--- /srv/reproducible-results/rbuild-debian/r-b-build.pLyCqxZf/b1/openlayers_2.13.1+ds2-10_amd64.changes +++ /srv/reproducible-results/rbuild-debian/r-b-build.pLyCqxZf/b2/openlayers_2.13.1+ds2-10_amd64.changes ├── Files │ @@ -1,2 +1,2 @@ │ │ - 2f894cbbe952d882ea4b0c5431d1cbc8 706880 javascript optional libjs-openlayers_2.13.1+ds2-10_all.deb │ + 1911fe32feb5191b50d7c2c66f2ddb29 781204 javascript optional libjs-openlayers_2.13.1+ds2-10_all.deb ├── libjs-openlayers_2.13.1+ds2-10_all.deb │ ├── file list │ │ @@ -1,3 +1,3 @@ │ │ -rw-r--r-- 0 0 0 4 2023-01-14 13:27:41.000000 debian-binary │ │ -rw-r--r-- 0 0 0 3684 2023-01-14 13:27:41.000000 control.tar.xz │ │ --rw-r--r-- 0 0 0 703004 2023-01-14 13:27:41.000000 data.tar.xz │ │ +-rw-r--r-- 0 0 0 777328 2023-01-14 13:27:41.000000 data.tar.xz │ ├── control.tar.xz │ │ ├── control.tar │ │ │ ├── ./md5sums │ │ │ │ ├── ./md5sums │ │ │ │ │┄ Files differ │ ├── data.tar.xz │ │ ├── data.tar │ │ │ ├── ./usr/share/javascript/openlayers/OpenLayers.js │ │ │ │ ├── js-beautify {} │ │ │ │ │ @@ -263,1776 +263,14 @@ │ │ │ │ │ source.hasOwnProperty && source.hasOwnProperty("toString")) { │ │ │ │ │ destination.toString = source.toString; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ return destination; │ │ │ │ │ }; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Console.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Console │ │ │ │ │ - * The OpenLayers.Console namespace is used for debugging and error logging. │ │ │ │ │ - * If the Firebug Lite (../Firebug/firebug.js) is included before this script, │ │ │ │ │ - * calls to OpenLayers.Console methods will get redirected to window.console. │ │ │ │ │ - * This makes use of the Firebug extension where available and allows for │ │ │ │ │ - * cross-browser debugging Firebug style. │ │ │ │ │ - * │ │ │ │ │ - * Note: │ │ │ │ │ - * Note that behavior will differ with the Firebug extention and Firebug Lite. │ │ │ │ │ - * Most notably, the Firebug Lite console does not currently allow for │ │ │ │ │ - * hyperlinks to code or for clicking on object to explore their properties. │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Console = { │ │ │ │ │ - /** │ │ │ │ │ - * Create empty functions for all console methods. The real value of these │ │ │ │ │ - * properties will be set if Firebug Lite (../Firebug/firebug.js script) is │ │ │ │ │ - * included. We explicitly require the Firebug Lite script to trigger │ │ │ │ │ - * functionality of the OpenLayers.Console methods. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIFunction: log │ │ │ │ │ - * Log an object in the console. The Firebug Lite console logs string │ │ │ │ │ - * representation of objects. Given multiple arguments, they will │ │ │ │ │ - * be cast to strings and logged with a space delimiter. If the first │ │ │ │ │ - * argument is a string with printf-like formatting, subsequent arguments │ │ │ │ │ - * will be used in string substitution. Any additional arguments (beyond │ │ │ │ │ - * the number substituted in a format string) will be appended in a space- │ │ │ │ │ - * delimited line. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * object - {Object} │ │ │ │ │ - */ │ │ │ │ │ - log: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIFunction: debug │ │ │ │ │ - * Writes a message to the console, including a hyperlink to the line │ │ │ │ │ - * where it was called. │ │ │ │ │ - * │ │ │ │ │ - * May be called with multiple arguments as with OpenLayers.Console.log(). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * object - {Object} │ │ │ │ │ - */ │ │ │ │ │ - debug: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIFunction: info │ │ │ │ │ - * Writes a message to the console with the visual "info" icon and color │ │ │ │ │ - * coding and a hyperlink to the line where it was called. │ │ │ │ │ - * │ │ │ │ │ - * May be called with multiple arguments as with OpenLayers.Console.log(). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * object - {Object} │ │ │ │ │ - */ │ │ │ │ │ - info: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIFunction: warn │ │ │ │ │ - * Writes a message to the console with the visual "warning" icon and │ │ │ │ │ - * color coding and a hyperlink to the line where it was called. │ │ │ │ │ - * │ │ │ │ │ - * May be called with multiple arguments as with OpenLayers.Console.log(). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * object - {Object} │ │ │ │ │ - */ │ │ │ │ │ - warn: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIFunction: error │ │ │ │ │ - * Writes a message to the console with the visual "error" icon and color │ │ │ │ │ - * coding and a hyperlink to the line where it was called. │ │ │ │ │ - * │ │ │ │ │ - * May be called with multiple arguments as with OpenLayers.Console.log(). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * object - {Object} │ │ │ │ │ - */ │ │ │ │ │ - error: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIFunction: userError │ │ │ │ │ - * A single interface for showing error messages to the user. The default │ │ │ │ │ - * behavior is a Javascript alert, though this can be overridden by │ │ │ │ │ - * reassigning OpenLayers.Console.userError to a different function. │ │ │ │ │ - * │ │ │ │ │ - * Expects a single error message │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * error - {Object} │ │ │ │ │ - */ │ │ │ │ │ - userError: function(error) { │ │ │ │ │ - alert(error); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIFunction: assert │ │ │ │ │ - * Tests that an expression is true. If not, it will write a message to │ │ │ │ │ - * the console and throw an exception. │ │ │ │ │ - * │ │ │ │ │ - * May be called with multiple arguments as with OpenLayers.Console.log(). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * object - {Object} │ │ │ │ │ - */ │ │ │ │ │ - assert: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIFunction: dir │ │ │ │ │ - * Prints an interactive listing of all properties of the object. This │ │ │ │ │ - * looks identical to the view that you would see in the DOM tab. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * object - {Object} │ │ │ │ │ - */ │ │ │ │ │ - dir: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIFunction: dirxml │ │ │ │ │ - * Prints the XML source tree of an HTML or XML element. This looks │ │ │ │ │ - * identical to the view that you would see in the HTML tab. You can click │ │ │ │ │ - * on any node to inspect it in the HTML tab. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * object - {Object} │ │ │ │ │ - */ │ │ │ │ │ - dirxml: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIFunction: trace │ │ │ │ │ - * Prints an interactive stack trace of JavaScript execution at the point │ │ │ │ │ - * where it is called. The stack trace details the functions on the stack, │ │ │ │ │ - * as well as the values that were passed as arguments to each function. │ │ │ │ │ - * You can click each function to take you to its source in the Script tab, │ │ │ │ │ - * and click each argument value to inspect it in the DOM or HTML tabs. │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - trace: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIFunction: group │ │ │ │ │ - * Writes a message to the console and opens a nested block to indent all │ │ │ │ │ - * future messages sent to the console. Call OpenLayers.Console.groupEnd() │ │ │ │ │ - * to close the block. │ │ │ │ │ - * │ │ │ │ │ - * May be called with multiple arguments as with OpenLayers.Console.log(). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * object - {Object} │ │ │ │ │ - */ │ │ │ │ │ - group: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIFunction: groupEnd │ │ │ │ │ - * Closes the most recently opened block created by a call to │ │ │ │ │ - * OpenLayers.Console.group │ │ │ │ │ - */ │ │ │ │ │ - groupEnd: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIFunction: time │ │ │ │ │ - * Creates a new timer under the given name. Call │ │ │ │ │ - * OpenLayers.Console.timeEnd(name) │ │ │ │ │ - * with the same name to stop the timer and print the time elapsed. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - */ │ │ │ │ │ - time: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIFunction: timeEnd │ │ │ │ │ - * Stops a timer created by a call to OpenLayers.Console.time(name) and │ │ │ │ │ - * writes the time elapsed. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - */ │ │ │ │ │ - timeEnd: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIFunction: profile │ │ │ │ │ - * Turns on the JavaScript profiler. The optional argument title would │ │ │ │ │ - * contain the text to be printed in the header of the profile report. │ │ │ │ │ - * │ │ │ │ │ - * This function is not currently implemented in Firebug Lite. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * title - {String} Optional title for the profiler │ │ │ │ │ - */ │ │ │ │ │ - profile: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIFunction: profileEnd │ │ │ │ │ - * Turns off the JavaScript profiler and prints its report. │ │ │ │ │ - * │ │ │ │ │ - * This function is not currently implemented in Firebug Lite. │ │ │ │ │ - */ │ │ │ │ │ - profileEnd: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIFunction: count │ │ │ │ │ - * Writes the number of times that the line of code where count was called │ │ │ │ │ - * was executed. The optional argument title will print a message in │ │ │ │ │ - * addition to the number of the count. │ │ │ │ │ - * │ │ │ │ │ - * This function is not currently implemented in Firebug Lite. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * title - {String} Optional title to be printed with count │ │ │ │ │ - */ │ │ │ │ │ - count: function() {}, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Console" │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Execute an anonymous function to extend the OpenLayers.Console namespace │ │ │ │ │ - * if the firebug.js script is included. This closure is used so that the │ │ │ │ │ - * "scripts" and "i" variables don't pollute the global namespace. │ │ │ │ │ - */ │ │ │ │ │ -(function() { │ │ │ │ │ - /** │ │ │ │ │ - * If Firebug Lite is included (before this script), re-route all │ │ │ │ │ - * OpenLayers.Console calls to the console object. │ │ │ │ │ - */ │ │ │ │ │ - var scripts = document.getElementsByTagName("script"); │ │ │ │ │ - for (var i = 0, len = scripts.length; i < len; ++i) { │ │ │ │ │ - if (scripts[i].src.indexOf("firebug.js") != -1) { │ │ │ │ │ - if (console) { │ │ │ │ │ - OpenLayers.Util.extend(OpenLayers.Console, console); │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ -})(); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Popup.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Popup │ │ │ │ │ - * A popup is a small div that can opened and closed on the map. │ │ │ │ │ - * Typically opened in response to clicking on a marker. │ │ │ │ │ - * See . Popup's don't require their own │ │ │ │ │ - * layer and are added the the map using the │ │ │ │ │ - * method. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * popup = new OpenLayers.Popup("chicken", │ │ │ │ │ - * new OpenLayers.LonLat(5,40), │ │ │ │ │ - * new OpenLayers.Size(200,200), │ │ │ │ │ - * "example popup", │ │ │ │ │ - * true); │ │ │ │ │ - * │ │ │ │ │ - * map.addPopup(popup); │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Popup = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: events │ │ │ │ │ - * {} custom event manager │ │ │ │ │ - */ │ │ │ │ │ - events: null, │ │ │ │ │ - │ │ │ │ │ - /** Property: id │ │ │ │ │ - * {String} the unique identifier assigned to this popup. │ │ │ │ │ - */ │ │ │ │ │ - id: "", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: lonlat │ │ │ │ │ - * {} the position of this popup on the map │ │ │ │ │ - */ │ │ │ │ │ - lonlat: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: div │ │ │ │ │ - * {DOMElement} the div that contains this popup. │ │ │ │ │ - */ │ │ │ │ │ - div: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: contentSize │ │ │ │ │ - * {} the width and height of the content. │ │ │ │ │ - */ │ │ │ │ │ - contentSize: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: size │ │ │ │ │ - * {} the width and height of the popup. │ │ │ │ │ - */ │ │ │ │ │ - size: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: contentHTML │ │ │ │ │ - * {String} An HTML string for this popup to display. │ │ │ │ │ - */ │ │ │ │ │ - contentHTML: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: backgroundColor │ │ │ │ │ - * {String} the background color used by the popup. │ │ │ │ │ - */ │ │ │ │ │ - backgroundColor: "", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: opacity │ │ │ │ │ - * {float} the opacity of this popup (between 0.0 and 1.0) │ │ │ │ │ - */ │ │ │ │ │ - opacity: "", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: border │ │ │ │ │ - * {String} the border size of the popup. (eg 2px) │ │ │ │ │ - */ │ │ │ │ │ - border: "", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: contentDiv │ │ │ │ │ - * {DOMElement} a reference to the element that holds the content of │ │ │ │ │ - * the div. │ │ │ │ │ - */ │ │ │ │ │ - contentDiv: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: groupDiv │ │ │ │ │ - * {DOMElement} First and only child of 'div'. The group Div contains the │ │ │ │ │ - * 'contentDiv' and the 'closeDiv'. │ │ │ │ │ - */ │ │ │ │ │ - groupDiv: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: closeDiv │ │ │ │ │ - * {DOMElement} the optional closer image │ │ │ │ │ - */ │ │ │ │ │ - closeDiv: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoSize │ │ │ │ │ - * {Boolean} Resize the popup to auto-fit the contents. │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - autoSize: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: minSize │ │ │ │ │ - * {} Minimum size allowed for the popup's contents. │ │ │ │ │ - */ │ │ │ │ │ - minSize: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: maxSize │ │ │ │ │ - * {} Maximum size allowed for the popup's contents. │ │ │ │ │ - */ │ │ │ │ │ - maxSize: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: displayClass │ │ │ │ │ - * {String} The CSS class of the popup. │ │ │ │ │ - */ │ │ │ │ │ - displayClass: "olPopup", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: contentDisplayClass │ │ │ │ │ - * {String} The CSS class of the popup content div. │ │ │ │ │ - */ │ │ │ │ │ - contentDisplayClass: "olPopupContent", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: padding │ │ │ │ │ - * {int or } An extra opportunity to specify internal │ │ │ │ │ - * padding of the content div inside the popup. This was originally │ │ │ │ │ - * confused with the css padding as specified in style.css's │ │ │ │ │ - * 'olPopupContent' class. We would like to get rid of this altogether, │ │ │ │ │ - * except that it does come in handy for the framed and anchoredbubble │ │ │ │ │ - * popups, who need to maintain yet another barrier between their │ │ │ │ │ - * content and the outer border of the popup itself. │ │ │ │ │ - * │ │ │ │ │ - * Note that in order to not break API, we must continue to support │ │ │ │ │ - * this property being set as an integer. Really, though, we'd like to │ │ │ │ │ - * have this specified as a Bounds object so that user can specify │ │ │ │ │ - * distinct left, top, right, bottom paddings. With the 3.0 release │ │ │ │ │ - * we can make this only a bounds. │ │ │ │ │ - */ │ │ │ │ │ - padding: 0, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: disableFirefoxOverflowHack │ │ │ │ │ - * {Boolean} The hack for overflow in Firefox causes all elements │ │ │ │ │ - * to be re-drawn, which causes Flash elements to be │ │ │ │ │ - * re-initialized, which is troublesome. │ │ │ │ │ - * With this property the hack can be disabled. │ │ │ │ │ - */ │ │ │ │ │ - disableFirefoxOverflowHack: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: fixPadding │ │ │ │ │ - * To be removed in 3.0, this function merely helps us to deal with the │ │ │ │ │ - * case where the user may have set an integer value for padding, │ │ │ │ │ - * instead of an object. │ │ │ │ │ - */ │ │ │ │ │ - fixPadding: function() { │ │ │ │ │ - if (typeof this.padding == "number") { │ │ │ │ │ - this.padding = new OpenLayers.Bounds( │ │ │ │ │ - this.padding, this.padding, this.padding, this.padding │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: panMapIfOutOfView │ │ │ │ │ - * {Boolean} When drawn, pan map such that the entire popup is visible in │ │ │ │ │ - * the current viewport (if necessary). │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - panMapIfOutOfView: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: keepInMap │ │ │ │ │ - * {Boolean} If panMapIfOutOfView is false, and this property is true, │ │ │ │ │ - * contrain the popup such that it always fits in the available map │ │ │ │ │ - * space. By default, this is not set on the base class. If you are │ │ │ │ │ - * creating popups that are near map edges and not allowing pannning, │ │ │ │ │ - * and especially if you have a popup which has a │ │ │ │ │ - * fixedRelativePosition, setting this to false may be a smart thing to │ │ │ │ │ - * do. Subclasses may want to override this setting. │ │ │ │ │ - * │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - keepInMap: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: closeOnMove │ │ │ │ │ - * {Boolean} When map pans, close the popup. │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - closeOnMove: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: map │ │ │ │ │ - * {} this gets set in Map.js when the popup is added to the map │ │ │ │ │ - */ │ │ │ │ │ - map: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Popup │ │ │ │ │ - * Create a popup. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} a unqiue identifier for this popup. If null is passed │ │ │ │ │ - * an identifier will be automatically generated. │ │ │ │ │ - * lonlat - {} The position on the map the popup will │ │ │ │ │ - * be shown. │ │ │ │ │ - * contentSize - {} The size of the content. │ │ │ │ │ - * contentHTML - {String} An HTML string to display inside the │ │ │ │ │ - * popup. │ │ │ │ │ - * closeBox - {Boolean} Whether to display a close box inside │ │ │ │ │ - * the popup. │ │ │ │ │ - * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, closeBox, closeBoxCallback) { │ │ │ │ │ - if (id == null) { │ │ │ │ │ - id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.id = id; │ │ │ │ │ - this.lonlat = lonlat; │ │ │ │ │ - │ │ │ │ │ - this.contentSize = (contentSize != null) ? contentSize : │ │ │ │ │ - new OpenLayers.Size( │ │ │ │ │ - OpenLayers.Popup.WIDTH, │ │ │ │ │ - OpenLayers.Popup.HEIGHT); │ │ │ │ │ - if (contentHTML != null) { │ │ │ │ │ - this.contentHTML = contentHTML; │ │ │ │ │ - } │ │ │ │ │ - this.backgroundColor = OpenLayers.Popup.COLOR; │ │ │ │ │ - this.opacity = OpenLayers.Popup.OPACITY; │ │ │ │ │ - this.border = OpenLayers.Popup.BORDER; │ │ │ │ │ - │ │ │ │ │ - this.div = OpenLayers.Util.createDiv(this.id, null, null, │ │ │ │ │ - null, null, null, "hidden"); │ │ │ │ │ - this.div.className = this.displayClass; │ │ │ │ │ - │ │ │ │ │ - var groupDivId = this.id + "_GroupDiv"; │ │ │ │ │ - this.groupDiv = OpenLayers.Util.createDiv(groupDivId, null, null, │ │ │ │ │ - null, "relative", null, │ │ │ │ │ - "hidden"); │ │ │ │ │ - │ │ │ │ │ - var id = this.div.id + "_contentDiv"; │ │ │ │ │ - this.contentDiv = OpenLayers.Util.createDiv(id, null, this.contentSize.clone(), │ │ │ │ │ - null, "relative"); │ │ │ │ │ - this.contentDiv.className = this.contentDisplayClass; │ │ │ │ │ - this.groupDiv.appendChild(this.contentDiv); │ │ │ │ │ - this.div.appendChild(this.groupDiv); │ │ │ │ │ - │ │ │ │ │ - if (closeBox) { │ │ │ │ │ - this.addCloseBox(closeBoxCallback); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.registerEvents(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * nullify references to prevent circular references and memory leaks │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - │ │ │ │ │ - this.id = null; │ │ │ │ │ - this.lonlat = null; │ │ │ │ │ - this.size = null; │ │ │ │ │ - this.contentHTML = null; │ │ │ │ │ - │ │ │ │ │ - this.backgroundColor = null; │ │ │ │ │ - this.opacity = null; │ │ │ │ │ - this.border = null; │ │ │ │ │ - │ │ │ │ │ - if (this.closeOnMove && this.map) { │ │ │ │ │ - this.map.events.unregister("movestart", this, this.hide); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.events.destroy(); │ │ │ │ │ - this.events = null; │ │ │ │ │ - │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - OpenLayers.Event.stopObservingElement(this.closeDiv); │ │ │ │ │ - this.groupDiv.removeChild(this.closeDiv); │ │ │ │ │ - } │ │ │ │ │ - this.closeDiv = null; │ │ │ │ │ - │ │ │ │ │ - this.div.removeChild(this.groupDiv); │ │ │ │ │ - this.groupDiv = null; │ │ │ │ │ - │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.removePopup(this); │ │ │ │ │ - } │ │ │ │ │ - this.map = null; │ │ │ │ │ - this.div = null; │ │ │ │ │ - │ │ │ │ │ - this.autoSize = null; │ │ │ │ │ - this.minSize = null; │ │ │ │ │ - this.maxSize = null; │ │ │ │ │ - this.padding = null; │ │ │ │ │ - this.panMapIfOutOfView = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Constructs the elements that make up the popup. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {} the position the popup in pixels. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} Reference to a div that contains the drawn popup │ │ │ │ │ - */ │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - if (px == null) { │ │ │ │ │ - if ((this.lonlat != null) && (this.map != null)) { │ │ │ │ │ - px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // this assumes that this.map already exists, which is okay because │ │ │ │ │ - // this.draw is only called once the popup has been added to the map. │ │ │ │ │ - if (this.closeOnMove) { │ │ │ │ │ - this.map.events.register("movestart", this, this.hide); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //listen to movestart, moveend to disable overflow (FF bug) │ │ │ │ │ - if (!this.disableFirefoxOverflowHack && OpenLayers.BROWSER_NAME == 'firefox') { │ │ │ │ │ - this.map.events.register("movestart", this, function() { │ │ │ │ │ - var style = document.defaultView.getComputedStyle( │ │ │ │ │ - this.contentDiv, null │ │ │ │ │ - ); │ │ │ │ │ - var currentOverflow = style.getPropertyValue("overflow"); │ │ │ │ │ - if (currentOverflow != "hidden") { │ │ │ │ │ - this.contentDiv._oldOverflow = currentOverflow; │ │ │ │ │ - this.contentDiv.style.overflow = "hidden"; │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - this.map.events.register("moveend", this, function() { │ │ │ │ │ - var oldOverflow = this.contentDiv._oldOverflow; │ │ │ │ │ - if (oldOverflow) { │ │ │ │ │ - this.contentDiv.style.overflow = oldOverflow; │ │ │ │ │ - this.contentDiv._oldOverflow = null; │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.moveTo(px); │ │ │ │ │ - if (!this.autoSize && !this.size) { │ │ │ │ │ - this.setSize(this.contentSize); │ │ │ │ │ - } │ │ │ │ │ - this.setBackgroundColor(); │ │ │ │ │ - this.setOpacity(); │ │ │ │ │ - this.setBorder(); │ │ │ │ │ - this.setContentHTML(); │ │ │ │ │ - │ │ │ │ │ - if (this.panMapIfOutOfView) { │ │ │ │ │ - this.panIntoView(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: updatePosition │ │ │ │ │ - * if the popup has a lonlat and its map members set, │ │ │ │ │ - * then have it move itself to its proper position │ │ │ │ │ - */ │ │ │ │ │ - updatePosition: function() { │ │ │ │ │ - if ((this.lonlat) && (this.map)) { │ │ │ │ │ - var px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ - if (px) { │ │ │ │ │ - this.moveTo(px); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {} the top and left position of the popup div. │ │ │ │ │ - */ │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - if ((px != null) && (this.div != null)) { │ │ │ │ │ - this.div.style.left = px.x + "px"; │ │ │ │ │ - this.div.style.top = px.y + "px"; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: visible │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Boolean indicating whether or not the popup is visible │ │ │ │ │ - */ │ │ │ │ │ - visible: function() { │ │ │ │ │ - return OpenLayers.Element.visible(this.div); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: toggle │ │ │ │ │ - * Toggles visibility of the popup. │ │ │ │ │ - */ │ │ │ │ │ - toggle: function() { │ │ │ │ │ - if (this.visible()) { │ │ │ │ │ - this.hide(); │ │ │ │ │ - } else { │ │ │ │ │ - this.show(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: show │ │ │ │ │ - * Makes the popup visible. │ │ │ │ │ - */ │ │ │ │ │ - show: function() { │ │ │ │ │ - this.div.style.display = ''; │ │ │ │ │ - │ │ │ │ │ - if (this.panMapIfOutOfView) { │ │ │ │ │ - this.panIntoView(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: hide │ │ │ │ │ - * Makes the popup invisible. │ │ │ │ │ - */ │ │ │ │ │ - hide: function() { │ │ │ │ │ - this.div.style.display = 'none'; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setSize │ │ │ │ │ - * Used to adjust the size of the popup. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * contentSize - {} the new size for the popup's │ │ │ │ │ - * contents div (in pixels). │ │ │ │ │ - */ │ │ │ │ │ - setSize: function(contentSize) { │ │ │ │ │ - this.size = contentSize.clone(); │ │ │ │ │ - │ │ │ │ │ - // if our contentDiv has a css 'padding' set on it by a stylesheet, we │ │ │ │ │ - // must add that to the desired "size". │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ - var wPadding = contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ - var hPadding = contentDivPadding.top + contentDivPadding.bottom; │ │ │ │ │ - │ │ │ │ │ - // take into account the popup's 'padding' property │ │ │ │ │ - this.fixPadding(); │ │ │ │ │ - wPadding += this.padding.left + this.padding.right; │ │ │ │ │ - hPadding += this.padding.top + this.padding.bottom; │ │ │ │ │ - │ │ │ │ │ - // make extra space for the close div │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - var closeDivWidth = parseInt(this.closeDiv.style.width); │ │ │ │ │ - wPadding += closeDivWidth + contentDivPadding.right; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //increase size of the main popup div to take into account the │ │ │ │ │ - // users's desired padding and close div. │ │ │ │ │ - this.size.w += wPadding; │ │ │ │ │ - this.size.h += hPadding; │ │ │ │ │ - │ │ │ │ │ - //now if our browser is IE, we need to actually make the contents │ │ │ │ │ - // div itself bigger to take its own padding into effect. this makes │ │ │ │ │ - // me want to shoot someone, but so it goes. │ │ │ │ │ - if (OpenLayers.BROWSER_NAME == "msie") { │ │ │ │ │ - this.contentSize.w += │ │ │ │ │ - contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ - this.contentSize.h += │ │ │ │ │ - contentDivPadding.bottom + contentDivPadding.top; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.div.style.width = this.size.w + "px"; │ │ │ │ │ - this.div.style.height = this.size.h + "px"; │ │ │ │ │ - } │ │ │ │ │ - if (this.contentDiv != null) { │ │ │ │ │ - this.contentDiv.style.width = contentSize.w + "px"; │ │ │ │ │ - this.contentDiv.style.height = contentSize.h + "px"; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: updateSize │ │ │ │ │ - * Auto size the popup so that it precisely fits its contents (as │ │ │ │ │ - * determined by this.contentDiv.innerHTML). Popup size will, of │ │ │ │ │ - * course, be limited by the available space on the current map │ │ │ │ │ - */ │ │ │ │ │ - updateSize: function() { │ │ │ │ │ - │ │ │ │ │ - // determine actual render dimensions of the contents by putting its │ │ │ │ │ - // contents into a fake contentDiv (for the CSS) and then measuring it │ │ │ │ │ - var preparedHTML = "
" + │ │ │ │ │ - this.contentDiv.innerHTML + │ │ │ │ │ - "
"; │ │ │ │ │ - │ │ │ │ │ - var containerElement = (this.map) ? this.map.div : document.body; │ │ │ │ │ - var realSize = OpenLayers.Util.getRenderedDimensions( │ │ │ │ │ - preparedHTML, null, { │ │ │ │ │ - displayClass: this.displayClass, │ │ │ │ │ - containerElement: containerElement │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - // is the "real" size of the div is safe to display in our map? │ │ │ │ │ - var safeSize = this.getSafeContentSize(realSize); │ │ │ │ │ - │ │ │ │ │ - var newSize = null; │ │ │ │ │ - if (safeSize.equals(realSize)) { │ │ │ │ │ - //real size of content is small enough to fit on the map, │ │ │ │ │ - // so we use real size. │ │ │ │ │ - newSize = realSize; │ │ │ │ │ - │ │ │ │ │ - } else { │ │ │ │ │ - │ │ │ │ │ - // make a new 'size' object with the clipped dimensions │ │ │ │ │ - // set or null if not clipped. │ │ │ │ │ - var fixedSize = { │ │ │ │ │ - w: (safeSize.w < realSize.w) ? safeSize.w : null, │ │ │ │ │ - h: (safeSize.h < realSize.h) ? safeSize.h : null │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - if (fixedSize.w && fixedSize.h) { │ │ │ │ │ - //content is too big in both directions, so we will use │ │ │ │ │ - // max popup size (safeSize), knowing well that it will │ │ │ │ │ - // overflow both ways. │ │ │ │ │ - newSize = safeSize; │ │ │ │ │ - } else { │ │ │ │ │ - //content is clipped in only one direction, so we need to │ │ │ │ │ - // run getRenderedDimensions() again with a fixed dimension │ │ │ │ │ - var clippedSize = OpenLayers.Util.getRenderedDimensions( │ │ │ │ │ - preparedHTML, fixedSize, { │ │ │ │ │ - displayClass: this.contentDisplayClass, │ │ │ │ │ - containerElement: containerElement │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - //if the clipped size is still the same as the safeSize, │ │ │ │ │ - // that means that our content must be fixed in the │ │ │ │ │ - // offending direction. If overflow is 'auto', this means │ │ │ │ │ - // we are going to have a scrollbar for sure, so we must │ │ │ │ │ - // adjust for that. │ │ │ │ │ - // │ │ │ │ │ - var currentOverflow = OpenLayers.Element.getStyle( │ │ │ │ │ - this.contentDiv, "overflow" │ │ │ │ │ - ); │ │ │ │ │ - if ((currentOverflow != "hidden") && │ │ │ │ │ - (clippedSize.equals(safeSize))) { │ │ │ │ │ - var scrollBar = OpenLayers.Util.getScrollbarWidth(); │ │ │ │ │ - if (fixedSize.w) { │ │ │ │ │ - clippedSize.h += scrollBar; │ │ │ │ │ - } else { │ │ │ │ │ - clippedSize.w += scrollBar; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - newSize = this.getSafeContentSize(clippedSize); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.setSize(newSize); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setBackgroundColor │ │ │ │ │ - * Sets the background color of the popup. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * color - {String} the background color. eg "#FFBBBB" │ │ │ │ │ - */ │ │ │ │ │ - setBackgroundColor: function(color) { │ │ │ │ │ - if (color != undefined) { │ │ │ │ │ - this.backgroundColor = color; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.div.style.backgroundColor = this.backgroundColor; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setOpacity │ │ │ │ │ - * Sets the opacity of the popup. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * opacity - {float} A value between 0.0 (transparent) and 1.0 (solid). │ │ │ │ │ - */ │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - if (opacity != undefined) { │ │ │ │ │ - this.opacity = opacity; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - // for Mozilla and Safari │ │ │ │ │ - this.div.style.opacity = this.opacity; │ │ │ │ │ - │ │ │ │ │ - // for IE │ │ │ │ │ - this.div.style.filter = 'alpha(opacity=' + this.opacity * 100 + ')'; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setBorder │ │ │ │ │ - * Sets the border style of the popup. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * border - {String} The border style value. eg 2px │ │ │ │ │ - */ │ │ │ │ │ - setBorder: function(border) { │ │ │ │ │ - if (border != undefined) { │ │ │ │ │ - this.border = border; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.div.style.border = this.border; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setContentHTML │ │ │ │ │ - * Allows the user to set the HTML content of the popup. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * contentHTML - {String} HTML for the div. │ │ │ │ │ - */ │ │ │ │ │ - setContentHTML: function(contentHTML) { │ │ │ │ │ - │ │ │ │ │ - if (contentHTML != null) { │ │ │ │ │ - this.contentHTML = contentHTML; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if ((this.contentDiv != null) && │ │ │ │ │ - (this.contentHTML != null) && │ │ │ │ │ - (this.contentHTML != this.contentDiv.innerHTML)) { │ │ │ │ │ - │ │ │ │ │ - this.contentDiv.innerHTML = this.contentHTML; │ │ │ │ │ - │ │ │ │ │ - if (this.autoSize) { │ │ │ │ │ - │ │ │ │ │ - //if popup has images, listen for when they finish │ │ │ │ │ - // loading and resize accordingly │ │ │ │ │ - this.registerImageListeners(); │ │ │ │ │ - │ │ │ │ │ - //auto size the popup to its current contents │ │ │ │ │ - this.updateSize(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: registerImageListeners │ │ │ │ │ - * Called when an image contained by the popup loaded. this function │ │ │ │ │ - * updates the popup size, then unregisters the image load listener. │ │ │ │ │ - */ │ │ │ │ │ - registerImageListeners: function() { │ │ │ │ │ - │ │ │ │ │ - // As the images load, this function will call updateSize() to │ │ │ │ │ - // resize the popup to fit the content div (which presumably is now │ │ │ │ │ - // bigger than when the image was not loaded). │ │ │ │ │ - // │ │ │ │ │ - // If the 'panMapIfOutOfView' property is set, we will pan the newly │ │ │ │ │ - // resized popup back into view. │ │ │ │ │ - // │ │ │ │ │ - // Note that this function, when called, will have 'popup' and │ │ │ │ │ - // 'img' properties in the context. │ │ │ │ │ - // │ │ │ │ │ - var onImgLoad = function() { │ │ │ │ │ - if (this.popup.id === null) { // this.popup has been destroyed! │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - this.popup.updateSize(); │ │ │ │ │ - │ │ │ │ │ - if (this.popup.visible() && this.popup.panMapIfOutOfView) { │ │ │ │ │ - this.popup.panIntoView(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Event.stopObserving( │ │ │ │ │ - this.img, "load", this.img._onImgLoad │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - //cycle through the images and if their size is 0x0, that means that │ │ │ │ │ - // they haven't been loaded yet, so we attach the listener, which │ │ │ │ │ - // will fire when the images finish loading and will resize the │ │ │ │ │ - // popup accordingly to its new size. │ │ │ │ │ - var images = this.contentDiv.getElementsByTagName("img"); │ │ │ │ │ - for (var i = 0, len = images.length; i < len; i++) { │ │ │ │ │ - var img = images[i]; │ │ │ │ │ - if (img.width == 0 || img.height == 0) { │ │ │ │ │ - │ │ │ │ │ - var context = { │ │ │ │ │ - 'popup': this, │ │ │ │ │ - 'img': img │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - //expando this function to the image itself before registering │ │ │ │ │ - // it. This way we can easily and properly unregister it. │ │ │ │ │ - img._onImgLoad = OpenLayers.Function.bind(onImgLoad, context); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Event.observe(img, 'load', img._onImgLoad); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getSafeContentSize │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * size - {} Desired size to make the popup. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} A size to make the popup which is neither smaller │ │ │ │ │ - * than the specified minimum size, nor bigger than the maximum │ │ │ │ │ - * size (which is calculated relative to the size of the viewport). │ │ │ │ │ - */ │ │ │ │ │ - getSafeContentSize: function(size) { │ │ │ │ │ - │ │ │ │ │ - var safeContentSize = size.clone(); │ │ │ │ │ - │ │ │ │ │ - // if our contentDiv has a css 'padding' set on it by a stylesheet, we │ │ │ │ │ - // must add that to the desired "size". │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ - var wPadding = contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ - var hPadding = contentDivPadding.top + contentDivPadding.bottom; │ │ │ │ │ - │ │ │ │ │ - // take into account the popup's 'padding' property │ │ │ │ │ - this.fixPadding(); │ │ │ │ │ - wPadding += this.padding.left + this.padding.right; │ │ │ │ │ - hPadding += this.padding.top + this.padding.bottom; │ │ │ │ │ - │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - var closeDivWidth = parseInt(this.closeDiv.style.width); │ │ │ │ │ - wPadding += closeDivWidth + contentDivPadding.right; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // prevent the popup from being smaller than a specified minimal size │ │ │ │ │ - if (this.minSize) { │ │ │ │ │ - safeContentSize.w = Math.max(safeContentSize.w, │ │ │ │ │ - (this.minSize.w - wPadding)); │ │ │ │ │ - safeContentSize.h = Math.max(safeContentSize.h, │ │ │ │ │ - (this.minSize.h - hPadding)); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // prevent the popup from being bigger than a specified maximum size │ │ │ │ │ - if (this.maxSize) { │ │ │ │ │ - safeContentSize.w = Math.min(safeContentSize.w, │ │ │ │ │ - (this.maxSize.w - wPadding)); │ │ │ │ │ - safeContentSize.h = Math.min(safeContentSize.h, │ │ │ │ │ - (this.maxSize.h - hPadding)); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //make sure the desired size to set doesn't result in a popup that │ │ │ │ │ - // is bigger than the map's viewport. │ │ │ │ │ - // │ │ │ │ │ - if (this.map && this.map.size) { │ │ │ │ │ - │ │ │ │ │ - var extraX = 0, │ │ │ │ │ - extraY = 0; │ │ │ │ │ - if (this.keepInMap && !this.panMapIfOutOfView) { │ │ │ │ │ - var px = this.map.getPixelFromLonLat(this.lonlat); │ │ │ │ │ - switch (this.relativePosition) { │ │ │ │ │ - case "tr": │ │ │ │ │ - extraX = px.x; │ │ │ │ │ - extraY = this.map.size.h - px.y; │ │ │ │ │ - break; │ │ │ │ │ - case "tl": │ │ │ │ │ - extraX = this.map.size.w - px.x; │ │ │ │ │ - extraY = this.map.size.h - px.y; │ │ │ │ │ - break; │ │ │ │ │ - case "bl": │ │ │ │ │ - extraX = this.map.size.w - px.x; │ │ │ │ │ - extraY = px.y; │ │ │ │ │ - break; │ │ │ │ │ - case "br": │ │ │ │ │ - extraX = px.x; │ │ │ │ │ - extraY = px.y; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - extraX = px.x; │ │ │ │ │ - extraY = this.map.size.h - px.y; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var maxY = this.map.size.h - │ │ │ │ │ - this.map.paddingForPopups.top - │ │ │ │ │ - this.map.paddingForPopups.bottom - │ │ │ │ │ - hPadding - extraY; │ │ │ │ │ - │ │ │ │ │ - var maxX = this.map.size.w - │ │ │ │ │ - this.map.paddingForPopups.left - │ │ │ │ │ - this.map.paddingForPopups.right - │ │ │ │ │ - wPadding - extraX; │ │ │ │ │ - │ │ │ │ │ - safeContentSize.w = Math.min(safeContentSize.w, maxX); │ │ │ │ │ - safeContentSize.h = Math.min(safeContentSize.h, maxY); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return safeContentSize; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getContentDivPadding │ │ │ │ │ - * Glorious, oh glorious hack in order to determine the css 'padding' of │ │ │ │ │ - * the contentDiv. IE/Opera return null here unless we actually add the │ │ │ │ │ - * popup's main 'div' element (which contains contentDiv) to the DOM. │ │ │ │ │ - * So we make it invisible and then add it to the document temporarily. │ │ │ │ │ - * │ │ │ │ │ - * Once we've taken the padding readings we need, we then remove it │ │ │ │ │ - * from the DOM (it will actually get added to the DOM in │ │ │ │ │ - * Map.js's addPopup) │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} │ │ │ │ │ - */ │ │ │ │ │ - getContentDivPadding: function() { │ │ │ │ │ - │ │ │ │ │ - //use cached value if we have it │ │ │ │ │ - var contentDivPadding = this._contentDivPadding; │ │ │ │ │ - if (!contentDivPadding) { │ │ │ │ │ - │ │ │ │ │ - if (this.div.parentNode == null) { │ │ │ │ │ - //make the div invisible and add it to the page │ │ │ │ │ - this.div.style.display = "none"; │ │ │ │ │ - document.body.appendChild(this.div); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //read the padding settings from css, put them in an OL.Bounds │ │ │ │ │ - contentDivPadding = new OpenLayers.Bounds( │ │ │ │ │ - OpenLayers.Element.getStyle(this.contentDiv, "padding-left"), │ │ │ │ │ - OpenLayers.Element.getStyle(this.contentDiv, "padding-bottom"), │ │ │ │ │ - OpenLayers.Element.getStyle(this.contentDiv, "padding-right"), │ │ │ │ │ - OpenLayers.Element.getStyle(this.contentDiv, "padding-top") │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - //cache the value │ │ │ │ │ - this._contentDivPadding = contentDivPadding; │ │ │ │ │ - │ │ │ │ │ - if (this.div.parentNode == document.body) { │ │ │ │ │ - //remove the div from the page and make it visible again │ │ │ │ │ - document.body.removeChild(this.div); │ │ │ │ │ - this.div.style.display = ""; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return contentDivPadding; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addCloseBox │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * callback - {Function} The callback to be called when the close button │ │ │ │ │ - * is clicked. │ │ │ │ │ - */ │ │ │ │ │ - addCloseBox: function(callback) { │ │ │ │ │ - │ │ │ │ │ - this.closeDiv = OpenLayers.Util.createDiv( │ │ │ │ │ - this.id + "_close", null, { │ │ │ │ │ - w: 17, │ │ │ │ │ - h: 17 │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - this.closeDiv.className = "olPopupCloseBox"; │ │ │ │ │ - │ │ │ │ │ - // use the content div's css padding to determine if we should │ │ │ │ │ - // padd the close div │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ - │ │ │ │ │ - this.closeDiv.style.right = contentDivPadding.right + "px"; │ │ │ │ │ - this.closeDiv.style.top = contentDivPadding.top + "px"; │ │ │ │ │ - this.groupDiv.appendChild(this.closeDiv); │ │ │ │ │ - │ │ │ │ │ - var closePopup = callback || function(e) { │ │ │ │ │ - this.hide(); │ │ │ │ │ - OpenLayers.Event.stop(e); │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Event.observe(this.closeDiv, "touchend", │ │ │ │ │ - OpenLayers.Function.bindAsEventListener(closePopup, this)); │ │ │ │ │ - OpenLayers.Event.observe(this.closeDiv, "click", │ │ │ │ │ - OpenLayers.Function.bindAsEventListener(closePopup, this)); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: panIntoView │ │ │ │ │ - * Pans the map such that the popup is totaly viewable (if necessary) │ │ │ │ │ - */ │ │ │ │ │ - panIntoView: function() { │ │ │ │ │ - │ │ │ │ │ - var mapSize = this.map.getSize(); │ │ │ │ │ - │ │ │ │ │ - //start with the top left corner of the popup, in px, │ │ │ │ │ - // relative to the viewport │ │ │ │ │ - var origTL = this.map.getViewPortPxFromLayerPx(new OpenLayers.Pixel( │ │ │ │ │ - parseInt(this.div.style.left), │ │ │ │ │ - parseInt(this.div.style.top) │ │ │ │ │ - )); │ │ │ │ │ - var newTL = origTL.clone(); │ │ │ │ │ - │ │ │ │ │ - //new left (compare to margins, using this.size to calculate right) │ │ │ │ │ - if (origTL.x < this.map.paddingForPopups.left) { │ │ │ │ │ - newTL.x = this.map.paddingForPopups.left; │ │ │ │ │ - } else │ │ │ │ │ - if ((origTL.x + this.size.w) > (mapSize.w - this.map.paddingForPopups.right)) { │ │ │ │ │ - newTL.x = mapSize.w - this.map.paddingForPopups.right - this.size.w; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //new top (compare to margins, using this.size to calculate bottom) │ │ │ │ │ - if (origTL.y < this.map.paddingForPopups.top) { │ │ │ │ │ - newTL.y = this.map.paddingForPopups.top; │ │ │ │ │ - } else │ │ │ │ │ - if ((origTL.y + this.size.h) > (mapSize.h - this.map.paddingForPopups.bottom)) { │ │ │ │ │ - newTL.y = mapSize.h - this.map.paddingForPopups.bottom - this.size.h; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var dx = origTL.x - newTL.x; │ │ │ │ │ - var dy = origTL.y - newTL.y; │ │ │ │ │ - │ │ │ │ │ - this.map.pan(dx, dy); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: registerEvents │ │ │ │ │ - * Registers events on the popup. │ │ │ │ │ - * │ │ │ │ │ - * Do this in a separate function so that subclasses can │ │ │ │ │ - * choose to override it if they wish to deal differently │ │ │ │ │ - * with mouse events │ │ │ │ │ - * │ │ │ │ │ - * Note in the following handler functions that some special │ │ │ │ │ - * care is needed to deal correctly with mousing and popups. │ │ │ │ │ - * │ │ │ │ │ - * Because the user might select the zoom-rectangle option and │ │ │ │ │ - * then drag it over a popup, we need a safe way to allow the │ │ │ │ │ - * mousemove and mouseup events to pass through the popup when │ │ │ │ │ - * they are initiated from outside. The same procedure is needed for │ │ │ │ │ - * touchmove and touchend events. │ │ │ │ │ - * │ │ │ │ │ - * Otherwise, we want to essentially kill the event propagation │ │ │ │ │ - * for all other events, though we have to do so carefully, │ │ │ │ │ - * without disabling basic html functionality, like clicking on │ │ │ │ │ - * hyperlinks or drag-selecting text. │ │ │ │ │ - */ │ │ │ │ │ - registerEvents: function() { │ │ │ │ │ - this.events = new OpenLayers.Events(this, this.div, null, true); │ │ │ │ │ - │ │ │ │ │ - function onTouchstart(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt, true); │ │ │ │ │ - } │ │ │ │ │ - this.events.on({ │ │ │ │ │ - "mousedown": this.onmousedown, │ │ │ │ │ - "mousemove": this.onmousemove, │ │ │ │ │ - "mouseup": this.onmouseup, │ │ │ │ │ - "click": this.onclick, │ │ │ │ │ - "mouseout": this.onmouseout, │ │ │ │ │ - "dblclick": this.ondblclick, │ │ │ │ │ - "touchstart": onTouchstart, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onmousedown │ │ │ │ │ - * When mouse goes down within the popup, make a note of │ │ │ │ │ - * it locally, and then do not propagate the mousedown │ │ │ │ │ - * (but do so safely so that user can select text inside) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - onmousedown: function(evt) { │ │ │ │ │ - this.mousedown = true; │ │ │ │ │ - OpenLayers.Event.stop(evt, true); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onmousemove │ │ │ │ │ - * If the drag was started within the popup, then │ │ │ │ │ - * do not propagate the mousemove (but do so safely │ │ │ │ │ - * so that user can select text inside) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - onmousemove: function(evt) { │ │ │ │ │ - if (this.mousedown) { │ │ │ │ │ - OpenLayers.Event.stop(evt, true); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onmouseup │ │ │ │ │ - * When mouse comes up within the popup, after going down │ │ │ │ │ - * in it, reset the flag, and then (once again) do not │ │ │ │ │ - * propagate the event, but do so safely so that user can │ │ │ │ │ - * select text inside │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - onmouseup: function(evt) { │ │ │ │ │ - if (this.mousedown) { │ │ │ │ │ - this.mousedown = false; │ │ │ │ │ - OpenLayers.Event.stop(evt, true); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onclick │ │ │ │ │ - * Ignore clicks, but allowing default browser handling │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - onclick: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt, true); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onmouseout │ │ │ │ │ - * When mouse goes out of the popup set the flag to false so that │ │ │ │ │ - * if they let go and then drag back in, we won't be confused. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - onmouseout: function(evt) { │ │ │ │ │ - this.mousedown = false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: ondblclick │ │ │ │ │ - * Ignore double-clicks, but allowing default browser handling │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - ondblclick: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt, true); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Popup.WIDTH = 200; │ │ │ │ │ -OpenLayers.Popup.HEIGHT = 200; │ │ │ │ │ -OpenLayers.Popup.COLOR = "white"; │ │ │ │ │ -OpenLayers.Popup.OPACITY = 1; │ │ │ │ │ -OpenLayers.Popup.BORDER = "0px"; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Util/vendorPrefix.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/SingleFile.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Util = OpenLayers.Util || {}; │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Util.vendorPrefix │ │ │ │ │ - * A collection of utility functions to detect vendor prefixed features │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Util.vendorPrefix = (function() { │ │ │ │ │ - "use strict"; │ │ │ │ │ - │ │ │ │ │ - var VENDOR_PREFIXES = ["", "O", "ms", "Moz", "Webkit"], │ │ │ │ │ - divStyle = document.createElement("div").style, │ │ │ │ │ - cssCache = {}, │ │ │ │ │ - jsCache = {}; │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Function: domToCss │ │ │ │ │ - * Converts a upper camel case DOM style property name to a CSS property │ │ │ │ │ - * i.e. transformOrigin -> transform-origin │ │ │ │ │ - * or WebkitTransformOrigin -> -webkit-transform-origin │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * prefixedDom - {String} The property to convert │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The CSS property │ │ │ │ │ - */ │ │ │ │ │ - function domToCss(prefixedDom) { │ │ │ │ │ - if (!prefixedDom) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - return prefixedDom. │ │ │ │ │ - replace(/([A-Z])/g, function(c) { │ │ │ │ │ - return "-" + c.toLowerCase(); │ │ │ │ │ - }). │ │ │ │ │ - replace(/^ms-/, "-ms-"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: css │ │ │ │ │ - * Detect which property is used for a CSS property │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * property - {String} The standard (unprefixed) CSS property name │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The standard CSS property, prefixed property or null if not │ │ │ │ │ - * supported │ │ │ │ │ - */ │ │ │ │ │ - function css(property) { │ │ │ │ │ - if (cssCache[property] === undefined) { │ │ │ │ │ - var domProperty = property. │ │ │ │ │ - replace(/(-[\s\S])/g, function(c) { │ │ │ │ │ - return c.charAt(1).toUpperCase(); │ │ │ │ │ - }); │ │ │ │ │ - var prefixedDom = style(domProperty); │ │ │ │ │ - cssCache[property] = domToCss(prefixedDom); │ │ │ │ │ - } │ │ │ │ │ - return cssCache[property]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: js │ │ │ │ │ - * Detect which property is used for a JS property/method │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} The object to test on │ │ │ │ │ - * property - {String} The standard (unprefixed) JS property name │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The standard JS property, prefixed property or null if not │ │ │ │ │ - * supported │ │ │ │ │ - */ │ │ │ │ │ - function js(obj, property) { │ │ │ │ │ - if (jsCache[property] === undefined) { │ │ │ │ │ - var tmpProp, │ │ │ │ │ - i = 0, │ │ │ │ │ - l = VENDOR_PREFIXES.length, │ │ │ │ │ - prefix, │ │ │ │ │ - isStyleObj = (typeof obj.cssText !== "undefined"); │ │ │ │ │ - │ │ │ │ │ - jsCache[property] = null; │ │ │ │ │ - for (; i < l; i++) { │ │ │ │ │ - prefix = VENDOR_PREFIXES[i]; │ │ │ │ │ - if (prefix) { │ │ │ │ │ - if (!isStyleObj) { │ │ │ │ │ - // js prefix should be lower-case, while style │ │ │ │ │ - // properties have upper case on first character │ │ │ │ │ - prefix = prefix.toLowerCase(); │ │ │ │ │ - } │ │ │ │ │ - tmpProp = prefix + property.charAt(0).toUpperCase() + property.slice(1); │ │ │ │ │ - } else { │ │ │ │ │ - tmpProp = property; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (obj[tmpProp] !== undefined) { │ │ │ │ │ - jsCache[property] = tmpProp; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return jsCache[property]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: style │ │ │ │ │ - * Detect which property is used for a DOM style property │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * property - {String} The standard (unprefixed) style property name │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The standard style property, prefixed property or null if not │ │ │ │ │ - * supported │ │ │ │ │ - */ │ │ │ │ │ - function style(property) { │ │ │ │ │ - return js(divStyle, property); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return { │ │ │ │ │ - css: css, │ │ │ │ │ - js: js, │ │ │ │ │ - style: style, │ │ │ │ │ - │ │ │ │ │ - // used for testing │ │ │ │ │ - cssCache: cssCache, │ │ │ │ │ - jsCache: jsCache │ │ │ │ │ - }; │ │ │ │ │ -}()); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Animation.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/SingleFile.js │ │ │ │ │ - * @requires OpenLayers/Util/vendorPrefix.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Animation │ │ │ │ │ - * A collection of utility functions for executing methods that repaint a │ │ │ │ │ - * portion of the browser window. These methods take advantage of the │ │ │ │ │ - * browser's scheduled repaints where requestAnimationFrame is available. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Animation = (function(window) { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: isNative │ │ │ │ │ - * {Boolean} true if a native requestAnimationFrame function is available │ │ │ │ │ - */ │ │ │ │ │ - var requestAnimationFrame = OpenLayers.Util.vendorPrefix.js(window, "requestAnimationFrame"); │ │ │ │ │ - var isNative = !!(requestAnimationFrame); │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Function: requestFrame │ │ │ │ │ - * Schedule a function to be called at the next available animation frame. │ │ │ │ │ - * Uses the native method where available. Where requestAnimationFrame is │ │ │ │ │ - * not available, setTimeout will be called with a 16ms delay. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * callback - {Function} The function to be called at the next animation frame. │ │ │ │ │ - * element - {DOMElement} Optional element that visually bounds the animation. │ │ │ │ │ - */ │ │ │ │ │ - var requestFrame = (function() { │ │ │ │ │ - var request = window[requestAnimationFrame] || │ │ │ │ │ - function(callback, element) { │ │ │ │ │ - window.setTimeout(callback, 16); │ │ │ │ │ - }; │ │ │ │ │ - // bind to window to avoid illegal invocation of native function │ │ │ │ │ - return function(callback, element) { │ │ │ │ │ - request.apply(window, [callback, element]); │ │ │ │ │ - }; │ │ │ │ │ - })(); │ │ │ │ │ - │ │ │ │ │ - // private variables for animation loops │ │ │ │ │ - var counter = 0; │ │ │ │ │ - var loops = {}; │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Function: start │ │ │ │ │ - * Executes a method with in series for some │ │ │ │ │ - * duration. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * callback - {Function} The function to be called at the next animation frame. │ │ │ │ │ - * duration - {Number} Optional duration for the loop. If not provided, the │ │ │ │ │ - * animation loop will execute indefinitely. │ │ │ │ │ - * element - {DOMElement} Optional element that visually bounds the animation. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number} Identifier for the animation loop. Used to stop animations with │ │ │ │ │ - * . │ │ │ │ │ - */ │ │ │ │ │ - function start(callback, duration, element) { │ │ │ │ │ - duration = duration > 0 ? duration : Number.POSITIVE_INFINITY; │ │ │ │ │ - var id = ++counter; │ │ │ │ │ - var start = +new Date; │ │ │ │ │ - loops[id] = function() { │ │ │ │ │ - if (loops[id] && +new Date - start <= duration) { │ │ │ │ │ - callback(); │ │ │ │ │ - if (loops[id]) { │ │ │ │ │ - requestFrame(loops[id], element); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - delete loops[id]; │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - requestFrame(loops[id], element); │ │ │ │ │ - return id; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Function: stop │ │ │ │ │ - * Terminates an animation loop started with . │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {Number} Identifier returned from . │ │ │ │ │ - */ │ │ │ │ │ - function stop(id) { │ │ │ │ │ - delete loops[id]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return { │ │ │ │ │ - isNative: isNative, │ │ │ │ │ - requestFrame: requestFrame, │ │ │ │ │ - start: start, │ │ │ │ │ - stop: stop │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ -})(window); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Kinetic.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Animation.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Kinetic = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: threshold │ │ │ │ │ - * In most cases changing the threshold isn't needed. │ │ │ │ │ - * In px/ms, default to 0. │ │ │ │ │ - */ │ │ │ │ │ - threshold: 0, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: deceleration │ │ │ │ │ - * {Float} the deseleration in px/ms², default to 0.0035. │ │ │ │ │ - */ │ │ │ │ │ - deceleration: 0.0035, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: nbPoints │ │ │ │ │ - * {Integer} the number of points we use to calculate the kinetic │ │ │ │ │ - * initial values. │ │ │ │ │ - */ │ │ │ │ │ - nbPoints: 100, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: delay │ │ │ │ │ - * {Float} time to consider to calculate the kinetic initial values. │ │ │ │ │ - * In ms, default to 200. │ │ │ │ │ - */ │ │ │ │ │ - delay: 200, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: points │ │ │ │ │ - * List of points use to calculate the kinetic initial values. │ │ │ │ │ - */ │ │ │ │ │ - points: undefined, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: timerId │ │ │ │ │ - * ID of the timer. │ │ │ │ │ - */ │ │ │ │ │ - timerId: undefined, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Kinetic │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: begin │ │ │ │ │ - * Begins the dragging. │ │ │ │ │ - */ │ │ │ │ │ - begin: function() { │ │ │ │ │ - OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ - this.timerId = undefined; │ │ │ │ │ - this.points = []; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: update │ │ │ │ │ - * Updates during the dragging. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xy - {} The new position. │ │ │ │ │ - */ │ │ │ │ │ - update: function(xy) { │ │ │ │ │ - this.points.unshift({ │ │ │ │ │ - xy: xy, │ │ │ │ │ - tick: new Date().getTime() │ │ │ │ │ - }); │ │ │ │ │ - if (this.points.length > this.nbPoints) { │ │ │ │ │ - this.points.pop(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: end │ │ │ │ │ - * Ends the dragging, start the kinetic. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xy - {} The last position. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object with two properties: "speed", and "theta". The │ │ │ │ │ - * "speed" and "theta" values are to be passed to the move │ │ │ │ │ - * function when starting the animation. │ │ │ │ │ - */ │ │ │ │ │ - end: function(xy) { │ │ │ │ │ - var last, now = new Date().getTime(); │ │ │ │ │ - for (var i = 0, l = this.points.length, point; i < l; i++) { │ │ │ │ │ - point = this.points[i]; │ │ │ │ │ - if (now - point.tick > this.delay) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - last = point; │ │ │ │ │ - } │ │ │ │ │ - if (!last) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var time = new Date().getTime() - last.tick; │ │ │ │ │ - var dist = Math.sqrt(Math.pow(xy.x - last.xy.x, 2) + │ │ │ │ │ - Math.pow(xy.y - last.xy.y, 2)); │ │ │ │ │ - var speed = dist / time; │ │ │ │ │ - if (speed == 0 || speed < this.threshold) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var theta = Math.asin((xy.y - last.xy.y) / dist); │ │ │ │ │ - if (last.xy.x <= xy.x) { │ │ │ │ │ - theta = Math.PI - theta; │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - speed: speed, │ │ │ │ │ - theta: theta │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * Launch the kinetic move pan. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * info - {Object} An object with two properties, "speed", and "theta". │ │ │ │ │ - * These values are those returned from the "end" call. │ │ │ │ │ - * callback - {Function} Function called on every step of the animation, │ │ │ │ │ - * receives x, y (values to pan), end (is the last point). │ │ │ │ │ - */ │ │ │ │ │ - move: function(info, callback) { │ │ │ │ │ - var v0 = info.speed; │ │ │ │ │ - var fx = Math.cos(info.theta); │ │ │ │ │ - var fy = -Math.sin(info.theta); │ │ │ │ │ - │ │ │ │ │ - var initialTime = new Date().getTime(); │ │ │ │ │ - │ │ │ │ │ - var lastX = 0; │ │ │ │ │ - var lastY = 0; │ │ │ │ │ - │ │ │ │ │ - var timerCallback = function() { │ │ │ │ │ - if (this.timerId == null) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var t = new Date().getTime() - initialTime; │ │ │ │ │ - │ │ │ │ │ - var p = (-this.deceleration * Math.pow(t, 2)) / 2.0 + v0 * t; │ │ │ │ │ - var x = p * fx; │ │ │ │ │ - var y = p * fy; │ │ │ │ │ - │ │ │ │ │ - var args = {}; │ │ │ │ │ - args.end = false; │ │ │ │ │ - var v = -this.deceleration * t + v0; │ │ │ │ │ - │ │ │ │ │ - if (v <= 0) { │ │ │ │ │ - OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - args.end = true; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - args.x = x - lastX; │ │ │ │ │ - args.y = y - lastY; │ │ │ │ │ - lastX = x; │ │ │ │ │ - lastY = y; │ │ │ │ │ - callback(args.x, args.y, args.end); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - this.timerId = OpenLayers.Animation.start( │ │ │ │ │ - OpenLayers.Function.bind(timerCallback, this) │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Kinetic" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ OpenLayers/BaseTypes.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -4005,14 +2243,268 @@ │ │ │ │ │ } │ │ │ │ │ return equals; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Size" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ + OpenLayers/Console.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Console │ │ │ │ │ + * The OpenLayers.Console namespace is used for debugging and error logging. │ │ │ │ │ + * If the Firebug Lite (../Firebug/firebug.js) is included before this script, │ │ │ │ │ + * calls to OpenLayers.Console methods will get redirected to window.console. │ │ │ │ │ + * This makes use of the Firebug extension where available and allows for │ │ │ │ │ + * cross-browser debugging Firebug style. │ │ │ │ │ + * │ │ │ │ │ + * Note: │ │ │ │ │ + * Note that behavior will differ with the Firebug extention and Firebug Lite. │ │ │ │ │ + * Most notably, the Firebug Lite console does not currently allow for │ │ │ │ │ + * hyperlinks to code or for clicking on object to explore their properties. │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Console = { │ │ │ │ │ + /** │ │ │ │ │ + * Create empty functions for all console methods. The real value of these │ │ │ │ │ + * properties will be set if Firebug Lite (../Firebug/firebug.js script) is │ │ │ │ │ + * included. We explicitly require the Firebug Lite script to trigger │ │ │ │ │ + * functionality of the OpenLayers.Console methods. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIFunction: log │ │ │ │ │ + * Log an object in the console. The Firebug Lite console logs string │ │ │ │ │ + * representation of objects. Given multiple arguments, they will │ │ │ │ │ + * be cast to strings and logged with a space delimiter. If the first │ │ │ │ │ + * argument is a string with printf-like formatting, subsequent arguments │ │ │ │ │ + * will be used in string substitution. Any additional arguments (beyond │ │ │ │ │ + * the number substituted in a format string) will be appended in a space- │ │ │ │ │ + * delimited line. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * object - {Object} │ │ │ │ │ + */ │ │ │ │ │ + log: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIFunction: debug │ │ │ │ │ + * Writes a message to the console, including a hyperlink to the line │ │ │ │ │ + * where it was called. │ │ │ │ │ + * │ │ │ │ │ + * May be called with multiple arguments as with OpenLayers.Console.log(). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * object - {Object} │ │ │ │ │ + */ │ │ │ │ │ + debug: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIFunction: info │ │ │ │ │ + * Writes a message to the console with the visual "info" icon and color │ │ │ │ │ + * coding and a hyperlink to the line where it was called. │ │ │ │ │ + * │ │ │ │ │ + * May be called with multiple arguments as with OpenLayers.Console.log(). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * object - {Object} │ │ │ │ │ + */ │ │ │ │ │ + info: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIFunction: warn │ │ │ │ │ + * Writes a message to the console with the visual "warning" icon and │ │ │ │ │ + * color coding and a hyperlink to the line where it was called. │ │ │ │ │ + * │ │ │ │ │ + * May be called with multiple arguments as with OpenLayers.Console.log(). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * object - {Object} │ │ │ │ │ + */ │ │ │ │ │ + warn: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIFunction: error │ │ │ │ │ + * Writes a message to the console with the visual "error" icon and color │ │ │ │ │ + * coding and a hyperlink to the line where it was called. │ │ │ │ │ + * │ │ │ │ │ + * May be called with multiple arguments as with OpenLayers.Console.log(). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * object - {Object} │ │ │ │ │ + */ │ │ │ │ │ + error: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIFunction: userError │ │ │ │ │ + * A single interface for showing error messages to the user. The default │ │ │ │ │ + * behavior is a Javascript alert, though this can be overridden by │ │ │ │ │ + * reassigning OpenLayers.Console.userError to a different function. │ │ │ │ │ + * │ │ │ │ │ + * Expects a single error message │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * error - {Object} │ │ │ │ │ + */ │ │ │ │ │ + userError: function(error) { │ │ │ │ │ + alert(error); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIFunction: assert │ │ │ │ │ + * Tests that an expression is true. If not, it will write a message to │ │ │ │ │ + * the console and throw an exception. │ │ │ │ │ + * │ │ │ │ │ + * May be called with multiple arguments as with OpenLayers.Console.log(). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * object - {Object} │ │ │ │ │ + */ │ │ │ │ │ + assert: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIFunction: dir │ │ │ │ │ + * Prints an interactive listing of all properties of the object. This │ │ │ │ │ + * looks identical to the view that you would see in the DOM tab. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * object - {Object} │ │ │ │ │ + */ │ │ │ │ │ + dir: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIFunction: dirxml │ │ │ │ │ + * Prints the XML source tree of an HTML or XML element. This looks │ │ │ │ │ + * identical to the view that you would see in the HTML tab. You can click │ │ │ │ │ + * on any node to inspect it in the HTML tab. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * object - {Object} │ │ │ │ │ + */ │ │ │ │ │ + dirxml: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIFunction: trace │ │ │ │ │ + * Prints an interactive stack trace of JavaScript execution at the point │ │ │ │ │ + * where it is called. The stack trace details the functions on the stack, │ │ │ │ │ + * as well as the values that were passed as arguments to each function. │ │ │ │ │ + * You can click each function to take you to its source in the Script tab, │ │ │ │ │ + * and click each argument value to inspect it in the DOM or HTML tabs. │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + trace: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIFunction: group │ │ │ │ │ + * Writes a message to the console and opens a nested block to indent all │ │ │ │ │ + * future messages sent to the console. Call OpenLayers.Console.groupEnd() │ │ │ │ │ + * to close the block. │ │ │ │ │ + * │ │ │ │ │ + * May be called with multiple arguments as with OpenLayers.Console.log(). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * object - {Object} │ │ │ │ │ + */ │ │ │ │ │ + group: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIFunction: groupEnd │ │ │ │ │ + * Closes the most recently opened block created by a call to │ │ │ │ │ + * OpenLayers.Console.group │ │ │ │ │ + */ │ │ │ │ │ + groupEnd: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIFunction: time │ │ │ │ │ + * Creates a new timer under the given name. Call │ │ │ │ │ + * OpenLayers.Console.timeEnd(name) │ │ │ │ │ + * with the same name to stop the timer and print the time elapsed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} │ │ │ │ │ + */ │ │ │ │ │ + time: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIFunction: timeEnd │ │ │ │ │ + * Stops a timer created by a call to OpenLayers.Console.time(name) and │ │ │ │ │ + * writes the time elapsed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} │ │ │ │ │ + */ │ │ │ │ │ + timeEnd: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIFunction: profile │ │ │ │ │ + * Turns on the JavaScript profiler. The optional argument title would │ │ │ │ │ + * contain the text to be printed in the header of the profile report. │ │ │ │ │ + * │ │ │ │ │ + * This function is not currently implemented in Firebug Lite. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * title - {String} Optional title for the profiler │ │ │ │ │ + */ │ │ │ │ │ + profile: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIFunction: profileEnd │ │ │ │ │ + * Turns off the JavaScript profiler and prints its report. │ │ │ │ │ + * │ │ │ │ │ + * This function is not currently implemented in Firebug Lite. │ │ │ │ │ + */ │ │ │ │ │ + profileEnd: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIFunction: count │ │ │ │ │ + * Writes the number of times that the line of code where count was called │ │ │ │ │ + * was executed. The optional argument title will print a message in │ │ │ │ │ + * addition to the number of the count. │ │ │ │ │ + * │ │ │ │ │ + * This function is not currently implemented in Firebug Lite. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * title - {String} Optional title to be printed with count │ │ │ │ │ + */ │ │ │ │ │ + count: function() {}, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Console" │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Execute an anonymous function to extend the OpenLayers.Console namespace │ │ │ │ │ + * if the firebug.js script is included. This closure is used so that the │ │ │ │ │ + * "scripts" and "i" variables don't pollute the global namespace. │ │ │ │ │ + */ │ │ │ │ │ +(function() { │ │ │ │ │ + /** │ │ │ │ │ + * If Firebug Lite is included (before this script), re-route all │ │ │ │ │ + * OpenLayers.Console calls to the console object. │ │ │ │ │ + */ │ │ │ │ │ + var scripts = document.getElementsByTagName("script"); │ │ │ │ │ + for (var i = 0, len = scripts.length; i < len; ++i) { │ │ │ │ │ + if (scripts[i].src.indexOf("firebug.js") != -1) { │ │ │ │ │ + if (console) { │ │ │ │ │ + OpenLayers.Util.extend(OpenLayers.Console, console); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ +})(); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ OpenLayers/Lang.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -5934,14 +4426,155 @@ │ │ │ │ │ } else { │ │ │ │ │ str += coordinate < 0 ? OpenLayers.i18n("S") : OpenLayers.i18n("N"); │ │ │ │ │ } │ │ │ │ │ return str; │ │ │ │ │ }; │ │ │ │ │ │ │ │ │ │ /* ====================================================================== │ │ │ │ │ + OpenLayers/Util/vendorPrefix.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/SingleFile.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Util = OpenLayers.Util || {}; │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Util.vendorPrefix │ │ │ │ │ + * A collection of utility functions to detect vendor prefixed features │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Util.vendorPrefix = (function() { │ │ │ │ │ + "use strict"; │ │ │ │ │ + │ │ │ │ │ + var VENDOR_PREFIXES = ["", "O", "ms", "Moz", "Webkit"], │ │ │ │ │ + divStyle = document.createElement("div").style, │ │ │ │ │ + cssCache = {}, │ │ │ │ │ + jsCache = {}; │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Function: domToCss │ │ │ │ │ + * Converts a upper camel case DOM style property name to a CSS property │ │ │ │ │ + * i.e. transformOrigin -> transform-origin │ │ │ │ │ + * or WebkitTransformOrigin -> -webkit-transform-origin │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * prefixedDom - {String} The property to convert │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The CSS property │ │ │ │ │ + */ │ │ │ │ │ + function domToCss(prefixedDom) { │ │ │ │ │ + if (!prefixedDom) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + return prefixedDom. │ │ │ │ │ + replace(/([A-Z])/g, function(c) { │ │ │ │ │ + return "-" + c.toLowerCase(); │ │ │ │ │ + }). │ │ │ │ │ + replace(/^ms-/, "-ms-"); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: css │ │ │ │ │ + * Detect which property is used for a CSS property │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * property - {String} The standard (unprefixed) CSS property name │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The standard CSS property, prefixed property or null if not │ │ │ │ │ + * supported │ │ │ │ │ + */ │ │ │ │ │ + function css(property) { │ │ │ │ │ + if (cssCache[property] === undefined) { │ │ │ │ │ + var domProperty = property. │ │ │ │ │ + replace(/(-[\s\S])/g, function(c) { │ │ │ │ │ + return c.charAt(1).toUpperCase(); │ │ │ │ │ + }); │ │ │ │ │ + var prefixedDom = style(domProperty); │ │ │ │ │ + cssCache[property] = domToCss(prefixedDom); │ │ │ │ │ + } │ │ │ │ │ + return cssCache[property]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: js │ │ │ │ │ + * Detect which property is used for a JS property/method │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} The object to test on │ │ │ │ │ + * property - {String} The standard (unprefixed) JS property name │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The standard JS property, prefixed property or null if not │ │ │ │ │ + * supported │ │ │ │ │ + */ │ │ │ │ │ + function js(obj, property) { │ │ │ │ │ + if (jsCache[property] === undefined) { │ │ │ │ │ + var tmpProp, │ │ │ │ │ + i = 0, │ │ │ │ │ + l = VENDOR_PREFIXES.length, │ │ │ │ │ + prefix, │ │ │ │ │ + isStyleObj = (typeof obj.cssText !== "undefined"); │ │ │ │ │ + │ │ │ │ │ + jsCache[property] = null; │ │ │ │ │ + for (; i < l; i++) { │ │ │ │ │ + prefix = VENDOR_PREFIXES[i]; │ │ │ │ │ + if (prefix) { │ │ │ │ │ + if (!isStyleObj) { │ │ │ │ │ + // js prefix should be lower-case, while style │ │ │ │ │ + // properties have upper case on first character │ │ │ │ │ + prefix = prefix.toLowerCase(); │ │ │ │ │ + } │ │ │ │ │ + tmpProp = prefix + property.charAt(0).toUpperCase() + property.slice(1); │ │ │ │ │ + } else { │ │ │ │ │ + tmpProp = property; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (obj[tmpProp] !== undefined) { │ │ │ │ │ + jsCache[property] = tmpProp; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return jsCache[property]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: style │ │ │ │ │ + * Detect which property is used for a DOM style property │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * property - {String} The standard (unprefixed) style property name │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The standard style property, prefixed property or null if not │ │ │ │ │ + * supported │ │ │ │ │ + */ │ │ │ │ │ + function style(property) { │ │ │ │ │ + return js(divStyle, property); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return { │ │ │ │ │ + css: css, │ │ │ │ │ + js: js, │ │ │ │ │ + style: style, │ │ │ │ │ + │ │ │ │ │ + // used for testing │ │ │ │ │ + cssCache: cssCache, │ │ │ │ │ + jsCache: jsCache │ │ │ │ │ + }; │ │ │ │ │ +}()); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ OpenLayers/Events.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -7110,14 +5743,120 @@ │ │ │ │ │ │ │ │ │ │ OpenLayers.Event.observe(element, 'MSPointerUp', cb); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Events" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ + OpenLayers/Animation.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/SingleFile.js │ │ │ │ │ + * @requires OpenLayers/Util/vendorPrefix.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Animation │ │ │ │ │ + * A collection of utility functions for executing methods that repaint a │ │ │ │ │ + * portion of the browser window. These methods take advantage of the │ │ │ │ │ + * browser's scheduled repaints where requestAnimationFrame is available. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Animation = (function(window) { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: isNative │ │ │ │ │ + * {Boolean} true if a native requestAnimationFrame function is available │ │ │ │ │ + */ │ │ │ │ │ + var requestAnimationFrame = OpenLayers.Util.vendorPrefix.js(window, "requestAnimationFrame"); │ │ │ │ │ + var isNative = !!(requestAnimationFrame); │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Function: requestFrame │ │ │ │ │ + * Schedule a function to be called at the next available animation frame. │ │ │ │ │ + * Uses the native method where available. Where requestAnimationFrame is │ │ │ │ │ + * not available, setTimeout will be called with a 16ms delay. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * callback - {Function} The function to be called at the next animation frame. │ │ │ │ │ + * element - {DOMElement} Optional element that visually bounds the animation. │ │ │ │ │ + */ │ │ │ │ │ + var requestFrame = (function() { │ │ │ │ │ + var request = window[requestAnimationFrame] || │ │ │ │ │ + function(callback, element) { │ │ │ │ │ + window.setTimeout(callback, 16); │ │ │ │ │ + }; │ │ │ │ │ + // bind to window to avoid illegal invocation of native function │ │ │ │ │ + return function(callback, element) { │ │ │ │ │ + request.apply(window, [callback, element]); │ │ │ │ │ + }; │ │ │ │ │ + })(); │ │ │ │ │ + │ │ │ │ │ + // private variables for animation loops │ │ │ │ │ + var counter = 0; │ │ │ │ │ + var loops = {}; │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Function: start │ │ │ │ │ + * Executes a method with in series for some │ │ │ │ │ + * duration. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * callback - {Function} The function to be called at the next animation frame. │ │ │ │ │ + * duration - {Number} Optional duration for the loop. If not provided, the │ │ │ │ │ + * animation loop will execute indefinitely. │ │ │ │ │ + * element - {DOMElement} Optional element that visually bounds the animation. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number} Identifier for the animation loop. Used to stop animations with │ │ │ │ │ + * . │ │ │ │ │ + */ │ │ │ │ │ + function start(callback, duration, element) { │ │ │ │ │ + duration = duration > 0 ? duration : Number.POSITIVE_INFINITY; │ │ │ │ │ + var id = ++counter; │ │ │ │ │ + var start = +new Date; │ │ │ │ │ + loops[id] = function() { │ │ │ │ │ + if (loops[id] && +new Date - start <= duration) { │ │ │ │ │ + callback(); │ │ │ │ │ + if (loops[id]) { │ │ │ │ │ + requestFrame(loops[id], element); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + delete loops[id]; │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + requestFrame(loops[id], element); │ │ │ │ │ + return id; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Function: stop │ │ │ │ │ + * Terminates an animation loop started with . │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {Number} Identifier returned from . │ │ │ │ │ + */ │ │ │ │ │ + function stop(id) { │ │ │ │ │ + delete loops[id]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return { │ │ │ │ │ + isNative: isNative, │ │ │ │ │ + requestFrame: requestFrame, │ │ │ │ │ + start: start, │ │ │ │ │ + stop: stop │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ +})(window); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ OpenLayers/Tween.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -10718,3715 +9457,1019 @@ │ │ │ │ │ OpenLayers.Map.TILE_WIDTH = 256; │ │ │ │ │ /** │ │ │ │ │ * Constant: TILE_HEIGHT │ │ │ │ │ * {Integer} 256 Default tile height (unless otherwise specified) │ │ │ │ │ */ │ │ │ │ │ OpenLayers.Map.TILE_HEIGHT = 256; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer.js │ │ │ │ │ + OpenLayers/Renderer.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Map.js │ │ │ │ │ - * @requires OpenLayers/Projection.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer │ │ │ │ │ + * Class: OpenLayers.Renderer │ │ │ │ │ + * This is the base class for all renderers. │ │ │ │ │ + * │ │ │ │ │ + * This is based on a merger code written by Paul Spencer and Bertil Chapuis. │ │ │ │ │ + * It is largely composed of virtual functions that are to be implemented │ │ │ │ │ + * in technology-specific subclasses, but there is some generic code too. │ │ │ │ │ + * │ │ │ │ │ + * The functions that *are* implemented here merely deal with the maintenance │ │ │ │ │ + * of the size and extent variables, as well as the cached 'resolution' │ │ │ │ │ + * value. │ │ │ │ │ + * │ │ │ │ │ + * A note to the user that all subclasses should use getResolution() instead │ │ │ │ │ + * of directly accessing this.resolution in order to correctly use the │ │ │ │ │ + * cacheing system. │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: id │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - id: null, │ │ │ │ │ +OpenLayers.Renderer = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: name │ │ │ │ │ - * {String} │ │ │ │ │ + * Property: container │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - name: null, │ │ │ │ │ + container: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: div │ │ │ │ │ + /** │ │ │ │ │ + * Property: root │ │ │ │ │ * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - div: null, │ │ │ │ │ + root: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: opacity │ │ │ │ │ - * {Float} The layer's opacity. Float number between 0.0 and 1.0. Default │ │ │ │ │ - * is 1. │ │ │ │ │ + /** │ │ │ │ │ + * Property: extent │ │ │ │ │ + * {} │ │ │ │ │ */ │ │ │ │ │ - opacity: 1, │ │ │ │ │ + extent: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: alwaysInRange │ │ │ │ │ - * {Boolean} If a layer's display should not be scale-based, this should │ │ │ │ │ - * be set to true. This will cause the layer, as an overlay, to always │ │ │ │ │ - * be 'active', by always returning true from the calculateInRange() │ │ │ │ │ - * function. │ │ │ │ │ - * │ │ │ │ │ - * If not explicitly specified for a layer, its value will be │ │ │ │ │ - * determined on startup in initResolutions() based on whether or not │ │ │ │ │ - * any scale-specific properties have been set as options on the │ │ │ │ │ - * layer. If no scale-specific options have been set on the layer, we │ │ │ │ │ - * assume that it should always be in range. │ │ │ │ │ - * │ │ │ │ │ - * See #987 for more info. │ │ │ │ │ + * Property: locked │ │ │ │ │ + * {Boolean} If the renderer is currently in a state where many things │ │ │ │ │ + * are changing, the 'locked' property is set to true. This means │ │ │ │ │ + * that renderers can expect at least one more drawFeature event to be │ │ │ │ │ + * called with the 'locked' property set to 'true': In some renderers, │ │ │ │ │ + * this might make sense to use as a 'only update local information' │ │ │ │ │ + * flag. │ │ │ │ │ */ │ │ │ │ │ - alwaysInRange: null, │ │ │ │ │ + locked: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: RESOLUTION_PROPERTIES │ │ │ │ │ - * {Array} The properties that are used for calculating resolutions │ │ │ │ │ - * information. │ │ │ │ │ + /** │ │ │ │ │ + * Property: size │ │ │ │ │ + * {} │ │ │ │ │ */ │ │ │ │ │ - RESOLUTION_PROPERTIES: [ │ │ │ │ │ - 'scales', 'resolutions', │ │ │ │ │ - 'maxScale', 'minScale', │ │ │ │ │ - 'maxResolution', 'minResolution', │ │ │ │ │ - 'numZoomLevels', 'maxZoomLevel' │ │ │ │ │ - ], │ │ │ │ │ + size: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {} │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * layer.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Listeners will be called with a reference to an event object. The │ │ │ │ │ - * properties of this event depends on exactly what happened. │ │ │ │ │ - * │ │ │ │ │ - * All event objects have at least the following properties: │ │ │ │ │ - * object - {Object} A reference to layer.events.object. │ │ │ │ │ - * element - {DOMElement} A reference to layer.events.element. │ │ │ │ │ - * │ │ │ │ │ - * Supported map event types: │ │ │ │ │ - * loadstart - Triggered when layer loading starts. When using a Vector │ │ │ │ │ - * layer with a Fixed or BBOX strategy, the event object includes │ │ │ │ │ - * a *filter* property holding the OpenLayers.Filter used when │ │ │ │ │ - * calling read on the protocol. │ │ │ │ │ - * loadend - Triggered when layer loading ends. When using a Vector layer │ │ │ │ │ - * with a Fixed or BBOX strategy, the event object includes a │ │ │ │ │ - * *response* property holding an OpenLayers.Protocol.Response object. │ │ │ │ │ - * visibilitychanged - Triggered when the layer's visibility property is │ │ │ │ │ - * changed, e.g. by turning the layer on or off in the layer switcher. │ │ │ │ │ - * Note that the actual visibility of the layer can also change if it │ │ │ │ │ - * gets out of range (see ). If you also want to catch │ │ │ │ │ - * these cases, register for the map's 'changelayer' event instead. │ │ │ │ │ - * move - Triggered when layer moves (triggered with every mousemove │ │ │ │ │ - * during a drag). │ │ │ │ │ - * moveend - Triggered when layer is done moving, object passed as │ │ │ │ │ - * argument has a zoomChanged boolean property which tells that the │ │ │ │ │ - * zoom has changed. │ │ │ │ │ - * added - Triggered after the layer is added to a map. Listeners will │ │ │ │ │ - * receive an object with a *map* property referencing the map and a │ │ │ │ │ - * *layer* property referencing the layer. │ │ │ │ │ - * removed - Triggered after the layer is removed from the map. Listeners │ │ │ │ │ - * will receive an object with a *map* property referencing the map and │ │ │ │ │ - * a *layer* property referencing the layer. │ │ │ │ │ + * Property: resolution │ │ │ │ │ + * {Float} cache of current map resolution │ │ │ │ │ */ │ │ │ │ │ - events: null, │ │ │ │ │ + resolution: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: map │ │ │ │ │ - * {} This variable is set when the layer is added to │ │ │ │ │ - * the map, via the accessor function setMap(). │ │ │ │ │ + * Property: map │ │ │ │ │ + * {} Reference to the map -- this is set in Vector's setMap() │ │ │ │ │ */ │ │ │ │ │ map: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} Whether or not the layer is a base layer. This should be set │ │ │ │ │ - * individually by all subclasses. Default is false │ │ │ │ │ - */ │ │ │ │ │ - isBaseLayer: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: alpha │ │ │ │ │ - * {Boolean} The layer's images have an alpha channel. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - alpha: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: displayInLayerSwitcher │ │ │ │ │ - * {Boolean} Display the layer's name in the layer switcher. Default is │ │ │ │ │ - * true. │ │ │ │ │ + * Property: featureDx │ │ │ │ │ + * {Number} Feature offset in x direction. Will be calculated for and │ │ │ │ │ + * applied to the current feature while rendering (see │ │ │ │ │ + * ). │ │ │ │ │ */ │ │ │ │ │ - displayInLayerSwitcher: true, │ │ │ │ │ + featureDx: 0, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: visibility │ │ │ │ │ - * {Boolean} The layer should be displayed in the map. Default is true. │ │ │ │ │ + * Constructor: OpenLayers.Renderer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * containerID - {} │ │ │ │ │ + * options - {Object} options for this renderer. See sublcasses for │ │ │ │ │ + * supported options. │ │ │ │ │ */ │ │ │ │ │ - visibility: true, │ │ │ │ │ + initialize: function(containerID, options) { │ │ │ │ │ + this.container = OpenLayers.Util.getElement(containerID); │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: attribution │ │ │ │ │ - * {String} Attribution string, displayed when an │ │ │ │ │ - * has been added to the map. │ │ │ │ │ - */ │ │ │ │ │ - attribution: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: inRange │ │ │ │ │ - * {Boolean} The current map resolution is within the layer's min/max │ │ │ │ │ - * range. This is set in whenever the zoom │ │ │ │ │ - * changes. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ */ │ │ │ │ │ - inRange: false, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.container = null; │ │ │ │ │ + this.extent = null; │ │ │ │ │ + this.size = null; │ │ │ │ │ + this.resolution = null; │ │ │ │ │ + this.map = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Propery: imageSize │ │ │ │ │ - * {} For layers with a gutter, the image is larger than │ │ │ │ │ - * the tile by twice the gutter in each dimension. │ │ │ │ │ - */ │ │ │ │ │ - imageSize: null, │ │ │ │ │ - │ │ │ │ │ - // OPTIONS │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: options │ │ │ │ │ - * {Object} An optional object whose properties will be set on the layer. │ │ │ │ │ - * Any of the layer properties can be set as a property of the options │ │ │ │ │ - * object and sent to the constructor when the layer is created. │ │ │ │ │ + * APIMethod: supported │ │ │ │ │ + * This should be overridden by specific subclasses │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the browser supports the renderer class │ │ │ │ │ */ │ │ │ │ │ - options: null, │ │ │ │ │ + supported: function() { │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: eventListeners │ │ │ │ │ - * {Object} If set as an option at construction, the eventListeners │ │ │ │ │ - * object will be registered with . Object │ │ │ │ │ - * structure must be a listeners object as shown in the example for │ │ │ │ │ - * the events.on method. │ │ │ │ │ + * Method: setExtent │ │ │ │ │ + * Set the visible part of the layer. │ │ │ │ │ + * │ │ │ │ │ + * Resolution has probably changed, so we nullify the resolution │ │ │ │ │ + * cache (this.resolution) -- this way it will be re-computed when │ │ │ │ │ + * next it is needed. │ │ │ │ │ + * We nullify the resolution cache (this.resolution) if resolutionChanged │ │ │ │ │ + * is set to true - this way it will be re-computed on the next │ │ │ │ │ + * getResolution() request. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * extent - {} │ │ │ │ │ + * resolutionChanged - {Boolean} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ + * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ + * False otherwise. │ │ │ │ │ */ │ │ │ │ │ - eventListeners: null, │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + this.extent = extent.clone(); │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + var ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ + extent = extent.scale(1 / ratio); │ │ │ │ │ + this.extent = extent.wrapDateLine(this.map.getMaxExtent()).scale(ratio); │ │ │ │ │ + } │ │ │ │ │ + if (resolutionChanged) { │ │ │ │ │ + this.resolution = null; │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: gutter │ │ │ │ │ - * {Integer} Determines the width (in pixels) of the gutter around image │ │ │ │ │ - * tiles to ignore. By setting this property to a non-zero value, │ │ │ │ │ - * images will be requested that are wider and taller than the tile │ │ │ │ │ - * size by a value of 2 x gutter. This allows artifacts of rendering │ │ │ │ │ - * at tile edges to be ignored. Set a gutter value that is equal to │ │ │ │ │ - * half the size of the widest symbol that needs to be displayed. │ │ │ │ │ - * Defaults to zero. Non-tiled layers always have zero gutter. │ │ │ │ │ + * Method: setSize │ │ │ │ │ + * Sets the size of the drawing surface. │ │ │ │ │ + * │ │ │ │ │ + * Resolution has probably changed, so we nullify the resolution │ │ │ │ │ + * cache (this.resolution) -- this way it will be re-computed when │ │ │ │ │ + * next it is needed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * size - {} │ │ │ │ │ */ │ │ │ │ │ - gutter: 0, │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + this.size = size.clone(); │ │ │ │ │ + this.resolution = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: projection │ │ │ │ │ - * {} or {} Specifies the projection of the layer. │ │ │ │ │ - * Can be set in the layer options. If not specified in the layer options, │ │ │ │ │ - * it is set to the default projection specified in the map, │ │ │ │ │ - * when the layer is added to the map. │ │ │ │ │ - * Projection along with default maxExtent and resolutions │ │ │ │ │ - * are set automatically with commercial baselayers in EPSG:3857, │ │ │ │ │ - * such as Google, Bing and OpenStreetMap, and do not need to be specified. │ │ │ │ │ - * Otherwise, if specifying projection, also set maxExtent, │ │ │ │ │ - * maxResolution or resolutions as appropriate. │ │ │ │ │ - * When using vector layers with strategies, layer projection should be set │ │ │ │ │ - * to the projection of the source data if that is different from the map default. │ │ │ │ │ - * │ │ │ │ │ - * Can be either a string or an object; │ │ │ │ │ - * if a string is passed, will be converted to an object when │ │ │ │ │ - * the layer is added to the map. │ │ │ │ │ + /** │ │ │ │ │ + * Method: getResolution │ │ │ │ │ + * Uses cached copy of resolution if available to minimize computing │ │ │ │ │ * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The current map's resolution │ │ │ │ │ */ │ │ │ │ │ - projection: null, │ │ │ │ │ + getResolution: function() { │ │ │ │ │ + this.resolution = this.resolution || this.map.getResolution(); │ │ │ │ │ + return this.resolution; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: units │ │ │ │ │ - * {String} The layer map units. Defaults to null. Possible values │ │ │ │ │ - * are 'degrees' (or 'dd'), 'm', 'ft', 'km', 'mi', 'inches'. │ │ │ │ │ - * Normally taken from the projection. │ │ │ │ │ - * Only required if both map and layers do not define a projection, │ │ │ │ │ - * or if they define a projection which does not define units. │ │ │ │ │ + * Method: drawFeature │ │ │ │ │ + * Draw the feature. The optional style argument can be used │ │ │ │ │ + * to override the feature's own style. This method should only │ │ │ │ │ + * be called from layer.drawFeature(). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {} │ │ │ │ │ + * style - {} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the feature has been drawn completely, false if not, │ │ │ │ │ + * undefined if the feature had no geometry │ │ │ │ │ */ │ │ │ │ │ - units: null, │ │ │ │ │ + drawFeature: function(feature, style) { │ │ │ │ │ + if (style == null) { │ │ │ │ │ + style = feature.style; │ │ │ │ │ + } │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + var bounds = feature.geometry.getBounds(); │ │ │ │ │ + if (bounds) { │ │ │ │ │ + var worldBounds; │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + worldBounds = this.map.getMaxExtent(); │ │ │ │ │ + } │ │ │ │ │ + if (!bounds.intersectsBounds(this.extent, { │ │ │ │ │ + worldBounds: worldBounds │ │ │ │ │ + })) { │ │ │ │ │ + style = { │ │ │ │ │ + display: "none" │ │ │ │ │ + }; │ │ │ │ │ + } else { │ │ │ │ │ + this.calculateFeatureDx(bounds, worldBounds); │ │ │ │ │ + } │ │ │ │ │ + var rendered = this.drawGeometry(feature.geometry, style, feature.id); │ │ │ │ │ + if (style.display != "none" && style.label && rendered !== false) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: scales │ │ │ │ │ - * {Array} An array of map scales in descending order. The values in the │ │ │ │ │ - * array correspond to the map scale denominator. Note that these │ │ │ │ │ - * values only make sense if the display (monitor) resolution of the │ │ │ │ │ - * client is correctly guessed by whomever is configuring the │ │ │ │ │ - * application. In addition, the units property must also be set. │ │ │ │ │ - * Use instead wherever possible. │ │ │ │ │ - */ │ │ │ │ │ - scales: null, │ │ │ │ │ + var location = feature.geometry.getCentroid(); │ │ │ │ │ + if (style.labelXOffset || style.labelYOffset) { │ │ │ │ │ + var xOffset = isNaN(style.labelXOffset) ? 0 : style.labelXOffset; │ │ │ │ │ + var yOffset = isNaN(style.labelYOffset) ? 0 : style.labelYOffset; │ │ │ │ │ + var res = this.getResolution(); │ │ │ │ │ + location.move(xOffset * res, yOffset * res); │ │ │ │ │ + } │ │ │ │ │ + this.drawText(feature.id, style, location); │ │ │ │ │ + } else { │ │ │ │ │ + this.removeText(feature.id); │ │ │ │ │ + } │ │ │ │ │ + return rendered; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: resolutions │ │ │ │ │ - * {Array} A list of map resolutions (map units per pixel) in descending │ │ │ │ │ - * order. If this is not set in the layer constructor, it will be set │ │ │ │ │ - * based on other resolution related properties (maxExtent, │ │ │ │ │ - * maxResolution, maxScale, etc.). │ │ │ │ │ + * Method: calculateFeatureDx │ │ │ │ │ + * {Number} Calculates the feature offset in x direction. Looking at the │ │ │ │ │ + * center of the feature bounds and the renderer extent, we calculate how │ │ │ │ │ + * many world widths the two are away from each other. This distance is │ │ │ │ │ + * used to shift the feature as close as possible to the center of the │ │ │ │ │ + * current enderer extent, which ensures that the feature is visible in the │ │ │ │ │ + * current viewport. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {} Bounds of the feature │ │ │ │ │ + * worldBounds - {} Bounds of the world │ │ │ │ │ */ │ │ │ │ │ - resolutions: null, │ │ │ │ │ + calculateFeatureDx: function(bounds, worldBounds) { │ │ │ │ │ + this.featureDx = 0; │ │ │ │ │ + if (worldBounds) { │ │ │ │ │ + var worldWidth = worldBounds.getWidth(), │ │ │ │ │ + rendererCenterX = (this.extent.left + this.extent.right) / 2, │ │ │ │ │ + featureCenterX = (bounds.left + bounds.right) / 2, │ │ │ │ │ + worldsAway = Math.round((featureCenterX - rendererCenterX) / worldWidth); │ │ │ │ │ + this.featureDx = worldsAway * worldWidth; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: maxExtent │ │ │ │ │ - * {|Array} If provided as an array, the array │ │ │ │ │ - * should consist of four values (left, bottom, right, top). │ │ │ │ │ - * The maximum extent for the layer. Defaults to null. │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawGeometry │ │ │ │ │ * │ │ │ │ │ - * The center of these bounds will not stray outside │ │ │ │ │ - * of the viewport extent during panning. In addition, if │ │ │ │ │ - * is set to false, data will not be │ │ │ │ │ - * requested that falls completely outside of these bounds. │ │ │ │ │ - */ │ │ │ │ │ - maxExtent: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: minExtent │ │ │ │ │ - * {|Array} If provided as an array, the array │ │ │ │ │ - * should consist of four values (left, bottom, right, top). │ │ │ │ │ - * The minimum extent for the layer. Defaults to null. │ │ │ │ │ + * Draw a geometry. This should only be called from the renderer itself. │ │ │ │ │ + * Use layer.drawFeature() from outside the renderer. │ │ │ │ │ + * virtual function │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {} │ │ │ │ │ */ │ │ │ │ │ - minExtent: null, │ │ │ │ │ + drawGeometry: function(geometry, style, featureId) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: maxResolution │ │ │ │ │ - * {Float} Default max is 360 deg / 256 px, which corresponds to │ │ │ │ │ - * zoom level 0 on gmaps. Specify a different value in the layer │ │ │ │ │ - * options if you are not using the default │ │ │ │ │ - * and displaying the whole world. │ │ │ │ │ + * Method: drawText │ │ │ │ │ + * Function for drawing text labels. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * style - │ │ │ │ │ + * location - {} │ │ │ │ │ */ │ │ │ │ │ - maxResolution: null, │ │ │ │ │ + drawText: function(featureId, style, location) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: minResolution │ │ │ │ │ - * {Float} │ │ │ │ │ + * Method: removeText │ │ │ │ │ + * Function for removing text labels. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureId - {String} │ │ │ │ │ */ │ │ │ │ │ - minResolution: null, │ │ │ │ │ + removeText: function(featureId) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: numZoomLevels │ │ │ │ │ - * {Integer} │ │ │ │ │ + * Method: clear │ │ │ │ │ + * Clear all vectors from the renderer. │ │ │ │ │ + * virtual function. │ │ │ │ │ */ │ │ │ │ │ - numZoomLevels: null, │ │ │ │ │ + clear: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: minScale │ │ │ │ │ - * {Float} │ │ │ │ │ + * Method: getFeatureIdFromEvent │ │ │ │ │ + * Returns a feature id from an event on the renderer. │ │ │ │ │ + * How this happens is specific to the renderer. This should be │ │ │ │ │ + * called from layer.getFeatureFromEvent(). │ │ │ │ │ + * Virtual function. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A feature id or undefined. │ │ │ │ │ */ │ │ │ │ │ - minScale: null, │ │ │ │ │ + getFeatureIdFromEvent: function(evt) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: maxScale │ │ │ │ │ - * {Float} │ │ │ │ │ + * Method: eraseFeatures │ │ │ │ │ + * This is called by the layer to erase features │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array()} │ │ │ │ │ */ │ │ │ │ │ - maxScale: null, │ │ │ │ │ + eraseFeatures: function(features) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + this.eraseGeometry(feature.geometry, feature.id); │ │ │ │ │ + this.removeText(feature.id); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: displayOutsideMaxExtent │ │ │ │ │ - * {Boolean} Request map tiles that are completely outside of the max │ │ │ │ │ - * extent for this layer. Defaults to false. │ │ │ │ │ + * Method: eraseGeometry │ │ │ │ │ + * Remove a geometry from the renderer (by id). │ │ │ │ │ + * virtual function. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ */ │ │ │ │ │ - displayOutsideMaxExtent: false, │ │ │ │ │ + eraseGeometry: function(geometry, featureId) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: wrapDateLine │ │ │ │ │ - * {Boolean} Wraps the world at the international dateline, so the map can │ │ │ │ │ - * be panned infinitely in longitudinal direction. Only use this on the │ │ │ │ │ - * base layer, and only if the layer's maxExtent equals the world bounds. │ │ │ │ │ - * #487 for more info. │ │ │ │ │ + * Method: moveRoot │ │ │ │ │ + * moves this renderer's root to a (different) renderer. │ │ │ │ │ + * To be implemented by subclasses that require a common renderer root for │ │ │ │ │ + * feature selection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * renderer - {} target renderer for the moved root │ │ │ │ │ */ │ │ │ │ │ - wrapDateLine: false, │ │ │ │ │ + moveRoot: function(renderer) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: metadata │ │ │ │ │ - * {Object} This object can be used to store additional information on a │ │ │ │ │ - * layer object. │ │ │ │ │ + * Method: getRenderLayerId │ │ │ │ │ + * Gets the layer that this renderer's output appears on. If moveRoot was │ │ │ │ │ + * used, this will be different from the id of the layer containing the │ │ │ │ │ + * features rendered by this renderer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} the id of the output layer. │ │ │ │ │ */ │ │ │ │ │ - metadata: null, │ │ │ │ │ + getRenderLayerId: function() { │ │ │ │ │ + return this.container.id; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer │ │ │ │ │ - * │ │ │ │ │ + * Method: applyDefaultSymbolizer │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} The layer name │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + * symbolizer - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - │ │ │ │ │ - this.metadata = {}; │ │ │ │ │ - │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - // make sure we respect alwaysInRange if set on the prototype │ │ │ │ │ - if (this.alwaysInRange != null) { │ │ │ │ │ - options.alwaysInRange = this.alwaysInRange; │ │ │ │ │ + applyDefaultSymbolizer: function(symbolizer) { │ │ │ │ │ + var result = OpenLayers.Util.extend({}, │ │ │ │ │ + OpenLayers.Renderer.defaultSymbolizer); │ │ │ │ │ + if (symbolizer.stroke === false) { │ │ │ │ │ + delete result.strokeWidth; │ │ │ │ │ + delete result.strokeColor; │ │ │ │ │ } │ │ │ │ │ - this.addOptions(options); │ │ │ │ │ + if (symbolizer.fill === false) { │ │ │ │ │ + delete result.fillColor; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.extend(result, symbolizer); │ │ │ │ │ + return result; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.name = name; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ - if (this.id == null) { │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.defaultSymbolizer │ │ │ │ │ + * {Object} Properties from this symbolizer will be applied to symbolizers │ │ │ │ │ + * with missing properties. This can also be used to set a global │ │ │ │ │ + * symbolizer default in OpenLayers. To be SLD 1.x compliant, add the │ │ │ │ │ + * following code before rendering any vector features: │ │ │ │ │ + * (code) │ │ │ │ │ + * OpenLayers.Renderer.defaultSymbolizer = { │ │ │ │ │ + * fillColor: "#808080", │ │ │ │ │ + * fillOpacity: 1, │ │ │ │ │ + * strokeColor: "#000000", │ │ │ │ │ + * strokeOpacity: 1, │ │ │ │ │ + * strokeWidth: 1, │ │ │ │ │ + * pointRadius: 3, │ │ │ │ │ + * graphicName: "square" │ │ │ │ │ + * }; │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.defaultSymbolizer = { │ │ │ │ │ + fillColor: "#000000", │ │ │ │ │ + strokeColor: "#000000", │ │ │ │ │ + strokeWidth: 2, │ │ │ │ │ + fillOpacity: 1, │ │ │ │ │ + strokeOpacity: 1, │ │ │ │ │ + pointRadius: 0, │ │ │ │ │ + labelAlign: 'cm' │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ │ │ │ │ │ - this.div = OpenLayers.Util.createDiv(this.id); │ │ │ │ │ - this.div.style.width = "100%"; │ │ │ │ │ - this.div.style.height = "100%"; │ │ │ │ │ - this.div.dir = "ltr"; │ │ │ │ │ │ │ │ │ │ - this.events = new OpenLayers.Events(this, this.div); │ │ │ │ │ - if (this.eventListeners instanceof Object) { │ │ │ │ │ - this.events.on(this.eventListeners); │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.symbol │ │ │ │ │ + * Coordinate arrays for well known (named) symbols. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.symbol = { │ │ │ │ │ + "star": [350, 75, 379, 161, 469, 161, 397, 215, 423, 301, 350, 250, 277, 301, │ │ │ │ │ + 303, 215, 231, 161, 321, 161, 350, 75 │ │ │ │ │ + ], │ │ │ │ │ + "cross": [4, 0, 6, 0, 6, 4, 10, 4, 10, 6, 6, 6, 6, 10, 4, 10, 4, 6, 0, 6, 0, 4, 4, 4, │ │ │ │ │ + 4, 0 │ │ │ │ │ + ], │ │ │ │ │ + "x": [0, 0, 25, 0, 50, 35, 75, 0, 100, 0, 65, 50, 100, 100, 75, 100, 50, 65, 25, 100, 0, 100, 35, 50, 0, 0], │ │ │ │ │ + "square": [0, 0, 0, 1, 1, 1, 1, 0, 0, 0], │ │ │ │ │ + "triangle": [0, 10, 10, 10, 5, 0, 0, 10] │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Feature.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Destroy is a destructor: this is to alleviate cyclic references which │ │ │ │ │ - * the Javascript garbage cleaner can not take care of on its own. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * setNewBaseLayer - {Boolean} Set a new base layer when this layer has │ │ │ │ │ - * been destroyed. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function(setNewBaseLayer) { │ │ │ │ │ - if (setNewBaseLayer == null) { │ │ │ │ │ - setNewBaseLayer = true; │ │ │ │ │ - } │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.removeLayer(this, setNewBaseLayer); │ │ │ │ │ - } │ │ │ │ │ - this.projection = null; │ │ │ │ │ - this.map = null; │ │ │ │ │ - this.name = null; │ │ │ │ │ - this.div = null; │ │ │ │ │ - this.options = null; │ │ │ │ │ │ │ │ │ │ - if (this.events) { │ │ │ │ │ - if (this.eventListeners) { │ │ │ │ │ - this.events.un(this.eventListeners); │ │ │ │ │ - } │ │ │ │ │ - this.events.destroy(); │ │ │ │ │ - } │ │ │ │ │ - this.eventListeners = null; │ │ │ │ │ - this.events = null; │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {} The layer to be cloned │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} An exact clone of this │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Feature │ │ │ │ │ + * Features are combinations of geography and attributes. The OpenLayers.Feature │ │ │ │ │ + * class specifically combines a marker and a lonlat. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Feature = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {} │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ + layer: null, │ │ │ │ │ │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer(this.name, this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: id │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + id: null, │ │ │ │ │ │ │ │ │ │ - // catch any randomly tagged-on properties │ │ │ │ │ - OpenLayers.Util.applyDefaults(obj, this); │ │ │ │ │ + /** │ │ │ │ │ + * Property: lonlat │ │ │ │ │ + * {} │ │ │ │ │ + */ │ │ │ │ │ + lonlat: null, │ │ │ │ │ │ │ │ │ │ - // a cloned layer should never have its map property set │ │ │ │ │ - // because it has not been added to a map yet. │ │ │ │ │ - obj.map = null; │ │ │ │ │ + /** │ │ │ │ │ + * Property: data │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ + data: null, │ │ │ │ │ │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: marker │ │ │ │ │ + * {} │ │ │ │ │ + */ │ │ │ │ │ + marker: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getOptions │ │ │ │ │ - * Extracts an object from the layer with the properties that were set as │ │ │ │ │ - * options, but updates them with the values currently set on the │ │ │ │ │ - * instance. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} the of the layer, representing the current state. │ │ │ │ │ + * APIProperty: popupClass │ │ │ │ │ + * {} The class which will be used to instantiate │ │ │ │ │ + * a new Popup. Default is . │ │ │ │ │ */ │ │ │ │ │ - getOptions: function() { │ │ │ │ │ - var options = {}; │ │ │ │ │ - for (var o in this.options) { │ │ │ │ │ - options[o] = this[o]; │ │ │ │ │ - } │ │ │ │ │ - return options; │ │ │ │ │ - }, │ │ │ │ │ + popupClass: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setName │ │ │ │ │ - * Sets the new layer name for this layer. Can trigger a changelayer event │ │ │ │ │ - * on the map. │ │ │ │ │ + * Property: popup │ │ │ │ │ + * {} │ │ │ │ │ + */ │ │ │ │ │ + popup: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Feature │ │ │ │ │ + * Constructor for features. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * newName - {String} The new name. │ │ │ │ │ + * layer - {} │ │ │ │ │ + * lonlat - {} │ │ │ │ │ + * data - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} │ │ │ │ │ */ │ │ │ │ │ - setName: function(newName) { │ │ │ │ │ - if (newName != this.name) { │ │ │ │ │ - this.name = newName; │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "name" │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + initialize: function(layer, lonlat, data) { │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + this.lonlat = lonlat; │ │ │ │ │ + this.data = (data != null) ? data : {}; │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: addOptions │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newOptions - {Object} │ │ │ │ │ - * reinitialize - {Boolean} If set to true, and if resolution options of the │ │ │ │ │ - * current baseLayer were changed, the map will be recentered to make │ │ │ │ │ - * sure that it is displayed with a valid resolution, and a │ │ │ │ │ - * changebaselayer event will be triggered. │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * nullify references to prevent circular references and memory leaks │ │ │ │ │ */ │ │ │ │ │ - addOptions: function(newOptions, reinitialize) { │ │ │ │ │ - │ │ │ │ │ - if (this.options == null) { │ │ │ │ │ - this.options = {}; │ │ │ │ │ - } │ │ │ │ │ + destroy: function() { │ │ │ │ │ │ │ │ │ │ - if (newOptions) { │ │ │ │ │ - // make sure this.projection references a projection object │ │ │ │ │ - if (typeof newOptions.projection == "string") { │ │ │ │ │ - newOptions.projection = new OpenLayers.Projection(newOptions.projection); │ │ │ │ │ - } │ │ │ │ │ - if (newOptions.projection) { │ │ │ │ │ - // get maxResolution, units and maxExtent from projection defaults if │ │ │ │ │ - // they are not defined already │ │ │ │ │ - OpenLayers.Util.applyDefaults(newOptions, │ │ │ │ │ - OpenLayers.Projection.defaults[newOptions.projection.getCode()]); │ │ │ │ │ - } │ │ │ │ │ - // allow array for extents │ │ │ │ │ - if (newOptions.maxExtent && !(newOptions.maxExtent instanceof OpenLayers.Bounds)) { │ │ │ │ │ - newOptions.maxExtent = new OpenLayers.Bounds(newOptions.maxExtent); │ │ │ │ │ - } │ │ │ │ │ - if (newOptions.minExtent && !(newOptions.minExtent instanceof OpenLayers.Bounds)) { │ │ │ │ │ - newOptions.minExtent = new OpenLayers.Bounds(newOptions.minExtent); │ │ │ │ │ + //remove the popup from the map │ │ │ │ │ + if ((this.layer != null) && (this.layer.map != null)) { │ │ │ │ │ + if (this.popup != null) { │ │ │ │ │ + this.layer.map.removePopup(this.popup); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // update our copy for clone │ │ │ │ │ - OpenLayers.Util.extend(this.options, newOptions); │ │ │ │ │ - │ │ │ │ │ - // add new options to this │ │ │ │ │ - OpenLayers.Util.extend(this, newOptions); │ │ │ │ │ - │ │ │ │ │ - // get the units from the projection, if we have a projection │ │ │ │ │ - // and it it has units │ │ │ │ │ - if (this.projection && this.projection.getUnits()) { │ │ │ │ │ - this.units = this.projection.getUnits(); │ │ │ │ │ + // remove the marker from the layer │ │ │ │ │ + if (this.layer != null && this.marker != null) { │ │ │ │ │ + this.layer.removeMarker(this.marker); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // re-initialize resolutions if necessary, i.e. if any of the │ │ │ │ │ - // properties of the "properties" array defined below is set │ │ │ │ │ - // in the new options │ │ │ │ │ - if (this.map) { │ │ │ │ │ - // store current resolution so we can try to restore it later │ │ │ │ │ - var resolution = this.map.getResolution(); │ │ │ │ │ - var properties = this.RESOLUTION_PROPERTIES.concat( │ │ │ │ │ - ["projection", "units", "minExtent", "maxExtent"] │ │ │ │ │ - ); │ │ │ │ │ - for (var o in newOptions) { │ │ │ │ │ - if (newOptions.hasOwnProperty(o) && │ │ │ │ │ - OpenLayers.Util.indexOf(properties, o) >= 0) { │ │ │ │ │ - │ │ │ │ │ - this.initResolutions(); │ │ │ │ │ - if (reinitialize && this.map.baseLayer === this) { │ │ │ │ │ - // update map position, and restore previous resolution │ │ │ │ │ - this.map.setCenter(this.map.getCenter(), │ │ │ │ │ - this.map.getZoomForResolution(resolution), │ │ │ │ │ - false, true │ │ │ │ │ - ); │ │ │ │ │ - // trigger a changebaselayer event to make sure that │ │ │ │ │ - // all controls (especially │ │ │ │ │ - // OpenLayers.Control.PanZoomBar) get notified of the │ │ │ │ │ - // new options │ │ │ │ │ - this.map.events.triggerEvent("changebaselayer", { │ │ │ │ │ - layer: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + this.layer = null; │ │ │ │ │ + this.id = null; │ │ │ │ │ + this.lonlat = null; │ │ │ │ │ + this.data = null; │ │ │ │ │ + if (this.marker != null) { │ │ │ │ │ + this.destroyMarker(this.marker); │ │ │ │ │ + this.marker = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.popup != null) { │ │ │ │ │ + this.destroyPopup(this.popup); │ │ │ │ │ + this.popup = null; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: onMapResize │ │ │ │ │ - * This function can be implemented by subclasses │ │ │ │ │ - */ │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - //this function can be implemented by subclasses │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: redraw │ │ │ │ │ - * Redraws the layer. Returns true if the layer was redrawn, false if not. │ │ │ │ │ - * │ │ │ │ │ + * Method: onScreen │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The layer was redrawn. │ │ │ │ │ + * {Boolean} Whether or not the feature is currently visible on screen │ │ │ │ │ + * (based on its 'lonlat' property) │ │ │ │ │ */ │ │ │ │ │ - redraw: function() { │ │ │ │ │ - var redrawn = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - │ │ │ │ │ - // min/max Range may have changed │ │ │ │ │ - this.inRange = this.calculateInRange(); │ │ │ │ │ - │ │ │ │ │ - // map's center might not yet be set │ │ │ │ │ - var extent = this.getExtent(); │ │ │ │ │ + onScreen: function() { │ │ │ │ │ │ │ │ │ │ - if (extent && this.inRange && this.visibility) { │ │ │ │ │ - var zoomChanged = true; │ │ │ │ │ - this.moveTo(extent, zoomChanged, false); │ │ │ │ │ - this.events.triggerEvent("moveend", { │ │ │ │ │ - "zoomChanged": zoomChanged │ │ │ │ │ - }); │ │ │ │ │ - redrawn = true; │ │ │ │ │ - } │ │ │ │ │ + var onScreen = false; │ │ │ │ │ + if ((this.layer != null) && (this.layer.map != null)) { │ │ │ │ │ + var screenBounds = this.layer.map.getExtent(); │ │ │ │ │ + onScreen = screenBounds.containsLonLat(this.lonlat); │ │ │ │ │ } │ │ │ │ │ - return redrawn; │ │ │ │ │ + return onScreen; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ + * Method: createMarker │ │ │ │ │ + * Based on the data associated with the Feature, create and return a marker object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} A Marker Object created from the 'lonlat' and 'icon' properties │ │ │ │ │ + * set in this.data. If no 'lonlat' is set, returns null. If no │ │ │ │ │ + * 'icon' is set, OpenLayers.Marker() will load the default image. │ │ │ │ │ + * │ │ │ │ │ + * Note - this.marker is set to return value │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {} │ │ │ │ │ - * zoomChanged - {Boolean} Tells when zoom has changed, as layers have to │ │ │ │ │ - * do some init work in that case. │ │ │ │ │ - * dragging - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - var display = this.visibility; │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - display = display && this.inRange; │ │ │ │ │ + createMarker: function() { │ │ │ │ │ + │ │ │ │ │ + if (this.lonlat != null) { │ │ │ │ │ + this.marker = new OpenLayers.Marker(this.lonlat, this.data.icon); │ │ │ │ │ } │ │ │ │ │ - this.display(display); │ │ │ │ │ + return this.marker; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveByPx │ │ │ │ │ - * Move the layer based on pixel vector. To be implemented by subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * dx - {Number} The x coord of the displacement vector. │ │ │ │ │ - * dy - {Number} The y coord of the displacement vector. │ │ │ │ │ + * Method: destroyMarker │ │ │ │ │ + * Destroys marker. │ │ │ │ │ + * If user overrides the createMarker() function, s/he should be able │ │ │ │ │ + * to also specify an alternative function for destroying it │ │ │ │ │ */ │ │ │ │ │ - moveByPx: function(dx, dy) {}, │ │ │ │ │ + destroyMarker: function() { │ │ │ │ │ + this.marker.destroy(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the layer. This is done through an accessor │ │ │ │ │ - * so that subclasses can override this and take special action once │ │ │ │ │ - * they have their map variable set. │ │ │ │ │ + * Method: createPopup │ │ │ │ │ + * Creates a popup object created from the 'lonlat', 'popupSize', │ │ │ │ │ + * and 'popupContentHTML' properties set in this.data. It uses │ │ │ │ │ + * this.marker.icon as default anchor. │ │ │ │ │ + * │ │ │ │ │ + * If no 'lonlat' is set, returns null. │ │ │ │ │ + * If no this.marker has been created, no anchor is sent. │ │ │ │ │ + * │ │ │ │ │ + * Note - the returned popup object is 'owned' by the feature, so you │ │ │ │ │ + * cannot use the popup's destroy method to discard the popup. │ │ │ │ │ + * Instead, you must use the feature's destroyPopup │ │ │ │ │ * │ │ │ │ │ - * Here we take care to bring over any of the necessary default │ │ │ │ │ - * properties from the map. │ │ │ │ │ + * Note - this.popup is set to return value │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * closeBox - {Boolean} create popup with closebox or not │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} Returns the created popup, which is also set │ │ │ │ │ + * as 'popup' property of this feature. Will be of whatever type │ │ │ │ │ + * specified by this feature's 'popupClass' property, but must be │ │ │ │ │ + * of type . │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {} │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - if (this.map == null) { │ │ │ │ │ - │ │ │ │ │ - this.map = map; │ │ │ │ │ - │ │ │ │ │ - // grab some essential layer data from the map if it hasn't already │ │ │ │ │ - // been set │ │ │ │ │ - this.maxExtent = this.maxExtent || this.map.maxExtent; │ │ │ │ │ - this.minExtent = this.minExtent || this.map.minExtent; │ │ │ │ │ + createPopup: function(closeBox) { │ │ │ │ │ │ │ │ │ │ - this.projection = this.projection || this.map.projection; │ │ │ │ │ - if (typeof this.projection == "string") { │ │ │ │ │ - this.projection = new OpenLayers.Projection(this.projection); │ │ │ │ │ + if (this.lonlat != null) { │ │ │ │ │ + if (!this.popup) { │ │ │ │ │ + var anchor = (this.marker) ? this.marker.icon : null; │ │ │ │ │ + var popupClass = this.popupClass ? │ │ │ │ │ + this.popupClass : OpenLayers.Popup.Anchored; │ │ │ │ │ + this.popup = new popupClass(this.id + "_popup", │ │ │ │ │ + this.lonlat, │ │ │ │ │ + this.data.popupSize, │ │ │ │ │ + this.data.popupContentHTML, │ │ │ │ │ + anchor, │ │ │ │ │ + closeBox); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // Check the projection to see if we can get units -- if not, refer │ │ │ │ │ - // to properties. │ │ │ │ │ - this.units = this.projection.getUnits() || │ │ │ │ │ - this.units || this.map.units; │ │ │ │ │ - │ │ │ │ │ - this.initResolutions(); │ │ │ │ │ - │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - this.inRange = this.calculateInRange(); │ │ │ │ │ - var show = ((this.visibility) && (this.inRange)); │ │ │ │ │ - this.div.style.display = show ? "" : "none"; │ │ │ │ │ + if (this.data.overflow != null) { │ │ │ │ │ + this.popup.contentDiv.style.overflow = this.data.overflow; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // deal with gutters │ │ │ │ │ - this.setTileSize(); │ │ │ │ │ + this.popup.feature = this; │ │ │ │ │ } │ │ │ │ │ + return this.popup; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: afterAdd │ │ │ │ │ - * Called at the end of the map.addLayer sequence. At this point, the map │ │ │ │ │ - * will have a base layer. To be overridden by subclasses. │ │ │ │ │ - */ │ │ │ │ │ - afterAdd: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: removeMap │ │ │ │ │ - * Just as setMap() allows each layer the possibility to take a │ │ │ │ │ - * personalized action on being added to the map, removeMap() allows │ │ │ │ │ - * each layer to take a personalized action on being removed from it. │ │ │ │ │ - * For now, this will be mostly unused, except for the EventPane layer, │ │ │ │ │ - * which needs this hook so that it can remove the special invisible │ │ │ │ │ - * pane. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {} │ │ │ │ │ + * Method: destroyPopup │ │ │ │ │ + * Destroys the popup created via createPopup. │ │ │ │ │ + * │ │ │ │ │ + * As with the marker, if user overrides the createPopup() function, s/he │ │ │ │ │ + * should also be able to override the destruction │ │ │ │ │ */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - //to be overridden by subclasses │ │ │ │ │ + destroyPopup: function() { │ │ │ │ │ + if (this.popup) { │ │ │ │ │ + this.popup.feature = null; │ │ │ │ │ + this.popup.destroy(); │ │ │ │ │ + this.popup = null; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getImageSize │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {} optional tile bounds, can be used │ │ │ │ │ - * by subclasses that have to deal with different tile sizes at the │ │ │ │ │ - * layer extent edges (e.g. Zoomify) │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} The size that the image should be, taking into │ │ │ │ │ - * account gutters. │ │ │ │ │ + CLASS_NAME: "OpenLayers.Feature" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Feature/Vector.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +// TRASH THIS │ │ │ │ │ +OpenLayers.State = { │ │ │ │ │ + /** states */ │ │ │ │ │ + UNKNOWN: 'Unknown', │ │ │ │ │ + INSERT: 'Insert', │ │ │ │ │ + UPDATE: 'Update', │ │ │ │ │ + DELETE: 'Delete' │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Feature.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Feature.Vector │ │ │ │ │ + * Vector features use the OpenLayers.Geometry classes as geometry description. │ │ │ │ │ + * They have an 'attributes' property, which is the data object, and a 'style' │ │ │ │ │ + * property, the default values of which are defined in the │ │ │ │ │ + * objects. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Feature.Vector = OpenLayers.Class(OpenLayers.Feature, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: fid │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - getImageSize: function(bounds) { │ │ │ │ │ - return (this.imageSize || this.tileSize); │ │ │ │ │ - }, │ │ │ │ │ + fid: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setTileSize │ │ │ │ │ - * Set the tile size based on the map size. This also sets layer.imageSize │ │ │ │ │ - * or use by Tile.Image. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * size - {} │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: geometry │ │ │ │ │ + * {} │ │ │ │ │ */ │ │ │ │ │ - setTileSize: function(size) { │ │ │ │ │ - var tileSize = (size) ? size : │ │ │ │ │ - ((this.tileSize) ? this.tileSize : │ │ │ │ │ - this.map.getTileSize()); │ │ │ │ │ - this.tileSize = tileSize; │ │ │ │ │ - if (this.gutter) { │ │ │ │ │ - // layers with gutters need non-null tile sizes │ │ │ │ │ - //if(tileSize == null) { │ │ │ │ │ - // OpenLayers.console.error("Error in layer.setMap() for " + │ │ │ │ │ - // this.name + ": layers with " + │ │ │ │ │ - // "gutters need non-null tile sizes"); │ │ │ │ │ - //} │ │ │ │ │ - this.imageSize = new OpenLayers.Size(tileSize.w + (2 * this.gutter), │ │ │ │ │ - tileSize.h + (2 * this.gutter)); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + geometry: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: attributes │ │ │ │ │ + * {Object} This object holds arbitrary, serializable properties that │ │ │ │ │ + * describe the feature. │ │ │ │ │ + */ │ │ │ │ │ + attributes: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getVisibility │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The layer should be displayed (if in range). │ │ │ │ │ + * Property: bounds │ │ │ │ │ + * {} The box bounding that feature's geometry, that │ │ │ │ │ + * property can be set by an object when │ │ │ │ │ + * deserializing the feature, so in most cases it represents an │ │ │ │ │ + * information set by the server. │ │ │ │ │ */ │ │ │ │ │ - getVisibility: function() { │ │ │ │ │ - return this.visibility; │ │ │ │ │ - }, │ │ │ │ │ + bounds: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setVisibility │ │ │ │ │ - * Set the visibility flag for the layer and hide/show & redraw │ │ │ │ │ - * accordingly. Fire event unless otherwise specified │ │ │ │ │ - * │ │ │ │ │ - * Note that visibility is no longer simply whether or not the layer's │ │ │ │ │ - * style.display is set to "block". Now we store a 'visibility' state │ │ │ │ │ - * property on the layer class, this allows us to remember whether or │ │ │ │ │ - * not we *desire* for a layer to be visible. In the case where the │ │ │ │ │ - * map's resolution is out of the layer's range, this desire may be │ │ │ │ │ - * subverted. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * visibility - {Boolean} Whether or not to display the layer (if in range) │ │ │ │ │ + * Property: state │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - setVisibility: function(visibility) { │ │ │ │ │ - if (visibility != this.visibility) { │ │ │ │ │ - this.visibility = visibility; │ │ │ │ │ - this.display(visibility); │ │ │ │ │ - this.redraw(); │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "visibility" │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("visibilitychanged"); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + state: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: display │ │ │ │ │ - * Hide or show the Layer. This is designed to be used internally, and │ │ │ │ │ - * is not generally the way to enable or disable the layer. For that, │ │ │ │ │ - * use the setVisibility function instead.. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * display - {Boolean} │ │ │ │ │ + * APIProperty: style │ │ │ │ │ + * {Object} │ │ │ │ │ */ │ │ │ │ │ - display: function(display) { │ │ │ │ │ - if (display != (this.div.style.display != "none")) { │ │ │ │ │ - this.div.style.display = (display && this.calculateInRange()) ? "block" : "none"; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + style: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: calculateInRange │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The layer is displayable at the current map's current │ │ │ │ │ - * resolution. Note that if 'alwaysInRange' is true for the layer, │ │ │ │ │ - * this function will always return true. │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String} If this property is set it will be taken into account by │ │ │ │ │ + * {} when upadting or deleting the feature. │ │ │ │ │ */ │ │ │ │ │ - calculateInRange: function() { │ │ │ │ │ - var inRange = false; │ │ │ │ │ + url: null, │ │ │ │ │ │ │ │ │ │ - if (this.alwaysInRange) { │ │ │ │ │ - inRange = true; │ │ │ │ │ - } else { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - var resolution = this.map.getResolution(); │ │ │ │ │ - inRange = ((resolution >= this.minResolution) && │ │ │ │ │ - (resolution <= this.maxResolution)); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return inRange; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: renderIntent │ │ │ │ │ + * {String} rendering intent currently being used │ │ │ │ │ + */ │ │ │ │ │ + renderIntent: "default", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: modified │ │ │ │ │ + * {Object} An object with the originals of the geometry and attributes of │ │ │ │ │ + * the feature, if they were changed. Currently this property is only read │ │ │ │ │ + * by , and written by │ │ │ │ │ + * , which sets the geometry property. │ │ │ │ │ + * Applications can set the originals of modified attributes in the │ │ │ │ │ + * attributes property. Note that applications have to check if this │ │ │ │ │ + * object and the attributes property is already created before using it. │ │ │ │ │ + * After a change made with ModifyFeature, this object could look like │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * { │ │ │ │ │ + * geometry: >Object │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * When an application has made changes to feature attributes, it could │ │ │ │ │ + * have set the attributes to something like this: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * { │ │ │ │ │ + * attributes: { │ │ │ │ │ + * myAttribute: "original" │ │ │ │ │ + * } │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Note that only checks for truthy values in │ │ │ │ │ + * *modified.geometry* and the attribute names in *modified.attributes*, │ │ │ │ │ + * but it is recommended to set the original values (and not just true) as │ │ │ │ │ + * attribute value, so applications could use this information to undo │ │ │ │ │ + * changes. │ │ │ │ │ + */ │ │ │ │ │ + modified: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setIsBaseLayer │ │ │ │ │ + * Constructor: OpenLayers.Feature.Vector │ │ │ │ │ + * Create a vector feature. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * isBaseLayer - {Boolean} │ │ │ │ │ + * geometry - {} The geometry that this feature │ │ │ │ │ + * represents. │ │ │ │ │ + * attributes - {Object} An optional object that will be mapped to the │ │ │ │ │ + * property. │ │ │ │ │ + * style - {Object} An optional style object. │ │ │ │ │ */ │ │ │ │ │ - setIsBaseLayer: function(isBaseLayer) { │ │ │ │ │ - if (isBaseLayer != this.isBaseLayer) { │ │ │ │ │ - this.isBaseLayer = isBaseLayer; │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changebaselayer", { │ │ │ │ │ - layer: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ + initialize: function(geometry, attributes, style) { │ │ │ │ │ + OpenLayers.Feature.prototype.initialize.apply(this, │ │ │ │ │ + [null, null, attributes]); │ │ │ │ │ + this.lonlat = null; │ │ │ │ │ + this.geometry = geometry ? geometry : null; │ │ │ │ │ + this.state = null; │ │ │ │ │ + this.attributes = {}; │ │ │ │ │ + if (attributes) { │ │ │ │ │ + this.attributes = OpenLayers.Util.extend(this.attributes, │ │ │ │ │ + attributes); │ │ │ │ │ } │ │ │ │ │ + this.style = style ? style : null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /********************************************************/ │ │ │ │ │ - /* */ │ │ │ │ │ - /* Baselayer Functions */ │ │ │ │ │ - /* */ │ │ │ │ │ - /********************************************************/ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: initResolutions │ │ │ │ │ - * This method's responsibility is to set up the 'resolutions' array │ │ │ │ │ - * for the layer -- this array is what the layer will use to interface │ │ │ │ │ - * between the zoom levels of the map and the resolution display │ │ │ │ │ - * of the layer. │ │ │ │ │ - * │ │ │ │ │ - * The user has several options that determine how the array is set up. │ │ │ │ │ - * │ │ │ │ │ - * For a detailed explanation, see the following wiki from the │ │ │ │ │ - * openlayers.org homepage: │ │ │ │ │ - * http://trac.openlayers.org/wiki/SettingZoomLevels │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * nullify references to prevent circular references and memory leaks │ │ │ │ │ */ │ │ │ │ │ - initResolutions: function() { │ │ │ │ │ - │ │ │ │ │ - // ok we want resolutions, here's our strategy: │ │ │ │ │ - // │ │ │ │ │ - // 1. if resolutions are defined in the layer config, use them │ │ │ │ │ - // 2. else, if scales are defined in the layer config then derive │ │ │ │ │ - // resolutions from these scales │ │ │ │ │ - // 3. else, attempt to calculate resolutions from maxResolution, │ │ │ │ │ - // minResolution, numZoomLevels, maxZoomLevel set in the │ │ │ │ │ - // layer config │ │ │ │ │ - // 4. if we still don't have resolutions, and if resolutions │ │ │ │ │ - // are defined in the same, use them │ │ │ │ │ - // 5. else, if scales are defined in the map then derive │ │ │ │ │ - // resolutions from these scales │ │ │ │ │ - // 6. else, attempt to calculate resolutions from maxResolution, │ │ │ │ │ - // minResolution, numZoomLevels, maxZoomLevel set in the │ │ │ │ │ - // map │ │ │ │ │ - // 7. hope for the best! │ │ │ │ │ - │ │ │ │ │ - var i, len, p; │ │ │ │ │ - var props = {}, │ │ │ │ │ - alwaysInRange = true; │ │ │ │ │ - │ │ │ │ │ - // get resolution data from layer config │ │ │ │ │ - // (we also set alwaysInRange in the layer as appropriate) │ │ │ │ │ - for (i = 0, len = this.RESOLUTION_PROPERTIES.length; i < len; i++) { │ │ │ │ │ - p = this.RESOLUTION_PROPERTIES[i]; │ │ │ │ │ - props[p] = this.options[p]; │ │ │ │ │ - if (alwaysInRange && this.options[p]) { │ │ │ │ │ - alwaysInRange = false; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.options.alwaysInRange == null) { │ │ │ │ │ - this.alwaysInRange = alwaysInRange; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // if we don't have resolutions then attempt to derive them from scales │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - props.resolutions = this.resolutionsFromScales(props.scales); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // if we still don't have resolutions then attempt to calculate them │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - props.resolutions = this.calculateResolutions(props); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // if we couldn't calculate resolutions then we look at we have │ │ │ │ │ - // in the map │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - for (i = 0, len = this.RESOLUTION_PROPERTIES.length; i < len; i++) { │ │ │ │ │ - p = this.RESOLUTION_PROPERTIES[i]; │ │ │ │ │ - props[p] = this.options[p] != null ? │ │ │ │ │ - this.options[p] : this.map[p]; │ │ │ │ │ - } │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - props.resolutions = this.resolutionsFromScales(props.scales); │ │ │ │ │ - } │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - props.resolutions = this.calculateResolutions(props); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // ok, we new need to set properties in the instance │ │ │ │ │ - │ │ │ │ │ - // get maxResolution from the config if it's defined there │ │ │ │ │ - var maxResolution; │ │ │ │ │ - if (this.options.maxResolution && │ │ │ │ │ - this.options.maxResolution !== "auto") { │ │ │ │ │ - maxResolution = this.options.maxResolution; │ │ │ │ │ - } │ │ │ │ │ - if (this.options.minScale) { │ │ │ │ │ - maxResolution = OpenLayers.Util.getResolutionFromScale( │ │ │ │ │ - this.options.minScale, this.units); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // get minResolution from the config if it's defined there │ │ │ │ │ - var minResolution; │ │ │ │ │ - if (this.options.minResolution && │ │ │ │ │ - this.options.minResolution !== "auto") { │ │ │ │ │ - minResolution = this.options.minResolution; │ │ │ │ │ - } │ │ │ │ │ - if (this.options.maxScale) { │ │ │ │ │ - minResolution = OpenLayers.Util.getResolutionFromScale( │ │ │ │ │ - this.options.maxScale, this.units); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (props.resolutions) { │ │ │ │ │ - │ │ │ │ │ - //sort resolutions array descendingly │ │ │ │ │ - props.resolutions.sort(function(a, b) { │ │ │ │ │ - return (b - a); │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - // if we still don't have a maxResolution get it from the │ │ │ │ │ - // resolutions array │ │ │ │ │ - if (!maxResolution) { │ │ │ │ │ - maxResolution = props.resolutions[0]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // if we still don't have a minResolution get it from the │ │ │ │ │ - // resolutions array │ │ │ │ │ - if (!minResolution) { │ │ │ │ │ - var lastIdx = props.resolutions.length - 1; │ │ │ │ │ - minResolution = props.resolutions[lastIdx]; │ │ │ │ │ - } │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.layer) { │ │ │ │ │ + this.layer.removeFeatures(this); │ │ │ │ │ + this.layer = null; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - this.resolutions = props.resolutions; │ │ │ │ │ - if (this.resolutions) { │ │ │ │ │ - len = this.resolutions.length; │ │ │ │ │ - this.scales = new Array(len); │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - this.scales[i] = OpenLayers.Util.getScaleFromResolution( │ │ │ │ │ - this.resolutions[i], this.units); │ │ │ │ │ - } │ │ │ │ │ - this.numZoomLevels = len; │ │ │ │ │ - } │ │ │ │ │ - this.minResolution = minResolution; │ │ │ │ │ - if (minResolution) { │ │ │ │ │ - this.maxScale = OpenLayers.Util.getScaleFromResolution( │ │ │ │ │ - minResolution, this.units); │ │ │ │ │ - } │ │ │ │ │ - this.maxResolution = maxResolution; │ │ │ │ │ - if (maxResolution) { │ │ │ │ │ - this.minScale = OpenLayers.Util.getScaleFromResolution( │ │ │ │ │ - maxResolution, this.units); │ │ │ │ │ - } │ │ │ │ │ + this.geometry = null; │ │ │ │ │ + this.modified = null; │ │ │ │ │ + OpenLayers.Feature.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: resolutionsFromScales │ │ │ │ │ - * Derive resolutions from scales. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * scales - {Array(Number)} Scales │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this vector feature. Does not set any non-standard │ │ │ │ │ + * properties. │ │ │ │ │ * │ │ │ │ │ - * Returns │ │ │ │ │ - * {Array(Number)} Resolutions │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} An exact clone of this vector feature. │ │ │ │ │ */ │ │ │ │ │ - resolutionsFromScales: function(scales) { │ │ │ │ │ - if (scales == null) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var resolutions, i, len; │ │ │ │ │ - len = scales.length; │ │ │ │ │ - resolutions = new Array(len); │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - resolutions[i] = OpenLayers.Util.getResolutionFromScale( │ │ │ │ │ - scales[i], this.units); │ │ │ │ │ - } │ │ │ │ │ - return resolutions; │ │ │ │ │ + clone: function() { │ │ │ │ │ + return new OpenLayers.Feature.Vector( │ │ │ │ │ + this.geometry ? this.geometry.clone() : null, │ │ │ │ │ + this.attributes, │ │ │ │ │ + this.style); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: calculateResolutions │ │ │ │ │ - * Calculate resolutions based on the provided properties. │ │ │ │ │ + * Method: onScreen │ │ │ │ │ + * Determine whether the feature is within the map viewport. This method │ │ │ │ │ + * tests for an intersection between the geometry and the viewport │ │ │ │ │ + * bounds. If a more effecient but less precise geometry bounds │ │ │ │ │ + * intersection is desired, call the method with the boundsOnly │ │ │ │ │ + * parameter true. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * props - {Object} Properties │ │ │ │ │ - * │ │ │ │ │ + * boundsOnly - {Boolean} Only test whether a feature's bounds intersects │ │ │ │ │ + * the viewport bounds. Default is false. If false, the feature's │ │ │ │ │ + * geometry must intersect the viewport for onScreen to return true. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array({Number})} Array of resolutions. │ │ │ │ │ + * {Boolean} The feature is currently visible on screen (optionally │ │ │ │ │ + * based on its bounds if boundsOnly is true). │ │ │ │ │ */ │ │ │ │ │ - calculateResolutions: function(props) { │ │ │ │ │ - │ │ │ │ │ - var viewSize, wRes, hRes; │ │ │ │ │ - │ │ │ │ │ - // determine maxResolution │ │ │ │ │ - var maxResolution = props.maxResolution; │ │ │ │ │ - if (props.minScale != null) { │ │ │ │ │ - maxResolution = │ │ │ │ │ - OpenLayers.Util.getResolutionFromScale(props.minScale, │ │ │ │ │ - this.units); │ │ │ │ │ - } else if (maxResolution == "auto" && this.maxExtent != null) { │ │ │ │ │ - viewSize = this.map.getSize(); │ │ │ │ │ - wRes = this.maxExtent.getWidth() / viewSize.w; │ │ │ │ │ - hRes = this.maxExtent.getHeight() / viewSize.h; │ │ │ │ │ - maxResolution = Math.max(wRes, hRes); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // determine minResolution │ │ │ │ │ - var minResolution = props.minResolution; │ │ │ │ │ - if (props.maxScale != null) { │ │ │ │ │ - minResolution = │ │ │ │ │ - OpenLayers.Util.getResolutionFromScale(props.maxScale, │ │ │ │ │ - this.units); │ │ │ │ │ - } else if (props.minResolution == "auto" && this.minExtent != null) { │ │ │ │ │ - viewSize = this.map.getSize(); │ │ │ │ │ - wRes = this.minExtent.getWidth() / viewSize.w; │ │ │ │ │ - hRes = this.minExtent.getHeight() / viewSize.h; │ │ │ │ │ - minResolution = Math.max(wRes, hRes); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (typeof maxResolution !== "number" && │ │ │ │ │ - typeof minResolution !== "number" && │ │ │ │ │ - this.maxExtent != null) { │ │ │ │ │ - // maxResolution for default grid sets assumes that at zoom │ │ │ │ │ - // level zero, the whole world fits on one tile. │ │ │ │ │ - var tileSize = this.map.getTileSize(); │ │ │ │ │ - maxResolution = Math.max( │ │ │ │ │ - this.maxExtent.getWidth() / tileSize.w, │ │ │ │ │ - this.maxExtent.getHeight() / tileSize.h │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // determine numZoomLevels │ │ │ │ │ - var maxZoomLevel = props.maxZoomLevel; │ │ │ │ │ - var numZoomLevels = props.numZoomLevels; │ │ │ │ │ - if (typeof minResolution === "number" && │ │ │ │ │ - typeof maxResolution === "number" && numZoomLevels === undefined) { │ │ │ │ │ - var ratio = maxResolution / minResolution; │ │ │ │ │ - numZoomLevels = Math.floor(Math.log(ratio) / Math.log(2)) + 1; │ │ │ │ │ - } else if (numZoomLevels === undefined && maxZoomLevel != null) { │ │ │ │ │ - numZoomLevels = maxZoomLevel + 1; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // are we able to calculate resolutions? │ │ │ │ │ - if (typeof numZoomLevels !== "number" || numZoomLevels <= 0 || │ │ │ │ │ - (typeof maxResolution !== "number" && │ │ │ │ │ - typeof minResolution !== "number")) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // now we have numZoomLevels and at least one of maxResolution │ │ │ │ │ - // or minResolution, we can populate the resolutions array │ │ │ │ │ - │ │ │ │ │ - var resolutions = new Array(numZoomLevels); │ │ │ │ │ - var base = 2; │ │ │ │ │ - if (typeof minResolution == "number" && │ │ │ │ │ - typeof maxResolution == "number") { │ │ │ │ │ - // if maxResolution and minResolution are set, we calculate │ │ │ │ │ - // the base for exponential scaling that starts at │ │ │ │ │ - // maxResolution and ends at minResolution in numZoomLevels │ │ │ │ │ - // steps. │ │ │ │ │ - base = Math.pow( │ │ │ │ │ - (maxResolution / minResolution), │ │ │ │ │ - (1 / (numZoomLevels - 1)) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var i; │ │ │ │ │ - if (typeof maxResolution === "number") { │ │ │ │ │ - for (i = 0; i < numZoomLevels; i++) { │ │ │ │ │ - resolutions[i] = maxResolution / Math.pow(base, i); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - for (i = 0; i < numZoomLevels; i++) { │ │ │ │ │ - resolutions[numZoomLevels - 1 - i] = │ │ │ │ │ - minResolution * Math.pow(base, i); │ │ │ │ │ + onScreen: function(boundsOnly) { │ │ │ │ │ + var onScreen = false; │ │ │ │ │ + if (this.layer && this.layer.map) { │ │ │ │ │ + var screenBounds = this.layer.map.getExtent(); │ │ │ │ │ + if (boundsOnly) { │ │ │ │ │ + var featureBounds = this.geometry.getBounds(); │ │ │ │ │ + onScreen = screenBounds.intersectsBounds(featureBounds); │ │ │ │ │ + } else { │ │ │ │ │ + var screenPoly = screenBounds.toGeometry(); │ │ │ │ │ + onScreen = screenPoly.intersects(this.geometry); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return resolutions; │ │ │ │ │ + return onScreen; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getResolution │ │ │ │ │ + * Method: getVisibility │ │ │ │ │ + * Determine whether the feature is displayed or not. It may not displayed │ │ │ │ │ + * because: │ │ │ │ │ + * - its style display property is set to 'none', │ │ │ │ │ + * - it doesn't belong to any layer, │ │ │ │ │ + * - the styleMap creates a symbolizer with display property set to 'none' │ │ │ │ │ + * for it, │ │ │ │ │ + * - the layer which it belongs to is not visible. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Float} The currently selected resolution of the map, taken from the │ │ │ │ │ - * resolutions array, indexed by current zoom level. │ │ │ │ │ + * {Boolean} The feature is currently displayed. │ │ │ │ │ */ │ │ │ │ │ - getResolution: function() { │ │ │ │ │ - var zoom = this.map.getZoom(); │ │ │ │ │ - return this.getResolutionForZoom(zoom); │ │ │ │ │ + getVisibility: function() { │ │ │ │ │ + return !(this.style && this.style.display == 'none' || │ │ │ │ │ + !this.layer || │ │ │ │ │ + this.layer && this.layer.styleMap && │ │ │ │ │ + this.layer.styleMap.createSymbolizer(this, this.renderIntent).display == 'none' || │ │ │ │ │ + this.layer && !this.layer.getVisibility()); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getExtent │ │ │ │ │ + /** │ │ │ │ │ + * Method: createMarker │ │ │ │ │ + * HACK - we need to decide if all vector features should be able to │ │ │ │ │ + * create markers │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {} A Bounds object which represents the lon/lat │ │ │ │ │ - * bounds of the current viewPort. │ │ │ │ │ + * {} For now just returns null │ │ │ │ │ */ │ │ │ │ │ - getExtent: function() { │ │ │ │ │ - // just use stock map calculateBounds function -- passing no arguments │ │ │ │ │ - // means it will user map's current center & resolution │ │ │ │ │ - // │ │ │ │ │ - return this.map.calculateBounds(); │ │ │ │ │ + createMarker: function() { │ │ │ │ │ + return null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getZoomForExtent │ │ │ │ │ + * Method: destroyMarker │ │ │ │ │ + * HACK - we need to decide if all vector features should be able to │ │ │ │ │ + * delete markers │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * extent - {} │ │ │ │ │ - * closest - {Boolean} Find the zoom level that most closely fits the │ │ │ │ │ - * specified bounds. Note that this may result in a zoom that does │ │ │ │ │ - * not exactly contain the entire extent. │ │ │ │ │ - * Default is false. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} The index of the zoomLevel (entry in the resolutions array) │ │ │ │ │ - * for the passed-in extent. We do this by calculating the ideal │ │ │ │ │ - * resolution for the given extent (based on the map size) and then │ │ │ │ │ - * calling getZoomForResolution(), passing along the 'closest' │ │ │ │ │ - * parameter. │ │ │ │ │ + * If user overrides the createMarker() function, s/he should be able │ │ │ │ │ + * to also specify an alternative function for destroying it │ │ │ │ │ */ │ │ │ │ │ - getZoomForExtent: function(extent, closest) { │ │ │ │ │ - var viewSize = this.map.getSize(); │ │ │ │ │ - var idealResolution = Math.max(extent.getWidth() / viewSize.w, │ │ │ │ │ - extent.getHeight() / viewSize.h); │ │ │ │ │ - │ │ │ │ │ - return this.getZoomForResolution(idealResolution, closest); │ │ │ │ │ + destroyMarker: function() { │ │ │ │ │ + // pass │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getDataExtent │ │ │ │ │ - * Calculates the max extent which includes all of the data for the layer. │ │ │ │ │ - * This function is to be implemented by subclasses. │ │ │ │ │ + /** │ │ │ │ │ + * Method: createPopup │ │ │ │ │ + * HACK - we need to decide if all vector features should be able to │ │ │ │ │ + * create popups │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {} │ │ │ │ │ + * {} For now just returns null │ │ │ │ │ */ │ │ │ │ │ - getDataExtent: function() { │ │ │ │ │ - //to be implemented by subclasses │ │ │ │ │ + createPopup: function() { │ │ │ │ │ + return null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getResolutionForZoom │ │ │ │ │ + * Method: atPoint │ │ │ │ │ + * Determins whether the feature intersects with the specified location. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * zoom - {Float} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * lonlat - {|Object} OpenLayers.LonLat or an │ │ │ │ │ + * object with a 'lon' and 'lat' properties. │ │ │ │ │ + * toleranceLon - {float} Optional tolerance in Geometric Coords │ │ │ │ │ + * toleranceLat - {float} Optional tolerance in Geographic Coords │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Float} A suitable resolution for the specified zoom. │ │ │ │ │ + * {Boolean} Whether or not the feature is at the specified location │ │ │ │ │ */ │ │ │ │ │ - getResolutionForZoom: function(zoom) { │ │ │ │ │ - zoom = Math.max(0, Math.min(zoom, this.resolutions.length - 1)); │ │ │ │ │ - var resolution; │ │ │ │ │ - if (this.map.fractionalZoom) { │ │ │ │ │ - var low = Math.floor(zoom); │ │ │ │ │ - var high = Math.ceil(zoom); │ │ │ │ │ - resolution = this.resolutions[low] - │ │ │ │ │ - ((zoom - low) * (this.resolutions[low] - this.resolutions[high])); │ │ │ │ │ - } else { │ │ │ │ │ - resolution = this.resolutions[Math.round(zoom)]; │ │ │ │ │ + atPoint: function(lonlat, toleranceLon, toleranceLat) { │ │ │ │ │ + var atPoint = false; │ │ │ │ │ + if (this.geometry) { │ │ │ │ │ + atPoint = this.geometry.atPoint(lonlat, toleranceLon, │ │ │ │ │ + toleranceLat); │ │ │ │ │ } │ │ │ │ │ - return resolution; │ │ │ │ │ + return atPoint; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getZoomForResolution │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resolution - {Float} │ │ │ │ │ - * closest - {Boolean} Find the zoom level that corresponds to the absolute │ │ │ │ │ - * closest resolution, which may result in a zoom whose corresponding │ │ │ │ │ - * resolution is actually smaller than we would have desired (if this │ │ │ │ │ - * is being called from a getZoomForExtent() call, then this means that │ │ │ │ │ - * the returned zoom index might not actually contain the entire │ │ │ │ │ - * extent specified... but it'll be close). │ │ │ │ │ - * Default is false. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} The index of the zoomLevel (entry in the resolutions array) │ │ │ │ │ - * that corresponds to the best fit resolution given the passed in │ │ │ │ │ - * value and the 'closest' specification. │ │ │ │ │ + * Method: destroyPopup │ │ │ │ │ + * HACK - we need to decide if all vector features should be able to │ │ │ │ │ + * delete popups │ │ │ │ │ */ │ │ │ │ │ - getZoomForResolution: function(resolution, closest) { │ │ │ │ │ - var zoom, i, len; │ │ │ │ │ - if (this.map.fractionalZoom) { │ │ │ │ │ - var lowZoom = 0; │ │ │ │ │ - var highZoom = this.resolutions.length - 1; │ │ │ │ │ - var highRes = this.resolutions[lowZoom]; │ │ │ │ │ - var lowRes = this.resolutions[highZoom]; │ │ │ │ │ - var res; │ │ │ │ │ - for (i = 0, len = this.resolutions.length; i < len; ++i) { │ │ │ │ │ - res = this.resolutions[i]; │ │ │ │ │ - if (res >= resolution) { │ │ │ │ │ - highRes = res; │ │ │ │ │ - lowZoom = i; │ │ │ │ │ - } │ │ │ │ │ - if (res <= resolution) { │ │ │ │ │ - lowRes = res; │ │ │ │ │ - highZoom = i; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var dRes = highRes - lowRes; │ │ │ │ │ - if (dRes > 0) { │ │ │ │ │ - zoom = lowZoom + ((highRes - resolution) / dRes); │ │ │ │ │ - } else { │ │ │ │ │ - zoom = lowZoom; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - var diff; │ │ │ │ │ - var minDiff = Number.POSITIVE_INFINITY; │ │ │ │ │ - for (i = 0, len = this.resolutions.length; i < len; i++) { │ │ │ │ │ - if (closest) { │ │ │ │ │ - diff = Math.abs(this.resolutions[i] - resolution); │ │ │ │ │ - if (diff > minDiff) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - minDiff = diff; │ │ │ │ │ - } else { │ │ │ │ │ - if (this.resolutions[i] < resolution) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - zoom = Math.max(0, i - 1); │ │ │ │ │ - } │ │ │ │ │ - return zoom; │ │ │ │ │ + destroyPopup: function() { │ │ │ │ │ + // pass │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getLonLatFromViewPortPx │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * viewPortPx - {|Object} An OpenLayers.Pixel or │ │ │ │ │ - * an object with a 'x' │ │ │ │ │ - * and 'y' properties. │ │ │ │ │ + * Method: move │ │ │ │ │ + * Moves the feature and redraws it at its new location │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} An OpenLayers.LonLat which is the passed-in │ │ │ │ │ - * view port , translated into lon/lat by the layer. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * location - { or } the │ │ │ │ │ + * location to which to move the feature. │ │ │ │ │ */ │ │ │ │ │ - getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ - var lonlat = null; │ │ │ │ │ - var map = this.map; │ │ │ │ │ - if (viewPortPx != null && map.minPx) { │ │ │ │ │ - var res = map.getResolution(); │ │ │ │ │ - var maxExtent = map.getMaxExtent({ │ │ │ │ │ - restricted: true │ │ │ │ │ - }); │ │ │ │ │ - var lon = (viewPortPx.x - map.minPx.x) * res + maxExtent.left; │ │ │ │ │ - var lat = (map.minPx.y - viewPortPx.y) * res + maxExtent.top; │ │ │ │ │ - lonlat = new OpenLayers.LonLat(lon, lat); │ │ │ │ │ + move: function(location) { │ │ │ │ │ │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - lonlat = lonlat.wrapDateLine(this.maxExtent); │ │ │ │ │ - } │ │ │ │ │ + if (!this.layer || !this.geometry.move) { │ │ │ │ │ + //do nothing if no layer or immoveable geometry │ │ │ │ │ + return undefined; │ │ │ │ │ } │ │ │ │ │ - return lonlat; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getViewPortPxFromLonLat │ │ │ │ │ - * Returns a pixel location given a map location. This method will return │ │ │ │ │ - * fractional pixel values. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * lonlat - {|Object} An OpenLayers.LonLat or │ │ │ │ │ - * an object with a 'lon' │ │ │ │ │ - * and 'lat' properties. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} An which is the passed-in │ │ │ │ │ - * lonlat translated into view port pixels. │ │ │ │ │ - */ │ │ │ │ │ - getViewPortPxFromLonLat: function(lonlat, resolution) { │ │ │ │ │ - var px = null; │ │ │ │ │ - if (lonlat != null) { │ │ │ │ │ - resolution = resolution || this.map.getResolution(); │ │ │ │ │ - var extent = this.map.calculateBounds(null, resolution); │ │ │ │ │ - px = new OpenLayers.Pixel( │ │ │ │ │ - (1 / resolution * (lonlat.lon - extent.left)), │ │ │ │ │ - (1 / resolution * (extent.top - lonlat.lat)) │ │ │ │ │ - ); │ │ │ │ │ + var pixel; │ │ │ │ │ + if (location.CLASS_NAME == "OpenLayers.LonLat") { │ │ │ │ │ + pixel = this.layer.getViewPortPxFromLonLat(location); │ │ │ │ │ + } else { │ │ │ │ │ + pixel = location; │ │ │ │ │ } │ │ │ │ │ - return px; │ │ │ │ │ + │ │ │ │ │ + var lastPixel = this.layer.getViewPortPxFromLonLat(this.geometry.getBounds().getCenterLonLat()); │ │ │ │ │ + var res = this.layer.map.getResolution(); │ │ │ │ │ + this.geometry.move(res * (pixel.x - lastPixel.x), │ │ │ │ │ + res * (lastPixel.y - pixel.y)); │ │ │ │ │ + this.layer.drawFeature(this); │ │ │ │ │ + return lastPixel; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setOpacity │ │ │ │ │ - * Sets the opacity for the entire layer (all images) │ │ │ │ │ - * │ │ │ │ │ + * Method: toState │ │ │ │ │ + * Sets the new state │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * opacity - {Float} │ │ │ │ │ - */ │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - if (opacity != this.opacity) { │ │ │ │ │ - this.opacity = opacity; │ │ │ │ │ - var childNodes = this.div.childNodes; │ │ │ │ │ - for (var i = 0, len = childNodes.length; i < len; ++i) { │ │ │ │ │ - var element = childNodes[i].firstChild || childNodes[i]; │ │ │ │ │ - var lastChild = childNodes[i].lastChild; │ │ │ │ │ - //TODO de-uglify this │ │ │ │ │ - if (lastChild && lastChild.nodeName.toLowerCase() === "iframe") { │ │ │ │ │ - element = lastChild.parentNode; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.modifyDOMElement(element, null, null, null, │ │ │ │ │ - null, null, null, opacity); │ │ │ │ │ - } │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "opacity" │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getZIndex │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} the z-index of this layer │ │ │ │ │ - */ │ │ │ │ │ - getZIndex: function() { │ │ │ │ │ - return this.div.style.zIndex; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setZIndex │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * zIndex - {Integer} │ │ │ │ │ - */ │ │ │ │ │ - setZIndex: function(zIndex) { │ │ │ │ │ - this.div.style.zIndex = zIndex; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: adjustBounds │ │ │ │ │ - * This function will take a bounds, and if wrapDateLine option is set │ │ │ │ │ - * on the layer, it will return a bounds which is wrapped around the │ │ │ │ │ - * world. We do not wrap for bounds which *cross* the │ │ │ │ │ - * maxExtent.left/right, only bounds which are entirely to the left │ │ │ │ │ - * or entirely to the right. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {} │ │ │ │ │ - */ │ │ │ │ │ - adjustBounds: function(bounds) { │ │ │ │ │ - │ │ │ │ │ - if (this.gutter) { │ │ │ │ │ - // Adjust the extent of a bounds in map units by the │ │ │ │ │ - // layer's gutter in pixels. │ │ │ │ │ - var mapGutter = this.gutter * this.map.getResolution(); │ │ │ │ │ - bounds = new OpenLayers.Bounds(bounds.left - mapGutter, │ │ │ │ │ - bounds.bottom - mapGutter, │ │ │ │ │ - bounds.right + mapGutter, │ │ │ │ │ - bounds.top + mapGutter); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - // wrap around the date line, within the limits of rounding error │ │ │ │ │ - var wrappingOptions = { │ │ │ │ │ - 'rightTolerance': this.getResolution(), │ │ │ │ │ - 'leftTolerance': this.getResolution() │ │ │ │ │ - }; │ │ │ │ │ - bounds = bounds.wrapDateLine(this.maxExtent, wrappingOptions); │ │ │ │ │ - │ │ │ │ │ - } │ │ │ │ │ - return bounds; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -// XMLHttpRequest.js Copyright (C) 2010 Sergey Ilinsky (http://www.ilinsky.com) │ │ │ │ │ -// │ │ │ │ │ -// Licensed under the Apache License, Version 2.0 (the "License"); │ │ │ │ │ -// you may not use this file except in compliance with the License. │ │ │ │ │ -// You may obtain a copy of the License at │ │ │ │ │ -// │ │ │ │ │ -// http://www.apache.org/licenses/LICENSE-2.0 │ │ │ │ │ -// │ │ │ │ │ -// Unless required by applicable law or agreed to in writing, software │ │ │ │ │ -// distributed under the License is distributed on an "AS IS" BASIS, │ │ │ │ │ -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. │ │ │ │ │ -// See the License for the specific language governing permissions and │ │ │ │ │ -// limitations under the License. │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Request.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -(function() { │ │ │ │ │ - │ │ │ │ │ - // Save reference to earlier defined object implementation (if any) │ │ │ │ │ - var oXMLHttpRequest = window.XMLHttpRequest; │ │ │ │ │ - │ │ │ │ │ - // Define on browser type │ │ │ │ │ - var bGecko = !!window.controllers, │ │ │ │ │ - bIE = window.document.all && !window.opera, │ │ │ │ │ - bIE7 = bIE && window.navigator.userAgent.match(/MSIE 7.0/); │ │ │ │ │ - │ │ │ │ │ - // Enables "XMLHttpRequest()" call next to "new XMLHttpReques()" │ │ │ │ │ - function fXMLHttpRequest() { │ │ │ │ │ - this._object = oXMLHttpRequest && !bIE7 ? new oXMLHttpRequest : new window.ActiveXObject("Microsoft.XMLHTTP"); │ │ │ │ │ - this._listeners = []; │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // Constructor │ │ │ │ │ - function cXMLHttpRequest() { │ │ │ │ │ - return new fXMLHttpRequest; │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype = fXMLHttpRequest.prototype; │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: Firefox with Firebug installed would break pages if not executed │ │ │ │ │ - if (bGecko && oXMLHttpRequest.wrapped) │ │ │ │ │ - cXMLHttpRequest.wrapped = oXMLHttpRequest.wrapped; │ │ │ │ │ - │ │ │ │ │ - // Constants │ │ │ │ │ - cXMLHttpRequest.UNSENT = 0; │ │ │ │ │ - cXMLHttpRequest.OPENED = 1; │ │ │ │ │ - cXMLHttpRequest.HEADERS_RECEIVED = 2; │ │ │ │ │ - cXMLHttpRequest.LOADING = 3; │ │ │ │ │ - cXMLHttpRequest.DONE = 4; │ │ │ │ │ - │ │ │ │ │ - // Public Properties │ │ │ │ │ - cXMLHttpRequest.prototype.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ - cXMLHttpRequest.prototype.responseText = ''; │ │ │ │ │ - cXMLHttpRequest.prototype.responseXML = null; │ │ │ │ │ - cXMLHttpRequest.prototype.status = 0; │ │ │ │ │ - cXMLHttpRequest.prototype.statusText = ''; │ │ │ │ │ - │ │ │ │ │ - // Priority proposal │ │ │ │ │ - cXMLHttpRequest.prototype.priority = "NORMAL"; │ │ │ │ │ - │ │ │ │ │ - // Instance-level Events Handlers │ │ │ │ │ - cXMLHttpRequest.prototype.onreadystatechange = null; │ │ │ │ │ - │ │ │ │ │ - // Class-level Events Handlers │ │ │ │ │ - cXMLHttpRequest.onreadystatechange = null; │ │ │ │ │ - cXMLHttpRequest.onopen = null; │ │ │ │ │ - cXMLHttpRequest.onsend = null; │ │ │ │ │ - cXMLHttpRequest.onabort = null; │ │ │ │ │ - │ │ │ │ │ - // Public Methods │ │ │ │ │ - cXMLHttpRequest.prototype.open = function(sMethod, sUrl, bAsync, sUser, sPassword) { │ │ │ │ │ - // Delete headers, required when object is reused │ │ │ │ │ - delete this._headers; │ │ │ │ │ - │ │ │ │ │ - // When bAsync parameter value is omitted, use true as default │ │ │ │ │ - if (arguments.length < 3) │ │ │ │ │ - bAsync = true; │ │ │ │ │ - │ │ │ │ │ - // Save async parameter for fixing Gecko bug with missing readystatechange in synchronous requests │ │ │ │ │ - this._async = bAsync; │ │ │ │ │ - │ │ │ │ │ - // Set the onreadystatechange handler │ │ │ │ │ - var oRequest = this, │ │ │ │ │ - nState = this.readyState, │ │ │ │ │ - fOnUnload; │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: IE - memory leak on page unload (inter-page leak) │ │ │ │ │ - if (bIE && bAsync) { │ │ │ │ │ - fOnUnload = function() { │ │ │ │ │ - if (nState != cXMLHttpRequest.DONE) { │ │ │ │ │ - fCleanTransport(oRequest); │ │ │ │ │ - // Safe to abort here since onreadystatechange handler removed │ │ │ │ │ - oRequest.abort(); │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - window.attachEvent("onunload", fOnUnload); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // Add method sniffer │ │ │ │ │ - if (cXMLHttpRequest.onopen) │ │ │ │ │ - cXMLHttpRequest.onopen.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - if (arguments.length > 4) │ │ │ │ │ - this._object.open(sMethod, sUrl, bAsync, sUser, sPassword); │ │ │ │ │ - else │ │ │ │ │ - if (arguments.length > 3) │ │ │ │ │ - this._object.open(sMethod, sUrl, bAsync, sUser); │ │ │ │ │ - else │ │ │ │ │ - this._object.open(sMethod, sUrl, bAsync); │ │ │ │ │ - │ │ │ │ │ - this.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ - fReadyStateChange(this); │ │ │ │ │ - │ │ │ │ │ - this._object.onreadystatechange = function() { │ │ │ │ │ - if (bGecko && !bAsync) │ │ │ │ │ - return; │ │ │ │ │ - │ │ │ │ │ - // Synchronize state │ │ │ │ │ - oRequest.readyState = oRequest._object.readyState; │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - fSynchronizeValues(oRequest); │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: Firefox fires unnecessary DONE when aborting │ │ │ │ │ - if (oRequest._aborted) { │ │ │ │ │ - // Reset readyState to UNSENT │ │ │ │ │ - oRequest.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ - │ │ │ │ │ - // Return now │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (oRequest.readyState == cXMLHttpRequest.DONE) { │ │ │ │ │ - // Free up queue │ │ │ │ │ - delete oRequest._data; │ │ │ │ │ - /* if (bAsync) │ │ │ │ │ - fQueue_remove(oRequest);*/ │ │ │ │ │ - // │ │ │ │ │ - fCleanTransport(oRequest); │ │ │ │ │ - // Uncomment this block if you need a fix for IE cache │ │ │ │ │ - /* │ │ │ │ │ - // BUGFIX: IE - cache issue │ │ │ │ │ - if (!oRequest._object.getResponseHeader("Date")) { │ │ │ │ │ - // Save object to cache │ │ │ │ │ - oRequest._cached = oRequest._object; │ │ │ │ │ - │ │ │ │ │ - // Instantiate a new transport object │ │ │ │ │ - cXMLHttpRequest.call(oRequest); │ │ │ │ │ - │ │ │ │ │ - // Re-send request │ │ │ │ │ - if (sUser) { │ │ │ │ │ - if (sPassword) │ │ │ │ │ - oRequest._object.open(sMethod, sUrl, bAsync, sUser, sPassword); │ │ │ │ │ - else │ │ │ │ │ - oRequest._object.open(sMethod, sUrl, bAsync, sUser); │ │ │ │ │ - } │ │ │ │ │ - else │ │ │ │ │ - oRequest._object.open(sMethod, sUrl, bAsync); │ │ │ │ │ - oRequest._object.setRequestHeader("If-Modified-Since", oRequest._cached.getResponseHeader("Last-Modified") || new window.Date(0)); │ │ │ │ │ - // Copy headers set │ │ │ │ │ - if (oRequest._headers) │ │ │ │ │ - for (var sHeader in oRequest._headers) │ │ │ │ │ - if (typeof oRequest._headers[sHeader] == "string") // Some frameworks prototype objects with functions │ │ │ │ │ - oRequest._object.setRequestHeader(sHeader, oRequest._headers[sHeader]); │ │ │ │ │ - │ │ │ │ │ - oRequest._object.onreadystatechange = function() { │ │ │ │ │ - // Synchronize state │ │ │ │ │ - oRequest.readyState = oRequest._object.readyState; │ │ │ │ │ - │ │ │ │ │ - if (oRequest._aborted) { │ │ │ │ │ - // │ │ │ │ │ - oRequest.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ - │ │ │ │ │ - // Return │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (oRequest.readyState == cXMLHttpRequest.DONE) { │ │ │ │ │ - // Clean Object │ │ │ │ │ - fCleanTransport(oRequest); │ │ │ │ │ - │ │ │ │ │ - // get cached request │ │ │ │ │ - if (oRequest.status == 304) │ │ │ │ │ - oRequest._object = oRequest._cached; │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - delete oRequest._cached; │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - fSynchronizeValues(oRequest); │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - fReadyStateChange(oRequest); │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: IE - memory leak in interrupted │ │ │ │ │ - if (bIE && bAsync) │ │ │ │ │ - window.detachEvent("onunload", fOnUnload); │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - oRequest._object.send(null); │ │ │ │ │ - │ │ │ │ │ - // Return now - wait until re-sent request is finished │ │ │ │ │ - return; │ │ │ │ │ - }; │ │ │ │ │ - */ │ │ │ │ │ - // BUGFIX: IE - memory leak in interrupted │ │ │ │ │ - if (bIE && bAsync) │ │ │ │ │ - window.detachEvent("onunload", fOnUnload); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: Some browsers (Internet Explorer, Gecko) fire OPEN readystate twice │ │ │ │ │ - if (nState != oRequest.readyState) │ │ │ │ │ - fReadyStateChange(oRequest); │ │ │ │ │ - │ │ │ │ │ - nState = oRequest.readyState; │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - function fXMLHttpRequest_send(oRequest) { │ │ │ │ │ - oRequest._object.send(oRequest._data); │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: Gecko - missing readystatechange calls in synchronous requests │ │ │ │ │ - if (bGecko && !oRequest._async) { │ │ │ │ │ - oRequest.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ - │ │ │ │ │ - // Synchronize state │ │ │ │ │ - fSynchronizeValues(oRequest); │ │ │ │ │ - │ │ │ │ │ - // Simulate missing states │ │ │ │ │ - while (oRequest.readyState < cXMLHttpRequest.DONE) { │ │ │ │ │ - oRequest.readyState++; │ │ │ │ │ - fReadyStateChange(oRequest); │ │ │ │ │ - // Check if we are aborted │ │ │ │ │ - if (oRequest._aborted) │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.send = function(vData) { │ │ │ │ │ - // Add method sniffer │ │ │ │ │ - if (cXMLHttpRequest.onsend) │ │ │ │ │ - cXMLHttpRequest.onsend.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - if (!arguments.length) │ │ │ │ │ - vData = null; │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: Safari - fails sending documents created/modified dynamically, so an explicit serialization required │ │ │ │ │ - // BUGFIX: IE - rewrites any custom mime-type to "text/xml" in case an XMLNode is sent │ │ │ │ │ - // BUGFIX: Gecko - fails sending Element (this is up to the implementation either to standard) │ │ │ │ │ - if (vData && vData.nodeType) { │ │ │ │ │ - vData = window.XMLSerializer ? new window.XMLSerializer().serializeToString(vData) : vData.xml; │ │ │ │ │ - if (!this._headers["Content-Type"]) │ │ │ │ │ - this._object.setRequestHeader("Content-Type", "application/xml"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this._data = vData; │ │ │ │ │ - /* │ │ │ │ │ - // Add to queue │ │ │ │ │ - if (this._async) │ │ │ │ │ - fQueue_add(this); │ │ │ │ │ - else*/ │ │ │ │ │ - fXMLHttpRequest_send(this); │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.abort = function() { │ │ │ │ │ - // Add method sniffer │ │ │ │ │ - if (cXMLHttpRequest.onabort) │ │ │ │ │ - cXMLHttpRequest.onabort.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: Gecko - unnecessary DONE when aborting │ │ │ │ │ - if (this.readyState > cXMLHttpRequest.UNSENT) │ │ │ │ │ - this._aborted = true; │ │ │ │ │ - │ │ │ │ │ - this._object.abort(); │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: IE - memory leak │ │ │ │ │ - fCleanTransport(this); │ │ │ │ │ - │ │ │ │ │ - this.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ - │ │ │ │ │ - delete this._data; │ │ │ │ │ - /* if (this._async) │ │ │ │ │ - fQueue_remove(this);*/ │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.getAllResponseHeaders = function() { │ │ │ │ │ - return this._object.getAllResponseHeaders(); │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.getResponseHeader = function(sName) { │ │ │ │ │ - return this._object.getResponseHeader(sName); │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.setRequestHeader = function(sName, sValue) { │ │ │ │ │ - // BUGFIX: IE - cache issue │ │ │ │ │ - if (!this._headers) │ │ │ │ │ - this._headers = {}; │ │ │ │ │ - this._headers[sName] = sValue; │ │ │ │ │ - │ │ │ │ │ - return this._object.setRequestHeader(sName, sValue); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // EventTarget interface implementation │ │ │ │ │ - cXMLHttpRequest.prototype.addEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ - for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ - if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) │ │ │ │ │ - return; │ │ │ │ │ - // Add listener │ │ │ │ │ - this._listeners.push([sName, fHandler, bUseCapture]); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - cXMLHttpRequest.prototype.removeEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ - for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ - if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) │ │ │ │ │ - break; │ │ │ │ │ - // Remove listener │ │ │ │ │ - if (oListener) │ │ │ │ │ - this._listeners.splice(nIndex, 1); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - cXMLHttpRequest.prototype.dispatchEvent = function(oEvent) { │ │ │ │ │ - var oEventPseudo = { │ │ │ │ │ - 'type': oEvent.type, │ │ │ │ │ - 'target': this, │ │ │ │ │ - 'currentTarget': this, │ │ │ │ │ - 'eventPhase': 2, │ │ │ │ │ - 'bubbles': oEvent.bubbles, │ │ │ │ │ - 'cancelable': oEvent.cancelable, │ │ │ │ │ - 'timeStamp': oEvent.timeStamp, │ │ │ │ │ - 'stopPropagation': function() {}, // There is no flow │ │ │ │ │ - 'preventDefault': function() {}, // There is no default action │ │ │ │ │ - 'initEvent': function() {} // Original event object should be initialized │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // Execute onreadystatechange │ │ │ │ │ - if (oEventPseudo.type == "readystatechange" && this.onreadystatechange) │ │ │ │ │ - (this.onreadystatechange.handleEvent || this.onreadystatechange).apply(this, [oEventPseudo]); │ │ │ │ │ - │ │ │ │ │ - // Execute listeners │ │ │ │ │ - for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ - if (oListener[0] == oEventPseudo.type && !oListener[2]) │ │ │ │ │ - (oListener[1].handleEvent || oListener[1]).apply(this, [oEventPseudo]); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - cXMLHttpRequest.prototype.toString = function() { │ │ │ │ │ - return '[' + "object" + ' ' + "XMLHttpRequest" + ']'; │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - cXMLHttpRequest.toString = function() { │ │ │ │ │ - return '[' + "XMLHttpRequest" + ']'; │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // Helper function │ │ │ │ │ - function fReadyStateChange(oRequest) { │ │ │ │ │ - // Sniffing code │ │ │ │ │ - if (cXMLHttpRequest.onreadystatechange) │ │ │ │ │ - cXMLHttpRequest.onreadystatechange.apply(oRequest); │ │ │ │ │ - │ │ │ │ │ - // Fake event │ │ │ │ │ - oRequest.dispatchEvent({ │ │ │ │ │ - 'type': "readystatechange", │ │ │ │ │ - 'bubbles': false, │ │ │ │ │ - 'cancelable': false, │ │ │ │ │ - 'timeStamp': new Date + 0 │ │ │ │ │ - }); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - function fGetDocument(oRequest) { │ │ │ │ │ - var oDocument = oRequest.responseXML, │ │ │ │ │ - sResponse = oRequest.responseText; │ │ │ │ │ - // Try parsing responseText │ │ │ │ │ - if (bIE && sResponse && oDocument && !oDocument.documentElement && oRequest.getResponseHeader("Content-Type").match(/[^\/]+\/[^\+]+\+xml/)) { │ │ │ │ │ - oDocument = new window.ActiveXObject("Microsoft.XMLDOM"); │ │ │ │ │ - oDocument.async = false; │ │ │ │ │ - oDocument.validateOnParse = false; │ │ │ │ │ - oDocument.loadXML(sResponse); │ │ │ │ │ - } │ │ │ │ │ - // Check if there is no error in document │ │ │ │ │ - if (oDocument) │ │ │ │ │ - if ((bIE && oDocument.parseError != 0) || !oDocument.documentElement || (oDocument.documentElement && oDocument.documentElement.tagName == "parsererror")) │ │ │ │ │ - return null; │ │ │ │ │ - return oDocument; │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - function fSynchronizeValues(oRequest) { │ │ │ │ │ - try { │ │ │ │ │ - oRequest.responseText = oRequest._object.responseText; │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - try { │ │ │ │ │ - oRequest.responseXML = fGetDocument(oRequest._object); │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - try { │ │ │ │ │ - oRequest.status = oRequest._object.status; │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - try { │ │ │ │ │ - oRequest.statusText = oRequest._object.statusText; │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - function fCleanTransport(oRequest) { │ │ │ │ │ - // BUGFIX: IE - memory leak (on-page leak) │ │ │ │ │ - oRequest._object.onreadystatechange = new window.Function; │ │ │ │ │ - }; │ │ │ │ │ - /* │ │ │ │ │ - // Queue manager │ │ │ │ │ - var oQueuePending = {"CRITICAL":[],"HIGH":[],"NORMAL":[],"LOW":[],"LOWEST":[]}, │ │ │ │ │ - aQueueRunning = []; │ │ │ │ │ - function fQueue_add(oRequest) { │ │ │ │ │ - oQueuePending[oRequest.priority in oQueuePending ? oRequest.priority : "NORMAL"].push(oRequest); │ │ │ │ │ - // │ │ │ │ │ - setTimeout(fQueue_process); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - function fQueue_remove(oRequest) { │ │ │ │ │ - for (var nIndex = 0, bFound = false; nIndex < aQueueRunning.length; nIndex++) │ │ │ │ │ - if (bFound) │ │ │ │ │ - aQueueRunning[nIndex - 1] = aQueueRunning[nIndex]; │ │ │ │ │ - else │ │ │ │ │ - if (aQueueRunning[nIndex] == oRequest) │ │ │ │ │ - bFound = true; │ │ │ │ │ - if (bFound) │ │ │ │ │ - aQueueRunning.length--; │ │ │ │ │ - // │ │ │ │ │ - setTimeout(fQueue_process); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - function fQueue_process() { │ │ │ │ │ - if (aQueueRunning.length < 6) { │ │ │ │ │ - for (var sPriority in oQueuePending) { │ │ │ │ │ - if (oQueuePending[sPriority].length) { │ │ │ │ │ - var oRequest = oQueuePending[sPriority][0]; │ │ │ │ │ - oQueuePending[sPriority] = oQueuePending[sPriority].slice(1); │ │ │ │ │ - // │ │ │ │ │ - aQueueRunning.push(oRequest); │ │ │ │ │ - // Send request │ │ │ │ │ - fXMLHttpRequest_send(oRequest); │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - */ │ │ │ │ │ - // Internet Explorer 5.0 (missing apply) │ │ │ │ │ - if (!window.Function.prototype.apply) { │ │ │ │ │ - window.Function.prototype.apply = function(oRequest, oArguments) { │ │ │ │ │ - if (!oArguments) │ │ │ │ │ - oArguments = []; │ │ │ │ │ - oRequest.__func = this; │ │ │ │ │ - oRequest.__func(oArguments[0], oArguments[1], oArguments[2], oArguments[3], oArguments[4]); │ │ │ │ │ - delete oRequest.__func; │ │ │ │ │ - }; │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // Register new object with window │ │ │ │ │ - /** │ │ │ │ │ - * Class: OpenLayers.Request.XMLHttpRequest │ │ │ │ │ - * Standard-compliant (W3C) cross-browser implementation of the │ │ │ │ │ - * XMLHttpRequest object. From │ │ │ │ │ - * http://code.google.com/p/xmlhttprequest/. │ │ │ │ │ - */ │ │ │ │ │ - if (!OpenLayers.Request) { │ │ │ │ │ - /** │ │ │ │ │ - * This allows for OpenLayers/Request.js to be included │ │ │ │ │ - * before or after this script. │ │ │ │ │ - */ │ │ │ │ │ - OpenLayers.Request = {}; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Request.XMLHttpRequest = cXMLHttpRequest; │ │ │ │ │ -})(); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Request.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ - * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * TODO: deprecate me │ │ │ │ │ - * Use OpenLayers.Request.proxy instead. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.ProxyHost = ""; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Request │ │ │ │ │ - * The OpenLayers.Request namespace contains convenience methods for working │ │ │ │ │ - * with XMLHttpRequests. These methods work with a cross-browser │ │ │ │ │ - * W3C compliant class. │ │ │ │ │ - */ │ │ │ │ │ -if (!OpenLayers.Request) { │ │ │ │ │ - /** │ │ │ │ │ - * This allows for OpenLayers/Request/XMLHttpRequest.js to be included │ │ │ │ │ - * before or after this script. │ │ │ │ │ - */ │ │ │ │ │ - OpenLayers.Request = {}; │ │ │ │ │ -} │ │ │ │ │ -OpenLayers.Util.extend(OpenLayers.Request, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: DEFAULT_CONFIG │ │ │ │ │ - * {Object} Default configuration for all requests. │ │ │ │ │ - */ │ │ │ │ │ - DEFAULT_CONFIG: { │ │ │ │ │ - method: "GET", │ │ │ │ │ - url: window.location.href, │ │ │ │ │ - async: true, │ │ │ │ │ - user: undefined, │ │ │ │ │ - password: undefined, │ │ │ │ │ - params: null, │ │ │ │ │ - proxy: OpenLayers.ProxyHost, │ │ │ │ │ - headers: {}, │ │ │ │ │ - data: null, │ │ │ │ │ - callback: function() {}, │ │ │ │ │ - success: null, │ │ │ │ │ - failure: null, │ │ │ │ │ - scope: null │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: URL_SPLIT_REGEX │ │ │ │ │ - */ │ │ │ │ │ - URL_SPLIT_REGEX: /([^:]*:)\/\/([^:]*:?[^@]*@)?([^:\/\?]*):?([^\/\?]*)/, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {} An events object that handles all │ │ │ │ │ - * events on the {} object. │ │ │ │ │ - * │ │ │ │ │ - * All event listeners will receive an event object with three properties: │ │ │ │ │ - * request - {} The request object. │ │ │ │ │ - * config - {Object} The config object sent to the specific request method. │ │ │ │ │ - * requestUrl - {String} The request url. │ │ │ │ │ - * │ │ │ │ │ - * Supported event types: │ │ │ │ │ - * complete - Triggered when we have a response from the request, if a │ │ │ │ │ - * listener returns false, no further response processing will take │ │ │ │ │ - * place. │ │ │ │ │ - * success - Triggered when the HTTP response has a success code (200-299). │ │ │ │ │ - * failure - Triggered when the HTTP response does not have a success code. │ │ │ │ │ - */ │ │ │ │ │ - events: new OpenLayers.Events(this), │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: makeSameOrigin │ │ │ │ │ - * Using the specified proxy, returns a same origin url of the provided url. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * url - {String} An arbitrary url │ │ │ │ │ - * proxy {String|Function} The proxy to use to make the provided url a │ │ │ │ │ - * same origin url. │ │ │ │ │ - * │ │ │ │ │ - * Returns │ │ │ │ │ - * {String} the same origin url. If no proxy is provided, the returned url │ │ │ │ │ - * will be the same as the provided url. │ │ │ │ │ - */ │ │ │ │ │ - makeSameOrigin: function(url, proxy) { │ │ │ │ │ - var sameOrigin = url.indexOf("http") !== 0; │ │ │ │ │ - var urlParts = !sameOrigin && url.match(this.URL_SPLIT_REGEX); │ │ │ │ │ - if (urlParts) { │ │ │ │ │ - var location = window.location; │ │ │ │ │ - sameOrigin = │ │ │ │ │ - urlParts[1] == location.protocol && │ │ │ │ │ - urlParts[3] == location.hostname; │ │ │ │ │ - var uPort = urlParts[4], │ │ │ │ │ - lPort = location.port; │ │ │ │ │ - if (uPort != 80 && uPort != "" || lPort != "80" && lPort != "") { │ │ │ │ │ - sameOrigin = sameOrigin && uPort == lPort; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!sameOrigin) { │ │ │ │ │ - if (proxy) { │ │ │ │ │ - if (typeof proxy == "function") { │ │ │ │ │ - url = proxy(url); │ │ │ │ │ - } else { │ │ │ │ │ - url = proxy + encodeURIComponent(url); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return url; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: issue │ │ │ │ │ - * Create a new XMLHttpRequest object, open it, set any headers, bind │ │ │ │ │ - * a callback to done state, and send any data. It is recommended that │ │ │ │ │ - * you use one , , , , , or . │ │ │ │ │ - * This method is only documented to provide detail on the configuration │ │ │ │ │ - * options available to all request methods. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Object containing properties for configuring the │ │ │ │ │ - * request. Allowed configuration properties are described below. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Allowed config properties: │ │ │ │ │ - * method - {String} One of GET, POST, PUT, DELETE, HEAD, or │ │ │ │ │ - * OPTIONS. Default is GET. │ │ │ │ │ - * url - {String} URL for the request. │ │ │ │ │ - * async - {Boolean} Open an asynchronous request. Default is true. │ │ │ │ │ - * user - {String} User for relevant authentication scheme. Set │ │ │ │ │ - * to null to clear current user. │ │ │ │ │ - * password - {String} Password for relevant authentication scheme. │ │ │ │ │ - * Set to null to clear current password. │ │ │ │ │ - * proxy - {String} Optional proxy. Defaults to │ │ │ │ │ - * . │ │ │ │ │ - * params - {Object} Any key:value pairs to be appended to the │ │ │ │ │ - * url as a query string. Assumes url doesn't already include a query │ │ │ │ │ - * string or hash. Typically, this is only appropriate for │ │ │ │ │ - * requests where the query string will be appended to the url. │ │ │ │ │ - * Parameter values that are arrays will be │ │ │ │ │ - * concatenated with a comma (note that this goes against form-encoding) │ │ │ │ │ - * as is done with . │ │ │ │ │ - * headers - {Object} Object with header:value pairs to be set on │ │ │ │ │ - * the request. │ │ │ │ │ - * data - {String | Document} Optional data to send with the request. │ │ │ │ │ - * Typically, this is only used with and requests. │ │ │ │ │ - * Make sure to provide the appropriate "Content-Type" header for your │ │ │ │ │ - * data. For and requests, the content type defaults to │ │ │ │ │ - * "application-xml". If your data is a different content type, or │ │ │ │ │ - * if you are using a different HTTP method, set the "Content-Type" │ │ │ │ │ - * header to match your data type. │ │ │ │ │ - * callback - {Function} Function to call when request is done. │ │ │ │ │ - * To determine if the request failed, check request.status (200 │ │ │ │ │ - * indicates success). │ │ │ │ │ - * success - {Function} Optional function to call if request status is in │ │ │ │ │ - * the 200s. This will be called in addition to callback above and │ │ │ │ │ - * would typically only be used as an alternative. │ │ │ │ │ - * failure - {Function} Optional function to call if request status is not │ │ │ │ │ - * in the 200s. This will be called in addition to callback above and │ │ │ │ │ - * would typically only be used as an alternative. │ │ │ │ │ - * scope - {Object} If callback is a public method on some object, │ │ │ │ │ - * set the scope to that object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. To abort the request before a response │ │ │ │ │ - * is received, call abort() on the request object. │ │ │ │ │ - */ │ │ │ │ │ - issue: function(config) { │ │ │ │ │ - // apply default config - proxy host may have changed │ │ │ │ │ - var defaultConfig = OpenLayers.Util.extend( │ │ │ │ │ - this.DEFAULT_CONFIG, { │ │ │ │ │ - proxy: OpenLayers.ProxyHost │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - config = config || {}; │ │ │ │ │ - config.headers = config.headers || {}; │ │ │ │ │ - config = OpenLayers.Util.applyDefaults(config, defaultConfig); │ │ │ │ │ - config.headers = OpenLayers.Util.applyDefaults(config.headers, defaultConfig.headers); │ │ │ │ │ - // Always set the "X-Requested-With" header to signal that this request │ │ │ │ │ - // was issued through the XHR-object. Since header keys are case │ │ │ │ │ - // insensitive and we want to allow overriding of the "X-Requested-With" │ │ │ │ │ - // header through the user we cannot use applyDefaults, but have to │ │ │ │ │ - // check manually whether we were called with a "X-Requested-With" │ │ │ │ │ - // header. │ │ │ │ │ - var customRequestedWithHeader = false, │ │ │ │ │ - headerKey; │ │ │ │ │ - for (headerKey in config.headers) { │ │ │ │ │ - if (config.headers.hasOwnProperty(headerKey)) { │ │ │ │ │ - if (headerKey.toLowerCase() === 'x-requested-with') { │ │ │ │ │ - customRequestedWithHeader = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (customRequestedWithHeader === false) { │ │ │ │ │ - // we did not have a custom "X-Requested-With" header │ │ │ │ │ - config.headers['X-Requested-With'] = 'XMLHttpRequest'; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // create request, open, and set headers │ │ │ │ │ - var request = new OpenLayers.Request.XMLHttpRequest(); │ │ │ │ │ - var url = OpenLayers.Util.urlAppend(config.url, │ │ │ │ │ - OpenLayers.Util.getParameterString(config.params || {})); │ │ │ │ │ - url = OpenLayers.Request.makeSameOrigin(url, config.proxy); │ │ │ │ │ - request.open( │ │ │ │ │ - config.method, url, config.async, config.user, config.password │ │ │ │ │ - ); │ │ │ │ │ - for (var header in config.headers) { │ │ │ │ │ - request.setRequestHeader(header, config.headers[header]); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var events = this.events; │ │ │ │ │ - │ │ │ │ │ - // we want to execute runCallbacks with "this" as the │ │ │ │ │ - // execution scope │ │ │ │ │ - var self = this; │ │ │ │ │ - │ │ │ │ │ - request.onreadystatechange = function() { │ │ │ │ │ - if (request.readyState == OpenLayers.Request.XMLHttpRequest.DONE) { │ │ │ │ │ - var proceed = events.triggerEvent( │ │ │ │ │ - "complete", { │ │ │ │ │ - request: request, │ │ │ │ │ - config: config, │ │ │ │ │ - requestUrl: url │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - if (proceed !== false) { │ │ │ │ │ - self.runCallbacks({ │ │ │ │ │ - request: request, │ │ │ │ │ - config: config, │ │ │ │ │ - requestUrl: url │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // send request (optionally with data) and return │ │ │ │ │ - // call in a timeout for asynchronous requests so the return is │ │ │ │ │ - // available before readyState == 4 for cached docs │ │ │ │ │ - if (config.async === false) { │ │ │ │ │ - request.send(config.data); │ │ │ │ │ - } else { │ │ │ │ │ - window.setTimeout(function() { │ │ │ │ │ - if (request.readyState !== 0) { // W3C: 0-UNSENT │ │ │ │ │ - request.send(config.data); │ │ │ │ │ - } │ │ │ │ │ - }, 0); │ │ │ │ │ - } │ │ │ │ │ - return request; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: runCallbacks │ │ │ │ │ - * Calls the complete, success and failure callbacks. Application │ │ │ │ │ - * can listen to the "complete" event, have the listener │ │ │ │ │ - * display a confirm window and always return false, and │ │ │ │ │ - * execute OpenLayers.Request.runCallbacks if the user │ │ │ │ │ - * hits "yes" in the confirm window. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Hash containing request, config and requestUrl keys │ │ │ │ │ - */ │ │ │ │ │ - runCallbacks: function(options) { │ │ │ │ │ - var request = options.request; │ │ │ │ │ - var config = options.config; │ │ │ │ │ - │ │ │ │ │ - // bind callbacks to readyState 4 (done) │ │ │ │ │ - var complete = (config.scope) ? │ │ │ │ │ - OpenLayers.Function.bind(config.callback, config.scope) : │ │ │ │ │ - config.callback; │ │ │ │ │ - │ │ │ │ │ - // optional success callback │ │ │ │ │ - var success; │ │ │ │ │ - if (config.success) { │ │ │ │ │ - success = (config.scope) ? │ │ │ │ │ - OpenLayers.Function.bind(config.success, config.scope) : │ │ │ │ │ - config.success; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // optional failure callback │ │ │ │ │ - var failure; │ │ │ │ │ - if (config.failure) { │ │ │ │ │ - failure = (config.scope) ? │ │ │ │ │ - OpenLayers.Function.bind(config.failure, config.scope) : │ │ │ │ │ - config.failure; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (OpenLayers.Util.createUrlObject(config.url).protocol == "file:" && │ │ │ │ │ - request.responseText) { │ │ │ │ │ - request.status = 200; │ │ │ │ │ - } │ │ │ │ │ - complete(request); │ │ │ │ │ - │ │ │ │ │ - if (!request.status || (request.status >= 200 && request.status < 300)) { │ │ │ │ │ - this.events.triggerEvent("success", options); │ │ │ │ │ - if (success) { │ │ │ │ │ - success(request); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ - this.events.triggerEvent("failure", options); │ │ │ │ │ - if (failure) { │ │ │ │ │ - failure(request); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: GET │ │ │ │ │ - * Send an HTTP GET request. Additional configuration properties are │ │ │ │ │ - * documented in the method, with the method property set │ │ │ │ │ - * to GET. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the method for documentation of allowed properties. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ - */ │ │ │ │ │ - GET: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "GET" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: POST │ │ │ │ │ - * Send a POST request. Additional configuration properties are │ │ │ │ │ - * documented in the method, with the method property set │ │ │ │ │ - * to POST and "Content-Type" header set to "application/xml". │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the method for documentation of allowed properties. The │ │ │ │ │ - * default "Content-Type" header will be set to "application-xml" if │ │ │ │ │ - * none is provided. This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ - */ │ │ │ │ │ - POST: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "POST" │ │ │ │ │ - }); │ │ │ │ │ - // set content type to application/xml if it isn't already set │ │ │ │ │ - config.headers = config.headers ? config.headers : {}; │ │ │ │ │ - if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ - config.headers["Content-Type"] = "application/xml"; │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: PUT │ │ │ │ │ - * Send an HTTP PUT request. Additional configuration properties are │ │ │ │ │ - * documented in the method, with the method property set │ │ │ │ │ - * to PUT and "Content-Type" header set to "application/xml". │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the method for documentation of allowed properties. The │ │ │ │ │ - * default "Content-Type" header will be set to "application-xml" if │ │ │ │ │ - * none is provided. This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ - */ │ │ │ │ │ - PUT: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "PUT" │ │ │ │ │ - }); │ │ │ │ │ - // set content type to application/xml if it isn't already set │ │ │ │ │ - config.headers = config.headers ? config.headers : {}; │ │ │ │ │ - if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ - config.headers["Content-Type"] = "application/xml"; │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: DELETE │ │ │ │ │ - * Send an HTTP DELETE request. Additional configuration properties are │ │ │ │ │ - * documented in the method, with the method property set │ │ │ │ │ - * to DELETE. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the method for documentation of allowed properties. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ - */ │ │ │ │ │ - DELETE: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "DELETE" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: HEAD │ │ │ │ │ - * Send an HTTP HEAD request. Additional configuration properties are │ │ │ │ │ - * documented in the method, with the method property set │ │ │ │ │ - * to HEAD. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the method for documentation of allowed properties. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ - */ │ │ │ │ │ - HEAD: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "HEAD" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: OPTIONS │ │ │ │ │ - * Send an HTTP OPTIONS request. Additional configuration properties are │ │ │ │ │ - * documented in the method, with the method property set │ │ │ │ │ - * to OPTIONS. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the method for documentation of allowed properties. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ - */ │ │ │ │ │ - OPTIONS: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "OPTIONS" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Icon.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Icon │ │ │ │ │ - * │ │ │ │ │ - * The icon represents a graphical icon on the screen. Typically used in │ │ │ │ │ - * conjunction with a to represent markers on a screen. │ │ │ │ │ - * │ │ │ │ │ - * An icon has a url, size and position. It also contains an offset which │ │ │ │ │ - * allows the center point to be represented correctly. This can be │ │ │ │ │ - * provided either as a fixed offset or a function provided to calculate │ │ │ │ │ - * the desired offset. │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Icon = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: url │ │ │ │ │ - * {String} image url │ │ │ │ │ - */ │ │ │ │ │ - url: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: size │ │ │ │ │ - * {|Object} An OpenLayers.Size or │ │ │ │ │ - * an object with a 'w' and 'h' properties. │ │ │ │ │ - */ │ │ │ │ │ - size: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: offset │ │ │ │ │ - * {|Object} distance in pixels to offset the │ │ │ │ │ - * image when being rendered. An OpenLayers.Pixel or an object │ │ │ │ │ - * with a 'x' and 'y' properties. │ │ │ │ │ - */ │ │ │ │ │ - offset: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: calculateOffset │ │ │ │ │ - * {Function} Function to calculate the offset (based on the size) │ │ │ │ │ - */ │ │ │ │ │ - calculateOffset: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: imageDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - imageDiv: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: px │ │ │ │ │ - * {|Object} An OpenLayers.Pixel or an object │ │ │ │ │ - * with a 'x' and 'y' properties. │ │ │ │ │ - */ │ │ │ │ │ - px: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Icon │ │ │ │ │ - * Creates an icon, which is an image tag in a div. │ │ │ │ │ - * │ │ │ │ │ - * url - {String} │ │ │ │ │ - * size - {|Object} An OpenLayers.Size or an │ │ │ │ │ - * object with a 'w' and 'h' │ │ │ │ │ - * properties. │ │ │ │ │ - * offset - {|Object} An OpenLayers.Pixel or an │ │ │ │ │ - * object with a 'x' and 'y' │ │ │ │ │ - * properties. │ │ │ │ │ - * calculateOffset - {Function} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(url, size, offset, calculateOffset) { │ │ │ │ │ - this.url = url; │ │ │ │ │ - this.size = size || { │ │ │ │ │ - w: 20, │ │ │ │ │ - h: 20 │ │ │ │ │ - }; │ │ │ │ │ - this.offset = offset || { │ │ │ │ │ - x: -(this.size.w / 2), │ │ │ │ │ - y: -(this.size.h / 2) │ │ │ │ │ - }; │ │ │ │ │ - this.calculateOffset = calculateOffset; │ │ │ │ │ - │ │ │ │ │ - var id = OpenLayers.Util.createUniqueID("OL_Icon_"); │ │ │ │ │ - this.imageDiv = OpenLayers.Util.createAlphaImageDiv(id); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Nullify references and remove event listeners to prevent circular │ │ │ │ │ - * references and memory leaks │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - // erase any drawn elements │ │ │ │ │ - this.erase(); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Event.stopObservingElement(this.imageDiv.firstChild); │ │ │ │ │ - this.imageDiv.innerHTML = ""; │ │ │ │ │ - this.imageDiv = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} A fresh copy of the icon. │ │ │ │ │ - */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - return new OpenLayers.Icon(this.url, │ │ │ │ │ - this.size, │ │ │ │ │ - this.offset, │ │ │ │ │ - this.calculateOffset); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setSize │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * size - {|Object} An OpenLayers.Size or │ │ │ │ │ - * an object with a 'w' and 'h' properties. │ │ │ │ │ - */ │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - if (size != null) { │ │ │ │ │ - this.size = size; │ │ │ │ │ - } │ │ │ │ │ - this.draw(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setUrl │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * url - {String} │ │ │ │ │ - */ │ │ │ │ │ - setUrl: function(url) { │ │ │ │ │ - if (url != null) { │ │ │ │ │ - this.url = url; │ │ │ │ │ - } │ │ │ │ │ - this.draw(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Move the div to the given pixel. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {|Object} An OpenLayers.Pixel or an │ │ │ │ │ - * object with a 'x' and 'y' properties. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A new DOM Image of this icon set at the location passed-in │ │ │ │ │ - */ │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, │ │ │ │ │ - null, │ │ │ │ │ - null, │ │ │ │ │ - this.size, │ │ │ │ │ - this.url, │ │ │ │ │ - "absolute"); │ │ │ │ │ - this.moveTo(px); │ │ │ │ │ - return this.imageDiv; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: erase │ │ │ │ │ - * Erase the underlying image element. │ │ │ │ │ - */ │ │ │ │ │ - erase: function() { │ │ │ │ │ - if (this.imageDiv != null && this.imageDiv.parentNode != null) { │ │ │ │ │ - OpenLayers.Element.remove(this.imageDiv); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setOpacity │ │ │ │ │ - * Change the icon's opacity │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * opacity - {float} │ │ │ │ │ - */ │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, null, │ │ │ │ │ - null, null, null, null, opacity); │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * move icon to passed in px. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {|Object} the pixel position to move to. │ │ │ │ │ - * An OpenLayers.Pixel or an object with a 'x' and 'y' properties. │ │ │ │ │ - */ │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - //if no px passed in, use stored location │ │ │ │ │ - if (px != null) { │ │ │ │ │ - this.px = px; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.imageDiv != null) { │ │ │ │ │ - if (this.px == null) { │ │ │ │ │ - this.display(false); │ │ │ │ │ - } else { │ │ │ │ │ - if (this.calculateOffset) { │ │ │ │ │ - this.offset = this.calculateOffset(this.size); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, { │ │ │ │ │ - x: this.px.x + this.offset.x, │ │ │ │ │ - y: this.px.y + this.offset.y │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: display │ │ │ │ │ - * Hide or show the icon │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * display - {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - display: function(display) { │ │ │ │ │ - this.imageDiv.style.display = (display) ? "" : "none"; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: isDrawn │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the icon is drawn. │ │ │ │ │ - */ │ │ │ │ │ - isDrawn: function() { │ │ │ │ │ - // nodeType 11 for ie, whose nodes *always* have a parentNode │ │ │ │ │ - // (of type document fragment) │ │ │ │ │ - var isDrawn = (this.imageDiv && this.imageDiv.parentNode && │ │ │ │ │ - (this.imageDiv.parentNode.nodeType != 11)); │ │ │ │ │ - │ │ │ │ │ - return isDrawn; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Icon" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Marker.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ - * @requires OpenLayers/Icon.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Marker │ │ │ │ │ - * Instances of OpenLayers.Marker are a combination of a │ │ │ │ │ - * and an . │ │ │ │ │ - * │ │ │ │ │ - * Markers are generally added to a special layer called │ │ │ │ │ - * . │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var markers = new OpenLayers.Layer.Markers( "Markers" ); │ │ │ │ │ - * map.addLayer(markers); │ │ │ │ │ - * │ │ │ │ │ - * var size = new OpenLayers.Size(21,25); │ │ │ │ │ - * var offset = new OpenLayers.Pixel(-(size.w/2), -size.h); │ │ │ │ │ - * var icon = new OpenLayers.Icon('http://www.openlayers.org/dev/img/marker.png', size, offset); │ │ │ │ │ - * markers.addMarker(new OpenLayers.Marker(new OpenLayers.LonLat(0,0),icon)); │ │ │ │ │ - * markers.addMarker(new OpenLayers.Marker(new OpenLayers.LonLat(0,0),icon.clone())); │ │ │ │ │ - * │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Note that if you pass an icon into the Marker constructor, it will take │ │ │ │ │ - * that icon and use it. This means that you should not share icons between │ │ │ │ │ - * markers -- you use them once, but you should clone() for any additional │ │ │ │ │ - * markers using that same icon. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Marker = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: icon │ │ │ │ │ - * {} The icon used by this marker. │ │ │ │ │ - */ │ │ │ │ │ - icon: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: lonlat │ │ │ │ │ - * {} location of object │ │ │ │ │ - */ │ │ │ │ │ - lonlat: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: events │ │ │ │ │ - * {} the event handler. │ │ │ │ │ - */ │ │ │ │ │ - events: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: map │ │ │ │ │ - * {} the map this marker is attached to │ │ │ │ │ - */ │ │ │ │ │ - map: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Marker │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * lonlat - {} the position of this marker │ │ │ │ │ - * icon - {} the icon for this marker │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(lonlat, icon) { │ │ │ │ │ - this.lonlat = lonlat; │ │ │ │ │ - │ │ │ │ │ - var newIcon = (icon) ? icon : OpenLayers.Marker.defaultIcon(); │ │ │ │ │ - if (this.icon == null) { │ │ │ │ │ - this.icon = newIcon; │ │ │ │ │ - } else { │ │ │ │ │ - this.icon.url = newIcon.url; │ │ │ │ │ - this.icon.size = newIcon.size; │ │ │ │ │ - this.icon.offset = newIcon.offset; │ │ │ │ │ - this.icon.calculateOffset = newIcon.calculateOffset; │ │ │ │ │ - } │ │ │ │ │ - this.events = new OpenLayers.Events(this, this.icon.imageDiv); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Destroy the marker. You must first remove the marker from any │ │ │ │ │ - * layer which it has been added to, or you will get buggy behavior. │ │ │ │ │ - * (This can not be done within the marker since the marker does not │ │ │ │ │ - * know which layer it is attached to.) │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - // erase any drawn features │ │ │ │ │ - this.erase(); │ │ │ │ │ - │ │ │ │ │ - this.map = null; │ │ │ │ │ - │ │ │ │ │ - this.events.destroy(); │ │ │ │ │ - this.events = null; │ │ │ │ │ - │ │ │ │ │ - if (this.icon != null) { │ │ │ │ │ - this.icon.destroy(); │ │ │ │ │ - this.icon = null; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Calls draw on the icon, and returns that output. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A new DOM Image with this marker's icon set at the │ │ │ │ │ - * location passed-in │ │ │ │ │ - */ │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - return this.icon.draw(px); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: erase │ │ │ │ │ - * Erases any drawn elements for this marker. │ │ │ │ │ - */ │ │ │ │ │ - erase: function() { │ │ │ │ │ - if (this.icon != null) { │ │ │ │ │ - this.icon.erase(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * Move the marker to the new location. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {|Object} the pixel position to move to. │ │ │ │ │ - * An OpenLayers.Pixel or an object with a 'x' and 'y' properties. │ │ │ │ │ - */ │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - if ((px != null) && (this.icon != null)) { │ │ │ │ │ - this.icon.moveTo(px); │ │ │ │ │ - } │ │ │ │ │ - this.lonlat = this.map.getLonLatFromLayerPx(px); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: isDrawn │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the marker is drawn. │ │ │ │ │ - */ │ │ │ │ │ - isDrawn: function() { │ │ │ │ │ - var isDrawn = (this.icon && this.icon.isDrawn()); │ │ │ │ │ - return isDrawn; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onScreen │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the marker is currently visible on screen. │ │ │ │ │ - */ │ │ │ │ │ - onScreen: function() { │ │ │ │ │ - │ │ │ │ │ - var onScreen = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - var screenBounds = this.map.getExtent(); │ │ │ │ │ - onScreen = screenBounds.containsLonLat(this.lonlat); │ │ │ │ │ - } │ │ │ │ │ - return onScreen; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: inflate │ │ │ │ │ - * Englarges the markers icon by the specified ratio. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * inflate - {float} the ratio to enlarge the marker by (passing 2 │ │ │ │ │ - * will double the size). │ │ │ │ │ - */ │ │ │ │ │ - inflate: function(inflate) { │ │ │ │ │ - if (this.icon) { │ │ │ │ │ - this.icon.setSize({ │ │ │ │ │ - w: this.icon.size.w * inflate, │ │ │ │ │ - h: this.icon.size.h * inflate │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setOpacity │ │ │ │ │ - * Change the opacity of the marker by changin the opacity of │ │ │ │ │ - * its icon │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * opacity - {float} Specified as fraction (0.4, etc) │ │ │ │ │ - */ │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - this.icon.setOpacity(opacity); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setUrl │ │ │ │ │ - * Change URL of the Icon Image. │ │ │ │ │ - * │ │ │ │ │ - * url - {String} │ │ │ │ │ - */ │ │ │ │ │ - setUrl: function(url) { │ │ │ │ │ - this.icon.setUrl(url); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: display │ │ │ │ │ - * Hide or show the icon │ │ │ │ │ - * │ │ │ │ │ - * display - {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - display: function(display) { │ │ │ │ │ - this.icon.display(display); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Marker" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Function: defaultIcon │ │ │ │ │ - * Creates a default . │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} A default OpenLayers.Icon to use for a marker │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Marker.defaultIcon = function() { │ │ │ │ │ - return new OpenLayers.Icon(OpenLayers.Util.getImageLocation("marker.png"), { │ │ │ │ │ - w: 21, │ │ │ │ │ - h: 25 │ │ │ │ │ - }, { │ │ │ │ │ - x: -10.5, │ │ │ │ │ - y: -25 │ │ │ │ │ - }); │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Handler │ │ │ │ │ - * Base class to construct a higher-level handler for event sequences. All │ │ │ │ │ - * handlers have activate and deactivate methods. In addition, they have │ │ │ │ │ - * methods named like browser events. When a handler is activated, any │ │ │ │ │ - * additional methods named like a browser event is registered as a │ │ │ │ │ - * listener for the corresponding event. When a handler is deactivated, │ │ │ │ │ - * those same methods are unregistered as event listeners. │ │ │ │ │ - * │ │ │ │ │ - * Handlers also typically have a callbacks object with keys named like │ │ │ │ │ - * the abstracted events or event sequences that they are in charge of │ │ │ │ │ - * handling. The controls that wrap handlers define the methods that │ │ │ │ │ - * correspond to these abstract events - so instead of listening for │ │ │ │ │ - * individual browser events, they only listen for the abstract events │ │ │ │ │ - * defined by the handler. │ │ │ │ │ - * │ │ │ │ │ - * Handlers are created by controls, which ultimately have the responsibility │ │ │ │ │ - * of making changes to the the state of the application. Handlers │ │ │ │ │ - * themselves may make temporary changes, but in general are expected to │ │ │ │ │ - * return the application in the same state that they found it. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: id │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - id: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: control │ │ │ │ │ - * {}. The control that initialized this handler. The │ │ │ │ │ - * control is assumed to have a valid map property - that map is used │ │ │ │ │ - * in the handler's own setMap method. │ │ │ │ │ - */ │ │ │ │ │ - control: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: map │ │ │ │ │ - * {} │ │ │ │ │ - */ │ │ │ │ │ - map: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: keyMask │ │ │ │ │ - * {Integer} Use bitwise operators and one or more of the OpenLayers.Handler │ │ │ │ │ - * constants to construct a keyMask. The keyMask is used by │ │ │ │ │ - * . If the keyMask matches the combination of keys │ │ │ │ │ - * down on an event, checkModifiers returns true. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * // handler only responds if the Shift key is down │ │ │ │ │ - * handler.keyMask = OpenLayers.Handler.MOD_SHIFT; │ │ │ │ │ - * │ │ │ │ │ - * // handler only responds if Ctrl-Shift is down │ │ │ │ │ - * handler.keyMask = OpenLayers.Handler.MOD_SHIFT | │ │ │ │ │ - * OpenLayers.Handler.MOD_CTRL; │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ - keyMask: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: active │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - active: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: evt │ │ │ │ │ - * {Event} This property references the last event handled by the handler. │ │ │ │ │ - * Note that this property is not part of the stable API. Use of the │ │ │ │ │ - * evt property should be restricted to controls in the library │ │ │ │ │ - * or other applications that are willing to update with changes to │ │ │ │ │ - * the OpenLayers code. │ │ │ │ │ - */ │ │ │ │ │ - evt: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: touch │ │ │ │ │ - * {Boolean} Indicates the support of touch events. When touch events are │ │ │ │ │ - * started touch will be true and all mouse related listeners will do │ │ │ │ │ - * nothing. │ │ │ │ │ - */ │ │ │ │ │ - touch: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Handler │ │ │ │ │ - * Construct a handler. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {} The control that initialized this │ │ │ │ │ - * handler. The control is assumed to have a valid map property; that │ │ │ │ │ - * map is used in the handler's own setMap method. If a map property │ │ │ │ │ - * is present in the options argument it will be used instead. │ │ │ │ │ - * callbacks - {Object} An object whose properties correspond to abstracted │ │ │ │ │ - * events or sequences of browser events. The values for these │ │ │ │ │ - * properties are functions defined by the control that get called by │ │ │ │ │ - * the handler. │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * the handler. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.control = control; │ │ │ │ │ - this.callbacks = callbacks; │ │ │ │ │ - │ │ │ │ │ - var map = this.map || control.map; │ │ │ │ │ - if (map) { │ │ │ │ │ - this.setMap(map); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.map = map; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: checkModifiers │ │ │ │ │ - * Check the keyMask on the handler. If no is set, this always │ │ │ │ │ - * returns true. If a is set and it matches the combination │ │ │ │ │ - * of keys down on an event, this returns true. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The keyMask matches the keys down on an event. │ │ │ │ │ - */ │ │ │ │ │ - checkModifiers: function(evt) { │ │ │ │ │ - if (this.keyMask == null) { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - /* calculate the keyboard modifier mask for this event */ │ │ │ │ │ - var keyModifiers = │ │ │ │ │ - (evt.shiftKey ? OpenLayers.Handler.MOD_SHIFT : 0) | │ │ │ │ │ - (evt.ctrlKey ? OpenLayers.Handler.MOD_CTRL : 0) | │ │ │ │ │ - (evt.altKey ? OpenLayers.Handler.MOD_ALT : 0) | │ │ │ │ │ - (evt.metaKey ? OpenLayers.Handler.MOD_META : 0); │ │ │ │ │ - │ │ │ │ │ - /* if it differs from the handler object's key mask, │ │ │ │ │ - bail out of the event handler */ │ │ │ │ │ - return (keyModifiers == this.keyMask); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Turn on the handler. Returns false if the handler was already active. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - // register for event handlers defined on this class. │ │ │ │ │ - var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.register(events[i], this[events[i]]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.active = true; │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Turn off the handler. Returns false if the handler was already inactive. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - // unregister event handlers defined on this class. │ │ │ │ │ - var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.unregister(events[i], this[events[i]]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.touch = false; │ │ │ │ │ - this.active = false; │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: startTouch │ │ │ │ │ - * Start touch events, this method must be called by subclasses in │ │ │ │ │ - * "touchstart" method. When touch events are started will be │ │ │ │ │ - * true and all mouse related listeners will do nothing. │ │ │ │ │ - */ │ │ │ │ │ - startTouch: function() { │ │ │ │ │ - if (!this.touch) { │ │ │ │ │ - this.touch = true; │ │ │ │ │ - var events = [ │ │ │ │ │ - "mousedown", "mouseup", "mousemove", "click", "dblclick", │ │ │ │ │ - "mouseout" │ │ │ │ │ - ]; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.unregister(events[i], this[events[i]]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: callback │ │ │ │ │ - * Trigger the control's named callback with the given arguments │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} The key for the callback that is one of the properties │ │ │ │ │ - * of the handler's callbacks object. │ │ │ │ │ - * args - {Array(*)} An array of arguments (any type) with which to call │ │ │ │ │ - * the callback (defined by the control). │ │ │ │ │ - */ │ │ │ │ │ - callback: function(name, args) { │ │ │ │ │ - if (name && this.callbacks[name]) { │ │ │ │ │ - this.callbacks[name].apply(this.control, args); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: register │ │ │ │ │ - * register an event on the map │ │ │ │ │ - */ │ │ │ │ │ - register: function(name, method) { │ │ │ │ │ - // TODO: deal with registerPriority in 3.0 │ │ │ │ │ - this.map.events.registerPriority(name, this, method); │ │ │ │ │ - this.map.events.registerPriority(name, this, this.setEvent); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: unregister │ │ │ │ │ - * unregister an event from the map │ │ │ │ │ - */ │ │ │ │ │ - unregister: function(name, method) { │ │ │ │ │ - this.map.events.unregister(name, this, method); │ │ │ │ │ - this.map.events.unregister(name, this, this.setEvent); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setEvent │ │ │ │ │ - * With each registered browser event, the handler sets its own evt │ │ │ │ │ - * property. This property can be accessed by controls if needed │ │ │ │ │ - * to get more information about the event that the handler is │ │ │ │ │ - * processing. │ │ │ │ │ - * │ │ │ │ │ - * This allows modifier keys on the event to be checked (alt, shift, ctrl, │ │ │ │ │ - * and meta cannot be checked with the keyboard handler). For a │ │ │ │ │ - * control to determine which modifier keys are associated with the │ │ │ │ │ - * event that a handler is currently processing, it should access │ │ │ │ │ - * (code)handler.evt.altKey || handler.evt.shiftKey || │ │ │ │ │ - * handler.evt.ctrlKey || handler.evt.metaKey(end). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event. │ │ │ │ │ - */ │ │ │ │ │ - setEvent: function(evt) { │ │ │ │ │ - this.evt = evt; │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Deconstruct the handler. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - // unregister event listeners │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - // eliminate circular references │ │ │ │ │ - this.control = this.map = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_NONE │ │ │ │ │ - * If set as the , returns false if any key is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_NONE = 0; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_SHIFT │ │ │ │ │ - * If set as the , returns false if Shift is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_SHIFT = 1; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_CTRL │ │ │ │ │ - * If set as the , returns false if Ctrl is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_CTRL = 2; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_ALT │ │ │ │ │ - * If set as the , returns false if Alt is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_ALT = 4; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_META │ │ │ │ │ - * If set as the , returns false if Cmd is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_META = 8; │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Feature.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Feature │ │ │ │ │ - * Features are combinations of geography and attributes. The OpenLayers.Feature │ │ │ │ │ - * class specifically combines a marker and a lonlat. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Feature = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {} │ │ │ │ │ - */ │ │ │ │ │ - layer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: id │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - id: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: lonlat │ │ │ │ │ - * {} │ │ │ │ │ - */ │ │ │ │ │ - lonlat: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: data │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ - data: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: marker │ │ │ │ │ - * {} │ │ │ │ │ - */ │ │ │ │ │ - marker: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: popupClass │ │ │ │ │ - * {} The class which will be used to instantiate │ │ │ │ │ - * a new Popup. Default is . │ │ │ │ │ - */ │ │ │ │ │ - popupClass: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: popup │ │ │ │ │ - * {} │ │ │ │ │ - */ │ │ │ │ │ - popup: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Feature │ │ │ │ │ - * Constructor for features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {} │ │ │ │ │ - * lonlat - {} │ │ │ │ │ - * data - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(layer, lonlat, data) { │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - this.lonlat = lonlat; │ │ │ │ │ - this.data = (data != null) ? data : {}; │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * nullify references to prevent circular references and memory leaks │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - │ │ │ │ │ - //remove the popup from the map │ │ │ │ │ - if ((this.layer != null) && (this.layer.map != null)) { │ │ │ │ │ - if (this.popup != null) { │ │ │ │ │ - this.layer.map.removePopup(this.popup); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // remove the marker from the layer │ │ │ │ │ - if (this.layer != null && this.marker != null) { │ │ │ │ │ - this.layer.removeMarker(this.marker); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.layer = null; │ │ │ │ │ - this.id = null; │ │ │ │ │ - this.lonlat = null; │ │ │ │ │ - this.data = null; │ │ │ │ │ - if (this.marker != null) { │ │ │ │ │ - this.destroyMarker(this.marker); │ │ │ │ │ - this.marker = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.popup != null) { │ │ │ │ │ - this.destroyPopup(this.popup); │ │ │ │ │ - this.popup = null; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onScreen │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the feature is currently visible on screen │ │ │ │ │ - * (based on its 'lonlat' property) │ │ │ │ │ - */ │ │ │ │ │ - onScreen: function() { │ │ │ │ │ - │ │ │ │ │ - var onScreen = false; │ │ │ │ │ - if ((this.layer != null) && (this.layer.map != null)) { │ │ │ │ │ - var screenBounds = this.layer.map.getExtent(); │ │ │ │ │ - onScreen = screenBounds.containsLonLat(this.lonlat); │ │ │ │ │ - } │ │ │ │ │ - return onScreen; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createMarker │ │ │ │ │ - * Based on the data associated with the Feature, create and return a marker object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} A Marker Object created from the 'lonlat' and 'icon' properties │ │ │ │ │ - * set in this.data. If no 'lonlat' is set, returns null. If no │ │ │ │ │ - * 'icon' is set, OpenLayers.Marker() will load the default image. │ │ │ │ │ - * │ │ │ │ │ - * Note - this.marker is set to return value │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - createMarker: function() { │ │ │ │ │ - │ │ │ │ │ - if (this.lonlat != null) { │ │ │ │ │ - this.marker = new OpenLayers.Marker(this.lonlat, this.data.icon); │ │ │ │ │ - } │ │ │ │ │ - return this.marker; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroyMarker │ │ │ │ │ - * Destroys marker. │ │ │ │ │ - * If user overrides the createMarker() function, s/he should be able │ │ │ │ │ - * to also specify an alternative function for destroying it │ │ │ │ │ - */ │ │ │ │ │ - destroyMarker: function() { │ │ │ │ │ - this.marker.destroy(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createPopup │ │ │ │ │ - * Creates a popup object created from the 'lonlat', 'popupSize', │ │ │ │ │ - * and 'popupContentHTML' properties set in this.data. It uses │ │ │ │ │ - * this.marker.icon as default anchor. │ │ │ │ │ - * │ │ │ │ │ - * If no 'lonlat' is set, returns null. │ │ │ │ │ - * If no this.marker has been created, no anchor is sent. │ │ │ │ │ - * │ │ │ │ │ - * Note - the returned popup object is 'owned' by the feature, so you │ │ │ │ │ - * cannot use the popup's destroy method to discard the popup. │ │ │ │ │ - * Instead, you must use the feature's destroyPopup │ │ │ │ │ - * │ │ │ │ │ - * Note - this.popup is set to return value │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * closeBox - {Boolean} create popup with closebox or not │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} Returns the created popup, which is also set │ │ │ │ │ - * as 'popup' property of this feature. Will be of whatever type │ │ │ │ │ - * specified by this feature's 'popupClass' property, but must be │ │ │ │ │ - * of type . │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - createPopup: function(closeBox) { │ │ │ │ │ - │ │ │ │ │ - if (this.lonlat != null) { │ │ │ │ │ - if (!this.popup) { │ │ │ │ │ - var anchor = (this.marker) ? this.marker.icon : null; │ │ │ │ │ - var popupClass = this.popupClass ? │ │ │ │ │ - this.popupClass : OpenLayers.Popup.Anchored; │ │ │ │ │ - this.popup = new popupClass(this.id + "_popup", │ │ │ │ │ - this.lonlat, │ │ │ │ │ - this.data.popupSize, │ │ │ │ │ - this.data.popupContentHTML, │ │ │ │ │ - anchor, │ │ │ │ │ - closeBox); │ │ │ │ │ - } │ │ │ │ │ - if (this.data.overflow != null) { │ │ │ │ │ - this.popup.contentDiv.style.overflow = this.data.overflow; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.popup.feature = this; │ │ │ │ │ - } │ │ │ │ │ - return this.popup; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroyPopup │ │ │ │ │ - * Destroys the popup created via createPopup. │ │ │ │ │ - * │ │ │ │ │ - * As with the marker, if user overrides the createPopup() function, s/he │ │ │ │ │ - * should also be able to override the destruction │ │ │ │ │ - */ │ │ │ │ │ - destroyPopup: function() { │ │ │ │ │ - if (this.popup) { │ │ │ │ │ - this.popup.feature = null; │ │ │ │ │ - this.popup.destroy(); │ │ │ │ │ - this.popup = null; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Feature" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Feature/Vector.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -// TRASH THIS │ │ │ │ │ -OpenLayers.State = { │ │ │ │ │ - /** states */ │ │ │ │ │ - UNKNOWN: 'Unknown', │ │ │ │ │ - INSERT: 'Insert', │ │ │ │ │ - UPDATE: 'Update', │ │ │ │ │ - DELETE: 'Delete' │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Feature.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Feature.Vector │ │ │ │ │ - * Vector features use the OpenLayers.Geometry classes as geometry description. │ │ │ │ │ - * They have an 'attributes' property, which is the data object, and a 'style' │ │ │ │ │ - * property, the default values of which are defined in the │ │ │ │ │ - * objects. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Feature.Vector = OpenLayers.Class(OpenLayers.Feature, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: fid │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - fid: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geometry │ │ │ │ │ - * {} │ │ │ │ │ - */ │ │ │ │ │ - geometry: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: attributes │ │ │ │ │ - * {Object} This object holds arbitrary, serializable properties that │ │ │ │ │ - * describe the feature. │ │ │ │ │ - */ │ │ │ │ │ - attributes: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: bounds │ │ │ │ │ - * {} The box bounding that feature's geometry, that │ │ │ │ │ - * property can be set by an object when │ │ │ │ │ - * deserializing the feature, so in most cases it represents an │ │ │ │ │ - * information set by the server. │ │ │ │ │ - */ │ │ │ │ │ - bounds: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: state │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - state: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: style │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ - style: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String} If this property is set it will be taken into account by │ │ │ │ │ - * {} when upadting or deleting the feature. │ │ │ │ │ - */ │ │ │ │ │ - url: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: renderIntent │ │ │ │ │ - * {String} rendering intent currently being used │ │ │ │ │ - */ │ │ │ │ │ - renderIntent: "default", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: modified │ │ │ │ │ - * {Object} An object with the originals of the geometry and attributes of │ │ │ │ │ - * the feature, if they were changed. Currently this property is only read │ │ │ │ │ - * by , and written by │ │ │ │ │ - * , which sets the geometry property. │ │ │ │ │ - * Applications can set the originals of modified attributes in the │ │ │ │ │ - * attributes property. Note that applications have to check if this │ │ │ │ │ - * object and the attributes property is already created before using it. │ │ │ │ │ - * After a change made with ModifyFeature, this object could look like │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * { │ │ │ │ │ - * geometry: >Object │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * When an application has made changes to feature attributes, it could │ │ │ │ │ - * have set the attributes to something like this: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * { │ │ │ │ │ - * attributes: { │ │ │ │ │ - * myAttribute: "original" │ │ │ │ │ - * } │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Note that only checks for truthy values in │ │ │ │ │ - * *modified.geometry* and the attribute names in *modified.attributes*, │ │ │ │ │ - * but it is recommended to set the original values (and not just true) as │ │ │ │ │ - * attribute value, so applications could use this information to undo │ │ │ │ │ - * changes. │ │ │ │ │ - */ │ │ │ │ │ - modified: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Feature.Vector │ │ │ │ │ - * Create a vector feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {} The geometry that this feature │ │ │ │ │ - * represents. │ │ │ │ │ - * attributes - {Object} An optional object that will be mapped to the │ │ │ │ │ - * property. │ │ │ │ │ - * style - {Object} An optional style object. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(geometry, attributes, style) { │ │ │ │ │ - OpenLayers.Feature.prototype.initialize.apply(this, │ │ │ │ │ - [null, null, attributes]); │ │ │ │ │ - this.lonlat = null; │ │ │ │ │ - this.geometry = geometry ? geometry : null; │ │ │ │ │ - this.state = null; │ │ │ │ │ - this.attributes = {}; │ │ │ │ │ - if (attributes) { │ │ │ │ │ - this.attributes = OpenLayers.Util.extend(this.attributes, │ │ │ │ │ - attributes); │ │ │ │ │ - } │ │ │ │ │ - this.style = style ? style : null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * nullify references to prevent circular references and memory leaks │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.layer) { │ │ │ │ │ - this.layer.removeFeatures(this); │ │ │ │ │ - this.layer = null; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.geometry = null; │ │ │ │ │ - this.modified = null; │ │ │ │ │ - OpenLayers.Feature.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this vector feature. Does not set any non-standard │ │ │ │ │ - * properties. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} An exact clone of this vector feature. │ │ │ │ │ - */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - return new OpenLayers.Feature.Vector( │ │ │ │ │ - this.geometry ? this.geometry.clone() : null, │ │ │ │ │ - this.attributes, │ │ │ │ │ - this.style); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onScreen │ │ │ │ │ - * Determine whether the feature is within the map viewport. This method │ │ │ │ │ - * tests for an intersection between the geometry and the viewport │ │ │ │ │ - * bounds. If a more effecient but less precise geometry bounds │ │ │ │ │ - * intersection is desired, call the method with the boundsOnly │ │ │ │ │ - * parameter true. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * boundsOnly - {Boolean} Only test whether a feature's bounds intersects │ │ │ │ │ - * the viewport bounds. Default is false. If false, the feature's │ │ │ │ │ - * geometry must intersect the viewport for onScreen to return true. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The feature is currently visible on screen (optionally │ │ │ │ │ - * based on its bounds if boundsOnly is true). │ │ │ │ │ - */ │ │ │ │ │ - onScreen: function(boundsOnly) { │ │ │ │ │ - var onScreen = false; │ │ │ │ │ - if (this.layer && this.layer.map) { │ │ │ │ │ - var screenBounds = this.layer.map.getExtent(); │ │ │ │ │ - if (boundsOnly) { │ │ │ │ │ - var featureBounds = this.geometry.getBounds(); │ │ │ │ │ - onScreen = screenBounds.intersectsBounds(featureBounds); │ │ │ │ │ - } else { │ │ │ │ │ - var screenPoly = screenBounds.toGeometry(); │ │ │ │ │ - onScreen = screenPoly.intersects(this.geometry); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return onScreen; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getVisibility │ │ │ │ │ - * Determine whether the feature is displayed or not. It may not displayed │ │ │ │ │ - * because: │ │ │ │ │ - * - its style display property is set to 'none', │ │ │ │ │ - * - it doesn't belong to any layer, │ │ │ │ │ - * - the styleMap creates a symbolizer with display property set to 'none' │ │ │ │ │ - * for it, │ │ │ │ │ - * - the layer which it belongs to is not visible. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The feature is currently displayed. │ │ │ │ │ - */ │ │ │ │ │ - getVisibility: function() { │ │ │ │ │ - return !(this.style && this.style.display == 'none' || │ │ │ │ │ - !this.layer || │ │ │ │ │ - this.layer && this.layer.styleMap && │ │ │ │ │ - this.layer.styleMap.createSymbolizer(this, this.renderIntent).display == 'none' || │ │ │ │ │ - this.layer && !this.layer.getVisibility()); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createMarker │ │ │ │ │ - * HACK - we need to decide if all vector features should be able to │ │ │ │ │ - * create markers │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} For now just returns null │ │ │ │ │ - */ │ │ │ │ │ - createMarker: function() { │ │ │ │ │ - return null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroyMarker │ │ │ │ │ - * HACK - we need to decide if all vector features should be able to │ │ │ │ │ - * delete markers │ │ │ │ │ - * │ │ │ │ │ - * If user overrides the createMarker() function, s/he should be able │ │ │ │ │ - * to also specify an alternative function for destroying it │ │ │ │ │ - */ │ │ │ │ │ - destroyMarker: function() { │ │ │ │ │ - // pass │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createPopup │ │ │ │ │ - * HACK - we need to decide if all vector features should be able to │ │ │ │ │ - * create popups │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} For now just returns null │ │ │ │ │ - */ │ │ │ │ │ - createPopup: function() { │ │ │ │ │ - return null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: atPoint │ │ │ │ │ - * Determins whether the feature intersects with the specified location. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * lonlat - {|Object} OpenLayers.LonLat or an │ │ │ │ │ - * object with a 'lon' and 'lat' properties. │ │ │ │ │ - * toleranceLon - {float} Optional tolerance in Geometric Coords │ │ │ │ │ - * toleranceLat - {float} Optional tolerance in Geographic Coords │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the feature is at the specified location │ │ │ │ │ - */ │ │ │ │ │ - atPoint: function(lonlat, toleranceLon, toleranceLat) { │ │ │ │ │ - var atPoint = false; │ │ │ │ │ - if (this.geometry) { │ │ │ │ │ - atPoint = this.geometry.atPoint(lonlat, toleranceLon, │ │ │ │ │ - toleranceLat); │ │ │ │ │ - } │ │ │ │ │ - return atPoint; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroyPopup │ │ │ │ │ - * HACK - we need to decide if all vector features should be able to │ │ │ │ │ - * delete popups │ │ │ │ │ - */ │ │ │ │ │ - destroyPopup: function() { │ │ │ │ │ - // pass │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * Moves the feature and redraws it at its new location │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * location - { or } the │ │ │ │ │ - * location to which to move the feature. │ │ │ │ │ - */ │ │ │ │ │ - move: function(location) { │ │ │ │ │ - │ │ │ │ │ - if (!this.layer || !this.geometry.move) { │ │ │ │ │ - //do nothing if no layer or immoveable geometry │ │ │ │ │ - return undefined; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var pixel; │ │ │ │ │ - if (location.CLASS_NAME == "OpenLayers.LonLat") { │ │ │ │ │ - pixel = this.layer.getViewPortPxFromLonLat(location); │ │ │ │ │ - } else { │ │ │ │ │ - pixel = location; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var lastPixel = this.layer.getViewPortPxFromLonLat(this.geometry.getBounds().getCenterLonLat()); │ │ │ │ │ - var res = this.layer.map.getResolution(); │ │ │ │ │ - this.geometry.move(res * (pixel.x - lastPixel.x), │ │ │ │ │ - res * (lastPixel.y - pixel.y)); │ │ │ │ │ - this.layer.drawFeature(this); │ │ │ │ │ - return lastPixel; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: toState │ │ │ │ │ - * Sets the new state │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * state - {String} │ │ │ │ │ + * state - {String} │ │ │ │ │ */ │ │ │ │ │ toState: function(state) { │ │ │ │ │ if (state == OpenLayers.State.UPDATE) { │ │ │ │ │ switch (this.state) { │ │ │ │ │ case OpenLayers.State.UNKNOWN: │ │ │ │ │ case OpenLayers.State.DELETE: │ │ │ │ │ this.state = state; │ │ │ │ │ @@ -15052,464 +11095,177 @@ │ │ │ │ │ * {Array} prefixes of the sld symbolizers. These are the │ │ │ │ │ * same as the main geometry types │ │ │ │ │ */ │ │ │ │ │ OpenLayers.Style.SYMBOLIZER_PREFIXES = ['Point', 'Line', 'Polygon', 'Text', │ │ │ │ │ 'Raster' │ │ │ │ │ ]; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Filter.js │ │ │ │ │ + OpenLayers/StyleMap.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ * @requires OpenLayers/Style.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Filter │ │ │ │ │ - * This class represents an OGC Filter. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Filter = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Filter │ │ │ │ │ - * This class represents a generic filter. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Remove reference to anything added. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: evaluate │ │ │ │ │ - * Evaluates this filter in a specific context. Instances or subclasses │ │ │ │ │ - * are supposed to override this method. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * context - {Object} Context to use in evaluating the filter. If a vector │ │ │ │ │ - * feature is provided, the feature.attributes will be used as context. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The filter applies. │ │ │ │ │ - */ │ │ │ │ │ - evaluate: function(context) { │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Clones this filter. Should be implemented by subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} Clone of this filter. │ │ │ │ │ - */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - return null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: toString │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} Include in your build to get a CQL │ │ │ │ │ - * representation of the filter returned. Otherwise "[Object object]" │ │ │ │ │ - * will be returned. │ │ │ │ │ - */ │ │ │ │ │ - toString: function() { │ │ │ │ │ - var string; │ │ │ │ │ - if (OpenLayers.Format && OpenLayers.Format.CQL) { │ │ │ │ │ - string = OpenLayers.Format.CQL.prototype.write(this); │ │ │ │ │ - } else { │ │ │ │ │ - string = Object.prototype.toString.call(this); │ │ │ │ │ - } │ │ │ │ │ - return string; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Filter" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format │ │ │ │ │ - * Base class for format reading/writing a variety of formats. Subclasses │ │ │ │ │ - * of OpenLayers.Format are expected to have read and write methods. │ │ │ │ │ + * Class: OpenLayers.StyleMap │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: options │ │ │ │ │ - * {Object} A reference to options passed to the constructor. │ │ │ │ │ - */ │ │ │ │ │ - options: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: externalProjection │ │ │ │ │ - * {} When passed a externalProjection and │ │ │ │ │ - * internalProjection, the format will reproject the geometries it │ │ │ │ │ - * reads or writes. The externalProjection is the projection used by │ │ │ │ │ - * the content which is passed into read or which comes out of write. │ │ │ │ │ - * In order to reproject, a projection transformation function for the │ │ │ │ │ - * specified projections must be available. This support may be │ │ │ │ │ - * provided via proj4js or via a custom transformation function. See │ │ │ │ │ - * {} for more information on │ │ │ │ │ - * custom transformations. │ │ │ │ │ - */ │ │ │ │ │ - externalProjection: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: internalProjection │ │ │ │ │ - * {} When passed a externalProjection and │ │ │ │ │ - * internalProjection, the format will reproject the geometries it │ │ │ │ │ - * reads or writes. The internalProjection is the projection used by │ │ │ │ │ - * the geometries which are returned by read or which are passed into │ │ │ │ │ - * write. In order to reproject, a projection transformation function │ │ │ │ │ - * for the specified projections must be available. This support may be │ │ │ │ │ - * provided via proj4js or via a custom transformation function. See │ │ │ │ │ - * {} for more information on │ │ │ │ │ - * custom transformations. │ │ │ │ │ - */ │ │ │ │ │ - internalProjection: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: data │ │ │ │ │ - * {Object} When is true, this is the parsed string sent to │ │ │ │ │ - * . │ │ │ │ │ - */ │ │ │ │ │ - data: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: keepData │ │ │ │ │ - * {Object} Maintain a reference () to the most recently read data. │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - keepData: false, │ │ │ │ │ +OpenLayers.StyleMap = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format │ │ │ │ │ - * Instances of this class are not useful. See one of the subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object with properties to set on the │ │ │ │ │ - * format │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * keepData - {Boolean} If true, upon , the data property will be │ │ │ │ │ - * set to the parsed object (e.g. the json or xml object). │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * An instance of OpenLayers.Format │ │ │ │ │ + * Property: styles │ │ │ │ │ + * {Object} Hash of {}, keyed by names of well known │ │ │ │ │ + * rendering intents (e.g. "default", "temporary", "select", "delete"). │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.options = options; │ │ │ │ │ - }, │ │ │ │ │ + styles: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up. │ │ │ │ │ + * Property: extendDefault │ │ │ │ │ + * {Boolean} if true, every render intent will extend the symbolizers │ │ │ │ │ + * specified for the "default" intent at rendering time. Otherwise, every │ │ │ │ │ + * rendering intent will be treated as a completely independent style. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() {}, │ │ │ │ │ + extendDefault: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read │ │ │ │ │ - * Read data from a string, and return an object whose type depends on the │ │ │ │ │ - * subclass. │ │ │ │ │ + * Constructor: OpenLayers.StyleMap │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {string} Data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * Depends on the subclass │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - throw new Error('Read not implemented.'); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: write │ │ │ │ │ - * Accept an object, and return a string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * object - {Object} Object to be serialized │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string representation of the object. │ │ │ │ │ - */ │ │ │ │ │ - write: function(object) { │ │ │ │ │ - throw new Error('Write not implemented.'); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/HTTPRequest.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.HTTPRequest │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.HTTPRequest = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: URL_HASH_FACTOR │ │ │ │ │ - * {Float} Used to hash URL param strings for multi-WMS server selection. │ │ │ │ │ - * Set to the Golden Ratio per Knuth's recommendation. │ │ │ │ │ - */ │ │ │ │ │ - URL_HASH_FACTOR: (Math.sqrt(5) - 1) / 2, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: url │ │ │ │ │ - * {Array(String) or String} This is either an array of url strings or │ │ │ │ │ - * a single url string. │ │ │ │ │ - */ │ │ │ │ │ - url: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: params │ │ │ │ │ - * {Object} Hashtable of key/value parameters │ │ │ │ │ - */ │ │ │ │ │ - params: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: reproject │ │ │ │ │ - * *Deprecated*. See http://docs.openlayers.org/library/spherical_mercator.html │ │ │ │ │ - * for information on the replacement for this functionality. │ │ │ │ │ - * {Boolean} Whether layer should reproject itself based on base layer │ │ │ │ │ - * locations. This allows reprojection onto commercial layers. │ │ │ │ │ - * Default is false: Most layers can't reproject, but layers │ │ │ │ │ - * which can create non-square geographic pixels can, like WMS. │ │ │ │ │ - * │ │ │ │ │ + * style - {Object} Optional. Either a style hash, or a style object, or │ │ │ │ │ + * a hash of style objects (style hashes) keyed by rendering │ │ │ │ │ + * intent. If just one style hash or style object is passed, │ │ │ │ │ + * this will be used for all known render intents (default, │ │ │ │ │ + * select, temporary) │ │ │ │ │ + * options - {Object} optional hash of additional options for this │ │ │ │ │ + * instance │ │ │ │ │ */ │ │ │ │ │ - reproject: false, │ │ │ │ │ + initialize: function(style, options) { │ │ │ │ │ + this.styles = { │ │ │ │ │ + "default": new OpenLayers.Style( │ │ │ │ │ + OpenLayers.Feature.Vector.style["default"]), │ │ │ │ │ + "select": new OpenLayers.Style( │ │ │ │ │ + OpenLayers.Feature.Vector.style["select"]), │ │ │ │ │ + "temporary": new OpenLayers.Style( │ │ │ │ │ + OpenLayers.Feature.Vector.style["temporary"]), │ │ │ │ │ + "delete": new OpenLayers.Style( │ │ │ │ │ + OpenLayers.Feature.Vector.style["delete"]) │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.HTTPRequest │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * url - {Array(String) or String} │ │ │ │ │ - * params - {Object} │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ - this.url = url; │ │ │ │ │ - if (!this.params) { │ │ │ │ │ - this.params = OpenLayers.Util.extend({}, params); │ │ │ │ │ + // take whatever the user passed as style parameter and convert it │ │ │ │ │ + // into parts of stylemap. │ │ │ │ │ + if (style instanceof OpenLayers.Style) { │ │ │ │ │ + // user passed a style object │ │ │ │ │ + this.styles["default"] = style; │ │ │ │ │ + this.styles["select"] = style; │ │ │ │ │ + this.styles["temporary"] = style; │ │ │ │ │ + this.styles["delete"] = style; │ │ │ │ │ + } else if (typeof style == "object") { │ │ │ │ │ + for (var key in style) { │ │ │ │ │ + if (style[key] instanceof OpenLayers.Style) { │ │ │ │ │ + // user passed a hash of style objects │ │ │ │ │ + this.styles[key] = style[key]; │ │ │ │ │ + } else if (typeof style[key] == "object") { │ │ │ │ │ + // user passsed a hash of style hashes │ │ │ │ │ + this.styles[key] = new OpenLayers.Style(style[key]); │ │ │ │ │ + } else { │ │ │ │ │ + // user passed a style hash (i.e. symbolizer) │ │ │ │ │ + this.styles["default"] = new OpenLayers.Style(style); │ │ │ │ │ + this.styles["select"] = new OpenLayers.Style(style); │ │ │ │ │ + this.styles["temporary"] = new OpenLayers.Style(style); │ │ │ │ │ + this.styles["delete"] = new OpenLayers.Style(style); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ destroy: function() { │ │ │ │ │ - this.url = null; │ │ │ │ │ - this.params = null; │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ + for (var key in this.styles) { │ │ │ │ │ + this.styles[key].destroy(); │ │ │ │ │ + } │ │ │ │ │ + this.styles = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ + * Method: createSymbolizer │ │ │ │ │ + * Creates the symbolizer for a feature for a render intent. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {Object} │ │ │ │ │ + * feature - {} The feature to evaluate the rules │ │ │ │ │ + * of the intended style against. │ │ │ │ │ + * intent - {String} The intent determines the symbolizer that will be │ │ │ │ │ + * used to draw the feature. Well known intents are "default" │ │ │ │ │ + * (for just drawing the features), "select" (for selected │ │ │ │ │ + * features) and "temporary" (for drawing features). │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {} An exact clone of this │ │ │ │ │ - * │ │ │ │ │ + * {Object} symbolizer hash │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.HTTPRequest(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.params, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ + createSymbolizer: function(feature, intent) { │ │ │ │ │ + if (!feature) { │ │ │ │ │ + feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ - │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setUrl │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newUrl - {String} │ │ │ │ │ - */ │ │ │ │ │ - setUrl: function(newUrl) { │ │ │ │ │ - this.url = newUrl; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: mergeNewParams │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newParams - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * redrawn: {Boolean} whether the layer was actually redrawn. │ │ │ │ │ - */ │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - this.params = OpenLayers.Util.extend(this.params, newParams); │ │ │ │ │ - var ret = this.redraw(); │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "params" │ │ │ │ │ - }); │ │ │ │ │ + if (!this.styles[intent]) { │ │ │ │ │ + intent = "default"; │ │ │ │ │ } │ │ │ │ │ - return ret; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: redraw │ │ │ │ │ - * Redraws the layer. Returns true if the layer was redrawn, false if not. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * force - {Boolean} Force redraw by adding random parameter. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The layer was redrawn. │ │ │ │ │ - */ │ │ │ │ │ - redraw: function(force) { │ │ │ │ │ - if (force) { │ │ │ │ │ - return this.mergeNewParams({ │ │ │ │ │ - "_olSalt": Math.random() │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - return OpenLayers.Layer.prototype.redraw.apply(this, []); │ │ │ │ │ + feature.renderIntent = intent; │ │ │ │ │ + var defaultSymbolizer = {}; │ │ │ │ │ + if (this.extendDefault && intent != "default") { │ │ │ │ │ + defaultSymbolizer = this.styles["default"].createSymbolizer(feature); │ │ │ │ │ } │ │ │ │ │ + return OpenLayers.Util.extend(defaultSymbolizer, │ │ │ │ │ + this.styles[intent].createSymbolizer(feature)); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: selectUrl │ │ │ │ │ - * selectUrl() implements the standard floating-point multiplicative │ │ │ │ │ - * hash function described by Knuth, and hashes the contents of the │ │ │ │ │ - * given param string into a float between 0 and 1. This float is then │ │ │ │ │ - * scaled to the size of the provided urls array, and used to select │ │ │ │ │ - * a URL. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * paramString - {String} │ │ │ │ │ - * urls - {Array(String)} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} An entry from the urls array, deterministically selected based │ │ │ │ │ - * on the paramString. │ │ │ │ │ - */ │ │ │ │ │ - selectUrl: function(paramString, urls) { │ │ │ │ │ - var product = 1; │ │ │ │ │ - for (var i = 0, len = paramString.length; i < len; i++) { │ │ │ │ │ - product *= paramString.charCodeAt(i) * this.URL_HASH_FACTOR; │ │ │ │ │ - product -= Math.floor(product); │ │ │ │ │ - } │ │ │ │ │ - return urls[Math.floor(product * urls.length)]; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getFullRequestString │ │ │ │ │ - * Combine url with layer's params and these newParams. │ │ │ │ │ - * │ │ │ │ │ - * does checking on the serverPath variable, allowing for cases when it │ │ │ │ │ - * is supplied with trailing ? or &, as well as cases where not. │ │ │ │ │ - * │ │ │ │ │ - * return in formatted string like this: │ │ │ │ │ - * "server?key1=value1&key2=value2&key3=value3" │ │ │ │ │ + * Method: addUniqueValueRules │ │ │ │ │ + * Convenience method to create comparison rules for unique values of a │ │ │ │ │ + * property. The rules will be added to the style object for a specified │ │ │ │ │ + * rendering intent. This method is a shortcut for creating something like │ │ │ │ │ + * the "unique value legends" familiar from well known desktop GIS systems │ │ │ │ │ * │ │ │ │ │ - * WARNING: The altUrl parameter is deprecated and will be removed in 3.0. │ │ │ │ │ - * │ │ │ │ │ * Parameters: │ │ │ │ │ - * newParams - {Object} │ │ │ │ │ - * altUrl - {String} Use this as the url instead of the layer's url │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} │ │ │ │ │ + * renderIntent - {String} rendering intent to add the rules to │ │ │ │ │ + * property - {String} values of feature attributes to create the │ │ │ │ │ + * rules for │ │ │ │ │ + * symbolizers - {Object} Hash of symbolizers, keyed by the desired │ │ │ │ │ + * property values │ │ │ │ │ + * context - {Object} An optional object with properties that │ │ │ │ │ + * symbolizers' property values should be evaluated │ │ │ │ │ + * against. If no context is specified, feature.attributes │ │ │ │ │ + * will be used │ │ │ │ │ */ │ │ │ │ │ - getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ - │ │ │ │ │ - // if not altUrl passed in, use layer's url │ │ │ │ │ - var url = altUrl || this.url; │ │ │ │ │ - │ │ │ │ │ - // create a new params hashtable with all the layer params and the │ │ │ │ │ - // new params together. then convert to string │ │ │ │ │ - var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ - allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ - var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ - │ │ │ │ │ - // if url is not a string, it should be an array of strings, │ │ │ │ │ - // in which case we will deterministically select one of them in │ │ │ │ │ - // order to evenly distribute requests to different urls. │ │ │ │ │ - // │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(paramsString, url); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // ignore parameters that are already in the url search string │ │ │ │ │ - var urlParams = │ │ │ │ │ - OpenLayers.Util.upperCaseObject(OpenLayers.Util.getParameters(url)); │ │ │ │ │ - for (var key in allParams) { │ │ │ │ │ - if (key.toUpperCase() in urlParams) { │ │ │ │ │ - delete allParams[key]; │ │ │ │ │ - } │ │ │ │ │ + addUniqueValueRules: function(renderIntent, property, symbolizers, context) { │ │ │ │ │ + var rules = []; │ │ │ │ │ + for (var value in symbolizers) { │ │ │ │ │ + rules.push(new OpenLayers.Rule({ │ │ │ │ │ + symbolizer: symbolizers[value], │ │ │ │ │ + context: context, │ │ │ │ │ + filter: new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.EQUAL_TO, │ │ │ │ │ + property: property, │ │ │ │ │ + value: value │ │ │ │ │ + }) │ │ │ │ │ + })); │ │ │ │ │ } │ │ │ │ │ - paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ - │ │ │ │ │ - return OpenLayers.Util.urlAppend(url, paramsString); │ │ │ │ │ + this.styles[renderIntent].addRules(rules); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.HTTPRequest" │ │ │ │ │ + CLASS_NAME: "OpenLayers.StyleMap" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Tile.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ @@ -15803,3091 +11559,1214 @@ │ │ │ │ │ clear: function(draw) { │ │ │ │ │ // to be extended by subclasses │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Tile" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Tile/Image.js │ │ │ │ │ + OpenLayers/Format.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Tile.js │ │ │ │ │ - * @requires OpenLayers/Animation.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ * @requires OpenLayers/Util.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Tile.Image │ │ │ │ │ - * Instances of OpenLayers.Tile.Image are used to manage the image tiles │ │ │ │ │ - * used by various layers. Create a new image tile with the │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ + * Class: OpenLayers.Format │ │ │ │ │ + * Base class for format reading/writing a variety of formats. Subclasses │ │ │ │ │ + * of OpenLayers.Format are expected to have read and write methods. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Tile.Image = OpenLayers.Class(OpenLayers.Tile, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {} An events object that handles all │ │ │ │ │ - * events on the tile. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * tile.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Supported event types (in addition to the events): │ │ │ │ │ - * beforeload - Triggered before an image is prepared for loading, when the │ │ │ │ │ - * url for the image is known already. Listeners may call on │ │ │ │ │ - * the tile instance. If they do so, that image will be used and no new │ │ │ │ │ - * one will be created. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String} The URL of the image being requested. No default. Filled in by │ │ │ │ │ - * layer.getURL() function. May be modified by loadstart listeners. │ │ │ │ │ - */ │ │ │ │ │ - url: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: imgDiv │ │ │ │ │ - * {HTMLImageElement} The image for this tile. │ │ │ │ │ - */ │ │ │ │ │ - imgDiv: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: frame │ │ │ │ │ - * {DOMElement} The image element is appended to the frame. Any gutter on │ │ │ │ │ - * the image will be hidden behind the frame. If no gutter is set, │ │ │ │ │ - * this will be null. │ │ │ │ │ - */ │ │ │ │ │ - frame: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: imageReloadAttempts │ │ │ │ │ - * {Integer} Attempts to load the image. │ │ │ │ │ - */ │ │ │ │ │ - imageReloadAttempts: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layerAlphaHack │ │ │ │ │ - * {Boolean} True if the png alpha hack needs to be applied on the layer's div. │ │ │ │ │ - */ │ │ │ │ │ - layerAlphaHack: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: asyncRequestId │ │ │ │ │ - * {Integer} ID of an request to see if request is still valid. This is a │ │ │ │ │ - * number which increments by 1 for each asynchronous request. │ │ │ │ │ - */ │ │ │ │ │ - asyncRequestId: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: maxGetUrlLength │ │ │ │ │ - * {Number} If set, requests that would result in GET urls with more │ │ │ │ │ - * characters than the number provided will be made using form-encoded │ │ │ │ │ - * HTTP POST. It is good practice to avoid urls that are longer than 2048 │ │ │ │ │ - * characters. │ │ │ │ │ - * │ │ │ │ │ - * Caution: │ │ │ │ │ - * Older versions of Gecko based browsers (e.g. Firefox < 3.5) and most │ │ │ │ │ - * Opera versions do not fully support this option. On all browsers, │ │ │ │ │ - * transition effects are not supported if POST requests are used. │ │ │ │ │ - */ │ │ │ │ │ - maxGetUrlLength: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: canvasContext │ │ │ │ │ - * {CanvasRenderingContext2D} A canvas context associated with │ │ │ │ │ - * the tile image. │ │ │ │ │ - */ │ │ │ │ │ - canvasContext: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: crossOriginKeyword │ │ │ │ │ - * The value of the crossorigin keyword to use when loading images. This is │ │ │ │ │ - * only relevant when using for tiles from remote │ │ │ │ │ - * origins and should be set to either 'anonymous' or 'use-credentials' │ │ │ │ │ - * for servers that send Access-Control-Allow-Origin headers with their │ │ │ │ │ - * tiles. │ │ │ │ │ - */ │ │ │ │ │ - crossOriginKeyword: null, │ │ │ │ │ - │ │ │ │ │ - /** TBD 3.0 - reorder the parameters to the init function to remove │ │ │ │ │ - * URL. the getUrl() function on the layer gets called on │ │ │ │ │ - * each draw(), so no need to specify it here. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Tile.Image │ │ │ │ │ - * Constructor for a new instance. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {} layer that the tile will go in. │ │ │ │ │ - * position - {} │ │ │ │ │ - * bounds - {} │ │ │ │ │ - * url - {} Deprecated. Remove me in 3.0. │ │ │ │ │ - * size - {} │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(layer, position, bounds, url, size, options) { │ │ │ │ │ - OpenLayers.Tile.prototype.initialize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - this.url = url; //deprecated remove me │ │ │ │ │ - │ │ │ │ │ - this.layerAlphaHack = this.layer.alpha && OpenLayers.Util.alphaHack(); │ │ │ │ │ - │ │ │ │ │ - if (this.maxGetUrlLength != null || this.layer.gutter || this.layerAlphaHack) { │ │ │ │ │ - // only create frame if it's needed │ │ │ │ │ - this.frame = document.createElement("div"); │ │ │ │ │ - this.frame.style.position = "absolute"; │ │ │ │ │ - this.frame.style.overflow = "hidden"; │ │ │ │ │ - } │ │ │ │ │ - if (this.maxGetUrlLength != null) { │ │ │ │ │ - OpenLayers.Util.extend(this, OpenLayers.Tile.Image.IFrame); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * nullify references to prevent circular references and memory leaks │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.imgDiv) { │ │ │ │ │ - this.clear(); │ │ │ │ │ - this.imgDiv = null; │ │ │ │ │ - this.frame = null; │ │ │ │ │ - } │ │ │ │ │ - // don't handle async requests any more │ │ │ │ │ - this.asyncRequestId = null; │ │ │ │ │ - OpenLayers.Tile.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Check that a tile should be drawn, and draw it. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Was a tile drawn? Or null if a beforedraw listener returned │ │ │ │ │ - * false. │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - var shouldDraw = OpenLayers.Tile.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (shouldDraw) { │ │ │ │ │ - // The layer's reproject option is deprecated. │ │ │ │ │ - if (this.layer != this.layer.map.baseLayer && this.layer.reproject) { │ │ │ │ │ - // getBoundsFromBaseLayer is defined in deprecated.js. │ │ │ │ │ - this.bounds = this.getBoundsFromBaseLayer(this.position); │ │ │ │ │ - } │ │ │ │ │ - if (this.isLoading) { │ │ │ │ │ - //if we're already loading, send 'reload' instead of 'loadstart'. │ │ │ │ │ - this._loadEvent = "reload"; │ │ │ │ │ - } else { │ │ │ │ │ - this.isLoading = true; │ │ │ │ │ - this._loadEvent = "loadstart"; │ │ │ │ │ - } │ │ │ │ │ - this.renderTile(); │ │ │ │ │ - this.positionTile(); │ │ │ │ │ - } else if (shouldDraw === false) { │ │ │ │ │ - this.unload(); │ │ │ │ │ - } │ │ │ │ │ - return shouldDraw; │ │ │ │ │ - }, │ │ │ │ │ +OpenLayers.Format = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: renderTile │ │ │ │ │ - * Internal function to actually initialize the image tile, │ │ │ │ │ - * position it correctly, and set its url. │ │ │ │ │ + * Property: options │ │ │ │ │ + * {Object} A reference to options passed to the constructor. │ │ │ │ │ */ │ │ │ │ │ - renderTile: function() { │ │ │ │ │ - if (this.layer.async) { │ │ │ │ │ - // Asynchronous image requests call the asynchronous getURL method │ │ │ │ │ - // on the layer to fetch an image that covers 'this.bounds'. │ │ │ │ │ - var id = this.asyncRequestId = (this.asyncRequestId || 0) + 1; │ │ │ │ │ - this.layer.getURLasync(this.bounds, function(url) { │ │ │ │ │ - if (id == this.asyncRequestId) { │ │ │ │ │ - this.url = url; │ │ │ │ │ - this.initImage(); │ │ │ │ │ - } │ │ │ │ │ - }, this); │ │ │ │ │ - } else { │ │ │ │ │ - // synchronous image requests get the url immediately. │ │ │ │ │ - this.url = this.layer.getURL(this.bounds); │ │ │ │ │ - this.initImage(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + options: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: positionTile │ │ │ │ │ - * Using the properties currenty set on the layer, position the tile correctly. │ │ │ │ │ - * This method is used both by the async and non-async versions of the Tile.Image │ │ │ │ │ - * code. │ │ │ │ │ - */ │ │ │ │ │ - positionTile: function() { │ │ │ │ │ - var style = this.getTile().style, │ │ │ │ │ - size = this.frame ? this.size : │ │ │ │ │ - this.layer.getImageSize(this.bounds), │ │ │ │ │ - ratio = 1; │ │ │ │ │ - if (this.layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ - ratio = this.layer.getServerResolution() / this.layer.map.getResolution(); │ │ │ │ │ - } │ │ │ │ │ - style.left = this.position.x + "px"; │ │ │ │ │ - style.top = this.position.y + "px"; │ │ │ │ │ - style.width = Math.round(ratio * size.w) + "px"; │ │ │ │ │ - style.height = Math.round(ratio * size.h) + "px"; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clear │ │ │ │ │ - * Remove the tile from the DOM, clear it of any image related data so that │ │ │ │ │ - * it can be reused in a new location. │ │ │ │ │ + * APIProperty: externalProjection │ │ │ │ │ + * {} When passed a externalProjection and │ │ │ │ │ + * internalProjection, the format will reproject the geometries it │ │ │ │ │ + * reads or writes. The externalProjection is the projection used by │ │ │ │ │ + * the content which is passed into read or which comes out of write. │ │ │ │ │ + * In order to reproject, a projection transformation function for the │ │ │ │ │ + * specified projections must be available. This support may be │ │ │ │ │ + * provided via proj4js or via a custom transformation function. See │ │ │ │ │ + * {} for more information on │ │ │ │ │ + * custom transformations. │ │ │ │ │ */ │ │ │ │ │ - clear: function() { │ │ │ │ │ - OpenLayers.Tile.prototype.clear.apply(this, arguments); │ │ │ │ │ - var img = this.imgDiv; │ │ │ │ │ - if (img) { │ │ │ │ │ - var tile = this.getTile(); │ │ │ │ │ - if (tile.parentNode === this.layer.div) { │ │ │ │ │ - this.layer.div.removeChild(tile); │ │ │ │ │ - } │ │ │ │ │ - this.setImgSrc(); │ │ │ │ │ - if (this.layerAlphaHack === true) { │ │ │ │ │ - img.style.filter = ""; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Element.removeClass(img, "olImageLoadError"); │ │ │ │ │ - } │ │ │ │ │ - this.canvasContext = null; │ │ │ │ │ - }, │ │ │ │ │ + externalProjection: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getImage │ │ │ │ │ - * Returns or creates and returns the tile image. │ │ │ │ │ + * APIProperty: internalProjection │ │ │ │ │ + * {} When passed a externalProjection and │ │ │ │ │ + * internalProjection, the format will reproject the geometries it │ │ │ │ │ + * reads or writes. The internalProjection is the projection used by │ │ │ │ │ + * the geometries which are returned by read or which are passed into │ │ │ │ │ + * write. In order to reproject, a projection transformation function │ │ │ │ │ + * for the specified projections must be available. This support may be │ │ │ │ │ + * provided via proj4js or via a custom transformation function. See │ │ │ │ │ + * {} for more information on │ │ │ │ │ + * custom transformations. │ │ │ │ │ */ │ │ │ │ │ - getImage: function() { │ │ │ │ │ - if (!this.imgDiv) { │ │ │ │ │ - this.imgDiv = OpenLayers.Tile.Image.IMAGE.cloneNode(false); │ │ │ │ │ - │ │ │ │ │ - var style = this.imgDiv.style; │ │ │ │ │ - if (this.frame) { │ │ │ │ │ - var left = 0, │ │ │ │ │ - top = 0; │ │ │ │ │ - if (this.layer.gutter) { │ │ │ │ │ - left = this.layer.gutter / this.layer.tileSize.w * 100; │ │ │ │ │ - top = this.layer.gutter / this.layer.tileSize.h * 100; │ │ │ │ │ - } │ │ │ │ │ - style.left = -left + "%"; │ │ │ │ │ - style.top = -top + "%"; │ │ │ │ │ - style.width = (2 * left + 100) + "%"; │ │ │ │ │ - style.height = (2 * top + 100) + "%"; │ │ │ │ │ - } │ │ │ │ │ - style.visibility = "hidden"; │ │ │ │ │ - style.opacity = 0; │ │ │ │ │ - if (this.layer.opacity < 1) { │ │ │ │ │ - style.filter = 'alpha(opacity=' + │ │ │ │ │ - (this.layer.opacity * 100) + │ │ │ │ │ - ')'; │ │ │ │ │ - } │ │ │ │ │ - style.position = "absolute"; │ │ │ │ │ - if (this.layerAlphaHack) { │ │ │ │ │ - // move the image out of sight │ │ │ │ │ - style.paddingTop = style.height; │ │ │ │ │ - style.height = "0"; │ │ │ │ │ - style.width = "100%"; │ │ │ │ │ - } │ │ │ │ │ - if (this.frame) { │ │ │ │ │ - this.frame.appendChild(this.imgDiv); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return this.imgDiv; │ │ │ │ │ - }, │ │ │ │ │ + internalProjection: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setImage │ │ │ │ │ - * Sets the image element for this tile. This method should only be called │ │ │ │ │ - * from beforeload listeners. │ │ │ │ │ - * │ │ │ │ │ - * Parameters │ │ │ │ │ - * img - {HTMLImageElement} The image to use for this tile. │ │ │ │ │ + * APIProperty: data │ │ │ │ │ + * {Object} When is true, this is the parsed string sent to │ │ │ │ │ + * . │ │ │ │ │ */ │ │ │ │ │ - setImage: function(img) { │ │ │ │ │ - this.imgDiv = img; │ │ │ │ │ - }, │ │ │ │ │ + data: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: initImage │ │ │ │ │ - * Creates the content for the frame on the tile. │ │ │ │ │ + * APIProperty: keepData │ │ │ │ │ + * {Object} Maintain a reference () to the most recently read data. │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - initImage: function() { │ │ │ │ │ - if (!this.url && !this.imgDiv) { │ │ │ │ │ - // fast path out - if there is no tile url and no previous image │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent('beforeload'); │ │ │ │ │ - this.layer.div.appendChild(this.getTile()); │ │ │ │ │ - this.events.triggerEvent(this._loadEvent); │ │ │ │ │ - var img = this.getImage(); │ │ │ │ │ - var src = img.getAttribute('src') || ''; │ │ │ │ │ - if (this.url && OpenLayers.Util.isEquivalentUrl(src, this.url)) { │ │ │ │ │ - this._loadTimeout = window.setTimeout( │ │ │ │ │ - OpenLayers.Function.bind(this.onImageLoad, this), 0 │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - this.stopLoading(); │ │ │ │ │ - if (this.crossOriginKeyword) { │ │ │ │ │ - img.removeAttribute("crossorigin"); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.observe(img, "load", │ │ │ │ │ - OpenLayers.Function.bind(this.onImageLoad, this) │ │ │ │ │ - ); │ │ │ │ │ - OpenLayers.Event.observe(img, "error", │ │ │ │ │ - OpenLayers.Function.bind(this.onImageError, this) │ │ │ │ │ - ); │ │ │ │ │ - this.imageReloadAttempts = 0; │ │ │ │ │ - this.setImgSrc(this.url); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + keepData: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setImgSrc │ │ │ │ │ - * Sets the source for the tile image │ │ │ │ │ + * Constructor: OpenLayers.Format │ │ │ │ │ + * Instances of this class are not useful. See one of the subclasses. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * url - {String} or undefined to hide the image │ │ │ │ │ - */ │ │ │ │ │ - setImgSrc: function(url) { │ │ │ │ │ - var img = this.imgDiv; │ │ │ │ │ - if (url) { │ │ │ │ │ - img.style.visibility = 'hidden'; │ │ │ │ │ - img.style.opacity = 0; │ │ │ │ │ - // don't set crossOrigin if the url is a data URL │ │ │ │ │ - if (this.crossOriginKeyword) { │ │ │ │ │ - if (url.substr(0, 5) !== 'data:') { │ │ │ │ │ - img.setAttribute("crossorigin", this.crossOriginKeyword); │ │ │ │ │ - } else { │ │ │ │ │ - img.removeAttribute("crossorigin"); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - img.src = url; │ │ │ │ │ - } else { │ │ │ │ │ - // Remove reference to the image, and leave it to the browser's │ │ │ │ │ - // caching and garbage collection. │ │ │ │ │ - this.stopLoading(); │ │ │ │ │ - this.imgDiv = null; │ │ │ │ │ - if (img.parentNode) { │ │ │ │ │ - img.parentNode.removeChild(img); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getTile │ │ │ │ │ - * Get the tile's markup. │ │ │ │ │ + * options - {Object} An optional object with properties to set on the │ │ │ │ │ + * format │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The tile's markup │ │ │ │ │ - */ │ │ │ │ │ - getTile: function() { │ │ │ │ │ - return this.frame ? this.frame : this.getImage(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createBackBuffer │ │ │ │ │ - * Create a backbuffer for this tile. A backbuffer isn't exactly a clone │ │ │ │ │ - * of the tile's markup, because we want to avoid the reloading of the │ │ │ │ │ - * image. So we clone the frame, and steal the image from the tile. │ │ │ │ │ + * Valid options: │ │ │ │ │ + * keepData - {Boolean} If true, upon , the data property will be │ │ │ │ │ + * set to the parsed object (e.g. the json or xml object). │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} The markup, or undefined if the tile has no image │ │ │ │ │ - * or if it's currently loading. │ │ │ │ │ - */ │ │ │ │ │ - createBackBuffer: function() { │ │ │ │ │ - if (!this.imgDiv || this.isLoading) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var backBuffer; │ │ │ │ │ - if (this.frame) { │ │ │ │ │ - backBuffer = this.frame.cloneNode(false); │ │ │ │ │ - backBuffer.appendChild(this.imgDiv); │ │ │ │ │ - } else { │ │ │ │ │ - backBuffer = this.imgDiv; │ │ │ │ │ - } │ │ │ │ │ - this.imgDiv = null; │ │ │ │ │ - return backBuffer; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onImageLoad │ │ │ │ │ - * Handler for the image onload event │ │ │ │ │ + * An instance of OpenLayers.Format │ │ │ │ │ */ │ │ │ │ │ - onImageLoad: function() { │ │ │ │ │ - var img = this.imgDiv; │ │ │ │ │ - this.stopLoading(); │ │ │ │ │ - img.style.visibility = 'inherit'; │ │ │ │ │ - img.style.opacity = this.layer.opacity; │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - this.canvasContext = null; │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ - │ │ │ │ │ - if (this.layerAlphaHack === true) { │ │ │ │ │ - img.style.filter = │ │ │ │ │ - "progid:DXImageTransform.Microsoft.AlphaImageLoader(src='" + │ │ │ │ │ - img.src + "', sizingMethod='scale')"; │ │ │ │ │ - } │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.options = options; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onImageError │ │ │ │ │ - * Handler for the image onerror event │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up. │ │ │ │ │ */ │ │ │ │ │ - onImageError: function() { │ │ │ │ │ - var img = this.imgDiv; │ │ │ │ │ - if (img.src != null) { │ │ │ │ │ - this.imageReloadAttempts++; │ │ │ │ │ - if (this.imageReloadAttempts <= OpenLayers.IMAGE_RELOAD_ATTEMPTS) { │ │ │ │ │ - this.setImgSrc(this.layer.getURL(this.bounds)); │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Element.addClass(img, "olImageLoadError"); │ │ │ │ │ - this.events.triggerEvent("loaderror"); │ │ │ │ │ - this.onImageLoad(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + destroy: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: stopLoading │ │ │ │ │ - * Stops a loading sequence so won't be executed. │ │ │ │ │ + * Method: read │ │ │ │ │ + * Read data from a string, and return an object whose type depends on the │ │ │ │ │ + * subclass. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {string} Data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * Depends on the subclass │ │ │ │ │ */ │ │ │ │ │ - stopLoading: function() { │ │ │ │ │ - OpenLayers.Event.stopObservingElement(this.imgDiv); │ │ │ │ │ - window.clearTimeout(this._loadTimeout); │ │ │ │ │ - delete this._loadTimeout; │ │ │ │ │ + read: function(data) { │ │ │ │ │ + throw new Error('Read not implemented.'); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getCanvasContext │ │ │ │ │ - * Returns a canvas context associated with the tile image (with │ │ │ │ │ - * the image drawn on it). │ │ │ │ │ - * Returns undefined if the browser does not support canvas, if │ │ │ │ │ - * the tile has no image or if it's currently loading. │ │ │ │ │ + * Method: write │ │ │ │ │ + * Accept an object, and return a string. │ │ │ │ │ * │ │ │ │ │ - * The function returns a canvas context instance but the │ │ │ │ │ - * underlying canvas is still available in the 'canvas' property: │ │ │ │ │ - * (code) │ │ │ │ │ - * var context = tile.getCanvasContext(); │ │ │ │ │ - * if (context) { │ │ │ │ │ - * var data = context.canvas.toDataURL('image/jpeg'); │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ + * Parameters: │ │ │ │ │ + * object - {Object} Object to be serialized │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * {String} A string representation of the object. │ │ │ │ │ */ │ │ │ │ │ - getCanvasContext: function() { │ │ │ │ │ - if (OpenLayers.CANVAS_SUPPORTED && this.imgDiv && !this.isLoading) { │ │ │ │ │ - if (!this.canvasContext) { │ │ │ │ │ - var canvas = document.createElement("canvas"); │ │ │ │ │ - canvas.width = this.size.w; │ │ │ │ │ - canvas.height = this.size.h; │ │ │ │ │ - this.canvasContext = canvas.getContext("2d"); │ │ │ │ │ - this.canvasContext.drawImage(this.imgDiv, 0, 0); │ │ │ │ │ - } │ │ │ │ │ - return this.canvasContext; │ │ │ │ │ - } │ │ │ │ │ + write: function(object) { │ │ │ │ │ + throw new Error('Write not implemented.'); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Tile.Image" │ │ │ │ │ - │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Tile.Image.IMAGE │ │ │ │ │ - * {HTMLImageElement} The image for a tile. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Tile.Image.IMAGE = (function() { │ │ │ │ │ - var img = new Image(); │ │ │ │ │ - img.className = "olTileImage"; │ │ │ │ │ - // avoid image gallery menu in IE6 │ │ │ │ │ - img.galleryImg = "no"; │ │ │ │ │ - return img; │ │ │ │ │ -}()); │ │ │ │ │ - │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Grid.js │ │ │ │ │ + OpenLayers/Popup.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/HTTPRequest.js │ │ │ │ │ - * @requires OpenLayers/Tile/Image.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.Grid │ │ │ │ │ - * Base class for layers that use a lattice of tiles. Create a new grid │ │ │ │ │ - * layer with the constructor. │ │ │ │ │ + * Class: OpenLayers.Popup │ │ │ │ │ + * A popup is a small div that can opened and closed on the map. │ │ │ │ │ + * Typically opened in response to clicking on a marker. │ │ │ │ │ + * See . Popup's don't require their own │ │ │ │ │ + * layer and are added the the map using the │ │ │ │ │ + * method. │ │ │ │ │ * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * popup = new OpenLayers.Popup("chicken", │ │ │ │ │ + * new OpenLayers.LonLat(5,40), │ │ │ │ │ + * new OpenLayers.Size(200,200), │ │ │ │ │ + * "example popup", │ │ │ │ │ + * true); │ │ │ │ │ + * │ │ │ │ │ + * map.addPopup(popup); │ │ │ │ │ + * (end) │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Grid = OpenLayers.Class(OpenLayers.Layer.HTTPRequest, { │ │ │ │ │ +OpenLayers.Popup = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: tileSize │ │ │ │ │ - * {} │ │ │ │ │ + /** │ │ │ │ │ + * Property: events │ │ │ │ │ + * {} custom event manager │ │ │ │ │ */ │ │ │ │ │ - tileSize: null, │ │ │ │ │ + events: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: tileOriginCorner │ │ │ │ │ - * {String} If the property is not provided, the tile origin │ │ │ │ │ - * will be derived from the layer's . The corner of the │ │ │ │ │ - * used is determined by this property. Acceptable values │ │ │ │ │ - * are "tl" (top left), "tr" (top right), "bl" (bottom left), and "br" │ │ │ │ │ - * (bottom right). Default is "bl". │ │ │ │ │ + /** Property: id │ │ │ │ │ + * {String} the unique identifier assigned to this popup. │ │ │ │ │ */ │ │ │ │ │ - tileOriginCorner: "bl", │ │ │ │ │ + id: "", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: tileOrigin │ │ │ │ │ - * {} Optional origin for aligning the grid of tiles. │ │ │ │ │ - * If provided, requests for tiles at all resolutions will be aligned │ │ │ │ │ - * with this location (no tiles shall overlap this location). If │ │ │ │ │ - * not provided, the grid of tiles will be aligned with the layer's │ │ │ │ │ - * . Default is ``null``. │ │ │ │ │ + /** │ │ │ │ │ + * Property: lonlat │ │ │ │ │ + * {} the position of this popup on the map │ │ │ │ │ */ │ │ │ │ │ - tileOrigin: null, │ │ │ │ │ + lonlat: null, │ │ │ │ │ │ │ │ │ │ - /** APIProperty: tileOptions │ │ │ │ │ - * {Object} optional configuration options for instances │ │ │ │ │ - * created by this Layer, if supported by the tile class. │ │ │ │ │ + /** │ │ │ │ │ + * Property: div │ │ │ │ │ + * {DOMElement} the div that contains this popup. │ │ │ │ │ */ │ │ │ │ │ - tileOptions: null, │ │ │ │ │ + div: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: tileClass │ │ │ │ │ - * {} The tile class to use for this layer. │ │ │ │ │ - * Defaults is OpenLayers.Tile.Image. │ │ │ │ │ + /** │ │ │ │ │ + * Property: contentSize │ │ │ │ │ + * {} the width and height of the content. │ │ │ │ │ */ │ │ │ │ │ - tileClass: OpenLayers.Tile.Image, │ │ │ │ │ + contentSize: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: grid │ │ │ │ │ - * {Array(Array())} This is an array of rows, each row is │ │ │ │ │ - * an array of tiles. │ │ │ │ │ + /** │ │ │ │ │ + * Property: size │ │ │ │ │ + * {} the width and height of the popup. │ │ │ │ │ */ │ │ │ │ │ - grid: null, │ │ │ │ │ + size: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: singleTile │ │ │ │ │ - * {Boolean} Moves the layer into single-tile mode, meaning that one tile │ │ │ │ │ - * will be loaded. The tile's size will be determined by the 'ratio' │ │ │ │ │ - * property. When the tile is dragged such that it does not cover the │ │ │ │ │ - * entire viewport, it is reloaded. │ │ │ │ │ + /** │ │ │ │ │ + * Property: contentHTML │ │ │ │ │ + * {String} An HTML string for this popup to display. │ │ │ │ │ */ │ │ │ │ │ - singleTile: false, │ │ │ │ │ + contentHTML: null, │ │ │ │ │ │ │ │ │ │ - /** APIProperty: ratio │ │ │ │ │ - * {Float} Used only when in single-tile mode, this specifies the │ │ │ │ │ - * ratio of the size of the single tile to the size of the map. │ │ │ │ │ - * Default value is 1.5. │ │ │ │ │ + /** │ │ │ │ │ + * Property: backgroundColor │ │ │ │ │ + * {String} the background color used by the popup. │ │ │ │ │ */ │ │ │ │ │ - ratio: 1.5, │ │ │ │ │ + backgroundColor: "", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: buffer │ │ │ │ │ - * {Integer} Used only when in gridded mode, this specifies the number of │ │ │ │ │ - * extra rows and colums of tiles on each side which will │ │ │ │ │ - * surround the minimum grid tiles to cover the map. │ │ │ │ │ - * For very slow loading layers, a larger value may increase │ │ │ │ │ - * performance somewhat when dragging, but will increase bandwidth │ │ │ │ │ - * use significantly. │ │ │ │ │ + /** │ │ │ │ │ + * Property: opacity │ │ │ │ │ + * {float} the opacity of this popup (between 0.0 and 1.0) │ │ │ │ │ */ │ │ │ │ │ - buffer: 0, │ │ │ │ │ + opacity: "", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: transitionEffect │ │ │ │ │ - * {String} The transition effect to use when the map is zoomed. │ │ │ │ │ - * Two posible values: │ │ │ │ │ - * │ │ │ │ │ - * "resize" - Existing tiles are resized on zoom to provide a visual │ │ │ │ │ - * effect of the zoom having taken place immediately. As the │ │ │ │ │ - * new tiles become available, they are drawn on top of the │ │ │ │ │ - * resized tiles (this is the default setting). │ │ │ │ │ - * "map-resize" - Existing tiles are resized on zoom and placed below the │ │ │ │ │ - * base layer. New tiles for the base layer will cover existing tiles. │ │ │ │ │ - * This setting is recommended when having an overlay duplicated during │ │ │ │ │ - * the transition is undesirable (e.g. street labels or big transparent │ │ │ │ │ - * fills). │ │ │ │ │ - * null - No transition effect. │ │ │ │ │ - * │ │ │ │ │ - * Using "resize" on non-opaque layers can cause undesired visual │ │ │ │ │ - * effects. Set transitionEffect to null in this case. │ │ │ │ │ + /** │ │ │ │ │ + * Property: border │ │ │ │ │ + * {String} the border size of the popup. (eg 2px) │ │ │ │ │ */ │ │ │ │ │ - transitionEffect: "resize", │ │ │ │ │ + border: "", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: numLoadingTiles │ │ │ │ │ - * {Integer} How many tiles are still loading? │ │ │ │ │ + /** │ │ │ │ │ + * Property: contentDiv │ │ │ │ │ + * {DOMElement} a reference to the element that holds the content of │ │ │ │ │ + * the div. │ │ │ │ │ */ │ │ │ │ │ - numLoadingTiles: 0, │ │ │ │ │ + contentDiv: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: serverResolutions │ │ │ │ │ - * {Array(Number}} This property is documented in subclasses as │ │ │ │ │ - * an API property. │ │ │ │ │ + /** │ │ │ │ │ + * Property: groupDiv │ │ │ │ │ + * {DOMElement} First and only child of 'div'. The group Div contains the │ │ │ │ │ + * 'contentDiv' and the 'closeDiv'. │ │ │ │ │ */ │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ + groupDiv: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: loading │ │ │ │ │ - * {Boolean} Indicates if tiles are being loaded. │ │ │ │ │ + /** │ │ │ │ │ + * Property: closeDiv │ │ │ │ │ + * {DOMElement} the optional closer image │ │ │ │ │ */ │ │ │ │ │ - loading: false, │ │ │ │ │ + closeDiv: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: backBuffer │ │ │ │ │ - * {DOMElement} The back buffer. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: autoSize │ │ │ │ │ + * {Boolean} Resize the popup to auto-fit the contents. │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - backBuffer: null, │ │ │ │ │ + autoSize: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: gridResolution │ │ │ │ │ - * {Number} The resolution of the current grid. Used for backbuffer and │ │ │ │ │ - * client zoom. This property is updated every time the grid is │ │ │ │ │ - * initialized. │ │ │ │ │ + * APIProperty: minSize │ │ │ │ │ + * {} Minimum size allowed for the popup's contents. │ │ │ │ │ */ │ │ │ │ │ - gridResolution: null, │ │ │ │ │ + minSize: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: backBufferResolution │ │ │ │ │ - * {Number} The resolution of the current back buffer. This property is │ │ │ │ │ - * updated each time a back buffer is created. │ │ │ │ │ + * APIProperty: maxSize │ │ │ │ │ + * {} Maximum size allowed for the popup's contents. │ │ │ │ │ */ │ │ │ │ │ - backBufferResolution: null, │ │ │ │ │ + maxSize: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: backBufferLonLat │ │ │ │ │ - * {Object} The top-left corner of the current back buffer. Includes lon │ │ │ │ │ - * and lat properties. This object is updated each time a back buffer │ │ │ │ │ - * is created. │ │ │ │ │ + /** │ │ │ │ │ + * Property: displayClass │ │ │ │ │ + * {String} The CSS class of the popup. │ │ │ │ │ */ │ │ │ │ │ - backBufferLonLat: null, │ │ │ │ │ + displayClass: "olPopup", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: backBufferTimerId │ │ │ │ │ - * {Number} The id of the back buffer timer. This timer is used to │ │ │ │ │ - * delay the removal of the back buffer, thereby preventing │ │ │ │ │ - * flash effects caused by tile animation. │ │ │ │ │ + /** │ │ │ │ │ + * Property: contentDisplayClass │ │ │ │ │ + * {String} The CSS class of the popup content div. │ │ │ │ │ */ │ │ │ │ │ - backBufferTimerId: null, │ │ │ │ │ + contentDisplayClass: "olPopupContent", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: removeBackBufferDelay │ │ │ │ │ - * {Number} Delay for removing the backbuffer when all tiles have finished │ │ │ │ │ - * loading. Can be set to 0 when no css opacity transitions for the │ │ │ │ │ - * olTileImage class are used. Default is 0 for layers, │ │ │ │ │ - * 2500 for tiled layers. See for more information on │ │ │ │ │ - * tile animation. │ │ │ │ │ + /** │ │ │ │ │ + * Property: padding │ │ │ │ │ + * {int or } An extra opportunity to specify internal │ │ │ │ │ + * padding of the content div inside the popup. This was originally │ │ │ │ │ + * confused with the css padding as specified in style.css's │ │ │ │ │ + * 'olPopupContent' class. We would like to get rid of this altogether, │ │ │ │ │ + * except that it does come in handy for the framed and anchoredbubble │ │ │ │ │ + * popups, who need to maintain yet another barrier between their │ │ │ │ │ + * content and the outer border of the popup itself. │ │ │ │ │ + * │ │ │ │ │ + * Note that in order to not break API, we must continue to support │ │ │ │ │ + * this property being set as an integer. Really, though, we'd like to │ │ │ │ │ + * have this specified as a Bounds object so that user can specify │ │ │ │ │ + * distinct left, top, right, bottom paddings. With the 3.0 release │ │ │ │ │ + * we can make this only a bounds. │ │ │ │ │ */ │ │ │ │ │ - removeBackBufferDelay: null, │ │ │ │ │ + padding: 0, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: className │ │ │ │ │ - * {String} Name of the class added to the layer div. If not set in the │ │ │ │ │ - * options passed to the constructor then className defaults to │ │ │ │ │ - * "olLayerGridSingleTile" for single tile layers (see ), │ │ │ │ │ - * and "olLayerGrid" for non single tile layers. │ │ │ │ │ - * │ │ │ │ │ - * Note: │ │ │ │ │ - * │ │ │ │ │ - * The displaying of tiles is not animated by default for single tile │ │ │ │ │ - * layers - OpenLayers' default theme (style.css) includes this: │ │ │ │ │ - * (code) │ │ │ │ │ - * .olLayerGrid .olTileImage { │ │ │ │ │ - * -webkit-transition: opacity 0.2s linear; │ │ │ │ │ - * -moz-transition: opacity 0.2s linear; │ │ │ │ │ - * -o-transition: opacity 0.2s linear; │ │ │ │ │ - * transition: opacity 0.2s linear; │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ - * To animate tile displaying for any grid layer the following │ │ │ │ │ - * CSS rule can be used: │ │ │ │ │ - * (code) │ │ │ │ │ - * .olTileImage { │ │ │ │ │ - * -webkit-transition: opacity 0.2s linear; │ │ │ │ │ - * -moz-transition: opacity 0.2s linear; │ │ │ │ │ - * -o-transition: opacity 0.2s linear; │ │ │ │ │ - * transition: opacity 0.2s linear; │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ - * In that case, to avoid flash effects, │ │ │ │ │ - * should not be zero. │ │ │ │ │ + /** │ │ │ │ │ + * Property: disableFirefoxOverflowHack │ │ │ │ │ + * {Boolean} The hack for overflow in Firefox causes all elements │ │ │ │ │ + * to be re-drawn, which causes Flash elements to be │ │ │ │ │ + * re-initialized, which is troublesome. │ │ │ │ │ + * With this property the hack can be disabled. │ │ │ │ │ */ │ │ │ │ │ - className: null, │ │ │ │ │ + disableFirefoxOverflowHack: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * layer.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Listeners will be called with a reference to an event object. The │ │ │ │ │ - * properties of this event depends on exactly what happened. │ │ │ │ │ - * │ │ │ │ │ - * All event objects have at least the following properties: │ │ │ │ │ - * object - {Object} A reference to layer.events.object. │ │ │ │ │ - * element - {DOMElement} A reference to layer.events.element. │ │ │ │ │ - * │ │ │ │ │ - * Supported event types: │ │ │ │ │ - * addtile - Triggered when a tile is added to this layer. Listeners receive │ │ │ │ │ - * an object as first argument, which has a tile property that │ │ │ │ │ - * references the tile that has been added. │ │ │ │ │ - * tileloadstart - Triggered when a tile starts loading. Listeners receive │ │ │ │ │ - * an object as first argument, which has a tile property that │ │ │ │ │ - * references the tile that starts loading. │ │ │ │ │ - * tileloaded - Triggered when each new tile is │ │ │ │ │ - * loaded, as a means of progress update to listeners. │ │ │ │ │ - * listeners can access 'numLoadingTiles' if they wish to keep │ │ │ │ │ - * track of the loading progress. Listeners are called with an object │ │ │ │ │ - * with a 'tile' property as first argument, making the loaded tile │ │ │ │ │ - * available to the listener, and an 'aborted' property, which will be │ │ │ │ │ - * true when loading was aborted and no tile data is available. │ │ │ │ │ - * tileerror - Triggered before the tileloaded event (i.e. when the tile is │ │ │ │ │ - * still hidden) if a tile failed to load. Listeners receive an object │ │ │ │ │ - * as first argument, which has a tile property that references the │ │ │ │ │ - * tile that could not be loaded. │ │ │ │ │ - * retile - Triggered when the layer recreates its tile grid. │ │ │ │ │ + * Method: fixPadding │ │ │ │ │ + * To be removed in 3.0, this function merely helps us to deal with the │ │ │ │ │ + * case where the user may have set an integer value for padding, │ │ │ │ │ + * instead of an object. │ │ │ │ │ */ │ │ │ │ │ + fixPadding: function() { │ │ │ │ │ + if (typeof this.padding == "number") { │ │ │ │ │ + this.padding = new OpenLayers.Bounds( │ │ │ │ │ + this.padding, this.padding, this.padding, this.padding │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: gridLayout │ │ │ │ │ - * {Object} Object containing properties tilelon, tilelat, startcol, │ │ │ │ │ - * startrow │ │ │ │ │ + * APIProperty: panMapIfOutOfView │ │ │ │ │ + * {Boolean} When drawn, pan map such that the entire popup is visible in │ │ │ │ │ + * the current viewport (if necessary). │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - gridLayout: null, │ │ │ │ │ + panMapIfOutOfView: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: rowSign │ │ │ │ │ - * {Number} 1 for grids starting at the top, -1 for grids starting at the │ │ │ │ │ - * bottom. This is used for several grid index and offset calculations. │ │ │ │ │ + * APIProperty: keepInMap │ │ │ │ │ + * {Boolean} If panMapIfOutOfView is false, and this property is true, │ │ │ │ │ + * contrain the popup such that it always fits in the available map │ │ │ │ │ + * space. By default, this is not set on the base class. If you are │ │ │ │ │ + * creating popups that are near map edges and not allowing pannning, │ │ │ │ │ + * and especially if you have a popup which has a │ │ │ │ │ + * fixedRelativePosition, setting this to false may be a smart thing to │ │ │ │ │ + * do. Subclasses may want to override this setting. │ │ │ │ │ + * │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - rowSign: null, │ │ │ │ │ + keepInMap: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: transitionendEvents │ │ │ │ │ - * {Array} Event names for transitionend │ │ │ │ │ + * APIProperty: closeOnMove │ │ │ │ │ + * {Boolean} When map pans, close the popup. │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - transitionendEvents: [ │ │ │ │ │ - 'transitionend', 'webkitTransitionEnd', 'otransitionend', │ │ │ │ │ - 'oTransitionEnd' │ │ │ │ │ - ], │ │ │ │ │ + closeOnMove: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Grid │ │ │ │ │ - * Create a new grid layer │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * url - {String} │ │ │ │ │ - * params - {Object} │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + /** │ │ │ │ │ + * Property: map │ │ │ │ │ + * {} this gets set in Map.js when the popup is added to the map │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.initialize.apply(this, │ │ │ │ │ - arguments); │ │ │ │ │ - this.grid = []; │ │ │ │ │ - this._removeBackBuffer = OpenLayers.Function.bind(this.removeBackBuffer, this); │ │ │ │ │ + map: null, │ │ │ │ │ │ │ │ │ │ - this.initProperties(); │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Popup │ │ │ │ │ + * Create a popup. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} a unqiue identifier for this popup. If null is passed │ │ │ │ │ + * an identifier will be automatically generated. │ │ │ │ │ + * lonlat - {} The position on the map the popup will │ │ │ │ │ + * be shown. │ │ │ │ │ + * contentSize - {} The size of the content. │ │ │ │ │ + * contentHTML - {String} An HTML string to display inside the │ │ │ │ │ + * popup. │ │ │ │ │ + * closeBox - {Boolean} Whether to display a close box inside │ │ │ │ │ + * the popup. │ │ │ │ │ + * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, closeBox, closeBoxCallback) { │ │ │ │ │ + if (id == null) { │ │ │ │ │ + id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - this.rowSign = this.tileOriginCorner.substr(0, 1) === "t" ? 1 : -1; │ │ │ │ │ + this.id = id; │ │ │ │ │ + this.lonlat = lonlat; │ │ │ │ │ + │ │ │ │ │ + this.contentSize = (contentSize != null) ? contentSize : │ │ │ │ │ + new OpenLayers.Size( │ │ │ │ │ + OpenLayers.Popup.WIDTH, │ │ │ │ │ + OpenLayers.Popup.HEIGHT); │ │ │ │ │ + if (contentHTML != null) { │ │ │ │ │ + this.contentHTML = contentHTML; │ │ │ │ │ + } │ │ │ │ │ + this.backgroundColor = OpenLayers.Popup.COLOR; │ │ │ │ │ + this.opacity = OpenLayers.Popup.OPACITY; │ │ │ │ │ + this.border = OpenLayers.Popup.BORDER; │ │ │ │ │ + │ │ │ │ │ + this.div = OpenLayers.Util.createDiv(this.id, null, null, │ │ │ │ │ + null, null, null, "hidden"); │ │ │ │ │ + this.div.className = this.displayClass; │ │ │ │ │ + │ │ │ │ │ + var groupDivId = this.id + "_GroupDiv"; │ │ │ │ │ + this.groupDiv = OpenLayers.Util.createDiv(groupDivId, null, null, │ │ │ │ │ + null, "relative", null, │ │ │ │ │ + "hidden"); │ │ │ │ │ + │ │ │ │ │ + var id = this.div.id + "_contentDiv"; │ │ │ │ │ + this.contentDiv = OpenLayers.Util.createDiv(id, null, this.contentSize.clone(), │ │ │ │ │ + null, "relative"); │ │ │ │ │ + this.contentDiv.className = this.contentDisplayClass; │ │ │ │ │ + this.groupDiv.appendChild(this.contentDiv); │ │ │ │ │ + this.div.appendChild(this.groupDiv); │ │ │ │ │ + │ │ │ │ │ + if (closeBox) { │ │ │ │ │ + this.addCloseBox(closeBoxCallback); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.registerEvents(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: initProperties │ │ │ │ │ - * Set any properties that depend on the value of singleTile. │ │ │ │ │ - * Currently sets removeBackBufferDelay and className │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * nullify references to prevent circular references and memory leaks │ │ │ │ │ */ │ │ │ │ │ - initProperties: function() { │ │ │ │ │ - if (this.options.removeBackBufferDelay === undefined) { │ │ │ │ │ - this.removeBackBufferDelay = this.singleTile ? 0 : 2500; │ │ │ │ │ + destroy: function() { │ │ │ │ │ + │ │ │ │ │ + this.id = null; │ │ │ │ │ + this.lonlat = null; │ │ │ │ │ + this.size = null; │ │ │ │ │ + this.contentHTML = null; │ │ │ │ │ + │ │ │ │ │ + this.backgroundColor = null; │ │ │ │ │ + this.opacity = null; │ │ │ │ │ + this.border = null; │ │ │ │ │ + │ │ │ │ │ + if (this.closeOnMove && this.map) { │ │ │ │ │ + this.map.events.unregister("movestart", this, this.hide); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - if (this.options.className === undefined) { │ │ │ │ │ - this.className = this.singleTile ? 'olLayerGridSingleTile' : │ │ │ │ │ - 'olLayerGrid'; │ │ │ │ │ + this.events.destroy(); │ │ │ │ │ + this.events = null; │ │ │ │ │ + │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + OpenLayers.Event.stopObservingElement(this.closeDiv); │ │ │ │ │ + this.groupDiv.removeChild(this.closeDiv); │ │ │ │ │ + } │ │ │ │ │ + this.closeDiv = null; │ │ │ │ │ + │ │ │ │ │ + this.div.removeChild(this.groupDiv); │ │ │ │ │ + this.groupDiv = null; │ │ │ │ │ + │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.removePopup(this); │ │ │ │ │ } │ │ │ │ │ + this.map = null; │ │ │ │ │ + this.div = null; │ │ │ │ │ + │ │ │ │ │ + this.autoSize = null; │ │ │ │ │ + this.minSize = null; │ │ │ │ │ + this.maxSize = null; │ │ │ │ │ + this.padding = null; │ │ │ │ │ + this.panMapIfOutOfView = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Constructs the elements that make up the popup. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {} The map. │ │ │ │ │ + * px - {} the position the popup in pixels. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} Reference to a div that contains the drawn popup │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.setMap.call(this, map); │ │ │ │ │ - OpenLayers.Element.addClass(this.div, this.className); │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + if (px == null) { │ │ │ │ │ + if ((this.lonlat != null) && (this.map != null)) { │ │ │ │ │ + px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // this assumes that this.map already exists, which is okay because │ │ │ │ │ + // this.draw is only called once the popup has been added to the map. │ │ │ │ │ + if (this.closeOnMove) { │ │ │ │ │ + this.map.events.register("movestart", this, this.hide); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //listen to movestart, moveend to disable overflow (FF bug) │ │ │ │ │ + if (!this.disableFirefoxOverflowHack && OpenLayers.BROWSER_NAME == 'firefox') { │ │ │ │ │ + this.map.events.register("movestart", this, function() { │ │ │ │ │ + var style = document.defaultView.getComputedStyle( │ │ │ │ │ + this.contentDiv, null │ │ │ │ │ + ); │ │ │ │ │ + var currentOverflow = style.getPropertyValue("overflow"); │ │ │ │ │ + if (currentOverflow != "hidden") { │ │ │ │ │ + this.contentDiv._oldOverflow = currentOverflow; │ │ │ │ │ + this.contentDiv.style.overflow = "hidden"; │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + this.map.events.register("moveend", this, function() { │ │ │ │ │ + var oldOverflow = this.contentDiv._oldOverflow; │ │ │ │ │ + if (oldOverflow) { │ │ │ │ │ + this.contentDiv.style.overflow = oldOverflow; │ │ │ │ │ + this.contentDiv._oldOverflow = null; │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.moveTo(px); │ │ │ │ │ + if (!this.autoSize && !this.size) { │ │ │ │ │ + this.setSize(this.contentSize); │ │ │ │ │ + } │ │ │ │ │ + this.setBackgroundColor(); │ │ │ │ │ + this.setOpacity(); │ │ │ │ │ + this.setBorder(); │ │ │ │ │ + this.setContentHTML(); │ │ │ │ │ + │ │ │ │ │ + if (this.panMapIfOutOfView) { │ │ │ │ │ + this.panIntoView(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeMap │ │ │ │ │ - * Called when the layer is removed from the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {} The map. │ │ │ │ │ + /** │ │ │ │ │ + * Method: updatePosition │ │ │ │ │ + * if the popup has a lonlat and its map members set, │ │ │ │ │ + * then have it move itself to its proper position │ │ │ │ │ */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - this.removeBackBuffer(); │ │ │ │ │ + updatePosition: function() { │ │ │ │ │ + if ((this.lonlat) && (this.map)) { │ │ │ │ │ + var px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ + if (px) { │ │ │ │ │ + this.moveTo(px); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Deconstruct the layer and clear the grid. │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {} the top and left position of the popup div. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.removeBackBuffer(); │ │ │ │ │ - this.clearGrid(); │ │ │ │ │ - │ │ │ │ │ - this.grid = null; │ │ │ │ │ - this.tileSize = null; │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.destroy.apply(this, arguments); │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + if ((px != null) && (this.div != null)) { │ │ │ │ │ + this.div.style.left = px.x + "px"; │ │ │ │ │ + this.div.style.top = px.y + "px"; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: mergeNewParams │ │ │ │ │ - * Refetches tiles with new params merged, keeping a backbuffer. Each │ │ │ │ │ - * loading new tile will have a css class of '.olTileReplacing'. If a │ │ │ │ │ - * stylesheet applies a 'display: none' style to that class, any fade-in │ │ │ │ │ - * transition will not apply, and backbuffers for each tile will be removed │ │ │ │ │ - * as soon as the tile is loaded. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newParams - {Object} │ │ │ │ │ + * Method: visible │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * redrawn: {Boolean} whether the layer was actually redrawn. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Boolean indicating whether or not the popup is visible │ │ │ │ │ */ │ │ │ │ │ + visible: function() { │ │ │ │ │ + return OpenLayers.Element.visible(this.div); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clearGrid │ │ │ │ │ - * Go through and remove all tiles from the grid, calling │ │ │ │ │ - * destroy() on each of them to kill circular references │ │ │ │ │ + * Method: toggle │ │ │ │ │ + * Toggles visibility of the popup. │ │ │ │ │ */ │ │ │ │ │ - clearGrid: function() { │ │ │ │ │ - if (this.grid) { │ │ │ │ │ - for (var iRow = 0, len = this.grid.length; iRow < len; iRow++) { │ │ │ │ │ - var row = this.grid[iRow]; │ │ │ │ │ - for (var iCol = 0, clen = row.length; iCol < clen; iCol++) { │ │ │ │ │ - var tile = row[iCol]; │ │ │ │ │ - this.destroyTile(tile); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.grid = []; │ │ │ │ │ - this.gridResolution = null; │ │ │ │ │ - this.gridLayout = null; │ │ │ │ │ + toggle: function() { │ │ │ │ │ + if (this.visible()) { │ │ │ │ │ + this.hide(); │ │ │ │ │ + } else { │ │ │ │ │ + this.show(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: addOptions │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newOptions - {Object} │ │ │ │ │ - * reinitialize - {Boolean} If set to true, and if resolution options of the │ │ │ │ │ - * current baseLayer were changed, the map will be recentered to make │ │ │ │ │ - * sure that it is displayed with a valid resolution, and a │ │ │ │ │ - * changebaselayer event will be triggered. │ │ │ │ │ + * Method: show │ │ │ │ │ + * Makes the popup visible. │ │ │ │ │ */ │ │ │ │ │ - addOptions: function(newOptions, reinitialize) { │ │ │ │ │ - var singleTileChanged = newOptions.singleTile !== undefined && │ │ │ │ │ - newOptions.singleTile !== this.singleTile; │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.addOptions.apply(this, arguments); │ │ │ │ │ - if (this.map && singleTileChanged) { │ │ │ │ │ - this.initProperties(); │ │ │ │ │ - this.clearGrid(); │ │ │ │ │ - this.tileSize = this.options.tileSize; │ │ │ │ │ - this.setTileSize(); │ │ │ │ │ - this.moveTo(null, true); │ │ │ │ │ + show: function() { │ │ │ │ │ + this.div.style.display = ''; │ │ │ │ │ + │ │ │ │ │ + if (this.panMapIfOutOfView) { │ │ │ │ │ + this.panIntoView(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ + * Method: hide │ │ │ │ │ + * Makes the popup invisible. │ │ │ │ │ + */ │ │ │ │ │ + hide: function() { │ │ │ │ │ + this.div.style.display = 'none'; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setSize │ │ │ │ │ + * Used to adjust the size of the popup. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {Object} Is this ever used? │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} An exact clone of this OpenLayers.Layer.Grid │ │ │ │ │ + * contentSize - {} the new size for the popup's │ │ │ │ │ + * contents div (in pixels). │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Grid(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.params, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ + setSize: function(contentSize) { │ │ │ │ │ + this.size = contentSize.clone(); │ │ │ │ │ │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.HTTPRequest.prototype.clone.apply(this, [obj]); │ │ │ │ │ + // if our contentDiv has a css 'padding' set on it by a stylesheet, we │ │ │ │ │ + // must add that to the desired "size". │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ + var wPadding = contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ + var hPadding = contentDivPadding.top + contentDivPadding.bottom; │ │ │ │ │ │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - if (this.tileSize != null) { │ │ │ │ │ - obj.tileSize = this.tileSize.clone(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // we do not want to copy reference to grid, so we make a new array │ │ │ │ │ - obj.grid = []; │ │ │ │ │ - obj.gridResolution = null; │ │ │ │ │ - // same for backbuffer │ │ │ │ │ - obj.backBuffer = null; │ │ │ │ │ - obj.backBufferTimerId = null; │ │ │ │ │ - obj.loading = false; │ │ │ │ │ - obj.numLoadingTiles = 0; │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * This function is called whenever the map is moved. All the moving │ │ │ │ │ - * of actual 'tiles' is done by the map, but moveTo's role is to accept │ │ │ │ │ - * a bounds and make sure the data that that bounds requires is pre-loaded. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {} │ │ │ │ │ - * zoomChanged - {Boolean} │ │ │ │ │ - * dragging - {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - bounds = bounds || this.map.getExtent(); │ │ │ │ │ - │ │ │ │ │ - if (bounds != null) { │ │ │ │ │ - │ │ │ │ │ - // if grid is empty or zoom has changed, we *must* re-tile │ │ │ │ │ - var forceReTile = !this.grid.length || zoomChanged; │ │ │ │ │ - │ │ │ │ │ - // total bounds of the tiles │ │ │ │ │ - var tilesBounds = this.getTilesBounds(); │ │ │ │ │ - │ │ │ │ │ - // the new map resolution │ │ │ │ │ - var resolution = this.map.getResolution(); │ │ │ │ │ - │ │ │ │ │ - // the server-supported resolution for the new map resolution │ │ │ │ │ - var serverResolution = this.getServerResolution(resolution); │ │ │ │ │ - │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - │ │ │ │ │ - // We want to redraw whenever even the slightest part of the │ │ │ │ │ - // current bounds is not contained by our tile. │ │ │ │ │ - // (thus, we do not specify partial -- its default is false) │ │ │ │ │ - │ │ │ │ │ - if (forceReTile || │ │ │ │ │ - (!dragging && !tilesBounds.containsBounds(bounds))) { │ │ │ │ │ - │ │ │ │ │ - // In single tile mode with no transition effect, we insert │ │ │ │ │ - // a non-scaled backbuffer when the layer is moved. But if │ │ │ │ │ - // a zoom occurs right after a move, i.e. before the new │ │ │ │ │ - // image is received, we need to remove the backbuffer, or │ │ │ │ │ - // an ill-positioned image will be visible during the zoom │ │ │ │ │ - // transition. │ │ │ │ │ - │ │ │ │ │ - if (zoomChanged && this.transitionEffect !== 'resize') { │ │ │ │ │ - this.removeBackBuffer(); │ │ │ │ │ - } │ │ │ │ │ + // take into account the popup's 'padding' property │ │ │ │ │ + this.fixPadding(); │ │ │ │ │ + wPadding += this.padding.left + this.padding.right; │ │ │ │ │ + hPadding += this.padding.top + this.padding.bottom; │ │ │ │ │ │ │ │ │ │ - if (!zoomChanged || this.transitionEffect === 'resize') { │ │ │ │ │ - this.applyBackBuffer(resolution); │ │ │ │ │ - } │ │ │ │ │ + // make extra space for the close div │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + var closeDivWidth = parseInt(this.closeDiv.style.width); │ │ │ │ │ + wPadding += closeDivWidth + contentDivPadding.right; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - this.initSingleTile(bounds); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ + //increase size of the main popup div to take into account the │ │ │ │ │ + // users's desired padding and close div. │ │ │ │ │ + this.size.w += wPadding; │ │ │ │ │ + this.size.h += hPadding; │ │ │ │ │ │ │ │ │ │ - // if the bounds have changed such that they are not even │ │ │ │ │ - // *partially* contained by our tiles (e.g. when user has │ │ │ │ │ - // programmatically panned to the other side of the earth on │ │ │ │ │ - // zoom level 18), then moveGriddedTiles could potentially have │ │ │ │ │ - // to run through thousands of cycles, so we want to reTile │ │ │ │ │ - // instead (thus, partial true). │ │ │ │ │ - forceReTile = forceReTile || │ │ │ │ │ - !tilesBounds.intersectsBounds(bounds, { │ │ │ │ │ - worldBounds: this.map.baseLayer.wrapDateLine && │ │ │ │ │ - this.map.getMaxExtent() │ │ │ │ │ - }); │ │ │ │ │ + //now if our browser is IE, we need to actually make the contents │ │ │ │ │ + // div itself bigger to take its own padding into effect. this makes │ │ │ │ │ + // me want to shoot someone, but so it goes. │ │ │ │ │ + if (OpenLayers.BROWSER_NAME == "msie") { │ │ │ │ │ + this.contentSize.w += │ │ │ │ │ + contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ + this.contentSize.h += │ │ │ │ │ + contentDivPadding.bottom + contentDivPadding.top; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (forceReTile) { │ │ │ │ │ - if (zoomChanged && (this.transitionEffect === 'resize' || │ │ │ │ │ - this.gridResolution === resolution)) { │ │ │ │ │ - this.applyBackBuffer(resolution); │ │ │ │ │ - } │ │ │ │ │ - this.initGriddedTiles(bounds); │ │ │ │ │ - } else { │ │ │ │ │ - this.moveGriddedTiles(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.div.style.width = this.size.w + "px"; │ │ │ │ │ + this.div.style.height = this.size.h + "px"; │ │ │ │ │ + } │ │ │ │ │ + if (this.contentDiv != null) { │ │ │ │ │ + this.contentDiv.style.width = contentSize.w + "px"; │ │ │ │ │ + this.contentDiv.style.height = contentSize.h + "px"; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getTileData │ │ │ │ │ - * Given a map location, retrieve a tile and the pixel offset within that │ │ │ │ │ - * tile corresponding to the location. If there is not an existing │ │ │ │ │ - * tile in the grid that covers the given location, null will be │ │ │ │ │ - * returned. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * loc - {} map location │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Object with the following properties: tile ({}), │ │ │ │ │ - * i ({Number} x-pixel offset from top left), and j ({Integer} y-pixel │ │ │ │ │ - * offset from top left). │ │ │ │ │ + * APIMethod: updateSize │ │ │ │ │ + * Auto size the popup so that it precisely fits its contents (as │ │ │ │ │ + * determined by this.contentDiv.innerHTML). Popup size will, of │ │ │ │ │ + * course, be limited by the available space on the current map │ │ │ │ │ */ │ │ │ │ │ - getTileData: function(loc) { │ │ │ │ │ - var data = null, │ │ │ │ │ - x = loc.lon, │ │ │ │ │ - y = loc.lat, │ │ │ │ │ - numRows = this.grid.length; │ │ │ │ │ + updateSize: function() { │ │ │ │ │ │ │ │ │ │ - if (this.map && numRows) { │ │ │ │ │ - var res = this.map.getResolution(), │ │ │ │ │ - tileWidth = this.tileSize.w, │ │ │ │ │ - tileHeight = this.tileSize.h, │ │ │ │ │ - bounds = this.grid[0][0].bounds, │ │ │ │ │ - left = bounds.left, │ │ │ │ │ - top = bounds.top; │ │ │ │ │ + // determine actual render dimensions of the contents by putting its │ │ │ │ │ + // contents into a fake contentDiv (for the CSS) and then measuring it │ │ │ │ │ + var preparedHTML = "
" + │ │ │ │ │ + this.contentDiv.innerHTML + │ │ │ │ │ + "
"; │ │ │ │ │ │ │ │ │ │ - if (x < left) { │ │ │ │ │ - // deal with multiple worlds │ │ │ │ │ - if (this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - var worldWidth = this.map.getMaxExtent().getWidth(); │ │ │ │ │ - var worldsAway = Math.ceil((left - x) / worldWidth); │ │ │ │ │ - x += worldWidth * worldsAway; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // tile distance to location (fractional number of tiles); │ │ │ │ │ - var dtx = (x - left) / (res * tileWidth); │ │ │ │ │ - var dty = (top - y) / (res * tileHeight); │ │ │ │ │ - // index of tile in grid │ │ │ │ │ - var col = Math.floor(dtx); │ │ │ │ │ - var row = Math.floor(dty); │ │ │ │ │ - if (row >= 0 && row < numRows) { │ │ │ │ │ - var tile = this.grid[row][col]; │ │ │ │ │ - if (tile) { │ │ │ │ │ - data = { │ │ │ │ │ - tile: tile, │ │ │ │ │ - // pixel index within tile │ │ │ │ │ - i: Math.floor((dtx - col) * tileWidth), │ │ │ │ │ - j: Math.floor((dty - row) * tileHeight) │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ + var containerElement = (this.map) ? this.map.div : document.body; │ │ │ │ │ + var realSize = OpenLayers.Util.getRenderedDimensions( │ │ │ │ │ + preparedHTML, null, { │ │ │ │ │ + displayClass: this.displayClass, │ │ │ │ │ + containerElement: containerElement │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return data; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroyTile │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * tile - {} │ │ │ │ │ - */ │ │ │ │ │ - destroyTile: function(tile) { │ │ │ │ │ - this.removeTileMonitoringHooks(tile); │ │ │ │ │ - tile.destroy(); │ │ │ │ │ - }, │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getServerResolution │ │ │ │ │ - * Return the closest server-supported resolution. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resolution - {Number} The base resolution. If undefined the │ │ │ │ │ - * map resolution is used. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number} The closest server resolution value. │ │ │ │ │ - */ │ │ │ │ │ - getServerResolution: function(resolution) { │ │ │ │ │ - var distance = Number.POSITIVE_INFINITY; │ │ │ │ │ - resolution = resolution || this.map.getResolution(); │ │ │ │ │ - if (this.serverResolutions && │ │ │ │ │ - OpenLayers.Util.indexOf(this.serverResolutions, resolution) === -1) { │ │ │ │ │ - var i, newDistance, newResolution, serverResolution; │ │ │ │ │ - for (i = this.serverResolutions.length - 1; i >= 0; i--) { │ │ │ │ │ - newResolution = this.serverResolutions[i]; │ │ │ │ │ - newDistance = Math.abs(newResolution - resolution); │ │ │ │ │ - if (newDistance > distance) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - distance = newDistance; │ │ │ │ │ - serverResolution = newResolution; │ │ │ │ │ - } │ │ │ │ │ - resolution = serverResolution; │ │ │ │ │ - } │ │ │ │ │ - return resolution; │ │ │ │ │ - }, │ │ │ │ │ + // is the "real" size of the div is safe to display in our map? │ │ │ │ │ + var safeSize = this.getSafeContentSize(realSize); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getServerZoom │ │ │ │ │ - * Return the zoom value corresponding to the best matching server │ │ │ │ │ - * resolution, taking into account and . │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number} The closest server supported zoom. This is not the map zoom │ │ │ │ │ - * level, but an index of the server's resolutions array. │ │ │ │ │ - */ │ │ │ │ │ - getServerZoom: function() { │ │ │ │ │ - var resolution = this.getServerResolution(); │ │ │ │ │ - return this.serverResolutions ? │ │ │ │ │ - OpenLayers.Util.indexOf(this.serverResolutions, resolution) : │ │ │ │ │ - this.map.getZoomForResolution(resolution) + (this.zoomOffset || 0); │ │ │ │ │ - }, │ │ │ │ │ + var newSize = null; │ │ │ │ │ + if (safeSize.equals(realSize)) { │ │ │ │ │ + //real size of content is small enough to fit on the map, │ │ │ │ │ + // so we use real size. │ │ │ │ │ + newSize = realSize; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: applyBackBuffer │ │ │ │ │ - * Create, insert, scale and position a back buffer for the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resolution - {Number} The resolution to transition to. │ │ │ │ │ - */ │ │ │ │ │ - applyBackBuffer: function(resolution) { │ │ │ │ │ - if (this.backBufferTimerId !== null) { │ │ │ │ │ - this.removeBackBuffer(); │ │ │ │ │ - } │ │ │ │ │ - var backBuffer = this.backBuffer; │ │ │ │ │ - if (!backBuffer) { │ │ │ │ │ - backBuffer = this.createBackBuffer(); │ │ │ │ │ - if (!backBuffer) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (resolution === this.gridResolution) { │ │ │ │ │ - this.div.insertBefore(backBuffer, this.div.firstChild); │ │ │ │ │ - } else { │ │ │ │ │ - this.map.baseLayer.div.parentNode.insertBefore(backBuffer, this.map.baseLayer.div); │ │ │ │ │ - } │ │ │ │ │ - this.backBuffer = backBuffer; │ │ │ │ │ + } else { │ │ │ │ │ │ │ │ │ │ - // set some information in the instance for subsequent │ │ │ │ │ - // calls to applyBackBuffer where the same back buffer │ │ │ │ │ - // is reused │ │ │ │ │ - var topLeftTileBounds = this.grid[0][0].bounds; │ │ │ │ │ - this.backBufferLonLat = { │ │ │ │ │ - lon: topLeftTileBounds.left, │ │ │ │ │ - lat: topLeftTileBounds.top │ │ │ │ │ + // make a new 'size' object with the clipped dimensions │ │ │ │ │ + // set or null if not clipped. │ │ │ │ │ + var fixedSize = { │ │ │ │ │ + w: (safeSize.w < realSize.w) ? safeSize.w : null, │ │ │ │ │ + h: (safeSize.h < realSize.h) ? safeSize.h : null │ │ │ │ │ }; │ │ │ │ │ - this.backBufferResolution = this.gridResolution; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var ratio = this.backBufferResolution / resolution; │ │ │ │ │ - │ │ │ │ │ - // scale the tiles inside the back buffer │ │ │ │ │ - var tiles = backBuffer.childNodes, │ │ │ │ │ - tile; │ │ │ │ │ - for (var i = tiles.length - 1; i >= 0; --i) { │ │ │ │ │ - tile = tiles[i]; │ │ │ │ │ - tile.style.top = ((ratio * tile._i * tile._h) | 0) + 'px'; │ │ │ │ │ - tile.style.left = ((ratio * tile._j * tile._w) | 0) + 'px'; │ │ │ │ │ - tile.style.width = Math.round(ratio * tile._w) + 'px'; │ │ │ │ │ - tile.style.height = Math.round(ratio * tile._h) + 'px'; │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - // and position it (based on the grid's top-left corner) │ │ │ │ │ - var position = this.getViewPortPxFromLonLat( │ │ │ │ │ - this.backBufferLonLat, resolution); │ │ │ │ │ - var leftOffset = this.map.layerContainerOriginPx.x; │ │ │ │ │ - var topOffset = this.map.layerContainerOriginPx.y; │ │ │ │ │ - backBuffer.style.left = Math.round(position.x - leftOffset) + 'px'; │ │ │ │ │ - backBuffer.style.top = Math.round(position.y - topOffset) + 'px'; │ │ │ │ │ - }, │ │ │ │ │ + if (fixedSize.w && fixedSize.h) { │ │ │ │ │ + //content is too big in both directions, so we will use │ │ │ │ │ + // max popup size (safeSize), knowing well that it will │ │ │ │ │ + // overflow both ways. │ │ │ │ │ + newSize = safeSize; │ │ │ │ │ + } else { │ │ │ │ │ + //content is clipped in only one direction, so we need to │ │ │ │ │ + // run getRenderedDimensions() again with a fixed dimension │ │ │ │ │ + var clippedSize = OpenLayers.Util.getRenderedDimensions( │ │ │ │ │ + preparedHTML, fixedSize, { │ │ │ │ │ + displayClass: this.contentDisplayClass, │ │ │ │ │ + containerElement: containerElement │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createBackBuffer │ │ │ │ │ - * Create a back buffer. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The DOM element for the back buffer, undefined if the │ │ │ │ │ - * grid isn't initialized yet. │ │ │ │ │ - */ │ │ │ │ │ - createBackBuffer: function() { │ │ │ │ │ - var backBuffer; │ │ │ │ │ - if (this.grid.length > 0) { │ │ │ │ │ - backBuffer = document.createElement('div'); │ │ │ │ │ - backBuffer.id = this.div.id + '_bb'; │ │ │ │ │ - backBuffer.className = 'olBackBuffer'; │ │ │ │ │ - backBuffer.style.position = 'absolute'; │ │ │ │ │ - var map = this.map; │ │ │ │ │ - backBuffer.style.zIndex = this.transitionEffect === 'resize' ? │ │ │ │ │ - this.getZIndex() - 1 : │ │ │ │ │ - // 'map-resize': │ │ │ │ │ - map.Z_INDEX_BASE.BaseLayer - │ │ │ │ │ - (map.getNumLayers() - map.getLayerIndex(this)); │ │ │ │ │ - for (var i = 0, lenI = this.grid.length; i < lenI; i++) { │ │ │ │ │ - for (var j = 0, lenJ = this.grid[i].length; j < lenJ; j++) { │ │ │ │ │ - var tile = this.grid[i][j], │ │ │ │ │ - markup = this.grid[i][j].createBackBuffer(); │ │ │ │ │ - if (markup) { │ │ │ │ │ - markup._i = i; │ │ │ │ │ - markup._j = j; │ │ │ │ │ - markup._w = tile.size.w; │ │ │ │ │ - markup._h = tile.size.h; │ │ │ │ │ - markup.id = tile.id + '_bb'; │ │ │ │ │ - backBuffer.appendChild(markup); │ │ │ │ │ + //if the clipped size is still the same as the safeSize, │ │ │ │ │ + // that means that our content must be fixed in the │ │ │ │ │ + // offending direction. If overflow is 'auto', this means │ │ │ │ │ + // we are going to have a scrollbar for sure, so we must │ │ │ │ │ + // adjust for that. │ │ │ │ │ + // │ │ │ │ │ + var currentOverflow = OpenLayers.Element.getStyle( │ │ │ │ │ + this.contentDiv, "overflow" │ │ │ │ │ + ); │ │ │ │ │ + if ((currentOverflow != "hidden") && │ │ │ │ │ + (clippedSize.equals(safeSize))) { │ │ │ │ │ + var scrollBar = OpenLayers.Util.getScrollbarWidth(); │ │ │ │ │ + if (fixedSize.w) { │ │ │ │ │ + clippedSize.h += scrollBar; │ │ │ │ │ + } else { │ │ │ │ │ + clippedSize.w += scrollBar; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return backBuffer; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeBackBuffer │ │ │ │ │ - * Remove back buffer from DOM. │ │ │ │ │ - */ │ │ │ │ │ - removeBackBuffer: function() { │ │ │ │ │ - if (this._transitionElement) { │ │ │ │ │ - for (var i = this.transitionendEvents.length - 1; i >= 0; --i) { │ │ │ │ │ - OpenLayers.Event.stopObserving(this._transitionElement, │ │ │ │ │ - this.transitionendEvents[i], this._removeBackBuffer); │ │ │ │ │ - } │ │ │ │ │ - delete this._transitionElement; │ │ │ │ │ - } │ │ │ │ │ - if (this.backBuffer) { │ │ │ │ │ - if (this.backBuffer.parentNode) { │ │ │ │ │ - this.backBuffer.parentNode.removeChild(this.backBuffer); │ │ │ │ │ - } │ │ │ │ │ - this.backBuffer = null; │ │ │ │ │ - this.backBufferResolution = null; │ │ │ │ │ - if (this.backBufferTimerId !== null) { │ │ │ │ │ - window.clearTimeout(this.backBufferTimerId); │ │ │ │ │ - this.backBufferTimerId = null; │ │ │ │ │ + newSize = this.getSafeContentSize(clippedSize); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + this.setSize(newSize); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveByPx │ │ │ │ │ - * Move the layer based on pixel vector. │ │ │ │ │ + * Method: setBackgroundColor │ │ │ │ │ + * Sets the background color of the popup. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * dx - {Number} │ │ │ │ │ - * dy - {Number} │ │ │ │ │ - */ │ │ │ │ │ - moveByPx: function(dx, dy) { │ │ │ │ │ - if (!this.singleTile) { │ │ │ │ │ - this.moveGriddedTiles(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setTileSize │ │ │ │ │ - * Check if we are in singleTile mode and if so, set the size as a ratio │ │ │ │ │ - * of the map size (as specified by the layer's 'ratio' property). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * size - {} │ │ │ │ │ + * color - {String} the background color. eg "#FFBBBB" │ │ │ │ │ */ │ │ │ │ │ - setTileSize: function(size) { │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - size.h = parseInt(size.h * this.ratio, 10); │ │ │ │ │ - size.w = parseInt(size.w * this.ratio, 10); │ │ │ │ │ + setBackgroundColor: function(color) { │ │ │ │ │ + if (color != undefined) { │ │ │ │ │ + this.backgroundColor = color; │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.setTileSize.apply(this, [size]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getTilesBounds │ │ │ │ │ - * Return the bounds of the tile grid. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} A Bounds object representing the bounds of all the │ │ │ │ │ - * currently loaded tiles (including those partially or not at all seen │ │ │ │ │ - * onscreen). │ │ │ │ │ - */ │ │ │ │ │ - getTilesBounds: function() { │ │ │ │ │ - var bounds = null; │ │ │ │ │ - │ │ │ │ │ - var length = this.grid.length; │ │ │ │ │ - if (length) { │ │ │ │ │ - var bottomLeftTileBounds = this.grid[length - 1][0].bounds, │ │ │ │ │ - width = this.grid[0].length * bottomLeftTileBounds.getWidth(), │ │ │ │ │ - height = this.grid.length * bottomLeftTileBounds.getHeight(); │ │ │ │ │ │ │ │ │ │ - bounds = new OpenLayers.Bounds(bottomLeftTileBounds.left, │ │ │ │ │ - bottomLeftTileBounds.bottom, │ │ │ │ │ - bottomLeftTileBounds.left + width, │ │ │ │ │ - bottomLeftTileBounds.bottom + height); │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.div.style.backgroundColor = this.backgroundColor; │ │ │ │ │ } │ │ │ │ │ - return bounds; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: initSingleTile │ │ │ │ │ + * Method: setOpacity │ │ │ │ │ + * Sets the opacity of the popup. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * opacity - {float} A value between 0.0 (transparent) and 1.0 (solid). │ │ │ │ │ */ │ │ │ │ │ - initSingleTile: function(bounds) { │ │ │ │ │ - this.events.triggerEvent("retile"); │ │ │ │ │ - │ │ │ │ │ - //determine new tile bounds │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var tileWidth = bounds.getWidth() * this.ratio; │ │ │ │ │ - var tileHeight = bounds.getHeight() * this.ratio; │ │ │ │ │ - │ │ │ │ │ - var tileBounds = │ │ │ │ │ - new OpenLayers.Bounds(center.lon - (tileWidth / 2), │ │ │ │ │ - center.lat - (tileHeight / 2), │ │ │ │ │ - center.lon + (tileWidth / 2), │ │ │ │ │ - center.lat + (tileHeight / 2)); │ │ │ │ │ - │ │ │ │ │ - var px = this.map.getLayerPxFromLonLat({ │ │ │ │ │ - lon: tileBounds.left, │ │ │ │ │ - lat: tileBounds.top │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - if (!this.grid.length) { │ │ │ │ │ - this.grid[0] = []; │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + if (opacity != undefined) { │ │ │ │ │ + this.opacity = opacity; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - var tile = this.grid[0][0]; │ │ │ │ │ - if (!tile) { │ │ │ │ │ - tile = this.addTile(tileBounds, px); │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + // for Mozilla and Safari │ │ │ │ │ + this.div.style.opacity = this.opacity; │ │ │ │ │ │ │ │ │ │ - this.addTileMonitoringHooks(tile); │ │ │ │ │ - tile.draw(); │ │ │ │ │ - this.grid[0][0] = tile; │ │ │ │ │ - } else { │ │ │ │ │ - tile.moveTo(tileBounds, px); │ │ │ │ │ + // for IE │ │ │ │ │ + this.div.style.filter = 'alpha(opacity=' + this.opacity * 100 + ')'; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - //remove all but our single tile │ │ │ │ │ - this.removeExcessTiles(1, 1); │ │ │ │ │ - │ │ │ │ │ - // store the resolution of the grid │ │ │ │ │ - this.gridResolution = this.getServerResolution(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateGridLayout │ │ │ │ │ - * Generate parameters for the grid layout. │ │ │ │ │ + /** │ │ │ │ │ + * Method: setBorder │ │ │ │ │ + * Sets the border style of the popup. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {|Object} OpenLayers.Bounds or an │ │ │ │ │ - * object with a 'left' and 'top' properties. │ │ │ │ │ - * origin - {|Object} OpenLayers.LonLat or an │ │ │ │ │ - * object with a 'lon' and 'lat' properties. │ │ │ │ │ - * resolution - {Number} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Object containing properties tilelon, tilelat, startcol, │ │ │ │ │ - * startrow │ │ │ │ │ + * border - {String} The border style value. eg 2px │ │ │ │ │ */ │ │ │ │ │ - calculateGridLayout: function(bounds, origin, resolution) { │ │ │ │ │ - var tilelon = resolution * this.tileSize.w; │ │ │ │ │ - var tilelat = resolution * this.tileSize.h; │ │ │ │ │ - │ │ │ │ │ - var offsetlon = bounds.left - origin.lon; │ │ │ │ │ - var tilecol = Math.floor(offsetlon / tilelon) - this.buffer; │ │ │ │ │ - │ │ │ │ │ - var rowSign = this.rowSign; │ │ │ │ │ - │ │ │ │ │ - var offsetlat = rowSign * (origin.lat - bounds.top + tilelat); │ │ │ │ │ - var tilerow = Math[~rowSign ? 'floor' : 'ceil'](offsetlat / tilelat) - this.buffer * rowSign; │ │ │ │ │ - │ │ │ │ │ - return { │ │ │ │ │ - tilelon: tilelon, │ │ │ │ │ - tilelat: tilelat, │ │ │ │ │ - startcol: tilecol, │ │ │ │ │ - startrow: tilerow │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ + setBorder: function(border) { │ │ │ │ │ + if (border != undefined) { │ │ │ │ │ + this.border = border; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getTileOrigin │ │ │ │ │ - * Determine the origin for aligning the grid of tiles. If a │ │ │ │ │ - * property is supplied, that will be returned. Otherwise, the origin │ │ │ │ │ - * will be derived from the layer's property. In this case, │ │ │ │ │ - * the tile origin will be the corner of the given by the │ │ │ │ │ - * property. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} The tile origin. │ │ │ │ │ - */ │ │ │ │ │ - getTileOrigin: function() { │ │ │ │ │ - var origin = this.tileOrigin; │ │ │ │ │ - if (!origin) { │ │ │ │ │ - var extent = this.getMaxExtent(); │ │ │ │ │ - var edges = ({ │ │ │ │ │ - "tl": ["left", "top"], │ │ │ │ │ - "tr": ["right", "top"], │ │ │ │ │ - "bl": ["left", "bottom"], │ │ │ │ │ - "br": ["right", "bottom"] │ │ │ │ │ - })[this.tileOriginCorner]; │ │ │ │ │ - origin = new OpenLayers.LonLat(extent[edges[0]], extent[edges[1]]); │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.div.style.border = this.border; │ │ │ │ │ } │ │ │ │ │ - return origin; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getTileBoundsForGridIndex │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * row - {Number} The row of the grid │ │ │ │ │ - * col - {Number} The column of the grid │ │ │ │ │ + * Method: setContentHTML │ │ │ │ │ + * Allows the user to set the HTML content of the popup. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} The bounds for the tile at (row, col) │ │ │ │ │ - */ │ │ │ │ │ - getTileBoundsForGridIndex: function(row, col) { │ │ │ │ │ - var origin = this.getTileOrigin(); │ │ │ │ │ - var tileLayout = this.gridLayout; │ │ │ │ │ - var tilelon = tileLayout.tilelon; │ │ │ │ │ - var tilelat = tileLayout.tilelat; │ │ │ │ │ - var startcol = tileLayout.startcol; │ │ │ │ │ - var startrow = tileLayout.startrow; │ │ │ │ │ - var rowSign = this.rowSign; │ │ │ │ │ - return new OpenLayers.Bounds( │ │ │ │ │ - origin.lon + (startcol + col) * tilelon, │ │ │ │ │ - origin.lat - (startrow + row * rowSign) * tilelat * rowSign, │ │ │ │ │ - origin.lon + (startcol + col + 1) * tilelon, │ │ │ │ │ - origin.lat - (startrow + (row - 1) * rowSign) * tilelat * rowSign │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: initGriddedTiles │ │ │ │ │ - * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {} │ │ │ │ │ + * contentHTML - {String} HTML for the div. │ │ │ │ │ */ │ │ │ │ │ - initGriddedTiles: function(bounds) { │ │ │ │ │ - this.events.triggerEvent("retile"); │ │ │ │ │ - │ │ │ │ │ - // work out mininum number of rows and columns; this is the number of │ │ │ │ │ - // tiles required to cover the viewport plus at least one for panning │ │ │ │ │ - │ │ │ │ │ - var viewSize = this.map.getSize(); │ │ │ │ │ - │ │ │ │ │ - var origin = this.getTileOrigin(); │ │ │ │ │ - var resolution = this.map.getResolution(), │ │ │ │ │ - serverResolution = this.getServerResolution(), │ │ │ │ │ - ratio = resolution / serverResolution, │ │ │ │ │ - tileSize = { │ │ │ │ │ - w: this.tileSize.w / ratio, │ │ │ │ │ - h: this.tileSize.h / ratio │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - var minRows = Math.ceil(viewSize.h / tileSize.h) + │ │ │ │ │ - 2 * this.buffer + 1; │ │ │ │ │ - var minCols = Math.ceil(viewSize.w / tileSize.w) + │ │ │ │ │ - 2 * this.buffer + 1; │ │ │ │ │ + setContentHTML: function(contentHTML) { │ │ │ │ │ │ │ │ │ │ - var tileLayout = this.calculateGridLayout(bounds, origin, serverResolution); │ │ │ │ │ - this.gridLayout = tileLayout; │ │ │ │ │ + if (contentHTML != null) { │ │ │ │ │ + this.contentHTML = contentHTML; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var tilelon = tileLayout.tilelon; │ │ │ │ │ - var tilelat = tileLayout.tilelat; │ │ │ │ │ + if ((this.contentDiv != null) && │ │ │ │ │ + (this.contentHTML != null) && │ │ │ │ │ + (this.contentHTML != this.contentDiv.innerHTML)) { │ │ │ │ │ │ │ │ │ │ - var layerContainerDivLeft = this.map.layerContainerOriginPx.x; │ │ │ │ │ - var layerContainerDivTop = this.map.layerContainerOriginPx.y; │ │ │ │ │ + this.contentDiv.innerHTML = this.contentHTML; │ │ │ │ │ │ │ │ │ │ - var tileBounds = this.getTileBoundsForGridIndex(0, 0); │ │ │ │ │ - var startPx = this.map.getViewPortPxFromLonLat( │ │ │ │ │ - new OpenLayers.LonLat(tileBounds.left, tileBounds.top) │ │ │ │ │ - ); │ │ │ │ │ - startPx.x = Math.round(startPx.x) - layerContainerDivLeft; │ │ │ │ │ - startPx.y = Math.round(startPx.y) - layerContainerDivTop; │ │ │ │ │ + if (this.autoSize) { │ │ │ │ │ │ │ │ │ │ - var tileData = [], │ │ │ │ │ - center = this.map.getCenter(); │ │ │ │ │ + //if popup has images, listen for when they finish │ │ │ │ │ + // loading and resize accordingly │ │ │ │ │ + this.registerImageListeners(); │ │ │ │ │ │ │ │ │ │ - var rowidx = 0; │ │ │ │ │ - do { │ │ │ │ │ - var row = this.grid[rowidx]; │ │ │ │ │ - if (!row) { │ │ │ │ │ - row = []; │ │ │ │ │ - this.grid.push(row); │ │ │ │ │ + //auto size the popup to its current contents │ │ │ │ │ + this.updateSize(); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var colidx = 0; │ │ │ │ │ - do { │ │ │ │ │ - tileBounds = this.getTileBoundsForGridIndex(rowidx, colidx); │ │ │ │ │ - var px = startPx.clone(); │ │ │ │ │ - px.x = px.x + colidx * Math.round(tileSize.w); │ │ │ │ │ - px.y = px.y + rowidx * Math.round(tileSize.h); │ │ │ │ │ - var tile = row[colidx]; │ │ │ │ │ - if (!tile) { │ │ │ │ │ - tile = this.addTile(tileBounds, px); │ │ │ │ │ - this.addTileMonitoringHooks(tile); │ │ │ │ │ - row.push(tile); │ │ │ │ │ - } else { │ │ │ │ │ - tile.moveTo(tileBounds, px, false); │ │ │ │ │ - } │ │ │ │ │ - var tileCenter = tileBounds.getCenterLonLat(); │ │ │ │ │ - tileData.push({ │ │ │ │ │ - tile: tile, │ │ │ │ │ - distance: Math.pow(tileCenter.lon - center.lon, 2) + │ │ │ │ │ - Math.pow(tileCenter.lat - center.lat, 2) │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - colidx += 1; │ │ │ │ │ - } while ((tileBounds.right <= bounds.right + tilelon * this.buffer) || │ │ │ │ │ - colidx < minCols); │ │ │ │ │ - │ │ │ │ │ - rowidx += 1; │ │ │ │ │ - } while ((tileBounds.bottom >= bounds.bottom - tilelat * this.buffer) || │ │ │ │ │ - rowidx < minRows); │ │ │ │ │ - │ │ │ │ │ - //shave off exceess rows and colums │ │ │ │ │ - this.removeExcessTiles(rowidx, colidx); │ │ │ │ │ - │ │ │ │ │ - var resolution = this.getServerResolution(); │ │ │ │ │ - // store the resolution of the grid │ │ │ │ │ - this.gridResolution = resolution; │ │ │ │ │ - │ │ │ │ │ - //now actually draw the tiles │ │ │ │ │ - tileData.sort(function(a, b) { │ │ │ │ │ - return a.distance - b.distance; │ │ │ │ │ - }); │ │ │ │ │ - for (var i = 0, ii = tileData.length; i < ii; ++i) { │ │ │ │ │ - tileData[i].tile.draw(); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getMaxExtent │ │ │ │ │ - * Get this layer's maximum extent. (Implemented as a getter for │ │ │ │ │ - * potential specific implementations in sub-classes.) │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} │ │ │ │ │ - */ │ │ │ │ │ - getMaxExtent: function() { │ │ │ │ │ - return this.maxExtent; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: addTile │ │ │ │ │ - * Create a tile, initialize it, and add it to the layer div. │ │ │ │ │ - * │ │ │ │ │ - * Parameters │ │ │ │ │ - * bounds - {} │ │ │ │ │ - * position - {} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} The added OpenLayers.Tile │ │ │ │ │ - */ │ │ │ │ │ - addTile: function(bounds, position) { │ │ │ │ │ - var tile = new this.tileClass( │ │ │ │ │ - this, position, bounds, null, this.tileSize, this.tileOptions │ │ │ │ │ - ); │ │ │ │ │ - this.events.triggerEvent("addtile", { │ │ │ │ │ - tile: tile │ │ │ │ │ - }); │ │ │ │ │ - return tile; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addTileMonitoringHooks │ │ │ │ │ - * This function takes a tile as input and adds the appropriate hooks to │ │ │ │ │ - * the tile so that the layer can keep track of the loading tiles. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * tile - {} │ │ │ │ │ + * Method: registerImageListeners │ │ │ │ │ + * Called when an image contained by the popup loaded. this function │ │ │ │ │ + * updates the popup size, then unregisters the image load listener. │ │ │ │ │ */ │ │ │ │ │ - addTileMonitoringHooks: function(tile) { │ │ │ │ │ - │ │ │ │ │ - var replacingCls = 'olTileReplacing'; │ │ │ │ │ + registerImageListeners: function() { │ │ │ │ │ │ │ │ │ │ - tile.onLoadStart = function() { │ │ │ │ │ - //if that was first tile then trigger a 'loadstart' on the layer │ │ │ │ │ - if (this.loading === false) { │ │ │ │ │ - this.loading = true; │ │ │ │ │ - this.events.triggerEvent("loadstart"); │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("tileloadstart", { │ │ │ │ │ - tile: tile │ │ │ │ │ - }); │ │ │ │ │ - this.numLoadingTiles++; │ │ │ │ │ - if (!this.singleTile && this.backBuffer && this.gridResolution === this.backBufferResolution) { │ │ │ │ │ - OpenLayers.Element.addClass(tile.getTile(), replacingCls); │ │ │ │ │ + // As the images load, this function will call updateSize() to │ │ │ │ │ + // resize the popup to fit the content div (which presumably is now │ │ │ │ │ + // bigger than when the image was not loaded). │ │ │ │ │ + // │ │ │ │ │ + // If the 'panMapIfOutOfView' property is set, we will pan the newly │ │ │ │ │ + // resized popup back into view. │ │ │ │ │ + // │ │ │ │ │ + // Note that this function, when called, will have 'popup' and │ │ │ │ │ + // 'img' properties in the context. │ │ │ │ │ + // │ │ │ │ │ + var onImgLoad = function() { │ │ │ │ │ + if (this.popup.id === null) { // this.popup has been destroyed! │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ - }; │ │ │ │ │ + this.popup.updateSize(); │ │ │ │ │ │ │ │ │ │ - tile.onLoadEnd = function(evt) { │ │ │ │ │ - this.numLoadingTiles--; │ │ │ │ │ - var aborted = evt.type === 'unload'; │ │ │ │ │ - this.events.triggerEvent("tileloaded", { │ │ │ │ │ - tile: tile, │ │ │ │ │ - aborted: aborted │ │ │ │ │ - }); │ │ │ │ │ - if (!this.singleTile && !aborted && this.backBuffer && this.gridResolution === this.backBufferResolution) { │ │ │ │ │ - var tileDiv = tile.getTile(); │ │ │ │ │ - if (OpenLayers.Element.getStyle(tileDiv, 'display') === 'none') { │ │ │ │ │ - var bufferTile = document.getElementById(tile.id + '_bb'); │ │ │ │ │ - if (bufferTile) { │ │ │ │ │ - bufferTile.parentNode.removeChild(bufferTile); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Element.removeClass(tileDiv, replacingCls); │ │ │ │ │ - } │ │ │ │ │ - //if that was the last tile, then trigger a 'loadend' on the layer │ │ │ │ │ - if (this.numLoadingTiles === 0) { │ │ │ │ │ - if (this.backBuffer) { │ │ │ │ │ - if (this.backBuffer.childNodes.length === 0) { │ │ │ │ │ - // no tiles transitioning, remove immediately │ │ │ │ │ - this.removeBackBuffer(); │ │ │ │ │ - } else { │ │ │ │ │ - // wait until transition has ended or delay has passed │ │ │ │ │ - this._transitionElement = aborted ? │ │ │ │ │ - this.div.lastChild : tile.imgDiv; │ │ │ │ │ - var transitionendEvents = this.transitionendEvents; │ │ │ │ │ - for (var i = transitionendEvents.length - 1; i >= 0; --i) { │ │ │ │ │ - OpenLayers.Event.observe(this._transitionElement, │ │ │ │ │ - transitionendEvents[i], │ │ │ │ │ - this._removeBackBuffer); │ │ │ │ │ - } │ │ │ │ │ - // the removal of the back buffer is delayed to prevent │ │ │ │ │ - // flash effects due to the animation of tile displaying │ │ │ │ │ - this.backBufferTimerId = window.setTimeout( │ │ │ │ │ - this._removeBackBuffer, this.removeBackBufferDelay │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.loading = false; │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ + if (this.popup.visible() && this.popup.panMapIfOutOfView) { │ │ │ │ │ + this.popup.panIntoView(); │ │ │ │ │ } │ │ │ │ │ - }; │ │ │ │ │ │ │ │ │ │ - tile.onLoadError = function() { │ │ │ │ │ - this.events.triggerEvent("tileerror", { │ │ │ │ │ - tile: tile │ │ │ │ │ - }); │ │ │ │ │ + OpenLayers.Event.stopObserving( │ │ │ │ │ + this.img, "load", this.img._onImgLoad │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ }; │ │ │ │ │ │ │ │ │ │ - tile.events.on({ │ │ │ │ │ - "loadstart": tile.onLoadStart, │ │ │ │ │ - "loadend": tile.onLoadEnd, │ │ │ │ │ - "unload": tile.onLoadEnd, │ │ │ │ │ - "loaderror": tile.onLoadError, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + //cycle through the images and if their size is 0x0, that means that │ │ │ │ │ + // they haven't been loaded yet, so we attach the listener, which │ │ │ │ │ + // will fire when the images finish loading and will resize the │ │ │ │ │ + // popup accordingly to its new size. │ │ │ │ │ + var images = this.contentDiv.getElementsByTagName("img"); │ │ │ │ │ + for (var i = 0, len = images.length; i < len; i++) { │ │ │ │ │ + var img = images[i]; │ │ │ │ │ + if (img.width == 0 || img.height == 0) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeTileMonitoringHooks │ │ │ │ │ - * This function takes a tile as input and removes the tile hooks │ │ │ │ │ - * that were added in addTileMonitoringHooks() │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * tile - {} │ │ │ │ │ - */ │ │ │ │ │ - removeTileMonitoringHooks: function(tile) { │ │ │ │ │ - tile.unload(); │ │ │ │ │ - tile.events.un({ │ │ │ │ │ - "loadstart": tile.onLoadStart, │ │ │ │ │ - "loadend": tile.onLoadEnd, │ │ │ │ │ - "unload": tile.onLoadEnd, │ │ │ │ │ - "loaderror": tile.onLoadError, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + var context = { │ │ │ │ │ + 'popup': this, │ │ │ │ │ + 'img': img │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveGriddedTiles │ │ │ │ │ - */ │ │ │ │ │ - moveGriddedTiles: function() { │ │ │ │ │ - var buffer = this.buffer + 1; │ │ │ │ │ - while (true) { │ │ │ │ │ - var tlTile = this.grid[0][0]; │ │ │ │ │ - var tlViewPort = { │ │ │ │ │ - x: tlTile.position.x + │ │ │ │ │ - this.map.layerContainerOriginPx.x, │ │ │ │ │ - y: tlTile.position.y + │ │ │ │ │ - this.map.layerContainerOriginPx.y │ │ │ │ │ - }; │ │ │ │ │ - var ratio = this.getServerResolution() / this.map.getResolution(); │ │ │ │ │ - var tileSize = { │ │ │ │ │ - w: Math.round(this.tileSize.w * ratio), │ │ │ │ │ - h: Math.round(this.tileSize.h * ratio) │ │ │ │ │ - }; │ │ │ │ │ - if (tlViewPort.x > -tileSize.w * (buffer - 1)) { │ │ │ │ │ - this.shiftColumn(true, tileSize); │ │ │ │ │ - } else if (tlViewPort.x < -tileSize.w * buffer) { │ │ │ │ │ - this.shiftColumn(false, tileSize); │ │ │ │ │ - } else if (tlViewPort.y > -tileSize.h * (buffer - 1)) { │ │ │ │ │ - this.shiftRow(true, tileSize); │ │ │ │ │ - } else if (tlViewPort.y < -tileSize.h * buffer) { │ │ │ │ │ - this.shiftRow(false, tileSize); │ │ │ │ │ - } else { │ │ │ │ │ - break; │ │ │ │ │ + //expando this function to the image itself before registering │ │ │ │ │ + // it. This way we can easily and properly unregister it. │ │ │ │ │ + img._onImgLoad = OpenLayers.Function.bind(onImgLoad, context); │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Event.observe(img, 'load', img._onImgLoad); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: shiftRow │ │ │ │ │ - * Shifty grid work │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getSafeContentSize │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * prepend - {Boolean} if true, prepend to beginning. │ │ │ │ │ - * if false, then append to end │ │ │ │ │ - * tileSize - {Object} rendered tile size; object with w and h properties │ │ │ │ │ + * size - {} Desired size to make the popup. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} A size to make the popup which is neither smaller │ │ │ │ │ + * than the specified minimum size, nor bigger than the maximum │ │ │ │ │ + * size (which is calculated relative to the size of the viewport). │ │ │ │ │ */ │ │ │ │ │ - shiftRow: function(prepend, tileSize) { │ │ │ │ │ - var grid = this.grid; │ │ │ │ │ - var rowIndex = prepend ? 0 : (grid.length - 1); │ │ │ │ │ - var sign = prepend ? -1 : 1; │ │ │ │ │ - var rowSign = this.rowSign; │ │ │ │ │ - var tileLayout = this.gridLayout; │ │ │ │ │ - tileLayout.startrow += sign * rowSign; │ │ │ │ │ - │ │ │ │ │ - var modelRow = grid[rowIndex]; │ │ │ │ │ - var row = grid[prepend ? 'pop' : 'shift'](); │ │ │ │ │ - for (var i = 0, len = row.length; i < len; i++) { │ │ │ │ │ - var tile = row[i]; │ │ │ │ │ - var position = modelRow[i].position.clone(); │ │ │ │ │ - position.y += tileSize.h * sign; │ │ │ │ │ - tile.moveTo(this.getTileBoundsForGridIndex(rowIndex, i), position); │ │ │ │ │ - } │ │ │ │ │ - grid[prepend ? 'unshift' : 'push'](row); │ │ │ │ │ - }, │ │ │ │ │ + getSafeContentSize: function(size) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: shiftColumn │ │ │ │ │ - * Shift grid work in the other dimension │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * prepend - {Boolean} if true, prepend to beginning. │ │ │ │ │ - * if false, then append to end │ │ │ │ │ - * tileSize - {Object} rendered tile size; object with w and h properties │ │ │ │ │ - */ │ │ │ │ │ - shiftColumn: function(prepend, tileSize) { │ │ │ │ │ - var grid = this.grid; │ │ │ │ │ - var colIndex = prepend ? 0 : (grid[0].length - 1); │ │ │ │ │ - var sign = prepend ? -1 : 1; │ │ │ │ │ - var tileLayout = this.gridLayout; │ │ │ │ │ - tileLayout.startcol += sign; │ │ │ │ │ + var safeContentSize = size.clone(); │ │ │ │ │ │ │ │ │ │ - for (var i = 0, len = grid.length; i < len; i++) { │ │ │ │ │ - var row = grid[i]; │ │ │ │ │ - var position = row[colIndex].position.clone(); │ │ │ │ │ - var tile = row[prepend ? 'pop' : 'shift'](); │ │ │ │ │ - position.x += tileSize.w * sign; │ │ │ │ │ - tile.moveTo(this.getTileBoundsForGridIndex(i, colIndex), position); │ │ │ │ │ - row[prepend ? 'unshift' : 'push'](tile); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + // if our contentDiv has a css 'padding' set on it by a stylesheet, we │ │ │ │ │ + // must add that to the desired "size". │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ + var wPadding = contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ + var hPadding = contentDivPadding.top + contentDivPadding.bottom; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeExcessTiles │ │ │ │ │ - * When the size of the map or the buffer changes, we may need to │ │ │ │ │ - * remove some excess rows and columns. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * rows - {Integer} Maximum number of rows we want our grid to have. │ │ │ │ │ - * columns - {Integer} Maximum number of columns we want our grid to have. │ │ │ │ │ - */ │ │ │ │ │ - removeExcessTiles: function(rows, columns) { │ │ │ │ │ - var i, l; │ │ │ │ │ + // take into account the popup's 'padding' property │ │ │ │ │ + this.fixPadding(); │ │ │ │ │ + wPadding += this.padding.left + this.padding.right; │ │ │ │ │ + hPadding += this.padding.top + this.padding.bottom; │ │ │ │ │ │ │ │ │ │ - // remove extra rows │ │ │ │ │ - while (this.grid.length > rows) { │ │ │ │ │ - var row = this.grid.pop(); │ │ │ │ │ - for (i = 0, l = row.length; i < l; i++) { │ │ │ │ │ - var tile = row[i]; │ │ │ │ │ - this.destroyTile(tile); │ │ │ │ │ - } │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + var closeDivWidth = parseInt(this.closeDiv.style.width); │ │ │ │ │ + wPadding += closeDivWidth + contentDivPadding.right; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // remove extra columns │ │ │ │ │ - for (i = 0, l = this.grid.length; i < l; i++) { │ │ │ │ │ - while (this.grid[i].length > columns) { │ │ │ │ │ - var row = this.grid[i]; │ │ │ │ │ - var tile = row.pop(); │ │ │ │ │ - this.destroyTile(tile); │ │ │ │ │ - } │ │ │ │ │ + // prevent the popup from being smaller than a specified minimal size │ │ │ │ │ + if (this.minSize) { │ │ │ │ │ + safeContentSize.w = Math.max(safeContentSize.w, │ │ │ │ │ + (this.minSize.w - wPadding)); │ │ │ │ │ + safeContentSize.h = Math.max(safeContentSize.h, │ │ │ │ │ + (this.minSize.h - hPadding)); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: onMapResize │ │ │ │ │ - * For singleTile layers, this will set a new tile size according to the │ │ │ │ │ - * dimensions of the map pane. │ │ │ │ │ - */ │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - this.clearGrid(); │ │ │ │ │ - this.setTileSize(); │ │ │ │ │ + // prevent the popup from being bigger than a specified maximum size │ │ │ │ │ + if (this.maxSize) { │ │ │ │ │ + safeContentSize.w = Math.min(safeContentSize.w, │ │ │ │ │ + (this.maxSize.w - wPadding)); │ │ │ │ │ + safeContentSize.h = Math.min(safeContentSize.h, │ │ │ │ │ + (this.maxSize.h - hPadding)); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getTileBounds │ │ │ │ │ - * Returns The tile bounds for a layer given a pixel location. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * viewPortPx - {} The location in the viewport. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} Bounds of the tile at the given pixel location. │ │ │ │ │ - */ │ │ │ │ │ - getTileBounds: function(viewPortPx) { │ │ │ │ │ - var maxExtent = this.maxExtent; │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var tileMapWidth = resolution * this.tileSize.w; │ │ │ │ │ - var tileMapHeight = resolution * this.tileSize.h; │ │ │ │ │ - var mapPoint = this.getLonLatFromViewPortPx(viewPortPx); │ │ │ │ │ - var tileLeft = maxExtent.left + (tileMapWidth * │ │ │ │ │ - Math.floor((mapPoint.lon - │ │ │ │ │ - maxExtent.left) / │ │ │ │ │ - tileMapWidth)); │ │ │ │ │ - var tileBottom = maxExtent.bottom + (tileMapHeight * │ │ │ │ │ - Math.floor((mapPoint.lat - │ │ │ │ │ - maxExtent.bottom) / │ │ │ │ │ - tileMapHeight)); │ │ │ │ │ - return new OpenLayers.Bounds(tileLeft, tileBottom, │ │ │ │ │ - tileLeft + tileMapWidth, │ │ │ │ │ - tileBottom + tileMapHeight); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Grid" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/TileManager.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - * @requires OpenLayers/BaseTypes.js │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Element.js │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ - * @requires OpenLayers/Tile/Image.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.TileManager │ │ │ │ │ - * Provides queueing of image requests and caching of image elements. │ │ │ │ │ - * │ │ │ │ │ - * Queueing avoids unnecessary image requests while changing zoom levels │ │ │ │ │ - * quickly, and helps improve dragging performance on mobile devices that show │ │ │ │ │ - * a lag in dragging when loading of new images starts. and │ │ │ │ │ - * are the configuration options to control this behavior. │ │ │ │ │ - * │ │ │ │ │ - * Caching avoids setting the src on image elements for images that have already │ │ │ │ │ - * been used. Several maps can share a TileManager instance, in which case each │ │ │ │ │ - * map gets its own tile queue, but all maps share the same tile cache. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.TileManager = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: cacheSize │ │ │ │ │ - * {Number} Number of image elements to keep referenced in this instance's │ │ │ │ │ - * cache for fast reuse. Default is 256. │ │ │ │ │ - */ │ │ │ │ │ - cacheSize: 256, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: tilesPerFrame │ │ │ │ │ - * {Number} Number of queued tiles to load per frame (see ). │ │ │ │ │ - * Default is 2. │ │ │ │ │ - */ │ │ │ │ │ - tilesPerFrame: 2, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: frameDelay │ │ │ │ │ - * {Number} Delay between tile loading frames (see ) in │ │ │ │ │ - * milliseconds. Default is 16. │ │ │ │ │ - */ │ │ │ │ │ - frameDelay: 16, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: moveDelay │ │ │ │ │ - * {Number} Delay in milliseconds after a map's move event before loading │ │ │ │ │ - * tiles. Default is 100. │ │ │ │ │ - */ │ │ │ │ │ - moveDelay: 100, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomDelay │ │ │ │ │ - * {Number} Delay in milliseconds after a map's zoomend event before loading │ │ │ │ │ - * tiles. Default is 200. │ │ │ │ │ - */ │ │ │ │ │ - zoomDelay: 200, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: maps │ │ │ │ │ - * {Array()} The maps to manage tiles on. │ │ │ │ │ - */ │ │ │ │ │ - maps: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: tileQueueId │ │ │ │ │ - * {Object} The ids of the loop, keyed by map id. │ │ │ │ │ - */ │ │ │ │ │ - tileQueueId: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: tileQueue │ │ │ │ │ - * {Object(Array())} Tiles queued for drawing, keyed by │ │ │ │ │ - * map id. │ │ │ │ │ - */ │ │ │ │ │ - tileQueue: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: tileCache │ │ │ │ │ - * {Object} Cached image elements, keyed by URL. │ │ │ │ │ - */ │ │ │ │ │ - tileCache: null, │ │ │ │ │ + //make sure the desired size to set doesn't result in a popup that │ │ │ │ │ + // is bigger than the map's viewport. │ │ │ │ │ + // │ │ │ │ │ + if (this.map && this.map.size) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: tileCacheIndex │ │ │ │ │ - * {Array(String)} URLs of cached tiles. First entry is the least recently │ │ │ │ │ - * used. │ │ │ │ │ - */ │ │ │ │ │ - tileCacheIndex: null, │ │ │ │ │ + var extraX = 0, │ │ │ │ │ + extraY = 0; │ │ │ │ │ + if (this.keepInMap && !this.panMapIfOutOfView) { │ │ │ │ │ + var px = this.map.getPixelFromLonLat(this.lonlat); │ │ │ │ │ + switch (this.relativePosition) { │ │ │ │ │ + case "tr": │ │ │ │ │ + extraX = px.x; │ │ │ │ │ + extraY = this.map.size.h - px.y; │ │ │ │ │ + break; │ │ │ │ │ + case "tl": │ │ │ │ │ + extraX = this.map.size.w - px.x; │ │ │ │ │ + extraY = this.map.size.h - px.y; │ │ │ │ │ + break; │ │ │ │ │ + case "bl": │ │ │ │ │ + extraX = this.map.size.w - px.x; │ │ │ │ │ + extraY = px.y; │ │ │ │ │ + break; │ │ │ │ │ + case "br": │ │ │ │ │ + extraX = px.x; │ │ │ │ │ + extraY = px.y; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + extraX = px.x; │ │ │ │ │ + extraY = this.map.size.h - px.y; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.TileManager │ │ │ │ │ - * Constructor for a new instance. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Configuration for this instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.maps = []; │ │ │ │ │ - this.tileQueueId = {}; │ │ │ │ │ - this.tileQueue = {}; │ │ │ │ │ - this.tileCache = {}; │ │ │ │ │ - this.tileCacheIndex = []; │ │ │ │ │ - }, │ │ │ │ │ + var maxY = this.map.size.h - │ │ │ │ │ + this.map.paddingForPopups.top - │ │ │ │ │ + this.map.paddingForPopups.bottom - │ │ │ │ │ + hPadding - extraY; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: addMap │ │ │ │ │ - * Binds this instance to a map │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {} │ │ │ │ │ - */ │ │ │ │ │ - addMap: function(map) { │ │ │ │ │ - if (this._destroyed || !OpenLayers.Layer.Grid) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - this.maps.push(map); │ │ │ │ │ - this.tileQueue[map.id] = []; │ │ │ │ │ - for (var i = 0, ii = map.layers.length; i < ii; ++i) { │ │ │ │ │ - this.addLayer({ │ │ │ │ │ - layer: map.layers[i] │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - map.events.on({ │ │ │ │ │ - move: this.move, │ │ │ │ │ - zoomend: this.zoomEnd, │ │ │ │ │ - changelayer: this.changeLayer, │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - preremovelayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + var maxX = this.map.size.w - │ │ │ │ │ + this.map.paddingForPopups.left - │ │ │ │ │ + this.map.paddingForPopups.right - │ │ │ │ │ + wPadding - extraX; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeMap │ │ │ │ │ - * Unbinds this instance from a map │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {} │ │ │ │ │ - */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - if (this._destroyed || !OpenLayers.Layer.Grid) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - window.clearTimeout(this.tileQueueId[map.id]); │ │ │ │ │ - if (map.layers) { │ │ │ │ │ - for (var i = 0, ii = map.layers.length; i < ii; ++i) { │ │ │ │ │ - this.removeLayer({ │ │ │ │ │ - layer: map.layers[i] │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (map.events) { │ │ │ │ │ - map.events.un({ │ │ │ │ │ - move: this.move, │ │ │ │ │ - zoomend: this.zoomEnd, │ │ │ │ │ - changelayer: this.changeLayer, │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - preremovelayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + safeContentSize.w = Math.min(safeContentSize.w, maxX); │ │ │ │ │ + safeContentSize.h = Math.min(safeContentSize.h, maxY); │ │ │ │ │ } │ │ │ │ │ - delete this.tileQueue[map.id]; │ │ │ │ │ - delete this.tileQueueId[map.id]; │ │ │ │ │ - OpenLayers.Util.removeItem(this.maps, map); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * Handles the map's move event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} Listener argument │ │ │ │ │ - */ │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - this.updateTimeout(evt.object, this.moveDelay, true); │ │ │ │ │ + return safeContentSize; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: zoomEnd │ │ │ │ │ - * Handles the map's zoomEnd event │ │ │ │ │ + * Method: getContentDivPadding │ │ │ │ │ + * Glorious, oh glorious hack in order to determine the css 'padding' of │ │ │ │ │ + * the contentDiv. IE/Opera return null here unless we actually add the │ │ │ │ │ + * popup's main 'div' element (which contains contentDiv) to the DOM. │ │ │ │ │ + * So we make it invisible and then add it to the document temporarily. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} Listener argument │ │ │ │ │ - */ │ │ │ │ │ - zoomEnd: function(evt) { │ │ │ │ │ - this.updateTimeout(evt.object, this.zoomDelay); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: changeLayer │ │ │ │ │ - * Handles the map's changeLayer event │ │ │ │ │ + * Once we've taken the padding readings we need, we then remove it │ │ │ │ │ + * from the DOM (it will actually get added to the DOM in │ │ │ │ │ + * Map.js's addPopup) │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} Listener argument │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} │ │ │ │ │ */ │ │ │ │ │ - changeLayer: function(evt) { │ │ │ │ │ - if (evt.property === 'visibility' || evt.property === 'params') { │ │ │ │ │ - this.updateTimeout(evt.object, 0); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + getContentDivPadding: function() { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: addLayer │ │ │ │ │ - * Handles the map's addlayer event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} The listener argument │ │ │ │ │ - */ │ │ │ │ │ - addLayer: function(evt) { │ │ │ │ │ - var layer = evt.layer; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ - layer.events.on({ │ │ │ │ │ - addtile: this.addTile, │ │ │ │ │ - retile: this.clearTileQueue, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - var i, j, tile; │ │ │ │ │ - for (i = layer.grid.length - 1; i >= 0; --i) { │ │ │ │ │ - for (j = layer.grid[i].length - 1; j >= 0; --j) { │ │ │ │ │ - tile = layer.grid[i][j]; │ │ │ │ │ - this.addTile({ │ │ │ │ │ - tile: tile │ │ │ │ │ - }); │ │ │ │ │ - if (tile.url && !tile.imgDiv) { │ │ │ │ │ - this.manageTileCache({ │ │ │ │ │ - object: tile │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + //use cached value if we have it │ │ │ │ │ + var contentDivPadding = this._contentDivPadding; │ │ │ │ │ + if (!contentDivPadding) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeLayer │ │ │ │ │ - * Handles the map's preremovelayer event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} The listener argument │ │ │ │ │ - */ │ │ │ │ │ - removeLayer: function(evt) { │ │ │ │ │ - var layer = evt.layer; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ - this.clearTileQueue({ │ │ │ │ │ - object: layer │ │ │ │ │ - }); │ │ │ │ │ - if (layer.events) { │ │ │ │ │ - layer.events.un({ │ │ │ │ │ - addtile: this.addTile, │ │ │ │ │ - retile: this.clearTileQueue, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - if (layer.grid) { │ │ │ │ │ - var i, j, tile; │ │ │ │ │ - for (i = layer.grid.length - 1; i >= 0; --i) { │ │ │ │ │ - for (j = layer.grid[i].length - 1; j >= 0; --j) { │ │ │ │ │ - tile = layer.grid[i][j]; │ │ │ │ │ - this.unloadTile({ │ │ │ │ │ - object: tile │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + if (this.div.parentNode == null) { │ │ │ │ │ + //make the div invisible and add it to the page │ │ │ │ │ + this.div.style.display = "none"; │ │ │ │ │ + document.body.appendChild(this.div); │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateTimeout │ │ │ │ │ - * Applies the or to the loop, │ │ │ │ │ - * and schedules more queue processing after if there are still │ │ │ │ │ - * tiles in the queue. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {} The map to update the timeout for │ │ │ │ │ - * delay - {Number} The delay to apply │ │ │ │ │ - * nice - {Boolean} If true, the timeout function will only be created if │ │ │ │ │ - * the tilequeue is not empty. This is used by the move handler to │ │ │ │ │ - * avoid impacts on dragging performance. For other events, the tile │ │ │ │ │ - * queue may not be populated yet, so we need to set the timer │ │ │ │ │ - * regardless of the queue size. │ │ │ │ │ - */ │ │ │ │ │ - updateTimeout: function(map, delay, nice) { │ │ │ │ │ - window.clearTimeout(this.tileQueueId[map.id]); │ │ │ │ │ - var tileQueue = this.tileQueue[map.id]; │ │ │ │ │ - if (!nice || tileQueue.length) { │ │ │ │ │ - this.tileQueueId[map.id] = window.setTimeout( │ │ │ │ │ - OpenLayers.Function.bind(function() { │ │ │ │ │ - this.drawTilesFromQueue(map); │ │ │ │ │ - if (tileQueue.length) { │ │ │ │ │ - this.updateTimeout(map, this.frameDelay); │ │ │ │ │ - } │ │ │ │ │ - }, this), delay │ │ │ │ │ + //read the padding settings from css, put them in an OL.Bounds │ │ │ │ │ + contentDivPadding = new OpenLayers.Bounds( │ │ │ │ │ + OpenLayers.Element.getStyle(this.contentDiv, "padding-left"), │ │ │ │ │ + OpenLayers.Element.getStyle(this.contentDiv, "padding-bottom"), │ │ │ │ │ + OpenLayers.Element.getStyle(this.contentDiv, "padding-right"), │ │ │ │ │ + OpenLayers.Element.getStyle(this.contentDiv, "padding-top") │ │ │ │ │ ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addTile │ │ │ │ │ - * Listener for the layer's addtile event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} The listener argument │ │ │ │ │ - */ │ │ │ │ │ - addTile: function(evt) { │ │ │ │ │ - if (evt.tile instanceof OpenLayers.Tile.Image) { │ │ │ │ │ - evt.tile.events.on({ │ │ │ │ │ - beforedraw: this.queueTileDraw, │ │ │ │ │ - beforeload: this.manageTileCache, │ │ │ │ │ - loadend: this.addToCache, │ │ │ │ │ - unload: this.unloadTile, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - // Layer has the wrong tile type, so don't handle it any longer │ │ │ │ │ - this.removeLayer({ │ │ │ │ │ - layer: evt.tile.layer │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: unloadTile │ │ │ │ │ - * Listener for the tile's unload event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} The listener argument │ │ │ │ │ - */ │ │ │ │ │ - unloadTile: function(evt) { │ │ │ │ │ - var tile = evt.object; │ │ │ │ │ - tile.events.un({ │ │ │ │ │ - beforedraw: this.queueTileDraw, │ │ │ │ │ - beforeload: this.manageTileCache, │ │ │ │ │ - loadend: this.addToCache, │ │ │ │ │ - unload: this.unloadTile, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Util.removeItem(this.tileQueue[tile.layer.map.id], tile); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: queueTileDraw │ │ │ │ │ - * Adds a tile to the queue that will draw it. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} Listener argument of the tile's beforedraw event │ │ │ │ │ - */ │ │ │ │ │ - queueTileDraw: function(evt) { │ │ │ │ │ - var tile = evt.object; │ │ │ │ │ - var queued = false; │ │ │ │ │ - var layer = tile.layer; │ │ │ │ │ - var url = layer.getURL(tile.bounds); │ │ │ │ │ - var img = this.tileCache[url]; │ │ │ │ │ - if (img && img.className !== 'olTileImage') { │ │ │ │ │ - // cached image no longer valid, e.g. because we're olTileReplacing │ │ │ │ │ - delete this.tileCache[url]; │ │ │ │ │ - OpenLayers.Util.removeItem(this.tileCacheIndex, url); │ │ │ │ │ - img = null; │ │ │ │ │ - } │ │ │ │ │ - // queue only if image with same url not cached already │ │ │ │ │ - if (layer.url && (layer.async || !img)) { │ │ │ │ │ - // add to queue only if not in queue already │ │ │ │ │ - var tileQueue = this.tileQueue[layer.map.id]; │ │ │ │ │ - if (!~OpenLayers.Util.indexOf(tileQueue, tile)) { │ │ │ │ │ - tileQueue.push(tile); │ │ │ │ │ - } │ │ │ │ │ - queued = true; │ │ │ │ │ - } │ │ │ │ │ - return !queued; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawTilesFromQueue │ │ │ │ │ - * Draws tiles from the tileQueue, and unqueues the tiles │ │ │ │ │ - */ │ │ │ │ │ - drawTilesFromQueue: function(map) { │ │ │ │ │ - var tileQueue = this.tileQueue[map.id]; │ │ │ │ │ - var limit = this.tilesPerFrame; │ │ │ │ │ - var animating = map.zoomTween && map.zoomTween.playing; │ │ │ │ │ - while (!animating && tileQueue.length && limit) { │ │ │ │ │ - tileQueue.shift().draw(true); │ │ │ │ │ - --limit; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + //cache the value │ │ │ │ │ + this._contentDivPadding = contentDivPadding; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: manageTileCache │ │ │ │ │ - * Adds, updates, removes and fetches cache entries. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} Listener argument of the tile's beforeload event │ │ │ │ │ - */ │ │ │ │ │ - manageTileCache: function(evt) { │ │ │ │ │ - var tile = evt.object; │ │ │ │ │ - var img = this.tileCache[tile.url]; │ │ │ │ │ - if (img) { │ │ │ │ │ - // if image is on its layer's backbuffer, remove it from backbuffer │ │ │ │ │ - if (img.parentNode && │ │ │ │ │ - OpenLayers.Element.hasClass(img.parentNode, 'olBackBuffer')) { │ │ │ │ │ - img.parentNode.removeChild(img); │ │ │ │ │ - img.id = null; │ │ │ │ │ - } │ │ │ │ │ - // only use image from cache if it is not on a layer already │ │ │ │ │ - if (!img.parentNode) { │ │ │ │ │ - img.style.visibility = 'hidden'; │ │ │ │ │ - img.style.opacity = 0; │ │ │ │ │ - tile.setImage(img); │ │ │ │ │ - // LRU - move tile to the end of the array to mark it as the most │ │ │ │ │ - // recently used │ │ │ │ │ - OpenLayers.Util.removeItem(this.tileCacheIndex, tile.url); │ │ │ │ │ - this.tileCacheIndex.push(tile.url); │ │ │ │ │ + if (this.div.parentNode == document.body) { │ │ │ │ │ + //remove the div from the page and make it visible again │ │ │ │ │ + document.body.removeChild(this.div); │ │ │ │ │ + this.div.style.display = ""; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return contentDivPadding; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: addToCache │ │ │ │ │ - * │ │ │ │ │ + * Method: addCloseBox │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} Listener argument for the tile's loadend event │ │ │ │ │ + * callback - {Function} The callback to be called when the close button │ │ │ │ │ + * is clicked. │ │ │ │ │ */ │ │ │ │ │ - addToCache: function(evt) { │ │ │ │ │ - var tile = evt.object; │ │ │ │ │ - if (!this.tileCache[tile.url]) { │ │ │ │ │ - if (!OpenLayers.Element.hasClass(tile.imgDiv, 'olImageLoadError')) { │ │ │ │ │ - if (this.tileCacheIndex.length >= this.cacheSize) { │ │ │ │ │ - delete this.tileCache[this.tileCacheIndex[0]]; │ │ │ │ │ - this.tileCacheIndex.shift(); │ │ │ │ │ - } │ │ │ │ │ - this.tileCache[tile.url] = tile.imgDiv; │ │ │ │ │ - this.tileCacheIndex.push(tile.url); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + addCloseBox: function(callback) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clearTileQueue │ │ │ │ │ - * Clears the tile queue from tiles of a specific layer │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} Listener argument of the layer's retile event │ │ │ │ │ - */ │ │ │ │ │ - clearTileQueue: function(evt) { │ │ │ │ │ - var layer = evt.object; │ │ │ │ │ - var tileQueue = this.tileQueue[layer.map.id]; │ │ │ │ │ - for (var i = tileQueue.length - 1; i >= 0; --i) { │ │ │ │ │ - if (tileQueue[i].layer === layer) { │ │ │ │ │ - tileQueue.splice(i, 1); │ │ │ │ │ + this.closeDiv = OpenLayers.Util.createDiv( │ │ │ │ │ + this.id + "_close", null, { │ │ │ │ │ + w: 17, │ │ │ │ │ + h: 17 │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i = this.maps.length - 1; i >= 0; --i) { │ │ │ │ │ - this.removeMap(this.maps[i]); │ │ │ │ │ - } │ │ │ │ │ - this.maps = null; │ │ │ │ │ - this.tileQueue = null; │ │ │ │ │ - this.tileQueueId = null; │ │ │ │ │ - this.tileCache = null; │ │ │ │ │ - this.tileCacheIndex = null; │ │ │ │ │ - this._destroyed = true; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control │ │ │ │ │ - * Controls affect the display or behavior of the map. They allow everything │ │ │ │ │ - * from panning and zooming to displaying a scale indicator. Controls by │ │ │ │ │ - * default are added to the map they are contained within however it is │ │ │ │ │ - * possible to add a control to an external div by passing the div in the │ │ │ │ │ - * options parameter. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * The following example shows how to add many of the common controls │ │ │ │ │ - * to a map. │ │ │ │ │ - * │ │ │ │ │ - * > var map = new OpenLayers.Map('map', { controls: [] }); │ │ │ │ │ - * > │ │ │ │ │ - * > map.addControl(new OpenLayers.Control.PanZoomBar()); │ │ │ │ │ - * > map.addControl(new OpenLayers.Control.LayerSwitcher({'ascending':false})); │ │ │ │ │ - * > map.addControl(new OpenLayers.Control.Permalink()); │ │ │ │ │ - * > map.addControl(new OpenLayers.Control.Permalink('permalink')); │ │ │ │ │ - * > map.addControl(new OpenLayers.Control.MousePosition()); │ │ │ │ │ - * > map.addControl(new OpenLayers.Control.OverviewMap()); │ │ │ │ │ - * > map.addControl(new OpenLayers.Control.KeyboardDefaults()); │ │ │ │ │ - * │ │ │ │ │ - * The next code fragment is a quick example of how to intercept │ │ │ │ │ - * shift-mouse click to display the extent of the bounding box │ │ │ │ │ - * dragged out by the user. Usually controls are not created │ │ │ │ │ - * in exactly this manner. See the source for a more complete │ │ │ │ │ - * example: │ │ │ │ │ - * │ │ │ │ │ - * > var control = new OpenLayers.Control(); │ │ │ │ │ - * > OpenLayers.Util.extend(control, { │ │ │ │ │ - * > draw: function () { │ │ │ │ │ - * > // this Handler.Box will intercept the shift-mousedown │ │ │ │ │ - * > // before Control.MouseDefault gets to see it │ │ │ │ │ - * > this.box = new OpenLayers.Handler.Box( control, │ │ │ │ │ - * > {"done": this.notice}, │ │ │ │ │ - * > {keyMask: OpenLayers.Handler.MOD_SHIFT}); │ │ │ │ │ - * > this.box.activate(); │ │ │ │ │ - * > }, │ │ │ │ │ - * > │ │ │ │ │ - * > notice: function (bounds) { │ │ │ │ │ - * > OpenLayers.Console.userError(bounds); │ │ │ │ │ - * > } │ │ │ │ │ - * > }); │ │ │ │ │ - * > map.addControl(control); │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: id │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - id: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: map │ │ │ │ │ - * {} this gets set in the addControl() function in │ │ │ │ │ - * OpenLayers.Map │ │ │ │ │ - */ │ │ │ │ │ - map: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: div │ │ │ │ │ - * {DOMElement} The element that contains the control, if not present the │ │ │ │ │ - * control is placed inside the map. │ │ │ │ │ - */ │ │ │ │ │ - div: null, │ │ │ │ │ + ); │ │ │ │ │ + this.closeDiv.className = "olPopupCloseBox"; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: type │ │ │ │ │ - * {Number} Controls can have a 'type'. The type determines the type of │ │ │ │ │ - * interactions which are possible with them when they are placed in an │ │ │ │ │ - * . │ │ │ │ │ - */ │ │ │ │ │ - type: null, │ │ │ │ │ + // use the content div's css padding to determine if we should │ │ │ │ │ + // padd the close div │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: allowSelection │ │ │ │ │ - * {Boolean} By default, controls do not allow selection, because │ │ │ │ │ - * it may interfere with map dragging. If this is true, OpenLayers │ │ │ │ │ - * will not prevent selection of the control. │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - allowSelection: false, │ │ │ │ │ + this.closeDiv.style.right = contentDivPadding.right + "px"; │ │ │ │ │ + this.closeDiv.style.top = contentDivPadding.top + "px"; │ │ │ │ │ + this.groupDiv.appendChild(this.closeDiv); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: displayClass │ │ │ │ │ - * {string} This property is used for CSS related to the drawing of the │ │ │ │ │ - * Control. │ │ │ │ │ - */ │ │ │ │ │ - displayClass: "", │ │ │ │ │ + var closePopup = callback || function(e) { │ │ │ │ │ + this.hide(); │ │ │ │ │ + OpenLayers.Event.stop(e); │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Event.observe(this.closeDiv, "touchend", │ │ │ │ │ + OpenLayers.Function.bindAsEventListener(closePopup, this)); │ │ │ │ │ + OpenLayers.Event.observe(this.closeDiv, "click", │ │ │ │ │ + OpenLayers.Function.bindAsEventListener(closePopup, this)); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: title │ │ │ │ │ - * {string} This property is used for showing a tooltip over the │ │ │ │ │ - * Control. │ │ │ │ │ + * Method: panIntoView │ │ │ │ │ + * Pans the map such that the popup is totaly viewable (if necessary) │ │ │ │ │ */ │ │ │ │ │ - title: "", │ │ │ │ │ + panIntoView: function() { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * false. │ │ │ │ │ - */ │ │ │ │ │ - autoActivate: false, │ │ │ │ │ + var mapSize = this.map.getSize(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: active │ │ │ │ │ - * {Boolean} The control is active (read-only). Use and │ │ │ │ │ - * to change control state. │ │ │ │ │ - */ │ │ │ │ │ - active: null, │ │ │ │ │ + //start with the top left corner of the popup, in px, │ │ │ │ │ + // relative to the viewport │ │ │ │ │ + var origTL = this.map.getViewPortPxFromLayerPx(new OpenLayers.Pixel( │ │ │ │ │ + parseInt(this.div.style.left), │ │ │ │ │ + parseInt(this.div.style.top) │ │ │ │ │ + )); │ │ │ │ │ + var newTL = origTL.clone(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: handlerOptions │ │ │ │ │ - * {Object} Used to set non-default properties on the control's handler │ │ │ │ │ - */ │ │ │ │ │ - handlerOptions: null, │ │ │ │ │ + //new left (compare to margins, using this.size to calculate right) │ │ │ │ │ + if (origTL.x < this.map.paddingForPopups.left) { │ │ │ │ │ + newTL.x = this.map.paddingForPopups.left; │ │ │ │ │ + } else │ │ │ │ │ + if ((origTL.x + this.size.w) > (mapSize.w - this.map.paddingForPopups.right)) { │ │ │ │ │ + newTL.x = mapSize.w - this.map.paddingForPopups.right - this.size.w; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: handler │ │ │ │ │ - * {} null │ │ │ │ │ - */ │ │ │ │ │ - handler: null, │ │ │ │ │ + //new top (compare to margins, using this.size to calculate bottom) │ │ │ │ │ + if (origTL.y < this.map.paddingForPopups.top) { │ │ │ │ │ + newTL.y = this.map.paddingForPopups.top; │ │ │ │ │ + } else │ │ │ │ │ + if ((origTL.y + this.size.h) > (mapSize.h - this.map.paddingForPopups.bottom)) { │ │ │ │ │ + newTL.y = mapSize.h - this.map.paddingForPopups.bottom - this.size.h; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: eventListeners │ │ │ │ │ - * {Object} If set as an option at construction, the eventListeners │ │ │ │ │ - * object will be registered with . Object │ │ │ │ │ - * structure must be a listeners object as shown in the example for │ │ │ │ │ - * the events.on method. │ │ │ │ │ - */ │ │ │ │ │ - eventListeners: null, │ │ │ │ │ + var dx = origTL.x - newTL.x; │ │ │ │ │ + var dy = origTL.y - newTL.y; │ │ │ │ │ + │ │ │ │ │ + this.map.pan(dx, dy); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Listeners will be called with a reference to an event object. The │ │ │ │ │ - * properties of this event depends on exactly what happened. │ │ │ │ │ - * │ │ │ │ │ - * All event objects have at least the following properties: │ │ │ │ │ - * object - {Object} A reference to control.events.object (a reference │ │ │ │ │ - * to the control). │ │ │ │ │ - * element - {DOMElement} A reference to control.events.element (which │ │ │ │ │ - * will be null unless documented otherwise). │ │ │ │ │ + * Method: registerEvents │ │ │ │ │ + * Registers events on the popup. │ │ │ │ │ * │ │ │ │ │ - * Supported map event types: │ │ │ │ │ - * activate - Triggered when activated. │ │ │ │ │ - * deactivate - Triggered when deactivated. │ │ │ │ │ - */ │ │ │ │ │ - events: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control │ │ │ │ │ - * Create an OpenLayers Control. The options passed as a parameter │ │ │ │ │ - * directly extend the control. For example passing the following: │ │ │ │ │ + * Do this in a separate function so that subclasses can │ │ │ │ │ + * choose to override it if they wish to deal differently │ │ │ │ │ + * with mouse events │ │ │ │ │ * │ │ │ │ │ - * > var control = new OpenLayers.Control({div: myDiv}); │ │ │ │ │ - * │ │ │ │ │ - * Overrides the default div attribute value of null. │ │ │ │ │ + * Note in the following handler functions that some special │ │ │ │ │ + * care is needed to deal correctly with mousing and popups. │ │ │ │ │ + * │ │ │ │ │ + * Because the user might select the zoom-rectangle option and │ │ │ │ │ + * then drag it over a popup, we need a safe way to allow the │ │ │ │ │ + * mousemove and mouseup events to pass through the popup when │ │ │ │ │ + * they are initiated from outside. The same procedure is needed for │ │ │ │ │ + * touchmove and touchend events. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * Otherwise, we want to essentially kill the event propagation │ │ │ │ │ + * for all other events, though we have to do so carefully, │ │ │ │ │ + * without disabling basic html functionality, like clicking on │ │ │ │ │ + * hyperlinks or drag-selecting text. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - // We do this before the extend so that instances can override │ │ │ │ │ - // className in options. │ │ │ │ │ - this.displayClass = │ │ │ │ │ - this.CLASS_NAME.replace("OpenLayers.", "ol").replace(/\./g, ""); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - │ │ │ │ │ - this.events = new OpenLayers.Events(this); │ │ │ │ │ - if (this.eventListeners instanceof Object) { │ │ │ │ │ - this.events.on(this.eventListeners); │ │ │ │ │ - } │ │ │ │ │ - if (this.id == null) { │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + registerEvents: function() { │ │ │ │ │ + this.events = new OpenLayers.Events(this, this.div, null, true); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * The destroy method is used to perform any clean up before the control │ │ │ │ │ - * is dereferenced. Typically this is where event listeners are removed │ │ │ │ │ - * to prevent memory leaks. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.events) { │ │ │ │ │ - if (this.eventListeners) { │ │ │ │ │ - this.events.un(this.eventListeners); │ │ │ │ │ - } │ │ │ │ │ - this.events.destroy(); │ │ │ │ │ - this.events = null; │ │ │ │ │ + function onTouchstart(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt, true); │ │ │ │ │ } │ │ │ │ │ - this.eventListeners = null; │ │ │ │ │ + this.events.on({ │ │ │ │ │ + "mousedown": this.onmousedown, │ │ │ │ │ + "mousemove": this.onmousemove, │ │ │ │ │ + "mouseup": this.onmouseup, │ │ │ │ │ + "click": this.onclick, │ │ │ │ │ + "mouseout": this.onmouseout, │ │ │ │ │ + "dblclick": this.ondblclick, │ │ │ │ │ + "touchstart": onTouchstart, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - // eliminate circular references │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.destroy(); │ │ │ │ │ - this.handler = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.handlers) { │ │ │ │ │ - for (var key in this.handlers) { │ │ │ │ │ - if (this.handlers.hasOwnProperty(key) && │ │ │ │ │ - typeof this.handlers[key].destroy == "function") { │ │ │ │ │ - this.handlers[key].destroy(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.handlers = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.removeControl(this); │ │ │ │ │ - this.map = null; │ │ │ │ │ - } │ │ │ │ │ - this.div = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the control. This is done through an accessor │ │ │ │ │ - * so that subclasses can override this and take special action once │ │ │ │ │ - * they have their map variable set. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.map = map; │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.setMap(map); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * The draw method is called when the control is ready to be displayed │ │ │ │ │ - * on the page. If a div has not been created one is created. Controls │ │ │ │ │ - * with a visual component will almost always want to override this method │ │ │ │ │ - * to customize the look of control. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {} The top-left pixel position of the control │ │ │ │ │ - * or null. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ - */ │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - if (this.div == null) { │ │ │ │ │ - this.div = OpenLayers.Util.createDiv(this.id); │ │ │ │ │ - this.div.className = this.displayClass; │ │ │ │ │ - if (!this.allowSelection) { │ │ │ │ │ - this.div.className += " olControlNoSelect"; │ │ │ │ │ - this.div.setAttribute("unselectable", "on", 0); │ │ │ │ │ - this.div.onselectstart = OpenLayers.Function.False; │ │ │ │ │ - } │ │ │ │ │ - if (this.title != "") { │ │ │ │ │ - this.div.title = this.title; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (px != null) { │ │ │ │ │ - this.position = px.clone(); │ │ │ │ │ - } │ │ │ │ │ - this.moveTo(this.position); │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * Sets the left and top style attributes to the passed in pixel │ │ │ │ │ - * coordinates. │ │ │ │ │ - * │ │ │ │ │ + * Method: onmousedown │ │ │ │ │ + * When mouse goes down within the popup, make a note of │ │ │ │ │ + * it locally, and then do not propagate the mousedown │ │ │ │ │ + * (but do so safely so that user can select text inside) │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * px - {} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - if ((px != null) && (this.div != null)) { │ │ │ │ │ - this.div.style.left = px.x + "px"; │ │ │ │ │ - this.div.style.top = px.y + "px"; │ │ │ │ │ - } │ │ │ │ │ + onmousedown: function(evt) { │ │ │ │ │ + this.mousedown = true; │ │ │ │ │ + OpenLayers.Event.stop(evt, true); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Explicitly activates a control and it's associated │ │ │ │ │ - * handler if one has been set. Controls can be │ │ │ │ │ - * deactivated by calling the deactivate() method. │ │ │ │ │ + /** │ │ │ │ │ + * Method: onmousemove │ │ │ │ │ + * If the drag was started within the popup, then │ │ │ │ │ + * do not propagate the mousemove (but do so safely │ │ │ │ │ + * so that user can select text inside) │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} True if the control was successfully activated or │ │ │ │ │ - * false if the control was already active. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.activate(); │ │ │ │ │ - } │ │ │ │ │ - this.active = true; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - OpenLayers.Element.addClass( │ │ │ │ │ - this.map.viewPortDiv, │ │ │ │ │ - this.displayClass.replace(/ /g, "") + "Active" │ │ │ │ │ - ); │ │ │ │ │ + onmousemove: function(evt) { │ │ │ │ │ + if (this.mousedown) { │ │ │ │ │ + OpenLayers.Event.stop(evt, true); │ │ │ │ │ } │ │ │ │ │ - this.events.triggerEvent("activate"); │ │ │ │ │ - return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivates a control and it's associated handler if any. The exact │ │ │ │ │ - * effect of this depends on the control itself. │ │ │ │ │ + /** │ │ │ │ │ + * Method: onmouseup │ │ │ │ │ + * When mouse comes up within the popup, after going down │ │ │ │ │ + * in it, reset the flag, and then (once again) do not │ │ │ │ │ + * propagate the event, but do so safely so that user can │ │ │ │ │ + * select text inside │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} True if the control was effectively deactivated or false │ │ │ │ │ - * if the control was already inactive. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.deactivate(); │ │ │ │ │ - } │ │ │ │ │ - this.active = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - OpenLayers.Element.removeClass( │ │ │ │ │ - this.map.viewPortDiv, │ │ │ │ │ - this.displayClass.replace(/ /g, "") + "Active" │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("deactivate"); │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Control.TYPE_BUTTON │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.TYPE_BUTTON = 1; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Control.TYPE_TOGGLE │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.TYPE_TOGGLE = 2; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Control.TYPE_TOOL │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.TYPE_TOOL = 3; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Protocol │ │ │ │ │ - * Abstract vector layer protocol class. Not to be instantiated directly. Use │ │ │ │ │ - * one of the protocol subclasses instead. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: format │ │ │ │ │ - * {} The format used by this protocol. │ │ │ │ │ - */ │ │ │ │ │ - format: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: options │ │ │ │ │ - * {Object} Any options sent to the constructor. │ │ │ │ │ - */ │ │ │ │ │ - options: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: autoDestroy │ │ │ │ │ - * {Boolean} The creator of the protocol can set autoDestroy to false │ │ │ │ │ - * to fully control when the protocol is destroyed. Defaults to │ │ │ │ │ - * true. │ │ │ │ │ - */ │ │ │ │ │ - autoDestroy: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultFilter │ │ │ │ │ - * {} Optional default filter to read requests │ │ │ │ │ - */ │ │ │ │ │ - defaultFilter: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol │ │ │ │ │ - * Abstract class for vector protocols. Create instances of a subclass. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.options = options; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mergeWithDefaultFilter │ │ │ │ │ - * Merge filter passed to the read method with the default one │ │ │ │ │ - * │ │ │ │ │ * Parameters: │ │ │ │ │ - * filter - {} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - mergeWithDefaultFilter: function(filter) { │ │ │ │ │ - var merged; │ │ │ │ │ - if (filter && this.defaultFilter) { │ │ │ │ │ - merged = new OpenLayers.Filter.Logical({ │ │ │ │ │ - type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ - filters: [this.defaultFilter, filter] │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - merged = filter || this.defaultFilter || undefined; │ │ │ │ │ + onmouseup: function(evt) { │ │ │ │ │ + if (this.mousedown) { │ │ │ │ │ + this.mousedown = false; │ │ │ │ │ + OpenLayers.Event.stop(evt, true); │ │ │ │ │ } │ │ │ │ │ - return merged; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up the protocol. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.options = null; │ │ │ │ │ - this.format = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Construct a request for reading new features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} An │ │ │ │ │ - * object, the same object will be passed to the callback function passed │ │ │ │ │ - * if one exists in the options object. │ │ │ │ │ - */ │ │ │ │ │ - read: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.filter = this.mergeWithDefaultFilter(options.filter); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: create │ │ │ │ │ - * Construct a request for writing newly created features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array({})} or │ │ │ │ │ - * {} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} An │ │ │ │ │ - * object, the same object will be passed to the callback function passed │ │ │ │ │ - * if one exists in the options object. │ │ │ │ │ - */ │ │ │ │ │ - create: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: update │ │ │ │ │ - * Construct a request updating modified features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array({})} or │ │ │ │ │ - * {} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} An │ │ │ │ │ - * object, the same object will be passed to the callback function passed │ │ │ │ │ - * if one exists in the options object. │ │ │ │ │ - */ │ │ │ │ │ - update: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: delete │ │ │ │ │ - * Construct a request deleting a removed feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} An │ │ │ │ │ - * object, the same object will be passed to the callback function passed │ │ │ │ │ - * if one exists in the options object. │ │ │ │ │ - */ │ │ │ │ │ - "delete": function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: commit │ │ │ │ │ - * Go over the features and for each take action │ │ │ │ │ - * based on the feature state. Possible actions are create, │ │ │ │ │ - * update and delete. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array({})} │ │ │ │ │ - * options - {Object} Object whose possible keys are "create", "update", │ │ │ │ │ - * "delete", "callback" and "scope", the values referenced by the │ │ │ │ │ - * first three are objects as passed to the "create", "update", and │ │ │ │ │ - * "delete" methods, the value referenced by the "callback" key is │ │ │ │ │ - * a function which is called when the commit operation is complete │ │ │ │ │ - * using the scope referenced by the "scope" key. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array({})} An array of │ │ │ │ │ - * objects. │ │ │ │ │ - */ │ │ │ │ │ - commit: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: abort │ │ │ │ │ - * Abort an ongoing request. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * response - {} │ │ │ │ │ - */ │ │ │ │ │ - abort: function(response) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createCallback │ │ │ │ │ - * Returns a function that applies the given public method with resp and │ │ │ │ │ - * options arguments. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * method - {Function} The method to be applied by the callback. │ │ │ │ │ - * response - {} The protocol response object. │ │ │ │ │ - * options - {Object} Options sent to the protocol method │ │ │ │ │ - */ │ │ │ │ │ - createCallback: function(method, response, options) { │ │ │ │ │ - return OpenLayers.Function.bind(function() { │ │ │ │ │ - method.apply(this, [response, options]); │ │ │ │ │ - }, this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Protocol.Response │ │ │ │ │ - * Protocols return Response objects to their users. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol.Response = OpenLayers.Class({ │ │ │ │ │ - /** │ │ │ │ │ - * Property: code │ │ │ │ │ - * {Number} - OpenLayers.Protocol.Response.SUCCESS or │ │ │ │ │ - * OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ - */ │ │ │ │ │ - code: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: requestType │ │ │ │ │ - * {String} The type of request this response corresponds to. Either │ │ │ │ │ - * "create", "read", "update" or "delete". │ │ │ │ │ - */ │ │ │ │ │ - requestType: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: last │ │ │ │ │ - * {Boolean} - true if this is the last response expected in a commit, │ │ │ │ │ - * false otherwise, defaults to true. │ │ │ │ │ - */ │ │ │ │ │ - last: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Array({})} or {} │ │ │ │ │ - * The features returned in the response by the server. Depending on the │ │ │ │ │ - * protocol's read payload, either features or data will be populated. │ │ │ │ │ - */ │ │ │ │ │ - features: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: data │ │ │ │ │ - * {Object} │ │ │ │ │ - * The data returned in the response by the server. Depending on the │ │ │ │ │ - * protocol's read payload, either features or data will be populated. │ │ │ │ │ - */ │ │ │ │ │ - data: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: reqFeatures │ │ │ │ │ - * {Array({})} or {} │ │ │ │ │ - * The features provided by the user and placed in the request by the │ │ │ │ │ - * protocol. │ │ │ │ │ - */ │ │ │ │ │ - reqFeatures: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: priv │ │ │ │ │ - */ │ │ │ │ │ - priv: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: error │ │ │ │ │ - * {Object} The error object in case a service exception was encountered. │ │ │ │ │ - */ │ │ │ │ │ - error: null, │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.Response │ │ │ │ │ - * │ │ │ │ │ + * Method: onclick │ │ │ │ │ + * Ignore clicks, but allowing default browser handling │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: success │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} - true on success, false otherwise │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - success: function() { │ │ │ │ │ - return this.code > 0; │ │ │ │ │ + onclick: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt, true); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.Response" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Protocol.Response.SUCCESS = 1; │ │ │ │ │ -OpenLayers.Protocol.Response.FAILURE = 0; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Symbolizer.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Symbolizer │ │ │ │ │ - * Base class representing a symbolizer used for feature rendering. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Symbolizer = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zIndex │ │ │ │ │ - * {Number} The zIndex determines the rendering order for a symbolizer. │ │ │ │ │ - * Symbolizers with larger zIndex values are rendered over symbolizers │ │ │ │ │ - * with smaller zIndex values. Default is 0. │ │ │ │ │ - */ │ │ │ │ │ - zIndex: 0, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Symbolizer │ │ │ │ │ - * Instances of this class are not useful. See one of the subclasses. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: onmouseout │ │ │ │ │ + * When mouse goes out of the popup set the flag to false so that │ │ │ │ │ + * if they let go and then drag back in, we won't be confused. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * config - {Object} An object containing properties to be set on the │ │ │ │ │ - * symbolizer. Any documented symbolizer property can be set at │ │ │ │ │ - * construction. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * A new symbolizer. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ - OpenLayers.Util.extend(this, config); │ │ │ │ │ + onmouseout: function(evt) { │ │ │ │ │ + this.mousedown = false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Create a copy of this symbolizer. │ │ │ │ │ - * │ │ │ │ │ - * Returns a symbolizer of the same type with the same properties. │ │ │ │ │ + * Method: ondblclick │ │ │ │ │ + * Ignore double-clicks, but allowing default browser handling │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - var Type = eval(this.CLASS_NAME); │ │ │ │ │ - return new Type(OpenLayers.Util.extend({}, this)); │ │ │ │ │ + ondblclick: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt, true); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Symbolizer" │ │ │ │ │ - │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup" │ │ │ │ │ }); │ │ │ │ │ │ │ │ │ │ +OpenLayers.Popup.WIDTH = 200; │ │ │ │ │ +OpenLayers.Popup.HEIGHT = 200; │ │ │ │ │ +OpenLayers.Popup.COLOR = "white"; │ │ │ │ │ +OpenLayers.Popup.OPACITY = 1; │ │ │ │ │ +OpenLayers.Popup.BORDER = "0px"; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Rule.js │ │ │ │ │ + OpenLayers/Filter.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ @@ -18895,236 +12774,86 @@ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ * @requires OpenLayers/Util.js │ │ │ │ │ * @requires OpenLayers/Style.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Rule │ │ │ │ │ - * This class represents an SLD Rule, as being used for rule-based SLD styling. │ │ │ │ │ + * Class: OpenLayers.Filter │ │ │ │ │ + * This class represents an OGC Filter. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Rule = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: id │ │ │ │ │ - * {String} A unique id for this session. │ │ │ │ │ - */ │ │ │ │ │ - id: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: name │ │ │ │ │ - * {String} name of this rule │ │ │ │ │ - */ │ │ │ │ │ - name: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: title │ │ │ │ │ - * {String} Title of this rule (set if included in SLD) │ │ │ │ │ - */ │ │ │ │ │ - title: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: description │ │ │ │ │ - * {String} Description of this rule (set if abstract is included in SLD) │ │ │ │ │ - */ │ │ │ │ │ - description: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: context │ │ │ │ │ - * {Object} An optional object with properties that the rule should be │ │ │ │ │ - * evaluated against. If no context is specified, feature.attributes will │ │ │ │ │ - * be used. │ │ │ │ │ - */ │ │ │ │ │ - context: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: filter │ │ │ │ │ - * {} Optional filter for the rule. │ │ │ │ │ - */ │ │ │ │ │ - filter: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: elseFilter │ │ │ │ │ - * {Boolean} Determines whether this rule is only to be applied only if │ │ │ │ │ - * no other rules match (ElseFilter according to the SLD specification). │ │ │ │ │ - * Default is false. For instances of OpenLayers.Rule, if elseFilter is │ │ │ │ │ - * false, the rule will always apply. For subclasses, the else property is │ │ │ │ │ - * ignored. │ │ │ │ │ - */ │ │ │ │ │ - elseFilter: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: symbolizer │ │ │ │ │ - * {Object} Symbolizer or hash of symbolizers for this rule. If hash of │ │ │ │ │ - * symbolizers, keys are one or more of ["Point", "Line", "Polygon"]. The │ │ │ │ │ - * latter if useful if it is required to style e.g. vertices of a line │ │ │ │ │ - * with a point symbolizer. Note, however, that this is not implemented │ │ │ │ │ - * yet in OpenLayers, but it is the way how symbolizers are defined in │ │ │ │ │ - * SLD. │ │ │ │ │ - */ │ │ │ │ │ - symbolizer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: symbolizers │ │ │ │ │ - * {Array} Collection of symbolizers associated with this rule. If │ │ │ │ │ - * provided at construction, the symbolizers array has precedence │ │ │ │ │ - * over the deprecated symbolizer property. Note that multiple │ │ │ │ │ - * symbolizers are not currently supported by the vector renderers. │ │ │ │ │ - * Rules with multiple symbolizers are currently only useful for │ │ │ │ │ - * maintaining elements in an SLD document. │ │ │ │ │ - */ │ │ │ │ │ - symbolizers: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: minScaleDenominator │ │ │ │ │ - * {Number} or {String} minimum scale at which to draw the feature. │ │ │ │ │ - * In the case of a String, this can be a combination of text and │ │ │ │ │ - * propertyNames in the form "literal ${propertyName}" │ │ │ │ │ - */ │ │ │ │ │ - minScaleDenominator: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: maxScaleDenominator │ │ │ │ │ - * {Number} or {String} maximum scale at which to draw the feature. │ │ │ │ │ - * In the case of a String, this can be a combination of text and │ │ │ │ │ - * propertyNames in the form "literal ${propertyName}" │ │ │ │ │ - */ │ │ │ │ │ - maxScaleDenominator: null, │ │ │ │ │ +OpenLayers.Filter = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Rule │ │ │ │ │ - * Creates a Rule. │ │ │ │ │ + * Constructor: OpenLayers.Filter │ │ │ │ │ + * This class represents a generic filter. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object with properties to set on the │ │ │ │ │ - * rule │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {} │ │ │ │ │ + * {} │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - this.symbolizer = {}; │ │ │ │ │ OpenLayers.Util.extend(this, options); │ │ │ │ │ - if (this.symbolizers) { │ │ │ │ │ - delete this.symbolizer; │ │ │ │ │ - } │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIMethod: destroy │ │ │ │ │ - * nullify references to prevent circular references and memory leaks │ │ │ │ │ + * Remove reference to anything added. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i in this.symbolizer) { │ │ │ │ │ - this.symbolizer[i] = null; │ │ │ │ │ - } │ │ │ │ │ - this.symbolizer = null; │ │ │ │ │ - delete this.symbolizers; │ │ │ │ │ - }, │ │ │ │ │ + destroy: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIMethod: evaluate │ │ │ │ │ - * evaluates this rule for a specific feature │ │ │ │ │ + * Evaluates this filter in a specific context. Instances or subclasses │ │ │ │ │ + * are supposed to override this method. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {} feature to apply the rule to. │ │ │ │ │ + * context - {Object} Context to use in evaluating the filter. If a vector │ │ │ │ │ + * feature is provided, the feature.attributes will be used as context. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} true if the rule applies, false if it does not. │ │ │ │ │ - * This rule is the default rule and always returns true. │ │ │ │ │ + * {Boolean} The filter applies. │ │ │ │ │ */ │ │ │ │ │ - evaluate: function(feature) { │ │ │ │ │ - var context = this.getContext(feature); │ │ │ │ │ - var applies = true; │ │ │ │ │ - │ │ │ │ │ - if (this.minScaleDenominator || this.maxScaleDenominator) { │ │ │ │ │ - var scale = feature.layer.map.getScale(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // check if within minScale/maxScale bounds │ │ │ │ │ - if (this.minScaleDenominator) { │ │ │ │ │ - applies = scale >= OpenLayers.Style.createLiteral( │ │ │ │ │ - this.minScaleDenominator, context); │ │ │ │ │ - } │ │ │ │ │ - if (applies && this.maxScaleDenominator) { │ │ │ │ │ - applies = scale < OpenLayers.Style.createLiteral( │ │ │ │ │ - this.maxScaleDenominator, context); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // check if optional filter applies │ │ │ │ │ - if (applies && this.filter) { │ │ │ │ │ - // feature id filters get the feature, others get the context │ │ │ │ │ - if (this.filter.CLASS_NAME == "OpenLayers.Filter.FeatureId") { │ │ │ │ │ - applies = this.filter.evaluate(feature); │ │ │ │ │ - } else { │ │ │ │ │ - applies = this.filter.evaluate(context); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return applies; │ │ │ │ │ + evaluate: function(context) { │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getContext │ │ │ │ │ - * Gets the context for evaluating this rule │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Clones this filter. Should be implemented by subclasses. │ │ │ │ │ * │ │ │ │ │ - * Paramters: │ │ │ │ │ - * feature - {} feature to take the context from if │ │ │ │ │ - * none is specified. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} Clone of this filter. │ │ │ │ │ */ │ │ │ │ │ - getContext: function(feature) { │ │ │ │ │ - var context = this.context; │ │ │ │ │ - if (!context) { │ │ │ │ │ - context = feature.attributes || feature.data; │ │ │ │ │ - } │ │ │ │ │ - if (typeof this.context == "function") { │ │ │ │ │ - context = this.context(feature); │ │ │ │ │ - } │ │ │ │ │ - return context; │ │ │ │ │ + clone: function() { │ │ │ │ │ + return null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Clones this rule. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: toString │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {} Clone of this rule. │ │ │ │ │ + * {String} Include in your build to get a CQL │ │ │ │ │ + * representation of the filter returned. Otherwise "[Object object]" │ │ │ │ │ + * will be returned. │ │ │ │ │ */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - var options = OpenLayers.Util.extend({}, this); │ │ │ │ │ - if (this.symbolizers) { │ │ │ │ │ - // clone symbolizers │ │ │ │ │ - var len = this.symbolizers.length; │ │ │ │ │ - options.symbolizers = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - options.symbolizers[i] = this.symbolizers[i].clone(); │ │ │ │ │ - } │ │ │ │ │ + toString: function() { │ │ │ │ │ + var string; │ │ │ │ │ + if (OpenLayers.Format && OpenLayers.Format.CQL) { │ │ │ │ │ + string = OpenLayers.Format.CQL.prototype.write(this); │ │ │ │ │ } else { │ │ │ │ │ - // clone symbolizer │ │ │ │ │ - options.symbolizer = {}; │ │ │ │ │ - var value, type; │ │ │ │ │ - for (var key in this.symbolizer) { │ │ │ │ │ - value = this.symbolizer[key]; │ │ │ │ │ - type = typeof value; │ │ │ │ │ - if (type === "object") { │ │ │ │ │ - options.symbolizer[key] = OpenLayers.Util.extend({}, value); │ │ │ │ │ - } else if (type === "string") { │ │ │ │ │ - options.symbolizer[key] = value; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + string = Object.prototype.toString.call(this); │ │ │ │ │ } │ │ │ │ │ - // clone filter │ │ │ │ │ - options.filter = this.filter && this.filter.clone(); │ │ │ │ │ - // clone context │ │ │ │ │ - options.context = this.context && OpenLayers.Util.extend({}, this.context); │ │ │ │ │ - return new OpenLayers.Rule(options); │ │ │ │ │ + return string; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Rule" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Filter" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Geometry.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ @@ -30673,14 +24402,937 @@ │ │ │ │ │ "ows": OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.WPSExecute" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ + OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +// XMLHttpRequest.js Copyright (C) 2010 Sergey Ilinsky (http://www.ilinsky.com) │ │ │ │ │ +// │ │ │ │ │ +// Licensed under the Apache License, Version 2.0 (the "License"); │ │ │ │ │ +// you may not use this file except in compliance with the License. │ │ │ │ │ +// You may obtain a copy of the License at │ │ │ │ │ +// │ │ │ │ │ +// http://www.apache.org/licenses/LICENSE-2.0 │ │ │ │ │ +// │ │ │ │ │ +// Unless required by applicable law or agreed to in writing, software │ │ │ │ │ +// distributed under the License is distributed on an "AS IS" BASIS, │ │ │ │ │ +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. │ │ │ │ │ +// See the License for the specific language governing permissions and │ │ │ │ │ +// limitations under the License. │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Request.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +(function() { │ │ │ │ │ + │ │ │ │ │ + // Save reference to earlier defined object implementation (if any) │ │ │ │ │ + var oXMLHttpRequest = window.XMLHttpRequest; │ │ │ │ │ + │ │ │ │ │ + // Define on browser type │ │ │ │ │ + var bGecko = !!window.controllers, │ │ │ │ │ + bIE = window.document.all && !window.opera, │ │ │ │ │ + bIE7 = bIE && window.navigator.userAgent.match(/MSIE 7.0/); │ │ │ │ │ + │ │ │ │ │ + // Enables "XMLHttpRequest()" call next to "new XMLHttpReques()" │ │ │ │ │ + function fXMLHttpRequest() { │ │ │ │ │ + this._object = oXMLHttpRequest && !bIE7 ? new oXMLHttpRequest : new window.ActiveXObject("Microsoft.XMLHTTP"); │ │ │ │ │ + this._listeners = []; │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // Constructor │ │ │ │ │ + function cXMLHttpRequest() { │ │ │ │ │ + return new fXMLHttpRequest; │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype = fXMLHttpRequest.prototype; │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: Firefox with Firebug installed would break pages if not executed │ │ │ │ │ + if (bGecko && oXMLHttpRequest.wrapped) │ │ │ │ │ + cXMLHttpRequest.wrapped = oXMLHttpRequest.wrapped; │ │ │ │ │ + │ │ │ │ │ + // Constants │ │ │ │ │ + cXMLHttpRequest.UNSENT = 0; │ │ │ │ │ + cXMLHttpRequest.OPENED = 1; │ │ │ │ │ + cXMLHttpRequest.HEADERS_RECEIVED = 2; │ │ │ │ │ + cXMLHttpRequest.LOADING = 3; │ │ │ │ │ + cXMLHttpRequest.DONE = 4; │ │ │ │ │ + │ │ │ │ │ + // Public Properties │ │ │ │ │ + cXMLHttpRequest.prototype.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + cXMLHttpRequest.prototype.responseText = ''; │ │ │ │ │ + cXMLHttpRequest.prototype.responseXML = null; │ │ │ │ │ + cXMLHttpRequest.prototype.status = 0; │ │ │ │ │ + cXMLHttpRequest.prototype.statusText = ''; │ │ │ │ │ + │ │ │ │ │ + // Priority proposal │ │ │ │ │ + cXMLHttpRequest.prototype.priority = "NORMAL"; │ │ │ │ │ + │ │ │ │ │ + // Instance-level Events Handlers │ │ │ │ │ + cXMLHttpRequest.prototype.onreadystatechange = null; │ │ │ │ │ + │ │ │ │ │ + // Class-level Events Handlers │ │ │ │ │ + cXMLHttpRequest.onreadystatechange = null; │ │ │ │ │ + cXMLHttpRequest.onopen = null; │ │ │ │ │ + cXMLHttpRequest.onsend = null; │ │ │ │ │ + cXMLHttpRequest.onabort = null; │ │ │ │ │ + │ │ │ │ │ + // Public Methods │ │ │ │ │ + cXMLHttpRequest.prototype.open = function(sMethod, sUrl, bAsync, sUser, sPassword) { │ │ │ │ │ + // Delete headers, required when object is reused │ │ │ │ │ + delete this._headers; │ │ │ │ │ + │ │ │ │ │ + // When bAsync parameter value is omitted, use true as default │ │ │ │ │ + if (arguments.length < 3) │ │ │ │ │ + bAsync = true; │ │ │ │ │ + │ │ │ │ │ + // Save async parameter for fixing Gecko bug with missing readystatechange in synchronous requests │ │ │ │ │ + this._async = bAsync; │ │ │ │ │ + │ │ │ │ │ + // Set the onreadystatechange handler │ │ │ │ │ + var oRequest = this, │ │ │ │ │ + nState = this.readyState, │ │ │ │ │ + fOnUnload; │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: IE - memory leak on page unload (inter-page leak) │ │ │ │ │ + if (bIE && bAsync) { │ │ │ │ │ + fOnUnload = function() { │ │ │ │ │ + if (nState != cXMLHttpRequest.DONE) { │ │ │ │ │ + fCleanTransport(oRequest); │ │ │ │ │ + // Safe to abort here since onreadystatechange handler removed │ │ │ │ │ + oRequest.abort(); │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + window.attachEvent("onunload", fOnUnload); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Add method sniffer │ │ │ │ │ + if (cXMLHttpRequest.onopen) │ │ │ │ │ + cXMLHttpRequest.onopen.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + if (arguments.length > 4) │ │ │ │ │ + this._object.open(sMethod, sUrl, bAsync, sUser, sPassword); │ │ │ │ │ + else │ │ │ │ │ + if (arguments.length > 3) │ │ │ │ │ + this._object.open(sMethod, sUrl, bAsync, sUser); │ │ │ │ │ + else │ │ │ │ │ + this._object.open(sMethod, sUrl, bAsync); │ │ │ │ │ + │ │ │ │ │ + this.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ + fReadyStateChange(this); │ │ │ │ │ + │ │ │ │ │ + this._object.onreadystatechange = function() { │ │ │ │ │ + if (bGecko && !bAsync) │ │ │ │ │ + return; │ │ │ │ │ + │ │ │ │ │ + // Synchronize state │ │ │ │ │ + oRequest.readyState = oRequest._object.readyState; │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + fSynchronizeValues(oRequest); │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: Firefox fires unnecessary DONE when aborting │ │ │ │ │ + if (oRequest._aborted) { │ │ │ │ │ + // Reset readyState to UNSENT │ │ │ │ │ + oRequest.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + │ │ │ │ │ + // Return now │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (oRequest.readyState == cXMLHttpRequest.DONE) { │ │ │ │ │ + // Free up queue │ │ │ │ │ + delete oRequest._data; │ │ │ │ │ + /* if (bAsync) │ │ │ │ │ + fQueue_remove(oRequest);*/ │ │ │ │ │ + // │ │ │ │ │ + fCleanTransport(oRequest); │ │ │ │ │ + // Uncomment this block if you need a fix for IE cache │ │ │ │ │ + /* │ │ │ │ │ + // BUGFIX: IE - cache issue │ │ │ │ │ + if (!oRequest._object.getResponseHeader("Date")) { │ │ │ │ │ + // Save object to cache │ │ │ │ │ + oRequest._cached = oRequest._object; │ │ │ │ │ + │ │ │ │ │ + // Instantiate a new transport object │ │ │ │ │ + cXMLHttpRequest.call(oRequest); │ │ │ │ │ + │ │ │ │ │ + // Re-send request │ │ │ │ │ + if (sUser) { │ │ │ │ │ + if (sPassword) │ │ │ │ │ + oRequest._object.open(sMethod, sUrl, bAsync, sUser, sPassword); │ │ │ │ │ + else │ │ │ │ │ + oRequest._object.open(sMethod, sUrl, bAsync, sUser); │ │ │ │ │ + } │ │ │ │ │ + else │ │ │ │ │ + oRequest._object.open(sMethod, sUrl, bAsync); │ │ │ │ │ + oRequest._object.setRequestHeader("If-Modified-Since", oRequest._cached.getResponseHeader("Last-Modified") || new window.Date(0)); │ │ │ │ │ + // Copy headers set │ │ │ │ │ + if (oRequest._headers) │ │ │ │ │ + for (var sHeader in oRequest._headers) │ │ │ │ │ + if (typeof oRequest._headers[sHeader] == "string") // Some frameworks prototype objects with functions │ │ │ │ │ + oRequest._object.setRequestHeader(sHeader, oRequest._headers[sHeader]); │ │ │ │ │ + │ │ │ │ │ + oRequest._object.onreadystatechange = function() { │ │ │ │ │ + // Synchronize state │ │ │ │ │ + oRequest.readyState = oRequest._object.readyState; │ │ │ │ │ + │ │ │ │ │ + if (oRequest._aborted) { │ │ │ │ │ + // │ │ │ │ │ + oRequest.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + │ │ │ │ │ + // Return │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (oRequest.readyState == cXMLHttpRequest.DONE) { │ │ │ │ │ + // Clean Object │ │ │ │ │ + fCleanTransport(oRequest); │ │ │ │ │ + │ │ │ │ │ + // get cached request │ │ │ │ │ + if (oRequest.status == 304) │ │ │ │ │ + oRequest._object = oRequest._cached; │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + delete oRequest._cached; │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + fSynchronizeValues(oRequest); │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + fReadyStateChange(oRequest); │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: IE - memory leak in interrupted │ │ │ │ │ + if (bIE && bAsync) │ │ │ │ │ + window.detachEvent("onunload", fOnUnload); │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + oRequest._object.send(null); │ │ │ │ │ + │ │ │ │ │ + // Return now - wait until re-sent request is finished │ │ │ │ │ + return; │ │ │ │ │ + }; │ │ │ │ │ + */ │ │ │ │ │ + // BUGFIX: IE - memory leak in interrupted │ │ │ │ │ + if (bIE && bAsync) │ │ │ │ │ + window.detachEvent("onunload", fOnUnload); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: Some browsers (Internet Explorer, Gecko) fire OPEN readystate twice │ │ │ │ │ + if (nState != oRequest.readyState) │ │ │ │ │ + fReadyStateChange(oRequest); │ │ │ │ │ + │ │ │ │ │ + nState = oRequest.readyState; │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fXMLHttpRequest_send(oRequest) { │ │ │ │ │ + oRequest._object.send(oRequest._data); │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: Gecko - missing readystatechange calls in synchronous requests │ │ │ │ │ + if (bGecko && !oRequest._async) { │ │ │ │ │ + oRequest.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ + │ │ │ │ │ + // Synchronize state │ │ │ │ │ + fSynchronizeValues(oRequest); │ │ │ │ │ + │ │ │ │ │ + // Simulate missing states │ │ │ │ │ + while (oRequest.readyState < cXMLHttpRequest.DONE) { │ │ │ │ │ + oRequest.readyState++; │ │ │ │ │ + fReadyStateChange(oRequest); │ │ │ │ │ + // Check if we are aborted │ │ │ │ │ + if (oRequest._aborted) │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.send = function(vData) { │ │ │ │ │ + // Add method sniffer │ │ │ │ │ + if (cXMLHttpRequest.onsend) │ │ │ │ │ + cXMLHttpRequest.onsend.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + if (!arguments.length) │ │ │ │ │ + vData = null; │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: Safari - fails sending documents created/modified dynamically, so an explicit serialization required │ │ │ │ │ + // BUGFIX: IE - rewrites any custom mime-type to "text/xml" in case an XMLNode is sent │ │ │ │ │ + // BUGFIX: Gecko - fails sending Element (this is up to the implementation either to standard) │ │ │ │ │ + if (vData && vData.nodeType) { │ │ │ │ │ + vData = window.XMLSerializer ? new window.XMLSerializer().serializeToString(vData) : vData.xml; │ │ │ │ │ + if (!this._headers["Content-Type"]) │ │ │ │ │ + this._object.setRequestHeader("Content-Type", "application/xml"); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this._data = vData; │ │ │ │ │ + /* │ │ │ │ │ + // Add to queue │ │ │ │ │ + if (this._async) │ │ │ │ │ + fQueue_add(this); │ │ │ │ │ + else*/ │ │ │ │ │ + fXMLHttpRequest_send(this); │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.abort = function() { │ │ │ │ │ + // Add method sniffer │ │ │ │ │ + if (cXMLHttpRequest.onabort) │ │ │ │ │ + cXMLHttpRequest.onabort.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: Gecko - unnecessary DONE when aborting │ │ │ │ │ + if (this.readyState > cXMLHttpRequest.UNSENT) │ │ │ │ │ + this._aborted = true; │ │ │ │ │ + │ │ │ │ │ + this._object.abort(); │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: IE - memory leak │ │ │ │ │ + fCleanTransport(this); │ │ │ │ │ + │ │ │ │ │ + this.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + │ │ │ │ │ + delete this._data; │ │ │ │ │ + /* if (this._async) │ │ │ │ │ + fQueue_remove(this);*/ │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.getAllResponseHeaders = function() { │ │ │ │ │ + return this._object.getAllResponseHeaders(); │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.getResponseHeader = function(sName) { │ │ │ │ │ + return this._object.getResponseHeader(sName); │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.setRequestHeader = function(sName, sValue) { │ │ │ │ │ + // BUGFIX: IE - cache issue │ │ │ │ │ + if (!this._headers) │ │ │ │ │ + this._headers = {}; │ │ │ │ │ + this._headers[sName] = sValue; │ │ │ │ │ + │ │ │ │ │ + return this._object.setRequestHeader(sName, sValue); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // EventTarget interface implementation │ │ │ │ │ + cXMLHttpRequest.prototype.addEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ + for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ + if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) │ │ │ │ │ + return; │ │ │ │ │ + // Add listener │ │ │ │ │ + this._listeners.push([sName, fHandler, bUseCapture]); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + cXMLHttpRequest.prototype.removeEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ + for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ + if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) │ │ │ │ │ + break; │ │ │ │ │ + // Remove listener │ │ │ │ │ + if (oListener) │ │ │ │ │ + this._listeners.splice(nIndex, 1); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + cXMLHttpRequest.prototype.dispatchEvent = function(oEvent) { │ │ │ │ │ + var oEventPseudo = { │ │ │ │ │ + 'type': oEvent.type, │ │ │ │ │ + 'target': this, │ │ │ │ │ + 'currentTarget': this, │ │ │ │ │ + 'eventPhase': 2, │ │ │ │ │ + 'bubbles': oEvent.bubbles, │ │ │ │ │ + 'cancelable': oEvent.cancelable, │ │ │ │ │ + 'timeStamp': oEvent.timeStamp, │ │ │ │ │ + 'stopPropagation': function() {}, // There is no flow │ │ │ │ │ + 'preventDefault': function() {}, // There is no default action │ │ │ │ │ + 'initEvent': function() {} // Original event object should be initialized │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // Execute onreadystatechange │ │ │ │ │ + if (oEventPseudo.type == "readystatechange" && this.onreadystatechange) │ │ │ │ │ + (this.onreadystatechange.handleEvent || this.onreadystatechange).apply(this, [oEventPseudo]); │ │ │ │ │ + │ │ │ │ │ + // Execute listeners │ │ │ │ │ + for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ + if (oListener[0] == oEventPseudo.type && !oListener[2]) │ │ │ │ │ + (oListener[1].handleEvent || oListener[1]).apply(this, [oEventPseudo]); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + cXMLHttpRequest.prototype.toString = function() { │ │ │ │ │ + return '[' + "object" + ' ' + "XMLHttpRequest" + ']'; │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + cXMLHttpRequest.toString = function() { │ │ │ │ │ + return '[' + "XMLHttpRequest" + ']'; │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // Helper function │ │ │ │ │ + function fReadyStateChange(oRequest) { │ │ │ │ │ + // Sniffing code │ │ │ │ │ + if (cXMLHttpRequest.onreadystatechange) │ │ │ │ │ + cXMLHttpRequest.onreadystatechange.apply(oRequest); │ │ │ │ │ + │ │ │ │ │ + // Fake event │ │ │ │ │ + oRequest.dispatchEvent({ │ │ │ │ │ + 'type': "readystatechange", │ │ │ │ │ + 'bubbles': false, │ │ │ │ │ + 'cancelable': false, │ │ │ │ │ + 'timeStamp': new Date + 0 │ │ │ │ │ + }); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fGetDocument(oRequest) { │ │ │ │ │ + var oDocument = oRequest.responseXML, │ │ │ │ │ + sResponse = oRequest.responseText; │ │ │ │ │ + // Try parsing responseText │ │ │ │ │ + if (bIE && sResponse && oDocument && !oDocument.documentElement && oRequest.getResponseHeader("Content-Type").match(/[^\/]+\/[^\+]+\+xml/)) { │ │ │ │ │ + oDocument = new window.ActiveXObject("Microsoft.XMLDOM"); │ │ │ │ │ + oDocument.async = false; │ │ │ │ │ + oDocument.validateOnParse = false; │ │ │ │ │ + oDocument.loadXML(sResponse); │ │ │ │ │ + } │ │ │ │ │ + // Check if there is no error in document │ │ │ │ │ + if (oDocument) │ │ │ │ │ + if ((bIE && oDocument.parseError != 0) || !oDocument.documentElement || (oDocument.documentElement && oDocument.documentElement.tagName == "parsererror")) │ │ │ │ │ + return null; │ │ │ │ │ + return oDocument; │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fSynchronizeValues(oRequest) { │ │ │ │ │ + try { │ │ │ │ │ + oRequest.responseText = oRequest._object.responseText; │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + try { │ │ │ │ │ + oRequest.responseXML = fGetDocument(oRequest._object); │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + try { │ │ │ │ │ + oRequest.status = oRequest._object.status; │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + try { │ │ │ │ │ + oRequest.statusText = oRequest._object.statusText; │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fCleanTransport(oRequest) { │ │ │ │ │ + // BUGFIX: IE - memory leak (on-page leak) │ │ │ │ │ + oRequest._object.onreadystatechange = new window.Function; │ │ │ │ │ + }; │ │ │ │ │ + /* │ │ │ │ │ + // Queue manager │ │ │ │ │ + var oQueuePending = {"CRITICAL":[],"HIGH":[],"NORMAL":[],"LOW":[],"LOWEST":[]}, │ │ │ │ │ + aQueueRunning = []; │ │ │ │ │ + function fQueue_add(oRequest) { │ │ │ │ │ + oQueuePending[oRequest.priority in oQueuePending ? oRequest.priority : "NORMAL"].push(oRequest); │ │ │ │ │ + // │ │ │ │ │ + setTimeout(fQueue_process); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fQueue_remove(oRequest) { │ │ │ │ │ + for (var nIndex = 0, bFound = false; nIndex < aQueueRunning.length; nIndex++) │ │ │ │ │ + if (bFound) │ │ │ │ │ + aQueueRunning[nIndex - 1] = aQueueRunning[nIndex]; │ │ │ │ │ + else │ │ │ │ │ + if (aQueueRunning[nIndex] == oRequest) │ │ │ │ │ + bFound = true; │ │ │ │ │ + if (bFound) │ │ │ │ │ + aQueueRunning.length--; │ │ │ │ │ + // │ │ │ │ │ + setTimeout(fQueue_process); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fQueue_process() { │ │ │ │ │ + if (aQueueRunning.length < 6) { │ │ │ │ │ + for (var sPriority in oQueuePending) { │ │ │ │ │ + if (oQueuePending[sPriority].length) { │ │ │ │ │ + var oRequest = oQueuePending[sPriority][0]; │ │ │ │ │ + oQueuePending[sPriority] = oQueuePending[sPriority].slice(1); │ │ │ │ │ + // │ │ │ │ │ + aQueueRunning.push(oRequest); │ │ │ │ │ + // Send request │ │ │ │ │ + fXMLHttpRequest_send(oRequest); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + */ │ │ │ │ │ + // Internet Explorer 5.0 (missing apply) │ │ │ │ │ + if (!window.Function.prototype.apply) { │ │ │ │ │ + window.Function.prototype.apply = function(oRequest, oArguments) { │ │ │ │ │ + if (!oArguments) │ │ │ │ │ + oArguments = []; │ │ │ │ │ + oRequest.__func = this; │ │ │ │ │ + oRequest.__func(oArguments[0], oArguments[1], oArguments[2], oArguments[3], oArguments[4]); │ │ │ │ │ + delete oRequest.__func; │ │ │ │ │ + }; │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // Register new object with window │ │ │ │ │ + /** │ │ │ │ │ + * Class: OpenLayers.Request.XMLHttpRequest │ │ │ │ │ + * Standard-compliant (W3C) cross-browser implementation of the │ │ │ │ │ + * XMLHttpRequest object. From │ │ │ │ │ + * http://code.google.com/p/xmlhttprequest/. │ │ │ │ │ + */ │ │ │ │ │ + if (!OpenLayers.Request) { │ │ │ │ │ + /** │ │ │ │ │ + * This allows for OpenLayers/Request.js to be included │ │ │ │ │ + * before or after this script. │ │ │ │ │ + */ │ │ │ │ │ + OpenLayers.Request = {}; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Request.XMLHttpRequest = cXMLHttpRequest; │ │ │ │ │ +})(); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Request.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ + * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * TODO: deprecate me │ │ │ │ │ + * Use OpenLayers.Request.proxy instead. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.ProxyHost = ""; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Request │ │ │ │ │ + * The OpenLayers.Request namespace contains convenience methods for working │ │ │ │ │ + * with XMLHttpRequests. These methods work with a cross-browser │ │ │ │ │ + * W3C compliant class. │ │ │ │ │ + */ │ │ │ │ │ +if (!OpenLayers.Request) { │ │ │ │ │ + /** │ │ │ │ │ + * This allows for OpenLayers/Request/XMLHttpRequest.js to be included │ │ │ │ │ + * before or after this script. │ │ │ │ │ + */ │ │ │ │ │ + OpenLayers.Request = {}; │ │ │ │ │ +} │ │ │ │ │ +OpenLayers.Util.extend(OpenLayers.Request, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: DEFAULT_CONFIG │ │ │ │ │ + * {Object} Default configuration for all requests. │ │ │ │ │ + */ │ │ │ │ │ + DEFAULT_CONFIG: { │ │ │ │ │ + method: "GET", │ │ │ │ │ + url: window.location.href, │ │ │ │ │ + async: true, │ │ │ │ │ + user: undefined, │ │ │ │ │ + password: undefined, │ │ │ │ │ + params: null, │ │ │ │ │ + proxy: OpenLayers.ProxyHost, │ │ │ │ │ + headers: {}, │ │ │ │ │ + data: null, │ │ │ │ │ + callback: function() {}, │ │ │ │ │ + success: null, │ │ │ │ │ + failure: null, │ │ │ │ │ + scope: null │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: URL_SPLIT_REGEX │ │ │ │ │ + */ │ │ │ │ │ + URL_SPLIT_REGEX: /([^:]*:)\/\/([^:]*:?[^@]*@)?([^:\/\?]*):?([^\/\?]*)/, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {} An events object that handles all │ │ │ │ │ + * events on the {} object. │ │ │ │ │ + * │ │ │ │ │ + * All event listeners will receive an event object with three properties: │ │ │ │ │ + * request - {} The request object. │ │ │ │ │ + * config - {Object} The config object sent to the specific request method. │ │ │ │ │ + * requestUrl - {String} The request url. │ │ │ │ │ + * │ │ │ │ │ + * Supported event types: │ │ │ │ │ + * complete - Triggered when we have a response from the request, if a │ │ │ │ │ + * listener returns false, no further response processing will take │ │ │ │ │ + * place. │ │ │ │ │ + * success - Triggered when the HTTP response has a success code (200-299). │ │ │ │ │ + * failure - Triggered when the HTTP response does not have a success code. │ │ │ │ │ + */ │ │ │ │ │ + events: new OpenLayers.Events(this), │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: makeSameOrigin │ │ │ │ │ + * Using the specified proxy, returns a same origin url of the provided url. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * url - {String} An arbitrary url │ │ │ │ │ + * proxy {String|Function} The proxy to use to make the provided url a │ │ │ │ │ + * same origin url. │ │ │ │ │ + * │ │ │ │ │ + * Returns │ │ │ │ │ + * {String} the same origin url. If no proxy is provided, the returned url │ │ │ │ │ + * will be the same as the provided url. │ │ │ │ │ + */ │ │ │ │ │ + makeSameOrigin: function(url, proxy) { │ │ │ │ │ + var sameOrigin = url.indexOf("http") !== 0; │ │ │ │ │ + var urlParts = !sameOrigin && url.match(this.URL_SPLIT_REGEX); │ │ │ │ │ + if (urlParts) { │ │ │ │ │ + var location = window.location; │ │ │ │ │ + sameOrigin = │ │ │ │ │ + urlParts[1] == location.protocol && │ │ │ │ │ + urlParts[3] == location.hostname; │ │ │ │ │ + var uPort = urlParts[4], │ │ │ │ │ + lPort = location.port; │ │ │ │ │ + if (uPort != 80 && uPort != "" || lPort != "80" && lPort != "") { │ │ │ │ │ + sameOrigin = sameOrigin && uPort == lPort; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!sameOrigin) { │ │ │ │ │ + if (proxy) { │ │ │ │ │ + if (typeof proxy == "function") { │ │ │ │ │ + url = proxy(url); │ │ │ │ │ + } else { │ │ │ │ │ + url = proxy + encodeURIComponent(url); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return url; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: issue │ │ │ │ │ + * Create a new XMLHttpRequest object, open it, set any headers, bind │ │ │ │ │ + * a callback to done state, and send any data. It is recommended that │ │ │ │ │ + * you use one , , , , , or . │ │ │ │ │ + * This method is only documented to provide detail on the configuration │ │ │ │ │ + * options available to all request methods. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Object containing properties for configuring the │ │ │ │ │ + * request. Allowed configuration properties are described below. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Allowed config properties: │ │ │ │ │ + * method - {String} One of GET, POST, PUT, DELETE, HEAD, or │ │ │ │ │ + * OPTIONS. Default is GET. │ │ │ │ │ + * url - {String} URL for the request. │ │ │ │ │ + * async - {Boolean} Open an asynchronous request. Default is true. │ │ │ │ │ + * user - {String} User for relevant authentication scheme. Set │ │ │ │ │ + * to null to clear current user. │ │ │ │ │ + * password - {String} Password for relevant authentication scheme. │ │ │ │ │ + * Set to null to clear current password. │ │ │ │ │ + * proxy - {String} Optional proxy. Defaults to │ │ │ │ │ + * . │ │ │ │ │ + * params - {Object} Any key:value pairs to be appended to the │ │ │ │ │ + * url as a query string. Assumes url doesn't already include a query │ │ │ │ │ + * string or hash. Typically, this is only appropriate for │ │ │ │ │ + * requests where the query string will be appended to the url. │ │ │ │ │ + * Parameter values that are arrays will be │ │ │ │ │ + * concatenated with a comma (note that this goes against form-encoding) │ │ │ │ │ + * as is done with . │ │ │ │ │ + * headers - {Object} Object with header:value pairs to be set on │ │ │ │ │ + * the request. │ │ │ │ │ + * data - {String | Document} Optional data to send with the request. │ │ │ │ │ + * Typically, this is only used with and requests. │ │ │ │ │ + * Make sure to provide the appropriate "Content-Type" header for your │ │ │ │ │ + * data. For and requests, the content type defaults to │ │ │ │ │ + * "application-xml". If your data is a different content type, or │ │ │ │ │ + * if you are using a different HTTP method, set the "Content-Type" │ │ │ │ │ + * header to match your data type. │ │ │ │ │ + * callback - {Function} Function to call when request is done. │ │ │ │ │ + * To determine if the request failed, check request.status (200 │ │ │ │ │ + * indicates success). │ │ │ │ │ + * success - {Function} Optional function to call if request status is in │ │ │ │ │ + * the 200s. This will be called in addition to callback above and │ │ │ │ │ + * would typically only be used as an alternative. │ │ │ │ │ + * failure - {Function} Optional function to call if request status is not │ │ │ │ │ + * in the 200s. This will be called in addition to callback above and │ │ │ │ │ + * would typically only be used as an alternative. │ │ │ │ │ + * scope - {Object} If callback is a public method on some object, │ │ │ │ │ + * set the scope to that object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLHttpRequest} Request object. To abort the request before a response │ │ │ │ │ + * is received, call abort() on the request object. │ │ │ │ │ + */ │ │ │ │ │ + issue: function(config) { │ │ │ │ │ + // apply default config - proxy host may have changed │ │ │ │ │ + var defaultConfig = OpenLayers.Util.extend( │ │ │ │ │ + this.DEFAULT_CONFIG, { │ │ │ │ │ + proxy: OpenLayers.ProxyHost │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + config = config || {}; │ │ │ │ │ + config.headers = config.headers || {}; │ │ │ │ │ + config = OpenLayers.Util.applyDefaults(config, defaultConfig); │ │ │ │ │ + config.headers = OpenLayers.Util.applyDefaults(config.headers, defaultConfig.headers); │ │ │ │ │ + // Always set the "X-Requested-With" header to signal that this request │ │ │ │ │ + // was issued through the XHR-object. Since header keys are case │ │ │ │ │ + // insensitive and we want to allow overriding of the "X-Requested-With" │ │ │ │ │ + // header through the user we cannot use applyDefaults, but have to │ │ │ │ │ + // check manually whether we were called with a "X-Requested-With" │ │ │ │ │ + // header. │ │ │ │ │ + var customRequestedWithHeader = false, │ │ │ │ │ + headerKey; │ │ │ │ │ + for (headerKey in config.headers) { │ │ │ │ │ + if (config.headers.hasOwnProperty(headerKey)) { │ │ │ │ │ + if (headerKey.toLowerCase() === 'x-requested-with') { │ │ │ │ │ + customRequestedWithHeader = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (customRequestedWithHeader === false) { │ │ │ │ │ + // we did not have a custom "X-Requested-With" header │ │ │ │ │ + config.headers['X-Requested-With'] = 'XMLHttpRequest'; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // create request, open, and set headers │ │ │ │ │ + var request = new OpenLayers.Request.XMLHttpRequest(); │ │ │ │ │ + var url = OpenLayers.Util.urlAppend(config.url, │ │ │ │ │ + OpenLayers.Util.getParameterString(config.params || {})); │ │ │ │ │ + url = OpenLayers.Request.makeSameOrigin(url, config.proxy); │ │ │ │ │ + request.open( │ │ │ │ │ + config.method, url, config.async, config.user, config.password │ │ │ │ │ + ); │ │ │ │ │ + for (var header in config.headers) { │ │ │ │ │ + request.setRequestHeader(header, config.headers[header]); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var events = this.events; │ │ │ │ │ + │ │ │ │ │ + // we want to execute runCallbacks with "this" as the │ │ │ │ │ + // execution scope │ │ │ │ │ + var self = this; │ │ │ │ │ + │ │ │ │ │ + request.onreadystatechange = function() { │ │ │ │ │ + if (request.readyState == OpenLayers.Request.XMLHttpRequest.DONE) { │ │ │ │ │ + var proceed = events.triggerEvent( │ │ │ │ │ + "complete", { │ │ │ │ │ + request: request, │ │ │ │ │ + config: config, │ │ │ │ │ + requestUrl: url │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + if (proceed !== false) { │ │ │ │ │ + self.runCallbacks({ │ │ │ │ │ + request: request, │ │ │ │ │ + config: config, │ │ │ │ │ + requestUrl: url │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // send request (optionally with data) and return │ │ │ │ │ + // call in a timeout for asynchronous requests so the return is │ │ │ │ │ + // available before readyState == 4 for cached docs │ │ │ │ │ + if (config.async === false) { │ │ │ │ │ + request.send(config.data); │ │ │ │ │ + } else { │ │ │ │ │ + window.setTimeout(function() { │ │ │ │ │ + if (request.readyState !== 0) { // W3C: 0-UNSENT │ │ │ │ │ + request.send(config.data); │ │ │ │ │ + } │ │ │ │ │ + }, 0); │ │ │ │ │ + } │ │ │ │ │ + return request; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: runCallbacks │ │ │ │ │ + * Calls the complete, success and failure callbacks. Application │ │ │ │ │ + * can listen to the "complete" event, have the listener │ │ │ │ │ + * display a confirm window and always return false, and │ │ │ │ │ + * execute OpenLayers.Request.runCallbacks if the user │ │ │ │ │ + * hits "yes" in the confirm window. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Hash containing request, config and requestUrl keys │ │ │ │ │ + */ │ │ │ │ │ + runCallbacks: function(options) { │ │ │ │ │ + var request = options.request; │ │ │ │ │ + var config = options.config; │ │ │ │ │ + │ │ │ │ │ + // bind callbacks to readyState 4 (done) │ │ │ │ │ + var complete = (config.scope) ? │ │ │ │ │ + OpenLayers.Function.bind(config.callback, config.scope) : │ │ │ │ │ + config.callback; │ │ │ │ │ + │ │ │ │ │ + // optional success callback │ │ │ │ │ + var success; │ │ │ │ │ + if (config.success) { │ │ │ │ │ + success = (config.scope) ? │ │ │ │ │ + OpenLayers.Function.bind(config.success, config.scope) : │ │ │ │ │ + config.success; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // optional failure callback │ │ │ │ │ + var failure; │ │ │ │ │ + if (config.failure) { │ │ │ │ │ + failure = (config.scope) ? │ │ │ │ │ + OpenLayers.Function.bind(config.failure, config.scope) : │ │ │ │ │ + config.failure; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (OpenLayers.Util.createUrlObject(config.url).protocol == "file:" && │ │ │ │ │ + request.responseText) { │ │ │ │ │ + request.status = 200; │ │ │ │ │ + } │ │ │ │ │ + complete(request); │ │ │ │ │ + │ │ │ │ │ + if (!request.status || (request.status >= 200 && request.status < 300)) { │ │ │ │ │ + this.events.triggerEvent("success", options); │ │ │ │ │ + if (success) { │ │ │ │ │ + success(request); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ + this.events.triggerEvent("failure", options); │ │ │ │ │ + if (failure) { │ │ │ │ │ + failure(request); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: GET │ │ │ │ │ + * Send an HTTP GET request. Additional configuration properties are │ │ │ │ │ + * documented in the method, with the method property set │ │ │ │ │ + * to GET. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the method for documentation of allowed properties. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ + */ │ │ │ │ │ + GET: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "GET" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: POST │ │ │ │ │ + * Send a POST request. Additional configuration properties are │ │ │ │ │ + * documented in the method, with the method property set │ │ │ │ │ + * to POST and "Content-Type" header set to "application/xml". │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the method for documentation of allowed properties. The │ │ │ │ │ + * default "Content-Type" header will be set to "application-xml" if │ │ │ │ │ + * none is provided. This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ + */ │ │ │ │ │ + POST: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "POST" │ │ │ │ │ + }); │ │ │ │ │ + // set content type to application/xml if it isn't already set │ │ │ │ │ + config.headers = config.headers ? config.headers : {}; │ │ │ │ │ + if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ + config.headers["Content-Type"] = "application/xml"; │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: PUT │ │ │ │ │ + * Send an HTTP PUT request. Additional configuration properties are │ │ │ │ │ + * documented in the method, with the method property set │ │ │ │ │ + * to PUT and "Content-Type" header set to "application/xml". │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the method for documentation of allowed properties. The │ │ │ │ │ + * default "Content-Type" header will be set to "application-xml" if │ │ │ │ │ + * none is provided. This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ + */ │ │ │ │ │ + PUT: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "PUT" │ │ │ │ │ + }); │ │ │ │ │ + // set content type to application/xml if it isn't already set │ │ │ │ │ + config.headers = config.headers ? config.headers : {}; │ │ │ │ │ + if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ + config.headers["Content-Type"] = "application/xml"; │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: DELETE │ │ │ │ │ + * Send an HTTP DELETE request. Additional configuration properties are │ │ │ │ │ + * documented in the method, with the method property set │ │ │ │ │ + * to DELETE. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the method for documentation of allowed properties. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ + */ │ │ │ │ │ + DELETE: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "DELETE" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: HEAD │ │ │ │ │ + * Send an HTTP HEAD request. Additional configuration properties are │ │ │ │ │ + * documented in the method, with the method property set │ │ │ │ │ + * to HEAD. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the method for documentation of allowed properties. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ + */ │ │ │ │ │ + HEAD: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "HEAD" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: OPTIONS │ │ │ │ │ + * Send an HTTP OPTIONS request. Additional configuration properties are │ │ │ │ │ + * documented in the method, with the method property set │ │ │ │ │ + * to OPTIONS. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the method for documentation of allowed properties. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ + */ │ │ │ │ │ + OPTIONS: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "OPTIONS" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ OpenLayers/WPSProcess.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -31180,14 +25832,517 @@ │ │ │ │ │ OpenLayers.Util.extend(this, options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.WPSProcess.ChainLink" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ + OpenLayers/Icon.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Icon │ │ │ │ │ + * │ │ │ │ │ + * The icon represents a graphical icon on the screen. Typically used in │ │ │ │ │ + * conjunction with a to represent markers on a screen. │ │ │ │ │ + * │ │ │ │ │ + * An icon has a url, size and position. It also contains an offset which │ │ │ │ │ + * allows the center point to be represented correctly. This can be │ │ │ │ │ + * provided either as a fixed offset or a function provided to calculate │ │ │ │ │ + * the desired offset. │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Icon = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: url │ │ │ │ │ + * {String} image url │ │ │ │ │ + */ │ │ │ │ │ + url: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: size │ │ │ │ │ + * {|Object} An OpenLayers.Size or │ │ │ │ │ + * an object with a 'w' and 'h' properties. │ │ │ │ │ + */ │ │ │ │ │ + size: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: offset │ │ │ │ │ + * {|Object} distance in pixels to offset the │ │ │ │ │ + * image when being rendered. An OpenLayers.Pixel or an object │ │ │ │ │ + * with a 'x' and 'y' properties. │ │ │ │ │ + */ │ │ │ │ │ + offset: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: calculateOffset │ │ │ │ │ + * {Function} Function to calculate the offset (based on the size) │ │ │ │ │ + */ │ │ │ │ │ + calculateOffset: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: imageDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + imageDiv: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: px │ │ │ │ │ + * {|Object} An OpenLayers.Pixel or an object │ │ │ │ │ + * with a 'x' and 'y' properties. │ │ │ │ │ + */ │ │ │ │ │ + px: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Icon │ │ │ │ │ + * Creates an icon, which is an image tag in a div. │ │ │ │ │ + * │ │ │ │ │ + * url - {String} │ │ │ │ │ + * size - {|Object} An OpenLayers.Size or an │ │ │ │ │ + * object with a 'w' and 'h' │ │ │ │ │ + * properties. │ │ │ │ │ + * offset - {|Object} An OpenLayers.Pixel or an │ │ │ │ │ + * object with a 'x' and 'y' │ │ │ │ │ + * properties. │ │ │ │ │ + * calculateOffset - {Function} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(url, size, offset, calculateOffset) { │ │ │ │ │ + this.url = url; │ │ │ │ │ + this.size = size || { │ │ │ │ │ + w: 20, │ │ │ │ │ + h: 20 │ │ │ │ │ + }; │ │ │ │ │ + this.offset = offset || { │ │ │ │ │ + x: -(this.size.w / 2), │ │ │ │ │ + y: -(this.size.h / 2) │ │ │ │ │ + }; │ │ │ │ │ + this.calculateOffset = calculateOffset; │ │ │ │ │ + │ │ │ │ │ + var id = OpenLayers.Util.createUniqueID("OL_Icon_"); │ │ │ │ │ + this.imageDiv = OpenLayers.Util.createAlphaImageDiv(id); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Nullify references and remove event listeners to prevent circular │ │ │ │ │ + * references and memory leaks │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + // erase any drawn elements │ │ │ │ │ + this.erase(); │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Event.stopObservingElement(this.imageDiv.firstChild); │ │ │ │ │ + this.imageDiv.innerHTML = ""; │ │ │ │ │ + this.imageDiv = null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clone │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} A fresh copy of the icon. │ │ │ │ │ + */ │ │ │ │ │ + clone: function() { │ │ │ │ │ + return new OpenLayers.Icon(this.url, │ │ │ │ │ + this.size, │ │ │ │ │ + this.offset, │ │ │ │ │ + this.calculateOffset); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setSize │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * size - {|Object} An OpenLayers.Size or │ │ │ │ │ + * an object with a 'w' and 'h' properties. │ │ │ │ │ + */ │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + if (size != null) { │ │ │ │ │ + this.size = size; │ │ │ │ │ + } │ │ │ │ │ + this.draw(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setUrl │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * url - {String} │ │ │ │ │ + */ │ │ │ │ │ + setUrl: function(url) { │ │ │ │ │ + if (url != null) { │ │ │ │ │ + this.url = url; │ │ │ │ │ + } │ │ │ │ │ + this.draw(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Move the div to the given pixel. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {|Object} An OpenLayers.Pixel or an │ │ │ │ │ + * object with a 'x' and 'y' properties. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A new DOM Image of this icon set at the location passed-in │ │ │ │ │ + */ │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, │ │ │ │ │ + null, │ │ │ │ │ + null, │ │ │ │ │ + this.size, │ │ │ │ │ + this.url, │ │ │ │ │ + "absolute"); │ │ │ │ │ + this.moveTo(px); │ │ │ │ │ + return this.imageDiv; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: erase │ │ │ │ │ + * Erase the underlying image element. │ │ │ │ │ + */ │ │ │ │ │ + erase: function() { │ │ │ │ │ + if (this.imageDiv != null && this.imageDiv.parentNode != null) { │ │ │ │ │ + OpenLayers.Element.remove(this.imageDiv); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setOpacity │ │ │ │ │ + * Change the icon's opacity │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * opacity - {float} │ │ │ │ │ + */ │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, null, │ │ │ │ │ + null, null, null, null, opacity); │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * move icon to passed in px. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {|Object} the pixel position to move to. │ │ │ │ │ + * An OpenLayers.Pixel or an object with a 'x' and 'y' properties. │ │ │ │ │ + */ │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + //if no px passed in, use stored location │ │ │ │ │ + if (px != null) { │ │ │ │ │ + this.px = px; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.imageDiv != null) { │ │ │ │ │ + if (this.px == null) { │ │ │ │ │ + this.display(false); │ │ │ │ │ + } else { │ │ │ │ │ + if (this.calculateOffset) { │ │ │ │ │ + this.offset = this.calculateOffset(this.size); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, { │ │ │ │ │ + x: this.px.x + this.offset.x, │ │ │ │ │ + y: this.px.y + this.offset.y │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: display │ │ │ │ │ + * Hide or show the icon │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * display - {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + display: function(display) { │ │ │ │ │ + this.imageDiv.style.display = (display) ? "" : "none"; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: isDrawn │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the icon is drawn. │ │ │ │ │ + */ │ │ │ │ │ + isDrawn: function() { │ │ │ │ │ + // nodeType 11 for ie, whose nodes *always* have a parentNode │ │ │ │ │ + // (of type document fragment) │ │ │ │ │ + var isDrawn = (this.imageDiv && this.imageDiv.parentNode && │ │ │ │ │ + (this.imageDiv.parentNode.nodeType != 11)); │ │ │ │ │ + │ │ │ │ │ + return isDrawn; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Icon" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Marker.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ + * @requires OpenLayers/Icon.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Marker │ │ │ │ │ + * Instances of OpenLayers.Marker are a combination of a │ │ │ │ │ + * and an . │ │ │ │ │ + * │ │ │ │ │ + * Markers are generally added to a special layer called │ │ │ │ │ + * . │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var markers = new OpenLayers.Layer.Markers( "Markers" ); │ │ │ │ │ + * map.addLayer(markers); │ │ │ │ │ + * │ │ │ │ │ + * var size = new OpenLayers.Size(21,25); │ │ │ │ │ + * var offset = new OpenLayers.Pixel(-(size.w/2), -size.h); │ │ │ │ │ + * var icon = new OpenLayers.Icon('http://www.openlayers.org/dev/img/marker.png', size, offset); │ │ │ │ │ + * markers.addMarker(new OpenLayers.Marker(new OpenLayers.LonLat(0,0),icon)); │ │ │ │ │ + * markers.addMarker(new OpenLayers.Marker(new OpenLayers.LonLat(0,0),icon.clone())); │ │ │ │ │ + * │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Note that if you pass an icon into the Marker constructor, it will take │ │ │ │ │ + * that icon and use it. This means that you should not share icons between │ │ │ │ │ + * markers -- you use them once, but you should clone() for any additional │ │ │ │ │ + * markers using that same icon. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Marker = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: icon │ │ │ │ │ + * {} The icon used by this marker. │ │ │ │ │ + */ │ │ │ │ │ + icon: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: lonlat │ │ │ │ │ + * {} location of object │ │ │ │ │ + */ │ │ │ │ │ + lonlat: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: events │ │ │ │ │ + * {} the event handler. │ │ │ │ │ + */ │ │ │ │ │ + events: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: map │ │ │ │ │ + * {} the map this marker is attached to │ │ │ │ │ + */ │ │ │ │ │ + map: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Marker │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * lonlat - {} the position of this marker │ │ │ │ │ + * icon - {} the icon for this marker │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(lonlat, icon) { │ │ │ │ │ + this.lonlat = lonlat; │ │ │ │ │ + │ │ │ │ │ + var newIcon = (icon) ? icon : OpenLayers.Marker.defaultIcon(); │ │ │ │ │ + if (this.icon == null) { │ │ │ │ │ + this.icon = newIcon; │ │ │ │ │ + } else { │ │ │ │ │ + this.icon.url = newIcon.url; │ │ │ │ │ + this.icon.size = newIcon.size; │ │ │ │ │ + this.icon.offset = newIcon.offset; │ │ │ │ │ + this.icon.calculateOffset = newIcon.calculateOffset; │ │ │ │ │ + } │ │ │ │ │ + this.events = new OpenLayers.Events(this, this.icon.imageDiv); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Destroy the marker. You must first remove the marker from any │ │ │ │ │ + * layer which it has been added to, or you will get buggy behavior. │ │ │ │ │ + * (This can not be done within the marker since the marker does not │ │ │ │ │ + * know which layer it is attached to.) │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + // erase any drawn features │ │ │ │ │ + this.erase(); │ │ │ │ │ + │ │ │ │ │ + this.map = null; │ │ │ │ │ + │ │ │ │ │ + this.events.destroy(); │ │ │ │ │ + this.events = null; │ │ │ │ │ + │ │ │ │ │ + if (this.icon != null) { │ │ │ │ │ + this.icon.destroy(); │ │ │ │ │ + this.icon = null; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Calls draw on the icon, and returns that output. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A new DOM Image with this marker's icon set at the │ │ │ │ │ + * location passed-in │ │ │ │ │ + */ │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + return this.icon.draw(px); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: erase │ │ │ │ │ + * Erases any drawn elements for this marker. │ │ │ │ │ + */ │ │ │ │ │ + erase: function() { │ │ │ │ │ + if (this.icon != null) { │ │ │ │ │ + this.icon.erase(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * Move the marker to the new location. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {|Object} the pixel position to move to. │ │ │ │ │ + * An OpenLayers.Pixel or an object with a 'x' and 'y' properties. │ │ │ │ │ + */ │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + if ((px != null) && (this.icon != null)) { │ │ │ │ │ + this.icon.moveTo(px); │ │ │ │ │ + } │ │ │ │ │ + this.lonlat = this.map.getLonLatFromLayerPx(px); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: isDrawn │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the marker is drawn. │ │ │ │ │ + */ │ │ │ │ │ + isDrawn: function() { │ │ │ │ │ + var isDrawn = (this.icon && this.icon.isDrawn()); │ │ │ │ │ + return isDrawn; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: onScreen │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the marker is currently visible on screen. │ │ │ │ │ + */ │ │ │ │ │ + onScreen: function() { │ │ │ │ │ + │ │ │ │ │ + var onScreen = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + var screenBounds = this.map.getExtent(); │ │ │ │ │ + onScreen = screenBounds.containsLonLat(this.lonlat); │ │ │ │ │ + } │ │ │ │ │ + return onScreen; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: inflate │ │ │ │ │ + * Englarges the markers icon by the specified ratio. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * inflate - {float} the ratio to enlarge the marker by (passing 2 │ │ │ │ │ + * will double the size). │ │ │ │ │ + */ │ │ │ │ │ + inflate: function(inflate) { │ │ │ │ │ + if (this.icon) { │ │ │ │ │ + this.icon.setSize({ │ │ │ │ │ + w: this.icon.size.w * inflate, │ │ │ │ │ + h: this.icon.size.h * inflate │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setOpacity │ │ │ │ │ + * Change the opacity of the marker by changin the opacity of │ │ │ │ │ + * its icon │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * opacity - {float} Specified as fraction (0.4, etc) │ │ │ │ │ + */ │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + this.icon.setOpacity(opacity); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setUrl │ │ │ │ │ + * Change URL of the Icon Image. │ │ │ │ │ + * │ │ │ │ │ + * url - {String} │ │ │ │ │ + */ │ │ │ │ │ + setUrl: function(url) { │ │ │ │ │ + this.icon.setUrl(url); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: display │ │ │ │ │ + * Hide or show the icon │ │ │ │ │ + * │ │ │ │ │ + * display - {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + display: function(display) { │ │ │ │ │ + this.icon.display(display); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Marker" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Function: defaultIcon │ │ │ │ │ + * Creates a default . │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} A default OpenLayers.Icon to use for a marker │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Marker.defaultIcon = function() { │ │ │ │ │ + return new OpenLayers.Icon(OpenLayers.Util.getImageLocation("marker.png"), { │ │ │ │ │ + w: 21, │ │ │ │ │ + h: 25 │ │ │ │ │ + }, { │ │ │ │ │ + x: -10.5, │ │ │ │ │ + y: -25 │ │ │ │ │ + }); │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ OpenLayers/Strategy.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -31305,18359 +26460,19971 @@ │ │ │ │ │ } │ │ │ │ │ return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Strategy" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/StyleMap.js │ │ │ │ │ + OpenLayers/Kinetic.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Style.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Animation.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.StyleMap │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.StyleMap = OpenLayers.Class({ │ │ │ │ │ +OpenLayers.Kinetic = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: styles │ │ │ │ │ - * {Object} Hash of {}, keyed by names of well known │ │ │ │ │ - * rendering intents (e.g. "default", "temporary", "select", "delete"). │ │ │ │ │ + * Property: threshold │ │ │ │ │ + * In most cases changing the threshold isn't needed. │ │ │ │ │ + * In px/ms, default to 0. │ │ │ │ │ */ │ │ │ │ │ - styles: null, │ │ │ │ │ + threshold: 0, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: extendDefault │ │ │ │ │ - * {Boolean} if true, every render intent will extend the symbolizers │ │ │ │ │ - * specified for the "default" intent at rendering time. Otherwise, every │ │ │ │ │ - * rendering intent will be treated as a completely independent style. │ │ │ │ │ + * Property: deceleration │ │ │ │ │ + * {Float} the deseleration in px/ms², default to 0.0035. │ │ │ │ │ */ │ │ │ │ │ - extendDefault: true, │ │ │ │ │ + deceleration: 0.0035, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.StyleMap │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * style - {Object} Optional. Either a style hash, or a style object, or │ │ │ │ │ - * a hash of style objects (style hashes) keyed by rendering │ │ │ │ │ - * intent. If just one style hash or style object is passed, │ │ │ │ │ - * this will be used for all known render intents (default, │ │ │ │ │ - * select, temporary) │ │ │ │ │ - * options - {Object} optional hash of additional options for this │ │ │ │ │ - * instance │ │ │ │ │ + * Property: nbPoints │ │ │ │ │ + * {Integer} the number of points we use to calculate the kinetic │ │ │ │ │ + * initial values. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(style, options) { │ │ │ │ │ - this.styles = { │ │ │ │ │ - "default": new OpenLayers.Style( │ │ │ │ │ - OpenLayers.Feature.Vector.style["default"]), │ │ │ │ │ - "select": new OpenLayers.Style( │ │ │ │ │ - OpenLayers.Feature.Vector.style["select"]), │ │ │ │ │ - "temporary": new OpenLayers.Style( │ │ │ │ │ - OpenLayers.Feature.Vector.style["temporary"]), │ │ │ │ │ - "delete": new OpenLayers.Style( │ │ │ │ │ - OpenLayers.Feature.Vector.style["delete"]) │ │ │ │ │ - }; │ │ │ │ │ + nbPoints: 100, │ │ │ │ │ │ │ │ │ │ - // take whatever the user passed as style parameter and convert it │ │ │ │ │ - // into parts of stylemap. │ │ │ │ │ - if (style instanceof OpenLayers.Style) { │ │ │ │ │ - // user passed a style object │ │ │ │ │ - this.styles["default"] = style; │ │ │ │ │ - this.styles["select"] = style; │ │ │ │ │ - this.styles["temporary"] = style; │ │ │ │ │ - this.styles["delete"] = style; │ │ │ │ │ - } else if (typeof style == "object") { │ │ │ │ │ - for (var key in style) { │ │ │ │ │ - if (style[key] instanceof OpenLayers.Style) { │ │ │ │ │ - // user passed a hash of style objects │ │ │ │ │ - this.styles[key] = style[key]; │ │ │ │ │ - } else if (typeof style[key] == "object") { │ │ │ │ │ - // user passsed a hash of style hashes │ │ │ │ │ - this.styles[key] = new OpenLayers.Style(style[key]); │ │ │ │ │ - } else { │ │ │ │ │ - // user passed a style hash (i.e. symbolizer) │ │ │ │ │ - this.styles["default"] = new OpenLayers.Style(style); │ │ │ │ │ - this.styles["select"] = new OpenLayers.Style(style); │ │ │ │ │ - this.styles["temporary"] = new OpenLayers.Style(style); │ │ │ │ │ - this.styles["delete"] = new OpenLayers.Style(style); │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: delay │ │ │ │ │ + * {Float} time to consider to calculate the kinetic initial values. │ │ │ │ │ + * In ms, default to 200. │ │ │ │ │ + */ │ │ │ │ │ + delay: 200, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: points │ │ │ │ │ + * List of points use to calculate the kinetic initial values. │ │ │ │ │ + */ │ │ │ │ │ + points: undefined, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: timerId │ │ │ │ │ + * ID of the timer. │ │ │ │ │ + */ │ │ │ │ │ + timerId: undefined, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Kinetic │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ OpenLayers.Util.extend(this, options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * Method: begin │ │ │ │ │ + * Begins the dragging. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var key in this.styles) { │ │ │ │ │ - this.styles[key].destroy(); │ │ │ │ │ + begin: function() { │ │ │ │ │ + OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ + this.timerId = undefined; │ │ │ │ │ + this.points = []; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: update │ │ │ │ │ + * Updates during the dragging. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * xy - {} The new position. │ │ │ │ │ + */ │ │ │ │ │ + update: function(xy) { │ │ │ │ │ + this.points.unshift({ │ │ │ │ │ + xy: xy, │ │ │ │ │ + tick: new Date().getTime() │ │ │ │ │ + }); │ │ │ │ │ + if (this.points.length > this.nbPoints) { │ │ │ │ │ + this.points.pop(); │ │ │ │ │ } │ │ │ │ │ - this.styles = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createSymbolizer │ │ │ │ │ - * Creates the symbolizer for a feature for a render intent. │ │ │ │ │ - * │ │ │ │ │ + * Method: end │ │ │ │ │ + * Ends the dragging, start the kinetic. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {} The feature to evaluate the rules │ │ │ │ │ - * of the intended style against. │ │ │ │ │ - * intent - {String} The intent determines the symbolizer that will be │ │ │ │ │ - * used to draw the feature. Well known intents are "default" │ │ │ │ │ - * (for just drawing the features), "select" (for selected │ │ │ │ │ - * features) and "temporary" (for drawing features). │ │ │ │ │ - * │ │ │ │ │ + * xy - {} The last position. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} symbolizer hash │ │ │ │ │ + * {Object} An object with two properties: "speed", and "theta". The │ │ │ │ │ + * "speed" and "theta" values are to be passed to the move │ │ │ │ │ + * function when starting the animation. │ │ │ │ │ */ │ │ │ │ │ - createSymbolizer: function(feature, intent) { │ │ │ │ │ - if (!feature) { │ │ │ │ │ - feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ + end: function(xy) { │ │ │ │ │ + var last, now = new Date().getTime(); │ │ │ │ │ + for (var i = 0, l = this.points.length, point; i < l; i++) { │ │ │ │ │ + point = this.points[i]; │ │ │ │ │ + if (now - point.tick > this.delay) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + last = point; │ │ │ │ │ } │ │ │ │ │ - if (!this.styles[intent]) { │ │ │ │ │ - intent = "default"; │ │ │ │ │ + if (!last) { │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ - feature.renderIntent = intent; │ │ │ │ │ - var defaultSymbolizer = {}; │ │ │ │ │ - if (this.extendDefault && intent != "default") { │ │ │ │ │ - defaultSymbolizer = this.styles["default"].createSymbolizer(feature); │ │ │ │ │ + var time = new Date().getTime() - last.tick; │ │ │ │ │ + var dist = Math.sqrt(Math.pow(xy.x - last.xy.x, 2) + │ │ │ │ │ + Math.pow(xy.y - last.xy.y, 2)); │ │ │ │ │ + var speed = dist / time; │ │ │ │ │ + if (speed == 0 || speed < this.threshold) { │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Util.extend(defaultSymbolizer, │ │ │ │ │ - this.styles[intent].createSymbolizer(feature)); │ │ │ │ │ + var theta = Math.asin((xy.y - last.xy.y) / dist); │ │ │ │ │ + if (last.xy.x <= xy.x) { │ │ │ │ │ + theta = Math.PI - theta; │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + speed: speed, │ │ │ │ │ + theta: theta │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: addUniqueValueRules │ │ │ │ │ - * Convenience method to create comparison rules for unique values of a │ │ │ │ │ - * property. The rules will be added to the style object for a specified │ │ │ │ │ - * rendering intent. This method is a shortcut for creating something like │ │ │ │ │ - * the "unique value legends" familiar from well known desktop GIS systems │ │ │ │ │ - * │ │ │ │ │ + * Method: move │ │ │ │ │ + * Launch the kinetic move pan. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * renderIntent - {String} rendering intent to add the rules to │ │ │ │ │ - * property - {String} values of feature attributes to create the │ │ │ │ │ - * rules for │ │ │ │ │ - * symbolizers - {Object} Hash of symbolizers, keyed by the desired │ │ │ │ │ - * property values │ │ │ │ │ - * context - {Object} An optional object with properties that │ │ │ │ │ - * symbolizers' property values should be evaluated │ │ │ │ │ - * against. If no context is specified, feature.attributes │ │ │ │ │ - * will be used │ │ │ │ │ + * info - {Object} An object with two properties, "speed", and "theta". │ │ │ │ │ + * These values are those returned from the "end" call. │ │ │ │ │ + * callback - {Function} Function called on every step of the animation, │ │ │ │ │ + * receives x, y (values to pan), end (is the last point). │ │ │ │ │ */ │ │ │ │ │ - addUniqueValueRules: function(renderIntent, property, symbolizers, context) { │ │ │ │ │ - var rules = []; │ │ │ │ │ - for (var value in symbolizers) { │ │ │ │ │ - rules.push(new OpenLayers.Rule({ │ │ │ │ │ - symbolizer: symbolizers[value], │ │ │ │ │ - context: context, │ │ │ │ │ - filter: new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.EQUAL_TO, │ │ │ │ │ - property: property, │ │ │ │ │ - value: value │ │ │ │ │ - }) │ │ │ │ │ - })); │ │ │ │ │ - } │ │ │ │ │ - this.styles[renderIntent].addRules(rules); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.StyleMap" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WPSDescribeProcess.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/OWSCommon/v1_1_0.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WPSDescribeProcess │ │ │ │ │ - * Read WPS DescribeProcess responses. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WPSDescribeProcess = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.XML, { │ │ │ │ │ + move: function(info, callback) { │ │ │ │ │ + var v0 = info.speed; │ │ │ │ │ + var fx = Math.cos(info.theta); │ │ │ │ │ + var fy = -Math.sin(info.theta); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: VERSION │ │ │ │ │ - * {String} 1.0.0 │ │ │ │ │ - */ │ │ │ │ │ - VERSION: "1.0.0", │ │ │ │ │ + var initialTime = new Date().getTime(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - wps: "http://www.opengis.net/wps/1.0.0", │ │ │ │ │ - ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ - }, │ │ │ │ │ + var lastX = 0; │ │ │ │ │ + var lastY = 0; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} Schema location │ │ │ │ │ - */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/wps/1.0.0 http://schemas.opengis.net/wps/1.0.0/wpsAll.xsd", │ │ │ │ │ + var timerCallback = function() { │ │ │ │ │ + if (this.timerId == null) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - */ │ │ │ │ │ - defaultPrefix: "wps", │ │ │ │ │ + var t = new Date().getTime() - initialTime; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ - */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ - }, │ │ │ │ │ + var p = (-this.deceleration * Math.pow(t, 2)) / 2.0 + v0 * t; │ │ │ │ │ + var x = p * fx; │ │ │ │ │ + var y = p * fy; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WPSDescribeProcess │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + var args = {}; │ │ │ │ │ + args.end = false; │ │ │ │ │ + var v = -this.deceleration * t + v0; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Parse a WPS DescribeProcess and return an object with its information. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ + if (v <= 0) { │ │ │ │ │ + OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + args.end = true; │ │ │ │ │ } │ │ │ │ │ - var info = {}; │ │ │ │ │ - this.readNode(data, info); │ │ │ │ │ - return info; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wps": { │ │ │ │ │ - "ProcessDescriptions": function(node, obj) { │ │ │ │ │ - obj.processDescriptions = {}; │ │ │ │ │ - this.readChildNodes(node, obj.processDescriptions); │ │ │ │ │ - }, │ │ │ │ │ - "ProcessDescription": function(node, processDescriptions) { │ │ │ │ │ - var processVersion = this.getAttributeNS(node, this.namespaces.wps, "processVersion"); │ │ │ │ │ - var processDescription = { │ │ │ │ │ - processVersion: processVersion, │ │ │ │ │ - statusSupported: (node.getAttribute("statusSupported") === "true"), │ │ │ │ │ - storeSupported: (node.getAttribute("storeSupported") === "true") │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, processDescription); │ │ │ │ │ - processDescriptions[processDescription.identifier] = processDescription; │ │ │ │ │ - }, │ │ │ │ │ - "DataInputs": function(node, processDescription) { │ │ │ │ │ - processDescription.dataInputs = []; │ │ │ │ │ - this.readChildNodes(node, processDescription.dataInputs); │ │ │ │ │ - }, │ │ │ │ │ - "ProcessOutputs": function(node, processDescription) { │ │ │ │ │ - processDescription.processOutputs = []; │ │ │ │ │ - this.readChildNodes(node, processDescription.processOutputs); │ │ │ │ │ - }, │ │ │ │ │ - "Output": function(node, processOutputs) { │ │ │ │ │ - var output = {}; │ │ │ │ │ - this.readChildNodes(node, output); │ │ │ │ │ - processOutputs.push(output); │ │ │ │ │ - }, │ │ │ │ │ - "ComplexOutput": function(node, output) { │ │ │ │ │ - output.complexOutput = {}; │ │ │ │ │ - this.readChildNodes(node, output.complexOutput); │ │ │ │ │ - }, │ │ │ │ │ - "LiteralOutput": function(node, output) { │ │ │ │ │ - output.literalOutput = {}; │ │ │ │ │ - this.readChildNodes(node, output.literalOutput); │ │ │ │ │ - }, │ │ │ │ │ - "Input": function(node, dataInputs) { │ │ │ │ │ - var input = { │ │ │ │ │ - maxOccurs: parseInt(node.getAttribute("maxOccurs")), │ │ │ │ │ - minOccurs: parseInt(node.getAttribute("minOccurs")) │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, input); │ │ │ │ │ - dataInputs.push(input); │ │ │ │ │ - }, │ │ │ │ │ - "BoundingBoxData": function(node, input) { │ │ │ │ │ - input.boundingBoxData = {}; │ │ │ │ │ - this.readChildNodes(node, input.boundingBoxData); │ │ │ │ │ - }, │ │ │ │ │ - "CRS": function(node, obj) { │ │ │ │ │ - if (!obj.CRSs) { │ │ │ │ │ - obj.CRSs = {}; │ │ │ │ │ - } │ │ │ │ │ - obj.CRSs[this.getChildValue(node)] = true; │ │ │ │ │ - }, │ │ │ │ │ - "LiteralData": function(node, input) { │ │ │ │ │ - input.literalData = {}; │ │ │ │ │ - this.readChildNodes(node, input.literalData); │ │ │ │ │ - }, │ │ │ │ │ - "ComplexData": function(node, input) { │ │ │ │ │ - input.complexData = {}; │ │ │ │ │ - this.readChildNodes(node, input.complexData); │ │ │ │ │ - }, │ │ │ │ │ - "Default": function(node, complexData) { │ │ │ │ │ - complexData["default"] = {}; │ │ │ │ │ - this.readChildNodes(node, complexData["default"]); │ │ │ │ │ - }, │ │ │ │ │ - "Supported": function(node, complexData) { │ │ │ │ │ - complexData["supported"] = {}; │ │ │ │ │ - this.readChildNodes(node, complexData["supported"]); │ │ │ │ │ - }, │ │ │ │ │ - "Format": function(node, obj) { │ │ │ │ │ - var format = {}; │ │ │ │ │ - this.readChildNodes(node, format); │ │ │ │ │ - if (!obj.formats) { │ │ │ │ │ - obj.formats = {}; │ │ │ │ │ - } │ │ │ │ │ - obj.formats[format.mimeType] = true; │ │ │ │ │ - }, │ │ │ │ │ - "MimeType": function(node, format) { │ │ │ │ │ - format.mimeType = this.getChildValue(node); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "ows": OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ - }, │ │ │ │ │ + args.x = x - lastX; │ │ │ │ │ + args.y = y - lastY; │ │ │ │ │ + lastX = x; │ │ │ │ │ + lastY = y; │ │ │ │ │ + callback(args.x, args.y, args.end); │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WPSDescribeProcess" │ │ │ │ │ + this.timerId = OpenLayers.Animation.start( │ │ │ │ │ + OpenLayers.Function.bind(timerCallback, this) │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Kinetic" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/WPSClient.js │ │ │ │ │ + OpenLayers/Layer.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/SingleFile.js │ │ │ │ │ - */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ - * @requires OpenLayers/WPSProcess.js │ │ │ │ │ - * @requires OpenLayers/Format/WPSDescribeProcess.js │ │ │ │ │ - * @requires OpenLayers/Request.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Map.js │ │ │ │ │ + * @requires OpenLayers/Projection.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.WPSClient │ │ │ │ │ - * High level API for interaction with Web Processing Services (WPS). │ │ │ │ │ - * An instance is used to create │ │ │ │ │ - * instances for servers known to the WPSClient. The WPSClient also caches │ │ │ │ │ - * DescribeProcess responses to reduce the number of requests sent to servers │ │ │ │ │ - * when processes are created. │ │ │ │ │ + * Class: OpenLayers.Layer │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.WPSClient = OpenLayers.Class({ │ │ │ │ │ +OpenLayers.Layer = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: servers │ │ │ │ │ - * {Object} Service metadata, keyed by a local identifier. │ │ │ │ │ - * │ │ │ │ │ - * Properties: │ │ │ │ │ - * url - {String} the url of the server │ │ │ │ │ - * version - {String} WPS version of the server │ │ │ │ │ - * processDescription - {Object} Cache of raw DescribeProcess │ │ │ │ │ - * responses, keyed by process identifier. │ │ │ │ │ + * APIProperty: id │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - servers: null, │ │ │ │ │ + id: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: name │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + name: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: div │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + div: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} The default WPS version to use if none is configured. Default │ │ │ │ │ - * is '1.0.0'. │ │ │ │ │ + * APIProperty: opacity │ │ │ │ │ + * {Float} The layer's opacity. Float number between 0.0 and 1.0. Default │ │ │ │ │ + * is 1. │ │ │ │ │ */ │ │ │ │ │ - version: '1.0.0', │ │ │ │ │ + opacity: 1, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: lazy │ │ │ │ │ - * {Boolean} Should the DescribeProcess be deferred until a process is │ │ │ │ │ - * fully configured? Default is false. │ │ │ │ │ + * APIProperty: alwaysInRange │ │ │ │ │ + * {Boolean} If a layer's display should not be scale-based, this should │ │ │ │ │ + * be set to true. This will cause the layer, as an overlay, to always │ │ │ │ │ + * be 'active', by always returning true from the calculateInRange() │ │ │ │ │ + * function. │ │ │ │ │ + * │ │ │ │ │ + * If not explicitly specified for a layer, its value will be │ │ │ │ │ + * determined on startup in initResolutions() based on whether or not │ │ │ │ │ + * any scale-specific properties have been set as options on the │ │ │ │ │ + * layer. If no scale-specific options have been set on the layer, we │ │ │ │ │ + * assume that it should always be in range. │ │ │ │ │ + * │ │ │ │ │ + * See #987 for more info. │ │ │ │ │ */ │ │ │ │ │ - lazy: false, │ │ │ │ │ + alwaysInRange: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: events │ │ │ │ │ - * {} │ │ │ │ │ - * │ │ │ │ │ - * Supported event types: │ │ │ │ │ - * describeprocess - Fires when the process description is available. │ │ │ │ │ - * Listeners receive an object with a 'raw' property holding the raw │ │ │ │ │ - * DescribeProcess response, and an 'identifier' property holding the │ │ │ │ │ - * process identifier of the described process. │ │ │ │ │ + * Constant: RESOLUTION_PROPERTIES │ │ │ │ │ + * {Array} The properties that are used for calculating resolutions │ │ │ │ │ + * information. │ │ │ │ │ */ │ │ │ │ │ - events: null, │ │ │ │ │ + RESOLUTION_PROPERTIES: [ │ │ │ │ │ + 'scales', 'resolutions', │ │ │ │ │ + 'maxScale', 'minScale', │ │ │ │ │ + 'maxResolution', 'minResolution', │ │ │ │ │ + 'numZoomLevels', 'maxZoomLevel' │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.WPSClient │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {} │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Object whose properties will be set on the instance. │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * layer.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ * │ │ │ │ │ - * Avaliable options: │ │ │ │ │ - * servers - {Object} Mandatory. Service metadata, keyed by a local │ │ │ │ │ - * identifier. Can either be a string with the service url or an │ │ │ │ │ - * object literal with additional metadata: │ │ │ │ │ + * Listeners will be called with a reference to an event object. The │ │ │ │ │ + * properties of this event depends on exactly what happened. │ │ │ │ │ * │ │ │ │ │ - * (code) │ │ │ │ │ - * servers: { │ │ │ │ │ - * local: '/geoserver/wps' │ │ │ │ │ - * }, { │ │ │ │ │ - * opengeo: { │ │ │ │ │ - * url: 'http://demo.opengeo.org/geoserver/wps', │ │ │ │ │ - * version: '1.0.0' │ │ │ │ │ - * } │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ + * All event objects have at least the following properties: │ │ │ │ │ + * object - {Object} A reference to layer.events.object. │ │ │ │ │ + * element - {DOMElement} A reference to layer.events.element. │ │ │ │ │ * │ │ │ │ │ - * lazy - {Boolean} Optional. Set to true if DescribeProcess should not be │ │ │ │ │ - * requested until a process is fully configured. Default is false. │ │ │ │ │ + * Supported map event types: │ │ │ │ │ + * loadstart - Triggered when layer loading starts. When using a Vector │ │ │ │ │ + * layer with a Fixed or BBOX strategy, the event object includes │ │ │ │ │ + * a *filter* property holding the OpenLayers.Filter used when │ │ │ │ │ + * calling read on the protocol. │ │ │ │ │ + * loadend - Triggered when layer loading ends. When using a Vector layer │ │ │ │ │ + * with a Fixed or BBOX strategy, the event object includes a │ │ │ │ │ + * *response* property holding an OpenLayers.Protocol.Response object. │ │ │ │ │ + * visibilitychanged - Triggered when the layer's visibility property is │ │ │ │ │ + * changed, e.g. by turning the layer on or off in the layer switcher. │ │ │ │ │ + * Note that the actual visibility of the layer can also change if it │ │ │ │ │ + * gets out of range (see ). If you also want to catch │ │ │ │ │ + * these cases, register for the map's 'changelayer' event instead. │ │ │ │ │ + * move - Triggered when layer moves (triggered with every mousemove │ │ │ │ │ + * during a drag). │ │ │ │ │ + * moveend - Triggered when layer is done moving, object passed as │ │ │ │ │ + * argument has a zoomChanged boolean property which tells that the │ │ │ │ │ + * zoom has changed. │ │ │ │ │ + * added - Triggered after the layer is added to a map. Listeners will │ │ │ │ │ + * receive an object with a *map* property referencing the map and a │ │ │ │ │ + * *layer* property referencing the layer. │ │ │ │ │ + * removed - Triggered after the layer is removed from the map. Listeners │ │ │ │ │ + * will receive an object with a *map* property referencing the map and │ │ │ │ │ + * a *layer* property referencing the layer. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.events = new OpenLayers.Events(this); │ │ │ │ │ - this.servers = {}; │ │ │ │ │ - for (var s in options.servers) { │ │ │ │ │ - this.servers[s] = typeof options.servers[s] == 'string' ? { │ │ │ │ │ - url: options.servers[s], │ │ │ │ │ - version: this.version, │ │ │ │ │ - processDescription: {} │ │ │ │ │ - } : options.servers[s]; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + events: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: execute │ │ │ │ │ - * Shortcut to execute a process with a single function call. This is │ │ │ │ │ - * equivalent to using and then calling execute on the │ │ │ │ │ - * process. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Options for the execute operation. │ │ │ │ │ - * │ │ │ │ │ - * Available options: │ │ │ │ │ - * server - {String} Mandatory. One of the local identifiers of the │ │ │ │ │ - * configured servers. │ │ │ │ │ - * process - {String} Mandatory. A process identifier known to the │ │ │ │ │ - * server. │ │ │ │ │ - * inputs - {Object} The inputs for the process, keyed by input identifier. │ │ │ │ │ - * For spatial data inputs, the value of an input is usually an │ │ │ │ │ - * , an or an array of │ │ │ │ │ - * geometries or features. │ │ │ │ │ - * output - {String} The identifier of an output to parse. Optional. If not │ │ │ │ │ - * provided, the first output will be parsed. │ │ │ │ │ - * success - {Function} Callback to call when the process is complete. │ │ │ │ │ - * This function is called with an outputs object as argument, which │ │ │ │ │ - * will have a property with the identifier of the requested output │ │ │ │ │ - * (e.g. 'result'). For processes that generate spatial output, the │ │ │ │ │ - * value will either be a single or an │ │ │ │ │ - * array of features. │ │ │ │ │ - * scope - {Object} Optional scope for the success callback. │ │ │ │ │ + * APIProperty: map │ │ │ │ │ + * {} This variable is set when the layer is added to │ │ │ │ │ + * the map, via the accessor function setMap(). │ │ │ │ │ */ │ │ │ │ │ - execute: function(options) { │ │ │ │ │ - var process = this.getProcess(options.server, options.process); │ │ │ │ │ - process.execute({ │ │ │ │ │ - inputs: options.inputs, │ │ │ │ │ - success: options.success, │ │ │ │ │ - scope: options.scope │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + map: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getProcess │ │ │ │ │ - * Creates an . │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * serverID - {String} Local identifier from the servers that this instance │ │ │ │ │ - * was constructed with. │ │ │ │ │ - * processID - {String} Process identifier known to the server. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} Whether or not the layer is a base layer. This should be set │ │ │ │ │ + * individually by all subclasses. Default is false │ │ │ │ │ */ │ │ │ │ │ - getProcess: function(serverID, processID) { │ │ │ │ │ - var process = new OpenLayers.WPSProcess({ │ │ │ │ │ - client: this, │ │ │ │ │ - server: serverID, │ │ │ │ │ - identifier: processID │ │ │ │ │ - }); │ │ │ │ │ - if (!this.lazy) { │ │ │ │ │ - process.describe(); │ │ │ │ │ - } │ │ │ │ │ - return process; │ │ │ │ │ - }, │ │ │ │ │ + isBaseLayer: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: describeProcess │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * serverID - {String} Identifier of the server │ │ │ │ │ - * processID - {String} Identifier of the requested process │ │ │ │ │ - * callback - {Function} Callback to call when the description is available │ │ │ │ │ - * scope - {Object} Optional execution scope for the callback function │ │ │ │ │ + * Property: alpha │ │ │ │ │ + * {Boolean} The layer's images have an alpha channel. Default is false. │ │ │ │ │ */ │ │ │ │ │ - describeProcess: function(serverID, processID, callback, scope) { │ │ │ │ │ - var server = this.servers[serverID]; │ │ │ │ │ - if (!server.processDescription[processID]) { │ │ │ │ │ - if (!(processID in server.processDescription)) { │ │ │ │ │ - // set to null so we know a describeFeature request is pending │ │ │ │ │ - server.processDescription[processID] = null; │ │ │ │ │ - OpenLayers.Request.GET({ │ │ │ │ │ - url: server.url, │ │ │ │ │ - params: { │ │ │ │ │ - SERVICE: 'WPS', │ │ │ │ │ - VERSION: server.version, │ │ │ │ │ - REQUEST: 'DescribeProcess', │ │ │ │ │ - IDENTIFIER: processID │ │ │ │ │ - }, │ │ │ │ │ - success: function(response) { │ │ │ │ │ - server.processDescription[processID] = response.responseText; │ │ │ │ │ - this.events.triggerEvent('describeprocess', { │ │ │ │ │ - identifier: processID, │ │ │ │ │ - raw: response.responseText │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - // pending request │ │ │ │ │ - this.events.register('describeprocess', this, function describe(evt) { │ │ │ │ │ - if (evt.identifier === processID) { │ │ │ │ │ - this.events.unregister('describeprocess', this, describe); │ │ │ │ │ - callback.call(scope, evt.raw); │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - window.setTimeout(function() { │ │ │ │ │ - callback.call(scope, server.processDescription[processID]); │ │ │ │ │ - }, 0); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + alpha: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: displayInLayerSwitcher │ │ │ │ │ + * {Boolean} Display the layer's name in the layer switcher. Default is │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.events.destroy(); │ │ │ │ │ - this.events = null; │ │ │ │ │ - this.servers = null; │ │ │ │ │ - }, │ │ │ │ │ + displayInLayerSwitcher: true, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: 'OpenLayers.WPSClient' │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: visibility │ │ │ │ │ + * {Boolean} The layer should be displayed in the map. Default is true. │ │ │ │ │ + */ │ │ │ │ │ + visibility: true, │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Renderer.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: attribution │ │ │ │ │ + * {String} Attribution string, displayed when an │ │ │ │ │ + * has been added to the map. │ │ │ │ │ + */ │ │ │ │ │ + attribution: null, │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: inRange │ │ │ │ │ + * {Boolean} The current map resolution is within the layer's min/max │ │ │ │ │ + * range. This is set in whenever the zoom │ │ │ │ │ + * changes. │ │ │ │ │ + */ │ │ │ │ │ + inRange: false, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Propery: imageSize │ │ │ │ │ + * {} For layers with a gutter, the image is larger than │ │ │ │ │ + * the tile by twice the gutter in each dimension. │ │ │ │ │ + */ │ │ │ │ │ + imageSize: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Renderer │ │ │ │ │ - * This is the base class for all renderers. │ │ │ │ │ - * │ │ │ │ │ - * This is based on a merger code written by Paul Spencer and Bertil Chapuis. │ │ │ │ │ - * It is largely composed of virtual functions that are to be implemented │ │ │ │ │ - * in technology-specific subclasses, but there is some generic code too. │ │ │ │ │ - * │ │ │ │ │ - * The functions that *are* implemented here merely deal with the maintenance │ │ │ │ │ - * of the size and extent variables, as well as the cached 'resolution' │ │ │ │ │ - * value. │ │ │ │ │ - * │ │ │ │ │ - * A note to the user that all subclasses should use getResolution() instead │ │ │ │ │ - * of directly accessing this.resolution in order to correctly use the │ │ │ │ │ - * cacheing system. │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer = OpenLayers.Class({ │ │ │ │ │ + // OPTIONS │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: container │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Property: options │ │ │ │ │ + * {Object} An optional object whose properties will be set on the layer. │ │ │ │ │ + * Any of the layer properties can be set as a property of the options │ │ │ │ │ + * object and sent to the constructor when the layer is created. │ │ │ │ │ */ │ │ │ │ │ - container: null, │ │ │ │ │ + options: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: root │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * APIProperty: eventListeners │ │ │ │ │ + * {Object} If set as an option at construction, the eventListeners │ │ │ │ │ + * object will be registered with . Object │ │ │ │ │ + * structure must be a listeners object as shown in the example for │ │ │ │ │ + * the events.on method. │ │ │ │ │ */ │ │ │ │ │ - root: null, │ │ │ │ │ + eventListeners: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: extent │ │ │ │ │ - * {} │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: gutter │ │ │ │ │ + * {Integer} Determines the width (in pixels) of the gutter around image │ │ │ │ │ + * tiles to ignore. By setting this property to a non-zero value, │ │ │ │ │ + * images will be requested that are wider and taller than the tile │ │ │ │ │ + * size by a value of 2 x gutter. This allows artifacts of rendering │ │ │ │ │ + * at tile edges to be ignored. Set a gutter value that is equal to │ │ │ │ │ + * half the size of the widest symbol that needs to be displayed. │ │ │ │ │ + * Defaults to zero. Non-tiled layers always have zero gutter. │ │ │ │ │ */ │ │ │ │ │ - extent: null, │ │ │ │ │ + gutter: 0, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: locked │ │ │ │ │ - * {Boolean} If the renderer is currently in a state where many things │ │ │ │ │ - * are changing, the 'locked' property is set to true. This means │ │ │ │ │ - * that renderers can expect at least one more drawFeature event to be │ │ │ │ │ - * called with the 'locked' property set to 'true': In some renderers, │ │ │ │ │ - * this might make sense to use as a 'only update local information' │ │ │ │ │ - * flag. │ │ │ │ │ + * APIProperty: projection │ │ │ │ │ + * {} or {} Specifies the projection of the layer. │ │ │ │ │ + * Can be set in the layer options. If not specified in the layer options, │ │ │ │ │ + * it is set to the default projection specified in the map, │ │ │ │ │ + * when the layer is added to the map. │ │ │ │ │ + * Projection along with default maxExtent and resolutions │ │ │ │ │ + * are set automatically with commercial baselayers in EPSG:3857, │ │ │ │ │ + * such as Google, Bing and OpenStreetMap, and do not need to be specified. │ │ │ │ │ + * Otherwise, if specifying projection, also set maxExtent, │ │ │ │ │ + * maxResolution or resolutions as appropriate. │ │ │ │ │ + * When using vector layers with strategies, layer projection should be set │ │ │ │ │ + * to the projection of the source data if that is different from the map default. │ │ │ │ │ + * │ │ │ │ │ + * Can be either a string or an object; │ │ │ │ │ + * if a string is passed, will be converted to an object when │ │ │ │ │ + * the layer is added to the map. │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - locked: false, │ │ │ │ │ + projection: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: size │ │ │ │ │ - * {} │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: units │ │ │ │ │ + * {String} The layer map units. Defaults to null. Possible values │ │ │ │ │ + * are 'degrees' (or 'dd'), 'm', 'ft', 'km', 'mi', 'inches'. │ │ │ │ │ + * Normally taken from the projection. │ │ │ │ │ + * Only required if both map and layers do not define a projection, │ │ │ │ │ + * or if they define a projection which does not define units. │ │ │ │ │ */ │ │ │ │ │ - size: null, │ │ │ │ │ + units: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: resolution │ │ │ │ │ - * {Float} cache of current map resolution │ │ │ │ │ + * APIProperty: scales │ │ │ │ │ + * {Array} An array of map scales in descending order. The values in the │ │ │ │ │ + * array correspond to the map scale denominator. Note that these │ │ │ │ │ + * values only make sense if the display (monitor) resolution of the │ │ │ │ │ + * client is correctly guessed by whomever is configuring the │ │ │ │ │ + * application. In addition, the units property must also be set. │ │ │ │ │ + * Use instead wherever possible. │ │ │ │ │ */ │ │ │ │ │ - resolution: null, │ │ │ │ │ + scales: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: map │ │ │ │ │ - * {} Reference to the map -- this is set in Vector's setMap() │ │ │ │ │ + * APIProperty: resolutions │ │ │ │ │ + * {Array} A list of map resolutions (map units per pixel) in descending │ │ │ │ │ + * order. If this is not set in the layer constructor, it will be set │ │ │ │ │ + * based on other resolution related properties (maxExtent, │ │ │ │ │ + * maxResolution, maxScale, etc.). │ │ │ │ │ */ │ │ │ │ │ - map: null, │ │ │ │ │ + resolutions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: featureDx │ │ │ │ │ - * {Number} Feature offset in x direction. Will be calculated for and │ │ │ │ │ - * applied to the current feature while rendering (see │ │ │ │ │ - * ). │ │ │ │ │ + * APIProperty: maxExtent │ │ │ │ │ + * {|Array} If provided as an array, the array │ │ │ │ │ + * should consist of four values (left, bottom, right, top). │ │ │ │ │ + * The maximum extent for the layer. Defaults to null. │ │ │ │ │ + * │ │ │ │ │ + * The center of these bounds will not stray outside │ │ │ │ │ + * of the viewport extent during panning. In addition, if │ │ │ │ │ + * is set to false, data will not be │ │ │ │ │ + * requested that falls completely outside of these bounds. │ │ │ │ │ */ │ │ │ │ │ - featureDx: 0, │ │ │ │ │ + maxExtent: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Renderer │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * containerID - {} │ │ │ │ │ - * options - {Object} options for this renderer. See sublcasses for │ │ │ │ │ - * supported options. │ │ │ │ │ + * APIProperty: minExtent │ │ │ │ │ + * {|Array} If provided as an array, the array │ │ │ │ │ + * should consist of four values (left, bottom, right, top). │ │ │ │ │ + * The minimum extent for the layer. Defaults to null. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(containerID, options) { │ │ │ │ │ - this.container = OpenLayers.Util.getElement(containerID); │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - }, │ │ │ │ │ + minExtent: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ + * APIProperty: maxResolution │ │ │ │ │ + * {Float} Default max is 360 deg / 256 px, which corresponds to │ │ │ │ │ + * zoom level 0 on gmaps. Specify a different value in the layer │ │ │ │ │ + * options if you are not using the default │ │ │ │ │ + * and displaying the whole world. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.container = null; │ │ │ │ │ - this.extent = null; │ │ │ │ │ - this.size = null; │ │ │ │ │ - this.resolution = null; │ │ │ │ │ - this.map = null; │ │ │ │ │ - }, │ │ │ │ │ + maxResolution: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: supported │ │ │ │ │ - * This should be overridden by specific subclasses │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the browser supports the renderer class │ │ │ │ │ + * APIProperty: minResolution │ │ │ │ │ + * {Float} │ │ │ │ │ */ │ │ │ │ │ - supported: function() { │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ + minResolution: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setExtent │ │ │ │ │ - * Set the visible part of the layer. │ │ │ │ │ - * │ │ │ │ │ - * Resolution has probably changed, so we nullify the resolution │ │ │ │ │ - * cache (this.resolution) -- this way it will be re-computed when │ │ │ │ │ - * next it is needed. │ │ │ │ │ - * We nullify the resolution cache (this.resolution) if resolutionChanged │ │ │ │ │ - * is set to true - this way it will be re-computed on the next │ │ │ │ │ - * getResolution() request. │ │ │ │ │ + * APIProperty: numZoomLevels │ │ │ │ │ + * {Integer} │ │ │ │ │ + */ │ │ │ │ │ + numZoomLevels: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: minScale │ │ │ │ │ + * {Float} │ │ │ │ │ + */ │ │ │ │ │ + minScale: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxScale │ │ │ │ │ + * {Float} │ │ │ │ │ + */ │ │ │ │ │ + maxScale: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: displayOutsideMaxExtent │ │ │ │ │ + * {Boolean} Request map tiles that are completely outside of the max │ │ │ │ │ + * extent for this layer. Defaults to false. │ │ │ │ │ + */ │ │ │ │ │ + displayOutsideMaxExtent: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: wrapDateLine │ │ │ │ │ + * {Boolean} Wraps the world at the international dateline, so the map can │ │ │ │ │ + * be panned infinitely in longitudinal direction. Only use this on the │ │ │ │ │ + * base layer, and only if the layer's maxExtent equals the world bounds. │ │ │ │ │ + * #487 for more info. │ │ │ │ │ + */ │ │ │ │ │ + wrapDateLine: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: metadata │ │ │ │ │ + * {Object} This object can be used to store additional information on a │ │ │ │ │ + * layer object. │ │ │ │ │ + */ │ │ │ │ │ + metadata: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * extent - {} │ │ │ │ │ - * resolutionChanged - {Boolean} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ - * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ - * False otherwise. │ │ │ │ │ + * name - {String} The layer name │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ */ │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - this.extent = extent.clone(); │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - var ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ - extent = extent.scale(1 / ratio); │ │ │ │ │ - this.extent = extent.wrapDateLine(this.map.getMaxExtent()).scale(ratio); │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + │ │ │ │ │ + this.metadata = {}; │ │ │ │ │ + │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + // make sure we respect alwaysInRange if set on the prototype │ │ │ │ │ + if (this.alwaysInRange != null) { │ │ │ │ │ + options.alwaysInRange = this.alwaysInRange; │ │ │ │ │ } │ │ │ │ │ - if (resolutionChanged) { │ │ │ │ │ - this.resolution = null; │ │ │ │ │ + this.addOptions(options); │ │ │ │ │ + │ │ │ │ │ + this.name = name; │ │ │ │ │ + │ │ │ │ │ + if (this.id == null) { │ │ │ │ │ + │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ + │ │ │ │ │ + this.div = OpenLayers.Util.createDiv(this.id); │ │ │ │ │ + this.div.style.width = "100%"; │ │ │ │ │ + this.div.style.height = "100%"; │ │ │ │ │ + this.div.dir = "ltr"; │ │ │ │ │ + │ │ │ │ │ + this.events = new OpenLayers.Events(this, this.div); │ │ │ │ │ + if (this.eventListeners instanceof Object) { │ │ │ │ │ + this.events.on(this.eventListeners); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ } │ │ │ │ │ - return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setSize │ │ │ │ │ - * Sets the size of the drawing surface. │ │ │ │ │ - * │ │ │ │ │ - * Resolution has probably changed, so we nullify the resolution │ │ │ │ │ - * cache (this.resolution) -- this way it will be re-computed when │ │ │ │ │ - * next it is needed. │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Destroy is a destructor: this is to alleviate cyclic references which │ │ │ │ │ + * the Javascript garbage cleaner can not take care of on its own. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * size - {} │ │ │ │ │ + * setNewBaseLayer - {Boolean} Set a new base layer when this layer has │ │ │ │ │ + * been destroyed. Default is true. │ │ │ │ │ */ │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - this.size = size.clone(); │ │ │ │ │ - this.resolution = null; │ │ │ │ │ - }, │ │ │ │ │ + destroy: function(setNewBaseLayer) { │ │ │ │ │ + if (setNewBaseLayer == null) { │ │ │ │ │ + setNewBaseLayer = true; │ │ │ │ │ + } │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.removeLayer(this, setNewBaseLayer); │ │ │ │ │ + } │ │ │ │ │ + this.projection = null; │ │ │ │ │ + this.map = null; │ │ │ │ │ + this.name = null; │ │ │ │ │ + this.div = null; │ │ │ │ │ + this.options = null; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getResolution │ │ │ │ │ - * Uses cached copy of resolution if available to minimize computing │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The current map's resolution │ │ │ │ │ - */ │ │ │ │ │ - getResolution: function() { │ │ │ │ │ - this.resolution = this.resolution || this.map.getResolution(); │ │ │ │ │ - return this.resolution; │ │ │ │ │ + if (this.events) { │ │ │ │ │ + if (this.eventListeners) { │ │ │ │ │ + this.events.un(this.eventListeners); │ │ │ │ │ + } │ │ │ │ │ + this.events.destroy(); │ │ │ │ │ + } │ │ │ │ │ + this.eventListeners = null; │ │ │ │ │ + this.events = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawFeature │ │ │ │ │ - * Draw the feature. The optional style argument can be used │ │ │ │ │ - * to override the feature's own style. This method should only │ │ │ │ │ - * be called from layer.drawFeature(). │ │ │ │ │ + * Method: clone │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {} │ │ │ │ │ - * style - {} │ │ │ │ │ - * │ │ │ │ │ + * obj - {} The layer to be cloned │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} true if the feature has been drawn completely, false if not, │ │ │ │ │ - * undefined if the feature had no geometry │ │ │ │ │ + * {} An exact clone of this │ │ │ │ │ */ │ │ │ │ │ - drawFeature: function(feature, style) { │ │ │ │ │ - if (style == null) { │ │ │ │ │ - style = feature.style; │ │ │ │ │ - } │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - var bounds = feature.geometry.getBounds(); │ │ │ │ │ - if (bounds) { │ │ │ │ │ - var worldBounds; │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - worldBounds = this.map.getMaxExtent(); │ │ │ │ │ - } │ │ │ │ │ - if (!bounds.intersectsBounds(this.extent, { │ │ │ │ │ - worldBounds: worldBounds │ │ │ │ │ - })) { │ │ │ │ │ - style = { │ │ │ │ │ - display: "none" │ │ │ │ │ - }; │ │ │ │ │ - } else { │ │ │ │ │ - this.calculateFeatureDx(bounds, worldBounds); │ │ │ │ │ - } │ │ │ │ │ - var rendered = this.drawGeometry(feature.geometry, style, feature.id); │ │ │ │ │ - if (style.display != "none" && style.label && rendered !== false) { │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ │ │ │ │ │ - var location = feature.geometry.getCentroid(); │ │ │ │ │ - if (style.labelXOffset || style.labelYOffset) { │ │ │ │ │ - var xOffset = isNaN(style.labelXOffset) ? 0 : style.labelXOffset; │ │ │ │ │ - var yOffset = isNaN(style.labelYOffset) ? 0 : style.labelYOffset; │ │ │ │ │ - var res = this.getResolution(); │ │ │ │ │ - location.move(xOffset * res, yOffset * res); │ │ │ │ │ - } │ │ │ │ │ - this.drawText(feature.id, style, location); │ │ │ │ │ - } else { │ │ │ │ │ - this.removeText(feature.id); │ │ │ │ │ - } │ │ │ │ │ - return rendered; │ │ │ │ │ - } │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer(this.name, this.getOptions()); │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + // catch any randomly tagged-on properties │ │ │ │ │ + OpenLayers.Util.applyDefaults(obj, this); │ │ │ │ │ + │ │ │ │ │ + // a cloned layer should never have its map property set │ │ │ │ │ + // because it has not been added to a map yet. │ │ │ │ │ + obj.map = null; │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: calculateFeatureDx │ │ │ │ │ - * {Number} Calculates the feature offset in x direction. Looking at the │ │ │ │ │ - * center of the feature bounds and the renderer extent, we calculate how │ │ │ │ │ - * many world widths the two are away from each other. This distance is │ │ │ │ │ - * used to shift the feature as close as possible to the center of the │ │ │ │ │ - * current enderer extent, which ensures that the feature is visible in the │ │ │ │ │ - * current viewport. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {} Bounds of the feature │ │ │ │ │ - * worldBounds - {} Bounds of the world │ │ │ │ │ + * Method: getOptions │ │ │ │ │ + * Extracts an object from the layer with the properties that were set as │ │ │ │ │ + * options, but updates them with the values currently set on the │ │ │ │ │ + * instance. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} the of the layer, representing the current state. │ │ │ │ │ */ │ │ │ │ │ - calculateFeatureDx: function(bounds, worldBounds) { │ │ │ │ │ - this.featureDx = 0; │ │ │ │ │ - if (worldBounds) { │ │ │ │ │ - var worldWidth = worldBounds.getWidth(), │ │ │ │ │ - rendererCenterX = (this.extent.left + this.extent.right) / 2, │ │ │ │ │ - featureCenterX = (bounds.left + bounds.right) / 2, │ │ │ │ │ - worldsAway = Math.round((featureCenterX - rendererCenterX) / worldWidth); │ │ │ │ │ - this.featureDx = worldsAway * worldWidth; │ │ │ │ │ + getOptions: function() { │ │ │ │ │ + var options = {}; │ │ │ │ │ + for (var o in this.options) { │ │ │ │ │ + options[o] = this[o]; │ │ │ │ │ } │ │ │ │ │ + return options; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawGeometry │ │ │ │ │ - * │ │ │ │ │ - * Draw a geometry. This should only be called from the renderer itself. │ │ │ │ │ - * Use layer.drawFeature() from outside the renderer. │ │ │ │ │ - * virtual function │ │ │ │ │ + * APIMethod: setName │ │ │ │ │ + * Sets the new layer name for this layer. Can trigger a changelayer event │ │ │ │ │ + * on the map. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {} │ │ │ │ │ + * newName - {String} The new name. │ │ │ │ │ */ │ │ │ │ │ - drawGeometry: function(geometry, style, featureId) {}, │ │ │ │ │ + setName: function(newName) { │ │ │ │ │ + if (newName != this.name) { │ │ │ │ │ + this.name = newName; │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "name" │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawText │ │ │ │ │ - * Function for drawing text labels. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ + * APIMethod: addOptions │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * style - │ │ │ │ │ - * location - {} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newOptions - {Object} │ │ │ │ │ + * reinitialize - {Boolean} If set to true, and if resolution options of the │ │ │ │ │ + * current baseLayer were changed, the map will be recentered to make │ │ │ │ │ + * sure that it is displayed with a valid resolution, and a │ │ │ │ │ + * changebaselayer event will be triggered. │ │ │ │ │ */ │ │ │ │ │ - drawText: function(featureId, style, location) {}, │ │ │ │ │ + addOptions: function(newOptions, reinitialize) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeText │ │ │ │ │ - * Function for removing text labels. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - */ │ │ │ │ │ - removeText: function(featureId) {}, │ │ │ │ │ + if (this.options == null) { │ │ │ │ │ + this.options = {}; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (newOptions) { │ │ │ │ │ + // make sure this.projection references a projection object │ │ │ │ │ + if (typeof newOptions.projection == "string") { │ │ │ │ │ + newOptions.projection = new OpenLayers.Projection(newOptions.projection); │ │ │ │ │ + } │ │ │ │ │ + if (newOptions.projection) { │ │ │ │ │ + // get maxResolution, units and maxExtent from projection defaults if │ │ │ │ │ + // they are not defined already │ │ │ │ │ + OpenLayers.Util.applyDefaults(newOptions, │ │ │ │ │ + OpenLayers.Projection.defaults[newOptions.projection.getCode()]); │ │ │ │ │ + } │ │ │ │ │ + // allow array for extents │ │ │ │ │ + if (newOptions.maxExtent && !(newOptions.maxExtent instanceof OpenLayers.Bounds)) { │ │ │ │ │ + newOptions.maxExtent = new OpenLayers.Bounds(newOptions.maxExtent); │ │ │ │ │ + } │ │ │ │ │ + if (newOptions.minExtent && !(newOptions.minExtent instanceof OpenLayers.Bounds)) { │ │ │ │ │ + newOptions.minExtent = new OpenLayers.Bounds(newOptions.minExtent); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // update our copy for clone │ │ │ │ │ + OpenLayers.Util.extend(this.options, newOptions); │ │ │ │ │ + │ │ │ │ │ + // add new options to this │ │ │ │ │ + OpenLayers.Util.extend(this, newOptions); │ │ │ │ │ + │ │ │ │ │ + // get the units from the projection, if we have a projection │ │ │ │ │ + // and it it has units │ │ │ │ │ + if (this.projection && this.projection.getUnits()) { │ │ │ │ │ + this.units = this.projection.getUnits(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // re-initialize resolutions if necessary, i.e. if any of the │ │ │ │ │ + // properties of the "properties" array defined below is set │ │ │ │ │ + // in the new options │ │ │ │ │ + if (this.map) { │ │ │ │ │ + // store current resolution so we can try to restore it later │ │ │ │ │ + var resolution = this.map.getResolution(); │ │ │ │ │ + var properties = this.RESOLUTION_PROPERTIES.concat( │ │ │ │ │ + ["projection", "units", "minExtent", "maxExtent"] │ │ │ │ │ + ); │ │ │ │ │ + for (var o in newOptions) { │ │ │ │ │ + if (newOptions.hasOwnProperty(o) && │ │ │ │ │ + OpenLayers.Util.indexOf(properties, o) >= 0) { │ │ │ │ │ + │ │ │ │ │ + this.initResolutions(); │ │ │ │ │ + if (reinitialize && this.map.baseLayer === this) { │ │ │ │ │ + // update map position, and restore previous resolution │ │ │ │ │ + this.map.setCenter(this.map.getCenter(), │ │ │ │ │ + this.map.getZoomForResolution(resolution), │ │ │ │ │ + false, true │ │ │ │ │ + ); │ │ │ │ │ + // trigger a changebaselayer event to make sure that │ │ │ │ │ + // all controls (especially │ │ │ │ │ + // OpenLayers.Control.PanZoomBar) get notified of the │ │ │ │ │ + // new options │ │ │ │ │ + this.map.events.triggerEvent("changebaselayer", { │ │ │ │ │ + layer: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clear │ │ │ │ │ - * Clear all vectors from the renderer. │ │ │ │ │ - * virtual function. │ │ │ │ │ + * APIMethod: onMapResize │ │ │ │ │ + * This function can be implemented by subclasses │ │ │ │ │ */ │ │ │ │ │ - clear: function() {}, │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + //this function can be implemented by subclasses │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureIdFromEvent │ │ │ │ │ - * Returns a feature id from an event on the renderer. │ │ │ │ │ - * How this happens is specific to the renderer. This should be │ │ │ │ │ - * called from layer.getFeatureFromEvent(). │ │ │ │ │ - * Virtual function. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {} │ │ │ │ │ + * APIMethod: redraw │ │ │ │ │ + * Redraws the layer. Returns true if the layer was redrawn, false if not. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A feature id or undefined. │ │ │ │ │ + * {Boolean} The layer was redrawn. │ │ │ │ │ */ │ │ │ │ │ - getFeatureIdFromEvent: function(evt) {}, │ │ │ │ │ + redraw: function() { │ │ │ │ │ + var redrawn = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: eraseFeatures │ │ │ │ │ - * This is called by the layer to erase features │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array()} │ │ │ │ │ - */ │ │ │ │ │ - eraseFeatures: function(features) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - this.eraseGeometry(feature.geometry, feature.id); │ │ │ │ │ - this.removeText(feature.id); │ │ │ │ │ + // min/max Range may have changed │ │ │ │ │ + this.inRange = this.calculateInRange(); │ │ │ │ │ + │ │ │ │ │ + // map's center might not yet be set │ │ │ │ │ + var extent = this.getExtent(); │ │ │ │ │ + │ │ │ │ │ + if (extent && this.inRange && this.visibility) { │ │ │ │ │ + var zoomChanged = true; │ │ │ │ │ + this.moveTo(extent, zoomChanged, false); │ │ │ │ │ + this.events.triggerEvent("moveend", { │ │ │ │ │ + "zoomChanged": zoomChanged │ │ │ │ │ + }); │ │ │ │ │ + redrawn = true; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return redrawn; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: eraseGeometry │ │ │ │ │ - * Remove a geometry from the renderer (by id). │ │ │ │ │ - * virtual function. │ │ │ │ │ + * Method: moveTo │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * bounds - {} │ │ │ │ │ + * zoomChanged - {Boolean} Tells when zoom has changed, as layers have to │ │ │ │ │ + * do some init work in that case. │ │ │ │ │ + * dragging - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - eraseGeometry: function(geometry, featureId) {}, │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + var display = this.visibility; │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + display = display && this.inRange; │ │ │ │ │ + } │ │ │ │ │ + this.display(display); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveRoot │ │ │ │ │ - * moves this renderer's root to a (different) renderer. │ │ │ │ │ - * To be implemented by subclasses that require a common renderer root for │ │ │ │ │ - * feature selection. │ │ │ │ │ - * │ │ │ │ │ + * Method: moveByPx │ │ │ │ │ + * Move the layer based on pixel vector. To be implemented by subclasses. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * renderer - {} target renderer for the moved root │ │ │ │ │ + * dx - {Number} The x coord of the displacement vector. │ │ │ │ │ + * dy - {Number} The y coord of the displacement vector. │ │ │ │ │ */ │ │ │ │ │ - moveRoot: function(renderer) {}, │ │ │ │ │ + moveByPx: function(dx, dy) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getRenderLayerId │ │ │ │ │ - * Gets the layer that this renderer's output appears on. If moveRoot was │ │ │ │ │ - * used, this will be different from the id of the layer containing the │ │ │ │ │ - * features rendered by this renderer. │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the layer. This is done through an accessor │ │ │ │ │ + * so that subclasses can override this and take special action once │ │ │ │ │ + * they have their map variable set. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} the id of the output layer. │ │ │ │ │ - */ │ │ │ │ │ - getRenderLayerId: function() { │ │ │ │ │ - return this.container.id; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: applyDefaultSymbolizer │ │ │ │ │ + * Here we take care to bring over any of the necessary default │ │ │ │ │ + * properties from the map. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * symbolizer - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} │ │ │ │ │ + * map - {} │ │ │ │ │ */ │ │ │ │ │ - applyDefaultSymbolizer: function(symbolizer) { │ │ │ │ │ - var result = OpenLayers.Util.extend({}, │ │ │ │ │ - OpenLayers.Renderer.defaultSymbolizer); │ │ │ │ │ - if (symbolizer.stroke === false) { │ │ │ │ │ - delete result.strokeWidth; │ │ │ │ │ - delete result.strokeColor; │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.fill === false) { │ │ │ │ │ - delete result.fillColor; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.extend(result, symbolizer); │ │ │ │ │ - return result; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.defaultSymbolizer │ │ │ │ │ - * {Object} Properties from this symbolizer will be applied to symbolizers │ │ │ │ │ - * with missing properties. This can also be used to set a global │ │ │ │ │ - * symbolizer default in OpenLayers. To be SLD 1.x compliant, add the │ │ │ │ │ - * following code before rendering any vector features: │ │ │ │ │ - * (code) │ │ │ │ │ - * OpenLayers.Renderer.defaultSymbolizer = { │ │ │ │ │ - * fillColor: "#808080", │ │ │ │ │ - * fillOpacity: 1, │ │ │ │ │ - * strokeColor: "#000000", │ │ │ │ │ - * strokeOpacity: 1, │ │ │ │ │ - * strokeWidth: 1, │ │ │ │ │ - * pointRadius: 3, │ │ │ │ │ - * graphicName: "square" │ │ │ │ │ - * }; │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.defaultSymbolizer = { │ │ │ │ │ - fillColor: "#000000", │ │ │ │ │ - strokeColor: "#000000", │ │ │ │ │ - strokeWidth: 2, │ │ │ │ │ - fillOpacity: 1, │ │ │ │ │ - strokeOpacity: 1, │ │ │ │ │ - pointRadius: 0, │ │ │ │ │ - labelAlign: 'cm' │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.symbol │ │ │ │ │ - * Coordinate arrays for well known (named) symbols. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.symbol = { │ │ │ │ │ - "star": [350, 75, 379, 161, 469, 161, 397, 215, 423, 301, 350, 250, 277, 301, │ │ │ │ │ - 303, 215, 231, 161, 321, 161, 350, 75 │ │ │ │ │ - ], │ │ │ │ │ - "cross": [4, 0, 6, 0, 6, 4, 10, 4, 10, 6, 6, 6, 6, 10, 4, 10, 4, 6, 0, 6, 0, 4, 4, 4, │ │ │ │ │ - 4, 0 │ │ │ │ │ - ], │ │ │ │ │ - "x": [0, 0, 25, 0, 50, 35, 75, 0, 100, 0, 65, 50, 100, 100, 75, 100, 50, 65, 25, 100, 0, 100, 35, 50, 0, 0], │ │ │ │ │ - "square": [0, 0, 0, 1, 1, 1, 1, 0, 0, 0], │ │ │ │ │ - "triangle": [0, 10, 10, 10, 5, 0, 0, 10] │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Spherical.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/SingleFile.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: Spherical │ │ │ │ │ - * The OpenLayers.Spherical namespace includes utility functions for │ │ │ │ │ - * calculations on the basis of a spherical earth (ignoring ellipsoidal │ │ │ │ │ - * effects), which is accurate enough for most purposes. │ │ │ │ │ - * │ │ │ │ │ - * Relevant links: │ │ │ │ │ - * * http://www.movable-type.co.uk/scripts/latlong.html │ │ │ │ │ - * * http://code.google.com/apis/maps/documentation/javascript/reference.html#spherical │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Spherical = OpenLayers.Spherical || {}; │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + if (this.map == null) { │ │ │ │ │ │ │ │ │ │ -OpenLayers.Spherical.DEFAULT_RADIUS = 6378137; │ │ │ │ │ + this.map = map; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * APIFunction: computeDistanceBetween │ │ │ │ │ - * Computes the distance between two LonLats. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * from - {} or {Object} Starting point. A LonLat or │ │ │ │ │ - * a JavaScript literal with lon lat properties. │ │ │ │ │ - * to - {} or {Object} Ending point. A LonLat or a │ │ │ │ │ - * JavaScript literal with lon lat properties. │ │ │ │ │ - * radius - {Float} The radius. Optional. Defaults to 6378137 meters. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The distance in meters. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Spherical.computeDistanceBetween = function(from, to, radius) { │ │ │ │ │ - var R = radius || OpenLayers.Spherical.DEFAULT_RADIUS; │ │ │ │ │ - var sinHalfDeltaLon = Math.sin(Math.PI * (to.lon - from.lon) / 360); │ │ │ │ │ - var sinHalfDeltaLat = Math.sin(Math.PI * (to.lat - from.lat) / 360); │ │ │ │ │ - var a = sinHalfDeltaLat * sinHalfDeltaLat + │ │ │ │ │ - sinHalfDeltaLon * sinHalfDeltaLon * Math.cos(Math.PI * from.lat / 180) * Math.cos(Math.PI * to.lat / 180); │ │ │ │ │ - return 2 * R * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a)); │ │ │ │ │ -}; │ │ │ │ │ + // grab some essential layer data from the map if it hasn't already │ │ │ │ │ + // been set │ │ │ │ │ + this.maxExtent = this.maxExtent || this.map.maxExtent; │ │ │ │ │ + this.minExtent = this.minExtent || this.map.minExtent; │ │ │ │ │ │ │ │ │ │ + this.projection = this.projection || this.map.projection; │ │ │ │ │ + if (typeof this.projection == "string") { │ │ │ │ │ + this.projection = new OpenLayers.Projection(this.projection); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * APIFunction: computeHeading │ │ │ │ │ - * Computes the heading from one LonLat to another LonLat. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * from - {} or {Object} Starting point. A LonLat or │ │ │ │ │ - * a JavaScript literal with lon lat properties. │ │ │ │ │ - * to - {} or {Object} Ending point. A LonLat or a │ │ │ │ │ - * JavaScript literal with lon lat properties. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The heading in degrees. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Spherical.computeHeading = function(from, to) { │ │ │ │ │ - var y = Math.sin(Math.PI * (from.lon - to.lon) / 180) * Math.cos(Math.PI * to.lat / 180); │ │ │ │ │ - var x = Math.cos(Math.PI * from.lat / 180) * Math.sin(Math.PI * to.lat / 180) - │ │ │ │ │ - Math.sin(Math.PI * from.lat / 180) * Math.cos(Math.PI * to.lat / 180) * Math.cos(Math.PI * (from.lon - to.lon) / 180); │ │ │ │ │ - return 180 * Math.atan2(y, x) / Math.PI; │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Symbolizer/Point.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + // Check the projection to see if we can get units -- if not, refer │ │ │ │ │ + // to properties. │ │ │ │ │ + this.units = this.projection.getUnits() || │ │ │ │ │ + this.units || this.map.units; │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + this.initResolutions(); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Symbolizer.js │ │ │ │ │ - */ │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + this.inRange = this.calculateInRange(); │ │ │ │ │ + var show = ((this.visibility) && (this.inRange)); │ │ │ │ │ + this.div.style.display = show ? "" : "none"; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Symbolizer.Point │ │ │ │ │ - * A symbolizer used to render point features. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Symbolizer.Point = OpenLayers.Class(OpenLayers.Symbolizer, { │ │ │ │ │ + // deal with gutters │ │ │ │ │ + this.setTileSize(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: strokeColor │ │ │ │ │ - * {String} Color for line stroke. This is a RGB hex value (e.g. "#ff0000" │ │ │ │ │ - * for red). │ │ │ │ │ - * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + * Method: afterAdd │ │ │ │ │ + * Called at the end of the map.addLayer sequence. At this point, the map │ │ │ │ │ + * will have a base layer. To be overridden by subclasses. │ │ │ │ │ */ │ │ │ │ │ + afterAdd: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: strokeOpacity │ │ │ │ │ - * {Number} Stroke opacity (0-1). │ │ │ │ │ + * APIMethod: removeMap │ │ │ │ │ + * Just as setMap() allows each layer the possibility to take a │ │ │ │ │ + * personalized action on being added to the map, removeMap() allows │ │ │ │ │ + * each layer to take a personalized action on being removed from it. │ │ │ │ │ + * For now, this will be mostly unused, except for the EventPane layer, │ │ │ │ │ + * which needs this hook so that it can remove the special invisible │ │ │ │ │ + * pane. │ │ │ │ │ * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {} │ │ │ │ │ */ │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + //to be overridden by subclasses │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: strokeWidth │ │ │ │ │ - * {Number} Pixel stroke width. │ │ │ │ │ + * APIMethod: getImageSize │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {} optional tile bounds, can be used │ │ │ │ │ + * by subclasses that have to deal with different tile sizes at the │ │ │ │ │ + * layer extent edges (e.g. Zoomify) │ │ │ │ │ * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} The size that the image should be, taking into │ │ │ │ │ + * account gutters. │ │ │ │ │ */ │ │ │ │ │ + getImageSize: function(bounds) { │ │ │ │ │ + return (this.imageSize || this.tileSize); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: strokeLinecap │ │ │ │ │ - * {String} Stroke cap type ("butt", "round", or "square"). │ │ │ │ │ + * APIMethod: setTileSize │ │ │ │ │ + * Set the tile size based on the map size. This also sets layer.imageSize │ │ │ │ │ + * or use by Tile.Image. │ │ │ │ │ * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * size - {} │ │ │ │ │ */ │ │ │ │ │ + setTileSize: function(size) { │ │ │ │ │ + var tileSize = (size) ? size : │ │ │ │ │ + ((this.tileSize) ? this.tileSize : │ │ │ │ │ + this.map.getTileSize()); │ │ │ │ │ + this.tileSize = tileSize; │ │ │ │ │ + if (this.gutter) { │ │ │ │ │ + // layers with gutters need non-null tile sizes │ │ │ │ │ + //if(tileSize == null) { │ │ │ │ │ + // OpenLayers.console.error("Error in layer.setMap() for " + │ │ │ │ │ + // this.name + ": layers with " + │ │ │ │ │ + // "gutters need non-null tile sizes"); │ │ │ │ │ + //} │ │ │ │ │ + this.imageSize = new OpenLayers.Size(tileSize.w + (2 * this.gutter), │ │ │ │ │ + tileSize.h + (2 * this.gutter)); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: strokeDashstyle │ │ │ │ │ - * {String} Stroke dash style according to the SLD spec. Note that the │ │ │ │ │ - * OpenLayers values for strokeDashstyle ("dot", "dash", "dashdot", │ │ │ │ │ - * "longdash", "longdashdot", or "solid") will not work in SLD, but │ │ │ │ │ - * most SLD patterns will render correctly in OpenLayers. │ │ │ │ │ + * APIMethod: getVisibility │ │ │ │ │ * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The layer should be displayed (if in range). │ │ │ │ │ */ │ │ │ │ │ + getVisibility: function() { │ │ │ │ │ + return this.visibility; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: fillColor │ │ │ │ │ - * {String} RGB hex fill color (e.g. "#ff0000" for red). │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setVisibility │ │ │ │ │ + * Set the visibility flag for the layer and hide/show & redraw │ │ │ │ │ + * accordingly. Fire event unless otherwise specified │ │ │ │ │ * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: fillOpacity │ │ │ │ │ - * {Number} Fill opacity (0-1). │ │ │ │ │ + * Note that visibility is no longer simply whether or not the layer's │ │ │ │ │ + * style.display is set to "block". Now we store a 'visibility' state │ │ │ │ │ + * property on the layer class, this allows us to remember whether or │ │ │ │ │ + * not we *desire* for a layer to be visible. In the case where the │ │ │ │ │ + * map's resolution is out of the layer's range, this desire may be │ │ │ │ │ + * subverted. │ │ │ │ │ * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * visibility - {Boolean} Whether or not to display the layer (if in range) │ │ │ │ │ */ │ │ │ │ │ + setVisibility: function(visibility) { │ │ │ │ │ + if (visibility != this.visibility) { │ │ │ │ │ + this.visibility = visibility; │ │ │ │ │ + this.display(visibility); │ │ │ │ │ + this.redraw(); │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "visibility" │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("visibilitychanged"); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: pointRadius │ │ │ │ │ - * {Number} Pixel point radius. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: display │ │ │ │ │ + * Hide or show the Layer. This is designed to be used internally, and │ │ │ │ │ + * is not generally the way to enable or disable the layer. For that, │ │ │ │ │ + * use the setVisibility function instead.. │ │ │ │ │ * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * display - {Boolean} │ │ │ │ │ */ │ │ │ │ │ + display: function(display) { │ │ │ │ │ + if (display != (this.div.style.display != "none")) { │ │ │ │ │ + this.div.style.display = (display && this.calculateInRange()) ? "block" : "none"; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: externalGraphic │ │ │ │ │ - * {String} Url to an external graphic that will be used for rendering │ │ │ │ │ - * points. │ │ │ │ │ + * APIMethod: calculateInRange │ │ │ │ │ * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The layer is displayable at the current map's current │ │ │ │ │ + * resolution. Note that if 'alwaysInRange' is true for the layer, │ │ │ │ │ + * this function will always return true. │ │ │ │ │ */ │ │ │ │ │ + calculateInRange: function() { │ │ │ │ │ + var inRange = false; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: graphicWidth │ │ │ │ │ - * {Number} Pixel width for sizing an external graphic. │ │ │ │ │ - * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ + if (this.alwaysInRange) { │ │ │ │ │ + inRange = true; │ │ │ │ │ + } else { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + var resolution = this.map.getResolution(); │ │ │ │ │ + inRange = ((resolution >= this.minResolution) && │ │ │ │ │ + (resolution <= this.maxResolution)); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return inRange; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: graphicHeight │ │ │ │ │ - * {Number} Pixel height for sizing an external graphic. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setIsBaseLayer │ │ │ │ │ * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * isBaseLayer - {Boolean} │ │ │ │ │ */ │ │ │ │ │ + setIsBaseLayer: function(isBaseLayer) { │ │ │ │ │ + if (isBaseLayer != this.isBaseLayer) { │ │ │ │ │ + this.isBaseLayer = isBaseLayer; │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changebaselayer", { │ │ │ │ │ + layer: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: graphicOpacity │ │ │ │ │ - * {Number} Opacity (0-1) for an external graphic. │ │ │ │ │ - * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ + /********************************************************/ │ │ │ │ │ + /* */ │ │ │ │ │ + /* Baselayer Functions */ │ │ │ │ │ + /* */ │ │ │ │ │ + /********************************************************/ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: graphicXOffset │ │ │ │ │ - * {Number} Pixel offset along the positive x axis for displacing an │ │ │ │ │ - * external graphic. │ │ │ │ │ + /** │ │ │ │ │ + * Method: initResolutions │ │ │ │ │ + * This method's responsibility is to set up the 'resolutions' array │ │ │ │ │ + * for the layer -- this array is what the layer will use to interface │ │ │ │ │ + * between the zoom levels of the map and the resolution display │ │ │ │ │ + * of the layer. │ │ │ │ │ * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + * The user has several options that determine how the array is set up. │ │ │ │ │ + * │ │ │ │ │ + * For a detailed explanation, see the following wiki from the │ │ │ │ │ + * openlayers.org homepage: │ │ │ │ │ + * http://trac.openlayers.org/wiki/SettingZoomLevels │ │ │ │ │ */ │ │ │ │ │ + initResolutions: function() { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: graphicYOffset │ │ │ │ │ - * {Number} Pixel offset along the positive y axis for displacing an │ │ │ │ │ - * external graphic. │ │ │ │ │ - * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ + // ok we want resolutions, here's our strategy: │ │ │ │ │ + // │ │ │ │ │ + // 1. if resolutions are defined in the layer config, use them │ │ │ │ │ + // 2. else, if scales are defined in the layer config then derive │ │ │ │ │ + // resolutions from these scales │ │ │ │ │ + // 3. else, attempt to calculate resolutions from maxResolution, │ │ │ │ │ + // minResolution, numZoomLevels, maxZoomLevel set in the │ │ │ │ │ + // layer config │ │ │ │ │ + // 4. if we still don't have resolutions, and if resolutions │ │ │ │ │ + // are defined in the same, use them │ │ │ │ │ + // 5. else, if scales are defined in the map then derive │ │ │ │ │ + // resolutions from these scales │ │ │ │ │ + // 6. else, attempt to calculate resolutions from maxResolution, │ │ │ │ │ + // minResolution, numZoomLevels, maxZoomLevel set in the │ │ │ │ │ + // map │ │ │ │ │ + // 7. hope for the best! │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: rotation │ │ │ │ │ - * {Number} The rotation of a graphic in the clockwise direction about its │ │ │ │ │ - * center point (or any point off center as specified by │ │ │ │ │ - * and ). │ │ │ │ │ - * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ + var i, len, p; │ │ │ │ │ + var props = {}, │ │ │ │ │ + alwaysInRange = true; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: graphicName │ │ │ │ │ - * {String} Named graphic to use when rendering points. Supported values │ │ │ │ │ - * include "circle", "square", "star", "x", "cross", and "triangle". │ │ │ │ │ - * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ + // get resolution data from layer config │ │ │ │ │ + // (we also set alwaysInRange in the layer as appropriate) │ │ │ │ │ + for (i = 0, len = this.RESOLUTION_PROPERTIES.length; i < len; i++) { │ │ │ │ │ + p = this.RESOLUTION_PROPERTIES[i]; │ │ │ │ │ + props[p] = this.options[p]; │ │ │ │ │ + if (alwaysInRange && this.options[p]) { │ │ │ │ │ + alwaysInRange = false; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.options.alwaysInRange == null) { │ │ │ │ │ + this.alwaysInRange = alwaysInRange; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Symbolizer.Point │ │ │ │ │ - * Create a symbolizer for rendering points. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} An object containing properties to be set on the │ │ │ │ │ - * symbolizer. Any documented symbolizer property can be set at │ │ │ │ │ - * construction. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * A new point symbolizer. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ - OpenLayers.Symbolizer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + // if we don't have resolutions then attempt to derive them from scales │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + props.resolutions = this.resolutionsFromScales(props.scales); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Symbolizer.Point" │ │ │ │ │ + // if we still don't have resolutions then attempt to calculate them │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + props.resolutions = this.calculateResolutions(props); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ + // if we couldn't calculate resolutions then we look at we have │ │ │ │ │ + // in the map │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + for (i = 0, len = this.RESOLUTION_PROPERTIES.length; i < len; i++) { │ │ │ │ │ + p = this.RESOLUTION_PROPERTIES[i]; │ │ │ │ │ + props[p] = this.options[p] != null ? │ │ │ │ │ + this.options[p] : this.map[p]; │ │ │ │ │ + } │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + props.resolutions = this.resolutionsFromScales(props.scales); │ │ │ │ │ + } │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + props.resolutions = this.calculateResolutions(props); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Symbolizer/Line.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + // ok, we new need to set properties in the instance │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + // get maxResolution from the config if it's defined there │ │ │ │ │ + var maxResolution; │ │ │ │ │ + if (this.options.maxResolution && │ │ │ │ │ + this.options.maxResolution !== "auto") { │ │ │ │ │ + maxResolution = this.options.maxResolution; │ │ │ │ │ + } │ │ │ │ │ + if (this.options.minScale) { │ │ │ │ │ + maxResolution = OpenLayers.Util.getResolutionFromScale( │ │ │ │ │ + this.options.minScale, this.units); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Symbolizer.js │ │ │ │ │ - */ │ │ │ │ │ + // get minResolution from the config if it's defined there │ │ │ │ │ + var minResolution; │ │ │ │ │ + if (this.options.minResolution && │ │ │ │ │ + this.options.minResolution !== "auto") { │ │ │ │ │ + minResolution = this.options.minResolution; │ │ │ │ │ + } │ │ │ │ │ + if (this.options.maxScale) { │ │ │ │ │ + minResolution = OpenLayers.Util.getResolutionFromScale( │ │ │ │ │ + this.options.maxScale, this.units); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Symbolizer.Line │ │ │ │ │ - * A symbolizer used to render line features. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Symbolizer.Line = OpenLayers.Class(OpenLayers.Symbolizer, { │ │ │ │ │ + if (props.resolutions) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: strokeColor │ │ │ │ │ - * {String} Color for line stroke. This is a RGB hex value (e.g. "#ff0000" │ │ │ │ │ - * for red). │ │ │ │ │ - * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ + //sort resolutions array descendingly │ │ │ │ │ + props.resolutions.sort(function(a, b) { │ │ │ │ │ + return (b - a); │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: strokeOpacity │ │ │ │ │ - * {Number} Stroke opacity (0-1). │ │ │ │ │ - * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ + // if we still don't have a maxResolution get it from the │ │ │ │ │ + // resolutions array │ │ │ │ │ + if (!maxResolution) { │ │ │ │ │ + maxResolution = props.resolutions[0]; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: strokeWidth │ │ │ │ │ - * {Number} Pixel stroke width. │ │ │ │ │ - * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ + // if we still don't have a minResolution get it from the │ │ │ │ │ + // resolutions array │ │ │ │ │ + if (!minResolution) { │ │ │ │ │ + var lastIdx = props.resolutions.length - 1; │ │ │ │ │ + minResolution = props.resolutions[lastIdx]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: strokeLinecap │ │ │ │ │ - * {String} Stroke cap type ("butt", "round", or "square"). │ │ │ │ │ - * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ + this.resolutions = props.resolutions; │ │ │ │ │ + if (this.resolutions) { │ │ │ │ │ + len = this.resolutions.length; │ │ │ │ │ + this.scales = new Array(len); │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + this.scales[i] = OpenLayers.Util.getScaleFromResolution( │ │ │ │ │ + this.resolutions[i], this.units); │ │ │ │ │ + } │ │ │ │ │ + this.numZoomLevels = len; │ │ │ │ │ + } │ │ │ │ │ + this.minResolution = minResolution; │ │ │ │ │ + if (minResolution) { │ │ │ │ │ + this.maxScale = OpenLayers.Util.getScaleFromResolution( │ │ │ │ │ + minResolution, this.units); │ │ │ │ │ + } │ │ │ │ │ + this.maxResolution = maxResolution; │ │ │ │ │ + if (maxResolution) { │ │ │ │ │ + this.minScale = OpenLayers.Util.getScaleFromResolution( │ │ │ │ │ + maxResolution, this.units); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: strokeDashstyle │ │ │ │ │ - * {String} Stroke dash style according to the SLD spec. Note that the │ │ │ │ │ - * OpenLayers values for strokeDashstyle ("dot", "dash", "dashdot", │ │ │ │ │ - * "longdash", "longdashdot", or "solid") will not work in SLD, but │ │ │ │ │ - * most SLD patterns will render correctly in OpenLayers. │ │ │ │ │ - * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + * Method: resolutionsFromScales │ │ │ │ │ + * Derive resolutions from scales. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * scales - {Array(Number)} Scales │ │ │ │ │ + * │ │ │ │ │ + * Returns │ │ │ │ │ + * {Array(Number)} Resolutions │ │ │ │ │ */ │ │ │ │ │ + resolutionsFromScales: function(scales) { │ │ │ │ │ + if (scales == null) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var resolutions, i, len; │ │ │ │ │ + len = scales.length; │ │ │ │ │ + resolutions = new Array(len); │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + resolutions[i] = OpenLayers.Util.getResolutionFromScale( │ │ │ │ │ + scales[i], this.units); │ │ │ │ │ + } │ │ │ │ │ + return resolutions; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Symbolizer.Line │ │ │ │ │ - * Create a symbolizer for rendering lines. │ │ │ │ │ + * Method: calculateResolutions │ │ │ │ │ + * Calculate resolutions based on the provided properties. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * config - {Object} An object containing properties to be set on the │ │ │ │ │ - * symbolizer. Any documented symbolizer property can be set at │ │ │ │ │ - * construction. │ │ │ │ │ + * props - {Object} Properties │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * A new line symbolizer. │ │ │ │ │ + * {Array({Number})} Array of resolutions. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ - OpenLayers.Symbolizer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + calculateResolutions: function(props) { │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Symbolizer.Line" │ │ │ │ │ + var viewSize, wRes, hRes; │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ + // determine maxResolution │ │ │ │ │ + var maxResolution = props.maxResolution; │ │ │ │ │ + if (props.minScale != null) { │ │ │ │ │ + maxResolution = │ │ │ │ │ + OpenLayers.Util.getResolutionFromScale(props.minScale, │ │ │ │ │ + this.units); │ │ │ │ │ + } else if (maxResolution == "auto" && this.maxExtent != null) { │ │ │ │ │ + viewSize = this.map.getSize(); │ │ │ │ │ + wRes = this.maxExtent.getWidth() / viewSize.w; │ │ │ │ │ + hRes = this.maxExtent.getHeight() / viewSize.h; │ │ │ │ │ + maxResolution = Math.max(wRes, hRes); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Symbolizer/Polygon.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + // determine minResolution │ │ │ │ │ + var minResolution = props.minResolution; │ │ │ │ │ + if (props.maxScale != null) { │ │ │ │ │ + minResolution = │ │ │ │ │ + OpenLayers.Util.getResolutionFromScale(props.maxScale, │ │ │ │ │ + this.units); │ │ │ │ │ + } else if (props.minResolution == "auto" && this.minExtent != null) { │ │ │ │ │ + viewSize = this.map.getSize(); │ │ │ │ │ + wRes = this.minExtent.getWidth() / viewSize.w; │ │ │ │ │ + hRes = this.minExtent.getHeight() / viewSize.h; │ │ │ │ │ + minResolution = Math.max(wRes, hRes); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + if (typeof maxResolution !== "number" && │ │ │ │ │ + typeof minResolution !== "number" && │ │ │ │ │ + this.maxExtent != null) { │ │ │ │ │ + // maxResolution for default grid sets assumes that at zoom │ │ │ │ │ + // level zero, the whole world fits on one tile. │ │ │ │ │ + var tileSize = this.map.getTileSize(); │ │ │ │ │ + maxResolution = Math.max( │ │ │ │ │ + this.maxExtent.getWidth() / tileSize.w, │ │ │ │ │ + this.maxExtent.getHeight() / tileSize.h │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Symbolizer.js │ │ │ │ │ - */ │ │ │ │ │ + // determine numZoomLevels │ │ │ │ │ + var maxZoomLevel = props.maxZoomLevel; │ │ │ │ │ + var numZoomLevels = props.numZoomLevels; │ │ │ │ │ + if (typeof minResolution === "number" && │ │ │ │ │ + typeof maxResolution === "number" && numZoomLevels === undefined) { │ │ │ │ │ + var ratio = maxResolution / minResolution; │ │ │ │ │ + numZoomLevels = Math.floor(Math.log(ratio) / Math.log(2)) + 1; │ │ │ │ │ + } else if (numZoomLevels === undefined && maxZoomLevel != null) { │ │ │ │ │ + numZoomLevels = maxZoomLevel + 1; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Symbolizer.Polygon │ │ │ │ │ - * A symbolizer used to render line features. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Symbolizer.Polygon = OpenLayers.Class(OpenLayers.Symbolizer, { │ │ │ │ │ + // are we able to calculate resolutions? │ │ │ │ │ + if (typeof numZoomLevels !== "number" || numZoomLevels <= 0 || │ │ │ │ │ + (typeof maxResolution !== "number" && │ │ │ │ │ + typeof minResolution !== "number")) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: strokeColor │ │ │ │ │ - * {String} Color for line stroke. This is a RGB hex value (e.g. "#ff0000" │ │ │ │ │ - * for red). │ │ │ │ │ - * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ + // now we have numZoomLevels and at least one of maxResolution │ │ │ │ │ + // or minResolution, we can populate the resolutions array │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: strokeOpacity │ │ │ │ │ - * {Number} Stroke opacity (0-1). │ │ │ │ │ - * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ + var resolutions = new Array(numZoomLevels); │ │ │ │ │ + var base = 2; │ │ │ │ │ + if (typeof minResolution == "number" && │ │ │ │ │ + typeof maxResolution == "number") { │ │ │ │ │ + // if maxResolution and minResolution are set, we calculate │ │ │ │ │ + // the base for exponential scaling that starts at │ │ │ │ │ + // maxResolution and ends at minResolution in numZoomLevels │ │ │ │ │ + // steps. │ │ │ │ │ + base = Math.pow( │ │ │ │ │ + (maxResolution / minResolution), │ │ │ │ │ + (1 / (numZoomLevels - 1)) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: strokeWidth │ │ │ │ │ - * {Number} Pixel stroke width. │ │ │ │ │ - * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ + var i; │ │ │ │ │ + if (typeof maxResolution === "number") { │ │ │ │ │ + for (i = 0; i < numZoomLevels; i++) { │ │ │ │ │ + resolutions[i] = maxResolution / Math.pow(base, i); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + for (i = 0; i < numZoomLevels; i++) { │ │ │ │ │ + resolutions[numZoomLevels - 1 - i] = │ │ │ │ │ + minResolution * Math.pow(base, i); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: strokeLinecap │ │ │ │ │ - * {String} Stroke cap type ("butt", "round", or "square"). │ │ │ │ │ - * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ + return resolutions; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: strokeDashstyle │ │ │ │ │ - * {String} Stroke dash style according to the SLD spec. Note that the │ │ │ │ │ - * OpenLayers values for strokeDashstyle ("dot", "dash", "dashdot", │ │ │ │ │ - * "longdash", "longdashdot", or "solid") will not work in SLD, but │ │ │ │ │ - * most SLD patterns will render correctly in OpenLayers. │ │ │ │ │ + * APIMethod: getResolution │ │ │ │ │ * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The currently selected resolution of the map, taken from the │ │ │ │ │ + * resolutions array, indexed by current zoom level. │ │ │ │ │ */ │ │ │ │ │ + getResolution: function() { │ │ │ │ │ + var zoom = this.map.getZoom(); │ │ │ │ │ + return this.getResolutionForZoom(zoom); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: fillColor │ │ │ │ │ - * {String} RGB hex fill color (e.g. "#ff0000" for red). │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getExtent │ │ │ │ │ * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} A Bounds object which represents the lon/lat │ │ │ │ │ + * bounds of the current viewPort. │ │ │ │ │ */ │ │ │ │ │ + getExtent: function() { │ │ │ │ │ + // just use stock map calculateBounds function -- passing no arguments │ │ │ │ │ + // means it will user map's current center & resolution │ │ │ │ │ + // │ │ │ │ │ + return this.map.calculateBounds(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: fillOpacity │ │ │ │ │ - * {Number} Fill opacity (0-1). │ │ │ │ │ + * APIMethod: getZoomForExtent │ │ │ │ │ * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Symbolizer.Polygon │ │ │ │ │ - * Create a symbolizer for rendering polygons. │ │ │ │ │ - * │ │ │ │ │ * Parameters: │ │ │ │ │ - * config - {Object} An object containing properties to be set on the │ │ │ │ │ - * symbolizer. Any documented symbolizer property can be set at │ │ │ │ │ - * construction. │ │ │ │ │ + * extent - {} │ │ │ │ │ + * closest - {Boolean} Find the zoom level that most closely fits the │ │ │ │ │ + * specified bounds. Note that this may result in a zoom that does │ │ │ │ │ + * not exactly contain the entire extent. │ │ │ │ │ + * Default is false. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * A new polygon symbolizer. │ │ │ │ │ + * {Integer} The index of the zoomLevel (entry in the resolutions array) │ │ │ │ │ + * for the passed-in extent. We do this by calculating the ideal │ │ │ │ │ + * resolution for the given extent (based on the map size) and then │ │ │ │ │ + * calling getZoomForResolution(), passing along the 'closest' │ │ │ │ │ + * parameter. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ - OpenLayers.Symbolizer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Symbolizer.Polygon" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Symbolizer/Text.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Symbolizer.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Symbolizer.Text │ │ │ │ │ - * A symbolizer used to render text labels for features. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Symbolizer.Text = OpenLayers.Class(OpenLayers.Symbolizer, { │ │ │ │ │ + getZoomForExtent: function(extent, closest) { │ │ │ │ │ + var viewSize = this.map.getSize(); │ │ │ │ │ + var idealResolution = Math.max(extent.getWidth() / viewSize.w, │ │ │ │ │ + extent.getHeight() / viewSize.h); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: label │ │ │ │ │ - * {String} The text for the label. │ │ │ │ │ - * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ + return this.getZoomForResolution(idealResolution, closest); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: fontFamily │ │ │ │ │ - * {String} The font family for the label. │ │ │ │ │ + * Method: getDataExtent │ │ │ │ │ + * Calculates the max extent which includes all of the data for the layer. │ │ │ │ │ + * This function is to be implemented by subclasses. │ │ │ │ │ * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} │ │ │ │ │ */ │ │ │ │ │ + getDataExtent: function() { │ │ │ │ │ + //to be implemented by subclasses │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: fontSize │ │ │ │ │ - * {String} The font size for the label. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getResolutionForZoom │ │ │ │ │ * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: fontWeight │ │ │ │ │ - * {String} The font weight for the label. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * zoom - {Float} │ │ │ │ │ * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} A suitable resolution for the specified zoom. │ │ │ │ │ */ │ │ │ │ │ + getResolutionForZoom: function(zoom) { │ │ │ │ │ + zoom = Math.max(0, Math.min(zoom, this.resolutions.length - 1)); │ │ │ │ │ + var resolution; │ │ │ │ │ + if (this.map.fractionalZoom) { │ │ │ │ │ + var low = Math.floor(zoom); │ │ │ │ │ + var high = Math.ceil(zoom); │ │ │ │ │ + resolution = this.resolutions[low] - │ │ │ │ │ + ((zoom - low) * (this.resolutions[low] - this.resolutions[high])); │ │ │ │ │ + } else { │ │ │ │ │ + resolution = this.resolutions[Math.round(zoom)]; │ │ │ │ │ + } │ │ │ │ │ + return resolution; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: fontStyle │ │ │ │ │ - * {String} The font style for the label. │ │ │ │ │ + * APIMethod: getZoomForResolution │ │ │ │ │ * │ │ │ │ │ - * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Symbolizer.Text │ │ │ │ │ - * Create a symbolizer for rendering text labels. │ │ │ │ │ - * │ │ │ │ │ * Parameters: │ │ │ │ │ - * config - {Object} An object containing properties to be set on the │ │ │ │ │ - * symbolizer. Any documented symbolizer property can be set at │ │ │ │ │ - * construction. │ │ │ │ │ - * │ │ │ │ │ + * resolution - {Float} │ │ │ │ │ + * closest - {Boolean} Find the zoom level that corresponds to the absolute │ │ │ │ │ + * closest resolution, which may result in a zoom whose corresponding │ │ │ │ │ + * resolution is actually smaller than we would have desired (if this │ │ │ │ │ + * is being called from a getZoomForExtent() call, then this means that │ │ │ │ │ + * the returned zoom index might not actually contain the entire │ │ │ │ │ + * extent specified... but it'll be close). │ │ │ │ │ + * Default is false. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * A new text symbolizer. │ │ │ │ │ + * {Integer} The index of the zoomLevel (entry in the resolutions array) │ │ │ │ │ + * that corresponds to the best fit resolution given the passed in │ │ │ │ │ + * value and the 'closest' specification. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ - OpenLayers.Symbolizer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + getZoomForResolution: function(resolution, closest) { │ │ │ │ │ + var zoom, i, len; │ │ │ │ │ + if (this.map.fractionalZoom) { │ │ │ │ │ + var lowZoom = 0; │ │ │ │ │ + var highZoom = this.resolutions.length - 1; │ │ │ │ │ + var highRes = this.resolutions[lowZoom]; │ │ │ │ │ + var lowRes = this.resolutions[highZoom]; │ │ │ │ │ + var res; │ │ │ │ │ + for (i = 0, len = this.resolutions.length; i < len; ++i) { │ │ │ │ │ + res = this.resolutions[i]; │ │ │ │ │ + if (res >= resolution) { │ │ │ │ │ + highRes = res; │ │ │ │ │ + lowZoom = i; │ │ │ │ │ + } │ │ │ │ │ + if (res <= resolution) { │ │ │ │ │ + lowRes = res; │ │ │ │ │ + highZoom = i; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var dRes = highRes - lowRes; │ │ │ │ │ + if (dRes > 0) { │ │ │ │ │ + zoom = lowZoom + ((highRes - resolution) / dRes); │ │ │ │ │ + } else { │ │ │ │ │ + zoom = lowZoom; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + var diff; │ │ │ │ │ + var minDiff = Number.POSITIVE_INFINITY; │ │ │ │ │ + for (i = 0, len = this.resolutions.length; i < len; i++) { │ │ │ │ │ + if (closest) { │ │ │ │ │ + diff = Math.abs(this.resolutions[i] - resolution); │ │ │ │ │ + if (diff > minDiff) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + minDiff = diff; │ │ │ │ │ + } else { │ │ │ │ │ + if (this.resolutions[i] < resolution) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + zoom = Math.max(0, i - 1); │ │ │ │ │ + } │ │ │ │ │ + return zoom; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Symbolizer.Text" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Symbolizer/Raster.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Symbolizer.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Symbolizer.Raster │ │ │ │ │ - * A symbolizer used to render raster images. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Symbolizer.Raster = OpenLayers.Class(OpenLayers.Symbolizer, { │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Symbolizer.Raster │ │ │ │ │ - * Create a symbolizer for rendering rasters. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getLonLatFromViewPortPx │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * config - {Object} An object containing properties to be set on the │ │ │ │ │ - * symbolizer. Any documented symbolizer property can be set at │ │ │ │ │ - * construction. │ │ │ │ │ + * viewPortPx - {|Object} An OpenLayers.Pixel or │ │ │ │ │ + * an object with a 'x' │ │ │ │ │ + * and 'y' properties. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * A new raster symbolizer. │ │ │ │ │ + * {} An OpenLayers.LonLat which is the passed-in │ │ │ │ │ + * view port , translated into lon/lat by the layer. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ - OpenLayers.Symbolizer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Symbolizer.Raster" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Style2.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Rule.js │ │ │ │ │ - * @requires OpenLayers/Symbolizer/Point.js │ │ │ │ │ - * @requires OpenLayers/Symbolizer/Line.js │ │ │ │ │ - * @requires OpenLayers/Symbolizer/Polygon.js │ │ │ │ │ - * @requires OpenLayers/Symbolizer/Text.js │ │ │ │ │ - * @requires OpenLayers/Symbolizer/Raster.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Style2 │ │ │ │ │ - * This class represents a collection of rules for rendering features. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Style2 = OpenLayers.Class({ │ │ │ │ │ + getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ + var lonlat = null; │ │ │ │ │ + var map = this.map; │ │ │ │ │ + if (viewPortPx != null && map.minPx) { │ │ │ │ │ + var res = map.getResolution(); │ │ │ │ │ + var maxExtent = map.getMaxExtent({ │ │ │ │ │ + restricted: true │ │ │ │ │ + }); │ │ │ │ │ + var lon = (viewPortPx.x - map.minPx.x) * res + maxExtent.left; │ │ │ │ │ + var lat = (map.minPx.y - viewPortPx.y) * res + maxExtent.top; │ │ │ │ │ + lonlat = new OpenLayers.LonLat(lon, lat); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: id │ │ │ │ │ - * {String} A unique id for this session. │ │ │ │ │ - */ │ │ │ │ │ - id: null, │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + lonlat = lonlat.wrapDateLine(this.maxExtent); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return lonlat; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: name │ │ │ │ │ - * {String} Style identifier. │ │ │ │ │ + * APIMethod: getViewPortPxFromLonLat │ │ │ │ │ + * Returns a pixel location given a map location. This method will return │ │ │ │ │ + * fractional pixel values. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * lonlat - {|Object} An OpenLayers.LonLat or │ │ │ │ │ + * an object with a 'lon' │ │ │ │ │ + * and 'lat' properties. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} An which is the passed-in │ │ │ │ │ + * lonlat translated into view port pixels. │ │ │ │ │ */ │ │ │ │ │ - name: null, │ │ │ │ │ + getViewPortPxFromLonLat: function(lonlat, resolution) { │ │ │ │ │ + var px = null; │ │ │ │ │ + if (lonlat != null) { │ │ │ │ │ + resolution = resolution || this.map.getResolution(); │ │ │ │ │ + var extent = this.map.calculateBounds(null, resolution); │ │ │ │ │ + px = new OpenLayers.Pixel( │ │ │ │ │ + (1 / resolution * (lonlat.lon - extent.left)), │ │ │ │ │ + (1 / resolution * (extent.top - lonlat.lat)) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return px; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: title │ │ │ │ │ - * {String} Title of this style. │ │ │ │ │ + * APIMethod: setOpacity │ │ │ │ │ + * Sets the opacity for the entire layer (all images) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * opacity - {Float} │ │ │ │ │ */ │ │ │ │ │ - title: null, │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + if (opacity != this.opacity) { │ │ │ │ │ + this.opacity = opacity; │ │ │ │ │ + var childNodes = this.div.childNodes; │ │ │ │ │ + for (var i = 0, len = childNodes.length; i < len; ++i) { │ │ │ │ │ + var element = childNodes[i].firstChild || childNodes[i]; │ │ │ │ │ + var lastChild = childNodes[i].lastChild; │ │ │ │ │ + //TODO de-uglify this │ │ │ │ │ + if (lastChild && lastChild.nodeName.toLowerCase() === "iframe") { │ │ │ │ │ + element = lastChild.parentNode; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.modifyDOMElement(element, null, null, null, │ │ │ │ │ + null, null, null, opacity); │ │ │ │ │ + } │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "opacity" │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: description │ │ │ │ │ - * {String} Description of this style. │ │ │ │ │ + * Method: getZIndex │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} the z-index of this layer │ │ │ │ │ */ │ │ │ │ │ - description: null, │ │ │ │ │ + getZIndex: function() { │ │ │ │ │ + return this.div.style.zIndex; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: layerName │ │ │ │ │ - * {} Name of the layer that this style belongs to, usually │ │ │ │ │ - * according to the NamedLayer attribute of an SLD document. │ │ │ │ │ + * Method: setZIndex │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * zIndex - {Integer} │ │ │ │ │ */ │ │ │ │ │ - layerName: null, │ │ │ │ │ + setZIndex: function(zIndex) { │ │ │ │ │ + this.div.style.zIndex = zIndex; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isDefault │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - isDefault: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: rules │ │ │ │ │ - * {Array()} Collection of rendering rules. │ │ │ │ │ - */ │ │ │ │ │ - rules: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Style2 │ │ │ │ │ - * Creates a style representing a collection of rendering rules. │ │ │ │ │ - * │ │ │ │ │ + * Method: adjustBounds │ │ │ │ │ + * This function will take a bounds, and if wrapDateLine option is set │ │ │ │ │ + * on the layer, it will return a bounds which is wrapped around the │ │ │ │ │ + * world. We do not wrap for bounds which *cross* the │ │ │ │ │ + * maxExtent.left/right, only bounds which are entirely to the left │ │ │ │ │ + * or entirely to the right. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * config - {Object} An object containing properties to be set on the │ │ │ │ │ - * style. Any documented properties may be set at construction. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} A new style object. │ │ │ │ │ + * bounds - {} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ - OpenLayers.Util.extend(this, config); │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ - }, │ │ │ │ │ + adjustBounds: function(bounds) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * nullify references to prevent circular references and memory leaks │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i = 0, len = this.rules.length; i < len; i++) { │ │ │ │ │ - this.rules[i].destroy(); │ │ │ │ │ + if (this.gutter) { │ │ │ │ │ + // Adjust the extent of a bounds in map units by the │ │ │ │ │ + // layer's gutter in pixels. │ │ │ │ │ + var mapGutter = this.gutter * this.map.getResolution(); │ │ │ │ │ + bounds = new OpenLayers.Bounds(bounds.left - mapGutter, │ │ │ │ │ + bounds.bottom - mapGutter, │ │ │ │ │ + bounds.right + mapGutter, │ │ │ │ │ + bounds.top + mapGutter); │ │ │ │ │ } │ │ │ │ │ - delete this.rules; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Clones this style. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} Clone of this style. │ │ │ │ │ - */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - var config = OpenLayers.Util.extend({}, this); │ │ │ │ │ - // clone rules │ │ │ │ │ - if (this.rules) { │ │ │ │ │ - config.rules = []; │ │ │ │ │ - for (var i = 0, len = this.rules.length; i < len; ++i) { │ │ │ │ │ - config.rules.push(this.rules[i].clone()); │ │ │ │ │ - } │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + // wrap around the date line, within the limits of rounding error │ │ │ │ │ + var wrappingOptions = { │ │ │ │ │ + 'rightTolerance': this.getResolution(), │ │ │ │ │ + 'leftTolerance': this.getResolution() │ │ │ │ │ + }; │ │ │ │ │ + bounds = bounds.wrapDateLine(this.maxExtent, wrappingOptions); │ │ │ │ │ + │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Style2(config); │ │ │ │ │ + return bounds; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Style2" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/TileCache.js │ │ │ │ │ + OpenLayers/Layer/HTTPRequest.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/Layer.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.TileCache │ │ │ │ │ - * A read only TileCache layer. Used to requests tiles cached by TileCache in │ │ │ │ │ - * a web accessible cache. This means that you have to pre-populate your │ │ │ │ │ - * cache before this layer can be used. It is meant only to read tiles │ │ │ │ │ - * created by TileCache, and not to make calls to TileCache for tile │ │ │ │ │ - * creation. Create a new instance with the │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ + * Class: OpenLayers.Layer.HTTPRequest │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.TileCache = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ +OpenLayers.Layer.HTTPRequest = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} Treat this layer as a base layer. Default is true. │ │ │ │ │ + * Constant: URL_HASH_FACTOR │ │ │ │ │ + * {Float} Used to hash URL param strings for multi-WMS server selection. │ │ │ │ │ + * Set to the Golden Ratio per Knuth's recommendation. │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + URL_HASH_FACTOR: (Math.sqrt(5) - 1) / 2, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: format │ │ │ │ │ - * {String} Mime type of the images returned. Default is image/png. │ │ │ │ │ + * Property: url │ │ │ │ │ + * {Array(String) or String} This is either an array of url strings or │ │ │ │ │ + * a single url string. │ │ │ │ │ */ │ │ │ │ │ - format: 'image/png', │ │ │ │ │ + url: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: serverResolutions │ │ │ │ │ - * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ - * property if the map resolutions differ from the server. This │ │ │ │ │ - * property serves two purposes. (a) can include │ │ │ │ │ - * resolutions that the server supports and that you don't want to │ │ │ │ │ - * provide with this layer. (b) The map can work with resolutions │ │ │ │ │ - * that aren't supported by the server, i.e. that aren't in │ │ │ │ │ - * . When the map is displayed in such a resolution │ │ │ │ │ - * data for the closest server-supported resolution is loaded and the │ │ │ │ │ - * layer div is stretched as necessary. │ │ │ │ │ + /** │ │ │ │ │ + * Property: params │ │ │ │ │ + * {Object} Hashtable of key/value parameters │ │ │ │ │ */ │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ + params: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: reproject │ │ │ │ │ + * *Deprecated*. See http://docs.openlayers.org/library/spherical_mercator.html │ │ │ │ │ + * for information on the replacement for this functionality. │ │ │ │ │ + * {Boolean} Whether layer should reproject itself based on base layer │ │ │ │ │ + * locations. This allows reprojection onto commercial layers. │ │ │ │ │ + * Default is false: Most layers can't reproject, but layers │ │ │ │ │ + * which can create non-square geographic pixels can, like WMS. │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + reproject: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.TileCache │ │ │ │ │ - * Create a new read only TileCache layer. │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Layer.HTTPRequest │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} Name of the layer displayed in the interface │ │ │ │ │ - * url - {String} Location of the web accessible cache (not the location of │ │ │ │ │ - * your tilecache script!) │ │ │ │ │ - * layername - {String} Layer name as defined in the TileCache │ │ │ │ │ - * configuration │ │ │ │ │ - * options - {Object} Optional object with properties to be set on the │ │ │ │ │ - * layer. Note that you should speficy your resolutions to match │ │ │ │ │ - * your TileCache configuration. This can be done by setting │ │ │ │ │ - * the resolutions array directly (here or on the map), by setting │ │ │ │ │ - * maxResolution and numZoomLevels, or by using scale based properties. │ │ │ │ │ + * name - {String} │ │ │ │ │ + * url - {Array(String) or String} │ │ │ │ │ + * params - {Object} │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, layername, options) { │ │ │ │ │ - this.layername = layername; │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, │ │ │ │ │ - [name, url, {}, options]); │ │ │ │ │ - this.extension = this.format.split('/')[1].toLowerCase(); │ │ │ │ │ - this.extension = (this.extension == 'jpg') ? 'jpeg' : this.extension; │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ + this.url = url; │ │ │ │ │ + if (!this.params) { │ │ │ │ │ + this.params = OpenLayers.Util.extend({}, params); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.url = null; │ │ │ │ │ + this.params = null; │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIMethod: clone │ │ │ │ │ - * obj - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {} An exact clone of this │ │ │ │ │ - * │ │ │ │ │ + * {} An exact clone of this │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ clone: function(obj) { │ │ │ │ │ │ │ │ │ │ if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.TileCache(this.name, │ │ │ │ │ + obj = new OpenLayers.Layer.HTTPRequest(this.name, │ │ │ │ │ this.url, │ │ │ │ │ - this.layername, │ │ │ │ │ + this.params, │ │ │ │ │ this.getOptions()); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ │ │ │ │ │ // copy/set any non-init, non-simple values here │ │ │ │ │ │ │ │ │ │ return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setUrl │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newUrl - {String} │ │ │ │ │ + */ │ │ │ │ │ + setUrl: function(newUrl) { │ │ │ │ │ + this.url = newUrl; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ + * APIMethod: mergeNewParams │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newParams - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * redrawn: {Boolean} whether the layer was actually redrawn. │ │ │ │ │ + */ │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + this.params = OpenLayers.Util.extend(this.params, newParams); │ │ │ │ │ + var ret = this.redraw(); │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "params" │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return ret; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: redraw │ │ │ │ │ + * Redraws the layer. Returns true if the layer was redrawn, false if not. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {} │ │ │ │ │ + * force - {Boolean} Force redraw by adding random parameter. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The layer was redrawn. │ │ │ │ │ + */ │ │ │ │ │ + redraw: function(force) { │ │ │ │ │ + if (force) { │ │ │ │ │ + return this.mergeNewParams({ │ │ │ │ │ + "_olSalt": Math.random() │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + return OpenLayers.Layer.prototype.redraw.apply(this, []); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: selectUrl │ │ │ │ │ + * selectUrl() implements the standard floating-point multiplicative │ │ │ │ │ + * hash function described by Knuth, and hashes the contents of the │ │ │ │ │ + * given param string into a float between 0 and 1. This float is then │ │ │ │ │ + * scaled to the size of the provided urls array, and used to select │ │ │ │ │ + * a URL. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * paramString - {String} │ │ │ │ │ + * urls - {Array(String)} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as parameters. │ │ │ │ │ + * {String} An entry from the urls array, deterministically selected based │ │ │ │ │ + * on the paramString. │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var bbox = this.maxExtent; │ │ │ │ │ - var size = this.tileSize; │ │ │ │ │ - var tileX = Math.round((bounds.left - bbox.left) / (res * size.w)); │ │ │ │ │ - var tileY = Math.round((bounds.bottom - bbox.bottom) / (res * size.h)); │ │ │ │ │ - var tileZ = this.serverResolutions != null ? │ │ │ │ │ - OpenLayers.Util.indexOf(this.serverResolutions, res) : │ │ │ │ │ - this.map.getZoom(); │ │ │ │ │ + selectUrl: function(paramString, urls) { │ │ │ │ │ + var product = 1; │ │ │ │ │ + for (var i = 0, len = paramString.length; i < len; i++) { │ │ │ │ │ + product *= paramString.charCodeAt(i) * this.URL_HASH_FACTOR; │ │ │ │ │ + product -= Math.floor(product); │ │ │ │ │ + } │ │ │ │ │ + return urls[Math.floor(product * urls.length)]; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var components = [ │ │ │ │ │ - this.layername, │ │ │ │ │ - OpenLayers.Number.zeroPad(tileZ, 2), │ │ │ │ │ - OpenLayers.Number.zeroPad(parseInt(tileX / 1000000), 3), │ │ │ │ │ - OpenLayers.Number.zeroPad((parseInt(tileX / 1000) % 1000), 3), │ │ │ │ │ - OpenLayers.Number.zeroPad((parseInt(tileX) % 1000), 3), │ │ │ │ │ - OpenLayers.Number.zeroPad(parseInt(tileY / 1000000), 3), │ │ │ │ │ - OpenLayers.Number.zeroPad((parseInt(tileY / 1000) % 1000), 3), │ │ │ │ │ - OpenLayers.Number.zeroPad((parseInt(tileY) % 1000), 3) + '.' + this.extension │ │ │ │ │ - ]; │ │ │ │ │ - var path = components.join('/'); │ │ │ │ │ - var url = this.url; │ │ │ │ │ + /** │ │ │ │ │ + * Method: getFullRequestString │ │ │ │ │ + * Combine url with layer's params and these newParams. │ │ │ │ │ + * │ │ │ │ │ + * does checking on the serverPath variable, allowing for cases when it │ │ │ │ │ + * is supplied with trailing ? or &, as well as cases where not. │ │ │ │ │ + * │ │ │ │ │ + * return in formatted string like this: │ │ │ │ │ + * "server?key1=value1&key2=value2&key3=value3" │ │ │ │ │ + * │ │ │ │ │ + * WARNING: The altUrl parameter is deprecated and will be removed in 3.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newParams - {Object} │ │ │ │ │ + * altUrl - {String} Use this as the url instead of the layer's url │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ + │ │ │ │ │ + // if not altUrl passed in, use layer's url │ │ │ │ │ + var url = altUrl || this.url; │ │ │ │ │ + │ │ │ │ │ + // create a new params hashtable with all the layer params and the │ │ │ │ │ + // new params together. then convert to string │ │ │ │ │ + var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ + allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ + var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + │ │ │ │ │ + // if url is not a string, it should be an array of strings, │ │ │ │ │ + // in which case we will deterministically select one of them in │ │ │ │ │ + // order to evenly distribute requests to different urls. │ │ │ │ │ + // │ │ │ │ │ if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(path, url); │ │ │ │ │ + url = this.selectUrl(paramsString, url); │ │ │ │ │ } │ │ │ │ │ - url = (url.charAt(url.length - 1) == '/') ? url : url + '/'; │ │ │ │ │ - return url + path; │ │ │ │ │ + │ │ │ │ │ + // ignore parameters that are already in the url search string │ │ │ │ │ + var urlParams = │ │ │ │ │ + OpenLayers.Util.upperCaseObject(OpenLayers.Util.getParameters(url)); │ │ │ │ │ + for (var key in allParams) { │ │ │ │ │ + if (key.toUpperCase() in urlParams) { │ │ │ │ │ + delete allParams[key]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Util.urlAppend(url, paramsString); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.TileCache" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.HTTPRequest" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/EventPane.js │ │ │ │ │ + OpenLayers/Tile/Image.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer.js │ │ │ │ │ + * @requires OpenLayers/Tile.js │ │ │ │ │ + * @requires OpenLayers/Animation.js │ │ │ │ │ * @requires OpenLayers/Util.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.EventPane │ │ │ │ │ - * Base class for 3rd party layers, providing a DOM element which isolates │ │ │ │ │ - * the 3rd-party layer from mouse events. │ │ │ │ │ - * Only used by Google layers. │ │ │ │ │ - * │ │ │ │ │ - * Automatically instantiated by the Google constructor, and not usually instantiated directly. │ │ │ │ │ + * Class: OpenLayers.Tile.Image │ │ │ │ │ + * Instances of OpenLayers.Tile.Image are used to manage the image tiles │ │ │ │ │ + * used by various layers. Create a new image tile with the │ │ │ │ │ + * constructor. │ │ │ │ │ * │ │ │ │ │ - * Create a new event pane layer with the │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ + * - │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.EventPane = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ +OpenLayers.Tile.Image = OpenLayers.Class(OpenLayers.Tile, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: smoothDragPan │ │ │ │ │ - * {Boolean} smoothDragPan determines whether non-public/internal API │ │ │ │ │ - * methods are used for better performance while dragging EventPane │ │ │ │ │ - * layers. When not in sphericalMercator mode, the smoother dragging │ │ │ │ │ - * doesn't actually move north/south directly with the number of │ │ │ │ │ - * pixels moved, resulting in a slight offset when you drag your mouse │ │ │ │ │ - * north south with this option on. If this visual disparity bothers │ │ │ │ │ - * you, you should turn this option off, or use spherical mercator. │ │ │ │ │ - * Default is on. │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {} An events object that handles all │ │ │ │ │ + * events on the tile. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * tile.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to the events): │ │ │ │ │ + * beforeload - Triggered before an image is prepared for loading, when the │ │ │ │ │ + * url for the image is known already. Listeners may call on │ │ │ │ │ + * the tile instance. If they do so, that image will be used and no new │ │ │ │ │ + * one will be created. │ │ │ │ │ */ │ │ │ │ │ - smoothDragPan: true, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: isBaseLayer │ │ │ │ │ - * {Boolean} EventPaned layers are always base layers, by necessity. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String} The URL of the image being requested. No default. Filled in by │ │ │ │ │ + * layer.getURL() function. May be modified by loadstart listeners. │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + url: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isFixed │ │ │ │ │ - * {Boolean} EventPaned layers are fixed by default. │ │ │ │ │ + /** │ │ │ │ │ + * Property: imgDiv │ │ │ │ │ + * {HTMLImageElement} The image for this tile. │ │ │ │ │ */ │ │ │ │ │ - isFixed: true, │ │ │ │ │ + imgDiv: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: pane │ │ │ │ │ - * {DOMElement} A reference to the element that controls the events. │ │ │ │ │ + * Property: frame │ │ │ │ │ + * {DOMElement} The image element is appended to the frame. Any gutter on │ │ │ │ │ + * the image will be hidden behind the frame. If no gutter is set, │ │ │ │ │ + * this will be null. │ │ │ │ │ */ │ │ │ │ │ - pane: null, │ │ │ │ │ + frame: null, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: imageReloadAttempts │ │ │ │ │ + * {Integer} Attempts to load the image. │ │ │ │ │ + */ │ │ │ │ │ + imageReloadAttempts: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: mapObject │ │ │ │ │ - * {Object} This is the object which will be used to load the 3rd party library │ │ │ │ │ - * in the case of the google layer, this will be of type GMap, │ │ │ │ │ - * in the case of the ve layer, this will be of type VEMap │ │ │ │ │ + * Property: layerAlphaHack │ │ │ │ │ + * {Boolean} True if the png alpha hack needs to be applied on the layer's div. │ │ │ │ │ */ │ │ │ │ │ - mapObject: null, │ │ │ │ │ + layerAlphaHack: null, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: asyncRequestId │ │ │ │ │ + * {Integer} ID of an request to see if request is still valid. This is a │ │ │ │ │ + * number which increments by 1 for each asynchronous request. │ │ │ │ │ + */ │ │ │ │ │ + asyncRequestId: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.EventPane │ │ │ │ │ - * Create a new event pane layer │ │ │ │ │ + * APIProperty: maxGetUrlLength │ │ │ │ │ + * {Number} If set, requests that would result in GET urls with more │ │ │ │ │ + * characters than the number provided will be made using form-encoded │ │ │ │ │ + * HTTP POST. It is good practice to avoid urls that are longer than 2048 │ │ │ │ │ + * characters. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + * Caution: │ │ │ │ │ + * Older versions of Gecko based browsers (e.g. Firefox < 3.5) and most │ │ │ │ │ + * Opera versions do not fully support this option. On all browsers, │ │ │ │ │ + * transition effects are not supported if POST requests are used. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (this.pane == null) { │ │ │ │ │ - this.pane = OpenLayers.Util.createDiv(this.div.id + "_EventPane"); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + maxGetUrlLength: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Deconstruct this layer. │ │ │ │ │ + * Property: canvasContext │ │ │ │ │ + * {CanvasRenderingContext2D} A canvas context associated with │ │ │ │ │ + * the tile image. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.mapObject = null; │ │ │ │ │ - this.pane = null; │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ + canvasContext: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the layer. This is done through an accessor │ │ │ │ │ - * so that subclasses can override this and take special action once │ │ │ │ │ - * they have their map variable set. │ │ │ │ │ - * │ │ │ │ │ + * APIProperty: crossOriginKeyword │ │ │ │ │ + * The value of the crossorigin keyword to use when loading images. This is │ │ │ │ │ + * only relevant when using for tiles from remote │ │ │ │ │ + * origins and should be set to either 'anonymous' or 'use-credentials' │ │ │ │ │ + * for servers that send Access-Control-Allow-Origin headers with their │ │ │ │ │ + * tiles. │ │ │ │ │ + */ │ │ │ │ │ + crossOriginKeyword: null, │ │ │ │ │ + │ │ │ │ │ + /** TBD 3.0 - reorder the parameters to the init function to remove │ │ │ │ │ + * URL. the getUrl() function on the layer gets called on │ │ │ │ │ + * each draw(), so no need to specify it here. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Tile.Image │ │ │ │ │ + * Constructor for a new instance. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {} │ │ │ │ │ + * layer - {} layer that the tile will go in. │ │ │ │ │ + * position - {} │ │ │ │ │ + * bounds - {} │ │ │ │ │ + * url - {} Deprecated. Remove me in 3.0. │ │ │ │ │ + * size - {} │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ + initialize: function(layer, position, bounds, url, size, options) { │ │ │ │ │ + OpenLayers.Tile.prototype.initialize.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - this.pane.style.zIndex = parseInt(this.div.style.zIndex) + 1; │ │ │ │ │ - this.pane.style.display = this.div.style.display; │ │ │ │ │ - this.pane.style.width = "100%"; │ │ │ │ │ - this.pane.style.height = "100%"; │ │ │ │ │ - if (OpenLayers.BROWSER_NAME == "msie") { │ │ │ │ │ - this.pane.style.background = │ │ │ │ │ - "url(" + OpenLayers.Util.getImageLocation("blank.gif") + ")"; │ │ │ │ │ - } │ │ │ │ │ + this.url = url; //deprecated remove me │ │ │ │ │ │ │ │ │ │ - if (this.isFixed) { │ │ │ │ │ - this.map.viewPortDiv.appendChild(this.pane); │ │ │ │ │ - } else { │ │ │ │ │ - this.map.layerContainerDiv.appendChild(this.pane); │ │ │ │ │ - } │ │ │ │ │ + this.layerAlphaHack = this.layer.alpha && OpenLayers.Util.alphaHack(); │ │ │ │ │ │ │ │ │ │ - // once our layer has been added to the map, we can load it │ │ │ │ │ - this.loadMapObject(); │ │ │ │ │ + if (this.maxGetUrlLength != null || this.layer.gutter || this.layerAlphaHack) { │ │ │ │ │ + // only create frame if it's needed │ │ │ │ │ + this.frame = document.createElement("div"); │ │ │ │ │ + this.frame.style.position = "absolute"; │ │ │ │ │ + this.frame.style.overflow = "hidden"; │ │ │ │ │ + } │ │ │ │ │ + if (this.maxGetUrlLength != null) { │ │ │ │ │ + OpenLayers.Util.extend(this, OpenLayers.Tile.Image.IFrame); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // if map didn't load, display warning │ │ │ │ │ - if (this.mapObject == null) { │ │ │ │ │ - this.loadWarningMessage(); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * nullify references to prevent circular references and memory leaks │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.imgDiv) { │ │ │ │ │ + this.clear(); │ │ │ │ │ + this.imgDiv = null; │ │ │ │ │ + this.frame = null; │ │ │ │ │ } │ │ │ │ │ + // don't handle async requests any more │ │ │ │ │ + this.asyncRequestId = null; │ │ │ │ │ + OpenLayers.Tile.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: removeMap │ │ │ │ │ - * On being removed from the map, we'll like to remove the invisible 'pane' │ │ │ │ │ - * div that we added to it on creation. │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Check that a tile should be drawn, and draw it. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Was a tile drawn? Or null if a beforedraw listener returned │ │ │ │ │ + * false. │ │ │ │ │ */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - if (this.pane && this.pane.parentNode) { │ │ │ │ │ - this.pane.parentNode.removeChild(this.pane); │ │ │ │ │ + draw: function() { │ │ │ │ │ + var shouldDraw = OpenLayers.Tile.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (shouldDraw) { │ │ │ │ │ + // The layer's reproject option is deprecated. │ │ │ │ │ + if (this.layer != this.layer.map.baseLayer && this.layer.reproject) { │ │ │ │ │ + // getBoundsFromBaseLayer is defined in deprecated.js. │ │ │ │ │ + this.bounds = this.getBoundsFromBaseLayer(this.position); │ │ │ │ │ + } │ │ │ │ │ + if (this.isLoading) { │ │ │ │ │ + //if we're already loading, send 'reload' instead of 'loadstart'. │ │ │ │ │ + this._loadEvent = "reload"; │ │ │ │ │ + } else { │ │ │ │ │ + this.isLoading = true; │ │ │ │ │ + this._loadEvent = "loadstart"; │ │ │ │ │ + } │ │ │ │ │ + this.renderTile(); │ │ │ │ │ + this.positionTile(); │ │ │ │ │ + } else if (shouldDraw === false) { │ │ │ │ │ + this.unload(); │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.prototype.removeMap.apply(this, arguments); │ │ │ │ │ + return shouldDraw; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: loadWarningMessage │ │ │ │ │ - * If we can't load the map lib, then display an error message to the │ │ │ │ │ - * user and tell them where to go for help. │ │ │ │ │ - * │ │ │ │ │ - * This function sets up the layout for the warning message. Each 3rd │ │ │ │ │ - * party layer must implement its own getWarningHTML() function to │ │ │ │ │ - * provide the actual warning message. │ │ │ │ │ + * Method: renderTile │ │ │ │ │ + * Internal function to actually initialize the image tile, │ │ │ │ │ + * position it correctly, and set its url. │ │ │ │ │ */ │ │ │ │ │ - loadWarningMessage: function() { │ │ │ │ │ - │ │ │ │ │ - this.div.style.backgroundColor = "darkblue"; │ │ │ │ │ - │ │ │ │ │ - var viewSize = this.map.getSize(); │ │ │ │ │ - │ │ │ │ │ - var msgW = Math.min(viewSize.w, 300); │ │ │ │ │ - var msgH = Math.min(viewSize.h, 200); │ │ │ │ │ - var size = new OpenLayers.Size(msgW, msgH); │ │ │ │ │ - │ │ │ │ │ - var centerPx = new OpenLayers.Pixel(viewSize.w / 2, viewSize.h / 2); │ │ │ │ │ - │ │ │ │ │ - var topLeft = centerPx.add(-size.w / 2, -size.h / 2); │ │ │ │ │ - │ │ │ │ │ - var div = OpenLayers.Util.createDiv(this.name + "_warning", │ │ │ │ │ - topLeft, │ │ │ │ │ - size, │ │ │ │ │ - null, │ │ │ │ │ - null, │ │ │ │ │ - null, │ │ │ │ │ - "auto"); │ │ │ │ │ - │ │ │ │ │ - div.style.padding = "7px"; │ │ │ │ │ - div.style.backgroundColor = "yellow"; │ │ │ │ │ + renderTile: function() { │ │ │ │ │ + if (this.layer.async) { │ │ │ │ │ + // Asynchronous image requests call the asynchronous getURL method │ │ │ │ │ + // on the layer to fetch an image that covers 'this.bounds'. │ │ │ │ │ + var id = this.asyncRequestId = (this.asyncRequestId || 0) + 1; │ │ │ │ │ + this.layer.getURLasync(this.bounds, function(url) { │ │ │ │ │ + if (id == this.asyncRequestId) { │ │ │ │ │ + this.url = url; │ │ │ │ │ + this.initImage(); │ │ │ │ │ + } │ │ │ │ │ + }, this); │ │ │ │ │ + } else { │ │ │ │ │ + // synchronous image requests get the url immediately. │ │ │ │ │ + this.url = this.layer.getURL(this.bounds); │ │ │ │ │ + this.initImage(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - div.innerHTML = this.getWarningHTML(); │ │ │ │ │ - this.div.appendChild(div); │ │ │ │ │ + /** │ │ │ │ │ + * Method: positionTile │ │ │ │ │ + * Using the properties currenty set on the layer, position the tile correctly. │ │ │ │ │ + * This method is used both by the async and non-async versions of the Tile.Image │ │ │ │ │ + * code. │ │ │ │ │ + */ │ │ │ │ │ + positionTile: function() { │ │ │ │ │ + var style = this.getTile().style, │ │ │ │ │ + size = this.frame ? this.size : │ │ │ │ │ + this.layer.getImageSize(this.bounds), │ │ │ │ │ + ratio = 1; │ │ │ │ │ + if (this.layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ + ratio = this.layer.getServerResolution() / this.layer.map.getResolution(); │ │ │ │ │ + } │ │ │ │ │ + style.left = this.position.x + "px"; │ │ │ │ │ + style.top = this.position.y + "px"; │ │ │ │ │ + style.width = Math.round(ratio * size.w) + "px"; │ │ │ │ │ + style.height = Math.round(ratio * size.h) + "px"; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getWarningHTML │ │ │ │ │ - * To be implemented by subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} String with information on why layer is broken, how to get │ │ │ │ │ - * it working. │ │ │ │ │ + * Method: clear │ │ │ │ │ + * Remove the tile from the DOM, clear it of any image related data so that │ │ │ │ │ + * it can be reused in a new location. │ │ │ │ │ */ │ │ │ │ │ - getWarningHTML: function() { │ │ │ │ │ - //should be implemented by subclasses │ │ │ │ │ - return ""; │ │ │ │ │ + clear: function() { │ │ │ │ │ + OpenLayers.Tile.prototype.clear.apply(this, arguments); │ │ │ │ │ + var img = this.imgDiv; │ │ │ │ │ + if (img) { │ │ │ │ │ + var tile = this.getTile(); │ │ │ │ │ + if (tile.parentNode === this.layer.div) { │ │ │ │ │ + this.layer.div.removeChild(tile); │ │ │ │ │ + } │ │ │ │ │ + this.setImgSrc(); │ │ │ │ │ + if (this.layerAlphaHack === true) { │ │ │ │ │ + img.style.filter = ""; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.removeClass(img, "olImageLoadError"); │ │ │ │ │ + } │ │ │ │ │ + this.canvasContext = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: display │ │ │ │ │ - * Set the display on the pane │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * display - {Boolean} │ │ │ │ │ + * Method: getImage │ │ │ │ │ + * Returns or creates and returns the tile image. │ │ │ │ │ */ │ │ │ │ │ - display: function(display) { │ │ │ │ │ - OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ - this.pane.style.display = this.div.style.display; │ │ │ │ │ + getImage: function() { │ │ │ │ │ + if (!this.imgDiv) { │ │ │ │ │ + this.imgDiv = OpenLayers.Tile.Image.IMAGE.cloneNode(false); │ │ │ │ │ + │ │ │ │ │ + var style = this.imgDiv.style; │ │ │ │ │ + if (this.frame) { │ │ │ │ │ + var left = 0, │ │ │ │ │ + top = 0; │ │ │ │ │ + if (this.layer.gutter) { │ │ │ │ │ + left = this.layer.gutter / this.layer.tileSize.w * 100; │ │ │ │ │ + top = this.layer.gutter / this.layer.tileSize.h * 100; │ │ │ │ │ + } │ │ │ │ │ + style.left = -left + "%"; │ │ │ │ │ + style.top = -top + "%"; │ │ │ │ │ + style.width = (2 * left + 100) + "%"; │ │ │ │ │ + style.height = (2 * top + 100) + "%"; │ │ │ │ │ + } │ │ │ │ │ + style.visibility = "hidden"; │ │ │ │ │ + style.opacity = 0; │ │ │ │ │ + if (this.layer.opacity < 1) { │ │ │ │ │ + style.filter = 'alpha(opacity=' + │ │ │ │ │ + (this.layer.opacity * 100) + │ │ │ │ │ + ')'; │ │ │ │ │ + } │ │ │ │ │ + style.position = "absolute"; │ │ │ │ │ + if (this.layerAlphaHack) { │ │ │ │ │ + // move the image out of sight │ │ │ │ │ + style.paddingTop = style.height; │ │ │ │ │ + style.height = "0"; │ │ │ │ │ + style.width = "100%"; │ │ │ │ │ + } │ │ │ │ │ + if (this.frame) { │ │ │ │ │ + this.frame.appendChild(this.imgDiv); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return this.imgDiv; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setZIndex │ │ │ │ │ - * Set the z-index order for the pane. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * zIndex - {int} │ │ │ │ │ + * APIMethod: setImage │ │ │ │ │ + * Sets the image element for this tile. This method should only be called │ │ │ │ │ + * from beforeload listeners. │ │ │ │ │ + * │ │ │ │ │ + * Parameters │ │ │ │ │ + * img - {HTMLImageElement} The image to use for this tile. │ │ │ │ │ */ │ │ │ │ │ - setZIndex: function(zIndex) { │ │ │ │ │ - OpenLayers.Layer.prototype.setZIndex.apply(this, arguments); │ │ │ │ │ - this.pane.style.zIndex = parseInt(this.div.style.zIndex) + 1; │ │ │ │ │ + setImage: function(img) { │ │ │ │ │ + this.imgDiv = img; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveByPx │ │ │ │ │ - * Move the layer based on pixel vector. To be implemented by subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * dx - {Number} The x coord of the displacement vector. │ │ │ │ │ - * dy - {Number} The y coord of the displacement vector. │ │ │ │ │ + * Method: initImage │ │ │ │ │ + * Creates the content for the frame on the tile. │ │ │ │ │ */ │ │ │ │ │ - moveByPx: function(dx, dy) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveByPx.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - if (this.dragPanMapObject) { │ │ │ │ │ - this.dragPanMapObject(dx, -dy); │ │ │ │ │ + initImage: function() { │ │ │ │ │ + if (!this.url && !this.imgDiv) { │ │ │ │ │ + // fast path out - if there is no tile url and no previous image │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent('beforeload'); │ │ │ │ │ + this.layer.div.appendChild(this.getTile()); │ │ │ │ │ + this.events.triggerEvent(this._loadEvent); │ │ │ │ │ + var img = this.getImage(); │ │ │ │ │ + var src = img.getAttribute('src') || ''; │ │ │ │ │ + if (this.url && OpenLayers.Util.isEquivalentUrl(src, this.url)) { │ │ │ │ │ + this._loadTimeout = window.setTimeout( │ │ │ │ │ + OpenLayers.Function.bind(this.onImageLoad, this), 0 │ │ │ │ │ + ); │ │ │ │ │ } else { │ │ │ │ │ - this.moveTo(this.map.getCachedCenter()); │ │ │ │ │ + this.stopLoading(); │ │ │ │ │ + if (this.crossOriginKeyword) { │ │ │ │ │ + img.removeAttribute("crossorigin"); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.observe(img, "load", │ │ │ │ │ + OpenLayers.Function.bind(this.onImageLoad, this) │ │ │ │ │ + ); │ │ │ │ │ + OpenLayers.Event.observe(img, "error", │ │ │ │ │ + OpenLayers.Function.bind(this.onImageError, this) │ │ │ │ │ + ); │ │ │ │ │ + this.imageReloadAttempts = 0; │ │ │ │ │ + this.setImgSrc(this.url); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * Handle calls to move the layer. │ │ │ │ │ - * │ │ │ │ │ + * Method: setImgSrc │ │ │ │ │ + * Sets the source for the tile image │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {} │ │ │ │ │ - * zoomChanged - {Boolean} │ │ │ │ │ - * dragging - {Boolean} │ │ │ │ │ + * url - {String} or undefined to hide the image │ │ │ │ │ */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - if (this.mapObject != null) { │ │ │ │ │ - │ │ │ │ │ - var newCenter = this.map.getCenter(); │ │ │ │ │ - var newZoom = this.map.getZoom(); │ │ │ │ │ - │ │ │ │ │ - if (newCenter != null) { │ │ │ │ │ - │ │ │ │ │ - var moOldCenter = this.getMapObjectCenter(); │ │ │ │ │ - var oldCenter = this.getOLLonLatFromMapObjectLonLat(moOldCenter); │ │ │ │ │ - │ │ │ │ │ - var moOldZoom = this.getMapObjectZoom(); │ │ │ │ │ - var oldZoom = this.getOLZoomFromMapObjectZoom(moOldZoom); │ │ │ │ │ - │ │ │ │ │ - if (!(newCenter.equals(oldCenter)) || newZoom != oldZoom) { │ │ │ │ │ - │ │ │ │ │ - if (!zoomChanged && oldCenter && this.dragPanMapObject && │ │ │ │ │ - this.smoothDragPan) { │ │ │ │ │ - var oldPx = this.map.getViewPortPxFromLonLat(oldCenter); │ │ │ │ │ - var newPx = this.map.getViewPortPxFromLonLat(newCenter); │ │ │ │ │ - this.dragPanMapObject(newPx.x - oldPx.x, oldPx.y - newPx.y); │ │ │ │ │ - } else { │ │ │ │ │ - var center = this.getMapObjectLonLatFromOLLonLat(newCenter); │ │ │ │ │ - var zoom = this.getMapObjectZoomFromOLZoom(newZoom); │ │ │ │ │ - this.setMapObjectCenter(center, zoom, dragging); │ │ │ │ │ - } │ │ │ │ │ + setImgSrc: function(url) { │ │ │ │ │ + var img = this.imgDiv; │ │ │ │ │ + if (url) { │ │ │ │ │ + img.style.visibility = 'hidden'; │ │ │ │ │ + img.style.opacity = 0; │ │ │ │ │ + // don't set crossOrigin if the url is a data URL │ │ │ │ │ + if (this.crossOriginKeyword) { │ │ │ │ │ + if (url.substr(0, 5) !== 'data:') { │ │ │ │ │ + img.setAttribute("crossorigin", this.crossOriginKeyword); │ │ │ │ │ + } else { │ │ │ │ │ + img.removeAttribute("crossorigin"); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + img.src = url; │ │ │ │ │ + } else { │ │ │ │ │ + // Remove reference to the image, and leave it to the browser's │ │ │ │ │ + // caching and garbage collection. │ │ │ │ │ + this.stopLoading(); │ │ │ │ │ + this.imgDiv = null; │ │ │ │ │ + if (img.parentNode) { │ │ │ │ │ + img.parentNode.removeChild(img); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ - /********************************************************/ │ │ │ │ │ - /* */ │ │ │ │ │ - /* Baselayer Functions */ │ │ │ │ │ - /* */ │ │ │ │ │ - /********************************************************/ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: getLonLatFromViewPortPx │ │ │ │ │ - * Get a map location from a pixel location │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * viewPortPx - {} │ │ │ │ │ + * Method: getTile │ │ │ │ │ + * Get the tile's markup. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {} An OpenLayers.LonLat which is the passed-in view │ │ │ │ │ - * port OpenLayers.Pixel, translated into lon/lat by map lib │ │ │ │ │ - * If the map lib is not loaded or not centered, returns null │ │ │ │ │ + * {DOMElement} The tile's markup │ │ │ │ │ */ │ │ │ │ │ - getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ - var lonlat = null; │ │ │ │ │ - if ((this.mapObject != null) && │ │ │ │ │ - (this.getMapObjectCenter() != null)) { │ │ │ │ │ - var moPixel = this.getMapObjectPixelFromOLPixel(viewPortPx); │ │ │ │ │ - var moLonLat = this.getMapObjectLonLatFromMapObjectPixel(moPixel); │ │ │ │ │ - lonlat = this.getOLLonLatFromMapObjectLonLat(moLonLat); │ │ │ │ │ - } │ │ │ │ │ - return lonlat; │ │ │ │ │ + getTile: function() { │ │ │ │ │ + return this.frame ? this.frame : this.getImage(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: getViewPortPxFromLonLat │ │ │ │ │ - * Get a pixel location from a map location │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * lonlat - {} │ │ │ │ │ + * Method: createBackBuffer │ │ │ │ │ + * Create a backbuffer for this tile. A backbuffer isn't exactly a clone │ │ │ │ │ + * of the tile's markup, because we want to avoid the reloading of the │ │ │ │ │ + * image. So we clone the frame, and steal the image from the tile. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {} An OpenLayers.Pixel which is the passed-in │ │ │ │ │ - * OpenLayers.LonLat, translated into view port pixels by map lib │ │ │ │ │ - * If map lib is not loaded or not centered, returns null │ │ │ │ │ + * {DOMElement} The markup, or undefined if the tile has no image │ │ │ │ │ + * or if it's currently loading. │ │ │ │ │ */ │ │ │ │ │ - getViewPortPxFromLonLat: function(lonlat) { │ │ │ │ │ - var viewPortPx = null; │ │ │ │ │ - if ((this.mapObject != null) && │ │ │ │ │ - (this.getMapObjectCenter() != null)) { │ │ │ │ │ - │ │ │ │ │ - var moLonLat = this.getMapObjectLonLatFromOLLonLat(lonlat); │ │ │ │ │ - var moPixel = this.getMapObjectPixelFromMapObjectLonLat(moLonLat); │ │ │ │ │ - │ │ │ │ │ - viewPortPx = this.getOLPixelFromMapObjectPixel(moPixel); │ │ │ │ │ + createBackBuffer: function() { │ │ │ │ │ + if (!this.imgDiv || this.isLoading) { │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ - return viewPortPx; │ │ │ │ │ + var backBuffer; │ │ │ │ │ + if (this.frame) { │ │ │ │ │ + backBuffer = this.frame.cloneNode(false); │ │ │ │ │ + backBuffer.appendChild(this.imgDiv); │ │ │ │ │ + } else { │ │ │ │ │ + backBuffer = this.imgDiv; │ │ │ │ │ + } │ │ │ │ │ + this.imgDiv = null; │ │ │ │ │ + return backBuffer; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /********************************************************/ │ │ │ │ │ - /* */ │ │ │ │ │ - /* Translation Functions */ │ │ │ │ │ - /* */ │ │ │ │ │ - /* The following functions translate Map Object and */ │ │ │ │ │ - /* OL formats for Pixel, LonLat */ │ │ │ │ │ - /* */ │ │ │ │ │ - /********************************************************/ │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - // TRANSLATION: MapObject LatLng <-> OpenLayers.LonLat │ │ │ │ │ - // │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: getOLLonLatFromMapObjectLonLat │ │ │ │ │ - * Get an OL style map location from a 3rd party style map location │ │ │ │ │ - * │ │ │ │ │ - * Parameters │ │ │ │ │ - * moLonLat - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} An OpenLayers.LonLat, translated from the passed in │ │ │ │ │ - * MapObject LonLat │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ + * Method: onImageLoad │ │ │ │ │ + * Handler for the image onload event │ │ │ │ │ */ │ │ │ │ │ - getOLLonLatFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - var olLonLat = null; │ │ │ │ │ - if (moLonLat != null) { │ │ │ │ │ - var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - olLonLat = new OpenLayers.LonLat(lon, lat); │ │ │ │ │ + onImageLoad: function() { │ │ │ │ │ + var img = this.imgDiv; │ │ │ │ │ + this.stopLoading(); │ │ │ │ │ + img.style.visibility = 'inherit'; │ │ │ │ │ + img.style.opacity = this.layer.opacity; │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + this.canvasContext = null; │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + │ │ │ │ │ + if (this.layerAlphaHack === true) { │ │ │ │ │ + img.style.filter = │ │ │ │ │ + "progid:DXImageTransform.Microsoft.AlphaImageLoader(src='" + │ │ │ │ │ + img.src + "', sizingMethod='scale')"; │ │ │ │ │ } │ │ │ │ │ - return olLonLat; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getMapObjectLonLatFromOLLonLat │ │ │ │ │ - * Get a 3rd party map location from an OL map location. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * olLonLat - {} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} A MapObject LonLat, translated from the passed in │ │ │ │ │ - * OpenLayers.LonLat │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ + * Method: onImageError │ │ │ │ │ + * Handler for the image onerror event │ │ │ │ │ */ │ │ │ │ │ - getMapObjectLonLatFromOLLonLat: function(olLonLat) { │ │ │ │ │ - var moLatLng = null; │ │ │ │ │ - if (olLonLat != null) { │ │ │ │ │ - moLatLng = this.getMapObjectLonLatFromLonLat(olLonLat.lon, │ │ │ │ │ - olLonLat.lat); │ │ │ │ │ + onImageError: function() { │ │ │ │ │ + var img = this.imgDiv; │ │ │ │ │ + if (img.src != null) { │ │ │ │ │ + this.imageReloadAttempts++; │ │ │ │ │ + if (this.imageReloadAttempts <= OpenLayers.IMAGE_RELOAD_ATTEMPTS) { │ │ │ │ │ + this.setImgSrc(this.layer.getURL(this.bounds)); │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Element.addClass(img, "olImageLoadError"); │ │ │ │ │ + this.events.triggerEvent("loaderror"); │ │ │ │ │ + this.onImageLoad(); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return moLatLng; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - // TRANSLATION: MapObject Pixel <-> OpenLayers.Pixel │ │ │ │ │ - // │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: getOLPixelFromMapObjectPixel │ │ │ │ │ - * Get an OL pixel location from a 3rd party pixel location. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moPixel - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} An OpenLayers.Pixel, translated from the passed in │ │ │ │ │ - * MapObject Pixel │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ + * Method: stopLoading │ │ │ │ │ + * Stops a loading sequence so won't be executed. │ │ │ │ │ */ │ │ │ │ │ - getOLPixelFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - var olPixel = null; │ │ │ │ │ - if (moPixel != null) { │ │ │ │ │ - var x = this.getXFromMapObjectPixel(moPixel); │ │ │ │ │ - var y = this.getYFromMapObjectPixel(moPixel); │ │ │ │ │ - olPixel = new OpenLayers.Pixel(x, y); │ │ │ │ │ - } │ │ │ │ │ - return olPixel; │ │ │ │ │ + stopLoading: function() { │ │ │ │ │ + OpenLayers.Event.stopObservingElement(this.imgDiv); │ │ │ │ │ + window.clearTimeout(this._loadTimeout); │ │ │ │ │ + delete this._loadTimeout; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getMapObjectPixelFromOLPixel │ │ │ │ │ - * Get a 3rd party pixel location from an OL pixel location │ │ │ │ │ + * APIMethod: getCanvasContext │ │ │ │ │ + * Returns a canvas context associated with the tile image (with │ │ │ │ │ + * the image drawn on it). │ │ │ │ │ + * Returns undefined if the browser does not support canvas, if │ │ │ │ │ + * the tile has no image or if it's currently loading. │ │ │ │ │ + * │ │ │ │ │ + * The function returns a canvas context instance but the │ │ │ │ │ + * underlying canvas is still available in the 'canvas' property: │ │ │ │ │ + * (code) │ │ │ │ │ + * var context = tile.getCanvasContext(); │ │ │ │ │ + * if (context) { │ │ │ │ │ + * var data = context.canvas.toDataURL('image/jpeg'); │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * olPixel - {} │ │ │ │ │ - * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} A MapObject Pixel, translated from the passed in │ │ │ │ │ - * OpenLayers.Pixel │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - getMapObjectPixelFromOLPixel: function(olPixel) { │ │ │ │ │ - var moPixel = null; │ │ │ │ │ - if (olPixel != null) { │ │ │ │ │ - moPixel = this.getMapObjectPixelFromXY(olPixel.x, olPixel.y); │ │ │ │ │ + getCanvasContext: function() { │ │ │ │ │ + if (OpenLayers.CANVAS_SUPPORTED && this.imgDiv && !this.isLoading) { │ │ │ │ │ + if (!this.canvasContext) { │ │ │ │ │ + var canvas = document.createElement("canvas"); │ │ │ │ │ + canvas.width = this.size.w; │ │ │ │ │ + canvas.height = this.size.h; │ │ │ │ │ + this.canvasContext = canvas.getContext("2d"); │ │ │ │ │ + this.canvasContext.drawImage(this.imgDiv, 0, 0); │ │ │ │ │ + } │ │ │ │ │ + return this.canvasContext; │ │ │ │ │ } │ │ │ │ │ - return moPixel; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.EventPane" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Tile.Image" │ │ │ │ │ + │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Tile.Image.IMAGE │ │ │ │ │ + * {HTMLImageElement} The image for a tile. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Tile.Image.IMAGE = (function() { │ │ │ │ │ + var img = new Image(); │ │ │ │ │ + img.className = "olTileImage"; │ │ │ │ │ + // avoid image gallery menu in IE6 │ │ │ │ │ + img.galleryImg = "no"; │ │ │ │ │ + return img; │ │ │ │ │ +}()); │ │ │ │ │ + │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/WorldWind.js │ │ │ │ │ + OpenLayers/Layer/Grid.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/Layer/HTTPRequest.js │ │ │ │ │ + * @requires OpenLayers/Tile/Image.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.WorldWind │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Layer.Grid │ │ │ │ │ + * Base class for layers that use a lattice of tiles. Create a new grid │ │ │ │ │ + * layer with the constructor. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ + * - │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.WorldWind = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ +OpenLayers.Layer.Grid = OpenLayers.Class(OpenLayers.Layer.HTTPRequest, { │ │ │ │ │ │ │ │ │ │ - DEFAULT_PARAMS: {}, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: tileSize │ │ │ │ │ + * {} │ │ │ │ │ + */ │ │ │ │ │ + tileSize: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} WorldWind layer is a base layer by default. │ │ │ │ │ + * Property: tileOriginCorner │ │ │ │ │ + * {String} If the property is not provided, the tile origin │ │ │ │ │ + * will be derived from the layer's . The corner of the │ │ │ │ │ + * used is determined by this property. Acceptable values │ │ │ │ │ + * are "tl" (top left), "tr" (top right), "bl" (bottom left), and "br" │ │ │ │ │ + * (bottom right). Default is "bl". │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + tileOriginCorner: "bl", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: lzd │ │ │ │ │ - * {Float} LevelZeroTileSizeDegrees │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: tileOrigin │ │ │ │ │ + * {} Optional origin for aligning the grid of tiles. │ │ │ │ │ + * If provided, requests for tiles at all resolutions will be aligned │ │ │ │ │ + * with this location (no tiles shall overlap this location). If │ │ │ │ │ + * not provided, the grid of tiles will be aligned with the layer's │ │ │ │ │ + * . Default is ``null``. │ │ │ │ │ */ │ │ │ │ │ - lzd: null, │ │ │ │ │ + tileOrigin: null, │ │ │ │ │ + │ │ │ │ │ + /** APIProperty: tileOptions │ │ │ │ │ + * {Object} optional configuration options for instances │ │ │ │ │ + * created by this Layer, if supported by the tile class. │ │ │ │ │ + */ │ │ │ │ │ + tileOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomLevels │ │ │ │ │ - * {Integer} Number of zoom levels. │ │ │ │ │ + * APIProperty: tileClass │ │ │ │ │ + * {} The tile class to use for this layer. │ │ │ │ │ + * Defaults is OpenLayers.Tile.Image. │ │ │ │ │ */ │ │ │ │ │ - zoomLevels: null, │ │ │ │ │ + tileClass: OpenLayers.Tile.Image, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.WorldWind │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} Name of Layer │ │ │ │ │ - * url - {String} Base URL │ │ │ │ │ - * lzd - {Float} Level zero tile size degrees │ │ │ │ │ - * zoomLevels - {Integer} number of zoom levels │ │ │ │ │ - * params - {Object} additional parameters │ │ │ │ │ - * options - {Object} additional options │ │ │ │ │ + * Property: grid │ │ │ │ │ + * {Array(Array())} This is an array of rows, each row is │ │ │ │ │ + * an array of tiles. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, lzd, zoomLevels, params, options) { │ │ │ │ │ - this.lzd = lzd; │ │ │ │ │ - this.zoomLevels = zoomLevels; │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - newArguments.push(name, url, params, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - this.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, this.DEFAULT_PARAMS │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ + grid: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getZoom │ │ │ │ │ - * Convert map zoom to WW zoom. │ │ │ │ │ + * APIProperty: singleTile │ │ │ │ │ + * {Boolean} Moves the layer into single-tile mode, meaning that one tile │ │ │ │ │ + * will be loaded. The tile's size will be determined by the 'ratio' │ │ │ │ │ + * property. When the tile is dragged such that it does not cover the │ │ │ │ │ + * entire viewport, it is reloaded. │ │ │ │ │ */ │ │ │ │ │ - getZoom: function() { │ │ │ │ │ - var zoom = this.map.getZoom(); │ │ │ │ │ - var extent = this.map.getMaxExtent(); │ │ │ │ │ - zoom = zoom - Math.log(this.maxResolution / (this.lzd / 512)) / Math.log(2); │ │ │ │ │ - return zoom; │ │ │ │ │ - }, │ │ │ │ │ + singleTile: false, │ │ │ │ │ + │ │ │ │ │ + /** APIProperty: ratio │ │ │ │ │ + * {Float} Used only when in single-tile mode, this specifies the │ │ │ │ │ + * ratio of the size of the single tile to the size of the map. │ │ │ │ │ + * Default value is 1.5. │ │ │ │ │ + */ │ │ │ │ │ + ratio: 1.5, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ + * APIProperty: buffer │ │ │ │ │ + * {Integer} Used only when in gridded mode, this specifies the number of │ │ │ │ │ + * extra rows and colums of tiles on each side which will │ │ │ │ │ + * surround the minimum grid tiles to cover the map. │ │ │ │ │ + * For very slow loading layers, a larger value may increase │ │ │ │ │ + * performance somewhat when dragging, but will increase bandwidth │ │ │ │ │ + * use significantly. │ │ │ │ │ + */ │ │ │ │ │ + buffer: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: transitionEffect │ │ │ │ │ + * {String} The transition effect to use when the map is zoomed. │ │ │ │ │ + * Two posible values: │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {} │ │ │ │ │ + * "resize" - Existing tiles are resized on zoom to provide a visual │ │ │ │ │ + * effect of the zoom having taken place immediately. As the │ │ │ │ │ + * new tiles become available, they are drawn on top of the │ │ │ │ │ + * resized tiles (this is the default setting). │ │ │ │ │ + * "map-resize" - Existing tiles are resized on zoom and placed below the │ │ │ │ │ + * base layer. New tiles for the base layer will cover existing tiles. │ │ │ │ │ + * This setting is recommended when having an overlay duplicated during │ │ │ │ │ + * the transition is undesirable (e.g. street labels or big transparent │ │ │ │ │ + * fills). │ │ │ │ │ + * null - No transition effect. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as │ │ │ │ │ - * parameters │ │ │ │ │ + * Using "resize" on non-opaque layers can cause undesired visual │ │ │ │ │ + * effects. Set transitionEffect to null in this case. │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var zoom = this.getZoom(); │ │ │ │ │ - var extent = this.map.getMaxExtent(); │ │ │ │ │ - var deg = this.lzd / Math.pow(2, this.getZoom()); │ │ │ │ │ - var x = Math.floor((bounds.left - extent.left) / deg); │ │ │ │ │ - var y = Math.floor((bounds.bottom - extent.bottom) / deg); │ │ │ │ │ - if (this.map.getResolution() <= (this.lzd / 512) && │ │ │ │ │ - this.getZoom() <= this.zoomLevels) { │ │ │ │ │ - return this.getFullRequestString({ │ │ │ │ │ - L: zoom, │ │ │ │ │ - X: x, │ │ │ │ │ - Y: y │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - return OpenLayers.Util.getImageLocation("blank.gif"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ + transitionEffect: "resize", │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.WorldWind" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/TMS.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: numLoadingTiles │ │ │ │ │ + * {Integer} How many tiles are still loading? │ │ │ │ │ + */ │ │ │ │ │ + numLoadingTiles: 0, │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: serverResolutions │ │ │ │ │ + * {Array(Number}} This property is documented in subclasses as │ │ │ │ │ + * an API property. │ │ │ │ │ + */ │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: loading │ │ │ │ │ + * {Boolean} Indicates if tiles are being loaded. │ │ │ │ │ + */ │ │ │ │ │ + loading: false, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: backBuffer │ │ │ │ │ + * {DOMElement} The back buffer. │ │ │ │ │ + */ │ │ │ │ │ + backBuffer: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.TMS │ │ │ │ │ - * Create a layer for accessing tiles from services that conform with the │ │ │ │ │ - * Tile Map Service Specification │ │ │ │ │ - * (http://wiki.osgeo.org/wiki/Tile_Map_Service_Specification). │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var layer = new OpenLayers.Layer.TMS( │ │ │ │ │ - * "My Layer", // name for display in LayerSwitcher │ │ │ │ │ - * "http://tilecache.osgeo.org/wms-c/Basic.py/", // service endpoint │ │ │ │ │ - * {layername: "basic", type: "png"} // required properties │ │ │ │ │ - * ); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.TMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + /** │ │ │ │ │ + * Property: gridResolution │ │ │ │ │ + * {Number} The resolution of the current grid. Used for backbuffer and │ │ │ │ │ + * client zoom. This property is updated every time the grid is │ │ │ │ │ + * initialized. │ │ │ │ │ + */ │ │ │ │ │ + gridResolution: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: serviceVersion │ │ │ │ │ - * {String} Service version for tile requests. Default is "1.0.0". │ │ │ │ │ + * Property: backBufferResolution │ │ │ │ │ + * {Number} The resolution of the current back buffer. This property is │ │ │ │ │ + * updated each time a back buffer is created. │ │ │ │ │ */ │ │ │ │ │ - serviceVersion: "1.0.0", │ │ │ │ │ + backBufferResolution: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: layername │ │ │ │ │ - * {String} The identifier for the as advertised by the service. │ │ │ │ │ - * For example, if the service advertises a with │ │ │ │ │ - * 'href="http://tms.osgeo.org/1.0.0/vmap0"', the property │ │ │ │ │ - * would be set to "vmap0". │ │ │ │ │ + * Property: backBufferLonLat │ │ │ │ │ + * {Object} The top-left corner of the current back buffer. Includes lon │ │ │ │ │ + * and lat properties. This object is updated each time a back buffer │ │ │ │ │ + * is created. │ │ │ │ │ */ │ │ │ │ │ - layername: null, │ │ │ │ │ + backBufferLonLat: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: type │ │ │ │ │ - * {String} The format extension corresponding to the requested tile image │ │ │ │ │ - * type. This is advertised in a element as the │ │ │ │ │ - * "extension" attribute. For example, if the service advertises a │ │ │ │ │ - * with , │ │ │ │ │ - * the property would be set to "jpg". │ │ │ │ │ + * Property: backBufferTimerId │ │ │ │ │ + * {Number} The id of the back buffer timer. This timer is used to │ │ │ │ │ + * delay the removal of the back buffer, thereby preventing │ │ │ │ │ + * flash effects caused by tile animation. │ │ │ │ │ */ │ │ │ │ │ - type: null, │ │ │ │ │ + backBufferTimerId: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} Make this layer a base layer. Default is true. Set false to │ │ │ │ │ - * use the layer as an overlay. │ │ │ │ │ + * APIProperty: removeBackBufferDelay │ │ │ │ │ + * {Number} Delay for removing the backbuffer when all tiles have finished │ │ │ │ │ + * loading. Can be set to 0 when no css opacity transitions for the │ │ │ │ │ + * olTileImage class are used. Default is 0 for layers, │ │ │ │ │ + * 2500 for tiled layers. See for more information on │ │ │ │ │ + * tile animation. │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + removeBackBufferDelay: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: tileOrigin │ │ │ │ │ - * {} Optional origin for aligning the grid of tiles. │ │ │ │ │ - * If provided, requests for tiles at all resolutions will be aligned │ │ │ │ │ - * with this location (no tiles shall overlap this location). If │ │ │ │ │ - * not provided, the grid of tiles will be aligned with the bottom-left │ │ │ │ │ - * corner of the map's . Default is ``null``. │ │ │ │ │ + * APIProperty: className │ │ │ │ │ + * {String} Name of the class added to the layer div. If not set in the │ │ │ │ │ + * options passed to the constructor then className defaults to │ │ │ │ │ + * "olLayerGridSingleTile" for single tile layers (see ), │ │ │ │ │ + * and "olLayerGrid" for non single tile layers. │ │ │ │ │ * │ │ │ │ │ - * Example: │ │ │ │ │ + * Note: │ │ │ │ │ + * │ │ │ │ │ + * The displaying of tiles is not animated by default for single tile │ │ │ │ │ + * layers - OpenLayers' default theme (style.css) includes this: │ │ │ │ │ * (code) │ │ │ │ │ - * var layer = new OpenLayers.Layer.TMS( │ │ │ │ │ - * "My Layer", │ │ │ │ │ - * "http://tilecache.osgeo.org/wms-c/Basic.py/", │ │ │ │ │ - * { │ │ │ │ │ - * layername: "basic", │ │ │ │ │ - * type: "png", │ │ │ │ │ - * // set if different than the bottom left of map.maxExtent │ │ │ │ │ - * tileOrigin: new OpenLayers.LonLat(-180, -90) │ │ │ │ │ - * } │ │ │ │ │ - * ); │ │ │ │ │ + * .olLayerGrid .olTileImage { │ │ │ │ │ + * -webkit-transition: opacity 0.2s linear; │ │ │ │ │ + * -moz-transition: opacity 0.2s linear; │ │ │ │ │ + * -o-transition: opacity 0.2s linear; │ │ │ │ │ + * transition: opacity 0.2s linear; │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ + * To animate tile displaying for any grid layer the following │ │ │ │ │ + * CSS rule can be used: │ │ │ │ │ + * (code) │ │ │ │ │ + * .olTileImage { │ │ │ │ │ + * -webkit-transition: opacity 0.2s linear; │ │ │ │ │ + * -moz-transition: opacity 0.2s linear; │ │ │ │ │ + * -o-transition: opacity 0.2s linear; │ │ │ │ │ + * transition: opacity 0.2s linear; │ │ │ │ │ + * } │ │ │ │ │ * (end) │ │ │ │ │ + * In that case, to avoid flash effects, │ │ │ │ │ + * should not be zero. │ │ │ │ │ */ │ │ │ │ │ - tileOrigin: null, │ │ │ │ │ + className: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: serverResolutions │ │ │ │ │ - * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ - * property if the map resolutions differ from the server. This │ │ │ │ │ - * property serves two purposes. (a) can include │ │ │ │ │ - * resolutions that the server supports and that you don't want to │ │ │ │ │ - * provide with this layer; you can also look at , which is │ │ │ │ │ - * an alternative to for that specific purpose. │ │ │ │ │ - * (b) The map can work with resolutions that aren't supported by │ │ │ │ │ - * the server, i.e. that aren't in . When the │ │ │ │ │ - * map is displayed in such a resolution data for the closest │ │ │ │ │ - * server-supported resolution is loaded and the layer div is │ │ │ │ │ - * stretched as necessary. │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * layer.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Listeners will be called with a reference to an event object. The │ │ │ │ │ + * properties of this event depends on exactly what happened. │ │ │ │ │ + * │ │ │ │ │ + * All event objects have at least the following properties: │ │ │ │ │ + * object - {Object} A reference to layer.events.object. │ │ │ │ │ + * element - {DOMElement} A reference to layer.events.element. │ │ │ │ │ + * │ │ │ │ │ + * Supported event types: │ │ │ │ │ + * addtile - Triggered when a tile is added to this layer. Listeners receive │ │ │ │ │ + * an object as first argument, which has a tile property that │ │ │ │ │ + * references the tile that has been added. │ │ │ │ │ + * tileloadstart - Triggered when a tile starts loading. Listeners receive │ │ │ │ │ + * an object as first argument, which has a tile property that │ │ │ │ │ + * references the tile that starts loading. │ │ │ │ │ + * tileloaded - Triggered when each new tile is │ │ │ │ │ + * loaded, as a means of progress update to listeners. │ │ │ │ │ + * listeners can access 'numLoadingTiles' if they wish to keep │ │ │ │ │ + * track of the loading progress. Listeners are called with an object │ │ │ │ │ + * with a 'tile' property as first argument, making the loaded tile │ │ │ │ │ + * available to the listener, and an 'aborted' property, which will be │ │ │ │ │ + * true when loading was aborted and no tile data is available. │ │ │ │ │ + * tileerror - Triggered before the tileloaded event (i.e. when the tile is │ │ │ │ │ + * still hidden) if a tile failed to load. Listeners receive an object │ │ │ │ │ + * as first argument, which has a tile property that references the │ │ │ │ │ + * tile that could not be loaded. │ │ │ │ │ + * retile - Triggered when the layer recreates its tile grid. │ │ │ │ │ */ │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomOffset │ │ │ │ │ - * {Number} If your cache has more zoom levels than you want to provide │ │ │ │ │ - * access to with this layer, supply a zoomOffset. This zoom offset │ │ │ │ │ - * is added to the current map zoom level to determine the level │ │ │ │ │ - * for a requested tile. For example, if you supply a zoomOffset │ │ │ │ │ - * of 3, when the map is at the zoom 0, tiles will be requested from │ │ │ │ │ - * level 3 of your cache. Default is 0 (assumes cache level and map │ │ │ │ │ - * zoom are equivalent). Using is an alternative to │ │ │ │ │ - * setting if you only want to expose a subset │ │ │ │ │ - * of the server resolutions. │ │ │ │ │ + * Property: gridLayout │ │ │ │ │ + * {Object} Object containing properties tilelon, tilelat, startcol, │ │ │ │ │ + * startrow │ │ │ │ │ */ │ │ │ │ │ - zoomOffset: 0, │ │ │ │ │ + gridLayout: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.TMS │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} Title to be displayed in a │ │ │ │ │ - * url - {String} Service endpoint (without the version number). E.g. │ │ │ │ │ - * "http://tms.osgeo.org/". │ │ │ │ │ - * options - {Object} Additional properties to be set on the layer. The │ │ │ │ │ - * and properties must be set here. │ │ │ │ │ + * Property: rowSign │ │ │ │ │ + * {Number} 1 for grids starting at the top, -1 for grids starting at the │ │ │ │ │ + * bottom. This is used for several grid index and offset calculations. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - newArguments.push(name, url, {}, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - }, │ │ │ │ │ + rowSign: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Create a complete copy of this layer. │ │ │ │ │ + * Property: transitionendEvents │ │ │ │ │ + * {Array} Event names for transitionend │ │ │ │ │ + */ │ │ │ │ │ + transitionendEvents: [ │ │ │ │ │ + 'transitionend', 'webkitTransitionEnd', 'otransitionend', │ │ │ │ │ + 'oTransitionEnd' │ │ │ │ │ + ], │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.Grid │ │ │ │ │ + * Create a new grid layer │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {Object} Should only be provided by subclasses that call this │ │ │ │ │ - * method. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} An exact clone of this │ │ │ │ │ + * name - {String} │ │ │ │ │ + * url - {String} │ │ │ │ │ + * params - {Object} │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.TMS(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.initialize.apply(this, │ │ │ │ │ + arguments); │ │ │ │ │ + this.grid = []; │ │ │ │ │ + this._removeBackBuffer = OpenLayers.Function.bind(this.removeBackBuffer, this); │ │ │ │ │ │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ + this.initProperties(); │ │ │ │ │ │ │ │ │ │ - return obj; │ │ │ │ │ + this.rowSign = this.tileOriginCorner.substr(0, 1) === "t" ? 1 : -1; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as │ │ │ │ │ - * parameters │ │ │ │ │ + * Method: initProperties │ │ │ │ │ + * Set any properties that depend on the value of singleTile. │ │ │ │ │ + * Currently sets removeBackBufferDelay and className │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((bounds.bottom - this.tileOrigin.lat) / (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getServerZoom(); │ │ │ │ │ - var path = this.serviceVersion + "/" + this.layername + "/" + z + "/" + x + "/" + y + "." + this.type; │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(path, url); │ │ │ │ │ + initProperties: function() { │ │ │ │ │ + if (this.options.removeBackBufferDelay === undefined) { │ │ │ │ │ + this.removeBackBufferDelay = this.singleTile ? 0 : 2500; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.options.className === undefined) { │ │ │ │ │ + this.className = this.singleTile ? 'olLayerGridSingleTile' : │ │ │ │ │ + 'olLayerGrid'; │ │ │ │ │ } │ │ │ │ │ - return url + path; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ + /** │ │ │ │ │ * Method: setMap │ │ │ │ │ - * When the layer is added to a map, then we can fetch our origin │ │ │ │ │ - * (if we don't have one.) │ │ │ │ │ - * │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {} │ │ │ │ │ + * map - {} The map. │ │ │ │ │ */ │ │ │ │ │ setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (!this.tileOrigin) { │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.map.maxExtent.left, │ │ │ │ │ - this.map.maxExtent.bottom); │ │ │ │ │ - } │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.setMap.call(this, map); │ │ │ │ │ + OpenLayers.Element.addClass(this.div, this.className); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.TMS" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/XYZ.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.XYZ │ │ │ │ │ - * The XYZ class is designed to make it easier for people who have tiles │ │ │ │ │ - * arranged by a standard XYZ grid. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.XYZ = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * Default is true, as this is designed to be a base tile source. │ │ │ │ │ + * Method: removeMap │ │ │ │ │ + * Called when the layer is removed from the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {} The map. │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + this.removeBackBuffer(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: sphericalMercator │ │ │ │ │ - * Whether the tile extents should be set to the defaults for │ │ │ │ │ - * spherical mercator. Useful for things like OpenStreetMap. │ │ │ │ │ - * Default is false, except for the OSM subclass. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Deconstruct the layer and clear the grid. │ │ │ │ │ */ │ │ │ │ │ - sphericalMercator: false, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.removeBackBuffer(); │ │ │ │ │ + this.clearGrid(); │ │ │ │ │ + │ │ │ │ │ + this.grid = null; │ │ │ │ │ + this.tileSize = null; │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomOffset │ │ │ │ │ - * {Number} If your cache has more zoom levels than you want to provide │ │ │ │ │ - * access to with this layer, supply a zoomOffset. This zoom offset │ │ │ │ │ - * is added to the current map zoom level to determine the level │ │ │ │ │ - * for a requested tile. For example, if you supply a zoomOffset │ │ │ │ │ - * of 3, when the map is at the zoom 0, tiles will be requested from │ │ │ │ │ - * level 3 of your cache. Default is 0 (assumes cache level and map │ │ │ │ │ - * zoom are equivalent). Using is an alternative to │ │ │ │ │ - * setting if you only want to expose a subset │ │ │ │ │ - * of the server resolutions. │ │ │ │ │ + * APIMethod: mergeNewParams │ │ │ │ │ + * Refetches tiles with new params merged, keeping a backbuffer. Each │ │ │ │ │ + * loading new tile will have a css class of '.olTileReplacing'. If a │ │ │ │ │ + * stylesheet applies a 'display: none' style to that class, any fade-in │ │ │ │ │ + * transition will not apply, and backbuffers for each tile will be removed │ │ │ │ │ + * as soon as the tile is loaded. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newParams - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * redrawn: {Boolean} whether the layer was actually redrawn. │ │ │ │ │ */ │ │ │ │ │ - zoomOffset: 0, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: serverResolutions │ │ │ │ │ - * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ - * property if the map resolutions differ from the server. This │ │ │ │ │ - * property serves two purposes. (a) can include │ │ │ │ │ - * resolutions that the server supports and that you don't want to │ │ │ │ │ - * provide with this layer; you can also look at , which is │ │ │ │ │ - * an alternative to for that specific purpose. │ │ │ │ │ - * (b) The map can work with resolutions that aren't supported by │ │ │ │ │ - * the server, i.e. that aren't in . When the │ │ │ │ │ - * map is displayed in such a resolution data for the closest │ │ │ │ │ - * server-supported resolution is loaded and the layer div is │ │ │ │ │ - * stretched as necessary. │ │ │ │ │ + * Method: clearGrid │ │ │ │ │ + * Go through and remove all tiles from the grid, calling │ │ │ │ │ + * destroy() on each of them to kill circular references │ │ │ │ │ */ │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ + clearGrid: function() { │ │ │ │ │ + if (this.grid) { │ │ │ │ │ + for (var iRow = 0, len = this.grid.length; iRow < len; iRow++) { │ │ │ │ │ + var row = this.grid[iRow]; │ │ │ │ │ + for (var iCol = 0, clen = row.length; iCol < clen; iCol++) { │ │ │ │ │ + var tile = row[iCol]; │ │ │ │ │ + this.destroyTile(tile); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.grid = []; │ │ │ │ │ + this.gridResolution = null; │ │ │ │ │ + this.gridLayout = null; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.XYZ │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: addOptions │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * url - {String} │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + * newOptions - {Object} │ │ │ │ │ + * reinitialize - {Boolean} If set to true, and if resolution options of the │ │ │ │ │ + * current baseLayer were changed, the map will be recentered to make │ │ │ │ │ + * sure that it is displayed with a valid resolution, and a │ │ │ │ │ + * changebaselayer event will be triggered. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - if (options && options.sphericalMercator || this.sphericalMercator) { │ │ │ │ │ - options = OpenLayers.Util.extend({ │ │ │ │ │ - projection: "EPSG:900913", │ │ │ │ │ - numZoomLevels: 19 │ │ │ │ │ - }, options); │ │ │ │ │ + addOptions: function(newOptions, reinitialize) { │ │ │ │ │ + var singleTileChanged = newOptions.singleTile !== undefined && │ │ │ │ │ + newOptions.singleTile !== this.singleTile; │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.addOptions.apply(this, arguments); │ │ │ │ │ + if (this.map && singleTileChanged) { │ │ │ │ │ + this.initProperties(); │ │ │ │ │ + this.clearGrid(); │ │ │ │ │ + this.tileSize = this.options.tileSize; │ │ │ │ │ + this.setTileSize(); │ │ │ │ │ + this.moveTo(null, true); │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [ │ │ │ │ │ - name || this.name, url || this.url, {}, │ │ │ │ │ - options │ │ │ │ │ - ]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIMethod: clone │ │ │ │ │ * Create a clone of this layer │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * obj - {Object} Is this ever used? │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {} An exact clone of this OpenLayers.Layer.XYZ │ │ │ │ │ + * {} An exact clone of this OpenLayers.Layer.Grid │ │ │ │ │ */ │ │ │ │ │ clone: function(obj) { │ │ │ │ │ │ │ │ │ │ if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.XYZ(this.name, │ │ │ │ │ + obj = new OpenLayers.Layer.Grid(this.name, │ │ │ │ │ this.url, │ │ │ │ │ + this.params, │ │ │ │ │ this.getOptions()); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + obj = OpenLayers.Layer.HTTPRequest.prototype.clone.apply(this, [obj]); │ │ │ │ │ + │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + if (this.tileSize != null) { │ │ │ │ │ + obj.tileSize = this.tileSize.clone(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // we do not want to copy reference to grid, so we make a new array │ │ │ │ │ + obj.grid = []; │ │ │ │ │ + obj.gridResolution = null; │ │ │ │ │ + // same for backbuffer │ │ │ │ │ + obj.backBuffer = null; │ │ │ │ │ + obj.backBufferTimerId = null; │ │ │ │ │ + obj.loading = false; │ │ │ │ │ + obj.numLoadingTiles = 0; │ │ │ │ │ │ │ │ │ │ return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * This function is called whenever the map is moved. All the moving │ │ │ │ │ + * of actual 'tiles' is done by the map, but moveTo's role is to accept │ │ │ │ │ + * a bounds and make sure the data that that bounds requires is pre-loaded. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * bounds - {} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as │ │ │ │ │ - * parameters │ │ │ │ │ + * zoomChanged - {Boolean} │ │ │ │ │ + * dragging - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var xyz = this.getXYZ(bounds); │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - var s = '' + xyz.x + xyz.y + xyz.z; │ │ │ │ │ - url = this.selectUrl(s, url); │ │ │ │ │ - } │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ │ │ │ │ │ - return OpenLayers.String.format(url, xyz); │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + bounds = bounds || this.map.getExtent(); │ │ │ │ │ + │ │ │ │ │ + if (bounds != null) { │ │ │ │ │ + │ │ │ │ │ + // if grid is empty or zoom has changed, we *must* re-tile │ │ │ │ │ + var forceReTile = !this.grid.length || zoomChanged; │ │ │ │ │ + │ │ │ │ │ + // total bounds of the tiles │ │ │ │ │ + var tilesBounds = this.getTilesBounds(); │ │ │ │ │ + │ │ │ │ │ + // the new map resolution │ │ │ │ │ + var resolution = this.map.getResolution(); │ │ │ │ │ + │ │ │ │ │ + // the server-supported resolution for the new map resolution │ │ │ │ │ + var serverResolution = this.getServerResolution(resolution); │ │ │ │ │ + │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + │ │ │ │ │ + // We want to redraw whenever even the slightest part of the │ │ │ │ │ + // current bounds is not contained by our tile. │ │ │ │ │ + // (thus, we do not specify partial -- its default is false) │ │ │ │ │ + │ │ │ │ │ + if (forceReTile || │ │ │ │ │ + (!dragging && !tilesBounds.containsBounds(bounds))) { │ │ │ │ │ + │ │ │ │ │ + // In single tile mode with no transition effect, we insert │ │ │ │ │ + // a non-scaled backbuffer when the layer is moved. But if │ │ │ │ │ + // a zoom occurs right after a move, i.e. before the new │ │ │ │ │ + // image is received, we need to remove the backbuffer, or │ │ │ │ │ + // an ill-positioned image will be visible during the zoom │ │ │ │ │ + // transition. │ │ │ │ │ + │ │ │ │ │ + if (zoomChanged && this.transitionEffect !== 'resize') { │ │ │ │ │ + this.removeBackBuffer(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (!zoomChanged || this.transitionEffect === 'resize') { │ │ │ │ │ + this.applyBackBuffer(resolution); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.initSingleTile(bounds); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + │ │ │ │ │ + // if the bounds have changed such that they are not even │ │ │ │ │ + // *partially* contained by our tiles (e.g. when user has │ │ │ │ │ + // programmatically panned to the other side of the earth on │ │ │ │ │ + // zoom level 18), then moveGriddedTiles could potentially have │ │ │ │ │ + // to run through thousands of cycles, so we want to reTile │ │ │ │ │ + // instead (thus, partial true). │ │ │ │ │ + forceReTile = forceReTile || │ │ │ │ │ + !tilesBounds.intersectsBounds(bounds, { │ │ │ │ │ + worldBounds: this.map.baseLayer.wrapDateLine && │ │ │ │ │ + this.map.getMaxExtent() │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + if (forceReTile) { │ │ │ │ │ + if (zoomChanged && (this.transitionEffect === 'resize' || │ │ │ │ │ + this.gridResolution === resolution)) { │ │ │ │ │ + this.applyBackBuffer(resolution); │ │ │ │ │ + } │ │ │ │ │ + this.initGriddedTiles(bounds); │ │ │ │ │ + } else { │ │ │ │ │ + this.moveGriddedTiles(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getXYZ │ │ │ │ │ - * Calculates x, y and z for the given bounds. │ │ │ │ │ + * Method: getTileData │ │ │ │ │ + * Given a map location, retrieve a tile and the pixel offset within that │ │ │ │ │ + * tile corresponding to the location. If there is not an existing │ │ │ │ │ + * tile in the grid that covers the given location, null will be │ │ │ │ │ + * returned. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {} │ │ │ │ │ + * loc - {} map location │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} - an object with x, y and z properties. │ │ │ │ │ + * {Object} Object with the following properties: tile ({}), │ │ │ │ │ + * i ({Number} x-pixel offset from top left), and j ({Integer} y-pixel │ │ │ │ │ + * offset from top left). │ │ │ │ │ */ │ │ │ │ │ - getXYZ: function(bounds) { │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.maxExtent.left) / │ │ │ │ │ - (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((this.maxExtent.top - bounds.top) / │ │ │ │ │ - (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getServerZoom(); │ │ │ │ │ + getTileData: function(loc) { │ │ │ │ │ + var data = null, │ │ │ │ │ + x = loc.lon, │ │ │ │ │ + y = loc.lat, │ │ │ │ │ + numRows = this.grid.length; │ │ │ │ │ │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - var limit = Math.pow(2, z); │ │ │ │ │ - x = ((x % limit) + limit) % limit; │ │ │ │ │ + if (this.map && numRows) { │ │ │ │ │ + var res = this.map.getResolution(), │ │ │ │ │ + tileWidth = this.tileSize.w, │ │ │ │ │ + tileHeight = this.tileSize.h, │ │ │ │ │ + bounds = this.grid[0][0].bounds, │ │ │ │ │ + left = bounds.left, │ │ │ │ │ + top = bounds.top; │ │ │ │ │ + │ │ │ │ │ + if (x < left) { │ │ │ │ │ + // deal with multiple worlds │ │ │ │ │ + if (this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + var worldWidth = this.map.getMaxExtent().getWidth(); │ │ │ │ │ + var worldsAway = Math.ceil((left - x) / worldWidth); │ │ │ │ │ + x += worldWidth * worldsAway; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // tile distance to location (fractional number of tiles); │ │ │ │ │ + var dtx = (x - left) / (res * tileWidth); │ │ │ │ │ + var dty = (top - y) / (res * tileHeight); │ │ │ │ │ + // index of tile in grid │ │ │ │ │ + var col = Math.floor(dtx); │ │ │ │ │ + var row = Math.floor(dty); │ │ │ │ │ + if (row >= 0 && row < numRows) { │ │ │ │ │ + var tile = this.grid[row][col]; │ │ │ │ │ + if (tile) { │ │ │ │ │ + data = { │ │ │ │ │ + tile: tile, │ │ │ │ │ + // pixel index within tile │ │ │ │ │ + i: Math.floor((dtx - col) * tileWidth), │ │ │ │ │ + j: Math.floor((dty - row) * tileHeight) │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return data; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return { │ │ │ │ │ - 'x': x, │ │ │ │ │ - 'y': y, │ │ │ │ │ - 'z': z │ │ │ │ │ - }; │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroyTile │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * tile - {} │ │ │ │ │ + */ │ │ │ │ │ + destroyTile: function(tile) { │ │ │ │ │ + this.removeTileMonitoringHooks(tile); │ │ │ │ │ + tile.destroy(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /* APIMethod: setMap │ │ │ │ │ - * When the layer is added to a map, then we can fetch our origin │ │ │ │ │ - * (if we don't have one.) │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: getServerResolution │ │ │ │ │ + * Return the closest server-supported resolution. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {} │ │ │ │ │ + * resolution - {Number} The base resolution. If undefined the │ │ │ │ │ + * map resolution is used. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number} The closest server resolution value. │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (!this.tileOrigin) { │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, │ │ │ │ │ - this.maxExtent.bottom); │ │ │ │ │ + getServerResolution: function(resolution) { │ │ │ │ │ + var distance = Number.POSITIVE_INFINITY; │ │ │ │ │ + resolution = resolution || this.map.getResolution(); │ │ │ │ │ + if (this.serverResolutions && │ │ │ │ │ + OpenLayers.Util.indexOf(this.serverResolutions, resolution) === -1) { │ │ │ │ │ + var i, newDistance, newResolution, serverResolution; │ │ │ │ │ + for (i = this.serverResolutions.length - 1; i >= 0; i--) { │ │ │ │ │ + newResolution = this.serverResolutions[i]; │ │ │ │ │ + newDistance = Math.abs(newResolution - resolution); │ │ │ │ │ + if (newDistance > distance) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + distance = newDistance; │ │ │ │ │ + serverResolution = newResolution; │ │ │ │ │ + } │ │ │ │ │ + resolution = serverResolution; │ │ │ │ │ } │ │ │ │ │ + return resolution; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.XYZ" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/ArcGISCache.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.ArcGISCache │ │ │ │ │ - * Layer for accessing cached map tiles from an ArcGIS Server style mapcache. │ │ │ │ │ - * Tile must already be cached for this layer to access it. This does not require │ │ │ │ │ - * ArcGIS Server itself. │ │ │ │ │ - * │ │ │ │ │ - * A few attempts have been made at this kind of layer before. See │ │ │ │ │ - * http://trac.osgeo.org/openlayers/ticket/1967 │ │ │ │ │ - * and │ │ │ │ │ - * http://trac.osgeo.org/openlayers/browser/sandbox/tschaub/arcgiscache/lib/OpenLayers/Layer/ArcGISCache.js │ │ │ │ │ - * │ │ │ │ │ - * Typically the problem encountered is that the tiles seem to "jump around". │ │ │ │ │ - * This is due to the fact that the actual max extent for the tiles on AGS layers │ │ │ │ │ - * changes at each zoom level due to the way these caches are constructed. │ │ │ │ │ - * We have attempted to use the resolutions, tile size, and tile origin │ │ │ │ │ - * from the cache meta data to make the appropriate changes to the max extent │ │ │ │ │ - * of the tile to compensate for this behavior. This must be done as zoom levels change │ │ │ │ │ - * and before tiles are requested, which is why methods from base classes are overridden. │ │ │ │ │ - * │ │ │ │ │ - * For reference, you can access mapcache meta data in two ways. For accessing a │ │ │ │ │ - * mapcache through ArcGIS Server, you can simply go to the landing page for the │ │ │ │ │ - * layer. (ie. http://services.arcgisonline.com/ArcGIS/rest/services/World_Street_Map/MapServer) │ │ │ │ │ - * For accessing it directly through HTTP, there should always be a conf.xml file │ │ │ │ │ - * in the root directory. │ │ │ │ │ - * (ie. http://serverx.esri.com/arcgiscache/DG_County_roads_yesA_backgroundDark/Layers/conf.xml) │ │ │ │ │ - * │ │ │ │ │ - *Inherits from: │ │ │ │ │ - * - │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.ArcGISCache = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String | Array} The base URL for the layer cache. You can also │ │ │ │ │ - * provide a list of URL strings for the layer if your cache is │ │ │ │ │ - * available from multiple origins. This must be set before the layer │ │ │ │ │ - * is drawn. │ │ │ │ │ + * Method: getServerZoom │ │ │ │ │ + * Return the zoom value corresponding to the best matching server │ │ │ │ │ + * resolution, taking into account and . │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number} The closest server supported zoom. This is not the map zoom │ │ │ │ │ + * level, but an index of the server's resolutions array. │ │ │ │ │ */ │ │ │ │ │ - url: null, │ │ │ │ │ + getServerZoom: function() { │ │ │ │ │ + var resolution = this.getServerResolution(); │ │ │ │ │ + return this.serverResolutions ? │ │ │ │ │ + OpenLayers.Util.indexOf(this.serverResolutions, resolution) : │ │ │ │ │ + this.map.getZoomForResolution(resolution) + (this.zoomOffset || 0); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: tileOrigin │ │ │ │ │ - * {} The location of the tile origin for the cache. │ │ │ │ │ - * An ArcGIS cache has it's origin at the upper-left (lowest x value │ │ │ │ │ - * and highest y value of the coordinate system). The units for the │ │ │ │ │ - * tile origin should be the same as the units for the cached data. │ │ │ │ │ + * Method: applyBackBuffer │ │ │ │ │ + * Create, insert, scale and position a back buffer for the layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resolution - {Number} The resolution to transition to. │ │ │ │ │ */ │ │ │ │ │ - tileOrigin: null, │ │ │ │ │ + applyBackBuffer: function(resolution) { │ │ │ │ │ + if (this.backBufferTimerId !== null) { │ │ │ │ │ + this.removeBackBuffer(); │ │ │ │ │ + } │ │ │ │ │ + var backBuffer = this.backBuffer; │ │ │ │ │ + if (!backBuffer) { │ │ │ │ │ + backBuffer = this.createBackBuffer(); │ │ │ │ │ + if (!backBuffer) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (resolution === this.gridResolution) { │ │ │ │ │ + this.div.insertBefore(backBuffer, this.div.firstChild); │ │ │ │ │ + } else { │ │ │ │ │ + this.map.baseLayer.div.parentNode.insertBefore(backBuffer, this.map.baseLayer.div); │ │ │ │ │ + } │ │ │ │ │ + this.backBuffer = backBuffer; │ │ │ │ │ + │ │ │ │ │ + // set some information in the instance for subsequent │ │ │ │ │ + // calls to applyBackBuffer where the same back buffer │ │ │ │ │ + // is reused │ │ │ │ │ + var topLeftTileBounds = this.grid[0][0].bounds; │ │ │ │ │ + this.backBufferLonLat = { │ │ │ │ │ + lon: topLeftTileBounds.left, │ │ │ │ │ + lat: topLeftTileBounds.top │ │ │ │ │ + }; │ │ │ │ │ + this.backBufferResolution = this.gridResolution; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var ratio = this.backBufferResolution / resolution; │ │ │ │ │ + │ │ │ │ │ + // scale the tiles inside the back buffer │ │ │ │ │ + var tiles = backBuffer.childNodes, │ │ │ │ │ + tile; │ │ │ │ │ + for (var i = tiles.length - 1; i >= 0; --i) { │ │ │ │ │ + tile = tiles[i]; │ │ │ │ │ + tile.style.top = ((ratio * tile._i * tile._h) | 0) + 'px'; │ │ │ │ │ + tile.style.left = ((ratio * tile._j * tile._w) | 0) + 'px'; │ │ │ │ │ + tile.style.width = Math.round(ratio * tile._w) + 'px'; │ │ │ │ │ + tile.style.height = Math.round(ratio * tile._h) + 'px'; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // and position it (based on the grid's top-left corner) │ │ │ │ │ + var position = this.getViewPortPxFromLonLat( │ │ │ │ │ + this.backBufferLonLat, resolution); │ │ │ │ │ + var leftOffset = this.map.layerContainerOriginPx.x; │ │ │ │ │ + var topOffset = this.map.layerContainerOriginPx.y; │ │ │ │ │ + backBuffer.style.left = Math.round(position.x - leftOffset) + 'px'; │ │ │ │ │ + backBuffer.style.top = Math.round(position.y - topOffset) + 'px'; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: tileSize │ │ │ │ │ - * {} This size of each tile. Defaults to 256 by 256 pixels. │ │ │ │ │ + * Method: createBackBuffer │ │ │ │ │ + * Create a back buffer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The DOM element for the back buffer, undefined if the │ │ │ │ │ + * grid isn't initialized yet. │ │ │ │ │ */ │ │ │ │ │ - tileSize: new OpenLayers.Size(256, 256), │ │ │ │ │ + createBackBuffer: function() { │ │ │ │ │ + var backBuffer; │ │ │ │ │ + if (this.grid.length > 0) { │ │ │ │ │ + backBuffer = document.createElement('div'); │ │ │ │ │ + backBuffer.id = this.div.id + '_bb'; │ │ │ │ │ + backBuffer.className = 'olBackBuffer'; │ │ │ │ │ + backBuffer.style.position = 'absolute'; │ │ │ │ │ + var map = this.map; │ │ │ │ │ + backBuffer.style.zIndex = this.transitionEffect === 'resize' ? │ │ │ │ │ + this.getZIndex() - 1 : │ │ │ │ │ + // 'map-resize': │ │ │ │ │ + map.Z_INDEX_BASE.BaseLayer - │ │ │ │ │ + (map.getNumLayers() - map.getLayerIndex(this)); │ │ │ │ │ + for (var i = 0, lenI = this.grid.length; i < lenI; i++) { │ │ │ │ │ + for (var j = 0, lenJ = this.grid[i].length; j < lenJ; j++) { │ │ │ │ │ + var tile = this.grid[i][j], │ │ │ │ │ + markup = this.grid[i][j].createBackBuffer(); │ │ │ │ │ + if (markup) { │ │ │ │ │ + markup._i = i; │ │ │ │ │ + markup._j = j; │ │ │ │ │ + markup._w = tile.size.w; │ │ │ │ │ + markup._h = tile.size.h; │ │ │ │ │ + markup.id = tile.id + '_bb'; │ │ │ │ │ + backBuffer.appendChild(markup); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return backBuffer; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: useAGS │ │ │ │ │ - * {Boolean} Indicates if we are going to be accessing the ArcGIS Server (AGS) │ │ │ │ │ - * cache via an AGS MapServer or directly through HTTP. When accessing via │ │ │ │ │ - * AGS the path structure uses a standard z/y/x structure. But AGS actually │ │ │ │ │ - * stores the tile images on disk using a hex based folder structure that looks │ │ │ │ │ - * like "http://example.com/mylayer/L00/R00000000/C00000000.png". Learn more │ │ │ │ │ - * about this here: │ │ │ │ │ - * http://blogs.esri.com/Support/blogs/mappingcenter/archive/2010/08/20/Checking-Your-Local-Cache-Folders.aspx │ │ │ │ │ - * Defaults to true; │ │ │ │ │ + * Method: removeBackBuffer │ │ │ │ │ + * Remove back buffer from DOM. │ │ │ │ │ */ │ │ │ │ │ - useArcGISServer: true, │ │ │ │ │ + removeBackBuffer: function() { │ │ │ │ │ + if (this._transitionElement) { │ │ │ │ │ + for (var i = this.transitionendEvents.length - 1; i >= 0; --i) { │ │ │ │ │ + OpenLayers.Event.stopObserving(this._transitionElement, │ │ │ │ │ + this.transitionendEvents[i], this._removeBackBuffer); │ │ │ │ │ + } │ │ │ │ │ + delete this._transitionElement; │ │ │ │ │ + } │ │ │ │ │ + if (this.backBuffer) { │ │ │ │ │ + if (this.backBuffer.parentNode) { │ │ │ │ │ + this.backBuffer.parentNode.removeChild(this.backBuffer); │ │ │ │ │ + } │ │ │ │ │ + this.backBuffer = null; │ │ │ │ │ + this.backBufferResolution = null; │ │ │ │ │ + if (this.backBufferTimerId !== null) { │ │ │ │ │ + window.clearTimeout(this.backBufferTimerId); │ │ │ │ │ + this.backBufferTimerId = null; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: type │ │ │ │ │ - * {String} Image type for the layer. This becomes the filename extension │ │ │ │ │ - * in tile requests. Default is "png" (generating a url like │ │ │ │ │ - * "http://example.com/mylayer/L00/R00000000/C00000000.png"). │ │ │ │ │ + * Method: moveByPx │ │ │ │ │ + * Move the layer based on pixel vector. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * dx - {Number} │ │ │ │ │ + * dy - {Number} │ │ │ │ │ */ │ │ │ │ │ - type: 'png', │ │ │ │ │ + moveByPx: function(dx, dy) { │ │ │ │ │ + if (!this.singleTile) { │ │ │ │ │ + this.moveGriddedTiles(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: useScales │ │ │ │ │ - * {Boolean} Optional override to indicate that the layer should use 'scale' information │ │ │ │ │ - * returned from the server capabilities object instead of 'resolution' information. │ │ │ │ │ - * This can be important if your tile server uses an unusual DPI for the tiles. │ │ │ │ │ + * APIMethod: setTileSize │ │ │ │ │ + * Check if we are in singleTile mode and if so, set the size as a ratio │ │ │ │ │ + * of the map size (as specified by the layer's 'ratio' property). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * size - {} │ │ │ │ │ */ │ │ │ │ │ - useScales: false, │ │ │ │ │ + setTileSize: function(size) { │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + size.h = parseInt(size.h * this.ratio, 10); │ │ │ │ │ + size.w = parseInt(size.w * this.ratio, 10); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.setTileSize.apply(this, [size]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: overrideDPI │ │ │ │ │ - * {Boolean} Optional override to change the OpenLayers.DOTS_PER_INCH setting based │ │ │ │ │ - * on the tile information in the server capabilities object. This can be useful │ │ │ │ │ - * if your server has a non-standard DPI setting on its tiles, and you're only using │ │ │ │ │ - * tiles with that DPI. This value is used while OpenLayers is calculating resolution │ │ │ │ │ - * using scales, and is not necessary if you have resolution information. (This is │ │ │ │ │ - * typically the case) Regardless, this setting can be useful, but is dangerous │ │ │ │ │ - * because it will impact other layers while calculating resolution. Only use this │ │ │ │ │ - * if you know what you are doing. (See OpenLayers.Util.getResolutionFromScale) │ │ │ │ │ + * APIMethod: getTilesBounds │ │ │ │ │ + * Return the bounds of the tile grid. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} A Bounds object representing the bounds of all the │ │ │ │ │ + * currently loaded tiles (including those partially or not at all seen │ │ │ │ │ + * onscreen). │ │ │ │ │ */ │ │ │ │ │ - overrideDPI: false, │ │ │ │ │ + getTilesBounds: function() { │ │ │ │ │ + var bounds = null; │ │ │ │ │ + │ │ │ │ │ + var length = this.grid.length; │ │ │ │ │ + if (length) { │ │ │ │ │ + var bottomLeftTileBounds = this.grid[length - 1][0].bounds, │ │ │ │ │ + width = this.grid[0].length * bottomLeftTileBounds.getWidth(), │ │ │ │ │ + height = this.grid.length * bottomLeftTileBounds.getHeight(); │ │ │ │ │ + │ │ │ │ │ + bounds = new OpenLayers.Bounds(bottomLeftTileBounds.left, │ │ │ │ │ + bottomLeftTileBounds.bottom, │ │ │ │ │ + bottomLeftTileBounds.left + width, │ │ │ │ │ + bottomLeftTileBounds.bottom + height); │ │ │ │ │ + } │ │ │ │ │ + return bounds; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.ArcGISCache │ │ │ │ │ - * Creates a new instance of this class │ │ │ │ │ + * Method: initSingleTile │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * url - {String} │ │ │ │ │ - * options - {Object} extra layer options │ │ │ │ │ + * bounds - {} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ + initSingleTile: function(bounds) { │ │ │ │ │ + this.events.triggerEvent("retile"); │ │ │ │ │ │ │ │ │ │ - if (this.resolutions) { │ │ │ │ │ - this.serverResolutions = this.resolutions; │ │ │ │ │ - this.maxExtent = this.getMaxExtentForResolution(this.resolutions[0]); │ │ │ │ │ - } │ │ │ │ │ + //determine new tile bounds │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var tileWidth = bounds.getWidth() * this.ratio; │ │ │ │ │ + var tileHeight = bounds.getHeight() * this.ratio; │ │ │ │ │ │ │ │ │ │ - // this block steps through translating the values from the server layer JSON │ │ │ │ │ - // capabilities object into values that we can use. This is also a helpful │ │ │ │ │ - // reference when configuring this layer directly. │ │ │ │ │ - if (this.layerInfo) { │ │ │ │ │ - // alias the object │ │ │ │ │ - var info = this.layerInfo; │ │ │ │ │ + var tileBounds = │ │ │ │ │ + new OpenLayers.Bounds(center.lon - (tileWidth / 2), │ │ │ │ │ + center.lat - (tileHeight / 2), │ │ │ │ │ + center.lon + (tileWidth / 2), │ │ │ │ │ + center.lat + (tileHeight / 2)); │ │ │ │ │ │ │ │ │ │ - // build our extents │ │ │ │ │ - var startingTileExtent = new OpenLayers.Bounds( │ │ │ │ │ - info.fullExtent.xmin, │ │ │ │ │ - info.fullExtent.ymin, │ │ │ │ │ - info.fullExtent.xmax, │ │ │ │ │ - info.fullExtent.ymax │ │ │ │ │ - ); │ │ │ │ │ + var px = this.map.getLayerPxFromLonLat({ │ │ │ │ │ + lon: tileBounds.left, │ │ │ │ │ + lat: tileBounds.top │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - // set our projection based on the given spatial reference. │ │ │ │ │ - // esri uses slightly different IDs, so this may not be comprehensive │ │ │ │ │ - this.projection = 'EPSG:' + info.spatialReference.wkid; │ │ │ │ │ - this.sphericalMercator = (info.spatialReference.wkid == 102100); │ │ │ │ │ + if (!this.grid.length) { │ │ │ │ │ + this.grid[0] = []; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // convert esri units into openlayers units (basic feet or meters only) │ │ │ │ │ - this.units = (info.units == "esriFeet") ? 'ft' : 'm'; │ │ │ │ │ + var tile = this.grid[0][0]; │ │ │ │ │ + if (!tile) { │ │ │ │ │ + tile = this.addTile(tileBounds, px); │ │ │ │ │ │ │ │ │ │ - // optional extended section based on whether or not the server returned │ │ │ │ │ - // specific tile information │ │ │ │ │ - if (!!info.tileInfo) { │ │ │ │ │ - // either set the tiles based on rows/columns, or specific width/height │ │ │ │ │ - this.tileSize = new OpenLayers.Size( │ │ │ │ │ - info.tileInfo.width || info.tileInfo.cols, │ │ │ │ │ - info.tileInfo.height || info.tileInfo.rows │ │ │ │ │ - ); │ │ │ │ │ + this.addTileMonitoringHooks(tile); │ │ │ │ │ + tile.draw(); │ │ │ │ │ + this.grid[0][0] = tile; │ │ │ │ │ + } else { │ │ │ │ │ + tile.moveTo(tileBounds, px); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // this must be set when manually configuring this layer │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat( │ │ │ │ │ - info.tileInfo.origin.x, │ │ │ │ │ - info.tileInfo.origin.y │ │ │ │ │ - ); │ │ │ │ │ + //remove all but our single tile │ │ │ │ │ + this.removeExcessTiles(1, 1); │ │ │ │ │ │ │ │ │ │ - var upperLeft = new OpenLayers.Geometry.Point( │ │ │ │ │ - startingTileExtent.left, │ │ │ │ │ - startingTileExtent.top │ │ │ │ │ - ); │ │ │ │ │ + // store the resolution of the grid │ │ │ │ │ + this.gridResolution = this.getServerResolution(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var bottomRight = new OpenLayers.Geometry.Point( │ │ │ │ │ - startingTileExtent.right, │ │ │ │ │ - startingTileExtent.bottom │ │ │ │ │ - ); │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateGridLayout │ │ │ │ │ + * Generate parameters for the grid layout. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {|Object} OpenLayers.Bounds or an │ │ │ │ │ + * object with a 'left' and 'top' properties. │ │ │ │ │ + * origin - {|Object} OpenLayers.LonLat or an │ │ │ │ │ + * object with a 'lon' and 'lat' properties. │ │ │ │ │ + * resolution - {Number} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Object containing properties tilelon, tilelat, startcol, │ │ │ │ │ + * startrow │ │ │ │ │ + */ │ │ │ │ │ + calculateGridLayout: function(bounds, origin, resolution) { │ │ │ │ │ + var tilelon = resolution * this.tileSize.w; │ │ │ │ │ + var tilelat = resolution * this.tileSize.h; │ │ │ │ │ │ │ │ │ │ - if (this.useScales) { │ │ │ │ │ - this.scales = []; │ │ │ │ │ - } else { │ │ │ │ │ - this.resolutions = []; │ │ │ │ │ - } │ │ │ │ │ + var offsetlon = bounds.left - origin.lon; │ │ │ │ │ + var tilecol = Math.floor(offsetlon / tilelon) - this.buffer; │ │ │ │ │ │ │ │ │ │ - this.lods = []; │ │ │ │ │ - for (var key in info.tileInfo.lods) { │ │ │ │ │ - if (info.tileInfo.lods.hasOwnProperty(key)) { │ │ │ │ │ - var lod = info.tileInfo.lods[key]; │ │ │ │ │ - if (this.useScales) { │ │ │ │ │ - this.scales.push(lod.scale); │ │ │ │ │ - } else { │ │ │ │ │ - this.resolutions.push(lod.resolution); │ │ │ │ │ - } │ │ │ │ │ + var rowSign = this.rowSign; │ │ │ │ │ │ │ │ │ │ - var start = this.getContainingTileCoords(upperLeft, lod.resolution); │ │ │ │ │ - lod.startTileCol = start.x; │ │ │ │ │ - lod.startTileRow = start.y; │ │ │ │ │ + var offsetlat = rowSign * (origin.lat - bounds.top + tilelat); │ │ │ │ │ + var tilerow = Math[~rowSign ? 'floor' : 'ceil'](offsetlat / tilelat) - this.buffer * rowSign; │ │ │ │ │ │ │ │ │ │ - var end = this.getContainingTileCoords(bottomRight, lod.resolution); │ │ │ │ │ - lod.endTileCol = end.x; │ │ │ │ │ - lod.endTileRow = end.y; │ │ │ │ │ - this.lods.push(lod); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + return { │ │ │ │ │ + tilelon: tilelon, │ │ │ │ │ + tilelat: tilelat, │ │ │ │ │ + startcol: tilecol, │ │ │ │ │ + startrow: tilerow │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - this.maxExtent = this.calculateMaxExtentWithLOD(this.lods[0]); │ │ │ │ │ - this.serverResolutions = this.resolutions; │ │ │ │ │ - if (this.overrideDPI && info.tileInfo.dpi) { │ │ │ │ │ - // see comment above for 'overrideDPI' │ │ │ │ │ - OpenLayers.DOTS_PER_INCH = info.tileInfo.dpi; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getTileOrigin │ │ │ │ │ + * Determine the origin for aligning the grid of tiles. If a │ │ │ │ │ + * property is supplied, that will be returned. Otherwise, the origin │ │ │ │ │ + * will be derived from the layer's property. In this case, │ │ │ │ │ + * the tile origin will be the corner of the given by the │ │ │ │ │ + * property. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} The tile origin. │ │ │ │ │ + */ │ │ │ │ │ + getTileOrigin: function() { │ │ │ │ │ + var origin = this.tileOrigin; │ │ │ │ │ + if (!origin) { │ │ │ │ │ + var extent = this.getMaxExtent(); │ │ │ │ │ + var edges = ({ │ │ │ │ │ + "tl": ["left", "top"], │ │ │ │ │ + "tr": ["right", "top"], │ │ │ │ │ + "bl": ["left", "bottom"], │ │ │ │ │ + "br": ["right", "bottom"] │ │ │ │ │ + })[this.tileOriginCorner]; │ │ │ │ │ + origin = new OpenLayers.LonLat(extent[edges[0]], extent[edges[1]]); │ │ │ │ │ } │ │ │ │ │ + return origin; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getContainingTileCoords │ │ │ │ │ - * Calculates the x/y pixel corresponding to the position of the tile │ │ │ │ │ - * that contains the given point and for the for the given resolution. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: getTileBoundsForGridIndex │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * point - {} │ │ │ │ │ - * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} The x/y pixel corresponding to the position │ │ │ │ │ - * of the upper left tile for the given resolution. │ │ │ │ │ + * row - {Number} The row of the grid │ │ │ │ │ + * col - {Number} The column of the grid │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} The bounds for the tile at (row, col) │ │ │ │ │ */ │ │ │ │ │ - getContainingTileCoords: function(point, res) { │ │ │ │ │ - return new OpenLayers.Pixel( │ │ │ │ │ - Math.max(Math.floor((point.x - this.tileOrigin.lon) / (this.tileSize.w * res)), 0), │ │ │ │ │ - Math.max(Math.floor((this.tileOrigin.lat - point.y) / (this.tileSize.h * res)), 0) │ │ │ │ │ + getTileBoundsForGridIndex: function(row, col) { │ │ │ │ │ + var origin = this.getTileOrigin(); │ │ │ │ │ + var tileLayout = this.gridLayout; │ │ │ │ │ + var tilelon = tileLayout.tilelon; │ │ │ │ │ + var tilelat = tileLayout.tilelat; │ │ │ │ │ + var startcol = tileLayout.startcol; │ │ │ │ │ + var startrow = tileLayout.startrow; │ │ │ │ │ + var rowSign = this.rowSign; │ │ │ │ │ + return new OpenLayers.Bounds( │ │ │ │ │ + origin.lon + (startcol + col) * tilelon, │ │ │ │ │ + origin.lat - (startrow + row * rowSign) * tilelat * rowSign, │ │ │ │ │ + origin.lon + (startcol + col + 1) * tilelon, │ │ │ │ │ + origin.lat - (startrow + (row - 1) * rowSign) * tilelat * rowSign │ │ │ │ │ ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateMaxExtentWithLOD │ │ │ │ │ - * Given a Level of Detail object from the server, this function │ │ │ │ │ - * calculates the actual max extent │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * lod - {Object} a Level of Detail Object from the server capabilities object │ │ │ │ │ - representing a particular zoom level │ │ │ │ │ + /** │ │ │ │ │ + * Method: initGriddedTiles │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} The actual extent of the tiles for the given zoom level │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {} │ │ │ │ │ */ │ │ │ │ │ - calculateMaxExtentWithLOD: function(lod) { │ │ │ │ │ - // the max extent we're provided with just overlaps some tiles │ │ │ │ │ - // our real extent is the bounds of all the tiles we touch │ │ │ │ │ + initGriddedTiles: function(bounds) { │ │ │ │ │ + this.events.triggerEvent("retile"); │ │ │ │ │ │ │ │ │ │ - var numTileCols = (lod.endTileCol - lod.startTileCol) + 1; │ │ │ │ │ - var numTileRows = (lod.endTileRow - lod.startTileRow) + 1; │ │ │ │ │ + // work out mininum number of rows and columns; this is the number of │ │ │ │ │ + // tiles required to cover the viewport plus at least one for panning │ │ │ │ │ │ │ │ │ │ - var minX = this.tileOrigin.lon + (lod.startTileCol * this.tileSize.w * lod.resolution); │ │ │ │ │ - var maxX = minX + (numTileCols * this.tileSize.w * lod.resolution); │ │ │ │ │ + var viewSize = this.map.getSize(); │ │ │ │ │ │ │ │ │ │ - var maxY = this.tileOrigin.lat - (lod.startTileRow * this.tileSize.h * lod.resolution); │ │ │ │ │ - var minY = maxY - (numTileRows * this.tileSize.h * lod.resolution); │ │ │ │ │ - return new OpenLayers.Bounds(minX, minY, maxX, maxY); │ │ │ │ │ - }, │ │ │ │ │ + var origin = this.getTileOrigin(); │ │ │ │ │ + var resolution = this.map.getResolution(), │ │ │ │ │ + serverResolution = this.getServerResolution(), │ │ │ │ │ + ratio = resolution / serverResolution, │ │ │ │ │ + tileSize = { │ │ │ │ │ + w: this.tileSize.w / ratio, │ │ │ │ │ + h: this.tileSize.h / ratio │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateMaxExtentWithExtent │ │ │ │ │ - * Given a 'suggested' max extent from the server, this function uses │ │ │ │ │ - * information about the actual tile sizes to determine the actual │ │ │ │ │ - * extent of the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * extent - {} The 'suggested' extent for the layer │ │ │ │ │ - * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} The actual extent of the tiles for the given zoom level │ │ │ │ │ - */ │ │ │ │ │ - calculateMaxExtentWithExtent: function(extent, res) { │ │ │ │ │ - var upperLeft = new OpenLayers.Geometry.Point(extent.left, extent.top); │ │ │ │ │ - var bottomRight = new OpenLayers.Geometry.Point(extent.right, extent.bottom); │ │ │ │ │ - var start = this.getContainingTileCoords(upperLeft, res); │ │ │ │ │ - var end = this.getContainingTileCoords(bottomRight, res); │ │ │ │ │ - var lod = { │ │ │ │ │ - resolution: res, │ │ │ │ │ - startTileCol: start.x, │ │ │ │ │ - startTileRow: start.y, │ │ │ │ │ - endTileCol: end.x, │ │ │ │ │ - endTileRow: end.y │ │ │ │ │ - }; │ │ │ │ │ - return this.calculateMaxExtentWithLOD(lod); │ │ │ │ │ + var minRows = Math.ceil(viewSize.h / tileSize.h) + │ │ │ │ │ + 2 * this.buffer + 1; │ │ │ │ │ + var minCols = Math.ceil(viewSize.w / tileSize.w) + │ │ │ │ │ + 2 * this.buffer + 1; │ │ │ │ │ + │ │ │ │ │ + var tileLayout = this.calculateGridLayout(bounds, origin, serverResolution); │ │ │ │ │ + this.gridLayout = tileLayout; │ │ │ │ │ + │ │ │ │ │ + var tilelon = tileLayout.tilelon; │ │ │ │ │ + var tilelat = tileLayout.tilelat; │ │ │ │ │ + │ │ │ │ │ + var layerContainerDivLeft = this.map.layerContainerOriginPx.x; │ │ │ │ │ + var layerContainerDivTop = this.map.layerContainerOriginPx.y; │ │ │ │ │ + │ │ │ │ │ + var tileBounds = this.getTileBoundsForGridIndex(0, 0); │ │ │ │ │ + var startPx = this.map.getViewPortPxFromLonLat( │ │ │ │ │ + new OpenLayers.LonLat(tileBounds.left, tileBounds.top) │ │ │ │ │ + ); │ │ │ │ │ + startPx.x = Math.round(startPx.x) - layerContainerDivLeft; │ │ │ │ │ + startPx.y = Math.round(startPx.y) - layerContainerDivTop; │ │ │ │ │ + │ │ │ │ │ + var tileData = [], │ │ │ │ │ + center = this.map.getCenter(); │ │ │ │ │ + │ │ │ │ │ + var rowidx = 0; │ │ │ │ │ + do { │ │ │ │ │ + var row = this.grid[rowidx]; │ │ │ │ │ + if (!row) { │ │ │ │ │ + row = []; │ │ │ │ │ + this.grid.push(row); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var colidx = 0; │ │ │ │ │ + do { │ │ │ │ │ + tileBounds = this.getTileBoundsForGridIndex(rowidx, colidx); │ │ │ │ │ + var px = startPx.clone(); │ │ │ │ │ + px.x = px.x + colidx * Math.round(tileSize.w); │ │ │ │ │ + px.y = px.y + rowidx * Math.round(tileSize.h); │ │ │ │ │ + var tile = row[colidx]; │ │ │ │ │ + if (!tile) { │ │ │ │ │ + tile = this.addTile(tileBounds, px); │ │ │ │ │ + this.addTileMonitoringHooks(tile); │ │ │ │ │ + row.push(tile); │ │ │ │ │ + } else { │ │ │ │ │ + tile.moveTo(tileBounds, px, false); │ │ │ │ │ + } │ │ │ │ │ + var tileCenter = tileBounds.getCenterLonLat(); │ │ │ │ │ + tileData.push({ │ │ │ │ │ + tile: tile, │ │ │ │ │ + distance: Math.pow(tileCenter.lon - center.lon, 2) + │ │ │ │ │ + Math.pow(tileCenter.lat - center.lat, 2) │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + colidx += 1; │ │ │ │ │ + } while ((tileBounds.right <= bounds.right + tilelon * this.buffer) || │ │ │ │ │ + colidx < minCols); │ │ │ │ │ + │ │ │ │ │ + rowidx += 1; │ │ │ │ │ + } while ((tileBounds.bottom >= bounds.bottom - tilelat * this.buffer) || │ │ │ │ │ + rowidx < minRows); │ │ │ │ │ + │ │ │ │ │ + //shave off exceess rows and colums │ │ │ │ │ + this.removeExcessTiles(rowidx, colidx); │ │ │ │ │ + │ │ │ │ │ + var resolution = this.getServerResolution(); │ │ │ │ │ + // store the resolution of the grid │ │ │ │ │ + this.gridResolution = resolution; │ │ │ │ │ + │ │ │ │ │ + //now actually draw the tiles │ │ │ │ │ + tileData.sort(function(a, b) { │ │ │ │ │ + return a.distance - b.distance; │ │ │ │ │ + }); │ │ │ │ │ + for (var i = 0, ii = tileData.length; i < ii; ++i) { │ │ │ │ │ + tileData[i].tile.draw(); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getUpperLeftTileCoord │ │ │ │ │ - * Calculates the x/y pixel corresponding to the position │ │ │ │ │ - * of the upper left tile for the given resolution. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} The x/y pixel corresponding to the position │ │ │ │ │ - * of the upper left tile for the given resolution. │ │ │ │ │ + /** │ │ │ │ │ + * Method: getMaxExtent │ │ │ │ │ + * Get this layer's maximum extent. (Implemented as a getter for │ │ │ │ │ + * potential specific implementations in sub-classes.) │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} │ │ │ │ │ */ │ │ │ │ │ - getUpperLeftTileCoord: function(res) { │ │ │ │ │ - var upperLeft = new OpenLayers.Geometry.Point( │ │ │ │ │ - this.maxExtent.left, │ │ │ │ │ - this.maxExtent.top); │ │ │ │ │ - return this.getContainingTileCoords(upperLeft, res); │ │ │ │ │ + getMaxExtent: function() { │ │ │ │ │ + return this.maxExtent; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getLowerRightTileCoord │ │ │ │ │ - * Calculates the x/y pixel corresponding to the position │ │ │ │ │ - * of the lower right tile for the given resolution. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} The x/y pixel corresponding to the position │ │ │ │ │ - * of the lower right tile for the given resolution. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: addTile │ │ │ │ │ + * Create a tile, initialize it, and add it to the layer div. │ │ │ │ │ + * │ │ │ │ │ + * Parameters │ │ │ │ │ + * bounds - {} │ │ │ │ │ + * position - {} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} The added OpenLayers.Tile │ │ │ │ │ */ │ │ │ │ │ - getLowerRightTileCoord: function(res) { │ │ │ │ │ - var bottomRight = new OpenLayers.Geometry.Point( │ │ │ │ │ - this.maxExtent.right, │ │ │ │ │ - this.maxExtent.bottom); │ │ │ │ │ - return this.getContainingTileCoords(bottomRight, res); │ │ │ │ │ + addTile: function(bounds, position) { │ │ │ │ │ + var tile = new this.tileClass( │ │ │ │ │ + this, position, bounds, null, this.tileSize, this.tileOptions │ │ │ │ │ + ); │ │ │ │ │ + this.events.triggerEvent("addtile", { │ │ │ │ │ + tile: tile │ │ │ │ │ + }); │ │ │ │ │ + return tile; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getMaxExtentForResolution │ │ │ │ │ - * Since the max extent of a set of tiles can change from zoom level │ │ │ │ │ - * to zoom level, we need to be able to calculate that max extent │ │ │ │ │ - * for a given resolution. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ + * Method: addTileMonitoringHooks │ │ │ │ │ + * This function takes a tile as input and adds the appropriate hooks to │ │ │ │ │ + * the tile so that the layer can keep track of the loading tiles. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} The extent for this resolution │ │ │ │ │ + * Parameters: │ │ │ │ │ + * tile - {} │ │ │ │ │ */ │ │ │ │ │ - getMaxExtentForResolution: function(res) { │ │ │ │ │ - var start = this.getUpperLeftTileCoord(res); │ │ │ │ │ - var end = this.getLowerRightTileCoord(res); │ │ │ │ │ + addTileMonitoringHooks: function(tile) { │ │ │ │ │ │ │ │ │ │ - var numTileCols = (end.x - start.x) + 1; │ │ │ │ │ - var numTileRows = (end.y - start.y) + 1; │ │ │ │ │ + var replacingCls = 'olTileReplacing'; │ │ │ │ │ │ │ │ │ │ - var minX = this.tileOrigin.lon + (start.x * this.tileSize.w * res); │ │ │ │ │ - var maxX = minX + (numTileCols * this.tileSize.w * res); │ │ │ │ │ + tile.onLoadStart = function() { │ │ │ │ │ + //if that was first tile then trigger a 'loadstart' on the layer │ │ │ │ │ + if (this.loading === false) { │ │ │ │ │ + this.loading = true; │ │ │ │ │ + this.events.triggerEvent("loadstart"); │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("tileloadstart", { │ │ │ │ │ + tile: tile │ │ │ │ │ + }); │ │ │ │ │ + this.numLoadingTiles++; │ │ │ │ │ + if (!this.singleTile && this.backBuffer && this.gridResolution === this.backBufferResolution) { │ │ │ │ │ + OpenLayers.Element.addClass(tile.getTile(), replacingCls); │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - var maxY = this.tileOrigin.lat - (start.y * this.tileSize.h * res); │ │ │ │ │ - var minY = maxY - (numTileRows * this.tileSize.h * res); │ │ │ │ │ - return new OpenLayers.Bounds(minX, minY, maxX, maxY); │ │ │ │ │ + tile.onLoadEnd = function(evt) { │ │ │ │ │ + this.numLoadingTiles--; │ │ │ │ │ + var aborted = evt.type === 'unload'; │ │ │ │ │ + this.events.triggerEvent("tileloaded", { │ │ │ │ │ + tile: tile, │ │ │ │ │ + aborted: aborted │ │ │ │ │ + }); │ │ │ │ │ + if (!this.singleTile && !aborted && this.backBuffer && this.gridResolution === this.backBufferResolution) { │ │ │ │ │ + var tileDiv = tile.getTile(); │ │ │ │ │ + if (OpenLayers.Element.getStyle(tileDiv, 'display') === 'none') { │ │ │ │ │ + var bufferTile = document.getElementById(tile.id + '_bb'); │ │ │ │ │ + if (bufferTile) { │ │ │ │ │ + bufferTile.parentNode.removeChild(bufferTile); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.removeClass(tileDiv, replacingCls); │ │ │ │ │ + } │ │ │ │ │ + //if that was the last tile, then trigger a 'loadend' on the layer │ │ │ │ │ + if (this.numLoadingTiles === 0) { │ │ │ │ │ + if (this.backBuffer) { │ │ │ │ │ + if (this.backBuffer.childNodes.length === 0) { │ │ │ │ │ + // no tiles transitioning, remove immediately │ │ │ │ │ + this.removeBackBuffer(); │ │ │ │ │ + } else { │ │ │ │ │ + // wait until transition has ended or delay has passed │ │ │ │ │ + this._transitionElement = aborted ? │ │ │ │ │ + this.div.lastChild : tile.imgDiv; │ │ │ │ │ + var transitionendEvents = this.transitionendEvents; │ │ │ │ │ + for (var i = transitionendEvents.length - 1; i >= 0; --i) { │ │ │ │ │ + OpenLayers.Event.observe(this._transitionElement, │ │ │ │ │ + transitionendEvents[i], │ │ │ │ │ + this._removeBackBuffer); │ │ │ │ │ + } │ │ │ │ │ + // the removal of the back buffer is delayed to prevent │ │ │ │ │ + // flash effects due to the animation of tile displaying │ │ │ │ │ + this.backBufferTimerId = window.setTimeout( │ │ │ │ │ + this._removeBackBuffer, this.removeBackBufferDelay │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.loading = false; │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + tile.onLoadError = function() { │ │ │ │ │ + this.events.triggerEvent("tileerror", { │ │ │ │ │ + tile: tile │ │ │ │ │ + }); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + tile.events.on({ │ │ │ │ │ + "loadstart": tile.onLoadStart, │ │ │ │ │ + "loadend": tile.onLoadEnd, │ │ │ │ │ + "unload": tile.onLoadEnd, │ │ │ │ │ + "loaderror": tile.onLoadError, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Returns an exact clone of this OpenLayers.Layer.ArcGISCache │ │ │ │ │ + * Method: removeTileMonitoringHooks │ │ │ │ │ + * This function takes a tile as input and removes the tile hooks │ │ │ │ │ + * that were added in addTileMonitoringHooks() │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * [obj] - {Object} optional object to assign the cloned instance to. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} clone of this instance │ │ │ │ │ + * tile - {} │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.ArcGISCache(this.name, this.url, this.options); │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + removeTileMonitoringHooks: function(tile) { │ │ │ │ │ + tile.unload(); │ │ │ │ │ + tile.events.un({ │ │ │ │ │ + "loadstart": tile.onLoadStart, │ │ │ │ │ + "loadend": tile.onLoadEnd, │ │ │ │ │ + "unload": tile.onLoadEnd, │ │ │ │ │ + "loaderror": tile.onLoadError, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: initGriddedTiles │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {} │ │ │ │ │ + * Method: moveGriddedTiles │ │ │ │ │ */ │ │ │ │ │ - initGriddedTiles: function(bounds) { │ │ │ │ │ - delete this._tileOrigin; │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initGriddedTiles.apply(this, arguments); │ │ │ │ │ + moveGriddedTiles: function() { │ │ │ │ │ + var buffer = this.buffer + 1; │ │ │ │ │ + while (true) { │ │ │ │ │ + var tlTile = this.grid[0][0]; │ │ │ │ │ + var tlViewPort = { │ │ │ │ │ + x: tlTile.position.x + │ │ │ │ │ + this.map.layerContainerOriginPx.x, │ │ │ │ │ + y: tlTile.position.y + │ │ │ │ │ + this.map.layerContainerOriginPx.y │ │ │ │ │ + }; │ │ │ │ │ + var ratio = this.getServerResolution() / this.map.getResolution(); │ │ │ │ │ + var tileSize = { │ │ │ │ │ + w: Math.round(this.tileSize.w * ratio), │ │ │ │ │ + h: Math.round(this.tileSize.h * ratio) │ │ │ │ │ + }; │ │ │ │ │ + if (tlViewPort.x > -tileSize.w * (buffer - 1)) { │ │ │ │ │ + this.shiftColumn(true, tileSize); │ │ │ │ │ + } else if (tlViewPort.x < -tileSize.w * buffer) { │ │ │ │ │ + this.shiftColumn(false, tileSize); │ │ │ │ │ + } else if (tlViewPort.y > -tileSize.h * (buffer - 1)) { │ │ │ │ │ + this.shiftRow(true, tileSize); │ │ │ │ │ + } else if (tlViewPort.y < -tileSize.h * buffer) { │ │ │ │ │ + this.shiftRow(false, tileSize); │ │ │ │ │ + } else { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getMaxExtent │ │ │ │ │ - * Get this layer's maximum extent. │ │ │ │ │ + * Method: shiftRow │ │ │ │ │ + * Shifty grid work │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * prepend - {Boolean} if true, prepend to beginning. │ │ │ │ │ + * if false, then append to end │ │ │ │ │ + * tileSize - {Object} rendered tile size; object with w and h properties │ │ │ │ │ */ │ │ │ │ │ - getMaxExtent: function() { │ │ │ │ │ - var resolution = this.map.getResolution(); │ │ │ │ │ - return this.maxExtent = this.getMaxExtentForResolution(resolution); │ │ │ │ │ + shiftRow: function(prepend, tileSize) { │ │ │ │ │ + var grid = this.grid; │ │ │ │ │ + var rowIndex = prepend ? 0 : (grid.length - 1); │ │ │ │ │ + var sign = prepend ? -1 : 1; │ │ │ │ │ + var rowSign = this.rowSign; │ │ │ │ │ + var tileLayout = this.gridLayout; │ │ │ │ │ + tileLayout.startrow += sign * rowSign; │ │ │ │ │ + │ │ │ │ │ + var modelRow = grid[rowIndex]; │ │ │ │ │ + var row = grid[prepend ? 'pop' : 'shift'](); │ │ │ │ │ + for (var i = 0, len = row.length; i < len; i++) { │ │ │ │ │ + var tile = row[i]; │ │ │ │ │ + var position = modelRow[i].position.clone(); │ │ │ │ │ + position.y += tileSize.h * sign; │ │ │ │ │ + tile.moveTo(this.getTileBoundsForGridIndex(rowIndex, i), position); │ │ │ │ │ + } │ │ │ │ │ + grid[prepend ? 'unshift' : 'push'](row); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getTileOrigin │ │ │ │ │ - * Determine the origin for aligning the grid of tiles. │ │ │ │ │ - * The origin will be derived from the layer's property. │ │ │ │ │ + * Method: shiftColumn │ │ │ │ │ + * Shift grid work in the other dimension │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} The tile origin. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * prepend - {Boolean} if true, prepend to beginning. │ │ │ │ │ + * if false, then append to end │ │ │ │ │ + * tileSize - {Object} rendered tile size; object with w and h properties │ │ │ │ │ */ │ │ │ │ │ - getTileOrigin: function() { │ │ │ │ │ - if (!this._tileOrigin) { │ │ │ │ │ - var extent = this.getMaxExtent(); │ │ │ │ │ - this._tileOrigin = new OpenLayers.LonLat(extent.left, extent.bottom); │ │ │ │ │ + shiftColumn: function(prepend, tileSize) { │ │ │ │ │ + var grid = this.grid; │ │ │ │ │ + var colIndex = prepend ? 0 : (grid[0].length - 1); │ │ │ │ │ + var sign = prepend ? -1 : 1; │ │ │ │ │ + var tileLayout = this.gridLayout; │ │ │ │ │ + tileLayout.startcol += sign; │ │ │ │ │ + │ │ │ │ │ + for (var i = 0, len = grid.length; i < len; i++) { │ │ │ │ │ + var row = grid[i]; │ │ │ │ │ + var position = row[colIndex].position.clone(); │ │ │ │ │ + var tile = row[prepend ? 'pop' : 'shift'](); │ │ │ │ │ + position.x += tileSize.w * sign; │ │ │ │ │ + tile.moveTo(this.getTileBoundsForGridIndex(i, colIndex), position); │ │ │ │ │ + row[prepend ? 'unshift' : 'push'](tile); │ │ │ │ │ } │ │ │ │ │ - return this._tileOrigin; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * Determine the URL for a tile given the tile bounds. This is should support │ │ │ │ │ - * urls that access tiles through an ArcGIS Server MapServer or directly through │ │ │ │ │ - * the hex folder structure using HTTP. Just be sure to set the useArcGISServer │ │ │ │ │ - * property appropriately! This is basically the same as │ │ │ │ │ - * 'OpenLayers.Layer.TMS.getURL', but with the addition of hex addressing, │ │ │ │ │ - * and tile rounding. │ │ │ │ │ - * │ │ │ │ │ + * Method: removeExcessTiles │ │ │ │ │ + * When the size of the map or the buffer changes, we may need to │ │ │ │ │ + * remove some excess rows and columns. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The URL for a tile based on given bounds. │ │ │ │ │ + * rows - {Integer} Maximum number of rows we want our grid to have. │ │ │ │ │ + * columns - {Integer} Maximum number of columns we want our grid to have. │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var res = this.getResolution(); │ │ │ │ │ - │ │ │ │ │ - // tile center │ │ │ │ │ - var originTileX = (this.tileOrigin.lon + (res * this.tileSize.w / 2)); │ │ │ │ │ - var originTileY = (this.tileOrigin.lat - (res * this.tileSize.h / 2)); │ │ │ │ │ - │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var point = { │ │ │ │ │ - x: center.lon, │ │ │ │ │ - y: center.lat │ │ │ │ │ - }; │ │ │ │ │ - var x = (Math.round(Math.abs((center.lon - originTileX) / (res * this.tileSize.w)))); │ │ │ │ │ - var y = (Math.round(Math.abs((originTileY - center.lat) / (res * this.tileSize.h)))); │ │ │ │ │ - var z = this.map.getZoom(); │ │ │ │ │ + removeExcessTiles: function(rows, columns) { │ │ │ │ │ + var i, l; │ │ │ │ │ │ │ │ │ │ - // this prevents us from getting pink tiles (non-existant tiles) │ │ │ │ │ - if (this.lods) { │ │ │ │ │ - var lod = this.lods[this.map.getZoom()]; │ │ │ │ │ - if ((x < lod.startTileCol || x > lod.endTileCol) || │ │ │ │ │ - (y < lod.startTileRow || y > lod.endTileRow)) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - var start = this.getUpperLeftTileCoord(res); │ │ │ │ │ - var end = this.getLowerRightTileCoord(res); │ │ │ │ │ - if ((x < start.x || x >= end.x) || │ │ │ │ │ - (y < start.y || y >= end.y)) { │ │ │ │ │ - return null; │ │ │ │ │ + // remove extra rows │ │ │ │ │ + while (this.grid.length > rows) { │ │ │ │ │ + var row = this.grid.pop(); │ │ │ │ │ + for (i = 0, l = row.length; i < l; i++) { │ │ │ │ │ + var tile = row[i]; │ │ │ │ │ + this.destroyTile(tile); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // Construct the url string │ │ │ │ │ - var url = this.url; │ │ │ │ │ - var s = '' + x + y + z; │ │ │ │ │ - │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(s, url); │ │ │ │ │ + // remove extra columns │ │ │ │ │ + for (i = 0, l = this.grid.length; i < l; i++) { │ │ │ │ │ + while (this.grid[i].length > columns) { │ │ │ │ │ + var row = this.grid[i]; │ │ │ │ │ + var tile = row.pop(); │ │ │ │ │ + this.destroyTile(tile); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Accessing tiles through ArcGIS Server uses a different path │ │ │ │ │ - // structure than direct access via the folder structure. │ │ │ │ │ - if (this.useArcGISServer) { │ │ │ │ │ - // AGS MapServers have pretty url access to tiles │ │ │ │ │ - url = url + '/tile/${z}/${y}/${x}'; │ │ │ │ │ - } else { │ │ │ │ │ - // The tile images are stored using hex values on disk. │ │ │ │ │ - x = 'C' + OpenLayers.Number.zeroPad(x, 8, 16); │ │ │ │ │ - y = 'R' + OpenLayers.Number.zeroPad(y, 8, 16); │ │ │ │ │ - z = 'L' + OpenLayers.Number.zeroPad(z, 2, 10); │ │ │ │ │ - url = url + '/${z}/${y}/${x}.' + this.type; │ │ │ │ │ + /** │ │ │ │ │ + * Method: onMapResize │ │ │ │ │ + * For singleTile layers, this will set a new tile size according to the │ │ │ │ │ + * dimensions of the map pane. │ │ │ │ │ + */ │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + this.clearGrid(); │ │ │ │ │ + this.setTileSize(); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Write the values into our formatted url │ │ │ │ │ - url = OpenLayers.String.format(url, { │ │ │ │ │ - 'x': x, │ │ │ │ │ - 'y': y, │ │ │ │ │ - 'z': z │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - return OpenLayers.Util.urlAppend( │ │ │ │ │ - url, OpenLayers.Util.getParameterString(this.params) │ │ │ │ │ - ); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getTileBounds │ │ │ │ │ + * Returns The tile bounds for a layer given a pixel location. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * viewPortPx - {} The location in the viewport. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} Bounds of the tile at the given pixel location. │ │ │ │ │ + */ │ │ │ │ │ + getTileBounds: function(viewPortPx) { │ │ │ │ │ + var maxExtent = this.maxExtent; │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var tileMapWidth = resolution * this.tileSize.w; │ │ │ │ │ + var tileMapHeight = resolution * this.tileSize.h; │ │ │ │ │ + var mapPoint = this.getLonLatFromViewPortPx(viewPortPx); │ │ │ │ │ + var tileLeft = maxExtent.left + (tileMapWidth * │ │ │ │ │ + Math.floor((mapPoint.lon - │ │ │ │ │ + maxExtent.left) / │ │ │ │ │ + tileMapWidth)); │ │ │ │ │ + var tileBottom = maxExtent.bottom + (tileMapHeight * │ │ │ │ │ + Math.floor((mapPoint.lat - │ │ │ │ │ + maxExtent.bottom) / │ │ │ │ │ + tileMapHeight)); │ │ │ │ │ + return new OpenLayers.Bounds(tileLeft, tileBottom, │ │ │ │ │ + tileLeft + tileMapWidth, │ │ │ │ │ + tileBottom + tileMapHeight); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: 'OpenLayers.Layer.ArcGISCache' │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Grid" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/ArcXML.js │ │ │ │ │ + OpenLayers/TileManager.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/MultiPolygon.js │ │ │ │ │ - * @requires OpenLayers/Geometry/LinearRing.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Element.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/Tile/Image.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.ArcXML │ │ │ │ │ - * Read/Write ArcXML. Create a new instance with the │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ + * Class: OpenLayers.TileManager │ │ │ │ │ + * Provides queueing of image requests and caching of image elements. │ │ │ │ │ + * │ │ │ │ │ + * Queueing avoids unnecessary image requests while changing zoom levels │ │ │ │ │ + * quickly, and helps improve dragging performance on mobile devices that show │ │ │ │ │ + * a lag in dragging when loading of new images starts. and │ │ │ │ │ + * are the configuration options to control this behavior. │ │ │ │ │ + * │ │ │ │ │ + * Caching avoids setting the src on image elements for images that have already │ │ │ │ │ + * been used. Several maps can share a TileManager instance, in which case each │ │ │ │ │ + * map gets its own tile queue, but all maps share the same tile cache. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.ArcXML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.TileManager = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: fontStyleKeys │ │ │ │ │ - * {Array} List of keys used in font styling. │ │ │ │ │ + * APIProperty: cacheSize │ │ │ │ │ + * {Number} Number of image elements to keep referenced in this instance's │ │ │ │ │ + * cache for fast reuse. Default is 256. │ │ │ │ │ */ │ │ │ │ │ - fontStyleKeys: [ │ │ │ │ │ - 'antialiasing', 'blockout', 'font', 'fontcolor', 'fontsize', 'fontstyle', │ │ │ │ │ - 'glowing', 'interval', 'outline', 'printmode', 'shadow', 'transparency' │ │ │ │ │ - ], │ │ │ │ │ + cacheSize: 256, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: request │ │ │ │ │ - * A get_image request destined for an ArcIMS server. │ │ │ │ │ + * APIProperty: tilesPerFrame │ │ │ │ │ + * {Number} Number of queued tiles to load per frame (see ). │ │ │ │ │ + * Default is 2. │ │ │ │ │ */ │ │ │ │ │ - request: null, │ │ │ │ │ + tilesPerFrame: 2, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: response │ │ │ │ │ - * A parsed response from an ArcIMS server. │ │ │ │ │ + * APIProperty: frameDelay │ │ │ │ │ + * {Number} Delay between tile loading frames (see ) in │ │ │ │ │ + * milliseconds. Default is 16. │ │ │ │ │ */ │ │ │ │ │ - response: null, │ │ │ │ │ + frameDelay: 16, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.ArcXML │ │ │ │ │ - * Create a new parser/writer for ArcXML. Create an instance of this class │ │ │ │ │ - * to begin authoring a request to an ArcIMS service. This is used │ │ │ │ │ - * primarily by the ArcIMS layer, but could be used to do other wild │ │ │ │ │ - * stuff, like geocoding. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * APIProperty: moveDelay │ │ │ │ │ + * {Number} Delay in milliseconds after a map's move event before loading │ │ │ │ │ + * tiles. Default is 100. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.request = new OpenLayers.Format.ArcXML.Request(); │ │ │ │ │ - this.response = new OpenLayers.Format.ArcXML.Response(); │ │ │ │ │ + moveDelay: 100, │ │ │ │ │ │ │ │ │ │ - if (options) { │ │ │ │ │ - if (options.requesttype == "feature") { │ │ │ │ │ - this.request.get_image = null; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomDelay │ │ │ │ │ + * {Number} Delay in milliseconds after a map's zoomend event before loading │ │ │ │ │ + * tiles. Default is 200. │ │ │ │ │ + */ │ │ │ │ │ + zoomDelay: 200, │ │ │ │ │ │ │ │ │ │ - var qry = this.request.get_feature.query; │ │ │ │ │ - this.addCoordSys(qry.featurecoordsys, options.featureCoordSys); │ │ │ │ │ - this.addCoordSys(qry.filtercoordsys, options.filterCoordSys); │ │ │ │ │ + /** │ │ │ │ │ + * Property: maps │ │ │ │ │ + * {Array()} The maps to manage tiles on. │ │ │ │ │ + */ │ │ │ │ │ + maps: null, │ │ │ │ │ │ │ │ │ │ - if (options.polygon) { │ │ │ │ │ - qry.isspatial = true; │ │ │ │ │ - qry.spatialfilter.polygon = options.polygon; │ │ │ │ │ - } else if (options.envelope) { │ │ │ │ │ - qry.isspatial = true; │ │ │ │ │ - qry.spatialfilter.envelope = { │ │ │ │ │ - minx: 0, │ │ │ │ │ - miny: 0, │ │ │ │ │ - maxx: 0, │ │ │ │ │ - maxy: 0 │ │ │ │ │ - }; │ │ │ │ │ - this.parseEnvelope(qry.spatialfilter.envelope, options.envelope); │ │ │ │ │ - } │ │ │ │ │ - } else if (options.requesttype == "image") { │ │ │ │ │ - this.request.get_feature = null; │ │ │ │ │ + /** │ │ │ │ │ + * Property: tileQueueId │ │ │ │ │ + * {Object} The ids of the loop, keyed by map id. │ │ │ │ │ + */ │ │ │ │ │ + tileQueueId: null, │ │ │ │ │ │ │ │ │ │ - var props = this.request.get_image.properties; │ │ │ │ │ - this.parseEnvelope(props.envelope, options.envelope); │ │ │ │ │ + /** │ │ │ │ │ + * Property: tileQueue │ │ │ │ │ + * {Object(Array())} Tiles queued for drawing, keyed by │ │ │ │ │ + * map id. │ │ │ │ │ + */ │ │ │ │ │ + tileQueue: null, │ │ │ │ │ │ │ │ │ │ - this.addLayers(props.layerlist, options.layers); │ │ │ │ │ - this.addImageSize(props.imagesize, options.tileSize); │ │ │ │ │ - this.addCoordSys(props.featurecoordsys, options.featureCoordSys); │ │ │ │ │ - this.addCoordSys(props.filtercoordsys, options.filterCoordSys); │ │ │ │ │ - } else { │ │ │ │ │ - // if an arcxml object is being created with no request type, it is │ │ │ │ │ - // probably going to consume a response, so do not throw an error if │ │ │ │ │ - // the requesttype is not defined │ │ │ │ │ - this.request = null; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: tileCache │ │ │ │ │ + * {Object} Cached image elements, keyed by URL. │ │ │ │ │ + */ │ │ │ │ │ + tileCache: null, │ │ │ │ │ │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + /** │ │ │ │ │ + * Property: tileCacheIndex │ │ │ │ │ + * {Array(String)} URLs of cached tiles. First entry is the least recently │ │ │ │ │ + * used. │ │ │ │ │ + */ │ │ │ │ │ + tileCacheIndex: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.TileManager │ │ │ │ │ + * Constructor for a new instance. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Configuration for this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.maps = []; │ │ │ │ │ + this.tileQueueId = {}; │ │ │ │ │ + this.tileQueue = {}; │ │ │ │ │ + this.tileCache = {}; │ │ │ │ │ + this.tileCacheIndex = []; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseEnvelope │ │ │ │ │ - * Parse an array of coordinates into an ArcXML envelope structure. │ │ │ │ │ + * Method: addMap │ │ │ │ │ + * Binds this instance to a map │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * env - {Object} An envelope object that will contain the parsed coordinates. │ │ │ │ │ - * arr - {Array(double)} An array of coordinates in the order: [ minx, miny, maxx, maxy ] │ │ │ │ │ + * map - {} │ │ │ │ │ */ │ │ │ │ │ - parseEnvelope: function(env, arr) { │ │ │ │ │ - if (arr && arr.length == 4) { │ │ │ │ │ - env.minx = arr[0]; │ │ │ │ │ - env.miny = arr[1]; │ │ │ │ │ - env.maxx = arr[2]; │ │ │ │ │ - env.maxy = arr[3]; │ │ │ │ │ + addMap: function(map) { │ │ │ │ │ + if (this._destroyed || !OpenLayers.Layer.Grid) { │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ + this.maps.push(map); │ │ │ │ │ + this.tileQueue[map.id] = []; │ │ │ │ │ + for (var i = 0, ii = map.layers.length; i < ii; ++i) { │ │ │ │ │ + this.addLayer({ │ │ │ │ │ + layer: map.layers[i] │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + map.events.on({ │ │ │ │ │ + move: this.move, │ │ │ │ │ + zoomend: this.zoomEnd, │ │ │ │ │ + changelayer: this.changeLayer, │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + preremovelayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: addLayers │ │ │ │ │ - * Add a collection of layers to another collection of layers. Each layer in the list is tuple of │ │ │ │ │ - * { id, visible }. These layer collections represent the │ │ │ │ │ - * /ARCXML/REQUEST/get_image/PROPERTIES/LAYERLIST/LAYERDEF items in ArcXML │ │ │ │ │ - * │ │ │ │ │ - * TODO: Add support for dynamic layer rendering. │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeMap │ │ │ │ │ + * Unbinds this instance from a map │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * ll - {Array({id,visible})} A list of layer definitions. │ │ │ │ │ - * lyrs - {Array({id,visible})} A list of layer definitions. │ │ │ │ │ + * map - {} │ │ │ │ │ */ │ │ │ │ │ - addLayers: function(ll, lyrs) { │ │ │ │ │ - for (var lind = 0, len = lyrs.length; lind < len; lind++) { │ │ │ │ │ - ll.push(lyrs[lind]); │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + if (this._destroyed || !OpenLayers.Layer.Grid) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + window.clearTimeout(this.tileQueueId[map.id]); │ │ │ │ │ + if (map.layers) { │ │ │ │ │ + for (var i = 0, ii = map.layers.length; i < ii; ++i) { │ │ │ │ │ + this.removeLayer({ │ │ │ │ │ + layer: map.layers[i] │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (map.events) { │ │ │ │ │ + map.events.un({ │ │ │ │ │ + move: this.move, │ │ │ │ │ + zoomend: this.zoomEnd, │ │ │ │ │ + changelayer: this.changeLayer, │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + preremovelayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ + delete this.tileQueue[map.id]; │ │ │ │ │ + delete this.tileQueueId[map.id]; │ │ │ │ │ + OpenLayers.Util.removeItem(this.maps, map); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: addImageSize │ │ │ │ │ - * Set the size of the requested image. │ │ │ │ │ + * Method: move │ │ │ │ │ + * Handles the map's move event │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * imsize - {Object} An ArcXML imagesize object. │ │ │ │ │ - * olsize - {} The image size to set. │ │ │ │ │ + * evt - {Object} Listener argument │ │ │ │ │ */ │ │ │ │ │ - addImageSize: function(imsize, olsize) { │ │ │ │ │ - if (olsize !== null) { │ │ │ │ │ - imsize.width = olsize.w; │ │ │ │ │ - imsize.height = olsize.h; │ │ │ │ │ - imsize.printwidth = olsize.w; │ │ │ │ │ - imsize.printheight = olsize.h; │ │ │ │ │ - } │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + this.updateTimeout(evt.object, this.moveDelay, true); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: addCoordSys │ │ │ │ │ - * Add the coordinate system information to an object. The object may be │ │ │ │ │ + * Method: zoomEnd │ │ │ │ │ + * Handles the map's zoomEnd event │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * featOrFilt - {Object} A featurecoordsys or filtercoordsys ArcXML structure. │ │ │ │ │ - * fsys - {String} or {} or {filtercoordsys} or │ │ │ │ │ - * {featurecoordsys} A projection representation. If it's a {String}, │ │ │ │ │ - * the value is assumed to be the SRID. If it's a {OpenLayers.Projection} │ │ │ │ │ - * AND Proj4js is available, the projection number and name are extracted │ │ │ │ │ - * from there. If it's a filter or feature ArcXML structure, it is copied. │ │ │ │ │ + * evt - {Object} Listener argument │ │ │ │ │ */ │ │ │ │ │ - addCoordSys: function(featOrFilt, fsys) { │ │ │ │ │ - if (typeof fsys == "string") { │ │ │ │ │ - featOrFilt.id = parseInt(fsys); │ │ │ │ │ - featOrFilt.string = fsys; │ │ │ │ │ - } │ │ │ │ │ - // is this a proj4js instance? │ │ │ │ │ - else if (typeof fsys == "object" && fsys.proj !== null) { │ │ │ │ │ - featOrFilt.id = fsys.proj.srsProjNumber; │ │ │ │ │ - featOrFilt.string = fsys.proj.srsCode; │ │ │ │ │ - } else { │ │ │ │ │ - featOrFilt = fsys; │ │ │ │ │ - } │ │ │ │ │ + zoomEnd: function(evt) { │ │ │ │ │ + this.updateTimeout(evt.object, this.zoomDelay); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: iserror │ │ │ │ │ - * Check to see if the response from the server was an error. │ │ │ │ │ + * Method: changeLayer │ │ │ │ │ + * Handles the map's changeLayer event │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. If nothing is supplied, │ │ │ │ │ - * the current response is examined. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true if the response was an error. │ │ │ │ │ + * evt - {Object} Listener argument │ │ │ │ │ */ │ │ │ │ │ - iserror: function(data) { │ │ │ │ │ - var ret = null; │ │ │ │ │ - │ │ │ │ │ - if (!data) { │ │ │ │ │ - ret = (this.response.error !== ''); │ │ │ │ │ - } else { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - var errorNodes = data.documentElement.getElementsByTagName("ERROR"); │ │ │ │ │ - ret = (errorNodes !== null && errorNodes.length > 0); │ │ │ │ │ + changeLayer: function(evt) { │ │ │ │ │ + if (evt.property === 'visibility' || evt.property === 'params') { │ │ │ │ │ + this.updateTimeout(evt.object, 0); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return ret; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read data from a string, and return an response. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * Method: addLayer │ │ │ │ │ + * Handles the map's addlayer event │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} An ArcXML response. Note that this response │ │ │ │ │ - * data may change in the future. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} The listener argument │ │ │ │ │ */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var arcNode = null; │ │ │ │ │ - if (data && data.documentElement) { │ │ │ │ │ - if (data.documentElement.nodeName == "ARCXML") { │ │ │ │ │ - arcNode = data.documentElement; │ │ │ │ │ - } else { │ │ │ │ │ - arcNode = data.documentElement.getElementsByTagName("ARCXML")[0]; │ │ │ │ │ + addLayer: function(evt) { │ │ │ │ │ + var layer = evt.layer; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ + layer.events.on({ │ │ │ │ │ + addtile: this.addTile, │ │ │ │ │ + retile: this.clearTileQueue, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + var i, j, tile; │ │ │ │ │ + for (i = layer.grid.length - 1; i >= 0; --i) { │ │ │ │ │ + for (j = layer.grid[i].length - 1; j >= 0; --j) { │ │ │ │ │ + tile = layer.grid[i][j]; │ │ │ │ │ + this.addTile({ │ │ │ │ │ + tile: tile │ │ │ │ │ + }); │ │ │ │ │ + if (tile.url && !tile.imgDiv) { │ │ │ │ │ + this.manageTileCache({ │ │ │ │ │ + object: tile │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // in Safari, arcNode will be there but will have a child named │ │ │ │ │ - // parsererror │ │ │ │ │ - if (!arcNode || arcNode.firstChild.nodeName === 'parsererror') { │ │ │ │ │ - var error, source; │ │ │ │ │ - try { │ │ │ │ │ - error = data.firstChild.nodeValue; │ │ │ │ │ - source = data.firstChild.childNodes[1].firstChild.nodeValue; │ │ │ │ │ - } catch (err) { │ │ │ │ │ - // pass │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeLayer │ │ │ │ │ + * Handles the map's preremovelayer event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} The listener argument │ │ │ │ │ + */ │ │ │ │ │ + removeLayer: function(evt) { │ │ │ │ │ + var layer = evt.layer; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ + this.clearTileQueue({ │ │ │ │ │ + object: layer │ │ │ │ │ + }); │ │ │ │ │ + if (layer.events) { │ │ │ │ │ + layer.events.un({ │ │ │ │ │ + addtile: this.addTile, │ │ │ │ │ + retile: this.clearTileQueue, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + if (layer.grid) { │ │ │ │ │ + var i, j, tile; │ │ │ │ │ + for (i = layer.grid.length - 1; i >= 0; --i) { │ │ │ │ │ + for (j = layer.grid[i].length - 1; j >= 0; --j) { │ │ │ │ │ + tile = layer.grid[i][j]; │ │ │ │ │ + this.unloadTile({ │ │ │ │ │ + object: tile │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - throw { │ │ │ │ │ - message: "Error parsing the ArcXML request", │ │ │ │ │ - error: error, │ │ │ │ │ - source: source │ │ │ │ │ - }; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var response = this.parseResponse(arcNode); │ │ │ │ │ - return response; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Generate an ArcXml document string for sending to an ArcIMS server. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string representing the ArcXML document request. │ │ │ │ │ + * Method: updateTimeout │ │ │ │ │ + * Applies the or to the loop, │ │ │ │ │ + * and schedules more queue processing after if there are still │ │ │ │ │ + * tiles in the queue. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {} The map to update the timeout for │ │ │ │ │ + * delay - {Number} The delay to apply │ │ │ │ │ + * nice - {Boolean} If true, the timeout function will only be created if │ │ │ │ │ + * the tilequeue is not empty. This is used by the move handler to │ │ │ │ │ + * avoid impacts on dragging performance. For other events, the tile │ │ │ │ │ + * queue may not be populated yet, so we need to set the timer │ │ │ │ │ + * regardless of the queue size. │ │ │ │ │ */ │ │ │ │ │ - write: function(request) { │ │ │ │ │ - if (!request) { │ │ │ │ │ - request = this.request; │ │ │ │ │ + updateTimeout: function(map, delay, nice) { │ │ │ │ │ + window.clearTimeout(this.tileQueueId[map.id]); │ │ │ │ │ + var tileQueue = this.tileQueue[map.id]; │ │ │ │ │ + if (!nice || tileQueue.length) { │ │ │ │ │ + this.tileQueueId[map.id] = window.setTimeout( │ │ │ │ │ + OpenLayers.Function.bind(function() { │ │ │ │ │ + this.drawTilesFromQueue(map); │ │ │ │ │ + if (tileQueue.length) { │ │ │ │ │ + this.updateTimeout(map, this.frameDelay); │ │ │ │ │ + } │ │ │ │ │ + }, this), delay │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - var root = this.createElementNS("", "ARCXML"); │ │ │ │ │ - root.setAttribute("version", "1.1"); │ │ │ │ │ - │ │ │ │ │ - var reqElem = this.createElementNS("", "REQUEST"); │ │ │ │ │ - │ │ │ │ │ - if (request.get_image != null) { │ │ │ │ │ - var getElem = this.createElementNS("", "GET_IMAGE"); │ │ │ │ │ - reqElem.appendChild(getElem); │ │ │ │ │ - │ │ │ │ │ - var propElem = this.createElementNS("", "PROPERTIES"); │ │ │ │ │ - getElem.appendChild(propElem); │ │ │ │ │ - │ │ │ │ │ - var props = request.get_image.properties; │ │ │ │ │ - if (props.featurecoordsys != null) { │ │ │ │ │ - var feat = this.createElementNS("", "FEATURECOORDSYS"); │ │ │ │ │ - propElem.appendChild(feat); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (props.featurecoordsys.id === 0) { │ │ │ │ │ - feat.setAttribute("string", props.featurecoordsys['string']); │ │ │ │ │ - } else { │ │ │ │ │ - feat.setAttribute("id", props.featurecoordsys.id); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: addTile │ │ │ │ │ + * Listener for the layer's addtile event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} The listener argument │ │ │ │ │ + */ │ │ │ │ │ + addTile: function(evt) { │ │ │ │ │ + if (evt.tile instanceof OpenLayers.Tile.Image) { │ │ │ │ │ + evt.tile.events.on({ │ │ │ │ │ + beforedraw: this.queueTileDraw, │ │ │ │ │ + beforeload: this.manageTileCache, │ │ │ │ │ + loadend: this.addToCache, │ │ │ │ │ + unload: this.unloadTile, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + // Layer has the wrong tile type, so don't handle it any longer │ │ │ │ │ + this.removeLayer({ │ │ │ │ │ + layer: evt.tile.layer │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (props.filtercoordsys != null) { │ │ │ │ │ - var filt = this.createElementNS("", "FILTERCOORDSYS"); │ │ │ │ │ - propElem.appendChild(filt); │ │ │ │ │ + /** │ │ │ │ │ + * Method: unloadTile │ │ │ │ │ + * Listener for the tile's unload event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} The listener argument │ │ │ │ │ + */ │ │ │ │ │ + unloadTile: function(evt) { │ │ │ │ │ + var tile = evt.object; │ │ │ │ │ + tile.events.un({ │ │ │ │ │ + beforedraw: this.queueTileDraw, │ │ │ │ │ + beforeload: this.manageTileCache, │ │ │ │ │ + loadend: this.addToCache, │ │ │ │ │ + unload: this.unloadTile, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Util.removeItem(this.tileQueue[tile.layer.map.id], tile); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (props.filtercoordsys.id === 0) { │ │ │ │ │ - filt.setAttribute("string", props.filtercoordsys.string); │ │ │ │ │ - } else { │ │ │ │ │ - filt.setAttribute("id", props.filtercoordsys.id); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: queueTileDraw │ │ │ │ │ + * Adds a tile to the queue that will draw it. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} Listener argument of the tile's beforedraw event │ │ │ │ │ + */ │ │ │ │ │ + queueTileDraw: function(evt) { │ │ │ │ │ + var tile = evt.object; │ │ │ │ │ + var queued = false; │ │ │ │ │ + var layer = tile.layer; │ │ │ │ │ + var url = layer.getURL(tile.bounds); │ │ │ │ │ + var img = this.tileCache[url]; │ │ │ │ │ + if (img && img.className !== 'olTileImage') { │ │ │ │ │ + // cached image no longer valid, e.g. because we're olTileReplacing │ │ │ │ │ + delete this.tileCache[url]; │ │ │ │ │ + OpenLayers.Util.removeItem(this.tileCacheIndex, url); │ │ │ │ │ + img = null; │ │ │ │ │ + } │ │ │ │ │ + // queue only if image with same url not cached already │ │ │ │ │ + if (layer.url && (layer.async || !img)) { │ │ │ │ │ + // add to queue only if not in queue already │ │ │ │ │ + var tileQueue = this.tileQueue[layer.map.id]; │ │ │ │ │ + if (!~OpenLayers.Util.indexOf(tileQueue, tile)) { │ │ │ │ │ + tileQueue.push(tile); │ │ │ │ │ } │ │ │ │ │ + queued = true; │ │ │ │ │ + } │ │ │ │ │ + return !queued; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (props.envelope != null) { │ │ │ │ │ - var env = this.createElementNS("", "ENVELOPE"); │ │ │ │ │ - propElem.appendChild(env); │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawTilesFromQueue │ │ │ │ │ + * Draws tiles from the tileQueue, and unqueues the tiles │ │ │ │ │ + */ │ │ │ │ │ + drawTilesFromQueue: function(map) { │ │ │ │ │ + var tileQueue = this.tileQueue[map.id]; │ │ │ │ │ + var limit = this.tilesPerFrame; │ │ │ │ │ + var animating = map.zoomTween && map.zoomTween.playing; │ │ │ │ │ + while (!animating && tileQueue.length && limit) { │ │ │ │ │ + tileQueue.shift().draw(true); │ │ │ │ │ + --limit; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - env.setAttribute("minx", props.envelope.minx); │ │ │ │ │ - env.setAttribute("miny", props.envelope.miny); │ │ │ │ │ - env.setAttribute("maxx", props.envelope.maxx); │ │ │ │ │ - env.setAttribute("maxy", props.envelope.maxy); │ │ │ │ │ + /** │ │ │ │ │ + * Method: manageTileCache │ │ │ │ │ + * Adds, updates, removes and fetches cache entries. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} Listener argument of the tile's beforeload event │ │ │ │ │ + */ │ │ │ │ │ + manageTileCache: function(evt) { │ │ │ │ │ + var tile = evt.object; │ │ │ │ │ + var img = this.tileCache[tile.url]; │ │ │ │ │ + if (img) { │ │ │ │ │ + // if image is on its layer's backbuffer, remove it from backbuffer │ │ │ │ │ + if (img.parentNode && │ │ │ │ │ + OpenLayers.Element.hasClass(img.parentNode, 'olBackBuffer')) { │ │ │ │ │ + img.parentNode.removeChild(img); │ │ │ │ │ + img.id = null; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var imagesz = this.createElementNS("", "IMAGESIZE"); │ │ │ │ │ - propElem.appendChild(imagesz); │ │ │ │ │ - │ │ │ │ │ - imagesz.setAttribute("height", props.imagesize.height); │ │ │ │ │ - imagesz.setAttribute("width", props.imagesize.width); │ │ │ │ │ - │ │ │ │ │ - if (props.imagesize.height != props.imagesize.printheight || │ │ │ │ │ - props.imagesize.width != props.imagesize.printwidth) { │ │ │ │ │ - imagesz.setAttribute("printheight", props.imagesize.printheight); │ │ │ │ │ - imagesz.setArrtibute("printwidth", props.imagesize.printwidth); │ │ │ │ │ + // only use image from cache if it is not on a layer already │ │ │ │ │ + if (!img.parentNode) { │ │ │ │ │ + img.style.visibility = 'hidden'; │ │ │ │ │ + img.style.opacity = 0; │ │ │ │ │ + tile.setImage(img); │ │ │ │ │ + // LRU - move tile to the end of the array to mark it as the most │ │ │ │ │ + // recently used │ │ │ │ │ + OpenLayers.Util.removeItem(this.tileCacheIndex, tile.url); │ │ │ │ │ + this.tileCacheIndex.push(tile.url); │ │ │ │ │ } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (props.background != null) { │ │ │ │ │ - var backgrnd = this.createElementNS("", "BACKGROUND"); │ │ │ │ │ - propElem.appendChild(backgrnd); │ │ │ │ │ - │ │ │ │ │ - backgrnd.setAttribute("color", │ │ │ │ │ - props.background.color.r + "," + │ │ │ │ │ - props.background.color.g + "," + │ │ │ │ │ - props.background.color.b); │ │ │ │ │ - │ │ │ │ │ - if (props.background.transcolor !== null) { │ │ │ │ │ - backgrnd.setAttribute("transcolor", │ │ │ │ │ - props.background.transcolor.r + "," + │ │ │ │ │ - props.background.transcolor.g + "," + │ │ │ │ │ - props.background.transcolor.b); │ │ │ │ │ + /** │ │ │ │ │ + * Method: addToCache │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} Listener argument for the tile's loadend event │ │ │ │ │ + */ │ │ │ │ │ + addToCache: function(evt) { │ │ │ │ │ + var tile = evt.object; │ │ │ │ │ + if (!this.tileCache[tile.url]) { │ │ │ │ │ + if (!OpenLayers.Element.hasClass(tile.imgDiv, 'olImageLoadError')) { │ │ │ │ │ + if (this.tileCacheIndex.length >= this.cacheSize) { │ │ │ │ │ + delete this.tileCache[this.tileCacheIndex[0]]; │ │ │ │ │ + this.tileCacheIndex.shift(); │ │ │ │ │ } │ │ │ │ │ + this.tileCache[tile.url] = tile.imgDiv; │ │ │ │ │ + this.tileCacheIndex.push(tile.url); │ │ │ │ │ } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (props.layerlist != null && props.layerlist.length > 0) { │ │ │ │ │ - var layerlst = this.createElementNS("", "LAYERLIST"); │ │ │ │ │ - propElem.appendChild(layerlst); │ │ │ │ │ - │ │ │ │ │ - for (var ld = 0; ld < props.layerlist.length; ld++) { │ │ │ │ │ - var ldef = this.createElementNS("", "LAYERDEF"); │ │ │ │ │ - layerlst.appendChild(ldef); │ │ │ │ │ - │ │ │ │ │ - ldef.setAttribute("id", props.layerlist[ld].id); │ │ │ │ │ - ldef.setAttribute("visible", props.layerlist[ld].visible); │ │ │ │ │ - │ │ │ │ │ - if (typeof props.layerlist[ld].query == "object") { │ │ │ │ │ - var query = props.layerlist[ld].query; │ │ │ │ │ - │ │ │ │ │ - if (query.where.length < 0) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var queryElem = null; │ │ │ │ │ - if (typeof query.spatialfilter == "boolean" && query.spatialfilter) { │ │ │ │ │ - // handle spatial filter madness │ │ │ │ │ - queryElem = this.createElementNS("", "SPATIALQUERY"); │ │ │ │ │ - } else { │ │ │ │ │ - queryElem = this.createElementNS("", "QUERY"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - queryElem.setAttribute("where", query.where); │ │ │ │ │ + /** │ │ │ │ │ + * Method: clearTileQueue │ │ │ │ │ + * Clears the tile queue from tiles of a specific layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} Listener argument of the layer's retile event │ │ │ │ │ + */ │ │ │ │ │ + clearTileQueue: function(evt) { │ │ │ │ │ + var layer = evt.object; │ │ │ │ │ + var tileQueue = this.tileQueue[layer.map.id]; │ │ │ │ │ + for (var i = tileQueue.length - 1; i >= 0; --i) { │ │ │ │ │ + if (tileQueue[i].layer === layer) { │ │ │ │ │ + tileQueue.splice(i, 1); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (typeof query.accuracy == "number" && query.accuracy > 0) { │ │ │ │ │ - queryElem.setAttribute("accuracy", query.accuracy); │ │ │ │ │ - } │ │ │ │ │ - if (typeof query.featurelimit == "number" && query.featurelimit < 2000) { │ │ │ │ │ - queryElem.setAttribute("featurelimit", query.featurelimit); │ │ │ │ │ - } │ │ │ │ │ - if (typeof query.subfields == "string" && query.subfields != "#ALL#") { │ │ │ │ │ - queryElem.setAttribute("subfields", query.subfields); │ │ │ │ │ - } │ │ │ │ │ - if (typeof query.joinexpression == "string" && query.joinexpression.length > 0) { │ │ │ │ │ - queryElem.setAttribute("joinexpression", query.joinexpression); │ │ │ │ │ - } │ │ │ │ │ - if (typeof query.jointables == "string" && query.jointables.length > 0) { │ │ │ │ │ - queryElem.setAttribute("jointables", query.jointables); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i = this.maps.length - 1; i >= 0; --i) { │ │ │ │ │ + this.removeMap(this.maps[i]); │ │ │ │ │ + } │ │ │ │ │ + this.maps = null; │ │ │ │ │ + this.tileQueue = null; │ │ │ │ │ + this.tileQueueId = null; │ │ │ │ │ + this.tileCache = null; │ │ │ │ │ + this.tileCacheIndex = null; │ │ │ │ │ + this._destroyed = true; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - ldef.appendChild(queryElem); │ │ │ │ │ - } │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Rule.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - if (typeof props.layerlist[ld].renderer == "object") { │ │ │ │ │ - this.addRenderer(ldef, props.layerlist[ld].renderer); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else if (request.get_feature != null) { │ │ │ │ │ - var getElem = this.createElementNS("", "GET_FEATURES"); │ │ │ │ │ - getElem.setAttribute("outputmode", "newxml"); │ │ │ │ │ - getElem.setAttribute("checkesc", "true"); │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - if (request.get_feature.geometry) { │ │ │ │ │ - getElem.setAttribute("geometry", request.get_feature.geometry); │ │ │ │ │ - } else { │ │ │ │ │ - getElem.setAttribute("geometry", "false"); │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - if (request.get_feature.compact) { │ │ │ │ │ - getElem.setAttribute("compact", request.get_feature.compact); │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + * @requires OpenLayers/Style.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (request.get_feature.featurelimit == "number") { │ │ │ │ │ - getElem.setAttribute("featurelimit", request.get_feature.featurelimit); │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Rule │ │ │ │ │ + * This class represents an SLD Rule, as being used for rule-based SLD styling. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Rule = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ - getElem.setAttribute("globalenvelope", "true"); │ │ │ │ │ - reqElem.appendChild(getElem); │ │ │ │ │ + /** │ │ │ │ │ + * Property: id │ │ │ │ │ + * {String} A unique id for this session. │ │ │ │ │ + */ │ │ │ │ │ + id: null, │ │ │ │ │ │ │ │ │ │ - if (request.get_feature.layer != null && request.get_feature.layer.length > 0) { │ │ │ │ │ - var lyrElem = this.createElementNS("", "LAYER"); │ │ │ │ │ - lyrElem.setAttribute("id", request.get_feature.layer); │ │ │ │ │ - getElem.appendChild(lyrElem); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: name │ │ │ │ │ + * {String} name of this rule │ │ │ │ │ + */ │ │ │ │ │ + name: null, │ │ │ │ │ │ │ │ │ │ - var fquery = request.get_feature.query; │ │ │ │ │ - if (fquery != null) { │ │ │ │ │ - var qElem = null; │ │ │ │ │ - if (fquery.isspatial) { │ │ │ │ │ - qElem = this.createElementNS("", "SPATIALQUERY"); │ │ │ │ │ - } else { │ │ │ │ │ - qElem = this.createElementNS("", "QUERY"); │ │ │ │ │ - } │ │ │ │ │ - getElem.appendChild(qElem); │ │ │ │ │ + /** │ │ │ │ │ + * Property: title │ │ │ │ │ + * {String} Title of this rule (set if included in SLD) │ │ │ │ │ + */ │ │ │ │ │ + title: null, │ │ │ │ │ │ │ │ │ │ - if (typeof fquery.accuracy == "number") { │ │ │ │ │ - qElem.setAttribute("accuracy", fquery.accuracy); │ │ │ │ │ - } │ │ │ │ │ - //qElem.setAttribute("featurelimit", "5"); │ │ │ │ │ + /** │ │ │ │ │ + * Property: description │ │ │ │ │ + * {String} Description of this rule (set if abstract is included in SLD) │ │ │ │ │ + */ │ │ │ │ │ + description: null, │ │ │ │ │ │ │ │ │ │ - if (fquery.featurecoordsys != null) { │ │ │ │ │ - var fcsElem1 = this.createElementNS("", "FEATURECOORDSYS"); │ │ │ │ │ + /** │ │ │ │ │ + * Property: context │ │ │ │ │ + * {Object} An optional object with properties that the rule should be │ │ │ │ │ + * evaluated against. If no context is specified, feature.attributes will │ │ │ │ │ + * be used. │ │ │ │ │ + */ │ │ │ │ │ + context: null, │ │ │ │ │ │ │ │ │ │ - if (fquery.featurecoordsys.id == 0) { │ │ │ │ │ - fcsElem1.setAttribute("string", fquery.featurecoordsys.string); │ │ │ │ │ - } else { │ │ │ │ │ - fcsElem1.setAttribute("id", fquery.featurecoordsys.id); │ │ │ │ │ - } │ │ │ │ │ - qElem.appendChild(fcsElem1); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: filter │ │ │ │ │ + * {} Optional filter for the rule. │ │ │ │ │ + */ │ │ │ │ │ + filter: null, │ │ │ │ │ │ │ │ │ │ - if (fquery.filtercoordsys != null) { │ │ │ │ │ - var fcsElem2 = this.createElementNS("", "FILTERCOORDSYS"); │ │ │ │ │ + /** │ │ │ │ │ + * Property: elseFilter │ │ │ │ │ + * {Boolean} Determines whether this rule is only to be applied only if │ │ │ │ │ + * no other rules match (ElseFilter according to the SLD specification). │ │ │ │ │ + * Default is false. For instances of OpenLayers.Rule, if elseFilter is │ │ │ │ │ + * false, the rule will always apply. For subclasses, the else property is │ │ │ │ │ + * ignored. │ │ │ │ │ + */ │ │ │ │ │ + elseFilter: false, │ │ │ │ │ │ │ │ │ │ - if (fquery.filtercoordsys.id === 0) { │ │ │ │ │ - fcsElem2.setAttribute("string", fquery.filtercoordsys.string); │ │ │ │ │ - } else { │ │ │ │ │ - fcsElem2.setAttribute("id", fquery.filtercoordsys.id); │ │ │ │ │ - } │ │ │ │ │ - qElem.appendChild(fcsElem2); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: symbolizer │ │ │ │ │ + * {Object} Symbolizer or hash of symbolizers for this rule. If hash of │ │ │ │ │ + * symbolizers, keys are one or more of ["Point", "Line", "Polygon"]. The │ │ │ │ │ + * latter if useful if it is required to style e.g. vertices of a line │ │ │ │ │ + * with a point symbolizer. Note, however, that this is not implemented │ │ │ │ │ + * yet in OpenLayers, but it is the way how symbolizers are defined in │ │ │ │ │ + * SLD. │ │ │ │ │ + */ │ │ │ │ │ + symbolizer: null, │ │ │ │ │ │ │ │ │ │ - if (fquery.buffer > 0) { │ │ │ │ │ - var bufElem = this.createElementNS("", "BUFFER"); │ │ │ │ │ - bufElem.setAttribute("distance", fquery.buffer); │ │ │ │ │ - qElem.appendChild(bufElem); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: symbolizers │ │ │ │ │ + * {Array} Collection of symbolizers associated with this rule. If │ │ │ │ │ + * provided at construction, the symbolizers array has precedence │ │ │ │ │ + * over the deprecated symbolizer property. Note that multiple │ │ │ │ │ + * symbolizers are not currently supported by the vector renderers. │ │ │ │ │ + * Rules with multiple symbolizers are currently only useful for │ │ │ │ │ + * maintaining elements in an SLD document. │ │ │ │ │ + */ │ │ │ │ │ + symbolizers: null, │ │ │ │ │ │ │ │ │ │ - if (fquery.isspatial) { │ │ │ │ │ - var spfElem = this.createElementNS("", "SPATIALFILTER"); │ │ │ │ │ - spfElem.setAttribute("relation", fquery.spatialfilter.relation); │ │ │ │ │ - qElem.appendChild(spfElem); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: minScaleDenominator │ │ │ │ │ + * {Number} or {String} minimum scale at which to draw the feature. │ │ │ │ │ + * In the case of a String, this can be a combination of text and │ │ │ │ │ + * propertyNames in the form "literal ${propertyName}" │ │ │ │ │ + */ │ │ │ │ │ + minScaleDenominator: null, │ │ │ │ │ │ │ │ │ │ - if (fquery.spatialfilter.envelope) { │ │ │ │ │ - var envElem = this.createElementNS("", "ENVELOPE"); │ │ │ │ │ - envElem.setAttribute("minx", fquery.spatialfilter.envelope.minx); │ │ │ │ │ - envElem.setAttribute("miny", fquery.spatialfilter.envelope.miny); │ │ │ │ │ - envElem.setAttribute("maxx", fquery.spatialfilter.envelope.maxx); │ │ │ │ │ - envElem.setAttribute("maxy", fquery.spatialfilter.envelope.maxy); │ │ │ │ │ - spfElem.appendChild(envElem); │ │ │ │ │ - } else if (typeof fquery.spatialfilter.polygon == "object") { │ │ │ │ │ - spfElem.appendChild(this.writePolygonGeometry(fquery.spatialfilter.polygon)); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxScaleDenominator │ │ │ │ │ + * {Number} or {String} maximum scale at which to draw the feature. │ │ │ │ │ + * In the case of a String, this can be a combination of text and │ │ │ │ │ + * propertyNames in the form "literal ${propertyName}" │ │ │ │ │ + */ │ │ │ │ │ + maxScaleDenominator: null, │ │ │ │ │ │ │ │ │ │ - if (fquery.where != null && fquery.where.length > 0) { │ │ │ │ │ - qElem.setAttribute("where", fquery.where); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Rule │ │ │ │ │ + * Creates a Rule. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object with properties to set on the │ │ │ │ │ + * rule │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.symbolizer = {}; │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + if (this.symbolizers) { │ │ │ │ │ + delete this.symbolizer; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - root.appendChild(reqElem); │ │ │ │ │ - │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [root]); │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ - addGroupRenderer: function(ldef, toprenderer) { │ │ │ │ │ - var topRelem = this.createElementNS("", "GROUPRENDERER"); │ │ │ │ │ - ldef.appendChild(topRelem); │ │ │ │ │ - │ │ │ │ │ - for (var rind = 0; rind < toprenderer.length; rind++) { │ │ │ │ │ - var renderer = toprenderer[rind]; │ │ │ │ │ - this.addRenderer(topRelem, renderer); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * nullify references to prevent circular references and memory leaks │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i in this.symbolizer) { │ │ │ │ │ + this.symbolizer[i] = null; │ │ │ │ │ } │ │ │ │ │ + this.symbolizer = null; │ │ │ │ │ + delete this.symbolizers; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: evaluate │ │ │ │ │ + * evaluates this rule for a specific feature │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {} feature to apply the rule to. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the rule applies, false if it does not. │ │ │ │ │ + * This rule is the default rule and always returns true. │ │ │ │ │ + */ │ │ │ │ │ + evaluate: function(feature) { │ │ │ │ │ + var context = this.getContext(feature); │ │ │ │ │ + var applies = true; │ │ │ │ │ │ │ │ │ │ - addRenderer: function(topRelem, renderer) { │ │ │ │ │ - if (OpenLayers.Util.isArray(renderer)) { │ │ │ │ │ - this.addGroupRenderer(topRelem, renderer); │ │ │ │ │ - } else { │ │ │ │ │ - var renderElem = this.createElementNS("", renderer.type.toUpperCase() + "RENDERER"); │ │ │ │ │ - topRelem.appendChild(renderElem); │ │ │ │ │ - │ │ │ │ │ - if (renderElem.tagName == "VALUEMAPRENDERER") { │ │ │ │ │ - this.addValueMapRenderer(renderElem, renderer); │ │ │ │ │ - } else if (renderElem.tagName == "VALUEMAPLABELRENDERER") { │ │ │ │ │ - this.addValueMapLabelRenderer(renderElem, renderer); │ │ │ │ │ - } else if (renderElem.tagName == "SIMPLELABELRENDERER") { │ │ │ │ │ - this.addSimpleLabelRenderer(renderElem, renderer); │ │ │ │ │ - } else if (renderElem.tagName == "SCALEDEPENDENTRENDERER") { │ │ │ │ │ - this.addScaleDependentRenderer(renderElem, renderer); │ │ │ │ │ - } │ │ │ │ │ + if (this.minScaleDenominator || this.maxScaleDenominator) { │ │ │ │ │ + var scale = feature.layer.map.getScale(); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ │ │ │ │ │ - addScaleDependentRenderer: function(renderElem, renderer) { │ │ │ │ │ - if (typeof renderer.lower == "string" || typeof renderer.lower == "number") { │ │ │ │ │ - renderElem.setAttribute("lower", renderer.lower); │ │ │ │ │ + // check if within minScale/maxScale bounds │ │ │ │ │ + if (this.minScaleDenominator) { │ │ │ │ │ + applies = scale >= OpenLayers.Style.createLiteral( │ │ │ │ │ + this.minScaleDenominator, context); │ │ │ │ │ } │ │ │ │ │ - if (typeof renderer.upper == "string" || typeof renderer.upper == "number") { │ │ │ │ │ - renderElem.setAttribute("upper", renderer.upper); │ │ │ │ │ + if (applies && this.maxScaleDenominator) { │ │ │ │ │ + applies = scale < OpenLayers.Style.createLiteral( │ │ │ │ │ + this.maxScaleDenominator, context); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - this.addRenderer(renderElem, renderer.renderer); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - addValueMapLabelRenderer: function(renderElem, renderer) { │ │ │ │ │ - renderElem.setAttribute("lookupfield", renderer.lookupfield); │ │ │ │ │ - renderElem.setAttribute("labelfield", renderer.labelfield); │ │ │ │ │ - │ │ │ │ │ - if (typeof renderer.exacts == "object") { │ │ │ │ │ - for (var ext = 0, extlen = renderer.exacts.length; ext < extlen; ext++) { │ │ │ │ │ - var exact = renderer.exacts[ext]; │ │ │ │ │ - │ │ │ │ │ - var eelem = this.createElementNS("", "EXACT"); │ │ │ │ │ - │ │ │ │ │ - if (typeof exact.value == "string") { │ │ │ │ │ - eelem.setAttribute("value", exact.value); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.label == "string") { │ │ │ │ │ - eelem.setAttribute("label", exact.label); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.method == "string") { │ │ │ │ │ - eelem.setAttribute("method", exact.method); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - renderElem.appendChild(eelem); │ │ │ │ │ - │ │ │ │ │ - if (typeof exact.symbol == "object") { │ │ │ │ │ - var selem = null; │ │ │ │ │ - │ │ │ │ │ - if (exact.symbol.type == "text") { │ │ │ │ │ - selem = this.createElementNS("", "TEXTSYMBOL"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (selem != null) { │ │ │ │ │ - var keys = this.fontStyleKeys; │ │ │ │ │ - for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ - var key = keys[i]; │ │ │ │ │ - if (exact.symbol[key]) { │ │ │ │ │ - selem.setAttribute(key, exact.symbol[key]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - eelem.appendChild(selem); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } // for each exact │ │ │ │ │ + // check if optional filter applies │ │ │ │ │ + if (applies && this.filter) { │ │ │ │ │ + // feature id filters get the feature, others get the context │ │ │ │ │ + if (this.filter.CLASS_NAME == "OpenLayers.Filter.FeatureId") { │ │ │ │ │ + applies = this.filter.evaluate(feature); │ │ │ │ │ + } else { │ │ │ │ │ + applies = this.filter.evaluate(context); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - addValueMapRenderer: function(renderElem, renderer) { │ │ │ │ │ - renderElem.setAttribute("lookupfield", renderer.lookupfield); │ │ │ │ │ - │ │ │ │ │ - if (typeof renderer.ranges == "object") { │ │ │ │ │ - for (var rng = 0, rnglen = renderer.ranges.length; rng < rnglen; rng++) { │ │ │ │ │ - var range = renderer.ranges[rng]; │ │ │ │ │ - │ │ │ │ │ - var relem = this.createElementNS("", "RANGE"); │ │ │ │ │ - relem.setAttribute("lower", range.lower); │ │ │ │ │ - relem.setAttribute("upper", range.upper); │ │ │ │ │ - │ │ │ │ │ - renderElem.appendChild(relem); │ │ │ │ │ - │ │ │ │ │ - if (typeof range.symbol == "object") { │ │ │ │ │ - var selem = null; │ │ │ │ │ - │ │ │ │ │ - if (range.symbol.type == "simplepolygon") { │ │ │ │ │ - selem = this.createElementNS("", "SIMPLEPOLYGONSYMBOL"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (selem != null) { │ │ │ │ │ - if (typeof range.symbol.boundarycolor == "string") { │ │ │ │ │ - selem.setAttribute("boundarycolor", range.symbol.boundarycolor); │ │ │ │ │ - } │ │ │ │ │ - if (typeof range.symbol.fillcolor == "string") { │ │ │ │ │ - selem.setAttribute("fillcolor", range.symbol.fillcolor); │ │ │ │ │ - } │ │ │ │ │ - if (typeof range.symbol.filltransparency == "number") { │ │ │ │ │ - selem.setAttribute("filltransparency", range.symbol.filltransparency); │ │ │ │ │ - } │ │ │ │ │ - relem.appendChild(selem); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } // for each range │ │ │ │ │ - } else if (typeof renderer.exacts == "object") { │ │ │ │ │ - for (var ext = 0, extlen = renderer.exacts.length; ext < extlen; ext++) { │ │ │ │ │ - var exact = renderer.exacts[ext]; │ │ │ │ │ - │ │ │ │ │ - var eelem = this.createElementNS("", "EXACT"); │ │ │ │ │ - if (typeof exact.value == "string") { │ │ │ │ │ - eelem.setAttribute("value", exact.value); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.label == "string") { │ │ │ │ │ - eelem.setAttribute("label", exact.label); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.method == "string") { │ │ │ │ │ - eelem.setAttribute("method", exact.method); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - renderElem.appendChild(eelem); │ │ │ │ │ - │ │ │ │ │ - if (typeof exact.symbol == "object") { │ │ │ │ │ - var selem = null; │ │ │ │ │ - │ │ │ │ │ - if (exact.symbol.type == "simplemarker") { │ │ │ │ │ - selem = this.createElementNS("", "SIMPLEMARKERSYMBOL"); │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - if (selem != null) { │ │ │ │ │ - if (typeof exact.symbol.antialiasing == "string") { │ │ │ │ │ - selem.setAttribute("antialiasing", exact.symbol.antialiasing); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.color == "string") { │ │ │ │ │ - selem.setAttribute("color", exact.symbol.color); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.outline == "string") { │ │ │ │ │ - selem.setAttribute("outline", exact.symbol.outline); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.overlap == "string") { │ │ │ │ │ - selem.setAttribute("overlap", exact.symbol.overlap); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.shadow == "string") { │ │ │ │ │ - selem.setAttribute("shadow", exact.symbol.shadow); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.transparency == "number") { │ │ │ │ │ - selem.setAttribute("transparency", exact.symbol.transparency); │ │ │ │ │ - } │ │ │ │ │ - //if (typeof exact.symbol.type == "string") │ │ │ │ │ - // selem.setAttribute("type", exact.symbol.type); │ │ │ │ │ - if (typeof exact.symbol.usecentroid == "string") { │ │ │ │ │ - selem.setAttribute("usecentroid", exact.symbol.usecentroid); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.width == "number") { │ │ │ │ │ - selem.setAttribute("width", exact.symbol.width); │ │ │ │ │ - } │ │ │ │ │ + return applies; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - eelem.appendChild(selem); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } // for each exact │ │ │ │ │ + /** │ │ │ │ │ + * Method: getContext │ │ │ │ │ + * Gets the context for evaluating this rule │ │ │ │ │ + * │ │ │ │ │ + * Paramters: │ │ │ │ │ + * feature - {} feature to take the context from if │ │ │ │ │ + * none is specified. │ │ │ │ │ + */ │ │ │ │ │ + getContext: function(feature) { │ │ │ │ │ + var context = this.context; │ │ │ │ │ + if (!context) { │ │ │ │ │ + context = feature.attributes || feature.data; │ │ │ │ │ + } │ │ │ │ │ + if (typeof this.context == "function") { │ │ │ │ │ + context = this.context(feature); │ │ │ │ │ } │ │ │ │ │ + return context; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ - addSimpleLabelRenderer: function(renderElem, renderer) { │ │ │ │ │ - renderElem.setAttribute("field", renderer.field); │ │ │ │ │ - var keys = ['featureweight', 'howmanylabels', 'labelbufferratio', │ │ │ │ │ - 'labelpriorities', 'labelweight', 'linelabelposition', │ │ │ │ │ - 'rotationalangles' │ │ │ │ │ - ]; │ │ │ │ │ - for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ - var key = keys[i]; │ │ │ │ │ - if (renderer[key]) { │ │ │ │ │ - renderElem.setAttribute(key, renderer[key]); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Clones this rule. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} Clone of this rule. │ │ │ │ │ + */ │ │ │ │ │ + clone: function() { │ │ │ │ │ + var options = OpenLayers.Util.extend({}, this); │ │ │ │ │ + if (this.symbolizers) { │ │ │ │ │ + // clone symbolizers │ │ │ │ │ + var len = this.symbolizers.length; │ │ │ │ │ + options.symbolizers = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + options.symbolizers[i] = this.symbolizers[i].clone(); │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (renderer.symbol.type == "text") { │ │ │ │ │ - var symbol = renderer.symbol; │ │ │ │ │ - var selem = this.createElementNS("", "TEXTSYMBOL"); │ │ │ │ │ - renderElem.appendChild(selem); │ │ │ │ │ - │ │ │ │ │ - var keys = this.fontStyleKeys; │ │ │ │ │ - for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ - var key = keys[i]; │ │ │ │ │ - if (symbol[key]) { │ │ │ │ │ - selem.setAttribute(key, renderer[key]); │ │ │ │ │ + } else { │ │ │ │ │ + // clone symbolizer │ │ │ │ │ + options.symbolizer = {}; │ │ │ │ │ + var value, type; │ │ │ │ │ + for (var key in this.symbolizer) { │ │ │ │ │ + value = this.symbolizer[key]; │ │ │ │ │ + type = typeof value; │ │ │ │ │ + if (type === "object") { │ │ │ │ │ + options.symbolizer[key] = OpenLayers.Util.extend({}, value); │ │ │ │ │ + } else if (type === "string") { │ │ │ │ │ + options.symbolizer[key] = value; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + // clone filter │ │ │ │ │ + options.filter = this.filter && this.filter.clone(); │ │ │ │ │ + // clone context │ │ │ │ │ + options.context = this.context && OpenLayers.Util.extend({}, this.context); │ │ │ │ │ + return new OpenLayers.Rule(options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - writePolygonGeometry: function(polygon) { │ │ │ │ │ - if (!(polygon instanceof OpenLayers.Geometry.Polygon)) { │ │ │ │ │ - throw { │ │ │ │ │ - message: 'Cannot write polygon geometry to ArcXML with an ' + │ │ │ │ │ - polygon.CLASS_NAME + ' object.', │ │ │ │ │ - geometry: polygon │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var polyElem = this.createElementNS("", "POLYGON"); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Rule" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WPSDescribeProcess.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - for (var ln = 0, lnlen = polygon.components.length; ln < lnlen; ln++) { │ │ │ │ │ - var ring = polygon.components[ln]; │ │ │ │ │ - var ringElem = this.createElementNS("", "RING"); │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - for (var rn = 0, rnlen = ring.components.length; rn < rnlen; rn++) { │ │ │ │ │ - var point = ring.components[rn]; │ │ │ │ │ - var pointElem = this.createElementNS("", "POINT"); │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Format/OWSCommon/v1_1_0.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - pointElem.setAttribute("x", point.x); │ │ │ │ │ - pointElem.setAttribute("y", point.y); │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WPSDescribeProcess │ │ │ │ │ + * Read WPS DescribeProcess responses. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WPSDescribeProcess = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ - ringElem.appendChild(pointElem); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Constant: VERSION │ │ │ │ │ + * {String} 1.0.0 │ │ │ │ │ + */ │ │ │ │ │ + VERSION: "1.0.0", │ │ │ │ │ │ │ │ │ │ - polyElem.appendChild(ringElem); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + wps: "http://www.opengis.net/wps/1.0.0", │ │ │ │ │ + ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return polyElem; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} Schema location │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: "http://www.opengis.net/wps/1.0.0 http://schemas.opengis.net/wps/1.0.0/wpsAll.xsd", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseResponse │ │ │ │ │ - * Take an ArcXML response, and parse in into this object's internal properties. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} The ArcXML response, as either a string or the │ │ │ │ │ - * top level DOMElement of the response. │ │ │ │ │ - */ │ │ │ │ │ - parseResponse: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - var newData = new OpenLayers.Format.XML(); │ │ │ │ │ - data = newData.read(data); │ │ │ │ │ - } │ │ │ │ │ - var response = new OpenLayers.Format.ArcXML.Response(); │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: "wps", │ │ │ │ │ │ │ │ │ │ - var errorNode = data.getElementsByTagName("ERROR"); │ │ │ │ │ + /** │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ + */ │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (errorNode != null && errorNode.length > 0) { │ │ │ │ │ - response.error = this.getChildValue(errorNode, "Unknown error."); │ │ │ │ │ - } else { │ │ │ │ │ - var responseNode = data.getElementsByTagName("RESPONSE"); │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WPSDescribeProcess │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (responseNode == null || responseNode.length == 0) { │ │ │ │ │ - response.error = "No RESPONSE tag found in ArcXML response."; │ │ │ │ │ - return response; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Parse a WPS DescribeProcess and return an object with its information. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var rtype = responseNode[0].firstChild.nodeName; │ │ │ │ │ - if (rtype == "#text") { │ │ │ │ │ - rtype = responseNode[0].firstChild.nextSibling.nodeName; │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ } │ │ │ │ │ + var info = {}; │ │ │ │ │ + this.readNode(data, info); │ │ │ │ │ + return info; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (rtype == "IMAGE") { │ │ │ │ │ - var envelopeNode = data.getElementsByTagName("ENVELOPE"); │ │ │ │ │ - var outputNode = data.getElementsByTagName("OUTPUT"); │ │ │ │ │ - │ │ │ │ │ - if (envelopeNode == null || envelopeNode.length == 0) { │ │ │ │ │ - response.error = "No ENVELOPE tag found in ArcXML response."; │ │ │ │ │ - } else if (outputNode == null || outputNode.length == 0) { │ │ │ │ │ - response.error = "No OUTPUT tag found in ArcXML response."; │ │ │ │ │ - } else { │ │ │ │ │ - var envAttr = this.parseAttributes(envelopeNode[0]); │ │ │ │ │ - var outputAttr = this.parseAttributes(outputNode[0]); │ │ │ │ │ - │ │ │ │ │ - if (typeof outputAttr.type == "string") { │ │ │ │ │ - response.image = { │ │ │ │ │ - envelope: envAttr, │ │ │ │ │ - output: { │ │ │ │ │ - type: outputAttr.type, │ │ │ │ │ - data: this.getChildValue(outputNode[0]) │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - } else { │ │ │ │ │ - response.image = { │ │ │ │ │ - envelope: envAttr, │ │ │ │ │ - output: outputAttr │ │ │ │ │ - }; │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wps": { │ │ │ │ │ + "ProcessDescriptions": function(node, obj) { │ │ │ │ │ + obj.processDescriptions = {}; │ │ │ │ │ + this.readChildNodes(node, obj.processDescriptions); │ │ │ │ │ + }, │ │ │ │ │ + "ProcessDescription": function(node, processDescriptions) { │ │ │ │ │ + var processVersion = this.getAttributeNS(node, this.namespaces.wps, "processVersion"); │ │ │ │ │ + var processDescription = { │ │ │ │ │ + processVersion: processVersion, │ │ │ │ │ + statusSupported: (node.getAttribute("statusSupported") === "true"), │ │ │ │ │ + storeSupported: (node.getAttribute("storeSupported") === "true") │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, processDescription); │ │ │ │ │ + processDescriptions[processDescription.identifier] = processDescription; │ │ │ │ │ + }, │ │ │ │ │ + "DataInputs": function(node, processDescription) { │ │ │ │ │ + processDescription.dataInputs = []; │ │ │ │ │ + this.readChildNodes(node, processDescription.dataInputs); │ │ │ │ │ + }, │ │ │ │ │ + "ProcessOutputs": function(node, processDescription) { │ │ │ │ │ + processDescription.processOutputs = []; │ │ │ │ │ + this.readChildNodes(node, processDescription.processOutputs); │ │ │ │ │ + }, │ │ │ │ │ + "Output": function(node, processOutputs) { │ │ │ │ │ + var output = {}; │ │ │ │ │ + this.readChildNodes(node, output); │ │ │ │ │ + processOutputs.push(output); │ │ │ │ │ + }, │ │ │ │ │ + "ComplexOutput": function(node, output) { │ │ │ │ │ + output.complexOutput = {}; │ │ │ │ │ + this.readChildNodes(node, output.complexOutput); │ │ │ │ │ + }, │ │ │ │ │ + "LiteralOutput": function(node, output) { │ │ │ │ │ + output.literalOutput = {}; │ │ │ │ │ + this.readChildNodes(node, output.literalOutput); │ │ │ │ │ + }, │ │ │ │ │ + "Input": function(node, dataInputs) { │ │ │ │ │ + var input = { │ │ │ │ │ + maxOccurs: parseInt(node.getAttribute("maxOccurs")), │ │ │ │ │ + minOccurs: parseInt(node.getAttribute("minOccurs")) │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, input); │ │ │ │ │ + dataInputs.push(input); │ │ │ │ │ + }, │ │ │ │ │ + "BoundingBoxData": function(node, input) { │ │ │ │ │ + input.boundingBoxData = {}; │ │ │ │ │ + this.readChildNodes(node, input.boundingBoxData); │ │ │ │ │ + }, │ │ │ │ │ + "CRS": function(node, obj) { │ │ │ │ │ + if (!obj.CRSs) { │ │ │ │ │ + obj.CRSs = {}; │ │ │ │ │ + } │ │ │ │ │ + obj.CRSs[this.getChildValue(node)] = true; │ │ │ │ │ + }, │ │ │ │ │ + "LiteralData": function(node, input) { │ │ │ │ │ + input.literalData = {}; │ │ │ │ │ + this.readChildNodes(node, input.literalData); │ │ │ │ │ + }, │ │ │ │ │ + "ComplexData": function(node, input) { │ │ │ │ │ + input.complexData = {}; │ │ │ │ │ + this.readChildNodes(node, input.complexData); │ │ │ │ │ + }, │ │ │ │ │ + "Default": function(node, complexData) { │ │ │ │ │ + complexData["default"] = {}; │ │ │ │ │ + this.readChildNodes(node, complexData["default"]); │ │ │ │ │ + }, │ │ │ │ │ + "Supported": function(node, complexData) { │ │ │ │ │ + complexData["supported"] = {}; │ │ │ │ │ + this.readChildNodes(node, complexData["supported"]); │ │ │ │ │ + }, │ │ │ │ │ + "Format": function(node, obj) { │ │ │ │ │ + var format = {}; │ │ │ │ │ + this.readChildNodes(node, format); │ │ │ │ │ + if (!obj.formats) { │ │ │ │ │ + obj.formats = {}; │ │ │ │ │ } │ │ │ │ │ + obj.formats[format.mimeType] = true; │ │ │ │ │ + }, │ │ │ │ │ + "MimeType": function(node, format) { │ │ │ │ │ + format.mimeType = this.getChildValue(node); │ │ │ │ │ } │ │ │ │ │ - } else if (rtype == "FEATURES") { │ │ │ │ │ - var features = responseNode[0].getElementsByTagName("FEATURES"); │ │ │ │ │ - │ │ │ │ │ - // get the feature count │ │ │ │ │ - var featureCount = features[0].getElementsByTagName("FEATURECOUNT"); │ │ │ │ │ - response.features.featurecount = featureCount[0].getAttribute("count"); │ │ │ │ │ + }, │ │ │ │ │ + "ows": OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (response.features.featurecount > 0) { │ │ │ │ │ - // get the feature envelope │ │ │ │ │ - var envelope = features[0].getElementsByTagName("ENVELOPE"); │ │ │ │ │ - response.features.envelope = this.parseAttributes(envelope[0], typeof(0)); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WPSDescribeProcess" │ │ │ │ │ │ │ │ │ │ - // get the field values per feature │ │ │ │ │ - var featureList = features[0].getElementsByTagName("FEATURE"); │ │ │ │ │ - for (var fn = 0; fn < featureList.length; fn++) { │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ - var fields = featureList[fn].getElementsByTagName("FIELD"); │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/WPSClient.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - for (var fdn = 0; fdn < fields.length; fdn++) { │ │ │ │ │ - var fieldName = fields[fdn].getAttribute("name"); │ │ │ │ │ - var fieldValue = fields[fdn].getAttribute("value"); │ │ │ │ │ - feature.attributes[fieldName] = fieldValue; │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - var geom = featureList[fn].getElementsByTagName("POLYGON"); │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/SingleFile.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (geom.length > 0) { │ │ │ │ │ - // if there is a polygon, create an openlayers polygon, and assign │ │ │ │ │ - // it to the .geometry property of the feature │ │ │ │ │ - var ring = geom[0].getElementsByTagName("RING"); │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ + * @requires OpenLayers/WPSProcess.js │ │ │ │ │ + * @requires OpenLayers/Format/WPSDescribeProcess.js │ │ │ │ │ + * @requires OpenLayers/Request.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - var polys = []; │ │ │ │ │ - for (var rn = 0; rn < ring.length; rn++) { │ │ │ │ │ - var linearRings = []; │ │ │ │ │ - linearRings.push(this.parsePointGeometry(ring[rn])); │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.WPSClient │ │ │ │ │ + * High level API for interaction with Web Processing Services (WPS). │ │ │ │ │ + * An instance is used to create │ │ │ │ │ + * instances for servers known to the WPSClient. The WPSClient also caches │ │ │ │ │ + * DescribeProcess responses to reduce the number of requests sent to servers │ │ │ │ │ + * when processes are created. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.WPSClient = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ - var holes = ring[rn].getElementsByTagName("HOLE"); │ │ │ │ │ - for (var hn = 0; hn < holes.length; hn++) { │ │ │ │ │ - linearRings.push(this.parsePointGeometry(holes[hn])); │ │ │ │ │ - } │ │ │ │ │ - holes = null; │ │ │ │ │ - polys.push(new OpenLayers.Geometry.Polygon(linearRings)); │ │ │ │ │ - linearRings = null; │ │ │ │ │ - } │ │ │ │ │ - ring = null; │ │ │ │ │ + /** │ │ │ │ │ + * Property: servers │ │ │ │ │ + * {Object} Service metadata, keyed by a local identifier. │ │ │ │ │ + * │ │ │ │ │ + * Properties: │ │ │ │ │ + * url - {String} the url of the server │ │ │ │ │ + * version - {String} WPS version of the server │ │ │ │ │ + * processDescription - {Object} Cache of raw DescribeProcess │ │ │ │ │ + * responses, keyed by process identifier. │ │ │ │ │ + */ │ │ │ │ │ + servers: null, │ │ │ │ │ │ │ │ │ │ - if (polys.length == 1) { │ │ │ │ │ - feature.geometry = polys[0]; │ │ │ │ │ - } else { │ │ │ │ │ - feature.geometry = new OpenLayers.Geometry.MultiPolygon(polys); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} The default WPS version to use if none is configured. Default │ │ │ │ │ + * is '1.0.0'. │ │ │ │ │ + */ │ │ │ │ │ + version: '1.0.0', │ │ │ │ │ │ │ │ │ │ - response.features.feature.push(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - response.error = "Unidentified response type."; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return response; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: lazy │ │ │ │ │ + * {Boolean} Should the DescribeProcess be deferred until a process is │ │ │ │ │ + * fully configured? Default is false. │ │ │ │ │ + */ │ │ │ │ │ + lazy: false, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: events │ │ │ │ │ + * {} │ │ │ │ │ + * │ │ │ │ │ + * Supported event types: │ │ │ │ │ + * describeprocess - Fires when the process description is available. │ │ │ │ │ + * Listeners receive an object with a 'raw' property holding the raw │ │ │ │ │ + * DescribeProcess response, and an 'identifier' property holding the │ │ │ │ │ + * process identifier of the described process. │ │ │ │ │ + */ │ │ │ │ │ + events: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseAttributes │ │ │ │ │ + * Constructor: OpenLayers.WPSClient │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {} An element to parse attributes from. │ │ │ │ │ + * options - {Object} Object whose properties will be set on the instance. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An attributes object, with properties set to attribute values. │ │ │ │ │ + * Avaliable options: │ │ │ │ │ + * servers - {Object} Mandatory. Service metadata, keyed by a local │ │ │ │ │ + * identifier. Can either be a string with the service url or an │ │ │ │ │ + * object literal with additional metadata: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * servers: { │ │ │ │ │ + * local: '/geoserver/wps' │ │ │ │ │ + * }, { │ │ │ │ │ + * opengeo: { │ │ │ │ │ + * url: 'http://demo.opengeo.org/geoserver/wps', │ │ │ │ │ + * version: '1.0.0' │ │ │ │ │ + * } │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * lazy - {Boolean} Optional. Set to true if DescribeProcess should not be │ │ │ │ │ + * requested until a process is fully configured. Default is false. │ │ │ │ │ */ │ │ │ │ │ - parseAttributes: function(node, type) { │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - for (var attr = 0; attr < node.attributes.length; attr++) { │ │ │ │ │ - if (type == "number") { │ │ │ │ │ - attributes[node.attributes[attr].nodeName] = parseFloat(node.attributes[attr].nodeValue); │ │ │ │ │ - } else { │ │ │ │ │ - attributes[node.attributes[attr].nodeName] = node.attributes[attr].nodeValue; │ │ │ │ │ - } │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.events = new OpenLayers.Events(this); │ │ │ │ │ + this.servers = {}; │ │ │ │ │ + for (var s in options.servers) { │ │ │ │ │ + this.servers[s] = typeof options.servers[s] == 'string' ? { │ │ │ │ │ + url: options.servers[s], │ │ │ │ │ + version: this.version, │ │ │ │ │ + processDescription: {} │ │ │ │ │ + } : options.servers[s]; │ │ │ │ │ } │ │ │ │ │ - return attributes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: execute │ │ │ │ │ + * Shortcut to execute a process with a single function call. This is │ │ │ │ │ + * equivalent to using and then calling execute on the │ │ │ │ │ + * process. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Options for the execute operation. │ │ │ │ │ + * │ │ │ │ │ + * Available options: │ │ │ │ │ + * server - {String} Mandatory. One of the local identifiers of the │ │ │ │ │ + * configured servers. │ │ │ │ │ + * process - {String} Mandatory. A process identifier known to the │ │ │ │ │ + * server. │ │ │ │ │ + * inputs - {Object} The inputs for the process, keyed by input identifier. │ │ │ │ │ + * For spatial data inputs, the value of an input is usually an │ │ │ │ │ + * , an or an array of │ │ │ │ │ + * geometries or features. │ │ │ │ │ + * output - {String} The identifier of an output to parse. Optional. If not │ │ │ │ │ + * provided, the first output will be parsed. │ │ │ │ │ + * success - {Function} Callback to call when the process is complete. │ │ │ │ │ + * This function is called with an outputs object as argument, which │ │ │ │ │ + * will have a property with the identifier of the requested output │ │ │ │ │ + * (e.g. 'result'). For processes that generate spatial output, the │ │ │ │ │ + * value will either be a single or an │ │ │ │ │ + * array of features. │ │ │ │ │ + * scope - {Object} Optional scope for the success callback. │ │ │ │ │ + */ │ │ │ │ │ + execute: function(options) { │ │ │ │ │ + var process = this.getProcess(options.server, options.process); │ │ │ │ │ + process.execute({ │ │ │ │ │ + inputs: options.inputs, │ │ │ │ │ + success: options.success, │ │ │ │ │ + scope: options.scope │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parsePointGeometry │ │ │ │ │ + * APIMethod: getProcess │ │ │ │ │ + * Creates an . │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {} An element to parse or arcxml data from. │ │ │ │ │ + * serverID - {String} Local identifier from the servers that this instance │ │ │ │ │ + * was constructed with. │ │ │ │ │ + * processID - {String} Process identifier known to the server. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {} A linear ring represented by the node's points. │ │ │ │ │ + * {} │ │ │ │ │ */ │ │ │ │ │ - parsePointGeometry: function(node) { │ │ │ │ │ - var ringPoints = []; │ │ │ │ │ - var coords = node.getElementsByTagName("COORDS"); │ │ │ │ │ - │ │ │ │ │ - if (coords.length > 0) { │ │ │ │ │ - // if coords is present, it's the only coords item │ │ │ │ │ - var coordArr = this.getChildValue(coords[0]); │ │ │ │ │ - coordArr = coordArr.split(/;/); │ │ │ │ │ - for (var cn = 0; cn < coordArr.length; cn++) { │ │ │ │ │ - var coordItems = coordArr[cn].split(/ /); │ │ │ │ │ - ringPoints.push(new OpenLayers.Geometry.Point(coordItems[0], coordItems[1])); │ │ │ │ │ - } │ │ │ │ │ - coords = null; │ │ │ │ │ - } else { │ │ │ │ │ - var point = node.getElementsByTagName("POINT"); │ │ │ │ │ - if (point.length > 0) { │ │ │ │ │ - for (var pn = 0; pn < point.length; pn++) { │ │ │ │ │ - ringPoints.push( │ │ │ │ │ - new OpenLayers.Geometry.Point( │ │ │ │ │ - parseFloat(point[pn].getAttribute("x")), │ │ │ │ │ - parseFloat(point[pn].getAttribute("y")) │ │ │ │ │ - ) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - point = null; │ │ │ │ │ + getProcess: function(serverID, processID) { │ │ │ │ │ + var process = new OpenLayers.WPSProcess({ │ │ │ │ │ + client: this, │ │ │ │ │ + server: serverID, │ │ │ │ │ + identifier: processID │ │ │ │ │ + }); │ │ │ │ │ + if (!this.lazy) { │ │ │ │ │ + process.describe(); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return new OpenLayers.Geometry.LinearRing(ringPoints); │ │ │ │ │ + return process; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.ArcXML" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Format.ArcXML.Request = OpenLayers.Class({ │ │ │ │ │ - initialize: function(params) { │ │ │ │ │ - var defaults = { │ │ │ │ │ - get_image: { │ │ │ │ │ - properties: { │ │ │ │ │ - background: null, │ │ │ │ │ - /*{ │ │ │ │ │ - color: { r:255, g:255, b:255 }, │ │ │ │ │ - transcolor: null │ │ │ │ │ - },*/ │ │ │ │ │ - draw: true, │ │ │ │ │ - envelope: { │ │ │ │ │ - minx: 0, │ │ │ │ │ - miny: 0, │ │ │ │ │ - maxx: 0, │ │ │ │ │ - maxy: 0 │ │ │ │ │ - }, │ │ │ │ │ - featurecoordsys: { │ │ │ │ │ - id: 0, │ │ │ │ │ - string: "", │ │ │ │ │ - datumtransformid: 0, │ │ │ │ │ - datumtransformstring: "" │ │ │ │ │ - }, │ │ │ │ │ - filtercoordsys: { │ │ │ │ │ - id: 0, │ │ │ │ │ - string: "", │ │ │ │ │ - datumtransformid: 0, │ │ │ │ │ - datumtransformstring: "" │ │ │ │ │ + /** │ │ │ │ │ + * Method: describeProcess │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * serverID - {String} Identifier of the server │ │ │ │ │ + * processID - {String} Identifier of the requested process │ │ │ │ │ + * callback - {Function} Callback to call when the description is available │ │ │ │ │ + * scope - {Object} Optional execution scope for the callback function │ │ │ │ │ + */ │ │ │ │ │ + describeProcess: function(serverID, processID, callback, scope) { │ │ │ │ │ + var server = this.servers[serverID]; │ │ │ │ │ + if (!server.processDescription[processID]) { │ │ │ │ │ + if (!(processID in server.processDescription)) { │ │ │ │ │ + // set to null so we know a describeFeature request is pending │ │ │ │ │ + server.processDescription[processID] = null; │ │ │ │ │ + OpenLayers.Request.GET({ │ │ │ │ │ + url: server.url, │ │ │ │ │ + params: { │ │ │ │ │ + SERVICE: 'WPS', │ │ │ │ │ + VERSION: server.version, │ │ │ │ │ + REQUEST: 'DescribeProcess', │ │ │ │ │ + IDENTIFIER: processID │ │ │ │ │ }, │ │ │ │ │ - imagesize: { │ │ │ │ │ - height: 0, │ │ │ │ │ - width: 0, │ │ │ │ │ - dpi: 96, │ │ │ │ │ - printheight: 0, │ │ │ │ │ - printwidth: 0, │ │ │ │ │ - scalesymbols: false │ │ │ │ │ + success: function(response) { │ │ │ │ │ + server.processDescription[processID] = response.responseText; │ │ │ │ │ + this.events.triggerEvent('describeprocess', { │ │ │ │ │ + identifier: processID, │ │ │ │ │ + raw: response.responseText │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ - layerlist: [], │ │ │ │ │ - /* no support for legends */ │ │ │ │ │ - output: { │ │ │ │ │ - baseurl: "", │ │ │ │ │ - legendbaseurl: "", │ │ │ │ │ - legendname: "", │ │ │ │ │ - legendpath: "", │ │ │ │ │ - legendurl: "", │ │ │ │ │ - name: "", │ │ │ │ │ - path: "", │ │ │ │ │ - type: "jpg", │ │ │ │ │ - url: "" │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + // pending request │ │ │ │ │ + this.events.register('describeprocess', this, function describe(evt) { │ │ │ │ │ + if (evt.identifier === processID) { │ │ │ │ │ + this.events.unregister('describeprocess', this, describe); │ │ │ │ │ + callback.call(scope, evt.raw); │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + window.setTimeout(function() { │ │ │ │ │ + callback.call(scope, server.processDescription[processID]); │ │ │ │ │ + }, 0); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - get_feature: { │ │ │ │ │ - layer: "", │ │ │ │ │ - query: { │ │ │ │ │ - isspatial: false, │ │ │ │ │ - featurecoordsys: { │ │ │ │ │ - id: 0, │ │ │ │ │ - string: "", │ │ │ │ │ - datumtransformid: 0, │ │ │ │ │ - datumtransformstring: "" │ │ │ │ │ - }, │ │ │ │ │ - filtercoordsys: { │ │ │ │ │ - id: 0, │ │ │ │ │ - string: "", │ │ │ │ │ - datumtransformid: 0, │ │ │ │ │ - datumtransformstring: "" │ │ │ │ │ - }, │ │ │ │ │ - buffer: 0, │ │ │ │ │ - where: "", │ │ │ │ │ - spatialfilter: { │ │ │ │ │ - relation: "envelope_intersection", │ │ │ │ │ - envelope: null │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.events.destroy(); │ │ │ │ │ + this.events = null; │ │ │ │ │ + this.servers = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - environment: { │ │ │ │ │ - separators: { │ │ │ │ │ - cs: " ", │ │ │ │ │ - ts: ";" │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + CLASS_NAME: 'OpenLayers.WPSClient' │ │ │ │ │ │ │ │ │ │ - layer: [], │ │ │ │ │ - workspaces: [] │ │ │ │ │ - }; │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Symbolizer.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - return OpenLayers.Util.extend(this, defaults); │ │ │ │ │ - }, │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.ArcXML.Request" │ │ │ │ │ -}); │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ -OpenLayers.Format.ArcXML.Response = OpenLayers.Class({ │ │ │ │ │ - initialize: function(params) { │ │ │ │ │ - var defaults = { │ │ │ │ │ - image: { │ │ │ │ │ - envelope: null, │ │ │ │ │ - output: '' │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Symbolizer │ │ │ │ │ + * Base class representing a symbolizer used for feature rendering. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Symbolizer = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ - features: { │ │ │ │ │ - featurecount: 0, │ │ │ │ │ - envelope: null, │ │ │ │ │ - feature: [] │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - error: '' │ │ │ │ │ - }; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zIndex │ │ │ │ │ + * {Number} The zIndex determines the rendering order for a symbolizer. │ │ │ │ │ + * Symbolizers with larger zIndex values are rendered over symbolizers │ │ │ │ │ + * with smaller zIndex values. Default is 0. │ │ │ │ │ + */ │ │ │ │ │ + zIndex: 0, │ │ │ │ │ │ │ │ │ │ - return OpenLayers.Util.extend(this, defaults); │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Symbolizer │ │ │ │ │ + * Instances of this class are not useful. See one of the subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} An object containing properties to be set on the │ │ │ │ │ + * symbolizer. Any documented symbolizer property can be set at │ │ │ │ │ + * construction. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * A new symbolizer. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ + OpenLayers.Util.extend(this, config); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.ArcXML.Response" │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Create a copy of this symbolizer. │ │ │ │ │ + * │ │ │ │ │ + * Returns a symbolizer of the same type with the same properties. │ │ │ │ │ + */ │ │ │ │ │ + clone: function() { │ │ │ │ │ + var Type = eval(this.CLASS_NAME); │ │ │ │ │ + return new Type(OpenLayers.Util.extend({}, this)); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Symbolizer" │ │ │ │ │ + │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/ArcIMS.js │ │ │ │ │ + OpenLayers/Symbolizer/Point.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ - * @requires OpenLayers/Format/ArcXML.js │ │ │ │ │ - * @requires OpenLayers/Request.js │ │ │ │ │ + * @requires OpenLayers/Symbolizer.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.ArcIMS │ │ │ │ │ - * Instances of OpenLayers.Layer.ArcIMS are used to display data from ESRI ArcIMS │ │ │ │ │ - * Mapping Services. Create a new ArcIMS layer with the │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ + * Class: OpenLayers.Symbolizer.Point │ │ │ │ │ + * A symbolizer used to render point features. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.ArcIMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ +OpenLayers.Symbolizer.Point = OpenLayers.Class(OpenLayers.Symbolizer, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: DEFAULT_PARAMS │ │ │ │ │ - * {Object} Default query string parameters. │ │ │ │ │ + * APIProperty: strokeColor │ │ │ │ │ + * {String} Color for line stroke. This is a RGB hex value (e.g. "#ff0000" │ │ │ │ │ + * for red). │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ */ │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - ClientVersion: "9.2", │ │ │ │ │ - ServiceName: '' │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: featureCoordSys │ │ │ │ │ - * {String} Code for feature coordinate system. Default is "4326". │ │ │ │ │ + * APIProperty: strokeOpacity │ │ │ │ │ + * {Number} Stroke opacity (0-1). │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ */ │ │ │ │ │ - featureCoordSys: "4326", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: filterCoordSys │ │ │ │ │ - * {String} Code for filter coordinate system. Default is "4326". │ │ │ │ │ + * APIProperty: strokeWidth │ │ │ │ │ + * {Number} Pixel stroke width. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ */ │ │ │ │ │ - filterCoordSys: "4326", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: layers │ │ │ │ │ - * {Array} An array of objects with layer properties. │ │ │ │ │ + * APIProperty: strokeLinecap │ │ │ │ │ + * {String} Stroke cap type ("butt", "round", or "square"). │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ */ │ │ │ │ │ - layers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: async │ │ │ │ │ - * {Boolean} Request images asynchronously. Default is true. │ │ │ │ │ + * Property: strokeDashstyle │ │ │ │ │ + * {String} Stroke dash style according to the SLD spec. Note that the │ │ │ │ │ + * OpenLayers values for strokeDashstyle ("dot", "dash", "dashdot", │ │ │ │ │ + * "longdash", "longdashdot", or "solid") will not work in SLD, but │ │ │ │ │ + * most SLD patterns will render correctly in OpenLayers. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ */ │ │ │ │ │ - async: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: name │ │ │ │ │ - * {String} Layer name. Default is "ArcIMS". │ │ │ │ │ + * APIProperty: fillColor │ │ │ │ │ + * {String} RGB hex fill color (e.g. "#ff0000" for red). │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ */ │ │ │ │ │ - name: "ArcIMS", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} The layer is a base layer. Default is true. │ │ │ │ │ + * APIProperty: fillOpacity │ │ │ │ │ + * {Number} Fill opacity (0-1). │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: DEFAULT_OPTIONS │ │ │ │ │ - * {Object} Default layers properties. │ │ │ │ │ + * APIProperty: pointRadius │ │ │ │ │ + * {Number} Pixel point radius. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ */ │ │ │ │ │ - DEFAULT_OPTIONS: { │ │ │ │ │ - tileSize: new OpenLayers.Size(512, 512), │ │ │ │ │ - featureCoordSys: "4326", │ │ │ │ │ - filterCoordSys: "4326", │ │ │ │ │ - layers: null, │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - async: true, │ │ │ │ │ - name: "ArcIMS" │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.ArcIMS │ │ │ │ │ - * Create a new ArcIMS layer object. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var arcims = new OpenLayers.Layer.ArcIMS( │ │ │ │ │ - * "Global Sample", │ │ │ │ │ - * "http://sample.avencia.com/servlet/com.esri.esrimap.Esrimap", │ │ │ │ │ - * { │ │ │ │ │ - * service: "OpenLayers_Sample", │ │ │ │ │ - * layers: [ │ │ │ │ │ - * // layers to manipulate │ │ │ │ │ - * {id: "1", visible: true} │ │ │ │ │ - * ] │ │ │ │ │ - * } │ │ │ │ │ - * ); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} A name for the layer │ │ │ │ │ - * url - {String} Base url for the ArcIMS server │ │ │ │ │ - * options - {Object} Optional object with properties to be set on the │ │ │ │ │ - * layer. │ │ │ │ │ + * APIProperty: externalGraphic │ │ │ │ │ + * {String} Url to an external graphic that will be used for rendering │ │ │ │ │ + * points. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ │ │ │ │ │ - this.tileSize = new OpenLayers.Size(512, 512); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: graphicWidth │ │ │ │ │ + * {Number} Pixel width for sizing an external graphic. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // parameters │ │ │ │ │ - this.params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - ServiceName: options.serviceName │ │ │ │ │ - }, │ │ │ │ │ - this.DEFAULT_PARAMS │ │ │ │ │ - ); │ │ │ │ │ - this.options = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options, this.DEFAULT_OPTIONS │ │ │ │ │ - ); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: graphicHeight │ │ │ │ │ + * {Number} Pixel height for sizing an external graphic. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply( │ │ │ │ │ - this, [name, url, this.params, options] │ │ │ │ │ - ); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: graphicOpacity │ │ │ │ │ + * {Number} Opacity (0-1) for an external graphic. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - //layer is transparent │ │ │ │ │ - if (this.transparent) { │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: graphicXOffset │ │ │ │ │ + * {Number} Pixel offset along the positive x axis for displacing an │ │ │ │ │ + * external graphic. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // unless explicitly set in options, make layer an overlay │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - this.isBaseLayer = false; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: graphicYOffset │ │ │ │ │ + * {Number} Pixel offset along the positive y axis for displacing an │ │ │ │ │ + * external graphic. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // jpegs can never be transparent, so intelligently switch the │ │ │ │ │ - // format, depending on the browser's capabilities │ │ │ │ │ - if (this.format == "image/jpeg") { │ │ │ │ │ - this.format = OpenLayers.Util.alphaHack() ? "image/gif" : "image/png"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: rotation │ │ │ │ │ + * {Number} The rotation of a graphic in the clockwise direction about its │ │ │ │ │ + * center point (or any point off center as specified by │ │ │ │ │ + * and ). │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // create an empty layer list if no layers specified in the options │ │ │ │ │ - if (this.options.layers === null) { │ │ │ │ │ - this.options.layers = []; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: graphicName │ │ │ │ │ + * {String} Named graphic to use when rendering points. Supported values │ │ │ │ │ + * include "circle", "square", "star", "x", "cross", and "triangle". │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * Return an image url this layer. │ │ │ │ │ + * Constructor: OpenLayers.Symbolizer.Point │ │ │ │ │ + * Create a symbolizer for rendering points. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {} A bounds representing the bbox for the │ │ │ │ │ - * request. │ │ │ │ │ + * config - {Object} An object containing properties to be set on the │ │ │ │ │ + * symbolizer. Any documented symbolizer property can be set at │ │ │ │ │ + * construction. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A string with the map image's url. │ │ │ │ │ + * A new point symbolizer. │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var url = ""; │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - │ │ │ │ │ - // create an arcxml request to generate the image │ │ │ │ │ - var axlReq = new OpenLayers.Format.ArcXML( │ │ │ │ │ - OpenLayers.Util.extend(this.options, { │ │ │ │ │ - requesttype: "image", │ │ │ │ │ - envelope: bounds.toArray(), │ │ │ │ │ - tileSize: this.tileSize │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ + OpenLayers.Symbolizer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // create a synchronous ajax request to get an arcims image │ │ │ │ │ - var req = new OpenLayers.Request.POST({ │ │ │ │ │ - url: this.getFullRequestString(), │ │ │ │ │ - data: axlReq.write(), │ │ │ │ │ - async: false │ │ │ │ │ - }); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Symbolizer.Point" │ │ │ │ │ │ │ │ │ │ - // if the response exists │ │ │ │ │ - if (req != null) { │ │ │ │ │ - var doc = req.responseXML; │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = req.responseText; │ │ │ │ │ - } │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Symbolizer/Line.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - // create a new arcxml format to read the response │ │ │ │ │ - var axlResp = new OpenLayers.Format.ArcXML(); │ │ │ │ │ - var arcxml = axlResp.read(doc); │ │ │ │ │ - url = this.getUrlOrImage(arcxml.image.output); │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - return url; │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Symbolizer.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Symbolizer.Line │ │ │ │ │ + * A symbolizer used to render line features. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Symbolizer.Line = OpenLayers.Class(OpenLayers.Symbolizer, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURLasync │ │ │ │ │ - * Get an image url this layer asynchronously, and execute a callback │ │ │ │ │ - * when the image url is generated. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {} A bounds representing the bbox for the │ │ │ │ │ - * request. │ │ │ │ │ - * callback - {Function} Function to call when image url is retrieved. │ │ │ │ │ - * scope - {Object} The scope of the callback method. │ │ │ │ │ + * APIProperty: strokeColor │ │ │ │ │ + * {String} Color for line stroke. This is a RGB hex value (e.g. "#ff0000" │ │ │ │ │ + * for red). │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ */ │ │ │ │ │ - getURLasync: function(bounds, callback, scope) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ │ │ │ │ │ - // create an arcxml request to generate the image │ │ │ │ │ - var axlReq = new OpenLayers.Format.ArcXML( │ │ │ │ │ - OpenLayers.Util.extend(this.options, { │ │ │ │ │ - requesttype: "image", │ │ │ │ │ - envelope: bounds.toArray(), │ │ │ │ │ - tileSize: this.tileSize │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: strokeOpacity │ │ │ │ │ + * {Number} Stroke opacity (0-1). │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // create an asynchronous ajax request to get an arcims image │ │ │ │ │ - OpenLayers.Request.POST({ │ │ │ │ │ - url: this.getFullRequestString(), │ │ │ │ │ - async: true, │ │ │ │ │ - data: axlReq.write(), │ │ │ │ │ - callback: function(req) { │ │ │ │ │ - // process the response from ArcIMS, and call the callback function │ │ │ │ │ - // to set the image URL │ │ │ │ │ - var doc = req.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = req.responseText; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: strokeWidth │ │ │ │ │ + * {Number} Pixel stroke width. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // create a new arcxml format to read the response │ │ │ │ │ - var axlResp = new OpenLayers.Format.ArcXML(); │ │ │ │ │ - var arcxml = axlResp.read(doc); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: strokeLinecap │ │ │ │ │ + * {String} Stroke cap type ("butt", "round", or "square"). │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - callback.call(scope, this.getUrlOrImage(arcxml.image.output)); │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: strokeDashstyle │ │ │ │ │ + * {String} Stroke dash style according to the SLD spec. Note that the │ │ │ │ │ + * OpenLayers values for strokeDashstyle ("dot", "dash", "dashdot", │ │ │ │ │ + * "longdash", "longdashdot", or "solid") will not work in SLD, but │ │ │ │ │ + * most SLD patterns will render correctly in OpenLayers. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getUrlOrImage │ │ │ │ │ - * Extract a url or image from the ArcXML image output. │ │ │ │ │ + * Constructor: OpenLayers.Symbolizer.Line │ │ │ │ │ + * Create a symbolizer for rendering lines. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * output - {Object} The image.output property of the object returned from │ │ │ │ │ - * the ArcXML format read method. │ │ │ │ │ + * config - {Object} An object containing properties to be set on the │ │ │ │ │ + * symbolizer. Any documented symbolizer property can be set at │ │ │ │ │ + * construction. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A URL for an image (potentially with the data protocol). │ │ │ │ │ + * A new line symbolizer. │ │ │ │ │ */ │ │ │ │ │ - getUrlOrImage: function(output) { │ │ │ │ │ - var ret = ""; │ │ │ │ │ - if (output.url) { │ │ │ │ │ - // If the image response output url is a string, then the image │ │ │ │ │ - // data is not inline. │ │ │ │ │ - ret = output.url; │ │ │ │ │ - } else if (output.data) { │ │ │ │ │ - // The image data is inline and base64 encoded, create a data │ │ │ │ │ - // url for the image. This will only work for small images, │ │ │ │ │ - // due to browser url length limits. │ │ │ │ │ - ret = "data:image/" + output.type + │ │ │ │ │ - ";base64," + output.data; │ │ │ │ │ - } │ │ │ │ │ - return ret; │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ + OpenLayers.Symbolizer.prototype.initialize.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Symbolizer.Line" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Symbolizer/Polygon.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Symbolizer.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Symbolizer.Polygon │ │ │ │ │ + * A symbolizer used to render line features. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Symbolizer.Polygon = OpenLayers.Class(OpenLayers.Symbolizer, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: setLayerQuery │ │ │ │ │ - * Set the query definition on this layer. Query definitions are used to │ │ │ │ │ - * render parts of the spatial data in an image, and can be used to │ │ │ │ │ - * filter features or layers in the ArcIMS service. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} The ArcIMS layer ID. │ │ │ │ │ - * querydef - {Object} The query definition to apply to this layer. │ │ │ │ │ + * APIProperty: strokeColor │ │ │ │ │ + * {String} Color for line stroke. This is a RGB hex value (e.g. "#ff0000" │ │ │ │ │ + * for red). │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ */ │ │ │ │ │ - setLayerQuery: function(id, querydef) { │ │ │ │ │ - // find the matching layer, if it exists │ │ │ │ │ - for (var lyr = 0; lyr < this.options.layers.length; lyr++) { │ │ │ │ │ - if (id == this.options.layers[lyr].id) { │ │ │ │ │ - // replace this layer definition │ │ │ │ │ - this.options.layers[lyr].query = querydef; │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - // no layer found, create a new definition │ │ │ │ │ - this.options.layers.push({ │ │ │ │ │ - id: id, │ │ │ │ │ - visible: true, │ │ │ │ │ - query: querydef │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: strokeOpacity │ │ │ │ │ + * {Number} Stroke opacity (0-1). │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureInfo │ │ │ │ │ - * Get feature information from ArcIMS. Using the applied geometry, apply │ │ │ │ │ - * the options to the query (buffer, area/envelope intersection), and │ │ │ │ │ - * query the ArcIMS service. │ │ │ │ │ - * │ │ │ │ │ - * A note about accuracy: │ │ │ │ │ - * ArcIMS interprets the accuracy attribute in feature requests to be │ │ │ │ │ - * something like the 'modulus' operator on feature coordinates, │ │ │ │ │ - * applied to the database geometry of the feature. It doesn't round, │ │ │ │ │ - * so your feature coordinates may be up to (1 x accuracy) offset from │ │ │ │ │ - * the actual feature coordinates. If the accuracy of the layer is not │ │ │ │ │ - * specified, the accuracy will be computed to be approximately 1 │ │ │ │ │ - * feature coordinate per screen pixel. │ │ │ │ │ + * APIProperty: strokeWidth │ │ │ │ │ + * {Number} Pixel stroke width. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: strokeLinecap │ │ │ │ │ + * {String} Stroke cap type ("butt", "round", or "square"). │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: strokeDashstyle │ │ │ │ │ + * {String} Stroke dash style according to the SLD spec. Note that the │ │ │ │ │ + * OpenLayers values for strokeDashstyle ("dot", "dash", "dashdot", │ │ │ │ │ + * "longdash", "longdashdot", or "solid") will not work in SLD, but │ │ │ │ │ + * most SLD patterns will render correctly in OpenLayers. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: fillColor │ │ │ │ │ + * {String} RGB hex fill color (e.g. "#ff0000" for red). │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: fillOpacity │ │ │ │ │ + * {Number} Fill opacity (0-1). │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Symbolizer.Polygon │ │ │ │ │ + * Create a symbolizer for rendering polygons. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {} or {} The │ │ │ │ │ - * geometry to use when making the query. This should be a closed │ │ │ │ │ - * polygon for behavior approximating a free selection. │ │ │ │ │ - * layer - {Object} The ArcIMS layer definition. This is an anonymous object │ │ │ │ │ - * that looks like: │ │ │ │ │ - * (code) │ │ │ │ │ - * { │ │ │ │ │ - * id: "ArcXML layer ID", // the ArcXML layer ID │ │ │ │ │ - * query: { │ │ │ │ │ - * where: "STATE = 'PA'", // the where clause of the query │ │ │ │ │ - * accuracy: 100 // the accuracy of the returned feature │ │ │ │ │ - * } │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ - * options - {Object} Object with non-default properties to set on the layer. │ │ │ │ │ - * Supported properties are buffer, callback, scope, and any other │ │ │ │ │ - * properties applicable to the ArcXML format. Set the 'callback' and │ │ │ │ │ - * 'scope' for an object and function to recieve the parsed features │ │ │ │ │ - * from ArcIMS. │ │ │ │ │ + * config - {Object} An object containing properties to be set on the │ │ │ │ │ + * symbolizer. Any documented symbolizer property can be set at │ │ │ │ │ + * construction. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * A new polygon symbolizer. │ │ │ │ │ */ │ │ │ │ │ - getFeatureInfo: function(geometry, layer, options) { │ │ │ │ │ - // set the buffer to 1 unit (dd/m/ft?) by default │ │ │ │ │ - var buffer = options.buffer || 1; │ │ │ │ │ - // empty callback by default │ │ │ │ │ - var callback = options.callback || function() {}; │ │ │ │ │ - // default scope is window (global) │ │ │ │ │ - var scope = options.scope || window; │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ + OpenLayers.Symbolizer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // apply these option to the request options │ │ │ │ │ - var requestOptions = {}; │ │ │ │ │ - OpenLayers.Util.extend(requestOptions, this.options); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Symbolizer.Polygon" │ │ │ │ │ │ │ │ │ │ - // this is a feature request │ │ │ │ │ - requestOptions.requesttype = "feature"; │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ - if (geometry instanceof OpenLayers.LonLat) { │ │ │ │ │ - // create an envelope if the geometry is really a lon/lat │ │ │ │ │ - requestOptions.polygon = null; │ │ │ │ │ - requestOptions.envelope = [ │ │ │ │ │ - geometry.lon - buffer, │ │ │ │ │ - geometry.lat - buffer, │ │ │ │ │ - geometry.lon + buffer, │ │ │ │ │ - geometry.lat + buffer │ │ │ │ │ - ]; │ │ │ │ │ - } else if (geometry instanceof OpenLayers.Geometry.Polygon) { │ │ │ │ │ - // use the polygon assigned, and empty the envelope │ │ │ │ │ - requestOptions.envelope = null; │ │ │ │ │ - requestOptions.polygon = geometry; │ │ │ │ │ - } │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Symbolizer/Text.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - // create an arcxml request to get feature requests │ │ │ │ │ - var arcxml = new OpenLayers.Format.ArcXML(requestOptions); │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - // apply any get feature options to the arcxml request │ │ │ │ │ - OpenLayers.Util.extend(arcxml.request.get_feature, options); │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Symbolizer.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - arcxml.request.get_feature.layer = layer.id; │ │ │ │ │ - if (typeof layer.query.accuracy == "number") { │ │ │ │ │ - // set the accuracy if it was specified │ │ │ │ │ - arcxml.request.get_feature.query.accuracy = layer.query.accuracy; │ │ │ │ │ - } else { │ │ │ │ │ - // guess that the accuracy is 1 per screen pixel │ │ │ │ │ - var mapCenter = this.map.getCenter(); │ │ │ │ │ - var viewPx = this.map.getViewPortPxFromLonLat(mapCenter); │ │ │ │ │ - viewPx.x++; │ │ │ │ │ - var mapOffCenter = this.map.getLonLatFromPixel(viewPx); │ │ │ │ │ - arcxml.request.get_feature.query.accuracy = mapOffCenter.lon - mapCenter.lon; │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Symbolizer.Text │ │ │ │ │ + * A symbolizer used to render text labels for features. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Symbolizer.Text = OpenLayers.Class(OpenLayers.Symbolizer, { │ │ │ │ │ │ │ │ │ │ - // set the get_feature query to be the same as the layer passed in │ │ │ │ │ - arcxml.request.get_feature.query.where = layer.query.where; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: label │ │ │ │ │ + * {String} The text for the label. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // use area_intersection │ │ │ │ │ - arcxml.request.get_feature.query.spatialfilter.relation = "area_intersection"; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: fontFamily │ │ │ │ │ + * {String} The font family for the label. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // create a new asynchronous request to get the feature info │ │ │ │ │ - OpenLayers.Request.POST({ │ │ │ │ │ - url: this.getFullRequestString({ │ │ │ │ │ - 'CustomService': 'Query' │ │ │ │ │ - }), │ │ │ │ │ - data: arcxml.write(), │ │ │ │ │ - callback: function(request) { │ │ │ │ │ - // parse the arcxml response │ │ │ │ │ - var response = arcxml.parseResponse(request.responseText); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: fontSize │ │ │ │ │ + * {String} The font size for the label. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (!arcxml.iserror()) { │ │ │ │ │ - // if the arcxml is not an error, call the callback with the features parsed │ │ │ │ │ - callback.call(scope, response.features); │ │ │ │ │ - } else { │ │ │ │ │ - // if the arcxml is an error, return null features selected │ │ │ │ │ - callback.call(scope, null); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: fontWeight │ │ │ │ │ + * {String} The font weight for the label. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ + * Property: fontStyle │ │ │ │ │ + * {String} The font style for the label. │ │ │ │ │ + * │ │ │ │ │ + * No default set here. Use OpenLayers.Renderer.defaultRenderer for defaults. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Symbolizer.Text │ │ │ │ │ + * Create a symbolizer for rendering text labels. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} An object containing properties to be set on the │ │ │ │ │ + * symbolizer. Any documented symbolizer property can be set at │ │ │ │ │ + * construction. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {} An exact clone of this layer │ │ │ │ │ + * A new text symbolizer. │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ + OpenLayers.Symbolizer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.ArcIMS(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ + CLASS_NAME: "OpenLayers.Symbolizer.Text" │ │ │ │ │ │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Symbolizer/Raster.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - return obj; │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Symbolizer.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Symbolizer.Raster │ │ │ │ │ + * A symbolizer used to render raster images. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Symbolizer.Raster = OpenLayers.Class(OpenLayers.Symbolizer, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Symbolizer.Raster │ │ │ │ │ + * Create a symbolizer for rendering rasters. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} An object containing properties to be set on the │ │ │ │ │ + * symbolizer. Any documented symbolizer property can be set at │ │ │ │ │ + * construction. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * A new raster symbolizer. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ + OpenLayers.Symbolizer.prototype.initialize.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.ArcIMS" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Symbolizer.Raster" │ │ │ │ │ + │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/OSM.js │ │ │ │ │ + OpenLayers/Style2.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Rule.js │ │ │ │ │ + * @requires OpenLayers/Symbolizer/Point.js │ │ │ │ │ + * @requires OpenLayers/Symbolizer/Line.js │ │ │ │ │ + * @requires OpenLayers/Symbolizer/Polygon.js │ │ │ │ │ + * @requires OpenLayers/Symbolizer/Text.js │ │ │ │ │ + * @requires OpenLayers/Symbolizer/Raster.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.OSM │ │ │ │ │ - * This layer allows accessing OpenStreetMap tiles. By default the OpenStreetMap │ │ │ │ │ - * hosted tile.openstreetmap.org Mapnik tileset is used. If you wish to use │ │ │ │ │ - * a different layer instead, you need to provide a different │ │ │ │ │ - * URL to the constructor. Here's an example for using OpenCycleMap: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * new OpenLayers.Layer.OSM("OpenCycleMap", │ │ │ │ │ - * ["http://a.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ - * "http://b.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ - * "http://c.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png"]); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ + * Class: OpenLayers.Style2 │ │ │ │ │ + * This class represents a collection of rules for rendering features. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.OSM = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ +OpenLayers.Style2 = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: id │ │ │ │ │ + * {String} A unique id for this session. │ │ │ │ │ + */ │ │ │ │ │ + id: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIProperty: name │ │ │ │ │ - * {String} The layer name. Defaults to "OpenStreetMap" if the first │ │ │ │ │ - * argument to the constructor is null or undefined. │ │ │ │ │ + * {String} Style identifier. │ │ │ │ │ */ │ │ │ │ │ - name: "OpenStreetMap", │ │ │ │ │ + name: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String} The tileset URL scheme. Defaults to │ │ │ │ │ - * : http://[a|b|c].tile.openstreetmap.org/${z}/${x}/${y}.png │ │ │ │ │ - * (the official OSM tileset) if the second argument to the constructor │ │ │ │ │ - * is null or undefined. To use another tileset you can have something │ │ │ │ │ - * like this: │ │ │ │ │ - * (code) │ │ │ │ │ - * new OpenLayers.Layer.OSM("OpenCycleMap", │ │ │ │ │ - * ["http://a.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ - * "http://b.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ - * "http://c.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png"]); │ │ │ │ │ - * (end) │ │ │ │ │ + * APIProperty: title │ │ │ │ │ + * {String} Title of this style. │ │ │ │ │ */ │ │ │ │ │ - url: [ │ │ │ │ │ - 'http://a.tile.openstreetmap.org/${z}/${x}/${y}.png', │ │ │ │ │ - 'http://b.tile.openstreetmap.org/${z}/${x}/${y}.png', │ │ │ │ │ - 'http://c.tile.openstreetmap.org/${z}/${x}/${y}.png' │ │ │ │ │ - ], │ │ │ │ │ + title: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: attribution │ │ │ │ │ - * {String} The layer attribution. │ │ │ │ │ + * APIProperty: description │ │ │ │ │ + * {String} Description of this style. │ │ │ │ │ */ │ │ │ │ │ - attribution: "© OpenStreetMap contributors", │ │ │ │ │ + description: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: sphericalMercator │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * APIProperty: layerName │ │ │ │ │ + * {} Name of the layer that this style belongs to, usually │ │ │ │ │ + * according to the NamedLayer attribute of an SLD document. │ │ │ │ │ */ │ │ │ │ │ - sphericalMercator: true, │ │ │ │ │ + layerName: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: wrapDateLine │ │ │ │ │ + * APIProperty: isDefault │ │ │ │ │ * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - wrapDateLine: true, │ │ │ │ │ + isDefault: false, │ │ │ │ │ │ │ │ │ │ - /** APIProperty: tileOptions │ │ │ │ │ - * {Object} optional configuration options for instances │ │ │ │ │ - * created by this Layer. Default is │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * {crossOriginKeyword: 'anonymous'} │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * When using OSM tilesets other than the default ones, it may be │ │ │ │ │ - * necessary to set this to │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * {crossOriginKeyword: null} │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * if the server does not send Access-Control-Allow-Origin headers. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: rules │ │ │ │ │ + * {Array()} Collection of rendering rules. │ │ │ │ │ */ │ │ │ │ │ - tileOptions: null, │ │ │ │ │ + rules: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.OSM │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Style2 │ │ │ │ │ + * Creates a style representing a collection of rendering rules. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} The layer name. │ │ │ │ │ - * url - {String} The tileset URL scheme. │ │ │ │ │ - * options - {Object} Configuration options for the layer. Any inherited │ │ │ │ │ - * layer option can be set in this object (e.g. │ │ │ │ │ - * ). │ │ │ │ │ + * config - {Object} An object containing properties to be set on the │ │ │ │ │ + * style. Any documented properties may be set at construction. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} A new style object. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - crossOriginKeyword: 'anonymous' │ │ │ │ │ - }, this.options && this.options.tileOptions); │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ + OpenLayers.Util.extend(this, config); │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * nullify references to prevent circular references and memory leaks │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i = 0, len = this.rules.length; i < len; i++) { │ │ │ │ │ + this.rules[i].destroy(); │ │ │ │ │ + } │ │ │ │ │ + delete this.rules; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clone │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Clones this style. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} Clone of this style. │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.OSM( │ │ │ │ │ - this.name, this.url, this.getOptions()); │ │ │ │ │ + clone: function() { │ │ │ │ │ + var config = OpenLayers.Util.extend({}, this); │ │ │ │ │ + // clone rules │ │ │ │ │ + if (this.rules) { │ │ │ │ │ + config.rules = []; │ │ │ │ │ + for (var i = 0, len = this.rules.length; i < len; ++i) { │ │ │ │ │ + config.rules.push(this.rules[i].clone()); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj; │ │ │ │ │ + return new OpenLayers.Style2(config); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.OSM" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Style2" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/MapGuide.js │ │ │ │ │ + OpenLayers/Control.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.MapGuide │ │ │ │ │ - * Instances of OpenLayers.Layer.MapGuide are used to display │ │ │ │ │ - * data from a MapGuide OS instance. │ │ │ │ │ + * Class: OpenLayers.Control │ │ │ │ │ + * Controls affect the display or behavior of the map. They allow everything │ │ │ │ │ + * from panning and zooming to displaying a scale indicator. Controls by │ │ │ │ │ + * default are added to the map they are contained within however it is │ │ │ │ │ + * possible to add a control to an external div by passing the div in the │ │ │ │ │ + * options parameter. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * The following example shows how to add many of the common controls │ │ │ │ │ + * to a map. │ │ │ │ │ + * │ │ │ │ │ + * > var map = new OpenLayers.Map('map', { controls: [] }); │ │ │ │ │ + * > │ │ │ │ │ + * > map.addControl(new OpenLayers.Control.PanZoomBar()); │ │ │ │ │ + * > map.addControl(new OpenLayers.Control.LayerSwitcher({'ascending':false})); │ │ │ │ │ + * > map.addControl(new OpenLayers.Control.Permalink()); │ │ │ │ │ + * > map.addControl(new OpenLayers.Control.Permalink('permalink')); │ │ │ │ │ + * > map.addControl(new OpenLayers.Control.MousePosition()); │ │ │ │ │ + * > map.addControl(new OpenLayers.Control.OverviewMap()); │ │ │ │ │ + * > map.addControl(new OpenLayers.Control.KeyboardDefaults()); │ │ │ │ │ * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ + * The next code fragment is a quick example of how to intercept │ │ │ │ │ + * shift-mouse click to display the extent of the bounding box │ │ │ │ │ + * dragged out by the user. Usually controls are not created │ │ │ │ │ + * in exactly this manner. See the source for a more complete │ │ │ │ │ + * example: │ │ │ │ │ + * │ │ │ │ │ + * > var control = new OpenLayers.Control(); │ │ │ │ │ + * > OpenLayers.Util.extend(control, { │ │ │ │ │ + * > draw: function () { │ │ │ │ │ + * > // this Handler.Box will intercept the shift-mousedown │ │ │ │ │ + * > // before Control.MouseDefault gets to see it │ │ │ │ │ + * > this.box = new OpenLayers.Handler.Box( control, │ │ │ │ │ + * > {"done": this.notice}, │ │ │ │ │ + * > {keyMask: OpenLayers.Handler.MOD_SHIFT}); │ │ │ │ │ + * > this.box.activate(); │ │ │ │ │ + * > }, │ │ │ │ │ + * > │ │ │ │ │ + * > notice: function (bounds) { │ │ │ │ │ + * > OpenLayers.Console.userError(bounds); │ │ │ │ │ + * > } │ │ │ │ │ + * > }); │ │ │ │ │ + * > map.addControl(control); │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.MapGuide = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ +OpenLayers.Control = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} Treat this layer as a base layer. Default is true. │ │ │ │ │ - **/ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + * Property: id │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + id: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: useHttpTile │ │ │ │ │ - * {Boolean} use a tile cache exposed directly via a webserver rather than the │ │ │ │ │ - * via mapguide server. This does require extra configuration on the Mapguide Server, │ │ │ │ │ - * and will only work when singleTile is false. The url for the layer must be set to the │ │ │ │ │ - * webserver path rather than the Mapguide mapagent. │ │ │ │ │ - * See http://trac.osgeo.org/mapguide/wiki/CodeSamples/Tiles/ServingTilesViaHttp │ │ │ │ │ - **/ │ │ │ │ │ - useHttpTile: false, │ │ │ │ │ + /** │ │ │ │ │ + * Property: map │ │ │ │ │ + * {} this gets set in the addControl() function in │ │ │ │ │ + * OpenLayers.Map │ │ │ │ │ + */ │ │ │ │ │ + map: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: singleTile │ │ │ │ │ - * {Boolean} use tile server or request single tile image. │ │ │ │ │ - **/ │ │ │ │ │ - singleTile: false, │ │ │ │ │ + * APIProperty: div │ │ │ │ │ + * {DOMElement} The element that contains the control, if not present the │ │ │ │ │ + * control is placed inside the map. │ │ │ │ │ + */ │ │ │ │ │ + div: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: useOverlay │ │ │ │ │ - * {Boolean} flag to indicate if the layer should be retrieved using │ │ │ │ │ - * GETMAPIMAGE (default) or using GETDYNAMICOVERLAY requests. │ │ │ │ │ - **/ │ │ │ │ │ - useOverlay: false, │ │ │ │ │ + * APIProperty: type │ │ │ │ │ + * {Number} Controls can have a 'type'. The type determines the type of │ │ │ │ │ + * interactions which are possible with them when they are placed in an │ │ │ │ │ + * . │ │ │ │ │ + */ │ │ │ │ │ + type: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: useAsyncOverlay │ │ │ │ │ - * {Boolean} indicates if the MapGuide site supports the asynchronous │ │ │ │ │ - * GETDYNAMICOVERLAY requests which is available in MapGuide Enterprise 2010 │ │ │ │ │ - * and MapGuide Open Source v2.0.3 or higher. The newer versions of MG │ │ │ │ │ - * is called asynchronously, allows selections to be drawn separately from │ │ │ │ │ - * the map and offers styling options. │ │ │ │ │ - * │ │ │ │ │ - * With older versions of MapGuide, set useAsyncOverlay=false. Note that in │ │ │ │ │ - * this case a synchronous AJAX call is issued and the mapname and session │ │ │ │ │ - * parameters must be used to initialize the layer, not the mapdefinition │ │ │ │ │ - * parameter. Also note that this will issue a synchronous AJAX request │ │ │ │ │ - * before the image request can be issued so the users browser may lock │ │ │ │ │ - * up if the MG Web tier does not respond in a timely fashion. │ │ │ │ │ - **/ │ │ │ │ │ - useAsyncOverlay: true, │ │ │ │ │ + * Property: allowSelection │ │ │ │ │ + * {Boolean} By default, controls do not allow selection, because │ │ │ │ │ + * it may interfere with map dragging. If this is true, OpenLayers │ │ │ │ │ + * will not prevent selection of the control. │ │ │ │ │ + * Default is false. │ │ │ │ │ + */ │ │ │ │ │ + allowSelection: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: TILE_PARAMS │ │ │ │ │ - * {Object} Hashtable of default parameter key/value pairs for tiled layer │ │ │ │ │ + /** │ │ │ │ │ + * Property: displayClass │ │ │ │ │ + * {string} This property is used for CSS related to the drawing of the │ │ │ │ │ + * Control. │ │ │ │ │ */ │ │ │ │ │ - TILE_PARAMS: { │ │ │ │ │ - operation: 'GETTILEIMAGE', │ │ │ │ │ - version: '1.2.0' │ │ │ │ │ - }, │ │ │ │ │ + displayClass: "", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: SINGLE_TILE_PARAMS │ │ │ │ │ - * {Object} Hashtable of default parameter key/value pairs for untiled layer │ │ │ │ │ + * APIProperty: title │ │ │ │ │ + * {string} This property is used for showing a tooltip over the │ │ │ │ │ + * Control. │ │ │ │ │ */ │ │ │ │ │ - SINGLE_TILE_PARAMS: { │ │ │ │ │ - operation: 'GETMAPIMAGE', │ │ │ │ │ - format: 'PNG', │ │ │ │ │ - locale: 'en', │ │ │ │ │ - clip: '1', │ │ │ │ │ - version: '1.0.0' │ │ │ │ │ - }, │ │ │ │ │ + title: "", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: OVERLAY_PARAMS │ │ │ │ │ - * {Object} Hashtable of default parameter key/value pairs for untiled layer │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * false. │ │ │ │ │ */ │ │ │ │ │ - OVERLAY_PARAMS: { │ │ │ │ │ - operation: 'GETDYNAMICMAPOVERLAYIMAGE', │ │ │ │ │ - format: 'PNG', │ │ │ │ │ - locale: 'en', │ │ │ │ │ - clip: '1', │ │ │ │ │ - version: '2.0.0' │ │ │ │ │ - }, │ │ │ │ │ + autoActivate: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: FOLDER_PARAMS │ │ │ │ │ - * {Object} Hashtable of parameter key/value pairs which describe │ │ │ │ │ - * the folder structure for tiles as configured in the mapguide │ │ │ │ │ - * serverconfig.ini section [TileServiceProperties] │ │ │ │ │ + * APIProperty: active │ │ │ │ │ + * {Boolean} The control is active (read-only). Use and │ │ │ │ │ + * to change control state. │ │ │ │ │ */ │ │ │ │ │ - FOLDER_PARAMS: { │ │ │ │ │ - tileColumnsPerFolder: 30, │ │ │ │ │ - tileRowsPerFolder: 30, │ │ │ │ │ - format: 'png', │ │ │ │ │ - querystring: null │ │ │ │ │ - }, │ │ │ │ │ + active: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultSize │ │ │ │ │ - * {} Tile size as produced by MapGuide server │ │ │ │ │ - **/ │ │ │ │ │ - defaultSize: new OpenLayers.Size(300, 300), │ │ │ │ │ + /** │ │ │ │ │ + * Property: handlerOptions │ │ │ │ │ + * {Object} Used to set non-default properties on the control's handler │ │ │ │ │ + */ │ │ │ │ │ + handlerOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: tileOriginCorner │ │ │ │ │ - * {String} MapGuide tile server uses top-left as tile origin │ │ │ │ │ - **/ │ │ │ │ │ - tileOriginCorner: "tl", │ │ │ │ │ + * Property: handler │ │ │ │ │ + * {} null │ │ │ │ │ + */ │ │ │ │ │ + handler: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.MapGuide │ │ │ │ │ - * Create a new Mapguide layer, either tiled or untiled. │ │ │ │ │ - * │ │ │ │ │ - * For tiled layers, the 'groupName' and 'mapDefinition' values │ │ │ │ │ - * must be specified as parameters in the constructor. │ │ │ │ │ + * APIProperty: eventListeners │ │ │ │ │ + * {Object} If set as an option at construction, the eventListeners │ │ │ │ │ + * object will be registered with . Object │ │ │ │ │ + * structure must be a listeners object as shown in the example for │ │ │ │ │ + * the events.on method. │ │ │ │ │ + */ │ │ │ │ │ + eventListeners: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ * │ │ │ │ │ - * For untiled base layers, specify either combination of 'mapName' and │ │ │ │ │ - * 'session', or 'mapDefinition' and 'locale'. │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ * │ │ │ │ │ - * For older versions of MapGuide and overlay layers, set useAsyncOverlay │ │ │ │ │ - * to false and in this case mapName and session are required parameters │ │ │ │ │ - * for the constructor. │ │ │ │ │ + * Listeners will be called with a reference to an event object. The │ │ │ │ │ + * properties of this event depends on exactly what happened. │ │ │ │ │ * │ │ │ │ │ - * NOTE: MapGuide OS uses a DPI value and degrees to meters conversion │ │ │ │ │ - * factor that are different than the defaults used in OpenLayers, │ │ │ │ │ - * so these must be adjusted accordingly in your application. │ │ │ │ │ - * See the MapGuide example for how to set these values for MGOS. │ │ │ │ │ + * All event objects have at least the following properties: │ │ │ │ │ + * object - {Object} A reference to control.events.object (a reference │ │ │ │ │ + * to the control). │ │ │ │ │ + * element - {DOMElement} A reference to control.events.element (which │ │ │ │ │ + * will be null unless documented otherwise). │ │ │ │ │ + * │ │ │ │ │ + * Supported map event types: │ │ │ │ │ + * activate - Triggered when activated. │ │ │ │ │ + * deactivate - Triggered when deactivated. │ │ │ │ │ + */ │ │ │ │ │ + events: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control │ │ │ │ │ + * Create an OpenLayers Control. The options passed as a parameter │ │ │ │ │ + * directly extend the control. For example passing the following: │ │ │ │ │ + * │ │ │ │ │ + * > var control = new OpenLayers.Control({div: myDiv}); │ │ │ │ │ * │ │ │ │ │ + * Overrides the default div attribute value of null. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} Name of the layer displayed in the interface │ │ │ │ │ - * url - {String} Location of the MapGuide mapagent executable │ │ │ │ │ - * (e.g. http://localhost:8008/mapguide/mapagent/mapagent.fcgi) │ │ │ │ │ - * params - {Object} hashtable of additional parameters to use. Some │ │ │ │ │ - * parameters may require additional code on the server. The ones that │ │ │ │ │ - * you may want to use are: │ │ │ │ │ - * - mapDefinition - {String} The MapGuide resource definition │ │ │ │ │ - * (e.g. Library://Samples/Gmap/Maps/gmapTiled.MapDefinition) │ │ │ │ │ - * - locale - Locale setting │ │ │ │ │ - * (for untiled overlays layers only) │ │ │ │ │ - * - mapName - {String} Name of the map as stored in the MapGuide session. │ │ │ │ │ - * (for untiled layers with a session parameter only) │ │ │ │ │ - * - session - { String} MapGuide session ID │ │ │ │ │ - * (for untiled overlays layers only) │ │ │ │ │ - * - basemaplayergroupname - {String} GroupName for tiled MapGuide layers only │ │ │ │ │ - * - format - Image format to be returned (for untiled overlay layers only) │ │ │ │ │ - * - showLayers - {String} A comma separated list of GUID's for the │ │ │ │ │ - * layers to display eg: 'cvc-xcv34,453-345-345sdf'. │ │ │ │ │ - * - hideLayers - {String} A comma separated list of GUID's for the │ │ │ │ │ - * layers to hide eg: 'cvc-xcv34,453-345-345sdf'. │ │ │ │ │ - * - showGroups - {String} A comma separated list of GUID's for the │ │ │ │ │ - * groups to display eg: 'cvc-xcv34,453-345-345sdf'. │ │ │ │ │ - * - hideGroups - {String} A comma separated list of GUID's for the │ │ │ │ │ - * groups to hide eg: 'cvc-xcv34,453-345-345sdf' │ │ │ │ │ - * - selectionXml - {String} A selection xml string Some server plumbing │ │ │ │ │ - * is required to read such a value. │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer; │ │ │ │ │ - * will vary depending if tiled or untiled maps are being requested │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + // We do this before the extend so that instances can override │ │ │ │ │ + // className in options. │ │ │ │ │ + this.displayClass = │ │ │ │ │ + this.CLASS_NAME.replace("OpenLayers.", "ol").replace(/\./g, ""); │ │ │ │ │ │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ │ │ │ │ │ - // unless explicitly set in options, if the layer is transparent, │ │ │ │ │ - // it will be an overlay │ │ │ │ │ - if (options == null || options.isBaseLayer == null) { │ │ │ │ │ - this.isBaseLayer = ((this.transparent != "true") && │ │ │ │ │ - (this.transparent != true)); │ │ │ │ │ + this.events = new OpenLayers.Events(this); │ │ │ │ │ + if (this.eventListeners instanceof Object) { │ │ │ │ │ + this.events.on(this.eventListeners); │ │ │ │ │ + } │ │ │ │ │ + if (this.id == null) { │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (options && options.useOverlay != null) { │ │ │ │ │ - this.useOverlay = options.useOverlay; │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * The destroy method is used to perform any clean up before the control │ │ │ │ │ + * is dereferenced. Typically this is where event listeners are removed │ │ │ │ │ + * to prevent memory leaks. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.events) { │ │ │ │ │ + if (this.eventListeners) { │ │ │ │ │ + this.events.un(this.eventListeners); │ │ │ │ │ + } │ │ │ │ │ + this.events.destroy(); │ │ │ │ │ + this.events = null; │ │ │ │ │ } │ │ │ │ │ + this.eventListeners = null; │ │ │ │ │ │ │ │ │ │ - //initialize for untiled layers │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - if (this.useOverlay) { │ │ │ │ │ - OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, │ │ │ │ │ - this.OVERLAY_PARAMS │ │ │ │ │ - ); │ │ │ │ │ - if (!this.useAsyncOverlay) { │ │ │ │ │ - this.params.version = "1.0.0"; │ │ │ │ │ + // eliminate circular references │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.destroy(); │ │ │ │ │ + this.handler = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.handlers) { │ │ │ │ │ + for (var key in this.handlers) { │ │ │ │ │ + if (this.handlers.hasOwnProperty(key) && │ │ │ │ │ + typeof this.handlers[key].destroy == "function") { │ │ │ │ │ + this.handlers[key].destroy(); │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, │ │ │ │ │ - this.SINGLE_TILE_PARAMS │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - //initialize for tiled layers │ │ │ │ │ - if (this.useHttpTile) { │ │ │ │ │ - OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, │ │ │ │ │ - this.FOLDER_PARAMS │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, │ │ │ │ │ - this.TILE_PARAMS │ │ │ │ │ - ); │ │ │ │ │ } │ │ │ │ │ - this.setTileSize(this.defaultSize); │ │ │ │ │ + this.handlers = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.removeControl(this); │ │ │ │ │ + this.map = null; │ │ │ │ │ } │ │ │ │ │ + this.div = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the control. This is done through an accessor │ │ │ │ │ + * so that subclasses can override this and take special action once │ │ │ │ │ + * they have their map variable set. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} An exact clone of this layer │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {} │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.MapGuide(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.params, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.map = map; │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.setMap(map); │ │ │ │ │ } │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * Return a query string for this layer │ │ │ │ │ + * Method: draw │ │ │ │ │ + * The draw method is called when the control is ready to be displayed │ │ │ │ │ + * on the page. If a div has not been created one is created. Controls │ │ │ │ │ + * with a visual component will almost always want to override this method │ │ │ │ │ + * to customize the look of control. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {} A bounds representing the bbox │ │ │ │ │ - * for the request │ │ │ │ │ + * px - {} The top-left pixel position of the control │ │ │ │ │ + * or null. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also │ │ │ │ │ - * the passed-in bounds and appropriate tile size specified │ │ │ │ │ - * as parameters. │ │ │ │ │ + * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var url; │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var mapSize = this.map.getSize(); │ │ │ │ │ - │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - //set up the call for GETMAPIMAGE or GETDYNAMICMAPOVERLAY with │ │ │ │ │ - //dynamic map parameters │ │ │ │ │ - var params = { │ │ │ │ │ - setdisplaydpi: OpenLayers.DOTS_PER_INCH, │ │ │ │ │ - setdisplayheight: mapSize.h * this.ratio, │ │ │ │ │ - setdisplaywidth: mapSize.w * this.ratio, │ │ │ │ │ - setviewcenterx: center.lon, │ │ │ │ │ - setviewcentery: center.lat, │ │ │ │ │ - setviewscale: this.map.getScale() │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - if (this.useOverlay && !this.useAsyncOverlay) { │ │ │ │ │ - //first we need to call GETVISIBLEMAPEXTENT to set the extent │ │ │ │ │ - var getVisParams = {}; │ │ │ │ │ - getVisParams = OpenLayers.Util.extend(getVisParams, params); │ │ │ │ │ - getVisParams.operation = "GETVISIBLEMAPEXTENT"; │ │ │ │ │ - getVisParams.version = "1.0.0"; │ │ │ │ │ - getVisParams.session = this.params.session; │ │ │ │ │ - getVisParams.mapName = this.params.mapName; │ │ │ │ │ - getVisParams.format = 'text/xml'; │ │ │ │ │ - url = this.getFullRequestString(getVisParams); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Request.GET({ │ │ │ │ │ - url: url, │ │ │ │ │ - async: false │ │ │ │ │ - }); │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + if (this.div == null) { │ │ │ │ │ + this.div = OpenLayers.Util.createDiv(this.id); │ │ │ │ │ + this.div.className = this.displayClass; │ │ │ │ │ + if (!this.allowSelection) { │ │ │ │ │ + this.div.className += " olControlNoSelect"; │ │ │ │ │ + this.div.setAttribute("unselectable", "on", 0); │ │ │ │ │ + this.div.onselectstart = OpenLayers.Function.False; │ │ │ │ │ } │ │ │ │ │ - //construct the full URL │ │ │ │ │ - url = this.getFullRequestString(params); │ │ │ │ │ - } else { │ │ │ │ │ - │ │ │ │ │ - //tiled version │ │ │ │ │ - var currentRes = this.map.getResolution(); │ │ │ │ │ - var colidx = Math.floor((bounds.left - this.maxExtent.left) / currentRes); │ │ │ │ │ - colidx = Math.round(colidx / this.tileSize.w); │ │ │ │ │ - var rowidx = Math.floor((this.maxExtent.top - bounds.top) / currentRes); │ │ │ │ │ - rowidx = Math.round(rowidx / this.tileSize.h); │ │ │ │ │ - │ │ │ │ │ - if (this.useHttpTile) { │ │ │ │ │ - url = this.getImageFilePath({ │ │ │ │ │ - tilecol: colidx, │ │ │ │ │ - tilerow: rowidx, │ │ │ │ │ - scaleindex: this.resolutions.length - this.map.zoom - 1 │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - } else { │ │ │ │ │ - url = this.getFullRequestString({ │ │ │ │ │ - tilecol: colidx, │ │ │ │ │ - tilerow: rowidx, │ │ │ │ │ - scaleindex: this.resolutions.length - this.map.zoom - 1 │ │ │ │ │ - }); │ │ │ │ │ + if (this.title != "") { │ │ │ │ │ + this.div.title = this.title; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return url; │ │ │ │ │ + if (px != null) { │ │ │ │ │ + this.position = px.clone(); │ │ │ │ │ + } │ │ │ │ │ + this.moveTo(this.position); │ │ │ │ │ + return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFullRequestString │ │ │ │ │ - * getFullRequestString on MapGuide layers is special, because we │ │ │ │ │ - * do a regular expression replace on ',' in parameters to '+'. │ │ │ │ │ - * This is why it is subclassed here. │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * Sets the left and top style attributes to the passed in pixel │ │ │ │ │ + * coordinates. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * altUrl - {String} Alternative base URL to use. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url appropriately encoded for MapGuide │ │ │ │ │ + * px - {} │ │ │ │ │ */ │ │ │ │ │ - getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ - // use layer's url unless altUrl passed in │ │ │ │ │ - var url = (altUrl == null) ? this.url : altUrl; │ │ │ │ │ - │ │ │ │ │ - // if url is not a string, it should be an array of strings, │ │ │ │ │ - // in which case we will randomly select one of them in order │ │ │ │ │ - // to evenly distribute requests to different urls. │ │ │ │ │ - if (typeof url == "object") { │ │ │ │ │ - url = url[Math.floor(Math.random() * url.length)]; │ │ │ │ │ - } │ │ │ │ │ - // requestString always starts with url │ │ │ │ │ - var requestString = url; │ │ │ │ │ - │ │ │ │ │ - // create a new params hashtable with all the layer params and the │ │ │ │ │ - // new params together. then convert to string │ │ │ │ │ - var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ - allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ - // ignore parameters that are already in the url search string │ │ │ │ │ - var urlParams = OpenLayers.Util.upperCaseObject( │ │ │ │ │ - OpenLayers.Util.getParameters(url)); │ │ │ │ │ - for (var key in allParams) { │ │ │ │ │ - if (key.toUpperCase() in urlParams) { │ │ │ │ │ - delete allParams[key]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ - │ │ │ │ │ - /* MapGuide needs '+' seperating things like bounds/height/width. │ │ │ │ │ - Since typically this is URL encoded, we use a slight hack: we │ │ │ │ │ - depend on the list-like functionality of getParameterString to │ │ │ │ │ - leave ',' only in the case of list items (since otherwise it is │ │ │ │ │ - encoded) then do a regular expression replace on the , characters │ │ │ │ │ - to '+' */ │ │ │ │ │ - paramsString = paramsString.replace(/,/g, "+"); │ │ │ │ │ - │ │ │ │ │ - if (paramsString != "") { │ │ │ │ │ - var lastServerChar = url.charAt(url.length - 1); │ │ │ │ │ - if ((lastServerChar == "&") || (lastServerChar == "?")) { │ │ │ │ │ - requestString += paramsString; │ │ │ │ │ - } else { │ │ │ │ │ - if (url.indexOf('?') == -1) { │ │ │ │ │ - //serverPath has no ? -- add one │ │ │ │ │ - requestString += '?' + paramsString; │ │ │ │ │ - } else { │ │ │ │ │ - //serverPath contains ?, so must already have paramsString at the end │ │ │ │ │ - requestString += '&' + paramsString; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + if ((px != null) && (this.div != null)) { │ │ │ │ │ + this.div.style.left = px.x + "px"; │ │ │ │ │ + this.div.style.top = px.y + "px"; │ │ │ │ │ } │ │ │ │ │ - return requestString; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getImageFilePath │ │ │ │ │ - * special handler to request mapguide tiles from an http exposed tilecache │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * altUrl - {String} Alternative base URL to use. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Explicitly activates a control and it's associated │ │ │ │ │ + * handler if one has been set. Controls can be │ │ │ │ │ + * deactivated by calling the deactivate() method. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A string with the url for the tile image │ │ │ │ │ + * {Boolean} True if the control was successfully activated or │ │ │ │ │ + * false if the control was already active. │ │ │ │ │ */ │ │ │ │ │ - getImageFilePath: function(newParams, altUrl) { │ │ │ │ │ - // use layer's url unless altUrl passed in │ │ │ │ │ - var url = (altUrl == null) ? this.url : altUrl; │ │ │ │ │ - │ │ │ │ │ - // if url is not a string, it should be an array of strings, │ │ │ │ │ - // in which case we will randomly select one of them in order │ │ │ │ │ - // to evenly distribute requests to different urls. │ │ │ │ │ - if (typeof url == "object") { │ │ │ │ │ - url = url[Math.floor(Math.random() * url.length)]; │ │ │ │ │ - } │ │ │ │ │ - // requestString always starts with url │ │ │ │ │ - var requestString = url; │ │ │ │ │ - │ │ │ │ │ - var tileRowGroup = ""; │ │ │ │ │ - var tileColGroup = ""; │ │ │ │ │ - │ │ │ │ │ - if (newParams.tilerow < 0) { │ │ │ │ │ - tileRowGroup = '-'; │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - if (newParams.tilerow == 0) { │ │ │ │ │ - tileRowGroup += '0'; │ │ │ │ │ - } else { │ │ │ │ │ - tileRowGroup += Math.floor(Math.abs(newParams.tilerow / this.params.tileRowsPerFolder)) * this.params.tileRowsPerFolder; │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.activate(); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - if (newParams.tilecol < 0) { │ │ │ │ │ - tileColGroup = '-'; │ │ │ │ │ + this.active = true; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + OpenLayers.Element.addClass( │ │ │ │ │ + this.map.viewPortDiv, │ │ │ │ │ + this.displayClass.replace(/ /g, "") + "Active" │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ + this.events.triggerEvent("activate"); │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (newParams.tilecol == 0) { │ │ │ │ │ - tileColGroup += '0'; │ │ │ │ │ - } else { │ │ │ │ │ - tileColGroup += Math.floor(Math.abs(newParams.tilecol / this.params.tileColumnsPerFolder)) * this.params.tileColumnsPerFolder; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivates a control and it's associated handler if any. The exact │ │ │ │ │ + * effect of this depends on the control itself. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} True if the control was effectively deactivated or false │ │ │ │ │ + * if the control was already inactive. │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.deactivate(); │ │ │ │ │ + } │ │ │ │ │ + this.active = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + OpenLayers.Element.removeClass( │ │ │ │ │ + this.map.viewPortDiv, │ │ │ │ │ + this.displayClass.replace(/ /g, "") + "Active" │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("deactivate"); │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var tilePath = '/S' + Math.floor(newParams.scaleindex) + │ │ │ │ │ - '/' + this.params.basemaplayergroupname + │ │ │ │ │ - '/R' + tileRowGroup + │ │ │ │ │ - '/C' + tileColGroup + │ │ │ │ │ - '/' + (newParams.tilerow % this.params.tileRowsPerFolder) + │ │ │ │ │ - '_' + (newParams.tilecol % this.params.tileColumnsPerFolder) + │ │ │ │ │ - '.' + this.params.format; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ - if (this.params.querystring) { │ │ │ │ │ - tilePath += "?" + this.params.querystring; │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Control.TYPE_BUTTON │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.TYPE_BUTTON = 1; │ │ │ │ │ │ │ │ │ │ - requestString += tilePath; │ │ │ │ │ - return requestString; │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Control.TYPE_TOGGLE │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.TYPE_TOGGLE = 2; │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.MapGuide" │ │ │ │ │ -}); │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Control.TYPE_TOOL │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.TYPE_TOOL = 3; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Vector.js │ │ │ │ │ + OpenLayers/Protocol.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer.js │ │ │ │ │ - * @requires OpenLayers/Renderer.js │ │ │ │ │ - * @requires OpenLayers/StyleMap.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Console.js │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.Vector │ │ │ │ │ - * Instances of OpenLayers.Layer.Vector are used to render vector data from │ │ │ │ │ - * a variety of sources. Create a new vector layer with the │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ + * Class: OpenLayers.Protocol │ │ │ │ │ + * Abstract vector layer protocol class. Not to be instantiated directly. Use │ │ │ │ │ + * one of the protocol subclasses instead. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Vector = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ +OpenLayers.Protocol = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {} │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * layer.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Listeners will be called with a reference to an event object. The │ │ │ │ │ - * properties of this event depends on exactly what happened. │ │ │ │ │ - * │ │ │ │ │ - * All event objects have at least the following properties: │ │ │ │ │ - * object - {Object} A reference to layer.events.object. │ │ │ │ │ - * element - {DOMElement} A reference to layer.events.element. │ │ │ │ │ + * Property: format │ │ │ │ │ + * {} The format used by this protocol. │ │ │ │ │ + */ │ │ │ │ │ + format: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: options │ │ │ │ │ + * {Object} Any options sent to the constructor. │ │ │ │ │ + */ │ │ │ │ │ + options: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: autoDestroy │ │ │ │ │ + * {Boolean} The creator of the protocol can set autoDestroy to false │ │ │ │ │ + * to fully control when the protocol is destroyed. Defaults to │ │ │ │ │ + * true. │ │ │ │ │ + */ │ │ │ │ │ + autoDestroy: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultFilter │ │ │ │ │ + * {} Optional default filter to read requests │ │ │ │ │ + */ │ │ │ │ │ + defaultFilter: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Protocol │ │ │ │ │ + * Abstract class for vector protocols. Create instances of a subclass. │ │ │ │ │ * │ │ │ │ │ - * Supported map event types (in addition to those from ): │ │ │ │ │ - * beforefeatureadded - Triggered before a feature is added. Listeners │ │ │ │ │ - * will receive an object with a *feature* property referencing the │ │ │ │ │ - * feature to be added. To stop the feature from being added, a │ │ │ │ │ - * listener should return false. │ │ │ │ │ - * beforefeaturesadded - Triggered before an array of features is added. │ │ │ │ │ - * Listeners will receive an object with a *features* property │ │ │ │ │ - * referencing the feature to be added. To stop the features from │ │ │ │ │ - * being added, a listener should return false. │ │ │ │ │ - * featureadded - Triggered after a feature is added. The event │ │ │ │ │ - * object passed to listeners will have a *feature* property with a │ │ │ │ │ - * reference to the added feature. │ │ │ │ │ - * featuresadded - Triggered after features are added. The event │ │ │ │ │ - * object passed to listeners will have a *features* property with a │ │ │ │ │ - * reference to an array of added features. │ │ │ │ │ - * beforefeatureremoved - Triggered before a feature is removed. Listeners │ │ │ │ │ - * will receive an object with a *feature* property referencing the │ │ │ │ │ - * feature to be removed. │ │ │ │ │ - * beforefeaturesremoved - Triggered before multiple features are removed. │ │ │ │ │ - * Listeners will receive an object with a *features* property │ │ │ │ │ - * referencing the features to be removed. │ │ │ │ │ - * featureremoved - Triggerd after a feature is removed. The event │ │ │ │ │ - * object passed to listeners will have a *feature* property with a │ │ │ │ │ - * reference to the removed feature. │ │ │ │ │ - * featuresremoved - Triggered after features are removed. The event │ │ │ │ │ - * object passed to listeners will have a *features* property with a │ │ │ │ │ - * reference to an array of removed features. │ │ │ │ │ - * beforefeatureselected - Triggered before a feature is selected. Listeners │ │ │ │ │ - * will receive an object with a *feature* property referencing the │ │ │ │ │ - * feature to be selected. To stop the feature from being selectd, a │ │ │ │ │ - * listener should return false. │ │ │ │ │ - * featureselected - Triggered after a feature is selected. Listeners │ │ │ │ │ - * will receive an object with a *feature* property referencing the │ │ │ │ │ - * selected feature. │ │ │ │ │ - * featureunselected - Triggered after a feature is unselected. │ │ │ │ │ - * Listeners will receive an object with a *feature* property │ │ │ │ │ - * referencing the unselected feature. │ │ │ │ │ - * beforefeaturemodified - Triggered when a feature is selected to │ │ │ │ │ - * be modified. Listeners will receive an object with a *feature* │ │ │ │ │ - * property referencing the selected feature. │ │ │ │ │ - * featuremodified - Triggered when a feature has been modified. │ │ │ │ │ - * Listeners will receive an object with a *feature* property referencing │ │ │ │ │ - * the modified feature. │ │ │ │ │ - * afterfeaturemodified - Triggered when a feature is finished being modified. │ │ │ │ │ - * Listeners will receive an object with a *feature* property referencing │ │ │ │ │ - * the modified feature. │ │ │ │ │ - * vertexmodified - Triggered when a vertex within any feature geometry │ │ │ │ │ - * has been modified. Listeners will receive an object with a │ │ │ │ │ - * *feature* property referencing the modified feature, a *vertex* │ │ │ │ │ - * property referencing the vertex modified (always a point geometry), │ │ │ │ │ - * and a *pixel* property referencing the pixel location of the │ │ │ │ │ - * modification. │ │ │ │ │ - * vertexremoved - Triggered when a vertex within any feature geometry │ │ │ │ │ - * has been deleted. Listeners will receive an object with a │ │ │ │ │ - * *feature* property referencing the modified feature, a *vertex* │ │ │ │ │ - * property referencing the vertex modified (always a point geometry), │ │ │ │ │ - * and a *pixel* property referencing the pixel location of the │ │ │ │ │ - * removal. │ │ │ │ │ - * sketchstarted - Triggered when a feature sketch bound for this layer │ │ │ │ │ - * is started. Listeners will receive an object with a *feature* │ │ │ │ │ - * property referencing the new sketch feature and a *vertex* property │ │ │ │ │ - * referencing the creation point. │ │ │ │ │ - * sketchmodified - Triggered when a feature sketch bound for this layer │ │ │ │ │ - * is modified. Listeners will receive an object with a *vertex* │ │ │ │ │ - * property referencing the modified vertex and a *feature* property │ │ │ │ │ - * referencing the sketch feature. │ │ │ │ │ - * sketchcomplete - Triggered when a feature sketch bound for this layer │ │ │ │ │ - * is complete. Listeners will receive an object with a *feature* │ │ │ │ │ - * property referencing the sketch feature. By returning false, a │ │ │ │ │ - * listener can stop the sketch feature from being added to the layer. │ │ │ │ │ - * refresh - Triggered when something wants a strategy to ask the protocol │ │ │ │ │ - * for a new set of features. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.options = options; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} The layer is a base layer. Default is false. Set this property │ │ │ │ │ - * in the layer options. │ │ │ │ │ + * Method: mergeWithDefaultFilter │ │ │ │ │ + * Merge filter passed to the read method with the default one │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {} │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: false, │ │ │ │ │ + mergeWithDefaultFilter: function(filter) { │ │ │ │ │ + var merged; │ │ │ │ │ + if (filter && this.defaultFilter) { │ │ │ │ │ + merged = new OpenLayers.Filter.Logical({ │ │ │ │ │ + type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ + filters: [this.defaultFilter, filter] │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + merged = filter || this.defaultFilter || undefined; │ │ │ │ │ + } │ │ │ │ │ + return merged; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isFixed │ │ │ │ │ - * {Boolean} Whether the layer remains in one place while dragging the │ │ │ │ │ - * map. Note that setting this to true will move the layer to the bottom │ │ │ │ │ - * of the layer stack. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up the protocol. │ │ │ │ │ */ │ │ │ │ │ - isFixed: false, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.options = null; │ │ │ │ │ + this.format = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: features │ │ │ │ │ - * {Array()} │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Construct a request for reading new features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} An │ │ │ │ │ + * object, the same object will be passed to the callback function passed │ │ │ │ │ + * if one exists in the options object. │ │ │ │ │ */ │ │ │ │ │ - features: null, │ │ │ │ │ + read: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.filter = this.mergeWithDefaultFilter(options.filter); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: filter │ │ │ │ │ - * {} The filter set in this layer, │ │ │ │ │ - * a strategy launching read requests can combined │ │ │ │ │ - * this filter with its own filter. │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: create │ │ │ │ │ + * Construct a request for writing newly created features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array({})} or │ │ │ │ │ + * {} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} An │ │ │ │ │ + * object, the same object will be passed to the callback function passed │ │ │ │ │ + * if one exists in the options object. │ │ │ │ │ */ │ │ │ │ │ - filter: null, │ │ │ │ │ + create: function() {}, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: selectedFeatures │ │ │ │ │ - * {Array()} │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: update │ │ │ │ │ + * Construct a request updating modified features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array({})} or │ │ │ │ │ + * {} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} An │ │ │ │ │ + * object, the same object will be passed to the callback function passed │ │ │ │ │ + * if one exists in the options object. │ │ │ │ │ */ │ │ │ │ │ - selectedFeatures: null, │ │ │ │ │ + update: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: unrenderedFeatures │ │ │ │ │ - * {Object} hash of features, keyed by feature.id, that the renderer │ │ │ │ │ - * failed to draw │ │ │ │ │ + * APIMethod: delete │ │ │ │ │ + * Construct a request deleting a removed feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} An │ │ │ │ │ + * object, the same object will be passed to the callback function passed │ │ │ │ │ + * if one exists in the options object. │ │ │ │ │ */ │ │ │ │ │ - unrenderedFeatures: null, │ │ │ │ │ + "delete": function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: reportError │ │ │ │ │ - * {Boolean} report friendly error message when loading of renderer │ │ │ │ │ - * fails. │ │ │ │ │ + * APIMethod: commit │ │ │ │ │ + * Go over the features and for each take action │ │ │ │ │ + * based on the feature state. Possible actions are create, │ │ │ │ │ + * update and delete. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array({})} │ │ │ │ │ + * options - {Object} Object whose possible keys are "create", "update", │ │ │ │ │ + * "delete", "callback" and "scope", the values referenced by the │ │ │ │ │ + * first three are objects as passed to the "create", "update", and │ │ │ │ │ + * "delete" methods, the value referenced by the "callback" key is │ │ │ │ │ + * a function which is called when the commit operation is complete │ │ │ │ │ + * using the scope referenced by the "scope" key. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array({})} An array of │ │ │ │ │ + * objects. │ │ │ │ │ */ │ │ │ │ │ - reportError: true, │ │ │ │ │ + commit: function() {}, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: style │ │ │ │ │ - * {Object} Default style for the layer │ │ │ │ │ + /** │ │ │ │ │ + * Method: abort │ │ │ │ │ + * Abort an ongoing request. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * response - {} │ │ │ │ │ */ │ │ │ │ │ - style: null, │ │ │ │ │ + abort: function(response) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: styleMap │ │ │ │ │ - * {} │ │ │ │ │ + * Method: createCallback │ │ │ │ │ + * Returns a function that applies the given public method with resp and │ │ │ │ │ + * options arguments. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * method - {Function} The method to be applied by the callback. │ │ │ │ │ + * response - {} The protocol response object. │ │ │ │ │ + * options - {Object} Options sent to the protocol method │ │ │ │ │ */ │ │ │ │ │ - styleMap: null, │ │ │ │ │ + createCallback: function(method, response, options) { │ │ │ │ │ + return OpenLayers.Function.bind(function() { │ │ │ │ │ + method.apply(this, [response, options]); │ │ │ │ │ + }, this); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.Response │ │ │ │ │ + * Protocols return Response objects to their users. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.Response = OpenLayers.Class({ │ │ │ │ │ /** │ │ │ │ │ - * Property: strategies │ │ │ │ │ - * {Array(})} Optional list of strategies for the layer. │ │ │ │ │ + * Property: code │ │ │ │ │ + * {Number} - OpenLayers.Protocol.Response.SUCCESS or │ │ │ │ │ + * OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ */ │ │ │ │ │ - strategies: null, │ │ │ │ │ + code: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: protocol │ │ │ │ │ - * {} Optional protocol for the layer. │ │ │ │ │ + * Property: requestType │ │ │ │ │ + * {String} The type of request this response corresponds to. Either │ │ │ │ │ + * "create", "read", "update" or "delete". │ │ │ │ │ */ │ │ │ │ │ - protocol: null, │ │ │ │ │ + requestType: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: renderers │ │ │ │ │ - * {Array(String)} List of supported Renderer classes. Add to this list to │ │ │ │ │ - * add support for additional renderers. This list is ordered: │ │ │ │ │ - * the first renderer which returns true for the 'supported()' │ │ │ │ │ - * method will be used, if not defined in the 'renderer' option. │ │ │ │ │ + * Property: last │ │ │ │ │ + * {Boolean} - true if this is the last response expected in a commit, │ │ │ │ │ + * false otherwise, defaults to true. │ │ │ │ │ */ │ │ │ │ │ - renderers: ['SVG', 'VML', 'Canvas'], │ │ │ │ │ + last: true, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: renderer │ │ │ │ │ - * {} │ │ │ │ │ + /** │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Array({})} or {} │ │ │ │ │ + * The features returned in the response by the server. Depending on the │ │ │ │ │ + * protocol's read payload, either features or data will be populated. │ │ │ │ │ */ │ │ │ │ │ - renderer: null, │ │ │ │ │ + features: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: rendererOptions │ │ │ │ │ - * {Object} Options for the renderer. See {} for │ │ │ │ │ - * supported options. │ │ │ │ │ + * Property: data │ │ │ │ │ + * {Object} │ │ │ │ │ + * The data returned in the response by the server. Depending on the │ │ │ │ │ + * protocol's read payload, either features or data will be populated. │ │ │ │ │ */ │ │ │ │ │ - rendererOptions: null, │ │ │ │ │ + data: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geometryType │ │ │ │ │ - * {String} geometryType allows you to limit the types of geometries this │ │ │ │ │ - * layer supports. This should be set to something like │ │ │ │ │ - * "OpenLayers.Geometry.Point" to limit types. │ │ │ │ │ + /** │ │ │ │ │ + * Property: reqFeatures │ │ │ │ │ + * {Array({})} or {} │ │ │ │ │ + * The features provided by the user and placed in the request by the │ │ │ │ │ + * protocol. │ │ │ │ │ */ │ │ │ │ │ - geometryType: null, │ │ │ │ │ + reqFeatures: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: drawn │ │ │ │ │ - * {Boolean} Whether the Vector Layer features have been drawn yet. │ │ │ │ │ + /** │ │ │ │ │ + * Property: priv │ │ │ │ │ */ │ │ │ │ │ - drawn: false, │ │ │ │ │ + priv: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: ratio │ │ │ │ │ - * {Float} This specifies the ratio of the size of the visiblity of the Vector Layer features to the size of the map. │ │ │ │ │ + /** │ │ │ │ │ + * Property: error │ │ │ │ │ + * {Object} The error object in case a service exception was encountered. │ │ │ │ │ */ │ │ │ │ │ - ratio: 1, │ │ │ │ │ + error: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Vector │ │ │ │ │ - * Create a new vector layer │ │ │ │ │ + * Constructor: OpenLayers.Protocol.Response │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} A name for the layer │ │ │ │ │ - * options - {Object} Optional object with non-default properties to set on │ │ │ │ │ - * the layer. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: success │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {} A new vector layer │ │ │ │ │ + * {Boolean} - true on success, false otherwise │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + success: function() { │ │ │ │ │ + return this.code > 0; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // allow user-set renderer, otherwise assign one │ │ │ │ │ - if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ - this.assignRenderer(); │ │ │ │ │ - } │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.Response" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ - // if no valid renderer found, display error │ │ │ │ │ - if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ - this.renderer = null; │ │ │ │ │ - this.displayError(); │ │ │ │ │ - } │ │ │ │ │ +OpenLayers.Protocol.Response.SUCCESS = 1; │ │ │ │ │ +OpenLayers.Protocol.Response.FAILURE = 0; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Handler.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - if (!this.styleMap) { │ │ │ │ │ - this.styleMap = new OpenLayers.StyleMap(); │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - this.features = []; │ │ │ │ │ - this.selectedFeatures = []; │ │ │ │ │ - this.unrenderedFeatures = {}; │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // Allow for custom layer behavior │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - for (var i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - this.strategies[i].setLayer(this); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Handler │ │ │ │ │ + * Base class to construct a higher-level handler for event sequences. All │ │ │ │ │ + * handlers have activate and deactivate methods. In addition, they have │ │ │ │ │ + * methods named like browser events. When a handler is activated, any │ │ │ │ │ + * additional methods named like a browser event is registered as a │ │ │ │ │ + * listener for the corresponding event. When a handler is deactivated, │ │ │ │ │ + * those same methods are unregistered as event listeners. │ │ │ │ │ + * │ │ │ │ │ + * Handlers also typically have a callbacks object with keys named like │ │ │ │ │ + * the abstracted events or event sequences that they are in charge of │ │ │ │ │ + * handling. The controls that wrap handlers define the methods that │ │ │ │ │ + * correspond to these abstract events - so instead of listening for │ │ │ │ │ + * individual browser events, they only listen for the abstract events │ │ │ │ │ + * defined by the handler. │ │ │ │ │ + * │ │ │ │ │ + * Handlers are created by controls, which ultimately have the responsibility │ │ │ │ │ + * of making changes to the the state of the application. Handlers │ │ │ │ │ + * themselves may make temporary changes, but in general are expected to │ │ │ │ │ + * return the application in the same state that they found it. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: id │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + id: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Destroy this layer │ │ │ │ │ + * APIProperty: control │ │ │ │ │ + * {}. The control that initialized this handler. The │ │ │ │ │ + * control is assumed to have a valid map property - that map is used │ │ │ │ │ + * in the handler's own setMap method. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoDestroy) { │ │ │ │ │ - strategy.destroy(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.strategies = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.protocol) { │ │ │ │ │ - if (this.protocol.autoDestroy) { │ │ │ │ │ - this.protocol.destroy(); │ │ │ │ │ - } │ │ │ │ │ - this.protocol = null; │ │ │ │ │ - } │ │ │ │ │ - this.destroyFeatures(); │ │ │ │ │ - this.features = null; │ │ │ │ │ - this.selectedFeatures = null; │ │ │ │ │ - this.unrenderedFeatures = null; │ │ │ │ │ - if (this.renderer) { │ │ │ │ │ - this.renderer.destroy(); │ │ │ │ │ - } │ │ │ │ │ - this.renderer = null; │ │ │ │ │ - this.geometryType = null; │ │ │ │ │ - this.drawn = null; │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + control: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer. │ │ │ │ │ - * │ │ │ │ │ - * Note: Features of the layer are also cloned. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} An exact clone of this layer │ │ │ │ │ + * Property: map │ │ │ │ │ + * {} │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ + map: null, │ │ │ │ │ │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Vector(this.name, this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: keyMask │ │ │ │ │ + * {Integer} Use bitwise operators and one or more of the OpenLayers.Handler │ │ │ │ │ + * constants to construct a keyMask. The keyMask is used by │ │ │ │ │ + * . If the keyMask matches the combination of keys │ │ │ │ │ + * down on an event, checkModifiers returns true. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * // handler only responds if the Shift key is down │ │ │ │ │ + * handler.keyMask = OpenLayers.Handler.MOD_SHIFT; │ │ │ │ │ + * │ │ │ │ │ + * // handler only responds if Ctrl-Shift is down │ │ │ │ │ + * handler.keyMask = OpenLayers.Handler.MOD_SHIFT | │ │ │ │ │ + * OpenLayers.Handler.MOD_CTRL; │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ + keyMask: null, │ │ │ │ │ │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ + /** │ │ │ │ │ + * Property: active │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + active: false, │ │ │ │ │ │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - var features = this.features; │ │ │ │ │ - var len = features.length; │ │ │ │ │ - var clonedFeatures = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - clonedFeatures[i] = features[i].clone(); │ │ │ │ │ - } │ │ │ │ │ - obj.features = clonedFeatures; │ │ │ │ │ + /** │ │ │ │ │ + * Property: evt │ │ │ │ │ + * {Event} This property references the last event handled by the handler. │ │ │ │ │ + * Note that this property is not part of the stable API. Use of the │ │ │ │ │ + * evt property should be restricted to controls in the library │ │ │ │ │ + * or other applications that are willing to update with changes to │ │ │ │ │ + * the OpenLayers code. │ │ │ │ │ + */ │ │ │ │ │ + evt: null, │ │ │ │ │ │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: touch │ │ │ │ │ + * {Boolean} Indicates the support of touch events. When touch events are │ │ │ │ │ + * started touch will be true and all mouse related listeners will do │ │ │ │ │ + * nothing. │ │ │ │ │ + */ │ │ │ │ │ + touch: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: refresh │ │ │ │ │ - * Ask the layer to request features again and redraw them. Triggers │ │ │ │ │ - * the refresh event if the layer is in range and visible. │ │ │ │ │ + * Constructor: OpenLayers.Handler │ │ │ │ │ + * Construct a handler. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {Object} Optional object with properties for any listener of │ │ │ │ │ - * the refresh event. │ │ │ │ │ + * control - {} The control that initialized this │ │ │ │ │ + * handler. The control is assumed to have a valid map property; that │ │ │ │ │ + * map is used in the handler's own setMap method. If a map property │ │ │ │ │ + * is present in the options argument it will be used instead. │ │ │ │ │ + * callbacks - {Object} An object whose properties correspond to abstracted │ │ │ │ │ + * events or sequences of browser events. The values for these │ │ │ │ │ + * properties are functions defined by the control that get called by │ │ │ │ │ + * the handler. │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * the handler. │ │ │ │ │ */ │ │ │ │ │ - refresh: function(obj) { │ │ │ │ │ - if (this.calculateInRange() && this.visibility) { │ │ │ │ │ - this.events.triggerEvent("refresh", obj); │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.control = control; │ │ │ │ │ + this.callbacks = callbacks; │ │ │ │ │ + │ │ │ │ │ + var map = this.map || control.map; │ │ │ │ │ + if (map) { │ │ │ │ │ + this.setMap(map); │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: assignRenderer │ │ │ │ │ - * Iterates through the available renderer implementations and selects │ │ │ │ │ - * and assigns the first one whose "supported()" function returns true. │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ */ │ │ │ │ │ - assignRenderer: function() { │ │ │ │ │ - for (var i = 0, len = this.renderers.length; i < len; i++) { │ │ │ │ │ - var rendererClass = this.renderers[i]; │ │ │ │ │ - var renderer = (typeof rendererClass == "function") ? │ │ │ │ │ - rendererClass : │ │ │ │ │ - OpenLayers.Renderer[rendererClass]; │ │ │ │ │ - if (renderer && renderer.prototype.supported()) { │ │ │ │ │ - this.renderer = new renderer(this.div, this.rendererOptions); │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.map = map; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: displayError │ │ │ │ │ - * Let the user know their browser isn't supported. │ │ │ │ │ + /** │ │ │ │ │ + * Method: checkModifiers │ │ │ │ │ + * Check the keyMask on the handler. If no is set, this always │ │ │ │ │ + * returns true. If a is set and it matches the combination │ │ │ │ │ + * of keys down on an event, this returns true. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The keyMask matches the keys down on an event. │ │ │ │ │ */ │ │ │ │ │ - displayError: function() { │ │ │ │ │ - if (this.reportError) { │ │ │ │ │ - OpenLayers.Console.userError(OpenLayers.i18n("browserNotSupported", { │ │ │ │ │ - renderers: this.renderers.join('\n') │ │ │ │ │ - })); │ │ │ │ │ + checkModifiers: function(evt) { │ │ │ │ │ + if (this.keyMask == null) { │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ + /* calculate the keyboard modifier mask for this event */ │ │ │ │ │ + var keyModifiers = │ │ │ │ │ + (evt.shiftKey ? OpenLayers.Handler.MOD_SHIFT : 0) | │ │ │ │ │ + (evt.ctrlKey ? OpenLayers.Handler.MOD_CTRL : 0) | │ │ │ │ │ + (evt.altKey ? OpenLayers.Handler.MOD_ALT : 0) | │ │ │ │ │ + (evt.metaKey ? OpenLayers.Handler.MOD_META : 0); │ │ │ │ │ + │ │ │ │ │ + /* if it differs from the handler object's key mask, │ │ │ │ │ + bail out of the event handler */ │ │ │ │ │ + return (keyModifiers == this.keyMask); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * The layer has been added to the map. │ │ │ │ │ - * │ │ │ │ │ - * If there is no renderer set, the layer can't be used. Remove it. │ │ │ │ │ - * Otherwise, give the renderer a reference to the map and set its size. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Turn on the handler. Returns false if the handler was already active. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was activated. │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - if (!this.renderer) { │ │ │ │ │ - this.map.removeLayer(this); │ │ │ │ │ - } else { │ │ │ │ │ - this.renderer.map = this.map; │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + // register for event handlers defined on this class. │ │ │ │ │ + var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.register(events[i], this[events[i]]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.active = true; │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var newSize = this.map.getSize(); │ │ │ │ │ - newSize.w = newSize.w * this.ratio; │ │ │ │ │ - newSize.h = newSize.h * this.ratio; │ │ │ │ │ - this.renderer.setSize(newSize); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Turn off the handler. Returns false if the handler was already inactive. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was deactivated. │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + // unregister event handlers defined on this class. │ │ │ │ │ + var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.unregister(events[i], this[events[i]]); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + this.touch = false; │ │ │ │ │ + this.active = false; │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: afterAdd │ │ │ │ │ - * Called at the end of the map.addLayer sequence. At this point, the map │ │ │ │ │ - * will have a base layer. Any autoActivate strategies will be │ │ │ │ │ - * activated here. │ │ │ │ │ + * Method: startTouch │ │ │ │ │ + * Start touch events, this method must be called by subclasses in │ │ │ │ │ + * "touchstart" method. When touch events are started will be │ │ │ │ │ + * true and all mouse related listeners will do nothing. │ │ │ │ │ */ │ │ │ │ │ - afterAdd: function() { │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoActivate) { │ │ │ │ │ - strategy.activate(); │ │ │ │ │ + startTouch: function() { │ │ │ │ │ + if (!this.touch) { │ │ │ │ │ + this.touch = true; │ │ │ │ │ + var events = [ │ │ │ │ │ + "mousedown", "mouseup", "mousemove", "click", "dblclick", │ │ │ │ │ + "mouseout" │ │ │ │ │ + ]; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.unregister(events[i], this[events[i]]); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removeMap │ │ │ │ │ - * The layer has been removed from the map. │ │ │ │ │ + * Method: callback │ │ │ │ │ + * Trigger the control's named callback with the given arguments │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {} │ │ │ │ │ + * name - {String} The key for the callback that is one of the properties │ │ │ │ │ + * of the handler's callbacks object. │ │ │ │ │ + * args - {Array(*)} An array of arguments (any type) with which to call │ │ │ │ │ + * the callback (defined by the control). │ │ │ │ │ */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - this.drawn = false; │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoActivate) { │ │ │ │ │ - strategy.deactivate(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + callback: function(name, args) { │ │ │ │ │ + if (name && this.callbacks[name]) { │ │ │ │ │ + this.callbacks[name].apply(this.control, args); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onMapResize │ │ │ │ │ - * Notify the renderer of the change in size. │ │ │ │ │ - * │ │ │ │ │ + * Method: register │ │ │ │ │ + * register an event on the map │ │ │ │ │ */ │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - OpenLayers.Layer.prototype.onMapResize.apply(this, arguments); │ │ │ │ │ + register: function(name, method) { │ │ │ │ │ + // TODO: deal with registerPriority in 3.0 │ │ │ │ │ + this.map.events.registerPriority(name, this, method); │ │ │ │ │ + this.map.events.registerPriority(name, this, this.setEvent); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var newSize = this.map.getSize(); │ │ │ │ │ - newSize.w = newSize.w * this.ratio; │ │ │ │ │ - newSize.h = newSize.h * this.ratio; │ │ │ │ │ - this.renderer.setSize(newSize); │ │ │ │ │ + /** │ │ │ │ │ + * Method: unregister │ │ │ │ │ + * unregister an event from the map │ │ │ │ │ + */ │ │ │ │ │ + unregister: function(name, method) { │ │ │ │ │ + this.map.events.unregister(name, this, method); │ │ │ │ │ + this.map.events.unregister(name, this, this.setEvent); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * Reset the vector layer's div so that it once again is lined up with │ │ │ │ │ - * the map. Notify the renderer of the change of extent, and in the │ │ │ │ │ - * case of a change of zoom level (resolution), have the │ │ │ │ │ - * renderer redraw features. │ │ │ │ │ - * │ │ │ │ │ - * If the layer has not yet been drawn, cycle through the layer's │ │ │ │ │ - * features and draw each one. │ │ │ │ │ - * │ │ │ │ │ + * Method: setEvent │ │ │ │ │ + * With each registered browser event, the handler sets its own evt │ │ │ │ │ + * property. This property can be accessed by controls if needed │ │ │ │ │ + * to get more information about the event that the handler is │ │ │ │ │ + * processing. │ │ │ │ │ + * │ │ │ │ │ + * This allows modifier keys on the event to be checked (alt, shift, ctrl, │ │ │ │ │ + * and meta cannot be checked with the keyboard handler). For a │ │ │ │ │ + * control to determine which modifier keys are associated with the │ │ │ │ │ + * event that a handler is currently processing, it should access │ │ │ │ │ + * (code)handler.evt.altKey || handler.evt.shiftKey || │ │ │ │ │ + * handler.evt.ctrlKey || handler.evt.metaKey(end). │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {} │ │ │ │ │ - * zoomChanged - {Boolean} │ │ │ │ │ - * dragging - {Boolean} │ │ │ │ │ + * evt - {Event} The browser event. │ │ │ │ │ */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + setEvent: function(evt) { │ │ │ │ │ + this.evt = evt; │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var coordSysUnchanged = true; │ │ │ │ │ - if (!dragging) { │ │ │ │ │ - this.renderer.root.style.visibility = 'hidden'; │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Deconstruct the handler. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + // unregister event listeners │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + // eliminate circular references │ │ │ │ │ + this.control = this.map = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var viewSize = this.map.getSize(), │ │ │ │ │ - viewWidth = viewSize.w, │ │ │ │ │ - viewHeight = viewSize.h, │ │ │ │ │ - offsetLeft = (viewWidth / 2 * this.ratio) - viewWidth / 2, │ │ │ │ │ - offsetTop = (viewHeight / 2 * this.ratio) - viewHeight / 2; │ │ │ │ │ - offsetLeft += this.map.layerContainerOriginPx.x; │ │ │ │ │ - offsetLeft = -Math.round(offsetLeft); │ │ │ │ │ - offsetTop += this.map.layerContainerOriginPx.y; │ │ │ │ │ - offsetTop = -Math.round(offsetTop); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ - this.div.style.left = offsetLeft + 'px'; │ │ │ │ │ - this.div.style.top = offsetTop + 'px'; │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_NONE │ │ │ │ │ + * If set as the , returns false if any key is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_NONE = 0; │ │ │ │ │ │ │ │ │ │ - var extent = this.map.getExtent().scale(this.ratio); │ │ │ │ │ - coordSysUnchanged = this.renderer.setExtent(extent, zoomChanged); │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_SHIFT │ │ │ │ │ + * If set as the , returns false if Shift is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_SHIFT = 1; │ │ │ │ │ │ │ │ │ │ - this.renderer.root.style.visibility = 'visible'; │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_CTRL │ │ │ │ │ + * If set as the , returns false if Ctrl is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_CTRL = 2; │ │ │ │ │ │ │ │ │ │ - // Force a reflow on gecko based browsers to prevent jump/flicker. │ │ │ │ │ - // This seems to happen on only certain configurations; it was originally │ │ │ │ │ - // noticed in FF 2.0 and Linux. │ │ │ │ │ - if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ - this.div.scrollLeft = this.div.scrollLeft; │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_ALT │ │ │ │ │ + * If set as the , returns false if Alt is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_ALT = 4; │ │ │ │ │ │ │ │ │ │ - if (!zoomChanged && coordSysUnchanged) { │ │ │ │ │ - for (var i in this.unrenderedFeatures) { │ │ │ │ │ - var feature = this.unrenderedFeatures[i]; │ │ │ │ │ - this.drawFeature(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!this.drawn || zoomChanged || !coordSysUnchanged) { │ │ │ │ │ - this.drawn = true; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0, len = this.features.length; i < len; i++) { │ │ │ │ │ - this.renderer.locked = (i !== (len - 1)); │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - this.drawFeature(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_META │ │ │ │ │ + * If set as the , returns false if Cmd is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_META = 8; │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Spherical.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/SingleFile.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: Spherical │ │ │ │ │ + * The OpenLayers.Spherical namespace includes utility functions for │ │ │ │ │ + * calculations on the basis of a spherical earth (ignoring ellipsoidal │ │ │ │ │ + * effects), which is accurate enough for most purposes. │ │ │ │ │ + * │ │ │ │ │ + * Relevant links: │ │ │ │ │ + * * http://www.movable-type.co.uk/scripts/latlong.html │ │ │ │ │ + * * http://code.google.com/apis/maps/documentation/javascript/reference.html#spherical │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Spherical = OpenLayers.Spherical || {}; │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Spherical.DEFAULT_RADIUS = 6378137; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * APIFunction: computeDistanceBetween │ │ │ │ │ + * Computes the distance between two LonLats. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * from - {} or {Object} Starting point. A LonLat or │ │ │ │ │ + * a JavaScript literal with lon lat properties. │ │ │ │ │ + * to - {} or {Object} Ending point. A LonLat or a │ │ │ │ │ + * JavaScript literal with lon lat properties. │ │ │ │ │ + * radius - {Float} The radius. Optional. Defaults to 6378137 meters. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The distance in meters. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Spherical.computeDistanceBetween = function(from, to, radius) { │ │ │ │ │ + var R = radius || OpenLayers.Spherical.DEFAULT_RADIUS; │ │ │ │ │ + var sinHalfDeltaLon = Math.sin(Math.PI * (to.lon - from.lon) / 360); │ │ │ │ │ + var sinHalfDeltaLat = Math.sin(Math.PI * (to.lat - from.lat) / 360); │ │ │ │ │ + var a = sinHalfDeltaLat * sinHalfDeltaLat + │ │ │ │ │ + sinHalfDeltaLon * sinHalfDeltaLon * Math.cos(Math.PI * from.lat / 180) * Math.cos(Math.PI * to.lat / 180); │ │ │ │ │ + return 2 * R * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a)); │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * APIFunction: computeHeading │ │ │ │ │ + * Computes the heading from one LonLat to another LonLat. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * from - {} or {Object} Starting point. A LonLat or │ │ │ │ │ + * a JavaScript literal with lon lat properties. │ │ │ │ │ + * to - {} or {Object} Ending point. A LonLat or a │ │ │ │ │ + * JavaScript literal with lon lat properties. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The heading in degrees. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Spherical.computeHeading = function(from, to) { │ │ │ │ │ + var y = Math.sin(Math.PI * (from.lon - to.lon) / 180) * Math.cos(Math.PI * to.lat / 180); │ │ │ │ │ + var x = Math.cos(Math.PI * from.lat / 180) * Math.sin(Math.PI * to.lat / 180) - │ │ │ │ │ + Math.sin(Math.PI * from.lat / 180) * Math.cos(Math.PI * to.lat / 180) * Math.cos(Math.PI * (from.lon - to.lon) / 180); │ │ │ │ │ + return 180 * Math.atan2(y, x) / Math.PI; │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Handler/Drag.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Handler.Drag │ │ │ │ │ + * The drag handler is used to deal with sequences of browser events related │ │ │ │ │ + * to dragging. The handler is used by controls that want to know when │ │ │ │ │ + * a drag sequence begins, when a drag is happening, and when it has │ │ │ │ │ + * finished. │ │ │ │ │ + * │ │ │ │ │ + * Controls that use the drag handler typically construct it with callbacks │ │ │ │ │ + * for 'down', 'move', and 'done'. Callbacks for these keys are called │ │ │ │ │ + * when the drag begins, with each move, and when the drag is done. In │ │ │ │ │ + * addition, controls can have callbacks keyed to 'up' and 'out' if they │ │ │ │ │ + * care to differentiate between the types of events that correspond with │ │ │ │ │ + * the end of a drag sequence. If no drag actually occurs (no mouse move) │ │ │ │ │ + * the 'down' and 'up' callbacks will be called, but not the 'done' │ │ │ │ │ + * callback. │ │ │ │ │ + * │ │ │ │ │ + * Create a new drag handler with the constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.Drag = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: display │ │ │ │ │ - * Hide or show the Layer │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * display - {Boolean} │ │ │ │ │ + * Property: started │ │ │ │ │ + * {Boolean} When a mousedown or touchstart event is received, we want to │ │ │ │ │ + * record it, but not set 'dragging' until the mouse moves after starting. │ │ │ │ │ */ │ │ │ │ │ - display: function(display) { │ │ │ │ │ - OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ - // we need to set the display style of the root in case it is attached │ │ │ │ │ - // to a foreign layer │ │ │ │ │ - var currentDisplay = this.div.style.display; │ │ │ │ │ - if (currentDisplay != this.renderer.root.style.display) { │ │ │ │ │ - this.renderer.root.style.display = currentDisplay; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + started: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: addFeatures │ │ │ │ │ - * Add Features to the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array()} │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * Property: stopDown │ │ │ │ │ + * {Boolean} Stop propagation of mousedown events from getting to listeners │ │ │ │ │ + * on the same element. Default is true. │ │ │ │ │ */ │ │ │ │ │ - addFeatures: function(features, options) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ - if (notify) { │ │ │ │ │ - var event = { │ │ │ │ │ - features: features │ │ │ │ │ - }; │ │ │ │ │ - var ret = this.events.triggerEvent("beforefeaturesadded", event); │ │ │ │ │ - if (ret === false) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - features = event.features; │ │ │ │ │ - } │ │ │ │ │ + stopDown: true, │ │ │ │ │ │ │ │ │ │ - // Track successfully added features for featuresadded event, since │ │ │ │ │ - // beforefeatureadded can veto single features. │ │ │ │ │ - var featuresAdded = []; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - if (i != (features.length - 1)) { │ │ │ │ │ - this.renderer.locked = true; │ │ │ │ │ - } else { │ │ │ │ │ - this.renderer.locked = false; │ │ │ │ │ - } │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ + /** │ │ │ │ │ + * Property: dragging │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + dragging: false, │ │ │ │ │ │ │ │ │ │ - if (this.geometryType && │ │ │ │ │ - !(feature.geometry instanceof this.geometryType)) { │ │ │ │ │ - throw new TypeError('addFeatures: component should be an ' + │ │ │ │ │ - this.geometryType.prototype.CLASS_NAME); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: last │ │ │ │ │ + * {} The last pixel location of the drag. │ │ │ │ │ + */ │ │ │ │ │ + last: null, │ │ │ │ │ │ │ │ │ │ - //give feature reference to its layer │ │ │ │ │ - feature.layer = this; │ │ │ │ │ + /** │ │ │ │ │ + * Property: start │ │ │ │ │ + * {} The first pixel location of the drag. │ │ │ │ │ + */ │ │ │ │ │ + start: null, │ │ │ │ │ │ │ │ │ │ - if (!feature.style && this.style) { │ │ │ │ │ - feature.style = OpenLayers.Util.extend({}, this.style); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: lastMoveEvt │ │ │ │ │ + * {Object} The last mousemove event that occurred. Used to │ │ │ │ │ + * position the map correctly when our "delay drag" │ │ │ │ │ + * timeout expired. │ │ │ │ │ + */ │ │ │ │ │ + lastMoveEvt: null, │ │ │ │ │ │ │ │ │ │ - if (notify) { │ │ │ │ │ - if (this.events.triggerEvent("beforefeatureadded", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) === false) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - this.preFeatureInsert(feature); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: oldOnselectstart │ │ │ │ │ + * {Function} │ │ │ │ │ + */ │ │ │ │ │ + oldOnselectstart: null, │ │ │ │ │ │ │ │ │ │ - featuresAdded.push(feature); │ │ │ │ │ - this.features.push(feature); │ │ │ │ │ - this.drawFeature(feature); │ │ │ │ │ + /** │ │ │ │ │ + * Property: interval │ │ │ │ │ + * {Integer} In order to increase performance, an interval (in │ │ │ │ │ + * milliseconds) can be set to reduce the number of drag events │ │ │ │ │ + * called. If set, a new drag event will not be set until the │ │ │ │ │ + * interval has passed. │ │ │ │ │ + * Defaults to 0, meaning no interval. │ │ │ │ │ + */ │ │ │ │ │ + interval: 0, │ │ │ │ │ │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureadded", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - this.onFeatureInsert(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: timeoutId │ │ │ │ │ + * {String} The id of the timeout used for the mousedown interval. │ │ │ │ │ + * This is "private", and should be left alone. │ │ │ │ │ + */ │ │ │ │ │ + timeoutId: null, │ │ │ │ │ │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresadded", { │ │ │ │ │ - features: featuresAdded │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: documentDrag │ │ │ │ │ + * {Boolean} If set to true, the handler will also handle mouse moves when │ │ │ │ │ + * the cursor has moved out of the map viewport. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + documentDrag: false, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: documentEvents │ │ │ │ │ + * {Boolean} Are we currently observing document events? │ │ │ │ │ + */ │ │ │ │ │ + documentEvents: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: removeFeatures │ │ │ │ │ - * Remove features from the layer. This erases any drawn features and │ │ │ │ │ - * removes them from the layer's control. The beforefeatureremoved │ │ │ │ │ - * and featureremoved events will be triggered for each feature. The │ │ │ │ │ - * featuresremoved event will be triggered after all features have │ │ │ │ │ - * been removed. To supress event triggering, use the silent option. │ │ │ │ │ + * Constructor: OpenLayers.Handler.Drag │ │ │ │ │ + * Returns OpenLayers.Handler.Drag │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array()} List of features to be │ │ │ │ │ - * removed. │ │ │ │ │ - * options - {Object} Optional properties for changing behavior of the │ │ │ │ │ - * removal. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * silent - {Boolean} Supress event triggering. Default is false. │ │ │ │ │ + * control - {} The control that is making use of │ │ │ │ │ + * this handler. If a handler is being used without a control, the │ │ │ │ │ + * handlers setMap method must be overridden to deal properly with │ │ │ │ │ + * the map. │ │ │ │ │ + * callbacks - {Object} An object containing a single function to be │ │ │ │ │ + * called when the drag operation is finished. The callback should │ │ │ │ │ + * expect to recieve a single argument, the pixel location of the event. │ │ │ │ │ + * Callbacks for 'move' and 'done' are supported. You can also speficy │ │ │ │ │ + * callbacks for 'down', 'up', and 'out' to respond to those events. │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - removeFeatures: function(features, options) { │ │ │ │ │ - if (!features || features.length === 0) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (features === this.features) { │ │ │ │ │ - return this.removeAllFeatures(options); │ │ │ │ │ - } │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ - if (features === this.selectedFeatures) { │ │ │ │ │ - features = features.slice(); │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + if (this.documentDrag === true) { │ │ │ │ │ + var me = this; │ │ │ │ │ + this._docMove = function(evt) { │ │ │ │ │ + me.mousemove({ │ │ │ │ │ + xy: { │ │ │ │ │ + x: evt.clientX, │ │ │ │ │ + y: evt.clientY │ │ │ │ │ + }, │ │ │ │ │ + element: document │ │ │ │ │ + }); │ │ │ │ │ + }; │ │ │ │ │ + this._docUp = function(evt) { │ │ │ │ │ + me.mouseup({ │ │ │ │ │ + xy: { │ │ │ │ │ + x: evt.clientX, │ │ │ │ │ + y: evt.clientY │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent( │ │ │ │ │ - "beforefeaturesremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: dragstart │ │ │ │ │ + * This private method is factorized from mousedown and touchstart methods │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ + */ │ │ │ │ │ + dragstart: function(evt) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + if (this.checkModifiers(evt) && │ │ │ │ │ + (OpenLayers.Event.isLeftClick(evt) || │ │ │ │ │ + OpenLayers.Event.isSingleTouch(evt))) { │ │ │ │ │ + this.started = true; │ │ │ │ │ + this.start = evt.xy; │ │ │ │ │ + this.last = evt.xy; │ │ │ │ │ + OpenLayers.Element.addClass( │ │ │ │ │ + this.map.viewPortDiv, "olDragDown" │ │ │ │ │ ); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - // We remain locked so long as we're not at 0 │ │ │ │ │ - // and the 'next' feature has a geometry. We do the geometry check │ │ │ │ │ - // because if all the features after the current one are 'null', we │ │ │ │ │ - // won't call eraseGeometry, so we break the 'renderer functions │ │ │ │ │ - // will always be called with locked=false *last*' rule. The end result │ │ │ │ │ - // is a possible gratiutious unlocking to save a loop through the rest │ │ │ │ │ - // of the list checking the remaining features every time. So long as │ │ │ │ │ - // null geoms are rare, this is probably okay. │ │ │ │ │ - if (i != 0 && features[i - 1].geometry) { │ │ │ │ │ - this.renderer.locked = true; │ │ │ │ │ - } else { │ │ │ │ │ - this.renderer.locked = false; │ │ │ │ │ - } │ │ │ │ │ + this.down(evt); │ │ │ │ │ + this.callback("down", [evt.xy]); │ │ │ │ │ │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ + // prevent document dragging │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ + if (!this.oldOnselectstart) { │ │ │ │ │ + this.oldOnselectstart = document.onselectstart ? │ │ │ │ │ + document.onselectstart : OpenLayers.Function.True; │ │ │ │ │ } │ │ │ │ │ + document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ │ │ │ │ │ - this.features = OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ - // feature has no layer at this point │ │ │ │ │ - feature.layer = null; │ │ │ │ │ + propagate = !this.stopDown; │ │ │ │ │ + } else { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + } │ │ │ │ │ + return propagate; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - this.renderer.eraseFeatures(feature); │ │ │ │ │ + /** │ │ │ │ │ + * Method: dragmove │ │ │ │ │ + * This private method is factorized from mousemove and touchmove methods │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ + */ │ │ │ │ │ + dragmove: function(evt) { │ │ │ │ │ + this.lastMoveEvt = evt; │ │ │ │ │ + if (this.started && !this.timeoutId && (evt.xy.x != this.last.x || │ │ │ │ │ + evt.xy.y != this.last.y)) { │ │ │ │ │ + if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ + if (evt.element === document) { │ │ │ │ │ + this.adjustXY(evt); │ │ │ │ │ + // do setEvent manually because the documentEvents are not │ │ │ │ │ + // registered with the map │ │ │ │ │ + this.setEvent(evt); │ │ │ │ │ + } else { │ │ │ │ │ + this.removeDocumentEvents(); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - //in the case that this feature is one of the selected features, │ │ │ │ │ - // remove it from that array as well. │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.selectedFeatures, feature) != -1) { │ │ │ │ │ - OpenLayers.Util.removeItem(this.selectedFeatures, feature); │ │ │ │ │ + if (this.interval > 0) { │ │ │ │ │ + this.timeoutId = setTimeout( │ │ │ │ │ + OpenLayers.Function.bind(this.removeTimeout, this), │ │ │ │ │ + this.interval); │ │ │ │ │ } │ │ │ │ │ + this.dragging = true; │ │ │ │ │ │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ + this.move(evt); │ │ │ │ │ + this.callback("move", [evt.xy]); │ │ │ │ │ + if (!this.oldOnselectstart) { │ │ │ │ │ + this.oldOnselectstart = document.onselectstart; │ │ │ │ │ + document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ } │ │ │ │ │ + this.last = evt.xy; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: removeAllFeatures │ │ │ │ │ - * Remove all features from the layer. │ │ │ │ │ + /** │ │ │ │ │ + * Method: dragend │ │ │ │ │ + * This private method is factorized from mouseup and touchend methods │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional properties for changing behavior of the │ │ │ │ │ - * removal. │ │ │ │ │ + * evt - {Event} The event │ │ │ │ │ * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * silent - {Boolean} Supress event triggering. Default is false. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - removeAllFeatures: function(options) { │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ - var features = this.features; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent( │ │ │ │ │ - "beforefeaturesremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ + dragend: function(evt) { │ │ │ │ │ + if (this.started) { │ │ │ │ │ + if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ + this.adjustXY(evt); │ │ │ │ │ + this.removeDocumentEvents(); │ │ │ │ │ } │ │ │ │ │ - feature.layer = null; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ + var dragged = (this.start != this.last); │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + OpenLayers.Element.removeClass( │ │ │ │ │ + this.map.viewPortDiv, "olDragDown" │ │ │ │ │ + ); │ │ │ │ │ + this.up(evt); │ │ │ │ │ + this.callback("up", [evt.xy]); │ │ │ │ │ + if (dragged) { │ │ │ │ │ + this.callback("done", [evt.xy]); │ │ │ │ │ } │ │ │ │ │ + document.onselectstart = this.oldOnselectstart; │ │ │ │ │ } │ │ │ │ │ - this.renderer.clear(); │ │ │ │ │ - this.features = []; │ │ │ │ │ - this.unrenderedFeatures = {}; │ │ │ │ │ - this.selectedFeatures = []; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroyFeatures │ │ │ │ │ - * Erase and destroy features on the layer. │ │ │ │ │ + * The four methods below (down, move, up, and out) are used by subclasses │ │ │ │ │ + * to do their own processing related to these mouse events. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: down │ │ │ │ │ + * This method is called during the handling of the mouse down event. │ │ │ │ │ + * Subclasses can do their own processing here. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array()} An optional array of │ │ │ │ │ - * features to destroy. If not supplied, all features on the layer │ │ │ │ │ - * will be destroyed. │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * evt - {Event} The mouse down event │ │ │ │ │ */ │ │ │ │ │ - destroyFeatures: function(features, options) { │ │ │ │ │ - var all = (features == undefined); // evaluates to true if │ │ │ │ │ - // features is null │ │ │ │ │ - if (all) { │ │ │ │ │ - features = this.features; │ │ │ │ │ - } │ │ │ │ │ - if (features) { │ │ │ │ │ - this.removeFeatures(features, options); │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - features[i].destroy(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + down: function(evt) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: drawFeature │ │ │ │ │ - * Draw (or redraw) a feature on the layer. If the optional style argument │ │ │ │ │ - * is included, this style will be used. If no style is included, the │ │ │ │ │ - * feature's style will be used. If the feature doesn't have a style, │ │ │ │ │ - * the layer's style will be used. │ │ │ │ │ - * │ │ │ │ │ - * This function is not designed to be used when adding features to │ │ │ │ │ - * the layer (use addFeatures instead). It is meant to be used when │ │ │ │ │ - * the style of a feature has changed, or in some other way needs to │ │ │ │ │ - * visually updated *after* it has already been added to a layer. You │ │ │ │ │ - * must add the feature to the layer for most layer-related events to │ │ │ │ │ - * happen. │ │ │ │ │ + * Method: move │ │ │ │ │ + * This method is called during the handling of the mouse move event. │ │ │ │ │ + * Subclasses can do their own processing here. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The mouse move event │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {} │ │ │ │ │ - * style - {String | Object} Named render intent or full symbolizer object. │ │ │ │ │ */ │ │ │ │ │ - drawFeature: function(feature, style) { │ │ │ │ │ - // don't try to draw the feature with the renderer if the layer is not │ │ │ │ │ - // drawn itself │ │ │ │ │ - if (!this.drawn) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (typeof style != "object") { │ │ │ │ │ - if (!style && feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ - style = "delete"; │ │ │ │ │ - } │ │ │ │ │ - var renderIntent = style || feature.renderIntent; │ │ │ │ │ - style = feature.style || this.style; │ │ │ │ │ - if (!style) { │ │ │ │ │ - style = this.styleMap.createSymbolizer(feature, renderIntent); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + move: function(evt) {}, │ │ │ │ │ │ │ │ │ │ - var drawn = this.renderer.drawFeature(feature, style); │ │ │ │ │ - //TODO remove the check for null when we get rid of Renderer.SVG │ │ │ │ │ - if (drawn === false || drawn === null) { │ │ │ │ │ - this.unrenderedFeatures[feature.id] = feature; │ │ │ │ │ - } else { │ │ │ │ │ - delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: up │ │ │ │ │ + * This method is called during the handling of the mouse up event. │ │ │ │ │ + * Subclasses can do their own processing here. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The mouse up event │ │ │ │ │ + */ │ │ │ │ │ + up: function(evt) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: out │ │ │ │ │ + * This method is called during the handling of the mouse out event. │ │ │ │ │ + * Subclasses can do their own processing here. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The mouse out event │ │ │ │ │ + */ │ │ │ │ │ + out: function(evt) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * The methods below are part of the magic of event handling. Because │ │ │ │ │ + * they are named like browser events, they are registered as listeners │ │ │ │ │ + * for the events they represent. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: mousedown │ │ │ │ │ + * Handle mousedown events │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ + */ │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + return this.dragstart(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: eraseFeatures │ │ │ │ │ - * Erase features from the layer. │ │ │ │ │ + * Method: touchstart │ │ │ │ │ + * Handle touchstart events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array()} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - eraseFeatures: function(features) { │ │ │ │ │ - this.renderer.eraseFeatures(features); │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + return this.dragstart(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureFromEvent │ │ │ │ │ - * Given an event, return a feature if the event occurred over one. │ │ │ │ │ - * Otherwise, return null. │ │ │ │ │ + * Method: mousemove │ │ │ │ │ + * Handle mousemove events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {} A feature if one was under the event. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - getFeatureFromEvent: function(evt) { │ │ │ │ │ - if (!this.renderer) { │ │ │ │ │ - throw new Error('getFeatureFromEvent called on layer with no ' + │ │ │ │ │ - 'renderer. This usually means you destroyed a ' + │ │ │ │ │ - 'layer, but not some handler which is associated ' + │ │ │ │ │ - 'with it.'); │ │ │ │ │ - } │ │ │ │ │ - var feature = null; │ │ │ │ │ - var featureId = this.renderer.getFeatureIdFromEvent(evt); │ │ │ │ │ - if (featureId) { │ │ │ │ │ - if (typeof featureId === "string") { │ │ │ │ │ - feature = this.getFeatureById(featureId); │ │ │ │ │ - } else { │ │ │ │ │ - feature = featureId; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return feature; │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + return this.dragmove(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getFeatureBy │ │ │ │ │ - * Given a property value, return the feature if it exists in the features array │ │ │ │ │ + * Method: touchmove │ │ │ │ │ + * Handle touchmove events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * property - {String} │ │ │ │ │ - * value - {String} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {} A feature corresponding to the given │ │ │ │ │ - * property value or null if there is no such feature. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - getFeatureBy: function(property, value) { │ │ │ │ │ - //TBD - would it be more efficient to use a hash for this.features? │ │ │ │ │ - var feature = null; │ │ │ │ │ - for (var i = 0, len = this.features.length; i < len; ++i) { │ │ │ │ │ - if (this.features[i][property] == value) { │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + return this.dragmove(evt); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeTimeout │ │ │ │ │ + * Private. Called by mousemove() to remove the drag timeout. │ │ │ │ │ + */ │ │ │ │ │ + removeTimeout: function() { │ │ │ │ │ + this.timeoutId = null; │ │ │ │ │ + // if timeout expires while we're still dragging (mouseup │ │ │ │ │ + // hasn't occurred) then call mousemove to move to the │ │ │ │ │ + // correct position │ │ │ │ │ + if (this.dragging) { │ │ │ │ │ + this.mousemove(this.lastMoveEvt); │ │ │ │ │ } │ │ │ │ │ - return feature; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getFeatureById │ │ │ │ │ - * Given a feature id, return the feature if it exists in the features array │ │ │ │ │ + * Method: mouseup │ │ │ │ │ + * Handle mouseup events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {} A feature corresponding to the given │ │ │ │ │ - * featureId or null if there is no such feature. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - getFeatureById: function(featureId) { │ │ │ │ │ - return this.getFeatureBy('id', featureId); │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + return this.dragend(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getFeatureByFid │ │ │ │ │ - * Given a feature fid, return the feature if it exists in the features array │ │ │ │ │ + * Method: touchend │ │ │ │ │ + * Handle touchend events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * featureFid - {String} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {} A feature corresponding to the given │ │ │ │ │ - * featureFid or null if there is no such feature. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - getFeatureByFid: function(featureFid) { │ │ │ │ │ - return this.getFeatureBy('fid', featureFid); │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + // override evt.xy with last position since touchend does not have │ │ │ │ │ + // any touch position │ │ │ │ │ + evt.xy = this.last; │ │ │ │ │ + return this.dragend(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getFeaturesByAttribute │ │ │ │ │ - * Returns an array of features that have the given attribute key set to the │ │ │ │ │ - * given value. Comparison of attribute values takes care of datatypes, e.g. │ │ │ │ │ - * the string '1234' is not equal to the number 1234. │ │ │ │ │ + * Method: mouseout │ │ │ │ │ + * Handle mouseout events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * attrName - {String} │ │ │ │ │ - * attrValue - {Mixed} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * Array({}) An array of features that have the │ │ │ │ │ - * passed named attribute set to the given value. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - getFeaturesByAttribute: function(attrName, attrValue) { │ │ │ │ │ - var i, │ │ │ │ │ - feature, │ │ │ │ │ - len = this.features.length, │ │ │ │ │ - foundFeatures = []; │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - if (feature && feature.attributes) { │ │ │ │ │ - if (feature.attributes[attrName] === attrValue) { │ │ │ │ │ - foundFeatures.push(feature); │ │ │ │ │ + mouseout: function(evt) { │ │ │ │ │ + if (this.started && OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ + if (this.documentDrag === true) { │ │ │ │ │ + this.addDocumentEvents(); │ │ │ │ │ + } else { │ │ │ │ │ + var dragged = (this.start != this.last); │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + OpenLayers.Element.removeClass( │ │ │ │ │ + this.map.viewPortDiv, "olDragDown" │ │ │ │ │ + ); │ │ │ │ │ + this.out(evt); │ │ │ │ │ + this.callback("out", []); │ │ │ │ │ + if (dragged) { │ │ │ │ │ + this.callback("done", [evt.xy]); │ │ │ │ │ + } │ │ │ │ │ + if (document.onselectstart) { │ │ │ │ │ + document.onselectstart = this.oldOnselectstart; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return foundFeatures; │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Unselect the selected features │ │ │ │ │ - * i.e. clears the featureSelection array │ │ │ │ │ - * change the style back │ │ │ │ │ - clearSelection: function() { │ │ │ │ │ + * Method: click │ │ │ │ │ + * The drag handler captures the click event. If something else registers │ │ │ │ │ + * for clicks on the same element, its listener will not be called │ │ │ │ │ + * after a drag. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ + */ │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + // let the click event propagate only if the mouse moved │ │ │ │ │ + return (this.start == this.last); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var vectorLayer = this.map.vectorLayer; │ │ │ │ │ - for (var i = 0; i < this.map.featureSelection.length; i++) { │ │ │ │ │ - var featureSelection = this.map.featureSelection[i]; │ │ │ │ │ - vectorLayer.drawFeature(featureSelection, vectorLayer.style); │ │ │ │ │ + /** │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activate the handler. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was successfully activated. │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + activated = true; │ │ │ │ │ } │ │ │ │ │ - this.map.featureSelection = []; │ │ │ │ │ + return activated; │ │ │ │ │ }, │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: onFeatureInsert │ │ │ │ │ - * method called after a feature is inserted. │ │ │ │ │ - * Does nothing by default. Override this if you │ │ │ │ │ - * need to do something on feature updates. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {} │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivate the handler. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ - onFeatureInsert: function(feature) {}, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + deactivated = true; │ │ │ │ │ + OpenLayers.Element.removeClass( │ │ │ │ │ + this.map.viewPortDiv, "olDragDown" │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: preFeatureInsert │ │ │ │ │ - * method called before a feature is inserted. │ │ │ │ │ - * Does nothing by default. Override this if you │ │ │ │ │ - * need to do something when features are first added to the │ │ │ │ │ - * layer, but before they are drawn, such as adjust the style. │ │ │ │ │ - * │ │ │ │ │ + * Method: adjustXY │ │ │ │ │ + * Converts event coordinates that are relative to the document body to │ │ │ │ │ + * ones that are relative to the map viewport. The latter is the default in │ │ │ │ │ + * OpenLayers. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {} │ │ │ │ │ + * evt - {Object} │ │ │ │ │ */ │ │ │ │ │ - preFeatureInsert: function(feature) {}, │ │ │ │ │ + adjustXY: function(evt) { │ │ │ │ │ + var pos = OpenLayers.Util.pagePosition(this.map.viewPortDiv); │ │ │ │ │ + evt.xy.x -= pos[0]; │ │ │ │ │ + evt.xy.y -= pos[1]; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getDataExtent │ │ │ │ │ - * Calculates the max extent which includes all of the features. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} or null if the layer has no features with │ │ │ │ │ - * geometries. │ │ │ │ │ + /** │ │ │ │ │ + * Method: addDocumentEvents │ │ │ │ │ + * Start observing document events when documentDrag is true and the mouse │ │ │ │ │ + * cursor leaves the map viewport while dragging. │ │ │ │ │ */ │ │ │ │ │ - getDataExtent: function() { │ │ │ │ │ - var maxExtent = null; │ │ │ │ │ - var features = this.features; │ │ │ │ │ - if (features && (features.length > 0)) { │ │ │ │ │ - var geometry = null; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - geometry = features[i].geometry; │ │ │ │ │ - if (geometry) { │ │ │ │ │ - if (maxExtent === null) { │ │ │ │ │ - maxExtent = new OpenLayers.Bounds(); │ │ │ │ │ - } │ │ │ │ │ - maxExtent.extend(geometry.getBounds()); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return maxExtent; │ │ │ │ │ + addDocumentEvents: function() { │ │ │ │ │ + OpenLayers.Element.addClass(document.body, "olDragDown"); │ │ │ │ │ + this.documentEvents = true; │ │ │ │ │ + OpenLayers.Event.observe(document, "mousemove", this._docMove); │ │ │ │ │ + OpenLayers.Event.observe(document, "mouseup", this._docUp); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Vector" │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeDocumentEvents │ │ │ │ │ + * Stops observing document events when documentDrag is true and the mouse │ │ │ │ │ + * cursor re-enters the map viewport while dragging. │ │ │ │ │ + */ │ │ │ │ │ + removeDocumentEvents: function() { │ │ │ │ │ + OpenLayers.Element.removeClass(document.body, "olDragDown"); │ │ │ │ │ + this.documentEvents = false; │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mousemove", this._docMove); │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mouseup", this._docUp); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Drag" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/PointTrack.js │ │ │ │ │ + OpenLayers/Handler/RegularPolygon.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ + * @requires OpenLayers/Handler/Drag.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.PointTrack │ │ │ │ │ - * Vector layer to display ordered point features as a line, creating one │ │ │ │ │ - * LineString feature for each pair of two points. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Handler.RegularPolygon │ │ │ │ │ + * Handler to draw a regular polygon on the map. Polygon is displayed on mouse │ │ │ │ │ + * down, moves or is modified on mouse move, and is finished on mouse up. │ │ │ │ │ + * The handler triggers callbacks for 'done' and 'cancel'. Create a new │ │ │ │ │ + * instance with the constructor. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ + * - │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.PointTrack = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ +OpenLayers.Handler.RegularPolygon = OpenLayers.Class(OpenLayers.Handler.Drag, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: dataFrom │ │ │ │ │ - * {} or │ │ │ │ │ - * {} optional. If the lines │ │ │ │ │ - * should get the data/attributes from one of the two points it is │ │ │ │ │ - * composed of, which one should it be? │ │ │ │ │ + * APIProperty: sides │ │ │ │ │ + * {Integer} Number of sides for the regular polygon. Needs to be greater │ │ │ │ │ + * than 2. Defaults to 4. │ │ │ │ │ */ │ │ │ │ │ - dataFrom: null, │ │ │ │ │ + sides: 4, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: styleFrom │ │ │ │ │ - * {} or │ │ │ │ │ - * {} optional. If the lines │ │ │ │ │ - * should get the style from one of the two points it is composed of, │ │ │ │ │ - * which one should it be? │ │ │ │ │ + * APIProperty: radius │ │ │ │ │ + * {Float} Optional radius in map units of the regular polygon. If this is │ │ │ │ │ + * set to some non-zero value, a polygon with a fixed radius will be │ │ │ │ │ + * drawn and dragged with mose movements. If this property is not │ │ │ │ │ + * set, dragging changes the radius of the polygon. Set to null by │ │ │ │ │ + * default. │ │ │ │ │ */ │ │ │ │ │ - styleFrom: null, │ │ │ │ │ + radius: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.PointTrack │ │ │ │ │ - * Constructor for a new OpenLayers.PointTrack instance. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} name of the layer │ │ │ │ │ - * options - {Object} Optional object with properties to tag onto the │ │ │ │ │ - * instance. │ │ │ │ │ + * APIProperty: snapAngle │ │ │ │ │ + * {Float} If set to a non-zero value, the handler will snap the polygon │ │ │ │ │ + * rotation to multiples of the snapAngle. Value is an angle measured │ │ │ │ │ + * in degrees counterclockwise from the positive x-axis. │ │ │ │ │ */ │ │ │ │ │ + snapAngle: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: addNodes │ │ │ │ │ - * Adds point features that will be used to create lines from, using point │ │ │ │ │ - * pairs. The first point of a pair will be the source node, the second │ │ │ │ │ - * will be the target node. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pointFeatures - {Array()} │ │ │ │ │ - * options - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Supported options: │ │ │ │ │ - * silent - {Boolean} true to suppress (before)feature(s)added events │ │ │ │ │ + * APIProperty: snapToggle │ │ │ │ │ + * {String} If set, snapToggle is checked on mouse events and will set │ │ │ │ │ + * the snap mode to the opposite of what it currently is. To disallow │ │ │ │ │ + * toggling between snap and non-snap mode, set freehandToggle to │ │ │ │ │ + * null. Acceptable toggle values are 'shiftKey', 'ctrlKey', and │ │ │ │ │ + * 'altKey'. Snap mode is only possible if this.snapAngle is set to a │ │ │ │ │ + * non-zero value. │ │ │ │ │ */ │ │ │ │ │ - addNodes: function(pointFeatures, options) { │ │ │ │ │ - if (pointFeatures.length < 2) { │ │ │ │ │ - throw new Error("At least two point features have to be added to " + │ │ │ │ │ - "create a line from"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var lines = new Array(pointFeatures.length - 1); │ │ │ │ │ - │ │ │ │ │ - var pointFeature, startPoint, endPoint; │ │ │ │ │ - for (var i = 0, len = pointFeatures.length; i < len; i++) { │ │ │ │ │ - pointFeature = pointFeatures[i]; │ │ │ │ │ - endPoint = pointFeature.geometry; │ │ │ │ │ - │ │ │ │ │ - if (!endPoint) { │ │ │ │ │ - var lonlat = pointFeature.lonlat; │ │ │ │ │ - endPoint = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ - } else if (endPoint.CLASS_NAME != "OpenLayers.Geometry.Point") { │ │ │ │ │ - throw new TypeError("Only features with point geometries are supported."); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (i > 0) { │ │ │ │ │ - var attributes = (this.dataFrom != null) ? │ │ │ │ │ - (pointFeatures[i + this.dataFrom].data || │ │ │ │ │ - pointFeatures[i + this.dataFrom].attributes) : │ │ │ │ │ - null; │ │ │ │ │ - var style = (this.styleFrom != null) ? │ │ │ │ │ - (pointFeatures[i + this.styleFrom].style) : │ │ │ │ │ - null; │ │ │ │ │ - var line = new OpenLayers.Geometry.LineString([startPoint, │ │ │ │ │ - endPoint │ │ │ │ │ - ]); │ │ │ │ │ - │ │ │ │ │ - lines[i - 1] = new OpenLayers.Feature.Vector(line, attributes, │ │ │ │ │ - style); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - startPoint = endPoint; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.addFeatures(lines, options); │ │ │ │ │ - }, │ │ │ │ │ + snapToggle: 'shiftKey', │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.PointTrack" │ │ │ │ │ -}); │ │ │ │ │ + /** │ │ │ │ │ + * Property: layerOptions │ │ │ │ │ + * {Object} Any optional properties to be set on the sketch layer. │ │ │ │ │ + */ │ │ │ │ │ + layerOptions: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Layer.PointTrack.SOURCE_NODE │ │ │ │ │ - * {Number} value for and │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.PointTrack.SOURCE_NODE = -1; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: persist │ │ │ │ │ + * {Boolean} Leave the feature rendered until clear is called. Default │ │ │ │ │ + * is false. If set to true, the feature remains rendered until │ │ │ │ │ + * clear is called, typically by deactivating the handler or starting │ │ │ │ │ + * another drawing. │ │ │ │ │ + */ │ │ │ │ │ + persist: false, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Layer.PointTrack.TARGET_NODE │ │ │ │ │ - * {Number} value for and │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.PointTrack.TARGET_NODE = 0; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: irregular │ │ │ │ │ + * {Boolean} Draw an irregular polygon instead of a regular polygon. │ │ │ │ │ + * Default is false. If true, the initial mouse down will represent │ │ │ │ │ + * one corner of the polygon bounds and with each mouse movement, the │ │ │ │ │ + * polygon will be stretched so the opposite corner of its bounds │ │ │ │ │ + * follows the mouse position. This property takes precedence over │ │ │ │ │ + * the radius property. If set to true, the radius property will │ │ │ │ │ + * be ignored. │ │ │ │ │ + */ │ │ │ │ │ + irregular: false, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Layer.PointTrack.dataFrom │ │ │ │ │ - * {Object} with the following keys - *deprecated* │ │ │ │ │ - * - SOURCE_NODE: take data/attributes from the source node of the line │ │ │ │ │ - * - TARGET_NODE: take data/attributes from the target node of the line │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.PointTrack.dataFrom = { │ │ │ │ │ - 'SOURCE_NODE': -1, │ │ │ │ │ - 'TARGET_NODE': 0 │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Markers.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: citeCompliant │ │ │ │ │ + * {Boolean} If set to true, coordinates of features drawn in a map extent │ │ │ │ │ + * crossing the date line won't exceed the world bounds. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + citeCompliant: false, │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: angle │ │ │ │ │ + * {Float} The angle from the origin (mouse down) to the current mouse │ │ │ │ │ + * position, in radians. This is measured counterclockwise from the │ │ │ │ │ + * positive x-axis. │ │ │ │ │ + */ │ │ │ │ │ + angle: null, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: fixedRadius │ │ │ │ │ + * {Boolean} The polygon has a fixed radius. True if a radius is set before │ │ │ │ │ + * drawing begins. False otherwise. │ │ │ │ │ + */ │ │ │ │ │ + fixedRadius: false, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: feature │ │ │ │ │ + * {} The currently drawn polygon feature │ │ │ │ │ + */ │ │ │ │ │ + feature: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.Markers │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.Markers = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ + /** │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {} The temporary drawing layer │ │ │ │ │ + */ │ │ │ │ │ + layer: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} Markers layer is never a base layer. │ │ │ │ │ + /** │ │ │ │ │ + * Property: origin │ │ │ │ │ + * {} Location of the first mouse down │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: false, │ │ │ │ │ + origin: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: markers │ │ │ │ │ - * {Array()} internal marker list │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Handler.RegularPolygon │ │ │ │ │ + * Create a new regular polygon handler. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * control - {} The control that owns this handler │ │ │ │ │ + * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ + * functions. Various callbacks described below. │ │ │ │ │ + * options - {Object} An object with properties to be set on the handler. │ │ │ │ │ + * If the options.sides property is not specified, the number of sides │ │ │ │ │ + * will default to 4. │ │ │ │ │ + * │ │ │ │ │ + * Named callbacks: │ │ │ │ │ + * create - Called when a sketch is first created. Callback called with │ │ │ │ │ + * the creation point geometry and sketch feature. │ │ │ │ │ + * done - Called when the sketch drawing is finished. The callback will │ │ │ │ │ + * recieve a single argument, the sketch geometry. │ │ │ │ │ + * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ + * cancel callback will receive a geometry. │ │ │ │ │ */ │ │ │ │ │ - markers: null, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ + this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style['default'], {}); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ + OpenLayers.Handler.Drag.prototype.initialize.apply(this, │ │ │ │ │ + [control, callbacks, options]); │ │ │ │ │ + this.options = (options) ? options : {}; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: drawn │ │ │ │ │ - * {Boolean} internal state of drawing. This is a workaround for the fact │ │ │ │ │ - * that the map does not call moveTo with a zoomChanged when the map is │ │ │ │ │ - * first starting up. This lets us catch the case where we have *never* │ │ │ │ │ - * drawn the layer, and draw it even if the zoom hasn't changed. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setOptions │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newOptions - {Object} │ │ │ │ │ */ │ │ │ │ │ - drawn: false, │ │ │ │ │ + setOptions: function(newOptions) { │ │ │ │ │ + OpenLayers.Util.extend(this.options, newOptions); │ │ │ │ │ + OpenLayers.Util.extend(this, newOptions); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Markers │ │ │ │ │ - * Create a Markers layer. │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Turn on the handler. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was successfully activated │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.markers = []; │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.Drag.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + // create temporary vector layer for rendering geometry sketch │ │ │ │ │ + var options = OpenLayers.Util.extend({ │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + // indicate that the temp vector layer will never be out of range │ │ │ │ │ + // without this, resolution properties must be specified at the │ │ │ │ │ + // map-level for this temporary layer to init its resolutions │ │ │ │ │ + // correctly │ │ │ │ │ + calculateInRange: OpenLayers.Function.True, │ │ │ │ │ + wrapDateLine: this.citeCompliant │ │ │ │ │ + }, this.layerOptions); │ │ │ │ │ + this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ + this.map.addLayer(this.layer); │ │ │ │ │ + activated = true; │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Turn off the handler. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was successfully deactivated │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.clearMarkers(); │ │ │ │ │ - this.markers = null; │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.Drag.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + // call the cancel callback if mid-drawing │ │ │ │ │ + if (this.dragging) { │ │ │ │ │ + this.cancel(); │ │ │ │ │ + } │ │ │ │ │ + // If a layer's map property is set to null, it means that that │ │ │ │ │ + // layer isn't added to the map. Since we ourself added the layer │ │ │ │ │ + // to the map in activate(), we can assume that if this.layer.map │ │ │ │ │ + // is null it means that the layer has been destroyed (as a result │ │ │ │ │ + // of map.destroy() for example. │ │ │ │ │ + if (this.layer.map != null) { │ │ │ │ │ + this.layer.destroy(false); │ │ │ │ │ + if (this.feature) { │ │ │ │ │ + this.feature.destroy(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.layer = null; │ │ │ │ │ + this.feature = null; │ │ │ │ │ + deactivated = true; │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setOpacity │ │ │ │ │ - * Sets the opacity for all the markers. │ │ │ │ │ - * │ │ │ │ │ + * Method: down │ │ │ │ │ + * Start drawing a new feature │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * opacity - {Float} │ │ │ │ │ + * evt - {Event} The drag start event │ │ │ │ │ */ │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - if (opacity != this.opacity) { │ │ │ │ │ - this.opacity = opacity; │ │ │ │ │ - for (var i = 0, len = this.markers.length; i < len; i++) { │ │ │ │ │ - this.markers[i].setOpacity(this.opacity); │ │ │ │ │ - } │ │ │ │ │ + down: function(evt) { │ │ │ │ │ + this.fixedRadius = !!(this.radius); │ │ │ │ │ + var maploc = this.layer.getLonLatFromViewPortPx(evt.xy); │ │ │ │ │ + this.origin = new OpenLayers.Geometry.Point(maploc.lon, maploc.lat); │ │ │ │ │ + // create the new polygon │ │ │ │ │ + if (!this.fixedRadius || this.irregular) { │ │ │ │ │ + // smallest radius should not be less one pixel in map units │ │ │ │ │ + // VML doesn't behave well with smaller │ │ │ │ │ + this.radius = this.map.getResolution(); │ │ │ │ │ } │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.clear(); │ │ │ │ │ + } │ │ │ │ │ + this.feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ + this.createGeometry(); │ │ │ │ │ + this.callback("create", [this.origin, this.feature]); │ │ │ │ │ + this.layer.addFeatures([this.feature], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.layer.drawFeature(this.feature, this.style); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ + /** │ │ │ │ │ + * Method: move │ │ │ │ │ + * Respond to drag move events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {} │ │ │ │ │ - * zoomChanged - {Boolean} │ │ │ │ │ - * dragging - {Boolean} │ │ │ │ │ + * evt - {Evt} The move event │ │ │ │ │ */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - if (zoomChanged || !this.drawn) { │ │ │ │ │ - for (var i = 0, len = this.markers.length; i < len; i++) { │ │ │ │ │ - this.drawMarker(this.markers[i]); │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + var maploc = this.layer.getLonLatFromViewPortPx(evt.xy); │ │ │ │ │ + var point = new OpenLayers.Geometry.Point(maploc.lon, maploc.lat); │ │ │ │ │ + if (this.irregular) { │ │ │ │ │ + var ry = Math.sqrt(2) * Math.abs(point.y - this.origin.y) / 2; │ │ │ │ │ + this.radius = Math.max(this.map.getResolution() / 2, ry); │ │ │ │ │ + } else if (this.fixedRadius) { │ │ │ │ │ + this.origin = point; │ │ │ │ │ + } else { │ │ │ │ │ + this.calculateAngle(point, evt); │ │ │ │ │ + this.radius = Math.max(this.map.getResolution() / 2, │ │ │ │ │ + point.distanceTo(this.origin)); │ │ │ │ │ + } │ │ │ │ │ + this.modifyGeometry(); │ │ │ │ │ + if (this.irregular) { │ │ │ │ │ + var dx = point.x - this.origin.x; │ │ │ │ │ + var dy = point.y - this.origin.y; │ │ │ │ │ + var ratio; │ │ │ │ │ + if (dy == 0) { │ │ │ │ │ + ratio = dx / (this.radius * Math.sqrt(2)); │ │ │ │ │ + } else { │ │ │ │ │ + ratio = dx / dy; │ │ │ │ │ } │ │ │ │ │ - this.drawn = true; │ │ │ │ │ + this.feature.geometry.resize(1, this.origin, ratio); │ │ │ │ │ + this.feature.geometry.move(dx / 2, dy / 2); │ │ │ │ │ } │ │ │ │ │ + this.layer.drawFeature(this.feature, this.style); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: addMarker │ │ │ │ │ + * Method: up │ │ │ │ │ + * Finish drawing the feature │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * marker - {} │ │ │ │ │ + * evt - {Event} The mouse up event │ │ │ │ │ */ │ │ │ │ │ - addMarker: function(marker) { │ │ │ │ │ - this.markers.push(marker); │ │ │ │ │ - │ │ │ │ │ - if (this.opacity < 1) { │ │ │ │ │ - marker.setOpacity(this.opacity); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.map && this.map.getExtent()) { │ │ │ │ │ - marker.map = this.map; │ │ │ │ │ - this.drawMarker(marker); │ │ │ │ │ + up: function(evt) { │ │ │ │ │ + this.finalize(); │ │ │ │ │ + // the mouseup method of superclass doesn't call the │ │ │ │ │ + // "done" callback if there's been no move between │ │ │ │ │ + // down and up │ │ │ │ │ + if (this.start == this.last) { │ │ │ │ │ + this.callback("done", [evt.xy]); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: removeMarker │ │ │ │ │ + * Method: out │ │ │ │ │ + * Finish drawing the feature. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * marker - {} │ │ │ │ │ + * evt - {Event} The mouse out event │ │ │ │ │ */ │ │ │ │ │ - removeMarker: function(marker) { │ │ │ │ │ - if (this.markers && this.markers.length) { │ │ │ │ │ - OpenLayers.Util.removeItem(this.markers, marker); │ │ │ │ │ - marker.erase(); │ │ │ │ │ + out: function(evt) { │ │ │ │ │ + this.finalize(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createGeometry │ │ │ │ │ + * Create the new polygon geometry. This is called at the start of the │ │ │ │ │ + * drag and at any point during the drag if the number of sides │ │ │ │ │ + * changes. │ │ │ │ │ + */ │ │ │ │ │ + createGeometry: function() { │ │ │ │ │ + this.angle = Math.PI * ((1 / this.sides) - (1 / 2)); │ │ │ │ │ + if (this.snapAngle) { │ │ │ │ │ + this.angle += this.snapAngle * (Math.PI / 180); │ │ │ │ │ } │ │ │ │ │ + this.feature.geometry = OpenLayers.Geometry.Polygon.createRegularPolygon( │ │ │ │ │ + this.origin, this.radius, this.sides, this.snapAngle │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clearMarkers │ │ │ │ │ - * This method removes all markers from a layer. The markers are not │ │ │ │ │ - * destroyed by this function, but are removed from the list of markers. │ │ │ │ │ + * Method: modifyGeometry │ │ │ │ │ + * Modify the polygon geometry in place. │ │ │ │ │ */ │ │ │ │ │ - clearMarkers: function() { │ │ │ │ │ - if (this.markers != null) { │ │ │ │ │ - while (this.markers.length > 0) { │ │ │ │ │ - this.removeMarker(this.markers[0]); │ │ │ │ │ - } │ │ │ │ │ + modifyGeometry: function() { │ │ │ │ │ + var angle, point; │ │ │ │ │ + var ring = this.feature.geometry.components[0]; │ │ │ │ │ + // if the number of sides ever changes, create a new geometry │ │ │ │ │ + if (ring.components.length != (this.sides + 1)) { │ │ │ │ │ + this.createGeometry(); │ │ │ │ │ + ring = this.feature.geometry.components[0]; │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0; i < this.sides; ++i) { │ │ │ │ │ + point = ring.components[i]; │ │ │ │ │ + angle = this.angle + (i * 2 * Math.PI / this.sides); │ │ │ │ │ + point.x = this.origin.x + (this.radius * Math.cos(angle)); │ │ │ │ │ + point.y = this.origin.y + (this.radius * Math.sin(angle)); │ │ │ │ │ + point.clearBounds(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawMarker │ │ │ │ │ - * Calculate the pixel location for the marker, create it, and │ │ │ │ │ - * add it to the layer's div │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateAngle │ │ │ │ │ + * Calculate the angle based on settings. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * marker - {} │ │ │ │ │ + * point - {} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - drawMarker: function(marker) { │ │ │ │ │ - var px = this.map.getLayerPxFromLonLat(marker.lonlat); │ │ │ │ │ - if (px == null) { │ │ │ │ │ - marker.display(false); │ │ │ │ │ + calculateAngle: function(point, evt) { │ │ │ │ │ + var alpha = Math.atan2(point.y - this.origin.y, │ │ │ │ │ + point.x - this.origin.x); │ │ │ │ │ + if (this.snapAngle && (this.snapToggle && !evt[this.snapToggle])) { │ │ │ │ │ + var snapAngleRad = (Math.PI / 180) * this.snapAngle; │ │ │ │ │ + this.angle = Math.round(alpha / snapAngleRad) * snapAngleRad; │ │ │ │ │ } else { │ │ │ │ │ - if (!marker.isDrawn()) { │ │ │ │ │ - var markerImg = marker.draw(px); │ │ │ │ │ - this.div.appendChild(markerImg); │ │ │ │ │ - } else if (marker.icon) { │ │ │ │ │ - marker.icon.moveTo(px); │ │ │ │ │ - } │ │ │ │ │ + this.angle = alpha; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getDataExtent │ │ │ │ │ - * Calculates the max extent which includes all of the markers. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: cancel │ │ │ │ │ + * Finish the geometry and call the "cancel" callback. │ │ │ │ │ */ │ │ │ │ │ - getDataExtent: function() { │ │ │ │ │ - var maxExtent = null; │ │ │ │ │ + cancel: function() { │ │ │ │ │ + // the polygon geometry gets cloned in the callback method │ │ │ │ │ + this.callback("cancel", null); │ │ │ │ │ + this.finalize(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (this.markers && (this.markers.length > 0)) { │ │ │ │ │ - var maxExtent = new OpenLayers.Bounds(); │ │ │ │ │ - for (var i = 0, len = this.markers.length; i < len; i++) { │ │ │ │ │ - var marker = this.markers[i]; │ │ │ │ │ - maxExtent.extend(marker.lonlat); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: finalize │ │ │ │ │ + * Finish the geometry and call the "done" callback. │ │ │ │ │ + */ │ │ │ │ │ + finalize: function() { │ │ │ │ │ + this.origin = null; │ │ │ │ │ + this.radius = this.options.radius; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clear │ │ │ │ │ + * Clear any rendered features on the temporary layer. This is called │ │ │ │ │ + * when the handler is deactivated, canceled, or done (unless persist │ │ │ │ │ + * is true). │ │ │ │ │ + */ │ │ │ │ │ + clear: function() { │ │ │ │ │ + if (this.layer) { │ │ │ │ │ + this.layer.renderer.clear(); │ │ │ │ │ + this.layer.destroyFeatures(); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return maxExtent; │ │ │ │ │ + /** │ │ │ │ │ + * Method: callback │ │ │ │ │ + * Trigger the control's named callback with the given arguments │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} The key for the callback that is one of the properties │ │ │ │ │ + * of the handler's callbacks object. │ │ │ │ │ + * args - {Array} An array of arguments with which to call the callback │ │ │ │ │ + * (defined by the control). │ │ │ │ │ + */ │ │ │ │ │ + callback: function(name, args) { │ │ │ │ │ + // override the callback method to always send the polygon geometry │ │ │ │ │ + if (this.callbacks[name]) { │ │ │ │ │ + this.callbacks[name].apply(this.control, │ │ │ │ │ + [this.feature.geometry.clone()]); │ │ │ │ │ + } │ │ │ │ │ + // since sketch features are added to the temporary layer │ │ │ │ │ + // they must be cleared here if done or cancel │ │ │ │ │ + if (!this.persist && (name == "done" || name == "cancel")) { │ │ │ │ │ + this.clear(); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Markers" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.RegularPolygon" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Boxes.js │ │ │ │ │ + OpenLayers/Handler/Point.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer.js │ │ │ │ │ - * @requires OpenLayers/Layer/Markers.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.Boxes │ │ │ │ │ - * Draw divs as 'boxes' on the layer. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Handler.Point │ │ │ │ │ + * Handler to draw a point on the map. Point is displayed on activation, │ │ │ │ │ + * moves on mouse move, and is finished on mouse up. The handler triggers │ │ │ │ │ + * callbacks for 'done', 'cancel', and 'modify'. The modify callback is │ │ │ │ │ + * called with each change in the sketch and will receive the latest point │ │ │ │ │ + * drawn. Create a new instance with the │ │ │ │ │ + * constructor. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ + * - │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Boxes = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ +OpenLayers.Handler.Point = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Boxes │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + * Property: point │ │ │ │ │ + * {} The currently drawn point │ │ │ │ │ */ │ │ │ │ │ + point: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawMarker │ │ │ │ │ - * Calculate the pixel location for the marker, create it, and │ │ │ │ │ - * add it to the layer's div │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * marker - {} │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {} The temporary drawing layer │ │ │ │ │ */ │ │ │ │ │ - drawMarker: function(marker) { │ │ │ │ │ - var topleft = this.map.getLayerPxFromLonLat({ │ │ │ │ │ - lon: marker.bounds.left, │ │ │ │ │ - lat: marker.bounds.top │ │ │ │ │ - }); │ │ │ │ │ - var botright = this.map.getLayerPxFromLonLat({ │ │ │ │ │ - lon: marker.bounds.right, │ │ │ │ │ - lat: marker.bounds.bottom │ │ │ │ │ - }); │ │ │ │ │ - if (botright == null || topleft == null) { │ │ │ │ │ - marker.display(false); │ │ │ │ │ - } else { │ │ │ │ │ - var markerDiv = marker.draw(topleft, { │ │ │ │ │ - w: Math.max(1, botright.x - topleft.x), │ │ │ │ │ - h: Math.max(1, botright.y - topleft.y) │ │ │ │ │ - }); │ │ │ │ │ - if (!marker.drawn) { │ │ │ │ │ - this.div.appendChild(markerDiv); │ │ │ │ │ - marker.drawn = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + layer: null, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: multi │ │ │ │ │ + * {Boolean} Cast features to multi-part geometries before passing to the │ │ │ │ │ + * layer. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + multi: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: removeMarker │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * marker - {} │ │ │ │ │ + * APIProperty: citeCompliant │ │ │ │ │ + * {Boolean} If set to true, coordinates of features drawn in a map extent │ │ │ │ │ + * crossing the date line won't exceed the world bounds. Default is false. │ │ │ │ │ */ │ │ │ │ │ - removeMarker: function(marker) { │ │ │ │ │ - OpenLayers.Util.removeItem(this.markers, marker); │ │ │ │ │ - if ((marker.div != null) && │ │ │ │ │ - (marker.div.parentNode == this.div)) { │ │ │ │ │ - this.div.removeChild(marker.div); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + citeCompliant: false, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Boxes" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Image.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: mouseDown │ │ │ │ │ + * {Boolean} The mouse is down │ │ │ │ │ + */ │ │ │ │ │ + mouseDown: false, │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: stoppedDown │ │ │ │ │ + * {Boolean} Indicate whether the last mousedown stopped the event │ │ │ │ │ + * propagation. │ │ │ │ │ + */ │ │ │ │ │ + stoppedDown: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer.js │ │ │ │ │ - * @requires OpenLayers/Tile/Image.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: lastDown │ │ │ │ │ + * {} Location of the last mouse down │ │ │ │ │ + */ │ │ │ │ │ + lastDown: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.Image │ │ │ │ │ - * Instances of OpenLayers.Layer.Image are used to display data from a web │ │ │ │ │ - * accessible image as a map layer. Create a new image layer with the │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.Image = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ + /** │ │ │ │ │ + * Property: lastUp │ │ │ │ │ + * {} │ │ │ │ │ + */ │ │ │ │ │ + lastUp: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: isBaseLayer │ │ │ │ │ - * {Boolean} The layer is a base layer. Default is true. Set this property │ │ │ │ │ - * in the layer options │ │ │ │ │ + * APIProperty: persist │ │ │ │ │ + * {Boolean} Leave the feature rendered until destroyFeature is called. │ │ │ │ │ + * Default is false. If set to true, the feature remains rendered until │ │ │ │ │ + * destroyFeature is called, typically by deactivating the handler or │ │ │ │ │ + * starting another drawing. │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + persist: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: url │ │ │ │ │ - * {String} URL of the image to use │ │ │ │ │ + * APIProperty: stopDown │ │ │ │ │ + * {Boolean} Stop event propagation on mousedown. Must be false to │ │ │ │ │ + * allow "pan while drawing". Defaults to false. │ │ │ │ │ */ │ │ │ │ │ - url: null, │ │ │ │ │ + stopDown: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: extent │ │ │ │ │ - * {} The image bounds in map units. This extent will │ │ │ │ │ - * also be used as the default maxExtent for the layer. If you wish │ │ │ │ │ - * to have a maxExtent that is different than the image extent, set the │ │ │ │ │ - * maxExtent property of the options argument (as with any other layer). │ │ │ │ │ + * APIPropery: stopUp │ │ │ │ │ + * {Boolean} Stop event propagation on mouse. Must be false to │ │ │ │ │ + * allow "pan while dragging". Defaults to fase. │ │ │ │ │ */ │ │ │ │ │ - extent: null, │ │ │ │ │ + stopUp: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: size │ │ │ │ │ - * {} The image size in pixels │ │ │ │ │ + * Property: layerOptions │ │ │ │ │ + * {Object} Any optional properties to be set on the sketch layer. │ │ │ │ │ */ │ │ │ │ │ - size: null, │ │ │ │ │ + layerOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: tile │ │ │ │ │ - * {} │ │ │ │ │ + * APIProperty: pixelTolerance │ │ │ │ │ + * {Number} Maximum number of pixels between down and up (mousedown │ │ │ │ │ + * and mouseup, or touchstart and touchend) for the handler to │ │ │ │ │ + * add a new point. If set to an integer value, if the │ │ │ │ │ + * displacement between down and up is great to this value │ │ │ │ │ + * no point will be added. Default value is 5. │ │ │ │ │ */ │ │ │ │ │ - tile: null, │ │ │ │ │ + pixelTolerance: 5, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: aspectRatio │ │ │ │ │ - * {Float} The ratio of height/width represented by a single pixel in the │ │ │ │ │ - * graphic │ │ │ │ │ + * Property: lastTouchPx │ │ │ │ │ + * {} The last pixel used to know the distance between │ │ │ │ │ + * two touches (for double touch). │ │ │ │ │ */ │ │ │ │ │ - aspectRatio: null, │ │ │ │ │ + lastTouchPx: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Image │ │ │ │ │ - * Create a new image layer │ │ │ │ │ + * Constructor: OpenLayers.Handler.Point │ │ │ │ │ + * Create a new point handler. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} A name for the layer. │ │ │ │ │ - * url - {String} Relative or absolute path to the image │ │ │ │ │ - * extent - {} The extent represented by the image │ │ │ │ │ - * size - {} The size (in pixels) of the image │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + * control - {} The control that owns this handler │ │ │ │ │ + * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ + * functions. Various callbacks described below. │ │ │ │ │ + * options - {Object} An optional object with properties to be set on the │ │ │ │ │ + * handler │ │ │ │ │ + * │ │ │ │ │ + * Named callbacks: │ │ │ │ │ + * create - Called when a sketch is first created. Callback called with │ │ │ │ │ + * the creation point geometry and sketch feature. │ │ │ │ │ + * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ + * geometry and the sketch feature. │ │ │ │ │ + * done - Called when the point drawing is finished. The callback will │ │ │ │ │ + * recieve a single argument, the point geometry. │ │ │ │ │ + * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ + * cancel callback will receive a geometry. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, extent, size, options) { │ │ │ │ │ - this.url = url; │ │ │ │ │ - this.extent = extent; │ │ │ │ │ - this.maxExtent = extent; │ │ │ │ │ - this.size = size; │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ + this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style['default'], {}); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - this.aspectRatio = (this.extent.getHeight() / this.size.h) / │ │ │ │ │ - (this.extent.getWidth() / this.size.w); │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Destroy this layer │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * turn on the handler │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.tile) { │ │ │ │ │ - this.removeTileMonitoringHooks(this.tile); │ │ │ │ │ - this.tile.destroy(); │ │ │ │ │ - this.tile = null; │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (!OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ + // create temporary vector layer for rendering geometry sketch │ │ │ │ │ + // TBD: this could be moved to initialize/destroy - setting visibility here │ │ │ │ │ + var options = OpenLayers.Util.extend({ │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + // indicate that the temp vector layer will never be out of range │ │ │ │ │ + // without this, resolution properties must be specified at the │ │ │ │ │ + // map-level for this temporary layer to init its resolutions │ │ │ │ │ + // correctly │ │ │ │ │ + calculateInRange: OpenLayers.Function.True, │ │ │ │ │ + wrapDateLine: this.citeCompliant │ │ │ │ │ + }, this.layerOptions); │ │ │ │ │ + this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ + this.map.addLayer(this.layer); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ - * │ │ │ │ │ - * Paramters: │ │ │ │ │ - * obj - {Object} An optional layer (is this ever used?) │ │ │ │ │ + * Method: createFeature │ │ │ │ │ + * Add temporary features │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} An exact copy of this layer │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {} A pixel location on the map. │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Image(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.extent, │ │ │ │ │ - this.size, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ - │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ + lonlat.lon, lonlat.lat │ │ │ │ │ + ); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.point]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setMap │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {} │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * turn off the handler │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - /** │ │ │ │ │ - * If nothing to do with resolutions has been set, assume a single │ │ │ │ │ - * resolution determined by ratio*extent/size - if an image has a │ │ │ │ │ - * pixel aspect ratio different than one (as calculated above), the │ │ │ │ │ - * image will be stretched in one dimension only. │ │ │ │ │ - */ │ │ │ │ │ - if (this.options.maxResolution == null) { │ │ │ │ │ - this.options.maxResolution = this.aspectRatio * │ │ │ │ │ - this.extent.getWidth() / │ │ │ │ │ - this.size.w; │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (!OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.cancel(); │ │ │ │ │ + // If a layer's map property is set to null, it means that that layer │ │ │ │ │ + // isn't added to the map. Since we ourself added the layer to the map │ │ │ │ │ + // in activate(), we can assume that if this.layer.map is null it means │ │ │ │ │ + // that the layer has been destroyed (as a result of map.destroy() for │ │ │ │ │ + // example. │ │ │ │ │ + if (this.layer.map != null) { │ │ │ │ │ + this.destroyFeature(true); │ │ │ │ │ + this.layer.destroy(false); │ │ │ │ │ + } │ │ │ │ │ + this.layer = null; │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * Create the tile for the image or resize it for the new resolution │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroyFeature │ │ │ │ │ + * Destroy the temporary geometries │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {} │ │ │ │ │ - * zoomChanged - {Boolean} │ │ │ │ │ - * dragging - {Boolean} │ │ │ │ │ + * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - var firstRendering = (this.tile == null); │ │ │ │ │ - │ │ │ │ │ - if (zoomChanged || firstRendering) { │ │ │ │ │ - │ │ │ │ │ - //determine new tile size │ │ │ │ │ - this.setTileSize(); │ │ │ │ │ - │ │ │ │ │ - //determine new position (upper left corner of new bounds) │ │ │ │ │ - var ulPx = this.map.getLayerPxFromLonLat({ │ │ │ │ │ - lon: this.extent.left, │ │ │ │ │ - lat: this.extent.top │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - if (firstRendering) { │ │ │ │ │ - //create the new tile │ │ │ │ │ - this.tile = new OpenLayers.Tile.Image(this, ulPx, this.extent, │ │ │ │ │ - null, this.tileSize); │ │ │ │ │ - this.addTileMonitoringHooks(this.tile); │ │ │ │ │ - } else { │ │ │ │ │ - //just resize the tile and set it's new position │ │ │ │ │ - this.tile.size = this.tileSize.clone(); │ │ │ │ │ - this.tile.position = ulPx.clone(); │ │ │ │ │ - } │ │ │ │ │ - this.tile.draw(); │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + if (this.layer && (force || !this.persist)) { │ │ │ │ │ + this.layer.destroyFeatures(); │ │ │ │ │ } │ │ │ │ │ + this.point = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Set the tile size based on the map size. │ │ │ │ │ + * Method: destroyPersistedFeature │ │ │ │ │ + * Destroy the persisted feature. │ │ │ │ │ */ │ │ │ │ │ - setTileSize: function() { │ │ │ │ │ - var tileWidth = this.extent.getWidth() / this.map.getResolution(); │ │ │ │ │ - var tileHeight = this.extent.getHeight() / this.map.getResolution(); │ │ │ │ │ - this.tileSize = new OpenLayers.Size(tileWidth, tileHeight); │ │ │ │ │ + destroyPersistedFeature: function() { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + if (layer && layer.features.length > 1) { │ │ │ │ │ + this.layer.features[0].destroy(); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: addTileMonitoringHooks │ │ │ │ │ - * This function takes a tile as input and adds the appropriate hooks to │ │ │ │ │ - * the tile so that the layer can keep track of the loading tiles. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * tile - {} │ │ │ │ │ + /** │ │ │ │ │ + * Method: finalize │ │ │ │ │ + * Finish the geometry and call the "done" callback. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * cancel - {Boolean} Call cancel instead of done callback. Default │ │ │ │ │ + * is false. │ │ │ │ │ */ │ │ │ │ │ - addTileMonitoringHooks: function(tile) { │ │ │ │ │ - tile.onLoadStart = function() { │ │ │ │ │ - this.events.triggerEvent("loadstart"); │ │ │ │ │ - }; │ │ │ │ │ - tile.events.register("loadstart", this, tile.onLoadStart); │ │ │ │ │ - │ │ │ │ │ - tile.onLoadEnd = function() { │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ - }; │ │ │ │ │ - tile.events.register("loadend", this, tile.onLoadEnd); │ │ │ │ │ - tile.events.register("unload", this, tile.onLoadEnd); │ │ │ │ │ + finalize: function(cancel) { │ │ │ │ │ + var key = cancel ? "cancel" : "done"; │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + this.lastDown = null; │ │ │ │ │ + this.lastUp = null; │ │ │ │ │ + this.lastTouchPx = null; │ │ │ │ │ + this.callback(key, [this.geometryClone()]); │ │ │ │ │ + this.destroyFeature(cancel); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeTileMonitoringHooks │ │ │ │ │ - * This function takes a tile as input and removes the tile hooks │ │ │ │ │ - * that were added in . │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * tile - {} │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: cancel │ │ │ │ │ + * Finish the geometry and call the "cancel" callback. │ │ │ │ │ */ │ │ │ │ │ - removeTileMonitoringHooks: function(tile) { │ │ │ │ │ - tile.unload(); │ │ │ │ │ - tile.events.un({ │ │ │ │ │ - "loadstart": tile.onLoadStart, │ │ │ │ │ - "loadend": tile.onLoadEnd, │ │ │ │ │ - "unload": tile.onLoadEnd, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + cancel: function() { │ │ │ │ │ + this.finalize(true); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setUrl │ │ │ │ │ + * Method: click │ │ │ │ │ + * Handle clicks. Clicks are stopped from propagating to other listeners │ │ │ │ │ + * on map.events or other dom elements. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * newUrl - {String} │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - setUrl: function(newUrl) { │ │ │ │ │ - this.url = newUrl; │ │ │ │ │ - this.tile.draw(); │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getURL │ │ │ │ │ - * The url we return is always the same (the image itself never changes) │ │ │ │ │ - * so we can ignore the bounds parameter (it will always be the same, │ │ │ │ │ - * anyways) │ │ │ │ │ + /** │ │ │ │ │ + * Method: dblclick │ │ │ │ │ + * Handle double-clicks. Double-clicks are stopped from propagating to other │ │ │ │ │ + * listeners on map.events or other dom elements. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {} │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - return this.url; │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Image" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/PointGrid.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.PointGrid │ │ │ │ │ - * A point grid layer dynamically generates a regularly spaced grid of point │ │ │ │ │ - * features. This is a specialty layer for cases where an application needs │ │ │ │ │ - * a regular grid of points. It can be used, for example, in an editing │ │ │ │ │ - * environment to snap to a grid. │ │ │ │ │ - * │ │ │ │ │ - * Create a new vector layer with the constructor. │ │ │ │ │ - * (code) │ │ │ │ │ - * // create a grid with points spaced at 10 map units │ │ │ │ │ - * var points = new OpenLayers.Layer.PointGrid({dx: 10, dy: 10}); │ │ │ │ │ - * │ │ │ │ │ - * // create a grid with different x/y spacing rotated 15 degrees clockwise. │ │ │ │ │ - * var points = new OpenLayers.Layer.PointGrid({dx: 5, dy: 10, rotation: 15}); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.PointGrid = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: dx │ │ │ │ │ - * {Number} Point grid spacing in the x-axis direction (map units). │ │ │ │ │ - * Read-only. Use the method to modify this value. │ │ │ │ │ - */ │ │ │ │ │ - dx: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: dy │ │ │ │ │ - * {Number} Point grid spacing in the y-axis direction (map units). │ │ │ │ │ - * Read-only. Use the method to modify this value. │ │ │ │ │ - */ │ │ │ │ │ - dy: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: ratio │ │ │ │ │ - * {Number} Ratio of the desired grid size to the map viewport size. │ │ │ │ │ - * Default is 1.5. Larger ratios mean the grid is recalculated less often │ │ │ │ │ - * while panning. The setting has precedence when determining │ │ │ │ │ - * grid size. Read-only. Use the method to modify this value. │ │ │ │ │ - */ │ │ │ │ │ - ratio: 1.5, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: maxFeatures │ │ │ │ │ - * {Number} The maximum number of points to generate in the grid. Default │ │ │ │ │ - * is 250. Read-only. Use the method to modify this value. │ │ │ │ │ - */ │ │ │ │ │ - maxFeatures: 250, │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: rotation │ │ │ │ │ - * {Number} Grid rotation (in degrees clockwise from the positive x-axis). │ │ │ │ │ - * Default is 0. Read-only. Use the method to modify this │ │ │ │ │ - * value. │ │ │ │ │ + * Method: modifyFeature │ │ │ │ │ + * Modify the existing geometry given a pixel location. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {} A pixel location on the map. │ │ │ │ │ */ │ │ │ │ │ - rotation: 0, │ │ │ │ │ + modifyFeature: function(pixel) { │ │ │ │ │ + if (!this.point) { │ │ │ │ │ + this.createFeature(pixel); │ │ │ │ │ + } │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point.geometry.x = lonlat.lon; │ │ │ │ │ + this.point.geometry.y = lonlat.lat; │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.point, false]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: origin │ │ │ │ │ - * {} Grid origin. The grid lattice will be aligned with │ │ │ │ │ - * the origin. If not set at construction, the center of the map's maximum │ │ │ │ │ - * extent is used. Read-only. Use the method to modify this │ │ │ │ │ - * value. │ │ │ │ │ + * Method: drawFeature │ │ │ │ │ + * Render features on the temporary layer. │ │ │ │ │ */ │ │ │ │ │ - origin: null, │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.point, this.style); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: gridBounds │ │ │ │ │ - * {} Internally cached grid bounds (with optional │ │ │ │ │ - * rotation applied). │ │ │ │ │ + * Method: getGeometry │ │ │ │ │ + * Return the sketch geometry. If is true, this will return │ │ │ │ │ + * a multi-part geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} │ │ │ │ │ */ │ │ │ │ │ - gridBounds: null, │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.point && this.point.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiPoint([geometry]); │ │ │ │ │ + } │ │ │ │ │ + return geometry; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.PointGrid │ │ │ │ │ - * Creates a new point grid layer. │ │ │ │ │ + * Method: geometryClone │ │ │ │ │ + * Return a clone of the relevant geometry. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} An object containing all configuration properties for │ │ │ │ │ - * the layer. The and properties are required to be set at │ │ │ │ │ - * construction. Any other layer properties may be set in this object. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ - config = config || {}; │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.initialize.apply(this, [config.name, config]); │ │ │ │ │ + geometryClone: function() { │ │ │ │ │ + var geom = this.getGeometry(); │ │ │ │ │ + return geom && geom.clone(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * The layer has been added to the map. │ │ │ │ │ + /** │ │ │ │ │ + * Method: mousedown │ │ │ │ │ + * Handle mousedown. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {} │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ - map.events.register("moveend", this, this.onMoveEnd); │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + return this.down(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removeMap │ │ │ │ │ - * The layer has been removed from the map. │ │ │ │ │ - * │ │ │ │ │ + * Method: touchstart │ │ │ │ │ + * Handle touchstart. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {} │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - map.events.unregister("moveend", this, this.onMoveEnd); │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments); │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + this.lastTouchPx = evt.xy; │ │ │ │ │ + return this.down(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setRatio │ │ │ │ │ - * Set the grid property and update the grid. Can only be called │ │ │ │ │ - * after the layer has been added to a map with a center/extent. │ │ │ │ │ - * │ │ │ │ │ + * Method: mousemove │ │ │ │ │ + * Handle mousemove. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * ratio - {Number} │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - setRatio: function(ratio) { │ │ │ │ │ - this.ratio = ratio; │ │ │ │ │ - this.updateGrid(true); │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + return this.move(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setMaxFeatures │ │ │ │ │ - * Set the grid property and update the grid. Can only be │ │ │ │ │ - * called after the layer has been added to a map with a center/extent. │ │ │ │ │ - * │ │ │ │ │ + * Method: touchmove │ │ │ │ │ + * Handle touchmove. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * maxFeatures - {Number} │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - setMaxFeatures: function(maxFeatures) { │ │ │ │ │ - this.maxFeatures = maxFeatures; │ │ │ │ │ - this.updateGrid(true); │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + this.lastTouchPx = evt.xy; │ │ │ │ │ + return this.move(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setSpacing │ │ │ │ │ - * Set the grid and properties and update the grid. If only one │ │ │ │ │ - * argument is provided, it will be set as and . Can only be │ │ │ │ │ - * called after the layer has been added to a map with a center/extent. │ │ │ │ │ - * │ │ │ │ │ + * Method: mouseup │ │ │ │ │ + * Handle mouseup. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * dx - {Number} │ │ │ │ │ - * dy - {Number} │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - setSpacing: function(dx, dy) { │ │ │ │ │ - this.dx = dx; │ │ │ │ │ - this.dy = dy || dx; │ │ │ │ │ - this.updateGrid(true); │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + return this.up(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setOrigin │ │ │ │ │ - * Set the grid property and update the grid. Can only be called │ │ │ │ │ - * after the layer has been added to a map with a center/extent. │ │ │ │ │ - * │ │ │ │ │ + * Method: touchend │ │ │ │ │ + * Handle touchend. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * origin - {} │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - setOrigin: function(origin) { │ │ │ │ │ - this.origin = origin; │ │ │ │ │ - this.updateGrid(true); │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + evt.xy = this.lastTouchPx; │ │ │ │ │ + return this.up(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getOrigin │ │ │ │ │ - * Get the grid property. │ │ │ │ │ + * Method: down │ │ │ │ │ + * Handle mousedown and touchstart. Adjust the geometry and redraw. │ │ │ │ │ + * Return determines whether to propagate the event on the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} The grid origin. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - getOrigin: function() { │ │ │ │ │ - if (!this.origin) { │ │ │ │ │ - this.origin = this.map.getExtent().getCenterLonLat(); │ │ │ │ │ + down: function(evt) { │ │ │ │ │ + this.mouseDown = true; │ │ │ │ │ + this.lastDown = evt.xy; │ │ │ │ │ + if (!this.touch) { // no point displayed until up on touch devices │ │ │ │ │ + this.modifyFeature(evt.xy); │ │ │ │ │ } │ │ │ │ │ - return this.origin; │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + return !this.stopDown; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setRotation │ │ │ │ │ - * Set the grid property and update the grid. Rotation values │ │ │ │ │ - * are in degrees clockwise from the positive x-axis (negative values │ │ │ │ │ - * for counter-clockwise rotation). Can only be called after the layer │ │ │ │ │ - * has been added to a map with a center/extent. │ │ │ │ │ - * │ │ │ │ │ + * Method: move │ │ │ │ │ + * Handle mousemove and touchmove. Adjust the geometry and redraw. │ │ │ │ │ + * Return determines whether to propagate the event on the map. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * rotation - {Number} Degrees clockwise from the positive x-axis. │ │ │ │ │ - */ │ │ │ │ │ - setRotation: function(rotation) { │ │ │ │ │ - this.rotation = rotation; │ │ │ │ │ - this.updateGrid(true); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onMoveEnd │ │ │ │ │ - * Listener for map "moveend" events. │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - onMoveEnd: function() { │ │ │ │ │ - this.updateGrid(); │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + if (!this.touch // no point displayed until up on touch devices │ │ │ │ │ + && │ │ │ │ │ + (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ + this.modifyFeature(evt.xy); │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getViewBounds │ │ │ │ │ - * Gets the (potentially rotated) view bounds for grid calculations. │ │ │ │ │ + * Method: up │ │ │ │ │ + * Handle mouseup and touchend. Send the latest point in the geometry to the control. │ │ │ │ │ + * Return determines whether to propagate the event on the map. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - getViewBounds: function() { │ │ │ │ │ - var bounds = this.map.getExtent(); │ │ │ │ │ - if (this.rotation) { │ │ │ │ │ - var origin = this.getOrigin(); │ │ │ │ │ - var rotationOrigin = new OpenLayers.Geometry.Point(origin.lon, origin.lat); │ │ │ │ │ - var rect = bounds.toGeometry(); │ │ │ │ │ - rect.rotate(-this.rotation, rotationOrigin); │ │ │ │ │ - bounds = rect.getBounds(); │ │ │ │ │ + up: function(evt) { │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + │ │ │ │ │ + // check keyboard modifiers │ │ │ │ │ + if (!this.checkModifiers(evt)) { │ │ │ │ │ + return true; │ │ │ │ │ + } │ │ │ │ │ + // ignore double-clicks │ │ │ │ │ + if (this.lastUp && this.lastUp.equals(evt.xy)) { │ │ │ │ │ + return true; │ │ │ │ │ + } │ │ │ │ │ + if (this.lastDown && this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ + this.pixelTolerance)) { │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy); │ │ │ │ │ + } │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature(); │ │ │ │ │ + } │ │ │ │ │ + this.lastUp = evt.xy; │ │ │ │ │ + this.finalize(); │ │ │ │ │ + return !this.stopUp; │ │ │ │ │ + } else { │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ - return bounds; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: updateGrid │ │ │ │ │ - * Update the grid. │ │ │ │ │ + * Method: mouseout │ │ │ │ │ + * Handle mouse out. For better user experience reset mouseDown │ │ │ │ │ + * and stoppedDown when the mouse leaves the map viewport. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * force - {Boolean} Update the grid even if the previous bounds are still │ │ │ │ │ - * valid. │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ */ │ │ │ │ │ - updateGrid: function(force) { │ │ │ │ │ - if (force || this.invalidBounds()) { │ │ │ │ │ - var viewBounds = this.getViewBounds(); │ │ │ │ │ - var origin = this.getOrigin(); │ │ │ │ │ - var rotationOrigin = new OpenLayers.Geometry.Point(origin.lon, origin.lat); │ │ │ │ │ - var viewBoundsWidth = viewBounds.getWidth(); │ │ │ │ │ - var viewBoundsHeight = viewBounds.getHeight(); │ │ │ │ │ - var aspectRatio = viewBoundsWidth / viewBoundsHeight; │ │ │ │ │ - var maxHeight = Math.sqrt(this.dx * this.dy * this.maxFeatures / aspectRatio); │ │ │ │ │ - var maxWidth = maxHeight * aspectRatio; │ │ │ │ │ - var gridWidth = Math.min(viewBoundsWidth * this.ratio, maxWidth); │ │ │ │ │ - var gridHeight = Math.min(viewBoundsHeight * this.ratio, maxHeight); │ │ │ │ │ - var center = viewBounds.getCenterLonLat(); │ │ │ │ │ - this.gridBounds = new OpenLayers.Bounds( │ │ │ │ │ - center.lon - (gridWidth / 2), │ │ │ │ │ - center.lat - (gridHeight / 2), │ │ │ │ │ - center.lon + (gridWidth / 2), │ │ │ │ │ - center.lat + (gridHeight / 2) │ │ │ │ │ - ); │ │ │ │ │ - var rows = Math.floor(gridHeight / this.dy); │ │ │ │ │ - var cols = Math.floor(gridWidth / this.dx); │ │ │ │ │ - var gridLeft = origin.lon + (this.dx * Math.ceil((this.gridBounds.left - origin.lon) / this.dx)); │ │ │ │ │ - var gridBottom = origin.lat + (this.dy * Math.ceil((this.gridBounds.bottom - origin.lat) / this.dy)); │ │ │ │ │ - var features = new Array(rows * cols); │ │ │ │ │ - var x, y, point; │ │ │ │ │ - for (var i = 0; i < cols; ++i) { │ │ │ │ │ - x = gridLeft + (i * this.dx); │ │ │ │ │ - for (var j = 0; j < rows; ++j) { │ │ │ │ │ - y = gridBottom + (j * this.dy); │ │ │ │ │ - point = new OpenLayers.Geometry.Point(x, y); │ │ │ │ │ - if (this.rotation) { │ │ │ │ │ - point.rotate(this.rotation, rotationOrigin); │ │ │ │ │ - } │ │ │ │ │ - features[(i * rows) + j] = new OpenLayers.Feature.Vector(point); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.destroyFeatures(this.features, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.addFeatures(features, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ + mouseout: function(evt) { │ │ │ │ │ + if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: invalidBounds │ │ │ │ │ - * Determine whether the previously generated point grid is invalid. │ │ │ │ │ - * This occurs when the map bounds extends beyond the previously │ │ │ │ │ - * generated grid bounds. │ │ │ │ │ + * Method: passesTolerance │ │ │ │ │ + * Determine whether the event is within the optional pixel tolerance. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * {Boolean} The event is within the pixel tolerance (if specified). │ │ │ │ │ */ │ │ │ │ │ - invalidBounds: function() { │ │ │ │ │ - return !this.gridBounds || !this.gridBounds.containsBounds(this.getViewBounds()); │ │ │ │ │ - }, │ │ │ │ │ + passesTolerance: function(pixel1, pixel2, tolerance) { │ │ │ │ │ + var passes = true; │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.PointGrid" │ │ │ │ │ + if (tolerance != null && pixel1 && pixel2) { │ │ │ │ │ + var dist = pixel1.distanceTo(pixel2); │ │ │ │ │ + if (dist > tolerance) { │ │ │ │ │ + passes = false; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return passes; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Point" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/GeoRSS.js │ │ │ │ │ + OpenLayers/Handler/Path.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Markers.js │ │ │ │ │ - * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ + * @requires OpenLayers/Handler/Point.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.GeoRSS │ │ │ │ │ - * Add GeoRSS Point features to your map. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Handler.Path │ │ │ │ │ + * Handler to draw a path on the map. Path is displayed on mouse down, │ │ │ │ │ + * moves on mouse move, and is finished on mouse up. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - │ │ │ │ │ + * - │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.GeoRSS = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ +OpenLayers.Handler.Path = OpenLayers.Class(OpenLayers.Handler.Point, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: location │ │ │ │ │ - * {String} store url of text file │ │ │ │ │ + /** │ │ │ │ │ + * Property: line │ │ │ │ │ + * {} │ │ │ │ │ */ │ │ │ │ │ - location: null, │ │ │ │ │ + line: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Array()} │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxVertices │ │ │ │ │ + * {Number} The maximum number of vertices which can be drawn by this │ │ │ │ │ + * handler. When the number of vertices reaches maxVertices, the │ │ │ │ │ + * geometry is automatically finalized. Default is null. │ │ │ │ │ */ │ │ │ │ │ - features: null, │ │ │ │ │ + maxVertices: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: formatOptions │ │ │ │ │ - * {Object} Hash of options which should be passed to the format when it is │ │ │ │ │ - * created. Must be passed in the constructor. │ │ │ │ │ + * Property: doubleTouchTolerance │ │ │ │ │ + * {Number} Maximum number of pixels between two touches for │ │ │ │ │ + * the gesture to be considered a "finalize feature" action. │ │ │ │ │ + * Default is 20. │ │ │ │ │ */ │ │ │ │ │ - formatOptions: null, │ │ │ │ │ + doubleTouchTolerance: 20, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: selectedFeature │ │ │ │ │ - * {} │ │ │ │ │ + /** │ │ │ │ │ + * Property: freehand │ │ │ │ │ + * {Boolean} In freehand mode, the handler starts the path on mouse down, │ │ │ │ │ + * adds a point for every mouse move, and finishes the path on mouse up. │ │ │ │ │ + * Outside of freehand mode, a point is added to the path on every mouse │ │ │ │ │ + * click and double-click finishes the path. │ │ │ │ │ */ │ │ │ │ │ - selectedFeature: null, │ │ │ │ │ + freehand: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: icon │ │ │ │ │ - * {}. This determines the Icon to be used on the map │ │ │ │ │ - * for this GeoRSS layer. │ │ │ │ │ + /** │ │ │ │ │ + * Property: freehandToggle │ │ │ │ │ + * {String} If set, freehandToggle is checked on mouse events and will set │ │ │ │ │ + * the freehand mode to the opposite of this.freehand. To disallow │ │ │ │ │ + * toggling between freehand and non-freehand mode, set freehandToggle to │ │ │ │ │ + * null. Acceptable toggle values are 'shiftKey', 'ctrlKey', and 'altKey'. │ │ │ │ │ */ │ │ │ │ │ - icon: null, │ │ │ │ │ + freehandToggle: 'shiftKey', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: popupSize │ │ │ │ │ - * {} This determines the size of GeoRSS popups. If │ │ │ │ │ - * not provided, defaults to 250px by 120px. │ │ │ │ │ + * Property: timerId │ │ │ │ │ + * {Integer} The timer used to test the double touch. │ │ │ │ │ */ │ │ │ │ │ - popupSize: null, │ │ │ │ │ + timerId: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: useFeedTitle │ │ │ │ │ - * {Boolean} Set layer.name to the first element in the feed. Default is true. │ │ │ │ │ + /** │ │ │ │ │ + * Property: redoStack │ │ │ │ │ + * {Array} Stack containing points removed with <undo>. │ │ │ │ │ */ │ │ │ │ │ - useFeedTitle: true, │ │ │ │ │ + redoStack: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.GeoRSS │ │ │ │ │ - * Create a GeoRSS Layer. │ │ │ │ │ + * Constructor: OpenLayers.Handler.Path │ │ │ │ │ + * Create a new path hander │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * location - {String} │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ + * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ + * functions. Various callbacks described below. │ │ │ │ │ + * options - {Object} An optional object with properties to be set on the │ │ │ │ │ + * handler │ │ │ │ │ + * │ │ │ │ │ + * Named callbacks: │ │ │ │ │ + * create - Called when a sketch is first created. Callback called with │ │ │ │ │ + * the creation point geometry and sketch feature. │ │ │ │ │ + * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ + * geometry and the sketch feature. │ │ │ │ │ + * point - Called as each point is added. Receives the new point geometry. │ │ │ │ │ + * done - Called when the point drawing is finished. The callback will │ │ │ │ │ + * recieve a single argument, the linestring geometry. │ │ │ │ │ + * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ + * cancel callback will receive a geometry. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, location, options) { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ - this.location = location; │ │ │ │ │ - this.features = []; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * Method: createFeature │ │ │ │ │ + * Add temporary geometries │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} The initial pixel location for the new │ │ │ │ │ + * feature. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - // Warning: Layer.Markers.destroy() must be called prior to calling │ │ │ │ │ - // clearFeatures() here, otherwise we leak memory. Indeed, if │ │ │ │ │ - // Layer.Markers.destroy() is called after clearFeatures(), it won't be │ │ │ │ │ - // able to remove the marker image elements from the layer's div since │ │ │ │ │ - // the markers will have been destroyed by clearFeatures(). │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.destroy.apply(this, arguments); │ │ │ │ │ - this.clearFeatures(); │ │ │ │ │ - this.features = null; │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ + lonlat.lon, lonlat.lat │ │ │ │ │ + ); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.line = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.LineString([this.point.geometry]) │ │ │ │ │ + ); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.line, this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: loadRSS │ │ │ │ │ - * Start the load of the RSS data. Don't do this when we first add the layer, │ │ │ │ │ - * since we may not be visible at any point, and it would therefore be a waste. │ │ │ │ │ + * Method: destroyFeature │ │ │ │ │ + * Destroy temporary geometries │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ */ │ │ │ │ │ - loadRSS: function() { │ │ │ │ │ - if (!this.loaded) { │ │ │ │ │ - this.events.triggerEvent("loadstart"); │ │ │ │ │ - OpenLayers.Request.GET({ │ │ │ │ │ - url: this.location, │ │ │ │ │ - success: this.parseData, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.loaded = true; │ │ │ │ │ - } │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + OpenLayers.Handler.Point.prototype.destroyFeature.call( │ │ │ │ │ + this, force); │ │ │ │ │ + this.line = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * If layer is visible and RSS has not been loaded, load RSS. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {Object} │ │ │ │ │ - * zoomChanged - {Object} │ │ │ │ │ - * minor - {Object} │ │ │ │ │ + * Method: destroyPersistedFeature │ │ │ │ │ + * Destroy the persisted feature. │ │ │ │ │ */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, minor) { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - if (this.visibility && !this.loaded) { │ │ │ │ │ - this.loadRSS(); │ │ │ │ │ + destroyPersistedFeature: function() { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + if (layer && layer.features.length > 2) { │ │ │ │ │ + this.layer.features[0].destroy(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseData │ │ │ │ │ - * Parse the data returned from the Events call. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * ajaxRequest - {<OpenLayers.Request.XMLHttpRequest>} │ │ │ │ │ + * Method: removePoint │ │ │ │ │ + * Destroy the temporary point. │ │ │ │ │ */ │ │ │ │ │ - parseData: function(ajaxRequest) { │ │ │ │ │ - var doc = ajaxRequest.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = OpenLayers.Format.XML.prototype.read(ajaxRequest.responseText); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.useFeedTitle) { │ │ │ │ │ - var name = null; │ │ │ │ │ - try { │ │ │ │ │ - name = doc.getElementsByTagNameNS('*', 'title')[0].firstChild.nodeValue; │ │ │ │ │ - } catch (e) { │ │ │ │ │ - try { │ │ │ │ │ - name = doc.getElementsByTagName('title')[0].firstChild.nodeValue; │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - } │ │ │ │ │ - if (name) { │ │ │ │ │ - this.setName(name); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - 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 format = new OpenLayers.Format.GeoRSS(options); │ │ │ │ │ - var features = format.read(doc); │ │ │ │ │ - │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - var data = {}; │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - │ │ │ │ │ - // we don't support features with no geometry in the GeoRSS │ │ │ │ │ - // layer at this time. │ │ │ │ │ - if (!feature.geometry) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var title = feature.attributes.title ? │ │ │ │ │ - feature.attributes.title : "Untitled"; │ │ │ │ │ - │ │ │ │ │ - var description = feature.attributes.description ? │ │ │ │ │ - feature.attributes.description : "No description."; │ │ │ │ │ - │ │ │ │ │ - var link = feature.attributes.link ? feature.attributes.link : ""; │ │ │ │ │ - │ │ │ │ │ - var location = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - data.icon = this.icon == null ? │ │ │ │ │ - OpenLayers.Marker.defaultIcon() : │ │ │ │ │ - this.icon.clone(); │ │ │ │ │ - │ │ │ │ │ - data.popupSize = this.popupSize ? │ │ │ │ │ - this.popupSize.clone() : │ │ │ │ │ - new OpenLayers.Size(250, 120); │ │ │ │ │ - │ │ │ │ │ - if (title || description) { │ │ │ │ │ - // we have supplemental data, store them. │ │ │ │ │ - data.title = title; │ │ │ │ │ - data.description = description; │ │ │ │ │ - │ │ │ │ │ - var contentHTML = '<div class="olLayerGeoRSSClose">[x]</div>'; │ │ │ │ │ - contentHTML += '<div class="olLayerGeoRSSTitle">'; │ │ │ │ │ - if (link) { │ │ │ │ │ - contentHTML += '<a class="link" href="' + link + '" target="_blank">'; │ │ │ │ │ - } │ │ │ │ │ - contentHTML += title; │ │ │ │ │ - if (link) { │ │ │ │ │ - contentHTML += '</a>'; │ │ │ │ │ - } │ │ │ │ │ - contentHTML += '</div>'; │ │ │ │ │ - contentHTML += '<div style="" class="olLayerGeoRSSDescription">'; │ │ │ │ │ - contentHTML += description; │ │ │ │ │ - contentHTML += '</div>'; │ │ │ │ │ - data['popupContentHTML'] = contentHTML; │ │ │ │ │ - } │ │ │ │ │ - var feature = new OpenLayers.Feature(this, location, data); │ │ │ │ │ - this.features.push(feature); │ │ │ │ │ - var marker = feature.createMarker(); │ │ │ │ │ - marker.events.register('click', feature, this.markerClick); │ │ │ │ │ - this.addMarker(marker); │ │ │ │ │ + removePoint: function() { │ │ │ │ │ + if (this.point) { │ │ │ │ │ + this.layer.removeFeatures([this.point]); │ │ │ │ │ } │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: markerClick │ │ │ │ │ + * Method: addPoint │ │ │ │ │ + * Add point to geometry. Send the point index to override │ │ │ │ │ + * the behavior of LinearRing that disregards adding duplicate points. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} The pixel location for the new point. │ │ │ │ │ */ │ │ │ │ │ - markerClick: function(evt) { │ │ │ │ │ - var sameMarkerClicked = (this == this.layer.selectedFeature); │ │ │ │ │ - this.layer.selectedFeature = (!sameMarkerClicked) ? this : null; │ │ │ │ │ - for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ - this.layer.map.removePopup(this.layer.map.popups[i]); │ │ │ │ │ - } │ │ │ │ │ - if (!sameMarkerClicked) { │ │ │ │ │ - var popup = this.createPopup(); │ │ │ │ │ - OpenLayers.Event.observe(popup.div, "click", │ │ │ │ │ - OpenLayers.Function.bind(function() { │ │ │ │ │ - for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ - this.layer.map.removePopup(this.layer.map.popups[i]); │ │ │ │ │ - } │ │ │ │ │ - }, this) │ │ │ │ │ - ); │ │ │ │ │ - this.layer.map.addPopup(popup); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ + addPoint: function(pixel) { │ │ │ │ │ + this.layer.removeFeatures([this.point]); │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat) │ │ │ │ │ + ); │ │ │ │ │ + this.line.geometry.addComponent( │ │ │ │ │ + this.point.geometry, this.line.geometry.components.length │ │ │ │ │ + ); │ │ │ │ │ + this.layer.addFeatures([this.point]); │ │ │ │ │ + this.callback("point", [this.point.geometry, this.getGeometry()]); │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + delete this.redoStack; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clearFeatures │ │ │ │ │ - * Destroy all features in this layer. │ │ │ │ │ + * Method: insertXY │ │ │ │ │ + * Insert a point in the current sketch given x & y coordinates. The new │ │ │ │ │ + * point is inserted immediately before the most recently drawn point. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * x - {Number} The x-coordinate of the point. │ │ │ │ │ + * y - {Number} The y-coordinate of the point. │ │ │ │ │ */ │ │ │ │ │ - clearFeatures: function() { │ │ │ │ │ - if (this.features != null) { │ │ │ │ │ - while (this.features.length > 0) { │ │ │ │ │ - var feature = this.features[0]; │ │ │ │ │ - OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ - feature.destroy(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + insertXY: function(x, y) { │ │ │ │ │ + this.line.geometry.addComponent( │ │ │ │ │ + new OpenLayers.Geometry.Point(x, y), │ │ │ │ │ + this.getCurrentPointIndex() │ │ │ │ │ + ); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + delete this.redoStack; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.GeoRSS" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/SphericalMercator.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer.js │ │ │ │ │ - * @requires OpenLayers/Projection.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.SphericalMercator │ │ │ │ │ - * A mixin for layers that wraps up the pieces neccesary to have a coordinate │ │ │ │ │ - * conversion for working with commercial APIs which use a spherical │ │ │ │ │ - * mercator projection. Using this layer as a base layer, additional │ │ │ │ │ - * layers can be used as overlays if they are in the same projection. │ │ │ │ │ - * │ │ │ │ │ - * A layer is given properties of this object by setting the sphericalMercator │ │ │ │ │ - * property to true. │ │ │ │ │ - * │ │ │ │ │ - * More projection information: │ │ │ │ │ - * - http://spatialreference.org/ref/user/google-projection/ │ │ │ │ │ - * │ │ │ │ │ - * Proj4 Text: │ │ │ │ │ - * +proj=merc +a=6378137 +b=6378137 +lat_ts=0.0 +lon_0=0.0 +x_0=0.0 +y_0=0 │ │ │ │ │ - * +k=1.0 +units=m +nadgrids=@null +no_defs │ │ │ │ │ - * │ │ │ │ │ - * WKT: │ │ │ │ │ - * 900913=PROJCS["WGS84 / Simple Mercator", GEOGCS["WGS 84", │ │ │ │ │ - * DATUM["WGS_1984", SPHEROID["WGS_1984", 6378137.0, 298.257223563]], │ │ │ │ │ - * PRIMEM["Greenwich", 0.0], UNIT["degree", 0.017453292519943295], │ │ │ │ │ - * AXIS["Longitude", EAST], AXIS["Latitude", NORTH]], │ │ │ │ │ - * PROJECTION["Mercator_1SP_Google"], │ │ │ │ │ - * PARAMETER["latitude_of_origin", 0.0], PARAMETER["central_meridian", 0.0], │ │ │ │ │ - * PARAMETER["scale_factor", 1.0], PARAMETER["false_easting", 0.0], │ │ │ │ │ - * PARAMETER["false_northing", 0.0], UNIT["m", 1.0], AXIS["x", EAST], │ │ │ │ │ - * AXIS["y", NORTH], AUTHORITY["EPSG","900913"]] │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.SphericalMercator = { │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: getExtent │ │ │ │ │ - * Get the map's extent. │ │ │ │ │ + * Method: insertDeltaXY │ │ │ │ │ + * Insert a point given offsets from the previously inserted point. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} The map extent. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * dx - {Number} The x-coordinate offset of the point. │ │ │ │ │ + * dy - {Number} The y-coordinate offset of the point. │ │ │ │ │ */ │ │ │ │ │ - getExtent: function() { │ │ │ │ │ - var extent = null; │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - extent = this.map.calculateBounds(); │ │ │ │ │ - } else { │ │ │ │ │ - extent = OpenLayers.Layer.FixedZoomLevels.prototype.getExtent.apply(this); │ │ │ │ │ + insertDeltaXY: function(dx, dy) { │ │ │ │ │ + var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ + var p0 = this.line.geometry.components[previousIndex]; │ │ │ │ │ + if (p0 && !isNaN(p0.x) && !isNaN(p0.y)) { │ │ │ │ │ + this.insertXY(p0.x + dx, p0.y + dy); │ │ │ │ │ } │ │ │ │ │ - return extent; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getLonLatFromViewPortPx │ │ │ │ │ - * Get a map location from a pixel location │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * viewPortPx - {<OpenLayers.Pixel>} │ │ │ │ │ + * Method: insertDirectionLength │ │ │ │ │ + * Insert a point in the current sketch given a direction and a length. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.LonLat>} An OpenLayers.LonLat which is the passed-in view │ │ │ │ │ - * port OpenLayers.Pixel, translated into lon/lat by map lib │ │ │ │ │ - * If the map lib is not loaded or not centered, returns null │ │ │ │ │ + * Parameters: │ │ │ │ │ + * direction - {Number} Degrees clockwise from the positive x-axis. │ │ │ │ │ + * length - {Number} Distance from the previously drawn point. │ │ │ │ │ */ │ │ │ │ │ - getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ - return OpenLayers.Layer.prototype.getLonLatFromViewPortPx.apply(this, arguments); │ │ │ │ │ + insertDirectionLength: function(direction, length) { │ │ │ │ │ + direction *= Math.PI / 180; │ │ │ │ │ + var dx = length * Math.cos(direction); │ │ │ │ │ + var dy = length * Math.sin(direction); │ │ │ │ │ + this.insertDeltaXY(dx, dy); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getViewPortPxFromLonLat │ │ │ │ │ - * Get a pixel location from a map location │ │ │ │ │ + * Method: insertDeflectionLength │ │ │ │ │ + * Insert a point in the current sketch given a deflection and a length. │ │ │ │ │ + * The deflection should be degrees clockwise from the previously │ │ │ │ │ + * digitized segment. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Pixel>} An OpenLayers.Pixel which is the passed-in │ │ │ │ │ - * OpenLayers.LonLat, translated into view port pixels by map lib │ │ │ │ │ - * If map lib is not loaded or not centered, returns null │ │ │ │ │ - */ │ │ │ │ │ - getViewPortPxFromLonLat: function(lonlat) { │ │ │ │ │ - return OpenLayers.Layer.prototype.getViewPortPxFromLonLat.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: initMercatorParameters │ │ │ │ │ - * Set up the mercator parameters on the layer: resolutions, │ │ │ │ │ - * projection, units. │ │ │ │ │ + * deflection - {Number} Degrees clockwise from the previous segment. │ │ │ │ │ + * length - {Number} Distance from the previously drawn point. │ │ │ │ │ */ │ │ │ │ │ - initMercatorParameters: function() { │ │ │ │ │ - // set up properties for Mercator - assume EPSG:900913 │ │ │ │ │ - this.RESOLUTIONS = []; │ │ │ │ │ - var maxResolution = 156543.03390625; │ │ │ │ │ - for (var zoom = 0; zoom <= this.MAX_ZOOM_LEVEL; ++zoom) { │ │ │ │ │ - this.RESOLUTIONS[zoom] = maxResolution / Math.pow(2, zoom); │ │ │ │ │ + insertDeflectionLength: function(deflection, length) { │ │ │ │ │ + var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ + if (previousIndex > 0) { │ │ │ │ │ + var p1 = this.line.geometry.components[previousIndex]; │ │ │ │ │ + var p0 = this.line.geometry.components[previousIndex - 1]; │ │ │ │ │ + var theta = Math.atan2(p1.y - p0.y, p1.x - p0.x); │ │ │ │ │ + this.insertDirectionLength( │ │ │ │ │ + (theta * 180 / Math.PI) + deflection, length │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - this.units = "m"; │ │ │ │ │ - this.projection = this.projection || "EPSG:900913"; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: forwardMercator │ │ │ │ │ - * Given a lon,lat in EPSG:4326, return a point in Spherical Mercator. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * lon - {float} │ │ │ │ │ - * lat - {float} │ │ │ │ │ + * Method: getCurrentPointIndex │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.LonLat>} The coordinates transformed to Mercator. │ │ │ │ │ + * {Number} The index of the most recently drawn point. │ │ │ │ │ */ │ │ │ │ │ - forwardMercator: (function() { │ │ │ │ │ - var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - var sm = new OpenLayers.Projection("EPSG:900913"); │ │ │ │ │ - return function(lon, lat) { │ │ │ │ │ - var point = OpenLayers.Projection.transform({ │ │ │ │ │ - x: lon, │ │ │ │ │ - y: lat │ │ │ │ │ - }, gg, sm); │ │ │ │ │ - return new OpenLayers.LonLat(point.x, point.y); │ │ │ │ │ - }; │ │ │ │ │ - })(), │ │ │ │ │ + getCurrentPointIndex: function() { │ │ │ │ │ + return this.line.geometry.components.length - 1; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: inverseMercator │ │ │ │ │ - * Given a x,y in Spherical Mercator, return a point in EPSG:4326. │ │ │ │ │ + * Method: undo │ │ │ │ │ + * Remove the most recently added point in the sketch geometry. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {float} A map x in Spherical Mercator. │ │ │ │ │ - * y - {float} A map y in Spherical Mercator. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.LonLat>} The coordinates transformed to EPSG:4326. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} A point was removed. │ │ │ │ │ */ │ │ │ │ │ - inverseMercator: (function() { │ │ │ │ │ - var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - var sm = new OpenLayers.Projection("EPSG:900913"); │ │ │ │ │ - return function(x, y) { │ │ │ │ │ - var point = OpenLayers.Projection.transform({ │ │ │ │ │ - x: x, │ │ │ │ │ - y: y │ │ │ │ │ - }, sm, gg); │ │ │ │ │ - return new OpenLayers.LonLat(point.x, point.y); │ │ │ │ │ - }; │ │ │ │ │ - })() │ │ │ │ │ - │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/FixedZoomLevels.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.FixedZoomLevels │ │ │ │ │ - * Some Layers will already have established zoom levels (like google │ │ │ │ │ - * or ve). Instead of trying to determine them and populate a resolutions[] │ │ │ │ │ - * Array with those values, we will hijack the resolution functionality │ │ │ │ │ - * here. │ │ │ │ │ - * │ │ │ │ │ - * When you subclass FixedZoomLevels: │ │ │ │ │ - * │ │ │ │ │ - * The initResolutions() call gets nullified, meaning no resolutions[] array │ │ │ │ │ - * is set up. Which would be a big problem getResolution() in Layer, since │ │ │ │ │ - * it merely takes map.zoom and indexes into resolutions[]... but.... │ │ │ │ │ - * │ │ │ │ │ - * The getResolution() call is also overridden. Instead of using the │ │ │ │ │ - * resolutions[] array, we simply calculate the current resolution based │ │ │ │ │ - * on the current extent and the current map size. But how will we be able │ │ │ │ │ - * to calculate the current extent without knowing the resolution...? │ │ │ │ │ - * │ │ │ │ │ - * The getExtent() function is also overridden. Instead of calculating extent │ │ │ │ │ - * based on the center point and the current resolution, we instead │ │ │ │ │ - * calculate the extent by getting the lonlats at the top-left and │ │ │ │ │ - * bottom-right by using the getLonLatFromViewPortPx() translation function, │ │ │ │ │ - * taken from the pixel locations (0,0) and the size of the map. But how │ │ │ │ │ - * will we be able to do lonlat-px translation without resolution....? │ │ │ │ │ - * │ │ │ │ │ - * The getZoomForResolution() method is overridden. Instead of indexing into │ │ │ │ │ - * the resolutions[] array, we call OpenLayers.Layer.getExent(), passing in │ │ │ │ │ - * the desired resolution. With this extent, we then call getZoomForExtent() │ │ │ │ │ - * │ │ │ │ │ - * │ │ │ │ │ - * Whenever you implement a layer using OpenLayers.Layer.FixedZoomLevels, │ │ │ │ │ - * it is your responsibility to provide the following three functions: │ │ │ │ │ - * │ │ │ │ │ - * - getLonLatFromViewPortPx │ │ │ │ │ - * - getViewPortPxFromLonLat │ │ │ │ │ - * - getZoomForExtent │ │ │ │ │ - * │ │ │ │ │ - * ...those three functions should generally be provided by any reasonable │ │ │ │ │ - * API that you might be working from. │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.FixedZoomLevels = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /********************************************************/ │ │ │ │ │ - /* */ │ │ │ │ │ - /* Baselayer Functions */ │ │ │ │ │ - /* */ │ │ │ │ │ - /* The following functions must all be implemented */ │ │ │ │ │ - /* by all base layers */ │ │ │ │ │ - /* */ │ │ │ │ │ - /********************************************************/ │ │ │ │ │ + undo: function() { │ │ │ │ │ + var geometry = this.line.geometry; │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var index = this.getCurrentPointIndex() - 1; │ │ │ │ │ + var target = components[index]; │ │ │ │ │ + var undone = geometry.removeComponent(target); │ │ │ │ │ + if (undone) { │ │ │ │ │ + // On touch devices, set the current ("mouse location") point to │ │ │ │ │ + // match the last digitized point. │ │ │ │ │ + if (this.touch && index > 0) { │ │ │ │ │ + components = geometry.components; // safety │ │ │ │ │ + var lastpt = components[index - 1]; │ │ │ │ │ + var curptidx = this.getCurrentPointIndex(); │ │ │ │ │ + var curpt = components[curptidx]; │ │ │ │ │ + curpt.x = lastpt.x; │ │ │ │ │ + curpt.y = lastpt.y; │ │ │ │ │ + } │ │ │ │ │ + if (!this.redoStack) { │ │ │ │ │ + this.redoStack = []; │ │ │ │ │ + } │ │ │ │ │ + this.redoStack.push(target); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + } │ │ │ │ │ + return undone; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.FixedZoomLevels │ │ │ │ │ - * Create a new fixed zoom levels layer. │ │ │ │ │ + * Method: redo │ │ │ │ │ + * Reinsert the most recently removed point resulting from an <undo> call. │ │ │ │ │ + * The undo stack is deleted whenever a point is added by other means. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} A point was added. │ │ │ │ │ */ │ │ │ │ │ - initialize: function() { │ │ │ │ │ - //this class is only just to add the following functions... │ │ │ │ │ - // nothing to actually do here... but it is probably a good │ │ │ │ │ - // idea to have layers that use these functions call this │ │ │ │ │ - // inititalize() anyways, in case at some point we decide we │ │ │ │ │ - // do want to put some functionality or state in here. │ │ │ │ │ + redo: function() { │ │ │ │ │ + var target = this.redoStack && this.redoStack.pop(); │ │ │ │ │ + if (target) { │ │ │ │ │ + this.line.geometry.addComponent(target, this.getCurrentPointIndex()); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + } │ │ │ │ │ + return !!target; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: initResolutions │ │ │ │ │ - * Populate the resolutions array │ │ │ │ │ + * Method: freehandMode │ │ │ │ │ + * Determine whether to behave in freehand mode or not. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - initResolutions: function() { │ │ │ │ │ - │ │ │ │ │ - var props = ['minZoomLevel', 'maxZoomLevel', 'numZoomLevels']; │ │ │ │ │ - │ │ │ │ │ - for (var i = 0, len = props.length; i < len; i++) { │ │ │ │ │ - var property = props[i]; │ │ │ │ │ - this[property] = (this.options[property] != null) ? │ │ │ │ │ - this.options[property] : │ │ │ │ │ - this.map[property]; │ │ │ │ │ - } │ │ │ │ │ + freehandMode: function(evt) { │ │ │ │ │ + return (this.freehandToggle && evt[this.freehandToggle]) ? │ │ │ │ │ + !this.freehand : this.freehand; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if ((this.minZoomLevel == null) || │ │ │ │ │ - (this.minZoomLevel < this.MIN_ZOOM_LEVEL)) { │ │ │ │ │ - this.minZoomLevel = this.MIN_ZOOM_LEVEL; │ │ │ │ │ + /** │ │ │ │ │ + * Method: modifyFeature │ │ │ │ │ + * Modify the existing geometry given the new point │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} The updated pixel location for the latest │ │ │ │ │ + * point. │ │ │ │ │ + * drawing - {Boolean} Indicate if we're currently drawing. │ │ │ │ │ + */ │ │ │ │ │ + modifyFeature: function(pixel, drawing) { │ │ │ │ │ + if (!this.line) { │ │ │ │ │ + this.createFeature(pixel); │ │ │ │ │ } │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point.geometry.x = lonlat.lon; │ │ │ │ │ + this.point.geometry.y = lonlat.lat; │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.getSketch(), drawing]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // │ │ │ │ │ - // At this point, we know what the minimum desired zoom level is, and │ │ │ │ │ - // we must calculate the total number of zoom levels. │ │ │ │ │ - // │ │ │ │ │ - // Because we allow for the setting of either the 'numZoomLevels' │ │ │ │ │ - // or the 'maxZoomLevel' properties... on either the layer or the │ │ │ │ │ - // map, we have to define some rules to see which we take into │ │ │ │ │ - // account first in this calculation. │ │ │ │ │ - // │ │ │ │ │ - // The following is the precedence list for these properties: │ │ │ │ │ - // │ │ │ │ │ - // (1) numZoomLevels set on layer │ │ │ │ │ - // (2) maxZoomLevel set on layer │ │ │ │ │ - // (3) numZoomLevels set on map │ │ │ │ │ - // (4) maxZoomLevel set on map* │ │ │ │ │ - // (5) none of the above* │ │ │ │ │ - // │ │ │ │ │ - // *Note that options (4) and (5) are only possible if the user │ │ │ │ │ - // _explicitly_ sets the 'numZoomLevels' property on the map to │ │ │ │ │ - // null, since it is set by default to 16. │ │ │ │ │ - // │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - // Note to future: In 3.0, I think we should remove the default │ │ │ │ │ - // value of 16 for map.numZoomLevels. Rather, I think that value │ │ │ │ │ - // should be set as a default on the Layer.WMS class. If someone │ │ │ │ │ - // creates a 3rd party layer and does not specify any 'minZoomLevel', │ │ │ │ │ - // 'maxZoomLevel', or 'numZoomLevels', and has not explicitly │ │ │ │ │ - // specified any of those on the map object either.. then I think │ │ │ │ │ - // it is fair to say that s/he wants all the zoom levels available. │ │ │ │ │ - // │ │ │ │ │ - // By making map.numZoomLevels *null* by default, that will be the │ │ │ │ │ - // case. As it is, I don't feel comfortable changing that right now │ │ │ │ │ - // as it would be a glaring API change and actually would probably │ │ │ │ │ - // break many peoples' codes. │ │ │ │ │ - // │ │ │ │ │ - │ │ │ │ │ - //the number of zoom levels we'd like to have. │ │ │ │ │ - var desiredZoomLevels; │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawFeature │ │ │ │ │ + * Render geometries on the temporary layer. │ │ │ │ │ + */ │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.line, this.style); │ │ │ │ │ + this.layer.drawFeature(this.point, this.style); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - //this is the maximum number of zoom levels the layer will allow, │ │ │ │ │ - // given the specified starting minimum zoom level. │ │ │ │ │ - var limitZoomLevels = this.MAX_ZOOM_LEVEL - this.minZoomLevel + 1; │ │ │ │ │ + /** │ │ │ │ │ + * Method: getSketch │ │ │ │ │ + * Return the sketch feature. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + getSketch: function() { │ │ │ │ │ + return this.line; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (((this.options.numZoomLevels == null) && │ │ │ │ │ - (this.options.maxZoomLevel != null)) // (2) │ │ │ │ │ - || │ │ │ │ │ - ((this.numZoomLevels == null) && │ │ │ │ │ - (this.maxZoomLevel != null)) // (4) │ │ │ │ │ - ) { │ │ │ │ │ - //calculate based on specified maxZoomLevel (on layer or map) │ │ │ │ │ - desiredZoomLevels = this.maxZoomLevel - this.minZoomLevel + 1; │ │ │ │ │ - } else { │ │ │ │ │ - //calculate based on specified numZoomLevels (on layer or map) │ │ │ │ │ - // this covers cases (1) and (3) │ │ │ │ │ - desiredZoomLevels = this.numZoomLevels; │ │ │ │ │ + /** │ │ │ │ │ + * Method: getGeometry │ │ │ │ │ + * Return the sketch geometry. If <multi> is true, this will return │ │ │ │ │ + * a multi-part geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.LineString>} │ │ │ │ │ + */ │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.line && this.line.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiLineString([geometry]); │ │ │ │ │ } │ │ │ │ │ + return geometry; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (desiredZoomLevels != null) { │ │ │ │ │ - //Now that we know what we would *like* the number of zoom levels │ │ │ │ │ - // to be, based on layer or map options, we have to make sure that │ │ │ │ │ - // it does not conflict with the actual limit, as specified by │ │ │ │ │ - // the constants on the layer itself (and calculated into the │ │ │ │ │ - // 'limitZoomLevels' variable). │ │ │ │ │ - this.numZoomLevels = Math.min(desiredZoomLevels, limitZoomLevels); │ │ │ │ │ + /** │ │ │ │ │ + * method: touchstart │ │ │ │ │ + * handle touchstart. │ │ │ │ │ + * │ │ │ │ │ + * parameters: │ │ │ │ │ + * evt - {event} the browser event │ │ │ │ │ + * │ │ │ │ │ + * returns: │ │ │ │ │ + * {boolean} allow event propagation │ │ │ │ │ + */ │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + if (this.timerId && │ │ │ │ │ + this.passesTolerance(this.lastTouchPx, evt.xy, │ │ │ │ │ + this.doubleTouchTolerance)) { │ │ │ │ │ + // double-tap, finalize the geometry │ │ │ │ │ + this.finishGeometry(); │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + return false; │ │ │ │ │ } else { │ │ │ │ │ - // case (5) -- neither 'numZoomLevels' not 'maxZoomLevel' was │ │ │ │ │ - // set on either the layer or the map. So we just use the │ │ │ │ │ - // maximum limit as calculated by the layer's constants. │ │ │ │ │ - this.numZoomLevels = limitZoomLevels; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //now that the 'numZoomLevels' is appropriately, safely set, │ │ │ │ │ - // we go back and re-calculate the 'maxZoomLevel'. │ │ │ │ │ - this.maxZoomLevel = this.minZoomLevel + this.numZoomLevels - 1; │ │ │ │ │ - │ │ │ │ │ - if (this.RESOLUTIONS != null) { │ │ │ │ │ - var resolutionsIndex = 0; │ │ │ │ │ - this.resolutions = []; │ │ │ │ │ - for (var i = this.minZoomLevel; i <= this.maxZoomLevel; i++) { │ │ │ │ │ - this.resolutions[resolutionsIndex++] = this.RESOLUTIONS[i]; │ │ │ │ │ + if (this.timerId) { │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ } │ │ │ │ │ - this.maxResolution = this.resolutions[0]; │ │ │ │ │ - this.minResolution = this.resolutions[this.resolutions.length - 1]; │ │ │ │ │ + this.timerId = window.setTimeout( │ │ │ │ │ + OpenLayers.Function.bind(function() { │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + }, this), 300); │ │ │ │ │ + return OpenLayers.Handler.Point.prototype.touchstart.call(this, evt); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getResolution │ │ │ │ │ - * Get the current map resolution │ │ │ │ │ + * Method: down │ │ │ │ │ + * Handle mousedown and touchstart. Add a new point to the geometry and │ │ │ │ │ + * render it. Return determines whether to propagate the event on the map. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} Map units per Pixel │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - getResolution: function() { │ │ │ │ │ - │ │ │ │ │ - if (this.resolutions != null) { │ │ │ │ │ - return OpenLayers.Layer.prototype.getResolution.apply(this, arguments); │ │ │ │ │ - } else { │ │ │ │ │ - var resolution = null; │ │ │ │ │ - │ │ │ │ │ - var viewSize = this.map.getSize(); │ │ │ │ │ - var extent = this.getExtent(); │ │ │ │ │ - │ │ │ │ │ - if ((viewSize != null) && (extent != null)) { │ │ │ │ │ - resolution = Math.max(extent.getWidth() / viewSize.w, │ │ │ │ │ - extent.getHeight() / viewSize.h); │ │ │ │ │ + down: function(evt) { │ │ │ │ │ + var stopDown = this.stopDown; │ │ │ │ │ + if (this.freehandMode(evt)) { │ │ │ │ │ + stopDown = true; │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ } │ │ │ │ │ - return resolution; │ │ │ │ │ } │ │ │ │ │ + if (!this.touch && (!this.lastDown || │ │ │ │ │ + !this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ + this.pixelTolerance))) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ + } │ │ │ │ │ + this.mouseDown = true; │ │ │ │ │ + this.lastDown = evt.xy; │ │ │ │ │ + this.stoppedDown = stopDown; │ │ │ │ │ + return !stopDown; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getExtent │ │ │ │ │ - * Calculates using px-> lonlat translation functions on tl and br │ │ │ │ │ - * corners of viewport │ │ │ │ │ + * Method: move │ │ │ │ │ + * Handle mousemove and touchmove. Adjust the geometry and redraw. │ │ │ │ │ + * Return determines whether to propagate the event on the map. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} A Bounds object which represents the lon/lat │ │ │ │ │ - * bounds of the current viewPort. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - getExtent: function() { │ │ │ │ │ - var size = this.map.getSize(); │ │ │ │ │ - var tl = this.getLonLatFromViewPortPx({ │ │ │ │ │ - x: 0, │ │ │ │ │ - y: 0 │ │ │ │ │ - }); │ │ │ │ │ - var br = this.getLonLatFromViewPortPx({ │ │ │ │ │ - x: size.w, │ │ │ │ │ - y: size.h │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - if ((tl != null) && (br != null)) { │ │ │ │ │ - return new OpenLayers.Bounds(tl.lon, br.lat, br.lon, tl.lat); │ │ │ │ │ - } else { │ │ │ │ │ - return null; │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature(); │ │ │ │ │ + } │ │ │ │ │ + if (this.maxVertices && this.line && │ │ │ │ │ + this.line.geometry.components.length === this.maxVertices) { │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize(); │ │ │ │ │ + } else { │ │ │ │ │ + this.addPoint(evt.xy); │ │ │ │ │ + } │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ } │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getZoomForResolution │ │ │ │ │ - * Get the zoom level for a given resolution │ │ │ │ │ - * │ │ │ │ │ + * Method: up │ │ │ │ │ + * Handle mouseup and touchend. Send the latest point in the geometry to │ │ │ │ │ + * the control. Return determines whether to propagate the event on the map. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resolution - {Float} │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} A suitable zoom level for the specified resolution. │ │ │ │ │ - * If no baselayer is set, returns null. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - getZoomForResolution: function(resolution) { │ │ │ │ │ - │ │ │ │ │ - if (this.resolutions != null) { │ │ │ │ │ - return OpenLayers.Layer.prototype.getZoomForResolution.apply(this, arguments); │ │ │ │ │ - } else { │ │ │ │ │ - var extent = OpenLayers.Layer.prototype.getExtent.apply(this, []); │ │ │ │ │ - return this.getZoomForExtent(extent); │ │ │ │ │ + up: function(evt) { │ │ │ │ │ + if (this.mouseDown && (!this.lastUp || !this.lastUp.equals(evt.xy))) { │ │ │ │ │ + if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature(); │ │ │ │ │ + } │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize(); │ │ │ │ │ + } else { │ │ │ │ │ + if (this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ + this.pixelTolerance)) { │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy); │ │ │ │ │ + } │ │ │ │ │ + if (this.lastUp == null && this.persist) { │ │ │ │ │ + this.destroyPersistedFeature(); │ │ │ │ │ + } │ │ │ │ │ + this.addPoint(evt.xy); │ │ │ │ │ + this.lastUp = evt.xy; │ │ │ │ │ + if (this.line.geometry.components.length === this.maxVertices + 1) { │ │ │ │ │ + this.finishGeometry(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + return !this.stopUp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /********************************************************/ │ │ │ │ │ - /* */ │ │ │ │ │ - /* Translation Functions */ │ │ │ │ │ - /* */ │ │ │ │ │ - /* The following functions translate GMaps and OL */ │ │ │ │ │ - /* formats for Pixel, LonLat, Bounds, and Zoom */ │ │ │ │ │ - /* */ │ │ │ │ │ - /********************************************************/ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - // TRANSLATION: MapObject Zoom <-> OpenLayers Zoom │ │ │ │ │ - // │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: getOLZoomFromMapObjectZoom │ │ │ │ │ - * Get the OL zoom index from the map object zoom level │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moZoom - {Integer} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} An OpenLayers Zoom level, translated from the passed in zoom │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ + * APIMethod: finishGeometry │ │ │ │ │ + * Finish the geometry and send it back to the control. │ │ │ │ │ */ │ │ │ │ │ - getOLZoomFromMapObjectZoom: function(moZoom) { │ │ │ │ │ - var zoom = null; │ │ │ │ │ - if (moZoom != null) { │ │ │ │ │ - zoom = moZoom - this.minZoomLevel; │ │ │ │ │ - if (this.map.baseLayer !== this) { │ │ │ │ │ - zoom = this.map.baseLayer.getZoomForResolution( │ │ │ │ │ - this.getResolutionForZoom(zoom) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return zoom; │ │ │ │ │ + finishGeometry: function() { │ │ │ │ │ + var index = this.line.geometry.components.length - 1; │ │ │ │ │ + this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getMapObjectZoomFromOLZoom │ │ │ │ │ - * Get the map object zoom level from the OL zoom level │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * olZoom - {Integer} │ │ │ │ │ + * Method: dblclick │ │ │ │ │ + * Handle double-clicks. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} A MapObject level, translated from the passed in olZoom │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - getMapObjectZoomFromOLZoom: function(olZoom) { │ │ │ │ │ - var zoom = null; │ │ │ │ │ - if (olZoom != null) { │ │ │ │ │ - zoom = olZoom + this.minZoomLevel; │ │ │ │ │ - if (this.map.baseLayer !== this) { │ │ │ │ │ - zoom = this.getZoomForResolution( │ │ │ │ │ - this.map.baseLayer.getResolutionForZoom(zoom) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + if (!this.freehandMode(evt)) { │ │ │ │ │ + this.finishGeometry(); │ │ │ │ │ } │ │ │ │ │ - return zoom; │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.FixedZoomLevels" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Path" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Google.js │ │ │ │ │ + OpenLayers/Handler/Polygon.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/SphericalMercator.js │ │ │ │ │ - * @requires OpenLayers/Layer/EventPane.js │ │ │ │ │ - * @requires OpenLayers/Layer/FixedZoomLevels.js │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Handler/Path.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.Google │ │ │ │ │ - * │ │ │ │ │ - * Provides a wrapper for Google's Maps API │ │ │ │ │ - * Normally the Terms of Use for this API do not allow wrapping, but Google │ │ │ │ │ - * have provided written consent to OpenLayers for this - see email in │ │ │ │ │ - * http://osgeo-org.1560.n6.nabble.com/Google-Maps-API-Terms-of-Use-changes-tp4910013p4911981.html │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Handler.Polygon │ │ │ │ │ + * Handler to draw a polygon on the map. Polygon is displayed on mouse down, │ │ │ │ │ + * moves on mouse move, and is finished on mouse up. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.SphericalMercator> │ │ │ │ │ - * - <OpenLayers.Layer.EventPane> │ │ │ │ │ - * - <OpenLayers.Layer.FixedZoomLevels> │ │ │ │ │ + * - <OpenLayers.Handler.Path> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Google = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Layer.EventPane, │ │ │ │ │ - OpenLayers.Layer.FixedZoomLevels, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: MIN_ZOOM_LEVEL │ │ │ │ │ - * {Integer} 0 │ │ │ │ │ - */ │ │ │ │ │ - MIN_ZOOM_LEVEL: 0, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: MAX_ZOOM_LEVEL │ │ │ │ │ - * {Integer} 21 │ │ │ │ │ - */ │ │ │ │ │ - MAX_ZOOM_LEVEL: 21, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: RESOLUTIONS │ │ │ │ │ - * {Array(Float)} Hardcode these resolutions so that they are more closely │ │ │ │ │ - * tied with the standard wms projection │ │ │ │ │ - */ │ │ │ │ │ - RESOLUTIONS: [ │ │ │ │ │ - 1.40625, │ │ │ │ │ - 0.703125, │ │ │ │ │ - 0.3515625, │ │ │ │ │ - 0.17578125, │ │ │ │ │ - 0.087890625, │ │ │ │ │ - 0.0439453125, │ │ │ │ │ - 0.02197265625, │ │ │ │ │ - 0.010986328125, │ │ │ │ │ - 0.0054931640625, │ │ │ │ │ - 0.00274658203125, │ │ │ │ │ - 0.001373291015625, │ │ │ │ │ - 0.0006866455078125, │ │ │ │ │ - 0.00034332275390625, │ │ │ │ │ - 0.000171661376953125, │ │ │ │ │ - 0.0000858306884765625, │ │ │ │ │ - 0.00004291534423828125, │ │ │ │ │ - 0.00002145767211914062, │ │ │ │ │ - 0.00001072883605957031, │ │ │ │ │ - 0.00000536441802978515, │ │ │ │ │ - 0.00000268220901489257, │ │ │ │ │ - 0.0000013411045074462891, │ │ │ │ │ - 0.00000067055225372314453 │ │ │ │ │ - ], │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: type │ │ │ │ │ - * {GMapType} │ │ │ │ │ - */ │ │ │ │ │ - type: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: wrapDateLine │ │ │ │ │ - * {Boolean} Allow user to pan forever east/west. Default is true. │ │ │ │ │ - * Setting this to false only restricts panning if │ │ │ │ │ - * <sphericalMercator> is true. │ │ │ │ │ - */ │ │ │ │ │ - wrapDateLine: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: sphericalMercator │ │ │ │ │ - * {Boolean} Should the map act as a mercator-projected map? This will │ │ │ │ │ - * cause all interactions with the map to be in the actual map │ │ │ │ │ - * projection, which allows support for vector drawing, overlaying │ │ │ │ │ - * other maps, etc. │ │ │ │ │ - */ │ │ │ │ │ - sphericalMercator: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {Number} The version of the Google Maps API │ │ │ │ │ - */ │ │ │ │ │ - version: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Google │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} A name for the layer. │ │ │ │ │ - * options - {Object} An optional object whose properties will be set │ │ │ │ │ - * on the layer. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - if (!options.version) { │ │ │ │ │ - options.version = typeof GMap2 === "function" ? "2" : "3"; │ │ │ │ │ - } │ │ │ │ │ - var mixin = OpenLayers.Layer.Google["v" + │ │ │ │ │ - options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (mixin) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, mixin); │ │ │ │ │ - } else { │ │ │ │ │ - throw "Unsupported Google Maps API version: " + options.version; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, mixin.DEFAULTS); │ │ │ │ │ - if (options.maxExtent) { │ │ │ │ │ - options.maxExtent = options.maxExtent.clone(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.initialize.apply(this, │ │ │ │ │ - [name, options]); │ │ │ │ │ - OpenLayers.Layer.FixedZoomLevels.prototype.initialize.apply(this, │ │ │ │ │ - [name, options]); │ │ │ │ │ +OpenLayers.Handler.Polygon = OpenLayers.Class(OpenLayers.Handler.Path, { │ │ │ │ │ │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - OpenLayers.Util.extend(this, OpenLayers.Layer.SphericalMercator); │ │ │ │ │ - this.initMercatorParameters(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: holeModifier │ │ │ │ │ + * {String} Key modifier to trigger hole digitizing. Acceptable values are │ │ │ │ │ + * "altKey", "shiftKey", or "ctrlKey". If not set, no hole digitizing │ │ │ │ │ + * will take place. Default is null. │ │ │ │ │ + */ │ │ │ │ │ + holeModifier: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Google>} An exact clone of this layer │ │ │ │ │ - */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - /** │ │ │ │ │ - * This method isn't intended to be called by a subclass and it │ │ │ │ │ - * doesn't call the same method on the superclass. We don't call │ │ │ │ │ - * the super's clone because we don't want properties that are set │ │ │ │ │ - * on this layer after initialize (i.e. this.mapObject etc.). │ │ │ │ │ - */ │ │ │ │ │ - return new OpenLayers.Layer.Google( │ │ │ │ │ - this.name, this.getOptions() │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: drawingHole │ │ │ │ │ + * {Boolean} Currently drawing an interior ring. │ │ │ │ │ + */ │ │ │ │ │ + drawingHole: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setVisibility │ │ │ │ │ - * Set the visibility flag for the layer and hide/show & redraw │ │ │ │ │ - * accordingly. Fire event unless otherwise specified │ │ │ │ │ - * │ │ │ │ │ - * Note that visibility is no longer simply whether or not the layer's │ │ │ │ │ - * style.display is set to "block". Now we store a 'visibility' state │ │ │ │ │ - * property on the layer class, this allows us to remember whether or │ │ │ │ │ - * not we *desire* for a layer to be visible. In the case where the │ │ │ │ │ - * map's resolution is out of the layer's range, this desire may be │ │ │ │ │ - * subverted. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * visible - {Boolean} Display the layer (if in range) │ │ │ │ │ - */ │ │ │ │ │ - setVisibility: function(visible) { │ │ │ │ │ - // sharing a map container, opacity has to be set per layer │ │ │ │ │ - var opacity = this.opacity == null ? 1 : this.opacity; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.setVisibility.apply(this, arguments); │ │ │ │ │ - this.setOpacity(opacity); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: polygon │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + polygon: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: display │ │ │ │ │ - * Hide or show the Layer │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * visible - {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - display: function(visible) { │ │ │ │ │ - if (!this._dragging) { │ │ │ │ │ - this.setGMapVisibility(visible); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.display.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Handler.Polygon │ │ │ │ │ + * Create a Polygon Handler. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ + * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ + * functions. Various callbacks described below. │ │ │ │ │ + * options - {Object} An optional object with properties to be set on the │ │ │ │ │ + * handler │ │ │ │ │ + * │ │ │ │ │ + * Named callbacks: │ │ │ │ │ + * create - Called when a sketch is first created. Callback called with │ │ │ │ │ + * the creation point geometry and sketch feature. │ │ │ │ │ + * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ + * geometry and the sketch feature. │ │ │ │ │ + * point - Called as each point is added. Receives the new point geometry. │ │ │ │ │ + * done - Called when the point drawing is finished. The callback will │ │ │ │ │ + * recieve a single argument, the polygon geometry. │ │ │ │ │ + * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ + * cancel callback will receive a geometry. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * zoomChanged - {Boolean} Tells when zoom has changed, as layers have to │ │ │ │ │ - * do some init work in that case. │ │ │ │ │ - * dragging - {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - this._dragging = dragging; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - delete this._dragging; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: createFeature │ │ │ │ │ + * Add temporary geometries │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} The initial pixel location for the new │ │ │ │ │ + * feature. │ │ │ │ │ + */ │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ + lonlat.lon, lonlat.lat │ │ │ │ │ + ); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.line = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.LinearRing([this.point.geometry]) │ │ │ │ │ + ); │ │ │ │ │ + this.polygon = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.Polygon([this.line.geometry]) │ │ │ │ │ + ); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.polygon, this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setOpacity │ │ │ │ │ - * Sets the opacity for the entire layer (all images) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * opacity - {Float} │ │ │ │ │ - */ │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - if (opacity !== this.opacity) { │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "opacity" │ │ │ │ │ + /** │ │ │ │ │ + * Method: addPoint │ │ │ │ │ + * Add point to geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} The pixel location for the new point. │ │ │ │ │ + */ │ │ │ │ │ + addPoint: function(pixel) { │ │ │ │ │ + if (!this.drawingHole && this.holeModifier && │ │ │ │ │ + this.evt && this.evt[this.holeModifier]) { │ │ │ │ │ + var geometry = this.point.geometry; │ │ │ │ │ + var features = this.control.layer.features; │ │ │ │ │ + var candidate, polygon; │ │ │ │ │ + // look for intersections, last drawn gets priority │ │ │ │ │ + for (var i = features.length - 1; i >= 0; --i) { │ │ │ │ │ + candidate = features[i].geometry; │ │ │ │ │ + if ((candidate instanceof OpenLayers.Geometry.Polygon || │ │ │ │ │ + candidate instanceof OpenLayers.Geometry.MultiPolygon) && │ │ │ │ │ + candidate.intersects(geometry)) { │ │ │ │ │ + polygon = features[i]; │ │ │ │ │ + this.control.layer.removeFeatures([polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ }); │ │ │ │ │ + this.control.layer.events.registerPriority( │ │ │ │ │ + "sketchcomplete", this, this.finalizeInteriorRing │ │ │ │ │ + ); │ │ │ │ │ + this.control.layer.events.registerPriority( │ │ │ │ │ + "sketchmodified", this, this.enforceTopology │ │ │ │ │ + ); │ │ │ │ │ + polygon.geometry.addComponent(this.line.geometry); │ │ │ │ │ + this.polygon = polygon; │ │ │ │ │ + this.drawingHole = true; │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ - this.opacity = opacity; │ │ │ │ │ - } │ │ │ │ │ - // Though this layer's opacity may not change, we're sharing a container │ │ │ │ │ - // and need to update the opacity for the entire container. │ │ │ │ │ - if (this.getVisibility()) { │ │ │ │ │ - var container = this.getMapContainer(); │ │ │ │ │ - OpenLayers.Util.modifyDOMElement( │ │ │ │ │ - container, null, null, null, null, null, null, opacity │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up this layer. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - /** │ │ │ │ │ - * We have to override this method because the event pane destroy │ │ │ │ │ - * deletes the mapObject reference before removing this layer from │ │ │ │ │ - * the map. │ │ │ │ │ - */ │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.setGMapVisibility(false); │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache && cache.count <= 1) { │ │ │ │ │ - this.removeGMapElements(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeGMapElements │ │ │ │ │ - * Remove all elements added to the dom. This should only be called if │ │ │ │ │ - * this is the last of the Google layers for the given map. │ │ │ │ │ - */ │ │ │ │ │ - removeGMapElements: function() { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - // remove shared elements from dom │ │ │ │ │ - var container = this.mapObject && this.getMapContainer(); │ │ │ │ │ - if (container && container.parentNode) { │ │ │ │ │ - container.parentNode.removeChild(container); │ │ │ │ │ - } │ │ │ │ │ - var termsOfUse = cache.termsOfUse; │ │ │ │ │ - if (termsOfUse && termsOfUse.parentNode) { │ │ │ │ │ - termsOfUse.parentNode.removeChild(termsOfUse); │ │ │ │ │ - } │ │ │ │ │ - var poweredBy = cache.poweredBy; │ │ │ │ │ - if (poweredBy && poweredBy.parentNode) { │ │ │ │ │ - poweredBy.parentNode.removeChild(poweredBy); │ │ │ │ │ - } │ │ │ │ │ - if (this.mapObject && window.google && google.maps && │ │ │ │ │ - google.maps.event && google.maps.event.clearListeners) { │ │ │ │ │ - google.maps.event.clearListeners(this.mapObject, 'tilesloaded'); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: removeMap │ │ │ │ │ - * On being removed from the map, also remove termsOfUse and poweredBy divs │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - // hide layer before removing │ │ │ │ │ - if (this.visibility && this.mapObject) { │ │ │ │ │ - this.setGMapVisibility(false); │ │ │ │ │ - } │ │ │ │ │ - // check to see if last Google layer in this map │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - if (cache.count <= 1) { │ │ │ │ │ - this.removeGMapElements(); │ │ │ │ │ - delete OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ - } else { │ │ │ │ │ - // decrement the layer count │ │ │ │ │ - --cache.count; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // remove references to gmap elements │ │ │ │ │ - delete this.termsOfUse; │ │ │ │ │ - delete this.poweredBy; │ │ │ │ │ - delete this.mapObject; │ │ │ │ │ - delete this.dragObject; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.removeMap.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ - // │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getOLBoundsFromMapObjectBounds │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moBounds - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} An <OpenLayers.Bounds>, translated from the │ │ │ │ │ - * passed-in MapObject Bounds. │ │ │ │ │ - * Returns null if null value is passed in. │ │ │ │ │ - */ │ │ │ │ │ - getOLBoundsFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - var olBounds = null; │ │ │ │ │ - if (moBounds != null) { │ │ │ │ │ - var sw = moBounds.getSouthWest(); │ │ │ │ │ - var ne = moBounds.getNorthEast(); │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - sw = this.forwardMercator(sw.lng(), sw.lat()); │ │ │ │ │ - ne = this.forwardMercator(ne.lng(), ne.lat()); │ │ │ │ │ - } else { │ │ │ │ │ - sw = new OpenLayers.LonLat(sw.lng(), sw.lat()); │ │ │ │ │ - ne = new OpenLayers.LonLat(ne.lng(), ne.lat()); │ │ │ │ │ - } │ │ │ │ │ - olBounds = new OpenLayers.Bounds(sw.lon, │ │ │ │ │ - sw.lat, │ │ │ │ │ - ne.lon, │ │ │ │ │ - ne.lat); │ │ │ │ │ } │ │ │ │ │ - return olBounds; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getWarningHTML │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} String with information on why layer is broken, how to get │ │ │ │ │ - * it working. │ │ │ │ │ - */ │ │ │ │ │ - getWarningHTML: function() { │ │ │ │ │ - return OpenLayers.i18n("googleWarning"); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Interface Controls * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - // Get&Set Center, Zoom │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectCenter │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The mapObject's current center in Map Object format │ │ │ │ │ - */ │ │ │ │ │ - getMapObjectCenter: function() { │ │ │ │ │ - return this.mapObject.getCenter(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectZoom │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} The mapObject's current zoom, in Map Object format │ │ │ │ │ - */ │ │ │ │ │ - getMapObjectZoom: function() { │ │ │ │ │ - return this.mapObject.getZoom(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Primitives * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - // LonLat │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getLongitudeFromMapObjectLonLat │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} Longitude of the given MapObject LonLat │ │ │ │ │ - */ │ │ │ │ │ - getLongitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - return this.sphericalMercator ? │ │ │ │ │ - this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lon : │ │ │ │ │ - moLonLat.lng(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getLatitudeFromMapObjectLonLat │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} Latitude of the given MapObject LonLat │ │ │ │ │ - */ │ │ │ │ │ - getLatitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - var lat = this.sphericalMercator ? │ │ │ │ │ - this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lat : │ │ │ │ │ - moLonLat.lat(); │ │ │ │ │ - return lat; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - // Pixel │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getXFromMapObjectPixel │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moPixel - {Object} MapObject Pixel format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} X value of the MapObject Pixel │ │ │ │ │ - */ │ │ │ │ │ - getXFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return moPixel.x; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getYFromMapObjectPixel │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moPixel - {Object} MapObject Pixel format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} Y value of the MapObject Pixel │ │ │ │ │ - */ │ │ │ │ │ - getYFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return moPixel.y; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Google" │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Property: OpenLayers.Layer.Google.cache │ │ │ │ │ - * {Object} Cache for elements that should only be created once per map. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.Google.cache = {}; │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Layer.Google.v2 │ │ │ │ │ - * │ │ │ │ │ - * Mixin providing functionality specific to the Google Maps API v2. │ │ │ │ │ - * │ │ │ │ │ - * This API has been deprecated by Google. │ │ │ │ │ - * Developers are encouraged to migrate to v3 of the API; support for this │ │ │ │ │ - * is provided by <OpenLayers.Layer.Google.v3> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.Google.v2 = { │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Handler.Path.prototype.addPoint.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: termsOfUse │ │ │ │ │ - * {DOMElement} Div for Google's copyright and terms of use link │ │ │ │ │ + * Method: getCurrentPointIndex │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number} The index of the most recently drawn point. │ │ │ │ │ */ │ │ │ │ │ - termsOfUse: null, │ │ │ │ │ + getCurrentPointIndex: function() { │ │ │ │ │ + return this.line.geometry.components.length - 2; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: poweredBy │ │ │ │ │ - * {DOMElement} Div for Google's powered by logo and link │ │ │ │ │ + * Method: enforceTopology │ │ │ │ │ + * Simple topology enforcement for drawing interior rings. Ensures vertices │ │ │ │ │ + * of interior rings are contained by exterior ring. Other topology │ │ │ │ │ + * rules are enforced in <finalizeInteriorRing> to allow drawing of │ │ │ │ │ + * rings that intersect only during the sketch (e.g. a "C" shaped ring │ │ │ │ │ + * that nearly encloses another ring). │ │ │ │ │ */ │ │ │ │ │ - poweredBy: null, │ │ │ │ │ + enforceTopology: function(event) { │ │ │ │ │ + var point = event.vertex; │ │ │ │ │ + var components = this.line.geometry.components; │ │ │ │ │ + // ensure that vertices of interior ring are contained by exterior ring │ │ │ │ │ + if (!this.polygon.geometry.intersects(point)) { │ │ │ │ │ + var last = components[components.length - 3]; │ │ │ │ │ + point.x = last.x; │ │ │ │ │ + point.y = last.y; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: dragObject │ │ │ │ │ - * {GDraggableObject} Since 2.93, Google has exposed the ability to get │ │ │ │ │ - * the maps GDraggableObject. We can now use this for smooth panning │ │ │ │ │ + * Method: finishGeometry │ │ │ │ │ + * Finish the geometry and send it back to the control. │ │ │ │ │ */ │ │ │ │ │ - dragObject: null, │ │ │ │ │ + finishGeometry: function() { │ │ │ │ │ + var index = this.line.geometry.components.length - 2; │ │ │ │ │ + this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: loadMapObject │ │ │ │ │ - * Load the GMap and register appropriate event listeners. If we can't │ │ │ │ │ - * load GMap2, then display a warning message. │ │ │ │ │ + /** │ │ │ │ │ + * Method: finalizeInteriorRing │ │ │ │ │ + * Enforces that new ring has some area and doesn't contain vertices of any │ │ │ │ │ + * other rings. │ │ │ │ │ */ │ │ │ │ │ - loadMapObject: function() { │ │ │ │ │ - if (!this.type) { │ │ │ │ │ - this.type = G_NORMAL_MAP; │ │ │ │ │ - } │ │ │ │ │ - var mapObject, termsOfUse, poweredBy; │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - // there are already Google layers added to this map │ │ │ │ │ - mapObject = cache.mapObject; │ │ │ │ │ - termsOfUse = cache.termsOfUse; │ │ │ │ │ - poweredBy = cache.poweredBy; │ │ │ │ │ - // increment the layer count │ │ │ │ │ - ++cache.count; │ │ │ │ │ - } else { │ │ │ │ │ - // this is the first Google layer for this map │ │ │ │ │ - │ │ │ │ │ - var container = this.map.viewPortDiv; │ │ │ │ │ - var div = document.createElement("div"); │ │ │ │ │ - div.id = this.map.id + "_GMap2Container"; │ │ │ │ │ - div.style.position = "absolute"; │ │ │ │ │ - div.style.width = "100%"; │ │ │ │ │ - div.style.height = "100%"; │ │ │ │ │ - container.appendChild(div); │ │ │ │ │ - │ │ │ │ │ - // create GMap and shuffle elements │ │ │ │ │ - try { │ │ │ │ │ - mapObject = new GMap2(div); │ │ │ │ │ - │ │ │ │ │ - // move the ToS and branding stuff up to the container div │ │ │ │ │ - termsOfUse = div.lastChild; │ │ │ │ │ - container.appendChild(termsOfUse); │ │ │ │ │ - termsOfUse.style.zIndex = "1100"; │ │ │ │ │ - termsOfUse.style.right = ""; │ │ │ │ │ - termsOfUse.style.bottom = ""; │ │ │ │ │ - termsOfUse.className = "olLayerGoogleCopyright"; │ │ │ │ │ - │ │ │ │ │ - poweredBy = div.lastChild; │ │ │ │ │ - container.appendChild(poweredBy); │ │ │ │ │ - poweredBy.style.zIndex = "1100"; │ │ │ │ │ - poweredBy.style.right = ""; │ │ │ │ │ - poweredBy.style.bottom = ""; │ │ │ │ │ - poweredBy.className = "olLayerGooglePoweredBy gmnoprint"; │ │ │ │ │ - │ │ │ │ │ - } catch (e) { │ │ │ │ │ - throw (e); │ │ │ │ │ + finalizeInteriorRing: function() { │ │ │ │ │ + var ring = this.line.geometry; │ │ │ │ │ + // ensure that ring has some area │ │ │ │ │ + var modified = (ring.getArea() !== 0); │ │ │ │ │ + if (modified) { │ │ │ │ │ + // ensure that new ring doesn't intersect any other rings │ │ │ │ │ + var rings = this.polygon.geometry.components; │ │ │ │ │ + for (var i = rings.length - 2; i >= 0; --i) { │ │ │ │ │ + if (ring.intersects(rings[i])) { │ │ │ │ │ + modified = false; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (modified) { │ │ │ │ │ + // ensure that new ring doesn't contain any other rings │ │ │ │ │ + var target; │ │ │ │ │ + outer: for (var i = rings.length - 2; i > 0; --i) { │ │ │ │ │ + var points = rings[i].components; │ │ │ │ │ + for (var j = 0, jj = points.length; j < jj; ++j) { │ │ │ │ │ + if (ring.containsPoint(points[j])) { │ │ │ │ │ + modified = false; │ │ │ │ │ + break outer; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - // cache elements for use by any other google layers added to │ │ │ │ │ - // this same map │ │ │ │ │ - OpenLayers.Layer.Google.cache[this.map.id] = { │ │ │ │ │ - mapObject: mapObject, │ │ │ │ │ - termsOfUse: termsOfUse, │ │ │ │ │ - poweredBy: poweredBy, │ │ │ │ │ - count: 1 │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.mapObject = mapObject; │ │ │ │ │ - this.termsOfUse = termsOfUse; │ │ │ │ │ - this.poweredBy = poweredBy; │ │ │ │ │ - │ │ │ │ │ - // ensure this layer type is one of the mapObject types │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.mapObject.getMapTypes(), │ │ │ │ │ - this.type) === -1) { │ │ │ │ │ - this.mapObject.addMapType(this.type); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - //since v 2.93 getDragObject is now available. │ │ │ │ │ - if (typeof mapObject.getDragObject == "function") { │ │ │ │ │ - this.dragObject = mapObject.getDragObject(); │ │ │ │ │ + if (modified) { │ │ │ │ │ + if (this.polygon.state !== OpenLayers.State.INSERT) { │ │ │ │ │ + this.polygon.state = OpenLayers.State.UPDATE; │ │ │ │ │ + } │ │ │ │ │ } else { │ │ │ │ │ - this.dragPanMapObject = null; │ │ │ │ │ + this.polygon.geometry.removeComponent(ring); │ │ │ │ │ } │ │ │ │ │ + this.restoreFeature(); │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (this.isBaseLayer === false) { │ │ │ │ │ - this.setGMapVisibility(this.div.style.display !== "none"); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: cancel │ │ │ │ │ + * Finish the geometry and call the "cancel" callback. │ │ │ │ │ + */ │ │ │ │ │ + cancel: function() { │ │ │ │ │ + if (this.drawingHole) { │ │ │ │ │ + this.polygon.geometry.removeComponent(this.line.geometry); │ │ │ │ │ + this.restoreFeature(true); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ + return OpenLayers.Handler.Path.prototype.cancel.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: onMapResize │ │ │ │ │ + * Method: restoreFeature │ │ │ │ │ + * Move the feature from the sketch layer to the target layer. │ │ │ │ │ + * │ │ │ │ │ + * Properties: │ │ │ │ │ + * cancel - {Boolean} Cancel drawing. If falsey, the "sketchcomplete" event │ │ │ │ │ + * will be fired. │ │ │ │ │ */ │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - // workaround for resizing of invisible or not yet fully loaded layers │ │ │ │ │ - // where GMap2.checkResize() does not work. We need to load the GMap │ │ │ │ │ - // for the old div size, then checkResize(), and then call │ │ │ │ │ - // layer.moveTo() to trigger GMap.setCenter() (which will finish │ │ │ │ │ - // the GMap initialization). │ │ │ │ │ - if (this.visibility && this.mapObject.isLoaded()) { │ │ │ │ │ - this.mapObject.checkResize(); │ │ │ │ │ - } else { │ │ │ │ │ - if (!this._resized) { │ │ │ │ │ - var layer = this; │ │ │ │ │ - var handle = GEvent.addListener(this.mapObject, "load", function() { │ │ │ │ │ - GEvent.removeListener(handle); │ │ │ │ │ - delete layer._resized; │ │ │ │ │ - layer.mapObject.checkResize(); │ │ │ │ │ - layer.moveTo(layer.map.getCenter(), layer.map.getZoom()); │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - this._resized = true; │ │ │ │ │ + restoreFeature: function(cancel) { │ │ │ │ │ + this.control.layer.events.unregister( │ │ │ │ │ + "sketchcomplete", this, this.finalizeInteriorRing │ │ │ │ │ + ); │ │ │ │ │ + this.control.layer.events.unregister( │ │ │ │ │ + "sketchmodified", this, this.enforceTopology │ │ │ │ │ + ); │ │ │ │ │ + this.layer.removeFeatures([this.polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.control.layer.addFeatures([this.polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.drawingHole = false; │ │ │ │ │ + if (!cancel) { │ │ │ │ │ + // Re-trigger "sketchcomplete" so other listeners can do their │ │ │ │ │ + // business. While this is somewhat sloppy (if a listener is │ │ │ │ │ + // registered with registerPriority - not common - between the start │ │ │ │ │ + // and end of a single ring drawing - very uncommon - it will be │ │ │ │ │ + // called twice). │ │ │ │ │ + // TODO: In 3.0, collapse sketch handlers into geometry specific │ │ │ │ │ + // drawing controls. │ │ │ │ │ + this.control.layer.events.triggerEvent( │ │ │ │ │ + "sketchcomplete", { │ │ │ │ │ + feature: this.polygon │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setGMapVisibility │ │ │ │ │ - * Display the GMap container and associated elements. │ │ │ │ │ - * │ │ │ │ │ + * Method: destroyFeature │ │ │ │ │ + * Destroy temporary geometries │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * visible - {Boolean} Display the GMap elements. │ │ │ │ │ + * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ */ │ │ │ │ │ - setGMapVisibility: function(visible) { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - var container = this.mapObject.getContainer(); │ │ │ │ │ - if (visible === true) { │ │ │ │ │ - this.mapObject.setMapType(this.type); │ │ │ │ │ - container.style.display = ""; │ │ │ │ │ - this.termsOfUse.style.left = ""; │ │ │ │ │ - this.termsOfUse.style.display = ""; │ │ │ │ │ - this.poweredBy.style.display = ""; │ │ │ │ │ - cache.displayed = this.id; │ │ │ │ │ - } else { │ │ │ │ │ - if (cache.displayed === this.id) { │ │ │ │ │ - delete cache.displayed; │ │ │ │ │ - } │ │ │ │ │ - if (!cache.displayed) { │ │ │ │ │ - container.style.display = "none"; │ │ │ │ │ - this.termsOfUse.style.display = "none"; │ │ │ │ │ - // move ToU far to the left in addition to setting display │ │ │ │ │ - // to "none", because at the end of the GMap2 load │ │ │ │ │ - // sequence, display: none will be unset and ToU would be │ │ │ │ │ - // visible after loading a map with a google layer that is │ │ │ │ │ - // initially hidden. │ │ │ │ │ - this.termsOfUse.style.left = "-9999px"; │ │ │ │ │ - this.poweredBy.style.display = "none"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + OpenLayers.Handler.Path.prototype.destroyFeature.call( │ │ │ │ │ + this, force); │ │ │ │ │ + this.polygon = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getMapContainer │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} the GMap container's div │ │ │ │ │ + * Method: drawFeature │ │ │ │ │ + * Render geometries on the temporary layer. │ │ │ │ │ */ │ │ │ │ │ - getMapContainer: function() { │ │ │ │ │ - return this.mapObject.getContainer(); │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.polygon, this.style); │ │ │ │ │ + this.layer.drawFeature(this.point, this.style); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - // │ │ │ │ │ - // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ - // │ │ │ │ │ + /** │ │ │ │ │ + * Method: getSketch │ │ │ │ │ + * Return the sketch feature. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + getSketch: function() { │ │ │ │ │ + return this.polygon; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectBoundsFromOLBounds │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * olBounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * │ │ │ │ │ + * Method: getGeometry │ │ │ │ │ + * Return the sketch geometry. If <multi> is true, this will return │ │ │ │ │ + * a multi-part geometry. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} A MapObject Bounds, translated from olBounds │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ + * {<OpenLayers.Geometry.Polygon>} │ │ │ │ │ */ │ │ │ │ │ - getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ - var moBounds = null; │ │ │ │ │ - if (olBounds != null) { │ │ │ │ │ - var sw = this.sphericalMercator ? │ │ │ │ │ - this.inverseMercator(olBounds.bottom, olBounds.left) : │ │ │ │ │ - new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ - var ne = this.sphericalMercator ? │ │ │ │ │ - this.inverseMercator(olBounds.top, olBounds.right) : │ │ │ │ │ - new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ - moBounds = new GLatLngBounds(new GLatLng(sw.lat, sw.lon), │ │ │ │ │ - new GLatLng(ne.lat, ne.lon)); │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.polygon && this.polygon.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiPolygon([geometry]); │ │ │ │ │ } │ │ │ │ │ - return moBounds; │ │ │ │ │ + return geometry; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Polygon" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Handler/MouseWheel.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Interface Controls * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ - │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - // Get&Set Center, Zoom │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Handler.MouseWheel │ │ │ │ │ + * Handler for wheel up/down events. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MouseWheel = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setMapObjectCenter │ │ │ │ │ - * Set the mapObject to the specified center and zoom │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * center - {Object} MapObject LonLat format │ │ │ │ │ - * zoom - {int} MapObject zoom format │ │ │ │ │ + * Property: wheelListener │ │ │ │ │ + * {function} │ │ │ │ │ */ │ │ │ │ │ - setMapObjectCenter: function(center, zoom) { │ │ │ │ │ - this.mapObject.setCenter(center, zoom); │ │ │ │ │ - }, │ │ │ │ │ + wheelListener: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: dragPanMapObject │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * dX - {Integer} │ │ │ │ │ - * dY - {Integer} │ │ │ │ │ + * Property: interval │ │ │ │ │ + * {Integer} In order to increase server performance, an interval (in │ │ │ │ │ + * milliseconds) can be set to reduce the number of up/down events │ │ │ │ │ + * called. If set, a new up/down event will not be set until the │ │ │ │ │ + * interval has passed. │ │ │ │ │ + * Defaults to 0, meaning no interval. │ │ │ │ │ */ │ │ │ │ │ - dragPanMapObject: function(dX, dY) { │ │ │ │ │ - this.dragObject.moveBy(new GSize(-dX, dY)); │ │ │ │ │ - }, │ │ │ │ │ + interval: 0, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: maxDelta │ │ │ │ │ + * {Integer} Maximum delta to collect before breaking from the current │ │ │ │ │ + * interval. In cumulative mode, this also limits the maximum delta │ │ │ │ │ + * returned from the handler. Default is Number.POSITIVE_INFINITY. │ │ │ │ │ + */ │ │ │ │ │ + maxDelta: Number.POSITIVE_INFINITY, │ │ │ │ │ │ │ │ │ │ - // LonLat - Pixel Translation │ │ │ │ │ + /** │ │ │ │ │ + * Property: delta │ │ │ │ │ + * {Integer} When interval is set, delta collects the mousewheel z-deltas │ │ │ │ │ + * of the events that occur within the interval. │ │ │ │ │ + * See also the cumulative option │ │ │ │ │ + */ │ │ │ │ │ + delta: 0, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectLonLatFromMapObjectPixel │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moPixel - {Object} MapObject Pixel format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject LonLat translated from MapObject Pixel │ │ │ │ │ + * Property: cumulative │ │ │ │ │ + * {Boolean} When interval is set: true to collect all the mousewheel │ │ │ │ │ + * z-deltas, false to only record the delta direction (positive or │ │ │ │ │ + * negative) │ │ │ │ │ */ │ │ │ │ │ - getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return this.mapObject.fromContainerPixelToLatLng(moPixel); │ │ │ │ │ - }, │ │ │ │ │ + cumulative: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectPixelFromMapObjectLonLat │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Handler.MouseWheel │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject Pixel transtlated from MapObject LonLat │ │ │ │ │ + * control - {<OpenLayers.Control>} │ │ │ │ │ + * callbacks - {Object} An object containing a single function to be │ │ │ │ │ + * called when the drag operation is finished. │ │ │ │ │ + * The callback should expect to recieve a single │ │ │ │ │ + * argument, the point geometry. │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - return this.mapObject.fromLatLngToContainerPixel(moLonLat); │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.wheelListener = OpenLayers.Function.bindAsEventListener( │ │ │ │ │ + this.onWheelEvent, this │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ + this.wheelListener = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Bounds │ │ │ │ │ + /** │ │ │ │ │ + * Mouse ScrollWheel code thanks to http://adomas.org/javascript-mouse-wheel/ │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectZoomFromMapObjectBounds │ │ │ │ │ + * Method: onWheelEvent │ │ │ │ │ + * Catch the wheel event and handle it xbrowserly │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * moBounds - {Object} MapObject Bounds format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject Zoom for specified MapObject Bounds │ │ │ │ │ + * e - {Event} │ │ │ │ │ */ │ │ │ │ │ - getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - return this.mapObject.getBoundsZoomLevel(moBounds); │ │ │ │ │ - }, │ │ │ │ │ + onWheelEvent: function(e) { │ │ │ │ │ │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Primitives * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ + // make sure we have a map and check keyboard modifiers │ │ │ │ │ + if (!this.map || !this.checkModifiers(e)) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ + // Ride up the element's DOM hierarchy to determine if it or any of │ │ │ │ │ + // its ancestors was: │ │ │ │ │ + // * specifically marked as scrollable (CSS overflow property) │ │ │ │ │ + // * one of our layer divs or a div marked as scrollable │ │ │ │ │ + // ('olScrollable' CSS class) │ │ │ │ │ + // * the map div │ │ │ │ │ + // │ │ │ │ │ + var overScrollableDiv = false; │ │ │ │ │ + var allowScroll = false; │ │ │ │ │ + var overMapDiv = false; │ │ │ │ │ │ │ │ │ │ - // LonLat │ │ │ │ │ + var elem = OpenLayers.Event.element(e); │ │ │ │ │ + while ((elem != null) && !overMapDiv && !overScrollableDiv) { │ │ │ │ │ + │ │ │ │ │ + if (!overScrollableDiv) { │ │ │ │ │ + try { │ │ │ │ │ + var overflow; │ │ │ │ │ + if (elem.currentStyle) { │ │ │ │ │ + overflow = elem.currentStyle["overflow"]; │ │ │ │ │ + } else { │ │ │ │ │ + var style = │ │ │ │ │ + document.defaultView.getComputedStyle(elem, null); │ │ │ │ │ + overflow = style.getPropertyValue("overflow"); │ │ │ │ │ + } │ │ │ │ │ + overScrollableDiv = (overflow && │ │ │ │ │ + (overflow == "auto") || (overflow == "scroll")); │ │ │ │ │ + } catch (err) { │ │ │ │ │ + //sometimes when scrolling in a popup, this causes │ │ │ │ │ + // obscure browser error │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (!allowScroll) { │ │ │ │ │ + allowScroll = OpenLayers.Element.hasClass(elem, 'olScrollable'); │ │ │ │ │ + if (!allowScroll) { │ │ │ │ │ + for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ + // Are we in the layer div? Note that we have two cases │ │ │ │ │ + // here: one is to catch EventPane layers, which have a │ │ │ │ │ + // pane above the layer (layer.pane) │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if (elem == layer.div || elem == layer.pane) { │ │ │ │ │ + allowScroll = true; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + overMapDiv = (elem == this.map.div); │ │ │ │ │ + │ │ │ │ │ + elem = elem.parentNode; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Logic below is the following: │ │ │ │ │ + // │ │ │ │ │ + // If we are over a scrollable div or not over the map div: │ │ │ │ │ + // * do nothing (let the browser handle scrolling) │ │ │ │ │ + // │ │ │ │ │ + // otherwise │ │ │ │ │ + // │ │ │ │ │ + // If we are over the layer div or a 'olScrollable' div: │ │ │ │ │ + // * zoom/in out │ │ │ │ │ + // then │ │ │ │ │ + // * kill event (so as not to also scroll the page after zooming) │ │ │ │ │ + // │ │ │ │ │ + // otherwise │ │ │ │ │ + // │ │ │ │ │ + // Kill the event (dont scroll the page if we wheel over the │ │ │ │ │ + // layerswitcher or the pan/zoom control) │ │ │ │ │ + // │ │ │ │ │ + if (!overScrollableDiv && overMapDiv) { │ │ │ │ │ + if (allowScroll) { │ │ │ │ │ + var delta = 0; │ │ │ │ │ + │ │ │ │ │ + if (e.wheelDelta) { │ │ │ │ │ + delta = e.wheelDelta; │ │ │ │ │ + if (delta % 160 === 0) { │ │ │ │ │ + // opera have steps of 160 instead of 120 │ │ │ │ │ + delta = delta * 0.75; │ │ │ │ │ + } │ │ │ │ │ + delta = delta / 120; │ │ │ │ │ + } else if (e.detail) { │ │ │ │ │ + // detail in Firefox on OS X is 1/3 of Windows │ │ │ │ │ + // so force delta 1 / -1 │ │ │ │ │ + delta = -(e.detail / Math.abs(e.detail)); │ │ │ │ │ + } │ │ │ │ │ + this.delta += delta; │ │ │ │ │ + │ │ │ │ │ + window.clearTimeout(this._timeoutId); │ │ │ │ │ + if (this.interval && Math.abs(this.delta) < this.maxDelta) { │ │ │ │ │ + // store e because window.event might change during delay │ │ │ │ │ + var evt = OpenLayers.Util.extend({}, e); │ │ │ │ │ + this._timeoutId = window.setTimeout( │ │ │ │ │ + OpenLayers.Function.bind(function() { │ │ │ │ │ + this.wheelZoom(evt); │ │ │ │ │ + }, this), │ │ │ │ │ + this.interval │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + this.wheelZoom(e); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stop(e); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectLonLatFromLonLat │ │ │ │ │ + * Method: wheelZoom │ │ │ │ │ + * Given the wheel event, we carry out the appropriate zooming in or out, │ │ │ │ │ + * based on the 'wheelDelta' or 'detail' property of the event. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * lon - {Float} │ │ │ │ │ - * lat - {Float} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject LonLat built from lon and lat params │ │ │ │ │ + * e - {Event} │ │ │ │ │ */ │ │ │ │ │ - getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ - var gLatLng; │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ - gLatLng = new GLatLng(lonlat.lat, lonlat.lon); │ │ │ │ │ - } else { │ │ │ │ │ - gLatLng = new GLatLng(lat, lon); │ │ │ │ │ + wheelZoom: function(e) { │ │ │ │ │ + var delta = this.delta; │ │ │ │ │ + this.delta = 0; │ │ │ │ │ + │ │ │ │ │ + if (delta) { │ │ │ │ │ + e.xy = this.map.events.getMousePosition(e); │ │ │ │ │ + if (delta < 0) { │ │ │ │ │ + this.callback("down", │ │ │ │ │ + [e, this.cumulative ? Math.max(-this.maxDelta, delta) : -1]); │ │ │ │ │ + } else { │ │ │ │ │ + this.callback("up", │ │ │ │ │ + [e, this.cumulative ? Math.min(this.maxDelta, delta) : 1]); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return gLatLng; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - // Pixel │ │ │ │ │ + /** │ │ │ │ │ + * Method: activate │ │ │ │ │ + */ │ │ │ │ │ + activate: function(evt) { │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + //register mousewheel events specifically on the window and document │ │ │ │ │ + var wheelListener = this.wheelListener; │ │ │ │ │ + OpenLayers.Event.observe(window, "DOMMouseScroll", wheelListener); │ │ │ │ │ + OpenLayers.Event.observe(window, "mousewheel", wheelListener); │ │ │ │ │ + OpenLayers.Event.observe(document, "mousewheel", wheelListener); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectPixelFromXY │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {Integer} │ │ │ │ │ - * y - {Integer} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject Pixel from x and y parameters │ │ │ │ │ + * Method: deactivate │ │ │ │ │ */ │ │ │ │ │ - getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ - return new GPoint(x, y); │ │ │ │ │ - } │ │ │ │ │ + deactivate: function(evt) { │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + // unregister mousewheel events specifically on the window and document │ │ │ │ │ + var wheelListener = this.wheelListener; │ │ │ │ │ + OpenLayers.Event.stopObserving(window, "DOMMouseScroll", wheelListener); │ │ │ │ │ + OpenLayers.Event.stopObserving(window, "mousewheel", wheelListener); │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mousewheel", wheelListener); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -}; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.MouseWheel" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/KaMap.js │ │ │ │ │ + OpenLayers/Handler/Pinch.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.KaMap │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Handler.Pinch │ │ │ │ │ + * The pinch handler is used to deal with sequences of browser events related │ │ │ │ │ + * to pinch gestures. The handler is used by controls that want to know │ │ │ │ │ + * when a pinch sequence begins, when a pinch is happening, and when it has │ │ │ │ │ + * finished. │ │ │ │ │ + * │ │ │ │ │ + * Controls that use the pinch handler typically construct it with callbacks │ │ │ │ │ + * for 'start', 'move', and 'done'. Callbacks for these keys are │ │ │ │ │ + * called when the pinch begins, with each change, and when the pinch is │ │ │ │ │ + * done. │ │ │ │ │ + * │ │ │ │ │ + * Create a new pinch handler with the <OpenLayers.Handler.Pinch> constructor. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.KaMap = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ +OpenLayers.Handler.Pinch = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} KaMap Layer is always a base layer │ │ │ │ │ + /** │ │ │ │ │ + * Property: started │ │ │ │ │ + * {Boolean} When a touchstart event is received, we want to record it, │ │ │ │ │ + * but not set 'pinching' until the touchmove get started after │ │ │ │ │ + * starting. │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + started: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: DEFAULT_PARAMS │ │ │ │ │ - * {Object} parameters set by default. The default parameters set │ │ │ │ │ - * the format via the 'i' parameter to 'jpeg'. │ │ │ │ │ + * Property: stopDown │ │ │ │ │ + * {Boolean} Stop propagation of touchstart events from getting to │ │ │ │ │ + * listeners on the same element. Default is false. │ │ │ │ │ */ │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - i: 'jpeg', │ │ │ │ │ - map: '' │ │ │ │ │ - }, │ │ │ │ │ + stopDown: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.KaMap │ │ │ │ │ - * │ │ │ │ │ + * Property: pinching │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + pinching: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: last │ │ │ │ │ + * {Object} Object that store informations related to pinch last touch. │ │ │ │ │ + */ │ │ │ │ │ + last: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: start │ │ │ │ │ + * {Object} Object that store informations related to pinch touchstart. │ │ │ │ │ + */ │ │ │ │ │ + start: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Handler.Pinch │ │ │ │ │ + * Returns OpenLayers.Handler.Pinch │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * url - {String} │ │ │ │ │ - * params - {Object} Parameters to be sent to the HTTP server in the │ │ │ │ │ - * query string for the tile. The format can be set via the 'i' │ │ │ │ │ - * parameter (defaults to jpg) , and the map should be set via │ │ │ │ │ - * the 'map' parameter. It has been reported that ka-Map may behave │ │ │ │ │ - * inconsistently if your format parameter does not match the format │ │ │ │ │ - * parameter configured in your config.php. (See ticket #327 for more │ │ │ │ │ - * information.) │ │ │ │ │ - * options - {Object} Additional options for the layer. Any of the │ │ │ │ │ - * APIProperties listed on this layer, and any layer types it │ │ │ │ │ - * extends, can be overridden through the options parameter. │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ + * this handler. If a handler is being used without a control, the │ │ │ │ │ + * handlers setMap method must be overridden to deal properly with │ │ │ │ │ + * the map. │ │ │ │ │ + * callbacks - {Object} An object containing functions to be called when │ │ │ │ │ + * the pinch operation start, change, or is finished. The callbacks │ │ │ │ │ + * should expect to receive an object argument, which contains │ │ │ │ │ + * information about scale, distance, and position of touch points. │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, this.DEFAULT_PARAMS │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * │ │ │ │ │ + * Method: touchstart │ │ │ │ │ + * Handle touchstart events │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as │ │ │ │ │ - * parameters │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var mapRes = this.map.getResolution(); │ │ │ │ │ - var scale = Math.round((this.map.getScale() * 10000)) / 10000; │ │ │ │ │ - var pX = Math.round(bounds.left / mapRes); │ │ │ │ │ - var pY = -Math.round(bounds.top / mapRes); │ │ │ │ │ - return this.getFullRequestString({ │ │ │ │ │ - t: pY, │ │ │ │ │ - l: pX, │ │ │ │ │ - s: scale │ │ │ │ │ - }); │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + if (OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ + this.started = true; │ │ │ │ │ + this.last = this.start = { │ │ │ │ │ + distance: this.getDistance(evt.touches), │ │ │ │ │ + delta: 0, │ │ │ │ │ + scale: 1 │ │ │ │ │ + }; │ │ │ │ │ + this.callback("start", [evt, this.start]); │ │ │ │ │ + propagate = !this.stopDown; │ │ │ │ │ + } else if (this.started) { │ │ │ │ │ + // Some webkit versions send fake single-touch events during │ │ │ │ │ + // multitouch, which cause the drag handler to trigger │ │ │ │ │ + return false; │ │ │ │ │ + } else { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + } │ │ │ │ │ + // prevent document dragging │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + return propagate; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateGridLayout │ │ │ │ │ - * ka-Map uses the center point of the map as an origin for │ │ │ │ │ - * its tiles. Override calculateGridLayout to center tiles │ │ │ │ │ - * correctly for this case. │ │ │ │ │ + /** │ │ │ │ │ + * Method: touchmove │ │ │ │ │ + * Handle touchmove events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bound>} │ │ │ │ │ - * origin - {<OpenLayers.LonLat>} │ │ │ │ │ - * resolution - {Number} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} Object containing properties tilelon, tilelat, startcol, │ │ │ │ │ - * startrow │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - calculateGridLayout: function(bounds, origin, resolution) { │ │ │ │ │ - var tilelon = resolution * this.tileSize.w; │ │ │ │ │ - var tilelat = resolution * this.tileSize.h; │ │ │ │ │ - │ │ │ │ │ - var offsetlon = bounds.left; │ │ │ │ │ - var tilecol = Math.floor(offsetlon / tilelon) - this.buffer; │ │ │ │ │ - │ │ │ │ │ - var offsetlat = bounds.top; │ │ │ │ │ - var tilerow = Math.floor(offsetlat / tilelat) + this.buffer; │ │ │ │ │ - │ │ │ │ │ - return { │ │ │ │ │ - tilelon: tilelon, │ │ │ │ │ - tilelat: tilelat, │ │ │ │ │ - startcol: tilecol, │ │ │ │ │ - startrow: tilerow │ │ │ │ │ - }; │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + if (this.started && OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ + this.pinching = true; │ │ │ │ │ + var current = this.getPinchData(evt); │ │ │ │ │ + this.callback("move", [evt, current]); │ │ │ │ │ + this.last = current; │ │ │ │ │ + // prevent document dragging │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + } else if (this.started) { │ │ │ │ │ + // Some webkit versions send fake single-touch events during │ │ │ │ │ + // multitouch, which cause the drag handler to trigger │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getTileBoundsForGridIndex │ │ │ │ │ + * Method: touchend │ │ │ │ │ + * Handle touchend events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * row - {Number} The row of the grid │ │ │ │ │ - * col - {Number} The column of the grid │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} The bounds for the tile at (row, col) │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - getTileBoundsForGridIndex: function(row, col) { │ │ │ │ │ - var origin = this.getTileOrigin(); │ │ │ │ │ - var tileLayout = this.gridLayout; │ │ │ │ │ - var tilelon = tileLayout.tilelon; │ │ │ │ │ - var tilelat = tileLayout.tilelat; │ │ │ │ │ - var minX = (tileLayout.startcol + col) * tilelon; │ │ │ │ │ - var minY = (tileLayout.startrow - row) * tilelat; │ │ │ │ │ - return new OpenLayers.Bounds( │ │ │ │ │ - minX, minY, │ │ │ │ │ - minX + tilelon, minY + tilelat │ │ │ │ │ - ); │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + if (this.started && !OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + this.callback("done", [evt, this.start, this.last]); │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} │ │ │ │ │ - * │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activate the handler. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Kamap>} An exact clone of this OpenLayers.Layer.KaMap │ │ │ │ │ + * {Boolean} The handler was successfully activated. │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.KaMap(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.params, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + activated = true; │ │ │ │ │ } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - if (this.tileSize != null) { │ │ │ │ │ - obj.tileSize = this.tileSize.clone(); │ │ │ │ │ + /** │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivate the handler. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was successfully deactivated. │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + deactivated = true; │ │ │ │ │ } │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // we do not want to copy reference to grid, so we make a new array │ │ │ │ │ - obj.grid = []; │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ + /** │ │ │ │ │ + * Method: getDistance │ │ │ │ │ + * Get the distance in pixels between two touches. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * touches - {Array(Object)} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number} The distance in pixels. │ │ │ │ │ + */ │ │ │ │ │ + getDistance: function(touches) { │ │ │ │ │ + var t0 = touches[0]; │ │ │ │ │ + var t1 = touches[1]; │ │ │ │ │ + return Math.sqrt( │ │ │ │ │ + Math.pow(t0.olClientX - t1.olClientX, 2) + │ │ │ │ │ + Math.pow(t0.olClientY - t1.olClientY, 2) │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getTileBounds │ │ │ │ │ - * Returns The tile bounds for a layer given a pixel location. │ │ │ │ │ + * Method: getPinchData │ │ │ │ │ + * Get informations about the pinch event. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * viewPortPx - {<OpenLayers.Pixel>} The location in the viewport. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} Bounds of the tile at the given pixel location. │ │ │ │ │ + * {Object} Object that contains data about the current pinch. │ │ │ │ │ */ │ │ │ │ │ - getTileBounds: function(viewPortPx) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var tileMapWidth = resolution * this.tileSize.w; │ │ │ │ │ - var tileMapHeight = resolution * this.tileSize.h; │ │ │ │ │ - var mapPoint = this.getLonLatFromViewPortPx(viewPortPx); │ │ │ │ │ - var tileLeft = tileMapWidth * Math.floor(mapPoint.lon / tileMapWidth); │ │ │ │ │ - var tileBottom = tileMapHeight * Math.floor(mapPoint.lat / tileMapHeight); │ │ │ │ │ - return new OpenLayers.Bounds(tileLeft, tileBottom, │ │ │ │ │ - tileLeft + tileMapWidth, │ │ │ │ │ - tileBottom + tileMapHeight); │ │ │ │ │ + getPinchData: function(evt) { │ │ │ │ │ + var distance = this.getDistance(evt.touches); │ │ │ │ │ + var scale = distance / this.start.distance; │ │ │ │ │ + return { │ │ │ │ │ + distance: distance, │ │ │ │ │ + delta: this.last.distance - distance, │ │ │ │ │ + scale: scale │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.KaMap" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Pinch" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/Text.js │ │ │ │ │ + OpenLayers/Handler/Box.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Handler/Drag.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.Text │ │ │ │ │ - * Read Text format. Create a new instance with the <OpenLayers.Format.Text> │ │ │ │ │ - * constructor. This reads text which is formatted like CSV text, using │ │ │ │ │ - * tabs as the seperator by default. It provides parsing of data originally │ │ │ │ │ - * used in the MapViewerService, described on the wiki. This Format is used │ │ │ │ │ - * by the <OpenLayers.Layer.Text> class. │ │ │ │ │ + * Class: OpenLayers.Handler.Box │ │ │ │ │ + * Handler for dragging a rectangle across the map. Box is displayed │ │ │ │ │ + * on mouse down, moves on mouse move, and is finished on mouse up. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.Text = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ +OpenLayers.Handler.Box = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: dragHandler │ │ │ │ │ + * {<OpenLayers.Handler.Drag>} │ │ │ │ │ + */ │ │ │ │ │ + dragHandler: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: defaultStyle │ │ │ │ │ - * defaultStyle allows one to control the default styling of the features. │ │ │ │ │ - * It should be a symbolizer hash. By default, this is set to match the │ │ │ │ │ - * Layer.Text behavior, which is to use the default OpenLayers Icon. │ │ │ │ │ + * APIProperty: boxDivClassName │ │ │ │ │ + * {String} The CSS class to use for drawing the box. Default is │ │ │ │ │ + * olHandlerBoxZoomBox │ │ │ │ │ */ │ │ │ │ │ - defaultStyle: null, │ │ │ │ │ + boxDivClassName: 'olHandlerBoxZoomBox', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: extractStyles │ │ │ │ │ - * set to true to extract styles from the TSV files, using information │ │ │ │ │ - * from the image or icon, iconSize and iconOffset fields. This will result │ │ │ │ │ - * in features with a symbolizer (style) property set, using the │ │ │ │ │ - * default symbolizer specified in <defaultStyle>. Set to false if you │ │ │ │ │ - * wish to use a styleMap or OpenLayers.Style options to style your │ │ │ │ │ - * layer instead. │ │ │ │ │ + * Property: boxOffsets │ │ │ │ │ + * {Object} Caches box offsets from css. This is used by the getBoxOffsets │ │ │ │ │ + * method. │ │ │ │ │ */ │ │ │ │ │ - extractStyles: true, │ │ │ │ │ + boxOffsets: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.Text │ │ │ │ │ - * Create a new parser for TSV Text. │ │ │ │ │ + * Constructor: OpenLayers.Handler.Box │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * control - {<OpenLayers.Control>} │ │ │ │ │ + * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ + * functions. Various callbacks described below. │ │ │ │ │ + * options - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Named callbacks: │ │ │ │ │ + * start - Called when the box drag operation starts. │ │ │ │ │ + * done - Called when the box drag operation is finished. │ │ │ │ │ + * The callback should expect to receive a single argument, the box │ │ │ │ │ + * bounds or a pixel. If the box dragging didn't span more than a 5 │ │ │ │ │ + * pixel distance, a pixel will be returned instead of a bounds object. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.dragHandler = new OpenLayers.Handler.Drag( │ │ │ │ │ + this, { │ │ │ │ │ + down: this.startBox, │ │ │ │ │ + move: this.moveBox, │ │ │ │ │ + out: this.removeBox, │ │ │ │ │ + up: this.endBox │ │ │ │ │ + }, { │ │ │ │ │ + keyMask: this.keyMask │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (options.extractStyles !== false) { │ │ │ │ │ - options.defaultStyle = { │ │ │ │ │ - 'externalGraphic': OpenLayers.Util.getImageLocation("marker.png"), │ │ │ │ │ - 'graphicWidth': 21, │ │ │ │ │ - 'graphicHeight': 25, │ │ │ │ │ - 'graphicXOffset': -10.5, │ │ │ │ │ - 'graphicYOffset': -12.5 │ │ │ │ │ - }; │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ + if (this.dragHandler) { │ │ │ │ │ + this.dragHandler.destroy(); │ │ │ │ │ + this.dragHandler = null; │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - OpenLayers.Format.prototype.initialize.apply(this, [options]); │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Handler.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (this.dragHandler) { │ │ │ │ │ + this.dragHandler.setMap(map); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Return a list of features from a Tab Seperated Values text string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * text - {String} │ │ │ │ │ + * Method: startBox │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ + * Parameters: │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} │ │ │ │ │ */ │ │ │ │ │ - read: function(text) { │ │ │ │ │ - var lines = text.split('\n'); │ │ │ │ │ - var columns; │ │ │ │ │ - var features = []; │ │ │ │ │ - // length - 1 to allow for trailing new line │ │ │ │ │ - for (var lcv = 0; lcv < (lines.length - 1); lcv++) { │ │ │ │ │ - var currLine = lines[lcv].replace(/^\s*/, '').replace(/\s*$/, ''); │ │ │ │ │ + startBox: function(xy) { │ │ │ │ │ + this.callback("start", []); │ │ │ │ │ + this.zoomBox = OpenLayers.Util.createDiv('zoomBox', { │ │ │ │ │ + x: -9999, │ │ │ │ │ + y: -9999 │ │ │ │ │ + }); │ │ │ │ │ + this.zoomBox.className = this.boxDivClassName; │ │ │ │ │ + this.zoomBox.style.zIndex = this.map.Z_INDEX_BASE["Popup"] - 1; │ │ │ │ │ │ │ │ │ │ - if (currLine.charAt(0) != '#') { │ │ │ │ │ - /* not a comment */ │ │ │ │ │ + this.map.viewPortDiv.appendChild(this.zoomBox); │ │ │ │ │ │ │ │ │ │ - if (!columns) { │ │ │ │ │ - //First line is columns │ │ │ │ │ - columns = currLine.split('\t'); │ │ │ │ │ - } else { │ │ │ │ │ - var vals = currLine.split('\t'); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point(0, 0); │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - var style = this.defaultStyle ? │ │ │ │ │ - OpenLayers.Util.applyDefaults({}, this.defaultStyle) : │ │ │ │ │ - null; │ │ │ │ │ - var icon, iconSize, iconOffset, overflow; │ │ │ │ │ - var set = false; │ │ │ │ │ - for (var valIndex = 0; valIndex < vals.length; valIndex++) { │ │ │ │ │ - if (vals[valIndex]) { │ │ │ │ │ - if (columns[valIndex] == 'point') { │ │ │ │ │ - var coords = vals[valIndex].split(','); │ │ │ │ │ - geometry.y = parseFloat(coords[0]); │ │ │ │ │ - geometry.x = parseFloat(coords[1]); │ │ │ │ │ - set = true; │ │ │ │ │ - } else if (columns[valIndex] == 'lat') { │ │ │ │ │ - geometry.y = parseFloat(vals[valIndex]); │ │ │ │ │ - set = true; │ │ │ │ │ - } else if (columns[valIndex] == 'lon') { │ │ │ │ │ - geometry.x = parseFloat(vals[valIndex]); │ │ │ │ │ - set = true; │ │ │ │ │ - } else if (columns[valIndex] == 'title') │ │ │ │ │ - attributes['title'] = vals[valIndex]; │ │ │ │ │ - else if (columns[valIndex] == 'image' || │ │ │ │ │ - columns[valIndex] == 'icon' && style) { │ │ │ │ │ - style['externalGraphic'] = vals[valIndex]; │ │ │ │ │ - } else if (columns[valIndex] == 'iconSize' && style) { │ │ │ │ │ - var size = vals[valIndex].split(','); │ │ │ │ │ - style['graphicWidth'] = parseFloat(size[0]); │ │ │ │ │ - style['graphicHeight'] = parseFloat(size[1]); │ │ │ │ │ - } else if (columns[valIndex] == 'iconOffset' && style) { │ │ │ │ │ - var offset = vals[valIndex].split(','); │ │ │ │ │ - style['graphicXOffset'] = parseFloat(offset[0]); │ │ │ │ │ - style['graphicYOffset'] = parseFloat(offset[1]); │ │ │ │ │ - } else if (columns[valIndex] == 'description') { │ │ │ │ │ - attributes['description'] = vals[valIndex]; │ │ │ │ │ - } else if (columns[valIndex] == 'overflow') { │ │ │ │ │ - attributes['overflow'] = vals[valIndex]; │ │ │ │ │ - } else { │ │ │ │ │ - // For StyleMap filtering, allow additional │ │ │ │ │ - // columns to be stored as attributes. │ │ │ │ │ - attributes[columns[valIndex]] = vals[valIndex]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (set) { │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry.transform(this.externalProjection, │ │ │ │ │ - this.internalProjection); │ │ │ │ │ - } │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry, attributes, style); │ │ │ │ │ - features.push(feature); │ │ │ │ │ - } │ │ │ │ │ + OpenLayers.Element.addClass( │ │ │ │ │ + this.map.viewPortDiv, "olDrawBox" │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveBox │ │ │ │ │ + */ │ │ │ │ │ + moveBox: function(xy) { │ │ │ │ │ + var startX = this.dragHandler.start.x; │ │ │ │ │ + var startY = this.dragHandler.start.y; │ │ │ │ │ + var deltaX = Math.abs(startX - xy.x); │ │ │ │ │ + var deltaY = Math.abs(startY - xy.y); │ │ │ │ │ + │ │ │ │ │ + var offset = this.getBoxOffsets(); │ │ │ │ │ + this.zoomBox.style.width = (deltaX + offset.width + 1) + "px"; │ │ │ │ │ + this.zoomBox.style.height = (deltaY + offset.height + 1) + "px"; │ │ │ │ │ + this.zoomBox.style.left = (xy.x < startX ? │ │ │ │ │ + startX - deltaX - offset.left : startX - offset.left) + "px"; │ │ │ │ │ + this.zoomBox.style.top = (xy.y < startY ? │ │ │ │ │ + startY - deltaY - offset.top : startY - offset.top) + "px"; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: endBox │ │ │ │ │ + */ │ │ │ │ │ + endBox: function(end) { │ │ │ │ │ + var result; │ │ │ │ │ + if (Math.abs(this.dragHandler.start.x - end.x) > 5 || │ │ │ │ │ + Math.abs(this.dragHandler.start.y - end.y) > 5) { │ │ │ │ │ + var start = this.dragHandler.start; │ │ │ │ │ + var top = Math.min(start.y, end.y); │ │ │ │ │ + var bottom = Math.max(start.y, end.y); │ │ │ │ │ + var left = Math.min(start.x, end.x); │ │ │ │ │ + var right = Math.max(start.x, end.x); │ │ │ │ │ + result = new OpenLayers.Bounds(left, bottom, right, top); │ │ │ │ │ + } else { │ │ │ │ │ + result = this.dragHandler.start.clone(); // i.e. OL.Pixel │ │ │ │ │ + } │ │ │ │ │ + this.removeBox(); │ │ │ │ │ + │ │ │ │ │ + this.callback("done", [result]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeBox │ │ │ │ │ + * Remove the zoombox from the screen and nullify our reference to it. │ │ │ │ │ + */ │ │ │ │ │ + removeBox: function() { │ │ │ │ │ + this.map.viewPortDiv.removeChild(this.zoomBox); │ │ │ │ │ + this.zoomBox = null; │ │ │ │ │ + this.boxOffsets = null; │ │ │ │ │ + OpenLayers.Element.removeClass( │ │ │ │ │ + this.map.viewPortDiv, "olDrawBox" │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: activate │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.dragHandler.activate(); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + if (this.dragHandler.deactivate()) { │ │ │ │ │ + if (this.zoomBox) { │ │ │ │ │ + this.removeBox(); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - return features; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.Text" │ │ │ │ │ + /** │ │ │ │ │ + * Method: getBoxOffsets │ │ │ │ │ + * Determines border offsets for a box, according to the box model. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} an object with the following offsets: │ │ │ │ │ + * - left │ │ │ │ │ + * - right │ │ │ │ │ + * - top │ │ │ │ │ + * - bottom │ │ │ │ │ + * - width │ │ │ │ │ + * - height │ │ │ │ │ + */ │ │ │ │ │ + getBoxOffsets: function() { │ │ │ │ │ + if (!this.boxOffsets) { │ │ │ │ │ + // Determine the box model. If the testDiv's clientWidth is 3, then │ │ │ │ │ + // the borders are outside and we are dealing with the w3c box │ │ │ │ │ + // model. Otherwise, the browser uses the traditional box model and │ │ │ │ │ + // the borders are inside the box bounds, leaving us with a │ │ │ │ │ + // clientWidth of 1. │ │ │ │ │ + var testDiv = document.createElement("div"); │ │ │ │ │ + //testDiv.style.visibility = "hidden"; │ │ │ │ │ + testDiv.style.position = "absolute"; │ │ │ │ │ + testDiv.style.border = "1px solid black"; │ │ │ │ │ + testDiv.style.width = "3px"; │ │ │ │ │ + document.body.appendChild(testDiv); │ │ │ │ │ + var w3cBoxModel = testDiv.clientWidth == 3; │ │ │ │ │ + document.body.removeChild(testDiv); │ │ │ │ │ + │ │ │ │ │ + var left = parseInt(OpenLayers.Element.getStyle(this.zoomBox, │ │ │ │ │ + "border-left-width")); │ │ │ │ │ + var right = parseInt(OpenLayers.Element.getStyle( │ │ │ │ │ + this.zoomBox, "border-right-width")); │ │ │ │ │ + var top = parseInt(OpenLayers.Element.getStyle(this.zoomBox, │ │ │ │ │ + "border-top-width")); │ │ │ │ │ + var bottom = parseInt(OpenLayers.Element.getStyle( │ │ │ │ │ + this.zoomBox, "border-bottom-width")); │ │ │ │ │ + this.boxOffsets = { │ │ │ │ │ + left: left, │ │ │ │ │ + right: right, │ │ │ │ │ + top: top, │ │ │ │ │ + bottom: bottom, │ │ │ │ │ + width: w3cBoxModel === false ? left + right : 0, │ │ │ │ │ + height: w3cBoxModel === false ? top + bottom : 0 │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + return this.boxOffsets; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Box" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Text.js │ │ │ │ │ + OpenLayers/Handler/Hover.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Markers.js │ │ │ │ │ - * @requires OpenLayers/Format/Text.js │ │ │ │ │ - * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.Text │ │ │ │ │ - * This layer creates markers given data in a text file. The <location> │ │ │ │ │ - * property of the layer (specified as a property of the options argument │ │ │ │ │ - * in the <OpenLayers.Layer.Text> constructor) points to a tab delimited │ │ │ │ │ - * file with data used to create markers. │ │ │ │ │ - * │ │ │ │ │ - * The first row of the data file should be a header line with the column names │ │ │ │ │ - * of the data. Each column should be delimited by a tab space. The │ │ │ │ │ - * possible columns are: │ │ │ │ │ - * - *point* lat,lon of the point where a marker is to be placed │ │ │ │ │ - * - *lat* Latitude of the point where a marker is to be placed │ │ │ │ │ - * - *lon* Longitude of the point where a marker is to be placed │ │ │ │ │ - * - *icon* or *image* URL of marker icon to use. │ │ │ │ │ - * - *iconSize* Size of Icon to use. │ │ │ │ │ - * - *iconOffset* Where the top-left corner of the icon is to be placed │ │ │ │ │ - * relative to the latitude and longitude of the point. │ │ │ │ │ - * - *title* The text of the 'title' is placed inside an 'h2' marker │ │ │ │ │ - * inside a popup, which opens when the marker is clicked. │ │ │ │ │ - * - *description* The text of the 'description' is placed below the h2 │ │ │ │ │ - * in the popup. this can be plain text or HTML. │ │ │ │ │ - * │ │ │ │ │ - * Example text file: │ │ │ │ │ - * (code) │ │ │ │ │ - * lat lon title description iconSize iconOffset icon │ │ │ │ │ - * 10 20 title description 21,25 -10,-25 http://www.openlayers.org/dev/img/marker.png │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Handler.Hover │ │ │ │ │ + * The hover handler is to be used to emulate mouseovers on objects │ │ │ │ │ + * on the map that aren't DOM elements. For example one can use │ │ │ │ │ + * this handler to send WMS/GetFeatureInfo requests as the user │ │ │ │ │ + * moves the mouve over the map. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Markers> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Text = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ +OpenLayers.Handler.Hover = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: location │ │ │ │ │ - * {String} URL of text file. Must be specified in the "options" argument │ │ │ │ │ - * of the constructor. Can not be changed once passed in. │ │ │ │ │ + * APIProperty: delay │ │ │ │ │ + * {Integer} - Number of milliseconds between mousemoves before │ │ │ │ │ + * the event is considered a hover. Default is 500. │ │ │ │ │ */ │ │ │ │ │ - location: null, │ │ │ │ │ + delay: 500, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Array(<OpenLayers.Feature>)} │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: pixelTolerance │ │ │ │ │ + * {Integer} - Maximum number of pixels between mousemoves for │ │ │ │ │ + * an event to be considered a hover. Default is null. │ │ │ │ │ */ │ │ │ │ │ - features: null, │ │ │ │ │ + pixelTolerance: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: formatOptions │ │ │ │ │ - * {Object} Hash of options which should be passed to the format when it is │ │ │ │ │ - * created. Must be passed in the constructor. │ │ │ │ │ + * APIProperty: stopMove │ │ │ │ │ + * {Boolean} - Stop other listeners from being notified on mousemoves. │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - formatOptions: null, │ │ │ │ │ + stopMove: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: selectedFeature │ │ │ │ │ - * {<OpenLayers.Feature>} │ │ │ │ │ + /** │ │ │ │ │ + * Property: px │ │ │ │ │ + * {<OpenLayers.Pixel>} - The location of the last mousemove, expressed │ │ │ │ │ + * in pixels. │ │ │ │ │ */ │ │ │ │ │ - selectedFeature: null, │ │ │ │ │ + px: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Text │ │ │ │ │ - * Create a text layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * options - {Object} Object with properties to be set on the layer. │ │ │ │ │ - * Must include <location> property. │ │ │ │ │ + * Property: timerId │ │ │ │ │ + * {Number} - The id of the timer. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.features = []; │ │ │ │ │ - }, │ │ │ │ │ + timerId: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ + * Constructor: OpenLayers.Handler.Hover │ │ │ │ │ + * Construct a hover handler. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that initialized this │ │ │ │ │ + * handler. The control is assumed to have a valid map property; that │ │ │ │ │ + * map is used in the handler's own setMap method. │ │ │ │ │ + * callbacks - {Object} An object with keys corresponding to callbacks │ │ │ │ │ + * that will be called by the handler. The callbacks should │ │ │ │ │ + * expect to receive a single argument, the event. Callbacks for │ │ │ │ │ + * 'move', the mouse is moving, and 'pause', the mouse is pausing, │ │ │ │ │ + * are supported. │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * the handler. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - // Warning: Layer.Markers.destroy() must be called prior to calling │ │ │ │ │ - // clearFeatures() here, otherwise we leak memory. Indeed, if │ │ │ │ │ - // Layer.Markers.destroy() is called after clearFeatures(), it won't be │ │ │ │ │ - // able to remove the marker image elements from the layer's div since │ │ │ │ │ - // the markers will have been destroyed by clearFeatures(). │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.destroy.apply(this, arguments); │ │ │ │ │ - this.clearFeatures(); │ │ │ │ │ - this.features = null; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: loadText │ │ │ │ │ - * Start the load of the Text data. Don't do this when we first add the layer, │ │ │ │ │ - * since we may not be visible at any point, and it would therefore be a waste. │ │ │ │ │ + * Method: mousemove │ │ │ │ │ + * Called when the mouse moves on the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - loadText: function() { │ │ │ │ │ - if (!this.loaded) { │ │ │ │ │ - if (this.location != null) { │ │ │ │ │ - │ │ │ │ │ - var onFail = function(e) { │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - this.events.triggerEvent("loadstart"); │ │ │ │ │ - OpenLayers.Request.GET({ │ │ │ │ │ - url: this.location, │ │ │ │ │ - success: this.parseData, │ │ │ │ │ - failure: onFail, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.loaded = true; │ │ │ │ │ - } │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + if (this.passesTolerance(evt.xy)) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.callback('move', [evt]); │ │ │ │ │ + this.px = evt.xy; │ │ │ │ │ + // clone the evt so original properties can be accessed even │ │ │ │ │ + // if the browser deletes them during the delay │ │ │ │ │ + evt = OpenLayers.Util.extend({}, evt); │ │ │ │ │ + this.timerId = window.setTimeout( │ │ │ │ │ + OpenLayers.Function.bind(this.delayedCall, this, evt), │ │ │ │ │ + this.delay │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ + return !this.stopMove; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * If layer is visible and Text has not been loaded, load Text. │ │ │ │ │ - * │ │ │ │ │ + * Method: mouseout │ │ │ │ │ + * Called when the mouse goes out of the map. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {Object} │ │ │ │ │ - * zoomChanged - {Object} │ │ │ │ │ - * minor - {Object} │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, minor) { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - if (this.visibility && !this.loaded) { │ │ │ │ │ - this.loadText(); │ │ │ │ │ + mouseout: function(evt) { │ │ │ │ │ + if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.callback('move', [evt]); │ │ │ │ │ } │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseData │ │ │ │ │ + * Method: passesTolerance │ │ │ │ │ + * Determine whether the mouse move is within the optional pixel tolerance. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * ajaxRequest - {<OpenLayers.Request.XMLHttpRequest>} │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The mouse move is within the pixel tolerance. │ │ │ │ │ */ │ │ │ │ │ - parseData: function(ajaxRequest) { │ │ │ │ │ - var text = ajaxRequest.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 parser = new OpenLayers.Format.Text(options); │ │ │ │ │ - var features = parser.read(text); │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - var data = {}; │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - var location; │ │ │ │ │ - var iconSize, iconOffset; │ │ │ │ │ - │ │ │ │ │ - location = new OpenLayers.LonLat(feature.geometry.x, │ │ │ │ │ - feature.geometry.y); │ │ │ │ │ - │ │ │ │ │ - if (feature.style.graphicWidth && │ │ │ │ │ - feature.style.graphicHeight) { │ │ │ │ │ - iconSize = new OpenLayers.Size( │ │ │ │ │ - feature.style.graphicWidth, │ │ │ │ │ - feature.style.graphicHeight); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // FIXME: At the moment, we only use this if we have an │ │ │ │ │ - // externalGraphic, because icon has no setOffset API Method. │ │ │ │ │ - /** │ │ │ │ │ - * FIXME FIRST!! │ │ │ │ │ - * The Text format does all sorts of parseFloating │ │ │ │ │ - * The result of a parseFloat for a bogus string is NaN. That │ │ │ │ │ - * means the three possible values here are undefined, NaN, or a │ │ │ │ │ - * number. The previous check was an identity check for null. This │ │ │ │ │ - * means it was failing for all undefined or NaN. A slightly better │ │ │ │ │ - * check is for undefined. An even better check is to see if the │ │ │ │ │ - * value is a number (see #1441). │ │ │ │ │ - */ │ │ │ │ │ - if (feature.style.graphicXOffset !== undefined && │ │ │ │ │ - feature.style.graphicYOffset !== undefined) { │ │ │ │ │ - iconOffset = new OpenLayers.Pixel( │ │ │ │ │ - feature.style.graphicXOffset, │ │ │ │ │ - feature.style.graphicYOffset); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (feature.style.externalGraphic != null) { │ │ │ │ │ - data.icon = new OpenLayers.Icon(feature.style.externalGraphic, │ │ │ │ │ - iconSize, │ │ │ │ │ - iconOffset); │ │ │ │ │ - } else { │ │ │ │ │ - data.icon = OpenLayers.Marker.defaultIcon(); │ │ │ │ │ - │ │ │ │ │ - //allows for the case where the image url is not │ │ │ │ │ - // specified but the size is. use a default icon │ │ │ │ │ - // but change the size │ │ │ │ │ - if (iconSize != null) { │ │ │ │ │ - data.icon.setSize(iconSize); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if ((feature.attributes.title != null) && │ │ │ │ │ - (feature.attributes.description != null)) { │ │ │ │ │ - data['popupContentHTML'] = │ │ │ │ │ - '<h2>' + feature.attributes.title + '</h2>' + │ │ │ │ │ - '<p>' + feature.attributes.description + '</p>'; │ │ │ │ │ + passesTolerance: function(px) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + if (this.pixelTolerance && this.px) { │ │ │ │ │ + var dpx = Math.sqrt( │ │ │ │ │ + Math.pow(this.px.x - px.x, 2) + │ │ │ │ │ + Math.pow(this.px.y - px.y, 2) │ │ │ │ │ + ); │ │ │ │ │ + if (dpx < this.pixelTolerance) { │ │ │ │ │ + passes = false; │ │ │ │ │ } │ │ │ │ │ + } │ │ │ │ │ + return passes; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - data['overflow'] = feature.attributes.overflow || "auto"; │ │ │ │ │ - │ │ │ │ │ - var markerFeature = new OpenLayers.Feature(this, location, data); │ │ │ │ │ - this.features.push(markerFeature); │ │ │ │ │ - var marker = markerFeature.createMarker(); │ │ │ │ │ - if ((feature.attributes.title != null) && │ │ │ │ │ - (feature.attributes.description != null)) { │ │ │ │ │ - marker.events.register('click', markerFeature, this.markerClick); │ │ │ │ │ - } │ │ │ │ │ - this.addMarker(marker); │ │ │ │ │ + /** │ │ │ │ │ + * Method: clearTimer │ │ │ │ │ + * Clear the timer and set <timerId> to null. │ │ │ │ │ + */ │ │ │ │ │ + clearTimer: function() { │ │ │ │ │ + if (this.timerId != null) { │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ } │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: markerClick │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * Method: delayedCall │ │ │ │ │ + * Triggers pause callback. │ │ │ │ │ * │ │ │ │ │ - * Context: │ │ │ │ │ - * - {<OpenLayers.Feature>} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - markerClick: function(evt) { │ │ │ │ │ - var sameMarkerClicked = (this == this.layer.selectedFeature); │ │ │ │ │ - this.layer.selectedFeature = (!sameMarkerClicked) ? this : null; │ │ │ │ │ - for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ - this.layer.map.removePopup(this.layer.map.popups[i]); │ │ │ │ │ - } │ │ │ │ │ - if (!sameMarkerClicked) { │ │ │ │ │ - this.layer.map.addPopup(this.createPopup()); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ + delayedCall: function(evt) { │ │ │ │ │ + this.callback('pause', [evt]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clearFeatures │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the handler. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ - clearFeatures: function() { │ │ │ │ │ - if (this.features != null) { │ │ │ │ │ - while (this.features.length > 0) { │ │ │ │ │ - var feature = this.features[0]; │ │ │ │ │ - OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ - feature.destroy(); │ │ │ │ │ - } │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + deactivated = true; │ │ │ │ │ } │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Text" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Hover" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Bing.js │ │ │ │ │ + OpenLayers/Handler/Click.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.Bing │ │ │ │ │ - * Bing layer using direct tile access as provided by Bing Maps REST Services. │ │ │ │ │ - * See http://msdn.microsoft.com/en-us/library/ff701713.aspx for more │ │ │ │ │ - * information. Note: Terms of Service compliant use requires the map to be │ │ │ │ │ - * configured with an <OpenLayers.Control.Attribution> control and the │ │ │ │ │ - * attribution placed on or near the map. │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Handler.Click │ │ │ │ │ + * A handler for mouse clicks. The intention of this handler is to give │ │ │ │ │ + * controls more flexibility with handling clicks. Browsers trigger │ │ │ │ │ + * click events twice for a double-click. In addition, the mousedown, │ │ │ │ │ + * mousemove, mouseup sequence fires a click event. With this handler, │ │ │ │ │ + * controls can decide whether to ignore clicks associated with a double │ │ │ │ │ + * click. By setting a <pixelTolerance>, controls can also ignore clicks │ │ │ │ │ + * that include a drag. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Handler.Click> constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.XYZ> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Bing = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ +OpenLayers.Handler.Click = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: delay │ │ │ │ │ + * {Number} Number of milliseconds between clicks before the event is │ │ │ │ │ + * considered a double-click. │ │ │ │ │ + */ │ │ │ │ │ + delay: 300, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: key │ │ │ │ │ - * {String} API key for Bing maps, get your own key │ │ │ │ │ - * at http://bingmapsportal.com/ . │ │ │ │ │ + * APIProperty: single │ │ │ │ │ + * {Boolean} Handle single clicks. Default is true. If false, clicks │ │ │ │ │ + * will not be reported. If true, single-clicks will be reported. │ │ │ │ │ */ │ │ │ │ │ - key: null, │ │ │ │ │ + single: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: serverResolutions │ │ │ │ │ - * {Array} the resolutions provided by the Bing servers. │ │ │ │ │ + * APIProperty: double │ │ │ │ │ + * {Boolean} Handle double-clicks. Default is false. │ │ │ │ │ */ │ │ │ │ │ - serverResolutions: [ │ │ │ │ │ - 156543.03390625, 78271.516953125, 39135.7584765625, │ │ │ │ │ - 19567.87923828125, 9783.939619140625, 4891.9698095703125, │ │ │ │ │ - 2445.9849047851562, 1222.9924523925781, 611.4962261962891, │ │ │ │ │ - 305.74811309814453, 152.87405654907226, 76.43702827453613, │ │ │ │ │ - 38.218514137268066, 19.109257068634033, 9.554628534317017, │ │ │ │ │ - 4.777314267158508, 2.388657133579254, 1.194328566789627, │ │ │ │ │ - 0.5971642833948135, 0.29858214169740677, 0.14929107084870338, │ │ │ │ │ - 0.07464553542435169 │ │ │ │ │ - ], │ │ │ │ │ + 'double': false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: attributionTemplate │ │ │ │ │ - * {String} │ │ │ │ │ + * APIProperty: pixelTolerance │ │ │ │ │ + * {Number} Maximum number of pixels between mouseup and mousedown for an │ │ │ │ │ + * event to be considered a click. Default is 0. If set to an │ │ │ │ │ + * integer value, clicks with a drag greater than the value will be │ │ │ │ │ + * ignored. This property can only be set when the handler is │ │ │ │ │ + * constructed. │ │ │ │ │ */ │ │ │ │ │ - attributionTemplate: '<span class="olBingAttribution ${type}">' + │ │ │ │ │ - '<div><a target="_blank" href="http://www.bing.com/maps/">' + │ │ │ │ │ - '<img src="${logo}" /></a></div>${copyrights}' + │ │ │ │ │ - '<a style="white-space: nowrap" target="_blank" ' + │ │ │ │ │ - 'href="http://www.microsoft.com/maps/product/terms.html">' + │ │ │ │ │ - 'Terms of Use</a></span>', │ │ │ │ │ + pixelTolerance: 0, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: metadata │ │ │ │ │ - * {Object} Metadata for this layer, as returned by the callback script │ │ │ │ │ + * APIProperty: dblclickTolerance │ │ │ │ │ + * {Number} Maximum distance in pixels between clicks for a sequence of │ │ │ │ │ + * events to be considered a double click. Default is 13. If the │ │ │ │ │ + * distance between two clicks is greater than this value, a double- │ │ │ │ │ + * click will not be fired. │ │ │ │ │ */ │ │ │ │ │ - metadata: null, │ │ │ │ │ + dblclickTolerance: 13, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: protocolRegex │ │ │ │ │ - * {RegExp} Regular expression to match and replace http: in bing urls │ │ │ │ │ + * APIProperty: stopSingle │ │ │ │ │ + * {Boolean} Stop other listeners from being notified of clicks. Default │ │ │ │ │ + * is false. If true, any listeners registered before this one for │ │ │ │ │ + * click or rightclick events will not be notified. │ │ │ │ │ */ │ │ │ │ │ - protocolRegex: /^http:/i, │ │ │ │ │ + stopSingle: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: type │ │ │ │ │ - * {String} The layer identifier. Any non-birdseye imageryType │ │ │ │ │ - * from http://msdn.microsoft.com/en-us/library/ff701716.aspx can be │ │ │ │ │ - * used. Default is "Road". │ │ │ │ │ + * APIProperty: stopDouble │ │ │ │ │ + * {Boolean} Stop other listeners from being notified of double-clicks. │ │ │ │ │ + * Default is false. If true, any click listeners registered before │ │ │ │ │ + * this one will not be notified of *any* double-click events. │ │ │ │ │ + * │ │ │ │ │ + * The one caveat with stopDouble is that given a map with two click │ │ │ │ │ + * handlers, one with stopDouble true and the other with stopSingle │ │ │ │ │ + * true, the stopSingle handler should be activated last to get │ │ │ │ │ + * uniform cross-browser performance. Since IE triggers one click │ │ │ │ │ + * with a dblclick and FF triggers two, if a stopSingle handler is │ │ │ │ │ + * activated first, all it gets in IE is a single click when the │ │ │ │ │ + * second handler stops propagation on the dblclick. │ │ │ │ │ */ │ │ │ │ │ - type: "Road", │ │ │ │ │ + stopDouble: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: culture │ │ │ │ │ - * {String} The culture identifier. See http://msdn.microsoft.com/en-us/library/ff701709.aspx │ │ │ │ │ - * for the definition and the possible values. Default is "en-US". │ │ │ │ │ + * Property: timerId │ │ │ │ │ + * {Number} The id of the timeout waiting to clear the <delayedCall>. │ │ │ │ │ */ │ │ │ │ │ - culture: "en-US", │ │ │ │ │ + timerId: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: metadataParams │ │ │ │ │ - * {Object} Optional url parameters for the Get Imagery Metadata request │ │ │ │ │ - * as described here: http://msdn.microsoft.com/en-us/library/ff701716.aspx │ │ │ │ │ + * Property: down │ │ │ │ │ + * {Object} Object that store relevant information about the last │ │ │ │ │ + * mousedown or touchstart. Its 'xy' OpenLayers.Pixel property gives │ │ │ │ │ + * the average location of the mouse/touch event. Its 'touches' │ │ │ │ │ + * property records clientX/clientY of each touches. │ │ │ │ │ */ │ │ │ │ │ - metadataParams: null, │ │ │ │ │ + down: null, │ │ │ │ │ │ │ │ │ │ - /** APIProperty: tileOptions │ │ │ │ │ - * {Object} optional configuration options for <OpenLayers.Tile> instances │ │ │ │ │ - * created by this Layer. Default is │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * {crossOriginKeyword: 'anonymous'} │ │ │ │ │ - * (end) │ │ │ │ │ + /** │ │ │ │ │ + * Property: last │ │ │ │ │ + * {Object} Object that store relevant information about the last │ │ │ │ │ + * mousemove or touchmove. Its 'xy' OpenLayers.Pixel property gives │ │ │ │ │ + * the average location of the mouse/touch event. Its 'touches' │ │ │ │ │ + * property records clientX/clientY of each touches. │ │ │ │ │ */ │ │ │ │ │ - tileOptions: null, │ │ │ │ │ + last: null, │ │ │ │ │ │ │ │ │ │ - /** APIProperty: protocol │ │ │ │ │ - * {String} Protocol to use to fetch Imagery Metadata, tiles and bing logo │ │ │ │ │ - * Can be 'http:' 'https:' or '' │ │ │ │ │ - * │ │ │ │ │ - * Warning: tiles may not be available under both HTTP and HTTPS protocols. │ │ │ │ │ - * Microsoft approved use of both HTTP and HTTPS urls for tiles. However │ │ │ │ │ - * this is undocumented and the Imagery Metadata API always returns HTTP │ │ │ │ │ - * urls. │ │ │ │ │ - * │ │ │ │ │ - * Default is '', unless when executed from a file:/// uri, in which case │ │ │ │ │ - * it is 'http:'. │ │ │ │ │ + /** │ │ │ │ │ + * Property: first │ │ │ │ │ + * {Object} When waiting for double clicks, this object will store │ │ │ │ │ + * information about the first click in a two click sequence. │ │ │ │ │ */ │ │ │ │ │ - protocol: ~window.location.href.indexOf('http') ? '' : 'http:', │ │ │ │ │ + first: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Bing │ │ │ │ │ - * Create a new Bing layer. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var road = new OpenLayers.Layer.Bing({ │ │ │ │ │ - * name: "My Bing Aerial Layer", │ │ │ │ │ - * type: "Aerial", │ │ │ │ │ - * key: "my-api-key-here", │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Configuration properties for the layer. │ │ │ │ │ - * │ │ │ │ │ - * Required configuration properties: │ │ │ │ │ - * key - {String} Bing Maps API key for your application. Get one at │ │ │ │ │ - * http://bingmapsportal.com/. │ │ │ │ │ - * type - {String} The layer identifier. Any non-birdseye imageryType │ │ │ │ │ - * from http://msdn.microsoft.com/en-us/library/ff701716.aspx can be │ │ │ │ │ - * used. │ │ │ │ │ - * │ │ │ │ │ - * Any other documented layer properties can be provided in the config object. │ │ │ │ │ + * Property: rightclickTimerId │ │ │ │ │ + * {Number} The id of the right mouse timeout waiting to clear the │ │ │ │ │ + * <delayedEvent>. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - sphericalMercator: true │ │ │ │ │ - }, options); │ │ │ │ │ - var name = options.name || "Bing " + (options.type || this.type); │ │ │ │ │ - │ │ │ │ │ - var newArgs = [name, null, options]; │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, newArgs); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - crossOriginKeyword: 'anonymous' │ │ │ │ │ - }, this.options.tileOptions); │ │ │ │ │ - this.loadMetadata(); │ │ │ │ │ - }, │ │ │ │ │ + rightclickTimerId: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: loadMetadata │ │ │ │ │ + * Constructor: OpenLayers.Handler.Click │ │ │ │ │ + * Create a new click handler. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ + * this handler. If a handler is being used without a control, the │ │ │ │ │ + * handler's setMap method must be overridden to deal properly with │ │ │ │ │ + * the map. │ │ │ │ │ + * callbacks - {Object} An object with keys corresponding to callbacks │ │ │ │ │ + * that will be called by the handler. The callbacks should │ │ │ │ │ + * expect to recieve a single argument, the click event. │ │ │ │ │ + * Callbacks for 'click' and 'dblclick' are supported. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * handler. │ │ │ │ │ */ │ │ │ │ │ - loadMetadata: function() { │ │ │ │ │ - this._callbackId = "_callback_" + this.id.replace(/\./g, "_"); │ │ │ │ │ - // link the processMetadata method to the global scope and bind it │ │ │ │ │ - // to this instance │ │ │ │ │ - window[this._callbackId] = OpenLayers.Function.bind( │ │ │ │ │ - OpenLayers.Layer.Bing.processMetadata, this │ │ │ │ │ - ); │ │ │ │ │ - var params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - key: this.key, │ │ │ │ │ - jsonp: this._callbackId, │ │ │ │ │ - include: "ImageryProviders" │ │ │ │ │ - }, this.metadataParams); │ │ │ │ │ - var url = this.protocol + "//dev.virtualearth.net/REST/v1/Imagery/Metadata/" + │ │ │ │ │ - this.type + "?" + OpenLayers.Util.getParameterString(params); │ │ │ │ │ - var script = document.createElement("script"); │ │ │ │ │ - script.type = "text/javascript"; │ │ │ │ │ - script.src = url; │ │ │ │ │ - script.id = this._callbackId; │ │ │ │ │ - document.getElementsByTagName("head")[0].appendChild(script); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: initLayer │ │ │ │ │ + * Method: touchstart │ │ │ │ │ + * Handle touchstart. │ │ │ │ │ * │ │ │ │ │ - * Sets layer properties according to the metadata provided by the API │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - initLayer: function() { │ │ │ │ │ - var res = this.metadata.resourceSets[0].resources[0]; │ │ │ │ │ - var url = res.imageUrl.replace("{quadkey}", "${quadkey}"); │ │ │ │ │ - url = url.replace("{culture}", this.culture); │ │ │ │ │ - url = url.replace(this.protocolRegex, this.protocol); │ │ │ │ │ - this.url = []; │ │ │ │ │ - for (var i = 0; i < res.imageUrlSubdomains.length; ++i) { │ │ │ │ │ - this.url.push(url.replace("{subdomain}", res.imageUrlSubdomains[i])); │ │ │ │ │ - } │ │ │ │ │ - this.addOptions({ │ │ │ │ │ - maxResolution: Math.min( │ │ │ │ │ - this.serverResolutions[res.zoomMin], │ │ │ │ │ - this.maxResolution || Number.POSITIVE_INFINITY │ │ │ │ │ - ), │ │ │ │ │ - numZoomLevels: Math.min( │ │ │ │ │ - res.zoomMax + 1 - res.zoomMin, this.numZoomLevels │ │ │ │ │ - ) │ │ │ │ │ - }, true); │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - this.redraw(); │ │ │ │ │ - } │ │ │ │ │ - this.updateAttribution(); │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + this.down = this.getEventInfo(evt); │ │ │ │ │ + this.last = this.getEventInfo(evt); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ + * Method: touchmove │ │ │ │ │ + * Store position of last move, because touchend event can have │ │ │ │ │ + * an empty "touches" property. │ │ │ │ │ * │ │ │ │ │ - * Paramters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - if (!this.url) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var xyz = this.getXYZ(bounds), │ │ │ │ │ - x = xyz.x, │ │ │ │ │ - y = xyz.y, │ │ │ │ │ - z = xyz.z; │ │ │ │ │ - var quadDigits = []; │ │ │ │ │ - for (var i = z; i > 0; --i) { │ │ │ │ │ - var digit = '0'; │ │ │ │ │ - var mask = 1 << (i - 1); │ │ │ │ │ - if ((x & mask) != 0) { │ │ │ │ │ - digit++; │ │ │ │ │ - } │ │ │ │ │ - if ((y & mask) != 0) { │ │ │ │ │ - digit++; │ │ │ │ │ - digit++; │ │ │ │ │ - } │ │ │ │ │ - quadDigits.push(digit); │ │ │ │ │ - } │ │ │ │ │ - var quadKey = quadDigits.join(""); │ │ │ │ │ - var url = this.selectUrl('' + x + y + z, this.url); │ │ │ │ │ - │ │ │ │ │ - return OpenLayers.String.format(url, { │ │ │ │ │ - 'quadkey': quadKey │ │ │ │ │ - }); │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + this.last = this.getEventInfo(evt); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: updateAttribution │ │ │ │ │ - * Updates the attribution according to the requirements outlined in │ │ │ │ │ - * http://gis.638310.n2.nabble.com/Bing-imagery-td5789168.html │ │ │ │ │ + * Method: touchend │ │ │ │ │ + * Correctly set event xy property, and add lastTouches to have │ │ │ │ │ + * touches property from last touchstart or touchmove │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - updateAttribution: function() { │ │ │ │ │ - var metadata = this.metadata; │ │ │ │ │ - if (!metadata.resourceSets || !this.map || !this.map.center) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var res = metadata.resourceSets[0].resources[0]; │ │ │ │ │ - var extent = this.map.getExtent().transform( │ │ │ │ │ - this.map.getProjectionObject(), │ │ │ │ │ - new OpenLayers.Projection("EPSG:4326") │ │ │ │ │ - ); │ │ │ │ │ - var providers = res.imageryProviders || [], │ │ │ │ │ - zoom = OpenLayers.Util.indexOf(this.serverResolutions, │ │ │ │ │ - this.getServerResolution()), │ │ │ │ │ - copyrights = "", │ │ │ │ │ - provider, i, ii, j, jj, bbox, coverage; │ │ │ │ │ - for (i = 0, ii = providers.length; i < ii; ++i) { │ │ │ │ │ - provider = providers[i]; │ │ │ │ │ - for (j = 0, jj = provider.coverageAreas.length; j < jj; ++j) { │ │ │ │ │ - coverage = provider.coverageAreas[j]; │ │ │ │ │ - // axis order provided is Y,X │ │ │ │ │ - bbox = OpenLayers.Bounds.fromArray(coverage.bbox, true); │ │ │ │ │ - if (extent.intersectsBounds(bbox) && │ │ │ │ │ - zoom <= coverage.zoomMax && zoom >= coverage.zoomMin) { │ │ │ │ │ - copyrights += provider.attribution + " "; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + // touchstart may not have been allowed to propagate │ │ │ │ │ + if (this.down) { │ │ │ │ │ + evt.xy = this.last.xy; │ │ │ │ │ + evt.lastTouches = this.last.touches; │ │ │ │ │ + this.handleSingle(evt); │ │ │ │ │ + this.down = null; │ │ │ │ │ } │ │ │ │ │ - var logo = metadata.brandLogoUri.replace(this.protocolRegex, this.protocol); │ │ │ │ │ - this.attribution = OpenLayers.String.format(this.attributionTemplate, { │ │ │ │ │ - type: this.type.toLowerCase(), │ │ │ │ │ - logo: logo, │ │ │ │ │ - copyrights: copyrights │ │ │ │ │ - }); │ │ │ │ │ - this.map && this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "attribution" │ │ │ │ │ - }); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ + * Method: mousedown │ │ │ │ │ + * Handle mousedown. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - setMap: function() { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.register("moveend", this, this.updateAttribution); │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + this.down = this.getEventInfo(evt); │ │ │ │ │ + this.last = this.getEventInfo(evt); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} │ │ │ │ │ + * Method: mouseup │ │ │ │ │ + * Handle mouseup. Installed to support collection of right mouse events. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Bing>} An exact clone of this <OpenLayers.Layer.Bing> │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Bing(this.options); │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + │ │ │ │ │ + // Collect right mouse clicks from the mouseup │ │ │ │ │ + // IE - ignores the second right click in mousedown so using │ │ │ │ │ + // mouseup instead │ │ │ │ │ + if (this.checkModifiers(evt) && this.control.handleRightClicks && │ │ │ │ │ + OpenLayers.Event.isRightClick(evt)) { │ │ │ │ │ + propagate = this.rightclick(evt); │ │ │ │ │ } │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - return obj; │ │ │ │ │ + │ │ │ │ │ + return propagate; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * Method: rightclick │ │ │ │ │ + * Handle rightclick. For a dblrightclick, we get two clicks so we need │ │ │ │ │ + * to always register for dblrightclick to properly handle single │ │ │ │ │ + * clicks. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.map && │ │ │ │ │ - this.map.events.unregister("moveend", this, this.updateAttribution); │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Bing" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Function: OpenLayers.Layer.Bing.processMetadata │ │ │ │ │ - * This function will be bound to an instance, linked to the global scope with │ │ │ │ │ - * an id, and called by the JSONP script returned by the API. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * metadata - {Object} metadata as returned by the API │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.Bing.processMetadata = function(metadata) { │ │ │ │ │ - this.metadata = metadata; │ │ │ │ │ - this.initLayer(); │ │ │ │ │ - var script = document.getElementById(this._callbackId); │ │ │ │ │ - script.parentNode.removeChild(script); │ │ │ │ │ - window[this._callbackId] = undefined; // cannot delete from window in IE │ │ │ │ │ - delete this._callbackId; │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Tile/UTFGrid.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Tile.js │ │ │ │ │ - * @requires OpenLayers/Format/JSON.js │ │ │ │ │ - * @requires OpenLayers/Request.js │ │ │ │ │ - */ │ │ │ │ │ + rightclick: function(evt) { │ │ │ │ │ + if (this.passesTolerance(evt)) { │ │ │ │ │ + if (this.rightclickTimerId != null) { │ │ │ │ │ + //Second click received before timeout this must be │ │ │ │ │ + // a double click │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.callback('dblrightclick', [evt]); │ │ │ │ │ + return !this.stopDouble; │ │ │ │ │ + } else { │ │ │ │ │ + //Set the rightclickTimerId, send evt only if double is │ │ │ │ │ + // true else trigger single │ │ │ │ │ + var clickEvent = this['double'] ? │ │ │ │ │ + OpenLayers.Util.extend({}, evt) : │ │ │ │ │ + this.callback('rightclick', [evt]); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Tile.UTFGrid │ │ │ │ │ - * Instances of OpenLayers.Tile.UTFGrid are used to manage │ │ │ │ │ - * UTFGrids. This is an unusual tile type in that it doesn't have a │ │ │ │ │ - * rendered image; only a 'hit grid' that can be used to │ │ │ │ │ - * look up feature attributes. │ │ │ │ │ - * │ │ │ │ │ - * See the <OpenLayers.Tile.UTFGrid> constructor for details on constructing a │ │ │ │ │ - * new instance. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Tile> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Tile.UTFGrid = OpenLayers.Class(OpenLayers.Tile, { │ │ │ │ │ + var delayedRightCall = OpenLayers.Function.bind( │ │ │ │ │ + this.delayedRightCall, │ │ │ │ │ + this, │ │ │ │ │ + clickEvent │ │ │ │ │ + ); │ │ │ │ │ + this.rightclickTimerId = window.setTimeout( │ │ │ │ │ + delayedRightCall, this.delay │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return !this.stopSingle; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: url │ │ │ │ │ - * {String} │ │ │ │ │ - * The URL of the UTFGrid file being requested. Provided by the <getURL> │ │ │ │ │ - * method. │ │ │ │ │ + /** │ │ │ │ │ + * Method: delayedRightCall │ │ │ │ │ + * Sets <rightclickTimerId> to null. And optionally triggers the │ │ │ │ │ + * rightclick callback if evt is set. │ │ │ │ │ */ │ │ │ │ │ - url: null, │ │ │ │ │ + delayedRightCall: function(evt) { │ │ │ │ │ + this.rightclickTimerId = null; │ │ │ │ │ + if (evt) { │ │ │ │ │ + this.callback('rightclick', [evt]); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: utfgridResolution │ │ │ │ │ - * {Number} │ │ │ │ │ - * Ratio of the pixel width to the width of a UTFGrid data point. If an │ │ │ │ │ - * entry in the grid represents a 4x4 block of pixels, the │ │ │ │ │ - * utfgridResolution would be 4. Default is 2. │ │ │ │ │ + * Method: click │ │ │ │ │ + * Handle click events from the browser. This is registered as a listener │ │ │ │ │ + * for click events and should not be called from other events in this │ │ │ │ │ + * handler. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - utfgridResolution: 2, │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + if (!this.last) { │ │ │ │ │ + this.last = this.getEventInfo(evt); │ │ │ │ │ + } │ │ │ │ │ + this.handleSingle(evt); │ │ │ │ │ + return !this.stopSingle; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: json │ │ │ │ │ - * {Object} │ │ │ │ │ - * Stores the parsed JSON tile data structure. │ │ │ │ │ + /** │ │ │ │ │ + * Method: dblclick │ │ │ │ │ + * Handle dblclick. For a dblclick, we get two clicks in some browsers │ │ │ │ │ + * (FF) and one in others (IE). So we need to always register for │ │ │ │ │ + * dblclick to properly handle single clicks. This method is registered │ │ │ │ │ + * as a listener for the dblclick browser event. It should *not* be │ │ │ │ │ + * called by other methods in this handler. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - json: null, │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + this.handleDouble(evt); │ │ │ │ │ + return !this.stopDouble; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: format │ │ │ │ │ - * {OpenLayers.Format.JSON} │ │ │ │ │ - * Parser instance used to parse JSON for cross browser support. The native │ │ │ │ │ - * JSON.parse method will be used where available (all except IE<8). │ │ │ │ │ + * Method: handleDouble │ │ │ │ │ + * Handle double-click sequence. │ │ │ │ │ */ │ │ │ │ │ - format: null, │ │ │ │ │ + handleDouble: function(evt) { │ │ │ │ │ + if (this.passesDblclickTolerance(evt)) { │ │ │ │ │ + if (this["double"]) { │ │ │ │ │ + this.callback("dblclick", [evt]); │ │ │ │ │ + } │ │ │ │ │ + // to prevent a dblclick from firing the click callback in IE │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Tile.UTFGrid │ │ │ │ │ - * Constructor for a new <OpenLayers.Tile.UTFGrid> instance. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer>} layer that the tile will go in. │ │ │ │ │ - * position - {<OpenLayers.Pixel>} │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * url - {<String>} Deprecated. Remove me in 3.0. │ │ │ │ │ - * size - {<OpenLayers.Size>} │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * Method: handleSingle │ │ │ │ │ + * Handle single click sequence. │ │ │ │ │ */ │ │ │ │ │ + handleSingle: function(evt) { │ │ │ │ │ + if (this.passesTolerance(evt)) { │ │ │ │ │ + if (this.timerId != null) { │ │ │ │ │ + // already received a click │ │ │ │ │ + if (this.last.touches && this.last.touches.length === 1) { │ │ │ │ │ + // touch device, no dblclick event - this may be a double │ │ │ │ │ + if (this["double"]) { │ │ │ │ │ + // on Android don't let the browser zoom on the page │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + } │ │ │ │ │ + this.handleDouble(evt); │ │ │ │ │ + } │ │ │ │ │ + // if we're not in a touch environment we clear the click timer │ │ │ │ │ + // if we've got a second touch, we'll get two touchend events │ │ │ │ │ + if (!this.last.touches || this.last.touches.length !== 2) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // remember the first click info so we can compare to the second │ │ │ │ │ + this.first = this.getEventInfo(evt); │ │ │ │ │ + // set the timer, send evt only if single is true │ │ │ │ │ + //use a clone of the event object because it will no longer │ │ │ │ │ + //be a valid event object in IE in the timer callback │ │ │ │ │ + var clickEvent = this.single ? │ │ │ │ │ + OpenLayers.Util.extend({}, evt) : null; │ │ │ │ │ + this.queuePotentialClick(clickEvent); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up. │ │ │ │ │ + * Method: queuePotentialClick │ │ │ │ │ + * This method is separated out largely to make testing easier (so we │ │ │ │ │ + * don't have to override window.setTimeout) │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.clear(); │ │ │ │ │ - OpenLayers.Tile.prototype.destroy.apply(this, arguments); │ │ │ │ │ + queuePotentialClick: function(evt) { │ │ │ │ │ + this.timerId = window.setTimeout( │ │ │ │ │ + OpenLayers.Function.bind(this.delayedCall, this, evt), │ │ │ │ │ + this.delay │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Check that a tile should be drawn, and draw it. │ │ │ │ │ - * In the case of UTFGrids, "drawing" it means fetching and │ │ │ │ │ - * parsing the json. │ │ │ │ │ - * │ │ │ │ │ + * Method: passesTolerance │ │ │ │ │ + * Determine whether the event is within the optional pixel tolerance. Note │ │ │ │ │ + * that the pixel tolerance check only works if mousedown events get to │ │ │ │ │ + * the listeners registered here. If they are stopped by other elements, │ │ │ │ │ + * the <pixelTolerance> will have no effect here (this method will always │ │ │ │ │ + * return true). │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Was a tile drawn? │ │ │ │ │ + * {Boolean} The click is within the pixel tolerance (if specified). │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - var drawn = OpenLayers.Tile.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (drawn) { │ │ │ │ │ - if (this.isLoading) { │ │ │ │ │ - this.abortLoading(); │ │ │ │ │ - //if we're already loading, send 'reload' instead of 'loadstart'. │ │ │ │ │ - this.events.triggerEvent("reload"); │ │ │ │ │ - } else { │ │ │ │ │ - this.isLoading = true; │ │ │ │ │ - this.events.triggerEvent("loadstart"); │ │ │ │ │ - } │ │ │ │ │ - this.url = this.layer.getURL(this.bounds); │ │ │ │ │ - │ │ │ │ │ - if (this.layer.useJSONP) { │ │ │ │ │ - // Use JSONP method to avoid xbrowser policy │ │ │ │ │ - var ols = new OpenLayers.Protocol.Script({ │ │ │ │ │ - url: this.url, │ │ │ │ │ - callback: function(response) { │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ - this.json = response.data; │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - ols.read(); │ │ │ │ │ - this.request = ols; │ │ │ │ │ - } else { │ │ │ │ │ - // Use standard XHR │ │ │ │ │ - this.request = OpenLayers.Request.GET({ │ │ │ │ │ - url: this.url, │ │ │ │ │ - callback: function(response) { │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ - if (response.status === 200) { │ │ │ │ │ - this.parseData(response.responseText); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + passesTolerance: function(evt) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + if (this.pixelTolerance != null && this.down && this.down.xy) { │ │ │ │ │ + passes = this.pixelTolerance >= this.down.xy.distanceTo(evt.xy); │ │ │ │ │ + // for touch environments, we also enforce that all touches │ │ │ │ │ + // start and end within the given tolerance to be considered a click │ │ │ │ │ + if (passes && this.touch && │ │ │ │ │ + this.down.touches.length === this.last.touches.length) { │ │ │ │ │ + // the touchend event doesn't come with touches, so we check │ │ │ │ │ + // down and last │ │ │ │ │ + for (var i = 0, ii = this.down.touches.length; i < ii; ++i) { │ │ │ │ │ + if (this.getTouchDistance( │ │ │ │ │ + this.down.touches[i], │ │ │ │ │ + this.last.touches[i] │ │ │ │ │ + ) > this.pixelTolerance) { │ │ │ │ │ + passes = false; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - this.unload(); │ │ │ │ │ } │ │ │ │ │ - return drawn; │ │ │ │ │ + return passes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: abortLoading │ │ │ │ │ - * Cancel a pending request. │ │ │ │ │ + /** │ │ │ │ │ + * Method: getTouchDistance │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The pixel displacement between two touches. │ │ │ │ │ */ │ │ │ │ │ - abortLoading: function() { │ │ │ │ │ - if (this.request) { │ │ │ │ │ - this.request.abort(); │ │ │ │ │ - delete this.request; │ │ │ │ │ - } │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ + getTouchDistance: function(from, to) { │ │ │ │ │ + return Math.sqrt( │ │ │ │ │ + Math.pow(from.clientX - to.clientX, 2) + │ │ │ │ │ + Math.pow(from.clientY - to.clientY, 2) │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureInfo │ │ │ │ │ - * Get feature information associated with a pixel offset. If the pixel │ │ │ │ │ - * offset corresponds to a feature, the returned object will have id │ │ │ │ │ - * and data properties. Otherwise, null will be returned. │ │ │ │ │ - * │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * i - {Number} X-axis pixel offset (from top left of tile) │ │ │ │ │ - * j - {Number} Y-axis pixel offset (from top left of tile) │ │ │ │ │ + * Method: passesDblclickTolerance │ │ │ │ │ + * Determine whether the event is within the optional double-cick pixel │ │ │ │ │ + * tolerance. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} Object with feature id and data properties corresponding to the │ │ │ │ │ - * given pixel offset. │ │ │ │ │ + * {Boolean} The click is within the double-click pixel tolerance. │ │ │ │ │ */ │ │ │ │ │ - getFeatureInfo: function(i, j) { │ │ │ │ │ - var info = null; │ │ │ │ │ - if (this.json) { │ │ │ │ │ - var id = this.getFeatureId(i, j); │ │ │ │ │ - if (id !== null) { │ │ │ │ │ - info = { │ │ │ │ │ - id: id, │ │ │ │ │ - data: this.json.data[id] │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ + passesDblclickTolerance: function(evt) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + if (this.down && this.first) { │ │ │ │ │ + passes = this.down.xy.distanceTo(this.first.xy) <= this.dblclickTolerance; │ │ │ │ │ } │ │ │ │ │ - return info; │ │ │ │ │ + return passes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureId │ │ │ │ │ - * Get the identifier for the feature associated with a pixel offset. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * i - {Number} X-axis pixel offset (from top left of tile) │ │ │ │ │ - * j - {Number} Y-axis pixel offset (from top left of tile) │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The feature identifier corresponding to the given pixel offset. │ │ │ │ │ - * Returns null if pixel doesn't correspond to a feature. │ │ │ │ │ + * Method: clearTimer │ │ │ │ │ + * Clear the timer and set <timerId> to null. │ │ │ │ │ */ │ │ │ │ │ - getFeatureId: function(i, j) { │ │ │ │ │ - var id = null; │ │ │ │ │ - if (this.json) { │ │ │ │ │ - var resolution = this.utfgridResolution; │ │ │ │ │ - var row = Math.floor(j / resolution); │ │ │ │ │ - var col = Math.floor(i / resolution); │ │ │ │ │ - var charCode = this.json.grid[row].charCodeAt(col); │ │ │ │ │ - var index = this.indexFromCharCode(charCode); │ │ │ │ │ - var keys = this.json.keys; │ │ │ │ │ - if (!isNaN(index) && (index in keys)) { │ │ │ │ │ - id = keys[index]; │ │ │ │ │ - } │ │ │ │ │ + clearTimer: function() { │ │ │ │ │ + if (this.timerId != null) { │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.rightclickTimerId != null) { │ │ │ │ │ + window.clearTimeout(this.rightclickTimerId); │ │ │ │ │ + this.rightclickTimerId = null; │ │ │ │ │ } │ │ │ │ │ - return id; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: indexFromCharCode │ │ │ │ │ - * Given a character code for one of the UTFGrid "grid" characters, │ │ │ │ │ - * resolve the integer index for the feature id in the UTFGrid "keys" │ │ │ │ │ - * array. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * charCode - {Integer} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} Index for the feature id from the keys array. │ │ │ │ │ + * Method: delayedCall │ │ │ │ │ + * Sets <timerId> to null. And optionally triggers the click callback if │ │ │ │ │ + * evt is set. │ │ │ │ │ */ │ │ │ │ │ - indexFromCharCode: function(charCode) { │ │ │ │ │ - if (charCode >= 93) { │ │ │ │ │ - charCode--; │ │ │ │ │ - } │ │ │ │ │ - if (charCode >= 35) { │ │ │ │ │ - charCode--; │ │ │ │ │ + delayedCall: function(evt) { │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + if (evt) { │ │ │ │ │ + this.callback("click", [evt]); │ │ │ │ │ } │ │ │ │ │ - return charCode - 32; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseData │ │ │ │ │ - * Parse the JSON from a request │ │ │ │ │ + * Method: getEventInfo │ │ │ │ │ + * This method allows us to store event information without storing the │ │ │ │ │ + * actual event. In touch devices (at least), the same event is │ │ │ │ │ + * modified between touchstart, touchmove, and touchend. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * str - {String} UTFGrid as a JSON string. │ │ │ │ │ - * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} parsed javascript data │ │ │ │ │ + * {Object} An object with event related info. │ │ │ │ │ */ │ │ │ │ │ - parseData: function(str) { │ │ │ │ │ - if (!this.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.JSON(); │ │ │ │ │ + getEventInfo: function(evt) { │ │ │ │ │ + var touches; │ │ │ │ │ + if (evt.touches) { │ │ │ │ │ + var len = evt.touches.length; │ │ │ │ │ + touches = new Array(len); │ │ │ │ │ + var touch; │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + touch = evt.touches[i]; │ │ │ │ │ + touches[i] = { │ │ │ │ │ + clientX: touch.olClientX, │ │ │ │ │ + clientY: touch.olClientY │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.json = this.format.read(str); │ │ │ │ │ + return { │ │ │ │ │ + xy: evt.xy, │ │ │ │ │ + touches: touches │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clear │ │ │ │ │ - * Delete data stored with this tile. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the handler. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ - clear: function() { │ │ │ │ │ - this.json = null; │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.down = null; │ │ │ │ │ + this.first = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + deactivated = true; │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Tile.UTFGrid" │ │ │ │ │ - │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Click" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/UTFGrid.js │ │ │ │ │ + OpenLayers/Handler/Feature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ - * @requires OpenLayers/Tile/UTFGrid.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.UTFGrid │ │ │ │ │ - * This Layer reads from UTFGrid tiled data sources. Since UTFGrids are │ │ │ │ │ - * essentially JSON-based ASCII art with attached attributes, they are not │ │ │ │ │ - * visibly rendered. In order to use them in the map, you must add a │ │ │ │ │ - * <OpenLayers.Control.UTFGrid> control as well. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Handler.Feature │ │ │ │ │ + * Handler to respond to mouse events related to a drawn feature. Callbacks │ │ │ │ │ + * with the following keys will be notified of the following events │ │ │ │ │ + * associated with features: click, clickout, over, out, and dblclick. │ │ │ │ │ * │ │ │ │ │ - * (start code) │ │ │ │ │ - * var world_utfgrid = new OpenLayers.Layer.UTFGrid({ │ │ │ │ │ - * url: "/tiles/world_utfgrid/${z}/${x}/${y}.json", │ │ │ │ │ - * utfgridResolution: 4, │ │ │ │ │ - * displayInLayerSwitcher: false │ │ │ │ │ - * ); │ │ │ │ │ - * map.addLayer(world_utfgrid); │ │ │ │ │ - * │ │ │ │ │ - * var control = new OpenLayers.Control.UTFGrid({ │ │ │ │ │ - * layers: [world_utfgrid], │ │ │ │ │ - * handlerMode: 'move', │ │ │ │ │ - * callback: function(dataLookup) { │ │ │ │ │ - * // do something with returned data │ │ │ │ │ - * } │ │ │ │ │ - * }) │ │ │ │ │ - * (end code) │ │ │ │ │ + * This handler stops event propagation for mousedown and mouseup if those │ │ │ │ │ + * browser events target features that can be selected. │ │ │ │ │ * │ │ │ │ │ - * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.XYZ> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.UTFGrid = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ +OpenLayers.Handler.Feature = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * Default is false, as UTFGrids are designed to be a transparent overlay layer. │ │ │ │ │ + * Property: EVENTMAP │ │ │ │ │ + * {Object} A object mapping the browser events to objects with callback │ │ │ │ │ + * keys for in and out. │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: false, │ │ │ │ │ + EVENTMAP: { │ │ │ │ │ + 'click': { │ │ │ │ │ + 'in': 'click', │ │ │ │ │ + 'out': 'clickout' │ │ │ │ │ + }, │ │ │ │ │ + 'mousemove': { │ │ │ │ │ + 'in': 'over', │ │ │ │ │ + 'out': 'out' │ │ │ │ │ + }, │ │ │ │ │ + 'dblclick': { │ │ │ │ │ + 'in': 'dblclick', │ │ │ │ │ + 'out': null │ │ │ │ │ + }, │ │ │ │ │ + 'mousedown': { │ │ │ │ │ + 'in': null, │ │ │ │ │ + 'out': null │ │ │ │ │ + }, │ │ │ │ │ + 'mouseup': { │ │ │ │ │ + 'in': null, │ │ │ │ │ + 'out': null │ │ │ │ │ + }, │ │ │ │ │ + 'touchstart': { │ │ │ │ │ + 'in': 'click', │ │ │ │ │ + 'out': 'clickout' │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: projection │ │ │ │ │ - * {<OpenLayers.Projection>} │ │ │ │ │ - * Source projection for the UTFGrids. Default is "EPSG:900913". │ │ │ │ │ + * Property: feature │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} The last feature that was hovered. │ │ │ │ │ */ │ │ │ │ │ - projection: new OpenLayers.Projection("EPSG:900913"), │ │ │ │ │ + feature: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: useJSONP │ │ │ │ │ - * {Boolean} │ │ │ │ │ - * Should we use a JSONP script approach instead of a standard AJAX call? │ │ │ │ │ - * │ │ │ │ │ - * Set to true for using utfgrids from another server. │ │ │ │ │ - * Avoids same-domain policy restrictions. │ │ │ │ │ - * Note that this only works if the server accepts │ │ │ │ │ - * the callback GET parameter and dynamically │ │ │ │ │ - * wraps the returned json in a function call. │ │ │ │ │ - * │ │ │ │ │ - * Default is false │ │ │ │ │ + * Property: lastFeature │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} The last feature that was handled. │ │ │ │ │ */ │ │ │ │ │ - useJSONP: false, │ │ │ │ │ + lastFeature: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String} │ │ │ │ │ - * URL tempate for UTFGrid tiles. Include x, y, and z parameters. │ │ │ │ │ - * E.g. "/tiles/${z}/${x}/${y}.json" │ │ │ │ │ + * Property: down │ │ │ │ │ + * {<OpenLayers.Pixel>} The location of the last mousedown. │ │ │ │ │ */ │ │ │ │ │ + down: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: utfgridResolution │ │ │ │ │ - * {Number} │ │ │ │ │ - * Ratio of the pixel width to the width of a UTFGrid data point. If an │ │ │ │ │ - * entry in the grid represents a 4x4 block of pixels, the │ │ │ │ │ - * utfgridResolution would be 4. Default is 2 (specified in │ │ │ │ │ - * <OpenLayers.Tile.UTFGrid>). │ │ │ │ │ + * Property: up │ │ │ │ │ + * {<OpenLayers.Pixel>} The location of the last mouseup. │ │ │ │ │ */ │ │ │ │ │ + up: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: tileClass │ │ │ │ │ - * {<OpenLayers.Tile>} The tile class to use for this layer. │ │ │ │ │ - * Defaults is <OpenLayers.Tile.UTFGrid>. │ │ │ │ │ + * Property: clickTolerance │ │ │ │ │ + * {Number} The number of pixels the mouse can move between mousedown │ │ │ │ │ + * and mouseup for the event to still be considered a click. │ │ │ │ │ + * Dragging the map should not trigger the click and clickout callbacks │ │ │ │ │ + * unless the map is moved by less than this tolerance. Defaults to 4. │ │ │ │ │ */ │ │ │ │ │ - tileClass: OpenLayers.Tile.UTFGrid, │ │ │ │ │ + clickTolerance: 4, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.UTFGrid │ │ │ │ │ - * Create a new UTFGrid layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Configuration properties for the layer. │ │ │ │ │ - * │ │ │ │ │ - * Required configuration properties: │ │ │ │ │ - * url - {String} The url template for UTFGrid tiles. See the <url> property. │ │ │ │ │ + * Property: geometryTypes │ │ │ │ │ + * To restrict dragging to a limited set of geometry types, send a list │ │ │ │ │ + * of strings corresponding to the geometry class names. │ │ │ │ │ + * │ │ │ │ │ + * @type Array(String) │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply( │ │ │ │ │ - this, [options.name, options.url, {}, options] │ │ │ │ │ - ); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - utfgridResolution: this.utfgridResolution │ │ │ │ │ - }, this.tileOptions); │ │ │ │ │ - }, │ │ │ │ │ + geometryTypes: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createBackBuffer │ │ │ │ │ - * The UTFGrid cannot create a back buffer, so this method is overriden. │ │ │ │ │ + * Property: stopClick │ │ │ │ │ + * {Boolean} If stopClick is set to true, handled clicks do not │ │ │ │ │ + * propagate to other click listeners. Otherwise, handled clicks │ │ │ │ │ + * do propagate. Unhandled clicks always propagate, whatever the │ │ │ │ │ + * value of stopClick. Defaults to true. │ │ │ │ │ */ │ │ │ │ │ - createBackBuffer: function() {}, │ │ │ │ │ + stopClick: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} Only used by a subclass of this layer. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.UTFGrid>} An exact clone of this OpenLayers.Layer.UTFGrid │ │ │ │ │ + * Property: stopDown │ │ │ │ │ + * {Boolean} If stopDown is set to true, handled mousedowns do not │ │ │ │ │ + * propagate to other mousedown listeners. Otherwise, handled │ │ │ │ │ + * mousedowns do propagate. Unhandled mousedowns always propagate, │ │ │ │ │ + * whatever the value of stopDown. Defaults to true. │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.UTFGrid(this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + stopDown: true, │ │ │ │ │ │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: stopUp │ │ │ │ │ + * {Boolean} If stopUp is set to true, handled mouseups do not │ │ │ │ │ + * propagate to other mouseup listeners. Otherwise, handled mouseups │ │ │ │ │ + * do propagate. Unhandled mouseups always propagate, whatever the │ │ │ │ │ + * value of stopUp. Defaults to false. │ │ │ │ │ + */ │ │ │ │ │ + stopUp: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: getFeatureInfo │ │ │ │ │ - * Get details about a feature associated with a map location. The object │ │ │ │ │ - * returned will have id and data properties. If the given location │ │ │ │ │ - * doesn't correspond to a feature, null will be returned. │ │ │ │ │ + * Constructor: OpenLayers.Handler.Feature │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * location - {<OpenLayers.LonLat>} map location │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Object representing the feature id and UTFGrid data │ │ │ │ │ - * corresponding to the given map location. Returns null if the given │ │ │ │ │ - * location doesn't hit a feature. │ │ │ │ │ + * control - {<OpenLayers.Control>} │ │ │ │ │ + * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ + * callbacks - {Object} An object with a 'over' property whos value is │ │ │ │ │ + * a function to be called when the mouse is over a feature. The │ │ │ │ │ + * callback should expect to recieve a single argument, the feature. │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - getFeatureInfo: function(location) { │ │ │ │ │ - var info = null; │ │ │ │ │ - var tileInfo = this.getTileData(location); │ │ │ │ │ - if (tileInfo && tileInfo.tile) { │ │ │ │ │ - info = tileInfo.tile.getFeatureInfo(tileInfo.i, tileInfo.j); │ │ │ │ │ - } │ │ │ │ │ - return info; │ │ │ │ │ + initialize: function(control, layer, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ + this.layer = layer; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getFeatureId │ │ │ │ │ - * Get the identifier for the feature associated with a map location. │ │ │ │ │ + * Method: touchstart │ │ │ │ │ + * Handle touchstart events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * location - {<OpenLayers.LonLat>} map location │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} The feature identifier corresponding to the given map location. │ │ │ │ │ - * Returns null if the location doesn't hit a feature. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - getFeatureId: function(location) { │ │ │ │ │ - var id = null; │ │ │ │ │ - var info = this.getTileData(location); │ │ │ │ │ - if (info.tile) { │ │ │ │ │ - id = info.tile.getFeatureId(info.i, info.j); │ │ │ │ │ - } │ │ │ │ │ - return id; │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + return OpenLayers.Event.isMultiTouch(evt) ? │ │ │ │ │ + true : this.mousedown(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.UTFGrid" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/WMS.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.WMS │ │ │ │ │ - * Instances of OpenLayers.Layer.WMS are used to display data from OGC Web │ │ │ │ │ - * Mapping Services. Create a new WMS layer with the <OpenLayers.Layer.WMS> │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.WMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Constant: DEFAULT_PARAMS │ │ │ │ │ - * {Object} Hashtable of default parameter key/value pairs │ │ │ │ │ + * Method: touchmove │ │ │ │ │ + * Handle touchmove events. We just prevent the browser default behavior, │ │ │ │ │ + * for Android Webkit not to select text when moving the finger after │ │ │ │ │ + * selecting a feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - service: "WMS", │ │ │ │ │ - version: "1.1.1", │ │ │ │ │ - request: "GetMap", │ │ │ │ │ - styles: "", │ │ │ │ │ - format: "image/jpeg" │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} Default is true for WMS layer │ │ │ │ │ - */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: encodeBBOX │ │ │ │ │ - * {Boolean} Should the BBOX commas be encoded? The WMS spec says 'no', │ │ │ │ │ - * but some services want it that way. Default false. │ │ │ │ │ - */ │ │ │ │ │ - encodeBBOX: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: noMagic │ │ │ │ │ - * {Boolean} If true, the image format will not be automagicaly switched │ │ │ │ │ - * from image/jpeg to image/png or image/gif when using │ │ │ │ │ - * TRANSPARENT=TRUE. Also isBaseLayer will not changed by the │ │ │ │ │ - * constructor. Default false. │ │ │ │ │ - */ │ │ │ │ │ - noMagic: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: yx │ │ │ │ │ - * {Object} Keys in this object are EPSG codes for which the axis order │ │ │ │ │ - * is to be reversed (yx instead of xy, LatLon instead of LonLat), with │ │ │ │ │ - * true as value. This is only relevant for WMS versions >= 1.3.0, and │ │ │ │ │ - * only if yx is not set in <OpenLayers.Projection.defaults> for the │ │ │ │ │ - * used projection. │ │ │ │ │ + * Method: mousedown │ │ │ │ │ + * Handle mouse down. Stop propagation if a feature is targeted by this │ │ │ │ │ + * event (stops map dragging during feature selection). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - yx: {}, │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + // Feature selection is only done with a left click. Other handlers may stop the │ │ │ │ │ + // propagation of left-click mousedown events but not right-click mousedown events. │ │ │ │ │ + // This mismatch causes problems when comparing the location of the down and up │ │ │ │ │ + // events in the click function so it is important ignore right-clicks. │ │ │ │ │ + if (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ + this.down = evt.xy; │ │ │ │ │ + } │ │ │ │ │ + return this.handle(evt) ? !this.stopDown : true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.WMS │ │ │ │ │ - * Create a new WMS layer object │ │ │ │ │ - * │ │ │ │ │ - * Examples: │ │ │ │ │ - * │ │ │ │ │ - * The code below creates a simple WMS layer using the image/jpeg format. │ │ │ │ │ - * (code) │ │ │ │ │ - * var wms = new OpenLayers.Layer.WMS("NASA Global Mosaic", │ │ │ │ │ - * "http://wms.jpl.nasa.gov/wms.cgi", │ │ │ │ │ - * {layers: "modis,global_mosaic"}); │ │ │ │ │ - * (end) │ │ │ │ │ - * Note the 3rd argument (params). Properties added to this object will be │ │ │ │ │ - * added to the WMS GetMap requests used for this layer's tiles. The only │ │ │ │ │ - * mandatory parameter is "layers". Other common WMS params include │ │ │ │ │ - * "transparent", "styles" and "format". Note that the "srs" param will │ │ │ │ │ - * always be ignored. Instead, it will be derived from the baseLayer's or │ │ │ │ │ - * map's projection. │ │ │ │ │ - * │ │ │ │ │ - * The code below creates a transparent WMS layer with additional options. │ │ │ │ │ - * (code) │ │ │ │ │ - * var wms = new OpenLayers.Layer.WMS("NASA Global Mosaic", │ │ │ │ │ - * "http://wms.jpl.nasa.gov/wms.cgi", │ │ │ │ │ - * { │ │ │ │ │ - * layers: "modis,global_mosaic", │ │ │ │ │ - * transparent: true │ │ │ │ │ - * }, { │ │ │ │ │ - * opacity: 0.5, │ │ │ │ │ - * singleTile: true │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * Note that by default, a WMS layer is configured as baseLayer. Setting │ │ │ │ │ - * the "transparent" param to true will apply some magic (see <noMagic>). │ │ │ │ │ - * The default image format changes from image/jpeg to image/png, and the │ │ │ │ │ - * layer is not configured as baseLayer. │ │ │ │ │ - * │ │ │ │ │ + * Method: mouseup │ │ │ │ │ + * Handle mouse up. Stop propagation if a feature is targeted by this │ │ │ │ │ + * event. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} A name for the layer │ │ │ │ │ - * url - {String} Base url for the WMS │ │ │ │ │ - * (e.g. http://wms.jpl.nasa.gov/wms.cgi) │ │ │ │ │ - * params - {Object} An object with key/value pairs representing the │ │ │ │ │ - * GetMap query string parameters and parameter values. │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer. │ │ │ │ │ - * These options include all properties listed above, plus the ones │ │ │ │ │ - * inherited from superclasses. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - //uppercase params │ │ │ │ │ - params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ - if (parseFloat(params.VERSION) >= 1.3 && !params.EXCEPTIONS) { │ │ │ │ │ - params.EXCEPTIONS = "INIMAGE"; │ │ │ │ │ - } │ │ │ │ │ - newArguments.push(name, url, params, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, │ │ │ │ │ - OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS) │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - //layer is transparent │ │ │ │ │ - if (!this.noMagic && this.params.TRANSPARENT && │ │ │ │ │ - this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ - │ │ │ │ │ - // unless explicitly set in options, make layer an overlay │ │ │ │ │ - if ((options == null) || (!options.isBaseLayer)) { │ │ │ │ │ - this.isBaseLayer = false; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // jpegs can never be transparent, so intelligently switch the │ │ │ │ │ - // format, depending on the browser's capabilities │ │ │ │ │ - if (this.params.FORMAT == "image/jpeg") { │ │ │ │ │ - this.params.FORMAT = OpenLayers.Util.alphaHack() ? "image/gif" : │ │ │ │ │ - "image/png"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + this.up = evt.xy; │ │ │ │ │ + return this.handle(evt) ? !this.stopUp : true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ + * Method: click │ │ │ │ │ + * Handle click. Call the "click" callback if click on a feature, │ │ │ │ │ + * or the "clickout" callback if click outside any feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.WMS>} An exact clone of this layer │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.WMS(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.params, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + return this.handle(evt) ? !this.stopClick : true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: reverseAxisOrder │ │ │ │ │ - * Returns true if the axis order is reversed for the WMS version and │ │ │ │ │ - * projection of the layer. │ │ │ │ │ + * Method: mousemove │ │ │ │ │ + * Handle mouse moves. Call the "over" callback if moving in to a feature, │ │ │ │ │ + * or the "out" callback if moving out of a feature. │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} true if the axis order is reversed, false otherwise. │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - reverseAxisOrder: function() { │ │ │ │ │ - var projCode = this.projection.getCode(); │ │ │ │ │ - return parseFloat(this.params.VERSION) >= 1.3 && │ │ │ │ │ - !!(this.yx[projCode] || (OpenLayers.Projection.defaults[projCode] && │ │ │ │ │ - OpenLayers.Projection.defaults[projCode].yx)); │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + if (!this.callbacks['over'] && !this.callbacks['out']) { │ │ │ │ │ + return true; │ │ │ │ │ + } │ │ │ │ │ + this.handle(evt); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * Return a GetMap query string for this layer │ │ │ │ │ + * Method: dblclick │ │ │ │ │ + * Handle dblclick. Call the "dblclick" callback if dblclick on a feature. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox for the │ │ │ │ │ - * request. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as │ │ │ │ │ - * parameters. │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - │ │ │ │ │ - var imageSize = this.getImageSize(); │ │ │ │ │ - var newParams = {}; │ │ │ │ │ - // WMS 1.3 introduced axis order │ │ │ │ │ - var reverseAxisOrder = this.reverseAxisOrder(); │ │ │ │ │ - newParams.BBOX = this.encodeBBOX ? │ │ │ │ │ - bounds.toBBOX(null, reverseAxisOrder) : │ │ │ │ │ - bounds.toArray(reverseAxisOrder); │ │ │ │ │ - newParams.WIDTH = imageSize.w; │ │ │ │ │ - newParams.HEIGHT = imageSize.h; │ │ │ │ │ - var requestString = this.getFullRequestString(newParams); │ │ │ │ │ - return requestString; │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + return !this.handle(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: mergeNewParams │ │ │ │ │ - * Catch changeParams and uppercase the new params to be merged in │ │ │ │ │ - * before calling changeParams on the super class. │ │ │ │ │ - * │ │ │ │ │ - * Once params have been changed, the tiles will be reloaded with │ │ │ │ │ - * the new parameters. │ │ │ │ │ - * │ │ │ │ │ + * Method: geometryTypeMatches │ │ │ │ │ + * Return true if the geometry type of the passed feature matches │ │ │ │ │ + * one of the geometry types in the geometryTypes array. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * newParams - {Object} Hashtable of new params to use │ │ │ │ │ + * feature - {<OpenLayers.Vector.Feature>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ - var newArguments = [upperParams]; │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, │ │ │ │ │ - newArguments); │ │ │ │ │ + geometryTypeMatches: function(feature) { │ │ │ │ │ + return this.geometryTypes == null || │ │ │ │ │ + OpenLayers.Util.indexOf(this.geometryTypes, │ │ │ │ │ + feature.geometry.CLASS_NAME) > -1; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getFullRequestString │ │ │ │ │ - * Combine the layer's url with its params and these newParams. │ │ │ │ │ - * │ │ │ │ │ - * Add the SRS parameter from projection -- this is probably │ │ │ │ │ - * more eloquently done via a setProjection() method, but this │ │ │ │ │ - * works for now and always. │ │ │ │ │ + /** │ │ │ │ │ + * Method: handle │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * newParams - {Object} │ │ │ │ │ - * altUrl - {String} Use this as the url instead of the layer's url │ │ │ │ │ - * │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} │ │ │ │ │ + * {Boolean} The event occurred over a relevant feature. │ │ │ │ │ */ │ │ │ │ │ - getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ - var mapProjection = this.map.getProjectionObject(); │ │ │ │ │ - var projectionCode = this.projection && this.projection.equals(mapProjection) ? │ │ │ │ │ - this.projection.getCode() : │ │ │ │ │ - mapProjection.getCode(); │ │ │ │ │ - var value = (projectionCode == "none") ? null : projectionCode; │ │ │ │ │ - if (parseFloat(this.params.VERSION) >= 1.3) { │ │ │ │ │ - this.params.CRS = value; │ │ │ │ │ - } else { │ │ │ │ │ - this.params.SRS = value; │ │ │ │ │ + handle: function(evt) { │ │ │ │ │ + if (this.feature && !this.feature.layer) { │ │ │ │ │ + // feature has been destroyed │ │ │ │ │ + this.feature = null; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - if (typeof this.params.TRANSPARENT == "boolean") { │ │ │ │ │ - newParams.TRANSPARENT = this.params.TRANSPARENT ? "TRUE" : "FALSE"; │ │ │ │ │ + var type = evt.type; │ │ │ │ │ + var handled = false; │ │ │ │ │ + var previouslyIn = !!(this.feature); // previously in a feature │ │ │ │ │ + var click = (type == "click" || type == "dblclick" || type == "touchstart"); │ │ │ │ │ + this.feature = this.layer.getFeatureFromEvent(evt); │ │ │ │ │ + if (this.feature && !this.feature.layer) { │ │ │ │ │ + // feature has been destroyed │ │ │ │ │ + this.feature = null; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.getFullRequestString.apply( │ │ │ │ │ - this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.WMS" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/KaMapCache.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ - * @requires OpenLayers/Layer/KaMap.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.KaMapCache │ │ │ │ │ - * │ │ │ │ │ - * This class is designed to talk directly to a web-accessible ka-Map │ │ │ │ │ - * cache generated by the precache2.php script. │ │ │ │ │ - * │ │ │ │ │ - * To create a a new KaMapCache layer, you must indicate also the "i" parameter │ │ │ │ │ - * (that will be used to calculate the file extension), and another special │ │ │ │ │ - * parameter, object names "metaTileSize", with "h" (height) and "w" (width) │ │ │ │ │ - * properties. │ │ │ │ │ - * │ │ │ │ │ - * // Create a new kaMapCache layer. │ │ │ │ │ - * var kamap_base = new OpenLayers.Layer.KaMapCache( │ │ │ │ │ - * "Satellite", │ │ │ │ │ - * "http://www.example.org/web/acessible/cache", │ │ │ │ │ - * {g: "satellite", map: "world", i: 'png24', metaTileSize: {w: 5, h: 5} } │ │ │ │ │ - * ); │ │ │ │ │ - * │ │ │ │ │ - * // Create an kaMapCache overlay layer (using "isBaseLayer: false"). │ │ │ │ │ - * // Forces the output to be a "gif", using the "i" parameter. │ │ │ │ │ - * var kamap_overlay = new OpenLayers.Layer.KaMapCache( │ │ │ │ │ - * "Streets", │ │ │ │ │ - * "http://www.example.org/web/acessible/cache", │ │ │ │ │ - * {g: "streets", map: "world", i: "gif", metaTileSize: {w: 5, h: 5} }, │ │ │ │ │ - * {isBaseLayer: false} │ │ │ │ │ - * ); │ │ │ │ │ - * │ │ │ │ │ - * The cache URLs must look like: │ │ │ │ │ - * var/cache/World/50000/Group_Name/def/t-440320/l20480 │ │ │ │ │ - * │ │ │ │ │ - * This means that the cache generated via tile.php will *not* work with │ │ │ │ │ - * this class, and should instead use the KaMap layer. │ │ │ │ │ - * │ │ │ │ │ - * More information is available in Ticket #1518. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.KaMap> │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.KaMapCache = OpenLayers.Class(OpenLayers.Layer.KaMap, { │ │ │ │ │ + if (this.lastFeature && !this.lastFeature.layer) { │ │ │ │ │ + // last feature has been destroyed │ │ │ │ │ + this.lastFeature = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.feature) { │ │ │ │ │ + if (type === "touchstart") { │ │ │ │ │ + // stop the event to prevent Android Webkit from │ │ │ │ │ + // "flashing" the map div │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + } │ │ │ │ │ + var inNew = (this.feature != this.lastFeature); │ │ │ │ │ + if (this.geometryTypeMatches(this.feature)) { │ │ │ │ │ + // in to a feature │ │ │ │ │ + if (previouslyIn && inNew) { │ │ │ │ │ + // out of last feature and in to another │ │ │ │ │ + if (this.lastFeature) { │ │ │ │ │ + this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ + } │ │ │ │ │ + this.triggerCallback(type, 'in', [this.feature]); │ │ │ │ │ + } else if (!previouslyIn || click) { │ │ │ │ │ + // in feature for the first time │ │ │ │ │ + this.triggerCallback(type, 'in', [this.feature]); │ │ │ │ │ + } │ │ │ │ │ + this.lastFeature = this.feature; │ │ │ │ │ + handled = true; │ │ │ │ │ + } else { │ │ │ │ │ + // not in to a feature │ │ │ │ │ + if (this.lastFeature && (previouslyIn && inNew || click)) { │ │ │ │ │ + // out of last feature for the first time │ │ │ │ │ + this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ + } │ │ │ │ │ + // next time the mouse goes in a feature whose geometry type │ │ │ │ │ + // doesn't match we don't want to call the 'out' callback │ │ │ │ │ + // again, so let's set this.feature to null so that │ │ │ │ │ + // previouslyIn will evaluate to false the next time │ │ │ │ │ + // we enter handle. Yes, a bit hackish... │ │ │ │ │ + this.feature = null; │ │ │ │ │ + } │ │ │ │ │ + } else if (this.lastFeature && (previouslyIn || click)) { │ │ │ │ │ + this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ + } │ │ │ │ │ + return handled; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: IMAGE_EXTENSIONS │ │ │ │ │ - * {Object} Simple hash map to convert format to extension. │ │ │ │ │ + * Method: triggerCallback │ │ │ │ │ + * Call the callback keyed in the event map with the supplied arguments. │ │ │ │ │ + * For click and clickout, the <clickTolerance> is checked first. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * type - {String} │ │ │ │ │ */ │ │ │ │ │ - IMAGE_EXTENSIONS: { │ │ │ │ │ - 'jpeg': 'jpg', │ │ │ │ │ - 'gif': 'gif', │ │ │ │ │ - 'png': 'png', │ │ │ │ │ - 'png8': 'png', │ │ │ │ │ - 'png24': 'png', │ │ │ │ │ - 'dithered': 'png' │ │ │ │ │ + triggerCallback: function(type, mode, args) { │ │ │ │ │ + var key = this.EVENTMAP[type][mode]; │ │ │ │ │ + if (key) { │ │ │ │ │ + if (type == 'click' && this.up && this.down) { │ │ │ │ │ + // for click/clickout, only trigger callback if tolerance is met │ │ │ │ │ + var dpx = Math.sqrt( │ │ │ │ │ + Math.pow(this.up.x - this.down.x, 2) + │ │ │ │ │ + Math.pow(this.up.y - this.down.y, 2) │ │ │ │ │ + ); │ │ │ │ │ + if (dpx <= this.clickTolerance) { │ │ │ │ │ + this.callback(key, args); │ │ │ │ │ + } │ │ │ │ │ + // we're done with this set of events now: clear the cached │ │ │ │ │ + // positions so we can't trip over them later (this can occur │ │ │ │ │ + // if one of the up/down events gets eaten before it gets to us │ │ │ │ │ + // but we still get the click) │ │ │ │ │ + this.up = this.down = null; │ │ │ │ │ + } else { │ │ │ │ │ + this.callback(key, args); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: DEFAULT_FORMAT │ │ │ │ │ - * {Object} Simple hash map to convert format to extension. │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Turn on the handler. Returns false if the handler was already active. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - DEFAULT_FORMAT: 'jpeg', │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerToTop(); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + "removelayer": this.handleMapEvents, │ │ │ │ │ + "changelayer": this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + activated = true; │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.KaMapCache │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * url - {String} │ │ │ │ │ - * params - {Object} Parameters to be sent to the HTTP server in the │ │ │ │ │ - * query string for the tile. The format can be set via the 'i' │ │ │ │ │ - * parameter (defaults to jpg) , and the map should be set via │ │ │ │ │ - * the 'map' parameter. It has been reported that ka-Map may behave │ │ │ │ │ - * inconsistently if your format parameter does not match the format │ │ │ │ │ - * parameter configured in your config.php. (See ticket #327 for more │ │ │ │ │ - * information.) │ │ │ │ │ - * options - {Object} Additional options for the layer. Any of the │ │ │ │ │ - * APIProperties listed on this layer, and any layer types it │ │ │ │ │ - * extends, can be overridden through the options parameter. │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Turn off the handler. Returns false if the handler was already active. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.KaMap.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.extension = this.IMAGE_EXTENSIONS[this.params.i.toLowerCase() || this.DEFAULT_FORMAT]; │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerBack(); │ │ │ │ │ + this.feature = null; │ │ │ │ │ + this.lastFeature = null; │ │ │ │ │ + this.down = null; │ │ │ │ │ + this.up = null; │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + "removelayer": this.handleMapEvents, │ │ │ │ │ + "changelayer": this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + deactivated = true; │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ + * Method: handleMapEvents │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as │ │ │ │ │ - * parameters │ │ │ │ │ + * evt - {Object} │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var mapRes = this.map.getResolution(); │ │ │ │ │ - var scale = Math.round((this.map.getScale() * 10000)) / 10000; │ │ │ │ │ - var pX = Math.round(bounds.left / mapRes); │ │ │ │ │ - var pY = -Math.round(bounds.top / mapRes); │ │ │ │ │ - │ │ │ │ │ - var metaX = Math.floor(pX / this.tileSize.w / this.params.metaTileSize.w) * this.tileSize.w * this.params.metaTileSize.w; │ │ │ │ │ - var metaY = Math.floor(pY / this.tileSize.h / this.params.metaTileSize.h) * this.tileSize.h * this.params.metaTileSize.h; │ │ │ │ │ + handleMapEvents: function(evt) { │ │ │ │ │ + if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ + this.moveLayerToTop(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var components = [ │ │ │ │ │ - "/", │ │ │ │ │ - this.params.map, │ │ │ │ │ - "/", │ │ │ │ │ - scale, │ │ │ │ │ - "/", │ │ │ │ │ - this.params.g.replace(/\s/g, '_'), │ │ │ │ │ - "/def/t", │ │ │ │ │ - metaY, │ │ │ │ │ - "/l", │ │ │ │ │ - metaX, │ │ │ │ │ - "/t", │ │ │ │ │ - pY, │ │ │ │ │ - "l", │ │ │ │ │ - pX, │ │ │ │ │ - ".", │ │ │ │ │ - this.extension │ │ │ │ │ - ]; │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveLayerToTop │ │ │ │ │ + * Moves the layer for this handler to the top, so mouse events can reach │ │ │ │ │ + * it. │ │ │ │ │ + */ │ │ │ │ │ + moveLayerToTop: function() { │ │ │ │ │ + var index = Math.max(this.map.Z_INDEX_BASE['Feature'] - 1, │ │ │ │ │ + this.layer.getZIndex()) + 1; │ │ │ │ │ + this.layer.setZIndex(index); │ │ │ │ │ │ │ │ │ │ - var url = this.url; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(components.join(''), url); │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveLayerBack │ │ │ │ │ + * Moves the layer back to the position determined by the map's layers │ │ │ │ │ + * array. │ │ │ │ │ + */ │ │ │ │ │ + moveLayerBack: function() { │ │ │ │ │ + var index = this.layer.getZIndex() - 1; │ │ │ │ │ + if (index >= this.map.Z_INDEX_BASE['Feature']) { │ │ │ │ │ + this.layer.setZIndex(index); │ │ │ │ │ + } else { │ │ │ │ │ + this.map.setLayerZIndex(this.layer, │ │ │ │ │ + this.map.getLayerIndex(this.layer)); │ │ │ │ │ } │ │ │ │ │ - return url + components.join(""); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.KaMapCache" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Feature" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/WMTS.js │ │ │ │ │ + OpenLayers/Handler/Keyboard.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.WMTS │ │ │ │ │ - * Instances of the WMTS class allow viewing of tiles from a service that │ │ │ │ │ - * implements the OGC WMTS specification version 1.0.0. │ │ │ │ │ + * Class: OpenLayers.handler.Keyboard │ │ │ │ │ + * A handler for keyboard events. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Handler.Keyboard> constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.WMTS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ +OpenLayers.Handler.Keyboard = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} The layer will be considered a base layer. Default is true. │ │ │ │ │ + /* http://www.quirksmode.org/js/keys.html explains key x-browser │ │ │ │ │ + key handling quirks in pretty nice detail */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: KEY_EVENTS │ │ │ │ │ + * keydown, keypress, keyup │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + KEY_EVENTS: ["keydown", "keyup"], │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} WMTS version. Default is "1.0.0". │ │ │ │ │ + /** │ │ │ │ │ + * Property: eventListener │ │ │ │ │ + * {Function} │ │ │ │ │ */ │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ + eventListener: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: requestEncoding │ │ │ │ │ - * {String} Request encoding. Can be "REST" or "KVP". Default is "KVP". │ │ │ │ │ + * Property: observeElement │ │ │ │ │ + * {DOMElement|String} The DOM element on which we listen for │ │ │ │ │ + * key events. Default to the document. │ │ │ │ │ */ │ │ │ │ │ - requestEncoding: "KVP", │ │ │ │ │ + observeElement: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String|Array(String)} The base URL or request URL template for the WMTS │ │ │ │ │ - * service. Must be provided. Array is only supported for base URLs, not │ │ │ │ │ - * for request URL templates. URL templates are only supported for │ │ │ │ │ - * REST <requestEncoding>. │ │ │ │ │ + * Constructor: OpenLayers.Handler.Keyboard │ │ │ │ │ + * Returns a new keyboard handler. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ + * this handler. If a handler is being used without a control, the │ │ │ │ │ + * handlers setMap method must be overridden to deal properly with │ │ │ │ │ + * the map. │ │ │ │ │ + * callbacks - {Object} An object containing a single function to be │ │ │ │ │ + * called when the drag operation is finished. The callback should │ │ │ │ │ + * expect to recieve a single argument, the pixel location of the event. │ │ │ │ │ + * Callbacks for 'keydown', 'keypress', and 'keyup' are supported. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * handler. │ │ │ │ │ */ │ │ │ │ │ - url: null, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + // cache the bound event listener method so it can be unobserved later │ │ │ │ │ + this.eventListener = OpenLayers.Function.bindAsEventListener( │ │ │ │ │ + this.handleKeyEvent, this │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: layer │ │ │ │ │ - * {String} The layer identifier advertised by the WMTS service. Must be │ │ │ │ │ - * provided. │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ - layer: null, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.eventListener = null; │ │ │ │ │ + OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: matrixSet │ │ │ │ │ - * {String} One of the advertised matrix set identifiers. Must be provided. │ │ │ │ │ + /** │ │ │ │ │ + * Method: activate │ │ │ │ │ */ │ │ │ │ │ - matrixSet: null, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.observeElement = this.observeElement || document; │ │ │ │ │ + for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ + OpenLayers.Event.observe( │ │ │ │ │ + this.observeElement, this.KEY_EVENTS[i], this.eventListener); │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: style │ │ │ │ │ - * {String} One of the advertised layer styles. Must be provided. │ │ │ │ │ + /** │ │ │ │ │ + * Method: deactivate │ │ │ │ │ */ │ │ │ │ │ - style: null, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ + OpenLayers.Event.stopObserving( │ │ │ │ │ + this.observeElement, this.KEY_EVENTS[i], this.eventListener); │ │ │ │ │ + } │ │ │ │ │ + deactivated = true; │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: format │ │ │ │ │ - * {String} The image MIME type. Default is "image/jpeg". │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleKeyEvent │ │ │ │ │ */ │ │ │ │ │ - format: "image/jpeg", │ │ │ │ │ + handleKeyEvent: function(evt) { │ │ │ │ │ + if (this.checkModifiers(evt)) { │ │ │ │ │ + this.callback(evt.type, [evt]); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Keyboard" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Renderer/Canvas.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Renderer.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Renderer.Canvas │ │ │ │ │ + * A renderer based on the 2D 'canvas' drawing element. │ │ │ │ │ + * │ │ │ │ │ + * Inherits: │ │ │ │ │ + * - <OpenLayers.Renderer> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.Canvas = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: tileOrigin │ │ │ │ │ - * {<OpenLayers.LonLat>} The top-left corner of the tile matrix in map │ │ │ │ │ - * units. If the tile origin for each matrix in a set is different, │ │ │ │ │ - * the <matrixIds> should include a topLeftCorner property. If │ │ │ │ │ - * not provided, the tile origin will default to the top left corner │ │ │ │ │ - * of the layer <maxExtent>. │ │ │ │ │ + * APIProperty: hitDetection │ │ │ │ │ + * {Boolean} Allow for hit detection of features. Default is true. │ │ │ │ │ */ │ │ │ │ │ - tileOrigin: null, │ │ │ │ │ + hitDetection: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: tileFullExtent │ │ │ │ │ - * {<OpenLayers.Bounds>} The full extent of the tile set. If not supplied, │ │ │ │ │ - * the layer's <maxExtent> property will be used. │ │ │ │ │ + * Property: hitOverflow │ │ │ │ │ + * {Number} The method for converting feature identifiers to color values │ │ │ │ │ + * supports 16777215 sequential values. Two features cannot be │ │ │ │ │ + * predictably detected if their identifiers differ by more than this │ │ │ │ │ + * value. The hitOverflow allows for bigger numbers (but the │ │ │ │ │ + * difference in values is still limited). │ │ │ │ │ */ │ │ │ │ │ - tileFullExtent: null, │ │ │ │ │ + hitOverflow: 0, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: formatSuffix │ │ │ │ │ - * {String} For REST request encoding, an image format suffix must be │ │ │ │ │ - * included in the request. If not provided, the suffix will be derived │ │ │ │ │ - * from the <format> property. │ │ │ │ │ + * Property: canvas │ │ │ │ │ + * {Canvas} The canvas context object. │ │ │ │ │ */ │ │ │ │ │ - formatSuffix: null, │ │ │ │ │ + canvas: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: matrixIds │ │ │ │ │ - * {Array} A list of tile matrix identifiers. If not provided, the matrix │ │ │ │ │ - * identifiers will be assumed to be integers corresponding to the │ │ │ │ │ - * map zoom level. If a list of strings is provided, each item should │ │ │ │ │ - * be the matrix identifier that corresponds to the map zoom level. │ │ │ │ │ - * Additionally, a list of objects can be provided. Each object should │ │ │ │ │ - * describe the matrix as presented in the WMTS capabilities. These │ │ │ │ │ - * objects should have the propertes shown below. │ │ │ │ │ - * │ │ │ │ │ - * Matrix properties: │ │ │ │ │ - * identifier - {String} The matrix identifier (required). │ │ │ │ │ - * scaleDenominator - {Number} The matrix scale denominator. │ │ │ │ │ - * topLeftCorner - {<OpenLayers.LonLat>} The top left corner of the │ │ │ │ │ - * matrix. Must be provided if different than the layer <tileOrigin>. │ │ │ │ │ - * tileWidth - {Number} The tile width for the matrix. Must be provided │ │ │ │ │ - * if different than the width given in the layer <tileSize>. │ │ │ │ │ - * tileHeight - {Number} The tile height for the matrix. Must be provided │ │ │ │ │ - * if different than the height given in the layer <tileSize>. │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Object} Internal object of feature/style pairs for use in redrawing the layer. │ │ │ │ │ */ │ │ │ │ │ - matrixIds: null, │ │ │ │ │ + features: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: dimensions │ │ │ │ │ - * {Array} For RESTful request encoding, extra dimensions may be specified. │ │ │ │ │ - * Items in this list should be property names in the <params> object. │ │ │ │ │ - * Values of extra dimensions will be determined from the corresponding │ │ │ │ │ - * values in the <params> object. │ │ │ │ │ + * Property: pendingRedraw │ │ │ │ │ + * {Boolean} The renderer needs a redraw call to render features added while │ │ │ │ │ + * the renderer was locked. │ │ │ │ │ */ │ │ │ │ │ - dimensions: null, │ │ │ │ │ + pendingRedraw: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: params │ │ │ │ │ - * {Object} Extra parameters to include in tile requests. For KVP │ │ │ │ │ - * <requestEncoding>, these properties will be encoded in the request │ │ │ │ │ - * query string. For REST <requestEncoding>, these properties will │ │ │ │ │ - * become part of the request path, with order determined by the │ │ │ │ │ - * <dimensions> list. │ │ │ │ │ + * Property: cachedSymbolBounds │ │ │ │ │ + * {Object} Internal cache of calculated symbol extents. │ │ │ │ │ */ │ │ │ │ │ - params: null, │ │ │ │ │ + cachedSymbolBounds: {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomOffset │ │ │ │ │ - * {Number} If your cache has more levels than you want to provide │ │ │ │ │ - * access to with this layer, supply a zoomOffset. This zoom offset │ │ │ │ │ - * is added to the current map zoom level to determine the level │ │ │ │ │ - * for a requested tile. For example, if you supply a zoomOffset │ │ │ │ │ - * of 3, when the map is at the zoom 0, tiles will be requested from │ │ │ │ │ - * level 3 of your cache. Default is 0 (assumes cache level and map │ │ │ │ │ - * zoom are equivalent). Additionally, if this layer is to be used │ │ │ │ │ - * as an overlay and the cache has fewer zoom levels than the base │ │ │ │ │ - * layer, you can supply a negative zoomOffset. For example, if a │ │ │ │ │ - * map zoom level of 1 corresponds to your cache level zero, you would │ │ │ │ │ - * supply a -1 zoomOffset (and set the maxResolution of the layer │ │ │ │ │ - * appropriately). The zoomOffset value has no effect if complete │ │ │ │ │ - * matrix definitions (including scaleDenominator) are supplied in │ │ │ │ │ - * the <matrixIds> property. Defaults to 0 (no zoom offset). │ │ │ │ │ + * Constructor: OpenLayers.Renderer.Canvas │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * containerID - {<String>} │ │ │ │ │ + * options - {Object} Optional properties to be set on the renderer. │ │ │ │ │ */ │ │ │ │ │ - zoomOffset: 0, │ │ │ │ │ + initialize: function(containerID, options) { │ │ │ │ │ + OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.root = document.createElement("canvas"); │ │ │ │ │ + this.container.appendChild(this.root); │ │ │ │ │ + this.canvas = this.root.getContext("2d"); │ │ │ │ │ + this.features = {}; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitCanvas = document.createElement("canvas"); │ │ │ │ │ + this.hitContext = this.hitCanvas.getContext("2d"); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: serverResolutions │ │ │ │ │ - * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ - * property if the map resolutions differ from the server. This │ │ │ │ │ - * property serves two purposes. (a) <serverResolutions> can include │ │ │ │ │ - * resolutions that the server supports and that you don't want to │ │ │ │ │ - * provide with this layer; you can also look at <zoomOffset>, which is │ │ │ │ │ - * an alternative to <serverResolutions> for that specific purpose. │ │ │ │ │ - * (b) The map can work with resolutions that aren't supported by │ │ │ │ │ - * the server, i.e. that aren't in <serverResolutions>. When the │ │ │ │ │ - * map is displayed in such a resolution data for the closest │ │ │ │ │ - * server-supported resolution is loaded and the layer div is │ │ │ │ │ - * stretched as necessary. │ │ │ │ │ + * Method: setExtent │ │ │ │ │ + * Set the visible part of the layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} │ │ │ │ │ + * resolutionChanged - {Boolean} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ + * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ + * False otherwise. │ │ │ │ │ */ │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ + setExtent: function() { │ │ │ │ │ + OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + // always redraw features │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: formatSuffixMap │ │ │ │ │ - * {Object} a map between WMTS 'format' request parameter and tile image file suffix │ │ │ │ │ + /** │ │ │ │ │ + * Method: eraseGeometry │ │ │ │ │ + * Erase a geometry from the renderer. Because the Canvas renderer has │ │ │ │ │ + * 'memory' of the features that it has drawn, we have to remove the │ │ │ │ │ + * feature so it doesn't redraw. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ */ │ │ │ │ │ - formatSuffixMap: { │ │ │ │ │ - "image/png": "png", │ │ │ │ │ - "image/png8": "png", │ │ │ │ │ - "image/png24": "png", │ │ │ │ │ - "image/png32": "png", │ │ │ │ │ - "png": "png", │ │ │ │ │ - "image/jpeg": "jpg", │ │ │ │ │ - "image/jpg": "jpg", │ │ │ │ │ - "jpeg": "jpg", │ │ │ │ │ - "jpg": "jpg" │ │ │ │ │ + eraseGeometry: function(geometry, featureId) { │ │ │ │ │ + this.eraseFeatures(this.features[featureId][0]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: matrix │ │ │ │ │ - * {Object} Matrix definition for the current map resolution. Updated by │ │ │ │ │ - * the <updateMatrixProperties> method. │ │ │ │ │ + * APIMethod: supported │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the browser supports the renderer class │ │ │ │ │ */ │ │ │ │ │ - matrix: null, │ │ │ │ │ + supported: function() { │ │ │ │ │ + return OpenLayers.CANVAS_SUPPORTED; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.WMTS │ │ │ │ │ - * Create a new WMTS layer. │ │ │ │ │ + * Method: setSize │ │ │ │ │ + * Sets the size of the drawing surface. │ │ │ │ │ * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var wmts = new OpenLayers.Layer.WMTS({ │ │ │ │ │ - * name: "My WMTS Layer", │ │ │ │ │ - * url: "http://example.com/wmts", │ │ │ │ │ - * layer: "layer_id", │ │ │ │ │ - * style: "default", │ │ │ │ │ - * matrixSet: "matrix_id" │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ + * Once the size is updated, redraw the canvas. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * config - {Object} Configuration properties for the layer. │ │ │ │ │ + * size - {<OpenLayers.Size>} │ │ │ │ │ + */ │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + this.size = size.clone(); │ │ │ │ │ + var root = this.root; │ │ │ │ │ + root.style.width = size.w + "px"; │ │ │ │ │ + root.style.height = size.h + "px"; │ │ │ │ │ + root.width = size.w; │ │ │ │ │ + root.height = size.h; │ │ │ │ │ + this.resolution = null; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + var hitCanvas = this.hitCanvas; │ │ │ │ │ + hitCanvas.style.width = size.w + "px"; │ │ │ │ │ + hitCanvas.style.height = size.h + "px"; │ │ │ │ │ + hitCanvas.width = size.w; │ │ │ │ │ + hitCanvas.height = size.h; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawFeature │ │ │ │ │ + * Draw the feature. Stores the feature in the features list, │ │ │ │ │ + * then redraws the layer. │ │ │ │ │ * │ │ │ │ │ - * Required configuration properties: │ │ │ │ │ - * url - {String} The base url for the service. See the <url> property. │ │ │ │ │ - * layer - {String} The layer identifier. See the <layer> property. │ │ │ │ │ - * style - {String} The layer style identifier. See the <style> property. │ │ │ │ │ - * matrixSet - {String} The tile matrix set identifier. See the <matrixSet> │ │ │ │ │ - * property. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * style - {<Object>} │ │ │ │ │ * │ │ │ │ │ - * Any other documented layer properties can be provided in the config object. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The feature has been drawn completely. If the feature has no │ │ │ │ │ + * geometry, undefined will be returned. If the feature is not rendered │ │ │ │ │ + * for other reasons, false will be returned. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ + drawFeature: function(feature, style) { │ │ │ │ │ + var rendered; │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + style = this.applyDefaultSymbolizer(style || feature.style); │ │ │ │ │ + // don't render if display none or feature outside extent │ │ │ │ │ + var bounds = feature.geometry.getBounds(); │ │ │ │ │ │ │ │ │ │ - // confirm required properties are supplied │ │ │ │ │ - var required = { │ │ │ │ │ - url: true, │ │ │ │ │ - layer: true, │ │ │ │ │ - style: true, │ │ │ │ │ - matrixSet: true │ │ │ │ │ - }; │ │ │ │ │ - for (var prop in required) { │ │ │ │ │ - if (!(prop in config)) { │ │ │ │ │ - throw new Error("Missing property '" + prop + "' in layer configuration."); │ │ │ │ │ + var worldBounds; │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + worldBounds = this.map.getMaxExtent(); │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - config.params = OpenLayers.Util.upperCaseObject(config.params); │ │ │ │ │ - var args = [config.name, config.url, config.params, config]; │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, args); │ │ │ │ │ │ │ │ │ │ + var intersects = bounds && bounds.intersectsBounds(this.extent, { │ │ │ │ │ + worldBounds: worldBounds │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - // determine format suffix (for REST) │ │ │ │ │ - if (!this.formatSuffix) { │ │ │ │ │ - this.formatSuffix = this.formatSuffixMap[this.format] || this.format.split("/").pop(); │ │ │ │ │ + rendered = (style.display !== "none") && !!bounds && intersects; │ │ │ │ │ + if (rendered) { │ │ │ │ │ + // keep track of what we have rendered for redraw │ │ │ │ │ + this.features[feature.id] = [feature, style]; │ │ │ │ │ + } else { │ │ │ │ │ + // remove from features tracked for redraw │ │ │ │ │ + delete(this.features[feature.id]); │ │ │ │ │ + } │ │ │ │ │ + this.pendingRedraw = true; │ │ │ │ │ + } │ │ │ │ │ + if (this.pendingRedraw && !this.locked) { │ │ │ │ │ + this.redraw(); │ │ │ │ │ + this.pendingRedraw = false; │ │ │ │ │ } │ │ │ │ │ + return rendered; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // expand matrixIds (may be array of string or array of object) │ │ │ │ │ - if (this.matrixIds) { │ │ │ │ │ - var len = this.matrixIds.length; │ │ │ │ │ - if (len && typeof this.matrixIds[0] === "string") { │ │ │ │ │ - var ids = this.matrixIds; │ │ │ │ │ - this.matrixIds = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - this.matrixIds[i] = { │ │ │ │ │ - identifier: ids[i] │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawGeometry │ │ │ │ │ + * Used when looping (in redraw) over the features; draws │ │ │ │ │ + * the canvas. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + */ │ │ │ │ │ + drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ + var className = geometry.CLASS_NAME; │ │ │ │ │ + if ((className == "OpenLayers.Geometry.Collection") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiPolygon")) { │ │ │ │ │ + for (var i = 0; i < geometry.components.length; i++) { │ │ │ │ │ + this.drawGeometry(geometry.components[i], style, featureId); │ │ │ │ │ } │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + this.drawPoint(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + this.drawLineString(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + this.drawPolygon(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ + * Method: drawExternalGraphic │ │ │ │ │ + * Called to draw External graphics. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ */ │ │ │ │ │ - setMap: function() { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + drawExternalGraphic: function(geometry, style, featureId) { │ │ │ │ │ + var img = new Image(); │ │ │ │ │ + │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + img.title = title; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ + var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ + style.graphicXOffset : -(0.5 * width); │ │ │ │ │ + var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ + style.graphicYOffset : -(0.5 * height); │ │ │ │ │ + │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ + │ │ │ │ │ + var onLoad = function() { │ │ │ │ │ + if (!this.features[featureId]) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ + if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ + var x = (p0 + xOffset) | 0; │ │ │ │ │ + var y = (p1 + yOffset) | 0; │ │ │ │ │ + var canvas = this.canvas; │ │ │ │ │ + canvas.globalAlpha = opacity; │ │ │ │ │ + var factor = OpenLayers.Renderer.Canvas.drawImageScaleFactor || │ │ │ │ │ + (OpenLayers.Renderer.Canvas.drawImageScaleFactor = │ │ │ │ │ + /android 2.1/.test(navigator.userAgent.toLowerCase()) ? │ │ │ │ │ + // 320 is the screen width of the G1 phone, for │ │ │ │ │ + // which drawImage works out of the box. │ │ │ │ │ + 320 / window.screen.width : 1 │ │ │ │ │ + ); │ │ │ │ │ + canvas.drawImage( │ │ │ │ │ + img, x * factor, y * factor, width * factor, height * factor │ │ │ │ │ + ); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId); │ │ │ │ │ + this.hitContext.fillRect(x, y, width, height); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + img.onload = OpenLayers.Function.bind(onLoad, this); │ │ │ │ │ + img.src = style.externalGraphic; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: updateMatrixProperties │ │ │ │ │ - * Called when map resolution changes to update matrix related properties. │ │ │ │ │ + * Method: drawNamedSymbol │ │ │ │ │ + * Called to draw Well Known Graphic Symbol Name. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ */ │ │ │ │ │ - updateMatrixProperties: function() { │ │ │ │ │ - this.matrix = this.getMatrix(); │ │ │ │ │ - if (this.matrix) { │ │ │ │ │ - if (this.matrix.topLeftCorner) { │ │ │ │ │ - this.tileOrigin = this.matrix.topLeftCorner; │ │ │ │ │ + drawNamedSymbol: function(geometry, style, featureId) { │ │ │ │ │ + var x, y, cx, cy, i, symbolBounds, scaling, angle; │ │ │ │ │ + var unscaledStrokeWidth; │ │ │ │ │ + var deg2rad = Math.PI / 180.0; │ │ │ │ │ + │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[style.graphicName]; │ │ │ │ │ + │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(style.graphicName + ' is not a valid symbol name'); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (!symbol.length || symbol.length < 2) return; │ │ │ │ │ + │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ + │ │ │ │ │ + if (isNaN(p0) || isNaN(p1)) return; │ │ │ │ │ + │ │ │ │ │ + // Use rounded line caps │ │ │ │ │ + this.canvas.lineCap = "round"; │ │ │ │ │ + this.canvas.lineJoin = "round"; │ │ │ │ │ + │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.lineCap = "round"; │ │ │ │ │ + this.hitContext.lineJoin = "round"; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Scale and rotate symbols, using precalculated bounds whenever possible. │ │ │ │ │ + if (style.graphicName in this.cachedSymbolBounds) { │ │ │ │ │ + symbolBounds = this.cachedSymbolBounds[style.graphicName]; │ │ │ │ │ + } else { │ │ │ │ │ + symbolBounds = new OpenLayers.Bounds(); │ │ │ │ │ + for (i = 0; i < symbol.length; i += 2) { │ │ │ │ │ + symbolBounds.extend(new OpenLayers.LonLat(symbol[i], symbol[i + 1])); │ │ │ │ │ } │ │ │ │ │ - if (this.matrix.tileWidth && this.matrix.tileHeight) { │ │ │ │ │ - this.tileSize = new OpenLayers.Size( │ │ │ │ │ - this.matrix.tileWidth, this.matrix.tileHeight │ │ │ │ │ - ); │ │ │ │ │ + this.cachedSymbolBounds[style.graphicName] = symbolBounds; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Push symbol scaling, translation and rotation onto the transformation stack in reverse order. │ │ │ │ │ + // Don't forget to apply all canvas transformations to the hitContext canvas as well(!) │ │ │ │ │ + this.canvas.save(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.save(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Step 3: place symbol at the desired location │ │ │ │ │ + this.canvas.translate(p0, p1); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.translate(p0, p1); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Step 2a. rotate the symbol if necessary │ │ │ │ │ + angle = deg2rad * style.rotation; // will be NaN when style.rotation is undefined. │ │ │ │ │ + if (!isNaN(angle)) { │ │ │ │ │ + this.canvas.rotate(angle); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.rotate(angle); │ │ │ │ │ } │ │ │ │ │ - if (!this.tileOrigin) { │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat( │ │ │ │ │ - this.maxExtent.left, this.maxExtent.top │ │ │ │ │ - ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // // Step 2: scale symbol such that pointRadius equals half the maximum symbol dimension. │ │ │ │ │ + scaling = 2.0 * style.pointRadius / Math.max(symbolBounds.getWidth(), symbolBounds.getHeight()); │ │ │ │ │ + this.canvas.scale(scaling, scaling); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.scale(scaling, scaling); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Step 1: center the symbol at the origin │ │ │ │ │ + cx = symbolBounds.getCenterLonLat().lon; │ │ │ │ │ + cy = symbolBounds.getCenterLonLat().lat; │ │ │ │ │ + this.canvas.translate(-cx, -cy); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.translate(-cx, -cy); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Don't forget to scale stroke widths, because they are affected by canvas scale transformations as well(!) │ │ │ │ │ + // Alternative: scale symbol coordinates manually, so stroke width scaling is not needed anymore. │ │ │ │ │ + unscaledStrokeWidth = style.strokeWidth; │ │ │ │ │ + style.strokeWidth = unscaledStrokeWidth / scaling; │ │ │ │ │ + │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.canvas.lineTo(x, y); │ │ │ │ │ } │ │ │ │ │ - if (!this.tileFullExtent) { │ │ │ │ │ - this.tileFullExtent = this.maxExtent; │ │ │ │ │ + this.canvas.closePath(); │ │ │ │ │ + this.canvas.fill(); │ │ │ │ │ + │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.hitContext.lineTo(x, y); │ │ │ │ │ + } │ │ │ │ │ + this.hitContext.closePath(); │ │ │ │ │ + this.hitContext.fill(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.canvas.lineTo(x, y); │ │ │ │ │ + } │ │ │ │ │ + this.canvas.closePath(); │ │ │ │ │ + this.canvas.stroke(); │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style, scaling); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.hitContext.moveTo(x, y); │ │ │ │ │ + this.hitContext.lineTo(x, y); │ │ │ │ │ + } │ │ │ │ │ + this.hitContext.closePath(); │ │ │ │ │ + this.hitContext.stroke(); │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + style.strokeWidth = unscaledStrokeWidth; │ │ │ │ │ + this.canvas.restore(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.restore(); │ │ │ │ │ + } │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * │ │ │ │ │ + * Method: setCanvasStyle │ │ │ │ │ + * Prepare the canvas for drawing by setting various global settings. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * zoomChanged - {Boolean} Tells when zoom has changed, as layers have to │ │ │ │ │ - * do some init work in that case. │ │ │ │ │ - * dragging - {Boolean} │ │ │ │ │ + * type - {String} one of 'stroke', 'fill', or 'reset' │ │ │ │ │ + * style - {Object} Symbolizer hash │ │ │ │ │ */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - if (zoomChanged || !this.matrix) { │ │ │ │ │ - this.updateMatrixProperties(); │ │ │ │ │ + setCanvasStyle: function(type, style) { │ │ │ │ │ + if (type === "fill") { │ │ │ │ │ + this.canvas.globalAlpha = style['fillOpacity']; │ │ │ │ │ + this.canvas.fillStyle = style['fillColor']; │ │ │ │ │ + } else if (type === "stroke") { │ │ │ │ │ + this.canvas.globalAlpha = style['strokeOpacity']; │ │ │ │ │ + this.canvas.strokeStyle = style['strokeColor']; │ │ │ │ │ + this.canvas.lineWidth = style['strokeWidth']; │ │ │ │ │ + } else { │ │ │ │ │ + this.canvas.globalAlpha = 0; │ │ │ │ │ + this.canvas.lineWidth = 1; │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.moveTo.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * │ │ │ │ │ + * Method: featureIdToHex │ │ │ │ │ + * Convert a feature ID string into an RGB hex string. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {Object} │ │ │ │ │ - * │ │ │ │ │ + * featureId - {String} Feature id │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.WMTS>} An exact clone of this <OpenLayers.Layer.WMTS> │ │ │ │ │ + * {String} RGB hex string. │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.WMTS(this.options); │ │ │ │ │ + featureIdToHex: function(featureId) { │ │ │ │ │ + var id = Number(featureId.split("_").pop()) + 1; // zero for no feature │ │ │ │ │ + if (id >= 16777216) { │ │ │ │ │ + this.hitOverflow = id - 16777215; │ │ │ │ │ + id = id % 16777216 + 1; │ │ │ │ │ } │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getIdentifier │ │ │ │ │ - * Get the current index in the matrixIds array. │ │ │ │ │ - */ │ │ │ │ │ - getIdentifier: function() { │ │ │ │ │ - return this.getServerZoom(); │ │ │ │ │ + var hex = "000000" + id.toString(16); │ │ │ │ │ + var len = hex.length; │ │ │ │ │ + hex = "#" + hex.substring(len - 6, len); │ │ │ │ │ + return hex; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getMatrix │ │ │ │ │ - * Get the appropriate matrix definition for the current map resolution. │ │ │ │ │ + * Method: setHitContextStyle │ │ │ │ │ + * Prepare the hit canvas for drawing by setting various global settings. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * type - {String} one of 'stroke', 'fill', or 'reset' │ │ │ │ │ + * featureId - {String} The feature id. │ │ │ │ │ + * symbolizer - {<OpenLayers.Symbolizer>} The symbolizer. │ │ │ │ │ */ │ │ │ │ │ - getMatrix: function() { │ │ │ │ │ - var matrix; │ │ │ │ │ - if (!this.matrixIds || this.matrixIds.length === 0) { │ │ │ │ │ - matrix = { │ │ │ │ │ - identifier: this.getIdentifier() │ │ │ │ │ - }; │ │ │ │ │ - } else { │ │ │ │ │ - // get appropriate matrix given the map scale if possible │ │ │ │ │ - if ("scaleDenominator" in this.matrixIds[0]) { │ │ │ │ │ - // scale denominator calculation based on WMTS spec │ │ │ │ │ - var denom = │ │ │ │ │ - OpenLayers.METERS_PER_INCH * │ │ │ │ │ - OpenLayers.INCHES_PER_UNIT[this.units] * │ │ │ │ │ - this.getServerResolution() / 0.28E-3; │ │ │ │ │ - var diff = Number.POSITIVE_INFINITY; │ │ │ │ │ - var delta; │ │ │ │ │ - for (var i = 0, ii = this.matrixIds.length; i < ii; ++i) { │ │ │ │ │ - delta = Math.abs(1 - (this.matrixIds[i].scaleDenominator / denom)); │ │ │ │ │ - if (delta < diff) { │ │ │ │ │ - diff = delta; │ │ │ │ │ - matrix = this.matrixIds[i]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + setHitContextStyle: function(type, featureId, symbolizer, strokeScaling) { │ │ │ │ │ + var hex = this.featureIdToHex(featureId); │ │ │ │ │ + if (type == "fill") { │ │ │ │ │ + this.hitContext.globalAlpha = 1.0; │ │ │ │ │ + this.hitContext.fillStyle = hex; │ │ │ │ │ + } else if (type == "stroke") { │ │ │ │ │ + this.hitContext.globalAlpha = 1.0; │ │ │ │ │ + this.hitContext.strokeStyle = hex; │ │ │ │ │ + // bump up stroke width to deal with antialiasing. If strokeScaling is defined, we're rendering a symbol │ │ │ │ │ + // on a transformed canvas, so the antialias width bump has to scale as well. │ │ │ │ │ + if (typeof strokeScaling === "undefined") { │ │ │ │ │ + this.hitContext.lineWidth = symbolizer.strokeWidth + 2; │ │ │ │ │ } else { │ │ │ │ │ - // fall back on zoom as index │ │ │ │ │ - matrix = this.matrixIds[this.getIdentifier()]; │ │ │ │ │ + if (!isNaN(strokeScaling)) { │ │ │ │ │ + this.hitContext.lineWidth = symbolizer.strokeWidth + 2.0 / strokeScaling; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + this.hitContext.globalAlpha = 0; │ │ │ │ │ + this.hitContext.lineWidth = 1; │ │ │ │ │ } │ │ │ │ │ - return matrix; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getTileInfo │ │ │ │ │ - * Get tile information for a given location at the current map resolution. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * loc - {<OpenLayers.LonLat} A location in map coordinates. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object with "col", "row", "i", and "j" properties. The col │ │ │ │ │ - * and row values are zero based tile indexes from the top left. The │ │ │ │ │ - * i and j values are the number of pixels to the left and top │ │ │ │ │ - * (respectively) of the given location within the target tile. │ │ │ │ │ - */ │ │ │ │ │ - getTileInfo: function(loc) { │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - │ │ │ │ │ - var fx = (loc.lon - this.tileOrigin.lon) / (res * this.tileSize.w); │ │ │ │ │ - var fy = (this.tileOrigin.lat - loc.lat) / (res * this.tileSize.h); │ │ │ │ │ - │ │ │ │ │ - var col = Math.floor(fx); │ │ │ │ │ - var row = Math.floor(fy); │ │ │ │ │ - │ │ │ │ │ - return { │ │ │ │ │ - col: col, │ │ │ │ │ - row: row, │ │ │ │ │ - i: Math.floor((fx - col) * this.tileSize.w), │ │ │ │ │ - j: Math.floor((fy - row) * this.tileSize.h) │ │ │ │ │ - }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * Method: drawPoint │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A URL for the tile corresponding to the given bounds. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var url = ""; │ │ │ │ │ - if (!this.tileFullExtent || this.tileFullExtent.intersectsBounds(bounds)) { │ │ │ │ │ - │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var info = this.getTileInfo(center); │ │ │ │ │ - var matrixId = this.matrix.identifier; │ │ │ │ │ - var dimensions = this.dimensions, │ │ │ │ │ - params; │ │ │ │ │ - │ │ │ │ │ - if (OpenLayers.Util.isArray(this.url)) { │ │ │ │ │ - url = this.selectUrl([ │ │ │ │ │ - this.version, this.style, this.matrixSet, │ │ │ │ │ - this.matrix.identifier, info.row, info.col │ │ │ │ │ - ].join(","), this.url); │ │ │ │ │ + drawPoint: function(geometry, style, featureId) { │ │ │ │ │ + if (style.graphic !== false) { │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + this.drawExternalGraphic(geometry, style, featureId); │ │ │ │ │ + } else if (style.graphicName && (style.graphicName != "circle")) { │ │ │ │ │ + this.drawNamedSymbol(geometry, style, featureId); │ │ │ │ │ } else { │ │ │ │ │ - url = this.url; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.requestEncoding.toUpperCase() === "REST") { │ │ │ │ │ - params = this.params; │ │ │ │ │ - if (url.indexOf("{") !== -1) { │ │ │ │ │ - var template = url.replace(/\{/g, "${"); │ │ │ │ │ - var context = { │ │ │ │ │ - // spec does not make clear if capital S or not │ │ │ │ │ - style: this.style, │ │ │ │ │ - Style: this.style, │ │ │ │ │ - TileMatrixSet: this.matrixSet, │ │ │ │ │ - TileMatrix: this.matrix.identifier, │ │ │ │ │ - TileRow: info.row, │ │ │ │ │ - TileCol: info.col │ │ │ │ │ - }; │ │ │ │ │ - if (dimensions) { │ │ │ │ │ - var dimension, i; │ │ │ │ │ - for (i = dimensions.length - 1; i >= 0; --i) { │ │ │ │ │ - dimension = dimensions[i]; │ │ │ │ │ - context[dimension] = params[dimension.toUpperCase()]; │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ + if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ + var twoPi = Math.PI * 2; │ │ │ │ │ + var radius = style.pointRadius; │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.canvas.fill(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.hitContext.fill(); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - url = OpenLayers.String.format(template, context); │ │ │ │ │ - } else { │ │ │ │ │ - // include 'version', 'layer' and 'style' in tile resource url │ │ │ │ │ - var path = this.version + "/" + this.layer + "/" + this.style + "/"; │ │ │ │ │ │ │ │ │ │ - // append optional dimension path elements │ │ │ │ │ - if (dimensions) { │ │ │ │ │ - for (var i = 0; i < dimensions.length; i++) { │ │ │ │ │ - if (params[dimensions[i]]) { │ │ │ │ │ - path = path + params[dimensions[i]] + "/"; │ │ │ │ │ - } │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.canvas.stroke(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.hitContext.stroke(); │ │ │ │ │ } │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // append other required path elements │ │ │ │ │ - path = path + this.matrixSet + "/" + this.matrix.identifier + │ │ │ │ │ - "/" + info.row + "/" + info.col + "." + this.formatSuffix; │ │ │ │ │ - │ │ │ │ │ - if (!url.match(/\/$/)) { │ │ │ │ │ - url = url + "/"; │ │ │ │ │ - } │ │ │ │ │ - url = url + path; │ │ │ │ │ } │ │ │ │ │ - } else if (this.requestEncoding.toUpperCase() === "KVP") { │ │ │ │ │ - │ │ │ │ │ - // assemble all required parameters │ │ │ │ │ - params = { │ │ │ │ │ - SERVICE: "WMTS", │ │ │ │ │ - REQUEST: "GetTile", │ │ │ │ │ - VERSION: this.version, │ │ │ │ │ - LAYER: this.layer, │ │ │ │ │ - STYLE: this.style, │ │ │ │ │ - TILEMATRIXSET: this.matrixSet, │ │ │ │ │ - TILEMATRIX: this.matrix.identifier, │ │ │ │ │ - TILEROW: info.row, │ │ │ │ │ - TILECOL: info.col, │ │ │ │ │ - FORMAT: this.format │ │ │ │ │ - }; │ │ │ │ │ - url = OpenLayers.Layer.Grid.prototype.getFullRequestString.apply(this, [params]); │ │ │ │ │ - │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return url; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: mergeNewParams │ │ │ │ │ - * Extend the existing layer <params> with new properties. Tiles will be │ │ │ │ │ - * reloaded with updated params in the request. │ │ │ │ │ + * Method: drawLineString │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newParams - {Object} Properties to extend to existing <params>. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ */ │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - if (this.requestEncoding.toUpperCase() === "KVP") { │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply( │ │ │ │ │ - this, [OpenLayers.Util.upperCaseObject(newParams)] │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + drawLineString: function(geometry, style, featureId) { │ │ │ │ │ + style = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + fill: false │ │ │ │ │ + }, style); │ │ │ │ │ + this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.WMTS" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/MapServer.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawLinearRing │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + drawLinearRing: function(geometry, style, featureId) { │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.renderPath(this.canvas, geometry, style, featureId, "fill"); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.renderPath(this.hitContext, geometry, style, featureId, "fill"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.renderPath(this.canvas, geometry, style, featureId, "stroke"); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ + this.renderPath(this.hitContext, geometry, style, featureId, "stroke"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.MapServer │ │ │ │ │ - * Instances of OpenLayers.Layer.MapServer are used to display │ │ │ │ │ - * data from a MapServer CGI instance. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.MapServer = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + /** │ │ │ │ │ + * Method: renderPath │ │ │ │ │ + * Render a path with stroke and optional fill. │ │ │ │ │ + */ │ │ │ │ │ + renderPath: function(context, geometry, style, featureId, type) { │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + context.beginPath(); │ │ │ │ │ + var start = this.getLocalXY(components[0]); │ │ │ │ │ + var x = start[0]; │ │ │ │ │ + var y = start[1]; │ │ │ │ │ + if (!isNaN(x) && !isNaN(y)) { │ │ │ │ │ + context.moveTo(start[0], start[1]); │ │ │ │ │ + for (var i = 1; i < len; ++i) { │ │ │ │ │ + var pt = this.getLocalXY(components[i]); │ │ │ │ │ + context.lineTo(pt[0], pt[1]); │ │ │ │ │ + } │ │ │ │ │ + if (type === "fill") { │ │ │ │ │ + context.fill(); │ │ │ │ │ + } else { │ │ │ │ │ + context.stroke(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: DEFAULT_PARAMS │ │ │ │ │ - * {Object} Hashtable of default parameter key/value pairs │ │ │ │ │ + * Method: drawPolygon │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ */ │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - mode: "map", │ │ │ │ │ - map_imagetype: "png" │ │ │ │ │ + drawPolygon: function(geometry, style, featureId) { │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + this.drawLinearRing(components[0], style, featureId); │ │ │ │ │ + // erase inner rings │ │ │ │ │ + for (var i = 1; i < len; ++i) { │ │ │ │ │ + /** │ │ │ │ │ + * Note that this is overly agressive. Here we punch holes through │ │ │ │ │ + * all previously rendered features on the same canvas. A better │ │ │ │ │ + * solution for polygons with interior rings would be to draw the │ │ │ │ │ + * polygon on a sketch canvas first. We could erase all holes │ │ │ │ │ + * there and then copy the drawing to the layer canvas. │ │ │ │ │ + * TODO: http://trac.osgeo.org/openlayers/ticket/3130 │ │ │ │ │ + */ │ │ │ │ │ + this.canvas.globalCompositeOperation = "destination-out"; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.globalCompositeOperation = "destination-out"; │ │ │ │ │ + } │ │ │ │ │ + this.drawLinearRing( │ │ │ │ │ + components[i], │ │ │ │ │ + OpenLayers.Util.applyDefaults({ │ │ │ │ │ + stroke: false, │ │ │ │ │ + fillOpacity: 1.0 │ │ │ │ │ + }, style), │ │ │ │ │ + featureId │ │ │ │ │ + ); │ │ │ │ │ + this.canvas.globalCompositeOperation = "source-over"; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.globalCompositeOperation = "source-over"; │ │ │ │ │ + } │ │ │ │ │ + this.drawLinearRing( │ │ │ │ │ + components[i], │ │ │ │ │ + OpenLayers.Util.applyDefaults({ │ │ │ │ │ + fill: false │ │ │ │ │ + }, style), │ │ │ │ │ + featureId │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.MapServer │ │ │ │ │ - * Create a new MapServer layer object │ │ │ │ │ + * Method: drawText │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} A name for the layer │ │ │ │ │ - * url - {String} Base url for the MapServer CGI │ │ │ │ │ - * (e.g. http://www2.dmsolutions.ca/cgi-bin/mapserv) │ │ │ │ │ - * params - {Object} An object with key/value pairs representing the │ │ │ │ │ - * GetMap query string parameters and parameter values. │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + * location - {<OpenLayers.Point>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - this.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, this.DEFAULT_PARAMS │ │ │ │ │ - ); │ │ │ │ │ + drawText: function(location, style) { │ │ │ │ │ + var pt = this.getLocalXY(location); │ │ │ │ │ │ │ │ │ │ - // unless explicitly set in options, if the layer is transparent, │ │ │ │ │ - // it will be an overlay │ │ │ │ │ - if (options == null || options.isBaseLayer == null) { │ │ │ │ │ - this.isBaseLayer = ((this.params.transparent != "true") && │ │ │ │ │ - (this.params.transparent != true)); │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ + this.canvas.fillStyle = style.fontColor; │ │ │ │ │ + this.canvas.globalAlpha = style.fontOpacity || 1.0; │ │ │ │ │ + var fontStyle = [style.fontStyle ? style.fontStyle : "normal", │ │ │ │ │ + "normal", // "font-variant" not supported │ │ │ │ │ + style.fontWeight ? style.fontWeight : "normal", │ │ │ │ │ + style.fontSize ? style.fontSize : "1em", │ │ │ │ │ + style.fontFamily ? style.fontFamily : "sans-serif" │ │ │ │ │ + ].join(" "); │ │ │ │ │ + var labelRows = style.label.split('\n'); │ │ │ │ │ + var numRows = labelRows.length; │ │ │ │ │ + if (this.canvas.fillText) { │ │ │ │ │ + // HTML5 │ │ │ │ │ + this.canvas.font = fontStyle; │ │ │ │ │ + this.canvas.textAlign = │ │ │ │ │ + OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[0]] || │ │ │ │ │ + "center"; │ │ │ │ │ + this.canvas.textBaseline = │ │ │ │ │ + OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[1]] || │ │ │ │ │ + "middle"; │ │ │ │ │ + var vfactor = │ │ │ │ │ + OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5; │ │ │ │ │ + } │ │ │ │ │ + var lineHeight = │ │ │ │ │ + this.canvas.measureText('Mg').height || │ │ │ │ │ + this.canvas.measureText('xx').width; │ │ │ │ │ + pt[1] += lineHeight * vfactor * (numRows - 1); │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + if (style.labelOutlineWidth) { │ │ │ │ │ + this.canvas.save(); │ │ │ │ │ + this.canvas.globalAlpha = style.labelOutlineOpacity || style.fontOpacity || 1.0; │ │ │ │ │ + this.canvas.strokeStyle = style.labelOutlineColor; │ │ │ │ │ + this.canvas.lineWidth = style.labelOutlineWidth; │ │ │ │ │ + this.canvas.strokeText(labelRows[i], pt[0], pt[1] + (lineHeight * i) + 1); │ │ │ │ │ + this.canvas.restore(); │ │ │ │ │ + } │ │ │ │ │ + this.canvas.fillText(labelRows[i], pt[0], pt[1] + (lineHeight * i)); │ │ │ │ │ + } │ │ │ │ │ + } else if (this.canvas.mozDrawText) { │ │ │ │ │ + // Mozilla pre-Gecko1.9.1 (<FF3.1) │ │ │ │ │ + this.canvas.mozTextStyle = fontStyle; │ │ │ │ │ + // No built-in text alignment, so we measure and adjust the position │ │ │ │ │ + var hfactor = │ │ │ │ │ + OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[0]]; │ │ │ │ │ + if (hfactor == null) { │ │ │ │ │ + hfactor = -.5; │ │ │ │ │ + } │ │ │ │ │ + var vfactor = │ │ │ │ │ + OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5; │ │ │ │ │ + } │ │ │ │ │ + var lineHeight = this.canvas.mozMeasureText('xx'); │ │ │ │ │ + pt[1] += lineHeight * (1 + (vfactor * numRows)); │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + var x = pt[0] + (hfactor * this.canvas.mozMeasureText(labelRows[i])); │ │ │ │ │ + var y = pt[1] + (i * lineHeight); │ │ │ │ │ + this.canvas.translate(x, y); │ │ │ │ │ + this.canvas.mozDrawText(labelRows[i]); │ │ │ │ │ + this.canvas.translate(-x, -y); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ + * Method: getLocalXY │ │ │ │ │ + * transform geographic xy into pixel xy │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.MapServer>} An exact clone of this layer │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.MapServer(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.params, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ + getLocalXY: function(point) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var extent = this.extent; │ │ │ │ │ + var x = ((point.x - this.featureDx) / resolution + (-extent.left / resolution)); │ │ │ │ │ + var y = ((extent.top / resolution) - point.y / resolution); │ │ │ │ │ + return [x, y]; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return obj; │ │ │ │ │ + /** │ │ │ │ │ + * Method: clear │ │ │ │ │ + * Clear all vectors from the renderer. │ │ │ │ │ + */ │ │ │ │ │ + clear: function() { │ │ │ │ │ + var height = this.root.height; │ │ │ │ │ + var width = this.root.width; │ │ │ │ │ + this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ + this.features = {}; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.clearRect(0, 0, width, height); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * Return a query string for this layer │ │ │ │ │ - * │ │ │ │ │ + * Method: getFeatureIdFromEvent │ │ │ │ │ + * Returns a feature id from an event on the renderer. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox │ │ │ │ │ - * for the request │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also │ │ │ │ │ - * the passed-in bounds and appropriate tile size specified │ │ │ │ │ - * as parameters. │ │ │ │ │ + * {<OpenLayers.Feature.Vector} A feature or undefined. This method returns a │ │ │ │ │ + * feature instead of a feature id to avoid an unnecessary lookup on the │ │ │ │ │ + * layer. │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - // Make a list, so that getFullRequestString uses literal "," │ │ │ │ │ - var extent = [bounds.left, bounds.bottom, bounds.right, bounds.top]; │ │ │ │ │ - │ │ │ │ │ - var imageSize = this.getImageSize(); │ │ │ │ │ - │ │ │ │ │ - // make lists, so that literal ','s are used │ │ │ │ │ - var url = this.getFullRequestString({ │ │ │ │ │ - mapext: extent, │ │ │ │ │ - imgext: extent, │ │ │ │ │ - map_size: [imageSize.w, imageSize.h], │ │ │ │ │ - imgx: imageSize.w / 2, │ │ │ │ │ - imgy: imageSize.h / 2, │ │ │ │ │ - imgxy: [imageSize.w, imageSize.h] │ │ │ │ │ - }); │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var featureId, feature; │ │ │ │ │ │ │ │ │ │ - return url; │ │ │ │ │ + if (this.hitDetection && this.root.style.display !== "none") { │ │ │ │ │ + // this dragging check should go in the feature handler │ │ │ │ │ + if (!this.map.dragging) { │ │ │ │ │ + var xy = evt.xy; │ │ │ │ │ + var x = xy.x | 0; │ │ │ │ │ + var y = xy.y | 0; │ │ │ │ │ + var data = this.hitContext.getImageData(x, y, 1, 1).data; │ │ │ │ │ + if (data[3] === 255) { // antialiased │ │ │ │ │ + var id = data[2] + (256 * (data[1] + (256 * data[0]))); │ │ │ │ │ + if (id) { │ │ │ │ │ + featureId = "OpenLayers_Feature_Vector_" + (id - 1 + this.hitOverflow); │ │ │ │ │ + try { │ │ │ │ │ + feature = this.features[featureId][0]; │ │ │ │ │ + } catch (err) { │ │ │ │ │ + // Because of antialiasing on the canvas, when the hit location is at a point where the edge of │ │ │ │ │ + // one symbol intersects the interior of another symbol, a wrong hit color (and therefore id) results. │ │ │ │ │ + // todo: set Antialiasing = 'off' on the hitContext as soon as browsers allow it. │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return feature; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getFullRequestString │ │ │ │ │ - * combine the layer's url with its params and these newParams. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newParams - {Object} New parameters that should be added to the │ │ │ │ │ - * request string. │ │ │ │ │ - * altUrl - {String} (optional) Replace the URL in the full request │ │ │ │ │ - * string with the provided URL. │ │ │ │ │ + /** │ │ │ │ │ + * Method: eraseFeatures │ │ │ │ │ + * This is called by the layer to erase features; removes the feature from │ │ │ │ │ + * the list, then redraws the layer. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters embedded in it. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ */ │ │ │ │ │ - getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ - // use layer's url unless altUrl passed in │ │ │ │ │ - var url = (altUrl == null) ? this.url : altUrl; │ │ │ │ │ - │ │ │ │ │ - // create a new params hashtable with all the layer params and the │ │ │ │ │ - // new params together. then convert to string │ │ │ │ │ - var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ - allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ - var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ - │ │ │ │ │ - // if url is not a string, it should be an array of strings, │ │ │ │ │ - // in which case we will deterministically select one of them in │ │ │ │ │ - // order to evenly distribute requests to different urls. │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(paramsString, url); │ │ │ │ │ + eraseFeatures: function(features) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // ignore parameters that are already in the url search string │ │ │ │ │ - var urlParams = OpenLayers.Util.upperCaseObject( │ │ │ │ │ - OpenLayers.Util.getParameters(url)); │ │ │ │ │ - for (var key in allParams) { │ │ │ │ │ - if (key.toUpperCase() in urlParams) { │ │ │ │ │ - delete allParams[key]; │ │ │ │ │ - } │ │ │ │ │ + for (var i = 0; i < features.length; ++i) { │ │ │ │ │ + delete this.features[features[i].id]; │ │ │ │ │ } │ │ │ │ │ - paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ - │ │ │ │ │ - // requestString always starts with url │ │ │ │ │ - var requestString = url; │ │ │ │ │ - │ │ │ │ │ - // MapServer needs '+' seperating things like bounds/height/width. │ │ │ │ │ - // Since typically this is URL encoded, we use a slight hack: we │ │ │ │ │ - // depend on the list-like functionality of getParameterString to │ │ │ │ │ - // leave ',' only in the case of list items (since otherwise it is │ │ │ │ │ - // encoded) then do a regular expression replace on the , characters │ │ │ │ │ - // to '+' │ │ │ │ │ - // │ │ │ │ │ - paramsString = paramsString.replace(/,/g, "+"); │ │ │ │ │ + this.redraw(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (paramsString != "") { │ │ │ │ │ - var lastServerChar = url.charAt(url.length - 1); │ │ │ │ │ - if ((lastServerChar == "&") || (lastServerChar == "?")) { │ │ │ │ │ - requestString += paramsString; │ │ │ │ │ - } else { │ │ │ │ │ - if (url.indexOf('?') == -1) { │ │ │ │ │ - //serverPath has no ? -- add one │ │ │ │ │ - requestString += '?' + paramsString; │ │ │ │ │ - } else { │ │ │ │ │ - //serverPath contains ?, so must already have paramsString at the end │ │ │ │ │ - requestString += '&' + paramsString; │ │ │ │ │ + /** │ │ │ │ │ + * Method: redraw │ │ │ │ │ + * The real 'meat' of the function: any time things have changed, │ │ │ │ │ + * redraw() can be called to loop over all the data and (you guessed │ │ │ │ │ + * it) redraw it. Unlike Elements-based Renderers, we can't interact │ │ │ │ │ + * with things once they're drawn, to remove them, for example, so │ │ │ │ │ + * instead we have to just clear everything and draw from scratch. │ │ │ │ │ + */ │ │ │ │ │ + redraw: function() { │ │ │ │ │ + if (!this.locked) { │ │ │ │ │ + var height = this.root.height; │ │ │ │ │ + var width = this.root.width; │ │ │ │ │ + this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.clearRect(0, 0, width, height); │ │ │ │ │ + } │ │ │ │ │ + var labelMap = []; │ │ │ │ │ + var feature, geometry, style; │ │ │ │ │ + var worldBounds = (this.map.baseLayer && this.map.baseLayer.wrapDateLine) && this.map.getMaxExtent(); │ │ │ │ │ + for (var id in this.features) { │ │ │ │ │ + if (!this.features.hasOwnProperty(id)) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + feature = this.features[id][0]; │ │ │ │ │ + geometry = feature.geometry; │ │ │ │ │ + this.calculateFeatureDx(geometry.getBounds(), worldBounds); │ │ │ │ │ + style = this.features[id][1]; │ │ │ │ │ + this.drawGeometry(geometry, style, feature.id); │ │ │ │ │ + if (style.label) { │ │ │ │ │ + labelMap.push([feature, style]); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + var item; │ │ │ │ │ + for (var i = 0, len = labelMap.length; i < len; ++i) { │ │ │ │ │ + item = labelMap[i]; │ │ │ │ │ + this.drawText(item[0].geometry.getCentroid(), item[1]); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return requestString; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.MapServer" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.Canvas" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.Canvas.LABEL_ALIGN │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.Canvas.LABEL_ALIGN = { │ │ │ │ │ + "l": "left", │ │ │ │ │ + "r": "right", │ │ │ │ │ + "t": "top", │ │ │ │ │ + "b": "bottom" │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.Canvas.LABEL_FACTOR │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.Canvas.LABEL_FACTOR = { │ │ │ │ │ + "l": 0, │ │ │ │ │ + "r": -1, │ │ │ │ │ + "t": 0, │ │ │ │ │ + "b": -1 │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.Canvas.drawImageScaleFactor │ │ │ │ │ + * {Number} Scale factor to apply to the canvas drawImage arguments. This │ │ │ │ │ + * is always 1 except for Android 2.1 devices, to work around │ │ │ │ │ + * http://code.google.com/p/android/issues/detail?id=5141. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.Canvas.drawImageScaleFactor = null; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/ArcGIS93Rest.js │ │ │ │ │ + OpenLayers/Renderer/Elements.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/Renderer.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.ArcGIS93Rest │ │ │ │ │ - * Instances of OpenLayers.Layer.ArcGIS93Rest are used to display data from │ │ │ │ │ - * ESRI ArcGIS Server 9.3 (and up?) Mapping Services using the REST API. │ │ │ │ │ - * Create a new ArcGIS93Rest layer with the <OpenLayers.Layer.ArcGIS93Rest> │ │ │ │ │ - * constructor. More detail on the REST API is available at │ │ │ │ │ - * http://sampleserver1.arcgisonline.com/ArcGIS/SDK/REST/index.html ; │ │ │ │ │ - * specifically, the URL provided to this layer should be an export service │ │ │ │ │ - * URL: http://sampleserver1.arcgisonline.com/ArcGIS/SDK/REST/export.html │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ + * Class: OpenLayers.ElementsIndexer │ │ │ │ │ + * This class takes care of figuring out which order elements should be │ │ │ │ │ + * placed in the DOM based on given indexing methods. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.ArcGIS93Rest = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ +OpenLayers.ElementsIndexer = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: DEFAULT_PARAMS │ │ │ │ │ - * {Object} Hashtable of default parameter key/value pairs │ │ │ │ │ + * Property: maxZIndex │ │ │ │ │ + * {Integer} This is the largest-most z-index value for a node │ │ │ │ │ + * contained within the indexer. │ │ │ │ │ */ │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - format: "png" │ │ │ │ │ - }, │ │ │ │ │ + maxZIndex: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} Default is true for ArcGIS93Rest layer │ │ │ │ │ + * Property: order │ │ │ │ │ + * {Array<String>} This is an array of node id's stored in the │ │ │ │ │ + * order that they should show up on screen. Id's higher up in the │ │ │ │ │ + * array (higher array index) represent nodes with higher z-indeces. │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + order: null, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: indices │ │ │ │ │ + * {Object} This is a hash that maps node ids to their z-index value │ │ │ │ │ + * stored in the indexer. This is done to make finding a nodes z-index │ │ │ │ │ + * value O(1). │ │ │ │ │ + */ │ │ │ │ │ + indices: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.ArcGIS93Rest │ │ │ │ │ - * Create a new ArcGIS93Rest layer object. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var arcims = new OpenLayers.Layer.ArcGIS93Rest("MyName", │ │ │ │ │ - * "http://sampleserver1.arcgisonline.com/ArcGIS/rest/services/Specialty/ESRI_StateCityHighway_USA/MapServer/export", │ │ │ │ │ - * { │ │ │ │ │ - * layers: "0,1,2" │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} A name for the layer │ │ │ │ │ - * url - {String} Base url for the ArcGIS server REST service │ │ │ │ │ - * options - {Object} An object with key/value pairs representing the │ │ │ │ │ - * options and option values. │ │ │ │ │ - * │ │ │ │ │ - * Valid Options: │ │ │ │ │ - * format - {String} MIME type of desired image type. │ │ │ │ │ - * layers - {String} Comma-separated list of layers to display. │ │ │ │ │ - * srs - {String} Projection ID. │ │ │ │ │ + * Property: compare │ │ │ │ │ + * {Function} This is the function used to determine placement of │ │ │ │ │ + * of a new node within the indexer. If null, this defaults to to │ │ │ │ │ + * the Z_ORDER_DRAWING_ORDER comparison method. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - //uppercase params │ │ │ │ │ - params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ - newArguments.push(name, url, params, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, │ │ │ │ │ - OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS) │ │ │ │ │ - ); │ │ │ │ │ + compare: null, │ │ │ │ │ │ │ │ │ │ - //layer is transparent │ │ │ │ │ - if (this.params.TRANSPARENT && │ │ │ │ │ - this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: initialize │ │ │ │ │ + * Create a new indexer with │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * yOrdering - {Boolean} Whether to use y-ordering. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(yOrdering) { │ │ │ │ │ │ │ │ │ │ - // unless explicitly set in options, make layer an overlay │ │ │ │ │ - if ((options == null) || (!options.isBaseLayer)) { │ │ │ │ │ - this.isBaseLayer = false; │ │ │ │ │ - } │ │ │ │ │ + this.compare = yOrdering ? │ │ │ │ │ + OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_Y_ORDER : │ │ │ │ │ + OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_DRAWING_ORDER; │ │ │ │ │ │ │ │ │ │ - // jpegs can never be transparent, so intelligently switch the │ │ │ │ │ - // format, depending on the browser's capabilities │ │ │ │ │ - if (this.params.FORMAT == "jpg") { │ │ │ │ │ - this.params.FORMAT = OpenLayers.Util.alphaHack() ? "gif" : │ │ │ │ │ - "png"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + this.clear(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.ArcGIS93Rest>} An exact clone of this layer │ │ │ │ │ + * APIMethod: insert │ │ │ │ │ + * Insert a new node into the indexer. In order to find the correct │ │ │ │ │ + * positioning for the node to be inserted, this method uses a binary │ │ │ │ │ + * search. This makes inserting O(log(n)). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newNode - {DOMElement} The new node to be inserted. │ │ │ │ │ + * │ │ │ │ │ + * Returns │ │ │ │ │ + * {DOMElement} the node before which we should insert our newNode, or │ │ │ │ │ + * null if newNode can just be appended. │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ + insert: function(newNode) { │ │ │ │ │ + // If the node is known to the indexer, remove it so we can │ │ │ │ │ + // recalculate where it should go. │ │ │ │ │ + if (this.exists(newNode)) { │ │ │ │ │ + this.remove(newNode); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.ArcGIS93Rest(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.params, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ + var nodeId = newNode.id; │ │ │ │ │ + │ │ │ │ │ + this.determineZIndex(newNode); │ │ │ │ │ + │ │ │ │ │ + var leftIndex = -1; │ │ │ │ │ + var rightIndex = this.order.length; │ │ │ │ │ + var middle; │ │ │ │ │ + │ │ │ │ │ + while (rightIndex - leftIndex > 1) { │ │ │ │ │ + middle = parseInt((leftIndex + rightIndex) / 2); │ │ │ │ │ + │ │ │ │ │ + var placement = this.compare(this, newNode, │ │ │ │ │ + OpenLayers.Util.getElement(this.order[middle])); │ │ │ │ │ + │ │ │ │ │ + if (placement > 0) { │ │ │ │ │ + leftIndex = middle; │ │ │ │ │ + } else { │ │ │ │ │ + rightIndex = middle; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + this.order.splice(rightIndex, 0, nodeId); │ │ │ │ │ + this.indices[nodeId] = this.getZIndex(newNode); │ │ │ │ │ │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ + // If the new node should be before another in the index │ │ │ │ │ + // order, return the node before which we have to insert the new one; │ │ │ │ │ + // else, return null to indicate that the new node can be appended. │ │ │ │ │ + return this.getNextElement(rightIndex); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return obj; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: remove │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The node to be removed. │ │ │ │ │ + */ │ │ │ │ │ + remove: function(node) { │ │ │ │ │ + var nodeId = node.id; │ │ │ │ │ + var arrayIndex = OpenLayers.Util.indexOf(this.order, nodeId); │ │ │ │ │ + if (arrayIndex >= 0) { │ │ │ │ │ + // Remove it from the order array, as well as deleting the node │ │ │ │ │ + // from the indeces hash. │ │ │ │ │ + this.order.splice(arrayIndex, 1); │ │ │ │ │ + delete this.indices[nodeId]; │ │ │ │ │ + │ │ │ │ │ + // Reset the maxium z-index based on the last item in the │ │ │ │ │ + // order array. │ │ │ │ │ + if (this.order.length > 0) { │ │ │ │ │ + var lastId = this.order[this.order.length - 1]; │ │ │ │ │ + this.maxZIndex = this.indices[lastId]; │ │ │ │ │ + } else { │ │ │ │ │ + this.maxZIndex = 0; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clear │ │ │ │ │ + */ │ │ │ │ │ + clear: function() { │ │ │ │ │ + this.order = []; │ │ │ │ │ + this.indices = {}; │ │ │ │ │ + this.maxZIndex = 0; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * Return an image url this layer. │ │ │ │ │ + * APIMethod: exists │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox for the │ │ │ │ │ - * request. │ │ │ │ │ + * node - {DOMElement} The node to test for existence. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A string with the map image's url. │ │ │ │ │ + * {Boolean} Whether or not the node exists in the indexer? │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ + exists: function(node) { │ │ │ │ │ + return (this.indices[node.id] != null); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // ArcGIS Server only wants the numeric portion of the projection ID. │ │ │ │ │ - var projWords = this.projection.getCode().split(":"); │ │ │ │ │ - var srid = projWords[projWords.length - 1]; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getZIndex │ │ │ │ │ + * Get the z-index value for the current node from the node data itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The node whose z-index to get. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} The z-index value for the specified node (from the node │ │ │ │ │ + * data itself). │ │ │ │ │ + */ │ │ │ │ │ + getZIndex: function(node) { │ │ │ │ │ + return node._style.graphicZIndex; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var imageSize = this.getImageSize(); │ │ │ │ │ - var newParams = { │ │ │ │ │ - 'BBOX': bounds.toBBOX(), │ │ │ │ │ - 'SIZE': imageSize.w + "," + imageSize.h, │ │ │ │ │ - // We always want image, the other options were json, image with a whole lotta html around it, etc. │ │ │ │ │ - 'F': "image", │ │ │ │ │ - 'BBOXSR': srid, │ │ │ │ │ - 'IMAGESR': srid │ │ │ │ │ - }; │ │ │ │ │ + /** │ │ │ │ │ + * Method: determineZIndex │ │ │ │ │ + * Determine the z-index for the current node if there isn't one, │ │ │ │ │ + * and set the maximum value if we've found a new maximum. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + determineZIndex: function(node) { │ │ │ │ │ + var zIndex = node._style.graphicZIndex; │ │ │ │ │ │ │ │ │ │ - // Now add the filter parameters. │ │ │ │ │ - if (this.layerDefs) { │ │ │ │ │ - var layerDefStrList = []; │ │ │ │ │ - var layerID; │ │ │ │ │ - for (layerID in this.layerDefs) { │ │ │ │ │ - if (this.layerDefs.hasOwnProperty(layerID)) { │ │ │ │ │ - if (this.layerDefs[layerID]) { │ │ │ │ │ - layerDefStrList.push(layerID); │ │ │ │ │ - layerDefStrList.push(":"); │ │ │ │ │ - layerDefStrList.push(this.layerDefs[layerID]); │ │ │ │ │ - layerDefStrList.push(";"); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (layerDefStrList.length > 0) { │ │ │ │ │ - newParams['LAYERDEFS'] = layerDefStrList.join(""); │ │ │ │ │ - } │ │ │ │ │ + // Everything must have a zIndex. If none is specified, │ │ │ │ │ + // this means the user *must* (hint: assumption) want this │ │ │ │ │ + // node to succomb to drawing order. To enforce drawing order │ │ │ │ │ + // over all indexing methods, we'll create a new z-index that's │ │ │ │ │ + // greater than any currently in the indexer. │ │ │ │ │ + if (zIndex == null) { │ │ │ │ │ + zIndex = this.maxZIndex; │ │ │ │ │ + node._style.graphicZIndex = zIndex; │ │ │ │ │ + } else if (zIndex > this.maxZIndex) { │ │ │ │ │ + this.maxZIndex = zIndex; │ │ │ │ │ } │ │ │ │ │ - var requestString = this.getFullRequestString(newParams); │ │ │ │ │ - return requestString; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setLayerFilter │ │ │ │ │ - * addTile creates a tile, initializes it, and adds it to the layer div. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getNextElement │ │ │ │ │ + * Get the next element in the order stack. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * id - {String} The id of the layer to which the filter applies. │ │ │ │ │ - * queryDef - {String} A sql-ish query filter, for more detail see the ESRI │ │ │ │ │ - * documentation at http://sampleserver1.arcgisonline.com/ArcGIS/SDK/REST/export.html │ │ │ │ │ + * index - {Integer} The index of the current node in this.order. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} the node following the index passed in, or │ │ │ │ │ + * null. │ │ │ │ │ */ │ │ │ │ │ - setLayerFilter: function(id, queryDef) { │ │ │ │ │ - if (!this.layerDefs) { │ │ │ │ │ - this.layerDefs = {}; │ │ │ │ │ - } │ │ │ │ │ - if (queryDef) { │ │ │ │ │ - this.layerDefs[id] = queryDef; │ │ │ │ │ + getNextElement: function(index) { │ │ │ │ │ + var nextIndex = index + 1; │ │ │ │ │ + if (nextIndex < this.order.length) { │ │ │ │ │ + var nextElement = OpenLayers.Util.getElement(this.order[nextIndex]); │ │ │ │ │ + if (nextElement == undefined) { │ │ │ │ │ + nextElement = this.getNextElement(nextIndex); │ │ │ │ │ + } │ │ │ │ │ + return nextElement; │ │ │ │ │ } else { │ │ │ │ │ - delete this.layerDefs[id]; │ │ │ │ │ + return null; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.ElementsIndexer" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.ElementsIndexer.IndexingMethods │ │ │ │ │ + * These are the compare methods for figuring out where a new node should be │ │ │ │ │ + * placed within the indexer. These methods are very similar to general │ │ │ │ │ + * sorting methods in that they return -1, 0, and 1 to specify the │ │ │ │ │ + * direction in which new nodes fall in the ordering. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.ElementsIndexer.IndexingMethods = { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: clearLayerFilter │ │ │ │ │ - * Clears layer filters, either from a specific layer, │ │ │ │ │ - * or all of them. │ │ │ │ │ - * │ │ │ │ │ + * Method: Z_ORDER │ │ │ │ │ + * This compare method is used by other comparison methods. │ │ │ │ │ + * It can be used individually for ordering, but is not recommended, │ │ │ │ │ + * because it doesn't subscribe to drawing order. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * id - {String} The id of the layer from which to remove any │ │ │ │ │ - * filter. If unspecified/blank, all filters │ │ │ │ │ - * will be removed. │ │ │ │ │ + * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ + * newNode - {DOMElement} │ │ │ │ │ + * nextNode - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} │ │ │ │ │ */ │ │ │ │ │ - clearLayerFilter: function(id) { │ │ │ │ │ - if (id) { │ │ │ │ │ - delete this.layerDefs[id]; │ │ │ │ │ - } else { │ │ │ │ │ - delete this.layerDefs; │ │ │ │ │ + Z_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var newZIndex = indexer.getZIndex(newNode); │ │ │ │ │ + │ │ │ │ │ + var returnVal = 0; │ │ │ │ │ + if (nextNode) { │ │ │ │ │ + var nextZIndex = indexer.getZIndex(nextNode); │ │ │ │ │ + returnVal = newZIndex - nextZIndex; │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + return returnVal; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: mergeNewParams │ │ │ │ │ - * Catch changeParams and uppercase the new params to be merged in │ │ │ │ │ - * before calling changeParams on the super class. │ │ │ │ │ - * │ │ │ │ │ - * Once params have been changed, the tiles will be reloaded with │ │ │ │ │ - * the new parameters. │ │ │ │ │ + * APIMethod: Z_ORDER_DRAWING_ORDER │ │ │ │ │ + * This method orders nodes by their z-index, but does so in a way │ │ │ │ │ + * that, if there are other nodes with the same z-index, the newest │ │ │ │ │ + * drawn will be the front most within that z-index. This is the │ │ │ │ │ + * default indexing method. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * newParams - {Object} Hashtable of new params to use │ │ │ │ │ + * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ + * newNode - {DOMElement} │ │ │ │ │ + * nextNode - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} │ │ │ │ │ */ │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ - var newArguments = [upperParams]; │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, │ │ │ │ │ - newArguments); │ │ │ │ │ - }, │ │ │ │ │ + Z_ORDER_DRAWING_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER( │ │ │ │ │ + indexer, │ │ │ │ │ + newNode, │ │ │ │ │ + nextNode │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.ArcGIS93Rest" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Zoomify.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + // Make Z_ORDER subscribe to drawing order by pushing it above │ │ │ │ │ + // all of the other nodes with the same z-index. │ │ │ │ │ + if (nextNode && returnVal == 0) { │ │ │ │ │ + returnVal = 1; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + return returnVal; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/* │ │ │ │ │ - * Development supported by a R&D grant DC08P02OUK006 - Old Maps Online │ │ │ │ │ - * (www.oldmapsonline.org) from Ministry of Culture of the Czech Republic. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: Z_ORDER_Y_ORDER │ │ │ │ │ + * This one should really be called Z_ORDER_Y_ORDER_DRAWING_ORDER, as it │ │ │ │ │ + * best describes which ordering methods have precedence (though, the │ │ │ │ │ + * name would be too long). This method orders nodes by their z-index, │ │ │ │ │ + * but does so in a way that, if there are other nodes with the same │ │ │ │ │ + * z-index, the nodes with the lower y position will be "closer" than │ │ │ │ │ + * those with a higher y position. If two nodes have the exact same y │ │ │ │ │ + * position, however, then this method will revert to using drawing │ │ │ │ │ + * order to decide placement. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ + * newNode - {DOMElement} │ │ │ │ │ + * nextNode - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} │ │ │ │ │ + */ │ │ │ │ │ + Z_ORDER_Y_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER( │ │ │ │ │ + indexer, │ │ │ │ │ + newNode, │ │ │ │ │ + nextNode │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ + if (nextNode && returnVal === 0) { │ │ │ │ │ + var result = nextNode._boundsBottom - newNode._boundsBottom; │ │ │ │ │ + returnVal = (result === 0) ? 1 : result; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ - */ │ │ │ │ │ + return returnVal; │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.Zoomify │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ + * Class: OpenLayers.Renderer.Elements │ │ │ │ │ + * This is another virtual class in that it should never be instantiated by │ │ │ │ │ + * itself as a Renderer. It exists because there is *tons* of shared │ │ │ │ │ + * functionality between different vector libraries which use nodes/elements │ │ │ │ │ + * as a base for rendering vectors. │ │ │ │ │ + * │ │ │ │ │ + * The highlevel bits of code that are implemented here are the adding and │ │ │ │ │ + * removing of geometries, which is essentially the same for any │ │ │ │ │ + * element-based renderer. The details of creating each node and drawing the │ │ │ │ │ + * paths are of course different, but the machinery is the same. │ │ │ │ │ + * │ │ │ │ │ + * Inherits: │ │ │ │ │ + * - <OpenLayers.Renderer> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Zoomify = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ +OpenLayers.Renderer.Elements = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: size │ │ │ │ │ - * {<OpenLayers.Size>} The Zoomify image size in pixels. │ │ │ │ │ + * Property: rendererRoot │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - size: null, │ │ │ │ │ + rendererRoot: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * Property: root │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + root: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: standardTileSize │ │ │ │ │ - * {Integer} The size of a standard (non-border) square tile in pixels. │ │ │ │ │ + * Property: vectorRoot │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - standardTileSize: 256, │ │ │ │ │ + vectorRoot: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: tileOriginCorner │ │ │ │ │ - * {String} This layer uses top-left as tile origin │ │ │ │ │ - **/ │ │ │ │ │ - tileOriginCorner: "tl", │ │ │ │ │ + /** │ │ │ │ │ + * Property: textRoot │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + textRoot: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: numberOfTiers │ │ │ │ │ - * {Integer} Depth of the Zoomify pyramid, number of tiers (zoom levels) │ │ │ │ │ - * - filled during Zoomify pyramid initialization. │ │ │ │ │ + * Property: xmlns │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - numberOfTiers: 0, │ │ │ │ │ + xmlns: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: tileCountUpToTier │ │ │ │ │ - * {Array(Integer)} Number of tiles up to the given tier of pyramid. │ │ │ │ │ - * - filled during Zoomify pyramid initialization. │ │ │ │ │ + * Property: xOffset │ │ │ │ │ + * {Number} Offset to apply to the renderer viewport translation in x │ │ │ │ │ + * direction. If the renderer extent's center is on the right of the │ │ │ │ │ + * dateline (i.e. exceeds the world bounds), we shift the viewport to the │ │ │ │ │ + * left by one world width. This avoids that features disappear from the │ │ │ │ │ + * map viewport. Because our dateline handling logic in other places │ │ │ │ │ + * ensures that extents crossing the dateline always have a center │ │ │ │ │ + * exceeding the world bounds on the left, we need this offset to make sure │ │ │ │ │ + * that the same is true for the renderer extent in pixel space as well. │ │ │ │ │ */ │ │ │ │ │ - tileCountUpToTier: null, │ │ │ │ │ + xOffset: 0, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: tierSizeInTiles │ │ │ │ │ - * {Array(<OpenLayers.Size>)} Size (in tiles) for each tier of pyramid. │ │ │ │ │ - * - filled during Zoomify pyramid initialization. │ │ │ │ │ + * Property: rightOfDateLine │ │ │ │ │ + * {Boolean} Keeps track of the location of the map extent relative to the │ │ │ │ │ + * date line. The <setExtent> method compares this value (which is the one │ │ │ │ │ + * from the previous <setExtent> call) with the current position of the map │ │ │ │ │ + * extent relative to the date line and updates the xOffset when the extent │ │ │ │ │ + * has moved from one side of the date line to the other. │ │ │ │ │ */ │ │ │ │ │ - tierSizeInTiles: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: tierImageSize │ │ │ │ │ - * {Array(<OpenLayers.Size>)} Image size in pixels for each pyramid tier. │ │ │ │ │ - * - filled during Zoomify pyramid initialization. │ │ │ │ │ + * Property: Indexer │ │ │ │ │ + * {<OpenLayers.ElementIndexer>} An instance of OpenLayers.ElementsIndexer │ │ │ │ │ + * created upon initialization if the zIndexing or yOrdering options │ │ │ │ │ + * passed to this renderer's constructor are set to true. │ │ │ │ │ */ │ │ │ │ │ - tierImageSize: null, │ │ │ │ │ + indexer: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Zoomify │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} A name for the layer. │ │ │ │ │ - * url - {String} - Relative or absolute path to the image or more │ │ │ │ │ - * precisly to the TileGroup[X] directories root. │ │ │ │ │ - * Flash plugin use the variable name "zoomifyImagePath" for this. │ │ │ │ │ - * size - {<OpenLayers.Size>} The size (in pixels) of the image. │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + * Constant: BACKGROUND_ID_SUFFIX │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, size, options) { │ │ │ │ │ + BACKGROUND_ID_SUFFIX: "_background", │ │ │ │ │ │ │ │ │ │ - // initilize the Zoomify pyramid for given size │ │ │ │ │ - this.initializeZoomify(size); │ │ │ │ │ + /** │ │ │ │ │ + * Constant: LABEL_ID_SUFFIX │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + LABEL_ID_SUFFIX: "_label", │ │ │ │ │ │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [ │ │ │ │ │ - name, url, size, {}, │ │ │ │ │ - options │ │ │ │ │ - ]); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Constant: LABEL_OUTLINE_SUFFIX │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + LABEL_OUTLINE_SUFFIX: "_outline", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: initializeZoomify │ │ │ │ │ - * It generates constants for all tiers of the Zoomify pyramid │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Renderer.Elements │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>} The size of the image in pixels │ │ │ │ │ + * containerID - {String} │ │ │ │ │ + * options - {Object} options for this renderer. │ │ │ │ │ * │ │ │ │ │ + * Supported options are: │ │ │ │ │ + * yOrdering - {Boolean} Whether to use y-ordering │ │ │ │ │ + * zIndexing - {Boolean} Whether to use z-indexing. Will be ignored │ │ │ │ │ + * if yOrdering is set to true. │ │ │ │ │ */ │ │ │ │ │ - initializeZoomify: function(size) { │ │ │ │ │ - │ │ │ │ │ - var imageSize = size.clone(); │ │ │ │ │ - this.size = size.clone(); │ │ │ │ │ - var tiles = new OpenLayers.Size( │ │ │ │ │ - Math.ceil(imageSize.w / this.standardTileSize), │ │ │ │ │ - Math.ceil(imageSize.h / this.standardTileSize) │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - this.tierSizeInTiles = [tiles]; │ │ │ │ │ - this.tierImageSize = [imageSize]; │ │ │ │ │ + initialize: function(containerID, options) { │ │ │ │ │ + OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - while (imageSize.w > this.standardTileSize || │ │ │ │ │ - imageSize.h > this.standardTileSize) { │ │ │ │ │ + this.rendererRoot = this.createRenderRoot(); │ │ │ │ │ + this.root = this.createRoot("_root"); │ │ │ │ │ + this.vectorRoot = this.createRoot("_vroot"); │ │ │ │ │ + this.textRoot = this.createRoot("_troot"); │ │ │ │ │ │ │ │ │ │ - imageSize = new OpenLayers.Size( │ │ │ │ │ - Math.floor(imageSize.w / 2), │ │ │ │ │ - Math.floor(imageSize.h / 2) │ │ │ │ │ - ); │ │ │ │ │ - tiles = new OpenLayers.Size( │ │ │ │ │ - Math.ceil(imageSize.w / this.standardTileSize), │ │ │ │ │ - Math.ceil(imageSize.h / this.standardTileSize) │ │ │ │ │ - ); │ │ │ │ │ - this.tierSizeInTiles.push(tiles); │ │ │ │ │ - this.tierImageSize.push(imageSize); │ │ │ │ │ - } │ │ │ │ │ + this.root.appendChild(this.vectorRoot); │ │ │ │ │ + this.root.appendChild(this.textRoot); │ │ │ │ │ │ │ │ │ │ - this.tierSizeInTiles.reverse(); │ │ │ │ │ - this.tierImageSize.reverse(); │ │ │ │ │ + this.rendererRoot.appendChild(this.root); │ │ │ │ │ + this.container.appendChild(this.rendererRoot); │ │ │ │ │ │ │ │ │ │ - this.numberOfTiers = this.tierSizeInTiles.length; │ │ │ │ │ - var resolutions = [1]; │ │ │ │ │ - this.tileCountUpToTier = [0]; │ │ │ │ │ - for (var i = 1; i < this.numberOfTiers; i++) { │ │ │ │ │ - resolutions.unshift(Math.pow(2, i)); │ │ │ │ │ - this.tileCountUpToTier.push( │ │ │ │ │ - this.tierSizeInTiles[i - 1].w * this.tierSizeInTiles[i - 1].h + │ │ │ │ │ - this.tileCountUpToTier[i - 1] │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (!this.serverResolutions) { │ │ │ │ │ - this.serverResolutions = resolutions; │ │ │ │ │ + if (options && (options.zIndexing || options.yOrdering)) { │ │ │ │ │ + this.indexer = new OpenLayers.ElementsIndexer(options.yOrdering); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod:destroy │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ destroy: function() { │ │ │ │ │ - // for now, nothing special to do here. │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.destroy.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - // Remove from memory the Zoomify pyramid - is that enough? │ │ │ │ │ - this.tileCountUpToTier.length = 0; │ │ │ │ │ - this.tierSizeInTiles.length = 0; │ │ │ │ │ - this.tierImageSize.length = 0; │ │ │ │ │ + this.clear(); │ │ │ │ │ + │ │ │ │ │ + this.rendererRoot = null; │ │ │ │ │ + this.root = null; │ │ │ │ │ + this.xmlns = null; │ │ │ │ │ │ │ │ │ │ + OpenLayers.Renderer.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Zoomify>} An exact clone of this <OpenLayers.Layer.Zoomify> │ │ │ │ │ + * Method: clear │ │ │ │ │ + * Remove all the elements from the root │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Zoomify(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.size, │ │ │ │ │ - this.options); │ │ │ │ │ + clear: function() { │ │ │ │ │ + var child; │ │ │ │ │ + var root = this.vectorRoot; │ │ │ │ │ + if (root) { │ │ │ │ │ + while (child = root.firstChild) { │ │ │ │ │ + root.removeChild(child); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + root = this.textRoot; │ │ │ │ │ + if (root) { │ │ │ │ │ + while (child = root.firstChild) { │ │ │ │ │ + root.removeChild(child); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + this.indexer.clear(); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ + * Method: setExtent │ │ │ │ │ + * Set the visible part of the layer. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} │ │ │ │ │ + * resolutionChanged - {Boolean} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as │ │ │ │ │ - * parameters │ │ │ │ │ + * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ + * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ + * False otherwise. │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((this.tileOrigin.lat - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getZoomForResolution(res); │ │ │ │ │ - │ │ │ │ │ - var tileIndex = x + y * this.tierSizeInTiles[z].w + this.tileCountUpToTier[z]; │ │ │ │ │ - var path = "TileGroup" + Math.floor((tileIndex) / 256) + │ │ │ │ │ - "/" + z + "-" + x + "-" + y + ".jpg"; │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(path, url); │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + var rightOfDateLine, │ │ │ │ │ + ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ + extent = extent.scale(1 / ratio), │ │ │ │ │ + world = this.map.getMaxExtent(); │ │ │ │ │ + if (world.right > extent.left && world.right < extent.right) { │ │ │ │ │ + rightOfDateLine = true; │ │ │ │ │ + } else if (world.left > extent.left && world.left < extent.right) { │ │ │ │ │ + rightOfDateLine = false; │ │ │ │ │ + } │ │ │ │ │ + if (rightOfDateLine !== this.rightOfDateLine || resolutionChanged) { │ │ │ │ │ + coordSysUnchanged = false; │ │ │ │ │ + this.xOffset = rightOfDateLine === true ? │ │ │ │ │ + world.getWidth() / resolution : 0; │ │ │ │ │ + } │ │ │ │ │ + this.rightOfDateLine = rightOfDateLine; │ │ │ │ │ } │ │ │ │ │ - return url + path; │ │ │ │ │ + return coordSysUnchanged; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getImageSize │ │ │ │ │ - * getImageSize returns size for a particular tile. If bounds are given as │ │ │ │ │ - * first argument, size is calculated (bottom-right tiles are non square). │ │ │ │ │ + /** │ │ │ │ │ + * Method: getNodeType │ │ │ │ │ + * This function is in charge of asking the specific renderer which type │ │ │ │ │ + * of node to create for the given geometry and style. All geometries │ │ │ │ │ + * in an Elements-based renderer consist of one node and some │ │ │ │ │ + * attributes. We have the nodeFactory() function which creates a node │ │ │ │ │ + * for us, but it takes a 'type' as input, and that is precisely what │ │ │ │ │ + * this function tells us. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The corresponding node type for the specified geometry │ │ │ │ │ + */ │ │ │ │ │ + getNodeType: function(geometry, style) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawGeometry │ │ │ │ │ + * Draw the geometry, creating new nodes, setting paths, setting style, │ │ │ │ │ + * setting featureId on the node. This method should only be called │ │ │ │ │ + * by the renderer itself. │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the geometry has been drawn completely; null if │ │ │ │ │ + * incomplete; false otherwise │ │ │ │ │ */ │ │ │ │ │ - getImageSize: function() { │ │ │ │ │ - if (arguments.length > 0) { │ │ │ │ │ - var bounds = this.adjustBounds(arguments[0]); │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((this.tileOrigin.lat - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getZoomForResolution(res); │ │ │ │ │ - var w = this.standardTileSize; │ │ │ │ │ - var h = this.standardTileSize; │ │ │ │ │ - if (x == this.tierSizeInTiles[z].w - 1) { │ │ │ │ │ - var w = this.tierImageSize[z].w % this.standardTileSize; │ │ │ │ │ + drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ + var className = geometry.CLASS_NAME; │ │ │ │ │ + var rendered = true; │ │ │ │ │ + if ((className == "OpenLayers.Geometry.Collection") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiPolygon")) { │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + rendered = this.drawGeometry( │ │ │ │ │ + geometry.components[i], style, featureId) && rendered; │ │ │ │ │ } │ │ │ │ │ - if (y == this.tierSizeInTiles[z].h - 1) { │ │ │ │ │ - var h = this.tierImageSize[z].h % this.standardTileSize; │ │ │ │ │ + return rendered; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + rendered = false; │ │ │ │ │ + var removeBackground = false; │ │ │ │ │ + if (style.display != "none") { │ │ │ │ │ + if (style.backgroundGraphic) { │ │ │ │ │ + this.redrawBackgroundNode(geometry.id, geometry, style, │ │ │ │ │ + featureId); │ │ │ │ │ + } else { │ │ │ │ │ + removeBackground = true; │ │ │ │ │ + } │ │ │ │ │ + rendered = this.redrawNode(geometry.id, geometry, style, │ │ │ │ │ + featureId); │ │ │ │ │ + } │ │ │ │ │ + if (rendered == false) { │ │ │ │ │ + var node = document.getElementById(geometry.id); │ │ │ │ │ + if (node) { │ │ │ │ │ + if (node._style.backgroundGraphic) { │ │ │ │ │ + removeBackground = true; │ │ │ │ │ + } │ │ │ │ │ + node.parentNode.removeChild(node); │ │ │ │ │ } │ │ │ │ │ - return (new OpenLayers.Size(w, h)); │ │ │ │ │ - } else { │ │ │ │ │ - return this.tileSize; │ │ │ │ │ } │ │ │ │ │ + if (removeBackground) { │ │ │ │ │ + var node = document.getElementById( │ │ │ │ │ + geometry.id + this.BACKGROUND_ID_SUFFIX); │ │ │ │ │ + if (node) { │ │ │ │ │ + node.parentNode.removeChild(node); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return rendered; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setMap │ │ │ │ │ - * When the layer is added to a map, then we can fetch our origin │ │ │ │ │ - * (if we don't have one.) │ │ │ │ │ - * │ │ │ │ │ + * Method: redrawNode │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * id - {String} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the complete geometry could be drawn, null if parts of │ │ │ │ │ + * the geometry could not be drawn, false otherwise │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.map.maxExtent.left, │ │ │ │ │ - this.map.maxExtent.top); │ │ │ │ │ - }, │ │ │ │ │ + redrawNode: function(id, geometry, style, featureId) { │ │ │ │ │ + style = this.applyDefaultSymbolizer(style); │ │ │ │ │ + // Get the node if it's already on the map. │ │ │ │ │ + var node = this.nodeFactory(id, this.getNodeType(geometry, style)); │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Zoomify" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Google/v3.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + // Set the data for the node, then draw it. │ │ │ │ │ + node._featureId = featureId; │ │ │ │ │ + node._boundsBottom = geometry.getBounds().bottom; │ │ │ │ │ + node._geometryClass = geometry.CLASS_NAME; │ │ │ │ │ + node._style = style; │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + var drawResult = this.drawGeometryNode(node, geometry, style); │ │ │ │ │ + if (drawResult === false) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ + node = drawResult.node; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/Google.js │ │ │ │ │ - */ │ │ │ │ │ + // Insert the node into the indexer so it can show us where to │ │ │ │ │ + // place it. Note that this operation is O(log(n)). If there's a │ │ │ │ │ + // performance problem (when dragging, for instance) this is │ │ │ │ │ + // likely where it would be. │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + var insert = this.indexer.insert(node); │ │ │ │ │ + if (insert) { │ │ │ │ │ + this.vectorRoot.insertBefore(node, insert); │ │ │ │ │ + } else { │ │ │ │ │ + this.vectorRoot.appendChild(node); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // if there's no indexer, simply append the node to root, │ │ │ │ │ + // but only if the node is a new one │ │ │ │ │ + if (node.parentNode !== this.vectorRoot) { │ │ │ │ │ + this.vectorRoot.appendChild(node); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Layer.Google.v3 │ │ │ │ │ - * │ │ │ │ │ - * Mixin providing functionality specific to the Google Maps API v3. │ │ │ │ │ - * │ │ │ │ │ - * To use this layer, you must include the GMaps v3 API in your html. │ │ │ │ │ - * │ │ │ │ │ - * Note that this layer configures the google.maps.map object with the │ │ │ │ │ - * "disableDefaultUI" option set to true. Using UI controls that the Google │ │ │ │ │ - * Maps API provides is not supported by the OpenLayers API. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.Google.v3 = { │ │ │ │ │ + this.postDraw(node); │ │ │ │ │ + │ │ │ │ │ + return drawResult.complete; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: DEFAULTS │ │ │ │ │ - * {Object} It is not recommended to change the properties set here. Note │ │ │ │ │ - * that Google.v3 layers only work when sphericalMercator is set to true. │ │ │ │ │ + * Method: redrawBackgroundNode │ │ │ │ │ + * Redraws the node using special 'background' style properties. Basically │ │ │ │ │ + * just calls redrawNode(), but instead of directly using the │ │ │ │ │ + * 'externalGraphic', 'graphicXOffset', 'graphicYOffset', and │ │ │ │ │ + * 'graphicZIndex' properties directly from the specified 'style' │ │ │ │ │ + * parameter, we create a new style object and set those properties │ │ │ │ │ + * from the corresponding 'background'-prefixed properties from │ │ │ │ │ + * specified 'style' parameter. │ │ │ │ │ * │ │ │ │ │ - * (code) │ │ │ │ │ - * { │ │ │ │ │ - * sphericalMercator: true, │ │ │ │ │ - * projection: "EPSG:900913" │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the complete geometry could be drawn, null if parts of │ │ │ │ │ + * the geometry could not be drawn, false otherwise │ │ │ │ │ */ │ │ │ │ │ - DEFAULTS: { │ │ │ │ │ - sphericalMercator: true, │ │ │ │ │ - projection: "EPSG:900913" │ │ │ │ │ + redrawBackgroundNode: function(id, geometry, style, featureId) { │ │ │ │ │ + var backgroundStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ + │ │ │ │ │ + // Set regular style attributes to apply to the background styles. │ │ │ │ │ + backgroundStyle.externalGraphic = backgroundStyle.backgroundGraphic; │ │ │ │ │ + backgroundStyle.graphicXOffset = backgroundStyle.backgroundXOffset; │ │ │ │ │ + backgroundStyle.graphicYOffset = backgroundStyle.backgroundYOffset; │ │ │ │ │ + backgroundStyle.graphicZIndex = backgroundStyle.backgroundGraphicZIndex; │ │ │ │ │ + backgroundStyle.graphicWidth = backgroundStyle.backgroundWidth || backgroundStyle.graphicWidth; │ │ │ │ │ + backgroundStyle.graphicHeight = backgroundStyle.backgroundHeight || backgroundStyle.graphicHeight; │ │ │ │ │ + │ │ │ │ │ + // Erase background styles. │ │ │ │ │ + backgroundStyle.backgroundGraphic = null; │ │ │ │ │ + backgroundStyle.backgroundXOffset = null; │ │ │ │ │ + backgroundStyle.backgroundYOffset = null; │ │ │ │ │ + backgroundStyle.backgroundGraphicZIndex = null; │ │ │ │ │ + │ │ │ │ │ + return this.redrawNode( │ │ │ │ │ + id + this.BACKGROUND_ID_SUFFIX, │ │ │ │ │ + geometry, │ │ │ │ │ + backgroundStyle, │ │ │ │ │ + null │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: animationEnabled │ │ │ │ │ - * {Boolean} If set to true, the transition between zoom levels will be │ │ │ │ │ - * animated (if supported by the GMaps API for the device used). Set to │ │ │ │ │ - * false to match the zooming experience of other layer types. Default │ │ │ │ │ - * is true. Note that the GMaps API does not give us control over zoom │ │ │ │ │ - * animation, so if set to false, when zooming, this will make the │ │ │ │ │ - * layer temporarily invisible, wait until GMaps reports the map being │ │ │ │ │ - * idle, and make it visible again. The result will be a blank layer │ │ │ │ │ - * for a few moments while zooming. │ │ │ │ │ + * Method: drawGeometryNode │ │ │ │ │ + * Given a node, draw a geometry on the specified layer. │ │ │ │ │ + * node and geometry are required arguments, style is optional. │ │ │ │ │ + * This method is only called by the render itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} a hash with properties "node" (the drawn node) and "complete" │ │ │ │ │ + * (null if parts of the geometry could not be drawn, false if nothing │ │ │ │ │ + * could be drawn) │ │ │ │ │ */ │ │ │ │ │ - animationEnabled: true, │ │ │ │ │ + drawGeometryNode: function(node, geometry, style) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: loadMapObject │ │ │ │ │ - * Load the GMap and register appropriate event listeners. │ │ │ │ │ - */ │ │ │ │ │ - loadMapObject: function() { │ │ │ │ │ - if (!this.type) { │ │ │ │ │ - this.type = google.maps.MapTypeId.ROADMAP; │ │ │ │ │ + var options = { │ │ │ │ │ + 'isFilled': style.fill === undefined ? │ │ │ │ │ + true : style.fill, │ │ │ │ │ + 'isStroked': style.stroke === undefined ? │ │ │ │ │ + !!style.strokeWidth : style.stroke │ │ │ │ │ + }; │ │ │ │ │ + var drawn; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.graphic === false) { │ │ │ │ │ + options.isFilled = false; │ │ │ │ │ + options.isStroked = false; │ │ │ │ │ + } │ │ │ │ │ + drawn = this.drawPoint(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + options.isFilled = false; │ │ │ │ │ + drawn = this.drawLineString(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + drawn = this.drawLinearRing(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + drawn = this.drawPolygon(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + drawn = this.drawRectangle(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ - var mapObject; │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - // there are already Google layers added to this map │ │ │ │ │ - mapObject = cache.mapObject; │ │ │ │ │ - // increment the layer count │ │ │ │ │ - ++cache.count; │ │ │ │ │ - } else { │ │ │ │ │ - // this is the first Google layer for this map │ │ │ │ │ - // create GMap │ │ │ │ │ - var center = this.map.getCenter(); │ │ │ │ │ - var container = document.createElement('div'); │ │ │ │ │ - container.className = "olForeignContainer"; │ │ │ │ │ - container.style.width = '100%'; │ │ │ │ │ - container.style.height = '100%'; │ │ │ │ │ - mapObject = new google.maps.Map(container, { │ │ │ │ │ - center: center ? │ │ │ │ │ - new google.maps.LatLng(center.lat, center.lon) : new google.maps.LatLng(0, 0), │ │ │ │ │ - zoom: this.map.getZoom() || 0, │ │ │ │ │ - mapTypeId: this.type, │ │ │ │ │ - disableDefaultUI: true, │ │ │ │ │ - keyboardShortcuts: false, │ │ │ │ │ - draggable: false, │ │ │ │ │ - disableDoubleClickZoom: true, │ │ │ │ │ - scrollwheel: false, │ │ │ │ │ - streetViewControl: false │ │ │ │ │ - }); │ │ │ │ │ - var googleControl = document.createElement('div'); │ │ │ │ │ - googleControl.style.width = '100%'; │ │ │ │ │ - googleControl.style.height = '100%'; │ │ │ │ │ - mapObject.controls[google.maps.ControlPosition.TOP_LEFT].push(googleControl); │ │ │ │ │ │ │ │ │ │ - // cache elements for use by any other google layers added to │ │ │ │ │ - // this same map │ │ │ │ │ - cache = { │ │ │ │ │ - googleControl: googleControl, │ │ │ │ │ - mapObject: mapObject, │ │ │ │ │ - count: 1 │ │ │ │ │ + node._options = options; │ │ │ │ │ + │ │ │ │ │ + //set style │ │ │ │ │ + //TBD simplify this │ │ │ │ │ + if (drawn != false) { │ │ │ │ │ + return { │ │ │ │ │ + node: this.setStyle(node, style, options, geometry), │ │ │ │ │ + complete: drawn │ │ │ │ │ }; │ │ │ │ │ - OpenLayers.Layer.Google.cache[this.map.id] = cache; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - this.mapObject = mapObject; │ │ │ │ │ - this.setGMapVisibility(this.visibility); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: onMapResize │ │ │ │ │ + * Method: postDraw │ │ │ │ │ + * Things that have do be done after the geometry node is appended │ │ │ │ │ + * to its parent node. To be overridden by subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - if (this.visibility) { │ │ │ │ │ - google.maps.event.trigger(this.mapObject, "resize"); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + postDraw: function(node) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setGMapVisibility │ │ │ │ │ - * Display the GMap container and associated elements. │ │ │ │ │ + * Method: drawPoint │ │ │ │ │ + * Virtual function for drawing Point Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * visible - {Boolean} Display the GMap elements. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the point │ │ │ │ │ */ │ │ │ │ │ - setGMapVisibility: function(visible) { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - var map = this.map; │ │ │ │ │ - if (cache) { │ │ │ │ │ - var type = this.type; │ │ │ │ │ - var layers = map.layers; │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = layers[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Google && │ │ │ │ │ - layer.visibility === true && layer.inRange === true) { │ │ │ │ │ - type = layer.type; │ │ │ │ │ - visible = true; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var container = this.mapObject.getDiv(); │ │ │ │ │ - if (visible === true) { │ │ │ │ │ - if (container.parentNode !== map.div) { │ │ │ │ │ - if (!cache.rendered) { │ │ │ │ │ - var me = this; │ │ │ │ │ - google.maps.event.addListenerOnce(this.mapObject, 'tilesloaded', function() { │ │ │ │ │ - cache.rendered = true; │ │ │ │ │ - me.setGMapVisibility(me.getVisibility()); │ │ │ │ │ - me.moveTo(me.map.getCenter()); │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - map.div.appendChild(container); │ │ │ │ │ - cache.googleControl.appendChild(map.viewPortDiv); │ │ │ │ │ - google.maps.event.trigger(this.mapObject, 'resize'); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.mapObject.setMapTypeId(type); │ │ │ │ │ - } else if (cache.googleControl.hasChildNodes()) { │ │ │ │ │ - map.div.appendChild(map.viewPortDiv); │ │ │ │ │ - map.div.removeChild(container); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + drawPoint: function(node, geometry) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getMapContainer │ │ │ │ │ + * Method: drawLineString │ │ │ │ │ + * Virtual function for drawing LineString Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} the GMap container's div │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components of │ │ │ │ │ + * the linestring, or false if nothing could be drawn │ │ │ │ │ */ │ │ │ │ │ - getMapContainer: function() { │ │ │ │ │ - return this.mapObject.getDiv(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ - // │ │ │ │ │ + drawLineString: function(node, geometry) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectBoundsFromOLBounds │ │ │ │ │ + * Method: drawLinearRing │ │ │ │ │ + * Virtual function for drawing LinearRing Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * olBounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} A MapObject Bounds, translated from olBounds │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ + * of the linear ring, or false if nothing could be drawn │ │ │ │ │ */ │ │ │ │ │ - getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ - var moBounds = null; │ │ │ │ │ - if (olBounds != null) { │ │ │ │ │ - var sw = this.sphericalMercator ? │ │ │ │ │ - this.inverseMercator(olBounds.bottom, olBounds.left) : │ │ │ │ │ - new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ - var ne = this.sphericalMercator ? │ │ │ │ │ - this.inverseMercator(olBounds.top, olBounds.right) : │ │ │ │ │ - new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ - moBounds = new google.maps.LatLngBounds( │ │ │ │ │ - new google.maps.LatLng(sw.lat, sw.lon), │ │ │ │ │ - new google.maps.LatLng(ne.lat, ne.lon) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return moBounds; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Interface Controls * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - // LonLat - Pixel Translation │ │ │ │ │ + drawLinearRing: function(node, geometry) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectLonLatFromMapObjectPixel │ │ │ │ │ + * Method: drawPolygon │ │ │ │ │ + * Virtual function for drawing Polygon Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moPixel - {Object} MapObject Pixel format │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} MapObject LonLat translated from MapObject Pixel │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ + * of the polygon, or false if nothing could be drawn │ │ │ │ │ */ │ │ │ │ │ - getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - var size = this.map.getSize(); │ │ │ │ │ - var lon = this.getLongitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ - var lat = this.getLatitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ - var res = this.map.getResolution(); │ │ │ │ │ + drawPolygon: function(node, geometry) {}, │ │ │ │ │ │ │ │ │ │ - var delta_x = moPixel.x - (size.w / 2); │ │ │ │ │ - var delta_y = moPixel.y - (size.h / 2); │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawRectangle │ │ │ │ │ + * Virtual function for drawing Rectangle Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the rectangle │ │ │ │ │ + */ │ │ │ │ │ + drawRectangle: function(node, geometry) {}, │ │ │ │ │ │ │ │ │ │ - var lonlat = new OpenLayers.LonLat( │ │ │ │ │ - lon + delta_x * res, │ │ │ │ │ - lat - delta_y * res │ │ │ │ │ - ); │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawCircle │ │ │ │ │ + * Virtual function for drawing Circle Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the circle │ │ │ │ │ + */ │ │ │ │ │ + drawCircle: function(node, geometry) {}, │ │ │ │ │ │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - lonlat = lonlat.wrapDateLine(this.maxExtent); │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeText │ │ │ │ │ + * Removes a label │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + removeText: function(featureId) { │ │ │ │ │ + var label = document.getElementById(featureId + this.LABEL_ID_SUFFIX); │ │ │ │ │ + if (label) { │ │ │ │ │ + this.textRoot.removeChild(label); │ │ │ │ │ + } │ │ │ │ │ + var outline = document.getElementById(featureId + this.LABEL_OUTLINE_SUFFIX); │ │ │ │ │ + if (outline) { │ │ │ │ │ + this.textRoot.removeChild(outline); │ │ │ │ │ } │ │ │ │ │ - return this.getMapObjectLonLatFromLonLat(lonlat.lon, lonlat.lat); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectPixelFromMapObjectLonLat │ │ │ │ │ + * Method: getFeatureIdFromEvent │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ - * │ │ │ │ │ + * evt - {Object} An <OpenLayers.Event> object │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} MapObject Pixel transtlated from MapObject LonLat │ │ │ │ │ + * {String} A feature id or undefined. │ │ │ │ │ */ │ │ │ │ │ - getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - var res = this.map.getResolution(); │ │ │ │ │ - var extent = this.map.getExtent(); │ │ │ │ │ - return this.getMapObjectPixelFromXY((1 / res * (lon - extent.left)), │ │ │ │ │ - (1 / res * (extent.top - lat))); │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var target = evt.target; │ │ │ │ │ + var useElement = target && target.correspondingUseElement; │ │ │ │ │ + var node = useElement ? useElement : (target || evt.srcElement); │ │ │ │ │ + return node._featureId; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setMapObjectCenter │ │ │ │ │ - * Set the mapObject to the specified center and zoom │ │ │ │ │ + * Method: eraseGeometry │ │ │ │ │ + * Erase a geometry from the renderer. In the case of a multi-geometry, │ │ │ │ │ + * we cycle through and recurse on ourselves. Otherwise, we look for a │ │ │ │ │ + * node with the geometry.id, destroy its geometry, and remove it from │ │ │ │ │ + * the DOM. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * center - {Object} MapObject LonLat format │ │ │ │ │ - * zoom - {int} MapObject zoom format │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ */ │ │ │ │ │ - setMapObjectCenter: function(center, zoom) { │ │ │ │ │ - if (this.animationEnabled === false && zoom != this.mapObject.zoom) { │ │ │ │ │ - var mapContainer = this.getMapContainer(); │ │ │ │ │ - google.maps.event.addListenerOnce( │ │ │ │ │ - this.mapObject, │ │ │ │ │ - "idle", │ │ │ │ │ - function() { │ │ │ │ │ - mapContainer.style.visibility = ""; │ │ │ │ │ + eraseGeometry: function(geometry, featureId) { │ │ │ │ │ + if ((geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ + (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ + (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPolygon") || │ │ │ │ │ + (geometry.CLASS_NAME == "OpenLayers.Geometry.Collection")) { │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + this.eraseGeometry(geometry.components[i], featureId); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + var element = OpenLayers.Util.getElement(geometry.id); │ │ │ │ │ + if (element && element.parentNode) { │ │ │ │ │ + if (element.geometry) { │ │ │ │ │ + element.geometry.destroy(); │ │ │ │ │ + element.geometry = null; │ │ │ │ │ } │ │ │ │ │ - ); │ │ │ │ │ - mapContainer.style.visibility = "hidden"; │ │ │ │ │ - } │ │ │ │ │ - this.mapObject.setOptions({ │ │ │ │ │ - center: center, │ │ │ │ │ - zoom: zoom │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + element.parentNode.removeChild(element); │ │ │ │ │ │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + this.indexer.remove(element); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // Bounds │ │ │ │ │ + if (element._style.backgroundGraphic) { │ │ │ │ │ + var backgroundId = geometry.id + this.BACKGROUND_ID_SUFFIX; │ │ │ │ │ + var bElem = OpenLayers.Util.getElement(backgroundId); │ │ │ │ │ + if (bElem && bElem.parentNode) { │ │ │ │ │ + // No need to destroy the geometry since the element and the background │ │ │ │ │ + // node share the same geometry. │ │ │ │ │ + bElem.parentNode.removeChild(bElem); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectZoomFromMapObjectBounds │ │ │ │ │ + * Method: nodeFactory │ │ │ │ │ + * Create new node of the specified type, with the (optional) specified id. │ │ │ │ │ + * │ │ │ │ │ + * If node already exists with same ID and a different type, we remove it │ │ │ │ │ + * and then call ourselves again to recreate it. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * moBounds - {Object} MapObject Bounds format │ │ │ │ │ + * id - {String} │ │ │ │ │ + * type - {String} type Kind of node to draw. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} MapObject Zoom for specified MapObject Bounds │ │ │ │ │ + * {DOMElement} A new node of the given type and id. │ │ │ │ │ */ │ │ │ │ │ - getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - return this.mapObject.getBoundsZoomLevel(moBounds); │ │ │ │ │ + nodeFactory: function(id, type) { │ │ │ │ │ + var node = OpenLayers.Util.getElement(id); │ │ │ │ │ + if (node) { │ │ │ │ │ + if (!this.nodeTypeCompare(node, type)) { │ │ │ │ │ + node.parentNode.removeChild(node); │ │ │ │ │ + node = this.nodeFactory(id, type); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + node = this.createNode(type, id); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Primitives * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - // LonLat │ │ │ │ │ + /** │ │ │ │ │ + * Method: nodeTypeCompare │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * type - {String} Kind of node │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ + * This function must be overridden by subclasses. │ │ │ │ │ + */ │ │ │ │ │ + nodeTypeCompare: function(node, type) {}, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectLonLatFromLonLat │ │ │ │ │ + /** │ │ │ │ │ + * Method: createNode │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * lon - {Float} │ │ │ │ │ - * lat - {Float} │ │ │ │ │ + * type - {String} Kind of node to draw. │ │ │ │ │ + * id - {String} Id for node. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} MapObject LonLat built from lon and lat params │ │ │ │ │ + * {DOMElement} A new node of the given type and id. │ │ │ │ │ + * This function must be overridden by subclasses. │ │ │ │ │ */ │ │ │ │ │ - getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ - var gLatLng; │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ - gLatLng = new google.maps.LatLng(lonlat.lat, lonlat.lon); │ │ │ │ │ - } else { │ │ │ │ │ - gLatLng = new google.maps.LatLng(lat, lon); │ │ │ │ │ + createNode: function(type, id) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveRoot │ │ │ │ │ + * moves this renderer's root to a different renderer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * renderer - {<OpenLayers.Renderer>} target renderer for the moved root │ │ │ │ │ + */ │ │ │ │ │ + moveRoot: function(renderer) { │ │ │ │ │ + var root = this.root; │ │ │ │ │ + if (renderer.root.parentNode == this.rendererRoot) { │ │ │ │ │ + root = renderer.root; │ │ │ │ │ } │ │ │ │ │ - return gLatLng; │ │ │ │ │ + root.parentNode.removeChild(root); │ │ │ │ │ + renderer.rendererRoot.appendChild(root); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - // Pixel │ │ │ │ │ + /** │ │ │ │ │ + * Method: getRenderLayerId │ │ │ │ │ + * Gets the layer that this renderer's output appears on. If moveRoot was │ │ │ │ │ + * used, this will be different from the id of the layer containing the │ │ │ │ │ + * features rendered by this renderer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} the id of the output layer. │ │ │ │ │ + */ │ │ │ │ │ + getRenderLayerId: function() { │ │ │ │ │ + return this.root.parentNode.parentNode.id; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectPixelFromXY │ │ │ │ │ + * Method: isComplexSymbol │ │ │ │ │ + * Determines if a symbol cannot be rendered using drawCircle │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * x - {Integer} │ │ │ │ │ - * y - {Integer} │ │ │ │ │ + * graphicName - {String} │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject Pixel from x and y parameters │ │ │ │ │ + * Returns │ │ │ │ │ + * {Boolean} true if the symbol is complex, false if not │ │ │ │ │ */ │ │ │ │ │ - getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ - return new google.maps.Point(x, y); │ │ │ │ │ - } │ │ │ │ │ + isComplexSymbol: function(graphicName) { │ │ │ │ │ + return (graphicName != "circle") && !!graphicName; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.Elements" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ -}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Vector/RootContainer.js │ │ │ │ │ + OpenLayers/Renderer/VML.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ + * @requires OpenLayers/Renderer/Elements.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.Vector.RootContainer │ │ │ │ │ - * A special layer type to combine multiple vector layers inside a single │ │ │ │ │ - * renderer root container. This class is not supposed to be instantiated │ │ │ │ │ - * from user space, it is a helper class for controls that require event │ │ │ │ │ - * processing for multiple vector layers. │ │ │ │ │ + * Class: OpenLayers.Renderer.VML │ │ │ │ │ + * Render vector features in browsers with VML capability. Construct a new │ │ │ │ │ + * VML renderer with the <OpenLayers.Renderer.VML> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Note that for all calculations in this class, we use (num | 0) to truncate a │ │ │ │ │ + * float value to an integer. This is done because it seems that VML doesn't │ │ │ │ │ + * support float values. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Vector> │ │ │ │ │ + * - <OpenLayers.Renderer.Elements> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Vector.RootContainer = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: displayInLayerSwitcher │ │ │ │ │ - * Set to false for this layer type │ │ │ │ │ - */ │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ +OpenLayers.Renderer.VML = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: layers │ │ │ │ │ - * Layers that are attached to this container. Required config option. │ │ │ │ │ + * Property: xmlns │ │ │ │ │ + * {String} XML Namespace URN │ │ │ │ │ */ │ │ │ │ │ - layers: null, │ │ │ │ │ + xmlns: "urn:schemas-microsoft-com:vml", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Vector.RootContainer │ │ │ │ │ - * Create a new root container for multiple vector layer. This constructor │ │ │ │ │ - * is not supposed to be used from user space, it is only to be used by │ │ │ │ │ - * controls that need feature selection across multiple vector layers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} A name for the layer │ │ │ │ │ - * options - {Object} Optional object with non-default properties to set on │ │ │ │ │ - * the layer. │ │ │ │ │ - * │ │ │ │ │ - * Required options properties: │ │ │ │ │ - * layers - {Array(<OpenLayers.Layer.Vector>)} The layers managed by this │ │ │ │ │ - * container │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Vector.RootContainer>} A new vector layer root │ │ │ │ │ - * container │ │ │ │ │ + * Property: symbolCache │ │ │ │ │ + * {DOMElement} node holding symbols. This hash is keyed by symbol name, │ │ │ │ │ + * and each value is a hash with a "path" and an "extent" property. │ │ │ │ │ */ │ │ │ │ │ + symbolCache: {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: display │ │ │ │ │ + * Property: offset │ │ │ │ │ + * {Object} Hash with "x" and "y" properties │ │ │ │ │ */ │ │ │ │ │ - display: function() {}, │ │ │ │ │ + offset: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureFromEvent │ │ │ │ │ - * walk through the layers to find the feature returned by the event │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Renderer.VML │ │ │ │ │ + * Create a new VML renderer. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} event object with a feature property │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * containerID - {String} The id for the element that contains the renderer │ │ │ │ │ */ │ │ │ │ │ - getFeatureFromEvent: function(evt) { │ │ │ │ │ - var layers = this.layers; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0; i < layers.length; i++) { │ │ │ │ │ - feature = layers[i].getFeatureFromEvent(evt); │ │ │ │ │ - if (feature) { │ │ │ │ │ - return feature; │ │ │ │ │ + initialize: function(containerID) { │ │ │ │ │ + if (!this.supported()) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (!document.namespaces.olv) { │ │ │ │ │ + document.namespaces.add("olv", this.xmlns); │ │ │ │ │ + var style = document.createStyleSheet(); │ │ │ │ │ + var shapes = ['shape', 'rect', 'oval', 'fill', 'stroke', 'imagedata', 'group', 'textbox']; │ │ │ │ │ + for (var i = 0, len = shapes.length; i < len; i++) { │ │ │ │ │ + │ │ │ │ │ + style.addRule('olv\\:' + shapes[i], "behavior: url(#default#VML); " + │ │ │ │ │ + "position: absolute; display: inline-block;"); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.initialize.apply(this, │ │ │ │ │ + arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * APIMethod: supported │ │ │ │ │ + * Determine whether a browser supports this renderer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The browser supports the VML renderer │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.collectRoots(); │ │ │ │ │ - map.events.register("changelayer", this, this.handleChangeLayer); │ │ │ │ │ + supported: function() { │ │ │ │ │ + return !!(document.namespaces); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removeMap │ │ │ │ │ - * │ │ │ │ │ + * Method: setExtent │ │ │ │ │ + * Set the renderer's extent │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} │ │ │ │ │ + * resolutionChanged - {Boolean} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ + * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - map.events.unregister("changelayer", this, this.handleChangeLayer); │ │ │ │ │ - this.resetRoots(); │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + │ │ │ │ │ + var left = (extent.left / resolution) | 0; │ │ │ │ │ + var top = (extent.top / resolution - this.size.h) | 0; │ │ │ │ │ + if (resolutionChanged || !this.offset) { │ │ │ │ │ + this.offset = { │ │ │ │ │ + x: left, │ │ │ │ │ + y: top │ │ │ │ │ + }; │ │ │ │ │ + left = 0; │ │ │ │ │ + top = 0; │ │ │ │ │ + } else { │ │ │ │ │ + left = left - this.offset.x; │ │ │ │ │ + top = top - this.offset.y; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + var org = (left - this.xOffset) + " " + top; │ │ │ │ │ + this.root.coordorigin = org; │ │ │ │ │ + var roots = [this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ + var root; │ │ │ │ │ + for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ + root = roots[i]; │ │ │ │ │ + │ │ │ │ │ + var size = this.size.w + " " + this.size.h; │ │ │ │ │ + root.coordsize = size; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: collectRoots │ │ │ │ │ - * Collects the root nodes of all layers this control is configured with │ │ │ │ │ - * and moveswien the nodes to this control's layer │ │ │ │ │ - */ │ │ │ │ │ - collectRoots: function() { │ │ │ │ │ - var layer; │ │ │ │ │ - // walk through all map layers, because we want to keep the order │ │ │ │ │ - for (var i = 0; i < this.map.layers.length; ++i) { │ │ │ │ │ - layer = this.map.layers[i]; │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ - layer.renderer.moveRoot(this.renderer); │ │ │ │ │ - } │ │ │ │ │ } │ │ │ │ │ + // flip the VML display Y axis upside down so it │ │ │ │ │ + // matches the display Y axis of the map │ │ │ │ │ + this.root.style.flip = "y"; │ │ │ │ │ + │ │ │ │ │ + return coordSysUnchanged; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: resetRoots │ │ │ │ │ - * Resets the root nodes back into the layers they belong to. │ │ │ │ │ + * Method: setSize │ │ │ │ │ + * Set the size of the drawing surface │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * size - {<OpenLayers.Size>} the size of the drawing surface │ │ │ │ │ */ │ │ │ │ │ - resetRoots: function() { │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = 0; i < this.layers.length; ++i) { │ │ │ │ │ - layer = this.layers[i]; │ │ │ │ │ - if (this.renderer && layer.renderer.getRenderLayerId() == this.id) { │ │ │ │ │ - this.renderer.moveRoot(layer.renderer); │ │ │ │ │ - } │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + // setting width and height on all roots to avoid flicker which we │ │ │ │ │ + // would get with 100% width and height on child roots │ │ │ │ │ + var roots = [ │ │ │ │ │ + this.rendererRoot, │ │ │ │ │ + this.root, │ │ │ │ │ + this.vectorRoot, │ │ │ │ │ + this.textRoot │ │ │ │ │ + ]; │ │ │ │ │ + var w = this.size.w + "px"; │ │ │ │ │ + var h = this.size.h + "px"; │ │ │ │ │ + var root; │ │ │ │ │ + for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ + root = roots[i]; │ │ │ │ │ + root.style.width = w; │ │ │ │ │ + root.style.height = h; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleChangeLayer │ │ │ │ │ - * Event handler for the map's changelayer event. We need to rebuild │ │ │ │ │ - * this container's layer dom if order of one of its layers changes. │ │ │ │ │ - * This handler is added with the setMap method, and removed with the │ │ │ │ │ - * removeMap method. │ │ │ │ │ - * │ │ │ │ │ + * Method: getNodeType │ │ │ │ │ + * Get the node type for a geometry and style │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The corresponding node type for the specified geometry │ │ │ │ │ */ │ │ │ │ │ - handleChangeLayer: function(evt) { │ │ │ │ │ - var layer = evt.layer; │ │ │ │ │ - if (evt.property == "order" && │ │ │ │ │ - OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ - this.resetRoots(); │ │ │ │ │ - this.collectRoots(); │ │ │ │ │ + getNodeType: function(geometry, style) { │ │ │ │ │ + var nodeType = null; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + nodeType = "olv:rect"; │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + nodeType = "olv:shape"; │ │ │ │ │ + } else { │ │ │ │ │ + nodeType = "olv:oval"; │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + nodeType = "olv:rect"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + case "OpenLayers.Geometry.Curve": │ │ │ │ │ + nodeType = "olv:shape"; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ + return nodeType; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Vector.RootContainer" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Popup/Anchored.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: setStyle │ │ │ │ │ + * Use to set all the style attributes to a VML node. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} An VML element to decorate │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * options - {Object} Currently supported options include │ │ │ │ │ + * 'isFilled' {Boolean} and │ │ │ │ │ + * 'isStroked' {Boolean} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + */ │ │ │ │ │ + setStyle: function(node, style, options, geometry) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + options = options || node._options; │ │ │ │ │ + var fillColor = style.fillColor; │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + node.title = title; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ + if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + options.isFilled = true; │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Popup.js │ │ │ │ │ - */ │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ + style.graphicXOffset : -(0.5 * width); │ │ │ │ │ + var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ + style.graphicYOffset : -(0.5 * height); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Popup.Anchored │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Popup> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Popup.Anchored = │ │ │ │ │ - OpenLayers.Class(OpenLayers.Popup, { │ │ │ │ │ + node.style.left = ((((geometry.x - this.featureDx) / resolution - this.offset.x) + xOffset) | 0) + "px"; │ │ │ │ │ + node.style.top = (((geometry.y / resolution - this.offset.y) - (yOffset + height)) | 0) + "px"; │ │ │ │ │ + node.style.width = width + "px"; │ │ │ │ │ + node.style.height = height + "px"; │ │ │ │ │ + node.style.flip = "y"; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: relativePosition │ │ │ │ │ - * {String} Relative position of the popup ("br", "tr", "tl" or "bl"). │ │ │ │ │ - */ │ │ │ │ │ - relativePosition: null, │ │ │ │ │ + // modify fillColor and options for stroke styling below │ │ │ │ │ + fillColor = "none"; │ │ │ │ │ + options.isStroked = false; │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + var cache = this.importSymbol(style.graphicName); │ │ │ │ │ + node.path = cache.path; │ │ │ │ │ + node.coordorigin = cache.left + "," + cache.bottom; │ │ │ │ │ + var size = cache.size; │ │ │ │ │ + node.coordsize = size + "," + size; │ │ │ │ │ + this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ + node.style.flip = "y"; │ │ │ │ │ + } else { │ │ │ │ │ + this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: keepInMap │ │ │ │ │ - * {Boolean} If panMapIfOutOfView is false, and this property is true, │ │ │ │ │ - * contrain the popup such that it always fits in the available map │ │ │ │ │ - * space. By default, this is set. If you are creating popups that are │ │ │ │ │ - * near map edges and not allowing pannning, and especially if you have │ │ │ │ │ - * a popup which has a fixedRelativePosition, setting this to false may │ │ │ │ │ - * be a smart thing to do. │ │ │ │ │ - * │ │ │ │ │ - * For anchored popups, default is true, since subclasses will │ │ │ │ │ - * usually want this functionality. │ │ │ │ │ - */ │ │ │ │ │ - keepInMap: true, │ │ │ │ │ + // fill │ │ │ │ │ + if (options.isFilled) { │ │ │ │ │ + node.fillcolor = fillColor; │ │ │ │ │ + } else { │ │ │ │ │ + node.filled = "false"; │ │ │ │ │ + } │ │ │ │ │ + var fills = node.getElementsByTagName("fill"); │ │ │ │ │ + var fill = (fills.length == 0) ? null : fills[0]; │ │ │ │ │ + if (!options.isFilled) { │ │ │ │ │ + if (fill) { │ │ │ │ │ + node.removeChild(fill); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + if (!fill) { │ │ │ │ │ + fill = this.createNode('olv:fill', node.id + "_fill"); │ │ │ │ │ + } │ │ │ │ │ + fill.opacity = style.fillOpacity; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: anchor │ │ │ │ │ - * {Object} Object to which we'll anchor the popup. Must expose a │ │ │ │ │ - * 'size' (<OpenLayers.Size>) and 'offset' (<OpenLayers.Pixel>). │ │ │ │ │ - */ │ │ │ │ │ - anchor: null, │ │ │ │ │ + if (node._geometryClass === "OpenLayers.Geometry.Point" && │ │ │ │ │ + style.externalGraphic) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Popup.Anchored │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ - * contentSize - {<OpenLayers.Size>} │ │ │ │ │ - * contentHTML - {String} │ │ │ │ │ - * anchor - {Object} Object which must expose a 'size' <OpenLayers.Size> │ │ │ │ │ - * and 'offset' <OpenLayers.Pixel> (generally an <OpenLayers.Icon>). │ │ │ │ │ - * closeBox - {Boolean} │ │ │ │ │ - * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, │ │ │ │ │ - closeBoxCallback) { │ │ │ │ │ - var newArguments = [ │ │ │ │ │ - id, lonlat, contentSize, contentHTML, closeBox, closeBoxCallback │ │ │ │ │ - ]; │ │ │ │ │ - OpenLayers.Popup.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + // override fillOpacity │ │ │ │ │ + if (style.graphicOpacity) { │ │ │ │ │ + fill.opacity = style.graphicOpacity; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - this.anchor = (anchor != null) ? anchor : │ │ │ │ │ - { │ │ │ │ │ - size: new OpenLayers.Size(0, 0), │ │ │ │ │ - offset: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ + fill.src = style.externalGraphic; │ │ │ │ │ + fill.type = "frame"; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.anchor = null; │ │ │ │ │ - this.relativePosition = null; │ │ │ │ │ + if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ + fill.aspect = "atmost"; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (fill.parentNode != node) { │ │ │ │ │ + node.appendChild(fill); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - OpenLayers.Popup.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + // additional rendering for rotated graphics or symbols │ │ │ │ │ + var rotation = style.rotation; │ │ │ │ │ + if ((rotation !== undefined || node._rotation !== undefined)) { │ │ │ │ │ + node._rotation = rotation; │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ + // make the fill fully transparent, because we now have │ │ │ │ │ + // the graphic as imagedata element. We cannot just remove │ │ │ │ │ + // the fill, because this is part of the hack described │ │ │ │ │ + // in graphicRotate │ │ │ │ │ + fill.opacity = 0; │ │ │ │ │ + } else if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ + node.style.rotation = rotation || 0; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: show │ │ │ │ │ - * Overridden from Popup since user might hide popup and then show() it │ │ │ │ │ - * in a new location (meaning we might want to update the relative │ │ │ │ │ - * position on the show) │ │ │ │ │ - */ │ │ │ │ │ - show: function() { │ │ │ │ │ - this.updatePosition(); │ │ │ │ │ - OpenLayers.Popup.prototype.show.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + // stroke │ │ │ │ │ + var strokes = node.getElementsByTagName("stroke"); │ │ │ │ │ + var stroke = (strokes.length == 0) ? null : strokes[0]; │ │ │ │ │ + if (!options.isStroked) { │ │ │ │ │ + node.stroked = false; │ │ │ │ │ + if (stroke) { │ │ │ │ │ + stroke.on = false; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + if (!stroke) { │ │ │ │ │ + stroke = this.createNode('olv:stroke', node.id + "_stroke"); │ │ │ │ │ + node.appendChild(stroke); │ │ │ │ │ + } │ │ │ │ │ + stroke.on = true; │ │ │ │ │ + stroke.color = style.strokeColor; │ │ │ │ │ + stroke.weight = style.strokeWidth + "px"; │ │ │ │ │ + stroke.opacity = style.strokeOpacity; │ │ │ │ │ + stroke.endcap = style.strokeLinecap == 'butt' ? 'flat' : │ │ │ │ │ + (style.strokeLinecap || 'round'); │ │ │ │ │ + if (style.strokeDashstyle) { │ │ │ │ │ + stroke.dashstyle = this.dashStyle(style); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * Since the popup is moving to a new px, it might need also to be moved │ │ │ │ │ - * relative to where the marker is. We first calculate the new │ │ │ │ │ - * relativePosition, and then we calculate the new px where we will │ │ │ │ │ - * put the popup, based on the new relative position. │ │ │ │ │ - * │ │ │ │ │ - * If the relativePosition has changed, we must also call │ │ │ │ │ - * updateRelativePosition() to make any visual changes to the popup │ │ │ │ │ - * which are associated with putting it in a new relativePosition. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - */ │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - var oldRelativePosition = this.relativePosition; │ │ │ │ │ - this.relativePosition = this.calculateRelativePosition(px); │ │ │ │ │ + if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ + node.style.cursor = style.cursor; │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - OpenLayers.Popup.prototype.moveTo.call(this, this.calculateNewPx(px)); │ │ │ │ │ + /** │ │ │ │ │ + * Method: graphicRotate │ │ │ │ │ + * If a point is to be styled with externalGraphic and rotation, VML fills │ │ │ │ │ + * cannot be used to display the graphic, because rotation of graphic │ │ │ │ │ + * fills is not supported by the VML implementation of Internet Explorer. │ │ │ │ │ + * This method creates a olv:imagedata element inside the VML node, │ │ │ │ │ + * DXImageTransform.Matrix and BasicImage filters for rotation and │ │ │ │ │ + * opacity, and a 3-step hack to remove rendering artefacts from the │ │ │ │ │ + * graphic and preserve the ability of graphics to trigger events. │ │ │ │ │ + * Finally, OpenLayers methods are used to determine the correct │ │ │ │ │ + * insertion point of the rotated image, because DXImageTransform.Matrix │ │ │ │ │ + * does the rotation without the ability to specify a rotation center │ │ │ │ │ + * point. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * xOffset - {Number} rotation center relative to image, x coordinate │ │ │ │ │ + * yOffset - {Number} rotation center relative to image, y coordinate │ │ │ │ │ + * style - {Object} │ │ │ │ │ + */ │ │ │ │ │ + graphicRotate: function(node, xOffset, yOffset, style) { │ │ │ │ │ + var style = style || node._style; │ │ │ │ │ + var rotation = style.rotation || 0; │ │ │ │ │ │ │ │ │ │ - //if this move has caused the popup to change its relative position, │ │ │ │ │ - // we need to make the appropriate cosmetic changes. │ │ │ │ │ - if (this.relativePosition != oldRelativePosition) { │ │ │ │ │ - this.updateRelativePosition(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + var aspectRatio, size; │ │ │ │ │ + if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ + // load the image to determine its size │ │ │ │ │ + var img = new Image(); │ │ │ │ │ + img.onreadystatechange = OpenLayers.Function.bind(function() { │ │ │ │ │ + if (img.readyState == "complete" || │ │ │ │ │ + img.readyState == "interactive") { │ │ │ │ │ + aspectRatio = img.width / img.height; │ │ │ │ │ + size = Math.max(style.pointRadius * 2, │ │ │ │ │ + style.graphicWidth || 0, │ │ │ │ │ + style.graphicHeight || 0); │ │ │ │ │ + xOffset = xOffset * aspectRatio; │ │ │ │ │ + style.graphicWidth = size * aspectRatio; │ │ │ │ │ + style.graphicHeight = size; │ │ │ │ │ + this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ + } │ │ │ │ │ + }, this); │ │ │ │ │ + img.src = style.externalGraphic; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setSize │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * contentSize - {<OpenLayers.Size>} the new size for the popup's │ │ │ │ │ - * contents div (in pixels). │ │ │ │ │ - */ │ │ │ │ │ - setSize: function(contentSize) { │ │ │ │ │ - OpenLayers.Popup.prototype.setSize.apply(this, arguments); │ │ │ │ │ + // will be called again by the onreadystate handler │ │ │ │ │ + return; │ │ │ │ │ + } else { │ │ │ │ │ + size = Math.max(style.graphicWidth, style.graphicHeight); │ │ │ │ │ + aspectRatio = style.graphicWidth / style.graphicHeight; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if ((this.lonlat) && (this.map)) { │ │ │ │ │ - var px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ - this.moveTo(px); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + var width = Math.round(style.graphicWidth || size * aspectRatio); │ │ │ │ │ + var height = Math.round(style.graphicHeight || size); │ │ │ │ │ + node.style.width = width + "px"; │ │ │ │ │ + node.style.height = height + "px"; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateRelativePosition │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The relative position ("br" "tr" "tl" "bl") at which the popup │ │ │ │ │ - * should be placed. │ │ │ │ │ - */ │ │ │ │ │ - calculateRelativePosition: function(px) { │ │ │ │ │ - var lonlat = this.map.getLonLatFromLayerPx(px); │ │ │ │ │ + // Three steps are required to remove artefacts for images with │ │ │ │ │ + // transparent backgrounds (resulting from using DXImageTransform │ │ │ │ │ + // filters on svg objects), while preserving awareness for browser │ │ │ │ │ + // events on images: │ │ │ │ │ + // - Use the fill as usual (like for unrotated images) to handle │ │ │ │ │ + // events │ │ │ │ │ + // - specify an imagedata element with the same src as the fill │ │ │ │ │ + // - style the imagedata element with an AlphaImageLoader filter │ │ │ │ │ + // with empty src │ │ │ │ │ + var image = document.getElementById(node.id + "_image"); │ │ │ │ │ + if (!image) { │ │ │ │ │ + image = this.createNode("olv:imagedata", node.id + "_image"); │ │ │ │ │ + node.appendChild(image); │ │ │ │ │ + } │ │ │ │ │ + image.style.width = width + "px"; │ │ │ │ │ + image.style.height = height + "px"; │ │ │ │ │ + image.src = style.externalGraphic; │ │ │ │ │ + image.style.filter = │ │ │ │ │ + "progid:DXImageTransform.Microsoft.AlphaImageLoader(" + │ │ │ │ │ + "src='', sizingMethod='scale')"; │ │ │ │ │ │ │ │ │ │ - var extent = this.map.getExtent(); │ │ │ │ │ - var quadrant = extent.determineQuadrant(lonlat); │ │ │ │ │ + var rot = rotation * Math.PI / 180; │ │ │ │ │ + var sintheta = Math.sin(rot); │ │ │ │ │ + var costheta = Math.cos(rot); │ │ │ │ │ │ │ │ │ │ - return OpenLayers.Bounds.oppositeQuadrant(quadrant); │ │ │ │ │ - }, │ │ │ │ │ + // do the rotation on the image │ │ │ │ │ + var filter = │ │ │ │ │ + "progid:DXImageTransform.Microsoft.Matrix(M11=" + costheta + │ │ │ │ │ + ",M12=" + (-sintheta) + ",M21=" + sintheta + ",M22=" + costheta + │ │ │ │ │ + ",SizingMethod='auto expand')\n"; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateRelativePosition │ │ │ │ │ - * The popup has been moved to a new relative location, so we may want to │ │ │ │ │ - * make some cosmetic adjustments to it. │ │ │ │ │ - * │ │ │ │ │ - * Note that in the classic Anchored popup, there is nothing to do │ │ │ │ │ - * here, since the popup looks exactly the same in all four positions. │ │ │ │ │ - * Subclasses such as Framed, however, will want to do something │ │ │ │ │ - * special here. │ │ │ │ │ - */ │ │ │ │ │ - updateRelativePosition: function() { │ │ │ │ │ - //to be overridden by subclasses │ │ │ │ │ - }, │ │ │ │ │ + // set the opacity (needed for the imagedata) │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ + if (opacity && opacity != 1) { │ │ │ │ │ + filter += │ │ │ │ │ + "progid:DXImageTransform.Microsoft.BasicImage(opacity=" + │ │ │ │ │ + opacity + ")\n"; │ │ │ │ │ + } │ │ │ │ │ + node.style.filter = filter; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateNewPx │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Pixel>} The the new px position of the popup on the screen │ │ │ │ │ - * relative to the passed-in px. │ │ │ │ │ - */ │ │ │ │ │ - calculateNewPx: function(px) { │ │ │ │ │ - var newPx = px.offset(this.anchor.offset); │ │ │ │ │ + // do the rotation again on a box, so we know the insertion point │ │ │ │ │ + var centerPoint = new OpenLayers.Geometry.Point(-xOffset, -yOffset); │ │ │ │ │ + var imgBox = new OpenLayers.Bounds(0, 0, width, height).toGeometry(); │ │ │ │ │ + imgBox.rotate(style.rotation, centerPoint); │ │ │ │ │ + var imgBounds = imgBox.getBounds(); │ │ │ │ │ │ │ │ │ │ - //use contentSize if size is not already set │ │ │ │ │ - var size = this.size || this.contentSize; │ │ │ │ │ + node.style.left = Math.round( │ │ │ │ │ + parseInt(node.style.left) + imgBounds.left) + "px"; │ │ │ │ │ + node.style.top = Math.round( │ │ │ │ │ + parseInt(node.style.top) - imgBounds.bottom) + "px"; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var top = (this.relativePosition.charAt(0) == 't'); │ │ │ │ │ - newPx.y += (top) ? -size.h : this.anchor.size.h; │ │ │ │ │ + /** │ │ │ │ │ + * Method: postDraw │ │ │ │ │ + * Does some node postprocessing to work around browser issues: │ │ │ │ │ + * - Some versions of Internet Explorer seem to be unable to set fillcolor │ │ │ │ │ + * and strokecolor to "none" correctly before the fill node is appended │ │ │ │ │ + * to a visible vml node. This method takes care of that and sets │ │ │ │ │ + * fillcolor and strokecolor again if needed. │ │ │ │ │ + * - In some cases, a node won't become visible after being drawn. Setting │ │ │ │ │ + * style.visibility to "visible" works around that. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + postDraw: function(node) { │ │ │ │ │ + node.style.visibility = "visible"; │ │ │ │ │ + var fillColor = node._style.fillColor; │ │ │ │ │ + var strokeColor = node._style.strokeColor; │ │ │ │ │ + if (fillColor == "none" && │ │ │ │ │ + node.fillcolor != fillColor) { │ │ │ │ │ + node.fillcolor = fillColor; │ │ │ │ │ + } │ │ │ │ │ + if (strokeColor == "none" && │ │ │ │ │ + node.strokecolor != strokeColor) { │ │ │ │ │ + node.strokecolor = strokeColor; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var left = (this.relativePosition.charAt(1) == 'l'); │ │ │ │ │ - newPx.x += (left) ? -size.w : this.anchor.size.w; │ │ │ │ │ │ │ │ │ │ - return newPx; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: setNodeDimension │ │ │ │ │ + * Get the geometry's bounds, convert it to our vml coordinate system, │ │ │ │ │ + * then set the node's position, size, and local coordinate system. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + */ │ │ │ │ │ + setNodeDimension: function(node, geometry) { │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup.Anchored" │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Popup/Framed.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + var bbox = geometry.getBounds(); │ │ │ │ │ + if (bbox) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + var scaledBox = │ │ │ │ │ + new OpenLayers.Bounds(((bbox.left - this.featureDx) / resolution - this.offset.x) | 0, │ │ │ │ │ + (bbox.bottom / resolution - this.offset.y) | 0, │ │ │ │ │ + ((bbox.right - this.featureDx) / resolution - this.offset.x) | 0, │ │ │ │ │ + (bbox.top / resolution - this.offset.y) | 0); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Popup/Anchored.js │ │ │ │ │ - */ │ │ │ │ │ + // Set the internal coordinate system to draw the path │ │ │ │ │ + node.style.left = scaledBox.left + "px"; │ │ │ │ │ + node.style.top = scaledBox.top + "px"; │ │ │ │ │ + node.style.width = scaledBox.getWidth() + "px"; │ │ │ │ │ + node.style.height = scaledBox.getHeight() + "px"; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Popup.Framed │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Popup.Anchored> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Popup.Framed = │ │ │ │ │ - OpenLayers.Class(OpenLayers.Popup.Anchored, { │ │ │ │ │ + node.coordorigin = scaledBox.left + " " + scaledBox.top; │ │ │ │ │ + node.coordsize = scaledBox.getWidth() + " " + scaledBox.getHeight(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: imageSrc │ │ │ │ │ - * {String} location of the image to be used as the popup frame │ │ │ │ │ - */ │ │ │ │ │ - imageSrc: null, │ │ │ │ │ + /** │ │ │ │ │ + * Method: dashStyle │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A VML compliant 'stroke-dasharray' value │ │ │ │ │ + */ │ │ │ │ │ + dashStyle: function(style) { │ │ │ │ │ + var dash = style.strokeDashstyle; │ │ │ │ │ + switch (dash) { │ │ │ │ │ + case 'solid': │ │ │ │ │ + case 'dot': │ │ │ │ │ + case 'dash': │ │ │ │ │ + case 'dashdot': │ │ │ │ │ + case 'longdash': │ │ │ │ │ + case 'longdashdot': │ │ │ │ │ + return dash; │ │ │ │ │ + default: │ │ │ │ │ + // very basic guessing of dash style patterns │ │ │ │ │ + var parts = dash.split(/[ ,]/); │ │ │ │ │ + if (parts.length == 2) { │ │ │ │ │ + if (1 * parts[0] >= 2 * parts[1]) { │ │ │ │ │ + return "longdash"; │ │ │ │ │ + } │ │ │ │ │ + return (parts[0] == 1 || parts[1] == 1) ? "dot" : "dash"; │ │ │ │ │ + } else if (parts.length == 4) { │ │ │ │ │ + return (1 * parts[0] >= 2 * parts[1]) ? "longdashdot" : │ │ │ │ │ + "dashdot"; │ │ │ │ │ + } │ │ │ │ │ + return "solid"; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: imageSize │ │ │ │ │ - * {<OpenLayers.Size>} Size (measured in pixels) of the image located │ │ │ │ │ - * by the 'imageSrc' property. │ │ │ │ │ - */ │ │ │ │ │ - imageSize: null, │ │ │ │ │ + /** │ │ │ │ │ + * Method: createNode │ │ │ │ │ + * Create a new node │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * type - {String} Kind of node to draw │ │ │ │ │ + * id - {String} Id for node │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A new node of the given type and id │ │ │ │ │ + */ │ │ │ │ │ + createNode: function(type, id) { │ │ │ │ │ + var node = document.createElement(type); │ │ │ │ │ + if (id) { │ │ │ │ │ + node.id = id; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isAlphaImage │ │ │ │ │ - * {Boolean} The image has some alpha and thus needs to use the alpha │ │ │ │ │ - * image hack. Note that setting this to true will have no noticeable │ │ │ │ │ - * effect in FF or IE7 browsers, but will all but crush the ie6 │ │ │ │ │ - * browser. │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - isAlphaImage: false, │ │ │ │ │ + // IE hack to make elements unselectable, to prevent 'blue flash' │ │ │ │ │ + // while dragging vectors; #1410 │ │ │ │ │ + node.unselectable = 'on'; │ │ │ │ │ + node.onselectstart = OpenLayers.Function.False; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: positionBlocks │ │ │ │ │ - * {Object} Hash of different position blocks (Object/Hashs). Each block │ │ │ │ │ - * will be keyed by a two-character 'relativePosition' │ │ │ │ │ - * code string (ie "tl", "tr", "bl", "br"). Block properties are │ │ │ │ │ - * 'offset', 'padding' (self-explanatory), and finally the 'blocks' │ │ │ │ │ - * parameter, which is an array of the block objects. │ │ │ │ │ - * │ │ │ │ │ - * Each block object must have 'size', 'anchor', and 'position' │ │ │ │ │ - * properties. │ │ │ │ │ - * │ │ │ │ │ - * Note that positionBlocks should never be modified at runtime. │ │ │ │ │ - */ │ │ │ │ │ - positionBlocks: null, │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: blocks │ │ │ │ │ - * {Array[Object]} Array of objects, each of which is one "block" of the │ │ │ │ │ - * popup. Each block has a 'div' and an 'image' property, both of │ │ │ │ │ - * which are DOMElements, and the latter of which is appended to the │ │ │ │ │ - * former. These are reused as the popup goes changing positions for │ │ │ │ │ - * great economy and elegance. │ │ │ │ │ - */ │ │ │ │ │ - blocks: null, │ │ │ │ │ + /** │ │ │ │ │ + * Method: nodeTypeCompare │ │ │ │ │ + * Determine whether a node is of a given type │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} An VML element │ │ │ │ │ + * type - {String} Kind of node │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ + */ │ │ │ │ │ + nodeTypeCompare: function(node, type) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: fixedRelativePosition │ │ │ │ │ - * {Boolean} We want the framed popup to work dynamically placed relative │ │ │ │ │ - * to its anchor but also in just one fixed position. A well designed │ │ │ │ │ - * framed popup will have the pixels and logic to display itself in │ │ │ │ │ - * any of the four relative positions, but (understandably), this will │ │ │ │ │ - * not be the case for all of them. By setting this property to 'true', │ │ │ │ │ - * framed popup will not recalculate for the best placement each time │ │ │ │ │ - * it's open, but will always open the same way. │ │ │ │ │ - * Note that if this is set to true, it is generally advisable to also │ │ │ │ │ - * set the 'panIntoView' property to true so that the popup can be │ │ │ │ │ - * scrolled into view (since it will often be offscreen on open) │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - fixedRelativePosition: false, │ │ │ │ │ + //split type │ │ │ │ │ + var subType = type; │ │ │ │ │ + var splitIndex = subType.indexOf(":"); │ │ │ │ │ + if (splitIndex != -1) { │ │ │ │ │ + subType = subType.substr(splitIndex + 1); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Popup.Framed │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ - * contentSize - {<OpenLayers.Size>} │ │ │ │ │ - * contentHTML - {String} │ │ │ │ │ - * anchor - {Object} Object to which we'll anchor the popup. Must expose │ │ │ │ │ - * a 'size' (<OpenLayers.Size>) and 'offset' (<OpenLayers.Pixel>) │ │ │ │ │ - * (Note that this is generally an <OpenLayers.Icon>). │ │ │ │ │ - * closeBox - {Boolean} │ │ │ │ │ - * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, │ │ │ │ │ - closeBoxCallback) { │ │ │ │ │ + //split nodeName │ │ │ │ │ + var nodeName = node.nodeName; │ │ │ │ │ + splitIndex = nodeName.indexOf(":"); │ │ │ │ │ + if (splitIndex != -1) { │ │ │ │ │ + nodeName = nodeName.substr(splitIndex + 1); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.initialize.apply(this, arguments); │ │ │ │ │ + return (subType == nodeName); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (this.fixedRelativePosition) { │ │ │ │ │ - //based on our decided relativePostion, set the current padding │ │ │ │ │ - // this keeps us from getting into trouble │ │ │ │ │ - this.updateRelativePosition(); │ │ │ │ │ + /** │ │ │ │ │ + * Method: createRenderRoot │ │ │ │ │ + * Create the renderer root │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The specific render engine's root element │ │ │ │ │ + */ │ │ │ │ │ + createRenderRoot: function() { │ │ │ │ │ + return this.nodeFactory(this.container.id + "_vmlRoot", "div"); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - //make calculateRelativePosition always return the specified │ │ │ │ │ - // fixed position. │ │ │ │ │ - this.calculateRelativePosition = function(px) { │ │ │ │ │ - return this.relativePosition; │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: createRoot │ │ │ │ │ + * Create the main root element │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * suffix - {String} suffix to append to the id │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + createRoot: function(suffix) { │ │ │ │ │ + return this.nodeFactory(this.container.id + suffix, "olv:group"); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.contentDiv.style.position = "absolute"; │ │ │ │ │ - this.contentDiv.style.zIndex = 1; │ │ │ │ │ + /************************************** │ │ │ │ │ + * * │ │ │ │ │ + * GEOMETRY DRAWING FUNCTIONS * │ │ │ │ │ + * * │ │ │ │ │ + **************************************/ │ │ │ │ │ │ │ │ │ │ - if (closeBox) { │ │ │ │ │ - this.closeDiv.style.zIndex = 1; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawPoint │ │ │ │ │ + * Render a point │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the point could not be drawn │ │ │ │ │ + */ │ │ │ │ │ + drawPoint: function(node, geometry) { │ │ │ │ │ + return this.drawCircle(node, geometry, 1); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.groupDiv.style.position = "absolute"; │ │ │ │ │ - this.groupDiv.style.top = "0px"; │ │ │ │ │ - this.groupDiv.style.left = "0px"; │ │ │ │ │ - this.groupDiv.style.height = "100%"; │ │ │ │ │ - this.groupDiv.style.width = "100%"; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawCircle │ │ │ │ │ + * Render a circle. │ │ │ │ │ + * Size and Center a circle given geometry (x,y center) and radius │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * radius - {float} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the circle could not ne drawn │ │ │ │ │ + */ │ │ │ │ │ + drawCircle: function(node, geometry, radius) { │ │ │ │ │ + if (!isNaN(geometry.x) && !isNaN(geometry.y)) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.imageSrc = null; │ │ │ │ │ - this.imageSize = null; │ │ │ │ │ - this.isAlphaImage = null; │ │ │ │ │ + node.style.left = ((((geometry.x - this.featureDx) / resolution - this.offset.x) | 0) - radius) + "px"; │ │ │ │ │ + node.style.top = (((geometry.y / resolution - this.offset.y) | 0) - radius) + "px"; │ │ │ │ │ │ │ │ │ │ - this.fixedRelativePosition = false; │ │ │ │ │ - this.positionBlocks = null; │ │ │ │ │ + var diameter = radius * 2; │ │ │ │ │ │ │ │ │ │ - //remove our blocks │ │ │ │ │ - for (var i = 0; i < this.blocks.length; i++) { │ │ │ │ │ - var block = this.blocks[i]; │ │ │ │ │ + node.style.width = diameter + "px"; │ │ │ │ │ + node.style.height = diameter + "px"; │ │ │ │ │ + return node; │ │ │ │ │ + } │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (block.image) { │ │ │ │ │ - block.div.removeChild(block.image); │ │ │ │ │ - } │ │ │ │ │ - block.image = null; │ │ │ │ │ │ │ │ │ │ - if (block.div) { │ │ │ │ │ - this.groupDiv.removeChild(block.div); │ │ │ │ │ - } │ │ │ │ │ - block.div = null; │ │ │ │ │ - } │ │ │ │ │ - this.blocks = null; │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setBackgroundColor │ │ │ │ │ - */ │ │ │ │ │ - setBackgroundColor: function(color) { │ │ │ │ │ - //does nothing since the framed popup's entire scheme is based on a │ │ │ │ │ - // an image -- changing the background color makes no sense. │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawLineString │ │ │ │ │ + * Render a linestring. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + drawLineString: function(node, geometry) { │ │ │ │ │ + return this.drawLine(node, geometry, false); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setBorder │ │ │ │ │ - */ │ │ │ │ │ - setBorder: function() { │ │ │ │ │ - //does nothing since the framed popup's entire scheme is based on a │ │ │ │ │ - // an image -- changing the popup's border makes no sense. │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawLinearRing │ │ │ │ │ + * Render a linearring │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + drawLinearRing: function(node, geometry) { │ │ │ │ │ + return this.drawLine(node, geometry, true); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setOpacity │ │ │ │ │ - * Sets the opacity of the popup. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * opacity - {float} A value between 0.0 (transparent) and 1.0 (solid). │ │ │ │ │ - */ │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - //does nothing since we suppose that we'll never apply an opacity │ │ │ │ │ - // to a framed popup │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: DrawLine │ │ │ │ │ + * Render a line. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * closeLine - {Boolean} Close the line? (make it a ring?) │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + drawLine: function(node, geometry, closeLine) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setSize │ │ │ │ │ - * Overridden here, because we need to update the blocks whenever the size │ │ │ │ │ - * of the popup has changed. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * contentSize - {<OpenLayers.Size>} the new size for the popup's │ │ │ │ │ - * contents div (in pixels). │ │ │ │ │ - */ │ │ │ │ │ - setSize: function(contentSize) { │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.setSize.apply(this, arguments); │ │ │ │ │ + this.setNodeDimension(node, geometry); │ │ │ │ │ │ │ │ │ │ - this.updateBlocks(); │ │ │ │ │ - }, │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var numComponents = geometry.components.length; │ │ │ │ │ + var parts = new Array(numComponents); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateRelativePosition │ │ │ │ │ - * When the relative position changes, we need to set the new padding │ │ │ │ │ - * BBOX on the popup, reposition the close div, and update the blocks. │ │ │ │ │ - */ │ │ │ │ │ - updateRelativePosition: function() { │ │ │ │ │ + var comp, x, y; │ │ │ │ │ + for (var i = 0; i < numComponents; i++) { │ │ │ │ │ + comp = geometry.components[i]; │ │ │ │ │ + x = ((comp.x - this.featureDx) / resolution - this.offset.x) | 0; │ │ │ │ │ + y = (comp.y / resolution - this.offset.y) | 0; │ │ │ │ │ + parts[i] = " " + x + "," + y + " l "; │ │ │ │ │ + } │ │ │ │ │ + var end = (closeLine) ? " x e" : " e"; │ │ │ │ │ + node.path = "m" + parts.join("") + end; │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - //update the padding │ │ │ │ │ - this.padding = this.positionBlocks[this.relativePosition].padding; │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawPolygon │ │ │ │ │ + * Render a polygon │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + drawPolygon: function(node, geometry) { │ │ │ │ │ + this.setNodeDimension(node, geometry); │ │ │ │ │ │ │ │ │ │ - //update the position of our close box to new padding │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - // use the content div's css padding to determine if we should │ │ │ │ │ - // padd the close div │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ │ │ │ │ │ - this.closeDiv.style.right = contentDivPadding.right + │ │ │ │ │ - this.padding.right + "px"; │ │ │ │ │ - this.closeDiv.style.top = contentDivPadding.top + │ │ │ │ │ - this.padding.top + "px"; │ │ │ │ │ + var path = []; │ │ │ │ │ + var j, jj, points, area, first, second, i, ii, comp, pathComp, x, y; │ │ │ │ │ + for (j = 0, jj = geometry.components.length; j < jj; j++) { │ │ │ │ │ + path.push("m"); │ │ │ │ │ + points = geometry.components[j].components; │ │ │ │ │ + // we only close paths of interior rings with area │ │ │ │ │ + area = (j === 0); │ │ │ │ │ + first = null; │ │ │ │ │ + second = null; │ │ │ │ │ + for (i = 0, ii = points.length; i < ii; i++) { │ │ │ │ │ + comp = points[i]; │ │ │ │ │ + x = ((comp.x - this.featureDx) / resolution - this.offset.x) | 0; │ │ │ │ │ + y = (comp.y / resolution - this.offset.y) | 0; │ │ │ │ │ + pathComp = " " + x + "," + y; │ │ │ │ │ + path.push(pathComp); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + path.push(" l"); │ │ │ │ │ + } │ │ │ │ │ + if (!area) { │ │ │ │ │ + // IE improperly renders sub-paths that have no area. │ │ │ │ │ + // Instead of checking the area of every ring, we confirm │ │ │ │ │ + // the ring has at least three distinct points. This does │ │ │ │ │ + // not catch all non-zero area cases, but it greatly improves │ │ │ │ │ + // interior ring digitizing and is a minor performance hit │ │ │ │ │ + // when rendering rings with many points. │ │ │ │ │ + if (!first) { │ │ │ │ │ + first = pathComp; │ │ │ │ │ + } else if (first != pathComp) { │ │ │ │ │ + if (!second) { │ │ │ │ │ + second = pathComp; │ │ │ │ │ + } else if (second != pathComp) { │ │ │ │ │ + // stop looking │ │ │ │ │ + area = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + path.push(area ? " x " : " "); │ │ │ │ │ + } │ │ │ │ │ + path.push("e"); │ │ │ │ │ + node.path = path.join(""); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.updateBlocks(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateNewPx │ │ │ │ │ - * Besides the standard offset as determined by the Anchored class, our │ │ │ │ │ - * Framed popups have a special 'offset' property for each of their │ │ │ │ │ - * positions, which is used to offset the popup relative to its anchor. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Pixel>} The the new px position of the popup on the screen │ │ │ │ │ - * relative to the passed-in px. │ │ │ │ │ - */ │ │ │ │ │ - calculateNewPx: function(px) { │ │ │ │ │ - var newPx = OpenLayers.Popup.Anchored.prototype.calculateNewPx.apply( │ │ │ │ │ - this, arguments │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - newPx = newPx.offset(this.positionBlocks[this.relativePosition].offset); │ │ │ │ │ - │ │ │ │ │ - return newPx; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createBlocks │ │ │ │ │ - */ │ │ │ │ │ - createBlocks: function() { │ │ │ │ │ - this.blocks = []; │ │ │ │ │ - │ │ │ │ │ - //since all positions contain the same number of blocks, we can │ │ │ │ │ - // just pick the first position and use its blocks array to create │ │ │ │ │ - // our blocks array │ │ │ │ │ - var firstPosition = null; │ │ │ │ │ - for (var key in this.positionBlocks) { │ │ │ │ │ - firstPosition = key; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawRectangle │ │ │ │ │ + * Render a rectangle │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + drawRectangle: function(node, geometry) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ │ │ │ │ │ - var position = this.positionBlocks[firstPosition]; │ │ │ │ │ - for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ + node.style.left = (((geometry.x - this.featureDx) / resolution - this.offset.x) | 0) + "px"; │ │ │ │ │ + node.style.top = ((geometry.y / resolution - this.offset.y) | 0) + "px"; │ │ │ │ │ + node.style.width = ((geometry.width / resolution) | 0) + "px"; │ │ │ │ │ + node.style.height = ((geometry.height / resolution) | 0) + "px"; │ │ │ │ │ │ │ │ │ │ - var block = {}; │ │ │ │ │ - this.blocks.push(block); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var divId = this.id + '_FrameDecorationDiv_' + i; │ │ │ │ │ - block.div = OpenLayers.Util.createDiv(divId, │ │ │ │ │ - null, null, null, "absolute", null, "hidden", null │ │ │ │ │ - ); │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawText │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * style - │ │ │ │ │ + * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + */ │ │ │ │ │ + drawText: function(featureId, style, location) { │ │ │ │ │ + var label = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX, "olv:rect"); │ │ │ │ │ + var textbox = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX + "_textbox", "olv:textbox"); │ │ │ │ │ │ │ │ │ │ - var imgId = this.id + '_FrameDecorationImg_' + i; │ │ │ │ │ - var imageCreator = │ │ │ │ │ - (this.isAlphaImage) ? OpenLayers.Util.createAlphaImageDiv : │ │ │ │ │ - OpenLayers.Util.createImage; │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + label.style.left = (((location.x - this.featureDx) / resolution - this.offset.x) | 0) + "px"; │ │ │ │ │ + label.style.top = ((location.y / resolution - this.offset.y) | 0) + "px"; │ │ │ │ │ + label.style.flip = "y"; │ │ │ │ │ │ │ │ │ │ - block.image = imageCreator(imgId, │ │ │ │ │ - null, this.imageSize, this.imageSrc, │ │ │ │ │ - "absolute", null, null, null │ │ │ │ │ - ); │ │ │ │ │ + textbox.innerText = style.label; │ │ │ │ │ │ │ │ │ │ - block.div.appendChild(block.image); │ │ │ │ │ - this.groupDiv.appendChild(block.div); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ + textbox.style.cursor = style.cursor; │ │ │ │ │ + } │ │ │ │ │ + if (style.fontColor) { │ │ │ │ │ + textbox.style.color = style.fontColor; │ │ │ │ │ + } │ │ │ │ │ + if (style.fontOpacity) { │ │ │ │ │ + textbox.style.filter = 'alpha(opacity=' + (style.fontOpacity * 100) + ')'; │ │ │ │ │ + } │ │ │ │ │ + if (style.fontFamily) { │ │ │ │ │ + textbox.style.fontFamily = style.fontFamily; │ │ │ │ │ + } │ │ │ │ │ + if (style.fontSize) { │ │ │ │ │ + textbox.style.fontSize = style.fontSize; │ │ │ │ │ + } │ │ │ │ │ + if (style.fontWeight) { │ │ │ │ │ + textbox.style.fontWeight = style.fontWeight; │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStyle) { │ │ │ │ │ + textbox.style.fontStyle = style.fontStyle; │ │ │ │ │ + } │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + label._featureId = featureId; │ │ │ │ │ + textbox._featureId = featureId; │ │ │ │ │ + textbox._geometry = location; │ │ │ │ │ + textbox._geometryClass = location.CLASS_NAME; │ │ │ │ │ + } │ │ │ │ │ + textbox.style.whiteSpace = "nowrap"; │ │ │ │ │ + // fun with IE: IE7 in standards compliant mode does not display any │ │ │ │ │ + // text with a left inset of 0. So we set this to 1px and subtract one │ │ │ │ │ + // pixel later when we set label.style.left │ │ │ │ │ + textbox.inset = "1px,0px,0px,0px"; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateBlocks │ │ │ │ │ - * Internal method, called on initialize and when the popup's relative │ │ │ │ │ - * position has changed. This function takes care of re-positioning │ │ │ │ │ - * the popup's blocks in their appropropriate places. │ │ │ │ │ - */ │ │ │ │ │ - updateBlocks: function() { │ │ │ │ │ - if (!this.blocks) { │ │ │ │ │ - this.createBlocks(); │ │ │ │ │ - } │ │ │ │ │ + if (!label.parentNode) { │ │ │ │ │ + label.appendChild(textbox); │ │ │ │ │ + this.textRoot.appendChild(label); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (this.size && this.relativePosition) { │ │ │ │ │ - var position = this.positionBlocks[this.relativePosition]; │ │ │ │ │ - for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ + var align = style.labelAlign || "cm"; │ │ │ │ │ + if (align.length == 1) { │ │ │ │ │ + align += "m"; │ │ │ │ │ + } │ │ │ │ │ + var xshift = textbox.clientWidth * │ │ │ │ │ + (OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(0, 1)]); │ │ │ │ │ + var yshift = textbox.clientHeight * │ │ │ │ │ + (OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(1, 1)]); │ │ │ │ │ + label.style.left = parseInt(label.style.left) - xshift - 1 + "px"; │ │ │ │ │ + label.style.top = parseInt(label.style.top) + yshift + "px"; │ │ │ │ │ │ │ │ │ │ - var positionBlock = position.blocks[i]; │ │ │ │ │ - var block = this.blocks[i]; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // adjust sizes │ │ │ │ │ - var l = positionBlock.anchor.left; │ │ │ │ │ - var b = positionBlock.anchor.bottom; │ │ │ │ │ - var r = positionBlock.anchor.right; │ │ │ │ │ - var t = positionBlock.anchor.top; │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveRoot │ │ │ │ │ + * moves this renderer's root to a different renderer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * renderer - {<OpenLayers.Renderer>} target renderer for the moved root │ │ │ │ │ + * root - {DOMElement} optional root node. To be used when this renderer │ │ │ │ │ + * holds roots from multiple layers to tell this method which one to │ │ │ │ │ + * detach │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if successful, false otherwise │ │ │ │ │ + */ │ │ │ │ │ + moveRoot: function(renderer) { │ │ │ │ │ + var layer = this.map.getLayer(renderer.container.id); │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Vector.RootContainer) { │ │ │ │ │ + layer = this.map.getLayer(this.container.id); │ │ │ │ │ + } │ │ │ │ │ + layer && layer.renderer.clear(); │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.moveRoot.apply(this, arguments); │ │ │ │ │ + layer && layer.redraw(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - //note that we use the isNaN() test here because if the │ │ │ │ │ - // size object is initialized with a "auto" parameter, the │ │ │ │ │ - // size constructor calls parseFloat() on the string, │ │ │ │ │ - // which will turn it into NaN │ │ │ │ │ - // │ │ │ │ │ - var w = (isNaN(positionBlock.size.w)) ? this.size.w - (r + l) : │ │ │ │ │ - positionBlock.size.w; │ │ │ │ │ + /** │ │ │ │ │ + * Method: importSymbol │ │ │ │ │ + * add a new symbol definition from the rendererer's symbol hash │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * graphicName - {String} name of the symbol to import │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} - hash of {DOMElement} "symbol" and {Number} "size" │ │ │ │ │ + */ │ │ │ │ │ + importSymbol: function(graphicName) { │ │ │ │ │ + var id = this.container.id + "-" + graphicName; │ │ │ │ │ │ │ │ │ │ - var h = (isNaN(positionBlock.size.h)) ? this.size.h - (b + t) : │ │ │ │ │ - positionBlock.size.h; │ │ │ │ │ + // check if symbol already exists in the cache │ │ │ │ │ + var cache = this.symbolCache[id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + return cache; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - block.div.style.width = (w < 0 ? 0 : w) + 'px'; │ │ │ │ │ - block.div.style.height = (h < 0 ? 0 : h) + 'px'; │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(graphicName + ' is not a valid symbol name'); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - block.div.style.left = (l != null) ? l + 'px' : ''; │ │ │ │ │ - block.div.style.bottom = (b != null) ? b + 'px' : ''; │ │ │ │ │ - block.div.style.right = (r != null) ? r + 'px' : ''; │ │ │ │ │ - block.div.style.top = (t != null) ? t + 'px' : ''; │ │ │ │ │ + var symbolExtent = new OpenLayers.Bounds( │ │ │ │ │ + Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ │ │ │ │ │ - block.image.style.left = positionBlock.position.x + 'px'; │ │ │ │ │ - block.image.style.top = positionBlock.position.y + 'px'; │ │ │ │ │ - } │ │ │ │ │ + var pathitems = ["m"]; │ │ │ │ │ + for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + var x = symbol[i]; │ │ │ │ │ + var y = symbol[i + 1]; │ │ │ │ │ + symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ + symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ + symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ + symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ │ │ │ │ │ - this.contentDiv.style.left = this.padding.left + "px"; │ │ │ │ │ - this.contentDiv.style.top = this.padding.top + "px"; │ │ │ │ │ + pathitems.push(x); │ │ │ │ │ + pathitems.push(y); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + pathitems.push("l"); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ + } │ │ │ │ │ + pathitems.push("x e"); │ │ │ │ │ + var path = pathitems.join(" "); │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup.Framed" │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Popup/FramedCloud.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + var diff = (symbolExtent.getWidth() - symbolExtent.getHeight()) / 2; │ │ │ │ │ + if (diff > 0) { │ │ │ │ │ + symbolExtent.bottom = symbolExtent.bottom - diff; │ │ │ │ │ + symbolExtent.top = symbolExtent.top + diff; │ │ │ │ │ + } else { │ │ │ │ │ + symbolExtent.left = symbolExtent.left + diff; │ │ │ │ │ + symbolExtent.right = symbolExtent.right - diff; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + cache = { │ │ │ │ │ + path: path, │ │ │ │ │ + size: symbolExtent.getWidth(), // equals getHeight() now │ │ │ │ │ + left: symbolExtent.left, │ │ │ │ │ + bottom: symbolExtent.bottom │ │ │ │ │ + }; │ │ │ │ │ + this.symbolCache[id] = cache; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Popup/Framed.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Bounds.js │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Pixel.js │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Size.js │ │ │ │ │ - */ │ │ │ │ │ + return cache; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.VML" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Popup.FramedCloud │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Popup.Framed> │ │ │ │ │ + * Constant: OpenLayers.Renderer.VML.LABEL_SHIFT │ │ │ │ │ + * {Object} │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Popup.FramedCloud = │ │ │ │ │ - OpenLayers.Class(OpenLayers.Popup.Framed, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: contentDisplayClass │ │ │ │ │ - * {String} The CSS class of the popup content div. │ │ │ │ │ - */ │ │ │ │ │ - contentDisplayClass: "olFramedCloudPopupContent", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoSize │ │ │ │ │ - * {Boolean} Framed Cloud is autosizing by default. │ │ │ │ │ - */ │ │ │ │ │ - autoSize: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: panMapIfOutOfView │ │ │ │ │ - * {Boolean} Framed Cloud does pan into view by default. │ │ │ │ │ - */ │ │ │ │ │ - panMapIfOutOfView: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: imageSize │ │ │ │ │ - * {<OpenLayers.Size>} │ │ │ │ │ - */ │ │ │ │ │ - imageSize: new OpenLayers.Size(1276, 736), │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isAlphaImage │ │ │ │ │ - * {Boolean} The FramedCloud does not use an alpha image (in honor of the │ │ │ │ │ - * good ie6 folk out there) │ │ │ │ │ - */ │ │ │ │ │ - isAlphaImage: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: fixedRelativePosition │ │ │ │ │ - * {Boolean} The Framed Cloud popup works in just one fixed position. │ │ │ │ │ - */ │ │ │ │ │ - fixedRelativePosition: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: positionBlocks │ │ │ │ │ - * {Object} Hash of differen position blocks, keyed by relativePosition │ │ │ │ │ - * two-character code string (ie "tl", "tr", "bl", "br") │ │ │ │ │ - */ │ │ │ │ │ - positionBlocks: { │ │ │ │ │ - "tl": { │ │ │ │ │ - 'offset': new OpenLayers.Pixel(44, 0), │ │ │ │ │ - 'padding': new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ - 'blocks': [{ // top-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { //top-right │ │ │ │ │ - size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { //bottom-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ - }, { //bottom-right │ │ │ │ │ - size: new OpenLayers.Size(22, 18), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -632) │ │ │ │ │ - }, { // stem │ │ │ │ │ - size: new OpenLayers.Size(81, 35), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -688) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - "tr": { │ │ │ │ │ - 'offset': new OpenLayers.Pixel(-45, 0), │ │ │ │ │ - 'padding': new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ - 'blocks': [{ // top-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { //top-right │ │ │ │ │ - size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { //bottom-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ - }, { //bottom-right │ │ │ │ │ - size: new OpenLayers.Size(22, 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -631) │ │ │ │ │ - }, { // stem │ │ │ │ │ - size: new OpenLayers.Size(81, 35), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, null, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-215, -687) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - "bl": { │ │ │ │ │ - 'offset': new OpenLayers.Pixel(45, 0), │ │ │ │ │ - 'padding': new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ - 'blocks': [{ // top-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { //top-right │ │ │ │ │ - size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { //bottom-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ - }, { //bottom-right │ │ │ │ │ - size: new OpenLayers.Size(22, 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ - }, { // stem │ │ │ │ │ - size: new OpenLayers.Size(81, 33), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, null, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-101, -674) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - "br": { │ │ │ │ │ - 'offset': new OpenLayers.Pixel(-44, 0), │ │ │ │ │ - 'padding': new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ - 'blocks': [{ // top-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { //top-right │ │ │ │ │ - size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { //bottom-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ - }, { //bottom-right │ │ │ │ │ - size: new OpenLayers.Size(22, 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ - }, { // stem │ │ │ │ │ - size: new OpenLayers.Size(81, 33), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, null, null, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-311, -674) │ │ │ │ │ - }] │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: minSize │ │ │ │ │ - * {<OpenLayers.Size>} │ │ │ │ │ - */ │ │ │ │ │ - minSize: new OpenLayers.Size(105, 10), │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: maxSize │ │ │ │ │ - * {<OpenLayers.Size>} │ │ │ │ │ - */ │ │ │ │ │ - maxSize: new OpenLayers.Size(1200, 660), │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Popup.FramedCloud │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ - * contentSize - {<OpenLayers.Size>} │ │ │ │ │ - * contentHTML - {String} │ │ │ │ │ - * anchor - {Object} Object to which we'll anchor the popup. Must expose │ │ │ │ │ - * a 'size' (<OpenLayers.Size>) and 'offset' (<OpenLayers.Pixel>) │ │ │ │ │ - * (Note that this is generally an <OpenLayers.Icon>). │ │ │ │ │ - * closeBox - {Boolean} │ │ │ │ │ - * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, │ │ │ │ │ - closeBoxCallback) { │ │ │ │ │ - │ │ │ │ │ - this.imageSrc = OpenLayers.Util.getImageLocation('cloud-popup-relative.png'); │ │ │ │ │ - OpenLayers.Popup.Framed.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.contentDiv.className = this.contentDisplayClass; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup.FramedCloud" │ │ │ │ │ - }); │ │ │ │ │ +OpenLayers.Renderer.VML.LABEL_SHIFT = { │ │ │ │ │ + "l": 0, │ │ │ │ │ + "c": .5, │ │ │ │ │ + "r": 1, │ │ │ │ │ + "t": 0, │ │ │ │ │ + "m": .5, │ │ │ │ │ + "b": 1 │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WFSCapabilities.js │ │ │ │ │ + OpenLayers/Renderer/SVG.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + * @requires OpenLayers/Renderer/Elements.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WFSCapabilities │ │ │ │ │ - * Read WFS Capabilities. │ │ │ │ │ + * Class: OpenLayers.Renderer.SVG │ │ │ │ │ * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + * Inherits: │ │ │ │ │ + * - <OpenLayers.Renderer.Elements> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WFSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ +OpenLayers.Renderer.SVG = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.1.0". │ │ │ │ │ + /** │ │ │ │ │ + * Property: xmlns │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - defaultVersion: "1.1.0", │ │ │ │ │ + xmlns: "http://www.w3.org/2000/svg", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WFSCapabilities │ │ │ │ │ - * Create a new parser for WFS capabilities. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * Property: xlinkns │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ + xlinkns: "http://www.w3.org/1999/xlink", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} List of named layers. │ │ │ │ │ + * Constant: MAX_PIXEL │ │ │ │ │ + * {Integer} Firefox has a limitation where values larger or smaller than │ │ │ │ │ + * about 15000 in an SVG document lock the browser up. This │ │ │ │ │ + * works around it. │ │ │ │ │ */ │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSCapabilities" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WPSCapabilities.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WPSCapabilities │ │ │ │ │ - * Read WPS Capabilities. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WPSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + MAX_PIXEL: 15000, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.0.0". │ │ │ │ │ + * Property: translationParameters │ │ │ │ │ + * {Object} Hash with "x" and "y" properties │ │ │ │ │ */ │ │ │ │ │ - defaultVersion: "1.0.0", │ │ │ │ │ + translationParameters: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WPSCapabilities │ │ │ │ │ - * Create a new parser for WPS Capabilities. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * Property: symbolMetrics │ │ │ │ │ + * {Object} Cache for symbol metrics according to their svg coordinate │ │ │ │ │ + * space. This is an object keyed by the symbol's id, and values are │ │ │ │ │ + * an array of [width, centerX, centerY]. │ │ │ │ │ */ │ │ │ │ │ + symbolMetrics: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return information about │ │ │ │ │ - * the service. │ │ │ │ │ + * Constructor: OpenLayers.Renderer.SVG │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Info about the WPS │ │ │ │ │ + * Parameters: │ │ │ │ │ + * containerID - {String} │ │ │ │ │ */ │ │ │ │ │ + initialize: function(containerID) { │ │ │ │ │ + if (!this.supported()) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.initialize.apply(this, │ │ │ │ │ + arguments); │ │ │ │ │ + this.translationParameters = { │ │ │ │ │ + x: 0, │ │ │ │ │ + y: 0 │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WPSCapabilities" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/Atom.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/GML/v3.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.Atom │ │ │ │ │ - * Read/write Atom feeds. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Format.AtomFeed> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.Atom = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. Properties │ │ │ │ │ - * of this object should not be set individually. Read-only. All │ │ │ │ │ - * XML subclasses should have their own namespaces object. Use │ │ │ │ │ - * <setNamespace> to add or set a namespace alias after construction. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - atom: "http://www.w3.org/2005/Atom", │ │ │ │ │ - georss: "http://www.georss.org/georss" │ │ │ │ │ + this.symbolMetrics = {}; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: feedTitle │ │ │ │ │ - * {String} Atom feed elements require a title. Default is "untitled". │ │ │ │ │ - */ │ │ │ │ │ - feedTitle: "untitled", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: defaultEntryTitle │ │ │ │ │ - * {String} Atom entry elements require a title. In cases where one is │ │ │ │ │ - * not provided in the feature attributes, this will be used. Default │ │ │ │ │ - * is "untitled". │ │ │ │ │ - */ │ │ │ │ │ - defaultEntryTitle: "untitled", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: gmlParse │ │ │ │ │ - * {Object} GML Format object for parsing features │ │ │ │ │ - * Non-API and only created if necessary │ │ │ │ │ - */ │ │ │ │ │ - gmlParser: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: xy │ │ │ │ │ - * {Boolean} Order of the GML coordinate: true:(x,y) or false:(y,x) │ │ │ │ │ - * For GeoRSS the default is (y,x), therefore: false │ │ │ │ │ + * APIMethod: supported │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the browser supports the SVG renderer │ │ │ │ │ */ │ │ │ │ │ - xy: false, │ │ │ │ │ + supported: function() { │ │ │ │ │ + var svgFeature = "http://www.w3.org/TR/SVG11/feature#"; │ │ │ │ │ + return (document.implementation && │ │ │ │ │ + (document.implementation.hasFeature("org.w3c.svg", "1.0") || │ │ │ │ │ + document.implementation.hasFeature(svgFeature + "SVG", "1.1") || │ │ │ │ │ + document.implementation.hasFeature(svgFeature + "BasicStructure", "1.1"))); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.AtomEntry │ │ │ │ │ - * Create a new parser for Atom. │ │ │ │ │ + * Method: inValidRange │ │ │ │ │ + * See #669 for more information │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * x - {Integer} │ │ │ │ │ + * y - {Integer} │ │ │ │ │ + * xyOnly - {Boolean} whether or not to just check for x and y, which means │ │ │ │ │ + * to not take the current translation parameters into account if true. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the 'x' and 'y' coordinates are in the │ │ │ │ │ + * valid range. │ │ │ │ │ */ │ │ │ │ │ + inValidRange: function(x, y, xyOnly) { │ │ │ │ │ + var left = x + (xyOnly ? 0 : this.translationParameters.x); │ │ │ │ │ + var top = y + (xyOnly ? 0 : this.translationParameters.y); │ │ │ │ │ + return (left >= -this.MAX_PIXEL && left <= this.MAX_PIXEL && │ │ │ │ │ + top >= -this.MAX_PIXEL && top <= this.MAX_PIXEL); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Return a list of features from an Atom feed or entry document. │ │ │ │ │ - │ │ │ │ │ + * Method: setExtent │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * doc - {Element} or {String} │ │ │ │ │ - * │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} │ │ │ │ │ + * resolutionChanged - {Boolean} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ + * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ + * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ + * False otherwise. │ │ │ │ │ */ │ │ │ │ │ - read: function(doc) { │ │ │ │ │ - if (typeof doc == "string") { │ │ │ │ │ - doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]); │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + var resolution = this.getResolution(), │ │ │ │ │ + left = -extent.left / resolution, │ │ │ │ │ + top = extent.top / resolution; │ │ │ │ │ + │ │ │ │ │ + // If the resolution has changed, start over changing the corner, because │ │ │ │ │ + // the features will redraw. │ │ │ │ │ + if (resolutionChanged) { │ │ │ │ │ + this.left = left; │ │ │ │ │ + this.top = top; │ │ │ │ │ + // Set the viewbox │ │ │ │ │ + var extentString = "0 0 " + this.size.w + " " + this.size.h; │ │ │ │ │ + │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "viewBox", extentString); │ │ │ │ │ + this.translate(this.xOffset, 0); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + var inRange = this.translate(left - this.left + this.xOffset, top - this.top); │ │ │ │ │ + if (!inRange) { │ │ │ │ │ + // recenter the coordinate system │ │ │ │ │ + this.setExtent(extent, true); │ │ │ │ │ + } │ │ │ │ │ + return coordSysUnchanged && inRange; │ │ │ │ │ } │ │ │ │ │ - return this.parseFeatures(doc); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Serialize or more feature nodes to Atom documents. │ │ │ │ │ - * │ │ │ │ │ + * Method: translate │ │ │ │ │ + * Transforms the SVG coordinate system │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {<OpenLayers.Feature.Vector>} or Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ - * │ │ │ │ │ + * x - {Float} │ │ │ │ │ + * y - {Float} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} an Atom entry document if passed one feature node, or a feed │ │ │ │ │ - * document if passed an array of feature nodes. │ │ │ │ │ + * {Boolean} true if the translation parameters are in the valid coordinates │ │ │ │ │ + * range, false otherwise. │ │ │ │ │ */ │ │ │ │ │ - write: function(features) { │ │ │ │ │ - var doc; │ │ │ │ │ - if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ - doc = this.createElementNSPlus("atom:feed"); │ │ │ │ │ - doc.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:title", { │ │ │ │ │ - value: this.feedTitle │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ - for (var i = 0, ii = features.length; i < ii; i++) { │ │ │ │ │ - doc.appendChild(this.buildEntryNode(features[i])); │ │ │ │ │ - } │ │ │ │ │ + translate: function(x, y) { │ │ │ │ │ + if (!this.inValidRange(x, y, true)) { │ │ │ │ │ + return false; │ │ │ │ │ } else { │ │ │ │ │ - doc = this.buildEntryNode(features); │ │ │ │ │ + var transformString = ""; │ │ │ │ │ + if (x || y) { │ │ │ │ │ + transformString = "translate(" + x + "," + y + ")"; │ │ │ │ │ + } │ │ │ │ │ + this.root.setAttributeNS(null, "transform", transformString); │ │ │ │ │ + this.translationParameters = { │ │ │ │ │ + x: x, │ │ │ │ │ + y: y │ │ │ │ │ + }; │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [doc]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildContentNode │ │ │ │ │ - * │ │ │ │ │ + * Method: setSize │ │ │ │ │ + * Sets the size of the drawing surface. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * content - {Object} │ │ │ │ │ - * │ │ │ │ │ + * size - {<OpenLayers.Size>} The size of the drawing surface │ │ │ │ │ + */ │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "width", this.size.w); │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "height", this.size.h); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getNodeType │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} an Atom content node. │ │ │ │ │ - * │ │ │ │ │ - * TODO: types other than text. │ │ │ │ │ + * {String} The corresponding node type for the specified geometry │ │ │ │ │ */ │ │ │ │ │ - buildContentNode: function(content) { │ │ │ │ │ - var node = this.createElementNSPlus("atom:content", { │ │ │ │ │ - attributes: { │ │ │ │ │ - type: content.type || null │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (content.src) { │ │ │ │ │ - node.setAttribute("src", content.src); │ │ │ │ │ - } else { │ │ │ │ │ - if (content.type == "text" || content.type == null) { │ │ │ │ │ - node.appendChild( │ │ │ │ │ - this.createTextNode(content.value) │ │ │ │ │ - ); │ │ │ │ │ - } else if (content.type == "html") { │ │ │ │ │ - if (typeof content.value != "string") { │ │ │ │ │ - throw "HTML content must be in form of an escaped string"; │ │ │ │ │ + getNodeType: function(geometry, style) { │ │ │ │ │ + var nodeType = null; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + nodeType = "image"; │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + nodeType = "svg"; │ │ │ │ │ + } else { │ │ │ │ │ + nodeType = "circle"; │ │ │ │ │ } │ │ │ │ │ - node.appendChild( │ │ │ │ │ - this.createTextNode(content.value) │ │ │ │ │ - ); │ │ │ │ │ - } else if (content.type == "xhtml") { │ │ │ │ │ - node.appendChild(content.value); │ │ │ │ │ - } else if (content.type == "xhtml" || │ │ │ │ │ - content.type.match(/(\+|\/)xml$/)) { │ │ │ │ │ - node.appendChild(content.value); │ │ │ │ │ - } else { // MUST be a valid Base64 encoding │ │ │ │ │ - node.appendChild( │ │ │ │ │ - this.createTextNode(content.value) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + nodeType = "rect"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + nodeType = "polyline"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + nodeType = "polygon"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + case "OpenLayers.Geometry.Curve": │ │ │ │ │ + nodeType = "path"; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ + return nodeType; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildEntryNode │ │ │ │ │ - * Build an Atom entry node from a feature object. │ │ │ │ │ + /** │ │ │ │ │ + * Method: setStyle │ │ │ │ │ + * Use to set all the style attributes to a SVG node. │ │ │ │ │ + * │ │ │ │ │ + * Takes care to adjust stroke width and point radius to be │ │ │ │ │ + * resolution-relative │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} an Atom entry node. │ │ │ │ │ - * │ │ │ │ │ - * These entries are geared for publication using AtomPub. │ │ │ │ │ - * │ │ │ │ │ - * TODO: support extension elements │ │ │ │ │ + * node - {SVGDomElement} An SVG element to decorate │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * options - {Object} Currently supported options include │ │ │ │ │ + * 'isFilled' {Boolean} and │ │ │ │ │ + * 'isStroked' {Boolean} │ │ │ │ │ */ │ │ │ │ │ - buildEntryNode: function(feature) { │ │ │ │ │ - var attrib = feature.attributes; │ │ │ │ │ - var atomAttrib = attrib.atom || {}; │ │ │ │ │ - var entryNode = this.createElementNSPlus("atom:entry"); │ │ │ │ │ + setStyle: function(node, style, options) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + options = options || node._options; │ │ │ │ │ │ │ │ │ │ - // atom:author │ │ │ │ │ - if (atomAttrib.authors) { │ │ │ │ │ - var authors = OpenLayers.Util.isArray(atomAttrib.authors) ? │ │ │ │ │ - atomAttrib.authors : [atomAttrib.authors]; │ │ │ │ │ - for (var i = 0, ii = authors.length; i < ii; i++) { │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.buildPersonConstructNode( │ │ │ │ │ - "author", authors[i] │ │ │ │ │ - ) │ │ │ │ │ - ); │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + node.setAttributeNS(null, "title", title); │ │ │ │ │ + //Standards-conformant SVG │ │ │ │ │ + // Prevent duplicate nodes. See issue https://github.com/openlayers/openlayers/issues/92 │ │ │ │ │ + var titleNode = node.getElementsByTagName("title"); │ │ │ │ │ + if (titleNode.length > 0) { │ │ │ │ │ + titleNode[0].firstChild.textContent = title; │ │ │ │ │ + } else { │ │ │ │ │ + var label = this.nodeFactory(null, "title"); │ │ │ │ │ + label.textContent = title; │ │ │ │ │ + node.appendChild(label); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // atom:category │ │ │ │ │ - if (atomAttrib.categories) { │ │ │ │ │ - var categories = OpenLayers.Util.isArray(atomAttrib.categories) ? │ │ │ │ │ - atomAttrib.categories : [atomAttrib.categories]; │ │ │ │ │ - var category; │ │ │ │ │ - for (var i = 0, ii = categories.length; i < ii; i++) { │ │ │ │ │ - category = categories[i]; │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:category", { │ │ │ │ │ - attributes: { │ │ │ │ │ - term: category.term, │ │ │ │ │ - scheme: category.scheme || null, │ │ │ │ │ - label: category.label || null │ │ │ │ │ - } │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + var r = parseFloat(node.getAttributeNS(null, "r")); │ │ │ │ │ + var widthFactor = 1; │ │ │ │ │ + var pos; │ │ │ │ │ + if (node._geometryClass == "OpenLayers.Geometry.Point" && r) { │ │ │ │ │ + node.style.visibility = ""; │ │ │ │ │ + if (style.graphic === false) { │ │ │ │ │ + node.style.visibility = "hidden"; │ │ │ │ │ + } else if (style.externalGraphic) { │ │ │ │ │ + pos = this.getPosition(node); │ │ │ │ │ + if (style.graphicWidth && style.graphicHeight) { │ │ │ │ │ + node.setAttributeNS(null, "preserveAspectRatio", "none"); │ │ │ │ │ + } │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ + var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ + style.graphicXOffset : -(0.5 * width); │ │ │ │ │ + var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ + style.graphicYOffset : -(0.5 * height); │ │ │ │ │ │ │ │ │ │ - // atom:content │ │ │ │ │ - if (atomAttrib.content) { │ │ │ │ │ - entryNode.appendChild(this.buildContentNode(atomAttrib.content)); │ │ │ │ │ - } │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ │ │ │ │ │ - // atom:contributor │ │ │ │ │ - if (atomAttrib.contributors) { │ │ │ │ │ - var contributors = OpenLayers.Util.isArray(atomAttrib.contributors) ? │ │ │ │ │ - atomAttrib.contributors : [atomAttrib.contributors]; │ │ │ │ │ - for (var i = 0, ii = contributors.length; i < ii; i++) { │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.buildPersonConstructNode( │ │ │ │ │ - "contributor", │ │ │ │ │ - contributors[i] │ │ │ │ │ - ) │ │ │ │ │ - ); │ │ │ │ │ + node.setAttributeNS(null, "x", (pos.x + xOffset).toFixed()); │ │ │ │ │ + node.setAttributeNS(null, "y", (pos.y + yOffset).toFixed()); │ │ │ │ │ + node.setAttributeNS(null, "width", width); │ │ │ │ │ + node.setAttributeNS(null, "height", height); │ │ │ │ │ + node.setAttributeNS(this.xlinkns, "xlink:href", style.externalGraphic); │ │ │ │ │ + node.setAttributeNS(null, "style", "opacity: " + opacity); │ │ │ │ │ + node.onclick = OpenLayers.Event.preventDefault; │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + // the symbol viewBox is three times as large as the symbol │ │ │ │ │ + var offset = style.pointRadius * 3; │ │ │ │ │ + var size = offset * 2; │ │ │ │ │ + var src = this.importSymbol(style.graphicName); │ │ │ │ │ + pos = this.getPosition(node); │ │ │ │ │ + widthFactor = this.symbolMetrics[src.id][0] * 3 / size; │ │ │ │ │ + │ │ │ │ │ + // remove the node from the dom before we modify it. This │ │ │ │ │ + // prevents various rendering issues in Safari and FF │ │ │ │ │ + var parent = node.parentNode; │ │ │ │ │ + var nextSibling = node.nextSibling; │ │ │ │ │ + if (parent) { │ │ │ │ │ + parent.removeChild(node); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // The more appropriate way to implement this would be use/defs, │ │ │ │ │ + // but due to various issues in several browsers, it is safer to │ │ │ │ │ + // copy the symbols instead of referencing them. │ │ │ │ │ + // See e.g. ticket http://trac.osgeo.org/openlayers/ticket/2985 │ │ │ │ │ + // and this email thread │ │ │ │ │ + // http://osgeo-org.1803224.n2.nabble.com/Select-Control-Ctrl-click-on-Feature-with-a-graphicName-opens-new-browser-window-tc5846039.html │ │ │ │ │ + node.firstChild && node.removeChild(node.firstChild); │ │ │ │ │ + node.appendChild(src.firstChild.cloneNode(true)); │ │ │ │ │ + node.setAttributeNS(null, "viewBox", src.getAttributeNS(null, "viewBox")); │ │ │ │ │ + │ │ │ │ │ + node.setAttributeNS(null, "width", size); │ │ │ │ │ + node.setAttributeNS(null, "height", size); │ │ │ │ │ + node.setAttributeNS(null, "x", pos.x - offset); │ │ │ │ │ + node.setAttributeNS(null, "y", pos.y - offset); │ │ │ │ │ + │ │ │ │ │ + // now that the node has all its new properties, insert it │ │ │ │ │ + // back into the dom where it was │ │ │ │ │ + if (nextSibling) { │ │ │ │ │ + parent.insertBefore(node, nextSibling); │ │ │ │ │ + } else if (parent) { │ │ │ │ │ + parent.appendChild(node); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "r", style.pointRadius); │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - // atom:id │ │ │ │ │ - if (feature.fid) { │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:id", { │ │ │ │ │ - value: feature.fid │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + var rotation = style.rotation; │ │ │ │ │ │ │ │ │ │ - // atom:link │ │ │ │ │ - if (atomAttrib.links) { │ │ │ │ │ - var links = OpenLayers.Util.isArray(atomAttrib.links) ? │ │ │ │ │ - atomAttrib.links : [atomAttrib.links]; │ │ │ │ │ - var link; │ │ │ │ │ - for (var i = 0, ii = links.length; i < ii; i++) { │ │ │ │ │ - link = links[i]; │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:link", { │ │ │ │ │ - attributes: { │ │ │ │ │ - href: link.href, │ │ │ │ │ - rel: link.rel || null, │ │ │ │ │ - type: link.type || null, │ │ │ │ │ - hreflang: link.hreflang || null, │ │ │ │ │ - title: link.title || null, │ │ │ │ │ - length: link.length || null │ │ │ │ │ - } │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ + if ((rotation !== undefined || node._rotation !== undefined) && pos) { │ │ │ │ │ + node._rotation = rotation; │ │ │ │ │ + rotation |= 0; │ │ │ │ │ + if (node.nodeName !== "svg") { │ │ │ │ │ + node.setAttributeNS(null, "transform", │ │ │ │ │ + "rotate(" + rotation + " " + pos.x + " " + │ │ │ │ │ + pos.y + ")"); │ │ │ │ │ + } else { │ │ │ │ │ + var metrics = this.symbolMetrics[src.id]; │ │ │ │ │ + node.firstChild.setAttributeNS(null, "transform", "rotate(" + │ │ │ │ │ + rotation + " " + │ │ │ │ │ + metrics[1] + " " + │ │ │ │ │ + metrics[2] + ")"); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // atom:published │ │ │ │ │ - if (atomAttrib.published) { │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:published", { │ │ │ │ │ - value: atomAttrib.published │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ + if (options.isFilled) { │ │ │ │ │ + node.setAttributeNS(null, "fill", style.fillColor); │ │ │ │ │ + node.setAttributeNS(null, "fill-opacity", style.fillOpacity); │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "fill", "none"); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // atom:rights │ │ │ │ │ - if (atomAttrib.rights) { │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:rights", { │ │ │ │ │ - value: atomAttrib.rights │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ + if (options.isStroked) { │ │ │ │ │ + node.setAttributeNS(null, "stroke", style.strokeColor); │ │ │ │ │ + node.setAttributeNS(null, "stroke-opacity", style.strokeOpacity); │ │ │ │ │ + node.setAttributeNS(null, "stroke-width", style.strokeWidth * widthFactor); │ │ │ │ │ + node.setAttributeNS(null, "stroke-linecap", style.strokeLinecap || "round"); │ │ │ │ │ + // Hard-coded linejoin for now, to make it look the same as in VML. │ │ │ │ │ + // There is no strokeLinejoin property yet for symbolizers. │ │ │ │ │ + node.setAttributeNS(null, "stroke-linejoin", "round"); │ │ │ │ │ + style.strokeDashstyle && node.setAttributeNS(null, │ │ │ │ │ + "stroke-dasharray", this.dashStyle(style, widthFactor)); │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "stroke", "none"); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // atom:source not implemented │ │ │ │ │ + if (style.pointerEvents) { │ │ │ │ │ + node.setAttributeNS(null, "pointer-events", style.pointerEvents); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // atom:summary │ │ │ │ │ - if (atomAttrib.summary || attrib.description) { │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:summary", { │ │ │ │ │ - value: atomAttrib.summary || attrib.description │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ + if (style.cursor != null) { │ │ │ │ │ + node.setAttributeNS(null, "cursor", style.cursor); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // atom:title │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:title", { │ │ │ │ │ - value: atomAttrib.title || attrib.title || this.defaultEntryTitle │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // atom:updated │ │ │ │ │ - if (atomAttrib.updated) { │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:updated", { │ │ │ │ │ - value: atomAttrib.updated │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ + /** │ │ │ │ │ + * Method: dashStyle │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * widthFactor - {Number} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A SVG compliant 'stroke-dasharray' value │ │ │ │ │ + */ │ │ │ │ │ + dashStyle: function(style, widthFactor) { │ │ │ │ │ + var w = style.strokeWidth * widthFactor; │ │ │ │ │ + var str = style.strokeDashstyle; │ │ │ │ │ + switch (str) { │ │ │ │ │ + case 'solid': │ │ │ │ │ + return 'none'; │ │ │ │ │ + case 'dot': │ │ │ │ │ + return [1, 4 * w].join(); │ │ │ │ │ + case 'dash': │ │ │ │ │ + return [4 * w, 4 * w].join(); │ │ │ │ │ + case 'dashdot': │ │ │ │ │ + return [4 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ + case 'longdash': │ │ │ │ │ + return [8 * w, 4 * w].join(); │ │ │ │ │ + case 'longdashdot': │ │ │ │ │ + return [8 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ + default: │ │ │ │ │ + return OpenLayers.String.trim(str).replace(/\s+/g, ","); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // georss:where │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - var whereNode = this.createElementNSPlus("georss:where"); │ │ │ │ │ - whereNode.appendChild( │ │ │ │ │ - this.buildGeometryNode(feature.geometry) │ │ │ │ │ - ); │ │ │ │ │ - entryNode.appendChild(whereNode); │ │ │ │ │ + /** │ │ │ │ │ + * Method: createNode │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * type - {String} Kind of node to draw │ │ │ │ │ + * id - {String} Id for node │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A new node of the given type and id │ │ │ │ │ + */ │ │ │ │ │ + createNode: function(type, id) { │ │ │ │ │ + var node = document.createElementNS(this.xmlns, type); │ │ │ │ │ + if (id) { │ │ │ │ │ + node.setAttributeNS(null, "id", id); │ │ │ │ │ } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return entryNode; │ │ │ │ │ + /** │ │ │ │ │ + * Method: nodeTypeCompare │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {SVGDomElement} An SVG element │ │ │ │ │ + * type - {String} Kind of node │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ + */ │ │ │ │ │ + nodeTypeCompare: function(node, type) { │ │ │ │ │ + return (type == node.nodeName); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: initGmlParser │ │ │ │ │ - * Creates a GML parser. │ │ │ │ │ + * Method: createRenderRoot │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The specific render engine's root element │ │ │ │ │ */ │ │ │ │ │ - initGmlParser: function() { │ │ │ │ │ - this.gmlParser = new OpenLayers.Format.GML.v3({ │ │ │ │ │ - xy: this.xy, │ │ │ │ │ - featureNS: "http://example.com#feature", │ │ │ │ │ - internalProjection: this.internalProjection, │ │ │ │ │ - externalProjection: this.externalProjection │ │ │ │ │ - }); │ │ │ │ │ + createRenderRoot: function() { │ │ │ │ │ + var svg = this.nodeFactory(this.container.id + "_svgRoot", "svg"); │ │ │ │ │ + svg.style.display = "block"; │ │ │ │ │ + return svg; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildGeometryNode │ │ │ │ │ - * builds a GeoRSS node with a given geometry │ │ │ │ │ - * │ │ │ │ │ + * Method: createRoot │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * suffix - {String} suffix to append to the id │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + createRoot: function(suffix) { │ │ │ │ │ + return this.nodeFactory(this.container.id + suffix, "g"); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createDefs │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} A gml node. │ │ │ │ │ + * {DOMElement} The element to which we'll add the symbol definitions │ │ │ │ │ */ │ │ │ │ │ - buildGeometryNode: function(geometry) { │ │ │ │ │ - if (!this.gmlParser) { │ │ │ │ │ - this.initGmlParser(); │ │ │ │ │ + createDefs: function() { │ │ │ │ │ + var defs = this.nodeFactory(this.container.id + "_defs", "defs"); │ │ │ │ │ + this.rendererRoot.appendChild(defs); │ │ │ │ │ + return defs; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /************************************** │ │ │ │ │ + * * │ │ │ │ │ + * GEOMETRY DRAWING FUNCTIONS * │ │ │ │ │ + * * │ │ │ │ │ + **************************************/ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawPoint │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the point │ │ │ │ │ + */ │ │ │ │ │ + drawPoint: function(node, geometry) { │ │ │ │ │ + return this.drawCircle(node, geometry, 1); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawCircle │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * radius - {Float} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the circle │ │ │ │ │ + */ │ │ │ │ │ + drawCircle: function(node, geometry, radius) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = ((geometry.x - this.featureDx) / resolution + this.left); │ │ │ │ │ + var y = (this.top - geometry.y / resolution); │ │ │ │ │ + │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + node.setAttributeNS(null, "cx", x); │ │ │ │ │ + node.setAttributeNS(null, "cy", y); │ │ │ │ │ + node.setAttributeNS(null, "r", radius); │ │ │ │ │ + return node; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - var node = this.gmlParser.writeNode("feature:_geometry", geometry); │ │ │ │ │ - return node.firstChild; │ │ │ │ │ + │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildPersonConstructNode │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * value - {Object} │ │ │ │ │ - * │ │ │ │ │ + * Method: drawLineString │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} an Atom person construct node. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * >>> buildPersonConstructNode("author", {name: "John Smith"}) │ │ │ │ │ - * {<author><name>John Smith</name></author>} │ │ │ │ │ - * │ │ │ │ │ - * TODO: how to specify extension elements? Add to the oNames array? │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components of │ │ │ │ │ + * the linestring, or false if nothing could be drawn │ │ │ │ │ */ │ │ │ │ │ - buildPersonConstructNode: function(name, value) { │ │ │ │ │ - var oNames = ["uri", "email"]; │ │ │ │ │ - var personNode = this.createElementNSPlus("atom:" + name); │ │ │ │ │ - personNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:name", { │ │ │ │ │ - value: value.name │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ - for (var i = 0, ii = oNames.length; i < ii; i++) { │ │ │ │ │ - if (value[oNames[i]]) { │ │ │ │ │ - personNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:" + oNames[i], { │ │ │ │ │ - value: value[oNames[i]] │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + drawLineString: function(node, geometry) { │ │ │ │ │ + var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ + if (componentsResult.path) { │ │ │ │ │ + node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ + return (componentsResult.complete ? node : null); │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - return personNode; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFirstChildValue │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ + * Method: drawLinearRing │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ * node - {DOMElement} │ │ │ │ │ - * nsuri - {String} Child node namespace uri ("*" for any). │ │ │ │ │ - * name - {String} Child node name. │ │ │ │ │ - * def - {String} Optional string default to return if no child found. │ │ │ │ │ - * │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} The value of the first child with the given tag name. Returns │ │ │ │ │ - * default value or empty string if none found. │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ + * of the linear ring, or false if nothing could be drawn │ │ │ │ │ */ │ │ │ │ │ - getFirstChildValue: function(node, nsuri, name, def) { │ │ │ │ │ - var value; │ │ │ │ │ - var nodes = this.getElementsByTagNameNS(node, nsuri, name); │ │ │ │ │ - if (nodes && nodes.length > 0) { │ │ │ │ │ - value = this.getChildValue(nodes[0], def); │ │ │ │ │ + drawLinearRing: function(node, geometry) { │ │ │ │ │ + var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ + if (componentsResult.path) { │ │ │ │ │ + node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ + return (componentsResult.complete ? node : null); │ │ │ │ │ } else { │ │ │ │ │ - value = def; │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - return value; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseFeature │ │ │ │ │ - * Parse feature from an Atom entry node.. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} An Atom entry or feed node. │ │ │ │ │ - * │ │ │ │ │ + * Method: drawPolygon │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ + * of the polygon, or false if nothing could be drawn │ │ │ │ │ */ │ │ │ │ │ - parseFeature: function(node) { │ │ │ │ │ - var atomAttrib = {}; │ │ │ │ │ - var value = null; │ │ │ │ │ - var nodes = null; │ │ │ │ │ - var attval = null; │ │ │ │ │ - var atomns = this.namespaces.atom; │ │ │ │ │ + drawPolygon: function(node, geometry) { │ │ │ │ │ + var d = ""; │ │ │ │ │ + var draw = true; │ │ │ │ │ + var complete = true; │ │ │ │ │ + var linearRingResult, path; │ │ │ │ │ + for (var j = 0, len = geometry.components.length; j < len; j++) { │ │ │ │ │ + d += " M"; │ │ │ │ │ + linearRingResult = this.getComponentsString( │ │ │ │ │ + geometry.components[j].components, " "); │ │ │ │ │ + path = linearRingResult.path; │ │ │ │ │ + if (path) { │ │ │ │ │ + d += " " + path; │ │ │ │ │ + complete = linearRingResult.complete && complete; │ │ │ │ │ + } else { │ │ │ │ │ + draw = false; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + d += " z"; │ │ │ │ │ + if (draw) { │ │ │ │ │ + node.setAttributeNS(null, "d", d); │ │ │ │ │ + node.setAttributeNS(null, "fill-rule", "evenodd"); │ │ │ │ │ + return complete ? node : null; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // atomAuthor* │ │ │ │ │ - this.parsePersonConstructs(node, "author", atomAttrib); │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawRectangle │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the rectangle │ │ │ │ │ + */ │ │ │ │ │ + drawRectangle: function(node, geometry) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = ((geometry.x - this.featureDx) / resolution + this.left); │ │ │ │ │ + var y = (this.top - geometry.y / resolution); │ │ │ │ │ │ │ │ │ │ - // atomCategory* │ │ │ │ │ - nodes = this.getElementsByTagNameNS(node, atomns, "category"); │ │ │ │ │ - if (nodes.length > 0) { │ │ │ │ │ - atomAttrib.categories = []; │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ - value = {}; │ │ │ │ │ - value.term = nodes[i].getAttribute("term"); │ │ │ │ │ - attval = nodes[i].getAttribute("scheme"); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value.scheme = attval; │ │ │ │ │ - } │ │ │ │ │ - attval = nodes[i].getAttribute("label"); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value.label = attval; │ │ │ │ │ - } │ │ │ │ │ - atomAttrib.categories.push(value); │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + node.setAttributeNS(null, "x", x); │ │ │ │ │ + node.setAttributeNS(null, "y", y); │ │ │ │ │ + node.setAttributeNS(null, "width", geometry.width / resolution); │ │ │ │ │ + node.setAttributeNS(null, "height", geometry.height / resolution); │ │ │ │ │ + return node; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // atomContent? │ │ │ │ │ - nodes = this.getElementsByTagNameNS(node, atomns, "content"); │ │ │ │ │ - if (nodes.length > 0) { │ │ │ │ │ - value = {}; │ │ │ │ │ - attval = nodes[0].getAttribute("type"); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value.type = attval; │ │ │ │ │ - } │ │ │ │ │ - attval = nodes[0].getAttribute("src"); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value.src = attval; │ │ │ │ │ - } else { │ │ │ │ │ - if (value.type == "text" || │ │ │ │ │ - value.type == "html" || │ │ │ │ │ - value.type == null) { │ │ │ │ │ - value.value = this.getFirstChildValue( │ │ │ │ │ - node, │ │ │ │ │ - atomns, │ │ │ │ │ - "content", │ │ │ │ │ - null │ │ │ │ │ - ); │ │ │ │ │ - } else if (value.type == "xhtml" || │ │ │ │ │ - value.type.match(/(\+|\/)xml$/)) { │ │ │ │ │ - value.value = this.getChildEl(nodes[0]); │ │ │ │ │ - } else { // MUST be base64 encoded │ │ │ │ │ - value.value = this.getFirstChildValue( │ │ │ │ │ - node, │ │ │ │ │ - atomns, │ │ │ │ │ - "content", │ │ │ │ │ - null │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - atomAttrib.content = value; │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawText │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * style - │ │ │ │ │ + * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + */ │ │ │ │ │ + drawText: function(featureId, style, location) { │ │ │ │ │ + var drawOutline = (!!style.labelOutlineWidth); │ │ │ │ │ + // First draw text in halo color and size and overlay the │ │ │ │ │ + // normal text afterwards │ │ │ │ │ + if (drawOutline) { │ │ │ │ │ + var outlineStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ + outlineStyle.fontColor = outlineStyle.labelOutlineColor; │ │ │ │ │ + outlineStyle.fontStrokeColor = outlineStyle.labelOutlineColor; │ │ │ │ │ + outlineStyle.fontStrokeWidth = style.labelOutlineWidth; │ │ │ │ │ + if (style.labelOutlineOpacity) { │ │ │ │ │ + outlineStyle.fontOpacity = style.labelOutlineOpacity; │ │ │ │ │ } │ │ │ │ │ + delete outlineStyle.labelOutlineWidth; │ │ │ │ │ + this.drawText(featureId, outlineStyle, location); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // atomContributor* │ │ │ │ │ - this.parsePersonConstructs(node, "contributor", atomAttrib); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ │ │ │ │ │ - // atomId │ │ │ │ │ - atomAttrib.id = this.getFirstChildValue(node, atomns, "id", null); │ │ │ │ │ + var x = ((location.x - this.featureDx) / resolution + this.left); │ │ │ │ │ + var y = (location.y / resolution - this.top); │ │ │ │ │ │ │ │ │ │ - // atomLink* │ │ │ │ │ - nodes = this.getElementsByTagNameNS(node, atomns, "link"); │ │ │ │ │ - if (nodes.length > 0) { │ │ │ │ │ - atomAttrib.links = new Array(nodes.length); │ │ │ │ │ + var suffix = (drawOutline) ? this.LABEL_OUTLINE_SUFFIX : this.LABEL_ID_SUFFIX; │ │ │ │ │ + var label = this.nodeFactory(featureId + suffix, "text"); │ │ │ │ │ + │ │ │ │ │ + label.setAttributeNS(null, "x", x); │ │ │ │ │ + label.setAttributeNS(null, "y", -y); │ │ │ │ │ + │ │ │ │ │ + if (style.fontColor) { │ │ │ │ │ + label.setAttributeNS(null, "fill", style.fontColor); │ │ │ │ │ } │ │ │ │ │ - var oAtts = ["rel", "type", "hreflang", "title", "length"]; │ │ │ │ │ - for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ - value = {}; │ │ │ │ │ - value.href = nodes[i].getAttribute("href"); │ │ │ │ │ - for (var j = 0, jj = oAtts.length; j < jj; j++) { │ │ │ │ │ - attval = nodes[i].getAttribute(oAtts[j]); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value[oAtts[j]] = attval; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - atomAttrib.links[i] = value; │ │ │ │ │ + if (style.fontStrokeColor) { │ │ │ │ │ + label.setAttributeNS(null, "stroke", style.fontStrokeColor); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // atomPublished? │ │ │ │ │ - value = this.getFirstChildValue(node, atomns, "published", null); │ │ │ │ │ - if (value) { │ │ │ │ │ - atomAttrib.published = value; │ │ │ │ │ + if (style.fontStrokeWidth) { │ │ │ │ │ + label.setAttributeNS(null, "stroke-width", style.fontStrokeWidth); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // atomRights? │ │ │ │ │ - value = this.getFirstChildValue(node, atomns, "rights", null); │ │ │ │ │ - if (value) { │ │ │ │ │ - atomAttrib.rights = value; │ │ │ │ │ + if (style.fontOpacity) { │ │ │ │ │ + label.setAttributeNS(null, "opacity", style.fontOpacity); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontFamily) { │ │ │ │ │ + label.setAttributeNS(null, "font-family", style.fontFamily); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontSize) { │ │ │ │ │ + label.setAttributeNS(null, "font-size", style.fontSize); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontWeight) { │ │ │ │ │ + label.setAttributeNS(null, "font-weight", style.fontWeight); │ │ │ │ │ } │ │ │ │ │ + if (style.fontStyle) { │ │ │ │ │ + label.setAttributeNS(null, "font-style", style.fontStyle); │ │ │ │ │ + } │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + label.setAttributeNS(null, "pointer-events", "visible"); │ │ │ │ │ + label._featureId = featureId; │ │ │ │ │ + } else { │ │ │ │ │ + label.setAttributeNS(null, "pointer-events", "none"); │ │ │ │ │ + } │ │ │ │ │ + var align = style.labelAlign || OpenLayers.Renderer.defaultSymbolizer.labelAlign; │ │ │ │ │ + label.setAttributeNS(null, "text-anchor", │ │ │ │ │ + OpenLayers.Renderer.SVG.LABEL_ALIGN[align[0]] || "middle"); │ │ │ │ │ │ │ │ │ │ - // atomSource? -- not implemented │ │ │ │ │ + if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ + label.setAttributeNS(null, "dominant-baseline", │ │ │ │ │ + OpenLayers.Renderer.SVG.LABEL_ALIGN[align[1]] || "central"); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // atomSummary? │ │ │ │ │ - value = this.getFirstChildValue(node, atomns, "summary", null); │ │ │ │ │ - if (value) { │ │ │ │ │ - atomAttrib.summary = value; │ │ │ │ │ + var labelRows = style.label.split('\n'); │ │ │ │ │ + var numRows = labelRows.length; │ │ │ │ │ + while (label.childNodes.length > numRows) { │ │ │ │ │ + label.removeChild(label.lastChild); │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + var tspan = this.nodeFactory(featureId + suffix + "_tspan_" + i, "tspan"); │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + tspan._featureId = featureId; │ │ │ │ │ + tspan._geometry = location; │ │ │ │ │ + tspan._geometryClass = location.CLASS_NAME; │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.IS_GECKO === false) { │ │ │ │ │ + tspan.setAttributeNS(null, "baseline-shift", │ │ │ │ │ + OpenLayers.Renderer.SVG.LABEL_VSHIFT[align[1]] || "-35%"); │ │ │ │ │ + } │ │ │ │ │ + tspan.setAttribute("x", x); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + var vfactor = OpenLayers.Renderer.SVG.LABEL_VFACTOR[align[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5; │ │ │ │ │ + } │ │ │ │ │ + tspan.setAttribute("dy", (vfactor * (numRows - 1)) + "em"); │ │ │ │ │ + } else { │ │ │ │ │ + tspan.setAttribute("dy", "1em"); │ │ │ │ │ + } │ │ │ │ │ + tspan.textContent = (labelRows[i] === '') ? ' ' : labelRows[i]; │ │ │ │ │ + if (!tspan.parentNode) { │ │ │ │ │ + label.appendChild(tspan); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // atomTitle │ │ │ │ │ - atomAttrib.title = this.getFirstChildValue( │ │ │ │ │ - node, atomns, "title", null │ │ │ │ │ - ); │ │ │ │ │ + if (!label.parentNode) { │ │ │ │ │ + this.textRoot.appendChild(label); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // atomUpdated │ │ │ │ │ - atomAttrib.updated = this.getFirstChildValue( │ │ │ │ │ - node, atomns, "updated", null │ │ │ │ │ - ); │ │ │ │ │ + /** │ │ │ │ │ + * Method: getComponentString │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * components - {Array(<OpenLayers.Geometry.Point>)} Array of points │ │ │ │ │ + * separator - {String} character between coordinate pairs. Defaults to "," │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} hash with properties "path" (the string created from the │ │ │ │ │ + * components and "complete" (false if the renderer was unable to │ │ │ │ │ + * draw all components) │ │ │ │ │ + */ │ │ │ │ │ + getComponentsString: function(components, separator) { │ │ │ │ │ + var renderCmp = []; │ │ │ │ │ + var complete = true; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + var strings = []; │ │ │ │ │ + var str, component; │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + component = components[i]; │ │ │ │ │ + renderCmp.push(component); │ │ │ │ │ + str = this.getShortString(component); │ │ │ │ │ + if (str) { │ │ │ │ │ + strings.push(str); │ │ │ │ │ + } else { │ │ │ │ │ + // The current component is outside the valid range. Let's │ │ │ │ │ + // see if the previous or next component is inside the range. │ │ │ │ │ + // If so, add the coordinate of the intersection with the │ │ │ │ │ + // valid range bounds. │ │ │ │ │ + if (i > 0) { │ │ │ │ │ + if (this.getShortString(components[i - 1])) { │ │ │ │ │ + strings.push(this.clipLine(components[i], │ │ │ │ │ + components[i - 1])); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (i < len - 1) { │ │ │ │ │ + if (this.getShortString(components[i + 1])) { │ │ │ │ │ + strings.push(this.clipLine(components[i], │ │ │ │ │ + components[i + 1])); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + complete = false; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var featureAttrib = { │ │ │ │ │ - title: atomAttrib.title, │ │ │ │ │ - description: atomAttrib.summary, │ │ │ │ │ - atom: atomAttrib │ │ │ │ │ + return { │ │ │ │ │ + path: strings.join(separator || ","), │ │ │ │ │ + complete: complete │ │ │ │ │ }; │ │ │ │ │ - var geometry = this.parseLocations(node)[0]; │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry, featureAttrib); │ │ │ │ │ - feature.fid = atomAttrib.id; │ │ │ │ │ - return feature; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseFeatures │ │ │ │ │ - * Return features from an Atom entry or feed. │ │ │ │ │ - * │ │ │ │ │ + * Method: clipLine │ │ │ │ │ + * Given two points (one inside the valid range, and one outside), │ │ │ │ │ + * clips the line betweeen the two points so that the new points are both │ │ │ │ │ + * inside the valid range. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} An Atom entry or feed node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ + * badComponent - {<OpenLayers.Geometry.Point>} original geometry of the │ │ │ │ │ + * invalid point │ │ │ │ │ + * goodComponent - {<OpenLayers.Geometry.Point>} original geometry of the │ │ │ │ │ + * valid point │ │ │ │ │ + * Returns │ │ │ │ │ + * {String} the SVG coordinate pair of the clipped point (like │ │ │ │ │ + * getShortString), or an empty string if both passed componets are at │ │ │ │ │ + * the same point. │ │ │ │ │ */ │ │ │ │ │ - parseFeatures: function(node) { │ │ │ │ │ - var features = []; │ │ │ │ │ - var entries = this.getElementsByTagNameNS( │ │ │ │ │ - node, this.namespaces.atom, "entry" │ │ │ │ │ - ); │ │ │ │ │ - if (entries.length == 0) { │ │ │ │ │ - entries = [node]; │ │ │ │ │ + clipLine: function(badComponent, goodComponent) { │ │ │ │ │ + if (goodComponent.equals(badComponent)) { │ │ │ │ │ + return ""; │ │ │ │ │ } │ │ │ │ │ - for (var i = 0, ii = entries.length; i < ii; i++) { │ │ │ │ │ - features.push(this.parseFeature(entries[i])); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var maxX = this.MAX_PIXEL - this.translationParameters.x; │ │ │ │ │ + var maxY = this.MAX_PIXEL - this.translationParameters.y; │ │ │ │ │ + var x1 = (goodComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y1 = this.top - goodComponent.y / resolution; │ │ │ │ │ + var x2 = (badComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y2 = this.top - badComponent.y / resolution; │ │ │ │ │ + var k; │ │ │ │ │ + if (x2 < -maxX || x2 > maxX) { │ │ │ │ │ + k = (y2 - y1) / (x2 - x1); │ │ │ │ │ + x2 = x2 < 0 ? -maxX : maxX; │ │ │ │ │ + y2 = y1 + (x2 - x1) * k; │ │ │ │ │ } │ │ │ │ │ - return features; │ │ │ │ │ + if (y2 < -maxY || y2 > maxY) { │ │ │ │ │ + k = (x2 - x1) / (y2 - y1); │ │ │ │ │ + y2 = y2 < 0 ? -maxY : maxY; │ │ │ │ │ + x2 = x1 + (y2 - y1) * k; │ │ │ │ │ + } │ │ │ │ │ + return x2 + "," + y2; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseLocations │ │ │ │ │ - * Parse the locations from an Atom entry or feed. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: getShortString │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} An Atom entry or feed node. │ │ │ │ │ - * │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * Array({<OpenLayers.Geometry>}) │ │ │ │ │ + * {String} or false if point is outside the valid range │ │ │ │ │ */ │ │ │ │ │ - parseLocations: function(node) { │ │ │ │ │ - var georssns = this.namespaces.georss; │ │ │ │ │ + getShortString: function(point) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = ((point.x - this.featureDx) / resolution + this.left); │ │ │ │ │ + var y = (this.top - point.y / resolution); │ │ │ │ │ │ │ │ │ │ - var locations = { │ │ │ │ │ - components: [] │ │ │ │ │ - }; │ │ │ │ │ - var where = this.getElementsByTagNameNS(node, georssns, "where"); │ │ │ │ │ - if (where && where.length > 0) { │ │ │ │ │ - if (!this.gmlParser) { │ │ │ │ │ - this.initGmlParser(); │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, ii = where.length; i < ii; i++) { │ │ │ │ │ - this.gmlParser.readChildNodes(where[i], locations); │ │ │ │ │ - } │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + return x + "," + y; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var components = locations.components; │ │ │ │ │ - var point = this.getElementsByTagNameNS(node, georssns, "point"); │ │ │ │ │ - if (point && point.length > 0) { │ │ │ │ │ - for (var i = 0, ii = point.length; i < ii; i++) { │ │ │ │ │ - var xy = OpenLayers.String.trim( │ │ │ │ │ - point[i].firstChild.nodeValue │ │ │ │ │ - ).split(/\s+/); │ │ │ │ │ - if (xy.length != 2) { │ │ │ │ │ - xy = OpenLayers.String.trim( │ │ │ │ │ - point[i].firstChild.nodeValue │ │ │ │ │ - ).split(/\s*,\s*/); │ │ │ │ │ - } │ │ │ │ │ - components.push(new OpenLayers.Geometry.Point(xy[1], xy[0])); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: getPosition │ │ │ │ │ + * Finds the position of an svg node. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} hash with x and y properties, representing the coordinates │ │ │ │ │ + * within the svg coordinate system │ │ │ │ │ + */ │ │ │ │ │ + getPosition: function(node) { │ │ │ │ │ + return ({ │ │ │ │ │ + x: parseFloat(node.getAttributeNS(null, "cx")), │ │ │ │ │ + y: parseFloat(node.getAttributeNS(null, "cy")) │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: importSymbol │ │ │ │ │ + * add a new symbol definition from the rendererer's symbol hash │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * graphicName - {String} name of the symbol to import │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} - the imported symbol │ │ │ │ │ + */ │ │ │ │ │ + importSymbol: function(graphicName) { │ │ │ │ │ + if (!this.defs) { │ │ │ │ │ + // create svg defs tag │ │ │ │ │ + this.defs = this.createDefs(); │ │ │ │ │ } │ │ │ │ │ + var id = this.container.id + "-" + graphicName; │ │ │ │ │ │ │ │ │ │ - var line = this.getElementsByTagNameNS(node, georssns, "line"); │ │ │ │ │ - if (line && line.length > 0) { │ │ │ │ │ - var coords; │ │ │ │ │ - var p; │ │ │ │ │ - var points; │ │ │ │ │ - for (var i = 0, ii = line.length; i < ii; i++) { │ │ │ │ │ - coords = OpenLayers.String.trim( │ │ │ │ │ - line[i].firstChild.nodeValue │ │ │ │ │ - ).split(/\s+/); │ │ │ │ │ - points = []; │ │ │ │ │ - for (var j = 0, jj = coords.length; j < jj; j += 2) { │ │ │ │ │ - p = new OpenLayers.Geometry.Point(coords[j + 1], coords[j]); │ │ │ │ │ - points.push(p); │ │ │ │ │ - } │ │ │ │ │ - components.push( │ │ │ │ │ - new OpenLayers.Geometry.LineString(points) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + // check if symbol already exists in the defs │ │ │ │ │ + var existing = document.getElementById(id); │ │ │ │ │ + if (existing != null) { │ │ │ │ │ + return existing; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - var polygon = this.getElementsByTagNameNS(node, georssns, "polygon"); │ │ │ │ │ - if (polygon && polygon.length > 0) { │ │ │ │ │ - var coords; │ │ │ │ │ - var p; │ │ │ │ │ - var points; │ │ │ │ │ - for (var i = 0, ii = polygon.length; i < ii; i++) { │ │ │ │ │ - coords = OpenLayers.String.trim( │ │ │ │ │ - polygon[i].firstChild.nodeValue │ │ │ │ │ - ).split(/\s+/); │ │ │ │ │ - points = []; │ │ │ │ │ - for (var j = 0, jj = coords.length; j < jj; j += 2) { │ │ │ │ │ - p = new OpenLayers.Geometry.Point(coords[j + 1], coords[j]); │ │ │ │ │ - points.push(p); │ │ │ │ │ - } │ │ │ │ │ - components.push( │ │ │ │ │ - new OpenLayers.Geometry.Polygon( │ │ │ │ │ - [new OpenLayers.Geometry.LinearRing(points)] │ │ │ │ │ - ) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(graphicName + ' is not a valid symbol name'); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - for (var i = 0, ii = components.length; i < ii; i++) { │ │ │ │ │ - if (components[i]) { │ │ │ │ │ - components[i].transform( │ │ │ │ │ - this.externalProjection, │ │ │ │ │ - this.internalProjection │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + var symbolNode = this.nodeFactory(id, "symbol"); │ │ │ │ │ + var node = this.nodeFactory(null, "polygon"); │ │ │ │ │ + symbolNode.appendChild(node); │ │ │ │ │ + var symbolExtent = new OpenLayers.Bounds( │ │ │ │ │ + Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ + │ │ │ │ │ + var points = []; │ │ │ │ │ + var x, y; │ │ │ │ │ + for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ + symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ + symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ + symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ + points.push(x, ",", y); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return components; │ │ │ │ │ + node.setAttributeNS(null, "points", points.join(" ")); │ │ │ │ │ + │ │ │ │ │ + var width = symbolExtent.getWidth(); │ │ │ │ │ + var height = symbolExtent.getHeight(); │ │ │ │ │ + // create a viewBox three times as large as the symbol itself, │ │ │ │ │ + // to allow for strokeWidth being displayed correctly at the corners. │ │ │ │ │ + var viewBox = [symbolExtent.left - width, │ │ │ │ │ + symbolExtent.bottom - height, width * 3, height * 3 │ │ │ │ │ + ]; │ │ │ │ │ + symbolNode.setAttributeNS(null, "viewBox", viewBox.join(" ")); │ │ │ │ │ + this.symbolMetrics[id] = [ │ │ │ │ │ + Math.max(width, height), │ │ │ │ │ + symbolExtent.getCenterLonLat().lon, │ │ │ │ │ + symbolExtent.getCenterLonLat().lat │ │ │ │ │ + ]; │ │ │ │ │ + │ │ │ │ │ + this.defs.appendChild(symbolNode); │ │ │ │ │ + return symbolNode; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parsePersonConstruct │ │ │ │ │ - * Parse Atom person constructs from an Atom entry node. │ │ │ │ │ - * │ │ │ │ │ + * Method: getFeatureIdFromEvent │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} An Atom entry or feed node. │ │ │ │ │ - * name - {String} Construcy name ("author" or "contributor") │ │ │ │ │ - * data = {Object} Object in which to put parsed persons. │ │ │ │ │ + * evt - {Object} An <OpenLayers.Event> object │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * An {Object}. │ │ │ │ │ + * {String} A feature id or undefined. │ │ │ │ │ */ │ │ │ │ │ - parsePersonConstructs: function(node, name, data) { │ │ │ │ │ - var persons = []; │ │ │ │ │ - var atomns = this.namespaces.atom; │ │ │ │ │ - var nodes = this.getElementsByTagNameNS(node, atomns, name); │ │ │ │ │ - var oAtts = ["uri", "email"]; │ │ │ │ │ - for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ - var value = {}; │ │ │ │ │ - value.name = this.getFirstChildValue( │ │ │ │ │ - nodes[i], │ │ │ │ │ - atomns, │ │ │ │ │ - "name", │ │ │ │ │ - null │ │ │ │ │ - ); │ │ │ │ │ - for (var j = 0, jj = oAtts.length; j < jj; j++) { │ │ │ │ │ - var attval = this.getFirstChildValue( │ │ │ │ │ - nodes[i], │ │ │ │ │ - atomns, │ │ │ │ │ - oAtts[j], │ │ │ │ │ - null); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value[oAtts[j]] = attval; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - persons.push(value); │ │ │ │ │ - } │ │ │ │ │ - if (persons.length > 0) { │ │ │ │ │ - data[name + "s"] = persons; │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var featureId = OpenLayers.Renderer.Elements.prototype.getFeatureIdFromEvent.apply(this, arguments); │ │ │ │ │ + if (!featureId) { │ │ │ │ │ + var target = evt.target; │ │ │ │ │ + featureId = target.parentNode && target != this.rendererRoot ? │ │ │ │ │ + target.parentNode._featureId : undefined; │ │ │ │ │ } │ │ │ │ │ + return featureId; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.Atom" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.SVG" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.SVG.LABEL_ALIGN │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_ALIGN = { │ │ │ │ │ + "l": "start", │ │ │ │ │ + "r": "end", │ │ │ │ │ + "b": "bottom", │ │ │ │ │ + "t": "hanging" │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.SVG.LABEL_VSHIFT │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_VSHIFT = { │ │ │ │ │ + // according to │ │ │ │ │ + // http://www.w3.org/Graphics/SVG/Test/20061213/htmlObjectHarness/full-text-align-02-b.html │ │ │ │ │ + // a baseline-shift of -70% shifts the text exactly from the │ │ │ │ │ + // bottom to the top of the baseline, so -35% moves the text to │ │ │ │ │ + // the center of the baseline. │ │ │ │ │ + "t": "-70%", │ │ │ │ │ + "b": "0" │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.SVG.LABEL_VFACTOR │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_VFACTOR = { │ │ │ │ │ + "t": 0, │ │ │ │ │ + "b": -1 │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Function: OpenLayers.Renderer.SVG.preventDefault │ │ │ │ │ + * *Deprecated*. Use <OpenLayers.Event.preventDefault> method instead. │ │ │ │ │ + * Used to prevent default events (especially opening images in a new tab on │ │ │ │ │ + * ctrl-click) from being executed for externalGraphic symbols │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.SVG.preventDefault = function(e) { │ │ │ │ │ + OpenLayers.Event.preventDefault(e); │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/SOSCapabilities.js │ │ │ │ │ + OpenLayers/Protocol/CSW.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + * @requires OpenLayers/Protocol.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.SOSCapabilities │ │ │ │ │ - * Read SOS Capabilities. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + * Class: OpenLayers.Protocol.CSW │ │ │ │ │ + * Used to create a versioned CSW protocol. Default version is 2.0.2. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.SOSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.0.0". │ │ │ │ │ - */ │ │ │ │ │ - defaultVersion: "1.0.0", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.SOSCapabilities │ │ │ │ │ - * Create a new parser for SOS Capabilities. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return information about │ │ │ │ │ - * the service (offering and observedProperty mostly). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Info about the SOS │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SOSCapabilities" │ │ │ │ │ +OpenLayers.Protocol.CSW = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options, OpenLayers.Protocol.CSW.DEFAULTS │ │ │ │ │ + ); │ │ │ │ │ + var cls = OpenLayers.Protocol.CSW["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported CSW version: " + options.version; │ │ │ │ │ + } │ │ │ │ │ + return new cls(options); │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Protocol.CSW.DEFAULTS │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.CSW.DEFAULTS = { │ │ │ │ │ + "version": "2.0.2" │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/SOSGetFeatureOfInterest.js │ │ │ │ │ + OpenLayers/Protocol/SOS.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/GML/v3.js │ │ │ │ │ + * @requires OpenLayers/Protocol.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.SOSGetFeatureOfInterest │ │ │ │ │ - * Read and write SOS GetFeatureOfInterest. This is used to get to │ │ │ │ │ - * the location of the features (stations). The stations can have 1 or more │ │ │ │ │ - * sensors. │ │ │ │ │ + * Function: OpenLayers.Protocol.SOS │ │ │ │ │ + * Used to create a versioned SOS protocol. Default version is 1.0.0. │ │ │ │ │ * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol>} An SOS protocol for the given version. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.SOSGetFeatureOfInterest = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.XML, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: VERSION │ │ │ │ │ - * {String} 1.0.0 │ │ │ │ │ - */ │ │ │ │ │ - VERSION: "1.0.0", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ - gml: "http://www.opengis.net/gml", │ │ │ │ │ - sa: "http://www.opengis.net/sampling/1.0", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} Schema location │ │ │ │ │ - */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/sos/1.0 http://schemas.opengis.net/sos/1.0.0/sosAll.xsd", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - */ │ │ │ │ │ - defaultPrefix: "sos", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ - */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.SOSGetFeatureOfInterest │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Parse a GetFeatureOfInterest response and return an array of features │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var info = { │ │ │ │ │ - features: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readNode(data, info); │ │ │ │ │ - │ │ │ │ │ - var features = []; │ │ │ │ │ - for (var i = 0, len = info.features.length; i < len; i++) { │ │ │ │ │ - var container = info.features[i]; │ │ │ │ │ - // reproject features if needed │ │ │ │ │ - if (this.internalProjection && this.externalProjection && │ │ │ │ │ - container.components[0]) { │ │ │ │ │ - container.components[0].transform( │ │ │ │ │ - this.externalProjection, this.internalProjection │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector( │ │ │ │ │ - container.components[0], container.attributes); │ │ │ │ │ - features.push(feature); │ │ │ │ │ - } │ │ │ │ │ - return features; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "sa": { │ │ │ │ │ - "SamplingPoint": function(node, obj) { │ │ │ │ │ - // sampling point can also be without a featureMember if │ │ │ │ │ - // there is only 1 │ │ │ │ │ - if (!obj.attributes) { │ │ │ │ │ - var feature = { │ │ │ │ │ - attributes: {} │ │ │ │ │ - }; │ │ │ │ │ - obj.features.push(feature); │ │ │ │ │ - obj = feature; │ │ │ │ │ - } │ │ │ │ │ - obj.attributes.id = this.getAttributeNS(node, │ │ │ │ │ - this.namespaces.gml, "id"); │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "position": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "FeatureCollection": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "featureMember": function(node, obj) { │ │ │ │ │ - var feature = { │ │ │ │ │ - attributes: {} │ │ │ │ │ - }; │ │ │ │ │ - obj.features.push(feature); │ │ │ │ │ - this.readChildNodes(node, feature); │ │ │ │ │ - }, │ │ │ │ │ - "name": function(node, obj) { │ │ │ │ │ - obj.attributes.name = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "pos": function(node, obj) { │ │ │ │ │ - // we need to parse the srsName to get to the │ │ │ │ │ - // externalProjection, that's why we cannot use │ │ │ │ │ - // GML v3 for this │ │ │ │ │ - if (!this.externalProjection) { │ │ │ │ │ - this.externalProjection = new OpenLayers.Projection( │ │ │ │ │ - node.getAttribute("srsName")); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Format.GML.v3.prototype.readers.gml.pos.apply( │ │ │ │ │ - this, [node, obj]); │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.GML.v3.prototype.readers.gml) │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ - */ │ │ │ │ │ - writers: { │ │ │ │ │ - "sos": { │ │ │ │ │ - "GetFeatureOfInterest": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("GetFeatureOfInterest", { │ │ │ │ │ - attributes: { │ │ │ │ │ - version: this.VERSION, │ │ │ │ │ - service: 'SOS', │ │ │ │ │ - "xsi:schemaLocation": this.schemaLocation │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - for (var i = 0, len = options.fois.length; i < len; i++) { │ │ │ │ │ - this.writeNode("FeatureOfInterestId", { │ │ │ │ │ - foi: options.fois[i] │ │ │ │ │ - }, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "FeatureOfInterestId": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("FeatureOfInterestId", { │ │ │ │ │ - value: options.foi │ │ │ │ │ - }); │ │ │ │ │ - return node; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SOSGetFeatureOfInterest" │ │ │ │ │ +OpenLayers.Protocol.SOS = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options, OpenLayers.Protocol.SOS.DEFAULTS │ │ │ │ │ + ); │ │ │ │ │ + var cls = OpenLayers.Protocol.SOS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported SOS version: " + options.version; │ │ │ │ │ + } │ │ │ │ │ + return new cls(options); │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Protocol.SOS.DEFAULTS │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.SOS.DEFAULTS = { │ │ │ │ │ + "version": "1.0.0" │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSDescribeLayer.js │ │ │ │ │ + OpenLayers/Protocol/WFS.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + * @requires OpenLayers/Protocol.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WMSDescribeLayer │ │ │ │ │ - * Read SLD WMS DescribeLayer response │ │ │ │ │ - * DescribeLayer is meant to couple WMS to WFS and WCS │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + * Class: OpenLayers.Protocol.WFS │ │ │ │ │ + * Used to create a versioned WFS protocol. Default version is 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol>} A WFS protocol of the given version. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var protocol = new OpenLayers.Protocol.WFS({ │ │ │ │ │ + * version: "1.1.0", │ │ │ │ │ + * url: "http://demo.opengeo.org/geoserver/wfs", │ │ │ │ │ + * featureType: "tasmania_roads", │ │ │ │ │ + * featureNS: "http://www.openplans.org/topp", │ │ │ │ │ + * geometryName: "the_geom" │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * See the protocols for specific WFS versions for more detail. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WMSDescribeLayer = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.1.1". │ │ │ │ │ - */ │ │ │ │ │ - defaultVersion: "1.1.1", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMSDescribeLayer │ │ │ │ │ - * Create a new parser for WMS DescribeLayer responses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read DescribeLayer data from a string, and return the response. │ │ │ │ │ - * The OGC currently defines 2 formats which are allowed for output, │ │ │ │ │ - * so we need to parse these 2 types │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} Array of {<LayerDescription>} objects which have: │ │ │ │ │ - * - {String} owsType: WFS/WCS │ │ │ │ │ - * - {String} owsURL: the online resource │ │ │ │ │ - * - {String} typeName: the name of the typename on the service │ │ │ │ │ - */ │ │ │ │ │ +OpenLayers.Protocol.WFS = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options, OpenLayers.Protocol.WFS.DEFAULTS │ │ │ │ │ + ); │ │ │ │ │ + var cls = OpenLayers.Protocol.WFS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported WFS version: " + options.version; │ │ │ │ │ + } │ │ │ │ │ + return new cls(options); │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSDescribeLayer" │ │ │ │ │ +/** │ │ │ │ │ + * Function: fromWMSLayer │ │ │ │ │ + * Convenience function to create a WFS protocol from a WMS layer. This makes │ │ │ │ │ + * the assumption that a WFS requests can be issued at the same URL as │ │ │ │ │ + * WMS requests and that a WFS featureType exists with the same name as the │ │ │ │ │ + * WMS layer. │ │ │ │ │ + * │ │ │ │ │ + * This function is designed to auto-configure <url>, <featureType>, │ │ │ │ │ + * <featurePrefix> and <srsName> for WFS <version> 1.1.0. Note that │ │ │ │ │ + * srsName matching with the WMS layer will not work with WFS 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer.WMS>} WMS layer that has a matching WFS │ │ │ │ │ + * FeatureType at the same server url with the same typename. │ │ │ │ │ + * options - {Object} Default properties to be set on the protocol. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.WFS>} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.WFS.fromWMSLayer = function(layer, options) { │ │ │ │ │ + var typeName, featurePrefix; │ │ │ │ │ + var param = layer.params["LAYERS"]; │ │ │ │ │ + var parts = (OpenLayers.Util.isArray(param) ? param[0] : param).split(":"); │ │ │ │ │ + if (parts.length > 1) { │ │ │ │ │ + featurePrefix = parts[0]; │ │ │ │ │ + } │ │ │ │ │ + typeName = parts.pop(); │ │ │ │ │ + var protocolOptions = { │ │ │ │ │ + url: layer.url, │ │ │ │ │ + featureType: typeName, │ │ │ │ │ + featurePrefix: featurePrefix, │ │ │ │ │ + srsName: layer.projection && layer.projection.getCode() || │ │ │ │ │ + layer.map && layer.map.getProjectionObject().getCode(), │ │ │ │ │ + version: "1.1.0" │ │ │ │ │ + }; │ │ │ │ │ + return new OpenLayers.Protocol.WFS(OpenLayers.Util.applyDefaults( │ │ │ │ │ + options, protocolOptions │ │ │ │ │ + )); │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Protocol.WFS.DEFAULTS │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.WFS.DEFAULTS = { │ │ │ │ │ + "version": "1.0.0" │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/SOSGetObservation.js │ │ │ │ │ + OpenLayers/Protocol/HTTP.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/SOSGetFeatureOfInterest.js │ │ │ │ │ + * @requires OpenLayers/Protocol.js │ │ │ │ │ + * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.SOSGetObservation │ │ │ │ │ - * Read and write SOS GetObersation (to get the actual values from a sensor) │ │ │ │ │ - * version 1.0.0 │ │ │ │ │ + * if application uses the query string, for example, for BBOX parameters, │ │ │ │ │ + * OpenLayers/Format/QueryStringFilter.js should be included in the build config file │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.HTTP │ │ │ │ │ + * A basic HTTP protocol for vector layers. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Protocol.HTTP> constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * - <OpenLayers.Protocol> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.SOSGetObservation = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Protocol.HTTP = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + * Property: url │ │ │ │ │ + * {String} Service URL, read-only, set through the options │ │ │ │ │ + * passed to constructor. │ │ │ │ │ */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - ows: "http://www.opengis.net/ows", │ │ │ │ │ - gml: "http://www.opengis.net/gml", │ │ │ │ │ - sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ - ogc: "http://www.opengis.net/ogc", │ │ │ │ │ - om: "http://www.opengis.net/om/1.0", │ │ │ │ │ - sa: "http://www.opengis.net/sampling/1.0", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ - }, │ │ │ │ │ + url: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ + * Property: headers │ │ │ │ │ + * {Object} HTTP request headers, read-only, set through the options │ │ │ │ │ + * passed to the constructor, │ │ │ │ │ + * Example: {'Content-Type': 'plain/text'} │ │ │ │ │ */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ - }, │ │ │ │ │ + headers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: VERSION │ │ │ │ │ - * {String} 1.0.0 │ │ │ │ │ + * Property: params │ │ │ │ │ + * {Object} Parameters of GET requests, read-only, set through the options │ │ │ │ │ + * passed to the constructor, │ │ │ │ │ + * Example: {'bbox': '5,5,5,5'} │ │ │ │ │ */ │ │ │ │ │ - VERSION: "1.0.0", │ │ │ │ │ + params: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} Schema location │ │ │ │ │ + * Property: callback │ │ │ │ │ + * {Object} Function to be called when the <read>, <create>, │ │ │ │ │ + * <update>, <delete> or <commit> operation completes, read-only, │ │ │ │ │ + * set through the options passed to the constructor. │ │ │ │ │ */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/sos/1.0 http://schemas.opengis.net/sos/1.0.0/sosGetObservation.xsd", │ │ │ │ │ + callback: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ + * Property: scope │ │ │ │ │ + * {Object} Callback execution scope, read-only, set through the │ │ │ │ │ + * options passed to the constructor. │ │ │ │ │ */ │ │ │ │ │ - defaultPrefix: "sos", │ │ │ │ │ + scope: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.SOSGetObservation │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * APIProperty: readWithPOST │ │ │ │ │ + * {Boolean} true if read operations are done with POST requests │ │ │ │ │ + * instead of GET, defaults to false. │ │ │ │ │ */ │ │ │ │ │ + readWithPOST: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object containing the measurements │ │ │ │ │ + * APIProperty: updateWithPOST │ │ │ │ │ + * {Boolean} true if update operations are done with POST requests │ │ │ │ │ + * defaults to false. │ │ │ │ │ */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ - } │ │ │ │ │ - var info = { │ │ │ │ │ - measurements: [], │ │ │ │ │ - observations: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readNode(data, info); │ │ │ │ │ - return info; │ │ │ │ │ - }, │ │ │ │ │ + updateWithPOST: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: write │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} An SOS GetObservation request XML string. │ │ │ │ │ + * APIProperty: deleteWithPOST │ │ │ │ │ + * {Boolean} true if delete operations are done with POST requests │ │ │ │ │ + * defaults to false. │ │ │ │ │ + * if true, POST data is set to output of format.write(). │ │ │ │ │ */ │ │ │ │ │ - write: function(options) { │ │ │ │ │ - var node = this.writeNode("sos:GetObservation", options); │ │ │ │ │ - node.setAttribute("xmlns:om", this.namespaces.om); │ │ │ │ │ - node.setAttribute("xmlns:ogc", this.namespaces.ogc); │ │ │ │ │ - this.setAttributeNS( │ │ │ │ │ - node, this.namespaces.xsi, │ │ │ │ │ - "xsi:schemaLocation", this.schemaLocation │ │ │ │ │ - ); │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [node]); │ │ │ │ │ - }, │ │ │ │ │ + deleteWithPOST: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ + * Property: wildcarded. │ │ │ │ │ + * {Boolean} If true percent signs are added around values │ │ │ │ │ + * read from LIKE filters, for example if the protocol │ │ │ │ │ + * read method is passed a LIKE filter whose property │ │ │ │ │ + * is "foo" and whose value is "bar" the string │ │ │ │ │ + * "foo__ilike=%bar%" will be sent in the query string; │ │ │ │ │ + * defaults to false. │ │ │ │ │ */ │ │ │ │ │ - readers: { │ │ │ │ │ - "om": { │ │ │ │ │ - "ObservationCollection": function(node, obj) { │ │ │ │ │ - obj.id = this.getAttributeNS(node, this.namespaces.gml, "id"); │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "member": function(node, observationCollection) { │ │ │ │ │ - this.readChildNodes(node, observationCollection); │ │ │ │ │ - }, │ │ │ │ │ - "Measurement": function(node, observationCollection) { │ │ │ │ │ - var measurement = {}; │ │ │ │ │ - observationCollection.measurements.push(measurement); │ │ │ │ │ - this.readChildNodes(node, measurement); │ │ │ │ │ - }, │ │ │ │ │ - "Observation": function(node, observationCollection) { │ │ │ │ │ - var observation = {}; │ │ │ │ │ - observationCollection.observations.push(observation); │ │ │ │ │ - this.readChildNodes(node, observation); │ │ │ │ │ - }, │ │ │ │ │ - "samplingTime": function(node, measurement) { │ │ │ │ │ - var samplingTime = {}; │ │ │ │ │ - measurement.samplingTime = samplingTime; │ │ │ │ │ - this.readChildNodes(node, samplingTime); │ │ │ │ │ - }, │ │ │ │ │ - "observedProperty": function(node, measurement) { │ │ │ │ │ - measurement.observedProperty = │ │ │ │ │ - this.getAttributeNS(node, this.namespaces.xlink, "href"); │ │ │ │ │ - this.readChildNodes(node, measurement); │ │ │ │ │ - }, │ │ │ │ │ - "procedure": function(node, measurement) { │ │ │ │ │ - measurement.procedure = │ │ │ │ │ - this.getAttributeNS(node, this.namespaces.xlink, "href"); │ │ │ │ │ - this.readChildNodes(node, measurement); │ │ │ │ │ - }, │ │ │ │ │ - "featureOfInterest": function(node, observation) { │ │ │ │ │ - var foi = { │ │ │ │ │ - features: [] │ │ │ │ │ - }; │ │ │ │ │ - observation.fois = []; │ │ │ │ │ - observation.fois.push(foi); │ │ │ │ │ - this.readChildNodes(node, foi); │ │ │ │ │ - // postprocessing to get actual features │ │ │ │ │ - var features = []; │ │ │ │ │ - for (var i = 0, len = foi.features.length; i < len; i++) { │ │ │ │ │ - var feature = foi.features[i]; │ │ │ │ │ - features.push(new OpenLayers.Feature.Vector( │ │ │ │ │ - feature.components[0], feature.attributes)); │ │ │ │ │ - } │ │ │ │ │ - foi.features = features; │ │ │ │ │ - }, │ │ │ │ │ - "result": function(node, measurement) { │ │ │ │ │ - var result = {}; │ │ │ │ │ - measurement.result = result; │ │ │ │ │ - if (this.getChildValue(node) !== '') { │ │ │ │ │ - result.value = this.getChildValue(node); │ │ │ │ │ - result.uom = node.getAttribute("uom"); │ │ │ │ │ - } else { │ │ │ │ │ - this.readChildNodes(node, result); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "sa": OpenLayers.Format.SOSGetFeatureOfInterest.prototype.readers.sa, │ │ │ │ │ - "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "TimeInstant": function(node, samplingTime) { │ │ │ │ │ - var timeInstant = {}; │ │ │ │ │ - samplingTime.timeInstant = timeInstant; │ │ │ │ │ - this.readChildNodes(node, timeInstant); │ │ │ │ │ - }, │ │ │ │ │ - "timePosition": function(node, timeInstant) { │ │ │ │ │ - timeInstant.timePosition = this.getChildValue(node); │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.SOSGetFeatureOfInterest.prototype.readers.gml) │ │ │ │ │ - }, │ │ │ │ │ + wildcarded: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ + * APIProperty: srsInBBOX │ │ │ │ │ + * {Boolean} Include the SRS identifier in BBOX query string parameter. │ │ │ │ │ + * Default is false. If true and the layer has a projection object set, │ │ │ │ │ + * any BBOX filter will be serialized with a fifth item identifying the │ │ │ │ │ + * projection. E.g. bbox=-1000,-1000,1000,1000,EPSG:900913 │ │ │ │ │ */ │ │ │ │ │ - writers: { │ │ │ │ │ - "sos": { │ │ │ │ │ - "GetObservation": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("GetObservation", { │ │ │ │ │ - attributes: { │ │ │ │ │ - version: this.VERSION, │ │ │ │ │ - service: 'SOS' │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - this.writeNode("offering", options, node); │ │ │ │ │ - if (options.eventTime) { │ │ │ │ │ - this.writeNode("eventTime", options, node); │ │ │ │ │ - } │ │ │ │ │ - for (var procedure in options.procedures) { │ │ │ │ │ - this.writeNode("procedure", options.procedures[procedure], node); │ │ │ │ │ - } │ │ │ │ │ - for (var observedProperty in options.observedProperties) { │ │ │ │ │ - this.writeNode("observedProperty", options.observedProperties[observedProperty], node); │ │ │ │ │ - } │ │ │ │ │ - if (options.foi) { │ │ │ │ │ - this.writeNode("featureOfInterest", options.foi, node); │ │ │ │ │ - } │ │ │ │ │ - this.writeNode("responseFormat", options, node); │ │ │ │ │ - if (options.resultModel) { │ │ │ │ │ - this.writeNode("resultModel", options, node); │ │ │ │ │ - } │ │ │ │ │ - if (options.responseMode) { │ │ │ │ │ - this.writeNode("responseMode", options, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "featureOfInterest": function(foi) { │ │ │ │ │ - var node = this.createElementNSPlus("featureOfInterest"); │ │ │ │ │ - this.writeNode("ObjectID", foi.objectId, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "ObjectID": function(options) { │ │ │ │ │ - return this.createElementNSPlus("ObjectID", { │ │ │ │ │ - value: options │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "responseFormat": function(options) { │ │ │ │ │ - return this.createElementNSPlus("responseFormat", { │ │ │ │ │ - value: options.responseFormat │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "procedure": function(procedure) { │ │ │ │ │ - return this.createElementNSPlus("procedure", { │ │ │ │ │ - value: procedure │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "offering": function(options) { │ │ │ │ │ - return this.createElementNSPlus("offering", { │ │ │ │ │ - value: options.offering │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "observedProperty": function(observedProperty) { │ │ │ │ │ - return this.createElementNSPlus("observedProperty", { │ │ │ │ │ - value: observedProperty │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "eventTime": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("eventTime"); │ │ │ │ │ - if (options.eventTime === 'latest') { │ │ │ │ │ - this.writeNode("ogc:TM_Equals", options, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "resultModel": function(options) { │ │ │ │ │ - return this.createElementNSPlus("resultModel", { │ │ │ │ │ - value: options.resultModel │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "responseMode": function(options) { │ │ │ │ │ - return this.createElementNSPlus("responseMode", { │ │ │ │ │ - value: options.responseMode │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "ogc": { │ │ │ │ │ - "TM_Equals": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:TM_Equals"); │ │ │ │ │ - this.writeNode("ogc:PropertyName", { │ │ │ │ │ - property: "urn:ogc:data:time:iso8601" │ │ │ │ │ - }, node); │ │ │ │ │ - if (options.eventTime === 'latest') { │ │ │ │ │ - this.writeNode("gml:TimeInstant", { │ │ │ │ │ - value: 'latest' │ │ │ │ │ - }, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "PropertyName": function(options) { │ │ │ │ │ - return this.createElementNSPlus("ogc:PropertyName", { │ │ │ │ │ - value: options.property │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "gml": { │ │ │ │ │ - "TimeInstant": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:TimeInstant"); │ │ │ │ │ - this.writeNode("gml:timePosition", options, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "timePosition": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:timePosition", { │ │ │ │ │ - value: options.value │ │ │ │ │ - }); │ │ │ │ │ - return node; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SOSGetObservation" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/Context.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.Context │ │ │ │ │ - * Base class for both Format.WMC and Format.OWSContext │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.Context = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + srsInBBOX: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: layerOptions │ │ │ │ │ - * {Object} Default options for layers created by the parser. These │ │ │ │ │ - * options are overridden by the options which are read from the │ │ │ │ │ - * capabilities document. │ │ │ │ │ + * Constructor: OpenLayers.Protocol.HTTP │ │ │ │ │ + * A class for giving layers generic HTTP protocol. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + * │ │ │ │ │ + * Valid options include: │ │ │ │ │ + * url - {String} │ │ │ │ │ + * headers - {Object} │ │ │ │ │ + * params - {Object} URL parameters for GET requests │ │ │ │ │ + * format - {<OpenLayers.Format>} │ │ │ │ │ + * callback - {Function} │ │ │ │ │ + * scope - {Object} │ │ │ │ │ */ │ │ │ │ │ - layerOptions: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + this.params = {}; │ │ │ │ │ + this.headers = {}; │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ + var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ + wildcarded: this.wildcarded, │ │ │ │ │ + srsInBBOX: this.srsInBBOX │ │ │ │ │ + }); │ │ │ │ │ + this.filterToParams = function(filter, params) { │ │ │ │ │ + return format.write(filter, params); │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: layerParams │ │ │ │ │ - * {Object} Default parameters for layers created by the parser. This │ │ │ │ │ - * can be used e.g. to override DEFAULT_PARAMS for │ │ │ │ │ - * OpenLayers.Layer.WMS. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up the protocol. │ │ │ │ │ */ │ │ │ │ │ - layerParams: null, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.params = null; │ │ │ │ │ + this.headers = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.Context │ │ │ │ │ - * Create a new parser for Context documents. │ │ │ │ │ + * APIMethod: filterToParams │ │ │ │ │ + * Optional method to translate an <OpenLayers.Filter> object into an object │ │ │ │ │ + * that can be serialized as request query string provided. If a custom │ │ │ │ │ + * method is not provided, the filter will be serialized using the │ │ │ │ │ + * <OpenLayers.Format.QueryStringFilter> class. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * filter - {<OpenLayers.Filter>} filter to convert. │ │ │ │ │ + * params - {Object} The parameters object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The resulting parameters object. │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIMethod: read │ │ │ │ │ - * Read Context data from a string, and return an object with map │ │ │ │ │ - * properties and a list of layers. │ │ │ │ │ + * Construct a request for reading new features. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * options - {Object} The options object must contain a map property. If │ │ │ │ │ - * the map property is a string, it must be the id of a dom element │ │ │ │ │ - * where the new map will be placed. If the map property is an │ │ │ │ │ - * <OpenLayers.Map>, the layers from the context document will be added │ │ │ │ │ - * to the map. │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * url - {String} Url for the request. │ │ │ │ │ + * params - {Object} Parameters to get serialized as a query string. │ │ │ │ │ + * headers - {Object} Headers to be set on the request. │ │ │ │ │ + * filter - {<OpenLayers.Filter>} Filter to get serialized as a │ │ │ │ │ + * query string. │ │ │ │ │ + * readWithPOST - {Boolean} If the request should be done with POST. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Map>} A map based on the context. │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} A response object, whose "priv" property │ │ │ │ │ + * references the HTTP request, this object is also passed to the │ │ │ │ │ + * callback function when the request completes, its "features" property │ │ │ │ │ + * is then populated with the features received from the server. │ │ │ │ │ */ │ │ │ │ │ - read: function(data, options) { │ │ │ │ │ - var context = OpenLayers.Format.XML.VersionedOGC.prototype.read.apply(this, │ │ │ │ │ - arguments); │ │ │ │ │ - var map; │ │ │ │ │ - if (options && options.map) { │ │ │ │ │ - this.context = context; │ │ │ │ │ - if (options.map instanceof OpenLayers.Map) { │ │ │ │ │ - map = this.mergeContextToMap(context, options.map); │ │ │ │ │ - } else { │ │ │ │ │ - var mapOptions = options.map; │ │ │ │ │ - if (OpenLayers.Util.isElement(mapOptions) || │ │ │ │ │ - typeof mapOptions == "string") { │ │ │ │ │ - // we assume mapOptions references a div │ │ │ │ │ - // element │ │ │ │ │ - mapOptions = { │ │ │ │ │ - div: mapOptions │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - map = this.contextToMap(context, mapOptions); │ │ │ │ │ - } │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options.params, this.options.params); │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + if (options.filter && this.filterToParams) { │ │ │ │ │ + options.params = this.filterToParams( │ │ │ │ │ + options.filter, options.params │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + var readWithPOST = (options.readWithPOST !== undefined) ? │ │ │ │ │ + options.readWithPOST : this.readWithPOST; │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + if (readWithPOST) { │ │ │ │ │ + var headers = options.headers || {}; │ │ │ │ │ + headers["Content-Type"] = "application/x-www-form-urlencoded"; │ │ │ │ │ + resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ + data: OpenLayers.Util.getParameterString(options.params), │ │ │ │ │ + headers: headers │ │ │ │ │ + }); │ │ │ │ │ } else { │ │ │ │ │ - // not documented as part of the API, provided as a non-API option │ │ │ │ │ - map = context; │ │ │ │ │ + resp.priv = OpenLayers.Request.GET({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - return map; │ │ │ │ │ + return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getLayerFromContext │ │ │ │ │ - * Create a WMS layer from a layerContext object. │ │ │ │ │ + * Method: handleRead │ │ │ │ │ + * Individual callbacks are created for read, create and update, should │ │ │ │ │ + * a subclass need to override each one separately. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layerContext - {Object} An object representing a WMS layer. │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * the user callback. │ │ │ │ │ + * options - {Object} The user options passed to the read call. │ │ │ │ │ + */ │ │ │ │ │ + handleRead: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: create │ │ │ │ │ + * Construct a request for writing newly created features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.WMS>} A WMS layer. │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, whose "priv" property references the HTTP request, this │ │ │ │ │ + * object is also passed to the callback function when the request │ │ │ │ │ + * completes, its "features" property is then populated with the │ │ │ │ │ + * the features received from the server. │ │ │ │ │ */ │ │ │ │ │ - getLayerFromContext: function(layerContext) { │ │ │ │ │ - var i, len; │ │ │ │ │ - // fill initial options object from layerContext │ │ │ │ │ - var options = { │ │ │ │ │ - queryable: layerContext.queryable, //keep queryable for api compatibility │ │ │ │ │ - visibility: layerContext.visibility, │ │ │ │ │ - maxExtent: layerContext.maxExtent, │ │ │ │ │ - metadata: OpenLayers.Util.applyDefaults(layerContext.metadata, { │ │ │ │ │ - styles: layerContext.styles, │ │ │ │ │ - formats: layerContext.formats, │ │ │ │ │ - "abstract": layerContext["abstract"], │ │ │ │ │ - dataURL: layerContext.dataURL │ │ │ │ │ - }), │ │ │ │ │ - numZoomLevels: layerContext.numZoomLevels, │ │ │ │ │ - units: layerContext.units, │ │ │ │ │ - isBaseLayer: layerContext.isBaseLayer, │ │ │ │ │ - opacity: layerContext.opacity, │ │ │ │ │ - displayInLayerSwitcher: layerContext.displayInLayerSwitcher, │ │ │ │ │ - singleTile: layerContext.singleTile, │ │ │ │ │ - tileSize: (layerContext.tileSize) ? │ │ │ │ │ - new OpenLayers.Size( │ │ │ │ │ - layerContext.tileSize.width, │ │ │ │ │ - layerContext.tileSize.height │ │ │ │ │ - ) : undefined, │ │ │ │ │ - minScale: layerContext.minScale || layerContext.maxScaleDenominator, │ │ │ │ │ - maxScale: layerContext.maxScale || layerContext.minScaleDenominator, │ │ │ │ │ - srs: layerContext.srs, │ │ │ │ │ - dimensions: layerContext.dimensions, │ │ │ │ │ - metadataURL: layerContext.metadataURL │ │ │ │ │ - }; │ │ │ │ │ - if (this.layerOptions) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.layerOptions); │ │ │ │ │ - } │ │ │ │ │ + create: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ │ │ │ │ │ - var params = { │ │ │ │ │ - layers: layerContext.name, │ │ │ │ │ - transparent: layerContext.transparent, │ │ │ │ │ - version: layerContext.version │ │ │ │ │ - }; │ │ │ │ │ - if (layerContext.formats && layerContext.formats.length > 0) { │ │ │ │ │ - // set default value for params if current attribute is not positionned │ │ │ │ │ - params.format = layerContext.formats[0].value; │ │ │ │ │ - for (i = 0, len = layerContext.formats.length; i < len; i++) { │ │ │ │ │ - var format = layerContext.formats[i]; │ │ │ │ │ - if (format.current == true) { │ │ │ │ │ - params.format = format.value; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (layerContext.styles && layerContext.styles.length > 0) { │ │ │ │ │ - for (i = 0, len = layerContext.styles.length; i < len; i++) { │ │ │ │ │ - var style = layerContext.styles[i]; │ │ │ │ │ - if (style.current == true) { │ │ │ │ │ - // three style types to consider │ │ │ │ │ - // 1) linked SLD │ │ │ │ │ - // 2) inline SLD │ │ │ │ │ - // 3) named style │ │ │ │ │ - if (style.href) { │ │ │ │ │ - params.sld = style.href; │ │ │ │ │ - } else if (style.body) { │ │ │ │ │ - params.sld_body = style.body; │ │ │ │ │ - } else { │ │ │ │ │ - params.styles = style.name; │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.layerParams) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(params, this.layerParams); │ │ │ │ │ - } │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: features, │ │ │ │ │ + requestType: "create" │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - var layer = null; │ │ │ │ │ - var service = layerContext.service; │ │ │ │ │ - if (service == OpenLayers.Format.Context.serviceTypes.WFS) { │ │ │ │ │ - options.strategies = [new OpenLayers.Strategy.BBOX()]; │ │ │ │ │ - options.protocol = new OpenLayers.Protocol.WFS({ │ │ │ │ │ - url: layerContext.url, │ │ │ │ │ - // since we do not know featureNS, let the protocol │ │ │ │ │ - // determine it automagically using featurePrefix │ │ │ │ │ - featurePrefix: layerContext.name.split(":")[0], │ │ │ │ │ - featureType: layerContext.name.split(":").pop() │ │ │ │ │ - }); │ │ │ │ │ - layer = new OpenLayers.Layer.Vector( │ │ │ │ │ - layerContext.title || layerContext.name, │ │ │ │ │ - options │ │ │ │ │ - ); │ │ │ │ │ - } else if (service == OpenLayers.Format.Context.serviceTypes.KML) { │ │ │ │ │ - // use a vector layer with an HTTP Protcol and a Fixed strategy │ │ │ │ │ - options.strategies = [new OpenLayers.Strategy.Fixed()]; │ │ │ │ │ - options.protocol = new OpenLayers.Protocol.HTTP({ │ │ │ │ │ - url: layerContext.url, │ │ │ │ │ - format: new OpenLayers.Format.KML() │ │ │ │ │ - }); │ │ │ │ │ - layer = new OpenLayers.Layer.Vector( │ │ │ │ │ - layerContext.title || layerContext.name, │ │ │ │ │ - options │ │ │ │ │ - ); │ │ │ │ │ - } else if (service == OpenLayers.Format.Context.serviceTypes.GML) { │ │ │ │ │ - // use a vector layer with a HTTP Protocol and a Fixed strategy │ │ │ │ │ - options.strategies = [new OpenLayers.Strategy.Fixed()]; │ │ │ │ │ - options.protocol = new OpenLayers.Protocol.HTTP({ │ │ │ │ │ - url: layerContext.url, │ │ │ │ │ - format: new OpenLayers.Format.GML() │ │ │ │ │ - }); │ │ │ │ │ - layer = new OpenLayers.Layer.Vector( │ │ │ │ │ - layerContext.title || layerContext.name, │ │ │ │ │ - options │ │ │ │ │ - ); │ │ │ │ │ - } else if (layerContext.features) { │ │ │ │ │ - // inline GML or KML features │ │ │ │ │ - layer = new OpenLayers.Layer.Vector( │ │ │ │ │ - layerContext.title || layerContext.name, │ │ │ │ │ - options │ │ │ │ │ - ); │ │ │ │ │ - layer.addFeatures(layerContext.features); │ │ │ │ │ - } else if (layerContext.categoryLayer !== true) { │ │ │ │ │ - layer = new OpenLayers.Layer.WMS( │ │ │ │ │ - layerContext.title || layerContext.name, │ │ │ │ │ - layerContext.url, │ │ │ │ │ - params, │ │ │ │ │ - options │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return layer; │ │ │ │ │ + resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleCreate, resp, options), │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(features) │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getLayersFromContext │ │ │ │ │ - * Create an array of layers from an array of layerContext objects. │ │ │ │ │ + * Method: handleCreate │ │ │ │ │ + * Called the the request issued by <create> is complete. May be overridden │ │ │ │ │ + * by subclasses. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layersContext - {Array(Object)} An array of objects representing layers. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Layer>)} An array of layers. │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the create call. │ │ │ │ │ */ │ │ │ │ │ - getLayersFromContext: function(layersContext) { │ │ │ │ │ - var layers = []; │ │ │ │ │ - for (var i = 0, len = layersContext.length; i < len; i++) { │ │ │ │ │ - var layer = this.getLayerFromContext(layersContext[i]); │ │ │ │ │ - if (layer !== null) { │ │ │ │ │ - layers.push(layer); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return layers; │ │ │ │ │ + handleCreate: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: contextToMap │ │ │ │ │ - * Create a map given a context object. │ │ │ │ │ + * APIMethod: update │ │ │ │ │ + * Construct a request updating modified feature. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * context - {Object} The context object. │ │ │ │ │ - * options - {Object} Default map options. │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Map>} A map based on the context object. │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, whose "priv" property references the HTTP request, this │ │ │ │ │ + * object is also passed to the callback function when the request │ │ │ │ │ + * completes, its "features" property is then populated with the │ │ │ │ │ + * the feature received from the server. │ │ │ │ │ */ │ │ │ │ │ - contextToMap: function(context, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - maxExtent: context.maxExtent, │ │ │ │ │ - projection: context.projection, │ │ │ │ │ - units: context.units │ │ │ │ │ - }, options); │ │ │ │ │ - │ │ │ │ │ - if (options.maxExtent) { │ │ │ │ │ - options.maxResolution = │ │ │ │ │ - options.maxExtent.getWidth() / OpenLayers.Map.TILE_WIDTH; │ │ │ │ │ - } │ │ │ │ │ + update: function(feature, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var url = options.url || │ │ │ │ │ + feature.url || │ │ │ │ │ + this.options.url + "/" + feature.fid; │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ │ │ │ │ │ - var metadata = { │ │ │ │ │ - contactInformation: context.contactInformation, │ │ │ │ │ - "abstract": context["abstract"], │ │ │ │ │ - keywords: context.keywords, │ │ │ │ │ - logo: context.logo, │ │ │ │ │ - descriptionURL: context.descriptionURL │ │ │ │ │ - }; │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: feature, │ │ │ │ │ + requestType: "update" │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - options.metadata = metadata; │ │ │ │ │ + var method = this.updateWithPOST ? "POST" : "PUT"; │ │ │ │ │ + resp.priv = OpenLayers.Request[method]({ │ │ │ │ │ + url: url, │ │ │ │ │ + callback: this.createCallback(this.handleUpdate, resp, options), │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(feature) │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - var map = new OpenLayers.Map(options); │ │ │ │ │ - map.addLayers(this.getLayersFromContext(context.layersContext)); │ │ │ │ │ - map.setCenter( │ │ │ │ │ - context.bounds.getCenterLonLat(), │ │ │ │ │ - map.getZoomForExtent(context.bounds, true) │ │ │ │ │ - ); │ │ │ │ │ - return map; │ │ │ │ │ + return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mergeContextToMap │ │ │ │ │ - * Add layers from a context object to a map. │ │ │ │ │ + * Method: handleUpdate │ │ │ │ │ + * Called the the request issued by <update> is complete. May be overridden │ │ │ │ │ + * by subclasses. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * context - {Object} The context object. │ │ │ │ │ - * map - {<OpenLayers.Map>} The map. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Map>} The same map with layers added. │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the update call. │ │ │ │ │ */ │ │ │ │ │ - mergeContextToMap: function(context, map) { │ │ │ │ │ - map.addLayers(this.getLayersFromContext(context.layersContext)); │ │ │ │ │ - return map; │ │ │ │ │ + handleUpdate: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Write a context document given a map. │ │ │ │ │ + * APIMethod: delete │ │ │ │ │ + * Construct a request deleting a removed feature. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {<OpenLayers.Map> | Object} A map or context object. │ │ │ │ │ - * options - {Object} Optional configuration object. │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A context document string. │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, whose "priv" property references the HTTP request, this │ │ │ │ │ + * object is also passed to the callback function when the request │ │ │ │ │ + * completes. │ │ │ │ │ */ │ │ │ │ │ - write: function(obj, options) { │ │ │ │ │ - obj = this.toContext(obj); │ │ │ │ │ - return OpenLayers.Format.XML.VersionedOGC.prototype.write.apply(this, │ │ │ │ │ - arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.Context" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Format.Context.serviceTypes │ │ │ │ │ - * Enumeration for service types │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.Context.serviceTypes = { │ │ │ │ │ - "WMS": "urn:ogc:serviceType:WMS", │ │ │ │ │ - "WFS": "urn:ogc:serviceType:WFS", │ │ │ │ │ - "WCS": "urn:ogc:serviceType:WCS", │ │ │ │ │ - "GML": "urn:ogc:serviceType:GML", │ │ │ │ │ - "SLD": "urn:ogc:serviceType:SLD", │ │ │ │ │ - "FES": "urn:ogc:serviceType:FES", │ │ │ │ │ - "KML": "urn:ogc:serviceType:KML" │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMC.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + "delete": function(feature, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var url = options.url || │ │ │ │ │ + feature.url || │ │ │ │ │ + this.options.url + "/" + feature.fid; │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: feature, │ │ │ │ │ + requestType: "delete" │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/Context.js │ │ │ │ │ - */ │ │ │ │ │ + var method = this.deleteWithPOST ? "POST" : "DELETE"; │ │ │ │ │ + var requestOptions = { │ │ │ │ │ + url: url, │ │ │ │ │ + callback: this.createCallback(this.handleDelete, resp, options), │ │ │ │ │ + headers: options.headers │ │ │ │ │ + }; │ │ │ │ │ + if (this.deleteWithPOST) { │ │ │ │ │ + requestOptions.data = this.format.write(feature); │ │ │ │ │ + } │ │ │ │ │ + resp.priv = OpenLayers.Request[method](requestOptions); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WMC │ │ │ │ │ - * Read and write Web Map Context documents. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.Context> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WMC = OpenLayers.Class(OpenLayers.Format.Context, { │ │ │ │ │ + return resp; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.1.0". │ │ │ │ │ + * Method: handleDelete │ │ │ │ │ + * Called the the request issued by <delete> is complete. May be overridden │ │ │ │ │ + * by subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the delete call. │ │ │ │ │ */ │ │ │ │ │ - defaultVersion: "1.1.0", │ │ │ │ │ + handleDelete: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMC │ │ │ │ │ - * Create a new parser for Web Map Context documents. │ │ │ │ │ + * Method: handleResponse │ │ │ │ │ + * Called by CRUD specific handlers. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the create, read, update, │ │ │ │ │ + * or delete call. │ │ │ │ │ */ │ │ │ │ │ + handleResponse: function(resp, options) { │ │ │ │ │ + var request = resp.priv; │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + // success │ │ │ │ │ + if (resp.requestType != "delete") { │ │ │ │ │ + resp.features = this.parseFeatures(request); │ │ │ │ │ + } │ │ │ │ │ + resp.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + } else { │ │ │ │ │ + // failure │ │ │ │ │ + resp.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + } │ │ │ │ │ + options.callback.call(options.scope, resp); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: layerToContext │ │ │ │ │ - * Create a layer context object given a wms layer object. │ │ │ │ │ + * Method: parseFeatures │ │ │ │ │ + * Read HTTP response body and return features. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.WMS>} The layer. │ │ │ │ │ + * request - {XMLHttpRequest} The request object │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} A layer context object. │ │ │ │ │ + * {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} Array of features or a single feature. │ │ │ │ │ */ │ │ │ │ │ - layerToContext: function(layer) { │ │ │ │ │ - var parser = this.getParser(); │ │ │ │ │ - var layerContext = { │ │ │ │ │ - queryable: layer.queryable, │ │ │ │ │ - visibility: layer.visibility, │ │ │ │ │ - name: layer.params["LAYERS"], │ │ │ │ │ - title: layer.name, │ │ │ │ │ - "abstract": layer.metadata["abstract"], │ │ │ │ │ - dataURL: layer.metadata.dataURL, │ │ │ │ │ - metadataURL: layer.metadataURL, │ │ │ │ │ - server: { │ │ │ │ │ - version: layer.params["VERSION"], │ │ │ │ │ - url: layer.url │ │ │ │ │ - }, │ │ │ │ │ - maxExtent: layer.maxExtent, │ │ │ │ │ - transparent: layer.params["TRANSPARENT"], │ │ │ │ │ - numZoomLevels: layer.numZoomLevels, │ │ │ │ │ - units: layer.units, │ │ │ │ │ - isBaseLayer: layer.isBaseLayer, │ │ │ │ │ - opacity: layer.opacity == 1 ? undefined : layer.opacity, │ │ │ │ │ - displayInLayerSwitcher: layer.displayInLayerSwitcher, │ │ │ │ │ - singleTile: layer.singleTile, │ │ │ │ │ - tileSize: (layer.singleTile || !layer.tileSize) ? │ │ │ │ │ - undefined : { │ │ │ │ │ - width: layer.tileSize.w, │ │ │ │ │ - height: layer.tileSize.h │ │ │ │ │ - }, │ │ │ │ │ - minScale: (layer.options.resolutions || │ │ │ │ │ - layer.options.scales || │ │ │ │ │ - layer.options.maxResolution || │ │ │ │ │ - layer.options.minScale) ? │ │ │ │ │ - layer.minScale : undefined, │ │ │ │ │ - maxScale: (layer.options.resolutions || │ │ │ │ │ - layer.options.scales || │ │ │ │ │ - layer.options.minResolution || │ │ │ │ │ - layer.options.maxScale) ? │ │ │ │ │ - layer.maxScale : undefined, │ │ │ │ │ - formats: [], │ │ │ │ │ - styles: [], │ │ │ │ │ - srs: layer.srs, │ │ │ │ │ - dimensions: layer.dimensions │ │ │ │ │ - }; │ │ │ │ │ + parseFeatures: function(request) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ + } │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + return this.format.read(doc); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: commit │ │ │ │ │ + * Iterate over each feature and take action based on the feature state. │ │ │ │ │ + * Possible actions are create, update and delete. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array({<OpenLayers.Feature.Vector>})} │ │ │ │ │ + * options - {Object} Optional object for setting up intermediate commit │ │ │ │ │ + * callbacks. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * create - {Object} Optional object to be passed to the <create> method. │ │ │ │ │ + * update - {Object} Optional object to be passed to the <update> method. │ │ │ │ │ + * delete - {Object} Optional object to be passed to the <delete> method. │ │ │ │ │ + * callback - {Function} Optional function to be called when the commit │ │ │ │ │ + * is complete. │ │ │ │ │ + * scope - {Object} Optional object to be set as the scope of the callback. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Protocol.Response>)} An array of response objects, │ │ │ │ │ + * one per request made to the server, each object's "priv" property │ │ │ │ │ + * references the corresponding HTTP request. │ │ │ │ │ + */ │ │ │ │ │ + commit: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = [], │ │ │ │ │ + nResponses = 0; │ │ │ │ │ │ │ │ │ │ - if (layer.metadata.servertitle) { │ │ │ │ │ - layerContext.server.title = layer.metadata.servertitle; │ │ │ │ │ + // Divide up features before issuing any requests. This properly │ │ │ │ │ + // counts requests in the event that any responses come in before │ │ │ │ │ + // all requests have been issued. │ │ │ │ │ + var types = {}; │ │ │ │ │ + types[OpenLayers.State.INSERT] = []; │ │ │ │ │ + types[OpenLayers.State.UPDATE] = []; │ │ │ │ │ + types[OpenLayers.State.DELETE] = []; │ │ │ │ │ + var feature, list, requestFeatures = []; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + list = types[feature.state]; │ │ │ │ │ + if (list) { │ │ │ │ │ + list.push(feature); │ │ │ │ │ + requestFeatures.push(feature); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + // tally up number of requests │ │ │ │ │ + var nRequests = (types[OpenLayers.State.INSERT].length > 0 ? 1 : 0) + │ │ │ │ │ + types[OpenLayers.State.UPDATE].length + │ │ │ │ │ + types[OpenLayers.State.DELETE].length; │ │ │ │ │ │ │ │ │ │ - if (layer.metadata.formats && layer.metadata.formats.length > 0) { │ │ │ │ │ - for (var i = 0, len = layer.metadata.formats.length; i < len; i++) { │ │ │ │ │ - var format = layer.metadata.formats[i]; │ │ │ │ │ - layerContext.formats.push({ │ │ │ │ │ - value: format.value, │ │ │ │ │ - current: (format.value == layer.params["FORMAT"]) │ │ │ │ │ - }); │ │ │ │ │ + // This response will be sent to the final callback after all the others │ │ │ │ │ + // have been fired. │ │ │ │ │ + var success = true; │ │ │ │ │ + var finalResponse = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: requestFeatures │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + function insertCallback(response) { │ │ │ │ │ + var len = response.features ? response.features.length : 0; │ │ │ │ │ + var fids = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + fids[i] = response.features[i].fid; │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - layerContext.formats.push({ │ │ │ │ │ - value: layer.params["FORMAT"], │ │ │ │ │ - current: true │ │ │ │ │ - }); │ │ │ │ │ + finalResponse.insertIds = fids; │ │ │ │ │ + callback.apply(this, [response]); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - if (layer.metadata.styles && layer.metadata.styles.length > 0) { │ │ │ │ │ - for (var i = 0, len = layer.metadata.styles.length; i < len; i++) { │ │ │ │ │ - var style = layer.metadata.styles[i]; │ │ │ │ │ - if ((style.href == layer.params["SLD"]) || │ │ │ │ │ - (style.body == layer.params["SLD_BODY"]) || │ │ │ │ │ - (style.name == layer.params["STYLES"])) { │ │ │ │ │ - style.current = true; │ │ │ │ │ - } else { │ │ │ │ │ - style.current = false; │ │ │ │ │ + function callback(response) { │ │ │ │ │ + this.callUserCallback(response, options); │ │ │ │ │ + success = success && response.success(); │ │ │ │ │ + nResponses++; │ │ │ │ │ + if (nResponses >= nRequests) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + finalResponse.code = success ? │ │ │ │ │ + OpenLayers.Protocol.Response.SUCCESS : │ │ │ │ │ + OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + options.callback.apply(options.scope, [finalResponse]); │ │ │ │ │ } │ │ │ │ │ - layerContext.styles.push(style); │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - layerContext.styles.push({ │ │ │ │ │ - href: layer.params["SLD"], │ │ │ │ │ - body: layer.params["SLD_BODY"], │ │ │ │ │ - name: layer.params["STYLES"] || parser.defaultStyleName, │ │ │ │ │ - title: parser.defaultStyleTitle, │ │ │ │ │ - current: true │ │ │ │ │ - }); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return layerContext; │ │ │ │ │ + // start issuing requests │ │ │ │ │ + var queue = types[OpenLayers.State.INSERT]; │ │ │ │ │ + if (queue.length > 0) { │ │ │ │ │ + resp.push(this.create( │ │ │ │ │ + queue, OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: insertCallback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options.create) │ │ │ │ │ + )); │ │ │ │ │ + } │ │ │ │ │ + queue = types[OpenLayers.State.UPDATE]; │ │ │ │ │ + for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ + resp.push(this.update( │ │ │ │ │ + queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: callback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options.update))); │ │ │ │ │ + } │ │ │ │ │ + queue = types[OpenLayers.State.DELETE]; │ │ │ │ │ + for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ + resp.push(this["delete"]( │ │ │ │ │ + queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: callback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options["delete"]))); │ │ │ │ │ + } │ │ │ │ │ + return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: toContext │ │ │ │ │ - * Create a context object free from layer given a map or a │ │ │ │ │ - * context object. │ │ │ │ │ + * APIMethod: abort │ │ │ │ │ + * Abort an ongoing request, the response object passed to │ │ │ │ │ + * this method must come from this HTTP protocol (as a result │ │ │ │ │ + * of a create, read, update, delete or commit operation). │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {<OpenLayers.Map> | Object} The map or context. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} A context object. │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ */ │ │ │ │ │ - toContext: function(obj) { │ │ │ │ │ - var context = {}; │ │ │ │ │ - var layers = obj.layers; │ │ │ │ │ - if (obj.CLASS_NAME == "OpenLayers.Map") { │ │ │ │ │ - var metadata = obj.metadata || {}; │ │ │ │ │ - context.size = obj.getSize(); │ │ │ │ │ - context.bounds = obj.getExtent(); │ │ │ │ │ - context.projection = obj.projection; │ │ │ │ │ - context.title = obj.title; │ │ │ │ │ - context.keywords = metadata.keywords; │ │ │ │ │ - context["abstract"] = metadata["abstract"]; │ │ │ │ │ - context.logo = metadata.logo; │ │ │ │ │ - context.descriptionURL = metadata.descriptionURL; │ │ │ │ │ - context.contactInformation = metadata.contactInformation; │ │ │ │ │ - context.maxExtent = obj.maxExtent; │ │ │ │ │ - } else { │ │ │ │ │ - // copy all obj properties except the "layers" property │ │ │ │ │ - OpenLayers.Util.applyDefaults(context, obj); │ │ │ │ │ - if (context.layers != undefined) { │ │ │ │ │ - delete(context.layers); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (context.layersContext == undefined) { │ │ │ │ │ - context.layersContext = []; │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + response.priv.abort(); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // let's convert layers into layersContext object (if any) │ │ │ │ │ - if (layers != undefined && OpenLayers.Util.isArray(layers)) { │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - var layer = layers[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.WMS) { │ │ │ │ │ - context.layersContext.push(this.layerToContext(layer)); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: callUserCallback │ │ │ │ │ + * This method is used from within the commit method each time an │ │ │ │ │ + * an HTTP response is received from the server, it is responsible │ │ │ │ │ + * for calling the user-supplied callbacks. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} │ │ │ │ │ + * options - {Object} The map of options passed to the commit call. │ │ │ │ │ + */ │ │ │ │ │ + callUserCallback: function(resp, options) { │ │ │ │ │ + var opt = options[resp.requestType]; │ │ │ │ │ + if (opt && opt.callback) { │ │ │ │ │ + opt.callback.call(opt.scope, resp); │ │ │ │ │ } │ │ │ │ │ - return context; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMC" │ │ │ │ │ - │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.HTTP" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/OSM.js │ │ │ │ │ + OpenLayers/Protocol/Script.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Protocol.js │ │ │ │ │ * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ - * @requires OpenLayers/Projection.js │ │ │ │ │ + * @requires OpenLayers/Format/GeoJSON.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.OSM │ │ │ │ │ - * OSM parser. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Format.OSM> constructor. │ │ │ │ │ +/** │ │ │ │ │ + * if application uses the query string, for example, for BBOX parameters, │ │ │ │ │ + * OpenLayers/Format/QueryStringFilter.js should be included in the build config file │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.Script │ │ │ │ │ + * A basic Script protocol for vector layers. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Protocol.Script> constructor. A script protocol is used to │ │ │ │ │ + * get around the same origin policy. It works with services that return │ │ │ │ │ + * JSONP - that is, JSON wrapped in a client-specified callback. The │ │ │ │ │ + * protocol handles fetching and parsing of feature data and sends parsed │ │ │ │ │ + * features to the <callback> configured with the protocol. The protocol │ │ │ │ │ + * expects features serialized as GeoJSON by default, but can be configured │ │ │ │ │ + * to work with other formats by setting the <format> property. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * - <OpenLayers.Protocol> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.OSM = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Protocol.Script = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: checkTags │ │ │ │ │ - * {Boolean} Should tags be checked to determine whether something │ │ │ │ │ - * should be treated as a seperate node. Will slow down parsing. │ │ │ │ │ - * Default is false. │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String} Service URL. The service is expected to return serialized │ │ │ │ │ + * features wrapped in a named callback (where the callback name is │ │ │ │ │ + * generated by this protocol). │ │ │ │ │ + * Read-only, set through the options passed to the constructor. │ │ │ │ │ */ │ │ │ │ │ - checkTags: false, │ │ │ │ │ + url: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: interestingTagsExclude │ │ │ │ │ - * {Array} List of tags to exclude from 'interesting' checks on nodes. │ │ │ │ │ - * Must be set when creating the format. Will only be used if checkTags │ │ │ │ │ - * is set. │ │ │ │ │ + * APIProperty: params │ │ │ │ │ + * {Object} Query string parameters to be appended to the URL. │ │ │ │ │ + * Read-only, set through the options passed to the constructor. │ │ │ │ │ + * Example: {maxFeatures: 50} │ │ │ │ │ */ │ │ │ │ │ - interestingTagsExclude: null, │ │ │ │ │ + params: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: areaTags │ │ │ │ │ - * {Array} List of tags indicating that something is an area. │ │ │ │ │ - * Must be set when creating the format. Will only be used if │ │ │ │ │ - * checkTags is true. │ │ │ │ │ + * APIProperty: callback │ │ │ │ │ + * {Object} Function to be called when the <read> operation completes. │ │ │ │ │ */ │ │ │ │ │ - areaTags: null, │ │ │ │ │ + callback: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.OSM │ │ │ │ │ - * Create a new parser for OSM. │ │ │ │ │ + * APIProperty: callbackTemplate │ │ │ │ │ + * {String} Template for creating a unique callback function name │ │ │ │ │ + * for the registry. Should include ${id}. The ${id} variable will be │ │ │ │ │ + * replaced with a string identifier prefixed with a "c" (e.g. c1, c2). │ │ │ │ │ + * Default is "OpenLayers.Protocol.Script.registry.${id}". │ │ │ │ │ + */ │ │ │ │ │ + callbackTemplate: "OpenLayers.Protocol.Script.registry.${id}", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: callbackKey │ │ │ │ │ + * {String} The name of the query string parameter that the service │ │ │ │ │ + * recognizes as the callback identifier. Default is "callback". │ │ │ │ │ + * This key is used to generate the URL for the script. For example │ │ │ │ │ + * setting <callbackKey> to "myCallback" would result in a URL like │ │ │ │ │ + * http://example.com/?myCallback=... │ │ │ │ │ + */ │ │ │ │ │ + callbackKey: "callback", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: callbackPrefix │ │ │ │ │ + * {String} Where a service requires that the callback query string │ │ │ │ │ + * parameter value is prefixed by some string, this value may be set. │ │ │ │ │ + * For example, setting <callbackPrefix> to "foo:" would result in a │ │ │ │ │ + * URL like http://example.com/?callback=foo:... Default is "". │ │ │ │ │ + */ │ │ │ │ │ + callbackPrefix: "", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: scope │ │ │ │ │ + * {Object} Optional ``this`` object for the callback. Read-only, set │ │ │ │ │ + * through the options passed to the constructor. │ │ │ │ │ + */ │ │ │ │ │ + scope: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: format │ │ │ │ │ + * {<OpenLayers.Format>} Format for parsing features. Default is an │ │ │ │ │ + * <OpenLayers.Format.GeoJSON> format. If an alternative is provided, │ │ │ │ │ + * the format's read method must take an object and return an array │ │ │ │ │ + * of features. │ │ │ │ │ + */ │ │ │ │ │ + format: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: pendingRequests │ │ │ │ │ + * {Object} References all pending requests. Property names are script │ │ │ │ │ + * identifiers and property values are script elements. │ │ │ │ │ + */ │ │ │ │ │ + pendingRequests: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: srsInBBOX │ │ │ │ │ + * {Boolean} Include the SRS identifier in BBOX query string parameter. │ │ │ │ │ + * Setting this property has no effect if a custom filterToParams method │ │ │ │ │ + * is provided. Default is false. If true and the layer has a │ │ │ │ │ + * projection object set, any BBOX filter will be serialized with a │ │ │ │ │ + * fifth item identifying the projection. │ │ │ │ │ + * E.g. bbox=-1000,-1000,1000,1000,EPSG:900913 │ │ │ │ │ + */ │ │ │ │ │ + srsInBBOX: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Protocol.Script │ │ │ │ │ + * A class for giving layers generic Script protocol. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + * │ │ │ │ │ + * Valid options include: │ │ │ │ │ + * url - {String} │ │ │ │ │ + * params - {Object} │ │ │ │ │ + * callback - {Function} │ │ │ │ │ + * scope - {Object} │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - var layer_defaults = { │ │ │ │ │ - 'interestingTagsExclude': ['source', 'source_ref', │ │ │ │ │ - 'source:ref', 'history', 'attribution', 'created_by' │ │ │ │ │ - ], │ │ │ │ │ - 'areaTags': ['area', 'building', 'leisure', 'tourism', 'ruins', │ │ │ │ │ - 'historic', 'landuse', 'military', 'natural', 'sport' │ │ │ │ │ - ] │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - layer_defaults = OpenLayers.Util.extend(layer_defaults, options); │ │ │ │ │ - │ │ │ │ │ - var interesting = {}; │ │ │ │ │ - for (var i = 0; i < layer_defaults.interestingTagsExclude.length; i++) { │ │ │ │ │ - interesting[layer_defaults.interestingTagsExclude[i]] = true; │ │ │ │ │ + options = options || {}; │ │ │ │ │ + this.params = {}; │ │ │ │ │ + this.pendingRequests = {}; │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (!this.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.GeoJSON(); │ │ │ │ │ } │ │ │ │ │ - layer_defaults.interestingTagsExclude = interesting; │ │ │ │ │ │ │ │ │ │ - var area = {}; │ │ │ │ │ - for (var i = 0; i < layer_defaults.areaTags.length; i++) { │ │ │ │ │ - area[layer_defaults.areaTags[i]] = true; │ │ │ │ │ + if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ + var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ + srsInBBOX: this.srsInBBOX │ │ │ │ │ + }); │ │ │ │ │ + this.filterToParams = function(filter, params) { │ │ │ │ │ + return format.write(filter, params); │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ - layer_defaults.areaTags = area; │ │ │ │ │ - │ │ │ │ │ - // OSM coordinates are always in longlat WGS84 │ │ │ │ │ - this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [layer_defaults]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIMethod: read │ │ │ │ │ - * Return a list of features from a OSM doc │ │ │ │ │ - │ │ │ │ │ + * Construct a request for reading new features. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * doc - {Element} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * url - {String} Url for the request. │ │ │ │ │ + * params - {Object} Parameters to get serialized as a query string. │ │ │ │ │ + * filter - {<OpenLayers.Filter>} Filter to get serialized as a │ │ │ │ │ + * query string. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} A response object, whose "priv" property │ │ │ │ │ + * references the injected script. This object is also passed to the │ │ │ │ │ + * callback function when the request completes, its "features" property │ │ │ │ │ + * is then populated with the features received from the server. │ │ │ │ │ */ │ │ │ │ │ - read: function(doc) { │ │ │ │ │ - if (typeof doc == "string") { │ │ │ │ │ - doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]); │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + options.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options.params, this.options.params │ │ │ │ │ + ); │ │ │ │ │ + if (options.filter && this.filterToParams) { │ │ │ │ │ + options.params = this.filterToParams( │ │ │ │ │ + options.filter, options.params │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + var request = this.createRequest( │ │ │ │ │ + options.url, │ │ │ │ │ + options.params, │ │ │ │ │ + OpenLayers.Function.bind(function(data) { │ │ │ │ │ + response.data = data; │ │ │ │ │ + this.handleRead(response, options); │ │ │ │ │ + }, this) │ │ │ │ │ + ); │ │ │ │ │ + response.priv = request; │ │ │ │ │ + return response; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var nodes = this.getNodes(doc); │ │ │ │ │ - var ways = this.getWays(doc); │ │ │ │ │ - │ │ │ │ │ - // Geoms will contain at least ways.length entries. │ │ │ │ │ - var feat_list = new Array(ways.length); │ │ │ │ │ - │ │ │ │ │ - for (var i = 0; i < ways.length; i++) { │ │ │ │ │ - // We know the minimal of this one ahead of time. (Could be -1 │ │ │ │ │ - // due to areas/polygons) │ │ │ │ │ - var point_list = new Array(ways[i].nodes.length); │ │ │ │ │ - │ │ │ │ │ - var poly = this.isWayArea(ways[i]) ? 1 : 0; │ │ │ │ │ - for (var j = 0; j < ways[i].nodes.length; j++) { │ │ │ │ │ - var node = nodes[ways[i].nodes[j]]; │ │ │ │ │ - │ │ │ │ │ - var point = new OpenLayers.Geometry.Point(node.lon, node.lat); │ │ │ │ │ - │ │ │ │ │ - // Since OSM is topological, we stash the node ID internally. │ │ │ │ │ - point.osm_id = parseInt(ways[i].nodes[j]); │ │ │ │ │ - point_list[j] = point; │ │ │ │ │ - │ │ │ │ │ - // We don't display nodes if they're used inside other │ │ │ │ │ - // elements. │ │ │ │ │ - node.used = true; │ │ │ │ │ - } │ │ │ │ │ - var geometry = null; │ │ │ │ │ - if (poly) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.Polygon( │ │ │ │ │ - new OpenLayers.Geometry.LinearRing(point_list)); │ │ │ │ │ - } else { │ │ │ │ │ - geometry = new OpenLayers.Geometry.LineString(point_list); │ │ │ │ │ - } │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry.transform(this.externalProjection, │ │ │ │ │ - this.internalProjection); │ │ │ │ │ - } │ │ │ │ │ - var feat = new OpenLayers.Feature.Vector(geometry, │ │ │ │ │ - ways[i].tags); │ │ │ │ │ - feat.osm_id = parseInt(ways[i].id); │ │ │ │ │ - feat.fid = "way." + feat.osm_id; │ │ │ │ │ - feat_list[i] = feat; │ │ │ │ │ - } │ │ │ │ │ - for (var node_id in nodes) { │ │ │ │ │ - var node = nodes[node_id]; │ │ │ │ │ - if (!node.used || this.checkTags) { │ │ │ │ │ - var tags = null; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: filterToParams │ │ │ │ │ + * Optional method to translate an <OpenLayers.Filter> object into an object │ │ │ │ │ + * that can be serialized as request query string provided. If a custom │ │ │ │ │ + * method is not provided, any filter will not be serialized. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {<OpenLayers.Filter>} filter to convert. │ │ │ │ │ + * params - {Object} The parameters object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The resulting parameters object. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (this.checkTags) { │ │ │ │ │ - var result = this.getTags(node.node, true); │ │ │ │ │ - if (node.used && !result[1]) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - tags = result[0]; │ │ │ │ │ - } else { │ │ │ │ │ - tags = this.getTags(node.node); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: createRequest │ │ │ │ │ + * Issues a request for features by creating injecting a script in the │ │ │ │ │ + * document head. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * url - {String} Service URL. │ │ │ │ │ + * params - {Object} Query string parameters. │ │ │ │ │ + * callback - {Function} Callback to be called with resulting data. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {HTMLScriptElement} The script pending execution. │ │ │ │ │ + */ │ │ │ │ │ + createRequest: function(url, params, callback) { │ │ │ │ │ + var id = OpenLayers.Protocol.Script.register(callback); │ │ │ │ │ + var name = OpenLayers.String.format(this.callbackTemplate, { │ │ │ │ │ + id: id │ │ │ │ │ + }); │ │ │ │ │ + params = OpenLayers.Util.extend({}, params); │ │ │ │ │ + params[this.callbackKey] = this.callbackPrefix + name; │ │ │ │ │ + url = OpenLayers.Util.urlAppend( │ │ │ │ │ + url, OpenLayers.Util.getParameterString(params) │ │ │ │ │ + ); │ │ │ │ │ + var script = document.createElement("script"); │ │ │ │ │ + script.type = "text/javascript"; │ │ │ │ │ + script.src = url; │ │ │ │ │ + script.id = "OpenLayers_Protocol_Script_" + id; │ │ │ │ │ + this.pendingRequests[script.id] = script; │ │ │ │ │ + var head = document.getElementsByTagName("head")[0]; │ │ │ │ │ + head.appendChild(script); │ │ │ │ │ + return script; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var feat = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.Point(node['lon'], node['lat']), │ │ │ │ │ - tags); │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - feat.geometry.transform(this.externalProjection, │ │ │ │ │ - this.internalProjection); │ │ │ │ │ - } │ │ │ │ │ - feat.osm_id = parseInt(node_id); │ │ │ │ │ - feat.fid = "node." + feat.osm_id; │ │ │ │ │ - feat_list.push(feat); │ │ │ │ │ - } │ │ │ │ │ - // Memory cleanup │ │ │ │ │ - node.node = null; │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroyRequest │ │ │ │ │ + * Remove a script node associated with a response from the document. Also │ │ │ │ │ + * unregisters the callback and removes the script from the │ │ │ │ │ + * <pendingRequests> object. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * script - {HTMLScriptElement} │ │ │ │ │ + */ │ │ │ │ │ + destroyRequest: function(script) { │ │ │ │ │ + OpenLayers.Protocol.Script.unregister(script.id.split("_").pop()); │ │ │ │ │ + delete this.pendingRequests[script.id]; │ │ │ │ │ + if (script.parentNode) { │ │ │ │ │ + script.parentNode.removeChild(script); │ │ │ │ │ } │ │ │ │ │ - return feat_list; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getNodes │ │ │ │ │ - * Return the node items from a doc. │ │ │ │ │ + * Method: handleRead │ │ │ │ │ + * Individual callbacks are created for read, create and update, should │ │ │ │ │ + * a subclass need to override each one separately. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * doc - {DOMElement} node to parse tags from │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * the user callback. │ │ │ │ │ + * options - {Object} The user options passed to the read call. │ │ │ │ │ */ │ │ │ │ │ - getNodes: function(doc) { │ │ │ │ │ - var node_list = doc.getElementsByTagName("node"); │ │ │ │ │ - var nodes = {}; │ │ │ │ │ - for (var i = 0; i < node_list.length; i++) { │ │ │ │ │ - var node = node_list[i]; │ │ │ │ │ - var id = node.getAttribute("id"); │ │ │ │ │ - nodes[id] = { │ │ │ │ │ - 'lat': node.getAttribute("lat"), │ │ │ │ │ - 'lon': node.getAttribute("lon"), │ │ │ │ │ - 'node': node │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - return nodes; │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + this.handleResponse(response, options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getWays │ │ │ │ │ - * Return the way items from a doc. │ │ │ │ │ + * Method: handleResponse │ │ │ │ │ + * Called by CRUD specific handlers. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * doc - {DOMElement} node to parse tags from │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the create, read, update, │ │ │ │ │ + * or delete call. │ │ │ │ │ */ │ │ │ │ │ - getWays: function(doc) { │ │ │ │ │ - var way_list = doc.getElementsByTagName("way"); │ │ │ │ │ - var return_ways = []; │ │ │ │ │ - for (var i = 0; i < way_list.length; i++) { │ │ │ │ │ - var way = way_list[i]; │ │ │ │ │ - var way_object = { │ │ │ │ │ - id: way.getAttribute("id") │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - way_object.tags = this.getTags(way); │ │ │ │ │ - │ │ │ │ │ - var node_list = way.getElementsByTagName("nd"); │ │ │ │ │ - │ │ │ │ │ - way_object.nodes = new Array(node_list.length); │ │ │ │ │ - │ │ │ │ │ - for (var j = 0; j < node_list.length; j++) { │ │ │ │ │ - way_object.nodes[j] = node_list[j].getAttribute("ref"); │ │ │ │ │ + handleResponse: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + if (response.data) { │ │ │ │ │ + response.features = this.parseFeatures(response.data); │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ } │ │ │ │ │ - return_ways.push(way_object); │ │ │ │ │ + this.destroyRequest(response.priv); │ │ │ │ │ + options.callback.call(options.scope, response); │ │ │ │ │ } │ │ │ │ │ - return return_ways; │ │ │ │ │ - │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getTags │ │ │ │ │ - * Return the tags list attached to a specific DOM element. │ │ │ │ │ + * Method: parseFeatures │ │ │ │ │ + * Read Script response body and return features. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * dom_node - {DOMElement} node to parse tags from │ │ │ │ │ - * interesting_tags - {Boolean} whether the return from this function should │ │ │ │ │ - * return a boolean indicating that it has 'interesting tags' -- │ │ │ │ │ - * tags like attribution and source are ignored. (To change the list │ │ │ │ │ - * of tags, see interestingTagsExclude) │ │ │ │ │ - * │ │ │ │ │ + * data - {Object} The data sent to the callback function by the server. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * tags - {Object} hash of tags │ │ │ │ │ - * interesting - {Boolean} if interesting_tags is passed, returns │ │ │ │ │ - * whether there are any interesting tags on this element. │ │ │ │ │ - */ │ │ │ │ │ - getTags: function(dom_node, interesting_tags) { │ │ │ │ │ - var tag_list = dom_node.getElementsByTagName("tag"); │ │ │ │ │ - var tags = {}; │ │ │ │ │ - var interesting = false; │ │ │ │ │ - for (var j = 0; j < tag_list.length; j++) { │ │ │ │ │ - var key = tag_list[j].getAttribute("k"); │ │ │ │ │ - tags[key] = tag_list[j].getAttribute("v"); │ │ │ │ │ - if (interesting_tags) { │ │ │ │ │ - if (!this.interestingTagsExclude[key]) { │ │ │ │ │ - interesting = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return interesting_tags ? [tags, interesting] : tags; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: isWayArea │ │ │ │ │ - * Given a way object from getWays, check whether the tags and geometry │ │ │ │ │ - * indicate something is an area. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} Array of features or a single feature. │ │ │ │ │ */ │ │ │ │ │ - isWayArea: function(way) { │ │ │ │ │ - var poly_shaped = false; │ │ │ │ │ - var poly_tags = false; │ │ │ │ │ - │ │ │ │ │ - if (way.nodes[0] == way.nodes[way.nodes.length - 1]) { │ │ │ │ │ - poly_shaped = true; │ │ │ │ │ - } │ │ │ │ │ - if (this.checkTags) { │ │ │ │ │ - for (var key in way.tags) { │ │ │ │ │ - if (this.areaTags[key]) { │ │ │ │ │ - poly_tags = true; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return poly_shaped && (this.checkTags ? poly_tags : true); │ │ │ │ │ + parseFeatures: function(data) { │ │ │ │ │ + return this.format.read(data); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Takes a list of features, returns a serialized OSM format file for use │ │ │ │ │ - * in tools like JOSM. │ │ │ │ │ + * APIMethod: abort │ │ │ │ │ + * Abort an ongoing request. If no response is provided, all pending │ │ │ │ │ + * requests will be aborted. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The response object returned │ │ │ │ │ + * from a <read> request. │ │ │ │ │ */ │ │ │ │ │ - write: function(features) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.osm_id = 1; │ │ │ │ │ - this.created_nodes = {}; │ │ │ │ │ - var root_node = this.createElementNS(null, "osm"); │ │ │ │ │ - root_node.setAttribute("version", "0.5"); │ │ │ │ │ - root_node.setAttribute("generator", "OpenLayers " + OpenLayers.VERSION_NUMBER); │ │ │ │ │ - │ │ │ │ │ - // Loop backwards, because the deserializer puts nodes last, and │ │ │ │ │ - // we want them first if possible │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - var nodes = this.createFeatureNodes(features[i]); │ │ │ │ │ - for (var j = 0; j < nodes.length; j++) { │ │ │ │ │ - root_node.appendChild(nodes[j]); │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + this.destroyRequest(response.priv); │ │ │ │ │ + } else { │ │ │ │ │ + for (var key in this.pendingRequests) { │ │ │ │ │ + this.destroyRequest(this.pendingRequests[key]); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [root_node]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createFeatureNodes │ │ │ │ │ - * Takes a feature, returns a list of nodes from size 0->n. │ │ │ │ │ - * Will include all pieces of the serialization that are required which │ │ │ │ │ - * have not already been created. Calls out to createXML based on geometry │ │ │ │ │ - * type. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up the protocol. │ │ │ │ │ */ │ │ │ │ │ - createFeatureNodes: function(feature) { │ │ │ │ │ - var nodes = []; │ │ │ │ │ - var className = feature.geometry.CLASS_NAME; │ │ │ │ │ - var type = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ - type = type.toLowerCase(); │ │ │ │ │ - var builder = this.createXML[type]; │ │ │ │ │ - if (builder) { │ │ │ │ │ - nodes = builder.apply(this, [feature]); │ │ │ │ │ - } │ │ │ │ │ - return nodes; │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.abort(); │ │ │ │ │ + delete this.params; │ │ │ │ │ + delete this.format; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createXML │ │ │ │ │ - * Takes a feature, returns a list of nodes from size 0->n. │ │ │ │ │ - * Will include all pieces of the serialization that are required which │ │ │ │ │ - * have not already been created. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - createXML: { │ │ │ │ │ - 'point': function(point) { │ │ │ │ │ - var id = null; │ │ │ │ │ - var geometry = point.geometry ? point.geometry : point; │ │ │ │ │ - │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry = geometry.clone(); │ │ │ │ │ - geometry.transform(this.internalProjection, │ │ │ │ │ - this.externalProjection); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var already_exists = false; // We don't return anything if the node │ │ │ │ │ - // has already been created │ │ │ │ │ - if (point.osm_id) { │ │ │ │ │ - id = point.osm_id; │ │ │ │ │ - if (this.created_nodes[id]) { │ │ │ │ │ - already_exists = true; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - id = -this.osm_id; │ │ │ │ │ - this.osm_id++; │ │ │ │ │ - } │ │ │ │ │ - if (already_exists) { │ │ │ │ │ - node = this.created_nodes[id]; │ │ │ │ │ - } else { │ │ │ │ │ - var node = this.createElementNS(null, "node"); │ │ │ │ │ - } │ │ │ │ │ - this.created_nodes[id] = node; │ │ │ │ │ - node.setAttribute("id", id); │ │ │ │ │ - node.setAttribute("lon", geometry.x); │ │ │ │ │ - node.setAttribute("lat", geometry.y); │ │ │ │ │ - if (point.attributes) { │ │ │ │ │ - this.serializeTags(point, node); │ │ │ │ │ - } │ │ │ │ │ - this.setState(point, node); │ │ │ │ │ - return already_exists ? [] : [node]; │ │ │ │ │ - }, │ │ │ │ │ - linestring: function(feature) { │ │ │ │ │ - var id; │ │ │ │ │ - var nodes = []; │ │ │ │ │ - var geometry = feature.geometry; │ │ │ │ │ - if (feature.osm_id) { │ │ │ │ │ - id = feature.osm_id; │ │ │ │ │ - } else { │ │ │ │ │ - id = -this.osm_id; │ │ │ │ │ - this.osm_id++; │ │ │ │ │ - } │ │ │ │ │ - var way = this.createElementNS(null, "way"); │ │ │ │ │ - way.setAttribute("id", id); │ │ │ │ │ - for (var i = 0; i < geometry.components.length; i++) { │ │ │ │ │ - var node = this.createXML['point'].apply(this, [geometry.components[i]]); │ │ │ │ │ - if (node.length) { │ │ │ │ │ - node = node[0]; │ │ │ │ │ - var node_ref = node.getAttribute("id"); │ │ │ │ │ - nodes.push(node); │ │ │ │ │ - } else { │ │ │ │ │ - node_ref = geometry.components[i].osm_id; │ │ │ │ │ - node = this.created_nodes[node_ref]; │ │ │ │ │ - } │ │ │ │ │ - this.setState(feature, node); │ │ │ │ │ - var nd_dom = this.createElementNS(null, "nd"); │ │ │ │ │ - nd_dom.setAttribute("ref", node_ref); │ │ │ │ │ - way.appendChild(nd_dom); │ │ │ │ │ - } │ │ │ │ │ - this.serializeTags(feature, way); │ │ │ │ │ - nodes.push(way); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.Script" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ - return nodes; │ │ │ │ │ - }, │ │ │ │ │ - polygon: function(feature) { │ │ │ │ │ - var attrs = OpenLayers.Util.extend({ │ │ │ │ │ - 'area': 'yes' │ │ │ │ │ - }, feature.attributes); │ │ │ │ │ - var feat = new OpenLayers.Feature.Vector(feature.geometry.components[0], attrs); │ │ │ │ │ - feat.osm_id = feature.osm_id; │ │ │ │ │ - return this.createXML['linestring'].apply(this, [feat]); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +(function() { │ │ │ │ │ + var o = OpenLayers.Protocol.Script; │ │ │ │ │ + var counter = 0; │ │ │ │ │ + o.registry = {}; │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: serializeTags │ │ │ │ │ - * Given a feature, serialize the attributes onto the given node. │ │ │ │ │ + * Function: OpenLayers.Protocol.Script.register │ │ │ │ │ + * Register a callback for a newly created script. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * node - {DOMNode} │ │ │ │ │ + * callback - {Function} The callback to be executed when the newly added │ │ │ │ │ + * script loads. This callback will be called with a single argument │ │ │ │ │ + * that is the JSON returned by the service. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number} An identifier for retrieving the registered callback. │ │ │ │ │ */ │ │ │ │ │ - serializeTags: function(feature, node) { │ │ │ │ │ - for (var key in feature.attributes) { │ │ │ │ │ - var tag = this.createElementNS(null, "tag"); │ │ │ │ │ - tag.setAttribute("k", key); │ │ │ │ │ - tag.setAttribute("v", feature.attributes[key]); │ │ │ │ │ - node.appendChild(tag); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + o.register = function(callback) { │ │ │ │ │ + var id = "c" + (++counter); │ │ │ │ │ + o.registry[id] = function() { │ │ │ │ │ + callback.apply(this, arguments); │ │ │ │ │ + }; │ │ │ │ │ + return id; │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setState │ │ │ │ │ - * OpenStreetMap has a convention that 'state' is stored for modification or deletion. │ │ │ │ │ - * This allows the file to be uploaded via JOSM or the bulk uploader tool. │ │ │ │ │ + * Function: OpenLayers.Protocol.Script.unregister │ │ │ │ │ + * Unregister a callback previously registered with the register function. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * node - {DOMNode} │ │ │ │ │ + * id - {Number} The identifer returned by the register function. │ │ │ │ │ */ │ │ │ │ │ - setState: function(feature, node) { │ │ │ │ │ - if (feature.state) { │ │ │ │ │ - var state = null; │ │ │ │ │ - switch (feature.state) { │ │ │ │ │ - case OpenLayers.State.UPDATE: │ │ │ │ │ - state = "modify"; │ │ │ │ │ - case OpenLayers.State.DELETE: │ │ │ │ │ - state = "delete"; │ │ │ │ │ - } │ │ │ │ │ - if (state) { │ │ │ │ │ - node.setAttribute("action", state); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.OSM" │ │ │ │ │ -}); │ │ │ │ │ + o.unregister = function(id) { │ │ │ │ │ + delete o.registry[id]; │ │ │ │ │ + }; │ │ │ │ │ +})(); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/OWSContext.js │ │ │ │ │ + OpenLayers/Protocol/WFS/v1.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/Context.js │ │ │ │ │ + * @requires OpenLayers/Protocol/WFS.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.OWSContext │ │ │ │ │ - * Read and write OWS Context documents. OWS Context documents are a │ │ │ │ │ - * preliminary OGC (Open Geospatial Consortium) standard for storing the │ │ │ │ │ - * state of a web mapping application. In a way it is the successor to │ │ │ │ │ - * Web Map Context (WMC), since it is more generic and more types of layers │ │ │ │ │ - * can be stored. Also, nesting of layers is supported since version 0.3.1. │ │ │ │ │ - * For more information see: http://www.ogcnetwork.net/context │ │ │ │ │ + * Class: OpenLayers.Protocol.WFS.v1 │ │ │ │ │ + * Abstract class for for v1.0.0 and v1.1.0 protocol. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.Context> │ │ │ │ │ + * - <OpenLayers.Protocol> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.OWSContext = OpenLayers.Class(OpenLayers.Format.Context, { │ │ │ │ │ +OpenLayers.Protocol.WFS.v1 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "0.3.1". │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} WFS version number. │ │ │ │ │ */ │ │ │ │ │ - defaultVersion: "0.3.1", │ │ │ │ │ + version: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.OWSContext │ │ │ │ │ - * Create a new parser for OWS Context documents. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * Property: srsName │ │ │ │ │ + * {String} Name of spatial reference system. Default is "EPSG:4326". │ │ │ │ │ */ │ │ │ │ │ + srsName: "EPSG:4326", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getVersion │ │ │ │ │ - * Returns the version to use. Subclasses can override this function │ │ │ │ │ - * if a different version detection is needed. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * root - {DOMElement} │ │ │ │ │ - * options - {Object} Optional configuration object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The version to use. │ │ │ │ │ + * Property: featureType │ │ │ │ │ + * {String} Local feature typeName. │ │ │ │ │ */ │ │ │ │ │ - getVersion: function(root, options) { │ │ │ │ │ - var version = OpenLayers.Format.XML.VersionedOGC.prototype.getVersion.apply( │ │ │ │ │ - this, arguments); │ │ │ │ │ - // 0.3.1 is backwards compatible with 0.3.0 │ │ │ │ │ - if (version === "0.3.0") { │ │ │ │ │ - version = this.defaultVersion; │ │ │ │ │ - } │ │ │ │ │ - return version; │ │ │ │ │ - }, │ │ │ │ │ + featureType: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: toContext │ │ │ │ │ - * Create a context object free from layer given a map or a │ │ │ │ │ - * context object. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {<OpenLayers.Map> | Object} The map or context. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} A context object. │ │ │ │ │ + * Property: featureNS │ │ │ │ │ + * {String} Feature namespace. │ │ │ │ │ */ │ │ │ │ │ - toContext: function(obj) { │ │ │ │ │ - var context = {}; │ │ │ │ │ - if (obj.CLASS_NAME == "OpenLayers.Map") { │ │ │ │ │ - context.bounds = obj.getExtent(); │ │ │ │ │ - context.maxExtent = obj.maxExtent; │ │ │ │ │ - context.projection = obj.projection; │ │ │ │ │ - context.size = obj.getSize(); │ │ │ │ │ - context.layers = obj.layers; │ │ │ │ │ - } │ │ │ │ │ - return context; │ │ │ │ │ - }, │ │ │ │ │ + featureNS: null, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.OWSContext" │ │ │ │ │ + /** │ │ │ │ │ + * Property: geometryName │ │ │ │ │ + * {String} Name of the geometry attribute for features. Default is │ │ │ │ │ + * "the_geom" for WFS <version> 1.0, and null for higher versions. │ │ │ │ │ + */ │ │ │ │ │ + geometryName: "the_geom", │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/EncodedPolyline.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: maxFeatures │ │ │ │ │ + * {Integer} Optional maximum number of features to retrieve. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: schema │ │ │ │ │ + * {String} Optional schema location that will be included in the │ │ │ │ │ + * schemaLocation attribute value. Note that the feature type schema │ │ │ │ │ + * is required for a strict XML validator (on transactions with an │ │ │ │ │ + * insert for example), but is *not* required by the WFS specification │ │ │ │ │ + * (since the server is supposed to know about feature type schemas). │ │ │ │ │ + */ │ │ │ │ │ + schema: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: featurePrefix │ │ │ │ │ + * {String} Namespace alias for feature type. Default is "feature". │ │ │ │ │ + */ │ │ │ │ │ + featurePrefix: "feature", │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.EncodedPolyline │ │ │ │ │ - * Class for reading and writing encoded polylines. Create a new instance │ │ │ │ │ - * with the <OpenLayers.Format.EncodedPolyline> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.EncodedPolyline = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ + /** │ │ │ │ │ + * Property: formatOptions │ │ │ │ │ + * {Object} Optional options for the format. If a format is not provided, │ │ │ │ │ + * this property can be used to extend the default format options. │ │ │ │ │ + */ │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readFormat │ │ │ │ │ + * {<OpenLayers.Format>} For WFS requests it is possible to get a │ │ │ │ │ + * different output format than GML. In that case, we cannot parse │ │ │ │ │ + * the response with the default format (WFST) and we need a different │ │ │ │ │ + * format for reading. │ │ │ │ │ + */ │ │ │ │ │ + readFormat: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: geometryType │ │ │ │ │ - * {String} Geometry type to output. One of: linestring (default), │ │ │ │ │ - * linearring, point, multipoint or polygon. If the geometryType is │ │ │ │ │ - * point, only the first point of the string is returned. │ │ │ │ │ + * Property: readOptions │ │ │ │ │ + * {Object} Optional object to pass to format's read. │ │ │ │ │ */ │ │ │ │ │ - geometryType: "linestring", │ │ │ │ │ + readOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.EncodedPolyline │ │ │ │ │ - * Create a new parser for encoded polylines │ │ │ │ │ + * Constructor: OpenLayers.Protocol.WFS │ │ │ │ │ + * A class for giving layers WFS protocol. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Format.EncodedPolyline>} A new encoded polylines parser. │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * url - {String} URL to send requests to (required). │ │ │ │ │ + * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ + * featureNS - {String} Feature namespace (required, but can be autodetected │ │ │ │ │ + * during the first query if GML is used as readFormat and │ │ │ │ │ + * featurePrefix is provided and matches the prefix used by the server │ │ │ │ │ + * for this featureType). │ │ │ │ │ + * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ + * for writing if featureNS is provided). Default is 'feature'. │ │ │ │ │ + * geometryName - {String} Name of geometry attribute. The default is │ │ │ │ │ + * 'the_geom' for WFS <version> 1.0, and null for higher versions. If │ │ │ │ │ + * null, it will be set to the name of the first geometry found in the │ │ │ │ │ + * first read operation. │ │ │ │ │ + * multi - {Boolean} If set to true, geometries will be casted to Multi │ │ │ │ │ + * geometries before they are written in a transaction. No casting will │ │ │ │ │ + * be done when reading features. │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.prototype.initialize.apply(this, [options]); │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!options.format) { │ │ │ │ │ + this.format = OpenLayers.Format.WFST(OpenLayers.Util.extend({ │ │ │ │ │ + version: this.version, │ │ │ │ │ + featureType: this.featureType, │ │ │ │ │ + featureNS: this.featureNS, │ │ │ │ │ + featurePrefix: this.featurePrefix, │ │ │ │ │ + geometryName: this.geometryName, │ │ │ │ │ + srsName: this.srsName, │ │ │ │ │ + schema: this.schema │ │ │ │ │ + }, this.formatOptions)); │ │ │ │ │ + } │ │ │ │ │ + if (!options.geometryName && parseFloat(this.format.version) > 1.0) { │ │ │ │ │ + this.setGeometryName(null); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up the protocol. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.options && !this.options.format) { │ │ │ │ │ + this.format.destroy(); │ │ │ │ │ + } │ │ │ │ │ + this.format = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIMethod: read │ │ │ │ │ - * Deserialize an encoded polyline string and return a vector feature. │ │ │ │ │ + * Construct a request for reading new features. Since WFS splits the │ │ │ │ │ + * basic CRUD operations into GetFeature requests (for read) and │ │ │ │ │ + * Transactions (for all others), this method does not make use of the │ │ │ │ │ + * format's read method (that is only about reading transaction │ │ │ │ │ + * responses). │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * encoded - {String} An encoded polyline string │ │ │ │ │ + * options - {Object} Options for the read operation, in addition to the │ │ │ │ │ + * options set on the instance (options set here will take precedence). │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A vector feature with a linestring. │ │ │ │ │ + * To use a configured protocol to get e.g. a WFS hit count, applications │ │ │ │ │ + * could do the following: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * protocol.read({ │ │ │ │ │ + * readOptions: {output: "object"}, │ │ │ │ │ + * resultType: "hits", │ │ │ │ │ + * maxFeatures: null, │ │ │ │ │ + * callback: function(resp) { │ │ │ │ │ + * // process resp.numberOfFeatures here │ │ │ │ │ + * } │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * To use a configured protocol to use WFS paging (if supported by the │ │ │ │ │ + * server), applications could do the following: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * protocol.read({ │ │ │ │ │ + * startIndex: 0, │ │ │ │ │ + * count: 50 │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * To limit the attributes returned by the GetFeature request, applications │ │ │ │ │ + * can use the propertyNames option to specify the properties to include in │ │ │ │ │ + * the response: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * protocol.read({ │ │ │ │ │ + * propertyNames: ["DURATION", "INTENSITY"] │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ */ │ │ │ │ │ - read: function(encoded) { │ │ │ │ │ - var geomType; │ │ │ │ │ - if (this.geometryType == "linestring") │ │ │ │ │ - geomType = OpenLayers.Geometry.LineString; │ │ │ │ │ - else if (this.geometryType == "linearring") │ │ │ │ │ - geomType = OpenLayers.Geometry.LinearRing; │ │ │ │ │ - else if (this.geometryType == "multipoint") │ │ │ │ │ - geomType = OpenLayers.Geometry.MultiPoint; │ │ │ │ │ - else if (this.geometryType != "point" && this.geometryType != "polygon") │ │ │ │ │ - return null; │ │ │ │ │ - │ │ │ │ │ - var flatPoints = this.decodeDeltas(encoded, 2); │ │ │ │ │ - var flatPointsLength = flatPoints.length; │ │ │ │ │ - │ │ │ │ │ - var pointGeometries = []; │ │ │ │ │ - for (var i = 0; i + 1 < flatPointsLength;) { │ │ │ │ │ - var y = flatPoints[i++], │ │ │ │ │ - x = flatPoints[i++]; │ │ │ │ │ - pointGeometries.push(new OpenLayers.Geometry.Point(x, y)); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - if (this.geometryType == "point") │ │ │ │ │ - return new OpenLayers.Feature.Vector( │ │ │ │ │ - pointGeometries[0] │ │ │ │ │ - ); │ │ │ │ │ + var data = OpenLayers.Format.XML.prototype.write.apply( │ │ │ │ │ + this.format, [this.format.writeNode("wfs:GetFeature", options)] │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - if (this.geometryType == "polygon") │ │ │ │ │ - return new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.Polygon([ │ │ │ │ │ - new OpenLayers.Geometry.LinearRing(pointGeometries) │ │ │ │ │ - ]) │ │ │ │ │ - ); │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: data │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - return new OpenLayers.Feature.Vector( │ │ │ │ │ - new geomType(pointGeometries) │ │ │ │ │ - ); │ │ │ │ │ + return response; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: decode │ │ │ │ │ - * Deserialize an encoded string and return an array of n-dimensional │ │ │ │ │ - * points. │ │ │ │ │ + * APIMethod: setFeatureType │ │ │ │ │ + * Change the feature type on the fly. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * encoded - {String} An encoded string │ │ │ │ │ - * dims - {int} The dimension of the points that are returned │ │ │ │ │ + * featureType - {String} Local (without prefix) feature typeName. │ │ │ │ │ + */ │ │ │ │ │ + setFeatureType: function(featureType) { │ │ │ │ │ + this.featureType = featureType; │ │ │ │ │ + this.format.featureType = featureType; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setGeometryName │ │ │ │ │ + * Sets the geometryName option after instantiation. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(Array(int))} An array containing n-dimensional arrays of │ │ │ │ │ - * coordinates. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometryName - {String} Name of geometry attribute. │ │ │ │ │ */ │ │ │ │ │ - decode: function(encoded, dims, opt_factor) { │ │ │ │ │ - var factor = opt_factor || 1e5; │ │ │ │ │ - var flatPoints = this.decodeDeltas(encoded, dims, factor); │ │ │ │ │ - var flatPointsLength = flatPoints.length; │ │ │ │ │ + setGeometryName: function(geometryName) { │ │ │ │ │ + this.geometryName = geometryName; │ │ │ │ │ + this.format.geometryName = geometryName; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var points = []; │ │ │ │ │ - for (var i = 0; i + (dims - 1) < flatPointsLength;) { │ │ │ │ │ - var point = []; │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleRead │ │ │ │ │ + * Deal with response from the read request. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ + * to the user callback. │ │ │ │ │ + * options - {Object} The user options passed to the read call. │ │ │ │ │ + */ │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ │ │ │ │ │ - for (var dim = 0; dim < dims; ++dim) { │ │ │ │ │ - point.push(flatPoints[i++]) │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + // success │ │ │ │ │ + var result = this.parseResponse(request, options.readOptions); │ │ │ │ │ + if (result && result.success !== false) { │ │ │ │ │ + if (options.readOptions && options.readOptions.output == "object") { │ │ │ │ │ + OpenLayers.Util.extend(response, result); │ │ │ │ │ + } else { │ │ │ │ │ + response.features = result; │ │ │ │ │ + } │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + } else { │ │ │ │ │ + // failure (service exception) │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + response.error = result; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // failure │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - points.push(point); │ │ │ │ │ + options.callback.call(options.scope, response); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return points; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Serialize a feature or array of features into a WKT string. │ │ │ │ │ + * Method: parseResponse │ │ │ │ │ + * Read HTTP response body and return features │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {<OpenLayers.Feature.Vector>|Array} A feature or array of │ │ │ │ │ - * features │ │ │ │ │ + * request - {XMLHttpRequest} The request object │ │ │ │ │ + * options - {Object} Optional object to pass to format's read │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} The WKT string representation of the input geometries │ │ │ │ │ + * {Object} or {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * An object with a features property, an array of features or a single │ │ │ │ │ + * feature. │ │ │ │ │ */ │ │ │ │ │ - write: function(features) { │ │ │ │ │ - var feature; │ │ │ │ │ - if (features.constructor == Array) │ │ │ │ │ - feature = features[0]; │ │ │ │ │ - else │ │ │ │ │ - feature = features; │ │ │ │ │ - │ │ │ │ │ - var geometry = feature.geometry; │ │ │ │ │ - var type = geometry.CLASS_NAME.split('.')[2].toLowerCase(); │ │ │ │ │ - │ │ │ │ │ - var pointGeometries; │ │ │ │ │ - if (type == "point") │ │ │ │ │ - pointGeometries = new Array(geometry); │ │ │ │ │ - else if (type == "linestring" || │ │ │ │ │ - type == "linearring" || │ │ │ │ │ - type == "multipoint") │ │ │ │ │ - pointGeometries = geometry.components; │ │ │ │ │ - else if (type == "polygon") │ │ │ │ │ - pointGeometries = geometry.components[0].components; │ │ │ │ │ - else │ │ │ │ │ + parseResponse: function(request, options) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ + } │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ return null; │ │ │ │ │ - │ │ │ │ │ - var flatPoints = []; │ │ │ │ │ - │ │ │ │ │ - var pointGeometriesLength = pointGeometries.length; │ │ │ │ │ - for (var i = 0; i < pointGeometriesLength; ++i) { │ │ │ │ │ - var pointGeometry = pointGeometries[i]; │ │ │ │ │ - flatPoints.push(pointGeometry.y); │ │ │ │ │ - flatPoints.push(pointGeometry.x); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return this.encodeDeltas(flatPoints, 2); │ │ │ │ │ + var result = (this.readFormat !== null) ? this.readFormat.read(doc) : │ │ │ │ │ + this.format.read(doc, options); │ │ │ │ │ + if (!this.featureNS) { │ │ │ │ │ + var format = this.readFormat || this.format; │ │ │ │ │ + this.featureNS = format.featureNS; │ │ │ │ │ + // no need to auto-configure again on subsequent reads │ │ │ │ │ + format.autoConfig = false; │ │ │ │ │ + if (!this.geometryName) { │ │ │ │ │ + this.setGeometryName(format.geometryName); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return result; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: encode │ │ │ │ │ - * Serialize an array of n-dimensional points and return an encoded string │ │ │ │ │ + * Method: commit │ │ │ │ │ + * Given a list of feature, assemble a batch request for update, create, │ │ │ │ │ + * and delete transactions. A commit call on the prototype amounts │ │ │ │ │ + * to writing a WFS transaction - so the write method on the format │ │ │ │ │ + * is used. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * points - {Array(Array(int))} An array containing n-dimensional │ │ │ │ │ - * arrays of coordinates │ │ │ │ │ - * dims - {int} The dimension of the points that should be read │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * options - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * nativeElements - {Array({Object})} Array of objects with information for writing │ │ │ │ │ + * out <Native> elements, these objects have vendorId, safeToIgnore and │ │ │ │ │ + * value properties. The <Native> element is intended to allow access to │ │ │ │ │ + * vendor specific capabilities of any particular web feature server or │ │ │ │ │ + * datastore. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} An encoded string │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} A response object with a features │ │ │ │ │ + * property containing any insertIds and a priv property referencing │ │ │ │ │ + * the XMLHttpRequest object. │ │ │ │ │ */ │ │ │ │ │ - encode: function(points, dims, opt_factor) { │ │ │ │ │ - var factor = opt_factor || 1e5; │ │ │ │ │ - var flatPoints = []; │ │ │ │ │ + commit: function(features, options) { │ │ │ │ │ │ │ │ │ │ - var pointsLength = points.length; │ │ │ │ │ - for (var i = 0; i < pointsLength; ++i) { │ │ │ │ │ - var point = points[i]; │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ │ │ │ │ │ - for (var dim = 0; dim < dims; ++dim) { │ │ │ │ │ - flatPoints.push(point[dim]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "commit", │ │ │ │ │ + reqFeatures: features │ │ │ │ │ + }); │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(features, options), │ │ │ │ │ + callback: this.createCallback(this.handleCommit, response, options) │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - return this.encodeDeltas(flatPoints, dims, factor); │ │ │ │ │ + return response; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: encodeDeltas │ │ │ │ │ - * Encode a list of n-dimensional points and return an encoded string │ │ │ │ │ - * │ │ │ │ │ - * Attention: This function will modify the passed array! │ │ │ │ │ - * │ │ │ │ │ + * Method: handleCommit │ │ │ │ │ + * Called when the commit request returns. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * numbers - {Array.<number>} A list of n-dimensional points. │ │ │ │ │ - * dimension - {number} The dimension of the points in the list. │ │ │ │ │ - * opt_factor - {number=} The factor by which the numbers will be │ │ │ │ │ - * multiplied. The remaining decimal places will get rounded away. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {string} The encoded string. │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ + * to the user callback. │ │ │ │ │ + * options - {Object} The user options passed to the commit call. │ │ │ │ │ */ │ │ │ │ │ - encodeDeltas: function(numbers, dimension, opt_factor) { │ │ │ │ │ - var factor = opt_factor || 1e5; │ │ │ │ │ - var d; │ │ │ │ │ + handleCommit: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ │ │ │ │ │ - var lastNumbers = new Array(dimension); │ │ │ │ │ - for (d = 0; d < dimension; ++d) { │ │ │ │ │ - lastNumbers[d] = 0; │ │ │ │ │ - } │ │ │ │ │ + // ensure that we have an xml doc │ │ │ │ │ + var data = request.responseXML; │ │ │ │ │ + if (!data || !data.documentElement) { │ │ │ │ │ + data = request.responseText; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var numbersLength = numbers.length; │ │ │ │ │ - for (var i = 0; i < numbersLength;) { │ │ │ │ │ - for (d = 0; d < dimension; ++d, ++i) { │ │ │ │ │ - var num = numbers[i]; │ │ │ │ │ - var delta = num - lastNumbers[d]; │ │ │ │ │ - lastNumbers[d] = num; │ │ │ │ │ + var obj = this.format.read(data) || {}; │ │ │ │ │ │ │ │ │ │ - numbers[i] = delta; │ │ │ │ │ + response.insertIds = obj.insertIds || []; │ │ │ │ │ + if (obj.success) { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + response.error = obj; │ │ │ │ │ } │ │ │ │ │ + options.callback.call(options.scope, response); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return this.encodeFloats(numbers, factor); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: decodeDeltas │ │ │ │ │ - * Decode a list of n-dimensional points from an encoded string │ │ │ │ │ - * │ │ │ │ │ + * Method: filterDelete │ │ │ │ │ + * Send a request that deletes all features by their filter. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * encoded - {string} An encoded string. │ │ │ │ │ - * dimension - {number} The dimension of the points in the encoded string. │ │ │ │ │ - * opt_factor - {number=} The factor by which the resulting numbers will │ │ │ │ │ - * be divided. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array.<number>} A list of n-dimensional points. │ │ │ │ │ + * filter - {<OpenLayers.Filter>} filter │ │ │ │ │ */ │ │ │ │ │ - decodeDeltas: function(encoded, dimension, opt_factor) { │ │ │ │ │ - var factor = opt_factor || 1e5; │ │ │ │ │ - var d; │ │ │ │ │ - │ │ │ │ │ - var lastNumbers = new Array(dimension); │ │ │ │ │ - for (d = 0; d < dimension; ++d) { │ │ │ │ │ - lastNumbers[d] = 0; │ │ │ │ │ - } │ │ │ │ │ + filterDelete: function(filter, options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ │ │ │ │ │ - var numbers = this.decodeFloats(encoded, factor); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "commit" │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - var numbersLength = numbers.length; │ │ │ │ │ - for (var i = 0; i < numbersLength;) { │ │ │ │ │ - for (d = 0; d < dimension; ++d, ++i) { │ │ │ │ │ - lastNumbers[d] += numbers[i]; │ │ │ │ │ + var root = this.format.createElementNSPlus("wfs:Transaction", { │ │ │ │ │ + attributes: { │ │ │ │ │ + service: "WFS", │ │ │ │ │ + version: this.version │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - numbers[i] = lastNumbers[d]; │ │ │ │ │ + var deleteNode = this.format.createElementNSPlus("wfs:Delete", { │ │ │ │ │ + attributes: { │ │ │ │ │ + typeName: (options.featureNS ? this.featurePrefix + ":" : "") + │ │ │ │ │ + options.featureType │ │ │ │ │ } │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + if (options.featureNS) { │ │ │ │ │ + deleteNode.setAttribute("xmlns:" + this.featurePrefix, options.featureNS); │ │ │ │ │ } │ │ │ │ │ + var filterNode = this.format.writeNode("ogc:Filter", filter); │ │ │ │ │ │ │ │ │ │ - return numbers; │ │ │ │ │ - }, │ │ │ │ │ + deleteNode.appendChild(filterNode); │ │ │ │ │ │ │ │ │ │ + root.appendChild(deleteNode); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: encodeFloats │ │ │ │ │ - * Encode a list of floating point numbers and return an encoded string │ │ │ │ │ - * │ │ │ │ │ - * Attention: This function will modify the passed array! │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * numbers - {Array.<number>} A list of floating point numbers. │ │ │ │ │ - * opt_factor - {number=} The factor by which the numbers will be │ │ │ │ │ - * multiplied. The remaining decimal places will get rounded away. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {string} The encoded string. │ │ │ │ │ - */ │ │ │ │ │ - encodeFloats: function(numbers, opt_factor) { │ │ │ │ │ - var factor = opt_factor || 1e5; │ │ │ │ │ + var data = OpenLayers.Format.XML.prototype.write.apply( │ │ │ │ │ + this.format, [root] │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - var numbersLength = numbers.length; │ │ │ │ │ - for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ - numbers[i] = Math.round(numbers[i] * factor); │ │ │ │ │ - } │ │ │ │ │ + return OpenLayers.Request.POST({ │ │ │ │ │ + url: this.url, │ │ │ │ │ + callback: options.callback || function() {}, │ │ │ │ │ + data: data │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - return this.encodeSignedIntegers(numbers); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: decodeFloats │ │ │ │ │ - * Decode a list of floating point numbers from an encoded string │ │ │ │ │ + * Method: abort │ │ │ │ │ + * Abort an ongoing request, the response object passed to │ │ │ │ │ + * this method must come from this protocol (as a result │ │ │ │ │ + * of a read, or commit operation). │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * encoded - {string} An encoded string. │ │ │ │ │ - * opt_factor - {number=} The factor by which the result will be divided. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array.<number>} A list of floating point numbers. │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ */ │ │ │ │ │ - decodeFloats: function(encoded, opt_factor) { │ │ │ │ │ - var factor = opt_factor || 1e5; │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + response.priv.abort(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var numbers = this.decodeSignedIntegers(encoded); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.WFS.v1" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Protocol/WFS/v1_1_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - var numbersLength = numbers.length; │ │ │ │ │ - for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ - numbers[i] /= factor; │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - return numbers; │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Protocol/WFS/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/WFST/v1_1_0.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.WFS.v1_1_0 │ │ │ │ │ + * A WFS v1.1.0 protocol for vector layers. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Protocol.WFS.v1_1_0> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Differences from the v1.0.0 protocol: │ │ │ │ │ + * - uses Filter Encoding 1.1.0 instead of 1.0.0 │ │ │ │ │ + * - uses GML 3 instead of 2 if no format is provided │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Protocol.WFS.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.WFS.v1_1_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} WFS version number. │ │ │ │ │ + */ │ │ │ │ │ + version: "1.1.0", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: encodeSignedIntegers │ │ │ │ │ - * Encode a list of signed integers and return an encoded string │ │ │ │ │ - * │ │ │ │ │ - * Attention: This function will modify the passed array! │ │ │ │ │ + * Constructor: OpenLayers.Protocol.WFS.v1_1_0 │ │ │ │ │ + * A class for giving layers WFS v1.1.0 protocol. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * numbers - {Array.<number>} A list of signed integers. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {string} The encoded string. │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ + * featureNS - {String} Feature namespace (optional). │ │ │ │ │ + * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ + * if featureNS is provided). Default is 'feature'. │ │ │ │ │ + * geometryName - {String} Name of geometry attribute. Default is 'the_geom'. │ │ │ │ │ + * outputFormat - {String} Optional output format to use for WFS GetFeature │ │ │ │ │ + * requests. This can be any format advertized by the WFS's │ │ │ │ │ + * GetCapabilities response. If set, an appropriate readFormat also │ │ │ │ │ + * has to be provided, unless outputFormat is GML3, GML2 or JSON. │ │ │ │ │ + * readFormat - {<OpenLayers.Format>} An appropriate format parser if │ │ │ │ │ + * outputFormat is none of GML3, GML2 or JSON. │ │ │ │ │ */ │ │ │ │ │ - encodeSignedIntegers: function(numbers) { │ │ │ │ │ - var numbersLength = numbers.length; │ │ │ │ │ - for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ - var num = numbers[i]; │ │ │ │ │ - │ │ │ │ │ - var signedNum = num << 1; │ │ │ │ │ - if (num < 0) { │ │ │ │ │ - signedNum = ~(signedNum); │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Protocol.WFS.v1.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (this.outputFormat && !this.readFormat) { │ │ │ │ │ + if (this.outputFormat.toLowerCase() == "gml2") { │ │ │ │ │ + this.readFormat = new OpenLayers.Format.GML.v2({ │ │ │ │ │ + featureType: this.featureType, │ │ │ │ │ + featureNS: this.featureNS, │ │ │ │ │ + geometryName: this.geometryName │ │ │ │ │ + }); │ │ │ │ │ + } else if (this.outputFormat.toLowerCase() == "json") { │ │ │ │ │ + this.readFormat = new OpenLayers.Format.GeoJSON(); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - numbers[i] = signedNum; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return this.encodeUnsignedIntegers(numbers); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.WFS.v1_1_0" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/GML/v2.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/GML/Base.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.GML.v2 │ │ │ │ │ + * Parses GML version 2. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.GML.Base> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.GML.v2 = OpenLayers.Class(OpenLayers.Format.GML.Base, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: decodeSignedIntegers │ │ │ │ │ - * Decode a list of signed integers from an encoded string │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} Schema location for a particular minor version. │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: "http://www.opengis.net/gml http://schemas.opengis.net/gml/2.1.2/feature.xsd", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.GML.v2 │ │ │ │ │ + * Create a parser for GML v2. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * encoded - {string} An encoded string. │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array.<number>} A list of signed integers. │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ + * featureNS - {String} Feature namespace (required). │ │ │ │ │ + * geometryName - {String} Geometry element name. │ │ │ │ │ */ │ │ │ │ │ - decodeSignedIntegers: function(encoded) { │ │ │ │ │ - var numbers = this.decodeUnsignedIntegers(encoded); │ │ │ │ │ - │ │ │ │ │ - var numbersLength = numbers.length; │ │ │ │ │ - for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ - var num = numbers[i]; │ │ │ │ │ - numbers[i] = (num & 1) ? ~(num >> 1) : (num >> 1); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return numbers; │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.GML.Base.prototype.initialize.apply(this, [options]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "outerBoundaryIs": function(node, container) { │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + container.outer = obj.components[0]; │ │ │ │ │ + }, │ │ │ │ │ + "innerBoundaryIs": function(node, container) { │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + container.inner.push(obj.components[0]); │ │ │ │ │ + }, │ │ │ │ │ + "Box": function(node, container) { │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + if (!container.components) { │ │ │ │ │ + container.components = []; │ │ │ │ │ + } │ │ │ │ │ + var min = obj.points[0]; │ │ │ │ │ + var max = obj.points[1]; │ │ │ │ │ + container.components.push( │ │ │ │ │ + new OpenLayers.Bounds(min.x, min.y, max.x, max.y) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.GML.Base.prototype.readers["gml"]), │ │ │ │ │ + "feature": OpenLayers.Format.GML.Base.prototype.readers["feature"], │ │ │ │ │ + "wfs": OpenLayers.Format.GML.Base.prototype.readers["wfs"] │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: encodeUnsignedIntegers │ │ │ │ │ - * Encode a list of unsigned integers and return an encoded string │ │ │ │ │ + * Method: write │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * numbers - {Array.<number>} A list of unsigned integers. │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>) | OpenLayers.Feature.Vector} │ │ │ │ │ + * An array of features or a single feature. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {string} The encoded string. │ │ │ │ │ + * {String} Given an array of features, a doc with a gml:featureMembers │ │ │ │ │ + * element will be returned. Given a single feature, a doc with a │ │ │ │ │ + * gml:featureMember element will be returned. │ │ │ │ │ */ │ │ │ │ │ - encodeUnsignedIntegers: function(numbers) { │ │ │ │ │ - var encoded = ''; │ │ │ │ │ - │ │ │ │ │ - var numbersLength = numbers.length; │ │ │ │ │ - for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ - encoded += this.encodeUnsignedInteger(numbers[i]); │ │ │ │ │ + write: function(features) { │ │ │ │ │ + var name; │ │ │ │ │ + if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ + // GML2 only has abstract feature collections │ │ │ │ │ + // wfs provides a feature collection from a well-known schema │ │ │ │ │ + name = "wfs:FeatureCollection"; │ │ │ │ │ + } else { │ │ │ │ │ + name = "gml:featureMember"; │ │ │ │ │ } │ │ │ │ │ + var root = this.writeNode(name, features); │ │ │ │ │ + this.setAttributeNS( │ │ │ │ │ + root, this.namespaces["xsi"], │ │ │ │ │ + "xsi:schemaLocation", this.schemaLocation │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - return encoded; │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [root]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: decodeUnsignedIntegers │ │ │ │ │ - * Decode a list of unsigned integers from an encoded string │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * encoded - {string} An encoded string. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array.<number>} A list of unsigned integers. │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ */ │ │ │ │ │ - decodeUnsignedIntegers: function(encoded) { │ │ │ │ │ - var numbers = []; │ │ │ │ │ + writers: { │ │ │ │ │ + "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "Point": function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:Point"); │ │ │ │ │ + this.writeNode("coordinates", [geometry], node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "coordinates": function(points) { │ │ │ │ │ + var numPoints = points.length; │ │ │ │ │ + var parts = new Array(numPoints); │ │ │ │ │ + var point; │ │ │ │ │ + for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ + point = points[i]; │ │ │ │ │ + if (this.xy) { │ │ │ │ │ + parts[i] = point.x + "," + point.y; │ │ │ │ │ + } else { │ │ │ │ │ + parts[i] = point.y + "," + point.x; │ │ │ │ │ + } │ │ │ │ │ + if (point.z != undefined) { // allow null or undefined │ │ │ │ │ + parts[i] += "," + point.z; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return this.createElementNSPlus("gml:coordinates", { │ │ │ │ │ + attributes: { │ │ │ │ │ + decimal: ".", │ │ │ │ │ + cs: ",", │ │ │ │ │ + ts: " " │ │ │ │ │ + }, │ │ │ │ │ + value: (numPoints == 1) ? parts[0] : parts.join(" ") │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "LineString": function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:LineString"); │ │ │ │ │ + this.writeNode("coordinates", geometry.components, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Polygon": function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:Polygon"); │ │ │ │ │ + this.writeNode("outerBoundaryIs", geometry.components[0], node); │ │ │ │ │ + for (var i = 1; i < geometry.components.length; ++i) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "innerBoundaryIs", geometry.components[i], node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "outerBoundaryIs": function(ring) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:outerBoundaryIs"); │ │ │ │ │ + this.writeNode("LinearRing", ring, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "innerBoundaryIs": function(ring) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:innerBoundaryIs"); │ │ │ │ │ + this.writeNode("LinearRing", ring, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "LinearRing": function(ring) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:LinearRing"); │ │ │ │ │ + this.writeNode("coordinates", ring.components, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Box": function(bounds) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:Box"); │ │ │ │ │ + this.writeNode("coordinates", [{ │ │ │ │ │ + x: bounds.left, │ │ │ │ │ + y: bounds.bottom │ │ │ │ │ + }, { │ │ │ │ │ + x: bounds.right, │ │ │ │ │ + y: bounds.top │ │ │ │ │ + }], node); │ │ │ │ │ + // srsName attribute is optional for gml:Box │ │ │ │ │ + if (this.srsName) { │ │ │ │ │ + node.setAttribute("srsName", this.srsName); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.GML.Base.prototype.writers["gml"]), │ │ │ │ │ + "feature": OpenLayers.Format.GML.Base.prototype.writers["feature"], │ │ │ │ │ + "wfs": OpenLayers.Format.GML.Base.prototype.writers["wfs"] │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var current = 0; │ │ │ │ │ - var shift = 0; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.GML.v2" │ │ │ │ │ │ │ │ │ │ - var encodedLength = encoded.length; │ │ │ │ │ - for (var i = 0; i < encodedLength; ++i) { │ │ │ │ │ - var b = encoded.charCodeAt(i) - 63; │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/Filter/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - current |= (b & 0x1f) << shift; │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - if (b < 0x20) { │ │ │ │ │ - numbers.push(current); │ │ │ │ │ - current = 0; │ │ │ │ │ - shift = 0; │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/GML/v2.js │ │ │ │ │ + * @requires OpenLayers/Format/Filter/v1.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.Filter.v1_0_0 │ │ │ │ │ + * Write ogc:Filter version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.GML.v2> │ │ │ │ │ + * - <OpenLayers.Format.Filter.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.Filter.v1_0_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.GML.v2, OpenLayers.Format.Filter.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: VERSION │ │ │ │ │ + * {String} 1.0.0 │ │ │ │ │ + */ │ │ │ │ │ + VERSION: "1.0.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} http://www.opengis.net/ogc/filter/1.0.0/filter.xsd │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: "http://www.opengis.net/ogc/filter/1.0.0/filter.xsd", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.Filter.v1_0_0 │ │ │ │ │ + * Instances of this class are not created directly. Use the │ │ │ │ │ + * <OpenLayers.Format.Filter> constructor instead. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.GML.v2.prototype.initialize.apply( │ │ │ │ │ + this, [options] │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "ogc": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "PropertyIsEqualTo": function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.EQUAL_TO │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + obj.filters.push(filter); │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsNotEqualTo": function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.NOT_EQUAL_TO │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + obj.filters.push(filter); │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsLike": function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.LIKE │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + var wildCard = node.getAttribute("wildCard"); │ │ │ │ │ + var singleChar = node.getAttribute("singleChar"); │ │ │ │ │ + var esc = node.getAttribute("escape"); │ │ │ │ │ + filter.value2regex(wildCard, singleChar, esc); │ │ │ │ │ + obj.filters.push(filter); │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.Filter.v1.prototype.readers["ogc"]), │ │ │ │ │ + "gml": OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ + "feature": OpenLayers.Format.GML.v2.prototype.readers["feature"] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ + */ │ │ │ │ │ + writers: { │ │ │ │ │ + "ogc": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "PropertyIsEqualTo": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsEqualTo"); │ │ │ │ │ + // no ogc:expression handling for PropertyName for now │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + // handle Literals or Functions for now │ │ │ │ │ + this.writeOgcExpression(filter.value, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsNotEqualTo": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsNotEqualTo"); │ │ │ │ │ + // no ogc:expression handling for PropertyName for now │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + // handle Literals or Functions for now │ │ │ │ │ + this.writeOgcExpression(filter.value, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsLike": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsLike", { │ │ │ │ │ + attributes: { │ │ │ │ │ + wildCard: "*", │ │ │ │ │ + singleChar: ".", │ │ │ │ │ + escape: "!" │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + // no ogc:expression handling for now │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + // convert regex string to ogc string │ │ │ │ │ + this.writeNode("Literal", filter.regex2value(), node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "BBOX": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:BBOX"); │ │ │ │ │ + // PropertyName is mandatory in 1.0.0, but e.g. GeoServer also │ │ │ │ │ + // accepts filters without it. When this is used with │ │ │ │ │ + // OpenLayers.Protocol.WFS, OpenLayers.Format.WFST will set a │ │ │ │ │ + // missing filter.property to the geometryName that is │ │ │ │ │ + // configured with the protocol, which defaults to "the_geom". │ │ │ │ │ + // So the only way to omit this mandatory property is to not │ │ │ │ │ + // set the property on the filter and to set the geometryName │ │ │ │ │ + // on the WFS protocol to null. The latter also happens when │ │ │ │ │ + // the protocol is configured without a geometryName and a │ │ │ │ │ + // featureNS. │ │ │ │ │ + filter.property && this.writeNode("PropertyName", filter, node); │ │ │ │ │ + var box = this.writeNode("gml:Box", filter.value, node); │ │ │ │ │ + if (filter.projection) { │ │ │ │ │ + box.setAttribute("srsName", filter.projection); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.Filter.v1.prototype.writers["ogc"]), │ │ │ │ │ + "gml": OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ + "feature": OpenLayers.Format.GML.v2.prototype.writers["feature"] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: writeSpatial │ │ │ │ │ + * │ │ │ │ │ + * Read a {<OpenLayers.Filter.Spatial>} filter and converts it into XML. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {<OpenLayers.Filter.Spatial>} The filter. │ │ │ │ │ + * name - {String} Name of the generated XML element. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The created XML element. │ │ │ │ │ + */ │ │ │ │ │ + writeSpatial: function(filter, name) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:" + name); │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + if (filter.value instanceof OpenLayers.Filter.Function) { │ │ │ │ │ + this.writeNode("Function", filter.value, node); │ │ │ │ │ } else { │ │ │ │ │ - shift += 5; │ │ │ │ │ + var child; │ │ │ │ │ + if (filter.value instanceof OpenLayers.Geometry) { │ │ │ │ │ + child = this.writeNode("feature:_geometry", filter.value).firstChild; │ │ │ │ │ + } else { │ │ │ │ │ + child = this.writeNode("gml:Box", filter.value); │ │ │ │ │ + } │ │ │ │ │ + if (filter.projection) { │ │ │ │ │ + child.setAttribute("srsName", filter.projection); │ │ │ │ │ + } │ │ │ │ │ + node.appendChild(child); │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return numbers; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.Filter.v1_0_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WFST/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WFST/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/Filter/v1_0_0.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WFST.v1_0_0 │ │ │ │ │ + * A format for creating WFS v1.0.0 transactions. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Format.WFST.v1_0_0> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.Filter.v1_0_0> │ │ │ │ │ + * - <OpenLayers.Format.WFST.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WFST.v1_0_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.Filter.v1_0_0, OpenLayers.Format.WFST.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} WFS version number. │ │ │ │ │ + */ │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: srsNameInQuery │ │ │ │ │ + * {Boolean} If true the reference system is passed in Query requests │ │ │ │ │ + * via the "srsName" attribute to the "wfs:Query" element, this │ │ │ │ │ + * property defaults to false as it isn't WFS 1.0.0 compliant. │ │ │ │ │ + */ │ │ │ │ │ + srsNameInQuery: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocations │ │ │ │ │ + * {Object} Properties are namespace aliases, values are schema locations. │ │ │ │ │ + */ │ │ │ │ │ + schemaLocations: { │ │ │ │ │ + "wfs": "http://schemas.opengis.net/wfs/1.0.0/WFS-transaction.xsd" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WFST.v1_0_0 │ │ │ │ │ + * A class for parsing and generating WFS v1.0.0 transactions. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + * │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ + * featureNS - {String} Feature namespace (optional). │ │ │ │ │ + * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ + * if featureNS is provided). Default is 'feature'. │ │ │ │ │ + * geometryName - {String} Name of geometry attribute. Default is 'the_geom'. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.Filter.v1_0_0.prototype.initialize.apply(this, [options]); │ │ │ │ │ + OpenLayers.Format.WFST.v1.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: readNode │ │ │ │ │ + * Shorthand for applying one of the named readers given the node │ │ │ │ │ + * namespace and local name. Readers take two args (node, obj) and │ │ │ │ │ + * generally extend or modify the second. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The node to be read (required). │ │ │ │ │ + * obj - {Object} The object to be modified (optional). │ │ │ │ │ + * first - {Boolean} Should be set to true for the first node read. This │ │ │ │ │ + * is usually the readNode call in the read method. Without this being │ │ │ │ │ + * set, auto-configured properties will stick on subsequent reads. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The input object, modified (or a new one if none was provided). │ │ │ │ │ + */ │ │ │ │ │ + readNode: function(node, obj, first) { │ │ │ │ │ + // Not the superclass, only the mixin classes inherit from │ │ │ │ │ + // Format.GML.v2. We need this because we don't want to get readNode │ │ │ │ │ + // from the superclass's superclass, which is OpenLayers.Format.XML. │ │ │ │ │ + return OpenLayers.Format.GML.v2.prototype.readNode.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "WFS_TransactionResponse": function(node, obj) { │ │ │ │ │ + obj.insertIds = []; │ │ │ │ │ + obj.success = false; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "InsertResult": function(node, container) { │ │ │ │ │ + var obj = { │ │ │ │ │ + fids: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + container.insertIds = container.insertIds.concat(obj.fids); │ │ │ │ │ + }, │ │ │ │ │ + "TransactionResult": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "Status": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "SUCCESS": function(node, obj) { │ │ │ │ │ + obj.success = true; │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WFST.v1.prototype.readers["wfs"]), │ │ │ │ │ + "gml": OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ + "feature": OpenLayers.Format.GML.v2.prototype.readers["feature"], │ │ │ │ │ + "ogc": OpenLayers.Format.Filter.v1_0_0.prototype.readers["ogc"] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ + */ │ │ │ │ │ + writers: { │ │ │ │ │ + "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "Query": function(options) { │ │ │ │ │ + options = OpenLayers.Util.extend({ │ │ │ │ │ + featureNS: this.featureNS, │ │ │ │ │ + featurePrefix: this.featurePrefix, │ │ │ │ │ + featureType: this.featureType, │ │ │ │ │ + srsName: this.srsName, │ │ │ │ │ + srsNameInQuery: this.srsNameInQuery │ │ │ │ │ + }, options); │ │ │ │ │ + var prefix = options.featurePrefix; │ │ │ │ │ + var node = this.createElementNSPlus("wfs:Query", { │ │ │ │ │ + attributes: { │ │ │ │ │ + typeName: (prefix ? prefix + ":" : "") + │ │ │ │ │ + options.featureType │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (options.srsNameInQuery && options.srsName) { │ │ │ │ │ + node.setAttribute("srsName", options.srsName); │ │ │ │ │ + } │ │ │ │ │ + if (options.featureNS) { │ │ │ │ │ + node.setAttribute("xmlns:" + prefix, options.featureNS); │ │ │ │ │ + } │ │ │ │ │ + if (options.propertyNames) { │ │ │ │ │ + for (var i = 0, len = options.propertyNames.length; i < len; i++) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "ogc:PropertyName", { │ │ │ │ │ + property: options.propertyNames[i] │ │ │ │ │ + }, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (options.filter) { │ │ │ │ │ + this.setFilterProperty(options.filter); │ │ │ │ │ + this.writeNode("ogc:Filter", options.filter, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WFST.v1.prototype.writers["wfs"]), │ │ │ │ │ + "gml": OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ + "feature": OpenLayers.Format.GML.v2.prototype.writers["feature"], │ │ │ │ │ + "ogc": OpenLayers.Format.Filter.v1_0_0.prototype.writers["ogc"] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFST.v1_0_0" │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Protocol/WFS/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Protocol/WFS/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/WFST/v1_0_0.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.WFS.v1_0_0 │ │ │ │ │ + * A WFS v1.0.0 protocol for vector layers. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Protocol.WFS.v1_0_0> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Protocol.WFS.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.WFS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} WFS version number. │ │ │ │ │ + */ │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: encodeFloat │ │ │ │ │ - * Encode one single floating point number and return an encoded string │ │ │ │ │ + * Constructor: OpenLayers.Protocol.WFS.v1_0_0 │ │ │ │ │ + * A class for giving layers WFS v1.0.0 protocol. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * num - {number} Floating point number that should be encoded. │ │ │ │ │ - * opt_factor - {number=} The factor by which num will be multiplied. │ │ │ │ │ - * The remaining decimal places will get rounded away. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {string} The encoded string. │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ + * featureNS - {String} Feature namespace (optional). │ │ │ │ │ + * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ + * if featureNS is provided). Default is 'feature'. │ │ │ │ │ + * geometryName - {String} Name of geometry attribute. Default is 'the_geom'. │ │ │ │ │ */ │ │ │ │ │ - encodeFloat: function(num, opt_factor) { │ │ │ │ │ - num = Math.round(num * (opt_factor || 1e5)); │ │ │ │ │ - return this.encodeSignedInteger(num); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.WFS.v1_0_0" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/SOSGetFeatureOfInterest.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Format/GML/v3.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.SOSGetFeatureOfInterest │ │ │ │ │ + * Read and write SOS GetFeatureOfInterest. This is used to get to │ │ │ │ │ + * the location of the features (stations). The stations can have 1 or more │ │ │ │ │ + * sensors. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.SOSGetFeatureOfInterest = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: VERSION │ │ │ │ │ + * {String} 1.0.0 │ │ │ │ │ + */ │ │ │ │ │ + VERSION: "1.0.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ + gml: "http://www.opengis.net/gml", │ │ │ │ │ + sa: "http://www.opengis.net/sampling/1.0", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} Schema location │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: "http://www.opengis.net/sos/1.0 http://schemas.opengis.net/sos/1.0.0/sosAll.xsd", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: "sos", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ + */ │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.SOSGetFeatureOfInterest │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Parse a GetFeatureOfInterest response and return an array of features │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var info = { │ │ │ │ │ + features: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readNode(data, info); │ │ │ │ │ + │ │ │ │ │ + var features = []; │ │ │ │ │ + for (var i = 0, len = info.features.length; i < len; i++) { │ │ │ │ │ + var container = info.features[i]; │ │ │ │ │ + // reproject features if needed │ │ │ │ │ + if (this.internalProjection && this.externalProjection && │ │ │ │ │ + container.components[0]) { │ │ │ │ │ + container.components[0].transform( │ │ │ │ │ + this.externalProjection, this.internalProjection │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector( │ │ │ │ │ + container.components[0], container.attributes); │ │ │ │ │ + features.push(feature); │ │ │ │ │ + } │ │ │ │ │ + return features; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "sa": { │ │ │ │ │ + "SamplingPoint": function(node, obj) { │ │ │ │ │ + // sampling point can also be without a featureMember if │ │ │ │ │ + // there is only 1 │ │ │ │ │ + if (!obj.attributes) { │ │ │ │ │ + var feature = { │ │ │ │ │ + attributes: {} │ │ │ │ │ + }; │ │ │ │ │ + obj.features.push(feature); │ │ │ │ │ + obj = feature; │ │ │ │ │ + } │ │ │ │ │ + obj.attributes.id = this.getAttributeNS(node, │ │ │ │ │ + this.namespaces.gml, "id"); │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "position": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "FeatureCollection": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "featureMember": function(node, obj) { │ │ │ │ │ + var feature = { │ │ │ │ │ + attributes: {} │ │ │ │ │ + }; │ │ │ │ │ + obj.features.push(feature); │ │ │ │ │ + this.readChildNodes(node, feature); │ │ │ │ │ + }, │ │ │ │ │ + "name": function(node, obj) { │ │ │ │ │ + obj.attributes.name = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "pos": function(node, obj) { │ │ │ │ │ + // we need to parse the srsName to get to the │ │ │ │ │ + // externalProjection, that's why we cannot use │ │ │ │ │ + // GML v3 for this │ │ │ │ │ + if (!this.externalProjection) { │ │ │ │ │ + this.externalProjection = new OpenLayers.Projection( │ │ │ │ │ + node.getAttribute("srsName")); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Format.GML.v3.prototype.readers.gml.pos.apply( │ │ │ │ │ + this, [node, obj]); │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.GML.v3.prototype.readers.gml) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ + */ │ │ │ │ │ + writers: { │ │ │ │ │ + "sos": { │ │ │ │ │ + "GetFeatureOfInterest": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("GetFeatureOfInterest", { │ │ │ │ │ + attributes: { │ │ │ │ │ + version: this.VERSION, │ │ │ │ │ + service: 'SOS', │ │ │ │ │ + "xsi:schemaLocation": this.schemaLocation │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + for (var i = 0, len = options.fois.length; i < len; i++) { │ │ │ │ │ + this.writeNode("FeatureOfInterestId", { │ │ │ │ │ + foi: options.fois[i] │ │ │ │ │ + }, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "FeatureOfInterestId": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("FeatureOfInterestId", { │ │ │ │ │ + value: options.foi │ │ │ │ │ + }); │ │ │ │ │ + return node; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SOSGetFeatureOfInterest" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Protocol/SOS/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Protocol/SOS.js │ │ │ │ │ + * @requires OpenLayers/Format/SOSGetFeatureOfInterest.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.SOS.v1_0_0 │ │ │ │ │ + * An SOS v1.0.0 Protocol for vector layers. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Protocol.SOS.v1_0_0> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Protocol> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.SOS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: decodeFloat │ │ │ │ │ - * Decode one single floating point number from an encoded string │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * encoded - {string} An encoded string. │ │ │ │ │ - * opt_factor - {number=} The factor by which the result will be divided. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {number} The decoded floating point number. │ │ │ │ │ + * APIProperty: fois │ │ │ │ │ + * {Array(String)} Array of features of interest (foi) │ │ │ │ │ */ │ │ │ │ │ - decodeFloat: function(encoded, opt_factor) { │ │ │ │ │ - var result = this.decodeSignedInteger(encoded); │ │ │ │ │ - return result / (opt_factor || 1e5); │ │ │ │ │ - }, │ │ │ │ │ + fois: null, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: formatOptions │ │ │ │ │ + * {Object} Optional options for the format. If a format is not provided, │ │ │ │ │ + * this property can be used to extend the default format options. │ │ │ │ │ + */ │ │ │ │ │ + formatOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: encodeSignedInteger │ │ │ │ │ - * Encode one single signed integer and return an encoded string │ │ │ │ │ + * Constructor: OpenLayers.Protocol.SOS │ │ │ │ │ + * A class for giving layers an SOS protocol. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * num - {number} Signed integer that should be encoded. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {string} The encoded string. │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * url - {String} URL to send requests to (required). │ │ │ │ │ + * fois - {Array} The features of interest (required). │ │ │ │ │ */ │ │ │ │ │ - encodeSignedInteger: function(num) { │ │ │ │ │ - var signedNum = num << 1; │ │ │ │ │ - if (num < 0) { │ │ │ │ │ - signedNum = ~(signedNum); │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!options.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.SOSGetFeatureOfInterest( │ │ │ │ │ + this.formatOptions); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return this.encodeUnsignedInteger(signedNum); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: decodeSignedInteger │ │ │ │ │ - * Decode one single signed integer from an encoded string │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * encoded - {string} An encoded string. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {number} The decoded signed integer. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up the protocol. │ │ │ │ │ */ │ │ │ │ │ - decodeSignedInteger: function(encoded) { │ │ │ │ │ - var result = this.decodeUnsignedInteger(encoded); │ │ │ │ │ - return ((result & 1) ? ~(result >> 1) : (result >> 1)); │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.options && !this.options.format) { │ │ │ │ │ + this.format.destroy(); │ │ │ │ │ + } │ │ │ │ │ + this.format = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Construct a request for reading new sensor positions. This is done by │ │ │ │ │ + * issuing one GetFeatureOfInterest request. │ │ │ │ │ + */ │ │ │ │ │ + read: function(options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + var format = this.format; │ │ │ │ │ + var data = OpenLayers.Format.XML.prototype.write.apply(format, │ │ │ │ │ + [format.writeNode("sos:GetFeatureOfInterest", { │ │ │ │ │ + fois: this.fois │ │ │ │ │ + })] │ │ │ │ │ + ); │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ + data: data │ │ │ │ │ + }); │ │ │ │ │ + return response; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: encodeUnsignedInteger │ │ │ │ │ - * Encode one single unsigned integer and return an encoded string │ │ │ │ │ + * Method: handleRead │ │ │ │ │ + * Deal with response from the read request. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * num - {number} Unsigned integer that should be encoded. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {string} The encoded string. │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ + * to the user callback. │ │ │ │ │ + * options - {Object} The user options passed to the read call. │ │ │ │ │ */ │ │ │ │ │ - encodeUnsignedInteger: function(num) { │ │ │ │ │ - var value, encoded = ''; │ │ │ │ │ - while (num >= 0x20) { │ │ │ │ │ - value = (0x20 | (num & 0x1f)) + 63; │ │ │ │ │ - encoded += (String.fromCharCode(value)); │ │ │ │ │ - num >>= 5; │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + // success │ │ │ │ │ + response.features = this.parseFeatures(request); │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + } else { │ │ │ │ │ + // failure │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + } │ │ │ │ │ + options.callback.call(options.scope, response); │ │ │ │ │ } │ │ │ │ │ - value = num + 63; │ │ │ │ │ - encoded += (String.fromCharCode(value)); │ │ │ │ │ - return encoded; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: decodeUnsignedInteger │ │ │ │ │ - * Decode one single unsigned integer from an encoded string │ │ │ │ │ + * Method: parseFeatures │ │ │ │ │ + * Read HTTP response body and return features │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * encoded - {string} An encoded string. │ │ │ │ │ + * request - {XMLHttpRequest} The request object │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {number} The decoded unsigned integer. │ │ │ │ │ + * {Array({<OpenLayers.Feature.Vector>})} Array of features │ │ │ │ │ */ │ │ │ │ │ - decodeUnsignedInteger: function(encoded) { │ │ │ │ │ - var result = 0; │ │ │ │ │ - var shift = 0; │ │ │ │ │ - │ │ │ │ │ - var encodedLength = encoded.length; │ │ │ │ │ - for (var i = 0; i < encodedLength; ++i) { │ │ │ │ │ - var b = encoded.charCodeAt(i) - 63; │ │ │ │ │ + parseFeatures: function(request) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ + } │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + return this.format.read(doc); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - result |= (b & 0x1f) << shift; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.SOS.v1_0_0" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/CSWGetRecords.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - if (b < 0x20) │ │ │ │ │ - break; │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - shift += 5; │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - return result; │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.CSWGetRecords │ │ │ │ │ + * Default version is 2.0.2. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Format>} A CSWGetRecords format of the given version. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.CSWGetRecords = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options, OpenLayers.Format.CSWGetRecords.DEFAULTS │ │ │ │ │ + ); │ │ │ │ │ + var cls = OpenLayers.Format.CSWGetRecords["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported CSWGetRecords version: " + options.version; │ │ │ │ │ + } │ │ │ │ │ + return new cls(options); │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.EncodedPolyline" │ │ │ │ │ -}); │ │ │ │ │ +/** │ │ │ │ │ + * Constant: DEFAULTS │ │ │ │ │ + * {Object} Default properties for the CSWGetRecords format. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.CSWGetRecords.DEFAULTS = { │ │ │ │ │ + "version": "2.0.2" │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSGetFeatureInfo.js │ │ │ │ │ + OpenLayers/Format/CSWGetRecords/v2_0_2.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Format/CSWGetRecords.js │ │ │ │ │ + * @requires OpenLayers/Format/Filter/v1_0_0.js │ │ │ │ │ + * @requires OpenLayers/Format/Filter/v1_1_0.js │ │ │ │ │ + * @requires OpenLayers/Format/OWSCommon/v1_0_0.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WMSGetFeatureInfo │ │ │ │ │ - * Class to read GetFeatureInfo responses from Web Mapping Services │ │ │ │ │ + * Class: OpenLayers.Format.CSWGetRecords.v2_0_2 │ │ │ │ │ + * A format for creating CSWGetRecords v2.0.2 transactions. │ │ │ │ │ + * Create a new instance with the │ │ │ │ │ + * <OpenLayers.Format.CSWGetRecords.v2_0_2> constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Format.XML> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WMSGetFeatureInfo = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Format.CSWGetRecords.v2_0_2 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: layerIdentifier │ │ │ │ │ - * {String} All xml nodes containing this search criteria will populate an │ │ │ │ │ - * internal array of layer nodes. │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ */ │ │ │ │ │ - layerIdentifier: '_layer', │ │ │ │ │ + namespaces: { │ │ │ │ │ + csw: "http://www.opengis.net/cat/csw/2.0.2", │ │ │ │ │ + dc: "http://purl.org/dc/elements/1.1/", │ │ │ │ │ + dct: "http://purl.org/dc/terms/", │ │ │ │ │ + gmd: "http://www.isotc211.org/2005/gmd", │ │ │ │ │ + geonet: "http://www.fao.org/geonetwork", │ │ │ │ │ + ogc: "http://www.opengis.net/ogc", │ │ │ │ │ + ows: "http://www.opengis.net/ows", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: featureIdentifier │ │ │ │ │ - * {String} All xml nodes containing this search criteria will populate an │ │ │ │ │ - * internal array of feature nodes for each layer node found. │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + * {String} The default prefix (used by Format.XML). │ │ │ │ │ */ │ │ │ │ │ - featureIdentifier: '_feature', │ │ │ │ │ + defaultPrefix: "csw", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} CSW version number. │ │ │ │ │ + */ │ │ │ │ │ + version: "2.0.2", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} http://www.opengis.net/cat/csw/2.0.2 │ │ │ │ │ + * http://schemas.opengis.net/csw/2.0.2/CSW-discovery.xsd │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: "http://www.opengis.net/cat/csw/2.0.2 http://schemas.opengis.net/csw/2.0.2/CSW-discovery.xsd", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: requestId │ │ │ │ │ + * {String} Value of the requestId attribute of the GetRecords element. │ │ │ │ │ + */ │ │ │ │ │ + requestId: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: resultType │ │ │ │ │ + * {String} Value of the resultType attribute of the GetRecords element, │ │ │ │ │ + * specifies the result type in the GetRecords response, "hits" is │ │ │ │ │ + * the default. │ │ │ │ │ + */ │ │ │ │ │ + resultType: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: outputFormat │ │ │ │ │ + * {String} Value of the outputFormat attribute of the GetRecords element, │ │ │ │ │ + * specifies the format of the GetRecords response, │ │ │ │ │ + * "application/xml" is the default. │ │ │ │ │ + */ │ │ │ │ │ + outputFormat: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: outputSchema │ │ │ │ │ + * {String} Value of the outputSchema attribute of the GetRecords element, │ │ │ │ │ + * specifies the schema of the GetRecords response. │ │ │ │ │ + */ │ │ │ │ │ + outputSchema: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: startPosition │ │ │ │ │ + * {String} Value of the startPosition attribute of the GetRecords element, │ │ │ │ │ + * specifies the start position (offset+1) for the GetRecords response, │ │ │ │ │ + * 1 is the default. │ │ │ │ │ + */ │ │ │ │ │ + startPosition: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxRecords │ │ │ │ │ + * {String} Value of the maxRecords attribute of the GetRecords element, │ │ │ │ │ + * specifies the maximum number of records in the GetRecords response, │ │ │ │ │ + * 10 is the default. │ │ │ │ │ + */ │ │ │ │ │ + maxRecords: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: DistributedSearch │ │ │ │ │ + * {String} Value of the csw:DistributedSearch element, used when writing │ │ │ │ │ + * a csw:GetRecords document. │ │ │ │ │ + */ │ │ │ │ │ + DistributedSearch: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: ResponseHandler │ │ │ │ │ + * {Array({String})} Values of the csw:ResponseHandler elements, used when │ │ │ │ │ + * writting a csw:GetRecords document. │ │ │ │ │ + */ │ │ │ │ │ + ResponseHandler: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: Query │ │ │ │ │ + * {String} Value of the csw:Query element, used when writing a csw:GetRecords │ │ │ │ │ + * document. │ │ │ │ │ + */ │ │ │ │ │ + Query: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Property: regExes │ │ │ │ │ * Compiled regular expressions for manipulating strings. │ │ │ │ │ */ │ │ │ │ │ regExes: { │ │ │ │ │ trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ removeSpace: (/\s*/g), │ │ │ │ │ splitSpace: (/\s+/), │ │ │ │ │ trimComma: (/\s*,\s*/g) │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: gmlFormat │ │ │ │ │ - * {<OpenLayers.Format.GML>} internal GML format for parsing geometries │ │ │ │ │ - * in msGMLOutput │ │ │ │ │ - */ │ │ │ │ │ - gmlFormat: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMSGetFeatureInfo │ │ │ │ │ - * Create a new parser for WMS GetFeatureInfo responses │ │ │ │ │ + * Constructor: OpenLayers.Format.CSWGetRecords.v2_0_2 │ │ │ │ │ + * A class for parsing and generating CSWGetRecords v2.0.2 transactions. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + * │ │ │ │ │ + * Valid options properties (documented as class properties): │ │ │ │ │ + * - requestId │ │ │ │ │ + * - resultType │ │ │ │ │ + * - outputFormat │ │ │ │ │ + * - outputSchema │ │ │ │ │ + * - startPosition │ │ │ │ │ + * - maxRecords │ │ │ │ │ + * - DistributedSearch │ │ │ │ │ + * - ResponseHandler │ │ │ │ │ + * - Query │ │ │ │ │ */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIMethod: read │ │ │ │ │ - * Read WMS GetFeatureInfo data from a string, and return an array of features │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ + * Parse the response from a GetRecords request. │ │ │ │ │ */ │ │ │ │ │ read: function(data) { │ │ │ │ │ - var result; │ │ │ │ │ if (typeof data == "string") { │ │ │ │ │ data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ } │ │ │ │ │ - var root = data.documentElement; │ │ │ │ │ - if (root) { │ │ │ │ │ - var scope = this; │ │ │ │ │ - var read = this["read_" + root.nodeName]; │ │ │ │ │ - if (read) { │ │ │ │ │ - result = read.call(this, root); │ │ │ │ │ - } else { │ │ │ │ │ - // fall-back to GML since this is a common output format for WMS │ │ │ │ │ - // GetFeatureInfo responses │ │ │ │ │ - result = new OpenLayers.Format.GML((this.options ? this.options : {})).read(data); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - result = data; │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ } │ │ │ │ │ - return result; │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readNode(data, obj); │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: read_msGMLOutput │ │ │ │ │ - * Parse msGMLOutput nodes. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ */ │ │ │ │ │ - read_msGMLOutput: function(data) { │ │ │ │ │ - var response = []; │ │ │ │ │ - var layerNodes = this.getSiblingNodesByTagCriteria(data, │ │ │ │ │ - this.layerIdentifier); │ │ │ │ │ - if (layerNodes) { │ │ │ │ │ - for (var i = 0, len = layerNodes.length; i < len; ++i) { │ │ │ │ │ - var node = layerNodes[i]; │ │ │ │ │ - var layerName = node.nodeName; │ │ │ │ │ - if (node.prefix) { │ │ │ │ │ - layerName = layerName.split(':')[1]; │ │ │ │ │ + readers: { │ │ │ │ │ + "csw": { │ │ │ │ │ + "GetRecordsResponse": function(node, obj) { │ │ │ │ │ + obj.records = []; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + var version = this.getAttributeNS(node, "", 'version'); │ │ │ │ │ + if (version != "") { │ │ │ │ │ + obj.version = version; │ │ │ │ │ } │ │ │ │ │ - var layerName = layerName.replace(this.layerIdentifier, ''); │ │ │ │ │ - var featureNodes = this.getSiblingNodesByTagCriteria(node, │ │ │ │ │ - this.featureIdentifier); │ │ │ │ │ - if (featureNodes) { │ │ │ │ │ - for (var j = 0; j < featureNodes.length; j++) { │ │ │ │ │ - var featureNode = featureNodes[j]; │ │ │ │ │ - var geomInfo = this.parseGeometry(featureNode); │ │ │ │ │ - var attributes = this.parseAttributes(featureNode); │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geomInfo.geometry, │ │ │ │ │ - attributes, null); │ │ │ │ │ - feature.bounds = geomInfo.bounds; │ │ │ │ │ - feature.type = layerName; │ │ │ │ │ - response.push(feature); │ │ │ │ │ + }, │ │ │ │ │ + "RequestId": function(node, obj) { │ │ │ │ │ + obj.RequestId = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "SearchStatus": function(node, obj) { │ │ │ │ │ + obj.SearchStatus = {}; │ │ │ │ │ + var timestamp = this.getAttributeNS(node, "", 'timestamp'); │ │ │ │ │ + if (timestamp != "") { │ │ │ │ │ + obj.SearchStatus.timestamp = timestamp; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "SearchResults": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + var attrs = node.attributes; │ │ │ │ │ + var SearchResults = {}; │ │ │ │ │ + for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ + if ((attrs[i].name == "numberOfRecordsMatched") || │ │ │ │ │ + (attrs[i].name == "numberOfRecordsReturned") || │ │ │ │ │ + (attrs[i].name == "nextRecord")) { │ │ │ │ │ + SearchResults[attrs[i].name] = parseInt(attrs[i].nodeValue); │ │ │ │ │ + } else { │ │ │ │ │ + SearchResults[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + obj.SearchResults = SearchResults; │ │ │ │ │ + }, │ │ │ │ │ + "SummaryRecord": function(node, obj) { │ │ │ │ │ + var record = { │ │ │ │ │ + type: "SummaryRecord" │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, record); │ │ │ │ │ + obj.records.push(record); │ │ │ │ │ + }, │ │ │ │ │ + "BriefRecord": function(node, obj) { │ │ │ │ │ + var record = { │ │ │ │ │ + type: "BriefRecord" │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, record); │ │ │ │ │ + obj.records.push(record); │ │ │ │ │ + }, │ │ │ │ │ + "DCMIRecord": function(node, obj) { │ │ │ │ │ + var record = { │ │ │ │ │ + type: "DCMIRecord" │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, record); │ │ │ │ │ + obj.records.push(record); │ │ │ │ │ + }, │ │ │ │ │ + "Record": function(node, obj) { │ │ │ │ │ + var record = { │ │ │ │ │ + type: "Record" │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, record); │ │ │ │ │ + obj.records.push(record); │ │ │ │ │ + }, │ │ │ │ │ + "*": function(node, obj) { │ │ │ │ │ + var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ + obj[name] = this.getChildValue(node); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "geonet": { │ │ │ │ │ + "info": function(node, obj) { │ │ │ │ │ + var gninfo = {}; │ │ │ │ │ + this.readChildNodes(node, gninfo); │ │ │ │ │ + obj.gninfo = gninfo; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "dc": { │ │ │ │ │ + // audience, contributor, coverage, creator, date, description, format, │ │ │ │ │ + // identifier, language, provenance, publisher, relation, rights, │ │ │ │ │ + // rightsHolder, source, subject, title, type, URI │ │ │ │ │ + "*": function(node, obj) { │ │ │ │ │ + var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ + if (!(OpenLayers.Util.isArray(obj[name]))) { │ │ │ │ │ + obj[name] = []; │ │ │ │ │ + } │ │ │ │ │ + var dc_element = {}; │ │ │ │ │ + var attrs = node.attributes; │ │ │ │ │ + for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ + dc_element[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ + } │ │ │ │ │ + dc_element.value = this.getChildValue(node); │ │ │ │ │ + if (dc_element.value != "") { │ │ │ │ │ + obj[name].push(dc_element); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return response; │ │ │ │ │ + }, │ │ │ │ │ + "dct": { │ │ │ │ │ + // abstract, modified, spatial │ │ │ │ │ + "*": function(node, obj) { │ │ │ │ │ + var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ + if (!(OpenLayers.Util.isArray(obj[name]))) { │ │ │ │ │ + obj[name] = []; │ │ │ │ │ + } │ │ │ │ │ + obj[name].push(this.getChildValue(node)); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "ows": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "BoundingBox": function(node, obj) { │ │ │ │ │ + if (obj.bounds) { │ │ │ │ │ + obj.BoundingBox = [{ │ │ │ │ │ + crs: obj.projection, │ │ │ │ │ + value: [ │ │ │ │ │ + obj.bounds.left, │ │ │ │ │ + obj.bounds.bottom, │ │ │ │ │ + obj.bounds.right, │ │ │ │ │ + obj.bounds.top │ │ │ │ │ + ] │ │ │ │ │ + }]; │ │ │ │ │ + delete obj.projection; │ │ │ │ │ + delete obj.bounds; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Format.OWSCommon.v1_0_0.prototype.readers["ows"]["BoundingBox"].apply( │ │ │ │ │ + this, arguments); │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.OWSCommon.v1_0_0.prototype.readers["ows"]) │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_FeatureInfoResponse │ │ │ │ │ - * Parse FeatureInfoResponse nodes. │ │ │ │ │ + * Method: write │ │ │ │ │ + * Given an configuration js object, write a CSWGetRecords request. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {DOMElement} │ │ │ │ │ + * options - {Object} A object mapping the request. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array} │ │ │ │ │ + * {String} A serialized CSWGetRecords request. │ │ │ │ │ */ │ │ │ │ │ - read_FeatureInfoResponse: function(data) { │ │ │ │ │ - var response = []; │ │ │ │ │ - var featureNodes = this.getElementsByTagNameNS(data, '*', │ │ │ │ │ - 'FIELDS'); │ │ │ │ │ - │ │ │ │ │ - for (var i = 0, len = featureNodes.length; i < len; i++) { │ │ │ │ │ - var featureNode = featureNodes[i]; │ │ │ │ │ - var geom = null; │ │ │ │ │ + write: function(options) { │ │ │ │ │ + var node = this.writeNode("csw:GetRecords", options); │ │ │ │ │ + node.setAttribute("xmlns:gmd", this.namespaces.gmd); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [node]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // attributes can be actual attributes on the FIELDS tag, │ │ │ │ │ - // or FIELD children │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - var j; │ │ │ │ │ - var jlen = featureNode.attributes.length; │ │ │ │ │ - if (jlen > 0) { │ │ │ │ │ - for (j = 0; j < jlen; j++) { │ │ │ │ │ - var attribute = featureNode.attributes[j]; │ │ │ │ │ - attributes[attribute.nodeName] = attribute.nodeValue; │ │ │ │ │ + /** │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ + */ │ │ │ │ │ + writers: { │ │ │ │ │ + "csw": { │ │ │ │ │ + "GetRecords": function(options) { │ │ │ │ │ + if (!options) { │ │ │ │ │ + options = {}; │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - var nodes = featureNode.childNodes; │ │ │ │ │ - for (j = 0, jlen = nodes.length; j < jlen; ++j) { │ │ │ │ │ - var node = nodes[j]; │ │ │ │ │ - if (node.nodeType != 3) { │ │ │ │ │ - attributes[node.getAttribute("name")] = │ │ │ │ │ - node.getAttribute("value"); │ │ │ │ │ + var node = this.createElementNSPlus("csw:GetRecords", { │ │ │ │ │ + attributes: { │ │ │ │ │ + service: "CSW", │ │ │ │ │ + version: this.version, │ │ │ │ │ + requestId: options.requestId || this.requestId, │ │ │ │ │ + resultType: options.resultType || this.resultType, │ │ │ │ │ + outputFormat: options.outputFormat || this.outputFormat, │ │ │ │ │ + outputSchema: options.outputSchema || this.outputSchema, │ │ │ │ │ + startPosition: options.startPosition || this.startPosition, │ │ │ │ │ + maxRecords: options.maxRecords || this.maxRecords │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (options.DistributedSearch || this.DistributedSearch) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "csw:DistributedSearch", │ │ │ │ │ + options.DistributedSearch || this.DistributedSearch, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + var ResponseHandler = options.ResponseHandler || this.ResponseHandler; │ │ │ │ │ + if (OpenLayers.Util.isArray(ResponseHandler) && ResponseHandler.length > 0) { │ │ │ │ │ + // ResponseHandler must be a non-empty array │ │ │ │ │ + for (var i = 0, len = ResponseHandler.length; i < len; i++) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "csw:ResponseHandler", │ │ │ │ │ + ResponseHandler[i], │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.writeNode("Query", options.Query || this.Query, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "DistributedSearch": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:DistributedSearch", { │ │ │ │ │ + attributes: { │ │ │ │ │ + hopCount: options.hopCount │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "ResponseHandler": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:ResponseHandler", { │ │ │ │ │ + value: options.value │ │ │ │ │ + }); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Query": function(options) { │ │ │ │ │ + if (!options) { │ │ │ │ │ + options = {}; │ │ │ │ │ + } │ │ │ │ │ + var node = this.createElementNSPlus("csw:Query", { │ │ │ │ │ + attributes: { │ │ │ │ │ + typeNames: options.typeNames || "csw:Record" │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + var ElementName = options.ElementName; │ │ │ │ │ + if (OpenLayers.Util.isArray(ElementName) && ElementName.length > 0) { │ │ │ │ │ + // ElementName must be a non-empty array │ │ │ │ │ + for (var i = 0, len = ElementName.length; i < len; i++) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "csw:ElementName", │ │ │ │ │ + ElementName[i], │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "csw:ElementSetName", │ │ │ │ │ + options.ElementSetName || { │ │ │ │ │ + value: 'summary' │ │ │ │ │ + }, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (options.Constraint) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "csw:Constraint", │ │ │ │ │ + options.Constraint, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (options.SortBy) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "ogc:SortBy", │ │ │ │ │ + options.SortBy, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "ElementName": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:ElementName", { │ │ │ │ │ + value: options.value │ │ │ │ │ + }); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "ElementSetName": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:ElementSetName", { │ │ │ │ │ + attributes: { │ │ │ │ │ + typeNames: options.typeNames │ │ │ │ │ + }, │ │ │ │ │ + value: options.value │ │ │ │ │ + }); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Constraint": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:Constraint", { │ │ │ │ │ + attributes: { │ │ │ │ │ + version: options.version │ │ │ │ │ } │ │ │ │ │ + }); │ │ │ │ │ + if (options.Filter) { │ │ │ │ │ + var format = new OpenLayers.Format.Filter({ │ │ │ │ │ + version: options.version │ │ │ │ │ + }); │ │ │ │ │ + node.appendChild(format.write(options.Filter)); │ │ │ │ │ + } else if (options.CqlText) { │ │ │ │ │ + var child = this.createElementNSPlus("CqlText", { │ │ │ │ │ + value: options.CqlText.value │ │ │ │ │ + }); │ │ │ │ │ + node.appendChild(child); │ │ │ │ │ } │ │ │ │ │ + return node; │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ + "ogc": OpenLayers.Format.Filter.v1_1_0.prototype.writers["ogc"] │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - response.push( │ │ │ │ │ - new OpenLayers.Feature.Vector(geom, attributes, null) │ │ │ │ │ - ); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.CSWGetRecords.v2_0_2" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Protocol/CSW/v2_0_2.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Protocol/CSW.js │ │ │ │ │ + * @requires OpenLayers/Format/CSWGetRecords/v2_0_2.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.CSW.v2_0_2 │ │ │ │ │ + * CS-W (Catalogue services for the Web) version 2.0.2 protocol. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Protocol> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.CSW.v2_0_2 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: formatOptions │ │ │ │ │ + * {Object} Optional options for the format. If a format is not provided, │ │ │ │ │ + * this property can be used to extend the default format options. │ │ │ │ │ + */ │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Protocol.CSW.v2_0_2 │ │ │ │ │ + * A class for CSW version 2.0.2 protocol management. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!options.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.CSWGetRecords.v2_0_2(OpenLayers.Util.extend({}, this.formatOptions)); │ │ │ │ │ } │ │ │ │ │ - return response; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getSiblingNodesByTagCriteria │ │ │ │ │ - * Recursively searches passed xml node and all it's descendant levels for │ │ │ │ │ - * nodes whose tagName contains the passed search string. This returns an │ │ │ │ │ - * array of all sibling nodes which match the criteria from the highest │ │ │ │ │ - * hierarchial level from which a match is found. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} An xml node │ │ │ │ │ - * criteria - {String} Search string which will match some part of a tagName │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * Array({DOMElement}) An array of sibling xml nodes │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up the protocol. │ │ │ │ │ */ │ │ │ │ │ - getSiblingNodesByTagCriteria: function(node, criteria) { │ │ │ │ │ - var nodes = []; │ │ │ │ │ - var children, tagName, n, matchNodes, child; │ │ │ │ │ - if (node && node.hasChildNodes()) { │ │ │ │ │ - children = node.childNodes; │ │ │ │ │ - n = children.length; │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.options && !this.options.format) { │ │ │ │ │ + this.format.destroy(); │ │ │ │ │ + } │ │ │ │ │ + this.format = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - for (var k = 0; k < n; k++) { │ │ │ │ │ - child = children[k]; │ │ │ │ │ - while (child && child.nodeType != 1) { │ │ │ │ │ - child = child.nextSibling; │ │ │ │ │ - k++; │ │ │ │ │ - } │ │ │ │ │ - tagName = (child ? child.nodeName : ''); │ │ │ │ │ - if (tagName.length > 0 && tagName.indexOf(criteria) > -1) { │ │ │ │ │ - nodes.push(child); │ │ │ │ │ - } else { │ │ │ │ │ - matchNodes = this.getSiblingNodesByTagCriteria( │ │ │ │ │ - child, criteria); │ │ │ │ │ + /** │ │ │ │ │ + * Method: read │ │ │ │ │ + * Construct a request for reading new records from the Catalogue. │ │ │ │ │ + */ │ │ │ │ │ + read: function(options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - if (matchNodes.length > 0) { │ │ │ │ │ - (nodes.length == 0) ? │ │ │ │ │ - nodes = matchNodes: nodes.push(matchNodes); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + var data = this.format.write(options.params || options); │ │ │ │ │ │ │ │ │ │ - } │ │ │ │ │ - return nodes; │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: data │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + return response; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseAttributes │ │ │ │ │ + * Method: handleRead │ │ │ │ │ + * Deal with response from the read request. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {<DOMElement>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An attributes object. │ │ │ │ │ - * │ │ │ │ │ - * Notes: │ │ │ │ │ - * Assumes that attributes are direct child xml nodes of the passed node │ │ │ │ │ - * and contain only a single text node. │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ + * to the user callback. │ │ │ │ │ + * This response is given a code property, and optionally a data property. │ │ │ │ │ + * The latter represents the CSW records as returned by the call to │ │ │ │ │ + * the CSW format read method. │ │ │ │ │ + * options - {Object} The user options passed to the read call. │ │ │ │ │ */ │ │ │ │ │ - parseAttributes: function(node) { │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - if (node.nodeType == 1) { │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - var n = children.length; │ │ │ │ │ - for (var i = 0; i < n; ++i) { │ │ │ │ │ - var child = children[i]; │ │ │ │ │ - if (child.nodeType == 1) { │ │ │ │ │ - var grandchildren = child.childNodes; │ │ │ │ │ - var name = (child.prefix) ? │ │ │ │ │ - child.nodeName.split(":")[1] : child.nodeName; │ │ │ │ │ - if (grandchildren.length == 0) { │ │ │ │ │ - attributes[name] = null; │ │ │ │ │ - } else if (grandchildren.length == 1) { │ │ │ │ │ - var grandchild = grandchildren[0]; │ │ │ │ │ - if (grandchild.nodeType == 3 || │ │ │ │ │ - grandchild.nodeType == 4) { │ │ │ │ │ - var value = grandchild.nodeValue.replace( │ │ │ │ │ - this.regExes.trimSpace, ""); │ │ │ │ │ - attributes[name] = value; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + // success │ │ │ │ │ + response.data = this.parseData(request); │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + } else { │ │ │ │ │ + // failure │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ } │ │ │ │ │ + options.callback.call(options.scope, response); │ │ │ │ │ } │ │ │ │ │ - return attributes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseGeometry │ │ │ │ │ - * Parse the geometry and the feature bounds out of the node using │ │ │ │ │ - * Format.GML │ │ │ │ │ + * Method: parseData │ │ │ │ │ + * Read HTTP response body and return records │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {<DOMElement>} │ │ │ │ │ + * request - {XMLHttpRequest} The request object │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} An object containing the geometry and the feature bounds │ │ │ │ │ + * {Object} The CSW records as returned by the call to the format read method. │ │ │ │ │ */ │ │ │ │ │ - parseGeometry: function(node) { │ │ │ │ │ - // we need to use the old Format.GML parser since we do not know the │ │ │ │ │ - // geometry name │ │ │ │ │ - if (!this.gmlFormat) { │ │ │ │ │ - this.gmlFormat = new OpenLayers.Format.GML(); │ │ │ │ │ + parseData: function(request) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ } │ │ │ │ │ - var feature = this.gmlFormat.parseFeature(node); │ │ │ │ │ - var geometry, bounds = null; │ │ │ │ │ - if (feature) { │ │ │ │ │ - geometry = feature.geometry && feature.geometry.clone(); │ │ │ │ │ - bounds = feature.bounds && feature.bounds.clone(); │ │ │ │ │ - feature.destroy(); │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null; │ │ │ │ │ } │ │ │ │ │ - return { │ │ │ │ │ - geometry: geometry, │ │ │ │ │ - bounds: bounds │ │ │ │ │ - }; │ │ │ │ │ + return this.format.read(doc); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSGetFeatureInfo" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.CSW.v2_0_2" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/CSWGetDomain.js │ │ │ │ │ + OpenLayers/Format/WCSCapabilities.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format.js │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.CSWGetDomain │ │ │ │ │ - * Default version is 2.0.2. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Format>} A CSWGetDomain format of the given version. │ │ │ │ │ + * Class: OpenLayers.Format.WCSCapabilities │ │ │ │ │ + * Read WCS Capabilities. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.CSWGetDomain = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options, OpenLayers.Format.CSWGetDomain.DEFAULTS │ │ │ │ │ - ); │ │ │ │ │ - var cls = OpenLayers.Format.CSWGetDomain["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported CSWGetDomain version: " + options.version; │ │ │ │ │ - } │ │ │ │ │ - return new cls(options); │ │ │ │ │ -}; │ │ │ │ │ +OpenLayers.Format.WCSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: DEFAULTS │ │ │ │ │ - * {Object} Default properties for the CSWGetDomain format. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.CSWGetDomain.DEFAULTS = { │ │ │ │ │ - "version": "2.0.2" │ │ │ │ │ -}; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.1.0". │ │ │ │ │ + */ │ │ │ │ │ + defaultVersion: "1.1.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WCSCapabilities │ │ │ │ │ + * Create a new parser for WCS capabilities. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return a list of coverages. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} List of named coverages. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WCSCapabilities" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/QueryStringFilter.js │ │ │ │ │ + OpenLayers/Format/WMSCapabilities.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Console.js │ │ │ │ │ - * @requires OpenLayers/Format.js │ │ │ │ │ - * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ - * @requires OpenLayers/Filter/Comparison.js │ │ │ │ │ - * @requires OpenLayers/Filter/Logical.js │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.QueryStringFilter │ │ │ │ │ - * Parser for reading a query string and creating a simple filter. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Format.WMSCapabilities │ │ │ │ │ + * Read WMS Capabilities. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format> │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.QueryStringFilter = (function() { │ │ │ │ │ +OpenLayers.Format.WMSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Map the OpenLayers.Filter.Comparison types to the operation strings of │ │ │ │ │ - * the protocol. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.1.1". │ │ │ │ │ */ │ │ │ │ │ - var cmpToStr = {}; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.EQUAL_TO] = "eq"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.NOT_EQUAL_TO] = "ne"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.LESS_THAN] = "lt"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO] = "lte"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.GREATER_THAN] = "gt"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO] = "gte"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.LIKE] = "ilike"; │ │ │ │ │ + defaultVersion: "1.1.1", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Function: regex2value │ │ │ │ │ - * Convert the value from a regular expression string to a LIKE/ILIKE │ │ │ │ │ - * string known to the web service. │ │ │ │ │ + * APIProperty: profile │ │ │ │ │ + * {String} If provided, use a custom profile. │ │ │ │ │ + * │ │ │ │ │ + * Currently supported profiles: │ │ │ │ │ + * - WMSC - parses vendor specific capabilities for WMS-C. │ │ │ │ │ + */ │ │ │ │ │ + profile: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMSCapabilities │ │ │ │ │ + * Create a new parser for WMS capabilities. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * value - {String} The regex string. │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} The converted string. │ │ │ │ │ + * {Array} List of named layers. │ │ │ │ │ */ │ │ │ │ │ - function regex2value(value) { │ │ │ │ │ - │ │ │ │ │ - // highly sensitive!! Do not change this without running the │ │ │ │ │ - // Protocol/HTTP.html unit tests │ │ │ │ │ - │ │ │ │ │ - // convert % to \% │ │ │ │ │ - value = value.replace(/%/g, "\\%"); │ │ │ │ │ - │ │ │ │ │ - // convert \\. to \\_ (\\.* occurences converted later) │ │ │ │ │ - value = value.replace(/\\\\\.(\*)?/g, function($0, $1) { │ │ │ │ │ - return $1 ? $0 : "\\\\_"; │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - // convert \\.* to \\% │ │ │ │ │ - value = value.replace(/\\\\\.\*/g, "\\\\%"); │ │ │ │ │ - │ │ │ │ │ - // convert . to _ (\. and .* occurences converted later) │ │ │ │ │ - value = value.replace(/(\\)?\.(\*)?/g, function($0, $1, $2) { │ │ │ │ │ - return $1 || $2 ? $0 : "_"; │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - // convert .* to % (\.* occurnces converted later) │ │ │ │ │ - value = value.replace(/(\\)?\.\*/g, function($0, $1) { │ │ │ │ │ - return $1 ? $0 : "%"; │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - // convert \. to . │ │ │ │ │ - value = value.replace(/\\\./g, "."); │ │ │ │ │ - │ │ │ │ │ - // replace \* with * (watching out for \\*) │ │ │ │ │ - value = value.replace(/(\\)?\\\*/g, function($0, $1) { │ │ │ │ │ - return $1 ? $0 : "*"; │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - return value; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: wildcarded. │ │ │ │ │ - * {Boolean} If true percent signs are added around values │ │ │ │ │ - * read from LIKE filters, for example if the protocol │ │ │ │ │ - * read method is passed a LIKE filter whose property │ │ │ │ │ - * is "foo" and whose value is "bar" the string │ │ │ │ │ - * "foo__ilike=%bar%" will be sent in the query string; │ │ │ │ │ - * defaults to false. │ │ │ │ │ - */ │ │ │ │ │ - wildcarded: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: srsInBBOX │ │ │ │ │ - * {Boolean} Include the SRS identifier in BBOX query string parameter. │ │ │ │ │ - * Default is false. If true and the layer has a projection object set, │ │ │ │ │ - * any BBOX filter will be serialized with a fifth item identifying the │ │ │ │ │ - * projection. E.g. bbox=-1000,-1000,1000,1000,EPSG:900913 │ │ │ │ │ - */ │ │ │ │ │ - srsInBBOX: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Serialize an <OpenLayers.Filter> objects using the "simple" filter syntax for │ │ │ │ │ - * query string parameters. This function must be called as a method of │ │ │ │ │ - * a protocol instance. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} filter to convert. │ │ │ │ │ - * params - {Object} The parameters object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The resulting parameters object. │ │ │ │ │ - */ │ │ │ │ │ - write: function(filter, params) { │ │ │ │ │ - params = params || {}; │ │ │ │ │ - var className = filter.CLASS_NAME; │ │ │ │ │ - var filterType = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ - switch (filterType) { │ │ │ │ │ - case "Spatial": │ │ │ │ │ - switch (filter.type) { │ │ │ │ │ - case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ - params.bbox = filter.value.toArray(); │ │ │ │ │ - if (this.srsInBBOX && filter.projection) { │ │ │ │ │ - params.bbox.push(filter.projection.getCode()); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Spatial.DWITHIN: │ │ │ │ │ - params.tolerance = filter.distance; │ │ │ │ │ - // no break here │ │ │ │ │ - case OpenLayers.Filter.Spatial.WITHIN: │ │ │ │ │ - params.lon = filter.value.x; │ │ │ │ │ - params.lat = filter.value.y; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - OpenLayers.Console.warn( │ │ │ │ │ - "Unknown spatial filter type " + filter.type); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "Comparison": │ │ │ │ │ - var op = cmpToStr[filter.type]; │ │ │ │ │ - if (op !== undefined) { │ │ │ │ │ - var value = filter.value; │ │ │ │ │ - if (filter.type == OpenLayers.Filter.Comparison.LIKE) { │ │ │ │ │ - value = regex2value(value); │ │ │ │ │ - if (this.wildcarded) { │ │ │ │ │ - value = "%" + value + "%"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - params[filter.property + "__" + op] = value; │ │ │ │ │ - params.queryable = params.queryable || []; │ │ │ │ │ - params.queryable.push(filter.property); │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Console.warn( │ │ │ │ │ - "Unknown comparison filter type " + filter.type); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "Logical": │ │ │ │ │ - if (filter.type === OpenLayers.Filter.Logical.AND) { │ │ │ │ │ - for (var i = 0, len = filter.filters.length; i < len; i++) { │ │ │ │ │ - params = this.write(filter.filters[i], params); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Console.warn( │ │ │ │ │ - "Unsupported logical filter type " + filter.type); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - OpenLayers.Console.warn("Unknown filter type " + filterType); │ │ │ │ │ - } │ │ │ │ │ - return params; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.QueryStringFilter" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities" │ │ │ │ │ │ │ │ │ │ -})(); │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/KML.js │ │ │ │ │ + OpenLayers/Format/ArcXML.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Date.js │ │ │ │ │ * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Collection.js │ │ │ │ │ - * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ - * @requires OpenLayers/Projection.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Geometry/MultiPolygon.js │ │ │ │ │ + * @requires OpenLayers/Geometry/LinearRing.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.KML │ │ │ │ │ - * Read/Write KML. Create a new instance with the <OpenLayers.Format.KML> │ │ │ │ │ - * constructor. │ │ │ │ │ + * Class: OpenLayers.Format.ArcXML │ │ │ │ │ + * Read/Write ArcXML. Create a new instance with the <OpenLayers.Format.ArcXML> │ │ │ │ │ + * constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Format.XML> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.KML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Format.ArcXML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + * Property: fontStyleKeys │ │ │ │ │ + * {Array} List of keys used in font styling. │ │ │ │ │ */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - kml: "http://www.opengis.net/kml/2.2", │ │ │ │ │ - gx: "http://www.google.com/kml/ext/2.2" │ │ │ │ │ - }, │ │ │ │ │ + fontStyleKeys: [ │ │ │ │ │ + 'antialiasing', 'blockout', 'font', 'fontcolor', 'fontsize', 'fontstyle', │ │ │ │ │ + 'glowing', 'interval', 'outline', 'printmode', 'shadow', 'transparency' │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: kmlns │ │ │ │ │ - * {String} KML Namespace to use. Defaults to 2.0 namespace. │ │ │ │ │ - */ │ │ │ │ │ - kmlns: "http://earth.google.com/kml/2.0", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: placemarksDesc │ │ │ │ │ - * {String} Name of the placemarks. Default is "No description available". │ │ │ │ │ - */ │ │ │ │ │ - placemarksDesc: "No description available", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: foldersName │ │ │ │ │ - * {String} Name of the folders. Default is "OpenLayers export". │ │ │ │ │ - * If set to null, no name element will be created. │ │ │ │ │ - */ │ │ │ │ │ - foldersName: "OpenLayers export", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: foldersDesc │ │ │ │ │ - * {String} Description of the folders. Default is "Exported on [date]." │ │ │ │ │ - * If set to null, no description element will be created. │ │ │ │ │ + * Property: request │ │ │ │ │ + * A get_image request destined for an ArcIMS server. │ │ │ │ │ */ │ │ │ │ │ - foldersDesc: "Exported on " + new Date(), │ │ │ │ │ + request: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: extractAttributes │ │ │ │ │ - * {Boolean} Extract attributes from KML. Default is true. │ │ │ │ │ - * Extracting styleUrls requires this to be set to true │ │ │ │ │ - * Note that currently only Data and SimpleData │ │ │ │ │ - * elements are handled. │ │ │ │ │ + * Property: response │ │ │ │ │ + * A parsed response from an ArcIMS server. │ │ │ │ │ */ │ │ │ │ │ - extractAttributes: true, │ │ │ │ │ + response: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: kvpAttributes │ │ │ │ │ - * {Boolean} Only used if extractAttributes is true. │ │ │ │ │ - * If set to true, attributes will be simple │ │ │ │ │ - * key-value pairs, compatible with other formats, │ │ │ │ │ - * Any displayName elements will be ignored. │ │ │ │ │ - * If set to false, attributes will be objects, │ │ │ │ │ - * retaining any displayName elements, but not │ │ │ │ │ - * compatible with other formats. Any CDATA in │ │ │ │ │ - * displayName will be read in as a string value. │ │ │ │ │ - * Default is false. │ │ │ │ │ + * Constructor: OpenLayers.Format.ArcXML │ │ │ │ │ + * Create a new parser/writer for ArcXML. Create an instance of this class │ │ │ │ │ + * to begin authoring a request to an ArcIMS service. This is used │ │ │ │ │ + * primarily by the ArcIMS layer, but could be used to do other wild │ │ │ │ │ + * stuff, like geocoding. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - kvpAttributes: false, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.request = new OpenLayers.Format.ArcXML.Request(); │ │ │ │ │ + this.response = new OpenLayers.Format.ArcXML.Response(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: extractStyles │ │ │ │ │ - * {Boolean} Extract styles from KML. Default is false. │ │ │ │ │ - * Extracting styleUrls also requires extractAttributes to be │ │ │ │ │ - * set to true │ │ │ │ │ - */ │ │ │ │ │ - extractStyles: false, │ │ │ │ │ + if (options) { │ │ │ │ │ + if (options.requesttype == "feature") { │ │ │ │ │ + this.request.get_image = null; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: extractTracks │ │ │ │ │ - * {Boolean} Extract gx:Track elements from Placemark elements. Default │ │ │ │ │ - * is false. If true, features will be generated for all points in │ │ │ │ │ - * all gx:Track elements. Features will have a when (Date) attribute │ │ │ │ │ - * based on when elements in the track. If tracks include angle │ │ │ │ │ - * elements, features will have heading, tilt, and roll attributes. │ │ │ │ │ - * If track point coordinates have three values, features will have │ │ │ │ │ - * an altitude attribute with the third coordinate value. │ │ │ │ │ - */ │ │ │ │ │ - extractTracks: false, │ │ │ │ │ + var qry = this.request.get_feature.query; │ │ │ │ │ + this.addCoordSys(qry.featurecoordsys, options.featureCoordSys); │ │ │ │ │ + this.addCoordSys(qry.filtercoordsys, options.filterCoordSys); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: trackAttributes │ │ │ │ │ - * {Array} If <extractTracks> is true, points within gx:Track elements will │ │ │ │ │ - * be parsed as features with when, heading, tilt, and roll attributes. │ │ │ │ │ - * Any additional attribute names can be provided in <trackAttributes>. │ │ │ │ │ - */ │ │ │ │ │ - trackAttributes: null, │ │ │ │ │ + if (options.polygon) { │ │ │ │ │ + qry.isspatial = true; │ │ │ │ │ + qry.spatialfilter.polygon = options.polygon; │ │ │ │ │ + } else if (options.envelope) { │ │ │ │ │ + qry.isspatial = true; │ │ │ │ │ + qry.spatialfilter.envelope = { │ │ │ │ │ + minx: 0, │ │ │ │ │ + miny: 0, │ │ │ │ │ + maxx: 0, │ │ │ │ │ + maxy: 0 │ │ │ │ │ + }; │ │ │ │ │ + this.parseEnvelope(qry.spatialfilter.envelope, options.envelope); │ │ │ │ │ + } │ │ │ │ │ + } else if (options.requesttype == "image") { │ │ │ │ │ + this.request.get_feature = null; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: internalns │ │ │ │ │ - * {String} KML Namespace to use -- defaults to the namespace of the │ │ │ │ │ - * Placemark node being parsed, but falls back to kmlns. │ │ │ │ │ - */ │ │ │ │ │ - internalns: null, │ │ │ │ │ + var props = this.request.get_image.properties; │ │ │ │ │ + this.parseEnvelope(props.envelope, options.envelope); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Array} Array of features │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - features: null, │ │ │ │ │ + this.addLayers(props.layerlist, options.layers); │ │ │ │ │ + this.addImageSize(props.imagesize, options.tileSize); │ │ │ │ │ + this.addCoordSys(props.featurecoordsys, options.featureCoordSys); │ │ │ │ │ + this.addCoordSys(props.filtercoordsys, options.filterCoordSys); │ │ │ │ │ + } else { │ │ │ │ │ + // if an arcxml object is being created with no request type, it is │ │ │ │ │ + // probably going to consume a response, so do not throw an error if │ │ │ │ │ + // the requesttype is not defined │ │ │ │ │ + this.request = null; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: styles │ │ │ │ │ - * {Object} Storage of style objects │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - styles: null, │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: styleBaseUrl │ │ │ │ │ - * {String} │ │ │ │ │ + * Method: parseEnvelope │ │ │ │ │ + * Parse an array of coordinates into an ArcXML envelope structure. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * env - {Object} An envelope object that will contain the parsed coordinates. │ │ │ │ │ + * arr - {Array(double)} An array of coordinates in the order: [ minx, miny, maxx, maxy ] │ │ │ │ │ */ │ │ │ │ │ - styleBaseUrl: "", │ │ │ │ │ + parseEnvelope: function(env, arr) { │ │ │ │ │ + if (arr && arr.length == 4) { │ │ │ │ │ + env.minx = arr[0]; │ │ │ │ │ + env.miny = arr[1]; │ │ │ │ │ + env.maxx = arr[2]; │ │ │ │ │ + env.maxy = arr[3]; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: fetched │ │ │ │ │ - * {Object} Storage of KML URLs that have been fetched before │ │ │ │ │ - * in order to prevent reloading them. │ │ │ │ │ + /** │ │ │ │ │ + * Method: addLayers │ │ │ │ │ + * Add a collection of layers to another collection of layers. Each layer in the list is tuple of │ │ │ │ │ + * { id, visible }. These layer collections represent the │ │ │ │ │ + * /ARCXML/REQUEST/get_image/PROPERTIES/LAYERLIST/LAYERDEF items in ArcXML │ │ │ │ │ + * │ │ │ │ │ + * TODO: Add support for dynamic layer rendering. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * ll - {Array({id,visible})} A list of layer definitions. │ │ │ │ │ + * lyrs - {Array({id,visible})} A list of layer definitions. │ │ │ │ │ */ │ │ │ │ │ - fetched: null, │ │ │ │ │ + addLayers: function(ll, lyrs) { │ │ │ │ │ + for (var lind = 0, len = lyrs.length; lind < len; lind++) { │ │ │ │ │ + ll.push(lyrs[lind]); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: maxDepth │ │ │ │ │ - * {Integer} Maximum depth for recursive loading external KML URLs │ │ │ │ │ - * Defaults to 0: do no external fetching │ │ │ │ │ + * Method: addImageSize │ │ │ │ │ + * Set the size of the requested image. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * imsize - {Object} An ArcXML imagesize object. │ │ │ │ │ + * olsize - {<OpenLayers.Size>} The image size to set. │ │ │ │ │ */ │ │ │ │ │ - maxDepth: 0, │ │ │ │ │ + addImageSize: function(imsize, olsize) { │ │ │ │ │ + if (olsize !== null) { │ │ │ │ │ + imsize.width = olsize.w; │ │ │ │ │ + imsize.height = olsize.h; │ │ │ │ │ + imsize.printwidth = olsize.w; │ │ │ │ │ + imsize.printheight = olsize.h; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.KML │ │ │ │ │ - * Create a new parser for KML. │ │ │ │ │ + * Method: addCoordSys │ │ │ │ │ + * Add the coordinate system information to an object. The object may be │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * featOrFilt - {Object} A featurecoordsys or filtercoordsys ArcXML structure. │ │ │ │ │ + * fsys - {String} or {<OpenLayers.Projection>} or {filtercoordsys} or │ │ │ │ │ + * {featurecoordsys} A projection representation. If it's a {String}, │ │ │ │ │ + * the value is assumed to be the SRID. If it's a {OpenLayers.Projection} │ │ │ │ │ + * AND Proj4js is available, the projection number and name are extracted │ │ │ │ │ + * from there. If it's a filter or feature ArcXML structure, it is copied. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - // compile regular expressions once instead of every time they are used │ │ │ │ │ - this.regExes = { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g), │ │ │ │ │ - kmlColor: (/(\w{2})(\w{2})(\w{2})(\w{2})/), │ │ │ │ │ - kmlIconPalette: (/root:\/\/icons\/palette-(\d+)(\.\w+)/), │ │ │ │ │ - straightBracket: (/\$\[(.*?)\]/g) │ │ │ │ │ - }; │ │ │ │ │ - // KML coordinates are always in longlat WGS84 │ │ │ │ │ - this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + addCoordSys: function(featOrFilt, fsys) { │ │ │ │ │ + if (typeof fsys == "string") { │ │ │ │ │ + featOrFilt.id = parseInt(fsys); │ │ │ │ │ + featOrFilt.string = fsys; │ │ │ │ │ + } │ │ │ │ │ + // is this a proj4js instance? │ │ │ │ │ + else if (typeof fsys == "object" && fsys.proj !== null) { │ │ │ │ │ + featOrFilt.id = fsys.proj.srsProjNumber; │ │ │ │ │ + featOrFilt.string = fsys.proj.srsCode; │ │ │ │ │ + } else { │ │ │ │ │ + featOrFilt = fsys; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read data from a string, and return a list of features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * APIMethod: iserror │ │ │ │ │ + * Check to see if the response from the server was an error. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. If nothing is supplied, │ │ │ │ │ + * the current response is examined. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} List of features. │ │ │ │ │ + * {Boolean} true if the response was an error. │ │ │ │ │ */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - this.features = []; │ │ │ │ │ - this.styles = {}; │ │ │ │ │ - this.fetched = {}; │ │ │ │ │ + iserror: function(data) { │ │ │ │ │ + var ret = null; │ │ │ │ │ │ │ │ │ │ - // Set default options │ │ │ │ │ - var options = { │ │ │ │ │ - depth: 0, │ │ │ │ │ - styleBaseUrl: this.styleBaseUrl │ │ │ │ │ - }; │ │ │ │ │ + if (!data) { │ │ │ │ │ + ret = (this.response.error !== ''); │ │ │ │ │ + } else { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + var errorNodes = data.documentElement.getElementsByTagName("ERROR"); │ │ │ │ │ + ret = (errorNodes !== null && errorNodes.length > 0); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - return this.parseData(data, options); │ │ │ │ │ + return ret; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseData │ │ │ │ │ - * Read data from a string, and return a list of features. │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read data from a string, and return an response. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * options - {Object} Hash of options │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} List of features. │ │ │ │ │ + * {<OpenLayers.Format.ArcXML.Response>} An ArcXML response. Note that this response │ │ │ │ │ + * data may change in the future. │ │ │ │ │ */ │ │ │ │ │ - parseData: function(data, options) { │ │ │ │ │ + read: function(data) { │ │ │ │ │ if (typeof data == "string") { │ │ │ │ │ data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // Loop throught the following node types in this order and │ │ │ │ │ - // process the nodes found │ │ │ │ │ - var types = ["Link", "NetworkLink", "Style", "StyleMap", "Placemark"]; │ │ │ │ │ - for (var i = 0, len = types.length; i < len; ++i) { │ │ │ │ │ - var type = types[i]; │ │ │ │ │ - │ │ │ │ │ - var nodes = this.getElementsByTagNameNS(data, "*", type); │ │ │ │ │ - │ │ │ │ │ - // skip to next type if no nodes are found │ │ │ │ │ - if (nodes.length == 0) { │ │ │ │ │ - continue; │ │ │ │ │ + var arcNode = null; │ │ │ │ │ + if (data && data.documentElement) { │ │ │ │ │ + if (data.documentElement.nodeName == "ARCXML") { │ │ │ │ │ + arcNode = data.documentElement; │ │ │ │ │ + } else { │ │ │ │ │ + arcNode = data.documentElement.getElementsByTagName("ARCXML")[0]; │ │ │ │ │ } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - switch (type.toLowerCase()) { │ │ │ │ │ - │ │ │ │ │ - // Fetch external links │ │ │ │ │ - case "link": │ │ │ │ │ - case "networklink": │ │ │ │ │ - this.parseLinks(nodes, options); │ │ │ │ │ - break; │ │ │ │ │ - │ │ │ │ │ - // parse style information │ │ │ │ │ - case "style": │ │ │ │ │ - if (this.extractStyles) { │ │ │ │ │ - this.parseStyles(nodes, options); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "stylemap": │ │ │ │ │ - if (this.extractStyles) { │ │ │ │ │ - this.parseStyleMaps(nodes, options); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - │ │ │ │ │ - // parse features │ │ │ │ │ - case "placemark": │ │ │ │ │ - this.parseFeatures(nodes, options); │ │ │ │ │ - break; │ │ │ │ │ + // in Safari, arcNode will be there but will have a child named │ │ │ │ │ + // parsererror │ │ │ │ │ + if (!arcNode || arcNode.firstChild.nodeName === 'parsererror') { │ │ │ │ │ + var error, source; │ │ │ │ │ + try { │ │ │ │ │ + error = data.firstChild.nodeValue; │ │ │ │ │ + source = data.firstChild.childNodes[1].firstChild.nodeValue; │ │ │ │ │ + } catch (err) { │ │ │ │ │ + // pass │ │ │ │ │ } │ │ │ │ │ + throw { │ │ │ │ │ + message: "Error parsing the ArcXML request", │ │ │ │ │ + error: error, │ │ │ │ │ + source: source │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return this.features; │ │ │ │ │ + var response = this.parseResponse(arcNode); │ │ │ │ │ + return response; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseLinks │ │ │ │ │ - * Finds URLs of linked KML documents and fetches them │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * nodes - {Array} of {DOMElement} data to read/parse. │ │ │ │ │ - * options - {Object} Hash of options │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Generate an ArcXml document string for sending to an ArcIMS server. │ │ │ │ │ * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string representing the ArcXML document request. │ │ │ │ │ */ │ │ │ │ │ - parseLinks: function(nodes, options) { │ │ │ │ │ - │ │ │ │ │ - // Fetch external links <NetworkLink> and <Link> │ │ │ │ │ - // Don't do anything if we have reached our maximum depth for recursion │ │ │ │ │ - if (options.depth >= this.maxDepth) { │ │ │ │ │ - return false; │ │ │ │ │ + write: function(request) { │ │ │ │ │ + if (!request) { │ │ │ │ │ + request = this.request; │ │ │ │ │ } │ │ │ │ │ + var root = this.createElementNS("", "ARCXML"); │ │ │ │ │ + root.setAttribute("version", "1.1"); │ │ │ │ │ │ │ │ │ │ - // increase depth │ │ │ │ │ - var newOptions = OpenLayers.Util.extend({}, options); │ │ │ │ │ - newOptions.depth++; │ │ │ │ │ + var reqElem = this.createElementNS("", "REQUEST"); │ │ │ │ │ │ │ │ │ │ - for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ - var href = this.parseProperty(nodes[i], "*", "href"); │ │ │ │ │ - if (href && !this.fetched[href]) { │ │ │ │ │ - this.fetched[href] = true; // prevent reloading the same urls │ │ │ │ │ - var data = this.fetchLink(href); │ │ │ │ │ - if (data) { │ │ │ │ │ - this.parseData(data, newOptions); │ │ │ │ │ + if (request.get_image != null) { │ │ │ │ │ + var getElem = this.createElementNS("", "GET_IMAGE"); │ │ │ │ │ + reqElem.appendChild(getElem); │ │ │ │ │ + │ │ │ │ │ + var propElem = this.createElementNS("", "PROPERTIES"); │ │ │ │ │ + getElem.appendChild(propElem); │ │ │ │ │ + │ │ │ │ │ + var props = request.get_image.properties; │ │ │ │ │ + if (props.featurecoordsys != null) { │ │ │ │ │ + var feat = this.createElementNS("", "FEATURECOORDSYS"); │ │ │ │ │ + propElem.appendChild(feat); │ │ │ │ │ + │ │ │ │ │ + if (props.featurecoordsys.id === 0) { │ │ │ │ │ + feat.setAttribute("string", props.featurecoordsys['string']); │ │ │ │ │ + } else { │ │ │ │ │ + feat.setAttribute("id", props.featurecoordsys.id); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - }, │ │ │ │ │ + if (props.filtercoordsys != null) { │ │ │ │ │ + var filt = this.createElementNS("", "FILTERCOORDSYS"); │ │ │ │ │ + propElem.appendChild(filt); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: fetchLink │ │ │ │ │ - * Fetches a URL and returns the result │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * href - {String} url to be fetched │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - fetchLink: function(href) { │ │ │ │ │ - var request = OpenLayers.Request.GET({ │ │ │ │ │ - url: href, │ │ │ │ │ - async: false │ │ │ │ │ - }); │ │ │ │ │ - if (request) { │ │ │ │ │ - return request.responseText; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + if (props.filtercoordsys.id === 0) { │ │ │ │ │ + filt.setAttribute("string", props.filtercoordsys.string); │ │ │ │ │ + } else { │ │ │ │ │ + filt.setAttribute("id", props.filtercoordsys.id); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseStyles │ │ │ │ │ - * Parses <Style> nodes │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * nodes - {Array} of {DOMElement} data to read/parse. │ │ │ │ │ - * options - {Object} Hash of options │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - parseStyles: function(nodes, options) { │ │ │ │ │ - for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ - var style = this.parseStyle(nodes[i]); │ │ │ │ │ - if (style) { │ │ │ │ │ - var styleName = (options.styleBaseUrl || "") + "#" + style.id; │ │ │ │ │ + if (props.envelope != null) { │ │ │ │ │ + var env = this.createElementNS("", "ENVELOPE"); │ │ │ │ │ + propElem.appendChild(env); │ │ │ │ │ │ │ │ │ │ - this.styles[styleName] = style; │ │ │ │ │ + env.setAttribute("minx", props.envelope.minx); │ │ │ │ │ + env.setAttribute("miny", props.envelope.miny); │ │ │ │ │ + env.setAttribute("maxx", props.envelope.maxx); │ │ │ │ │ + env.setAttribute("maxy", props.envelope.maxy); │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseKmlColor │ │ │ │ │ - * Parses a kml color (in 'aabbggrr' format) and returns the corresponding │ │ │ │ │ - * color and opacity or null if the color is invalid. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * kmlColor - {String} a kml formated color │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ - parseKmlColor: function(kmlColor) { │ │ │ │ │ - var color = null; │ │ │ │ │ - if (kmlColor) { │ │ │ │ │ - var matches = kmlColor.match(this.regExes.kmlColor); │ │ │ │ │ - if (matches) { │ │ │ │ │ - color = { │ │ │ │ │ - color: '#' + matches[4] + matches[3] + matches[2], │ │ │ │ │ - opacity: parseInt(matches[1], 16) / 255 │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return color; │ │ │ │ │ - }, │ │ │ │ │ + var imagesz = this.createElementNS("", "IMAGESIZE"); │ │ │ │ │ + propElem.appendChild(imagesz); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseStyle │ │ │ │ │ - * Parses the children of a <Style> node and builds the style hash │ │ │ │ │ - * accordingly │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} <Style> node │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - parseStyle: function(node) { │ │ │ │ │ - var style = {}; │ │ │ │ │ + imagesz.setAttribute("height", props.imagesize.height); │ │ │ │ │ + imagesz.setAttribute("width", props.imagesize.width); │ │ │ │ │ │ │ │ │ │ - var types = ["LineStyle", "PolyStyle", "IconStyle", "BalloonStyle", │ │ │ │ │ - "LabelStyle" │ │ │ │ │ - ]; │ │ │ │ │ - var type, styleTypeNode, nodeList, geometry, parser; │ │ │ │ │ - for (var i = 0, len = types.length; i < len; ++i) { │ │ │ │ │ - type = types[i]; │ │ │ │ │ - styleTypeNode = this.getElementsByTagNameNS(node, "*", type)[0]; │ │ │ │ │ - if (!styleTypeNode) { │ │ │ │ │ - continue; │ │ │ │ │ + if (props.imagesize.height != props.imagesize.printheight || │ │ │ │ │ + props.imagesize.width != props.imagesize.printwidth) { │ │ │ │ │ + imagesz.setAttribute("printheight", props.imagesize.printheight); │ │ │ │ │ + imagesz.setArrtibute("printwidth", props.imagesize.printwidth); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // only deal with first geometry of this type │ │ │ │ │ - switch (type.toLowerCase()) { │ │ │ │ │ - case "linestyle": │ │ │ │ │ - var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ - var color = this.parseKmlColor(kmlColor); │ │ │ │ │ - if (color) { │ │ │ │ │ - style["strokeColor"] = color.color; │ │ │ │ │ - style["strokeOpacity"] = color.opacity; │ │ │ │ │ - } │ │ │ │ │ + if (props.background != null) { │ │ │ │ │ + var backgrnd = this.createElementNS("", "BACKGROUND"); │ │ │ │ │ + propElem.appendChild(backgrnd); │ │ │ │ │ │ │ │ │ │ - var width = this.parseProperty(styleTypeNode, "*", "width"); │ │ │ │ │ - if (width) { │ │ │ │ │ - style["strokeWidth"] = width; │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ + backgrnd.setAttribute("color", │ │ │ │ │ + props.background.color.r + "," + │ │ │ │ │ + props.background.color.g + "," + │ │ │ │ │ + props.background.color.b); │ │ │ │ │ │ │ │ │ │ - case "polystyle": │ │ │ │ │ - var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ - var color = this.parseKmlColor(kmlColor); │ │ │ │ │ - if (color) { │ │ │ │ │ - style["fillOpacity"] = color.opacity; │ │ │ │ │ - style["fillColor"] = color.color; │ │ │ │ │ - } │ │ │ │ │ - // Check if fill is disabled │ │ │ │ │ - var fill = this.parseProperty(styleTypeNode, "*", "fill"); │ │ │ │ │ - if (fill == "0") { │ │ │ │ │ - style["fillColor"] = "none"; │ │ │ │ │ - } │ │ │ │ │ - // Check if outline is disabled │ │ │ │ │ - var outline = this.parseProperty(styleTypeNode, "*", "outline"); │ │ │ │ │ - if (outline == "0") { │ │ │ │ │ - style["strokeWidth"] = "0"; │ │ │ │ │ - } │ │ │ │ │ + if (props.background.transcolor !== null) { │ │ │ │ │ + backgrnd.setAttribute("transcolor", │ │ │ │ │ + props.background.transcolor.r + "," + │ │ │ │ │ + props.background.transcolor.g + "," + │ │ │ │ │ + props.background.transcolor.b); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - break; │ │ │ │ │ + if (props.layerlist != null && props.layerlist.length > 0) { │ │ │ │ │ + var layerlst = this.createElementNS("", "LAYERLIST"); │ │ │ │ │ + propElem.appendChild(layerlst); │ │ │ │ │ │ │ │ │ │ - case "iconstyle": │ │ │ │ │ - // set scale │ │ │ │ │ - var scale = parseFloat(this.parseProperty(styleTypeNode, │ │ │ │ │ - "*", "scale") || 1); │ │ │ │ │ + for (var ld = 0; ld < props.layerlist.length; ld++) { │ │ │ │ │ + var ldef = this.createElementNS("", "LAYERDEF"); │ │ │ │ │ + layerlst.appendChild(ldef); │ │ │ │ │ │ │ │ │ │ - // set default width and height of icon │ │ │ │ │ - var width = 32 * scale; │ │ │ │ │ - var height = 32 * scale; │ │ │ │ │ + ldef.setAttribute("id", props.layerlist[ld].id); │ │ │ │ │ + ldef.setAttribute("visible", props.layerlist[ld].visible); │ │ │ │ │ │ │ │ │ │ - var iconNode = this.getElementsByTagNameNS(styleTypeNode, │ │ │ │ │ - "*", │ │ │ │ │ - "Icon")[0]; │ │ │ │ │ - if (iconNode) { │ │ │ │ │ - var href = this.parseProperty(iconNode, "*", "href"); │ │ │ │ │ - if (href) { │ │ │ │ │ + if (typeof props.layerlist[ld].query == "object") { │ │ │ │ │ + var query = props.layerlist[ld].query; │ │ │ │ │ │ │ │ │ │ - var w = this.parseProperty(iconNode, "*", "w"); │ │ │ │ │ - var h = this.parseProperty(iconNode, "*", "h"); │ │ │ │ │ + if (query.where.length < 0) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // Settings for Google specific icons that are 64x64 │ │ │ │ │ - // We set the width and height to 64 and halve the │ │ │ │ │ - // scale to prevent icons from being too big │ │ │ │ │ - var google = "http://maps.google.com/mapfiles/kml"; │ │ │ │ │ - if (OpenLayers.String.startsWith( │ │ │ │ │ - href, google) && !w && !h) { │ │ │ │ │ - w = 64; │ │ │ │ │ - h = 64; │ │ │ │ │ - scale = scale / 2; │ │ │ │ │ - } │ │ │ │ │ + var queryElem = null; │ │ │ │ │ + if (typeof query.spatialfilter == "boolean" && query.spatialfilter) { │ │ │ │ │ + // handle spatial filter madness │ │ │ │ │ + queryElem = this.createElementNS("", "SPATIALQUERY"); │ │ │ │ │ + } else { │ │ │ │ │ + queryElem = this.createElementNS("", "QUERY"); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // if only dimension is defined, make sure the │ │ │ │ │ - // other one has the same value │ │ │ │ │ - w = w || h; │ │ │ │ │ - h = h || w; │ │ │ │ │ + queryElem.setAttribute("where", query.where); │ │ │ │ │ │ │ │ │ │ - if (w) { │ │ │ │ │ - width = parseInt(w) * scale; │ │ │ │ │ - } │ │ │ │ │ + if (typeof query.accuracy == "number" && query.accuracy > 0) { │ │ │ │ │ + queryElem.setAttribute("accuracy", query.accuracy); │ │ │ │ │ + } │ │ │ │ │ + if (typeof query.featurelimit == "number" && query.featurelimit < 2000) { │ │ │ │ │ + queryElem.setAttribute("featurelimit", query.featurelimit); │ │ │ │ │ + } │ │ │ │ │ + if (typeof query.subfields == "string" && query.subfields != "#ALL#") { │ │ │ │ │ + queryElem.setAttribute("subfields", query.subfields); │ │ │ │ │ + } │ │ │ │ │ + if (typeof query.joinexpression == "string" && query.joinexpression.length > 0) { │ │ │ │ │ + queryElem.setAttribute("joinexpression", query.joinexpression); │ │ │ │ │ + } │ │ │ │ │ + if (typeof query.jointables == "string" && query.jointables.length > 0) { │ │ │ │ │ + queryElem.setAttribute("jointables", query.jointables); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (h) { │ │ │ │ │ - height = parseInt(h) * scale; │ │ │ │ │ - } │ │ │ │ │ + ldef.appendChild(queryElem); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // support for internal icons │ │ │ │ │ - // (/root://icons/palette-x.png) │ │ │ │ │ - // x and y tell the position on the palette: │ │ │ │ │ - // - in pixels │ │ │ │ │ - // - starting from the left bottom │ │ │ │ │ - // We translate that to a position in the list │ │ │ │ │ - // and request the appropriate icon from the │ │ │ │ │ - // google maps website │ │ │ │ │ - var matches = href.match(this.regExes.kmlIconPalette); │ │ │ │ │ - if (matches) { │ │ │ │ │ - var palette = matches[1]; │ │ │ │ │ - var file_extension = matches[2]; │ │ │ │ │ + if (typeof props.layerlist[ld].renderer == "object") { │ │ │ │ │ + this.addRenderer(ldef, props.layerlist[ld].renderer); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else if (request.get_feature != null) { │ │ │ │ │ + var getElem = this.createElementNS("", "GET_FEATURES"); │ │ │ │ │ + getElem.setAttribute("outputmode", "newxml"); │ │ │ │ │ + getElem.setAttribute("checkesc", "true"); │ │ │ │ │ │ │ │ │ │ - var x = this.parseProperty(iconNode, "*", "x"); │ │ │ │ │ - var y = this.parseProperty(iconNode, "*", "y"); │ │ │ │ │ + if (request.get_feature.geometry) { │ │ │ │ │ + getElem.setAttribute("geometry", request.get_feature.geometry); │ │ │ │ │ + } else { │ │ │ │ │ + getElem.setAttribute("geometry", "false"); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var posX = x ? x / 32 : 0; │ │ │ │ │ - var posY = y ? (7 - y / 32) : 7; │ │ │ │ │ + if (request.get_feature.compact) { │ │ │ │ │ + getElem.setAttribute("compact", request.get_feature.compact); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var pos = posY * 8 + posX; │ │ │ │ │ - href = "http://maps.google.com/mapfiles/kml/pal" + │ │ │ │ │ - palette + "/icon" + pos + file_extension; │ │ │ │ │ - } │ │ │ │ │ + if (request.get_feature.featurelimit == "number") { │ │ │ │ │ + getElem.setAttribute("featurelimit", request.get_feature.featurelimit); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - style["graphicOpacity"] = 1; // fully opaque │ │ │ │ │ - style["externalGraphic"] = href; │ │ │ │ │ - } │ │ │ │ │ + getElem.setAttribute("globalenvelope", "true"); │ │ │ │ │ + reqElem.appendChild(getElem); │ │ │ │ │ │ │ │ │ │ - } │ │ │ │ │ + if (request.get_feature.layer != null && request.get_feature.layer.length > 0) { │ │ │ │ │ + var lyrElem = this.createElementNS("", "LAYER"); │ │ │ │ │ + lyrElem.setAttribute("id", request.get_feature.layer); │ │ │ │ │ + getElem.appendChild(lyrElem); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ + var fquery = request.get_feature.query; │ │ │ │ │ + if (fquery != null) { │ │ │ │ │ + var qElem = null; │ │ │ │ │ + if (fquery.isspatial) { │ │ │ │ │ + qElem = this.createElementNS("", "SPATIALQUERY"); │ │ │ │ │ + } else { │ │ │ │ │ + qElem = this.createElementNS("", "QUERY"); │ │ │ │ │ + } │ │ │ │ │ + getElem.appendChild(qElem); │ │ │ │ │ │ │ │ │ │ - // hotSpots define the offset for an Icon │ │ │ │ │ - var hotSpotNode = this.getElementsByTagNameNS(styleTypeNode, │ │ │ │ │ - "*", │ │ │ │ │ - "hotSpot")[0]; │ │ │ │ │ - if (hotSpotNode) { │ │ │ │ │ - var x = parseFloat(hotSpotNode.getAttribute("x")); │ │ │ │ │ - var y = parseFloat(hotSpotNode.getAttribute("y")); │ │ │ │ │ + if (typeof fquery.accuracy == "number") { │ │ │ │ │ + qElem.setAttribute("accuracy", fquery.accuracy); │ │ │ │ │ + } │ │ │ │ │ + //qElem.setAttribute("featurelimit", "5"); │ │ │ │ │ │ │ │ │ │ - var xUnits = hotSpotNode.getAttribute("xunits"); │ │ │ │ │ - if (xUnits == "pixels") { │ │ │ │ │ - style["graphicXOffset"] = -x * scale; │ │ │ │ │ - } else if (xUnits == "insetPixels") { │ │ │ │ │ - style["graphicXOffset"] = -width + (x * scale); │ │ │ │ │ - } else if (xUnits == "fraction") { │ │ │ │ │ - style["graphicXOffset"] = -width * x; │ │ │ │ │ - } │ │ │ │ │ + if (fquery.featurecoordsys != null) { │ │ │ │ │ + var fcsElem1 = this.createElementNS("", "FEATURECOORDSYS"); │ │ │ │ │ │ │ │ │ │ - var yUnits = hotSpotNode.getAttribute("yunits"); │ │ │ │ │ - if (yUnits == "pixels") { │ │ │ │ │ - style["graphicYOffset"] = -height + (y * scale) + 1; │ │ │ │ │ - } else if (yUnits == "insetPixels") { │ │ │ │ │ - style["graphicYOffset"] = -(y * scale) + 1; │ │ │ │ │ - } else if (yUnits == "fraction") { │ │ │ │ │ - style["graphicYOffset"] = -height * (1 - y) + 1; │ │ │ │ │ - } │ │ │ │ │ + if (fquery.featurecoordsys.id == 0) { │ │ │ │ │ + fcsElem1.setAttribute("string", fquery.featurecoordsys.string); │ │ │ │ │ + } else { │ │ │ │ │ + fcsElem1.setAttribute("id", fquery.featurecoordsys.id); │ │ │ │ │ } │ │ │ │ │ + qElem.appendChild(fcsElem1); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - style["graphicWidth"] = width; │ │ │ │ │ - style["graphicHeight"] = height; │ │ │ │ │ - break; │ │ │ │ │ + if (fquery.filtercoordsys != null) { │ │ │ │ │ + var fcsElem2 = this.createElementNS("", "FILTERCOORDSYS"); │ │ │ │ │ │ │ │ │ │ - case "balloonstyle": │ │ │ │ │ - var balloonStyle = OpenLayers.Util.getXmlNodeValue( │ │ │ │ │ - styleTypeNode); │ │ │ │ │ - if (balloonStyle) { │ │ │ │ │ - style["balloonStyle"] = balloonStyle.replace( │ │ │ │ │ - this.regExes.straightBracket, "${$1}"); │ │ │ │ │ + if (fquery.filtercoordsys.id === 0) { │ │ │ │ │ + fcsElem2.setAttribute("string", fquery.filtercoordsys.string); │ │ │ │ │ + } else { │ │ │ │ │ + fcsElem2.setAttribute("id", fquery.filtercoordsys.id); │ │ │ │ │ } │ │ │ │ │ - break; │ │ │ │ │ - case "labelstyle": │ │ │ │ │ - var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ - var color = this.parseKmlColor(kmlColor); │ │ │ │ │ - if (color) { │ │ │ │ │ - style["fontColor"] = color.color; │ │ │ │ │ - style["fontOpacity"] = color.opacity; │ │ │ │ │ + qElem.appendChild(fcsElem2); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (fquery.buffer > 0) { │ │ │ │ │ + var bufElem = this.createElementNS("", "BUFFER"); │ │ │ │ │ + bufElem.setAttribute("distance", fquery.buffer); │ │ │ │ │ + qElem.appendChild(bufElem); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (fquery.isspatial) { │ │ │ │ │ + var spfElem = this.createElementNS("", "SPATIALFILTER"); │ │ │ │ │ + spfElem.setAttribute("relation", fquery.spatialfilter.relation); │ │ │ │ │ + qElem.appendChild(spfElem); │ │ │ │ │ + │ │ │ │ │ + if (fquery.spatialfilter.envelope) { │ │ │ │ │ + var envElem = this.createElementNS("", "ENVELOPE"); │ │ │ │ │ + envElem.setAttribute("minx", fquery.spatialfilter.envelope.minx); │ │ │ │ │ + envElem.setAttribute("miny", fquery.spatialfilter.envelope.miny); │ │ │ │ │ + envElem.setAttribute("maxx", fquery.spatialfilter.envelope.maxx); │ │ │ │ │ + envElem.setAttribute("maxy", fquery.spatialfilter.envelope.maxy); │ │ │ │ │ + spfElem.appendChild(envElem); │ │ │ │ │ + } else if (typeof fquery.spatialfilter.polygon == "object") { │ │ │ │ │ + spfElem.appendChild(this.writePolygonGeometry(fquery.spatialfilter.polygon)); │ │ │ │ │ } │ │ │ │ │ - break; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - default: │ │ │ │ │ + if (fquery.where != null && fquery.where.length > 0) { │ │ │ │ │ + qElem.setAttribute("where", fquery.where); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // Some polygons have no line color, so we use the fillColor for that │ │ │ │ │ - if (!style["strokeColor"] && style["fillColor"]) { │ │ │ │ │ - style["strokeColor"] = style["fillColor"]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var id = node.getAttribute("id"); │ │ │ │ │ - if (id && style) { │ │ │ │ │ - style.id = id; │ │ │ │ │ - } │ │ │ │ │ + root.appendChild(reqElem); │ │ │ │ │ │ │ │ │ │ - return style; │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [root]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseStyleMaps │ │ │ │ │ - * Parses <StyleMap> nodes, but only uses the 'normal' key │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * nodes - {Array} of {DOMElement} data to read/parse. │ │ │ │ │ - * options - {Object} Hash of options │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - parseStyleMaps: function(nodes, options) { │ │ │ │ │ - // Only the default or "normal" part of the StyleMap is processed now │ │ │ │ │ - // To do the select or "highlight" bit, we'd need to change lots more │ │ │ │ │ │ │ │ │ │ - for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ - var node = nodes[i]; │ │ │ │ │ - var pairs = this.getElementsByTagNameNS(node, "*", │ │ │ │ │ - "Pair"); │ │ │ │ │ + addGroupRenderer: function(ldef, toprenderer) { │ │ │ │ │ + var topRelem = this.createElementNS("", "GROUPRENDERER"); │ │ │ │ │ + ldef.appendChild(topRelem); │ │ │ │ │ │ │ │ │ │ - var id = node.getAttribute("id"); │ │ │ │ │ - for (var j = 0, jlen = pairs.length; j < jlen; j++) { │ │ │ │ │ - var pair = pairs[j]; │ │ │ │ │ - // Use the shortcut in the SLD format to quickly retrieve the │ │ │ │ │ - // value of a node. Maybe it's good to have a method in │ │ │ │ │ - // Format.XML to do this │ │ │ │ │ - var key = this.parseProperty(pair, "*", "key"); │ │ │ │ │ - var styleUrl = this.parseProperty(pair, "*", "styleUrl"); │ │ │ │ │ + for (var rind = 0; rind < toprenderer.length; rind++) { │ │ │ │ │ + var renderer = toprenderer[rind]; │ │ │ │ │ + this.addRenderer(topRelem, renderer); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (styleUrl && key == "normal") { │ │ │ │ │ - this.styles[(options.styleBaseUrl || "") + "#" + id] = │ │ │ │ │ - this.styles[(options.styleBaseUrl || "") + styleUrl]; │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - // TODO: implement the "select" part │ │ │ │ │ - //if (styleUrl && key == "highlight") { │ │ │ │ │ - //} │ │ │ │ │ + addRenderer: function(topRelem, renderer) { │ │ │ │ │ + if (OpenLayers.Util.isArray(renderer)) { │ │ │ │ │ + this.addGroupRenderer(topRelem, renderer); │ │ │ │ │ + } else { │ │ │ │ │ + var renderElem = this.createElementNS("", renderer.type.toUpperCase() + "RENDERER"); │ │ │ │ │ + topRelem.appendChild(renderElem); │ │ │ │ │ │ │ │ │ │ + if (renderElem.tagName == "VALUEMAPRENDERER") { │ │ │ │ │ + this.addValueMapRenderer(renderElem, renderer); │ │ │ │ │ + } else if (renderElem.tagName == "VALUEMAPLABELRENDERER") { │ │ │ │ │ + this.addValueMapLabelRenderer(renderElem, renderer); │ │ │ │ │ + } else if (renderElem.tagName == "SIMPLELABELRENDERER") { │ │ │ │ │ + this.addSimpleLabelRenderer(renderElem, renderer); │ │ │ │ │ + } else if (renderElem.tagName == "SCALEDEPENDENTRENDERER") { │ │ │ │ │ + this.addScaleDependentRenderer(renderElem, renderer); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + addScaleDependentRenderer: function(renderElem, renderer) { │ │ │ │ │ + if (typeof renderer.lower == "string" || typeof renderer.lower == "number") { │ │ │ │ │ + renderElem.setAttribute("lower", renderer.lower); │ │ │ │ │ + } │ │ │ │ │ + if (typeof renderer.upper == "string" || typeof renderer.upper == "number") { │ │ │ │ │ + renderElem.setAttribute("upper", renderer.upper); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ + this.addRenderer(renderElem, renderer.renderer); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseFeatures │ │ │ │ │ - * Loop through all Placemark nodes and parse them. │ │ │ │ │ - * Will create a list of features │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * nodes - {Array} of {DOMElement} data to read/parse. │ │ │ │ │ - * options - {Object} Hash of options │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - parseFeatures: function(nodes, options) { │ │ │ │ │ - var features = []; │ │ │ │ │ - for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ - var featureNode = nodes[i]; │ │ │ │ │ - var feature = this.parseFeature.apply(this, [featureNode]); │ │ │ │ │ - if (feature) { │ │ │ │ │ + addValueMapLabelRenderer: function(renderElem, renderer) { │ │ │ │ │ + renderElem.setAttribute("lookupfield", renderer.lookupfield); │ │ │ │ │ + renderElem.setAttribute("labelfield", renderer.labelfield); │ │ │ │ │ │ │ │ │ │ - // Create reference to styleUrl │ │ │ │ │ - if (this.extractStyles && feature.attributes && │ │ │ │ │ - feature.attributes.styleUrl) { │ │ │ │ │ - feature.style = this.getStyle(feature.attributes.styleUrl, options); │ │ │ │ │ + if (typeof renderer.exacts == "object") { │ │ │ │ │ + for (var ext = 0, extlen = renderer.exacts.length; ext < extlen; ext++) { │ │ │ │ │ + var exact = renderer.exacts[ext]; │ │ │ │ │ + │ │ │ │ │ + var eelem = this.createElementNS("", "EXACT"); │ │ │ │ │ + │ │ │ │ │ + if (typeof exact.value == "string") { │ │ │ │ │ + eelem.setAttribute("value", exact.value); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.label == "string") { │ │ │ │ │ + eelem.setAttribute("label", exact.label); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.method == "string") { │ │ │ │ │ + eelem.setAttribute("method", exact.method); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - if (this.extractStyles) { │ │ │ │ │ - // Make sure that <Style> nodes within a placemark are │ │ │ │ │ - // processed as well │ │ │ │ │ - var inlineStyleNode = this.getElementsByTagNameNS(featureNode, │ │ │ │ │ - "*", │ │ │ │ │ - "Style")[0]; │ │ │ │ │ - if (inlineStyleNode) { │ │ │ │ │ - var inlineStyle = this.parseStyle(inlineStyleNode); │ │ │ │ │ - if (inlineStyle) { │ │ │ │ │ - feature.style = OpenLayers.Util.extend( │ │ │ │ │ - feature.style, inlineStyle │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + renderElem.appendChild(eelem); │ │ │ │ │ + │ │ │ │ │ + if (typeof exact.symbol == "object") { │ │ │ │ │ + var selem = null; │ │ │ │ │ + │ │ │ │ │ + if (exact.symbol.type == "text") { │ │ │ │ │ + selem = this.createElementNS("", "TEXTSYMBOL"); │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - // check if gx:Track elements should be parsed │ │ │ │ │ - if (this.extractTracks) { │ │ │ │ │ - var tracks = this.getElementsByTagNameNS( │ │ │ │ │ - featureNode, this.namespaces.gx, "Track" │ │ │ │ │ - ); │ │ │ │ │ - if (tracks && tracks.length > 0) { │ │ │ │ │ - var track = tracks[0]; │ │ │ │ │ - var container = { │ │ │ │ │ - features: [], │ │ │ │ │ - feature: feature │ │ │ │ │ - }; │ │ │ │ │ - this.readNode(track, container); │ │ │ │ │ - if (container.features.length > 0) { │ │ │ │ │ - features.push.apply(features, container.features); │ │ │ │ │ + if (selem != null) { │ │ │ │ │ + var keys = this.fontStyleKeys; │ │ │ │ │ + for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ + var key = keys[i]; │ │ │ │ │ + if (exact.symbol[key]) { │ │ │ │ │ + selem.setAttribute(key, exact.symbol[key]); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + eelem.appendChild(selem); │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - // add feature to list of features │ │ │ │ │ - features.push(feature); │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - throw "Bad Placemark: " + i; │ │ │ │ │ - } │ │ │ │ │ + } // for each exact │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // add new features to existing feature list │ │ │ │ │ - this.features = this.features.concat(features); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "kml": { │ │ │ │ │ - "when": function(node, container) { │ │ │ │ │ - container.whens.push(OpenLayers.Date.parse( │ │ │ │ │ - this.getChildValue(node) │ │ │ │ │ - )); │ │ │ │ │ - }, │ │ │ │ │ - "_trackPointAttribute": function(node, container) { │ │ │ │ │ - var name = node.nodeName.split(":").pop(); │ │ │ │ │ - container.attributes[name].push(this.getChildValue(node)); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "gx": { │ │ │ │ │ - "Track": function(node, container) { │ │ │ │ │ - var obj = { │ │ │ │ │ - whens: [], │ │ │ │ │ - points: [], │ │ │ │ │ - angles: [] │ │ │ │ │ - }; │ │ │ │ │ - if (this.trackAttributes) { │ │ │ │ │ - var name; │ │ │ │ │ - obj.attributes = {}; │ │ │ │ │ - for (var i = 0, ii = this.trackAttributes.length; i < ii; ++i) { │ │ │ │ │ - name = this.trackAttributes[i]; │ │ │ │ │ - obj.attributes[name] = []; │ │ │ │ │ - if (!(name in this.readers.kml)) { │ │ │ │ │ - this.readers.kml[name] = this.readers.kml._trackPointAttribute; │ │ │ │ │ + addValueMapRenderer: function(renderElem, renderer) { │ │ │ │ │ + renderElem.setAttribute("lookupfield", renderer.lookupfield); │ │ │ │ │ + │ │ │ │ │ + if (typeof renderer.ranges == "object") { │ │ │ │ │ + for (var rng = 0, rnglen = renderer.ranges.length; rng < rnglen; rng++) { │ │ │ │ │ + var range = renderer.ranges[rng]; │ │ │ │ │ + │ │ │ │ │ + var relem = this.createElementNS("", "RANGE"); │ │ │ │ │ + relem.setAttribute("lower", range.lower); │ │ │ │ │ + relem.setAttribute("upper", range.upper); │ │ │ │ │ + │ │ │ │ │ + renderElem.appendChild(relem); │ │ │ │ │ + │ │ │ │ │ + if (typeof range.symbol == "object") { │ │ │ │ │ + var selem = null; │ │ │ │ │ + │ │ │ │ │ + if (range.symbol.type == "simplepolygon") { │ │ │ │ │ + selem = this.createElementNS("", "SIMPLEPOLYGONSYMBOL"); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (selem != null) { │ │ │ │ │ + if (typeof range.symbol.boundarycolor == "string") { │ │ │ │ │ + selem.setAttribute("boundarycolor", range.symbol.boundarycolor); │ │ │ │ │ + } │ │ │ │ │ + if (typeof range.symbol.fillcolor == "string") { │ │ │ │ │ + selem.setAttribute("fillcolor", range.symbol.fillcolor); │ │ │ │ │ + } │ │ │ │ │ + if (typeof range.symbol.filltransparency == "number") { │ │ │ │ │ + selem.setAttribute("filltransparency", range.symbol.filltransparency); │ │ │ │ │ } │ │ │ │ │ + relem.appendChild(selem); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - if (obj.whens.length !== obj.points.length) { │ │ │ │ │ - throw new Error("gx:Track with unequal number of when (" + │ │ │ │ │ - obj.whens.length + ") and gx:coord (" + │ │ │ │ │ - obj.points.length + ") elements."); │ │ │ │ │ + } // for each range │ │ │ │ │ + } else if (typeof renderer.exacts == "object") { │ │ │ │ │ + for (var ext = 0, extlen = renderer.exacts.length; ext < extlen; ext++) { │ │ │ │ │ + var exact = renderer.exacts[ext]; │ │ │ │ │ + │ │ │ │ │ + var eelem = this.createElementNS("", "EXACT"); │ │ │ │ │ + if (typeof exact.value == "string") { │ │ │ │ │ + eelem.setAttribute("value", exact.value); │ │ │ │ │ } │ │ │ │ │ - var hasAngles = obj.angles.length > 0; │ │ │ │ │ - if (hasAngles && obj.whens.length !== obj.angles.length) { │ │ │ │ │ - throw new Error("gx:Track with unequal number of when (" + │ │ │ │ │ - obj.whens.length + ") and gx:angles (" + │ │ │ │ │ - obj.angles.length + ") elements."); │ │ │ │ │ + if (typeof exact.label == "string") { │ │ │ │ │ + eelem.setAttribute("label", exact.label); │ │ │ │ │ } │ │ │ │ │ - var feature, point, angles; │ │ │ │ │ - for (var i = 0, ii = obj.whens.length; i < ii; ++i) { │ │ │ │ │ - feature = container.feature.clone(); │ │ │ │ │ - feature.fid = container.feature.fid || container.feature.id; │ │ │ │ │ - point = obj.points[i]; │ │ │ │ │ - feature.geometry = point; │ │ │ │ │ - if ("z" in point) { │ │ │ │ │ - feature.attributes.altitude = point.z; │ │ │ │ │ - } │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - feature.geometry.transform( │ │ │ │ │ - this.externalProjection, this.internalProjection │ │ │ │ │ - ); │ │ │ │ │ + if (typeof exact.method == "string") { │ │ │ │ │ + eelem.setAttribute("method", exact.method); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + renderElem.appendChild(eelem); │ │ │ │ │ + │ │ │ │ │ + if (typeof exact.symbol == "object") { │ │ │ │ │ + var selem = null; │ │ │ │ │ + │ │ │ │ │ + if (exact.symbol.type == "simplemarker") { │ │ │ │ │ + selem = this.createElementNS("", "SIMPLEMARKERSYMBOL"); │ │ │ │ │ } │ │ │ │ │ - if (this.trackAttributes) { │ │ │ │ │ - for (var j = 0, jj = this.trackAttributes.length; j < jj; ++j) { │ │ │ │ │ - var name = this.trackAttributes[j]; │ │ │ │ │ - feature.attributes[name] = obj.attributes[name][i]; │ │ │ │ │ + │ │ │ │ │ + if (selem != null) { │ │ │ │ │ + if (typeof exact.symbol.antialiasing == "string") { │ │ │ │ │ + selem.setAttribute("antialiasing", exact.symbol.antialiasing); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.color == "string") { │ │ │ │ │ + selem.setAttribute("color", exact.symbol.color); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.outline == "string") { │ │ │ │ │ + selem.setAttribute("outline", exact.symbol.outline); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.overlap == "string") { │ │ │ │ │ + selem.setAttribute("overlap", exact.symbol.overlap); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.shadow == "string") { │ │ │ │ │ + selem.setAttribute("shadow", exact.symbol.shadow); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.transparency == "number") { │ │ │ │ │ + selem.setAttribute("transparency", exact.symbol.transparency); │ │ │ │ │ + } │ │ │ │ │ + //if (typeof exact.symbol.type == "string") │ │ │ │ │ + // selem.setAttribute("type", exact.symbol.type); │ │ │ │ │ + if (typeof exact.symbol.usecentroid == "string") { │ │ │ │ │ + selem.setAttribute("usecentroid", exact.symbol.usecentroid); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.width == "number") { │ │ │ │ │ + selem.setAttribute("width", exact.symbol.width); │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + eelem.appendChild(selem); │ │ │ │ │ } │ │ │ │ │ - feature.attributes.when = obj.whens[i]; │ │ │ │ │ - feature.attributes.trackId = container.feature.id; │ │ │ │ │ - if (hasAngles) { │ │ │ │ │ - angles = obj.angles[i]; │ │ │ │ │ - feature.attributes.heading = parseFloat(angles[0]); │ │ │ │ │ - feature.attributes.tilt = parseFloat(angles[1]); │ │ │ │ │ - feature.attributes.roll = parseFloat(angles[2]); │ │ │ │ │ - } │ │ │ │ │ - container.features.push(feature); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "coord": function(node, container) { │ │ │ │ │ - var str = this.getChildValue(node); │ │ │ │ │ - var coords = str.replace(this.regExes.trimSpace, "").split(/\s+/); │ │ │ │ │ - var point = new OpenLayers.Geometry.Point(coords[0], coords[1]); │ │ │ │ │ - if (coords.length > 2) { │ │ │ │ │ - point.z = parseFloat(coords[2]); │ │ │ │ │ } │ │ │ │ │ - container.points.push(point); │ │ │ │ │ - }, │ │ │ │ │ - "angles": function(node, container) { │ │ │ │ │ - var str = this.getChildValue(node); │ │ │ │ │ - var parts = str.replace(this.regExes.trimSpace, "").split(/\s+/); │ │ │ │ │ - container.angles.push(parts); │ │ │ │ │ - } │ │ │ │ │ + } // for each exact │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseFeature │ │ │ │ │ - * This function is the core of the KML parsing code in OpenLayers. │ │ │ │ │ - * It creates the geometries that are then attached to the returned │ │ │ │ │ - * feature, and calls parseAttributes() to get attribute data out. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A vector feature. │ │ │ │ │ - */ │ │ │ │ │ - parseFeature: function(node) { │ │ │ │ │ - // only accept one geometry per feature - look for highest "order" │ │ │ │ │ - var order = ["MultiGeometry", "Polygon", "LineString", "Point"]; │ │ │ │ │ - var type, nodeList, geometry, parser; │ │ │ │ │ - for (var i = 0, len = order.length; i < len; ++i) { │ │ │ │ │ - type = order[i]; │ │ │ │ │ - this.internalns = node.namespaceURI ? │ │ │ │ │ - node.namespaceURI : this.kmlns; │ │ │ │ │ - nodeList = this.getElementsByTagNameNS(node, │ │ │ │ │ - this.internalns, type); │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - // only deal with first geometry of this type │ │ │ │ │ - var parser = this.parseGeometry[type.toLowerCase()]; │ │ │ │ │ - if (parser) { │ │ │ │ │ - geometry = parser.apply(this, [nodeList[0]]); │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry.transform(this.externalProjection, │ │ │ │ │ - this.internalProjection); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - throw new TypeError("Unsupported geometry type: " + type); │ │ │ │ │ - } │ │ │ │ │ - // stop looking for different geometry types │ │ │ │ │ - break; │ │ │ │ │ + │ │ │ │ │ + addSimpleLabelRenderer: function(renderElem, renderer) { │ │ │ │ │ + renderElem.setAttribute("field", renderer.field); │ │ │ │ │ + var keys = ['featureweight', 'howmanylabels', 'labelbufferratio', │ │ │ │ │ + 'labelpriorities', 'labelweight', 'linelabelposition', │ │ │ │ │ + 'rotationalangles' │ │ │ │ │ + ]; │ │ │ │ │ + for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ + var key = keys[i]; │ │ │ │ │ + if (renderer[key]) { │ │ │ │ │ + renderElem.setAttribute(key, renderer[key]); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // construct feature (optionally with attributes) │ │ │ │ │ - var attributes; │ │ │ │ │ - if (this.extractAttributes) { │ │ │ │ │ - attributes = this.parseAttributes(node); │ │ │ │ │ - } │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry, attributes); │ │ │ │ │ + if (renderer.symbol.type == "text") { │ │ │ │ │ + var symbol = renderer.symbol; │ │ │ │ │ + var selem = this.createElementNS("", "TEXTSYMBOL"); │ │ │ │ │ + renderElem.appendChild(selem); │ │ │ │ │ │ │ │ │ │ - var fid = node.getAttribute("id") || node.getAttribute("name"); │ │ │ │ │ - if (fid != null) { │ │ │ │ │ - feature.fid = fid; │ │ │ │ │ + var keys = this.fontStyleKeys; │ │ │ │ │ + for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ + var key = keys[i]; │ │ │ │ │ + if (symbol[key]) { │ │ │ │ │ + selem.setAttribute(key, renderer[key]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return feature; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getStyle │ │ │ │ │ - * Retrieves a style from a style hash using styleUrl as the key │ │ │ │ │ - * If the styleUrl doesn't exist yet, we try to fetch it │ │ │ │ │ - * Internet │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * styleUrl - {String} URL of style │ │ │ │ │ - * options - {Object} Hash of options │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} - (reference to) Style hash │ │ │ │ │ - */ │ │ │ │ │ - getStyle: function(styleUrl, options) { │ │ │ │ │ + writePolygonGeometry: function(polygon) { │ │ │ │ │ + if (!(polygon instanceof OpenLayers.Geometry.Polygon)) { │ │ │ │ │ + throw { │ │ │ │ │ + message: 'Cannot write polygon geometry to ArcXML with an ' + │ │ │ │ │ + polygon.CLASS_NAME + ' object.', │ │ │ │ │ + geometry: polygon │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var styleBaseUrl = OpenLayers.Util.removeTail(styleUrl); │ │ │ │ │ + var polyElem = this.createElementNS("", "POLYGON"); │ │ │ │ │ │ │ │ │ │ - var newOptions = OpenLayers.Util.extend({}, options); │ │ │ │ │ - newOptions.depth++; │ │ │ │ │ - newOptions.styleBaseUrl = styleBaseUrl; │ │ │ │ │ + for (var ln = 0, lnlen = polygon.components.length; ln < lnlen; ln++) { │ │ │ │ │ + var ring = polygon.components[ln]; │ │ │ │ │ + var ringElem = this.createElementNS("", "RING"); │ │ │ │ │ │ │ │ │ │ - // Fetch remote Style URLs (if not fetched before) │ │ │ │ │ - if (!this.styles[styleUrl] && │ │ │ │ │ - !OpenLayers.String.startsWith(styleUrl, "#") && │ │ │ │ │ - newOptions.depth <= this.maxDepth && │ │ │ │ │ - !this.fetched[styleBaseUrl]) { │ │ │ │ │ + for (var rn = 0, rnlen = ring.components.length; rn < rnlen; rn++) { │ │ │ │ │ + var point = ring.components[rn]; │ │ │ │ │ + var pointElem = this.createElementNS("", "POINT"); │ │ │ │ │ │ │ │ │ │ - var data = this.fetchLink(styleBaseUrl); │ │ │ │ │ - if (data) { │ │ │ │ │ - this.parseData(data, newOptions); │ │ │ │ │ + pointElem.setAttribute("x", point.x); │ │ │ │ │ + pointElem.setAttribute("y", point.y); │ │ │ │ │ + │ │ │ │ │ + ringElem.appendChild(pointElem); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ + polyElem.appendChild(ringElem); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // return requested style │ │ │ │ │ - var style = OpenLayers.Util.extend({}, this.styles[styleUrl]); │ │ │ │ │ - return style; │ │ │ │ │ + return polyElem; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: parseGeometry │ │ │ │ │ - * Properties of this object are the functions that parse geometries based │ │ │ │ │ - * on their type. │ │ │ │ │ + * Method: parseResponse │ │ │ │ │ + * Take an ArcXML response, and parse in into this object's internal properties. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} The ArcXML response, as either a string or the │ │ │ │ │ + * top level DOMElement of the response. │ │ │ │ │ */ │ │ │ │ │ - parseGeometry: { │ │ │ │ │ + parseResponse: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + var newData = new OpenLayers.Format.XML(); │ │ │ │ │ + data = newData.read(data); │ │ │ │ │ + } │ │ │ │ │ + var response = new OpenLayers.Format.ArcXML.Response(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseGeometry.point │ │ │ │ │ - * Given a KML node representing a point geometry, create an OpenLayers │ │ │ │ │ - * point geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A KML Point node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} A point geometry. │ │ │ │ │ - */ │ │ │ │ │ - point: function(node) { │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, this.internalns, │ │ │ │ │ - "coordinates"); │ │ │ │ │ - var coords = []; │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - var coordString = nodeList[0].firstChild.nodeValue; │ │ │ │ │ - coordString = coordString.replace(this.regExes.removeSpace, ""); │ │ │ │ │ - coords = coordString.split(","); │ │ │ │ │ + var errorNode = data.getElementsByTagName("ERROR"); │ │ │ │ │ + │ │ │ │ │ + if (errorNode != null && errorNode.length > 0) { │ │ │ │ │ + response.error = this.getChildValue(errorNode, "Unknown error."); │ │ │ │ │ + } else { │ │ │ │ │ + var responseNode = data.getElementsByTagName("RESPONSE"); │ │ │ │ │ + │ │ │ │ │ + if (responseNode == null || responseNode.length == 0) { │ │ │ │ │ + response.error = "No RESPONSE tag found in ArcXML response."; │ │ │ │ │ + return response; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - var point = null; │ │ │ │ │ - if (coords.length > 1) { │ │ │ │ │ - // preserve third dimension │ │ │ │ │ - if (coords.length == 2) { │ │ │ │ │ - coords[2] = null; │ │ │ │ │ - } │ │ │ │ │ - point = new OpenLayers.Geometry.Point(coords[0], coords[1], │ │ │ │ │ - coords[2]); │ │ │ │ │ - } else { │ │ │ │ │ - throw "Bad coordinate string: " + coordString; │ │ │ │ │ + var rtype = responseNode[0].firstChild.nodeName; │ │ │ │ │ + if (rtype == "#text") { │ │ │ │ │ + rtype = responseNode[0].firstChild.nextSibling.nodeName; │ │ │ │ │ } │ │ │ │ │ - return point; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseGeometry.linestring │ │ │ │ │ - * Given a KML node representing a linestring geometry, create an │ │ │ │ │ - * OpenLayers linestring geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A KML LineString node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.LineString>} A linestring geometry. │ │ │ │ │ - */ │ │ │ │ │ - linestring: function(node, ring) { │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, this.internalns, │ │ │ │ │ - "coordinates"); │ │ │ │ │ - var line = null; │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - var coordString = this.getChildValue(nodeList[0]); │ │ │ │ │ + if (rtype == "IMAGE") { │ │ │ │ │ + var envelopeNode = data.getElementsByTagName("ENVELOPE"); │ │ │ │ │ + var outputNode = data.getElementsByTagName("OUTPUT"); │ │ │ │ │ │ │ │ │ │ - coordString = coordString.replace(this.regExes.trimSpace, │ │ │ │ │ - ""); │ │ │ │ │ - coordString = coordString.replace(this.regExes.trimComma, │ │ │ │ │ - ","); │ │ │ │ │ - var pointList = coordString.split(this.regExes.splitSpace); │ │ │ │ │ - var numPoints = pointList.length; │ │ │ │ │ - var points = new Array(numPoints); │ │ │ │ │ - var coords, numCoords; │ │ │ │ │ - for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ - coords = pointList[i].split(","); │ │ │ │ │ - numCoords = coords.length; │ │ │ │ │ - if (numCoords > 1) { │ │ │ │ │ - if (coords.length == 2) { │ │ │ │ │ - coords[2] = null; │ │ │ │ │ - } │ │ │ │ │ - points[i] = new OpenLayers.Geometry.Point(coords[0], │ │ │ │ │ - coords[1], │ │ │ │ │ - coords[2]); │ │ │ │ │ - } else { │ │ │ │ │ - throw "Bad LineString point coordinates: " + │ │ │ │ │ - pointList[i]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (numPoints) { │ │ │ │ │ - if (ring) { │ │ │ │ │ - line = new OpenLayers.Geometry.LinearRing(points); │ │ │ │ │ - } else { │ │ │ │ │ - line = new OpenLayers.Geometry.LineString(points); │ │ │ │ │ - } │ │ │ │ │ + if (envelopeNode == null || envelopeNode.length == 0) { │ │ │ │ │ + response.error = "No ENVELOPE tag found in ArcXML response."; │ │ │ │ │ + } else if (outputNode == null || outputNode.length == 0) { │ │ │ │ │ + response.error = "No OUTPUT tag found in ArcXML response."; │ │ │ │ │ } else { │ │ │ │ │ - throw "Bad LineString coordinates: " + coordString; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return line; │ │ │ │ │ - }, │ │ │ │ │ + var envAttr = this.parseAttributes(envelopeNode[0]); │ │ │ │ │ + var outputAttr = this.parseAttributes(outputNode[0]); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseGeometry.polygon │ │ │ │ │ - * Given a KML node representing a polygon geometry, create an │ │ │ │ │ - * OpenLayers polygon geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A KML Polygon node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Polygon>} A polygon geometry. │ │ │ │ │ - */ │ │ │ │ │ - polygon: function(node) { │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, this.internalns, │ │ │ │ │ - "LinearRing"); │ │ │ │ │ - var numRings = nodeList.length; │ │ │ │ │ - var components = new Array(numRings); │ │ │ │ │ - if (numRings > 0) { │ │ │ │ │ - // this assumes exterior ring first, inner rings after │ │ │ │ │ - var ring; │ │ │ │ │ - for (var i = 0, len = nodeList.length; i < len; ++i) { │ │ │ │ │ - ring = this.parseGeometry.linestring.apply(this, │ │ │ │ │ - [nodeList[i], true]); │ │ │ │ │ - if (ring) { │ │ │ │ │ - components[i] = ring; │ │ │ │ │ + if (typeof outputAttr.type == "string") { │ │ │ │ │ + response.image = { │ │ │ │ │ + envelope: envAttr, │ │ │ │ │ + output: { │ │ │ │ │ + type: outputAttr.type, │ │ │ │ │ + data: this.getChildValue(outputNode[0]) │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ } else { │ │ │ │ │ - throw "Bad LinearRing geometry: " + i; │ │ │ │ │ + response.image = { │ │ │ │ │ + envelope: envAttr, │ │ │ │ │ + output: outputAttr │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Geometry.Polygon(components); │ │ │ │ │ - }, │ │ │ │ │ + } else if (rtype == "FEATURES") { │ │ │ │ │ + var features = responseNode[0].getElementsByTagName("FEATURES"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseGeometry.multigeometry │ │ │ │ │ - * Given a KML node representing a multigeometry, create an │ │ │ │ │ - * OpenLayers geometry collection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A KML MultiGeometry node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Collection>} A geometry collection. │ │ │ │ │ - */ │ │ │ │ │ - multigeometry: function(node) { │ │ │ │ │ - var child, parser; │ │ │ │ │ - var parts = []; │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ - child = children[i]; │ │ │ │ │ - if (child.nodeType == 1) { │ │ │ │ │ - var type = (child.prefix) ? │ │ │ │ │ - child.nodeName.split(":")[1] : │ │ │ │ │ - child.nodeName; │ │ │ │ │ - var parser = this.parseGeometry[type.toLowerCase()]; │ │ │ │ │ - if (parser) { │ │ │ │ │ - parts.push(parser.apply(this, [child])); │ │ │ │ │ + // get the feature count │ │ │ │ │ + var featureCount = features[0].getElementsByTagName("FEATURECOUNT"); │ │ │ │ │ + response.features.featurecount = featureCount[0].getAttribute("count"); │ │ │ │ │ + │ │ │ │ │ + if (response.features.featurecount > 0) { │ │ │ │ │ + // get the feature envelope │ │ │ │ │ + var envelope = features[0].getElementsByTagName("ENVELOPE"); │ │ │ │ │ + response.features.envelope = this.parseAttributes(envelope[0], typeof(0)); │ │ │ │ │ + │ │ │ │ │ + // get the field values per feature │ │ │ │ │ + var featureList = features[0].getElementsByTagName("FEATURE"); │ │ │ │ │ + for (var fn = 0; fn < featureList.length; fn++) { │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ + var fields = featureList[fn].getElementsByTagName("FIELD"); │ │ │ │ │ + │ │ │ │ │ + for (var fdn = 0; fdn < fields.length; fdn++) { │ │ │ │ │ + var fieldName = fields[fdn].getAttribute("name"); │ │ │ │ │ + var fieldValue = fields[fdn].getAttribute("value"); │ │ │ │ │ + feature.attributes[fieldName] = fieldValue; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var geom = featureList[fn].getElementsByTagName("POLYGON"); │ │ │ │ │ + │ │ │ │ │ + if (geom.length > 0) { │ │ │ │ │ + // if there is a polygon, create an openlayers polygon, and assign │ │ │ │ │ + // it to the .geometry property of the feature │ │ │ │ │ + var ring = geom[0].getElementsByTagName("RING"); │ │ │ │ │ + │ │ │ │ │ + var polys = []; │ │ │ │ │ + for (var rn = 0; rn < ring.length; rn++) { │ │ │ │ │ + var linearRings = []; │ │ │ │ │ + linearRings.push(this.parsePointGeometry(ring[rn])); │ │ │ │ │ + │ │ │ │ │ + var holes = ring[rn].getElementsByTagName("HOLE"); │ │ │ │ │ + for (var hn = 0; hn < holes.length; hn++) { │ │ │ │ │ + linearRings.push(this.parsePointGeometry(holes[hn])); │ │ │ │ │ + } │ │ │ │ │ + holes = null; │ │ │ │ │ + polys.push(new OpenLayers.Geometry.Polygon(linearRings)); │ │ │ │ │ + linearRings = null; │ │ │ │ │ + } │ │ │ │ │ + ring = null; │ │ │ │ │ + │ │ │ │ │ + if (polys.length == 1) { │ │ │ │ │ + feature.geometry = polys[0]; │ │ │ │ │ + } else { │ │ │ │ │ + feature.geometry = new OpenLayers.Geometry.MultiPolygon(polys); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + response.features.feature.push(feature); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + response.error = "Unidentified response type."; │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Geometry.Collection(parts); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ + return response; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ * Method: parseAttributes │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ + * node - {<DOMElement>} An element to parse attributes from. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} An attributes object. │ │ │ │ │ + * {Object} An attributes object, with properties set to attribute values. │ │ │ │ │ */ │ │ │ │ │ - parseAttributes: function(node) { │ │ │ │ │ + parseAttributes: function(node, type) { │ │ │ │ │ var attributes = {}; │ │ │ │ │ - │ │ │ │ │ - // Extended Data is parsed first. │ │ │ │ │ - var edNodes = node.getElementsByTagName("ExtendedData"); │ │ │ │ │ - if (edNodes.length) { │ │ │ │ │ - attributes = this.parseExtendedData(edNodes[0]); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // assume attribute nodes are type 1 children with a type 3 or 4 child │ │ │ │ │ - var child, grandchildren, grandchild; │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - │ │ │ │ │ - for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ - child = children[i]; │ │ │ │ │ - if (child.nodeType == 1) { │ │ │ │ │ - grandchildren = child.childNodes; │ │ │ │ │ - if (grandchildren.length >= 1 && grandchildren.length <= 3) { │ │ │ │ │ - var grandchild; │ │ │ │ │ - switch (grandchildren.length) { │ │ │ │ │ - case 1: │ │ │ │ │ - grandchild = grandchildren[0]; │ │ │ │ │ - break; │ │ │ │ │ - case 2: │ │ │ │ │ - var c1 = grandchildren[0]; │ │ │ │ │ - var c2 = grandchildren[1]; │ │ │ │ │ - grandchild = (c1.nodeType == 3 || c1.nodeType == 4) ? │ │ │ │ │ - c1 : c2; │ │ │ │ │ - break; │ │ │ │ │ - case 3: │ │ │ │ │ - default: │ │ │ │ │ - grandchild = grandchildren[1]; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - if (grandchild.nodeType == 3 || grandchild.nodeType == 4) { │ │ │ │ │ - var name = (child.prefix) ? │ │ │ │ │ - child.nodeName.split(":")[1] : │ │ │ │ │ - child.nodeName; │ │ │ │ │ - var value = OpenLayers.Util.getXmlNodeValue(grandchild); │ │ │ │ │ - if (value) { │ │ │ │ │ - value = value.replace(this.regExes.trimSpace, ""); │ │ │ │ │ - attributes[name] = value; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + for (var attr = 0; attr < node.attributes.length; attr++) { │ │ │ │ │ + if (type == "number") { │ │ │ │ │ + attributes[node.attributes[attr].nodeName] = parseFloat(node.attributes[attr].nodeValue); │ │ │ │ │ + } else { │ │ │ │ │ + attributes[node.attributes[attr].nodeName] = node.attributes[attr].nodeValue; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ return attributes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: parseExtendedData │ │ │ │ │ - * Parse ExtendedData from KML. Limited support for schemas/datatypes. │ │ │ │ │ - * See http://code.google.com/apis/kml/documentation/kmlreference.html#extendeddata │ │ │ │ │ - * for more information on extendeddata. │ │ │ │ │ + * Method: parsePointGeometry │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {<DOMElement>} An element to parse <COORDS> or <POINT> arcxml data from. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.LinearRing>} A linear ring represented by the node's points. │ │ │ │ │ */ │ │ │ │ │ - parseExtendedData: function(node) { │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - var i, len, data, key; │ │ │ │ │ - var dataNodes = node.getElementsByTagName("Data"); │ │ │ │ │ - for (i = 0, len = dataNodes.length; i < len; i++) { │ │ │ │ │ - data = dataNodes[i]; │ │ │ │ │ - key = data.getAttribute("name"); │ │ │ │ │ - var ed = {}; │ │ │ │ │ - var valueNode = data.getElementsByTagName("value"); │ │ │ │ │ - if (valueNode.length) { │ │ │ │ │ - ed['value'] = this.getChildValue(valueNode[0]); │ │ │ │ │ + parsePointGeometry: function(node) { │ │ │ │ │ + var ringPoints = []; │ │ │ │ │ + var coords = node.getElementsByTagName("COORDS"); │ │ │ │ │ + │ │ │ │ │ + if (coords.length > 0) { │ │ │ │ │ + // if coords is present, it's the only coords item │ │ │ │ │ + var coordArr = this.getChildValue(coords[0]); │ │ │ │ │ + coordArr = coordArr.split(/;/); │ │ │ │ │ + for (var cn = 0; cn < coordArr.length; cn++) { │ │ │ │ │ + var coordItems = coordArr[cn].split(/ /); │ │ │ │ │ + ringPoints.push(new OpenLayers.Geometry.Point(coordItems[0], coordItems[1])); │ │ │ │ │ } │ │ │ │ │ - if (this.kvpAttributes) { │ │ │ │ │ - attributes[key] = ed['value']; │ │ │ │ │ - } else { │ │ │ │ │ - var nameNode = data.getElementsByTagName("displayName"); │ │ │ │ │ - if (nameNode.length) { │ │ │ │ │ - ed['displayName'] = this.getChildValue(nameNode[0]); │ │ │ │ │ + coords = null; │ │ │ │ │ + } else { │ │ │ │ │ + var point = node.getElementsByTagName("POINT"); │ │ │ │ │ + if (point.length > 0) { │ │ │ │ │ + for (var pn = 0; pn < point.length; pn++) { │ │ │ │ │ + ringPoints.push( │ │ │ │ │ + new OpenLayers.Geometry.Point( │ │ │ │ │ + parseFloat(point[pn].getAttribute("x")), │ │ │ │ │ + parseFloat(point[pn].getAttribute("y")) │ │ │ │ │ + ) │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - attributes[key] = ed; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var simpleDataNodes = node.getElementsByTagName("SimpleData"); │ │ │ │ │ - for (i = 0, len = simpleDataNodes.length; i < len; i++) { │ │ │ │ │ - var ed = {}; │ │ │ │ │ - data = simpleDataNodes[i]; │ │ │ │ │ - key = data.getAttribute("name"); │ │ │ │ │ - ed['value'] = this.getChildValue(data); │ │ │ │ │ - if (this.kvpAttributes) { │ │ │ │ │ - attributes[key] = ed['value']; │ │ │ │ │ - } else { │ │ │ │ │ - ed['displayName'] = key; │ │ │ │ │ - attributes[key] = ed; │ │ │ │ │ } │ │ │ │ │ + point = null; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return attributes; │ │ │ │ │ + return new OpenLayers.Geometry.LinearRing(ringPoints); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseProperty │ │ │ │ │ - * Convenience method to find a node and return its value │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xmlNode - {<DOMElement>} │ │ │ │ │ - * namespace - {String} namespace of the node to find │ │ │ │ │ - * tagName - {String} name of the property to parse │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The value for the requested property (defaults to null) │ │ │ │ │ - */ │ │ │ │ │ - parseProperty: function(xmlNode, namespace, tagName) { │ │ │ │ │ - var value; │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(xmlNode, namespace, tagName); │ │ │ │ │ - try { │ │ │ │ │ - value = OpenLayers.Util.getXmlNodeValue(nodeList[0]); │ │ │ │ │ - } catch (e) { │ │ │ │ │ - value = null; │ │ │ │ │ - } │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.ArcXML" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ - return value; │ │ │ │ │ +OpenLayers.Format.ArcXML.Request = OpenLayers.Class({ │ │ │ │ │ + initialize: function(params) { │ │ │ │ │ + var defaults = { │ │ │ │ │ + get_image: { │ │ │ │ │ + properties: { │ │ │ │ │ + background: null, │ │ │ │ │ + /*{ │ │ │ │ │ + color: { r:255, g:255, b:255 }, │ │ │ │ │ + transcolor: null │ │ │ │ │ + },*/ │ │ │ │ │ + draw: true, │ │ │ │ │ + envelope: { │ │ │ │ │ + minx: 0, │ │ │ │ │ + miny: 0, │ │ │ │ │ + maxx: 0, │ │ │ │ │ + maxy: 0 │ │ │ │ │ + }, │ │ │ │ │ + featurecoordsys: { │ │ │ │ │ + id: 0, │ │ │ │ │ + string: "", │ │ │ │ │ + datumtransformid: 0, │ │ │ │ │ + datumtransformstring: "" │ │ │ │ │ + }, │ │ │ │ │ + filtercoordsys: { │ │ │ │ │ + id: 0, │ │ │ │ │ + string: "", │ │ │ │ │ + datumtransformid: 0, │ │ │ │ │ + datumtransformstring: "" │ │ │ │ │ + }, │ │ │ │ │ + imagesize: { │ │ │ │ │ + height: 0, │ │ │ │ │ + width: 0, │ │ │ │ │ + dpi: 96, │ │ │ │ │ + printheight: 0, │ │ │ │ │ + printwidth: 0, │ │ │ │ │ + scalesymbols: false │ │ │ │ │ + }, │ │ │ │ │ + layerlist: [], │ │ │ │ │ + /* no support for legends */ │ │ │ │ │ + output: { │ │ │ │ │ + baseurl: "", │ │ │ │ │ + legendbaseurl: "", │ │ │ │ │ + legendname: "", │ │ │ │ │ + legendpath: "", │ │ │ │ │ + legendurl: "", │ │ │ │ │ + name: "", │ │ │ │ │ + path: "", │ │ │ │ │ + type: "jpg", │ │ │ │ │ + url: "" │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + get_feature: { │ │ │ │ │ + layer: "", │ │ │ │ │ + query: { │ │ │ │ │ + isspatial: false, │ │ │ │ │ + featurecoordsys: { │ │ │ │ │ + id: 0, │ │ │ │ │ + string: "", │ │ │ │ │ + datumtransformid: 0, │ │ │ │ │ + datumtransformstring: "" │ │ │ │ │ + }, │ │ │ │ │ + filtercoordsys: { │ │ │ │ │ + id: 0, │ │ │ │ │ + string: "", │ │ │ │ │ + datumtransformid: 0, │ │ │ │ │ + datumtransformstring: "" │ │ │ │ │ + }, │ │ │ │ │ + buffer: 0, │ │ │ │ │ + where: "", │ │ │ │ │ + spatialfilter: { │ │ │ │ │ + relation: "envelope_intersection", │ │ │ │ │ + envelope: null │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + environment: { │ │ │ │ │ + separators: { │ │ │ │ │ + cs: " ", │ │ │ │ │ + ts: ";" │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + layer: [], │ │ │ │ │ + workspaces: [] │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Util.extend(this, defaults); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.ArcXML.Request" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Format.ArcXML.Response = OpenLayers.Class({ │ │ │ │ │ + initialize: function(params) { │ │ │ │ │ + var defaults = { │ │ │ │ │ + image: { │ │ │ │ │ + envelope: null, │ │ │ │ │ + output: '' │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + features: { │ │ │ │ │ + featurecount: 0, │ │ │ │ │ + envelope: null, │ │ │ │ │ + feature: [] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + error: '' │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Util.extend(this, defaults); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.ArcXML.Response" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/Text.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.Text │ │ │ │ │ + * Read Text format. Create a new instance with the <OpenLayers.Format.Text> │ │ │ │ │ + * constructor. This reads text which is formatted like CSV text, using │ │ │ │ │ + * tabs as the seperator by default. It provides parsing of data originally │ │ │ │ │ + * used in the MapViewerService, described on the wiki. This Format is used │ │ │ │ │ + * by the <OpenLayers.Layer.Text> class. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.Text = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Accept Feature Collection, and return a string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A KML string. │ │ │ │ │ + * APIProperty: defaultStyle │ │ │ │ │ + * defaultStyle allows one to control the default styling of the features. │ │ │ │ │ + * It should be a symbolizer hash. By default, this is set to match the │ │ │ │ │ + * Layer.Text behavior, which is to use the default OpenLayers Icon. │ │ │ │ │ */ │ │ │ │ │ - write: function(features) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "kml"); │ │ │ │ │ - var folder = this.createFolderXML(); │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - folder.appendChild(this.createPlacemarkXML(features[i])); │ │ │ │ │ - } │ │ │ │ │ - kml.appendChild(folder); │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [kml]); │ │ │ │ │ - }, │ │ │ │ │ + defaultStyle: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createFolderXML │ │ │ │ │ - * Creates and returns a KML folder node │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * APIProperty: extractStyles │ │ │ │ │ + * set to true to extract styles from the TSV files, using information │ │ │ │ │ + * from the image or icon, iconSize and iconOffset fields. This will result │ │ │ │ │ + * in features with a symbolizer (style) property set, using the │ │ │ │ │ + * default symbolizer specified in <defaultStyle>. Set to false if you │ │ │ │ │ + * wish to use a styleMap or OpenLayers.Style options to style your │ │ │ │ │ + * layer instead. │ │ │ │ │ */ │ │ │ │ │ - createFolderXML: function() { │ │ │ │ │ - // Folder │ │ │ │ │ - var folder = this.createElementNS(this.kmlns, "Folder"); │ │ │ │ │ + extractStyles: true, │ │ │ │ │ │ │ │ │ │ - // Folder name │ │ │ │ │ - if (this.foldersName) { │ │ │ │ │ - var folderName = this.createElementNS(this.kmlns, "name"); │ │ │ │ │ - var folderNameText = this.createTextNode(this.foldersName); │ │ │ │ │ - folderName.appendChild(folderNameText); │ │ │ │ │ - folder.appendChild(folderName); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.Text │ │ │ │ │ + * Create a new parser for TSV Text. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ │ │ │ │ │ - // Folder description │ │ │ │ │ - if (this.foldersDesc) { │ │ │ │ │ - var folderDesc = this.createElementNS(this.kmlns, "description"); │ │ │ │ │ - var folderDescText = this.createTextNode(this.foldersDesc); │ │ │ │ │ - folderDesc.appendChild(folderDescText); │ │ │ │ │ - folder.appendChild(folderDesc); │ │ │ │ │ + if (options.extractStyles !== false) { │ │ │ │ │ + options.defaultStyle = { │ │ │ │ │ + 'externalGraphic': OpenLayers.Util.getImageLocation("marker.png"), │ │ │ │ │ + 'graphicWidth': 21, │ │ │ │ │ + 'graphicHeight': 25, │ │ │ │ │ + 'graphicXOffset': -10.5, │ │ │ │ │ + 'graphicYOffset': -12.5 │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return folder; │ │ │ │ │ + OpenLayers.Format.prototype.initialize.apply(this, [options]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createPlacemarkXML │ │ │ │ │ - * Creates and returns a KML placemark node representing the given feature. │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Return a list of features from a Tab Seperated Values text string. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * │ │ │ │ │ + * text - {String} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ */ │ │ │ │ │ - createPlacemarkXML: function(feature) { │ │ │ │ │ - // Placemark name │ │ │ │ │ - var placemarkName = this.createElementNS(this.kmlns, "name"); │ │ │ │ │ - var label = (feature.style && feature.style.label) ? feature.style.label : feature.id; │ │ │ │ │ - var name = feature.attributes.name || label; │ │ │ │ │ - placemarkName.appendChild(this.createTextNode(name)); │ │ │ │ │ + read: function(text) { │ │ │ │ │ + var lines = text.split('\n'); │ │ │ │ │ + var columns; │ │ │ │ │ + var features = []; │ │ │ │ │ + // length - 1 to allow for trailing new line │ │ │ │ │ + for (var lcv = 0; lcv < (lines.length - 1); lcv++) { │ │ │ │ │ + var currLine = lines[lcv].replace(/^\s*/, '').replace(/\s*$/, ''); │ │ │ │ │ │ │ │ │ │ - // Placemark description │ │ │ │ │ - var placemarkDesc = this.createElementNS(this.kmlns, "description"); │ │ │ │ │ - var desc = feature.attributes.description || this.placemarksDesc; │ │ │ │ │ - placemarkDesc.appendChild(this.createTextNode(desc)); │ │ │ │ │ + if (currLine.charAt(0) != '#') { │ │ │ │ │ + /* not a comment */ │ │ │ │ │ │ │ │ │ │ - // Placemark │ │ │ │ │ - var placemarkNode = this.createElementNS(this.kmlns, "Placemark"); │ │ │ │ │ - if (feature.fid != null) { │ │ │ │ │ - placemarkNode.setAttribute("id", feature.fid); │ │ │ │ │ + if (!columns) { │ │ │ │ │ + //First line is columns │ │ │ │ │ + columns = currLine.split('\t'); │ │ │ │ │ + } else { │ │ │ │ │ + var vals = currLine.split('\t'); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point(0, 0); │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + var style = this.defaultStyle ? │ │ │ │ │ + OpenLayers.Util.applyDefaults({}, this.defaultStyle) : │ │ │ │ │ + null; │ │ │ │ │ + var icon, iconSize, iconOffset, overflow; │ │ │ │ │ + var set = false; │ │ │ │ │ + for (var valIndex = 0; valIndex < vals.length; valIndex++) { │ │ │ │ │ + if (vals[valIndex]) { │ │ │ │ │ + if (columns[valIndex] == 'point') { │ │ │ │ │ + var coords = vals[valIndex].split(','); │ │ │ │ │ + geometry.y = parseFloat(coords[0]); │ │ │ │ │ + geometry.x = parseFloat(coords[1]); │ │ │ │ │ + set = true; │ │ │ │ │ + } else if (columns[valIndex] == 'lat') { │ │ │ │ │ + geometry.y = parseFloat(vals[valIndex]); │ │ │ │ │ + set = true; │ │ │ │ │ + } else if (columns[valIndex] == 'lon') { │ │ │ │ │ + geometry.x = parseFloat(vals[valIndex]); │ │ │ │ │ + set = true; │ │ │ │ │ + } else if (columns[valIndex] == 'title') │ │ │ │ │ + attributes['title'] = vals[valIndex]; │ │ │ │ │ + else if (columns[valIndex] == 'image' || │ │ │ │ │ + columns[valIndex] == 'icon' && style) { │ │ │ │ │ + style['externalGraphic'] = vals[valIndex]; │ │ │ │ │ + } else if (columns[valIndex] == 'iconSize' && style) { │ │ │ │ │ + var size = vals[valIndex].split(','); │ │ │ │ │ + style['graphicWidth'] = parseFloat(size[0]); │ │ │ │ │ + style['graphicHeight'] = parseFloat(size[1]); │ │ │ │ │ + } else if (columns[valIndex] == 'iconOffset' && style) { │ │ │ │ │ + var offset = vals[valIndex].split(','); │ │ │ │ │ + style['graphicXOffset'] = parseFloat(offset[0]); │ │ │ │ │ + style['graphicYOffset'] = parseFloat(offset[1]); │ │ │ │ │ + } else if (columns[valIndex] == 'description') { │ │ │ │ │ + attributes['description'] = vals[valIndex]; │ │ │ │ │ + } else if (columns[valIndex] == 'overflow') { │ │ │ │ │ + attributes['overflow'] = vals[valIndex]; │ │ │ │ │ + } else { │ │ │ │ │ + // For StyleMap filtering, allow additional │ │ │ │ │ + // columns to be stored as attributes. │ │ │ │ │ + attributes[columns[valIndex]] = vals[valIndex]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (set) { │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry.transform(this.externalProjection, │ │ │ │ │ + this.internalProjection); │ │ │ │ │ + } │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry, attributes, style); │ │ │ │ │ + features.push(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - placemarkNode.appendChild(placemarkName); │ │ │ │ │ - placemarkNode.appendChild(placemarkDesc); │ │ │ │ │ + return features; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Geometry node (Point, LineString, etc. nodes) │ │ │ │ │ - var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ - placemarkNode.appendChild(geometryNode); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.Text" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/Context.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - // output attributes as extendedData │ │ │ │ │ - if (feature.attributes) { │ │ │ │ │ - var edNode = this.buildExtendedData(feature.attributes); │ │ │ │ │ - if (edNode) { │ │ │ │ │ - placemarkNode.appendChild(edNode); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - return placemarkNode; │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.Context │ │ │ │ │ + * Base class for both Format.WMC and Format.OWSContext │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.Context = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildGeometryNode │ │ │ │ │ - * Builds and returns a KML geometry node with the given geometry. │ │ │ │ │ - * │ │ │ │ │ + * Property: layerOptions │ │ │ │ │ + * {Object} Default options for layers created by the parser. These │ │ │ │ │ + * options are overridden by the options which are read from the │ │ │ │ │ + * capabilities document. │ │ │ │ │ + */ │ │ │ │ │ + layerOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: layerParams │ │ │ │ │ + * {Object} Default parameters for layers created by the parser. This │ │ │ │ │ + * can be used e.g. to override DEFAULT_PARAMS for │ │ │ │ │ + * OpenLayers.Layer.WMS. │ │ │ │ │ + */ │ │ │ │ │ + layerParams: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.Context │ │ │ │ │ + * Create a new parser for Context documents. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read Context data from a string, and return an object with map │ │ │ │ │ + * properties and a list of layers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * options - {Object} The options object must contain a map property. If │ │ │ │ │ + * the map property is a string, it must be the id of a dom element │ │ │ │ │ + * where the new map will be placed. If the map property is an │ │ │ │ │ + * <OpenLayers.Map>, the layers from the context document will be added │ │ │ │ │ + * to the map. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * {<OpenLayers.Map>} A map based on the context. │ │ │ │ │ */ │ │ │ │ │ - buildGeometryNode: function(geometry) { │ │ │ │ │ - var className = geometry.CLASS_NAME; │ │ │ │ │ - var type = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ - var builder = this.buildGeometry[type.toLowerCase()]; │ │ │ │ │ - var node = null; │ │ │ │ │ - if (builder) { │ │ │ │ │ - node = builder.apply(this, [geometry]); │ │ │ │ │ + read: function(data, options) { │ │ │ │ │ + var context = OpenLayers.Format.XML.VersionedOGC.prototype.read.apply(this, │ │ │ │ │ + arguments); │ │ │ │ │ + var map; │ │ │ │ │ + if (options && options.map) { │ │ │ │ │ + this.context = context; │ │ │ │ │ + if (options.map instanceof OpenLayers.Map) { │ │ │ │ │ + map = this.mergeContextToMap(context, options.map); │ │ │ │ │ + } else { │ │ │ │ │ + var mapOptions = options.map; │ │ │ │ │ + if (OpenLayers.Util.isElement(mapOptions) || │ │ │ │ │ + typeof mapOptions == "string") { │ │ │ │ │ + // we assume mapOptions references a div │ │ │ │ │ + // element │ │ │ │ │ + mapOptions = { │ │ │ │ │ + div: mapOptions │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + map = this.contextToMap(context, mapOptions); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // not documented as part of the API, provided as a non-API option │ │ │ │ │ + map = context; │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ + return map; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: buildGeometry │ │ │ │ │ - * Object containing methods to do the actual geometry node building │ │ │ │ │ - * based on geometry type. │ │ │ │ │ + * Method: getLayerFromContext │ │ │ │ │ + * Create a WMS layer from a layerContext object. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layerContext - {Object} An object representing a WMS layer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.WMS>} A WMS layer. │ │ │ │ │ */ │ │ │ │ │ - buildGeometry: { │ │ │ │ │ - // TBD: Anybody care about namespace aliases here (these nodes have │ │ │ │ │ - // no prefixes)? │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.point │ │ │ │ │ - * Given an OpenLayers point geometry, create a KML point. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Point>} A point geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML point node. │ │ │ │ │ - */ │ │ │ │ │ - point: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "Point"); │ │ │ │ │ - kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - return kml; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.multipoint │ │ │ │ │ - * Given an OpenLayers multipoint geometry, create a KML │ │ │ │ │ - * GeometryCollection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Point>} A multipoint geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ - */ │ │ │ │ │ - multipoint: function(geometry) { │ │ │ │ │ - return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.linestring │ │ │ │ │ - * Given an OpenLayers linestring geometry, create a KML linestring. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.LineString>} A linestring geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML linestring node. │ │ │ │ │ - */ │ │ │ │ │ - linestring: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "LineString"); │ │ │ │ │ - kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - return kml; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.multilinestring │ │ │ │ │ - * Given an OpenLayers multilinestring geometry, create a KML │ │ │ │ │ - * GeometryCollection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Point>} A multilinestring geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ - */ │ │ │ │ │ - multilinestring: function(geometry) { │ │ │ │ │ - return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.linearring │ │ │ │ │ - * Given an OpenLayers linearring geometry, create a KML linearring. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.LinearRing>} A linearring geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML linearring node. │ │ │ │ │ - */ │ │ │ │ │ - linearring: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "LinearRing"); │ │ │ │ │ - kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - return kml; │ │ │ │ │ - }, │ │ │ │ │ + getLayerFromContext: function(layerContext) { │ │ │ │ │ + var i, len; │ │ │ │ │ + // fill initial options object from layerContext │ │ │ │ │ + var options = { │ │ │ │ │ + queryable: layerContext.queryable, //keep queryable for api compatibility │ │ │ │ │ + visibility: layerContext.visibility, │ │ │ │ │ + maxExtent: layerContext.maxExtent, │ │ │ │ │ + metadata: OpenLayers.Util.applyDefaults(layerContext.metadata, { │ │ │ │ │ + styles: layerContext.styles, │ │ │ │ │ + formats: layerContext.formats, │ │ │ │ │ + "abstract": layerContext["abstract"], │ │ │ │ │ + dataURL: layerContext.dataURL │ │ │ │ │ + }), │ │ │ │ │ + numZoomLevels: layerContext.numZoomLevels, │ │ │ │ │ + units: layerContext.units, │ │ │ │ │ + isBaseLayer: layerContext.isBaseLayer, │ │ │ │ │ + opacity: layerContext.opacity, │ │ │ │ │ + displayInLayerSwitcher: layerContext.displayInLayerSwitcher, │ │ │ │ │ + singleTile: layerContext.singleTile, │ │ │ │ │ + tileSize: (layerContext.tileSize) ? │ │ │ │ │ + new OpenLayers.Size( │ │ │ │ │ + layerContext.tileSize.width, │ │ │ │ │ + layerContext.tileSize.height │ │ │ │ │ + ) : undefined, │ │ │ │ │ + minScale: layerContext.minScale || layerContext.maxScaleDenominator, │ │ │ │ │ + maxScale: layerContext.maxScale || layerContext.minScaleDenominator, │ │ │ │ │ + srs: layerContext.srs, │ │ │ │ │ + dimensions: layerContext.dimensions, │ │ │ │ │ + metadataURL: layerContext.metadataURL │ │ │ │ │ + }; │ │ │ │ │ + if (this.layerOptions) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.layerOptions); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.polygon │ │ │ │ │ - * Given an OpenLayers polygon geometry, create a KML polygon. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Polygon>} A polygon geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML polygon node. │ │ │ │ │ - */ │ │ │ │ │ - polygon: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "Polygon"); │ │ │ │ │ - var rings = geometry.components; │ │ │ │ │ - var ringMember, ringGeom, type; │ │ │ │ │ - for (var i = 0, len = rings.length; i < len; ++i) { │ │ │ │ │ - type = (i == 0) ? "outerBoundaryIs" : "innerBoundaryIs"; │ │ │ │ │ - ringMember = this.createElementNS(this.kmlns, type); │ │ │ │ │ - ringGeom = this.buildGeometry.linearring.apply(this, │ │ │ │ │ - [rings[i]]); │ │ │ │ │ - ringMember.appendChild(ringGeom); │ │ │ │ │ - kml.appendChild(ringMember); │ │ │ │ │ + var params = { │ │ │ │ │ + layers: layerContext.name, │ │ │ │ │ + transparent: layerContext.transparent, │ │ │ │ │ + version: layerContext.version │ │ │ │ │ + }; │ │ │ │ │ + if (layerContext.formats && layerContext.formats.length > 0) { │ │ │ │ │ + // set default value for params if current attribute is not positionned │ │ │ │ │ + params.format = layerContext.formats[0].value; │ │ │ │ │ + for (i = 0, len = layerContext.formats.length; i < len; i++) { │ │ │ │ │ + var format = layerContext.formats[i]; │ │ │ │ │ + if (format.current == true) { │ │ │ │ │ + params.format = format.value; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return kml; │ │ │ │ │ - }, │ │ │ │ │ + } │ │ │ │ │ + if (layerContext.styles && layerContext.styles.length > 0) { │ │ │ │ │ + for (i = 0, len = layerContext.styles.length; i < len; i++) { │ │ │ │ │ + var style = layerContext.styles[i]; │ │ │ │ │ + if (style.current == true) { │ │ │ │ │ + // three style types to consider │ │ │ │ │ + // 1) linked SLD │ │ │ │ │ + // 2) inline SLD │ │ │ │ │ + // 3) named style │ │ │ │ │ + if (style.href) { │ │ │ │ │ + params.sld = style.href; │ │ │ │ │ + } else if (style.body) { │ │ │ │ │ + params.sld_body = style.body; │ │ │ │ │ + } else { │ │ │ │ │ + params.styles = style.name; │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.layerParams) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(params, this.layerParams); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.multipolygon │ │ │ │ │ - * Given an OpenLayers multipolygon geometry, create a KML │ │ │ │ │ - * GeometryCollection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Point>} A multipolygon geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ - */ │ │ │ │ │ - multipolygon: function(geometry) { │ │ │ │ │ - return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ - }, │ │ │ │ │ + var layer = null; │ │ │ │ │ + var service = layerContext.service; │ │ │ │ │ + if (service == OpenLayers.Format.Context.serviceTypes.WFS) { │ │ │ │ │ + options.strategies = [new OpenLayers.Strategy.BBOX()]; │ │ │ │ │ + options.protocol = new OpenLayers.Protocol.WFS({ │ │ │ │ │ + url: layerContext.url, │ │ │ │ │ + // since we do not know featureNS, let the protocol │ │ │ │ │ + // determine it automagically using featurePrefix │ │ │ │ │ + featurePrefix: layerContext.name.split(":")[0], │ │ │ │ │ + featureType: layerContext.name.split(":").pop() │ │ │ │ │ + }); │ │ │ │ │ + layer = new OpenLayers.Layer.Vector( │ │ │ │ │ + layerContext.title || layerContext.name, │ │ │ │ │ + options │ │ │ │ │ + ); │ │ │ │ │ + } else if (service == OpenLayers.Format.Context.serviceTypes.KML) { │ │ │ │ │ + // use a vector layer with an HTTP Protcol and a Fixed strategy │ │ │ │ │ + options.strategies = [new OpenLayers.Strategy.Fixed()]; │ │ │ │ │ + options.protocol = new OpenLayers.Protocol.HTTP({ │ │ │ │ │ + url: layerContext.url, │ │ │ │ │ + format: new OpenLayers.Format.KML() │ │ │ │ │ + }); │ │ │ │ │ + layer = new OpenLayers.Layer.Vector( │ │ │ │ │ + layerContext.title || layerContext.name, │ │ │ │ │ + options │ │ │ │ │ + ); │ │ │ │ │ + } else if (service == OpenLayers.Format.Context.serviceTypes.GML) { │ │ │ │ │ + // use a vector layer with a HTTP Protocol and a Fixed strategy │ │ │ │ │ + options.strategies = [new OpenLayers.Strategy.Fixed()]; │ │ │ │ │ + options.protocol = new OpenLayers.Protocol.HTTP({ │ │ │ │ │ + url: layerContext.url, │ │ │ │ │ + format: new OpenLayers.Format.GML() │ │ │ │ │ + }); │ │ │ │ │ + layer = new OpenLayers.Layer.Vector( │ │ │ │ │ + layerContext.title || layerContext.name, │ │ │ │ │ + options │ │ │ │ │ + ); │ │ │ │ │ + } else if (layerContext.features) { │ │ │ │ │ + // inline GML or KML features │ │ │ │ │ + layer = new OpenLayers.Layer.Vector( │ │ │ │ │ + layerContext.title || layerContext.name, │ │ │ │ │ + options │ │ │ │ │ + ); │ │ │ │ │ + layer.addFeatures(layerContext.features); │ │ │ │ │ + } else if (layerContext.categoryLayer !== true) { │ │ │ │ │ + layer = new OpenLayers.Layer.WMS( │ │ │ │ │ + layerContext.title || layerContext.name, │ │ │ │ │ + layerContext.url, │ │ │ │ │ + params, │ │ │ │ │ + options │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return layer; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.collection │ │ │ │ │ - * Given an OpenLayers geometry collection, create a KML MultiGeometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Collection>} A geometry collection. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML MultiGeometry node. │ │ │ │ │ - */ │ │ │ │ │ - collection: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "MultiGeometry"); │ │ │ │ │ - var child; │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; ++i) { │ │ │ │ │ - child = this.buildGeometryNode.apply(this, │ │ │ │ │ - [geometry.components[i]]); │ │ │ │ │ - if (child) { │ │ │ │ │ - kml.appendChild(child); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: getLayersFromContext │ │ │ │ │ + * Create an array of layers from an array of layerContext objects. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layersContext - {Array(Object)} An array of objects representing layers. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Layer>)} An array of layers. │ │ │ │ │ + */ │ │ │ │ │ + getLayersFromContext: function(layersContext) { │ │ │ │ │ + var layers = []; │ │ │ │ │ + for (var i = 0, len = layersContext.length; i < len; i++) { │ │ │ │ │ + var layer = this.getLayerFromContext(layersContext[i]); │ │ │ │ │ + if (layer !== null) { │ │ │ │ │ + layers.push(layer); │ │ │ │ │ } │ │ │ │ │ - return kml; │ │ │ │ │ } │ │ │ │ │ + return layers; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildCoordinatesNode │ │ │ │ │ - * Builds and returns the KML coordinates node with the given geometry │ │ │ │ │ - * <coordinates>...</coordinates> │ │ │ │ │ - * │ │ │ │ │ + * Method: contextToMap │ │ │ │ │ + * Create a map given a context object. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * context - {Object} The context object. │ │ │ │ │ + * options - {Object} Default map options. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * {<OpenLayers.Map>} A map based on the context object. │ │ │ │ │ */ │ │ │ │ │ - buildCoordinatesNode: function(geometry) { │ │ │ │ │ - var coordinatesNode = this.createElementNS(this.kmlns, "coordinates"); │ │ │ │ │ + contextToMap: function(context, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + maxExtent: context.maxExtent, │ │ │ │ │ + projection: context.projection, │ │ │ │ │ + units: context.units │ │ │ │ │ + }, options); │ │ │ │ │ │ │ │ │ │ - var path; │ │ │ │ │ - var points = geometry.components; │ │ │ │ │ - if (points) { │ │ │ │ │ - // LineString or LinearRing │ │ │ │ │ - var point; │ │ │ │ │ - var numPoints = points.length; │ │ │ │ │ - var parts = new Array(numPoints); │ │ │ │ │ - for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ - point = points[i]; │ │ │ │ │ - parts[i] = this.buildCoordinates(point); │ │ │ │ │ - } │ │ │ │ │ - path = parts.join(" "); │ │ │ │ │ - } else { │ │ │ │ │ - // Point │ │ │ │ │ - path = this.buildCoordinates(geometry); │ │ │ │ │ + if (options.maxExtent) { │ │ │ │ │ + options.maxResolution = │ │ │ │ │ + options.maxExtent.getWidth() / OpenLayers.Map.TILE_WIDTH; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - var txtNode = this.createTextNode(path); │ │ │ │ │ - coordinatesNode.appendChild(txtNode); │ │ │ │ │ + var metadata = { │ │ │ │ │ + contactInformation: context.contactInformation, │ │ │ │ │ + "abstract": context["abstract"], │ │ │ │ │ + keywords: context.keywords, │ │ │ │ │ + logo: context.logo, │ │ │ │ │ + descriptionURL: context.descriptionURL │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - return coordinatesNode; │ │ │ │ │ + options.metadata = metadata; │ │ │ │ │ + │ │ │ │ │ + var map = new OpenLayers.Map(options); │ │ │ │ │ + map.addLayers(this.getLayersFromContext(context.layersContext)); │ │ │ │ │ + map.setCenter( │ │ │ │ │ + context.bounds.getCenterLonLat(), │ │ │ │ │ + map.getZoomForExtent(context.bounds, true) │ │ │ │ │ + ); │ │ │ │ │ + return map; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildCoordinates │ │ │ │ │ + * Method: mergeContextToMap │ │ │ │ │ + * Add layers from a context object to a map. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * context - {Object} The context object. │ │ │ │ │ + * map - {<OpenLayers.Map>} The map. │ │ │ │ │ * │ │ │ │ │ - * Returns │ │ │ │ │ - * {String} a coordinate pair │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Map>} The same map with layers added. │ │ │ │ │ */ │ │ │ │ │ - buildCoordinates: function(point) { │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - point = point.clone(); │ │ │ │ │ - point.transform(this.internalProjection, │ │ │ │ │ - this.externalProjection); │ │ │ │ │ - } │ │ │ │ │ - return point.x + "," + point.y; │ │ │ │ │ + mergeContextToMap: function(context, map) { │ │ │ │ │ + map.addLayers(this.getLayersFromContext(context.layersContext)); │ │ │ │ │ + return map; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildExtendedData │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Write a context document given a map. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * attributes - {Object} │ │ │ │ │ + * obj - {<OpenLayers.Map> | Object} A map or context object. │ │ │ │ │ + * options - {Object} Optional configuration object. │ │ │ │ │ * │ │ │ │ │ - * Returns │ │ │ │ │ - * {DOMElement} A KML ExtendedData node or {null} if no attributes. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A context document string. │ │ │ │ │ */ │ │ │ │ │ - buildExtendedData: function(attributes) { │ │ │ │ │ - var extendedData = this.createElementNS(this.kmlns, "ExtendedData"); │ │ │ │ │ - for (var attributeName in attributes) { │ │ │ │ │ - // empty, name, description, styleUrl attributes ignored │ │ │ │ │ - if (attributes[attributeName] && attributeName != "name" && attributeName != "description" && attributeName != "styleUrl") { │ │ │ │ │ - var data = this.createElementNS(this.kmlns, "Data"); │ │ │ │ │ - data.setAttribute("name", attributeName); │ │ │ │ │ - var value = this.createElementNS(this.kmlns, "value"); │ │ │ │ │ - if (typeof attributes[attributeName] == "object") { │ │ │ │ │ - // cater for object attributes with 'value' properties │ │ │ │ │ - // other object properties will output an empty node │ │ │ │ │ - if (attributes[attributeName].value) { │ │ │ │ │ - value.appendChild(this.createTextNode(attributes[attributeName].value)); │ │ │ │ │ - } │ │ │ │ │ - if (attributes[attributeName].displayName) { │ │ │ │ │ - var displayName = this.createElementNS(this.kmlns, "displayName"); │ │ │ │ │ - // displayName always written as CDATA │ │ │ │ │ - displayName.appendChild(this.getXMLDoc().createCDATASection(attributes[attributeName].displayName)); │ │ │ │ │ - data.appendChild(displayName); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - value.appendChild(this.createTextNode(attributes[attributeName])); │ │ │ │ │ - } │ │ │ │ │ - data.appendChild(value); │ │ │ │ │ - extendedData.appendChild(data); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.isSimpleContent(extendedData)) { │ │ │ │ │ - return null; │ │ │ │ │ - } else { │ │ │ │ │ - return extendedData; │ │ │ │ │ - } │ │ │ │ │ + write: function(obj, options) { │ │ │ │ │ + obj = this.toContext(obj); │ │ │ │ │ + return OpenLayers.Format.XML.VersionedOGC.prototype.write.apply(this, │ │ │ │ │ + arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.KML" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.Context" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Format.Context.serviceTypes │ │ │ │ │ + * Enumeration for service types │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.Context.serviceTypes = { │ │ │ │ │ + "WMS": "urn:ogc:serviceType:WMS", │ │ │ │ │ + "WFS": "urn:ogc:serviceType:WFS", │ │ │ │ │ + "WCS": "urn:ogc:serviceType:WCS", │ │ │ │ │ + "GML": "urn:ogc:serviceType:GML", │ │ │ │ │ + "SLD": "urn:ogc:serviceType:SLD", │ │ │ │ │ + "FES": "urn:ogc:serviceType:FES", │ │ │ │ │ + "KML": "urn:ogc:serviceType:KML" │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WCSCapabilities.js │ │ │ │ │ + OpenLayers/Format/OWSContext.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + * @requires OpenLayers/Format/Context.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WCSCapabilities │ │ │ │ │ - * Read WCS Capabilities. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Format.OWSContext │ │ │ │ │ + * Read and write OWS Context documents. OWS Context documents are a │ │ │ │ │ + * preliminary OGC (Open Geospatial Consortium) standard for storing the │ │ │ │ │ + * state of a web mapping application. In a way it is the successor to │ │ │ │ │ + * Web Map Context (WMC), since it is more generic and more types of layers │ │ │ │ │ + * can be stored. Also, nesting of layers is supported since version 0.3.1. │ │ │ │ │ + * For more information see: http://www.ogcnetwork.net/context │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + * - <OpenLayers.Format.Context> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WCSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ +OpenLayers.Format.OWSContext = OpenLayers.Class(OpenLayers.Format.Context, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.1.0". │ │ │ │ │ + * {String} Version number to assume if none found. Default is "0.3.1". │ │ │ │ │ */ │ │ │ │ │ - defaultVersion: "1.1.0", │ │ │ │ │ + defaultVersion: "0.3.1", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WCSCapabilities │ │ │ │ │ - * Create a new parser for WCS capabilities. │ │ │ │ │ + * Constructor: OpenLayers.Format.OWSContext │ │ │ │ │ + * Create a new parser for OWS Context documents. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * options - {Object} An optional object whose properties will be set on │ │ │ │ │ * this instance. │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return a list of coverages. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * Method: getVersion │ │ │ │ │ + * Returns the version to use. Subclasses can override this function │ │ │ │ │ + * if a different version detection is needed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * root - {DOMElement} │ │ │ │ │ + * options - {Object} Optional configuration object. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array} List of named coverages. │ │ │ │ │ + * {String} The version to use. │ │ │ │ │ */ │ │ │ │ │ + getVersion: function(root, options) { │ │ │ │ │ + var version = OpenLayers.Format.XML.VersionedOGC.prototype.getVersion.apply( │ │ │ │ │ + this, arguments); │ │ │ │ │ + // 0.3.1 is backwards compatible with 0.3.0 │ │ │ │ │ + if (version === "0.3.0") { │ │ │ │ │ + version = this.defaultVersion; │ │ │ │ │ + } │ │ │ │ │ + return version; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WCSCapabilities" │ │ │ │ │ + /** │ │ │ │ │ + * Method: toContext │ │ │ │ │ + * Create a context object free from layer given a map or a │ │ │ │ │ + * context object. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {<OpenLayers.Map> | Object} The map or context. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} A context object. │ │ │ │ │ + */ │ │ │ │ │ + toContext: function(obj) { │ │ │ │ │ + var context = {}; │ │ │ │ │ + if (obj.CLASS_NAME == "OpenLayers.Map") { │ │ │ │ │ + context.bounds = obj.getExtent(); │ │ │ │ │ + context.maxExtent = obj.maxExtent; │ │ │ │ │ + context.projection = obj.projection; │ │ │ │ │ + context.size = obj.getSize(); │ │ │ │ │ + context.layers = obj.layers; │ │ │ │ │ + } │ │ │ │ │ + return context; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.OWSContext" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Format/GPX.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ @@ -50043,491 +46810,833 @@ │ │ │ │ │ node.appendChild(desc); │ │ │ │ │ // TBD - deal with remaining (non name/description) attributes. │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.GPX" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/GeoRSS.js │ │ │ │ │ + OpenLayers/Format/WMC.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ + * @requires OpenLayers/Format/Context.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.GeoRSS │ │ │ │ │ - * Read/write GeoRSS parser. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Format.GeoRSS> constructor. │ │ │ │ │ + * Class: OpenLayers.Format.WMC │ │ │ │ │ + * Read and write Web Map Context documents. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * - <OpenLayers.Format.Context> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.GeoRSS = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Format.WMC = OpenLayers.Class(OpenLayers.Format.Context, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: rssns │ │ │ │ │ - * {String} RSS namespace to use. Defaults to │ │ │ │ │ - * "http://backend.userland.com/rss2" │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.1.0". │ │ │ │ │ */ │ │ │ │ │ - rssns: "http://backend.userland.com/rss2", │ │ │ │ │ + defaultVersion: "1.1.0", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: featurens │ │ │ │ │ - * {String} Feature Attributes namespace. Defaults to │ │ │ │ │ - * "http://mapserver.gis.umn.edu/mapserver" │ │ │ │ │ + * Constructor: OpenLayers.Format.WMC │ │ │ │ │ + * Create a new parser for Web Map Context documents. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - featureNS: "http://mapserver.gis.umn.edu/mapserver", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: georssns │ │ │ │ │ - * {String} GeoRSS namespace to use. Defaults to │ │ │ │ │ - * "http://www.georss.org/georss" │ │ │ │ │ + * Method: layerToContext │ │ │ │ │ + * Create a layer context object given a wms layer object. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer.WMS>} The layer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} A layer context object. │ │ │ │ │ */ │ │ │ │ │ - georssns: "http://www.georss.org/georss", │ │ │ │ │ + layerToContext: function(layer) { │ │ │ │ │ + var parser = this.getParser(); │ │ │ │ │ + var layerContext = { │ │ │ │ │ + queryable: layer.queryable, │ │ │ │ │ + visibility: layer.visibility, │ │ │ │ │ + name: layer.params["LAYERS"], │ │ │ │ │ + title: layer.name, │ │ │ │ │ + "abstract": layer.metadata["abstract"], │ │ │ │ │ + dataURL: layer.metadata.dataURL, │ │ │ │ │ + metadataURL: layer.metadataURL, │ │ │ │ │ + server: { │ │ │ │ │ + version: layer.params["VERSION"], │ │ │ │ │ + url: layer.url │ │ │ │ │ + }, │ │ │ │ │ + maxExtent: layer.maxExtent, │ │ │ │ │ + transparent: layer.params["TRANSPARENT"], │ │ │ │ │ + numZoomLevels: layer.numZoomLevels, │ │ │ │ │ + units: layer.units, │ │ │ │ │ + isBaseLayer: layer.isBaseLayer, │ │ │ │ │ + opacity: layer.opacity == 1 ? undefined : layer.opacity, │ │ │ │ │ + displayInLayerSwitcher: layer.displayInLayerSwitcher, │ │ │ │ │ + singleTile: layer.singleTile, │ │ │ │ │ + tileSize: (layer.singleTile || !layer.tileSize) ? │ │ │ │ │ + undefined : { │ │ │ │ │ + width: layer.tileSize.w, │ │ │ │ │ + height: layer.tileSize.h │ │ │ │ │ + }, │ │ │ │ │ + minScale: (layer.options.resolutions || │ │ │ │ │ + layer.options.scales || │ │ │ │ │ + layer.options.maxResolution || │ │ │ │ │ + layer.options.minScale) ? │ │ │ │ │ + layer.minScale : undefined, │ │ │ │ │ + maxScale: (layer.options.resolutions || │ │ │ │ │ + layer.options.scales || │ │ │ │ │ + layer.options.minResolution || │ │ │ │ │ + layer.options.maxScale) ? │ │ │ │ │ + layer.maxScale : undefined, │ │ │ │ │ + formats: [], │ │ │ │ │ + styles: [], │ │ │ │ │ + srs: layer.srs, │ │ │ │ │ + dimensions: layer.dimensions │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geons │ │ │ │ │ - * {String} W3C Geo namespace to use. Defaults to │ │ │ │ │ - * "http://www.w3.org/2003/01/geo/wgs84_pos#" │ │ │ │ │ - */ │ │ │ │ │ - geons: "http://www.w3.org/2003/01/geo/wgs84_pos#", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: featureTitle │ │ │ │ │ - * {String} Default title for features. Defaults to "Untitled" │ │ │ │ │ - */ │ │ │ │ │ - featureTitle: "Untitled", │ │ │ │ │ + if (layer.metadata.servertitle) { │ │ │ │ │ + layerContext.server.title = layer.metadata.servertitle; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: featureDescription │ │ │ │ │ - * {String} Default description for features. Defaults to "No Description" │ │ │ │ │ - */ │ │ │ │ │ - featureDescription: "No Description", │ │ │ │ │ + if (layer.metadata.formats && layer.metadata.formats.length > 0) { │ │ │ │ │ + for (var i = 0, len = layer.metadata.formats.length; i < len; i++) { │ │ │ │ │ + var format = layer.metadata.formats[i]; │ │ │ │ │ + layerContext.formats.push({ │ │ │ │ │ + value: format.value, │ │ │ │ │ + current: (format.value == layer.params["FORMAT"]) │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + layerContext.formats.push({ │ │ │ │ │ + value: layer.params["FORMAT"], │ │ │ │ │ + current: true │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (layer.metadata.styles && layer.metadata.styles.length > 0) { │ │ │ │ │ + for (var i = 0, len = layer.metadata.styles.length; i < len; i++) { │ │ │ │ │ + var style = layer.metadata.styles[i]; │ │ │ │ │ + if ((style.href == layer.params["SLD"]) || │ │ │ │ │ + (style.body == layer.params["SLD_BODY"]) || │ │ │ │ │ + (style.name == layer.params["STYLES"])) { │ │ │ │ │ + style.current = true; │ │ │ │ │ + } else { │ │ │ │ │ + style.current = false; │ │ │ │ │ + } │ │ │ │ │ + layerContext.styles.push(style); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + layerContext.styles.push({ │ │ │ │ │ + href: layer.params["SLD"], │ │ │ │ │ + body: layer.params["SLD_BODY"], │ │ │ │ │ + name: layer.params["STYLES"] || parser.defaultStyleName, │ │ │ │ │ + title: parser.defaultStyleTitle, │ │ │ │ │ + current: true │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return layerContext; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: gmlParse │ │ │ │ │ - * {Object} GML Format object for parsing features │ │ │ │ │ - * Non-API and only created if necessary │ │ │ │ │ + * Method: toContext │ │ │ │ │ + * Create a context object free from layer given a map or a │ │ │ │ │ + * context object. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {<OpenLayers.Map> | Object} The map or context. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} A context object. │ │ │ │ │ */ │ │ │ │ │ - gmlParser: null, │ │ │ │ │ + toContext: function(obj) { │ │ │ │ │ + var context = {}; │ │ │ │ │ + var layers = obj.layers; │ │ │ │ │ + if (obj.CLASS_NAME == "OpenLayers.Map") { │ │ │ │ │ + var metadata = obj.metadata || {}; │ │ │ │ │ + context.size = obj.getSize(); │ │ │ │ │ + context.bounds = obj.getExtent(); │ │ │ │ │ + context.projection = obj.projection; │ │ │ │ │ + context.title = obj.title; │ │ │ │ │ + context.keywords = metadata.keywords; │ │ │ │ │ + context["abstract"] = metadata["abstract"]; │ │ │ │ │ + context.logo = metadata.logo; │ │ │ │ │ + context.descriptionURL = metadata.descriptionURL; │ │ │ │ │ + context.contactInformation = metadata.contactInformation; │ │ │ │ │ + context.maxExtent = obj.maxExtent; │ │ │ │ │ + } else { │ │ │ │ │ + // copy all obj properties except the "layers" property │ │ │ │ │ + OpenLayers.Util.applyDefaults(context, obj); │ │ │ │ │ + if (context.layers != undefined) { │ │ │ │ │ + delete(context.layers); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (context.layersContext == undefined) { │ │ │ │ │ + context.layersContext = []; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // let's convert layers into layersContext object (if any) │ │ │ │ │ + if (layers != undefined && OpenLayers.Util.isArray(layers)) { │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + var layer = layers[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.WMS) { │ │ │ │ │ + context.layersContext.push(this.layerToContext(layer)); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return context; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMC" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WPSCapabilities.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WPSCapabilities │ │ │ │ │ + * Read WPS Capabilities. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WPSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: xy │ │ │ │ │ - * {Boolean} Order of the GML coordinate: true:(x,y) or false:(y,x) │ │ │ │ │ - * For GeoRSS the default is (y,x), therefore: false │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.0.0". │ │ │ │ │ */ │ │ │ │ │ - xy: false, │ │ │ │ │ + defaultVersion: "1.0.0", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.GeoRSS │ │ │ │ │ - * Create a new parser for GeoRSS. │ │ │ │ │ + * Constructor: OpenLayers.Format.WPSCapabilities │ │ │ │ │ + * Create a new parser for WPS Capabilities. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * options - {Object} An optional object whose properties will be set on │ │ │ │ │ * this instance. │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createGeometryFromItem │ │ │ │ │ - * Return a geometry from a GeoRSS Item. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * item - {DOMElement} A GeoRSS item node. │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return information about │ │ │ │ │ + * the service. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} A geometry representing the node. │ │ │ │ │ + * {Object} Info about the WPS │ │ │ │ │ */ │ │ │ │ │ - createGeometryFromItem: function(item) { │ │ │ │ │ - var point = this.getElementsByTagNameNS(item, this.georssns, "point"); │ │ │ │ │ - var lat = this.getElementsByTagNameNS(item, this.geons, 'lat'); │ │ │ │ │ - var lon = this.getElementsByTagNameNS(item, this.geons, 'long'); │ │ │ │ │ │ │ │ │ │ - var line = this.getElementsByTagNameNS(item, │ │ │ │ │ - this.georssns, │ │ │ │ │ - "line"); │ │ │ │ │ - var polygon = this.getElementsByTagNameNS(item, │ │ │ │ │ - this.georssns, │ │ │ │ │ - "polygon"); │ │ │ │ │ - var where = this.getElementsByTagNameNS(item, │ │ │ │ │ - this.georssns, │ │ │ │ │ - "where"); │ │ │ │ │ - var box = this.getElementsByTagNameNS(item, │ │ │ │ │ - this.georssns, │ │ │ │ │ - "box"); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WPSCapabilities" │ │ │ │ │ │ │ │ │ │ - if (point.length > 0 || (lat.length > 0 && lon.length > 0)) { │ │ │ │ │ - var location; │ │ │ │ │ - if (point.length > 0) { │ │ │ │ │ - location = OpenLayers.String.trim( │ │ │ │ │ - point[0].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ - if (location.length != 2) { │ │ │ │ │ - location = OpenLayers.String.trim( │ │ │ │ │ - point[0].firstChild.nodeValue).split(/\s*,\s*/); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - location = [parseFloat(lat[0].firstChild.nodeValue), │ │ │ │ │ - parseFloat(lon[0].firstChild.nodeValue) │ │ │ │ │ - ]; │ │ │ │ │ - } │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WMSGetFeatureInfo.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point(location[1], location[0]); │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - } else if (line.length > 0) { │ │ │ │ │ - var coords = OpenLayers.String.trim(this.getChildValue(line[0])).split(/\s+/); │ │ │ │ │ - var components = []; │ │ │ │ │ - var point; │ │ │ │ │ - for (var i = 0, len = coords.length; i < len; i += 2) { │ │ │ │ │ - point = new OpenLayers.Geometry.Point(coords[i + 1], coords[i]); │ │ │ │ │ - components.push(point); │ │ │ │ │ - } │ │ │ │ │ - geometry = new OpenLayers.Geometry.LineString(components); │ │ │ │ │ - } else if (polygon.length > 0) { │ │ │ │ │ - var coords = OpenLayers.String.trim(this.getChildValue(polygon[0])).split(/\s+/); │ │ │ │ │ - var components = []; │ │ │ │ │ - var point; │ │ │ │ │ - for (var i = 0, len = coords.length; i < len; i += 2) { │ │ │ │ │ - point = new OpenLayers.Geometry.Point(coords[i + 1], coords[i]); │ │ │ │ │ - components.push(point); │ │ │ │ │ - } │ │ │ │ │ - geometry = new OpenLayers.Geometry.Polygon([new OpenLayers.Geometry.LinearRing(components)]); │ │ │ │ │ - } else if (where.length > 0) { │ │ │ │ │ - if (!this.gmlParser) { │ │ │ │ │ - this.gmlParser = new OpenLayers.Format.GML({ │ │ │ │ │ - 'xy': this.xy │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - var feature = this.gmlParser.parseFeature(where[0]); │ │ │ │ │ - geometry = feature.geometry; │ │ │ │ │ - } else if (box.length > 0) { │ │ │ │ │ - var coords = OpenLayers.String.trim(box[0].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ - var components = []; │ │ │ │ │ - var point; │ │ │ │ │ - if (coords.length > 3) { │ │ │ │ │ - point = new OpenLayers.Geometry.Point(coords[1], coords[0]); │ │ │ │ │ - components.push(point); │ │ │ │ │ - point = new OpenLayers.Geometry.Point(coords[1], coords[2]); │ │ │ │ │ - components.push(point); │ │ │ │ │ - point = new OpenLayers.Geometry.Point(coords[3], coords[2]); │ │ │ │ │ - components.push(point); │ │ │ │ │ - point = new OpenLayers.Geometry.Point(coords[3], coords[0]); │ │ │ │ │ - components.push(point); │ │ │ │ │ - point = new OpenLayers.Geometry.Point(coords[1], coords[0]); │ │ │ │ │ - components.push(point); │ │ │ │ │ - } │ │ │ │ │ - geometry = new OpenLayers.Geometry.Polygon([new OpenLayers.Geometry.LinearRing(components)]); │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (geometry && this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry.transform(this.externalProjection, │ │ │ │ │ - this.internalProjection); │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WMSGetFeatureInfo │ │ │ │ │ + * Class to read GetFeatureInfo responses from Web Mapping Services │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WMSGetFeatureInfo = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ - return geometry; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layerIdentifier │ │ │ │ │ + * {String} All xml nodes containing this search criteria will populate an │ │ │ │ │ + * internal array of layer nodes. │ │ │ │ │ + */ │ │ │ │ │ + layerIdentifier: '_layer', │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: featureIdentifier │ │ │ │ │ + * {String} All xml nodes containing this search criteria will populate an │ │ │ │ │ + * internal array of feature nodes for each layer node found. │ │ │ │ │ + */ │ │ │ │ │ + featureIdentifier: '_feature', │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ + */ │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createFeatureFromItem │ │ │ │ │ - * Return a feature from a GeoRSS Item. │ │ │ │ │ + * Property: gmlFormat │ │ │ │ │ + * {<OpenLayers.Format.GML>} internal GML format for parsing geometries │ │ │ │ │ + * in msGMLOutput │ │ │ │ │ + */ │ │ │ │ │ + gmlFormat: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMSGetFeatureInfo │ │ │ │ │ + * Create a new parser for WMS GetFeatureInfo responses │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * item - {DOMElement} A GeoRSS item node. │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read WMS GetFeatureInfo data from a string, and return an array of features │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A feature representing the item. │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ */ │ │ │ │ │ - createFeatureFromItem: function(item) { │ │ │ │ │ - var geometry = this.createGeometryFromItem(item); │ │ │ │ │ - │ │ │ │ │ - /* Provide defaults for title and description */ │ │ │ │ │ - var title = this._getChildValue(item, "*", "title", this.featureTitle); │ │ │ │ │ - │ │ │ │ │ - /* First try RSS descriptions, then Atom summaries */ │ │ │ │ │ - var description = this._getChildValue( │ │ │ │ │ - item, "*", "description", │ │ │ │ │ - this._getChildValue(item, "*", "content", │ │ │ │ │ - this._getChildValue(item, "*", "summary", this.featureDescription))); │ │ │ │ │ - │ │ │ │ │ - /* If no link URL is found in the first child node, try the │ │ │ │ │ - href attribute */ │ │ │ │ │ - var link = this._getChildValue(item, "*", "link"); │ │ │ │ │ - if (!link) { │ │ │ │ │ - try { │ │ │ │ │ - link = this.getElementsByTagNameNS(item, "*", "link")[0].getAttribute("href"); │ │ │ │ │ - } catch (e) { │ │ │ │ │ - link = null; │ │ │ │ │ + read: function(data) { │ │ │ │ │ + var result; │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + var root = data.documentElement; │ │ │ │ │ + if (root) { │ │ │ │ │ + var scope = this; │ │ │ │ │ + var read = this["read_" + root.nodeName]; │ │ │ │ │ + if (read) { │ │ │ │ │ + result = read.call(this, root); │ │ │ │ │ + } else { │ │ │ │ │ + // fall-back to GML since this is a common output format for WMS │ │ │ │ │ + // GetFeatureInfo responses │ │ │ │ │ + result = new OpenLayers.Format.GML((this.options ? this.options : {})).read(data); │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + result = data; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var id = this._getChildValue(item, "*", "id", null); │ │ │ │ │ - │ │ │ │ │ - var data = { │ │ │ │ │ - "title": title, │ │ │ │ │ - "description": description, │ │ │ │ │ - "link": link │ │ │ │ │ - }; │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry, data); │ │ │ │ │ - feature.fid = id; │ │ │ │ │ - return feature; │ │ │ │ │ + return result; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: _getChildValue │ │ │ │ │ + * Method: read_msGMLOutput │ │ │ │ │ + * Parse msGMLOutput nodes. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * nsuri - {String} Child node namespace uri ("*" for any). │ │ │ │ │ - * name - {String} Child node name. │ │ │ │ │ - * def - {String} Optional string default to return if no child found. │ │ │ │ │ + * data - {DOMElement} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} The value of the first child with the given tag name. Returns │ │ │ │ │ - * default value or empty string if none found. │ │ │ │ │ + * {Array} │ │ │ │ │ */ │ │ │ │ │ - _getChildValue: function(node, nsuri, name, def) { │ │ │ │ │ - var value; │ │ │ │ │ - var eles = this.getElementsByTagNameNS(node, nsuri, name); │ │ │ │ │ - if (eles && eles[0] && eles[0].firstChild && │ │ │ │ │ - eles[0].firstChild.nodeValue) { │ │ │ │ │ - value = this.getChildValue(eles[0]); │ │ │ │ │ - } else { │ │ │ │ │ - value = (def == undefined) ? "" : def; │ │ │ │ │ + read_msGMLOutput: function(data) { │ │ │ │ │ + var response = []; │ │ │ │ │ + var layerNodes = this.getSiblingNodesByTagCriteria(data, │ │ │ │ │ + this.layerIdentifier); │ │ │ │ │ + if (layerNodes) { │ │ │ │ │ + for (var i = 0, len = layerNodes.length; i < len; ++i) { │ │ │ │ │ + var node = layerNodes[i]; │ │ │ │ │ + var layerName = node.nodeName; │ │ │ │ │ + if (node.prefix) { │ │ │ │ │ + layerName = layerName.split(':')[1]; │ │ │ │ │ + } │ │ │ │ │ + var layerName = layerName.replace(this.layerIdentifier, ''); │ │ │ │ │ + var featureNodes = this.getSiblingNodesByTagCriteria(node, │ │ │ │ │ + this.featureIdentifier); │ │ │ │ │ + if (featureNodes) { │ │ │ │ │ + for (var j = 0; j < featureNodes.length; j++) { │ │ │ │ │ + var featureNode = featureNodes[j]; │ │ │ │ │ + var geomInfo = this.parseGeometry(featureNode); │ │ │ │ │ + var attributes = this.parseAttributes(featureNode); │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geomInfo.geometry, │ │ │ │ │ + attributes, null); │ │ │ │ │ + feature.bounds = geomInfo.bounds; │ │ │ │ │ + feature.type = layerName; │ │ │ │ │ + response.push(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return value; │ │ │ │ │ + return response; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Return a list of features from a GeoRSS doc │ │ │ │ │ + * Method: read_FeatureInfoResponse │ │ │ │ │ + * Parse FeatureInfoResponse nodes. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * doc - {Element} │ │ │ │ │ + * data - {DOMElement} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * {Array} │ │ │ │ │ */ │ │ │ │ │ - read: function(doc) { │ │ │ │ │ - if (typeof doc == "string") { │ │ │ │ │ - doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]); │ │ │ │ │ - } │ │ │ │ │ + read_FeatureInfoResponse: function(data) { │ │ │ │ │ + var response = []; │ │ │ │ │ + var featureNodes = this.getElementsByTagNameNS(data, '*', │ │ │ │ │ + 'FIELDS'); │ │ │ │ │ │ │ │ │ │ - /* Try RSS items first, then Atom entries */ │ │ │ │ │ - var itemlist = null; │ │ │ │ │ - itemlist = this.getElementsByTagNameNS(doc, '*', 'item'); │ │ │ │ │ - if (itemlist.length == 0) { │ │ │ │ │ - itemlist = this.getElementsByTagNameNS(doc, '*', 'entry'); │ │ │ │ │ - } │ │ │ │ │ + for (var i = 0, len = featureNodes.length; i < len; i++) { │ │ │ │ │ + var featureNode = featureNodes[i]; │ │ │ │ │ + var geom = null; │ │ │ │ │ │ │ │ │ │ - var numItems = itemlist.length; │ │ │ │ │ - var features = new Array(numItems); │ │ │ │ │ - for (var i = 0; i < numItems; i++) { │ │ │ │ │ - features[i] = this.createFeatureFromItem(itemlist[i]); │ │ │ │ │ + // attributes can be actual attributes on the FIELDS tag, │ │ │ │ │ + // or FIELD children │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + var j; │ │ │ │ │ + var jlen = featureNode.attributes.length; │ │ │ │ │ + if (jlen > 0) { │ │ │ │ │ + for (j = 0; j < jlen; j++) { │ │ │ │ │ + var attribute = featureNode.attributes[j]; │ │ │ │ │ + attributes[attribute.nodeName] = attribute.nodeValue; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + var nodes = featureNode.childNodes; │ │ │ │ │ + for (j = 0, jlen = nodes.length; j < jlen; ++j) { │ │ │ │ │ + var node = nodes[j]; │ │ │ │ │ + if (node.nodeType != 3) { │ │ │ │ │ + attributes[node.getAttribute("name")] = │ │ │ │ │ + node.getAttribute("value"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + response.push( │ │ │ │ │ + new OpenLayers.Feature.Vector(geom, attributes, null) │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - return features; │ │ │ │ │ + return response; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Accept Feature Collection, and return a string. │ │ │ │ │ + * Method: getSiblingNodesByTagCriteria │ │ │ │ │ + * Recursively searches passed xml node and all it's descendant levels for │ │ │ │ │ + * nodes whose tagName contains the passed search string. This returns an │ │ │ │ │ + * array of all sibling nodes which match the criteria from the highest │ │ │ │ │ + * hierarchial level from which a match is found. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} List of features to serialize into a string. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} An xml node │ │ │ │ │ + * criteria - {String} Search string which will match some part of a tagName │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * Array({DOMElement}) An array of sibling xml nodes │ │ │ │ │ */ │ │ │ │ │ - write: function(features) { │ │ │ │ │ - var georss; │ │ │ │ │ - if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ - georss = this.createElementNS(this.rssns, "rss"); │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - georss.appendChild(this.createFeatureXML(features[i])); │ │ │ │ │ + getSiblingNodesByTagCriteria: function(node, criteria) { │ │ │ │ │ + var nodes = []; │ │ │ │ │ + var children, tagName, n, matchNodes, child; │ │ │ │ │ + if (node && node.hasChildNodes()) { │ │ │ │ │ + children = node.childNodes; │ │ │ │ │ + n = children.length; │ │ │ │ │ + │ │ │ │ │ + for (var k = 0; k < n; k++) { │ │ │ │ │ + child = children[k]; │ │ │ │ │ + while (child && child.nodeType != 1) { │ │ │ │ │ + child = child.nextSibling; │ │ │ │ │ + k++; │ │ │ │ │ + } │ │ │ │ │ + tagName = (child ? child.nodeName : ''); │ │ │ │ │ + if (tagName.length > 0 && tagName.indexOf(criteria) > -1) { │ │ │ │ │ + nodes.push(child); │ │ │ │ │ + } else { │ │ │ │ │ + matchNodes = this.getSiblingNodesByTagCriteria( │ │ │ │ │ + child, criteria); │ │ │ │ │ + │ │ │ │ │ + if (matchNodes.length > 0) { │ │ │ │ │ + (nodes.length == 0) ? │ │ │ │ │ + nodes = matchNodes: nodes.push(matchNodes); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - georss = this.createFeatureXML(features); │ │ │ │ │ + │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [georss]); │ │ │ │ │ + return nodes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createFeatureXML │ │ │ │ │ - * Accept an <OpenLayers.Feature.Vector>, and build a geometry for it. │ │ │ │ │ - * │ │ │ │ │ + * Method: parseAttributes │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * node - {<DOMElement>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * {Object} An attributes object. │ │ │ │ │ + * │ │ │ │ │ + * Notes: │ │ │ │ │ + * Assumes that attributes are direct child xml nodes of the passed node │ │ │ │ │ + * and contain only a single text node. │ │ │ │ │ */ │ │ │ │ │ - createFeatureXML: function(feature) { │ │ │ │ │ - var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ - var featureNode = this.createElementNS(this.rssns, "item"); │ │ │ │ │ - var titleNode = this.createElementNS(this.rssns, "title"); │ │ │ │ │ - titleNode.appendChild(this.createTextNode(feature.attributes.title ? feature.attributes.title : "")); │ │ │ │ │ - var descNode = this.createElementNS(this.rssns, "description"); │ │ │ │ │ - descNode.appendChild(this.createTextNode(feature.attributes.description ? feature.attributes.description : "")); │ │ │ │ │ - featureNode.appendChild(titleNode); │ │ │ │ │ - featureNode.appendChild(descNode); │ │ │ │ │ - if (feature.attributes.link) { │ │ │ │ │ - var linkNode = this.createElementNS(this.rssns, "link"); │ │ │ │ │ - linkNode.appendChild(this.createTextNode(feature.attributes.link)); │ │ │ │ │ - featureNode.appendChild(linkNode); │ │ │ │ │ - } │ │ │ │ │ - for (var attr in feature.attributes) { │ │ │ │ │ - if (attr == "link" || attr == "title" || attr == "description") { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - var attrText = this.createTextNode(feature.attributes[attr]); │ │ │ │ │ - var nodename = attr; │ │ │ │ │ - if (attr.search(":") != -1) { │ │ │ │ │ - nodename = attr.split(":")[1]; │ │ │ │ │ + parseAttributes: function(node) { │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + if (node.nodeType == 1) { │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + var n = children.length; │ │ │ │ │ + for (var i = 0; i < n; ++i) { │ │ │ │ │ + var child = children[i]; │ │ │ │ │ + if (child.nodeType == 1) { │ │ │ │ │ + var grandchildren = child.childNodes; │ │ │ │ │ + var name = (child.prefix) ? │ │ │ │ │ + child.nodeName.split(":")[1] : child.nodeName; │ │ │ │ │ + if (grandchildren.length == 0) { │ │ │ │ │ + attributes[name] = null; │ │ │ │ │ + } else if (grandchildren.length == 1) { │ │ │ │ │ + var grandchild = grandchildren[0]; │ │ │ │ │ + if (grandchild.nodeType == 3 || │ │ │ │ │ + grandchild.nodeType == 4) { │ │ │ │ │ + var value = grandchild.nodeValue.replace( │ │ │ │ │ + this.regExes.trimSpace, ""); │ │ │ │ │ + attributes[name] = value; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var attrContainer = this.createElementNS(this.featureNS, "feature:" + nodename); │ │ │ │ │ - attrContainer.appendChild(attrText); │ │ │ │ │ - featureNode.appendChild(attrContainer); │ │ │ │ │ } │ │ │ │ │ - featureNode.appendChild(geometryNode); │ │ │ │ │ - return featureNode; │ │ │ │ │ + return attributes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometryNode │ │ │ │ │ - * builds a GeoRSS node with a given geometry │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseGeometry │ │ │ │ │ + * Parse the geometry and the feature bounds out of the node using │ │ │ │ │ + * Format.GML │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * node - {<DOMElement>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} A gml node. │ │ │ │ │ + * {Object} An object containing the geometry and the feature bounds │ │ │ │ │ */ │ │ │ │ │ - buildGeometryNode: function(geometry) { │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry = geometry.clone(); │ │ │ │ │ - geometry.transform(this.internalProjection, │ │ │ │ │ - this.externalProjection); │ │ │ │ │ - } │ │ │ │ │ - var node; │ │ │ │ │ - // match Polygon │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.Polygon") { │ │ │ │ │ - node = this.createElementNS(this.georssns, 'georss:polygon'); │ │ │ │ │ - │ │ │ │ │ - node.appendChild(this.buildCoordinatesNode(geometry.components[0])); │ │ │ │ │ - } │ │ │ │ │ - // match LineString │ │ │ │ │ - else if (geometry.CLASS_NAME == "OpenLayers.Geometry.LineString") { │ │ │ │ │ - node = this.createElementNS(this.georssns, 'georss:line'); │ │ │ │ │ - │ │ │ │ │ - node.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + parseGeometry: function(node) { │ │ │ │ │ + // we need to use the old Format.GML parser since we do not know the │ │ │ │ │ + // geometry name │ │ │ │ │ + if (!this.gmlFormat) { │ │ │ │ │ + this.gmlFormat = new OpenLayers.Format.GML(); │ │ │ │ │ } │ │ │ │ │ - // match Point │ │ │ │ │ - else if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - node = this.createElementNS(this.georssns, 'georss:point'); │ │ │ │ │ - node.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - } else { │ │ │ │ │ - throw "Couldn't parse " + geometry.CLASS_NAME; │ │ │ │ │ + var feature = this.gmlFormat.parseFeature(node); │ │ │ │ │ + var geometry, bounds = null; │ │ │ │ │ + if (feature) { │ │ │ │ │ + geometry = feature.geometry && feature.geometry.clone(); │ │ │ │ │ + bounds = feature.bounds && feature.bounds.clone(); │ │ │ │ │ + feature.destroy(); │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ + return { │ │ │ │ │ + geometry: geometry, │ │ │ │ │ + bounds: bounds │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildCoordinatesNode │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - */ │ │ │ │ │ - buildCoordinatesNode: function(geometry) { │ │ │ │ │ - var points = null; │ │ │ │ │ - │ │ │ │ │ - if (geometry.components) { │ │ │ │ │ - points = geometry.components; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var path; │ │ │ │ │ - if (points) { │ │ │ │ │ - var numPoints = points.length; │ │ │ │ │ - var parts = new Array(numPoints); │ │ │ │ │ - for (var i = 0; i < numPoints; i++) { │ │ │ │ │ - parts[i] = points[i].y + " " + points[i].x; │ │ │ │ │ - } │ │ │ │ │ - path = parts.join(" "); │ │ │ │ │ - } else { │ │ │ │ │ - path = geometry.y + " " + geometry.x; │ │ │ │ │ - } │ │ │ │ │ - return this.createTextNode(path); │ │ │ │ │ - }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSGetFeatureInfo" │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.GeoRSS" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSCapabilities.js │ │ │ │ │ + OpenLayers/Format/WFSDescribeFeatureType.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Format/OGCExceptionReport.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WMSCapabilities │ │ │ │ │ - * Read WMS Capabilities. │ │ │ │ │ + * Class: OpenLayers.Format.WFSDescribeFeatureType │ │ │ │ │ + * Read WFS DescribeFeatureType response │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WMSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ +OpenLayers.Format.WFSDescribeFeatureType = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.1.1". │ │ │ │ │ - */ │ │ │ │ │ - defaultVersion: "1.1.1", │ │ │ │ │ + /** │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ + */ │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g) │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: profile │ │ │ │ │ - * {String} If provided, use a custom profile. │ │ │ │ │ - * │ │ │ │ │ - * Currently supported profiles: │ │ │ │ │ - * - WMSC - parses vendor specific capabilities for WMS-C. │ │ │ │ │ - */ │ │ │ │ │ - profile: null, │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + xsd: "http://www.w3.org/2001/XMLSchema" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMSCapabilities │ │ │ │ │ - * Create a new parser for WMS capabilities. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WFSDescribeFeatureType │ │ │ │ │ + * Create a new parser for WFS DescribeFeatureType responses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} List of named layers. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "xsd": { │ │ │ │ │ + "schema": function(node, obj) { │ │ │ │ │ + var complexTypes = []; │ │ │ │ │ + var customTypes = {}; │ │ │ │ │ + var schema = { │ │ │ │ │ + complexTypes: complexTypes, │ │ │ │ │ + customTypes: customTypes │ │ │ │ │ + }; │ │ │ │ │ + var i, len; │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities" │ │ │ │ │ + this.readChildNodes(node, schema); │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ + var attributes = node.attributes; │ │ │ │ │ + var attr, name; │ │ │ │ │ + for (i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ + attr = attributes[i]; │ │ │ │ │ + name = attr.name; │ │ │ │ │ + if (name.indexOf("xmlns") === 0) { │ │ │ │ │ + this.setNamespace(name.split(":")[1] || "", attr.value); │ │ │ │ │ + } else { │ │ │ │ │ + obj[name] = attr.value; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + obj.featureTypes = complexTypes; │ │ │ │ │ + obj.targetPrefix = this.namespaceAlias[obj.targetNamespace]; │ │ │ │ │ + │ │ │ │ │ + // map complexTypes to names of customTypes │ │ │ │ │ + var complexType, customType; │ │ │ │ │ + for (i = 0, len = complexTypes.length; i < len; ++i) { │ │ │ │ │ + complexType = complexTypes[i]; │ │ │ │ │ + customType = customTypes[complexType.typeName]; │ │ │ │ │ + if (customTypes[complexType.typeName]) { │ │ │ │ │ + complexType.typeName = customType.name; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "complexType": function(node, obj) { │ │ │ │ │ + var complexType = { │ │ │ │ │ + // this is a temporary typeName, it will be overwritten by │ │ │ │ │ + // the schema reader with the metadata found in the │ │ │ │ │ + // customTypes hash │ │ │ │ │ + "typeName": node.getAttribute("name") │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, complexType); │ │ │ │ │ + obj.complexTypes.push(complexType); │ │ │ │ │ + }, │ │ │ │ │ + "complexContent": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "extension": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "sequence": function(node, obj) { │ │ │ │ │ + var sequence = { │ │ │ │ │ + elements: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, sequence); │ │ │ │ │ + obj.properties = sequence.elements; │ │ │ │ │ + }, │ │ │ │ │ + "element": function(node, obj) { │ │ │ │ │ + var type; │ │ │ │ │ + if (obj.elements) { │ │ │ │ │ + var element = {}; │ │ │ │ │ + var attributes = node.attributes; │ │ │ │ │ + var attr; │ │ │ │ │ + for (var i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ + attr = attributes[i]; │ │ │ │ │ + element[attr.name] = attr.value; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + type = element.type; │ │ │ │ │ + if (!type) { │ │ │ │ │ + type = {}; │ │ │ │ │ + this.readChildNodes(node, type); │ │ │ │ │ + element.restriction = type; │ │ │ │ │ + element.type = type.base; │ │ │ │ │ + } │ │ │ │ │ + var fullType = type.base || type; │ │ │ │ │ + element.localType = fullType.split(":").pop(); │ │ │ │ │ + obj.elements.push(element); │ │ │ │ │ + this.readChildNodes(node, element); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (obj.complexTypes) { │ │ │ │ │ + type = node.getAttribute("type"); │ │ │ │ │ + var localType = type.split(":").pop(); │ │ │ │ │ + obj.customTypes[localType] = { │ │ │ │ │ + "name": node.getAttribute("name"), │ │ │ │ │ + "type": type │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "annotation": function(node, obj) { │ │ │ │ │ + obj.annotation = {}; │ │ │ │ │ + this.readChildNodes(node, obj.annotation); │ │ │ │ │ + }, │ │ │ │ │ + "appinfo": function(node, obj) { │ │ │ │ │ + if (!obj.appinfo) { │ │ │ │ │ + obj.appinfo = []; │ │ │ │ │ + } │ │ │ │ │ + obj.appinfo.push(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "documentation": function(node, obj) { │ │ │ │ │ + if (!obj.documentation) { │ │ │ │ │ + obj.documentation = []; │ │ │ │ │ + } │ │ │ │ │ + var value = this.getChildValue(node); │ │ │ │ │ + obj.documentation.push({ │ │ │ │ │ + lang: node.getAttribute("xml:lang"), │ │ │ │ │ + textContent: value.replace(this.regExes.trimSpace, "") │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "simpleType": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "restriction": function(node, obj) { │ │ │ │ │ + obj.base = node.getAttribute("base"); │ │ │ │ │ + this.readRestriction(node, obj); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: readRestriction │ │ │ │ │ + * Reads restriction defined in the child nodes of a restriction element │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} the node to parse │ │ │ │ │ + * obj - {Object} the object that receives the read result │ │ │ │ │ + */ │ │ │ │ │ + readRestriction: function(node, obj) { │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + var child, nodeName, value; │ │ │ │ │ + for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ + child = children[i]; │ │ │ │ │ + if (child.nodeType == 1) { │ │ │ │ │ + nodeName = child.nodeName.split(":").pop(); │ │ │ │ │ + value = child.getAttribute("value"); │ │ │ │ │ + if (!obj[nodeName]) { │ │ │ │ │ + obj[nodeName] = value; │ │ │ │ │ + } else { │ │ │ │ │ + if (typeof obj[nodeName] == "string") { │ │ │ │ │ + obj[nodeName] = [obj[nodeName]]; │ │ │ │ │ + } │ │ │ │ │ + obj[nodeName].push(value); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: read │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {DOMElement|String} A WFS DescribeFeatureType document. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object representing the WFS DescribeFeatureType response. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + var schema = {}; │ │ │ │ │ + if (data.nodeName.split(":").pop() === 'ExceptionReport') { │ │ │ │ │ + // an exception must have occurred, so parse it │ │ │ │ │ + var parser = new OpenLayers.Format.OGCExceptionReport(); │ │ │ │ │ + schema.error = parser.read(data); │ │ │ │ │ + } else { │ │ │ │ │ + this.readNode(data, schema); │ │ │ │ │ + } │ │ │ │ │ + return schema; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSDescribeFeatureType" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/CSWGetDomain.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.CSWGetDomain │ │ │ │ │ + * Default version is 2.0.2. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Format>} A CSWGetDomain format of the given version. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.CSWGetDomain = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options, OpenLayers.Format.CSWGetDomain.DEFAULTS │ │ │ │ │ + ); │ │ │ │ │ + var cls = OpenLayers.Format.CSWGetDomain["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported CSWGetDomain version: " + options.version; │ │ │ │ │ + } │ │ │ │ │ + return new cls(options); │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: DEFAULTS │ │ │ │ │ + * {Object} Default properties for the CSWGetDomain format. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.CSWGetDomain.DEFAULTS = { │ │ │ │ │ + "version": "2.0.2" │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Format/WMTSCapabilities.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ @@ -50752,51 +47861,1329 @@ │ │ │ │ │ ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.WMTSCapabilities" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/CSWGetRecords.js │ │ │ │ │ + OpenLayers/Format/OSM.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ + * @requires OpenLayers/Projection.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.OSM │ │ │ │ │ + * OSM parser. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Format.OSM> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.OSM = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: checkTags │ │ │ │ │ + * {Boolean} Should tags be checked to determine whether something │ │ │ │ │ + * should be treated as a seperate node. Will slow down parsing. │ │ │ │ │ + * Default is false. │ │ │ │ │ + */ │ │ │ │ │ + checkTags: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: interestingTagsExclude │ │ │ │ │ + * {Array} List of tags to exclude from 'interesting' checks on nodes. │ │ │ │ │ + * Must be set when creating the format. Will only be used if checkTags │ │ │ │ │ + * is set. │ │ │ │ │ + */ │ │ │ │ │ + interestingTagsExclude: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: areaTags │ │ │ │ │ + * {Array} List of tags indicating that something is an area. │ │ │ │ │ + * Must be set when creating the format. Will only be used if │ │ │ │ │ + * checkTags is true. │ │ │ │ │ + */ │ │ │ │ │ + areaTags: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.OSM │ │ │ │ │ + * Create a new parser for OSM. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + var layer_defaults = { │ │ │ │ │ + 'interestingTagsExclude': ['source', 'source_ref', │ │ │ │ │ + 'source:ref', 'history', 'attribution', 'created_by' │ │ │ │ │ + ], │ │ │ │ │ + 'areaTags': ['area', 'building', 'leisure', 'tourism', 'ruins', │ │ │ │ │ + 'historic', 'landuse', 'military', 'natural', 'sport' │ │ │ │ │ + ] │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + layer_defaults = OpenLayers.Util.extend(layer_defaults, options); │ │ │ │ │ + │ │ │ │ │ + var interesting = {}; │ │ │ │ │ + for (var i = 0; i < layer_defaults.interestingTagsExclude.length; i++) { │ │ │ │ │ + interesting[layer_defaults.interestingTagsExclude[i]] = true; │ │ │ │ │ + } │ │ │ │ │ + layer_defaults.interestingTagsExclude = interesting; │ │ │ │ │ + │ │ │ │ │ + var area = {}; │ │ │ │ │ + for (var i = 0; i < layer_defaults.areaTags.length; i++) { │ │ │ │ │ + area[layer_defaults.areaTags[i]] = true; │ │ │ │ │ + } │ │ │ │ │ + layer_defaults.areaTags = area; │ │ │ │ │ + │ │ │ │ │ + // OSM coordinates are always in longlat WGS84 │ │ │ │ │ + this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [layer_defaults]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Return a list of features from a OSM doc │ │ │ │ │ + │ │ │ │ │ + * Parameters: │ │ │ │ │ + * doc - {Element} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ + */ │ │ │ │ │ + read: function(doc) { │ │ │ │ │ + if (typeof doc == "string") { │ │ │ │ │ + doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var nodes = this.getNodes(doc); │ │ │ │ │ + var ways = this.getWays(doc); │ │ │ │ │ + │ │ │ │ │ + // Geoms will contain at least ways.length entries. │ │ │ │ │ + var feat_list = new Array(ways.length); │ │ │ │ │ + │ │ │ │ │ + for (var i = 0; i < ways.length; i++) { │ │ │ │ │ + // We know the minimal of this one ahead of time. (Could be -1 │ │ │ │ │ + // due to areas/polygons) │ │ │ │ │ + var point_list = new Array(ways[i].nodes.length); │ │ │ │ │ + │ │ │ │ │ + var poly = this.isWayArea(ways[i]) ? 1 : 0; │ │ │ │ │ + for (var j = 0; j < ways[i].nodes.length; j++) { │ │ │ │ │ + var node = nodes[ways[i].nodes[j]]; │ │ │ │ │ + │ │ │ │ │ + var point = new OpenLayers.Geometry.Point(node.lon, node.lat); │ │ │ │ │ + │ │ │ │ │ + // Since OSM is topological, we stash the node ID internally. │ │ │ │ │ + point.osm_id = parseInt(ways[i].nodes[j]); │ │ │ │ │ + point_list[j] = point; │ │ │ │ │ + │ │ │ │ │ + // We don't display nodes if they're used inside other │ │ │ │ │ + // elements. │ │ │ │ │ + node.used = true; │ │ │ │ │ + } │ │ │ │ │ + var geometry = null; │ │ │ │ │ + if (poly) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.Polygon( │ │ │ │ │ + new OpenLayers.Geometry.LinearRing(point_list)); │ │ │ │ │ + } else { │ │ │ │ │ + geometry = new OpenLayers.Geometry.LineString(point_list); │ │ │ │ │ + } │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry.transform(this.externalProjection, │ │ │ │ │ + this.internalProjection); │ │ │ │ │ + } │ │ │ │ │ + var feat = new OpenLayers.Feature.Vector(geometry, │ │ │ │ │ + ways[i].tags); │ │ │ │ │ + feat.osm_id = parseInt(ways[i].id); │ │ │ │ │ + feat.fid = "way." + feat.osm_id; │ │ │ │ │ + feat_list[i] = feat; │ │ │ │ │ + } │ │ │ │ │ + for (var node_id in nodes) { │ │ │ │ │ + var node = nodes[node_id]; │ │ │ │ │ + if (!node.used || this.checkTags) { │ │ │ │ │ + var tags = null; │ │ │ │ │ + │ │ │ │ │ + if (this.checkTags) { │ │ │ │ │ + var result = this.getTags(node.node, true); │ │ │ │ │ + if (node.used && !result[1]) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + tags = result[0]; │ │ │ │ │ + } else { │ │ │ │ │ + tags = this.getTags(node.node); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var feat = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.Point(node['lon'], node['lat']), │ │ │ │ │ + tags); │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + feat.geometry.transform(this.externalProjection, │ │ │ │ │ + this.internalProjection); │ │ │ │ │ + } │ │ │ │ │ + feat.osm_id = parseInt(node_id); │ │ │ │ │ + feat.fid = "node." + feat.osm_id; │ │ │ │ │ + feat_list.push(feat); │ │ │ │ │ + } │ │ │ │ │ + // Memory cleanup │ │ │ │ │ + node.node = null; │ │ │ │ │ + } │ │ │ │ │ + return feat_list; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getNodes │ │ │ │ │ + * Return the node items from a doc. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * doc - {DOMElement} node to parse tags from │ │ │ │ │ + */ │ │ │ │ │ + getNodes: function(doc) { │ │ │ │ │ + var node_list = doc.getElementsByTagName("node"); │ │ │ │ │ + var nodes = {}; │ │ │ │ │ + for (var i = 0; i < node_list.length; i++) { │ │ │ │ │ + var node = node_list[i]; │ │ │ │ │ + var id = node.getAttribute("id"); │ │ │ │ │ + nodes[id] = { │ │ │ │ │ + 'lat': node.getAttribute("lat"), │ │ │ │ │ + 'lon': node.getAttribute("lon"), │ │ │ │ │ + 'node': node │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + return nodes; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getWays │ │ │ │ │ + * Return the way items from a doc. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * doc - {DOMElement} node to parse tags from │ │ │ │ │ + */ │ │ │ │ │ + getWays: function(doc) { │ │ │ │ │ + var way_list = doc.getElementsByTagName("way"); │ │ │ │ │ + var return_ways = []; │ │ │ │ │ + for (var i = 0; i < way_list.length; i++) { │ │ │ │ │ + var way = way_list[i]; │ │ │ │ │ + var way_object = { │ │ │ │ │ + id: way.getAttribute("id") │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + way_object.tags = this.getTags(way); │ │ │ │ │ + │ │ │ │ │ + var node_list = way.getElementsByTagName("nd"); │ │ │ │ │ + │ │ │ │ │ + way_object.nodes = new Array(node_list.length); │ │ │ │ │ + │ │ │ │ │ + for (var j = 0; j < node_list.length; j++) { │ │ │ │ │ + way_object.nodes[j] = node_list[j].getAttribute("ref"); │ │ │ │ │ + } │ │ │ │ │ + return_ways.push(way_object); │ │ │ │ │ + } │ │ │ │ │ + return return_ways; │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getTags │ │ │ │ │ + * Return the tags list attached to a specific DOM element. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * dom_node - {DOMElement} node to parse tags from │ │ │ │ │ + * interesting_tags - {Boolean} whether the return from this function should │ │ │ │ │ + * return a boolean indicating that it has 'interesting tags' -- │ │ │ │ │ + * tags like attribution and source are ignored. (To change the list │ │ │ │ │ + * of tags, see interestingTagsExclude) │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * tags - {Object} hash of tags │ │ │ │ │ + * interesting - {Boolean} if interesting_tags is passed, returns │ │ │ │ │ + * whether there are any interesting tags on this element. │ │ │ │ │ + */ │ │ │ │ │ + getTags: function(dom_node, interesting_tags) { │ │ │ │ │ + var tag_list = dom_node.getElementsByTagName("tag"); │ │ │ │ │ + var tags = {}; │ │ │ │ │ + var interesting = false; │ │ │ │ │ + for (var j = 0; j < tag_list.length; j++) { │ │ │ │ │ + var key = tag_list[j].getAttribute("k"); │ │ │ │ │ + tags[key] = tag_list[j].getAttribute("v"); │ │ │ │ │ + if (interesting_tags) { │ │ │ │ │ + if (!this.interestingTagsExclude[key]) { │ │ │ │ │ + interesting = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return interesting_tags ? [tags, interesting] : tags; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: isWayArea │ │ │ │ │ + * Given a way object from getWays, check whether the tags and geometry │ │ │ │ │ + * indicate something is an area. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + isWayArea: function(way) { │ │ │ │ │ + var poly_shaped = false; │ │ │ │ │ + var poly_tags = false; │ │ │ │ │ + │ │ │ │ │ + if (way.nodes[0] == way.nodes[way.nodes.length - 1]) { │ │ │ │ │ + poly_shaped = true; │ │ │ │ │ + } │ │ │ │ │ + if (this.checkTags) { │ │ │ │ │ + for (var key in way.tags) { │ │ │ │ │ + if (this.areaTags[key]) { │ │ │ │ │ + poly_tags = true; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return poly_shaped && (this.checkTags ? poly_tags : true); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Takes a list of features, returns a serialized OSM format file for use │ │ │ │ │ + * in tools like JOSM. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + */ │ │ │ │ │ + write: function(features) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.osm_id = 1; │ │ │ │ │ + this.created_nodes = {}; │ │ │ │ │ + var root_node = this.createElementNS(null, "osm"); │ │ │ │ │ + root_node.setAttribute("version", "0.5"); │ │ │ │ │ + root_node.setAttribute("generator", "OpenLayers " + OpenLayers.VERSION_NUMBER); │ │ │ │ │ + │ │ │ │ │ + // Loop backwards, because the deserializer puts nodes last, and │ │ │ │ │ + // we want them first if possible │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + var nodes = this.createFeatureNodes(features[i]); │ │ │ │ │ + for (var j = 0; j < nodes.length; j++) { │ │ │ │ │ + root_node.appendChild(nodes[j]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [root_node]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createFeatureNodes │ │ │ │ │ + * Takes a feature, returns a list of nodes from size 0->n. │ │ │ │ │ + * Will include all pieces of the serialization that are required which │ │ │ │ │ + * have not already been created. Calls out to createXML based on geometry │ │ │ │ │ + * type. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + createFeatureNodes: function(feature) { │ │ │ │ │ + var nodes = []; │ │ │ │ │ + var className = feature.geometry.CLASS_NAME; │ │ │ │ │ + var type = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ + type = type.toLowerCase(); │ │ │ │ │ + var builder = this.createXML[type]; │ │ │ │ │ + if (builder) { │ │ │ │ │ + nodes = builder.apply(this, [feature]); │ │ │ │ │ + } │ │ │ │ │ + return nodes; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createXML │ │ │ │ │ + * Takes a feature, returns a list of nodes from size 0->n. │ │ │ │ │ + * Will include all pieces of the serialization that are required which │ │ │ │ │ + * have not already been created. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + createXML: { │ │ │ │ │ + 'point': function(point) { │ │ │ │ │ + var id = null; │ │ │ │ │ + var geometry = point.geometry ? point.geometry : point; │ │ │ │ │ + │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry = geometry.clone(); │ │ │ │ │ + geometry.transform(this.internalProjection, │ │ │ │ │ + this.externalProjection); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var already_exists = false; // We don't return anything if the node │ │ │ │ │ + // has already been created │ │ │ │ │ + if (point.osm_id) { │ │ │ │ │ + id = point.osm_id; │ │ │ │ │ + if (this.created_nodes[id]) { │ │ │ │ │ + already_exists = true; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + id = -this.osm_id; │ │ │ │ │ + this.osm_id++; │ │ │ │ │ + } │ │ │ │ │ + if (already_exists) { │ │ │ │ │ + node = this.created_nodes[id]; │ │ │ │ │ + } else { │ │ │ │ │ + var node = this.createElementNS(null, "node"); │ │ │ │ │ + } │ │ │ │ │ + this.created_nodes[id] = node; │ │ │ │ │ + node.setAttribute("id", id); │ │ │ │ │ + node.setAttribute("lon", geometry.x); │ │ │ │ │ + node.setAttribute("lat", geometry.y); │ │ │ │ │ + if (point.attributes) { │ │ │ │ │ + this.serializeTags(point, node); │ │ │ │ │ + } │ │ │ │ │ + this.setState(point, node); │ │ │ │ │ + return already_exists ? [] : [node]; │ │ │ │ │ + }, │ │ │ │ │ + linestring: function(feature) { │ │ │ │ │ + var id; │ │ │ │ │ + var nodes = []; │ │ │ │ │ + var geometry = feature.geometry; │ │ │ │ │ + if (feature.osm_id) { │ │ │ │ │ + id = feature.osm_id; │ │ │ │ │ + } else { │ │ │ │ │ + id = -this.osm_id; │ │ │ │ │ + this.osm_id++; │ │ │ │ │ + } │ │ │ │ │ + var way = this.createElementNS(null, "way"); │ │ │ │ │ + way.setAttribute("id", id); │ │ │ │ │ + for (var i = 0; i < geometry.components.length; i++) { │ │ │ │ │ + var node = this.createXML['point'].apply(this, [geometry.components[i]]); │ │ │ │ │ + if (node.length) { │ │ │ │ │ + node = node[0]; │ │ │ │ │ + var node_ref = node.getAttribute("id"); │ │ │ │ │ + nodes.push(node); │ │ │ │ │ + } else { │ │ │ │ │ + node_ref = geometry.components[i].osm_id; │ │ │ │ │ + node = this.created_nodes[node_ref]; │ │ │ │ │ + } │ │ │ │ │ + this.setState(feature, node); │ │ │ │ │ + var nd_dom = this.createElementNS(null, "nd"); │ │ │ │ │ + nd_dom.setAttribute("ref", node_ref); │ │ │ │ │ + way.appendChild(nd_dom); │ │ │ │ │ + } │ │ │ │ │ + this.serializeTags(feature, way); │ │ │ │ │ + nodes.push(way); │ │ │ │ │ + │ │ │ │ │ + return nodes; │ │ │ │ │ + }, │ │ │ │ │ + polygon: function(feature) { │ │ │ │ │ + var attrs = OpenLayers.Util.extend({ │ │ │ │ │ + 'area': 'yes' │ │ │ │ │ + }, feature.attributes); │ │ │ │ │ + var feat = new OpenLayers.Feature.Vector(feature.geometry.components[0], attrs); │ │ │ │ │ + feat.osm_id = feature.osm_id; │ │ │ │ │ + return this.createXML['linestring'].apply(this, [feat]); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: serializeTags │ │ │ │ │ + * Given a feature, serialize the attributes onto the given node. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * node - {DOMNode} │ │ │ │ │ + */ │ │ │ │ │ + serializeTags: function(feature, node) { │ │ │ │ │ + for (var key in feature.attributes) { │ │ │ │ │ + var tag = this.createElementNS(null, "tag"); │ │ │ │ │ + tag.setAttribute("k", key); │ │ │ │ │ + tag.setAttribute("v", feature.attributes[key]); │ │ │ │ │ + node.appendChild(tag); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setState │ │ │ │ │ + * OpenStreetMap has a convention that 'state' is stored for modification or deletion. │ │ │ │ │ + * This allows the file to be uploaded via JOSM or the bulk uploader tool. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * node - {DOMNode} │ │ │ │ │ + */ │ │ │ │ │ + setState: function(feature, node) { │ │ │ │ │ + if (feature.state) { │ │ │ │ │ + var state = null; │ │ │ │ │ + switch (feature.state) { │ │ │ │ │ + case OpenLayers.State.UPDATE: │ │ │ │ │ + state = "modify"; │ │ │ │ │ + case OpenLayers.State.DELETE: │ │ │ │ │ + state = "delete"; │ │ │ │ │ + } │ │ │ │ │ + if (state) { │ │ │ │ │ + node.setAttribute("action", state); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.OSM" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WFS.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/GML.js │ │ │ │ │ + * @requires OpenLayers/Console.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WFS │ │ │ │ │ + * Read/Write WFS. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.GML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WFS = OpenLayers.Class(OpenLayers.Format.GML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {<OpenLayers.Layer>} │ │ │ │ │ + */ │ │ │ │ │ + layer: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: wfsns │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + wfsns: "http://www.opengis.net/wfs", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: ogcns │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + ogcns: "http://www.opengis.net/ogc", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WFS │ │ │ │ │ + * Create a WFS-T formatter. This requires a layer: that layer should │ │ │ │ │ + * have two properties: geometry_column and typename. The parser │ │ │ │ │ + * for this format is subclassed entirely from GML: There is a writer │ │ │ │ │ + * only, which uses most of the code from the GML layer, and wraps │ │ │ │ │ + * it in transactional elements. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} │ │ │ │ │ + * layer - {<OpenLayers.Layer>} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options, layer) { │ │ │ │ │ + OpenLayers.Format.GML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + if (this.layer.featureNS) { │ │ │ │ │ + this.featureNS = this.layer.featureNS; │ │ │ │ │ + } │ │ │ │ │ + if (this.layer.options.geometry_column) { │ │ │ │ │ + this.geometryName = this.layer.options.geometry_column; │ │ │ │ │ + } │ │ │ │ │ + if (this.layer.options.typename) { │ │ │ │ │ + this.featureName = this.layer.options.typename; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: write │ │ │ │ │ + * Takes a feature list, and generates a WFS-T Transaction │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + */ │ │ │ │ │ + write: function(features) { │ │ │ │ │ + │ │ │ │ │ + var transaction = this.createElementNS(this.wfsns, 'wfs:Transaction'); │ │ │ │ │ + transaction.setAttribute("version", "1.0.0"); │ │ │ │ │ + transaction.setAttribute("service", "WFS"); │ │ │ │ │ + for (var i = 0; i < features.length; i++) { │ │ │ │ │ + switch (features[i].state) { │ │ │ │ │ + case OpenLayers.State.INSERT: │ │ │ │ │ + transaction.appendChild(this.insert(features[i])); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.State.UPDATE: │ │ │ │ │ + transaction.appendChild(this.update(features[i])); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.State.DELETE: │ │ │ │ │ + transaction.appendChild(this.remove(features[i])); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [transaction]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createFeatureXML │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + createFeatureXML: function(feature) { │ │ │ │ │ + var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ + var geomContainer = this.createElementNS(this.featureNS, "feature:" + this.geometryName); │ │ │ │ │ + geomContainer.appendChild(geometryNode); │ │ │ │ │ + var featureContainer = this.createElementNS(this.featureNS, "feature:" + this.featureName); │ │ │ │ │ + featureContainer.appendChild(geomContainer); │ │ │ │ │ + for (var attr in feature.attributes) { │ │ │ │ │ + var attrText = this.createTextNode(feature.attributes[attr]); │ │ │ │ │ + var nodename = attr; │ │ │ │ │ + if (attr.search(":") != -1) { │ │ │ │ │ + nodename = attr.split(":")[1]; │ │ │ │ │ + } │ │ │ │ │ + var attrContainer = this.createElementNS(this.featureNS, "feature:" + nodename); │ │ │ │ │ + attrContainer.appendChild(attrText); │ │ │ │ │ + featureContainer.appendChild(attrContainer); │ │ │ │ │ + } │ │ │ │ │ + return featureContainer; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: insert │ │ │ │ │ + * Takes a feature, and generates a WFS-T Transaction "Insert" │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + insert: function(feature) { │ │ │ │ │ + var insertNode = this.createElementNS(this.wfsns, 'wfs:Insert'); │ │ │ │ │ + insertNode.appendChild(this.createFeatureXML(feature)); │ │ │ │ │ + return insertNode; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: update │ │ │ │ │ + * Takes a feature, and generates a WFS-T Transaction "Update" │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + update: function(feature) { │ │ │ │ │ + if (!feature.fid) { │ │ │ │ │ + OpenLayers.Console.userError(OpenLayers.i18n("noFID")); │ │ │ │ │ + } │ │ │ │ │ + var updateNode = this.createElementNS(this.wfsns, 'wfs:Update'); │ │ │ │ │ + updateNode.setAttribute("typeName", this.featurePrefix + ':' + this.featureName); │ │ │ │ │ + updateNode.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ + │ │ │ │ │ + var propertyNode = this.createElementNS(this.wfsns, 'wfs:Property'); │ │ │ │ │ + var nameNode = this.createElementNS(this.wfsns, 'wfs:Name'); │ │ │ │ │ + │ │ │ │ │ + var txtNode = this.createTextNode(this.geometryName); │ │ │ │ │ + nameNode.appendChild(txtNode); │ │ │ │ │ + propertyNode.appendChild(nameNode); │ │ │ │ │ + │ │ │ │ │ + var valueNode = this.createElementNS(this.wfsns, 'wfs:Value'); │ │ │ │ │ + │ │ │ │ │ + var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ + │ │ │ │ │ + if (feature.layer) { │ │ │ │ │ + geometryNode.setAttribute( │ │ │ │ │ + "srsName", feature.layer.projection.getCode() │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + valueNode.appendChild(geometryNode); │ │ │ │ │ + │ │ │ │ │ + propertyNode.appendChild(valueNode); │ │ │ │ │ + updateNode.appendChild(propertyNode); │ │ │ │ │ + │ │ │ │ │ + // add in attributes │ │ │ │ │ + for (var propName in feature.attributes) { │ │ │ │ │ + propertyNode = this.createElementNS(this.wfsns, 'wfs:Property'); │ │ │ │ │ + nameNode = this.createElementNS(this.wfsns, 'wfs:Name'); │ │ │ │ │ + nameNode.appendChild(this.createTextNode(propName)); │ │ │ │ │ + propertyNode.appendChild(nameNode); │ │ │ │ │ + valueNode = this.createElementNS(this.wfsns, 'wfs:Value'); │ │ │ │ │ + valueNode.appendChild(this.createTextNode(feature.attributes[propName])); │ │ │ │ │ + propertyNode.appendChild(valueNode); │ │ │ │ │ + updateNode.appendChild(propertyNode); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + var filterNode = this.createElementNS(this.ogcns, 'ogc:Filter'); │ │ │ │ │ + var filterIdNode = this.createElementNS(this.ogcns, 'ogc:FeatureId'); │ │ │ │ │ + filterIdNode.setAttribute("fid", feature.fid); │ │ │ │ │ + filterNode.appendChild(filterIdNode); │ │ │ │ │ + updateNode.appendChild(filterNode); │ │ │ │ │ + │ │ │ │ │ + return updateNode; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: remove │ │ │ │ │ + * Takes a feature, and generates a WFS-T Transaction "Delete" │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + remove: function(feature) { │ │ │ │ │ + if (!feature.fid) { │ │ │ │ │ + OpenLayers.Console.userError(OpenLayers.i18n("noFID")); │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + var deleteNode = this.createElementNS(this.wfsns, 'wfs:Delete'); │ │ │ │ │ + deleteNode.setAttribute("typeName", this.featurePrefix + ':' + this.featureName); │ │ │ │ │ + deleteNode.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ + │ │ │ │ │ + var filterNode = this.createElementNS(this.ogcns, 'ogc:Filter'); │ │ │ │ │ + var filterIdNode = this.createElementNS(this.ogcns, 'ogc:FeatureId'); │ │ │ │ │ + filterIdNode.setAttribute("fid", feature.fid); │ │ │ │ │ + filterNode.appendChild(filterIdNode); │ │ │ │ │ + deleteNode.appendChild(filterNode); │ │ │ │ │ + │ │ │ │ │ + return deleteNode; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Remove ciruclar ref to layer │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.layer = null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFS" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/EncodedPolyline.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Format.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.CSWGetRecords │ │ │ │ │ - * Default version is 2.0.2. │ │ │ │ │ + * Class: OpenLayers.Format.EncodedPolyline │ │ │ │ │ + * Class for reading and writing encoded polylines. Create a new instance │ │ │ │ │ + * with the <OpenLayers.Format.EncodedPolyline> constructor. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Format>} A CSWGetRecords format of the given version. │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.CSWGetRecords = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options, OpenLayers.Format.CSWGetRecords.DEFAULTS │ │ │ │ │ - ); │ │ │ │ │ - var cls = OpenLayers.Format.CSWGetRecords["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported CSWGetRecords version: " + options.version; │ │ │ │ │ - } │ │ │ │ │ - return new cls(options); │ │ │ │ │ -}; │ │ │ │ │ +OpenLayers.Format.EncodedPolyline = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: geometryType │ │ │ │ │ + * {String} Geometry type to output. One of: linestring (default), │ │ │ │ │ + * linearring, point, multipoint or polygon. If the geometryType is │ │ │ │ │ + * point, only the first point of the string is returned. │ │ │ │ │ + */ │ │ │ │ │ + geometryType: "linestring", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.EncodedPolyline │ │ │ │ │ + * Create a new parser for encoded polylines │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Format.EncodedPolyline>} A new encoded polylines parser. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Deserialize an encoded polyline string and return a vector feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * encoded - {String} An encoded polyline string │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A vector feature with a linestring. │ │ │ │ │ + */ │ │ │ │ │ + read: function(encoded) { │ │ │ │ │ + var geomType; │ │ │ │ │ + if (this.geometryType == "linestring") │ │ │ │ │ + geomType = OpenLayers.Geometry.LineString; │ │ │ │ │ + else if (this.geometryType == "linearring") │ │ │ │ │ + geomType = OpenLayers.Geometry.LinearRing; │ │ │ │ │ + else if (this.geometryType == "multipoint") │ │ │ │ │ + geomType = OpenLayers.Geometry.MultiPoint; │ │ │ │ │ + else if (this.geometryType != "point" && this.geometryType != "polygon") │ │ │ │ │ + return null; │ │ │ │ │ + │ │ │ │ │ + var flatPoints = this.decodeDeltas(encoded, 2); │ │ │ │ │ + var flatPointsLength = flatPoints.length; │ │ │ │ │ + │ │ │ │ │ + var pointGeometries = []; │ │ │ │ │ + for (var i = 0; i + 1 < flatPointsLength;) { │ │ │ │ │ + var y = flatPoints[i++], │ │ │ │ │ + x = flatPoints[i++]; │ │ │ │ │ + pointGeometries.push(new OpenLayers.Geometry.Point(x, y)); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + if (this.geometryType == "point") │ │ │ │ │ + return new OpenLayers.Feature.Vector( │ │ │ │ │ + pointGeometries[0] │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + if (this.geometryType == "polygon") │ │ │ │ │ + return new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.Polygon([ │ │ │ │ │ + new OpenLayers.Geometry.LinearRing(pointGeometries) │ │ │ │ │ + ]) │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + return new OpenLayers.Feature.Vector( │ │ │ │ │ + new geomType(pointGeometries) │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: decode │ │ │ │ │ + * Deserialize an encoded string and return an array of n-dimensional │ │ │ │ │ + * points. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * encoded - {String} An encoded string │ │ │ │ │ + * dims - {int} The dimension of the points that are returned │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(Array(int))} An array containing n-dimensional arrays of │ │ │ │ │ + * coordinates. │ │ │ │ │ + */ │ │ │ │ │ + decode: function(encoded, dims, opt_factor) { │ │ │ │ │ + var factor = opt_factor || 1e5; │ │ │ │ │ + var flatPoints = this.decodeDeltas(encoded, dims, factor); │ │ │ │ │ + var flatPointsLength = flatPoints.length; │ │ │ │ │ + │ │ │ │ │ + var points = []; │ │ │ │ │ + for (var i = 0; i + (dims - 1) < flatPointsLength;) { │ │ │ │ │ + var point = []; │ │ │ │ │ + │ │ │ │ │ + for (var dim = 0; dim < dims; ++dim) { │ │ │ │ │ + point.push(flatPoints[i++]) │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + points.push(point); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return points; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Serialize a feature or array of features into a WKT string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {<OpenLayers.Feature.Vector>|Array} A feature or array of │ │ │ │ │ + * features │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The WKT string representation of the input geometries │ │ │ │ │ + */ │ │ │ │ │ + write: function(features) { │ │ │ │ │ + var feature; │ │ │ │ │ + if (features.constructor == Array) │ │ │ │ │ + feature = features[0]; │ │ │ │ │ + else │ │ │ │ │ + feature = features; │ │ │ │ │ + │ │ │ │ │ + var geometry = feature.geometry; │ │ │ │ │ + var type = geometry.CLASS_NAME.split('.')[2].toLowerCase(); │ │ │ │ │ + │ │ │ │ │ + var pointGeometries; │ │ │ │ │ + if (type == "point") │ │ │ │ │ + pointGeometries = new Array(geometry); │ │ │ │ │ + else if (type == "linestring" || │ │ │ │ │ + type == "linearring" || │ │ │ │ │ + type == "multipoint") │ │ │ │ │ + pointGeometries = geometry.components; │ │ │ │ │ + else if (type == "polygon") │ │ │ │ │ + pointGeometries = geometry.components[0].components; │ │ │ │ │ + else │ │ │ │ │ + return null; │ │ │ │ │ + │ │ │ │ │ + var flatPoints = []; │ │ │ │ │ + │ │ │ │ │ + var pointGeometriesLength = pointGeometries.length; │ │ │ │ │ + for (var i = 0; i < pointGeometriesLength; ++i) { │ │ │ │ │ + var pointGeometry = pointGeometries[i]; │ │ │ │ │ + flatPoints.push(pointGeometry.y); │ │ │ │ │ + flatPoints.push(pointGeometry.x); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return this.encodeDeltas(flatPoints, 2); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: encode │ │ │ │ │ + * Serialize an array of n-dimensional points and return an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * points - {Array(Array(int))} An array containing n-dimensional │ │ │ │ │ + * arrays of coordinates │ │ │ │ │ + * dims - {int} The dimension of the points that should be read │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} An encoded string │ │ │ │ │ + */ │ │ │ │ │ + encode: function(points, dims, opt_factor) { │ │ │ │ │ + var factor = opt_factor || 1e5; │ │ │ │ │ + var flatPoints = []; │ │ │ │ │ + │ │ │ │ │ + var pointsLength = points.length; │ │ │ │ │ + for (var i = 0; i < pointsLength; ++i) { │ │ │ │ │ + var point = points[i]; │ │ │ │ │ + │ │ │ │ │ + for (var dim = 0; dim < dims; ++dim) { │ │ │ │ │ + flatPoints.push(point[dim]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return this.encodeDeltas(flatPoints, dims, factor); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: encodeDeltas │ │ │ │ │ + * Encode a list of n-dimensional points and return an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Attention: This function will modify the passed array! │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * numbers - {Array.<number>} A list of n-dimensional points. │ │ │ │ │ + * dimension - {number} The dimension of the points in the list. │ │ │ │ │ + * opt_factor - {number=} The factor by which the numbers will be │ │ │ │ │ + * multiplied. The remaining decimal places will get rounded away. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {string} The encoded string. │ │ │ │ │ + */ │ │ │ │ │ + encodeDeltas: function(numbers, dimension, opt_factor) { │ │ │ │ │ + var factor = opt_factor || 1e5; │ │ │ │ │ + var d; │ │ │ │ │ + │ │ │ │ │ + var lastNumbers = new Array(dimension); │ │ │ │ │ + for (d = 0; d < dimension; ++d) { │ │ │ │ │ + lastNumbers[d] = 0; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var numbersLength = numbers.length; │ │ │ │ │ + for (var i = 0; i < numbersLength;) { │ │ │ │ │ + for (d = 0; d < dimension; ++d, ++i) { │ │ │ │ │ + var num = numbers[i]; │ │ │ │ │ + var delta = num - lastNumbers[d]; │ │ │ │ │ + lastNumbers[d] = num; │ │ │ │ │ + │ │ │ │ │ + numbers[i] = delta; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return this.encodeFloats(numbers, factor); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: decodeDeltas │ │ │ │ │ + * Decode a list of n-dimensional points from an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * encoded - {string} An encoded string. │ │ │ │ │ + * dimension - {number} The dimension of the points in the encoded string. │ │ │ │ │ + * opt_factor - {number=} The factor by which the resulting numbers will │ │ │ │ │ + * be divided. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array.<number>} A list of n-dimensional points. │ │ │ │ │ + */ │ │ │ │ │ + decodeDeltas: function(encoded, dimension, opt_factor) { │ │ │ │ │ + var factor = opt_factor || 1e5; │ │ │ │ │ + var d; │ │ │ │ │ + │ │ │ │ │ + var lastNumbers = new Array(dimension); │ │ │ │ │ + for (d = 0; d < dimension; ++d) { │ │ │ │ │ + lastNumbers[d] = 0; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var numbers = this.decodeFloats(encoded, factor); │ │ │ │ │ + │ │ │ │ │ + var numbersLength = numbers.length; │ │ │ │ │ + for (var i = 0; i < numbersLength;) { │ │ │ │ │ + for (d = 0; d < dimension; ++d, ++i) { │ │ │ │ │ + lastNumbers[d] += numbers[i]; │ │ │ │ │ + │ │ │ │ │ + numbers[i] = lastNumbers[d]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return numbers; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: encodeFloats │ │ │ │ │ + * Encode a list of floating point numbers and return an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Attention: This function will modify the passed array! │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * numbers - {Array.<number>} A list of floating point numbers. │ │ │ │ │ + * opt_factor - {number=} The factor by which the numbers will be │ │ │ │ │ + * multiplied. The remaining decimal places will get rounded away. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {string} The encoded string. │ │ │ │ │ + */ │ │ │ │ │ + encodeFloats: function(numbers, opt_factor) { │ │ │ │ │ + var factor = opt_factor || 1e5; │ │ │ │ │ + │ │ │ │ │ + var numbersLength = numbers.length; │ │ │ │ │ + for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ + numbers[i] = Math.round(numbers[i] * factor); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return this.encodeSignedIntegers(numbers); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: decodeFloats │ │ │ │ │ + * Decode a list of floating point numbers from an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * encoded - {string} An encoded string. │ │ │ │ │ + * opt_factor - {number=} The factor by which the result will be divided. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array.<number>} A list of floating point numbers. │ │ │ │ │ + */ │ │ │ │ │ + decodeFloats: function(encoded, opt_factor) { │ │ │ │ │ + var factor = opt_factor || 1e5; │ │ │ │ │ + │ │ │ │ │ + var numbers = this.decodeSignedIntegers(encoded); │ │ │ │ │ + │ │ │ │ │ + var numbersLength = numbers.length; │ │ │ │ │ + for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ + numbers[i] /= factor; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return numbers; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: encodeSignedIntegers │ │ │ │ │ + * Encode a list of signed integers and return an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Attention: This function will modify the passed array! │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * numbers - {Array.<number>} A list of signed integers. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {string} The encoded string. │ │ │ │ │ + */ │ │ │ │ │ + encodeSignedIntegers: function(numbers) { │ │ │ │ │ + var numbersLength = numbers.length; │ │ │ │ │ + for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ + var num = numbers[i]; │ │ │ │ │ + │ │ │ │ │ + var signedNum = num << 1; │ │ │ │ │ + if (num < 0) { │ │ │ │ │ + signedNum = ~(signedNum); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + numbers[i] = signedNum; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return this.encodeUnsignedIntegers(numbers); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: decodeSignedIntegers │ │ │ │ │ + * Decode a list of signed integers from an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * encoded - {string} An encoded string. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array.<number>} A list of signed integers. │ │ │ │ │ + */ │ │ │ │ │ + decodeSignedIntegers: function(encoded) { │ │ │ │ │ + var numbers = this.decodeUnsignedIntegers(encoded); │ │ │ │ │ + │ │ │ │ │ + var numbersLength = numbers.length; │ │ │ │ │ + for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ + var num = numbers[i]; │ │ │ │ │ + numbers[i] = (num & 1) ? ~(num >> 1) : (num >> 1); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return numbers; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: encodeUnsignedIntegers │ │ │ │ │ + * Encode a list of unsigned integers and return an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * numbers - {Array.<number>} A list of unsigned integers. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {string} The encoded string. │ │ │ │ │ + */ │ │ │ │ │ + encodeUnsignedIntegers: function(numbers) { │ │ │ │ │ + var encoded = ''; │ │ │ │ │ + │ │ │ │ │ + var numbersLength = numbers.length; │ │ │ │ │ + for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ + encoded += this.encodeUnsignedInteger(numbers[i]); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return encoded; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: decodeUnsignedIntegers │ │ │ │ │ + * Decode a list of unsigned integers from an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * encoded - {string} An encoded string. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array.<number>} A list of unsigned integers. │ │ │ │ │ + */ │ │ │ │ │ + decodeUnsignedIntegers: function(encoded) { │ │ │ │ │ + var numbers = []; │ │ │ │ │ + │ │ │ │ │ + var current = 0; │ │ │ │ │ + var shift = 0; │ │ │ │ │ + │ │ │ │ │ + var encodedLength = encoded.length; │ │ │ │ │ + for (var i = 0; i < encodedLength; ++i) { │ │ │ │ │ + var b = encoded.charCodeAt(i) - 63; │ │ │ │ │ + │ │ │ │ │ + current |= (b & 0x1f) << shift; │ │ │ │ │ + │ │ │ │ │ + if (b < 0x20) { │ │ │ │ │ + numbers.push(current); │ │ │ │ │ + current = 0; │ │ │ │ │ + shift = 0; │ │ │ │ │ + } else { │ │ │ │ │ + shift += 5; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return numbers; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: encodeFloat │ │ │ │ │ + * Encode one single floating point number and return an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * num - {number} Floating point number that should be encoded. │ │ │ │ │ + * opt_factor - {number=} The factor by which num will be multiplied. │ │ │ │ │ + * The remaining decimal places will get rounded away. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {string} The encoded string. │ │ │ │ │ + */ │ │ │ │ │ + encodeFloat: function(num, opt_factor) { │ │ │ │ │ + num = Math.round(num * (opt_factor || 1e5)); │ │ │ │ │ + return this.encodeSignedInteger(num); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: decodeFloat │ │ │ │ │ + * Decode one single floating point number from an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * encoded - {string} An encoded string. │ │ │ │ │ + * opt_factor - {number=} The factor by which the result will be divided. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {number} The decoded floating point number. │ │ │ │ │ + */ │ │ │ │ │ + decodeFloat: function(encoded, opt_factor) { │ │ │ │ │ + var result = this.decodeSignedInteger(encoded); │ │ │ │ │ + return result / (opt_factor || 1e5); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: encodeSignedInteger │ │ │ │ │ + * Encode one single signed integer and return an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * num - {number} Signed integer that should be encoded. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {string} The encoded string. │ │ │ │ │ + */ │ │ │ │ │ + encodeSignedInteger: function(num) { │ │ │ │ │ + var signedNum = num << 1; │ │ │ │ │ + if (num < 0) { │ │ │ │ │ + signedNum = ~(signedNum); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return this.encodeUnsignedInteger(signedNum); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: decodeSignedInteger │ │ │ │ │ + * Decode one single signed integer from an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * encoded - {string} An encoded string. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {number} The decoded signed integer. │ │ │ │ │ + */ │ │ │ │ │ + decodeSignedInteger: function(encoded) { │ │ │ │ │ + var result = this.decodeUnsignedInteger(encoded); │ │ │ │ │ + return ((result & 1) ? ~(result >> 1) : (result >> 1)); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: encodeUnsignedInteger │ │ │ │ │ + * Encode one single unsigned integer and return an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * num - {number} Unsigned integer that should be encoded. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {string} The encoded string. │ │ │ │ │ + */ │ │ │ │ │ + encodeUnsignedInteger: function(num) { │ │ │ │ │ + var value, encoded = ''; │ │ │ │ │ + while (num >= 0x20) { │ │ │ │ │ + value = (0x20 | (num & 0x1f)) + 63; │ │ │ │ │ + encoded += (String.fromCharCode(value)); │ │ │ │ │ + num >>= 5; │ │ │ │ │ + } │ │ │ │ │ + value = num + 63; │ │ │ │ │ + encoded += (String.fromCharCode(value)); │ │ │ │ │ + return encoded; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: decodeUnsignedInteger │ │ │ │ │ + * Decode one single unsigned integer from an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * encoded - {string} An encoded string. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {number} The decoded unsigned integer. │ │ │ │ │ + */ │ │ │ │ │ + decodeUnsignedInteger: function(encoded) { │ │ │ │ │ + var result = 0; │ │ │ │ │ + var shift = 0; │ │ │ │ │ + │ │ │ │ │ + var encodedLength = encoded.length; │ │ │ │ │ + for (var i = 0; i < encodedLength; ++i) { │ │ │ │ │ + var b = encoded.charCodeAt(i) - 63; │ │ │ │ │ + │ │ │ │ │ + result |= (b & 0x1f) << shift; │ │ │ │ │ + │ │ │ │ │ + if (b < 0x20) │ │ │ │ │ + break; │ │ │ │ │ + │ │ │ │ │ + shift += 5; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return result; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.EncodedPolyline" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/SOSCapabilities.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: DEFAULTS │ │ │ │ │ - * {Object} Default properties for the CSWGetRecords format. │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.CSWGetRecords.DEFAULTS = { │ │ │ │ │ - "version": "2.0.2" │ │ │ │ │ -}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.SOSCapabilities │ │ │ │ │ + * Read SOS Capabilities. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.SOSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.0.0". │ │ │ │ │ + */ │ │ │ │ │ + defaultVersion: "1.0.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.SOSCapabilities │ │ │ │ │ + * Create a new parser for SOS Capabilities. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return information about │ │ │ │ │ + * the service (offering and observedProperty mostly). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Info about the SOS │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SOSCapabilities" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Format/CQL.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ @@ -51246,283 +49633,1623 @@ │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.CQL" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ })(); │ │ │ │ │ │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WFSDescribeFeatureType.js │ │ │ │ │ + OpenLayers/Format/WMSDescribeLayer.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/OGCExceptionReport.js │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WFSDescribeFeatureType │ │ │ │ │ - * Read WFS DescribeFeatureType response │ │ │ │ │ + * Class: OpenLayers.Format.WMSDescribeLayer │ │ │ │ │ + * Read SLD WMS DescribeLayer response │ │ │ │ │ + * DescribeLayer is meant to couple WMS to WFS and WCS │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WFSDescribeFeatureType = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.XML, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ - */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g) │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - xsd: "http://www.w3.org/2001/XMLSchema" │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WFSDescribeFeatureType │ │ │ │ │ - * Create a new parser for WFS DescribeFeatureType responses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "xsd": { │ │ │ │ │ - "schema": function(node, obj) { │ │ │ │ │ - var complexTypes = []; │ │ │ │ │ - var customTypes = {}; │ │ │ │ │ - var schema = { │ │ │ │ │ - complexTypes: complexTypes, │ │ │ │ │ - customTypes: customTypes │ │ │ │ │ - }; │ │ │ │ │ - var i, len; │ │ │ │ │ - │ │ │ │ │ - this.readChildNodes(node, schema); │ │ │ │ │ - │ │ │ │ │ - var attributes = node.attributes; │ │ │ │ │ - var attr, name; │ │ │ │ │ - for (i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ - attr = attributes[i]; │ │ │ │ │ - name = attr.name; │ │ │ │ │ - if (name.indexOf("xmlns") === 0) { │ │ │ │ │ - this.setNamespace(name.split(":")[1] || "", attr.value); │ │ │ │ │ - } else { │ │ │ │ │ - obj[name] = attr.value; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - obj.featureTypes = complexTypes; │ │ │ │ │ - obj.targetPrefix = this.namespaceAlias[obj.targetNamespace]; │ │ │ │ │ - │ │ │ │ │ - // map complexTypes to names of customTypes │ │ │ │ │ - var complexType, customType; │ │ │ │ │ - for (i = 0, len = complexTypes.length; i < len; ++i) { │ │ │ │ │ - complexType = complexTypes[i]; │ │ │ │ │ - customType = customTypes[complexType.typeName]; │ │ │ │ │ - if (customTypes[complexType.typeName]) { │ │ │ │ │ - complexType.typeName = customType.name; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "complexType": function(node, obj) { │ │ │ │ │ - var complexType = { │ │ │ │ │ - // this is a temporary typeName, it will be overwritten by │ │ │ │ │ - // the schema reader with the metadata found in the │ │ │ │ │ - // customTypes hash │ │ │ │ │ - "typeName": node.getAttribute("name") │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, complexType); │ │ │ │ │ - obj.complexTypes.push(complexType); │ │ │ │ │ - }, │ │ │ │ │ - "complexContent": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "extension": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "sequence": function(node, obj) { │ │ │ │ │ - var sequence = { │ │ │ │ │ - elements: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, sequence); │ │ │ │ │ - obj.properties = sequence.elements; │ │ │ │ │ - }, │ │ │ │ │ - "element": function(node, obj) { │ │ │ │ │ - var type; │ │ │ │ │ - if (obj.elements) { │ │ │ │ │ - var element = {}; │ │ │ │ │ - var attributes = node.attributes; │ │ │ │ │ - var attr; │ │ │ │ │ - for (var i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ - attr = attributes[i]; │ │ │ │ │ - element[attr.name] = attr.value; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - type = element.type; │ │ │ │ │ - if (!type) { │ │ │ │ │ - type = {}; │ │ │ │ │ - this.readChildNodes(node, type); │ │ │ │ │ - element.restriction = type; │ │ │ │ │ - element.type = type.base; │ │ │ │ │ - } │ │ │ │ │ - var fullType = type.base || type; │ │ │ │ │ - element.localType = fullType.split(":").pop(); │ │ │ │ │ - obj.elements.push(element); │ │ │ │ │ - this.readChildNodes(node, element); │ │ │ │ │ - } │ │ │ │ │ +OpenLayers.Format.WMSDescribeLayer = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ │ │ │ │ │ - if (obj.complexTypes) { │ │ │ │ │ - type = node.getAttribute("type"); │ │ │ │ │ - var localType = type.split(":").pop(); │ │ │ │ │ - obj.customTypes[localType] = { │ │ │ │ │ - "name": node.getAttribute("name"), │ │ │ │ │ - "type": type │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "annotation": function(node, obj) { │ │ │ │ │ - obj.annotation = {}; │ │ │ │ │ - this.readChildNodes(node, obj.annotation); │ │ │ │ │ - }, │ │ │ │ │ - "appinfo": function(node, obj) { │ │ │ │ │ - if (!obj.appinfo) { │ │ │ │ │ - obj.appinfo = []; │ │ │ │ │ - } │ │ │ │ │ - obj.appinfo.push(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "documentation": function(node, obj) { │ │ │ │ │ - if (!obj.documentation) { │ │ │ │ │ - obj.documentation = []; │ │ │ │ │ - } │ │ │ │ │ - var value = this.getChildValue(node); │ │ │ │ │ - obj.documentation.push({ │ │ │ │ │ - lang: node.getAttribute("xml:lang"), │ │ │ │ │ - textContent: value.replace(this.regExes.trimSpace, "") │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "simpleType": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "restriction": function(node, obj) { │ │ │ │ │ - obj.base = node.getAttribute("base"); │ │ │ │ │ - this.readRestriction(node, obj); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.1.1". │ │ │ │ │ + */ │ │ │ │ │ + defaultVersion: "1.1.1", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: readRestriction │ │ │ │ │ - * Reads restriction defined in the child nodes of a restriction element │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} the node to parse │ │ │ │ │ - * obj - {Object} the object that receives the read result │ │ │ │ │ - */ │ │ │ │ │ - readRestriction: function(node, obj) { │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - var child, nodeName, value; │ │ │ │ │ - for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ - child = children[i]; │ │ │ │ │ - if (child.nodeType == 1) { │ │ │ │ │ - nodeName = child.nodeName.split(":").pop(); │ │ │ │ │ - value = child.getAttribute("value"); │ │ │ │ │ - if (!obj[nodeName]) { │ │ │ │ │ - obj[nodeName] = value; │ │ │ │ │ - } else { │ │ │ │ │ - if (typeof obj[nodeName] == "string") { │ │ │ │ │ - obj[nodeName] = [obj[nodeName]]; │ │ │ │ │ - } │ │ │ │ │ - obj[nodeName].push(value); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMSDescribeLayer │ │ │ │ │ + * Create a new parser for WMS DescribeLayer responses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {DOMElement|String} A WFS DescribeFeatureType document. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object representing the WFS DescribeFeatureType response. │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ - } │ │ │ │ │ - var schema = {}; │ │ │ │ │ - if (data.nodeName.split(":").pop() === 'ExceptionReport') { │ │ │ │ │ - // an exception must have occurred, so parse it │ │ │ │ │ - var parser = new OpenLayers.Format.OGCExceptionReport(); │ │ │ │ │ - schema.error = parser.read(data); │ │ │ │ │ - } else { │ │ │ │ │ - this.readNode(data, schema); │ │ │ │ │ - } │ │ │ │ │ - return schema; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read DescribeLayer data from a string, and return the response. │ │ │ │ │ + * The OGC currently defines 2 formats which are allowed for output, │ │ │ │ │ + * so we need to parse these 2 types │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} Array of {<LayerDescription>} objects which have: │ │ │ │ │ + * - {String} owsType: WFS/WCS │ │ │ │ │ + * - {String} owsURL: the online resource │ │ │ │ │ + * - {String} typeName: the name of the typename on the service │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSDescribeFeatureType" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSDescribeLayer" │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/SLD.js │ │ │ │ │ + OpenLayers/Format/KML.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ - * @requires OpenLayers/Style.js │ │ │ │ │ - * @requires OpenLayers/Rule.js │ │ │ │ │ - * @requires OpenLayers/Filter/FeatureId.js │ │ │ │ │ - * @requires OpenLayers/Filter/Logical.js │ │ │ │ │ - * @requires OpenLayers/Filter/Comparison.js │ │ │ │ │ - * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Date.js │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Collection.js │ │ │ │ │ + * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ + * @requires OpenLayers/Projection.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.SLD │ │ │ │ │ - * Read/Write SLD. Create a new instance with the <OpenLayers.Format.SLD> │ │ │ │ │ - * constructor. │ │ │ │ │ + * Class: OpenLayers.Format.KML │ │ │ │ │ + * Read/Write KML. Create a new instance with the <OpenLayers.Format.KML> │ │ │ │ │ + * constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.SLD = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ +OpenLayers.Format.KML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: profile │ │ │ │ │ - * {String} If provided, use a custom profile. │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + kml: "http://www.opengis.net/kml/2.2", │ │ │ │ │ + gx: "http://www.google.com/kml/ext/2.2" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: kmlns │ │ │ │ │ + * {String} KML Namespace to use. Defaults to 2.0 namespace. │ │ │ │ │ + */ │ │ │ │ │ + kmlns: "http://earth.google.com/kml/2.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: placemarksDesc │ │ │ │ │ + * {String} Name of the placemarks. Default is "No description available". │ │ │ │ │ + */ │ │ │ │ │ + placemarksDesc: "No description available", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: foldersName │ │ │ │ │ + * {String} Name of the folders. Default is "OpenLayers export". │ │ │ │ │ + * If set to null, no name element will be created. │ │ │ │ │ + */ │ │ │ │ │ + foldersName: "OpenLayers export", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: foldersDesc │ │ │ │ │ + * {String} Description of the folders. Default is "Exported on [date]." │ │ │ │ │ + * If set to null, no description element will be created. │ │ │ │ │ + */ │ │ │ │ │ + foldersDesc: "Exported on " + new Date(), │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: extractAttributes │ │ │ │ │ + * {Boolean} Extract attributes from KML. Default is true. │ │ │ │ │ + * Extracting styleUrls requires this to be set to true │ │ │ │ │ + * Note that currently only Data and SimpleData │ │ │ │ │ + * elements are handled. │ │ │ │ │ + */ │ │ │ │ │ + extractAttributes: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: kvpAttributes │ │ │ │ │ + * {Boolean} Only used if extractAttributes is true. │ │ │ │ │ + * If set to true, attributes will be simple │ │ │ │ │ + * key-value pairs, compatible with other formats, │ │ │ │ │ + * Any displayName elements will be ignored. │ │ │ │ │ + * If set to false, attributes will be objects, │ │ │ │ │ + * retaining any displayName elements, but not │ │ │ │ │ + * compatible with other formats. Any CDATA in │ │ │ │ │ + * displayName will be read in as a string value. │ │ │ │ │ + * Default is false. │ │ │ │ │ + */ │ │ │ │ │ + kvpAttributes: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: extractStyles │ │ │ │ │ + * {Boolean} Extract styles from KML. Default is false. │ │ │ │ │ + * Extracting styleUrls also requires extractAttributes to be │ │ │ │ │ + * set to true │ │ │ │ │ + */ │ │ │ │ │ + extractStyles: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: extractTracks │ │ │ │ │ + * {Boolean} Extract gx:Track elements from Placemark elements. Default │ │ │ │ │ + * is false. If true, features will be generated for all points in │ │ │ │ │ + * all gx:Track elements. Features will have a when (Date) attribute │ │ │ │ │ + * based on when elements in the track. If tracks include angle │ │ │ │ │ + * elements, features will have heading, tilt, and roll attributes. │ │ │ │ │ + * If track point coordinates have three values, features will have │ │ │ │ │ + * an altitude attribute with the third coordinate value. │ │ │ │ │ + */ │ │ │ │ │ + extractTracks: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: trackAttributes │ │ │ │ │ + * {Array} If <extractTracks> is true, points within gx:Track elements will │ │ │ │ │ + * be parsed as features with when, heading, tilt, and roll attributes. │ │ │ │ │ + * Any additional attribute names can be provided in <trackAttributes>. │ │ │ │ │ + */ │ │ │ │ │ + trackAttributes: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: internalns │ │ │ │ │ + * {String} KML Namespace to use -- defaults to the namespace of the │ │ │ │ │ + * Placemark node being parsed, but falls back to kmlns. │ │ │ │ │ + */ │ │ │ │ │ + internalns: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Array} Array of features │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + features: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: styles │ │ │ │ │ + * {Object} Storage of style objects │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + styles: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: styleBaseUrl │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + styleBaseUrl: "", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: fetched │ │ │ │ │ + * {Object} Storage of KML URLs that have been fetched before │ │ │ │ │ + * in order to prevent reloading them. │ │ │ │ │ + */ │ │ │ │ │ + fetched: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxDepth │ │ │ │ │ + * {Integer} Maximum depth for recursive loading external KML URLs │ │ │ │ │ + * Defaults to 0: do no external fetching │ │ │ │ │ + */ │ │ │ │ │ + maxDepth: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.KML │ │ │ │ │ + * Create a new parser for KML. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + // compile regular expressions once instead of every time they are used │ │ │ │ │ + this.regExes = { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g), │ │ │ │ │ + kmlColor: (/(\w{2})(\w{2})(\w{2})(\w{2})/), │ │ │ │ │ + kmlIconPalette: (/root:\/\/icons\/palette-(\d+)(\.\w+)/), │ │ │ │ │ + straightBracket: (/\$\[(.*?)\]/g) │ │ │ │ │ + }; │ │ │ │ │ + // KML coordinates are always in longlat WGS84 │ │ │ │ │ + this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read data from a string, and return a list of features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} List of features. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + this.features = []; │ │ │ │ │ + this.styles = {}; │ │ │ │ │ + this.fetched = {}; │ │ │ │ │ + │ │ │ │ │ + // Set default options │ │ │ │ │ + var options = { │ │ │ │ │ + depth: 0, │ │ │ │ │ + styleBaseUrl: this.styleBaseUrl │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + return this.parseData(data, options); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseData │ │ │ │ │ + * Read data from a string, and return a list of features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * options - {Object} Hash of options │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} List of features. │ │ │ │ │ + */ │ │ │ │ │ + parseData: function(data, options) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Loop throught the following node types in this order and │ │ │ │ │ + // process the nodes found │ │ │ │ │ + var types = ["Link", "NetworkLink", "Style", "StyleMap", "Placemark"]; │ │ │ │ │ + for (var i = 0, len = types.length; i < len; ++i) { │ │ │ │ │ + var type = types[i]; │ │ │ │ │ + │ │ │ │ │ + var nodes = this.getElementsByTagNameNS(data, "*", type); │ │ │ │ │ + │ │ │ │ │ + // skip to next type if no nodes are found │ │ │ │ │ + if (nodes.length == 0) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + switch (type.toLowerCase()) { │ │ │ │ │ + │ │ │ │ │ + // Fetch external links │ │ │ │ │ + case "link": │ │ │ │ │ + case "networklink": │ │ │ │ │ + this.parseLinks(nodes, options); │ │ │ │ │ + break; │ │ │ │ │ + │ │ │ │ │ + // parse style information │ │ │ │ │ + case "style": │ │ │ │ │ + if (this.extractStyles) { │ │ │ │ │ + this.parseStyles(nodes, options); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "stylemap": │ │ │ │ │ + if (this.extractStyles) { │ │ │ │ │ + this.parseStyleMaps(nodes, options); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + │ │ │ │ │ + // parse features │ │ │ │ │ + case "placemark": │ │ │ │ │ + this.parseFeatures(nodes, options); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return this.features; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseLinks │ │ │ │ │ + * Finds URLs of linked KML documents and fetches them │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * nodes - {Array} of {DOMElement} data to read/parse. │ │ │ │ │ + * options - {Object} Hash of options │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + parseLinks: function(nodes, options) { │ │ │ │ │ + │ │ │ │ │ + // Fetch external links <NetworkLink> and <Link> │ │ │ │ │ + // Don't do anything if we have reached our maximum depth for recursion │ │ │ │ │ + if (options.depth >= this.maxDepth) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // increase depth │ │ │ │ │ + var newOptions = OpenLayers.Util.extend({}, options); │ │ │ │ │ + newOptions.depth++; │ │ │ │ │ + │ │ │ │ │ + for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ + var href = this.parseProperty(nodes[i], "*", "href"); │ │ │ │ │ + if (href && !this.fetched[href]) { │ │ │ │ │ + this.fetched[href] = true; // prevent reloading the same urls │ │ │ │ │ + var data = this.fetchLink(href); │ │ │ │ │ + if (data) { │ │ │ │ │ + this.parseData(data, newOptions); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: fetchLink │ │ │ │ │ + * Fetches a URL and returns the result │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * href - {String} url to be fetched │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + fetchLink: function(href) { │ │ │ │ │ + var request = OpenLayers.Request.GET({ │ │ │ │ │ + url: href, │ │ │ │ │ + async: false │ │ │ │ │ + }); │ │ │ │ │ + if (request) { │ │ │ │ │ + return request.responseText; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseStyles │ │ │ │ │ + * Parses <Style> nodes │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * nodes - {Array} of {DOMElement} data to read/parse. │ │ │ │ │ + * options - {Object} Hash of options │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + parseStyles: function(nodes, options) { │ │ │ │ │ + for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ + var style = this.parseStyle(nodes[i]); │ │ │ │ │ + if (style) { │ │ │ │ │ + var styleName = (options.styleBaseUrl || "") + "#" + style.id; │ │ │ │ │ + │ │ │ │ │ + this.styles[styleName] = style; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseKmlColor │ │ │ │ │ + * Parses a kml color (in 'aabbggrr' format) and returns the corresponding │ │ │ │ │ + * color and opacity or null if the color is invalid. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * kmlColor - {String} a kml formated color │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ + parseKmlColor: function(kmlColor) { │ │ │ │ │ + var color = null; │ │ │ │ │ + if (kmlColor) { │ │ │ │ │ + var matches = kmlColor.match(this.regExes.kmlColor); │ │ │ │ │ + if (matches) { │ │ │ │ │ + color = { │ │ │ │ │ + color: '#' + matches[4] + matches[3] + matches[2], │ │ │ │ │ + opacity: parseInt(matches[1], 16) / 255 │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return color; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseStyle │ │ │ │ │ + * Parses the children of a <Style> node and builds the style hash │ │ │ │ │ + * accordingly │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} <Style> node │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + parseStyle: function(node) { │ │ │ │ │ + var style = {}; │ │ │ │ │ + │ │ │ │ │ + var types = ["LineStyle", "PolyStyle", "IconStyle", "BalloonStyle", │ │ │ │ │ + "LabelStyle" │ │ │ │ │ + ]; │ │ │ │ │ + var type, styleTypeNode, nodeList, geometry, parser; │ │ │ │ │ + for (var i = 0, len = types.length; i < len; ++i) { │ │ │ │ │ + type = types[i]; │ │ │ │ │ + styleTypeNode = this.getElementsByTagNameNS(node, "*", type)[0]; │ │ │ │ │ + if (!styleTypeNode) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // only deal with first geometry of this type │ │ │ │ │ + switch (type.toLowerCase()) { │ │ │ │ │ + case "linestyle": │ │ │ │ │ + var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ + var color = this.parseKmlColor(kmlColor); │ │ │ │ │ + if (color) { │ │ │ │ │ + style["strokeColor"] = color.color; │ │ │ │ │ + style["strokeOpacity"] = color.opacity; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var width = this.parseProperty(styleTypeNode, "*", "width"); │ │ │ │ │ + if (width) { │ │ │ │ │ + style["strokeWidth"] = width; │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + │ │ │ │ │ + case "polystyle": │ │ │ │ │ + var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ + var color = this.parseKmlColor(kmlColor); │ │ │ │ │ + if (color) { │ │ │ │ │ + style["fillOpacity"] = color.opacity; │ │ │ │ │ + style["fillColor"] = color.color; │ │ │ │ │ + } │ │ │ │ │ + // Check if fill is disabled │ │ │ │ │ + var fill = this.parseProperty(styleTypeNode, "*", "fill"); │ │ │ │ │ + if (fill == "0") { │ │ │ │ │ + style["fillColor"] = "none"; │ │ │ │ │ + } │ │ │ │ │ + // Check if outline is disabled │ │ │ │ │ + var outline = this.parseProperty(styleTypeNode, "*", "outline"); │ │ │ │ │ + if (outline == "0") { │ │ │ │ │ + style["strokeWidth"] = "0"; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + break; │ │ │ │ │ + │ │ │ │ │ + case "iconstyle": │ │ │ │ │ + // set scale │ │ │ │ │ + var scale = parseFloat(this.parseProperty(styleTypeNode, │ │ │ │ │ + "*", "scale") || 1); │ │ │ │ │ + │ │ │ │ │ + // set default width and height of icon │ │ │ │ │ + var width = 32 * scale; │ │ │ │ │ + var height = 32 * scale; │ │ │ │ │ + │ │ │ │ │ + var iconNode = this.getElementsByTagNameNS(styleTypeNode, │ │ │ │ │ + "*", │ │ │ │ │ + "Icon")[0]; │ │ │ │ │ + if (iconNode) { │ │ │ │ │ + var href = this.parseProperty(iconNode, "*", "href"); │ │ │ │ │ + if (href) { │ │ │ │ │ + │ │ │ │ │ + var w = this.parseProperty(iconNode, "*", "w"); │ │ │ │ │ + var h = this.parseProperty(iconNode, "*", "h"); │ │ │ │ │ + │ │ │ │ │ + // Settings for Google specific icons that are 64x64 │ │ │ │ │ + // We set the width and height to 64 and halve the │ │ │ │ │ + // scale to prevent icons from being too big │ │ │ │ │ + var google = "http://maps.google.com/mapfiles/kml"; │ │ │ │ │ + if (OpenLayers.String.startsWith( │ │ │ │ │ + href, google) && !w && !h) { │ │ │ │ │ + w = 64; │ │ │ │ │ + h = 64; │ │ │ │ │ + scale = scale / 2; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // if only dimension is defined, make sure the │ │ │ │ │ + // other one has the same value │ │ │ │ │ + w = w || h; │ │ │ │ │ + h = h || w; │ │ │ │ │ + │ │ │ │ │ + if (w) { │ │ │ │ │ + width = parseInt(w) * scale; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (h) { │ │ │ │ │ + height = parseInt(h) * scale; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // support for internal icons │ │ │ │ │ + // (/root://icons/palette-x.png) │ │ │ │ │ + // x and y tell the position on the palette: │ │ │ │ │ + // - in pixels │ │ │ │ │ + // - starting from the left bottom │ │ │ │ │ + // We translate that to a position in the list │ │ │ │ │ + // and request the appropriate icon from the │ │ │ │ │ + // google maps website │ │ │ │ │ + var matches = href.match(this.regExes.kmlIconPalette); │ │ │ │ │ + if (matches) { │ │ │ │ │ + var palette = matches[1]; │ │ │ │ │ + var file_extension = matches[2]; │ │ │ │ │ + │ │ │ │ │ + var x = this.parseProperty(iconNode, "*", "x"); │ │ │ │ │ + var y = this.parseProperty(iconNode, "*", "y"); │ │ │ │ │ + │ │ │ │ │ + var posX = x ? x / 32 : 0; │ │ │ │ │ + var posY = y ? (7 - y / 32) : 7; │ │ │ │ │ + │ │ │ │ │ + var pos = posY * 8 + posX; │ │ │ │ │ + href = "http://maps.google.com/mapfiles/kml/pal" + │ │ │ │ │ + palette + "/icon" + pos + file_extension; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + style["graphicOpacity"] = 1; // fully opaque │ │ │ │ │ + style["externalGraphic"] = href; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + // hotSpots define the offset for an Icon │ │ │ │ │ + var hotSpotNode = this.getElementsByTagNameNS(styleTypeNode, │ │ │ │ │ + "*", │ │ │ │ │ + "hotSpot")[0]; │ │ │ │ │ + if (hotSpotNode) { │ │ │ │ │ + var x = parseFloat(hotSpotNode.getAttribute("x")); │ │ │ │ │ + var y = parseFloat(hotSpotNode.getAttribute("y")); │ │ │ │ │ + │ │ │ │ │ + var xUnits = hotSpotNode.getAttribute("xunits"); │ │ │ │ │ + if (xUnits == "pixels") { │ │ │ │ │ + style["graphicXOffset"] = -x * scale; │ │ │ │ │ + } else if (xUnits == "insetPixels") { │ │ │ │ │ + style["graphicXOffset"] = -width + (x * scale); │ │ │ │ │ + } else if (xUnits == "fraction") { │ │ │ │ │ + style["graphicXOffset"] = -width * x; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var yUnits = hotSpotNode.getAttribute("yunits"); │ │ │ │ │ + if (yUnits == "pixels") { │ │ │ │ │ + style["graphicYOffset"] = -height + (y * scale) + 1; │ │ │ │ │ + } else if (yUnits == "insetPixels") { │ │ │ │ │ + style["graphicYOffset"] = -(y * scale) + 1; │ │ │ │ │ + } else if (yUnits == "fraction") { │ │ │ │ │ + style["graphicYOffset"] = -height * (1 - y) + 1; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + style["graphicWidth"] = width; │ │ │ │ │ + style["graphicHeight"] = height; │ │ │ │ │ + break; │ │ │ │ │ + │ │ │ │ │ + case "balloonstyle": │ │ │ │ │ + var balloonStyle = OpenLayers.Util.getXmlNodeValue( │ │ │ │ │ + styleTypeNode); │ │ │ │ │ + if (balloonStyle) { │ │ │ │ │ + style["balloonStyle"] = balloonStyle.replace( │ │ │ │ │ + this.regExes.straightBracket, "${$1}"); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "labelstyle": │ │ │ │ │ + var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ + var color = this.parseKmlColor(kmlColor); │ │ │ │ │ + if (color) { │ │ │ │ │ + style["fontColor"] = color.color; │ │ │ │ │ + style["fontOpacity"] = color.opacity; │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + │ │ │ │ │ + default: │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Some polygons have no line color, so we use the fillColor for that │ │ │ │ │ + if (!style["strokeColor"] && style["fillColor"]) { │ │ │ │ │ + style["strokeColor"] = style["fillColor"]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var id = node.getAttribute("id"); │ │ │ │ │ + if (id && style) { │ │ │ │ │ + style.id = id; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return style; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseStyleMaps │ │ │ │ │ + * Parses <StyleMap> nodes, but only uses the 'normal' key │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * nodes - {Array} of {DOMElement} data to read/parse. │ │ │ │ │ + * options - {Object} Hash of options │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + parseStyleMaps: function(nodes, options) { │ │ │ │ │ + // Only the default or "normal" part of the StyleMap is processed now │ │ │ │ │ + // To do the select or "highlight" bit, we'd need to change lots more │ │ │ │ │ + │ │ │ │ │ + for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ + var node = nodes[i]; │ │ │ │ │ + var pairs = this.getElementsByTagNameNS(node, "*", │ │ │ │ │ + "Pair"); │ │ │ │ │ + │ │ │ │ │ + var id = node.getAttribute("id"); │ │ │ │ │ + for (var j = 0, jlen = pairs.length; j < jlen; j++) { │ │ │ │ │ + var pair = pairs[j]; │ │ │ │ │ + // Use the shortcut in the SLD format to quickly retrieve the │ │ │ │ │ + // value of a node. Maybe it's good to have a method in │ │ │ │ │ + // Format.XML to do this │ │ │ │ │ + var key = this.parseProperty(pair, "*", "key"); │ │ │ │ │ + var styleUrl = this.parseProperty(pair, "*", "styleUrl"); │ │ │ │ │ + │ │ │ │ │ + if (styleUrl && key == "normal") { │ │ │ │ │ + this.styles[(options.styleBaseUrl || "") + "#" + id] = │ │ │ │ │ + this.styles[(options.styleBaseUrl || "") + styleUrl]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // TODO: implement the "select" part │ │ │ │ │ + //if (styleUrl && key == "highlight") { │ │ │ │ │ + //} │ │ │ │ │ + │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseFeatures │ │ │ │ │ + * Loop through all Placemark nodes and parse them. │ │ │ │ │ + * Will create a list of features │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * nodes - {Array} of {DOMElement} data to read/parse. │ │ │ │ │ + * options - {Object} Hash of options │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + parseFeatures: function(nodes, options) { │ │ │ │ │ + var features = []; │ │ │ │ │ + for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ + var featureNode = nodes[i]; │ │ │ │ │ + var feature = this.parseFeature.apply(this, [featureNode]); │ │ │ │ │ + if (feature) { │ │ │ │ │ + │ │ │ │ │ + // Create reference to styleUrl │ │ │ │ │ + if (this.extractStyles && feature.attributes && │ │ │ │ │ + feature.attributes.styleUrl) { │ │ │ │ │ + feature.style = this.getStyle(feature.attributes.styleUrl, options); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.extractStyles) { │ │ │ │ │ + // Make sure that <Style> nodes within a placemark are │ │ │ │ │ + // processed as well │ │ │ │ │ + var inlineStyleNode = this.getElementsByTagNameNS(featureNode, │ │ │ │ │ + "*", │ │ │ │ │ + "Style")[0]; │ │ │ │ │ + if (inlineStyleNode) { │ │ │ │ │ + var inlineStyle = this.parseStyle(inlineStyleNode); │ │ │ │ │ + if (inlineStyle) { │ │ │ │ │ + feature.style = OpenLayers.Util.extend( │ │ │ │ │ + feature.style, inlineStyle │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // check if gx:Track elements should be parsed │ │ │ │ │ + if (this.extractTracks) { │ │ │ │ │ + var tracks = this.getElementsByTagNameNS( │ │ │ │ │ + featureNode, this.namespaces.gx, "Track" │ │ │ │ │ + ); │ │ │ │ │ + if (tracks && tracks.length > 0) { │ │ │ │ │ + var track = tracks[0]; │ │ │ │ │ + var container = { │ │ │ │ │ + features: [], │ │ │ │ │ + feature: feature │ │ │ │ │ + }; │ │ │ │ │ + this.readNode(track, container); │ │ │ │ │ + if (container.features.length > 0) { │ │ │ │ │ + features.push.apply(features, container.features); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // add feature to list of features │ │ │ │ │ + features.push(feature); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + throw "Bad Placemark: " + i; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // add new features to existing feature list │ │ │ │ │ + this.features = this.features.concat(features); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "kml": { │ │ │ │ │ + "when": function(node, container) { │ │ │ │ │ + container.whens.push(OpenLayers.Date.parse( │ │ │ │ │ + this.getChildValue(node) │ │ │ │ │ + )); │ │ │ │ │ + }, │ │ │ │ │ + "_trackPointAttribute": function(node, container) { │ │ │ │ │ + var name = node.nodeName.split(":").pop(); │ │ │ │ │ + container.attributes[name].push(this.getChildValue(node)); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "gx": { │ │ │ │ │ + "Track": function(node, container) { │ │ │ │ │ + var obj = { │ │ │ │ │ + whens: [], │ │ │ │ │ + points: [], │ │ │ │ │ + angles: [] │ │ │ │ │ + }; │ │ │ │ │ + if (this.trackAttributes) { │ │ │ │ │ + var name; │ │ │ │ │ + obj.attributes = {}; │ │ │ │ │ + for (var i = 0, ii = this.trackAttributes.length; i < ii; ++i) { │ │ │ │ │ + name = this.trackAttributes[i]; │ │ │ │ │ + obj.attributes[name] = []; │ │ │ │ │ + if (!(name in this.readers.kml)) { │ │ │ │ │ + this.readers.kml[name] = this.readers.kml._trackPointAttribute; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + if (obj.whens.length !== obj.points.length) { │ │ │ │ │ + throw new Error("gx:Track with unequal number of when (" + │ │ │ │ │ + obj.whens.length + ") and gx:coord (" + │ │ │ │ │ + obj.points.length + ") elements."); │ │ │ │ │ + } │ │ │ │ │ + var hasAngles = obj.angles.length > 0; │ │ │ │ │ + if (hasAngles && obj.whens.length !== obj.angles.length) { │ │ │ │ │ + throw new Error("gx:Track with unequal number of when (" + │ │ │ │ │ + obj.whens.length + ") and gx:angles (" + │ │ │ │ │ + obj.angles.length + ") elements."); │ │ │ │ │ + } │ │ │ │ │ + var feature, point, angles; │ │ │ │ │ + for (var i = 0, ii = obj.whens.length; i < ii; ++i) { │ │ │ │ │ + feature = container.feature.clone(); │ │ │ │ │ + feature.fid = container.feature.fid || container.feature.id; │ │ │ │ │ + point = obj.points[i]; │ │ │ │ │ + feature.geometry = point; │ │ │ │ │ + if ("z" in point) { │ │ │ │ │ + feature.attributes.altitude = point.z; │ │ │ │ │ + } │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + feature.geometry.transform( │ │ │ │ │ + this.externalProjection, this.internalProjection │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (this.trackAttributes) { │ │ │ │ │ + for (var j = 0, jj = this.trackAttributes.length; j < jj; ++j) { │ │ │ │ │ + var name = this.trackAttributes[j]; │ │ │ │ │ + feature.attributes[name] = obj.attributes[name][i]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + feature.attributes.when = obj.whens[i]; │ │ │ │ │ + feature.attributes.trackId = container.feature.id; │ │ │ │ │ + if (hasAngles) { │ │ │ │ │ + angles = obj.angles[i]; │ │ │ │ │ + feature.attributes.heading = parseFloat(angles[0]); │ │ │ │ │ + feature.attributes.tilt = parseFloat(angles[1]); │ │ │ │ │ + feature.attributes.roll = parseFloat(angles[2]); │ │ │ │ │ + } │ │ │ │ │ + container.features.push(feature); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "coord": function(node, container) { │ │ │ │ │ + var str = this.getChildValue(node); │ │ │ │ │ + var coords = str.replace(this.regExes.trimSpace, "").split(/\s+/); │ │ │ │ │ + var point = new OpenLayers.Geometry.Point(coords[0], coords[1]); │ │ │ │ │ + if (coords.length > 2) { │ │ │ │ │ + point.z = parseFloat(coords[2]); │ │ │ │ │ + } │ │ │ │ │ + container.points.push(point); │ │ │ │ │ + }, │ │ │ │ │ + "angles": function(node, container) { │ │ │ │ │ + var str = this.getChildValue(node); │ │ │ │ │ + var parts = str.replace(this.regExes.trimSpace, "").split(/\s+/); │ │ │ │ │ + container.angles.push(parts); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseFeature │ │ │ │ │ + * This function is the core of the KML parsing code in OpenLayers. │ │ │ │ │ + * It creates the geometries that are then attached to the returned │ │ │ │ │ + * feature, and calls parseAttributes() to get attribute data out. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A vector feature. │ │ │ │ │ + */ │ │ │ │ │ + parseFeature: function(node) { │ │ │ │ │ + // only accept one geometry per feature - look for highest "order" │ │ │ │ │ + var order = ["MultiGeometry", "Polygon", "LineString", "Point"]; │ │ │ │ │ + var type, nodeList, geometry, parser; │ │ │ │ │ + for (var i = 0, len = order.length; i < len; ++i) { │ │ │ │ │ + type = order[i]; │ │ │ │ │ + this.internalns = node.namespaceURI ? │ │ │ │ │ + node.namespaceURI : this.kmlns; │ │ │ │ │ + nodeList = this.getElementsByTagNameNS(node, │ │ │ │ │ + this.internalns, type); │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + // only deal with first geometry of this type │ │ │ │ │ + var parser = this.parseGeometry[type.toLowerCase()]; │ │ │ │ │ + if (parser) { │ │ │ │ │ + geometry = parser.apply(this, [nodeList[0]]); │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry.transform(this.externalProjection, │ │ │ │ │ + this.internalProjection); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + throw new TypeError("Unsupported geometry type: " + type); │ │ │ │ │ + } │ │ │ │ │ + // stop looking for different geometry types │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // construct feature (optionally with attributes) │ │ │ │ │ + var attributes; │ │ │ │ │ + if (this.extractAttributes) { │ │ │ │ │ + attributes = this.parseAttributes(node); │ │ │ │ │ + } │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry, attributes); │ │ │ │ │ + │ │ │ │ │ + var fid = node.getAttribute("id") || node.getAttribute("name"); │ │ │ │ │ + if (fid != null) { │ │ │ │ │ + feature.fid = fid; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return feature; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getStyle │ │ │ │ │ + * Retrieves a style from a style hash using styleUrl as the key │ │ │ │ │ + * If the styleUrl doesn't exist yet, we try to fetch it │ │ │ │ │ + * Internet │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * styleUrl - {String} URL of style │ │ │ │ │ + * options - {Object} Hash of options │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} - (reference to) Style hash │ │ │ │ │ + */ │ │ │ │ │ + getStyle: function(styleUrl, options) { │ │ │ │ │ + │ │ │ │ │ + var styleBaseUrl = OpenLayers.Util.removeTail(styleUrl); │ │ │ │ │ + │ │ │ │ │ + var newOptions = OpenLayers.Util.extend({}, options); │ │ │ │ │ + newOptions.depth++; │ │ │ │ │ + newOptions.styleBaseUrl = styleBaseUrl; │ │ │ │ │ + │ │ │ │ │ + // Fetch remote Style URLs (if not fetched before) │ │ │ │ │ + if (!this.styles[styleUrl] && │ │ │ │ │ + !OpenLayers.String.startsWith(styleUrl, "#") && │ │ │ │ │ + newOptions.depth <= this.maxDepth && │ │ │ │ │ + !this.fetched[styleBaseUrl]) { │ │ │ │ │ + │ │ │ │ │ + var data = this.fetchLink(styleBaseUrl); │ │ │ │ │ + if (data) { │ │ │ │ │ + this.parseData(data, newOptions); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // return requested style │ │ │ │ │ + var style = OpenLayers.Util.extend({}, this.styles[styleUrl]); │ │ │ │ │ + return style; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: parseGeometry │ │ │ │ │ + * Properties of this object are the functions that parse geometries based │ │ │ │ │ + * on their type. │ │ │ │ │ + */ │ │ │ │ │ + parseGeometry: { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseGeometry.point │ │ │ │ │ + * Given a KML node representing a point geometry, create an OpenLayers │ │ │ │ │ + * point geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A KML Point node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Point>} A point geometry. │ │ │ │ │ + */ │ │ │ │ │ + point: function(node) { │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, this.internalns, │ │ │ │ │ + "coordinates"); │ │ │ │ │ + var coords = []; │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + var coordString = nodeList[0].firstChild.nodeValue; │ │ │ │ │ + coordString = coordString.replace(this.regExes.removeSpace, ""); │ │ │ │ │ + coords = coordString.split(","); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var point = null; │ │ │ │ │ + if (coords.length > 1) { │ │ │ │ │ + // preserve third dimension │ │ │ │ │ + if (coords.length == 2) { │ │ │ │ │ + coords[2] = null; │ │ │ │ │ + } │ │ │ │ │ + point = new OpenLayers.Geometry.Point(coords[0], coords[1], │ │ │ │ │ + coords[2]); │ │ │ │ │ + } else { │ │ │ │ │ + throw "Bad coordinate string: " + coordString; │ │ │ │ │ + } │ │ │ │ │ + return point; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseGeometry.linestring │ │ │ │ │ + * Given a KML node representing a linestring geometry, create an │ │ │ │ │ + * OpenLayers linestring geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A KML LineString node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.LineString>} A linestring geometry. │ │ │ │ │ + */ │ │ │ │ │ + linestring: function(node, ring) { │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, this.internalns, │ │ │ │ │ + "coordinates"); │ │ │ │ │ + var line = null; │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + var coordString = this.getChildValue(nodeList[0]); │ │ │ │ │ + │ │ │ │ │ + coordString = coordString.replace(this.regExes.trimSpace, │ │ │ │ │ + ""); │ │ │ │ │ + coordString = coordString.replace(this.regExes.trimComma, │ │ │ │ │ + ","); │ │ │ │ │ + var pointList = coordString.split(this.regExes.splitSpace); │ │ │ │ │ + var numPoints = pointList.length; │ │ │ │ │ + var points = new Array(numPoints); │ │ │ │ │ + var coords, numCoords; │ │ │ │ │ + for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ + coords = pointList[i].split(","); │ │ │ │ │ + numCoords = coords.length; │ │ │ │ │ + if (numCoords > 1) { │ │ │ │ │ + if (coords.length == 2) { │ │ │ │ │ + coords[2] = null; │ │ │ │ │ + } │ │ │ │ │ + points[i] = new OpenLayers.Geometry.Point(coords[0], │ │ │ │ │ + coords[1], │ │ │ │ │ + coords[2]); │ │ │ │ │ + } else { │ │ │ │ │ + throw "Bad LineString point coordinates: " + │ │ │ │ │ + pointList[i]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (numPoints) { │ │ │ │ │ + if (ring) { │ │ │ │ │ + line = new OpenLayers.Geometry.LinearRing(points); │ │ │ │ │ + } else { │ │ │ │ │ + line = new OpenLayers.Geometry.LineString(points); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + throw "Bad LineString coordinates: " + coordString; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return line; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseGeometry.polygon │ │ │ │ │ + * Given a KML node representing a polygon geometry, create an │ │ │ │ │ + * OpenLayers polygon geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A KML Polygon node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Polygon>} A polygon geometry. │ │ │ │ │ + */ │ │ │ │ │ + polygon: function(node) { │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, this.internalns, │ │ │ │ │ + "LinearRing"); │ │ │ │ │ + var numRings = nodeList.length; │ │ │ │ │ + var components = new Array(numRings); │ │ │ │ │ + if (numRings > 0) { │ │ │ │ │ + // this assumes exterior ring first, inner rings after │ │ │ │ │ + var ring; │ │ │ │ │ + for (var i = 0, len = nodeList.length; i < len; ++i) { │ │ │ │ │ + ring = this.parseGeometry.linestring.apply(this, │ │ │ │ │ + [nodeList[i], true]); │ │ │ │ │ + if (ring) { │ │ │ │ │ + components[i] = ring; │ │ │ │ │ + } else { │ │ │ │ │ + throw "Bad LinearRing geometry: " + i; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.Polygon(components); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseGeometry.multigeometry │ │ │ │ │ + * Given a KML node representing a multigeometry, create an │ │ │ │ │ + * OpenLayers geometry collection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A KML MultiGeometry node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Collection>} A geometry collection. │ │ │ │ │ + */ │ │ │ │ │ + multigeometry: function(node) { │ │ │ │ │ + var child, parser; │ │ │ │ │ + var parts = []; │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ + child = children[i]; │ │ │ │ │ + if (child.nodeType == 1) { │ │ │ │ │ + var type = (child.prefix) ? │ │ │ │ │ + child.nodeName.split(":")[1] : │ │ │ │ │ + child.nodeName; │ │ │ │ │ + var parser = this.parseGeometry[type.toLowerCase()]; │ │ │ │ │ + if (parser) { │ │ │ │ │ + parts.push(parser.apply(this, [child])); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.Collection(parts); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseAttributes │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An attributes object. │ │ │ │ │ + */ │ │ │ │ │ + parseAttributes: function(node) { │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + │ │ │ │ │ + // Extended Data is parsed first. │ │ │ │ │ + var edNodes = node.getElementsByTagName("ExtendedData"); │ │ │ │ │ + if (edNodes.length) { │ │ │ │ │ + attributes = this.parseExtendedData(edNodes[0]); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // assume attribute nodes are type 1 children with a type 3 or 4 child │ │ │ │ │ + var child, grandchildren, grandchild; │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + │ │ │ │ │ + for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ + child = children[i]; │ │ │ │ │ + if (child.nodeType == 1) { │ │ │ │ │ + grandchildren = child.childNodes; │ │ │ │ │ + if (grandchildren.length >= 1 && grandchildren.length <= 3) { │ │ │ │ │ + var grandchild; │ │ │ │ │ + switch (grandchildren.length) { │ │ │ │ │ + case 1: │ │ │ │ │ + grandchild = grandchildren[0]; │ │ │ │ │ + break; │ │ │ │ │ + case 2: │ │ │ │ │ + var c1 = grandchildren[0]; │ │ │ │ │ + var c2 = grandchildren[1]; │ │ │ │ │ + grandchild = (c1.nodeType == 3 || c1.nodeType == 4) ? │ │ │ │ │ + c1 : c2; │ │ │ │ │ + break; │ │ │ │ │ + case 3: │ │ │ │ │ + default: │ │ │ │ │ + grandchild = grandchildren[1]; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + if (grandchild.nodeType == 3 || grandchild.nodeType == 4) { │ │ │ │ │ + var name = (child.prefix) ? │ │ │ │ │ + child.nodeName.split(":")[1] : │ │ │ │ │ + child.nodeName; │ │ │ │ │ + var value = OpenLayers.Util.getXmlNodeValue(grandchild); │ │ │ │ │ + if (value) { │ │ │ │ │ + value = value.replace(this.regExes.trimSpace, ""); │ │ │ │ │ + attributes[name] = value; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return attributes; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseExtendedData │ │ │ │ │ + * Parse ExtendedData from KML. Limited support for schemas/datatypes. │ │ │ │ │ + * See http://code.google.com/apis/kml/documentation/kmlreference.html#extendeddata │ │ │ │ │ + * for more information on extendeddata. │ │ │ │ │ + */ │ │ │ │ │ + parseExtendedData: function(node) { │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + var i, len, data, key; │ │ │ │ │ + var dataNodes = node.getElementsByTagName("Data"); │ │ │ │ │ + for (i = 0, len = dataNodes.length; i < len; i++) { │ │ │ │ │ + data = dataNodes[i]; │ │ │ │ │ + key = data.getAttribute("name"); │ │ │ │ │ + var ed = {}; │ │ │ │ │ + var valueNode = data.getElementsByTagName("value"); │ │ │ │ │ + if (valueNode.length) { │ │ │ │ │ + ed['value'] = this.getChildValue(valueNode[0]); │ │ │ │ │ + } │ │ │ │ │ + if (this.kvpAttributes) { │ │ │ │ │ + attributes[key] = ed['value']; │ │ │ │ │ + } else { │ │ │ │ │ + var nameNode = data.getElementsByTagName("displayName"); │ │ │ │ │ + if (nameNode.length) { │ │ │ │ │ + ed['displayName'] = this.getChildValue(nameNode[0]); │ │ │ │ │ + } │ │ │ │ │ + attributes[key] = ed; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var simpleDataNodes = node.getElementsByTagName("SimpleData"); │ │ │ │ │ + for (i = 0, len = simpleDataNodes.length; i < len; i++) { │ │ │ │ │ + var ed = {}; │ │ │ │ │ + data = simpleDataNodes[i]; │ │ │ │ │ + key = data.getAttribute("name"); │ │ │ │ │ + ed['value'] = this.getChildValue(data); │ │ │ │ │ + if (this.kvpAttributes) { │ │ │ │ │ + attributes[key] = ed['value']; │ │ │ │ │ + } else { │ │ │ │ │ + ed['displayName'] = key; │ │ │ │ │ + attributes[key] = ed; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return attributes; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseProperty │ │ │ │ │ + * Convenience method to find a node and return its value │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * xmlNode - {<DOMElement>} │ │ │ │ │ + * namespace - {String} namespace of the node to find │ │ │ │ │ + * tagName - {String} name of the property to parse │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The value for the requested property (defaults to null) │ │ │ │ │ + */ │ │ │ │ │ + parseProperty: function(xmlNode, namespace, tagName) { │ │ │ │ │ + var value; │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(xmlNode, namespace, tagName); │ │ │ │ │ + try { │ │ │ │ │ + value = OpenLayers.Util.getXmlNodeValue(nodeList[0]); │ │ │ │ │ + } catch (e) { │ │ │ │ │ + value = null; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return value; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Accept Feature Collection, and return a string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A KML string. │ │ │ │ │ + */ │ │ │ │ │ + write: function(features) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ + } │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "kml"); │ │ │ │ │ + var folder = this.createFolderXML(); │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + folder.appendChild(this.createPlacemarkXML(features[i])); │ │ │ │ │ + } │ │ │ │ │ + kml.appendChild(folder); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [kml]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createFolderXML │ │ │ │ │ + * Creates and returns a KML folder node │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + createFolderXML: function() { │ │ │ │ │ + // Folder │ │ │ │ │ + var folder = this.createElementNS(this.kmlns, "Folder"); │ │ │ │ │ + │ │ │ │ │ + // Folder name │ │ │ │ │ + if (this.foldersName) { │ │ │ │ │ + var folderName = this.createElementNS(this.kmlns, "name"); │ │ │ │ │ + var folderNameText = this.createTextNode(this.foldersName); │ │ │ │ │ + folderName.appendChild(folderNameText); │ │ │ │ │ + folder.appendChild(folderName); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Folder description │ │ │ │ │ + if (this.foldersDesc) { │ │ │ │ │ + var folderDesc = this.createElementNS(this.kmlns, "description"); │ │ │ │ │ + var folderDescText = this.createTextNode(this.foldersDesc); │ │ │ │ │ + folderDesc.appendChild(folderDescText); │ │ │ │ │ + folder.appendChild(folderDesc); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return folder; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createPlacemarkXML │ │ │ │ │ + * Creates and returns a KML placemark node representing the given feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + createPlacemarkXML: function(feature) { │ │ │ │ │ + // Placemark name │ │ │ │ │ + var placemarkName = this.createElementNS(this.kmlns, "name"); │ │ │ │ │ + var label = (feature.style && feature.style.label) ? feature.style.label : feature.id; │ │ │ │ │ + var name = feature.attributes.name || label; │ │ │ │ │ + placemarkName.appendChild(this.createTextNode(name)); │ │ │ │ │ + │ │ │ │ │ + // Placemark description │ │ │ │ │ + var placemarkDesc = this.createElementNS(this.kmlns, "description"); │ │ │ │ │ + var desc = feature.attributes.description || this.placemarksDesc; │ │ │ │ │ + placemarkDesc.appendChild(this.createTextNode(desc)); │ │ │ │ │ + │ │ │ │ │ + // Placemark │ │ │ │ │ + var placemarkNode = this.createElementNS(this.kmlns, "Placemark"); │ │ │ │ │ + if (feature.fid != null) { │ │ │ │ │ + placemarkNode.setAttribute("id", feature.fid); │ │ │ │ │ + } │ │ │ │ │ + placemarkNode.appendChild(placemarkName); │ │ │ │ │ + placemarkNode.appendChild(placemarkDesc); │ │ │ │ │ + │ │ │ │ │ + // Geometry node (Point, LineString, etc. nodes) │ │ │ │ │ + var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ + placemarkNode.appendChild(geometryNode); │ │ │ │ │ + │ │ │ │ │ + // output attributes as extendedData │ │ │ │ │ + if (feature.attributes) { │ │ │ │ │ + var edNode = this.buildExtendedData(feature.attributes); │ │ │ │ │ + if (edNode) { │ │ │ │ │ + placemarkNode.appendChild(edNode); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return placemarkNode; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometryNode │ │ │ │ │ + * Builds and returns a KML geometry node with the given geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + buildGeometryNode: function(geometry) { │ │ │ │ │ + var className = geometry.CLASS_NAME; │ │ │ │ │ + var type = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ + var builder = this.buildGeometry[type.toLowerCase()]; │ │ │ │ │ + var node = null; │ │ │ │ │ + if (builder) { │ │ │ │ │ + node = builder.apply(this, [geometry]); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: buildGeometry │ │ │ │ │ + * Object containing methods to do the actual geometry node building │ │ │ │ │ + * based on geometry type. │ │ │ │ │ + */ │ │ │ │ │ + buildGeometry: { │ │ │ │ │ + // TBD: Anybody care about namespace aliases here (these nodes have │ │ │ │ │ + // no prefixes)? │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.point │ │ │ │ │ + * Given an OpenLayers point geometry, create a KML point. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Point>} A point geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML point node. │ │ │ │ │ + */ │ │ │ │ │ + point: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "Point"); │ │ │ │ │ + kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + return kml; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.multipoint │ │ │ │ │ + * Given an OpenLayers multipoint geometry, create a KML │ │ │ │ │ + * GeometryCollection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Point>} A multipoint geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ + */ │ │ │ │ │ + multipoint: function(geometry) { │ │ │ │ │ + return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.linestring │ │ │ │ │ + * Given an OpenLayers linestring geometry, create a KML linestring. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.LineString>} A linestring geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML linestring node. │ │ │ │ │ + */ │ │ │ │ │ + linestring: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "LineString"); │ │ │ │ │ + kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + return kml; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.multilinestring │ │ │ │ │ + * Given an OpenLayers multilinestring geometry, create a KML │ │ │ │ │ + * GeometryCollection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Point>} A multilinestring geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ + */ │ │ │ │ │ + multilinestring: function(geometry) { │ │ │ │ │ + return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.linearring │ │ │ │ │ + * Given an OpenLayers linearring geometry, create a KML linearring. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.LinearRing>} A linearring geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML linearring node. │ │ │ │ │ + */ │ │ │ │ │ + linearring: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "LinearRing"); │ │ │ │ │ + kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + return kml; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.polygon │ │ │ │ │ + * Given an OpenLayers polygon geometry, create a KML polygon. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Polygon>} A polygon geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML polygon node. │ │ │ │ │ + */ │ │ │ │ │ + polygon: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "Polygon"); │ │ │ │ │ + var rings = geometry.components; │ │ │ │ │ + var ringMember, ringGeom, type; │ │ │ │ │ + for (var i = 0, len = rings.length; i < len; ++i) { │ │ │ │ │ + type = (i == 0) ? "outerBoundaryIs" : "innerBoundaryIs"; │ │ │ │ │ + ringMember = this.createElementNS(this.kmlns, type); │ │ │ │ │ + ringGeom = this.buildGeometry.linearring.apply(this, │ │ │ │ │ + [rings[i]]); │ │ │ │ │ + ringMember.appendChild(ringGeom); │ │ │ │ │ + kml.appendChild(ringMember); │ │ │ │ │ + } │ │ │ │ │ + return kml; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.multipolygon │ │ │ │ │ + * Given an OpenLayers multipolygon geometry, create a KML │ │ │ │ │ + * GeometryCollection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Point>} A multipolygon geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ + */ │ │ │ │ │ + multipolygon: function(geometry) { │ │ │ │ │ + return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.collection │ │ │ │ │ + * Given an OpenLayers geometry collection, create a KML MultiGeometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Collection>} A geometry collection. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML MultiGeometry node. │ │ │ │ │ + */ │ │ │ │ │ + collection: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "MultiGeometry"); │ │ │ │ │ + var child; │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; ++i) { │ │ │ │ │ + child = this.buildGeometryNode.apply(this, │ │ │ │ │ + [geometry.components[i]]); │ │ │ │ │ + if (child) { │ │ │ │ │ + kml.appendChild(child); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return kml; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildCoordinatesNode │ │ │ │ │ + * Builds and returns the KML coordinates node with the given geometry │ │ │ │ │ + * <coordinates>...</coordinates> │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + buildCoordinatesNode: function(geometry) { │ │ │ │ │ + var coordinatesNode = this.createElementNS(this.kmlns, "coordinates"); │ │ │ │ │ + │ │ │ │ │ + var path; │ │ │ │ │ + var points = geometry.components; │ │ │ │ │ + if (points) { │ │ │ │ │ + // LineString or LinearRing │ │ │ │ │ + var point; │ │ │ │ │ + var numPoints = points.length; │ │ │ │ │ + var parts = new Array(numPoints); │ │ │ │ │ + for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ + point = points[i]; │ │ │ │ │ + parts[i] = this.buildCoordinates(point); │ │ │ │ │ + } │ │ │ │ │ + path = parts.join(" "); │ │ │ │ │ + } else { │ │ │ │ │ + // Point │ │ │ │ │ + path = this.buildCoordinates(geometry); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var txtNode = this.createTextNode(path); │ │ │ │ │ + coordinatesNode.appendChild(txtNode); │ │ │ │ │ + │ │ │ │ │ + return coordinatesNode; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildCoordinates │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * │ │ │ │ │ + * Returns │ │ │ │ │ + * {String} a coordinate pair │ │ │ │ │ + */ │ │ │ │ │ + buildCoordinates: function(point) { │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + point = point.clone(); │ │ │ │ │ + point.transform(this.internalProjection, │ │ │ │ │ + this.externalProjection); │ │ │ │ │ + } │ │ │ │ │ + return point.x + "," + point.y; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildExtendedData │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * attributes - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns │ │ │ │ │ + * {DOMElement} A KML ExtendedData node or {null} if no attributes. │ │ │ │ │ + */ │ │ │ │ │ + buildExtendedData: function(attributes) { │ │ │ │ │ + var extendedData = this.createElementNS(this.kmlns, "ExtendedData"); │ │ │ │ │ + for (var attributeName in attributes) { │ │ │ │ │ + // empty, name, description, styleUrl attributes ignored │ │ │ │ │ + if (attributes[attributeName] && attributeName != "name" && attributeName != "description" && attributeName != "styleUrl") { │ │ │ │ │ + var data = this.createElementNS(this.kmlns, "Data"); │ │ │ │ │ + data.setAttribute("name", attributeName); │ │ │ │ │ + var value = this.createElementNS(this.kmlns, "value"); │ │ │ │ │ + if (typeof attributes[attributeName] == "object") { │ │ │ │ │ + // cater for object attributes with 'value' properties │ │ │ │ │ + // other object properties will output an empty node │ │ │ │ │ + if (attributes[attributeName].value) { │ │ │ │ │ + value.appendChild(this.createTextNode(attributes[attributeName].value)); │ │ │ │ │ + } │ │ │ │ │ + if (attributes[attributeName].displayName) { │ │ │ │ │ + var displayName = this.createElementNS(this.kmlns, "displayName"); │ │ │ │ │ + // displayName always written as CDATA │ │ │ │ │ + displayName.appendChild(this.getXMLDoc().createCDATASection(attributes[attributeName].displayName)); │ │ │ │ │ + data.appendChild(displayName); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + value.appendChild(this.createTextNode(attributes[attributeName])); │ │ │ │ │ + } │ │ │ │ │ + data.appendChild(value); │ │ │ │ │ + extendedData.appendChild(data); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.isSimpleContent(extendedData)) { │ │ │ │ │ + return null; │ │ │ │ │ + } else { │ │ │ │ │ + return extendedData; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.KML" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/SLD.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + * @requires OpenLayers/Style.js │ │ │ │ │ + * @requires OpenLayers/Rule.js │ │ │ │ │ + * @requires OpenLayers/Filter/FeatureId.js │ │ │ │ │ + * @requires OpenLayers/Filter/Logical.js │ │ │ │ │ + * @requires OpenLayers/Filter/Comparison.js │ │ │ │ │ + * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.SLD │ │ │ │ │ + * Read/Write SLD. Create a new instance with the <OpenLayers.Format.SLD> │ │ │ │ │ + * constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.SLD = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: profile │ │ │ │ │ + * {String} If provided, use a custom profile. │ │ │ │ │ * │ │ │ │ │ * Currently supported profiles: │ │ │ │ │ * - GeoServer - parses GeoServer vendor specific capabilities for SLD. │ │ │ │ │ */ │ │ │ │ │ profile: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ @@ -51569,748 +51296,1817 @@ │ │ │ │ │ * Returns: │ │ │ │ │ * {Object} An object representing the SLD. │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.SLD" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WFS.js │ │ │ │ │ + OpenLayers/Format/QueryStringFilter.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/GML.js │ │ │ │ │ * @requires OpenLayers/Console.js │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Format.js │ │ │ │ │ + * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ + * @requires OpenLayers/Filter/Comparison.js │ │ │ │ │ + * @requires OpenLayers/Filter/Logical.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WFS │ │ │ │ │ - * Read/Write WFS. │ │ │ │ │ + * Class: OpenLayers.Format.QueryStringFilter │ │ │ │ │ + * Parser for reading a query string and creating a simple filter. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.GML> │ │ │ │ │ + * - <OpenLayers.Format> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WFS = OpenLayers.Class(OpenLayers.Format.GML, { │ │ │ │ │ +OpenLayers.Format.QueryStringFilter = (function() { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer>} │ │ │ │ │ + * Map the OpenLayers.Filter.Comparison types to the operation strings of │ │ │ │ │ + * the protocol. │ │ │ │ │ */ │ │ │ │ │ - layer: null, │ │ │ │ │ + var cmpToStr = {}; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.EQUAL_TO] = "eq"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.NOT_EQUAL_TO] = "ne"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.LESS_THAN] = "lt"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO] = "lte"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.GREATER_THAN] = "gt"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO] = "gte"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.LIKE] = "ilike"; │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: wfsns │ │ │ │ │ - * {String} │ │ │ │ │ + * Function: regex2value │ │ │ │ │ + * Convert the value from a regular expression string to a LIKE/ILIKE │ │ │ │ │ + * string known to the web service. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * value - {String} The regex string. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The converted string. │ │ │ │ │ */ │ │ │ │ │ - wfsns: "http://www.opengis.net/wfs", │ │ │ │ │ + function regex2value(value) { │ │ │ │ │ + │ │ │ │ │ + // highly sensitive!! Do not change this without running the │ │ │ │ │ + // Protocol/HTTP.html unit tests │ │ │ │ │ + │ │ │ │ │ + // convert % to \% │ │ │ │ │ + value = value.replace(/%/g, "\\%"); │ │ │ │ │ + │ │ │ │ │ + // convert \\. to \\_ (\\.* occurences converted later) │ │ │ │ │ + value = value.replace(/\\\\\.(\*)?/g, function($0, $1) { │ │ │ │ │ + return $1 ? $0 : "\\\\_"; │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + // convert \\.* to \\% │ │ │ │ │ + value = value.replace(/\\\\\.\*/g, "\\\\%"); │ │ │ │ │ + │ │ │ │ │ + // convert . to _ (\. and .* occurences converted later) │ │ │ │ │ + value = value.replace(/(\\)?\.(\*)?/g, function($0, $1, $2) { │ │ │ │ │ + return $1 || $2 ? $0 : "_"; │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + // convert .* to % (\.* occurnces converted later) │ │ │ │ │ + value = value.replace(/(\\)?\.\*/g, function($0, $1) { │ │ │ │ │ + return $1 ? $0 : "%"; │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + // convert \. to . │ │ │ │ │ + value = value.replace(/\\\./g, "."); │ │ │ │ │ + │ │ │ │ │ + // replace \* with * (watching out for \\*) │ │ │ │ │ + value = value.replace(/(\\)?\\\*/g, function($0, $1) { │ │ │ │ │ + return $1 ? $0 : "*"; │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + return value; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: wildcarded. │ │ │ │ │ + * {Boolean} If true percent signs are added around values │ │ │ │ │ + * read from LIKE filters, for example if the protocol │ │ │ │ │ + * read method is passed a LIKE filter whose property │ │ │ │ │ + * is "foo" and whose value is "bar" the string │ │ │ │ │ + * "foo__ilike=%bar%" will be sent in the query string; │ │ │ │ │ + * defaults to false. │ │ │ │ │ + */ │ │ │ │ │ + wildcarded: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: srsInBBOX │ │ │ │ │ + * {Boolean} Include the SRS identifier in BBOX query string parameter. │ │ │ │ │ + * Default is false. If true and the layer has a projection object set, │ │ │ │ │ + * any BBOX filter will be serialized with a fifth item identifying the │ │ │ │ │ + * projection. E.g. bbox=-1000,-1000,1000,1000,EPSG:900913 │ │ │ │ │ + */ │ │ │ │ │ + srsInBBOX: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Serialize an <OpenLayers.Filter> objects using the "simple" filter syntax for │ │ │ │ │ + * query string parameters. This function must be called as a method of │ │ │ │ │ + * a protocol instance. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {<OpenLayers.Filter>} filter to convert. │ │ │ │ │ + * params - {Object} The parameters object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The resulting parameters object. │ │ │ │ │ + */ │ │ │ │ │ + write: function(filter, params) { │ │ │ │ │ + params = params || {}; │ │ │ │ │ + var className = filter.CLASS_NAME; │ │ │ │ │ + var filterType = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ + switch (filterType) { │ │ │ │ │ + case "Spatial": │ │ │ │ │ + switch (filter.type) { │ │ │ │ │ + case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ + params.bbox = filter.value.toArray(); │ │ │ │ │ + if (this.srsInBBOX && filter.projection) { │ │ │ │ │ + params.bbox.push(filter.projection.getCode()); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Spatial.DWITHIN: │ │ │ │ │ + params.tolerance = filter.distance; │ │ │ │ │ + // no break here │ │ │ │ │ + case OpenLayers.Filter.Spatial.WITHIN: │ │ │ │ │ + params.lon = filter.value.x; │ │ │ │ │ + params.lat = filter.value.y; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + OpenLayers.Console.warn( │ │ │ │ │ + "Unknown spatial filter type " + filter.type); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "Comparison": │ │ │ │ │ + var op = cmpToStr[filter.type]; │ │ │ │ │ + if (op !== undefined) { │ │ │ │ │ + var value = filter.value; │ │ │ │ │ + if (filter.type == OpenLayers.Filter.Comparison.LIKE) { │ │ │ │ │ + value = regex2value(value); │ │ │ │ │ + if (this.wildcarded) { │ │ │ │ │ + value = "%" + value + "%"; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + params[filter.property + "__" + op] = value; │ │ │ │ │ + params.queryable = params.queryable || []; │ │ │ │ │ + params.queryable.push(filter.property); │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Console.warn( │ │ │ │ │ + "Unknown comparison filter type " + filter.type); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "Logical": │ │ │ │ │ + if (filter.type === OpenLayers.Filter.Logical.AND) { │ │ │ │ │ + for (var i = 0, len = filter.filters.length; i < len; i++) { │ │ │ │ │ + params = this.write(filter.filters[i], params); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Console.warn( │ │ │ │ │ + "Unsupported logical filter type " + filter.type); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + OpenLayers.Console.warn("Unknown filter type " + filterType); │ │ │ │ │ + } │ │ │ │ │ + return params; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.QueryStringFilter" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +})(); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/GeoRSS.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.GeoRSS │ │ │ │ │ + * Read/write GeoRSS parser. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Format.GeoRSS> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.GeoRSS = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: ogcns │ │ │ │ │ - * {String} │ │ │ │ │ + * APIProperty: rssns │ │ │ │ │ + * {String} RSS namespace to use. Defaults to │ │ │ │ │ + * "http://backend.userland.com/rss2" │ │ │ │ │ */ │ │ │ │ │ - ogcns: "http://www.opengis.net/ogc", │ │ │ │ │ + rssns: "http://backend.userland.com/rss2", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WFS │ │ │ │ │ - * Create a WFS-T formatter. This requires a layer: that layer should │ │ │ │ │ - * have two properties: geometry_column and typename. The parser │ │ │ │ │ - * for this format is subclassed entirely from GML: There is a writer │ │ │ │ │ - * only, which uses most of the code from the GML layer, and wraps │ │ │ │ │ - * it in transactional elements. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ - * layer - {<OpenLayers.Layer>} │ │ │ │ │ + * APIProperty: featurens │ │ │ │ │ + * {String} Feature Attributes namespace. Defaults to │ │ │ │ │ + * "http://mapserver.gis.umn.edu/mapserver" │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options, layer) { │ │ │ │ │ - OpenLayers.Format.GML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - if (this.layer.featureNS) { │ │ │ │ │ - this.featureNS = this.layer.featureNS; │ │ │ │ │ - } │ │ │ │ │ - if (this.layer.options.geometry_column) { │ │ │ │ │ - this.geometryName = this.layer.options.geometry_column; │ │ │ │ │ - } │ │ │ │ │ - if (this.layer.options.typename) { │ │ │ │ │ - this.featureName = this.layer.options.typename; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + featureNS: "http://mapserver.gis.umn.edu/mapserver", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: write │ │ │ │ │ - * Takes a feature list, and generates a WFS-T Transaction │ │ │ │ │ + * APIProperty: georssns │ │ │ │ │ + * {String} GeoRSS namespace to use. Defaults to │ │ │ │ │ + * "http://www.georss.org/georss" │ │ │ │ │ + */ │ │ │ │ │ + georssns: "http://www.georss.org/georss", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: geons │ │ │ │ │ + * {String} W3C Geo namespace to use. Defaults to │ │ │ │ │ + * "http://www.w3.org/2003/01/geo/wgs84_pos#" │ │ │ │ │ + */ │ │ │ │ │ + geons: "http://www.w3.org/2003/01/geo/wgs84_pos#", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: featureTitle │ │ │ │ │ + * {String} Default title for features. Defaults to "Untitled" │ │ │ │ │ + */ │ │ │ │ │ + featureTitle: "Untitled", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: featureDescription │ │ │ │ │ + * {String} Default description for features. Defaults to "No Description" │ │ │ │ │ + */ │ │ │ │ │ + featureDescription: "No Description", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: gmlParse │ │ │ │ │ + * {Object} GML Format object for parsing features │ │ │ │ │ + * Non-API and only created if necessary │ │ │ │ │ + */ │ │ │ │ │ + gmlParser: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: xy │ │ │ │ │ + * {Boolean} Order of the GML coordinate: true:(x,y) or false:(y,x) │ │ │ │ │ + * For GeoRSS the default is (y,x), therefore: false │ │ │ │ │ + */ │ │ │ │ │ + xy: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.GeoRSS │ │ │ │ │ + * Create a new parser for GeoRSS. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - write: function(features) { │ │ │ │ │ │ │ │ │ │ - var transaction = this.createElementNS(this.wfsns, 'wfs:Transaction'); │ │ │ │ │ - transaction.setAttribute("version", "1.0.0"); │ │ │ │ │ - transaction.setAttribute("service", "WFS"); │ │ │ │ │ - for (var i = 0; i < features.length; i++) { │ │ │ │ │ - switch (features[i].state) { │ │ │ │ │ - case OpenLayers.State.INSERT: │ │ │ │ │ - transaction.appendChild(this.insert(features[i])); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.State.UPDATE: │ │ │ │ │ - transaction.appendChild(this.update(features[i])); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.State.DELETE: │ │ │ │ │ - transaction.appendChild(this.remove(features[i])); │ │ │ │ │ - break; │ │ │ │ │ + /** │ │ │ │ │ + * Method: createGeometryFromItem │ │ │ │ │ + * Return a geometry from a GeoRSS Item. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * item - {DOMElement} A GeoRSS item node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry>} A geometry representing the node. │ │ │ │ │ + */ │ │ │ │ │ + createGeometryFromItem: function(item) { │ │ │ │ │ + var point = this.getElementsByTagNameNS(item, this.georssns, "point"); │ │ │ │ │ + var lat = this.getElementsByTagNameNS(item, this.geons, 'lat'); │ │ │ │ │ + var lon = this.getElementsByTagNameNS(item, this.geons, 'long'); │ │ │ │ │ + │ │ │ │ │ + var line = this.getElementsByTagNameNS(item, │ │ │ │ │ + this.georssns, │ │ │ │ │ + "line"); │ │ │ │ │ + var polygon = this.getElementsByTagNameNS(item, │ │ │ │ │ + this.georssns, │ │ │ │ │ + "polygon"); │ │ │ │ │ + var where = this.getElementsByTagNameNS(item, │ │ │ │ │ + this.georssns, │ │ │ │ │ + "where"); │ │ │ │ │ + var box = this.getElementsByTagNameNS(item, │ │ │ │ │ + this.georssns, │ │ │ │ │ + "box"); │ │ │ │ │ + │ │ │ │ │ + if (point.length > 0 || (lat.length > 0 && lon.length > 0)) { │ │ │ │ │ + var location; │ │ │ │ │ + if (point.length > 0) { │ │ │ │ │ + location = OpenLayers.String.trim( │ │ │ │ │ + point[0].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ + if (location.length != 2) { │ │ │ │ │ + location = OpenLayers.String.trim( │ │ │ │ │ + point[0].firstChild.nodeValue).split(/\s*,\s*/); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + location = [parseFloat(lat[0].firstChild.nodeValue), │ │ │ │ │ + parseFloat(lon[0].firstChild.nodeValue) │ │ │ │ │ + ]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point(location[1], location[0]); │ │ │ │ │ + │ │ │ │ │ + } else if (line.length > 0) { │ │ │ │ │ + var coords = OpenLayers.String.trim(this.getChildValue(line[0])).split(/\s+/); │ │ │ │ │ + var components = []; │ │ │ │ │ + var point; │ │ │ │ │ + for (var i = 0, len = coords.length; i < len; i += 2) { │ │ │ │ │ + point = new OpenLayers.Geometry.Point(coords[i + 1], coords[i]); │ │ │ │ │ + components.push(point); │ │ │ │ │ + } │ │ │ │ │ + geometry = new OpenLayers.Geometry.LineString(components); │ │ │ │ │ + } else if (polygon.length > 0) { │ │ │ │ │ + var coords = OpenLayers.String.trim(this.getChildValue(polygon[0])).split(/\s+/); │ │ │ │ │ + var components = []; │ │ │ │ │ + var point; │ │ │ │ │ + for (var i = 0, len = coords.length; i < len; i += 2) { │ │ │ │ │ + point = new OpenLayers.Geometry.Point(coords[i + 1], coords[i]); │ │ │ │ │ + components.push(point); │ │ │ │ │ + } │ │ │ │ │ + geometry = new OpenLayers.Geometry.Polygon([new OpenLayers.Geometry.LinearRing(components)]); │ │ │ │ │ + } else if (where.length > 0) { │ │ │ │ │ + if (!this.gmlParser) { │ │ │ │ │ + this.gmlParser = new OpenLayers.Format.GML({ │ │ │ │ │ + 'xy': this.xy │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + var feature = this.gmlParser.parseFeature(where[0]); │ │ │ │ │ + geometry = feature.geometry; │ │ │ │ │ + } else if (box.length > 0) { │ │ │ │ │ + var coords = OpenLayers.String.trim(box[0].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ + var components = []; │ │ │ │ │ + var point; │ │ │ │ │ + if (coords.length > 3) { │ │ │ │ │ + point = new OpenLayers.Geometry.Point(coords[1], coords[0]); │ │ │ │ │ + components.push(point); │ │ │ │ │ + point = new OpenLayers.Geometry.Point(coords[1], coords[2]); │ │ │ │ │ + components.push(point); │ │ │ │ │ + point = new OpenLayers.Geometry.Point(coords[3], coords[2]); │ │ │ │ │ + components.push(point); │ │ │ │ │ + point = new OpenLayers.Geometry.Point(coords[3], coords[0]); │ │ │ │ │ + components.push(point); │ │ │ │ │ + point = new OpenLayers.Geometry.Point(coords[1], coords[0]); │ │ │ │ │ + components.push(point); │ │ │ │ │ } │ │ │ │ │ + geometry = new OpenLayers.Geometry.Polygon([new OpenLayers.Geometry.LinearRing(components)]); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [transaction]); │ │ │ │ │ + if (geometry && this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry.transform(this.externalProjection, │ │ │ │ │ + this.internalProjection); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return geometry; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createFeatureXML │ │ │ │ │ + * Method: createFeatureFromItem │ │ │ │ │ + * Return a feature from a GeoRSS Item. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * item - {DOMElement} A GeoRSS item node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A feature representing the item. │ │ │ │ │ */ │ │ │ │ │ - createFeatureXML: function(feature) { │ │ │ │ │ - var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ - var geomContainer = this.createElementNS(this.featureNS, "feature:" + this.geometryName); │ │ │ │ │ - geomContainer.appendChild(geometryNode); │ │ │ │ │ - var featureContainer = this.createElementNS(this.featureNS, "feature:" + this.featureName); │ │ │ │ │ - featureContainer.appendChild(geomContainer); │ │ │ │ │ - for (var attr in feature.attributes) { │ │ │ │ │ - var attrText = this.createTextNode(feature.attributes[attr]); │ │ │ │ │ - var nodename = attr; │ │ │ │ │ - if (attr.search(":") != -1) { │ │ │ │ │ - nodename = attr.split(":")[1]; │ │ │ │ │ + createFeatureFromItem: function(item) { │ │ │ │ │ + var geometry = this.createGeometryFromItem(item); │ │ │ │ │ + │ │ │ │ │ + /* Provide defaults for title and description */ │ │ │ │ │ + var title = this._getChildValue(item, "*", "title", this.featureTitle); │ │ │ │ │ + │ │ │ │ │ + /* First try RSS descriptions, then Atom summaries */ │ │ │ │ │ + var description = this._getChildValue( │ │ │ │ │ + item, "*", "description", │ │ │ │ │ + this._getChildValue(item, "*", "content", │ │ │ │ │ + this._getChildValue(item, "*", "summary", this.featureDescription))); │ │ │ │ │ + │ │ │ │ │ + /* If no link URL is found in the first child node, try the │ │ │ │ │ + href attribute */ │ │ │ │ │ + var link = this._getChildValue(item, "*", "link"); │ │ │ │ │ + if (!link) { │ │ │ │ │ + try { │ │ │ │ │ + link = this.getElementsByTagNameNS(item, "*", "link")[0].getAttribute("href"); │ │ │ │ │ + } catch (e) { │ │ │ │ │ + link = null; │ │ │ │ │ } │ │ │ │ │ - var attrContainer = this.createElementNS(this.featureNS, "feature:" + nodename); │ │ │ │ │ - attrContainer.appendChild(attrText); │ │ │ │ │ - featureContainer.appendChild(attrContainer); │ │ │ │ │ } │ │ │ │ │ - return featureContainer; │ │ │ │ │ + │ │ │ │ │ + var id = this._getChildValue(item, "*", "id", null); │ │ │ │ │ + │ │ │ │ │ + var data = { │ │ │ │ │ + "title": title, │ │ │ │ │ + "description": description, │ │ │ │ │ + "link": link │ │ │ │ │ + }; │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry, data); │ │ │ │ │ + feature.fid = id; │ │ │ │ │ + return feature; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: insert │ │ │ │ │ - * Takes a feature, and generates a WFS-T Transaction "Insert" │ │ │ │ │ + * Method: _getChildValue │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * nsuri - {String} Child node namespace uri ("*" for any). │ │ │ │ │ + * name - {String} Child node name. │ │ │ │ │ + * def - {String} Optional string default to return if no child found. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The value of the first child with the given tag name. Returns │ │ │ │ │ + * default value or empty string if none found. │ │ │ │ │ */ │ │ │ │ │ - insert: function(feature) { │ │ │ │ │ - var insertNode = this.createElementNS(this.wfsns, 'wfs:Insert'); │ │ │ │ │ - insertNode.appendChild(this.createFeatureXML(feature)); │ │ │ │ │ - return insertNode; │ │ │ │ │ + _getChildValue: function(node, nsuri, name, def) { │ │ │ │ │ + var value; │ │ │ │ │ + var eles = this.getElementsByTagNameNS(node, nsuri, name); │ │ │ │ │ + if (eles && eles[0] && eles[0].firstChild && │ │ │ │ │ + eles[0].firstChild.nodeValue) { │ │ │ │ │ + value = this.getChildValue(eles[0]); │ │ │ │ │ + } else { │ │ │ │ │ + value = (def == undefined) ? "" : def; │ │ │ │ │ + } │ │ │ │ │ + return value; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: update │ │ │ │ │ - * Takes a feature, and generates a WFS-T Transaction "Update" │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Return a list of features from a GeoRSS doc │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * doc - {Element} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ */ │ │ │ │ │ - update: function(feature) { │ │ │ │ │ - if (!feature.fid) { │ │ │ │ │ - OpenLayers.Console.userError(OpenLayers.i18n("noFID")); │ │ │ │ │ + read: function(doc) { │ │ │ │ │ + if (typeof doc == "string") { │ │ │ │ │ + doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]); │ │ │ │ │ } │ │ │ │ │ - var updateNode = this.createElementNS(this.wfsns, 'wfs:Update'); │ │ │ │ │ - updateNode.setAttribute("typeName", this.featurePrefix + ':' + this.featureName); │ │ │ │ │ - updateNode.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ - │ │ │ │ │ - var propertyNode = this.createElementNS(this.wfsns, 'wfs:Property'); │ │ │ │ │ - var nameNode = this.createElementNS(this.wfsns, 'wfs:Name'); │ │ │ │ │ - │ │ │ │ │ - var txtNode = this.createTextNode(this.geometryName); │ │ │ │ │ - nameNode.appendChild(txtNode); │ │ │ │ │ - propertyNode.appendChild(nameNode); │ │ │ │ │ │ │ │ │ │ - var valueNode = this.createElementNS(this.wfsns, 'wfs:Value'); │ │ │ │ │ - │ │ │ │ │ - var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ - │ │ │ │ │ - if (feature.layer) { │ │ │ │ │ - geometryNode.setAttribute( │ │ │ │ │ - "srsName", feature.layer.projection.getCode() │ │ │ │ │ - ); │ │ │ │ │ + /* Try RSS items first, then Atom entries */ │ │ │ │ │ + var itemlist = null; │ │ │ │ │ + itemlist = this.getElementsByTagNameNS(doc, '*', 'item'); │ │ │ │ │ + if (itemlist.length == 0) { │ │ │ │ │ + itemlist = this.getElementsByTagNameNS(doc, '*', 'entry'); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - valueNode.appendChild(geometryNode); │ │ │ │ │ - │ │ │ │ │ - propertyNode.appendChild(valueNode); │ │ │ │ │ - updateNode.appendChild(propertyNode); │ │ │ │ │ - │ │ │ │ │ - // add in attributes │ │ │ │ │ - for (var propName in feature.attributes) { │ │ │ │ │ - propertyNode = this.createElementNS(this.wfsns, 'wfs:Property'); │ │ │ │ │ - nameNode = this.createElementNS(this.wfsns, 'wfs:Name'); │ │ │ │ │ - nameNode.appendChild(this.createTextNode(propName)); │ │ │ │ │ - propertyNode.appendChild(nameNode); │ │ │ │ │ - valueNode = this.createElementNS(this.wfsns, 'wfs:Value'); │ │ │ │ │ - valueNode.appendChild(this.createTextNode(feature.attributes[propName])); │ │ │ │ │ - propertyNode.appendChild(valueNode); │ │ │ │ │ - updateNode.appendChild(propertyNode); │ │ │ │ │ + var numItems = itemlist.length; │ │ │ │ │ + var features = new Array(numItems); │ │ │ │ │ + for (var i = 0; i < numItems; i++) { │ │ │ │ │ + features[i] = this.createFeatureFromItem(itemlist[i]); │ │ │ │ │ } │ │ │ │ │ + return features; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ - var filterNode = this.createElementNS(this.ogcns, 'ogc:Filter'); │ │ │ │ │ - var filterIdNode = this.createElementNS(this.ogcns, 'ogc:FeatureId'); │ │ │ │ │ - filterIdNode.setAttribute("fid", feature.fid); │ │ │ │ │ - filterNode.appendChild(filterIdNode); │ │ │ │ │ - updateNode.appendChild(filterNode); │ │ │ │ │ - │ │ │ │ │ - return updateNode; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Accept Feature Collection, and return a string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} List of features to serialize into a string. │ │ │ │ │ + */ │ │ │ │ │ + write: function(features) { │ │ │ │ │ + var georss; │ │ │ │ │ + if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ + georss = this.createElementNS(this.rssns, "rss"); │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + georss.appendChild(this.createFeatureXML(features[i])); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + georss = this.createFeatureXML(features); │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [georss]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: remove │ │ │ │ │ - * Takes a feature, and generates a WFS-T Transaction "Delete" │ │ │ │ │ - * │ │ │ │ │ + * Method: createFeatureXML │ │ │ │ │ + * Accept an <OpenLayers.Feature.Vector>, and build a geometry for it. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - remove: function(feature) { │ │ │ │ │ - if (!feature.fid) { │ │ │ │ │ - OpenLayers.Console.userError(OpenLayers.i18n("noFID")); │ │ │ │ │ - return false; │ │ │ │ │ + createFeatureXML: function(feature) { │ │ │ │ │ + var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ + var featureNode = this.createElementNS(this.rssns, "item"); │ │ │ │ │ + var titleNode = this.createElementNS(this.rssns, "title"); │ │ │ │ │ + titleNode.appendChild(this.createTextNode(feature.attributes.title ? feature.attributes.title : "")); │ │ │ │ │ + var descNode = this.createElementNS(this.rssns, "description"); │ │ │ │ │ + descNode.appendChild(this.createTextNode(feature.attributes.description ? feature.attributes.description : "")); │ │ │ │ │ + featureNode.appendChild(titleNode); │ │ │ │ │ + featureNode.appendChild(descNode); │ │ │ │ │ + if (feature.attributes.link) { │ │ │ │ │ + var linkNode = this.createElementNS(this.rssns, "link"); │ │ │ │ │ + linkNode.appendChild(this.createTextNode(feature.attributes.link)); │ │ │ │ │ + featureNode.appendChild(linkNode); │ │ │ │ │ } │ │ │ │ │ - var deleteNode = this.createElementNS(this.wfsns, 'wfs:Delete'); │ │ │ │ │ - deleteNode.setAttribute("typeName", this.featurePrefix + ':' + this.featureName); │ │ │ │ │ - deleteNode.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ + for (var attr in feature.attributes) { │ │ │ │ │ + if (attr == "link" || attr == "title" || attr == "description") { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + var attrText = this.createTextNode(feature.attributes[attr]); │ │ │ │ │ + var nodename = attr; │ │ │ │ │ + if (attr.search(":") != -1) { │ │ │ │ │ + nodename = attr.split(":")[1]; │ │ │ │ │ + } │ │ │ │ │ + var attrContainer = this.createElementNS(this.featureNS, "feature:" + nodename); │ │ │ │ │ + attrContainer.appendChild(attrText); │ │ │ │ │ + featureNode.appendChild(attrContainer); │ │ │ │ │ + } │ │ │ │ │ + featureNode.appendChild(geometryNode); │ │ │ │ │ + return featureNode; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var filterNode = this.createElementNS(this.ogcns, 'ogc:Filter'); │ │ │ │ │ - var filterIdNode = this.createElementNS(this.ogcns, 'ogc:FeatureId'); │ │ │ │ │ - filterIdNode.setAttribute("fid", feature.fid); │ │ │ │ │ - filterNode.appendChild(filterIdNode); │ │ │ │ │ - deleteNode.appendChild(filterNode); │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometryNode │ │ │ │ │ + * builds a GeoRSS node with a given geometry │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A gml node. │ │ │ │ │ + */ │ │ │ │ │ + buildGeometryNode: function(geometry) { │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry = geometry.clone(); │ │ │ │ │ + geometry.transform(this.internalProjection, │ │ │ │ │ + this.externalProjection); │ │ │ │ │ + } │ │ │ │ │ + var node; │ │ │ │ │ + // match Polygon │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.Polygon") { │ │ │ │ │ + node = this.createElementNS(this.georssns, 'georss:polygon'); │ │ │ │ │ │ │ │ │ │ - return deleteNode; │ │ │ │ │ + node.appendChild(this.buildCoordinatesNode(geometry.components[0])); │ │ │ │ │ + } │ │ │ │ │ + // match LineString │ │ │ │ │ + else if (geometry.CLASS_NAME == "OpenLayers.Geometry.LineString") { │ │ │ │ │ + node = this.createElementNS(this.georssns, 'georss:line'); │ │ │ │ │ + │ │ │ │ │ + node.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + } │ │ │ │ │ + // match Point │ │ │ │ │ + else if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + node = this.createElementNS(this.georssns, 'georss:point'); │ │ │ │ │ + node.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + } else { │ │ │ │ │ + throw "Couldn't parse " + geometry.CLASS_NAME; │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Remove ciruclar ref to layer │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildCoordinatesNode │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.layer = null; │ │ │ │ │ + buildCoordinatesNode: function(geometry) { │ │ │ │ │ + var points = null; │ │ │ │ │ + │ │ │ │ │ + if (geometry.components) { │ │ │ │ │ + points = geometry.components; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var path; │ │ │ │ │ + if (points) { │ │ │ │ │ + var numPoints = points.length; │ │ │ │ │ + var parts = new Array(numPoints); │ │ │ │ │ + for (var i = 0; i < numPoints; i++) { │ │ │ │ │ + parts[i] = points[i].y + " " + points[i].x; │ │ │ │ │ + } │ │ │ │ │ + path = parts.join(" "); │ │ │ │ │ + } else { │ │ │ │ │ + path = geometry.y + " " + geometry.x; │ │ │ │ │ + } │ │ │ │ │ + return this.createTextNode(path); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFS" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.GeoRSS" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/XLS.js │ │ │ │ │ + OpenLayers/Format/WFSCapabilities.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.XLS │ │ │ │ │ - * Read/Write XLS (OpenLS). Create a new instance with the <OpenLayers.Format.XLS> │ │ │ │ │ - * constructor. Currently only implemented for Location Utility Services, more │ │ │ │ │ - * specifically only for Geocoding. No support for Reverse Geocoding as yet. │ │ │ │ │ + * Class: OpenLayers.Format.WFSCapabilities │ │ │ │ │ + * Read WFS Capabilities. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.XLS = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ +OpenLayers.Format.WFSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIProperty: defaultVersion │ │ │ │ │ * {String} Version number to assume if none found. Default is "1.1.0". │ │ │ │ │ */ │ │ │ │ │ defaultVersion: "1.1.0", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: stringifyOutput │ │ │ │ │ - * {Boolean} If true, write will return a string otherwise a DOMElement. │ │ │ │ │ - * Default is true. │ │ │ │ │ + * Constructor: OpenLayers.Format.WFSCapabilities │ │ │ │ │ + * Create a new parser for WFS capabilities. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - stringifyOutput: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.XLS │ │ │ │ │ - * Create a new parser for XLS. │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} List of named layers. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSCapabilities" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/Atom.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Format/GML/v3.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.Atom │ │ │ │ │ + * Read/write Atom feeds. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Format.AtomFeed> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.Atom = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. Properties │ │ │ │ │ + * of this object should not be set individually. Read-only. All │ │ │ │ │ + * XML subclasses should have their own namespaces object. Use │ │ │ │ │ + * <setNamespace> to add or set a namespace alias after construction. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + atom: "http://www.w3.org/2005/Atom", │ │ │ │ │ + georss: "http://www.georss.org/georss" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: feedTitle │ │ │ │ │ + * {String} Atom feed elements require a title. Default is "untitled". │ │ │ │ │ + */ │ │ │ │ │ + feedTitle: "untitled", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultEntryTitle │ │ │ │ │ + * {String} Atom entry elements require a title. In cases where one is │ │ │ │ │ + * not provided in the feature attributes, this will be used. Default │ │ │ │ │ + * is "untitled". │ │ │ │ │ + */ │ │ │ │ │ + defaultEntryTitle: "untitled", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: gmlParse │ │ │ │ │ + * {Object} GML Format object for parsing features │ │ │ │ │ + * Non-API and only created if necessary │ │ │ │ │ + */ │ │ │ │ │ + gmlParser: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: xy │ │ │ │ │ + * {Boolean} Order of the GML coordinate: true:(x,y) or false:(y,x) │ │ │ │ │ + * For GeoRSS the default is (y,x), therefore: false │ │ │ │ │ + */ │ │ │ │ │ + xy: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.AtomEntry │ │ │ │ │ + * Create a new parser for Atom. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * options - {Object} An optional object whose properties will be set on │ │ │ │ │ * this instance. │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Return a list of features from an Atom feed or entry document. │ │ │ │ │ + │ │ │ │ │ + * Parameters: │ │ │ │ │ + * doc - {Element} or {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ + */ │ │ │ │ │ + read: function(doc) { │ │ │ │ │ + if (typeof doc == "string") { │ │ │ │ │ + doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]); │ │ │ │ │ + } │ │ │ │ │ + return this.parseFeatures(doc); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ * APIMethod: write │ │ │ │ │ - * Write out an XLS request. │ │ │ │ │ + * Serialize or more feature nodes to Atom documents. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * request - {Object} An object representing the LUS request. │ │ │ │ │ - * options - {Object} Optional configuration object. │ │ │ │ │ + * features - {<OpenLayers.Feature.Vector>} or Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} An XLS document string. │ │ │ │ │ + * {String} an Atom entry document if passed one feature node, or a feed │ │ │ │ │ + * document if passed an array of feature nodes. │ │ │ │ │ */ │ │ │ │ │ + write: function(features) { │ │ │ │ │ + var doc; │ │ │ │ │ + if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ + doc = this.createElementNSPlus("atom:feed"); │ │ │ │ │ + doc.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:title", { │ │ │ │ │ + value: this.feedTitle │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + for (var i = 0, ii = features.length; i < ii; i++) { │ │ │ │ │ + doc.appendChild(this.buildEntryNode(features[i])); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + doc = this.buildEntryNode(features); │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [doc]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read an XLS doc and return an object representing the result. │ │ │ │ │ + * Method: buildContentNode │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {String | DOMElement} Data to read. │ │ │ │ │ - * options - {Object} Options for the reader. │ │ │ │ │ + * content - {Object} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} An object representing the GeocodeResponse. │ │ │ │ │ + * {DOMElement} an Atom content node. │ │ │ │ │ + * │ │ │ │ │ + * TODO: types other than text. │ │ │ │ │ + */ │ │ │ │ │ + buildContentNode: function(content) { │ │ │ │ │ + var node = this.createElementNSPlus("atom:content", { │ │ │ │ │ + attributes: { │ │ │ │ │ + type: content.type || null │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (content.src) { │ │ │ │ │ + node.setAttribute("src", content.src); │ │ │ │ │ + } else { │ │ │ │ │ + if (content.type == "text" || content.type == null) { │ │ │ │ │ + node.appendChild( │ │ │ │ │ + this.createTextNode(content.value) │ │ │ │ │ + ); │ │ │ │ │ + } else if (content.type == "html") { │ │ │ │ │ + if (typeof content.value != "string") { │ │ │ │ │ + throw "HTML content must be in form of an escaped string"; │ │ │ │ │ + } │ │ │ │ │ + node.appendChild( │ │ │ │ │ + this.createTextNode(content.value) │ │ │ │ │ + ); │ │ │ │ │ + } else if (content.type == "xhtml") { │ │ │ │ │ + node.appendChild(content.value); │ │ │ │ │ + } else if (content.type == "xhtml" || │ │ │ │ │ + content.type.match(/(\+|\/)xml$/)) { │ │ │ │ │ + node.appendChild(content.value); │ │ │ │ │ + } else { // MUST be a valid Base64 encoding │ │ │ │ │ + node.appendChild( │ │ │ │ │ + this.createTextNode(content.value) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildEntryNode │ │ │ │ │ + * Build an Atom entry node from a feature object. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} an Atom entry node. │ │ │ │ │ + * │ │ │ │ │ + * These entries are geared for publication using AtomPub. │ │ │ │ │ + * │ │ │ │ │ + * TODO: support extension elements │ │ │ │ │ */ │ │ │ │ │ + buildEntryNode: function(feature) { │ │ │ │ │ + var attrib = feature.attributes; │ │ │ │ │ + var atomAttrib = attrib.atom || {}; │ │ │ │ │ + var entryNode = this.createElementNSPlus("atom:entry"); │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.XLS" │ │ │ │ │ + // atom:author │ │ │ │ │ + if (atomAttrib.authors) { │ │ │ │ │ + var authors = OpenLayers.Util.isArray(atomAttrib.authors) ? │ │ │ │ │ + atomAttrib.authors : [atomAttrib.authors]; │ │ │ │ │ + for (var i = 0, ii = authors.length; i < ii; i++) { │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.buildPersonConstructNode( │ │ │ │ │ + "author", authors[i] │ │ │ │ │ + ) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atom:category │ │ │ │ │ + if (atomAttrib.categories) { │ │ │ │ │ + var categories = OpenLayers.Util.isArray(atomAttrib.categories) ? │ │ │ │ │ + atomAttrib.categories : [atomAttrib.categories]; │ │ │ │ │ + var category; │ │ │ │ │ + for (var i = 0, ii = categories.length; i < ii; i++) { │ │ │ │ │ + category = categories[i]; │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:category", { │ │ │ │ │ + attributes: { │ │ │ │ │ + term: category.term, │ │ │ │ │ + scheme: category.scheme || null, │ │ │ │ │ + label: category.label || null │ │ │ │ │ + } │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atom:content │ │ │ │ │ + if (atomAttrib.content) { │ │ │ │ │ + entryNode.appendChild(this.buildContentNode(atomAttrib.content)); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atom:contributor │ │ │ │ │ + if (atomAttrib.contributors) { │ │ │ │ │ + var contributors = OpenLayers.Util.isArray(atomAttrib.contributors) ? │ │ │ │ │ + atomAttrib.contributors : [atomAttrib.contributors]; │ │ │ │ │ + for (var i = 0, ii = contributors.length; i < ii; i++) { │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.buildPersonConstructNode( │ │ │ │ │ + "contributor", │ │ │ │ │ + contributors[i] │ │ │ │ │ + ) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atom:id │ │ │ │ │ + if (feature.fid) { │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:id", { │ │ │ │ │ + value: feature.fid │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atom:link │ │ │ │ │ + if (atomAttrib.links) { │ │ │ │ │ + var links = OpenLayers.Util.isArray(atomAttrib.links) ? │ │ │ │ │ + atomAttrib.links : [atomAttrib.links]; │ │ │ │ │ + var link; │ │ │ │ │ + for (var i = 0, ii = links.length; i < ii; i++) { │ │ │ │ │ + link = links[i]; │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:link", { │ │ │ │ │ + attributes: { │ │ │ │ │ + href: link.href, │ │ │ │ │ + rel: link.rel || null, │ │ │ │ │ + type: link.type || null, │ │ │ │ │ + hreflang: link.hreflang || null, │ │ │ │ │ + title: link.title || null, │ │ │ │ │ + length: link.length || null │ │ │ │ │ + } │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atom:published │ │ │ │ │ + if (atomAttrib.published) { │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:published", { │ │ │ │ │ + value: atomAttrib.published │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atom:rights │ │ │ │ │ + if (atomAttrib.rights) { │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:rights", { │ │ │ │ │ + value: atomAttrib.rights │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atom:source not implemented │ │ │ │ │ + │ │ │ │ │ + // atom:summary │ │ │ │ │ + if (atomAttrib.summary || attrib.description) { │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:summary", { │ │ │ │ │ + value: atomAttrib.summary || attrib.description │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atom:title │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:title", { │ │ │ │ │ + value: atomAttrib.title || attrib.title || this.defaultEntryTitle │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + // atom:updated │ │ │ │ │ + if (atomAttrib.updated) { │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:updated", { │ │ │ │ │ + value: atomAttrib.updated │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // georss:where │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + var whereNode = this.createElementNSPlus("georss:where"); │ │ │ │ │ + whereNode.appendChild( │ │ │ │ │ + this.buildGeometryNode(feature.geometry) │ │ │ │ │ + ); │ │ │ │ │ + entryNode.appendChild(whereNode); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return entryNode; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: initGmlParser │ │ │ │ │ + * Creates a GML parser. │ │ │ │ │ + */ │ │ │ │ │ + initGmlParser: function() { │ │ │ │ │ + this.gmlParser = new OpenLayers.Format.GML.v3({ │ │ │ │ │ + xy: this.xy, │ │ │ │ │ + featureNS: "http://example.com#feature", │ │ │ │ │ + internalProjection: this.internalProjection, │ │ │ │ │ + externalProjection: this.externalProjection │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometryNode │ │ │ │ │ + * builds a GeoRSS node with a given geometry │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A gml node. │ │ │ │ │ + */ │ │ │ │ │ + buildGeometryNode: function(geometry) { │ │ │ │ │ + if (!this.gmlParser) { │ │ │ │ │ + this.initGmlParser(); │ │ │ │ │ + } │ │ │ │ │ + var node = this.gmlParser.writeNode("feature:_geometry", geometry); │ │ │ │ │ + return node.firstChild; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildPersonConstructNode │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} │ │ │ │ │ + * value - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} an Atom person construct node. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * >>> buildPersonConstructNode("author", {name: "John Smith"}) │ │ │ │ │ + * {<author><name>John Smith</name></author>} │ │ │ │ │ + * │ │ │ │ │ + * TODO: how to specify extension elements? Add to the oNames array? │ │ │ │ │ + */ │ │ │ │ │ + buildPersonConstructNode: function(name, value) { │ │ │ │ │ + var oNames = ["uri", "email"]; │ │ │ │ │ + var personNode = this.createElementNSPlus("atom:" + name); │ │ │ │ │ + personNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:name", { │ │ │ │ │ + value: value.name │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + for (var i = 0, ii = oNames.length; i < ii; i++) { │ │ │ │ │ + if (value[oNames[i]]) { │ │ │ │ │ + personNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:" + oNames[i], { │ │ │ │ │ + value: value[oNames[i]] │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return personNode; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getFirstChildValue │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * nsuri - {String} Child node namespace uri ("*" for any). │ │ │ │ │ + * name - {String} Child node name. │ │ │ │ │ + * def - {String} Optional string default to return if no child found. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The value of the first child with the given tag name. Returns │ │ │ │ │ + * default value or empty string if none found. │ │ │ │ │ + */ │ │ │ │ │ + getFirstChildValue: function(node, nsuri, name, def) { │ │ │ │ │ + var value; │ │ │ │ │ + var nodes = this.getElementsByTagNameNS(node, nsuri, name); │ │ │ │ │ + if (nodes && nodes.length > 0) { │ │ │ │ │ + value = this.getChildValue(nodes[0], def); │ │ │ │ │ + } else { │ │ │ │ │ + value = def; │ │ │ │ │ + } │ │ │ │ │ + return value; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseFeature │ │ │ │ │ + * Parse feature from an Atom entry node.. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} An Atom entry or feed node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + parseFeature: function(node) { │ │ │ │ │ + var atomAttrib = {}; │ │ │ │ │ + var value = null; │ │ │ │ │ + var nodes = null; │ │ │ │ │ + var attval = null; │ │ │ │ │ + var atomns = this.namespaces.atom; │ │ │ │ │ + │ │ │ │ │ + // atomAuthor* │ │ │ │ │ + this.parsePersonConstructs(node, "author", atomAttrib); │ │ │ │ │ + │ │ │ │ │ + // atomCategory* │ │ │ │ │ + nodes = this.getElementsByTagNameNS(node, atomns, "category"); │ │ │ │ │ + if (nodes.length > 0) { │ │ │ │ │ + atomAttrib.categories = []; │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ + value = {}; │ │ │ │ │ + value.term = nodes[i].getAttribute("term"); │ │ │ │ │ + attval = nodes[i].getAttribute("scheme"); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value.scheme = attval; │ │ │ │ │ + } │ │ │ │ │ + attval = nodes[i].getAttribute("label"); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value.label = attval; │ │ │ │ │ + } │ │ │ │ │ + atomAttrib.categories.push(value); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atomContent? │ │ │ │ │ + nodes = this.getElementsByTagNameNS(node, atomns, "content"); │ │ │ │ │ + if (nodes.length > 0) { │ │ │ │ │ + value = {}; │ │ │ │ │ + attval = nodes[0].getAttribute("type"); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value.type = attval; │ │ │ │ │ + } │ │ │ │ │ + attval = nodes[0].getAttribute("src"); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value.src = attval; │ │ │ │ │ + } else { │ │ │ │ │ + if (value.type == "text" || │ │ │ │ │ + value.type == "html" || │ │ │ │ │ + value.type == null) { │ │ │ │ │ + value.value = this.getFirstChildValue( │ │ │ │ │ + node, │ │ │ │ │ + atomns, │ │ │ │ │ + "content", │ │ │ │ │ + null │ │ │ │ │ + ); │ │ │ │ │ + } else if (value.type == "xhtml" || │ │ │ │ │ + value.type.match(/(\+|\/)xml$/)) { │ │ │ │ │ + value.value = this.getChildEl(nodes[0]); │ │ │ │ │ + } else { // MUST be base64 encoded │ │ │ │ │ + value.value = this.getFirstChildValue( │ │ │ │ │ + node, │ │ │ │ │ + atomns, │ │ │ │ │ + "content", │ │ │ │ │ + null │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + atomAttrib.content = value; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atomContributor* │ │ │ │ │ + this.parsePersonConstructs(node, "contributor", atomAttrib); │ │ │ │ │ + │ │ │ │ │ + // atomId │ │ │ │ │ + atomAttrib.id = this.getFirstChildValue(node, atomns, "id", null); │ │ │ │ │ + │ │ │ │ │ + // atomLink* │ │ │ │ │ + nodes = this.getElementsByTagNameNS(node, atomns, "link"); │ │ │ │ │ + if (nodes.length > 0) { │ │ │ │ │ + atomAttrib.links = new Array(nodes.length); │ │ │ │ │ + } │ │ │ │ │ + var oAtts = ["rel", "type", "hreflang", "title", "length"]; │ │ │ │ │ + for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ + value = {}; │ │ │ │ │ + value.href = nodes[i].getAttribute("href"); │ │ │ │ │ + for (var j = 0, jj = oAtts.length; j < jj; j++) { │ │ │ │ │ + attval = nodes[i].getAttribute(oAtts[j]); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value[oAtts[j]] = attval; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + atomAttrib.links[i] = value; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atomPublished? │ │ │ │ │ + value = this.getFirstChildValue(node, atomns, "published", null); │ │ │ │ │ + if (value) { │ │ │ │ │ + atomAttrib.published = value; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atomRights? │ │ │ │ │ + value = this.getFirstChildValue(node, atomns, "rights", null); │ │ │ │ │ + if (value) { │ │ │ │ │ + atomAttrib.rights = value; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atomSource? -- not implemented │ │ │ │ │ + │ │ │ │ │ + // atomSummary? │ │ │ │ │ + value = this.getFirstChildValue(node, atomns, "summary", null); │ │ │ │ │ + if (value) { │ │ │ │ │ + atomAttrib.summary = value; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atomTitle │ │ │ │ │ + atomAttrib.title = this.getFirstChildValue( │ │ │ │ │ + node, atomns, "title", null │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + // atomUpdated │ │ │ │ │ + atomAttrib.updated = this.getFirstChildValue( │ │ │ │ │ + node, atomns, "updated", null │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + var featureAttrib = { │ │ │ │ │ + title: atomAttrib.title, │ │ │ │ │ + description: atomAttrib.summary, │ │ │ │ │ + atom: atomAttrib │ │ │ │ │ + }; │ │ │ │ │ + var geometry = this.parseLocations(node)[0]; │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry, featureAttrib); │ │ │ │ │ + feature.fid = atomAttrib.id; │ │ │ │ │ + return feature; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseFeatures │ │ │ │ │ + * Return features from an Atom entry or feed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} An Atom entry or feed node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ + */ │ │ │ │ │ + parseFeatures: function(node) { │ │ │ │ │ + var features = []; │ │ │ │ │ + var entries = this.getElementsByTagNameNS( │ │ │ │ │ + node, this.namespaces.atom, "entry" │ │ │ │ │ + ); │ │ │ │ │ + if (entries.length == 0) { │ │ │ │ │ + entries = [node]; │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, ii = entries.length; i < ii; i++) { │ │ │ │ │ + features.push(this.parseFeature(entries[i])); │ │ │ │ │ + } │ │ │ │ │ + return features; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseLocations │ │ │ │ │ + * Parse the locations from an Atom entry or feed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} An Atom entry or feed node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * Array({<OpenLayers.Geometry>}) │ │ │ │ │ + */ │ │ │ │ │ + parseLocations: function(node) { │ │ │ │ │ + var georssns = this.namespaces.georss; │ │ │ │ │ + │ │ │ │ │ + var locations = { │ │ │ │ │ + components: [] │ │ │ │ │ + }; │ │ │ │ │ + var where = this.getElementsByTagNameNS(node, georssns, "where"); │ │ │ │ │ + if (where && where.length > 0) { │ │ │ │ │ + if (!this.gmlParser) { │ │ │ │ │ + this.initGmlParser(); │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, ii = where.length; i < ii; i++) { │ │ │ │ │ + this.gmlParser.readChildNodes(where[i], locations); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var components = locations.components; │ │ │ │ │ + var point = this.getElementsByTagNameNS(node, georssns, "point"); │ │ │ │ │ + if (point && point.length > 0) { │ │ │ │ │ + for (var i = 0, ii = point.length; i < ii; i++) { │ │ │ │ │ + var xy = OpenLayers.String.trim( │ │ │ │ │ + point[i].firstChild.nodeValue │ │ │ │ │ + ).split(/\s+/); │ │ │ │ │ + if (xy.length != 2) { │ │ │ │ │ + xy = OpenLayers.String.trim( │ │ │ │ │ + point[i].firstChild.nodeValue │ │ │ │ │ + ).split(/\s*,\s*/); │ │ │ │ │ + } │ │ │ │ │ + components.push(new OpenLayers.Geometry.Point(xy[1], xy[0])); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var line = this.getElementsByTagNameNS(node, georssns, "line"); │ │ │ │ │ + if (line && line.length > 0) { │ │ │ │ │ + var coords; │ │ │ │ │ + var p; │ │ │ │ │ + var points; │ │ │ │ │ + for (var i = 0, ii = line.length; i < ii; i++) { │ │ │ │ │ + coords = OpenLayers.String.trim( │ │ │ │ │ + line[i].firstChild.nodeValue │ │ │ │ │ + ).split(/\s+/); │ │ │ │ │ + points = []; │ │ │ │ │ + for (var j = 0, jj = coords.length; j < jj; j += 2) { │ │ │ │ │ + p = new OpenLayers.Geometry.Point(coords[j + 1], coords[j]); │ │ │ │ │ + points.push(p); │ │ │ │ │ + } │ │ │ │ │ + components.push( │ │ │ │ │ + new OpenLayers.Geometry.LineString(points) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var polygon = this.getElementsByTagNameNS(node, georssns, "polygon"); │ │ │ │ │ + if (polygon && polygon.length > 0) { │ │ │ │ │ + var coords; │ │ │ │ │ + var p; │ │ │ │ │ + var points; │ │ │ │ │ + for (var i = 0, ii = polygon.length; i < ii; i++) { │ │ │ │ │ + coords = OpenLayers.String.trim( │ │ │ │ │ + polygon[i].firstChild.nodeValue │ │ │ │ │ + ).split(/\s+/); │ │ │ │ │ + points = []; │ │ │ │ │ + for (var j = 0, jj = coords.length; j < jj; j += 2) { │ │ │ │ │ + p = new OpenLayers.Geometry.Point(coords[j + 1], coords[j]); │ │ │ │ │ + points.push(p); │ │ │ │ │ + } │ │ │ │ │ + components.push( │ │ │ │ │ + new OpenLayers.Geometry.Polygon( │ │ │ │ │ + [new OpenLayers.Geometry.LinearRing(points)] │ │ │ │ │ + ) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + for (var i = 0, ii = components.length; i < ii; i++) { │ │ │ │ │ + if (components[i]) { │ │ │ │ │ + components[i].transform( │ │ │ │ │ + this.externalProjection, │ │ │ │ │ + this.internalProjection │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return components; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parsePersonConstruct │ │ │ │ │ + * Parse Atom person constructs from an Atom entry node. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} An Atom entry or feed node. │ │ │ │ │ + * name - {String} Construcy name ("author" or "contributor") │ │ │ │ │ + * data = {Object} Object in which to put parsed persons. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * An {Object}. │ │ │ │ │ + */ │ │ │ │ │ + parsePersonConstructs: function(node, name, data) { │ │ │ │ │ + var persons = []; │ │ │ │ │ + var atomns = this.namespaces.atom; │ │ │ │ │ + var nodes = this.getElementsByTagNameNS(node, atomns, name); │ │ │ │ │ + var oAtts = ["uri", "email"]; │ │ │ │ │ + for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ + var value = {}; │ │ │ │ │ + value.name = this.getFirstChildValue( │ │ │ │ │ + nodes[i], │ │ │ │ │ + atomns, │ │ │ │ │ + "name", │ │ │ │ │ + null │ │ │ │ │ + ); │ │ │ │ │ + for (var j = 0, jj = oAtts.length; j < jj; j++) { │ │ │ │ │ + var attval = this.getFirstChildValue( │ │ │ │ │ + nodes[i], │ │ │ │ │ + atomns, │ │ │ │ │ + oAtts[j], │ │ │ │ │ + null); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value[oAtts[j]] = attval; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + persons.push(value); │ │ │ │ │ + } │ │ │ │ │ + if (persons.length > 0) { │ │ │ │ │ + data[name + "s"] = persons; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.Atom" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/GML/v2.js │ │ │ │ │ + OpenLayers/Format/SOSGetObservation.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/GML/Base.js │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Format/SOSGetFeatureOfInterest.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.GML.v2 │ │ │ │ │ - * Parses GML version 2. │ │ │ │ │ + * Class: OpenLayers.Format.SOSGetObservation │ │ │ │ │ + * Read and write SOS GetObersation (to get the actual values from a sensor) │ │ │ │ │ + * version 1.0.0 │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.GML.Base> │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.GML.v2 = OpenLayers.Class(OpenLayers.Format.GML.Base, { │ │ │ │ │ +OpenLayers.Format.SOSGetObservation = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + ows: "http://www.opengis.net/ows", │ │ │ │ │ + gml: "http://www.opengis.net/gml", │ │ │ │ │ + sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ + ogc: "http://www.opengis.net/ogc", │ │ │ │ │ + om: "http://www.opengis.net/om/1.0", │ │ │ │ │ + sa: "http://www.opengis.net/sampling/1.0", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ + */ │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: VERSION │ │ │ │ │ + * {String} 1.0.0 │ │ │ │ │ + */ │ │ │ │ │ + VERSION: "1.0.0", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Property: schemaLocation │ │ │ │ │ - * {String} Schema location for a particular minor version. │ │ │ │ │ + * {String} Schema location │ │ │ │ │ */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/gml http://schemas.opengis.net/gml/2.1.2/feature.xsd", │ │ │ │ │ + schemaLocation: "http://www.opengis.net/sos/1.0 http://schemas.opengis.net/sos/1.0.0/sosGetObservation.xsd", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.GML.v2 │ │ │ │ │ - * Create a parser for GML v2. │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: "sos", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.SOSGetObservation │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * options - {Object} An optional object whose properties will be set on │ │ │ │ │ * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: read │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ - * featureNS - {String} Feature namespace (required). │ │ │ │ │ - * geometryName - {String} Geometry element name. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object containing the measurements │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.GML.Base.prototype.initialize.apply(this, [options]); │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + var info = { │ │ │ │ │ + measurements: [], │ │ │ │ │ + observations: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readNode(data, info); │ │ │ │ │ + return info; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: write │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} An SOS GetObservation request XML string. │ │ │ │ │ + */ │ │ │ │ │ + write: function(options) { │ │ │ │ │ + var node = this.writeNode("sos:GetObservation", options); │ │ │ │ │ + node.setAttribute("xmlns:om", this.namespaces.om); │ │ │ │ │ + node.setAttribute("xmlns:ogc", this.namespaces.ogc); │ │ │ │ │ + this.setAttributeNS( │ │ │ │ │ + node, this.namespaces.xsi, │ │ │ │ │ + "xsi:schemaLocation", this.schemaLocation │ │ │ │ │ + ); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [node]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Property: readers │ │ │ │ │ * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ * be applied when a namespaced node is found matching the function │ │ │ │ │ * name. The function will be applied in the scope of this parser │ │ │ │ │ * with two arguments: the node being read and a context object passed │ │ │ │ │ * from the parent. │ │ │ │ │ */ │ │ │ │ │ readers: { │ │ │ │ │ - "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "outerBoundaryIs": function(node, container) { │ │ │ │ │ - var obj = {}; │ │ │ │ │ + "om": { │ │ │ │ │ + "ObservationCollection": function(node, obj) { │ │ │ │ │ + obj.id = this.getAttributeNS(node, this.namespaces.gml, "id"); │ │ │ │ │ this.readChildNodes(node, obj); │ │ │ │ │ - container.outer = obj.components[0]; │ │ │ │ │ }, │ │ │ │ │ - "innerBoundaryIs": function(node, container) { │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - container.inner.push(obj.components[0]); │ │ │ │ │ + "member": function(node, observationCollection) { │ │ │ │ │ + this.readChildNodes(node, observationCollection); │ │ │ │ │ }, │ │ │ │ │ - "Box": function(node, container) { │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - if (!container.components) { │ │ │ │ │ - container.components = []; │ │ │ │ │ + "Measurement": function(node, observationCollection) { │ │ │ │ │ + var measurement = {}; │ │ │ │ │ + observationCollection.measurements.push(measurement); │ │ │ │ │ + this.readChildNodes(node, measurement); │ │ │ │ │ + }, │ │ │ │ │ + "Observation": function(node, observationCollection) { │ │ │ │ │ + var observation = {}; │ │ │ │ │ + observationCollection.observations.push(observation); │ │ │ │ │ + this.readChildNodes(node, observation); │ │ │ │ │ + }, │ │ │ │ │ + "samplingTime": function(node, measurement) { │ │ │ │ │ + var samplingTime = {}; │ │ │ │ │ + measurement.samplingTime = samplingTime; │ │ │ │ │ + this.readChildNodes(node, samplingTime); │ │ │ │ │ + }, │ │ │ │ │ + "observedProperty": function(node, measurement) { │ │ │ │ │ + measurement.observedProperty = │ │ │ │ │ + this.getAttributeNS(node, this.namespaces.xlink, "href"); │ │ │ │ │ + this.readChildNodes(node, measurement); │ │ │ │ │ + }, │ │ │ │ │ + "procedure": function(node, measurement) { │ │ │ │ │ + measurement.procedure = │ │ │ │ │ + this.getAttributeNS(node, this.namespaces.xlink, "href"); │ │ │ │ │ + this.readChildNodes(node, measurement); │ │ │ │ │ + }, │ │ │ │ │ + "featureOfInterest": function(node, observation) { │ │ │ │ │ + var foi = { │ │ │ │ │ + features: [] │ │ │ │ │ + }; │ │ │ │ │ + observation.fois = []; │ │ │ │ │ + observation.fois.push(foi); │ │ │ │ │ + this.readChildNodes(node, foi); │ │ │ │ │ + // postprocessing to get actual features │ │ │ │ │ + var features = []; │ │ │ │ │ + for (var i = 0, len = foi.features.length; i < len; i++) { │ │ │ │ │ + var feature = foi.features[i]; │ │ │ │ │ + features.push(new OpenLayers.Feature.Vector( │ │ │ │ │ + feature.components[0], feature.attributes)); │ │ │ │ │ + } │ │ │ │ │ + foi.features = features; │ │ │ │ │ + }, │ │ │ │ │ + "result": function(node, measurement) { │ │ │ │ │ + var result = {}; │ │ │ │ │ + measurement.result = result; │ │ │ │ │ + if (this.getChildValue(node) !== '') { │ │ │ │ │ + result.value = this.getChildValue(node); │ │ │ │ │ + result.uom = node.getAttribute("uom"); │ │ │ │ │ + } else { │ │ │ │ │ + this.readChildNodes(node, result); │ │ │ │ │ } │ │ │ │ │ - var min = obj.points[0]; │ │ │ │ │ - var max = obj.points[1]; │ │ │ │ │ - container.components.push( │ │ │ │ │ - new OpenLayers.Bounds(min.x, min.y, max.x, max.y) │ │ │ │ │ - ); │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.GML.Base.prototype.readers["gml"]), │ │ │ │ │ - "feature": OpenLayers.Format.GML.Base.prototype.readers["feature"], │ │ │ │ │ - "wfs": OpenLayers.Format.GML.Base.prototype.readers["wfs"] │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: write │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>) | OpenLayers.Feature.Vector} │ │ │ │ │ - * An array of features or a single feature. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} Given an array of features, a doc with a gml:featureMembers │ │ │ │ │ - * element will be returned. Given a single feature, a doc with a │ │ │ │ │ - * gml:featureMember element will be returned. │ │ │ │ │ - */ │ │ │ │ │ - write: function(features) { │ │ │ │ │ - var name; │ │ │ │ │ - if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ - // GML2 only has abstract feature collections │ │ │ │ │ - // wfs provides a feature collection from a well-known schema │ │ │ │ │ - name = "wfs:FeatureCollection"; │ │ │ │ │ - } else { │ │ │ │ │ - name = "gml:featureMember"; │ │ │ │ │ - } │ │ │ │ │ - var root = this.writeNode(name, features); │ │ │ │ │ - this.setAttributeNS( │ │ │ │ │ - root, this.namespaces["xsi"], │ │ │ │ │ - "xsi:schemaLocation", this.schemaLocation │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [root]); │ │ │ │ │ + }, │ │ │ │ │ + "sa": OpenLayers.Format.SOSGetFeatureOfInterest.prototype.readers.sa, │ │ │ │ │ + "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "TimeInstant": function(node, samplingTime) { │ │ │ │ │ + var timeInstant = {}; │ │ │ │ │ + samplingTime.timeInstant = timeInstant; │ │ │ │ │ + this.readChildNodes(node, timeInstant); │ │ │ │ │ + }, │ │ │ │ │ + "timePosition": function(node, timeInstant) { │ │ │ │ │ + timeInstant.timePosition = this.getChildValue(node); │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.SOSGetFeatureOfInterest.prototype.readers.gml) │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Property: writers │ │ │ │ │ * As a compliment to the readers property, this structure contains public │ │ │ │ │ * writing functions grouped by namespace alias and named like the │ │ │ │ │ * node names they produce. │ │ │ │ │ */ │ │ │ │ │ writers: { │ │ │ │ │ - "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "Point": function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:Point"); │ │ │ │ │ - this.writeNode("coordinates", [geometry], node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "coordinates": function(points) { │ │ │ │ │ - var numPoints = points.length; │ │ │ │ │ - var parts = new Array(numPoints); │ │ │ │ │ - var point; │ │ │ │ │ - for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ - point = points[i]; │ │ │ │ │ - if (this.xy) { │ │ │ │ │ - parts[i] = point.x + "," + point.y; │ │ │ │ │ - } else { │ │ │ │ │ - parts[i] = point.y + "," + point.x; │ │ │ │ │ - } │ │ │ │ │ - if (point.z != undefined) { // allow null or undefined │ │ │ │ │ - parts[i] += "," + point.z; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return this.createElementNSPlus("gml:coordinates", { │ │ │ │ │ + "sos": { │ │ │ │ │ + "GetObservation": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("GetObservation", { │ │ │ │ │ attributes: { │ │ │ │ │ - decimal: ".", │ │ │ │ │ - cs: ",", │ │ │ │ │ - ts: " " │ │ │ │ │ - }, │ │ │ │ │ - value: (numPoints == 1) ? parts[0] : parts.join(" ") │ │ │ │ │ + version: this.VERSION, │ │ │ │ │ + service: 'SOS' │ │ │ │ │ + } │ │ │ │ │ }); │ │ │ │ │ + this.writeNode("offering", options, node); │ │ │ │ │ + if (options.eventTime) { │ │ │ │ │ + this.writeNode("eventTime", options, node); │ │ │ │ │ + } │ │ │ │ │ + for (var procedure in options.procedures) { │ │ │ │ │ + this.writeNode("procedure", options.procedures[procedure], node); │ │ │ │ │ + } │ │ │ │ │ + for (var observedProperty in options.observedProperties) { │ │ │ │ │ + this.writeNode("observedProperty", options.observedProperties[observedProperty], node); │ │ │ │ │ + } │ │ │ │ │ + if (options.foi) { │ │ │ │ │ + this.writeNode("featureOfInterest", options.foi, node); │ │ │ │ │ + } │ │ │ │ │ + this.writeNode("responseFormat", options, node); │ │ │ │ │ + if (options.resultModel) { │ │ │ │ │ + this.writeNode("resultModel", options, node); │ │ │ │ │ + } │ │ │ │ │ + if (options.responseMode) { │ │ │ │ │ + this.writeNode("responseMode", options, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ - "LineString": function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:LineString"); │ │ │ │ │ - this.writeNode("coordinates", geometry.components, node); │ │ │ │ │ + "featureOfInterest": function(foi) { │ │ │ │ │ + var node = this.createElementNSPlus("featureOfInterest"); │ │ │ │ │ + this.writeNode("ObjectID", foi.objectId, node); │ │ │ │ │ return node; │ │ │ │ │ }, │ │ │ │ │ - "Polygon": function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:Polygon"); │ │ │ │ │ - this.writeNode("outerBoundaryIs", geometry.components[0], node); │ │ │ │ │ - for (var i = 1; i < geometry.components.length; ++i) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "innerBoundaryIs", geometry.components[i], node │ │ │ │ │ - ); │ │ │ │ │ + "ObjectID": function(options) { │ │ │ │ │ + return this.createElementNSPlus("ObjectID", { │ │ │ │ │ + value: options │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "responseFormat": function(options) { │ │ │ │ │ + return this.createElementNSPlus("responseFormat", { │ │ │ │ │ + value: options.responseFormat │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "procedure": function(procedure) { │ │ │ │ │ + return this.createElementNSPlus("procedure", { │ │ │ │ │ + value: procedure │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "offering": function(options) { │ │ │ │ │ + return this.createElementNSPlus("offering", { │ │ │ │ │ + value: options.offering │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "observedProperty": function(observedProperty) { │ │ │ │ │ + return this.createElementNSPlus("observedProperty", { │ │ │ │ │ + value: observedProperty │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "eventTime": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("eventTime"); │ │ │ │ │ + if (options.eventTime === 'latest') { │ │ │ │ │ + this.writeNode("ogc:TM_Equals", options, node); │ │ │ │ │ } │ │ │ │ │ return node; │ │ │ │ │ }, │ │ │ │ │ - "outerBoundaryIs": function(ring) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:outerBoundaryIs"); │ │ │ │ │ - this.writeNode("LinearRing", ring, node); │ │ │ │ │ - return node; │ │ │ │ │ + "resultModel": function(options) { │ │ │ │ │ + return this.createElementNSPlus("resultModel", { │ │ │ │ │ + value: options.resultModel │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ - "innerBoundaryIs": function(ring) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:innerBoundaryIs"); │ │ │ │ │ - this.writeNode("LinearRing", ring, node); │ │ │ │ │ + "responseMode": function(options) { │ │ │ │ │ + return this.createElementNSPlus("responseMode", { │ │ │ │ │ + value: options.responseMode │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "ogc": { │ │ │ │ │ + "TM_Equals": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:TM_Equals"); │ │ │ │ │ + this.writeNode("ogc:PropertyName", { │ │ │ │ │ + property: "urn:ogc:data:time:iso8601" │ │ │ │ │ + }, node); │ │ │ │ │ + if (options.eventTime === 'latest') { │ │ │ │ │ + this.writeNode("gml:TimeInstant", { │ │ │ │ │ + value: 'latest' │ │ │ │ │ + }, node); │ │ │ │ │ + } │ │ │ │ │ return node; │ │ │ │ │ }, │ │ │ │ │ - "LinearRing": function(ring) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:LinearRing"); │ │ │ │ │ - this.writeNode("coordinates", ring.components, node); │ │ │ │ │ + "PropertyName": function(options) { │ │ │ │ │ + return this.createElementNSPlus("ogc:PropertyName", { │ │ │ │ │ + value: options.property │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "gml": { │ │ │ │ │ + "TimeInstant": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:TimeInstant"); │ │ │ │ │ + this.writeNode("gml:timePosition", options, node); │ │ │ │ │ return node; │ │ │ │ │ }, │ │ │ │ │ - "Box": function(bounds) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:Box"); │ │ │ │ │ - this.writeNode("coordinates", [{ │ │ │ │ │ - x: bounds.left, │ │ │ │ │ - y: bounds.bottom │ │ │ │ │ - }, { │ │ │ │ │ - x: bounds.right, │ │ │ │ │ - y: bounds.top │ │ │ │ │ - }], node); │ │ │ │ │ - // srsName attribute is optional for gml:Box │ │ │ │ │ - if (this.srsName) { │ │ │ │ │ - node.setAttribute("srsName", this.srsName); │ │ │ │ │ - } │ │ │ │ │ + "timePosition": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:timePosition", { │ │ │ │ │ + value: options.value │ │ │ │ │ + }); │ │ │ │ │ return node; │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.GML.Base.prototype.writers["gml"]), │ │ │ │ │ - "feature": OpenLayers.Format.GML.Base.prototype.writers["feature"], │ │ │ │ │ - "wfs": OpenLayers.Format.GML.Base.prototype.writers["wfs"] │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.GML.v2" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SOSGetObservation" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/Filter/v1_0_0.js │ │ │ │ │ + OpenLayers/Format/XLS.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/GML/v2.js │ │ │ │ │ - * @requires OpenLayers/Format/Filter/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.Filter.v1_0_0 │ │ │ │ │ - * Write ogc:Filter version 1.0.0. │ │ │ │ │ + * Class: OpenLayers.Format.XLS │ │ │ │ │ + * Read/Write XLS (OpenLS). Create a new instance with the <OpenLayers.Format.XLS> │ │ │ │ │ + * constructor. Currently only implemented for Location Utility Services, more │ │ │ │ │ + * specifically only for Geocoding. No support for Reverse Geocoding as yet. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.GML.v2> │ │ │ │ │ - * - <OpenLayers.Format.Filter.v1> │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.Filter.v1_0_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.GML.v2, OpenLayers.Format.Filter.v1, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: VERSION │ │ │ │ │ - * {String} 1.0.0 │ │ │ │ │ - */ │ │ │ │ │ - VERSION: "1.0.0", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} http://www.opengis.net/ogc/filter/1.0.0/filter.xsd │ │ │ │ │ - */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/ogc/filter/1.0.0/filter.xsd", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.Filter.v1_0_0 │ │ │ │ │ - * Instances of this class are not created directly. Use the │ │ │ │ │ - * <OpenLayers.Format.Filter> constructor instead. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.GML.v2.prototype.initialize.apply( │ │ │ │ │ - this, [options] │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ +OpenLayers.Format.XLS = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "ogc": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "PropertyIsEqualTo": function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.EQUAL_TO │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - obj.filters.push(filter); │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsNotEqualTo": function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.NOT_EQUAL_TO │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - obj.filters.push(filter); │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsLike": function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.LIKE │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - var wildCard = node.getAttribute("wildCard"); │ │ │ │ │ - var singleChar = node.getAttribute("singleChar"); │ │ │ │ │ - var esc = node.getAttribute("escape"); │ │ │ │ │ - filter.value2regex(wildCard, singleChar, esc); │ │ │ │ │ - obj.filters.push(filter); │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.Filter.v1.prototype.readers["ogc"]), │ │ │ │ │ - "gml": OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ - "feature": OpenLayers.Format.GML.v2.prototype.readers["feature"] │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.1.0". │ │ │ │ │ + */ │ │ │ │ │ + defaultVersion: "1.1.0", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ - */ │ │ │ │ │ - writers: { │ │ │ │ │ - "ogc": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "PropertyIsEqualTo": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsEqualTo"); │ │ │ │ │ - // no ogc:expression handling for PropertyName for now │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - // handle Literals or Functions for now │ │ │ │ │ - this.writeOgcExpression(filter.value, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsNotEqualTo": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsNotEqualTo"); │ │ │ │ │ - // no ogc:expression handling for PropertyName for now │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - // handle Literals or Functions for now │ │ │ │ │ - this.writeOgcExpression(filter.value, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsLike": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsLike", { │ │ │ │ │ - attributes: { │ │ │ │ │ - wildCard: "*", │ │ │ │ │ - singleChar: ".", │ │ │ │ │ - escape: "!" │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - // no ogc:expression handling for now │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - // convert regex string to ogc string │ │ │ │ │ - this.writeNode("Literal", filter.regex2value(), node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "BBOX": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:BBOX"); │ │ │ │ │ - // PropertyName is mandatory in 1.0.0, but e.g. GeoServer also │ │ │ │ │ - // accepts filters without it. When this is used with │ │ │ │ │ - // OpenLayers.Protocol.WFS, OpenLayers.Format.WFST will set a │ │ │ │ │ - // missing filter.property to the geometryName that is │ │ │ │ │ - // configured with the protocol, which defaults to "the_geom". │ │ │ │ │ - // So the only way to omit this mandatory property is to not │ │ │ │ │ - // set the property on the filter and to set the geometryName │ │ │ │ │ - // on the WFS protocol to null. The latter also happens when │ │ │ │ │ - // the protocol is configured without a geometryName and a │ │ │ │ │ - // featureNS. │ │ │ │ │ - filter.property && this.writeNode("PropertyName", filter, node); │ │ │ │ │ - var box = this.writeNode("gml:Box", filter.value, node); │ │ │ │ │ - if (filter.projection) { │ │ │ │ │ - box.setAttribute("srsName", filter.projection); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.Filter.v1.prototype.writers["ogc"]), │ │ │ │ │ - "gml": OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ - "feature": OpenLayers.Format.GML.v2.prototype.writers["feature"] │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: stringifyOutput │ │ │ │ │ + * {Boolean} If true, write will return a string otherwise a DOMElement. │ │ │ │ │ + * Default is true. │ │ │ │ │ + */ │ │ │ │ │ + stringifyOutput: true, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: writeSpatial │ │ │ │ │ - * │ │ │ │ │ - * Read a {<OpenLayers.Filter.Spatial>} filter and converts it into XML. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter.Spatial>} The filter. │ │ │ │ │ - * name - {String} Name of the generated XML element. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The created XML element. │ │ │ │ │ - */ │ │ │ │ │ - writeSpatial: function(filter, name) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:" + name); │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - if (filter.value instanceof OpenLayers.Filter.Function) { │ │ │ │ │ - this.writeNode("Function", filter.value, node); │ │ │ │ │ - } else { │ │ │ │ │ - var child; │ │ │ │ │ - if (filter.value instanceof OpenLayers.Geometry) { │ │ │ │ │ - child = this.writeNode("feature:_geometry", filter.value).firstChild; │ │ │ │ │ - } else { │ │ │ │ │ - child = this.writeNode("gml:Box", filter.value); │ │ │ │ │ - } │ │ │ │ │ - if (filter.projection) { │ │ │ │ │ - child.setAttribute("srsName", filter.projection); │ │ │ │ │ - } │ │ │ │ │ - node.appendChild(child); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.XLS │ │ │ │ │ + * Create a new parser for XLS. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Write out an XLS request. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * request - {Object} An object representing the LUS request. │ │ │ │ │ + * options - {Object} Optional configuration object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} An XLS document string. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.Filter.v1_0_0" │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read an XLS doc and return an object representing the result. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String | DOMElement} Data to read. │ │ │ │ │ + * options - {Object} Options for the reader. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object representing the GeocodeResponse. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.XLS" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/CSWGetRecords/v2_0_2.js │ │ │ │ │ + OpenLayers/Format/CSWGetDomain/v2_0_2.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/CSWGetRecords.js │ │ │ │ │ - * @requires OpenLayers/Format/Filter/v1_0_0.js │ │ │ │ │ - * @requires OpenLayers/Format/Filter/v1_1_0.js │ │ │ │ │ - * @requires OpenLayers/Format/OWSCommon/v1_0_0.js │ │ │ │ │ + * @requires OpenLayers/Format/CSWGetDomain.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.CSWGetRecords.v2_0_2 │ │ │ │ │ - * A format for creating CSWGetRecords v2.0.2 transactions. │ │ │ │ │ + * Class: OpenLayers.Format.CSWGetDomain.v2_0_2 │ │ │ │ │ + * A format for creating CSWGetDomain v2.0.2 transactions. │ │ │ │ │ * Create a new instance with the │ │ │ │ │ - * <OpenLayers.Format.CSWGetRecords.v2_0_2> constructor. │ │ │ │ │ + * <OpenLayers.Format.CSWGetDomain.v2_0_2> constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Format.XML> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.CSWGetRecords.v2_0_2 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Format.CSWGetDomain.v2_0_2 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Property: namespaces │ │ │ │ │ * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ */ │ │ │ │ │ namespaces: { │ │ │ │ │ - csw: "http://www.opengis.net/cat/csw/2.0.2", │ │ │ │ │ - dc: "http://purl.org/dc/elements/1.1/", │ │ │ │ │ - dct: "http://purl.org/dc/terms/", │ │ │ │ │ - gmd: "http://www.isotc211.org/2005/gmd", │ │ │ │ │ - geonet: "http://www.fao.org/geonetwork", │ │ │ │ │ - ogc: "http://www.opengis.net/ogc", │ │ │ │ │ - ows: "http://www.opengis.net/ows", │ │ │ │ │ xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + csw: "http://www.opengis.net/cat/csw/2.0.2" │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Property: defaultPrefix │ │ │ │ │ * {String} The default prefix (used by Format.XML). │ │ │ │ │ */ │ │ │ │ │ defaultPrefix: "csw", │ │ │ │ │ @@ -52325,116 +53121,43 @@ │ │ │ │ │ * Property: schemaLocation │ │ │ │ │ * {String} http://www.opengis.net/cat/csw/2.0.2 │ │ │ │ │ * http://schemas.opengis.net/csw/2.0.2/CSW-discovery.xsd │ │ │ │ │ */ │ │ │ │ │ schemaLocation: "http://www.opengis.net/cat/csw/2.0.2 http://schemas.opengis.net/csw/2.0.2/CSW-discovery.xsd", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: requestId │ │ │ │ │ - * {String} Value of the requestId attribute of the GetRecords element. │ │ │ │ │ - */ │ │ │ │ │ - requestId: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: resultType │ │ │ │ │ - * {String} Value of the resultType attribute of the GetRecords element, │ │ │ │ │ - * specifies the result type in the GetRecords response, "hits" is │ │ │ │ │ - * the default. │ │ │ │ │ - */ │ │ │ │ │ - resultType: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: outputFormat │ │ │ │ │ - * {String} Value of the outputFormat attribute of the GetRecords element, │ │ │ │ │ - * specifies the format of the GetRecords response, │ │ │ │ │ - * "application/xml" is the default. │ │ │ │ │ - */ │ │ │ │ │ - outputFormat: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: outputSchema │ │ │ │ │ - * {String} Value of the outputSchema attribute of the GetRecords element, │ │ │ │ │ - * specifies the schema of the GetRecords response. │ │ │ │ │ - */ │ │ │ │ │ - outputSchema: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: startPosition │ │ │ │ │ - * {String} Value of the startPosition attribute of the GetRecords element, │ │ │ │ │ - * specifies the start position (offset+1) for the GetRecords response, │ │ │ │ │ - * 1 is the default. │ │ │ │ │ - */ │ │ │ │ │ - startPosition: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: maxRecords │ │ │ │ │ - * {String} Value of the maxRecords attribute of the GetRecords element, │ │ │ │ │ - * specifies the maximum number of records in the GetRecords response, │ │ │ │ │ - * 10 is the default. │ │ │ │ │ - */ │ │ │ │ │ - maxRecords: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: DistributedSearch │ │ │ │ │ - * {String} Value of the csw:DistributedSearch element, used when writing │ │ │ │ │ - * a csw:GetRecords document. │ │ │ │ │ - */ │ │ │ │ │ - DistributedSearch: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: ResponseHandler │ │ │ │ │ - * {Array({String})} Values of the csw:ResponseHandler elements, used when │ │ │ │ │ - * writting a csw:GetRecords document. │ │ │ │ │ - */ │ │ │ │ │ - ResponseHandler: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: Query │ │ │ │ │ - * {String} Value of the csw:Query element, used when writing a csw:GetRecords │ │ │ │ │ - * document. │ │ │ │ │ + * APIProperty: PropertyName │ │ │ │ │ + * {String} Value of the csw:PropertyName element, used when │ │ │ │ │ + * writing a GetDomain document. │ │ │ │ │ */ │ │ │ │ │ - Query: null, │ │ │ │ │ + PropertyName: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ + * APIProperty: ParameterName │ │ │ │ │ + * {String} Value of the csw:ParameterName element, used when │ │ │ │ │ + * writing a GetDomain document. │ │ │ │ │ */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ - }, │ │ │ │ │ + ParameterName: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.CSWGetRecords.v2_0_2 │ │ │ │ │ - * A class for parsing and generating CSWGetRecords v2.0.2 transactions. │ │ │ │ │ + * Constructor: OpenLayers.Format.CSWGetDomain.v2_0_2 │ │ │ │ │ + * A class for parsing and generating CSWGetDomain v2.0.2 transactions. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ * instance. │ │ │ │ │ * │ │ │ │ │ - * Valid options properties (documented as class properties): │ │ │ │ │ - * - requestId │ │ │ │ │ - * - resultType │ │ │ │ │ - * - outputFormat │ │ │ │ │ - * - outputSchema │ │ │ │ │ - * - startPosition │ │ │ │ │ - * - maxRecords │ │ │ │ │ - * - DistributedSearch │ │ │ │ │ - * - ResponseHandler │ │ │ │ │ - * - Query │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * - PropertyName │ │ │ │ │ + * - ParameterName │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIMethod: read │ │ │ │ │ - * Parse the response from a GetRecords request. │ │ │ │ │ + * Parse the response from a GetDomain request. │ │ │ │ │ */ │ │ │ │ │ read: function(data) { │ │ │ │ │ if (typeof data == "string") { │ │ │ │ │ data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ } │ │ │ │ │ if (data && data.nodeType == 9) { │ │ │ │ │ data = data.documentElement; │ │ │ │ │ @@ -52450,296 +53173,152 @@ │ │ │ │ │ * be applied when a namespaced node is found matching the function │ │ │ │ │ * name. The function will be applied in the scope of this parser │ │ │ │ │ * with two arguments: the node being read and a context object passed │ │ │ │ │ * from the parent. │ │ │ │ │ */ │ │ │ │ │ readers: { │ │ │ │ │ "csw": { │ │ │ │ │ - "GetRecordsResponse": function(node, obj) { │ │ │ │ │ - obj.records = []; │ │ │ │ │ + "GetDomainResponse": function(node, obj) { │ │ │ │ │ this.readChildNodes(node, obj); │ │ │ │ │ - var version = this.getAttributeNS(node, "", 'version'); │ │ │ │ │ - if (version != "") { │ │ │ │ │ - obj.version = version; │ │ │ │ │ + }, │ │ │ │ │ + "DomainValues": function(node, obj) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(obj.DomainValues))) { │ │ │ │ │ + obj.DomainValues = []; │ │ │ │ │ + } │ │ │ │ │ + var attrs = node.attributes; │ │ │ │ │ + var domainValue = {}; │ │ │ │ │ + for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ + domainValue[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ } │ │ │ │ │ + this.readChildNodes(node, domainValue); │ │ │ │ │ + obj.DomainValues.push(domainValue); │ │ │ │ │ }, │ │ │ │ │ - "RequestId": function(node, obj) { │ │ │ │ │ - obj.RequestId = this.getChildValue(node); │ │ │ │ │ + "PropertyName": function(node, obj) { │ │ │ │ │ + obj.PropertyName = this.getChildValue(node); │ │ │ │ │ }, │ │ │ │ │ - "SearchStatus": function(node, obj) { │ │ │ │ │ - obj.SearchStatus = {}; │ │ │ │ │ - var timestamp = this.getAttributeNS(node, "", 'timestamp'); │ │ │ │ │ - if (timestamp != "") { │ │ │ │ │ - obj.SearchStatus.timestamp = timestamp; │ │ │ │ │ + "ParameterName": function(node, obj) { │ │ │ │ │ + obj.ParameterName = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "ListOfValues": function(node, obj) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(obj.ListOfValues))) { │ │ │ │ │ + obj.ListOfValues = []; │ │ │ │ │ } │ │ │ │ │ + this.readChildNodes(node, obj.ListOfValues); │ │ │ │ │ }, │ │ │ │ │ - "SearchResults": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ + "Value": function(node, obj) { │ │ │ │ │ var attrs = node.attributes; │ │ │ │ │ - var SearchResults = {}; │ │ │ │ │ + var value = {}; │ │ │ │ │ for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - if ((attrs[i].name == "numberOfRecordsMatched") || │ │ │ │ │ - (attrs[i].name == "numberOfRecordsReturned") || │ │ │ │ │ - (attrs[i].name == "nextRecord")) { │ │ │ │ │ - SearchResults[attrs[i].name] = parseInt(attrs[i].nodeValue); │ │ │ │ │ - } else { │ │ │ │ │ - SearchResults[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ - } │ │ │ │ │ + value[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ } │ │ │ │ │ - obj.SearchResults = SearchResults; │ │ │ │ │ + value.value = this.getChildValue(node); │ │ │ │ │ + obj.push({ │ │ │ │ │ + Value: value │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ - "SummaryRecord": function(node, obj) { │ │ │ │ │ - var record = { │ │ │ │ │ - type: "SummaryRecord" │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, record); │ │ │ │ │ - obj.records.push(record); │ │ │ │ │ + "ConceptualScheme": function(node, obj) { │ │ │ │ │ + obj.ConceptualScheme = {}; │ │ │ │ │ + this.readChildNodes(node, obj.ConceptualScheme); │ │ │ │ │ }, │ │ │ │ │ - "BriefRecord": function(node, obj) { │ │ │ │ │ - var record = { │ │ │ │ │ - type: "BriefRecord" │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, record); │ │ │ │ │ - obj.records.push(record); │ │ │ │ │ + "Name": function(node, obj) { │ │ │ │ │ + obj.Name = this.getChildValue(node); │ │ │ │ │ }, │ │ │ │ │ - "DCMIRecord": function(node, obj) { │ │ │ │ │ - var record = { │ │ │ │ │ - type: "DCMIRecord" │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, record); │ │ │ │ │ - obj.records.push(record); │ │ │ │ │ + "Document": function(node, obj) { │ │ │ │ │ + obj.Document = this.getChildValue(node); │ │ │ │ │ }, │ │ │ │ │ - "Record": function(node, obj) { │ │ │ │ │ - var record = { │ │ │ │ │ - type: "Record" │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, record); │ │ │ │ │ - obj.records.push(record); │ │ │ │ │ + "Authority": function(node, obj) { │ │ │ │ │ + obj.Authority = this.getChildValue(node); │ │ │ │ │ }, │ │ │ │ │ - "*": function(node, obj) { │ │ │ │ │ - var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ - obj[name] = this.getChildValue(node); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "geonet": { │ │ │ │ │ - "info": function(node, obj) { │ │ │ │ │ - var gninfo = {}; │ │ │ │ │ - this.readChildNodes(node, gninfo); │ │ │ │ │ - obj.gninfo = gninfo; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "dc": { │ │ │ │ │ - // audience, contributor, coverage, creator, date, description, format, │ │ │ │ │ - // identifier, language, provenance, publisher, relation, rights, │ │ │ │ │ - // rightsHolder, source, subject, title, type, URI │ │ │ │ │ - "*": function(node, obj) { │ │ │ │ │ - var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ - if (!(OpenLayers.Util.isArray(obj[name]))) { │ │ │ │ │ - obj[name] = []; │ │ │ │ │ - } │ │ │ │ │ - var dc_element = {}; │ │ │ │ │ + "RangeOfValues": function(node, obj) { │ │ │ │ │ + obj.RangeOfValues = {}; │ │ │ │ │ + this.readChildNodes(node, obj.RangeOfValues); │ │ │ │ │ + }, │ │ │ │ │ + "MinValue": function(node, obj) { │ │ │ │ │ var attrs = node.attributes; │ │ │ │ │ + var value = {}; │ │ │ │ │ for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - dc_element[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ - } │ │ │ │ │ - dc_element.value = this.getChildValue(node); │ │ │ │ │ - if (dc_element.value != "") { │ │ │ │ │ - obj[name].push(dc_element); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "dct": { │ │ │ │ │ - // abstract, modified, spatial │ │ │ │ │ - "*": function(node, obj) { │ │ │ │ │ - var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ - if (!(OpenLayers.Util.isArray(obj[name]))) { │ │ │ │ │ - obj[name] = []; │ │ │ │ │ + value[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ } │ │ │ │ │ - obj[name].push(this.getChildValue(node)); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "ows": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "BoundingBox": function(node, obj) { │ │ │ │ │ - if (obj.bounds) { │ │ │ │ │ - obj.BoundingBox = [{ │ │ │ │ │ - crs: obj.projection, │ │ │ │ │ - value: [ │ │ │ │ │ - obj.bounds.left, │ │ │ │ │ - obj.bounds.bottom, │ │ │ │ │ - obj.bounds.right, │ │ │ │ │ - obj.bounds.top │ │ │ │ │ - ] │ │ │ │ │ - }]; │ │ │ │ │ - delete obj.projection; │ │ │ │ │ - delete obj.bounds; │ │ │ │ │ + value.value = this.getChildValue(node); │ │ │ │ │ + obj.MinValue = value; │ │ │ │ │ + }, │ │ │ │ │ + "MaxValue": function(node, obj) { │ │ │ │ │ + var attrs = node.attributes; │ │ │ │ │ + var value = {}; │ │ │ │ │ + for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ + value[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Format.OWSCommon.v1_0_0.prototype.readers["ows"]["BoundingBox"].apply( │ │ │ │ │ - this, arguments); │ │ │ │ │ + value.value = this.getChildValue(node); │ │ │ │ │ + obj.MaxValue = value; │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.OWSCommon.v1_0_0.prototype.readers["ows"]) │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: write │ │ │ │ │ - * Given an configuration js object, write a CSWGetRecords request. │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Given an configuration js object, write a CSWGetDomain request. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * options - {Object} A object mapping the request. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A serialized CSWGetRecords request. │ │ │ │ │ + * {String} A serialized CSWGetDomain request. │ │ │ │ │ */ │ │ │ │ │ write: function(options) { │ │ │ │ │ - var node = this.writeNode("csw:GetRecords", options); │ │ │ │ │ - node.setAttribute("xmlns:gmd", this.namespaces.gmd); │ │ │ │ │ + var node = this.writeNode("csw:GetDomain", options); │ │ │ │ │ return OpenLayers.Format.XML.prototype.write.apply(this, [node]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Property: writers │ │ │ │ │ * As a compliment to the readers property, this structure contains public │ │ │ │ │ * writing functions grouped by namespace alias and named like the │ │ │ │ │ * node names they produce. │ │ │ │ │ */ │ │ │ │ │ writers: { │ │ │ │ │ "csw": { │ │ │ │ │ - "GetRecords": function(options) { │ │ │ │ │ - if (!options) { │ │ │ │ │ - options = {}; │ │ │ │ │ - } │ │ │ │ │ - var node = this.createElementNSPlus("csw:GetRecords", { │ │ │ │ │ + "GetDomain": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:GetDomain", { │ │ │ │ │ attributes: { │ │ │ │ │ service: "CSW", │ │ │ │ │ - version: this.version, │ │ │ │ │ - requestId: options.requestId || this.requestId, │ │ │ │ │ - resultType: options.resultType || this.resultType, │ │ │ │ │ - outputFormat: options.outputFormat || this.outputFormat, │ │ │ │ │ - outputSchema: options.outputSchema || this.outputSchema, │ │ │ │ │ - startPosition: options.startPosition || this.startPosition, │ │ │ │ │ - maxRecords: options.maxRecords || this.maxRecords │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (options.DistributedSearch || this.DistributedSearch) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "csw:DistributedSearch", │ │ │ │ │ - options.DistributedSearch || this.DistributedSearch, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - var ResponseHandler = options.ResponseHandler || this.ResponseHandler; │ │ │ │ │ - if (OpenLayers.Util.isArray(ResponseHandler) && ResponseHandler.length > 0) { │ │ │ │ │ - // ResponseHandler must be a non-empty array │ │ │ │ │ - for (var i = 0, len = ResponseHandler.length; i < len; i++) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "csw:ResponseHandler", │ │ │ │ │ - ResponseHandler[i], │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.writeNode("Query", options.Query || this.Query, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "DistributedSearch": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:DistributedSearch", { │ │ │ │ │ - attributes: { │ │ │ │ │ - hopCount: options.hopCount │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "ResponseHandler": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:ResponseHandler", { │ │ │ │ │ - value: options.value │ │ │ │ │ - }); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Query": function(options) { │ │ │ │ │ - if (!options) { │ │ │ │ │ - options = {}; │ │ │ │ │ - } │ │ │ │ │ - var node = this.createElementNSPlus("csw:Query", { │ │ │ │ │ - attributes: { │ │ │ │ │ - typeNames: options.typeNames || "csw:Record" │ │ │ │ │ + version: this.version │ │ │ │ │ } │ │ │ │ │ }); │ │ │ │ │ - var ElementName = options.ElementName; │ │ │ │ │ - if (OpenLayers.Util.isArray(ElementName) && ElementName.length > 0) { │ │ │ │ │ - // ElementName must be a non-empty array │ │ │ │ │ - for (var i = 0, len = ElementName.length; i < len; i++) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "csw:ElementName", │ │ │ │ │ - ElementName[i], │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "csw:ElementSetName", │ │ │ │ │ - options.ElementSetName || { │ │ │ │ │ - value: 'summary' │ │ │ │ │ - }, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (options.Constraint) { │ │ │ │ │ + if (options.PropertyName || this.PropertyName) { │ │ │ │ │ this.writeNode( │ │ │ │ │ - "csw:Constraint", │ │ │ │ │ - options.Constraint, │ │ │ │ │ + "csw:PropertyName", │ │ │ │ │ + options.PropertyName || this.PropertyName, │ │ │ │ │ node │ │ │ │ │ ); │ │ │ │ │ - } │ │ │ │ │ - if (options.SortBy) { │ │ │ │ │ + } else if (options.ParameterName || this.ParameterName) { │ │ │ │ │ this.writeNode( │ │ │ │ │ - "ogc:SortBy", │ │ │ │ │ - options.SortBy, │ │ │ │ │ + "csw:ParameterName", │ │ │ │ │ + options.ParameterName || this.ParameterName, │ │ │ │ │ node │ │ │ │ │ ); │ │ │ │ │ } │ │ │ │ │ + this.readChildNodes(node, options); │ │ │ │ │ return node; │ │ │ │ │ }, │ │ │ │ │ - "ElementName": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:ElementName", { │ │ │ │ │ - value: options.value │ │ │ │ │ - }); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "ElementSetName": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:ElementSetName", { │ │ │ │ │ - attributes: { │ │ │ │ │ - typeNames: options.typeNames │ │ │ │ │ - }, │ │ │ │ │ - value: options.value │ │ │ │ │ + "PropertyName": function(value) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:PropertyName", { │ │ │ │ │ + value: value │ │ │ │ │ }); │ │ │ │ │ return node; │ │ │ │ │ }, │ │ │ │ │ - "Constraint": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:Constraint", { │ │ │ │ │ - attributes: { │ │ │ │ │ - version: options.version │ │ │ │ │ - } │ │ │ │ │ + "ParameterName": function(value) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:ParameterName", { │ │ │ │ │ + value: value │ │ │ │ │ }); │ │ │ │ │ - if (options.Filter) { │ │ │ │ │ - var format = new OpenLayers.Format.Filter({ │ │ │ │ │ - version: options.version │ │ │ │ │ - }); │ │ │ │ │ - node.appendChild(format.write(options.Filter)); │ │ │ │ │ - } else if (options.CqlText) { │ │ │ │ │ - var child = this.createElementNSPlus("CqlText", { │ │ │ │ │ - value: options.CqlText.value │ │ │ │ │ - }); │ │ │ │ │ - node.appendChild(child); │ │ │ │ │ - } │ │ │ │ │ return node; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "ogc": OpenLayers.Format.Filter.v1_1_0.prototype.writers["ogc"] │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.CSWGetRecords.v2_0_2" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.CSWGetDomain.v2_0_2" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Format/SLD/v1.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ @@ -54139,102 +54718,1908 @@ │ │ │ │ │ * this instance. │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.SLD.v1_0_0" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/OWSContext/v0_3_1.js │ │ │ │ │ + OpenLayers/Format/SLD/v1_0_0_GeoServer.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/KML.js │ │ │ │ │ - * @requires OpenLayers/Format/GML.js │ │ │ │ │ - * @requires OpenLayers/Format/GML/v2.js │ │ │ │ │ * @requires OpenLayers/Format/SLD/v1_0_0.js │ │ │ │ │ - * @requires OpenLayers/Format/OWSContext.js │ │ │ │ │ - * @requires OpenLayers/Format/OWSCommon/v1_0_0.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.OWSContext.v0_3_1 │ │ │ │ │ - * Read and write OWSContext version 0.3.1. │ │ │ │ │ + * Class: OpenLayers.Format.SLD/v1_0_0_GeoServer │ │ │ │ │ + * Read and write SLD version 1.0.0 with GeoServer-specific enhanced options. │ │ │ │ │ + * See http://svn.osgeo.org/geotools/trunk/modules/extension/xsd/xsd-sld/src/main/resources/org/geotools/sld/bindings/StyledLayerDescriptor.xsd │ │ │ │ │ + * for more information. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * - <OpenLayers.Format.SLD.v1_0_0> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.OWSContext.v0_3_1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - owc: "http://www.opengis.net/ows-context", │ │ │ │ │ - gml: "http://www.opengis.net/gml", │ │ │ │ │ - kml: "http://www.opengis.net/kml/2.2", │ │ │ │ │ - ogc: "http://www.opengis.net/ogc", │ │ │ │ │ - ows: "http://www.opengis.net/ows", │ │ │ │ │ - sld: "http://www.opengis.net/sld", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: VERSION │ │ │ │ │ - * {String} 0.3.1 │ │ │ │ │ - */ │ │ │ │ │ - VERSION: "0.3.1", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} Schema location │ │ │ │ │ - */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/ows-context http://www.ogcnetwork.net/schemas/owc/0.3.1/owsContext.xsd", │ │ │ │ │ +OpenLayers.Format.SLD.v1_0_0_GeoServer = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.SLD.v1_0_0, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - * {String} Default namespace prefix to use. │ │ │ │ │ - */ │ │ │ │ │ - defaultPrefix: "owc", │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} The specific parser version. │ │ │ │ │ + */ │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: extractAttributes │ │ │ │ │ - * {Boolean} Extract attributes from GML. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - extractAttributes: true, │ │ │ │ │ + /** │ │ │ │ │ + * Property: profile │ │ │ │ │ + * {String} The specific profile │ │ │ │ │ + */ │ │ │ │ │ + profile: "GeoServer", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: xy │ │ │ │ │ - * {Boolean} Order of the GML coordinate true:(x,y) or false:(y,x) │ │ │ │ │ - * Changing is not recommended, a new Format should be instantiated. │ │ │ │ │ - */ │ │ │ │ │ - xy: true, │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.SLD.v1_0_0_GeoServer │ │ │ │ │ + * Create a new parser for GeoServer-enhanced SLD version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ - */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "sld": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "Priority": function(node, obj) { │ │ │ │ │ + var value = this.readers.ogc._expression.call(this, node); │ │ │ │ │ + if (value) { │ │ │ │ │ + obj.priority = value; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "VendorOption": function(node, obj) { │ │ │ │ │ + if (!obj.vendorOptions) { │ │ │ │ │ + obj.vendorOptions = {}; │ │ │ │ │ + } │ │ │ │ │ + obj.vendorOptions[node.getAttribute("name")] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "TextSymbolizer": function(node, rule) { │ │ │ │ │ + OpenLayers.Format.SLD.v1_0_0.prototype.readers.sld.TextSymbolizer.apply(this, arguments); │ │ │ │ │ + var symbolizer = this.multipleSymbolizers ? rule.symbolizers[rule.symbolizers.length - 1] : rule.symbolizer["Text"]; │ │ │ │ │ + if (symbolizer.graphic === undefined) { │ │ │ │ │ + symbolizer.graphic = false; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.SLD.v1_0_0.prototype.readers["sld"]) │ │ │ │ │ + }, OpenLayers.Format.SLD.v1_0_0.prototype.readers), │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: featureNS │ │ │ │ │ - * {String} The namespace uri to use for writing InlineGeometry │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ + */ │ │ │ │ │ + writers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "sld": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "Priority": function(priority) { │ │ │ │ │ + return this.writers.sld._OGCExpression.call( │ │ │ │ │ + this, "sld:Priority", priority │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + "VendorOption": function(option) { │ │ │ │ │ + return this.createElementNSPlus("sld:VendorOption", { │ │ │ │ │ + attributes: { │ │ │ │ │ + name: option.name │ │ │ │ │ + }, │ │ │ │ │ + value: option.value │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "TextSymbolizer": function(symbolizer) { │ │ │ │ │ + var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ + var node = writers["sld"]["TextSymbolizer"].apply(this, arguments); │ │ │ │ │ + if (symbolizer.graphic !== false && (symbolizer.externalGraphic || symbolizer.graphicName)) { │ │ │ │ │ + this.writeNode("Graphic", symbolizer, node); │ │ │ │ │ + } │ │ │ │ │ + if ("priority" in symbolizer) { │ │ │ │ │ + this.writeNode("Priority", symbolizer.priority, node); │ │ │ │ │ + } │ │ │ │ │ + return this.addVendorOptions(node, symbolizer); │ │ │ │ │ + }, │ │ │ │ │ + "PointSymbolizer": function(symbolizer) { │ │ │ │ │ + var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ + var node = writers["sld"]["PointSymbolizer"].apply(this, arguments); │ │ │ │ │ + return this.addVendorOptions(node, symbolizer); │ │ │ │ │ + }, │ │ │ │ │ + "LineSymbolizer": function(symbolizer) { │ │ │ │ │ + var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ + var node = writers["sld"]["LineSymbolizer"].apply(this, arguments); │ │ │ │ │ + return this.addVendorOptions(node, symbolizer); │ │ │ │ │ + }, │ │ │ │ │ + "PolygonSymbolizer": function(symbolizer) { │ │ │ │ │ + var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ + var node = writers["sld"]["PolygonSymbolizer"].apply(this, arguments); │ │ │ │ │ + return this.addVendorOptions(node, symbolizer); │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.SLD.v1_0_0.prototype.writers["sld"]) │ │ │ │ │ + }, OpenLayers.Format.SLD.v1_0_0.prototype.writers), │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: addVendorOptions │ │ │ │ │ + * Add in the VendorOption tags and return the node again. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A DOM node. │ │ │ │ │ + * symbolizer - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A DOM node. │ │ │ │ │ + */ │ │ │ │ │ + addVendorOptions: function(node, symbolizer) { │ │ │ │ │ + var options = symbolizer.vendorOptions; │ │ │ │ │ + if (options) { │ │ │ │ │ + for (var key in symbolizer.vendorOptions) { │ │ │ │ │ + this.writeNode("VendorOption", { │ │ │ │ │ + name: key, │ │ │ │ │ + value: symbolizer.vendorOptions[key] │ │ │ │ │ + }, node); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SLD.v1_0_0_GeoServer" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/ArcXML/Features.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/ArcXML.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.ArcXML.Features │ │ │ │ │ + * Read/Write ArcXML features. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Format.ArcXML.Features> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.ArcXML.Features = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.ArcXML.Features │ │ │ │ │ + * Create a new parser/writer for ArcXML Features. Create an instance of this class │ │ │ │ │ + * to get a set of features from an ArcXML response. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read data from a string of ArcXML, and return a set of OpenLayers features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} A collection of features. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + var axl = new OpenLayers.Format.ArcXML(); │ │ │ │ │ + var parsed = axl.read(data); │ │ │ │ │ + │ │ │ │ │ + return parsed.features.feature; │ │ │ │ │ + } │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WFSCapabilities/v1.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WFSCapabilities.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WFSCapabilities.v1 │ │ │ │ │ + * Abstract class not to be instantiated directly. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WFSCapabilities.v1 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + wfs: "http://www.opengis.net/wfs", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + ows: "http://www.opengis.net/ows" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: errorProperty │ │ │ │ │ + * {String} Which property of the returned object to check for in order to │ │ │ │ │ + * determine whether or not parsing has failed. In the case that the │ │ │ │ │ + * errorProperty is undefined on the returned object, the document will be │ │ │ │ │ + * run through an OGCExceptionReport parser. │ │ │ │ │ + */ │ │ │ │ │ + errorProperty: "featureTypeList", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: "wfs", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WFSCapabilities.v1_1 │ │ │ │ │ + * Create an instance of one of the subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} List of named layers. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + var raw = data; │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + return capabilities; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wfs": { │ │ │ │ │ + "WFS_Capabilities": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "FeatureTypeList": function(node, request) { │ │ │ │ │ + request.featureTypeList = { │ │ │ │ │ + featureTypes: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, request.featureTypeList); │ │ │ │ │ + }, │ │ │ │ │ + "FeatureType": function(node, featureTypeList) { │ │ │ │ │ + var featureType = {}; │ │ │ │ │ + this.readChildNodes(node, featureType); │ │ │ │ │ + featureTypeList.featureTypes.push(featureType); │ │ │ │ │ + }, │ │ │ │ │ + "Name": function(node, obj) { │ │ │ │ │ + var name = this.getChildValue(node); │ │ │ │ │ + if (name) { │ │ │ │ │ + var parts = name.split(":"); │ │ │ │ │ + obj.name = parts.pop(); │ │ │ │ │ + if (parts.length > 0) { │ │ │ │ │ + obj.featureNS = this.lookupNamespaceURI(node, parts[0]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Title": function(node, obj) { │ │ │ │ │ + var title = this.getChildValue(node); │ │ │ │ │ + if (title) { │ │ │ │ │ + obj.title = title; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Abstract": function(node, obj) { │ │ │ │ │ + var abst = this.getChildValue(node); │ │ │ │ │ + if (abst) { │ │ │ │ │ + obj["abstract"] = abst; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WFSCapabilities/v1_1_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WFSCapabilities/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/OWSCommon/v1.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WFSCapabilities/v1_1_0 │ │ │ │ │ + * Read WFS Capabilities version 1.1.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WFSCapabilities> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WFSCapabilities.v1_1_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WFSCapabilities.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ + */ │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WFSCapabilities.v1_1_0 │ │ │ │ │ + * Create a new parser for WFS capabilities version 1.1.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "DefaultSRS": function(node, obj) { │ │ │ │ │ + var defaultSRS = this.getChildValue(node); │ │ │ │ │ + if (defaultSRS) { │ │ │ │ │ + obj.srs = defaultSRS; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WFSCapabilities.v1.prototype.readers["wfs"]), │ │ │ │ │ + "ows": OpenLayers.Format.OWSCommon.v1.prototype.readers.ows │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1_1_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WFSCapabilities/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WFSCapabilities/v1.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WFSCapabilities/v1_0_0 │ │ │ │ │ + * Read WFS Capabilities version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WFSCapabilities.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WFSCapabilities.v1_0_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WFSCapabilities.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WFSCapabilities.v1_0_0 │ │ │ │ │ + * Create a new parser for WFS capabilities version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "Service": function(node, capabilities) { │ │ │ │ │ + capabilities.service = {}; │ │ │ │ │ + this.readChildNodes(node, capabilities.service); │ │ │ │ │ + }, │ │ │ │ │ + "Fees": function(node, service) { │ │ │ │ │ + var fees = this.getChildValue(node); │ │ │ │ │ + if (fees && fees.toLowerCase() != "none") { │ │ │ │ │ + service.fees = fees; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "AccessConstraints": function(node, service) { │ │ │ │ │ + var constraints = this.getChildValue(node); │ │ │ │ │ + if (constraints && constraints.toLowerCase() != "none") { │ │ │ │ │ + service.accessConstraints = constraints; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "OnlineResource": function(node, service) { │ │ │ │ │ + var onlineResource = this.getChildValue(node); │ │ │ │ │ + if (onlineResource && onlineResource.toLowerCase() != "none") { │ │ │ │ │ + service.onlineResource = onlineResource; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Keywords": function(node, service) { │ │ │ │ │ + var keywords = this.getChildValue(node); │ │ │ │ │ + if (keywords && keywords.toLowerCase() != "none") { │ │ │ │ │ + service.keywords = keywords.split(', '); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Capability": function(node, capabilities) { │ │ │ │ │ + capabilities.capability = {}; │ │ │ │ │ + this.readChildNodes(node, capabilities.capability); │ │ │ │ │ + }, │ │ │ │ │ + "Request": function(node, obj) { │ │ │ │ │ + obj.request = {}; │ │ │ │ │ + this.readChildNodes(node, obj.request); │ │ │ │ │ + }, │ │ │ │ │ + "GetFeature": function(node, request) { │ │ │ │ │ + request.getfeature = { │ │ │ │ │ + href: {}, // DCPType │ │ │ │ │ + formats: [] // ResultFormat │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, request.getfeature); │ │ │ │ │ + }, │ │ │ │ │ + "ResultFormat": function(node, obj) { │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + var childNode; │ │ │ │ │ + for (var i = 0; i < children.length; i++) { │ │ │ │ │ + childNode = children[i]; │ │ │ │ │ + if (childNode.nodeType == 1) { │ │ │ │ │ + obj.formats.push(childNode.nodeName); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "DCPType": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "HTTP": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj.href); │ │ │ │ │ + }, │ │ │ │ │ + "Get": function(node, obj) { │ │ │ │ │ + obj.get = node.getAttribute("onlineResource"); │ │ │ │ │ + }, │ │ │ │ │ + "Post": function(node, obj) { │ │ │ │ │ + obj.post = node.getAttribute("onlineResource"); │ │ │ │ │ + }, │ │ │ │ │ + "SRS": function(node, obj) { │ │ │ │ │ + var srs = this.getChildValue(node); │ │ │ │ │ + if (srs) { │ │ │ │ │ + obj.srs = srs; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WFSCapabilities.v1.prototype.readers["wfs"]) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1_0_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WMSCapabilities/v1.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WMSCapabilities.js │ │ │ │ │ + * @requires OpenLayers/Format/OGCExceptionReport.js │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WMSCapabilities.v1 │ │ │ │ │ + * Abstract class not to be instantiated directly. Creates │ │ │ │ │ + * the common parts for both WMS 1.1.X and WMS 1.3.X. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + wms: "http://www.opengis.net/wms", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: "wms", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMSCapabilities.v1 │ │ │ │ │ + * Create an instance of one of the subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} List of named layers. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + var raw = data; │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + if (capabilities.service === undefined) { │ │ │ │ │ + // an exception must have occurred, so parse it │ │ │ │ │ + var parser = new OpenLayers.Format.OGCExceptionReport(); │ │ │ │ │ + capabilities.error = parser.read(raw); │ │ │ │ │ + } │ │ │ │ │ + return capabilities; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wms": { │ │ │ │ │ + "Service": function(node, obj) { │ │ │ │ │ + obj.service = {}; │ │ │ │ │ + this.readChildNodes(node, obj.service); │ │ │ │ │ + }, │ │ │ │ │ + "Name": function(node, obj) { │ │ │ │ │ + obj.name = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Title": function(node, obj) { │ │ │ │ │ + obj.title = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Abstract": function(node, obj) { │ │ │ │ │ + obj["abstract"] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "BoundingBox": function(node, obj) { │ │ │ │ │ + var bbox = {}; │ │ │ │ │ + bbox.bbox = [ │ │ │ │ │ + parseFloat(node.getAttribute("minx")), │ │ │ │ │ + parseFloat(node.getAttribute("miny")), │ │ │ │ │ + parseFloat(node.getAttribute("maxx")), │ │ │ │ │ + parseFloat(node.getAttribute("maxy")) │ │ │ │ │ + ]; │ │ │ │ │ + var res = { │ │ │ │ │ + x: parseFloat(node.getAttribute("resx")), │ │ │ │ │ + y: parseFloat(node.getAttribute("resy")) │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + if (!(isNaN(res.x) && isNaN(res.y))) { │ │ │ │ │ + bbox.res = res; │ │ │ │ │ + } │ │ │ │ │ + // return the bbox so that descendant classes can set the │ │ │ │ │ + // CRS and SRS and add it to the obj │ │ │ │ │ + return bbox; │ │ │ │ │ + }, │ │ │ │ │ + "OnlineResource": function(node, obj) { │ │ │ │ │ + obj.href = this.getAttributeNS(node, this.namespaces.xlink, │ │ │ │ │ + "href"); │ │ │ │ │ + }, │ │ │ │ │ + "ContactInformation": function(node, obj) { │ │ │ │ │ + obj.contactInformation = {}; │ │ │ │ │ + this.readChildNodes(node, obj.contactInformation); │ │ │ │ │ + }, │ │ │ │ │ + "ContactPersonPrimary": function(node, obj) { │ │ │ │ │ + obj.personPrimary = {}; │ │ │ │ │ + this.readChildNodes(node, obj.personPrimary); │ │ │ │ │ + }, │ │ │ │ │ + "ContactPerson": function(node, obj) { │ │ │ │ │ + obj.person = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "ContactOrganization": function(node, obj) { │ │ │ │ │ + obj.organization = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "ContactPosition": function(node, obj) { │ │ │ │ │ + obj.position = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "ContactAddress": function(node, obj) { │ │ │ │ │ + obj.contactAddress = {}; │ │ │ │ │ + this.readChildNodes(node, obj.contactAddress); │ │ │ │ │ + }, │ │ │ │ │ + "AddressType": function(node, obj) { │ │ │ │ │ + obj.type = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Address": function(node, obj) { │ │ │ │ │ + obj.address = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "City": function(node, obj) { │ │ │ │ │ + obj.city = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "StateOrProvince": function(node, obj) { │ │ │ │ │ + obj.stateOrProvince = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "PostCode": function(node, obj) { │ │ │ │ │ + obj.postcode = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Country": function(node, obj) { │ │ │ │ │ + obj.country = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "ContactVoiceTelephone": function(node, obj) { │ │ │ │ │ + obj.phone = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "ContactFacsimileTelephone": function(node, obj) { │ │ │ │ │ + obj.fax = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "ContactElectronicMailAddress": function(node, obj) { │ │ │ │ │ + obj.email = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Fees": function(node, obj) { │ │ │ │ │ + var fees = this.getChildValue(node); │ │ │ │ │ + if (fees && fees.toLowerCase() != "none") { │ │ │ │ │ + obj.fees = fees; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "AccessConstraints": function(node, obj) { │ │ │ │ │ + var constraints = this.getChildValue(node); │ │ │ │ │ + if (constraints && constraints.toLowerCase() != "none") { │ │ │ │ │ + obj.accessConstraints = constraints; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Capability": function(node, obj) { │ │ │ │ │ + obj.capability = { │ │ │ │ │ + nestedLayers: [], │ │ │ │ │ + layers: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.capability); │ │ │ │ │ + }, │ │ │ │ │ + "Request": function(node, obj) { │ │ │ │ │ + obj.request = {}; │ │ │ │ │ + this.readChildNodes(node, obj.request); │ │ │ │ │ + }, │ │ │ │ │ + "GetCapabilities": function(node, obj) { │ │ │ │ │ + obj.getcapabilities = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getcapabilities); │ │ │ │ │ + }, │ │ │ │ │ + "Format": function(node, obj) { │ │ │ │ │ + if (OpenLayers.Util.isArray(obj.formats)) { │ │ │ │ │ + obj.formats.push(this.getChildValue(node)); │ │ │ │ │ + } else { │ │ │ │ │ + obj.format = this.getChildValue(node); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "DCPType": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "HTTP": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "Get": function(node, obj) { │ │ │ │ │ + obj.get = {}; │ │ │ │ │ + this.readChildNodes(node, obj.get); │ │ │ │ │ + // backwards compatibility │ │ │ │ │ + if (!obj.href) { │ │ │ │ │ + obj.href = obj.get.href; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Post": function(node, obj) { │ │ │ │ │ + obj.post = {}; │ │ │ │ │ + this.readChildNodes(node, obj.post); │ │ │ │ │ + // backwards compatibility │ │ │ │ │ + if (!obj.href) { │ │ │ │ │ + obj.href = obj.get.href; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "GetMap": function(node, obj) { │ │ │ │ │ + obj.getmap = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getmap); │ │ │ │ │ + }, │ │ │ │ │ + "GetFeatureInfo": function(node, obj) { │ │ │ │ │ + obj.getfeatureinfo = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getfeatureinfo); │ │ │ │ │ + }, │ │ │ │ │ + "Exception": function(node, obj) { │ │ │ │ │ + obj.exception = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.exception); │ │ │ │ │ + }, │ │ │ │ │ + "Layer": function(node, obj) { │ │ │ │ │ + var parentLayer, capability; │ │ │ │ │ + if (obj.capability) { │ │ │ │ │ + capability = obj.capability; │ │ │ │ │ + parentLayer = obj; │ │ │ │ │ + } else { │ │ │ │ │ + capability = obj; │ │ │ │ │ + } │ │ │ │ │ + var attrNode = node.getAttributeNode("queryable"); │ │ │ │ │ + var queryable = (attrNode && attrNode.specified) ? │ │ │ │ │ + node.getAttribute("queryable") : null; │ │ │ │ │ + attrNode = node.getAttributeNode("cascaded"); │ │ │ │ │ + var cascaded = (attrNode && attrNode.specified) ? │ │ │ │ │ + node.getAttribute("cascaded") : null; │ │ │ │ │ + attrNode = node.getAttributeNode("opaque"); │ │ │ │ │ + var opaque = (attrNode && attrNode.specified) ? │ │ │ │ │ + node.getAttribute('opaque') : null; │ │ │ │ │ + var noSubsets = node.getAttribute('noSubsets'); │ │ │ │ │ + var fixedWidth = node.getAttribute('fixedWidth'); │ │ │ │ │ + var fixedHeight = node.getAttribute('fixedHeight'); │ │ │ │ │ + var parent = parentLayer || {}, │ │ │ │ │ + extend = OpenLayers.Util.extend; │ │ │ │ │ + var layer = { │ │ │ │ │ + nestedLayers: [], │ │ │ │ │ + styles: parentLayer ? [].concat(parentLayer.styles) : [], │ │ │ │ │ + srs: parentLayer ? extend({}, parent.srs) : {}, │ │ │ │ │ + metadataURLs: [], │ │ │ │ │ + bbox: parentLayer ? extend({}, parent.bbox) : {}, │ │ │ │ │ + llbbox: parent.llbbox, │ │ │ │ │ + dimensions: parentLayer ? extend({}, parent.dimensions) : {}, │ │ │ │ │ + authorityURLs: parentLayer ? extend({}, parent.authorityURLs) : {}, │ │ │ │ │ + identifiers: {}, │ │ │ │ │ + keywords: [], │ │ │ │ │ + queryable: (queryable && queryable !== "") ? │ │ │ │ │ + (queryable === "1" || queryable === "true") : (parent.queryable || false), │ │ │ │ │ + cascaded: (cascaded !== null) ? parseInt(cascaded) : (parent.cascaded || 0), │ │ │ │ │ + opaque: opaque ? │ │ │ │ │ + (opaque === "1" || opaque === "true") : (parent.opaque || false), │ │ │ │ │ + noSubsets: (noSubsets !== null) ? │ │ │ │ │ + (noSubsets === "1" || noSubsets === "true") : (parent.noSubsets || false), │ │ │ │ │ + fixedWidth: (fixedWidth != null) ? │ │ │ │ │ + parseInt(fixedWidth) : (parent.fixedWidth || 0), │ │ │ │ │ + fixedHeight: (fixedHeight != null) ? │ │ │ │ │ + parseInt(fixedHeight) : (parent.fixedHeight || 0), │ │ │ │ │ + minScale: parent.minScale, │ │ │ │ │ + maxScale: parent.maxScale, │ │ │ │ │ + attribution: parent.attribution │ │ │ │ │ + }; │ │ │ │ │ + obj.nestedLayers.push(layer); │ │ │ │ │ + layer.capability = capability; │ │ │ │ │ + this.readChildNodes(node, layer); │ │ │ │ │ + delete layer.capability; │ │ │ │ │ + if (layer.name) { │ │ │ │ │ + var parts = layer.name.split(":"), │ │ │ │ │ + request = capability.request, │ │ │ │ │ + gfi = request.getfeatureinfo; │ │ │ │ │ + if (parts.length > 0) { │ │ │ │ │ + layer.prefix = parts[0]; │ │ │ │ │ + } │ │ │ │ │ + capability.layers.push(layer); │ │ │ │ │ + if (layer.formats === undefined) { │ │ │ │ │ + layer.formats = request.getmap.formats; │ │ │ │ │ + } │ │ │ │ │ + if (layer.infoFormats === undefined && gfi) { │ │ │ │ │ + layer.infoFormats = gfi.formats; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Attribution": function(node, obj) { │ │ │ │ │ + obj.attribution = {}; │ │ │ │ │ + this.readChildNodes(node, obj.attribution); │ │ │ │ │ + }, │ │ │ │ │ + "LogoURL": function(node, obj) { │ │ │ │ │ + obj.logo = { │ │ │ │ │ + width: node.getAttribute("width"), │ │ │ │ │ + height: node.getAttribute("height") │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.logo); │ │ │ │ │ + }, │ │ │ │ │ + "Style": function(node, obj) { │ │ │ │ │ + var style = {}; │ │ │ │ │ + obj.styles.push(style); │ │ │ │ │ + this.readChildNodes(node, style); │ │ │ │ │ + }, │ │ │ │ │ + "LegendURL": function(node, obj) { │ │ │ │ │ + var legend = { │ │ │ │ │ + width: node.getAttribute("width"), │ │ │ │ │ + height: node.getAttribute("height") │ │ │ │ │ + }; │ │ │ │ │ + obj.legend = legend; │ │ │ │ │ + this.readChildNodes(node, legend); │ │ │ │ │ + }, │ │ │ │ │ + "MetadataURL": function(node, obj) { │ │ │ │ │ + var metadataURL = { │ │ │ │ │ + type: node.getAttribute("type") │ │ │ │ │ + }; │ │ │ │ │ + obj.metadataURLs.push(metadataURL); │ │ │ │ │ + this.readChildNodes(node, metadataURL); │ │ │ │ │ + }, │ │ │ │ │ + "DataURL": function(node, obj) { │ │ │ │ │ + obj.dataURL = {}; │ │ │ │ │ + this.readChildNodes(node, obj.dataURL); │ │ │ │ │ + }, │ │ │ │ │ + "FeatureListURL": function(node, obj) { │ │ │ │ │ + obj.featureListURL = {}; │ │ │ │ │ + this.readChildNodes(node, obj.featureListURL); │ │ │ │ │ + }, │ │ │ │ │ + "AuthorityURL": function(node, obj) { │ │ │ │ │ + var name = node.getAttribute("name"); │ │ │ │ │ + var authority = {}; │ │ │ │ │ + this.readChildNodes(node, authority); │ │ │ │ │ + obj.authorityURLs[name] = authority.href; │ │ │ │ │ + }, │ │ │ │ │ + "Identifier": function(node, obj) { │ │ │ │ │ + var authority = node.getAttribute("authority"); │ │ │ │ │ + obj.identifiers[authority] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "KeywordList": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "SRS": function(node, obj) { │ │ │ │ │ + obj.srs[this.getChildValue(node)] = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WMSCapabilities/v1_1.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WMSCapabilities/v1.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WMSCapabilities.v1_1 │ │ │ │ │ + * Abstract class not to be instantiated directly. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WMSCapabilities.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_1 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WMSCapabilities.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "WMT_MS_Capabilities": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "Keyword": function(node, obj) { │ │ │ │ │ + if (obj.keywords) { │ │ │ │ │ + obj.keywords.push(this.getChildValue(node)); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "DescribeLayer": function(node, obj) { │ │ │ │ │ + obj.describelayer = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.describelayer); │ │ │ │ │ + }, │ │ │ │ │ + "GetLegendGraphic": function(node, obj) { │ │ │ │ │ + obj.getlegendgraphic = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getlegendgraphic); │ │ │ │ │ + }, │ │ │ │ │ + "GetStyles": function(node, obj) { │ │ │ │ │ + obj.getstyles = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getstyles); │ │ │ │ │ + }, │ │ │ │ │ + "PutStyles": function(node, obj) { │ │ │ │ │ + obj.putstyles = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.putstyles); │ │ │ │ │ + }, │ │ │ │ │ + "UserDefinedSymbolization": function(node, obj) { │ │ │ │ │ + var userSymbols = { │ │ │ │ │ + supportSLD: parseInt(node.getAttribute("SupportSLD")) == 1, │ │ │ │ │ + userLayer: parseInt(node.getAttribute("UserLayer")) == 1, │ │ │ │ │ + userStyle: parseInt(node.getAttribute("UserStyle")) == 1, │ │ │ │ │ + remoteWFS: parseInt(node.getAttribute("RemoteWFS")) == 1 │ │ │ │ │ + }; │ │ │ │ │ + obj.userSymbols = userSymbols; │ │ │ │ │ + }, │ │ │ │ │ + "LatLonBoundingBox": function(node, obj) { │ │ │ │ │ + obj.llbbox = [ │ │ │ │ │ + parseFloat(node.getAttribute("minx")), │ │ │ │ │ + parseFloat(node.getAttribute("miny")), │ │ │ │ │ + parseFloat(node.getAttribute("maxx")), │ │ │ │ │ + parseFloat(node.getAttribute("maxy")) │ │ │ │ │ + ]; │ │ │ │ │ + }, │ │ │ │ │ + "BoundingBox": function(node, obj) { │ │ │ │ │ + var bbox = OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"].BoundingBox.apply(this, [node, obj]); │ │ │ │ │ + bbox.srs = node.getAttribute("SRS"); │ │ │ │ │ + obj.bbox[bbox.srs] = bbox; │ │ │ │ │ + }, │ │ │ │ │ + "ScaleHint": function(node, obj) { │ │ │ │ │ + var min = node.getAttribute("min"); │ │ │ │ │ + var max = node.getAttribute("max"); │ │ │ │ │ + var rad2 = Math.pow(2, 0.5); │ │ │ │ │ + var ipm = OpenLayers.INCHES_PER_UNIT["m"]; │ │ │ │ │ + if (min != 0) { │ │ │ │ │ + obj.maxScale = parseFloat( │ │ │ │ │ + ((min / rad2) * ipm * │ │ │ │ │ + OpenLayers.DOTS_PER_INCH).toPrecision(13) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (max != Number.POSITIVE_INFINITY) { │ │ │ │ │ + obj.minScale = parseFloat( │ │ │ │ │ + ((max / rad2) * ipm * │ │ │ │ │ + OpenLayers.DOTS_PER_INCH).toPrecision(13) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Dimension": function(node, obj) { │ │ │ │ │ + var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ + var dim = { │ │ │ │ │ + name: name, │ │ │ │ │ + units: node.getAttribute("units"), │ │ │ │ │ + unitsymbol: node.getAttribute("unitSymbol") │ │ │ │ │ + }; │ │ │ │ │ + obj.dimensions[dim.name] = dim; │ │ │ │ │ + }, │ │ │ │ │ + "Extent": function(node, obj) { │ │ │ │ │ + var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ + if (name in obj["dimensions"]) { │ │ │ │ │ + var extent = obj.dimensions[name]; │ │ │ │ │ + extent.nearestVal = │ │ │ │ │ + node.getAttribute("nearestValue") === "1"; │ │ │ │ │ + extent.multipleVal = │ │ │ │ │ + node.getAttribute("multipleValues") === "1"; │ │ │ │ │ + extent.current = node.getAttribute("current") === "1"; │ │ │ │ │ + extent["default"] = node.getAttribute("default") || ""; │ │ │ │ │ + var values = this.getChildValue(node); │ │ │ │ │ + extent.values = values.split(","); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"]) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WMSCapabilities/v1_1_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WMSCapabilities/v1_1.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WMSCapabilities/v1_1_0 │ │ │ │ │ + * Read WMS Capabilities version 1.1.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WMSCapabilities.v1_1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_1_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WMSCapabilities.v1_1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} The specific parser version. │ │ │ │ │ + */ │ │ │ │ │ + version: "1.1.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMSCapabilities.v1_1_0 │ │ │ │ │ + * Create a new parser for WMS capabilities version 1.1.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "SRS": function(node, obj) { │ │ │ │ │ + var srs = this.getChildValue(node); │ │ │ │ │ + var values = srs.split(/ +/); │ │ │ │ │ + for (var i = 0, len = values.length; i < len; i++) { │ │ │ │ │ + obj.srs[values[i]] = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1_1.prototype.readers["wms"]) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WMSCapabilities/v1_1_1.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WMSCapabilities/v1_1.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WMSCapabilities/v1_1_1 │ │ │ │ │ + * Read WMS Capabilities version 1.1.1. │ │ │ │ │ + * │ │ │ │ │ + * Note on <ScaleHint> parsing: If the 'min' attribute is set to "0", no │ │ │ │ │ + * maxScale will be set on the layer object. If the 'max' attribute is set to │ │ │ │ │ + * "Infinity", no minScale will be set. This makes it easy to create proper │ │ │ │ │ + * {<OpenLayers.Layer.WMS>} configurations directly from the layer object │ │ │ │ │ + * literals returned by this format, because no minScale/maxScale modifications │ │ │ │ │ + * need to be made. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WMSCapabilities.v1_1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_1_1 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WMSCapabilities.v1_1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} The specific parser version. │ │ │ │ │ + */ │ │ │ │ │ + version: "1.1.1", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMSCapabilities.v1_1_1 │ │ │ │ │ + * Create a new parser for WMS capabilities version 1.1.1. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "SRS": function(node, obj) { │ │ │ │ │ + obj.srs[this.getChildValue(node)] = true; │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1_1.prototype.readers["wms"]) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_1" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WMSCapabilities/v1_1_1_WMSC.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WMSCapabilities/v1_1_1.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WMSCapabilities/v1_1_1_WMSC │ │ │ │ │ + * Read WMS-C Capabilities version 1.1.1. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WMSCapabilities.v1_1_1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_1_1_WMSC = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WMSCapabilities.v1_1_1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} The specific parser version. │ │ │ │ │ + */ │ │ │ │ │ + version: "1.1.1", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: profile │ │ │ │ │ + * {String} The specific profile │ │ │ │ │ + */ │ │ │ │ │ + profile: "WMSC", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMSCapabilities.v1_1_1 │ │ │ │ │ + * Create a new parser for WMS-C capabilities version 1.1.1. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "VendorSpecificCapabilities": function(node, obj) { │ │ │ │ │ + obj.vendorSpecific = { │ │ │ │ │ + tileSets: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.vendorSpecific); │ │ │ │ │ + }, │ │ │ │ │ + "TileSet": function(node, vendorSpecific) { │ │ │ │ │ + var tileset = { │ │ │ │ │ + srs: {}, │ │ │ │ │ + bbox: {}, │ │ │ │ │ + resolutions: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, tileset); │ │ │ │ │ + vendorSpecific.tileSets.push(tileset); │ │ │ │ │ + }, │ │ │ │ │ + "Resolutions": function(node, tileset) { │ │ │ │ │ + var res = this.getChildValue(node).split(" "); │ │ │ │ │ + for (var i = 0, len = res.length; i < len; i++) { │ │ │ │ │ + if (res[i] != "") { │ │ │ │ │ + tileset.resolutions.push(parseFloat(res[i])); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Width": function(node, tileset) { │ │ │ │ │ + tileset.width = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "Height": function(node, tileset) { │ │ │ │ │ + tileset.height = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "Layers": function(node, tileset) { │ │ │ │ │ + tileset.layers = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Styles": function(node, tileset) { │ │ │ │ │ + tileset.styles = this.getChildValue(node); │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1_1_1.prototype.readers["wms"]) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_1_WMSC" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WMSCapabilities/v1_3.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WMSCapabilities/v1.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WMSCapabilities/v1_3 │ │ │ │ │ + * Abstract base class for WMS Capabilities version 1.3.X. │ │ │ │ │ + * SLD 1.1.0 adds in the extra operations DescribeLayer and GetLegendGraphic, │ │ │ │ │ + * see: http://schemas.opengis.net/sld/1.1.0/sld_capabilities.xsd │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WMSCapabilities.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_3 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WMSCapabilities.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "WMS_Capabilities": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "LayerLimit": function(node, obj) { │ │ │ │ │ + obj.layerLimit = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "MaxWidth": function(node, obj) { │ │ │ │ │ + obj.maxWidth = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "MaxHeight": function(node, obj) { │ │ │ │ │ + obj.maxHeight = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "BoundingBox": function(node, obj) { │ │ │ │ │ + var bbox = OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"].BoundingBox.apply(this, [node, obj]); │ │ │ │ │ + bbox.srs = node.getAttribute("CRS"); │ │ │ │ │ + obj.bbox[bbox.srs] = bbox; │ │ │ │ │ + }, │ │ │ │ │ + "CRS": function(node, obj) { │ │ │ │ │ + // CRS is the synonym of SRS │ │ │ │ │ + this.readers.wms.SRS.apply(this, [node, obj]); │ │ │ │ │ + }, │ │ │ │ │ + "EX_GeographicBoundingBox": function(node, obj) { │ │ │ │ │ + // replacement of LatLonBoundingBox │ │ │ │ │ + obj.llbbox = []; │ │ │ │ │ + this.readChildNodes(node, obj.llbbox); │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + "westBoundLongitude": function(node, obj) { │ │ │ │ │ + obj[0] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "eastBoundLongitude": function(node, obj) { │ │ │ │ │ + obj[2] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "southBoundLatitude": function(node, obj) { │ │ │ │ │ + obj[1] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "northBoundLatitude": function(node, obj) { │ │ │ │ │ + obj[3] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "MinScaleDenominator": function(node, obj) { │ │ │ │ │ + obj.maxScale = parseFloat(this.getChildValue(node)).toPrecision(16); │ │ │ │ │ + }, │ │ │ │ │ + "MaxScaleDenominator": function(node, obj) { │ │ │ │ │ + obj.minScale = parseFloat(this.getChildValue(node)).toPrecision(16); │ │ │ │ │ + }, │ │ │ │ │ + "Dimension": function(node, obj) { │ │ │ │ │ + // dimension has extra attributes: default, multipleValues, │ │ │ │ │ + // nearestValue, current which used to be part of Extent. It now │ │ │ │ │ + // also contains the values. │ │ │ │ │ + var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ + var dim = { │ │ │ │ │ + name: name, │ │ │ │ │ + units: node.getAttribute("units"), │ │ │ │ │ + unitsymbol: node.getAttribute("unitSymbol"), │ │ │ │ │ + nearestVal: node.getAttribute("nearestValue") === "1", │ │ │ │ │ + multipleVal: node.getAttribute("multipleValues") === "1", │ │ │ │ │ + "default": node.getAttribute("default") || "", │ │ │ │ │ + current: node.getAttribute("current") === "1", │ │ │ │ │ + values: this.getChildValue(node).split(",") │ │ │ │ │ + │ │ │ │ │ + }; │ │ │ │ │ + // Theoretically there can be more dimensions with the same │ │ │ │ │ + // name, but with a different unit. Until we meet such a case, │ │ │ │ │ + // let's just keep the same structure as the WMS 1.1 │ │ │ │ │ + // GetCapabilities parser uses. We will store the last │ │ │ │ │ + // one encountered. │ │ │ │ │ + obj.dimensions[dim.name] = dim; │ │ │ │ │ + }, │ │ │ │ │ + "Keyword": function(node, obj) { │ │ │ │ │ + // TODO: should we change the structure of keyword in v1.js? │ │ │ │ │ + // Make it an object with a value instead of a string? │ │ │ │ │ + var keyword = { │ │ │ │ │ + value: this.getChildValue(node), │ │ │ │ │ + vocabulary: node.getAttribute("vocabulary") │ │ │ │ │ + }; │ │ │ │ │ + if (obj.keywords) { │ │ │ │ │ + obj.keywords.push(keyword); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"]), │ │ │ │ │ + "sld": { │ │ │ │ │ + "UserDefinedSymbolization": function(node, obj) { │ │ │ │ │ + this.readers.wms.UserDefinedSymbolization.apply(this, [node, obj]); │ │ │ │ │ + // add the two extra attributes │ │ │ │ │ + obj.userSymbols.inlineFeature = parseInt(node.getAttribute("InlineFeature")) == 1; │ │ │ │ │ + obj.userSymbols.remoteWCS = parseInt(node.getAttribute("RemoteWCS")) == 1; │ │ │ │ │ + }, │ │ │ │ │ + "DescribeLayer": function(node, obj) { │ │ │ │ │ + this.readers.wms.DescribeLayer.apply(this, [node, obj]); │ │ │ │ │ + }, │ │ │ │ │ + "GetLegendGraphic": function(node, obj) { │ │ │ │ │ + this.readers.wms.GetLegendGraphic.apply(this, [node, obj]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_3" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WMSCapabilities/v1_3_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WMSCapabilities/v1_3.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WMSCapabilities/v1_3_0 │ │ │ │ │ + * Read WMS Capabilities version 1.3.0. │ │ │ │ │ + * SLD 1.1.0 adds in the extra operations DescribeLayer and GetLegendGraphic, │ │ │ │ │ + * see: http://schemas.opengis.net/sld/1.1.0/sld_capabilities.xsd │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WMSCapabilities.v1_3> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_3_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WMSCapabilities.v1_3, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} The specific parser version. │ │ │ │ │ + */ │ │ │ │ │ + version: "1.3.0", │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_3_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WMTSCapabilities/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WMTSCapabilities.js │ │ │ │ │ + * @requires OpenLayers/Format/OWSCommon/v1_1_0.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WMTSCapabilities.v1_0_0 │ │ │ │ │ + * Read WMTS Capabilities version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WMTSCapabilities> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WMTSCapabilities.v1_0_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.OWSCommon.v1_1_0, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} The parser version ("1.0.0"). │ │ │ │ │ + */ │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ + wmts: "http://www.opengis.net/wmts/1.0", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: yx │ │ │ │ │ + * {Object} Members in the yx object are used to determine if a CRS URN │ │ │ │ │ + * corresponds to a CRS with y,x axis order. Member names are CRS URNs │ │ │ │ │ + * and values are boolean. Defaults come from the │ │ │ │ │ + * <OpenLayers.Format.WMTSCapabilities> prototype. │ │ │ │ │ + */ │ │ │ │ │ + yx: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + * {String} The default namespace alias for creating element nodes. │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: "wmts", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMTSCapabilities.v1_0_0 │ │ │ │ │ + * Create a new parser for WMTS capabilities version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.options = options; │ │ │ │ │ + var yx = OpenLayers.Util.extend({}, OpenLayers.Format.WMTSCapabilities.prototype.yx); │ │ │ │ │ + this.yx = OpenLayers.Util.extend(yx, this.yx); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return info about the WMTS. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Information about the SOS service. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + capabilities.version = this.version; │ │ │ │ │ + return capabilities; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wmts": { │ │ │ │ │ + "Capabilities": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "Contents": function(node, obj) { │ │ │ │ │ + obj.contents = {}; │ │ │ │ │ + obj.contents.layers = []; │ │ │ │ │ + obj.contents.tileMatrixSets = {}; │ │ │ │ │ + this.readChildNodes(node, obj.contents); │ │ │ │ │ + }, │ │ │ │ │ + "Layer": function(node, obj) { │ │ │ │ │ + var layer = { │ │ │ │ │ + styles: [], │ │ │ │ │ + formats: [], │ │ │ │ │ + dimensions: [], │ │ │ │ │ + tileMatrixSetLinks: [] │ │ │ │ │ + }; │ │ │ │ │ + layer.layers = []; │ │ │ │ │ + this.readChildNodes(node, layer); │ │ │ │ │ + obj.layers.push(layer); │ │ │ │ │ + }, │ │ │ │ │ + "Style": function(node, obj) { │ │ │ │ │ + var style = {}; │ │ │ │ │ + style.isDefault = (node.getAttribute("isDefault") === "true"); │ │ │ │ │ + this.readChildNodes(node, style); │ │ │ │ │ + obj.styles.push(style); │ │ │ │ │ + }, │ │ │ │ │ + "Format": function(node, obj) { │ │ │ │ │ + obj.formats.push(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "TileMatrixSetLink": function(node, obj) { │ │ │ │ │ + var tileMatrixSetLink = {}; │ │ │ │ │ + this.readChildNodes(node, tileMatrixSetLink); │ │ │ │ │ + obj.tileMatrixSetLinks.push(tileMatrixSetLink); │ │ │ │ │ + }, │ │ │ │ │ + "TileMatrixSet": function(node, obj) { │ │ │ │ │ + // node could be child of wmts:Contents or wmts:TileMatrixSetLink │ │ │ │ │ + // duck type wmts:Contents by looking for layers │ │ │ │ │ + if (obj.layers) { │ │ │ │ │ + // TileMatrixSet as object type in schema │ │ │ │ │ + var tileMatrixSet = { │ │ │ │ │ + matrixIds: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, tileMatrixSet); │ │ │ │ │ + obj.tileMatrixSets[tileMatrixSet.identifier] = tileMatrixSet; │ │ │ │ │ + } else { │ │ │ │ │ + // TileMatrixSet as string type in schema │ │ │ │ │ + obj.tileMatrixSet = this.getChildValue(node); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "TileMatrix": function(node, obj) { │ │ │ │ │ + var tileMatrix = { │ │ │ │ │ + supportedCRS: obj.supportedCRS │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, tileMatrix); │ │ │ │ │ + obj.matrixIds.push(tileMatrix); │ │ │ │ │ + }, │ │ │ │ │ + "ScaleDenominator": function(node, obj) { │ │ │ │ │ + obj.scaleDenominator = parseFloat(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "TopLeftCorner": function(node, obj) { │ │ │ │ │ + var topLeftCorner = this.getChildValue(node); │ │ │ │ │ + var coords = topLeftCorner.split(" "); │ │ │ │ │ + // decide on axis order for the given CRS │ │ │ │ │ + var yx; │ │ │ │ │ + if (obj.supportedCRS) { │ │ │ │ │ + // extract out version from URN │ │ │ │ │ + var crs = obj.supportedCRS.replace( │ │ │ │ │ + /urn:ogc:def:crs:(\w+):.+:(\w+)$/, │ │ │ │ │ + "urn:ogc:def:crs:$1::$2" │ │ │ │ │ + ); │ │ │ │ │ + yx = !!this.yx[crs]; │ │ │ │ │ + } │ │ │ │ │ + if (yx) { │ │ │ │ │ + obj.topLeftCorner = new OpenLayers.LonLat( │ │ │ │ │ + coords[1], coords[0] │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + obj.topLeftCorner = new OpenLayers.LonLat( │ │ │ │ │ + coords[0], coords[1] │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "TileWidth": function(node, obj) { │ │ │ │ │ + obj.tileWidth = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "TileHeight": function(node, obj) { │ │ │ │ │ + obj.tileHeight = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "MatrixWidth": function(node, obj) { │ │ │ │ │ + obj.matrixWidth = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "MatrixHeight": function(node, obj) { │ │ │ │ │ + obj.matrixHeight = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "ResourceURL": function(node, obj) { │ │ │ │ │ + obj.resourceUrl = obj.resourceUrl || {}; │ │ │ │ │ + var resourceType = node.getAttribute("resourceType"); │ │ │ │ │ + if (!obj.resourceUrls) { │ │ │ │ │ + obj.resourceUrls = []; │ │ │ │ │ + } │ │ │ │ │ + var resourceUrl = obj.resourceUrl[resourceType] = { │ │ │ │ │ + format: node.getAttribute("format"), │ │ │ │ │ + template: node.getAttribute("template"), │ │ │ │ │ + resourceType: resourceType │ │ │ │ │ + }; │ │ │ │ │ + obj.resourceUrls.push(resourceUrl); │ │ │ │ │ + }, │ │ │ │ │ + // not used for now, can be added in the future though │ │ │ │ │ + /*"Themes": function(node, obj) { │ │ │ │ │ + obj.themes = []; │ │ │ │ │ + this.readChildNodes(node, obj.themes); │ │ │ │ │ + }, │ │ │ │ │ + "Theme": function(node, obj) { │ │ │ │ │ + var theme = {}; │ │ │ │ │ + this.readChildNodes(node, theme); │ │ │ │ │ + obj.push(theme); │ │ │ │ │ + },*/ │ │ │ │ │ + "WSDL": function(node, obj) { │ │ │ │ │ + obj.wsdl = {}; │ │ │ │ │ + obj.wsdl.href = node.getAttribute("xlink:href"); │ │ │ │ │ + // TODO: other attributes of <WSDL> element │ │ │ │ │ + }, │ │ │ │ │ + "ServiceMetadataURL": function(node, obj) { │ │ │ │ │ + obj.serviceMetadataUrl = {}; │ │ │ │ │ + obj.serviceMetadataUrl.href = node.getAttribute("xlink:href"); │ │ │ │ │ + // TODO: other attributes of <ServiceMetadataURL> element │ │ │ │ │ + }, │ │ │ │ │ + "LegendURL": function(node, obj) { │ │ │ │ │ + obj.legend = {}; │ │ │ │ │ + obj.legend.href = node.getAttribute("xlink:href"); │ │ │ │ │ + obj.legend.format = node.getAttribute("format"); │ │ │ │ │ + }, │ │ │ │ │ + "Dimension": function(node, obj) { │ │ │ │ │ + var dimension = { │ │ │ │ │ + values: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, dimension); │ │ │ │ │ + obj.dimensions.push(dimension); │ │ │ │ │ + }, │ │ │ │ │ + "Default": function(node, obj) { │ │ │ │ │ + obj["default"] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Value": function(node, obj) { │ │ │ │ │ + obj.values.push(this.getChildValue(node)); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "ows": OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMTSCapabilities.v1_0_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WPSCapabilities/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WPSCapabilities.js │ │ │ │ │ + * @requires OpenLayers/Format/OWSCommon/v1_1_0.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WPSCapabilities.v1_0_0 │ │ │ │ │ + * Read WPS Capabilities version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WPSCapabilities.v1_0_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ + wps: "http://www.opengis.net/wps/1.0.0", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ + */ │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WPSCapabilities.v1_0_0 │ │ │ │ │ + * Create a new parser for WPS capabilities version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return info about the WPS. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Information about the WPS service. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + return capabilities; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wps": { │ │ │ │ │ + "Capabilities": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "ProcessOfferings": function(node, obj) { │ │ │ │ │ + obj.processOfferings = {}; │ │ │ │ │ + this.readChildNodes(node, obj.processOfferings); │ │ │ │ │ + }, │ │ │ │ │ + "Process": function(node, processOfferings) { │ │ │ │ │ + var processVersion = this.getAttributeNS(node, this.namespaces.wps, "processVersion"); │ │ │ │ │ + var process = { │ │ │ │ │ + processVersion: processVersion │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, process); │ │ │ │ │ + processOfferings[process.identifier] = process; │ │ │ │ │ + }, │ │ │ │ │ + "Languages": function(node, obj) { │ │ │ │ │ + obj.languages = []; │ │ │ │ │ + this.readChildNodes(node, obj.languages); │ │ │ │ │ + }, │ │ │ │ │ + "Default": function(node, languages) { │ │ │ │ │ + var language = { │ │ │ │ │ + isDefault: true │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, language); │ │ │ │ │ + languages.push(language); │ │ │ │ │ + }, │ │ │ │ │ + "Supported": function(node, languages) { │ │ │ │ │ + var language = {}; │ │ │ │ │ + this.readChildNodes(node, language); │ │ │ │ │ + languages.push(language); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "ows": OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WPSCapabilities.v1_0_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/OWSContext/v0_3_1.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Format/KML.js │ │ │ │ │ + * @requires OpenLayers/Format/GML.js │ │ │ │ │ + * @requires OpenLayers/Format/GML/v2.js │ │ │ │ │ + * @requires OpenLayers/Format/SLD/v1_0_0.js │ │ │ │ │ + * @requires OpenLayers/Format/OWSContext.js │ │ │ │ │ + * @requires OpenLayers/Format/OWSCommon/v1_0_0.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.OWSContext.v0_3_1 │ │ │ │ │ + * Read and write OWSContext version 0.3.1. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.OWSContext.v0_3_1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + owc: "http://www.opengis.net/ows-context", │ │ │ │ │ + gml: "http://www.opengis.net/gml", │ │ │ │ │ + kml: "http://www.opengis.net/kml/2.2", │ │ │ │ │ + ogc: "http://www.opengis.net/ogc", │ │ │ │ │ + ows: "http://www.opengis.net/ows", │ │ │ │ │ + sld: "http://www.opengis.net/sld", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: VERSION │ │ │ │ │ + * {String} 0.3.1 │ │ │ │ │ + */ │ │ │ │ │ + VERSION: "0.3.1", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} Schema location │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: "http://www.opengis.net/ows-context http://www.ogcnetwork.net/schemas/owc/0.3.1/owsContext.xsd", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + * {String} Default namespace prefix to use. │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: "owc", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: extractAttributes │ │ │ │ │ + * {Boolean} Extract attributes from GML. Default is true. │ │ │ │ │ + */ │ │ │ │ │ + extractAttributes: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: xy │ │ │ │ │ + * {Boolean} Order of the GML coordinate true:(x,y) or false:(y,x) │ │ │ │ │ + * Changing is not recommended, a new Format should be instantiated. │ │ │ │ │ + */ │ │ │ │ │ + xy: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ + */ │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: featureNS │ │ │ │ │ + * {String} The namespace uri to use for writing InlineGeometry │ │ │ │ │ + */ │ │ │ │ │ featureNS: "http://mapserver.gis.umn.edu/mapserver", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Property: featureType │ │ │ │ │ * {String} The name to use as the feature type when writing out │ │ │ │ │ * InlineGeometry │ │ │ │ │ */ │ │ │ │ │ @@ -54761,376 +57146,143 @@ │ │ │ │ │ "feature": OpenLayers.Format.GML.v2.prototype.writers.feature │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.OWSContext.v0_3_1" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/XLS/v1.js │ │ │ │ │ + OpenLayers/Format/WMSDescribeLayer/v1_1.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XLS.js │ │ │ │ │ - * @requires OpenLayers/Format/GML/v3.js │ │ │ │ │ + * @requires OpenLayers/Format/WMSDescribeLayer.js │ │ │ │ │ + * @requires OpenLayers/Format/OGCExceptionReport.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.XLS.v1 │ │ │ │ │ - * Superclass for XLS version 1 parsers. Only supports GeocodeRequest for now. │ │ │ │ │ + * Class: OpenLayers.Format.WMSDescribeLayer.v1_1_1 │ │ │ │ │ + * Read SLD WMS DescribeLayer response for WMS 1.1.X │ │ │ │ │ + * WMS 1.1.X is tightly coupled to SLD 1.0.0 │ │ │ │ │ + * │ │ │ │ │ + * Example DescribeLayer request: │ │ │ │ │ + * http://demo.opengeo.org/geoserver/wms?request=DescribeLayer&version=1.1.1&layers=topp:states │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * - <OpenLayers.Format.WMSDescribeLayer> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.XLS.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - xls: "http://www.opengis.net/xls", │ │ │ │ │ - gml: "http://www.opengis.net/gml", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ - */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: xy │ │ │ │ │ - * {Boolean} Order of the GML coordinate true:(x,y) or false:(y,x) │ │ │ │ │ - * Changing is not recommended, a new Format should be instantiated. │ │ │ │ │ - */ │ │ │ │ │ - xy: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - */ │ │ │ │ │ - defaultPrefix: "xls", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} Schema location for a particular minor version. │ │ │ │ │ - */ │ │ │ │ │ - schemaLocation: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.XLS.v1 │ │ │ │ │ - * Instances of this class are not created directly. Use the │ │ │ │ │ - * <OpenLayers.Format.XLS> constructor instead. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: read │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {DOMElement} An XLS document element. │ │ │ │ │ - * options - {Object} Options for the reader. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object representing the XLSResponse. │ │ │ │ │ - */ │ │ │ │ │ - read: function(data, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var xls = {}; │ │ │ │ │ - this.readChildNodes(data, xls); │ │ │ │ │ - return xls; │ │ │ │ │ - }, │ │ │ │ │ +OpenLayers.Format.WMSDescribeLayer.v1_1_1 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WMSDescribeLayer, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "xls": { │ │ │ │ │ - "XLS": function(node, xls) { │ │ │ │ │ - xls.version = node.getAttribute("version"); │ │ │ │ │ - this.readChildNodes(node, xls); │ │ │ │ │ - }, │ │ │ │ │ - "Response": function(node, xls) { │ │ │ │ │ - this.readChildNodes(node, xls); │ │ │ │ │ - }, │ │ │ │ │ - "GeocodeResponse": function(node, xls) { │ │ │ │ │ - xls.responseLists = []; │ │ │ │ │ - this.readChildNodes(node, xls); │ │ │ │ │ - }, │ │ │ │ │ - "GeocodeResponseList": function(node, xls) { │ │ │ │ │ - var responseList = { │ │ │ │ │ - features: [], │ │ │ │ │ - numberOfGeocodedAddresses: parseInt(node.getAttribute("numberOfGeocodedAddresses")) │ │ │ │ │ - }; │ │ │ │ │ - xls.responseLists.push(responseList); │ │ │ │ │ - this.readChildNodes(node, responseList); │ │ │ │ │ - }, │ │ │ │ │ - "GeocodedAddress": function(node, responseList) { │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ - responseList.features.push(feature); │ │ │ │ │ - this.readChildNodes(node, feature); │ │ │ │ │ - // post-process geometry │ │ │ │ │ - feature.geometry = feature.components[0]; │ │ │ │ │ - }, │ │ │ │ │ - "GeocodeMatchCode": function(node, feature) { │ │ │ │ │ - feature.attributes.matchCode = { │ │ │ │ │ - accuracy: parseFloat(node.getAttribute("accuracy")), │ │ │ │ │ - matchType: node.getAttribute("matchType") │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ - "Address": function(node, feature) { │ │ │ │ │ - var address = { │ │ │ │ │ - countryCode: node.getAttribute("countryCode"), │ │ │ │ │ - addressee: node.getAttribute("addressee"), │ │ │ │ │ - street: [], │ │ │ │ │ - place: [] │ │ │ │ │ - }; │ │ │ │ │ - feature.attributes.address = address; │ │ │ │ │ - this.readChildNodes(node, address); │ │ │ │ │ - }, │ │ │ │ │ - "freeFormAddress": function(node, address) { │ │ │ │ │ - address.freeFormAddress = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "StreetAddress": function(node, address) { │ │ │ │ │ - this.readChildNodes(node, address); │ │ │ │ │ - }, │ │ │ │ │ - "Building": function(node, address) { │ │ │ │ │ - address.building = { │ │ │ │ │ - 'number': node.getAttribute("number"), │ │ │ │ │ - subdivision: node.getAttribute("subdivision"), │ │ │ │ │ - buildingName: node.getAttribute("buildingName") │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ - "Street": function(node, address) { │ │ │ │ │ - // only support the built-in primitive type for now │ │ │ │ │ - address.street.push(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "Place": function(node, address) { │ │ │ │ │ - // type is one of CountrySubdivision, │ │ │ │ │ - // CountrySecondarySubdivision, Municipality or │ │ │ │ │ - // MunicipalitySubdivision │ │ │ │ │ - address.place[node.getAttribute("type")] = │ │ │ │ │ - this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "PostalCode": function(node, address) { │ │ │ │ │ - address.postalCode = this.getChildValue(node); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMSDescribeLayer │ │ │ │ │ + * Create a new parser for WMS DescribeLayer responses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.WMSDescribeLayer.prototype.initialize.apply(this, │ │ │ │ │ + [options]); │ │ │ │ │ }, │ │ │ │ │ - "gml": OpenLayers.Format.GML.v3.prototype.readers.gml │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: write │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * request - {Object} An object representing the geocode request. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The root of an XLS document. │ │ │ │ │ - */ │ │ │ │ │ - write: function(request) { │ │ │ │ │ - return this.writers.xls.XLS.apply(this, [request]); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ - */ │ │ │ │ │ - writers: { │ │ │ │ │ - "xls": { │ │ │ │ │ - "XLS": function(request) { │ │ │ │ │ - var root = this.createElementNSPlus( │ │ │ │ │ - "xls:XLS", { │ │ │ │ │ - attributes: { │ │ │ │ │ - "version": this.VERSION, │ │ │ │ │ - "xsi:schemaLocation": this.schemaLocation │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read DescribeLayer data from a string, and return the response. │ │ │ │ │ + * The OGC defines 2 formats which are allowed for output, │ │ │ │ │ + * so we need to parse these 2 types for version 1.1.X │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Object with a layerDescriptions property, which holds an Array │ │ │ │ │ + * of {<LayerDescription>} objects which have: │ │ │ │ │ + * - {String} owsType: WFS/WCS │ │ │ │ │ + * - {String} owsURL: the online resource │ │ │ │ │ + * - {String} typeName: the name of the typename on the owsType service │ │ │ │ │ + * - {String} layerName: the name of the WMS layer we did a lookup for │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + var root = data.documentElement; │ │ │ │ │ + var children = root.childNodes; │ │ │ │ │ + var describelayer = { │ │ │ │ │ + layerDescriptions: [] │ │ │ │ │ + }; │ │ │ │ │ + var childNode, nodeName; │ │ │ │ │ + for (var i = 0; i < children.length; ++i) { │ │ │ │ │ + childNode = children[i]; │ │ │ │ │ + nodeName = childNode.nodeName; │ │ │ │ │ + if (nodeName == 'LayerDescription') { │ │ │ │ │ + var layerName = childNode.getAttribute('name'); │ │ │ │ │ + var owsType = ''; │ │ │ │ │ + var owsURL = ''; │ │ │ │ │ + var typeName = ''; │ │ │ │ │ + // check for owsType and owsURL attributes │ │ │ │ │ + if (childNode.getAttribute('owsType')) { │ │ │ │ │ + owsType = childNode.getAttribute('owsType'); │ │ │ │ │ + owsURL = childNode.getAttribute('owsURL'); │ │ │ │ │ + } else { │ │ │ │ │ + // look for wfs or wcs attribute │ │ │ │ │ + if (childNode.getAttribute('wfs') != '') { │ │ │ │ │ + owsType = 'WFS'; │ │ │ │ │ + owsURL = childNode.getAttribute('wfs'); │ │ │ │ │ + } else if (childNode.getAttribute('wcs') != '') { │ │ │ │ │ + owsType = 'WCS'; │ │ │ │ │ + owsURL = childNode.getAttribute('wcs'); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - ); │ │ │ │ │ - this.writeNode("RequestHeader", request.header, root); │ │ │ │ │ - this.writeNode("Request", request, root); │ │ │ │ │ - return root; │ │ │ │ │ - }, │ │ │ │ │ - "RequestHeader": function(header) { │ │ │ │ │ - return this.createElementNSPlus("xls:RequestHeader"); │ │ │ │ │ - }, │ │ │ │ │ - "Request": function(request) { │ │ │ │ │ - var node = this.createElementNSPlus("xls:Request", { │ │ │ │ │ - attributes: { │ │ │ │ │ - methodName: "GeocodeRequest", │ │ │ │ │ - requestID: request.requestID || "", │ │ │ │ │ - version: this.VERSION │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - this.writeNode("GeocodeRequest", request.addresses, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "GeocodeRequest": function(addresses) { │ │ │ │ │ - var node = this.createElementNSPlus("xls:GeocodeRequest"); │ │ │ │ │ - for (var i = 0, len = addresses.length; i < len; i++) { │ │ │ │ │ - this.writeNode("Address", addresses[i], node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Address": function(address) { │ │ │ │ │ - var node = this.createElementNSPlus("xls:Address", { │ │ │ │ │ - attributes: { │ │ │ │ │ - countryCode: address.countryCode │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (address.freeFormAddress) { │ │ │ │ │ - this.writeNode("freeFormAddress", address.freeFormAddress, node); │ │ │ │ │ - } else { │ │ │ │ │ - if (address.street) { │ │ │ │ │ - this.writeNode("StreetAddress", address, node); │ │ │ │ │ - } │ │ │ │ │ - if (address.municipality) { │ │ │ │ │ - this.writeNode("Municipality", address.municipality, node); │ │ │ │ │ - } │ │ │ │ │ - if (address.countrySubdivision) { │ │ │ │ │ - this.writeNode("CountrySubdivision", address.countrySubdivision, node); │ │ │ │ │ - } │ │ │ │ │ - if (address.postalCode) { │ │ │ │ │ - this.writeNode("PostalCode", address.postalCode, node); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "freeFormAddress": function(freeFormAddress) { │ │ │ │ │ - return this.createElementNSPlus("freeFormAddress", { │ │ │ │ │ - value: freeFormAddress │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "StreetAddress": function(address) { │ │ │ │ │ - var node = this.createElementNSPlus("xls:StreetAddress"); │ │ │ │ │ - if (address.building) { │ │ │ │ │ - this.writeNode(node, "Building", address.building); │ │ │ │ │ - } │ │ │ │ │ - var street = address.street; │ │ │ │ │ - if (!(OpenLayers.Util.isArray(street))) { │ │ │ │ │ - street = [street]; │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = street.length; i < len; i++) { │ │ │ │ │ - this.writeNode("Street", street[i], node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Building": function(building) { │ │ │ │ │ - return this.createElementNSPlus("xls:Building", { │ │ │ │ │ - attributes: { │ │ │ │ │ - "number": building["number"], │ │ │ │ │ - "subdivision": building.subdivision, │ │ │ │ │ - "buildingName": building.buildingName │ │ │ │ │ + // look for Query child │ │ │ │ │ + var query = childNode.getElementsByTagName('Query'); │ │ │ │ │ + if (query.length > 0) { │ │ │ │ │ + typeName = query[0].getAttribute('typeName'); │ │ │ │ │ + if (!typeName) { │ │ │ │ │ + // because of Ionic bug │ │ │ │ │ + typeName = query[0].getAttribute('typename'); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "Street": function(street) { │ │ │ │ │ - return this.createElementNSPlus("xls:Street", { │ │ │ │ │ - value: street │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "Municipality": function(municipality) { │ │ │ │ │ - return this.createElementNSPlus("xls:Place", { │ │ │ │ │ - attributes: { │ │ │ │ │ - type: "Municipality" │ │ │ │ │ - }, │ │ │ │ │ - value: municipality │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "CountrySubdivision": function(countrySubdivision) { │ │ │ │ │ - return this.createElementNSPlus("xls:Place", { │ │ │ │ │ - attributes: { │ │ │ │ │ - type: "CountrySubdivision" │ │ │ │ │ - }, │ │ │ │ │ - value: countrySubdivision │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "PostalCode": function(postalCode) { │ │ │ │ │ - return this.createElementNSPlus("xls:PostalCode", { │ │ │ │ │ - value: postalCode │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.XLS.v1" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/XLS/v1_1_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XLS/v1.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.XLS.v1_1_0 │ │ │ │ │ - * Read / write XLS version 1.1.0. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XLS.v1> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.XLS.v1_1_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.XLS.v1, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: VERSION │ │ │ │ │ - * {String} 1.1 │ │ │ │ │ - */ │ │ │ │ │ - VERSION: "1.1", │ │ │ │ │ + var layerDescription = { │ │ │ │ │ + layerName: layerName, │ │ │ │ │ + owsType: owsType, │ │ │ │ │ + owsURL: owsURL, │ │ │ │ │ + typeName: typeName │ │ │ │ │ + }; │ │ │ │ │ + describelayer.layerDescriptions.push(layerDescription); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} http://www.opengis.net/xls │ │ │ │ │ - * http://schemas.opengis.net/ols/1.1.0/LocationUtilityService.xsd │ │ │ │ │ - */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/xls http://schemas.opengis.net/ols/1.1.0/LocationUtilityService.xsd", │ │ │ │ │ + //TODO do this in deprecated.js instead: │ │ │ │ │ + // array style index for backwards compatibility │ │ │ │ │ + describelayer.length = describelayer.layerDescriptions.length; │ │ │ │ │ + describelayer[describelayer.length - 1] = layerDescription; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.XLS.v1_1_0 │ │ │ │ │ - * Instances of this class are not created directly. Use the │ │ │ │ │ - * <OpenLayers.Format.XLS> constructor instead. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + } else if (nodeName == 'ServiceException') { │ │ │ │ │ + // an exception must have occurred, so parse it │ │ │ │ │ + var parser = new OpenLayers.Format.OGCExceptionReport(); │ │ │ │ │ + return { │ │ │ │ │ + error: parser.read(data) │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return describelayer; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.XLS.v1_1_0" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSDescribeLayer.v1_1_1" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ │ │ │ │ │ -// Support non standard implementation │ │ │ │ │ -OpenLayers.Format.XLS.v1_1 = OpenLayers.Format.XLS.v1_1_0; │ │ │ │ │ +// Version alias - workaround for http://trac.osgeo.org/mapserver/ticket/2257 │ │ │ │ │ +OpenLayers.Format.WMSDescribeLayer.v1_1_0 = │ │ │ │ │ + OpenLayers.Format.WMSDescribeLayer.v1_1_1; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Format/WCSCapabilities/v1.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ @@ -55183,14 +57335,127 @@ │ │ │ │ │ return capabilities; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WCSCapabilities/v1_1_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WCSCapabilities/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/OWSCommon/v1_1_0.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WCSCapabilities/v1_1_0 │ │ │ │ │ + * Read WCS Capabilities version 1.1.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WCSCapabilities.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WCSCapabilities.v1_1_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WCSCapabilities.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + wcs: "http://www.opengis.net/wcs/1.1", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + ows: "http://www.opengis.net/ows/1.1" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: errorProperty │ │ │ │ │ + * {String} Which property of the returned object to check for in order to │ │ │ │ │ + * determine whether or not parsing has failed. In the case that the │ │ │ │ │ + * errorProperty is undefined on the returned object, the document will be │ │ │ │ │ + * run through an OGCExceptionReport parser. │ │ │ │ │ + */ │ │ │ │ │ + errorProperty: "operationsMetadata", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WCSCapabilities.v1_1_0 │ │ │ │ │ + * Create a new parser for WCS capabilities version 1.1.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wcs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + // In 1.0.0, this was WCS_Capabilties, in 1.1.0, it's Capabilities │ │ │ │ │ + "Capabilities": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "Contents": function(node, request) { │ │ │ │ │ + request.contentMetadata = []; │ │ │ │ │ + this.readChildNodes(node, request.contentMetadata); │ │ │ │ │ + }, │ │ │ │ │ + "CoverageSummary": function(node, contentMetadata) { │ │ │ │ │ + var coverageSummary = {}; │ │ │ │ │ + // Read the summary: │ │ │ │ │ + this.readChildNodes(node, coverageSummary); │ │ │ │ │ + │ │ │ │ │ + // Add it to the contentMetadata array: │ │ │ │ │ + contentMetadata.push(coverageSummary); │ │ │ │ │ + }, │ │ │ │ │ + "Identifier": function(node, coverageSummary) { │ │ │ │ │ + coverageSummary.identifier = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Title": function(node, coverageSummary) { │ │ │ │ │ + coverageSummary.title = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Abstract": function(node, coverageSummary) { │ │ │ │ │ + coverageSummary["abstract"] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "SupportedCRS": function(node, coverageSummary) { │ │ │ │ │ + var crs = this.getChildValue(node); │ │ │ │ │ + if (crs) { │ │ │ │ │ + if (!coverageSummary.supportedCRS) { │ │ │ │ │ + coverageSummary.supportedCRS = []; │ │ │ │ │ + } │ │ │ │ │ + coverageSummary.supportedCRS.push(crs); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "SupportedFormat": function(node, coverageSummary) { │ │ │ │ │ + var format = this.getChildValue(node); │ │ │ │ │ + if (format) { │ │ │ │ │ + if (!coverageSummary.supportedFormat) { │ │ │ │ │ + coverageSummary.supportedFormat = []; │ │ │ │ │ + } │ │ │ │ │ + coverageSummary.supportedFormat.push(format); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WCSCapabilities.v1.prototype.readers["wcs"]), │ │ │ │ │ + "ows": OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1_1_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ OpenLayers/Format/WCSCapabilities/v1_0_0.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -55357,1270 +57622,376 @@ │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1_0_0" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WCSCapabilities/v1_1_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WCSCapabilities/v1.js │ │ │ │ │ - * @requires OpenLayers/Format/OWSCommon/v1_1_0.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WCSCapabilities/v1_1_0 │ │ │ │ │ - * Read WCS Capabilities version 1.1.0. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WCSCapabilities.v1> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WCSCapabilities.v1_1_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WCSCapabilities.v1, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - wcs: "http://www.opengis.net/wcs/1.1", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - ows: "http://www.opengis.net/ows/1.1" │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: errorProperty │ │ │ │ │ - * {String} Which property of the returned object to check for in order to │ │ │ │ │ - * determine whether or not parsing has failed. In the case that the │ │ │ │ │ - * errorProperty is undefined on the returned object, the document will be │ │ │ │ │ - * run through an OGCExceptionReport parser. │ │ │ │ │ - */ │ │ │ │ │ - errorProperty: "operationsMetadata", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WCSCapabilities.v1_1_0 │ │ │ │ │ - * Create a new parser for WCS capabilities version 1.1.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wcs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - // In 1.0.0, this was WCS_Capabilties, in 1.1.0, it's Capabilities │ │ │ │ │ - "Capabilities": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "Contents": function(node, request) { │ │ │ │ │ - request.contentMetadata = []; │ │ │ │ │ - this.readChildNodes(node, request.contentMetadata); │ │ │ │ │ - }, │ │ │ │ │ - "CoverageSummary": function(node, contentMetadata) { │ │ │ │ │ - var coverageSummary = {}; │ │ │ │ │ - // Read the summary: │ │ │ │ │ - this.readChildNodes(node, coverageSummary); │ │ │ │ │ - │ │ │ │ │ - // Add it to the contentMetadata array: │ │ │ │ │ - contentMetadata.push(coverageSummary); │ │ │ │ │ - }, │ │ │ │ │ - "Identifier": function(node, coverageSummary) { │ │ │ │ │ - coverageSummary.identifier = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Title": function(node, coverageSummary) { │ │ │ │ │ - coverageSummary.title = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Abstract": function(node, coverageSummary) { │ │ │ │ │ - coverageSummary["abstract"] = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "SupportedCRS": function(node, coverageSummary) { │ │ │ │ │ - var crs = this.getChildValue(node); │ │ │ │ │ - if (crs) { │ │ │ │ │ - if (!coverageSummary.supportedCRS) { │ │ │ │ │ - coverageSummary.supportedCRS = []; │ │ │ │ │ - } │ │ │ │ │ - coverageSummary.supportedCRS.push(crs); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "SupportedFormat": function(node, coverageSummary) { │ │ │ │ │ - var format = this.getChildValue(node); │ │ │ │ │ - if (format) { │ │ │ │ │ - if (!coverageSummary.supportedFormat) { │ │ │ │ │ - coverageSummary.supportedFormat = []; │ │ │ │ │ - } │ │ │ │ │ - coverageSummary.supportedFormat.push(format); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WCSCapabilities.v1.prototype.readers["wcs"]), │ │ │ │ │ - "ows": OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1_1_0" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/SLD/v1_0_0_GeoServer.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/SLD/v1_0_0.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.SLD/v1_0_0_GeoServer │ │ │ │ │ - * Read and write SLD version 1.0.0 with GeoServer-specific enhanced options. │ │ │ │ │ - * See http://svn.osgeo.org/geotools/trunk/modules/extension/xsd/xsd-sld/src/main/resources/org/geotools/sld/bindings/StyledLayerDescriptor.xsd │ │ │ │ │ - * for more information. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.SLD.v1_0_0> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.SLD.v1_0_0_GeoServer = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.SLD.v1_0_0, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} The specific parser version. │ │ │ │ │ - */ │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: profile │ │ │ │ │ - * {String} The specific profile │ │ │ │ │ - */ │ │ │ │ │ - profile: "GeoServer", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.SLD.v1_0_0_GeoServer │ │ │ │ │ - * Create a new parser for GeoServer-enhanced SLD version 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "sld": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "Priority": function(node, obj) { │ │ │ │ │ - var value = this.readers.ogc._expression.call(this, node); │ │ │ │ │ - if (value) { │ │ │ │ │ - obj.priority = value; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "VendorOption": function(node, obj) { │ │ │ │ │ - if (!obj.vendorOptions) { │ │ │ │ │ - obj.vendorOptions = {}; │ │ │ │ │ - } │ │ │ │ │ - obj.vendorOptions[node.getAttribute("name")] = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "TextSymbolizer": function(node, rule) { │ │ │ │ │ - OpenLayers.Format.SLD.v1_0_0.prototype.readers.sld.TextSymbolizer.apply(this, arguments); │ │ │ │ │ - var symbolizer = this.multipleSymbolizers ? rule.symbolizers[rule.symbolizers.length - 1] : rule.symbolizer["Text"]; │ │ │ │ │ - if (symbolizer.graphic === undefined) { │ │ │ │ │ - symbolizer.graphic = false; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.SLD.v1_0_0.prototype.readers["sld"]) │ │ │ │ │ - }, OpenLayers.Format.SLD.v1_0_0.prototype.readers), │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ - */ │ │ │ │ │ - writers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "sld": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "Priority": function(priority) { │ │ │ │ │ - return this.writers.sld._OGCExpression.call( │ │ │ │ │ - this, "sld:Priority", priority │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - "VendorOption": function(option) { │ │ │ │ │ - return this.createElementNSPlus("sld:VendorOption", { │ │ │ │ │ - attributes: { │ │ │ │ │ - name: option.name │ │ │ │ │ - }, │ │ │ │ │ - value: option.value │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "TextSymbolizer": function(symbolizer) { │ │ │ │ │ - var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ - var node = writers["sld"]["TextSymbolizer"].apply(this, arguments); │ │ │ │ │ - if (symbolizer.graphic !== false && (symbolizer.externalGraphic || symbolizer.graphicName)) { │ │ │ │ │ - this.writeNode("Graphic", symbolizer, node); │ │ │ │ │ - } │ │ │ │ │ - if ("priority" in symbolizer) { │ │ │ │ │ - this.writeNode("Priority", symbolizer.priority, node); │ │ │ │ │ - } │ │ │ │ │ - return this.addVendorOptions(node, symbolizer); │ │ │ │ │ - }, │ │ │ │ │ - "PointSymbolizer": function(symbolizer) { │ │ │ │ │ - var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ - var node = writers["sld"]["PointSymbolizer"].apply(this, arguments); │ │ │ │ │ - return this.addVendorOptions(node, symbolizer); │ │ │ │ │ - }, │ │ │ │ │ - "LineSymbolizer": function(symbolizer) { │ │ │ │ │ - var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ - var node = writers["sld"]["LineSymbolizer"].apply(this, arguments); │ │ │ │ │ - return this.addVendorOptions(node, symbolizer); │ │ │ │ │ - }, │ │ │ │ │ - "PolygonSymbolizer": function(symbolizer) { │ │ │ │ │ - var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ - var node = writers["sld"]["PolygonSymbolizer"].apply(this, arguments); │ │ │ │ │ - return this.addVendorOptions(node, symbolizer); │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.SLD.v1_0_0.prototype.writers["sld"]) │ │ │ │ │ - }, OpenLayers.Format.SLD.v1_0_0.prototype.writers), │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addVendorOptions │ │ │ │ │ - * Add in the VendorOption tags and return the node again. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A DOM node. │ │ │ │ │ - * symbolizer - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A DOM node. │ │ │ │ │ - */ │ │ │ │ │ - addVendorOptions: function(node, symbolizer) { │ │ │ │ │ - var options = symbolizer.vendorOptions; │ │ │ │ │ - if (options) { │ │ │ │ │ - for (var key in symbolizer.vendorOptions) { │ │ │ │ │ - this.writeNode("VendorOption", { │ │ │ │ │ - name: key, │ │ │ │ │ - value: symbolizer.vendorOptions[key] │ │ │ │ │ - }, node); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SLD.v1_0_0_GeoServer" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/CSWGetDomain/v2_0_2.js │ │ │ │ │ + OpenLayers/Format/XLS/v1.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/CSWGetDomain.js │ │ │ │ │ + * @requires OpenLayers/Format/XLS.js │ │ │ │ │ + * @requires OpenLayers/Format/GML/v3.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.CSWGetDomain.v2_0_2 │ │ │ │ │ - * A format for creating CSWGetDomain v2.0.2 transactions. │ │ │ │ │ - * Create a new instance with the │ │ │ │ │ - * <OpenLayers.Format.CSWGetDomain.v2_0_2> constructor. │ │ │ │ │ + * Class: OpenLayers.Format.XLS.v1 │ │ │ │ │ + * Superclass for XLS version 1 parsers. Only supports GeocodeRequest for now. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Format.XML> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.CSWGetDomain.v2_0_2 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Format.XLS.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Property: namespaces │ │ │ │ │ * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ */ │ │ │ │ │ namespaces: { │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - csw: "http://www.opengis.net/cat/csw/2.0.2" │ │ │ │ │ + xls: "http://www.opengis.net/xls", │ │ │ │ │ + gml: "http://www.opengis.net/gml", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - * {String} The default prefix (used by Format.XML). │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ */ │ │ │ │ │ - defaultPrefix: "csw", │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} CSW version number. │ │ │ │ │ + * APIProperty: xy │ │ │ │ │ + * {Boolean} Order of the GML coordinate true:(x,y) or false:(y,x) │ │ │ │ │ + * Changing is not recommended, a new Format should be instantiated. │ │ │ │ │ */ │ │ │ │ │ - version: "2.0.2", │ │ │ │ │ + xy: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} http://www.opengis.net/cat/csw/2.0.2 │ │ │ │ │ - * http://schemas.opengis.net/csw/2.0.2/CSW-discovery.xsd │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/cat/csw/2.0.2 http://schemas.opengis.net/csw/2.0.2/CSW-discovery.xsd", │ │ │ │ │ + defaultPrefix: "xls", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: PropertyName │ │ │ │ │ - * {String} Value of the csw:PropertyName element, used when │ │ │ │ │ - * writing a GetDomain document. │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} Schema location for a particular minor version. │ │ │ │ │ */ │ │ │ │ │ - PropertyName: null, │ │ │ │ │ + schemaLocation: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: ParameterName │ │ │ │ │ - * {String} Value of the csw:ParameterName element, used when │ │ │ │ │ - * writing a GetDomain document. │ │ │ │ │ + * Constructor: OpenLayers.Format.XLS.v1 │ │ │ │ │ + * Instances of this class are not created directly. Use the │ │ │ │ │ + * <OpenLayers.Format.XLS> constructor instead. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - ParameterName: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.CSWGetDomain.v2_0_2 │ │ │ │ │ - * A class for parsing and generating CSWGetDomain v2.0.2 transactions. │ │ │ │ │ + * Method: read │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * data - {DOMElement} An XLS document element. │ │ │ │ │ + * options - {Object} Options for the reader. │ │ │ │ │ * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * - PropertyName │ │ │ │ │ - * - ParameterName │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Parse the response from a GetDomain request. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object representing the XLSResponse. │ │ │ │ │ */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ - } │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readNode(data, obj); │ │ │ │ │ - return obj; │ │ │ │ │ + read: function(data, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var xls = {}; │ │ │ │ │ + this.readChildNodes(data, xls); │ │ │ │ │ + return xls; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Property: readers │ │ │ │ │ * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ * be applied when a namespaced node is found matching the function │ │ │ │ │ * name. The function will be applied in the scope of this parser │ │ │ │ │ * with two arguments: the node being read and a context object passed │ │ │ │ │ * from the parent. │ │ │ │ │ */ │ │ │ │ │ readers: { │ │ │ │ │ - "csw": { │ │ │ │ │ - "GetDomainResponse": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ + "xls": { │ │ │ │ │ + "XLS": function(node, xls) { │ │ │ │ │ + xls.version = node.getAttribute("version"); │ │ │ │ │ + this.readChildNodes(node, xls); │ │ │ │ │ }, │ │ │ │ │ - "DomainValues": function(node, obj) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(obj.DomainValues))) { │ │ │ │ │ - obj.DomainValues = []; │ │ │ │ │ - } │ │ │ │ │ - var attrs = node.attributes; │ │ │ │ │ - var domainValue = {}; │ │ │ │ │ - for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - domainValue[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ - } │ │ │ │ │ - this.readChildNodes(node, domainValue); │ │ │ │ │ - obj.DomainValues.push(domainValue); │ │ │ │ │ + "Response": function(node, xls) { │ │ │ │ │ + this.readChildNodes(node, xls); │ │ │ │ │ }, │ │ │ │ │ - "PropertyName": function(node, obj) { │ │ │ │ │ - obj.PropertyName = this.getChildValue(node); │ │ │ │ │ + "GeocodeResponse": function(node, xls) { │ │ │ │ │ + xls.responseLists = []; │ │ │ │ │ + this.readChildNodes(node, xls); │ │ │ │ │ }, │ │ │ │ │ - "ParameterName": function(node, obj) { │ │ │ │ │ - obj.ParameterName = this.getChildValue(node); │ │ │ │ │ + "GeocodeResponseList": function(node, xls) { │ │ │ │ │ + var responseList = { │ │ │ │ │ + features: [], │ │ │ │ │ + numberOfGeocodedAddresses: parseInt(node.getAttribute("numberOfGeocodedAddresses")) │ │ │ │ │ + }; │ │ │ │ │ + xls.responseLists.push(responseList); │ │ │ │ │ + this.readChildNodes(node, responseList); │ │ │ │ │ }, │ │ │ │ │ - "ListOfValues": function(node, obj) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(obj.ListOfValues))) { │ │ │ │ │ - obj.ListOfValues = []; │ │ │ │ │ - } │ │ │ │ │ - this.readChildNodes(node, obj.ListOfValues); │ │ │ │ │ + "GeocodedAddress": function(node, responseList) { │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ + responseList.features.push(feature); │ │ │ │ │ + this.readChildNodes(node, feature); │ │ │ │ │ + // post-process geometry │ │ │ │ │ + feature.geometry = feature.components[0]; │ │ │ │ │ }, │ │ │ │ │ - "Value": function(node, obj) { │ │ │ │ │ - var attrs = node.attributes; │ │ │ │ │ - var value = {}; │ │ │ │ │ - for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - value[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ - } │ │ │ │ │ - value.value = this.getChildValue(node); │ │ │ │ │ - obj.push({ │ │ │ │ │ - Value: value │ │ │ │ │ - }); │ │ │ │ │ + "GeocodeMatchCode": function(node, feature) { │ │ │ │ │ + feature.attributes.matchCode = { │ │ │ │ │ + accuracy: parseFloat(node.getAttribute("accuracy")), │ │ │ │ │ + matchType: node.getAttribute("matchType") │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ - "ConceptualScheme": function(node, obj) { │ │ │ │ │ - obj.ConceptualScheme = {}; │ │ │ │ │ - this.readChildNodes(node, obj.ConceptualScheme); │ │ │ │ │ + "Address": function(node, feature) { │ │ │ │ │ + var address = { │ │ │ │ │ + countryCode: node.getAttribute("countryCode"), │ │ │ │ │ + addressee: node.getAttribute("addressee"), │ │ │ │ │ + street: [], │ │ │ │ │ + place: [] │ │ │ │ │ + }; │ │ │ │ │ + feature.attributes.address = address; │ │ │ │ │ + this.readChildNodes(node, address); │ │ │ │ │ }, │ │ │ │ │ - "Name": function(node, obj) { │ │ │ │ │ - obj.Name = this.getChildValue(node); │ │ │ │ │ + "freeFormAddress": function(node, address) { │ │ │ │ │ + address.freeFormAddress = this.getChildValue(node); │ │ │ │ │ }, │ │ │ │ │ - "Document": function(node, obj) { │ │ │ │ │ - obj.Document = this.getChildValue(node); │ │ │ │ │ + "StreetAddress": function(node, address) { │ │ │ │ │ + this.readChildNodes(node, address); │ │ │ │ │ }, │ │ │ │ │ - "Authority": function(node, obj) { │ │ │ │ │ - obj.Authority = this.getChildValue(node); │ │ │ │ │ + "Building": function(node, address) { │ │ │ │ │ + address.building = { │ │ │ │ │ + 'number': node.getAttribute("number"), │ │ │ │ │ + subdivision: node.getAttribute("subdivision"), │ │ │ │ │ + buildingName: node.getAttribute("buildingName") │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ - "RangeOfValues": function(node, obj) { │ │ │ │ │ - obj.RangeOfValues = {}; │ │ │ │ │ - this.readChildNodes(node, obj.RangeOfValues); │ │ │ │ │ + "Street": function(node, address) { │ │ │ │ │ + // only support the built-in primitive type for now │ │ │ │ │ + address.street.push(this.getChildValue(node)); │ │ │ │ │ }, │ │ │ │ │ - "MinValue": function(node, obj) { │ │ │ │ │ - var attrs = node.attributes; │ │ │ │ │ - var value = {}; │ │ │ │ │ - for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - value[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ - } │ │ │ │ │ - value.value = this.getChildValue(node); │ │ │ │ │ - obj.MinValue = value; │ │ │ │ │ + "Place": function(node, address) { │ │ │ │ │ + // type is one of CountrySubdivision, │ │ │ │ │ + // CountrySecondarySubdivision, Municipality or │ │ │ │ │ + // MunicipalitySubdivision │ │ │ │ │ + address.place[node.getAttribute("type")] = │ │ │ │ │ + this.getChildValue(node); │ │ │ │ │ }, │ │ │ │ │ - "MaxValue": function(node, obj) { │ │ │ │ │ - var attrs = node.attributes; │ │ │ │ │ - var value = {}; │ │ │ │ │ - for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - value[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ - } │ │ │ │ │ - value.value = this.getChildValue(node); │ │ │ │ │ - obj.MaxValue = value; │ │ │ │ │ + "PostalCode": function(node, address) { │ │ │ │ │ + address.postalCode = this.getChildValue(node); │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ + "gml": OpenLayers.Format.GML.v3.prototype.readers.gml │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Given an configuration js object, write a CSWGetDomain request. │ │ │ │ │ + * Method: write │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} A object mapping the request. │ │ │ │ │ + * request - {Object} An object representing the geocode request. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A serialized CSWGetDomain request. │ │ │ │ │ + * {DOMElement} The root of an XLS document. │ │ │ │ │ */ │ │ │ │ │ - write: function(options) { │ │ │ │ │ - var node = this.writeNode("csw:GetDomain", options); │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [node]); │ │ │ │ │ + write: function(request) { │ │ │ │ │ + return this.writers.xls.XLS.apply(this, [request]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Property: writers │ │ │ │ │ * As a compliment to the readers property, this structure contains public │ │ │ │ │ * writing functions grouped by namespace alias and named like the │ │ │ │ │ * node names they produce. │ │ │ │ │ */ │ │ │ │ │ writers: { │ │ │ │ │ - "csw": { │ │ │ │ │ - "GetDomain": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:GetDomain", { │ │ │ │ │ + "xls": { │ │ │ │ │ + "XLS": function(request) { │ │ │ │ │ + var root = this.createElementNSPlus( │ │ │ │ │ + "xls:XLS", { │ │ │ │ │ + attributes: { │ │ │ │ │ + "version": this.VERSION, │ │ │ │ │ + "xsi:schemaLocation": this.schemaLocation │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + this.writeNode("RequestHeader", request.header, root); │ │ │ │ │ + this.writeNode("Request", request, root); │ │ │ │ │ + return root; │ │ │ │ │ + }, │ │ │ │ │ + "RequestHeader": function(header) { │ │ │ │ │ + return this.createElementNSPlus("xls:RequestHeader"); │ │ │ │ │ + }, │ │ │ │ │ + "Request": function(request) { │ │ │ │ │ + var node = this.createElementNSPlus("xls:Request", { │ │ │ │ │ attributes: { │ │ │ │ │ - service: "CSW", │ │ │ │ │ - version: this.version │ │ │ │ │ + methodName: "GeocodeRequest", │ │ │ │ │ + requestID: request.requestID || "", │ │ │ │ │ + version: this.VERSION │ │ │ │ │ } │ │ │ │ │ }); │ │ │ │ │ - if (options.PropertyName || this.PropertyName) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "csw:PropertyName", │ │ │ │ │ - options.PropertyName || this.PropertyName, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } else if (options.ParameterName || this.ParameterName) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "csw:ParameterName", │ │ │ │ │ - options.ParameterName || this.ParameterName, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - this.readChildNodes(node, options); │ │ │ │ │ + this.writeNode("GeocodeRequest", request.addresses, node); │ │ │ │ │ return node; │ │ │ │ │ }, │ │ │ │ │ - "PropertyName": function(value) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:PropertyName", { │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ + "GeocodeRequest": function(addresses) { │ │ │ │ │ + var node = this.createElementNSPlus("xls:GeocodeRequest"); │ │ │ │ │ + for (var i = 0, len = addresses.length; i < len; i++) { │ │ │ │ │ + this.writeNode("Address", addresses[i], node); │ │ │ │ │ + } │ │ │ │ │ return node; │ │ │ │ │ }, │ │ │ │ │ - "ParameterName": function(value) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:ParameterName", { │ │ │ │ │ - value: value │ │ │ │ │ + "Address": function(address) { │ │ │ │ │ + var node = this.createElementNSPlus("xls:Address", { │ │ │ │ │ + attributes: { │ │ │ │ │ + countryCode: address.countryCode │ │ │ │ │ + } │ │ │ │ │ }); │ │ │ │ │ - return node; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.CSWGetDomain.v2_0_2" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMTSCapabilities/v1_0_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WMTSCapabilities.js │ │ │ │ │ - * @requires OpenLayers/Format/OWSCommon/v1_1_0.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WMTSCapabilities.v1_0_0 │ │ │ │ │ - * Read WMTS Capabilities version 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMTSCapabilities> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WMTSCapabilities.v1_0_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.OWSCommon.v1_1_0, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} The parser version ("1.0.0"). │ │ │ │ │ - */ │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ - wmts: "http://www.opengis.net/wmts/1.0", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: yx │ │ │ │ │ - * {Object} Members in the yx object are used to determine if a CRS URN │ │ │ │ │ - * corresponds to a CRS with y,x axis order. Member names are CRS URNs │ │ │ │ │ - * and values are boolean. Defaults come from the │ │ │ │ │ - * <OpenLayers.Format.WMTSCapabilities> prototype. │ │ │ │ │ - */ │ │ │ │ │ - yx: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - * {String} The default namespace alias for creating element nodes. │ │ │ │ │ - */ │ │ │ │ │ - defaultPrefix: "wmts", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMTSCapabilities.v1_0_0 │ │ │ │ │ - * Create a new parser for WMTS capabilities version 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.options = options; │ │ │ │ │ - var yx = OpenLayers.Util.extend({}, OpenLayers.Format.WMTSCapabilities.prototype.yx); │ │ │ │ │ - this.yx = OpenLayers.Util.extend(yx, this.yx); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return info about the WMTS. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Information about the SOS service. │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ - } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - capabilities.version = this.version; │ │ │ │ │ - return capabilities; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wmts": { │ │ │ │ │ - "Capabilities": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "Contents": function(node, obj) { │ │ │ │ │ - obj.contents = {}; │ │ │ │ │ - obj.contents.layers = []; │ │ │ │ │ - obj.contents.tileMatrixSets = {}; │ │ │ │ │ - this.readChildNodes(node, obj.contents); │ │ │ │ │ - }, │ │ │ │ │ - "Layer": function(node, obj) { │ │ │ │ │ - var layer = { │ │ │ │ │ - styles: [], │ │ │ │ │ - formats: [], │ │ │ │ │ - dimensions: [], │ │ │ │ │ - tileMatrixSetLinks: [] │ │ │ │ │ - }; │ │ │ │ │ - layer.layers = []; │ │ │ │ │ - this.readChildNodes(node, layer); │ │ │ │ │ - obj.layers.push(layer); │ │ │ │ │ - }, │ │ │ │ │ - "Style": function(node, obj) { │ │ │ │ │ - var style = {}; │ │ │ │ │ - style.isDefault = (node.getAttribute("isDefault") === "true"); │ │ │ │ │ - this.readChildNodes(node, style); │ │ │ │ │ - obj.styles.push(style); │ │ │ │ │ - }, │ │ │ │ │ - "Format": function(node, obj) { │ │ │ │ │ - obj.formats.push(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "TileMatrixSetLink": function(node, obj) { │ │ │ │ │ - var tileMatrixSetLink = {}; │ │ │ │ │ - this.readChildNodes(node, tileMatrixSetLink); │ │ │ │ │ - obj.tileMatrixSetLinks.push(tileMatrixSetLink); │ │ │ │ │ - }, │ │ │ │ │ - "TileMatrixSet": function(node, obj) { │ │ │ │ │ - // node could be child of wmts:Contents or wmts:TileMatrixSetLink │ │ │ │ │ - // duck type wmts:Contents by looking for layers │ │ │ │ │ - if (obj.layers) { │ │ │ │ │ - // TileMatrixSet as object type in schema │ │ │ │ │ - var tileMatrixSet = { │ │ │ │ │ - matrixIds: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, tileMatrixSet); │ │ │ │ │ - obj.tileMatrixSets[tileMatrixSet.identifier] = tileMatrixSet; │ │ │ │ │ - } else { │ │ │ │ │ - // TileMatrixSet as string type in schema │ │ │ │ │ - obj.tileMatrixSet = this.getChildValue(node); │ │ │ │ │ + if (address.freeFormAddress) { │ │ │ │ │ + this.writeNode("freeFormAddress", address.freeFormAddress, node); │ │ │ │ │ + } else { │ │ │ │ │ + if (address.street) { │ │ │ │ │ + this.writeNode("StreetAddress", address, node); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "TileMatrix": function(node, obj) { │ │ │ │ │ - var tileMatrix = { │ │ │ │ │ - supportedCRS: obj.supportedCRS │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, tileMatrix); │ │ │ │ │ - obj.matrixIds.push(tileMatrix); │ │ │ │ │ - }, │ │ │ │ │ - "ScaleDenominator": function(node, obj) { │ │ │ │ │ - obj.scaleDenominator = parseFloat(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "TopLeftCorner": function(node, obj) { │ │ │ │ │ - var topLeftCorner = this.getChildValue(node); │ │ │ │ │ - var coords = topLeftCorner.split(" "); │ │ │ │ │ - // decide on axis order for the given CRS │ │ │ │ │ - var yx; │ │ │ │ │ - if (obj.supportedCRS) { │ │ │ │ │ - // extract out version from URN │ │ │ │ │ - var crs = obj.supportedCRS.replace( │ │ │ │ │ - /urn:ogc:def:crs:(\w+):.+:(\w+)$/, │ │ │ │ │ - "urn:ogc:def:crs:$1::$2" │ │ │ │ │ - ); │ │ │ │ │ - yx = !!this.yx[crs]; │ │ │ │ │ + if (address.municipality) { │ │ │ │ │ + this.writeNode("Municipality", address.municipality, node); │ │ │ │ │ } │ │ │ │ │ - if (yx) { │ │ │ │ │ - obj.topLeftCorner = new OpenLayers.LonLat( │ │ │ │ │ - coords[1], coords[0] │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - obj.topLeftCorner = new OpenLayers.LonLat( │ │ │ │ │ - coords[0], coords[1] │ │ │ │ │ - ); │ │ │ │ │ + if (address.countrySubdivision) { │ │ │ │ │ + this.writeNode("CountrySubdivision", address.countrySubdivision, node); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "TileWidth": function(node, obj) { │ │ │ │ │ - obj.tileWidth = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "TileHeight": function(node, obj) { │ │ │ │ │ - obj.tileHeight = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "MatrixWidth": function(node, obj) { │ │ │ │ │ - obj.matrixWidth = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "MatrixHeight": function(node, obj) { │ │ │ │ │ - obj.matrixHeight = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "ResourceURL": function(node, obj) { │ │ │ │ │ - obj.resourceUrl = obj.resourceUrl || {}; │ │ │ │ │ - var resourceType = node.getAttribute("resourceType"); │ │ │ │ │ - if (!obj.resourceUrls) { │ │ │ │ │ - obj.resourceUrls = []; │ │ │ │ │ + if (address.postalCode) { │ │ │ │ │ + this.writeNode("PostalCode", address.postalCode, node); │ │ │ │ │ } │ │ │ │ │ - var resourceUrl = obj.resourceUrl[resourceType] = { │ │ │ │ │ - format: node.getAttribute("format"), │ │ │ │ │ - template: node.getAttribute("template"), │ │ │ │ │ - resourceType: resourceType │ │ │ │ │ - }; │ │ │ │ │ - obj.resourceUrls.push(resourceUrl); │ │ │ │ │ - }, │ │ │ │ │ - // not used for now, can be added in the future though │ │ │ │ │ - /*"Themes": function(node, obj) { │ │ │ │ │ - obj.themes = []; │ │ │ │ │ - this.readChildNodes(node, obj.themes); │ │ │ │ │ - }, │ │ │ │ │ - "Theme": function(node, obj) { │ │ │ │ │ - var theme = {}; │ │ │ │ │ - this.readChildNodes(node, theme); │ │ │ │ │ - obj.push(theme); │ │ │ │ │ - },*/ │ │ │ │ │ - "WSDL": function(node, obj) { │ │ │ │ │ - obj.wsdl = {}; │ │ │ │ │ - obj.wsdl.href = node.getAttribute("xlink:href"); │ │ │ │ │ - // TODO: other attributes of <WSDL> element │ │ │ │ │ - }, │ │ │ │ │ - "ServiceMetadataURL": function(node, obj) { │ │ │ │ │ - obj.serviceMetadataUrl = {}; │ │ │ │ │ - obj.serviceMetadataUrl.href = node.getAttribute("xlink:href"); │ │ │ │ │ - // TODO: other attributes of <ServiceMetadataURL> element │ │ │ │ │ - }, │ │ │ │ │ - "LegendURL": function(node, obj) { │ │ │ │ │ - obj.legend = {}; │ │ │ │ │ - obj.legend.href = node.getAttribute("xlink:href"); │ │ │ │ │ - obj.legend.format = node.getAttribute("format"); │ │ │ │ │ - }, │ │ │ │ │ - "Dimension": function(node, obj) { │ │ │ │ │ - var dimension = { │ │ │ │ │ - values: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, dimension); │ │ │ │ │ - obj.dimensions.push(dimension); │ │ │ │ │ - }, │ │ │ │ │ - "Default": function(node, obj) { │ │ │ │ │ - obj["default"] = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Value": function(node, obj) { │ │ │ │ │ - obj.values.push(this.getChildValue(node)); │ │ │ │ │ } │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ - "ows": OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMTSCapabilities.v1_0_0" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WPSCapabilities/v1_0_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WPSCapabilities.js │ │ │ │ │ - * @requires OpenLayers/Format/OWSCommon/v1_1_0.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WPSCapabilities.v1_0_0 │ │ │ │ │ - * Read WPS Capabilities version 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WPSCapabilities.v1_0_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.XML, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ - wps: "http://www.opengis.net/wps/1.0.0", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ - */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WPSCapabilities.v1_0_0 │ │ │ │ │ - * Create a new parser for WPS capabilities version 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return info about the WPS. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Information about the WPS service. │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ - } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - return capabilities; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wps": { │ │ │ │ │ - "Capabilities": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "ProcessOfferings": function(node, obj) { │ │ │ │ │ - obj.processOfferings = {}; │ │ │ │ │ - this.readChildNodes(node, obj.processOfferings); │ │ │ │ │ - }, │ │ │ │ │ - "Process": function(node, processOfferings) { │ │ │ │ │ - var processVersion = this.getAttributeNS(node, this.namespaces.wps, "processVersion"); │ │ │ │ │ - var process = { │ │ │ │ │ - processVersion: processVersion │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, process); │ │ │ │ │ - processOfferings[process.identifier] = process; │ │ │ │ │ - }, │ │ │ │ │ - "Languages": function(node, obj) { │ │ │ │ │ - obj.languages = []; │ │ │ │ │ - this.readChildNodes(node, obj.languages); │ │ │ │ │ - }, │ │ │ │ │ - "Default": function(node, languages) { │ │ │ │ │ - var language = { │ │ │ │ │ - isDefault: true │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, language); │ │ │ │ │ - languages.push(language); │ │ │ │ │ - }, │ │ │ │ │ - "Supported": function(node, languages) { │ │ │ │ │ - var language = {}; │ │ │ │ │ - this.readChildNodes(node, language); │ │ │ │ │ - languages.push(language); │ │ │ │ │ + "freeFormAddress": function(freeFormAddress) { │ │ │ │ │ + return this.createElementNSPlus("freeFormAddress", { │ │ │ │ │ + value: freeFormAddress │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "StreetAddress": function(address) { │ │ │ │ │ + var node = this.createElementNSPlus("xls:StreetAddress"); │ │ │ │ │ + if (address.building) { │ │ │ │ │ + this.writeNode(node, "Building", address.building); │ │ │ │ │ + } │ │ │ │ │ + var street = address.street; │ │ │ │ │ + if (!(OpenLayers.Util.isArray(street))) { │ │ │ │ │ + street = [street]; │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, len = street.length; i < len; i++) { │ │ │ │ │ + this.writeNode("Street", street[i], node); │ │ │ │ │ } │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ - "ows": OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WPSCapabilities.v1_0_0" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSDescribeLayer/v1_1.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WMSDescribeLayer.js │ │ │ │ │ - * @requires OpenLayers/Format/OGCExceptionReport.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WMSDescribeLayer.v1_1_1 │ │ │ │ │ - * Read SLD WMS DescribeLayer response for WMS 1.1.X │ │ │ │ │ - * WMS 1.1.X is tightly coupled to SLD 1.0.0 │ │ │ │ │ - * │ │ │ │ │ - * Example DescribeLayer request: │ │ │ │ │ - * http://demo.opengeo.org/geoserver/wms?request=DescribeLayer&version=1.1.1&layers=topp:states │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMSDescribeLayer> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WMSDescribeLayer.v1_1_1 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WMSDescribeLayer, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMSDescribeLayer │ │ │ │ │ - * Create a new parser for WMS DescribeLayer responses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.WMSDescribeLayer.prototype.initialize.apply(this, │ │ │ │ │ - [options]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read DescribeLayer data from a string, and return the response. │ │ │ │ │ - * The OGC defines 2 formats which are allowed for output, │ │ │ │ │ - * so we need to parse these 2 types for version 1.1.X │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Object with a layerDescriptions property, which holds an Array │ │ │ │ │ - * of {<LayerDescription>} objects which have: │ │ │ │ │ - * - {String} owsType: WFS/WCS │ │ │ │ │ - * - {String} owsURL: the online resource │ │ │ │ │ - * - {String} typeName: the name of the typename on the owsType service │ │ │ │ │ - * - {String} layerName: the name of the WMS layer we did a lookup for │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - var root = data.documentElement; │ │ │ │ │ - var children = root.childNodes; │ │ │ │ │ - var describelayer = { │ │ │ │ │ - layerDescriptions: [] │ │ │ │ │ - }; │ │ │ │ │ - var childNode, nodeName; │ │ │ │ │ - for (var i = 0; i < children.length; ++i) { │ │ │ │ │ - childNode = children[i]; │ │ │ │ │ - nodeName = childNode.nodeName; │ │ │ │ │ - if (nodeName == 'LayerDescription') { │ │ │ │ │ - var layerName = childNode.getAttribute('name'); │ │ │ │ │ - var owsType = ''; │ │ │ │ │ - var owsURL = ''; │ │ │ │ │ - var typeName = ''; │ │ │ │ │ - // check for owsType and owsURL attributes │ │ │ │ │ - if (childNode.getAttribute('owsType')) { │ │ │ │ │ - owsType = childNode.getAttribute('owsType'); │ │ │ │ │ - owsURL = childNode.getAttribute('owsURL'); │ │ │ │ │ - } else { │ │ │ │ │ - // look for wfs or wcs attribute │ │ │ │ │ - if (childNode.getAttribute('wfs') != '') { │ │ │ │ │ - owsType = 'WFS'; │ │ │ │ │ - owsURL = childNode.getAttribute('wfs'); │ │ │ │ │ - } else if (childNode.getAttribute('wcs') != '') { │ │ │ │ │ - owsType = 'WCS'; │ │ │ │ │ - owsURL = childNode.getAttribute('wcs'); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // look for Query child │ │ │ │ │ - var query = childNode.getElementsByTagName('Query'); │ │ │ │ │ - if (query.length > 0) { │ │ │ │ │ - typeName = query[0].getAttribute('typeName'); │ │ │ │ │ - if (!typeName) { │ │ │ │ │ - // because of Ionic bug │ │ │ │ │ - typeName = query[0].getAttribute('typename'); │ │ │ │ │ - } │ │ │ │ │ + "Building": function(building) { │ │ │ │ │ + return this.createElementNSPlus("xls:Building", { │ │ │ │ │ + attributes: { │ │ │ │ │ + "number": building["number"], │ │ │ │ │ + "subdivision": building.subdivision, │ │ │ │ │ + "buildingName": building.buildingName │ │ │ │ │ } │ │ │ │ │ - var layerDescription = { │ │ │ │ │ - layerName: layerName, │ │ │ │ │ - owsType: owsType, │ │ │ │ │ - owsURL: owsURL, │ │ │ │ │ - typeName: typeName │ │ │ │ │ - }; │ │ │ │ │ - describelayer.layerDescriptions.push(layerDescription); │ │ │ │ │ - │ │ │ │ │ - //TODO do this in deprecated.js instead: │ │ │ │ │ - // array style index for backwards compatibility │ │ │ │ │ - describelayer.length = describelayer.layerDescriptions.length; │ │ │ │ │ - describelayer[describelayer.length - 1] = layerDescription; │ │ │ │ │ - │ │ │ │ │ - } else if (nodeName == 'ServiceException') { │ │ │ │ │ - // an exception must have occurred, so parse it │ │ │ │ │ - var parser = new OpenLayers.Format.OGCExceptionReport(); │ │ │ │ │ - return { │ │ │ │ │ - error: parser.read(data) │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "Street": function(street) { │ │ │ │ │ + return this.createElementNSPlus("xls:Street", { │ │ │ │ │ + value: street │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "Municipality": function(municipality) { │ │ │ │ │ + return this.createElementNSPlus("xls:Place", { │ │ │ │ │ + attributes: { │ │ │ │ │ + type: "Municipality" │ │ │ │ │ + }, │ │ │ │ │ + value: municipality │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "CountrySubdivision": function(countrySubdivision) { │ │ │ │ │ + return this.createElementNSPlus("xls:Place", { │ │ │ │ │ + attributes: { │ │ │ │ │ + type: "CountrySubdivision" │ │ │ │ │ + }, │ │ │ │ │ + value: countrySubdivision │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "PostalCode": function(postalCode) { │ │ │ │ │ + return this.createElementNSPlus("xls:PostalCode", { │ │ │ │ │ + value: postalCode │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - return describelayer; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSDescribeLayer.v1_1_1" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ -// Version alias - workaround for http://trac.osgeo.org/mapserver/ticket/2257 │ │ │ │ │ -OpenLayers.Format.WMSDescribeLayer.v1_1_0 = │ │ │ │ │ - OpenLayers.Format.WMSDescribeLayer.v1_1_1; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/ArcXML/Features.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/ArcXML.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.ArcXML.Features │ │ │ │ │ - * Read/Write ArcXML features. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Format.ArcXML.Features> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.ArcXML.Features = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.ArcXML.Features │ │ │ │ │ - * Create a new parser/writer for ArcXML Features. Create an instance of this class │ │ │ │ │ - * to get a set of features from an ArcXML response. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read data from a string of ArcXML, and return a set of OpenLayers features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} A collection of features. │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - var axl = new OpenLayers.Format.ArcXML(); │ │ │ │ │ - var parsed = axl.read(data); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.XLS.v1" │ │ │ │ │ │ │ │ │ │ - return parsed.features.feature; │ │ │ │ │ - } │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WFST/v1_0_0.js │ │ │ │ │ + OpenLayers/Format/XLS/v1_1_0.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/WFST/v1.js │ │ │ │ │ - * @requires OpenLayers/Format/Filter/v1_0_0.js │ │ │ │ │ + * @requires OpenLayers/Format/XLS/v1.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WFST.v1_0_0 │ │ │ │ │ - * A format for creating WFS v1.0.0 transactions. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Format.WFST.v1_0_0> constructor. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Format.XLS.v1_1_0 │ │ │ │ │ + * Read / write XLS version 1.1.0. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.Filter.v1_0_0> │ │ │ │ │ - * - <OpenLayers.Format.WFST.v1> │ │ │ │ │ + * - <OpenLayers.Format.XLS.v1> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WFST.v1_0_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.Filter.v1_0_0, OpenLayers.Format.WFST.v1, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} WFS version number. │ │ │ │ │ - */ │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: srsNameInQuery │ │ │ │ │ - * {Boolean} If true the reference system is passed in Query requests │ │ │ │ │ - * via the "srsName" attribute to the "wfs:Query" element, this │ │ │ │ │ - * property defaults to false as it isn't WFS 1.0.0 compliant. │ │ │ │ │ - */ │ │ │ │ │ - srsNameInQuery: false, │ │ │ │ │ +OpenLayers.Format.XLS.v1_1_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.XLS.v1, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: schemaLocations │ │ │ │ │ - * {Object} Properties are namespace aliases, values are schema locations. │ │ │ │ │ + * Constant: VERSION │ │ │ │ │ + * {String} 1.1 │ │ │ │ │ */ │ │ │ │ │ - schemaLocations: { │ │ │ │ │ - "wfs": "http://schemas.opengis.net/wfs/1.0.0/WFS-transaction.xsd" │ │ │ │ │ - }, │ │ │ │ │ + VERSION: "1.1", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WFST.v1_0_0 │ │ │ │ │ - * A class for parsing and generating WFS v1.0.0 transactions. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ - * featureNS - {String} Feature namespace (optional). │ │ │ │ │ - * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ - * if featureNS is provided). Default is 'feature'. │ │ │ │ │ - * geometryName - {String} Name of geometry attribute. Default is 'the_geom'. │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} http://www.opengis.net/xls │ │ │ │ │ + * http://schemas.opengis.net/ols/1.1.0/LocationUtilityService.xsd │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.Filter.v1_0_0.prototype.initialize.apply(this, [options]); │ │ │ │ │ - OpenLayers.Format.WFST.v1.prototype.initialize.apply(this, [options]); │ │ │ │ │ - }, │ │ │ │ │ + schemaLocation: "http://www.opengis.net/xls http://schemas.opengis.net/ols/1.1.0/LocationUtilityService.xsd", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: readNode │ │ │ │ │ - * Shorthand for applying one of the named readers given the node │ │ │ │ │ - * namespace and local name. Readers take two args (node, obj) and │ │ │ │ │ - * generally extend or modify the second. │ │ │ │ │ + * Constructor: OpenLayers.Format.XLS.v1_1_0 │ │ │ │ │ + * Instances of this class are not created directly. Use the │ │ │ │ │ + * <OpenLayers.Format.XLS> constructor instead. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} The node to be read (required). │ │ │ │ │ - * obj - {Object} The object to be modified (optional). │ │ │ │ │ - * first - {Boolean} Should be set to true for the first node read. This │ │ │ │ │ - * is usually the readNode call in the read method. Without this being │ │ │ │ │ - * set, auto-configured properties will stick on subsequent reads. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The input object, modified (or a new one if none was provided). │ │ │ │ │ - */ │ │ │ │ │ - readNode: function(node, obj, first) { │ │ │ │ │ - // Not the superclass, only the mixin classes inherit from │ │ │ │ │ - // Format.GML.v2. We need this because we don't want to get readNode │ │ │ │ │ - // from the superclass's superclass, which is OpenLayers.Format.XML. │ │ │ │ │ - return OpenLayers.Format.GML.v2.prototype.readNode.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "WFS_TransactionResponse": function(node, obj) { │ │ │ │ │ - obj.insertIds = []; │ │ │ │ │ - obj.success = false; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "InsertResult": function(node, container) { │ │ │ │ │ - var obj = { │ │ │ │ │ - fids: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - container.insertIds = container.insertIds.concat(obj.fids); │ │ │ │ │ - }, │ │ │ │ │ - "TransactionResult": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "Status": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "SUCCESS": function(node, obj) { │ │ │ │ │ - obj.success = true; │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WFST.v1.prototype.readers["wfs"]), │ │ │ │ │ - "gml": OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ - "feature": OpenLayers.Format.GML.v2.prototype.readers["feature"], │ │ │ │ │ - "ogc": OpenLayers.Format.Filter.v1_0_0.prototype.readers["ogc"] │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ - */ │ │ │ │ │ - writers: { │ │ │ │ │ - "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "Query": function(options) { │ │ │ │ │ - options = OpenLayers.Util.extend({ │ │ │ │ │ - featureNS: this.featureNS, │ │ │ │ │ - featurePrefix: this.featurePrefix, │ │ │ │ │ - featureType: this.featureType, │ │ │ │ │ - srsName: this.srsName, │ │ │ │ │ - srsNameInQuery: this.srsNameInQuery │ │ │ │ │ - }, options); │ │ │ │ │ - var prefix = options.featurePrefix; │ │ │ │ │ - var node = this.createElementNSPlus("wfs:Query", { │ │ │ │ │ - attributes: { │ │ │ │ │ - typeName: (prefix ? prefix + ":" : "") + │ │ │ │ │ - options.featureType │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (options.srsNameInQuery && options.srsName) { │ │ │ │ │ - node.setAttribute("srsName", options.srsName); │ │ │ │ │ - } │ │ │ │ │ - if (options.featureNS) { │ │ │ │ │ - node.setAttribute("xmlns:" + prefix, options.featureNS); │ │ │ │ │ - } │ │ │ │ │ - if (options.propertyNames) { │ │ │ │ │ - for (var i = 0, len = options.propertyNames.length; i < len; i++) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "ogc:PropertyName", { │ │ │ │ │ - property: options.propertyNames[i] │ │ │ │ │ - }, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (options.filter) { │ │ │ │ │ - this.setFilterProperty(options.filter); │ │ │ │ │ - this.writeNode("ogc:Filter", options.filter, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WFST.v1.prototype.writers["wfs"]), │ │ │ │ │ - "gml": OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ - "feature": OpenLayers.Format.GML.v2.prototype.writers["feature"], │ │ │ │ │ - "ogc": OpenLayers.Format.Filter.v1_0_0.prototype.writers["ogc"] │ │ │ │ │ - }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.XLS.v1_1_0" │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFST.v1_0_0" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +// Support non standard implementation │ │ │ │ │ +OpenLayers.Format.XLS.v1_1 = OpenLayers.Format.XLS.v1_1_0; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Format/SOSCapabilities/v1_0_0.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ @@ -58049,101 +59420,143 @@ │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.WMC.v1" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMC/v1_0_0.js │ │ │ │ │ + OpenLayers/Format/WMC/v1_1_0.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Format/WMC/v1.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WMC.v1_0_0 │ │ │ │ │ - * Read and write WMC version 1.0.0. │ │ │ │ │ + * Class: OpenLayers.Format.WMC.v1_1_0 │ │ │ │ │ + * Read and write WMC version 1.1.0. │ │ │ │ │ + * │ │ │ │ │ + * Differences between 1.1.0 and 1.0.0: │ │ │ │ │ + * - 1.1.0 Layers have optional sld:MinScaleDenominator and │ │ │ │ │ + * sld:MaxScaleDenominator │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Format.WMC.v1> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WMC.v1_0_0 = OpenLayers.Class( │ │ │ │ │ +OpenLayers.Format.WMC.v1_1_0 = OpenLayers.Class( │ │ │ │ │ OpenLayers.Format.WMC.v1, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Constant: VERSION │ │ │ │ │ - * {String} 1.0.0 │ │ │ │ │ + * {String} 1.1.0 │ │ │ │ │ */ │ │ │ │ │ - VERSION: "1.0.0", │ │ │ │ │ + VERSION: "1.1.0", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Property: schemaLocation │ │ │ │ │ * {String} http://www.opengis.net/context │ │ │ │ │ - * http://schemas.opengis.net/context/1.0.0/context.xsd │ │ │ │ │ + * http://schemas.opengis.net/context/1.1.0/context.xsd │ │ │ │ │ */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/context http://schemas.opengis.net/context/1.0.0/context.xsd", │ │ │ │ │ + schemaLocation: "http://www.opengis.net/context http://schemas.opengis.net/context/1.1.0/context.xsd", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMC.v1_0_0 │ │ │ │ │ + * Constructor: OpenLayers.Format.WMC.v1_1_0 │ │ │ │ │ * Instances of this class are not created directly. Use the │ │ │ │ │ * <OpenLayers.Format.WMC> constructor instead. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * options - {Object} An optional object whose properties will be set on │ │ │ │ │ * this instance. │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ OpenLayers.Format.WMC.v1.prototype.initialize.apply( │ │ │ │ │ this, [options] │ │ │ │ │ ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ + * Method: read_sld_MinScaleDenominator │ │ │ │ │ + * Read a sld:MinScaleDenominator node. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layerContext - {Object} An object representing a layer. │ │ │ │ │ + * node - {Element} An element node. │ │ │ │ │ + */ │ │ │ │ │ + read_sld_MinScaleDenominator: function(layerContext, node) { │ │ │ │ │ + var minScaleDenominator = parseFloat(this.getChildValue(node)); │ │ │ │ │ + if (minScaleDenominator > 0) { │ │ │ │ │ + layerContext.maxScale = minScaleDenominator; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: read_sld_MaxScaleDenominator │ │ │ │ │ + * Read a sld:MaxScaleDenominator node. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layerContext - {Object} An object representing a layer. │ │ │ │ │ + * node - {Element} An element node. │ │ │ │ │ + */ │ │ │ │ │ + read_sld_MaxScaleDenominator: function(layerContext, node) { │ │ │ │ │ + layerContext.minScale = parseFloat(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ * Method: read_wmc_SRS │ │ │ │ │ */ │ │ │ │ │ read_wmc_SRS: function(layerContext, node) { │ │ │ │ │ - var srs = this.getChildValue(node); │ │ │ │ │ - if (typeof layerContext.projections != "object") { │ │ │ │ │ - layerContext.projections = {}; │ │ │ │ │ - } │ │ │ │ │ - var values = srs.split(/ +/); │ │ │ │ │ - for (var i = 0, len = values.length; i < len; i++) { │ │ │ │ │ - layerContext.projections[values[i]] = true; │ │ │ │ │ + if (!("srs" in layerContext)) { │ │ │ │ │ + layerContext.srs = {}; │ │ │ │ │ } │ │ │ │ │ + layerContext.srs[this.getChildValue(node)] = true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Method: write_wmc_Layer │ │ │ │ │ * Create a Layer node given a layer context object. This method adds │ │ │ │ │ - * elements specific to version 1.0.0. │ │ │ │ │ + * elements specific to version 1.1.0. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * context - {Object} A layer context object.} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ * {Element} A WMC Layer element node. │ │ │ │ │ */ │ │ │ │ │ write_wmc_Layer: function(context) { │ │ │ │ │ var node = OpenLayers.Format.WMC.v1.prototype.write_wmc_Layer.apply( │ │ │ │ │ this, [context] │ │ │ │ │ ); │ │ │ │ │ │ │ │ │ │ + // min/max scale denominator elements go before the 4th element in v1 │ │ │ │ │ + if (context.maxScale) { │ │ │ │ │ + var minSD = this.createElementNS( │ │ │ │ │ + this.namespaces.sld, "sld:MinScaleDenominator" │ │ │ │ │ + ); │ │ │ │ │ + minSD.appendChild(this.createTextNode(context.maxScale.toPrecision(16))); │ │ │ │ │ + node.appendChild(minSD); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (context.minScale) { │ │ │ │ │ + var maxSD = this.createElementNS( │ │ │ │ │ + this.namespaces.sld, "sld:MaxScaleDenominator" │ │ │ │ │ + ); │ │ │ │ │ + maxSD.appendChild(this.createTextNode(context.minScale.toPrecision(16))); │ │ │ │ │ + node.appendChild(maxSD); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ // optional SRS element(s) │ │ │ │ │ if (context.srs) { │ │ │ │ │ - var projections = []; │ │ │ │ │ for (var name in context.srs) { │ │ │ │ │ - projections.push(name); │ │ │ │ │ + node.appendChild(this.createElementDefaultNS("SRS", name)); │ │ │ │ │ } │ │ │ │ │ - node.appendChild(this.createElementDefaultNS("SRS", projections.join(" "))); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ // optional FormatList element │ │ │ │ │ node.appendChild(this.write_wmc_FormatList(context)); │ │ │ │ │ │ │ │ │ │ // optional StyleList element │ │ │ │ │ node.appendChild(this.write_wmc_StyleList(context)); │ │ │ │ │ @@ -58151,149 +59564,110 @@ │ │ │ │ │ // optional DimensionList element │ │ │ │ │ if (context.dimensions) { │ │ │ │ │ node.appendChild(this.write_wmc_DimensionList(context)); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ // OpenLayers specific properties go in an Extension element │ │ │ │ │ node.appendChild(this.write_wmc_LayerExtension(context)); │ │ │ │ │ + │ │ │ │ │ + return node; │ │ │ │ │ + │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMC.v1_0_0" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMC.v1_1_0" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMC/v1_1_0.js │ │ │ │ │ + OpenLayers/Format/WMC/v1_0_0.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Format/WMC/v1.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WMC.v1_1_0 │ │ │ │ │ - * Read and write WMC version 1.1.0. │ │ │ │ │ - * │ │ │ │ │ - * Differences between 1.1.0 and 1.0.0: │ │ │ │ │ - * - 1.1.0 Layers have optional sld:MinScaleDenominator and │ │ │ │ │ - * sld:MaxScaleDenominator │ │ │ │ │ + * Class: OpenLayers.Format.WMC.v1_0_0 │ │ │ │ │ + * Read and write WMC version 1.0.0. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Format.WMC.v1> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WMC.v1_1_0 = OpenLayers.Class( │ │ │ │ │ +OpenLayers.Format.WMC.v1_0_0 = OpenLayers.Class( │ │ │ │ │ OpenLayers.Format.WMC.v1, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Constant: VERSION │ │ │ │ │ - * {String} 1.1.0 │ │ │ │ │ + * {String} 1.0.0 │ │ │ │ │ */ │ │ │ │ │ - VERSION: "1.1.0", │ │ │ │ │ + VERSION: "1.0.0", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Property: schemaLocation │ │ │ │ │ * {String} http://www.opengis.net/context │ │ │ │ │ - * http://schemas.opengis.net/context/1.1.0/context.xsd │ │ │ │ │ + * http://schemas.opengis.net/context/1.0.0/context.xsd │ │ │ │ │ */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/context http://schemas.opengis.net/context/1.1.0/context.xsd", │ │ │ │ │ + schemaLocation: "http://www.opengis.net/context http://schemas.opengis.net/context/1.0.0/context.xsd", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMC.v1_1_0 │ │ │ │ │ + * Constructor: OpenLayers.Format.WMC.v1_0_0 │ │ │ │ │ * Instances of this class are not created directly. Use the │ │ │ │ │ * <OpenLayers.Format.WMC> constructor instead. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * options - {Object} An optional object whose properties will be set on │ │ │ │ │ * this instance. │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ OpenLayers.Format.WMC.v1.prototype.initialize.apply( │ │ │ │ │ this, [options] │ │ │ │ │ ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_sld_MinScaleDenominator │ │ │ │ │ - * Read a sld:MinScaleDenominator node. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layerContext - {Object} An object representing a layer. │ │ │ │ │ - * node - {Element} An element node. │ │ │ │ │ - */ │ │ │ │ │ - read_sld_MinScaleDenominator: function(layerContext, node) { │ │ │ │ │ - var minScaleDenominator = parseFloat(this.getChildValue(node)); │ │ │ │ │ - if (minScaleDenominator > 0) { │ │ │ │ │ - layerContext.maxScale = minScaleDenominator; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_sld_MaxScaleDenominator │ │ │ │ │ - * Read a sld:MaxScaleDenominator node. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layerContext - {Object} An object representing a layer. │ │ │ │ │ - * node - {Element} An element node. │ │ │ │ │ - */ │ │ │ │ │ - read_sld_MaxScaleDenominator: function(layerContext, node) { │ │ │ │ │ - layerContext.minScale = parseFloat(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ * Method: read_wmc_SRS │ │ │ │ │ */ │ │ │ │ │ read_wmc_SRS: function(layerContext, node) { │ │ │ │ │ - if (!("srs" in layerContext)) { │ │ │ │ │ - layerContext.srs = {}; │ │ │ │ │ + var srs = this.getChildValue(node); │ │ │ │ │ + if (typeof layerContext.projections != "object") { │ │ │ │ │ + layerContext.projections = {}; │ │ │ │ │ + } │ │ │ │ │ + var values = srs.split(/ +/); │ │ │ │ │ + for (var i = 0, len = values.length; i < len; i++) { │ │ │ │ │ + layerContext.projections[values[i]] = true; │ │ │ │ │ } │ │ │ │ │ - layerContext.srs[this.getChildValue(node)] = true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Method: write_wmc_Layer │ │ │ │ │ * Create a Layer node given a layer context object. This method adds │ │ │ │ │ - * elements specific to version 1.1.0. │ │ │ │ │ + * elements specific to version 1.0.0. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * context - {Object} A layer context object.} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ * {Element} A WMC Layer element node. │ │ │ │ │ */ │ │ │ │ │ write_wmc_Layer: function(context) { │ │ │ │ │ var node = OpenLayers.Format.WMC.v1.prototype.write_wmc_Layer.apply( │ │ │ │ │ this, [context] │ │ │ │ │ ); │ │ │ │ │ │ │ │ │ │ - // min/max scale denominator elements go before the 4th element in v1 │ │ │ │ │ - if (context.maxScale) { │ │ │ │ │ - var minSD = this.createElementNS( │ │ │ │ │ - this.namespaces.sld, "sld:MinScaleDenominator" │ │ │ │ │ - ); │ │ │ │ │ - minSD.appendChild(this.createTextNode(context.maxScale.toPrecision(16))); │ │ │ │ │ - node.appendChild(minSD); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (context.minScale) { │ │ │ │ │ - var maxSD = this.createElementNS( │ │ │ │ │ - this.namespaces.sld, "sld:MaxScaleDenominator" │ │ │ │ │ - ); │ │ │ │ │ - maxSD.appendChild(this.createTextNode(context.minScale.toPrecision(16))); │ │ │ │ │ - node.appendChild(maxSD); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ // optional SRS element(s) │ │ │ │ │ if (context.srs) { │ │ │ │ │ + var projections = []; │ │ │ │ │ for (var name in context.srs) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS("SRS", name)); │ │ │ │ │ + projections.push(name); │ │ │ │ │ } │ │ │ │ │ + node.appendChild(this.createElementDefaultNS("SRS", projections.join(" "))); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ // optional FormatList element │ │ │ │ │ node.appendChild(this.write_wmc_FormatList(context)); │ │ │ │ │ │ │ │ │ │ // optional StyleList element │ │ │ │ │ node.appendChild(this.write_wmc_StyleList(context)); │ │ │ │ │ @@ -58301,1241 +59675,229 @@ │ │ │ │ │ // optional DimensionList element │ │ │ │ │ if (context.dimensions) { │ │ │ │ │ node.appendChild(this.write_wmc_DimensionList(context)); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ // OpenLayers specific properties go in an Extension element │ │ │ │ │ node.appendChild(this.write_wmc_LayerExtension(context)); │ │ │ │ │ - │ │ │ │ │ - return node; │ │ │ │ │ - │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMC.v1_1_0" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMC.v1_0_0" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSCapabilities/v1.js │ │ │ │ │ + OpenLayers/Events/buttonclick.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/WMSCapabilities.js │ │ │ │ │ - * @requires OpenLayers/Format/OGCExceptionReport.js │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WMSCapabilities.v1 │ │ │ │ │ - * Abstract class not to be instantiated directly. Creates │ │ │ │ │ - * the common parts for both WMS 1.1.X and WMS 1.3.X. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * Class: OpenLayers.Events.buttonclick │ │ │ │ │ + * Extension event type for handling buttons on top of a dom element. This │ │ │ │ │ + * event type fires "buttonclick" on its <target> when a button was │ │ │ │ │ + * clicked. Buttons are detected by the "olButton" class. │ │ │ │ │ + * │ │ │ │ │ + * This event type makes sure that button clicks do not interfere with other │ │ │ │ │ + * events that are registered on the same <element>. │ │ │ │ │ + * │ │ │ │ │ + * Event types provided by this extension: │ │ │ │ │ + * - *buttonclick* Triggered when a button is clicked. Listeners receive an │ │ │ │ │ + * object with a *buttonElement* property referencing the dom element of │ │ │ │ │ + * the clicked button, and an *buttonXY* property with the click position │ │ │ │ │ + * relative to the button. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Events.buttonclick = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - wms: "http://www.opengis.net/wms", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: target │ │ │ │ │ + * {<OpenLayers.Events>} The events instance that the buttonclick event will │ │ │ │ │ + * be triggered on. │ │ │ │ │ + */ │ │ │ │ │ + target: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - */ │ │ │ │ │ - defaultPrefix: "wms", │ │ │ │ │ + /** │ │ │ │ │ + * Property: events │ │ │ │ │ + * {Array} Events to observe and conditionally stop from propagating when │ │ │ │ │ + * an element with the olButton class (or its olAlphaImg child) is │ │ │ │ │ + * clicked. │ │ │ │ │ + */ │ │ │ │ │ + events: [ │ │ │ │ │ + 'mousedown', 'mouseup', 'click', 'dblclick', │ │ │ │ │ + 'touchstart', 'touchmove', 'touchend', 'keydown' │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMSCapabilities.v1 │ │ │ │ │ - * Create an instance of one of the subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: startRegEx │ │ │ │ │ + * {RegExp} Regular expression to test Event.type for events that start │ │ │ │ │ + * a buttonclick sequence. │ │ │ │ │ + */ │ │ │ │ │ + startRegEx: /^mousedown|touchstart$/, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} List of named layers. │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - var raw = data; │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ + /** │ │ │ │ │ + * Property: cancelRegEx │ │ │ │ │ + * {RegExp} Regular expression to test Event.type for events that cancel │ │ │ │ │ + * a buttonclick sequence. │ │ │ │ │ + */ │ │ │ │ │ + cancelRegEx: /^touchmove$/, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: completeRegEx │ │ │ │ │ + * {RegExp} Regular expression to test Event.type for events that complete │ │ │ │ │ + * a buttonclick sequence. │ │ │ │ │ + */ │ │ │ │ │ + completeRegEx: /^mouseup|touchend$/, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: startEvt │ │ │ │ │ + * {Event} The event that started the click sequence │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Events.buttonclick │ │ │ │ │ + * Construct a buttonclick event type. Applications are not supposed to │ │ │ │ │ + * create instances of this class - they are created on demand by │ │ │ │ │ + * <OpenLayers.Events> instances. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * target - {<OpenLayers.Events>} The events instance that the buttonclick │ │ │ │ │ + * event will be triggered on. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(target) { │ │ │ │ │ + this.target = target; │ │ │ │ │ + for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ + this.target.register(this.events[i], this, this.buttonClick, { │ │ │ │ │ + extension: true │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ + this.target.unregister(this.events[i], this, this.buttonClick); │ │ │ │ │ + } │ │ │ │ │ + delete this.target; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getPressedButton │ │ │ │ │ + * Get the pressed button, if any. Returns undefined if no button │ │ │ │ │ + * was pressed. │ │ │ │ │ + * │ │ │ │ │ + * Arguments: │ │ │ │ │ + * element - {DOMElement} The event target. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The button element, or undefined. │ │ │ │ │ + */ │ │ │ │ │ + getPressedButton: function(element) { │ │ │ │ │ + var depth = 3, // limit the search depth │ │ │ │ │ + button; │ │ │ │ │ + do { │ │ │ │ │ + if (OpenLayers.Element.hasClass(element, "olButton")) { │ │ │ │ │ + // hit! │ │ │ │ │ + button = element; │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - if (capabilities.service === undefined) { │ │ │ │ │ - // an exception must have occurred, so parse it │ │ │ │ │ - var parser = new OpenLayers.Format.OGCExceptionReport(); │ │ │ │ │ - capabilities.error = parser.read(raw); │ │ │ │ │ + element = element.parentNode; │ │ │ │ │ + } while (--depth > 0 && element); │ │ │ │ │ + return button; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: ignore │ │ │ │ │ + * Check for event target elements that should be ignored by OpenLayers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * element - {DOMElement} The event target. │ │ │ │ │ + */ │ │ │ │ │ + ignore: function(element) { │ │ │ │ │ + var depth = 3, │ │ │ │ │ + ignore = false; │ │ │ │ │ + do { │ │ │ │ │ + if (element.nodeName.toLowerCase() === 'a') { │ │ │ │ │ + ignore = true; │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ - return capabilities; │ │ │ │ │ - }, │ │ │ │ │ + element = element.parentNode; │ │ │ │ │ + } while (--depth > 0 && element); │ │ │ │ │ + return ignore; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wms": { │ │ │ │ │ - "Service": function(node, obj) { │ │ │ │ │ - obj.service = {}; │ │ │ │ │ - this.readChildNodes(node, obj.service); │ │ │ │ │ - }, │ │ │ │ │ - "Name": function(node, obj) { │ │ │ │ │ - obj.name = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Title": function(node, obj) { │ │ │ │ │ - obj.title = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Abstract": function(node, obj) { │ │ │ │ │ - obj["abstract"] = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "BoundingBox": function(node, obj) { │ │ │ │ │ - var bbox = {}; │ │ │ │ │ - bbox.bbox = [ │ │ │ │ │ - parseFloat(node.getAttribute("minx")), │ │ │ │ │ - parseFloat(node.getAttribute("miny")), │ │ │ │ │ - parseFloat(node.getAttribute("maxx")), │ │ │ │ │ - parseFloat(node.getAttribute("maxy")) │ │ │ │ │ - ]; │ │ │ │ │ - var res = { │ │ │ │ │ - x: parseFloat(node.getAttribute("resx")), │ │ │ │ │ - y: parseFloat(node.getAttribute("resy")) │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - if (!(isNaN(res.x) && isNaN(res.y))) { │ │ │ │ │ - bbox.res = res; │ │ │ │ │ - } │ │ │ │ │ - // return the bbox so that descendant classes can set the │ │ │ │ │ - // CRS and SRS and add it to the obj │ │ │ │ │ - return bbox; │ │ │ │ │ - }, │ │ │ │ │ - "OnlineResource": function(node, obj) { │ │ │ │ │ - obj.href = this.getAttributeNS(node, this.namespaces.xlink, │ │ │ │ │ - "href"); │ │ │ │ │ - }, │ │ │ │ │ - "ContactInformation": function(node, obj) { │ │ │ │ │ - obj.contactInformation = {}; │ │ │ │ │ - this.readChildNodes(node, obj.contactInformation); │ │ │ │ │ - }, │ │ │ │ │ - "ContactPersonPrimary": function(node, obj) { │ │ │ │ │ - obj.personPrimary = {}; │ │ │ │ │ - this.readChildNodes(node, obj.personPrimary); │ │ │ │ │ - }, │ │ │ │ │ - "ContactPerson": function(node, obj) { │ │ │ │ │ - obj.person = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "ContactOrganization": function(node, obj) { │ │ │ │ │ - obj.organization = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "ContactPosition": function(node, obj) { │ │ │ │ │ - obj.position = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "ContactAddress": function(node, obj) { │ │ │ │ │ - obj.contactAddress = {}; │ │ │ │ │ - this.readChildNodes(node, obj.contactAddress); │ │ │ │ │ - }, │ │ │ │ │ - "AddressType": function(node, obj) { │ │ │ │ │ - obj.type = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Address": function(node, obj) { │ │ │ │ │ - obj.address = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "City": function(node, obj) { │ │ │ │ │ - obj.city = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "StateOrProvince": function(node, obj) { │ │ │ │ │ - obj.stateOrProvince = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "PostCode": function(node, obj) { │ │ │ │ │ - obj.postcode = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Country": function(node, obj) { │ │ │ │ │ - obj.country = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "ContactVoiceTelephone": function(node, obj) { │ │ │ │ │ - obj.phone = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "ContactFacsimileTelephone": function(node, obj) { │ │ │ │ │ - obj.fax = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "ContactElectronicMailAddress": function(node, obj) { │ │ │ │ │ - obj.email = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Fees": function(node, obj) { │ │ │ │ │ - var fees = this.getChildValue(node); │ │ │ │ │ - if (fees && fees.toLowerCase() != "none") { │ │ │ │ │ - obj.fees = fees; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "AccessConstraints": function(node, obj) { │ │ │ │ │ - var constraints = this.getChildValue(node); │ │ │ │ │ - if (constraints && constraints.toLowerCase() != "none") { │ │ │ │ │ - obj.accessConstraints = constraints; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Capability": function(node, obj) { │ │ │ │ │ - obj.capability = { │ │ │ │ │ - nestedLayers: [], │ │ │ │ │ - layers: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.capability); │ │ │ │ │ - }, │ │ │ │ │ - "Request": function(node, obj) { │ │ │ │ │ - obj.request = {}; │ │ │ │ │ - this.readChildNodes(node, obj.request); │ │ │ │ │ - }, │ │ │ │ │ - "GetCapabilities": function(node, obj) { │ │ │ │ │ - obj.getcapabilities = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getcapabilities); │ │ │ │ │ - }, │ │ │ │ │ - "Format": function(node, obj) { │ │ │ │ │ - if (OpenLayers.Util.isArray(obj.formats)) { │ │ │ │ │ - obj.formats.push(this.getChildValue(node)); │ │ │ │ │ - } else { │ │ │ │ │ - obj.format = this.getChildValue(node); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "DCPType": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "HTTP": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "Get": function(node, obj) { │ │ │ │ │ - obj.get = {}; │ │ │ │ │ - this.readChildNodes(node, obj.get); │ │ │ │ │ - // backwards compatibility │ │ │ │ │ - if (!obj.href) { │ │ │ │ │ - obj.href = obj.get.href; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Post": function(node, obj) { │ │ │ │ │ - obj.post = {}; │ │ │ │ │ - this.readChildNodes(node, obj.post); │ │ │ │ │ - // backwards compatibility │ │ │ │ │ - if (!obj.href) { │ │ │ │ │ - obj.href = obj.get.href; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "GetMap": function(node, obj) { │ │ │ │ │ - obj.getmap = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getmap); │ │ │ │ │ - }, │ │ │ │ │ - "GetFeatureInfo": function(node, obj) { │ │ │ │ │ - obj.getfeatureinfo = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getfeatureinfo); │ │ │ │ │ - }, │ │ │ │ │ - "Exception": function(node, obj) { │ │ │ │ │ - obj.exception = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.exception); │ │ │ │ │ - }, │ │ │ │ │ - "Layer": function(node, obj) { │ │ │ │ │ - var parentLayer, capability; │ │ │ │ │ - if (obj.capability) { │ │ │ │ │ - capability = obj.capability; │ │ │ │ │ - parentLayer = obj; │ │ │ │ │ - } else { │ │ │ │ │ - capability = obj; │ │ │ │ │ - } │ │ │ │ │ - var attrNode = node.getAttributeNode("queryable"); │ │ │ │ │ - var queryable = (attrNode && attrNode.specified) ? │ │ │ │ │ - node.getAttribute("queryable") : null; │ │ │ │ │ - attrNode = node.getAttributeNode("cascaded"); │ │ │ │ │ - var cascaded = (attrNode && attrNode.specified) ? │ │ │ │ │ - node.getAttribute("cascaded") : null; │ │ │ │ │ - attrNode = node.getAttributeNode("opaque"); │ │ │ │ │ - var opaque = (attrNode && attrNode.specified) ? │ │ │ │ │ - node.getAttribute('opaque') : null; │ │ │ │ │ - var noSubsets = node.getAttribute('noSubsets'); │ │ │ │ │ - var fixedWidth = node.getAttribute('fixedWidth'); │ │ │ │ │ - var fixedHeight = node.getAttribute('fixedHeight'); │ │ │ │ │ - var parent = parentLayer || {}, │ │ │ │ │ - extend = OpenLayers.Util.extend; │ │ │ │ │ - var layer = { │ │ │ │ │ - nestedLayers: [], │ │ │ │ │ - styles: parentLayer ? [].concat(parentLayer.styles) : [], │ │ │ │ │ - srs: parentLayer ? extend({}, parent.srs) : {}, │ │ │ │ │ - metadataURLs: [], │ │ │ │ │ - bbox: parentLayer ? extend({}, parent.bbox) : {}, │ │ │ │ │ - llbbox: parent.llbbox, │ │ │ │ │ - dimensions: parentLayer ? extend({}, parent.dimensions) : {}, │ │ │ │ │ - authorityURLs: parentLayer ? extend({}, parent.authorityURLs) : {}, │ │ │ │ │ - identifiers: {}, │ │ │ │ │ - keywords: [], │ │ │ │ │ - queryable: (queryable && queryable !== "") ? │ │ │ │ │ - (queryable === "1" || queryable === "true") : (parent.queryable || false), │ │ │ │ │ - cascaded: (cascaded !== null) ? parseInt(cascaded) : (parent.cascaded || 0), │ │ │ │ │ - opaque: opaque ? │ │ │ │ │ - (opaque === "1" || opaque === "true") : (parent.opaque || false), │ │ │ │ │ - noSubsets: (noSubsets !== null) ? │ │ │ │ │ - (noSubsets === "1" || noSubsets === "true") : (parent.noSubsets || false), │ │ │ │ │ - fixedWidth: (fixedWidth != null) ? │ │ │ │ │ - parseInt(fixedWidth) : (parent.fixedWidth || 0), │ │ │ │ │ - fixedHeight: (fixedHeight != null) ? │ │ │ │ │ - parseInt(fixedHeight) : (parent.fixedHeight || 0), │ │ │ │ │ - minScale: parent.minScale, │ │ │ │ │ - maxScale: parent.maxScale, │ │ │ │ │ - attribution: parent.attribution │ │ │ │ │ - }; │ │ │ │ │ - obj.nestedLayers.push(layer); │ │ │ │ │ - layer.capability = capability; │ │ │ │ │ - this.readChildNodes(node, layer); │ │ │ │ │ - delete layer.capability; │ │ │ │ │ - if (layer.name) { │ │ │ │ │ - var parts = layer.name.split(":"), │ │ │ │ │ - request = capability.request, │ │ │ │ │ - gfi = request.getfeatureinfo; │ │ │ │ │ - if (parts.length > 0) { │ │ │ │ │ - layer.prefix = parts[0]; │ │ │ │ │ - } │ │ │ │ │ - capability.layers.push(layer); │ │ │ │ │ - if (layer.formats === undefined) { │ │ │ │ │ - layer.formats = request.getmap.formats; │ │ │ │ │ - } │ │ │ │ │ - if (layer.infoFormats === undefined && gfi) { │ │ │ │ │ - layer.infoFormats = gfi.formats; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Attribution": function(node, obj) { │ │ │ │ │ - obj.attribution = {}; │ │ │ │ │ - this.readChildNodes(node, obj.attribution); │ │ │ │ │ - }, │ │ │ │ │ - "LogoURL": function(node, obj) { │ │ │ │ │ - obj.logo = { │ │ │ │ │ - width: node.getAttribute("width"), │ │ │ │ │ - height: node.getAttribute("height") │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.logo); │ │ │ │ │ - }, │ │ │ │ │ - "Style": function(node, obj) { │ │ │ │ │ - var style = {}; │ │ │ │ │ - obj.styles.push(style); │ │ │ │ │ - this.readChildNodes(node, style); │ │ │ │ │ - }, │ │ │ │ │ - "LegendURL": function(node, obj) { │ │ │ │ │ - var legend = { │ │ │ │ │ - width: node.getAttribute("width"), │ │ │ │ │ - height: node.getAttribute("height") │ │ │ │ │ - }; │ │ │ │ │ - obj.legend = legend; │ │ │ │ │ - this.readChildNodes(node, legend); │ │ │ │ │ - }, │ │ │ │ │ - "MetadataURL": function(node, obj) { │ │ │ │ │ - var metadataURL = { │ │ │ │ │ - type: node.getAttribute("type") │ │ │ │ │ - }; │ │ │ │ │ - obj.metadataURLs.push(metadataURL); │ │ │ │ │ - this.readChildNodes(node, metadataURL); │ │ │ │ │ - }, │ │ │ │ │ - "DataURL": function(node, obj) { │ │ │ │ │ - obj.dataURL = {}; │ │ │ │ │ - this.readChildNodes(node, obj.dataURL); │ │ │ │ │ - }, │ │ │ │ │ - "FeatureListURL": function(node, obj) { │ │ │ │ │ - obj.featureListURL = {}; │ │ │ │ │ - this.readChildNodes(node, obj.featureListURL); │ │ │ │ │ - }, │ │ │ │ │ - "AuthorityURL": function(node, obj) { │ │ │ │ │ - var name = node.getAttribute("name"); │ │ │ │ │ - var authority = {}; │ │ │ │ │ - this.readChildNodes(node, authority); │ │ │ │ │ - obj.authorityURLs[name] = authority.href; │ │ │ │ │ - }, │ │ │ │ │ - "Identifier": function(node, obj) { │ │ │ │ │ - var authority = node.getAttribute("authority"); │ │ │ │ │ - obj.identifiers[authority] = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "KeywordList": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "SRS": function(node, obj) { │ │ │ │ │ - obj.srs[this.getChildValue(node)] = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSCapabilities/v1_1.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WMSCapabilities/v1.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WMSCapabilities.v1_1 │ │ │ │ │ - * Abstract class not to be instantiated directly. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMSCapabilities.v1> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_1 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WMSCapabilities.v1, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "WMT_MS_Capabilities": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "Keyword": function(node, obj) { │ │ │ │ │ - if (obj.keywords) { │ │ │ │ │ - obj.keywords.push(this.getChildValue(node)); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "DescribeLayer": function(node, obj) { │ │ │ │ │ - obj.describelayer = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.describelayer); │ │ │ │ │ - }, │ │ │ │ │ - "GetLegendGraphic": function(node, obj) { │ │ │ │ │ - obj.getlegendgraphic = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getlegendgraphic); │ │ │ │ │ - }, │ │ │ │ │ - "GetStyles": function(node, obj) { │ │ │ │ │ - obj.getstyles = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getstyles); │ │ │ │ │ - }, │ │ │ │ │ - "PutStyles": function(node, obj) { │ │ │ │ │ - obj.putstyles = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.putstyles); │ │ │ │ │ - }, │ │ │ │ │ - "UserDefinedSymbolization": function(node, obj) { │ │ │ │ │ - var userSymbols = { │ │ │ │ │ - supportSLD: parseInt(node.getAttribute("SupportSLD")) == 1, │ │ │ │ │ - userLayer: parseInt(node.getAttribute("UserLayer")) == 1, │ │ │ │ │ - userStyle: parseInt(node.getAttribute("UserStyle")) == 1, │ │ │ │ │ - remoteWFS: parseInt(node.getAttribute("RemoteWFS")) == 1 │ │ │ │ │ - }; │ │ │ │ │ - obj.userSymbols = userSymbols; │ │ │ │ │ - }, │ │ │ │ │ - "LatLonBoundingBox": function(node, obj) { │ │ │ │ │ - obj.llbbox = [ │ │ │ │ │ - parseFloat(node.getAttribute("minx")), │ │ │ │ │ - parseFloat(node.getAttribute("miny")), │ │ │ │ │ - parseFloat(node.getAttribute("maxx")), │ │ │ │ │ - parseFloat(node.getAttribute("maxy")) │ │ │ │ │ - ]; │ │ │ │ │ - }, │ │ │ │ │ - "BoundingBox": function(node, obj) { │ │ │ │ │ - var bbox = OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"].BoundingBox.apply(this, [node, obj]); │ │ │ │ │ - bbox.srs = node.getAttribute("SRS"); │ │ │ │ │ - obj.bbox[bbox.srs] = bbox; │ │ │ │ │ - }, │ │ │ │ │ - "ScaleHint": function(node, obj) { │ │ │ │ │ - var min = node.getAttribute("min"); │ │ │ │ │ - var max = node.getAttribute("max"); │ │ │ │ │ - var rad2 = Math.pow(2, 0.5); │ │ │ │ │ - var ipm = OpenLayers.INCHES_PER_UNIT["m"]; │ │ │ │ │ - if (min != 0) { │ │ │ │ │ - obj.maxScale = parseFloat( │ │ │ │ │ - ((min / rad2) * ipm * │ │ │ │ │ - OpenLayers.DOTS_PER_INCH).toPrecision(13) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (max != Number.POSITIVE_INFINITY) { │ │ │ │ │ - obj.minScale = parseFloat( │ │ │ │ │ - ((max / rad2) * ipm * │ │ │ │ │ - OpenLayers.DOTS_PER_INCH).toPrecision(13) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Dimension": function(node, obj) { │ │ │ │ │ - var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ - var dim = { │ │ │ │ │ - name: name, │ │ │ │ │ - units: node.getAttribute("units"), │ │ │ │ │ - unitsymbol: node.getAttribute("unitSymbol") │ │ │ │ │ - }; │ │ │ │ │ - obj.dimensions[dim.name] = dim; │ │ │ │ │ - }, │ │ │ │ │ - "Extent": function(node, obj) { │ │ │ │ │ - var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ - if (name in obj["dimensions"]) { │ │ │ │ │ - var extent = obj.dimensions[name]; │ │ │ │ │ - extent.nearestVal = │ │ │ │ │ - node.getAttribute("nearestValue") === "1"; │ │ │ │ │ - extent.multipleVal = │ │ │ │ │ - node.getAttribute("multipleValues") === "1"; │ │ │ │ │ - extent.current = node.getAttribute("current") === "1"; │ │ │ │ │ - extent["default"] = node.getAttribute("default") || ""; │ │ │ │ │ - var values = this.getChildValue(node); │ │ │ │ │ - extent.values = values.split(","); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"]) │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSCapabilities/v1_1_1.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WMSCapabilities/v1_1.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WMSCapabilities/v1_1_1 │ │ │ │ │ - * Read WMS Capabilities version 1.1.1. │ │ │ │ │ - * │ │ │ │ │ - * Note on <ScaleHint> parsing: If the 'min' attribute is set to "0", no │ │ │ │ │ - * maxScale will be set on the layer object. If the 'max' attribute is set to │ │ │ │ │ - * "Infinity", no minScale will be set. This makes it easy to create proper │ │ │ │ │ - * {<OpenLayers.Layer.WMS>} configurations directly from the layer object │ │ │ │ │ - * literals returned by this format, because no minScale/maxScale modifications │ │ │ │ │ - * need to be made. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMSCapabilities.v1_1> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_1_1 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WMSCapabilities.v1_1, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} The specific parser version. │ │ │ │ │ - */ │ │ │ │ │ - version: "1.1.1", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMSCapabilities.v1_1_1 │ │ │ │ │ - * Create a new parser for WMS capabilities version 1.1.1. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "SRS": function(node, obj) { │ │ │ │ │ - obj.srs[this.getChildValue(node)] = true; │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1_1.prototype.readers["wms"]) │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_1" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSCapabilities/v1_3.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WMSCapabilities/v1.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WMSCapabilities/v1_3 │ │ │ │ │ - * Abstract base class for WMS Capabilities version 1.3.X. │ │ │ │ │ - * SLD 1.1.0 adds in the extra operations DescribeLayer and GetLegendGraphic, │ │ │ │ │ - * see: http://schemas.opengis.net/sld/1.1.0/sld_capabilities.xsd │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMSCapabilities.v1> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_3 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WMSCapabilities.v1, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "WMS_Capabilities": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "LayerLimit": function(node, obj) { │ │ │ │ │ - obj.layerLimit = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "MaxWidth": function(node, obj) { │ │ │ │ │ - obj.maxWidth = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "MaxHeight": function(node, obj) { │ │ │ │ │ - obj.maxHeight = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "BoundingBox": function(node, obj) { │ │ │ │ │ - var bbox = OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"].BoundingBox.apply(this, [node, obj]); │ │ │ │ │ - bbox.srs = node.getAttribute("CRS"); │ │ │ │ │ - obj.bbox[bbox.srs] = bbox; │ │ │ │ │ - }, │ │ │ │ │ - "CRS": function(node, obj) { │ │ │ │ │ - // CRS is the synonym of SRS │ │ │ │ │ - this.readers.wms.SRS.apply(this, [node, obj]); │ │ │ │ │ - }, │ │ │ │ │ - "EX_GeographicBoundingBox": function(node, obj) { │ │ │ │ │ - // replacement of LatLonBoundingBox │ │ │ │ │ - obj.llbbox = []; │ │ │ │ │ - this.readChildNodes(node, obj.llbbox); │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - "westBoundLongitude": function(node, obj) { │ │ │ │ │ - obj[0] = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "eastBoundLongitude": function(node, obj) { │ │ │ │ │ - obj[2] = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "southBoundLatitude": function(node, obj) { │ │ │ │ │ - obj[1] = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "northBoundLatitude": function(node, obj) { │ │ │ │ │ - obj[3] = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "MinScaleDenominator": function(node, obj) { │ │ │ │ │ - obj.maxScale = parseFloat(this.getChildValue(node)).toPrecision(16); │ │ │ │ │ - }, │ │ │ │ │ - "MaxScaleDenominator": function(node, obj) { │ │ │ │ │ - obj.minScale = parseFloat(this.getChildValue(node)).toPrecision(16); │ │ │ │ │ - }, │ │ │ │ │ - "Dimension": function(node, obj) { │ │ │ │ │ - // dimension has extra attributes: default, multipleValues, │ │ │ │ │ - // nearestValue, current which used to be part of Extent. It now │ │ │ │ │ - // also contains the values. │ │ │ │ │ - var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ - var dim = { │ │ │ │ │ - name: name, │ │ │ │ │ - units: node.getAttribute("units"), │ │ │ │ │ - unitsymbol: node.getAttribute("unitSymbol"), │ │ │ │ │ - nearestVal: node.getAttribute("nearestValue") === "1", │ │ │ │ │ - multipleVal: node.getAttribute("multipleValues") === "1", │ │ │ │ │ - "default": node.getAttribute("default") || "", │ │ │ │ │ - current: node.getAttribute("current") === "1", │ │ │ │ │ - values: this.getChildValue(node).split(",") │ │ │ │ │ - │ │ │ │ │ - }; │ │ │ │ │ - // Theoretically there can be more dimensions with the same │ │ │ │ │ - // name, but with a different unit. Until we meet such a case, │ │ │ │ │ - // let's just keep the same structure as the WMS 1.1 │ │ │ │ │ - // GetCapabilities parser uses. We will store the last │ │ │ │ │ - // one encountered. │ │ │ │ │ - obj.dimensions[dim.name] = dim; │ │ │ │ │ - }, │ │ │ │ │ - "Keyword": function(node, obj) { │ │ │ │ │ - // TODO: should we change the structure of keyword in v1.js? │ │ │ │ │ - // Make it an object with a value instead of a string? │ │ │ │ │ - var keyword = { │ │ │ │ │ - value: this.getChildValue(node), │ │ │ │ │ - vocabulary: node.getAttribute("vocabulary") │ │ │ │ │ - }; │ │ │ │ │ - if (obj.keywords) { │ │ │ │ │ - obj.keywords.push(keyword); │ │ │ │ │ + /** │ │ │ │ │ + * Method: buttonClick │ │ │ │ │ + * Check if a button was clicked, and fire the buttonclick event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + buttonClick: function(evt) { │ │ │ │ │ + var propagate = true, │ │ │ │ │ + element = OpenLayers.Event.element(evt); │ │ │ │ │ + if (element && (OpenLayers.Event.isLeftClick(evt) || !~evt.type.indexOf("mouse"))) { │ │ │ │ │ + // was a button pressed? │ │ │ │ │ + var button = this.getPressedButton(element); │ │ │ │ │ + if (button) { │ │ │ │ │ + if (evt.type === "keydown") { │ │ │ │ │ + switch (evt.keyCode) { │ │ │ │ │ + case OpenLayers.Event.KEY_RETURN: │ │ │ │ │ + case OpenLayers.Event.KEY_SPACE: │ │ │ │ │ + this.target.triggerEvent("buttonclick", { │ │ │ │ │ + buttonElement: button │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false; │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"]), │ │ │ │ │ - "sld": { │ │ │ │ │ - "UserDefinedSymbolization": function(node, obj) { │ │ │ │ │ - this.readers.wms.UserDefinedSymbolization.apply(this, [node, obj]); │ │ │ │ │ - // add the two extra attributes │ │ │ │ │ - obj.userSymbols.inlineFeature = parseInt(node.getAttribute("InlineFeature")) == 1; │ │ │ │ │ - obj.userSymbols.remoteWCS = parseInt(node.getAttribute("RemoteWCS")) == 1; │ │ │ │ │ - }, │ │ │ │ │ - "DescribeLayer": function(node, obj) { │ │ │ │ │ - this.readers.wms.DescribeLayer.apply(this, [node, obj]); │ │ │ │ │ - }, │ │ │ │ │ - "GetLegendGraphic": function(node, obj) { │ │ │ │ │ - this.readers.wms.GetLegendGraphic.apply(this, [node, obj]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_3" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSCapabilities/v1_3_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WMSCapabilities/v1_3.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WMSCapabilities/v1_3_0 │ │ │ │ │ - * Read WMS Capabilities version 1.3.0. │ │ │ │ │ - * SLD 1.1.0 adds in the extra operations DescribeLayer and GetLegendGraphic, │ │ │ │ │ - * see: http://schemas.opengis.net/sld/1.1.0/sld_capabilities.xsd │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMSCapabilities.v1_3> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_3_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WMSCapabilities.v1_3, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} The specific parser version. │ │ │ │ │ - */ │ │ │ │ │ - version: "1.3.0", │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_3_0" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSCapabilities/v1_1_1_WMSC.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WMSCapabilities/v1_1_1.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WMSCapabilities/v1_1_1_WMSC │ │ │ │ │ - * Read WMS-C Capabilities version 1.1.1. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMSCapabilities.v1_1_1> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_1_1_WMSC = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WMSCapabilities.v1_1_1, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} The specific parser version. │ │ │ │ │ - */ │ │ │ │ │ - version: "1.1.1", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: profile │ │ │ │ │ - * {String} The specific profile │ │ │ │ │ - */ │ │ │ │ │ - profile: "WMSC", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMSCapabilities.v1_1_1 │ │ │ │ │ - * Create a new parser for WMS-C capabilities version 1.1.1. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + } else if (this.startEvt) { │ │ │ │ │ + if (this.completeRegEx.test(evt.type)) { │ │ │ │ │ + var pos = OpenLayers.Util.pagePosition(button); │ │ │ │ │ + var viewportElement = OpenLayers.Util.getViewportElement(); │ │ │ │ │ + var scrollTop = window.pageYOffset || viewportElement.scrollTop; │ │ │ │ │ + var scrollLeft = window.pageXOffset || viewportElement.scrollLeft; │ │ │ │ │ + pos[0] = pos[0] - scrollLeft; │ │ │ │ │ + pos[1] = pos[1] - scrollTop; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "VendorSpecificCapabilities": function(node, obj) { │ │ │ │ │ - obj.vendorSpecific = { │ │ │ │ │ - tileSets: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.vendorSpecific); │ │ │ │ │ - }, │ │ │ │ │ - "TileSet": function(node, vendorSpecific) { │ │ │ │ │ - var tileset = { │ │ │ │ │ - srs: {}, │ │ │ │ │ - bbox: {}, │ │ │ │ │ - resolutions: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, tileset); │ │ │ │ │ - vendorSpecific.tileSets.push(tileset); │ │ │ │ │ - }, │ │ │ │ │ - "Resolutions": function(node, tileset) { │ │ │ │ │ - var res = this.getChildValue(node).split(" "); │ │ │ │ │ - for (var i = 0, len = res.length; i < len; i++) { │ │ │ │ │ - if (res[i] != "") { │ │ │ │ │ - tileset.resolutions.push(parseFloat(res[i])); │ │ │ │ │ - } │ │ │ │ │ + this.target.triggerEvent("buttonclick", { │ │ │ │ │ + buttonElement: button, │ │ │ │ │ + buttonXY: { │ │ │ │ │ + x: this.startEvt.clientX - pos[0], │ │ │ │ │ + y: this.startEvt.clientY - pos[1] │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "Width": function(node, tileset) { │ │ │ │ │ - tileset.width = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "Height": function(node, tileset) { │ │ │ │ │ - tileset.height = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "Layers": function(node, tileset) { │ │ │ │ │ - tileset.layers = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Styles": function(node, tileset) { │ │ │ │ │ - tileset.styles = this.getChildValue(node); │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1_1_1.prototype.readers["wms"]) │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_1_WMSC" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSCapabilities/v1_1_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WMSCapabilities/v1_1.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WMSCapabilities/v1_1_0 │ │ │ │ │ - * Read WMS Capabilities version 1.1.0. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMSCapabilities.v1_1> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_1_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WMSCapabilities.v1_1, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} The specific parser version. │ │ │ │ │ - */ │ │ │ │ │ - version: "1.1.0", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMSCapabilities.v1_1_0 │ │ │ │ │ - * Create a new parser for WMS capabilities version 1.1.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "SRS": function(node, obj) { │ │ │ │ │ - var srs = this.getChildValue(node); │ │ │ │ │ - var values = srs.split(/ +/); │ │ │ │ │ - for (var i = 0, len = values.length; i < len; i++) { │ │ │ │ │ - obj.srs[values[i]] = true; │ │ │ │ │ + if (this.cancelRegEx.test(evt.type)) { │ │ │ │ │ + delete this.startEvt; │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false; │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1_1.prototype.readers["wms"]) │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_0" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WFSCapabilities/v1.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WFSCapabilities.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WFSCapabilities.v1 │ │ │ │ │ - * Abstract class not to be instantiated directly. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WFSCapabilities.v1 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.XML, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - wfs: "http://www.opengis.net/wfs", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - ows: "http://www.opengis.net/ows" │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: errorProperty │ │ │ │ │ - * {String} Which property of the returned object to check for in order to │ │ │ │ │ - * determine whether or not parsing has failed. In the case that the │ │ │ │ │ - * errorProperty is undefined on the returned object, the document will be │ │ │ │ │ - * run through an OGCExceptionReport parser. │ │ │ │ │ - */ │ │ │ │ │ - errorProperty: "featureTypeList", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - */ │ │ │ │ │ - defaultPrefix: "wfs", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WFSCapabilities.v1_1 │ │ │ │ │ - * Create an instance of one of the subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} List of named layers. │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - var raw = data; │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ - } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - return capabilities; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wfs": { │ │ │ │ │ - "WFS_Capabilities": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "FeatureTypeList": function(node, request) { │ │ │ │ │ - request.featureTypeList = { │ │ │ │ │ - featureTypes: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, request.featureTypeList); │ │ │ │ │ - }, │ │ │ │ │ - "FeatureType": function(node, featureTypeList) { │ │ │ │ │ - var featureType = {}; │ │ │ │ │ - this.readChildNodes(node, featureType); │ │ │ │ │ - featureTypeList.featureTypes.push(featureType); │ │ │ │ │ - }, │ │ │ │ │ - "Name": function(node, obj) { │ │ │ │ │ - var name = this.getChildValue(node); │ │ │ │ │ - if (name) { │ │ │ │ │ - var parts = name.split(":"); │ │ │ │ │ - obj.name = parts.pop(); │ │ │ │ │ - if (parts.length > 0) { │ │ │ │ │ - obj.featureNS = this.lookupNamespaceURI(node, parts[0]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Title": function(node, obj) { │ │ │ │ │ - var title = this.getChildValue(node); │ │ │ │ │ - if (title) { │ │ │ │ │ - obj.title = title; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Abstract": function(node, obj) { │ │ │ │ │ - var abst = this.getChildValue(node); │ │ │ │ │ - if (abst) { │ │ │ │ │ - obj["abstract"] = abst; │ │ │ │ │ - } │ │ │ │ │ + if (this.startRegEx.test(evt.type)) { │ │ │ │ │ + this.startEvt = evt; │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false; │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + propagate = !this.ignore(OpenLayers.Event.element(evt)); │ │ │ │ │ + delete this.startEvt; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WFSCapabilities/v1_0_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WFSCapabilities/v1.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WFSCapabilities/v1_0_0 │ │ │ │ │ - * Read WFS Capabilities version 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WFSCapabilities.v1> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WFSCapabilities.v1_0_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WFSCapabilities.v1, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WFSCapabilities.v1_0_0 │ │ │ │ │ - * Create a new parser for WFS capabilities version 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "Service": function(node, capabilities) { │ │ │ │ │ - capabilities.service = {}; │ │ │ │ │ - this.readChildNodes(node, capabilities.service); │ │ │ │ │ - }, │ │ │ │ │ - "Fees": function(node, service) { │ │ │ │ │ - var fees = this.getChildValue(node); │ │ │ │ │ - if (fees && fees.toLowerCase() != "none") { │ │ │ │ │ - service.fees = fees; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "AccessConstraints": function(node, service) { │ │ │ │ │ - var constraints = this.getChildValue(node); │ │ │ │ │ - if (constraints && constraints.toLowerCase() != "none") { │ │ │ │ │ - service.accessConstraints = constraints; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "OnlineResource": function(node, service) { │ │ │ │ │ - var onlineResource = this.getChildValue(node); │ │ │ │ │ - if (onlineResource && onlineResource.toLowerCase() != "none") { │ │ │ │ │ - service.onlineResource = onlineResource; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Keywords": function(node, service) { │ │ │ │ │ - var keywords = this.getChildValue(node); │ │ │ │ │ - if (keywords && keywords.toLowerCase() != "none") { │ │ │ │ │ - service.keywords = keywords.split(', '); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Capability": function(node, capabilities) { │ │ │ │ │ - capabilities.capability = {}; │ │ │ │ │ - this.readChildNodes(node, capabilities.capability); │ │ │ │ │ - }, │ │ │ │ │ - "Request": function(node, obj) { │ │ │ │ │ - obj.request = {}; │ │ │ │ │ - this.readChildNodes(node, obj.request); │ │ │ │ │ - }, │ │ │ │ │ - "GetFeature": function(node, request) { │ │ │ │ │ - request.getfeature = { │ │ │ │ │ - href: {}, // DCPType │ │ │ │ │ - formats: [] // ResultFormat │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, request.getfeature); │ │ │ │ │ - }, │ │ │ │ │ - "ResultFormat": function(node, obj) { │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - var childNode; │ │ │ │ │ - for (var i = 0; i < children.length; i++) { │ │ │ │ │ - childNode = children[i]; │ │ │ │ │ - if (childNode.nodeType == 1) { │ │ │ │ │ - obj.formats.push(childNode.nodeName); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "DCPType": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "HTTP": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj.href); │ │ │ │ │ - }, │ │ │ │ │ - "Get": function(node, obj) { │ │ │ │ │ - obj.get = node.getAttribute("onlineResource"); │ │ │ │ │ - }, │ │ │ │ │ - "Post": function(node, obj) { │ │ │ │ │ - obj.post = node.getAttribute("onlineResource"); │ │ │ │ │ - }, │ │ │ │ │ - "SRS": function(node, obj) { │ │ │ │ │ - var srs = this.getChildValue(node); │ │ │ │ │ - if (srs) { │ │ │ │ │ - obj.srs = srs; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WFSCapabilities.v1.prototype.readers["wfs"]) │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1_0_0" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WFSCapabilities/v1_1_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WFSCapabilities/v1.js │ │ │ │ │ - * @requires OpenLayers/Format/OWSCommon/v1.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WFSCapabilities/v1_1_0 │ │ │ │ │ - * Read WFS Capabilities version 1.1.0. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WFSCapabilities> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WFSCapabilities.v1_1_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WFSCapabilities.v1, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ - */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WFSCapabilities.v1_1_0 │ │ │ │ │ - * Create a new parser for WFS capabilities version 1.1.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "DefaultSRS": function(node, obj) { │ │ │ │ │ - var defaultSRS = this.getChildValue(node); │ │ │ │ │ - if (defaultSRS) { │ │ │ │ │ - obj.srs = defaultSRS; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WFSCapabilities.v1.prototype.readers["wfs"]), │ │ │ │ │ - "ows": OpenLayers.Format.OWSCommon.v1.prototype.readers.ows │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1_1_0" │ │ │ │ │ + } │ │ │ │ │ + return propagate; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Events/featureclick.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ @@ -59879,6316 +60241,4865 @@ │ │ │ │ │ * Extension event type for handling leaving a feature. │ │ │ │ │ * │ │ │ │ │ * Event types provided by this extension: │ │ │ │ │ * - featureout │ │ │ │ │ */ │ │ │ │ │ OpenLayers.Events.featureout = OpenLayers.Events.featureclick; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Events/buttonclick.js │ │ │ │ │ + OpenLayers/Control/ZoomBox.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Handler/Box.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Events.buttonclick │ │ │ │ │ - * Extension event type for handling buttons on top of a dom element. This │ │ │ │ │ - * event type fires "buttonclick" on its <target> when a button was │ │ │ │ │ - * clicked. Buttons are detected by the "olButton" class. │ │ │ │ │ - * │ │ │ │ │ - * This event type makes sure that button clicks do not interfere with other │ │ │ │ │ - * events that are registered on the same <element>. │ │ │ │ │ + * Class: OpenLayers.Control.ZoomBox │ │ │ │ │ + * The ZoomBox control enables zooming directly to a given extent, by drawing │ │ │ │ │ + * a box on the map. The box is drawn by holding down shift, whilst dragging │ │ │ │ │ + * the mouse. │ │ │ │ │ * │ │ │ │ │ - * Event types provided by this extension: │ │ │ │ │ - * - *buttonclick* Triggered when a button is clicked. Listeners receive an │ │ │ │ │ - * object with a *buttonElement* property referencing the dom element of │ │ │ │ │ - * the clicked button, and an *buttonXY* property with the click position │ │ │ │ │ - * relative to the button. │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Events.buttonclick = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: target │ │ │ │ │ - * {<OpenLayers.Events>} The events instance that the buttonclick event will │ │ │ │ │ - * be triggered on. │ │ │ │ │ - */ │ │ │ │ │ - target: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: events │ │ │ │ │ - * {Array} Events to observe and conditionally stop from propagating when │ │ │ │ │ - * an element with the olButton class (or its olAlphaImg child) is │ │ │ │ │ - * clicked. │ │ │ │ │ - */ │ │ │ │ │ - events: [ │ │ │ │ │ - 'mousedown', 'mouseup', 'click', 'dblclick', │ │ │ │ │ - 'touchstart', 'touchmove', 'touchend', 'keydown' │ │ │ │ │ - ], │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: startRegEx │ │ │ │ │ - * {RegExp} Regular expression to test Event.type for events that start │ │ │ │ │ - * a buttonclick sequence. │ │ │ │ │ - */ │ │ │ │ │ - startRegEx: /^mousedown|touchstart$/, │ │ │ │ │ - │ │ │ │ │ +OpenLayers.Control.ZoomBox = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ /** │ │ │ │ │ - * Property: cancelRegEx │ │ │ │ │ - * {RegExp} Regular expression to test Event.type for events that cancel │ │ │ │ │ - * a buttonclick sequence. │ │ │ │ │ + * Property: type │ │ │ │ │ + * {OpenLayers.Control.TYPE} │ │ │ │ │ */ │ │ │ │ │ - cancelRegEx: /^touchmove$/, │ │ │ │ │ + type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: completeRegEx │ │ │ │ │ - * {RegExp} Regular expression to test Event.type for events that complete │ │ │ │ │ - * a buttonclick sequence. │ │ │ │ │ + * Property: out │ │ │ │ │ + * {Boolean} Should the control be used for zooming out? │ │ │ │ │ */ │ │ │ │ │ - completeRegEx: /^mouseup|touchend$/, │ │ │ │ │ + out: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: startEvt │ │ │ │ │ - * {Event} The event that started the click sequence │ │ │ │ │ + * APIProperty: keyMask │ │ │ │ │ + * {Integer} Zoom only occurs if the keyMask matches the combination of │ │ │ │ │ + * keys down. Use bitwise operators and one or more of the │ │ │ │ │ + * <OpenLayers.Handler> constants to construct a keyMask. Leave null if │ │ │ │ │ + * not used mask. Default is null. │ │ │ │ │ */ │ │ │ │ │ + keyMask: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Events.buttonclick │ │ │ │ │ - * Construct a buttonclick event type. Applications are not supposed to │ │ │ │ │ - * create instances of this class - they are created on demand by │ │ │ │ │ - * <OpenLayers.Events> instances. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * target - {<OpenLayers.Events>} The events instance that the buttonclick │ │ │ │ │ - * event will be triggered on. │ │ │ │ │ + * APIProperty: alwaysZoom │ │ │ │ │ + * {Boolean} Always zoom in/out when box drawn, even if the zoom level does │ │ │ │ │ + * not change. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(target) { │ │ │ │ │ - this.target = target; │ │ │ │ │ - for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ - this.target.register(this.events[i], this, this.buttonClick, { │ │ │ │ │ - extension: true │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + alwaysZoom: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * APIProperty: zoomOnClick │ │ │ │ │ + * {Boolean} Should we zoom when no box was dragged, i.e. the user only │ │ │ │ │ + * clicked? Default is true. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ - this.target.unregister(this.events[i], this, this.buttonClick); │ │ │ │ │ - } │ │ │ │ │ - delete this.target; │ │ │ │ │ - }, │ │ │ │ │ + zoomOnClick: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getPressedButton │ │ │ │ │ - * Get the pressed button, if any. Returns undefined if no button │ │ │ │ │ - * was pressed. │ │ │ │ │ - * │ │ │ │ │ - * Arguments: │ │ │ │ │ - * element - {DOMElement} The event target. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The button element, or undefined. │ │ │ │ │ + * Method: draw │ │ │ │ │ */ │ │ │ │ │ - getPressedButton: function(element) { │ │ │ │ │ - var depth = 3, // limit the search depth │ │ │ │ │ - button; │ │ │ │ │ - do { │ │ │ │ │ - if (OpenLayers.Element.hasClass(element, "olButton")) { │ │ │ │ │ - // hit! │ │ │ │ │ - button = element; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - element = element.parentNode; │ │ │ │ │ - } while (--depth > 0 && element); │ │ │ │ │ - return button; │ │ │ │ │ + draw: function() { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Box(this, { │ │ │ │ │ + done: this.zoomBox │ │ │ │ │ + }, { │ │ │ │ │ + keyMask: this.keyMask │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: ignore │ │ │ │ │ - * Check for event target elements that should be ignored by OpenLayers. │ │ │ │ │ + * Method: zoomBox │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * element - {DOMElement} The event target. │ │ │ │ │ + * position - {<OpenLayers.Bounds>} or {<OpenLayers.Pixel>} │ │ │ │ │ */ │ │ │ │ │ - ignore: function(element) { │ │ │ │ │ - var depth = 3, │ │ │ │ │ - ignore = false; │ │ │ │ │ - do { │ │ │ │ │ - if (element.nodeName.toLowerCase() === 'a') { │ │ │ │ │ - ignore = true; │ │ │ │ │ - break; │ │ │ │ │ + zoomBox: function(position) { │ │ │ │ │ + if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ + var bounds, │ │ │ │ │ + targetCenterPx = position.getCenterPixel(); │ │ │ │ │ + if (!this.out) { │ │ │ │ │ + var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.left, │ │ │ │ │ + y: position.bottom │ │ │ │ │ + }); │ │ │ │ │ + var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.right, │ │ │ │ │ + y: position.top │ │ │ │ │ + }); │ │ │ │ │ + bounds = new OpenLayers.Bounds(minXY.lon, minXY.lat, │ │ │ │ │ + maxXY.lon, maxXY.lat); │ │ │ │ │ + } else { │ │ │ │ │ + var pixWidth = position.right - position.left; │ │ │ │ │ + var pixHeight = position.bottom - position.top; │ │ │ │ │ + var zoomFactor = Math.min((this.map.size.h / pixHeight), │ │ │ │ │ + (this.map.size.w / pixWidth)); │ │ │ │ │ + var extent = this.map.getExtent(); │ │ │ │ │ + var center = this.map.getLonLatFromPixel(targetCenterPx); │ │ │ │ │ + var xmin = center.lon - (extent.getWidth() / 2) * zoomFactor; │ │ │ │ │ + var xmax = center.lon + (extent.getWidth() / 2) * zoomFactor; │ │ │ │ │ + var ymin = center.lat - (extent.getHeight() / 2) * zoomFactor; │ │ │ │ │ + var ymax = center.lat + (extent.getHeight() / 2) * zoomFactor; │ │ │ │ │ + bounds = new OpenLayers.Bounds(xmin, ymin, xmax, ymax); │ │ │ │ │ } │ │ │ │ │ - element = element.parentNode; │ │ │ │ │ - } while (--depth > 0 && element); │ │ │ │ │ - return ignore; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buttonClick │ │ │ │ │ - * Check if a button was clicked, and fire the buttonclick event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - buttonClick: function(evt) { │ │ │ │ │ - var propagate = true, │ │ │ │ │ - element = OpenLayers.Event.element(evt); │ │ │ │ │ - if (element && (OpenLayers.Event.isLeftClick(evt) || !~evt.type.indexOf("mouse"))) { │ │ │ │ │ - // was a button pressed? │ │ │ │ │ - var button = this.getPressedButton(element); │ │ │ │ │ - if (button) { │ │ │ │ │ - if (evt.type === "keydown") { │ │ │ │ │ - switch (evt.keyCode) { │ │ │ │ │ - case OpenLayers.Event.KEY_RETURN: │ │ │ │ │ - case OpenLayers.Event.KEY_SPACE: │ │ │ │ │ - this.target.triggerEvent("buttonclick", { │ │ │ │ │ - buttonElement: button │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } else if (this.startEvt) { │ │ │ │ │ - if (this.completeRegEx.test(evt.type)) { │ │ │ │ │ - var pos = OpenLayers.Util.pagePosition(button); │ │ │ │ │ - var viewportElement = OpenLayers.Util.getViewportElement(); │ │ │ │ │ - var scrollTop = window.pageYOffset || viewportElement.scrollTop; │ │ │ │ │ - var scrollLeft = window.pageXOffset || viewportElement.scrollLeft; │ │ │ │ │ - pos[0] = pos[0] - scrollLeft; │ │ │ │ │ - pos[1] = pos[1] - scrollTop; │ │ │ │ │ - │ │ │ │ │ - this.target.triggerEvent("buttonclick", { │ │ │ │ │ - buttonElement: button, │ │ │ │ │ - buttonXY: { │ │ │ │ │ - x: this.startEvt.clientX - pos[0], │ │ │ │ │ - y: this.startEvt.clientY - pos[1] │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - if (this.cancelRegEx.test(evt.type)) { │ │ │ │ │ - delete this.startEvt; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false; │ │ │ │ │ - } │ │ │ │ │ - if (this.startRegEx.test(evt.type)) { │ │ │ │ │ - this.startEvt = evt; │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false; │ │ │ │ │ - } │ │ │ │ │ + // always zoom in/out │ │ │ │ │ + var lastZoom = this.map.getZoom(), │ │ │ │ │ + size = this.map.getSize(), │ │ │ │ │ + centerPx = { │ │ │ │ │ + x: size.w / 2, │ │ │ │ │ + y: size.h / 2 │ │ │ │ │ + }, │ │ │ │ │ + zoom = this.map.getZoomForExtent(bounds), │ │ │ │ │ + oldRes = this.map.getResolution(), │ │ │ │ │ + newRes = this.map.getResolutionForZoom(zoom); │ │ │ │ │ + if (oldRes == newRes) { │ │ │ │ │ + this.map.setCenter(this.map.getLonLatFromPixel(targetCenterPx)); │ │ │ │ │ } else { │ │ │ │ │ - propagate = !this.ignore(OpenLayers.Event.element(evt)); │ │ │ │ │ - delete this.startEvt; │ │ │ │ │ + var zoomOriginPx = { │ │ │ │ │ + x: (oldRes * targetCenterPx.x - newRes * centerPx.x) / │ │ │ │ │ + (oldRes - newRes), │ │ │ │ │ + y: (oldRes * targetCenterPx.y - newRes * centerPx.y) / │ │ │ │ │ + (oldRes - newRes) │ │ │ │ │ + }; │ │ │ │ │ + this.map.zoomTo(zoom, zoomOriginPx); │ │ │ │ │ + } │ │ │ │ │ + if (lastZoom == this.map.getZoom() && this.alwaysZoom == true) { │ │ │ │ │ + this.map.zoomTo(lastZoom + (this.out ? -1 : 1)); │ │ │ │ │ + } │ │ │ │ │ + } else if (this.zoomOnClick) { // it's a pixel │ │ │ │ │ + if (!this.out) { │ │ │ │ │ + this.map.zoomTo(this.map.getZoom() + 1, position); │ │ │ │ │ + } else { │ │ │ │ │ + this.map.zoomTo(this.map.getZoom() - 1, position); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return propagate; │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomBox" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Point.js │ │ │ │ │ + OpenLayers/Control/DragPan.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Handler/Drag.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Handler.Point │ │ │ │ │ - * Handler to draw a point on the map. Point is displayed on activation, │ │ │ │ │ - * moves on mouse move, and is finished on mouse up. The handler triggers │ │ │ │ │ - * callbacks for 'done', 'cancel', and 'modify'. The modify callback is │ │ │ │ │ - * called with each change in the sketch and will receive the latest point │ │ │ │ │ - * drawn. Create a new instance with the <OpenLayers.Handler.Point> │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Control.DragPan │ │ │ │ │ + * The DragPan control pans the map with a drag of the mouse. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.Point = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: point │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} The currently drawn point │ │ │ │ │ - */ │ │ │ │ │ - point: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} The temporary drawing layer │ │ │ │ │ - */ │ │ │ │ │ - layer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: multi │ │ │ │ │ - * {Boolean} Cast features to multi-part geometries before passing to the │ │ │ │ │ - * layer. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - multi: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: citeCompliant │ │ │ │ │ - * {Boolean} If set to true, coordinates of features drawn in a map extent │ │ │ │ │ - * crossing the date line won't exceed the world bounds. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - citeCompliant: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: mouseDown │ │ │ │ │ - * {Boolean} The mouse is down │ │ │ │ │ - */ │ │ │ │ │ - mouseDown: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: stoppedDown │ │ │ │ │ - * {Boolean} Indicate whether the last mousedown stopped the event │ │ │ │ │ - * propagation. │ │ │ │ │ - */ │ │ │ │ │ - stoppedDown: null, │ │ │ │ │ +OpenLayers.Control.DragPan = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: lastDown │ │ │ │ │ - * {<OpenLayers.Pixel>} Location of the last mouse down │ │ │ │ │ + /** │ │ │ │ │ + * Property: type │ │ │ │ │ + * {OpenLayers.Control.TYPES} │ │ │ │ │ */ │ │ │ │ │ - lastDown: null, │ │ │ │ │ + type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: lastUp │ │ │ │ │ - * {<OpenLayers.Pixel>} │ │ │ │ │ + * Property: panned │ │ │ │ │ + * {Boolean} The map moved. │ │ │ │ │ */ │ │ │ │ │ - lastUp: null, │ │ │ │ │ + panned: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: persist │ │ │ │ │ - * {Boolean} Leave the feature rendered until destroyFeature is called. │ │ │ │ │ - * Default is false. If set to true, the feature remains rendered until │ │ │ │ │ - * destroyFeature is called, typically by deactivating the handler or │ │ │ │ │ - * starting another drawing. │ │ │ │ │ + * Property: interval │ │ │ │ │ + * {Integer} The number of milliseconds that should ellapse before │ │ │ │ │ + * panning the map again. Defaults to 0 milliseconds, which means that │ │ │ │ │ + * no separate cycle is used for panning. In most cases you won't want │ │ │ │ │ + * to change this value. For slow machines/devices larger values can be │ │ │ │ │ + * tried out. │ │ │ │ │ */ │ │ │ │ │ - persist: false, │ │ │ │ │ + interval: 0, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: stopDown │ │ │ │ │ - * {Boolean} Stop event propagation on mousedown. Must be false to │ │ │ │ │ - * allow "pan while drawing". Defaults to false. │ │ │ │ │ + * APIProperty: documentDrag │ │ │ │ │ + * {Boolean} If set to true, mouse dragging will continue even if the │ │ │ │ │ + * mouse cursor leaves the map viewport. Default is false. │ │ │ │ │ */ │ │ │ │ │ - stopDown: false, │ │ │ │ │ + documentDrag: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIPropery: stopUp │ │ │ │ │ - * {Boolean} Stop event propagation on mouse. Must be false to │ │ │ │ │ - * allow "pan while dragging". Defaults to fase. │ │ │ │ │ + * Property: kinetic │ │ │ │ │ + * {<OpenLayers.Kinetic>} The OpenLayers.Kinetic object. │ │ │ │ │ */ │ │ │ │ │ - stopUp: false, │ │ │ │ │ + kinetic: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: layerOptions │ │ │ │ │ - * {Object} Any optional properties to be set on the sketch layer. │ │ │ │ │ + * APIProperty: enableKinetic │ │ │ │ │ + * {Boolean} Set this option to enable "kinetic dragging". Can be │ │ │ │ │ + * set to true or to an object. If set to an object this │ │ │ │ │ + * object will be passed to the {<OpenLayers.Kinetic>} │ │ │ │ │ + * constructor. Defaults to true. │ │ │ │ │ + * To get kinetic dragging, ensure that OpenLayers/Kinetic.js is │ │ │ │ │ + * included in your build config. │ │ │ │ │ */ │ │ │ │ │ - layerOptions: null, │ │ │ │ │ + enableKinetic: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: pixelTolerance │ │ │ │ │ - * {Number} Maximum number of pixels between down and up (mousedown │ │ │ │ │ - * and mouseup, or touchstart and touchend) for the handler to │ │ │ │ │ - * add a new point. If set to an integer value, if the │ │ │ │ │ - * displacement between down and up is great to this value │ │ │ │ │ - * no point will be added. Default value is 5. │ │ │ │ │ + * APIProperty: kineticInterval │ │ │ │ │ + * {Integer} Interval in milliseconds between 2 steps in the "kinetic │ │ │ │ │ + * scrolling". Applies only if enableKinetic is set. Defaults │ │ │ │ │ + * to 10 milliseconds. │ │ │ │ │ */ │ │ │ │ │ - pixelTolerance: 5, │ │ │ │ │ + kineticInterval: 10, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: lastTouchPx │ │ │ │ │ - * {<OpenLayers.Pixel>} The last pixel used to know the distance between │ │ │ │ │ - * two touches (for double touch). │ │ │ │ │ - */ │ │ │ │ │ - lastTouchPx: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Point │ │ │ │ │ - * Create a new point handler. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ - * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ - * functions. Various callbacks described below. │ │ │ │ │ - * options - {Object} An optional object with properties to be set on the │ │ │ │ │ - * handler │ │ │ │ │ - * │ │ │ │ │ - * Named callbacks: │ │ │ │ │ - * create - Called when a sketch is first created. Callback called with │ │ │ │ │ - * the creation point geometry and sketch feature. │ │ │ │ │ - * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ - * geometry and the sketch feature. │ │ │ │ │ - * done - Called when the point drawing is finished. The callback will │ │ │ │ │ - * recieve a single argument, the point geometry. │ │ │ │ │ - * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ - * cancel callback will receive a geometry. │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Creates a Drag handler, using <panMap> and │ │ │ │ │ + * <panMapDone> as callbacks. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ - this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style['default'], {}); │ │ │ │ │ + draw: function() { │ │ │ │ │ + if (this.enableKinetic && OpenLayers.Kinetic) { │ │ │ │ │ + var config = { │ │ │ │ │ + interval: this.kineticInterval │ │ │ │ │ + }; │ │ │ │ │ + if (typeof this.enableKinetic === "object") { │ │ │ │ │ + config = OpenLayers.Util.extend(config, this.enableKinetic); │ │ │ │ │ + } │ │ │ │ │ + this.kinetic = new OpenLayers.Kinetic(config); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.handler = new OpenLayers.Handler.Drag(this, { │ │ │ │ │ + "move": this.panMap, │ │ │ │ │ + "done": this.panMapDone, │ │ │ │ │ + "down": this.panMapStart │ │ │ │ │ + }, { │ │ │ │ │ + interval: this.interval, │ │ │ │ │ + documentDrag: this.documentDrag │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * turn on the handler │ │ │ │ │ + * Method: panMapStart │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (!OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - return false; │ │ │ │ │ + panMapStart: function() { │ │ │ │ │ + if (this.kinetic) { │ │ │ │ │ + this.kinetic.begin(); │ │ │ │ │ } │ │ │ │ │ - // create temporary vector layer for rendering geometry sketch │ │ │ │ │ - // TBD: this could be moved to initialize/destroy - setting visibility here │ │ │ │ │ - var options = OpenLayers.Util.extend({ │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - // indicate that the temp vector layer will never be out of range │ │ │ │ │ - // without this, resolution properties must be specified at the │ │ │ │ │ - // map-level for this temporary layer to init its resolutions │ │ │ │ │ - // correctly │ │ │ │ │ - calculateInRange: OpenLayers.Function.True, │ │ │ │ │ - wrapDateLine: this.citeCompliant │ │ │ │ │ - }, this.layerOptions); │ │ │ │ │ - this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ - this.map.addLayer(this.layer); │ │ │ │ │ - return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createFeature │ │ │ │ │ - * Add temporary features │ │ │ │ │ + * Method: panMap │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} A pixel location on the map. │ │ │ │ │ - */ │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ - lonlat.lon, lonlat.lat │ │ │ │ │ - ); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.point]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.point], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * turn off the handler │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} Pixel of the mouse position │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (!OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - this.cancel(); │ │ │ │ │ - // If a layer's map property is set to null, it means that that layer │ │ │ │ │ - // isn't added to the map. Since we ourself added the layer to the map │ │ │ │ │ - // in activate(), we can assume that if this.layer.map is null it means │ │ │ │ │ - // that the layer has been destroyed (as a result of map.destroy() for │ │ │ │ │ - // example. │ │ │ │ │ - if (this.layer.map != null) { │ │ │ │ │ - this.destroyFeature(true); │ │ │ │ │ - this.layer.destroy(false); │ │ │ │ │ + panMap: function(xy) { │ │ │ │ │ + if (this.kinetic) { │ │ │ │ │ + this.kinetic.update(xy); │ │ │ │ │ } │ │ │ │ │ - this.layer = null; │ │ │ │ │ - return true; │ │ │ │ │ + this.panned = true; │ │ │ │ │ + this.map.pan( │ │ │ │ │ + this.handler.last.x - xy.x, │ │ │ │ │ + this.handler.last.y - xy.y, { │ │ │ │ │ + dragging: true, │ │ │ │ │ + animate: false │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroyFeature │ │ │ │ │ - * Destroy the temporary geometries │ │ │ │ │ + * Method: panMapDone │ │ │ │ │ + * Finish the panning operation. Only call setCenter (through <panMap>) │ │ │ │ │ + * if the map has actually been moved. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} Pixel of the mouse position │ │ │ │ │ */ │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - if (this.layer && (force || !this.persist)) { │ │ │ │ │ - this.layer.destroyFeatures(); │ │ │ │ │ + panMapDone: function(xy) { │ │ │ │ │ + if (this.panned) { │ │ │ │ │ + var res = null; │ │ │ │ │ + if (this.kinetic) { │ │ │ │ │ + res = this.kinetic.end(xy); │ │ │ │ │ + } │ │ │ │ │ + this.map.pan( │ │ │ │ │ + this.handler.last.x - xy.x, │ │ │ │ │ + this.handler.last.y - xy.y, { │ │ │ │ │ + dragging: !!res, │ │ │ │ │ + animate: false │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + if (res) { │ │ │ │ │ + var self = this; │ │ │ │ │ + this.kinetic.move(res, function(x, y, end) { │ │ │ │ │ + self.map.pan(x, y, { │ │ │ │ │ + dragging: !end, │ │ │ │ │ + animate: false │ │ │ │ │ + }); │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + this.panned = false; │ │ │ │ │ } │ │ │ │ │ - this.point = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroyPersistedFeature │ │ │ │ │ - * Destroy the persisted feature. │ │ │ │ │ - */ │ │ │ │ │ - destroyPersistedFeature: function() { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - if (layer && layer.features.length > 1) { │ │ │ │ │ - this.layer.features[0].destroy(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.DragPan" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/Scale.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.Scale │ │ │ │ │ + * The Scale control displays the current map scale as a ratio (e.g. Scale = │ │ │ │ │ + * 1:1M). By default it is displayed in the lower right corner of the map. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.Scale = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: finalize │ │ │ │ │ - * Finish the geometry and call the "done" callback. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * cancel - {Boolean} Call cancel instead of done callback. Default │ │ │ │ │ - * is false. │ │ │ │ │ + * Property: element │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - finalize: function(cancel) { │ │ │ │ │ - var key = cancel ? "cancel" : "done"; │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - this.lastDown = null; │ │ │ │ │ - this.lastUp = null; │ │ │ │ │ - this.lastTouchPx = null; │ │ │ │ │ - this.callback(key, [this.geometryClone()]); │ │ │ │ │ - this.destroyFeature(cancel); │ │ │ │ │ - }, │ │ │ │ │ + element: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: cancel │ │ │ │ │ - * Finish the geometry and call the "cancel" callback. │ │ │ │ │ + * APIProperty: geodesic │ │ │ │ │ + * {Boolean} Use geodesic measurement. Default is false. The recommended │ │ │ │ │ + * setting for maps in EPSG:4326 is false, and true EPSG:900913. If set to │ │ │ │ │ + * true, the scale will be calculated based on the horizontal size of the │ │ │ │ │ + * pixel in the center of the map viewport. │ │ │ │ │ */ │ │ │ │ │ - cancel: function() { │ │ │ │ │ - this.finalize(true); │ │ │ │ │ - }, │ │ │ │ │ + geodesic: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: click │ │ │ │ │ - * Handle clicks. Clicks are stopped from propagating to other listeners │ │ │ │ │ - * on map.events or other dom elements. │ │ │ │ │ + * Constructor: OpenLayers.Control.Scale │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * element - {DOMElement} │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - return false; │ │ │ │ │ + initialize: function(element, options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.element = OpenLayers.Util.getElement(element); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: dblclick │ │ │ │ │ - * Handle double-clicks. Double-clicks are stopped from propagating to other │ │ │ │ │ - * listeners on map.events or other dom elements. │ │ │ │ │ + * Method: draw │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - return false; │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (!this.element) { │ │ │ │ │ + this.element = document.createElement("div"); │ │ │ │ │ + this.div.appendChild(this.element); │ │ │ │ │ + } │ │ │ │ │ + this.map.events.register('moveend', this, this.updateScale); │ │ │ │ │ + this.updateScale(); │ │ │ │ │ + return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: modifyFeature │ │ │ │ │ - * Modify the existing geometry given a pixel location. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} A pixel location on the map. │ │ │ │ │ + * Method: updateScale │ │ │ │ │ */ │ │ │ │ │ - modifyFeature: function(pixel) { │ │ │ │ │ - if (!this.point) { │ │ │ │ │ - this.createFeature(pixel); │ │ │ │ │ + updateScale: function() { │ │ │ │ │ + var scale; │ │ │ │ │ + if (this.geodesic === true) { │ │ │ │ │ + var units = this.map.getUnits(); │ │ │ │ │ + if (!units) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var inches = OpenLayers.INCHES_PER_UNIT; │ │ │ │ │ + scale = (this.map.getGeodesicPixelSize().w || 0.000001) * │ │ │ │ │ + inches["km"] * OpenLayers.DOTS_PER_INCH; │ │ │ │ │ + } else { │ │ │ │ │ + scale = this.map.getScale(); │ │ │ │ │ } │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point.geometry.x = lonlat.lon; │ │ │ │ │ - this.point.geometry.y = lonlat.lat; │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.point, false]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawFeature │ │ │ │ │ - * Render features on the temporary layer. │ │ │ │ │ - */ │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.point, this.style); │ │ │ │ │ - }, │ │ │ │ │ + if (!scale) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getGeometry │ │ │ │ │ - * Return the sketch geometry. If <multi> is true, this will return │ │ │ │ │ - * a multi-part geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} │ │ │ │ │ - */ │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.point && this.point.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiPoint([geometry]); │ │ │ │ │ + if (scale >= 9500 && scale <= 950000) { │ │ │ │ │ + scale = Math.round(scale / 1000) + "K"; │ │ │ │ │ + } else if (scale >= 950000) { │ │ │ │ │ + scale = Math.round(scale / 1000000) + "M"; │ │ │ │ │ + } else { │ │ │ │ │ + scale = Math.round(scale); │ │ │ │ │ } │ │ │ │ │ - return geometry; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: geometryClone │ │ │ │ │ - * Return a clone of the relevant geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} │ │ │ │ │ - */ │ │ │ │ │ - geometryClone: function() { │ │ │ │ │ - var geom = this.getGeometry(); │ │ │ │ │ - return geom && geom.clone(); │ │ │ │ │ + this.element.innerHTML = OpenLayers.i18n("Scale = 1 : ${scaleDenom}", { │ │ │ │ │ + 'scaleDenom': scale │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: mousedown │ │ │ │ │ - * Handle mousedown. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Scale" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/PinchZoom.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Handler/Pinch.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.PinchZoom │ │ │ │ │ + * │ │ │ │ │ + * Inherits: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.PinchZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: type │ │ │ │ │ + * {OpenLayers.Control.TYPES} │ │ │ │ │ */ │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - return this.down(evt); │ │ │ │ │ - }, │ │ │ │ │ + type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchstart │ │ │ │ │ - * Handle touchstart. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * Property: pinchOrigin │ │ │ │ │ + * {Object} Cached object representing the pinch start (in pixels). │ │ │ │ │ */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - this.lastTouchPx = evt.xy; │ │ │ │ │ - return this.down(evt); │ │ │ │ │ - }, │ │ │ │ │ + pinchOrigin: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mousemove │ │ │ │ │ - * Handle mousemove. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * Property: currentCenter │ │ │ │ │ + * {Object} Cached object representing the latest pinch center (in pixels). │ │ │ │ │ */ │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - return this.move(evt); │ │ │ │ │ - }, │ │ │ │ │ + currentCenter: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchmove │ │ │ │ │ - * Handle touchmove. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - this.lastTouchPx = evt.xy; │ │ │ │ │ - return this.move(evt); │ │ │ │ │ - }, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mouseup │ │ │ │ │ - * Handle mouseup. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * APIProperty: preserveCenter │ │ │ │ │ + * {Boolean} Set this to true if you don't want the map center to change │ │ │ │ │ + * while pinching. For example you may want to set preserveCenter to │ │ │ │ │ + * true when the user location is being watched and you want to preserve │ │ │ │ │ + * the user location at the center of the map even if he zooms in or │ │ │ │ │ + * out using pinch. This property's value can be changed any time on an │ │ │ │ │ + * existing instance. Default is false. │ │ │ │ │ */ │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - return this.up(evt); │ │ │ │ │ - }, │ │ │ │ │ + preserveCenter: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchend │ │ │ │ │ - * Handle touchend. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * APIProperty: handlerOptions │ │ │ │ │ + * {Object} Used to set non-default properties on the pinch handler │ │ │ │ │ */ │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - evt.xy = this.lastTouchPx; │ │ │ │ │ - return this.up(evt); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: down │ │ │ │ │ - * Handle mousedown and touchstart. Adjust the geometry and redraw. │ │ │ │ │ - * Return determines whether to propagate the event on the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ + * Constructor: OpenLayers.Control.PinchZoom │ │ │ │ │ + * Create a control for zooming with pinch gestures. This works on devices │ │ │ │ │ + * with multi-touch support. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * the control │ │ │ │ │ */ │ │ │ │ │ - down: function(evt) { │ │ │ │ │ - this.mouseDown = true; │ │ │ │ │ - this.lastDown = evt.xy; │ │ │ │ │ - if (!this.touch) { // no point displayed until up on touch devices │ │ │ │ │ - this.modifyFeature(evt.xy); │ │ │ │ │ - } │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - return !this.stopDown; │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.handler = new OpenLayers.Handler.Pinch(this, { │ │ │ │ │ + start: this.pinchStart, │ │ │ │ │ + move: this.pinchMove, │ │ │ │ │ + done: this.pinchDone │ │ │ │ │ + }, this.handlerOptions); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * Handle mousemove and touchmove. Adjust the geometry and redraw. │ │ │ │ │ - * Return determines whether to propagate the event on the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ + * Method: pinchStart │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * pinchData - {Object} pinch data object related to the current touchmove │ │ │ │ │ + * of the pinch gesture. This give us the current scale of the pinch. │ │ │ │ │ */ │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - if (!this.touch // no point displayed until up on touch devices │ │ │ │ │ - && │ │ │ │ │ - (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ - this.modifyFeature(evt.xy); │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ + pinchStart: function(evt, pinchData) { │ │ │ │ │ + var xy = (this.preserveCenter) ? │ │ │ │ │ + this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ + this.pinchOrigin = xy; │ │ │ │ │ + this.currentCenter = xy; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: up │ │ │ │ │ - * Handle mouseup and touchend. Send the latest point in the geometry to the control. │ │ │ │ │ - * Return determines whether to propagate the event on the map. │ │ │ │ │ + * Method: pinchMove │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * pinchData - {Object} pinch data object related to the current touchmove │ │ │ │ │ + * of the pinch gesture. This give us the current scale of the pinch. │ │ │ │ │ */ │ │ │ │ │ - up: function(evt) { │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ + pinchMove: function(evt, pinchData) { │ │ │ │ │ + var scale = pinchData.scale; │ │ │ │ │ + var containerOrigin = this.map.layerContainerOriginPx; │ │ │ │ │ + var pinchOrigin = this.pinchOrigin; │ │ │ │ │ + var current = (this.preserveCenter) ? │ │ │ │ │ + this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ │ │ │ │ │ - // check keyboard modifiers │ │ │ │ │ - if (!this.checkModifiers(evt)) { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - // ignore double-clicks │ │ │ │ │ - if (this.lastUp && this.lastUp.equals(evt.xy)) { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - if (this.lastDown && this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ - this.pixelTolerance)) { │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy); │ │ │ │ │ - } │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature(); │ │ │ │ │ - } │ │ │ │ │ - this.lastUp = evt.xy; │ │ │ │ │ - this.finalize(); │ │ │ │ │ - return !this.stopUp; │ │ │ │ │ - } else { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ + var dx = Math.round((containerOrigin.x + current.x - pinchOrigin.x) + (scale - 1) * (containerOrigin.x - pinchOrigin.x)); │ │ │ │ │ + var dy = Math.round((containerOrigin.y + current.y - pinchOrigin.y) + (scale - 1) * (containerOrigin.y - pinchOrigin.y)); │ │ │ │ │ + │ │ │ │ │ + this.map.applyTransform(dx, dy, scale); │ │ │ │ │ + this.currentCenter = current; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mouseout │ │ │ │ │ - * Handle mouse out. For better user experience reset mouseDown │ │ │ │ │ - * and stoppedDown when the mouse leaves the map viewport. │ │ │ │ │ + * Method: pinchDone │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * start - {Object} pinch data object related to the touchstart event that │ │ │ │ │ + * started the pinch gesture. │ │ │ │ │ + * last - {Object} pinch data object related to the last touchmove event │ │ │ │ │ + * of the pinch gesture. This give us the final scale of the pinch. │ │ │ │ │ */ │ │ │ │ │ - mouseout: function(evt) { │ │ │ │ │ - if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + pinchDone: function(evt, start, last) { │ │ │ │ │ + this.map.applyTransform(); │ │ │ │ │ + var zoom = this.map.getZoomForResolution(this.map.getResolution() / last.scale, true); │ │ │ │ │ + if (zoom !== this.map.getZoom() || !this.currentCenter.equals(this.pinchOrigin)) { │ │ │ │ │ + var resolution = this.map.getResolutionForZoom(zoom); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: passesTolerance │ │ │ │ │ - * Determine whether the event is within the optional pixel tolerance. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The event is within the pixel tolerance (if specified). │ │ │ │ │ - */ │ │ │ │ │ - passesTolerance: function(pixel1, pixel2, tolerance) { │ │ │ │ │ - var passes = true; │ │ │ │ │ + var location = this.map.getLonLatFromPixel(this.pinchOrigin); │ │ │ │ │ + var zoomPixel = this.currentCenter; │ │ │ │ │ + var size = this.map.getSize(); │ │ │ │ │ │ │ │ │ │ - if (tolerance != null && pixel1 && pixel2) { │ │ │ │ │ - var dist = pixel1.distanceTo(pixel2); │ │ │ │ │ - if (dist > tolerance) { │ │ │ │ │ - passes = false; │ │ │ │ │ - } │ │ │ │ │ + location.lon += resolution * ((size.w / 2) - zoomPixel.x); │ │ │ │ │ + location.lat -= resolution * ((size.h / 2) - zoomPixel.y); │ │ │ │ │ + │ │ │ │ │ + // Force a reflow before calling setCenter. This is to work │ │ │ │ │ + // around an issue occuring in iOS. │ │ │ │ │ + // │ │ │ │ │ + // See https://github.com/openlayers/openlayers/pull/351. │ │ │ │ │ + // │ │ │ │ │ + // Without a reflow setting the layer container div's top left │ │ │ │ │ + // style properties to "0px" - as done in Map.moveTo when zoom │ │ │ │ │ + // is changed - won't actually correctly reposition the layer │ │ │ │ │ + // container div. │ │ │ │ │ + // │ │ │ │ │ + // Also, we need to use a statement that the Google Closure │ │ │ │ │ + // compiler won't optimize away. │ │ │ │ │ + this.map.div.clientWidth = this.map.div.clientWidth; │ │ │ │ │ + │ │ │ │ │ + this.map.setCenter(location, zoom); │ │ │ │ │ } │ │ │ │ │ - return passes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Point" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.PinchZoom" │ │ │ │ │ + │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Feature.js │ │ │ │ │ + OpenLayers/Control/TouchNavigation.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Control/DragPan.js │ │ │ │ │ + * @requires OpenLayers/Control/PinchZoom.js │ │ │ │ │ + * @requires OpenLayers/Handler/Click.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Handler.Feature │ │ │ │ │ - * Handler to respond to mouse events related to a drawn feature. Callbacks │ │ │ │ │ - * with the following keys will be notified of the following events │ │ │ │ │ - * associated with features: click, clickout, over, out, and dblclick. │ │ │ │ │ + * Class: OpenLayers.Control.TouchNavigation │ │ │ │ │ + * The navigation control handles map browsing with touch events (dragging, │ │ │ │ │ + * double-tapping, tap with two fingers, and pinch zoom). Create a new │ │ │ │ │ + * control with the <OpenLayers.Control.TouchNavigation> constructor. │ │ │ │ │ * │ │ │ │ │ - * This handler stops event propagation for mousedown and mouseup if those │ │ │ │ │ - * browser events target features that can be selected. │ │ │ │ │ + * If you’re only targeting touch enabled devices with your mapping application, │ │ │ │ │ + * you can create a map with only a TouchNavigation control. The │ │ │ │ │ + * <OpenLayers.Control.Navigation> control is mobile ready by default, but │ │ │ │ │ + * you can generate a smaller build of the library by only including this │ │ │ │ │ + * touch navigation control if you aren't concerned about mouse interaction. │ │ │ │ │ * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ + * Inherits: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.Feature = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: EVENTMAP │ │ │ │ │ - * {Object} A object mapping the browser events to objects with callback │ │ │ │ │ - * keys for in and out. │ │ │ │ │ - */ │ │ │ │ │ - EVENTMAP: { │ │ │ │ │ - 'click': { │ │ │ │ │ - 'in': 'click', │ │ │ │ │ - 'out': 'clickout' │ │ │ │ │ - }, │ │ │ │ │ - 'mousemove': { │ │ │ │ │ - 'in': 'over', │ │ │ │ │ - 'out': 'out' │ │ │ │ │ - }, │ │ │ │ │ - 'dblclick': { │ │ │ │ │ - 'in': 'dblclick', │ │ │ │ │ - 'out': null │ │ │ │ │ - }, │ │ │ │ │ - 'mousedown': { │ │ │ │ │ - 'in': null, │ │ │ │ │ - 'out': null │ │ │ │ │ - }, │ │ │ │ │ - 'mouseup': { │ │ │ │ │ - 'in': null, │ │ │ │ │ - 'out': null │ │ │ │ │ - }, │ │ │ │ │ - 'touchstart': { │ │ │ │ │ - 'in': 'click', │ │ │ │ │ - 'out': 'clickout' │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: feature │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} The last feature that was hovered. │ │ │ │ │ - */ │ │ │ │ │ - feature: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: lastFeature │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} The last feature that was handled. │ │ │ │ │ - */ │ │ │ │ │ - lastFeature: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: down │ │ │ │ │ - * {<OpenLayers.Pixel>} The location of the last mousedown. │ │ │ │ │ - */ │ │ │ │ │ - down: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: up │ │ │ │ │ - * {<OpenLayers.Pixel>} The location of the last mouseup. │ │ │ │ │ - */ │ │ │ │ │ - up: null, │ │ │ │ │ +OpenLayers.Control.TouchNavigation = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: clickTolerance │ │ │ │ │ - * {Number} The number of pixels the mouse can move between mousedown │ │ │ │ │ - * and mouseup for the event to still be considered a click. │ │ │ │ │ - * Dragging the map should not trigger the click and clickout callbacks │ │ │ │ │ - * unless the map is moved by less than this tolerance. Defaults to 4. │ │ │ │ │ + * Property: dragPan │ │ │ │ │ + * {<OpenLayers.Control.DragPan>} │ │ │ │ │ */ │ │ │ │ │ - clickTolerance: 4, │ │ │ │ │ + dragPan: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: geometryTypes │ │ │ │ │ - * To restrict dragging to a limited set of geometry types, send a list │ │ │ │ │ - * of strings corresponding to the geometry class names. │ │ │ │ │ - * │ │ │ │ │ - * @type Array(String) │ │ │ │ │ + * APIProperty: dragPanOptions │ │ │ │ │ + * {Object} Options passed to the DragPan control. │ │ │ │ │ */ │ │ │ │ │ - geometryTypes: null, │ │ │ │ │ + dragPanOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: stopClick │ │ │ │ │ - * {Boolean} If stopClick is set to true, handled clicks do not │ │ │ │ │ - * propagate to other click listeners. Otherwise, handled clicks │ │ │ │ │ - * do propagate. Unhandled clicks always propagate, whatever the │ │ │ │ │ - * value of stopClick. Defaults to true. │ │ │ │ │ + * Property: pinchZoom │ │ │ │ │ + * {<OpenLayers.Control.PinchZoom>} │ │ │ │ │ */ │ │ │ │ │ - stopClick: true, │ │ │ │ │ + pinchZoom: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: stopDown │ │ │ │ │ - * {Boolean} If stopDown is set to true, handled mousedowns do not │ │ │ │ │ - * propagate to other mousedown listeners. Otherwise, handled │ │ │ │ │ - * mousedowns do propagate. Unhandled mousedowns always propagate, │ │ │ │ │ - * whatever the value of stopDown. Defaults to true. │ │ │ │ │ + * APIProperty: pinchZoomOptions │ │ │ │ │ + * {Object} Options passed to the PinchZoom control. │ │ │ │ │ */ │ │ │ │ │ - stopDown: true, │ │ │ │ │ + pinchZoomOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: stopUp │ │ │ │ │ - * {Boolean} If stopUp is set to true, handled mouseups do not │ │ │ │ │ - * propagate to other mouseup listeners. Otherwise, handled mouseups │ │ │ │ │ - * do propagate. Unhandled mouseups always propagate, whatever the │ │ │ │ │ - * value of stopUp. Defaults to false. │ │ │ │ │ + * APIProperty: clickHandlerOptions │ │ │ │ │ + * {Object} Options passed to the Click handler. │ │ │ │ │ */ │ │ │ │ │ - stopUp: false, │ │ │ │ │ + clickHandlerOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Feature │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} │ │ │ │ │ - * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ - * callbacks - {Object} An object with a 'over' property whos value is │ │ │ │ │ - * a function to be called when the mouse is over a feature. The │ │ │ │ │ - * callback should expect to recieve a single argument, the feature. │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * APIProperty: documentDrag │ │ │ │ │ + * {Boolean} Allow panning of the map by dragging outside map viewport. │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(control, layer, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - }, │ │ │ │ │ + documentDrag: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchstart │ │ │ │ │ - * Handle touchstart events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - return OpenLayers.Event.isMultiTouch(evt) ? │ │ │ │ │ - true : this.mousedown(evt); │ │ │ │ │ - }, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchmove │ │ │ │ │ - * Handle touchmove events. We just prevent the browser default behavior, │ │ │ │ │ - * for Android Webkit not to select text when moving the finger after │ │ │ │ │ - * selecting a feature. │ │ │ │ │ + * Constructor: OpenLayers.Control.TouchNavigation │ │ │ │ │ + * Create a new navigation control │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * the control │ │ │ │ │ */ │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.handlers = {}; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mousedown │ │ │ │ │ - * Handle mouse down. Stop propagation if a feature is targeted by this │ │ │ │ │ - * event (stops map dragging during feature selection). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * The destroy method is used to perform any clean up before the control │ │ │ │ │ + * is dereferenced. Typically this is where event listeners are removed │ │ │ │ │ + * to prevent memory leaks. │ │ │ │ │ */ │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - // Feature selection is only done with a left click. Other handlers may stop the │ │ │ │ │ - // propagation of left-click mousedown events but not right-click mousedown events. │ │ │ │ │ - // This mismatch causes problems when comparing the location of the down and up │ │ │ │ │ - // events in the click function so it is important ignore right-clicks. │ │ │ │ │ - if (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ - this.down = evt.xy; │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + if (this.dragPan) { │ │ │ │ │ + this.dragPan.destroy(); │ │ │ │ │ } │ │ │ │ │ - return this.handle(evt) ? !this.stopDown : true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mouseup │ │ │ │ │ - * Handle mouse up. Stop propagation if a feature is targeted by this │ │ │ │ │ - * event. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - this.up = evt.xy; │ │ │ │ │ - return this.handle(evt) ? !this.stopUp : true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: click │ │ │ │ │ - * Handle click. Call the "click" callback if click on a feature, │ │ │ │ │ - * or the "clickout" callback if click outside any feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - return this.handle(evt) ? !this.stopClick : true; │ │ │ │ │ + this.dragPan = null; │ │ │ │ │ + if (this.pinchZoom) { │ │ │ │ │ + this.pinchZoom.destroy(); │ │ │ │ │ + delete this.pinchZoom; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mousemove │ │ │ │ │ - * Handle mouse moves. Call the "over" callback if moving in to a feature, │ │ │ │ │ - * or the "out" callback if moving out of a feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * Method: activate │ │ │ │ │ */ │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - if (!this.callbacks['over'] && !this.callbacks['out']) { │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.dragPan.activate(); │ │ │ │ │ + this.handlers.click.activate(); │ │ │ │ │ + this.pinchZoom.activate(); │ │ │ │ │ return true; │ │ │ │ │ } │ │ │ │ │ - this.handle(evt); │ │ │ │ │ - return true; │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: dblclick │ │ │ │ │ - * Handle dblclick. Call the "dblclick" callback if dblclick on a feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * Method: deactivate │ │ │ │ │ */ │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - return !this.handle(evt); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.dragPan.deactivate(); │ │ │ │ │ + this.handlers.click.deactivate(); │ │ │ │ │ + this.pinchZoom.deactivate(); │ │ │ │ │ + return true; │ │ │ │ │ + } │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: geometryTypeMatches │ │ │ │ │ - * Return true if the geometry type of the passed feature matches │ │ │ │ │ - * one of the geometry types in the geometryTypes array. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Vector.Feature>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * Method: draw │ │ │ │ │ */ │ │ │ │ │ - geometryTypeMatches: function(feature) { │ │ │ │ │ - return this.geometryTypes == null || │ │ │ │ │ - OpenLayers.Util.indexOf(this.geometryTypes, │ │ │ │ │ - feature.geometry.CLASS_NAME) > -1; │ │ │ │ │ + draw: function() { │ │ │ │ │ + var clickCallbacks = { │ │ │ │ │ + click: this.defaultClick, │ │ │ │ │ + dblclick: this.defaultDblClick │ │ │ │ │ + }; │ │ │ │ │ + var clickOptions = OpenLayers.Util.extend({ │ │ │ │ │ + "double": true, │ │ │ │ │ + stopDouble: true, │ │ │ │ │ + pixelTolerance: 2 │ │ │ │ │ + }, this.clickHandlerOptions); │ │ │ │ │ + this.handlers.click = new OpenLayers.Handler.Click( │ │ │ │ │ + this, clickCallbacks, clickOptions │ │ │ │ │ + ); │ │ │ │ │ + this.dragPan = new OpenLayers.Control.DragPan( │ │ │ │ │ + OpenLayers.Util.extend({ │ │ │ │ │ + map: this.map, │ │ │ │ │ + documentDrag: this.documentDrag │ │ │ │ │ + }, this.dragPanOptions) │ │ │ │ │ + ); │ │ │ │ │ + this.dragPan.draw(); │ │ │ │ │ + this.pinchZoom = new OpenLayers.Control.PinchZoom( │ │ │ │ │ + OpenLayers.Util.extend({ │ │ │ │ │ + map: this.map │ │ │ │ │ + }, this.pinchZoomOptions) │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handle │ │ │ │ │ + * Method: defaultClick │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The event occurred over a relevant feature. │ │ │ │ │ - */ │ │ │ │ │ - handle: function(evt) { │ │ │ │ │ - if (this.feature && !this.feature.layer) { │ │ │ │ │ - // feature has been destroyed │ │ │ │ │ - this.feature = null; │ │ │ │ │ - } │ │ │ │ │ - var type = evt.type; │ │ │ │ │ - var handled = false; │ │ │ │ │ - var previouslyIn = !!(this.feature); // previously in a feature │ │ │ │ │ - var click = (type == "click" || type == "dblclick" || type == "touchstart"); │ │ │ │ │ - this.feature = this.layer.getFeatureFromEvent(evt); │ │ │ │ │ - if (this.feature && !this.feature.layer) { │ │ │ │ │ - // feature has been destroyed │ │ │ │ │ - this.feature = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.lastFeature && !this.lastFeature.layer) { │ │ │ │ │ - // last feature has been destroyed │ │ │ │ │ - this.lastFeature = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.feature) { │ │ │ │ │ - if (type === "touchstart") { │ │ │ │ │ - // stop the event to prevent Android Webkit from │ │ │ │ │ - // "flashing" the map div │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ - } │ │ │ │ │ - var inNew = (this.feature != this.lastFeature); │ │ │ │ │ - if (this.geometryTypeMatches(this.feature)) { │ │ │ │ │ - // in to a feature │ │ │ │ │ - if (previouslyIn && inNew) { │ │ │ │ │ - // out of last feature and in to another │ │ │ │ │ - if (this.lastFeature) { │ │ │ │ │ - this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ - } │ │ │ │ │ - this.triggerCallback(type, 'in', [this.feature]); │ │ │ │ │ - } else if (!previouslyIn || click) { │ │ │ │ │ - // in feature for the first time │ │ │ │ │ - this.triggerCallback(type, 'in', [this.feature]); │ │ │ │ │ - } │ │ │ │ │ - this.lastFeature = this.feature; │ │ │ │ │ - handled = true; │ │ │ │ │ - } else { │ │ │ │ │ - // not in to a feature │ │ │ │ │ - if (this.lastFeature && (previouslyIn && inNew || click)) { │ │ │ │ │ - // out of last feature for the first time │ │ │ │ │ - this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ - } │ │ │ │ │ - // next time the mouse goes in a feature whose geometry type │ │ │ │ │ - // doesn't match we don't want to call the 'out' callback │ │ │ │ │ - // again, so let's set this.feature to null so that │ │ │ │ │ - // previouslyIn will evaluate to false the next time │ │ │ │ │ - // we enter handle. Yes, a bit hackish... │ │ │ │ │ - this.feature = null; │ │ │ │ │ - } │ │ │ │ │ - } else if (this.lastFeature && (previouslyIn || click)) { │ │ │ │ │ - this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ - } │ │ │ │ │ - return handled; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: triggerCallback │ │ │ │ │ - * Call the callback keyed in the event map with the supplied arguments. │ │ │ │ │ - * For click and clickout, the <clickTolerance> is checked first. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * type - {String} │ │ │ │ │ - */ │ │ │ │ │ - triggerCallback: function(type, mode, args) { │ │ │ │ │ - var key = this.EVENTMAP[type][mode]; │ │ │ │ │ - if (key) { │ │ │ │ │ - if (type == 'click' && this.up && this.down) { │ │ │ │ │ - // for click/clickout, only trigger callback if tolerance is met │ │ │ │ │ - var dpx = Math.sqrt( │ │ │ │ │ - Math.pow(this.up.x - this.down.x, 2) + │ │ │ │ │ - Math.pow(this.up.y - this.down.y, 2) │ │ │ │ │ - ); │ │ │ │ │ - if (dpx <= this.clickTolerance) { │ │ │ │ │ - this.callback(key, args); │ │ │ │ │ - } │ │ │ │ │ - // we're done with this set of events now: clear the cached │ │ │ │ │ - // positions so we can't trip over them later (this can occur │ │ │ │ │ - // if one of the up/down events gets eaten before it gets to us │ │ │ │ │ - // but we still get the click) │ │ │ │ │ - this.up = this.down = null; │ │ │ │ │ - } else { │ │ │ │ │ - this.callback(key, args); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Turn on the handler. Returns false if the handler was already active. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerToTop(); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - "removelayer": this.handleMapEvents, │ │ │ │ │ - "changelayer": this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - activated = true; │ │ │ │ │ + defaultClick: function(evt) { │ │ │ │ │ + if (evt.lastTouches && evt.lastTouches.length == 2) { │ │ │ │ │ + this.map.zoomOut(); │ │ │ │ │ } │ │ │ │ │ - return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Turn off the handler. Returns false if the handler was already active. │ │ │ │ │ + * Method: defaultDblClick │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerBack(); │ │ │ │ │ - this.feature = null; │ │ │ │ │ - this.lastFeature = null; │ │ │ │ │ - this.down = null; │ │ │ │ │ - this.up = null; │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - "removelayer": this.handleMapEvents, │ │ │ │ │ - "changelayer": this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - deactivated = true; │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleMapEvents │ │ │ │ │ - * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ - */ │ │ │ │ │ - handleMapEvents: function(evt) { │ │ │ │ │ - if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ - this.moveLayerToTop(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveLayerToTop │ │ │ │ │ - * Moves the layer for this handler to the top, so mouse events can reach │ │ │ │ │ - * it. │ │ │ │ │ - */ │ │ │ │ │ - moveLayerToTop: function() { │ │ │ │ │ - var index = Math.max(this.map.Z_INDEX_BASE['Feature'] - 1, │ │ │ │ │ - this.layer.getZIndex()) + 1; │ │ │ │ │ - this.layer.setZIndex(index); │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveLayerBack │ │ │ │ │ - * Moves the layer back to the position determined by the map's layers │ │ │ │ │ - * array. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - moveLayerBack: function() { │ │ │ │ │ - var index = this.layer.getZIndex() - 1; │ │ │ │ │ - if (index >= this.map.Z_INDEX_BASE['Feature']) { │ │ │ │ │ - this.layer.setZIndex(index); │ │ │ │ │ - } else { │ │ │ │ │ - this.map.setLayerZIndex(this.layer, │ │ │ │ │ - this.map.getLayerIndex(this.layer)); │ │ │ │ │ - } │ │ │ │ │ + defaultDblClick: function(evt) { │ │ │ │ │ + this.map.zoomTo(this.map.zoom + 1, evt.xy); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Feature" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.TouchNavigation" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Drag.js │ │ │ │ │ + OpenLayers/Control/KeyboardDefaults.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Handler/Keyboard.js │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Handler.Drag │ │ │ │ │ - * The drag handler is used to deal with sequences of browser events related │ │ │ │ │ - * to dragging. The handler is used by controls that want to know when │ │ │ │ │ - * a drag sequence begins, when a drag is happening, and when it has │ │ │ │ │ - * finished. │ │ │ │ │ - * │ │ │ │ │ - * Controls that use the drag handler typically construct it with callbacks │ │ │ │ │ - * for 'down', 'move', and 'done'. Callbacks for these keys are called │ │ │ │ │ - * when the drag begins, with each move, and when the drag is done. In │ │ │ │ │ - * addition, controls can have callbacks keyed to 'up' and 'out' if they │ │ │ │ │ - * care to differentiate between the types of events that correspond with │ │ │ │ │ - * the end of a drag sequence. If no drag actually occurs (no mouse move) │ │ │ │ │ - * the 'down' and 'up' callbacks will be called, but not the 'done' │ │ │ │ │ - * callback. │ │ │ │ │ - * │ │ │ │ │ - * Create a new drag handler with the <OpenLayers.Handler.Drag> constructor. │ │ │ │ │ + * Class: OpenLayers.Control.KeyboardDefaults │ │ │ │ │ + * The KeyboardDefaults control adds panning and zooming functions, controlled │ │ │ │ │ + * with the keyboard. By default arrow keys pan, +/- keys zoom & Page Up/Page │ │ │ │ │ + * Down/Home/End scroll by three quarters of a page. │ │ │ │ │ + * │ │ │ │ │ + * This control has no visible appearance. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.Drag = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: started │ │ │ │ │ - * {Boolean} When a mousedown or touchstart event is received, we want to │ │ │ │ │ - * record it, but not set 'dragging' until the mouse moves after starting. │ │ │ │ │ - */ │ │ │ │ │ - started: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: stopDown │ │ │ │ │ - * {Boolean} Stop propagation of mousedown events from getting to listeners │ │ │ │ │ - * on the same element. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - stopDown: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: dragging │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - dragging: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: last │ │ │ │ │ - * {<OpenLayers.Pixel>} The last pixel location of the drag. │ │ │ │ │ - */ │ │ │ │ │ - last: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: start │ │ │ │ │ - * {<OpenLayers.Pixel>} The first pixel location of the drag. │ │ │ │ │ - */ │ │ │ │ │ - start: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: lastMoveEvt │ │ │ │ │ - * {Object} The last mousemove event that occurred. Used to │ │ │ │ │ - * position the map correctly when our "delay drag" │ │ │ │ │ - * timeout expired. │ │ │ │ │ - */ │ │ │ │ │ - lastMoveEvt: null, │ │ │ │ │ +OpenLayers.Control.KeyboardDefaults = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: oldOnselectstart │ │ │ │ │ - * {Function} │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ - oldOnselectstart: null, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: interval │ │ │ │ │ - * {Integer} In order to increase performance, an interval (in │ │ │ │ │ - * milliseconds) can be set to reduce the number of drag events │ │ │ │ │ - * called. If set, a new drag event will not be set until the │ │ │ │ │ - * interval has passed. │ │ │ │ │ - * Defaults to 0, meaning no interval. │ │ │ │ │ + * APIProperty: slideFactor │ │ │ │ │ + * Pixels to slide by. │ │ │ │ │ */ │ │ │ │ │ - interval: 0, │ │ │ │ │ + slideFactor: 75, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: timeoutId │ │ │ │ │ - * {String} The id of the timeout used for the mousedown interval. │ │ │ │ │ - * This is "private", and should be left alone. │ │ │ │ │ + * APIProperty: observeElement │ │ │ │ │ + * {DOMelement|String} The DOM element to handle keys for. You │ │ │ │ │ + * can use the map div here, to have the navigation keys │ │ │ │ │ + * work when the map div has the focus. If undefined the │ │ │ │ │ + * document is used. │ │ │ │ │ */ │ │ │ │ │ - timeoutId: null, │ │ │ │ │ + observeElement: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: documentDrag │ │ │ │ │ - * {Boolean} If set to true, the handler will also handle mouse moves when │ │ │ │ │ - * the cursor has moved out of the map viewport. Default is false. │ │ │ │ │ + * Constructor: OpenLayers.Control.KeyboardDefaults │ │ │ │ │ */ │ │ │ │ │ - documentDrag: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: documentEvents │ │ │ │ │ - * {Boolean} Are we currently observing document events? │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Create handler. │ │ │ │ │ */ │ │ │ │ │ - documentEvents: null, │ │ │ │ │ + draw: function() { │ │ │ │ │ + var observeElement = this.observeElement || document; │ │ │ │ │ + this.handler = new OpenLayers.Handler.Keyboard(this, { │ │ │ │ │ + "keydown": this.defaultKeyPress │ │ │ │ │ + }, { │ │ │ │ │ + observeElement: observeElement │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Drag │ │ │ │ │ - * Returns OpenLayers.Handler.Drag │ │ │ │ │ + * Method: defaultKeyPress │ │ │ │ │ + * When handling the key event, we only use evt.keyCode. This holds │ │ │ │ │ + * some drawbacks, though we get around them below. When interpretting │ │ │ │ │ + * the keycodes below (including the comments associated with them), │ │ │ │ │ + * consult the URL below. For instance, the Safari browser returns │ │ │ │ │ + * "IE keycodes", and so is supported by any keycode labeled "IE". │ │ │ │ │ * │ │ │ │ │ + * Very informative URL: │ │ │ │ │ + * http://unixpapa.com/js/key.html │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ - * this handler. If a handler is being used without a control, the │ │ │ │ │ - * handlers setMap method must be overridden to deal properly with │ │ │ │ │ - * the map. │ │ │ │ │ - * callbacks - {Object} An object containing a single function to be │ │ │ │ │ - * called when the drag operation is finished. The callback should │ │ │ │ │ - * expect to recieve a single argument, the pixel location of the event. │ │ │ │ │ - * Callbacks for 'move' and 'done' are supported. You can also speficy │ │ │ │ │ - * callbacks for 'down', 'up', and 'out' to respond to those events. │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + defaultKeyPress: function(evt) { │ │ │ │ │ + var size, handled = true; │ │ │ │ │ │ │ │ │ │ - if (this.documentDrag === true) { │ │ │ │ │ - var me = this; │ │ │ │ │ - this._docMove = function(evt) { │ │ │ │ │ - me.mousemove({ │ │ │ │ │ - xy: { │ │ │ │ │ - x: evt.clientX, │ │ │ │ │ - y: evt.clientY │ │ │ │ │ - }, │ │ │ │ │ - element: document │ │ │ │ │ - }); │ │ │ │ │ - }; │ │ │ │ │ - this._docUp = function(evt) { │ │ │ │ │ - me.mouseup({ │ │ │ │ │ - xy: { │ │ │ │ │ - x: evt.clientX, │ │ │ │ │ - y: evt.clientY │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - }; │ │ │ │ │ + var target = OpenLayers.Event.element(evt); │ │ │ │ │ + if (target && │ │ │ │ │ + (target.tagName == 'INPUT' || │ │ │ │ │ + target.tagName == 'TEXTAREA' || │ │ │ │ │ + target.tagName == 'SELECT')) { │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: dragstart │ │ │ │ │ - * This private method is factorized from mousedown and touchstart methods │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - dragstart: function(evt) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - if (this.checkModifiers(evt) && │ │ │ │ │ - (OpenLayers.Event.isLeftClick(evt) || │ │ │ │ │ - OpenLayers.Event.isSingleTouch(evt))) { │ │ │ │ │ - this.started = true; │ │ │ │ │ - this.start = evt.xy; │ │ │ │ │ - this.last = evt.xy; │ │ │ │ │ - OpenLayers.Element.addClass( │ │ │ │ │ - this.map.viewPortDiv, "olDragDown" │ │ │ │ │ - ); │ │ │ │ │ - this.down(evt); │ │ │ │ │ - this.callback("down", [evt.xy]); │ │ │ │ │ │ │ │ │ │ - // prevent document dragging │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + switch (evt.keyCode) { │ │ │ │ │ + case OpenLayers.Event.KEY_LEFT: │ │ │ │ │ + this.map.pan(-this.slideFactor, 0); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Event.KEY_RIGHT: │ │ │ │ │ + this.map.pan(this.slideFactor, 0); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Event.KEY_UP: │ │ │ │ │ + this.map.pan(0, -this.slideFactor); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Event.KEY_DOWN: │ │ │ │ │ + this.map.pan(0, this.slideFactor); │ │ │ │ │ + break; │ │ │ │ │ │ │ │ │ │ - if (!this.oldOnselectstart) { │ │ │ │ │ - this.oldOnselectstart = document.onselectstart ? │ │ │ │ │ - document.onselectstart : OpenLayers.Function.True; │ │ │ │ │ - } │ │ │ │ │ - document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ + case 33: // Page Up. Same in all browsers. │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + this.map.pan(0, -0.75 * size.h); │ │ │ │ │ + break; │ │ │ │ │ + case 34: // Page Down. Same in all browsers. │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + this.map.pan(0, 0.75 * size.h); │ │ │ │ │ + break; │ │ │ │ │ + case 35: // End. Same in all browsers. │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + this.map.pan(0.75 * size.w, 0); │ │ │ │ │ + break; │ │ │ │ │ + case 36: // Home. Same in all browsers. │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + this.map.pan(-0.75 * size.w, 0); │ │ │ │ │ + break; │ │ │ │ │ │ │ │ │ │ - propagate = !this.stopDown; │ │ │ │ │ - } else { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ + case 43: // +/= (ASCII), keypad + (ASCII, Opera) │ │ │ │ │ + case 61: // +/= (Mozilla, Opera, some ASCII) │ │ │ │ │ + case 187: // +/= (IE) │ │ │ │ │ + case 107: // keypad + (IE, Mozilla) │ │ │ │ │ + this.map.zoomIn(); │ │ │ │ │ + break; │ │ │ │ │ + case 45: // -/_ (ASCII, Opera), keypad - (ASCII, Opera) │ │ │ │ │ + case 109: // -/_ (Mozilla), keypad - (Mozilla, IE) │ │ │ │ │ + case 189: // -/_ (IE) │ │ │ │ │ + case 95: // -/_ (some ASCII) │ │ │ │ │ + this.map.zoomOut(); │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + handled = false; │ │ │ │ │ } │ │ │ │ │ - return propagate; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: dragmove │ │ │ │ │ - * This private method is factorized from mousemove and touchmove methods │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - dragmove: function(evt) { │ │ │ │ │ - this.lastMoveEvt = evt; │ │ │ │ │ - if (this.started && !this.timeoutId && (evt.xy.x != this.last.x || │ │ │ │ │ - evt.xy.y != this.last.y)) { │ │ │ │ │ - if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ - if (evt.element === document) { │ │ │ │ │ - this.adjustXY(evt); │ │ │ │ │ - // do setEvent manually because the documentEvents are not │ │ │ │ │ - // registered with the map │ │ │ │ │ - this.setEvent(evt); │ │ │ │ │ - } else { │ │ │ │ │ - this.removeDocumentEvents(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.interval > 0) { │ │ │ │ │ - this.timeoutId = setTimeout( │ │ │ │ │ - OpenLayers.Function.bind(this.removeTimeout, this), │ │ │ │ │ - this.interval); │ │ │ │ │ - } │ │ │ │ │ - this.dragging = true; │ │ │ │ │ - │ │ │ │ │ - this.move(evt); │ │ │ │ │ - this.callback("move", [evt.xy]); │ │ │ │ │ - if (!this.oldOnselectstart) { │ │ │ │ │ - this.oldOnselectstart = document.onselectstart; │ │ │ │ │ - document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ - } │ │ │ │ │ - this.last = evt.xy; │ │ │ │ │ + if (handled) { │ │ │ │ │ + // prevent browser default not to move the page │ │ │ │ │ + // when moving the page with the keyboard │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ } │ │ │ │ │ - return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: dragend │ │ │ │ │ - * This private method is factorized from mouseup and touchend methods │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - dragend: function(evt) { │ │ │ │ │ - if (this.started) { │ │ │ │ │ - if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ - this.adjustXY(evt); │ │ │ │ │ - this.removeDocumentEvents(); │ │ │ │ │ - } │ │ │ │ │ - var dragged = (this.start != this.last); │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - OpenLayers.Element.removeClass( │ │ │ │ │ - this.map.viewPortDiv, "olDragDown" │ │ │ │ │ - ); │ │ │ │ │ - this.up(evt); │ │ │ │ │ - this.callback("up", [evt.xy]); │ │ │ │ │ - if (dragged) { │ │ │ │ │ - this.callback("done", [evt.xy]); │ │ │ │ │ - } │ │ │ │ │ - document.onselectstart = this.oldOnselectstart; │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.KeyboardDefaults" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/ArgParser.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * The four methods below (down, move, up, and out) are used by subclasses │ │ │ │ │ - * to do their own processing related to these mouse events. │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: down │ │ │ │ │ - * This method is called during the handling of the mouse down event. │ │ │ │ │ - * Subclasses can do their own processing here. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The mouse down event │ │ │ │ │ - */ │ │ │ │ │ - down: function(evt) {}, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * This method is called during the handling of the mouse move event. │ │ │ │ │ - * Subclasses can do their own processing here. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The mouse move event │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - move: function(evt) {}, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: up │ │ │ │ │ - * This method is called during the handling of the mouse up event. │ │ │ │ │ - * Subclasses can do their own processing here. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The mouse up event │ │ │ │ │ - */ │ │ │ │ │ - up: function(evt) {}, │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.ArgParser │ │ │ │ │ + * The ArgParser control adds location bar query string parsing functionality │ │ │ │ │ + * to an OpenLayers Map. │ │ │ │ │ + * When added to a Map control, on a page load/refresh, the Map will │ │ │ │ │ + * automatically take the href string and parse it for lon, lat, zoom, and │ │ │ │ │ + * layers information. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.ArgParser = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: out │ │ │ │ │ - * This method is called during the handling of the mouse out event. │ │ │ │ │ - * Subclasses can do their own processing here. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The mouse out event │ │ │ │ │ + * Property: center │ │ │ │ │ + * {<OpenLayers.LonLat>} │ │ │ │ │ */ │ │ │ │ │ - out: function(evt) {}, │ │ │ │ │ + center: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * The methods below are part of the magic of event handling. Because │ │ │ │ │ - * they are named like browser events, they are registered as listeners │ │ │ │ │ - * for the events they represent. │ │ │ │ │ + * Property: zoom │ │ │ │ │ + * {int} │ │ │ │ │ */ │ │ │ │ │ + zoom: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mousedown │ │ │ │ │ - * Handle mousedown events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * Property: layers │ │ │ │ │ + * {String} Each character represents the state of the corresponding layer │ │ │ │ │ + * on the map. │ │ │ │ │ */ │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - return this.dragstart(evt); │ │ │ │ │ - }, │ │ │ │ │ + layers: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchstart │ │ │ │ │ - * Handle touchstart events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: displayProjection │ │ │ │ │ + * {<OpenLayers.Projection>} Requires proj4js support. │ │ │ │ │ + * Projection used when reading the coordinates from the URL. This will │ │ │ │ │ + * reproject the map coordinates from the URL into the map's │ │ │ │ │ + * projection. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * If you are using this functionality, be aware that any permalink │ │ │ │ │ + * which is added to the map will determine the coordinate type which │ │ │ │ │ + * is read from the URL, which means you should not add permalinks with │ │ │ │ │ + * different displayProjections to the same map. │ │ │ │ │ */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - return this.dragstart(evt); │ │ │ │ │ - }, │ │ │ │ │ + displayProjection: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mousemove │ │ │ │ │ - * Handle mousemove events │ │ │ │ │ + * Constructor: OpenLayers.Control.ArgParser │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - return this.dragmove(evt); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchmove │ │ │ │ │ - * Handle touchmove events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * Method: getParameters │ │ │ │ │ */ │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - return this.dragmove(evt); │ │ │ │ │ - }, │ │ │ │ │ + getParameters: function(url) { │ │ │ │ │ + url = url || window.location.href; │ │ │ │ │ + var parameters = OpenLayers.Util.getParameters(url); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeTimeout │ │ │ │ │ - * Private. Called by mousemove() to remove the drag timeout. │ │ │ │ │ - */ │ │ │ │ │ - removeTimeout: function() { │ │ │ │ │ - this.timeoutId = null; │ │ │ │ │ - // if timeout expires while we're still dragging (mouseup │ │ │ │ │ - // hasn't occurred) then call mousemove to move to the │ │ │ │ │ - // correct position │ │ │ │ │ - if (this.dragging) { │ │ │ │ │ - this.mousemove(this.lastMoveEvt); │ │ │ │ │ + // If we have an anchor in the url use it to split the url │ │ │ │ │ + var index = url.indexOf('#'); │ │ │ │ │ + if (index > 0) { │ │ │ │ │ + // create an url to parse on the getParameters │ │ │ │ │ + url = '?' + url.substring(index + 1, url.length); │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Util.extend(parameters, │ │ │ │ │ + OpenLayers.Util.getParameters(url)); │ │ │ │ │ } │ │ │ │ │ + return parameters; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mouseup │ │ │ │ │ - * Handle mouseup events │ │ │ │ │ - * │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the control. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - return this.dragend(evt); │ │ │ │ │ - }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchend │ │ │ │ │ - * Handle touchend events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - // override evt.xy with last position since touchend does not have │ │ │ │ │ - // any touch position │ │ │ │ │ - evt.xy = this.last; │ │ │ │ │ - return this.dragend(evt); │ │ │ │ │ - }, │ │ │ │ │ + //make sure we dont already have an arg parser attached │ │ │ │ │ + for (var i = 0, len = this.map.controls.length; i < len; i++) { │ │ │ │ │ + var control = this.map.controls[i]; │ │ │ │ │ + if ((control != this) && │ │ │ │ │ + (control.CLASS_NAME == "OpenLayers.Control.ArgParser")) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: mouseout │ │ │ │ │ - * Handle mouseout events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - mouseout: function(evt) { │ │ │ │ │ - if (this.started && OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ - if (this.documentDrag === true) { │ │ │ │ │ - this.addDocumentEvents(); │ │ │ │ │ - } else { │ │ │ │ │ - var dragged = (this.start != this.last); │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - OpenLayers.Element.removeClass( │ │ │ │ │ - this.map.viewPortDiv, "olDragDown" │ │ │ │ │ - ); │ │ │ │ │ - this.out(evt); │ │ │ │ │ - this.callback("out", []); │ │ │ │ │ - if (dragged) { │ │ │ │ │ - this.callback("done", [evt.xy]); │ │ │ │ │ - } │ │ │ │ │ - if (document.onselectstart) { │ │ │ │ │ - document.onselectstart = this.oldOnselectstart; │ │ │ │ │ + // If a second argparser is added to the map, then we │ │ │ │ │ + // override the displayProjection to be the one added to the │ │ │ │ │ + // map. │ │ │ │ │ + if (control.displayProjection != this.displayProjection) { │ │ │ │ │ + this.displayProjection = control.displayProjection; │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ + if (i == this.map.controls.length) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: click │ │ │ │ │ - * The drag handler captures the click event. If something else registers │ │ │ │ │ - * for clicks on the same element, its listener will not be called │ │ │ │ │ - * after a drag. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - // let the click event propagate only if the mouse moved │ │ │ │ │ - return (this.start == this.last); │ │ │ │ │ - }, │ │ │ │ │ + var args = this.getParameters(); │ │ │ │ │ + // Be careful to set layer first, to not trigger unnecessary layer loads │ │ │ │ │ + if (args.layers) { │ │ │ │ │ + this.layers = args.layers; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Activate the handler. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - activated = true; │ │ │ │ │ + // when we add a new layer, set its visibility │ │ │ │ │ + this.map.events.register('addlayer', this, │ │ │ │ │ + this.configureLayers); │ │ │ │ │ + this.configureLayers(); │ │ │ │ │ + } │ │ │ │ │ + if (args.lat && args.lon) { │ │ │ │ │ + this.center = new OpenLayers.LonLat(parseFloat(args.lon), │ │ │ │ │ + parseFloat(args.lat)); │ │ │ │ │ + if (args.zoom) { │ │ │ │ │ + this.zoom = parseFloat(args.zoom); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // when we add a new baselayer to see when we can set the center │ │ │ │ │ + this.map.events.register('changebaselayer', this, │ │ │ │ │ + this.setCenter); │ │ │ │ │ + this.setCenter(); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Deactivate the handler. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully deactivated. │ │ │ │ │ + /** │ │ │ │ │ + * Method: setCenter │ │ │ │ │ + * As soon as a baseLayer has been loaded, we center and zoom │ │ │ │ │ + * ...and remove the handler. │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - deactivated = true; │ │ │ │ │ - OpenLayers.Element.removeClass( │ │ │ │ │ - this.map.viewPortDiv, "olDragDown" │ │ │ │ │ - ); │ │ │ │ │ + setCenter: function() { │ │ │ │ │ + │ │ │ │ │ + if (this.map.baseLayer) { │ │ │ │ │ + //dont need to listen for this one anymore │ │ │ │ │ + this.map.events.unregister('changebaselayer', this, │ │ │ │ │ + this.setCenter); │ │ │ │ │ + │ │ │ │ │ + if (this.displayProjection) { │ │ │ │ │ + this.center.transform(this.displayProjection, │ │ │ │ │ + this.map.getProjectionObject()); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.map.setCenter(this.center, this.zoom); │ │ │ │ │ } │ │ │ │ │ - return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: adjustXY │ │ │ │ │ - * Converts event coordinates that are relative to the document body to │ │ │ │ │ - * ones that are relative to the map viewport. The latter is the default in │ │ │ │ │ - * OpenLayers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ + /** │ │ │ │ │ + * Method: configureLayers │ │ │ │ │ + * As soon as all the layers are loaded, cycle through them and │ │ │ │ │ + * hide or show them. │ │ │ │ │ */ │ │ │ │ │ - adjustXY: function(evt) { │ │ │ │ │ - var pos = OpenLayers.Util.pagePosition(this.map.viewPortDiv); │ │ │ │ │ - evt.xy.x -= pos[0]; │ │ │ │ │ - evt.xy.y -= pos[1]; │ │ │ │ │ - }, │ │ │ │ │ + configureLayers: function() { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: addDocumentEvents │ │ │ │ │ - * Start observing document events when documentDrag is true and the mouse │ │ │ │ │ - * cursor leaves the map viewport while dragging. │ │ │ │ │ - */ │ │ │ │ │ - addDocumentEvents: function() { │ │ │ │ │ - OpenLayers.Element.addClass(document.body, "olDragDown"); │ │ │ │ │ - this.documentEvents = true; │ │ │ │ │ - OpenLayers.Event.observe(document, "mousemove", this._docMove); │ │ │ │ │ - OpenLayers.Event.observe(document, "mouseup", this._docUp); │ │ │ │ │ - }, │ │ │ │ │ + if (this.layers.length == this.map.layers.length) { │ │ │ │ │ + this.map.events.unregister('addlayer', this, this.configureLayers); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeDocumentEvents │ │ │ │ │ - * Stops observing document events when documentDrag is true and the mouse │ │ │ │ │ - * cursor re-enters the map viewport while dragging. │ │ │ │ │ - */ │ │ │ │ │ - removeDocumentEvents: function() { │ │ │ │ │ - OpenLayers.Element.removeClass(document.body, "olDragDown"); │ │ │ │ │ - this.documentEvents = false; │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mousemove", this._docMove); │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mouseup", this._docUp); │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + var c = this.layers.charAt(i); │ │ │ │ │ + │ │ │ │ │ + if (c == "B") { │ │ │ │ │ + this.map.setBaseLayer(layer); │ │ │ │ │ + } else if ((c == "T") || (c == "F")) { │ │ │ │ │ + layer.setVisibility(c == "T"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Drag" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ArgParser" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/RegularPolygon.js │ │ │ │ │ + OpenLayers/Control/WMTSGetFeatureInfo.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler/Drag.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Handler/Click.js │ │ │ │ │ + * @requires OpenLayers/Handler/Hover.js │ │ │ │ │ + * @requires OpenLayers/Request.js │ │ │ │ │ + * @requires OpenLayers/Format/WMSGetFeatureInfo.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Handler.RegularPolygon │ │ │ │ │ - * Handler to draw a regular polygon on the map. Polygon is displayed on mouse │ │ │ │ │ - * down, moves or is modified on mouse move, and is finished on mouse up. │ │ │ │ │ - * The handler triggers callbacks for 'done' and 'cancel'. Create a new │ │ │ │ │ - * instance with the <OpenLayers.Handler.RegularPolygon> constructor. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Control.WMTSGetFeatureInfo │ │ │ │ │ + * The WMTSGetFeatureInfo control uses a WMTS query to get information about a │ │ │ │ │ + * point on the map. The information may be in a display-friendly format │ │ │ │ │ + * such as HTML, or a machine-friendly format such as GML, depending on the │ │ │ │ │ + * server's capabilities and the client's configuration. This control │ │ │ │ │ + * handles click or hover events, attempts to parse the results using an │ │ │ │ │ + * OpenLayers.Format, and fires a 'getfeatureinfo' event for each layer │ │ │ │ │ + * queried. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler.Drag> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.RegularPolygon = OpenLayers.Class(OpenLayers.Handler.Drag, { │ │ │ │ │ +OpenLayers.Control.WMTSGetFeatureInfo = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: sides │ │ │ │ │ - * {Integer} Number of sides for the regular polygon. Needs to be greater │ │ │ │ │ - * than 2. Defaults to 4. │ │ │ │ │ + * APIProperty: hover │ │ │ │ │ + * {Boolean} Send GetFeatureInfo requests when mouse stops moving. │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - sides: 4, │ │ │ │ │ + hover: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: radius │ │ │ │ │ - * {Float} Optional radius in map units of the regular polygon. If this is │ │ │ │ │ - * set to some non-zero value, a polygon with a fixed radius will be │ │ │ │ │ - * drawn and dragged with mose movements. If this property is not │ │ │ │ │ - * set, dragging changes the radius of the polygon. Set to null by │ │ │ │ │ - * default. │ │ │ │ │ + * Property: requestEncoding │ │ │ │ │ + * {String} One of "KVP" or "REST". Only KVP encoding is supported at this │ │ │ │ │ + * time. │ │ │ │ │ */ │ │ │ │ │ - radius: null, │ │ │ │ │ + requestEncoding: "KVP", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: snapAngle │ │ │ │ │ - * {Float} If set to a non-zero value, the handler will snap the polygon │ │ │ │ │ - * rotation to multiples of the snapAngle. Value is an angle measured │ │ │ │ │ - * in degrees counterclockwise from the positive x-axis. │ │ │ │ │ + * APIProperty: drillDown │ │ │ │ │ + * {Boolean} Drill down over all WMTS layers in the map. When │ │ │ │ │ + * using drillDown mode, hover is not possible. A getfeatureinfo event │ │ │ │ │ + * will be fired for each layer queried. │ │ │ │ │ */ │ │ │ │ │ - snapAngle: null, │ │ │ │ │ + drillDown: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: snapToggle │ │ │ │ │ - * {String} If set, snapToggle is checked on mouse events and will set │ │ │ │ │ - * the snap mode to the opposite of what it currently is. To disallow │ │ │ │ │ - * toggling between snap and non-snap mode, set freehandToggle to │ │ │ │ │ - * null. Acceptable toggle values are 'shiftKey', 'ctrlKey', and │ │ │ │ │ - * 'altKey'. Snap mode is only possible if this.snapAngle is set to a │ │ │ │ │ - * non-zero value. │ │ │ │ │ + * APIProperty: maxFeatures │ │ │ │ │ + * {Integer} Maximum number of features to return from a WMTS query. This │ │ │ │ │ + * sets the feature_count parameter on WMTS GetFeatureInfo │ │ │ │ │ + * requests. │ │ │ │ │ */ │ │ │ │ │ - snapToggle: 'shiftKey', │ │ │ │ │ + maxFeatures: 10, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: layerOptions │ │ │ │ │ - * {Object} Any optional properties to be set on the sketch layer. │ │ │ │ │ + /** APIProperty: clickCallback │ │ │ │ │ + * {String} The click callback to register in the │ │ │ │ │ + * {<OpenLayers.Handler.Click>} object created when the hover │ │ │ │ │ + * option is set to false. Default is "click". │ │ │ │ │ */ │ │ │ │ │ - layerOptions: null, │ │ │ │ │ + clickCallback: "click", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: persist │ │ │ │ │ - * {Boolean} Leave the feature rendered until clear is called. Default │ │ │ │ │ - * is false. If set to true, the feature remains rendered until │ │ │ │ │ - * clear is called, typically by deactivating the handler or starting │ │ │ │ │ - * another drawing. │ │ │ │ │ + * Property: layers │ │ │ │ │ + * {Array(<OpenLayers.Layer.WMTS>)} The layers to query for feature info. │ │ │ │ │ + * If omitted, all map WMTS layers will be considered. │ │ │ │ │ */ │ │ │ │ │ - persist: false, │ │ │ │ │ + layers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: irregular │ │ │ │ │ - * {Boolean} Draw an irregular polygon instead of a regular polygon. │ │ │ │ │ - * Default is false. If true, the initial mouse down will represent │ │ │ │ │ - * one corner of the polygon bounds and with each mouse movement, the │ │ │ │ │ - * polygon will be stretched so the opposite corner of its bounds │ │ │ │ │ - * follows the mouse position. This property takes precedence over │ │ │ │ │ - * the radius property. If set to true, the radius property will │ │ │ │ │ - * be ignored. │ │ │ │ │ + * APIProperty: queryVisible │ │ │ │ │ + * {Boolean} Filter out hidden layers when searching the map for layers to │ │ │ │ │ + * query. Default is true. │ │ │ │ │ */ │ │ │ │ │ - irregular: false, │ │ │ │ │ + queryVisible: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: citeCompliant │ │ │ │ │ - * {Boolean} If set to true, coordinates of features drawn in a map extent │ │ │ │ │ - * crossing the date line won't exceed the world bounds. Default is false. │ │ │ │ │ + * Property: infoFormat │ │ │ │ │ + * {String} The mimetype to request from the server │ │ │ │ │ */ │ │ │ │ │ - citeCompliant: false, │ │ │ │ │ + infoFormat: 'text/html', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: angle │ │ │ │ │ - * {Float} The angle from the origin (mouse down) to the current mouse │ │ │ │ │ - * position, in radians. This is measured counterclockwise from the │ │ │ │ │ - * positive x-axis. │ │ │ │ │ + * Property: vendorParams │ │ │ │ │ + * {Object} Additional parameters that will be added to the request, for │ │ │ │ │ + * WMTS implementations that support them. This could e.g. look like │ │ │ │ │ + * (start code) │ │ │ │ │ + * { │ │ │ │ │ + * radius: 5 │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ */ │ │ │ │ │ - angle: null, │ │ │ │ │ + vendorParams: {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: fixedRadius │ │ │ │ │ - * {Boolean} The polygon has a fixed radius. True if a radius is set before │ │ │ │ │ - * drawing begins. False otherwise. │ │ │ │ │ + * Property: format │ │ │ │ │ + * {<OpenLayers.Format>} A format for parsing GetFeatureInfo responses. │ │ │ │ │ + * Default is <OpenLayers.Format.WMSGetFeatureInfo>. │ │ │ │ │ */ │ │ │ │ │ - fixedRadius: false, │ │ │ │ │ + format: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: feature │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} The currently drawn polygon feature │ │ │ │ │ + * Property: formatOptions │ │ │ │ │ + * {Object} Optional properties to set on the format (if one is not provided │ │ │ │ │ + * in the <format> property. │ │ │ │ │ */ │ │ │ │ │ - feature: null, │ │ │ │ │ + formatOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} The temporary drawing layer │ │ │ │ │ + * APIProperty: handlerOptions │ │ │ │ │ + * {Object} Additional options for the handlers used by this control, e.g. │ │ │ │ │ + * (start code) │ │ │ │ │ + * { │ │ │ │ │ + * "click": {delay: 100}, │ │ │ │ │ + * "hover": {delay: 300} │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ */ │ │ │ │ │ - layer: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: origin │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} Location of the first mouse down │ │ │ │ │ + * Property: handler │ │ │ │ │ + * {Object} Reference to the <OpenLayers.Handler> for this control │ │ │ │ │ */ │ │ │ │ │ - origin: null, │ │ │ │ │ + handler: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.RegularPolygon │ │ │ │ │ - * Create a new regular polygon handler. │ │ │ │ │ + * Property: hoverRequest │ │ │ │ │ + * {<OpenLayers.Request>} contains the currently running hover request │ │ │ │ │ + * (if any). │ │ │ │ │ + */ │ │ │ │ │ + hoverRequest: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ - * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ - * functions. Various callbacks described below. │ │ │ │ │ - * options - {Object} An object with properties to be set on the handler. │ │ │ │ │ - * If the options.sides property is not specified, the number of sides │ │ │ │ │ - * will default to 4. │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ * │ │ │ │ │ - * Named callbacks: │ │ │ │ │ - * create - Called when a sketch is first created. Callback called with │ │ │ │ │ - * the creation point geometry and sketch feature. │ │ │ │ │ - * done - Called when the sketch drawing is finished. The callback will │ │ │ │ │ - * recieve a single argument, the sketch geometry. │ │ │ │ │ - * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ - * cancel callback will receive a geometry. │ │ │ │ │ + * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ + * beforegetfeatureinfo - Triggered before each request is sent. │ │ │ │ │ + * The event object has an *xy* property with the position of the │ │ │ │ │ + * mouse click or hover event that triggers the request and a *layer* │ │ │ │ │ + * property referencing the layer about to be queried. If a listener │ │ │ │ │ + * returns false, the request will not be issued. │ │ │ │ │ + * getfeatureinfo - Triggered when a GetFeatureInfo response is received. │ │ │ │ │ + * The event object has a *text* property with the body of the │ │ │ │ │ + * response (String), a *features* property with an array of the │ │ │ │ │ + * parsed features, an *xy* property with the position of the mouse │ │ │ │ │ + * click or hover event that triggered the request, a *layer* property │ │ │ │ │ + * referencing the layer queried and a *request* property with the │ │ │ │ │ + * request itself. If drillDown is set to true, one event will be fired │ │ │ │ │ + * for each layer queried. │ │ │ │ │ + * exception - Triggered when a GetFeatureInfo request fails (with a │ │ │ │ │ + * status other than 200) or whenparsing fails. Listeners will receive │ │ │ │ │ + * an event with *request*, *xy*, and *layer* properties. In the case │ │ │ │ │ + * of a parsing error, the event will also contain an *error* property. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ - this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style['default'], {}); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Handler.Drag.prototype.initialize.apply(this, │ │ │ │ │ - [control, callbacks, options]); │ │ │ │ │ - this.options = (options) ? options : {}; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setOptions │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newOptions - {Object} │ │ │ │ │ + /** │ │ │ │ │ + * Property: pending │ │ │ │ │ + * {Number} The number of pending requests. │ │ │ │ │ */ │ │ │ │ │ - setOptions: function(newOptions) { │ │ │ │ │ - OpenLayers.Util.extend(this.options, newOptions); │ │ │ │ │ - OpenLayers.Util.extend(this, newOptions); │ │ │ │ │ - }, │ │ │ │ │ + pending: 0, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Turn on the handler. │ │ │ │ │ + * Constructor: <OpenLayers.Control.WMTSGetFeatureInfo> │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully activated │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.Drag.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - // create temporary vector layer for rendering geometry sketch │ │ │ │ │ - var options = OpenLayers.Util.extend({ │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - // indicate that the temp vector layer will never be out of range │ │ │ │ │ - // without this, resolution properties must be specified at the │ │ │ │ │ - // map-level for this temporary layer to init its resolutions │ │ │ │ │ - // correctly │ │ │ │ │ - calculateInRange: OpenLayers.Function.True, │ │ │ │ │ - wrapDateLine: this.citeCompliant │ │ │ │ │ - }, this.layerOptions); │ │ │ │ │ - this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ - this.map.addLayer(this.layer); │ │ │ │ │ - activated = true; │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + │ │ │ │ │ + if (!this.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.WMSGetFeatureInfo( │ │ │ │ │ + options.formatOptions │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.drillDown === true) { │ │ │ │ │ + this.hover = false; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Hover( │ │ │ │ │ + this, { │ │ │ │ │ + move: this.cancelHover, │ │ │ │ │ + pause: this.getInfoForHover │ │ │ │ │ + }, │ │ │ │ │ + OpenLayers.Util.extend( │ │ │ │ │ + this.handlerOptions.hover || {}, { │ │ │ │ │ + delay: 250 │ │ │ │ │ + } │ │ │ │ │ + ) │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + var callbacks = {}; │ │ │ │ │ + callbacks[this.clickCallback] = this.getInfoForClick; │ │ │ │ │ + this.handler = new OpenLayers.Handler.Click( │ │ │ │ │ + this, callbacks, this.handlerOptions.click || {} │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Turn off the handler. │ │ │ │ │ + * Method: getInfoForClick │ │ │ │ │ + * Called on click │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully deactivated │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.Drag.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - // call the cancel callback if mid-drawing │ │ │ │ │ - if (this.dragging) { │ │ │ │ │ - this.cancel(); │ │ │ │ │ - } │ │ │ │ │ - // If a layer's map property is set to null, it means that that │ │ │ │ │ - // layer isn't added to the map. Since we ourself added the layer │ │ │ │ │ - // to the map in activate(), we can assume that if this.layer.map │ │ │ │ │ - // is null it means that the layer has been destroyed (as a result │ │ │ │ │ - // of map.destroy() for example. │ │ │ │ │ - if (this.layer.map != null) { │ │ │ │ │ - this.layer.destroy(false); │ │ │ │ │ - if (this.feature) { │ │ │ │ │ - this.feature.destroy(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.layer = null; │ │ │ │ │ - this.feature = null; │ │ │ │ │ - deactivated = true; │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ + getInfoForClick: function(evt) { │ │ │ │ │ + this.request(evt.xy, {}); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: down │ │ │ │ │ - * Start drawing a new feature │ │ │ │ │ + * Method: getInfoForHover │ │ │ │ │ + * Pause callback for the hover handler │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} The drag start event │ │ │ │ │ + * evt - {Object} │ │ │ │ │ */ │ │ │ │ │ - down: function(evt) { │ │ │ │ │ - this.fixedRadius = !!(this.radius); │ │ │ │ │ - var maploc = this.layer.getLonLatFromViewPortPx(evt.xy); │ │ │ │ │ - this.origin = new OpenLayers.Geometry.Point(maploc.lon, maploc.lat); │ │ │ │ │ - // create the new polygon │ │ │ │ │ - if (!this.fixedRadius || this.irregular) { │ │ │ │ │ - // smallest radius should not be less one pixel in map units │ │ │ │ │ - // VML doesn't behave well with smaller │ │ │ │ │ - this.radius = this.map.getResolution(); │ │ │ │ │ - } │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.clear(); │ │ │ │ │ - } │ │ │ │ │ - this.feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ - this.createGeometry(); │ │ │ │ │ - this.callback("create", [this.origin, this.feature]); │ │ │ │ │ - this.layer.addFeatures([this.feature], { │ │ │ │ │ - silent: true │ │ │ │ │ + getInfoForHover: function(evt) { │ │ │ │ │ + this.request(evt.xy, { │ │ │ │ │ + hover: true │ │ │ │ │ }); │ │ │ │ │ - this.layer.drawFeature(this.feature, this.style); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * Respond to drag move events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Evt} The move event │ │ │ │ │ + * Method: cancelHover │ │ │ │ │ + * Cancel callback for the hover handler │ │ │ │ │ */ │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - var maploc = this.layer.getLonLatFromViewPortPx(evt.xy); │ │ │ │ │ - var point = new OpenLayers.Geometry.Point(maploc.lon, maploc.lat); │ │ │ │ │ - if (this.irregular) { │ │ │ │ │ - var ry = Math.sqrt(2) * Math.abs(point.y - this.origin.y) / 2; │ │ │ │ │ - this.radius = Math.max(this.map.getResolution() / 2, ry); │ │ │ │ │ - } else if (this.fixedRadius) { │ │ │ │ │ - this.origin = point; │ │ │ │ │ - } else { │ │ │ │ │ - this.calculateAngle(point, evt); │ │ │ │ │ - this.radius = Math.max(this.map.getResolution() / 2, │ │ │ │ │ - point.distanceTo(this.origin)); │ │ │ │ │ - } │ │ │ │ │ - this.modifyGeometry(); │ │ │ │ │ - if (this.irregular) { │ │ │ │ │ - var dx = point.x - this.origin.x; │ │ │ │ │ - var dy = point.y - this.origin.y; │ │ │ │ │ - var ratio; │ │ │ │ │ - if (dy == 0) { │ │ │ │ │ - ratio = dx / (this.radius * Math.sqrt(2)); │ │ │ │ │ - } else { │ │ │ │ │ - ratio = dx / dy; │ │ │ │ │ + cancelHover: function() { │ │ │ │ │ + if (this.hoverRequest) { │ │ │ │ │ + --this.pending; │ │ │ │ │ + if (this.pending <= 0) { │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + this.pending = 0; │ │ │ │ │ } │ │ │ │ │ - this.feature.geometry.resize(1, this.origin, ratio); │ │ │ │ │ - this.feature.geometry.move(dx / 2, dy / 2); │ │ │ │ │ + this.hoverRequest.abort(); │ │ │ │ │ + this.hoverRequest = null; │ │ │ │ │ } │ │ │ │ │ - this.layer.drawFeature(this.feature, this.style); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: up │ │ │ │ │ - * Finish drawing the feature │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The mouse up event │ │ │ │ │ + * Method: findLayers │ │ │ │ │ + * Internal method to get the layers, independent of whether we are │ │ │ │ │ + * inspecting the map or using a client-provided array │ │ │ │ │ */ │ │ │ │ │ - up: function(evt) { │ │ │ │ │ - this.finalize(); │ │ │ │ │ - // the mouseup method of superclass doesn't call the │ │ │ │ │ - // "done" callback if there's been no move between │ │ │ │ │ - // down and up │ │ │ │ │ - if (this.start == this.last) { │ │ │ │ │ - this.callback("done", [evt.xy]); │ │ │ │ │ + findLayers: function() { │ │ │ │ │ + var candidates = this.layers || this.map.layers; │ │ │ │ │ + var layers = []; │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = candidates[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.WMTS && │ │ │ │ │ + layer.requestEncoding === this.requestEncoding && │ │ │ │ │ + (!this.queryVisible || layer.getVisibility())) { │ │ │ │ │ + layers.push(layer); │ │ │ │ │ + if (!this.drillDown || this.hover) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return layers; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: out │ │ │ │ │ - * Finish drawing the feature. │ │ │ │ │ + * Method: buildRequestOptions │ │ │ │ │ + * Build an object with the relevant options for the GetFeatureInfo request. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} The mouse out event │ │ │ │ │ - */ │ │ │ │ │ - out: function(evt) { │ │ │ │ │ - this.finalize(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createGeometry │ │ │ │ │ - * Create the new polygon geometry. This is called at the start of the │ │ │ │ │ - * drag and at any point during the drag if the number of sides │ │ │ │ │ - * changes. │ │ │ │ │ + * layer - {<OpenLayers.Layer.WMTS>} A WMTS layer. │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} The position on the map where the │ │ │ │ │ + * mouse event occurred. │ │ │ │ │ */ │ │ │ │ │ - createGeometry: function() { │ │ │ │ │ - this.angle = Math.PI * ((1 / this.sides) - (1 / 2)); │ │ │ │ │ - if (this.snapAngle) { │ │ │ │ │ - this.angle += this.snapAngle * (Math.PI / 180); │ │ │ │ │ - } │ │ │ │ │ - this.feature.geometry = OpenLayers.Geometry.Polygon.createRegularPolygon( │ │ │ │ │ - this.origin, this.radius, this.sides, this.snapAngle │ │ │ │ │ + buildRequestOptions: function(layer, xy) { │ │ │ │ │ + var loc = this.map.getLonLatFromPixel(xy); │ │ │ │ │ + var getTileUrl = layer.getURL( │ │ │ │ │ + new OpenLayers.Bounds(loc.lon, loc.lat, loc.lon, loc.lat) │ │ │ │ │ ); │ │ │ │ │ + var params = OpenLayers.Util.getParameters(getTileUrl); │ │ │ │ │ + var tileInfo = layer.getTileInfo(loc); │ │ │ │ │ + OpenLayers.Util.extend(params, { │ │ │ │ │ + service: "WMTS", │ │ │ │ │ + version: layer.version, │ │ │ │ │ + request: "GetFeatureInfo", │ │ │ │ │ + infoFormat: this.infoFormat, │ │ │ │ │ + i: tileInfo.i, │ │ │ │ │ + j: tileInfo.j │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Util.applyDefaults(params, this.vendorParams); │ │ │ │ │ + return { │ │ │ │ │ + url: OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url, │ │ │ │ │ + params: OpenLayers.Util.upperCaseObject(params), │ │ │ │ │ + callback: function(request) { │ │ │ │ │ + this.handleResponse(xy, request, layer); │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: modifyGeometry │ │ │ │ │ - * Modify the polygon geometry in place. │ │ │ │ │ + * Method: request │ │ │ │ │ + * Sends a GetFeatureInfo request to the WMTS │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} The position on the map where the mouse event │ │ │ │ │ + * occurred. │ │ │ │ │ + * options - {Object} additional options for this method. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * - *hover* {Boolean} true if we do the request for the hover handler │ │ │ │ │ */ │ │ │ │ │ - modifyGeometry: function() { │ │ │ │ │ - var angle, point; │ │ │ │ │ - var ring = this.feature.geometry.components[0]; │ │ │ │ │ - // if the number of sides ever changes, create a new geometry │ │ │ │ │ - if (ring.components.length != (this.sides + 1)) { │ │ │ │ │ - this.createGeometry(); │ │ │ │ │ - ring = this.feature.geometry.components[0]; │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0; i < this.sides; ++i) { │ │ │ │ │ - point = ring.components[i]; │ │ │ │ │ - angle = this.angle + (i * 2 * Math.PI / this.sides); │ │ │ │ │ - point.x = this.origin.x + (this.radius * Math.cos(angle)); │ │ │ │ │ - point.y = this.origin.y + (this.radius * Math.sin(angle)); │ │ │ │ │ - point.clearBounds(); │ │ │ │ │ + request: function(xy, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var layers = this.findLayers(); │ │ │ │ │ + if (layers.length > 0) { │ │ │ │ │ + var issue, layer; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + layer = layers[i]; │ │ │ │ │ + issue = this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ + xy: xy, │ │ │ │ │ + layer: layer │ │ │ │ │ + }); │ │ │ │ │ + if (issue !== false) { │ │ │ │ │ + ++this.pending; │ │ │ │ │ + var requestOptions = this.buildRequestOptions(layer, xy); │ │ │ │ │ + var request = OpenLayers.Request.GET(requestOptions); │ │ │ │ │ + if (options.hover === true) { │ │ │ │ │ + this.hoverRequest = request; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.pending > 0) { │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: calculateAngle │ │ │ │ │ - * Calculate the angle based on settings. │ │ │ │ │ - * │ │ │ │ │ + * Method: handleResponse │ │ │ │ │ + * Handler for the GetFeatureInfo response. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} The position on the map where the mouse event │ │ │ │ │ + * occurred. │ │ │ │ │ + * request - {XMLHttpRequest} The request object. │ │ │ │ │ + * layer - {<OpenLayers.Layer.WMTS>} The queried layer. │ │ │ │ │ */ │ │ │ │ │ - calculateAngle: function(point, evt) { │ │ │ │ │ - var alpha = Math.atan2(point.y - this.origin.y, │ │ │ │ │ - point.x - this.origin.x); │ │ │ │ │ - if (this.snapAngle && (this.snapToggle && !evt[this.snapToggle])) { │ │ │ │ │ - var snapAngleRad = (Math.PI / 180) * this.snapAngle; │ │ │ │ │ - this.angle = Math.round(alpha / snapAngleRad) * snapAngleRad; │ │ │ │ │ + handleResponse: function(xy, request, layer) { │ │ │ │ │ + --this.pending; │ │ │ │ │ + if (this.pending <= 0) { │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + this.pending = 0; │ │ │ │ │ + } │ │ │ │ │ + if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ + this.events.triggerEvent("exception", { │ │ │ │ │ + xy: xy, │ │ │ │ │ + request: request, │ │ │ │ │ + layer: layer │ │ │ │ │ + }); │ │ │ │ │ } else { │ │ │ │ │ - this.angle = alpha; │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ + } │ │ │ │ │ + var features, except; │ │ │ │ │ + try { │ │ │ │ │ + features = this.format.read(doc); │ │ │ │ │ + } catch (error) { │ │ │ │ │ + except = true; │ │ │ │ │ + this.events.triggerEvent("exception", { │ │ │ │ │ + xy: xy, │ │ │ │ │ + request: request, │ │ │ │ │ + error: error, │ │ │ │ │ + layer: layer │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + if (!except) { │ │ │ │ │ + this.events.triggerEvent("getfeatureinfo", { │ │ │ │ │ + text: request.responseText, │ │ │ │ │ + features: features, │ │ │ │ │ + request: request, │ │ │ │ │ + xy: xy, │ │ │ │ │ + layer: layer │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: cancel │ │ │ │ │ - * Finish the geometry and call the "cancel" callback. │ │ │ │ │ - */ │ │ │ │ │ - cancel: function() { │ │ │ │ │ - // the polygon geometry gets cloned in the callback method │ │ │ │ │ - this.callback("cancel", null); │ │ │ │ │ - this.finalize(); │ │ │ │ │ - }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.WMTSGetFeatureInfo" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/Button.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.Button │ │ │ │ │ + * The Button control is a very simple push-button, for use with │ │ │ │ │ + * <OpenLayers.Control.Panel>. │ │ │ │ │ + * When clicked, the function trigger() is executed. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + * │ │ │ │ │ + * Use: │ │ │ │ │ + * (code) │ │ │ │ │ + * var button = new OpenLayers.Control.Button({ │ │ │ │ │ + * displayClass: "MyButton", trigger: myFunction │ │ │ │ │ + * }); │ │ │ │ │ + * panel.addControls([button]); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Will create a button with CSS class MyButtonItemInactive, that │ │ │ │ │ + * will call the function MyFunction() when clicked. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.Button = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ /** │ │ │ │ │ - * Method: finalize │ │ │ │ │ - * Finish the geometry and call the "done" callback. │ │ │ │ │ + * Property: type │ │ │ │ │ + * {Integer} OpenLayers.Control.TYPE_BUTTON. │ │ │ │ │ */ │ │ │ │ │ - finalize: function() { │ │ │ │ │ - this.origin = null; │ │ │ │ │ - this.radius = this.options.radius; │ │ │ │ │ - }, │ │ │ │ │ + type: OpenLayers.Control.TYPE_BUTTON, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clear │ │ │ │ │ - * Clear any rendered features on the temporary layer. This is called │ │ │ │ │ - * when the handler is deactivated, canceled, or done (unless persist │ │ │ │ │ - * is true). │ │ │ │ │ + * Method: trigger │ │ │ │ │ + * Called by a control panel when the button is clicked. │ │ │ │ │ */ │ │ │ │ │ - clear: function() { │ │ │ │ │ - if (this.layer) { │ │ │ │ │ - this.layer.renderer.clear(); │ │ │ │ │ - this.layer.destroyFeatures(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + trigger: function() {}, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Button" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/ZoomOut.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control/Button.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.ZoomOut │ │ │ │ │ + * The ZoomOut control is a button to decrease the zoom level of a map. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.ZoomOut = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: callback │ │ │ │ │ - * Trigger the control's named callback with the given arguments │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} The key for the callback that is one of the properties │ │ │ │ │ - * of the handler's callbacks object. │ │ │ │ │ - * args - {Array} An array of arguments with which to call the callback │ │ │ │ │ - * (defined by the control). │ │ │ │ │ + * Method: trigger │ │ │ │ │ */ │ │ │ │ │ - callback: function(name, args) { │ │ │ │ │ - // override the callback method to always send the polygon geometry │ │ │ │ │ - if (this.callbacks[name]) { │ │ │ │ │ - this.callbacks[name].apply(this.control, │ │ │ │ │ - [this.feature.geometry.clone()]); │ │ │ │ │ - } │ │ │ │ │ - // since sketch features are added to the temporary layer │ │ │ │ │ - // they must be cleared here if done or cancel │ │ │ │ │ - if (!this.persist && (name == "done" || name == "cancel")) { │ │ │ │ │ - this.clear(); │ │ │ │ │ + trigger: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.zoomOut(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.RegularPolygon" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomOut" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Click.js │ │ │ │ │ + OpenLayers/Layer/WMS.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Handler.Click │ │ │ │ │ - * A handler for mouse clicks. The intention of this handler is to give │ │ │ │ │ - * controls more flexibility with handling clicks. Browsers trigger │ │ │ │ │ - * click events twice for a double-click. In addition, the mousedown, │ │ │ │ │ - * mousemove, mouseup sequence fires a click event. With this handler, │ │ │ │ │ - * controls can decide whether to ignore clicks associated with a double │ │ │ │ │ - * click. By setting a <pixelTolerance>, controls can also ignore clicks │ │ │ │ │ - * that include a drag. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Handler.Click> constructor. │ │ │ │ │ + * Class: OpenLayers.Layer.WMS │ │ │ │ │ + * Instances of OpenLayers.Layer.WMS are used to display data from OGC Web │ │ │ │ │ + * Mapping Services. Create a new WMS layer with the <OpenLayers.Layer.WMS> │ │ │ │ │ + * constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.Click = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: delay │ │ │ │ │ - * {Number} Number of milliseconds between clicks before the event is │ │ │ │ │ - * considered a double-click. │ │ │ │ │ - */ │ │ │ │ │ - delay: 300, │ │ │ │ │ +OpenLayers.Layer.WMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: single │ │ │ │ │ - * {Boolean} Handle single clicks. Default is true. If false, clicks │ │ │ │ │ - * will not be reported. If true, single-clicks will be reported. │ │ │ │ │ + * Constant: DEFAULT_PARAMS │ │ │ │ │ + * {Object} Hashtable of default parameter key/value pairs │ │ │ │ │ */ │ │ │ │ │ - single: true, │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + service: "WMS", │ │ │ │ │ + version: "1.1.1", │ │ │ │ │ + request: "GetMap", │ │ │ │ │ + styles: "", │ │ │ │ │ + format: "image/jpeg" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: double │ │ │ │ │ - * {Boolean} Handle double-clicks. Default is false. │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} Default is true for WMS layer │ │ │ │ │ */ │ │ │ │ │ - 'double': false, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: pixelTolerance │ │ │ │ │ - * {Number} Maximum number of pixels between mouseup and mousedown for an │ │ │ │ │ - * event to be considered a click. Default is 0. If set to an │ │ │ │ │ - * integer value, clicks with a drag greater than the value will be │ │ │ │ │ - * ignored. This property can only be set when the handler is │ │ │ │ │ - * constructed. │ │ │ │ │ + * APIProperty: encodeBBOX │ │ │ │ │ + * {Boolean} Should the BBOX commas be encoded? The WMS spec says 'no', │ │ │ │ │ + * but some services want it that way. Default false. │ │ │ │ │ */ │ │ │ │ │ - pixelTolerance: 0, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: dblclickTolerance │ │ │ │ │ - * {Number} Maximum distance in pixels between clicks for a sequence of │ │ │ │ │ - * events to be considered a double click. Default is 13. If the │ │ │ │ │ - * distance between two clicks is greater than this value, a double- │ │ │ │ │ - * click will not be fired. │ │ │ │ │ - */ │ │ │ │ │ - dblclickTolerance: 13, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: stopSingle │ │ │ │ │ - * {Boolean} Stop other listeners from being notified of clicks. Default │ │ │ │ │ - * is false. If true, any listeners registered before this one for │ │ │ │ │ - * click or rightclick events will not be notified. │ │ │ │ │ - */ │ │ │ │ │ - stopSingle: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: stopDouble │ │ │ │ │ - * {Boolean} Stop other listeners from being notified of double-clicks. │ │ │ │ │ - * Default is false. If true, any click listeners registered before │ │ │ │ │ - * this one will not be notified of *any* double-click events. │ │ │ │ │ - * │ │ │ │ │ - * The one caveat with stopDouble is that given a map with two click │ │ │ │ │ - * handlers, one with stopDouble true and the other with stopSingle │ │ │ │ │ - * true, the stopSingle handler should be activated last to get │ │ │ │ │ - * uniform cross-browser performance. Since IE triggers one click │ │ │ │ │ - * with a dblclick and FF triggers two, if a stopSingle handler is │ │ │ │ │ - * activated first, all it gets in IE is a single click when the │ │ │ │ │ - * second handler stops propagation on the dblclick. │ │ │ │ │ - */ │ │ │ │ │ - stopDouble: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: timerId │ │ │ │ │ - * {Number} The id of the timeout waiting to clear the <delayedCall>. │ │ │ │ │ - */ │ │ │ │ │ - timerId: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: down │ │ │ │ │ - * {Object} Object that store relevant information about the last │ │ │ │ │ - * mousedown or touchstart. Its 'xy' OpenLayers.Pixel property gives │ │ │ │ │ - * the average location of the mouse/touch event. Its 'touches' │ │ │ │ │ - * property records clientX/clientY of each touches. │ │ │ │ │ - */ │ │ │ │ │ - down: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: last │ │ │ │ │ - * {Object} Object that store relevant information about the last │ │ │ │ │ - * mousemove or touchmove. Its 'xy' OpenLayers.Pixel property gives │ │ │ │ │ - * the average location of the mouse/touch event. Its 'touches' │ │ │ │ │ - * property records clientX/clientY of each touches. │ │ │ │ │ - */ │ │ │ │ │ - last: null, │ │ │ │ │ + encodeBBOX: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: first │ │ │ │ │ - * {Object} When waiting for double clicks, this object will store │ │ │ │ │ - * information about the first click in a two click sequence. │ │ │ │ │ - */ │ │ │ │ │ - first: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: rightclickTimerId │ │ │ │ │ - * {Number} The id of the right mouse timeout waiting to clear the │ │ │ │ │ - * <delayedEvent>. │ │ │ │ │ + * APIProperty: noMagic │ │ │ │ │ + * {Boolean} If true, the image format will not be automagicaly switched │ │ │ │ │ + * from image/jpeg to image/png or image/gif when using │ │ │ │ │ + * TRANSPARENT=TRUE. Also isBaseLayer will not changed by the │ │ │ │ │ + * constructor. Default false. │ │ │ │ │ */ │ │ │ │ │ - rightclickTimerId: null, │ │ │ │ │ + noMagic: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Click │ │ │ │ │ - * Create a new click handler. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ - * this handler. If a handler is being used without a control, the │ │ │ │ │ - * handler's setMap method must be overridden to deal properly with │ │ │ │ │ - * the map. │ │ │ │ │ - * callbacks - {Object} An object with keys corresponding to callbacks │ │ │ │ │ - * that will be called by the handler. The callbacks should │ │ │ │ │ - * expect to recieve a single argument, the click event. │ │ │ │ │ - * Callbacks for 'click' and 'dblclick' are supported. │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * handler. │ │ │ │ │ + * Property: yx │ │ │ │ │ + * {Object} Keys in this object are EPSG codes for which the axis order │ │ │ │ │ + * is to be reversed (yx instead of xy, LatLon instead of LonLat), with │ │ │ │ │ + * true as value. This is only relevant for WMS versions >= 1.3.0, and │ │ │ │ │ + * only if yx is not set in <OpenLayers.Projection.defaults> for the │ │ │ │ │ + * used projection. │ │ │ │ │ */ │ │ │ │ │ + yx: {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchstart │ │ │ │ │ - * Handle touchstart. │ │ │ │ │ + * Constructor: OpenLayers.Layer.WMS │ │ │ │ │ + * Create a new WMS layer object │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ - */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - this.down = this.getEventInfo(evt); │ │ │ │ │ - this.last = this.getEventInfo(evt); │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchmove │ │ │ │ │ - * Store position of last move, because touchend event can have │ │ │ │ │ - * an empty "touches" property. │ │ │ │ │ + * Examples: │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ - */ │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - this.last = this.getEventInfo(evt); │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchend │ │ │ │ │ - * Correctly set event xy property, and add lastTouches to have │ │ │ │ │ - * touches property from last touchstart or touchmove │ │ │ │ │ + * The code below creates a simple WMS layer using the image/jpeg format. │ │ │ │ │ + * (code) │ │ │ │ │ + * var wms = new OpenLayers.Layer.WMS("NASA Global Mosaic", │ │ │ │ │ + * "http://wms.jpl.nasa.gov/wms.cgi", │ │ │ │ │ + * {layers: "modis,global_mosaic"}); │ │ │ │ │ + * (end) │ │ │ │ │ + * Note the 3rd argument (params). Properties added to this object will be │ │ │ │ │ + * added to the WMS GetMap requests used for this layer's tiles. The only │ │ │ │ │ + * mandatory parameter is "layers". Other common WMS params include │ │ │ │ │ + * "transparent", "styles" and "format". Note that the "srs" param will │ │ │ │ │ + * always be ignored. Instead, it will be derived from the baseLayer's or │ │ │ │ │ + * map's projection. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ - */ │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - // touchstart may not have been allowed to propagate │ │ │ │ │ - if (this.down) { │ │ │ │ │ - evt.xy = this.last.xy; │ │ │ │ │ - evt.lastTouches = this.last.touches; │ │ │ │ │ - this.handleSingle(evt); │ │ │ │ │ - this.down = null; │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mousedown │ │ │ │ │ - * Handle mousedown. │ │ │ │ │ + * The code below creates a transparent WMS layer with additional options. │ │ │ │ │ + * (code) │ │ │ │ │ + * var wms = new OpenLayers.Layer.WMS("NASA Global Mosaic", │ │ │ │ │ + * "http://wms.jpl.nasa.gov/wms.cgi", │ │ │ │ │ + * { │ │ │ │ │ + * layers: "modis,global_mosaic", │ │ │ │ │ + * transparent: true │ │ │ │ │ + * }, { │ │ │ │ │ + * opacity: 0.5, │ │ │ │ │ + * singleTile: true │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * Note that by default, a WMS layer is configured as baseLayer. Setting │ │ │ │ │ + * the "transparent" param to true will apply some magic (see <noMagic>). │ │ │ │ │ + * The default image format changes from image/jpeg to image/png, and the │ │ │ │ │ + * layer is not configured as baseLayer. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ - */ │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - this.down = this.getEventInfo(evt); │ │ │ │ │ - this.last = this.getEventInfo(evt); │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mouseup │ │ │ │ │ - * Handle mouseup. Installed to support collection of right mouse events. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} A name for the layer │ │ │ │ │ + * url - {String} Base url for the WMS │ │ │ │ │ + * (e.g. http://wms.jpl.nasa.gov/wms.cgi) │ │ │ │ │ + * params - {Object} An object with key/value pairs representing the │ │ │ │ │ + * GetMap query string parameters and parameter values. │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer. │ │ │ │ │ + * These options include all properties listed above, plus the ones │ │ │ │ │ + * inherited from superclasses. │ │ │ │ │ */ │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - │ │ │ │ │ - // Collect right mouse clicks from the mouseup │ │ │ │ │ - // IE - ignores the second right click in mousedown so using │ │ │ │ │ - // mouseup instead │ │ │ │ │ - if (this.checkModifiers(evt) && this.control.handleRightClicks && │ │ │ │ │ - OpenLayers.Event.isRightClick(evt)) { │ │ │ │ │ - propagate = this.rightclick(evt); │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + //uppercase params │ │ │ │ │ + params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ + if (parseFloat(params.VERSION) >= 1.3 && !params.EXCEPTIONS) { │ │ │ │ │ + params.EXCEPTIONS = "INIMAGE"; │ │ │ │ │ } │ │ │ │ │ + newArguments.push(name, url, params, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, │ │ │ │ │ + OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS) │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - return propagate; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: rightclick │ │ │ │ │ - * Handle rightclick. For a dblrightclick, we get two clicks so we need │ │ │ │ │ - * to always register for dblrightclick to properly handle single │ │ │ │ │ - * clicks. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ - */ │ │ │ │ │ - rightclick: function(evt) { │ │ │ │ │ - if (this.passesTolerance(evt)) { │ │ │ │ │ - if (this.rightclickTimerId != null) { │ │ │ │ │ - //Second click received before timeout this must be │ │ │ │ │ - // a double click │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.callback('dblrightclick', [evt]); │ │ │ │ │ - return !this.stopDouble; │ │ │ │ │ - } else { │ │ │ │ │ - //Set the rightclickTimerId, send evt only if double is │ │ │ │ │ - // true else trigger single │ │ │ │ │ - var clickEvent = this['double'] ? │ │ │ │ │ - OpenLayers.Util.extend({}, evt) : │ │ │ │ │ - this.callback('rightclick', [evt]); │ │ │ │ │ + //layer is transparent │ │ │ │ │ + if (!this.noMagic && this.params.TRANSPARENT && │ │ │ │ │ + this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ │ │ │ │ │ - var delayedRightCall = OpenLayers.Function.bind( │ │ │ │ │ - this.delayedRightCall, │ │ │ │ │ - this, │ │ │ │ │ - clickEvent │ │ │ │ │ - ); │ │ │ │ │ - this.rightclickTimerId = window.setTimeout( │ │ │ │ │ - delayedRightCall, this.delay │ │ │ │ │ - ); │ │ │ │ │ + // unless explicitly set in options, make layer an overlay │ │ │ │ │ + if ((options == null) || (!options.isBaseLayer)) { │ │ │ │ │ + this.isBaseLayer = false; │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return !this.stopSingle; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: delayedRightCall │ │ │ │ │ - * Sets <rightclickTimerId> to null. And optionally triggers the │ │ │ │ │ - * rightclick callback if evt is set. │ │ │ │ │ - */ │ │ │ │ │ - delayedRightCall: function(evt) { │ │ │ │ │ - this.rightclickTimerId = null; │ │ │ │ │ - if (evt) { │ │ │ │ │ - this.callback('rightclick', [evt]); │ │ │ │ │ + // jpegs can never be transparent, so intelligently switch the │ │ │ │ │ + // format, depending on the browser's capabilities │ │ │ │ │ + if (this.params.FORMAT == "image/jpeg") { │ │ │ │ │ + this.params.FORMAT = OpenLayers.Util.alphaHack() ? "image/gif" : │ │ │ │ │ + "image/png"; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: click │ │ │ │ │ - * Handle click events from the browser. This is registered as a listener │ │ │ │ │ - * for click events and should not be called from other events in this │ │ │ │ │ - * handler. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ - */ │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - if (!this.last) { │ │ │ │ │ - this.last = this.getEventInfo(evt); │ │ │ │ │ - } │ │ │ │ │ - this.handleSingle(evt); │ │ │ │ │ - return !this.stopSingle; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: dblclick │ │ │ │ │ - * Handle dblclick. For a dblclick, we get two clicks in some browsers │ │ │ │ │ - * (FF) and one in others (IE). So we need to always register for │ │ │ │ │ - * dblclick to properly handle single clicks. This method is registered │ │ │ │ │ - * as a listener for the dblclick browser event. It should *not* be │ │ │ │ │ - * called by other methods in this handler. │ │ │ │ │ - * │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ + * {<OpenLayers.Layer.WMS>} An exact clone of this layer │ │ │ │ │ */ │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - this.handleDouble(evt); │ │ │ │ │ - return !this.stopDouble; │ │ │ │ │ - }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleDouble │ │ │ │ │ - * Handle double-click sequence. │ │ │ │ │ - */ │ │ │ │ │ - handleDouble: function(evt) { │ │ │ │ │ - if (this.passesDblclickTolerance(evt)) { │ │ │ │ │ - if (this["double"]) { │ │ │ │ │ - this.callback("dblclick", [evt]); │ │ │ │ │ - } │ │ │ │ │ - // to prevent a dblclick from firing the click callback in IE │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.WMS(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.params, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleSingle │ │ │ │ │ - * Handle single click sequence. │ │ │ │ │ - */ │ │ │ │ │ - handleSingle: function(evt) { │ │ │ │ │ - if (this.passesTolerance(evt)) { │ │ │ │ │ - if (this.timerId != null) { │ │ │ │ │ - // already received a click │ │ │ │ │ - if (this.last.touches && this.last.touches.length === 1) { │ │ │ │ │ - // touch device, no dblclick event - this may be a double │ │ │ │ │ - if (this["double"]) { │ │ │ │ │ - // on Android don't let the browser zoom on the page │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ - } │ │ │ │ │ - this.handleDouble(evt); │ │ │ │ │ - } │ │ │ │ │ - // if we're not in a touch environment we clear the click timer │ │ │ │ │ - // if we've got a second touch, we'll get two touchend events │ │ │ │ │ - if (!this.last.touches || this.last.touches.length !== 2) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - // remember the first click info so we can compare to the second │ │ │ │ │ - this.first = this.getEventInfo(evt); │ │ │ │ │ - // set the timer, send evt only if single is true │ │ │ │ │ - //use a clone of the event object because it will no longer │ │ │ │ │ - //be a valid event object in IE in the timer callback │ │ │ │ │ - var clickEvent = this.single ? │ │ │ │ │ - OpenLayers.Util.extend({}, evt) : null; │ │ │ │ │ - this.queuePotentialClick(clickEvent); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: queuePotentialClick │ │ │ │ │ - * This method is separated out largely to make testing easier (so we │ │ │ │ │ - * don't have to override window.setTimeout) │ │ │ │ │ - */ │ │ │ │ │ - queuePotentialClick: function(evt) { │ │ │ │ │ - this.timerId = window.setTimeout( │ │ │ │ │ - OpenLayers.Function.bind(this.delayedCall, this, evt), │ │ │ │ │ - this.delay │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: passesTolerance │ │ │ │ │ - * Determine whether the event is within the optional pixel tolerance. Note │ │ │ │ │ - * that the pixel tolerance check only works if mousedown events get to │ │ │ │ │ - * the listeners registered here. If they are stopped by other elements, │ │ │ │ │ - * the <pixelTolerance> will have no effect here (this method will always │ │ │ │ │ - * return true). │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The click is within the pixel tolerance (if specified). │ │ │ │ │ - */ │ │ │ │ │ - passesTolerance: function(evt) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - if (this.pixelTolerance != null && this.down && this.down.xy) { │ │ │ │ │ - passes = this.pixelTolerance >= this.down.xy.distanceTo(evt.xy); │ │ │ │ │ - // for touch environments, we also enforce that all touches │ │ │ │ │ - // start and end within the given tolerance to be considered a click │ │ │ │ │ - if (passes && this.touch && │ │ │ │ │ - this.down.touches.length === this.last.touches.length) { │ │ │ │ │ - // the touchend event doesn't come with touches, so we check │ │ │ │ │ - // down and last │ │ │ │ │ - for (var i = 0, ii = this.down.touches.length; i < ii; ++i) { │ │ │ │ │ - if (this.getTouchDistance( │ │ │ │ │ - this.down.touches[i], │ │ │ │ │ - this.last.touches[i] │ │ │ │ │ - ) > this.pixelTolerance) { │ │ │ │ │ - passes = false; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return passes; │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getTouchDistance │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: reverseAxisOrder │ │ │ │ │ + * Returns true if the axis order is reversed for the WMS version and │ │ │ │ │ + * projection of the layer. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The pixel displacement between two touches. │ │ │ │ │ + * {Boolean} true if the axis order is reversed, false otherwise. │ │ │ │ │ */ │ │ │ │ │ - getTouchDistance: function(from, to) { │ │ │ │ │ - return Math.sqrt( │ │ │ │ │ - Math.pow(from.clientX - to.clientX, 2) + │ │ │ │ │ - Math.pow(from.clientY - to.clientY, 2) │ │ │ │ │ - ); │ │ │ │ │ + reverseAxisOrder: function() { │ │ │ │ │ + var projCode = this.projection.getCode(); │ │ │ │ │ + return parseFloat(this.params.VERSION) >= 1.3 && │ │ │ │ │ + !!(this.yx[projCode] || (OpenLayers.Projection.defaults[projCode] && │ │ │ │ │ + OpenLayers.Projection.defaults[projCode].yx)); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: passesDblclickTolerance │ │ │ │ │ - * Determine whether the event is within the optional double-cick pixel │ │ │ │ │ - * tolerance. │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * Return a GetMap query string for this layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox for the │ │ │ │ │ + * request. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The click is within the double-click pixel tolerance. │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as │ │ │ │ │ + * parameters. │ │ │ │ │ */ │ │ │ │ │ - passesDblclickTolerance: function(evt) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - if (this.down && this.first) { │ │ │ │ │ - passes = this.down.xy.distanceTo(this.first.xy) <= this.dblclickTolerance; │ │ │ │ │ - } │ │ │ │ │ - return passes; │ │ │ │ │ - }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clearTimer │ │ │ │ │ - * Clear the timer and set <timerId> to null. │ │ │ │ │ - */ │ │ │ │ │ - clearTimer: function() { │ │ │ │ │ - if (this.timerId != null) { │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.rightclickTimerId != null) { │ │ │ │ │ - window.clearTimeout(this.rightclickTimerId); │ │ │ │ │ - this.rightclickTimerId = null; │ │ │ │ │ - } │ │ │ │ │ + var imageSize = this.getImageSize(); │ │ │ │ │ + var newParams = {}; │ │ │ │ │ + // WMS 1.3 introduced axis order │ │ │ │ │ + var reverseAxisOrder = this.reverseAxisOrder(); │ │ │ │ │ + newParams.BBOX = this.encodeBBOX ? │ │ │ │ │ + bounds.toBBOX(null, reverseAxisOrder) : │ │ │ │ │ + bounds.toArray(reverseAxisOrder); │ │ │ │ │ + newParams.WIDTH = imageSize.w; │ │ │ │ │ + newParams.HEIGHT = imageSize.h; │ │ │ │ │ + var requestString = this.getFullRequestString(newParams); │ │ │ │ │ + return requestString; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: delayedCall │ │ │ │ │ - * Sets <timerId> to null. And optionally triggers the click callback if │ │ │ │ │ - * evt is set. │ │ │ │ │ + * APIMethod: mergeNewParams │ │ │ │ │ + * Catch changeParams and uppercase the new params to be merged in │ │ │ │ │ + * before calling changeParams on the super class. │ │ │ │ │ + * │ │ │ │ │ + * Once params have been changed, the tiles will be reloaded with │ │ │ │ │ + * the new parameters. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newParams - {Object} Hashtable of new params to use │ │ │ │ │ */ │ │ │ │ │ - delayedCall: function(evt) { │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - if (evt) { │ │ │ │ │ - this.callback("click", [evt]); │ │ │ │ │ - } │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ + var newArguments = [upperParams]; │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, │ │ │ │ │ + newArguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getEventInfo │ │ │ │ │ - * This method allows us to store event information without storing the │ │ │ │ │ - * actual event. In touch devices (at least), the same event is │ │ │ │ │ - * modified between touchstart, touchmove, and touchend. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getFullRequestString │ │ │ │ │ + * Combine the layer's url with its params and these newParams. │ │ │ │ │ + * │ │ │ │ │ + * Add the SRS parameter from projection -- this is probably │ │ │ │ │ + * more eloquently done via a setProjection() method, but this │ │ │ │ │ + * works for now and always. │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newParams - {Object} │ │ │ │ │ + * altUrl - {String} Use this as the url instead of the layer's url │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} An object with event related info. │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - getEventInfo: function(evt) { │ │ │ │ │ - var touches; │ │ │ │ │ - if (evt.touches) { │ │ │ │ │ - var len = evt.touches.length; │ │ │ │ │ - touches = new Array(len); │ │ │ │ │ - var touch; │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - touch = evt.touches[i]; │ │ │ │ │ - touches[i] = { │ │ │ │ │ - clientX: touch.olClientX, │ │ │ │ │ - clientY: touch.olClientY │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ + getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ + var mapProjection = this.map.getProjectionObject(); │ │ │ │ │ + var projectionCode = this.projection && this.projection.equals(mapProjection) ? │ │ │ │ │ + this.projection.getCode() : │ │ │ │ │ + mapProjection.getCode(); │ │ │ │ │ + var value = (projectionCode == "none") ? null : projectionCode; │ │ │ │ │ + if (parseFloat(this.params.VERSION) >= 1.3) { │ │ │ │ │ + this.params.CRS = value; │ │ │ │ │ + } else { │ │ │ │ │ + this.params.SRS = value; │ │ │ │ │ } │ │ │ │ │ - return { │ │ │ │ │ - xy: evt.xy, │ │ │ │ │ - touches: touches │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the handler. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.down = null; │ │ │ │ │ - this.first = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - deactivated = true; │ │ │ │ │ + if (typeof this.params.TRANSPARENT == "boolean") { │ │ │ │ │ + newParams.TRANSPARENT = this.params.TRANSPARENT ? "TRUE" : "FALSE"; │ │ │ │ │ } │ │ │ │ │ - return deactivated; │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.getFullRequestString.apply( │ │ │ │ │ + this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Click" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.WMS" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Path.js │ │ │ │ │ + OpenLayers/Control/SLDSelect.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Layer/WMS.js │ │ │ │ │ + * @requires OpenLayers/Handler/RegularPolygon.js │ │ │ │ │ + * @requires OpenLayers/Handler/Polygon.js │ │ │ │ │ + * @requires OpenLayers/Handler/Path.js │ │ │ │ │ + * @requires OpenLayers/Handler/Click.js │ │ │ │ │ + * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ + * @requires OpenLayers/Format/SLD/v1_0_0.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Handler.Path │ │ │ │ │ - * Handler to draw a path on the map. Path is displayed on mouse down, │ │ │ │ │ - * moves on mouse move, and is finished on mouse up. │ │ │ │ │ + * Class: OpenLayers.Control.SLDSelect │ │ │ │ │ + * Perform selections on WMS layers using Styled Layer Descriptor (SLD) │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler.Point> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.Path = OpenLayers.Class(OpenLayers.Handler.Point, { │ │ │ │ │ +OpenLayers.Control.SLDSelect = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ + * selected - Triggered when a selection occurs. Listeners receive an │ │ │ │ │ + * event with *filters* and *layer* properties. Filters will be an │ │ │ │ │ + * array of OpenLayers.Filter objects created in order to perform │ │ │ │ │ + * the particular selection. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: line │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * APIProperty: clearOnDeactivate │ │ │ │ │ + * {Boolean} Should the selection be cleared when the control is │ │ │ │ │ + * deactivated. Default value is false. │ │ │ │ │ */ │ │ │ │ │ - line: null, │ │ │ │ │ + clearOnDeactivate: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: maxVertices │ │ │ │ │ - * {Number} The maximum number of vertices which can be drawn by this │ │ │ │ │ - * handler. When the number of vertices reaches maxVertices, the │ │ │ │ │ - * geometry is automatically finalized. Default is null. │ │ │ │ │ + * APIProperty: layers │ │ │ │ │ + * {Array(<OpenLayers.Layer.WMS>)} The WMS layers this control will work │ │ │ │ │ + * on. │ │ │ │ │ */ │ │ │ │ │ - maxVertices: null, │ │ │ │ │ + layers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: doubleTouchTolerance │ │ │ │ │ - * {Number} Maximum number of pixels between two touches for │ │ │ │ │ - * the gesture to be considered a "finalize feature" action. │ │ │ │ │ - * Default is 20. │ │ │ │ │ + * Property: callbacks │ │ │ │ │ + * {Object} The functions that are sent to the handler for callback │ │ │ │ │ */ │ │ │ │ │ - doubleTouchTolerance: 20, │ │ │ │ │ + callbacks: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: freehand │ │ │ │ │ - * {Boolean} In freehand mode, the handler starts the path on mouse down, │ │ │ │ │ - * adds a point for every mouse move, and finishes the path on mouse up. │ │ │ │ │ - * Outside of freehand mode, a point is added to the path on every mouse │ │ │ │ │ - * click and double-click finishes the path. │ │ │ │ │ + * APIProperty: selectionSymbolizer │ │ │ │ │ + * {Object} Determines the styling of the selected objects. Default is │ │ │ │ │ + * a selection in red. │ │ │ │ │ */ │ │ │ │ │ - freehand: false, │ │ │ │ │ + selectionSymbolizer: { │ │ │ │ │ + 'Polygon': { │ │ │ │ │ + fillColor: '#FF0000', │ │ │ │ │ + stroke: false │ │ │ │ │ + }, │ │ │ │ │ + 'Line': { │ │ │ │ │ + strokeColor: '#FF0000', │ │ │ │ │ + strokeWidth: 2 │ │ │ │ │ + }, │ │ │ │ │ + 'Point': { │ │ │ │ │ + graphicName: 'square', │ │ │ │ │ + fillColor: '#FF0000', │ │ │ │ │ + pointRadius: 5 │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: freehandToggle │ │ │ │ │ - * {String} If set, freehandToggle is checked on mouse events and will set │ │ │ │ │ - * the freehand mode to the opposite of this.freehand. To disallow │ │ │ │ │ - * toggling between freehand and non-freehand mode, set freehandToggle to │ │ │ │ │ - * null. Acceptable toggle values are 'shiftKey', 'ctrlKey', and 'altKey'. │ │ │ │ │ + * APIProperty: layerOptions │ │ │ │ │ + * {Object} The options to apply to the selection layer, by default the │ │ │ │ │ + * selection layer will be kept out of the layer switcher. │ │ │ │ │ */ │ │ │ │ │ - freehandToggle: 'shiftKey', │ │ │ │ │ + layerOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: timerId │ │ │ │ │ - * {Integer} The timer used to test the double touch. │ │ │ │ │ + * APIProperty: handlerOptions │ │ │ │ │ + * {Object} Used to set non-default properties on the control's handler │ │ │ │ │ */ │ │ │ │ │ - timerId: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: redoStack │ │ │ │ │ - * {Array} Stack containing points removed with <undo>. │ │ │ │ │ + * APIProperty: sketchStyle │ │ │ │ │ + * {<OpenLayers.Style>|Object} Style or symbolizer to use for the sketch │ │ │ │ │ + * handler. The recommended way of styling the sketch layer, however, is │ │ │ │ │ + * to configure an <OpenLayers.StyleMap> in the layerOptions of the │ │ │ │ │ + * <handlerOptions>: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * new OpenLayers.Control.SLDSelect(OpenLayers.Handler.Path, { │ │ │ │ │ + * handlerOptions: { │ │ │ │ │ + * layerOptions: { │ │ │ │ │ + * styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ + * "default": {strokeColor: "yellow"} │ │ │ │ │ + * }) │ │ │ │ │ + * } │ │ │ │ │ + * } │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ */ │ │ │ │ │ - redoStack: null, │ │ │ │ │ + sketchStyle: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Path │ │ │ │ │ - * Create a new path hander │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ - * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ - * functions. Various callbacks described below. │ │ │ │ │ - * options - {Object} An optional object with properties to be set on the │ │ │ │ │ - * handler │ │ │ │ │ - * │ │ │ │ │ - * Named callbacks: │ │ │ │ │ - * create - Called when a sketch is first created. Callback called with │ │ │ │ │ - * the creation point geometry and sketch feature. │ │ │ │ │ - * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ - * geometry and the sketch feature. │ │ │ │ │ - * point - Called as each point is added. Receives the new point geometry. │ │ │ │ │ - * done - Called when the point drawing is finished. The callback will │ │ │ │ │ - * recieve a single argument, the linestring geometry. │ │ │ │ │ - * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ - * cancel callback will receive a geometry. │ │ │ │ │ + * APIProperty: wfsCache │ │ │ │ │ + * {Object} Cache to use for storing parsed results from │ │ │ │ │ + * <OpenLayers.Format.WFSDescribeFeatureType.read>. If not provided, │ │ │ │ │ + * these will be cached on the prototype. │ │ │ │ │ */ │ │ │ │ │ + wfsCache: {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createFeature │ │ │ │ │ - * Add temporary geometries │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} The initial pixel location for the new │ │ │ │ │ - * feature. │ │ │ │ │ + * APIProperty: layerCache │ │ │ │ │ + * {Object} Cache to use for storing references to the selection layers. │ │ │ │ │ + * Normally each source layer will have exactly 1 selection layer of │ │ │ │ │ + * type OpenLayers.Layer.WMS. If not provided, layers will │ │ │ │ │ + * be cached on the prototype. Note that if <clearOnDeactivate> is │ │ │ │ │ + * true, the layer will no longer be cached after deactivating the │ │ │ │ │ + * control. │ │ │ │ │ */ │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ - lonlat.lon, lonlat.lat │ │ │ │ │ - ); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.line = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.LineString([this.point.geometry]) │ │ │ │ │ - ); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.line, this.point], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + layerCache: {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroyFeature │ │ │ │ │ - * Destroy temporary geometries │ │ │ │ │ + * Constructor: OpenLayers.Control.SLDSelect │ │ │ │ │ + * Create a new control for selecting features in WMS layers using │ │ │ │ │ + * Styled Layer Descriptor (SLD). │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ + * handler - {<OpenLayers.Class>} A sketch handler class. This determines │ │ │ │ │ + * the type of selection, e.g. box (<OpenLayers.Handler.Box>), point │ │ │ │ │ + * (<OpenLayers.Handler.Point>), path (<OpenLayers.Handler.Path>) or │ │ │ │ │ + * polygon (<OpenLayers.Handler.Polygon>) selection. To use circle │ │ │ │ │ + * type selection, use <OpenLayers.Handler.RegularPolygon> and pass │ │ │ │ │ + * the number of desired sides (e.g. 40) as "sides" property to the │ │ │ │ │ + * <handlerOptions>. │ │ │ │ │ + * options - {Object} An object containing all configuration properties for │ │ │ │ │ + * the control. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * layers - Array({<OpenLayers.Layer.WMS>}) The layers to perform the │ │ │ │ │ + * selection on. │ │ │ │ │ */ │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - OpenLayers.Handler.Point.prototype.destroyFeature.call( │ │ │ │ │ - this, force); │ │ │ │ │ - this.line = null; │ │ │ │ │ - }, │ │ │ │ │ + initialize: function(handler, options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroyPersistedFeature │ │ │ │ │ - * Destroy the persisted feature. │ │ │ │ │ - */ │ │ │ │ │ - destroyPersistedFeature: function() { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - if (layer && layer.features.length > 2) { │ │ │ │ │ - this.layer.features[0].destroy(); │ │ │ │ │ + this.callbacks = OpenLayers.Util.extend({ │ │ │ │ │ + done: this.select, │ │ │ │ │ + click: this.select │ │ │ │ │ + }, this.callbacks); │ │ │ │ │ + this.handlerOptions = this.handlerOptions || {}; │ │ │ │ │ + this.layerOptions = OpenLayers.Util.applyDefaults(this.layerOptions, { │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + tileOptions: { │ │ │ │ │ + maxGetUrlLength: 2048 │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (this.sketchStyle) { │ │ │ │ │ + this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.handlerOptions.layerOptions, { │ │ │ │ │ + styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ + "default": this.sketchStyle │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ + this.handler = new handler(this, this.callbacks, this.handlerOptions); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removePoint │ │ │ │ │ - * Destroy the temporary point. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Take care of things that are not handled in superclass. │ │ │ │ │ */ │ │ │ │ │ - removePoint: function() { │ │ │ │ │ - if (this.point) { │ │ │ │ │ - this.layer.removeFeatures([this.point]); │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var key in this.layerCache) { │ │ │ │ │ + delete this.layerCache[key]; │ │ │ │ │ } │ │ │ │ │ + for (var key in this.wfsCache) { │ │ │ │ │ + delete this.wfsCache[key]; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: addPoint │ │ │ │ │ - * Add point to geometry. Send the point index to override │ │ │ │ │ - * the behavior of LinearRing that disregards adding duplicate points. │ │ │ │ │ + * Method: coupleLayerVisiblity │ │ │ │ │ + * Couple the selection layer and the source layer with respect to │ │ │ │ │ + * layer visibility. So if the source layer is turned off, the │ │ │ │ │ + * selection layer is also turned off. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} The pixel location for the new point. │ │ │ │ │ - */ │ │ │ │ │ - addPoint: function(pixel) { │ │ │ │ │ - this.layer.removeFeatures([this.point]); │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat) │ │ │ │ │ - ); │ │ │ │ │ - this.line.geometry.addComponent( │ │ │ │ │ - this.point.geometry, this.line.geometry.components.length │ │ │ │ │ - ); │ │ │ │ │ - this.layer.addFeatures([this.point]); │ │ │ │ │ - this.callback("point", [this.point.geometry, this.getGeometry()]); │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - delete this.redoStack; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: insertXY │ │ │ │ │ - * Insert a point in the current sketch given x & y coordinates. The new │ │ │ │ │ - * point is inserted immediately before the most recently drawn point. │ │ │ │ │ + * Context: │ │ │ │ │ + * - {<OpenLayers.Layer>} │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * x - {Number} The x-coordinate of the point. │ │ │ │ │ - * y - {Number} The y-coordinate of the point. │ │ │ │ │ + * evt - {Object} │ │ │ │ │ */ │ │ │ │ │ - insertXY: function(x, y) { │ │ │ │ │ - this.line.geometry.addComponent( │ │ │ │ │ - new OpenLayers.Geometry.Point(x, y), │ │ │ │ │ - this.getCurrentPointIndex() │ │ │ │ │ - ); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - delete this.redoStack; │ │ │ │ │ + coupleLayerVisiblity: function(evt) { │ │ │ │ │ + this.setVisibility(evt.object.getVisibility()); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: insertDeltaXY │ │ │ │ │ - * Insert a point given offsets from the previously inserted point. │ │ │ │ │ + * Method: createSelectionLayer │ │ │ │ │ + * Creates a "clone" from the source layer in which the selection can │ │ │ │ │ + * be drawn. This ensures both the source layer and the selection are │ │ │ │ │ + * visible and not only the selection. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * dx - {Number} The x-coordinate offset of the point. │ │ │ │ │ - * dy - {Number} The y-coordinate offset of the point. │ │ │ │ │ + * source - {<OpenLayers.Layer.WMS>} The source layer on which the selection │ │ │ │ │ + * is performed. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.WMS>} A WMS layer with maxGetUrlLength configured to 2048 │ │ │ │ │ + * since SLD selections can easily get quite long. │ │ │ │ │ */ │ │ │ │ │ - insertDeltaXY: function(dx, dy) { │ │ │ │ │ - var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ - var p0 = this.line.geometry.components[previousIndex]; │ │ │ │ │ - if (p0 && !isNaN(p0.x) && !isNaN(p0.y)) { │ │ │ │ │ - this.insertXY(p0.x + dx, p0.y + dy); │ │ │ │ │ + createSelectionLayer: function(source) { │ │ │ │ │ + // check if we already have a selection layer for the source layer │ │ │ │ │ + var selectionLayer; │ │ │ │ │ + if (!this.layerCache[source.id]) { │ │ │ │ │ + selectionLayer = new OpenLayers.Layer.WMS(source.name, │ │ │ │ │ + source.url, source.params, │ │ │ │ │ + OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.layerOptions, │ │ │ │ │ + source.getOptions()) │ │ │ │ │ + ); │ │ │ │ │ + this.layerCache[source.id] = selectionLayer; │ │ │ │ │ + // make sure the layers are coupled wrt visibility, but only │ │ │ │ │ + // if they are not displayed in the layer switcher, because in │ │ │ │ │ + // that case the user cannot control visibility. │ │ │ │ │ + if (this.layerOptions.displayInLayerSwitcher === false) { │ │ │ │ │ + source.events.on({ │ │ │ │ │ + "visibilitychanged": this.coupleLayerVisiblity, │ │ │ │ │ + scope: selectionLayer │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + this.map.addLayer(selectionLayer); │ │ │ │ │ + } else { │ │ │ │ │ + selectionLayer = this.layerCache[source.id]; │ │ │ │ │ } │ │ │ │ │ + return selectionLayer; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: insertDirectionLength │ │ │ │ │ - * Insert a point in the current sketch given a direction and a length. │ │ │ │ │ + * Method: createSLD │ │ │ │ │ + * Create the SLD document for the layer using the supplied filters. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * direction - {Number} Degrees clockwise from the positive x-axis. │ │ │ │ │ - * length - {Number} Distance from the previously drawn point. │ │ │ │ │ - */ │ │ │ │ │ - insertDirectionLength: function(direction, length) { │ │ │ │ │ - direction *= Math.PI / 180; │ │ │ │ │ - var dx = length * Math.cos(direction); │ │ │ │ │ - var dy = length * Math.sin(direction); │ │ │ │ │ - this.insertDeltaXY(dx, dy); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: insertDeflectionLength │ │ │ │ │ - * Insert a point in the current sketch given a deflection and a length. │ │ │ │ │ - * The deflection should be degrees clockwise from the previously │ │ │ │ │ - * digitized segment. │ │ │ │ │ + * layer - {<OpenLayers.Layer.WMS>} │ │ │ │ │ + * filters - Array({<OpenLayers.Filter>}) The filters to be applied. │ │ │ │ │ + * geometryAttributes - Array({Object}) The geometry attributes of the │ │ │ │ │ + * layer. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * deflection - {Number} Degrees clockwise from the previous segment. │ │ │ │ │ - * length - {Number} Distance from the previously drawn point. │ │ │ │ │ - */ │ │ │ │ │ - insertDeflectionLength: function(deflection, length) { │ │ │ │ │ - var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ - if (previousIndex > 0) { │ │ │ │ │ - var p1 = this.line.geometry.components[previousIndex]; │ │ │ │ │ - var p0 = this.line.geometry.components[previousIndex - 1]; │ │ │ │ │ - var theta = Math.atan2(p1.y - p0.y, p1.x - p0.x); │ │ │ │ │ - this.insertDirectionLength( │ │ │ │ │ - (theta * 180 / Math.PI) + deflection, length │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getCurrentPointIndex │ │ │ │ │ - * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Number} The index of the most recently drawn point. │ │ │ │ │ - */ │ │ │ │ │ - getCurrentPointIndex: function() { │ │ │ │ │ - return this.line.geometry.components.length - 1; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: undo │ │ │ │ │ - * Remove the most recently added point in the sketch geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} A point was removed. │ │ │ │ │ + * {String} The SLD document generated as a string. │ │ │ │ │ */ │ │ │ │ │ - undo: function() { │ │ │ │ │ - var geometry = this.line.geometry; │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var index = this.getCurrentPointIndex() - 1; │ │ │ │ │ - var target = components[index]; │ │ │ │ │ - var undone = geometry.removeComponent(target); │ │ │ │ │ - if (undone) { │ │ │ │ │ - // On touch devices, set the current ("mouse location") point to │ │ │ │ │ - // match the last digitized point. │ │ │ │ │ - if (this.touch && index > 0) { │ │ │ │ │ - components = geometry.components; // safety │ │ │ │ │ - var lastpt = components[index - 1]; │ │ │ │ │ - var curptidx = this.getCurrentPointIndex(); │ │ │ │ │ - var curpt = components[curptidx]; │ │ │ │ │ - curpt.x = lastpt.x; │ │ │ │ │ - curpt.y = lastpt.y; │ │ │ │ │ - } │ │ │ │ │ - if (!this.redoStack) { │ │ │ │ │ - this.redoStack = []; │ │ │ │ │ + createSLD: function(layer, filters, geometryAttributes) { │ │ │ │ │ + var sld = { │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + namedLayers: {} │ │ │ │ │ + }; │ │ │ │ │ + var layerNames = [layer.params.LAYERS].join(",").split(","); │ │ │ │ │ + for (var i = 0, len = layerNames.length; i < len; i++) { │ │ │ │ │ + var name = layerNames[i]; │ │ │ │ │ + sld.namedLayers[name] = { │ │ │ │ │ + name: name, │ │ │ │ │ + userStyles: [] │ │ │ │ │ + }; │ │ │ │ │ + var symbolizer = this.selectionSymbolizer; │ │ │ │ │ + var geometryAttribute = geometryAttributes[i]; │ │ │ │ │ + if (geometryAttribute.type.indexOf('Polygon') >= 0) { │ │ │ │ │ + symbolizer = { │ │ │ │ │ + Polygon: this.selectionSymbolizer['Polygon'] │ │ │ │ │ + }; │ │ │ │ │ + } else if (geometryAttribute.type.indexOf('LineString') >= 0) { │ │ │ │ │ + symbolizer = { │ │ │ │ │ + Line: this.selectionSymbolizer['Line'] │ │ │ │ │ + }; │ │ │ │ │ + } else if (geometryAttribute.type.indexOf('Point') >= 0) { │ │ │ │ │ + symbolizer = { │ │ │ │ │ + Point: this.selectionSymbolizer['Point'] │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ - this.redoStack.push(target); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ + var filter = filters[i]; │ │ │ │ │ + sld.namedLayers[name].userStyles.push({ │ │ │ │ │ + name: 'default', │ │ │ │ │ + rules: [ │ │ │ │ │ + new OpenLayers.Rule({ │ │ │ │ │ + symbolizer: symbolizer, │ │ │ │ │ + filter: filter, │ │ │ │ │ + maxScaleDenominator: layer.options.minScale │ │ │ │ │ + }) │ │ │ │ │ + ] │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - return undone; │ │ │ │ │ + return new OpenLayers.Format.SLD({ │ │ │ │ │ + srsName: this.map.getProjection() │ │ │ │ │ + }).write(sld); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: redo │ │ │ │ │ - * Reinsert the most recently removed point resulting from an <undo> call. │ │ │ │ │ - * The undo stack is deleted whenever a point is added by other means. │ │ │ │ │ + * Method: parseDescribeLayer │ │ │ │ │ + * Parse the SLD WMS DescribeLayer response and issue the corresponding │ │ │ │ │ + * WFS DescribeFeatureType request │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} A point was added. │ │ │ │ │ + * request - {XMLHttpRequest} The request object. │ │ │ │ │ */ │ │ │ │ │ - redo: function() { │ │ │ │ │ - var target = this.redoStack && this.redoStack.pop(); │ │ │ │ │ - if (target) { │ │ │ │ │ - this.line.geometry.addComponent(target, this.getCurrentPointIndex()); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ + parseDescribeLayer: function(request) { │ │ │ │ │ + var format = new OpenLayers.Format.WMSDescribeLayer(); │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ } │ │ │ │ │ - return !!target; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: freehandMode │ │ │ │ │ - * Determine whether to behave in freehand mode or not. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - freehandMode: function(evt) { │ │ │ │ │ - return (this.freehandToggle && evt[this.freehandToggle]) ? │ │ │ │ │ - !this.freehand : this.freehand; │ │ │ │ │ + var describeLayer = format.read(doc); │ │ │ │ │ + var typeNames = []; │ │ │ │ │ + var url = null; │ │ │ │ │ + for (var i = 0, len = describeLayer.length; i < len; i++) { │ │ │ │ │ + // perform a WFS DescribeFeatureType request │ │ │ │ │ + if (describeLayer[i].owsType == "WFS") { │ │ │ │ │ + typeNames.push(describeLayer[i].typeName); │ │ │ │ │ + url = describeLayer[i].owsURL; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var options = { │ │ │ │ │ + url: url, │ │ │ │ │ + params: { │ │ │ │ │ + SERVICE: "WFS", │ │ │ │ │ + TYPENAME: typeNames.toString(), │ │ │ │ │ + REQUEST: "DescribeFeatureType", │ │ │ │ │ + VERSION: "1.0.0" │ │ │ │ │ + }, │ │ │ │ │ + callback: function(request) { │ │ │ │ │ + var format = new OpenLayers.Format.WFSDescribeFeatureType(); │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ + } │ │ │ │ │ + var describeFeatureType = format.read(doc); │ │ │ │ │ + this.control.wfsCache[this.layer.id] = describeFeatureType; │ │ │ │ │ + this.control._queue && this.control.applySelection(); │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Request.GET(options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: modifyFeature │ │ │ │ │ - * Modify the existing geometry given the new point │ │ │ │ │ + * Method: getGeometryAttributes │ │ │ │ │ + * Look up the geometry attributes from the WFS DescribeFeatureType response │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} The updated pixel location for the latest │ │ │ │ │ - * point. │ │ │ │ │ - * drawing - {Boolean} Indicate if we're currently drawing. │ │ │ │ │ - */ │ │ │ │ │ - modifyFeature: function(pixel, drawing) { │ │ │ │ │ - if (!this.line) { │ │ │ │ │ - this.createFeature(pixel); │ │ │ │ │ - } │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point.geometry.x = lonlat.lon; │ │ │ │ │ - this.point.geometry.y = lonlat.lat; │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.getSketch(), drawing]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawFeature │ │ │ │ │ - * Render geometries on the temporary layer. │ │ │ │ │ - */ │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.line, this.style); │ │ │ │ │ - this.layer.drawFeature(this.point, this.style); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getSketch │ │ │ │ │ - * Return the sketch feature. │ │ │ │ │ + * layer - {<OpenLayers.Layer.WMS>} The layer for which to look up the │ │ │ │ │ + * geometry attributes. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * Array({Object}) Array of geometry attributes │ │ │ │ │ */ │ │ │ │ │ - getSketch: function() { │ │ │ │ │ - return this.line; │ │ │ │ │ + getGeometryAttributes: function(layer) { │ │ │ │ │ + var result = []; │ │ │ │ │ + var cache = this.wfsCache[layer.id]; │ │ │ │ │ + for (var i = 0, len = cache.featureTypes.length; i < len; i++) { │ │ │ │ │ + var typeName = cache.featureTypes[i]; │ │ │ │ │ + var properties = typeName.properties; │ │ │ │ │ + for (var j = 0, lenj = properties.length; j < lenj; j++) { │ │ │ │ │ + var property = properties[j]; │ │ │ │ │ + var type = property.type; │ │ │ │ │ + if ((type.indexOf('LineString') >= 0) || │ │ │ │ │ + (type.indexOf('GeometryAssociationType') >= 0) || │ │ │ │ │ + (type.indexOf('GeometryPropertyType') >= 0) || │ │ │ │ │ + (type.indexOf('Point') >= 0) || │ │ │ │ │ + (type.indexOf('Polygon') >= 0)) { │ │ │ │ │ + result.push(property); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return result; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getGeometry │ │ │ │ │ - * Return the sketch geometry. If <multi> is true, this will return │ │ │ │ │ - * a multi-part geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.LineString>} │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Activate the control. Activating the control will perform a SLD WMS │ │ │ │ │ + * DescribeLayer request followed by a WFS DescribeFeatureType request │ │ │ │ │ + * so that the proper symbolizers can be chosen based on the geometry │ │ │ │ │ + * type. │ │ │ │ │ */ │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.line && this.line.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiLineString([geometry]); │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Control.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.layers[i]; │ │ │ │ │ + if (layer && !this.wfsCache[layer.id]) { │ │ │ │ │ + var options = { │ │ │ │ │ + url: layer.url, │ │ │ │ │ + params: { │ │ │ │ │ + SERVICE: "WMS", │ │ │ │ │ + VERSION: layer.params.VERSION, │ │ │ │ │ + LAYERS: layer.params.LAYERS, │ │ │ │ │ + REQUEST: "DescribeLayer" │ │ │ │ │ + }, │ │ │ │ │ + callback: this.parseDescribeLayer, │ │ │ │ │ + scope: { │ │ │ │ │ + layer: layer, │ │ │ │ │ + control: this │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Request.GET(options); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return geometry; │ │ │ │ │ + return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * method: touchstart │ │ │ │ │ - * handle touchstart. │ │ │ │ │ - * │ │ │ │ │ - * parameters: │ │ │ │ │ - * evt - {event} the browser event │ │ │ │ │ - * │ │ │ │ │ - * returns: │ │ │ │ │ - * {boolean} allow event propagation │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the control. If clearOnDeactivate is true, remove the │ │ │ │ │ + * selection layer(s). │ │ │ │ │ */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - if (this.timerId && │ │ │ │ │ - this.passesTolerance(this.lastTouchPx, evt.xy, │ │ │ │ │ - this.doubleTouchTolerance)) { │ │ │ │ │ - // double-tap, finalize the geometry │ │ │ │ │ - this.finishGeometry(); │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - return false; │ │ │ │ │ - } else { │ │ │ │ │ - if (this.timerId) { │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Control.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.layers[i]; │ │ │ │ │ + if (layer && this.clearOnDeactivate === true) { │ │ │ │ │ + var layerCache = this.layerCache; │ │ │ │ │ + var selectionLayer = layerCache[layer.id]; │ │ │ │ │ + if (selectionLayer) { │ │ │ │ │ + layer.events.un({ │ │ │ │ │ + "visibilitychanged": this.coupleLayerVisiblity, │ │ │ │ │ + scope: selectionLayer │ │ │ │ │ + }); │ │ │ │ │ + selectionLayer.destroy(); │ │ │ │ │ + delete layerCache[layer.id]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.timerId = window.setTimeout( │ │ │ │ │ - OpenLayers.Function.bind(function() { │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - }, this), 300); │ │ │ │ │ - return OpenLayers.Handler.Point.prototype.touchstart.call(this, evt); │ │ │ │ │ } │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: down │ │ │ │ │ - * Handle mousedown and touchstart. Add a new point to the geometry and │ │ │ │ │ - * render it. Return determines whether to propagate the event on the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ + * APIMethod: setLayers │ │ │ │ │ + * Set the layers on which the selection should be performed. Call the │ │ │ │ │ + * setLayers method if the layer(s) to be used change and the same │ │ │ │ │ + * control should be used on a new set of layers. │ │ │ │ │ + * If the control is already active, it will be active after the new │ │ │ │ │ + * set of layers is set. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layers - {Array(<OpenLayers.Layer.WMS>)} The new set of layers on which │ │ │ │ │ + * the selection should be performed. │ │ │ │ │ */ │ │ │ │ │ - down: function(evt) { │ │ │ │ │ - var stopDown = this.stopDown; │ │ │ │ │ - if (this.freehandMode(evt)) { │ │ │ │ │ - stopDown = true; │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!this.touch && (!this.lastDown || │ │ │ │ │ - !this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ - this.pixelTolerance))) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ + setLayers: function(layers) { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.layers = layers; │ │ │ │ │ + this.activate(); │ │ │ │ │ + } else { │ │ │ │ │ + this.layers = layers; │ │ │ │ │ } │ │ │ │ │ - this.mouseDown = true; │ │ │ │ │ - this.lastDown = evt.xy; │ │ │ │ │ - this.stoppedDown = stopDown; │ │ │ │ │ - return !stopDown; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * Handle mousemove and touchmove. Adjust the geometry and redraw. │ │ │ │ │ - * Return determines whether to propagate the event on the map. │ │ │ │ │ - * │ │ │ │ │ + * Function: createFilter │ │ │ │ │ + * Create the filter to be used in the SLD. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ + * geometryAttribute - {Object} Used to get the name of the geometry │ │ │ │ │ + * attribute which is needed for constructing the spatial filter. │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} The geometry to use. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter.Spatial>} The spatial filter created. │ │ │ │ │ */ │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature(); │ │ │ │ │ + createFilter: function(geometryAttribute, geometry) { │ │ │ │ │ + var filter = null; │ │ │ │ │ + if (this.handler instanceof OpenLayers.Handler.RegularPolygon) { │ │ │ │ │ + // box │ │ │ │ │ + if (this.handler.irregular === true) { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry.getBounds() │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - if (this.maxVertices && this.line && │ │ │ │ │ - this.line.geometry.components.length === this.maxVertices) { │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize(); │ │ │ │ │ + } else if (this.handler instanceof OpenLayers.Handler.Polygon) { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry │ │ │ │ │ + }); │ │ │ │ │ + } else if (this.handler instanceof OpenLayers.Handler.Path) { │ │ │ │ │ + // if source layer is point based, use DWITHIN instead │ │ │ │ │ + if (geometryAttribute.type.indexOf('Point') >= 0) { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + distance: this.map.getExtent().getWidth() * 0.01, │ │ │ │ │ + distanceUnits: this.map.getUnits(), │ │ │ │ │ + value: geometry │ │ │ │ │ + }); │ │ │ │ │ } else { │ │ │ │ │ - this.addPoint(evt.xy); │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } else if (this.handler instanceof OpenLayers.Handler.Click) { │ │ │ │ │ + if (geometryAttribute.type.indexOf('Polygon') >= 0) { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + distance: this.map.getExtent().getWidth() * 0.01, │ │ │ │ │ + distanceUnits: this.map.getUnits(), │ │ │ │ │ + value: geometry │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ } │ │ │ │ │ - return true; │ │ │ │ │ + return filter; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: up │ │ │ │ │ - * Handle mouseup and touchend. Send the latest point in the geometry to │ │ │ │ │ - * the control. Return determines whether to propagate the event on the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ + * Method: select │ │ │ │ │ + * When the handler is done, use SLD_BODY on the selection layer to │ │ │ │ │ + * display the selection in the map. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {Object} or {<OpenLayers.Geometry>} │ │ │ │ │ */ │ │ │ │ │ - up: function(evt) { │ │ │ │ │ - if (this.mouseDown && (!this.lastUp || !this.lastUp.equals(evt.xy))) { │ │ │ │ │ - if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature(); │ │ │ │ │ - } │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize(); │ │ │ │ │ - } else { │ │ │ │ │ - if (this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ - this.pixelTolerance)) { │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy); │ │ │ │ │ - } │ │ │ │ │ - if (this.lastUp == null && this.persist) { │ │ │ │ │ - this.destroyPersistedFeature(); │ │ │ │ │ - } │ │ │ │ │ - this.addPoint(evt.xy); │ │ │ │ │ - this.lastUp = evt.xy; │ │ │ │ │ - if (this.line.geometry.components.length === this.maxVertices + 1) { │ │ │ │ │ - this.finishGeometry(); │ │ │ │ │ + select: function(geometry) { │ │ │ │ │ + this._queue = function() { │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.layers[i]; │ │ │ │ │ + var geometryAttributes = this.getGeometryAttributes(layer); │ │ │ │ │ + var filters = []; │ │ │ │ │ + for (var j = 0, lenj = geometryAttributes.length; j < lenj; j++) { │ │ │ │ │ + var geometryAttribute = geometryAttributes[j]; │ │ │ │ │ + if (geometryAttribute !== null) { │ │ │ │ │ + // from the click handler we will not get an actual │ │ │ │ │ + // geometry so transform │ │ │ │ │ + if (!(geometry instanceof OpenLayers.Geometry)) { │ │ │ │ │ + var point = this.map.getLonLatFromPixel( │ │ │ │ │ + geometry.xy); │ │ │ │ │ + geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ + point.lon, point.lat); │ │ │ │ │ + } │ │ │ │ │ + var filter = this.createFilter(geometryAttribute, │ │ │ │ │ + geometry); │ │ │ │ │ + if (filter !== null) { │ │ │ │ │ + filters.push(filter); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - return !this.stopUp; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: finishGeometry │ │ │ │ │ - * Finish the geometry and send it back to the control. │ │ │ │ │ - */ │ │ │ │ │ - finishGeometry: function() { │ │ │ │ │ - var index = this.line.geometry.components.length - 1; │ │ │ │ │ - this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize(); │ │ │ │ │ + var selectionLayer = this.createSelectionLayer(layer); │ │ │ │ │ + │ │ │ │ │ + this.events.triggerEvent("selected", { │ │ │ │ │ + layer: layer, │ │ │ │ │ + filters: filters │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + var sld = this.createSLD(layer, filters, geometryAttributes); │ │ │ │ │ + │ │ │ │ │ + selectionLayer.mergeNewParams({ │ │ │ │ │ + SLD_BODY: sld │ │ │ │ │ + }); │ │ │ │ │ + delete this._queue; │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + this.applySelection(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: dblclick │ │ │ │ │ - * Handle double-clicks. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * Method: applySelection │ │ │ │ │ + * Checks if all required wfs data is cached, and applies the selection │ │ │ │ │ */ │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - if (!this.freehandMode(evt)) { │ │ │ │ │ - this.finishGeometry(); │ │ │ │ │ + applySelection: function() { │ │ │ │ │ + var canApply = true; │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + if (!this.wfsCache[this.layers[i].id]) { │ │ │ │ │ + canApply = false; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return false; │ │ │ │ │ + canApply && this._queue.call(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Path" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.SLDSelect" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Polygon.js │ │ │ │ │ + OpenLayers/Control/PanZoom.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler/Path.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Handler.Polygon │ │ │ │ │ - * Handler to draw a polygon on the map. Polygon is displayed on mouse down, │ │ │ │ │ - * moves on mouse move, and is finished on mouse up. │ │ │ │ │ + * Class: OpenLayers.Control.PanZoom │ │ │ │ │ + * The PanZoom is a visible control, composed of a │ │ │ │ │ + * <OpenLayers.Control.PanPanel> and a <OpenLayers.Control.ZoomPanel>. By │ │ │ │ │ + * default it is drawn in the upper left corner of the map. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler.Path> │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.Polygon = OpenLayers.Class(OpenLayers.Handler.Path, { │ │ │ │ │ +OpenLayers.Control.PanZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: holeModifier │ │ │ │ │ - * {String} Key modifier to trigger hole digitizing. Acceptable values are │ │ │ │ │ - * "altKey", "shiftKey", or "ctrlKey". If not set, no hole digitizing │ │ │ │ │ - * will take place. Default is null. │ │ │ │ │ + * APIProperty: slideFactor │ │ │ │ │ + * {Integer} Number of pixels by which we'll pan the map in any direction │ │ │ │ │ + * on clicking the arrow buttons. If you want to pan by some ratio │ │ │ │ │ + * of the map dimensions, use <slideRatio> instead. │ │ │ │ │ */ │ │ │ │ │ - holeModifier: null, │ │ │ │ │ + slideFactor: 50, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: drawingHole │ │ │ │ │ - * {Boolean} Currently drawing an interior ring. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: slideRatio │ │ │ │ │ + * {Number} The fraction of map width/height by which we'll pan the map │ │ │ │ │ + * on clicking the arrow buttons. Default is null. If set, will │ │ │ │ │ + * override <slideFactor>. E.g. if slideRatio is .5, then the Pan Up │ │ │ │ │ + * button will pan up half the map height. │ │ │ │ │ */ │ │ │ │ │ - drawingHole: false, │ │ │ │ │ + slideRatio: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: polygon │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + /** │ │ │ │ │ + * Property: buttons │ │ │ │ │ + * {Array(DOMElement)} Array of Button Divs │ │ │ │ │ */ │ │ │ │ │ - polygon: null, │ │ │ │ │ + buttons: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Polygon │ │ │ │ │ - * Create a Polygon Handler. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ - * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ - * functions. Various callbacks described below. │ │ │ │ │ - * options - {Object} An optional object with properties to be set on the │ │ │ │ │ - * handler │ │ │ │ │ - * │ │ │ │ │ - * Named callbacks: │ │ │ │ │ - * create - Called when a sketch is first created. Callback called with │ │ │ │ │ - * the creation point geometry and sketch feature. │ │ │ │ │ - * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ - * geometry and the sketch feature. │ │ │ │ │ - * point - Called as each point is added. Receives the new point geometry. │ │ │ │ │ - * done - Called when the point drawing is finished. The callback will │ │ │ │ │ - * recieve a single argument, the polygon geometry. │ │ │ │ │ - * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ - * cancel callback will receive a geometry. │ │ │ │ │ + /** │ │ │ │ │ + * Property: position │ │ │ │ │ + * {<OpenLayers.Pixel>} │ │ │ │ │ */ │ │ │ │ │ + position: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createFeature │ │ │ │ │ - * Add temporary geometries │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Control.PanZoom │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} The initial pixel location for the new │ │ │ │ │ - * feature. │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ - lonlat.lon, lonlat.lat │ │ │ │ │ - ); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.line = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.LinearRing([this.point.geometry]) │ │ │ │ │ - ); │ │ │ │ │ - this.polygon = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.Polygon([this.line.geometry]) │ │ │ │ │ - ); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.polygon, this.point], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.position = new OpenLayers.Pixel(OpenLayers.Control.PanZoom.X, │ │ │ │ │ + OpenLayers.Control.PanZoom.Y); │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: addPoint │ │ │ │ │ - * Add point to geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} The pixel location for the new point. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ */ │ │ │ │ │ - addPoint: function(pixel) { │ │ │ │ │ - if (!this.drawingHole && this.holeModifier && │ │ │ │ │ - this.evt && this.evt[this.holeModifier]) { │ │ │ │ │ - var geometry = this.point.geometry; │ │ │ │ │ - var features = this.control.layer.features; │ │ │ │ │ - var candidate, polygon; │ │ │ │ │ - // look for intersections, last drawn gets priority │ │ │ │ │ - for (var i = features.length - 1; i >= 0; --i) { │ │ │ │ │ - candidate = features[i].geometry; │ │ │ │ │ - if ((candidate instanceof OpenLayers.Geometry.Polygon || │ │ │ │ │ - candidate instanceof OpenLayers.Geometry.MultiPolygon) && │ │ │ │ │ - candidate.intersects(geometry)) { │ │ │ │ │ - polygon = features[i]; │ │ │ │ │ - this.control.layer.removeFeatures([polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.control.layer.events.registerPriority( │ │ │ │ │ - "sketchcomplete", this, this.finalizeInteriorRing │ │ │ │ │ - ); │ │ │ │ │ - this.control.layer.events.registerPriority( │ │ │ │ │ - "sketchmodified", this, this.enforceTopology │ │ │ │ │ - ); │ │ │ │ │ - polygon.geometry.addComponent(this.line.geometry); │ │ │ │ │ - this.polygon = polygon; │ │ │ │ │ - this.drawingHole = true; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Handler.Path.prototype.addPoint.apply(this, arguments); │ │ │ │ │ + this.removeButtons(); │ │ │ │ │ + this.buttons = null; │ │ │ │ │ + this.position = null; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getCurrentPointIndex │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number} The index of the most recently drawn point. │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * │ │ │ │ │ + * Properties: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - getCurrentPointIndex: function() { │ │ │ │ │ - return this.line.geometry.components.length - 2; │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: enforceTopology │ │ │ │ │ - * Simple topology enforcement for drawing interior rings. Ensures vertices │ │ │ │ │ - * of interior rings are contained by exterior ring. Other topology │ │ │ │ │ - * rules are enforced in <finalizeInteriorRing> to allow drawing of │ │ │ │ │ - * rings that intersect only during the sketch (e.g. a "C" shaped ring │ │ │ │ │ - * that nearly encloses another ring). │ │ │ │ │ + * Method: draw │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A reference to the container div for the PanZoom control. │ │ │ │ │ */ │ │ │ │ │ - enforceTopology: function(event) { │ │ │ │ │ - var point = event.vertex; │ │ │ │ │ - var components = this.line.geometry.components; │ │ │ │ │ - // ensure that vertices of interior ring are contained by exterior ring │ │ │ │ │ - if (!this.polygon.geometry.intersects(point)) { │ │ │ │ │ - var last = components[components.length - 3]; │ │ │ │ │ - point.x = last.x; │ │ │ │ │ - point.y = last.y; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + // initialize our internal div │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + px = this.position; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: finishGeometry │ │ │ │ │ - * Finish the geometry and send it back to the control. │ │ │ │ │ - */ │ │ │ │ │ - finishGeometry: function() { │ │ │ │ │ - var index = this.line.geometry.components.length - 2; │ │ │ │ │ - this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize(); │ │ │ │ │ - }, │ │ │ │ │ + // place the controls │ │ │ │ │ + this.buttons = []; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: finalizeInteriorRing │ │ │ │ │ - * Enforces that new ring has some area and doesn't contain vertices of any │ │ │ │ │ - * other rings. │ │ │ │ │ - */ │ │ │ │ │ - finalizeInteriorRing: function() { │ │ │ │ │ - var ring = this.line.geometry; │ │ │ │ │ - // ensure that ring has some area │ │ │ │ │ - var modified = (ring.getArea() !== 0); │ │ │ │ │ - if (modified) { │ │ │ │ │ - // ensure that new ring doesn't intersect any other rings │ │ │ │ │ - var rings = this.polygon.geometry.components; │ │ │ │ │ - for (var i = rings.length - 2; i >= 0; --i) { │ │ │ │ │ - if (ring.intersects(rings[i])) { │ │ │ │ │ - modified = false; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (modified) { │ │ │ │ │ - // ensure that new ring doesn't contain any other rings │ │ │ │ │ - var target; │ │ │ │ │ - outer: for (var i = rings.length - 2; i > 0; --i) { │ │ │ │ │ - var points = rings[i].components; │ │ │ │ │ - for (var j = 0, jj = points.length; j < jj; ++j) { │ │ │ │ │ - if (ring.containsPoint(points[j])) { │ │ │ │ │ - modified = false; │ │ │ │ │ - break outer; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (modified) { │ │ │ │ │ - if (this.polygon.state !== OpenLayers.State.INSERT) { │ │ │ │ │ - this.polygon.state = OpenLayers.State.UPDATE; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.polygon.geometry.removeComponent(ring); │ │ │ │ │ - } │ │ │ │ │ - this.restoreFeature(); │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ + var sz = { │ │ │ │ │ + w: 18, │ │ │ │ │ + h: 18 │ │ │ │ │ + }; │ │ │ │ │ + var centered = new OpenLayers.Pixel(px.x + sz.w / 2, px.y); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: cancel │ │ │ │ │ - * Finish the geometry and call the "cancel" callback. │ │ │ │ │ - */ │ │ │ │ │ - cancel: function() { │ │ │ │ │ - if (this.drawingHole) { │ │ │ │ │ - this.polygon.geometry.removeComponent(this.line.geometry); │ │ │ │ │ - this.restoreFeature(true); │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Handler.Path.prototype.cancel.apply(this, arguments); │ │ │ │ │ + this._addButton("panup", "north-mini.png", centered, sz); │ │ │ │ │ + px.y = centered.y + sz.h; │ │ │ │ │ + this._addButton("panleft", "west-mini.png", px, sz); │ │ │ │ │ + this._addButton("panright", "east-mini.png", px.add(sz.w, 0), sz); │ │ │ │ │ + this._addButton("pandown", "south-mini.png", │ │ │ │ │ + centered.add(0, sz.h * 2), sz); │ │ │ │ │ + this._addButton("zoomin", "zoom-plus-mini.png", │ │ │ │ │ + centered.add(0, sz.h * 3 + 5), sz); │ │ │ │ │ + this._addButton("zoomworld", "zoom-world-mini.png", │ │ │ │ │ + centered.add(0, sz.h * 4 + 5), sz); │ │ │ │ │ + this._addButton("zoomout", "zoom-minus-mini.png", │ │ │ │ │ + centered.add(0, sz.h * 5 + 5), sz); │ │ │ │ │ + return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: restoreFeature │ │ │ │ │ - * Move the feature from the sketch layer to the target layer. │ │ │ │ │ - * │ │ │ │ │ - * Properties: │ │ │ │ │ - * cancel - {Boolean} Cancel drawing. If falsey, the "sketchcomplete" event │ │ │ │ │ - * will be fired. │ │ │ │ │ + * Method: _addButton │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} │ │ │ │ │ + * img - {String} │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} │ │ │ │ │ + * sz - {<OpenLayers.Size>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A Div (an alphaImageDiv, to be precise) that contains the │ │ │ │ │ + * image of the button, and has all the proper event handlers set. │ │ │ │ │ */ │ │ │ │ │ - restoreFeature: function(cancel) { │ │ │ │ │ - this.control.layer.events.unregister( │ │ │ │ │ - "sketchcomplete", this, this.finalizeInteriorRing │ │ │ │ │ - ); │ │ │ │ │ - this.control.layer.events.unregister( │ │ │ │ │ - "sketchmodified", this, this.enforceTopology │ │ │ │ │ - ); │ │ │ │ │ - this.layer.removeFeatures([this.polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.control.layer.addFeatures([this.polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.drawingHole = false; │ │ │ │ │ - if (!cancel) { │ │ │ │ │ - // Re-trigger "sketchcomplete" so other listeners can do their │ │ │ │ │ - // business. While this is somewhat sloppy (if a listener is │ │ │ │ │ - // registered with registerPriority - not common - between the start │ │ │ │ │ - // and end of a single ring drawing - very uncommon - it will be │ │ │ │ │ - // called twice). │ │ │ │ │ - // TODO: In 3.0, collapse sketch handlers into geometry specific │ │ │ │ │ - // drawing controls. │ │ │ │ │ - this.control.layer.events.triggerEvent( │ │ │ │ │ - "sketchcomplete", { │ │ │ │ │ - feature: this.polygon │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + _addButton: function(id, img, xy, sz) { │ │ │ │ │ + var imgLocation = OpenLayers.Util.getImageLocation(img); │ │ │ │ │ + var btn = OpenLayers.Util.createAlphaImageDiv( │ │ │ │ │ + this.id + "_" + id, │ │ │ │ │ + xy, sz, imgLocation, "absolute"); │ │ │ │ │ + btn.style.cursor = "pointer"; │ │ │ │ │ + //we want to add the outer div │ │ │ │ │ + this.div.appendChild(btn); │ │ │ │ │ + btn.action = id; │ │ │ │ │ + btn.className = "olButton"; │ │ │ │ │ + │ │ │ │ │ + //we want to remember/reference the outer div │ │ │ │ │ + this.buttons.push(btn); │ │ │ │ │ + return btn; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroyFeature │ │ │ │ │ - * Destroy temporary geometries │ │ │ │ │ - * │ │ │ │ │ + * Method: _removeButton │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ + * btn - {Object} │ │ │ │ │ */ │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - OpenLayers.Handler.Path.prototype.destroyFeature.call( │ │ │ │ │ - this, force); │ │ │ │ │ - this.polygon = null; │ │ │ │ │ + _removeButton: function(btn) { │ │ │ │ │ + this.div.removeChild(btn); │ │ │ │ │ + OpenLayers.Util.removeItem(this.buttons, btn); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawFeature │ │ │ │ │ - * Render geometries on the temporary layer. │ │ │ │ │ + * Method: removeButtons │ │ │ │ │ */ │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.polygon, this.style); │ │ │ │ │ - this.layer.drawFeature(this.point, this.style); │ │ │ │ │ + removeButtons: function() { │ │ │ │ │ + for (var i = this.buttons.length - 1; i >= 0; --i) { │ │ │ │ │ + this._removeButton(this.buttons[i]); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getSketch │ │ │ │ │ - * Return the sketch feature. │ │ │ │ │ + * Method: onButtonClick │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - getSketch: function() { │ │ │ │ │ - return this.polygon; │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + var btn = evt.buttonElement; │ │ │ │ │ + switch (btn.action) { │ │ │ │ │ + case "panup": │ │ │ │ │ + this.map.pan(0, -this.getSlideFactor("h")); │ │ │ │ │ + break; │ │ │ │ │ + case "pandown": │ │ │ │ │ + this.map.pan(0, this.getSlideFactor("h")); │ │ │ │ │ + break; │ │ │ │ │ + case "panleft": │ │ │ │ │ + this.map.pan(-this.getSlideFactor("w"), 0); │ │ │ │ │ + break; │ │ │ │ │ + case "panright": │ │ │ │ │ + this.map.pan(this.getSlideFactor("w"), 0); │ │ │ │ │ + break; │ │ │ │ │ + case "zoomin": │ │ │ │ │ + this.map.zoomIn(); │ │ │ │ │ + break; │ │ │ │ │ + case "zoomout": │ │ │ │ │ + this.map.zoomOut(); │ │ │ │ │ + break; │ │ │ │ │ + case "zoomworld": │ │ │ │ │ + this.map.zoomToMaxExtent(); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getGeometry │ │ │ │ │ - * Return the sketch geometry. If <multi> is true, this will return │ │ │ │ │ - * a multi-part geometry. │ │ │ │ │ + * Method: getSlideFactor │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * dim - {String} "w" or "h" (for width or height). │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Polygon>} │ │ │ │ │ + * {Number} The slide factor for panning in the requested direction. │ │ │ │ │ */ │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.polygon && this.polygon.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiPolygon([geometry]); │ │ │ │ │ - } │ │ │ │ │ - return geometry; │ │ │ │ │ + getSlideFactor: function(dim) { │ │ │ │ │ + return this.slideRatio ? │ │ │ │ │ + this.map.getSize()[dim] * this.slideRatio : │ │ │ │ │ + this.slideFactor; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Polygon" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.PanZoom" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: X │ │ │ │ │ + * {Integer} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.PanZoom.X = 4; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: Y │ │ │ │ │ + * {Integer} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.PanZoom.Y = 4; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/MouseWheel.js │ │ │ │ │ + OpenLayers/Control/WMSGetFeatureInfo.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Handler/Click.js │ │ │ │ │ + * @requires OpenLayers/Handler/Hover.js │ │ │ │ │ + * @requires OpenLayers/Request.js │ │ │ │ │ + * @requires OpenLayers/Format/WMSGetFeatureInfo.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Handler.MouseWheel │ │ │ │ │ - * Handler for wheel up/down events. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Control.WMSGetFeatureInfo │ │ │ │ │ + * The WMSGetFeatureInfo control uses a WMS query to get information about a point on the map. The │ │ │ │ │ + * information may be in a display-friendly format such as HTML, or a machine-friendly format such │ │ │ │ │ + * as GML, depending on the server's capabilities and the client's configuration. This control │ │ │ │ │ + * handles click or hover events, attempts to parse the results using an OpenLayers.Format, and │ │ │ │ │ + * fires a 'getfeatureinfo' event with the click position, the raw body of the response, and an │ │ │ │ │ + * array of features if it successfully read the response. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.MouseWheel = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - /** │ │ │ │ │ - * Property: wheelListener │ │ │ │ │ - * {function} │ │ │ │ │ +OpenLayers.Control.WMSGetFeatureInfo = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: hover │ │ │ │ │ + * {Boolean} Send GetFeatureInfo requests when mouse stops moving. │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - wheelListener: null, │ │ │ │ │ + hover: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: interval │ │ │ │ │ - * {Integer} In order to increase server performance, an interval (in │ │ │ │ │ - * milliseconds) can be set to reduce the number of up/down events │ │ │ │ │ - * called. If set, a new up/down event will not be set until the │ │ │ │ │ - * interval has passed. │ │ │ │ │ - * Defaults to 0, meaning no interval. │ │ │ │ │ + * APIProperty: drillDown │ │ │ │ │ + * {Boolean} Drill down over all WMS layers in the map. When │ │ │ │ │ + * using drillDown mode, hover is not possible, and an infoFormat that │ │ │ │ │ + * returns parseable features is required. Default is false. │ │ │ │ │ */ │ │ │ │ │ - interval: 0, │ │ │ │ │ + drillDown: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: maxDelta │ │ │ │ │ - * {Integer} Maximum delta to collect before breaking from the current │ │ │ │ │ - * interval. In cumulative mode, this also limits the maximum delta │ │ │ │ │ - * returned from the handler. Default is Number.POSITIVE_INFINITY. │ │ │ │ │ + * APIProperty: maxFeatures │ │ │ │ │ + * {Integer} Maximum number of features to return from a WMS query. This │ │ │ │ │ + * sets the feature_count parameter on WMS GetFeatureInfo │ │ │ │ │ + * requests. │ │ │ │ │ */ │ │ │ │ │ - maxDelta: Number.POSITIVE_INFINITY, │ │ │ │ │ + maxFeatures: 10, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: delta │ │ │ │ │ - * {Integer} When interval is set, delta collects the mousewheel z-deltas │ │ │ │ │ - * of the events that occur within the interval. │ │ │ │ │ - * See also the cumulative option │ │ │ │ │ + * APIProperty: clickCallback │ │ │ │ │ + * {String} The click callback to register in the │ │ │ │ │ + * {<OpenLayers.Handler.Click>} object created when the hover │ │ │ │ │ + * option is set to false. Default is "click". │ │ │ │ │ */ │ │ │ │ │ - delta: 0, │ │ │ │ │ + clickCallback: "click", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: cumulative │ │ │ │ │ - * {Boolean} When interval is set: true to collect all the mousewheel │ │ │ │ │ - * z-deltas, false to only record the delta direction (positive or │ │ │ │ │ - * negative) │ │ │ │ │ + * APIProperty: output │ │ │ │ │ + * {String} Either "features" or "object". When triggering a getfeatureinfo │ │ │ │ │ + * request should we pass on an array of features or an object with with │ │ │ │ │ + * a "features" property and other properties (such as the url of the │ │ │ │ │ + * WMS). Default is "features". │ │ │ │ │ */ │ │ │ │ │ - cumulative: true, │ │ │ │ │ + output: "features", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.MouseWheel │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} │ │ │ │ │ - * callbacks - {Object} An object containing a single function to be │ │ │ │ │ - * called when the drag operation is finished. │ │ │ │ │ - * The callback should expect to recieve a single │ │ │ │ │ - * argument, the point geometry. │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * APIProperty: layers │ │ │ │ │ + * {Array(<OpenLayers.Layer.WMS>)} The layers to query for feature info. │ │ │ │ │ + * If omitted, all map WMS layers with a url that matches this <url> or │ │ │ │ │ + * <layerUrls> will be considered. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.wheelListener = OpenLayers.Function.bindAsEventListener( │ │ │ │ │ - this.onWheelEvent, this │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ + layers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * APIProperty: queryVisible │ │ │ │ │ + * {Boolean} If true, filter out hidden layers when searching the map for │ │ │ │ │ + * layers to query. Default is false. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ - this.wheelListener = null; │ │ │ │ │ - }, │ │ │ │ │ + queryVisible: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Mouse ScrollWheel code thanks to http://adomas.org/javascript-mouse-wheel/ │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String} The URL of the WMS service to use. If not provided, the url │ │ │ │ │ + * of the first eligible layer will be used. │ │ │ │ │ */ │ │ │ │ │ + url: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: onWheelEvent │ │ │ │ │ - * Catch the wheel event and handle it xbrowserly │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * e - {Event} │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layerUrls │ │ │ │ │ + * {Array(String)} Optional list of urls for layers that should be queried. │ │ │ │ │ + * This can be used when the layer url differs from the url used for │ │ │ │ │ + * making GetFeatureInfo requests (in the case of a layer using cached │ │ │ │ │ + * tiles). │ │ │ │ │ */ │ │ │ │ │ - onWheelEvent: function(e) { │ │ │ │ │ + layerUrls: null, │ │ │ │ │ │ │ │ │ │ - // make sure we have a map and check keyboard modifiers │ │ │ │ │ - if (!this.map || !this.checkModifiers(e)) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: infoFormat │ │ │ │ │ + * {String} The mimetype to request from the server. If you are using │ │ │ │ │ + * drillDown mode and have multiple servers that do not share a common │ │ │ │ │ + * infoFormat, you can override the control's infoFormat by providing an │ │ │ │ │ + * INFO_FORMAT parameter in your <OpenLayers.Layer.WMS> instance(s). │ │ │ │ │ + */ │ │ │ │ │ + infoFormat: 'text/html', │ │ │ │ │ │ │ │ │ │ - // Ride up the element's DOM hierarchy to determine if it or any of │ │ │ │ │ - // its ancestors was: │ │ │ │ │ - // * specifically marked as scrollable (CSS overflow property) │ │ │ │ │ - // * one of our layer divs or a div marked as scrollable │ │ │ │ │ - // ('olScrollable' CSS class) │ │ │ │ │ - // * the map div │ │ │ │ │ - // │ │ │ │ │ - var overScrollableDiv = false; │ │ │ │ │ - var allowScroll = false; │ │ │ │ │ - var overMapDiv = false; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: vendorParams │ │ │ │ │ + * {Object} Additional parameters that will be added to the request, for │ │ │ │ │ + * WMS implementations that support them. This could e.g. look like │ │ │ │ │ + * (start code) │ │ │ │ │ + * { │ │ │ │ │ + * radius: 5 │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ + vendorParams: {}, │ │ │ │ │ │ │ │ │ │ - var elem = OpenLayers.Event.element(e); │ │ │ │ │ - while ((elem != null) && !overMapDiv && !overScrollableDiv) { │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: format │ │ │ │ │ + * {<OpenLayers.Format>} A format for parsing GetFeatureInfo responses. │ │ │ │ │ + * Default is <OpenLayers.Format.WMSGetFeatureInfo>. │ │ │ │ │ + */ │ │ │ │ │ + format: null, │ │ │ │ │ │ │ │ │ │ - if (!overScrollableDiv) { │ │ │ │ │ - try { │ │ │ │ │ - var overflow; │ │ │ │ │ - if (elem.currentStyle) { │ │ │ │ │ - overflow = elem.currentStyle["overflow"]; │ │ │ │ │ - } else { │ │ │ │ │ - var style = │ │ │ │ │ - document.defaultView.getComputedStyle(elem, null); │ │ │ │ │ - overflow = style.getPropertyValue("overflow"); │ │ │ │ │ - } │ │ │ │ │ - overScrollableDiv = (overflow && │ │ │ │ │ - (overflow == "auto") || (overflow == "scroll")); │ │ │ │ │ - } catch (err) { │ │ │ │ │ - //sometimes when scrolling in a popup, this causes │ │ │ │ │ - // obscure browser error │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: formatOptions │ │ │ │ │ + * {Object} Optional properties to set on the format (if one is not provided │ │ │ │ │ + * in the <format> property. │ │ │ │ │ + */ │ │ │ │ │ + formatOptions: null, │ │ │ │ │ │ │ │ │ │ - if (!allowScroll) { │ │ │ │ │ - allowScroll = OpenLayers.Element.hasClass(elem, 'olScrollable'); │ │ │ │ │ - if (!allowScroll) { │ │ │ │ │ - for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ - // Are we in the layer div? Note that we have two cases │ │ │ │ │ - // here: one is to catch EventPane layers, which have a │ │ │ │ │ - // pane above the layer (layer.pane) │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if (elem == layer.div || elem == layer.pane) { │ │ │ │ │ - allowScroll = true; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - overMapDiv = (elem == this.map.div); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: handlerOptions │ │ │ │ │ + * {Object} Additional options for the handlers used by this control, e.g. │ │ │ │ │ + * (start code) │ │ │ │ │ + * { │ │ │ │ │ + * "click": {delay: 100}, │ │ │ │ │ + * "hover": {delay: 300} │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - elem = elem.parentNode; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: handler │ │ │ │ │ + * {Object} Reference to the <OpenLayers.Handler> for this control │ │ │ │ │ + */ │ │ │ │ │ + handler: null, │ │ │ │ │ │ │ │ │ │ - // Logic below is the following: │ │ │ │ │ - // │ │ │ │ │ - // If we are over a scrollable div or not over the map div: │ │ │ │ │ - // * do nothing (let the browser handle scrolling) │ │ │ │ │ - // │ │ │ │ │ - // otherwise │ │ │ │ │ - // │ │ │ │ │ - // If we are over the layer div or a 'olScrollable' div: │ │ │ │ │ - // * zoom/in out │ │ │ │ │ - // then │ │ │ │ │ - // * kill event (so as not to also scroll the page after zooming) │ │ │ │ │ - // │ │ │ │ │ - // otherwise │ │ │ │ │ - // │ │ │ │ │ - // Kill the event (dont scroll the page if we wheel over the │ │ │ │ │ - // layerswitcher or the pan/zoom control) │ │ │ │ │ - // │ │ │ │ │ - if (!overScrollableDiv && overMapDiv) { │ │ │ │ │ - if (allowScroll) { │ │ │ │ │ - var delta = 0; │ │ │ │ │ + /** │ │ │ │ │ + * Property: hoverRequest │ │ │ │ │ + * {<OpenLayers.Request>} contains the currently running hover request │ │ │ │ │ + * (if any). │ │ │ │ │ + */ │ │ │ │ │ + hoverRequest: null, │ │ │ │ │ │ │ │ │ │ - if (e.wheelDelta) { │ │ │ │ │ - delta = e.wheelDelta; │ │ │ │ │ - if (delta % 160 === 0) { │ │ │ │ │ - // opera have steps of 160 instead of 120 │ │ │ │ │ - delta = delta * 0.75; │ │ │ │ │ - } │ │ │ │ │ - delta = delta / 120; │ │ │ │ │ - } else if (e.detail) { │ │ │ │ │ - // detail in Firefox on OS X is 1/3 of Windows │ │ │ │ │ - // so force delta 1 / -1 │ │ │ │ │ - delta = -(e.detail / Math.abs(e.detail)); │ │ │ │ │ - } │ │ │ │ │ - this.delta += delta; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ + * beforegetfeatureinfo - Triggered before the request is sent. │ │ │ │ │ + * The event object has an *xy* property with the position of the │ │ │ │ │ + * mouse click or hover event that triggers the request. │ │ │ │ │ + * nogetfeatureinfo - no queryable layers were found. │ │ │ │ │ + * getfeatureinfo - Triggered when a GetFeatureInfo response is received. │ │ │ │ │ + * The event object has a *text* property with the body of the │ │ │ │ │ + * response (String), a *features* property with an array of the │ │ │ │ │ + * parsed features, an *xy* property with the position of the mouse │ │ │ │ │ + * click or hover event that triggered the request, and a *request* │ │ │ │ │ + * property with the request itself. If drillDown is set to true and │ │ │ │ │ + * multiple requests were issued to collect feature info from all │ │ │ │ │ + * layers, *text* and *request* will only contain the response body │ │ │ │ │ + * and request object of the last request. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - window.clearTimeout(this._timeoutId); │ │ │ │ │ - if (this.interval && Math.abs(this.delta) < this.maxDelta) { │ │ │ │ │ - // store e because window.event might change during delay │ │ │ │ │ - var evt = OpenLayers.Util.extend({}, e); │ │ │ │ │ - this._timeoutId = window.setTimeout( │ │ │ │ │ - OpenLayers.Function.bind(function() { │ │ │ │ │ - this.wheelZoom(evt); │ │ │ │ │ - }, this), │ │ │ │ │ - this.interval │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - this.wheelZoom(e); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.stop(e); │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: <OpenLayers.Control.WMSGetFeatureInfo> │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + │ │ │ │ │ + if (!this.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.WMSGetFeatureInfo( │ │ │ │ │ + options.formatOptions │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.drillDown === true) { │ │ │ │ │ + this.hover = false; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Hover( │ │ │ │ │ + this, { │ │ │ │ │ + 'move': this.cancelHover, │ │ │ │ │ + 'pause': this.getInfoForHover │ │ │ │ │ + }, │ │ │ │ │ + OpenLayers.Util.extend(this.handlerOptions.hover || {}, { │ │ │ │ │ + 'delay': 250 │ │ │ │ │ + })); │ │ │ │ │ + } else { │ │ │ │ │ + var callbacks = {}; │ │ │ │ │ + callbacks[this.clickCallback] = this.getInfoForClick; │ │ │ │ │ + this.handler = new OpenLayers.Handler.Click( │ │ │ │ │ + this, callbacks, this.handlerOptions.click || {}); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: wheelZoom │ │ │ │ │ - * Given the wheel event, we carry out the appropriate zooming in or out, │ │ │ │ │ - * based on the 'wheelDelta' or 'detail' property of the event. │ │ │ │ │ - * │ │ │ │ │ + * Method: getInfoForClick │ │ │ │ │ + * Called on click │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * e - {Event} │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - wheelZoom: function(e) { │ │ │ │ │ - var delta = this.delta; │ │ │ │ │ - this.delta = 0; │ │ │ │ │ - │ │ │ │ │ - if (delta) { │ │ │ │ │ - e.xy = this.map.events.getMousePosition(e); │ │ │ │ │ - if (delta < 0) { │ │ │ │ │ - this.callback("down", │ │ │ │ │ - [e, this.cumulative ? Math.max(-this.maxDelta, delta) : -1]); │ │ │ │ │ - } else { │ │ │ │ │ - this.callback("up", │ │ │ │ │ - [e, this.cumulative ? Math.min(this.maxDelta, delta) : 1]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + getInfoForClick: function(evt) { │ │ │ │ │ + this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ + xy: evt.xy │ │ │ │ │ + }); │ │ │ │ │ + // Set the cursor to "wait" to tell the user we're working on their │ │ │ │ │ + // click. │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + this.request(evt.xy, {}); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: activate │ │ │ │ │ + * Method: getInfoForHover │ │ │ │ │ + * Pause callback for the hover handler │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} │ │ │ │ │ */ │ │ │ │ │ - activate: function(evt) { │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - //register mousewheel events specifically on the window and document │ │ │ │ │ - var wheelListener = this.wheelListener; │ │ │ │ │ - OpenLayers.Event.observe(window, "DOMMouseScroll", wheelListener); │ │ │ │ │ - OpenLayers.Event.observe(window, "mousewheel", wheelListener); │ │ │ │ │ - OpenLayers.Event.observe(document, "mousewheel", wheelListener); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ + getInfoForHover: function(evt) { │ │ │ │ │ + this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ + xy: evt.xy │ │ │ │ │ + }); │ │ │ │ │ + this.request(evt.xy, { │ │ │ │ │ + hover: true │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ + * Method: cancelHover │ │ │ │ │ + * Cancel callback for the hover handler │ │ │ │ │ */ │ │ │ │ │ - deactivate: function(evt) { │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - // unregister mousewheel events specifically on the window and document │ │ │ │ │ - var wheelListener = this.wheelListener; │ │ │ │ │ - OpenLayers.Event.stopObserving(window, "DOMMouseScroll", wheelListener); │ │ │ │ │ - OpenLayers.Event.stopObserving(window, "mousewheel", wheelListener); │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mousewheel", wheelListener); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ + cancelHover: function() { │ │ │ │ │ + if (this.hoverRequest) { │ │ │ │ │ + this.hoverRequest.abort(); │ │ │ │ │ + this.hoverRequest = null; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.MouseWheel" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Keyboard.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.handler.Keyboard │ │ │ │ │ - * A handler for keyboard events. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Handler.Keyboard> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.Keyboard = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - │ │ │ │ │ - /* http://www.quirksmode.org/js/keys.html explains key x-browser │ │ │ │ │ - key handling quirks in pretty nice detail */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: KEY_EVENTS │ │ │ │ │ - * keydown, keypress, keyup │ │ │ │ │ + /** │ │ │ │ │ + * Method: findLayers │ │ │ │ │ + * Internal method to get the layers, independent of whether we are │ │ │ │ │ + * inspecting the map or using a client-provided array │ │ │ │ │ */ │ │ │ │ │ - KEY_EVENTS: ["keydown", "keyup"], │ │ │ │ │ + findLayers: function() { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: eventListener │ │ │ │ │ - * {Function} │ │ │ │ │ - */ │ │ │ │ │ - eventListener: null, │ │ │ │ │ + var candidates = this.layers || this.map.layers; │ │ │ │ │ + var layers = []; │ │ │ │ │ + var layer, url; │ │ │ │ │ + for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = candidates[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.WMS && │ │ │ │ │ + (!this.queryVisible || layer.getVisibility())) { │ │ │ │ │ + url = OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url; │ │ │ │ │ + // if the control was not configured with a url, set it │ │ │ │ │ + // to the first layer url │ │ │ │ │ + if (this.drillDown === false && !this.url) { │ │ │ │ │ + this.url = url; │ │ │ │ │ + } │ │ │ │ │ + if (this.drillDown === true || this.urlMatches(url)) { │ │ │ │ │ + layers.push(layer); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return layers; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: observeElement │ │ │ │ │ - * {DOMElement|String} The DOM element on which we listen for │ │ │ │ │ - * key events. Default to the document. │ │ │ │ │ + * Method: urlMatches │ │ │ │ │ + * Test to see if the provided url matches either the control <url> or one │ │ │ │ │ + * of the <layerUrls>. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * url - {String} The url to test. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The provided url matches the control <url> or one of the │ │ │ │ │ + * <layerUrls>. │ │ │ │ │ */ │ │ │ │ │ - observeElement: null, │ │ │ │ │ + urlMatches: function(url) { │ │ │ │ │ + var matches = OpenLayers.Util.isEquivalentUrl(this.url, url); │ │ │ │ │ + if (!matches && this.layerUrls) { │ │ │ │ │ + for (var i = 0, len = this.layerUrls.length; i < len; ++i) { │ │ │ │ │ + if (OpenLayers.Util.isEquivalentUrl(this.layerUrls[i], url)) { │ │ │ │ │ + matches = true; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return matches; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Keyboard │ │ │ │ │ - * Returns a new keyboard handler. │ │ │ │ │ - * │ │ │ │ │ + * Method: buildWMSOptions │ │ │ │ │ + * Build an object with the relevant WMS options for the GetFeatureInfo request │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ - * this handler. If a handler is being used without a control, the │ │ │ │ │ - * handlers setMap method must be overridden to deal properly with │ │ │ │ │ - * the map. │ │ │ │ │ - * callbacks - {Object} An object containing a single function to be │ │ │ │ │ - * called when the drag operation is finished. The callback should │ │ │ │ │ - * expect to recieve a single argument, the pixel location of the event. │ │ │ │ │ - * Callbacks for 'keydown', 'keypress', and 'keyup' are supported. │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * handler. │ │ │ │ │ + * url - {String} The url to be used for sending the request │ │ │ │ │ + * layers - {Array(<OpenLayers.Layer.WMS)} An array of layers │ │ │ │ │ + * clickPosition - {<OpenLayers.Pixel>} The position on the map where the mouse │ │ │ │ │ + * event occurred. │ │ │ │ │ + * format - {String} The format from the corresponding GetMap request │ │ │ │ │ */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - // cache the bound event listener method so it can be unobserved later │ │ │ │ │ - this.eventListener = OpenLayers.Function.bindAsEventListener( │ │ │ │ │ - this.handleKeyEvent, this │ │ │ │ │ - ); │ │ │ │ │ + buildWMSOptions: function(url, layers, clickPosition, format) { │ │ │ │ │ + var layerNames = [], │ │ │ │ │ + styleNames = []; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + if (layers[i].params.LAYERS != null) { │ │ │ │ │ + layerNames = layerNames.concat(layers[i].params.LAYERS); │ │ │ │ │ + styleNames = styleNames.concat(this.getStyleNames(layers[i])); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var firstLayer = layers[0]; │ │ │ │ │ + // use the firstLayer's projection if it matches the map projection - │ │ │ │ │ + // this assumes that all layers will be available in this projection │ │ │ │ │ + var projection = this.map.getProjection(); │ │ │ │ │ + var layerProj = firstLayer.projection; │ │ │ │ │ + if (layerProj && layerProj.equals(this.map.getProjectionObject())) { │ │ │ │ │ + projection = layerProj.getCode(); │ │ │ │ │ + } │ │ │ │ │ + var params = OpenLayers.Util.extend({ │ │ │ │ │ + service: "WMS", │ │ │ │ │ + version: firstLayer.params.VERSION, │ │ │ │ │ + request: "GetFeatureInfo", │ │ │ │ │ + exceptions: firstLayer.params.EXCEPTIONS, │ │ │ │ │ + bbox: this.map.getExtent().toBBOX(null, │ │ │ │ │ + firstLayer.reverseAxisOrder()), │ │ │ │ │ + feature_count: this.maxFeatures, │ │ │ │ │ + height: this.map.getSize().h, │ │ │ │ │ + width: this.map.getSize().w, │ │ │ │ │ + format: format, │ │ │ │ │ + info_format: firstLayer.params.INFO_FORMAT || this.infoFormat │ │ │ │ │ + }, (parseFloat(firstLayer.params.VERSION) >= 1.3) ? { │ │ │ │ │ + crs: projection, │ │ │ │ │ + i: parseInt(clickPosition.x), │ │ │ │ │ + j: parseInt(clickPosition.y) │ │ │ │ │ + } : { │ │ │ │ │ + srs: projection, │ │ │ │ │ + x: parseInt(clickPosition.x), │ │ │ │ │ + y: parseInt(clickPosition.y) │ │ │ │ │ + }); │ │ │ │ │ + if (layerNames.length != 0) { │ │ │ │ │ + params = OpenLayers.Util.extend({ │ │ │ │ │ + layers: layerNames, │ │ │ │ │ + query_layers: layerNames, │ │ │ │ │ + styles: styleNames │ │ │ │ │ + }, params); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.applyDefaults(params, this.vendorParams); │ │ │ │ │ + return { │ │ │ │ │ + url: url, │ │ │ │ │ + params: OpenLayers.Util.upperCaseObject(params), │ │ │ │ │ + callback: function(request) { │ │ │ │ │ + this.handleResponse(clickPosition, request, url); │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * Method: getStyleNames │ │ │ │ │ + * Gets the STYLES parameter for the layer. Make sure the STYLES parameter │ │ │ │ │ + * matches the LAYERS parameter │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer.WMS>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(String)} The STYLES parameter │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.eventListener = null; │ │ │ │ │ - OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ + getStyleNames: function(layer) { │ │ │ │ │ + // in the event of a WMS layer bundling multiple layers but not │ │ │ │ │ + // specifying styles,we need the same number of commas to specify │ │ │ │ │ + // the default style for each of the layers. We can't just leave it │ │ │ │ │ + // blank as we may be including other layers that do specify styles. │ │ │ │ │ + var styleNames; │ │ │ │ │ + if (layer.params.STYLES) { │ │ │ │ │ + styleNames = layer.params.STYLES; │ │ │ │ │ + } else { │ │ │ │ │ + if (OpenLayers.Util.isArray(layer.params.LAYERS)) { │ │ │ │ │ + styleNames = new Array(layer.params.LAYERS.length); │ │ │ │ │ + } else { // Assume it's a String │ │ │ │ │ + styleNames = layer.params.LAYERS.replace(/[^,]/g, ""); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return styleNames; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: activate │ │ │ │ │ + * Method: request │ │ │ │ │ + * Sends a GetFeatureInfo request to the WMS │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * clickPosition - {<OpenLayers.Pixel>} The position on the map where the │ │ │ │ │ + * mouse event occurred. │ │ │ │ │ + * options - {Object} additional options for this method. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * - *hover* {Boolean} true if we do the request for the hover handler │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.observeElement = this.observeElement || document; │ │ │ │ │ - for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ - OpenLayers.Event.observe( │ │ │ │ │ - this.observeElement, this.KEY_EVENTS[i], this.eventListener); │ │ │ │ │ + request: function(clickPosition, options) { │ │ │ │ │ + var layers = this.findLayers(); │ │ │ │ │ + if (layers.length == 0) { │ │ │ │ │ + this.events.triggerEvent("nogetfeatureinfo"); │ │ │ │ │ + // Reset the cursor. │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + options = options || {}; │ │ │ │ │ + if (this.drillDown === false) { │ │ │ │ │ + var wmsOptions = this.buildWMSOptions(this.url, layers, │ │ │ │ │ + clickPosition, layers[0].params.FORMAT); │ │ │ │ │ + var request = OpenLayers.Request.GET(wmsOptions); │ │ │ │ │ + │ │ │ │ │ + if (options.hover === true) { │ │ │ │ │ + this.hoverRequest = request; │ │ │ │ │ } │ │ │ │ │ - return true; │ │ │ │ │ } else { │ │ │ │ │ - return false; │ │ │ │ │ + this._requestCount = 0; │ │ │ │ │ + this._numRequests = 0; │ │ │ │ │ + this.features = []; │ │ │ │ │ + // group according to service url to combine requests │ │ │ │ │ + var services = {}, │ │ │ │ │ + url; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + var layer = layers[i]; │ │ │ │ │ + var service, found = false; │ │ │ │ │ + url = OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url; │ │ │ │ │ + if (url in services) { │ │ │ │ │ + services[url].push(layer); │ │ │ │ │ + } else { │ │ │ │ │ + this._numRequests++; │ │ │ │ │ + services[url] = [layer]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var layers; │ │ │ │ │ + for (var url in services) { │ │ │ │ │ + layers = services[url]; │ │ │ │ │ + var wmsOptions = this.buildWMSOptions(url, layers, │ │ │ │ │ + clickPosition, layers[0].params.FORMAT); │ │ │ │ │ + OpenLayers.Request.GET(wmsOptions); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ + * Method: triggerGetFeatureInfo │ │ │ │ │ + * Trigger the getfeatureinfo event when all is done │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * request - {XMLHttpRequest} The request object │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} The position on the map where the │ │ │ │ │ + * mouse event occurred. │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} or │ │ │ │ │ + * {Array({Object}) when output is "object". The object has a url and a │ │ │ │ │ + * features property which contains an array of features. │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ - OpenLayers.Event.stopObserving( │ │ │ │ │ - this.observeElement, this.KEY_EVENTS[i], this.eventListener); │ │ │ │ │ - } │ │ │ │ │ - deactivated = true; │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ + triggerGetFeatureInfo: function(request, xy, features) { │ │ │ │ │ + this.events.triggerEvent("getfeatureinfo", { │ │ │ │ │ + text: request.responseText, │ │ │ │ │ + features: features, │ │ │ │ │ + request: request, │ │ │ │ │ + xy: xy │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + // Reset the cursor. │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleKeyEvent │ │ │ │ │ + * Method: handleResponse │ │ │ │ │ + * Handler for the GetFeatureInfo response. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} The position on the map where the │ │ │ │ │ + * mouse event occurred. │ │ │ │ │ + * request - {XMLHttpRequest} The request object. │ │ │ │ │ + * url - {String} The url which was used for this request. │ │ │ │ │ */ │ │ │ │ │ - handleKeyEvent: function(evt) { │ │ │ │ │ - if (this.checkModifiers(evt)) { │ │ │ │ │ - this.callback(evt.type, [evt]); │ │ │ │ │ + handleResponse: function(xy, request, url) { │ │ │ │ │ + │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ + } │ │ │ │ │ + var features = this.format.read(doc); │ │ │ │ │ + if (this.drillDown === false) { │ │ │ │ │ + this.triggerGetFeatureInfo(request, xy, features); │ │ │ │ │ + } else { │ │ │ │ │ + this._requestCount++; │ │ │ │ │ + if (this.output === "object") { │ │ │ │ │ + this._features = (this._features || []).concat({ │ │ │ │ │ + url: url, │ │ │ │ │ + features: features │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + this._features = (this._features || []).concat(features); │ │ │ │ │ + } │ │ │ │ │ + if (this._requestCount === this._numRequests) { │ │ │ │ │ + this.triggerGetFeatureInfo(request, xy, this._features.concat()); │ │ │ │ │ + delete this._features; │ │ │ │ │ + delete this._requestCount; │ │ │ │ │ + delete this._numRequests; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Keyboard" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.WMSGetFeatureInfo" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Hover.js │ │ │ │ │ + OpenLayers/Control/ModifyFeature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Handler/Drag.js │ │ │ │ │ + * @requires OpenLayers/Handler/Keyboard.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Handler.Hover │ │ │ │ │ - * The hover handler is to be used to emulate mouseovers on objects │ │ │ │ │ - * on the map that aren't DOM elements. For example one can use │ │ │ │ │ - * this handler to send WMS/GetFeatureInfo requests as the user │ │ │ │ │ - * moves the mouve over the map. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ + * Class: OpenLayers.Control.ModifyFeature │ │ │ │ │ + * Control to modify features. When activated, a click renders the vertices │ │ │ │ │ + * of a feature - these vertices can then be dragged. By default, the │ │ │ │ │ + * delete key will delete the vertex under the mouse. New features are │ │ │ │ │ + * added by dragging "virtual vertices" between vertices. Create a new │ │ │ │ │ + * control with the <OpenLayers.Control.ModifyFeature> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits From: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.Hover = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ +OpenLayers.Control.ModifyFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: delay │ │ │ │ │ - * {Integer} - Number of milliseconds between mousemoves before │ │ │ │ │ - * the event is considered a hover. Default is 500. │ │ │ │ │ + * APIProperty: documentDrag │ │ │ │ │ + * {Boolean} If set to true, dragging vertices will continue even if the │ │ │ │ │ + * mouse cursor leaves the map viewport. Default is false. │ │ │ │ │ */ │ │ │ │ │ - delay: 500, │ │ │ │ │ + documentDrag: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: pixelTolerance │ │ │ │ │ - * {Integer} - Maximum number of pixels between mousemoves for │ │ │ │ │ - * an event to be considered a hover. Default is null. │ │ │ │ │ + * APIProperty: geometryTypes │ │ │ │ │ + * {Array(String)} To restrict modification to a limited set of geometry │ │ │ │ │ + * types, send a list of strings corresponding to the geometry class │ │ │ │ │ + * names. │ │ │ │ │ */ │ │ │ │ │ - pixelTolerance: null, │ │ │ │ │ + geometryTypes: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: stopMove │ │ │ │ │ - * {Boolean} - Stop other listeners from being notified on mousemoves. │ │ │ │ │ - * Default is false. │ │ │ │ │ + * APIProperty: clickout │ │ │ │ │ + * {Boolean} Unselect features when clicking outside any feature. │ │ │ │ │ + * Default is true. │ │ │ │ │ */ │ │ │ │ │ - stopMove: false, │ │ │ │ │ + clickout: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: px │ │ │ │ │ - * {<OpenLayers.Pixel>} - The location of the last mousemove, expressed │ │ │ │ │ - * in pixels. │ │ │ │ │ + * APIProperty: toggle │ │ │ │ │ + * {Boolean} Unselect a selected feature on click. │ │ │ │ │ + * Default is true. │ │ │ │ │ */ │ │ │ │ │ - px: null, │ │ │ │ │ + toggle: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: timerId │ │ │ │ │ - * {Number} - The id of the timer. │ │ │ │ │ + * APIProperty: standalone │ │ │ │ │ + * {Boolean} Set to true to create a control without SelectFeature │ │ │ │ │ + * capabilities. Default is false. If standalone is true, to modify │ │ │ │ │ + * a feature, call the <selectFeature> method with the target feature. │ │ │ │ │ + * Note that you must call the <unselectFeature> method to finish │ │ │ │ │ + * feature modification in standalone mode (before starting to modify │ │ │ │ │ + * another feature). │ │ │ │ │ */ │ │ │ │ │ - timerId: null, │ │ │ │ │ + standalone: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Hover │ │ │ │ │ - * Construct a hover handler. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that initialized this │ │ │ │ │ - * handler. The control is assumed to have a valid map property; that │ │ │ │ │ - * map is used in the handler's own setMap method. │ │ │ │ │ - * callbacks - {Object} An object with keys corresponding to callbacks │ │ │ │ │ - * that will be called by the handler. The callbacks should │ │ │ │ │ - * expect to receive a single argument, the event. Callbacks for │ │ │ │ │ - * 'move', the mouse is moving, and 'pause', the mouse is pausing, │ │ │ │ │ - * are supported. │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * the handler. │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} │ │ │ │ │ */ │ │ │ │ │ + layer: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mousemove │ │ │ │ │ - * Called when the mouse moves on the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ + * Property: feature │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} Feature currently available for modification. │ │ │ │ │ */ │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - if (this.passesTolerance(evt.xy)) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.callback('move', [evt]); │ │ │ │ │ - this.px = evt.xy; │ │ │ │ │ - // clone the evt so original properties can be accessed even │ │ │ │ │ - // if the browser deletes them during the delay │ │ │ │ │ - evt = OpenLayers.Util.extend({}, evt); │ │ │ │ │ - this.timerId = window.setTimeout( │ │ │ │ │ - OpenLayers.Function.bind(this.delayedCall, this, evt), │ │ │ │ │ - this.delay │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return !this.stopMove; │ │ │ │ │ - }, │ │ │ │ │ + feature: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mouseout │ │ │ │ │ - * Called when the mouse goes out of the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ + * Property: vertex │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} Vertex currently being modified. │ │ │ │ │ */ │ │ │ │ │ - mouseout: function(evt) { │ │ │ │ │ - if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.callback('move', [evt]); │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ + vertex: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: passesTolerance │ │ │ │ │ - * Determine whether the mouse move is within the optional pixel tolerance. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The mouse move is within the pixel tolerance. │ │ │ │ │ + * Property: vertices │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} Verticies currently available │ │ │ │ │ + * for dragging. │ │ │ │ │ */ │ │ │ │ │ - passesTolerance: function(px) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - if (this.pixelTolerance && this.px) { │ │ │ │ │ - var dpx = Math.sqrt( │ │ │ │ │ - Math.pow(this.px.x - px.x, 2) + │ │ │ │ │ - Math.pow(this.px.y - px.y, 2) │ │ │ │ │ - ); │ │ │ │ │ - if (dpx < this.pixelTolerance) { │ │ │ │ │ - passes = false; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return passes; │ │ │ │ │ - }, │ │ │ │ │ + vertices: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clearTimer │ │ │ │ │ - * Clear the timer and set <timerId> to null. │ │ │ │ │ + * Property: virtualVertices │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} Virtual vertices in the middle │ │ │ │ │ + * of each edge. │ │ │ │ │ */ │ │ │ │ │ - clearTimer: function() { │ │ │ │ │ - if (this.timerId != null) { │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + virtualVertices: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: delayedCall │ │ │ │ │ - * Triggers pause callback. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ + * Property: handlers │ │ │ │ │ + * {Object} │ │ │ │ │ */ │ │ │ │ │ - delayedCall: function(evt) { │ │ │ │ │ - this.callback('pause', [evt]); │ │ │ │ │ - }, │ │ │ │ │ + handlers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the handler. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully deactivated. │ │ │ │ │ + * APIProperty: deleteCodes │ │ │ │ │ + * {Array(Integer)} Keycodes for deleting verticies. Set to null to disable │ │ │ │ │ + * vertex deltion by keypress. If non-null, keypresses with codes │ │ │ │ │ + * in this array will delete vertices under the mouse. Default │ │ │ │ │ + * is 46 and 68, the 'delete' and lowercase 'd' keys. │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - deactivated = true; │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Hover" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Pinch.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + deleteCodes: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: virtualStyle │ │ │ │ │ + * {Object} A symbolizer to be used for virtual vertices. │ │ │ │ │ + */ │ │ │ │ │ + virtualStyle: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Handler.Pinch │ │ │ │ │ - * The pinch handler is used to deal with sequences of browser events related │ │ │ │ │ - * to pinch gestures. The handler is used by controls that want to know │ │ │ │ │ - * when a pinch sequence begins, when a pinch is happening, and when it has │ │ │ │ │ - * finished. │ │ │ │ │ - * │ │ │ │ │ - * Controls that use the pinch handler typically construct it with callbacks │ │ │ │ │ - * for 'start', 'move', and 'done'. Callbacks for these keys are │ │ │ │ │ - * called when the pinch begins, with each change, and when the pinch is │ │ │ │ │ - * done. │ │ │ │ │ - * │ │ │ │ │ - * Create a new pinch handler with the <OpenLayers.Handler.Pinch> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.Pinch = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: vertexRenderIntent │ │ │ │ │ + * {String} The renderIntent to use for vertices. If no <virtualStyle> is │ │ │ │ │ + * provided, this renderIntent will also be used for virtual vertices, with │ │ │ │ │ + * a fillOpacity and strokeOpacity of 0.3. Default is null, which means │ │ │ │ │ + * that the layer's default style will be used for vertices. │ │ │ │ │ + */ │ │ │ │ │ + vertexRenderIntent: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: started │ │ │ │ │ - * {Boolean} When a touchstart event is received, we want to record it, │ │ │ │ │ - * but not set 'pinching' until the touchmove get started after │ │ │ │ │ - * starting. │ │ │ │ │ + * APIProperty: mode │ │ │ │ │ + * {Integer} Bitfields specifying the modification mode. Defaults to │ │ │ │ │ + * OpenLayers.Control.ModifyFeature.RESHAPE. To set the mode to a │ │ │ │ │ + * combination of options, use the | operator. For example, to allow │ │ │ │ │ + * the control to both resize and rotate features, use the following │ │ │ │ │ + * syntax │ │ │ │ │ + * (code) │ │ │ │ │ + * control.mode = OpenLayers.Control.ModifyFeature.RESIZE | │ │ │ │ │ + * OpenLayers.Control.ModifyFeature.ROTATE; │ │ │ │ │ + * (end) │ │ │ │ │ */ │ │ │ │ │ - started: false, │ │ │ │ │ + mode: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: stopDown │ │ │ │ │ - * {Boolean} Stop propagation of touchstart events from getting to │ │ │ │ │ - * listeners on the same element. Default is false. │ │ │ │ │ + * APIProperty: createVertices │ │ │ │ │ + * {Boolean} Create new vertices by dragging the virtual vertices │ │ │ │ │ + * in the middle of each edge. Default is true. │ │ │ │ │ */ │ │ │ │ │ - stopDown: false, │ │ │ │ │ + createVertices: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: pinching │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * Property: modified │ │ │ │ │ + * {Boolean} The currently selected feature has been modified. │ │ │ │ │ */ │ │ │ │ │ - pinching: false, │ │ │ │ │ + modified: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: last │ │ │ │ │ - * {Object} Object that store informations related to pinch last touch. │ │ │ │ │ + * Property: radiusHandle │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A handle for rotating/resizing a feature. │ │ │ │ │ */ │ │ │ │ │ - last: null, │ │ │ │ │ + radiusHandle: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: start │ │ │ │ │ - * {Object} Object that store informations related to pinch touchstart. │ │ │ │ │ + * Property: dragHandle │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A handle for dragging a feature. │ │ │ │ │ */ │ │ │ │ │ - start: null, │ │ │ │ │ + dragHandle: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Pinch │ │ │ │ │ - * Returns OpenLayers.Handler.Pinch │ │ │ │ │ + * APIProperty: onModificationStart │ │ │ │ │ + * {Function} *Deprecated*. Register for "beforefeaturemodified" instead. │ │ │ │ │ + * The "beforefeaturemodified" event is triggered on the layer before │ │ │ │ │ + * any modification begins. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ - * this handler. If a handler is being used without a control, the │ │ │ │ │ - * handlers setMap method must be overridden to deal properly with │ │ │ │ │ - * the map. │ │ │ │ │ - * callbacks - {Object} An object containing functions to be called when │ │ │ │ │ - * the pinch operation start, change, or is finished. The callbacks │ │ │ │ │ - * should expect to receive an object argument, which contains │ │ │ │ │ - * information about scale, distance, and position of touch points. │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * Optional function to be called when a feature is selected │ │ │ │ │ + * to be modified. The function should expect to be called with a │ │ │ │ │ + * feature. This could be used for example to allow to lock the │ │ │ │ │ + * feature on server-side. │ │ │ │ │ */ │ │ │ │ │ + onModificationStart: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchstart │ │ │ │ │ - * Handle touchstart events │ │ │ │ │ + * APIProperty: onModification │ │ │ │ │ + * {Function} *Deprecated*. Register for "featuremodified" instead. │ │ │ │ │ + * The "featuremodified" event is triggered on the layer with each │ │ │ │ │ + * feature modification. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * Optional function to be called when a feature has been │ │ │ │ │ + * modified. The function should expect to be called with a feature. │ │ │ │ │ + */ │ │ │ │ │ + onModification: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: onModificationEnd │ │ │ │ │ + * {Function} *Deprecated*. Register for "afterfeaturemodified" instead. │ │ │ │ │ + * The "afterfeaturemodified" event is triggered on the layer after │ │ │ │ │ + * a feature has been modified. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * Optional function to be called when a feature is finished │ │ │ │ │ + * being modified. The function should expect to be called with a │ │ │ │ │ + * feature. │ │ │ │ │ */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - if (OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ - this.started = true; │ │ │ │ │ - this.last = this.start = { │ │ │ │ │ - distance: this.getDistance(evt.touches), │ │ │ │ │ - delta: 0, │ │ │ │ │ - scale: 1 │ │ │ │ │ - }; │ │ │ │ │ - this.callback("start", [evt, this.start]); │ │ │ │ │ - propagate = !this.stopDown; │ │ │ │ │ - } else if (this.started) { │ │ │ │ │ - // Some webkit versions send fake single-touch events during │ │ │ │ │ - // multitouch, which cause the drag handler to trigger │ │ │ │ │ - return false; │ │ │ │ │ - } else { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - } │ │ │ │ │ - // prevent document dragging │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ - return propagate; │ │ │ │ │ - }, │ │ │ │ │ + onModificationEnd: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchmove │ │ │ │ │ - * Handle touchmove events │ │ │ │ │ + * Constructor: OpenLayers.Control.ModifyFeature │ │ │ │ │ + * Create a new modify feature control. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * layer - {<OpenLayers.Layer.Vector>} Layer that contains features that │ │ │ │ │ + * will be modified. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * control. │ │ │ │ │ */ │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - if (this.started && OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ - this.pinching = true; │ │ │ │ │ - var current = this.getPinchData(evt); │ │ │ │ │ - this.callback("move", [evt, current]); │ │ │ │ │ - this.last = current; │ │ │ │ │ - // prevent document dragging │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - } else if (this.started) { │ │ │ │ │ - // Some webkit versions send fake single-touch events during │ │ │ │ │ - // multitouch, which cause the drag handler to trigger │ │ │ │ │ - return false; │ │ │ │ │ + initialize: function(layer, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + this.virtualVertices = []; │ │ │ │ │ + this.virtualStyle = OpenLayers.Util.extend({}, │ │ │ │ │ + this.layer.style || │ │ │ │ │ + this.layer.styleMap.createSymbolizer(null, options.vertexRenderIntent) │ │ │ │ │ + ); │ │ │ │ │ + this.virtualStyle.fillOpacity = 0.3; │ │ │ │ │ + this.virtualStyle.strokeOpacity = 0.3; │ │ │ │ │ + this.deleteCodes = [46, 68]; │ │ │ │ │ + this.mode = OpenLayers.Control.ModifyFeature.RESHAPE; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!(OpenLayers.Util.isArray(this.deleteCodes))) { │ │ │ │ │ + this.deleteCodes = [this.deleteCodes]; │ │ │ │ │ } │ │ │ │ │ - return true; │ │ │ │ │ + │ │ │ │ │ + // configure the drag handler │ │ │ │ │ + var dragCallbacks = { │ │ │ │ │ + down: function(pixel) { │ │ │ │ │ + this.vertex = null; │ │ │ │ │ + var feature = this.layer.getFeatureFromEvent( │ │ │ │ │ + this.handlers.drag.evt); │ │ │ │ │ + if (feature) { │ │ │ │ │ + this.dragStart(feature); │ │ │ │ │ + } else if (this.clickout) { │ │ │ │ │ + this._unselect = this.feature; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + move: function(pixel) { │ │ │ │ │ + delete this._unselect; │ │ │ │ │ + if (this.vertex) { │ │ │ │ │ + this.dragVertex(this.vertex, pixel); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + up: function() { │ │ │ │ │ + this.handlers.drag.stopDown = false; │ │ │ │ │ + if (this._unselect) { │ │ │ │ │ + this.unselectFeature(this._unselect); │ │ │ │ │ + delete this._unselect; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + done: function(pixel) { │ │ │ │ │ + if (this.vertex) { │ │ │ │ │ + this.dragComplete(this.vertex); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + var dragOptions = { │ │ │ │ │ + documentDrag: this.documentDrag, │ │ │ │ │ + stopDown: false │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // configure the keyboard handler │ │ │ │ │ + var keyboardOptions = { │ │ │ │ │ + keydown: this.handleKeypress │ │ │ │ │ + }; │ │ │ │ │ + this.handlers = { │ │ │ │ │ + keyboard: new OpenLayers.Handler.Keyboard(this, keyboardOptions), │ │ │ │ │ + drag: new OpenLayers.Handler.Drag(this, dragCallbacks, dragOptions) │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchend │ │ │ │ │ - * Handle touchend events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Take care of things that are not handled in superclass. │ │ │ │ │ */ │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - if (this.started && !OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - this.callback("done", [evt, this.start, this.last]); │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - return false; │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + "removelayer": this.handleMapEvents, │ │ │ │ │ + "changelayer": this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - return true; │ │ │ │ │ + this.layer = null; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, []); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Activate the handler. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Activate the control. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully activated. │ │ │ │ │ + * {Boolean} Successfully activated the control. │ │ │ │ │ */ │ │ │ │ │ activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - activated = true; │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ + this.moveLayerToTop(); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + "removelayer": this.handleMapEvents, │ │ │ │ │ + "changelayer": this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + return (this.handlers.keyboard.activate() && │ │ │ │ │ + this.handlers.drag.activate() && │ │ │ │ │ + OpenLayers.Control.prototype.activate.apply(this, arguments)); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Deactivate the handler. │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the control. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully deactivated. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Successfully deactivated the control. │ │ │ │ │ */ │ │ │ │ │ deactivate: function() { │ │ │ │ │ var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ + // the return from the controls is unimportant in this case │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerBack(); │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + "removelayer": this.handleMapEvents, │ │ │ │ │ + "changelayer": this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.layer.removeFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.layer.removeFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + this.handlers.drag.deactivate(); │ │ │ │ │ + this.handlers.keyboard.deactivate(); │ │ │ │ │ + var feature = this.feature; │ │ │ │ │ + if (feature && feature.geometry && feature.layer) { │ │ │ │ │ + this.unselectFeature(feature); │ │ │ │ │ + } │ │ │ │ │ deactivated = true; │ │ │ │ │ } │ │ │ │ │ return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getDistance │ │ │ │ │ - * Get the distance in pixels between two touches. │ │ │ │ │ + * Method: beforeSelectFeature │ │ │ │ │ + * Called before a feature is selected. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * touches - {Array(Object)} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number} The distance in pixels. │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} The feature about to be selected. │ │ │ │ │ */ │ │ │ │ │ - getDistance: function(touches) { │ │ │ │ │ - var t0 = touches[0]; │ │ │ │ │ - var t1 = touches[1]; │ │ │ │ │ - return Math.sqrt( │ │ │ │ │ - Math.pow(t0.olClientX - t1.olClientX, 2) + │ │ │ │ │ - Math.pow(t0.olClientY - t1.olClientY, 2) │ │ │ │ │ + beforeSelectFeature: function(feature) { │ │ │ │ │ + return this.layer.events.triggerEvent( │ │ │ │ │ + "beforefeaturemodified", { │ │ │ │ │ + feature: feature │ │ │ │ │ + } │ │ │ │ │ ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: getPinchData │ │ │ │ │ - * Get informations about the pinch event. │ │ │ │ │ + * APIMethod: selectFeature │ │ │ │ │ + * Select a feature for modification in standalone mode. In non-standalone │ │ │ │ │ + * mode, this method is called when a feature is selected by clicking. │ │ │ │ │ + * Register a listener to the beforefeaturemodified event and return false │ │ │ │ │ + * to prevent feature modification. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Object that contains data about the current pinch. │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} the selected feature. │ │ │ │ │ */ │ │ │ │ │ - getPinchData: function(evt) { │ │ │ │ │ - var distance = this.getDistance(evt.touches); │ │ │ │ │ - var scale = distance / this.start.distance; │ │ │ │ │ - return { │ │ │ │ │ - distance: distance, │ │ │ │ │ - delta: this.last.distance - distance, │ │ │ │ │ - scale: scale │ │ │ │ │ - }; │ │ │ │ │ + selectFeature: function(feature) { │ │ │ │ │ + if (this.feature === feature || │ │ │ │ │ + (this.geometryTypes && OpenLayers.Util.indexOf(this.geometryTypes, │ │ │ │ │ + feature.geometry.CLASS_NAME) == -1)) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (this.beforeSelectFeature(feature) !== false) { │ │ │ │ │ + if (this.feature) { │ │ │ │ │ + this.unselectFeature(this.feature); │ │ │ │ │ + } │ │ │ │ │ + this.feature = feature; │ │ │ │ │ + this.layer.selectedFeatures.push(feature); │ │ │ │ │ + this.layer.drawFeature(feature, 'select'); │ │ │ │ │ + this.modified = false; │ │ │ │ │ + this.resetVertices(); │ │ │ │ │ + this.onModificationStart(this.feature); │ │ │ │ │ + } │ │ │ │ │ + // keep track of geometry modifications │ │ │ │ │ + var modified = feature.modified; │ │ │ │ │ + if (feature.geometry && !(modified && modified.geometry)) { │ │ │ │ │ + this._originalGeometry = feature.geometry.clone(); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Pinch" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Box.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ - * @requires OpenLayers/Handler/Drag.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Handler.Box │ │ │ │ │ - * Handler for dragging a rectangle across the map. Box is displayed │ │ │ │ │ - * on mouse down, moves on mouse move, and is finished on mouse up. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.Box = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: dragHandler │ │ │ │ │ - * {<OpenLayers.Handler.Drag>} │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: unselectFeature │ │ │ │ │ + * Called when the select feature control unselects a feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} The unselected feature. │ │ │ │ │ */ │ │ │ │ │ - dragHandler: null, │ │ │ │ │ + unselectFeature: function(feature) { │ │ │ │ │ + this.layer.removeFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + this.layer.destroyFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.virtualVertices = []; │ │ │ │ │ + if (this.dragHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.dragHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + delete this.dragHandle; │ │ │ │ │ + } │ │ │ │ │ + if (this.radiusHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + delete this.radiusHandle; │ │ │ │ │ + } │ │ │ │ │ + this.layer.drawFeature(this.feature, 'default'); │ │ │ │ │ + this.feature = null; │ │ │ │ │ + OpenLayers.Util.removeItem(this.layer.selectedFeatures, feature); │ │ │ │ │ + this.onModificationEnd(feature); │ │ │ │ │ + this.layer.events.triggerEvent("afterfeaturemodified", { │ │ │ │ │ + feature: feature, │ │ │ │ │ + modified: this.modified │ │ │ │ │ + }); │ │ │ │ │ + this.modified = false; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: boxDivClassName │ │ │ │ │ - * {String} The CSS class to use for drawing the box. Default is │ │ │ │ │ - * olHandlerBoxZoomBox │ │ │ │ │ + * Method: dragStart │ │ │ │ │ + * Called by the drag handler before a feature is dragged. This method is │ │ │ │ │ + * used to differentiate between points and vertices │ │ │ │ │ + * of higher order geometries. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} The point or vertex about to be │ │ │ │ │ + * dragged. │ │ │ │ │ */ │ │ │ │ │ - boxDivClassName: 'olHandlerBoxZoomBox', │ │ │ │ │ + dragStart: function(feature) { │ │ │ │ │ + var isPoint = feature.geometry.CLASS_NAME == │ │ │ │ │ + 'OpenLayers.Geometry.Point'; │ │ │ │ │ + if (!this.standalone && │ │ │ │ │ + ((!feature._sketch && isPoint) || !feature._sketch)) { │ │ │ │ │ + if (this.toggle && this.feature === feature) { │ │ │ │ │ + // mark feature for unselection │ │ │ │ │ + this._unselect = feature; │ │ │ │ │ + } │ │ │ │ │ + this.selectFeature(feature); │ │ │ │ │ + } │ │ │ │ │ + if (feature._sketch || isPoint) { │ │ │ │ │ + // feature is a drag or virtual handle or point │ │ │ │ │ + this.vertex = feature; │ │ │ │ │ + this.handlers.drag.stopDown = true; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: boxOffsets │ │ │ │ │ - * {Object} Caches box offsets from css. This is used by the getBoxOffsets │ │ │ │ │ - * method. │ │ │ │ │ + * Method: dragVertex │ │ │ │ │ + * Called by the drag handler with each drag move of a vertex. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * vertex - {<OpenLayers.Feature.Vector>} The vertex being dragged. │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} Pixel location of the mouse event. │ │ │ │ │ */ │ │ │ │ │ - boxOffsets: null, │ │ │ │ │ + dragVertex: function(vertex, pixel) { │ │ │ │ │ + var pos = this.map.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geom = vertex.geometry; │ │ │ │ │ + geom.move(pos.lon - geom.x, pos.lat - geom.y); │ │ │ │ │ + this.modified = true; │ │ │ │ │ + /** │ │ │ │ │ + * Five cases: │ │ │ │ │ + * 1) dragging a simple point │ │ │ │ │ + * 2) dragging a virtual vertex │ │ │ │ │ + * 3) dragging a drag handle │ │ │ │ │ + * 4) dragging a real vertex │ │ │ │ │ + * 5) dragging a radius handle │ │ │ │ │ + */ │ │ │ │ │ + if (this.feature.geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + // dragging a simple point │ │ │ │ │ + this.layer.events.triggerEvent("vertexmodified", { │ │ │ │ │ + vertex: vertex.geometry, │ │ │ │ │ + feature: this.feature, │ │ │ │ │ + pixel: pixel │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + if (vertex._index) { │ │ │ │ │ + // dragging a virtual vertex │ │ │ │ │ + vertex.geometry.parent.addComponent(vertex.geometry, │ │ │ │ │ + vertex._index); │ │ │ │ │ + // move from virtual to real vertex │ │ │ │ │ + delete vertex._index; │ │ │ │ │ + OpenLayers.Util.removeItem(this.virtualVertices, vertex); │ │ │ │ │ + this.vertices.push(vertex); │ │ │ │ │ + } else if (vertex == this.dragHandle) { │ │ │ │ │ + // dragging a drag handle │ │ │ │ │ + this.layer.removeFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + if (this.radiusHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.radiusHandle = null; │ │ │ │ │ + } │ │ │ │ │ + } else if (vertex !== this.radiusHandle) { │ │ │ │ │ + // dragging a real vertex │ │ │ │ │ + this.layer.events.triggerEvent("vertexmodified", { │ │ │ │ │ + vertex: vertex.geometry, │ │ │ │ │ + feature: this.feature, │ │ │ │ │ + pixel: pixel │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + // dragging a radius handle - no special treatment │ │ │ │ │ + if (this.virtualVertices.length > 0) { │ │ │ │ │ + this.layer.destroyFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.virtualVertices = []; │ │ │ │ │ + } │ │ │ │ │ + this.layer.drawFeature(this.feature, this.standalone ? undefined : │ │ │ │ │ + 'select'); │ │ │ │ │ + } │ │ │ │ │ + // keep the vertex on top so it gets the mouseout after dragging │ │ │ │ │ + // this should be removed in favor of an option to draw under or │ │ │ │ │ + // maintain node z-index │ │ │ │ │ + this.layer.drawFeature(vertex); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Box │ │ │ │ │ + * Method: dragComplete │ │ │ │ │ + * Called by the drag handler when the feature dragging is complete. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} │ │ │ │ │ - * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ - * functions. Various callbacks described below. │ │ │ │ │ - * options - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Named callbacks: │ │ │ │ │ - * start - Called when the box drag operation starts. │ │ │ │ │ - * done - Called when the box drag operation is finished. │ │ │ │ │ - * The callback should expect to receive a single argument, the box │ │ │ │ │ - * bounds or a pixel. If the box dragging didn't span more than a 5 │ │ │ │ │ - * pixel distance, a pixel will be returned instead of a bounds object. │ │ │ │ │ + * vertex - {<OpenLayers.Feature.Vector>} The vertex being dragged. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.dragHandler = new OpenLayers.Handler.Drag( │ │ │ │ │ - this, { │ │ │ │ │ - down: this.startBox, │ │ │ │ │ - move: this.moveBox, │ │ │ │ │ - out: this.removeBox, │ │ │ │ │ - up: this.endBox │ │ │ │ │ - }, { │ │ │ │ │ - keyMask: this.keyMask │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ + dragComplete: function(vertex) { │ │ │ │ │ + this.resetVertices(); │ │ │ │ │ + this.setFeatureState(); │ │ │ │ │ + this.onModification(this.feature); │ │ │ │ │ + this.layer.events.triggerEvent("featuremodified", { │ │ │ │ │ + feature: this.feature │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * Method: setFeatureState │ │ │ │ │ + * Called when the feature is modified. If the current state is not │ │ │ │ │ + * INSERT or DELETE, the state is set to UPDATE. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ - if (this.dragHandler) { │ │ │ │ │ - this.dragHandler.destroy(); │ │ │ │ │ - this.dragHandler = null; │ │ │ │ │ + setFeatureState: function() { │ │ │ │ │ + if (this.feature.state != OpenLayers.State.INSERT && │ │ │ │ │ + this.feature.state != OpenLayers.State.DELETE) { │ │ │ │ │ + this.feature.state = OpenLayers.State.UPDATE; │ │ │ │ │ + if (this.modified && this._originalGeometry) { │ │ │ │ │ + var feature = this.feature; │ │ │ │ │ + feature.modified = OpenLayers.Util.extend(feature.modified, { │ │ │ │ │ + geometry: this._originalGeometry │ │ │ │ │ + }); │ │ │ │ │ + delete this._originalGeometry; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ + * Method: resetVertices │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Handler.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (this.dragHandler) { │ │ │ │ │ - this.dragHandler.setMap(map); │ │ │ │ │ + resetVertices: function() { │ │ │ │ │ + if (this.vertices.length > 0) { │ │ │ │ │ + this.layer.removeFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + } │ │ │ │ │ + if (this.virtualVertices.length > 0) { │ │ │ │ │ + this.layer.removeFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.virtualVertices = []; │ │ │ │ │ + } │ │ │ │ │ + if (this.dragHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.dragHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.dragHandle = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.radiusHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.radiusHandle = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.feature && │ │ │ │ │ + this.feature.geometry.CLASS_NAME != "OpenLayers.Geometry.Point") { │ │ │ │ │ + if ((this.mode & OpenLayers.Control.ModifyFeature.DRAG)) { │ │ │ │ │ + this.collectDragHandle(); │ │ │ │ │ + } │ │ │ │ │ + if ((this.mode & (OpenLayers.Control.ModifyFeature.ROTATE | │ │ │ │ │ + OpenLayers.Control.ModifyFeature.RESIZE))) { │ │ │ │ │ + this.collectRadiusHandle(); │ │ │ │ │ + } │ │ │ │ │ + if (this.mode & OpenLayers.Control.ModifyFeature.RESHAPE) { │ │ │ │ │ + // Don't collect vertices when we're resizing │ │ │ │ │ + if (!(this.mode & OpenLayers.Control.ModifyFeature.RESIZE)) { │ │ │ │ │ + this.collectVertices(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: startBox │ │ │ │ │ + * Method: handleKeypress │ │ │ │ │ + * Called by the feature handler on keypress. This is used to delete │ │ │ │ │ + * vertices. If the <deleteCode> property is set, vertices will │ │ │ │ │ + * be deleted when a feature is selected for modification and │ │ │ │ │ + * the mouse is over a vertex. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} │ │ │ │ │ + * evt - {Event} Keypress event. │ │ │ │ │ */ │ │ │ │ │ - startBox: function(xy) { │ │ │ │ │ - this.callback("start", []); │ │ │ │ │ - this.zoomBox = OpenLayers.Util.createDiv('zoomBox', { │ │ │ │ │ - x: -9999, │ │ │ │ │ - y: -9999 │ │ │ │ │ - }); │ │ │ │ │ - this.zoomBox.className = this.boxDivClassName; │ │ │ │ │ - this.zoomBox.style.zIndex = this.map.Z_INDEX_BASE["Popup"] - 1; │ │ │ │ │ - │ │ │ │ │ - this.map.viewPortDiv.appendChild(this.zoomBox); │ │ │ │ │ + handleKeypress: function(evt) { │ │ │ │ │ + var code = evt.keyCode; │ │ │ │ │ │ │ │ │ │ - OpenLayers.Element.addClass( │ │ │ │ │ - this.map.viewPortDiv, "olDrawBox" │ │ │ │ │ - ); │ │ │ │ │ + // check for delete key │ │ │ │ │ + if (this.feature && │ │ │ │ │ + OpenLayers.Util.indexOf(this.deleteCodes, code) != -1) { │ │ │ │ │ + var vertex = this.layer.getFeatureFromEvent(this.handlers.drag.evt); │ │ │ │ │ + if (vertex && │ │ │ │ │ + OpenLayers.Util.indexOf(this.vertices, vertex) != -1 && │ │ │ │ │ + !this.handlers.drag.dragging && vertex.geometry.parent) { │ │ │ │ │ + // remove the vertex │ │ │ │ │ + vertex.geometry.parent.removeComponent(vertex.geometry); │ │ │ │ │ + this.layer.events.triggerEvent("vertexremoved", { │ │ │ │ │ + vertex: vertex.geometry, │ │ │ │ │ + feature: this.feature, │ │ │ │ │ + pixel: evt.xy │ │ │ │ │ + }); │ │ │ │ │ + this.layer.drawFeature(this.feature, this.standalone ? │ │ │ │ │ + undefined : 'select'); │ │ │ │ │ + this.modified = true; │ │ │ │ │ + this.resetVertices(); │ │ │ │ │ + this.setFeatureState(); │ │ │ │ │ + this.onModification(this.feature); │ │ │ │ │ + this.layer.events.triggerEvent("featuremodified", { │ │ │ │ │ + feature: this.feature │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveBox │ │ │ │ │ + * Method: collectVertices │ │ │ │ │ + * Collect the vertices from the modifiable feature's geometry and push │ │ │ │ │ + * them on to the control's vertices array. │ │ │ │ │ */ │ │ │ │ │ - moveBox: function(xy) { │ │ │ │ │ - var startX = this.dragHandler.start.x; │ │ │ │ │ - var startY = this.dragHandler.start.y; │ │ │ │ │ - var deltaX = Math.abs(startX - xy.x); │ │ │ │ │ - var deltaY = Math.abs(startY - xy.y); │ │ │ │ │ + collectVertices: function() { │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + this.virtualVertices = []; │ │ │ │ │ + var control = this; │ │ │ │ │ │ │ │ │ │ - var offset = this.getBoxOffsets(); │ │ │ │ │ - this.zoomBox.style.width = (deltaX + offset.width + 1) + "px"; │ │ │ │ │ - this.zoomBox.style.height = (deltaY + offset.height + 1) + "px"; │ │ │ │ │ - this.zoomBox.style.left = (xy.x < startX ? │ │ │ │ │ - startX - deltaX - offset.left : startX - offset.left) + "px"; │ │ │ │ │ - this.zoomBox.style.top = (xy.y < startY ? │ │ │ │ │ - startY - deltaY - offset.top : startY - offset.top) + "px"; │ │ │ │ │ + function collectComponentVertices(geometry) { │ │ │ │ │ + var i, vertex, component, len; │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + vertex = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + vertex._sketch = true; │ │ │ │ │ + vertex.renderIntent = control.vertexRenderIntent; │ │ │ │ │ + control.vertices.push(vertex); │ │ │ │ │ + } else { │ │ │ │ │ + var numVert = geometry.components.length; │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ + numVert -= 1; │ │ │ │ │ + } │ │ │ │ │ + for (i = 0; i < numVert; ++i) { │ │ │ │ │ + component = geometry.components[i]; │ │ │ │ │ + if (component.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + vertex = new OpenLayers.Feature.Vector(component); │ │ │ │ │ + vertex._sketch = true; │ │ │ │ │ + vertex.renderIntent = control.vertexRenderIntent; │ │ │ │ │ + control.vertices.push(vertex); │ │ │ │ │ + } else { │ │ │ │ │ + collectComponentVertices(component); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // add virtual vertices in the middle of each edge │ │ │ │ │ + if (control.createVertices && geometry.CLASS_NAME != "OpenLayers.Geometry.MultiPoint") { │ │ │ │ │ + for (i = 0, len = geometry.components.length; i < len - 1; ++i) { │ │ │ │ │ + var prevVertex = geometry.components[i]; │ │ │ │ │ + var nextVertex = geometry.components[i + 1]; │ │ │ │ │ + if (prevVertex.CLASS_NAME == "OpenLayers.Geometry.Point" && │ │ │ │ │ + nextVertex.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + var x = (prevVertex.x + nextVertex.x) / 2; │ │ │ │ │ + var y = (prevVertex.y + nextVertex.y) / 2; │ │ │ │ │ + var point = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.Point(x, y), │ │ │ │ │ + null, control.virtualStyle │ │ │ │ │ + ); │ │ │ │ │ + // set the virtual parent and intended index │ │ │ │ │ + point.geometry.parent = geometry; │ │ │ │ │ + point._index = i + 1; │ │ │ │ │ + point._sketch = true; │ │ │ │ │ + control.virtualVertices.push(point); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + collectComponentVertices.call(this, this.feature.geometry); │ │ │ │ │ + this.layer.addFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.layer.addFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: endBox │ │ │ │ │ + * Method: collectDragHandle │ │ │ │ │ + * Collect the drag handle for the selected geometry. │ │ │ │ │ */ │ │ │ │ │ - endBox: function(end) { │ │ │ │ │ - var result; │ │ │ │ │ - if (Math.abs(this.dragHandler.start.x - end.x) > 5 || │ │ │ │ │ - Math.abs(this.dragHandler.start.y - end.y) > 5) { │ │ │ │ │ - var start = this.dragHandler.start; │ │ │ │ │ - var top = Math.min(start.y, end.y); │ │ │ │ │ - var bottom = Math.max(start.y, end.y); │ │ │ │ │ - var left = Math.min(start.x, end.x); │ │ │ │ │ - var right = Math.max(start.x, end.x); │ │ │ │ │ - result = new OpenLayers.Bounds(left, bottom, right, top); │ │ │ │ │ - } else { │ │ │ │ │ - result = this.dragHandler.start.clone(); // i.e. OL.Pixel │ │ │ │ │ - } │ │ │ │ │ - this.removeBox(); │ │ │ │ │ - │ │ │ │ │ - this.callback("done", [result]); │ │ │ │ │ + collectDragHandle: function() { │ │ │ │ │ + var geometry = this.feature.geometry; │ │ │ │ │ + var center = geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + var originGeometry = new OpenLayers.Geometry.Point( │ │ │ │ │ + center.lon, center.lat │ │ │ │ │ + ); │ │ │ │ │ + var origin = new OpenLayers.Feature.Vector(originGeometry); │ │ │ │ │ + originGeometry.move = function(x, y) { │ │ │ │ │ + OpenLayers.Geometry.Point.prototype.move.call(this, x, y); │ │ │ │ │ + geometry.move(x, y); │ │ │ │ │ + }; │ │ │ │ │ + origin._sketch = true; │ │ │ │ │ + this.dragHandle = origin; │ │ │ │ │ + this.dragHandle.renderIntent = this.vertexRenderIntent; │ │ │ │ │ + this.layer.addFeatures([this.dragHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removeBox │ │ │ │ │ - * Remove the zoombox from the screen and nullify our reference to it. │ │ │ │ │ + * Method: collectRadiusHandle │ │ │ │ │ + * Collect the radius handle for the selected geometry. │ │ │ │ │ */ │ │ │ │ │ - removeBox: function() { │ │ │ │ │ - this.map.viewPortDiv.removeChild(this.zoomBox); │ │ │ │ │ - this.zoomBox = null; │ │ │ │ │ - this.boxOffsets = null; │ │ │ │ │ - OpenLayers.Element.removeClass( │ │ │ │ │ - this.map.viewPortDiv, "olDrawBox" │ │ │ │ │ + collectRadiusHandle: function() { │ │ │ │ │ + var geometry = this.feature.geometry; │ │ │ │ │ + var bounds = geometry.getBounds(); │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var originGeometry = new OpenLayers.Geometry.Point( │ │ │ │ │ + center.lon, center.lat │ │ │ │ │ + ); │ │ │ │ │ + var radiusGeometry = new OpenLayers.Geometry.Point( │ │ │ │ │ + bounds.right, bounds.bottom │ │ │ │ │ ); │ │ │ │ │ + var radius = new OpenLayers.Feature.Vector(radiusGeometry); │ │ │ │ │ + var resize = (this.mode & OpenLayers.Control.ModifyFeature.RESIZE); │ │ │ │ │ + var reshape = (this.mode & OpenLayers.Control.ModifyFeature.RESHAPE); │ │ │ │ │ + var rotate = (this.mode & OpenLayers.Control.ModifyFeature.ROTATE); │ │ │ │ │ │ │ │ │ │ + radiusGeometry.move = function(x, y) { │ │ │ │ │ + OpenLayers.Geometry.Point.prototype.move.call(this, x, y); │ │ │ │ │ + var dx1 = this.x - originGeometry.x; │ │ │ │ │ + var dy1 = this.y - originGeometry.y; │ │ │ │ │ + var dx0 = dx1 - x; │ │ │ │ │ + var dy0 = dy1 - y; │ │ │ │ │ + if (rotate) { │ │ │ │ │ + var a0 = Math.atan2(dy0, dx0); │ │ │ │ │ + var a1 = Math.atan2(dy1, dx1); │ │ │ │ │ + var angle = a1 - a0; │ │ │ │ │ + angle *= 180 / Math.PI; │ │ │ │ │ + geometry.rotate(angle, originGeometry); │ │ │ │ │ + } │ │ │ │ │ + if (resize) { │ │ │ │ │ + var scale, ratio; │ │ │ │ │ + // 'resize' together with 'reshape' implies that the aspect │ │ │ │ │ + // ratio of the geometry will not be preserved whilst resizing │ │ │ │ │ + if (reshape) { │ │ │ │ │ + scale = dy1 / dy0; │ │ │ │ │ + ratio = (dx1 / dx0) / scale; │ │ │ │ │ + } else { │ │ │ │ │ + var l0 = Math.sqrt((dx0 * dx0) + (dy0 * dy0)); │ │ │ │ │ + var l1 = Math.sqrt((dx1 * dx1) + (dy1 * dy1)); │ │ │ │ │ + scale = l1 / l0; │ │ │ │ │ + } │ │ │ │ │ + geometry.resize(scale, originGeometry, ratio); │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + radius._sketch = true; │ │ │ │ │ + this.radiusHandle = radius; │ │ │ │ │ + this.radiusHandle.renderIntent = this.vertexRenderIntent; │ │ │ │ │ + this.layer.addFeatures([this.radiusHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: activate │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the control and all handlers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} The control's map. │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.dragHandler.activate(); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.handlers.drag.setMap(map); │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ + * Method: handleMapEvents │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - if (this.dragHandler.deactivate()) { │ │ │ │ │ - if (this.zoomBox) { │ │ │ │ │ - this.removeBox(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ + handleMapEvents: function(evt) { │ │ │ │ │ + if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ + this.moveLayerToTop(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getBoxOffsets │ │ │ │ │ - * Determines border offsets for a box, according to the box model. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} an object with the following offsets: │ │ │ │ │ - * - left │ │ │ │ │ - * - right │ │ │ │ │ - * - top │ │ │ │ │ - * - bottom │ │ │ │ │ - * - width │ │ │ │ │ - * - height │ │ │ │ │ + * Method: moveLayerToTop │ │ │ │ │ + * Moves the layer for this handler to the top, so mouse events can reach │ │ │ │ │ + * it. │ │ │ │ │ */ │ │ │ │ │ - getBoxOffsets: function() { │ │ │ │ │ - if (!this.boxOffsets) { │ │ │ │ │ - // Determine the box model. If the testDiv's clientWidth is 3, then │ │ │ │ │ - // the borders are outside and we are dealing with the w3c box │ │ │ │ │ - // model. Otherwise, the browser uses the traditional box model and │ │ │ │ │ - // the borders are inside the box bounds, leaving us with a │ │ │ │ │ - // clientWidth of 1. │ │ │ │ │ - var testDiv = document.createElement("div"); │ │ │ │ │ - //testDiv.style.visibility = "hidden"; │ │ │ │ │ - testDiv.style.position = "absolute"; │ │ │ │ │ - testDiv.style.border = "1px solid black"; │ │ │ │ │ - testDiv.style.width = "3px"; │ │ │ │ │ - document.body.appendChild(testDiv); │ │ │ │ │ - var w3cBoxModel = testDiv.clientWidth == 3; │ │ │ │ │ - document.body.removeChild(testDiv); │ │ │ │ │ + moveLayerToTop: function() { │ │ │ │ │ + var index = Math.max(this.map.Z_INDEX_BASE['Feature'] - 1, │ │ │ │ │ + this.layer.getZIndex()) + 1; │ │ │ │ │ + this.layer.setZIndex(index); │ │ │ │ │ │ │ │ │ │ - var left = parseInt(OpenLayers.Element.getStyle(this.zoomBox, │ │ │ │ │ - "border-left-width")); │ │ │ │ │ - var right = parseInt(OpenLayers.Element.getStyle( │ │ │ │ │ - this.zoomBox, "border-right-width")); │ │ │ │ │ - var top = parseInt(OpenLayers.Element.getStyle(this.zoomBox, │ │ │ │ │ - "border-top-width")); │ │ │ │ │ - var bottom = parseInt(OpenLayers.Element.getStyle( │ │ │ │ │ - this.zoomBox, "border-bottom-width")); │ │ │ │ │ - this.boxOffsets = { │ │ │ │ │ - left: left, │ │ │ │ │ - right: right, │ │ │ │ │ - top: top, │ │ │ │ │ - bottom: bottom, │ │ │ │ │ - width: w3cBoxModel === false ? left + right : 0, │ │ │ │ │ - height: w3cBoxModel === false ? top + bottom : 0 │ │ │ │ │ - }; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveLayerBack │ │ │ │ │ + * Moves the layer back to the position determined by the map's layers │ │ │ │ │ + * array. │ │ │ │ │ + */ │ │ │ │ │ + moveLayerBack: function() { │ │ │ │ │ + var index = this.layer.getZIndex() - 1; │ │ │ │ │ + if (index >= this.map.Z_INDEX_BASE['Feature']) { │ │ │ │ │ + this.layer.setZIndex(index); │ │ │ │ │ + } else { │ │ │ │ │ + this.map.setLayerZIndex(this.layer, │ │ │ │ │ + this.map.getLayerIndex(this.layer)); │ │ │ │ │ } │ │ │ │ │ - return this.boxOffsets; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Box" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ModifyFeature" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: RESHAPE │ │ │ │ │ + * {Integer} Constant used to make the control work in reshape mode │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.ModifyFeature.RESHAPE = 1; │ │ │ │ │ +/** │ │ │ │ │ + * Constant: RESIZE │ │ │ │ │ + * {Integer} Constant used to make the control work in resize mode │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.ModifyFeature.RESIZE = 2; │ │ │ │ │ +/** │ │ │ │ │ + * Constant: ROTATE │ │ │ │ │ + * {Integer} Constant used to make the control work in rotate mode │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.ModifyFeature.ROTATE = 4; │ │ │ │ │ +/** │ │ │ │ │ + * Constant: DRAG │ │ │ │ │ + * {Integer} Constant used to make the control work in drag mode │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.ModifyFeature.DRAG = 8; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Marker/Box.js │ │ │ │ │ + OpenLayers/Control/ZoomIn.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control/Button.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Marker.js │ │ │ │ │ + * Class: OpenLayers.Control.ZoomIn │ │ │ │ │ + * The ZoomIn control is a button to increase the zoom level of a map. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ +OpenLayers.Control.ZoomIn = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: trigger │ │ │ │ │ + */ │ │ │ │ │ + trigger: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.zoomIn(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomIn" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/Zoom.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Marker.Box │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.Zoom │ │ │ │ │ + * The Zoom control is a pair of +/- links for zooming in and out. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Marker> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Marker.Box = OpenLayers.Class(OpenLayers.Marker, { │ │ │ │ │ +OpenLayers.Control.Zoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: bounds │ │ │ │ │ - * {<OpenLayers.Bounds>} │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomInText │ │ │ │ │ + * {String} │ │ │ │ │ + * Text for zoom-in link. Default is "+". │ │ │ │ │ */ │ │ │ │ │ - bounds: null, │ │ │ │ │ + zoomInText: "+", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: div │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomInId │ │ │ │ │ + * {String} │ │ │ │ │ + * Instead of having the control create a zoom in link, you can provide │ │ │ │ │ + * the identifier for an anchor element already added to the document. │ │ │ │ │ + * By default, an element with id "olZoomInLink" will be searched for │ │ │ │ │ + * and used if it exists. │ │ │ │ │ */ │ │ │ │ │ - div: null, │ │ │ │ │ + zoomInId: "olZoomInLink", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Marker.Box │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * borderColor - {String} │ │ │ │ │ - * borderWidth - {int} │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomOutText │ │ │ │ │ + * {String} │ │ │ │ │ + * Text for zoom-out link. Default is "\u2212". │ │ │ │ │ */ │ │ │ │ │ - initialize: function(bounds, borderColor, borderWidth) { │ │ │ │ │ - this.bounds = bounds; │ │ │ │ │ - this.div = OpenLayers.Util.createDiv(); │ │ │ │ │ - this.div.style.overflow = 'hidden'; │ │ │ │ │ - this.events = new OpenLayers.Events(this, this.div); │ │ │ │ │ - this.setBorder(borderColor, borderWidth); │ │ │ │ │ - }, │ │ │ │ │ + zoomOutText: "\u2212", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * APIProperty: zoomOutId │ │ │ │ │ + * {String} │ │ │ │ │ + * Instead of having the control create a zoom out link, you can provide │ │ │ │ │ + * the identifier for an anchor element already added to the document. │ │ │ │ │ + * By default, an element with id "olZoomOutLink" will be searched for │ │ │ │ │ + * and used if it exists. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ + zoomOutId: "olZoomOutLink", │ │ │ │ │ │ │ │ │ │ - this.bounds = null; │ │ │ │ │ - this.div = null; │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A reference to the DOMElement containing the zoom links. │ │ │ │ │ + */ │ │ │ │ │ + draw: function() { │ │ │ │ │ + var div = OpenLayers.Control.prototype.draw.apply(this), │ │ │ │ │ + links = this.getOrCreateLinks(div), │ │ │ │ │ + zoomIn = links.zoomIn, │ │ │ │ │ + zoomOut = links.zoomOut, │ │ │ │ │ + eventsInstance = this.map.events; │ │ │ │ │ │ │ │ │ │ - OpenLayers.Marker.prototype.destroy.apply(this, arguments); │ │ │ │ │ + if (zoomOut.parentNode !== div) { │ │ │ │ │ + eventsInstance = this.events; │ │ │ │ │ + eventsInstance.attachToElement(zoomOut.parentNode); │ │ │ │ │ + } │ │ │ │ │ + eventsInstance.register("buttonclick", this, this.onZoomClick); │ │ │ │ │ + │ │ │ │ │ + this.zoomInLink = zoomIn; │ │ │ │ │ + this.zoomOutLink = zoomOut; │ │ │ │ │ + return div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setBorder │ │ │ │ │ - * Allow the user to change the box's color and border width │ │ │ │ │ + /** │ │ │ │ │ + * Method: getOrCreateLinks │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * color - {String} Default is "red" │ │ │ │ │ - * width - {int} Default is 2 │ │ │ │ │ + * el - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Return: │ │ │ │ │ + * {Object} Object with zoomIn and zoomOut properties referencing links. │ │ │ │ │ */ │ │ │ │ │ - setBorder: function(color, width) { │ │ │ │ │ - if (!color) { │ │ │ │ │ - color = "red"; │ │ │ │ │ + getOrCreateLinks: function(el) { │ │ │ │ │ + var zoomIn = document.getElementById(this.zoomInId), │ │ │ │ │ + zoomOut = document.getElementById(this.zoomOutId); │ │ │ │ │ + if (!zoomIn) { │ │ │ │ │ + zoomIn = document.createElement("a"); │ │ │ │ │ + zoomIn.href = "#zoomIn"; │ │ │ │ │ + zoomIn.appendChild(document.createTextNode(this.zoomInText)); │ │ │ │ │ + zoomIn.className = "olControlZoomIn"; │ │ │ │ │ + el.appendChild(zoomIn); │ │ │ │ │ } │ │ │ │ │ - if (!width) { │ │ │ │ │ - width = 2; │ │ │ │ │ + OpenLayers.Element.addClass(zoomIn, "olButton"); │ │ │ │ │ + if (!zoomOut) { │ │ │ │ │ + zoomOut = document.createElement("a"); │ │ │ │ │ + zoomOut.href = "#zoomOut"; │ │ │ │ │ + zoomOut.appendChild(document.createTextNode(this.zoomOutText)); │ │ │ │ │ + zoomOut.className = "olControlZoomOut"; │ │ │ │ │ + el.appendChild(zoomOut); │ │ │ │ │ } │ │ │ │ │ - this.div.style.border = width + "px solid " + color; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - * sz - {<OpenLayers.Size>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A new DOM Image with this marker's icon set at the │ │ │ │ │ - * location passed-in │ │ │ │ │ - */ │ │ │ │ │ - draw: function(px, sz) { │ │ │ │ │ - OpenLayers.Util.modifyDOMElement(this.div, null, px, sz); │ │ │ │ │ - return this.div; │ │ │ │ │ + OpenLayers.Element.addClass(zoomOut, "olButton"); │ │ │ │ │ + return { │ │ │ │ │ + zoomIn: zoomIn, │ │ │ │ │ + zoomOut: zoomOut │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onScreen │ │ │ │ │ - * │ │ │ │ │ - * Rreturn: │ │ │ │ │ - * {Boolean} Whether or not the marker is currently visible on screen. │ │ │ │ │ + * Method: onZoomClick │ │ │ │ │ + * Called when zoomin/out link is clicked. │ │ │ │ │ */ │ │ │ │ │ - onScreen: function() { │ │ │ │ │ - var onScreen = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - var screenBounds = this.map.getExtent(); │ │ │ │ │ - onScreen = screenBounds.containsBounds(this.bounds, true, true); │ │ │ │ │ + onZoomClick: function(evt) { │ │ │ │ │ + var button = evt.buttonElement; │ │ │ │ │ + if (button === this.zoomInLink) { │ │ │ │ │ + this.map.zoomIn(); │ │ │ │ │ + } else if (button === this.zoomOutLink) { │ │ │ │ │ + this.map.zoomOut(); │ │ │ │ │ } │ │ │ │ │ - return onScreen; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: display │ │ │ │ │ - * Hide or show the icon │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * display - {Boolean} │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Clean up. │ │ │ │ │ */ │ │ │ │ │ - display: function(display) { │ │ │ │ │ - this.div.style.display = (display) ? "" : "none"; │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onZoomClick); │ │ │ │ │ + } │ │ │ │ │ + delete this.zoomInLink; │ │ │ │ │ + delete this.zoomOutLink; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Marker.Box" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Zoom" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy/Cluster.js │ │ │ │ │ + OpenLayers/Control/DrawFeature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Strategy.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Strategy.Cluster │ │ │ │ │ - * Strategy for vector feature clustering. │ │ │ │ │ + * Class: OpenLayers.Control.DrawFeature │ │ │ │ │ + * The DrawFeature control draws point, line or polygon features on a vector │ │ │ │ │ + * layer when active. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Strategy> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Strategy.Cluster = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ +OpenLayers.Control.DrawFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: distance │ │ │ │ │ - * {Integer} Pixel distance between features that should be considered a │ │ │ │ │ - * single cluster. Default is 20 pixels. │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} │ │ │ │ │ */ │ │ │ │ │ - distance: 20, │ │ │ │ │ + layer: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: threshold │ │ │ │ │ - * {Integer} Optional threshold below which original features will be │ │ │ │ │ - * added to the layer instead of clusters. For example, a threshold │ │ │ │ │ - * of 3 would mean that any time there are 2 or fewer features in │ │ │ │ │ - * a cluster, those features will be added directly to the layer instead │ │ │ │ │ - * of a cluster representing those features. Default is null (which is │ │ │ │ │ - * equivalent to 1 - meaning that clusters may contain just one feature). │ │ │ │ │ + * Property: callbacks │ │ │ │ │ + * {Object} The functions that are sent to the handler for callback │ │ │ │ │ */ │ │ │ │ │ - threshold: null, │ │ │ │ │ + callbacks: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} Cached features. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ + * featureadded - Triggered when a feature is added │ │ │ │ │ */ │ │ │ │ │ - features: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: clusters │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} Calculated clusters. │ │ │ │ │ + * APIProperty: multi │ │ │ │ │ + * {Boolean} Cast features to multi-part geometries before passing to the │ │ │ │ │ + * layer. Default is false. │ │ │ │ │ */ │ │ │ │ │ - clusters: null, │ │ │ │ │ + multi: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: clustering │ │ │ │ │ - * {Boolean} The strategy is currently clustering features. │ │ │ │ │ + * APIProperty: featureAdded │ │ │ │ │ + * {Function} Called after each feature is added │ │ │ │ │ */ │ │ │ │ │ - clustering: false, │ │ │ │ │ + featureAdded: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: resolution │ │ │ │ │ - * {Float} The resolution (map units per pixel) of the current cluster set. │ │ │ │ │ + * APIProperty: handlerOptions │ │ │ │ │ + * {Object} Used to set non-default properties on the control's handler │ │ │ │ │ */ │ │ │ │ │ - resolution: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy.Cluster │ │ │ │ │ - * Create a new clustering strategy. │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Control.DrawFeature │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ + * handler - {<OpenLayers.Handler>} │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "beforefeaturesadded": this.cacheFeatures, │ │ │ │ │ - "featuresremoved": this.clearCache, │ │ │ │ │ - "moveend": this.cluster, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ - * tear-down. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.clearCache(); │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - "beforefeaturesadded": this.cacheFeatures, │ │ │ │ │ - "featuresremoved": this.clearCache, │ │ │ │ │ - "moveend": this.cluster, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: cacheFeatures │ │ │ │ │ - * Cache features before they are added to the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * event - {Object} The event that this was listening for. This will come │ │ │ │ │ - * with a batch of features to be clustered. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} False to stop features from being added to the layer. │ │ │ │ │ - */ │ │ │ │ │ - cacheFeatures: function(event) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - if (!this.clustering) { │ │ │ │ │ - this.clearCache(); │ │ │ │ │ - this.features = event.features; │ │ │ │ │ - this.cluster(); │ │ │ │ │ - propagate = false; │ │ │ │ │ - } │ │ │ │ │ - return propagate; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clearCache │ │ │ │ │ - * Clear out the cached features. │ │ │ │ │ - */ │ │ │ │ │ - clearCache: function() { │ │ │ │ │ - if (!this.clustering) { │ │ │ │ │ - this.features = null; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: cluster │ │ │ │ │ - * Cluster features based on some threshold distance. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * event - {Object} The event received when cluster is called as a │ │ │ │ │ - * result of a moveend event. │ │ │ │ │ - */ │ │ │ │ │ - cluster: function(event) { │ │ │ │ │ - if ((!event || event.zoomChanged) && this.features) { │ │ │ │ │ - var resolution = this.layer.map.getResolution(); │ │ │ │ │ - if (resolution != this.resolution || !this.clustersExist()) { │ │ │ │ │ - this.resolution = resolution; │ │ │ │ │ - var clusters = []; │ │ │ │ │ - var feature, clustered, cluster; │ │ │ │ │ - for (var i = 0; i < this.features.length; ++i) { │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - clustered = false; │ │ │ │ │ - for (var j = clusters.length - 1; j >= 0; --j) { │ │ │ │ │ - cluster = clusters[j]; │ │ │ │ │ - if (this.shouldCluster(cluster, feature)) { │ │ │ │ │ - this.addToCluster(cluster, feature); │ │ │ │ │ - clustered = true; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!clustered) { │ │ │ │ │ - clusters.push(this.createCluster(this.features[i])); │ │ │ │ │ + initialize: function(layer, handler, options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.callbacks = OpenLayers.Util.extend({ │ │ │ │ │ + done: this.drawFeature, │ │ │ │ │ + modify: function(vertex, feature) { │ │ │ │ │ + this.layer.events.triggerEvent( │ │ │ │ │ + "sketchmodified", { │ │ │ │ │ + vertex: vertex, │ │ │ │ │ + feature: feature │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.clustering = true; │ │ │ │ │ - this.layer.removeAllFeatures(); │ │ │ │ │ - this.clustering = false; │ │ │ │ │ - if (clusters.length > 0) { │ │ │ │ │ - if (this.threshold > 1) { │ │ │ │ │ - var clone = clusters.slice(); │ │ │ │ │ - clusters = []; │ │ │ │ │ - var candidate; │ │ │ │ │ - for (var i = 0, len = clone.length; i < len; ++i) { │ │ │ │ │ - candidate = clone[i]; │ │ │ │ │ - if (candidate.attributes.count < this.threshold) { │ │ │ │ │ - Array.prototype.push.apply(clusters, candidate.cluster); │ │ │ │ │ - } else { │ │ │ │ │ - clusters.push(candidate); │ │ │ │ │ - } │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + create: function(vertex, feature) { │ │ │ │ │ + this.layer.events.triggerEvent( │ │ │ │ │ + "sketchstarted", { │ │ │ │ │ + vertex: vertex, │ │ │ │ │ + feature: feature │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - this.clustering = true; │ │ │ │ │ - // A legitimate feature addition could occur during this │ │ │ │ │ - // addFeatures call. For clustering to behave well, features │ │ │ │ │ - // should be removed from a layer before requesting a new batch. │ │ │ │ │ - this.layer.addFeatures(clusters); │ │ │ │ │ - this.clustering = false; │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - this.clusters = clusters; │ │ │ │ │ + }, │ │ │ │ │ + this.callbacks │ │ │ │ │ + ); │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + this.handlerOptions = this.handlerOptions || {}; │ │ │ │ │ + this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.handlerOptions.layerOptions, { │ │ │ │ │ + renderers: layer.renderers, │ │ │ │ │ + rendererOptions: layer.rendererOptions │ │ │ │ │ } │ │ │ │ │ + ); │ │ │ │ │ + if (!("multi" in this.handlerOptions)) { │ │ │ │ │ + this.handlerOptions.multi = this.multi; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clustersExist │ │ │ │ │ - * Determine whether calculated clusters are already on the layer. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The calculated clusters are already on the layer. │ │ │ │ │ - */ │ │ │ │ │ - clustersExist: function() { │ │ │ │ │ - var exist = false; │ │ │ │ │ - if (this.clusters && this.clusters.length > 0 && │ │ │ │ │ - this.clusters.length == this.layer.features.length) { │ │ │ │ │ - exist = true; │ │ │ │ │ - for (var i = 0; i < this.clusters.length; ++i) { │ │ │ │ │ - if (this.clusters[i] != this.layer.features[i]) { │ │ │ │ │ - exist = false; │ │ │ │ │ - break; │ │ │ │ │ + var sketchStyle = this.layer.styleMap && this.layer.styleMap.styles.temporary; │ │ │ │ │ + if (sketchStyle) { │ │ │ │ │ + this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.handlerOptions.layerOptions, { │ │ │ │ │ + styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ + "default": sketchStyle │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - return exist; │ │ │ │ │ + this.handler = new handler(this, this.callbacks, this.handlerOptions); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: shouldCluster │ │ │ │ │ - * Determine whether to include a feature in a given cluster. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * cluster - {<OpenLayers.Feature.Vector>} A cluster. │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} A feature. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The feature should be included in the cluster. │ │ │ │ │ + * Method: drawFeature │ │ │ │ │ */ │ │ │ │ │ - shouldCluster: function(cluster, feature) { │ │ │ │ │ - var cc = cluster.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - var fc = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - var distance = ( │ │ │ │ │ - Math.sqrt( │ │ │ │ │ - Math.pow((cc.lon - fc.lon), 2) + Math.pow((cc.lat - fc.lat), 2) │ │ │ │ │ - ) / this.resolution │ │ │ │ │ + drawFeature: function(geometry) { │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + var proceed = this.layer.events.triggerEvent( │ │ │ │ │ + "sketchcomplete", { │ │ │ │ │ + feature: feature │ │ │ │ │ + } │ │ │ │ │ ); │ │ │ │ │ - return (distance <= this.distance); │ │ │ │ │ + if (proceed !== false) { │ │ │ │ │ + feature.state = OpenLayers.State.INSERT; │ │ │ │ │ + this.layer.addFeatures([feature]); │ │ │ │ │ + this.featureAdded(feature); │ │ │ │ │ + this.events.triggerEvent("featureadded", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: addToCluster │ │ │ │ │ - * Add a feature to a cluster. │ │ │ │ │ + * APIMethod: insertXY │ │ │ │ │ + * Insert a point in the current sketch given x & y coordinates. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * cluster - {<OpenLayers.Feature.Vector>} A cluster. │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} A feature. │ │ │ │ │ + * x - {Number} The x-coordinate of the point. │ │ │ │ │ + * y - {Number} The y-coordinate of the point. │ │ │ │ │ */ │ │ │ │ │ - addToCluster: function(cluster, feature) { │ │ │ │ │ - cluster.cluster.push(feature); │ │ │ │ │ - cluster.attributes.count += 1; │ │ │ │ │ + insertXY: function(x, y) { │ │ │ │ │ + if (this.handler && this.handler.line) { │ │ │ │ │ + this.handler.insertXY(x, y); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createCluster │ │ │ │ │ - * Given a feature, create a cluster. │ │ │ │ │ + * APIMethod: insertDeltaXY │ │ │ │ │ + * Insert a point given offsets from the previously inserted point. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A cluster. │ │ │ │ │ + * dx - {Number} The x-coordinate offset of the point. │ │ │ │ │ + * dy - {Number} The y-coordinate offset of the point. │ │ │ │ │ */ │ │ │ │ │ - createCluster: function(feature) { │ │ │ │ │ - var center = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - var cluster = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.Point(center.lon, center.lat), { │ │ │ │ │ - count: 1 │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - cluster.cluster = [feature]; │ │ │ │ │ - return cluster; │ │ │ │ │ + insertDeltaXY: function(dx, dy) { │ │ │ │ │ + if (this.handler && this.handler.line) { │ │ │ │ │ + this.handler.insertDeltaXY(dx, dy); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Cluster" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy/Paging.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Strategy.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Strategy.Paging │ │ │ │ │ - * Strategy for vector feature paging │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Strategy> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Strategy.Paging = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} Cached features. │ │ │ │ │ - */ │ │ │ │ │ - features: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: length │ │ │ │ │ - * {Integer} Number of features per page. Default is 10. │ │ │ │ │ - */ │ │ │ │ │ - length: 10, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: num │ │ │ │ │ - * {Integer} The currently displayed page number. │ │ │ │ │ - */ │ │ │ │ │ - num: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: paging │ │ │ │ │ - * {Boolean} The strategy is currently changing pages. │ │ │ │ │ - */ │ │ │ │ │ - paging: false, │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy.Paging │ │ │ │ │ - * Create a new paging strategy. │ │ │ │ │ + * APIMethod: insertDirectionLength │ │ │ │ │ + * Insert a point in the current sketch given a direction and a length. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "beforefeaturesadded": this.cacheFeatures, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ - * tear-down. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ + * direction - {Number} Degrees clockwise from the positive x-axis. │ │ │ │ │ + * length - {Number} Distance from the previously drawn point. │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.clearCache(); │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - "beforefeaturesadded": this.cacheFeatures, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + insertDirectionLength: function(direction, length) { │ │ │ │ │ + if (this.handler && this.handler.line) { │ │ │ │ │ + this.handler.insertDirectionLength(direction, length); │ │ │ │ │ } │ │ │ │ │ - return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: cacheFeatures │ │ │ │ │ - * Cache features before they are added to the layer. │ │ │ │ │ + * APIMethod: insertDeflectionLength │ │ │ │ │ + * Insert a point in the current sketch given a deflection and a length. │ │ │ │ │ + * The deflection should be degrees clockwise from the previously │ │ │ │ │ + * digitized segment. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * event - {Object} The event that this was listening for. This will come │ │ │ │ │ - * with a batch of features to be paged. │ │ │ │ │ - */ │ │ │ │ │ - cacheFeatures: function(event) { │ │ │ │ │ - if (!this.paging) { │ │ │ │ │ - this.clearCache(); │ │ │ │ │ - this.features = event.features; │ │ │ │ │ - this.pageNext(event); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clearCache │ │ │ │ │ - * Clear out the cached features. This destroys features, assuming │ │ │ │ │ - * nothing else has a reference. │ │ │ │ │ + * deflection - {Number} Degrees clockwise from the previous segment. │ │ │ │ │ + * length - {Number} Distance from the previously drawn point. │ │ │ │ │ */ │ │ │ │ │ - clearCache: function() { │ │ │ │ │ - if (this.features) { │ │ │ │ │ - for (var i = 0; i < this.features.length; ++i) { │ │ │ │ │ - this.features[i].destroy(); │ │ │ │ │ - } │ │ │ │ │ + insertDeflectionLength: function(deflection, length) { │ │ │ │ │ + if (this.handler && this.handler.line) { │ │ │ │ │ + this.handler.insertDeflectionLength(deflection, length); │ │ │ │ │ } │ │ │ │ │ - this.features = null; │ │ │ │ │ - this.num = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: pageCount │ │ │ │ │ - * Get the total count of pages given the current cache of features. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} The page count. │ │ │ │ │ - */ │ │ │ │ │ - pageCount: function() { │ │ │ │ │ - var numFeatures = this.features ? this.features.length : 0; │ │ │ │ │ - return Math.ceil(numFeatures / this.length); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: pageNum │ │ │ │ │ - * Get the zero based page number. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} The current page number being displayed. │ │ │ │ │ - */ │ │ │ │ │ - pageNum: function() { │ │ │ │ │ - return this.num; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: pageLength │ │ │ │ │ - * Gets or sets page length. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newLength - {Integer} Optional length to be set. │ │ │ │ │ + * APIMethod: undo │ │ │ │ │ + * Remove the most recently added point in the current sketch geometry. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} The length of a page (number of features per page). │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} An edit was undone. │ │ │ │ │ */ │ │ │ │ │ - pageLength: function(newLength) { │ │ │ │ │ - if (newLength && newLength > 0) { │ │ │ │ │ - this.length = newLength; │ │ │ │ │ - } │ │ │ │ │ - return this.length; │ │ │ │ │ + undo: function() { │ │ │ │ │ + return this.handler.undo && this.handler.undo(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: pageNext │ │ │ │ │ - * Display the next page of features. │ │ │ │ │ + * APIMethod: redo │ │ │ │ │ + * Reinsert the most recently removed point resulting from an <undo> call. │ │ │ │ │ + * The undo stack is deleted whenever a point is added by other means. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} A new page was displayed. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} An edit was redone. │ │ │ │ │ */ │ │ │ │ │ - pageNext: function(event) { │ │ │ │ │ - var changed = false; │ │ │ │ │ - if (this.features) { │ │ │ │ │ - if (this.num === null) { │ │ │ │ │ - this.num = -1; │ │ │ │ │ - } │ │ │ │ │ - var start = (this.num + 1) * this.length; │ │ │ │ │ - changed = this.page(start, event); │ │ │ │ │ - } │ │ │ │ │ - return changed; │ │ │ │ │ + redo: function() { │ │ │ │ │ + return this.handler.redo && this.handler.redo(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: pagePrevious │ │ │ │ │ - * Display the previous page of features. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} A new page was displayed. │ │ │ │ │ + * APIMethod: finishSketch │ │ │ │ │ + * Finishes the sketch without including the currently drawn point. │ │ │ │ │ + * This method can be called to terminate drawing programmatically │ │ │ │ │ + * instead of waiting for the user to end the sketch. │ │ │ │ │ */ │ │ │ │ │ - pagePrevious: function() { │ │ │ │ │ - var changed = false; │ │ │ │ │ - if (this.features) { │ │ │ │ │ - if (this.num === null) { │ │ │ │ │ - this.num = this.pageCount(); │ │ │ │ │ - } │ │ │ │ │ - var start = (this.num - 1) * this.length; │ │ │ │ │ - changed = this.page(start); │ │ │ │ │ - } │ │ │ │ │ - return changed; │ │ │ │ │ + finishSketch: function() { │ │ │ │ │ + this.handler.finishGeometry(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: page │ │ │ │ │ - * Display the page starting at the given index from the cache. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} A new page was displayed. │ │ │ │ │ + * APIMethod: cancel │ │ │ │ │ + * Cancel the current sketch. This removes the current sketch and keeps │ │ │ │ │ + * the drawing control active. │ │ │ │ │ */ │ │ │ │ │ - page: function(start, event) { │ │ │ │ │ - var changed = false; │ │ │ │ │ - if (this.features) { │ │ │ │ │ - if (start >= 0 && start < this.features.length) { │ │ │ │ │ - var num = Math.floor(start / this.length); │ │ │ │ │ - if (num != this.num) { │ │ │ │ │ - this.paging = true; │ │ │ │ │ - var features = this.features.slice(start, start + this.length); │ │ │ │ │ - this.layer.removeFeatures(this.layer.features); │ │ │ │ │ - this.num = num; │ │ │ │ │ - // modify the event if any │ │ │ │ │ - if (event && event.features) { │ │ │ │ │ - // this.was called by an event listener │ │ │ │ │ - event.features = features; │ │ │ │ │ - } else { │ │ │ │ │ - // this was called directly on the strategy │ │ │ │ │ - this.layer.addFeatures(features); │ │ │ │ │ - } │ │ │ │ │ - this.paging = false; │ │ │ │ │ - changed = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return changed; │ │ │ │ │ + cancel: function() { │ │ │ │ │ + this.handler.cancel(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Paging" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.DrawFeature" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy/Filter.js │ │ │ │ │ + OpenLayers/Control/Permalink.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Strategy.js │ │ │ │ │ - * @requires OpenLayers/Filter.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Control/ArgParser.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Strategy.Filter │ │ │ │ │ - * Strategy for limiting features that get added to a layer by │ │ │ │ │ - * evaluating a filter. The strategy maintains a cache of │ │ │ │ │ - * all features until removeFeatures is called on the layer. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Control.Permalink │ │ │ │ │ + * The Permalink control is hyperlink that will return the user to the │ │ │ │ │ + * current map view. By default it is drawn in the lower right corner of the │ │ │ │ │ + * map. The href is updated as the map is zoomed, panned and whilst layers │ │ │ │ │ + * are switched. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Strategy> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Strategy.Filter = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ +OpenLayers.Control.Permalink = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: filter │ │ │ │ │ - * {<OpenLayers.Filter>} Filter for limiting features sent to the layer. │ │ │ │ │ - * Use the <setFilter> method to update this filter after construction. │ │ │ │ │ + * APIProperty: argParserClass │ │ │ │ │ + * {Class} The ArgParser control class (not instance) to use with this │ │ │ │ │ + * control. │ │ │ │ │ */ │ │ │ │ │ - filter: null, │ │ │ │ │ + argParserClass: OpenLayers.Control.ArgParser, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: cache │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} List of currently cached │ │ │ │ │ - * features. │ │ │ │ │ + /** │ │ │ │ │ + * Property: element │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - cache: null, │ │ │ │ │ + element: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: caching │ │ │ │ │ - * {Boolean} The filter is currently caching features. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: anchor │ │ │ │ │ + * {Boolean} This option changes 3 things: │ │ │ │ │ + * the character '#' is used in place of the character '?', │ │ │ │ │ + * the window.href is updated if no element is provided. │ │ │ │ │ + * When this option is set to true it's not recommend to provide │ │ │ │ │ + * a base without provide an element. │ │ │ │ │ */ │ │ │ │ │ - caching: false, │ │ │ │ │ + anchor: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy.Filter │ │ │ │ │ - * Create a new filter strategy. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: base │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ + base: '', │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ - * By default, this strategy automatically activates itself when a layer │ │ │ │ │ - * is added to a map. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} True if the strategy was successfully activated or false if │ │ │ │ │ - * the strategy was already active. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: displayProjection │ │ │ │ │ + * {<OpenLayers.Projection>} Requires proj4js support. Projection used │ │ │ │ │ + * when creating the coordinates in the link. This will reproject the │ │ │ │ │ + * map coordinates into display coordinates. If you are using this │ │ │ │ │ + * functionality, the permalink which is last added to the map will │ │ │ │ │ + * determine the coordinate type which is read from the URL, which │ │ │ │ │ + * means you should not add permalinks with different │ │ │ │ │ + * displayProjections to the same map. │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.cache = []; │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "beforefeaturesadded": this.handleAdd, │ │ │ │ │ - "beforefeaturesremoved": this.handleRemove, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ + displayProjection: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the strategy. Clear the feature cache. │ │ │ │ │ + * Constructor: OpenLayers.Control.Permalink │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} True if the strategy was successfully deactivated or false if │ │ │ │ │ - * the strategy was already inactive. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - this.cache = null; │ │ │ │ │ - if (this.layer && this.layer.events) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - "beforefeaturesadded": this.handleAdd, │ │ │ │ │ - "beforefeaturesremoved": this.handleRemove, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Strategy.prototype.deactivate.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleAdd │ │ │ │ │ + * Parameters: │ │ │ │ │ + * element - {DOMElement} │ │ │ │ │ + * base - {String} │ │ │ │ │ + * options - {Object} options to the control. │ │ │ │ │ + * │ │ │ │ │ + * Or for anchor: │ │ │ │ │ + * options - {Object} options to the control. │ │ │ │ │ */ │ │ │ │ │ - handleAdd: function(event) { │ │ │ │ │ - if (!this.caching && this.filter) { │ │ │ │ │ - var features = event.features; │ │ │ │ │ - event.features = []; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0, ii = features.length; i < ii; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (this.filter.evaluate(feature)) { │ │ │ │ │ - event.features.push(feature); │ │ │ │ │ - } else { │ │ │ │ │ - this.cache.push(feature); │ │ │ │ │ - } │ │ │ │ │ + initialize: function(element, base, options) { │ │ │ │ │ + if (element !== null && typeof element == 'object' && !OpenLayers.Util.isElement(element)) { │ │ │ │ │ + options = element; │ │ │ │ │ + this.base = document.location.href; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (this.element != null) { │ │ │ │ │ + this.element = OpenLayers.Util.getElement(this.element); │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.element = OpenLayers.Util.getElement(element); │ │ │ │ │ + this.base = base || document.location.href; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleRemove │ │ │ │ │ - */ │ │ │ │ │ - handleRemove: function(event) { │ │ │ │ │ - if (!this.caching) { │ │ │ │ │ - this.cache = []; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setFilter │ │ │ │ │ - * Update the filter for this strategy. This will re-evaluate │ │ │ │ │ - * any features on the layer and in the cache. Only features │ │ │ │ │ - * for which filter.evalute(feature) returns true will be │ │ │ │ │ - * added to the layer. Others will be cached by the strategy. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} A filter for evaluating features. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ */ │ │ │ │ │ - setFilter: function(filter) { │ │ │ │ │ - this.filter = filter; │ │ │ │ │ - var previousCache = this.cache; │ │ │ │ │ - this.cache = []; │ │ │ │ │ - // look through layer for features to remove from layer │ │ │ │ │ - this.handleAdd({ │ │ │ │ │ - features: this.layer.features │ │ │ │ │ - }); │ │ │ │ │ - // cache now contains features to remove from layer │ │ │ │ │ - if (this.cache.length > 0) { │ │ │ │ │ - this.caching = true; │ │ │ │ │ - this.layer.removeFeatures(this.cache.slice()); │ │ │ │ │ - this.caching = false; │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.element && this.element.parentNode == this.div) { │ │ │ │ │ + this.div.removeChild(this.element); │ │ │ │ │ + this.element = null; │ │ │ │ │ } │ │ │ │ │ - // now look through previous cache for features to add to layer │ │ │ │ │ - if (previousCache.length > 0) { │ │ │ │ │ - var event = { │ │ │ │ │ - features: previousCache │ │ │ │ │ - }; │ │ │ │ │ - this.handleAdd(event); │ │ │ │ │ - if (event.features.length > 0) { │ │ │ │ │ - // event has features to add to layer │ │ │ │ │ - this.caching = true; │ │ │ │ │ - this.layer.addFeatures(event.features); │ │ │ │ │ - this.caching = false; │ │ │ │ │ - } │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister('moveend', this, this.updateLink); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Filter" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy/Refresh.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Strategy.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Strategy.Refresh │ │ │ │ │ - * A strategy that refreshes the layer. By default the strategy waits for a │ │ │ │ │ - * call to <refresh> before refreshing. By configuring the strategy with │ │ │ │ │ - * the <interval> option, refreshing can take place automatically. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Strategy> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Strategy.Refresh = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: force │ │ │ │ │ - * {Boolean} Force a refresh on the layer. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - force: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: interval │ │ │ │ │ - * {Number} Auto-refresh. Default is 0. If > 0, layer will be refreshed │ │ │ │ │ - * every N milliseconds. │ │ │ │ │ - */ │ │ │ │ │ - interval: 0, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: timer │ │ │ │ │ - * {Number} The id of the timer. │ │ │ │ │ - */ │ │ │ │ │ - timer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy.Refresh │ │ │ │ │ - * Create a new Refresh strategy. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} True if the strategy was successfully activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - if (this.layer.visibility === true) { │ │ │ │ │ - this.start(); │ │ │ │ │ - } │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "visibilitychanged": this.reset, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ - * tear-down. │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the control. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} True if the strategy was successfully deactivated. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.stop(); │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - "visibilitychanged": this.reset, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: reset │ │ │ │ │ - * Start or cancel the refresh interval depending on the visibility of │ │ │ │ │ - * the layer. │ │ │ │ │ - */ │ │ │ │ │ - reset: function() { │ │ │ │ │ - if (this.layer.visibility === true) { │ │ │ │ │ - this.start(); │ │ │ │ │ - } else { │ │ │ │ │ - this.stop(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + //make sure we have an arg parser attached │ │ │ │ │ + for (var i = 0, len = this.map.controls.length; i < len; i++) { │ │ │ │ │ + var control = this.map.controls[i]; │ │ │ │ │ + if (control.CLASS_NAME == this.argParserClass.CLASS_NAME) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: start │ │ │ │ │ - * Start the refresh interval. │ │ │ │ │ - */ │ │ │ │ │ - start: function() { │ │ │ │ │ - if (this.interval && typeof this.interval === "number" && │ │ │ │ │ - this.interval > 0) { │ │ │ │ │ + // If a permalink is added to the map, and an ArgParser already │ │ │ │ │ + // exists, we override the displayProjection to be the one │ │ │ │ │ + // on the permalink. │ │ │ │ │ + if (control.displayProjection != this.displayProjection) { │ │ │ │ │ + this.displayProjection = control.displayProjection; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - this.timer = window.setInterval( │ │ │ │ │ - OpenLayers.Function.bind(this.refresh, this), │ │ │ │ │ - this.interval); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: refresh │ │ │ │ │ - * Tell the strategy to refresh which will refresh the layer. │ │ │ │ │ - */ │ │ │ │ │ - refresh: function() { │ │ │ │ │ - if (this.layer && this.layer.refresh && │ │ │ │ │ - typeof this.layer.refresh == "function") { │ │ │ │ │ - │ │ │ │ │ - this.layer.refresh({ │ │ │ │ │ - force: this.force │ │ │ │ │ - }); │ │ │ │ │ + if (i == this.map.controls.length) { │ │ │ │ │ + this.map.addControl(new this.argParserClass({ │ │ │ │ │ + 'displayProjection': this.displayProjection │ │ │ │ │ + })); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: stop │ │ │ │ │ - * Cancels the refresh interval. │ │ │ │ │ - */ │ │ │ │ │ - stop: function() { │ │ │ │ │ - if (this.timer !== null) { │ │ │ │ │ - window.clearInterval(this.timer); │ │ │ │ │ - this.timer = null; │ │ │ │ │ - } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Refresh" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy/Save.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Strategy.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Strategy.Save │ │ │ │ │ - * A strategy that commits newly created or modified features. By default │ │ │ │ │ - * the strategy waits for a call to <save> before persisting changes. By │ │ │ │ │ - * configuring the strategy with the <auto> option, changes can be saved │ │ │ │ │ - * automatically. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Strategy> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Strategy.Save = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} An events object that handles all │ │ │ │ │ - * events on the strategy object. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * strategy.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ + * Method: draw │ │ │ │ │ * │ │ │ │ │ - * Supported event types: │ │ │ │ │ - * start - Triggered before saving │ │ │ │ │ - * success - Triggered after a successful transaction │ │ │ │ │ - * fail - Triggered after a failed transaction │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for triggering this protocol │ │ │ │ │ - * events. │ │ │ │ │ - */ │ │ │ │ │ - events: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: auto │ │ │ │ │ - * {Boolean | Number} Auto-save. Default is false. If true, features will be │ │ │ │ │ - * saved immediately after being added to the layer and with each │ │ │ │ │ - * modification or deletion. If auto is a number, features will be │ │ │ │ │ - * saved on an interval provided by the value (in seconds). │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - auto: false, │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: timer │ │ │ │ │ - * {Number} The id of the timer. │ │ │ │ │ - */ │ │ │ │ │ - timer: null, │ │ │ │ │ + if (!this.element && !this.anchor) { │ │ │ │ │ + this.element = document.createElement("a"); │ │ │ │ │ + this.element.innerHTML = OpenLayers.i18n("Permalink"); │ │ │ │ │ + this.element.href = ""; │ │ │ │ │ + this.div.appendChild(this.element); │ │ │ │ │ + } │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + 'moveend': this.updateLink, │ │ │ │ │ + 'changelayer': this.updateLink, │ │ │ │ │ + 'changebaselayer': this.updateLink, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy.Save │ │ │ │ │ - * Create a new Save strategy. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Strategy.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.events = new OpenLayers.Events(this); │ │ │ │ │ - }, │ │ │ │ │ + // Make it so there is at least a link even though the map may not have │ │ │ │ │ + // moved yet. │ │ │ │ │ + this.updateLink(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - if (this.auto) { │ │ │ │ │ - if (typeof this.auto === "number") { │ │ │ │ │ - this.timer = window.setInterval( │ │ │ │ │ - OpenLayers.Function.bind(this.save, this), │ │ │ │ │ - this.auto * 1000 │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "featureadded": this.triggerSave, │ │ │ │ │ - "afterfeaturemodified": this.triggerSave, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ + return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ - * tear-down. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ + * Method: updateLink │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - if (this.auto) { │ │ │ │ │ - if (typeof this.auto === "number") { │ │ │ │ │ - window.clearInterval(this.timer); │ │ │ │ │ - } else { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - "featureadded": this.triggerSave, │ │ │ │ │ - "afterfeaturemodified": this.triggerSave, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + updateLink: function() { │ │ │ │ │ + var separator = this.anchor ? '#' : '?'; │ │ │ │ │ + var href = this.base; │ │ │ │ │ + var anchor = null; │ │ │ │ │ + if (href.indexOf("#") != -1 && this.anchor == false) { │ │ │ │ │ + anchor = href.substring(href.indexOf("#"), href.length); │ │ │ │ │ } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: triggerSave │ │ │ │ │ - * Registered as a listener. Calls save if a feature has insert, update, │ │ │ │ │ - * or delete state. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * event - {Object} The event this function is listening for. │ │ │ │ │ - */ │ │ │ │ │ - triggerSave: function(event) { │ │ │ │ │ - var feature = event.feature; │ │ │ │ │ - if (feature.state === OpenLayers.State.INSERT || │ │ │ │ │ - feature.state === OpenLayers.State.UPDATE || │ │ │ │ │ - feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ - this.save([event.feature]); │ │ │ │ │ + if (href.indexOf(separator) != -1) { │ │ │ │ │ + href = href.substring(0, href.indexOf(separator)); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: save │ │ │ │ │ - * Tell the layer protocol to commit unsaved features. If the layer │ │ │ │ │ - * projection differs from the map projection, features will be │ │ │ │ │ - * transformed into the layer projection before being committed. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array} Features to be saved. If null, then default is all │ │ │ │ │ - * features in the layer. Features are assumed to be in the map │ │ │ │ │ - * projection. │ │ │ │ │ - */ │ │ │ │ │ - save: function(features) { │ │ │ │ │ - if (!features) { │ │ │ │ │ - features = this.layer.features; │ │ │ │ │ + var splits = href.split("#"); │ │ │ │ │ + href = splits[0] + separator + OpenLayers.Util.getParameterString(this.createParams()); │ │ │ │ │ + if (anchor) { │ │ │ │ │ + href += anchor; │ │ │ │ │ } │ │ │ │ │ - this.events.triggerEvent("start", { │ │ │ │ │ - features: features │ │ │ │ │ - }); │ │ │ │ │ - var remote = this.layer.projection; │ │ │ │ │ - var local = this.layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var len = features.length; │ │ │ │ │ - var clones = new Array(len); │ │ │ │ │ - var orig, clone; │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - orig = features[i]; │ │ │ │ │ - clone = orig.clone(); │ │ │ │ │ - clone.fid = orig.fid; │ │ │ │ │ - clone.state = orig.state; │ │ │ │ │ - if (orig.url) { │ │ │ │ │ - clone.url = orig.url; │ │ │ │ │ - } │ │ │ │ │ - clone._original = orig; │ │ │ │ │ - clone.geometry.transform(local, remote); │ │ │ │ │ - clones[i] = clone; │ │ │ │ │ - } │ │ │ │ │ - features = clones; │ │ │ │ │ + if (this.anchor && !this.element) { │ │ │ │ │ + window.location.href = href; │ │ │ │ │ + } else { │ │ │ │ │ + this.element.href = href; │ │ │ │ │ } │ │ │ │ │ - this.layer.protocol.commit(features, { │ │ │ │ │ - callback: this.onCommit, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onCommit │ │ │ │ │ - * Called after protocol commit. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: createParams │ │ │ │ │ + * Creates the parameters that need to be encoded into the permalink url. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} A response object. │ │ │ │ │ + * center - {<OpenLayers.LonLat>} center to encode in the permalink. │ │ │ │ │ + * Defaults to the current map center. │ │ │ │ │ + * zoom - {Integer} zoom level to encode in the permalink. Defaults to the │ │ │ │ │ + * current map zoom level. │ │ │ │ │ + * layers - {Array(<OpenLayers.Layer>)} layers to encode in the permalink. │ │ │ │ │ + * Defaults to the current map layers. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Hash of parameters that will be url-encoded into the │ │ │ │ │ + * permalink. │ │ │ │ │ */ │ │ │ │ │ - onCommit: function(response) { │ │ │ │ │ - var evt = { │ │ │ │ │ - "response": response │ │ │ │ │ - }; │ │ │ │ │ - if (response.success()) { │ │ │ │ │ - var features = response.reqFeatures; │ │ │ │ │ - // deal with inserts, updates, and deletes │ │ │ │ │ - var state, feature; │ │ │ │ │ - var destroys = []; │ │ │ │ │ - var insertIds = response.insertIds || []; │ │ │ │ │ - var j = 0; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - // if projection was different, we may be dealing with clones │ │ │ │ │ - feature = feature._original || feature; │ │ │ │ │ - state = feature.state; │ │ │ │ │ - if (state) { │ │ │ │ │ - if (state == OpenLayers.State.DELETE) { │ │ │ │ │ - destroys.push(feature); │ │ │ │ │ - } else if (state == OpenLayers.State.INSERT) { │ │ │ │ │ - feature.fid = insertIds[j]; │ │ │ │ │ - ++j; │ │ │ │ │ - } │ │ │ │ │ - feature.state = null; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (destroys.length > 0) { │ │ │ │ │ - this.layer.destroyFeatures(destroys); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.events.triggerEvent("success", evt); │ │ │ │ │ - │ │ │ │ │ - } else { │ │ │ │ │ - this.events.triggerEvent("fail", evt); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Save" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy/Fixed.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + createParams: function(center, zoom, layers) { │ │ │ │ │ + center = center || this.map.getCenter(); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Strategy.js │ │ │ │ │ - */ │ │ │ │ │ + var params = OpenLayers.Util.getParameters(this.base); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Strategy.Fixed │ │ │ │ │ - * A simple strategy that requests features once and never requests new data. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Strategy> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Strategy.Fixed = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + // If there's still no center, map is not initialized yet. │ │ │ │ │ + // Break out of this function, and simply return the params from the │ │ │ │ │ + // base link. │ │ │ │ │ + if (center) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: preload │ │ │ │ │ - * {Boolean} Load data before layer made visible. Enabling this may result │ │ │ │ │ - * in considerable overhead if your application loads many data layers │ │ │ │ │ - * that are not visible by default. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - preload: false, │ │ │ │ │ + //zoom │ │ │ │ │ + params.zoom = zoom || this.map.getZoom(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy.Fixed │ │ │ │ │ - * Create a new Fixed strategy. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ + //lon,lat │ │ │ │ │ + var lat = center.lat; │ │ │ │ │ + var lon = center.lon; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Activate the strategy: load data or add listener to load when visible │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} True if the strategy was successfully activated or false if │ │ │ │ │ - * the strategy was already active. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "refresh": this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - if (this.layer.visibility == true || this.preload) { │ │ │ │ │ - this.load(); │ │ │ │ │ - } else { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "visibilitychanged": this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + if (this.displayProjection) { │ │ │ │ │ + var mapPosition = OpenLayers.Projection.transform({ │ │ │ │ │ + x: lon, │ │ │ │ │ + y: lat │ │ │ │ │ + }, │ │ │ │ │ + this.map.getProjectionObject(), │ │ │ │ │ + this.displayProjection); │ │ │ │ │ + lon = mapPosition.x; │ │ │ │ │ + lat = mapPosition.y; │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Deactivate the strategy. Undo what is done in <activate>. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - "refresh": this.load, │ │ │ │ │ - "visibilitychanged": this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ + params.lat = Math.round(lat * 100000) / 100000; │ │ │ │ │ + params.lon = Math.round(lon * 100000) / 100000; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: load │ │ │ │ │ - * Tells protocol to load data and unhooks the visibilitychanged event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} options to pass to protocol read. │ │ │ │ │ - */ │ │ │ │ │ - load: function(options) { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - layer.events.triggerEvent("loadstart", { │ │ │ │ │ - filter: layer.filter │ │ │ │ │ - }); │ │ │ │ │ - layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: this.merge, │ │ │ │ │ - filter: layer.filter, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options)); │ │ │ │ │ - layer.events.un({ │ │ │ │ │ - "visibilitychanged": this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + //layers │ │ │ │ │ + layers = layers || this.map.layers; │ │ │ │ │ + params.layers = ''; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + var layer = layers[i]; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: merge │ │ │ │ │ - * Add all features to the layer. │ │ │ │ │ - * If the layer projection differs from the map projection, features │ │ │ │ │ - * will be transformed from the layer projection to the map projection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object passed │ │ │ │ │ - * by the protocol. │ │ │ │ │ - */ │ │ │ │ │ - merge: function(resp) { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - layer.destroyFeatures(); │ │ │ │ │ - var features = resp.features; │ │ │ │ │ - if (features && features.length > 0) { │ │ │ │ │ - var remote = layer.projection; │ │ │ │ │ - var local = layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var geom; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - geom = features[i].geometry; │ │ │ │ │ - if (geom) { │ │ │ │ │ - geom.transform(remote, local); │ │ │ │ │ - } │ │ │ │ │ + if (layer.isBaseLayer) { │ │ │ │ │ + params.layers += (layer == this.map.baseLayer) ? "B" : "0"; │ │ │ │ │ + } else { │ │ │ │ │ + params.layers += (layer.getVisibility()) ? "T" : "F"; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - layer.addFeatures(features); │ │ │ │ │ } │ │ │ │ │ - layer.events.triggerEvent("loadend", { │ │ │ │ │ - response: resp │ │ │ │ │ - }); │ │ │ │ │ + │ │ │ │ │ + return params; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Fixed" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Permalink" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy/BBOX.js │ │ │ │ │ + OpenLayers/Control/ZoomToMaxExtent.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Strategy.js │ │ │ │ │ - * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ + * @requires OpenLayers/Control/Button.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Strategy.BBOX │ │ │ │ │ - * A simple strategy that reads new features when the viewport invalidates │ │ │ │ │ - * some bounds. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Control.ZoomToMaxExtent │ │ │ │ │ + * The ZoomToMaxExtent control is a button that zooms out to the maximum │ │ │ │ │ + * extent of the map. It is designed to be used with a │ │ │ │ │ + * <OpenLayers.Control.Panel>. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Strategy> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Strategy.BBOX = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: bounds │ │ │ │ │ - * {<OpenLayers.Bounds>} The current data bounds (in the same projection │ │ │ │ │ - * as the layer - not always the same projection as the map). │ │ │ │ │ - */ │ │ │ │ │ - bounds: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: resolution │ │ │ │ │ - * {Float} The current data resolution. │ │ │ │ │ - */ │ │ │ │ │ - resolution: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: ratio │ │ │ │ │ - * {Float} The ratio of the data bounds to the viewport bounds (in each │ │ │ │ │ - * dimension). Default is 2. │ │ │ │ │ - */ │ │ │ │ │ - ratio: 2, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: resFactor │ │ │ │ │ - * {Float} Optional factor used to determine when previously requested │ │ │ │ │ - * features are invalid. If set, the resFactor will be compared to the │ │ │ │ │ - * resolution of the previous request to the current map resolution. │ │ │ │ │ - * If resFactor > (old / new) and 1/resFactor < (old / new). If you │ │ │ │ │ - * set a resFactor of 1, data will be requested every time the │ │ │ │ │ - * resolution changes. If you set a resFactor of 3, data will be │ │ │ │ │ - * requested if the old resolution is 3 times the new, or if the new is │ │ │ │ │ - * 3 times the old. If the old bounds do not contain the new bounds │ │ │ │ │ - * new data will always be requested (with or without considering │ │ │ │ │ - * resFactor). │ │ │ │ │ - */ │ │ │ │ │ - resFactor: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: response │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} The protocol response object returned │ │ │ │ │ - * by the layer protocol. │ │ │ │ │ - */ │ │ │ │ │ - response: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy.BBOX │ │ │ │ │ - * Create a new BBOX strategy. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Set up strategy with regard to reading new batches of remote data. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "moveend": this.update, │ │ │ │ │ - "refresh": this.update, │ │ │ │ │ - "visibilitychanged": this.update, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.update(); │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Tear down strategy with regard to reading new batches of remote data. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - "moveend": this.update, │ │ │ │ │ - "refresh": this.update, │ │ │ │ │ - "visibilitychanged": this.update, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: update │ │ │ │ │ - * Callback function called on "moveend" or "refresh" layer events. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will determine │ │ │ │ │ - * the behaviour of this Strategy │ │ │ │ │ - * │ │ │ │ │ - * Valid options include: │ │ │ │ │ - * force - {Boolean} if true, new data must be unconditionally read. │ │ │ │ │ - * noAbort - {Boolean} if true, do not abort previous requests. │ │ │ │ │ - */ │ │ │ │ │ - update: function(options) { │ │ │ │ │ - var mapBounds = this.getMapBounds(); │ │ │ │ │ - if (mapBounds !== null && ((options && options.force) || │ │ │ │ │ - (this.layer.visibility && this.layer.calculateInRange() && this.invalidBounds(mapBounds)))) { │ │ │ │ │ - this.calculateBounds(mapBounds); │ │ │ │ │ - this.resolution = this.layer.map.getResolution(); │ │ │ │ │ - this.triggerRead(options); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getMapBounds │ │ │ │ │ - * Get the map bounds expressed in the same projection as this layer. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} Map bounds in the projection of the layer. │ │ │ │ │ - */ │ │ │ │ │ - getMapBounds: function() { │ │ │ │ │ - if (this.layer.map === null) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - var bounds = this.layer.map.getExtent(); │ │ │ │ │ - if (bounds && !this.layer.projection.equals( │ │ │ │ │ - this.layer.map.getProjectionObject())) { │ │ │ │ │ - bounds = bounds.clone().transform( │ │ │ │ │ - this.layer.map.getProjectionObject(), this.layer.projection │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return bounds; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: invalidBounds │ │ │ │ │ - * Determine whether the previously requested set of features is invalid. │ │ │ │ │ - * This occurs when the new map bounds do not contain the previously │ │ │ │ │ - * requested bounds. In addition, if <resFactor> is set, it will be │ │ │ │ │ - * considered. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * mapBounds - {<OpenLayers.Bounds>} the current map extent, will be │ │ │ │ │ - * retrieved from the map object if not provided │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - invalidBounds: function(mapBounds) { │ │ │ │ │ - if (!mapBounds) { │ │ │ │ │ - mapBounds = this.getMapBounds(); │ │ │ │ │ - } │ │ │ │ │ - var invalid = !this.bounds || !this.bounds.containsBounds(mapBounds); │ │ │ │ │ - if (!invalid && this.resFactor) { │ │ │ │ │ - var ratio = this.resolution / this.layer.map.getResolution(); │ │ │ │ │ - invalid = (ratio >= this.resFactor || ratio <= (1 / this.resFactor)); │ │ │ │ │ - } │ │ │ │ │ - return invalid; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateBounds │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * mapBounds - {<OpenLayers.Bounds>} the current map extent, will be │ │ │ │ │ - * retrieved from the map object if not provided │ │ │ │ │ - */ │ │ │ │ │ - calculateBounds: function(mapBounds) { │ │ │ │ │ - if (!mapBounds) { │ │ │ │ │ - mapBounds = this.getMapBounds(); │ │ │ │ │ - } │ │ │ │ │ - var center = mapBounds.getCenterLonLat(); │ │ │ │ │ - var dataWidth = mapBounds.getWidth() * this.ratio; │ │ │ │ │ - var dataHeight = mapBounds.getHeight() * this.ratio; │ │ │ │ │ - this.bounds = new OpenLayers.Bounds( │ │ │ │ │ - center.lon - (dataWidth / 2), │ │ │ │ │ - center.lat - (dataHeight / 2), │ │ │ │ │ - center.lon + (dataWidth / 2), │ │ │ │ │ - center.lat + (dataHeight / 2) │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: triggerRead │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Additional options for the protocol's read method │ │ │ │ │ - * (optional) │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} The protocol response object │ │ │ │ │ - * returned by the layer protocol. │ │ │ │ │ - */ │ │ │ │ │ - triggerRead: function(options) { │ │ │ │ │ - if (this.response && !(options && options.noAbort === true)) { │ │ │ │ │ - this.layer.protocol.abort(this.response); │ │ │ │ │ - this.layer.events.triggerEvent("loadend"); │ │ │ │ │ - } │ │ │ │ │ - var evt = { │ │ │ │ │ - filter: this.createFilter() │ │ │ │ │ - }; │ │ │ │ │ - this.layer.events.triggerEvent("loadstart", evt); │ │ │ │ │ - this.response = this.layer.protocol.read( │ │ │ │ │ - OpenLayers.Util.applyDefaults({ │ │ │ │ │ - filter: evt.filter, │ │ │ │ │ - callback: this.merge, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options)); │ │ │ │ │ - }, │ │ │ │ │ +OpenLayers.Control.ZoomToMaxExtent = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createFilter │ │ │ │ │ - * Creates a spatial BBOX filter. If the layer that this strategy belongs │ │ │ │ │ - * to has a filter property, this filter will be combined with the BBOX │ │ │ │ │ - * filter. │ │ │ │ │ + * Method: trigger │ │ │ │ │ * │ │ │ │ │ - * Returns │ │ │ │ │ - * {<OpenLayers.Filter>} The filter object. │ │ │ │ │ - */ │ │ │ │ │ - createFilter: function() { │ │ │ │ │ - var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ - value: this.bounds, │ │ │ │ │ - projection: this.layer.projection │ │ │ │ │ - }); │ │ │ │ │ - if (this.layer.filter) { │ │ │ │ │ - filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ - type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ - filters: [this.layer.filter, filter] │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return filter; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: merge │ │ │ │ │ - * Given a list of features, determine which ones to add to the layer. │ │ │ │ │ - * If the layer projection differs from the map projection, features │ │ │ │ │ - * will be transformed from the layer projection to the map projection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object passed │ │ │ │ │ - * by the protocol. │ │ │ │ │ + * Called whenever this control is being rendered inside of a panel and a │ │ │ │ │ + * click occurs on this controls element. Actually zooms to the maximum │ │ │ │ │ + * extent of this controls map. │ │ │ │ │ */ │ │ │ │ │ - merge: function(resp) { │ │ │ │ │ - this.layer.destroyFeatures(); │ │ │ │ │ - if (resp.success()) { │ │ │ │ │ - var features = resp.features; │ │ │ │ │ - if (features && features.length > 0) { │ │ │ │ │ - var remote = this.layer.projection; │ │ │ │ │ - var local = this.layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var geom; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - geom = features[i].geometry; │ │ │ │ │ - if (geom) { │ │ │ │ │ - geom.transform(remote, local); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.layer.addFeatures(features); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.bounds = null; │ │ │ │ │ + trigger: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.zoomToMaxExtent(); │ │ │ │ │ } │ │ │ │ │ - this.response = null; │ │ │ │ │ - this.layer.events.triggerEvent("loadend", { │ │ │ │ │ - response: resp │ │ │ │ │ - }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.BBOX" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomToMaxExtent" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Control/Panel.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ @@ -66620,792 +65531,1796 @@ │ │ │ │ │ return this.getControlsBy("CLASS_NAME", match); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.Panel" │ │ │ │ │ }); │ │ │ │ │ │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/ZoomBox.js │ │ │ │ │ + OpenLayers/Control/Pan.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Handler/Box.js │ │ │ │ │ + * @requires OpenLayers/Control/Button.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.ZoomBox │ │ │ │ │ - * The ZoomBox control enables zooming directly to a given extent, by drawing │ │ │ │ │ - * a box on the map. The box is drawn by holding down shift, whilst dragging │ │ │ │ │ - * the mouse. │ │ │ │ │ + * Class: OpenLayers.Control.Pan │ │ │ │ │ + * The Pan control is a single button to pan the map in one direction. For │ │ │ │ │ + * a more complete control see <OpenLayers.Control.PanPanel>. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.ZoomBox = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - /** │ │ │ │ │ - * Property: type │ │ │ │ │ - * {OpenLayers.Control.TYPE} │ │ │ │ │ - */ │ │ │ │ │ - type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ +OpenLayers.Control.Pan = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: out │ │ │ │ │ - * {Boolean} Should the control be used for zooming out? │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: slideFactor │ │ │ │ │ + * {Integer} Number of pixels by which we'll pan the map in any direction │ │ │ │ │ + * on clicking the arrow buttons, defaults to 50. If you want to pan │ │ │ │ │ + * by some ratio of the map dimensions, use <slideRatio> instead. │ │ │ │ │ */ │ │ │ │ │ - out: false, │ │ │ │ │ + slideFactor: 50, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: keyMask │ │ │ │ │ - * {Integer} Zoom only occurs if the keyMask matches the combination of │ │ │ │ │ - * keys down. Use bitwise operators and one or more of the │ │ │ │ │ - * <OpenLayers.Handler> constants to construct a keyMask. Leave null if │ │ │ │ │ - * not used mask. Default is null. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: slideRatio │ │ │ │ │ + * {Number} The fraction of map width/height by which we'll pan the map │ │ │ │ │ + * on clicking the arrow buttons. Default is null. If set, will │ │ │ │ │ + * override <slideFactor>. E.g. if slideRatio is .5, then Pan Up will │ │ │ │ │ + * pan up half the map height. │ │ │ │ │ */ │ │ │ │ │ - keyMask: null, │ │ │ │ │ + slideRatio: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: alwaysZoom │ │ │ │ │ - * {Boolean} Always zoom in/out when box drawn, even if the zoom level does │ │ │ │ │ - * not change. │ │ │ │ │ + /** │ │ │ │ │ + * Property: direction │ │ │ │ │ + * {String} in {'North', 'South', 'East', 'West'} │ │ │ │ │ */ │ │ │ │ │ - alwaysZoom: false, │ │ │ │ │ + direction: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomOnClick │ │ │ │ │ - * {Boolean} Should we zoom when no box was dragged, i.e. the user only │ │ │ │ │ - * clicked? Default is true. │ │ │ │ │ + * Constructor: OpenLayers.Control.Pan │ │ │ │ │ + * Control which handles the panning (in any of the cardinal directions) │ │ │ │ │ + * of the map by a set px distance. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * direction - {String} The direction this button should pan. │ │ │ │ │ + * options - {Object} An optional object whose properties will be used │ │ │ │ │ + * to extend the control. │ │ │ │ │ */ │ │ │ │ │ - zoomOnClick: true, │ │ │ │ │ + initialize: function(direction, options) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Box(this, { │ │ │ │ │ - done: this.zoomBox │ │ │ │ │ - }, { │ │ │ │ │ - keyMask: this.keyMask │ │ │ │ │ - }); │ │ │ │ │ + this.direction = direction; │ │ │ │ │ + this.CLASS_NAME += this.direction; │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: zoomBox │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * position - {<OpenLayers.Bounds>} or {<OpenLayers.Pixel>} │ │ │ │ │ + * Method: trigger │ │ │ │ │ */ │ │ │ │ │ - zoomBox: function(position) { │ │ │ │ │ - if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ - var bounds, │ │ │ │ │ - targetCenterPx = position.getCenterPixel(); │ │ │ │ │ - if (!this.out) { │ │ │ │ │ - var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.left, │ │ │ │ │ - y: position.bottom │ │ │ │ │ - }); │ │ │ │ │ - var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.right, │ │ │ │ │ - y: position.top │ │ │ │ │ - }); │ │ │ │ │ - bounds = new OpenLayers.Bounds(minXY.lon, minXY.lat, │ │ │ │ │ - maxXY.lon, maxXY.lat); │ │ │ │ │ - } else { │ │ │ │ │ - var pixWidth = position.right - position.left; │ │ │ │ │ - var pixHeight = position.bottom - position.top; │ │ │ │ │ - var zoomFactor = Math.min((this.map.size.h / pixHeight), │ │ │ │ │ - (this.map.size.w / pixWidth)); │ │ │ │ │ - var extent = this.map.getExtent(); │ │ │ │ │ - var center = this.map.getLonLatFromPixel(targetCenterPx); │ │ │ │ │ - var xmin = center.lon - (extent.getWidth() / 2) * zoomFactor; │ │ │ │ │ - var xmax = center.lon + (extent.getWidth() / 2) * zoomFactor; │ │ │ │ │ - var ymin = center.lat - (extent.getHeight() / 2) * zoomFactor; │ │ │ │ │ - var ymax = center.lat + (extent.getHeight() / 2) * zoomFactor; │ │ │ │ │ - bounds = new OpenLayers.Bounds(xmin, ymin, xmax, ymax); │ │ │ │ │ - } │ │ │ │ │ - // always zoom in/out │ │ │ │ │ - var lastZoom = this.map.getZoom(), │ │ │ │ │ - size = this.map.getSize(), │ │ │ │ │ - centerPx = { │ │ │ │ │ - x: size.w / 2, │ │ │ │ │ - y: size.h / 2 │ │ │ │ │ - }, │ │ │ │ │ - zoom = this.map.getZoomForExtent(bounds), │ │ │ │ │ - oldRes = this.map.getResolution(), │ │ │ │ │ - newRes = this.map.getResolutionForZoom(zoom); │ │ │ │ │ - if (oldRes == newRes) { │ │ │ │ │ - this.map.setCenter(this.map.getLonLatFromPixel(targetCenterPx)); │ │ │ │ │ - } else { │ │ │ │ │ - var zoomOriginPx = { │ │ │ │ │ - x: (oldRes * targetCenterPx.x - newRes * centerPx.x) / │ │ │ │ │ - (oldRes - newRes), │ │ │ │ │ - y: (oldRes * targetCenterPx.y - newRes * centerPx.y) / │ │ │ │ │ - (oldRes - newRes) │ │ │ │ │ - }; │ │ │ │ │ - this.map.zoomTo(zoom, zoomOriginPx); │ │ │ │ │ - } │ │ │ │ │ - if (lastZoom == this.map.getZoom() && this.alwaysZoom == true) { │ │ │ │ │ - this.map.zoomTo(lastZoom + (this.out ? -1 : 1)); │ │ │ │ │ - } │ │ │ │ │ - } else if (this.zoomOnClick) { // it's a pixel │ │ │ │ │ - if (!this.out) { │ │ │ │ │ - this.map.zoomTo(this.map.getZoom() + 1, position); │ │ │ │ │ - } else { │ │ │ │ │ - this.map.zoomTo(this.map.getZoom() - 1, position); │ │ │ │ │ + trigger: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + var getSlideFactor = OpenLayers.Function.bind(function(dim) { │ │ │ │ │ + return this.slideRatio ? │ │ │ │ │ + this.map.getSize()[dim] * this.slideRatio : │ │ │ │ │ + this.slideFactor; │ │ │ │ │ + }, this); │ │ │ │ │ + │ │ │ │ │ + switch (this.direction) { │ │ │ │ │ + case OpenLayers.Control.Pan.NORTH: │ │ │ │ │ + this.map.pan(0, -getSlideFactor("h")); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Control.Pan.SOUTH: │ │ │ │ │ + this.map.pan(0, getSlideFactor("h")); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Control.Pan.WEST: │ │ │ │ │ + this.map.pan(-getSlideFactor("w"), 0); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Control.Pan.EAST: │ │ │ │ │ + this.map.pan(getSlideFactor("w"), 0); │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomBox" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Pan" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Control.Pan.NORTH = "North"; │ │ │ │ │ +OpenLayers.Control.Pan.SOUTH = "South"; │ │ │ │ │ +OpenLayers.Control.Pan.EAST = "East"; │ │ │ │ │ +OpenLayers.Control.Pan.WEST = "West"; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/DragPan.js │ │ │ │ │ + OpenLayers/Layer/Vector.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Handler/Drag.js │ │ │ │ │ + * @requires OpenLayers/Layer.js │ │ │ │ │ + * @requires OpenLayers/Renderer.js │ │ │ │ │ + * @requires OpenLayers/StyleMap.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Console.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.DragPan │ │ │ │ │ - * The DragPan control pans the map with a drag of the mouse. │ │ │ │ │ + * Class: OpenLayers.Layer.Vector │ │ │ │ │ + * Instances of OpenLayers.Layer.Vector are used to render vector data from │ │ │ │ │ + * a variety of sources. Create a new vector layer with the │ │ │ │ │ + * <OpenLayers.Layer.Vector> constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ + * - <OpenLayers.Layer> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.DragPan = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Layer.Vector = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * layer.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Listeners will be called with a reference to an event object. The │ │ │ │ │ + * properties of this event depends on exactly what happened. │ │ │ │ │ + * │ │ │ │ │ + * All event objects have at least the following properties: │ │ │ │ │ + * object - {Object} A reference to layer.events.object. │ │ │ │ │ + * element - {DOMElement} A reference to layer.events.element. │ │ │ │ │ + * │ │ │ │ │ + * Supported map event types (in addition to those from <OpenLayers.Layer.events>): │ │ │ │ │ + * beforefeatureadded - Triggered before a feature is added. Listeners │ │ │ │ │ + * will receive an object with a *feature* property referencing the │ │ │ │ │ + * feature to be added. To stop the feature from being added, a │ │ │ │ │ + * listener should return false. │ │ │ │ │ + * beforefeaturesadded - Triggered before an array of features is added. │ │ │ │ │ + * Listeners will receive an object with a *features* property │ │ │ │ │ + * referencing the feature to be added. To stop the features from │ │ │ │ │ + * being added, a listener should return false. │ │ │ │ │ + * featureadded - Triggered after a feature is added. The event │ │ │ │ │ + * object passed to listeners will have a *feature* property with a │ │ │ │ │ + * reference to the added feature. │ │ │ │ │ + * featuresadded - Triggered after features are added. The event │ │ │ │ │ + * object passed to listeners will have a *features* property with a │ │ │ │ │ + * reference to an array of added features. │ │ │ │ │ + * beforefeatureremoved - Triggered before a feature is removed. Listeners │ │ │ │ │ + * will receive an object with a *feature* property referencing the │ │ │ │ │ + * feature to be removed. │ │ │ │ │ + * beforefeaturesremoved - Triggered before multiple features are removed. │ │ │ │ │ + * Listeners will receive an object with a *features* property │ │ │ │ │ + * referencing the features to be removed. │ │ │ │ │ + * featureremoved - Triggerd after a feature is removed. The event │ │ │ │ │ + * object passed to listeners will have a *feature* property with a │ │ │ │ │ + * reference to the removed feature. │ │ │ │ │ + * featuresremoved - Triggered after features are removed. The event │ │ │ │ │ + * object passed to listeners will have a *features* property with a │ │ │ │ │ + * reference to an array of removed features. │ │ │ │ │ + * beforefeatureselected - Triggered before a feature is selected. Listeners │ │ │ │ │ + * will receive an object with a *feature* property referencing the │ │ │ │ │ + * feature to be selected. To stop the feature from being selectd, a │ │ │ │ │ + * listener should return false. │ │ │ │ │ + * featureselected - Triggered after a feature is selected. Listeners │ │ │ │ │ + * will receive an object with a *feature* property referencing the │ │ │ │ │ + * selected feature. │ │ │ │ │ + * featureunselected - Triggered after a feature is unselected. │ │ │ │ │ + * Listeners will receive an object with a *feature* property │ │ │ │ │ + * referencing the unselected feature. │ │ │ │ │ + * beforefeaturemodified - Triggered when a feature is selected to │ │ │ │ │ + * be modified. Listeners will receive an object with a *feature* │ │ │ │ │ + * property referencing the selected feature. │ │ │ │ │ + * featuremodified - Triggered when a feature has been modified. │ │ │ │ │ + * Listeners will receive an object with a *feature* property referencing │ │ │ │ │ + * the modified feature. │ │ │ │ │ + * afterfeaturemodified - Triggered when a feature is finished being modified. │ │ │ │ │ + * Listeners will receive an object with a *feature* property referencing │ │ │ │ │ + * the modified feature. │ │ │ │ │ + * vertexmodified - Triggered when a vertex within any feature geometry │ │ │ │ │ + * has been modified. Listeners will receive an object with a │ │ │ │ │ + * *feature* property referencing the modified feature, a *vertex* │ │ │ │ │ + * property referencing the vertex modified (always a point geometry), │ │ │ │ │ + * and a *pixel* property referencing the pixel location of the │ │ │ │ │ + * modification. │ │ │ │ │ + * vertexremoved - Triggered when a vertex within any feature geometry │ │ │ │ │ + * has been deleted. Listeners will receive an object with a │ │ │ │ │ + * *feature* property referencing the modified feature, a *vertex* │ │ │ │ │ + * property referencing the vertex modified (always a point geometry), │ │ │ │ │ + * and a *pixel* property referencing the pixel location of the │ │ │ │ │ + * removal. │ │ │ │ │ + * sketchstarted - Triggered when a feature sketch bound for this layer │ │ │ │ │ + * is started. Listeners will receive an object with a *feature* │ │ │ │ │ + * property referencing the new sketch feature and a *vertex* property │ │ │ │ │ + * referencing the creation point. │ │ │ │ │ + * sketchmodified - Triggered when a feature sketch bound for this layer │ │ │ │ │ + * is modified. Listeners will receive an object with a *vertex* │ │ │ │ │ + * property referencing the modified vertex and a *feature* property │ │ │ │ │ + * referencing the sketch feature. │ │ │ │ │ + * sketchcomplete - Triggered when a feature sketch bound for this layer │ │ │ │ │ + * is complete. Listeners will receive an object with a *feature* │ │ │ │ │ + * property referencing the sketch feature. By returning false, a │ │ │ │ │ + * listener can stop the sketch feature from being added to the layer. │ │ │ │ │ + * refresh - Triggered when something wants a strategy to ask the protocol │ │ │ │ │ + * for a new set of features. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} The layer is a base layer. Default is false. Set this property │ │ │ │ │ + * in the layer options. │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: type │ │ │ │ │ - * {OpenLayers.Control.TYPES} │ │ │ │ │ + * APIProperty: isFixed │ │ │ │ │ + * {Boolean} Whether the layer remains in one place while dragging the │ │ │ │ │ + * map. Note that setting this to true will move the layer to the bottom │ │ │ │ │ + * of the layer stack. │ │ │ │ │ */ │ │ │ │ │ - type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ + isFixed: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: features │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + */ │ │ │ │ │ + features: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: filter │ │ │ │ │ + * {<OpenLayers.Filter>} The filter set in this layer, │ │ │ │ │ + * a strategy launching read requests can combined │ │ │ │ │ + * this filter with its own filter. │ │ │ │ │ + */ │ │ │ │ │ + filter: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: selectedFeatures │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + */ │ │ │ │ │ + selectedFeatures: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: panned │ │ │ │ │ - * {Boolean} The map moved. │ │ │ │ │ + * Property: unrenderedFeatures │ │ │ │ │ + * {Object} hash of features, keyed by feature.id, that the renderer │ │ │ │ │ + * failed to draw │ │ │ │ │ */ │ │ │ │ │ - panned: false, │ │ │ │ │ + unrenderedFeatures: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: interval │ │ │ │ │ - * {Integer} The number of milliseconds that should ellapse before │ │ │ │ │ - * panning the map again. Defaults to 0 milliseconds, which means that │ │ │ │ │ - * no separate cycle is used for panning. In most cases you won't want │ │ │ │ │ - * to change this value. For slow machines/devices larger values can be │ │ │ │ │ - * tried out. │ │ │ │ │ + * APIProperty: reportError │ │ │ │ │ + * {Boolean} report friendly error message when loading of renderer │ │ │ │ │ + * fails. │ │ │ │ │ */ │ │ │ │ │ - interval: 0, │ │ │ │ │ + reportError: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: style │ │ │ │ │ + * {Object} Default style for the layer │ │ │ │ │ + */ │ │ │ │ │ + style: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: documentDrag │ │ │ │ │ - * {Boolean} If set to true, mouse dragging will continue even if the │ │ │ │ │ - * mouse cursor leaves the map viewport. Default is false. │ │ │ │ │ + * Property: styleMap │ │ │ │ │ + * {<OpenLayers.StyleMap>} │ │ │ │ │ */ │ │ │ │ │ - documentDrag: false, │ │ │ │ │ + styleMap: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: kinetic │ │ │ │ │ - * {<OpenLayers.Kinetic>} The OpenLayers.Kinetic object. │ │ │ │ │ + * Property: strategies │ │ │ │ │ + * {Array(<OpenLayers.Strategy>})} Optional list of strategies for the layer. │ │ │ │ │ */ │ │ │ │ │ - kinetic: null, │ │ │ │ │ + strategies: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: enableKinetic │ │ │ │ │ - * {Boolean} Set this option to enable "kinetic dragging". Can be │ │ │ │ │ - * set to true or to an object. If set to an object this │ │ │ │ │ - * object will be passed to the {<OpenLayers.Kinetic>} │ │ │ │ │ - * constructor. Defaults to true. │ │ │ │ │ - * To get kinetic dragging, ensure that OpenLayers/Kinetic.js is │ │ │ │ │ - * included in your build config. │ │ │ │ │ + * Property: protocol │ │ │ │ │ + * {<OpenLayers.Protocol>} Optional protocol for the layer. │ │ │ │ │ */ │ │ │ │ │ - enableKinetic: true, │ │ │ │ │ + protocol: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: kineticInterval │ │ │ │ │ - * {Integer} Interval in milliseconds between 2 steps in the "kinetic │ │ │ │ │ - * scrolling". Applies only if enableKinetic is set. Defaults │ │ │ │ │ - * to 10 milliseconds. │ │ │ │ │ + * Property: renderers │ │ │ │ │ + * {Array(String)} List of supported Renderer classes. Add to this list to │ │ │ │ │ + * add support for additional renderers. This list is ordered: │ │ │ │ │ + * the first renderer which returns true for the 'supported()' │ │ │ │ │ + * method will be used, if not defined in the 'renderer' option. │ │ │ │ │ */ │ │ │ │ │ - kineticInterval: 10, │ │ │ │ │ + renderers: ['SVG', 'VML', 'Canvas'], │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: renderer │ │ │ │ │ + * {<OpenLayers.Renderer>} │ │ │ │ │ + */ │ │ │ │ │ + renderer: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Creates a Drag handler, using <panMap> and │ │ │ │ │ - * <panMapDone> as callbacks. │ │ │ │ │ + * APIProperty: rendererOptions │ │ │ │ │ + * {Object} Options for the renderer. See {<OpenLayers.Renderer>} for │ │ │ │ │ + * supported options. │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - if (this.enableKinetic && OpenLayers.Kinetic) { │ │ │ │ │ - var config = { │ │ │ │ │ - interval: this.kineticInterval │ │ │ │ │ - }; │ │ │ │ │ - if (typeof this.enableKinetic === "object") { │ │ │ │ │ - config = OpenLayers.Util.extend(config, this.enableKinetic); │ │ │ │ │ - } │ │ │ │ │ - this.kinetic = new OpenLayers.Kinetic(config); │ │ │ │ │ - } │ │ │ │ │ - this.handler = new OpenLayers.Handler.Drag(this, { │ │ │ │ │ - "move": this.panMap, │ │ │ │ │ - "done": this.panMapDone, │ │ │ │ │ - "down": this.panMapStart │ │ │ │ │ - }, { │ │ │ │ │ - interval: this.interval, │ │ │ │ │ - documentDrag: this.documentDrag │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + rendererOptions: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: panMapStart │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: geometryType │ │ │ │ │ + * {String} geometryType allows you to limit the types of geometries this │ │ │ │ │ + * layer supports. This should be set to something like │ │ │ │ │ + * "OpenLayers.Geometry.Point" to limit types. │ │ │ │ │ */ │ │ │ │ │ - panMapStart: function() { │ │ │ │ │ - if (this.kinetic) { │ │ │ │ │ - this.kinetic.begin(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + geometryType: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: drawn │ │ │ │ │ + * {Boolean} Whether the Vector Layer features have been drawn yet. │ │ │ │ │ + */ │ │ │ │ │ + drawn: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: ratio │ │ │ │ │ + * {Float} This specifies the ratio of the size of the visiblity of the Vector Layer features to the size of the map. │ │ │ │ │ + */ │ │ │ │ │ + ratio: 1, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: panMap │ │ │ │ │ + * Constructor: OpenLayers.Layer.Vector │ │ │ │ │ + * Create a new vector layer │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} Pixel of the mouse position │ │ │ │ │ + * name - {String} A name for the layer │ │ │ │ │ + * options - {Object} Optional object with non-default properties to set on │ │ │ │ │ + * the layer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} A new vector layer │ │ │ │ │ */ │ │ │ │ │ - panMap: function(xy) { │ │ │ │ │ - if (this.kinetic) { │ │ │ │ │ - this.kinetic.update(xy); │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + // allow user-set renderer, otherwise assign one │ │ │ │ │ + if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ + this.assignRenderer(); │ │ │ │ │ } │ │ │ │ │ - this.panned = true; │ │ │ │ │ - this.map.pan( │ │ │ │ │ - this.handler.last.x - xy.x, │ │ │ │ │ - this.handler.last.y - xy.y, { │ │ │ │ │ - dragging: true, │ │ │ │ │ - animate: false │ │ │ │ │ + │ │ │ │ │ + // if no valid renderer found, display error │ │ │ │ │ + if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ + this.renderer = null; │ │ │ │ │ + this.displayError(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (!this.styleMap) { │ │ │ │ │ + this.styleMap = new OpenLayers.StyleMap(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.features = []; │ │ │ │ │ + this.selectedFeatures = []; │ │ │ │ │ + this.unrenderedFeatures = {}; │ │ │ │ │ + │ │ │ │ │ + // Allow for custom layer behavior │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + for (var i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + this.strategies[i].setLayer(this); │ │ │ │ │ } │ │ │ │ │ - ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: panMapDone │ │ │ │ │ - * Finish the panning operation. Only call setCenter (through <panMap>) │ │ │ │ │ - * if the map has actually been moved. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} Pixel of the mouse position │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Destroy this layer │ │ │ │ │ */ │ │ │ │ │ - panMapDone: function(xy) { │ │ │ │ │ - if (this.panned) { │ │ │ │ │ - var res = null; │ │ │ │ │ - if (this.kinetic) { │ │ │ │ │ - res = this.kinetic.end(xy); │ │ │ │ │ - } │ │ │ │ │ - this.map.pan( │ │ │ │ │ - this.handler.last.x - xy.x, │ │ │ │ │ - this.handler.last.y - xy.y, { │ │ │ │ │ - dragging: !!res, │ │ │ │ │ - animate: false │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoDestroy) { │ │ │ │ │ + strategy.destroy(); │ │ │ │ │ } │ │ │ │ │ - ); │ │ │ │ │ - if (res) { │ │ │ │ │ - var self = this; │ │ │ │ │ - this.kinetic.move(res, function(x, y, end) { │ │ │ │ │ - self.map.pan(x, y, { │ │ │ │ │ - dragging: !end, │ │ │ │ │ - animate: false │ │ │ │ │ - }); │ │ │ │ │ - }); │ │ │ │ │ } │ │ │ │ │ - this.panned = false; │ │ │ │ │ + this.strategies = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.protocol) { │ │ │ │ │ + if (this.protocol.autoDestroy) { │ │ │ │ │ + this.protocol.destroy(); │ │ │ │ │ + } │ │ │ │ │ + this.protocol = null; │ │ │ │ │ + } │ │ │ │ │ + this.destroyFeatures(); │ │ │ │ │ + this.features = null; │ │ │ │ │ + this.selectedFeatures = null; │ │ │ │ │ + this.unrenderedFeatures = null; │ │ │ │ │ + if (this.renderer) { │ │ │ │ │ + this.renderer.destroy(); │ │ │ │ │ } │ │ │ │ │ + this.renderer = null; │ │ │ │ │ + this.geometryType = null; │ │ │ │ │ + this.drawn = null; │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.DragPan" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Navigation.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer. │ │ │ │ │ + * │ │ │ │ │ + * Note: Features of the layer are also cloned. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} An exact clone of this layer │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control/ZoomBox.js │ │ │ │ │ - * @requires OpenLayers/Control/DragPan.js │ │ │ │ │ - * @requires OpenLayers/Handler/MouseWheel.js │ │ │ │ │ - * @requires OpenLayers/Handler/Click.js │ │ │ │ │ - */ │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Vector(this.name, this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Navigation │ │ │ │ │ - * The navigation control handles map browsing with mouse events (dragging, │ │ │ │ │ - * double-clicking, and scrolling the wheel). Create a new navigation │ │ │ │ │ - * control with the <OpenLayers.Control.Navigation> control. │ │ │ │ │ - * │ │ │ │ │ - * Note that this control is added to the map by default (if no controls │ │ │ │ │ - * array is sent in the options object to the <OpenLayers.Map> │ │ │ │ │ - * constructor). │ │ │ │ │ - * │ │ │ │ │ - * Inherits: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Navigation = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: dragPan │ │ │ │ │ - * {<OpenLayers.Control.DragPan>} │ │ │ │ │ - */ │ │ │ │ │ - dragPan: null, │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + var features = this.features; │ │ │ │ │ + var len = features.length; │ │ │ │ │ + var clonedFeatures = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + clonedFeatures[i] = features[i].clone(); │ │ │ │ │ + } │ │ │ │ │ + obj.features = clonedFeatures; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: dragPanOptions │ │ │ │ │ - * {Object} Options passed to the DragPan control. │ │ │ │ │ - */ │ │ │ │ │ - dragPanOptions: null, │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: pinchZoom │ │ │ │ │ - * {<OpenLayers.Control.PinchZoom>} │ │ │ │ │ + * Method: refresh │ │ │ │ │ + * Ask the layer to request features again and redraw them. Triggers │ │ │ │ │ + * the refresh event if the layer is in range and visible. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} Optional object with properties for any listener of │ │ │ │ │ + * the refresh event. │ │ │ │ │ */ │ │ │ │ │ - pinchZoom: null, │ │ │ │ │ + refresh: function(obj) { │ │ │ │ │ + if (this.calculateInRange() && this.visibility) { │ │ │ │ │ + this.events.triggerEvent("refresh", obj); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: pinchZoomOptions │ │ │ │ │ - * {Object} Options passed to the PinchZoom control. │ │ │ │ │ + /** │ │ │ │ │ + * Method: assignRenderer │ │ │ │ │ + * Iterates through the available renderer implementations and selects │ │ │ │ │ + * and assigns the first one whose "supported()" function returns true. │ │ │ │ │ */ │ │ │ │ │ - pinchZoomOptions: null, │ │ │ │ │ + assignRenderer: function() { │ │ │ │ │ + for (var i = 0, len = this.renderers.length; i < len; i++) { │ │ │ │ │ + var rendererClass = this.renderers[i]; │ │ │ │ │ + var renderer = (typeof rendererClass == "function") ? │ │ │ │ │ + rendererClass : │ │ │ │ │ + OpenLayers.Renderer[rendererClass]; │ │ │ │ │ + if (renderer && renderer.prototype.supported()) { │ │ │ │ │ + this.renderer = new renderer(this.div, this.rendererOptions); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: documentDrag │ │ │ │ │ - * {Boolean} Allow panning of the map by dragging outside map viewport. │ │ │ │ │ - * Default is false. │ │ │ │ │ + /** │ │ │ │ │ + * Method: displayError │ │ │ │ │ + * Let the user know their browser isn't supported. │ │ │ │ │ */ │ │ │ │ │ - documentDrag: false, │ │ │ │ │ + displayError: function() { │ │ │ │ │ + if (this.reportError) { │ │ │ │ │ + OpenLayers.Console.userError(OpenLayers.i18n("browserNotSupported", { │ │ │ │ │ + renderers: this.renderers.join('\n') │ │ │ │ │ + })); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: zoomBox │ │ │ │ │ - * {<OpenLayers.Control.ZoomBox>} │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * The layer has been added to the map. │ │ │ │ │ + * │ │ │ │ │ + * If there is no renderer set, the layer can't be used. Remove it. │ │ │ │ │ + * Otherwise, give the renderer a reference to the map and set its size. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - zoomBox: null, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomBoxEnabled │ │ │ │ │ - * {Boolean} Whether the user can draw a box to zoom │ │ │ │ │ - */ │ │ │ │ │ - zoomBoxEnabled: true, │ │ │ │ │ + if (!this.renderer) { │ │ │ │ │ + this.map.removeLayer(this); │ │ │ │ │ + } else { │ │ │ │ │ + this.renderer.map = this.map; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomWheelEnabled │ │ │ │ │ - * {Boolean} Whether the mousewheel should zoom the map │ │ │ │ │ - */ │ │ │ │ │ - zoomWheelEnabled: true, │ │ │ │ │ + var newSize = this.map.getSize(); │ │ │ │ │ + newSize.w = newSize.w * this.ratio; │ │ │ │ │ + newSize.h = newSize.h * this.ratio; │ │ │ │ │ + this.renderer.setSize(newSize); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: mouseWheelOptions │ │ │ │ │ - * {Object} Options passed to the MouseWheel control (only useful if │ │ │ │ │ - * <zoomWheelEnabled> is set to true). Default is no options for maps │ │ │ │ │ - * with fractionalZoom set to true, otherwise │ │ │ │ │ - * {cumulative: false, interval: 50, maxDelta: 6} │ │ │ │ │ + * Method: afterAdd │ │ │ │ │ + * Called at the end of the map.addLayer sequence. At this point, the map │ │ │ │ │ + * will have a base layer. Any autoActivate strategies will be │ │ │ │ │ + * activated here. │ │ │ │ │ */ │ │ │ │ │ - mouseWheelOptions: null, │ │ │ │ │ + afterAdd: function() { │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoActivate) { │ │ │ │ │ + strategy.activate(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: handleRightClicks │ │ │ │ │ - * {Boolean} Whether or not to handle right clicks. Default is false. │ │ │ │ │ + * Method: removeMap │ │ │ │ │ + * The layer has been removed from the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - handleRightClicks: false, │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + this.drawn = false; │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoActivate) { │ │ │ │ │ + strategy.deactivate(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomBoxKeyMask │ │ │ │ │ - * {Integer} <OpenLayers.Handler> key code of the key, which has to be │ │ │ │ │ - * pressed, while drawing the zoom box with the mouse on the screen. │ │ │ │ │ - * You should probably set handleRightClicks to true if you use this │ │ │ │ │ - * with MOD_CTRL, to disable the context menu for machines which use │ │ │ │ │ - * CTRL-Click as a right click. │ │ │ │ │ - * Default: <OpenLayers.Handler.MOD_SHIFT> │ │ │ │ │ + * Method: onMapResize │ │ │ │ │ + * Notify the renderer of the change in size. │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - zoomBoxKeyMask: OpenLayers.Handler.MOD_SHIFT, │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + OpenLayers.Layer.prototype.onMapResize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + var newSize = this.map.getSize(); │ │ │ │ │ + newSize.w = newSize.w * this.ratio; │ │ │ │ │ + newSize.h = newSize.h * this.ratio; │ │ │ │ │ + this.renderer.setSize(newSize); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * Reset the vector layer's div so that it once again is lined up with │ │ │ │ │ + * the map. Notify the renderer of the change of extent, and in the │ │ │ │ │ + * case of a change of zoom level (resolution), have the │ │ │ │ │ + * renderer redraw features. │ │ │ │ │ + * │ │ │ │ │ + * If the layer has not yet been drawn, cycle through the layer's │ │ │ │ │ + * features and draw each one. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * zoomChanged - {Boolean} │ │ │ │ │ + * dragging - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Navigation │ │ │ │ │ - * Create a new navigation control │ │ │ │ │ + var coordSysUnchanged = true; │ │ │ │ │ + if (!dragging) { │ │ │ │ │ + this.renderer.root.style.visibility = 'hidden'; │ │ │ │ │ + │ │ │ │ │ + var viewSize = this.map.getSize(), │ │ │ │ │ + viewWidth = viewSize.w, │ │ │ │ │ + viewHeight = viewSize.h, │ │ │ │ │ + offsetLeft = (viewWidth / 2 * this.ratio) - viewWidth / 2, │ │ │ │ │ + offsetTop = (viewHeight / 2 * this.ratio) - viewHeight / 2; │ │ │ │ │ + offsetLeft += this.map.layerContainerOriginPx.x; │ │ │ │ │ + offsetLeft = -Math.round(offsetLeft); │ │ │ │ │ + offsetTop += this.map.layerContainerOriginPx.y; │ │ │ │ │ + offsetTop = -Math.round(offsetTop); │ │ │ │ │ + │ │ │ │ │ + this.div.style.left = offsetLeft + 'px'; │ │ │ │ │ + this.div.style.top = offsetTop + 'px'; │ │ │ │ │ + │ │ │ │ │ + var extent = this.map.getExtent().scale(this.ratio); │ │ │ │ │ + coordSysUnchanged = this.renderer.setExtent(extent, zoomChanged); │ │ │ │ │ + │ │ │ │ │ + this.renderer.root.style.visibility = 'visible'; │ │ │ │ │ + │ │ │ │ │ + // Force a reflow on gecko based browsers to prevent jump/flicker. │ │ │ │ │ + // This seems to happen on only certain configurations; it was originally │ │ │ │ │ + // noticed in FF 2.0 and Linux. │ │ │ │ │ + if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ + this.div.scrollLeft = this.div.scrollLeft; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (!zoomChanged && coordSysUnchanged) { │ │ │ │ │ + for (var i in this.unrenderedFeatures) { │ │ │ │ │ + var feature = this.unrenderedFeatures[i]; │ │ │ │ │ + this.drawFeature(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!this.drawn || zoomChanged || !coordSysUnchanged) { │ │ │ │ │ + this.drawn = true; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0, len = this.features.length; i < len; i++) { │ │ │ │ │ + this.renderer.locked = (i !== (len - 1)); │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + this.drawFeature(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: display │ │ │ │ │ + * Hide or show the Layer │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * the control │ │ │ │ │ + * display - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.handlers = {}; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ + display: function(display) { │ │ │ │ │ + OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ + // we need to set the display style of the root in case it is attached │ │ │ │ │ + // to a foreign layer │ │ │ │ │ + var currentDisplay = this.div.style.display; │ │ │ │ │ + if (currentDisplay != this.renderer.root.style.display) { │ │ │ │ │ + this.renderer.root.style.display = currentDisplay; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * The destroy method is used to perform any clean up before the control │ │ │ │ │ - * is dereferenced. Typically this is where event listeners are removed │ │ │ │ │ - * to prevent memory leaks. │ │ │ │ │ + * APIMethod: addFeatures │ │ │ │ │ + * Add Features to the layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - │ │ │ │ │ - if (this.dragPan) { │ │ │ │ │ - this.dragPan.destroy(); │ │ │ │ │ + addFeatures: function(features, options) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ } │ │ │ │ │ - this.dragPan = null; │ │ │ │ │ │ │ │ │ │ - if (this.zoomBox) { │ │ │ │ │ - this.zoomBox.destroy(); │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ + if (notify) { │ │ │ │ │ + var event = { │ │ │ │ │ + features: features │ │ │ │ │ + }; │ │ │ │ │ + var ret = this.events.triggerEvent("beforefeaturesadded", event); │ │ │ │ │ + if (ret === false) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + features = event.features; │ │ │ │ │ } │ │ │ │ │ - this.zoomBox = null; │ │ │ │ │ │ │ │ │ │ - if (this.pinchZoom) { │ │ │ │ │ - this.pinchZoom.destroy(); │ │ │ │ │ + // Track successfully added features for featuresadded event, since │ │ │ │ │ + // beforefeatureadded can veto single features. │ │ │ │ │ + var featuresAdded = []; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + if (i != (features.length - 1)) { │ │ │ │ │ + this.renderer.locked = true; │ │ │ │ │ + } else { │ │ │ │ │ + this.renderer.locked = false; │ │ │ │ │ + } │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + │ │ │ │ │ + if (this.geometryType && │ │ │ │ │ + !(feature.geometry instanceof this.geometryType)) { │ │ │ │ │ + throw new TypeError('addFeatures: component should be an ' + │ │ │ │ │ + this.geometryType.prototype.CLASS_NAME); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //give feature reference to its layer │ │ │ │ │ + feature.layer = this; │ │ │ │ │ + │ │ │ │ │ + if (!feature.style && this.style) { │ │ │ │ │ + feature.style = OpenLayers.Util.extend({}, this.style); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (notify) { │ │ │ │ │ + if (this.events.triggerEvent("beforefeatureadded", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) === false) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + this.preFeatureInsert(feature); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + featuresAdded.push(feature); │ │ │ │ │ + this.features.push(feature); │ │ │ │ │ + this.drawFeature(feature); │ │ │ │ │ + │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureadded", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.onFeatureInsert(feature); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.pinchZoom = null; │ │ │ │ │ │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresadded", { │ │ │ │ │ + features: featuresAdded │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: activate │ │ │ │ │ + * APIMethod: removeFeatures │ │ │ │ │ + * Remove features from the layer. This erases any drawn features and │ │ │ │ │ + * removes them from the layer's control. The beforefeatureremoved │ │ │ │ │ + * and featureremoved events will be triggered for each feature. The │ │ │ │ │ + * featuresremoved event will be triggered after all features have │ │ │ │ │ + * been removed. To supress event triggering, use the silent option. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} List of features to be │ │ │ │ │ + * removed. │ │ │ │ │ + * options - {Object} Optional properties for changing behavior of the │ │ │ │ │ + * removal. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * silent - {Boolean} Supress event triggering. Default is false. │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - this.dragPan.activate(); │ │ │ │ │ - if (this.zoomWheelEnabled) { │ │ │ │ │ - this.handlers.wheel.activate(); │ │ │ │ │ + removeFeatures: function(features, options) { │ │ │ │ │ + if (!features || features.length === 0) { │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ - this.handlers.click.activate(); │ │ │ │ │ - if (this.zoomBoxEnabled) { │ │ │ │ │ - this.zoomBox.activate(); │ │ │ │ │ + if (features === this.features) { │ │ │ │ │ + return this.removeAllFeatures(options); │ │ │ │ │ } │ │ │ │ │ - if (this.pinchZoom) { │ │ │ │ │ - this.pinchZoom.activate(); │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ + } │ │ │ │ │ + if (features === this.selectedFeatures) { │ │ │ │ │ + features = features.slice(); │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Control.prototype.activate.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.pinchZoom) { │ │ │ │ │ - this.pinchZoom.deactivate(); │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ + │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent( │ │ │ │ │ + "beforefeaturesremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + // We remain locked so long as we're not at 0 │ │ │ │ │ + // and the 'next' feature has a geometry. We do the geometry check │ │ │ │ │ + // because if all the features after the current one are 'null', we │ │ │ │ │ + // won't call eraseGeometry, so we break the 'renderer functions │ │ │ │ │ + // will always be called with locked=false *last*' rule. The end result │ │ │ │ │ + // is a possible gratiutious unlocking to save a loop through the rest │ │ │ │ │ + // of the list checking the remaining features every time. So long as │ │ │ │ │ + // null geoms are rare, this is probably okay. │ │ │ │ │ + if (i != 0 && features[i - 1].geometry) { │ │ │ │ │ + this.renderer.locked = true; │ │ │ │ │ + } else { │ │ │ │ │ + this.renderer.locked = false; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ + │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.features = OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ + // feature has no layer at this point │ │ │ │ │ + feature.layer = null; │ │ │ │ │ + │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + this.renderer.eraseFeatures(feature); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //in the case that this feature is one of the selected features, │ │ │ │ │ + // remove it from that array as well. │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.selectedFeatures, feature) != -1) { │ │ │ │ │ + OpenLayers.Util.removeItem(this.selectedFeatures, feature); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - this.zoomBox.deactivate(); │ │ │ │ │ - this.dragPan.deactivate(); │ │ │ │ │ - this.handlers.click.deactivate(); │ │ │ │ │ - this.handlers.wheel.deactivate(); │ │ │ │ │ - return OpenLayers.Control.prototype.deactivate.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: removeAllFeatures │ │ │ │ │ + * Remove all features from the layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional properties for changing behavior of the │ │ │ │ │ + * removal. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * silent - {Boolean} Supress event triggering. Default is false. │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - // disable right mouse context menu for support of right click events │ │ │ │ │ - if (this.handleRightClicks) { │ │ │ │ │ - this.map.viewPortDiv.oncontextmenu = OpenLayers.Function.False; │ │ │ │ │ + removeAllFeatures: function(options) { │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ + var features = this.features; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent( │ │ │ │ │ + "beforefeaturesremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var clickCallbacks = { │ │ │ │ │ - 'click': this.defaultClick, │ │ │ │ │ - 'dblclick': this.defaultDblClick, │ │ │ │ │ - 'dblrightclick': this.defaultDblRightClick │ │ │ │ │ - }; │ │ │ │ │ - var clickOptions = { │ │ │ │ │ - 'double': true, │ │ │ │ │ - 'stopDouble': true │ │ │ │ │ - }; │ │ │ │ │ - this.handlers.click = new OpenLayers.Handler.Click( │ │ │ │ │ - this, clickCallbacks, clickOptions │ │ │ │ │ - ); │ │ │ │ │ - this.dragPan = new OpenLayers.Control.DragPan( │ │ │ │ │ - OpenLayers.Util.extend({ │ │ │ │ │ - map: this.map, │ │ │ │ │ - documentDrag: this.documentDrag │ │ │ │ │ - }, this.dragPanOptions) │ │ │ │ │ - ); │ │ │ │ │ - this.zoomBox = new OpenLayers.Control.ZoomBox({ │ │ │ │ │ - map: this.map, │ │ │ │ │ - keyMask: this.zoomBoxKeyMask │ │ │ │ │ - }); │ │ │ │ │ - this.dragPan.draw(); │ │ │ │ │ - this.zoomBox.draw(); │ │ │ │ │ - var wheelOptions = this.map.fractionalZoom ? {} : { │ │ │ │ │ - cumulative: false, │ │ │ │ │ - interval: 50, │ │ │ │ │ - maxDelta: 6 │ │ │ │ │ - }; │ │ │ │ │ - this.handlers.wheel = new OpenLayers.Handler.MouseWheel( │ │ │ │ │ - this, { │ │ │ │ │ - up: this.wheelUp, │ │ │ │ │ - down: this.wheelDown │ │ │ │ │ - }, │ │ │ │ │ - OpenLayers.Util.extend(wheelOptions, this.mouseWheelOptions) │ │ │ │ │ - ); │ │ │ │ │ - if (OpenLayers.Control.PinchZoom) { │ │ │ │ │ - this.pinchZoom = new OpenLayers.Control.PinchZoom( │ │ │ │ │ - OpenLayers.Util.extend({ │ │ │ │ │ - map: this.map │ │ │ │ │ - }, this.pinchZoomOptions)); │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + feature.layer = null; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.renderer.clear(); │ │ │ │ │ + this.features = []; │ │ │ │ │ + this.unrenderedFeatures = {}; │ │ │ │ │ + this.selectedFeatures = []; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: defaultClick │ │ │ │ │ + * APIMethod: destroyFeatures │ │ │ │ │ + * Erase and destroy features on the layer. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} An optional array of │ │ │ │ │ + * features to destroy. If not supplied, all features on the layer │ │ │ │ │ + * will be destroyed. │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - defaultClick: function(evt) { │ │ │ │ │ - if (evt.lastTouches && evt.lastTouches.length == 2) { │ │ │ │ │ - this.map.zoomOut(); │ │ │ │ │ + destroyFeatures: function(features, options) { │ │ │ │ │ + var all = (features == undefined); // evaluates to true if │ │ │ │ │ + // features is null │ │ │ │ │ + if (all) { │ │ │ │ │ + features = this.features; │ │ │ │ │ + } │ │ │ │ │ + if (features) { │ │ │ │ │ + this.removeFeatures(features, options); │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + features[i].destroy(); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: defaultDblClick │ │ │ │ │ + * APIMethod: drawFeature │ │ │ │ │ + * Draw (or redraw) a feature on the layer. If the optional style argument │ │ │ │ │ + * is included, this style will be used. If no style is included, the │ │ │ │ │ + * feature's style will be used. If the feature doesn't have a style, │ │ │ │ │ + * the layer's style will be used. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * This function is not designed to be used when adding features to │ │ │ │ │ + * the layer (use addFeatures instead). It is meant to be used when │ │ │ │ │ + * the style of a feature has changed, or in some other way needs to │ │ │ │ │ + * visually updated *after* it has already been added to a layer. You │ │ │ │ │ + * must add the feature to the layer for most layer-related events to │ │ │ │ │ + * happen. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * style - {String | Object} Named render intent or full symbolizer object. │ │ │ │ │ */ │ │ │ │ │ - defaultDblClick: function(evt) { │ │ │ │ │ - this.map.zoomTo(this.map.zoom + 1, evt.xy); │ │ │ │ │ + drawFeature: function(feature, style) { │ │ │ │ │ + // don't try to draw the feature with the renderer if the layer is not │ │ │ │ │ + // drawn itself │ │ │ │ │ + if (!this.drawn) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (typeof style != "object") { │ │ │ │ │ + if (!style && feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ + style = "delete"; │ │ │ │ │ + } │ │ │ │ │ + var renderIntent = style || feature.renderIntent; │ │ │ │ │ + style = feature.style || this.style; │ │ │ │ │ + if (!style) { │ │ │ │ │ + style = this.styleMap.createSymbolizer(feature, renderIntent); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var drawn = this.renderer.drawFeature(feature, style); │ │ │ │ │ + //TODO remove the check for null when we get rid of Renderer.SVG │ │ │ │ │ + if (drawn === false || drawn === null) { │ │ │ │ │ + this.unrenderedFeatures[feature.id] = feature; │ │ │ │ │ + } else { │ │ │ │ │ + delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: defaultDblRightClick │ │ │ │ │ - * │ │ │ │ │ + * Method: eraseFeatures │ │ │ │ │ + * Erase features from the layer. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ */ │ │ │ │ │ - defaultDblRightClick: function(evt) { │ │ │ │ │ - this.map.zoomTo(this.map.zoom - 1, evt.xy); │ │ │ │ │ + eraseFeatures: function(features) { │ │ │ │ │ + this.renderer.eraseFeatures(features); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: wheelChange │ │ │ │ │ + * Method: getFeatureFromEvent │ │ │ │ │ + * Given an event, return a feature if the event occurred over one. │ │ │ │ │ + * Otherwise, return null. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * deltaZ - {Integer} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A feature if one was under the event. │ │ │ │ │ */ │ │ │ │ │ - wheelChange: function(evt, deltaZ) { │ │ │ │ │ - if (!this.map.fractionalZoom) { │ │ │ │ │ - deltaZ = Math.round(deltaZ); │ │ │ │ │ + getFeatureFromEvent: function(evt) { │ │ │ │ │ + if (!this.renderer) { │ │ │ │ │ + throw new Error('getFeatureFromEvent called on layer with no ' + │ │ │ │ │ + 'renderer. This usually means you destroyed a ' + │ │ │ │ │ + 'layer, but not some handler which is associated ' + │ │ │ │ │ + 'with it.'); │ │ │ │ │ } │ │ │ │ │ - var currentZoom = this.map.getZoom(), │ │ │ │ │ - newZoom = currentZoom + deltaZ; │ │ │ │ │ - newZoom = Math.max(newZoom, 0); │ │ │ │ │ - newZoom = Math.min(newZoom, this.map.getNumZoomLevels()); │ │ │ │ │ - if (newZoom === currentZoom) { │ │ │ │ │ - return; │ │ │ │ │ + var feature = null; │ │ │ │ │ + var featureId = this.renderer.getFeatureIdFromEvent(evt); │ │ │ │ │ + if (featureId) { │ │ │ │ │ + if (typeof featureId === "string") { │ │ │ │ │ + feature = this.getFeatureById(featureId); │ │ │ │ │ + } else { │ │ │ │ │ + feature = featureId; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.map.zoomTo(newZoom, evt.xy); │ │ │ │ │ + return feature; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: wheelUp │ │ │ │ │ - * User spun scroll wheel up │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getFeatureBy │ │ │ │ │ + * Given a property value, return the feature if it exists in the features array │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * delta - {Integer} │ │ │ │ │ + * property - {String} │ │ │ │ │ + * value - {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A feature corresponding to the given │ │ │ │ │ + * property value or null if there is no such feature. │ │ │ │ │ */ │ │ │ │ │ - wheelUp: function(evt, delta) { │ │ │ │ │ - this.wheelChange(evt, delta || 1); │ │ │ │ │ + getFeatureBy: function(property, value) { │ │ │ │ │ + //TBD - would it be more efficient to use a hash for this.features? │ │ │ │ │ + var feature = null; │ │ │ │ │ + for (var i = 0, len = this.features.length; i < len; ++i) { │ │ │ │ │ + if (this.features[i][property] == value) { │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return feature; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: wheelDown │ │ │ │ │ - * User spun scroll wheel down │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getFeatureById │ │ │ │ │ + * Given a feature id, return the feature if it exists in the features array │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * delta - {Integer} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A feature corresponding to the given │ │ │ │ │ + * featureId or null if there is no such feature. │ │ │ │ │ */ │ │ │ │ │ - wheelDown: function(evt, delta) { │ │ │ │ │ - this.wheelChange(evt, delta || -1); │ │ │ │ │ + getFeatureById: function(featureId) { │ │ │ │ │ + return this.getFeatureBy('id', featureId); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: disableZoomBox │ │ │ │ │ + * APIMethod: getFeatureByFid │ │ │ │ │ + * Given a feature fid, return the feature if it exists in the features array │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureFid - {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A feature corresponding to the given │ │ │ │ │ + * featureFid or null if there is no such feature. │ │ │ │ │ */ │ │ │ │ │ - disableZoomBox: function() { │ │ │ │ │ - this.zoomBoxEnabled = false; │ │ │ │ │ - this.zoomBox.deactivate(); │ │ │ │ │ + getFeatureByFid: function(featureFid) { │ │ │ │ │ + return this.getFeatureBy('fid', featureFid); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: enableZoomBox │ │ │ │ │ + * APIMethod: getFeaturesByAttribute │ │ │ │ │ + * Returns an array of features that have the given attribute key set to the │ │ │ │ │ + * given value. Comparison of attribute values takes care of datatypes, e.g. │ │ │ │ │ + * the string '1234' is not equal to the number 1234. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * attrName - {String} │ │ │ │ │ + * attrValue - {Mixed} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * Array({<OpenLayers.Feature.Vector>}) An array of features that have the │ │ │ │ │ + * passed named attribute set to the given value. │ │ │ │ │ */ │ │ │ │ │ - enableZoomBox: function() { │ │ │ │ │ - this.zoomBoxEnabled = true; │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.zoomBox.activate(); │ │ │ │ │ + getFeaturesByAttribute: function(attrName, attrValue) { │ │ │ │ │ + var i, │ │ │ │ │ + feature, │ │ │ │ │ + len = this.features.length, │ │ │ │ │ + foundFeatures = []; │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + if (feature && feature.attributes) { │ │ │ │ │ + if (feature.attributes[attrName] === attrValue) { │ │ │ │ │ + foundFeatures.push(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return foundFeatures; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: disableZoomWheel │ │ │ │ │ - */ │ │ │ │ │ + * Unselect the selected features │ │ │ │ │ + * i.e. clears the featureSelection array │ │ │ │ │ + * change the style back │ │ │ │ │ + clearSelection: function() { │ │ │ │ │ │ │ │ │ │ - disableZoomWheel: function() { │ │ │ │ │ - this.zoomWheelEnabled = false; │ │ │ │ │ - this.handlers.wheel.deactivate(); │ │ │ │ │ + var vectorLayer = this.map.vectorLayer; │ │ │ │ │ + for (var i = 0; i < this.map.featureSelection.length; i++) { │ │ │ │ │ + var featureSelection = this.map.featureSelection[i]; │ │ │ │ │ + vectorLayer.drawFeature(featureSelection, vectorLayer.style); │ │ │ │ │ + } │ │ │ │ │ + this.map.featureSelection = []; │ │ │ │ │ }, │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: enableZoomWheel │ │ │ │ │ + * APIMethod: onFeatureInsert │ │ │ │ │ + * method called after a feature is inserted. │ │ │ │ │ + * Does nothing by default. Override this if you │ │ │ │ │ + * need to do something on feature updates. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ + onFeatureInsert: function(feature) {}, │ │ │ │ │ │ │ │ │ │ - enableZoomWheel: function() { │ │ │ │ │ - this.zoomWheelEnabled = true; │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.handlers.wheel.activate(); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: preFeatureInsert │ │ │ │ │ + * method called before a feature is inserted. │ │ │ │ │ + * Does nothing by default. Override this if you │ │ │ │ │ + * need to do something when features are first added to the │ │ │ │ │ + * layer, but before they are drawn, such as adjust the style. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + preFeatureInsert: function(feature) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getDataExtent │ │ │ │ │ + * Calculates the max extent which includes all of the features. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} or null if the layer has no features with │ │ │ │ │ + * geometries. │ │ │ │ │ + */ │ │ │ │ │ + getDataExtent: function() { │ │ │ │ │ + var maxExtent = null; │ │ │ │ │ + var features = this.features; │ │ │ │ │ + if (features && (features.length > 0)) { │ │ │ │ │ + var geometry = null; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + geometry = features[i].geometry; │ │ │ │ │ + if (geometry) { │ │ │ │ │ + if (maxExtent === null) { │ │ │ │ │ + maxExtent = new OpenLayers.Bounds(); │ │ │ │ │ + } │ │ │ │ │ + maxExtent.extend(geometry.getBounds()); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return maxExtent; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Navigation" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Vector" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/NavToolbar.js │ │ │ │ │ + OpenLayers/Control/Snapping.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Control/Panel.js │ │ │ │ │ - * @requires OpenLayers/Control/Navigation.js │ │ │ │ │ - * @requires OpenLayers/Control/ZoomBox.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.NavToolbar │ │ │ │ │ - * This Toolbar is an alternative to the Navigation control that displays │ │ │ │ │ - * the state of the control, and provides a UI for changing state to │ │ │ │ │ - * use the zoomBox via a Panel control. │ │ │ │ │ + * Class: OpenLayers.Control.Snapping │ │ │ │ │ + * Acts as a snapping agent while editing vector features. │ │ │ │ │ * │ │ │ │ │ - * If you wish to change the properties of the Navigation control used │ │ │ │ │ - * in the NavToolbar, see: │ │ │ │ │ - * http://trac.openlayers.org/wiki/Toolbars#SubclassingNavToolbar │ │ │ │ │ - * │ │ │ │ │ - * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control.Panel> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.NavToolbar = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ +OpenLayers.Control.Snapping = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ + * beforesnap - Triggered before a snap occurs. Listeners receive an │ │ │ │ │ + * event object with *point*, *x*, *y*, *distance*, *layer*, and │ │ │ │ │ + * *snapType* properties. The point property will be original point │ │ │ │ │ + * geometry considered for snapping. The x and y properties represent │ │ │ │ │ + * coordinates the point will receive. The distance is the distance │ │ │ │ │ + * of the snap. The layer is the target layer. The snapType property │ │ │ │ │ + * will be one of "node", "vertex", or "edge". Return false to stop │ │ │ │ │ + * snapping from occurring. │ │ │ │ │ + * snap - Triggered when a snap occurs. Listeners receive an event with │ │ │ │ │ + * *point*, *snapType*, *layer*, and *distance* properties. The point │ │ │ │ │ + * will be the location snapped to. The snapType will be one of "node", │ │ │ │ │ + * "vertex", or "edge". The layer will be the target layer. The │ │ │ │ │ + * distance will be the distance of the snap in map units. │ │ │ │ │ + * unsnap - Triggered when a vertex is unsnapped. Listeners receive an │ │ │ │ │ + * event with a *point* property. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.NavToolbar │ │ │ │ │ - * Add our two mousedefaults controls. │ │ │ │ │ + * CONSTANT: DEFAULTS │ │ │ │ │ + * Default target properties. │ │ │ │ │ + */ │ │ │ │ │ + DEFAULTS: { │ │ │ │ │ + tolerance: 10, │ │ │ │ │ + node: true, │ │ │ │ │ + edge: true, │ │ │ │ │ + vertex: true │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: greedy │ │ │ │ │ + * {Boolean} Snap to closest feature in first layer with an eligible │ │ │ │ │ + * feature. Default is true. │ │ │ │ │ + */ │ │ │ │ │ + greedy: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: precedence │ │ │ │ │ + * {Array} List representing precedence of different snapping types. │ │ │ │ │ + * Default is "node", "vertex", "edge". │ │ │ │ │ + */ │ │ │ │ │ + precedence: ["node", "vertex", "edge"], │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: resolution │ │ │ │ │ + * {Float} The map resolution for the previously considered snap. │ │ │ │ │ + */ │ │ │ │ │ + resolution: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: geoToleranceCache │ │ │ │ │ + * {Object} A cache of geo-tolerances. Tolerance values (in map units) are │ │ │ │ │ + * calculated when the map resolution changes. │ │ │ │ │ + */ │ │ │ │ │ + geoToleranceCache: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} The current editable layer. Set at │ │ │ │ │ + * construction or after construction with <setLayer>. │ │ │ │ │ + */ │ │ │ │ │ + layer: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: feature │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} The current editable feature. │ │ │ │ │ + */ │ │ │ │ │ + feature: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: point │ │ │ │ │ + * {<OpenLayers.Geometry.Point>} The currently snapped vertex. │ │ │ │ │ + */ │ │ │ │ │ + point: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.Snapping │ │ │ │ │ + * Creates a new snapping control. A control is constructed with an editable │ │ │ │ │ + * layer and a set of configuration objects for target layers. While the │ │ │ │ │ + * control is active, dragging vertices while drawing new features or │ │ │ │ │ + * modifying existing features on the editable layer will engage │ │ │ │ │ + * snapping to features on the target layers. Whether a vertex snaps to │ │ │ │ │ + * a feature on a target layer depends on the target layer configuration. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be used │ │ │ │ │ - * to extend the control. │ │ │ │ │ + * options - {Object} An object containing all configuration properties for │ │ │ │ │ + * the control. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * layer - {<OpenLayers.Layer.Vector>} The editable layer. Features from this │ │ │ │ │ + * layer that are digitized or modified may have vertices snapped to │ │ │ │ │ + * features from any of the target layers. │ │ │ │ │ + * targets - {Array(Object | OpenLayers.Layer.Vector)} A list of objects for │ │ │ │ │ + * configuring target layers. See valid properties of the target │ │ │ │ │ + * objects below. If the items in the targets list are vector layers │ │ │ │ │ + * (instead of configuration objects), the defaults from the <defaults> │ │ │ │ │ + * property will apply. The editable layer itself may be a target │ │ │ │ │ + * layer, allowing newly created or edited features to be snapped to │ │ │ │ │ + * existing features from the same layer. If no targets are provided │ │ │ │ │ + * the layer given in the constructor (as <layer>) will become the │ │ │ │ │ + * initial target. │ │ │ │ │ + * defaults - {Object} An object with default properties to be applied │ │ │ │ │ + * to all target objects. │ │ │ │ │ + * greedy - {Boolean} Snap to closest feature in first target layer that │ │ │ │ │ + * applies. Default is true. If false, all features in all target │ │ │ │ │ + * layers will be checked and the closest feature in all target layers │ │ │ │ │ + * will be chosen. The greedy property determines if the order of the │ │ │ │ │ + * target layers is significant. By default, the order of the target │ │ │ │ │ + * layers is significant where layers earlier in the target layer list │ │ │ │ │ + * have precedence over layers later in the list. Within a single │ │ │ │ │ + * layer, the closest feature is always chosen for snapping. This │ │ │ │ │ + * property only determines whether the search for a closer feature │ │ │ │ │ + * continues after an eligible feature is found in a target layer. │ │ │ │ │ + * │ │ │ │ │ + * Valid target properties: │ │ │ │ │ + * layer - {<OpenLayers.Layer.Vector>} A target layer. Features from this │ │ │ │ │ + * layer will be eligible to act as snapping target for the editable │ │ │ │ │ + * layer. │ │ │ │ │ + * tolerance - {Float} The distance (in pixels) at which snapping may occur. │ │ │ │ │ + * Default is 10. │ │ │ │ │ + * node - {Boolean} Snap to nodes (first or last point in a geometry) in │ │ │ │ │ + * target layer. Default is true. │ │ │ │ │ + * nodeTolerance - {Float} Optional distance at which snapping may occur │ │ │ │ │ + * for nodes specifically. If none is provided, <tolerance> will be │ │ │ │ │ + * used. │ │ │ │ │ + * vertex - {Boolean} Snap to vertices in target layer. Default is true. │ │ │ │ │ + * vertexTolerance - {Float} Optional distance at which snapping may occur │ │ │ │ │ + * for vertices specifically. If none is provided, <tolerance> will be │ │ │ │ │ + * used. │ │ │ │ │ + * edge - {Boolean} Snap to edges in target layer. Default is true. │ │ │ │ │ + * edgeTolerance - {Float} Optional distance at which snapping may occur │ │ │ │ │ + * for edges specifically. If none is provided, <tolerance> will be │ │ │ │ │ + * used. │ │ │ │ │ + * filter - {<OpenLayers.Filter>} Optional filter to evaluate to determine if │ │ │ │ │ + * feature is eligible for snapping. If filter evaluates to true for a │ │ │ │ │ + * target feature a vertex may be snapped to the feature. │ │ │ │ │ + * minResolution - {Number} If a minResolution is provided, snapping to this │ │ │ │ │ + * target will only be considered if the map resolution is greater than │ │ │ │ │ + * or equal to this value (the minResolution is inclusive). Default is │ │ │ │ │ + * no minimum resolution limit. │ │ │ │ │ + * maxResolution - {Number} If a maxResolution is provided, snapping to this │ │ │ │ │ + * target will only be considered if the map resolution is strictly │ │ │ │ │ + * less than this value (the maxResolution is exclusive). Default is │ │ │ │ │ + * no maximum resolution limit. │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.addControls([ │ │ │ │ │ - new OpenLayers.Control.Navigation(), │ │ │ │ │ - new OpenLayers.Control.ZoomBox() │ │ │ │ │ - ]); │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.options = options || {}; // TODO: this could be done by the super │ │ │ │ │ + │ │ │ │ │ + // set the editable layer if provided │ │ │ │ │ + if (this.options.layer) { │ │ │ │ │ + this.setLayer(this.options.layer); │ │ │ │ │ + } │ │ │ │ │ + // configure target layers │ │ │ │ │ + var defaults = OpenLayers.Util.extend({}, this.options.defaults); │ │ │ │ │ + this.defaults = OpenLayers.Util.applyDefaults(defaults, this.DEFAULTS); │ │ │ │ │ + this.setTargets(this.options.targets); │ │ │ │ │ + if (this.targets.length === 0 && this.layer) { │ │ │ │ │ + this.addTargetLayer(this.layer); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.geoToleranceCache = {}; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * calls the default draw, and then activates mouse defaults. │ │ │ │ │ + * APIMethod: setLayer │ │ │ │ │ + * Set the editable layer. Call the setLayer method if the editable layer │ │ │ │ │ + * changes and the same control should be used on a new editable layer. │ │ │ │ │ + * If the control is already active, it will be active after the new │ │ │ │ │ + * layer is set. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer.Vector>} The new editable layer. │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - var div = OpenLayers.Control.Panel.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (this.defaultControl === null) { │ │ │ │ │ - this.defaultControl = this.controls[0]; │ │ │ │ │ + setLayer: function(layer) { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + this.activate(); │ │ │ │ │ + } else { │ │ │ │ │ + this.layer = layer; │ │ │ │ │ } │ │ │ │ │ - return div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.NavToolbar" │ │ │ │ │ + /** │ │ │ │ │ + * Method: setTargets │ │ │ │ │ + * Set the targets for the snapping agent. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * targets - {Array} An array of target configs or target layers. │ │ │ │ │ + */ │ │ │ │ │ + setTargets: function(targets) { │ │ │ │ │ + this.targets = []; │ │ │ │ │ + if (targets && targets.length) { │ │ │ │ │ + var target; │ │ │ │ │ + for (var i = 0, len = targets.length; i < len; ++i) { │ │ │ │ │ + target = targets[i]; │ │ │ │ │ + if (target instanceof OpenLayers.Layer.Vector) { │ │ │ │ │ + this.addTargetLayer(target); │ │ │ │ │ + } else { │ │ │ │ │ + this.addTarget(target); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: addTargetLayer │ │ │ │ │ + * Add a target layer with the default target config. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer.Vector>} A target layer. │ │ │ │ │ + */ │ │ │ │ │ + addTargetLayer: function(layer) { │ │ │ │ │ + this.addTarget({ │ │ │ │ │ + layer: layer │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: addTarget │ │ │ │ │ + * Add a configured target layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * target - {Object} A target config. │ │ │ │ │ + */ │ │ │ │ │ + addTarget: function(target) { │ │ │ │ │ + target = OpenLayers.Util.applyDefaults(target, this.defaults); │ │ │ │ │ + target.nodeTolerance = target.nodeTolerance || target.tolerance; │ │ │ │ │ + target.vertexTolerance = target.vertexTolerance || target.tolerance; │ │ │ │ │ + target.edgeTolerance = target.edgeTolerance || target.tolerance; │ │ │ │ │ + this.targets.push(target); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeTargetLayer │ │ │ │ │ + * Remove a target layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer.Vector>} The target layer to remove. │ │ │ │ │ + */ │ │ │ │ │ + removeTargetLayer: function(layer) { │ │ │ │ │ + var target; │ │ │ │ │ + for (var i = this.targets.length - 1; i >= 0; --i) { │ │ │ │ │ + target = this.targets[i]; │ │ │ │ │ + if (target.layer === layer) { │ │ │ │ │ + this.removeTarget(target); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeTarget │ │ │ │ │ + * Remove a target. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * target - {Object} A target config. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} The targets array. │ │ │ │ │ + */ │ │ │ │ │ + removeTarget: function(target) { │ │ │ │ │ + return OpenLayers.Util.removeItem(this.targets, target); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Activate the control. Activating the control registers listeners for │ │ │ │ │ + * editing related events so that during feature creation and │ │ │ │ │ + * modification, moving vertices will trigger snapping. │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Control.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + if (this.layer && this.layer.events) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + sketchstarted: this.onSketchModified, │ │ │ │ │ + sketchmodified: this.onSketchModified, │ │ │ │ │ + vertexmodified: this.onVertexModified, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the control. Deactivating the control unregisters listeners │ │ │ │ │ + * so feature editing may proceed without engaging the snapping agent. │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Control.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + if (this.layer && this.layer.events) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + sketchstarted: this.onSketchModified, │ │ │ │ │ + sketchmodified: this.onSketchModified, │ │ │ │ │ + vertexmodified: this.onVertexModified, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.feature = null; │ │ │ │ │ + this.point = null; │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: onSketchModified │ │ │ │ │ + * Registered as a listener for the sketchmodified event on the editable │ │ │ │ │ + * layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * event - {Object} The sketch modified event. │ │ │ │ │ + */ │ │ │ │ │ + onSketchModified: function(event) { │ │ │ │ │ + this.feature = event.feature; │ │ │ │ │ + this.considerSnapping(event.vertex, event.vertex); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: onVertexModified │ │ │ │ │ + * Registered as a listener for the vertexmodified event on the editable │ │ │ │ │ + * layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * event - {Object} The vertex modified event. │ │ │ │ │ + */ │ │ │ │ │ + onVertexModified: function(event) { │ │ │ │ │ + this.feature = event.feature; │ │ │ │ │ + var loc = this.layer.map.getLonLatFromViewPortPx(event.pixel); │ │ │ │ │ + this.considerSnapping( │ │ │ │ │ + event.vertex, new OpenLayers.Geometry.Point(loc.lon, loc.lat) │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: considerSnapping │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} The vertex to be snapped (or │ │ │ │ │ + * unsnapped). │ │ │ │ │ + * loc - {<OpenLayers.Geometry.Point>} The location of the mouse in map │ │ │ │ │ + * coords. │ │ │ │ │ + */ │ │ │ │ │ + considerSnapping: function(point, loc) { │ │ │ │ │ + var best = { │ │ │ │ │ + rank: Number.POSITIVE_INFINITY, │ │ │ │ │ + dist: Number.POSITIVE_INFINITY, │ │ │ │ │ + x: null, │ │ │ │ │ + y: null │ │ │ │ │ + }; │ │ │ │ │ + var snapped = false; │ │ │ │ │ + var result, target; │ │ │ │ │ + for (var i = 0, len = this.targets.length; i < len; ++i) { │ │ │ │ │ + target = this.targets[i]; │ │ │ │ │ + result = this.testTarget(target, loc); │ │ │ │ │ + if (result) { │ │ │ │ │ + if (this.greedy) { │ │ │ │ │ + best = result; │ │ │ │ │ + best.target = target; │ │ │ │ │ + snapped = true; │ │ │ │ │ + break; │ │ │ │ │ + } else { │ │ │ │ │ + if ((result.rank < best.rank) || │ │ │ │ │ + (result.rank === best.rank && result.dist < best.dist)) { │ │ │ │ │ + best = result; │ │ │ │ │ + best.target = target; │ │ │ │ │ + snapped = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (snapped) { │ │ │ │ │ + var proceed = this.events.triggerEvent("beforesnap", { │ │ │ │ │ + point: point, │ │ │ │ │ + x: best.x, │ │ │ │ │ + y: best.y, │ │ │ │ │ + distance: best.dist, │ │ │ │ │ + layer: best.target.layer, │ │ │ │ │ + snapType: this.precedence[best.rank] │ │ │ │ │ + }); │ │ │ │ │ + if (proceed !== false) { │ │ │ │ │ + point.x = best.x; │ │ │ │ │ + point.y = best.y; │ │ │ │ │ + this.point = point; │ │ │ │ │ + this.events.triggerEvent("snap", { │ │ │ │ │ + point: point, │ │ │ │ │ + snapType: this.precedence[best.rank], │ │ │ │ │ + layer: best.target.layer, │ │ │ │ │ + distance: best.dist │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + snapped = false; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.point && !snapped) { │ │ │ │ │ + point.x = loc.x; │ │ │ │ │ + point.y = loc.y; │ │ │ │ │ + this.point = null; │ │ │ │ │ + this.events.triggerEvent("unsnap", { │ │ │ │ │ + point: point │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: testTarget │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * target - {Object} Object with target layer configuration. │ │ │ │ │ + * loc - {<OpenLayers.Geometry.Point>} The location of the mouse in map │ │ │ │ │ + * coords. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} A result object with rank, dist, x, and y properties. │ │ │ │ │ + * Returns null if candidate is not eligible for snapping. │ │ │ │ │ + */ │ │ │ │ │ + testTarget: function(target, loc) { │ │ │ │ │ + var resolution = this.layer.map.getResolution(); │ │ │ │ │ + if ("minResolution" in target) { │ │ │ │ │ + if (resolution < target.minResolution) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if ("maxResolution" in target) { │ │ │ │ │ + if (resolution >= target.maxResolution) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var tolerance = { │ │ │ │ │ + node: this.getGeoTolerance(target.nodeTolerance, resolution), │ │ │ │ │ + vertex: this.getGeoTolerance(target.vertexTolerance, resolution), │ │ │ │ │ + edge: this.getGeoTolerance(target.edgeTolerance, resolution) │ │ │ │ │ + }; │ │ │ │ │ + // this could be cached if we don't support setting tolerance values directly │ │ │ │ │ + var maxTolerance = Math.max( │ │ │ │ │ + tolerance.node, tolerance.vertex, tolerance.edge │ │ │ │ │ + ); │ │ │ │ │ + var result = { │ │ │ │ │ + rank: Number.POSITIVE_INFINITY, │ │ │ │ │ + dist: Number.POSITIVE_INFINITY │ │ │ │ │ + }; │ │ │ │ │ + var eligible = false; │ │ │ │ │ + var features = target.layer.features; │ │ │ │ │ + var feature, type, vertices, vertex, closest, dist, found; │ │ │ │ │ + var numTypes = this.precedence.length; │ │ │ │ │ + var ll = new OpenLayers.LonLat(loc.x, loc.y); │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (feature !== this.feature && !feature._sketch && │ │ │ │ │ + feature.state !== OpenLayers.State.DELETE && │ │ │ │ │ + (!target.filter || target.filter.evaluate(feature))) { │ │ │ │ │ + if (feature.atPoint(ll, maxTolerance, maxTolerance)) { │ │ │ │ │ + for (var j = 0, stop = Math.min(result.rank + 1, numTypes); j < stop; ++j) { │ │ │ │ │ + type = this.precedence[j]; │ │ │ │ │ + if (target[type]) { │ │ │ │ │ + if (type === "edge") { │ │ │ │ │ + closest = feature.geometry.distanceTo(loc, { │ │ │ │ │ + details: true │ │ │ │ │ + }); │ │ │ │ │ + dist = closest.distance; │ │ │ │ │ + if (dist <= tolerance[type] && dist < result.dist) { │ │ │ │ │ + result = { │ │ │ │ │ + rank: j, │ │ │ │ │ + dist: dist, │ │ │ │ │ + x: closest.x0, │ │ │ │ │ + y: closest.y0 // closest coords on feature │ │ │ │ │ + }; │ │ │ │ │ + eligible = true; │ │ │ │ │ + // don't look for lower precedence types for this feature │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // look for nodes or vertices │ │ │ │ │ + vertices = feature.geometry.getVertices(type === "node"); │ │ │ │ │ + found = false; │ │ │ │ │ + for (var k = 0, klen = vertices.length; k < klen; ++k) { │ │ │ │ │ + vertex = vertices[k]; │ │ │ │ │ + dist = vertex.distanceTo(loc); │ │ │ │ │ + if (dist <= tolerance[type] && │ │ │ │ │ + (j < result.rank || (j === result.rank && dist < result.dist))) { │ │ │ │ │ + result = { │ │ │ │ │ + rank: j, │ │ │ │ │ + dist: dist, │ │ │ │ │ + x: vertex.x, │ │ │ │ │ + y: vertex.y │ │ │ │ │ + }; │ │ │ │ │ + eligible = true; │ │ │ │ │ + found = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (found) { │ │ │ │ │ + // don't look for lower precedence types for this feature │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return eligible ? result : null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getGeoTolerance │ │ │ │ │ + * Calculate a tolerance in map units given a tolerance in pixels. This │ │ │ │ │ + * takes advantage of the <geoToleranceCache> when the map resolution │ │ │ │ │ + * has not changed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * tolerance - {Number} A tolerance value in pixels. │ │ │ │ │ + * resolution - {Number} Map resolution. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number} A tolerance value in map units. │ │ │ │ │ + */ │ │ │ │ │ + getGeoTolerance: function(tolerance, resolution) { │ │ │ │ │ + if (resolution !== this.resolution) { │ │ │ │ │ + this.resolution = resolution; │ │ │ │ │ + this.geoToleranceCache = {}; │ │ │ │ │ + } │ │ │ │ │ + var geoTolerance = this.geoToleranceCache[tolerance]; │ │ │ │ │ + if (geoTolerance === undefined) { │ │ │ │ │ + geoTolerance = tolerance * resolution; │ │ │ │ │ + this.geoToleranceCache[tolerance] = geoTolerance; │ │ │ │ │ + } │ │ │ │ │ + return geoTolerance; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Clean up the control. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.deactivate(); // TODO: this should be handled by the super │ │ │ │ │ + } │ │ │ │ │ + delete this.layer; │ │ │ │ │ + delete this.targets; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.call(this); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Snapping" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/CacheRead.js │ │ │ │ │ + OpenLayers/Control/CacheWrite.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Request.js │ │ │ │ │ + * @requires OpenLayers/Console.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.CacheRead │ │ │ │ │ - * A control for using image tiles cached with <OpenLayers.Control.CacheWrite> │ │ │ │ │ - * from the browser's local storage. │ │ │ │ │ + * Class: OpenLayers.Control.CacheWrite │ │ │ │ │ + * A control for caching image tiles in the browser's local storage. The │ │ │ │ │ + * <OpenLayers.Control.CacheRead> control is used to fetch and use the cached │ │ │ │ │ + * tile images. │ │ │ │ │ + * │ │ │ │ │ + * Note: Before using this control on any layer that is not your own, make sure │ │ │ │ │ + * that the terms of service of the tile provider allow local storage of tiles. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.CacheRead = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Control.CacheWrite = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: fetchEvent │ │ │ │ │ - * {String} The layer event to listen to for replacing remote resource tile │ │ │ │ │ - * URLs with cached data URIs. Supported values are "tileerror" (try │ │ │ │ │ - * remote first, fall back to cached) and "tileloadstart" (try cache │ │ │ │ │ - * first, fall back to remote). Default is "tileloadstart". │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ * │ │ │ │ │ - * Note that "tileerror" will not work for CORS enabled images (see │ │ │ │ │ - * https://developer.mozilla.org/en/CORS_Enabled_Image), i.e. layers │ │ │ │ │ - * configured with a <OpenLayers.Tile.Image.crossOriginKeyword> in │ │ │ │ │ - * <OpenLayers.Layer.Grid.tileOptions>. │ │ │ │ │ + * To register events in the constructor, configure <eventListeners>. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ + * cachefull - Triggered when the cache is full. Listeners receive an │ │ │ │ │ + * object with a tile property as first argument. The tile references │ │ │ │ │ + * the tile that couldn't be cached. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: eventListeners │ │ │ │ │ + * {Object} Object with event listeners, keyed by event name. An optional │ │ │ │ │ + * scope property defines the scope that listeners will be executed in. │ │ │ │ │ */ │ │ │ │ │ - fetchEvent: "tileloadstart", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIProperty: layers │ │ │ │ │ - * {Array(<OpenLayers.Layer.Grid>)}. Optional. If provided, only these │ │ │ │ │ - * layers will receive tiles from the cache. │ │ │ │ │ + * {Array(<OpenLayers.Layer.Grid>)}. Optional. If provided, caching │ │ │ │ │ + * will be enabled for these layers only, otherwise for all cacheable │ │ │ │ │ + * layers. │ │ │ │ │ */ │ │ │ │ │ layers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ + * APIProperty: imageFormat │ │ │ │ │ + * {String} The image format used for caching. The default is "image/png". │ │ │ │ │ + * Supported formats depend on the user agent. If an unsupported │ │ │ │ │ + * <imageFormat> is provided, "image/png" will be used. For aerial │ │ │ │ │ + * imagery, "image/jpeg" is recommended. │ │ │ │ │ */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ + imageFormat: "image/png", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.CacheRead │ │ │ │ │ + * Property: quotaRegEx │ │ │ │ │ + * {RegExp} │ │ │ │ │ + */ │ │ │ │ │ + quotaRegEx: (/quota/i), │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.CacheWrite │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Object with API properties for this control │ │ │ │ │ + * options - {Object} Object with API properties for this control. │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Method: setMap │ │ │ │ │ * Set the map property for the control. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ @@ -67434,54 +67349,110 @@ │ │ │ │ │ * will be cached. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * evt - {Object} Object with a layer property referencing an │ │ │ │ │ * <OpenLayers.Layer> instance │ │ │ │ │ */ │ │ │ │ │ addLayer: function(evt) { │ │ │ │ │ - evt.layer.events.register(this.fetchEvent, this, this.fetch); │ │ │ │ │ + evt.layer.events.on({ │ │ │ │ │ + tileloadstart: this.makeSameOrigin, │ │ │ │ │ + tileloaded: this.onTileLoaded, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Method: removeLayer │ │ │ │ │ * Removes a layer from the control. Once removed, tiles requested for this │ │ │ │ │ * layer will no longer be cached. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * evt - {Object} Object with a layer property referencing an │ │ │ │ │ * <OpenLayers.Layer> instance │ │ │ │ │ */ │ │ │ │ │ removeLayer: function(evt) { │ │ │ │ │ - evt.layer.events.unregister(this.fetchEvent, this, this.fetch); │ │ │ │ │ + evt.layer.events.un({ │ │ │ │ │ + tileloadstart: this.makeSameOrigin, │ │ │ │ │ + tileloaded: this.onTileLoaded, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: fetch │ │ │ │ │ - * Listener to the <fetchEvent> event. Replaces a tile's url with a data │ │ │ │ │ - * URI from the cache. │ │ │ │ │ + * Method: makeSameOrigin │ │ │ │ │ + * If the tile does not have CORS image loading enabled and is from a │ │ │ │ │ + * different origin, use OpenLayers.ProxyHost to make it a same origin url. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} Event object with a tile property. │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - fetch: function(evt) { │ │ │ │ │ - if (this.active && window.localStorage && │ │ │ │ │ - evt.tile instanceof OpenLayers.Tile.Image) { │ │ │ │ │ - var tile = evt.tile, │ │ │ │ │ - url = tile.url; │ │ │ │ │ - // deal with modified tile urls when both CacheWrite and CacheRead │ │ │ │ │ - // are active │ │ │ │ │ - if (!tile.layer.crossOriginKeyword && OpenLayers.ProxyHost && │ │ │ │ │ - url.indexOf(OpenLayers.ProxyHost) === 0) { │ │ │ │ │ - url = OpenLayers.Control.CacheWrite.urlMap[url]; │ │ │ │ │ + makeSameOrigin: function(evt) { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + var tile = evt.tile; │ │ │ │ │ + if (tile instanceof OpenLayers.Tile.Image && │ │ │ │ │ + !tile.crossOriginKeyword && │ │ │ │ │ + tile.url.substr(0, 5) !== "data:") { │ │ │ │ │ + var sameOriginUrl = OpenLayers.Request.makeSameOrigin( │ │ │ │ │ + tile.url, OpenLayers.ProxyHost │ │ │ │ │ + ); │ │ │ │ │ + OpenLayers.Control.CacheWrite.urlMap[sameOriginUrl] = tile.url; │ │ │ │ │ + tile.url = sameOriginUrl; │ │ │ │ │ } │ │ │ │ │ - var dataURI = window.localStorage.getItem("olCache_" + url); │ │ │ │ │ - if (dataURI) { │ │ │ │ │ - tile.url = dataURI; │ │ │ │ │ - if (evt.type === "tileerror") { │ │ │ │ │ - tile.setImgSrc(dataURI); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: onTileLoaded │ │ │ │ │ + * Decides whether a tile can be cached and calls the cache method. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + onTileLoaded: function(evt) { │ │ │ │ │ + if (this.active && !evt.aborted && │ │ │ │ │ + evt.tile instanceof OpenLayers.Tile.Image && │ │ │ │ │ + evt.tile.url.substr(0, 5) !== 'data:') { │ │ │ │ │ + this.cache({ │ │ │ │ │ + tile: evt.tile │ │ │ │ │ + }); │ │ │ │ │ + delete OpenLayers.Control.CacheWrite.urlMap[evt.tile.url]; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: cache │ │ │ │ │ + * Adds a tile to the cache. When the cache is full, the "cachefull" event │ │ │ │ │ + * is triggered. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} Object with a tile property, tile being the │ │ │ │ │ + * <OpenLayers.Tile.Image> with the data to add to the cache │ │ │ │ │ + */ │ │ │ │ │ + cache: function(obj) { │ │ │ │ │ + if (window.localStorage) { │ │ │ │ │ + var tile = obj.tile; │ │ │ │ │ + try { │ │ │ │ │ + var canvasContext = tile.getCanvasContext(); │ │ │ │ │ + if (canvasContext) { │ │ │ │ │ + var urlMap = OpenLayers.Control.CacheWrite.urlMap; │ │ │ │ │ + var url = urlMap[tile.url] || tile.url; │ │ │ │ │ + window.localStorage.setItem( │ │ │ │ │ + "olCache_" + url, │ │ │ │ │ + canvasContext.canvas.toDataURL(this.imageFormat) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } catch (e) { │ │ │ │ │ + // local storage full or CORS violation │ │ │ │ │ + var reason = e.name || e.message; │ │ │ │ │ + if (reason && this.quotaRegEx.test(reason)) { │ │ │ │ │ + this.events.triggerEvent("cachefull", { │ │ │ │ │ + tile: tile │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Console.error(e.toString()); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Method: destroy │ │ │ │ │ @@ -67504,3950 +67475,1332 @@ │ │ │ │ │ removeLayer: this.removeLayer, │ │ │ │ │ scope: this │ │ │ │ │ }); │ │ │ │ │ } │ │ │ │ │ OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.CacheRead" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.CacheWrite" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * APIFunction: OpenLayers.Control.CacheWrite.clearCache │ │ │ │ │ + * Clears all tiles cached with <OpenLayers.Control.CacheWrite> from the cache. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.CacheWrite.clearCache = function() { │ │ │ │ │ + if (!window.localStorage) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var i, key; │ │ │ │ │ + for (i = window.localStorage.length - 1; i >= 0; --i) { │ │ │ │ │ + key = window.localStorage.key(i); │ │ │ │ │ + if (key.substr(0, 8) === "olCache_") { │ │ │ │ │ + window.localStorage.removeItem(key); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Property: OpenLayers.Control.CacheWrite.urlMap │ │ │ │ │ + * {Object} Mapping of same origin urls to cache url keys. Entries will be │ │ │ │ │ + * deleted as soon as a tile was cached. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.CacheWrite.urlMap = {}; │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/ModifyFeature.js │ │ │ │ │ + OpenLayers/Control/PanPanel.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Handler/Drag.js │ │ │ │ │ - * @requires OpenLayers/Handler/Keyboard.js │ │ │ │ │ + * @requires OpenLayers/Control/Panel.js │ │ │ │ │ + * @requires OpenLayers/Control/Pan.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.ModifyFeature │ │ │ │ │ - * Control to modify features. When activated, a click renders the vertices │ │ │ │ │ - * of a feature - these vertices can then be dragged. By default, the │ │ │ │ │ - * delete key will delete the vertex under the mouse. New features are │ │ │ │ │ - * added by dragging "virtual vertices" between vertices. Create a new │ │ │ │ │ - * control with the <OpenLayers.Control.ModifyFeature> constructor. │ │ │ │ │ + * Class: OpenLayers.Control.PanPanel │ │ │ │ │ + * The PanPanel is visible control for panning the map North, South, East or │ │ │ │ │ + * West in small steps. By default it is drawn in the top left corner of the │ │ │ │ │ + * map. │ │ │ │ │ * │ │ │ │ │ - * Inherits From: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ + * Note: │ │ │ │ │ + * If you wish to use this class with the default images and you want │ │ │ │ │ + * it to look nice in ie6, you should add the following, conditionally │ │ │ │ │ + * added css stylesheet to your HTML file: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * <!--[if lte IE 6]> │ │ │ │ │ + * <link rel="stylesheet" href="../theme/default/ie6-style.css" type="text/css" /> │ │ │ │ │ + * <![endif]--> │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control.Panel> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.ModifyFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Control.PanPanel = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: documentDrag │ │ │ │ │ - * {Boolean} If set to true, dragging vertices will continue even if the │ │ │ │ │ - * mouse cursor leaves the map viewport. Default is false. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: slideFactor │ │ │ │ │ + * {Integer} Number of pixels by which we'll pan the map in any direction │ │ │ │ │ + * on clicking the arrow buttons, defaults to 50. If you want to pan │ │ │ │ │ + * by some ratio of the map dimensions, use <slideRatio> instead. │ │ │ │ │ */ │ │ │ │ │ - documentDrag: false, │ │ │ │ │ + slideFactor: 50, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geometryTypes │ │ │ │ │ - * {Array(String)} To restrict modification to a limited set of geometry │ │ │ │ │ - * types, send a list of strings corresponding to the geometry class │ │ │ │ │ - * names. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: slideRatio │ │ │ │ │ + * {Number} The fraction of map width/height by which we'll pan the map │ │ │ │ │ + * on clicking the arrow buttons. Default is null. If set, will │ │ │ │ │ + * override <slideFactor>. E.g. if slideRatio is .5, then Pan Up will │ │ │ │ │ + * pan up half the map height. │ │ │ │ │ */ │ │ │ │ │ - geometryTypes: null, │ │ │ │ │ + slideRatio: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: clickout │ │ │ │ │ - * {Boolean} Unselect features when clicking outside any feature. │ │ │ │ │ - * Default is true. │ │ │ │ │ + * Constructor: OpenLayers.Control.PanPanel │ │ │ │ │ + * Add the four directional pan buttons. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be used │ │ │ │ │ + * to extend the control. │ │ │ │ │ */ │ │ │ │ │ - clickout: true, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ + var options = { │ │ │ │ │ + slideFactor: this.slideFactor, │ │ │ │ │ + slideRatio: this.slideRatio │ │ │ │ │ + }; │ │ │ │ │ + this.addControls([ │ │ │ │ │ + new OpenLayers.Control.Pan(OpenLayers.Control.Pan.NORTH, options), │ │ │ │ │ + new OpenLayers.Control.Pan(OpenLayers.Control.Pan.SOUTH, options), │ │ │ │ │ + new OpenLayers.Control.Pan(OpenLayers.Control.Pan.EAST, options), │ │ │ │ │ + new OpenLayers.Control.Pan(OpenLayers.Control.Pan.WEST, options) │ │ │ │ │ + ]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: toggle │ │ │ │ │ - * {Boolean} Unselect a selected feature on click. │ │ │ │ │ - * Default is true. │ │ │ │ │ - */ │ │ │ │ │ - toggle: true, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.PanPanel" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/MousePosition.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: standalone │ │ │ │ │ - * {Boolean} Set to true to create a control without SelectFeature │ │ │ │ │ - * capabilities. Default is false. If standalone is true, to modify │ │ │ │ │ - * a feature, call the <selectFeature> method with the target feature. │ │ │ │ │ - * Note that you must call the <unselectFeature> method to finish │ │ │ │ │ - * feature modification in standalone mode (before starting to modify │ │ │ │ │ - * another feature). │ │ │ │ │ - */ │ │ │ │ │ - standalone: false, │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.MousePosition │ │ │ │ │ + * The MousePosition control displays geographic coordinates of the mouse │ │ │ │ │ + * pointer, as it is moved about the map. │ │ │ │ │ + * │ │ │ │ │ + * You can use the <prefix>- or <suffix>-properties to provide more information │ │ │ │ │ + * about the displayed coordinates to the user: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * var mousePositionCtrl = new OpenLayers.Control.MousePosition({ │ │ │ │ │ + * prefix: '<a target="_blank" ' + │ │ │ │ │ + * 'href="http://spatialreference.org/ref/epsg/4326/">' + │ │ │ │ │ + * 'EPSG:4326</a> coordinates: ' │ │ │ │ │ + * } │ │ │ │ │ + * ); │ │ │ │ │ + * (end code) │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.MousePosition = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ - layer: null, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: feature │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} Feature currently available for modification. │ │ │ │ │ + * Property: element │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - feature: null, │ │ │ │ │ + element: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: vertex │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} Vertex currently being modified. │ │ │ │ │ + * APIProperty: prefix │ │ │ │ │ + * {String} A string to be prepended to the current pointers coordinates │ │ │ │ │ + * when it is rendered. Defaults to the empty string ''. │ │ │ │ │ */ │ │ │ │ │ - vertex: null, │ │ │ │ │ + prefix: '', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: vertices │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} Verticies currently available │ │ │ │ │ - * for dragging. │ │ │ │ │ + * APIProperty: separator │ │ │ │ │ + * {String} A string to be used to seperate the two coordinates from each │ │ │ │ │ + * other. Defaults to the string ', ', which will result in a │ │ │ │ │ + * rendered coordinate of e.g. '42.12, 21.22'. │ │ │ │ │ */ │ │ │ │ │ - vertices: null, │ │ │ │ │ + separator: ', ', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: virtualVertices │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} Virtual vertices in the middle │ │ │ │ │ - * of each edge. │ │ │ │ │ + * APIProperty: suffix │ │ │ │ │ + * {String} A string to be appended to the current pointers coordinates │ │ │ │ │ + * when it is rendered. Defaults to the empty string ''. │ │ │ │ │ */ │ │ │ │ │ - virtualVertices: null, │ │ │ │ │ + suffix: '', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: handlers │ │ │ │ │ - * {Object} │ │ │ │ │ + * APIProperty: numDigits │ │ │ │ │ + * {Integer} The number of digits each coordinate shall have when being │ │ │ │ │ + * rendered, Defaults to 5. │ │ │ │ │ */ │ │ │ │ │ - handlers: null, │ │ │ │ │ + numDigits: 5, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: deleteCodes │ │ │ │ │ - * {Array(Integer)} Keycodes for deleting verticies. Set to null to disable │ │ │ │ │ - * vertex deltion by keypress. If non-null, keypresses with codes │ │ │ │ │ - * in this array will delete vertices under the mouse. Default │ │ │ │ │ - * is 46 and 68, the 'delete' and lowercase 'd' keys. │ │ │ │ │ + * APIProperty: granularity │ │ │ │ │ + * {Integer} │ │ │ │ │ */ │ │ │ │ │ - deleteCodes: null, │ │ │ │ │ + granularity: 10, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: virtualStyle │ │ │ │ │ - * {Object} A symbolizer to be used for virtual vertices. │ │ │ │ │ + * APIProperty: emptyString │ │ │ │ │ + * {String} Set this to some value to set when the mouse is outside the │ │ │ │ │ + * map. │ │ │ │ │ */ │ │ │ │ │ - virtualStyle: null, │ │ │ │ │ + emptyString: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: vertexRenderIntent │ │ │ │ │ - * {String} The renderIntent to use for vertices. If no <virtualStyle> is │ │ │ │ │ - * provided, this renderIntent will also be used for virtual vertices, with │ │ │ │ │ - * a fillOpacity and strokeOpacity of 0.3. Default is null, which means │ │ │ │ │ - * that the layer's default style will be used for vertices. │ │ │ │ │ + * Property: lastXy │ │ │ │ │ + * {<OpenLayers.Pixel>} │ │ │ │ │ */ │ │ │ │ │ - vertexRenderIntent: null, │ │ │ │ │ + lastXy: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: mode │ │ │ │ │ - * {Integer} Bitfields specifying the modification mode. Defaults to │ │ │ │ │ - * OpenLayers.Control.ModifyFeature.RESHAPE. To set the mode to a │ │ │ │ │ - * combination of options, use the | operator. For example, to allow │ │ │ │ │ - * the control to both resize and rotate features, use the following │ │ │ │ │ - * syntax │ │ │ │ │ - * (code) │ │ │ │ │ - * control.mode = OpenLayers.Control.ModifyFeature.RESIZE | │ │ │ │ │ - * OpenLayers.Control.ModifyFeature.ROTATE; │ │ │ │ │ - * (end) │ │ │ │ │ + * APIProperty: displayProjection │ │ │ │ │ + * {<OpenLayers.Projection>} The projection in which the mouse position is │ │ │ │ │ + * displayed. │ │ │ │ │ */ │ │ │ │ │ - mode: null, │ │ │ │ │ + displayProjection: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: createVertices │ │ │ │ │ - * {Boolean} Create new vertices by dragging the virtual vertices │ │ │ │ │ - * in the middle of each edge. Default is true. │ │ │ │ │ + * Constructor: OpenLayers.Control.MousePosition │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Options for control. │ │ │ │ │ */ │ │ │ │ │ - createVertices: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: modified │ │ │ │ │ - * {Boolean} The currently selected feature has been modified. │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ - modified: false, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: radiusHandle │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A handle for rotating/resizing a feature. │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ */ │ │ │ │ │ - radiusHandle: null, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.map.events.register('mousemove', this, this.redraw); │ │ │ │ │ + this.map.events.register('mouseout', this, this.reset); │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: dragHandle │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A handle for dragging a feature. │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ */ │ │ │ │ │ - dragHandle: null, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.map.events.unregister('mousemove', this, this.redraw); │ │ │ │ │ + this.map.events.unregister('mouseout', this, this.reset); │ │ │ │ │ + this.element.innerHTML = ""; │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: onModificationStart │ │ │ │ │ - * {Function} *Deprecated*. Register for "beforefeaturemodified" instead. │ │ │ │ │ - * The "beforefeaturemodified" event is triggered on the layer before │ │ │ │ │ - * any modification begins. │ │ │ │ │ - * │ │ │ │ │ - * Optional function to be called when a feature is selected │ │ │ │ │ - * to be modified. The function should expect to be called with a │ │ │ │ │ - * feature. This could be used for example to allow to lock the │ │ │ │ │ - * feature on server-side. │ │ │ │ │ + * Method: draw │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - onModificationStart: function() {}, │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + if (!this.element) { │ │ │ │ │ + this.div.left = ""; │ │ │ │ │ + this.div.top = ""; │ │ │ │ │ + this.element = this.div; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return this.div; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: onModification │ │ │ │ │ - * {Function} *Deprecated*. Register for "featuremodified" instead. │ │ │ │ │ - * The "featuremodified" event is triggered on the layer with each │ │ │ │ │ - * feature modification. │ │ │ │ │ - * │ │ │ │ │ - * Optional function to be called when a feature has been │ │ │ │ │ - * modified. The function should expect to be called with a feature. │ │ │ │ │ + * Method: redraw │ │ │ │ │ */ │ │ │ │ │ - onModification: function() {}, │ │ │ │ │ + redraw: function(evt) { │ │ │ │ │ + │ │ │ │ │ + var lonLat; │ │ │ │ │ + │ │ │ │ │ + if (evt == null) { │ │ │ │ │ + this.reset(); │ │ │ │ │ + return; │ │ │ │ │ + } else { │ │ │ │ │ + if (this.lastXy == null || │ │ │ │ │ + Math.abs(evt.xy.x - this.lastXy.x) > this.granularity || │ │ │ │ │ + Math.abs(evt.xy.y - this.lastXy.y) > this.granularity) { │ │ │ │ │ + this.lastXy = evt.xy; │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + lonLat = this.map.getLonLatFromPixel(evt.xy); │ │ │ │ │ + if (!lonLat) { │ │ │ │ │ + // map has not yet been properly initialized │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (this.displayProjection) { │ │ │ │ │ + lonLat.transform(this.map.getProjectionObject(), │ │ │ │ │ + this.displayProjection); │ │ │ │ │ + } │ │ │ │ │ + this.lastXy = evt.xy; │ │ │ │ │ + │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var newHtml = this.formatOutput(lonLat); │ │ │ │ │ + │ │ │ │ │ + if (newHtml != this.element.innerHTML) { │ │ │ │ │ + this.element.innerHTML = newHtml; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: onModificationEnd │ │ │ │ │ - * {Function} *Deprecated*. Register for "afterfeaturemodified" instead. │ │ │ │ │ - * The "afterfeaturemodified" event is triggered on the layer after │ │ │ │ │ - * a feature has been modified. │ │ │ │ │ - * │ │ │ │ │ - * Optional function to be called when a feature is finished │ │ │ │ │ - * being modified. The function should expect to be called with a │ │ │ │ │ - * feature. │ │ │ │ │ + * Method: reset │ │ │ │ │ */ │ │ │ │ │ - onModificationEnd: function() {}, │ │ │ │ │ + reset: function(evt) { │ │ │ │ │ + if (this.emptyString != null) { │ │ │ │ │ + this.element.innerHTML = this.emptyString; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.ModifyFeature │ │ │ │ │ - * Create a new modify feature control. │ │ │ │ │ + * Method: formatOutput │ │ │ │ │ + * Override to provide custom display output │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.Vector>} Layer that contains features that │ │ │ │ │ - * will be modified. │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * control. │ │ │ │ │ + * lonLat - {<OpenLayers.LonLat>} Location to display │ │ │ │ │ */ │ │ │ │ │ - initialize: function(layer, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - this.virtualVertices = []; │ │ │ │ │ - this.virtualStyle = OpenLayers.Util.extend({}, │ │ │ │ │ - this.layer.style || │ │ │ │ │ - this.layer.styleMap.createSymbolizer(null, options.vertexRenderIntent) │ │ │ │ │ - ); │ │ │ │ │ - this.virtualStyle.fillOpacity = 0.3; │ │ │ │ │ - this.virtualStyle.strokeOpacity = 0.3; │ │ │ │ │ - this.deleteCodes = [46, 68]; │ │ │ │ │ - this.mode = OpenLayers.Control.ModifyFeature.RESHAPE; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!(OpenLayers.Util.isArray(this.deleteCodes))) { │ │ │ │ │ - this.deleteCodes = [this.deleteCodes]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // configure the drag handler │ │ │ │ │ - var dragCallbacks = { │ │ │ │ │ - down: function(pixel) { │ │ │ │ │ - this.vertex = null; │ │ │ │ │ - var feature = this.layer.getFeatureFromEvent( │ │ │ │ │ - this.handlers.drag.evt); │ │ │ │ │ - if (feature) { │ │ │ │ │ - this.dragStart(feature); │ │ │ │ │ - } else if (this.clickout) { │ │ │ │ │ - this._unselect = this.feature; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - move: function(pixel) { │ │ │ │ │ - delete this._unselect; │ │ │ │ │ - if (this.vertex) { │ │ │ │ │ - this.dragVertex(this.vertex, pixel); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - up: function() { │ │ │ │ │ - this.handlers.drag.stopDown = false; │ │ │ │ │ - if (this._unselect) { │ │ │ │ │ - this.unselectFeature(this._unselect); │ │ │ │ │ - delete this._unselect; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - done: function(pixel) { │ │ │ │ │ - if (this.vertex) { │ │ │ │ │ - this.dragComplete(this.vertex); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - var dragOptions = { │ │ │ │ │ - documentDrag: this.documentDrag, │ │ │ │ │ - stopDown: false │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // configure the keyboard handler │ │ │ │ │ - var keyboardOptions = { │ │ │ │ │ - keydown: this.handleKeypress │ │ │ │ │ - }; │ │ │ │ │ - this.handlers = { │ │ │ │ │ - keyboard: new OpenLayers.Handler.Keyboard(this, keyboardOptions), │ │ │ │ │ - drag: new OpenLayers.Handler.Drag(this, dragCallbacks, dragOptions) │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Take care of things that are not handled in superclass. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - "removelayer": this.handleMapEvents, │ │ │ │ │ - "changelayer": this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - this.layer = null; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, []); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Activate the control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Successfully activated the control. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - this.moveLayerToTop(); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - "removelayer": this.handleMapEvents, │ │ │ │ │ - "changelayer": this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - return (this.handlers.keyboard.activate() && │ │ │ │ │ - this.handlers.drag.activate() && │ │ │ │ │ - OpenLayers.Control.prototype.activate.apply(this, arguments)); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Successfully deactivated the control. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - // the return from the controls is unimportant in this case │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerBack(); │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - "removelayer": this.handleMapEvents, │ │ │ │ │ - "changelayer": this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.layer.removeFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.layer.removeFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - this.handlers.drag.deactivate(); │ │ │ │ │ - this.handlers.keyboard.deactivate(); │ │ │ │ │ - var feature = this.feature; │ │ │ │ │ - if (feature && feature.geometry && feature.layer) { │ │ │ │ │ - this.unselectFeature(feature); │ │ │ │ │ - } │ │ │ │ │ - deactivated = true; │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: beforeSelectFeature │ │ │ │ │ - * Called before a feature is selected. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} The feature about to be selected. │ │ │ │ │ - */ │ │ │ │ │ - beforeSelectFeature: function(feature) { │ │ │ │ │ - return this.layer.events.triggerEvent( │ │ │ │ │ - "beforefeaturemodified", { │ │ │ │ │ - feature: feature │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: selectFeature │ │ │ │ │ - * Select a feature for modification in standalone mode. In non-standalone │ │ │ │ │ - * mode, this method is called when a feature is selected by clicking. │ │ │ │ │ - * Register a listener to the beforefeaturemodified event and return false │ │ │ │ │ - * to prevent feature modification. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} the selected feature. │ │ │ │ │ - */ │ │ │ │ │ - selectFeature: function(feature) { │ │ │ │ │ - if (this.feature === feature || │ │ │ │ │ - (this.geometryTypes && OpenLayers.Util.indexOf(this.geometryTypes, │ │ │ │ │ - feature.geometry.CLASS_NAME) == -1)) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (this.beforeSelectFeature(feature) !== false) { │ │ │ │ │ - if (this.feature) { │ │ │ │ │ - this.unselectFeature(this.feature); │ │ │ │ │ - } │ │ │ │ │ - this.feature = feature; │ │ │ │ │ - this.layer.selectedFeatures.push(feature); │ │ │ │ │ - this.layer.drawFeature(feature, 'select'); │ │ │ │ │ - this.modified = false; │ │ │ │ │ - this.resetVertices(); │ │ │ │ │ - this.onModificationStart(this.feature); │ │ │ │ │ - } │ │ │ │ │ - // keep track of geometry modifications │ │ │ │ │ - var modified = feature.modified; │ │ │ │ │ - if (feature.geometry && !(modified && modified.geometry)) { │ │ │ │ │ - this._originalGeometry = feature.geometry.clone(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: unselectFeature │ │ │ │ │ - * Called when the select feature control unselects a feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} The unselected feature. │ │ │ │ │ - */ │ │ │ │ │ - unselectFeature: function(feature) { │ │ │ │ │ - this.layer.removeFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - this.layer.destroyFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.virtualVertices = []; │ │ │ │ │ - if (this.dragHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.dragHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - delete this.dragHandle; │ │ │ │ │ - } │ │ │ │ │ - if (this.radiusHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - delete this.radiusHandle; │ │ │ │ │ - } │ │ │ │ │ - this.layer.drawFeature(this.feature, 'default'); │ │ │ │ │ - this.feature = null; │ │ │ │ │ - OpenLayers.Util.removeItem(this.layer.selectedFeatures, feature); │ │ │ │ │ - this.onModificationEnd(feature); │ │ │ │ │ - this.layer.events.triggerEvent("afterfeaturemodified", { │ │ │ │ │ - feature: feature, │ │ │ │ │ - modified: this.modified │ │ │ │ │ - }); │ │ │ │ │ - this.modified = false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: dragStart │ │ │ │ │ - * Called by the drag handler before a feature is dragged. This method is │ │ │ │ │ - * used to differentiate between points and vertices │ │ │ │ │ - * of higher order geometries. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} The point or vertex about to be │ │ │ │ │ - * dragged. │ │ │ │ │ - */ │ │ │ │ │ - dragStart: function(feature) { │ │ │ │ │ - var isPoint = feature.geometry.CLASS_NAME == │ │ │ │ │ - 'OpenLayers.Geometry.Point'; │ │ │ │ │ - if (!this.standalone && │ │ │ │ │ - ((!feature._sketch && isPoint) || !feature._sketch)) { │ │ │ │ │ - if (this.toggle && this.feature === feature) { │ │ │ │ │ - // mark feature for unselection │ │ │ │ │ - this._unselect = feature; │ │ │ │ │ - } │ │ │ │ │ - this.selectFeature(feature); │ │ │ │ │ - } │ │ │ │ │ - if (feature._sketch || isPoint) { │ │ │ │ │ - // feature is a drag or virtual handle or point │ │ │ │ │ - this.vertex = feature; │ │ │ │ │ - this.handlers.drag.stopDown = true; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: dragVertex │ │ │ │ │ - * Called by the drag handler with each drag move of a vertex. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * vertex - {<OpenLayers.Feature.Vector>} The vertex being dragged. │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} Pixel location of the mouse event. │ │ │ │ │ - */ │ │ │ │ │ - dragVertex: function(vertex, pixel) { │ │ │ │ │ - var pos = this.map.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geom = vertex.geometry; │ │ │ │ │ - geom.move(pos.lon - geom.x, pos.lat - geom.y); │ │ │ │ │ - this.modified = true; │ │ │ │ │ - /** │ │ │ │ │ - * Five cases: │ │ │ │ │ - * 1) dragging a simple point │ │ │ │ │ - * 2) dragging a virtual vertex │ │ │ │ │ - * 3) dragging a drag handle │ │ │ │ │ - * 4) dragging a real vertex │ │ │ │ │ - * 5) dragging a radius handle │ │ │ │ │ - */ │ │ │ │ │ - if (this.feature.geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - // dragging a simple point │ │ │ │ │ - this.layer.events.triggerEvent("vertexmodified", { │ │ │ │ │ - vertex: vertex.geometry, │ │ │ │ │ - feature: this.feature, │ │ │ │ │ - pixel: pixel │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - if (vertex._index) { │ │ │ │ │ - // dragging a virtual vertex │ │ │ │ │ - vertex.geometry.parent.addComponent(vertex.geometry, │ │ │ │ │ - vertex._index); │ │ │ │ │ - // move from virtual to real vertex │ │ │ │ │ - delete vertex._index; │ │ │ │ │ - OpenLayers.Util.removeItem(this.virtualVertices, vertex); │ │ │ │ │ - this.vertices.push(vertex); │ │ │ │ │ - } else if (vertex == this.dragHandle) { │ │ │ │ │ - // dragging a drag handle │ │ │ │ │ - this.layer.removeFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - if (this.radiusHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.radiusHandle = null; │ │ │ │ │ - } │ │ │ │ │ - } else if (vertex !== this.radiusHandle) { │ │ │ │ │ - // dragging a real vertex │ │ │ │ │ - this.layer.events.triggerEvent("vertexmodified", { │ │ │ │ │ - vertex: vertex.geometry, │ │ │ │ │ - feature: this.feature, │ │ │ │ │ - pixel: pixel │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - // dragging a radius handle - no special treatment │ │ │ │ │ - if (this.virtualVertices.length > 0) { │ │ │ │ │ - this.layer.destroyFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.virtualVertices = []; │ │ │ │ │ - } │ │ │ │ │ - this.layer.drawFeature(this.feature, this.standalone ? undefined : │ │ │ │ │ - 'select'); │ │ │ │ │ - } │ │ │ │ │ - // keep the vertex on top so it gets the mouseout after dragging │ │ │ │ │ - // this should be removed in favor of an option to draw under or │ │ │ │ │ - // maintain node z-index │ │ │ │ │ - this.layer.drawFeature(vertex); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: dragComplete │ │ │ │ │ - * Called by the drag handler when the feature dragging is complete. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * vertex - {<OpenLayers.Feature.Vector>} The vertex being dragged. │ │ │ │ │ - */ │ │ │ │ │ - dragComplete: function(vertex) { │ │ │ │ │ - this.resetVertices(); │ │ │ │ │ - this.setFeatureState(); │ │ │ │ │ - this.onModification(this.feature); │ │ │ │ │ - this.layer.events.triggerEvent("featuremodified", { │ │ │ │ │ - feature: this.feature │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setFeatureState │ │ │ │ │ - * Called when the feature is modified. If the current state is not │ │ │ │ │ - * INSERT or DELETE, the state is set to UPDATE. │ │ │ │ │ - */ │ │ │ │ │ - setFeatureState: function() { │ │ │ │ │ - if (this.feature.state != OpenLayers.State.INSERT && │ │ │ │ │ - this.feature.state != OpenLayers.State.DELETE) { │ │ │ │ │ - this.feature.state = OpenLayers.State.UPDATE; │ │ │ │ │ - if (this.modified && this._originalGeometry) { │ │ │ │ │ - var feature = this.feature; │ │ │ │ │ - feature.modified = OpenLayers.Util.extend(feature.modified, { │ │ │ │ │ - geometry: this._originalGeometry │ │ │ │ │ - }); │ │ │ │ │ - delete this._originalGeometry; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: resetVertices │ │ │ │ │ - */ │ │ │ │ │ - resetVertices: function() { │ │ │ │ │ - if (this.vertices.length > 0) { │ │ │ │ │ - this.layer.removeFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - } │ │ │ │ │ - if (this.virtualVertices.length > 0) { │ │ │ │ │ - this.layer.removeFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.virtualVertices = []; │ │ │ │ │ - } │ │ │ │ │ - if (this.dragHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.dragHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.dragHandle = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.radiusHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.radiusHandle = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.feature && │ │ │ │ │ - this.feature.geometry.CLASS_NAME != "OpenLayers.Geometry.Point") { │ │ │ │ │ - if ((this.mode & OpenLayers.Control.ModifyFeature.DRAG)) { │ │ │ │ │ - this.collectDragHandle(); │ │ │ │ │ - } │ │ │ │ │ - if ((this.mode & (OpenLayers.Control.ModifyFeature.ROTATE | │ │ │ │ │ - OpenLayers.Control.ModifyFeature.RESIZE))) { │ │ │ │ │ - this.collectRadiusHandle(); │ │ │ │ │ - } │ │ │ │ │ - if (this.mode & OpenLayers.Control.ModifyFeature.RESHAPE) { │ │ │ │ │ - // Don't collect vertices when we're resizing │ │ │ │ │ - if (!(this.mode & OpenLayers.Control.ModifyFeature.RESIZE)) { │ │ │ │ │ - this.collectVertices(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleKeypress │ │ │ │ │ - * Called by the feature handler on keypress. This is used to delete │ │ │ │ │ - * vertices. If the <deleteCode> property is set, vertices will │ │ │ │ │ - * be deleted when a feature is selected for modification and │ │ │ │ │ - * the mouse is over a vertex. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} Keypress event. │ │ │ │ │ - */ │ │ │ │ │ - handleKeypress: function(evt) { │ │ │ │ │ - var code = evt.keyCode; │ │ │ │ │ - │ │ │ │ │ - // check for delete key │ │ │ │ │ - if (this.feature && │ │ │ │ │ - OpenLayers.Util.indexOf(this.deleteCodes, code) != -1) { │ │ │ │ │ - var vertex = this.layer.getFeatureFromEvent(this.handlers.drag.evt); │ │ │ │ │ - if (vertex && │ │ │ │ │ - OpenLayers.Util.indexOf(this.vertices, vertex) != -1 && │ │ │ │ │ - !this.handlers.drag.dragging && vertex.geometry.parent) { │ │ │ │ │ - // remove the vertex │ │ │ │ │ - vertex.geometry.parent.removeComponent(vertex.geometry); │ │ │ │ │ - this.layer.events.triggerEvent("vertexremoved", { │ │ │ │ │ - vertex: vertex.geometry, │ │ │ │ │ - feature: this.feature, │ │ │ │ │ - pixel: evt.xy │ │ │ │ │ - }); │ │ │ │ │ - this.layer.drawFeature(this.feature, this.standalone ? │ │ │ │ │ - undefined : 'select'); │ │ │ │ │ - this.modified = true; │ │ │ │ │ - this.resetVertices(); │ │ │ │ │ - this.setFeatureState(); │ │ │ │ │ - this.onModification(this.feature); │ │ │ │ │ - this.layer.events.triggerEvent("featuremodified", { │ │ │ │ │ - feature: this.feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: collectVertices │ │ │ │ │ - * Collect the vertices from the modifiable feature's geometry and push │ │ │ │ │ - * them on to the control's vertices array. │ │ │ │ │ - */ │ │ │ │ │ - collectVertices: function() { │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - this.virtualVertices = []; │ │ │ │ │ - var control = this; │ │ │ │ │ - │ │ │ │ │ - function collectComponentVertices(geometry) { │ │ │ │ │ - var i, vertex, component, len; │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - vertex = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - vertex._sketch = true; │ │ │ │ │ - vertex.renderIntent = control.vertexRenderIntent; │ │ │ │ │ - control.vertices.push(vertex); │ │ │ │ │ - } else { │ │ │ │ │ - var numVert = geometry.components.length; │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ - numVert -= 1; │ │ │ │ │ - } │ │ │ │ │ - for (i = 0; i < numVert; ++i) { │ │ │ │ │ - component = geometry.components[i]; │ │ │ │ │ - if (component.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - vertex = new OpenLayers.Feature.Vector(component); │ │ │ │ │ - vertex._sketch = true; │ │ │ │ │ - vertex.renderIntent = control.vertexRenderIntent; │ │ │ │ │ - control.vertices.push(vertex); │ │ │ │ │ - } else { │ │ │ │ │ - collectComponentVertices(component); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // add virtual vertices in the middle of each edge │ │ │ │ │ - if (control.createVertices && geometry.CLASS_NAME != "OpenLayers.Geometry.MultiPoint") { │ │ │ │ │ - for (i = 0, len = geometry.components.length; i < len - 1; ++i) { │ │ │ │ │ - var prevVertex = geometry.components[i]; │ │ │ │ │ - var nextVertex = geometry.components[i + 1]; │ │ │ │ │ - if (prevVertex.CLASS_NAME == "OpenLayers.Geometry.Point" && │ │ │ │ │ - nextVertex.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - var x = (prevVertex.x + nextVertex.x) / 2; │ │ │ │ │ - var y = (prevVertex.y + nextVertex.y) / 2; │ │ │ │ │ - var point = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.Point(x, y), │ │ │ │ │ - null, control.virtualStyle │ │ │ │ │ - ); │ │ │ │ │ - // set the virtual parent and intended index │ │ │ │ │ - point.geometry.parent = geometry; │ │ │ │ │ - point._index = i + 1; │ │ │ │ │ - point._sketch = true; │ │ │ │ │ - control.virtualVertices.push(point); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - collectComponentVertices.call(this, this.feature.geometry); │ │ │ │ │ - this.layer.addFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.layer.addFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: collectDragHandle │ │ │ │ │ - * Collect the drag handle for the selected geometry. │ │ │ │ │ - */ │ │ │ │ │ - collectDragHandle: function() { │ │ │ │ │ - var geometry = this.feature.geometry; │ │ │ │ │ - var center = geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - var originGeometry = new OpenLayers.Geometry.Point( │ │ │ │ │ - center.lon, center.lat │ │ │ │ │ - ); │ │ │ │ │ - var origin = new OpenLayers.Feature.Vector(originGeometry); │ │ │ │ │ - originGeometry.move = function(x, y) { │ │ │ │ │ - OpenLayers.Geometry.Point.prototype.move.call(this, x, y); │ │ │ │ │ - geometry.move(x, y); │ │ │ │ │ - }; │ │ │ │ │ - origin._sketch = true; │ │ │ │ │ - this.dragHandle = origin; │ │ │ │ │ - this.dragHandle.renderIntent = this.vertexRenderIntent; │ │ │ │ │ - this.layer.addFeatures([this.dragHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: collectRadiusHandle │ │ │ │ │ - * Collect the radius handle for the selected geometry. │ │ │ │ │ - */ │ │ │ │ │ - collectRadiusHandle: function() { │ │ │ │ │ - var geometry = this.feature.geometry; │ │ │ │ │ - var bounds = geometry.getBounds(); │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var originGeometry = new OpenLayers.Geometry.Point( │ │ │ │ │ - center.lon, center.lat │ │ │ │ │ - ); │ │ │ │ │ - var radiusGeometry = new OpenLayers.Geometry.Point( │ │ │ │ │ - bounds.right, bounds.bottom │ │ │ │ │ - ); │ │ │ │ │ - var radius = new OpenLayers.Feature.Vector(radiusGeometry); │ │ │ │ │ - var resize = (this.mode & OpenLayers.Control.ModifyFeature.RESIZE); │ │ │ │ │ - var reshape = (this.mode & OpenLayers.Control.ModifyFeature.RESHAPE); │ │ │ │ │ - var rotate = (this.mode & OpenLayers.Control.ModifyFeature.ROTATE); │ │ │ │ │ - │ │ │ │ │ - radiusGeometry.move = function(x, y) { │ │ │ │ │ - OpenLayers.Geometry.Point.prototype.move.call(this, x, y); │ │ │ │ │ - var dx1 = this.x - originGeometry.x; │ │ │ │ │ - var dy1 = this.y - originGeometry.y; │ │ │ │ │ - var dx0 = dx1 - x; │ │ │ │ │ - var dy0 = dy1 - y; │ │ │ │ │ - if (rotate) { │ │ │ │ │ - var a0 = Math.atan2(dy0, dx0); │ │ │ │ │ - var a1 = Math.atan2(dy1, dx1); │ │ │ │ │ - var angle = a1 - a0; │ │ │ │ │ - angle *= 180 / Math.PI; │ │ │ │ │ - geometry.rotate(angle, originGeometry); │ │ │ │ │ - } │ │ │ │ │ - if (resize) { │ │ │ │ │ - var scale, ratio; │ │ │ │ │ - // 'resize' together with 'reshape' implies that the aspect │ │ │ │ │ - // ratio of the geometry will not be preserved whilst resizing │ │ │ │ │ - if (reshape) { │ │ │ │ │ - scale = dy1 / dy0; │ │ │ │ │ - ratio = (dx1 / dx0) / scale; │ │ │ │ │ - } else { │ │ │ │ │ - var l0 = Math.sqrt((dx0 * dx0) + (dy0 * dy0)); │ │ │ │ │ - var l1 = Math.sqrt((dx1 * dx1) + (dy1 * dy1)); │ │ │ │ │ - scale = l1 / l0; │ │ │ │ │ - } │ │ │ │ │ - geometry.resize(scale, originGeometry, ratio); │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - radius._sketch = true; │ │ │ │ │ - this.radiusHandle = radius; │ │ │ │ │ - this.radiusHandle.renderIntent = this.vertexRenderIntent; │ │ │ │ │ - this.layer.addFeatures([this.radiusHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the control and all handlers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} The control's map. │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.handlers.drag.setMap(map); │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleMapEvents │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ - */ │ │ │ │ │ - handleMapEvents: function(evt) { │ │ │ │ │ - if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ - this.moveLayerToTop(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveLayerToTop │ │ │ │ │ - * Moves the layer for this handler to the top, so mouse events can reach │ │ │ │ │ - * it. │ │ │ │ │ - */ │ │ │ │ │ - moveLayerToTop: function() { │ │ │ │ │ - var index = Math.max(this.map.Z_INDEX_BASE['Feature'] - 1, │ │ │ │ │ - this.layer.getZIndex()) + 1; │ │ │ │ │ - this.layer.setZIndex(index); │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveLayerBack │ │ │ │ │ - * Moves the layer back to the position determined by the map's layers │ │ │ │ │ - * array. │ │ │ │ │ - */ │ │ │ │ │ - moveLayerBack: function() { │ │ │ │ │ - var index = this.layer.getZIndex() - 1; │ │ │ │ │ - if (index >= this.map.Z_INDEX_BASE['Feature']) { │ │ │ │ │ - this.layer.setZIndex(index); │ │ │ │ │ - } else { │ │ │ │ │ - this.map.setLayerZIndex(this.layer, │ │ │ │ │ - this.map.getLayerIndex(this.layer)); │ │ │ │ │ - } │ │ │ │ │ + formatOutput: function(lonLat) { │ │ │ │ │ + var digits = parseInt(this.numDigits); │ │ │ │ │ + var newHtml = │ │ │ │ │ + this.prefix + │ │ │ │ │ + lonLat.lon.toFixed(digits) + │ │ │ │ │ + this.separator + │ │ │ │ │ + lonLat.lat.toFixed(digits) + │ │ │ │ │ + this.suffix; │ │ │ │ │ + return newHtml; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ModifyFeature" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.MousePosition" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: RESHAPE │ │ │ │ │ - * {Integer} Constant used to make the control work in reshape mode │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ModifyFeature.RESHAPE = 1; │ │ │ │ │ -/** │ │ │ │ │ - * Constant: RESIZE │ │ │ │ │ - * {Integer} Constant used to make the control work in resize mode │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ModifyFeature.RESIZE = 2; │ │ │ │ │ -/** │ │ │ │ │ - * Constant: ROTATE │ │ │ │ │ - * {Integer} Constant used to make the control work in rotate mode │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ModifyFeature.ROTATE = 4; │ │ │ │ │ -/** │ │ │ │ │ - * Constant: DRAG │ │ │ │ │ - * {Integer} Constant used to make the control work in drag mode │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ModifyFeature.DRAG = 8; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/DrawFeature.js │ │ │ │ │ + OpenLayers/Control/Graticule.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Rule.js │ │ │ │ │ + * @requires OpenLayers/StyleMap.js │ │ │ │ │ + * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.DrawFeature │ │ │ │ │ - * The DrawFeature control draws point, line or polygon features on a vector │ │ │ │ │ - * layer when active. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Control.Graticule │ │ │ │ │ + * The Graticule displays a grid of latitude/longitude lines reprojected on │ │ │ │ │ + * the map. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.DrawFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - layer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: callbacks │ │ │ │ │ - * {Object} The functions that are sent to the handler for callback │ │ │ │ │ - */ │ │ │ │ │ - callbacks: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * featureadded - Triggered when a feature is added │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: multi │ │ │ │ │ - * {Boolean} Cast features to multi-part geometries before passing to the │ │ │ │ │ - * layer. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - multi: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: featureAdded │ │ │ │ │ - * {Function} Called after each feature is added │ │ │ │ │ - */ │ │ │ │ │ - featureAdded: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: handlerOptions │ │ │ │ │ - * {Object} Used to set non-default properties on the control's handler │ │ │ │ │ - */ │ │ │ │ │ +OpenLayers.Control.Graticule = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.DrawFeature │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ - * handler - {<OpenLayers.Handler>} │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(layer, handler, options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.callbacks = OpenLayers.Util.extend({ │ │ │ │ │ - done: this.drawFeature, │ │ │ │ │ - modify: function(vertex, feature) { │ │ │ │ │ - this.layer.events.triggerEvent( │ │ │ │ │ - "sketchmodified", { │ │ │ │ │ - vertex: vertex, │ │ │ │ │ - feature: feature │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - create: function(vertex, feature) { │ │ │ │ │ - this.layer.events.triggerEvent( │ │ │ │ │ - "sketchstarted", { │ │ │ │ │ - vertex: vertex, │ │ │ │ │ - feature: feature │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - this.callbacks │ │ │ │ │ - ); │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - this.handlerOptions = this.handlerOptions || {}; │ │ │ │ │ - this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.handlerOptions.layerOptions, { │ │ │ │ │ - renderers: layer.renderers, │ │ │ │ │ - rendererOptions: layer.rendererOptions │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - if (!("multi" in this.handlerOptions)) { │ │ │ │ │ - this.handlerOptions.multi = this.multi; │ │ │ │ │ - } │ │ │ │ │ - var sketchStyle = this.layer.styleMap && this.layer.styleMap.styles.temporary; │ │ │ │ │ - if (sketchStyle) { │ │ │ │ │ - this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.handlerOptions.layerOptions, { │ │ │ │ │ - styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ - "default": sketchStyle │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - this.handler = new handler(this, this.callbacks, this.handlerOptions); │ │ │ │ │ - }, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawFeature │ │ │ │ │ + * APIProperty: intervals │ │ │ │ │ + * {Array(Float)} A list of possible graticule widths in degrees. │ │ │ │ │ */ │ │ │ │ │ - drawFeature: function(geometry) { │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - var proceed = this.layer.events.triggerEvent( │ │ │ │ │ - "sketchcomplete", { │ │ │ │ │ - feature: feature │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - if (proceed !== false) { │ │ │ │ │ - feature.state = OpenLayers.State.INSERT; │ │ │ │ │ - this.layer.addFeatures([feature]); │ │ │ │ │ - this.featureAdded(feature); │ │ │ │ │ - this.events.triggerEvent("featureadded", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + intervals: [45, 30, 20, 10, 5, 2, 1, │ │ │ │ │ + 0.5, 0.2, 0.1, 0.05, 0.01, │ │ │ │ │ + 0.005, 0.002, 0.001 │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: insertXY │ │ │ │ │ - * Insert a point in the current sketch given x & y coordinates. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {Number} The x-coordinate of the point. │ │ │ │ │ - * y - {Number} The y-coordinate of the point. │ │ │ │ │ + * APIProperty: displayInLayerSwitcher │ │ │ │ │ + * {Boolean} Allows the Graticule control to be switched on and off by │ │ │ │ │ + * LayerSwitcher control. Defaults is true. │ │ │ │ │ */ │ │ │ │ │ - insertXY: function(x, y) { │ │ │ │ │ - if (this.handler && this.handler.line) { │ │ │ │ │ - this.handler.insertXY(x, y); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + displayInLayerSwitcher: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: insertDeltaXY │ │ │ │ │ - * Insert a point given offsets from the previously inserted point. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * dx - {Number} The x-coordinate offset of the point. │ │ │ │ │ - * dy - {Number} The y-coordinate offset of the point. │ │ │ │ │ + * APIProperty: visible │ │ │ │ │ + * {Boolean} should the graticule be initially visible (default=true) │ │ │ │ │ */ │ │ │ │ │ - insertDeltaXY: function(dx, dy) { │ │ │ │ │ - if (this.handler && this.handler.line) { │ │ │ │ │ - this.handler.insertDeltaXY(dx, dy); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + visible: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: insertDirectionLength │ │ │ │ │ - * Insert a point in the current sketch given a direction and a length. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * direction - {Number} Degrees clockwise from the positive x-axis. │ │ │ │ │ - * length - {Number} Distance from the previously drawn point. │ │ │ │ │ + * APIProperty: numPoints │ │ │ │ │ + * {Integer} The number of points to use in each graticule line. Higher │ │ │ │ │ + * numbers result in a smoother curve for projected maps │ │ │ │ │ */ │ │ │ │ │ - insertDirectionLength: function(direction, length) { │ │ │ │ │ - if (this.handler && this.handler.line) { │ │ │ │ │ - this.handler.insertDirectionLength(direction, length); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + numPoints: 50, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: insertDeflectionLength │ │ │ │ │ - * Insert a point in the current sketch given a deflection and a length. │ │ │ │ │ - * The deflection should be degrees clockwise from the previously │ │ │ │ │ - * digitized segment. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * deflection - {Number} Degrees clockwise from the previous segment. │ │ │ │ │ - * length - {Number} Distance from the previously drawn point. │ │ │ │ │ + * APIProperty: targetSize │ │ │ │ │ + * {Integer} The maximum size of the grid in pixels on the map │ │ │ │ │ */ │ │ │ │ │ - insertDeflectionLength: function(deflection, length) { │ │ │ │ │ - if (this.handler && this.handler.line) { │ │ │ │ │ - this.handler.insertDeflectionLength(deflection, length); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + targetSize: 200, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: undo │ │ │ │ │ - * Remove the most recently added point in the current sketch geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} An edit was undone. │ │ │ │ │ + * APIProperty: layerName │ │ │ │ │ + * {String} The name to be displayed in the layer switcher, default is set │ │ │ │ │ + * by {<OpenLayers.Lang>}. │ │ │ │ │ */ │ │ │ │ │ - undo: function() { │ │ │ │ │ - return this.handler.undo && this.handler.undo(); │ │ │ │ │ - }, │ │ │ │ │ + layerName: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: redo │ │ │ │ │ - * Reinsert the most recently removed point resulting from an <undo> call. │ │ │ │ │ - * The undo stack is deleted whenever a point is added by other means. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} An edit was redone. │ │ │ │ │ + * APIProperty: labelled │ │ │ │ │ + * {Boolean} Should the graticule lines be labelled?. default=true │ │ │ │ │ */ │ │ │ │ │ - redo: function() { │ │ │ │ │ - return this.handler.redo && this.handler.redo(); │ │ │ │ │ - }, │ │ │ │ │ + labelled: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: finishSketch │ │ │ │ │ - * Finishes the sketch without including the currently drawn point. │ │ │ │ │ - * This method can be called to terminate drawing programmatically │ │ │ │ │ - * instead of waiting for the user to end the sketch. │ │ │ │ │ + * APIProperty: labelFormat │ │ │ │ │ + * {String} the format of the labels, default = 'dm'. See │ │ │ │ │ + * <OpenLayers.Util.getFormattedLonLat> for other options. │ │ │ │ │ */ │ │ │ │ │ - finishSketch: function() { │ │ │ │ │ - this.handler.finishGeometry(); │ │ │ │ │ - }, │ │ │ │ │ + labelFormat: 'dm', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: cancel │ │ │ │ │ - * Cancel the current sketch. This removes the current sketch and keeps │ │ │ │ │ - * the drawing control active. │ │ │ │ │ + * APIProperty: lineSymbolizer │ │ │ │ │ + * {symbolizer} the symbolizer used to render lines │ │ │ │ │ */ │ │ │ │ │ - cancel: function() { │ │ │ │ │ - this.handler.cancel(); │ │ │ │ │ + lineSymbolizer: { │ │ │ │ │ + strokeColor: "#333", │ │ │ │ │ + strokeWidth: 1, │ │ │ │ │ + strokeOpacity: 0.5 │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.DrawFeature" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/ArgParser.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.ArgParser │ │ │ │ │ - * The ArgParser control adds location bar query string parsing functionality │ │ │ │ │ - * to an OpenLayers Map. │ │ │ │ │ - * When added to a Map control, on a page load/refresh, the Map will │ │ │ │ │ - * automatically take the href string and parse it for lon, lat, zoom, and │ │ │ │ │ - * layers information. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ArgParser = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: center │ │ │ │ │ - * {<OpenLayers.LonLat>} │ │ │ │ │ - */ │ │ │ │ │ - center: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: zoom │ │ │ │ │ - * {int} │ │ │ │ │ - */ │ │ │ │ │ - zoom: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layers │ │ │ │ │ - * {String} Each character represents the state of the corresponding layer │ │ │ │ │ - * on the map. │ │ │ │ │ - */ │ │ │ │ │ - layers: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: displayProjection │ │ │ │ │ - * {<OpenLayers.Projection>} Requires proj4js support. │ │ │ │ │ - * Projection used when reading the coordinates from the URL. This will │ │ │ │ │ - * reproject the map coordinates from the URL into the map's │ │ │ │ │ - * projection. │ │ │ │ │ - * │ │ │ │ │ - * If you are using this functionality, be aware that any permalink │ │ │ │ │ - * which is added to the map will determine the coordinate type which │ │ │ │ │ - * is read from the URL, which means you should not add permalinks with │ │ │ │ │ - * different displayProjections to the same map. │ │ │ │ │ - */ │ │ │ │ │ - displayProjection: null, │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.ArgParser │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * APIProperty: labelSymbolizer │ │ │ │ │ + * {symbolizer} the symbolizer used to render labels │ │ │ │ │ */ │ │ │ │ │ + labelSymbolizer: {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getParameters │ │ │ │ │ + * Property: gratLayer │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} vector layer used to draw the graticule on │ │ │ │ │ */ │ │ │ │ │ - getParameters: function(url) { │ │ │ │ │ - url = url || window.location.href; │ │ │ │ │ - var parameters = OpenLayers.Util.getParameters(url); │ │ │ │ │ - │ │ │ │ │ - // If we have an anchor in the url use it to split the url │ │ │ │ │ - var index = url.indexOf('#'); │ │ │ │ │ - if (index > 0) { │ │ │ │ │ - // create an url to parse on the getParameters │ │ │ │ │ - url = '?' + url.substring(index + 1, url.length); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Util.extend(parameters, │ │ │ │ │ - OpenLayers.Util.getParameters(url)); │ │ │ │ │ - } │ │ │ │ │ - return parameters; │ │ │ │ │ - }, │ │ │ │ │ + gratLayer: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the control. │ │ │ │ │ + * Constructor: OpenLayers.Control.Graticule │ │ │ │ │ + * Create a new graticule control to display a grid of latitude longitude │ │ │ │ │ + * lines. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - //make sure we dont already have an arg parser attached │ │ │ │ │ - for (var i = 0, len = this.map.controls.length; i < len; i++) { │ │ │ │ │ - var control = this.map.controls[i]; │ │ │ │ │ - if ((control != this) && │ │ │ │ │ - (control.CLASS_NAME == "OpenLayers.Control.ArgParser")) { │ │ │ │ │ - │ │ │ │ │ - // If a second argparser is added to the map, then we │ │ │ │ │ - // override the displayProjection to be the one added to the │ │ │ │ │ - // map. │ │ │ │ │ - if (control.displayProjection != this.displayProjection) { │ │ │ │ │ - this.displayProjection = control.displayProjection; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (i == this.map.controls.length) { │ │ │ │ │ - │ │ │ │ │ - var args = this.getParameters(); │ │ │ │ │ - // Be careful to set layer first, to not trigger unnecessary layer loads │ │ │ │ │ - if (args.layers) { │ │ │ │ │ - this.layers = args.layers; │ │ │ │ │ - │ │ │ │ │ - // when we add a new layer, set its visibility │ │ │ │ │ - this.map.events.register('addlayer', this, │ │ │ │ │ - this.configureLayers); │ │ │ │ │ - this.configureLayers(); │ │ │ │ │ - } │ │ │ │ │ - if (args.lat && args.lon) { │ │ │ │ │ - this.center = new OpenLayers.LonLat(parseFloat(args.lon), │ │ │ │ │ - parseFloat(args.lat)); │ │ │ │ │ - if (args.zoom) { │ │ │ │ │ - this.zoom = parseFloat(args.zoom); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // when we add a new baselayer to see when we can set the center │ │ │ │ │ - this.map.events.register('changebaselayer', this, │ │ │ │ │ - this.setCenter); │ │ │ │ │ - this.setCenter(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setCenter │ │ │ │ │ - * As soon as a baseLayer has been loaded, we center and zoom │ │ │ │ │ - * ...and remove the handler. │ │ │ │ │ - */ │ │ │ │ │ - setCenter: function() { │ │ │ │ │ - │ │ │ │ │ - if (this.map.baseLayer) { │ │ │ │ │ - //dont need to listen for this one anymore │ │ │ │ │ - this.map.events.unregister('changebaselayer', this, │ │ │ │ │ - this.setCenter); │ │ │ │ │ - │ │ │ │ │ - if (this.displayProjection) { │ │ │ │ │ - this.center.transform(this.displayProjection, │ │ │ │ │ - this.map.getProjectionObject()); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.map.setCenter(this.center, this.zoom); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: configureLayers │ │ │ │ │ - * As soon as all the layers are loaded, cycle through them and │ │ │ │ │ - * hide or show them. │ │ │ │ │ + * options - {Object} An optional object whose properties will be used │ │ │ │ │ + * to extend the control. │ │ │ │ │ */ │ │ │ │ │ - configureLayers: function() { │ │ │ │ │ - │ │ │ │ │ - if (this.layers.length == this.map.layers.length) { │ │ │ │ │ - this.map.events.unregister('addlayer', this, this.configureLayers); │ │ │ │ │ - │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - var c = this.layers.charAt(i); │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.layerName = options.layerName || OpenLayers.i18n("Graticule"); │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ │ │ │ │ │ - if (c == "B") { │ │ │ │ │ - this.map.setBaseLayer(layer); │ │ │ │ │ - } else if ((c == "T") || (c == "F")) { │ │ │ │ │ - layer.setVisibility(c == "T"); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + this.labelSymbolizer.stroke = false; │ │ │ │ │ + this.labelSymbolizer.fill = false; │ │ │ │ │ + this.labelSymbolizer.label = "${label}"; │ │ │ │ │ + this.labelSymbolizer.labelAlign = "${labelAlign}"; │ │ │ │ │ + this.labelSymbolizer.labelXOffset = "${xOffset}"; │ │ │ │ │ + this.labelSymbolizer.labelYOffset = "${yOffset}"; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ArgParser" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Geolocate.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Projection.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Geolocate │ │ │ │ │ - * The Geolocate control wraps w3c geolocation API into control that can be │ │ │ │ │ - * bound to a map, and generate events on location update │ │ │ │ │ - * │ │ │ │ │ - * To use this control requires to load the proj4js library if the projection │ │ │ │ │ - * of the map is not EPSG:4326 or EPSG:900913. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Geolocate = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * locationupdated - Triggered when browser return a new position. Listeners will │ │ │ │ │ - * receive an object with a 'position' property which is the browser.geolocation.position │ │ │ │ │ - * native object, as well as a 'point' property which is the location transformed in the │ │ │ │ │ - * current map projection. │ │ │ │ │ - * locationfailed - Triggered when geolocation has failed │ │ │ │ │ - * locationuncapable - Triggered when control is activated on a browser │ │ │ │ │ - * which doesn't support geolocation │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: geolocation │ │ │ │ │ - * {Object} The geolocation engine, as a property to be possibly mocked. │ │ │ │ │ - * This is set lazily to avoid a memory leak in IE9. │ │ │ │ │ - */ │ │ │ │ │ - geolocation: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: available │ │ │ │ │ - * {Boolean} The navigator.geolocation object is available. │ │ │ │ │ - */ │ │ │ │ │ - available: ('geolocation' in navigator), │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: bind │ │ │ │ │ - * {Boolean} If true, map center will be set on location update. │ │ │ │ │ - */ │ │ │ │ │ - bind: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: watch │ │ │ │ │ - * {Boolean} If true, position will be update regularly. │ │ │ │ │ - */ │ │ │ │ │ - watch: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geolocationOptions │ │ │ │ │ - * {Object} Options to pass to the navigator's geolocation API. See │ │ │ │ │ - * <http://dev.w3.org/geo/api/spec-source.html>. No specific │ │ │ │ │ - * option is passed to the geolocation API by default. │ │ │ │ │ - */ │ │ │ │ │ - geolocationOptions: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Geolocate │ │ │ │ │ - * Create a new control to deal with browser geolocation API │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ */ │ │ │ │ │ destroy: function() { │ │ │ │ │ this.deactivate(); │ │ │ │ │ OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Activates the control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The control was effectively activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (this.available && !this.geolocation) { │ │ │ │ │ - // set lazily to avoid IE9 memory leak │ │ │ │ │ - this.geolocation = navigator.geolocation; │ │ │ │ │ - } │ │ │ │ │ - if (!this.geolocation) { │ │ │ │ │ - this.events.triggerEvent("locationuncapable"); │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - if (this.watch) { │ │ │ │ │ - this.watchId = this.geolocation.watchPosition( │ │ │ │ │ - OpenLayers.Function.bind(this.geolocate, this), │ │ │ │ │ - OpenLayers.Function.bind(this.failure, this), │ │ │ │ │ - this.geolocationOptions │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - this.getCurrentLocation(); │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ + if (this.gratLayer) { │ │ │ │ │ + this.gratLayer.destroy(); │ │ │ │ │ + this.gratLayer = null; │ │ │ │ │ } │ │ │ │ │ - return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Deactivates the control. │ │ │ │ │ + * Method: draw │ │ │ │ │ * │ │ │ │ │ + * initializes the graticule layer and does the initial update │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The control was effectively deactivated. │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active && this.watchId !== null) { │ │ │ │ │ - this.geolocation.clearWatch(this.watchId); │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (!this.gratLayer) { │ │ │ │ │ + var gratStyle = new OpenLayers.Style({}, { │ │ │ │ │ + rules: [new OpenLayers.Rule({ │ │ │ │ │ + 'symbolizer': { │ │ │ │ │ + "Point": this.labelSymbolizer, │ │ │ │ │ + "Line": this.lineSymbolizer │ │ │ │ │ + } │ │ │ │ │ + })] │ │ │ │ │ + }); │ │ │ │ │ + this.gratLayer = new OpenLayers.Layer.Vector(this.layerName, { │ │ │ │ │ + styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ + 'default': gratStyle │ │ │ │ │ + }), │ │ │ │ │ + visibility: this.visible, │ │ │ │ │ + displayInLayerSwitcher: this.displayInLayerSwitcher │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Control.prototype.deactivate.apply( │ │ │ │ │ - this, arguments │ │ │ │ │ - ); │ │ │ │ │ + return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: geolocate │ │ │ │ │ - * Activates the control. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ */ │ │ │ │ │ - geolocate: function(position) { │ │ │ │ │ - var center = new OpenLayers.LonLat( │ │ │ │ │ - position.coords.longitude, │ │ │ │ │ - position.coords.latitude │ │ │ │ │ - ).transform( │ │ │ │ │ - new OpenLayers.Projection("EPSG:4326"), │ │ │ │ │ - this.map.getProjectionObject() │ │ │ │ │ - ); │ │ │ │ │ - if (this.bind) { │ │ │ │ │ - this.map.setCenter(center); │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.map.addLayer(this.gratLayer); │ │ │ │ │ + this.map.events.register('moveend', this, this.update); │ │ │ │ │ + this.update(); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - this.events.triggerEvent("locationupdated", { │ │ │ │ │ - position: position, │ │ │ │ │ - point: new OpenLayers.Geometry.Point( │ │ │ │ │ - center.lon, center.lat │ │ │ │ │ - ) │ │ │ │ │ - }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getCurrentLocation │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Returns true if a event will be fired (successfull │ │ │ │ │ - * registration) │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ */ │ │ │ │ │ - getCurrentLocation: function() { │ │ │ │ │ - if (!this.active || this.watch) { │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.map.events.unregister('moveend', this, this.update); │ │ │ │ │ + this.map.removeLayer(this.gratLayer); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ return false; │ │ │ │ │ } │ │ │ │ │ - this.geolocation.getCurrentPosition( │ │ │ │ │ - OpenLayers.Function.bind(this.geolocate, this), │ │ │ │ │ - OpenLayers.Function.bind(this.failure, this), │ │ │ │ │ - this.geolocationOptions │ │ │ │ │ - ); │ │ │ │ │ - return true; │ │ │ │ │ }, │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: failure │ │ │ │ │ - * method called on browser's geolocation failure │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - failure: function(error) { │ │ │ │ │ - this.events.triggerEvent("locationfailed", { │ │ │ │ │ - error: error │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Geolocate" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/CacheWrite.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Request.js │ │ │ │ │ - * @requires OpenLayers/Console.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.CacheWrite │ │ │ │ │ - * A control for caching image tiles in the browser's local storage. The │ │ │ │ │ - * <OpenLayers.Control.CacheRead> control is used to fetch and use the cached │ │ │ │ │ - * tile images. │ │ │ │ │ - * │ │ │ │ │ - * Note: Before using this control on any layer that is not your own, make sure │ │ │ │ │ - * that the terms of service of the tile provider allow local storage of tiles. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.CacheWrite = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * To register events in the constructor, configure <eventListeners>. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ + * Method: update │ │ │ │ │ * │ │ │ │ │ - * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * cachefull - Triggered when the cache is full. Listeners receive an │ │ │ │ │ - * object with a tile property as first argument. The tile references │ │ │ │ │ - * the tile that couldn't be cached. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: eventListeners │ │ │ │ │ - * {Object} Object with event listeners, keyed by event name. An optional │ │ │ │ │ - * scope property defines the scope that listeners will be executed in. │ │ │ │ │ + * calculates the grid to be displayed and actually draws it │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ + update: function() { │ │ │ │ │ + //wait for the map to be initialized before proceeding │ │ │ │ │ + var mapBounds = this.map.getExtent(); │ │ │ │ │ + if (!mapBounds) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: layers │ │ │ │ │ - * {Array(<OpenLayers.Layer.Grid>)}. Optional. If provided, caching │ │ │ │ │ - * will be enabled for these layers only, otherwise for all cacheable │ │ │ │ │ - * layers. │ │ │ │ │ - */ │ │ │ │ │ - layers: null, │ │ │ │ │ + //clear out the old grid │ │ │ │ │ + this.gratLayer.destroyFeatures(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: imageFormat │ │ │ │ │ - * {String} The image format used for caching. The default is "image/png". │ │ │ │ │ - * Supported formats depend on the user agent. If an unsupported │ │ │ │ │ - * <imageFormat> is provided, "image/png" will be used. For aerial │ │ │ │ │ - * imagery, "image/jpeg" is recommended. │ │ │ │ │ - */ │ │ │ │ │ - imageFormat: "image/png", │ │ │ │ │ + //get the projection objects required │ │ │ │ │ + var llProj = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + var mapProj = this.map.getProjectionObject(); │ │ │ │ │ + var mapRes = this.map.getResolution(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: quotaRegEx │ │ │ │ │ - * {RegExp} │ │ │ │ │ - */ │ │ │ │ │ - quotaRegEx: (/quota/i), │ │ │ │ │ + //if the map is in lon/lat, then the lines are straight and only one │ │ │ │ │ + //point is required │ │ │ │ │ + if (mapProj.proj && mapProj.proj.projName == "longlat") { │ │ │ │ │ + this.numPoints = 1; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.CacheWrite │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Object with API properties for this control. │ │ │ │ │ - */ │ │ │ │ │ + //get the map center in EPSG:4326 │ │ │ │ │ + var mapCenter = this.map.getCenter(); //lon and lat here are really map x and y │ │ │ │ │ + var mapCenterLL = new OpenLayers.Pixel(mapCenter.lon, mapCenter.lat); │ │ │ │ │ + OpenLayers.Projection.transform(mapCenterLL, mapProj, llProj); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the control. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - var i, layers = this.layers || map.layers; │ │ │ │ │ - for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - this.addLayer({ │ │ │ │ │ - layer: layers[i] │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - if (!this.layers) { │ │ │ │ │ - map.events.on({ │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - removeLayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ + /* This block of code determines the lon/lat interval to use for the │ │ │ │ │ + * grid by calculating the diagonal size of one grid cell at the map │ │ │ │ │ + * center. Iterates through the intervals array until the diagonal │ │ │ │ │ + * length is less than the targetSize option. │ │ │ │ │ + */ │ │ │ │ │ + //find lat/lon interval that results in a grid of less than the target size │ │ │ │ │ + var testSq = this.targetSize * mapRes; │ │ │ │ │ + testSq *= testSq; //compare squares rather than doing a square root to save time │ │ │ │ │ + var llInterval; │ │ │ │ │ + for (var i = 0; i < this.intervals.length; ++i) { │ │ │ │ │ + llInterval = this.intervals[i]; //could do this for both x and y?? │ │ │ │ │ + var delta = llInterval / 2; │ │ │ │ │ + var p1 = mapCenterLL.offset({ │ │ │ │ │ + x: -delta, │ │ │ │ │ + y: -delta │ │ │ │ │ + }); //test coords in EPSG:4326 space │ │ │ │ │ + var p2 = mapCenterLL.offset({ │ │ │ │ │ + x: delta, │ │ │ │ │ + y: delta │ │ │ │ │ }); │ │ │ │ │ + OpenLayers.Projection.transform(p1, llProj, mapProj); // convert them back to map projection │ │ │ │ │ + OpenLayers.Projection.transform(p2, llProj, mapProj); │ │ │ │ │ + var distSq = (p1.x - p2.x) * (p1.x - p2.x) + (p1.y - p2.y) * (p1.y - p2.y); │ │ │ │ │ + if (distSq <= testSq) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addLayer │ │ │ │ │ - * Adds a layer to the control. Once added, tiles requested for this layer │ │ │ │ │ - * will be cached. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} Object with a layer property referencing an │ │ │ │ │ - * <OpenLayers.Layer> instance │ │ │ │ │ - */ │ │ │ │ │ - addLayer: function(evt) { │ │ │ │ │ - evt.layer.events.on({ │ │ │ │ │ - tileloadstart: this.makeSameOrigin, │ │ │ │ │ - tileloaded: this.onTileLoaded, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + //alert(llInterval); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeLayer │ │ │ │ │ - * Removes a layer from the control. Once removed, tiles requested for this │ │ │ │ │ - * layer will no longer be cached. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} Object with a layer property referencing an │ │ │ │ │ - * <OpenLayers.Layer> instance │ │ │ │ │ - */ │ │ │ │ │ - removeLayer: function(evt) { │ │ │ │ │ - evt.layer.events.un({ │ │ │ │ │ - tileloadstart: this.makeSameOrigin, │ │ │ │ │ - tileloaded: this.onTileLoaded, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + //round the LL center to an even number based on the interval │ │ │ │ │ + mapCenterLL.x = Math.floor(mapCenterLL.x / llInterval) * llInterval; │ │ │ │ │ + mapCenterLL.y = Math.floor(mapCenterLL.y / llInterval) * llInterval; │ │ │ │ │ + //TODO adjust for minutses/seconds? │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: makeSameOrigin │ │ │ │ │ - * If the tile does not have CORS image loading enabled and is from a │ │ │ │ │ - * different origin, use OpenLayers.ProxyHost to make it a same origin url. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - */ │ │ │ │ │ - makeSameOrigin: function(evt) { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - var tile = evt.tile; │ │ │ │ │ - if (tile instanceof OpenLayers.Tile.Image && │ │ │ │ │ - !tile.crossOriginKeyword && │ │ │ │ │ - tile.url.substr(0, 5) !== "data:") { │ │ │ │ │ - var sameOriginUrl = OpenLayers.Request.makeSameOrigin( │ │ │ │ │ - tile.url, OpenLayers.ProxyHost │ │ │ │ │ - ); │ │ │ │ │ - OpenLayers.Control.CacheWrite.urlMap[sameOriginUrl] = tile.url; │ │ │ │ │ - tile.url = sameOriginUrl; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /* The following 2 blocks calculate the nodes of the grid along a │ │ │ │ │ + * line of constant longitude (then latitiude) running through the │ │ │ │ │ + * center of the map until it reaches the map edge. The calculation │ │ │ │ │ + * goes from the center in both directions to the edge. │ │ │ │ │ + */ │ │ │ │ │ + //get the central longitude line, increment the latitude │ │ │ │ │ + var iter = 0; │ │ │ │ │ + var centerLonPoints = [mapCenterLL.clone()]; │ │ │ │ │ + var newPoint = mapCenterLL.clone(); │ │ │ │ │ + var mapXY; │ │ │ │ │ + do { │ │ │ │ │ + newPoint = newPoint.offset({ │ │ │ │ │ + x: 0, │ │ │ │ │ + y: llInterval │ │ │ │ │ + }); │ │ │ │ │ + mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ + centerLonPoints.unshift(newPoint); │ │ │ │ │ + } while (mapBounds.containsPixel(mapXY) && ++iter < 1000); │ │ │ │ │ + newPoint = mapCenterLL.clone(); │ │ │ │ │ + do { │ │ │ │ │ + newPoint = newPoint.offset({ │ │ │ │ │ + x: 0, │ │ │ │ │ + y: -llInterval │ │ │ │ │ + }); │ │ │ │ │ + mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ + centerLonPoints.push(newPoint); │ │ │ │ │ + } while (mapBounds.containsPixel(mapXY) && ++iter < 1000); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: onTileLoaded │ │ │ │ │ - * Decides whether a tile can be cached and calls the cache method. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - onTileLoaded: function(evt) { │ │ │ │ │ - if (this.active && !evt.aborted && │ │ │ │ │ - evt.tile instanceof OpenLayers.Tile.Image && │ │ │ │ │ - evt.tile.url.substr(0, 5) !== 'data:') { │ │ │ │ │ - this.cache({ │ │ │ │ │ - tile: evt.tile │ │ │ │ │ + //get the central latitude line, increment the longitude │ │ │ │ │ + iter = 0; │ │ │ │ │ + var centerLatPoints = [mapCenterLL.clone()]; │ │ │ │ │ + newPoint = mapCenterLL.clone(); │ │ │ │ │ + do { │ │ │ │ │ + newPoint = newPoint.offset({ │ │ │ │ │ + x: -llInterval, │ │ │ │ │ + y: 0 │ │ │ │ │ }); │ │ │ │ │ - delete OpenLayers.Control.CacheWrite.urlMap[evt.tile.url]; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ + centerLatPoints.unshift(newPoint); │ │ │ │ │ + } while (mapBounds.containsPixel(mapXY) && ++iter < 1000); │ │ │ │ │ + newPoint = mapCenterLL.clone(); │ │ │ │ │ + do { │ │ │ │ │ + newPoint = newPoint.offset({ │ │ │ │ │ + x: llInterval, │ │ │ │ │ + y: 0 │ │ │ │ │ + }); │ │ │ │ │ + mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ + centerLatPoints.push(newPoint); │ │ │ │ │ + } while (mapBounds.containsPixel(mapXY) && ++iter < 1000); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: cache │ │ │ │ │ - * Adds a tile to the cache. When the cache is full, the "cachefull" event │ │ │ │ │ - * is triggered. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} Object with a tile property, tile being the │ │ │ │ │ - * <OpenLayers.Tile.Image> with the data to add to the cache │ │ │ │ │ - */ │ │ │ │ │ - cache: function(obj) { │ │ │ │ │ - if (window.localStorage) { │ │ │ │ │ - var tile = obj.tile; │ │ │ │ │ - try { │ │ │ │ │ - var canvasContext = tile.getCanvasContext(); │ │ │ │ │ - if (canvasContext) { │ │ │ │ │ - var urlMap = OpenLayers.Control.CacheWrite.urlMap; │ │ │ │ │ - var url = urlMap[tile.url] || tile.url; │ │ │ │ │ - window.localStorage.setItem( │ │ │ │ │ - "olCache_" + url, │ │ │ │ │ - canvasContext.canvas.toDataURL(this.imageFormat) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } catch (e) { │ │ │ │ │ - // local storage full or CORS violation │ │ │ │ │ - var reason = e.name || e.message; │ │ │ │ │ - if (reason && this.quotaRegEx.test(reason)) { │ │ │ │ │ - this.events.triggerEvent("cachefull", { │ │ │ │ │ - tile: tile │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Console.error(e.toString()); │ │ │ │ │ + //now generate a line for each node in the central lat and lon lines │ │ │ │ │ + //first loop over constant longitude │ │ │ │ │ + var lines = []; │ │ │ │ │ + for (var i = 0; i < centerLatPoints.length; ++i) { │ │ │ │ │ + var lon = centerLatPoints[i].x; │ │ │ │ │ + var pointList = []; │ │ │ │ │ + var labelPoint = null; │ │ │ │ │ + var latEnd = Math.min(centerLonPoints[0].y, 90); │ │ │ │ │ + var latStart = Math.max(centerLonPoints[centerLonPoints.length - 1].y, -90); │ │ │ │ │ + var latDelta = (latEnd - latStart) / this.numPoints; │ │ │ │ │ + var lat = latStart; │ │ │ │ │ + for (var j = 0; j <= this.numPoints; ++j) { │ │ │ │ │ + var gridPoint = new OpenLayers.Geometry.Point(lon, lat); │ │ │ │ │ + gridPoint.transform(llProj, mapProj); │ │ │ │ │ + pointList.push(gridPoint); │ │ │ │ │ + lat += latDelta; │ │ │ │ │ + if (gridPoint.y >= mapBounds.bottom && !labelPoint) { │ │ │ │ │ + labelPoint = gridPoint; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * The destroy method is used to perform any clean up before the control │ │ │ │ │ - * is dereferenced. Typically this is where event listeners are removed │ │ │ │ │ - * to prevent memory leaks. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.layers || this.map) { │ │ │ │ │ - var i, layers = this.layers || this.map.layers; │ │ │ │ │ - for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - this.removeLayer({ │ │ │ │ │ - layer: layers[i] │ │ │ │ │ - }); │ │ │ │ │ + if (this.labelled) { │ │ │ │ │ + //keep track of when this grid line crosses the map bounds to set │ │ │ │ │ + //the label position │ │ │ │ │ + //labels along the bottom, add 10 pixel offset up into the map │ │ │ │ │ + //TODO add option for labels on top │ │ │ │ │ + var labelPos = new OpenLayers.Geometry.Point(labelPoint.x, mapBounds.bottom); │ │ │ │ │ + var labelAttrs = { │ │ │ │ │ + value: lon, │ │ │ │ │ + label: this.labelled ? OpenLayers.Util.getFormattedLonLat(lon, "lon", this.labelFormat) : "", │ │ │ │ │ + labelAlign: "cb", │ │ │ │ │ + xOffset: 0, │ │ │ │ │ + yOffset: 2 │ │ │ │ │ + }; │ │ │ │ │ + this.gratLayer.addFeatures(new OpenLayers.Feature.Vector(labelPos, labelAttrs)); │ │ │ │ │ } │ │ │ │ │ + var geom = new OpenLayers.Geometry.LineString(pointList); │ │ │ │ │ + lines.push(new OpenLayers.Feature.Vector(geom)); │ │ │ │ │ } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - removeLayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.CacheWrite" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * APIFunction: OpenLayers.Control.CacheWrite.clearCache │ │ │ │ │ - * Clears all tiles cached with <OpenLayers.Control.CacheWrite> from the cache. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.CacheWrite.clearCache = function() { │ │ │ │ │ - if (!window.localStorage) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var i, key; │ │ │ │ │ - for (i = window.localStorage.length - 1; i >= 0; --i) { │ │ │ │ │ - key = window.localStorage.key(i); │ │ │ │ │ - if (key.substr(0, 8) === "olCache_") { │ │ │ │ │ - window.localStorage.removeItem(key); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Property: OpenLayers.Control.CacheWrite.urlMap │ │ │ │ │ - * {Object} Mapping of same origin urls to cache url keys. Entries will be │ │ │ │ │ - * deleted as soon as a tile was cached. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.CacheWrite.urlMap = {}; │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Button.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Button │ │ │ │ │ - * The Button control is a very simple push-button, for use with │ │ │ │ │ - * <OpenLayers.Control.Panel>. │ │ │ │ │ - * When clicked, the function trigger() is executed. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - * │ │ │ │ │ - * Use: │ │ │ │ │ - * (code) │ │ │ │ │ - * var button = new OpenLayers.Control.Button({ │ │ │ │ │ - * displayClass: "MyButton", trigger: myFunction │ │ │ │ │ - * }); │ │ │ │ │ - * panel.addControls([button]); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Will create a button with CSS class MyButtonItemInactive, that │ │ │ │ │ - * will call the function MyFunction() when clicked. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Button = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - /** │ │ │ │ │ - * Property: type │ │ │ │ │ - * {Integer} OpenLayers.Control.TYPE_BUTTON. │ │ │ │ │ - */ │ │ │ │ │ - type: OpenLayers.Control.TYPE_BUTTON, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: trigger │ │ │ │ │ - * Called by a control panel when the button is clicked. │ │ │ │ │ - */ │ │ │ │ │ - trigger: function() {}, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Button" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Pan.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control/Button.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Pan │ │ │ │ │ - * The Pan control is a single button to pan the map in one direction. For │ │ │ │ │ - * a more complete control see <OpenLayers.Control.PanPanel>. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Pan = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: slideFactor │ │ │ │ │ - * {Integer} Number of pixels by which we'll pan the map in any direction │ │ │ │ │ - * on clicking the arrow buttons, defaults to 50. If you want to pan │ │ │ │ │ - * by some ratio of the map dimensions, use <slideRatio> instead. │ │ │ │ │ - */ │ │ │ │ │ - slideFactor: 50, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: slideRatio │ │ │ │ │ - * {Number} The fraction of map width/height by which we'll pan the map │ │ │ │ │ - * on clicking the arrow buttons. Default is null. If set, will │ │ │ │ │ - * override <slideFactor>. E.g. if slideRatio is .5, then Pan Up will │ │ │ │ │ - * pan up half the map height. │ │ │ │ │ - */ │ │ │ │ │ - slideRatio: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: direction │ │ │ │ │ - * {String} in {'North', 'South', 'East', 'West'} │ │ │ │ │ - */ │ │ │ │ │ - direction: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Pan │ │ │ │ │ - * Control which handles the panning (in any of the cardinal directions) │ │ │ │ │ - * of the map by a set px distance. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * direction - {String} The direction this button should pan. │ │ │ │ │ - * options - {Object} An optional object whose properties will be used │ │ │ │ │ - * to extend the control. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(direction, options) { │ │ │ │ │ - │ │ │ │ │ - this.direction = direction; │ │ │ │ │ - this.CLASS_NAME += this.direction; │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: trigger │ │ │ │ │ - */ │ │ │ │ │ - trigger: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - var getSlideFactor = OpenLayers.Function.bind(function(dim) { │ │ │ │ │ - return this.slideRatio ? │ │ │ │ │ - this.map.getSize()[dim] * this.slideRatio : │ │ │ │ │ - this.slideFactor; │ │ │ │ │ - }, this); │ │ │ │ │ │ │ │ │ │ - switch (this.direction) { │ │ │ │ │ - case OpenLayers.Control.Pan.NORTH: │ │ │ │ │ - this.map.pan(0, -getSlideFactor("h")); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Control.Pan.SOUTH: │ │ │ │ │ - this.map.pan(0, getSlideFactor("h")); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Control.Pan.WEST: │ │ │ │ │ - this.map.pan(-getSlideFactor("w"), 0); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Control.Pan.EAST: │ │ │ │ │ - this.map.pan(getSlideFactor("w"), 0); │ │ │ │ │ - break; │ │ │ │ │ + //now draw the lines of constant latitude │ │ │ │ │ + for (var j = 0; j < centerLonPoints.length; ++j) { │ │ │ │ │ + lat = centerLonPoints[j].y; │ │ │ │ │ + if (lat < -90 || lat > 90) { //latitudes only valid between -90 and 90 │ │ │ │ │ + continue; │ │ │ │ │ } │ │ │ │ │ + var pointList = []; │ │ │ │ │ + var lonStart = centerLatPoints[0].x; │ │ │ │ │ + var lonEnd = centerLatPoints[centerLatPoints.length - 1].x; │ │ │ │ │ + var lonDelta = (lonEnd - lonStart) / this.numPoints; │ │ │ │ │ + var lon = lonStart; │ │ │ │ │ + var labelPoint = null; │ │ │ │ │ + for (var i = 0; i <= this.numPoints; ++i) { │ │ │ │ │ + var gridPoint = new OpenLayers.Geometry.Point(lon, lat); │ │ │ │ │ + gridPoint.transform(llProj, mapProj); │ │ │ │ │ + pointList.push(gridPoint); │ │ │ │ │ + lon += lonDelta; │ │ │ │ │ + if (gridPoint.x < mapBounds.right) { │ │ │ │ │ + labelPoint = gridPoint; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.labelled) { │ │ │ │ │ + //keep track of when this grid line crosses the map bounds to set │ │ │ │ │ + //the label position │ │ │ │ │ + //labels along the right, 30 pixel offset left into the map │ │ │ │ │ + //TODO add option for labels on left │ │ │ │ │ + var labelPos = new OpenLayers.Geometry.Point(mapBounds.right, labelPoint.y); │ │ │ │ │ + var labelAttrs = { │ │ │ │ │ + value: lat, │ │ │ │ │ + label: this.labelled ? OpenLayers.Util.getFormattedLonLat(lat, "lat", this.labelFormat) : "", │ │ │ │ │ + labelAlign: "rb", │ │ │ │ │ + xOffset: -2, │ │ │ │ │ + yOffset: 2 │ │ │ │ │ + }; │ │ │ │ │ + this.gratLayer.addFeatures(new OpenLayers.Feature.Vector(labelPos, labelAttrs)); │ │ │ │ │ + } │ │ │ │ │ + var geom = new OpenLayers.Geometry.LineString(pointList); │ │ │ │ │ + lines.push(new OpenLayers.Feature.Vector(geom)); │ │ │ │ │ } │ │ │ │ │ + this.gratLayer.addFeatures(lines); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Pan" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Control.Pan.NORTH = "North"; │ │ │ │ │ -OpenLayers.Control.Pan.SOUTH = "South"; │ │ │ │ │ -OpenLayers.Control.Pan.EAST = "East"; │ │ │ │ │ -OpenLayers.Control.Pan.WEST = "West"; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/PanPanel.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control/Panel.js │ │ │ │ │ - * @requires OpenLayers/Control/Pan.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.PanPanel │ │ │ │ │ - * The PanPanel is visible control for panning the map North, South, East or │ │ │ │ │ - * West in small steps. By default it is drawn in the top left corner of the │ │ │ │ │ - * map. │ │ │ │ │ - * │ │ │ │ │ - * Note: │ │ │ │ │ - * If you wish to use this class with the default images and you want │ │ │ │ │ - * it to look nice in ie6, you should add the following, conditionally │ │ │ │ │ - * added css stylesheet to your HTML file: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * <!--[if lte IE 6]> │ │ │ │ │ - * <link rel="stylesheet" href="../theme/default/ie6-style.css" type="text/css" /> │ │ │ │ │ - * <![endif]--> │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control.Panel> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.PanPanel = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: slideFactor │ │ │ │ │ - * {Integer} Number of pixels by which we'll pan the map in any direction │ │ │ │ │ - * on clicking the arrow buttons, defaults to 50. If you want to pan │ │ │ │ │ - * by some ratio of the map dimensions, use <slideRatio> instead. │ │ │ │ │ - */ │ │ │ │ │ - slideFactor: 50, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: slideRatio │ │ │ │ │ - * {Number} The fraction of map width/height by which we'll pan the map │ │ │ │ │ - * on clicking the arrow buttons. Default is null. If set, will │ │ │ │ │ - * override <slideFactor>. E.g. if slideRatio is .5, then Pan Up will │ │ │ │ │ - * pan up half the map height. │ │ │ │ │ - */ │ │ │ │ │ - slideRatio: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.PanPanel │ │ │ │ │ - * Add the four directional pan buttons. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be used │ │ │ │ │ - * to extend the control. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ - var options = { │ │ │ │ │ - slideFactor: this.slideFactor, │ │ │ │ │ - slideRatio: this.slideRatio │ │ │ │ │ - }; │ │ │ │ │ - this.addControls([ │ │ │ │ │ - new OpenLayers.Control.Pan(OpenLayers.Control.Pan.NORTH, options), │ │ │ │ │ - new OpenLayers.Control.Pan(OpenLayers.Control.Pan.SOUTH, options), │ │ │ │ │ - new OpenLayers.Control.Pan(OpenLayers.Control.Pan.EAST, options), │ │ │ │ │ - new OpenLayers.Control.Pan(OpenLayers.Control.Pan.WEST, options) │ │ │ │ │ - ]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.PanPanel" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/ZoomIn.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control/Button.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.ZoomIn │ │ │ │ │ - * The ZoomIn control is a button to increase the zoom level of a map. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ZoomIn = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: trigger │ │ │ │ │ - */ │ │ │ │ │ - trigger: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.zoomIn(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomIn" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/ZoomOut.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control/Button.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.ZoomOut │ │ │ │ │ - * The ZoomOut control is a button to decrease the zoom level of a map. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ZoomOut = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: trigger │ │ │ │ │ - */ │ │ │ │ │ - trigger: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.zoomOut(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomOut" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Graticule" │ │ │ │ │ }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/ZoomToMaxExtent.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control/Button.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.ZoomToMaxExtent │ │ │ │ │ - * The ZoomToMaxExtent control is a button that zooms out to the maximum │ │ │ │ │ - * extent of the map. It is designed to be used with a │ │ │ │ │ - * <OpenLayers.Control.Panel>. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ZoomToMaxExtent = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: trigger │ │ │ │ │ - * │ │ │ │ │ - * Called whenever this control is being rendered inside of a panel and a │ │ │ │ │ - * click occurs on this controls element. Actually zooms to the maximum │ │ │ │ │ - * extent of this controls map. │ │ │ │ │ - */ │ │ │ │ │ - trigger: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.zoomToMaxExtent(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomToMaxExtent" │ │ │ │ │ -}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/ZoomPanel.js │ │ │ │ │ + OpenLayers/Control/Navigation.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Control/Panel.js │ │ │ │ │ - * @requires OpenLayers/Control/ZoomIn.js │ │ │ │ │ - * @requires OpenLayers/Control/ZoomOut.js │ │ │ │ │ - * @requires OpenLayers/Control/ZoomToMaxExtent.js │ │ │ │ │ + * @requires OpenLayers/Control/ZoomBox.js │ │ │ │ │ + * @requires OpenLayers/Control/DragPan.js │ │ │ │ │ + * @requires OpenLayers/Handler/MouseWheel.js │ │ │ │ │ + * @requires OpenLayers/Handler/Click.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.ZoomPanel │ │ │ │ │ - * The ZoomPanel control is a compact collecton of 3 zoom controls: a │ │ │ │ │ - * <OpenLayers.Control.ZoomIn>, a <OpenLayers.Control.ZoomToMaxExtent>, and a │ │ │ │ │ - * <OpenLayers.Control.ZoomOut>. By default it is drawn in the upper left │ │ │ │ │ - * corner of the map. │ │ │ │ │ - * │ │ │ │ │ - * Note: │ │ │ │ │ - * If you wish to use this class with the default images and you want │ │ │ │ │ - * it to look nice in ie6, you should add the following, conditionally │ │ │ │ │ - * added css stylesheet to your HTML file: │ │ │ │ │ + * Class: OpenLayers.Control.Navigation │ │ │ │ │ + * The navigation control handles map browsing with mouse events (dragging, │ │ │ │ │ + * double-clicking, and scrolling the wheel). Create a new navigation │ │ │ │ │ + * control with the <OpenLayers.Control.Navigation> control. │ │ │ │ │ * │ │ │ │ │ - * (code) │ │ │ │ │ - * <!--[if lte IE 6]> │ │ │ │ │ - * <link rel="stylesheet" href="../theme/default/ie6-style.css" type="text/css" /> │ │ │ │ │ - * <![endif]--> │ │ │ │ │ - * (end) │ │ │ │ │ + * Note that this control is added to the map by default (if no controls │ │ │ │ │ + * array is sent in the options object to the <OpenLayers.Map> │ │ │ │ │ + * constructor). │ │ │ │ │ * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control.Panel> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ZoomPanel = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.ZoomPanel │ │ │ │ │ - * Add the three zooming controls. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be used │ │ │ │ │ - * to extend the control. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.addControls([ │ │ │ │ │ - new OpenLayers.Control.ZoomIn(), │ │ │ │ │ - new OpenLayers.Control.ZoomToMaxExtent(), │ │ │ │ │ - new OpenLayers.Control.ZoomOut() │ │ │ │ │ - ]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomPanel" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/GetFeature.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Handler/Click.js │ │ │ │ │ - * @requires OpenLayers/Handler/Box.js │ │ │ │ │ - * @requires OpenLayers/Handler/Hover.js │ │ │ │ │ - * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.GetFeature │ │ │ │ │ - * Gets vector features for locations underneath the mouse cursor. Can be │ │ │ │ │ - * configured to act on click, hover or dragged boxes. Uses an │ │ │ │ │ - * <OpenLayers.Protocol> that supports spatial filters to retrieve │ │ │ │ │ - * features from a server and fires events that notify applications of the │ │ │ │ │ - * selected features. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ + * Inherits: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.GetFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: protocol │ │ │ │ │ - * {<OpenLayers.Protocol>} Required. The protocol used for fetching │ │ │ │ │ - * features. │ │ │ │ │ - */ │ │ │ │ │ - protocol: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: multipleKey │ │ │ │ │ - * {String} An event modifier ('altKey' or 'shiftKey') that temporarily sets │ │ │ │ │ - * the <multiple> property to true. Default is null. │ │ │ │ │ - */ │ │ │ │ │ - multipleKey: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: toggleKey │ │ │ │ │ - * {String} An event modifier ('altKey' or 'shiftKey') that temporarily sets │ │ │ │ │ - * the <toggle> property to true. Default is null. │ │ │ │ │ - */ │ │ │ │ │ - toggleKey: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: modifiers │ │ │ │ │ - * {Object} The event modifiers to use, according to the current event │ │ │ │ │ - * being handled by this control's handlers │ │ │ │ │ - */ │ │ │ │ │ - modifiers: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: multiple │ │ │ │ │ - * {Boolean} Allow selection of multiple geometries. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - multiple: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: click │ │ │ │ │ - * {Boolean} Use a click handler for selecting/unselecting features. If │ │ │ │ │ - * both <click> and <box> are set to true, the click handler takes │ │ │ │ │ - * precedence over the box handler if a box with zero extent was │ │ │ │ │ - * selected. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - click: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: single │ │ │ │ │ - * {Boolean} Tells whether select by click should select a single │ │ │ │ │ - * feature. If set to false, all matching features are selected. │ │ │ │ │ - * If set to true, only the best matching feature is selected. │ │ │ │ │ - * This option has an effect only of the <click> option is set │ │ │ │ │ - * to true. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - single: true, │ │ │ │ │ +OpenLayers.Control.Navigation = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: clickout │ │ │ │ │ - * {Boolean} Unselect features when clicking outside any feature. │ │ │ │ │ - * Applies only if <click> is true. Default is true. │ │ │ │ │ + /** │ │ │ │ │ + * Property: dragPan │ │ │ │ │ + * {<OpenLayers.Control.DragPan>} │ │ │ │ │ */ │ │ │ │ │ - clickout: true, │ │ │ │ │ + dragPan: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: toggle │ │ │ │ │ - * {Boolean} Unselect a selected feature on click. Applies only if │ │ │ │ │ - * <click> is true. Default is false. │ │ │ │ │ + * APIProperty: dragPanOptions │ │ │ │ │ + * {Object} Options passed to the DragPan control. │ │ │ │ │ */ │ │ │ │ │ - toggle: false, │ │ │ │ │ + dragPanOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: clickTolerance │ │ │ │ │ - * {Integer} Tolerance for the filter query in pixels. This has the │ │ │ │ │ - * same effect as the tolerance parameter on WMS GetFeatureInfo │ │ │ │ │ - * requests. Will be ignored for box selections. Applies only if │ │ │ │ │ - * <click> or <hover> is true. Default is 5. Note that this not │ │ │ │ │ - * only affects requests on click, but also on hover. │ │ │ │ │ + * Property: pinchZoom │ │ │ │ │ + * {<OpenLayers.Control.PinchZoom>} │ │ │ │ │ */ │ │ │ │ │ - clickTolerance: 5, │ │ │ │ │ + pinchZoom: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: hover │ │ │ │ │ - * {Boolean} Send feature requests on mouse moves. Default is false. │ │ │ │ │ + * APIProperty: pinchZoomOptions │ │ │ │ │ + * {Object} Options passed to the PinchZoom control. │ │ │ │ │ */ │ │ │ │ │ - hover: false, │ │ │ │ │ + pinchZoomOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: box │ │ │ │ │ - * {Boolean} Allow feature selection by drawing a box. If set to │ │ │ │ │ - * true set <click> to false to disable the click handler and │ │ │ │ │ - * rely on the box handler only, even for "zero extent" boxes. │ │ │ │ │ - * See the description of the <click> option for additional │ │ │ │ │ - * information. Default is false. │ │ │ │ │ + * APIProperty: documentDrag │ │ │ │ │ + * {Boolean} Allow panning of the map by dragging outside map viewport. │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - box: false, │ │ │ │ │ + documentDrag: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: maxFeatures │ │ │ │ │ - * {Integer} Maximum number of features to return from a query in single mode │ │ │ │ │ - * if supported by the <protocol>. This set of features is then used to │ │ │ │ │ - * determine the best match client-side. Default is 10. │ │ │ │ │ + /** │ │ │ │ │ + * Property: zoomBox │ │ │ │ │ + * {<OpenLayers.Control.ZoomBox>} │ │ │ │ │ */ │ │ │ │ │ - maxFeatures: 10, │ │ │ │ │ + zoomBox: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Object} Hash of {<OpenLayers.Feature.Vector>}, keyed by fid, holding │ │ │ │ │ - * the currently selected features │ │ │ │ │ + * APIProperty: zoomBoxEnabled │ │ │ │ │ + * {Boolean} Whether the user can draw a box to zoom │ │ │ │ │ */ │ │ │ │ │ - features: null, │ │ │ │ │ + zoomBoxEnabled: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Proeprty: hoverFeature │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} The feature currently selected by the │ │ │ │ │ - * hover handler │ │ │ │ │ + * APIProperty: zoomWheelEnabled │ │ │ │ │ + * {Boolean} Whether the mousewheel should zoom the map │ │ │ │ │ */ │ │ │ │ │ - hoverFeature: null, │ │ │ │ │ + zoomWheelEnabled: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: handlerOptions │ │ │ │ │ - * {Object} Additional options for the handlers used by this control. This │ │ │ │ │ - * is a hash with the keys "click", "box" and "hover". │ │ │ │ │ + * Property: mouseWheelOptions │ │ │ │ │ + * {Object} Options passed to the MouseWheel control (only useful if │ │ │ │ │ + * <zoomWheelEnabled> is set to true). Default is no options for maps │ │ │ │ │ + * with fractionalZoom set to true, otherwise │ │ │ │ │ + * {cumulative: false, interval: 50, maxDelta: 6} │ │ │ │ │ */ │ │ │ │ │ + mouseWheelOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: handlers │ │ │ │ │ - * {Object} Object with references to multiple <OpenLayers.Handler> │ │ │ │ │ - * instances. │ │ │ │ │ + * APIProperty: handleRightClicks │ │ │ │ │ + * {Boolean} Whether or not to handle right clicks. Default is false. │ │ │ │ │ */ │ │ │ │ │ - handlers: null, │ │ │ │ │ + handleRightClicks: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: hoverResponse │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} The response object associated with │ │ │ │ │ - * the currently running hover request (if any). │ │ │ │ │ + * APIProperty: zoomBoxKeyMask │ │ │ │ │ + * {Integer} <OpenLayers.Handler> key code of the key, which has to be │ │ │ │ │ + * pressed, while drawing the zoom box with the mouse on the screen. │ │ │ │ │ + * You should probably set handleRightClicks to true if you use this │ │ │ │ │ + * with MOD_CTRL, to disable the context menu for machines which use │ │ │ │ │ + * CTRL-Click as a right click. │ │ │ │ │ + * Default: <OpenLayers.Handler.MOD_SHIFT> │ │ │ │ │ */ │ │ │ │ │ - hoverResponse: null, │ │ │ │ │ + zoomBoxKeyMask: OpenLayers.Handler.MOD_SHIFT, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: filterType │ │ │ │ │ - * {<String>} The type of filter to use when sending off a request. │ │ │ │ │ - * Possible values: │ │ │ │ │ - * OpenLayers.Filter.Spatial.<BBOX|INTERSECTS|WITHIN|CONTAINS> │ │ │ │ │ - * Defaults to: OpenLayers.Filter.Spatial.BBOX │ │ │ │ │ - */ │ │ │ │ │ - filterType: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * beforefeatureselected - Triggered when <click> is true before a │ │ │ │ │ - * feature is selected. The event object has a feature property with │ │ │ │ │ - * the feature about to select │ │ │ │ │ - * featureselected - Triggered when <click> is true and a feature is │ │ │ │ │ - * selected. The event object has a feature property with the │ │ │ │ │ - * selected feature │ │ │ │ │ - * beforefeaturesselected - Triggered when <click> is true before a │ │ │ │ │ - * set of features is selected. The event object is an array of │ │ │ │ │ - * feature properties with the features about to be selected. │ │ │ │ │ - * Return false after receiving this event to discontinue processing │ │ │ │ │ - * of all featureselected events and the featuresselected event. │ │ │ │ │ - * featuresselected - Triggered when <click> is true and a set of │ │ │ │ │ - * features is selected. The event object is an array of feature │ │ │ │ │ - * properties of the selected features │ │ │ │ │ - * featureunselected - Triggered when <click> is true and a feature is │ │ │ │ │ - * unselected. The event object has a feature property with the │ │ │ │ │ - * unselected feature │ │ │ │ │ - * clickout - Triggered when when <click> is true and no feature was │ │ │ │ │ - * selected. │ │ │ │ │ - * hoverfeature - Triggered when <hover> is true and the mouse has │ │ │ │ │ - * stopped over a feature │ │ │ │ │ - * outfeature - Triggered when <hover> is true and the mouse moves │ │ │ │ │ - * moved away from a hover-selected feature │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ + autoActivate: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.GetFeature │ │ │ │ │ - * Create a new control for fetching remote features. │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Control.Navigation │ │ │ │ │ + * Create a new navigation control │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} A configuration object which at least has to contain │ │ │ │ │ - * a <protocol> property (if not, it has to be set before a request is │ │ │ │ │ - * made) │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * the control │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - │ │ │ │ │ - this.features = {}; │ │ │ │ │ - │ │ │ │ │ this.handlers = {}; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (this.click) { │ │ │ │ │ - this.handlers.click = new OpenLayers.Handler.Click(this, { │ │ │ │ │ - click: this.selectClick │ │ │ │ │ - }, this.handlerOptions.click || {}); │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * The destroy method is used to perform any clean up before the control │ │ │ │ │ + * is dereferenced. Typically this is where event listeners are removed │ │ │ │ │ + * to prevent memory leaks. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + │ │ │ │ │ + if (this.dragPan) { │ │ │ │ │ + this.dragPan.destroy(); │ │ │ │ │ } │ │ │ │ │ + this.dragPan = null; │ │ │ │ │ │ │ │ │ │ - if (this.box) { │ │ │ │ │ - this.handlers.box = new OpenLayers.Handler.Box( │ │ │ │ │ - this, { │ │ │ │ │ - done: this.selectBox │ │ │ │ │ - }, │ │ │ │ │ - OpenLayers.Util.extend(this.handlerOptions.box, { │ │ │ │ │ - boxDivClassName: "olHandlerBoxSelectFeature" │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ + if (this.zoomBox) { │ │ │ │ │ + this.zoomBox.destroy(); │ │ │ │ │ } │ │ │ │ │ + this.zoomBox = null; │ │ │ │ │ │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - this.handlers.hover = new OpenLayers.Handler.Hover( │ │ │ │ │ - this, { │ │ │ │ │ - 'move': this.cancelHover, │ │ │ │ │ - 'pause': this.selectHover │ │ │ │ │ - }, │ │ │ │ │ - OpenLayers.Util.extend(this.handlerOptions.hover, { │ │ │ │ │ - 'delay': 250, │ │ │ │ │ - 'pixelTolerance': 2 │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ + if (this.pinchZoom) { │ │ │ │ │ + this.pinchZoom.destroy(); │ │ │ │ │ } │ │ │ │ │ + this.pinchZoom = null; │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Method: activate │ │ │ │ │ - * Activates the control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The control was effectively activated. │ │ │ │ │ */ │ │ │ │ │ activate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - for (var i in this.handlers) { │ │ │ │ │ - this.handlers[i].activate(); │ │ │ │ │ - } │ │ │ │ │ + this.dragPan.activate(); │ │ │ │ │ + if (this.zoomWheelEnabled) { │ │ │ │ │ + this.handlers.wheel.activate(); │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Control.prototype.activate.apply( │ │ │ │ │ - this, arguments │ │ │ │ │ - ); │ │ │ │ │ + this.handlers.click.activate(); │ │ │ │ │ + if (this.zoomBoxEnabled) { │ │ │ │ │ + this.zoomBox.activate(); │ │ │ │ │ + } │ │ │ │ │ + if (this.pinchZoom) { │ │ │ │ │ + this.pinchZoom.activate(); │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Control.prototype.activate.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Method: deactivate │ │ │ │ │ - * Deactivates the control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The control was effectively deactivated. │ │ │ │ │ */ │ │ │ │ │ deactivate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - for (var i in this.handlers) { │ │ │ │ │ - this.handlers[i].deactivate(); │ │ │ │ │ - } │ │ │ │ │ + if (this.pinchZoom) { │ │ │ │ │ + this.pinchZoom.deactivate(); │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Control.prototype.deactivate.apply( │ │ │ │ │ - this, arguments │ │ │ │ │ - ); │ │ │ │ │ + this.zoomBox.deactivate(); │ │ │ │ │ + this.dragPan.deactivate(); │ │ │ │ │ + this.handlers.click.deactivate(); │ │ │ │ │ + this.handlers.wheel.deactivate(); │ │ │ │ │ + return OpenLayers.Control.prototype.deactivate.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: selectClick │ │ │ │ │ - * Called on click │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ + * Method: draw │ │ │ │ │ */ │ │ │ │ │ - selectClick: function(evt) { │ │ │ │ │ - var bounds = this.pixelToBounds(evt.xy); │ │ │ │ │ + draw: function() { │ │ │ │ │ + // disable right mouse context menu for support of right click events │ │ │ │ │ + if (this.handleRightClicks) { │ │ │ │ │ + this.map.viewPortDiv.oncontextmenu = OpenLayers.Function.False; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - this.setModifiers(evt); │ │ │ │ │ - this.request(bounds, { │ │ │ │ │ - single: this.single │ │ │ │ │ + var clickCallbacks = { │ │ │ │ │ + 'click': this.defaultClick, │ │ │ │ │ + 'dblclick': this.defaultDblClick, │ │ │ │ │ + 'dblrightclick': this.defaultDblRightClick │ │ │ │ │ + }; │ │ │ │ │ + var clickOptions = { │ │ │ │ │ + 'double': true, │ │ │ │ │ + 'stopDouble': true │ │ │ │ │ + }; │ │ │ │ │ + this.handlers.click = new OpenLayers.Handler.Click( │ │ │ │ │ + this, clickCallbacks, clickOptions │ │ │ │ │ + ); │ │ │ │ │ + this.dragPan = new OpenLayers.Control.DragPan( │ │ │ │ │ + OpenLayers.Util.extend({ │ │ │ │ │ + map: this.map, │ │ │ │ │ + documentDrag: this.documentDrag │ │ │ │ │ + }, this.dragPanOptions) │ │ │ │ │ + ); │ │ │ │ │ + this.zoomBox = new OpenLayers.Control.ZoomBox({ │ │ │ │ │ + map: this.map, │ │ │ │ │ + keyMask: this.zoomBoxKeyMask │ │ │ │ │ }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: selectBox │ │ │ │ │ - * Callback from the handlers.box set up when <box> selection is on │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * position - {<OpenLayers.Bounds>|Object} An OpenLayers.Bounds or │ │ │ │ │ - * an object with a 'left', 'bottom', 'right' and 'top' properties. │ │ │ │ │ - */ │ │ │ │ │ - selectBox: function(position) { │ │ │ │ │ - var bounds; │ │ │ │ │ - if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ - var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.left, │ │ │ │ │ - y: position.bottom │ │ │ │ │ - }); │ │ │ │ │ - var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.right, │ │ │ │ │ - y: position.top │ │ │ │ │ - }); │ │ │ │ │ - bounds = new OpenLayers.Bounds( │ │ │ │ │ - minXY.lon, minXY.lat, maxXY.lon, maxXY.lat │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - } else { │ │ │ │ │ - if (this.click) { │ │ │ │ │ - // box without extent - let the click handler take care of it │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - bounds = this.pixelToBounds(position); │ │ │ │ │ + this.dragPan.draw(); │ │ │ │ │ + this.zoomBox.draw(); │ │ │ │ │ + var wheelOptions = this.map.fractionalZoom ? {} : { │ │ │ │ │ + cumulative: false, │ │ │ │ │ + interval: 50, │ │ │ │ │ + maxDelta: 6 │ │ │ │ │ + }; │ │ │ │ │ + this.handlers.wheel = new OpenLayers.Handler.MouseWheel( │ │ │ │ │ + this, { │ │ │ │ │ + up: this.wheelUp, │ │ │ │ │ + down: this.wheelDown │ │ │ │ │ + }, │ │ │ │ │ + OpenLayers.Util.extend(wheelOptions, this.mouseWheelOptions) │ │ │ │ │ + ); │ │ │ │ │ + if (OpenLayers.Control.PinchZoom) { │ │ │ │ │ + this.pinchZoom = new OpenLayers.Control.PinchZoom( │ │ │ │ │ + OpenLayers.Util.extend({ │ │ │ │ │ + map: this.map │ │ │ │ │ + }, this.pinchZoomOptions)); │ │ │ │ │ } │ │ │ │ │ - this.setModifiers(this.handlers.box.dragHandler.evt); │ │ │ │ │ - this.request(bounds); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: selectHover │ │ │ │ │ - * Callback from the handlers.hover set up when <hover> selection is on │ │ │ │ │ + * Method: defaultClick │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} event object with an xy property │ │ │ │ │ - */ │ │ │ │ │ - selectHover: function(evt) { │ │ │ │ │ - var bounds = this.pixelToBounds(evt.xy); │ │ │ │ │ - this.request(bounds, { │ │ │ │ │ - single: true, │ │ │ │ │ - hover: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: cancelHover │ │ │ │ │ - * Callback from the handlers.hover set up when <hover> selection is on │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - cancelHover: function() { │ │ │ │ │ - if (this.hoverResponse) { │ │ │ │ │ - this.protocol.abort(this.hoverResponse); │ │ │ │ │ - this.hoverResponse = null; │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + defaultClick: function(evt) { │ │ │ │ │ + if (evt.lastTouches && evt.lastTouches.length == 2) { │ │ │ │ │ + this.map.zoomOut(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: request │ │ │ │ │ - * Sends a GetFeature request to the WFS │ │ │ │ │ + * Method: defaultDblClick │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} bounds for the request's BBOX filter │ │ │ │ │ - * options - {Object} additional options for this method. │ │ │ │ │ - * │ │ │ │ │ - * Supported options include: │ │ │ │ │ - * single - {Boolean} A single feature should be returned. │ │ │ │ │ - * Note that this will be ignored if the protocol does not │ │ │ │ │ - * return the geometries of the features. │ │ │ │ │ - * hover - {Boolean} Do the request for the hover handler. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - request: function(bounds, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: this.filterType, │ │ │ │ │ - value: bounds │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - // Set the cursor to "wait" to tell the user we're working. │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - │ │ │ │ │ - var response = this.protocol.read({ │ │ │ │ │ - maxFeatures: options.single == true ? this.maxFeatures : undefined, │ │ │ │ │ - filter: filter, │ │ │ │ │ - callback: function(result) { │ │ │ │ │ - if (result.success()) { │ │ │ │ │ - if (result.features.length) { │ │ │ │ │ - if (options.single == true) { │ │ │ │ │ - this.selectBestFeature(result.features, │ │ │ │ │ - bounds.getCenterLonLat(), options); │ │ │ │ │ - } else { │ │ │ │ │ - this.select(result.features); │ │ │ │ │ - } │ │ │ │ │ - } else if (options.hover) { │ │ │ │ │ - this.hoverSelect(); │ │ │ │ │ - } else { │ │ │ │ │ - this.events.triggerEvent("clickout"); │ │ │ │ │ - if (this.clickout) { │ │ │ │ │ - this.unselectAll(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // Reset the cursor. │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - if (options.hover == true) { │ │ │ │ │ - this.hoverResponse = response; │ │ │ │ │ - } │ │ │ │ │ + defaultDblClick: function(evt) { │ │ │ │ │ + this.map.zoomTo(this.map.zoom + 1, evt.xy); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: selectBestFeature │ │ │ │ │ - * Selects the feature from an array of features that is the best match │ │ │ │ │ - * for the click position. │ │ │ │ │ + * Method: defaultDblRightClick │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ - * clickPosition - {<OpenLayers.LonLat>} │ │ │ │ │ - * options - {Object} additional options for this method │ │ │ │ │ - * │ │ │ │ │ - * Supported options include: │ │ │ │ │ - * hover - {Boolean} Do the selection for the hover handler. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - selectBestFeature: function(features, clickPosition, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - if (features.length) { │ │ │ │ │ - var point = new OpenLayers.Geometry.Point(clickPosition.lon, │ │ │ │ │ - clickPosition.lat); │ │ │ │ │ - var feature, resultFeature, dist; │ │ │ │ │ - var minDist = Number.MAX_VALUE; │ │ │ │ │ - for (var i = 0; i < features.length; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - dist = point.distanceTo(feature.geometry, { │ │ │ │ │ - edge: false │ │ │ │ │ - }); │ │ │ │ │ - if (dist < minDist) { │ │ │ │ │ - minDist = dist; │ │ │ │ │ - resultFeature = feature; │ │ │ │ │ - if (minDist == 0) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (options.hover == true) { │ │ │ │ │ - this.hoverSelect(resultFeature); │ │ │ │ │ - } else { │ │ │ │ │ - this.select(resultFeature || features); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + defaultDblRightClick: function(evt) { │ │ │ │ │ + this.map.zoomTo(this.map.zoom - 1, evt.xy); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setModifiers │ │ │ │ │ - * Sets the multiple and toggle modifiers according to the current event │ │ │ │ │ - * │ │ │ │ │ + * Method: wheelChange │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * deltaZ - {Integer} │ │ │ │ │ */ │ │ │ │ │ - setModifiers: function(evt) { │ │ │ │ │ - this.modifiers = { │ │ │ │ │ - multiple: this.multiple || (this.multipleKey && evt[this.multipleKey]), │ │ │ │ │ - toggle: this.toggle || (this.toggleKey && evt[this.toggleKey]) │ │ │ │ │ - }; │ │ │ │ │ + wheelChange: function(evt, deltaZ) { │ │ │ │ │ + if (!this.map.fractionalZoom) { │ │ │ │ │ + deltaZ = Math.round(deltaZ); │ │ │ │ │ + } │ │ │ │ │ + var currentZoom = this.map.getZoom(), │ │ │ │ │ + newZoom = currentZoom + deltaZ; │ │ │ │ │ + newZoom = Math.max(newZoom, 0); │ │ │ │ │ + newZoom = Math.min(newZoom, this.map.getNumZoomLevels()); │ │ │ │ │ + if (newZoom === currentZoom) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + this.map.zoomTo(newZoom, evt.xy); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: select │ │ │ │ │ - * Add feature to the hash of selected features and trigger the │ │ │ │ │ - * featureselected and featuresselected events. │ │ │ │ │ + /** │ │ │ │ │ + * Method: wheelUp │ │ │ │ │ + * User spun scroll wheel up │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {<OpenLayers.Feature.Vector>} or an array of features │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * delta - {Integer} │ │ │ │ │ */ │ │ │ │ │ - select: function(features) { │ │ │ │ │ - if (!this.modifiers.multiple && !this.modifiers.toggle) { │ │ │ │ │ - this.unselectAll(); │ │ │ │ │ - } │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var cont = this.events.triggerEvent("beforefeaturesselected", { │ │ │ │ │ - features: features │ │ │ │ │ - }); │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - var selectedFeatures = []; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (this.features[feature.fid || feature.id]) { │ │ │ │ │ - if (this.modifiers.toggle) { │ │ │ │ │ - this.unselect(this.features[feature.fid || feature.id]); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - cont = this.events.triggerEvent("beforefeatureselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - this.features[feature.fid || feature.id] = feature; │ │ │ │ │ - selectedFeatures.push(feature); │ │ │ │ │ - │ │ │ │ │ - this.events.triggerEvent("featureselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("featuresselected", { │ │ │ │ │ - features: selectedFeatures │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ + wheelUp: function(evt, delta) { │ │ │ │ │ + this.wheelChange(evt, delta || 1); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: hoverSelect │ │ │ │ │ - * Sets/unsets the <hoverFeature> │ │ │ │ │ + /** │ │ │ │ │ + * Method: wheelDown │ │ │ │ │ + * User spun scroll wheel down │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} the feature to hover-select. │ │ │ │ │ - * If none is provided, the current <hoverFeature> will be nulled and │ │ │ │ │ - * the outfeature event will be triggered. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * delta - {Integer} │ │ │ │ │ */ │ │ │ │ │ - hoverSelect: function(feature) { │ │ │ │ │ - var fid = feature ? feature.fid || feature.id : null; │ │ │ │ │ - var hfid = this.hoverFeature ? │ │ │ │ │ - this.hoverFeature.fid || this.hoverFeature.id : null; │ │ │ │ │ - │ │ │ │ │ - if (hfid && hfid != fid) { │ │ │ │ │ - this.events.triggerEvent("outfeature", { │ │ │ │ │ - feature: this.hoverFeature │ │ │ │ │ - }); │ │ │ │ │ - this.hoverFeature = null; │ │ │ │ │ - } │ │ │ │ │ - if (fid && fid != hfid) { │ │ │ │ │ - this.events.triggerEvent("hoverfeature", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - this.hoverFeature = feature; │ │ │ │ │ - } │ │ │ │ │ + wheelDown: function(evt, delta) { │ │ │ │ │ + this.wheelChange(evt, delta || -1); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: unselect │ │ │ │ │ - * Remove feature from the hash of selected features and trigger the │ │ │ │ │ - * featureunselected event. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * Method: disableZoomBox │ │ │ │ │ */ │ │ │ │ │ - unselect: function(feature) { │ │ │ │ │ - delete this.features[feature.fid || feature.id]; │ │ │ │ │ - this.events.triggerEvent("featureunselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ + disableZoomBox: function() { │ │ │ │ │ + this.zoomBoxEnabled = false; │ │ │ │ │ + this.zoomBox.deactivate(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: unselectAll │ │ │ │ │ - * Unselect all selected features. │ │ │ │ │ + * Method: enableZoomBox │ │ │ │ │ */ │ │ │ │ │ - unselectAll: function() { │ │ │ │ │ - // we'll want an option to supress notification here │ │ │ │ │ - for (var fid in this.features) { │ │ │ │ │ - this.unselect(this.features[fid]); │ │ │ │ │ + enableZoomBox: function() { │ │ │ │ │ + this.zoomBoxEnabled = true; │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.zoomBox.activate(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the control. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + /** │ │ │ │ │ + * Method: disableZoomWheel │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - for (var i in this.handlers) { │ │ │ │ │ - this.handlers[i].setMap(map); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + disableZoomWheel: function() { │ │ │ │ │ + this.zoomWheelEnabled = false; │ │ │ │ │ + this.handlers.wheel.deactivate(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: pixelToBounds │ │ │ │ │ - * Takes a pixel as argument and creates bounds after adding the │ │ │ │ │ - * <clickTolerance>. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} │ │ │ │ │ + * Method: enableZoomWheel │ │ │ │ │ */ │ │ │ │ │ - pixelToBounds: function(pixel) { │ │ │ │ │ - var llPx = pixel.add(-this.clickTolerance / 2, this.clickTolerance / 2); │ │ │ │ │ - var urPx = pixel.add(this.clickTolerance / 2, -this.clickTolerance / 2); │ │ │ │ │ - var ll = this.map.getLonLatFromPixel(llPx); │ │ │ │ │ - var ur = this.map.getLonLatFromPixel(urPx); │ │ │ │ │ - return new OpenLayers.Bounds(ll.lon, ll.lat, ur.lon, ur.lat); │ │ │ │ │ + │ │ │ │ │ + enableZoomWheel: function() { │ │ │ │ │ + this.zoomWheelEnabled = true; │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.handlers.wheel.activate(); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.GetFeature" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Navigation" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Snapping.js │ │ │ │ │ + OpenLayers/Control/NavToolbar.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ + * @requires OpenLayers/Control/Panel.js │ │ │ │ │ + * @requires OpenLayers/Control/Navigation.js │ │ │ │ │ + * @requires OpenLayers/Control/ZoomBox.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.Snapping │ │ │ │ │ - * Acts as a snapping agent while editing vector features. │ │ │ │ │ + * Class: OpenLayers.Control.NavToolbar │ │ │ │ │ + * This Toolbar is an alternative to the Navigation control that displays │ │ │ │ │ + * the state of the control, and provides a UI for changing state to │ │ │ │ │ + * use the zoomBox via a Panel control. │ │ │ │ │ * │ │ │ │ │ + * If you wish to change the properties of the Navigation control used │ │ │ │ │ + * in the NavToolbar, see: │ │ │ │ │ + * http://trac.openlayers.org/wiki/Toolbars#SubclassingNavToolbar │ │ │ │ │ + * │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ + * - <OpenLayers.Control.Panel> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.Snapping = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * beforesnap - Triggered before a snap occurs. Listeners receive an │ │ │ │ │ - * event object with *point*, *x*, *y*, *distance*, *layer*, and │ │ │ │ │ - * *snapType* properties. The point property will be original point │ │ │ │ │ - * geometry considered for snapping. The x and y properties represent │ │ │ │ │ - * coordinates the point will receive. The distance is the distance │ │ │ │ │ - * of the snap. The layer is the target layer. The snapType property │ │ │ │ │ - * will be one of "node", "vertex", or "edge". Return false to stop │ │ │ │ │ - * snapping from occurring. │ │ │ │ │ - * snap - Triggered when a snap occurs. Listeners receive an event with │ │ │ │ │ - * *point*, *snapType*, *layer*, and *distance* properties. The point │ │ │ │ │ - * will be the location snapped to. The snapType will be one of "node", │ │ │ │ │ - * "vertex", or "edge". The layer will be the target layer. The │ │ │ │ │ - * distance will be the distance of the snap in map units. │ │ │ │ │ - * unsnap - Triggered when a vertex is unsnapped. Listeners receive an │ │ │ │ │ - * event with a *point* property. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * CONSTANT: DEFAULTS │ │ │ │ │ - * Default target properties. │ │ │ │ │ - */ │ │ │ │ │ - DEFAULTS: { │ │ │ │ │ - tolerance: 10, │ │ │ │ │ - node: true, │ │ │ │ │ - edge: true, │ │ │ │ │ - vertex: true │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: greedy │ │ │ │ │ - * {Boolean} Snap to closest feature in first layer with an eligible │ │ │ │ │ - * feature. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - greedy: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: precedence │ │ │ │ │ - * {Array} List representing precedence of different snapping types. │ │ │ │ │ - * Default is "node", "vertex", "edge". │ │ │ │ │ - */ │ │ │ │ │ - precedence: ["node", "vertex", "edge"], │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: resolution │ │ │ │ │ - * {Float} The map resolution for the previously considered snap. │ │ │ │ │ - */ │ │ │ │ │ - resolution: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: geoToleranceCache │ │ │ │ │ - * {Object} A cache of geo-tolerances. Tolerance values (in map units) are │ │ │ │ │ - * calculated when the map resolution changes. │ │ │ │ │ - */ │ │ │ │ │ - geoToleranceCache: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} The current editable layer. Set at │ │ │ │ │ - * construction or after construction with <setLayer>. │ │ │ │ │ - */ │ │ │ │ │ - layer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: feature │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} The current editable feature. │ │ │ │ │ - */ │ │ │ │ │ - feature: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: point │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} The currently snapped vertex. │ │ │ │ │ - */ │ │ │ │ │ - point: null, │ │ │ │ │ +OpenLayers.Control.NavToolbar = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Snapping │ │ │ │ │ - * Creates a new snapping control. A control is constructed with an editable │ │ │ │ │ - * layer and a set of configuration objects for target layers. While the │ │ │ │ │ - * control is active, dragging vertices while drawing new features or │ │ │ │ │ - * modifying existing features on the editable layer will engage │ │ │ │ │ - * snapping to features on the target layers. Whether a vertex snaps to │ │ │ │ │ - * a feature on a target layer depends on the target layer configuration. │ │ │ │ │ + * Constructor: OpenLayers.Control.NavToolbar │ │ │ │ │ + * Add our two mousedefaults controls. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An object containing all configuration properties for │ │ │ │ │ - * the control. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * layer - {<OpenLayers.Layer.Vector>} The editable layer. Features from this │ │ │ │ │ - * layer that are digitized or modified may have vertices snapped to │ │ │ │ │ - * features from any of the target layers. │ │ │ │ │ - * targets - {Array(Object | OpenLayers.Layer.Vector)} A list of objects for │ │ │ │ │ - * configuring target layers. See valid properties of the target │ │ │ │ │ - * objects below. If the items in the targets list are vector layers │ │ │ │ │ - * (instead of configuration objects), the defaults from the <defaults> │ │ │ │ │ - * property will apply. The editable layer itself may be a target │ │ │ │ │ - * layer, allowing newly created or edited features to be snapped to │ │ │ │ │ - * existing features from the same layer. If no targets are provided │ │ │ │ │ - * the layer given in the constructor (as <layer>) will become the │ │ │ │ │ - * initial target. │ │ │ │ │ - * defaults - {Object} An object with default properties to be applied │ │ │ │ │ - * to all target objects. │ │ │ │ │ - * greedy - {Boolean} Snap to closest feature in first target layer that │ │ │ │ │ - * applies. Default is true. If false, all features in all target │ │ │ │ │ - * layers will be checked and the closest feature in all target layers │ │ │ │ │ - * will be chosen. The greedy property determines if the order of the │ │ │ │ │ - * target layers is significant. By default, the order of the target │ │ │ │ │ - * layers is significant where layers earlier in the target layer list │ │ │ │ │ - * have precedence over layers later in the list. Within a single │ │ │ │ │ - * layer, the closest feature is always chosen for snapping. This │ │ │ │ │ - * property only determines whether the search for a closer feature │ │ │ │ │ - * continues after an eligible feature is found in a target layer. │ │ │ │ │ - * │ │ │ │ │ - * Valid target properties: │ │ │ │ │ - * layer - {<OpenLayers.Layer.Vector>} A target layer. Features from this │ │ │ │ │ - * layer will be eligible to act as snapping target for the editable │ │ │ │ │ - * layer. │ │ │ │ │ - * tolerance - {Float} The distance (in pixels) at which snapping may occur. │ │ │ │ │ - * Default is 10. │ │ │ │ │ - * node - {Boolean} Snap to nodes (first or last point in a geometry) in │ │ │ │ │ - * target layer. Default is true. │ │ │ │ │ - * nodeTolerance - {Float} Optional distance at which snapping may occur │ │ │ │ │ - * for nodes specifically. If none is provided, <tolerance> will be │ │ │ │ │ - * used. │ │ │ │ │ - * vertex - {Boolean} Snap to vertices in target layer. Default is true. │ │ │ │ │ - * vertexTolerance - {Float} Optional distance at which snapping may occur │ │ │ │ │ - * for vertices specifically. If none is provided, <tolerance> will be │ │ │ │ │ - * used. │ │ │ │ │ - * edge - {Boolean} Snap to edges in target layer. Default is true. │ │ │ │ │ - * edgeTolerance - {Float} Optional distance at which snapping may occur │ │ │ │ │ - * for edges specifically. If none is provided, <tolerance> will be │ │ │ │ │ - * used. │ │ │ │ │ - * filter - {<OpenLayers.Filter>} Optional filter to evaluate to determine if │ │ │ │ │ - * feature is eligible for snapping. If filter evaluates to true for a │ │ │ │ │ - * target feature a vertex may be snapped to the feature. │ │ │ │ │ - * minResolution - {Number} If a minResolution is provided, snapping to this │ │ │ │ │ - * target will only be considered if the map resolution is greater than │ │ │ │ │ - * or equal to this value (the minResolution is inclusive). Default is │ │ │ │ │ - * no minimum resolution limit. │ │ │ │ │ - * maxResolution - {Number} If a maxResolution is provided, snapping to this │ │ │ │ │ - * target will only be considered if the map resolution is strictly │ │ │ │ │ - * less than this value (the maxResolution is exclusive). Default is │ │ │ │ │ - * no maximum resolution limit. │ │ │ │ │ + * options - {Object} An optional object whose properties will be used │ │ │ │ │ + * to extend the control. │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.options = options || {}; // TODO: this could be done by the super │ │ │ │ │ - │ │ │ │ │ - // set the editable layer if provided │ │ │ │ │ - if (this.options.layer) { │ │ │ │ │ - this.setLayer(this.options.layer); │ │ │ │ │ - } │ │ │ │ │ - // configure target layers │ │ │ │ │ - var defaults = OpenLayers.Util.extend({}, this.options.defaults); │ │ │ │ │ - this.defaults = OpenLayers.Util.applyDefaults(defaults, this.DEFAULTS); │ │ │ │ │ - this.setTargets(this.options.targets); │ │ │ │ │ - if (this.targets.length === 0 && this.layer) { │ │ │ │ │ - this.addTargetLayer(this.layer); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.geoToleranceCache = {}; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setLayer │ │ │ │ │ - * Set the editable layer. Call the setLayer method if the editable layer │ │ │ │ │ - * changes and the same control should be used on a new editable layer. │ │ │ │ │ - * If the control is already active, it will be active after the new │ │ │ │ │ - * layer is set. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.Vector>} The new editable layer. │ │ │ │ │ - */ │ │ │ │ │ - setLayer: function(layer) { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - this.activate(); │ │ │ │ │ - } else { │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setTargets │ │ │ │ │ - * Set the targets for the snapping agent. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * targets - {Array} An array of target configs or target layers. │ │ │ │ │ - */ │ │ │ │ │ - setTargets: function(targets) { │ │ │ │ │ - this.targets = []; │ │ │ │ │ - if (targets && targets.length) { │ │ │ │ │ - var target; │ │ │ │ │ - for (var i = 0, len = targets.length; i < len; ++i) { │ │ │ │ │ - target = targets[i]; │ │ │ │ │ - if (target instanceof OpenLayers.Layer.Vector) { │ │ │ │ │ - this.addTargetLayer(target); │ │ │ │ │ - } else { │ │ │ │ │ - this.addTarget(target); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addTargetLayer │ │ │ │ │ - * Add a target layer with the default target config. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.Vector>} A target layer. │ │ │ │ │ - */ │ │ │ │ │ - addTargetLayer: function(layer) { │ │ │ │ │ - this.addTarget({ │ │ │ │ │ - layer: layer │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addTarget │ │ │ │ │ - * Add a configured target layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * target - {Object} A target config. │ │ │ │ │ - */ │ │ │ │ │ - addTarget: function(target) { │ │ │ │ │ - target = OpenLayers.Util.applyDefaults(target, this.defaults); │ │ │ │ │ - target.nodeTolerance = target.nodeTolerance || target.tolerance; │ │ │ │ │ - target.vertexTolerance = target.vertexTolerance || target.tolerance; │ │ │ │ │ - target.edgeTolerance = target.edgeTolerance || target.tolerance; │ │ │ │ │ - this.targets.push(target); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeTargetLayer │ │ │ │ │ - * Remove a target layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.Vector>} The target layer to remove. │ │ │ │ │ - */ │ │ │ │ │ - removeTargetLayer: function(layer) { │ │ │ │ │ - var target; │ │ │ │ │ - for (var i = this.targets.length - 1; i >= 0; --i) { │ │ │ │ │ - target = this.targets[i]; │ │ │ │ │ - if (target.layer === layer) { │ │ │ │ │ - this.removeTarget(target); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeTarget │ │ │ │ │ - * Remove a target. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * target - {Object} A target config. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} The targets array. │ │ │ │ │ - */ │ │ │ │ │ - removeTarget: function(target) { │ │ │ │ │ - return OpenLayers.Util.removeItem(this.targets, target); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Activate the control. Activating the control registers listeners for │ │ │ │ │ - * editing related events so that during feature creation and │ │ │ │ │ - * modification, moving vertices will trigger snapping. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Control.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - if (this.layer && this.layer.events) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - sketchstarted: this.onSketchModified, │ │ │ │ │ - sketchmodified: this.onSketchModified, │ │ │ │ │ - vertexmodified: this.onVertexModified, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the control. Deactivating the control unregisters listeners │ │ │ │ │ - * so feature editing may proceed without engaging the snapping agent. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Control.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - if (this.layer && this.layer.events) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - sketchstarted: this.onSketchModified, │ │ │ │ │ - sketchmodified: this.onSketchModified, │ │ │ │ │ - vertexmodified: this.onVertexModified, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.feature = null; │ │ │ │ │ - this.point = null; │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onSketchModified │ │ │ │ │ - * Registered as a listener for the sketchmodified event on the editable │ │ │ │ │ - * layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * event - {Object} The sketch modified event. │ │ │ │ │ - */ │ │ │ │ │ - onSketchModified: function(event) { │ │ │ │ │ - this.feature = event.feature; │ │ │ │ │ - this.considerSnapping(event.vertex, event.vertex); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onVertexModified │ │ │ │ │ - * Registered as a listener for the vertexmodified event on the editable │ │ │ │ │ - * layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * event - {Object} The vertex modified event. │ │ │ │ │ - */ │ │ │ │ │ - onVertexModified: function(event) { │ │ │ │ │ - this.feature = event.feature; │ │ │ │ │ - var loc = this.layer.map.getLonLatFromViewPortPx(event.pixel); │ │ │ │ │ - this.considerSnapping( │ │ │ │ │ - event.vertex, new OpenLayers.Geometry.Point(loc.lon, loc.lat) │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: considerSnapping │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} The vertex to be snapped (or │ │ │ │ │ - * unsnapped). │ │ │ │ │ - * loc - {<OpenLayers.Geometry.Point>} The location of the mouse in map │ │ │ │ │ - * coords. │ │ │ │ │ - */ │ │ │ │ │ - considerSnapping: function(point, loc) { │ │ │ │ │ - var best = { │ │ │ │ │ - rank: Number.POSITIVE_INFINITY, │ │ │ │ │ - dist: Number.POSITIVE_INFINITY, │ │ │ │ │ - x: null, │ │ │ │ │ - y: null │ │ │ │ │ - }; │ │ │ │ │ - var snapped = false; │ │ │ │ │ - var result, target; │ │ │ │ │ - for (var i = 0, len = this.targets.length; i < len; ++i) { │ │ │ │ │ - target = this.targets[i]; │ │ │ │ │ - result = this.testTarget(target, loc); │ │ │ │ │ - if (result) { │ │ │ │ │ - if (this.greedy) { │ │ │ │ │ - best = result; │ │ │ │ │ - best.target = target; │ │ │ │ │ - snapped = true; │ │ │ │ │ - break; │ │ │ │ │ - } else { │ │ │ │ │ - if ((result.rank < best.rank) || │ │ │ │ │ - (result.rank === best.rank && result.dist < best.dist)) { │ │ │ │ │ - best = result; │ │ │ │ │ - best.target = target; │ │ │ │ │ - snapped = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (snapped) { │ │ │ │ │ - var proceed = this.events.triggerEvent("beforesnap", { │ │ │ │ │ - point: point, │ │ │ │ │ - x: best.x, │ │ │ │ │ - y: best.y, │ │ │ │ │ - distance: best.dist, │ │ │ │ │ - layer: best.target.layer, │ │ │ │ │ - snapType: this.precedence[best.rank] │ │ │ │ │ - }); │ │ │ │ │ - if (proceed !== false) { │ │ │ │ │ - point.x = best.x; │ │ │ │ │ - point.y = best.y; │ │ │ │ │ - this.point = point; │ │ │ │ │ - this.events.triggerEvent("snap", { │ │ │ │ │ - point: point, │ │ │ │ │ - snapType: this.precedence[best.rank], │ │ │ │ │ - layer: best.target.layer, │ │ │ │ │ - distance: best.dist │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - snapped = false; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.point && !snapped) { │ │ │ │ │ - point.x = loc.x; │ │ │ │ │ - point.y = loc.y; │ │ │ │ │ - this.point = null; │ │ │ │ │ - this.events.triggerEvent("unsnap", { │ │ │ │ │ - point: point │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: testTarget │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * target - {Object} Object with target layer configuration. │ │ │ │ │ - * loc - {<OpenLayers.Geometry.Point>} The location of the mouse in map │ │ │ │ │ - * coords. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} A result object with rank, dist, x, and y properties. │ │ │ │ │ - * Returns null if candidate is not eligible for snapping. │ │ │ │ │ - */ │ │ │ │ │ - testTarget: function(target, loc) { │ │ │ │ │ - var resolution = this.layer.map.getResolution(); │ │ │ │ │ - if ("minResolution" in target) { │ │ │ │ │ - if (resolution < target.minResolution) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if ("maxResolution" in target) { │ │ │ │ │ - if (resolution >= target.maxResolution) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var tolerance = { │ │ │ │ │ - node: this.getGeoTolerance(target.nodeTolerance, resolution), │ │ │ │ │ - vertex: this.getGeoTolerance(target.vertexTolerance, resolution), │ │ │ │ │ - edge: this.getGeoTolerance(target.edgeTolerance, resolution) │ │ │ │ │ - }; │ │ │ │ │ - // this could be cached if we don't support setting tolerance values directly │ │ │ │ │ - var maxTolerance = Math.max( │ │ │ │ │ - tolerance.node, tolerance.vertex, tolerance.edge │ │ │ │ │ - ); │ │ │ │ │ - var result = { │ │ │ │ │ - rank: Number.POSITIVE_INFINITY, │ │ │ │ │ - dist: Number.POSITIVE_INFINITY │ │ │ │ │ - }; │ │ │ │ │ - var eligible = false; │ │ │ │ │ - var features = target.layer.features; │ │ │ │ │ - var feature, type, vertices, vertex, closest, dist, found; │ │ │ │ │ - var numTypes = this.precedence.length; │ │ │ │ │ - var ll = new OpenLayers.LonLat(loc.x, loc.y); │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (feature !== this.feature && !feature._sketch && │ │ │ │ │ - feature.state !== OpenLayers.State.DELETE && │ │ │ │ │ - (!target.filter || target.filter.evaluate(feature))) { │ │ │ │ │ - if (feature.atPoint(ll, maxTolerance, maxTolerance)) { │ │ │ │ │ - for (var j = 0, stop = Math.min(result.rank + 1, numTypes); j < stop; ++j) { │ │ │ │ │ - type = this.precedence[j]; │ │ │ │ │ - if (target[type]) { │ │ │ │ │ - if (type === "edge") { │ │ │ │ │ - closest = feature.geometry.distanceTo(loc, { │ │ │ │ │ - details: true │ │ │ │ │ - }); │ │ │ │ │ - dist = closest.distance; │ │ │ │ │ - if (dist <= tolerance[type] && dist < result.dist) { │ │ │ │ │ - result = { │ │ │ │ │ - rank: j, │ │ │ │ │ - dist: dist, │ │ │ │ │ - x: closest.x0, │ │ │ │ │ - y: closest.y0 // closest coords on feature │ │ │ │ │ - }; │ │ │ │ │ - eligible = true; │ │ │ │ │ - // don't look for lower precedence types for this feature │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - // look for nodes or vertices │ │ │ │ │ - vertices = feature.geometry.getVertices(type === "node"); │ │ │ │ │ - found = false; │ │ │ │ │ - for (var k = 0, klen = vertices.length; k < klen; ++k) { │ │ │ │ │ - vertex = vertices[k]; │ │ │ │ │ - dist = vertex.distanceTo(loc); │ │ │ │ │ - if (dist <= tolerance[type] && │ │ │ │ │ - (j < result.rank || (j === result.rank && dist < result.dist))) { │ │ │ │ │ - result = { │ │ │ │ │ - rank: j, │ │ │ │ │ - dist: dist, │ │ │ │ │ - x: vertex.x, │ │ │ │ │ - y: vertex.y │ │ │ │ │ - }; │ │ │ │ │ - eligible = true; │ │ │ │ │ - found = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (found) { │ │ │ │ │ - // don't look for lower precedence types for this feature │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return eligible ? result : null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getGeoTolerance │ │ │ │ │ - * Calculate a tolerance in map units given a tolerance in pixels. This │ │ │ │ │ - * takes advantage of the <geoToleranceCache> when the map resolution │ │ │ │ │ - * has not changed. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * tolerance - {Number} A tolerance value in pixels. │ │ │ │ │ - * resolution - {Number} Map resolution. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number} A tolerance value in map units. │ │ │ │ │ - */ │ │ │ │ │ - getGeoTolerance: function(tolerance, resolution) { │ │ │ │ │ - if (resolution !== this.resolution) { │ │ │ │ │ - this.resolution = resolution; │ │ │ │ │ - this.geoToleranceCache = {}; │ │ │ │ │ - } │ │ │ │ │ - var geoTolerance = this.geoToleranceCache[tolerance]; │ │ │ │ │ - if (geoTolerance === undefined) { │ │ │ │ │ - geoTolerance = tolerance * resolution; │ │ │ │ │ - this.geoToleranceCache[tolerance] = geoTolerance; │ │ │ │ │ - } │ │ │ │ │ - return geoTolerance; │ │ │ │ │ + OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.addControls([ │ │ │ │ │ + new OpenLayers.Control.Navigation(), │ │ │ │ │ + new OpenLayers.Control.ZoomBox() │ │ │ │ │ + ]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Clean up the control. │ │ │ │ │ + * Method: draw │ │ │ │ │ + * calls the default draw, and then activates mouse defaults. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.deactivate(); // TODO: this should be handled by the super │ │ │ │ │ + draw: function() { │ │ │ │ │ + var div = OpenLayers.Control.Panel.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (this.defaultControl === null) { │ │ │ │ │ + this.defaultControl = this.controls[0]; │ │ │ │ │ } │ │ │ │ │ - delete this.layer; │ │ │ │ │ - delete this.targets; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.call(this); │ │ │ │ │ + return div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Snapping" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.NavToolbar" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/WMTSGetFeatureInfo.js │ │ │ │ │ + OpenLayers/Control/CacheRead.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Handler/Click.js │ │ │ │ │ - * @requires OpenLayers/Handler/Hover.js │ │ │ │ │ - * @requires OpenLayers/Request.js │ │ │ │ │ - * @requires OpenLayers/Format/WMSGetFeatureInfo.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.WMTSGetFeatureInfo │ │ │ │ │ - * The WMTSGetFeatureInfo control uses a WMTS query to get information about a │ │ │ │ │ - * point on the map. The information may be in a display-friendly format │ │ │ │ │ - * such as HTML, or a machine-friendly format such as GML, depending on the │ │ │ │ │ - * server's capabilities and the client's configuration. This control │ │ │ │ │ - * handles click or hover events, attempts to parse the results using an │ │ │ │ │ - * OpenLayers.Format, and fires a 'getfeatureinfo' event for each layer │ │ │ │ │ - * queried. │ │ │ │ │ + * Class: OpenLayers.Control.CacheRead │ │ │ │ │ + * A control for using image tiles cached with <OpenLayers.Control.CacheWrite> │ │ │ │ │ + * from the browser's local storage. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.WMTSGetFeatureInfo = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: hover │ │ │ │ │ - * {Boolean} Send GetFeatureInfo requests when mouse stops moving. │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - hover: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: requestEncoding │ │ │ │ │ - * {String} One of "KVP" or "REST". Only KVP encoding is supported at this │ │ │ │ │ - * time. │ │ │ │ │ - */ │ │ │ │ │ - requestEncoding: "KVP", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: drillDown │ │ │ │ │ - * {Boolean} Drill down over all WMTS layers in the map. When │ │ │ │ │ - * using drillDown mode, hover is not possible. A getfeatureinfo event │ │ │ │ │ - * will be fired for each layer queried. │ │ │ │ │ - */ │ │ │ │ │ - drillDown: false, │ │ │ │ │ +OpenLayers.Control.CacheRead = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: maxFeatures │ │ │ │ │ - * {Integer} Maximum number of features to return from a WMTS query. This │ │ │ │ │ - * sets the feature_count parameter on WMTS GetFeatureInfo │ │ │ │ │ - * requests. │ │ │ │ │ - */ │ │ │ │ │ - maxFeatures: 10, │ │ │ │ │ - │ │ │ │ │ - /** APIProperty: clickCallback │ │ │ │ │ - * {String} The click callback to register in the │ │ │ │ │ - * {<OpenLayers.Handler.Click>} object created when the hover │ │ │ │ │ - * option is set to false. Default is "click". │ │ │ │ │ + * APIProperty: fetchEvent │ │ │ │ │ + * {String} The layer event to listen to for replacing remote resource tile │ │ │ │ │ + * URLs with cached data URIs. Supported values are "tileerror" (try │ │ │ │ │ + * remote first, fall back to cached) and "tileloadstart" (try cache │ │ │ │ │ + * first, fall back to remote). Default is "tileloadstart". │ │ │ │ │ + * │ │ │ │ │ + * Note that "tileerror" will not work for CORS enabled images (see │ │ │ │ │ + * https://developer.mozilla.org/en/CORS_Enabled_Image), i.e. layers │ │ │ │ │ + * configured with a <OpenLayers.Tile.Image.crossOriginKeyword> in │ │ │ │ │ + * <OpenLayers.Layer.Grid.tileOptions>. │ │ │ │ │ */ │ │ │ │ │ - clickCallback: "click", │ │ │ │ │ + fetchEvent: "tileloadstart", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: layers │ │ │ │ │ - * {Array(<OpenLayers.Layer.WMTS>)} The layers to query for feature info. │ │ │ │ │ - * If omitted, all map WMTS layers will be considered. │ │ │ │ │ + * APIProperty: layers │ │ │ │ │ + * {Array(<OpenLayers.Layer.Grid>)}. Optional. If provided, only these │ │ │ │ │ + * layers will receive tiles from the cache. │ │ │ │ │ */ │ │ │ │ │ layers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: queryVisible │ │ │ │ │ - * {Boolean} Filter out hidden layers when searching the map for layers to │ │ │ │ │ - * query. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - queryVisible: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: infoFormat │ │ │ │ │ - * {String} The mimetype to request from the server │ │ │ │ │ - */ │ │ │ │ │ - infoFormat: 'text/html', │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: vendorParams │ │ │ │ │ - * {Object} Additional parameters that will be added to the request, for │ │ │ │ │ - * WMTS implementations that support them. This could e.g. look like │ │ │ │ │ - * (start code) │ │ │ │ │ - * { │ │ │ │ │ - * radius: 5 │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ - vendorParams: {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: format │ │ │ │ │ - * {<OpenLayers.Format>} A format for parsing GetFeatureInfo responses. │ │ │ │ │ - * Default is <OpenLayers.Format.WMSGetFeatureInfo>. │ │ │ │ │ - */ │ │ │ │ │ - format: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: formatOptions │ │ │ │ │ - * {Object} Optional properties to set on the format (if one is not provided │ │ │ │ │ - * in the <format> property. │ │ │ │ │ - */ │ │ │ │ │ - formatOptions: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: handlerOptions │ │ │ │ │ - * {Object} Additional options for the handlers used by this control, e.g. │ │ │ │ │ - * (start code) │ │ │ │ │ - * { │ │ │ │ │ - * "click": {delay: 100}, │ │ │ │ │ - * "hover": {delay: 300} │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: handler │ │ │ │ │ - * {Object} Reference to the <OpenLayers.Handler> for this control │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ - handler: null, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: hoverRequest │ │ │ │ │ - * {<OpenLayers.Request>} contains the currently running hover request │ │ │ │ │ - * (if any). │ │ │ │ │ - */ │ │ │ │ │ - hoverRequest: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ + * Constructor: OpenLayers.Control.CacheRead │ │ │ │ │ * │ │ │ │ │ - * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * beforegetfeatureinfo - Triggered before each request is sent. │ │ │ │ │ - * The event object has an *xy* property with the position of the │ │ │ │ │ - * mouse click or hover event that triggers the request and a *layer* │ │ │ │ │ - * property referencing the layer about to be queried. If a listener │ │ │ │ │ - * returns false, the request will not be issued. │ │ │ │ │ - * getfeatureinfo - Triggered when a GetFeatureInfo response is received. │ │ │ │ │ - * The event object has a *text* property with the body of the │ │ │ │ │ - * response (String), a *features* property with an array of the │ │ │ │ │ - * parsed features, an *xy* property with the position of the mouse │ │ │ │ │ - * click or hover event that triggered the request, a *layer* property │ │ │ │ │ - * referencing the layer queried and a *request* property with the │ │ │ │ │ - * request itself. If drillDown is set to true, one event will be fired │ │ │ │ │ - * for each layer queried. │ │ │ │ │ - * exception - Triggered when a GetFeatureInfo request fails (with a │ │ │ │ │ - * status other than 200) or whenparsing fails. Listeners will receive │ │ │ │ │ - * an event with *request*, *xy*, and *layer* properties. In the case │ │ │ │ │ - * of a parsing error, the event will also contain an *error* property. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Object with API properties for this control │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: pending │ │ │ │ │ - * {Number} The number of pending requests. │ │ │ │ │ - */ │ │ │ │ │ - pending: 0, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: <OpenLayers.Control.WMTSGetFeatureInfo> │ │ │ │ │ - * │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the control. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - │ │ │ │ │ - if (!this.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.WMSGetFeatureInfo( │ │ │ │ │ - options.formatOptions │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.drillDown === true) { │ │ │ │ │ - this.hover = false; │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + var i, layers = this.layers || map.layers; │ │ │ │ │ + for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + this.addLayer({ │ │ │ │ │ + layer: layers[i] │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Hover( │ │ │ │ │ - this, { │ │ │ │ │ - move: this.cancelHover, │ │ │ │ │ - pause: this.getInfoForHover │ │ │ │ │ - }, │ │ │ │ │ - OpenLayers.Util.extend( │ │ │ │ │ - this.handlerOptions.hover || {}, { │ │ │ │ │ - delay: 250 │ │ │ │ │ - } │ │ │ │ │ - ) │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - var callbacks = {}; │ │ │ │ │ - callbacks[this.clickCallback] = this.getInfoForClick; │ │ │ │ │ - this.handler = new OpenLayers.Handler.Click( │ │ │ │ │ - this, callbacks, this.handlerOptions.click || {} │ │ │ │ │ - ); │ │ │ │ │ + if (!this.layers) { │ │ │ │ │ + map.events.on({ │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + removeLayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getInfoForClick │ │ │ │ │ - * Called on click │ │ │ │ │ + * Method: addLayer │ │ │ │ │ + * Adds a layer to the control. Once added, tiles requested for this layer │ │ │ │ │ + * will be cached. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ + * evt - {Object} Object with a layer property referencing an │ │ │ │ │ + * <OpenLayers.Layer> instance │ │ │ │ │ */ │ │ │ │ │ - getInfoForClick: function(evt) { │ │ │ │ │ - this.request(evt.xy, {}); │ │ │ │ │ + addLayer: function(evt) { │ │ │ │ │ + evt.layer.events.register(this.fetchEvent, this, this.fetch); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getInfoForHover │ │ │ │ │ - * Pause callback for the hover handler │ │ │ │ │ + * Method: removeLayer │ │ │ │ │ + * Removes a layer from the control. Once removed, tiles requested for this │ │ │ │ │ + * layer will no longer be cached. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ - */ │ │ │ │ │ - getInfoForHover: function(evt) { │ │ │ │ │ - this.request(evt.xy, { │ │ │ │ │ - hover: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: cancelHover │ │ │ │ │ - * Cancel callback for the hover handler │ │ │ │ │ - */ │ │ │ │ │ - cancelHover: function() { │ │ │ │ │ - if (this.hoverRequest) { │ │ │ │ │ - --this.pending; │ │ │ │ │ - if (this.pending <= 0) { │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - this.pending = 0; │ │ │ │ │ - } │ │ │ │ │ - this.hoverRequest.abort(); │ │ │ │ │ - this.hoverRequest = null; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: findLayers │ │ │ │ │ - * Internal method to get the layers, independent of whether we are │ │ │ │ │ - * inspecting the map or using a client-provided array │ │ │ │ │ + * evt - {Object} Object with a layer property referencing an │ │ │ │ │ + * <OpenLayers.Layer> instance │ │ │ │ │ */ │ │ │ │ │ - findLayers: function() { │ │ │ │ │ - var candidates = this.layers || this.map.layers; │ │ │ │ │ - var layers = []; │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = candidates[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.WMTS && │ │ │ │ │ - layer.requestEncoding === this.requestEncoding && │ │ │ │ │ - (!this.queryVisible || layer.getVisibility())) { │ │ │ │ │ - layers.push(layer); │ │ │ │ │ - if (!this.drillDown || this.hover) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return layers; │ │ │ │ │ + removeLayer: function(evt) { │ │ │ │ │ + evt.layer.events.unregister(this.fetchEvent, this, this.fetch); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildRequestOptions │ │ │ │ │ - * Build an object with the relevant options for the GetFeatureInfo request. │ │ │ │ │ + * Method: fetch │ │ │ │ │ + * Listener to the <fetchEvent> event. Replaces a tile's url with a data │ │ │ │ │ + * URI from the cache. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.WMTS>} A WMTS layer. │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} The position on the map where the │ │ │ │ │ - * mouse event occurred. │ │ │ │ │ - */ │ │ │ │ │ - buildRequestOptions: function(layer, xy) { │ │ │ │ │ - var loc = this.map.getLonLatFromPixel(xy); │ │ │ │ │ - var getTileUrl = layer.getURL( │ │ │ │ │ - new OpenLayers.Bounds(loc.lon, loc.lat, loc.lon, loc.lat) │ │ │ │ │ - ); │ │ │ │ │ - var params = OpenLayers.Util.getParameters(getTileUrl); │ │ │ │ │ - var tileInfo = layer.getTileInfo(loc); │ │ │ │ │ - OpenLayers.Util.extend(params, { │ │ │ │ │ - service: "WMTS", │ │ │ │ │ - version: layer.version, │ │ │ │ │ - request: "GetFeatureInfo", │ │ │ │ │ - infoFormat: this.infoFormat, │ │ │ │ │ - i: tileInfo.i, │ │ │ │ │ - j: tileInfo.j │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Util.applyDefaults(params, this.vendorParams); │ │ │ │ │ - return { │ │ │ │ │ - url: OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url, │ │ │ │ │ - params: OpenLayers.Util.upperCaseObject(params), │ │ │ │ │ - callback: function(request) { │ │ │ │ │ - this.handleResponse(xy, request, layer); │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: request │ │ │ │ │ - * Sends a GetFeatureInfo request to the WMTS │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} The position on the map where the mouse event │ │ │ │ │ - * occurred. │ │ │ │ │ - * options - {Object} additional options for this method. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * - *hover* {Boolean} true if we do the request for the hover handler │ │ │ │ │ + * evt - {Object} Event object with a tile property. │ │ │ │ │ */ │ │ │ │ │ - request: function(xy, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var layers = this.findLayers(); │ │ │ │ │ - if (layers.length > 0) { │ │ │ │ │ - var issue, layer; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - layer = layers[i]; │ │ │ │ │ - issue = this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ - xy: xy, │ │ │ │ │ - layer: layer │ │ │ │ │ - }); │ │ │ │ │ - if (issue !== false) { │ │ │ │ │ - ++this.pending; │ │ │ │ │ - var requestOptions = this.buildRequestOptions(layer, xy); │ │ │ │ │ - var request = OpenLayers.Request.GET(requestOptions); │ │ │ │ │ - if (options.hover === true) { │ │ │ │ │ - this.hoverRequest = request; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + fetch: function(evt) { │ │ │ │ │ + if (this.active && window.localStorage && │ │ │ │ │ + evt.tile instanceof OpenLayers.Tile.Image) { │ │ │ │ │ + var tile = evt.tile, │ │ │ │ │ + url = tile.url; │ │ │ │ │ + // deal with modified tile urls when both CacheWrite and CacheRead │ │ │ │ │ + // are active │ │ │ │ │ + if (!tile.layer.crossOriginKeyword && OpenLayers.ProxyHost && │ │ │ │ │ + url.indexOf(OpenLayers.ProxyHost) === 0) { │ │ │ │ │ + url = OpenLayers.Control.CacheWrite.urlMap[url]; │ │ │ │ │ } │ │ │ │ │ - if (this.pending > 0) { │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + var dataURI = window.localStorage.getItem("olCache_" + url); │ │ │ │ │ + if (dataURI) { │ │ │ │ │ + tile.url = dataURI; │ │ │ │ │ + if (evt.type === "tileerror") { │ │ │ │ │ + tile.setImgSrc(dataURI); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleResponse │ │ │ │ │ - * Handler for the GetFeatureInfo response. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} The position on the map where the mouse event │ │ │ │ │ - * occurred. │ │ │ │ │ - * request - {XMLHttpRequest} The request object. │ │ │ │ │ - * layer - {<OpenLayers.Layer.WMTS>} The queried layer. │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * The destroy method is used to perform any clean up before the control │ │ │ │ │ + * is dereferenced. Typically this is where event listeners are removed │ │ │ │ │ + * to prevent memory leaks. │ │ │ │ │ */ │ │ │ │ │ - handleResponse: function(xy, request, layer) { │ │ │ │ │ - --this.pending; │ │ │ │ │ - if (this.pending <= 0) { │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - this.pending = 0; │ │ │ │ │ - } │ │ │ │ │ - if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ - this.events.triggerEvent("exception", { │ │ │ │ │ - xy: xy, │ │ │ │ │ - request: request, │ │ │ │ │ - layer: layer │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ - } │ │ │ │ │ - var features, except; │ │ │ │ │ - try { │ │ │ │ │ - features = this.format.read(doc); │ │ │ │ │ - } catch (error) { │ │ │ │ │ - except = true; │ │ │ │ │ - this.events.triggerEvent("exception", { │ │ │ │ │ - xy: xy, │ │ │ │ │ - request: request, │ │ │ │ │ - error: error, │ │ │ │ │ - layer: layer │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - if (!except) { │ │ │ │ │ - this.events.triggerEvent("getfeatureinfo", { │ │ │ │ │ - text: request.responseText, │ │ │ │ │ - features: features, │ │ │ │ │ - request: request, │ │ │ │ │ - xy: xy, │ │ │ │ │ - layer: layer │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.layers || this.map) { │ │ │ │ │ + var i, layers = this.layers || this.map.layers; │ │ │ │ │ + for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + this.removeLayer({ │ │ │ │ │ + layer: layers[i] │ │ │ │ │ }); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.WMTSGetFeatureInfo" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/PinchZoom.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Handler/Pinch.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.PinchZoom │ │ │ │ │ - * │ │ │ │ │ - * Inherits: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.PinchZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: type │ │ │ │ │ - * {OpenLayers.Control.TYPES} │ │ │ │ │ - */ │ │ │ │ │ - type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: pinchOrigin │ │ │ │ │ - * {Object} Cached object representing the pinch start (in pixels). │ │ │ │ │ - */ │ │ │ │ │ - pinchOrigin: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: currentCenter │ │ │ │ │ - * {Object} Cached object representing the latest pinch center (in pixels). │ │ │ │ │ - */ │ │ │ │ │ - currentCenter: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ - */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: preserveCenter │ │ │ │ │ - * {Boolean} Set this to true if you don't want the map center to change │ │ │ │ │ - * while pinching. For example you may want to set preserveCenter to │ │ │ │ │ - * true when the user location is being watched and you want to preserve │ │ │ │ │ - * the user location at the center of the map even if he zooms in or │ │ │ │ │ - * out using pinch. This property's value can be changed any time on an │ │ │ │ │ - * existing instance. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - preserveCenter: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: handlerOptions │ │ │ │ │ - * {Object} Used to set non-default properties on the pinch handler │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.PinchZoom │ │ │ │ │ - * Create a control for zooming with pinch gestures. This works on devices │ │ │ │ │ - * with multi-touch support. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * the control │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.handler = new OpenLayers.Handler.Pinch(this, { │ │ │ │ │ - start: this.pinchStart, │ │ │ │ │ - move: this.pinchMove, │ │ │ │ │ - done: this.pinchDone │ │ │ │ │ - }, this.handlerOptions); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: pinchStart │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * pinchData - {Object} pinch data object related to the current touchmove │ │ │ │ │ - * of the pinch gesture. This give us the current scale of the pinch. │ │ │ │ │ - */ │ │ │ │ │ - pinchStart: function(evt, pinchData) { │ │ │ │ │ - var xy = (this.preserveCenter) ? │ │ │ │ │ - this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ - this.pinchOrigin = xy; │ │ │ │ │ - this.currentCenter = xy; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: pinchMove │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * pinchData - {Object} pinch data object related to the current touchmove │ │ │ │ │ - * of the pinch gesture. This give us the current scale of the pinch. │ │ │ │ │ - */ │ │ │ │ │ - pinchMove: function(evt, pinchData) { │ │ │ │ │ - var scale = pinchData.scale; │ │ │ │ │ - var containerOrigin = this.map.layerContainerOriginPx; │ │ │ │ │ - var pinchOrigin = this.pinchOrigin; │ │ │ │ │ - var current = (this.preserveCenter) ? │ │ │ │ │ - this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ - │ │ │ │ │ - var dx = Math.round((containerOrigin.x + current.x - pinchOrigin.x) + (scale - 1) * (containerOrigin.x - pinchOrigin.x)); │ │ │ │ │ - var dy = Math.round((containerOrigin.y + current.y - pinchOrigin.y) + (scale - 1) * (containerOrigin.y - pinchOrigin.y)); │ │ │ │ │ - │ │ │ │ │ - this.map.applyTransform(dx, dy, scale); │ │ │ │ │ - this.currentCenter = current; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: pinchDone │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * start - {Object} pinch data object related to the touchstart event that │ │ │ │ │ - * started the pinch gesture. │ │ │ │ │ - * last - {Object} pinch data object related to the last touchmove event │ │ │ │ │ - * of the pinch gesture. This give us the final scale of the pinch. │ │ │ │ │ - */ │ │ │ │ │ - pinchDone: function(evt, start, last) { │ │ │ │ │ - this.map.applyTransform(); │ │ │ │ │ - var zoom = this.map.getZoomForResolution(this.map.getResolution() / last.scale, true); │ │ │ │ │ - if (zoom !== this.map.getZoom() || !this.currentCenter.equals(this.pinchOrigin)) { │ │ │ │ │ - var resolution = this.map.getResolutionForZoom(zoom); │ │ │ │ │ - │ │ │ │ │ - var location = this.map.getLonLatFromPixel(this.pinchOrigin); │ │ │ │ │ - var zoomPixel = this.currentCenter; │ │ │ │ │ - var size = this.map.getSize(); │ │ │ │ │ - │ │ │ │ │ - location.lon += resolution * ((size.w / 2) - zoomPixel.x); │ │ │ │ │ - location.lat -= resolution * ((size.h / 2) - zoomPixel.y); │ │ │ │ │ - │ │ │ │ │ - // Force a reflow before calling setCenter. This is to work │ │ │ │ │ - // around an issue occuring in iOS. │ │ │ │ │ - // │ │ │ │ │ - // See https://github.com/openlayers/openlayers/pull/351. │ │ │ │ │ - // │ │ │ │ │ - // Without a reflow setting the layer container div's top left │ │ │ │ │ - // style properties to "0px" - as done in Map.moveTo when zoom │ │ │ │ │ - // is changed - won't actually correctly reposition the layer │ │ │ │ │ - // container div. │ │ │ │ │ - // │ │ │ │ │ - // Also, we need to use a statement that the Google Closure │ │ │ │ │ - // compiler won't optimize away. │ │ │ │ │ - this.map.div.clientWidth = this.map.div.clientWidth; │ │ │ │ │ - │ │ │ │ │ - this.map.setCenter(location, zoom); │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + removeLayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.PinchZoom" │ │ │ │ │ - │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.CacheRead" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Control/OverviewMap.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ @@ -72207,1255 +69560,539 @@ │ │ │ │ │ }; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: 'OpenLayers.Control.OverviewMap' │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/WMSGetFeatureInfo.js │ │ │ │ │ + OpenLayers/Control/LayerSwitcher.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Handler/Click.js │ │ │ │ │ - * @requires OpenLayers/Handler/Hover.js │ │ │ │ │ - * @requires OpenLayers/Request.js │ │ │ │ │ - * @requires OpenLayers/Format/WMSGetFeatureInfo.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.WMSGetFeatureInfo │ │ │ │ │ - * The WMSGetFeatureInfo control uses a WMS query to get information about a point on the map. The │ │ │ │ │ - * information may be in a display-friendly format such as HTML, or a machine-friendly format such │ │ │ │ │ - * as GML, depending on the server's capabilities and the client's configuration. This control │ │ │ │ │ - * handles click or hover events, attempts to parse the results using an OpenLayers.Format, and │ │ │ │ │ - * fires a 'getfeatureinfo' event with the click position, the raw body of the response, and an │ │ │ │ │ - * array of features if it successfully read the response. │ │ │ │ │ + * Class: OpenLayers.Control.LayerSwitcher │ │ │ │ │ + * The LayerSwitcher control displays a table of contents for the map. This │ │ │ │ │ + * allows the user interface to switch between BaseLasyers and to show or hide │ │ │ │ │ + * Overlays. By default the switcher is shown minimized on the right edge of │ │ │ │ │ + * the map, the user may expand it by clicking on the handle. │ │ │ │ │ + * │ │ │ │ │ + * To create the LayerSwitcher outside of the map, pass the Id of a html div │ │ │ │ │ + * as the first argument to the constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.WMSGetFeatureInfo = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Control.LayerSwitcher = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: layerStates │ │ │ │ │ + * {Array(Object)} Basically a copy of the "state" of the map's layers │ │ │ │ │ + * the last time the control was drawn. We have this in order to avoid │ │ │ │ │ + * unnecessarily redrawing the control. │ │ │ │ │ + */ │ │ │ │ │ + layerStates: null, │ │ │ │ │ + │ │ │ │ │ + // DOM Elements │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: hover │ │ │ │ │ - * {Boolean} Send GetFeatureInfo requests when mouse stops moving. │ │ │ │ │ - * Default is false. │ │ │ │ │ + * Property: layersDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - hover: false, │ │ │ │ │ + layersDiv: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: drillDown │ │ │ │ │ - * {Boolean} Drill down over all WMS layers in the map. When │ │ │ │ │ - * using drillDown mode, hover is not possible, and an infoFormat that │ │ │ │ │ - * returns parseable features is required. Default is false. │ │ │ │ │ + * Property: baseLayersDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - drillDown: false, │ │ │ │ │ + baseLayersDiv: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: maxFeatures │ │ │ │ │ - * {Integer} Maximum number of features to return from a WMS query. This │ │ │ │ │ - * sets the feature_count parameter on WMS GetFeatureInfo │ │ │ │ │ - * requests. │ │ │ │ │ + * Property: baseLayers │ │ │ │ │ + * {Array(Object)} │ │ │ │ │ */ │ │ │ │ │ - maxFeatures: 10, │ │ │ │ │ + baseLayers: null, │ │ │ │ │ + │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: clickCallback │ │ │ │ │ - * {String} The click callback to register in the │ │ │ │ │ - * {<OpenLayers.Handler.Click>} object created when the hover │ │ │ │ │ - * option is set to false. Default is "click". │ │ │ │ │ + * Property: dataLbl │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - clickCallback: "click", │ │ │ │ │ + dataLbl: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: output │ │ │ │ │ - * {String} Either "features" or "object". When triggering a getfeatureinfo │ │ │ │ │ - * request should we pass on an array of features or an object with with │ │ │ │ │ - * a "features" property and other properties (such as the url of the │ │ │ │ │ - * WMS). Default is "features". │ │ │ │ │ + * Property: dataLayersDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - output: "features", │ │ │ │ │ + dataLayersDiv: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: layers │ │ │ │ │ - * {Array(<OpenLayers.Layer.WMS>)} The layers to query for feature info. │ │ │ │ │ - * If omitted, all map WMS layers with a url that matches this <url> or │ │ │ │ │ - * <layerUrls> will be considered. │ │ │ │ │ - */ │ │ │ │ │ - layers: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: queryVisible │ │ │ │ │ - * {Boolean} If true, filter out hidden layers when searching the map for │ │ │ │ │ - * layers to query. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - queryVisible: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String} The URL of the WMS service to use. If not provided, the url │ │ │ │ │ - * of the first eligible layer will be used. │ │ │ │ │ - */ │ │ │ │ │ - url: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: layerUrls │ │ │ │ │ - * {Array(String)} Optional list of urls for layers that should be queried. │ │ │ │ │ - * This can be used when the layer url differs from the url used for │ │ │ │ │ - * making GetFeatureInfo requests (in the case of a layer using cached │ │ │ │ │ - * tiles). │ │ │ │ │ - */ │ │ │ │ │ - layerUrls: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: infoFormat │ │ │ │ │ - * {String} The mimetype to request from the server. If you are using │ │ │ │ │ - * drillDown mode and have multiple servers that do not share a common │ │ │ │ │ - * infoFormat, you can override the control's infoFormat by providing an │ │ │ │ │ - * INFO_FORMAT parameter in your <OpenLayers.Layer.WMS> instance(s). │ │ │ │ │ - */ │ │ │ │ │ - infoFormat: 'text/html', │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: vendorParams │ │ │ │ │ - * {Object} Additional parameters that will be added to the request, for │ │ │ │ │ - * WMS implementations that support them. This could e.g. look like │ │ │ │ │ - * (start code) │ │ │ │ │ - * { │ │ │ │ │ - * radius: 5 │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ - vendorParams: {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: format │ │ │ │ │ - * {<OpenLayers.Format>} A format for parsing GetFeatureInfo responses. │ │ │ │ │ - * Default is <OpenLayers.Format.WMSGetFeatureInfo>. │ │ │ │ │ - */ │ │ │ │ │ - format: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: formatOptions │ │ │ │ │ - * {Object} Optional properties to set on the format (if one is not provided │ │ │ │ │ - * in the <format> property. │ │ │ │ │ + * Property: dataLayers │ │ │ │ │ + * {Array(Object)} │ │ │ │ │ */ │ │ │ │ │ - formatOptions: null, │ │ │ │ │ + dataLayers: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: handlerOptions │ │ │ │ │ - * {Object} Additional options for the handlers used by this control, e.g. │ │ │ │ │ - * (start code) │ │ │ │ │ - * { │ │ │ │ │ - * "click": {delay: 100}, │ │ │ │ │ - * "hover": {delay: 300} │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: handler │ │ │ │ │ - * {Object} Reference to the <OpenLayers.Handler> for this control │ │ │ │ │ + * Property: minimizeDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - handler: null, │ │ │ │ │ + minimizeDiv: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: hoverRequest │ │ │ │ │ - * {<OpenLayers.Request>} contains the currently running hover request │ │ │ │ │ - * (if any). │ │ │ │ │ + * Property: maximizeDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - hoverRequest: null, │ │ │ │ │ + maximizeDiv: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * beforegetfeatureinfo - Triggered before the request is sent. │ │ │ │ │ - * The event object has an *xy* property with the position of the │ │ │ │ │ - * mouse click or hover event that triggers the request. │ │ │ │ │ - * nogetfeatureinfo - no queryable layers were found. │ │ │ │ │ - * getfeatureinfo - Triggered when a GetFeatureInfo response is received. │ │ │ │ │ - * The event object has a *text* property with the body of the │ │ │ │ │ - * response (String), a *features* property with an array of the │ │ │ │ │ - * parsed features, an *xy* property with the position of the mouse │ │ │ │ │ - * click or hover event that triggered the request, and a *request* │ │ │ │ │ - * property with the request itself. If drillDown is set to true and │ │ │ │ │ - * multiple requests were issued to collect feature info from all │ │ │ │ │ - * layers, *text* and *request* will only contain the response body │ │ │ │ │ - * and request object of the last request. │ │ │ │ │ + * APIProperty: ascending │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ + ascending: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: <OpenLayers.Control.WMSGetFeatureInfo> │ │ │ │ │ + * Constructor: OpenLayers.Control.LayerSwitcher │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * options - {Object} │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.layerStates = []; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ │ │ │ │ │ - if (!this.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.WMSGetFeatureInfo( │ │ │ │ │ - options.formatOptions │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + //clear out layers info and unregister their events │ │ │ │ │ + this.clearLayersArray("base"); │ │ │ │ │ + this.clearLayersArray("data"); │ │ │ │ │ │ │ │ │ │ - if (this.drillDown === true) { │ │ │ │ │ - this.hover = false; │ │ │ │ │ - } │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + buttonclick: this.onButtonClick, │ │ │ │ │ + addlayer: this.redraw, │ │ │ │ │ + changelayer: this.redraw, │ │ │ │ │ + removelayer: this.redraw, │ │ │ │ │ + changebaselayer: this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Hover( │ │ │ │ │ - this, { │ │ │ │ │ - 'move': this.cancelHover, │ │ │ │ │ - 'pause': this.getInfoForHover │ │ │ │ │ - }, │ │ │ │ │ - OpenLayers.Util.extend(this.handlerOptions.hover || {}, { │ │ │ │ │ - 'delay': 250 │ │ │ │ │ - })); │ │ │ │ │ - } else { │ │ │ │ │ - var callbacks = {}; │ │ │ │ │ - callbacks[this.clickCallback] = this.getInfoForClick; │ │ │ │ │ - this.handler = new OpenLayers.Handler.Click( │ │ │ │ │ - this, callbacks, this.handlerOptions.click || {}); │ │ │ │ │ - } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getInfoForClick │ │ │ │ │ - * Called on click │ │ │ │ │ + * Method: setMap │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ + * Properties: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - getInfoForClick: function(evt) { │ │ │ │ │ - this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ - xy: evt.xy │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + addlayer: this.redraw, │ │ │ │ │ + changelayer: this.redraw, │ │ │ │ │ + removelayer: this.redraw, │ │ │ │ │ + changebaselayer: this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ }); │ │ │ │ │ - // Set the cursor to "wait" to tell the user we're working on their │ │ │ │ │ - // click. │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - this.request(evt.xy, {}); │ │ │ │ │ + if (this.outsideViewport) { │ │ │ │ │ + this.events.attachToElement(this.div); │ │ │ │ │ + this.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ + } else { │ │ │ │ │ + this.map.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getInfoForHover │ │ │ │ │ - * Pause callback for the hover handler │ │ │ │ │ + * Method: draw │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A reference to the DIV DOMElement containing the │ │ │ │ │ + * switcher tabs. │ │ │ │ │ */ │ │ │ │ │ - getInfoForHover: function(evt) { │ │ │ │ │ - this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ - xy: evt.xy │ │ │ │ │ - }); │ │ │ │ │ - this.request(evt.xy, { │ │ │ │ │ - hover: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: cancelHover │ │ │ │ │ - * Cancel callback for the hover handler │ │ │ │ │ - */ │ │ │ │ │ - cancelHover: function() { │ │ │ │ │ - if (this.hoverRequest) { │ │ │ │ │ - this.hoverRequest.abort(); │ │ │ │ │ - this.hoverRequest = null; │ │ │ │ │ + // create layout divs │ │ │ │ │ + this.loadContents(); │ │ │ │ │ + │ │ │ │ │ + // set mode to minimize │ │ │ │ │ + if (!this.outsideViewport) { │ │ │ │ │ + this.minimizeControl(); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: findLayers │ │ │ │ │ - * Internal method to get the layers, independent of whether we are │ │ │ │ │ - * inspecting the map or using a client-provided array │ │ │ │ │ - */ │ │ │ │ │ - findLayers: function() { │ │ │ │ │ + // populate div with current info │ │ │ │ │ + this.redraw(); │ │ │ │ │ │ │ │ │ │ - var candidates = this.layers || this.map.layers; │ │ │ │ │ - var layers = []; │ │ │ │ │ - var layer, url; │ │ │ │ │ - for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = candidates[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.WMS && │ │ │ │ │ - (!this.queryVisible || layer.getVisibility())) { │ │ │ │ │ - url = OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url; │ │ │ │ │ - // if the control was not configured with a url, set it │ │ │ │ │ - // to the first layer url │ │ │ │ │ - if (this.drillDown === false && !this.url) { │ │ │ │ │ - this.url = url; │ │ │ │ │ - } │ │ │ │ │ - if (this.drillDown === true || this.urlMatches(url)) { │ │ │ │ │ - layers.push(layer); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return layers; │ │ │ │ │ + return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: urlMatches │ │ │ │ │ - * Test to see if the provided url matches either the control <url> or one │ │ │ │ │ - * of the <layerUrls>. │ │ │ │ │ + * Method: onButtonClick │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * url - {String} The url to test. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The provided url matches the control <url> or one of the │ │ │ │ │ - * <layerUrls>. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - urlMatches: function(url) { │ │ │ │ │ - var matches = OpenLayers.Util.isEquivalentUrl(this.url, url); │ │ │ │ │ - if (!matches && this.layerUrls) { │ │ │ │ │ - for (var i = 0, len = this.layerUrls.length; i < len; ++i) { │ │ │ │ │ - if (OpenLayers.Util.isEquivalentUrl(this.layerUrls[i], url)) { │ │ │ │ │ - matches = true; │ │ │ │ │ - break; │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + var button = evt.buttonElement; │ │ │ │ │ + if (button === this.minimizeDiv) { │ │ │ │ │ + this.minimizeControl(); │ │ │ │ │ + } else if (button === this.maximizeDiv) { │ │ │ │ │ + this.maximizeControl(); │ │ │ │ │ + } else if (button._layerSwitcher === this.id) { │ │ │ │ │ + if (button["for"]) { │ │ │ │ │ + button = document.getElementById(button["for"]); │ │ │ │ │ + } │ │ │ │ │ + if (!button.disabled) { │ │ │ │ │ + if (button.type == "radio") { │ │ │ │ │ + button.checked = true; │ │ │ │ │ + this.map.setBaseLayer(this.map.getLayer(button._layer)); │ │ │ │ │ + } else { │ │ │ │ │ + button.checked = !button.checked; │ │ │ │ │ + this.updateMap(); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return matches; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildWMSOptions │ │ │ │ │ - * Build an object with the relevant WMS options for the GetFeatureInfo request │ │ │ │ │ + * Method: clearLayersArray │ │ │ │ │ + * User specifies either "base" or "data". we then clear all the │ │ │ │ │ + * corresponding listeners, the div, and reinitialize a new array. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * url - {String} The url to be used for sending the request │ │ │ │ │ - * layers - {Array(<OpenLayers.Layer.WMS)} An array of layers │ │ │ │ │ - * clickPosition - {<OpenLayers.Pixel>} The position on the map where the mouse │ │ │ │ │ - * event occurred. │ │ │ │ │ - * format - {String} The format from the corresponding GetMap request │ │ │ │ │ + * layersType - {String} │ │ │ │ │ */ │ │ │ │ │ - buildWMSOptions: function(url, layers, clickPosition, format) { │ │ │ │ │ - var layerNames = [], │ │ │ │ │ - styleNames = []; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - if (layers[i].params.LAYERS != null) { │ │ │ │ │ - layerNames = layerNames.concat(layers[i].params.LAYERS); │ │ │ │ │ - styleNames = styleNames.concat(this.getStyleNames(layers[i])); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var firstLayer = layers[0]; │ │ │ │ │ - // use the firstLayer's projection if it matches the map projection - │ │ │ │ │ - // this assumes that all layers will be available in this projection │ │ │ │ │ - var projection = this.map.getProjection(); │ │ │ │ │ - var layerProj = firstLayer.projection; │ │ │ │ │ - if (layerProj && layerProj.equals(this.map.getProjectionObject())) { │ │ │ │ │ - projection = layerProj.getCode(); │ │ │ │ │ - } │ │ │ │ │ - var params = OpenLayers.Util.extend({ │ │ │ │ │ - service: "WMS", │ │ │ │ │ - version: firstLayer.params.VERSION, │ │ │ │ │ - request: "GetFeatureInfo", │ │ │ │ │ - exceptions: firstLayer.params.EXCEPTIONS, │ │ │ │ │ - bbox: this.map.getExtent().toBBOX(null, │ │ │ │ │ - firstLayer.reverseAxisOrder()), │ │ │ │ │ - feature_count: this.maxFeatures, │ │ │ │ │ - height: this.map.getSize().h, │ │ │ │ │ - width: this.map.getSize().w, │ │ │ │ │ - format: format, │ │ │ │ │ - info_format: firstLayer.params.INFO_FORMAT || this.infoFormat │ │ │ │ │ - }, (parseFloat(firstLayer.params.VERSION) >= 1.3) ? { │ │ │ │ │ - crs: projection, │ │ │ │ │ - i: parseInt(clickPosition.x), │ │ │ │ │ - j: parseInt(clickPosition.y) │ │ │ │ │ - } : { │ │ │ │ │ - srs: projection, │ │ │ │ │ - x: parseInt(clickPosition.x), │ │ │ │ │ - y: parseInt(clickPosition.y) │ │ │ │ │ - }); │ │ │ │ │ - if (layerNames.length != 0) { │ │ │ │ │ - params = OpenLayers.Util.extend({ │ │ │ │ │ - layers: layerNames, │ │ │ │ │ - query_layers: layerNames, │ │ │ │ │ - styles: styleNames │ │ │ │ │ - }, params); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.applyDefaults(params, this.vendorParams); │ │ │ │ │ - return { │ │ │ │ │ - url: url, │ │ │ │ │ - params: OpenLayers.Util.upperCaseObject(params), │ │ │ │ │ - callback: function(request) { │ │ │ │ │ - this.handleResponse(clickPosition, request, url); │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }; │ │ │ │ │ + clearLayersArray: function(layersType) { │ │ │ │ │ + this[layersType + "LayersDiv"].innerHTML = ""; │ │ │ │ │ + this[layersType + "Layers"] = []; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getStyleNames │ │ │ │ │ - * Gets the STYLES parameter for the layer. Make sure the STYLES parameter │ │ │ │ │ - * matches the LAYERS parameter │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.WMS>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(String)} The STYLES parameter │ │ │ │ │ - */ │ │ │ │ │ - getStyleNames: function(layer) { │ │ │ │ │ - // in the event of a WMS layer bundling multiple layers but not │ │ │ │ │ - // specifying styles,we need the same number of commas to specify │ │ │ │ │ - // the default style for each of the layers. We can't just leave it │ │ │ │ │ - // blank as we may be including other layers that do specify styles. │ │ │ │ │ - var styleNames; │ │ │ │ │ - if (layer.params.STYLES) { │ │ │ │ │ - styleNames = layer.params.STYLES; │ │ │ │ │ - } else { │ │ │ │ │ - if (OpenLayers.Util.isArray(layer.params.LAYERS)) { │ │ │ │ │ - styleNames = new Array(layer.params.LAYERS.length); │ │ │ │ │ - } else { // Assume it's a String │ │ │ │ │ - styleNames = layer.params.LAYERS.replace(/[^,]/g, ""); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return styleNames; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: request │ │ │ │ │ - * Sends a GetFeatureInfo request to the WMS │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * clickPosition - {<OpenLayers.Pixel>} The position on the map where the │ │ │ │ │ - * mouse event occurred. │ │ │ │ │ - * options - {Object} additional options for this method. │ │ │ │ │ + * Method: checkRedraw │ │ │ │ │ + * Checks if the layer state has changed since the last redraw() call. │ │ │ │ │ * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * - *hover* {Boolean} true if we do the request for the hover handler │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The layer state changed since the last redraw() call. │ │ │ │ │ */ │ │ │ │ │ - request: function(clickPosition, options) { │ │ │ │ │ - var layers = this.findLayers(); │ │ │ │ │ - if (layers.length == 0) { │ │ │ │ │ - this.events.triggerEvent("nogetfeatureinfo"); │ │ │ │ │ - // Reset the cursor. │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - return; │ │ │ │ │ + checkRedraw: function() { │ │ │ │ │ + if (!this.layerStates.length || │ │ │ │ │ + (this.map.layers.length != this.layerStates.length)) { │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - options = options || {}; │ │ │ │ │ - if (this.drillDown === false) { │ │ │ │ │ - var wmsOptions = this.buildWMSOptions(this.url, layers, │ │ │ │ │ - clickPosition, layers[0].params.FORMAT); │ │ │ │ │ - var request = OpenLayers.Request.GET(wmsOptions); │ │ │ │ │ - │ │ │ │ │ - if (options.hover === true) { │ │ │ │ │ - this.hoverRequest = request; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this._requestCount = 0; │ │ │ │ │ - this._numRequests = 0; │ │ │ │ │ - this.features = []; │ │ │ │ │ - // group according to service url to combine requests │ │ │ │ │ - var services = {}, │ │ │ │ │ - url; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - var layer = layers[i]; │ │ │ │ │ - var service, found = false; │ │ │ │ │ - url = OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url; │ │ │ │ │ - if (url in services) { │ │ │ │ │ - services[url].push(layer); │ │ │ │ │ - } else { │ │ │ │ │ - this._numRequests++; │ │ │ │ │ - services[url] = [layer]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var layers; │ │ │ │ │ - for (var url in services) { │ │ │ │ │ - layers = services[url]; │ │ │ │ │ - var wmsOptions = this.buildWMSOptions(url, layers, │ │ │ │ │ - clickPosition, layers[0].params.FORMAT); │ │ │ │ │ - OpenLayers.Request.GET(wmsOptions); │ │ │ │ │ + for (var i = 0, len = this.layerStates.length; i < len; i++) { │ │ │ │ │ + var layerState = this.layerStates[i]; │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if ((layerState.name != layer.name) || │ │ │ │ │ + (layerState.inRange != layer.inRange) || │ │ │ │ │ + (layerState.id != layer.id) || │ │ │ │ │ + (layerState.visibility != layer.visibility)) { │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: triggerGetFeatureInfo │ │ │ │ │ - * Trigger the getfeatureinfo event when all is done │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * request - {XMLHttpRequest} The request object │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} The position on the map where the │ │ │ │ │ - * mouse event occurred. │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} or │ │ │ │ │ - * {Array({Object}) when output is "object". The object has a url and a │ │ │ │ │ - * features property which contains an array of features. │ │ │ │ │ - */ │ │ │ │ │ - triggerGetFeatureInfo: function(request, xy, features) { │ │ │ │ │ - this.events.triggerEvent("getfeatureinfo", { │ │ │ │ │ - text: request.responseText, │ │ │ │ │ - features: features, │ │ │ │ │ - request: request, │ │ │ │ │ - xy: xy │ │ │ │ │ - }); │ │ │ │ │ │ │ │ │ │ - // Reset the cursor. │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleResponse │ │ │ │ │ - * Handler for the GetFeatureInfo response. │ │ │ │ │ + * Method: redraw │ │ │ │ │ + * Goes through and takes the current state of the Map and rebuilds the │ │ │ │ │ + * control to display that state. Groups base layers into a │ │ │ │ │ + * radio-button group and lists each data layer with a checkbox. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} The position on the map where the │ │ │ │ │ - * mouse event occurred. │ │ │ │ │ - * request - {XMLHttpRequest} The request object. │ │ │ │ │ - * url - {String} The url which was used for this request. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ */ │ │ │ │ │ - handleResponse: function(xy, request, url) { │ │ │ │ │ - │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ - } │ │ │ │ │ - var features = this.format.read(doc); │ │ │ │ │ - if (this.drillDown === false) { │ │ │ │ │ - this.triggerGetFeatureInfo(request, xy, features); │ │ │ │ │ - } else { │ │ │ │ │ - this._requestCount++; │ │ │ │ │ - if (this.output === "object") { │ │ │ │ │ - this._features = (this._features || []).concat({ │ │ │ │ │ - url: url, │ │ │ │ │ - features: features │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - this._features = (this._features || []).concat(features); │ │ │ │ │ - } │ │ │ │ │ - if (this._requestCount === this._numRequests) { │ │ │ │ │ - this.triggerGetFeatureInfo(request, xy, this._features.concat()); │ │ │ │ │ - delete this._features; │ │ │ │ │ - delete this._requestCount; │ │ │ │ │ - delete this._numRequests; │ │ │ │ │ - } │ │ │ │ │ + redraw: function() { │ │ │ │ │ + //if the state hasn't changed since last redraw, no need │ │ │ │ │ + // to do anything. Just return the existing div. │ │ │ │ │ + if (!this.checkRedraw()) { │ │ │ │ │ + return this.div; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.WMSGetFeatureInfo" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/EditingToolbar.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control/Panel.js │ │ │ │ │ - * @requires OpenLayers/Control/Navigation.js │ │ │ │ │ - * @requires OpenLayers/Control/DrawFeature.js │ │ │ │ │ - * @requires OpenLayers/Handler/Point.js │ │ │ │ │ - * @requires OpenLayers/Handler/Path.js │ │ │ │ │ - * @requires OpenLayers/Handler/Polygon.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.EditingToolbar │ │ │ │ │ - * The EditingToolbar is a panel of 4 controls to draw polygons, lines, │ │ │ │ │ - * points, or to navigate the map by panning. By default it appears in the │ │ │ │ │ - * upper right corner of the map. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control.Panel> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.EditingToolbar = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Control.Panel, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: citeCompliant │ │ │ │ │ - * {Boolean} If set to true, coordinates of features drawn in a map extent │ │ │ │ │ - * crossing the date line won't exceed the world bounds. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - citeCompliant: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.EditingToolbar │ │ │ │ │ - * Create an editing toolbar for a given layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(layer, options) { │ │ │ │ │ - OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ - │ │ │ │ │ - this.addControls( │ │ │ │ │ - [new OpenLayers.Control.Navigation()] │ │ │ │ │ - ); │ │ │ │ │ - var controls = [ │ │ │ │ │ - new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Point, { │ │ │ │ │ - displayClass: 'olControlDrawFeaturePoint', │ │ │ │ │ - handlerOptions: { │ │ │ │ │ - citeCompliant: this.citeCompliant │ │ │ │ │ - } │ │ │ │ │ - }), │ │ │ │ │ - new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Path, { │ │ │ │ │ - displayClass: 'olControlDrawFeaturePath', │ │ │ │ │ - handlerOptions: { │ │ │ │ │ - citeCompliant: this.citeCompliant │ │ │ │ │ - } │ │ │ │ │ - }), │ │ │ │ │ - new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Polygon, { │ │ │ │ │ - displayClass: 'olControlDrawFeaturePolygon', │ │ │ │ │ - handlerOptions: { │ │ │ │ │ - citeCompliant: this.citeCompliant │ │ │ │ │ - } │ │ │ │ │ - }) │ │ │ │ │ - ]; │ │ │ │ │ - this.addControls(controls); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * calls the default draw, and then activates mouse defaults. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - var div = OpenLayers.Control.Panel.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (this.defaultControl === null) { │ │ │ │ │ - this.defaultControl = this.controls[0]; │ │ │ │ │ - } │ │ │ │ │ - return div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.EditingToolbar" │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Attribution.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Attribution │ │ │ │ │ - * The attribution control adds attribution from layers to the map display. │ │ │ │ │ - * It uses 'attribution' property of each layer. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Attribution = │ │ │ │ │ - OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: separator │ │ │ │ │ - * {String} String used to separate layers. │ │ │ │ │ - */ │ │ │ │ │ - separator: ", ", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: template │ │ │ │ │ - * {String} Template for the attribution. This has to include the substring │ │ │ │ │ - * "${layers}", which will be replaced by the layer specific │ │ │ │ │ - * attributions, separated by <separator>. The default is "${layers}". │ │ │ │ │ - */ │ │ │ │ │ - template: "${layers}", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Attribution │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Options for control. │ │ │ │ │ - */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Destroy control. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - "removelayer": this.updateAttribution, │ │ │ │ │ - "addlayer": this.updateAttribution, │ │ │ │ │ - "changelayer": this.updateAttribution, │ │ │ │ │ - "changebaselayer": this.updateAttribution, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + //clear out previous layers │ │ │ │ │ + this.clearLayersArray("base"); │ │ │ │ │ + this.clearLayersArray("data"); │ │ │ │ │ │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + var containsOverlays = false; │ │ │ │ │ + var containsBaseLayers = false; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Initialize control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + // Save state -- for checking layer if the map state changed. │ │ │ │ │ + // We save this before redrawing, because in the process of redrawing │ │ │ │ │ + // we will trigger more visibility changes, and we want to not redraw │ │ │ │ │ + // and enter an infinite loop. │ │ │ │ │ + var len = this.map.layers.length; │ │ │ │ │ + this.layerStates = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + this.layerStates[i] = { │ │ │ │ │ + 'name': layer.name, │ │ │ │ │ + 'visibility': layer.visibility, │ │ │ │ │ + 'inRange': layer.inRange, │ │ │ │ │ + 'id': layer.id │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - 'changebaselayer': this.updateAttribution, │ │ │ │ │ - 'changelayer': this.updateAttribution, │ │ │ │ │ - 'addlayer': this.updateAttribution, │ │ │ │ │ - 'removelayer': this.updateAttribution, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.updateAttribution(); │ │ │ │ │ + var layers = this.map.layers.slice(); │ │ │ │ │ + if (!this.ascending) { │ │ │ │ │ + layers.reverse(); │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + var layer = layers[i]; │ │ │ │ │ + var baseLayer = layer.isBaseLayer; │ │ │ │ │ │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ + if (layer.displayInLayerSwitcher) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateAttribution │ │ │ │ │ - * Update attribution string. │ │ │ │ │ - */ │ │ │ │ │ - updateAttribution: function() { │ │ │ │ │ - var attributions = []; │ │ │ │ │ - if (this.map && this.map.layers) { │ │ │ │ │ - for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if (layer.attribution && layer.getVisibility()) { │ │ │ │ │ - // add attribution only if attribution text is unique │ │ │ │ │ - if (OpenLayers.Util.indexOf( │ │ │ │ │ - attributions, layer.attribution) === -1) { │ │ │ │ │ - attributions.push(layer.attribution); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + if (baseLayer) { │ │ │ │ │ + containsBaseLayers = true; │ │ │ │ │ + } else { │ │ │ │ │ + containsOverlays = true; │ │ │ │ │ } │ │ │ │ │ - this.div.innerHTML = OpenLayers.String.format(this.template, { │ │ │ │ │ - layers: attributions.join(this.separator) │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Attribution" │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/PanZoom.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + // only check a baselayer if it is *the* baselayer, check data │ │ │ │ │ + // layers if they are visible │ │ │ │ │ + var checked = (baseLayer) ? (layer == this.map.baseLayer) : │ │ │ │ │ + layer.getVisibility(); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ - */ │ │ │ │ │ + // create input element │ │ │ │ │ + var inputElem = document.createElement("input"), │ │ │ │ │ + // The input shall have an id attribute so we can use │ │ │ │ │ + // labels to interact with them. │ │ │ │ │ + inputId = OpenLayers.Util.createUniqueID( │ │ │ │ │ + this.id + "_input_" │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.PanZoom │ │ │ │ │ - * The PanZoom is a visible control, composed of a │ │ │ │ │ - * <OpenLayers.Control.PanPanel> and a <OpenLayers.Control.ZoomPanel>. By │ │ │ │ │ - * default it is drawn in the upper left corner of the map. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.PanZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + inputElem.id = inputId; │ │ │ │ │ + inputElem.name = (baseLayer) ? this.id + "_baseLayers" : layer.name; │ │ │ │ │ + inputElem.type = (baseLayer) ? "radio" : "checkbox"; │ │ │ │ │ + inputElem.value = layer.name; │ │ │ │ │ + inputElem.checked = checked; │ │ │ │ │ + inputElem.defaultChecked = checked; │ │ │ │ │ + inputElem.className = "olButton"; │ │ │ │ │ + inputElem._layer = layer.id; │ │ │ │ │ + inputElem._layerSwitcher = this.id; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: slideFactor │ │ │ │ │ - * {Integer} Number of pixels by which we'll pan the map in any direction │ │ │ │ │ - * on clicking the arrow buttons. If you want to pan by some ratio │ │ │ │ │ - * of the map dimensions, use <slideRatio> instead. │ │ │ │ │ - */ │ │ │ │ │ - slideFactor: 50, │ │ │ │ │ + if (!baseLayer && !layer.inRange) { │ │ │ │ │ + inputElem.disabled = true; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: slideRatio │ │ │ │ │ - * {Number} The fraction of map width/height by which we'll pan the map │ │ │ │ │ - * on clicking the arrow buttons. Default is null. If set, will │ │ │ │ │ - * override <slideFactor>. E.g. if slideRatio is .5, then the Pan Up │ │ │ │ │ - * button will pan up half the map height. │ │ │ │ │ - */ │ │ │ │ │ - slideRatio: null, │ │ │ │ │ + // create span │ │ │ │ │ + var labelSpan = document.createElement("label"); │ │ │ │ │ + // this isn't the DOM attribute 'for', but an arbitrary name we │ │ │ │ │ + // use to find the appropriate input element in <onButtonClick> │ │ │ │ │ + labelSpan["for"] = inputElem.id; │ │ │ │ │ + OpenLayers.Element.addClass(labelSpan, "labelSpan olButton"); │ │ │ │ │ + labelSpan._layer = layer.id; │ │ │ │ │ + labelSpan._layerSwitcher = this.id; │ │ │ │ │ + if (!baseLayer && !layer.inRange) { │ │ │ │ │ + labelSpan.style.color = "gray"; │ │ │ │ │ + } │ │ │ │ │ + labelSpan.innerHTML = layer.name; │ │ │ │ │ + labelSpan.style.verticalAlign = (baseLayer) ? "bottom" : │ │ │ │ │ + "baseline"; │ │ │ │ │ + // create line break │ │ │ │ │ + var br = document.createElement("br"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: buttons │ │ │ │ │ - * {Array(DOMElement)} Array of Button Divs │ │ │ │ │ - */ │ │ │ │ │ - buttons: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: position │ │ │ │ │ - * {<OpenLayers.Pixel>} │ │ │ │ │ - */ │ │ │ │ │ - position: null, │ │ │ │ │ + var groupArray = (baseLayer) ? this.baseLayers : │ │ │ │ │ + this.dataLayers; │ │ │ │ │ + groupArray.push({ │ │ │ │ │ + 'layer': layer, │ │ │ │ │ + 'inputElem': inputElem, │ │ │ │ │ + 'labelSpan': labelSpan │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.PanZoom │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.position = new OpenLayers.Pixel(OpenLayers.Control.PanZoom.X, │ │ │ │ │ - OpenLayers.Control.PanZoom.Y); │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ + var groupDiv = (baseLayer) ? this.baseLayersDiv : │ │ │ │ │ + this.dataLayersDiv; │ │ │ │ │ + groupDiv.appendChild(inputElem); │ │ │ │ │ + groupDiv.appendChild(labelSpan); │ │ │ │ │ + groupDiv.appendChild(br); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.removeButtons(); │ │ │ │ │ - this.buttons = null; │ │ │ │ │ - this.position = null; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * │ │ │ │ │ - * Properties: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A reference to the container div for the PanZoom control. │ │ │ │ │ - */ │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - // initialize our internal div │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - px = this.position; │ │ │ │ │ - │ │ │ │ │ - // place the controls │ │ │ │ │ - this.buttons = []; │ │ │ │ │ + // if no overlays, dont display the overlay label │ │ │ │ │ + this.dataLbl.style.display = (containsOverlays) ? "" : "none"; │ │ │ │ │ │ │ │ │ │ - var sz = { │ │ │ │ │ - w: 18, │ │ │ │ │ - h: 18 │ │ │ │ │ - }; │ │ │ │ │ - var centered = new OpenLayers.Pixel(px.x + sz.w / 2, px.y); │ │ │ │ │ + // if no baselayers, dont display the baselayer label │ │ │ │ │ + this.baseLbl.style.display = (containsBaseLayers) ? "" : "none"; │ │ │ │ │ │ │ │ │ │ - this._addButton("panup", "north-mini.png", centered, sz); │ │ │ │ │ - px.y = centered.y + sz.h; │ │ │ │ │ - this._addButton("panleft", "west-mini.png", px, sz); │ │ │ │ │ - this._addButton("panright", "east-mini.png", px.add(sz.w, 0), sz); │ │ │ │ │ - this._addButton("pandown", "south-mini.png", │ │ │ │ │ - centered.add(0, sz.h * 2), sz); │ │ │ │ │ - this._addButton("zoomin", "zoom-plus-mini.png", │ │ │ │ │ - centered.add(0, sz.h * 3 + 5), sz); │ │ │ │ │ - this._addButton("zoomworld", "zoom-world-mini.png", │ │ │ │ │ - centered.add(0, sz.h * 4 + 5), sz); │ │ │ │ │ - this._addButton("zoomout", "zoom-minus-mini.png", │ │ │ │ │ - centered.add(0, sz.h * 5 + 5), sz); │ │ │ │ │ return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: _addButton │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * img - {String} │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} │ │ │ │ │ - * sz - {<OpenLayers.Size>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A Div (an alphaImageDiv, to be precise) that contains the │ │ │ │ │ - * image of the button, and has all the proper event handlers set. │ │ │ │ │ - */ │ │ │ │ │ - _addButton: function(id, img, xy, sz) { │ │ │ │ │ - var imgLocation = OpenLayers.Util.getImageLocation(img); │ │ │ │ │ - var btn = OpenLayers.Util.createAlphaImageDiv( │ │ │ │ │ - this.id + "_" + id, │ │ │ │ │ - xy, sz, imgLocation, "absolute"); │ │ │ │ │ - btn.style.cursor = "pointer"; │ │ │ │ │ - //we want to add the outer div │ │ │ │ │ - this.div.appendChild(btn); │ │ │ │ │ - btn.action = id; │ │ │ │ │ - btn.className = "olButton"; │ │ │ │ │ - │ │ │ │ │ - //we want to remember/reference the outer div │ │ │ │ │ - this.buttons.push(btn); │ │ │ │ │ - return btn; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: _removeButton │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * btn - {Object} │ │ │ │ │ + * Method: updateMap │ │ │ │ │ + * Cycles through the loaded data and base layer input arrays and makes │ │ │ │ │ + * the necessary calls to the Map object such that that the map's │ │ │ │ │ + * visual state corresponds to what the user has selected in │ │ │ │ │ + * the control. │ │ │ │ │ */ │ │ │ │ │ - _removeButton: function(btn) { │ │ │ │ │ - this.div.removeChild(btn); │ │ │ │ │ - OpenLayers.Util.removeItem(this.buttons, btn); │ │ │ │ │ - }, │ │ │ │ │ + updateMap: function() { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeButtons │ │ │ │ │ - */ │ │ │ │ │ - removeButtons: function() { │ │ │ │ │ - for (var i = this.buttons.length - 1; i >= 0; --i) { │ │ │ │ │ - this._removeButton(this.buttons[i]); │ │ │ │ │ + // set the newly selected base layer │ │ │ │ │ + for (var i = 0, len = this.baseLayers.length; i < len; i++) { │ │ │ │ │ + var layerEntry = this.baseLayers[i]; │ │ │ │ │ + if (layerEntry.inputElem.checked) { │ │ │ │ │ + this.map.setBaseLayer(layerEntry.layer, false); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: onButtonClick │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - var btn = evt.buttonElement; │ │ │ │ │ - switch (btn.action) { │ │ │ │ │ - case "panup": │ │ │ │ │ - this.map.pan(0, -this.getSlideFactor("h")); │ │ │ │ │ - break; │ │ │ │ │ - case "pandown": │ │ │ │ │ - this.map.pan(0, this.getSlideFactor("h")); │ │ │ │ │ - break; │ │ │ │ │ - case "panleft": │ │ │ │ │ - this.map.pan(-this.getSlideFactor("w"), 0); │ │ │ │ │ - break; │ │ │ │ │ - case "panright": │ │ │ │ │ - this.map.pan(this.getSlideFactor("w"), 0); │ │ │ │ │ - break; │ │ │ │ │ - case "zoomin": │ │ │ │ │ - this.map.zoomIn(); │ │ │ │ │ - break; │ │ │ │ │ - case "zoomout": │ │ │ │ │ - this.map.zoomOut(); │ │ │ │ │ - break; │ │ │ │ │ - case "zoomworld": │ │ │ │ │ - this.map.zoomToMaxExtent(); │ │ │ │ │ - break; │ │ │ │ │ + // set the correct visibilities for the overlays │ │ │ │ │ + for (var i = 0, len = this.dataLayers.length; i < len; i++) { │ │ │ │ │ + var layerEntry = this.dataLayers[i]; │ │ │ │ │ + layerEntry.layer.setVisibility(layerEntry.inputElem.checked); │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getSlideFactor │ │ │ │ │ + * Method: maximizeControl │ │ │ │ │ + * Set up the labels and divs for the control │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * dim - {String} "w" or "h" (for width or height). │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number} The slide factor for panning in the requested direction. │ │ │ │ │ - */ │ │ │ │ │ - getSlideFactor: function(dim) { │ │ │ │ │ - return this.slideRatio ? │ │ │ │ │ - this.map.getSize()[dim] * this.slideRatio : │ │ │ │ │ - this.slideFactor; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.PanZoom" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: X │ │ │ │ │ - * {Integer} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.PanZoom.X = 4; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: Y │ │ │ │ │ - * {Integer} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.PanZoom.Y = 4; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Permalink.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Control/ArgParser.js │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Permalink │ │ │ │ │ - * The Permalink control is hyperlink that will return the user to the │ │ │ │ │ - * current map view. By default it is drawn in the lower right corner of the │ │ │ │ │ - * map. The href is updated as the map is zoomed, panned and whilst layers │ │ │ │ │ - * are switched. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Permalink = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: argParserClass │ │ │ │ │ - * {Class} The ArgParser control class (not instance) to use with this │ │ │ │ │ - * control. │ │ │ │ │ - */ │ │ │ │ │ - argParserClass: OpenLayers.Control.ArgParser, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: element │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - element: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: anchor │ │ │ │ │ - * {Boolean} This option changes 3 things: │ │ │ │ │ - * the character '#' is used in place of the character '?', │ │ │ │ │ - * the window.href is updated if no element is provided. │ │ │ │ │ - * When this option is set to true it's not recommend to provide │ │ │ │ │ - * a base without provide an element. │ │ │ │ │ - */ │ │ │ │ │ - anchor: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: base │ │ │ │ │ - * {String} │ │ │ │ │ + * e - {Event} │ │ │ │ │ */ │ │ │ │ │ - base: '', │ │ │ │ │ + maximizeControl: function(e) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: displayProjection │ │ │ │ │ - * {<OpenLayers.Projection>} Requires proj4js support. Projection used │ │ │ │ │ - * when creating the coordinates in the link. This will reproject the │ │ │ │ │ - * map coordinates into display coordinates. If you are using this │ │ │ │ │ - * functionality, the permalink which is last added to the map will │ │ │ │ │ - * determine the coordinate type which is read from the URL, which │ │ │ │ │ - * means you should not add permalinks with different │ │ │ │ │ - * displayProjections to the same map. │ │ │ │ │ - */ │ │ │ │ │ - displayProjection: null, │ │ │ │ │ + // set the div's width and height to empty values, so │ │ │ │ │ + // the div dimensions can be controlled by CSS │ │ │ │ │ + this.div.style.width = ""; │ │ │ │ │ + this.div.style.height = ""; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Permalink │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * element - {DOMElement} │ │ │ │ │ - * base - {String} │ │ │ │ │ - * options - {Object} options to the control. │ │ │ │ │ - * │ │ │ │ │ - * Or for anchor: │ │ │ │ │ - * options - {Object} options to the control. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(element, base, options) { │ │ │ │ │ - if (element !== null && typeof element == 'object' && !OpenLayers.Util.isElement(element)) { │ │ │ │ │ - options = element; │ │ │ │ │ - this.base = document.location.href; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (this.element != null) { │ │ │ │ │ - this.element = OpenLayers.Util.getElement(this.element); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.element = OpenLayers.Util.getElement(element); │ │ │ │ │ - this.base = base || document.location.href; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + this.showControls(false); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.element && this.element.parentNode == this.div) { │ │ │ │ │ - this.div.removeChild(this.element); │ │ │ │ │ - this.element = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister('moveend', this, this.updateLink); │ │ │ │ │ + if (e != null) { │ │ │ │ │ + OpenLayers.Event.stop(e); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the control. │ │ │ │ │ - * │ │ │ │ │ + * Method: minimizeControl │ │ │ │ │ + * Hide all the contents of the control, shrink the size, │ │ │ │ │ + * add the maximize icon │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * e - {Event} │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + minimizeControl: function(e) { │ │ │ │ │ │ │ │ │ │ - //make sure we have an arg parser attached │ │ │ │ │ - for (var i = 0, len = this.map.controls.length; i < len; i++) { │ │ │ │ │ - var control = this.map.controls[i]; │ │ │ │ │ - if (control.CLASS_NAME == this.argParserClass.CLASS_NAME) { │ │ │ │ │ + // to minimize the control we set its div's width │ │ │ │ │ + // and height to 0px, we cannot just set "display" │ │ │ │ │ + // to "none" because it would hide the maximize │ │ │ │ │ + // div │ │ │ │ │ + this.div.style.width = "0px"; │ │ │ │ │ + this.div.style.height = "0px"; │ │ │ │ │ │ │ │ │ │ - // If a permalink is added to the map, and an ArgParser already │ │ │ │ │ - // exists, we override the displayProjection to be the one │ │ │ │ │ - // on the permalink. │ │ │ │ │ - if (control.displayProjection != this.displayProjection) { │ │ │ │ │ - this.displayProjection = control.displayProjection; │ │ │ │ │ - } │ │ │ │ │ + this.showControls(true); │ │ │ │ │ │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (i == this.map.controls.length) { │ │ │ │ │ - this.map.addControl(new this.argParserClass({ │ │ │ │ │ - 'displayProjection': this.displayProjection │ │ │ │ │ - })); │ │ │ │ │ + if (e != null) { │ │ │ │ │ + OpenLayers.Event.stop(e); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ + * Method: showControls │ │ │ │ │ + * Hide/Show all LayerSwitcher controls depending on whether we are │ │ │ │ │ + * minimized or not │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * minimize - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - if (!this.element && !this.anchor) { │ │ │ │ │ - this.element = document.createElement("a"); │ │ │ │ │ - this.element.innerHTML = OpenLayers.i18n("Permalink"); │ │ │ │ │ - this.element.href = ""; │ │ │ │ │ - this.div.appendChild(this.element); │ │ │ │ │ - } │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - 'moveend': this.updateLink, │ │ │ │ │ - 'changelayer': this.updateLink, │ │ │ │ │ - 'changebaselayer': this.updateLink, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + showControls: function(minimize) { │ │ │ │ │ │ │ │ │ │ - // Make it so there is at least a link even though the map may not have │ │ │ │ │ - // moved yet. │ │ │ │ │ - this.updateLink(); │ │ │ │ │ + this.maximizeDiv.style.display = minimize ? "" : "none"; │ │ │ │ │ + this.minimizeDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ │ │ │ │ │ - return this.div; │ │ │ │ │ + this.layersDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: updateLink │ │ │ │ │ + * Method: loadContents │ │ │ │ │ + * Set up the labels and divs for the control │ │ │ │ │ */ │ │ │ │ │ - updateLink: function() { │ │ │ │ │ - var separator = this.anchor ? '#' : '?'; │ │ │ │ │ - var href = this.base; │ │ │ │ │ - var anchor = null; │ │ │ │ │ - if (href.indexOf("#") != -1 && this.anchor == false) { │ │ │ │ │ - anchor = href.substring(href.indexOf("#"), href.length); │ │ │ │ │ - } │ │ │ │ │ - if (href.indexOf(separator) != -1) { │ │ │ │ │ - href = href.substring(0, href.indexOf(separator)); │ │ │ │ │ - } │ │ │ │ │ - var splits = href.split("#"); │ │ │ │ │ - href = splits[0] + separator + OpenLayers.Util.getParameterString(this.createParams()); │ │ │ │ │ - if (anchor) { │ │ │ │ │ - href += anchor; │ │ │ │ │ - } │ │ │ │ │ - if (this.anchor && !this.element) { │ │ │ │ │ - window.location.href = href; │ │ │ │ │ - } else { │ │ │ │ │ - this.element.href = href; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + loadContents: function() { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: createParams │ │ │ │ │ - * Creates the parameters that need to be encoded into the permalink url. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * center - {<OpenLayers.LonLat>} center to encode in the permalink. │ │ │ │ │ - * Defaults to the current map center. │ │ │ │ │ - * zoom - {Integer} zoom level to encode in the permalink. Defaults to the │ │ │ │ │ - * current map zoom level. │ │ │ │ │ - * layers - {Array(<OpenLayers.Layer>)} layers to encode in the permalink. │ │ │ │ │ - * Defaults to the current map layers. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Hash of parameters that will be url-encoded into the │ │ │ │ │ - * permalink. │ │ │ │ │ - */ │ │ │ │ │ - createParams: function(center, zoom, layers) { │ │ │ │ │ - center = center || this.map.getCenter(); │ │ │ │ │ + // layers list div │ │ │ │ │ + this.layersDiv = document.createElement("div"); │ │ │ │ │ + this.layersDiv.id = this.id + "_layersDiv"; │ │ │ │ │ + OpenLayers.Element.addClass(this.layersDiv, "layersDiv"); │ │ │ │ │ │ │ │ │ │ - var params = OpenLayers.Util.getParameters(this.base); │ │ │ │ │ + this.baseLbl = document.createElement("div"); │ │ │ │ │ + this.baseLbl.innerHTML = OpenLayers.i18n("Base Layer"); │ │ │ │ │ + OpenLayers.Element.addClass(this.baseLbl, "baseLbl"); │ │ │ │ │ │ │ │ │ │ - // If there's still no center, map is not initialized yet. │ │ │ │ │ - // Break out of this function, and simply return the params from the │ │ │ │ │ - // base link. │ │ │ │ │ - if (center) { │ │ │ │ │ + this.baseLayersDiv = document.createElement("div"); │ │ │ │ │ + OpenLayers.Element.addClass(this.baseLayersDiv, "baseLayersDiv"); │ │ │ │ │ │ │ │ │ │ - //zoom │ │ │ │ │ - params.zoom = zoom || this.map.getZoom(); │ │ │ │ │ + this.dataLbl = document.createElement("div"); │ │ │ │ │ + this.dataLbl.innerHTML = OpenLayers.i18n("Overlays"); │ │ │ │ │ + OpenLayers.Element.addClass(this.dataLbl, "dataLbl"); │ │ │ │ │ │ │ │ │ │ - //lon,lat │ │ │ │ │ - var lat = center.lat; │ │ │ │ │ - var lon = center.lon; │ │ │ │ │ + this.dataLayersDiv = document.createElement("div"); │ │ │ │ │ + OpenLayers.Element.addClass(this.dataLayersDiv, "dataLayersDiv"); │ │ │ │ │ │ │ │ │ │ - if (this.displayProjection) { │ │ │ │ │ - var mapPosition = OpenLayers.Projection.transform({ │ │ │ │ │ - x: lon, │ │ │ │ │ - y: lat │ │ │ │ │ - }, │ │ │ │ │ - this.map.getProjectionObject(), │ │ │ │ │ - this.displayProjection); │ │ │ │ │ - lon = mapPosition.x; │ │ │ │ │ - lat = mapPosition.y; │ │ │ │ │ - } │ │ │ │ │ - params.lat = Math.round(lat * 100000) / 100000; │ │ │ │ │ - params.lon = Math.round(lon * 100000) / 100000; │ │ │ │ │ + if (this.ascending) { │ │ │ │ │ + this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ + } else { │ │ │ │ │ + this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - //layers │ │ │ │ │ - layers = layers || this.map.layers; │ │ │ │ │ - params.layers = ''; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - var layer = layers[i]; │ │ │ │ │ + this.div.appendChild(this.layersDiv); │ │ │ │ │ │ │ │ │ │ - if (layer.isBaseLayer) { │ │ │ │ │ - params.layers += (layer == this.map.baseLayer) ? "B" : "0"; │ │ │ │ │ - } else { │ │ │ │ │ - params.layers += (layer.getVisibility()) ? "T" : "F"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + // maximize button div │ │ │ │ │ + var img = OpenLayers.Util.getImageLocation('layer-switcher-maximize.png'); │ │ │ │ │ + this.maximizeDiv = OpenLayers.Util.createAlphaImageDiv( │ │ │ │ │ + "OpenLayers_Control_MaximizeDiv", │ │ │ │ │ + null, │ │ │ │ │ + null, │ │ │ │ │ + img, │ │ │ │ │ + "absolute"); │ │ │ │ │ + OpenLayers.Element.addClass(this.maximizeDiv, "maximizeDiv olButton"); │ │ │ │ │ + this.maximizeDiv.style.display = "none"; │ │ │ │ │ │ │ │ │ │ - return params; │ │ │ │ │ + this.div.appendChild(this.maximizeDiv); │ │ │ │ │ + │ │ │ │ │ + // minimize button div │ │ │ │ │ + var img = OpenLayers.Util.getImageLocation('layer-switcher-minimize.png'); │ │ │ │ │ + this.minimizeDiv = OpenLayers.Util.createAlphaImageDiv( │ │ │ │ │ + "OpenLayers_Control_MinimizeDiv", │ │ │ │ │ + null, │ │ │ │ │ + null, │ │ │ │ │ + img, │ │ │ │ │ + "absolute"); │ │ │ │ │ + OpenLayers.Element.addClass(this.minimizeDiv, "minimizeDiv olButton"); │ │ │ │ │ + this.minimizeDiv.style.display = "none"; │ │ │ │ │ + │ │ │ │ │ + this.div.appendChild(this.minimizeDiv); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Permalink" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.LayerSwitcher" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Control/Split.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ @@ -73959,433 +70596,14 @@ │ │ │ │ │ } │ │ │ │ │ OpenLayers.Control.prototype.destroy.call(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.Split" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/PanZoomBar.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control/PanZoom.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.PanZoomBar │ │ │ │ │ - * The PanZoomBar is a visible control composed of a │ │ │ │ │ - * <OpenLayers.Control.PanPanel> and a <OpenLayers.Control.ZoomBar>. │ │ │ │ │ - * By default it is displayed in the upper left corner of the map as 4 │ │ │ │ │ - * directional arrows above a vertical slider. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control.PanZoom> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.PanZoomBar = OpenLayers.Class(OpenLayers.Control.PanZoom, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomStopWidth │ │ │ │ │ - */ │ │ │ │ │ - zoomStopWidth: 18, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomStopHeight │ │ │ │ │ - */ │ │ │ │ │ - zoomStopHeight: 11, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: slider │ │ │ │ │ - */ │ │ │ │ │ - slider: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: sliderEvents │ │ │ │ │ - * {<OpenLayers.Events>} │ │ │ │ │ - */ │ │ │ │ │ - sliderEvents: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: zoombarDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - zoombarDiv: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomWorldIcon │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - zoomWorldIcon: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: panIcons │ │ │ │ │ - * {Boolean} Set this property to false not to display the pan icons. If │ │ │ │ │ - * false the zoom world icon is placed under the zoom bar. Defaults to │ │ │ │ │ - * true. │ │ │ │ │ - */ │ │ │ │ │ - panIcons: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: forceFixedZoomLevel │ │ │ │ │ - * {Boolean} Force a fixed zoom level even though the map has │ │ │ │ │ - * fractionalZoom │ │ │ │ │ - */ │ │ │ │ │ - forceFixedZoomLevel: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: mouseDragStart │ │ │ │ │ - * {<OpenLayers.Pixel>} │ │ │ │ │ - */ │ │ │ │ │ - mouseDragStart: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: deltaY │ │ │ │ │ - * {Number} The cumulative vertical pixel offset during a zoom bar drag. │ │ │ │ │ - */ │ │ │ │ │ - deltaY: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: zoomStart │ │ │ │ │ - * {<OpenLayers.Pixel>} │ │ │ │ │ - */ │ │ │ │ │ - zoomStart: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.PanZoomBar │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - │ │ │ │ │ - this._removeZoomBar(); │ │ │ │ │ - │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - "changebaselayer": this.redraw, │ │ │ │ │ - "updatesize": this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Control.PanZoom.prototype.destroy.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - delete this.mouseDragStart; │ │ │ │ │ - delete this.zoomStart; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.PanZoom.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - "changebaselayer": this.redraw, │ │ │ │ │ - "updatesize": this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: redraw │ │ │ │ │ - * clear the div and start over. │ │ │ │ │ - */ │ │ │ │ │ - redraw: function() { │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.removeButtons(); │ │ │ │ │ - this._removeZoomBar(); │ │ │ │ │ - } │ │ │ │ │ - this.draw(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - */ │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - // initialize our internal div │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - px = this.position.clone(); │ │ │ │ │ - │ │ │ │ │ - // place the controls │ │ │ │ │ - this.buttons = []; │ │ │ │ │ - │ │ │ │ │ - var sz = { │ │ │ │ │ - w: 18, │ │ │ │ │ - h: 18 │ │ │ │ │ - }; │ │ │ │ │ - if (this.panIcons) { │ │ │ │ │ - var centered = new OpenLayers.Pixel(px.x + sz.w / 2, px.y); │ │ │ │ │ - var wposition = sz.w; │ │ │ │ │ - │ │ │ │ │ - if (this.zoomWorldIcon) { │ │ │ │ │ - centered = new OpenLayers.Pixel(px.x + sz.w, px.y); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this._addButton("panup", "north-mini.png", centered, sz); │ │ │ │ │ - px.y = centered.y + sz.h; │ │ │ │ │ - this._addButton("panleft", "west-mini.png", px, sz); │ │ │ │ │ - if (this.zoomWorldIcon) { │ │ │ │ │ - this._addButton("zoomworld", "zoom-world-mini.png", px.add(sz.w, 0), sz); │ │ │ │ │ - │ │ │ │ │ - wposition *= 2; │ │ │ │ │ - } │ │ │ │ │ - this._addButton("panright", "east-mini.png", px.add(wposition, 0), sz); │ │ │ │ │ - this._addButton("pandown", "south-mini.png", centered.add(0, sz.h * 2), sz); │ │ │ │ │ - this._addButton("zoomin", "zoom-plus-mini.png", centered.add(0, sz.h * 3 + 5), sz); │ │ │ │ │ - centered = this._addZoomBar(centered.add(0, sz.h * 4 + 5)); │ │ │ │ │ - this._addButton("zoomout", "zoom-minus-mini.png", centered, sz); │ │ │ │ │ - } else { │ │ │ │ │ - this._addButton("zoomin", "zoom-plus-mini.png", px, sz); │ │ │ │ │ - centered = this._addZoomBar(px.add(0, sz.h)); │ │ │ │ │ - this._addButton("zoomout", "zoom-minus-mini.png", centered, sz); │ │ │ │ │ - if (this.zoomWorldIcon) { │ │ │ │ │ - centered = centered.add(0, sz.h + 3); │ │ │ │ │ - this._addButton("zoomworld", "zoom-world-mini.png", centered, sz); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: _addZoomBar │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * centered - {<OpenLayers.Pixel>} where zoombar drawing is to start. │ │ │ │ │ - */ │ │ │ │ │ - _addZoomBar: function(centered) { │ │ │ │ │ - var imgLocation = OpenLayers.Util.getImageLocation("slider.png"); │ │ │ │ │ - var id = this.id + "_" + this.map.id; │ │ │ │ │ - var minZoom = this.map.getMinZoom(); │ │ │ │ │ - var zoomsToEnd = this.map.getNumZoomLevels() - 1 - this.map.getZoom(); │ │ │ │ │ - var slider = OpenLayers.Util.createAlphaImageDiv(id, │ │ │ │ │ - centered.add(-1, zoomsToEnd * this.zoomStopHeight), { │ │ │ │ │ - w: 20, │ │ │ │ │ - h: 9 │ │ │ │ │ - }, │ │ │ │ │ - imgLocation, │ │ │ │ │ - "absolute"); │ │ │ │ │ - slider.style.cursor = "move"; │ │ │ │ │ - this.slider = slider; │ │ │ │ │ - │ │ │ │ │ - this.sliderEvents = new OpenLayers.Events(this, slider, null, true, { │ │ │ │ │ - includeXY: true │ │ │ │ │ - }); │ │ │ │ │ - this.sliderEvents.on({ │ │ │ │ │ - "touchstart": this.zoomBarDown, │ │ │ │ │ - "touchmove": this.zoomBarDrag, │ │ │ │ │ - "touchend": this.zoomBarUp, │ │ │ │ │ - "mousedown": this.zoomBarDown, │ │ │ │ │ - "mousemove": this.zoomBarDrag, │ │ │ │ │ - "mouseup": this.zoomBarUp │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - var sz = { │ │ │ │ │ - w: this.zoomStopWidth, │ │ │ │ │ - h: this.zoomStopHeight * (this.map.getNumZoomLevels() - minZoom) │ │ │ │ │ - }; │ │ │ │ │ - var imgLocation = OpenLayers.Util.getImageLocation("zoombar.png"); │ │ │ │ │ - var div = null; │ │ │ │ │ - │ │ │ │ │ - if (OpenLayers.Util.alphaHack()) { │ │ │ │ │ - var id = this.id + "_" + this.map.id; │ │ │ │ │ - div = OpenLayers.Util.createAlphaImageDiv(id, centered, { │ │ │ │ │ - w: sz.w, │ │ │ │ │ - h: this.zoomStopHeight │ │ │ │ │ - }, │ │ │ │ │ - imgLocation, │ │ │ │ │ - "absolute", null, "crop"); │ │ │ │ │ - div.style.height = sz.h + "px"; │ │ │ │ │ - } else { │ │ │ │ │ - div = OpenLayers.Util.createDiv( │ │ │ │ │ - 'OpenLayers_Control_PanZoomBar_Zoombar' + this.map.id, │ │ │ │ │ - centered, │ │ │ │ │ - sz, │ │ │ │ │ - imgLocation); │ │ │ │ │ - } │ │ │ │ │ - div.style.cursor = "pointer"; │ │ │ │ │ - div.className = "olButton"; │ │ │ │ │ - this.zoombarDiv = div; │ │ │ │ │ - │ │ │ │ │ - this.div.appendChild(div); │ │ │ │ │ - │ │ │ │ │ - this.startTop = parseInt(div.style.top); │ │ │ │ │ - this.div.appendChild(slider); │ │ │ │ │ - │ │ │ │ │ - this.map.events.register("zoomend", this, this.moveZoomBar); │ │ │ │ │ - │ │ │ │ │ - centered = centered.add(0, │ │ │ │ │ - this.zoomStopHeight * (this.map.getNumZoomLevels() - minZoom)); │ │ │ │ │ - return centered; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: _removeZoomBar │ │ │ │ │ - */ │ │ │ │ │ - _removeZoomBar: function() { │ │ │ │ │ - this.sliderEvents.un({ │ │ │ │ │ - "touchstart": this.zoomBarDown, │ │ │ │ │ - "touchmove": this.zoomBarDrag, │ │ │ │ │ - "touchend": this.zoomBarUp, │ │ │ │ │ - "mousedown": this.zoomBarDown, │ │ │ │ │ - "mousemove": this.zoomBarDrag, │ │ │ │ │ - "mouseup": this.zoomBarUp │ │ │ │ │ - }); │ │ │ │ │ - this.sliderEvents.destroy(); │ │ │ │ │ - │ │ │ │ │ - this.div.removeChild(this.zoombarDiv); │ │ │ │ │ - this.zoombarDiv = null; │ │ │ │ │ - this.div.removeChild(this.slider); │ │ │ │ │ - this.slider = null; │ │ │ │ │ - │ │ │ │ │ - this.map.events.unregister("zoomend", this, this.moveZoomBar); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onButtonClick │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - OpenLayers.Control.PanZoom.prototype.onButtonClick.apply(this, arguments); │ │ │ │ │ - if (evt.buttonElement === this.zoombarDiv) { │ │ │ │ │ - var levels = evt.buttonXY.y / this.zoomStopHeight; │ │ │ │ │ - if (this.forceFixedZoomLevel || !this.map.fractionalZoom) { │ │ │ │ │ - levels = Math.floor(levels); │ │ │ │ │ - } │ │ │ │ │ - var zoom = (this.map.getNumZoomLevels() - 1) - levels; │ │ │ │ │ - zoom = Math.min(Math.max(zoom, 0), this.map.getNumZoomLevels() - 1); │ │ │ │ │ - this.map.zoomTo(zoom); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: passEventToSlider │ │ │ │ │ - * This function is used to pass events that happen on the div, or the map, │ │ │ │ │ - * through to the slider, which then does its moving thing. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - */ │ │ │ │ │ - passEventToSlider: function(evt) { │ │ │ │ │ - this.sliderEvents.handleBrowserEvent(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /* │ │ │ │ │ - * Method: zoomBarDown │ │ │ │ │ - * event listener for clicks on the slider │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - */ │ │ │ │ │ - zoomBarDown: function(evt) { │ │ │ │ │ - if (!OpenLayers.Event.isLeftClick(evt) && !OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - "touchmove": this.passEventToSlider, │ │ │ │ │ - "mousemove": this.passEventToSlider, │ │ │ │ │ - "mouseup": this.passEventToSlider, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.mouseDragStart = evt.xy.clone(); │ │ │ │ │ - this.zoomStart = evt.xy.clone(); │ │ │ │ │ - this.div.style.cursor = "move"; │ │ │ │ │ - // reset the div offsets just in case the div moved │ │ │ │ │ - this.zoombarDiv.offsets = null; │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /* │ │ │ │ │ - * Method: zoomBarDrag │ │ │ │ │ - * This is what happens when a click has occurred, and the client is │ │ │ │ │ - * dragging. Here we must ensure that the slider doesn't go beyond the │ │ │ │ │ - * bottom/top of the zoombar div, as well as moving the slider to its new │ │ │ │ │ - * visual location │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - */ │ │ │ │ │ - zoomBarDrag: function(evt) { │ │ │ │ │ - if (this.mouseDragStart != null) { │ │ │ │ │ - var deltaY = this.mouseDragStart.y - evt.xy.y; │ │ │ │ │ - var offsets = OpenLayers.Util.pagePosition(this.zoombarDiv); │ │ │ │ │ - if ((evt.clientY - offsets[1]) > 0 && │ │ │ │ │ - (evt.clientY - offsets[1]) < parseInt(this.zoombarDiv.style.height) - 2) { │ │ │ │ │ - var newTop = parseInt(this.slider.style.top) - deltaY; │ │ │ │ │ - this.slider.style.top = newTop + "px"; │ │ │ │ │ - this.mouseDragStart = evt.xy.clone(); │ │ │ │ │ - } │ │ │ │ │ - // set cumulative displacement │ │ │ │ │ - this.deltaY = this.zoomStart.y - evt.xy.y; │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /* │ │ │ │ │ - * Method: zoomBarUp │ │ │ │ │ - * Perform cleanup when a mouseup event is received -- discover new zoom │ │ │ │ │ - * level and switch to it. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - */ │ │ │ │ │ - zoomBarUp: function(evt) { │ │ │ │ │ - if (!OpenLayers.Event.isLeftClick(evt) && evt.type !== "touchend") { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (this.mouseDragStart) { │ │ │ │ │ - this.div.style.cursor = ""; │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - "touchmove": this.passEventToSlider, │ │ │ │ │ - "mouseup": this.passEventToSlider, │ │ │ │ │ - "mousemove": this.passEventToSlider, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - var zoomLevel = this.map.zoom; │ │ │ │ │ - if (!this.forceFixedZoomLevel && this.map.fractionalZoom) { │ │ │ │ │ - zoomLevel += this.deltaY / this.zoomStopHeight; │ │ │ │ │ - zoomLevel = Math.min(Math.max(zoomLevel, 0), │ │ │ │ │ - this.map.getNumZoomLevels() - 1); │ │ │ │ │ - } else { │ │ │ │ │ - zoomLevel += this.deltaY / this.zoomStopHeight; │ │ │ │ │ - zoomLevel = Math.max(Math.round(zoomLevel), 0); │ │ │ │ │ - } │ │ │ │ │ - this.map.zoomTo(zoomLevel); │ │ │ │ │ - this.mouseDragStart = null; │ │ │ │ │ - this.zoomStart = null; │ │ │ │ │ - this.deltaY = 0; │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /* │ │ │ │ │ - * Method: moveZoomBar │ │ │ │ │ - * Change the location of the slider to match the current zoom level. │ │ │ │ │ - */ │ │ │ │ │ - moveZoomBar: function() { │ │ │ │ │ - var newTop = │ │ │ │ │ - ((this.map.getNumZoomLevels() - 1) - this.map.getZoom()) * │ │ │ │ │ - this.zoomStopHeight + this.startTop + 1; │ │ │ │ │ - this.slider.style.top = newTop + "px"; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.PanZoomBar" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ OpenLayers/Control/DragFeature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -74749,238 +70967,14 @@ │ │ │ │ │ this.handlers.feature.setMap(map); │ │ │ │ │ OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.DragFeature" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/ScaleLine.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.ScaleLine │ │ │ │ │ - * The ScaleLine displays a small line indicator representing the current │ │ │ │ │ - * map scale on the map. By default it is drawn in the lower left corner of │ │ │ │ │ - * the map. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - * │ │ │ │ │ - * Is a very close copy of: │ │ │ │ │ - * - <OpenLayers.Control.Scale> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ScaleLine = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: maxWidth │ │ │ │ │ - * {Integer} Maximum width of the scale line in pixels. Default is 100. │ │ │ │ │ - */ │ │ │ │ │ - maxWidth: 100, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: topOutUnits │ │ │ │ │ - * {String} Units for zoomed out on top bar. Default is km. │ │ │ │ │ - */ │ │ │ │ │ - topOutUnits: "km", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: topInUnits │ │ │ │ │ - * {String} Units for zoomed in on top bar. Default is m. │ │ │ │ │ - */ │ │ │ │ │ - topInUnits: "m", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: bottomOutUnits │ │ │ │ │ - * {String} Units for zoomed out on bottom bar. Default is mi. │ │ │ │ │ - */ │ │ │ │ │ - bottomOutUnits: "mi", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: bottomInUnits │ │ │ │ │ - * {String} Units for zoomed in on bottom bar. Default is ft. │ │ │ │ │ - */ │ │ │ │ │ - bottomInUnits: "ft", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: eTop │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - eTop: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: eBottom │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - eBottom: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geodesic │ │ │ │ │ - * {Boolean} Use geodesic measurement. Default is false. The recommended │ │ │ │ │ - * setting for maps in EPSG:4326 is false, and true EPSG:900913. If set to │ │ │ │ │ - * true, the scale will be calculated based on the horizontal size of the │ │ │ │ │ - * pixel in the center of the map viewport. │ │ │ │ │ - */ │ │ │ │ │ - geodesic: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.ScaleLine │ │ │ │ │ - * Create a new scale line control. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be used │ │ │ │ │ - * to extend the control. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (!this.eTop) { │ │ │ │ │ - // stick in the top bar │ │ │ │ │ - this.eTop = document.createElement("div"); │ │ │ │ │ - this.eTop.className = this.displayClass + "Top"; │ │ │ │ │ - var theLen = this.topInUnits.length; │ │ │ │ │ - this.div.appendChild(this.eTop); │ │ │ │ │ - if ((this.topOutUnits == "") || (this.topInUnits == "")) { │ │ │ │ │ - this.eTop.style.visibility = "hidden"; │ │ │ │ │ - } else { │ │ │ │ │ - this.eTop.style.visibility = "visible"; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // and the bottom bar │ │ │ │ │ - this.eBottom = document.createElement("div"); │ │ │ │ │ - this.eBottom.className = this.displayClass + "Bottom"; │ │ │ │ │ - this.div.appendChild(this.eBottom); │ │ │ │ │ - if ((this.bottomOutUnits == "") || (this.bottomInUnits == "")) { │ │ │ │ │ - this.eBottom.style.visibility = "hidden"; │ │ │ │ │ - } else { │ │ │ │ │ - this.eBottom.style.visibility = "visible"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.map.events.register('moveend', this, this.update); │ │ │ │ │ - this.update(); │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getBarLen │ │ │ │ │ - * Given a number, round it down to the nearest 1,2,5 times a power of 10. │ │ │ │ │ - * That seems a fairly useful set of number groups to use. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * maxLen - {float} the number we're rounding down from │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} the rounded number (less than or equal to maxLen) │ │ │ │ │ - */ │ │ │ │ │ - getBarLen: function(maxLen) { │ │ │ │ │ - // nearest power of 10 lower than maxLen │ │ │ │ │ - var digits = parseInt(Math.log(maxLen) / Math.log(10)); │ │ │ │ │ - var pow10 = Math.pow(10, digits); │ │ │ │ │ - │ │ │ │ │ - // ok, find first character │ │ │ │ │ - var firstChar = parseInt(maxLen / pow10); │ │ │ │ │ - │ │ │ │ │ - // right, put it into the correct bracket │ │ │ │ │ - var barLen; │ │ │ │ │ - if (firstChar > 5) { │ │ │ │ │ - barLen = 5; │ │ │ │ │ - } else if (firstChar > 2) { │ │ │ │ │ - barLen = 2; │ │ │ │ │ - } else { │ │ │ │ │ - barLen = 1; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // scale it up the correct power of 10 │ │ │ │ │ - return barLen * pow10; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: update │ │ │ │ │ - * Update the size of the bars, and the labels they contain. │ │ │ │ │ - */ │ │ │ │ │ - update: function() { │ │ │ │ │ - var res = this.map.getResolution(); │ │ │ │ │ - if (!res) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var curMapUnits = this.map.getUnits(); │ │ │ │ │ - var inches = OpenLayers.INCHES_PER_UNIT; │ │ │ │ │ - │ │ │ │ │ - // convert maxWidth to map units │ │ │ │ │ - var maxSizeData = this.maxWidth * res * inches[curMapUnits]; │ │ │ │ │ - var geodesicRatio = 1; │ │ │ │ │ - if (this.geodesic === true) { │ │ │ │ │ - var maxSizeGeodesic = (this.map.getGeodesicPixelSize().w || │ │ │ │ │ - 0.000001) * this.maxWidth; │ │ │ │ │ - var maxSizeKilometers = maxSizeData / inches["km"]; │ │ │ │ │ - geodesicRatio = maxSizeGeodesic / maxSizeKilometers; │ │ │ │ │ - maxSizeData *= geodesicRatio; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // decide whether to use large or small scale units │ │ │ │ │ - var topUnits; │ │ │ │ │ - var bottomUnits; │ │ │ │ │ - if (maxSizeData > 100000) { │ │ │ │ │ - topUnits = this.topOutUnits; │ │ │ │ │ - bottomUnits = this.bottomOutUnits; │ │ │ │ │ - } else { │ │ │ │ │ - topUnits = this.topInUnits; │ │ │ │ │ - bottomUnits = this.bottomInUnits; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // and to map units units │ │ │ │ │ - var topMax = maxSizeData / inches[topUnits]; │ │ │ │ │ - var bottomMax = maxSizeData / inches[bottomUnits]; │ │ │ │ │ - │ │ │ │ │ - // now trim this down to useful block length │ │ │ │ │ - var topRounded = this.getBarLen(topMax); │ │ │ │ │ - var bottomRounded = this.getBarLen(bottomMax); │ │ │ │ │ - │ │ │ │ │ - // and back to display units │ │ │ │ │ - topMax = topRounded / inches[curMapUnits] * inches[topUnits]; │ │ │ │ │ - bottomMax = bottomRounded / inches[curMapUnits] * inches[bottomUnits]; │ │ │ │ │ - │ │ │ │ │ - // and to pixel units │ │ │ │ │ - var topPx = topMax / res / geodesicRatio; │ │ │ │ │ - var bottomPx = bottomMax / res / geodesicRatio; │ │ │ │ │ - │ │ │ │ │ - // now set the pixel widths │ │ │ │ │ - // and the values inside them │ │ │ │ │ - │ │ │ │ │ - if (this.eBottom.style.visibility == "visible") { │ │ │ │ │ - this.eBottom.style.width = Math.round(bottomPx) + "px"; │ │ │ │ │ - this.eBottom.innerHTML = bottomRounded + " " + bottomUnits; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.eTop.style.visibility == "visible") { │ │ │ │ │ - this.eTop.style.width = Math.round(topPx) + "px"; │ │ │ │ │ - this.eTop.innerHTML = topRounded + " " + topUnits; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ScaleLine" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ OpenLayers/Control/TransformFeature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -75615,781 +71609,1433 @@ │ │ │ │ │ this.dragControl = null; │ │ │ │ │ OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.TransformFeature" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/KeyboardDefaults.js │ │ │ │ │ + OpenLayers/Control/GetFeature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Handler/Keyboard.js │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ + * @requires OpenLayers/Handler/Click.js │ │ │ │ │ + * @requires OpenLayers/Handler/Box.js │ │ │ │ │ + * @requires OpenLayers/Handler/Hover.js │ │ │ │ │ + * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.KeyboardDefaults │ │ │ │ │ - * The KeyboardDefaults control adds panning and zooming functions, controlled │ │ │ │ │ - * with the keyboard. By default arrow keys pan, +/- keys zoom & Page Up/Page │ │ │ │ │ - * Down/Home/End scroll by three quarters of a page. │ │ │ │ │ - * │ │ │ │ │ - * This control has no visible appearance. │ │ │ │ │ + * Class: OpenLayers.Control.GetFeature │ │ │ │ │ + * Gets vector features for locations underneath the mouse cursor. Can be │ │ │ │ │ + * configured to act on click, hover or dragged boxes. Uses an │ │ │ │ │ + * <OpenLayers.Protocol> that supports spatial filters to retrieve │ │ │ │ │ + * features from a server and fires events that notify applications of the │ │ │ │ │ + * selected features. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.KeyboardDefaults = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Control.GetFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ + * APIProperty: protocol │ │ │ │ │ + * {<OpenLayers.Protocol>} Required. The protocol used for fetching │ │ │ │ │ + * features. │ │ │ │ │ */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ + protocol: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: slideFactor │ │ │ │ │ - * Pixels to slide by. │ │ │ │ │ + * APIProperty: multipleKey │ │ │ │ │ + * {String} An event modifier ('altKey' or 'shiftKey') that temporarily sets │ │ │ │ │ + * the <multiple> property to true. Default is null. │ │ │ │ │ */ │ │ │ │ │ - slideFactor: 75, │ │ │ │ │ + multipleKey: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: observeElement │ │ │ │ │ - * {DOMelement|String} The DOM element to handle keys for. You │ │ │ │ │ - * can use the map div here, to have the navigation keys │ │ │ │ │ - * work when the map div has the focus. If undefined the │ │ │ │ │ - * document is used. │ │ │ │ │ + * APIProperty: toggleKey │ │ │ │ │ + * {String} An event modifier ('altKey' or 'shiftKey') that temporarily sets │ │ │ │ │ + * the <toggle> property to true. Default is null. │ │ │ │ │ */ │ │ │ │ │ - observeElement: null, │ │ │ │ │ + toggleKey: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.KeyboardDefaults │ │ │ │ │ + * Property: modifiers │ │ │ │ │ + * {Object} The event modifiers to use, according to the current event │ │ │ │ │ + * being handled by this control's handlers │ │ │ │ │ */ │ │ │ │ │ + modifiers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Create handler. │ │ │ │ │ + * APIProperty: multiple │ │ │ │ │ + * {Boolean} Allow selection of multiple geometries. Default is false. │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - var observeElement = this.observeElement || document; │ │ │ │ │ - this.handler = new OpenLayers.Handler.Keyboard(this, { │ │ │ │ │ - "keydown": this.defaultKeyPress │ │ │ │ │ - }, { │ │ │ │ │ - observeElement: observeElement │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + multiple: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: defaultKeyPress │ │ │ │ │ - * When handling the key event, we only use evt.keyCode. This holds │ │ │ │ │ - * some drawbacks, though we get around them below. When interpretting │ │ │ │ │ - * the keycodes below (including the comments associated with them), │ │ │ │ │ - * consult the URL below. For instance, the Safari browser returns │ │ │ │ │ - * "IE keycodes", and so is supported by any keycode labeled "IE". │ │ │ │ │ - * │ │ │ │ │ - * Very informative URL: │ │ │ │ │ - * http://unixpapa.com/js/key.html │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * APIProperty: click │ │ │ │ │ + * {Boolean} Use a click handler for selecting/unselecting features. If │ │ │ │ │ + * both <click> and <box> are set to true, the click handler takes │ │ │ │ │ + * precedence over the box handler if a box with zero extent was │ │ │ │ │ + * selected. Default is true. │ │ │ │ │ */ │ │ │ │ │ - defaultKeyPress: function(evt) { │ │ │ │ │ - var size, handled = true; │ │ │ │ │ + click: true, │ │ │ │ │ │ │ │ │ │ - var target = OpenLayers.Event.element(evt); │ │ │ │ │ - if (target && │ │ │ │ │ - (target.tagName == 'INPUT' || │ │ │ │ │ - target.tagName == 'TEXTAREA' || │ │ │ │ │ - target.tagName == 'SELECT')) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: single │ │ │ │ │ + * {Boolean} Tells whether select by click should select a single │ │ │ │ │ + * feature. If set to false, all matching features are selected. │ │ │ │ │ + * If set to true, only the best matching feature is selected. │ │ │ │ │ + * This option has an effect only of the <click> option is set │ │ │ │ │ + * to true. Default is true. │ │ │ │ │ + */ │ │ │ │ │ + single: true, │ │ │ │ │ │ │ │ │ │ - switch (evt.keyCode) { │ │ │ │ │ - case OpenLayers.Event.KEY_LEFT: │ │ │ │ │ - this.map.pan(-this.slideFactor, 0); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Event.KEY_RIGHT: │ │ │ │ │ - this.map.pan(this.slideFactor, 0); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Event.KEY_UP: │ │ │ │ │ - this.map.pan(0, -this.slideFactor); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Event.KEY_DOWN: │ │ │ │ │ - this.map.pan(0, this.slideFactor); │ │ │ │ │ - break; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: clickout │ │ │ │ │ + * {Boolean} Unselect features when clicking outside any feature. │ │ │ │ │ + * Applies only if <click> is true. Default is true. │ │ │ │ │ + */ │ │ │ │ │ + clickout: true, │ │ │ │ │ │ │ │ │ │ - case 33: // Page Up. Same in all browsers. │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - this.map.pan(0, -0.75 * size.h); │ │ │ │ │ - break; │ │ │ │ │ - case 34: // Page Down. Same in all browsers. │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - this.map.pan(0, 0.75 * size.h); │ │ │ │ │ - break; │ │ │ │ │ - case 35: // End. Same in all browsers. │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - this.map.pan(0.75 * size.w, 0); │ │ │ │ │ - break; │ │ │ │ │ - case 36: // Home. Same in all browsers. │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - this.map.pan(-0.75 * size.w, 0); │ │ │ │ │ - break; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: toggle │ │ │ │ │ + * {Boolean} Unselect a selected feature on click. Applies only if │ │ │ │ │ + * <click> is true. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + toggle: false, │ │ │ │ │ │ │ │ │ │ - case 43: // +/= (ASCII), keypad + (ASCII, Opera) │ │ │ │ │ - case 61: // +/= (Mozilla, Opera, some ASCII) │ │ │ │ │ - case 187: // +/= (IE) │ │ │ │ │ - case 107: // keypad + (IE, Mozilla) │ │ │ │ │ - this.map.zoomIn(); │ │ │ │ │ - break; │ │ │ │ │ - case 45: // -/_ (ASCII, Opera), keypad - (ASCII, Opera) │ │ │ │ │ - case 109: // -/_ (Mozilla), keypad - (Mozilla, IE) │ │ │ │ │ - case 189: // -/_ (IE) │ │ │ │ │ - case 95: // -/_ (some ASCII) │ │ │ │ │ - this.map.zoomOut(); │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - handled = false; │ │ │ │ │ - } │ │ │ │ │ - if (handled) { │ │ │ │ │ - // prevent browser default not to move the page │ │ │ │ │ - // when moving the page with the keyboard │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: clickTolerance │ │ │ │ │ + * {Integer} Tolerance for the filter query in pixels. This has the │ │ │ │ │ + * same effect as the tolerance parameter on WMS GetFeatureInfo │ │ │ │ │ + * requests. Will be ignored for box selections. Applies only if │ │ │ │ │ + * <click> or <hover> is true. Default is 5. Note that this not │ │ │ │ │ + * only affects requests on click, but also on hover. │ │ │ │ │ + */ │ │ │ │ │ + clickTolerance: 5, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.KeyboardDefaults" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/UTFGrid.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: hover │ │ │ │ │ + * {Boolean} Send feature requests on mouse moves. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + hover: false, │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: box │ │ │ │ │ + * {Boolean} Allow feature selection by drawing a box. If set to │ │ │ │ │ + * true set <click> to false to disable the click handler and │ │ │ │ │ + * rely on the box handler only, even for "zero extent" boxes. │ │ │ │ │ + * See the description of the <click> option for additional │ │ │ │ │ + * information. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + box: false, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Handler/Hover.js │ │ │ │ │ - * @requires OpenLayers/Handler/Click.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxFeatures │ │ │ │ │ + * {Integer} Maximum number of features to return from a query in single mode │ │ │ │ │ + * if supported by the <protocol>. This set of features is then used to │ │ │ │ │ + * determine the best match client-side. Default is 10. │ │ │ │ │ + */ │ │ │ │ │ + maxFeatures: 10, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.UTFGrid │ │ │ │ │ - * │ │ │ │ │ - * This Control provides behavior associated with UTFGrid Layers. │ │ │ │ │ - * These 'hit grids' provide underlying feature attributes without │ │ │ │ │ - * calling the server (again). This control allows Mousemove, Hovering │ │ │ │ │ - * and Click events to trigger callbacks that use the attributes in │ │ │ │ │ - * whatever way you need. │ │ │ │ │ - * │ │ │ │ │ - * The most common example may be a UTFGrid layer containing feature │ │ │ │ │ - * attributes that are displayed in a div as you mouseover. │ │ │ │ │ - * │ │ │ │ │ - * Example Code: │ │ │ │ │ - * │ │ │ │ │ - * (start code) │ │ │ │ │ - * var world_utfgrid = new OpenLayers.Layer.UTFGrid( │ │ │ │ │ - * 'UTFGrid Layer', │ │ │ │ │ - * "http://tiles/world_utfgrid/${z}/${x}/${y}.json" │ │ │ │ │ - * ); │ │ │ │ │ - * map.addLayer(world_utfgrid); │ │ │ │ │ - * │ │ │ │ │ - * var control = new OpenLayers.Control.UTFGrid({ │ │ │ │ │ - * layers: [world_utfgrid], │ │ │ │ │ - * handlerMode: 'move', │ │ │ │ │ - * callback: function(infoLookup) { │ │ │ │ │ - * // do something with returned data │ │ │ │ │ - * │ │ │ │ │ - * } │ │ │ │ │ - * }) │ │ │ │ │ - * (end code) │ │ │ │ │ - * │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.UTFGrid = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + /** │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Object} Hash of {<OpenLayers.Feature.Vector>}, keyed by fid, holding │ │ │ │ │ + * the currently selected features │ │ │ │ │ + */ │ │ │ │ │ + features: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ + * Proeprty: hoverFeature │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} The feature currently selected by the │ │ │ │ │ + * hover handler │ │ │ │ │ */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ + hoverFeature: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: Layers │ │ │ │ │ - * List of layers to consider. Must be Layer.UTFGrids │ │ │ │ │ - * `null` is the default indicating all UTFGrid Layers are queried. │ │ │ │ │ - * {Array} <OpenLayers.Layer.UTFGrid> │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: handlerOptions │ │ │ │ │ + * {Object} Additional options for the handlers used by this control. This │ │ │ │ │ + * is a hash with the keys "click", "box" and "hover". │ │ │ │ │ */ │ │ │ │ │ - layers: null, │ │ │ │ │ │ │ │ │ │ - /* Property: defaultHandlerOptions │ │ │ │ │ - * The default opts passed to the handler constructors │ │ │ │ │ + /** │ │ │ │ │ + * Property: handlers │ │ │ │ │ + * {Object} Object with references to multiple <OpenLayers.Handler> │ │ │ │ │ + * instances. │ │ │ │ │ */ │ │ │ │ │ - defaultHandlerOptions: { │ │ │ │ │ - 'delay': 300, │ │ │ │ │ - 'pixelTolerance': 4, │ │ │ │ │ - 'stopMove': false, │ │ │ │ │ - 'single': true, │ │ │ │ │ - 'double': false, │ │ │ │ │ - 'stopSingle': false, │ │ │ │ │ - 'stopDouble': false │ │ │ │ │ - }, │ │ │ │ │ + handlers: null, │ │ │ │ │ │ │ │ │ │ - /* APIProperty: handlerMode │ │ │ │ │ - * Defaults to 'click'. Can be 'hover' or 'move'. │ │ │ │ │ + /** │ │ │ │ │ + * Property: hoverResponse │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} The response object associated with │ │ │ │ │ + * the currently running hover request (if any). │ │ │ │ │ */ │ │ │ │ │ - handlerMode: 'click', │ │ │ │ │ + hoverResponse: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setHandler │ │ │ │ │ - * sets this.handlerMode and calls resetHandler() │ │ │ │ │ + * Property: filterType │ │ │ │ │ + * {<String>} The type of filter to use when sending off a request. │ │ │ │ │ + * Possible values: │ │ │ │ │ + * OpenLayers.Filter.Spatial.<BBOX|INTERSECTS|WITHIN|CONTAINS> │ │ │ │ │ + * Defaults to: OpenLayers.Filter.Spatial.BBOX │ │ │ │ │ + */ │ │ │ │ │ + filterType: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * hm - {String} Handler Mode string; 'click', 'hover' or 'move'. │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ + * beforefeatureselected - Triggered when <click> is true before a │ │ │ │ │ + * feature is selected. The event object has a feature property with │ │ │ │ │ + * the feature about to select │ │ │ │ │ + * featureselected - Triggered when <click> is true and a feature is │ │ │ │ │ + * selected. The event object has a feature property with the │ │ │ │ │ + * selected feature │ │ │ │ │ + * beforefeaturesselected - Triggered when <click> is true before a │ │ │ │ │ + * set of features is selected. The event object is an array of │ │ │ │ │ + * feature properties with the features about to be selected. │ │ │ │ │ + * Return false after receiving this event to discontinue processing │ │ │ │ │ + * of all featureselected events and the featuresselected event. │ │ │ │ │ + * featuresselected - Triggered when <click> is true and a set of │ │ │ │ │ + * features is selected. The event object is an array of feature │ │ │ │ │ + * properties of the selected features │ │ │ │ │ + * featureunselected - Triggered when <click> is true and a feature is │ │ │ │ │ + * unselected. The event object has a feature property with the │ │ │ │ │ + * unselected feature │ │ │ │ │ + * clickout - Triggered when when <click> is true and no feature was │ │ │ │ │ + * selected. │ │ │ │ │ + * hoverfeature - Triggered when <hover> is true and the mouse has │ │ │ │ │ + * stopped over a feature │ │ │ │ │ + * outfeature - Triggered when <hover> is true and the mouse moves │ │ │ │ │ + * moved away from a hover-selected feature │ │ │ │ │ */ │ │ │ │ │ - setHandler: function(hm) { │ │ │ │ │ - this.handlerMode = hm; │ │ │ │ │ - this.resetHandler(); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: resetHandler │ │ │ │ │ - * Deactivates the old hanlder and creates a new │ │ │ │ │ - * <OpenLayers.Handler> based on the mode specified in │ │ │ │ │ - * this.handlerMode │ │ │ │ │ + * Constructor: OpenLayers.Control.GetFeature │ │ │ │ │ + * Create a new control for fetching remote features. │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} A configuration object which at least has to contain │ │ │ │ │ + * a <protocol> property (if not, it has to be set before a request is │ │ │ │ │ + * made) │ │ │ │ │ */ │ │ │ │ │ - resetHandler: function() { │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.deactivate(); │ │ │ │ │ - this.handler.destroy(); │ │ │ │ │ - this.handler = null; │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + │ │ │ │ │ + this.features = {}; │ │ │ │ │ + │ │ │ │ │ + this.handlers = {}; │ │ │ │ │ + │ │ │ │ │ + if (this.click) { │ │ │ │ │ + this.handlers.click = new OpenLayers.Handler.Click(this, { │ │ │ │ │ + click: this.selectClick │ │ │ │ │ + }, this.handlerOptions.click || {}); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - if (this.handlerMode == 'hover') { │ │ │ │ │ - // Handle this event on hover │ │ │ │ │ - this.handler = new OpenLayers.Handler.Hover( │ │ │ │ │ + if (this.box) { │ │ │ │ │ + this.handlers.box = new OpenLayers.Handler.Box( │ │ │ │ │ this, { │ │ │ │ │ - 'pause': this.handleEvent, │ │ │ │ │ - 'move': this.reset │ │ │ │ │ + done: this.selectBox │ │ │ │ │ }, │ │ │ │ │ - this.handlerOptions │ │ │ │ │ + OpenLayers.Util.extend(this.handlerOptions.box, { │ │ │ │ │ + boxDivClassName: "olHandlerBoxSelectFeature" │ │ │ │ │ + }) │ │ │ │ │ ); │ │ │ │ │ - } else if (this.handlerMode == 'click') { │ │ │ │ │ - // Handle this event on click │ │ │ │ │ - this.handler = new OpenLayers.Handler.Click( │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + this.handlers.hover = new OpenLayers.Handler.Hover( │ │ │ │ │ this, { │ │ │ │ │ - 'click': this.handleEvent │ │ │ │ │ - }, this.handlerOptions │ │ │ │ │ - ); │ │ │ │ │ - } else if (this.handlerMode == 'move') { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Hover( │ │ │ │ │ - this, │ │ │ │ │ - // Handle this event while hovering OR moving │ │ │ │ │ - { │ │ │ │ │ - 'pause': this.handleEvent, │ │ │ │ │ - 'move': this.handleEvent │ │ │ │ │ + 'move': this.cancelHover, │ │ │ │ │ + 'pause': this.selectHover │ │ │ │ │ }, │ │ │ │ │ - this.handlerOptions │ │ │ │ │ + OpenLayers.Util.extend(this.handlerOptions.hover, { │ │ │ │ │ + 'delay': 250, │ │ │ │ │ + 'pixelTolerance': 2 │ │ │ │ │ + }) │ │ │ │ │ ); │ │ │ │ │ } │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activates the control. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The control was effectively activated. │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + for (var i in this.handlers) { │ │ │ │ │ + this.handlers[i].activate(); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return OpenLayers.Control.prototype.activate.apply( │ │ │ │ │ + this, arguments │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: <OpenLayers.Control.UTFGrid> │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivates the control. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The control was effectively deactivated. │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + for (var i in this.handlers) { │ │ │ │ │ + this.handlers[i].deactivate(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Control.prototype.deactivate.apply( │ │ │ │ │ + this, arguments │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: selectClick │ │ │ │ │ + * Called on click │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.handlerOptions = options.handlerOptions || this.defaultHandlerOptions; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.resetHandler(); │ │ │ │ │ + selectClick: function(evt) { │ │ │ │ │ + var bounds = this.pixelToBounds(evt.xy); │ │ │ │ │ + │ │ │ │ │ + this.setModifiers(evt); │ │ │ │ │ + this.request(bounds, { │ │ │ │ │ + single: this.single │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleEvent │ │ │ │ │ - * Internal method called when specified event is triggered. │ │ │ │ │ - * │ │ │ │ │ - * This method does several things: │ │ │ │ │ - * │ │ │ │ │ - * Gets the lonLat of the event. │ │ │ │ │ - * │ │ │ │ │ - * Loops through the appropriate hit grid layers and gathers the attributes. │ │ │ │ │ + * Method: selectBox │ │ │ │ │ + * Callback from the handlers.box set up when <box> selection is on │ │ │ │ │ * │ │ │ │ │ - * Passes the attributes to the callback │ │ │ │ │ + * Parameters: │ │ │ │ │ + * position - {<OpenLayers.Bounds>|Object} An OpenLayers.Bounds or │ │ │ │ │ + * an object with a 'left', 'bottom', 'right' and 'top' properties. │ │ │ │ │ + */ │ │ │ │ │ + selectBox: function(position) { │ │ │ │ │ + var bounds; │ │ │ │ │ + if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ + var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.left, │ │ │ │ │ + y: position.bottom │ │ │ │ │ + }); │ │ │ │ │ + var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.right, │ │ │ │ │ + y: position.top │ │ │ │ │ + }); │ │ │ │ │ + bounds = new OpenLayers.Bounds( │ │ │ │ │ + minXY.lon, minXY.lat, maxXY.lon, maxXY.lat │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + } else { │ │ │ │ │ + if (this.click) { │ │ │ │ │ + // box without extent - let the click handler take care of it │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + bounds = this.pixelToBounds(position); │ │ │ │ │ + } │ │ │ │ │ + this.setModifiers(this.handlers.box.dragHandler.evt); │ │ │ │ │ + this.request(bounds); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: selectHover │ │ │ │ │ + * Callback from the handlers.hover set up when <hover> selection is on │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ + * evt - {Object} event object with an xy property │ │ │ │ │ */ │ │ │ │ │ - handleEvent: function(evt) { │ │ │ │ │ - if (evt == null) { │ │ │ │ │ - this.reset(); │ │ │ │ │ - return; │ │ │ │ │ + selectHover: function(evt) { │ │ │ │ │ + var bounds = this.pixelToBounds(evt.xy); │ │ │ │ │ + this.request(bounds, { │ │ │ │ │ + single: true, │ │ │ │ │ + hover: true │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: cancelHover │ │ │ │ │ + * Callback from the handlers.hover set up when <hover> selection is on │ │ │ │ │ + */ │ │ │ │ │ + cancelHover: function() { │ │ │ │ │ + if (this.hoverResponse) { │ │ │ │ │ + this.protocol.abort(this.hoverResponse); │ │ │ │ │ + this.hoverResponse = null; │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var lonLat = this.map.getLonLatFromPixel(evt.xy); │ │ │ │ │ - if (!lonLat) { │ │ │ │ │ - return; │ │ │ │ │ + /** │ │ │ │ │ + * Method: request │ │ │ │ │ + * Sends a GetFeature request to the WFS │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} bounds for the request's BBOX filter │ │ │ │ │ + * options - {Object} additional options for this method. │ │ │ │ │ + * │ │ │ │ │ + * Supported options include: │ │ │ │ │ + * single - {Boolean} A single feature should be returned. │ │ │ │ │ + * Note that this will be ignored if the protocol does not │ │ │ │ │ + * return the geometries of the features. │ │ │ │ │ + * hover - {Boolean} Do the request for the hover handler. │ │ │ │ │ + */ │ │ │ │ │ + request: function(bounds, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: this.filterType, │ │ │ │ │ + value: bounds │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + // Set the cursor to "wait" to tell the user we're working. │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + │ │ │ │ │ + var response = this.protocol.read({ │ │ │ │ │ + maxFeatures: options.single == true ? this.maxFeatures : undefined, │ │ │ │ │ + filter: filter, │ │ │ │ │ + callback: function(result) { │ │ │ │ │ + if (result.success()) { │ │ │ │ │ + if (result.features.length) { │ │ │ │ │ + if (options.single == true) { │ │ │ │ │ + this.selectBestFeature(result.features, │ │ │ │ │ + bounds.getCenterLonLat(), options); │ │ │ │ │ + } else { │ │ │ │ │ + this.select(result.features); │ │ │ │ │ + } │ │ │ │ │ + } else if (options.hover) { │ │ │ │ │ + this.hoverSelect(); │ │ │ │ │ + } else { │ │ │ │ │ + this.events.triggerEvent("clickout"); │ │ │ │ │ + if (this.clickout) { │ │ │ │ │ + this.unselectAll(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // Reset the cursor. │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + if (options.hover == true) { │ │ │ │ │ + this.hoverResponse = response; │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var layers = this.findLayers(); │ │ │ │ │ - if (layers.length > 0) { │ │ │ │ │ - var infoLookup = {}; │ │ │ │ │ - var layer, idx; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - layer = layers[i]; │ │ │ │ │ - idx = OpenLayers.Util.indexOf(this.map.layers, layer); │ │ │ │ │ - infoLookup[idx] = layer.getFeatureInfo(lonLat); │ │ │ │ │ + /** │ │ │ │ │ + * Method: selectBestFeature │ │ │ │ │ + * Selects the feature from an array of features that is the best match │ │ │ │ │ + * for the click position. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * clickPosition - {<OpenLayers.LonLat>} │ │ │ │ │ + * options - {Object} additional options for this method │ │ │ │ │ + * │ │ │ │ │ + * Supported options include: │ │ │ │ │ + * hover - {Boolean} Do the selection for the hover handler. │ │ │ │ │ + */ │ │ │ │ │ + selectBestFeature: function(features, clickPosition, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + if (features.length) { │ │ │ │ │ + var point = new OpenLayers.Geometry.Point(clickPosition.lon, │ │ │ │ │ + clickPosition.lat); │ │ │ │ │ + var feature, resultFeature, dist; │ │ │ │ │ + var minDist = Number.MAX_VALUE; │ │ │ │ │ + for (var i = 0; i < features.length; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + dist = point.distanceTo(feature.geometry, { │ │ │ │ │ + edge: false │ │ │ │ │ + }); │ │ │ │ │ + if (dist < minDist) { │ │ │ │ │ + minDist = dist; │ │ │ │ │ + resultFeature = feature; │ │ │ │ │ + if (minDist == 0) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (options.hover == true) { │ │ │ │ │ + this.hoverSelect(resultFeature); │ │ │ │ │ + } else { │ │ │ │ │ + this.select(resultFeature || features); │ │ │ │ │ } │ │ │ │ │ - this.callback(infoLookup, lonLat, evt.xy); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: callback │ │ │ │ │ - * Function to be called when a mouse event corresponds with a location that │ │ │ │ │ - * includes data in one of the configured UTFGrid layers. │ │ │ │ │ - * │ │ │ │ │ + * Method: setModifiers │ │ │ │ │ + * Sets the multiple and toggle modifiers according to the current event │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * infoLookup - {Object} Keys of this object are layer indexes and can be │ │ │ │ │ - * used to resolve a layer in the map.layers array. The structure of │ │ │ │ │ - * the property values depend on the data included in the underlying │ │ │ │ │ - * UTFGrid and may be any valid JSON type. │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - callback: function(infoLookup) { │ │ │ │ │ - // to be provided in the constructor │ │ │ │ │ + setModifiers: function(evt) { │ │ │ │ │ + this.modifiers = { │ │ │ │ │ + multiple: this.multiple || (this.multipleKey && evt[this.multipleKey]), │ │ │ │ │ + toggle: this.toggle || (this.toggleKey && evt[this.toggleKey]) │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: reset │ │ │ │ │ - * Calls the callback with null. │ │ │ │ │ + * Method: select │ │ │ │ │ + * Add feature to the hash of selected features and trigger the │ │ │ │ │ + * featureselected and featuresselected events. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {<OpenLayers.Feature.Vector>} or an array of features │ │ │ │ │ */ │ │ │ │ │ - reset: function(evt) { │ │ │ │ │ - this.callback(null); │ │ │ │ │ + select: function(features) { │ │ │ │ │ + if (!this.modifiers.multiple && !this.modifiers.toggle) { │ │ │ │ │ + this.unselectAll(); │ │ │ │ │ + } │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var cont = this.events.triggerEvent("beforefeaturesselected", { │ │ │ │ │ + features: features │ │ │ │ │ + }); │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + var selectedFeatures = []; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (this.features[feature.fid || feature.id]) { │ │ │ │ │ + if (this.modifiers.toggle) { │ │ │ │ │ + this.unselect(this.features[feature.fid || feature.id]); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + cont = this.events.triggerEvent("beforefeatureselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + this.features[feature.fid || feature.id] = feature; │ │ │ │ │ + selectedFeatures.push(feature); │ │ │ │ │ + │ │ │ │ │ + this.events.triggerEvent("featureselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("featuresselected", { │ │ │ │ │ + features: selectedFeatures │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: findLayers │ │ │ │ │ - * Internal method to get the layers, independent of whether we are │ │ │ │ │ - * inspecting the map or using a client-provided array │ │ │ │ │ - * │ │ │ │ │ - * The default value of this.layers is null; this causes the │ │ │ │ │ - * findLayers method to return ALL UTFGrid layers encountered. │ │ │ │ │ - * │ │ │ │ │ + * Method: hoverSelect │ │ │ │ │ + * Sets/unsets the <hoverFeature> │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * None │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} the feature to hover-select. │ │ │ │ │ + * If none is provided, the current <hoverFeature> will be nulled and │ │ │ │ │ + * the outfeature event will be triggered. │ │ │ │ │ + */ │ │ │ │ │ + hoverSelect: function(feature) { │ │ │ │ │ + var fid = feature ? feature.fid || feature.id : null; │ │ │ │ │ + var hfid = this.hoverFeature ? │ │ │ │ │ + this.hoverFeature.fid || this.hoverFeature.id : null; │ │ │ │ │ + │ │ │ │ │ + if (hfid && hfid != fid) { │ │ │ │ │ + this.events.triggerEvent("outfeature", { │ │ │ │ │ + feature: this.hoverFeature │ │ │ │ │ + }); │ │ │ │ │ + this.hoverFeature = null; │ │ │ │ │ + } │ │ │ │ │ + if (fid && fid != hfid) { │ │ │ │ │ + this.events.triggerEvent("hoverfeature", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.hoverFeature = feature; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: unselect │ │ │ │ │ + * Remove feature from the hash of selected features and trigger the │ │ │ │ │ + * featureunselected event. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} Layers to handle on each event │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - findLayers: function() { │ │ │ │ │ - var candidates = this.layers || this.map.layers; │ │ │ │ │ - var layers = []; │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = candidates[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.UTFGrid) { │ │ │ │ │ - layers.push(layer); │ │ │ │ │ - } │ │ │ │ │ + unselect: function(feature) { │ │ │ │ │ + delete this.features[feature.fid || feature.id]; │ │ │ │ │ + this.events.triggerEvent("featureunselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: unselectAll │ │ │ │ │ + * Unselect all selected features. │ │ │ │ │ + */ │ │ │ │ │ + unselectAll: function() { │ │ │ │ │ + // we'll want an option to supress notification here │ │ │ │ │ + for (var fid in this.features) { │ │ │ │ │ + this.unselect(this.features[fid]); │ │ │ │ │ } │ │ │ │ │ - return layers; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.UTFGrid" │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the control. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + for (var i in this.handlers) { │ │ │ │ │ + this.handlers[i].setMap(map); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: pixelToBounds │ │ │ │ │ + * Takes a pixel as argument and creates bounds after adding the │ │ │ │ │ + * <clickTolerance>. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} │ │ │ │ │ + */ │ │ │ │ │ + pixelToBounds: function(pixel) { │ │ │ │ │ + var llPx = pixel.add(-this.clickTolerance / 2, this.clickTolerance / 2); │ │ │ │ │ + var urPx = pixel.add(this.clickTolerance / 2, -this.clickTolerance / 2); │ │ │ │ │ + var ll = this.map.getLonLatFromPixel(llPx); │ │ │ │ │ + var ur = this.map.getLonLatFromPixel(urPx); │ │ │ │ │ + return new OpenLayers.Bounds(ll.lon, ll.lat, ur.lon, ur.lat); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.GetFeature" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/MousePosition.js │ │ │ │ │ + OpenLayers/Control/PanZoomBar.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Control/PanZoom.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.MousePosition │ │ │ │ │ - * The MousePosition control displays geographic coordinates of the mouse │ │ │ │ │ - * pointer, as it is moved about the map. │ │ │ │ │ - * │ │ │ │ │ - * You can use the <prefix>- or <suffix>-properties to provide more information │ │ │ │ │ - * about the displayed coordinates to the user: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * var mousePositionCtrl = new OpenLayers.Control.MousePosition({ │ │ │ │ │ - * prefix: '<a target="_blank" ' + │ │ │ │ │ - * 'href="http://spatialreference.org/ref/epsg/4326/">' + │ │ │ │ │ - * 'EPSG:4326</a> coordinates: ' │ │ │ │ │ - * } │ │ │ │ │ - * ); │ │ │ │ │ - * (end code) │ │ │ │ │ + * Class: OpenLayers.Control.PanZoomBar │ │ │ │ │ + * The PanZoomBar is a visible control composed of a │ │ │ │ │ + * <OpenLayers.Control.PanPanel> and a <OpenLayers.Control.ZoomBar>. │ │ │ │ │ + * By default it is displayed in the upper left corner of the map as 4 │ │ │ │ │ + * directional arrows above a vertical slider. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ + * - <OpenLayers.Control.PanZoom> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.MousePosition = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Control.PanZoomBar = OpenLayers.Class(OpenLayers.Control.PanZoom, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomStopWidth │ │ │ │ │ */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ + zoomStopWidth: 18, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: element │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomStopHeight │ │ │ │ │ */ │ │ │ │ │ - element: null, │ │ │ │ │ + zoomStopHeight: 11, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: prefix │ │ │ │ │ - * {String} A string to be prepended to the current pointers coordinates │ │ │ │ │ - * when it is rendered. Defaults to the empty string ''. │ │ │ │ │ + /** │ │ │ │ │ + * Property: slider │ │ │ │ │ */ │ │ │ │ │ - prefix: '', │ │ │ │ │ + slider: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: separator │ │ │ │ │ - * {String} A string to be used to seperate the two coordinates from each │ │ │ │ │ - * other. Defaults to the string ', ', which will result in a │ │ │ │ │ - * rendered coordinate of e.g. '42.12, 21.22'. │ │ │ │ │ + /** │ │ │ │ │ + * Property: sliderEvents │ │ │ │ │ + * {<OpenLayers.Events>} │ │ │ │ │ */ │ │ │ │ │ - separator: ', ', │ │ │ │ │ + sliderEvents: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: suffix │ │ │ │ │ - * {String} A string to be appended to the current pointers coordinates │ │ │ │ │ - * when it is rendered. Defaults to the empty string ''. │ │ │ │ │ + /** │ │ │ │ │ + * Property: zoombarDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - suffix: '', │ │ │ │ │ + zoombarDiv: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: numDigits │ │ │ │ │ - * {Integer} The number of digits each coordinate shall have when being │ │ │ │ │ - * rendered, Defaults to 5. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomWorldIcon │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - numDigits: 5, │ │ │ │ │ + zoomWorldIcon: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: granularity │ │ │ │ │ - * {Integer} │ │ │ │ │ + * APIProperty: panIcons │ │ │ │ │ + * {Boolean} Set this property to false not to display the pan icons. If │ │ │ │ │ + * false the zoom world icon is placed under the zoom bar. Defaults to │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ - granularity: 10, │ │ │ │ │ + panIcons: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: emptyString │ │ │ │ │ - * {String} Set this to some value to set when the mouse is outside the │ │ │ │ │ - * map. │ │ │ │ │ + * APIProperty: forceFixedZoomLevel │ │ │ │ │ + * {Boolean} Force a fixed zoom level even though the map has │ │ │ │ │ + * fractionalZoom │ │ │ │ │ */ │ │ │ │ │ - emptyString: null, │ │ │ │ │ + forceFixedZoomLevel: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: lastXy │ │ │ │ │ + * Property: mouseDragStart │ │ │ │ │ * {<OpenLayers.Pixel>} │ │ │ │ │ */ │ │ │ │ │ - lastXy: null, │ │ │ │ │ + mouseDragStart: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: displayProjection │ │ │ │ │ - * {<OpenLayers.Projection>} The projection in which the mouse position is │ │ │ │ │ - * displayed. │ │ │ │ │ + * Property: deltaY │ │ │ │ │ + * {Number} The cumulative vertical pixel offset during a zoom bar drag. │ │ │ │ │ */ │ │ │ │ │ - displayProjection: null, │ │ │ │ │ + deltaY: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.MousePosition │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Options for control. │ │ │ │ │ + * Property: zoomStart │ │ │ │ │ + * {<OpenLayers.Pixel>} │ │ │ │ │ */ │ │ │ │ │ + zoomStart: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * Constructor: OpenLayers.Control.PanZoomBar │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ */ │ │ │ │ │ destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + this._removeZoomBar(); │ │ │ │ │ + │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + "changebaselayer": this.redraw, │ │ │ │ │ + "updatesize": this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Control.PanZoom.prototype.destroy.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + delete this.mouseDragStart; │ │ │ │ │ + delete this.zoomStart; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.map.events.register('mousemove', this, this.redraw); │ │ │ │ │ - this.map.events.register('mouseout', this, this.reset); │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.PanZoom.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + "changebaselayer": this.redraw, │ │ │ │ │ + "updatesize": this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: redraw │ │ │ │ │ + * clear the div and start over. │ │ │ │ │ + */ │ │ │ │ │ + redraw: function() { │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.removeButtons(); │ │ │ │ │ + this._removeZoomBar(); │ │ │ │ │ } │ │ │ │ │ + this.draw(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ + * Method: draw │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.map.events.unregister('mousemove', this, this.redraw); │ │ │ │ │ - this.map.events.unregister('mouseout', this, this.reset); │ │ │ │ │ - this.element.innerHTML = ""; │ │ │ │ │ - return true; │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + // initialize our internal div │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + px = this.position.clone(); │ │ │ │ │ + │ │ │ │ │ + // place the controls │ │ │ │ │ + this.buttons = []; │ │ │ │ │ + │ │ │ │ │ + var sz = { │ │ │ │ │ + w: 18, │ │ │ │ │ + h: 18 │ │ │ │ │ + }; │ │ │ │ │ + if (this.panIcons) { │ │ │ │ │ + var centered = new OpenLayers.Pixel(px.x + sz.w / 2, px.y); │ │ │ │ │ + var wposition = sz.w; │ │ │ │ │ + │ │ │ │ │ + if (this.zoomWorldIcon) { │ │ │ │ │ + centered = new OpenLayers.Pixel(px.x + sz.w, px.y); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this._addButton("panup", "north-mini.png", centered, sz); │ │ │ │ │ + px.y = centered.y + sz.h; │ │ │ │ │ + this._addButton("panleft", "west-mini.png", px, sz); │ │ │ │ │ + if (this.zoomWorldIcon) { │ │ │ │ │ + this._addButton("zoomworld", "zoom-world-mini.png", px.add(sz.w, 0), sz); │ │ │ │ │ + │ │ │ │ │ + wposition *= 2; │ │ │ │ │ + } │ │ │ │ │ + this._addButton("panright", "east-mini.png", px.add(wposition, 0), sz); │ │ │ │ │ + this._addButton("pandown", "south-mini.png", centered.add(0, sz.h * 2), sz); │ │ │ │ │ + this._addButton("zoomin", "zoom-plus-mini.png", centered.add(0, sz.h * 3 + 5), sz); │ │ │ │ │ + centered = this._addZoomBar(centered.add(0, sz.h * 4 + 5)); │ │ │ │ │ + this._addButton("zoomout", "zoom-minus-mini.png", centered, sz); │ │ │ │ │ } else { │ │ │ │ │ - return false; │ │ │ │ │ + this._addButton("zoomin", "zoom-plus-mini.png", px, sz); │ │ │ │ │ + centered = this._addZoomBar(px.add(0, sz.h)); │ │ │ │ │ + this._addButton("zoomout", "zoom-minus-mini.png", centered, sz); │ │ │ │ │ + if (this.zoomWorldIcon) { │ │ │ │ │ + centered = centered.add(0, sz.h + 3); │ │ │ │ │ + this._addButton("zoomworld", "zoom-world-mini.png", centered, sz); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + /** │ │ │ │ │ + * Method: _addZoomBar │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * centered - {<OpenLayers.Pixel>} where zoombar drawing is to start. │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + _addZoomBar: function(centered) { │ │ │ │ │ + var imgLocation = OpenLayers.Util.getImageLocation("slider.png"); │ │ │ │ │ + var id = this.id + "_" + this.map.id; │ │ │ │ │ + var minZoom = this.map.getMinZoom(); │ │ │ │ │ + var zoomsToEnd = this.map.getNumZoomLevels() - 1 - this.map.getZoom(); │ │ │ │ │ + var slider = OpenLayers.Util.createAlphaImageDiv(id, │ │ │ │ │ + centered.add(-1, zoomsToEnd * this.zoomStopHeight), { │ │ │ │ │ + w: 20, │ │ │ │ │ + h: 9 │ │ │ │ │ + }, │ │ │ │ │ + imgLocation, │ │ │ │ │ + "absolute"); │ │ │ │ │ + slider.style.cursor = "move"; │ │ │ │ │ + this.slider = slider; │ │ │ │ │ │ │ │ │ │ - if (!this.element) { │ │ │ │ │ - this.div.left = ""; │ │ │ │ │ - this.div.top = ""; │ │ │ │ │ - this.element = this.div; │ │ │ │ │ + this.sliderEvents = new OpenLayers.Events(this, slider, null, true, { │ │ │ │ │ + includeXY: true │ │ │ │ │ + }); │ │ │ │ │ + this.sliderEvents.on({ │ │ │ │ │ + "touchstart": this.zoomBarDown, │ │ │ │ │ + "touchmove": this.zoomBarDrag, │ │ │ │ │ + "touchend": this.zoomBarUp, │ │ │ │ │ + "mousedown": this.zoomBarDown, │ │ │ │ │ + "mousemove": this.zoomBarDrag, │ │ │ │ │ + "mouseup": this.zoomBarUp │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + var sz = { │ │ │ │ │ + w: this.zoomStopWidth, │ │ │ │ │ + h: this.zoomStopHeight * (this.map.getNumZoomLevels() - minZoom) │ │ │ │ │ + }; │ │ │ │ │ + var imgLocation = OpenLayers.Util.getImageLocation("zoombar.png"); │ │ │ │ │ + var div = null; │ │ │ │ │ + │ │ │ │ │ + if (OpenLayers.Util.alphaHack()) { │ │ │ │ │ + var id = this.id + "_" + this.map.id; │ │ │ │ │ + div = OpenLayers.Util.createAlphaImageDiv(id, centered, { │ │ │ │ │ + w: sz.w, │ │ │ │ │ + h: this.zoomStopHeight │ │ │ │ │ + }, │ │ │ │ │ + imgLocation, │ │ │ │ │ + "absolute", null, "crop"); │ │ │ │ │ + div.style.height = sz.h + "px"; │ │ │ │ │ + } else { │ │ │ │ │ + div = OpenLayers.Util.createDiv( │ │ │ │ │ + 'OpenLayers_Control_PanZoomBar_Zoombar' + this.map.id, │ │ │ │ │ + centered, │ │ │ │ │ + sz, │ │ │ │ │ + imgLocation); │ │ │ │ │ } │ │ │ │ │ + div.style.cursor = "pointer"; │ │ │ │ │ + div.className = "olButton"; │ │ │ │ │ + this.zoombarDiv = div; │ │ │ │ │ │ │ │ │ │ - return this.div; │ │ │ │ │ + this.div.appendChild(div); │ │ │ │ │ + │ │ │ │ │ + this.startTop = parseInt(div.style.top); │ │ │ │ │ + this.div.appendChild(slider); │ │ │ │ │ + │ │ │ │ │ + this.map.events.register("zoomend", this, this.moveZoomBar); │ │ │ │ │ + │ │ │ │ │ + centered = centered.add(0, │ │ │ │ │ + this.zoomStopHeight * (this.map.getNumZoomLevels() - minZoom)); │ │ │ │ │ + return centered; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: redraw │ │ │ │ │ + * Method: _removeZoomBar │ │ │ │ │ */ │ │ │ │ │ - redraw: function(evt) { │ │ │ │ │ + _removeZoomBar: function() { │ │ │ │ │ + this.sliderEvents.un({ │ │ │ │ │ + "touchstart": this.zoomBarDown, │ │ │ │ │ + "touchmove": this.zoomBarDrag, │ │ │ │ │ + "touchend": this.zoomBarUp, │ │ │ │ │ + "mousedown": this.zoomBarDown, │ │ │ │ │ + "mousemove": this.zoomBarDrag, │ │ │ │ │ + "mouseup": this.zoomBarUp │ │ │ │ │ + }); │ │ │ │ │ + this.sliderEvents.destroy(); │ │ │ │ │ │ │ │ │ │ - var lonLat; │ │ │ │ │ + this.div.removeChild(this.zoombarDiv); │ │ │ │ │ + this.zoombarDiv = null; │ │ │ │ │ + this.div.removeChild(this.slider); │ │ │ │ │ + this.slider = null; │ │ │ │ │ │ │ │ │ │ - if (evt == null) { │ │ │ │ │ - this.reset(); │ │ │ │ │ - return; │ │ │ │ │ - } else { │ │ │ │ │ - if (this.lastXy == null || │ │ │ │ │ - Math.abs(evt.xy.x - this.lastXy.x) > this.granularity || │ │ │ │ │ - Math.abs(evt.xy.y - this.lastXy.y) > this.granularity) { │ │ │ │ │ - this.lastXy = evt.xy; │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ + this.map.events.unregister("zoomend", this, this.moveZoomBar); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - lonLat = this.map.getLonLatFromPixel(evt.xy); │ │ │ │ │ - if (!lonLat) { │ │ │ │ │ - // map has not yet been properly initialized │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (this.displayProjection) { │ │ │ │ │ - lonLat.transform(this.map.getProjectionObject(), │ │ │ │ │ - this.displayProjection); │ │ │ │ │ + /** │ │ │ │ │ + * Method: onButtonClick │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + OpenLayers.Control.PanZoom.prototype.onButtonClick.apply(this, arguments); │ │ │ │ │ + if (evt.buttonElement === this.zoombarDiv) { │ │ │ │ │ + var levels = evt.buttonXY.y / this.zoomStopHeight; │ │ │ │ │ + if (this.forceFixedZoomLevel || !this.map.fractionalZoom) { │ │ │ │ │ + levels = Math.floor(levels); │ │ │ │ │ } │ │ │ │ │ - this.lastXy = evt.xy; │ │ │ │ │ - │ │ │ │ │ + var zoom = (this.map.getNumZoomLevels() - 1) - levels; │ │ │ │ │ + zoom = Math.min(Math.max(zoom, 0), this.map.getNumZoomLevels() - 1); │ │ │ │ │ + this.map.zoomTo(zoom); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var newHtml = this.formatOutput(lonLat); │ │ │ │ │ + /** │ │ │ │ │ + * Method: passEventToSlider │ │ │ │ │ + * This function is used to pass events that happen on the div, or the map, │ │ │ │ │ + * through to the slider, which then does its moving thing. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ + */ │ │ │ │ │ + passEventToSlider: function(evt) { │ │ │ │ │ + this.sliderEvents.handleBrowserEvent(evt); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (newHtml != this.element.innerHTML) { │ │ │ │ │ - this.element.innerHTML = newHtml; │ │ │ │ │ + /* │ │ │ │ │ + * Method: zoomBarDown │ │ │ │ │ + * event listener for clicks on the slider │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ + */ │ │ │ │ │ + zoomBarDown: function(evt) { │ │ │ │ │ + if (!OpenLayers.Event.isLeftClick(evt) && !OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + "touchmove": this.passEventToSlider, │ │ │ │ │ + "mousemove": this.passEventToSlider, │ │ │ │ │ + "mouseup": this.passEventToSlider, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.mouseDragStart = evt.xy.clone(); │ │ │ │ │ + this.zoomStart = evt.xy.clone(); │ │ │ │ │ + this.div.style.cursor = "move"; │ │ │ │ │ + // reset the div offsets just in case the div moved │ │ │ │ │ + this.zoombarDiv.offsets = null; │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: reset │ │ │ │ │ + /* │ │ │ │ │ + * Method: zoomBarDrag │ │ │ │ │ + * This is what happens when a click has occurred, and the client is │ │ │ │ │ + * dragging. Here we must ensure that the slider doesn't go beyond the │ │ │ │ │ + * bottom/top of the zoombar div, as well as moving the slider to its new │ │ │ │ │ + * visual location │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - reset: function(evt) { │ │ │ │ │ - if (this.emptyString != null) { │ │ │ │ │ - this.element.innerHTML = this.emptyString; │ │ │ │ │ + zoomBarDrag: function(evt) { │ │ │ │ │ + if (this.mouseDragStart != null) { │ │ │ │ │ + var deltaY = this.mouseDragStart.y - evt.xy.y; │ │ │ │ │ + var offsets = OpenLayers.Util.pagePosition(this.zoombarDiv); │ │ │ │ │ + if ((evt.clientY - offsets[1]) > 0 && │ │ │ │ │ + (evt.clientY - offsets[1]) < parseInt(this.zoombarDiv.style.height) - 2) { │ │ │ │ │ + var newTop = parseInt(this.slider.style.top) - deltaY; │ │ │ │ │ + this.slider.style.top = newTop + "px"; │ │ │ │ │ + this.mouseDragStart = evt.xy.clone(); │ │ │ │ │ + } │ │ │ │ │ + // set cumulative displacement │ │ │ │ │ + this.deltaY = this.zoomStart.y - evt.xy.y; │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: formatOutput │ │ │ │ │ - * Override to provide custom display output │ │ │ │ │ + /* │ │ │ │ │ + * Method: zoomBarUp │ │ │ │ │ + * Perform cleanup when a mouseup event is received -- discover new zoom │ │ │ │ │ + * level and switch to it. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * lonLat - {<OpenLayers.LonLat>} Location to display │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - formatOutput: function(lonLat) { │ │ │ │ │ - var digits = parseInt(this.numDigits); │ │ │ │ │ - var newHtml = │ │ │ │ │ - this.prefix + │ │ │ │ │ - lonLat.lon.toFixed(digits) + │ │ │ │ │ - this.separator + │ │ │ │ │ - lonLat.lat.toFixed(digits) + │ │ │ │ │ - this.suffix; │ │ │ │ │ - return newHtml; │ │ │ │ │ + zoomBarUp: function(evt) { │ │ │ │ │ + if (!OpenLayers.Event.isLeftClick(evt) && evt.type !== "touchend") { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (this.mouseDragStart) { │ │ │ │ │ + this.div.style.cursor = ""; │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + "touchmove": this.passEventToSlider, │ │ │ │ │ + "mouseup": this.passEventToSlider, │ │ │ │ │ + "mousemove": this.passEventToSlider, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + var zoomLevel = this.map.zoom; │ │ │ │ │ + if (!this.forceFixedZoomLevel && this.map.fractionalZoom) { │ │ │ │ │ + zoomLevel += this.deltaY / this.zoomStopHeight; │ │ │ │ │ + zoomLevel = Math.min(Math.max(zoomLevel, 0), │ │ │ │ │ + this.map.getNumZoomLevels() - 1); │ │ │ │ │ + } else { │ │ │ │ │ + zoomLevel += this.deltaY / this.zoomStopHeight; │ │ │ │ │ + zoomLevel = Math.max(Math.round(zoomLevel), 0); │ │ │ │ │ + } │ │ │ │ │ + this.map.zoomTo(zoomLevel); │ │ │ │ │ + this.mouseDragStart = null; │ │ │ │ │ + this.zoomStart = null; │ │ │ │ │ + this.deltaY = 0; │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.MousePosition" │ │ │ │ │ + /* │ │ │ │ │ + * Method: moveZoomBar │ │ │ │ │ + * Change the location of the slider to match the current zoom level. │ │ │ │ │ + */ │ │ │ │ │ + moveZoomBar: function() { │ │ │ │ │ + var newTop = │ │ │ │ │ + ((this.map.getNumZoomLevels() - 1) - this.map.getZoom()) * │ │ │ │ │ + this.zoomStopHeight + this.startTop + 1; │ │ │ │ │ + this.slider.style.top = newTop + "px"; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.PanZoomBar" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Zoom.js │ │ │ │ │ + OpenLayers/Control/ScaleLine.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.Zoom │ │ │ │ │ - * The Zoom control is a pair of +/- links for zooming in and out. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Control.ScaleLine │ │ │ │ │ + * The ScaleLine displays a small line indicator representing the current │ │ │ │ │ + * map scale on the map. By default it is drawn in the lower left corner of │ │ │ │ │ + * the map. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ + * │ │ │ │ │ + * Is a very close copy of: │ │ │ │ │ + * - <OpenLayers.Control.Scale> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.Zoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Control.ScaleLine = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomInText │ │ │ │ │ - * {String} │ │ │ │ │ - * Text for zoom-in link. Default is "+". │ │ │ │ │ + * Property: maxWidth │ │ │ │ │ + * {Integer} Maximum width of the scale line in pixels. Default is 100. │ │ │ │ │ */ │ │ │ │ │ - zoomInText: "+", │ │ │ │ │ + maxWidth: 100, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomInId │ │ │ │ │ - * {String} │ │ │ │ │ - * Instead of having the control create a zoom in link, you can provide │ │ │ │ │ - * the identifier for an anchor element already added to the document. │ │ │ │ │ - * By default, an element with id "olZoomInLink" will be searched for │ │ │ │ │ - * and used if it exists. │ │ │ │ │ + * Property: topOutUnits │ │ │ │ │ + * {String} Units for zoomed out on top bar. Default is km. │ │ │ │ │ */ │ │ │ │ │ - zoomInId: "olZoomInLink", │ │ │ │ │ + topOutUnits: "km", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomOutText │ │ │ │ │ - * {String} │ │ │ │ │ - * Text for zoom-out link. Default is "\u2212". │ │ │ │ │ + * Property: topInUnits │ │ │ │ │ + * {String} Units for zoomed in on top bar. Default is m. │ │ │ │ │ */ │ │ │ │ │ - zoomOutText: "\u2212", │ │ │ │ │ + topInUnits: "m", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomOutId │ │ │ │ │ - * {String} │ │ │ │ │ - * Instead of having the control create a zoom out link, you can provide │ │ │ │ │ - * the identifier for an anchor element already added to the document. │ │ │ │ │ - * By default, an element with id "olZoomOutLink" will be searched for │ │ │ │ │ - * and used if it exists. │ │ │ │ │ + * Property: bottomOutUnits │ │ │ │ │ + * {String} Units for zoomed out on bottom bar. Default is mi. │ │ │ │ │ + */ │ │ │ │ │ + bottomOutUnits: "mi", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: bottomInUnits │ │ │ │ │ + * {String} Units for zoomed in on bottom bar. Default is ft. │ │ │ │ │ + */ │ │ │ │ │ + bottomInUnits: "ft", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: eTop │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + eTop: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: eBottom │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + eBottom: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: geodesic │ │ │ │ │ + * {Boolean} Use geodesic measurement. Default is false. The recommended │ │ │ │ │ + * setting for maps in EPSG:4326 is false, and true EPSG:900913. If set to │ │ │ │ │ + * true, the scale will be calculated based on the horizontal size of the │ │ │ │ │ + * pixel in the center of the map viewport. │ │ │ │ │ + */ │ │ │ │ │ + geodesic: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.ScaleLine │ │ │ │ │ + * Create a new scale line control. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be used │ │ │ │ │ + * to extend the control. │ │ │ │ │ */ │ │ │ │ │ - zoomOutId: "olZoomOutLink", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Method: draw │ │ │ │ │ - * │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} A reference to the DOMElement containing the zoom links. │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ draw: function() { │ │ │ │ │ - var div = OpenLayers.Control.prototype.draw.apply(this), │ │ │ │ │ - links = this.getOrCreateLinks(div), │ │ │ │ │ - zoomIn = links.zoomIn, │ │ │ │ │ - zoomOut = links.zoomOut, │ │ │ │ │ - eventsInstance = this.map.events; │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (!this.eTop) { │ │ │ │ │ + // stick in the top bar │ │ │ │ │ + this.eTop = document.createElement("div"); │ │ │ │ │ + this.eTop.className = this.displayClass + "Top"; │ │ │ │ │ + var theLen = this.topInUnits.length; │ │ │ │ │ + this.div.appendChild(this.eTop); │ │ │ │ │ + if ((this.topOutUnits == "") || (this.topInUnits == "")) { │ │ │ │ │ + this.eTop.style.visibility = "hidden"; │ │ │ │ │ + } else { │ │ │ │ │ + this.eTop.style.visibility = "visible"; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (zoomOut.parentNode !== div) { │ │ │ │ │ - eventsInstance = this.events; │ │ │ │ │ - eventsInstance.attachToElement(zoomOut.parentNode); │ │ │ │ │ + // and the bottom bar │ │ │ │ │ + this.eBottom = document.createElement("div"); │ │ │ │ │ + this.eBottom.className = this.displayClass + "Bottom"; │ │ │ │ │ + this.div.appendChild(this.eBottom); │ │ │ │ │ + if ((this.bottomOutUnits == "") || (this.bottomInUnits == "")) { │ │ │ │ │ + this.eBottom.style.visibility = "hidden"; │ │ │ │ │ + } else { │ │ │ │ │ + this.eBottom.style.visibility = "visible"; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - eventsInstance.register("buttonclick", this, this.onZoomClick); │ │ │ │ │ + this.map.events.register('moveend', this, this.update); │ │ │ │ │ + this.update(); │ │ │ │ │ + return this.div; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.zoomInLink = zoomIn; │ │ │ │ │ - this.zoomOutLink = zoomOut; │ │ │ │ │ - return div; │ │ │ │ │ + /** │ │ │ │ │ + * Method: getBarLen │ │ │ │ │ + * Given a number, round it down to the nearest 1,2,5 times a power of 10. │ │ │ │ │ + * That seems a fairly useful set of number groups to use. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * maxLen - {float} the number we're rounding down from │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} the rounded number (less than or equal to maxLen) │ │ │ │ │ + */ │ │ │ │ │ + getBarLen: function(maxLen) { │ │ │ │ │ + // nearest power of 10 lower than maxLen │ │ │ │ │ + var digits = parseInt(Math.log(maxLen) / Math.log(10)); │ │ │ │ │ + var pow10 = Math.pow(10, digits); │ │ │ │ │ + │ │ │ │ │ + // ok, find first character │ │ │ │ │ + var firstChar = parseInt(maxLen / pow10); │ │ │ │ │ + │ │ │ │ │ + // right, put it into the correct bracket │ │ │ │ │ + var barLen; │ │ │ │ │ + if (firstChar > 5) { │ │ │ │ │ + barLen = 5; │ │ │ │ │ + } else if (firstChar > 2) { │ │ │ │ │ + barLen = 2; │ │ │ │ │ + } else { │ │ │ │ │ + barLen = 1; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // scale it up the correct power of 10 │ │ │ │ │ + return barLen * pow10; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getOrCreateLinks │ │ │ │ │ - * │ │ │ │ │ + * Method: update │ │ │ │ │ + * Update the size of the bars, and the labels they contain. │ │ │ │ │ + */ │ │ │ │ │ + update: function() { │ │ │ │ │ + var res = this.map.getResolution(); │ │ │ │ │ + if (!res) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var curMapUnits = this.map.getUnits(); │ │ │ │ │ + var inches = OpenLayers.INCHES_PER_UNIT; │ │ │ │ │ + │ │ │ │ │ + // convert maxWidth to map units │ │ │ │ │ + var maxSizeData = this.maxWidth * res * inches[curMapUnits]; │ │ │ │ │ + var geodesicRatio = 1; │ │ │ │ │ + if (this.geodesic === true) { │ │ │ │ │ + var maxSizeGeodesic = (this.map.getGeodesicPixelSize().w || │ │ │ │ │ + 0.000001) * this.maxWidth; │ │ │ │ │ + var maxSizeKilometers = maxSizeData / inches["km"]; │ │ │ │ │ + geodesicRatio = maxSizeGeodesic / maxSizeKilometers; │ │ │ │ │ + maxSizeData *= geodesicRatio; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // decide whether to use large or small scale units │ │ │ │ │ + var topUnits; │ │ │ │ │ + var bottomUnits; │ │ │ │ │ + if (maxSizeData > 100000) { │ │ │ │ │ + topUnits = this.topOutUnits; │ │ │ │ │ + bottomUnits = this.bottomOutUnits; │ │ │ │ │ + } else { │ │ │ │ │ + topUnits = this.topInUnits; │ │ │ │ │ + bottomUnits = this.bottomInUnits; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // and to map units units │ │ │ │ │ + var topMax = maxSizeData / inches[topUnits]; │ │ │ │ │ + var bottomMax = maxSizeData / inches[bottomUnits]; │ │ │ │ │ + │ │ │ │ │ + // now trim this down to useful block length │ │ │ │ │ + var topRounded = this.getBarLen(topMax); │ │ │ │ │ + var bottomRounded = this.getBarLen(bottomMax); │ │ │ │ │ + │ │ │ │ │ + // and back to display units │ │ │ │ │ + topMax = topRounded / inches[curMapUnits] * inches[topUnits]; │ │ │ │ │ + bottomMax = bottomRounded / inches[curMapUnits] * inches[bottomUnits]; │ │ │ │ │ + │ │ │ │ │ + // and to pixel units │ │ │ │ │ + var topPx = topMax / res / geodesicRatio; │ │ │ │ │ + var bottomPx = bottomMax / res / geodesicRatio; │ │ │ │ │ + │ │ │ │ │ + // now set the pixel widths │ │ │ │ │ + // and the values inside them │ │ │ │ │ + │ │ │ │ │ + if (this.eBottom.style.visibility == "visible") { │ │ │ │ │ + this.eBottom.style.width = Math.round(bottomPx) + "px"; │ │ │ │ │ + this.eBottom.innerHTML = bottomRounded + " " + bottomUnits; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.eTop.style.visibility == "visible") { │ │ │ │ │ + this.eTop.style.width = Math.round(topPx) + "px"; │ │ │ │ │ + this.eTop.innerHTML = topRounded + " " + topUnits; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ScaleLine" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/Vector/RootContainer.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.Vector.RootContainer │ │ │ │ │ + * A special layer type to combine multiple vector layers inside a single │ │ │ │ │ + * renderer root container. This class is not supposed to be instantiated │ │ │ │ │ + * from user space, it is a helper class for controls that require event │ │ │ │ │ + * processing for multiple vector layers. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Vector> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Vector.RootContainer = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: displayInLayerSwitcher │ │ │ │ │ + * Set to false for this layer type │ │ │ │ │ + */ │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layers │ │ │ │ │ + * Layers that are attached to this container. Required config option. │ │ │ │ │ + */ │ │ │ │ │ + layers: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.Vector.RootContainer │ │ │ │ │ + * Create a new root container for multiple vector layer. This constructor │ │ │ │ │ + * is not supposed to be used from user space, it is only to be used by │ │ │ │ │ + * controls that need feature selection across multiple vector layers. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * el - {DOMElement} │ │ │ │ │ + * name - {String} A name for the layer │ │ │ │ │ + * options - {Object} Optional object with non-default properties to set on │ │ │ │ │ + * the layer. │ │ │ │ │ + * │ │ │ │ │ + * Required options properties: │ │ │ │ │ + * layers - {Array(<OpenLayers.Layer.Vector>)} The layers managed by this │ │ │ │ │ + * container │ │ │ │ │ * │ │ │ │ │ - * Return: │ │ │ │ │ - * {Object} Object with zoomIn and zoomOut properties referencing links. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Vector.RootContainer>} A new vector layer root │ │ │ │ │ + * container │ │ │ │ │ */ │ │ │ │ │ - getOrCreateLinks: function(el) { │ │ │ │ │ - var zoomIn = document.getElementById(this.zoomInId), │ │ │ │ │ - zoomOut = document.getElementById(this.zoomOutId); │ │ │ │ │ - if (!zoomIn) { │ │ │ │ │ - zoomIn = document.createElement("a"); │ │ │ │ │ - zoomIn.href = "#zoomIn"; │ │ │ │ │ - zoomIn.appendChild(document.createTextNode(this.zoomInText)); │ │ │ │ │ - zoomIn.className = "olControlZoomIn"; │ │ │ │ │ - el.appendChild(zoomIn); │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: display │ │ │ │ │ + */ │ │ │ │ │ + display: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getFeatureFromEvent │ │ │ │ │ + * walk through the layers to find the feature returned by the event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} event object with a feature property │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + getFeatureFromEvent: function(evt) { │ │ │ │ │ + var layers = this.layers; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0; i < layers.length; i++) { │ │ │ │ │ + feature = layers[i].getFeatureFromEvent(evt); │ │ │ │ │ + if (feature) { │ │ │ │ │ + return feature; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Element.addClass(zoomIn, "olButton"); │ │ │ │ │ - if (!zoomOut) { │ │ │ │ │ - zoomOut = document.createElement("a"); │ │ │ │ │ - zoomOut.href = "#zoomOut"; │ │ │ │ │ - zoomOut.appendChild(document.createTextNode(this.zoomOutText)); │ │ │ │ │ - zoomOut.className = "olControlZoomOut"; │ │ │ │ │ - el.appendChild(zoomOut); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.collectRoots(); │ │ │ │ │ + map.events.register("changelayer", this, this.handleChangeLayer); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeMap │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + map.events.unregister("changelayer", this, this.handleChangeLayer); │ │ │ │ │ + this.resetRoots(); │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: collectRoots │ │ │ │ │ + * Collects the root nodes of all layers this control is configured with │ │ │ │ │ + * and moveswien the nodes to this control's layer │ │ │ │ │ + */ │ │ │ │ │ + collectRoots: function() { │ │ │ │ │ + var layer; │ │ │ │ │ + // walk through all map layers, because we want to keep the order │ │ │ │ │ + for (var i = 0; i < this.map.layers.length; ++i) { │ │ │ │ │ + layer = this.map.layers[i]; │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ + layer.renderer.moveRoot(this.renderer); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Element.addClass(zoomOut, "olButton"); │ │ │ │ │ - return { │ │ │ │ │ - zoomIn: zoomIn, │ │ │ │ │ - zoomOut: zoomOut │ │ │ │ │ - }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onZoomClick │ │ │ │ │ - * Called when zoomin/out link is clicked. │ │ │ │ │ + * Method: resetRoots │ │ │ │ │ + * Resets the root nodes back into the layers they belong to. │ │ │ │ │ */ │ │ │ │ │ - onZoomClick: function(evt) { │ │ │ │ │ - var button = evt.buttonElement; │ │ │ │ │ - if (button === this.zoomInLink) { │ │ │ │ │ - this.map.zoomIn(); │ │ │ │ │ - } else if (button === this.zoomOutLink) { │ │ │ │ │ - this.map.zoomOut(); │ │ │ │ │ + resetRoots: function() { │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = 0; i < this.layers.length; ++i) { │ │ │ │ │ + layer = this.layers[i]; │ │ │ │ │ + if (this.renderer && layer.renderer.getRenderLayerId() == this.id) { │ │ │ │ │ + this.renderer.moveRoot(layer.renderer); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Clean up. │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleChangeLayer │ │ │ │ │ + * Event handler for the map's changelayer event. We need to rebuild │ │ │ │ │ + * this container's layer dom if order of one of its layers changes. │ │ │ │ │ + * This handler is added with the setMap method, and removed with the │ │ │ │ │ + * removeMap method. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onZoomClick); │ │ │ │ │ + handleChangeLayer: function(evt) { │ │ │ │ │ + var layer = evt.layer; │ │ │ │ │ + if (evt.property == "order" && │ │ │ │ │ + OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ + this.resetRoots(); │ │ │ │ │ + this.collectRoots(); │ │ │ │ │ } │ │ │ │ │ - delete this.zoomInLink; │ │ │ │ │ - delete this.zoomOutLink; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Zoom" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Vector.RootContainer" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Control/SelectFeature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ @@ -77051,1557 +73697,1084 @@ │ │ │ │ │ this.activate(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.SelectFeature" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/NavigationHistory.js │ │ │ │ │ + OpenLayers/Control/EditingToolbar.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control/Panel.js │ │ │ │ │ + * @requires OpenLayers/Control/Navigation.js │ │ │ │ │ + * @requires OpenLayers/Control/DrawFeature.js │ │ │ │ │ + * @requires OpenLayers/Handler/Point.js │ │ │ │ │ + * @requires OpenLayers/Handler/Path.js │ │ │ │ │ + * @requires OpenLayers/Handler/Polygon.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.EditingToolbar │ │ │ │ │ + * The EditingToolbar is a panel of 4 controls to draw polygons, lines, │ │ │ │ │ + * points, or to navigate the map by panning. By default it appears in the │ │ │ │ │ + * upper right corner of the map. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control.Panel> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.EditingToolbar = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Control.Panel, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: citeCompliant │ │ │ │ │ + * {Boolean} If set to true, coordinates of features drawn in a map extent │ │ │ │ │ + * crossing the date line won't exceed the world bounds. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + citeCompliant: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.EditingToolbar │ │ │ │ │ + * Create an editing toolbar for a given layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ + * options - {Object} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(layer, options) { │ │ │ │ │ + OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ + │ │ │ │ │ + this.addControls( │ │ │ │ │ + [new OpenLayers.Control.Navigation()] │ │ │ │ │ + ); │ │ │ │ │ + var controls = [ │ │ │ │ │ + new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Point, { │ │ │ │ │ + displayClass: 'olControlDrawFeaturePoint', │ │ │ │ │ + handlerOptions: { │ │ │ │ │ + citeCompliant: this.citeCompliant │ │ │ │ │ + } │ │ │ │ │ + }), │ │ │ │ │ + new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Path, { │ │ │ │ │ + displayClass: 'olControlDrawFeaturePath', │ │ │ │ │ + handlerOptions: { │ │ │ │ │ + citeCompliant: this.citeCompliant │ │ │ │ │ + } │ │ │ │ │ + }), │ │ │ │ │ + new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Polygon, { │ │ │ │ │ + displayClass: 'olControlDrawFeaturePolygon', │ │ │ │ │ + handlerOptions: { │ │ │ │ │ + citeCompliant: this.citeCompliant │ │ │ │ │ + } │ │ │ │ │ + }) │ │ │ │ │ + ]; │ │ │ │ │ + this.addControls(controls); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * calls the default draw, and then activates mouse defaults. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + draw: function() { │ │ │ │ │ + var div = OpenLayers.Control.Panel.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (this.defaultControl === null) { │ │ │ │ │ + this.defaultControl = this.controls[0]; │ │ │ │ │ + } │ │ │ │ │ + return div; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.EditingToolbar" │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/Attribution.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Control/Button.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.NavigationHistory │ │ │ │ │ - * A navigation history control. This is a meta-control, that creates two │ │ │ │ │ - * dependent controls: <previous> and <next>. Call the trigger method │ │ │ │ │ - * on the <previous> and <next> controls to restore previous and next │ │ │ │ │ - * history states. The previous and next controls will become active │ │ │ │ │ - * when there are available states to restore and will become deactive │ │ │ │ │ - * when there are no states to restore. │ │ │ │ │ + * Class: OpenLayers.Control.Attribution │ │ │ │ │ + * The attribution control adds attribution from layers to the map display. │ │ │ │ │ + * It uses 'attribution' property of each layer. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.NavigationHistory = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Control.Attribution = │ │ │ │ │ + OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: type │ │ │ │ │ - * {String} Note that this control is not intended to be added directly │ │ │ │ │ - * to a control panel. Instead, add the sub-controls previous and │ │ │ │ │ - * next. These sub-controls are button type controls that activate │ │ │ │ │ - * and deactivate themselves. If this parent control is added to │ │ │ │ │ - * a panel, it will act as a toggle. │ │ │ │ │ - */ │ │ │ │ │ - type: OpenLayers.Control.TYPE_TOGGLE, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: separator │ │ │ │ │ + * {String} String used to separate layers. │ │ │ │ │ + */ │ │ │ │ │ + separator: ", ", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: previous │ │ │ │ │ - * {<OpenLayers.Control>} A button type control whose trigger method restores │ │ │ │ │ - * the previous state managed by this control. │ │ │ │ │ - */ │ │ │ │ │ - previous: null, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: template │ │ │ │ │ + * {String} Template for the attribution. This has to include the substring │ │ │ │ │ + * "${layers}", which will be replaced by the layer specific │ │ │ │ │ + * attributions, separated by <separator>. The default is "${layers}". │ │ │ │ │ + */ │ │ │ │ │ + template: "${layers}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: previousOptions │ │ │ │ │ - * {Object} Set this property on the options argument of the constructor │ │ │ │ │ - * to set optional properties on the <previous> control. │ │ │ │ │ - */ │ │ │ │ │ - previousOptions: null, │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.Attribution │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Options for control. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: next │ │ │ │ │ - * {<OpenLayers.Control>} A button type control whose trigger method restores │ │ │ │ │ - * the next state managed by this control. │ │ │ │ │ - */ │ │ │ │ │ - next: null, │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Destroy control. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + "removelayer": this.updateAttribution, │ │ │ │ │ + "addlayer": this.updateAttribution, │ │ │ │ │ + "changelayer": this.updateAttribution, │ │ │ │ │ + "changebaselayer": this.updateAttribution, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: nextOptions │ │ │ │ │ - * {Object} Set this property on the options argument of the constructor │ │ │ │ │ - * to set optional properties on the <next> control. │ │ │ │ │ - */ │ │ │ │ │ - nextOptions: null, │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: limit │ │ │ │ │ - * {Integer} Optional limit on the number of history items to retain. If │ │ │ │ │ - * null, there is no limit. Default is 50. │ │ │ │ │ - */ │ │ │ │ │ - limit: 50, │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Initialize control. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ + */ │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + 'changebaselayer': this.updateAttribution, │ │ │ │ │ + 'changelayer': this.updateAttribution, │ │ │ │ │ + 'addlayer': this.updateAttribution, │ │ │ │ │ + 'removelayer': this.updateAttribution, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.updateAttribution(); │ │ │ │ │ + │ │ │ │ │ + return this.div; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateAttribution │ │ │ │ │ + * Update attribution string. │ │ │ │ │ + */ │ │ │ │ │ + updateAttribution: function() { │ │ │ │ │ + var attributions = []; │ │ │ │ │ + if (this.map && this.map.layers) { │ │ │ │ │ + for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if (layer.attribution && layer.getVisibility()) { │ │ │ │ │ + // add attribution only if attribution text is unique │ │ │ │ │ + if (OpenLayers.Util.indexOf( │ │ │ │ │ + attributions, layer.attribution) === -1) { │ │ │ │ │ + attributions.push(layer.attribution); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.div.innerHTML = OpenLayers.String.format(this.template, { │ │ │ │ │ + layers: attributions.join(this.separator) │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Attribution" │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/Geolocate.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Projection.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.Geolocate │ │ │ │ │ + * The Geolocate control wraps w3c geolocation API into control that can be │ │ │ │ │ + * bound to a map, and generate events on location update │ │ │ │ │ + * │ │ │ │ │ + * To use this control requires to load the proj4js library if the projection │ │ │ │ │ + * of the map is not EPSG:4326 or EPSG:900913. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.Geolocate = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ + * locationupdated - Triggered when browser return a new position. Listeners will │ │ │ │ │ + * receive an object with a 'position' property which is the browser.geolocation.position │ │ │ │ │ + * native object, as well as a 'point' property which is the location transformed in the │ │ │ │ │ + * current map projection. │ │ │ │ │ + * locationfailed - Triggered when geolocation has failed │ │ │ │ │ + * locationuncapable - Triggered when control is activated on a browser │ │ │ │ │ + * which doesn't support geolocation │ │ │ │ │ */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: clearOnDeactivate │ │ │ │ │ - * {Boolean} Clear the history when the control is deactivated. Default │ │ │ │ │ - * is false. │ │ │ │ │ + * Property: geolocation │ │ │ │ │ + * {Object} The geolocation engine, as a property to be possibly mocked. │ │ │ │ │ + * This is set lazily to avoid a memory leak in IE9. │ │ │ │ │ */ │ │ │ │ │ - clearOnDeactivate: false, │ │ │ │ │ + geolocation: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: registry │ │ │ │ │ - * {Object} An object with keys corresponding to event types. Values │ │ │ │ │ - * are functions that return an object representing the current state. │ │ │ │ │ + * Property: available │ │ │ │ │ + * {Boolean} The navigator.geolocation object is available. │ │ │ │ │ */ │ │ │ │ │ - registry: null, │ │ │ │ │ + available: ('geolocation' in navigator), │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: nextStack │ │ │ │ │ - * {Array} Array of items in the history. │ │ │ │ │ + * APIProperty: bind │ │ │ │ │ + * {Boolean} If true, map center will be set on location update. │ │ │ │ │ */ │ │ │ │ │ - nextStack: null, │ │ │ │ │ + bind: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: previousStack │ │ │ │ │ - * {Array} List of items in the history. First item represents the current │ │ │ │ │ - * state. │ │ │ │ │ + * APIProperty: watch │ │ │ │ │ + * {Boolean} If true, position will be update regularly. │ │ │ │ │ */ │ │ │ │ │ - previousStack: null, │ │ │ │ │ + watch: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: listeners │ │ │ │ │ - * {Object} An object containing properties corresponding to event types. │ │ │ │ │ - * This object is used to configure the control and is modified on │ │ │ │ │ - * construction. │ │ │ │ │ + * APIProperty: geolocationOptions │ │ │ │ │ + * {Object} Options to pass to the navigator's geolocation API. See │ │ │ │ │ + * <http://dev.w3.org/geo/api/spec-source.html>. No specific │ │ │ │ │ + * option is passed to the geolocation API by default. │ │ │ │ │ */ │ │ │ │ │ - listeners: null, │ │ │ │ │ + geolocationOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: restoring │ │ │ │ │ - * {Boolean} Currently restoring a history state. This is set to true │ │ │ │ │ - * before calling restore and set to false after restore returns. │ │ │ │ │ + * Constructor: OpenLayers.Control.Geolocate │ │ │ │ │ + * Create a new control to deal with browser geolocation API │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - restoring: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.NavigationHistory │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be used │ │ │ │ │ - * to extend the control. │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - │ │ │ │ │ - this.registry = OpenLayers.Util.extend({ │ │ │ │ │ - "moveend": this.getState │ │ │ │ │ - }, this.registry); │ │ │ │ │ - │ │ │ │ │ - var previousOptions = { │ │ │ │ │ - trigger: OpenLayers.Function.bind(this.previousTrigger, this), │ │ │ │ │ - displayClass: this.displayClass + " " + this.displayClass + "Previous" │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Util.extend(previousOptions, this.previousOptions); │ │ │ │ │ - this.previous = new OpenLayers.Control.Button(previousOptions); │ │ │ │ │ - │ │ │ │ │ - var nextOptions = { │ │ │ │ │ - trigger: OpenLayers.Function.bind(this.nextTrigger, this), │ │ │ │ │ - displayClass: this.displayClass + " " + this.displayClass + "Next" │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Util.extend(nextOptions, this.nextOptions); │ │ │ │ │ - this.next = new OpenLayers.Control.Button(nextOptions); │ │ │ │ │ - │ │ │ │ │ - this.clear(); │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onPreviousChange │ │ │ │ │ - * Called when the previous history stack changes. │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activates the control. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * state - {Object} An object representing the state to be restored │ │ │ │ │ - * if previous is triggered again or null if no previous states remain. │ │ │ │ │ - * length - {Integer} The number of remaining previous states that can │ │ │ │ │ - * be restored. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The control was effectively activated. │ │ │ │ │ */ │ │ │ │ │ - onPreviousChange: function(state, length) { │ │ │ │ │ - if (state && !this.previous.active) { │ │ │ │ │ - this.previous.activate(); │ │ │ │ │ - } else if (!state && this.previous.active) { │ │ │ │ │ - this.previous.deactivate(); │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (this.available && !this.geolocation) { │ │ │ │ │ + // set lazily to avoid IE9 memory leak │ │ │ │ │ + this.geolocation = navigator.geolocation; │ │ │ │ │ + } │ │ │ │ │ + if (!this.geolocation) { │ │ │ │ │ + this.events.triggerEvent("locationuncapable"); │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + if (this.watch) { │ │ │ │ │ + this.watchId = this.geolocation.watchPosition( │ │ │ │ │ + OpenLayers.Function.bind(this.geolocate, this), │ │ │ │ │ + OpenLayers.Function.bind(this.failure, this), │ │ │ │ │ + this.geolocationOptions │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + this.getCurrentLocation(); │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onNextChange │ │ │ │ │ - * Called when the next history stack changes. │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivates the control. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * state - {Object} An object representing the state to be restored │ │ │ │ │ - * if next is triggered again or null if no next states remain. │ │ │ │ │ - * length - {Integer} The number of remaining next states that can │ │ │ │ │ - * be restored. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The control was effectively deactivated. │ │ │ │ │ */ │ │ │ │ │ - onNextChange: function(state, length) { │ │ │ │ │ - if (state && !this.next.active) { │ │ │ │ │ - this.next.activate(); │ │ │ │ │ - } else if (!state && this.next.active) { │ │ │ │ │ - this.next.deactivate(); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active && this.watchId !== null) { │ │ │ │ │ + this.geolocation.clearWatch(this.watchId); │ │ │ │ │ } │ │ │ │ │ + return OpenLayers.Control.prototype.deactivate.apply( │ │ │ │ │ + this, arguments │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Destroy the control. │ │ │ │ │ + * Method: geolocate │ │ │ │ │ + * Activates the control. │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this); │ │ │ │ │ - this.previous.destroy(); │ │ │ │ │ - this.next.destroy(); │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - for (var prop in this) { │ │ │ │ │ - this[prop] = null; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the control and <previous> and <next> child │ │ │ │ │ - * controls. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.map = map; │ │ │ │ │ - this.next.setMap(map); │ │ │ │ │ - this.previous.setMap(map); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Called when the control is added to the map. │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - this.next.draw(); │ │ │ │ │ - this.previous.draw(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: previousTrigger │ │ │ │ │ - * Restore the previous state. If no items are in the previous history │ │ │ │ │ - * stack, this has no effect. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Item representing state that was restored. Undefined if no │ │ │ │ │ - * items are in the previous history stack. │ │ │ │ │ - */ │ │ │ │ │ - previousTrigger: function() { │ │ │ │ │ - var current = this.previousStack.shift(); │ │ │ │ │ - var state = this.previousStack.shift(); │ │ │ │ │ - if (state != undefined) { │ │ │ │ │ - this.nextStack.unshift(current); │ │ │ │ │ - this.previousStack.unshift(state); │ │ │ │ │ - this.restoring = true; │ │ │ │ │ - this.restore(state); │ │ │ │ │ - this.restoring = false; │ │ │ │ │ - this.onNextChange(this.nextStack[0], this.nextStack.length); │ │ │ │ │ - this.onPreviousChange( │ │ │ │ │ - this.previousStack[1], this.previousStack.length - 1 │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - this.previousStack.unshift(current); │ │ │ │ │ - } │ │ │ │ │ - return state; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: nextTrigger │ │ │ │ │ - * Restore the next state. If no items are in the next history │ │ │ │ │ - * stack, this has no effect. The next history stack is populated │ │ │ │ │ - * as states are restored from the previous history stack. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Item representing state that was restored. Undefined if no │ │ │ │ │ - * items are in the next history stack. │ │ │ │ │ - */ │ │ │ │ │ - nextTrigger: function() { │ │ │ │ │ - var state = this.nextStack.shift(); │ │ │ │ │ - if (state != undefined) { │ │ │ │ │ - this.previousStack.unshift(state); │ │ │ │ │ - this.restoring = true; │ │ │ │ │ - this.restore(state); │ │ │ │ │ - this.restoring = false; │ │ │ │ │ - this.onNextChange(this.nextStack[0], this.nextStack.length); │ │ │ │ │ - this.onPreviousChange( │ │ │ │ │ - this.previousStack[1], this.previousStack.length - 1 │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return state; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clear │ │ │ │ │ - * Clear history. │ │ │ │ │ - */ │ │ │ │ │ - clear: function() { │ │ │ │ │ - this.previousStack = []; │ │ │ │ │ - this.previous.deactivate(); │ │ │ │ │ - this.nextStack = []; │ │ │ │ │ - this.next.deactivate(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getState │ │ │ │ │ - * Get the current state and return it. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object representing the current state. │ │ │ │ │ - */ │ │ │ │ │ - getState: function() { │ │ │ │ │ - return { │ │ │ │ │ - center: this.map.getCenter(), │ │ │ │ │ - resolution: this.map.getResolution(), │ │ │ │ │ - projection: this.map.getProjectionObject(), │ │ │ │ │ - units: this.map.getProjectionObject().getUnits() || │ │ │ │ │ - this.map.units || this.map.baseLayer.units │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: restore │ │ │ │ │ - * Update the state with the given object. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * state - {Object} An object representing the state to restore. │ │ │ │ │ - */ │ │ │ │ │ - restore: function(state) { │ │ │ │ │ - var center, zoom; │ │ │ │ │ - if (this.map.getProjectionObject() == state.projection) { │ │ │ │ │ - zoom = this.map.getZoomForResolution(state.resolution); │ │ │ │ │ - center = state.center; │ │ │ │ │ - } else { │ │ │ │ │ - center = state.center.clone(); │ │ │ │ │ - center.transform(state.projection, this.map.getProjectionObject()); │ │ │ │ │ - var sourceUnits = state.units; │ │ │ │ │ - var targetUnits = this.map.getProjectionObject().getUnits() || │ │ │ │ │ - this.map.units || this.map.baseLayer.units; │ │ │ │ │ - var resolutionFactor = sourceUnits && targetUnits ? │ │ │ │ │ - OpenLayers.INCHES_PER_UNIT[sourceUnits] / OpenLayers.INCHES_PER_UNIT[targetUnits] : 1; │ │ │ │ │ - zoom = this.map.getZoomForResolution(resolutionFactor * state.resolution); │ │ │ │ │ - } │ │ │ │ │ - this.map.setCenter(center, zoom); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setListeners │ │ │ │ │ - * Sets functions to be registered in the listeners object. │ │ │ │ │ - */ │ │ │ │ │ - setListeners: function() { │ │ │ │ │ - this.listeners = {}; │ │ │ │ │ - for (var type in this.registry) { │ │ │ │ │ - this.listeners[type] = OpenLayers.Function.bind(function() { │ │ │ │ │ - if (!this.restoring) { │ │ │ │ │ - var state = this.registry[type].apply(this, arguments); │ │ │ │ │ - this.previousStack.unshift(state); │ │ │ │ │ - if (this.previousStack.length > 1) { │ │ │ │ │ - this.onPreviousChange( │ │ │ │ │ - this.previousStack[1], this.previousStack.length - 1 │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (this.previousStack.length > (this.limit + 1)) { │ │ │ │ │ - this.previousStack.pop(); │ │ │ │ │ - } │ │ │ │ │ - if (this.nextStack.length > 0) { │ │ │ │ │ - this.nextStack = []; │ │ │ │ │ - this.onNextChange(null, 0); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - }, this); │ │ │ │ │ + geolocate: function(position) { │ │ │ │ │ + var center = new OpenLayers.LonLat( │ │ │ │ │ + position.coords.longitude, │ │ │ │ │ + position.coords.latitude │ │ │ │ │ + ).transform( │ │ │ │ │ + new OpenLayers.Projection("EPSG:4326"), │ │ │ │ │ + this.map.getProjectionObject() │ │ │ │ │ + ); │ │ │ │ │ + if (this.bind) { │ │ │ │ │ + this.map.setCenter(center); │ │ │ │ │ } │ │ │ │ │ + this.events.triggerEvent("locationupdated", { │ │ │ │ │ + position: position, │ │ │ │ │ + point: new OpenLayers.Geometry.Point( │ │ │ │ │ + center.lon, center.lat │ │ │ │ │ + ) │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Activate the control. This registers any listeners. │ │ │ │ │ + * APIMethod: getCurrentLocation │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Control successfully activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this)) { │ │ │ │ │ - if (this.listeners == null) { │ │ │ │ │ - this.setListeners(); │ │ │ │ │ - } │ │ │ │ │ - for (var type in this.listeners) { │ │ │ │ │ - this.map.events.register(type, this, this.listeners[type]); │ │ │ │ │ - } │ │ │ │ │ - activated = true; │ │ │ │ │ - if (this.previousStack.length == 0) { │ │ │ │ │ - this.initStack(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: initStack │ │ │ │ │ - * Called after the control is activated if the previous history stack is │ │ │ │ │ - * empty. │ │ │ │ │ + * {Boolean} Returns true if a event will be fired (successfull │ │ │ │ │ + * registration) │ │ │ │ │ */ │ │ │ │ │ - initStack: function() { │ │ │ │ │ - if (this.map.getCenter()) { │ │ │ │ │ - this.listeners.moveend(); │ │ │ │ │ + getCurrentLocation: function() { │ │ │ │ │ + if (!this.active || this.watch) { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ + this.geolocation.getCurrentPosition( │ │ │ │ │ + OpenLayers.Function.bind(this.geolocate, this), │ │ │ │ │ + OpenLayers.Function.bind(this.failure, this), │ │ │ │ │ + this.geolocationOptions │ │ │ │ │ + ); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the control. This unregisters any listeners. │ │ │ │ │ + * Method: failure │ │ │ │ │ + * method called on browser's geolocation failure │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Control successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this)) { │ │ │ │ │ - for (var type in this.listeners) { │ │ │ │ │ - this.map.events.unregister( │ │ │ │ │ - type, this, this.listeners[type] │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (this.clearOnDeactivate) { │ │ │ │ │ - this.clear(); │ │ │ │ │ - } │ │ │ │ │ - deactivated = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ + failure: function(error) { │ │ │ │ │ + this.events.triggerEvent("locationfailed", { │ │ │ │ │ + error: error │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.NavigationHistory" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Geolocate" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/TouchNavigation.js │ │ │ │ │ + OpenLayers/Control/UTFGrid.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Control/DragPan.js │ │ │ │ │ - * @requires OpenLayers/Control/PinchZoom.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Handler/Hover.js │ │ │ │ │ * @requires OpenLayers/Handler/Click.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.TouchNavigation │ │ │ │ │ - * The navigation control handles map browsing with touch events (dragging, │ │ │ │ │ - * double-tapping, tap with two fingers, and pinch zoom). Create a new │ │ │ │ │ - * control with the <OpenLayers.Control.TouchNavigation> constructor. │ │ │ │ │ + * Class: OpenLayers.Control.UTFGrid │ │ │ │ │ * │ │ │ │ │ - * If you’re only targeting touch enabled devices with your mapping application, │ │ │ │ │ - * you can create a map with only a TouchNavigation control. The │ │ │ │ │ - * <OpenLayers.Control.Navigation> control is mobile ready by default, but │ │ │ │ │ - * you can generate a smaller build of the library by only including this │ │ │ │ │ - * touch navigation control if you aren't concerned about mouse interaction. │ │ │ │ │ + * This Control provides behavior associated with UTFGrid Layers. │ │ │ │ │ + * These 'hit grids' provide underlying feature attributes without │ │ │ │ │ + * calling the server (again). This control allows Mousemove, Hovering │ │ │ │ │ + * and Click events to trigger callbacks that use the attributes in │ │ │ │ │ + * whatever way you need. │ │ │ │ │ * │ │ │ │ │ - * Inherits: │ │ │ │ │ + * The most common example may be a UTFGrid layer containing feature │ │ │ │ │ + * attributes that are displayed in a div as you mouseover. │ │ │ │ │ + * │ │ │ │ │ + * Example Code: │ │ │ │ │ + * │ │ │ │ │ + * (start code) │ │ │ │ │ + * var world_utfgrid = new OpenLayers.Layer.UTFGrid( │ │ │ │ │ + * 'UTFGrid Layer', │ │ │ │ │ + * "http://tiles/world_utfgrid/${z}/${x}/${y}.json" │ │ │ │ │ + * ); │ │ │ │ │ + * map.addLayer(world_utfgrid); │ │ │ │ │ + * │ │ │ │ │ + * var control = new OpenLayers.Control.UTFGrid({ │ │ │ │ │ + * layers: [world_utfgrid], │ │ │ │ │ + * handlerMode: 'move', │ │ │ │ │ + * callback: function(infoLookup) { │ │ │ │ │ + * // do something with returned data │ │ │ │ │ + * │ │ │ │ │ + * } │ │ │ │ │ + * }) │ │ │ │ │ + * (end code) │ │ │ │ │ + * │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.TouchNavigation = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Control.UTFGrid = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: dragPan │ │ │ │ │ - * {<OpenLayers.Control.DragPan>} │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ - dragPan: null, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: dragPanOptions │ │ │ │ │ - * {Object} Options passed to the DragPan control. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: Layers │ │ │ │ │ + * List of layers to consider. Must be Layer.UTFGrids │ │ │ │ │ + * `null` is the default indicating all UTFGrid Layers are queried. │ │ │ │ │ + * {Array} <OpenLayers.Layer.UTFGrid> │ │ │ │ │ */ │ │ │ │ │ - dragPanOptions: null, │ │ │ │ │ + layers: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: pinchZoom │ │ │ │ │ - * {<OpenLayers.Control.PinchZoom>} │ │ │ │ │ + /* Property: defaultHandlerOptions │ │ │ │ │ + * The default opts passed to the handler constructors │ │ │ │ │ */ │ │ │ │ │ - pinchZoom: null, │ │ │ │ │ + defaultHandlerOptions: { │ │ │ │ │ + 'delay': 300, │ │ │ │ │ + 'pixelTolerance': 4, │ │ │ │ │ + 'stopMove': false, │ │ │ │ │ + 'single': true, │ │ │ │ │ + 'double': false, │ │ │ │ │ + 'stopSingle': false, │ │ │ │ │ + 'stopDouble': false │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: pinchZoomOptions │ │ │ │ │ - * {Object} Options passed to the PinchZoom control. │ │ │ │ │ + /* APIProperty: handlerMode │ │ │ │ │ + * Defaults to 'click'. Can be 'hover' or 'move'. │ │ │ │ │ */ │ │ │ │ │ - pinchZoomOptions: null, │ │ │ │ │ + handlerMode: 'click', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: clickHandlerOptions │ │ │ │ │ - * {Object} Options passed to the Click handler. │ │ │ │ │ + * APIMethod: setHandler │ │ │ │ │ + * sets this.handlerMode and calls resetHandler() │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * hm - {String} Handler Mode string; 'click', 'hover' or 'move'. │ │ │ │ │ */ │ │ │ │ │ - clickHandlerOptions: null, │ │ │ │ │ + setHandler: function(hm) { │ │ │ │ │ + this.handlerMode = hm; │ │ │ │ │ + this.resetHandler(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: documentDrag │ │ │ │ │ - * {Boolean} Allow panning of the map by dragging outside map viewport. │ │ │ │ │ - * Default is false. │ │ │ │ │ + * Method: resetHandler │ │ │ │ │ + * Deactivates the old hanlder and creates a new │ │ │ │ │ + * <OpenLayers.Handler> based on the mode specified in │ │ │ │ │ + * this.handlerMode │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - documentDrag: false, │ │ │ │ │ + resetHandler: function() { │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.deactivate(); │ │ │ │ │ + this.handler.destroy(); │ │ │ │ │ + this.handler = null; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ - */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ + if (this.handlerMode == 'hover') { │ │ │ │ │ + // Handle this event on hover │ │ │ │ │ + this.handler = new OpenLayers.Handler.Hover( │ │ │ │ │ + this, { │ │ │ │ │ + 'pause': this.handleEvent, │ │ │ │ │ + 'move': this.reset │ │ │ │ │ + }, │ │ │ │ │ + this.handlerOptions │ │ │ │ │ + ); │ │ │ │ │ + } else if (this.handlerMode == 'click') { │ │ │ │ │ + // Handle this event on click │ │ │ │ │ + this.handler = new OpenLayers.Handler.Click( │ │ │ │ │ + this, { │ │ │ │ │ + 'click': this.handleEvent │ │ │ │ │ + }, this.handlerOptions │ │ │ │ │ + ); │ │ │ │ │ + } else if (this.handlerMode == 'move') { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Hover( │ │ │ │ │ + this, │ │ │ │ │ + // Handle this event while hovering OR moving │ │ │ │ │ + { │ │ │ │ │ + 'pause': this.handleEvent, │ │ │ │ │ + 'move': this.handleEvent │ │ │ │ │ + }, │ │ │ │ │ + this.handlerOptions │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.TouchNavigation │ │ │ │ │ - * Create a new navigation control │ │ │ │ │ + * Constructor: <OpenLayers.Control.UTFGrid> │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * the control │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - this.handlers = {}; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.handlerOptions = options.handlerOptions || this.defaultHandlerOptions; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.resetHandler(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * The destroy method is used to perform any clean up before the control │ │ │ │ │ - * is dereferenced. Typically this is where event listeners are removed │ │ │ │ │ - * to prevent memory leaks. │ │ │ │ │ + * Method: handleEvent │ │ │ │ │ + * Internal method called when specified event is triggered. │ │ │ │ │ + * │ │ │ │ │ + * This method does several things: │ │ │ │ │ + * │ │ │ │ │ + * Gets the lonLat of the event. │ │ │ │ │ + * │ │ │ │ │ + * Loops through the appropriate hit grid layers and gathers the attributes. │ │ │ │ │ + * │ │ │ │ │ + * Passes the attributes to the callback │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - if (this.dragPan) { │ │ │ │ │ - this.dragPan.destroy(); │ │ │ │ │ - } │ │ │ │ │ - this.dragPan = null; │ │ │ │ │ - if (this.pinchZoom) { │ │ │ │ │ - this.pinchZoom.destroy(); │ │ │ │ │ - delete this.pinchZoom; │ │ │ │ │ + handleEvent: function(evt) { │ │ │ │ │ + if (evt == null) { │ │ │ │ │ + this.reset(); │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.dragPan.activate(); │ │ │ │ │ - this.handlers.click.activate(); │ │ │ │ │ - this.pinchZoom.activate(); │ │ │ │ │ - return true; │ │ │ │ │ + var lonLat = this.map.getLonLatFromPixel(evt.xy); │ │ │ │ │ + if (!lonLat) { │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.dragPan.deactivate(); │ │ │ │ │ - this.handlers.click.deactivate(); │ │ │ │ │ - this.pinchZoom.deactivate(); │ │ │ │ │ - return true; │ │ │ │ │ + var layers = this.findLayers(); │ │ │ │ │ + if (layers.length > 0) { │ │ │ │ │ + var infoLookup = {}; │ │ │ │ │ + var layer, idx; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + layer = layers[i]; │ │ │ │ │ + idx = OpenLayers.Util.indexOf(this.map.layers, layer); │ │ │ │ │ + infoLookup[idx] = layer.getFeatureInfo(lonLat); │ │ │ │ │ + } │ │ │ │ │ + this.callback(infoLookup, lonLat, evt.xy); │ │ │ │ │ } │ │ │ │ │ - return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ + * APIMethod: callback │ │ │ │ │ + * Function to be called when a mouse event corresponds with a location that │ │ │ │ │ + * includes data in one of the configured UTFGrid layers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * infoLookup - {Object} Keys of this object are layer indexes and can be │ │ │ │ │ + * used to resolve a layer in the map.layers array. The structure of │ │ │ │ │ + * the property values depend on the data included in the underlying │ │ │ │ │ + * UTFGrid and may be any valid JSON type. │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - var clickCallbacks = { │ │ │ │ │ - click: this.defaultClick, │ │ │ │ │ - dblclick: this.defaultDblClick │ │ │ │ │ - }; │ │ │ │ │ - var clickOptions = OpenLayers.Util.extend({ │ │ │ │ │ - "double": true, │ │ │ │ │ - stopDouble: true, │ │ │ │ │ - pixelTolerance: 2 │ │ │ │ │ - }, this.clickHandlerOptions); │ │ │ │ │ - this.handlers.click = new OpenLayers.Handler.Click( │ │ │ │ │ - this, clickCallbacks, clickOptions │ │ │ │ │ - ); │ │ │ │ │ - this.dragPan = new OpenLayers.Control.DragPan( │ │ │ │ │ - OpenLayers.Util.extend({ │ │ │ │ │ - map: this.map, │ │ │ │ │ - documentDrag: this.documentDrag │ │ │ │ │ - }, this.dragPanOptions) │ │ │ │ │ - ); │ │ │ │ │ - this.dragPan.draw(); │ │ │ │ │ - this.pinchZoom = new OpenLayers.Control.PinchZoom( │ │ │ │ │ - OpenLayers.Util.extend({ │ │ │ │ │ - map: this.map │ │ │ │ │ - }, this.pinchZoomOptions) │ │ │ │ │ - ); │ │ │ │ │ + callback: function(infoLookup) { │ │ │ │ │ + // to be provided in the constructor │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: defaultClick │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * Method: reset │ │ │ │ │ + * Calls the callback with null. │ │ │ │ │ */ │ │ │ │ │ - defaultClick: function(evt) { │ │ │ │ │ - if (evt.lastTouches && evt.lastTouches.length == 2) { │ │ │ │ │ - this.map.zoomOut(); │ │ │ │ │ - } │ │ │ │ │ + reset: function(evt) { │ │ │ │ │ + this.callback(null); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: defaultDblClick │ │ │ │ │ + * Method: findLayers │ │ │ │ │ + * Internal method to get the layers, independent of whether we are │ │ │ │ │ + * inspecting the map or using a client-provided array │ │ │ │ │ + * │ │ │ │ │ + * The default value of this.layers is null; this causes the │ │ │ │ │ + * findLayers method to return ALL UTFGrid layers encountered. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * None │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} Layers to handle on each event │ │ │ │ │ */ │ │ │ │ │ - defaultDblClick: function(evt) { │ │ │ │ │ - this.map.zoomTo(this.map.zoom + 1, evt.xy); │ │ │ │ │ + findLayers: function() { │ │ │ │ │ + var candidates = this.layers || this.map.layers; │ │ │ │ │ + var layers = []; │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = candidates[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.UTFGrid) { │ │ │ │ │ + layers.push(layer); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return layers; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.TouchNavigation" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.UTFGrid" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/LayerSwitcher.js │ │ │ │ │ + OpenLayers/Control/NavigationHistory.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ + * @requires OpenLayers/Control/Button.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.LayerSwitcher │ │ │ │ │ - * The LayerSwitcher control displays a table of contents for the map. This │ │ │ │ │ - * allows the user interface to switch between BaseLasyers and to show or hide │ │ │ │ │ - * Overlays. By default the switcher is shown minimized on the right edge of │ │ │ │ │ - * the map, the user may expand it by clicking on the handle. │ │ │ │ │ - * │ │ │ │ │ - * To create the LayerSwitcher outside of the map, pass the Id of a html div │ │ │ │ │ - * as the first argument to the constructor. │ │ │ │ │ + * Class: OpenLayers.Control.NavigationHistory │ │ │ │ │ + * A navigation history control. This is a meta-control, that creates two │ │ │ │ │ + * dependent controls: <previous> and <next>. Call the trigger method │ │ │ │ │ + * on the <previous> and <next> controls to restore previous and next │ │ │ │ │ + * history states. The previous and next controls will become active │ │ │ │ │ + * when there are available states to restore and will become deactive │ │ │ │ │ + * when there are no states to restore. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.LayerSwitcher = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Control.NavigationHistory = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: layerStates │ │ │ │ │ - * {Array(Object)} Basically a copy of the "state" of the map's layers │ │ │ │ │ - * the last time the control was drawn. We have this in order to avoid │ │ │ │ │ - * unnecessarily redrawing the control. │ │ │ │ │ + /** │ │ │ │ │ + * Property: type │ │ │ │ │ + * {String} Note that this control is not intended to be added directly │ │ │ │ │ + * to a control panel. Instead, add the sub-controls previous and │ │ │ │ │ + * next. These sub-controls are button type controls that activate │ │ │ │ │ + * and deactivate themselves. If this parent control is added to │ │ │ │ │ + * a panel, it will act as a toggle. │ │ │ │ │ */ │ │ │ │ │ - layerStates: null, │ │ │ │ │ - │ │ │ │ │ - // DOM Elements │ │ │ │ │ + type: OpenLayers.Control.TYPE_TOGGLE, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: layersDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * APIProperty: previous │ │ │ │ │ + * {<OpenLayers.Control>} A button type control whose trigger method restores │ │ │ │ │ + * the previous state managed by this control. │ │ │ │ │ */ │ │ │ │ │ - layersDiv: null, │ │ │ │ │ + previous: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: baseLayersDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * APIProperty: previousOptions │ │ │ │ │ + * {Object} Set this property on the options argument of the constructor │ │ │ │ │ + * to set optional properties on the <previous> control. │ │ │ │ │ */ │ │ │ │ │ - baseLayersDiv: null, │ │ │ │ │ + previousOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: baseLayers │ │ │ │ │ - * {Array(Object)} │ │ │ │ │ + * APIProperty: next │ │ │ │ │ + * {<OpenLayers.Control>} A button type control whose trigger method restores │ │ │ │ │ + * the next state managed by this control. │ │ │ │ │ */ │ │ │ │ │ - baseLayers: null, │ │ │ │ │ - │ │ │ │ │ + next: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: dataLbl │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * APIProperty: nextOptions │ │ │ │ │ + * {Object} Set this property on the options argument of the constructor │ │ │ │ │ + * to set optional properties on the <next> control. │ │ │ │ │ */ │ │ │ │ │ - dataLbl: null, │ │ │ │ │ + nextOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: dataLayersDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * APIProperty: limit │ │ │ │ │ + * {Integer} Optional limit on the number of history items to retain. If │ │ │ │ │ + * null, there is no limit. Default is 50. │ │ │ │ │ */ │ │ │ │ │ - dataLayersDiv: null, │ │ │ │ │ + limit: 50, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: dataLayers │ │ │ │ │ - * {Array(Object)} │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ - dataLayers: null, │ │ │ │ │ - │ │ │ │ │ + autoActivate: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: minimizeDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Property: clearOnDeactivate │ │ │ │ │ + * {Boolean} Clear the history when the control is deactivated. Default │ │ │ │ │ + * is false. │ │ │ │ │ */ │ │ │ │ │ - minimizeDiv: null, │ │ │ │ │ + clearOnDeactivate: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: maximizeDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Property: registry │ │ │ │ │ + * {Object} An object with keys corresponding to event types. Values │ │ │ │ │ + * are functions that return an object representing the current state. │ │ │ │ │ */ │ │ │ │ │ - maximizeDiv: null, │ │ │ │ │ + registry: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: ascending │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * Property: nextStack │ │ │ │ │ + * {Array} Array of items in the history. │ │ │ │ │ */ │ │ │ │ │ - ascending: true, │ │ │ │ │ + nextStack: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.LayerSwitcher │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * Property: previousStack │ │ │ │ │ + * {Array} List of items in the history. First item represents the current │ │ │ │ │ + * state. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.layerStates = []; │ │ │ │ │ - }, │ │ │ │ │ + previousStack: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ + * Property: listeners │ │ │ │ │ + * {Object} An object containing properties corresponding to event types. │ │ │ │ │ + * This object is used to configure the control and is modified on │ │ │ │ │ + * construction. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - │ │ │ │ │ - //clear out layers info and unregister their events │ │ │ │ │ - this.clearLayersArray("base"); │ │ │ │ │ - this.clearLayersArray("data"); │ │ │ │ │ - │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - buttonclick: this.onButtonClick, │ │ │ │ │ - addlayer: this.redraw, │ │ │ │ │ - changelayer: this.redraw, │ │ │ │ │ - removelayer: this.redraw, │ │ │ │ │ - changebaselayer: this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + listeners: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * │ │ │ │ │ - * Properties: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * Property: restoring │ │ │ │ │ + * {Boolean} Currently restoring a history state. This is set to true │ │ │ │ │ + * before calling restore and set to false after restore returns. │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - addlayer: this.redraw, │ │ │ │ │ - changelayer: this.redraw, │ │ │ │ │ - removelayer: this.redraw, │ │ │ │ │ - changebaselayer: this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - if (this.outsideViewport) { │ │ │ │ │ - this.events.attachToElement(this.div); │ │ │ │ │ - this.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ - } else { │ │ │ │ │ - this.map.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + restoring: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A reference to the DIV DOMElement containing the │ │ │ │ │ - * switcher tabs. │ │ │ │ │ + * Constructor: OpenLayers.Control.NavigationHistory │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be used │ │ │ │ │ + * to extend the control. │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this); │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ │ │ │ │ │ - // create layout divs │ │ │ │ │ - this.loadContents(); │ │ │ │ │ + this.registry = OpenLayers.Util.extend({ │ │ │ │ │ + "moveend": this.getState │ │ │ │ │ + }, this.registry); │ │ │ │ │ │ │ │ │ │ - // set mode to minimize │ │ │ │ │ - if (!this.outsideViewport) { │ │ │ │ │ - this.minimizeControl(); │ │ │ │ │ - } │ │ │ │ │ + var previousOptions = { │ │ │ │ │ + trigger: OpenLayers.Function.bind(this.previousTrigger, this), │ │ │ │ │ + displayClass: this.displayClass + " " + this.displayClass + "Previous" │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Util.extend(previousOptions, this.previousOptions); │ │ │ │ │ + this.previous = new OpenLayers.Control.Button(previousOptions); │ │ │ │ │ │ │ │ │ │ - // populate div with current info │ │ │ │ │ - this.redraw(); │ │ │ │ │ + var nextOptions = { │ │ │ │ │ + trigger: OpenLayers.Function.bind(this.nextTrigger, this), │ │ │ │ │ + displayClass: this.displayClass + " " + this.displayClass + "Next" │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Util.extend(nextOptions, this.nextOptions); │ │ │ │ │ + this.next = new OpenLayers.Control.Button(nextOptions); │ │ │ │ │ │ │ │ │ │ - return this.div; │ │ │ │ │ + this.clear(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onButtonClick │ │ │ │ │ + * Method: onPreviousChange │ │ │ │ │ + * Called when the previous history stack changes. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * state - {Object} An object representing the state to be restored │ │ │ │ │ + * if previous is triggered again or null if no previous states remain. │ │ │ │ │ + * length - {Integer} The number of remaining previous states that can │ │ │ │ │ + * be restored. │ │ │ │ │ */ │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - var button = evt.buttonElement; │ │ │ │ │ - if (button === this.minimizeDiv) { │ │ │ │ │ - this.minimizeControl(); │ │ │ │ │ - } else if (button === this.maximizeDiv) { │ │ │ │ │ - this.maximizeControl(); │ │ │ │ │ - } else if (button._layerSwitcher === this.id) { │ │ │ │ │ - if (button["for"]) { │ │ │ │ │ - button = document.getElementById(button["for"]); │ │ │ │ │ - } │ │ │ │ │ - if (!button.disabled) { │ │ │ │ │ - if (button.type == "radio") { │ │ │ │ │ - button.checked = true; │ │ │ │ │ - this.map.setBaseLayer(this.map.getLayer(button._layer)); │ │ │ │ │ - } else { │ │ │ │ │ - button.checked = !button.checked; │ │ │ │ │ - this.updateMap(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + onPreviousChange: function(state, length) { │ │ │ │ │ + if (state && !this.previous.active) { │ │ │ │ │ + this.previous.activate(); │ │ │ │ │ + } else if (!state && this.previous.active) { │ │ │ │ │ + this.previous.deactivate(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clearLayersArray │ │ │ │ │ - * User specifies either "base" or "data". we then clear all the │ │ │ │ │ - * corresponding listeners, the div, and reinitialize a new array. │ │ │ │ │ + * Method: onNextChange │ │ │ │ │ + * Called when the next history stack changes. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layersType - {String} │ │ │ │ │ + * state - {Object} An object representing the state to be restored │ │ │ │ │ + * if next is triggered again or null if no next states remain. │ │ │ │ │ + * length - {Integer} The number of remaining next states that can │ │ │ │ │ + * be restored. │ │ │ │ │ */ │ │ │ │ │ - clearLayersArray: function(layersType) { │ │ │ │ │ - this[layersType + "LayersDiv"].innerHTML = ""; │ │ │ │ │ - this[layersType + "Layers"] = []; │ │ │ │ │ + onNextChange: function(state, length) { │ │ │ │ │ + if (state && !this.next.active) { │ │ │ │ │ + this.next.activate(); │ │ │ │ │ + } else if (!state && this.next.active) { │ │ │ │ │ + this.next.deactivate(); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: checkRedraw │ │ │ │ │ - * Checks if the layer state has changed since the last redraw() call. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The layer state changed since the last redraw() call. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Destroy the control. │ │ │ │ │ */ │ │ │ │ │ - checkRedraw: function() { │ │ │ │ │ - if (!this.layerStates.length || │ │ │ │ │ - (this.map.layers.length != this.layerStates.length)) { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - for (var i = 0, len = this.layerStates.length; i < len; i++) { │ │ │ │ │ - var layerState = this.layerStates[i]; │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if ((layerState.name != layer.name) || │ │ │ │ │ - (layerState.inRange != layer.inRange) || │ │ │ │ │ - (layerState.id != layer.id) || │ │ │ │ │ - (layerState.visibility != layer.visibility)) { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ + destroy: function() { │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this); │ │ │ │ │ + this.previous.destroy(); │ │ │ │ │ + this.next.destroy(); │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + for (var prop in this) { │ │ │ │ │ + this[prop] = null; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: redraw │ │ │ │ │ - * Goes through and takes the current state of the Map and rebuilds the │ │ │ │ │ - * control to display that state. Groups base layers into a │ │ │ │ │ - * radio-button group and lists each data layer with a checkbox. │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the control and <previous> and <next> child │ │ │ │ │ + * controls. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - redraw: function() { │ │ │ │ │ - //if the state hasn't changed since last redraw, no need │ │ │ │ │ - // to do anything. Just return the existing div. │ │ │ │ │ - if (!this.checkRedraw()) { │ │ │ │ │ - return this.div; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //clear out previous layers │ │ │ │ │ - this.clearLayersArray("base"); │ │ │ │ │ - this.clearLayersArray("data"); │ │ │ │ │ - │ │ │ │ │ - var containsOverlays = false; │ │ │ │ │ - var containsBaseLayers = false; │ │ │ │ │ - │ │ │ │ │ - // Save state -- for checking layer if the map state changed. │ │ │ │ │ - // We save this before redrawing, because in the process of redrawing │ │ │ │ │ - // we will trigger more visibility changes, and we want to not redraw │ │ │ │ │ - // and enter an infinite loop. │ │ │ │ │ - var len = this.map.layers.length; │ │ │ │ │ - this.layerStates = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - this.layerStates[i] = { │ │ │ │ │ - 'name': layer.name, │ │ │ │ │ - 'visibility': layer.visibility, │ │ │ │ │ - 'inRange': layer.inRange, │ │ │ │ │ - 'id': layer.id │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var layers = this.map.layers.slice(); │ │ │ │ │ - if (!this.ascending) { │ │ │ │ │ - layers.reverse(); │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - var layer = layers[i]; │ │ │ │ │ - var baseLayer = layer.isBaseLayer; │ │ │ │ │ - │ │ │ │ │ - if (layer.displayInLayerSwitcher) { │ │ │ │ │ - │ │ │ │ │ - if (baseLayer) { │ │ │ │ │ - containsBaseLayers = true; │ │ │ │ │ - } else { │ │ │ │ │ - containsOverlays = true; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // only check a baselayer if it is *the* baselayer, check data │ │ │ │ │ - // layers if they are visible │ │ │ │ │ - var checked = (baseLayer) ? (layer == this.map.baseLayer) : │ │ │ │ │ - layer.getVisibility(); │ │ │ │ │ - │ │ │ │ │ - // create input element │ │ │ │ │ - var inputElem = document.createElement("input"), │ │ │ │ │ - // The input shall have an id attribute so we can use │ │ │ │ │ - // labels to interact with them. │ │ │ │ │ - inputId = OpenLayers.Util.createUniqueID( │ │ │ │ │ - this.id + "_input_" │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - inputElem.id = inputId; │ │ │ │ │ - inputElem.name = (baseLayer) ? this.id + "_baseLayers" : layer.name; │ │ │ │ │ - inputElem.type = (baseLayer) ? "radio" : "checkbox"; │ │ │ │ │ - inputElem.value = layer.name; │ │ │ │ │ - inputElem.checked = checked; │ │ │ │ │ - inputElem.defaultChecked = checked; │ │ │ │ │ - inputElem.className = "olButton"; │ │ │ │ │ - inputElem._layer = layer.id; │ │ │ │ │ - inputElem._layerSwitcher = this.id; │ │ │ │ │ - │ │ │ │ │ - if (!baseLayer && !layer.inRange) { │ │ │ │ │ - inputElem.disabled = true; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // create span │ │ │ │ │ - var labelSpan = document.createElement("label"); │ │ │ │ │ - // this isn't the DOM attribute 'for', but an arbitrary name we │ │ │ │ │ - // use to find the appropriate input element in <onButtonClick> │ │ │ │ │ - labelSpan["for"] = inputElem.id; │ │ │ │ │ - OpenLayers.Element.addClass(labelSpan, "labelSpan olButton"); │ │ │ │ │ - labelSpan._layer = layer.id; │ │ │ │ │ - labelSpan._layerSwitcher = this.id; │ │ │ │ │ - if (!baseLayer && !layer.inRange) { │ │ │ │ │ - labelSpan.style.color = "gray"; │ │ │ │ │ - } │ │ │ │ │ - labelSpan.innerHTML = layer.name; │ │ │ │ │ - labelSpan.style.verticalAlign = (baseLayer) ? "bottom" : │ │ │ │ │ - "baseline"; │ │ │ │ │ - // create line break │ │ │ │ │ - var br = document.createElement("br"); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - var groupArray = (baseLayer) ? this.baseLayers : │ │ │ │ │ - this.dataLayers; │ │ │ │ │ - groupArray.push({ │ │ │ │ │ - 'layer': layer, │ │ │ │ │ - 'inputElem': inputElem, │ │ │ │ │ - 'labelSpan': labelSpan │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - var groupDiv = (baseLayer) ? this.baseLayersDiv : │ │ │ │ │ - this.dataLayersDiv; │ │ │ │ │ - groupDiv.appendChild(inputElem); │ │ │ │ │ - groupDiv.appendChild(labelSpan); │ │ │ │ │ - groupDiv.appendChild(br); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // if no overlays, dont display the overlay label │ │ │ │ │ - this.dataLbl.style.display = (containsOverlays) ? "" : "none"; │ │ │ │ │ - │ │ │ │ │ - // if no baselayers, dont display the baselayer label │ │ │ │ │ - this.baseLbl.style.display = (containsBaseLayers) ? "" : "none"; │ │ │ │ │ - │ │ │ │ │ - return this.div; │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.map = map; │ │ │ │ │ + this.next.setMap(map); │ │ │ │ │ + this.previous.setMap(map); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: updateMap │ │ │ │ │ - * Cycles through the loaded data and base layer input arrays and makes │ │ │ │ │ - * the necessary calls to the Map object such that that the map's │ │ │ │ │ - * visual state corresponds to what the user has selected in │ │ │ │ │ - * the control. │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Called when the control is added to the map. │ │ │ │ │ */ │ │ │ │ │ - updateMap: function() { │ │ │ │ │ - │ │ │ │ │ - // set the newly selected base layer │ │ │ │ │ - for (var i = 0, len = this.baseLayers.length; i < len; i++) { │ │ │ │ │ - var layerEntry = this.baseLayers[i]; │ │ │ │ │ - if (layerEntry.inputElem.checked) { │ │ │ │ │ - this.map.setBaseLayer(layerEntry.layer, false); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // set the correct visibilities for the overlays │ │ │ │ │ - for (var i = 0, len = this.dataLayers.length; i < len; i++) { │ │ │ │ │ - var layerEntry = this.dataLayers[i]; │ │ │ │ │ - layerEntry.layer.setVisibility(layerEntry.inputElem.checked); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + this.next.draw(); │ │ │ │ │ + this.previous.draw(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: maximizeControl │ │ │ │ │ - * Set up the labels and divs for the control │ │ │ │ │ + * Method: previousTrigger │ │ │ │ │ + * Restore the previous state. If no items are in the previous history │ │ │ │ │ + * stack, this has no effect. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * e - {Event} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Item representing state that was restored. Undefined if no │ │ │ │ │ + * items are in the previous history stack. │ │ │ │ │ */ │ │ │ │ │ - maximizeControl: function(e) { │ │ │ │ │ - │ │ │ │ │ - // set the div's width and height to empty values, so │ │ │ │ │ - // the div dimensions can be controlled by CSS │ │ │ │ │ - this.div.style.width = ""; │ │ │ │ │ - this.div.style.height = ""; │ │ │ │ │ - │ │ │ │ │ - this.showControls(false); │ │ │ │ │ - │ │ │ │ │ - if (e != null) { │ │ │ │ │ - OpenLayers.Event.stop(e); │ │ │ │ │ + previousTrigger: function() { │ │ │ │ │ + var current = this.previousStack.shift(); │ │ │ │ │ + var state = this.previousStack.shift(); │ │ │ │ │ + if (state != undefined) { │ │ │ │ │ + this.nextStack.unshift(current); │ │ │ │ │ + this.previousStack.unshift(state); │ │ │ │ │ + this.restoring = true; │ │ │ │ │ + this.restore(state); │ │ │ │ │ + this.restoring = false; │ │ │ │ │ + this.onNextChange(this.nextStack[0], this.nextStack.length); │ │ │ │ │ + this.onPreviousChange( │ │ │ │ │ + this.previousStack[1], this.previousStack.length - 1 │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + this.previousStack.unshift(current); │ │ │ │ │ } │ │ │ │ │ + return state; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: minimizeControl │ │ │ │ │ - * Hide all the contents of the control, shrink the size, │ │ │ │ │ - * add the maximize icon │ │ │ │ │ + * APIMethod: nextTrigger │ │ │ │ │ + * Restore the next state. If no items are in the next history │ │ │ │ │ + * stack, this has no effect. The next history stack is populated │ │ │ │ │ + * as states are restored from the previous history stack. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * e - {Event} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Item representing state that was restored. Undefined if no │ │ │ │ │ + * items are in the next history stack. │ │ │ │ │ */ │ │ │ │ │ - minimizeControl: function(e) { │ │ │ │ │ - │ │ │ │ │ - // to minimize the control we set its div's width │ │ │ │ │ - // and height to 0px, we cannot just set "display" │ │ │ │ │ - // to "none" because it would hide the maximize │ │ │ │ │ - // div │ │ │ │ │ - this.div.style.width = "0px"; │ │ │ │ │ - this.div.style.height = "0px"; │ │ │ │ │ - │ │ │ │ │ - this.showControls(true); │ │ │ │ │ - │ │ │ │ │ - if (e != null) { │ │ │ │ │ - OpenLayers.Event.stop(e); │ │ │ │ │ + nextTrigger: function() { │ │ │ │ │ + var state = this.nextStack.shift(); │ │ │ │ │ + if (state != undefined) { │ │ │ │ │ + this.previousStack.unshift(state); │ │ │ │ │ + this.restoring = true; │ │ │ │ │ + this.restore(state); │ │ │ │ │ + this.restoring = false; │ │ │ │ │ + this.onNextChange(this.nextStack[0], this.nextStack.length); │ │ │ │ │ + this.onPreviousChange( │ │ │ │ │ + this.previousStack[1], this.previousStack.length - 1 │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ + return state; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: showControls │ │ │ │ │ - * Hide/Show all LayerSwitcher controls depending on whether we are │ │ │ │ │ - * minimized or not │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * minimize - {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - showControls: function(minimize) { │ │ │ │ │ - │ │ │ │ │ - this.maximizeDiv.style.display = minimize ? "" : "none"; │ │ │ │ │ - this.minimizeDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ - │ │ │ │ │ - this.layersDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: loadContents │ │ │ │ │ - * Set up the labels and divs for the control │ │ │ │ │ + * APIMethod: clear │ │ │ │ │ + * Clear history. │ │ │ │ │ */ │ │ │ │ │ - loadContents: function() { │ │ │ │ │ - │ │ │ │ │ - // layers list div │ │ │ │ │ - this.layersDiv = document.createElement("div"); │ │ │ │ │ - this.layersDiv.id = this.id + "_layersDiv"; │ │ │ │ │ - OpenLayers.Element.addClass(this.layersDiv, "layersDiv"); │ │ │ │ │ - │ │ │ │ │ - this.baseLbl = document.createElement("div"); │ │ │ │ │ - this.baseLbl.innerHTML = OpenLayers.i18n("Base Layer"); │ │ │ │ │ - OpenLayers.Element.addClass(this.baseLbl, "baseLbl"); │ │ │ │ │ - │ │ │ │ │ - this.baseLayersDiv = document.createElement("div"); │ │ │ │ │ - OpenLayers.Element.addClass(this.baseLayersDiv, "baseLayersDiv"); │ │ │ │ │ - │ │ │ │ │ - this.dataLbl = document.createElement("div"); │ │ │ │ │ - this.dataLbl.innerHTML = OpenLayers.i18n("Overlays"); │ │ │ │ │ - OpenLayers.Element.addClass(this.dataLbl, "dataLbl"); │ │ │ │ │ - │ │ │ │ │ - this.dataLayersDiv = document.createElement("div"); │ │ │ │ │ - OpenLayers.Element.addClass(this.dataLayersDiv, "dataLayersDiv"); │ │ │ │ │ - │ │ │ │ │ - if (this.ascending) { │ │ │ │ │ - this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ - } else { │ │ │ │ │ - this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.div.appendChild(this.layersDiv); │ │ │ │ │ - │ │ │ │ │ - // maximize button div │ │ │ │ │ - var img = OpenLayers.Util.getImageLocation('layer-switcher-maximize.png'); │ │ │ │ │ - this.maximizeDiv = OpenLayers.Util.createAlphaImageDiv( │ │ │ │ │ - "OpenLayers_Control_MaximizeDiv", │ │ │ │ │ - null, │ │ │ │ │ - null, │ │ │ │ │ - img, │ │ │ │ │ - "absolute"); │ │ │ │ │ - OpenLayers.Element.addClass(this.maximizeDiv, "maximizeDiv olButton"); │ │ │ │ │ - this.maximizeDiv.style.display = "none"; │ │ │ │ │ - │ │ │ │ │ - this.div.appendChild(this.maximizeDiv); │ │ │ │ │ - │ │ │ │ │ - // minimize button div │ │ │ │ │ - var img = OpenLayers.Util.getImageLocation('layer-switcher-minimize.png'); │ │ │ │ │ - this.minimizeDiv = OpenLayers.Util.createAlphaImageDiv( │ │ │ │ │ - "OpenLayers_Control_MinimizeDiv", │ │ │ │ │ - null, │ │ │ │ │ - null, │ │ │ │ │ - img, │ │ │ │ │ - "absolute"); │ │ │ │ │ - OpenLayers.Element.addClass(this.minimizeDiv, "minimizeDiv olButton"); │ │ │ │ │ - this.minimizeDiv.style.display = "none"; │ │ │ │ │ - │ │ │ │ │ - this.div.appendChild(this.minimizeDiv); │ │ │ │ │ + clear: function() { │ │ │ │ │ + this.previousStack = []; │ │ │ │ │ + this.previous.deactivate(); │ │ │ │ │ + this.nextStack = []; │ │ │ │ │ + this.next.deactivate(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.LayerSwitcher" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Graticule.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - * @requires OpenLayers/Rule.js │ │ │ │ │ - * @requires OpenLayers/StyleMap.js │ │ │ │ │ - * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Graticule │ │ │ │ │ - * The Graticule displays a grid of latitude/longitude lines reprojected on │ │ │ │ │ - * the map. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Graticule = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ - */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: intervals │ │ │ │ │ - * {Array(Float)} A list of possible graticule widths in degrees. │ │ │ │ │ - */ │ │ │ │ │ - intervals: [45, 30, 20, 10, 5, 2, 1, │ │ │ │ │ - 0.5, 0.2, 0.1, 0.05, 0.01, │ │ │ │ │ - 0.005, 0.002, 0.001 │ │ │ │ │ - ], │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: displayInLayerSwitcher │ │ │ │ │ - * {Boolean} Allows the Graticule control to be switched on and off by │ │ │ │ │ - * LayerSwitcher control. Defaults is true. │ │ │ │ │ - */ │ │ │ │ │ - displayInLayerSwitcher: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: visible │ │ │ │ │ - * {Boolean} should the graticule be initially visible (default=true) │ │ │ │ │ - */ │ │ │ │ │ - visible: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: numPoints │ │ │ │ │ - * {Integer} The number of points to use in each graticule line. Higher │ │ │ │ │ - * numbers result in a smoother curve for projected maps │ │ │ │ │ - */ │ │ │ │ │ - numPoints: 50, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: targetSize │ │ │ │ │ - * {Integer} The maximum size of the grid in pixels on the map │ │ │ │ │ - */ │ │ │ │ │ - targetSize: 200, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: layerName │ │ │ │ │ - * {String} The name to be displayed in the layer switcher, default is set │ │ │ │ │ - * by {<OpenLayers.Lang>}. │ │ │ │ │ - */ │ │ │ │ │ - layerName: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: labelled │ │ │ │ │ - * {Boolean} Should the graticule lines be labelled?. default=true │ │ │ │ │ - */ │ │ │ │ │ - labelled: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: labelFormat │ │ │ │ │ - * {String} the format of the labels, default = 'dm'. See │ │ │ │ │ - * <OpenLayers.Util.getFormattedLonLat> for other options. │ │ │ │ │ - */ │ │ │ │ │ - labelFormat: 'dm', │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: lineSymbolizer │ │ │ │ │ - * {symbolizer} the symbolizer used to render lines │ │ │ │ │ + * Method: getState │ │ │ │ │ + * Get the current state and return it. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object representing the current state. │ │ │ │ │ */ │ │ │ │ │ - lineSymbolizer: { │ │ │ │ │ - strokeColor: "#333", │ │ │ │ │ - strokeWidth: 1, │ │ │ │ │ - strokeOpacity: 0.5 │ │ │ │ │ + getState: function() { │ │ │ │ │ + return { │ │ │ │ │ + center: this.map.getCenter(), │ │ │ │ │ + resolution: this.map.getResolution(), │ │ │ │ │ + projection: this.map.getProjectionObject(), │ │ │ │ │ + units: this.map.getProjectionObject().getUnits() || │ │ │ │ │ + this.map.units || this.map.baseLayer.units │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: labelSymbolizer │ │ │ │ │ - * {symbolizer} the symbolizer used to render labels │ │ │ │ │ - */ │ │ │ │ │ - labelSymbolizer: {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: gratLayer │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} vector layer used to draw the graticule on │ │ │ │ │ - */ │ │ │ │ │ - gratLayer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Graticule │ │ │ │ │ - * Create a new graticule control to display a grid of latitude longitude │ │ │ │ │ - * lines. │ │ │ │ │ - * │ │ │ │ │ + * Method: restore │ │ │ │ │ + * Update the state with the given object. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be used │ │ │ │ │ - * to extend the control. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.layerName = options.layerName || OpenLayers.i18n("Graticule"); │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - │ │ │ │ │ - this.labelSymbolizer.stroke = false; │ │ │ │ │ - this.labelSymbolizer.fill = false; │ │ │ │ │ - this.labelSymbolizer.label = "${label}"; │ │ │ │ │ - this.labelSymbolizer.labelAlign = "${labelAlign}"; │ │ │ │ │ - this.labelSymbolizer.labelXOffset = "${xOffset}"; │ │ │ │ │ - this.labelSymbolizer.labelYOffset = "${yOffset}"; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ + * state - {Object} An object representing the state to restore. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - if (this.gratLayer) { │ │ │ │ │ - this.gratLayer.destroy(); │ │ │ │ │ - this.gratLayer = null; │ │ │ │ │ + restore: function(state) { │ │ │ │ │ + var center, zoom; │ │ │ │ │ + if (this.map.getProjectionObject() == state.projection) { │ │ │ │ │ + zoom = this.map.getZoomForResolution(state.resolution); │ │ │ │ │ + center = state.center; │ │ │ │ │ + } else { │ │ │ │ │ + center = state.center.clone(); │ │ │ │ │ + center.transform(state.projection, this.map.getProjectionObject()); │ │ │ │ │ + var sourceUnits = state.units; │ │ │ │ │ + var targetUnits = this.map.getProjectionObject().getUnits() || │ │ │ │ │ + this.map.units || this.map.baseLayer.units; │ │ │ │ │ + var resolutionFactor = sourceUnits && targetUnits ? │ │ │ │ │ + OpenLayers.INCHES_PER_UNIT[sourceUnits] / OpenLayers.INCHES_PER_UNIT[targetUnits] : 1; │ │ │ │ │ + zoom = this.map.getZoomForResolution(resolutionFactor * state.resolution); │ │ │ │ │ } │ │ │ │ │ + this.map.setCenter(center, zoom); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * │ │ │ │ │ - * initializes the graticule layer and does the initial update │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Method: setListeners │ │ │ │ │ + * Sets functions to be registered in the listeners object. │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (!this.gratLayer) { │ │ │ │ │ - var gratStyle = new OpenLayers.Style({}, { │ │ │ │ │ - rules: [new OpenLayers.Rule({ │ │ │ │ │ - 'symbolizer': { │ │ │ │ │ - "Point": this.labelSymbolizer, │ │ │ │ │ - "Line": this.lineSymbolizer │ │ │ │ │ + setListeners: function() { │ │ │ │ │ + this.listeners = {}; │ │ │ │ │ + for (var type in this.registry) { │ │ │ │ │ + this.listeners[type] = OpenLayers.Function.bind(function() { │ │ │ │ │ + if (!this.restoring) { │ │ │ │ │ + var state = this.registry[type].apply(this, arguments); │ │ │ │ │ + this.previousStack.unshift(state); │ │ │ │ │ + if (this.previousStack.length > 1) { │ │ │ │ │ + this.onPreviousChange( │ │ │ │ │ + this.previousStack[1], this.previousStack.length - 1 │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - })] │ │ │ │ │ - }); │ │ │ │ │ - this.gratLayer = new OpenLayers.Layer.Vector(this.layerName, { │ │ │ │ │ - styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ - 'default': gratStyle │ │ │ │ │ - }), │ │ │ │ │ - visibility: this.visible, │ │ │ │ │ - displayInLayerSwitcher: this.displayInLayerSwitcher │ │ │ │ │ - }); │ │ │ │ │ + if (this.previousStack.length > (this.limit + 1)) { │ │ │ │ │ + this.previousStack.pop(); │ │ │ │ │ + } │ │ │ │ │ + if (this.nextStack.length > 0) { │ │ │ │ │ + this.nextStack = []; │ │ │ │ │ + this.onNextChange(null, 0); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ + }, this); │ │ │ │ │ } │ │ │ │ │ - return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIMethod: activate │ │ │ │ │ + * Activate the control. This registers any listeners. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Control successfully activated. │ │ │ │ │ */ │ │ │ │ │ activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.map.addLayer(this.gratLayer); │ │ │ │ │ - this.map.events.register('moveend', this, this.update); │ │ │ │ │ - this.update(); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this)) { │ │ │ │ │ + if (this.listeners == null) { │ │ │ │ │ + this.setListeners(); │ │ │ │ │ + } │ │ │ │ │ + for (var type in this.listeners) { │ │ │ │ │ + this.map.events.register(type, this, this.listeners[type]); │ │ │ │ │ + } │ │ │ │ │ + activated = true; │ │ │ │ │ + if (this.previousStack.length == 0) { │ │ │ │ │ + this.initStack(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ + * Method: initStack │ │ │ │ │ + * Called after the control is activated if the previous history stack is │ │ │ │ │ + * empty. │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.map.events.unregister('moveend', this, this.update); │ │ │ │ │ - this.map.removeLayer(this.gratLayer); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ + initStack: function() { │ │ │ │ │ + if (this.map.getCenter()) { │ │ │ │ │ + this.listeners.moveend(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: update │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the control. This unregisters any listeners. │ │ │ │ │ * │ │ │ │ │ - * calculates the grid to be displayed and actually draws it │ │ │ │ │ - * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * {Boolean} Control successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ - update: function() { │ │ │ │ │ - //wait for the map to be initialized before proceeding │ │ │ │ │ - var mapBounds = this.map.getExtent(); │ │ │ │ │ - if (!mapBounds) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //clear out the old grid │ │ │ │ │ - this.gratLayer.destroyFeatures(); │ │ │ │ │ - │ │ │ │ │ - //get the projection objects required │ │ │ │ │ - var llProj = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - var mapProj = this.map.getProjectionObject(); │ │ │ │ │ - var mapRes = this.map.getResolution(); │ │ │ │ │ - │ │ │ │ │ - //if the map is in lon/lat, then the lines are straight and only one │ │ │ │ │ - //point is required │ │ │ │ │ - if (mapProj.proj && mapProj.proj.projName == "longlat") { │ │ │ │ │ - this.numPoints = 1; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //get the map center in EPSG:4326 │ │ │ │ │ - var mapCenter = this.map.getCenter(); //lon and lat here are really map x and y │ │ │ │ │ - var mapCenterLL = new OpenLayers.Pixel(mapCenter.lon, mapCenter.lat); │ │ │ │ │ - OpenLayers.Projection.transform(mapCenterLL, mapProj, llProj); │ │ │ │ │ - │ │ │ │ │ - /* This block of code determines the lon/lat interval to use for the │ │ │ │ │ - * grid by calculating the diagonal size of one grid cell at the map │ │ │ │ │ - * center. Iterates through the intervals array until the diagonal │ │ │ │ │ - * length is less than the targetSize option. │ │ │ │ │ - */ │ │ │ │ │ - //find lat/lon interval that results in a grid of less than the target size │ │ │ │ │ - var testSq = this.targetSize * mapRes; │ │ │ │ │ - testSq *= testSq; //compare squares rather than doing a square root to save time │ │ │ │ │ - var llInterval; │ │ │ │ │ - for (var i = 0; i < this.intervals.length; ++i) { │ │ │ │ │ - llInterval = this.intervals[i]; //could do this for both x and y?? │ │ │ │ │ - var delta = llInterval / 2; │ │ │ │ │ - var p1 = mapCenterLL.offset({ │ │ │ │ │ - x: -delta, │ │ │ │ │ - y: -delta │ │ │ │ │ - }); //test coords in EPSG:4326 space │ │ │ │ │ - var p2 = mapCenterLL.offset({ │ │ │ │ │ - x: delta, │ │ │ │ │ - y: delta │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Projection.transform(p1, llProj, mapProj); // convert them back to map projection │ │ │ │ │ - OpenLayers.Projection.transform(p2, llProj, mapProj); │ │ │ │ │ - var distSq = (p1.x - p2.x) * (p1.x - p2.x) + (p1.y - p2.y) * (p1.y - p2.y); │ │ │ │ │ - if (distSq <= testSq) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - //alert(llInterval); │ │ │ │ │ - │ │ │ │ │ - //round the LL center to an even number based on the interval │ │ │ │ │ - mapCenterLL.x = Math.floor(mapCenterLL.x / llInterval) * llInterval; │ │ │ │ │ - mapCenterLL.y = Math.floor(mapCenterLL.y / llInterval) * llInterval; │ │ │ │ │ - //TODO adjust for minutses/seconds? │ │ │ │ │ - │ │ │ │ │ - /* The following 2 blocks calculate the nodes of the grid along a │ │ │ │ │ - * line of constant longitude (then latitiude) running through the │ │ │ │ │ - * center of the map until it reaches the map edge. The calculation │ │ │ │ │ - * goes from the center in both directions to the edge. │ │ │ │ │ - */ │ │ │ │ │ - //get the central longitude line, increment the latitude │ │ │ │ │ - var iter = 0; │ │ │ │ │ - var centerLonPoints = [mapCenterLL.clone()]; │ │ │ │ │ - var newPoint = mapCenterLL.clone(); │ │ │ │ │ - var mapXY; │ │ │ │ │ - do { │ │ │ │ │ - newPoint = newPoint.offset({ │ │ │ │ │ - x: 0, │ │ │ │ │ - y: llInterval │ │ │ │ │ - }); │ │ │ │ │ - mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ - centerLonPoints.unshift(newPoint); │ │ │ │ │ - } while (mapBounds.containsPixel(mapXY) && ++iter < 1000); │ │ │ │ │ - newPoint = mapCenterLL.clone(); │ │ │ │ │ - do { │ │ │ │ │ - newPoint = newPoint.offset({ │ │ │ │ │ - x: 0, │ │ │ │ │ - y: -llInterval │ │ │ │ │ - }); │ │ │ │ │ - mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ - centerLonPoints.push(newPoint); │ │ │ │ │ - } while (mapBounds.containsPixel(mapXY) && ++iter < 1000); │ │ │ │ │ - │ │ │ │ │ - //get the central latitude line, increment the longitude │ │ │ │ │ - iter = 0; │ │ │ │ │ - var centerLatPoints = [mapCenterLL.clone()]; │ │ │ │ │ - newPoint = mapCenterLL.clone(); │ │ │ │ │ - do { │ │ │ │ │ - newPoint = newPoint.offset({ │ │ │ │ │ - x: -llInterval, │ │ │ │ │ - y: 0 │ │ │ │ │ - }); │ │ │ │ │ - mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ - centerLatPoints.unshift(newPoint); │ │ │ │ │ - } while (mapBounds.containsPixel(mapXY) && ++iter < 1000); │ │ │ │ │ - newPoint = mapCenterLL.clone(); │ │ │ │ │ - do { │ │ │ │ │ - newPoint = newPoint.offset({ │ │ │ │ │ - x: llInterval, │ │ │ │ │ - y: 0 │ │ │ │ │ - }); │ │ │ │ │ - mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ - centerLatPoints.push(newPoint); │ │ │ │ │ - } while (mapBounds.containsPixel(mapXY) && ++iter < 1000); │ │ │ │ │ - │ │ │ │ │ - //now generate a line for each node in the central lat and lon lines │ │ │ │ │ - //first loop over constant longitude │ │ │ │ │ - var lines = []; │ │ │ │ │ - for (var i = 0; i < centerLatPoints.length; ++i) { │ │ │ │ │ - var lon = centerLatPoints[i].x; │ │ │ │ │ - var pointList = []; │ │ │ │ │ - var labelPoint = null; │ │ │ │ │ - var latEnd = Math.min(centerLonPoints[0].y, 90); │ │ │ │ │ - var latStart = Math.max(centerLonPoints[centerLonPoints.length - 1].y, -90); │ │ │ │ │ - var latDelta = (latEnd - latStart) / this.numPoints; │ │ │ │ │ - var lat = latStart; │ │ │ │ │ - for (var j = 0; j <= this.numPoints; ++j) { │ │ │ │ │ - var gridPoint = new OpenLayers.Geometry.Point(lon, lat); │ │ │ │ │ - gridPoint.transform(llProj, mapProj); │ │ │ │ │ - pointList.push(gridPoint); │ │ │ │ │ - lat += latDelta; │ │ │ │ │ - if (gridPoint.y >= mapBounds.bottom && !labelPoint) { │ │ │ │ │ - labelPoint = gridPoint; │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this)) { │ │ │ │ │ + for (var type in this.listeners) { │ │ │ │ │ + this.map.events.unregister( │ │ │ │ │ + type, this, this.listeners[type] │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - if (this.labelled) { │ │ │ │ │ - //keep track of when this grid line crosses the map bounds to set │ │ │ │ │ - //the label position │ │ │ │ │ - //labels along the bottom, add 10 pixel offset up into the map │ │ │ │ │ - //TODO add option for labels on top │ │ │ │ │ - var labelPos = new OpenLayers.Geometry.Point(labelPoint.x, mapBounds.bottom); │ │ │ │ │ - var labelAttrs = { │ │ │ │ │ - value: lon, │ │ │ │ │ - label: this.labelled ? OpenLayers.Util.getFormattedLonLat(lon, "lon", this.labelFormat) : "", │ │ │ │ │ - labelAlign: "cb", │ │ │ │ │ - xOffset: 0, │ │ │ │ │ - yOffset: 2 │ │ │ │ │ - }; │ │ │ │ │ - this.gratLayer.addFeatures(new OpenLayers.Feature.Vector(labelPos, labelAttrs)); │ │ │ │ │ - } │ │ │ │ │ - var geom = new OpenLayers.Geometry.LineString(pointList); │ │ │ │ │ - lines.push(new OpenLayers.Feature.Vector(geom)); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //now draw the lines of constant latitude │ │ │ │ │ - for (var j = 0; j < centerLonPoints.length; ++j) { │ │ │ │ │ - lat = centerLonPoints[j].y; │ │ │ │ │ - if (lat < -90 || lat > 90) { //latitudes only valid between -90 and 90 │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - var pointList = []; │ │ │ │ │ - var lonStart = centerLatPoints[0].x; │ │ │ │ │ - var lonEnd = centerLatPoints[centerLatPoints.length - 1].x; │ │ │ │ │ - var lonDelta = (lonEnd - lonStart) / this.numPoints; │ │ │ │ │ - var lon = lonStart; │ │ │ │ │ - var labelPoint = null; │ │ │ │ │ - for (var i = 0; i <= this.numPoints; ++i) { │ │ │ │ │ - var gridPoint = new OpenLayers.Geometry.Point(lon, lat); │ │ │ │ │ - gridPoint.transform(llProj, mapProj); │ │ │ │ │ - pointList.push(gridPoint); │ │ │ │ │ - lon += lonDelta; │ │ │ │ │ - if (gridPoint.x < mapBounds.right) { │ │ │ │ │ - labelPoint = gridPoint; │ │ │ │ │ + if (this.clearOnDeactivate) { │ │ │ │ │ + this.clear(); │ │ │ │ │ } │ │ │ │ │ + deactivated = true; │ │ │ │ │ } │ │ │ │ │ - if (this.labelled) { │ │ │ │ │ - //keep track of when this grid line crosses the map bounds to set │ │ │ │ │ - //the label position │ │ │ │ │ - //labels along the right, 30 pixel offset left into the map │ │ │ │ │ - //TODO add option for labels on left │ │ │ │ │ - var labelPos = new OpenLayers.Geometry.Point(mapBounds.right, labelPoint.y); │ │ │ │ │ - var labelAttrs = { │ │ │ │ │ - value: lat, │ │ │ │ │ - label: this.labelled ? OpenLayers.Util.getFormattedLonLat(lat, "lat", this.labelFormat) : "", │ │ │ │ │ - labelAlign: "rb", │ │ │ │ │ - xOffset: -2, │ │ │ │ │ - yOffset: 2 │ │ │ │ │ - }; │ │ │ │ │ - this.gratLayer.addFeatures(new OpenLayers.Feature.Vector(labelPos, labelAttrs)); │ │ │ │ │ - } │ │ │ │ │ - var geom = new OpenLayers.Geometry.LineString(pointList); │ │ │ │ │ - lines.push(new OpenLayers.Feature.Vector(geom)); │ │ │ │ │ } │ │ │ │ │ - this.gratLayer.addFeatures(lines); │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Graticule" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.NavigationHistory" │ │ │ │ │ }); │ │ │ │ │ │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Control/Measure.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ @@ -78982,6961 +75155,10788 @@ │ │ │ │ │ } │ │ │ │ │ return length; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.Measure" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Scale.js │ │ │ │ │ + OpenLayers/Control/ZoomPanel.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Control/Panel.js │ │ │ │ │ + * @requires OpenLayers/Control/ZoomIn.js │ │ │ │ │ + * @requires OpenLayers/Control/ZoomOut.js │ │ │ │ │ + * @requires OpenLayers/Control/ZoomToMaxExtent.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.Scale │ │ │ │ │ - * The Scale control displays the current map scale as a ratio (e.g. Scale = │ │ │ │ │ - * 1:1M). By default it is displayed in the lower right corner of the map. │ │ │ │ │ + * Class: OpenLayers.Control.ZoomPanel │ │ │ │ │ + * The ZoomPanel control is a compact collecton of 3 zoom controls: a │ │ │ │ │ + * <OpenLayers.Control.ZoomIn>, a <OpenLayers.Control.ZoomToMaxExtent>, and a │ │ │ │ │ + * <OpenLayers.Control.ZoomOut>. By default it is drawn in the upper left │ │ │ │ │ + * corner of the map. │ │ │ │ │ * │ │ │ │ │ + * Note: │ │ │ │ │ + * If you wish to use this class with the default images and you want │ │ │ │ │ + * it to look nice in ie6, you should add the following, conditionally │ │ │ │ │ + * added css stylesheet to your HTML file: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * <!--[if lte IE 6]> │ │ │ │ │ + * <link rel="stylesheet" href="../theme/default/ie6-style.css" type="text/css" /> │ │ │ │ │ + * <![endif]--> │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ + * - <OpenLayers.Control.Panel> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.Scale = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: element │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - element: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geodesic │ │ │ │ │ - * {Boolean} Use geodesic measurement. Default is false. The recommended │ │ │ │ │ - * setting for maps in EPSG:4326 is false, and true EPSG:900913. If set to │ │ │ │ │ - * true, the scale will be calculated based on the horizontal size of the │ │ │ │ │ - * pixel in the center of the map viewport. │ │ │ │ │ - */ │ │ │ │ │ - geodesic: false, │ │ │ │ │ +OpenLayers.Control.ZoomPanel = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Scale │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Control.ZoomPanel │ │ │ │ │ + * Add the three zooming controls. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * element - {DOMElement} │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(element, options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.element = OpenLayers.Util.getElement(element); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (!this.element) { │ │ │ │ │ - this.element = document.createElement("div"); │ │ │ │ │ - this.div.appendChild(this.element); │ │ │ │ │ - } │ │ │ │ │ - this.map.events.register('moveend', this, this.updateScale); │ │ │ │ │ - this.updateScale(); │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateScale │ │ │ │ │ + * options - {Object} An optional object whose properties will be used │ │ │ │ │ + * to extend the control. │ │ │ │ │ */ │ │ │ │ │ - updateScale: function() { │ │ │ │ │ - var scale; │ │ │ │ │ - if (this.geodesic === true) { │ │ │ │ │ - var units = this.map.getUnits(); │ │ │ │ │ - if (!units) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var inches = OpenLayers.INCHES_PER_UNIT; │ │ │ │ │ - scale = (this.map.getGeodesicPixelSize().w || 0.000001) * │ │ │ │ │ - inches["km"] * OpenLayers.DOTS_PER_INCH; │ │ │ │ │ - } else { │ │ │ │ │ - scale = this.map.getScale(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (!scale) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (scale >= 9500 && scale <= 950000) { │ │ │ │ │ - scale = Math.round(scale / 1000) + "K"; │ │ │ │ │ - } else if (scale >= 950000) { │ │ │ │ │ - scale = Math.round(scale / 1000000) + "M"; │ │ │ │ │ - } else { │ │ │ │ │ - scale = Math.round(scale); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.element.innerHTML = OpenLayers.i18n("Scale = 1 : ${scaleDenom}", { │ │ │ │ │ - 'scaleDenom': scale │ │ │ │ │ - }); │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.addControls([ │ │ │ │ │ + new OpenLayers.Control.ZoomIn(), │ │ │ │ │ + new OpenLayers.Control.ZoomToMaxExtent(), │ │ │ │ │ + new OpenLayers.Control.ZoomOut() │ │ │ │ │ + ]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Scale" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomPanel" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/SLDSelect.js │ │ │ │ │ + OpenLayers/Tile/UTFGrid.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Layer/WMS.js │ │ │ │ │ - * @requires OpenLayers/Handler/RegularPolygon.js │ │ │ │ │ - * @requires OpenLayers/Handler/Polygon.js │ │ │ │ │ - * @requires OpenLayers/Handler/Path.js │ │ │ │ │ - * @requires OpenLayers/Handler/Click.js │ │ │ │ │ - * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ - * @requires OpenLayers/Format/SLD/v1_0_0.js │ │ │ │ │ + * @requires OpenLayers/Tile.js │ │ │ │ │ + * @requires OpenLayers/Format/JSON.js │ │ │ │ │ + * @requires OpenLayers/Request.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.SLDSelect │ │ │ │ │ - * Perform selections on WMS layers using Styled Layer Descriptor (SLD) │ │ │ │ │ + * Class: OpenLayers.Tile.UTFGrid │ │ │ │ │ + * Instances of OpenLayers.Tile.UTFGrid are used to manage │ │ │ │ │ + * UTFGrids. This is an unusual tile type in that it doesn't have a │ │ │ │ │ + * rendered image; only a 'hit grid' that can be used to │ │ │ │ │ + * look up feature attributes. │ │ │ │ │ + * │ │ │ │ │ + * See the <OpenLayers.Tile.UTFGrid> constructor for details on constructing a │ │ │ │ │ + * new instance. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ + * - <OpenLayers.Tile> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.SLDSelect = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Tile.UTFGrid = OpenLayers.Class(OpenLayers.Tile, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * selected - Triggered when a selection occurs. Listeners receive an │ │ │ │ │ - * event with *filters* and *layer* properties. Filters will be an │ │ │ │ │ - * array of OpenLayers.Filter objects created in order to perform │ │ │ │ │ - * the particular selection. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: clearOnDeactivate │ │ │ │ │ - * {Boolean} Should the selection be cleared when the control is │ │ │ │ │ - * deactivated. Default value is false. │ │ │ │ │ - */ │ │ │ │ │ - clearOnDeactivate: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: layers │ │ │ │ │ - * {Array(<OpenLayers.Layer.WMS>)} The WMS layers this control will work │ │ │ │ │ - * on. │ │ │ │ │ - */ │ │ │ │ │ - layers: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: callbacks │ │ │ │ │ - * {Object} The functions that are sent to the handler for callback │ │ │ │ │ + * Property: url │ │ │ │ │ + * {String} │ │ │ │ │ + * The URL of the UTFGrid file being requested. Provided by the <getURL> │ │ │ │ │ + * method. │ │ │ │ │ */ │ │ │ │ │ - callbacks: null, │ │ │ │ │ + url: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: selectionSymbolizer │ │ │ │ │ - * {Object} Determines the styling of the selected objects. Default is │ │ │ │ │ - * a selection in red. │ │ │ │ │ + * Property: utfgridResolution │ │ │ │ │ + * {Number} │ │ │ │ │ + * Ratio of the pixel width to the width of a UTFGrid data point. If an │ │ │ │ │ + * entry in the grid represents a 4x4 block of pixels, the │ │ │ │ │ + * utfgridResolution would be 4. Default is 2. │ │ │ │ │ */ │ │ │ │ │ - selectionSymbolizer: { │ │ │ │ │ - 'Polygon': { │ │ │ │ │ - fillColor: '#FF0000', │ │ │ │ │ - stroke: false │ │ │ │ │ - }, │ │ │ │ │ - 'Line': { │ │ │ │ │ - strokeColor: '#FF0000', │ │ │ │ │ - strokeWidth: 2 │ │ │ │ │ - }, │ │ │ │ │ - 'Point': { │ │ │ │ │ - graphicName: 'square', │ │ │ │ │ - fillColor: '#FF0000', │ │ │ │ │ - pointRadius: 5 │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + utfgridResolution: 2, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: layerOptions │ │ │ │ │ - * {Object} The options to apply to the selection layer, by default the │ │ │ │ │ - * selection layer will be kept out of the layer switcher. │ │ │ │ │ + /** │ │ │ │ │ + * Property: json │ │ │ │ │ + * {Object} │ │ │ │ │ + * Stores the parsed JSON tile data structure. │ │ │ │ │ */ │ │ │ │ │ - layerOptions: null, │ │ │ │ │ + json: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: handlerOptions │ │ │ │ │ - * {Object} Used to set non-default properties on the control's handler │ │ │ │ │ + /** │ │ │ │ │ + * Property: format │ │ │ │ │ + * {OpenLayers.Format.JSON} │ │ │ │ │ + * Parser instance used to parse JSON for cross browser support. The native │ │ │ │ │ + * JSON.parse method will be used where available (all except IE<8). │ │ │ │ │ */ │ │ │ │ │ + format: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: sketchStyle │ │ │ │ │ - * {<OpenLayers.Style>|Object} Style or symbolizer to use for the sketch │ │ │ │ │ - * handler. The recommended way of styling the sketch layer, however, is │ │ │ │ │ - * to configure an <OpenLayers.StyleMap> in the layerOptions of the │ │ │ │ │ - * <handlerOptions>: │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Tile.UTFGrid │ │ │ │ │ + * Constructor for a new <OpenLayers.Tile.UTFGrid> instance. │ │ │ │ │ * │ │ │ │ │ - * (code) │ │ │ │ │ - * new OpenLayers.Control.SLDSelect(OpenLayers.Handler.Path, { │ │ │ │ │ - * handlerOptions: { │ │ │ │ │ - * layerOptions: { │ │ │ │ │ - * styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ - * "default": {strokeColor: "yellow"} │ │ │ │ │ - * }) │ │ │ │ │ - * } │ │ │ │ │ - * } │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ - sketchStyle: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: wfsCache │ │ │ │ │ - * {Object} Cache to use for storing parsed results from │ │ │ │ │ - * <OpenLayers.Format.WFSDescribeFeatureType.read>. If not provided, │ │ │ │ │ - * these will be cached on the prototype. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer>} layer that the tile will go in. │ │ │ │ │ + * position - {<OpenLayers.Pixel>} │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * url - {<String>} Deprecated. Remove me in 3.0. │ │ │ │ │ + * size - {<OpenLayers.Size>} │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - wfsCache: {}, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: layerCache │ │ │ │ │ - * {Object} Cache to use for storing references to the selection layers. │ │ │ │ │ - * Normally each source layer will have exactly 1 selection layer of │ │ │ │ │ - * type OpenLayers.Layer.WMS. If not provided, layers will │ │ │ │ │ - * be cached on the prototype. Note that if <clearOnDeactivate> is │ │ │ │ │ - * true, the layer will no longer be cached after deactivating the │ │ │ │ │ - * control. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up. │ │ │ │ │ */ │ │ │ │ │ - layerCache: {}, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.clear(); │ │ │ │ │ + OpenLayers.Tile.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.SLDSelect │ │ │ │ │ - * Create a new control for selecting features in WMS layers using │ │ │ │ │ - * Styled Layer Descriptor (SLD). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * handler - {<OpenLayers.Class>} A sketch handler class. This determines │ │ │ │ │ - * the type of selection, e.g. box (<OpenLayers.Handler.Box>), point │ │ │ │ │ - * (<OpenLayers.Handler.Point>), path (<OpenLayers.Handler.Path>) or │ │ │ │ │ - * polygon (<OpenLayers.Handler.Polygon>) selection. To use circle │ │ │ │ │ - * type selection, use <OpenLayers.Handler.RegularPolygon> and pass │ │ │ │ │ - * the number of desired sides (e.g. 40) as "sides" property to the │ │ │ │ │ - * <handlerOptions>. │ │ │ │ │ - * options - {Object} An object containing all configuration properties for │ │ │ │ │ - * the control. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * layers - Array({<OpenLayers.Layer.WMS>}) The layers to perform the │ │ │ │ │ - * selection on. │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Check that a tile should be drawn, and draw it. │ │ │ │ │ + * In the case of UTFGrids, "drawing" it means fetching and │ │ │ │ │ + * parsing the json. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Was a tile drawn? │ │ │ │ │ */ │ │ │ │ │ - initialize: function(handler, options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + draw: function() { │ │ │ │ │ + var drawn = OpenLayers.Tile.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (drawn) { │ │ │ │ │ + if (this.isLoading) { │ │ │ │ │ + this.abortLoading(); │ │ │ │ │ + //if we're already loading, send 'reload' instead of 'loadstart'. │ │ │ │ │ + this.events.triggerEvent("reload"); │ │ │ │ │ + } else { │ │ │ │ │ + this.isLoading = true; │ │ │ │ │ + this.events.triggerEvent("loadstart"); │ │ │ │ │ + } │ │ │ │ │ + this.url = this.layer.getURL(this.bounds); │ │ │ │ │ │ │ │ │ │ - this.callbacks = OpenLayers.Util.extend({ │ │ │ │ │ - done: this.select, │ │ │ │ │ - click: this.select │ │ │ │ │ - }, this.callbacks); │ │ │ │ │ - this.handlerOptions = this.handlerOptions || {}; │ │ │ │ │ - this.layerOptions = OpenLayers.Util.applyDefaults(this.layerOptions, { │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - tileOptions: { │ │ │ │ │ - maxGetUrlLength: 2048 │ │ │ │ │ + if (this.layer.useJSONP) { │ │ │ │ │ + // Use JSONP method to avoid xbrowser policy │ │ │ │ │ + var ols = new OpenLayers.Protocol.Script({ │ │ │ │ │ + url: this.url, │ │ │ │ │ + callback: function(response) { │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + this.json = response.data; │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + ols.read(); │ │ │ │ │ + this.request = ols; │ │ │ │ │ + } else { │ │ │ │ │ + // Use standard XHR │ │ │ │ │ + this.request = OpenLayers.Request.GET({ │ │ │ │ │ + url: this.url, │ │ │ │ │ + callback: function(response) { │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + if (response.status === 200) { │ │ │ │ │ + this.parseData(response.responseText); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - }); │ │ │ │ │ - if (this.sketchStyle) { │ │ │ │ │ - this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.handlerOptions.layerOptions, { │ │ │ │ │ - styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ - "default": this.sketchStyle │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ + } else { │ │ │ │ │ + this.unload(); │ │ │ │ │ } │ │ │ │ │ - this.handler = new handler(this, this.callbacks, this.handlerOptions); │ │ │ │ │ + return drawn; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Take care of things that are not handled in superclass. │ │ │ │ │ + * Method: abortLoading │ │ │ │ │ + * Cancel a pending request. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var key in this.layerCache) { │ │ │ │ │ - delete this.layerCache[key]; │ │ │ │ │ - } │ │ │ │ │ - for (var key in this.wfsCache) { │ │ │ │ │ - delete this.wfsCache[key]; │ │ │ │ │ + abortLoading: function() { │ │ │ │ │ + if (this.request) { │ │ │ │ │ + this.request.abort(); │ │ │ │ │ + delete this.request; │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: coupleLayerVisiblity │ │ │ │ │ - * Couple the selection layer and the source layer with respect to │ │ │ │ │ - * layer visibility. So if the source layer is turned off, the │ │ │ │ │ - * selection layer is also turned off. │ │ │ │ │ - * │ │ │ │ │ - * Context: │ │ │ │ │ - * - {<OpenLayers.Layer>} │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ - */ │ │ │ │ │ - coupleLayerVisiblity: function(evt) { │ │ │ │ │ - this.setVisibility(evt.object.getVisibility()); │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createSelectionLayer │ │ │ │ │ - * Creates a "clone" from the source layer in which the selection can │ │ │ │ │ - * be drawn. This ensures both the source layer and the selection are │ │ │ │ │ - * visible and not only the selection. │ │ │ │ │ + * Method: getFeatureInfo │ │ │ │ │ + * Get feature information associated with a pixel offset. If the pixel │ │ │ │ │ + * offset corresponds to a feature, the returned object will have id │ │ │ │ │ + * and data properties. Otherwise, null will be returned. │ │ │ │ │ + * │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * source - {<OpenLayers.Layer.WMS>} The source layer on which the selection │ │ │ │ │ - * is performed. │ │ │ │ │ + * i - {Number} X-axis pixel offset (from top left of tile) │ │ │ │ │ + * j - {Number} Y-axis pixel offset (from top left of tile) │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.WMS>} A WMS layer with maxGetUrlLength configured to 2048 │ │ │ │ │ - * since SLD selections can easily get quite long. │ │ │ │ │ + * {Object} Object with feature id and data properties corresponding to the │ │ │ │ │ + * given pixel offset. │ │ │ │ │ */ │ │ │ │ │ - createSelectionLayer: function(source) { │ │ │ │ │ - // check if we already have a selection layer for the source layer │ │ │ │ │ - var selectionLayer; │ │ │ │ │ - if (!this.layerCache[source.id]) { │ │ │ │ │ - selectionLayer = new OpenLayers.Layer.WMS(source.name, │ │ │ │ │ - source.url, source.params, │ │ │ │ │ - OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.layerOptions, │ │ │ │ │ - source.getOptions()) │ │ │ │ │ - ); │ │ │ │ │ - this.layerCache[source.id] = selectionLayer; │ │ │ │ │ - // make sure the layers are coupled wrt visibility, but only │ │ │ │ │ - // if they are not displayed in the layer switcher, because in │ │ │ │ │ - // that case the user cannot control visibility. │ │ │ │ │ - if (this.layerOptions.displayInLayerSwitcher === false) { │ │ │ │ │ - source.events.on({ │ │ │ │ │ - "visibilitychanged": this.coupleLayerVisiblity, │ │ │ │ │ - scope: selectionLayer │ │ │ │ │ - }); │ │ │ │ │ + getFeatureInfo: function(i, j) { │ │ │ │ │ + var info = null; │ │ │ │ │ + if (this.json) { │ │ │ │ │ + var id = this.getFeatureId(i, j); │ │ │ │ │ + if (id !== null) { │ │ │ │ │ + info = { │ │ │ │ │ + id: id, │ │ │ │ │ + data: this.json.data[id] │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ - this.map.addLayer(selectionLayer); │ │ │ │ │ - } else { │ │ │ │ │ - selectionLayer = this.layerCache[source.id]; │ │ │ │ │ } │ │ │ │ │ - return selectionLayer; │ │ │ │ │ + return info; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createSLD │ │ │ │ │ - * Create the SLD document for the layer using the supplied filters. │ │ │ │ │ + * Method: getFeatureId │ │ │ │ │ + * Get the identifier for the feature associated with a pixel offset. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.WMS>} │ │ │ │ │ - * filters - Array({<OpenLayers.Filter>}) The filters to be applied. │ │ │ │ │ - * geometryAttributes - Array({Object}) The geometry attributes of the │ │ │ │ │ - * layer. │ │ │ │ │ + * i - {Number} X-axis pixel offset (from top left of tile) │ │ │ │ │ + * j - {Number} Y-axis pixel offset (from top left of tile) │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} The SLD document generated as a string. │ │ │ │ │ + * {Object} The feature identifier corresponding to the given pixel offset. │ │ │ │ │ + * Returns null if pixel doesn't correspond to a feature. │ │ │ │ │ */ │ │ │ │ │ - createSLD: function(layer, filters, geometryAttributes) { │ │ │ │ │ - var sld = { │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - namedLayers: {} │ │ │ │ │ - }; │ │ │ │ │ - var layerNames = [layer.params.LAYERS].join(",").split(","); │ │ │ │ │ - for (var i = 0, len = layerNames.length; i < len; i++) { │ │ │ │ │ - var name = layerNames[i]; │ │ │ │ │ - sld.namedLayers[name] = { │ │ │ │ │ - name: name, │ │ │ │ │ - userStyles: [] │ │ │ │ │ - }; │ │ │ │ │ - var symbolizer = this.selectionSymbolizer; │ │ │ │ │ - var geometryAttribute = geometryAttributes[i]; │ │ │ │ │ - if (geometryAttribute.type.indexOf('Polygon') >= 0) { │ │ │ │ │ - symbolizer = { │ │ │ │ │ - Polygon: this.selectionSymbolizer['Polygon'] │ │ │ │ │ - }; │ │ │ │ │ - } else if (geometryAttribute.type.indexOf('LineString') >= 0) { │ │ │ │ │ - symbolizer = { │ │ │ │ │ - Line: this.selectionSymbolizer['Line'] │ │ │ │ │ - }; │ │ │ │ │ - } else if (geometryAttribute.type.indexOf('Point') >= 0) { │ │ │ │ │ - symbolizer = { │ │ │ │ │ - Point: this.selectionSymbolizer['Point'] │ │ │ │ │ - }; │ │ │ │ │ + getFeatureId: function(i, j) { │ │ │ │ │ + var id = null; │ │ │ │ │ + if (this.json) { │ │ │ │ │ + var resolution = this.utfgridResolution; │ │ │ │ │ + var row = Math.floor(j / resolution); │ │ │ │ │ + var col = Math.floor(i / resolution); │ │ │ │ │ + var charCode = this.json.grid[row].charCodeAt(col); │ │ │ │ │ + var index = this.indexFromCharCode(charCode); │ │ │ │ │ + var keys = this.json.keys; │ │ │ │ │ + if (!isNaN(index) && (index in keys)) { │ │ │ │ │ + id = keys[index]; │ │ │ │ │ } │ │ │ │ │ - var filter = filters[i]; │ │ │ │ │ - sld.namedLayers[name].userStyles.push({ │ │ │ │ │ - name: 'default', │ │ │ │ │ - rules: [ │ │ │ │ │ - new OpenLayers.Rule({ │ │ │ │ │ - symbolizer: symbolizer, │ │ │ │ │ - filter: filter, │ │ │ │ │ - maxScaleDenominator: layer.options.minScale │ │ │ │ │ - }) │ │ │ │ │ - ] │ │ │ │ │ - }); │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Format.SLD({ │ │ │ │ │ - srsName: this.map.getProjection() │ │ │ │ │ - }).write(sld); │ │ │ │ │ + return id; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseDescribeLayer │ │ │ │ │ - * Parse the SLD WMS DescribeLayer response and issue the corresponding │ │ │ │ │ - * WFS DescribeFeatureType request │ │ │ │ │ + * Method: indexFromCharCode │ │ │ │ │ + * Given a character code for one of the UTFGrid "grid" characters, │ │ │ │ │ + * resolve the integer index for the feature id in the UTFGrid "keys" │ │ │ │ │ + * array. │ │ │ │ │ * │ │ │ │ │ - * request - {XMLHttpRequest} The request object. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * charCode - {Integer} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} Index for the feature id from the keys array. │ │ │ │ │ */ │ │ │ │ │ - parseDescribeLayer: function(request) { │ │ │ │ │ - var format = new OpenLayers.Format.WMSDescribeLayer(); │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ + indexFromCharCode: function(charCode) { │ │ │ │ │ + if (charCode >= 93) { │ │ │ │ │ + charCode--; │ │ │ │ │ } │ │ │ │ │ - var describeLayer = format.read(doc); │ │ │ │ │ - var typeNames = []; │ │ │ │ │ - var url = null; │ │ │ │ │ - for (var i = 0, len = describeLayer.length; i < len; i++) { │ │ │ │ │ - // perform a WFS DescribeFeatureType request │ │ │ │ │ - if (describeLayer[i].owsType == "WFS") { │ │ │ │ │ - typeNames.push(describeLayer[i].typeName); │ │ │ │ │ - url = describeLayer[i].owsURL; │ │ │ │ │ - } │ │ │ │ │ + if (charCode >= 35) { │ │ │ │ │ + charCode--; │ │ │ │ │ } │ │ │ │ │ - var options = { │ │ │ │ │ - url: url, │ │ │ │ │ - params: { │ │ │ │ │ - SERVICE: "WFS", │ │ │ │ │ - TYPENAME: typeNames.toString(), │ │ │ │ │ - REQUEST: "DescribeFeatureType", │ │ │ │ │ - VERSION: "1.0.0" │ │ │ │ │ - }, │ │ │ │ │ - callback: function(request) { │ │ │ │ │ - var format = new OpenLayers.Format.WFSDescribeFeatureType(); │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ - } │ │ │ │ │ - var describeFeatureType = format.read(doc); │ │ │ │ │ - this.control.wfsCache[this.layer.id] = describeFeatureType; │ │ │ │ │ - this.control._queue && this.control.applySelection(); │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Request.GET(options); │ │ │ │ │ + return charCode - 32; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getGeometryAttributes │ │ │ │ │ - * Look up the geometry attributes from the WFS DescribeFeatureType response │ │ │ │ │ + * Method: parseData │ │ │ │ │ + * Parse the JSON from a request │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.WMS>} The layer for which to look up the │ │ │ │ │ - * geometry attributes. │ │ │ │ │ - * │ │ │ │ │ + * str - {String} UTFGrid as a JSON string. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * Array({Object}) Array of geometry attributes │ │ │ │ │ + * {Object} parsed javascript data │ │ │ │ │ */ │ │ │ │ │ - getGeometryAttributes: function(layer) { │ │ │ │ │ - var result = []; │ │ │ │ │ - var cache = this.wfsCache[layer.id]; │ │ │ │ │ - for (var i = 0, len = cache.featureTypes.length; i < len; i++) { │ │ │ │ │ - var typeName = cache.featureTypes[i]; │ │ │ │ │ - var properties = typeName.properties; │ │ │ │ │ - for (var j = 0, lenj = properties.length; j < lenj; j++) { │ │ │ │ │ - var property = properties[j]; │ │ │ │ │ - var type = property.type; │ │ │ │ │ - if ((type.indexOf('LineString') >= 0) || │ │ │ │ │ - (type.indexOf('GeometryAssociationType') >= 0) || │ │ │ │ │ - (type.indexOf('GeometryPropertyType') >= 0) || │ │ │ │ │ - (type.indexOf('Point') >= 0) || │ │ │ │ │ - (type.indexOf('Polygon') >= 0)) { │ │ │ │ │ - result.push(property); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + parseData: function(str) { │ │ │ │ │ + if (!this.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.JSON(); │ │ │ │ │ } │ │ │ │ │ - return result; │ │ │ │ │ + this.json = this.format.read(str); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Activate the control. Activating the control will perform a SLD WMS │ │ │ │ │ - * DescribeLayer request followed by a WFS DescribeFeatureType request │ │ │ │ │ - * so that the proper symbolizers can be chosen based on the geometry │ │ │ │ │ - * type. │ │ │ │ │ + /** │ │ │ │ │ + * Method: clear │ │ │ │ │ + * Delete data stored with this tile. │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Control.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.layers[i]; │ │ │ │ │ - if (layer && !this.wfsCache[layer.id]) { │ │ │ │ │ - var options = { │ │ │ │ │ - url: layer.url, │ │ │ │ │ - params: { │ │ │ │ │ - SERVICE: "WMS", │ │ │ │ │ - VERSION: layer.params.VERSION, │ │ │ │ │ - LAYERS: layer.params.LAYERS, │ │ │ │ │ - REQUEST: "DescribeLayer" │ │ │ │ │ - }, │ │ │ │ │ - callback: this.parseDescribeLayer, │ │ │ │ │ - scope: { │ │ │ │ │ - layer: layer, │ │ │ │ │ - control: this │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Request.GET(options); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ + clear: function() { │ │ │ │ │ + this.json = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Tile.UTFGrid" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Tile/Image/IFrame.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Tile/Image.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Tile.Image.IFrame │ │ │ │ │ + * Mixin for tiles that use form-encoded POST requests to get images from │ │ │ │ │ + * remote services. Images will be loaded using HTTP-POST into an IFrame. │ │ │ │ │ + * │ │ │ │ │ + * This mixin will be applied to <OpenLayers.Tile.Image> instances │ │ │ │ │ + * configured with <OpenLayers.Tile.Image.maxGetUrlLength> set. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Tile.Image.IFrame = { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the control. If clearOnDeactivate is true, remove the │ │ │ │ │ - * selection layer(s). │ │ │ │ │ + * Property: useIFrame │ │ │ │ │ + * {Boolean} true if we are currently using an IFrame to render POST │ │ │ │ │ + * responses, false if we are using an img element to render GET responses. │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Control.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.layers[i]; │ │ │ │ │ - if (layer && this.clearOnDeactivate === true) { │ │ │ │ │ - var layerCache = this.layerCache; │ │ │ │ │ - var selectionLayer = layerCache[layer.id]; │ │ │ │ │ - if (selectionLayer) { │ │ │ │ │ - layer.events.un({ │ │ │ │ │ - "visibilitychanged": this.coupleLayerVisiblity, │ │ │ │ │ - scope: selectionLayer │ │ │ │ │ - }); │ │ │ │ │ - selectionLayer.destroy(); │ │ │ │ │ - delete layerCache[layer.id]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ + useIFrame: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setLayers │ │ │ │ │ - * Set the layers on which the selection should be performed. Call the │ │ │ │ │ - * setLayers method if the layer(s) to be used change and the same │ │ │ │ │ - * control should be used on a new set of layers. │ │ │ │ │ - * If the control is already active, it will be active after the new │ │ │ │ │ - * set of layers is set. │ │ │ │ │ + * Property: blankImageUrl │ │ │ │ │ + * {String} Using a data scheme url is not supported by all browsers, but │ │ │ │ │ + * we don't care because we either set it as css backgroundImage, or the │ │ │ │ │ + * image's display style is set to "none" when we use it. │ │ │ │ │ + */ │ │ │ │ │ + blankImageUrl: "data:image/gif;base64,R0lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAQAIBRAA7", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Set useIFrame in the instance, and operate the image/iframe switch. │ │ │ │ │ + * Then call Tile.Image.draw. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layers - {Array(<OpenLayers.Layer.WMS>)} The new set of layers on which │ │ │ │ │ - * the selection should be performed. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - setLayers: function(layers) { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.layers = layers; │ │ │ │ │ - this.activate(); │ │ │ │ │ - } else { │ │ │ │ │ - this.layers = layers; │ │ │ │ │ + draw: function() { │ │ │ │ │ + var draw = OpenLayers.Tile.Image.prototype.shouldDraw.call(this); │ │ │ │ │ + if (draw) { │ │ │ │ │ + │ │ │ │ │ + // this.url isn't set to the currect value yet, so we call getURL │ │ │ │ │ + // on the layer and store the result in a local variable │ │ │ │ │ + var url = this.layer.getURL(this.bounds); │ │ │ │ │ + │ │ │ │ │ + var usedIFrame = this.useIFrame; │ │ │ │ │ + this.useIFrame = this.maxGetUrlLength !== null && │ │ │ │ │ + !this.layer.async && │ │ │ │ │ + url.length > this.maxGetUrlLength; │ │ │ │ │ + │ │ │ │ │ + var fromIFrame = usedIFrame && !this.useIFrame; │ │ │ │ │ + var toIFrame = !usedIFrame && this.useIFrame; │ │ │ │ │ + │ │ │ │ │ + if (fromIFrame || toIFrame) { │ │ │ │ │ + │ │ │ │ │ + // Switching between GET (image) and POST (iframe). │ │ │ │ │ + │ │ │ │ │ + // We remove the imgDiv (really either an image or an iframe) │ │ │ │ │ + // from the frame and set it to null to make sure initImage │ │ │ │ │ + // will call getImage. │ │ │ │ │ + │ │ │ │ │ + if (this.imgDiv && this.imgDiv.parentNode === this.frame) { │ │ │ │ │ + this.frame.removeChild(this.imgDiv); │ │ │ │ │ + } │ │ │ │ │ + this.imgDiv = null; │ │ │ │ │ + │ │ │ │ │ + // And if we had an iframe we also remove the event pane. │ │ │ │ │ + │ │ │ │ │ + if (fromIFrame) { │ │ │ │ │ + this.frame.removeChild(this.frame.firstChild); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return OpenLayers.Tile.Image.prototype.draw.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Function: createFilter │ │ │ │ │ - * Create the filter to be used in the SLD. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometryAttribute - {Object} Used to get the name of the geometry │ │ │ │ │ - * attribute which is needed for constructing the spatial filter. │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} The geometry to use. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Filter.Spatial>} The spatial filter created. │ │ │ │ │ + * Method: getImage │ │ │ │ │ + * Creates the content for the frame on the tile. │ │ │ │ │ */ │ │ │ │ │ - createFilter: function(geometryAttribute, geometry) { │ │ │ │ │ - var filter = null; │ │ │ │ │ - if (this.handler instanceof OpenLayers.Handler.RegularPolygon) { │ │ │ │ │ - // box │ │ │ │ │ - if (this.handler.irregular === true) { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry.getBounds() │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry │ │ │ │ │ - }); │ │ │ │ │ + getImage: function() { │ │ │ │ │ + if (this.useIFrame === true) { │ │ │ │ │ + if (!this.frame.childNodes.length) { │ │ │ │ │ + var eventPane = document.createElement("div"), │ │ │ │ │ + style = eventPane.style; │ │ │ │ │ + style.position = "absolute"; │ │ │ │ │ + style.width = "100%"; │ │ │ │ │ + style.height = "100%"; │ │ │ │ │ + style.zIndex = 1; │ │ │ │ │ + style.backgroundImage = "url(" + this.blankImageUrl + ")"; │ │ │ │ │ + this.frame.appendChild(eventPane); │ │ │ │ │ } │ │ │ │ │ - } else if (this.handler instanceof OpenLayers.Handler.Polygon) { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry │ │ │ │ │ - }); │ │ │ │ │ - } else if (this.handler instanceof OpenLayers.Handler.Path) { │ │ │ │ │ - // if source layer is point based, use DWITHIN instead │ │ │ │ │ - if (geometryAttribute.type.indexOf('Point') >= 0) { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - distance: this.map.getExtent().getWidth() * 0.01, │ │ │ │ │ - distanceUnits: this.map.getUnits(), │ │ │ │ │ - value: geometry │ │ │ │ │ - }); │ │ │ │ │ + │ │ │ │ │ + var id = this.id + '_iFrame', │ │ │ │ │ + iframe; │ │ │ │ │ + if (parseFloat(navigator.appVersion.split("MSIE")[1]) < 9) { │ │ │ │ │ + // Older IE versions do not set the name attribute of an iFrame │ │ │ │ │ + // properly via DOM manipulation, so we need to do it on our own with │ │ │ │ │ + // this hack. │ │ │ │ │ + iframe = document.createElement('<iframe name="' + id + '">'); │ │ │ │ │ + │ │ │ │ │ + // IFrames in older IE versions are not transparent, if you set │ │ │ │ │ + // the backgroundColor transparent. This is a workaround to get │ │ │ │ │ + // transparent iframes. │ │ │ │ │ + iframe.style.backgroundColor = '#FFFFFF'; │ │ │ │ │ + iframe.style.filter = 'chroma(color=#FFFFFF)'; │ │ │ │ │ } else { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry │ │ │ │ │ - }); │ │ │ │ │ + iframe = document.createElement('iframe'); │ │ │ │ │ + iframe.style.backgroundColor = 'transparent'; │ │ │ │ │ + │ │ │ │ │ + // iframe.name needs to be an unique id, otherwise it │ │ │ │ │ + // could happen that other iframes are overwritten. │ │ │ │ │ + iframe.name = id; │ │ │ │ │ } │ │ │ │ │ - } else if (this.handler instanceof OpenLayers.Handler.Click) { │ │ │ │ │ - if (geometryAttribute.type.indexOf('Polygon') >= 0) { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - distance: this.map.getExtent().getWidth() * 0.01, │ │ │ │ │ - distanceUnits: this.map.getUnits(), │ │ │ │ │ - value: geometry │ │ │ │ │ - }); │ │ │ │ │ + │ │ │ │ │ + // some special properties to avoid scaling the images and scrollbars │ │ │ │ │ + // in the iframe │ │ │ │ │ + iframe.scrolling = 'no'; │ │ │ │ │ + iframe.marginWidth = '0px'; │ │ │ │ │ + iframe.marginHeight = '0px'; │ │ │ │ │ + iframe.frameBorder = '0'; │ │ │ │ │ + │ │ │ │ │ + iframe.style.position = "absolute"; │ │ │ │ │ + iframe.style.width = "100%"; │ │ │ │ │ + iframe.style.height = "100%"; │ │ │ │ │ + │ │ │ │ │ + if (this.layer.opacity < 1) { │ │ │ │ │ + OpenLayers.Util.modifyDOMElement(iframe, null, null, null, │ │ │ │ │ + null, null, null, this.layer.opacity); │ │ │ │ │ } │ │ │ │ │ + this.frame.appendChild(iframe); │ │ │ │ │ + this.imgDiv = iframe; │ │ │ │ │ + return iframe; │ │ │ │ │ + } else { │ │ │ │ │ + return OpenLayers.Tile.Image.prototype.getImage.apply(this, arguments); │ │ │ │ │ } │ │ │ │ │ - return filter; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: select │ │ │ │ │ - * When the handler is done, use SLD_BODY on the selection layer to │ │ │ │ │ - * display the selection in the map. │ │ │ │ │ + * Method: createRequestForm │ │ │ │ │ + * Create the html <form> element with width, height, bbox and all │ │ │ │ │ + * parameters specified in the layer params. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {Object} or {<OpenLayers.Geometry>} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The form element which sends the HTTP-POST request to the │ │ │ │ │ + * WMS. │ │ │ │ │ */ │ │ │ │ │ - select: function(geometry) { │ │ │ │ │ - this._queue = function() { │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.layers[i]; │ │ │ │ │ - var geometryAttributes = this.getGeometryAttributes(layer); │ │ │ │ │ - var filters = []; │ │ │ │ │ - for (var j = 0, lenj = geometryAttributes.length; j < lenj; j++) { │ │ │ │ │ - var geometryAttribute = geometryAttributes[j]; │ │ │ │ │ - if (geometryAttribute !== null) { │ │ │ │ │ - // from the click handler we will not get an actual │ │ │ │ │ - // geometry so transform │ │ │ │ │ - if (!(geometry instanceof OpenLayers.Geometry)) { │ │ │ │ │ - var point = this.map.getLonLatFromPixel( │ │ │ │ │ - geometry.xy); │ │ │ │ │ - geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ - point.lon, point.lat); │ │ │ │ │ - } │ │ │ │ │ - var filter = this.createFilter(geometryAttribute, │ │ │ │ │ - geometry); │ │ │ │ │ - if (filter !== null) { │ │ │ │ │ - filters.push(filter); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + createRequestForm: function() { │ │ │ │ │ + // creation of the form element │ │ │ │ │ + var form = document.createElement('form'); │ │ │ │ │ + form.method = 'POST'; │ │ │ │ │ + var cacheId = this.layer.params["_OLSALT"]; │ │ │ │ │ + cacheId = (cacheId ? cacheId + "_" : "") + this.bounds.toBBOX(); │ │ │ │ │ + form.action = OpenLayers.Util.urlAppend(this.layer.url, cacheId); │ │ │ │ │ + form.target = this.id + '_iFrame'; │ │ │ │ │ │ │ │ │ │ - var selectionLayer = this.createSelectionLayer(layer); │ │ │ │ │ + // adding all parameters in layer params as hidden fields to the html │ │ │ │ │ + // form element │ │ │ │ │ + var imageSize = this.layer.getImageSize(), │ │ │ │ │ + params = OpenLayers.Util.getParameters(this.url), │ │ │ │ │ + field; │ │ │ │ │ │ │ │ │ │ - this.events.triggerEvent("selected", { │ │ │ │ │ - layer: layer, │ │ │ │ │ - filters: filters │ │ │ │ │ - }); │ │ │ │ │ + for (var par in params) { │ │ │ │ │ + field = document.createElement('input'); │ │ │ │ │ + field.type = 'hidden'; │ │ │ │ │ + field.name = par; │ │ │ │ │ + field.value = params[par]; │ │ │ │ │ + form.appendChild(field); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var sld = this.createSLD(layer, filters, geometryAttributes); │ │ │ │ │ + return form; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - selectionLayer.mergeNewParams({ │ │ │ │ │ - SLD_BODY: sld │ │ │ │ │ - }); │ │ │ │ │ - delete this._queue; │ │ │ │ │ + /** │ │ │ │ │ + * Method: setImgSrc │ │ │ │ │ + * Sets the source for the tile image │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * url - {String} │ │ │ │ │ + */ │ │ │ │ │ + setImgSrc: function(url) { │ │ │ │ │ + if (this.useIFrame === true) { │ │ │ │ │ + if (url) { │ │ │ │ │ + var form = this.createRequestForm(); │ │ │ │ │ + this.frame.appendChild(form); │ │ │ │ │ + form.submit(); │ │ │ │ │ + this.frame.removeChild(form); │ │ │ │ │ + } else if (this.imgDiv.parentNode === this.frame) { │ │ │ │ │ + // we don't reuse iframes to avoid caching issues │ │ │ │ │ + this.frame.removeChild(this.imgDiv); │ │ │ │ │ + this.imgDiv = null; │ │ │ │ │ } │ │ │ │ │ - }; │ │ │ │ │ - this.applySelection(); │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Tile.Image.prototype.setImgSrc.apply(this, arguments); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: applySelection │ │ │ │ │ - * Checks if all required wfs data is cached, and applies the selection │ │ │ │ │ + * Method: onImageLoad │ │ │ │ │ + * Handler for the image onload event │ │ │ │ │ */ │ │ │ │ │ - applySelection: function() { │ │ │ │ │ - var canApply = true; │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - if (!this.wfsCache[this.layers[i].id]) { │ │ │ │ │ - canApply = false; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ + onImageLoad: function() { │ │ │ │ │ + //TODO de-uglify opacity handling │ │ │ │ │ + OpenLayers.Tile.Image.prototype.onImageLoad.apply(this, arguments); │ │ │ │ │ + if (this.useIFrame === true) { │ │ │ │ │ + this.imgDiv.style.opacity = 1; │ │ │ │ │ + this.frame.style.opacity = this.layer.opacity; │ │ │ │ │ } │ │ │ │ │ - canApply && this._queue.call(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.SLDSelect" │ │ │ │ │ -}); │ │ │ │ │ + /** │ │ │ │ │ + * Method: createBackBuffer │ │ │ │ │ + * Override createBackBuffer to do nothing when we use an iframe. Moving an │ │ │ │ │ + * iframe from one element to another makes it necessary to reload the iframe │ │ │ │ │ + * because its content is lost. So we just give up. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + createBackBuffer: function() { │ │ │ │ │ + var backBuffer; │ │ │ │ │ + if (this.useIFrame === false) { │ │ │ │ │ + backBuffer = OpenLayers.Tile.Image.prototype.createBackBuffer.call(this); │ │ │ │ │ + } │ │ │ │ │ + return backBuffer; │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Renderer/Canvas.js │ │ │ │ │ + OpenLayers/Strategy/BBOX.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Renderer.js │ │ │ │ │ + * @requires OpenLayers/Strategy.js │ │ │ │ │ + * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Renderer.Canvas │ │ │ │ │ - * A renderer based on the 2D 'canvas' drawing element. │ │ │ │ │ - * │ │ │ │ │ - * Inherits: │ │ │ │ │ - * - <OpenLayers.Renderer> │ │ │ │ │ + * Class: OpenLayers.Strategy.BBOX │ │ │ │ │ + * A simple strategy that reads new features when the viewport invalidates │ │ │ │ │ + * some bounds. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Strategy> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Renderer.Canvas = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: hitDetection │ │ │ │ │ - * {Boolean} Allow for hit detection of features. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - hitDetection: true, │ │ │ │ │ +OpenLayers.Strategy.BBOX = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: hitOverflow │ │ │ │ │ - * {Number} The method for converting feature identifiers to color values │ │ │ │ │ - * supports 16777215 sequential values. Two features cannot be │ │ │ │ │ - * predictably detected if their identifiers differ by more than this │ │ │ │ │ - * value. The hitOverflow allows for bigger numbers (but the │ │ │ │ │ - * difference in values is still limited). │ │ │ │ │ + * Property: bounds │ │ │ │ │ + * {<OpenLayers.Bounds>} The current data bounds (in the same projection │ │ │ │ │ + * as the layer - not always the same projection as the map). │ │ │ │ │ */ │ │ │ │ │ - hitOverflow: 0, │ │ │ │ │ + bounds: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: canvas │ │ │ │ │ - * {Canvas} The canvas context object. │ │ │ │ │ + /** │ │ │ │ │ + * Property: resolution │ │ │ │ │ + * {Float} The current data resolution. │ │ │ │ │ */ │ │ │ │ │ - canvas: null, │ │ │ │ │ + resolution: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Object} Internal object of feature/style pairs for use in redrawing the layer. │ │ │ │ │ + * APIProperty: ratio │ │ │ │ │ + * {Float} The ratio of the data bounds to the viewport bounds (in each │ │ │ │ │ + * dimension). Default is 2. │ │ │ │ │ */ │ │ │ │ │ - features: null, │ │ │ │ │ + ratio: 2, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: pendingRedraw │ │ │ │ │ - * {Boolean} The renderer needs a redraw call to render features added while │ │ │ │ │ - * the renderer was locked. │ │ │ │ │ + /** │ │ │ │ │ + * Property: resFactor │ │ │ │ │ + * {Float} Optional factor used to determine when previously requested │ │ │ │ │ + * features are invalid. If set, the resFactor will be compared to the │ │ │ │ │ + * resolution of the previous request to the current map resolution. │ │ │ │ │ + * If resFactor > (old / new) and 1/resFactor < (old / new). If you │ │ │ │ │ + * set a resFactor of 1, data will be requested every time the │ │ │ │ │ + * resolution changes. If you set a resFactor of 3, data will be │ │ │ │ │ + * requested if the old resolution is 3 times the new, or if the new is │ │ │ │ │ + * 3 times the old. If the old bounds do not contain the new bounds │ │ │ │ │ + * new data will always be requested (with or without considering │ │ │ │ │ + * resFactor). │ │ │ │ │ */ │ │ │ │ │ - pendingRedraw: false, │ │ │ │ │ + resFactor: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: cachedSymbolBounds │ │ │ │ │ - * {Object} Internal cache of calculated symbol extents. │ │ │ │ │ + * Property: response │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} The protocol response object returned │ │ │ │ │ + * by the layer protocol. │ │ │ │ │ */ │ │ │ │ │ - cachedSymbolBounds: {}, │ │ │ │ │ + response: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Renderer.Canvas │ │ │ │ │ + * Constructor: OpenLayers.Strategy.BBOX │ │ │ │ │ + * Create a new BBOX strategy. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * containerID - {<String>} │ │ │ │ │ - * options - {Object} Optional properties to be set on the renderer. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(containerID, options) { │ │ │ │ │ - OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.root = document.createElement("canvas"); │ │ │ │ │ - this.container.appendChild(this.root); │ │ │ │ │ - this.canvas = this.root.getContext("2d"); │ │ │ │ │ - this.features = {}; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitCanvas = document.createElement("canvas"); │ │ │ │ │ - this.hitContext = this.hitCanvas.getContext("2d"); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setExtent │ │ │ │ │ - * Set the visible part of the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} │ │ │ │ │ - * resolutionChanged - {Boolean} │ │ │ │ │ - * │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Set up strategy with regard to reading new batches of remote data. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ - * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ - * False otherwise. │ │ │ │ │ + * {Boolean} The strategy was successfully activated. │ │ │ │ │ */ │ │ │ │ │ - setExtent: function() { │ │ │ │ │ - OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - // always redraw features │ │ │ │ │ - return false; │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "moveend": this.update, │ │ │ │ │ + "refresh": this.update, │ │ │ │ │ + "visibilitychanged": this.update, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.update(); │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: eraseGeometry │ │ │ │ │ - * Erase a geometry from the renderer. Because the Canvas renderer has │ │ │ │ │ - * 'memory' of the features that it has drawn, we have to remove the │ │ │ │ │ - * feature so it doesn't redraw. │ │ │ │ │ + /** │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Tear down strategy with regard to reading new batches of remote data. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ - eraseGeometry: function(geometry, featureId) { │ │ │ │ │ - this.eraseFeatures(this.features[featureId][0]); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + "moveend": this.update, │ │ │ │ │ + "refresh": this.update, │ │ │ │ │ + "visibilitychanged": this.update, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: supported │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the browser supports the renderer class │ │ │ │ │ + * Method: update │ │ │ │ │ + * Callback function called on "moveend" or "refresh" layer events. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will determine │ │ │ │ │ + * the behaviour of this Strategy │ │ │ │ │ + * │ │ │ │ │ + * Valid options include: │ │ │ │ │ + * force - {Boolean} if true, new data must be unconditionally read. │ │ │ │ │ + * noAbort - {Boolean} if true, do not abort previous requests. │ │ │ │ │ */ │ │ │ │ │ - supported: function() { │ │ │ │ │ - return OpenLayers.CANVAS_SUPPORTED; │ │ │ │ │ + update: function(options) { │ │ │ │ │ + var mapBounds = this.getMapBounds(); │ │ │ │ │ + if (mapBounds !== null && ((options && options.force) || │ │ │ │ │ + (this.layer.visibility && this.layer.calculateInRange() && this.invalidBounds(mapBounds)))) { │ │ │ │ │ + this.calculateBounds(mapBounds); │ │ │ │ │ + this.resolution = this.layer.map.getResolution(); │ │ │ │ │ + this.triggerRead(options); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setSize │ │ │ │ │ - * Sets the size of the drawing surface. │ │ │ │ │ - * │ │ │ │ │ - * Once the size is updated, redraw the canvas. │ │ │ │ │ + * Method: getMapBounds │ │ │ │ │ + * Get the map bounds expressed in the same projection as this layer. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} Map bounds in the projection of the layer. │ │ │ │ │ */ │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - this.size = size.clone(); │ │ │ │ │ - var root = this.root; │ │ │ │ │ - root.style.width = size.w + "px"; │ │ │ │ │ - root.style.height = size.h + "px"; │ │ │ │ │ - root.width = size.w; │ │ │ │ │ - root.height = size.h; │ │ │ │ │ - this.resolution = null; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - var hitCanvas = this.hitCanvas; │ │ │ │ │ - hitCanvas.style.width = size.w + "px"; │ │ │ │ │ - hitCanvas.style.height = size.h + "px"; │ │ │ │ │ - hitCanvas.width = size.w; │ │ │ │ │ - hitCanvas.height = size.h; │ │ │ │ │ + getMapBounds: function() { │ │ │ │ │ + if (this.layer.map === null) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + var bounds = this.layer.map.getExtent(); │ │ │ │ │ + if (bounds && !this.layer.projection.equals( │ │ │ │ │ + this.layer.map.getProjectionObject())) { │ │ │ │ │ + bounds = bounds.clone().transform( │ │ │ │ │ + this.layer.map.getProjectionObject(), this.layer.projection │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ + return bounds; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawFeature │ │ │ │ │ - * Draw the feature. Stores the feature in the features list, │ │ │ │ │ - * then redraws the layer. │ │ │ │ │ + * Method: invalidBounds │ │ │ │ │ + * Determine whether the previously requested set of features is invalid. │ │ │ │ │ + * This occurs when the new map bounds do not contain the previously │ │ │ │ │ + * requested bounds. In addition, if <resFactor> is set, it will be │ │ │ │ │ + * considered. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * style - {<Object>} │ │ │ │ │ + * mapBounds - {<OpenLayers.Bounds>} the current map extent, will be │ │ │ │ │ + * retrieved from the map object if not provided │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The feature has been drawn completely. If the feature has no │ │ │ │ │ - * geometry, undefined will be returned. If the feature is not rendered │ │ │ │ │ - * for other reasons, false will be returned. │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - drawFeature: function(feature, style) { │ │ │ │ │ - var rendered; │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - style = this.applyDefaultSymbolizer(style || feature.style); │ │ │ │ │ - // don't render if display none or feature outside extent │ │ │ │ │ - var bounds = feature.geometry.getBounds(); │ │ │ │ │ - │ │ │ │ │ - var worldBounds; │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - worldBounds = this.map.getMaxExtent(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var intersects = bounds && bounds.intersectsBounds(this.extent, { │ │ │ │ │ - worldBounds: worldBounds │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - rendered = (style.display !== "none") && !!bounds && intersects; │ │ │ │ │ - if (rendered) { │ │ │ │ │ - // keep track of what we have rendered for redraw │ │ │ │ │ - this.features[feature.id] = [feature, style]; │ │ │ │ │ - } else { │ │ │ │ │ - // remove from features tracked for redraw │ │ │ │ │ - delete(this.features[feature.id]); │ │ │ │ │ - } │ │ │ │ │ - this.pendingRedraw = true; │ │ │ │ │ + invalidBounds: function(mapBounds) { │ │ │ │ │ + if (!mapBounds) { │ │ │ │ │ + mapBounds = this.getMapBounds(); │ │ │ │ │ } │ │ │ │ │ - if (this.pendingRedraw && !this.locked) { │ │ │ │ │ - this.redraw(); │ │ │ │ │ - this.pendingRedraw = false; │ │ │ │ │ + var invalid = !this.bounds || !this.bounds.containsBounds(mapBounds); │ │ │ │ │ + if (!invalid && this.resFactor) { │ │ │ │ │ + var ratio = this.resolution / this.layer.map.getResolution(); │ │ │ │ │ + invalid = (ratio >= this.resFactor || ratio <= (1 / this.resFactor)); │ │ │ │ │ } │ │ │ │ │ - return rendered; │ │ │ │ │ + return invalid; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawGeometry │ │ │ │ │ - * Used when looping (in redraw) over the features; draws │ │ │ │ │ - * the canvas. │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateBounds │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ + * mapBounds - {<OpenLayers.Bounds>} the current map extent, will be │ │ │ │ │ + * retrieved from the map object if not provided │ │ │ │ │ */ │ │ │ │ │ - drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ - var className = geometry.CLASS_NAME; │ │ │ │ │ - if ((className == "OpenLayers.Geometry.Collection") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiPolygon")) { │ │ │ │ │ - for (var i = 0; i < geometry.components.length; i++) { │ │ │ │ │ - this.drawGeometry(geometry.components[i], style, featureId); │ │ │ │ │ - } │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - this.drawPoint(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - this.drawLineString(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - this.drawPolygon(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break; │ │ │ │ │ + calculateBounds: function(mapBounds) { │ │ │ │ │ + if (!mapBounds) { │ │ │ │ │ + mapBounds = this.getMapBounds(); │ │ │ │ │ } │ │ │ │ │ + var center = mapBounds.getCenterLonLat(); │ │ │ │ │ + var dataWidth = mapBounds.getWidth() * this.ratio; │ │ │ │ │ + var dataHeight = mapBounds.getHeight() * this.ratio; │ │ │ │ │ + this.bounds = new OpenLayers.Bounds( │ │ │ │ │ + center.lon - (dataWidth / 2), │ │ │ │ │ + center.lat - (dataHeight / 2), │ │ │ │ │ + center.lon + (dataWidth / 2), │ │ │ │ │ + center.lat + (dataHeight / 2) │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawExternalGraphic │ │ │ │ │ - * Called to draw External graphics. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * Method: triggerRead │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Additional options for the protocol's read method │ │ │ │ │ + * (optional) │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} The protocol response object │ │ │ │ │ + * returned by the layer protocol. │ │ │ │ │ */ │ │ │ │ │ - drawExternalGraphic: function(geometry, style, featureId) { │ │ │ │ │ - var img = new Image(); │ │ │ │ │ - │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - img.title = title; │ │ │ │ │ + triggerRead: function(options) { │ │ │ │ │ + if (this.response && !(options && options.noAbort === true)) { │ │ │ │ │ + this.layer.protocol.abort(this.response); │ │ │ │ │ + this.layer.events.triggerEvent("loadend"); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ - style.graphicXOffset : -(0.5 * width); │ │ │ │ │ - var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ - style.graphicYOffset : -(0.5 * height); │ │ │ │ │ - │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ - │ │ │ │ │ - var onLoad = function() { │ │ │ │ │ - if (!this.features[featureId]) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ - if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ - var x = (p0 + xOffset) | 0; │ │ │ │ │ - var y = (p1 + yOffset) | 0; │ │ │ │ │ - var canvas = this.canvas; │ │ │ │ │ - canvas.globalAlpha = opacity; │ │ │ │ │ - var factor = OpenLayers.Renderer.Canvas.drawImageScaleFactor || │ │ │ │ │ - (OpenLayers.Renderer.Canvas.drawImageScaleFactor = │ │ │ │ │ - /android 2.1/.test(navigator.userAgent.toLowerCase()) ? │ │ │ │ │ - // 320 is the screen width of the G1 phone, for │ │ │ │ │ - // which drawImage works out of the box. │ │ │ │ │ - 320 / window.screen.width : 1 │ │ │ │ │ - ); │ │ │ │ │ - canvas.drawImage( │ │ │ │ │ - img, x * factor, y * factor, width * factor, height * factor │ │ │ │ │ - ); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId); │ │ │ │ │ - this.hitContext.fillRect(x, y, width, height); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + var evt = { │ │ │ │ │ + filter: this.createFilter() │ │ │ │ │ }; │ │ │ │ │ - │ │ │ │ │ - img.onload = OpenLayers.Function.bind(onLoad, this); │ │ │ │ │ - img.src = style.externalGraphic; │ │ │ │ │ + this.layer.events.triggerEvent("loadstart", evt); │ │ │ │ │ + this.response = this.layer.protocol.read( │ │ │ │ │ + OpenLayers.Util.applyDefaults({ │ │ │ │ │ + filter: evt.filter, │ │ │ │ │ + callback: this.merge, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options)); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawNamedSymbol │ │ │ │ │ - * Called to draw Well Known Graphic Symbol Name. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ + * Method: createFilter │ │ │ │ │ + * Creates a spatial BBOX filter. If the layer that this strategy belongs │ │ │ │ │ + * to has a filter property, this filter will be combined with the BBOX │ │ │ │ │ + * filter. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * Returns │ │ │ │ │ + * {<OpenLayers.Filter>} The filter object. │ │ │ │ │ */ │ │ │ │ │ - drawNamedSymbol: function(geometry, style, featureId) { │ │ │ │ │ - var x, y, cx, cy, i, symbolBounds, scaling, angle; │ │ │ │ │ - var unscaledStrokeWidth; │ │ │ │ │ - var deg2rad = Math.PI / 180.0; │ │ │ │ │ - │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[style.graphicName]; │ │ │ │ │ - │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(style.graphicName + ' is not a valid symbol name'); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (!symbol.length || symbol.length < 2) return; │ │ │ │ │ - │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ - │ │ │ │ │ - if (isNaN(p0) || isNaN(p1)) return; │ │ │ │ │ - │ │ │ │ │ - // Use rounded line caps │ │ │ │ │ - this.canvas.lineCap = "round"; │ │ │ │ │ - this.canvas.lineJoin = "round"; │ │ │ │ │ - │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.lineCap = "round"; │ │ │ │ │ - this.hitContext.lineJoin = "round"; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // Scale and rotate symbols, using precalculated bounds whenever possible. │ │ │ │ │ - if (style.graphicName in this.cachedSymbolBounds) { │ │ │ │ │ - symbolBounds = this.cachedSymbolBounds[style.graphicName]; │ │ │ │ │ - } else { │ │ │ │ │ - symbolBounds = new OpenLayers.Bounds(); │ │ │ │ │ - for (i = 0; i < symbol.length; i += 2) { │ │ │ │ │ - symbolBounds.extend(new OpenLayers.LonLat(symbol[i], symbol[i + 1])); │ │ │ │ │ - } │ │ │ │ │ - this.cachedSymbolBounds[style.graphicName] = symbolBounds; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // Push symbol scaling, translation and rotation onto the transformation stack in reverse order. │ │ │ │ │ - // Don't forget to apply all canvas transformations to the hitContext canvas as well(!) │ │ │ │ │ - this.canvas.save(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.save(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // Step 3: place symbol at the desired location │ │ │ │ │ - this.canvas.translate(p0, p1); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.translate(p0, p1); │ │ │ │ │ + createFilter: function() { │ │ │ │ │ + var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ + value: this.bounds, │ │ │ │ │ + projection: this.layer.projection │ │ │ │ │ + }); │ │ │ │ │ + if (this.layer.filter) { │ │ │ │ │ + filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ + type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ + filters: [this.layer.filter, filter] │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ + return filter; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Step 2a. rotate the symbol if necessary │ │ │ │ │ - angle = deg2rad * style.rotation; // will be NaN when style.rotation is undefined. │ │ │ │ │ - if (!isNaN(angle)) { │ │ │ │ │ - this.canvas.rotate(angle); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.rotate(angle); │ │ │ │ │ + /** │ │ │ │ │ + * Method: merge │ │ │ │ │ + * Given a list of features, determine which ones to add to the layer. │ │ │ │ │ + * If the layer projection differs from the map projection, features │ │ │ │ │ + * will be transformed from the layer projection to the map projection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object passed │ │ │ │ │ + * by the protocol. │ │ │ │ │ + */ │ │ │ │ │ + merge: function(resp) { │ │ │ │ │ + this.layer.destroyFeatures(); │ │ │ │ │ + if (resp.success()) { │ │ │ │ │ + var features = resp.features; │ │ │ │ │ + if (features && features.length > 0) { │ │ │ │ │ + var remote = this.layer.projection; │ │ │ │ │ + var local = this.layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var geom; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + geom = features[i].geometry; │ │ │ │ │ + if (geom) { │ │ │ │ │ + geom.transform(remote, local); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.layer.addFeatures(features); │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + this.bounds = null; │ │ │ │ │ } │ │ │ │ │ + this.response = null; │ │ │ │ │ + this.layer.events.triggerEvent("loadend", { │ │ │ │ │ + response: resp │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // // Step 2: scale symbol such that pointRadius equals half the maximum symbol dimension. │ │ │ │ │ - scaling = 2.0 * style.pointRadius / Math.max(symbolBounds.getWidth(), symbolBounds.getHeight()); │ │ │ │ │ - this.canvas.scale(scaling, scaling); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.scale(scaling, scaling); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // Step 1: center the symbol at the origin │ │ │ │ │ - cx = symbolBounds.getCenterLonLat().lon; │ │ │ │ │ - cy = symbolBounds.getCenterLonLat().lat; │ │ │ │ │ - this.canvas.translate(-cx, -cy); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.translate(-cx, -cy); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // Don't forget to scale stroke widths, because they are affected by canvas scale transformations as well(!) │ │ │ │ │ - // Alternative: scale symbol coordinates manually, so stroke width scaling is not needed anymore. │ │ │ │ │ - unscaledStrokeWidth = style.strokeWidth; │ │ │ │ │ - style.strokeWidth = unscaledStrokeWidth / scaling; │ │ │ │ │ - │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.canvas.lineTo(x, y); │ │ │ │ │ - } │ │ │ │ │ - this.canvas.closePath(); │ │ │ │ │ - this.canvas.fill(); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.BBOX" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Strategy/Filter.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.hitContext.lineTo(x, y); │ │ │ │ │ - } │ │ │ │ │ - this.hitContext.closePath(); │ │ │ │ │ - this.hitContext.fill(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.canvas.lineTo(x, y); │ │ │ │ │ - } │ │ │ │ │ - this.canvas.closePath(); │ │ │ │ │ - this.canvas.stroke(); │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Strategy.js │ │ │ │ │ + * @requires OpenLayers/Filter.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Strategy.Filter │ │ │ │ │ + * Strategy for limiting features that get added to a layer by │ │ │ │ │ + * evaluating a filter. The strategy maintains a cache of │ │ │ │ │ + * all features until removeFeatures is called on the layer. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Strategy> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Strategy.Filter = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style, scaling); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.hitContext.moveTo(x, y); │ │ │ │ │ - this.hitContext.lineTo(x, y); │ │ │ │ │ - } │ │ │ │ │ - this.hitContext.closePath(); │ │ │ │ │ - this.hitContext.stroke(); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: filter │ │ │ │ │ + * {<OpenLayers.Filter>} Filter for limiting features sent to the layer. │ │ │ │ │ + * Use the <setFilter> method to update this filter after construction. │ │ │ │ │ + */ │ │ │ │ │ + filter: null, │ │ │ │ │ │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: cache │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} List of currently cached │ │ │ │ │ + * features. │ │ │ │ │ + */ │ │ │ │ │ + cache: null, │ │ │ │ │ │ │ │ │ │ - style.strokeWidth = unscaledStrokeWidth; │ │ │ │ │ - this.canvas.restore(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.restore(); │ │ │ │ │ - } │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: caching │ │ │ │ │ + * {Boolean} The filter is currently caching features. │ │ │ │ │ + */ │ │ │ │ │ + caching: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setCanvasStyle │ │ │ │ │ - * Prepare the canvas for drawing by setting various global settings. │ │ │ │ │ + * Constructor: OpenLayers.Strategy.Filter │ │ │ │ │ + * Create a new filter strategy. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * type - {String} one of 'stroke', 'fill', or 'reset' │ │ │ │ │ - * style - {Object} Symbolizer hash │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ */ │ │ │ │ │ - setCanvasStyle: function(type, style) { │ │ │ │ │ - if (type === "fill") { │ │ │ │ │ - this.canvas.globalAlpha = style['fillOpacity']; │ │ │ │ │ - this.canvas.fillStyle = style['fillColor']; │ │ │ │ │ - } else if (type === "stroke") { │ │ │ │ │ - this.canvas.globalAlpha = style['strokeOpacity']; │ │ │ │ │ - this.canvas.strokeStyle = style['strokeColor']; │ │ │ │ │ - this.canvas.lineWidth = style['strokeWidth']; │ │ │ │ │ - } else { │ │ │ │ │ - this.canvas.globalAlpha = 0; │ │ │ │ │ - this.canvas.lineWidth = 1; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: featureIdToHex │ │ │ │ │ - * Convert a feature ID string into an RGB hex string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureId - {String} Feature id │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ + * By default, this strategy automatically activates itself when a layer │ │ │ │ │ + * is added to a map. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} RGB hex string. │ │ │ │ │ + * {Boolean} True if the strategy was successfully activated or false if │ │ │ │ │ + * the strategy was already active. │ │ │ │ │ */ │ │ │ │ │ - featureIdToHex: function(featureId) { │ │ │ │ │ - var id = Number(featureId.split("_").pop()) + 1; // zero for no feature │ │ │ │ │ - if (id >= 16777216) { │ │ │ │ │ - this.hitOverflow = id - 16777215; │ │ │ │ │ - id = id % 16777216 + 1; │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.cache = []; │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "beforefeaturesadded": this.handleAdd, │ │ │ │ │ + "beforefeaturesremoved": this.handleRemove, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - var hex = "000000" + id.toString(16); │ │ │ │ │ - var len = hex.length; │ │ │ │ │ - hex = "#" + hex.substring(len - 6, len); │ │ │ │ │ - return hex; │ │ │ │ │ + return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setHitContextStyle │ │ │ │ │ - * Prepare the hit canvas for drawing by setting various global settings. │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the strategy. Clear the feature cache. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * type - {String} one of 'stroke', 'fill', or 'reset' │ │ │ │ │ - * featureId - {String} The feature id. │ │ │ │ │ - * symbolizer - {<OpenLayers.Symbolizer>} The symbolizer. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} True if the strategy was successfully deactivated or false if │ │ │ │ │ + * the strategy was already inactive. │ │ │ │ │ */ │ │ │ │ │ - setHitContextStyle: function(type, featureId, symbolizer, strokeScaling) { │ │ │ │ │ - var hex = this.featureIdToHex(featureId); │ │ │ │ │ - if (type == "fill") { │ │ │ │ │ - this.hitContext.globalAlpha = 1.0; │ │ │ │ │ - this.hitContext.fillStyle = hex; │ │ │ │ │ - } else if (type == "stroke") { │ │ │ │ │ - this.hitContext.globalAlpha = 1.0; │ │ │ │ │ - this.hitContext.strokeStyle = hex; │ │ │ │ │ - // bump up stroke width to deal with antialiasing. If strokeScaling is defined, we're rendering a symbol │ │ │ │ │ - // on a transformed canvas, so the antialias width bump has to scale as well. │ │ │ │ │ - if (typeof strokeScaling === "undefined") { │ │ │ │ │ - this.hitContext.lineWidth = symbolizer.strokeWidth + 2; │ │ │ │ │ - } else { │ │ │ │ │ - if (!isNaN(strokeScaling)) { │ │ │ │ │ - this.hitContext.lineWidth = symbolizer.strokeWidth + 2.0 / strokeScaling; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.hitContext.globalAlpha = 0; │ │ │ │ │ - this.hitContext.lineWidth = 1; │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + this.cache = null; │ │ │ │ │ + if (this.layer && this.layer.events) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + "beforefeaturesadded": this.handleAdd, │ │ │ │ │ + "beforefeaturesremoved": this.handleRemove, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ + return OpenLayers.Strategy.prototype.deactivate.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPoint │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * Method: handleAdd │ │ │ │ │ */ │ │ │ │ │ - drawPoint: function(geometry, style, featureId) { │ │ │ │ │ - if (style.graphic !== false) { │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - this.drawExternalGraphic(geometry, style, featureId); │ │ │ │ │ - } else if (style.graphicName && (style.graphicName != "circle")) { │ │ │ │ │ - this.drawNamedSymbol(geometry, style, featureId); │ │ │ │ │ - } else { │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ - if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ - var twoPi = Math.PI * 2; │ │ │ │ │ - var radius = style.pointRadius; │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.canvas.fill(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.hitContext.fill(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.canvas.stroke(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.hitContext.stroke(); │ │ │ │ │ - } │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ - } │ │ │ │ │ + handleAdd: function(event) { │ │ │ │ │ + if (!this.caching && this.filter) { │ │ │ │ │ + var features = event.features; │ │ │ │ │ + event.features = []; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0, ii = features.length; i < ii; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (this.filter.evaluate(feature)) { │ │ │ │ │ + event.features.push(feature); │ │ │ │ │ + } else { │ │ │ │ │ + this.cache.push(feature); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLineString │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * Method: handleRemove │ │ │ │ │ */ │ │ │ │ │ - drawLineString: function(geometry, style, featureId) { │ │ │ │ │ - style = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - fill: false │ │ │ │ │ - }, style); │ │ │ │ │ - this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ + handleRemove: function(event) { │ │ │ │ │ + if (!this.caching) { │ │ │ │ │ + this.cache = []; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawLinearRing │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setFilter │ │ │ │ │ + * Update the filter for this strategy. This will re-evaluate │ │ │ │ │ + * any features on the layer and in the cache. Only features │ │ │ │ │ + * for which filter.evalute(feature) returns true will be │ │ │ │ │ + * added to the layer. Others will be cached by the strategy. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {<OpenLayers.Filter>} A filter for evaluating features. │ │ │ │ │ */ │ │ │ │ │ - drawLinearRing: function(geometry, style, featureId) { │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.renderPath(this.canvas, geometry, style, featureId, "fill"); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.renderPath(this.hitContext, geometry, style, featureId, "fill"); │ │ │ │ │ - } │ │ │ │ │ + setFilter: function(filter) { │ │ │ │ │ + this.filter = filter; │ │ │ │ │ + var previousCache = this.cache; │ │ │ │ │ + this.cache = []; │ │ │ │ │ + // look through layer for features to remove from layer │ │ │ │ │ + this.handleAdd({ │ │ │ │ │ + features: this.layer.features │ │ │ │ │ + }); │ │ │ │ │ + // cache now contains features to remove from layer │ │ │ │ │ + if (this.cache.length > 0) { │ │ │ │ │ + this.caching = true; │ │ │ │ │ + this.layer.removeFeatures(this.cache.slice()); │ │ │ │ │ + this.caching = false; │ │ │ │ │ } │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.renderPath(this.canvas, geometry, style, featureId, "stroke"); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ - this.renderPath(this.hitContext, geometry, style, featureId, "stroke"); │ │ │ │ │ + // now look through previous cache for features to add to layer │ │ │ │ │ + if (previousCache.length > 0) { │ │ │ │ │ + var event = { │ │ │ │ │ + features: previousCache │ │ │ │ │ + }; │ │ │ │ │ + this.handleAdd(event); │ │ │ │ │ + if (event.features.length > 0) { │ │ │ │ │ + // event has features to add to layer │ │ │ │ │ + this.caching = true; │ │ │ │ │ + this.layer.addFeatures(event.features); │ │ │ │ │ + this.caching = false; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Filter" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Strategy/Refresh.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Strategy.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Strategy.Refresh │ │ │ │ │ + * A strategy that refreshes the layer. By default the strategy waits for a │ │ │ │ │ + * call to <refresh> before refreshing. By configuring the strategy with │ │ │ │ │ + * the <interval> option, refreshing can take place automatically. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Strategy> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Strategy.Refresh = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: renderPath │ │ │ │ │ - * Render a path with stroke and optional fill. │ │ │ │ │ + * Property: force │ │ │ │ │ + * {Boolean} Force a refresh on the layer. Default is false. │ │ │ │ │ */ │ │ │ │ │ - renderPath: function(context, geometry, style, featureId, type) { │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - context.beginPath(); │ │ │ │ │ - var start = this.getLocalXY(components[0]); │ │ │ │ │ - var x = start[0]; │ │ │ │ │ - var y = start[1]; │ │ │ │ │ - if (!isNaN(x) && !isNaN(y)) { │ │ │ │ │ - context.moveTo(start[0], start[1]); │ │ │ │ │ - for (var i = 1; i < len; ++i) { │ │ │ │ │ - var pt = this.getLocalXY(components[i]); │ │ │ │ │ - context.lineTo(pt[0], pt[1]); │ │ │ │ │ - } │ │ │ │ │ - if (type === "fill") { │ │ │ │ │ - context.fill(); │ │ │ │ │ - } else { │ │ │ │ │ - context.stroke(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + force: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPolygon │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * Property: interval │ │ │ │ │ + * {Number} Auto-refresh. Default is 0. If > 0, layer will be refreshed │ │ │ │ │ + * every N milliseconds. │ │ │ │ │ */ │ │ │ │ │ - drawPolygon: function(geometry, style, featureId) { │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - this.drawLinearRing(components[0], style, featureId); │ │ │ │ │ - // erase inner rings │ │ │ │ │ - for (var i = 1; i < len; ++i) { │ │ │ │ │ - /** │ │ │ │ │ - * Note that this is overly agressive. Here we punch holes through │ │ │ │ │ - * all previously rendered features on the same canvas. A better │ │ │ │ │ - * solution for polygons with interior rings would be to draw the │ │ │ │ │ - * polygon on a sketch canvas first. We could erase all holes │ │ │ │ │ - * there and then copy the drawing to the layer canvas. │ │ │ │ │ - * TODO: http://trac.osgeo.org/openlayers/ticket/3130 │ │ │ │ │ - */ │ │ │ │ │ - this.canvas.globalCompositeOperation = "destination-out"; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.globalCompositeOperation = "destination-out"; │ │ │ │ │ - } │ │ │ │ │ - this.drawLinearRing( │ │ │ │ │ - components[i], │ │ │ │ │ - OpenLayers.Util.applyDefaults({ │ │ │ │ │ - stroke: false, │ │ │ │ │ - fillOpacity: 1.0 │ │ │ │ │ - }, style), │ │ │ │ │ - featureId │ │ │ │ │ - ); │ │ │ │ │ - this.canvas.globalCompositeOperation = "source-over"; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.globalCompositeOperation = "source-over"; │ │ │ │ │ - } │ │ │ │ │ - this.drawLinearRing( │ │ │ │ │ - components[i], │ │ │ │ │ - OpenLayers.Util.applyDefaults({ │ │ │ │ │ - fill: false │ │ │ │ │ - }, style), │ │ │ │ │ - featureId │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + interval: 0, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawText │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ + * Property: timer │ │ │ │ │ + * {Number} The id of the timer. │ │ │ │ │ + */ │ │ │ │ │ + timer: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Strategy.Refresh │ │ │ │ │ + * Create a new Refresh strategy. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * location - {<OpenLayers.Point>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ */ │ │ │ │ │ - drawText: function(location, style) { │ │ │ │ │ - var pt = this.getLocalXY(location); │ │ │ │ │ │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ - this.canvas.fillStyle = style.fontColor; │ │ │ │ │ - this.canvas.globalAlpha = style.fontOpacity || 1.0; │ │ │ │ │ - var fontStyle = [style.fontStyle ? style.fontStyle : "normal", │ │ │ │ │ - "normal", // "font-variant" not supported │ │ │ │ │ - style.fontWeight ? style.fontWeight : "normal", │ │ │ │ │ - style.fontSize ? style.fontSize : "1em", │ │ │ │ │ - style.fontFamily ? style.fontFamily : "sans-serif" │ │ │ │ │ - ].join(" "); │ │ │ │ │ - var labelRows = style.label.split('\n'); │ │ │ │ │ - var numRows = labelRows.length; │ │ │ │ │ - if (this.canvas.fillText) { │ │ │ │ │ - // HTML5 │ │ │ │ │ - this.canvas.font = fontStyle; │ │ │ │ │ - this.canvas.textAlign = │ │ │ │ │ - OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[0]] || │ │ │ │ │ - "center"; │ │ │ │ │ - this.canvas.textBaseline = │ │ │ │ │ - OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[1]] || │ │ │ │ │ - "middle"; │ │ │ │ │ - var vfactor = │ │ │ │ │ - OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5; │ │ │ │ │ - } │ │ │ │ │ - var lineHeight = │ │ │ │ │ - this.canvas.measureText('Mg').height || │ │ │ │ │ - this.canvas.measureText('xx').width; │ │ │ │ │ - pt[1] += lineHeight * vfactor * (numRows - 1); │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - if (style.labelOutlineWidth) { │ │ │ │ │ - this.canvas.save(); │ │ │ │ │ - this.canvas.globalAlpha = style.labelOutlineOpacity || style.fontOpacity || 1.0; │ │ │ │ │ - this.canvas.strokeStyle = style.labelOutlineColor; │ │ │ │ │ - this.canvas.lineWidth = style.labelOutlineWidth; │ │ │ │ │ - this.canvas.strokeText(labelRows[i], pt[0], pt[1] + (lineHeight * i) + 1); │ │ │ │ │ - this.canvas.restore(); │ │ │ │ │ - } │ │ │ │ │ - this.canvas.fillText(labelRows[i], pt[0], pt[1] + (lineHeight * i)); │ │ │ │ │ - } │ │ │ │ │ - } else if (this.canvas.mozDrawText) { │ │ │ │ │ - // Mozilla pre-Gecko1.9.1 (<FF3.1) │ │ │ │ │ - this.canvas.mozTextStyle = fontStyle; │ │ │ │ │ - // No built-in text alignment, so we measure and adjust the position │ │ │ │ │ - var hfactor = │ │ │ │ │ - OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[0]]; │ │ │ │ │ - if (hfactor == null) { │ │ │ │ │ - hfactor = -.5; │ │ │ │ │ - } │ │ │ │ │ - var vfactor = │ │ │ │ │ - OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5; │ │ │ │ │ - } │ │ │ │ │ - var lineHeight = this.canvas.mozMeasureText('xx'); │ │ │ │ │ - pt[1] += lineHeight * (1 + (vfactor * numRows)); │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - var x = pt[0] + (hfactor * this.canvas.mozMeasureText(labelRows[i])); │ │ │ │ │ - var y = pt[1] + (i * lineHeight); │ │ │ │ │ - this.canvas.translate(x, y); │ │ │ │ │ - this.canvas.mozDrawText(labelRows[i]); │ │ │ │ │ - this.canvas.translate(-x, -y); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} True if the strategy was successfully activated. │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + if (this.layer.visibility === true) { │ │ │ │ │ + this.start(); │ │ │ │ │ } │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "visibilitychanged": this.reset, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ + return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getLocalXY │ │ │ │ │ - * transform geographic xy into pixel xy │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ + * tear-down. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} True if the strategy was successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ - getLocalXY: function(point) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var extent = this.extent; │ │ │ │ │ - var x = ((point.x - this.featureDx) / resolution + (-extent.left / resolution)); │ │ │ │ │ - var y = ((extent.top / resolution) - point.y / resolution); │ │ │ │ │ - return [x, y]; │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.stop(); │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + "visibilitychanged": this.reset, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clear │ │ │ │ │ - * Clear all vectors from the renderer. │ │ │ │ │ + * Method: reset │ │ │ │ │ + * Start or cancel the refresh interval depending on the visibility of │ │ │ │ │ + * the layer. │ │ │ │ │ */ │ │ │ │ │ - clear: function() { │ │ │ │ │ - var height = this.root.height; │ │ │ │ │ - var width = this.root.width; │ │ │ │ │ - this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ - this.features = {}; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.clearRect(0, 0, width, height); │ │ │ │ │ + reset: function() { │ │ │ │ │ + if (this.layer.visibility === true) { │ │ │ │ │ + this.start(); │ │ │ │ │ + } else { │ │ │ │ │ + this.stop(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureIdFromEvent │ │ │ │ │ - * Returns a feature id from an event on the renderer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector} A feature or undefined. This method returns a │ │ │ │ │ - * feature instead of a feature id to avoid an unnecessary lookup on the │ │ │ │ │ - * layer. │ │ │ │ │ + * Method: start │ │ │ │ │ + * Start the refresh interval. │ │ │ │ │ */ │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var featureId, feature; │ │ │ │ │ + start: function() { │ │ │ │ │ + if (this.interval && typeof this.interval === "number" && │ │ │ │ │ + this.interval > 0) { │ │ │ │ │ │ │ │ │ │ - if (this.hitDetection && this.root.style.display !== "none") { │ │ │ │ │ - // this dragging check should go in the feature handler │ │ │ │ │ - if (!this.map.dragging) { │ │ │ │ │ - var xy = evt.xy; │ │ │ │ │ - var x = xy.x | 0; │ │ │ │ │ - var y = xy.y | 0; │ │ │ │ │ - var data = this.hitContext.getImageData(x, y, 1, 1).data; │ │ │ │ │ - if (data[3] === 255) { // antialiased │ │ │ │ │ - var id = data[2] + (256 * (data[1] + (256 * data[0]))); │ │ │ │ │ - if (id) { │ │ │ │ │ - featureId = "OpenLayers_Feature_Vector_" + (id - 1 + this.hitOverflow); │ │ │ │ │ - try { │ │ │ │ │ - feature = this.features[featureId][0]; │ │ │ │ │ - } catch (err) { │ │ │ │ │ - // Because of antialiasing on the canvas, when the hit location is at a point where the edge of │ │ │ │ │ - // one symbol intersects the interior of another symbol, a wrong hit color (and therefore id) results. │ │ │ │ │ - // todo: set Antialiasing = 'off' on the hitContext as soon as browsers allow it. │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + this.timer = window.setInterval( │ │ │ │ │ + OpenLayers.Function.bind(this.refresh, this), │ │ │ │ │ + this.interval); │ │ │ │ │ } │ │ │ │ │ - return feature; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: eraseFeatures │ │ │ │ │ - * This is called by the layer to erase features; removes the feature from │ │ │ │ │ - * the list, then redraws the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * APIMethod: refresh │ │ │ │ │ + * Tell the strategy to refresh which will refresh the layer. │ │ │ │ │ */ │ │ │ │ │ - eraseFeatures: function(features) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0; i < features.length; ++i) { │ │ │ │ │ - delete this.features[features[i].id]; │ │ │ │ │ + refresh: function() { │ │ │ │ │ + if (this.layer && this.layer.refresh && │ │ │ │ │ + typeof this.layer.refresh == "function") { │ │ │ │ │ + │ │ │ │ │ + this.layer.refresh({ │ │ │ │ │ + force: this.force │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - this.redraw(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: redraw │ │ │ │ │ - * The real 'meat' of the function: any time things have changed, │ │ │ │ │ - * redraw() can be called to loop over all the data and (you guessed │ │ │ │ │ - * it) redraw it. Unlike Elements-based Renderers, we can't interact │ │ │ │ │ - * with things once they're drawn, to remove them, for example, so │ │ │ │ │ - * instead we have to just clear everything and draw from scratch. │ │ │ │ │ + * Method: stop │ │ │ │ │ + * Cancels the refresh interval. │ │ │ │ │ */ │ │ │ │ │ - redraw: function() { │ │ │ │ │ - if (!this.locked) { │ │ │ │ │ - var height = this.root.height; │ │ │ │ │ - var width = this.root.width; │ │ │ │ │ - this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.clearRect(0, 0, width, height); │ │ │ │ │ - } │ │ │ │ │ - var labelMap = []; │ │ │ │ │ - var feature, geometry, style; │ │ │ │ │ - var worldBounds = (this.map.baseLayer && this.map.baseLayer.wrapDateLine) && this.map.getMaxExtent(); │ │ │ │ │ - for (var id in this.features) { │ │ │ │ │ - if (!this.features.hasOwnProperty(id)) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - feature = this.features[id][0]; │ │ │ │ │ - geometry = feature.geometry; │ │ │ │ │ - this.calculateFeatureDx(geometry.getBounds(), worldBounds); │ │ │ │ │ - style = this.features[id][1]; │ │ │ │ │ - this.drawGeometry(geometry, style, feature.id); │ │ │ │ │ - if (style.label) { │ │ │ │ │ - labelMap.push([feature, style]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var item; │ │ │ │ │ - for (var i = 0, len = labelMap.length; i < len; ++i) { │ │ │ │ │ - item = labelMap[i]; │ │ │ │ │ - this.drawText(item[0].geometry.getCentroid(), item[1]); │ │ │ │ │ - } │ │ │ │ │ + stop: function() { │ │ │ │ │ + if (this.timer !== null) { │ │ │ │ │ + window.clearInterval(this.timer); │ │ │ │ │ + this.timer = null; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.Canvas" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Refresh" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.Canvas.LABEL_ALIGN │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.Canvas.LABEL_ALIGN = { │ │ │ │ │ - "l": "left", │ │ │ │ │ - "r": "right", │ │ │ │ │ - "t": "top", │ │ │ │ │ - "b": "bottom" │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.Canvas.LABEL_FACTOR │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.Canvas.LABEL_FACTOR = { │ │ │ │ │ - "l": 0, │ │ │ │ │ - "r": -1, │ │ │ │ │ - "t": 0, │ │ │ │ │ - "b": -1 │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.Canvas.drawImageScaleFactor │ │ │ │ │ - * {Number} Scale factor to apply to the canvas drawImage arguments. This │ │ │ │ │ - * is always 1 except for Android 2.1 devices, to work around │ │ │ │ │ - * http://code.google.com/p/android/issues/detail?id=5141. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.Canvas.drawImageScaleFactor = null; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Renderer/Elements.js │ │ │ │ │ + OpenLayers/Strategy/Cluster.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Renderer.js │ │ │ │ │ + * @requires OpenLayers/Strategy.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.ElementsIndexer │ │ │ │ │ - * This class takes care of figuring out which order elements should be │ │ │ │ │ - * placed in the DOM based on given indexing methods. │ │ │ │ │ + * Class: OpenLayers.Strategy.Cluster │ │ │ │ │ + * Strategy for vector feature clustering. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Strategy> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.ElementsIndexer = OpenLayers.Class({ │ │ │ │ │ +OpenLayers.Strategy.Cluster = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: maxZIndex │ │ │ │ │ - * {Integer} This is the largest-most z-index value for a node │ │ │ │ │ - * contained within the indexer. │ │ │ │ │ + * APIProperty: distance │ │ │ │ │ + * {Integer} Pixel distance between features that should be considered a │ │ │ │ │ + * single cluster. Default is 20 pixels. │ │ │ │ │ */ │ │ │ │ │ - maxZIndex: null, │ │ │ │ │ + distance: 20, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: order │ │ │ │ │ - * {Array<String>} This is an array of node id's stored in the │ │ │ │ │ - * order that they should show up on screen. Id's higher up in the │ │ │ │ │ - * array (higher array index) represent nodes with higher z-indeces. │ │ │ │ │ + * APIProperty: threshold │ │ │ │ │ + * {Integer} Optional threshold below which original features will be │ │ │ │ │ + * added to the layer instead of clusters. For example, a threshold │ │ │ │ │ + * of 3 would mean that any time there are 2 or fewer features in │ │ │ │ │ + * a cluster, those features will be added directly to the layer instead │ │ │ │ │ + * of a cluster representing those features. Default is null (which is │ │ │ │ │ + * equivalent to 1 - meaning that clusters may contain just one feature). │ │ │ │ │ */ │ │ │ │ │ - order: null, │ │ │ │ │ + threshold: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: indices │ │ │ │ │ - * {Object} This is a hash that maps node ids to their z-index value │ │ │ │ │ - * stored in the indexer. This is done to make finding a nodes z-index │ │ │ │ │ - * value O(1). │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} Cached features. │ │ │ │ │ */ │ │ │ │ │ - indices: null, │ │ │ │ │ + features: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: compare │ │ │ │ │ - * {Function} This is the function used to determine placement of │ │ │ │ │ - * of a new node within the indexer. If null, this defaults to to │ │ │ │ │ - * the Z_ORDER_DRAWING_ORDER comparison method. │ │ │ │ │ + * Property: clusters │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} Calculated clusters. │ │ │ │ │ */ │ │ │ │ │ - compare: null, │ │ │ │ │ + clusters: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: initialize │ │ │ │ │ - * Create a new indexer with │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * yOrdering - {Boolean} Whether to use y-ordering. │ │ │ │ │ + * Property: clustering │ │ │ │ │ + * {Boolean} The strategy is currently clustering features. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(yOrdering) { │ │ │ │ │ - │ │ │ │ │ - this.compare = yOrdering ? │ │ │ │ │ - OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_Y_ORDER : │ │ │ │ │ - OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_DRAWING_ORDER; │ │ │ │ │ - │ │ │ │ │ - this.clear(); │ │ │ │ │ - }, │ │ │ │ │ + clustering: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: insert │ │ │ │ │ - * Insert a new node into the indexer. In order to find the correct │ │ │ │ │ - * positioning for the node to be inserted, this method uses a binary │ │ │ │ │ - * search. This makes inserting O(log(n)). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newNode - {DOMElement} The new node to be inserted. │ │ │ │ │ - * │ │ │ │ │ - * Returns │ │ │ │ │ - * {DOMElement} the node before which we should insert our newNode, or │ │ │ │ │ - * null if newNode can just be appended. │ │ │ │ │ + * Property: resolution │ │ │ │ │ + * {Float} The resolution (map units per pixel) of the current cluster set. │ │ │ │ │ */ │ │ │ │ │ - insert: function(newNode) { │ │ │ │ │ - // If the node is known to the indexer, remove it so we can │ │ │ │ │ - // recalculate where it should go. │ │ │ │ │ - if (this.exists(newNode)) { │ │ │ │ │ - this.remove(newNode); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var nodeId = newNode.id; │ │ │ │ │ - │ │ │ │ │ - this.determineZIndex(newNode); │ │ │ │ │ - │ │ │ │ │ - var leftIndex = -1; │ │ │ │ │ - var rightIndex = this.order.length; │ │ │ │ │ - var middle; │ │ │ │ │ - │ │ │ │ │ - while (rightIndex - leftIndex > 1) { │ │ │ │ │ - middle = parseInt((leftIndex + rightIndex) / 2); │ │ │ │ │ - │ │ │ │ │ - var placement = this.compare(this, newNode, │ │ │ │ │ - OpenLayers.Util.getElement(this.order[middle])); │ │ │ │ │ - │ │ │ │ │ - if (placement > 0) { │ │ │ │ │ - leftIndex = middle; │ │ │ │ │ - } else { │ │ │ │ │ - rightIndex = middle; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.order.splice(rightIndex, 0, nodeId); │ │ │ │ │ - this.indices[nodeId] = this.getZIndex(newNode); │ │ │ │ │ - │ │ │ │ │ - // If the new node should be before another in the index │ │ │ │ │ - // order, return the node before which we have to insert the new one; │ │ │ │ │ - // else, return null to indicate that the new node can be appended. │ │ │ │ │ - return this.getNextElement(rightIndex); │ │ │ │ │ - }, │ │ │ │ │ + resolution: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: remove │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Strategy.Cluster │ │ │ │ │ + * Create a new clustering strategy. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} The node to be removed. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ */ │ │ │ │ │ - remove: function(node) { │ │ │ │ │ - var nodeId = node.id; │ │ │ │ │ - var arrayIndex = OpenLayers.Util.indexOf(this.order, nodeId); │ │ │ │ │ - if (arrayIndex >= 0) { │ │ │ │ │ - // Remove it from the order array, as well as deleting the node │ │ │ │ │ - // from the indeces hash. │ │ │ │ │ - this.order.splice(arrayIndex, 1); │ │ │ │ │ - delete this.indices[nodeId]; │ │ │ │ │ - │ │ │ │ │ - // Reset the maxium z-index based on the last item in the │ │ │ │ │ - // order array. │ │ │ │ │ - if (this.order.length > 0) { │ │ │ │ │ - var lastId = this.order[this.order.length - 1]; │ │ │ │ │ - this.maxZIndex = this.indices[lastId]; │ │ │ │ │ - } else { │ │ │ │ │ - this.maxZIndex = 0; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clear │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The strategy was successfully activated. │ │ │ │ │ */ │ │ │ │ │ - clear: function() { │ │ │ │ │ - this.order = []; │ │ │ │ │ - this.indices = {}; │ │ │ │ │ - this.maxZIndex = 0; │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "beforefeaturesadded": this.cacheFeatures, │ │ │ │ │ + "featuresremoved": this.clearCache, │ │ │ │ │ + "moveend": this.cluster, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: exists │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} The node to test for existence. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ + * tear-down. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Whether or not the node exists in the indexer? │ │ │ │ │ + * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ - exists: function(node) { │ │ │ │ │ - return (this.indices[node.id] != null); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.clearCache(); │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + "beforefeaturesadded": this.cacheFeatures, │ │ │ │ │ + "featuresremoved": this.clearCache, │ │ │ │ │ + "moveend": this.cluster, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getZIndex │ │ │ │ │ - * Get the z-index value for the current node from the node data itself. │ │ │ │ │ - * │ │ │ │ │ + * Method: cacheFeatures │ │ │ │ │ + * Cache features before they are added to the layer. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} The node whose z-index to get. │ │ │ │ │ - * │ │ │ │ │ + * event - {Object} The event that this was listening for. This will come │ │ │ │ │ + * with a batch of features to be clustered. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Integer} The z-index value for the specified node (from the node │ │ │ │ │ - * data itself). │ │ │ │ │ + * {Boolean} False to stop features from being added to the layer. │ │ │ │ │ */ │ │ │ │ │ - getZIndex: function(node) { │ │ │ │ │ - return node._style.graphicZIndex; │ │ │ │ │ + cacheFeatures: function(event) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + if (!this.clustering) { │ │ │ │ │ + this.clearCache(); │ │ │ │ │ + this.features = event.features; │ │ │ │ │ + this.cluster(); │ │ │ │ │ + propagate = false; │ │ │ │ │ + } │ │ │ │ │ + return propagate; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: determineZIndex │ │ │ │ │ - * Determine the z-index for the current node if there isn't one, │ │ │ │ │ - * and set the maximum value if we've found a new maximum. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ + * Method: clearCache │ │ │ │ │ + * Clear out the cached features. │ │ │ │ │ */ │ │ │ │ │ - determineZIndex: function(node) { │ │ │ │ │ - var zIndex = node._style.graphicZIndex; │ │ │ │ │ - │ │ │ │ │ - // Everything must have a zIndex. If none is specified, │ │ │ │ │ - // this means the user *must* (hint: assumption) want this │ │ │ │ │ - // node to succomb to drawing order. To enforce drawing order │ │ │ │ │ - // over all indexing methods, we'll create a new z-index that's │ │ │ │ │ - // greater than any currently in the indexer. │ │ │ │ │ - if (zIndex == null) { │ │ │ │ │ - zIndex = this.maxZIndex; │ │ │ │ │ - node._style.graphicZIndex = zIndex; │ │ │ │ │ - } else if (zIndex > this.maxZIndex) { │ │ │ │ │ - this.maxZIndex = zIndex; │ │ │ │ │ + clearCache: function() { │ │ │ │ │ + if (!this.clustering) { │ │ │ │ │ + this.features = null; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getNextElement │ │ │ │ │ - * Get the next element in the order stack. │ │ │ │ │ - * │ │ │ │ │ + * Method: cluster │ │ │ │ │ + * Cluster features based on some threshold distance. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * index - {Integer} The index of the current node in this.order. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} the node following the index passed in, or │ │ │ │ │ - * null. │ │ │ │ │ + * event - {Object} The event received when cluster is called as a │ │ │ │ │ + * result of a moveend event. │ │ │ │ │ */ │ │ │ │ │ - getNextElement: function(index) { │ │ │ │ │ - var nextIndex = index + 1; │ │ │ │ │ - if (nextIndex < this.order.length) { │ │ │ │ │ - var nextElement = OpenLayers.Util.getElement(this.order[nextIndex]); │ │ │ │ │ - if (nextElement == undefined) { │ │ │ │ │ - nextElement = this.getNextElement(nextIndex); │ │ │ │ │ - } │ │ │ │ │ - return nextElement; │ │ │ │ │ - } else { │ │ │ │ │ - return null; │ │ │ │ │ + cluster: function(event) { │ │ │ │ │ + if ((!event || event.zoomChanged) && this.features) { │ │ │ │ │ + var resolution = this.layer.map.getResolution(); │ │ │ │ │ + if (resolution != this.resolution || !this.clustersExist()) { │ │ │ │ │ + this.resolution = resolution; │ │ │ │ │ + var clusters = []; │ │ │ │ │ + var feature, clustered, cluster; │ │ │ │ │ + for (var i = 0; i < this.features.length; ++i) { │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + clustered = false; │ │ │ │ │ + for (var j = clusters.length - 1; j >= 0; --j) { │ │ │ │ │ + cluster = clusters[j]; │ │ │ │ │ + if (this.shouldCluster(cluster, feature)) { │ │ │ │ │ + this.addToCluster(cluster, feature); │ │ │ │ │ + clustered = true; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!clustered) { │ │ │ │ │ + clusters.push(this.createCluster(this.features[i])); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.clustering = true; │ │ │ │ │ + this.layer.removeAllFeatures(); │ │ │ │ │ + this.clustering = false; │ │ │ │ │ + if (clusters.length > 0) { │ │ │ │ │ + if (this.threshold > 1) { │ │ │ │ │ + var clone = clusters.slice(); │ │ │ │ │ + clusters = []; │ │ │ │ │ + var candidate; │ │ │ │ │ + for (var i = 0, len = clone.length; i < len; ++i) { │ │ │ │ │ + candidate = clone[i]; │ │ │ │ │ + if (candidate.attributes.count < this.threshold) { │ │ │ │ │ + Array.prototype.push.apply(clusters, candidate.cluster); │ │ │ │ │ + } else { │ │ │ │ │ + clusters.push(candidate); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.clustering = true; │ │ │ │ │ + // A legitimate feature addition could occur during this │ │ │ │ │ + // addFeatures call. For clustering to behave well, features │ │ │ │ │ + // should be removed from a layer before requesting a new batch. │ │ │ │ │ + this.layer.addFeatures(clusters); │ │ │ │ │ + this.clustering = false; │ │ │ │ │ + } │ │ │ │ │ + this.clusters = clusters; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.ElementsIndexer" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.ElementsIndexer.IndexingMethods │ │ │ │ │ - * These are the compare methods for figuring out where a new node should be │ │ │ │ │ - * placed within the indexer. These methods are very similar to general │ │ │ │ │ - * sorting methods in that they return -1, 0, and 1 to specify the │ │ │ │ │ - * direction in which new nodes fall in the ordering. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.ElementsIndexer.IndexingMethods = { │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: Z_ORDER │ │ │ │ │ - * This compare method is used by other comparison methods. │ │ │ │ │ - * It can be used individually for ordering, but is not recommended, │ │ │ │ │ - * because it doesn't subscribe to drawing order. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ - * newNode - {DOMElement} │ │ │ │ │ - * nextNode - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ + * Method: clustersExist │ │ │ │ │ + * Determine whether calculated clusters are already on the layer. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Integer} │ │ │ │ │ + * {Boolean} The calculated clusters are already on the layer. │ │ │ │ │ */ │ │ │ │ │ - Z_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var newZIndex = indexer.getZIndex(newNode); │ │ │ │ │ - │ │ │ │ │ - var returnVal = 0; │ │ │ │ │ - if (nextNode) { │ │ │ │ │ - var nextZIndex = indexer.getZIndex(nextNode); │ │ │ │ │ - returnVal = newZIndex - nextZIndex; │ │ │ │ │ + clustersExist: function() { │ │ │ │ │ + var exist = false; │ │ │ │ │ + if (this.clusters && this.clusters.length > 0 && │ │ │ │ │ + this.clusters.length == this.layer.features.length) { │ │ │ │ │ + exist = true; │ │ │ │ │ + for (var i = 0; i < this.clusters.length; ++i) { │ │ │ │ │ + if (this.clusters[i] != this.layer.features[i]) { │ │ │ │ │ + exist = false; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return returnVal; │ │ │ │ │ + return exist; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: Z_ORDER_DRAWING_ORDER │ │ │ │ │ - * This method orders nodes by their z-index, but does so in a way │ │ │ │ │ - * that, if there are other nodes with the same z-index, the newest │ │ │ │ │ - * drawn will be the front most within that z-index. This is the │ │ │ │ │ - * default indexing method. │ │ │ │ │ - * │ │ │ │ │ + * Method: shouldCluster │ │ │ │ │ + * Determine whether to include a feature in a given cluster. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ - * newNode - {DOMElement} │ │ │ │ │ - * nextNode - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ + * cluster - {<OpenLayers.Feature.Vector>} A cluster. │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} A feature. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Integer} │ │ │ │ │ + * {Boolean} The feature should be included in the cluster. │ │ │ │ │ */ │ │ │ │ │ - Z_ORDER_DRAWING_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER( │ │ │ │ │ - indexer, │ │ │ │ │ - newNode, │ │ │ │ │ - nextNode │ │ │ │ │ + shouldCluster: function(cluster, feature) { │ │ │ │ │ + var cc = cluster.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + var fc = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + var distance = ( │ │ │ │ │ + Math.sqrt( │ │ │ │ │ + Math.pow((cc.lon - fc.lon), 2) + Math.pow((cc.lat - fc.lat), 2) │ │ │ │ │ + ) / this.resolution │ │ │ │ │ ); │ │ │ │ │ + return (distance <= this.distance); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Make Z_ORDER subscribe to drawing order by pushing it above │ │ │ │ │ - // all of the other nodes with the same z-index. │ │ │ │ │ - if (nextNode && returnVal == 0) { │ │ │ │ │ - returnVal = 1; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return returnVal; │ │ │ │ │ + /** │ │ │ │ │ + * Method: addToCluster │ │ │ │ │ + * Add a feature to a cluster. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * cluster - {<OpenLayers.Feature.Vector>} A cluster. │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} A feature. │ │ │ │ │ + */ │ │ │ │ │ + addToCluster: function(cluster, feature) { │ │ │ │ │ + cluster.cluster.push(feature); │ │ │ │ │ + cluster.attributes.count += 1; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: Z_ORDER_Y_ORDER │ │ │ │ │ - * This one should really be called Z_ORDER_Y_ORDER_DRAWING_ORDER, as it │ │ │ │ │ - * best describes which ordering methods have precedence (though, the │ │ │ │ │ - * name would be too long). This method orders nodes by their z-index, │ │ │ │ │ - * but does so in a way that, if there are other nodes with the same │ │ │ │ │ - * z-index, the nodes with the lower y position will be "closer" than │ │ │ │ │ - * those with a higher y position. If two nodes have the exact same y │ │ │ │ │ - * position, however, then this method will revert to using drawing │ │ │ │ │ - * order to decide placement. │ │ │ │ │ - * │ │ │ │ │ + * Method: createCluster │ │ │ │ │ + * Given a feature, create a cluster. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ - * newNode - {DOMElement} │ │ │ │ │ - * nextNode - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Integer} │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A cluster. │ │ │ │ │ */ │ │ │ │ │ - Z_ORDER_Y_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER( │ │ │ │ │ - indexer, │ │ │ │ │ - newNode, │ │ │ │ │ - nextNode │ │ │ │ │ + createCluster: function(feature) { │ │ │ │ │ + var center = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + var cluster = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.Point(center.lon, center.lat), { │ │ │ │ │ + count: 1 │ │ │ │ │ + } │ │ │ │ │ ); │ │ │ │ │ + cluster.cluster = [feature]; │ │ │ │ │ + return cluster; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (nextNode && returnVal === 0) { │ │ │ │ │ - var result = nextNode._boundsBottom - newNode._boundsBottom; │ │ │ │ │ - returnVal = (result === 0) ? 1 : result; │ │ │ │ │ - } │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Cluster" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Strategy/Save.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - return returnVal; │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Renderer.Elements │ │ │ │ │ - * This is another virtual class in that it should never be instantiated by │ │ │ │ │ - * itself as a Renderer. It exists because there is *tons* of shared │ │ │ │ │ - * functionality between different vector libraries which use nodes/elements │ │ │ │ │ - * as a base for rendering vectors. │ │ │ │ │ - * │ │ │ │ │ - * The highlevel bits of code that are implemented here are the adding and │ │ │ │ │ - * removing of geometries, which is essentially the same for any │ │ │ │ │ - * element-based renderer. The details of creating each node and drawing the │ │ │ │ │ - * paths are of course different, but the machinery is the same. │ │ │ │ │ - * │ │ │ │ │ - * Inherits: │ │ │ │ │ - * - <OpenLayers.Renderer> │ │ │ │ │ + * @requires OpenLayers/Strategy.js │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Renderer.Elements = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: rendererRoot │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - rendererRoot: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: root │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - root: null, │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Strategy.Save │ │ │ │ │ + * A strategy that commits newly created or modified features. By default │ │ │ │ │ + * the strategy waits for a call to <save> before persisting changes. By │ │ │ │ │ + * configuring the strategy with the <auto> option, changes can be saved │ │ │ │ │ + * automatically. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Strategy> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Strategy.Save = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: vectorRoot │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} An events object that handles all │ │ │ │ │ + * events on the strategy object. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * strategy.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types: │ │ │ │ │ + * start - Triggered before saving │ │ │ │ │ + * success - Triggered after a successful transaction │ │ │ │ │ + * fail - Triggered after a failed transaction │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - vectorRoot: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: textRoot │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + /** │ │ │ │ │ + * Property: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for triggering this protocol │ │ │ │ │ + * events. │ │ │ │ │ */ │ │ │ │ │ - textRoot: null, │ │ │ │ │ + events: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: xmlns │ │ │ │ │ - * {String} │ │ │ │ │ + * APIProperty: auto │ │ │ │ │ + * {Boolean | Number} Auto-save. Default is false. If true, features will be │ │ │ │ │ + * saved immediately after being added to the layer and with each │ │ │ │ │ + * modification or deletion. If auto is a number, features will be │ │ │ │ │ + * saved on an interval provided by the value (in seconds). │ │ │ │ │ */ │ │ │ │ │ - xmlns: null, │ │ │ │ │ + auto: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: xOffset │ │ │ │ │ - * {Number} Offset to apply to the renderer viewport translation in x │ │ │ │ │ - * direction. If the renderer extent's center is on the right of the │ │ │ │ │ - * dateline (i.e. exceeds the world bounds), we shift the viewport to the │ │ │ │ │ - * left by one world width. This avoids that features disappear from the │ │ │ │ │ - * map viewport. Because our dateline handling logic in other places │ │ │ │ │ - * ensures that extents crossing the dateline always have a center │ │ │ │ │ - * exceeding the world bounds on the left, we need this offset to make sure │ │ │ │ │ - * that the same is true for the renderer extent in pixel space as well. │ │ │ │ │ + * Property: timer │ │ │ │ │ + * {Number} The id of the timer. │ │ │ │ │ */ │ │ │ │ │ - xOffset: 0, │ │ │ │ │ + timer: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: rightOfDateLine │ │ │ │ │ - * {Boolean} Keeps track of the location of the map extent relative to the │ │ │ │ │ - * date line. The <setExtent> method compares this value (which is the one │ │ │ │ │ - * from the previous <setExtent> call) with the current position of the map │ │ │ │ │ - * extent relative to the date line and updates the xOffset when the extent │ │ │ │ │ - * has moved from one side of the date line to the other. │ │ │ │ │ + * Constructor: OpenLayers.Strategy.Save │ │ │ │ │ + * Create a new Save strategy. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Strategy.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.events = new OpenLayers.Events(this); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: Indexer │ │ │ │ │ - * {<OpenLayers.ElementIndexer>} An instance of OpenLayers.ElementsIndexer │ │ │ │ │ - * created upon initialization if the zIndexing or yOrdering options │ │ │ │ │ - * passed to this renderer's constructor are set to true. │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The strategy was successfully activated. │ │ │ │ │ */ │ │ │ │ │ - indexer: null, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + if (this.auto) { │ │ │ │ │ + if (typeof this.auto === "number") { │ │ │ │ │ + this.timer = window.setInterval( │ │ │ │ │ + OpenLayers.Function.bind(this.save, this), │ │ │ │ │ + this.auto * 1000 │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "featureadded": this.triggerSave, │ │ │ │ │ + "afterfeaturemodified": this.triggerSave, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: BACKGROUND_ID_SUFFIX │ │ │ │ │ - * {String} │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ + * tear-down. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ - BACKGROUND_ID_SUFFIX: "_background", │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + if (this.auto) { │ │ │ │ │ + if (typeof this.auto === "number") { │ │ │ │ │ + window.clearInterval(this.timer); │ │ │ │ │ + } else { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + "featureadded": this.triggerSave, │ │ │ │ │ + "afterfeaturemodified": this.triggerSave, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: LABEL_ID_SUFFIX │ │ │ │ │ - * {String} │ │ │ │ │ + * Method: triggerSave │ │ │ │ │ + * Registered as a listener. Calls save if a feature has insert, update, │ │ │ │ │ + * or delete state. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * event - {Object} The event this function is listening for. │ │ │ │ │ */ │ │ │ │ │ - LABEL_ID_SUFFIX: "_label", │ │ │ │ │ + triggerSave: function(event) { │ │ │ │ │ + var feature = event.feature; │ │ │ │ │ + if (feature.state === OpenLayers.State.INSERT || │ │ │ │ │ + feature.state === OpenLayers.State.UPDATE || │ │ │ │ │ + feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ + this.save([event.feature]); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: LABEL_OUTLINE_SUFFIX │ │ │ │ │ - * {String} │ │ │ │ │ + * APIMethod: save │ │ │ │ │ + * Tell the layer protocol to commit unsaved features. If the layer │ │ │ │ │ + * projection differs from the map projection, features will be │ │ │ │ │ + * transformed into the layer projection before being committed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array} Features to be saved. If null, then default is all │ │ │ │ │ + * features in the layer. Features are assumed to be in the map │ │ │ │ │ + * projection. │ │ │ │ │ */ │ │ │ │ │ - LABEL_OUTLINE_SUFFIX: "_outline", │ │ │ │ │ + save: function(features) { │ │ │ │ │ + if (!features) { │ │ │ │ │ + features = this.layer.features; │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("start", { │ │ │ │ │ + features: features │ │ │ │ │ + }); │ │ │ │ │ + var remote = this.layer.projection; │ │ │ │ │ + var local = this.layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var len = features.length; │ │ │ │ │ + var clones = new Array(len); │ │ │ │ │ + var orig, clone; │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + orig = features[i]; │ │ │ │ │ + clone = orig.clone(); │ │ │ │ │ + clone.fid = orig.fid; │ │ │ │ │ + clone.state = orig.state; │ │ │ │ │ + if (orig.url) { │ │ │ │ │ + clone.url = orig.url; │ │ │ │ │ + } │ │ │ │ │ + clone._original = orig; │ │ │ │ │ + clone.geometry.transform(local, remote); │ │ │ │ │ + clones[i] = clone; │ │ │ │ │ + } │ │ │ │ │ + features = clones; │ │ │ │ │ + } │ │ │ │ │ + this.layer.protocol.commit(features, { │ │ │ │ │ + callback: this.onCommit, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Renderer.Elements │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * containerID - {String} │ │ │ │ │ - * options - {Object} options for this renderer. │ │ │ │ │ + * Method: onCommit │ │ │ │ │ + * Called after protocol commit. │ │ │ │ │ * │ │ │ │ │ - * Supported options are: │ │ │ │ │ - * yOrdering - {Boolean} Whether to use y-ordering │ │ │ │ │ - * zIndexing - {Boolean} Whether to use z-indexing. Will be ignored │ │ │ │ │ - * if yOrdering is set to true. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} A response object. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(containerID, options) { │ │ │ │ │ - OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - this.rendererRoot = this.createRenderRoot(); │ │ │ │ │ - this.root = this.createRoot("_root"); │ │ │ │ │ - this.vectorRoot = this.createRoot("_vroot"); │ │ │ │ │ - this.textRoot = this.createRoot("_troot"); │ │ │ │ │ + onCommit: function(response) { │ │ │ │ │ + var evt = { │ │ │ │ │ + "response": response │ │ │ │ │ + }; │ │ │ │ │ + if (response.success()) { │ │ │ │ │ + var features = response.reqFeatures; │ │ │ │ │ + // deal with inserts, updates, and deletes │ │ │ │ │ + var state, feature; │ │ │ │ │ + var destroys = []; │ │ │ │ │ + var insertIds = response.insertIds || []; │ │ │ │ │ + var j = 0; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + // if projection was different, we may be dealing with clones │ │ │ │ │ + feature = feature._original || feature; │ │ │ │ │ + state = feature.state; │ │ │ │ │ + if (state) { │ │ │ │ │ + if (state == OpenLayers.State.DELETE) { │ │ │ │ │ + destroys.push(feature); │ │ │ │ │ + } else if (state == OpenLayers.State.INSERT) { │ │ │ │ │ + feature.fid = insertIds[j]; │ │ │ │ │ + ++j; │ │ │ │ │ + } │ │ │ │ │ + feature.state = null; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - this.root.appendChild(this.vectorRoot); │ │ │ │ │ - this.root.appendChild(this.textRoot); │ │ │ │ │ + if (destroys.length > 0) { │ │ │ │ │ + this.layer.destroyFeatures(destroys); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - this.rendererRoot.appendChild(this.root); │ │ │ │ │ - this.container.appendChild(this.rendererRoot); │ │ │ │ │ + this.events.triggerEvent("success", evt); │ │ │ │ │ │ │ │ │ │ - if (options && (options.zIndexing || options.yOrdering)) { │ │ │ │ │ - this.indexer = new OpenLayers.ElementsIndexer(options.yOrdering); │ │ │ │ │ + } else { │ │ │ │ │ + this.events.triggerEvent("fail", evt); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Save" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Strategy/Paging.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Strategy.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Strategy.Paging │ │ │ │ │ + * Strategy for vector feature paging │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Strategy> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Strategy.Paging = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} Cached features. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - │ │ │ │ │ - this.clear(); │ │ │ │ │ + features: null, │ │ │ │ │ │ │ │ │ │ - this.rendererRoot = null; │ │ │ │ │ - this.root = null; │ │ │ │ │ - this.xmlns = null; │ │ │ │ │ + /** │ │ │ │ │ + * Property: length │ │ │ │ │ + * {Integer} Number of features per page. Default is 10. │ │ │ │ │ + */ │ │ │ │ │ + length: 10, │ │ │ │ │ │ │ │ │ │ - OpenLayers.Renderer.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: num │ │ │ │ │ + * {Integer} The currently displayed page number. │ │ │ │ │ + */ │ │ │ │ │ + num: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clear │ │ │ │ │ - * Remove all the elements from the root │ │ │ │ │ + * Property: paging │ │ │ │ │ + * {Boolean} The strategy is currently changing pages. │ │ │ │ │ */ │ │ │ │ │ - clear: function() { │ │ │ │ │ - var child; │ │ │ │ │ - var root = this.vectorRoot; │ │ │ │ │ - if (root) { │ │ │ │ │ - while (child = root.firstChild) { │ │ │ │ │ - root.removeChild(child); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - root = this.textRoot; │ │ │ │ │ - if (root) { │ │ │ │ │ - while (child = root.firstChild) { │ │ │ │ │ - root.removeChild(child); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - this.indexer.clear(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + paging: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setExtent │ │ │ │ │ - * Set the visible part of the layer. │ │ │ │ │ + * Constructor: OpenLayers.Strategy.Paging │ │ │ │ │ + * Create a new paging strategy. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} │ │ │ │ │ - * resolutionChanged - {Boolean} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ - * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ - * False otherwise. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ */ │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - var rightOfDateLine, │ │ │ │ │ - ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ - extent = extent.scale(1 / ratio), │ │ │ │ │ - world = this.map.getMaxExtent(); │ │ │ │ │ - if (world.right > extent.left && world.right < extent.right) { │ │ │ │ │ - rightOfDateLine = true; │ │ │ │ │ - } else if (world.left > extent.left && world.left < extent.right) { │ │ │ │ │ - rightOfDateLine = false; │ │ │ │ │ - } │ │ │ │ │ - if (rightOfDateLine !== this.rightOfDateLine || resolutionChanged) { │ │ │ │ │ - coordSysUnchanged = false; │ │ │ │ │ - this.xOffset = rightOfDateLine === true ? │ │ │ │ │ - world.getWidth() / resolution : 0; │ │ │ │ │ - } │ │ │ │ │ - this.rightOfDateLine = rightOfDateLine; │ │ │ │ │ - } │ │ │ │ │ - return coordSysUnchanged; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getNodeType │ │ │ │ │ - * This function is in charge of asking the specific renderer which type │ │ │ │ │ - * of node to create for the given geometry and style. All geometries │ │ │ │ │ - * in an Elements-based renderer consist of one node and some │ │ │ │ │ - * attributes. We have the nodeFactory() function which creates a node │ │ │ │ │ - * for us, but it takes a 'type' as input, and that is precisely what │ │ │ │ │ - * this function tells us. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} The corresponding node type for the specified geometry │ │ │ │ │ + * {Boolean} The strategy was successfully activated. │ │ │ │ │ */ │ │ │ │ │ - getNodeType: function(geometry, style) {}, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "beforefeaturesadded": this.cacheFeatures, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawGeometry │ │ │ │ │ - * Draw the geometry, creating new nodes, setting paths, setting style, │ │ │ │ │ - * setting featureId on the node. This method should only be called │ │ │ │ │ - * by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ + * tear-down. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} true if the geometry has been drawn completely; null if │ │ │ │ │ - * incomplete; false otherwise │ │ │ │ │ + * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ - drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ - var className = geometry.CLASS_NAME; │ │ │ │ │ - var rendered = true; │ │ │ │ │ - if ((className == "OpenLayers.Geometry.Collection") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiPolygon")) { │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - rendered = this.drawGeometry( │ │ │ │ │ - geometry.components[i], style, featureId) && rendered; │ │ │ │ │ - } │ │ │ │ │ - return rendered; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - rendered = false; │ │ │ │ │ - var removeBackground = false; │ │ │ │ │ - if (style.display != "none") { │ │ │ │ │ - if (style.backgroundGraphic) { │ │ │ │ │ - this.redrawBackgroundNode(geometry.id, geometry, style, │ │ │ │ │ - featureId); │ │ │ │ │ - } else { │ │ │ │ │ - removeBackground = true; │ │ │ │ │ - } │ │ │ │ │ - rendered = this.redrawNode(geometry.id, geometry, style, │ │ │ │ │ - featureId); │ │ │ │ │ - } │ │ │ │ │ - if (rendered == false) { │ │ │ │ │ - var node = document.getElementById(geometry.id); │ │ │ │ │ - if (node) { │ │ │ │ │ - if (node._style.backgroundGraphic) { │ │ │ │ │ - removeBackground = true; │ │ │ │ │ - } │ │ │ │ │ - node.parentNode.removeChild(node); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (removeBackground) { │ │ │ │ │ - var node = document.getElementById( │ │ │ │ │ - geometry.id + this.BACKGROUND_ID_SUFFIX); │ │ │ │ │ - if (node) { │ │ │ │ │ - node.parentNode.removeChild(node); │ │ │ │ │ - } │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.clearCache(); │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + "beforefeaturesadded": this.cacheFeatures, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - return rendered; │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: redrawNode │ │ │ │ │ - * │ │ │ │ │ + * Method: cacheFeatures │ │ │ │ │ + * Cache features before they are added to the layer. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true if the complete geometry could be drawn, null if parts of │ │ │ │ │ - * the geometry could not be drawn, false otherwise │ │ │ │ │ + * event - {Object} The event that this was listening for. This will come │ │ │ │ │ + * with a batch of features to be paged. │ │ │ │ │ */ │ │ │ │ │ - redrawNode: function(id, geometry, style, featureId) { │ │ │ │ │ - style = this.applyDefaultSymbolizer(style); │ │ │ │ │ - // Get the node if it's already on the map. │ │ │ │ │ - var node = this.nodeFactory(id, this.getNodeType(geometry, style)); │ │ │ │ │ - │ │ │ │ │ - // Set the data for the node, then draw it. │ │ │ │ │ - node._featureId = featureId; │ │ │ │ │ - node._boundsBottom = geometry.getBounds().bottom; │ │ │ │ │ - node._geometryClass = geometry.CLASS_NAME; │ │ │ │ │ - node._style = style; │ │ │ │ │ - │ │ │ │ │ - var drawResult = this.drawGeometryNode(node, geometry, style); │ │ │ │ │ - if (drawResult === false) { │ │ │ │ │ - return false; │ │ │ │ │ + cacheFeatures: function(event) { │ │ │ │ │ + if (!this.paging) { │ │ │ │ │ + this.clearCache(); │ │ │ │ │ + this.features = event.features; │ │ │ │ │ + this.pageNext(event); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - node = drawResult.node; │ │ │ │ │ - │ │ │ │ │ - // Insert the node into the indexer so it can show us where to │ │ │ │ │ - // place it. Note that this operation is O(log(n)). If there's a │ │ │ │ │ - // performance problem (when dragging, for instance) this is │ │ │ │ │ - // likely where it would be. │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - var insert = this.indexer.insert(node); │ │ │ │ │ - if (insert) { │ │ │ │ │ - this.vectorRoot.insertBefore(node, insert); │ │ │ │ │ - } else { │ │ │ │ │ - this.vectorRoot.appendChild(node); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - // if there's no indexer, simply append the node to root, │ │ │ │ │ - // but only if the node is a new one │ │ │ │ │ - if (node.parentNode !== this.vectorRoot) { │ │ │ │ │ - this.vectorRoot.appendChild(node); │ │ │ │ │ + /** │ │ │ │ │ + * Method: clearCache │ │ │ │ │ + * Clear out the cached features. This destroys features, assuming │ │ │ │ │ + * nothing else has a reference. │ │ │ │ │ + */ │ │ │ │ │ + clearCache: function() { │ │ │ │ │ + if (this.features) { │ │ │ │ │ + for (var i = 0; i < this.features.length; ++i) { │ │ │ │ │ + this.features[i].destroy(); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - this.postDraw(node); │ │ │ │ │ - │ │ │ │ │ - return drawResult.complete; │ │ │ │ │ + this.features = null; │ │ │ │ │ + this.num = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: redrawBackgroundNode │ │ │ │ │ - * Redraws the node using special 'background' style properties. Basically │ │ │ │ │ - * just calls redrawNode(), but instead of directly using the │ │ │ │ │ - * 'externalGraphic', 'graphicXOffset', 'graphicYOffset', and │ │ │ │ │ - * 'graphicZIndex' properties directly from the specified 'style' │ │ │ │ │ - * parameter, we create a new style object and set those properties │ │ │ │ │ - * from the corresponding 'background'-prefixed properties from │ │ │ │ │ - * specified 'style' parameter. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: pageCount │ │ │ │ │ + * Get the total count of pages given the current cache of features. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} true if the complete geometry could be drawn, null if parts of │ │ │ │ │ - * the geometry could not be drawn, false otherwise │ │ │ │ │ + * {Integer} The page count. │ │ │ │ │ */ │ │ │ │ │ - redrawBackgroundNode: function(id, geometry, style, featureId) { │ │ │ │ │ - var backgroundStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ - │ │ │ │ │ - // Set regular style attributes to apply to the background styles. │ │ │ │ │ - backgroundStyle.externalGraphic = backgroundStyle.backgroundGraphic; │ │ │ │ │ - backgroundStyle.graphicXOffset = backgroundStyle.backgroundXOffset; │ │ │ │ │ - backgroundStyle.graphicYOffset = backgroundStyle.backgroundYOffset; │ │ │ │ │ - backgroundStyle.graphicZIndex = backgroundStyle.backgroundGraphicZIndex; │ │ │ │ │ - backgroundStyle.graphicWidth = backgroundStyle.backgroundWidth || backgroundStyle.graphicWidth; │ │ │ │ │ - backgroundStyle.graphicHeight = backgroundStyle.backgroundHeight || backgroundStyle.graphicHeight; │ │ │ │ │ - │ │ │ │ │ - // Erase background styles. │ │ │ │ │ - backgroundStyle.backgroundGraphic = null; │ │ │ │ │ - backgroundStyle.backgroundXOffset = null; │ │ │ │ │ - backgroundStyle.backgroundYOffset = null; │ │ │ │ │ - backgroundStyle.backgroundGraphicZIndex = null; │ │ │ │ │ - │ │ │ │ │ - return this.redrawNode( │ │ │ │ │ - id + this.BACKGROUND_ID_SUFFIX, │ │ │ │ │ - geometry, │ │ │ │ │ - backgroundStyle, │ │ │ │ │ - null │ │ │ │ │ - ); │ │ │ │ │ + pageCount: function() { │ │ │ │ │ + var numFeatures = this.features ? this.features.length : 0; │ │ │ │ │ + return Math.ceil(numFeatures / this.length); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawGeometryNode │ │ │ │ │ - * Given a node, draw a geometry on the specified layer. │ │ │ │ │ - * node and geometry are required arguments, style is optional. │ │ │ │ │ - * This method is only called by the render itself. │ │ │ │ │ + * APIMethod: pageNum │ │ │ │ │ + * Get the zero based page number. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} a hash with properties "node" (the drawn node) and "complete" │ │ │ │ │ - * (null if parts of the geometry could not be drawn, false if nothing │ │ │ │ │ - * could be drawn) │ │ │ │ │ + * {Integer} The current page number being displayed. │ │ │ │ │ */ │ │ │ │ │ - drawGeometryNode: function(node, geometry, style) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - │ │ │ │ │ - var options = { │ │ │ │ │ - 'isFilled': style.fill === undefined ? │ │ │ │ │ - true : style.fill, │ │ │ │ │ - 'isStroked': style.stroke === undefined ? │ │ │ │ │ - !!style.strokeWidth : style.stroke │ │ │ │ │ - }; │ │ │ │ │ - var drawn; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.graphic === false) { │ │ │ │ │ - options.isFilled = false; │ │ │ │ │ - options.isStroked = false; │ │ │ │ │ - } │ │ │ │ │ - drawn = this.drawPoint(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - options.isFilled = false; │ │ │ │ │ - drawn = this.drawLineString(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - drawn = this.drawLinearRing(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - drawn = this.drawPolygon(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - drawn = this.drawRectangle(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - node._options = options; │ │ │ │ │ - │ │ │ │ │ - //set style │ │ │ │ │ - //TBD simplify this │ │ │ │ │ - if (drawn != false) { │ │ │ │ │ - return { │ │ │ │ │ - node: this.setStyle(node, style, options, geometry), │ │ │ │ │ - complete: drawn │ │ │ │ │ - }; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ + pageNum: function() { │ │ │ │ │ + return this.num; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: postDraw │ │ │ │ │ - * Things that have do be done after the geometry node is appended │ │ │ │ │ - * to its parent node. To be overridden by subclasses. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: pageLength │ │ │ │ │ + * Gets or sets page length. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ + * newLength - {Integer} Optional length to be set. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} The length of a page (number of features per page). │ │ │ │ │ */ │ │ │ │ │ - postDraw: function(node) {}, │ │ │ │ │ + pageLength: function(newLength) { │ │ │ │ │ + if (newLength && newLength > 0) { │ │ │ │ │ + this.length = newLength; │ │ │ │ │ + } │ │ │ │ │ + return this.length; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPoint │ │ │ │ │ - * Virtual function for drawing Point Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: pageNext │ │ │ │ │ + * Display the next page of features. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the point │ │ │ │ │ + * {Boolean} A new page was displayed. │ │ │ │ │ */ │ │ │ │ │ - drawPoint: function(node, geometry) {}, │ │ │ │ │ + pageNext: function(event) { │ │ │ │ │ + var changed = false; │ │ │ │ │ + if (this.features) { │ │ │ │ │ + if (this.num === null) { │ │ │ │ │ + this.num = -1; │ │ │ │ │ + } │ │ │ │ │ + var start = (this.num + 1) * this.length; │ │ │ │ │ + changed = this.page(start, event); │ │ │ │ │ + } │ │ │ │ │ + return changed; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLineString │ │ │ │ │ - * Virtual function for drawing LineString Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: pagePrevious │ │ │ │ │ + * Display the previous page of features. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components of │ │ │ │ │ - * the linestring, or false if nothing could be drawn │ │ │ │ │ + * {Boolean} A new page was displayed. │ │ │ │ │ */ │ │ │ │ │ - drawLineString: function(node, geometry) {}, │ │ │ │ │ + pagePrevious: function() { │ │ │ │ │ + var changed = false; │ │ │ │ │ + if (this.features) { │ │ │ │ │ + if (this.num === null) { │ │ │ │ │ + this.num = this.pageCount(); │ │ │ │ │ + } │ │ │ │ │ + var start = (this.num - 1) * this.length; │ │ │ │ │ + changed = this.page(start); │ │ │ │ │ + } │ │ │ │ │ + return changed; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLinearRing │ │ │ │ │ - * Virtual function for drawing LinearRing Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * Method: page │ │ │ │ │ + * Display the page starting at the given index from the cache. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ - * of the linear ring, or false if nothing could be drawn │ │ │ │ │ + * {Boolean} A new page was displayed. │ │ │ │ │ */ │ │ │ │ │ - drawLinearRing: function(node, geometry) {}, │ │ │ │ │ + page: function(start, event) { │ │ │ │ │ + var changed = false; │ │ │ │ │ + if (this.features) { │ │ │ │ │ + if (start >= 0 && start < this.features.length) { │ │ │ │ │ + var num = Math.floor(start / this.length); │ │ │ │ │ + if (num != this.num) { │ │ │ │ │ + this.paging = true; │ │ │ │ │ + var features = this.features.slice(start, start + this.length); │ │ │ │ │ + this.layer.removeFeatures(this.layer.features); │ │ │ │ │ + this.num = num; │ │ │ │ │ + // modify the event if any │ │ │ │ │ + if (event && event.features) { │ │ │ │ │ + // this.was called by an event listener │ │ │ │ │ + event.features = features; │ │ │ │ │ + } else { │ │ │ │ │ + // this was called directly on the strategy │ │ │ │ │ + this.layer.addFeatures(features); │ │ │ │ │ + } │ │ │ │ │ + this.paging = false; │ │ │ │ │ + changed = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return changed; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Paging" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Strategy/Fixed.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Strategy.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Strategy.Fixed │ │ │ │ │ + * A simple strategy that requests features once and never requests new data. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Strategy> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Strategy.Fixed = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPolygon │ │ │ │ │ - * Virtual function for drawing Polygon Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ - * of the polygon, or false if nothing could be drawn │ │ │ │ │ + * APIProperty: preload │ │ │ │ │ + * {Boolean} Load data before layer made visible. Enabling this may result │ │ │ │ │ + * in considerable overhead if your application loads many data layers │ │ │ │ │ + * that are not visible by default. Default is false. │ │ │ │ │ */ │ │ │ │ │ - drawPolygon: function(node, geometry) {}, │ │ │ │ │ + preload: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawRectangle │ │ │ │ │ - * Virtual function for drawing Rectangle Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the rectangle │ │ │ │ │ + * Constructor: OpenLayers.Strategy.Fixed │ │ │ │ │ + * Create a new Fixed strategy. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ */ │ │ │ │ │ - drawRectangle: function(node, geometry) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawCircle │ │ │ │ │ - * Virtual function for drawing Circle Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activate the strategy: load data or add listener to load when visible │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the circle │ │ │ │ │ + * {Boolean} True if the strategy was successfully activated or false if │ │ │ │ │ + * the strategy was already active. │ │ │ │ │ */ │ │ │ │ │ - drawCircle: function(node, geometry) {}, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "refresh": this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + if (this.layer.visibility == true || this.preload) { │ │ │ │ │ + this.load(); │ │ │ │ │ + } else { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "visibilitychanged": this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removeText │ │ │ │ │ - * Removes a label │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivate the strategy. Undo what is done in <activate>. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ - removeText: function(featureId) { │ │ │ │ │ - var label = document.getElementById(featureId + this.LABEL_ID_SUFFIX); │ │ │ │ │ - if (label) { │ │ │ │ │ - this.textRoot.removeChild(label); │ │ │ │ │ - } │ │ │ │ │ - var outline = document.getElementById(featureId + this.LABEL_OUTLINE_SUFFIX); │ │ │ │ │ - if (outline) { │ │ │ │ │ - this.textRoot.removeChild(outline); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + "refresh": this.load, │ │ │ │ │ + "visibilitychanged": this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureIdFromEvent │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} An <OpenLayers.Event> object │ │ │ │ │ + * Method: load │ │ │ │ │ + * Tells protocol to load data and unhooks the visibilitychanged event │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A feature id or undefined. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} options to pass to protocol read. │ │ │ │ │ */ │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var target = evt.target; │ │ │ │ │ - var useElement = target && target.correspondingUseElement; │ │ │ │ │ - var node = useElement ? useElement : (target || evt.srcElement); │ │ │ │ │ - return node._featureId; │ │ │ │ │ + load: function(options) { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + layer.events.triggerEvent("loadstart", { │ │ │ │ │ + filter: layer.filter │ │ │ │ │ + }); │ │ │ │ │ + layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: this.merge, │ │ │ │ │ + filter: layer.filter, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options)); │ │ │ │ │ + layer.events.un({ │ │ │ │ │ + "visibilitychanged": this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: eraseGeometry │ │ │ │ │ - * Erase a geometry from the renderer. In the case of a multi-geometry, │ │ │ │ │ - * we cycle through and recurse on ourselves. Otherwise, we look for a │ │ │ │ │ - * node with the geometry.id, destroy its geometry, and remove it from │ │ │ │ │ - * the DOM. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: merge │ │ │ │ │ + * Add all features to the layer. │ │ │ │ │ + * If the layer projection differs from the map projection, features │ │ │ │ │ + * will be transformed from the layer projection to the map projection. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object passed │ │ │ │ │ + * by the protocol. │ │ │ │ │ */ │ │ │ │ │ - eraseGeometry: function(geometry, featureId) { │ │ │ │ │ - if ((geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ - (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ - (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPolygon") || │ │ │ │ │ - (geometry.CLASS_NAME == "OpenLayers.Geometry.Collection")) { │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - this.eraseGeometry(geometry.components[i], featureId); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - var element = OpenLayers.Util.getElement(geometry.id); │ │ │ │ │ - if (element && element.parentNode) { │ │ │ │ │ - if (element.geometry) { │ │ │ │ │ - element.geometry.destroy(); │ │ │ │ │ - element.geometry = null; │ │ │ │ │ - } │ │ │ │ │ - element.parentNode.removeChild(element); │ │ │ │ │ - │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - this.indexer.remove(element); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (element._style.backgroundGraphic) { │ │ │ │ │ - var backgroundId = geometry.id + this.BACKGROUND_ID_SUFFIX; │ │ │ │ │ - var bElem = OpenLayers.Util.getElement(backgroundId); │ │ │ │ │ - if (bElem && bElem.parentNode) { │ │ │ │ │ - // No need to destroy the geometry since the element and the background │ │ │ │ │ - // node share the same geometry. │ │ │ │ │ - bElem.parentNode.removeChild(bElem); │ │ │ │ │ + merge: function(resp) { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + layer.destroyFeatures(); │ │ │ │ │ + var features = resp.features; │ │ │ │ │ + if (features && features.length > 0) { │ │ │ │ │ + var remote = layer.projection; │ │ │ │ │ + var local = layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var geom; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + geom = features[i].geometry; │ │ │ │ │ + if (geom) { │ │ │ │ │ + geom.transform(remote, local); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + layer.addFeatures(features); │ │ │ │ │ } │ │ │ │ │ + layer.events.triggerEvent("loadend", { │ │ │ │ │ + response: resp │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Fixed" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Marker/Box.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Marker.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Marker.Box │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Marker> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Marker.Box = OpenLayers.Class(OpenLayers.Marker, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: nodeFactory │ │ │ │ │ - * Create new node of the specified type, with the (optional) specified id. │ │ │ │ │ - * │ │ │ │ │ - * If node already exists with same ID and a different type, we remove it │ │ │ │ │ - * and then call ourselves again to recreate it. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * type - {String} type Kind of node to draw. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A new node of the given type and id. │ │ │ │ │ + * Property: bounds │ │ │ │ │ + * {<OpenLayers.Bounds>} │ │ │ │ │ */ │ │ │ │ │ - nodeFactory: function(id, type) { │ │ │ │ │ - var node = OpenLayers.Util.getElement(id); │ │ │ │ │ - if (node) { │ │ │ │ │ - if (!this.nodeTypeCompare(node, type)) { │ │ │ │ │ - node.parentNode.removeChild(node); │ │ │ │ │ - node = this.nodeFactory(id, type); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - node = this.createNode(type, id); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ + bounds: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: nodeTypeCompare │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * type - {String} Kind of node │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ - * This function must be overridden by subclasses. │ │ │ │ │ + * Property: div │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - nodeTypeCompare: function(node, type) {}, │ │ │ │ │ + div: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createNode │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Marker.Box │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * type - {String} Kind of node to draw. │ │ │ │ │ - * id - {String} Id for node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A new node of the given type and id. │ │ │ │ │ - * This function must be overridden by subclasses. │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * borderColor - {String} │ │ │ │ │ + * borderWidth - {int} │ │ │ │ │ */ │ │ │ │ │ - createNode: function(type, id) {}, │ │ │ │ │ + initialize: function(bounds, borderColor, borderWidth) { │ │ │ │ │ + this.bounds = bounds; │ │ │ │ │ + this.div = OpenLayers.Util.createDiv(); │ │ │ │ │ + this.div.style.overflow = 'hidden'; │ │ │ │ │ + this.events = new OpenLayers.Events(this, this.div); │ │ │ │ │ + this.setBorder(borderColor, borderWidth); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveRoot │ │ │ │ │ - * moves this renderer's root to a different renderer. │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + │ │ │ │ │ + this.bounds = null; │ │ │ │ │ + this.div = null; │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Marker.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setBorder │ │ │ │ │ + * Allow the user to change the box's color and border width │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * renderer - {<OpenLayers.Renderer>} target renderer for the moved root │ │ │ │ │ + * color - {String} Default is "red" │ │ │ │ │ + * width - {int} Default is 2 │ │ │ │ │ */ │ │ │ │ │ - moveRoot: function(renderer) { │ │ │ │ │ - var root = this.root; │ │ │ │ │ - if (renderer.root.parentNode == this.rendererRoot) { │ │ │ │ │ - root = renderer.root; │ │ │ │ │ + setBorder: function(color, width) { │ │ │ │ │ + if (!color) { │ │ │ │ │ + color = "red"; │ │ │ │ │ } │ │ │ │ │ - root.parentNode.removeChild(root); │ │ │ │ │ - renderer.rendererRoot.appendChild(root); │ │ │ │ │ + if (!width) { │ │ │ │ │ + width = 2; │ │ │ │ │ + } │ │ │ │ │ + this.div.style.border = width + "px solid " + color; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ + * sz - {<OpenLayers.Size>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A new DOM Image with this marker's icon set at the │ │ │ │ │ + * location passed-in │ │ │ │ │ + */ │ │ │ │ │ + draw: function(px, sz) { │ │ │ │ │ + OpenLayers.Util.modifyDOMElement(this.div, null, px, sz); │ │ │ │ │ + return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getRenderLayerId │ │ │ │ │ - * Gets the layer that this renderer's output appears on. If moveRoot was │ │ │ │ │ - * used, this will be different from the id of the layer containing the │ │ │ │ │ - * features rendered by this renderer. │ │ │ │ │ + * Method: onScreen │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} the id of the output layer. │ │ │ │ │ + * Rreturn: │ │ │ │ │ + * {Boolean} Whether or not the marker is currently visible on screen. │ │ │ │ │ */ │ │ │ │ │ - getRenderLayerId: function() { │ │ │ │ │ - return this.root.parentNode.parentNode.id; │ │ │ │ │ + onScreen: function() { │ │ │ │ │ + var onScreen = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + var screenBounds = this.map.getExtent(); │ │ │ │ │ + onScreen = screenBounds.containsBounds(this.bounds, true, true); │ │ │ │ │ + } │ │ │ │ │ + return onScreen; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: isComplexSymbol │ │ │ │ │ - * Determines if a symbol cannot be rendered using drawCircle │ │ │ │ │ + * Method: display │ │ │ │ │ + * Hide or show the icon │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * graphicName - {String} │ │ │ │ │ - * │ │ │ │ │ - * Returns │ │ │ │ │ - * {Boolean} true if the symbol is complex, false if not │ │ │ │ │ + * display - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - isComplexSymbol: function(graphicName) { │ │ │ │ │ - return (graphicName != "circle") && !!graphicName; │ │ │ │ │ + display: function(display) { │ │ │ │ │ + this.div.style.display = (display) ? "" : "none"; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.Elements" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Marker.Box" │ │ │ │ │ }); │ │ │ │ │ │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Renderer/VML.js │ │ │ │ │ + OpenLayers/Layer/XYZ.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Renderer/Elements.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Renderer.VML │ │ │ │ │ - * Render vector features in browsers with VML capability. Construct a new │ │ │ │ │ - * VML renderer with the <OpenLayers.Renderer.VML> constructor. │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.XYZ │ │ │ │ │ + * The XYZ class is designed to make it easier for people who have tiles │ │ │ │ │ + * arranged by a standard XYZ grid. │ │ │ │ │ * │ │ │ │ │ - * Note that for all calculations in this class, we use (num | 0) to truncate a │ │ │ │ │ - * float value to an integer. This is done because it seems that VML doesn't │ │ │ │ │ - * support float values. │ │ │ │ │ - * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Renderer.Elements> │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Renderer.VML = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ +OpenLayers.Layer.XYZ = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: xmlns │ │ │ │ │ - * {String} XML Namespace URN │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * Default is true, as this is designed to be a base tile source. │ │ │ │ │ */ │ │ │ │ │ - xmlns: "urn:schemas-microsoft-com:vml", │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: symbolCache │ │ │ │ │ - * {DOMElement} node holding symbols. This hash is keyed by symbol name, │ │ │ │ │ - * and each value is a hash with a "path" and an "extent" property. │ │ │ │ │ + * APIProperty: sphericalMercator │ │ │ │ │ + * Whether the tile extents should be set to the defaults for │ │ │ │ │ + * spherical mercator. Useful for things like OpenStreetMap. │ │ │ │ │ + * Default is false, except for the OSM subclass. │ │ │ │ │ */ │ │ │ │ │ - symbolCache: {}, │ │ │ │ │ + sphericalMercator: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: offset │ │ │ │ │ - * {Object} Hash with "x" and "y" properties │ │ │ │ │ + * APIProperty: zoomOffset │ │ │ │ │ + * {Number} If your cache has more zoom levels than you want to provide │ │ │ │ │ + * access to with this layer, supply a zoomOffset. This zoom offset │ │ │ │ │ + * is added to the current map zoom level to determine the level │ │ │ │ │ + * for a requested tile. For example, if you supply a zoomOffset │ │ │ │ │ + * of 3, when the map is at the zoom 0, tiles will be requested from │ │ │ │ │ + * level 3 of your cache. Default is 0 (assumes cache level and map │ │ │ │ │ + * zoom are equivalent). Using <zoomOffset> is an alternative to │ │ │ │ │ + * setting <serverResolutions> if you only want to expose a subset │ │ │ │ │ + * of the server resolutions. │ │ │ │ │ */ │ │ │ │ │ - offset: null, │ │ │ │ │ + zoomOffset: 0, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Renderer.VML │ │ │ │ │ - * Create a new VML renderer. │ │ │ │ │ + * APIProperty: serverResolutions │ │ │ │ │ + * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ + * property if the map resolutions differ from the server. This │ │ │ │ │ + * property serves two purposes. (a) <serverResolutions> can include │ │ │ │ │ + * resolutions that the server supports and that you don't want to │ │ │ │ │ + * provide with this layer; you can also look at <zoomOffset>, which is │ │ │ │ │ + * an alternative to <serverResolutions> for that specific purpose. │ │ │ │ │ + * (b) The map can work with resolutions that aren't supported by │ │ │ │ │ + * the server, i.e. that aren't in <serverResolutions>. When the │ │ │ │ │ + * map is displayed in such a resolution data for the closest │ │ │ │ │ + * server-supported resolution is loaded and the layer div is │ │ │ │ │ + * stretched as necessary. │ │ │ │ │ + */ │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.XYZ │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * containerID - {String} The id for the element that contains the renderer │ │ │ │ │ + * name - {String} │ │ │ │ │ + * url - {String} │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ */ │ │ │ │ │ - initialize: function(containerID) { │ │ │ │ │ - if (!this.supported()) { │ │ │ │ │ - return; │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + if (options && options.sphericalMercator || this.sphericalMercator) { │ │ │ │ │ + options = OpenLayers.Util.extend({ │ │ │ │ │ + projection: "EPSG:900913", │ │ │ │ │ + numZoomLevels: 19 │ │ │ │ │ + }, options); │ │ │ │ │ } │ │ │ │ │ - if (!document.namespaces.olv) { │ │ │ │ │ - document.namespaces.add("olv", this.xmlns); │ │ │ │ │ - var style = document.createStyleSheet(); │ │ │ │ │ - var shapes = ['shape', 'rect', 'oval', 'fill', 'stroke', 'imagedata', 'group', 'textbox']; │ │ │ │ │ - for (var i = 0, len = shapes.length; i < len; i++) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [ │ │ │ │ │ + name || this.name, url || this.url, {}, │ │ │ │ │ + options │ │ │ │ │ + ]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - style.addRule('olv\\:' + shapes[i], "behavior: url(#default#VML); " + │ │ │ │ │ - "position: absolute; display: inline-block;"); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} Is this ever used? │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.XYZ>} An exact clone of this OpenLayers.Layer.XYZ │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.XYZ(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.initialize.apply(this, │ │ │ │ │ - arguments); │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: supported │ │ │ │ │ - * Determine whether a browser supports this renderer. │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The browser supports the VML renderer │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as │ │ │ │ │ + * parameters │ │ │ │ │ */ │ │ │ │ │ - supported: function() { │ │ │ │ │ - return !!(document.namespaces); │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var xyz = this.getXYZ(bounds); │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + var s = '' + xyz.x + xyz.y + xyz.z; │ │ │ │ │ + url = this.selectUrl(s, url); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.String.format(url, xyz); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setExtent │ │ │ │ │ - * Set the renderer's extent │ │ │ │ │ + * Method: getXYZ │ │ │ │ │ + * Calculates x, y and z for the given bounds. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} │ │ │ │ │ - * resolutionChanged - {Boolean} │ │ │ │ │ - * │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ - * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ + * {Object} - an object with x, y and z properties. │ │ │ │ │ */ │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ + getXYZ: function(bounds) { │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.maxExtent.left) / │ │ │ │ │ + (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((this.maxExtent.top - bounds.top) / │ │ │ │ │ + (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getServerZoom(); │ │ │ │ │ │ │ │ │ │ - var left = (extent.left / resolution) | 0; │ │ │ │ │ - var top = (extent.top / resolution - this.size.h) | 0; │ │ │ │ │ - if (resolutionChanged || !this.offset) { │ │ │ │ │ - this.offset = { │ │ │ │ │ - x: left, │ │ │ │ │ - y: top │ │ │ │ │ - }; │ │ │ │ │ - left = 0; │ │ │ │ │ - top = 0; │ │ │ │ │ - } else { │ │ │ │ │ - left = left - this.offset.x; │ │ │ │ │ - top = top - this.offset.y; │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + var limit = Math.pow(2, z); │ │ │ │ │ + x = ((x % limit) + limit) % limit; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ + return { │ │ │ │ │ + 'x': x, │ │ │ │ │ + 'y': y, │ │ │ │ │ + 'z': z │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var org = (left - this.xOffset) + " " + top; │ │ │ │ │ - this.root.coordorigin = org; │ │ │ │ │ - var roots = [this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ - var root; │ │ │ │ │ - for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ - root = roots[i]; │ │ │ │ │ + /* APIMethod: setMap │ │ │ │ │ + * When the layer is added to a map, then we can fetch our origin │ │ │ │ │ + * (if we don't have one.) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (!this.tileOrigin) { │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, │ │ │ │ │ + this.maxExtent.bottom); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var size = this.size.w + " " + this.size.h; │ │ │ │ │ - root.coordsize = size; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.XYZ" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/Markers.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - } │ │ │ │ │ - // flip the VML display Y axis upside down so it │ │ │ │ │ - // matches the display Y axis of the map │ │ │ │ │ - this.root.style.flip = "y"; │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.Markers │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Markers = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} Markers layer is never a base layer. │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: markers │ │ │ │ │ + * {Array(<OpenLayers.Marker>)} internal marker list │ │ │ │ │ + */ │ │ │ │ │ + markers: null, │ │ │ │ │ │ │ │ │ │ - return coordSysUnchanged; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: drawn │ │ │ │ │ + * {Boolean} internal state of drawing. This is a workaround for the fact │ │ │ │ │ + * that the map does not call moveTo with a zoomChanged when the map is │ │ │ │ │ + * first starting up. This lets us catch the case where we have *never* │ │ │ │ │ + * drawn the layer, and draw it even if the zoom hasn't changed. │ │ │ │ │ + */ │ │ │ │ │ + drawn: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setSize │ │ │ │ │ - * Set the size of the drawing surface │ │ │ │ │ + * Constructor: OpenLayers.Layer.Markers │ │ │ │ │ + * Create a Markers layer. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>} the size of the drawing surface │ │ │ │ │ + * name - {String} │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ */ │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.markers = []; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // setting width and height on all roots to avoid flicker which we │ │ │ │ │ - // would get with 100% width and height on child roots │ │ │ │ │ - var roots = [ │ │ │ │ │ - this.rendererRoot, │ │ │ │ │ - this.root, │ │ │ │ │ - this.vectorRoot, │ │ │ │ │ - this.textRoot │ │ │ │ │ - ]; │ │ │ │ │ - var w = this.size.w + "px"; │ │ │ │ │ - var h = this.size.h + "px"; │ │ │ │ │ - var root; │ │ │ │ │ - for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ - root = roots[i]; │ │ │ │ │ - root.style.width = w; │ │ │ │ │ - root.style.height = h; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.clearMarkers(); │ │ │ │ │ + this.markers = null; │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getNodeType │ │ │ │ │ - * Get the node type for a geometry and style │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: setOpacity │ │ │ │ │ + * Sets the opacity for all the markers. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The corresponding node type for the specified geometry │ │ │ │ │ + * opacity - {Float} │ │ │ │ │ */ │ │ │ │ │ - getNodeType: function(geometry, style) { │ │ │ │ │ - var nodeType = null; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - nodeType = "olv:rect"; │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - nodeType = "olv:shape"; │ │ │ │ │ - } else { │ │ │ │ │ - nodeType = "olv:oval"; │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - nodeType = "olv:rect"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - case "OpenLayers.Geometry.Curve": │ │ │ │ │ - nodeType = "olv:shape"; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break; │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + if (opacity != this.opacity) { │ │ │ │ │ + this.opacity = opacity; │ │ │ │ │ + for (var i = 0, len = this.markers.length; i < len; i++) { │ │ │ │ │ + this.markers[i].setOpacity(this.opacity); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return nodeType; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setStyle │ │ │ │ │ - * Use to set all the style attributes to a VML node. │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} An VML element to decorate │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * options - {Object} Currently supported options include │ │ │ │ │ - * 'isFilled' {Boolean} and │ │ │ │ │ - * 'isStroked' {Boolean} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * zoomChanged - {Boolean} │ │ │ │ │ + * dragging - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - setStyle: function(node, style, options, geometry) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - options = options || node._options; │ │ │ │ │ - var fillColor = style.fillColor; │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - node.title = title; │ │ │ │ │ + if (zoomChanged || !this.drawn) { │ │ │ │ │ + for (var i = 0, len = this.markers.length; i < len; i++) { │ │ │ │ │ + this.drawMarker(this.markers[i]); │ │ │ │ │ + } │ │ │ │ │ + this.drawn = true; │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - options.isFilled = true; │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ - style.graphicXOffset : -(0.5 * width); │ │ │ │ │ - var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ - style.graphicYOffset : -(0.5 * height); │ │ │ │ │ - │ │ │ │ │ - node.style.left = ((((geometry.x - this.featureDx) / resolution - this.offset.x) + xOffset) | 0) + "px"; │ │ │ │ │ - node.style.top = (((geometry.y / resolution - this.offset.y) - (yOffset + height)) | 0) + "px"; │ │ │ │ │ - node.style.width = width + "px"; │ │ │ │ │ - node.style.height = height + "px"; │ │ │ │ │ - node.style.flip = "y"; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: addMarker │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * marker - {<OpenLayers.Marker>} │ │ │ │ │ + */ │ │ │ │ │ + addMarker: function(marker) { │ │ │ │ │ + this.markers.push(marker); │ │ │ │ │ │ │ │ │ │ - // modify fillColor and options for stroke styling below │ │ │ │ │ - fillColor = "none"; │ │ │ │ │ - options.isStroked = false; │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - var cache = this.importSymbol(style.graphicName); │ │ │ │ │ - node.path = cache.path; │ │ │ │ │ - node.coordorigin = cache.left + "," + cache.bottom; │ │ │ │ │ - var size = cache.size; │ │ │ │ │ - node.coordsize = size + "," + size; │ │ │ │ │ - this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ - node.style.flip = "y"; │ │ │ │ │ - } else { │ │ │ │ │ - this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ - } │ │ │ │ │ + if (this.opacity < 1) { │ │ │ │ │ + marker.setOpacity(this.opacity); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // fill │ │ │ │ │ - if (options.isFilled) { │ │ │ │ │ - node.fillcolor = fillColor; │ │ │ │ │ - } else { │ │ │ │ │ - node.filled = "false"; │ │ │ │ │ + if (this.map && this.map.getExtent()) { │ │ │ │ │ + marker.map = this.map; │ │ │ │ │ + this.drawMarker(marker); │ │ │ │ │ } │ │ │ │ │ - var fills = node.getElementsByTagName("fill"); │ │ │ │ │ - var fill = (fills.length == 0) ? null : fills[0]; │ │ │ │ │ - if (!options.isFilled) { │ │ │ │ │ - if (fill) { │ │ │ │ │ - node.removeChild(fill); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - if (!fill) { │ │ │ │ │ - fill = this.createNode('olv:fill', node.id + "_fill"); │ │ │ │ │ - } │ │ │ │ │ - fill.opacity = style.fillOpacity; │ │ │ │ │ - │ │ │ │ │ - if (node._geometryClass === "OpenLayers.Geometry.Point" && │ │ │ │ │ - style.externalGraphic) { │ │ │ │ │ - │ │ │ │ │ - // override fillOpacity │ │ │ │ │ - if (style.graphicOpacity) { │ │ │ │ │ - fill.opacity = style.graphicOpacity; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - fill.src = style.externalGraphic; │ │ │ │ │ - fill.type = "frame"; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ - fill.aspect = "atmost"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (fill.parentNode != node) { │ │ │ │ │ - node.appendChild(fill); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: removeMarker │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * marker - {<OpenLayers.Marker>} │ │ │ │ │ + */ │ │ │ │ │ + removeMarker: function(marker) { │ │ │ │ │ + if (this.markers && this.markers.length) { │ │ │ │ │ + OpenLayers.Util.removeItem(this.markers, marker); │ │ │ │ │ + marker.erase(); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // additional rendering for rotated graphics or symbols │ │ │ │ │ - var rotation = style.rotation; │ │ │ │ │ - if ((rotation !== undefined || node._rotation !== undefined)) { │ │ │ │ │ - node._rotation = rotation; │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ - // make the fill fully transparent, because we now have │ │ │ │ │ - // the graphic as imagedata element. We cannot just remove │ │ │ │ │ - // the fill, because this is part of the hack described │ │ │ │ │ - // in graphicRotate │ │ │ │ │ - fill.opacity = 0; │ │ │ │ │ - } else if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ - node.style.rotation = rotation || 0; │ │ │ │ │ + /** │ │ │ │ │ + * Method: clearMarkers │ │ │ │ │ + * This method removes all markers from a layer. The markers are not │ │ │ │ │ + * destroyed by this function, but are removed from the list of markers. │ │ │ │ │ + */ │ │ │ │ │ + clearMarkers: function() { │ │ │ │ │ + if (this.markers != null) { │ │ │ │ │ + while (this.markers.length > 0) { │ │ │ │ │ + this.removeMarker(this.markers[0]); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // stroke │ │ │ │ │ - var strokes = node.getElementsByTagName("stroke"); │ │ │ │ │ - var stroke = (strokes.length == 0) ? null : strokes[0]; │ │ │ │ │ - if (!options.isStroked) { │ │ │ │ │ - node.stroked = false; │ │ │ │ │ - if (stroke) { │ │ │ │ │ - stroke.on = false; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawMarker │ │ │ │ │ + * Calculate the pixel location for the marker, create it, and │ │ │ │ │ + * add it to the layer's div │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * marker - {<OpenLayers.Marker>} │ │ │ │ │ + */ │ │ │ │ │ + drawMarker: function(marker) { │ │ │ │ │ + var px = this.map.getLayerPxFromLonLat(marker.lonlat); │ │ │ │ │ + if (px == null) { │ │ │ │ │ + marker.display(false); │ │ │ │ │ } else { │ │ │ │ │ - if (!stroke) { │ │ │ │ │ - stroke = this.createNode('olv:stroke', node.id + "_stroke"); │ │ │ │ │ - node.appendChild(stroke); │ │ │ │ │ - } │ │ │ │ │ - stroke.on = true; │ │ │ │ │ - stroke.color = style.strokeColor; │ │ │ │ │ - stroke.weight = style.strokeWidth + "px"; │ │ │ │ │ - stroke.opacity = style.strokeOpacity; │ │ │ │ │ - stroke.endcap = style.strokeLinecap == 'butt' ? 'flat' : │ │ │ │ │ - (style.strokeLinecap || 'round'); │ │ │ │ │ - if (style.strokeDashstyle) { │ │ │ │ │ - stroke.dashstyle = this.dashStyle(style); │ │ │ │ │ + if (!marker.isDrawn()) { │ │ │ │ │ + var markerImg = marker.draw(px); │ │ │ │ │ + this.div.appendChild(markerImg); │ │ │ │ │ + } else if (marker.icon) { │ │ │ │ │ + marker.icon.moveTo(px); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ - node.style.cursor = style.cursor; │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: graphicRotate │ │ │ │ │ - * If a point is to be styled with externalGraphic and rotation, VML fills │ │ │ │ │ - * cannot be used to display the graphic, because rotation of graphic │ │ │ │ │ - * fills is not supported by the VML implementation of Internet Explorer. │ │ │ │ │ - * This method creates a olv:imagedata element inside the VML node, │ │ │ │ │ - * DXImageTransform.Matrix and BasicImage filters for rotation and │ │ │ │ │ - * opacity, and a 3-step hack to remove rendering artefacts from the │ │ │ │ │ - * graphic and preserve the ability of graphics to trigger events. │ │ │ │ │ - * Finally, OpenLayers methods are used to determine the correct │ │ │ │ │ - * insertion point of the rotated image, because DXImageTransform.Matrix │ │ │ │ │ - * does the rotation without the ability to specify a rotation center │ │ │ │ │ - * point. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getDataExtent │ │ │ │ │ + * Calculates the max extent which includes all of the markers. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * xOffset - {Number} rotation center relative to image, x coordinate │ │ │ │ │ - * yOffset - {Number} rotation center relative to image, y coordinate │ │ │ │ │ - * style - {Object} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} │ │ │ │ │ */ │ │ │ │ │ - graphicRotate: function(node, xOffset, yOffset, style) { │ │ │ │ │ - var style = style || node._style; │ │ │ │ │ - var rotation = style.rotation || 0; │ │ │ │ │ - │ │ │ │ │ - var aspectRatio, size; │ │ │ │ │ - if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ - // load the image to determine its size │ │ │ │ │ - var img = new Image(); │ │ │ │ │ - img.onreadystatechange = OpenLayers.Function.bind(function() { │ │ │ │ │ - if (img.readyState == "complete" || │ │ │ │ │ - img.readyState == "interactive") { │ │ │ │ │ - aspectRatio = img.width / img.height; │ │ │ │ │ - size = Math.max(style.pointRadius * 2, │ │ │ │ │ - style.graphicWidth || 0, │ │ │ │ │ - style.graphicHeight || 0); │ │ │ │ │ - xOffset = xOffset * aspectRatio; │ │ │ │ │ - style.graphicWidth = size * aspectRatio; │ │ │ │ │ - style.graphicHeight = size; │ │ │ │ │ - this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ - } │ │ │ │ │ - }, this); │ │ │ │ │ - img.src = style.externalGraphic; │ │ │ │ │ + getDataExtent: function() { │ │ │ │ │ + var maxExtent = null; │ │ │ │ │ │ │ │ │ │ - // will be called again by the onreadystate handler │ │ │ │ │ - return; │ │ │ │ │ - } else { │ │ │ │ │ - size = Math.max(style.graphicWidth, style.graphicHeight); │ │ │ │ │ - aspectRatio = style.graphicWidth / style.graphicHeight; │ │ │ │ │ + if (this.markers && (this.markers.length > 0)) { │ │ │ │ │ + var maxExtent = new OpenLayers.Bounds(); │ │ │ │ │ + for (var i = 0, len = this.markers.length; i < len; i++) { │ │ │ │ │ + var marker = this.markers[i]; │ │ │ │ │ + maxExtent.extend(marker.lonlat); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - var width = Math.round(style.graphicWidth || size * aspectRatio); │ │ │ │ │ - var height = Math.round(style.graphicHeight || size); │ │ │ │ │ - node.style.width = width + "px"; │ │ │ │ │ - node.style.height = height + "px"; │ │ │ │ │ + return maxExtent; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Three steps are required to remove artefacts for images with │ │ │ │ │ - // transparent backgrounds (resulting from using DXImageTransform │ │ │ │ │ - // filters on svg objects), while preserving awareness for browser │ │ │ │ │ - // events on images: │ │ │ │ │ - // - Use the fill as usual (like for unrotated images) to handle │ │ │ │ │ - // events │ │ │ │ │ - // - specify an imagedata element with the same src as the fill │ │ │ │ │ - // - style the imagedata element with an AlphaImageLoader filter │ │ │ │ │ - // with empty src │ │ │ │ │ - var image = document.getElementById(node.id + "_image"); │ │ │ │ │ - if (!image) { │ │ │ │ │ - image = this.createNode("olv:imagedata", node.id + "_image"); │ │ │ │ │ - node.appendChild(image); │ │ │ │ │ - } │ │ │ │ │ - image.style.width = width + "px"; │ │ │ │ │ - image.style.height = height + "px"; │ │ │ │ │ - image.src = style.externalGraphic; │ │ │ │ │ - image.style.filter = │ │ │ │ │ - "progid:DXImageTransform.Microsoft.AlphaImageLoader(" + │ │ │ │ │ - "src='', sizingMethod='scale')"; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Markers" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/Text.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - var rot = rotation * Math.PI / 180; │ │ │ │ │ - var sintheta = Math.sin(rot); │ │ │ │ │ - var costheta = Math.cos(rot); │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - // do the rotation on the image │ │ │ │ │ - var filter = │ │ │ │ │ - "progid:DXImageTransform.Microsoft.Matrix(M11=" + costheta + │ │ │ │ │ - ",M12=" + (-sintheta) + ",M21=" + sintheta + ",M22=" + costheta + │ │ │ │ │ - ",SizingMethod='auto expand')\n"; │ │ │ │ │ │ │ │ │ │ - // set the opacity (needed for the imagedata) │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ - if (opacity && opacity != 1) { │ │ │ │ │ - filter += │ │ │ │ │ - "progid:DXImageTransform.Microsoft.BasicImage(opacity=" + │ │ │ │ │ - opacity + ")\n"; │ │ │ │ │ - } │ │ │ │ │ - node.style.filter = filter; │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/Markers.js │ │ │ │ │ + * @requires OpenLayers/Format/Text.js │ │ │ │ │ + * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // do the rotation again on a box, so we know the insertion point │ │ │ │ │ - var centerPoint = new OpenLayers.Geometry.Point(-xOffset, -yOffset); │ │ │ │ │ - var imgBox = new OpenLayers.Bounds(0, 0, width, height).toGeometry(); │ │ │ │ │ - imgBox.rotate(style.rotation, centerPoint); │ │ │ │ │ - var imgBounds = imgBox.getBounds(); │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.Text │ │ │ │ │ + * This layer creates markers given data in a text file. The <location> │ │ │ │ │ + * property of the layer (specified as a property of the options argument │ │ │ │ │ + * in the <OpenLayers.Layer.Text> constructor) points to a tab delimited │ │ │ │ │ + * file with data used to create markers. │ │ │ │ │ + * │ │ │ │ │ + * The first row of the data file should be a header line with the column names │ │ │ │ │ + * of the data. Each column should be delimited by a tab space. The │ │ │ │ │ + * possible columns are: │ │ │ │ │ + * - *point* lat,lon of the point where a marker is to be placed │ │ │ │ │ + * - *lat* Latitude of the point where a marker is to be placed │ │ │ │ │ + * - *lon* Longitude of the point where a marker is to be placed │ │ │ │ │ + * - *icon* or *image* URL of marker icon to use. │ │ │ │ │ + * - *iconSize* Size of Icon to use. │ │ │ │ │ + * - *iconOffset* Where the top-left corner of the icon is to be placed │ │ │ │ │ + * relative to the latitude and longitude of the point. │ │ │ │ │ + * - *title* The text of the 'title' is placed inside an 'h2' marker │ │ │ │ │ + * inside a popup, which opens when the marker is clicked. │ │ │ │ │ + * - *description* The text of the 'description' is placed below the h2 │ │ │ │ │ + * in the popup. this can be plain text or HTML. │ │ │ │ │ + * │ │ │ │ │ + * Example text file: │ │ │ │ │ + * (code) │ │ │ │ │ + * lat lon title description iconSize iconOffset icon │ │ │ │ │ + * 10 20 title description 21,25 -10,-25 http://www.openlayers.org/dev/img/marker.png │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Markers> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Text = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ │ │ │ │ │ - node.style.left = Math.round( │ │ │ │ │ - parseInt(node.style.left) + imgBounds.left) + "px"; │ │ │ │ │ - node.style.top = Math.round( │ │ │ │ │ - parseInt(node.style.top) - imgBounds.bottom) + "px"; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: location │ │ │ │ │ + * {String} URL of text file. Must be specified in the "options" argument │ │ │ │ │ + * of the constructor. Can not be changed once passed in. │ │ │ │ │ + */ │ │ │ │ │ + location: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Array(<OpenLayers.Feature>)} │ │ │ │ │ + */ │ │ │ │ │ + features: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: postDraw │ │ │ │ │ - * Does some node postprocessing to work around browser issues: │ │ │ │ │ - * - Some versions of Internet Explorer seem to be unable to set fillcolor │ │ │ │ │ - * and strokecolor to "none" correctly before the fill node is appended │ │ │ │ │ - * to a visible vml node. This method takes care of that and sets │ │ │ │ │ - * fillcolor and strokecolor again if needed. │ │ │ │ │ - * - In some cases, a node won't become visible after being drawn. Setting │ │ │ │ │ - * style.visibility to "visible" works around that. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ + * APIProperty: formatOptions │ │ │ │ │ + * {Object} Hash of options which should be passed to the format when it is │ │ │ │ │ + * created. Must be passed in the constructor. │ │ │ │ │ */ │ │ │ │ │ - postDraw: function(node) { │ │ │ │ │ - node.style.visibility = "visible"; │ │ │ │ │ - var fillColor = node._style.fillColor; │ │ │ │ │ - var strokeColor = node._style.strokeColor; │ │ │ │ │ - if (fillColor == "none" && │ │ │ │ │ - node.fillcolor != fillColor) { │ │ │ │ │ - node.fillcolor = fillColor; │ │ │ │ │ - } │ │ │ │ │ - if (strokeColor == "none" && │ │ │ │ │ - node.strokecolor != strokeColor) { │ │ │ │ │ - node.strokecolor = strokeColor; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + formatOptions: null, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: selectedFeature │ │ │ │ │ + * {<OpenLayers.Feature>} │ │ │ │ │ + */ │ │ │ │ │ + selectedFeature: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setNodeDimension │ │ │ │ │ - * Get the geometry's bounds, convert it to our vml coordinate system, │ │ │ │ │ - * then set the node's position, size, and local coordinate system. │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Layer.Text │ │ │ │ │ + * Create a text layer. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * name - {String} │ │ │ │ │ + * options - {Object} Object with properties to be set on the layer. │ │ │ │ │ + * Must include <location> property. │ │ │ │ │ */ │ │ │ │ │ - setNodeDimension: function(node, geometry) { │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.features = []; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var bbox = geometry.getBounds(); │ │ │ │ │ - if (bbox) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + // Warning: Layer.Markers.destroy() must be called prior to calling │ │ │ │ │ + // clearFeatures() here, otherwise we leak memory. Indeed, if │ │ │ │ │ + // Layer.Markers.destroy() is called after clearFeatures(), it won't be │ │ │ │ │ + // able to remove the marker image elements from the layer's div since │ │ │ │ │ + // the markers will have been destroyed by clearFeatures(). │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.destroy.apply(this, arguments); │ │ │ │ │ + this.clearFeatures(); │ │ │ │ │ + this.features = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var scaledBox = │ │ │ │ │ - new OpenLayers.Bounds(((bbox.left - this.featureDx) / resolution - this.offset.x) | 0, │ │ │ │ │ - (bbox.bottom / resolution - this.offset.y) | 0, │ │ │ │ │ - ((bbox.right - this.featureDx) / resolution - this.offset.x) | 0, │ │ │ │ │ - (bbox.top / resolution - this.offset.y) | 0); │ │ │ │ │ + /** │ │ │ │ │ + * Method: loadText │ │ │ │ │ + * Start the load of the Text data. Don't do this when we first add the layer, │ │ │ │ │ + * since we may not be visible at any point, and it would therefore be a waste. │ │ │ │ │ + */ │ │ │ │ │ + loadText: function() { │ │ │ │ │ + if (!this.loaded) { │ │ │ │ │ + if (this.location != null) { │ │ │ │ │ │ │ │ │ │ - // Set the internal coordinate system to draw the path │ │ │ │ │ - node.style.left = scaledBox.left + "px"; │ │ │ │ │ - node.style.top = scaledBox.top + "px"; │ │ │ │ │ - node.style.width = scaledBox.getWidth() + "px"; │ │ │ │ │ - node.style.height = scaledBox.getHeight() + "px"; │ │ │ │ │ + var onFail = function(e) { │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - node.coordorigin = scaledBox.left + " " + scaledBox.top; │ │ │ │ │ - node.coordsize = scaledBox.getWidth() + " " + scaledBox.getHeight(); │ │ │ │ │ + this.events.triggerEvent("loadstart"); │ │ │ │ │ + OpenLayers.Request.GET({ │ │ │ │ │ + url: this.location, │ │ │ │ │ + success: this.parseData, │ │ │ │ │ + failure: onFail, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.loaded = true; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: dashStyle │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * If layer is visible and Text has not been loaded, load Text. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A VML compliant 'stroke-dasharray' value │ │ │ │ │ + * bounds - {Object} │ │ │ │ │ + * zoomChanged - {Object} │ │ │ │ │ + * minor - {Object} │ │ │ │ │ */ │ │ │ │ │ - dashStyle: function(style) { │ │ │ │ │ - var dash = style.strokeDashstyle; │ │ │ │ │ - switch (dash) { │ │ │ │ │ - case 'solid': │ │ │ │ │ - case 'dot': │ │ │ │ │ - case 'dash': │ │ │ │ │ - case 'dashdot': │ │ │ │ │ - case 'longdash': │ │ │ │ │ - case 'longdashdot': │ │ │ │ │ - return dash; │ │ │ │ │ - default: │ │ │ │ │ - // very basic guessing of dash style patterns │ │ │ │ │ - var parts = dash.split(/[ ,]/); │ │ │ │ │ - if (parts.length == 2) { │ │ │ │ │ - if (1 * parts[0] >= 2 * parts[1]) { │ │ │ │ │ - return "longdash"; │ │ │ │ │ - } │ │ │ │ │ - return (parts[0] == 1 || parts[1] == 1) ? "dot" : "dash"; │ │ │ │ │ - } else if (parts.length == 4) { │ │ │ │ │ - return (1 * parts[0] >= 2 * parts[1]) ? "longdashdot" : │ │ │ │ │ - "dashdot"; │ │ │ │ │ - } │ │ │ │ │ - return "solid"; │ │ │ │ │ + moveTo: function(bounds, zoomChanged, minor) { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + if (this.visibility && !this.loaded) { │ │ │ │ │ + this.loadText(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createNode │ │ │ │ │ - * Create a new node │ │ │ │ │ + * Method: parseData │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * type - {String} Kind of node to draw │ │ │ │ │ - * id - {String} Id for node │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A new node of the given type and id │ │ │ │ │ + * ajaxRequest - {<OpenLayers.Request.XMLHttpRequest>} │ │ │ │ │ */ │ │ │ │ │ - createNode: function(type, id) { │ │ │ │ │ - var node = document.createElement(type); │ │ │ │ │ - if (id) { │ │ │ │ │ - node.id = id; │ │ │ │ │ + parseData: function(ajaxRequest) { │ │ │ │ │ + var text = ajaxRequest.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(); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // IE hack to make elements unselectable, to prevent 'blue flash' │ │ │ │ │ - // while dragging vectors; #1410 │ │ │ │ │ - node.unselectable = 'on'; │ │ │ │ │ - node.onselectstart = OpenLayers.Function.False; │ │ │ │ │ + var parser = new OpenLayers.Format.Text(options); │ │ │ │ │ + var features = parser.read(text); │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + var data = {}; │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + var location; │ │ │ │ │ + var iconSize, iconOffset; │ │ │ │ │ │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ + location = new OpenLayers.LonLat(feature.geometry.x, │ │ │ │ │ + feature.geometry.y); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: nodeTypeCompare │ │ │ │ │ - * Determine whether a node is of a given type │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} An VML element │ │ │ │ │ - * type - {String} Kind of node │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ - */ │ │ │ │ │ - nodeTypeCompare: function(node, type) { │ │ │ │ │ + if (feature.style.graphicWidth && │ │ │ │ │ + feature.style.graphicHeight) { │ │ │ │ │ + iconSize = new OpenLayers.Size( │ │ │ │ │ + feature.style.graphicWidth, │ │ │ │ │ + feature.style.graphicHeight); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - //split type │ │ │ │ │ - var subType = type; │ │ │ │ │ - var splitIndex = subType.indexOf(":"); │ │ │ │ │ - if (splitIndex != -1) { │ │ │ │ │ - subType = subType.substr(splitIndex + 1); │ │ │ │ │ - } │ │ │ │ │ + // FIXME: At the moment, we only use this if we have an │ │ │ │ │ + // externalGraphic, because icon has no setOffset API Method. │ │ │ │ │ + /** │ │ │ │ │ + * FIXME FIRST!! │ │ │ │ │ + * The Text format does all sorts of parseFloating │ │ │ │ │ + * The result of a parseFloat for a bogus string is NaN. That │ │ │ │ │ + * means the three possible values here are undefined, NaN, or a │ │ │ │ │ + * number. The previous check was an identity check for null. This │ │ │ │ │ + * means it was failing for all undefined or NaN. A slightly better │ │ │ │ │ + * check is for undefined. An even better check is to see if the │ │ │ │ │ + * value is a number (see #1441). │ │ │ │ │ + */ │ │ │ │ │ + if (feature.style.graphicXOffset !== undefined && │ │ │ │ │ + feature.style.graphicYOffset !== undefined) { │ │ │ │ │ + iconOffset = new OpenLayers.Pixel( │ │ │ │ │ + feature.style.graphicXOffset, │ │ │ │ │ + feature.style.graphicYOffset); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - //split nodeName │ │ │ │ │ - var nodeName = node.nodeName; │ │ │ │ │ - splitIndex = nodeName.indexOf(":"); │ │ │ │ │ - if (splitIndex != -1) { │ │ │ │ │ - nodeName = nodeName.substr(splitIndex + 1); │ │ │ │ │ - } │ │ │ │ │ + if (feature.style.externalGraphic != null) { │ │ │ │ │ + data.icon = new OpenLayers.Icon(feature.style.externalGraphic, │ │ │ │ │ + iconSize, │ │ │ │ │ + iconOffset); │ │ │ │ │ + } else { │ │ │ │ │ + data.icon = OpenLayers.Marker.defaultIcon(); │ │ │ │ │ │ │ │ │ │ - return (subType == nodeName); │ │ │ │ │ - }, │ │ │ │ │ + //allows for the case where the image url is not │ │ │ │ │ + // specified but the size is. use a default icon │ │ │ │ │ + // but change the size │ │ │ │ │ + if (iconSize != null) { │ │ │ │ │ + data.icon.setSize(iconSize); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createRenderRoot │ │ │ │ │ - * Create the renderer root │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The specific render engine's root element │ │ │ │ │ - */ │ │ │ │ │ - createRenderRoot: function() { │ │ │ │ │ - return this.nodeFactory(this.container.id + "_vmlRoot", "div"); │ │ │ │ │ + if ((feature.attributes.title != null) && │ │ │ │ │ + (feature.attributes.description != null)) { │ │ │ │ │ + data['popupContentHTML'] = │ │ │ │ │ + '<h2>' + feature.attributes.title + '</h2>' + │ │ │ │ │ + '<p>' + feature.attributes.description + '</p>'; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + data['overflow'] = feature.attributes.overflow || "auto"; │ │ │ │ │ + │ │ │ │ │ + var markerFeature = new OpenLayers.Feature(this, location, data); │ │ │ │ │ + this.features.push(markerFeature); │ │ │ │ │ + var marker = markerFeature.createMarker(); │ │ │ │ │ + if ((feature.attributes.title != null) && │ │ │ │ │ + (feature.attributes.description != null)) { │ │ │ │ │ + marker.events.register('click', markerFeature, this.markerClick); │ │ │ │ │ + } │ │ │ │ │ + this.addMarker(marker); │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createRoot │ │ │ │ │ - * Create the main root element │ │ │ │ │ + * Property: markerClick │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * suffix - {String} suffix to append to the id │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Context: │ │ │ │ │ + * - {<OpenLayers.Feature>} │ │ │ │ │ */ │ │ │ │ │ - createRoot: function(suffix) { │ │ │ │ │ - return this.nodeFactory(this.container.id + suffix, "olv:group"); │ │ │ │ │ + markerClick: function(evt) { │ │ │ │ │ + var sameMarkerClicked = (this == this.layer.selectedFeature); │ │ │ │ │ + this.layer.selectedFeature = (!sameMarkerClicked) ? this : null; │ │ │ │ │ + for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ + this.layer.map.removePopup(this.layer.map.popups[i]); │ │ │ │ │ + } │ │ │ │ │ + if (!sameMarkerClicked) { │ │ │ │ │ + this.layer.map.addPopup(this.createPopup()); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /************************************** │ │ │ │ │ - * * │ │ │ │ │ - * GEOMETRY DRAWING FUNCTIONS * │ │ │ │ │ - * * │ │ │ │ │ - **************************************/ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPoint │ │ │ │ │ - * Render a point │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or false if the point could not be drawn │ │ │ │ │ + * Method: clearFeatures │ │ │ │ │ */ │ │ │ │ │ - drawPoint: function(node, geometry) { │ │ │ │ │ - return this.drawCircle(node, geometry, 1); │ │ │ │ │ + clearFeatures: function() { │ │ │ │ │ + if (this.features != null) { │ │ │ │ │ + while (this.features.length > 0) { │ │ │ │ │ + var feature = this.features[0]; │ │ │ │ │ + OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ + feature.destroy(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawCircle │ │ │ │ │ - * Render a circle. │ │ │ │ │ - * Size and Center a circle given geometry (x,y center) and radius │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * radius - {float} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or false if the circle could not ne drawn │ │ │ │ │ - */ │ │ │ │ │ - drawCircle: function(node, geometry, radius) { │ │ │ │ │ - if (!isNaN(geometry.x) && !isNaN(geometry.y)) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Text" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/ArcGISCache.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - node.style.left = ((((geometry.x - this.featureDx) / resolution - this.offset.x) | 0) - radius) + "px"; │ │ │ │ │ - node.style.top = (((geometry.y / resolution - this.offset.y) | 0) - radius) + "px"; │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - var diameter = radius * 2; │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - node.style.width = diameter + "px"; │ │ │ │ │ - node.style.height = diameter + "px"; │ │ │ │ │ - return node; │ │ │ │ │ - } │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.ArcGISCache │ │ │ │ │ + * Layer for accessing cached map tiles from an ArcGIS Server style mapcache. │ │ │ │ │ + * Tile must already be cached for this layer to access it. This does not require │ │ │ │ │ + * ArcGIS Server itself. │ │ │ │ │ + * │ │ │ │ │ + * A few attempts have been made at this kind of layer before. See │ │ │ │ │ + * http://trac.osgeo.org/openlayers/ticket/1967 │ │ │ │ │ + * and │ │ │ │ │ + * http://trac.osgeo.org/openlayers/browser/sandbox/tschaub/arcgiscache/lib/OpenLayers/Layer/ArcGISCache.js │ │ │ │ │ + * │ │ │ │ │ + * Typically the problem encountered is that the tiles seem to "jump around". │ │ │ │ │ + * This is due to the fact that the actual max extent for the tiles on AGS layers │ │ │ │ │ + * changes at each zoom level due to the way these caches are constructed. │ │ │ │ │ + * We have attempted to use the resolutions, tile size, and tile origin │ │ │ │ │ + * from the cache meta data to make the appropriate changes to the max extent │ │ │ │ │ + * of the tile to compensate for this behavior. This must be done as zoom levels change │ │ │ │ │ + * and before tiles are requested, which is why methods from base classes are overridden. │ │ │ │ │ + * │ │ │ │ │ + * For reference, you can access mapcache meta data in two ways. For accessing a │ │ │ │ │ + * mapcache through ArcGIS Server, you can simply go to the landing page for the │ │ │ │ │ + * layer. (ie. http://services.arcgisonline.com/ArcGIS/rest/services/World_Street_Map/MapServer) │ │ │ │ │ + * For accessing it directly through HTTP, there should always be a conf.xml file │ │ │ │ │ + * in the root directory. │ │ │ │ │ + * (ie. http://serverx.esri.com/arcgiscache/DG_County_roads_yesA_backgroundDark/Layers/conf.xml) │ │ │ │ │ + * │ │ │ │ │ + *Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.XYZ> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.ArcGISCache = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String | Array} The base URL for the layer cache. You can also │ │ │ │ │ + * provide a list of URL strings for the layer if your cache is │ │ │ │ │ + * available from multiple origins. This must be set before the layer │ │ │ │ │ + * is drawn. │ │ │ │ │ + */ │ │ │ │ │ + url: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLineString │ │ │ │ │ - * Render a linestring. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * APIProperty: tileOrigin │ │ │ │ │ + * {<OpenLayers.LonLat>} The location of the tile origin for the cache. │ │ │ │ │ + * An ArcGIS cache has it's origin at the upper-left (lowest x value │ │ │ │ │ + * and highest y value of the coordinate system). The units for the │ │ │ │ │ + * tile origin should be the same as the units for the cached data. │ │ │ │ │ */ │ │ │ │ │ - drawLineString: function(node, geometry) { │ │ │ │ │ - return this.drawLine(node, geometry, false); │ │ │ │ │ - }, │ │ │ │ │ + tileOrigin: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLinearRing │ │ │ │ │ - * Render a linearring │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * APIProperty: tileSize │ │ │ │ │ + * {<OpenLayers.Size>} This size of each tile. Defaults to 256 by 256 pixels. │ │ │ │ │ */ │ │ │ │ │ - drawLinearRing: function(node, geometry) { │ │ │ │ │ - return this.drawLine(node, geometry, true); │ │ │ │ │ - }, │ │ │ │ │ + tileSize: new OpenLayers.Size(256, 256), │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: DrawLine │ │ │ │ │ - * Render a line. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * closeLine - {Boolean} Close the line? (make it a ring?) │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * APIProperty: useAGS │ │ │ │ │ + * {Boolean} Indicates if we are going to be accessing the ArcGIS Server (AGS) │ │ │ │ │ + * cache via an AGS MapServer or directly through HTTP. When accessing via │ │ │ │ │ + * AGS the path structure uses a standard z/y/x structure. But AGS actually │ │ │ │ │ + * stores the tile images on disk using a hex based folder structure that looks │ │ │ │ │ + * like "http://example.com/mylayer/L00/R00000000/C00000000.png". Learn more │ │ │ │ │ + * about this here: │ │ │ │ │ + * http://blogs.esri.com/Support/blogs/mappingcenter/archive/2010/08/20/Checking-Your-Local-Cache-Folders.aspx │ │ │ │ │ + * Defaults to true; │ │ │ │ │ */ │ │ │ │ │ - drawLine: function(node, geometry, closeLine) { │ │ │ │ │ + useArcGISServer: true, │ │ │ │ │ │ │ │ │ │ - this.setNodeDimension(node, geometry); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: type │ │ │ │ │ + * {String} Image type for the layer. This becomes the filename extension │ │ │ │ │ + * in tile requests. Default is "png" (generating a url like │ │ │ │ │ + * "http://example.com/mylayer/L00/R00000000/C00000000.png"). │ │ │ │ │ + */ │ │ │ │ │ + type: 'png', │ │ │ │ │ │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var numComponents = geometry.components.length; │ │ │ │ │ - var parts = new Array(numComponents); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: useScales │ │ │ │ │ + * {Boolean} Optional override to indicate that the layer should use 'scale' information │ │ │ │ │ + * returned from the server capabilities object instead of 'resolution' information. │ │ │ │ │ + * This can be important if your tile server uses an unusual DPI for the tiles. │ │ │ │ │ + */ │ │ │ │ │ + useScales: false, │ │ │ │ │ │ │ │ │ │ - var comp, x, y; │ │ │ │ │ - for (var i = 0; i < numComponents; i++) { │ │ │ │ │ - comp = geometry.components[i]; │ │ │ │ │ - x = ((comp.x - this.featureDx) / resolution - this.offset.x) | 0; │ │ │ │ │ - y = (comp.y / resolution - this.offset.y) | 0; │ │ │ │ │ - parts[i] = " " + x + "," + y + " l "; │ │ │ │ │ - } │ │ │ │ │ - var end = (closeLine) ? " x e" : " e"; │ │ │ │ │ - node.path = "m" + parts.join("") + end; │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: overrideDPI │ │ │ │ │ + * {Boolean} Optional override to change the OpenLayers.DOTS_PER_INCH setting based │ │ │ │ │ + * on the tile information in the server capabilities object. This can be useful │ │ │ │ │ + * if your server has a non-standard DPI setting on its tiles, and you're only using │ │ │ │ │ + * tiles with that DPI. This value is used while OpenLayers is calculating resolution │ │ │ │ │ + * using scales, and is not necessary if you have resolution information. (This is │ │ │ │ │ + * typically the case) Regardless, this setting can be useful, but is dangerous │ │ │ │ │ + * because it will impact other layers while calculating resolution. Only use this │ │ │ │ │ + * if you know what you are doing. (See OpenLayers.Util.getResolutionFromScale) │ │ │ │ │ + */ │ │ │ │ │ + overrideDPI: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPolygon │ │ │ │ │ - * Render a polygon │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * Constructor: OpenLayers.Layer.ArcGISCache │ │ │ │ │ + * Creates a new instance of this class │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} │ │ │ │ │ + * url - {String} │ │ │ │ │ + * options - {Object} extra layer options │ │ │ │ │ */ │ │ │ │ │ - drawPolygon: function(node, geometry) { │ │ │ │ │ - this.setNodeDimension(node, geometry); │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ + if (this.resolutions) { │ │ │ │ │ + this.serverResolutions = this.resolutions; │ │ │ │ │ + this.maxExtent = this.getMaxExtentForResolution(this.resolutions[0]); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var path = []; │ │ │ │ │ - var j, jj, points, area, first, second, i, ii, comp, pathComp, x, y; │ │ │ │ │ - for (j = 0, jj = geometry.components.length; j < jj; j++) { │ │ │ │ │ - path.push("m"); │ │ │ │ │ - points = geometry.components[j].components; │ │ │ │ │ - // we only close paths of interior rings with area │ │ │ │ │ - area = (j === 0); │ │ │ │ │ - first = null; │ │ │ │ │ - second = null; │ │ │ │ │ - for (i = 0, ii = points.length; i < ii; i++) { │ │ │ │ │ - comp = points[i]; │ │ │ │ │ - x = ((comp.x - this.featureDx) / resolution - this.offset.x) | 0; │ │ │ │ │ - y = (comp.y / resolution - this.offset.y) | 0; │ │ │ │ │ - pathComp = " " + x + "," + y; │ │ │ │ │ - path.push(pathComp); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - path.push(" l"); │ │ │ │ │ + // this block steps through translating the values from the server layer JSON │ │ │ │ │ + // capabilities object into values that we can use. This is also a helpful │ │ │ │ │ + // reference when configuring this layer directly. │ │ │ │ │ + if (this.layerInfo) { │ │ │ │ │ + // alias the object │ │ │ │ │ + var info = this.layerInfo; │ │ │ │ │ + │ │ │ │ │ + // build our extents │ │ │ │ │ + var startingTileExtent = new OpenLayers.Bounds( │ │ │ │ │ + info.fullExtent.xmin, │ │ │ │ │ + info.fullExtent.ymin, │ │ │ │ │ + info.fullExtent.xmax, │ │ │ │ │ + info.fullExtent.ymax │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + // set our projection based on the given spatial reference. │ │ │ │ │ + // esri uses slightly different IDs, so this may not be comprehensive │ │ │ │ │ + this.projection = 'EPSG:' + info.spatialReference.wkid; │ │ │ │ │ + this.sphericalMercator = (info.spatialReference.wkid == 102100); │ │ │ │ │ + │ │ │ │ │ + // convert esri units into openlayers units (basic feet or meters only) │ │ │ │ │ + this.units = (info.units == "esriFeet") ? 'ft' : 'm'; │ │ │ │ │ + │ │ │ │ │ + // optional extended section based on whether or not the server returned │ │ │ │ │ + // specific tile information │ │ │ │ │ + if (!!info.tileInfo) { │ │ │ │ │ + // either set the tiles based on rows/columns, or specific width/height │ │ │ │ │ + this.tileSize = new OpenLayers.Size( │ │ │ │ │ + info.tileInfo.width || info.tileInfo.cols, │ │ │ │ │ + info.tileInfo.height || info.tileInfo.rows │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + // this must be set when manually configuring this layer │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat( │ │ │ │ │ + info.tileInfo.origin.x, │ │ │ │ │ + info.tileInfo.origin.y │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + var upperLeft = new OpenLayers.Geometry.Point( │ │ │ │ │ + startingTileExtent.left, │ │ │ │ │ + startingTileExtent.top │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + var bottomRight = new OpenLayers.Geometry.Point( │ │ │ │ │ + startingTileExtent.right, │ │ │ │ │ + startingTileExtent.bottom │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + if (this.useScales) { │ │ │ │ │ + this.scales = []; │ │ │ │ │ + } else { │ │ │ │ │ + this.resolutions = []; │ │ │ │ │ } │ │ │ │ │ - if (!area) { │ │ │ │ │ - // IE improperly renders sub-paths that have no area. │ │ │ │ │ - // Instead of checking the area of every ring, we confirm │ │ │ │ │ - // the ring has at least three distinct points. This does │ │ │ │ │ - // not catch all non-zero area cases, but it greatly improves │ │ │ │ │ - // interior ring digitizing and is a minor performance hit │ │ │ │ │ - // when rendering rings with many points. │ │ │ │ │ - if (!first) { │ │ │ │ │ - first = pathComp; │ │ │ │ │ - } else if (first != pathComp) { │ │ │ │ │ - if (!second) { │ │ │ │ │ - second = pathComp; │ │ │ │ │ - } else if (second != pathComp) { │ │ │ │ │ - // stop looking │ │ │ │ │ - area = true; │ │ │ │ │ + │ │ │ │ │ + this.lods = []; │ │ │ │ │ + for (var key in info.tileInfo.lods) { │ │ │ │ │ + if (info.tileInfo.lods.hasOwnProperty(key)) { │ │ │ │ │ + var lod = info.tileInfo.lods[key]; │ │ │ │ │ + if (this.useScales) { │ │ │ │ │ + this.scales.push(lod.scale); │ │ │ │ │ + } else { │ │ │ │ │ + this.resolutions.push(lod.resolution); │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + var start = this.getContainingTileCoords(upperLeft, lod.resolution); │ │ │ │ │ + lod.startTileCol = start.x; │ │ │ │ │ + lod.startTileRow = start.y; │ │ │ │ │ + │ │ │ │ │ + var end = this.getContainingTileCoords(bottomRight, lod.resolution); │ │ │ │ │ + lod.endTileCol = end.x; │ │ │ │ │ + lod.endTileRow = end.y; │ │ │ │ │ + this.lods.push(lod); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + this.maxExtent = this.calculateMaxExtentWithLOD(this.lods[0]); │ │ │ │ │ + this.serverResolutions = this.resolutions; │ │ │ │ │ + if (this.overrideDPI && info.tileInfo.dpi) { │ │ │ │ │ + // see comment above for 'overrideDPI' │ │ │ │ │ + OpenLayers.DOTS_PER_INCH = info.tileInfo.dpi; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - path.push(area ? " x " : " "); │ │ │ │ │ } │ │ │ │ │ - path.push("e"); │ │ │ │ │ - node.path = path.join(""); │ │ │ │ │ - return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawRectangle │ │ │ │ │ - * Render a rectangle │ │ │ │ │ + /** │ │ │ │ │ + * Method: getContainingTileCoords │ │ │ │ │ + * Calculates the x/y pixel corresponding to the position of the tile │ │ │ │ │ + * that contains the given point and for the for the given resolution. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Pixel>} The x/y pixel corresponding to the position │ │ │ │ │ + * of the upper left tile for the given resolution. │ │ │ │ │ */ │ │ │ │ │ - drawRectangle: function(node, geometry) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ + getContainingTileCoords: function(point, res) { │ │ │ │ │ + return new OpenLayers.Pixel( │ │ │ │ │ + Math.max(Math.floor((point.x - this.tileOrigin.lon) / (this.tileSize.w * res)), 0), │ │ │ │ │ + Math.max(Math.floor((this.tileOrigin.lat - point.y) / (this.tileSize.h * res)), 0) │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - node.style.left = (((geometry.x - this.featureDx) / resolution - this.offset.x) | 0) + "px"; │ │ │ │ │ - node.style.top = ((geometry.y / resolution - this.offset.y) | 0) + "px"; │ │ │ │ │ - node.style.width = ((geometry.width / resolution) | 0) + "px"; │ │ │ │ │ - node.style.height = ((geometry.height / resolution) | 0) + "px"; │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateMaxExtentWithLOD │ │ │ │ │ + * Given a Level of Detail object from the server, this function │ │ │ │ │ + * calculates the actual max extent │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * lod - {Object} a Level of Detail Object from the server capabilities object │ │ │ │ │ + representing a particular zoom level │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} The actual extent of the tiles for the given zoom level │ │ │ │ │ + */ │ │ │ │ │ + calculateMaxExtentWithLOD: function(lod) { │ │ │ │ │ + // the max extent we're provided with just overlaps some tiles │ │ │ │ │ + // our real extent is the bounds of all the tiles we touch │ │ │ │ │ │ │ │ │ │ - return node; │ │ │ │ │ + var numTileCols = (lod.endTileCol - lod.startTileCol) + 1; │ │ │ │ │ + var numTileRows = (lod.endTileRow - lod.startTileRow) + 1; │ │ │ │ │ + │ │ │ │ │ + var minX = this.tileOrigin.lon + (lod.startTileCol * this.tileSize.w * lod.resolution); │ │ │ │ │ + var maxX = minX + (numTileCols * this.tileSize.w * lod.resolution); │ │ │ │ │ + │ │ │ │ │ + var maxY = this.tileOrigin.lat - (lod.startTileRow * this.tileSize.h * lod.resolution); │ │ │ │ │ + var minY = maxY - (numTileRows * this.tileSize.h * lod.resolution); │ │ │ │ │ + return new OpenLayers.Bounds(minX, minY, maxX, maxY); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawText │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateMaxExtentWithExtent │ │ │ │ │ + * Given a 'suggested' max extent from the server, this function uses │ │ │ │ │ + * information about the actual tile sizes to determine the actual │ │ │ │ │ + * extent of the layer. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * style - │ │ │ │ │ - * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} The 'suggested' extent for the layer │ │ │ │ │ + * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} The actual extent of the tiles for the given zoom level │ │ │ │ │ */ │ │ │ │ │ - drawText: function(featureId, style, location) { │ │ │ │ │ - var label = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX, "olv:rect"); │ │ │ │ │ - var textbox = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX + "_textbox", "olv:textbox"); │ │ │ │ │ + calculateMaxExtentWithExtent: function(extent, res) { │ │ │ │ │ + var upperLeft = new OpenLayers.Geometry.Point(extent.left, extent.top); │ │ │ │ │ + var bottomRight = new OpenLayers.Geometry.Point(extent.right, extent.bottom); │ │ │ │ │ + var start = this.getContainingTileCoords(upperLeft, res); │ │ │ │ │ + var end = this.getContainingTileCoords(bottomRight, res); │ │ │ │ │ + var lod = { │ │ │ │ │ + resolution: res, │ │ │ │ │ + startTileCol: start.x, │ │ │ │ │ + startTileRow: start.y, │ │ │ │ │ + endTileCol: end.x, │ │ │ │ │ + endTileRow: end.y │ │ │ │ │ + }; │ │ │ │ │ + return this.calculateMaxExtentWithLOD(lod); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - label.style.left = (((location.x - this.featureDx) / resolution - this.offset.x) | 0) + "px"; │ │ │ │ │ - label.style.top = ((location.y / resolution - this.offset.y) | 0) + "px"; │ │ │ │ │ - label.style.flip = "y"; │ │ │ │ │ + /** │ │ │ │ │ + * Method: getUpperLeftTileCoord │ │ │ │ │ + * Calculates the x/y pixel corresponding to the position │ │ │ │ │ + * of the upper left tile for the given resolution. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Pixel>} The x/y pixel corresponding to the position │ │ │ │ │ + * of the upper left tile for the given resolution. │ │ │ │ │ + */ │ │ │ │ │ + getUpperLeftTileCoord: function(res) { │ │ │ │ │ + var upperLeft = new OpenLayers.Geometry.Point( │ │ │ │ │ + this.maxExtent.left, │ │ │ │ │ + this.maxExtent.top); │ │ │ │ │ + return this.getContainingTileCoords(upperLeft, res); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - textbox.innerText = style.label; │ │ │ │ │ + /** │ │ │ │ │ + * Method: getLowerRightTileCoord │ │ │ │ │ + * Calculates the x/y pixel corresponding to the position │ │ │ │ │ + * of the lower right tile for the given resolution. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Pixel>} The x/y pixel corresponding to the position │ │ │ │ │ + * of the lower right tile for the given resolution. │ │ │ │ │ + */ │ │ │ │ │ + getLowerRightTileCoord: function(res) { │ │ │ │ │ + var bottomRight = new OpenLayers.Geometry.Point( │ │ │ │ │ + this.maxExtent.right, │ │ │ │ │ + this.maxExtent.bottom); │ │ │ │ │ + return this.getContainingTileCoords(bottomRight, res); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ - textbox.style.cursor = style.cursor; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontColor) { │ │ │ │ │ - textbox.style.color = style.fontColor; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontOpacity) { │ │ │ │ │ - textbox.style.filter = 'alpha(opacity=' + (style.fontOpacity * 100) + ')'; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontFamily) { │ │ │ │ │ - textbox.style.fontFamily = style.fontFamily; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontSize) { │ │ │ │ │ - textbox.style.fontSize = style.fontSize; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontWeight) { │ │ │ │ │ - textbox.style.fontWeight = style.fontWeight; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStyle) { │ │ │ │ │ - textbox.style.fontStyle = style.fontStyle; │ │ │ │ │ - } │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - label._featureId = featureId; │ │ │ │ │ - textbox._featureId = featureId; │ │ │ │ │ - textbox._geometry = location; │ │ │ │ │ - textbox._geometryClass = location.CLASS_NAME; │ │ │ │ │ - } │ │ │ │ │ - textbox.style.whiteSpace = "nowrap"; │ │ │ │ │ - // fun with IE: IE7 in standards compliant mode does not display any │ │ │ │ │ - // text with a left inset of 0. So we set this to 1px and subtract one │ │ │ │ │ - // pixel later when we set label.style.left │ │ │ │ │ - textbox.inset = "1px,0px,0px,0px"; │ │ │ │ │ + /** │ │ │ │ │ + * Method: getMaxExtentForResolution │ │ │ │ │ + * Since the max extent of a set of tiles can change from zoom level │ │ │ │ │ + * to zoom level, we need to be able to calculate that max extent │ │ │ │ │ + * for a given resolution. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} The extent for this resolution │ │ │ │ │ + */ │ │ │ │ │ + getMaxExtentForResolution: function(res) { │ │ │ │ │ + var start = this.getUpperLeftTileCoord(res); │ │ │ │ │ + var end = this.getLowerRightTileCoord(res); │ │ │ │ │ │ │ │ │ │ - if (!label.parentNode) { │ │ │ │ │ - label.appendChild(textbox); │ │ │ │ │ - this.textRoot.appendChild(label); │ │ │ │ │ - } │ │ │ │ │ + var numTileCols = (end.x - start.x) + 1; │ │ │ │ │ + var numTileRows = (end.y - start.y) + 1; │ │ │ │ │ │ │ │ │ │ - var align = style.labelAlign || "cm"; │ │ │ │ │ - if (align.length == 1) { │ │ │ │ │ - align += "m"; │ │ │ │ │ - } │ │ │ │ │ - var xshift = textbox.clientWidth * │ │ │ │ │ - (OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(0, 1)]); │ │ │ │ │ - var yshift = textbox.clientHeight * │ │ │ │ │ - (OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(1, 1)]); │ │ │ │ │ - label.style.left = parseInt(label.style.left) - xshift - 1 + "px"; │ │ │ │ │ - label.style.top = parseInt(label.style.top) + yshift + "px"; │ │ │ │ │ + var minX = this.tileOrigin.lon + (start.x * this.tileSize.w * res); │ │ │ │ │ + var maxX = minX + (numTileCols * this.tileSize.w * res); │ │ │ │ │ │ │ │ │ │ + var maxY = this.tileOrigin.lat - (start.y * this.tileSize.h * res); │ │ │ │ │ + var minY = maxY - (numTileRows * this.tileSize.h * res); │ │ │ │ │ + return new OpenLayers.Bounds(minX, minY, maxX, maxY); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveRoot │ │ │ │ │ - * moves this renderer's root to a different renderer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * renderer - {<OpenLayers.Renderer>} target renderer for the moved root │ │ │ │ │ - * root - {DOMElement} optional root node. To be used when this renderer │ │ │ │ │ - * holds roots from multiple layers to tell this method which one to │ │ │ │ │ - * detach │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Returns an exact clone of this OpenLayers.Layer.ArcGISCache │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true if successful, false otherwise │ │ │ │ │ + * Parameters: │ │ │ │ │ + * [obj] - {Object} optional object to assign the cloned instance to. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.ArcGISCache>} clone of this instance │ │ │ │ │ */ │ │ │ │ │ - moveRoot: function(renderer) { │ │ │ │ │ - var layer = this.map.getLayer(renderer.container.id); │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Vector.RootContainer) { │ │ │ │ │ - layer = this.map.getLayer(this.container.id); │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.ArcGISCache(this.name, this.url, this.options); │ │ │ │ │ } │ │ │ │ │ - layer && layer.renderer.clear(); │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.moveRoot.apply(this, arguments); │ │ │ │ │ - layer && layer.redraw(); │ │ │ │ │ + return OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: importSymbol │ │ │ │ │ - * add a new symbol definition from the rendererer's symbol hash │ │ │ │ │ + * Method: initGriddedTiles │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * graphicName - {String} name of the symbol to import │ │ │ │ │ - * │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + */ │ │ │ │ │ + initGriddedTiles: function(bounds) { │ │ │ │ │ + delete this._tileOrigin; │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initGriddedTiles.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getMaxExtent │ │ │ │ │ + * Get this layer's maximum extent. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} - hash of {DOMElement} "symbol" and {Number} "size" │ │ │ │ │ + * {<OpenLayers.Bounds>} │ │ │ │ │ */ │ │ │ │ │ - importSymbol: function(graphicName) { │ │ │ │ │ - var id = this.container.id + "-" + graphicName; │ │ │ │ │ + getMaxExtent: function() { │ │ │ │ │ + var resolution = this.map.getResolution(); │ │ │ │ │ + return this.maxExtent = this.getMaxExtentForResolution(resolution); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // check if symbol already exists in the cache │ │ │ │ │ - var cache = this.symbolCache[id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - return cache; │ │ │ │ │ + /** │ │ │ │ │ + * Method: getTileOrigin │ │ │ │ │ + * Determine the origin for aligning the grid of tiles. │ │ │ │ │ + * The origin will be derived from the layer's <maxExtent> property. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.LonLat>} The tile origin. │ │ │ │ │ + */ │ │ │ │ │ + getTileOrigin: function() { │ │ │ │ │ + if (!this._tileOrigin) { │ │ │ │ │ + var extent = this.getMaxExtent(); │ │ │ │ │ + this._tileOrigin = new OpenLayers.LonLat(extent.left, extent.bottom); │ │ │ │ │ } │ │ │ │ │ + return this._tileOrigin; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(graphicName + ' is not a valid symbol name'); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * Determine the URL for a tile given the tile bounds. This is should support │ │ │ │ │ + * urls that access tiles through an ArcGIS Server MapServer or directly through │ │ │ │ │ + * the hex folder structure using HTTP. Just be sure to set the useArcGISServer │ │ │ │ │ + * property appropriately! This is basically the same as │ │ │ │ │ + * 'OpenLayers.Layer.TMS.getURL', but with the addition of hex addressing, │ │ │ │ │ + * and tile rounding. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The URL for a tile based on given bounds. │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var res = this.getResolution(); │ │ │ │ │ │ │ │ │ │ - var symbolExtent = new OpenLayers.Bounds( │ │ │ │ │ - Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ + // tile center │ │ │ │ │ + var originTileX = (this.tileOrigin.lon + (res * this.tileSize.w / 2)); │ │ │ │ │ + var originTileY = (this.tileOrigin.lat - (res * this.tileSize.h / 2)); │ │ │ │ │ │ │ │ │ │ - var pathitems = ["m"]; │ │ │ │ │ - for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - var x = symbol[i]; │ │ │ │ │ - var y = symbol[i + 1]; │ │ │ │ │ - symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ - symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ - symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ - symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var point = { │ │ │ │ │ + x: center.lon, │ │ │ │ │ + y: center.lat │ │ │ │ │ + }; │ │ │ │ │ + var x = (Math.round(Math.abs((center.lon - originTileX) / (res * this.tileSize.w)))); │ │ │ │ │ + var y = (Math.round(Math.abs((originTileY - center.lat) / (res * this.tileSize.h)))); │ │ │ │ │ + var z = this.map.getZoom(); │ │ │ │ │ │ │ │ │ │ - pathitems.push(x); │ │ │ │ │ - pathitems.push(y); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - pathitems.push("l"); │ │ │ │ │ + // this prevents us from getting pink tiles (non-existant tiles) │ │ │ │ │ + if (this.lods) { │ │ │ │ │ + var lod = this.lods[this.map.getZoom()]; │ │ │ │ │ + if ((x < lod.startTileCol || x > lod.endTileCol) || │ │ │ │ │ + (y < lod.startTileRow || y > lod.endTileRow)) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + var start = this.getUpperLeftTileCoord(res); │ │ │ │ │ + var end = this.getLowerRightTileCoord(res); │ │ │ │ │ + if ((x < start.x || x >= end.x) || │ │ │ │ │ + (y < start.y || y >= end.y)) { │ │ │ │ │ + return null; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - pathitems.push("x e"); │ │ │ │ │ - var path = pathitems.join(" "); │ │ │ │ │ │ │ │ │ │ - var diff = (symbolExtent.getWidth() - symbolExtent.getHeight()) / 2; │ │ │ │ │ - if (diff > 0) { │ │ │ │ │ - symbolExtent.bottom = symbolExtent.bottom - diff; │ │ │ │ │ - symbolExtent.top = symbolExtent.top + diff; │ │ │ │ │ + // Construct the url string │ │ │ │ │ + var url = this.url; │ │ │ │ │ + var s = '' + x + y + z; │ │ │ │ │ + │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(s, url); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Accessing tiles through ArcGIS Server uses a different path │ │ │ │ │ + // structure than direct access via the folder structure. │ │ │ │ │ + if (this.useArcGISServer) { │ │ │ │ │ + // AGS MapServers have pretty url access to tiles │ │ │ │ │ + url = url + '/tile/${z}/${y}/${x}'; │ │ │ │ │ } else { │ │ │ │ │ - symbolExtent.left = symbolExtent.left + diff; │ │ │ │ │ - symbolExtent.right = symbolExtent.right - diff; │ │ │ │ │ + // The tile images are stored using hex values on disk. │ │ │ │ │ + x = 'C' + OpenLayers.Number.zeroPad(x, 8, 16); │ │ │ │ │ + y = 'R' + OpenLayers.Number.zeroPad(y, 8, 16); │ │ │ │ │ + z = 'L' + OpenLayers.Number.zeroPad(z, 2, 10); │ │ │ │ │ + url = url + '/${z}/${y}/${x}.' + this.type; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - cache = { │ │ │ │ │ - path: path, │ │ │ │ │ - size: symbolExtent.getWidth(), // equals getHeight() now │ │ │ │ │ - left: symbolExtent.left, │ │ │ │ │ - bottom: symbolExtent.bottom │ │ │ │ │ - }; │ │ │ │ │ - this.symbolCache[id] = cache; │ │ │ │ │ + // Write the values into our formatted url │ │ │ │ │ + url = OpenLayers.String.format(url, { │ │ │ │ │ + 'x': x, │ │ │ │ │ + 'y': y, │ │ │ │ │ + 'z': z │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - return cache; │ │ │ │ │ + return OpenLayers.Util.urlAppend( │ │ │ │ │ + url, OpenLayers.Util.getParameterString(this.params) │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.VML" │ │ │ │ │ + CLASS_NAME: 'OpenLayers.Layer.ArcGISCache' │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.VML.LABEL_SHIFT │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.VML.LABEL_SHIFT = { │ │ │ │ │ - "l": 0, │ │ │ │ │ - "c": .5, │ │ │ │ │ - "r": 1, │ │ │ │ │ - "t": 0, │ │ │ │ │ - "m": .5, │ │ │ │ │ - "b": 1 │ │ │ │ │ -}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Renderer/SVG.js │ │ │ │ │ + OpenLayers/Layer/ArcIMS.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Renderer/Elements.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/Format/ArcXML.js │ │ │ │ │ + * @requires OpenLayers/Request.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Renderer.SVG │ │ │ │ │ + * Class: OpenLayers.Layer.ArcIMS │ │ │ │ │ + * Instances of OpenLayers.Layer.ArcIMS are used to display data from ESRI ArcIMS │ │ │ │ │ + * Mapping Services. Create a new ArcIMS layer with the <OpenLayers.Layer.ArcIMS> │ │ │ │ │ + * constructor. │ │ │ │ │ * │ │ │ │ │ - * Inherits: │ │ │ │ │ - * - <OpenLayers.Renderer.Elements> │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Renderer.SVG = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ +OpenLayers.Layer.ArcIMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: xmlns │ │ │ │ │ - * {String} │ │ │ │ │ + /** │ │ │ │ │ + * Constant: DEFAULT_PARAMS │ │ │ │ │ + * {Object} Default query string parameters. │ │ │ │ │ */ │ │ │ │ │ - xmlns: "http://www.w3.org/2000/svg", │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + ClientVersion: "9.2", │ │ │ │ │ + ServiceName: '' │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: xlinkns │ │ │ │ │ - * {String} │ │ │ │ │ + * APIProperty: featureCoordSys │ │ │ │ │ + * {String} Code for feature coordinate system. Default is "4326". │ │ │ │ │ */ │ │ │ │ │ - xlinkns: "http://www.w3.org/1999/xlink", │ │ │ │ │ + featureCoordSys: "4326", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: MAX_PIXEL │ │ │ │ │ - * {Integer} Firefox has a limitation where values larger or smaller than │ │ │ │ │ - * about 15000 in an SVG document lock the browser up. This │ │ │ │ │ - * works around it. │ │ │ │ │ + * APIProperty: filterCoordSys │ │ │ │ │ + * {String} Code for filter coordinate system. Default is "4326". │ │ │ │ │ */ │ │ │ │ │ - MAX_PIXEL: 15000, │ │ │ │ │ + filterCoordSys: "4326", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: translationParameters │ │ │ │ │ - * {Object} Hash with "x" and "y" properties │ │ │ │ │ + * APIProperty: layers │ │ │ │ │ + * {Array} An array of objects with layer properties. │ │ │ │ │ */ │ │ │ │ │ - translationParameters: null, │ │ │ │ │ + layers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: symbolMetrics │ │ │ │ │ - * {Object} Cache for symbol metrics according to their svg coordinate │ │ │ │ │ - * space. This is an object keyed by the symbol's id, and values are │ │ │ │ │ - * an array of [width, centerX, centerY]. │ │ │ │ │ + * APIProperty: async │ │ │ │ │ + * {Boolean} Request images asynchronously. Default is true. │ │ │ │ │ */ │ │ │ │ │ - symbolMetrics: null, │ │ │ │ │ + async: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Renderer.SVG │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * containerID - {String} │ │ │ │ │ + * APIProperty: name │ │ │ │ │ + * {String} Layer name. Default is "ArcIMS". │ │ │ │ │ */ │ │ │ │ │ - initialize: function(containerID) { │ │ │ │ │ - if (!this.supported()) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.initialize.apply(this, │ │ │ │ │ - arguments); │ │ │ │ │ - this.translationParameters = { │ │ │ │ │ - x: 0, │ │ │ │ │ - y: 0 │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - this.symbolMetrics = {}; │ │ │ │ │ - }, │ │ │ │ │ + name: "ArcIMS", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: supported │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the browser supports the SVG renderer │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} The layer is a base layer. Default is true. │ │ │ │ │ */ │ │ │ │ │ - supported: function() { │ │ │ │ │ - var svgFeature = "http://www.w3.org/TR/SVG11/feature#"; │ │ │ │ │ - return (document.implementation && │ │ │ │ │ - (document.implementation.hasFeature("org.w3c.svg", "1.0") || │ │ │ │ │ - document.implementation.hasFeature(svgFeature + "SVG", "1.1") || │ │ │ │ │ - document.implementation.hasFeature(svgFeature + "BasicStructure", "1.1"))); │ │ │ │ │ - }, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: inValidRange │ │ │ │ │ - * See #669 for more information │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {Integer} │ │ │ │ │ - * y - {Integer} │ │ │ │ │ - * xyOnly - {Boolean} whether or not to just check for x and y, which means │ │ │ │ │ - * to not take the current translation parameters into account if true. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the 'x' and 'y' coordinates are in the │ │ │ │ │ - * valid range. │ │ │ │ │ + * Constant: DEFAULT_OPTIONS │ │ │ │ │ + * {Object} Default layers properties. │ │ │ │ │ */ │ │ │ │ │ - inValidRange: function(x, y, xyOnly) { │ │ │ │ │ - var left = x + (xyOnly ? 0 : this.translationParameters.x); │ │ │ │ │ - var top = y + (xyOnly ? 0 : this.translationParameters.y); │ │ │ │ │ - return (left >= -this.MAX_PIXEL && left <= this.MAX_PIXEL && │ │ │ │ │ - top >= -this.MAX_PIXEL && top <= this.MAX_PIXEL); │ │ │ │ │ + DEFAULT_OPTIONS: { │ │ │ │ │ + tileSize: new OpenLayers.Size(512, 512), │ │ │ │ │ + featureCoordSys: "4326", │ │ │ │ │ + filterCoordSys: "4326", │ │ │ │ │ + layers: null, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + async: true, │ │ │ │ │ + name: "ArcIMS" │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setExtent │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Layer.ArcIMS │ │ │ │ │ + * Create a new ArcIMS layer object. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var arcims = new OpenLayers.Layer.ArcIMS( │ │ │ │ │ + * "Global Sample", │ │ │ │ │ + * "http://sample.avencia.com/servlet/com.esri.esrimap.Esrimap", │ │ │ │ │ + * { │ │ │ │ │ + * service: "OpenLayers_Sample", │ │ │ │ │ + * layers: [ │ │ │ │ │ + * // layers to manipulate │ │ │ │ │ + * {id: "1", visible: true} │ │ │ │ │ + * ] │ │ │ │ │ + * } │ │ │ │ │ + * ); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} │ │ │ │ │ - * resolutionChanged - {Boolean} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ - * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ - * False otherwise. │ │ │ │ │ + * name - {String} A name for the layer │ │ │ │ │ + * url - {String} Base url for the ArcIMS server │ │ │ │ │ + * options - {Object} Optional object with properties to be set on the │ │ │ │ │ + * layer. │ │ │ │ │ */ │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ │ │ │ │ │ - var resolution = this.getResolution(), │ │ │ │ │ - left = -extent.left / resolution, │ │ │ │ │ - top = extent.top / resolution; │ │ │ │ │ + this.tileSize = new OpenLayers.Size(512, 512); │ │ │ │ │ │ │ │ │ │ - // If the resolution has changed, start over changing the corner, because │ │ │ │ │ - // the features will redraw. │ │ │ │ │ - if (resolutionChanged) { │ │ │ │ │ - this.left = left; │ │ │ │ │ - this.top = top; │ │ │ │ │ - // Set the viewbox │ │ │ │ │ - var extentString = "0 0 " + this.size.w + " " + this.size.h; │ │ │ │ │ + // parameters │ │ │ │ │ + this.params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + ServiceName: options.serviceName │ │ │ │ │ + }, │ │ │ │ │ + this.DEFAULT_PARAMS │ │ │ │ │ + ); │ │ │ │ │ + this.options = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options, this.DEFAULT_OPTIONS │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "viewBox", extentString); │ │ │ │ │ - this.translate(this.xOffset, 0); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - var inRange = this.translate(left - this.left + this.xOffset, top - this.top); │ │ │ │ │ - if (!inRange) { │ │ │ │ │ - // recenter the coordinate system │ │ │ │ │ - this.setExtent(extent, true); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply( │ │ │ │ │ + this, [name, url, this.params, options] │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + //layer is transparent │ │ │ │ │ + if (this.transparent) { │ │ │ │ │ + │ │ │ │ │ + // unless explicitly set in options, make layer an overlay │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + this.isBaseLayer = false; │ │ │ │ │ } │ │ │ │ │ - return coordSysUnchanged && inRange; │ │ │ │ │ + │ │ │ │ │ + // jpegs can never be transparent, so intelligently switch the │ │ │ │ │ + // format, depending on the browser's capabilities │ │ │ │ │ + if (this.format == "image/jpeg") { │ │ │ │ │ + this.format = OpenLayers.Util.alphaHack() ? "image/gif" : "image/png"; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // create an empty layer list if no layers specified in the options │ │ │ │ │ + if (this.options.layers === null) { │ │ │ │ │ + this.options.layers = []; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: translate │ │ │ │ │ - * Transforms the SVG coordinate system │ │ │ │ │ - * │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * Return an image url this layer. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * x - {Float} │ │ │ │ │ - * y - {Float} │ │ │ │ │ - * │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox for the │ │ │ │ │ + * request. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} true if the translation parameters are in the valid coordinates │ │ │ │ │ - * range, false otherwise. │ │ │ │ │ + * {String} A string with the map image's url. │ │ │ │ │ */ │ │ │ │ │ - translate: function(x, y) { │ │ │ │ │ - if (!this.inValidRange(x, y, true)) { │ │ │ │ │ - return false; │ │ │ │ │ - } else { │ │ │ │ │ - var transformString = ""; │ │ │ │ │ - if (x || y) { │ │ │ │ │ - transformString = "translate(" + x + "," + y + ")"; │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var url = ""; │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + │ │ │ │ │ + // create an arcxml request to generate the image │ │ │ │ │ + var axlReq = new OpenLayers.Format.ArcXML( │ │ │ │ │ + OpenLayers.Util.extend(this.options, { │ │ │ │ │ + requesttype: "image", │ │ │ │ │ + envelope: bounds.toArray(), │ │ │ │ │ + tileSize: this.tileSize │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + // create a synchronous ajax request to get an arcims image │ │ │ │ │ + var req = new OpenLayers.Request.POST({ │ │ │ │ │ + url: this.getFullRequestString(), │ │ │ │ │ + data: axlReq.write(), │ │ │ │ │ + async: false │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + // if the response exists │ │ │ │ │ + if (req != null) { │ │ │ │ │ + var doc = req.responseXML; │ │ │ │ │ + │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = req.responseText; │ │ │ │ │ } │ │ │ │ │ - this.root.setAttributeNS(null, "transform", transformString); │ │ │ │ │ - this.translationParameters = { │ │ │ │ │ - x: x, │ │ │ │ │ - y: y │ │ │ │ │ - }; │ │ │ │ │ - return true; │ │ │ │ │ + │ │ │ │ │ + // create a new arcxml format to read the response │ │ │ │ │ + var axlResp = new OpenLayers.Format.ArcXML(); │ │ │ │ │ + var arcxml = axlResp.read(doc); │ │ │ │ │ + url = this.getUrlOrImage(arcxml.image.output); │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + return url; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: setSize │ │ │ │ │ - * Sets the size of the drawing surface. │ │ │ │ │ - * │ │ │ │ │ + * Method: getURLasync │ │ │ │ │ + * Get an image url this layer asynchronously, and execute a callback │ │ │ │ │ + * when the image url is generated. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>} The size of the drawing surface │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox for the │ │ │ │ │ + * request. │ │ │ │ │ + * callback - {Function} Function to call when image url is retrieved. │ │ │ │ │ + * scope - {Object} The scope of the callback method. │ │ │ │ │ */ │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ + getURLasync: function(bounds, callback, scope) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "width", this.size.w); │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "height", this.size.h); │ │ │ │ │ + // create an arcxml request to generate the image │ │ │ │ │ + var axlReq = new OpenLayers.Format.ArcXML( │ │ │ │ │ + OpenLayers.Util.extend(this.options, { │ │ │ │ │ + requesttype: "image", │ │ │ │ │ + envelope: bounds.toArray(), │ │ │ │ │ + tileSize: this.tileSize │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + // create an asynchronous ajax request to get an arcims image │ │ │ │ │ + OpenLayers.Request.POST({ │ │ │ │ │ + url: this.getFullRequestString(), │ │ │ │ │ + async: true, │ │ │ │ │ + data: axlReq.write(), │ │ │ │ │ + callback: function(req) { │ │ │ │ │ + // process the response from ArcIMS, and call the callback function │ │ │ │ │ + // to set the image URL │ │ │ │ │ + var doc = req.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = req.responseText; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // create a new arcxml format to read the response │ │ │ │ │ + var axlResp = new OpenLayers.Format.ArcXML(); │ │ │ │ │ + var arcxml = axlResp.read(doc); │ │ │ │ │ + │ │ │ │ │ + callback.call(scope, this.getUrlOrImage(arcxml.image.output)); │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getNodeType │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: getUrlOrImage │ │ │ │ │ + * Extract a url or image from the ArcXML image output. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * │ │ │ │ │ + * output - {Object} The image.output property of the object returned from │ │ │ │ │ + * the ArcXML format read method. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} The corresponding node type for the specified geometry │ │ │ │ │ + * {String} A URL for an image (potentially with the data protocol). │ │ │ │ │ */ │ │ │ │ │ - getNodeType: function(geometry, style) { │ │ │ │ │ - var nodeType = null; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - nodeType = "image"; │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - nodeType = "svg"; │ │ │ │ │ - } else { │ │ │ │ │ - nodeType = "circle"; │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - nodeType = "rect"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - nodeType = "polyline"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - nodeType = "polygon"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - case "OpenLayers.Geometry.Curve": │ │ │ │ │ - nodeType = "path"; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break; │ │ │ │ │ + getUrlOrImage: function(output) { │ │ │ │ │ + var ret = ""; │ │ │ │ │ + if (output.url) { │ │ │ │ │ + // If the image response output url is a string, then the image │ │ │ │ │ + // data is not inline. │ │ │ │ │ + ret = output.url; │ │ │ │ │ + } else if (output.data) { │ │ │ │ │ + // The image data is inline and base64 encoded, create a data │ │ │ │ │ + // url for the image. This will only work for small images, │ │ │ │ │ + // due to browser url length limits. │ │ │ │ │ + ret = "data:image/" + output.type + │ │ │ │ │ + ";base64," + output.data; │ │ │ │ │ } │ │ │ │ │ - return nodeType; │ │ │ │ │ + return ret; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setStyle │ │ │ │ │ - * Use to set all the style attributes to a SVG node. │ │ │ │ │ - * │ │ │ │ │ - * Takes care to adjust stroke width and point radius to be │ │ │ │ │ - * resolution-relative │ │ │ │ │ + /** │ │ │ │ │ + * Method: setLayerQuery │ │ │ │ │ + * Set the query definition on this layer. Query definitions are used to │ │ │ │ │ + * render parts of the spatial data in an image, and can be used to │ │ │ │ │ + * filter features or layers in the ArcIMS service. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {SVGDomElement} An SVG element to decorate │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * options - {Object} Currently supported options include │ │ │ │ │ - * 'isFilled' {Boolean} and │ │ │ │ │ - * 'isStroked' {Boolean} │ │ │ │ │ + * id - {String} The ArcIMS layer ID. │ │ │ │ │ + * querydef - {Object} The query definition to apply to this layer. │ │ │ │ │ */ │ │ │ │ │ - setStyle: function(node, style, options) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - options = options || node._options; │ │ │ │ │ - │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - node.setAttributeNS(null, "title", title); │ │ │ │ │ - //Standards-conformant SVG │ │ │ │ │ - // Prevent duplicate nodes. See issue https://github.com/openlayers/openlayers/issues/92 │ │ │ │ │ - var titleNode = node.getElementsByTagName("title"); │ │ │ │ │ - if (titleNode.length > 0) { │ │ │ │ │ - titleNode[0].firstChild.textContent = title; │ │ │ │ │ - } else { │ │ │ │ │ - var label = this.nodeFactory(null, "title"); │ │ │ │ │ - label.textContent = title; │ │ │ │ │ - node.appendChild(label); │ │ │ │ │ + setLayerQuery: function(id, querydef) { │ │ │ │ │ + // find the matching layer, if it exists │ │ │ │ │ + for (var lyr = 0; lyr < this.options.layers.length; lyr++) { │ │ │ │ │ + if (id == this.options.layers[lyr].id) { │ │ │ │ │ + // replace this layer definition │ │ │ │ │ + this.options.layers[lyr].query = querydef; │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - var r = parseFloat(node.getAttributeNS(null, "r")); │ │ │ │ │ - var widthFactor = 1; │ │ │ │ │ - var pos; │ │ │ │ │ - if (node._geometryClass == "OpenLayers.Geometry.Point" && r) { │ │ │ │ │ - node.style.visibility = ""; │ │ │ │ │ - if (style.graphic === false) { │ │ │ │ │ - node.style.visibility = "hidden"; │ │ │ │ │ - } else if (style.externalGraphic) { │ │ │ │ │ - pos = this.getPosition(node); │ │ │ │ │ - if (style.graphicWidth && style.graphicHeight) { │ │ │ │ │ - node.setAttributeNS(null, "preserveAspectRatio", "none"); │ │ │ │ │ - } │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ - style.graphicXOffset : -(0.5 * width); │ │ │ │ │ - var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ - style.graphicYOffset : -(0.5 * height); │ │ │ │ │ + // no layer found, create a new definition │ │ │ │ │ + this.options.layers.push({ │ │ │ │ │ + id: id, │ │ │ │ │ + visible: true, │ │ │ │ │ + query: querydef │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ + /** │ │ │ │ │ + * Method: getFeatureInfo │ │ │ │ │ + * Get feature information from ArcIMS. Using the applied geometry, apply │ │ │ │ │ + * the options to the query (buffer, area/envelope intersection), and │ │ │ │ │ + * query the ArcIMS service. │ │ │ │ │ + * │ │ │ │ │ + * A note about accuracy: │ │ │ │ │ + * ArcIMS interprets the accuracy attribute in feature requests to be │ │ │ │ │ + * something like the 'modulus' operator on feature coordinates, │ │ │ │ │ + * applied to the database geometry of the feature. It doesn't round, │ │ │ │ │ + * so your feature coordinates may be up to (1 x accuracy) offset from │ │ │ │ │ + * the actual feature coordinates. If the accuracy of the layer is not │ │ │ │ │ + * specified, the accuracy will be computed to be approximately 1 │ │ │ │ │ + * feature coordinate per screen pixel. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.LonLat>} or {<OpenLayers.Geometry.Polygon>} The │ │ │ │ │ + * geometry to use when making the query. This should be a closed │ │ │ │ │ + * polygon for behavior approximating a free selection. │ │ │ │ │ + * layer - {Object} The ArcIMS layer definition. This is an anonymous object │ │ │ │ │ + * that looks like: │ │ │ │ │ + * (code) │ │ │ │ │ + * { │ │ │ │ │ + * id: "ArcXML layer ID", // the ArcXML layer ID │ │ │ │ │ + * query: { │ │ │ │ │ + * where: "STATE = 'PA'", // the where clause of the query │ │ │ │ │ + * accuracy: 100 // the accuracy of the returned feature │ │ │ │ │ + * } │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ + * options - {Object} Object with non-default properties to set on the layer. │ │ │ │ │ + * Supported properties are buffer, callback, scope, and any other │ │ │ │ │ + * properties applicable to the ArcXML format. Set the 'callback' and │ │ │ │ │ + * 'scope' for an object and function to recieve the parsed features │ │ │ │ │ + * from ArcIMS. │ │ │ │ │ + */ │ │ │ │ │ + getFeatureInfo: function(geometry, layer, options) { │ │ │ │ │ + // set the buffer to 1 unit (dd/m/ft?) by default │ │ │ │ │ + var buffer = options.buffer || 1; │ │ │ │ │ + // empty callback by default │ │ │ │ │ + var callback = options.callback || function() {}; │ │ │ │ │ + // default scope is window (global) │ │ │ │ │ + var scope = options.scope || window; │ │ │ │ │ │ │ │ │ │ - node.setAttributeNS(null, "x", (pos.x + xOffset).toFixed()); │ │ │ │ │ - node.setAttributeNS(null, "y", (pos.y + yOffset).toFixed()); │ │ │ │ │ - node.setAttributeNS(null, "width", width); │ │ │ │ │ - node.setAttributeNS(null, "height", height); │ │ │ │ │ - node.setAttributeNS(this.xlinkns, "xlink:href", style.externalGraphic); │ │ │ │ │ - node.setAttributeNS(null, "style", "opacity: " + opacity); │ │ │ │ │ - node.onclick = OpenLayers.Event.preventDefault; │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - // the symbol viewBox is three times as large as the symbol │ │ │ │ │ - var offset = style.pointRadius * 3; │ │ │ │ │ - var size = offset * 2; │ │ │ │ │ - var src = this.importSymbol(style.graphicName); │ │ │ │ │ - pos = this.getPosition(node); │ │ │ │ │ - widthFactor = this.symbolMetrics[src.id][0] * 3 / size; │ │ │ │ │ + // apply these option to the request options │ │ │ │ │ + var requestOptions = {}; │ │ │ │ │ + OpenLayers.Util.extend(requestOptions, this.options); │ │ │ │ │ │ │ │ │ │ - // remove the node from the dom before we modify it. This │ │ │ │ │ - // prevents various rendering issues in Safari and FF │ │ │ │ │ - var parent = node.parentNode; │ │ │ │ │ - var nextSibling = node.nextSibling; │ │ │ │ │ - if (parent) { │ │ │ │ │ - parent.removeChild(node); │ │ │ │ │ - } │ │ │ │ │ + // this is a feature request │ │ │ │ │ + requestOptions.requesttype = "feature"; │ │ │ │ │ │ │ │ │ │ - // The more appropriate way to implement this would be use/defs, │ │ │ │ │ - // but due to various issues in several browsers, it is safer to │ │ │ │ │ - // copy the symbols instead of referencing them. │ │ │ │ │ - // See e.g. ticket http://trac.osgeo.org/openlayers/ticket/2985 │ │ │ │ │ - // and this email thread │ │ │ │ │ - // http://osgeo-org.1803224.n2.nabble.com/Select-Control-Ctrl-click-on-Feature-with-a-graphicName-opens-new-browser-window-tc5846039.html │ │ │ │ │ - node.firstChild && node.removeChild(node.firstChild); │ │ │ │ │ - node.appendChild(src.firstChild.cloneNode(true)); │ │ │ │ │ - node.setAttributeNS(null, "viewBox", src.getAttributeNS(null, "viewBox")); │ │ │ │ │ + if (geometry instanceof OpenLayers.LonLat) { │ │ │ │ │ + // create an envelope if the geometry is really a lon/lat │ │ │ │ │ + requestOptions.polygon = null; │ │ │ │ │ + requestOptions.envelope = [ │ │ │ │ │ + geometry.lon - buffer, │ │ │ │ │ + geometry.lat - buffer, │ │ │ │ │ + geometry.lon + buffer, │ │ │ │ │ + geometry.lat + buffer │ │ │ │ │ + ]; │ │ │ │ │ + } else if (geometry instanceof OpenLayers.Geometry.Polygon) { │ │ │ │ │ + // use the polygon assigned, and empty the envelope │ │ │ │ │ + requestOptions.envelope = null; │ │ │ │ │ + requestOptions.polygon = geometry; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - node.setAttributeNS(null, "width", size); │ │ │ │ │ - node.setAttributeNS(null, "height", size); │ │ │ │ │ - node.setAttributeNS(null, "x", pos.x - offset); │ │ │ │ │ - node.setAttributeNS(null, "y", pos.y - offset); │ │ │ │ │ + // create an arcxml request to get feature requests │ │ │ │ │ + var arcxml = new OpenLayers.Format.ArcXML(requestOptions); │ │ │ │ │ │ │ │ │ │ - // now that the node has all its new properties, insert it │ │ │ │ │ - // back into the dom where it was │ │ │ │ │ - if (nextSibling) { │ │ │ │ │ - parent.insertBefore(node, nextSibling); │ │ │ │ │ - } else if (parent) { │ │ │ │ │ - parent.appendChild(node); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "r", style.pointRadius); │ │ │ │ │ - } │ │ │ │ │ + // apply any get feature options to the arcxml request │ │ │ │ │ + OpenLayers.Util.extend(arcxml.request.get_feature, options); │ │ │ │ │ │ │ │ │ │ - var rotation = style.rotation; │ │ │ │ │ + arcxml.request.get_feature.layer = layer.id; │ │ │ │ │ + if (typeof layer.query.accuracy == "number") { │ │ │ │ │ + // set the accuracy if it was specified │ │ │ │ │ + arcxml.request.get_feature.query.accuracy = layer.query.accuracy; │ │ │ │ │ + } else { │ │ │ │ │ + // guess that the accuracy is 1 per screen pixel │ │ │ │ │ + var mapCenter = this.map.getCenter(); │ │ │ │ │ + var viewPx = this.map.getViewPortPxFromLonLat(mapCenter); │ │ │ │ │ + viewPx.x++; │ │ │ │ │ + var mapOffCenter = this.map.getLonLatFromPixel(viewPx); │ │ │ │ │ + arcxml.request.get_feature.query.accuracy = mapOffCenter.lon - mapCenter.lon; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if ((rotation !== undefined || node._rotation !== undefined) && pos) { │ │ │ │ │ - node._rotation = rotation; │ │ │ │ │ - rotation |= 0; │ │ │ │ │ - if (node.nodeName !== "svg") { │ │ │ │ │ - node.setAttributeNS(null, "transform", │ │ │ │ │ - "rotate(" + rotation + " " + pos.x + " " + │ │ │ │ │ - pos.y + ")"); │ │ │ │ │ + // set the get_feature query to be the same as the layer passed in │ │ │ │ │ + arcxml.request.get_feature.query.where = layer.query.where; │ │ │ │ │ + │ │ │ │ │ + // use area_intersection │ │ │ │ │ + arcxml.request.get_feature.query.spatialfilter.relation = "area_intersection"; │ │ │ │ │ + │ │ │ │ │ + // create a new asynchronous request to get the feature info │ │ │ │ │ + OpenLayers.Request.POST({ │ │ │ │ │ + url: this.getFullRequestString({ │ │ │ │ │ + 'CustomService': 'Query' │ │ │ │ │ + }), │ │ │ │ │ + data: arcxml.write(), │ │ │ │ │ + callback: function(request) { │ │ │ │ │ + // parse the arcxml response │ │ │ │ │ + var response = arcxml.parseResponse(request.responseText); │ │ │ │ │ + │ │ │ │ │ + if (!arcxml.iserror()) { │ │ │ │ │ + // if the arcxml is not an error, call the callback with the features parsed │ │ │ │ │ + callback.call(scope, response.features); │ │ │ │ │ } else { │ │ │ │ │ - var metrics = this.symbolMetrics[src.id]; │ │ │ │ │ - node.firstChild.setAttributeNS(null, "transform", "rotate(" + │ │ │ │ │ - rotation + " " + │ │ │ │ │ - metrics[1] + " " + │ │ │ │ │ - metrics[2] + ")"); │ │ │ │ │ + // if the arcxml is an error, return null features selected │ │ │ │ │ + callback.call(scope, null); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (options.isFilled) { │ │ │ │ │ - node.setAttributeNS(null, "fill", style.fillColor); │ │ │ │ │ - node.setAttributeNS(null, "fill-opacity", style.fillOpacity); │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "fill", "none"); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.ArcIMS>} An exact clone of this layer │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ │ │ │ │ │ - if (options.isStroked) { │ │ │ │ │ - node.setAttributeNS(null, "stroke", style.strokeColor); │ │ │ │ │ - node.setAttributeNS(null, "stroke-opacity", style.strokeOpacity); │ │ │ │ │ - node.setAttributeNS(null, "stroke-width", style.strokeWidth * widthFactor); │ │ │ │ │ - node.setAttributeNS(null, "stroke-linecap", style.strokeLinecap || "round"); │ │ │ │ │ - // Hard-coded linejoin for now, to make it look the same as in VML. │ │ │ │ │ - // There is no strokeLinejoin property yet for symbolizers. │ │ │ │ │ - node.setAttributeNS(null, "stroke-linejoin", "round"); │ │ │ │ │ - style.strokeDashstyle && node.setAttributeNS(null, │ │ │ │ │ - "stroke-dasharray", this.dashStyle(style, widthFactor)); │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "stroke", "none"); │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.ArcIMS(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - if (style.pointerEvents) { │ │ │ │ │ - node.setAttributeNS(null, "pointer-events", style.pointerEvents); │ │ │ │ │ - } │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ │ │ │ │ │ - if (style.cursor != null) { │ │ │ │ │ - node.setAttributeNS(null, "cursor", style.cursor); │ │ │ │ │ - } │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ │ │ │ │ │ - return node; │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: dashStyle │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * widthFactor - {Number} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A SVG compliant 'stroke-dasharray' value │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.ArcIMS" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/PointGrid.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.PointGrid │ │ │ │ │ + * A point grid layer dynamically generates a regularly spaced grid of point │ │ │ │ │ + * features. This is a specialty layer for cases where an application needs │ │ │ │ │ + * a regular grid of points. It can be used, for example, in an editing │ │ │ │ │ + * environment to snap to a grid. │ │ │ │ │ + * │ │ │ │ │ + * Create a new vector layer with the <OpenLayers.Layer.PointGrid> constructor. │ │ │ │ │ + * (code) │ │ │ │ │ + * // create a grid with points spaced at 10 map units │ │ │ │ │ + * var points = new OpenLayers.Layer.PointGrid({dx: 10, dy: 10}); │ │ │ │ │ + * │ │ │ │ │ + * // create a grid with different x/y spacing rotated 15 degrees clockwise. │ │ │ │ │ + * var points = new OpenLayers.Layer.PointGrid({dx: 5, dy: 10, rotation: 15}); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Vector> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.PointGrid = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: dx │ │ │ │ │ + * {Number} Point grid spacing in the x-axis direction (map units). │ │ │ │ │ + * Read-only. Use the <setSpacing> method to modify this value. │ │ │ │ │ */ │ │ │ │ │ - dashStyle: function(style, widthFactor) { │ │ │ │ │ - var w = style.strokeWidth * widthFactor; │ │ │ │ │ - var str = style.strokeDashstyle; │ │ │ │ │ - switch (str) { │ │ │ │ │ - case 'solid': │ │ │ │ │ - return 'none'; │ │ │ │ │ - case 'dot': │ │ │ │ │ - return [1, 4 * w].join(); │ │ │ │ │ - case 'dash': │ │ │ │ │ - return [4 * w, 4 * w].join(); │ │ │ │ │ - case 'dashdot': │ │ │ │ │ - return [4 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ - case 'longdash': │ │ │ │ │ - return [8 * w, 4 * w].join(); │ │ │ │ │ - case 'longdashdot': │ │ │ │ │ - return [8 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ - default: │ │ │ │ │ - return OpenLayers.String.trim(str).replace(/\s+/g, ","); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + dx: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createNode │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: dy │ │ │ │ │ + * {Number} Point grid spacing in the y-axis direction (map units). │ │ │ │ │ + * Read-only. Use the <setSpacing> method to modify this value. │ │ │ │ │ + */ │ │ │ │ │ + dy: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: ratio │ │ │ │ │ + * {Number} Ratio of the desired grid size to the map viewport size. │ │ │ │ │ + * Default is 1.5. Larger ratios mean the grid is recalculated less often │ │ │ │ │ + * while panning. The <maxFeatures> setting has precedence when determining │ │ │ │ │ + * grid size. Read-only. Use the <setRatio> method to modify this value. │ │ │ │ │ + */ │ │ │ │ │ + ratio: 1.5, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxFeatures │ │ │ │ │ + * {Number} The maximum number of points to generate in the grid. Default │ │ │ │ │ + * is 250. Read-only. Use the <setMaxFeatures> method to modify this value. │ │ │ │ │ + */ │ │ │ │ │ + maxFeatures: 250, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: rotation │ │ │ │ │ + * {Number} Grid rotation (in degrees clockwise from the positive x-axis). │ │ │ │ │ + * Default is 0. Read-only. Use the <setRotation> method to modify this │ │ │ │ │ + * value. │ │ │ │ │ + */ │ │ │ │ │ + rotation: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: origin │ │ │ │ │ + * {<OpenLayers.LonLat>} Grid origin. The grid lattice will be aligned with │ │ │ │ │ + * the origin. If not set at construction, the center of the map's maximum │ │ │ │ │ + * extent is used. Read-only. Use the <setOrigin> method to modify this │ │ │ │ │ + * value. │ │ │ │ │ + */ │ │ │ │ │ + origin: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: gridBounds │ │ │ │ │ + * {<OpenLayers.Bounds>} Internally cached grid bounds (with optional │ │ │ │ │ + * rotation applied). │ │ │ │ │ + */ │ │ │ │ │ + gridBounds: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.PointGrid │ │ │ │ │ + * Creates a new point grid layer. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * type - {String} Kind of node to draw │ │ │ │ │ - * id - {String} Id for node │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A new node of the given type and id │ │ │ │ │ + * config - {Object} An object containing all configuration properties for │ │ │ │ │ + * the layer. The <dx> and <dy> properties are required to be set at │ │ │ │ │ + * construction. Any other layer properties may be set in this object. │ │ │ │ │ */ │ │ │ │ │ - createNode: function(type, id) { │ │ │ │ │ - var node = document.createElementNS(this.xmlns, type); │ │ │ │ │ - if (id) { │ │ │ │ │ - node.setAttributeNS(null, "id", id); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ + config = config || {}; │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.initialize.apply(this, [config.name, config]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: nodeTypeCompare │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * The layer has been added to the map. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {SVGDomElement} An SVG element │ │ │ │ │ - * type - {String} Kind of node │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - nodeTypeCompare: function(node, type) { │ │ │ │ │ - return (type == node.nodeName); │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ + map.events.register("moveend", this, this.onMoveEnd); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createRenderRoot │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The specific render engine's root element │ │ │ │ │ + * Method: removeMap │ │ │ │ │ + * The layer has been removed from the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - createRenderRoot: function() { │ │ │ │ │ - var svg = this.nodeFactory(this.container.id + "_svgRoot", "svg"); │ │ │ │ │ - svg.style.display = "block"; │ │ │ │ │ - return svg; │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + map.events.unregister("moveend", this, this.onMoveEnd); │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createRoot │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: setRatio │ │ │ │ │ + * Set the grid <ratio> property and update the grid. Can only be called │ │ │ │ │ + * after the layer has been added to a map with a center/extent. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * suffix - {String} suffix to append to the id │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * ratio - {Number} │ │ │ │ │ */ │ │ │ │ │ - createRoot: function(suffix) { │ │ │ │ │ - return this.nodeFactory(this.container.id + suffix, "g"); │ │ │ │ │ + setRatio: function(ratio) { │ │ │ │ │ + this.ratio = ratio; │ │ │ │ │ + this.updateGrid(true); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createDefs │ │ │ │ │ + * APIMethod: setMaxFeatures │ │ │ │ │ + * Set the grid <maxFeatures> property and update the grid. Can only be │ │ │ │ │ + * called after the layer has been added to a map with a center/extent. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The element to which we'll add the symbol definitions │ │ │ │ │ + * Parameters: │ │ │ │ │ + * maxFeatures - {Number} │ │ │ │ │ */ │ │ │ │ │ - createDefs: function() { │ │ │ │ │ - var defs = this.nodeFactory(this.container.id + "_defs", "defs"); │ │ │ │ │ - this.rendererRoot.appendChild(defs); │ │ │ │ │ - return defs; │ │ │ │ │ + setMaxFeatures: function(maxFeatures) { │ │ │ │ │ + this.maxFeatures = maxFeatures; │ │ │ │ │ + this.updateGrid(true); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /************************************** │ │ │ │ │ - * * │ │ │ │ │ - * GEOMETRY DRAWING FUNCTIONS * │ │ │ │ │ - * * │ │ │ │ │ - **************************************/ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPoint │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the point │ │ │ │ │ + * APIMethod: setSpacing │ │ │ │ │ + * Set the grid <dx> and <dy> properties and update the grid. If only one │ │ │ │ │ + * argument is provided, it will be set as <dx> and <dy>. Can only be │ │ │ │ │ + * called after the layer has been added to a map with a center/extent. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * dx - {Number} │ │ │ │ │ + * dy - {Number} │ │ │ │ │ */ │ │ │ │ │ - drawPoint: function(node, geometry) { │ │ │ │ │ - return this.drawCircle(node, geometry, 1); │ │ │ │ │ + setSpacing: function(dx, dy) { │ │ │ │ │ + this.dx = dx; │ │ │ │ │ + this.dy = dy || dx; │ │ │ │ │ + this.updateGrid(true); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawCircle │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * radius - {Float} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the circle │ │ │ │ │ + * APIMethod: setOrigin │ │ │ │ │ + * Set the grid <origin> property and update the grid. Can only be called │ │ │ │ │ + * after the layer has been added to a map with a center/extent. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * origin - {<OpenLayers.LonLat>} │ │ │ │ │ */ │ │ │ │ │ - drawCircle: function(node, geometry, radius) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = ((geometry.x - this.featureDx) / resolution + this.left); │ │ │ │ │ - var y = (this.top - geometry.y / resolution); │ │ │ │ │ - │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - node.setAttributeNS(null, "cx", x); │ │ │ │ │ - node.setAttributeNS(null, "cy", y); │ │ │ │ │ - node.setAttributeNS(null, "r", radius); │ │ │ │ │ - return node; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ + setOrigin: function(origin) { │ │ │ │ │ + this.origin = origin; │ │ │ │ │ + this.updateGrid(true); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLineString │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getOrigin │ │ │ │ │ + * Get the grid <origin> property. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components of │ │ │ │ │ - * the linestring, or false if nothing could be drawn │ │ │ │ │ + * {<OpenLayers.LonLat>} The grid origin. │ │ │ │ │ */ │ │ │ │ │ - drawLineString: function(node, geometry) { │ │ │ │ │ - var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ - if (componentsResult.path) { │ │ │ │ │ - node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ - return (componentsResult.complete ? node : null); │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ + getOrigin: function() { │ │ │ │ │ + if (!this.origin) { │ │ │ │ │ + this.origin = this.map.getExtent().getCenterLonLat(); │ │ │ │ │ } │ │ │ │ │ + return this.origin; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLinearRing │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ - * of the linear ring, or false if nothing could be drawn │ │ │ │ │ + * APIMethod: setRotation │ │ │ │ │ + * Set the grid <rotation> property and update the grid. Rotation values │ │ │ │ │ + * are in degrees clockwise from the positive x-axis (negative values │ │ │ │ │ + * for counter-clockwise rotation). Can only be called after the layer │ │ │ │ │ + * has been added to a map with a center/extent. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * rotation - {Number} Degrees clockwise from the positive x-axis. │ │ │ │ │ */ │ │ │ │ │ - drawLinearRing: function(node, geometry) { │ │ │ │ │ - var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ - if (componentsResult.path) { │ │ │ │ │ - node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ - return (componentsResult.complete ? node : null); │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ + setRotation: function(rotation) { │ │ │ │ │ + this.rotation = rotation; │ │ │ │ │ + this.updateGrid(true); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPolygon │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ - * of the polygon, or false if nothing could be drawn │ │ │ │ │ + * Method: onMoveEnd │ │ │ │ │ + * Listener for map "moveend" events. │ │ │ │ │ */ │ │ │ │ │ - drawPolygon: function(node, geometry) { │ │ │ │ │ - var d = ""; │ │ │ │ │ - var draw = true; │ │ │ │ │ - var complete = true; │ │ │ │ │ - var linearRingResult, path; │ │ │ │ │ - for (var j = 0, len = geometry.components.length; j < len; j++) { │ │ │ │ │ - d += " M"; │ │ │ │ │ - linearRingResult = this.getComponentsString( │ │ │ │ │ - geometry.components[j].components, " "); │ │ │ │ │ - path = linearRingResult.path; │ │ │ │ │ - if (path) { │ │ │ │ │ - d += " " + path; │ │ │ │ │ - complete = linearRingResult.complete && complete; │ │ │ │ │ - } else { │ │ │ │ │ - draw = false; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - d += " z"; │ │ │ │ │ - if (draw) { │ │ │ │ │ - node.setAttributeNS(null, "d", d); │ │ │ │ │ - node.setAttributeNS(null, "fill-rule", "evenodd"); │ │ │ │ │ - return complete ? node : null; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ + onMoveEnd: function() { │ │ │ │ │ + this.updateGrid(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawRectangle │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * Method: getViewBounds │ │ │ │ │ + * Gets the (potentially rotated) view bounds for grid calculations. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the rectangle │ │ │ │ │ + * {<OpenLayers.Bounds>} │ │ │ │ │ */ │ │ │ │ │ - drawRectangle: function(node, geometry) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = ((geometry.x - this.featureDx) / resolution + this.left); │ │ │ │ │ - var y = (this.top - geometry.y / resolution); │ │ │ │ │ - │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - node.setAttributeNS(null, "x", x); │ │ │ │ │ - node.setAttributeNS(null, "y", y); │ │ │ │ │ - node.setAttributeNS(null, "width", geometry.width / resolution); │ │ │ │ │ - node.setAttributeNS(null, "height", geometry.height / resolution); │ │ │ │ │ - return node; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ + getViewBounds: function() { │ │ │ │ │ + var bounds = this.map.getExtent(); │ │ │ │ │ + if (this.rotation) { │ │ │ │ │ + var origin = this.getOrigin(); │ │ │ │ │ + var rotationOrigin = new OpenLayers.Geometry.Point(origin.lon, origin.lat); │ │ │ │ │ + var rect = bounds.toGeometry(); │ │ │ │ │ + rect.rotate(-this.rotation, rotationOrigin); │ │ │ │ │ + bounds = rect.getBounds(); │ │ │ │ │ } │ │ │ │ │ + return bounds; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawText │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ + * Method: updateGrid │ │ │ │ │ + * Update the grid. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * style - │ │ │ │ │ - * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * force - {Boolean} Update the grid even if the previous bounds are still │ │ │ │ │ + * valid. │ │ │ │ │ */ │ │ │ │ │ - drawText: function(featureId, style, location) { │ │ │ │ │ - var drawOutline = (!!style.labelOutlineWidth); │ │ │ │ │ - // First draw text in halo color and size and overlay the │ │ │ │ │ - // normal text afterwards │ │ │ │ │ - if (drawOutline) { │ │ │ │ │ - var outlineStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ - outlineStyle.fontColor = outlineStyle.labelOutlineColor; │ │ │ │ │ - outlineStyle.fontStrokeColor = outlineStyle.labelOutlineColor; │ │ │ │ │ - outlineStyle.fontStrokeWidth = style.labelOutlineWidth; │ │ │ │ │ - if (style.labelOutlineOpacity) { │ │ │ │ │ - outlineStyle.fontOpacity = style.labelOutlineOpacity; │ │ │ │ │ - } │ │ │ │ │ - delete outlineStyle.labelOutlineWidth; │ │ │ │ │ - this.drawText(featureId, outlineStyle, location); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - │ │ │ │ │ - var x = ((location.x - this.featureDx) / resolution + this.left); │ │ │ │ │ - var y = (location.y / resolution - this.top); │ │ │ │ │ - │ │ │ │ │ - var suffix = (drawOutline) ? this.LABEL_OUTLINE_SUFFIX : this.LABEL_ID_SUFFIX; │ │ │ │ │ - var label = this.nodeFactory(featureId + suffix, "text"); │ │ │ │ │ - │ │ │ │ │ - label.setAttributeNS(null, "x", x); │ │ │ │ │ - label.setAttributeNS(null, "y", -y); │ │ │ │ │ - │ │ │ │ │ - if (style.fontColor) { │ │ │ │ │ - label.setAttributeNS(null, "fill", style.fontColor); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStrokeColor) { │ │ │ │ │ - label.setAttributeNS(null, "stroke", style.fontStrokeColor); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStrokeWidth) { │ │ │ │ │ - label.setAttributeNS(null, "stroke-width", style.fontStrokeWidth); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontOpacity) { │ │ │ │ │ - label.setAttributeNS(null, "opacity", style.fontOpacity); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontFamily) { │ │ │ │ │ - label.setAttributeNS(null, "font-family", style.fontFamily); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontSize) { │ │ │ │ │ - label.setAttributeNS(null, "font-size", style.fontSize); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontWeight) { │ │ │ │ │ - label.setAttributeNS(null, "font-weight", style.fontWeight); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStyle) { │ │ │ │ │ - label.setAttributeNS(null, "font-style", style.fontStyle); │ │ │ │ │ - } │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - label.setAttributeNS(null, "pointer-events", "visible"); │ │ │ │ │ - label._featureId = featureId; │ │ │ │ │ - } else { │ │ │ │ │ - label.setAttributeNS(null, "pointer-events", "none"); │ │ │ │ │ - } │ │ │ │ │ - var align = style.labelAlign || OpenLayers.Renderer.defaultSymbolizer.labelAlign; │ │ │ │ │ - label.setAttributeNS(null, "text-anchor", │ │ │ │ │ - OpenLayers.Renderer.SVG.LABEL_ALIGN[align[0]] || "middle"); │ │ │ │ │ - │ │ │ │ │ - if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ - label.setAttributeNS(null, "dominant-baseline", │ │ │ │ │ - OpenLayers.Renderer.SVG.LABEL_ALIGN[align[1]] || "central"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var labelRows = style.label.split('\n'); │ │ │ │ │ - var numRows = labelRows.length; │ │ │ │ │ - while (label.childNodes.length > numRows) { │ │ │ │ │ - label.removeChild(label.lastChild); │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - var tspan = this.nodeFactory(featureId + suffix + "_tspan_" + i, "tspan"); │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - tspan._featureId = featureId; │ │ │ │ │ - tspan._geometry = location; │ │ │ │ │ - tspan._geometryClass = location.CLASS_NAME; │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.IS_GECKO === false) { │ │ │ │ │ - tspan.setAttributeNS(null, "baseline-shift", │ │ │ │ │ - OpenLayers.Renderer.SVG.LABEL_VSHIFT[align[1]] || "-35%"); │ │ │ │ │ - } │ │ │ │ │ - tspan.setAttribute("x", x); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - var vfactor = OpenLayers.Renderer.SVG.LABEL_VFACTOR[align[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5; │ │ │ │ │ + updateGrid: function(force) { │ │ │ │ │ + if (force || this.invalidBounds()) { │ │ │ │ │ + var viewBounds = this.getViewBounds(); │ │ │ │ │ + var origin = this.getOrigin(); │ │ │ │ │ + var rotationOrigin = new OpenLayers.Geometry.Point(origin.lon, origin.lat); │ │ │ │ │ + var viewBoundsWidth = viewBounds.getWidth(); │ │ │ │ │ + var viewBoundsHeight = viewBounds.getHeight(); │ │ │ │ │ + var aspectRatio = viewBoundsWidth / viewBoundsHeight; │ │ │ │ │ + var maxHeight = Math.sqrt(this.dx * this.dy * this.maxFeatures / aspectRatio); │ │ │ │ │ + var maxWidth = maxHeight * aspectRatio; │ │ │ │ │ + var gridWidth = Math.min(viewBoundsWidth * this.ratio, maxWidth); │ │ │ │ │ + var gridHeight = Math.min(viewBoundsHeight * this.ratio, maxHeight); │ │ │ │ │ + var center = viewBounds.getCenterLonLat(); │ │ │ │ │ + this.gridBounds = new OpenLayers.Bounds( │ │ │ │ │ + center.lon - (gridWidth / 2), │ │ │ │ │ + center.lat - (gridHeight / 2), │ │ │ │ │ + center.lon + (gridWidth / 2), │ │ │ │ │ + center.lat + (gridHeight / 2) │ │ │ │ │ + ); │ │ │ │ │ + var rows = Math.floor(gridHeight / this.dy); │ │ │ │ │ + var cols = Math.floor(gridWidth / this.dx); │ │ │ │ │ + var gridLeft = origin.lon + (this.dx * Math.ceil((this.gridBounds.left - origin.lon) / this.dx)); │ │ │ │ │ + var gridBottom = origin.lat + (this.dy * Math.ceil((this.gridBounds.bottom - origin.lat) / this.dy)); │ │ │ │ │ + var features = new Array(rows * cols); │ │ │ │ │ + var x, y, point; │ │ │ │ │ + for (var i = 0; i < cols; ++i) { │ │ │ │ │ + x = gridLeft + (i * this.dx); │ │ │ │ │ + for (var j = 0; j < rows; ++j) { │ │ │ │ │ + y = gridBottom + (j * this.dy); │ │ │ │ │ + point = new OpenLayers.Geometry.Point(x, y); │ │ │ │ │ + if (this.rotation) { │ │ │ │ │ + point.rotate(this.rotation, rotationOrigin); │ │ │ │ │ + } │ │ │ │ │ + features[(i * rows) + j] = new OpenLayers.Feature.Vector(point); │ │ │ │ │ } │ │ │ │ │ - tspan.setAttribute("dy", (vfactor * (numRows - 1)) + "em"); │ │ │ │ │ - } else { │ │ │ │ │ - tspan.setAttribute("dy", "1em"); │ │ │ │ │ - } │ │ │ │ │ - tspan.textContent = (labelRows[i] === '') ? ' ' : labelRows[i]; │ │ │ │ │ - if (!tspan.parentNode) { │ │ │ │ │ - label.appendChild(tspan); │ │ │ │ │ } │ │ │ │ │ + this.destroyFeatures(this.features, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.addFeatures(features, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (!label.parentNode) { │ │ │ │ │ - this.textRoot.appendChild(label); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: invalidBounds │ │ │ │ │ + * Determine whether the previously generated point grid is invalid. │ │ │ │ │ + * This occurs when the map bounds extends beyond the previously │ │ │ │ │ + * generated grid bounds. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + invalidBounds: function() { │ │ │ │ │ + return !this.gridBounds || !this.gridBounds.containsBounds(this.getViewBounds()); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.PointGrid" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/TileCache.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.TileCache │ │ │ │ │ + * A read only TileCache layer. Used to requests tiles cached by TileCache in │ │ │ │ │ + * a web accessible cache. This means that you have to pre-populate your │ │ │ │ │ + * cache before this layer can be used. It is meant only to read tiles │ │ │ │ │ + * created by TileCache, and not to make calls to TileCache for tile │ │ │ │ │ + * creation. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Layer.TileCache> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.TileCache = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: getComponentString │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} Treat this layer as a base layer. Default is true. │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: format │ │ │ │ │ + * {String} Mime type of the images returned. Default is image/png. │ │ │ │ │ + */ │ │ │ │ │ + format: 'image/png', │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: serverResolutions │ │ │ │ │ + * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ + * property if the map resolutions differ from the server. This │ │ │ │ │ + * property serves two purposes. (a) <serverResolutions> can include │ │ │ │ │ + * resolutions that the server supports and that you don't want to │ │ │ │ │ + * provide with this layer. (b) The map can work with resolutions │ │ │ │ │ + * that aren't supported by the server, i.e. that aren't in │ │ │ │ │ + * <serverResolutions>. When the map is displayed in such a resolution │ │ │ │ │ + * data for the closest server-supported resolution is loaded and the │ │ │ │ │ + * layer div is stretched as necessary. │ │ │ │ │ + */ │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.TileCache │ │ │ │ │ + * Create a new read only TileCache layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} Name of the layer displayed in the interface │ │ │ │ │ + * url - {String} Location of the web accessible cache (not the location of │ │ │ │ │ + * your tilecache script!) │ │ │ │ │ + * layername - {String} Layer name as defined in the TileCache │ │ │ │ │ + * configuration │ │ │ │ │ + * options - {Object} Optional object with properties to be set on the │ │ │ │ │ + * layer. Note that you should speficy your resolutions to match │ │ │ │ │ + * your TileCache configuration. This can be done by setting │ │ │ │ │ + * the resolutions array directly (here or on the map), by setting │ │ │ │ │ + * maxResolution and numZoomLevels, or by using scale based properties. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, layername, options) { │ │ │ │ │ + this.layername = layername; │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, │ │ │ │ │ + [name, url, {}, options]); │ │ │ │ │ + this.extension = this.format.split('/')[1].toLowerCase(); │ │ │ │ │ + this.extension = (this.extension == 'jpg') ? 'jpeg' : this.extension; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * obj - {Object} │ │ │ │ │ * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.TileCache>} An exact clone of this │ │ │ │ │ + * <OpenLayers.Layer.TileCache> │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.TileCache(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.layername, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * components - {Array(<OpenLayers.Geometry.Point>)} Array of points │ │ │ │ │ - * separator - {String} character between coordinate pairs. Defaults to "," │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} hash with properties "path" (the string created from the │ │ │ │ │ - * components and "complete" (false if the renderer was unable to │ │ │ │ │ - * draw all components) │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as parameters. │ │ │ │ │ */ │ │ │ │ │ - getComponentsString: function(components, separator) { │ │ │ │ │ - var renderCmp = []; │ │ │ │ │ - var complete = true; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - var strings = []; │ │ │ │ │ - var str, component; │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - component = components[i]; │ │ │ │ │ - renderCmp.push(component); │ │ │ │ │ - str = this.getShortString(component); │ │ │ │ │ - if (str) { │ │ │ │ │ - strings.push(str); │ │ │ │ │ - } else { │ │ │ │ │ - // The current component is outside the valid range. Let's │ │ │ │ │ - // see if the previous or next component is inside the range. │ │ │ │ │ - // If so, add the coordinate of the intersection with the │ │ │ │ │ - // valid range bounds. │ │ │ │ │ - if (i > 0) { │ │ │ │ │ - if (this.getShortString(components[i - 1])) { │ │ │ │ │ - strings.push(this.clipLine(components[i], │ │ │ │ │ - components[i - 1])); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (i < len - 1) { │ │ │ │ │ - if (this.getShortString(components[i + 1])) { │ │ │ │ │ - strings.push(this.clipLine(components[i], │ │ │ │ │ - components[i + 1])); │ │ │ │ │ - } │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var bbox = this.maxExtent; │ │ │ │ │ + var size = this.tileSize; │ │ │ │ │ + var tileX = Math.round((bounds.left - bbox.left) / (res * size.w)); │ │ │ │ │ + var tileY = Math.round((bounds.bottom - bbox.bottom) / (res * size.h)); │ │ │ │ │ + var tileZ = this.serverResolutions != null ? │ │ │ │ │ + OpenLayers.Util.indexOf(this.serverResolutions, res) : │ │ │ │ │ + this.map.getZoom(); │ │ │ │ │ + │ │ │ │ │ + var components = [ │ │ │ │ │ + this.layername, │ │ │ │ │ + OpenLayers.Number.zeroPad(tileZ, 2), │ │ │ │ │ + OpenLayers.Number.zeroPad(parseInt(tileX / 1000000), 3), │ │ │ │ │ + OpenLayers.Number.zeroPad((parseInt(tileX / 1000) % 1000), 3), │ │ │ │ │ + OpenLayers.Number.zeroPad((parseInt(tileX) % 1000), 3), │ │ │ │ │ + OpenLayers.Number.zeroPad(parseInt(tileY / 1000000), 3), │ │ │ │ │ + OpenLayers.Number.zeroPad((parseInt(tileY / 1000) % 1000), 3), │ │ │ │ │ + OpenLayers.Number.zeroPad((parseInt(tileY) % 1000), 3) + '.' + this.extension │ │ │ │ │ + ]; │ │ │ │ │ + var path = components.join('/'); │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(path, url); │ │ │ │ │ + } │ │ │ │ │ + url = (url.charAt(url.length - 1) == '/') ? url : url + '/'; │ │ │ │ │ + return url + path; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.TileCache" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/WMTS.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.WMTS │ │ │ │ │ + * Instances of the WMTS class allow viewing of tiles from a service that │ │ │ │ │ + * implements the OGC WMTS specification version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.WMTS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} The layer will be considered a base layer. Default is true. │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} WMTS version. Default is "1.0.0". │ │ │ │ │ + */ │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: requestEncoding │ │ │ │ │ + * {String} Request encoding. Can be "REST" or "KVP". Default is "KVP". │ │ │ │ │ + */ │ │ │ │ │ + requestEncoding: "KVP", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String|Array(String)} The base URL or request URL template for the WMTS │ │ │ │ │ + * service. Must be provided. Array is only supported for base URLs, not │ │ │ │ │ + * for request URL templates. URL templates are only supported for │ │ │ │ │ + * REST <requestEncoding>. │ │ │ │ │ + */ │ │ │ │ │ + url: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layer │ │ │ │ │ + * {String} The layer identifier advertised by the WMTS service. Must be │ │ │ │ │ + * provided. │ │ │ │ │ + */ │ │ │ │ │ + layer: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: matrixSet │ │ │ │ │ + * {String} One of the advertised matrix set identifiers. Must be provided. │ │ │ │ │ + */ │ │ │ │ │ + matrixSet: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: style │ │ │ │ │ + * {String} One of the advertised layer styles. Must be provided. │ │ │ │ │ + */ │ │ │ │ │ + style: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: format │ │ │ │ │ + * {String} The image MIME type. Default is "image/jpeg". │ │ │ │ │ + */ │ │ │ │ │ + format: "image/jpeg", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: tileOrigin │ │ │ │ │ + * {<OpenLayers.LonLat>} The top-left corner of the tile matrix in map │ │ │ │ │ + * units. If the tile origin for each matrix in a set is different, │ │ │ │ │ + * the <matrixIds> should include a topLeftCorner property. If │ │ │ │ │ + * not provided, the tile origin will default to the top left corner │ │ │ │ │ + * of the layer <maxExtent>. │ │ │ │ │ + */ │ │ │ │ │ + tileOrigin: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: tileFullExtent │ │ │ │ │ + * {<OpenLayers.Bounds>} The full extent of the tile set. If not supplied, │ │ │ │ │ + * the layer's <maxExtent> property will be used. │ │ │ │ │ + */ │ │ │ │ │ + tileFullExtent: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: formatSuffix │ │ │ │ │ + * {String} For REST request encoding, an image format suffix must be │ │ │ │ │ + * included in the request. If not provided, the suffix will be derived │ │ │ │ │ + * from the <format> property. │ │ │ │ │ + */ │ │ │ │ │ + formatSuffix: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: matrixIds │ │ │ │ │ + * {Array} A list of tile matrix identifiers. If not provided, the matrix │ │ │ │ │ + * identifiers will be assumed to be integers corresponding to the │ │ │ │ │ + * map zoom level. If a list of strings is provided, each item should │ │ │ │ │ + * be the matrix identifier that corresponds to the map zoom level. │ │ │ │ │ + * Additionally, a list of objects can be provided. Each object should │ │ │ │ │ + * describe the matrix as presented in the WMTS capabilities. These │ │ │ │ │ + * objects should have the propertes shown below. │ │ │ │ │ + * │ │ │ │ │ + * Matrix properties: │ │ │ │ │ + * identifier - {String} The matrix identifier (required). │ │ │ │ │ + * scaleDenominator - {Number} The matrix scale denominator. │ │ │ │ │ + * topLeftCorner - {<OpenLayers.LonLat>} The top left corner of the │ │ │ │ │ + * matrix. Must be provided if different than the layer <tileOrigin>. │ │ │ │ │ + * tileWidth - {Number} The tile width for the matrix. Must be provided │ │ │ │ │ + * if different than the width given in the layer <tileSize>. │ │ │ │ │ + * tileHeight - {Number} The tile height for the matrix. Must be provided │ │ │ │ │ + * if different than the height given in the layer <tileSize>. │ │ │ │ │ + */ │ │ │ │ │ + matrixIds: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: dimensions │ │ │ │ │ + * {Array} For RESTful request encoding, extra dimensions may be specified. │ │ │ │ │ + * Items in this list should be property names in the <params> object. │ │ │ │ │ + * Values of extra dimensions will be determined from the corresponding │ │ │ │ │ + * values in the <params> object. │ │ │ │ │ + */ │ │ │ │ │ + dimensions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: params │ │ │ │ │ + * {Object} Extra parameters to include in tile requests. For KVP │ │ │ │ │ + * <requestEncoding>, these properties will be encoded in the request │ │ │ │ │ + * query string. For REST <requestEncoding>, these properties will │ │ │ │ │ + * become part of the request path, with order determined by the │ │ │ │ │ + * <dimensions> list. │ │ │ │ │ + */ │ │ │ │ │ + params: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomOffset │ │ │ │ │ + * {Number} If your cache has more levels than you want to provide │ │ │ │ │ + * access to with this layer, supply a zoomOffset. This zoom offset │ │ │ │ │ + * is added to the current map zoom level to determine the level │ │ │ │ │ + * for a requested tile. For example, if you supply a zoomOffset │ │ │ │ │ + * of 3, when the map is at the zoom 0, tiles will be requested from │ │ │ │ │ + * level 3 of your cache. Default is 0 (assumes cache level and map │ │ │ │ │ + * zoom are equivalent). Additionally, if this layer is to be used │ │ │ │ │ + * as an overlay and the cache has fewer zoom levels than the base │ │ │ │ │ + * layer, you can supply a negative zoomOffset. For example, if a │ │ │ │ │ + * map zoom level of 1 corresponds to your cache level zero, you would │ │ │ │ │ + * supply a -1 zoomOffset (and set the maxResolution of the layer │ │ │ │ │ + * appropriately). The zoomOffset value has no effect if complete │ │ │ │ │ + * matrix definitions (including scaleDenominator) are supplied in │ │ │ │ │ + * the <matrixIds> property. Defaults to 0 (no zoom offset). │ │ │ │ │ + */ │ │ │ │ │ + zoomOffset: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: serverResolutions │ │ │ │ │ + * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ + * property if the map resolutions differ from the server. This │ │ │ │ │ + * property serves two purposes. (a) <serverResolutions> can include │ │ │ │ │ + * resolutions that the server supports and that you don't want to │ │ │ │ │ + * provide with this layer; you can also look at <zoomOffset>, which is │ │ │ │ │ + * an alternative to <serverResolutions> for that specific purpose. │ │ │ │ │ + * (b) The map can work with resolutions that aren't supported by │ │ │ │ │ + * the server, i.e. that aren't in <serverResolutions>. When the │ │ │ │ │ + * map is displayed in such a resolution data for the closest │ │ │ │ │ + * server-supported resolution is loaded and the layer div is │ │ │ │ │ + * stretched as necessary. │ │ │ │ │ + */ │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: formatSuffixMap │ │ │ │ │ + * {Object} a map between WMTS 'format' request parameter and tile image file suffix │ │ │ │ │ + */ │ │ │ │ │ + formatSuffixMap: { │ │ │ │ │ + "image/png": "png", │ │ │ │ │ + "image/png8": "png", │ │ │ │ │ + "image/png24": "png", │ │ │ │ │ + "image/png32": "png", │ │ │ │ │ + "png": "png", │ │ │ │ │ + "image/jpeg": "jpg", │ │ │ │ │ + "image/jpg": "jpg", │ │ │ │ │ + "jpeg": "jpg", │ │ │ │ │ + "jpg": "jpg" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: matrix │ │ │ │ │ + * {Object} Matrix definition for the current map resolution. Updated by │ │ │ │ │ + * the <updateMatrixProperties> method. │ │ │ │ │ + */ │ │ │ │ │ + matrix: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.WMTS │ │ │ │ │ + * Create a new WMTS layer. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var wmts = new OpenLayers.Layer.WMTS({ │ │ │ │ │ + * name: "My WMTS Layer", │ │ │ │ │ + * url: "http://example.com/wmts", │ │ │ │ │ + * layer: "layer_id", │ │ │ │ │ + * style: "default", │ │ │ │ │ + * matrixSet: "matrix_id" │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Configuration properties for the layer. │ │ │ │ │ + * │ │ │ │ │ + * Required configuration properties: │ │ │ │ │ + * url - {String} The base url for the service. See the <url> property. │ │ │ │ │ + * layer - {String} The layer identifier. See the <layer> property. │ │ │ │ │ + * style - {String} The layer style identifier. See the <style> property. │ │ │ │ │ + * matrixSet - {String} The tile matrix set identifier. See the <matrixSet> │ │ │ │ │ + * property. │ │ │ │ │ + * │ │ │ │ │ + * Any other documented layer properties can be provided in the config object. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ + │ │ │ │ │ + // confirm required properties are supplied │ │ │ │ │ + var required = { │ │ │ │ │ + url: true, │ │ │ │ │ + layer: true, │ │ │ │ │ + style: true, │ │ │ │ │ + matrixSet: true │ │ │ │ │ + }; │ │ │ │ │ + for (var prop in required) { │ │ │ │ │ + if (!(prop in config)) { │ │ │ │ │ + throw new Error("Missing property '" + prop + "' in layer configuration."); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + config.params = OpenLayers.Util.upperCaseObject(config.params); │ │ │ │ │ + var args = [config.name, config.url, config.params, config]; │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, args); │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + // determine format suffix (for REST) │ │ │ │ │ + if (!this.formatSuffix) { │ │ │ │ │ + this.formatSuffix = this.formatSuffixMap[this.format] || this.format.split("/").pop(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // expand matrixIds (may be array of string or array of object) │ │ │ │ │ + if (this.matrixIds) { │ │ │ │ │ + var len = this.matrixIds.length; │ │ │ │ │ + if (len && typeof this.matrixIds[0] === "string") { │ │ │ │ │ + var ids = this.matrixIds; │ │ │ │ │ + this.matrixIds = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + this.matrixIds[i] = { │ │ │ │ │ + identifier: ids[i] │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ - complete = false; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return { │ │ │ │ │ - path: strings.join(separator || ","), │ │ │ │ │ - complete: complete │ │ │ │ │ - }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clipLine │ │ │ │ │ - * Given two points (one inside the valid range, and one outside), │ │ │ │ │ - * clips the line betweeen the two points so that the new points are both │ │ │ │ │ - * inside the valid range. │ │ │ │ │ + * Method: setMap │ │ │ │ │ + */ │ │ │ │ │ + setMap: function() { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateMatrixProperties │ │ │ │ │ + * Called when map resolution changes to update matrix related properties. │ │ │ │ │ + */ │ │ │ │ │ + updateMatrixProperties: function() { │ │ │ │ │ + this.matrix = this.getMatrix(); │ │ │ │ │ + if (this.matrix) { │ │ │ │ │ + if (this.matrix.topLeftCorner) { │ │ │ │ │ + this.tileOrigin = this.matrix.topLeftCorner; │ │ │ │ │ + } │ │ │ │ │ + if (this.matrix.tileWidth && this.matrix.tileHeight) { │ │ │ │ │ + this.tileSize = new OpenLayers.Size( │ │ │ │ │ + this.matrix.tileWidth, this.matrix.tileHeight │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (!this.tileOrigin) { │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat( │ │ │ │ │ + this.maxExtent.left, this.maxExtent.top │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (!this.tileFullExtent) { │ │ │ │ │ + this.tileFullExtent = this.maxExtent; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * badComponent - {<OpenLayers.Geometry.Point>} original geometry of the │ │ │ │ │ - * invalid point │ │ │ │ │ - * goodComponent - {<OpenLayers.Geometry.Point>} original geometry of the │ │ │ │ │ - * valid point │ │ │ │ │ - * Returns │ │ │ │ │ - * {String} the SVG coordinate pair of the clipped point (like │ │ │ │ │ - * getShortString), or an empty string if both passed componets are at │ │ │ │ │ - * the same point. │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * zoomChanged - {Boolean} Tells when zoom has changed, as layers have to │ │ │ │ │ + * do some init work in that case. │ │ │ │ │ + * dragging - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - clipLine: function(badComponent, goodComponent) { │ │ │ │ │ - if (goodComponent.equals(badComponent)) { │ │ │ │ │ - return ""; │ │ │ │ │ - } │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var maxX = this.MAX_PIXEL - this.translationParameters.x; │ │ │ │ │ - var maxY = this.MAX_PIXEL - this.translationParameters.y; │ │ │ │ │ - var x1 = (goodComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y1 = this.top - goodComponent.y / resolution; │ │ │ │ │ - var x2 = (badComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y2 = this.top - badComponent.y / resolution; │ │ │ │ │ - var k; │ │ │ │ │ - if (x2 < -maxX || x2 > maxX) { │ │ │ │ │ - k = (y2 - y1) / (x2 - x1); │ │ │ │ │ - x2 = x2 < 0 ? -maxX : maxX; │ │ │ │ │ - y2 = y1 + (x2 - x1) * k; │ │ │ │ │ - } │ │ │ │ │ - if (y2 < -maxY || y2 > maxY) { │ │ │ │ │ - k = (x2 - x1) / (y2 - y1); │ │ │ │ │ - y2 = y2 < 0 ? -maxY : maxY; │ │ │ │ │ - x2 = x1 + (y2 - y1) * k; │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + if (zoomChanged || !this.matrix) { │ │ │ │ │ + this.updateMatrixProperties(); │ │ │ │ │ } │ │ │ │ │ - return x2 + "," + y2; │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.moveTo.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getShortString │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * obj - {Object} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} or false if point is outside the valid range │ │ │ │ │ + * {<OpenLayers.Layer.WMTS>} An exact clone of this <OpenLayers.Layer.WMTS> │ │ │ │ │ */ │ │ │ │ │ - getShortString: function(point) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = ((point.x - this.featureDx) / resolution + this.left); │ │ │ │ │ - var y = (this.top - point.y / resolution); │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.WMTS(this.options); │ │ │ │ │ + } │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - return x + "," + y; │ │ │ │ │ + /** │ │ │ │ │ + * Method: getIdentifier │ │ │ │ │ + * Get the current index in the matrixIds array. │ │ │ │ │ + */ │ │ │ │ │ + getIdentifier: function() { │ │ │ │ │ + return this.getServerZoom(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getMatrix │ │ │ │ │ + * Get the appropriate matrix definition for the current map resolution. │ │ │ │ │ + */ │ │ │ │ │ + getMatrix: function() { │ │ │ │ │ + var matrix; │ │ │ │ │ + if (!this.matrixIds || this.matrixIds.length === 0) { │ │ │ │ │ + matrix = { │ │ │ │ │ + identifier: this.getIdentifier() │ │ │ │ │ + }; │ │ │ │ │ } else { │ │ │ │ │ - return false; │ │ │ │ │ + // get appropriate matrix given the map scale if possible │ │ │ │ │ + if ("scaleDenominator" in this.matrixIds[0]) { │ │ │ │ │ + // scale denominator calculation based on WMTS spec │ │ │ │ │ + var denom = │ │ │ │ │ + OpenLayers.METERS_PER_INCH * │ │ │ │ │ + OpenLayers.INCHES_PER_UNIT[this.units] * │ │ │ │ │ + this.getServerResolution() / 0.28E-3; │ │ │ │ │ + var diff = Number.POSITIVE_INFINITY; │ │ │ │ │ + var delta; │ │ │ │ │ + for (var i = 0, ii = this.matrixIds.length; i < ii; ++i) { │ │ │ │ │ + delta = Math.abs(1 - (this.matrixIds[i].scaleDenominator / denom)); │ │ │ │ │ + if (delta < diff) { │ │ │ │ │ + diff = delta; │ │ │ │ │ + matrix = this.matrixIds[i]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // fall back on zoom as index │ │ │ │ │ + matrix = this.matrixIds[this.getIdentifier()]; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return matrix; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getPosition │ │ │ │ │ - * Finds the position of an svg node. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: getTileInfo │ │ │ │ │ + * Get tile information for a given location at the current map resolution. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ + * loc - {<OpenLayers.LonLat} A location in map coordinates. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} hash with x and y properties, representing the coordinates │ │ │ │ │ - * within the svg coordinate system │ │ │ │ │ + * {Object} An object with "col", "row", "i", and "j" properties. The col │ │ │ │ │ + * and row values are zero based tile indexes from the top left. The │ │ │ │ │ + * i and j values are the number of pixels to the left and top │ │ │ │ │ + * (respectively) of the given location within the target tile. │ │ │ │ │ */ │ │ │ │ │ - getPosition: function(node) { │ │ │ │ │ - return ({ │ │ │ │ │ - x: parseFloat(node.getAttributeNS(null, "cx")), │ │ │ │ │ - y: parseFloat(node.getAttributeNS(null, "cy")) │ │ │ │ │ - }); │ │ │ │ │ + getTileInfo: function(loc) { │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + │ │ │ │ │ + var fx = (loc.lon - this.tileOrigin.lon) / (res * this.tileSize.w); │ │ │ │ │ + var fy = (this.tileOrigin.lat - loc.lat) / (res * this.tileSize.h); │ │ │ │ │ + │ │ │ │ │ + var col = Math.floor(fx); │ │ │ │ │ + var row = Math.floor(fy); │ │ │ │ │ + │ │ │ │ │ + return { │ │ │ │ │ + col: col, │ │ │ │ │ + row: row, │ │ │ │ │ + i: Math.floor((fx - col) * this.tileSize.w), │ │ │ │ │ + j: Math.floor((fy - row) * this.tileSize.h) │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: importSymbol │ │ │ │ │ - * add a new symbol definition from the rendererer's symbol hash │ │ │ │ │ + * Method: getURL │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * graphicName - {String} name of the symbol to import │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} - the imported symbol │ │ │ │ │ + * {String} A URL for the tile corresponding to the given bounds. │ │ │ │ │ */ │ │ │ │ │ - importSymbol: function(graphicName) { │ │ │ │ │ - if (!this.defs) { │ │ │ │ │ - // create svg defs tag │ │ │ │ │ - this.defs = this.createDefs(); │ │ │ │ │ - } │ │ │ │ │ - var id = this.container.id + "-" + graphicName; │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var url = ""; │ │ │ │ │ + if (!this.tileFullExtent || this.tileFullExtent.intersectsBounds(bounds)) { │ │ │ │ │ │ │ │ │ │ - // check if symbol already exists in the defs │ │ │ │ │ - var existing = document.getElementById(id); │ │ │ │ │ - if (existing != null) { │ │ │ │ │ - return existing; │ │ │ │ │ - } │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var info = this.getTileInfo(center); │ │ │ │ │ + var matrixId = this.matrix.identifier; │ │ │ │ │ + var dimensions = this.dimensions, │ │ │ │ │ + params; │ │ │ │ │ │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(graphicName + ' is not a valid symbol name'); │ │ │ │ │ - } │ │ │ │ │ + if (OpenLayers.Util.isArray(this.url)) { │ │ │ │ │ + url = this.selectUrl([ │ │ │ │ │ + this.version, this.style, this.matrixSet, │ │ │ │ │ + this.matrix.identifier, info.row, info.col │ │ │ │ │ + ].join(","), this.url); │ │ │ │ │ + } else { │ │ │ │ │ + url = this.url; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var symbolNode = this.nodeFactory(id, "symbol"); │ │ │ │ │ - var node = this.nodeFactory(null, "polygon"); │ │ │ │ │ - symbolNode.appendChild(node); │ │ │ │ │ - var symbolExtent = new OpenLayers.Bounds( │ │ │ │ │ - Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ + if (this.requestEncoding.toUpperCase() === "REST") { │ │ │ │ │ + params = this.params; │ │ │ │ │ + if (url.indexOf("{") !== -1) { │ │ │ │ │ + var template = url.replace(/\{/g, "${"); │ │ │ │ │ + var context = { │ │ │ │ │ + // spec does not make clear if capital S or not │ │ │ │ │ + style: this.style, │ │ │ │ │ + Style: this.style, │ │ │ │ │ + TileMatrixSet: this.matrixSet, │ │ │ │ │ + TileMatrix: this.matrix.identifier, │ │ │ │ │ + TileRow: info.row, │ │ │ │ │ + TileCol: info.col │ │ │ │ │ + }; │ │ │ │ │ + if (dimensions) { │ │ │ │ │ + var dimension, i; │ │ │ │ │ + for (i = dimensions.length - 1; i >= 0; --i) { │ │ │ │ │ + dimension = dimensions[i]; │ │ │ │ │ + context[dimension] = params[dimension.toUpperCase()]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + url = OpenLayers.String.format(template, context); │ │ │ │ │ + } else { │ │ │ │ │ + // include 'version', 'layer' and 'style' in tile resource url │ │ │ │ │ + var path = this.version + "/" + this.layer + "/" + this.style + "/"; │ │ │ │ │ │ │ │ │ │ - var points = []; │ │ │ │ │ - var x, y; │ │ │ │ │ - for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ - symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ - symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ - symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ - points.push(x, ",", y); │ │ │ │ │ - } │ │ │ │ │ + // append optional dimension path elements │ │ │ │ │ + if (dimensions) { │ │ │ │ │ + for (var i = 0; i < dimensions.length; i++) { │ │ │ │ │ + if (params[dimensions[i]]) { │ │ │ │ │ + path = path + params[dimensions[i]] + "/"; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - node.setAttributeNS(null, "points", points.join(" ")); │ │ │ │ │ + // append other required path elements │ │ │ │ │ + path = path + this.matrixSet + "/" + this.matrix.identifier + │ │ │ │ │ + "/" + info.row + "/" + info.col + "." + this.formatSuffix; │ │ │ │ │ │ │ │ │ │ - var width = symbolExtent.getWidth(); │ │ │ │ │ - var height = symbolExtent.getHeight(); │ │ │ │ │ - // create a viewBox three times as large as the symbol itself, │ │ │ │ │ - // to allow for strokeWidth being displayed correctly at the corners. │ │ │ │ │ - var viewBox = [symbolExtent.left - width, │ │ │ │ │ - symbolExtent.bottom - height, width * 3, height * 3 │ │ │ │ │ - ]; │ │ │ │ │ - symbolNode.setAttributeNS(null, "viewBox", viewBox.join(" ")); │ │ │ │ │ - this.symbolMetrics[id] = [ │ │ │ │ │ - Math.max(width, height), │ │ │ │ │ - symbolExtent.getCenterLonLat().lon, │ │ │ │ │ - symbolExtent.getCenterLonLat().lat │ │ │ │ │ - ]; │ │ │ │ │ + if (!url.match(/\/$/)) { │ │ │ │ │ + url = url + "/"; │ │ │ │ │ + } │ │ │ │ │ + url = url + path; │ │ │ │ │ + } │ │ │ │ │ + } else if (this.requestEncoding.toUpperCase() === "KVP") { │ │ │ │ │ │ │ │ │ │ - this.defs.appendChild(symbolNode); │ │ │ │ │ - return symbolNode; │ │ │ │ │ + // assemble all required parameters │ │ │ │ │ + params = { │ │ │ │ │ + SERVICE: "WMTS", │ │ │ │ │ + REQUEST: "GetTile", │ │ │ │ │ + VERSION: this.version, │ │ │ │ │ + LAYER: this.layer, │ │ │ │ │ + STYLE: this.style, │ │ │ │ │ + TILEMATRIXSET: this.matrixSet, │ │ │ │ │ + TILEMATRIX: this.matrix.identifier, │ │ │ │ │ + TILEROW: info.row, │ │ │ │ │ + TILECOL: info.col, │ │ │ │ │ + FORMAT: this.format │ │ │ │ │ + }; │ │ │ │ │ + url = OpenLayers.Layer.Grid.prototype.getFullRequestString.apply(this, [params]); │ │ │ │ │ + │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return url; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureIdFromEvent │ │ │ │ │ + * APIMethod: mergeNewParams │ │ │ │ │ + * Extend the existing layer <params> with new properties. Tiles will be │ │ │ │ │ + * reloaded with updated params in the request. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} An <OpenLayers.Event> object │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A feature id or undefined. │ │ │ │ │ + * newParams - {Object} Properties to extend to existing <params>. │ │ │ │ │ */ │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var featureId = OpenLayers.Renderer.Elements.prototype.getFeatureIdFromEvent.apply(this, arguments); │ │ │ │ │ - if (!featureId) { │ │ │ │ │ - var target = evt.target; │ │ │ │ │ - featureId = target.parentNode && target != this.rendererRoot ? │ │ │ │ │ - target.parentNode._featureId : undefined; │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + if (this.requestEncoding.toUpperCase() === "KVP") { │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply( │ │ │ │ │ + this, [OpenLayers.Util.upperCaseObject(newParams)] │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - return featureId; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.SVG" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.WMTS" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.SVG.LABEL_ALIGN │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_ALIGN = { │ │ │ │ │ - "l": "start", │ │ │ │ │ - "r": "end", │ │ │ │ │ - "b": "bottom", │ │ │ │ │ - "t": "hanging" │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.SVG.LABEL_VSHIFT │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_VSHIFT = { │ │ │ │ │ - // according to │ │ │ │ │ - // http://www.w3.org/Graphics/SVG/Test/20061213/htmlObjectHarness/full-text-align-02-b.html │ │ │ │ │ - // a baseline-shift of -70% shifts the text exactly from the │ │ │ │ │ - // bottom to the top of the baseline, so -35% moves the text to │ │ │ │ │ - // the center of the baseline. │ │ │ │ │ - "t": "-70%", │ │ │ │ │ - "b": "0" │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.SVG.LABEL_VFACTOR │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_VFACTOR = { │ │ │ │ │ - "t": 0, │ │ │ │ │ - "b": -1 │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Function: OpenLayers.Renderer.SVG.preventDefault │ │ │ │ │ - * *Deprecated*. Use <OpenLayers.Event.preventDefault> method instead. │ │ │ │ │ - * Used to prevent default events (especially opening images in a new tab on │ │ │ │ │ - * ctrl-click) from being executed for externalGraphic symbols │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.SVG.preventDefault = function(e) { │ │ │ │ │ - OpenLayers.Event.preventDefault(e); │ │ │ │ │ -}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Tile/Image/IFrame.js │ │ │ │ │ + OpenLayers/Layer/EventPane.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Tile/Image.js │ │ │ │ │ + * @requires OpenLayers/Layer.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: OpenLayers.Tile.Image.IFrame │ │ │ │ │ - * Mixin for tiles that use form-encoded POST requests to get images from │ │ │ │ │ - * remote services. Images will be loaded using HTTP-POST into an IFrame. │ │ │ │ │ + * Class: OpenLayers.Layer.EventPane │ │ │ │ │ + * Base class for 3rd party layers, providing a DOM element which isolates │ │ │ │ │ + * the 3rd-party layer from mouse events. │ │ │ │ │ + * Only used by Google layers. │ │ │ │ │ * │ │ │ │ │ - * This mixin will be applied to <OpenLayers.Tile.Image> instances │ │ │ │ │ - * configured with <OpenLayers.Tile.Image.maxGetUrlLength> set. │ │ │ │ │ + * Automatically instantiated by the Google constructor, and not usually instantiated directly. │ │ │ │ │ + * │ │ │ │ │ + * Create a new event pane layer with the │ │ │ │ │ + * <OpenLayers.Layer.EventPane> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Tile.Image.IFrame = { │ │ │ │ │ +OpenLayers.Layer.EventPane = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: useIFrame │ │ │ │ │ - * {Boolean} true if we are currently using an IFrame to render POST │ │ │ │ │ - * responses, false if we are using an img element to render GET responses. │ │ │ │ │ + * APIProperty: smoothDragPan │ │ │ │ │ + * {Boolean} smoothDragPan determines whether non-public/internal API │ │ │ │ │ + * methods are used for better performance while dragging EventPane │ │ │ │ │ + * layers. When not in sphericalMercator mode, the smoother dragging │ │ │ │ │ + * doesn't actually move north/south directly with the number of │ │ │ │ │ + * pixels moved, resulting in a slight offset when you drag your mouse │ │ │ │ │ + * north south with this option on. If this visual disparity bothers │ │ │ │ │ + * you, you should turn this option off, or use spherical mercator. │ │ │ │ │ + * Default is on. │ │ │ │ │ */ │ │ │ │ │ - useIFrame: null, │ │ │ │ │ + smoothDragPan: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: blankImageUrl │ │ │ │ │ - * {String} Using a data scheme url is not supported by all browsers, but │ │ │ │ │ - * we don't care because we either set it as css backgroundImage, or the │ │ │ │ │ - * image's display style is set to "none" when we use it. │ │ │ │ │ + * Property: isBaseLayer │ │ │ │ │ + * {Boolean} EventPaned layers are always base layers, by necessity. │ │ │ │ │ */ │ │ │ │ │ - blankImageUrl: "data:image/gif;base64,R0lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAQAIBRAA7", │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Set useIFrame in the instance, and operate the image/iframe switch. │ │ │ │ │ - * Then call Tile.Image.draw. │ │ │ │ │ + * APIProperty: isFixed │ │ │ │ │ + * {Boolean} EventPaned layers are fixed by default. │ │ │ │ │ + */ │ │ │ │ │ + isFixed: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: pane │ │ │ │ │ + * {DOMElement} A reference to the element that controls the events. │ │ │ │ │ + */ │ │ │ │ │ + pane: null, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: mapObject │ │ │ │ │ + * {Object} This is the object which will be used to load the 3rd party library │ │ │ │ │ + * in the case of the google layer, this will be of type GMap, │ │ │ │ │ + * in the case of the ve layer, this will be of type VEMap │ │ │ │ │ + */ │ │ │ │ │ + mapObject: null, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.EventPane │ │ │ │ │ + * Create a new event pane layer │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - var draw = OpenLayers.Tile.Image.prototype.shouldDraw.call(this); │ │ │ │ │ - if (draw) { │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (this.pane == null) { │ │ │ │ │ + this.pane = OpenLayers.Util.createDiv(this.div.id + "_EventPane"); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // this.url isn't set to the currect value yet, so we call getURL │ │ │ │ │ - // on the layer and store the result in a local variable │ │ │ │ │ - var url = this.layer.getURL(this.bounds); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Deconstruct this layer. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.mapObject = null; │ │ │ │ │ + this.pane = null; │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var usedIFrame = this.useIFrame; │ │ │ │ │ - this.useIFrame = this.maxGetUrlLength !== null && │ │ │ │ │ - !this.layer.async && │ │ │ │ │ - url.length > this.maxGetUrlLength; │ │ │ │ │ │ │ │ │ │ - var fromIFrame = usedIFrame && !this.useIFrame; │ │ │ │ │ - var toIFrame = !usedIFrame && this.useIFrame; │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the layer. This is done through an accessor │ │ │ │ │ + * so that subclasses can override this and take special action once │ │ │ │ │ + * they have their map variable set. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - if (fromIFrame || toIFrame) { │ │ │ │ │ + this.pane.style.zIndex = parseInt(this.div.style.zIndex) + 1; │ │ │ │ │ + this.pane.style.display = this.div.style.display; │ │ │ │ │ + this.pane.style.width = "100%"; │ │ │ │ │ + this.pane.style.height = "100%"; │ │ │ │ │ + if (OpenLayers.BROWSER_NAME == "msie") { │ │ │ │ │ + this.pane.style.background = │ │ │ │ │ + "url(" + OpenLayers.Util.getImageLocation("blank.gif") + ")"; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // Switching between GET (image) and POST (iframe). │ │ │ │ │ + if (this.isFixed) { │ │ │ │ │ + this.map.viewPortDiv.appendChild(this.pane); │ │ │ │ │ + } else { │ │ │ │ │ + this.map.layerContainerDiv.appendChild(this.pane); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // We remove the imgDiv (really either an image or an iframe) │ │ │ │ │ - // from the frame and set it to null to make sure initImage │ │ │ │ │ - // will call getImage. │ │ │ │ │ + // once our layer has been added to the map, we can load it │ │ │ │ │ + this.loadMapObject(); │ │ │ │ │ │ │ │ │ │ - if (this.imgDiv && this.imgDiv.parentNode === this.frame) { │ │ │ │ │ - this.frame.removeChild(this.imgDiv); │ │ │ │ │ - } │ │ │ │ │ - this.imgDiv = null; │ │ │ │ │ + // if map didn't load, display warning │ │ │ │ │ + if (this.mapObject == null) { │ │ │ │ │ + this.loadWarningMessage(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // And if we had an iframe we also remove the event pane. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: removeMap │ │ │ │ │ + * On being removed from the map, we'll like to remove the invisible 'pane' │ │ │ │ │ + * div that we added to it on creation. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + if (this.pane && this.pane.parentNode) { │ │ │ │ │ + this.pane.parentNode.removeChild(this.pane); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.prototype.removeMap.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (fromIFrame) { │ │ │ │ │ - this.frame.removeChild(this.frame.firstChild); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: loadWarningMessage │ │ │ │ │ + * If we can't load the map lib, then display an error message to the │ │ │ │ │ + * user and tell them where to go for help. │ │ │ │ │ + * │ │ │ │ │ + * This function sets up the layout for the warning message. Each 3rd │ │ │ │ │ + * party layer must implement its own getWarningHTML() function to │ │ │ │ │ + * provide the actual warning message. │ │ │ │ │ + */ │ │ │ │ │ + loadWarningMessage: function() { │ │ │ │ │ + │ │ │ │ │ + this.div.style.backgroundColor = "darkblue"; │ │ │ │ │ + │ │ │ │ │ + var viewSize = this.map.getSize(); │ │ │ │ │ + │ │ │ │ │ + var msgW = Math.min(viewSize.w, 300); │ │ │ │ │ + var msgH = Math.min(viewSize.h, 200); │ │ │ │ │ + var size = new OpenLayers.Size(msgW, msgH); │ │ │ │ │ + │ │ │ │ │ + var centerPx = new OpenLayers.Pixel(viewSize.w / 2, viewSize.h / 2); │ │ │ │ │ + │ │ │ │ │ + var topLeft = centerPx.add(-size.w / 2, -size.h / 2); │ │ │ │ │ + │ │ │ │ │ + var div = OpenLayers.Util.createDiv(this.name + "_warning", │ │ │ │ │ + topLeft, │ │ │ │ │ + size, │ │ │ │ │ + null, │ │ │ │ │ + null, │ │ │ │ │ + null, │ │ │ │ │ + "auto"); │ │ │ │ │ + │ │ │ │ │ + div.style.padding = "7px"; │ │ │ │ │ + div.style.backgroundColor = "yellow"; │ │ │ │ │ + │ │ │ │ │ + div.innerHTML = this.getWarningHTML(); │ │ │ │ │ + this.div.appendChild(div); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getWarningHTML │ │ │ │ │ + * To be implemented by subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} String with information on why layer is broken, how to get │ │ │ │ │ + * it working. │ │ │ │ │ + */ │ │ │ │ │ + getWarningHTML: function() { │ │ │ │ │ + //should be implemented by subclasses │ │ │ │ │ + return ""; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: display │ │ │ │ │ + * Set the display on the pane │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * display - {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + display: function(display) { │ │ │ │ │ + OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ + this.pane.style.display = this.div.style.display; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setZIndex │ │ │ │ │ + * Set the z-index order for the pane. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * zIndex - {int} │ │ │ │ │ + */ │ │ │ │ │ + setZIndex: function(zIndex) { │ │ │ │ │ + OpenLayers.Layer.prototype.setZIndex.apply(this, arguments); │ │ │ │ │ + this.pane.style.zIndex = parseInt(this.div.style.zIndex) + 1; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveByPx │ │ │ │ │ + * Move the layer based on pixel vector. To be implemented by subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * dx - {Number} The x coord of the displacement vector. │ │ │ │ │ + * dy - {Number} The y coord of the displacement vector. │ │ │ │ │ + */ │ │ │ │ │ + moveByPx: function(dx, dy) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveByPx.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + if (this.dragPanMapObject) { │ │ │ │ │ + this.dragPanMapObject(dx, -dy); │ │ │ │ │ + } else { │ │ │ │ │ + this.moveTo(this.map.getCachedCenter()); │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Tile.Image.prototype.draw.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getImage │ │ │ │ │ - * Creates the content for the frame on the tile. │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * Handle calls to move the layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * zoomChanged - {Boolean} │ │ │ │ │ + * dragging - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - getImage: function() { │ │ │ │ │ - if (this.useIFrame === true) { │ │ │ │ │ - if (!this.frame.childNodes.length) { │ │ │ │ │ - var eventPane = document.createElement("div"), │ │ │ │ │ - style = eventPane.style; │ │ │ │ │ - style.position = "absolute"; │ │ │ │ │ - style.width = "100%"; │ │ │ │ │ - style.height = "100%"; │ │ │ │ │ - style.zIndex = 1; │ │ │ │ │ - style.backgroundImage = "url(" + this.blankImageUrl + ")"; │ │ │ │ │ - this.frame.appendChild(eventPane); │ │ │ │ │ - } │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - var id = this.id + '_iFrame', │ │ │ │ │ - iframe; │ │ │ │ │ - if (parseFloat(navigator.appVersion.split("MSIE")[1]) < 9) { │ │ │ │ │ - // Older IE versions do not set the name attribute of an iFrame │ │ │ │ │ - // properly via DOM manipulation, so we need to do it on our own with │ │ │ │ │ - // this hack. │ │ │ │ │ - iframe = document.createElement('<iframe name="' + id + '">'); │ │ │ │ │ + if (this.mapObject != null) { │ │ │ │ │ │ │ │ │ │ - // IFrames in older IE versions are not transparent, if you set │ │ │ │ │ - // the backgroundColor transparent. This is a workaround to get │ │ │ │ │ - // transparent iframes. │ │ │ │ │ - iframe.style.backgroundColor = '#FFFFFF'; │ │ │ │ │ - iframe.style.filter = 'chroma(color=#FFFFFF)'; │ │ │ │ │ - } else { │ │ │ │ │ - iframe = document.createElement('iframe'); │ │ │ │ │ - iframe.style.backgroundColor = 'transparent'; │ │ │ │ │ + var newCenter = this.map.getCenter(); │ │ │ │ │ + var newZoom = this.map.getZoom(); │ │ │ │ │ │ │ │ │ │ - // iframe.name needs to be an unique id, otherwise it │ │ │ │ │ - // could happen that other iframes are overwritten. │ │ │ │ │ - iframe.name = id; │ │ │ │ │ - } │ │ │ │ │ + if (newCenter != null) { │ │ │ │ │ │ │ │ │ │ - // some special properties to avoid scaling the images and scrollbars │ │ │ │ │ - // in the iframe │ │ │ │ │ - iframe.scrolling = 'no'; │ │ │ │ │ - iframe.marginWidth = '0px'; │ │ │ │ │ - iframe.marginHeight = '0px'; │ │ │ │ │ - iframe.frameBorder = '0'; │ │ │ │ │ + var moOldCenter = this.getMapObjectCenter(); │ │ │ │ │ + var oldCenter = this.getOLLonLatFromMapObjectLonLat(moOldCenter); │ │ │ │ │ │ │ │ │ │ - iframe.style.position = "absolute"; │ │ │ │ │ - iframe.style.width = "100%"; │ │ │ │ │ - iframe.style.height = "100%"; │ │ │ │ │ + var moOldZoom = this.getMapObjectZoom(); │ │ │ │ │ + var oldZoom = this.getOLZoomFromMapObjectZoom(moOldZoom); │ │ │ │ │ │ │ │ │ │ - if (this.layer.opacity < 1) { │ │ │ │ │ - OpenLayers.Util.modifyDOMElement(iframe, null, null, null, │ │ │ │ │ - null, null, null, this.layer.opacity); │ │ │ │ │ + if (!(newCenter.equals(oldCenter)) || newZoom != oldZoom) { │ │ │ │ │ + │ │ │ │ │ + if (!zoomChanged && oldCenter && this.dragPanMapObject && │ │ │ │ │ + this.smoothDragPan) { │ │ │ │ │ + var oldPx = this.map.getViewPortPxFromLonLat(oldCenter); │ │ │ │ │ + var newPx = this.map.getViewPortPxFromLonLat(newCenter); │ │ │ │ │ + this.dragPanMapObject(newPx.x - oldPx.x, oldPx.y - newPx.y); │ │ │ │ │ + } else { │ │ │ │ │ + var center = this.getMapObjectLonLatFromOLLonLat(newCenter); │ │ │ │ │ + var zoom = this.getMapObjectZoomFromOLZoom(newZoom); │ │ │ │ │ + this.setMapObjectCenter(center, zoom, dragging); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.frame.appendChild(iframe); │ │ │ │ │ - this.imgDiv = iframe; │ │ │ │ │ - return iframe; │ │ │ │ │ - } else { │ │ │ │ │ - return OpenLayers.Tile.Image.prototype.getImage.apply(this, arguments); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ + /********************************************************/ │ │ │ │ │ + /* */ │ │ │ │ │ + /* Baselayer Functions */ │ │ │ │ │ + /* */ │ │ │ │ │ + /********************************************************/ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: createRequestForm │ │ │ │ │ - * Create the html <form> element with width, height, bbox and all │ │ │ │ │ - * parameters specified in the layer params. │ │ │ │ │ + * Method: getLonLatFromViewPortPx │ │ │ │ │ + * Get a map location from a pixel location │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * viewPortPx - {<OpenLayers.Pixel>} │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The form element which sends the HTTP-POST request to the │ │ │ │ │ - * WMS. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.LonLat>} An OpenLayers.LonLat which is the passed-in view │ │ │ │ │ + * port OpenLayers.Pixel, translated into lon/lat by map lib │ │ │ │ │ + * If the map lib is not loaded or not centered, returns null │ │ │ │ │ */ │ │ │ │ │ - createRequestForm: function() { │ │ │ │ │ - // creation of the form element │ │ │ │ │ - var form = document.createElement('form'); │ │ │ │ │ - form.method = 'POST'; │ │ │ │ │ - var cacheId = this.layer.params["_OLSALT"]; │ │ │ │ │ - cacheId = (cacheId ? cacheId + "_" : "") + this.bounds.toBBOX(); │ │ │ │ │ - form.action = OpenLayers.Util.urlAppend(this.layer.url, cacheId); │ │ │ │ │ - form.target = this.id + '_iFrame'; │ │ │ │ │ + getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ + var lonlat = null; │ │ │ │ │ + if ((this.mapObject != null) && │ │ │ │ │ + (this.getMapObjectCenter() != null)) { │ │ │ │ │ + var moPixel = this.getMapObjectPixelFromOLPixel(viewPortPx); │ │ │ │ │ + var moLonLat = this.getMapObjectLonLatFromMapObjectPixel(moPixel); │ │ │ │ │ + lonlat = this.getOLLonLatFromMapObjectLonLat(moLonLat); │ │ │ │ │ + } │ │ │ │ │ + return lonlat; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // adding all parameters in layer params as hidden fields to the html │ │ │ │ │ - // form element │ │ │ │ │ - var imageSize = this.layer.getImageSize(), │ │ │ │ │ - params = OpenLayers.Util.getParameters(this.url), │ │ │ │ │ - field; │ │ │ │ │ │ │ │ │ │ - for (var par in params) { │ │ │ │ │ - field = document.createElement('input'); │ │ │ │ │ - field.type = 'hidden'; │ │ │ │ │ - field.name = par; │ │ │ │ │ - field.value = params[par]; │ │ │ │ │ - form.appendChild(field); │ │ │ │ │ + /** │ │ │ │ │ + * Method: getViewPortPxFromLonLat │ │ │ │ │ + * Get a pixel location from a map location │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Pixel>} An OpenLayers.Pixel which is the passed-in │ │ │ │ │ + * OpenLayers.LonLat, translated into view port pixels by map lib │ │ │ │ │ + * If map lib is not loaded or not centered, returns null │ │ │ │ │ + */ │ │ │ │ │ + getViewPortPxFromLonLat: function(lonlat) { │ │ │ │ │ + var viewPortPx = null; │ │ │ │ │ + if ((this.mapObject != null) && │ │ │ │ │ + (this.getMapObjectCenter() != null)) { │ │ │ │ │ + │ │ │ │ │ + var moLonLat = this.getMapObjectLonLatFromOLLonLat(lonlat); │ │ │ │ │ + var moPixel = this.getMapObjectPixelFromMapObjectLonLat(moLonLat); │ │ │ │ │ + │ │ │ │ │ + viewPortPx = this.getOLPixelFromMapObjectPixel(moPixel); │ │ │ │ │ } │ │ │ │ │ + return viewPortPx; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return form; │ │ │ │ │ + /********************************************************/ │ │ │ │ │ + /* */ │ │ │ │ │ + /* Translation Functions */ │ │ │ │ │ + /* */ │ │ │ │ │ + /* The following functions translate Map Object and */ │ │ │ │ │ + /* OL formats for Pixel, LonLat */ │ │ │ │ │ + /* */ │ │ │ │ │ + /********************************************************/ │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + // TRANSLATION: MapObject LatLng <-> OpenLayers.LonLat │ │ │ │ │ + // │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getOLLonLatFromMapObjectLonLat │ │ │ │ │ + * Get an OL style map location from a 3rd party style map location │ │ │ │ │ + * │ │ │ │ │ + * Parameters │ │ │ │ │ + * moLonLat - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.LonLat>} An OpenLayers.LonLat, translated from the passed in │ │ │ │ │ + * MapObject LonLat │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ + */ │ │ │ │ │ + getOLLonLatFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + var olLonLat = null; │ │ │ │ │ + if (moLonLat != null) { │ │ │ │ │ + var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + olLonLat = new OpenLayers.LonLat(lon, lat); │ │ │ │ │ + } │ │ │ │ │ + return olLonLat; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setImgSrc │ │ │ │ │ - * Sets the source for the tile image │ │ │ │ │ + * Method: getMapObjectLonLatFromOLLonLat │ │ │ │ │ + * Get a 3rd party map location from an OL map location. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * url - {String} │ │ │ │ │ + * olLonLat - {<OpenLayers.LonLat>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} A MapObject LonLat, translated from the passed in │ │ │ │ │ + * OpenLayers.LonLat │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ */ │ │ │ │ │ - setImgSrc: function(url) { │ │ │ │ │ - if (this.useIFrame === true) { │ │ │ │ │ - if (url) { │ │ │ │ │ - var form = this.createRequestForm(); │ │ │ │ │ - this.frame.appendChild(form); │ │ │ │ │ - form.submit(); │ │ │ │ │ - this.frame.removeChild(form); │ │ │ │ │ - } else if (this.imgDiv.parentNode === this.frame) { │ │ │ │ │ - // we don't reuse iframes to avoid caching issues │ │ │ │ │ - this.frame.removeChild(this.imgDiv); │ │ │ │ │ - this.imgDiv = null; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Tile.Image.prototype.setImgSrc.apply(this, arguments); │ │ │ │ │ + getMapObjectLonLatFromOLLonLat: function(olLonLat) { │ │ │ │ │ + var moLatLng = null; │ │ │ │ │ + if (olLonLat != null) { │ │ │ │ │ + moLatLng = this.getMapObjectLonLatFromLonLat(olLonLat.lon, │ │ │ │ │ + olLonLat.lat); │ │ │ │ │ } │ │ │ │ │ + return moLatLng; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + // TRANSLATION: MapObject Pixel <-> OpenLayers.Pixel │ │ │ │ │ + // │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: onImageLoad │ │ │ │ │ - * Handler for the image onload event │ │ │ │ │ + * Method: getOLPixelFromMapObjectPixel │ │ │ │ │ + * Get an OL pixel location from a 3rd party pixel location. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moPixel - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Pixel>} An OpenLayers.Pixel, translated from the passed in │ │ │ │ │ + * MapObject Pixel │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ */ │ │ │ │ │ - onImageLoad: function() { │ │ │ │ │ - //TODO de-uglify opacity handling │ │ │ │ │ - OpenLayers.Tile.Image.prototype.onImageLoad.apply(this, arguments); │ │ │ │ │ - if (this.useIFrame === true) { │ │ │ │ │ - this.imgDiv.style.opacity = 1; │ │ │ │ │ - this.frame.style.opacity = this.layer.opacity; │ │ │ │ │ + getOLPixelFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + var olPixel = null; │ │ │ │ │ + if (moPixel != null) { │ │ │ │ │ + var x = this.getXFromMapObjectPixel(moPixel); │ │ │ │ │ + var y = this.getYFromMapObjectPixel(moPixel); │ │ │ │ │ + olPixel = new OpenLayers.Pixel(x, y); │ │ │ │ │ } │ │ │ │ │ + return olPixel; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createBackBuffer │ │ │ │ │ - * Override createBackBuffer to do nothing when we use an iframe. Moving an │ │ │ │ │ - * iframe from one element to another makes it necessary to reload the iframe │ │ │ │ │ - * because its content is lost. So we just give up. │ │ │ │ │ + * Method: getMapObjectPixelFromOLPixel │ │ │ │ │ + * Get a 3rd party pixel location from an OL pixel location │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * olPixel - {<OpenLayers.Pixel>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * {Object} A MapObject Pixel, translated from the passed in │ │ │ │ │ + * OpenLayers.Pixel │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ */ │ │ │ │ │ - createBackBuffer: function() { │ │ │ │ │ - var backBuffer; │ │ │ │ │ - if (this.useIFrame === false) { │ │ │ │ │ - backBuffer = OpenLayers.Tile.Image.prototype.createBackBuffer.call(this); │ │ │ │ │ + getMapObjectPixelFromOLPixel: function(olPixel) { │ │ │ │ │ + var moPixel = null; │ │ │ │ │ + if (olPixel != null) { │ │ │ │ │ + moPixel = this.getMapObjectPixelFromXY(olPixel.x, olPixel.y); │ │ │ │ │ } │ │ │ │ │ - return backBuffer; │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ + return moPixel; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.EventPane" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/SOS.js │ │ │ │ │ + OpenLayers/Layer/PointTrack.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Protocol.js │ │ │ │ │ + * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Function: OpenLayers.Protocol.SOS │ │ │ │ │ - * Used to create a versioned SOS protocol. Default version is 1.0.0. │ │ │ │ │ + * Class: OpenLayers.Layer.PointTrack │ │ │ │ │ + * Vector layer to display ordered point features as a line, creating one │ │ │ │ │ + * LineString feature for each pair of two points. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol>} An SOS protocol for the given version. │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Vector> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.SOS = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options, OpenLayers.Protocol.SOS.DEFAULTS │ │ │ │ │ - ); │ │ │ │ │ - var cls = OpenLayers.Protocol.SOS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported SOS version: " + options.version; │ │ │ │ │ - } │ │ │ │ │ - return new cls(options); │ │ │ │ │ -}; │ │ │ │ │ +OpenLayers.Layer.PointTrack = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Protocol.SOS.DEFAULTS │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol.SOS.DEFAULTS = { │ │ │ │ │ - "version": "1.0.0" │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/CSW.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: dataFrom │ │ │ │ │ + * {<OpenLayers.Layer.PointTrack.TARGET_NODE>} or │ │ │ │ │ + * {<OpenLayers.Layer.PointTrack.SOURCE_NODE>} optional. If the lines │ │ │ │ │ + * should get the data/attributes from one of the two points it is │ │ │ │ │ + * composed of, which one should it be? │ │ │ │ │ + */ │ │ │ │ │ + dataFrom: null, │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: styleFrom │ │ │ │ │ + * {<OpenLayers.Layer.PointTrack.TARGET_NODE>} or │ │ │ │ │ + * {<OpenLayers.Layer.PointTrack.SOURCE_NODE>} optional. If the lines │ │ │ │ │ + * should get the style from one of the two points it is composed of, │ │ │ │ │ + * which one should it be? │ │ │ │ │ + */ │ │ │ │ │ + styleFrom: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.PointTrack │ │ │ │ │ + * Constructor for a new OpenLayers.PointTrack instance. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} name of the layer │ │ │ │ │ + * options - {Object} Optional object with properties to tag onto the │ │ │ │ │ + * instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: addNodes │ │ │ │ │ + * Adds point features that will be used to create lines from, using point │ │ │ │ │ + * pairs. The first point of a pair will be the source node, the second │ │ │ │ │ + * will be the target node. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pointFeatures - {Array(<OpenLayers.Feature>)} │ │ │ │ │ + * options - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Supported options: │ │ │ │ │ + * silent - {Boolean} true to suppress (before)feature(s)added events │ │ │ │ │ + */ │ │ │ │ │ + addNodes: function(pointFeatures, options) { │ │ │ │ │ + if (pointFeatures.length < 2) { │ │ │ │ │ + throw new Error("At least two point features have to be added to " + │ │ │ │ │ + "create a line from"); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var lines = new Array(pointFeatures.length - 1); │ │ │ │ │ + │ │ │ │ │ + var pointFeature, startPoint, endPoint; │ │ │ │ │ + for (var i = 0, len = pointFeatures.length; i < len; i++) { │ │ │ │ │ + pointFeature = pointFeatures[i]; │ │ │ │ │ + endPoint = pointFeature.geometry; │ │ │ │ │ + │ │ │ │ │ + if (!endPoint) { │ │ │ │ │ + var lonlat = pointFeature.lonlat; │ │ │ │ │ + endPoint = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ + } else if (endPoint.CLASS_NAME != "OpenLayers.Geometry.Point") { │ │ │ │ │ + throw new TypeError("Only features with point geometries are supported."); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (i > 0) { │ │ │ │ │ + var attributes = (this.dataFrom != null) ? │ │ │ │ │ + (pointFeatures[i + this.dataFrom].data || │ │ │ │ │ + pointFeatures[i + this.dataFrom].attributes) : │ │ │ │ │ + null; │ │ │ │ │ + var style = (this.styleFrom != null) ? │ │ │ │ │ + (pointFeatures[i + this.styleFrom].style) : │ │ │ │ │ + null; │ │ │ │ │ + var line = new OpenLayers.Geometry.LineString([startPoint, │ │ │ │ │ + endPoint │ │ │ │ │ + ]); │ │ │ │ │ + │ │ │ │ │ + lines[i - 1] = new OpenLayers.Feature.Vector(line, attributes, │ │ │ │ │ + style); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + startPoint = endPoint; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.addFeatures(lines, options); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.PointTrack" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Protocol.js │ │ │ │ │ + * Constant: OpenLayers.Layer.PointTrack.SOURCE_NODE │ │ │ │ │ + * {Number} value for <OpenLayers.Layer.PointTrack.dataFrom> and │ │ │ │ │ + * <OpenLayers.Layer.PointTrack.styleFrom> │ │ │ │ │ */ │ │ │ │ │ +OpenLayers.Layer.PointTrack.SOURCE_NODE = -1; │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Protocol.CSW │ │ │ │ │ - * Used to create a versioned CSW protocol. Default version is 2.0.2. │ │ │ │ │ + * Constant: OpenLayers.Layer.PointTrack.TARGET_NODE │ │ │ │ │ + * {Number} value for <OpenLayers.Layer.PointTrack.dataFrom> and │ │ │ │ │ + * <OpenLayers.Layer.PointTrack.styleFrom> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.CSW = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options, OpenLayers.Protocol.CSW.DEFAULTS │ │ │ │ │ - ); │ │ │ │ │ - var cls = OpenLayers.Protocol.CSW["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported CSW version: " + options.version; │ │ │ │ │ - } │ │ │ │ │ - return new cls(options); │ │ │ │ │ -}; │ │ │ │ │ +OpenLayers.Layer.PointTrack.TARGET_NODE = 0; │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: OpenLayers.Protocol.CSW.DEFAULTS │ │ │ │ │ + * Constant: OpenLayers.Layer.PointTrack.dataFrom │ │ │ │ │ + * {Object} with the following keys - *deprecated* │ │ │ │ │ + * - SOURCE_NODE: take data/attributes from the source node of the line │ │ │ │ │ + * - TARGET_NODE: take data/attributes from the target node of the line │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.CSW.DEFAULTS = { │ │ │ │ │ - "version": "2.0.2" │ │ │ │ │ +OpenLayers.Layer.PointTrack.dataFrom = { │ │ │ │ │ + 'SOURCE_NODE': -1, │ │ │ │ │ + 'TARGET_NODE': 0 │ │ │ │ │ }; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/WFS.js │ │ │ │ │ + OpenLayers/Layer/FixedZoomLevels.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Protocol.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Protocol.WFS │ │ │ │ │ - * Used to create a versioned WFS protocol. Default version is 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol>} A WFS protocol of the given version. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var protocol = new OpenLayers.Protocol.WFS({ │ │ │ │ │ - * version: "1.1.0", │ │ │ │ │ - * url: "http://demo.opengeo.org/geoserver/wfs", │ │ │ │ │ - * featureType: "tasmania_roads", │ │ │ │ │ - * featureNS: "http://www.openplans.org/topp", │ │ │ │ │ - * geometryName: "the_geom" │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * See the protocols for specific WFS versions for more detail. │ │ │ │ │ + * @requires OpenLayers/Layer.js │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.WFS = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options, OpenLayers.Protocol.WFS.DEFAULTS │ │ │ │ │ - ); │ │ │ │ │ - var cls = OpenLayers.Protocol.WFS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported WFS version: " + options.version; │ │ │ │ │ - } │ │ │ │ │ - return new cls(options); │ │ │ │ │ -}; │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Function: fromWMSLayer │ │ │ │ │ - * Convenience function to create a WFS protocol from a WMS layer. This makes │ │ │ │ │ - * the assumption that a WFS requests can be issued at the same URL as │ │ │ │ │ - * WMS requests and that a WFS featureType exists with the same name as the │ │ │ │ │ - * WMS layer. │ │ │ │ │ - * │ │ │ │ │ - * This function is designed to auto-configure <url>, <featureType>, │ │ │ │ │ - * <featurePrefix> and <srsName> for WFS <version> 1.1.0. Note that │ │ │ │ │ - * srsName matching with the WMS layer will not work with WFS 1.0.0. │ │ │ │ │ + * Class: OpenLayers.Layer.FixedZoomLevels │ │ │ │ │ + * Some Layers will already have established zoom levels (like google │ │ │ │ │ + * or ve). Instead of trying to determine them and populate a resolutions[] │ │ │ │ │ + * Array with those values, we will hijack the resolution functionality │ │ │ │ │ + * here. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.WMS>} WMS layer that has a matching WFS │ │ │ │ │ - * FeatureType at the same server url with the same typename. │ │ │ │ │ - * options - {Object} Default properties to be set on the protocol. │ │ │ │ │ + * When you subclass FixedZoomLevels: │ │ │ │ │ + * │ │ │ │ │ + * The initResolutions() call gets nullified, meaning no resolutions[] array │ │ │ │ │ + * is set up. Which would be a big problem getResolution() in Layer, since │ │ │ │ │ + * it merely takes map.zoom and indexes into resolutions[]... but.... │ │ │ │ │ + * │ │ │ │ │ + * The getResolution() call is also overridden. Instead of using the │ │ │ │ │ + * resolutions[] array, we simply calculate the current resolution based │ │ │ │ │ + * on the current extent and the current map size. But how will we be able │ │ │ │ │ + * to calculate the current extent without knowing the resolution...? │ │ │ │ │ + * │ │ │ │ │ + * The getExtent() function is also overridden. Instead of calculating extent │ │ │ │ │ + * based on the center point and the current resolution, we instead │ │ │ │ │ + * calculate the extent by getting the lonlats at the top-left and │ │ │ │ │ + * bottom-right by using the getLonLatFromViewPortPx() translation function, │ │ │ │ │ + * taken from the pixel locations (0,0) and the size of the map. But how │ │ │ │ │ + * will we be able to do lonlat-px translation without resolution....? │ │ │ │ │ + * │ │ │ │ │ + * The getZoomForResolution() method is overridden. Instead of indexing into │ │ │ │ │ + * the resolutions[] array, we call OpenLayers.Layer.getExent(), passing in │ │ │ │ │ + * the desired resolution. With this extent, we then call getZoomForExtent() │ │ │ │ │ + * │ │ │ │ │ + * │ │ │ │ │ + * Whenever you implement a layer using OpenLayers.Layer.FixedZoomLevels, │ │ │ │ │ + * it is your responsibility to provide the following three functions: │ │ │ │ │ + * │ │ │ │ │ + * - getLonLatFromViewPortPx │ │ │ │ │ + * - getViewPortPxFromLonLat │ │ │ │ │ + * - getZoomForExtent │ │ │ │ │ + * │ │ │ │ │ + * ...those three functions should generally be provided by any reasonable │ │ │ │ │ + * API that you might be working from. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.WFS>} │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.WFS.fromWMSLayer = function(layer, options) { │ │ │ │ │ - var typeName, featurePrefix; │ │ │ │ │ - var param = layer.params["LAYERS"]; │ │ │ │ │ - var parts = (OpenLayers.Util.isArray(param) ? param[0] : param).split(":"); │ │ │ │ │ - if (parts.length > 1) { │ │ │ │ │ - featurePrefix = parts[0]; │ │ │ │ │ - } │ │ │ │ │ - typeName = parts.pop(); │ │ │ │ │ - var protocolOptions = { │ │ │ │ │ - url: layer.url, │ │ │ │ │ - featureType: typeName, │ │ │ │ │ - featurePrefix: featurePrefix, │ │ │ │ │ - srsName: layer.projection && layer.projection.getCode() || │ │ │ │ │ - layer.map && layer.map.getProjectionObject().getCode(), │ │ │ │ │ - version: "1.1.0" │ │ │ │ │ - }; │ │ │ │ │ - return new OpenLayers.Protocol.WFS(OpenLayers.Util.applyDefaults( │ │ │ │ │ - options, protocolOptions │ │ │ │ │ - )); │ │ │ │ │ -}; │ │ │ │ │ +OpenLayers.Layer.FixedZoomLevels = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /********************************************************/ │ │ │ │ │ + /* */ │ │ │ │ │ + /* Baselayer Functions */ │ │ │ │ │ + /* */ │ │ │ │ │ + /* The following functions must all be implemented */ │ │ │ │ │ + /* by all base layers */ │ │ │ │ │ + /* */ │ │ │ │ │ + /********************************************************/ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.FixedZoomLevels │ │ │ │ │ + * Create a new fixed zoom levels layer. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function() { │ │ │ │ │ + //this class is only just to add the following functions... │ │ │ │ │ + // nothing to actually do here... but it is probably a good │ │ │ │ │ + // idea to have layers that use these functions call this │ │ │ │ │ + // inititalize() anyways, in case at some point we decide we │ │ │ │ │ + // do want to put some functionality or state in here. │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: initResolutions │ │ │ │ │ + * Populate the resolutions array │ │ │ │ │ + */ │ │ │ │ │ + initResolutions: function() { │ │ │ │ │ + │ │ │ │ │ + var props = ['minZoomLevel', 'maxZoomLevel', 'numZoomLevels']; │ │ │ │ │ + │ │ │ │ │ + for (var i = 0, len = props.length; i < len; i++) { │ │ │ │ │ + var property = props[i]; │ │ │ │ │ + this[property] = (this.options[property] != null) ? │ │ │ │ │ + this.options[property] : │ │ │ │ │ + this.map[property]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if ((this.minZoomLevel == null) || │ │ │ │ │ + (this.minZoomLevel < this.MIN_ZOOM_LEVEL)) { │ │ │ │ │ + this.minZoomLevel = this.MIN_ZOOM_LEVEL; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + // At this point, we know what the minimum desired zoom level is, and │ │ │ │ │ + // we must calculate the total number of zoom levels. │ │ │ │ │ + // │ │ │ │ │ + // Because we allow for the setting of either the 'numZoomLevels' │ │ │ │ │ + // or the 'maxZoomLevel' properties... on either the layer or the │ │ │ │ │ + // map, we have to define some rules to see which we take into │ │ │ │ │ + // account first in this calculation. │ │ │ │ │ + // │ │ │ │ │ + // The following is the precedence list for these properties: │ │ │ │ │ + // │ │ │ │ │ + // (1) numZoomLevels set on layer │ │ │ │ │ + // (2) maxZoomLevel set on layer │ │ │ │ │ + // (3) numZoomLevels set on map │ │ │ │ │ + // (4) maxZoomLevel set on map* │ │ │ │ │ + // (5) none of the above* │ │ │ │ │ + // │ │ │ │ │ + // *Note that options (4) and (5) are only possible if the user │ │ │ │ │ + // _explicitly_ sets the 'numZoomLevels' property on the map to │ │ │ │ │ + // null, since it is set by default to 16. │ │ │ │ │ + // │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + // Note to future: In 3.0, I think we should remove the default │ │ │ │ │ + // value of 16 for map.numZoomLevels. Rather, I think that value │ │ │ │ │ + // should be set as a default on the Layer.WMS class. If someone │ │ │ │ │ + // creates a 3rd party layer and does not specify any 'minZoomLevel', │ │ │ │ │ + // 'maxZoomLevel', or 'numZoomLevels', and has not explicitly │ │ │ │ │ + // specified any of those on the map object either.. then I think │ │ │ │ │ + // it is fair to say that s/he wants all the zoom levels available. │ │ │ │ │ + // │ │ │ │ │ + // By making map.numZoomLevels *null* by default, that will be the │ │ │ │ │ + // case. As it is, I don't feel comfortable changing that right now │ │ │ │ │ + // as it would be a glaring API change and actually would probably │ │ │ │ │ + // break many peoples' codes. │ │ │ │ │ + // │ │ │ │ │ + │ │ │ │ │ + //the number of zoom levels we'd like to have. │ │ │ │ │ + var desiredZoomLevels; │ │ │ │ │ + │ │ │ │ │ + //this is the maximum number of zoom levels the layer will allow, │ │ │ │ │ + // given the specified starting minimum zoom level. │ │ │ │ │ + var limitZoomLevels = this.MAX_ZOOM_LEVEL - this.minZoomLevel + 1; │ │ │ │ │ + │ │ │ │ │ + if (((this.options.numZoomLevels == null) && │ │ │ │ │ + (this.options.maxZoomLevel != null)) // (2) │ │ │ │ │ + || │ │ │ │ │ + ((this.numZoomLevels == null) && │ │ │ │ │ + (this.maxZoomLevel != null)) // (4) │ │ │ │ │ + ) { │ │ │ │ │ + //calculate based on specified maxZoomLevel (on layer or map) │ │ │ │ │ + desiredZoomLevels = this.maxZoomLevel - this.minZoomLevel + 1; │ │ │ │ │ + } else { │ │ │ │ │ + //calculate based on specified numZoomLevels (on layer or map) │ │ │ │ │ + // this covers cases (1) and (3) │ │ │ │ │ + desiredZoomLevels = this.numZoomLevels; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (desiredZoomLevels != null) { │ │ │ │ │ + //Now that we know what we would *like* the number of zoom levels │ │ │ │ │ + // to be, based on layer or map options, we have to make sure that │ │ │ │ │ + // it does not conflict with the actual limit, as specified by │ │ │ │ │ + // the constants on the layer itself (and calculated into the │ │ │ │ │ + // 'limitZoomLevels' variable). │ │ │ │ │ + this.numZoomLevels = Math.min(desiredZoomLevels, limitZoomLevels); │ │ │ │ │ + } else { │ │ │ │ │ + // case (5) -- neither 'numZoomLevels' not 'maxZoomLevel' was │ │ │ │ │ + // set on either the layer or the map. So we just use the │ │ │ │ │ + // maximum limit as calculated by the layer's constants. │ │ │ │ │ + this.numZoomLevels = limitZoomLevels; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //now that the 'numZoomLevels' is appropriately, safely set, │ │ │ │ │ + // we go back and re-calculate the 'maxZoomLevel'. │ │ │ │ │ + this.maxZoomLevel = this.minZoomLevel + this.numZoomLevels - 1; │ │ │ │ │ + │ │ │ │ │ + if (this.RESOLUTIONS != null) { │ │ │ │ │ + var resolutionsIndex = 0; │ │ │ │ │ + this.resolutions = []; │ │ │ │ │ + for (var i = this.minZoomLevel; i <= this.maxZoomLevel; i++) { │ │ │ │ │ + this.resolutions[resolutionsIndex++] = this.RESOLUTIONS[i]; │ │ │ │ │ + } │ │ │ │ │ + this.maxResolution = this.resolutions[0]; │ │ │ │ │ + this.minResolution = this.resolutions[this.resolutions.length - 1]; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getResolution │ │ │ │ │ + * Get the current map resolution │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} Map units per Pixel │ │ │ │ │ + */ │ │ │ │ │ + getResolution: function() { │ │ │ │ │ + │ │ │ │ │ + if (this.resolutions != null) { │ │ │ │ │ + return OpenLayers.Layer.prototype.getResolution.apply(this, arguments); │ │ │ │ │ + } else { │ │ │ │ │ + var resolution = null; │ │ │ │ │ + │ │ │ │ │ + var viewSize = this.map.getSize(); │ │ │ │ │ + var extent = this.getExtent(); │ │ │ │ │ + │ │ │ │ │ + if ((viewSize != null) && (extent != null)) { │ │ │ │ │ + resolution = Math.max(extent.getWidth() / viewSize.w, │ │ │ │ │ + extent.getHeight() / viewSize.h); │ │ │ │ │ + } │ │ │ │ │ + return resolution; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getExtent │ │ │ │ │ + * Calculates using px-> lonlat translation functions on tl and br │ │ │ │ │ + * corners of viewport │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} A Bounds object which represents the lon/lat │ │ │ │ │ + * bounds of the current viewPort. │ │ │ │ │ + */ │ │ │ │ │ + getExtent: function() { │ │ │ │ │ + var size = this.map.getSize(); │ │ │ │ │ + var tl = this.getLonLatFromViewPortPx({ │ │ │ │ │ + x: 0, │ │ │ │ │ + y: 0 │ │ │ │ │ + }); │ │ │ │ │ + var br = this.getLonLatFromViewPortPx({ │ │ │ │ │ + x: size.w, │ │ │ │ │ + y: size.h │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + if ((tl != null) && (br != null)) { │ │ │ │ │ + return new OpenLayers.Bounds(tl.lon, br.lat, br.lon, tl.lat); │ │ │ │ │ + } else { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getZoomForResolution │ │ │ │ │ + * Get the zoom level for a given resolution │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resolution - {Float} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} A suitable zoom level for the specified resolution. │ │ │ │ │ + * If no baselayer is set, returns null. │ │ │ │ │ + */ │ │ │ │ │ + getZoomForResolution: function(resolution) { │ │ │ │ │ + │ │ │ │ │ + if (this.resolutions != null) { │ │ │ │ │ + return OpenLayers.Layer.prototype.getZoomForResolution.apply(this, arguments); │ │ │ │ │ + } else { │ │ │ │ │ + var extent = OpenLayers.Layer.prototype.getExtent.apply(this, []); │ │ │ │ │ + return this.getZoomForExtent(extent); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /********************************************************/ │ │ │ │ │ + /* */ │ │ │ │ │ + /* Translation Functions */ │ │ │ │ │ + /* */ │ │ │ │ │ + /* The following functions translate GMaps and OL */ │ │ │ │ │ + /* formats for Pixel, LonLat, Bounds, and Zoom */ │ │ │ │ │ + /* */ │ │ │ │ │ + /********************************************************/ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + // TRANSLATION: MapObject Zoom <-> OpenLayers Zoom │ │ │ │ │ + // │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getOLZoomFromMapObjectZoom │ │ │ │ │ + * Get the OL zoom index from the map object zoom level │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moZoom - {Integer} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} An OpenLayers Zoom level, translated from the passed in zoom │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ + */ │ │ │ │ │ + getOLZoomFromMapObjectZoom: function(moZoom) { │ │ │ │ │ + var zoom = null; │ │ │ │ │ + if (moZoom != null) { │ │ │ │ │ + zoom = moZoom - this.minZoomLevel; │ │ │ │ │ + if (this.map.baseLayer !== this) { │ │ │ │ │ + zoom = this.map.baseLayer.getZoomForResolution( │ │ │ │ │ + this.getResolutionForZoom(zoom) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return zoom; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getMapObjectZoomFromOLZoom │ │ │ │ │ + * Get the map object zoom level from the OL zoom level │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * olZoom - {Integer} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} A MapObject level, translated from the passed in olZoom │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ + */ │ │ │ │ │ + getMapObjectZoomFromOLZoom: function(olZoom) { │ │ │ │ │ + var zoom = null; │ │ │ │ │ + if (olZoom != null) { │ │ │ │ │ + zoom = olZoom + this.minZoomLevel; │ │ │ │ │ + if (this.map.baseLayer !== this) { │ │ │ │ │ + zoom = this.getZoomForResolution( │ │ │ │ │ + this.map.baseLayer.getResolutionForZoom(zoom) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return zoom; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.FixedZoomLevels" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Protocol.WFS.DEFAULTS │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol.WFS.DEFAULTS = { │ │ │ │ │ - "version": "1.0.0" │ │ │ │ │ -}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/HTTP.js │ │ │ │ │ + OpenLayers/Layer/Zoomify.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Protocol.js │ │ │ │ │ - * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ +/* │ │ │ │ │ + * Development supported by a R&D grant DC08P02OUK006 - Old Maps Online │ │ │ │ │ + * (www.oldmapsonline.org) from Ministry of Culture of the Czech Republic. │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * if application uses the query string, for example, for BBOX parameters, │ │ │ │ │ - * OpenLayers/Format/QueryStringFilter.js should be included in the build config file │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Protocol.HTTP │ │ │ │ │ - * A basic HTTP protocol for vector layers. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Protocol.HTTP> constructor. │ │ │ │ │ + * Class: OpenLayers.Layer.Zoomify │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol> │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.HTTP = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ +OpenLayers.Layer.Zoomify = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: url │ │ │ │ │ - * {String} Service URL, read-only, set through the options │ │ │ │ │ - * passed to constructor. │ │ │ │ │ + * Property: size │ │ │ │ │ + * {<OpenLayers.Size>} The Zoomify image size in pixels. │ │ │ │ │ */ │ │ │ │ │ - url: null, │ │ │ │ │ + size: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: headers │ │ │ │ │ - * {Object} HTTP request headers, read-only, set through the options │ │ │ │ │ - * passed to the constructor, │ │ │ │ │ - * Example: {'Content-Type': 'plain/text'} │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - headers: null, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: params │ │ │ │ │ - * {Object} Parameters of GET requests, read-only, set through the options │ │ │ │ │ - * passed to the constructor, │ │ │ │ │ - * Example: {'bbox': '5,5,5,5'} │ │ │ │ │ + * Property: standardTileSize │ │ │ │ │ + * {Integer} The size of a standard (non-border) square tile in pixels. │ │ │ │ │ */ │ │ │ │ │ - params: null, │ │ │ │ │ + standardTileSize: 256, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: callback │ │ │ │ │ - * {Object} Function to be called when the <read>, <create>, │ │ │ │ │ - * <update>, <delete> or <commit> operation completes, read-only, │ │ │ │ │ - * set through the options passed to the constructor. │ │ │ │ │ - */ │ │ │ │ │ - callback: null, │ │ │ │ │ + /** │ │ │ │ │ + * Property: tileOriginCorner │ │ │ │ │ + * {String} This layer uses top-left as tile origin │ │ │ │ │ + **/ │ │ │ │ │ + tileOriginCorner: "tl", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: scope │ │ │ │ │ - * {Object} Callback execution scope, read-only, set through the │ │ │ │ │ - * options passed to the constructor. │ │ │ │ │ + * Property: numberOfTiers │ │ │ │ │ + * {Integer} Depth of the Zoomify pyramid, number of tiers (zoom levels) │ │ │ │ │ + * - filled during Zoomify pyramid initialization. │ │ │ │ │ */ │ │ │ │ │ - scope: null, │ │ │ │ │ + numberOfTiers: 0, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: readWithPOST │ │ │ │ │ - * {Boolean} true if read operations are done with POST requests │ │ │ │ │ - * instead of GET, defaults to false. │ │ │ │ │ + * Property: tileCountUpToTier │ │ │ │ │ + * {Array(Integer)} Number of tiles up to the given tier of pyramid. │ │ │ │ │ + * - filled during Zoomify pyramid initialization. │ │ │ │ │ */ │ │ │ │ │ - readWithPOST: false, │ │ │ │ │ + tileCountUpToTier: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: updateWithPOST │ │ │ │ │ - * {Boolean} true if update operations are done with POST requests │ │ │ │ │ - * defaults to false. │ │ │ │ │ + * Property: tierSizeInTiles │ │ │ │ │ + * {Array(<OpenLayers.Size>)} Size (in tiles) for each tier of pyramid. │ │ │ │ │ + * - filled during Zoomify pyramid initialization. │ │ │ │ │ */ │ │ │ │ │ - updateWithPOST: false, │ │ │ │ │ + tierSizeInTiles: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: deleteWithPOST │ │ │ │ │ - * {Boolean} true if delete operations are done with POST requests │ │ │ │ │ - * defaults to false. │ │ │ │ │ - * if true, POST data is set to output of format.write(). │ │ │ │ │ + * Property: tierImageSize │ │ │ │ │ + * {Array(<OpenLayers.Size>)} Image size in pixels for each pyramid tier. │ │ │ │ │ + * - filled during Zoomify pyramid initialization. │ │ │ │ │ */ │ │ │ │ │ - deleteWithPOST: false, │ │ │ │ │ + tierImageSize: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: wildcarded. │ │ │ │ │ - * {Boolean} If true percent signs are added around values │ │ │ │ │ - * read from LIKE filters, for example if the protocol │ │ │ │ │ - * read method is passed a LIKE filter whose property │ │ │ │ │ - * is "foo" and whose value is "bar" the string │ │ │ │ │ - * "foo__ilike=%bar%" will be sent in the query string; │ │ │ │ │ - * defaults to false. │ │ │ │ │ + * Constructor: OpenLayers.Layer.Zoomify │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} A name for the layer. │ │ │ │ │ + * url - {String} - Relative or absolute path to the image or more │ │ │ │ │ + * precisly to the TileGroup[X] directories root. │ │ │ │ │ + * Flash plugin use the variable name "zoomifyImagePath" for this. │ │ │ │ │ + * size - {<OpenLayers.Size>} The size (in pixels) of the image. │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ */ │ │ │ │ │ - wildcarded: false, │ │ │ │ │ + initialize: function(name, url, size, options) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: srsInBBOX │ │ │ │ │ - * {Boolean} Include the SRS identifier in BBOX query string parameter. │ │ │ │ │ - * Default is false. If true and the layer has a projection object set, │ │ │ │ │ - * any BBOX filter will be serialized with a fifth item identifying the │ │ │ │ │ - * projection. E.g. bbox=-1000,-1000,1000,1000,EPSG:900913 │ │ │ │ │ - */ │ │ │ │ │ - srsInBBOX: false, │ │ │ │ │ + // initilize the Zoomify pyramid for given size │ │ │ │ │ + this.initializeZoomify(size); │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [ │ │ │ │ │ + name, url, size, {}, │ │ │ │ │ + options │ │ │ │ │ + ]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.HTTP │ │ │ │ │ - * A class for giving layers generic HTTP protocol. │ │ │ │ │ + * Method: initializeZoomify │ │ │ │ │ + * It generates constants for all tiers of the Zoomify pyramid │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * size - {<OpenLayers.Size>} The size of the image in pixels │ │ │ │ │ * │ │ │ │ │ - * Valid options include: │ │ │ │ │ - * url - {String} │ │ │ │ │ - * headers - {Object} │ │ │ │ │ - * params - {Object} URL parameters for GET requests │ │ │ │ │ - * format - {<OpenLayers.Format>} │ │ │ │ │ - * callback - {Function} │ │ │ │ │ - * scope - {Object} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - this.params = {}; │ │ │ │ │ - this.headers = {}; │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ + initializeZoomify: function(size) { │ │ │ │ │ │ │ │ │ │ - if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ - var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ - wildcarded: this.wildcarded, │ │ │ │ │ - srsInBBOX: this.srsInBBOX │ │ │ │ │ - }); │ │ │ │ │ - this.filterToParams = function(filter, params) { │ │ │ │ │ - return format.write(filter, params); │ │ │ │ │ - }; │ │ │ │ │ + var imageSize = size.clone(); │ │ │ │ │ + this.size = size.clone(); │ │ │ │ │ + var tiles = new OpenLayers.Size( │ │ │ │ │ + Math.ceil(imageSize.w / this.standardTileSize), │ │ │ │ │ + Math.ceil(imageSize.h / this.standardTileSize) │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + this.tierSizeInTiles = [tiles]; │ │ │ │ │ + this.tierImageSize = [imageSize]; │ │ │ │ │ + │ │ │ │ │ + while (imageSize.w > this.standardTileSize || │ │ │ │ │ + imageSize.h > this.standardTileSize) { │ │ │ │ │ + │ │ │ │ │ + imageSize = new OpenLayers.Size( │ │ │ │ │ + Math.floor(imageSize.w / 2), │ │ │ │ │ + Math.floor(imageSize.h / 2) │ │ │ │ │ + ); │ │ │ │ │ + tiles = new OpenLayers.Size( │ │ │ │ │ + Math.ceil(imageSize.w / this.standardTileSize), │ │ │ │ │ + Math.ceil(imageSize.h / this.standardTileSize) │ │ │ │ │ + ); │ │ │ │ │ + this.tierSizeInTiles.push(tiles); │ │ │ │ │ + this.tierImageSize.push(imageSize); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.tierSizeInTiles.reverse(); │ │ │ │ │ + this.tierImageSize.reverse(); │ │ │ │ │ + │ │ │ │ │ + this.numberOfTiers = this.tierSizeInTiles.length; │ │ │ │ │ + var resolutions = [1]; │ │ │ │ │ + this.tileCountUpToTier = [0]; │ │ │ │ │ + for (var i = 1; i < this.numberOfTiers; i++) { │ │ │ │ │ + resolutions.unshift(Math.pow(2, i)); │ │ │ │ │ + this.tileCountUpToTier.push( │ │ │ │ │ + this.tierSizeInTiles[i - 1].w * this.tierSizeInTiles[i - 1].h + │ │ │ │ │ + this.tileCountUpToTier[i - 1] │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (!this.serverResolutions) { │ │ │ │ │ + this.serverResolutions = resolutions; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up the protocol. │ │ │ │ │ + * APIMethod:destroy │ │ │ │ │ */ │ │ │ │ │ destroy: function() { │ │ │ │ │ - this.params = null; │ │ │ │ │ - this.headers = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ + // for now, nothing special to do here. │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.destroy.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + // Remove from memory the Zoomify pyramid - is that enough? │ │ │ │ │ + this.tileCountUpToTier.length = 0; │ │ │ │ │ + this.tierSizeInTiles.length = 0; │ │ │ │ │ + this.tierImageSize.length = 0; │ │ │ │ │ + │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: filterToParams │ │ │ │ │ - * Optional method to translate an <OpenLayers.Filter> object into an object │ │ │ │ │ - * that can be serialized as request query string provided. If a custom │ │ │ │ │ - * method is not provided, the filter will be serialized using the │ │ │ │ │ - * <OpenLayers.Format.QueryStringFilter> class. │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} filter to convert. │ │ │ │ │ - * params - {Object} The parameters object. │ │ │ │ │ + * obj - {Object} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} The resulting parameters object. │ │ │ │ │ + * {<OpenLayers.Layer.Zoomify>} An exact clone of this <OpenLayers.Layer.Zoomify> │ │ │ │ │ */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Zoomify(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.size, │ │ │ │ │ + this.options); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Construct a request for reading new features. │ │ │ │ │ + * Method: getURL │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * url - {String} Url for the request. │ │ │ │ │ - * params - {Object} Parameters to get serialized as a query string. │ │ │ │ │ - * headers - {Object} Headers to be set on the request. │ │ │ │ │ - * filter - {<OpenLayers.Filter>} Filter to get serialized as a │ │ │ │ │ - * query string. │ │ │ │ │ - * readWithPOST - {Boolean} If the request should be done with POST. │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} A response object, whose "priv" property │ │ │ │ │ - * references the HTTP request, this object is also passed to the │ │ │ │ │ - * callback function when the request completes, its "features" property │ │ │ │ │ - * is then populated with the features received from the server. │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as │ │ │ │ │ + * parameters │ │ │ │ │ */ │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options.params, this.options.params); │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - if (options.filter && this.filterToParams) { │ │ │ │ │ - options.params = this.filterToParams( │ │ │ │ │ - options.filter, options.params │ │ │ │ │ - ); │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((this.tileOrigin.lat - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getZoomForResolution(res); │ │ │ │ │ + │ │ │ │ │ + var tileIndex = x + y * this.tierSizeInTiles[z].w + this.tileCountUpToTier[z]; │ │ │ │ │ + var path = "TileGroup" + Math.floor((tileIndex) / 256) + │ │ │ │ │ + "/" + z + "-" + x + "-" + y + ".jpg"; │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(path, url); │ │ │ │ │ } │ │ │ │ │ - var readWithPOST = (options.readWithPOST !== undefined) ? │ │ │ │ │ - options.readWithPOST : this.readWithPOST; │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - if (readWithPOST) { │ │ │ │ │ - var headers = options.headers || {}; │ │ │ │ │ - headers["Content-Type"] = "application/x-www-form-urlencoded"; │ │ │ │ │ - resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ - data: OpenLayers.Util.getParameterString(options.params), │ │ │ │ │ - headers: headers │ │ │ │ │ - }); │ │ │ │ │ + return url + path; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getImageSize │ │ │ │ │ + * getImageSize returns size for a particular tile. If bounds are given as │ │ │ │ │ + * first argument, size is calculated (bottom-right tiles are non square). │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + getImageSize: function() { │ │ │ │ │ + if (arguments.length > 0) { │ │ │ │ │ + var bounds = this.adjustBounds(arguments[0]); │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((this.tileOrigin.lat - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getZoomForResolution(res); │ │ │ │ │ + var w = this.standardTileSize; │ │ │ │ │ + var h = this.standardTileSize; │ │ │ │ │ + if (x == this.tierSizeInTiles[z].w - 1) { │ │ │ │ │ + var w = this.tierImageSize[z].w % this.standardTileSize; │ │ │ │ │ + } │ │ │ │ │ + if (y == this.tierSizeInTiles[z].h - 1) { │ │ │ │ │ + var h = this.tierImageSize[z].h % this.standardTileSize; │ │ │ │ │ + } │ │ │ │ │ + return (new OpenLayers.Size(w, h)); │ │ │ │ │ } else { │ │ │ │ │ - resp.priv = OpenLayers.Request.GET({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers │ │ │ │ │ - }); │ │ │ │ │ + return this.tileSize; │ │ │ │ │ } │ │ │ │ │ - return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleRead │ │ │ │ │ - * Individual callbacks are created for read, create and update, should │ │ │ │ │ - * a subclass need to override each one separately. │ │ │ │ │ + * APIMethod: setMap │ │ │ │ │ + * When the layer is added to a map, then we can fetch our origin │ │ │ │ │ + * (if we don't have one.) │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * the user callback. │ │ │ │ │ - * options - {Object} The user options passed to the read call. │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - handleRead: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options); │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.map.maxExtent.left, │ │ │ │ │ + this.map.maxExtent.top); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Zoomify" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/Boxes.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer.js │ │ │ │ │ + * @requires OpenLayers/Layer/Markers.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.Boxes │ │ │ │ │ + * Draw divs as 'boxes' on the layer. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Markers> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Boxes = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: create │ │ │ │ │ - * Construct a request for writing newly created features. │ │ │ │ │ + * Constructor: OpenLayers.Layer.Boxes │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, whose "priv" property references the HTTP request, this │ │ │ │ │ - * object is also passed to the callback function when the request │ │ │ │ │ - * completes, its "features" property is then populated with the │ │ │ │ │ - * the features received from the server. │ │ │ │ │ + * name - {String} │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ */ │ │ │ │ │ - create: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: features, │ │ │ │ │ - requestType: "create" │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawMarker │ │ │ │ │ + * Calculate the pixel location for the marker, create it, and │ │ │ │ │ + * add it to the layer's div │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * marker - {<OpenLayers.Marker.Box>} │ │ │ │ │ + */ │ │ │ │ │ + drawMarker: function(marker) { │ │ │ │ │ + var topleft = this.map.getLayerPxFromLonLat({ │ │ │ │ │ + lon: marker.bounds.left, │ │ │ │ │ + lat: marker.bounds.top │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ - resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleCreate, resp, options), │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(features) │ │ │ │ │ + var botright = this.map.getLayerPxFromLonLat({ │ │ │ │ │ + lon: marker.bounds.right, │ │ │ │ │ + lat: marker.bounds.bottom │ │ │ │ │ }); │ │ │ │ │ + if (botright == null || topleft == null) { │ │ │ │ │ + marker.display(false); │ │ │ │ │ + } else { │ │ │ │ │ + var markerDiv = marker.draw(topleft, { │ │ │ │ │ + w: Math.max(1, botright.x - topleft.x), │ │ │ │ │ + h: Math.max(1, botright.y - topleft.y) │ │ │ │ │ + }); │ │ │ │ │ + if (!marker.drawn) { │ │ │ │ │ + this.div.appendChild(markerDiv); │ │ │ │ │ + marker.drawn = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return resp; │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: removeMarker │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * marker - {<OpenLayers.Marker.Box>} │ │ │ │ │ + */ │ │ │ │ │ + removeMarker: function(marker) { │ │ │ │ │ + OpenLayers.Util.removeItem(this.markers, marker); │ │ │ │ │ + if ((marker.div != null) && │ │ │ │ │ + (marker.div.parentNode == this.div)) { │ │ │ │ │ + this.div.removeChild(marker.div); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Boxes" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/Bing.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.Bing │ │ │ │ │ + * Bing layer using direct tile access as provided by Bing Maps REST Services. │ │ │ │ │ + * See http://msdn.microsoft.com/en-us/library/ff701713.aspx for more │ │ │ │ │ + * information. Note: Terms of Service compliant use requires the map to be │ │ │ │ │ + * configured with an <OpenLayers.Control.Attribution> control and the │ │ │ │ │ + * attribution placed on or near the map. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.XYZ> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Bing = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: handleCreate │ │ │ │ │ - * Called the the request issued by <create> is complete. May be overridden │ │ │ │ │ - * by subclasses. │ │ │ │ │ + * Property: key │ │ │ │ │ + * {String} API key for Bing maps, get your own key │ │ │ │ │ + * at http://bingmapsportal.com/ . │ │ │ │ │ + */ │ │ │ │ │ + key: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: serverResolutions │ │ │ │ │ + * {Array} the resolutions provided by the Bing servers. │ │ │ │ │ + */ │ │ │ │ │ + serverResolutions: [ │ │ │ │ │ + 156543.03390625, 78271.516953125, 39135.7584765625, │ │ │ │ │ + 19567.87923828125, 9783.939619140625, 4891.9698095703125, │ │ │ │ │ + 2445.9849047851562, 1222.9924523925781, 611.4962261962891, │ │ │ │ │ + 305.74811309814453, 152.87405654907226, 76.43702827453613, │ │ │ │ │ + 38.218514137268066, 19.109257068634033, 9.554628534317017, │ │ │ │ │ + 4.777314267158508, 2.388657133579254, 1.194328566789627, │ │ │ │ │ + 0.5971642833948135, 0.29858214169740677, 0.14929107084870338, │ │ │ │ │ + 0.07464553542435169 │ │ │ │ │ + ], │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: attributionTemplate │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + attributionTemplate: '<span class="olBingAttribution ${type}">' + │ │ │ │ │ + '<div><a target="_blank" href="http://www.bing.com/maps/">' + │ │ │ │ │ + '<img src="${logo}" /></a></div>${copyrights}' + │ │ │ │ │ + '<a style="white-space: nowrap" target="_blank" ' + │ │ │ │ │ + 'href="http://www.microsoft.com/maps/product/terms.html">' + │ │ │ │ │ + 'Terms of Use</a></span>', │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: metadata │ │ │ │ │ + * {Object} Metadata for this layer, as returned by the callback script │ │ │ │ │ + */ │ │ │ │ │ + metadata: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: protocolRegex │ │ │ │ │ + * {RegExp} Regular expression to match and replace http: in bing urls │ │ │ │ │ + */ │ │ │ │ │ + protocolRegex: /^http:/i, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: type │ │ │ │ │ + * {String} The layer identifier. Any non-birdseye imageryType │ │ │ │ │ + * from http://msdn.microsoft.com/en-us/library/ff701716.aspx can be │ │ │ │ │ + * used. Default is "Road". │ │ │ │ │ + */ │ │ │ │ │ + type: "Road", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: culture │ │ │ │ │ + * {String} The culture identifier. See http://msdn.microsoft.com/en-us/library/ff701709.aspx │ │ │ │ │ + * for the definition and the possible values. Default is "en-US". │ │ │ │ │ + */ │ │ │ │ │ + culture: "en-US", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: metadataParams │ │ │ │ │ + * {Object} Optional url parameters for the Get Imagery Metadata request │ │ │ │ │ + * as described here: http://msdn.microsoft.com/en-us/library/ff701716.aspx │ │ │ │ │ + */ │ │ │ │ │ + metadataParams: null, │ │ │ │ │ + │ │ │ │ │ + /** APIProperty: tileOptions │ │ │ │ │ + * {Object} optional configuration options for <OpenLayers.Tile> instances │ │ │ │ │ + * created by this Layer. Default is │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * {crossOriginKeyword: 'anonymous'} │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ + tileOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** APIProperty: protocol │ │ │ │ │ + * {String} Protocol to use to fetch Imagery Metadata, tiles and bing logo │ │ │ │ │ + * Can be 'http:' 'https:' or '' │ │ │ │ │ + * │ │ │ │ │ + * Warning: tiles may not be available under both HTTP and HTTPS protocols. │ │ │ │ │ + * Microsoft approved use of both HTTP and HTTPS urls for tiles. However │ │ │ │ │ + * this is undocumented and the Imagery Metadata API always returns HTTP │ │ │ │ │ + * urls. │ │ │ │ │ + * │ │ │ │ │ + * Default is '', unless when executed from a file:/// uri, in which case │ │ │ │ │ + * it is 'http:'. │ │ │ │ │ + */ │ │ │ │ │ + protocol: ~window.location.href.indexOf('http') ? '' : 'http:', │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.Bing │ │ │ │ │ + * Create a new Bing layer. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var road = new OpenLayers.Layer.Bing({ │ │ │ │ │ + * name: "My Bing Aerial Layer", │ │ │ │ │ + * type: "Aerial", │ │ │ │ │ + * key: "my-api-key-here", │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the create call. │ │ │ │ │ + * options - {Object} Configuration properties for the layer. │ │ │ │ │ + * │ │ │ │ │ + * Required configuration properties: │ │ │ │ │ + * key - {String} Bing Maps API key for your application. Get one at │ │ │ │ │ + * http://bingmapsportal.com/. │ │ │ │ │ + * type - {String} The layer identifier. Any non-birdseye imageryType │ │ │ │ │ + * from http://msdn.microsoft.com/en-us/library/ff701716.aspx can be │ │ │ │ │ + * used. │ │ │ │ │ + * │ │ │ │ │ + * Any other documented layer properties can be provided in the config object. │ │ │ │ │ */ │ │ │ │ │ - handleCreate: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options); │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + sphericalMercator: true │ │ │ │ │ + }, options); │ │ │ │ │ + var name = options.name || "Bing " + (options.type || this.type); │ │ │ │ │ + │ │ │ │ │ + var newArgs = [name, null, options]; │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, newArgs); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + crossOriginKeyword: 'anonymous' │ │ │ │ │ + }, this.options.tileOptions); │ │ │ │ │ + this.loadMetadata(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: update │ │ │ │ │ - * Construct a request updating modified feature. │ │ │ │ │ + * Method: loadMetadata │ │ │ │ │ + */ │ │ │ │ │ + loadMetadata: function() { │ │ │ │ │ + this._callbackId = "_callback_" + this.id.replace(/\./g, "_"); │ │ │ │ │ + // link the processMetadata method to the global scope and bind it │ │ │ │ │ + // to this instance │ │ │ │ │ + window[this._callbackId] = OpenLayers.Function.bind( │ │ │ │ │ + OpenLayers.Layer.Bing.processMetadata, this │ │ │ │ │ + ); │ │ │ │ │ + var params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + key: this.key, │ │ │ │ │ + jsonp: this._callbackId, │ │ │ │ │ + include: "ImageryProviders" │ │ │ │ │ + }, this.metadataParams); │ │ │ │ │ + var url = this.protocol + "//dev.virtualearth.net/REST/v1/Imagery/Metadata/" + │ │ │ │ │ + this.type + "?" + OpenLayers.Util.getParameterString(params); │ │ │ │ │ + var script = document.createElement("script"); │ │ │ │ │ + script.type = "text/javascript"; │ │ │ │ │ + script.src = url; │ │ │ │ │ + script.id = this._callbackId; │ │ │ │ │ + document.getElementsByTagName("head")[0].appendChild(script); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: initLayer │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ + * Sets layer properties according to the metadata provided by the API │ │ │ │ │ + */ │ │ │ │ │ + initLayer: function() { │ │ │ │ │ + var res = this.metadata.resourceSets[0].resources[0]; │ │ │ │ │ + var url = res.imageUrl.replace("{quadkey}", "${quadkey}"); │ │ │ │ │ + url = url.replace("{culture}", this.culture); │ │ │ │ │ + url = url.replace(this.protocolRegex, this.protocol); │ │ │ │ │ + this.url = []; │ │ │ │ │ + for (var i = 0; i < res.imageUrlSubdomains.length; ++i) { │ │ │ │ │ + this.url.push(url.replace("{subdomain}", res.imageUrlSubdomains[i])); │ │ │ │ │ + } │ │ │ │ │ + this.addOptions({ │ │ │ │ │ + maxResolution: Math.min( │ │ │ │ │ + this.serverResolutions[res.zoomMin], │ │ │ │ │ + this.maxResolution || Number.POSITIVE_INFINITY │ │ │ │ │ + ), │ │ │ │ │ + numZoomLevels: Math.min( │ │ │ │ │ + res.zoomMax + 1 - res.zoomMin, this.numZoomLevels │ │ │ │ │ + ) │ │ │ │ │ + }, true); │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + this.redraw(); │ │ │ │ │ + } │ │ │ │ │ + this.updateAttribution(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, whose "priv" property references the HTTP request, this │ │ │ │ │ - * object is also passed to the callback function when the request │ │ │ │ │ - * completes, its "features" property is then populated with the │ │ │ │ │ - * the feature received from the server. │ │ │ │ │ + * Paramters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ */ │ │ │ │ │ - update: function(feature, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var url = options.url || │ │ │ │ │ - feature.url || │ │ │ │ │ - this.options.url + "/" + feature.fid; │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + if (!this.url) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var xyz = this.getXYZ(bounds), │ │ │ │ │ + x = xyz.x, │ │ │ │ │ + y = xyz.y, │ │ │ │ │ + z = xyz.z; │ │ │ │ │ + var quadDigits = []; │ │ │ │ │ + for (var i = z; i > 0; --i) { │ │ │ │ │ + var digit = '0'; │ │ │ │ │ + var mask = 1 << (i - 1); │ │ │ │ │ + if ((x & mask) != 0) { │ │ │ │ │ + digit++; │ │ │ │ │ + } │ │ │ │ │ + if ((y & mask) != 0) { │ │ │ │ │ + digit++; │ │ │ │ │ + digit++; │ │ │ │ │ + } │ │ │ │ │ + quadDigits.push(digit); │ │ │ │ │ + } │ │ │ │ │ + var quadKey = quadDigits.join(""); │ │ │ │ │ + var url = this.selectUrl('' + x + y + z, this.url); │ │ │ │ │ │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: feature, │ │ │ │ │ - requestType: "update" │ │ │ │ │ + return OpenLayers.String.format(url, { │ │ │ │ │ + 'quadkey': quadKey │ │ │ │ │ }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var method = this.updateWithPOST ? "POST" : "PUT"; │ │ │ │ │ - resp.priv = OpenLayers.Request[method]({ │ │ │ │ │ - url: url, │ │ │ │ │ - callback: this.createCallback(this.handleUpdate, resp, options), │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(feature) │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateAttribution │ │ │ │ │ + * Updates the attribution according to the requirements outlined in │ │ │ │ │ + * http://gis.638310.n2.nabble.com/Bing-imagery-td5789168.html │ │ │ │ │ + */ │ │ │ │ │ + updateAttribution: function() { │ │ │ │ │ + var metadata = this.metadata; │ │ │ │ │ + if (!metadata.resourceSets || !this.map || !this.map.center) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var res = metadata.resourceSets[0].resources[0]; │ │ │ │ │ + var extent = this.map.getExtent().transform( │ │ │ │ │ + this.map.getProjectionObject(), │ │ │ │ │ + new OpenLayers.Projection("EPSG:4326") │ │ │ │ │ + ); │ │ │ │ │ + var providers = res.imageryProviders || [], │ │ │ │ │ + zoom = OpenLayers.Util.indexOf(this.serverResolutions, │ │ │ │ │ + this.getServerResolution()), │ │ │ │ │ + copyrights = "", │ │ │ │ │ + provider, i, ii, j, jj, bbox, coverage; │ │ │ │ │ + for (i = 0, ii = providers.length; i < ii; ++i) { │ │ │ │ │ + provider = providers[i]; │ │ │ │ │ + for (j = 0, jj = provider.coverageAreas.length; j < jj; ++j) { │ │ │ │ │ + coverage = provider.coverageAreas[j]; │ │ │ │ │ + // axis order provided is Y,X │ │ │ │ │ + bbox = OpenLayers.Bounds.fromArray(coverage.bbox, true); │ │ │ │ │ + if (extent.intersectsBounds(bbox) && │ │ │ │ │ + zoom <= coverage.zoomMax && zoom >= coverage.zoomMin) { │ │ │ │ │ + copyrights += provider.attribution + " "; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var logo = metadata.brandLogoUri.replace(this.protocolRegex, this.protocol); │ │ │ │ │ + this.attribution = OpenLayers.String.format(this.attributionTemplate, { │ │ │ │ │ + type: this.type.toLowerCase(), │ │ │ │ │ + logo: logo, │ │ │ │ │ + copyrights: copyrights │ │ │ │ │ }); │ │ │ │ │ + this.map && this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "attribution" │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return resp; │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + */ │ │ │ │ │ + setMap: function() { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.register("moveend", this, this.updateAttribution); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleUpdate │ │ │ │ │ - * Called the the request issued by <update> is complete. May be overridden │ │ │ │ │ - * by subclasses. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the update call. │ │ │ │ │ + * obj - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Bing>} An exact clone of this <OpenLayers.Layer.Bing> │ │ │ │ │ */ │ │ │ │ │ - handleUpdate: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options); │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Bing(this.options); │ │ │ │ │ + } │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: delete │ │ │ │ │ - * Construct a request deleting a removed feature. │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.map && │ │ │ │ │ + this.map.events.unregister("moveend", this, this.updateAttribution); │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Bing" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Function: OpenLayers.Layer.Bing.processMetadata │ │ │ │ │ + * This function will be bound to an instance, linked to the global scope with │ │ │ │ │ + * an id, and called by the JSONP script returned by the API. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * metadata - {Object} metadata as returned by the API │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Bing.processMetadata = function(metadata) { │ │ │ │ │ + this.metadata = metadata; │ │ │ │ │ + this.initLayer(); │ │ │ │ │ + var script = document.getElementById(this._callbackId); │ │ │ │ │ + script.parentNode.removeChild(script); │ │ │ │ │ + window[this._callbackId] = undefined; // cannot delete from window in IE │ │ │ │ │ + delete this._callbackId; │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/MapGuide.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.MapGuide │ │ │ │ │ + * Instances of OpenLayers.Layer.MapGuide are used to display │ │ │ │ │ + * data from a MapGuide OS instance. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.MapGuide = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} Treat this layer as a base layer. Default is true. │ │ │ │ │ + **/ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: useHttpTile │ │ │ │ │ + * {Boolean} use a tile cache exposed directly via a webserver rather than the │ │ │ │ │ + * via mapguide server. This does require extra configuration on the Mapguide Server, │ │ │ │ │ + * and will only work when singleTile is false. The url for the layer must be set to the │ │ │ │ │ + * webserver path rather than the Mapguide mapagent. │ │ │ │ │ + * See http://trac.osgeo.org/mapguide/wiki/CodeSamples/Tiles/ServingTilesViaHttp │ │ │ │ │ + **/ │ │ │ │ │ + useHttpTile: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: singleTile │ │ │ │ │ + * {Boolean} use tile server or request single tile image. │ │ │ │ │ + **/ │ │ │ │ │ + singleTile: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: useOverlay │ │ │ │ │ + * {Boolean} flag to indicate if the layer should be retrieved using │ │ │ │ │ + * GETMAPIMAGE (default) or using GETDYNAMICOVERLAY requests. │ │ │ │ │ + **/ │ │ │ │ │ + useOverlay: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: useAsyncOverlay │ │ │ │ │ + * {Boolean} indicates if the MapGuide site supports the asynchronous │ │ │ │ │ + * GETDYNAMICOVERLAY requests which is available in MapGuide Enterprise 2010 │ │ │ │ │ + * and MapGuide Open Source v2.0.3 or higher. The newer versions of MG │ │ │ │ │ + * is called asynchronously, allows selections to be drawn separately from │ │ │ │ │ + * the map and offers styling options. │ │ │ │ │ + * │ │ │ │ │ + * With older versions of MapGuide, set useAsyncOverlay=false. Note that in │ │ │ │ │ + * this case a synchronous AJAX call is issued and the mapname and session │ │ │ │ │ + * parameters must be used to initialize the layer, not the mapdefinition │ │ │ │ │ + * parameter. Also note that this will issue a synchronous AJAX request │ │ │ │ │ + * before the image request can be issued so the users browser may lock │ │ │ │ │ + * up if the MG Web tier does not respond in a timely fashion. │ │ │ │ │ + **/ │ │ │ │ │ + useAsyncOverlay: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: TILE_PARAMS │ │ │ │ │ + * {Object} Hashtable of default parameter key/value pairs for tiled layer │ │ │ │ │ + */ │ │ │ │ │ + TILE_PARAMS: { │ │ │ │ │ + operation: 'GETTILEIMAGE', │ │ │ │ │ + version: '1.2.0' │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: SINGLE_TILE_PARAMS │ │ │ │ │ + * {Object} Hashtable of default parameter key/value pairs for untiled layer │ │ │ │ │ + */ │ │ │ │ │ + SINGLE_TILE_PARAMS: { │ │ │ │ │ + operation: 'GETMAPIMAGE', │ │ │ │ │ + format: 'PNG', │ │ │ │ │ + locale: 'en', │ │ │ │ │ + clip: '1', │ │ │ │ │ + version: '1.0.0' │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: OVERLAY_PARAMS │ │ │ │ │ + * {Object} Hashtable of default parameter key/value pairs for untiled layer │ │ │ │ │ + */ │ │ │ │ │ + OVERLAY_PARAMS: { │ │ │ │ │ + operation: 'GETDYNAMICMAPOVERLAYIMAGE', │ │ │ │ │ + format: 'PNG', │ │ │ │ │ + locale: 'en', │ │ │ │ │ + clip: '1', │ │ │ │ │ + version: '2.0.0' │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: FOLDER_PARAMS │ │ │ │ │ + * {Object} Hashtable of parameter key/value pairs which describe │ │ │ │ │ + * the folder structure for tiles as configured in the mapguide │ │ │ │ │ + * serverconfig.ini section [TileServiceProperties] │ │ │ │ │ + */ │ │ │ │ │ + FOLDER_PARAMS: { │ │ │ │ │ + tileColumnsPerFolder: 30, │ │ │ │ │ + tileRowsPerFolder: 30, │ │ │ │ │ + format: 'png', │ │ │ │ │ + querystring: null │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultSize │ │ │ │ │ + * {<OpenLayers.Size>} Tile size as produced by MapGuide server │ │ │ │ │ + **/ │ │ │ │ │ + defaultSize: new OpenLayers.Size(300, 300), │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: tileOriginCorner │ │ │ │ │ + * {String} MapGuide tile server uses top-left as tile origin │ │ │ │ │ + **/ │ │ │ │ │ + tileOriginCorner: "tl", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.MapGuide │ │ │ │ │ + * Create a new Mapguide layer, either tiled or untiled. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ + * For tiled layers, the 'groupName' and 'mapDefinition' values │ │ │ │ │ + * must be specified as parameters in the constructor. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, whose "priv" property references the HTTP request, this │ │ │ │ │ - * object is also passed to the callback function when the request │ │ │ │ │ - * completes. │ │ │ │ │ + * For untiled base layers, specify either combination of 'mapName' and │ │ │ │ │ + * 'session', or 'mapDefinition' and 'locale'. │ │ │ │ │ + * │ │ │ │ │ + * For older versions of MapGuide and overlay layers, set useAsyncOverlay │ │ │ │ │ + * to false and in this case mapName and session are required parameters │ │ │ │ │ + * for the constructor. │ │ │ │ │ + * │ │ │ │ │ + * NOTE: MapGuide OS uses a DPI value and degrees to meters conversion │ │ │ │ │ + * factor that are different than the defaults used in OpenLayers, │ │ │ │ │ + * so these must be adjusted accordingly in your application. │ │ │ │ │ + * See the MapGuide example for how to set these values for MGOS. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} Name of the layer displayed in the interface │ │ │ │ │ + * url - {String} Location of the MapGuide mapagent executable │ │ │ │ │ + * (e.g. http://localhost:8008/mapguide/mapagent/mapagent.fcgi) │ │ │ │ │ + * params - {Object} hashtable of additional parameters to use. Some │ │ │ │ │ + * parameters may require additional code on the server. The ones that │ │ │ │ │ + * you may want to use are: │ │ │ │ │ + * - mapDefinition - {String} The MapGuide resource definition │ │ │ │ │ + * (e.g. Library://Samples/Gmap/Maps/gmapTiled.MapDefinition) │ │ │ │ │ + * - locale - Locale setting │ │ │ │ │ + * (for untiled overlays layers only) │ │ │ │ │ + * - mapName - {String} Name of the map as stored in the MapGuide session. │ │ │ │ │ + * (for untiled layers with a session parameter only) │ │ │ │ │ + * - session - { String} MapGuide session ID │ │ │ │ │ + * (for untiled overlays layers only) │ │ │ │ │ + * - basemaplayergroupname - {String} GroupName for tiled MapGuide layers only │ │ │ │ │ + * - format - Image format to be returned (for untiled overlay layers only) │ │ │ │ │ + * - showLayers - {String} A comma separated list of GUID's for the │ │ │ │ │ + * layers to display eg: 'cvc-xcv34,453-345-345sdf'. │ │ │ │ │ + * - hideLayers - {String} A comma separated list of GUID's for the │ │ │ │ │ + * layers to hide eg: 'cvc-xcv34,453-345-345sdf'. │ │ │ │ │ + * - showGroups - {String} A comma separated list of GUID's for the │ │ │ │ │ + * groups to display eg: 'cvc-xcv34,453-345-345sdf'. │ │ │ │ │ + * - hideGroups - {String} A comma separated list of GUID's for the │ │ │ │ │ + * groups to hide eg: 'cvc-xcv34,453-345-345sdf' │ │ │ │ │ + * - selectionXml - {String} A selection xml string Some server plumbing │ │ │ │ │ + * is required to read such a value. │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer; │ │ │ │ │ + * will vary depending if tiled or untiled maps are being requested │ │ │ │ │ */ │ │ │ │ │ - "delete": function(feature, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var url = options.url || │ │ │ │ │ - feature.url || │ │ │ │ │ - this.options.url + "/" + feature.fid; │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: feature, │ │ │ │ │ - requestType: "delete" │ │ │ │ │ - }); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - var method = this.deleteWithPOST ? "POST" : "DELETE"; │ │ │ │ │ - var requestOptions = { │ │ │ │ │ - url: url, │ │ │ │ │ - callback: this.createCallback(this.handleDelete, resp, options), │ │ │ │ │ - headers: options.headers │ │ │ │ │ - }; │ │ │ │ │ - if (this.deleteWithPOST) { │ │ │ │ │ - requestOptions.data = this.format.write(feature); │ │ │ │ │ + // unless explicitly set in options, if the layer is transparent, │ │ │ │ │ + // it will be an overlay │ │ │ │ │ + if (options == null || options.isBaseLayer == null) { │ │ │ │ │ + this.isBaseLayer = ((this.transparent != "true") && │ │ │ │ │ + (this.transparent != true)); │ │ │ │ │ } │ │ │ │ │ - resp.priv = OpenLayers.Request[method](requestOptions); │ │ │ │ │ │ │ │ │ │ - return resp; │ │ │ │ │ + if (options && options.useOverlay != null) { │ │ │ │ │ + this.useOverlay = options.useOverlay; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //initialize for untiled layers │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + if (this.useOverlay) { │ │ │ │ │ + OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, │ │ │ │ │ + this.OVERLAY_PARAMS │ │ │ │ │ + ); │ │ │ │ │ + if (!this.useAsyncOverlay) { │ │ │ │ │ + this.params.version = "1.0.0"; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, │ │ │ │ │ + this.SINGLE_TILE_PARAMS │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + //initialize for tiled layers │ │ │ │ │ + if (this.useHttpTile) { │ │ │ │ │ + OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, │ │ │ │ │ + this.FOLDER_PARAMS │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, │ │ │ │ │ + this.TILE_PARAMS │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + this.setTileSize(this.defaultSize); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleDelete │ │ │ │ │ - * Called the the request issued by <delete> is complete. May be overridden │ │ │ │ │ - * by subclasses. │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the delete call. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.MapGuide>} An exact clone of this layer │ │ │ │ │ */ │ │ │ │ │ - handleDelete: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options); │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.MapGuide(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.params, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleResponse │ │ │ │ │ - * Called by CRUD specific handlers. │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * Return a query string for this layer │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the create, read, update, │ │ │ │ │ - * or delete call. │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox │ │ │ │ │ + * for the request │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the layer's url and parameters and also │ │ │ │ │ + * the passed-in bounds and appropriate tile size specified │ │ │ │ │ + * as parameters. │ │ │ │ │ */ │ │ │ │ │ - handleResponse: function(resp, options) { │ │ │ │ │ - var request = resp.priv; │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - // success │ │ │ │ │ - if (resp.requestType != "delete") { │ │ │ │ │ - resp.features = this.parseFeatures(request); │ │ │ │ │ - } │ │ │ │ │ - resp.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var url; │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var mapSize = this.map.getSize(); │ │ │ │ │ + │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + //set up the call for GETMAPIMAGE or GETDYNAMICMAPOVERLAY with │ │ │ │ │ + //dynamic map parameters │ │ │ │ │ + var params = { │ │ │ │ │ + setdisplaydpi: OpenLayers.DOTS_PER_INCH, │ │ │ │ │ + setdisplayheight: mapSize.h * this.ratio, │ │ │ │ │ + setdisplaywidth: mapSize.w * this.ratio, │ │ │ │ │ + setviewcenterx: center.lon, │ │ │ │ │ + setviewcentery: center.lat, │ │ │ │ │ + setviewscale: this.map.getScale() │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + if (this.useOverlay && !this.useAsyncOverlay) { │ │ │ │ │ + //first we need to call GETVISIBLEMAPEXTENT to set the extent │ │ │ │ │ + var getVisParams = {}; │ │ │ │ │ + getVisParams = OpenLayers.Util.extend(getVisParams, params); │ │ │ │ │ + getVisParams.operation = "GETVISIBLEMAPEXTENT"; │ │ │ │ │ + getVisParams.version = "1.0.0"; │ │ │ │ │ + getVisParams.session = this.params.session; │ │ │ │ │ + getVisParams.mapName = this.params.mapName; │ │ │ │ │ + getVisParams.format = 'text/xml'; │ │ │ │ │ + url = this.getFullRequestString(getVisParams); │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Request.GET({ │ │ │ │ │ + url: url, │ │ │ │ │ + async: false │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + //construct the full URL │ │ │ │ │ + url = this.getFullRequestString(params); │ │ │ │ │ + } else { │ │ │ │ │ + │ │ │ │ │ + //tiled version │ │ │ │ │ + var currentRes = this.map.getResolution(); │ │ │ │ │ + var colidx = Math.floor((bounds.left - this.maxExtent.left) / currentRes); │ │ │ │ │ + colidx = Math.round(colidx / this.tileSize.w); │ │ │ │ │ + var rowidx = Math.floor((this.maxExtent.top - bounds.top) / currentRes); │ │ │ │ │ + rowidx = Math.round(rowidx / this.tileSize.h); │ │ │ │ │ + │ │ │ │ │ + if (this.useHttpTile) { │ │ │ │ │ + url = this.getImageFilePath({ │ │ │ │ │ + tilecol: colidx, │ │ │ │ │ + tilerow: rowidx, │ │ │ │ │ + scaleindex: this.resolutions.length - this.map.zoom - 1 │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ } else { │ │ │ │ │ - // failure │ │ │ │ │ - resp.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + url = this.getFullRequestString({ │ │ │ │ │ + tilecol: colidx, │ │ │ │ │ + tilerow: rowidx, │ │ │ │ │ + scaleindex: this.resolutions.length - this.map.zoom - 1 │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - options.callback.call(options.scope, resp); │ │ │ │ │ } │ │ │ │ │ + return url; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseFeatures │ │ │ │ │ - * Read HTTP response body and return features. │ │ │ │ │ + * Method: getFullRequestString │ │ │ │ │ + * getFullRequestString on MapGuide layers is special, because we │ │ │ │ │ + * do a regular expression replace on ',' in parameters to '+'. │ │ │ │ │ + * This is why it is subclassed here. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * request - {XMLHttpRequest} The request object │ │ │ │ │ + * altUrl - {String} Alternative base URL to use. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} Array of features or a single feature. │ │ │ │ │ + * {String} A string with the layer's url appropriately encoded for MapGuide │ │ │ │ │ */ │ │ │ │ │ - parseFeatures: function(request) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ + getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ + // use layer's url unless altUrl passed in │ │ │ │ │ + var url = (altUrl == null) ? this.url : altUrl; │ │ │ │ │ + │ │ │ │ │ + // if url is not a string, it should be an array of strings, │ │ │ │ │ + // in which case we will randomly select one of them in order │ │ │ │ │ + // to evenly distribute requests to different urls. │ │ │ │ │ + if (typeof url == "object") { │ │ │ │ │ + url = url[Math.floor(Math.random() * url.length)]; │ │ │ │ │ } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null; │ │ │ │ │ + // requestString always starts with url │ │ │ │ │ + var requestString = url; │ │ │ │ │ + │ │ │ │ │ + // create a new params hashtable with all the layer params and the │ │ │ │ │ + // new params together. then convert to string │ │ │ │ │ + var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ + allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ + // ignore parameters that are already in the url search string │ │ │ │ │ + var urlParams = OpenLayers.Util.upperCaseObject( │ │ │ │ │ + OpenLayers.Util.getParameters(url)); │ │ │ │ │ + for (var key in allParams) { │ │ │ │ │ + if (key.toUpperCase() in urlParams) { │ │ │ │ │ + delete allParams[key]; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return this.format.read(doc); │ │ │ │ │ + var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + │ │ │ │ │ + /* MapGuide needs '+' seperating things like bounds/height/width. │ │ │ │ │ + Since typically this is URL encoded, we use a slight hack: we │ │ │ │ │ + depend on the list-like functionality of getParameterString to │ │ │ │ │ + leave ',' only in the case of list items (since otherwise it is │ │ │ │ │ + encoded) then do a regular expression replace on the , characters │ │ │ │ │ + to '+' */ │ │ │ │ │ + paramsString = paramsString.replace(/,/g, "+"); │ │ │ │ │ + │ │ │ │ │ + if (paramsString != "") { │ │ │ │ │ + var lastServerChar = url.charAt(url.length - 1); │ │ │ │ │ + if ((lastServerChar == "&") || (lastServerChar == "?")) { │ │ │ │ │ + requestString += paramsString; │ │ │ │ │ + } else { │ │ │ │ │ + if (url.indexOf('?') == -1) { │ │ │ │ │ + //serverPath has no ? -- add one │ │ │ │ │ + requestString += '?' + paramsString; │ │ │ │ │ + } else { │ │ │ │ │ + //serverPath contains ?, so must already have paramsString at the end │ │ │ │ │ + requestString += '&' + paramsString; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return requestString; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: commit │ │ │ │ │ - * Iterate over each feature and take action based on the feature state. │ │ │ │ │ - * Possible actions are create, update and delete. │ │ │ │ │ + /** │ │ │ │ │ + * Method: getImageFilePath │ │ │ │ │ + * special handler to request mapguide tiles from an http exposed tilecache │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array({<OpenLayers.Feature.Vector>})} │ │ │ │ │ - * options - {Object} Optional object for setting up intermediate commit │ │ │ │ │ - * callbacks. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * create - {Object} Optional object to be passed to the <create> method. │ │ │ │ │ - * update - {Object} Optional object to be passed to the <update> method. │ │ │ │ │ - * delete - {Object} Optional object to be passed to the <delete> method. │ │ │ │ │ - * callback - {Function} Optional function to be called when the commit │ │ │ │ │ - * is complete. │ │ │ │ │ - * scope - {Object} Optional object to be set as the scope of the callback. │ │ │ │ │ + * altUrl - {String} Alternative base URL to use. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Protocol.Response>)} An array of response objects, │ │ │ │ │ - * one per request made to the server, each object's "priv" property │ │ │ │ │ - * references the corresponding HTTP request. │ │ │ │ │ + * {String} A string with the url for the tile image │ │ │ │ │ */ │ │ │ │ │ - commit: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = [], │ │ │ │ │ - nResponses = 0; │ │ │ │ │ + getImageFilePath: function(newParams, altUrl) { │ │ │ │ │ + // use layer's url unless altUrl passed in │ │ │ │ │ + var url = (altUrl == null) ? this.url : altUrl; │ │ │ │ │ │ │ │ │ │ - // Divide up features before issuing any requests. This properly │ │ │ │ │ - // counts requests in the event that any responses come in before │ │ │ │ │ - // all requests have been issued. │ │ │ │ │ - var types = {}; │ │ │ │ │ - types[OpenLayers.State.INSERT] = []; │ │ │ │ │ - types[OpenLayers.State.UPDATE] = []; │ │ │ │ │ - types[OpenLayers.State.DELETE] = []; │ │ │ │ │ - var feature, list, requestFeatures = []; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - list = types[feature.state]; │ │ │ │ │ - if (list) { │ │ │ │ │ - list.push(feature); │ │ │ │ │ - requestFeatures.push(feature); │ │ │ │ │ - } │ │ │ │ │ + // if url is not a string, it should be an array of strings, │ │ │ │ │ + // in which case we will randomly select one of them in order │ │ │ │ │ + // to evenly distribute requests to different urls. │ │ │ │ │ + if (typeof url == "object") { │ │ │ │ │ + url = url[Math.floor(Math.random() * url.length)]; │ │ │ │ │ } │ │ │ │ │ - // tally up number of requests │ │ │ │ │ - var nRequests = (types[OpenLayers.State.INSERT].length > 0 ? 1 : 0) + │ │ │ │ │ - types[OpenLayers.State.UPDATE].length + │ │ │ │ │ - types[OpenLayers.State.DELETE].length; │ │ │ │ │ + // requestString always starts with url │ │ │ │ │ + var requestString = url; │ │ │ │ │ │ │ │ │ │ - // This response will be sent to the final callback after all the others │ │ │ │ │ - // have been fired. │ │ │ │ │ - var success = true; │ │ │ │ │ - var finalResponse = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: requestFeatures │ │ │ │ │ - }); │ │ │ │ │ + var tileRowGroup = ""; │ │ │ │ │ + var tileColGroup = ""; │ │ │ │ │ │ │ │ │ │ - function insertCallback(response) { │ │ │ │ │ - var len = response.features ? response.features.length : 0; │ │ │ │ │ - var fids = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - fids[i] = response.features[i].fid; │ │ │ │ │ - } │ │ │ │ │ - finalResponse.insertIds = fids; │ │ │ │ │ - callback.apply(this, [response]); │ │ │ │ │ + if (newParams.tilerow < 0) { │ │ │ │ │ + tileRowGroup = '-'; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - function callback(response) { │ │ │ │ │ - this.callUserCallback(response, options); │ │ │ │ │ - success = success && response.success(); │ │ │ │ │ - nResponses++; │ │ │ │ │ - if (nResponses >= nRequests) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - finalResponse.code = success ? │ │ │ │ │ - OpenLayers.Protocol.Response.SUCCESS : │ │ │ │ │ - OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - options.callback.apply(options.scope, [finalResponse]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + if (newParams.tilerow == 0) { │ │ │ │ │ + tileRowGroup += '0'; │ │ │ │ │ + } else { │ │ │ │ │ + tileRowGroup += Math.floor(Math.abs(newParams.tilerow / this.params.tileRowsPerFolder)) * this.params.tileRowsPerFolder; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // start issuing requests │ │ │ │ │ - var queue = types[OpenLayers.State.INSERT]; │ │ │ │ │ - if (queue.length > 0) { │ │ │ │ │ - resp.push(this.create( │ │ │ │ │ - queue, OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: insertCallback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options.create) │ │ │ │ │ - )); │ │ │ │ │ + if (newParams.tilecol < 0) { │ │ │ │ │ + tileColGroup = '-'; │ │ │ │ │ } │ │ │ │ │ - queue = types[OpenLayers.State.UPDATE]; │ │ │ │ │ - for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ - resp.push(this.update( │ │ │ │ │ - queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: callback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options.update))); │ │ │ │ │ + │ │ │ │ │ + if (newParams.tilecol == 0) { │ │ │ │ │ + tileColGroup += '0'; │ │ │ │ │ + } else { │ │ │ │ │ + tileColGroup += Math.floor(Math.abs(newParams.tilecol / this.params.tileColumnsPerFolder)) * this.params.tileColumnsPerFolder; │ │ │ │ │ } │ │ │ │ │ - queue = types[OpenLayers.State.DELETE]; │ │ │ │ │ - for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ - resp.push(this["delete"]( │ │ │ │ │ - queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: callback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options["delete"]))); │ │ │ │ │ + │ │ │ │ │ + var tilePath = '/S' + Math.floor(newParams.scaleindex) + │ │ │ │ │ + '/' + this.params.basemaplayergroupname + │ │ │ │ │ + '/R' + tileRowGroup + │ │ │ │ │ + '/C' + tileColGroup + │ │ │ │ │ + '/' + (newParams.tilerow % this.params.tileRowsPerFolder) + │ │ │ │ │ + '_' + (newParams.tilecol % this.params.tileColumnsPerFolder) + │ │ │ │ │ + '.' + this.params.format; │ │ │ │ │ + │ │ │ │ │ + if (this.params.querystring) { │ │ │ │ │ + tilePath += "?" + this.params.querystring; │ │ │ │ │ } │ │ │ │ │ - return resp; │ │ │ │ │ + │ │ │ │ │ + requestString += tilePath; │ │ │ │ │ + return requestString; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.MapGuide" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/TMS.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.TMS │ │ │ │ │ + * Create a layer for accessing tiles from services that conform with the │ │ │ │ │ + * Tile Map Service Specification │ │ │ │ │ + * (http://wiki.osgeo.org/wiki/Tile_Map_Service_Specification). │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var layer = new OpenLayers.Layer.TMS( │ │ │ │ │ + * "My Layer", // name for display in LayerSwitcher │ │ │ │ │ + * "http://tilecache.osgeo.org/wms-c/Basic.py/", // service endpoint │ │ │ │ │ + * {layername: "basic", type: "png"} // required properties │ │ │ │ │ + * ); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.TMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: abort │ │ │ │ │ - * Abort an ongoing request, the response object passed to │ │ │ │ │ - * this method must come from this HTTP protocol (as a result │ │ │ │ │ - * of a create, read, update, delete or commit operation). │ │ │ │ │ + * APIProperty: serviceVersion │ │ │ │ │ + * {String} Service version for tile requests. Default is "1.0.0". │ │ │ │ │ + */ │ │ │ │ │ + serviceVersion: "1.0.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layername │ │ │ │ │ + * {String} The identifier for the <TileMap> as advertised by the service. │ │ │ │ │ + * For example, if the service advertises a <TileMap> with │ │ │ │ │ + * 'href="http://tms.osgeo.org/1.0.0/vmap0"', the <layername> property │ │ │ │ │ + * would be set to "vmap0". │ │ │ │ │ + */ │ │ │ │ │ + layername: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: type │ │ │ │ │ + * {String} The format extension corresponding to the requested tile image │ │ │ │ │ + * type. This is advertised in a <TileFormat> element as the │ │ │ │ │ + * "extension" attribute. For example, if the service advertises a │ │ │ │ │ + * <TileMap> with <TileFormat width="256" height="256" mime-type="image/jpeg" extension="jpg" />, │ │ │ │ │ + * the <type> property would be set to "jpg". │ │ │ │ │ + */ │ │ │ │ │ + type: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} Make this layer a base layer. Default is true. Set false to │ │ │ │ │ + * use the layer as an overlay. │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: tileOrigin │ │ │ │ │ + * {<OpenLayers.LonLat>} Optional origin for aligning the grid of tiles. │ │ │ │ │ + * If provided, requests for tiles at all resolutions will be aligned │ │ │ │ │ + * with this location (no tiles shall overlap this location). If │ │ │ │ │ + * not provided, the grid of tiles will be aligned with the bottom-left │ │ │ │ │ + * corner of the map's <maxExtent>. Default is ``null``. │ │ │ │ │ * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var layer = new OpenLayers.Layer.TMS( │ │ │ │ │ + * "My Layer", │ │ │ │ │ + * "http://tilecache.osgeo.org/wms-c/Basic.py/", │ │ │ │ │ + * { │ │ │ │ │ + * layername: "basic", │ │ │ │ │ + * type: "png", │ │ │ │ │ + * // set if different than the bottom left of map.maxExtent │ │ │ │ │ + * tileOrigin: new OpenLayers.LonLat(-180, -90) │ │ │ │ │ + * } │ │ │ │ │ + * ); │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ + tileOrigin: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: serverResolutions │ │ │ │ │ + * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ + * property if the map resolutions differ from the server. This │ │ │ │ │ + * property serves two purposes. (a) <serverResolutions> can include │ │ │ │ │ + * resolutions that the server supports and that you don't want to │ │ │ │ │ + * provide with this layer; you can also look at <zoomOffset>, which is │ │ │ │ │ + * an alternative to <serverResolutions> for that specific purpose. │ │ │ │ │ + * (b) The map can work with resolutions that aren't supported by │ │ │ │ │ + * the server, i.e. that aren't in <serverResolutions>. When the │ │ │ │ │ + * map is displayed in such a resolution data for the closest │ │ │ │ │ + * server-supported resolution is loaded and the layer div is │ │ │ │ │ + * stretched as necessary. │ │ │ │ │ + */ │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomOffset │ │ │ │ │ + * {Number} If your cache has more zoom levels than you want to provide │ │ │ │ │ + * access to with this layer, supply a zoomOffset. This zoom offset │ │ │ │ │ + * is added to the current map zoom level to determine the level │ │ │ │ │ + * for a requested tile. For example, if you supply a zoomOffset │ │ │ │ │ + * of 3, when the map is at the zoom 0, tiles will be requested from │ │ │ │ │ + * level 3 of your cache. Default is 0 (assumes cache level and map │ │ │ │ │ + * zoom are equivalent). Using <zoomOffset> is an alternative to │ │ │ │ │ + * setting <serverResolutions> if you only want to expose a subset │ │ │ │ │ + * of the server resolutions. │ │ │ │ │ + */ │ │ │ │ │ + zoomOffset: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.TMS │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ + * name - {String} Title to be displayed in a <OpenLayers.Control.LayerSwitcher> │ │ │ │ │ + * url - {String} Service endpoint (without the version number). E.g. │ │ │ │ │ + * "http://tms.osgeo.org/". │ │ │ │ │ + * options - {Object} Additional properties to be set on the layer. The │ │ │ │ │ + * <layername> and <type> properties must be set here. │ │ │ │ │ */ │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - response.priv.abort(); │ │ │ │ │ - } │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + newArguments.push(name, url, {}, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: callUserCallback │ │ │ │ │ - * This method is used from within the commit method each time an │ │ │ │ │ - * an HTTP response is received from the server, it is responsible │ │ │ │ │ - * for calling the user-supplied callbacks. │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Create a complete copy of this layer. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} │ │ │ │ │ - * options - {Object} The map of options passed to the commit call. │ │ │ │ │ + * obj - {Object} Should only be provided by subclasses that call this │ │ │ │ │ + * method. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.TMS>} An exact clone of this <OpenLayers.Layer.TMS> │ │ │ │ │ */ │ │ │ │ │ - callUserCallback: function(resp, options) { │ │ │ │ │ - var opt = options[resp.requestType]; │ │ │ │ │ - if (opt && opt.callback) { │ │ │ │ │ - opt.callback.call(opt.scope, resp); │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.TMS(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.HTTP" │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as │ │ │ │ │ + * parameters │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((bounds.bottom - this.tileOrigin.lat) / (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getServerZoom(); │ │ │ │ │ + var path = this.serviceVersion + "/" + this.layername + "/" + z + "/" + x + "/" + y + "." + this.type; │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(path, url); │ │ │ │ │ + } │ │ │ │ │ + return url + path; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * When the layer is added to a map, then we can fetch our origin │ │ │ │ │ + * (if we don't have one.) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (!this.tileOrigin) { │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.map.maxExtent.left, │ │ │ │ │ + this.map.maxExtent.bottom); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.TMS" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/Script.js │ │ │ │ │ + OpenLayers/Layer/OSM.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Protocol.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Format/GeoJSON.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * if application uses the query string, for example, for BBOX parameters, │ │ │ │ │ - * OpenLayers/Format/QueryStringFilter.js should be included in the build config file │ │ │ │ │ + * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Protocol.Script │ │ │ │ │ - * A basic Script protocol for vector layers. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Protocol.Script> constructor. A script protocol is used to │ │ │ │ │ - * get around the same origin policy. It works with services that return │ │ │ │ │ - * JSONP - that is, JSON wrapped in a client-specified callback. The │ │ │ │ │ - * protocol handles fetching and parsing of feature data and sends parsed │ │ │ │ │ - * features to the <callback> configured with the protocol. The protocol │ │ │ │ │ - * expects features serialized as GeoJSON by default, but can be configured │ │ │ │ │ - * to work with other formats by setting the <format> property. │ │ │ │ │ + * Class: OpenLayers.Layer.OSM │ │ │ │ │ + * This layer allows accessing OpenStreetMap tiles. By default the OpenStreetMap │ │ │ │ │ + * hosted tile.openstreetmap.org Mapnik tileset is used. If you wish to use │ │ │ │ │ + * a different layer instead, you need to provide a different │ │ │ │ │ + * URL to the constructor. Here's an example for using OpenCycleMap: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * new OpenLayers.Layer.OSM("OpenCycleMap", │ │ │ │ │ + * ["http://a.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ + * "http://b.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ + * "http://c.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png"]); │ │ │ │ │ + * (end) │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol> │ │ │ │ │ + * - <OpenLayers.Layer.XYZ> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.Script = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ +OpenLayers.Layer.OSM = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String} Service URL. The service is expected to return serialized │ │ │ │ │ - * features wrapped in a named callback (where the callback name is │ │ │ │ │ - * generated by this protocol). │ │ │ │ │ - * Read-only, set through the options passed to the constructor. │ │ │ │ │ + * APIProperty: name │ │ │ │ │ + * {String} The layer name. Defaults to "OpenStreetMap" if the first │ │ │ │ │ + * argument to the constructor is null or undefined. │ │ │ │ │ */ │ │ │ │ │ - url: null, │ │ │ │ │ + name: "OpenStreetMap", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: params │ │ │ │ │ - * {Object} Query string parameters to be appended to the URL. │ │ │ │ │ - * Read-only, set through the options passed to the constructor. │ │ │ │ │ - * Example: {maxFeatures: 50} │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String} The tileset URL scheme. Defaults to │ │ │ │ │ + * : http://[a|b|c].tile.openstreetmap.org/${z}/${x}/${y}.png │ │ │ │ │ + * (the official OSM tileset) if the second argument to the constructor │ │ │ │ │ + * is null or undefined. To use another tileset you can have something │ │ │ │ │ + * like this: │ │ │ │ │ + * (code) │ │ │ │ │ + * new OpenLayers.Layer.OSM("OpenCycleMap", │ │ │ │ │ + * ["http://a.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ + * "http://b.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ + * "http://c.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png"]); │ │ │ │ │ + * (end) │ │ │ │ │ */ │ │ │ │ │ - params: null, │ │ │ │ │ + url: [ │ │ │ │ │ + 'http://a.tile.openstreetmap.org/${z}/${x}/${y}.png', │ │ │ │ │ + 'http://b.tile.openstreetmap.org/${z}/${x}/${y}.png', │ │ │ │ │ + 'http://c.tile.openstreetmap.org/${z}/${x}/${y}.png' │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: callback │ │ │ │ │ - * {Object} Function to be called when the <read> operation completes. │ │ │ │ │ + * Property: attribution │ │ │ │ │ + * {String} The layer attribution. │ │ │ │ │ */ │ │ │ │ │ - callback: null, │ │ │ │ │ + attribution: "© <a href='http://www.openstreetmap.org/copyright'>OpenStreetMap</a> contributors", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: callbackTemplate │ │ │ │ │ - * {String} Template for creating a unique callback function name │ │ │ │ │ - * for the registry. Should include ${id}. The ${id} variable will be │ │ │ │ │ - * replaced with a string identifier prefixed with a "c" (e.g. c1, c2). │ │ │ │ │ - * Default is "OpenLayers.Protocol.Script.registry.${id}". │ │ │ │ │ + * Property: sphericalMercator │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - callbackTemplate: "OpenLayers.Protocol.Script.registry.${id}", │ │ │ │ │ + sphericalMercator: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: callbackKey │ │ │ │ │ - * {String} The name of the query string parameter that the service │ │ │ │ │ - * recognizes as the callback identifier. Default is "callback". │ │ │ │ │ - * This key is used to generate the URL for the script. For example │ │ │ │ │ - * setting <callbackKey> to "myCallback" would result in a URL like │ │ │ │ │ - * http://example.com/?myCallback=... │ │ │ │ │ + * Property: wrapDateLine │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - callbackKey: "callback", │ │ │ │ │ + wrapDateLine: true, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: callbackPrefix │ │ │ │ │ - * {String} Where a service requires that the callback query string │ │ │ │ │ - * parameter value is prefixed by some string, this value may be set. │ │ │ │ │ - * For example, setting <callbackPrefix> to "foo:" would result in a │ │ │ │ │ - * URL like http://example.com/?callback=foo:... Default is "". │ │ │ │ │ + /** APIProperty: tileOptions │ │ │ │ │ + * {Object} optional configuration options for <OpenLayers.Tile> instances │ │ │ │ │ + * created by this Layer. Default is │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * {crossOriginKeyword: 'anonymous'} │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * When using OSM tilesets other than the default ones, it may be │ │ │ │ │ + * necessary to set this to │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * {crossOriginKeyword: null} │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * if the server does not send Access-Control-Allow-Origin headers. │ │ │ │ │ */ │ │ │ │ │ - callbackPrefix: "", │ │ │ │ │ + tileOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: scope │ │ │ │ │ - * {Object} Optional ``this`` object for the callback. Read-only, set │ │ │ │ │ - * through the options passed to the constructor. │ │ │ │ │ + * Constructor: OpenLayers.Layer.OSM │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} The layer name. │ │ │ │ │ + * url - {String} The tileset URL scheme. │ │ │ │ │ + * options - {Object} Configuration options for the layer. Any inherited │ │ │ │ │ + * layer option can be set in this object (e.g. │ │ │ │ │ + * <OpenLayers.Layer.Grid.buffer>). │ │ │ │ │ */ │ │ │ │ │ - scope: null, │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + crossOriginKeyword: 'anonymous' │ │ │ │ │ + }, this.options && this.options.tileOptions); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: format │ │ │ │ │ - * {<OpenLayers.Format>} Format for parsing features. Default is an │ │ │ │ │ - * <OpenLayers.Format.GeoJSON> format. If an alternative is provided, │ │ │ │ │ - * the format's read method must take an object and return an array │ │ │ │ │ - * of features. │ │ │ │ │ + * Method: clone │ │ │ │ │ */ │ │ │ │ │ - format: null, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.OSM( │ │ │ │ │ + this.name, this.url, this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.OSM" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/SphericalMercator.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer.js │ │ │ │ │ + * @requires OpenLayers/Projection.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.SphericalMercator │ │ │ │ │ + * A mixin for layers that wraps up the pieces neccesary to have a coordinate │ │ │ │ │ + * conversion for working with commercial APIs which use a spherical │ │ │ │ │ + * mercator projection. Using this layer as a base layer, additional │ │ │ │ │ + * layers can be used as overlays if they are in the same projection. │ │ │ │ │ + * │ │ │ │ │ + * A layer is given properties of this object by setting the sphericalMercator │ │ │ │ │ + * property to true. │ │ │ │ │ + * │ │ │ │ │ + * More projection information: │ │ │ │ │ + * - http://spatialreference.org/ref/user/google-projection/ │ │ │ │ │ + * │ │ │ │ │ + * Proj4 Text: │ │ │ │ │ + * +proj=merc +a=6378137 +b=6378137 +lat_ts=0.0 +lon_0=0.0 +x_0=0.0 +y_0=0 │ │ │ │ │ + * +k=1.0 +units=m +nadgrids=@null +no_defs │ │ │ │ │ + * │ │ │ │ │ + * WKT: │ │ │ │ │ + * 900913=PROJCS["WGS84 / Simple Mercator", GEOGCS["WGS 84", │ │ │ │ │ + * DATUM["WGS_1984", SPHEROID["WGS_1984", 6378137.0, 298.257223563]], │ │ │ │ │ + * PRIMEM["Greenwich", 0.0], UNIT["degree", 0.017453292519943295], │ │ │ │ │ + * AXIS["Longitude", EAST], AXIS["Latitude", NORTH]], │ │ │ │ │ + * PROJECTION["Mercator_1SP_Google"], │ │ │ │ │ + * PARAMETER["latitude_of_origin", 0.0], PARAMETER["central_meridian", 0.0], │ │ │ │ │ + * PARAMETER["scale_factor", 1.0], PARAMETER["false_easting", 0.0], │ │ │ │ │ + * PARAMETER["false_northing", 0.0], UNIT["m", 1.0], AXIS["x", EAST], │ │ │ │ │ + * AXIS["y", NORTH], AUTHORITY["EPSG","900913"]] │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.SphericalMercator = { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: pendingRequests │ │ │ │ │ - * {Object} References all pending requests. Property names are script │ │ │ │ │ - * identifiers and property values are script elements. │ │ │ │ │ + * Method: getExtent │ │ │ │ │ + * Get the map's extent. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} The map extent. │ │ │ │ │ */ │ │ │ │ │ - pendingRequests: null, │ │ │ │ │ + getExtent: function() { │ │ │ │ │ + var extent = null; │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + extent = this.map.calculateBounds(); │ │ │ │ │ + } else { │ │ │ │ │ + extent = OpenLayers.Layer.FixedZoomLevels.prototype.getExtent.apply(this); │ │ │ │ │ + } │ │ │ │ │ + return extent; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: srsInBBOX │ │ │ │ │ - * {Boolean} Include the SRS identifier in BBOX query string parameter. │ │ │ │ │ - * Setting this property has no effect if a custom filterToParams method │ │ │ │ │ - * is provided. Default is false. If true and the layer has a │ │ │ │ │ - * projection object set, any BBOX filter will be serialized with a │ │ │ │ │ - * fifth item identifying the projection. │ │ │ │ │ - * E.g. bbox=-1000,-1000,1000,1000,EPSG:900913 │ │ │ │ │ + * Method: getLonLatFromViewPortPx │ │ │ │ │ + * Get a map location from a pixel location │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * viewPortPx - {<OpenLayers.Pixel>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.LonLat>} An OpenLayers.LonLat which is the passed-in view │ │ │ │ │ + * port OpenLayers.Pixel, translated into lon/lat by map lib │ │ │ │ │ + * If the map lib is not loaded or not centered, returns null │ │ │ │ │ */ │ │ │ │ │ - srsInBBOX: false, │ │ │ │ │ + getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ + return OpenLayers.Layer.prototype.getLonLatFromViewPortPx.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.Script │ │ │ │ │ - * A class for giving layers generic Script protocol. │ │ │ │ │ + * Method: getViewPortPxFromLonLat │ │ │ │ │ + * Get a pixel location from a map location │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ * │ │ │ │ │ - * Valid options include: │ │ │ │ │ - * url - {String} │ │ │ │ │ - * params - {Object} │ │ │ │ │ - * callback - {Function} │ │ │ │ │ - * scope - {Object} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Pixel>} An OpenLayers.Pixel which is the passed-in │ │ │ │ │ + * OpenLayers.LonLat, translated into view port pixels by map lib │ │ │ │ │ + * If map lib is not loaded or not centered, returns null │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - this.params = {}; │ │ │ │ │ - this.pendingRequests = {}; │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (!this.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.GeoJSON(); │ │ │ │ │ - } │ │ │ │ │ + getViewPortPxFromLonLat: function(lonlat) { │ │ │ │ │ + return OpenLayers.Layer.prototype.getViewPortPxFromLonLat.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ - var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ - srsInBBOX: this.srsInBBOX │ │ │ │ │ - }); │ │ │ │ │ - this.filterToParams = function(filter, params) { │ │ │ │ │ - return format.write(filter, params); │ │ │ │ │ - }; │ │ │ │ │ + /** │ │ │ │ │ + * Method: initMercatorParameters │ │ │ │ │ + * Set up the mercator parameters on the layer: resolutions, │ │ │ │ │ + * projection, units. │ │ │ │ │ + */ │ │ │ │ │ + initMercatorParameters: function() { │ │ │ │ │ + // set up properties for Mercator - assume EPSG:900913 │ │ │ │ │ + this.RESOLUTIONS = []; │ │ │ │ │ + var maxResolution = 156543.03390625; │ │ │ │ │ + for (var zoom = 0; zoom <= this.MAX_ZOOM_LEVEL; ++zoom) { │ │ │ │ │ + this.RESOLUTIONS[zoom] = maxResolution / Math.pow(2, zoom); │ │ │ │ │ } │ │ │ │ │ + this.units = "m"; │ │ │ │ │ + this.projection = this.projection || "EPSG:900913"; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Construct a request for reading new features. │ │ │ │ │ + * APIMethod: forwardMercator │ │ │ │ │ + * Given a lon,lat in EPSG:4326, return a point in Spherical Mercator. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * url - {String} Url for the request. │ │ │ │ │ - * params - {Object} Parameters to get serialized as a query string. │ │ │ │ │ - * filter - {<OpenLayers.Filter>} Filter to get serialized as a │ │ │ │ │ - * query string. │ │ │ │ │ + * lon - {float} │ │ │ │ │ + * lat - {float} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.LonLat>} The coordinates transformed to Mercator. │ │ │ │ │ + */ │ │ │ │ │ + forwardMercator: (function() { │ │ │ │ │ + var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + var sm = new OpenLayers.Projection("EPSG:900913"); │ │ │ │ │ + return function(lon, lat) { │ │ │ │ │ + var point = OpenLayers.Projection.transform({ │ │ │ │ │ + x: lon, │ │ │ │ │ + y: lat │ │ │ │ │ + }, gg, sm); │ │ │ │ │ + return new OpenLayers.LonLat(point.x, point.y); │ │ │ │ │ + }; │ │ │ │ │ + })(), │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: inverseMercator │ │ │ │ │ + * Given a x,y in Spherical Mercator, return a point in EPSG:4326. │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * x - {float} A map x in Spherical Mercator. │ │ │ │ │ + * y - {float} A map y in Spherical Mercator. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} A response object, whose "priv" property │ │ │ │ │ - * references the injected script. This object is also passed to the │ │ │ │ │ - * callback function when the request completes, its "features" property │ │ │ │ │ - * is then populated with the features received from the server. │ │ │ │ │ + * {<OpenLayers.LonLat>} The coordinates transformed to EPSG:4326. │ │ │ │ │ */ │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - options.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options.params, this.options.params │ │ │ │ │ + inverseMercator: (function() { │ │ │ │ │ + var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + var sm = new OpenLayers.Projection("EPSG:900913"); │ │ │ │ │ + return function(x, y) { │ │ │ │ │ + var point = OpenLayers.Projection.transform({ │ │ │ │ │ + x: x, │ │ │ │ │ + y: y │ │ │ │ │ + }, sm, gg); │ │ │ │ │ + return new OpenLayers.LonLat(point.x, point.y); │ │ │ │ │ + }; │ │ │ │ │ + })() │ │ │ │ │ + │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/WorldWind.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.WorldWind │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.WorldWind = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + │ │ │ │ │ + DEFAULT_PARAMS: {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} WorldWind layer is a base layer by default. │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: lzd │ │ │ │ │ + * {Float} LevelZeroTileSizeDegrees │ │ │ │ │ + */ │ │ │ │ │ + lzd: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomLevels │ │ │ │ │ + * {Integer} Number of zoom levels. │ │ │ │ │ + */ │ │ │ │ │ + zoomLevels: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.WorldWind │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} Name of Layer │ │ │ │ │ + * url - {String} Base URL │ │ │ │ │ + * lzd - {Float} Level zero tile size degrees │ │ │ │ │ + * zoomLevels - {Integer} number of zoom levels │ │ │ │ │ + * params - {Object} additional parameters │ │ │ │ │ + * options - {Object} additional options │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, lzd, zoomLevels, params, options) { │ │ │ │ │ + this.lzd = lzd; │ │ │ │ │ + this.zoomLevels = zoomLevels; │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + newArguments.push(name, url, params, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + this.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, this.DEFAULT_PARAMS │ │ │ │ │ ); │ │ │ │ │ - if (options.filter && this.filterToParams) { │ │ │ │ │ - options.params = this.filterToParams( │ │ │ │ │ - options.filter, options.params │ │ │ │ │ - ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getZoom │ │ │ │ │ + * Convert map zoom to WW zoom. │ │ │ │ │ + */ │ │ │ │ │ + getZoom: function() { │ │ │ │ │ + var zoom = this.map.getZoom(); │ │ │ │ │ + var extent = this.map.getMaxExtent(); │ │ │ │ │ + zoom = zoom - Math.log(this.maxResolution / (this.lzd / 512)) / Math.log(2); │ │ │ │ │ + return zoom; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as │ │ │ │ │ + * parameters │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var zoom = this.getZoom(); │ │ │ │ │ + var extent = this.map.getMaxExtent(); │ │ │ │ │ + var deg = this.lzd / Math.pow(2, this.getZoom()); │ │ │ │ │ + var x = Math.floor((bounds.left - extent.left) / deg); │ │ │ │ │ + var y = Math.floor((bounds.bottom - extent.bottom) / deg); │ │ │ │ │ + if (this.map.getResolution() <= (this.lzd / 512) && │ │ │ │ │ + this.getZoom() <= this.zoomLevels) { │ │ │ │ │ + return this.getFullRequestString({ │ │ │ │ │ + L: zoom, │ │ │ │ │ + X: x, │ │ │ │ │ + Y: y │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + return OpenLayers.Util.getImageLocation("blank.gif"); │ │ │ │ │ } │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - var request = this.createRequest( │ │ │ │ │ - options.url, │ │ │ │ │ - options.params, │ │ │ │ │ - OpenLayers.Function.bind(function(data) { │ │ │ │ │ - response.data = data; │ │ │ │ │ - this.handleRead(response, options); │ │ │ │ │ - }, this) │ │ │ │ │ - ); │ │ │ │ │ - response.priv = request; │ │ │ │ │ - return response; │ │ │ │ │ + │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.WorldWind" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/KaMap.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.KaMap │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.KaMap = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: filterToParams │ │ │ │ │ - * Optional method to translate an <OpenLayers.Filter> object into an object │ │ │ │ │ - * that can be serialized as request query string provided. If a custom │ │ │ │ │ - * method is not provided, any filter will not be serialized. │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} KaMap Layer is always a base layer │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: DEFAULT_PARAMS │ │ │ │ │ + * {Object} parameters set by default. The default parameters set │ │ │ │ │ + * the format via the 'i' parameter to 'jpeg'. │ │ │ │ │ + */ │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + i: 'jpeg', │ │ │ │ │ + map: '' │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.KaMap │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} filter to convert. │ │ │ │ │ - * params - {Object} The parameters object. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} │ │ │ │ │ + * url - {String} │ │ │ │ │ + * params - {Object} Parameters to be sent to the HTTP server in the │ │ │ │ │ + * query string for the tile. The format can be set via the 'i' │ │ │ │ │ + * parameter (defaults to jpg) , and the map should be set via │ │ │ │ │ + * the 'map' parameter. It has been reported that ka-Map may behave │ │ │ │ │ + * inconsistently if your format parameter does not match the format │ │ │ │ │ + * parameter configured in your config.php. (See ticket #327 for more │ │ │ │ │ + * information.) │ │ │ │ │ + * options - {Object} Additional options for the layer. Any of the │ │ │ │ │ + * APIProperties listed on this layer, and any layer types it │ │ │ │ │ + * extends, can be overridden through the options parameter. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, this.DEFAULT_PARAMS │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The resulting parameters object. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as │ │ │ │ │ + * parameters │ │ │ │ │ */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var mapRes = this.map.getResolution(); │ │ │ │ │ + var scale = Math.round((this.map.getScale() * 10000)) / 10000; │ │ │ │ │ + var pX = Math.round(bounds.left / mapRes); │ │ │ │ │ + var pY = -Math.round(bounds.top / mapRes); │ │ │ │ │ + return this.getFullRequestString({ │ │ │ │ │ + t: pY, │ │ │ │ │ + l: pX, │ │ │ │ │ + s: scale │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createRequest │ │ │ │ │ - * Issues a request for features by creating injecting a script in the │ │ │ │ │ - * document head. │ │ │ │ │ + * Method: calculateGridLayout │ │ │ │ │ + * ka-Map uses the center point of the map as an origin for │ │ │ │ │ + * its tiles. Override calculateGridLayout to center tiles │ │ │ │ │ + * correctly for this case. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * url - {String} Service URL. │ │ │ │ │ - * params - {Object} Query string parameters. │ │ │ │ │ - * callback - {Function} Callback to be called with resulting data. │ │ │ │ │ + * bounds - {<OpenLayers.Bound>} │ │ │ │ │ + * origin - {<OpenLayers.LonLat>} │ │ │ │ │ + * resolution - {Number} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {HTMLScriptElement} The script pending execution. │ │ │ │ │ + * {Object} Object containing properties tilelon, tilelat, startcol, │ │ │ │ │ + * startrow │ │ │ │ │ */ │ │ │ │ │ - createRequest: function(url, params, callback) { │ │ │ │ │ - var id = OpenLayers.Protocol.Script.register(callback); │ │ │ │ │ - var name = OpenLayers.String.format(this.callbackTemplate, { │ │ │ │ │ - id: id │ │ │ │ │ - }); │ │ │ │ │ - params = OpenLayers.Util.extend({}, params); │ │ │ │ │ - params[this.callbackKey] = this.callbackPrefix + name; │ │ │ │ │ - url = OpenLayers.Util.urlAppend( │ │ │ │ │ - url, OpenLayers.Util.getParameterString(params) │ │ │ │ │ - ); │ │ │ │ │ - var script = document.createElement("script"); │ │ │ │ │ - script.type = "text/javascript"; │ │ │ │ │ - script.src = url; │ │ │ │ │ - script.id = "OpenLayers_Protocol_Script_" + id; │ │ │ │ │ - this.pendingRequests[script.id] = script; │ │ │ │ │ - var head = document.getElementsByTagName("head")[0]; │ │ │ │ │ - head.appendChild(script); │ │ │ │ │ - return script; │ │ │ │ │ + calculateGridLayout: function(bounds, origin, resolution) { │ │ │ │ │ + var tilelon = resolution * this.tileSize.w; │ │ │ │ │ + var tilelat = resolution * this.tileSize.h; │ │ │ │ │ + │ │ │ │ │ + var offsetlon = bounds.left; │ │ │ │ │ + var tilecol = Math.floor(offsetlon / tilelon) - this.buffer; │ │ │ │ │ + │ │ │ │ │ + var offsetlat = bounds.top; │ │ │ │ │ + var tilerow = Math.floor(offsetlat / tilelat) + this.buffer; │ │ │ │ │ + │ │ │ │ │ + return { │ │ │ │ │ + tilelon: tilelon, │ │ │ │ │ + tilelat: tilelat, │ │ │ │ │ + startcol: tilecol, │ │ │ │ │ + startrow: tilerow │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroyRequest │ │ │ │ │ - * Remove a script node associated with a response from the document. Also │ │ │ │ │ - * unregisters the callback and removes the script from the │ │ │ │ │ - * <pendingRequests> object. │ │ │ │ │ + /** │ │ │ │ │ + * Method: getTileBoundsForGridIndex │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * script - {HTMLScriptElement} │ │ │ │ │ + * row - {Number} The row of the grid │ │ │ │ │ + * col - {Number} The column of the grid │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} The bounds for the tile at (row, col) │ │ │ │ │ */ │ │ │ │ │ - destroyRequest: function(script) { │ │ │ │ │ - OpenLayers.Protocol.Script.unregister(script.id.split("_").pop()); │ │ │ │ │ - delete this.pendingRequests[script.id]; │ │ │ │ │ - if (script.parentNode) { │ │ │ │ │ - script.parentNode.removeChild(script); │ │ │ │ │ + getTileBoundsForGridIndex: function(row, col) { │ │ │ │ │ + var origin = this.getTileOrigin(); │ │ │ │ │ + var tileLayout = this.gridLayout; │ │ │ │ │ + var tilelon = tileLayout.tilelon; │ │ │ │ │ + var tilelat = tileLayout.tilelat; │ │ │ │ │ + var minX = (tileLayout.startcol + col) * tilelon; │ │ │ │ │ + var minY = (tileLayout.startrow - row) * tilelat; │ │ │ │ │ + return new OpenLayers.Bounds( │ │ │ │ │ + minX, minY, │ │ │ │ │ + minX + tilelon, minY + tilelat │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Kamap>} An exact clone of this OpenLayers.Layer.KaMap │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.KaMap(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.params, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + if (this.tileSize != null) { │ │ │ │ │ + obj.tileSize = this.tileSize.clone(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // we do not want to copy reference to grid, so we make a new array │ │ │ │ │ + obj.grid = []; │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleRead │ │ │ │ │ - * Individual callbacks are created for read, create and update, should │ │ │ │ │ - * a subclass need to override each one separately. │ │ │ │ │ + * APIMethod: getTileBounds │ │ │ │ │ + * Returns The tile bounds for a layer given a pixel location. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * the user callback. │ │ │ │ │ - * options - {Object} The user options passed to the read call. │ │ │ │ │ + * viewPortPx - {<OpenLayers.Pixel>} The location in the viewport. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} Bounds of the tile at the given pixel location. │ │ │ │ │ */ │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - this.handleResponse(response, options); │ │ │ │ │ + getTileBounds: function(viewPortPx) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var tileMapWidth = resolution * this.tileSize.w; │ │ │ │ │ + var tileMapHeight = resolution * this.tileSize.h; │ │ │ │ │ + var mapPoint = this.getLonLatFromViewPortPx(viewPortPx); │ │ │ │ │ + var tileLeft = tileMapWidth * Math.floor(mapPoint.lon / tileMapWidth); │ │ │ │ │ + var tileBottom = tileMapHeight * Math.floor(mapPoint.lat / tileMapHeight); │ │ │ │ │ + return new OpenLayers.Bounds(tileLeft, tileBottom, │ │ │ │ │ + tileLeft + tileMapWidth, │ │ │ │ │ + tileBottom + tileMapHeight); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.KaMap" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/KaMapCache.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/Layer/KaMap.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.KaMapCache │ │ │ │ │ + * │ │ │ │ │ + * This class is designed to talk directly to a web-accessible ka-Map │ │ │ │ │ + * cache generated by the precache2.php script. │ │ │ │ │ + * │ │ │ │ │ + * To create a a new KaMapCache layer, you must indicate also the "i" parameter │ │ │ │ │ + * (that will be used to calculate the file extension), and another special │ │ │ │ │ + * parameter, object names "metaTileSize", with "h" (height) and "w" (width) │ │ │ │ │ + * properties. │ │ │ │ │ + * │ │ │ │ │ + * // Create a new kaMapCache layer. │ │ │ │ │ + * var kamap_base = new OpenLayers.Layer.KaMapCache( │ │ │ │ │ + * "Satellite", │ │ │ │ │ + * "http://www.example.org/web/acessible/cache", │ │ │ │ │ + * {g: "satellite", map: "world", i: 'png24', metaTileSize: {w: 5, h: 5} } │ │ │ │ │ + * ); │ │ │ │ │ + * │ │ │ │ │ + * // Create an kaMapCache overlay layer (using "isBaseLayer: false"). │ │ │ │ │ + * // Forces the output to be a "gif", using the "i" parameter. │ │ │ │ │ + * var kamap_overlay = new OpenLayers.Layer.KaMapCache( │ │ │ │ │ + * "Streets", │ │ │ │ │ + * "http://www.example.org/web/acessible/cache", │ │ │ │ │ + * {g: "streets", map: "world", i: "gif", metaTileSize: {w: 5, h: 5} }, │ │ │ │ │ + * {isBaseLayer: false} │ │ │ │ │ + * ); │ │ │ │ │ + * │ │ │ │ │ + * The cache URLs must look like: │ │ │ │ │ + * var/cache/World/50000/Group_Name/def/t-440320/l20480 │ │ │ │ │ + * │ │ │ │ │ + * This means that the cache generated via tile.php will *not* work with │ │ │ │ │ + * this class, and should instead use the KaMap layer. │ │ │ │ │ + * │ │ │ │ │ + * More information is available in Ticket #1518. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.KaMap> │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.KaMapCache = OpenLayers.Class(OpenLayers.Layer.KaMap, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: IMAGE_EXTENSIONS │ │ │ │ │ + * {Object} Simple hash map to convert format to extension. │ │ │ │ │ + */ │ │ │ │ │ + IMAGE_EXTENSIONS: { │ │ │ │ │ + 'jpeg': 'jpg', │ │ │ │ │ + 'gif': 'gif', │ │ │ │ │ + 'png': 'png', │ │ │ │ │ + 'png8': 'png', │ │ │ │ │ + 'png24': 'png', │ │ │ │ │ + 'dithered': 'png' │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleResponse │ │ │ │ │ - * Called by CRUD specific handlers. │ │ │ │ │ - * │ │ │ │ │ + * Constant: DEFAULT_FORMAT │ │ │ │ │ + * {Object} Simple hash map to convert format to extension. │ │ │ │ │ + */ │ │ │ │ │ + DEFAULT_FORMAT: 'jpeg', │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.KaMapCache │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the create, read, update, │ │ │ │ │ - * or delete call. │ │ │ │ │ + * name - {String} │ │ │ │ │ + * url - {String} │ │ │ │ │ + * params - {Object} Parameters to be sent to the HTTP server in the │ │ │ │ │ + * query string for the tile. The format can be set via the 'i' │ │ │ │ │ + * parameter (defaults to jpg) , and the map should be set via │ │ │ │ │ + * the 'map' parameter. It has been reported that ka-Map may behave │ │ │ │ │ + * inconsistently if your format parameter does not match the format │ │ │ │ │ + * parameter configured in your config.php. (See ticket #327 for more │ │ │ │ │ + * information.) │ │ │ │ │ + * options - {Object} Additional options for the layer. Any of the │ │ │ │ │ + * APIProperties listed on this layer, and any layer types it │ │ │ │ │ + * extends, can be overridden through the options parameter. │ │ │ │ │ */ │ │ │ │ │ - handleResponse: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - if (response.data) { │ │ │ │ │ - response.features = this.parseFeatures(response.data); │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - } │ │ │ │ │ - this.destroyRequest(response.priv); │ │ │ │ │ - options.callback.call(options.scope, response); │ │ │ │ │ - } │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.KaMap.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.extension = this.IMAGE_EXTENSIONS[this.params.i.toLowerCase() || this.DEFAULT_FORMAT]; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseFeatures │ │ │ │ │ - * Read Script response body and return features. │ │ │ │ │ - * │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {Object} The data sent to the callback function by the server. │ │ │ │ │ - * │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} Array of features or a single feature. │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as │ │ │ │ │ + * parameters │ │ │ │ │ */ │ │ │ │ │ - parseFeatures: function(data) { │ │ │ │ │ - return this.format.read(data); │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var mapRes = this.map.getResolution(); │ │ │ │ │ + var scale = Math.round((this.map.getScale() * 10000)) / 10000; │ │ │ │ │ + var pX = Math.round(bounds.left / mapRes); │ │ │ │ │ + var pY = -Math.round(bounds.top / mapRes); │ │ │ │ │ + │ │ │ │ │ + var metaX = Math.floor(pX / this.tileSize.w / this.params.metaTileSize.w) * this.tileSize.w * this.params.metaTileSize.w; │ │ │ │ │ + var metaY = Math.floor(pY / this.tileSize.h / this.params.metaTileSize.h) * this.tileSize.h * this.params.metaTileSize.h; │ │ │ │ │ + │ │ │ │ │ + var components = [ │ │ │ │ │ + "/", │ │ │ │ │ + this.params.map, │ │ │ │ │ + "/", │ │ │ │ │ + scale, │ │ │ │ │ + "/", │ │ │ │ │ + this.params.g.replace(/\s/g, '_'), │ │ │ │ │ + "/def/t", │ │ │ │ │ + metaY, │ │ │ │ │ + "/l", │ │ │ │ │ + metaX, │ │ │ │ │ + "/t", │ │ │ │ │ + pY, │ │ │ │ │ + "l", │ │ │ │ │ + pX, │ │ │ │ │ + ".", │ │ │ │ │ + this.extension │ │ │ │ │ + ]; │ │ │ │ │ + │ │ │ │ │ + var url = this.url; │ │ │ │ │ + │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(components.join(''), url); │ │ │ │ │ + } │ │ │ │ │ + return url + components.join(""); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.KaMapCache" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/ArcGIS93Rest.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.ArcGIS93Rest │ │ │ │ │ + * Instances of OpenLayers.Layer.ArcGIS93Rest are used to display data from │ │ │ │ │ + * ESRI ArcGIS Server 9.3 (and up?) Mapping Services using the REST API. │ │ │ │ │ + * Create a new ArcGIS93Rest layer with the <OpenLayers.Layer.ArcGIS93Rest> │ │ │ │ │ + * constructor. More detail on the REST API is available at │ │ │ │ │ + * http://sampleserver1.arcgisonline.com/ArcGIS/SDK/REST/index.html ; │ │ │ │ │ + * specifically, the URL provided to this layer should be an export service │ │ │ │ │ + * URL: http://sampleserver1.arcgisonline.com/ArcGIS/SDK/REST/export.html │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.ArcGIS93Rest = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: DEFAULT_PARAMS │ │ │ │ │ + * {Object} Hashtable of default parameter key/value pairs │ │ │ │ │ + */ │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + format: "png" │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: abort │ │ │ │ │ - * Abort an ongoing request. If no response is provided, all pending │ │ │ │ │ - * requests will be aborted. │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} Default is true for ArcGIS93Rest layer │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.ArcGIS93Rest │ │ │ │ │ + * Create a new ArcGIS93Rest layer object. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var arcims = new OpenLayers.Layer.ArcGIS93Rest("MyName", │ │ │ │ │ + * "http://sampleserver1.arcgisonline.com/ArcGIS/rest/services/Specialty/ESRI_StateCityHighway_USA/MapServer/export", │ │ │ │ │ + * { │ │ │ │ │ + * layers: "0,1,2" │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The response object returned │ │ │ │ │ - * from a <read> request. │ │ │ │ │ + * name - {String} A name for the layer │ │ │ │ │ + * url - {String} Base url for the ArcGIS server REST service │ │ │ │ │ + * options - {Object} An object with key/value pairs representing the │ │ │ │ │ + * options and option values. │ │ │ │ │ + * │ │ │ │ │ + * Valid Options: │ │ │ │ │ + * format - {String} MIME type of desired image type. │ │ │ │ │ + * layers - {String} Comma-separated list of layers to display. │ │ │ │ │ + * srs - {String} Projection ID. │ │ │ │ │ */ │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - this.destroyRequest(response.priv); │ │ │ │ │ - } else { │ │ │ │ │ - for (var key in this.pendingRequests) { │ │ │ │ │ - this.destroyRequest(this.pendingRequests[key]); │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + //uppercase params │ │ │ │ │ + params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ + newArguments.push(name, url, params, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, │ │ │ │ │ + OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS) │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + //layer is transparent │ │ │ │ │ + if (this.params.TRANSPARENT && │ │ │ │ │ + this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ + │ │ │ │ │ + // unless explicitly set in options, make layer an overlay │ │ │ │ │ + if ((options == null) || (!options.isBaseLayer)) { │ │ │ │ │ + this.isBaseLayer = false; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // jpegs can never be transparent, so intelligently switch the │ │ │ │ │ + // format, depending on the browser's capabilities │ │ │ │ │ + if (this.params.FORMAT == "jpg") { │ │ │ │ │ + this.params.FORMAT = OpenLayers.Util.alphaHack() ? "gif" : │ │ │ │ │ + "png"; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up the protocol. │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.ArcGIS93Rest>} An exact clone of this layer │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.abort(); │ │ │ │ │ - delete this.params; │ │ │ │ │ - delete this.format; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ - }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.Script" │ │ │ │ │ -}); │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.ArcGIS93Rest(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.params, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -(function() { │ │ │ │ │ - var o = OpenLayers.Protocol.Script; │ │ │ │ │ - var counter = 0; │ │ │ │ │ - o.registry = {}; │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Function: OpenLayers.Protocol.Script.register │ │ │ │ │ - * Register a callback for a newly created script. │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * Return an image url this layer. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * callback - {Function} The callback to be executed when the newly added │ │ │ │ │ - * script loads. This callback will be called with a single argument │ │ │ │ │ - * that is the JSON returned by the service. │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox for the │ │ │ │ │ + * request. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Number} An identifier for retrieving the registered callback. │ │ │ │ │ + * {String} A string with the map image's url. │ │ │ │ │ */ │ │ │ │ │ - o.register = function(callback) { │ │ │ │ │ - var id = "c" + (++counter); │ │ │ │ │ - o.registry[id] = function() { │ │ │ │ │ - callback.apply(this, arguments); │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + │ │ │ │ │ + // ArcGIS Server only wants the numeric portion of the projection ID. │ │ │ │ │ + var projWords = this.projection.getCode().split(":"); │ │ │ │ │ + var srid = projWords[projWords.length - 1]; │ │ │ │ │ + │ │ │ │ │ + var imageSize = this.getImageSize(); │ │ │ │ │ + var newParams = { │ │ │ │ │ + 'BBOX': bounds.toBBOX(), │ │ │ │ │ + 'SIZE': imageSize.w + "," + imageSize.h, │ │ │ │ │ + // We always want image, the other options were json, image with a whole lotta html around it, etc. │ │ │ │ │ + 'F': "image", │ │ │ │ │ + 'BBOXSR': srid, │ │ │ │ │ + 'IMAGESR': srid │ │ │ │ │ }; │ │ │ │ │ - return id; │ │ │ │ │ - }; │ │ │ │ │ + │ │ │ │ │ + // Now add the filter parameters. │ │ │ │ │ + if (this.layerDefs) { │ │ │ │ │ + var layerDefStrList = []; │ │ │ │ │ + var layerID; │ │ │ │ │ + for (layerID in this.layerDefs) { │ │ │ │ │ + if (this.layerDefs.hasOwnProperty(layerID)) { │ │ │ │ │ + if (this.layerDefs[layerID]) { │ │ │ │ │ + layerDefStrList.push(layerID); │ │ │ │ │ + layerDefStrList.push(":"); │ │ │ │ │ + layerDefStrList.push(this.layerDefs[layerID]); │ │ │ │ │ + layerDefStrList.push(";"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (layerDefStrList.length > 0) { │ │ │ │ │ + newParams['LAYERDEFS'] = layerDefStrList.join(""); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var requestString = this.getFullRequestString(newParams); │ │ │ │ │ + return requestString; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Function: OpenLayers.Protocol.Script.unregister │ │ │ │ │ - * Unregister a callback previously registered with the register function. │ │ │ │ │ + * Method: setLayerFilter │ │ │ │ │ + * addTile creates a tile, initializes it, and adds it to the layer div. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * id - {Number} The identifer returned by the register function. │ │ │ │ │ + * id - {String} The id of the layer to which the filter applies. │ │ │ │ │ + * queryDef - {String} A sql-ish query filter, for more detail see the ESRI │ │ │ │ │ + * documentation at http://sampleserver1.arcgisonline.com/ArcGIS/SDK/REST/export.html │ │ │ │ │ */ │ │ │ │ │ - o.unregister = function(id) { │ │ │ │ │ - delete o.registry[id]; │ │ │ │ │ - }; │ │ │ │ │ -})(); │ │ │ │ │ + setLayerFilter: function(id, queryDef) { │ │ │ │ │ + if (!this.layerDefs) { │ │ │ │ │ + this.layerDefs = {}; │ │ │ │ │ + } │ │ │ │ │ + if (queryDef) { │ │ │ │ │ + this.layerDefs[id] = queryDef; │ │ │ │ │ + } else { │ │ │ │ │ + delete this.layerDefs[id]; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clearLayerFilter │ │ │ │ │ + * Clears layer filters, either from a specific layer, │ │ │ │ │ + * or all of them. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} The id of the layer from which to remove any │ │ │ │ │ + * filter. If unspecified/blank, all filters │ │ │ │ │ + * will be removed. │ │ │ │ │ + */ │ │ │ │ │ + clearLayerFilter: function(id) { │ │ │ │ │ + if (id) { │ │ │ │ │ + delete this.layerDefs[id]; │ │ │ │ │ + } else { │ │ │ │ │ + delete this.layerDefs; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: mergeNewParams │ │ │ │ │ + * Catch changeParams and uppercase the new params to be merged in │ │ │ │ │ + * before calling changeParams on the super class. │ │ │ │ │ + * │ │ │ │ │ + * Once params have been changed, the tiles will be reloaded with │ │ │ │ │ + * the new parameters. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newParams - {Object} Hashtable of new params to use │ │ │ │ │ + */ │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ + var newArguments = [upperParams]; │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, │ │ │ │ │ + newArguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.ArcGIS93Rest" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/WFS/v1.js │ │ │ │ │ + OpenLayers/Layer/Image.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Protocol/WFS.js │ │ │ │ │ + * @requires OpenLayers/Layer.js │ │ │ │ │ + * @requires OpenLayers/Tile/Image.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Protocol.WFS.v1 │ │ │ │ │ - * Abstract class for for v1.0.0 and v1.1.0 protocol. │ │ │ │ │ + * Class: OpenLayers.Layer.Image │ │ │ │ │ + * Instances of OpenLayers.Layer.Image are used to display data from a web │ │ │ │ │ + * accessible image as a map layer. Create a new image layer with the │ │ │ │ │ + * <OpenLayers.Layer.Image> constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol> │ │ │ │ │ + * - <OpenLayers.Layer> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.WFS.v1 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} WFS version number. │ │ │ │ │ - */ │ │ │ │ │ - version: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: srsName │ │ │ │ │ - * {String} Name of spatial reference system. Default is "EPSG:4326". │ │ │ │ │ - */ │ │ │ │ │ - srsName: "EPSG:4326", │ │ │ │ │ +OpenLayers.Layer.Image = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: featureType │ │ │ │ │ - * {String} Local feature typeName. │ │ │ │ │ + * Property: isBaseLayer │ │ │ │ │ + * {Boolean} The layer is a base layer. Default is true. Set this property │ │ │ │ │ + * in the layer options │ │ │ │ │ */ │ │ │ │ │ - featureType: null, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: featureNS │ │ │ │ │ - * {String} Feature namespace. │ │ │ │ │ + * Property: url │ │ │ │ │ + * {String} URL of the image to use │ │ │ │ │ */ │ │ │ │ │ - featureNS: null, │ │ │ │ │ + url: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: geometryName │ │ │ │ │ - * {String} Name of the geometry attribute for features. Default is │ │ │ │ │ - * "the_geom" for WFS <version> 1.0, and null for higher versions. │ │ │ │ │ + * Property: extent │ │ │ │ │ + * {<OpenLayers.Bounds>} The image bounds in map units. This extent will │ │ │ │ │ + * also be used as the default maxExtent for the layer. If you wish │ │ │ │ │ + * to have a maxExtent that is different than the image extent, set the │ │ │ │ │ + * maxExtent property of the options argument (as with any other layer). │ │ │ │ │ */ │ │ │ │ │ - geometryName: "the_geom", │ │ │ │ │ + extent: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: maxFeatures │ │ │ │ │ - * {Integer} Optional maximum number of features to retrieve. │ │ │ │ │ + * Property: size │ │ │ │ │ + * {<OpenLayers.Size>} The image size in pixels │ │ │ │ │ */ │ │ │ │ │ + size: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: schema │ │ │ │ │ - * {String} Optional schema location that will be included in the │ │ │ │ │ - * schemaLocation attribute value. Note that the feature type schema │ │ │ │ │ - * is required for a strict XML validator (on transactions with an │ │ │ │ │ - * insert for example), but is *not* required by the WFS specification │ │ │ │ │ - * (since the server is supposed to know about feature type schemas). │ │ │ │ │ + * Property: tile │ │ │ │ │ + * {<OpenLayers.Tile.Image>} │ │ │ │ │ */ │ │ │ │ │ - schema: null, │ │ │ │ │ + tile: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: featurePrefix │ │ │ │ │ - * {String} Namespace alias for feature type. Default is "feature". │ │ │ │ │ + * Property: aspectRatio │ │ │ │ │ + * {Float} The ratio of height/width represented by a single pixel in the │ │ │ │ │ + * graphic │ │ │ │ │ */ │ │ │ │ │ - featurePrefix: "feature", │ │ │ │ │ + aspectRatio: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: formatOptions │ │ │ │ │ - * {Object} Optional options for the format. If a format is not provided, │ │ │ │ │ - * this property can be used to extend the default format options. │ │ │ │ │ + * Constructor: OpenLayers.Layer.Image │ │ │ │ │ + * Create a new image layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} A name for the layer. │ │ │ │ │ + * url - {String} Relative or absolute path to the image │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} The extent represented by the image │ │ │ │ │ + * size - {<OpenLayers.Size>} The size (in pixels) of the image │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ */ │ │ │ │ │ - formatOptions: null, │ │ │ │ │ + initialize: function(name, url, extent, size, options) { │ │ │ │ │ + this.url = url; │ │ │ │ │ + this.extent = extent; │ │ │ │ │ + this.maxExtent = extent; │ │ │ │ │ + this.size = size; │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readFormat │ │ │ │ │ - * {<OpenLayers.Format>} For WFS requests it is possible to get a │ │ │ │ │ - * different output format than GML. In that case, we cannot parse │ │ │ │ │ - * the response with the default format (WFST) and we need a different │ │ │ │ │ - * format for reading. │ │ │ │ │ - */ │ │ │ │ │ - readFormat: null, │ │ │ │ │ + this.aspectRatio = (this.extent.getHeight() / this.size.h) / │ │ │ │ │ + (this.extent.getWidth() / this.size.w); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: readOptions │ │ │ │ │ - * {Object} Optional object to pass to format's read. │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Destroy this layer │ │ │ │ │ */ │ │ │ │ │ - readOptions: null, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.tile) { │ │ │ │ │ + this.removeTileMonitoringHooks(this.tile); │ │ │ │ │ + this.tile.destroy(); │ │ │ │ │ + this.tile = null; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.WFS │ │ │ │ │ - * A class for giving layers WFS protocol. │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * Paramters: │ │ │ │ │ + * obj - {Object} An optional layer (is this ever used?) │ │ │ │ │ * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * url - {String} URL to send requests to (required). │ │ │ │ │ - * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ - * featureNS - {String} Feature namespace (required, but can be autodetected │ │ │ │ │ - * during the first query if GML is used as readFormat and │ │ │ │ │ - * featurePrefix is provided and matches the prefix used by the server │ │ │ │ │ - * for this featureType). │ │ │ │ │ - * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ - * for writing if featureNS is provided). Default is 'feature'. │ │ │ │ │ - * geometryName - {String} Name of geometry attribute. The default is │ │ │ │ │ - * 'the_geom' for WFS <version> 1.0, and null for higher versions. If │ │ │ │ │ - * null, it will be set to the name of the first geometry found in the │ │ │ │ │ - * first read operation. │ │ │ │ │ - * multi - {Boolean} If set to true, geometries will be casted to Multi │ │ │ │ │ - * geometries before they are written in a transaction. No casting will │ │ │ │ │ - * be done when reading features. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Image>} An exact copy of this layer │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!options.format) { │ │ │ │ │ - this.format = OpenLayers.Format.WFST(OpenLayers.Util.extend({ │ │ │ │ │ - version: this.version, │ │ │ │ │ - featureType: this.featureType, │ │ │ │ │ - featureNS: this.featureNS, │ │ │ │ │ - featurePrefix: this.featurePrefix, │ │ │ │ │ - geometryName: this.geometryName, │ │ │ │ │ - srsName: this.srsName, │ │ │ │ │ - schema: this.schema │ │ │ │ │ - }, this.formatOptions)); │ │ │ │ │ - } │ │ │ │ │ - if (!options.geometryName && parseFloat(this.format.version) > 1.0) { │ │ │ │ │ - this.setGeometryName(null); │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Image(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.extent, │ │ │ │ │ + this.size, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ + │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up the protocol. │ │ │ │ │ + * APIMethod: setMap │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.options && !this.options.format) { │ │ │ │ │ - this.format.destroy(); │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + /** │ │ │ │ │ + * If nothing to do with resolutions has been set, assume a single │ │ │ │ │ + * resolution determined by ratio*extent/size - if an image has a │ │ │ │ │ + * pixel aspect ratio different than one (as calculated above), the │ │ │ │ │ + * image will be stretched in one dimension only. │ │ │ │ │ + */ │ │ │ │ │ + if (this.options.maxResolution == null) { │ │ │ │ │ + this.options.maxResolution = this.aspectRatio * │ │ │ │ │ + this.extent.getWidth() / │ │ │ │ │ + this.size.w; │ │ │ │ │ } │ │ │ │ │ - this.format = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ + OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Construct a request for reading new features. Since WFS splits the │ │ │ │ │ - * basic CRUD operations into GetFeature requests (for read) and │ │ │ │ │ - * Transactions (for all others), this method does not make use of the │ │ │ │ │ - * format's read method (that is only about reading transaction │ │ │ │ │ - * responses). │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * Create the tile for the image or resize it for the new resolution │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Options for the read operation, in addition to the │ │ │ │ │ - * options set on the instance (options set here will take precedence). │ │ │ │ │ - * │ │ │ │ │ - * To use a configured protocol to get e.g. a WFS hit count, applications │ │ │ │ │ - * could do the following: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * protocol.read({ │ │ │ │ │ - * readOptions: {output: "object"}, │ │ │ │ │ - * resultType: "hits", │ │ │ │ │ - * maxFeatures: null, │ │ │ │ │ - * callback: function(resp) { │ │ │ │ │ - * // process resp.numberOfFeatures here │ │ │ │ │ - * } │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * To use a configured protocol to use WFS paging (if supported by the │ │ │ │ │ - * server), applications could do the following: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * protocol.read({ │ │ │ │ │ - * startIndex: 0, │ │ │ │ │ - * count: 50 │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * To limit the attributes returned by the GetFeature request, applications │ │ │ │ │ - * can use the propertyNames option to specify the properties to include in │ │ │ │ │ - * the response: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * protocol.read({ │ │ │ │ │ - * propertyNames: ["DURATION", "INTENSITY"] │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * zoomChanged - {Boolean} │ │ │ │ │ + * dragging - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - var data = OpenLayers.Format.XML.prototype.write.apply( │ │ │ │ │ - this.format, [this.format.writeNode("wfs:GetFeature", options)] │ │ │ │ │ - ); │ │ │ │ │ + var firstRendering = (this.tile == null); │ │ │ │ │ │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: data │ │ │ │ │ - }); │ │ │ │ │ + if (zoomChanged || firstRendering) { │ │ │ │ │ │ │ │ │ │ - return response; │ │ │ │ │ + //determine new tile size │ │ │ │ │ + this.setTileSize(); │ │ │ │ │ + │ │ │ │ │ + //determine new position (upper left corner of new bounds) │ │ │ │ │ + var ulPx = this.map.getLayerPxFromLonLat({ │ │ │ │ │ + lon: this.extent.left, │ │ │ │ │ + lat: this.extent.top │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + if (firstRendering) { │ │ │ │ │ + //create the new tile │ │ │ │ │ + this.tile = new OpenLayers.Tile.Image(this, ulPx, this.extent, │ │ │ │ │ + null, this.tileSize); │ │ │ │ │ + this.addTileMonitoringHooks(this.tile); │ │ │ │ │ + } else { │ │ │ │ │ + //just resize the tile and set it's new position │ │ │ │ │ + this.tile.size = this.tileSize.clone(); │ │ │ │ │ + this.tile.position = ulPx.clone(); │ │ │ │ │ + } │ │ │ │ │ + this.tile.draw(); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setFeatureType │ │ │ │ │ - * Change the feature type on the fly. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureType - {String} Local (without prefix) feature typeName. │ │ │ │ │ + * Set the tile size based on the map size. │ │ │ │ │ */ │ │ │ │ │ - setFeatureType: function(featureType) { │ │ │ │ │ - this.featureType = featureType; │ │ │ │ │ - this.format.featureType = featureType; │ │ │ │ │ + setTileSize: function() { │ │ │ │ │ + var tileWidth = this.extent.getWidth() / this.map.getResolution(); │ │ │ │ │ + var tileHeight = this.extent.getHeight() / this.map.getResolution(); │ │ │ │ │ + this.tileSize = new OpenLayers.Size(tileWidth, tileHeight); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: addTileMonitoringHooks │ │ │ │ │ + * This function takes a tile as input and adds the appropriate hooks to │ │ │ │ │ + * the tile so that the layer can keep track of the loading tiles. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * tile - {<OpenLayers.Tile>} │ │ │ │ │ + */ │ │ │ │ │ + addTileMonitoringHooks: function(tile) { │ │ │ │ │ + tile.onLoadStart = function() { │ │ │ │ │ + this.events.triggerEvent("loadstart"); │ │ │ │ │ + }; │ │ │ │ │ + tile.events.register("loadstart", this, tile.onLoadStart); │ │ │ │ │ + │ │ │ │ │ + tile.onLoadEnd = function() { │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + }; │ │ │ │ │ + tile.events.register("loadend", this, tile.onLoadEnd); │ │ │ │ │ + tile.events.register("unload", this, tile.onLoadEnd); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeTileMonitoringHooks │ │ │ │ │ + * This function takes a tile as input and removes the tile hooks │ │ │ │ │ + * that were added in <addTileMonitoringHooks>. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * tile - {<OpenLayers.Tile>} │ │ │ │ │ + */ │ │ │ │ │ + removeTileMonitoringHooks: function(tile) { │ │ │ │ │ + tile.unload(); │ │ │ │ │ + tile.events.un({ │ │ │ │ │ + "loadstart": tile.onLoadStart, │ │ │ │ │ + "loadend": tile.onLoadEnd, │ │ │ │ │ + "unload": tile.onLoadEnd, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setGeometryName │ │ │ │ │ - * Sets the geometryName option after instantiation. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: setUrl │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometryName - {String} Name of geometry attribute. │ │ │ │ │ + * newUrl - {String} │ │ │ │ │ */ │ │ │ │ │ - setGeometryName: function(geometryName) { │ │ │ │ │ - this.geometryName = geometryName; │ │ │ │ │ - this.format.geometryName = geometryName; │ │ │ │ │ + setUrl: function(newUrl) { │ │ │ │ │ + this.url = newUrl; │ │ │ │ │ + this.tile.draw(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleRead │ │ │ │ │ - * Deal with response from the read request. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getURL │ │ │ │ │ + * The url we return is always the same (the image itself never changes) │ │ │ │ │ + * so we can ignore the bounds parameter (it will always be the same, │ │ │ │ │ + * anyways) │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ - * to the user callback. │ │ │ │ │ - * options - {Object} The user options passed to the read call. │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ */ │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + return this.url; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - // success │ │ │ │ │ - var result = this.parseResponse(request, options.readOptions); │ │ │ │ │ - if (result && result.success !== false) { │ │ │ │ │ - if (options.readOptions && options.readOptions.output == "object") { │ │ │ │ │ - OpenLayers.Util.extend(response, result); │ │ │ │ │ - } else { │ │ │ │ │ - response.features = result; │ │ │ │ │ - } │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Image" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/Google.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/SphericalMercator.js │ │ │ │ │ + * @requires OpenLayers/Layer/EventPane.js │ │ │ │ │ + * @requires OpenLayers/Layer/FixedZoomLevels.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.Google │ │ │ │ │ + * │ │ │ │ │ + * Provides a wrapper for Google's Maps API │ │ │ │ │ + * Normally the Terms of Use for this API do not allow wrapping, but Google │ │ │ │ │ + * have provided written consent to OpenLayers for this - see email in │ │ │ │ │ + * http://osgeo-org.1560.n6.nabble.com/Google-Maps-API-Terms-of-Use-changes-tp4910013p4911981.html │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.SphericalMercator> │ │ │ │ │ + * - <OpenLayers.Layer.EventPane> │ │ │ │ │ + * - <OpenLayers.Layer.FixedZoomLevels> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Google = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Layer.EventPane, │ │ │ │ │ + OpenLayers.Layer.FixedZoomLevels, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: MIN_ZOOM_LEVEL │ │ │ │ │ + * {Integer} 0 │ │ │ │ │ + */ │ │ │ │ │ + MIN_ZOOM_LEVEL: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: MAX_ZOOM_LEVEL │ │ │ │ │ + * {Integer} 21 │ │ │ │ │ + */ │ │ │ │ │ + MAX_ZOOM_LEVEL: 21, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: RESOLUTIONS │ │ │ │ │ + * {Array(Float)} Hardcode these resolutions so that they are more closely │ │ │ │ │ + * tied with the standard wms projection │ │ │ │ │ + */ │ │ │ │ │ + RESOLUTIONS: [ │ │ │ │ │ + 1.40625, │ │ │ │ │ + 0.703125, │ │ │ │ │ + 0.3515625, │ │ │ │ │ + 0.17578125, │ │ │ │ │ + 0.087890625, │ │ │ │ │ + 0.0439453125, │ │ │ │ │ + 0.02197265625, │ │ │ │ │ + 0.010986328125, │ │ │ │ │ + 0.0054931640625, │ │ │ │ │ + 0.00274658203125, │ │ │ │ │ + 0.001373291015625, │ │ │ │ │ + 0.0006866455078125, │ │ │ │ │ + 0.00034332275390625, │ │ │ │ │ + 0.000171661376953125, │ │ │ │ │ + 0.0000858306884765625, │ │ │ │ │ + 0.00004291534423828125, │ │ │ │ │ + 0.00002145767211914062, │ │ │ │ │ + 0.00001072883605957031, │ │ │ │ │ + 0.00000536441802978515, │ │ │ │ │ + 0.00000268220901489257, │ │ │ │ │ + 0.0000013411045074462891, │ │ │ │ │ + 0.00000067055225372314453 │ │ │ │ │ + ], │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: type │ │ │ │ │ + * {GMapType} │ │ │ │ │ + */ │ │ │ │ │ + type: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: wrapDateLine │ │ │ │ │ + * {Boolean} Allow user to pan forever east/west. Default is true. │ │ │ │ │ + * Setting this to false only restricts panning if │ │ │ │ │ + * <sphericalMercator> is true. │ │ │ │ │ + */ │ │ │ │ │ + wrapDateLine: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: sphericalMercator │ │ │ │ │ + * {Boolean} Should the map act as a mercator-projected map? This will │ │ │ │ │ + * cause all interactions with the map to be in the actual map │ │ │ │ │ + * projection, which allows support for vector drawing, overlaying │ │ │ │ │ + * other maps, etc. │ │ │ │ │ + */ │ │ │ │ │ + sphericalMercator: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {Number} The version of the Google Maps API │ │ │ │ │ + */ │ │ │ │ │ + version: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.Google │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} A name for the layer. │ │ │ │ │ + * options - {Object} An optional object whose properties will be set │ │ │ │ │ + * on the layer. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + if (!options.version) { │ │ │ │ │ + options.version = typeof GMap2 === "function" ? "2" : "3"; │ │ │ │ │ + } │ │ │ │ │ + var mixin = OpenLayers.Layer.Google["v" + │ │ │ │ │ + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (mixin) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, mixin); │ │ │ │ │ + } else { │ │ │ │ │ + throw "Unsupported Google Maps API version: " + options.version; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, mixin.DEFAULTS); │ │ │ │ │ + if (options.maxExtent) { │ │ │ │ │ + options.maxExtent = options.maxExtent.clone(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.initialize.apply(this, │ │ │ │ │ + [name, options]); │ │ │ │ │ + OpenLayers.Layer.FixedZoomLevels.prototype.initialize.apply(this, │ │ │ │ │ + [name, options]); │ │ │ │ │ + │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + OpenLayers.Util.extend(this, OpenLayers.Layer.SphericalMercator); │ │ │ │ │ + this.initMercatorParameters(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Google>} An exact clone of this layer │ │ │ │ │ + */ │ │ │ │ │ + clone: function() { │ │ │ │ │ + /** │ │ │ │ │ + * This method isn't intended to be called by a subclass and it │ │ │ │ │ + * doesn't call the same method on the superclass. We don't call │ │ │ │ │ + * the super's clone because we don't want properties that are set │ │ │ │ │ + * on this layer after initialize (i.e. this.mapObject etc.). │ │ │ │ │ + */ │ │ │ │ │ + return new OpenLayers.Layer.Google( │ │ │ │ │ + this.name, this.getOptions() │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setVisibility │ │ │ │ │ + * Set the visibility flag for the layer and hide/show & redraw │ │ │ │ │ + * accordingly. Fire event unless otherwise specified │ │ │ │ │ + * │ │ │ │ │ + * Note that visibility is no longer simply whether or not the layer's │ │ │ │ │ + * style.display is set to "block". Now we store a 'visibility' state │ │ │ │ │ + * property on the layer class, this allows us to remember whether or │ │ │ │ │ + * not we *desire* for a layer to be visible. In the case where the │ │ │ │ │ + * map's resolution is out of the layer's range, this desire may be │ │ │ │ │ + * subverted. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * visible - {Boolean} Display the layer (if in range) │ │ │ │ │ + */ │ │ │ │ │ + setVisibility: function(visible) { │ │ │ │ │ + // sharing a map container, opacity has to be set per layer │ │ │ │ │ + var opacity = this.opacity == null ? 1 : this.opacity; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.setVisibility.apply(this, arguments); │ │ │ │ │ + this.setOpacity(opacity); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: display │ │ │ │ │ + * Hide or show the Layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * visible - {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + display: function(visible) { │ │ │ │ │ + if (!this._dragging) { │ │ │ │ │ + this.setGMapVisibility(visible); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.display.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * zoomChanged - {Boolean} Tells when zoom has changed, as layers have to │ │ │ │ │ + * do some init work in that case. │ │ │ │ │ + * dragging - {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + this._dragging = dragging; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + delete this._dragging; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setOpacity │ │ │ │ │ + * Sets the opacity for the entire layer (all images) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * opacity - {Float} │ │ │ │ │ + */ │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + if (opacity !== this.opacity) { │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "opacity" │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + this.opacity = opacity; │ │ │ │ │ + } │ │ │ │ │ + // Though this layer's opacity may not change, we're sharing a container │ │ │ │ │ + // and need to update the opacity for the entire container. │ │ │ │ │ + if (this.getVisibility()) { │ │ │ │ │ + var container = this.getMapContainer(); │ │ │ │ │ + OpenLayers.Util.modifyDOMElement( │ │ │ │ │ + container, null, null, null, null, null, null, opacity │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up this layer. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + /** │ │ │ │ │ + * We have to override this method because the event pane destroy │ │ │ │ │ + * deletes the mapObject reference before removing this layer from │ │ │ │ │ + * the map. │ │ │ │ │ + */ │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.setGMapVisibility(false); │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache && cache.count <= 1) { │ │ │ │ │ + this.removeGMapElements(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeGMapElements │ │ │ │ │ + * Remove all elements added to the dom. This should only be called if │ │ │ │ │ + * this is the last of the Google layers for the given map. │ │ │ │ │ + */ │ │ │ │ │ + removeGMapElements: function() { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + // remove shared elements from dom │ │ │ │ │ + var container = this.mapObject && this.getMapContainer(); │ │ │ │ │ + if (container && container.parentNode) { │ │ │ │ │ + container.parentNode.removeChild(container); │ │ │ │ │ + } │ │ │ │ │ + var termsOfUse = cache.termsOfUse; │ │ │ │ │ + if (termsOfUse && termsOfUse.parentNode) { │ │ │ │ │ + termsOfUse.parentNode.removeChild(termsOfUse); │ │ │ │ │ + } │ │ │ │ │ + var poweredBy = cache.poweredBy; │ │ │ │ │ + if (poweredBy && poweredBy.parentNode) { │ │ │ │ │ + poweredBy.parentNode.removeChild(poweredBy); │ │ │ │ │ + } │ │ │ │ │ + if (this.mapObject && window.google && google.maps && │ │ │ │ │ + google.maps.event && google.maps.event.clearListeners) { │ │ │ │ │ + google.maps.event.clearListeners(this.mapObject, 'tilesloaded'); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: removeMap │ │ │ │ │ + * On being removed from the map, also remove termsOfUse and poweredBy divs │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + // hide layer before removing │ │ │ │ │ + if (this.visibility && this.mapObject) { │ │ │ │ │ + this.setGMapVisibility(false); │ │ │ │ │ + } │ │ │ │ │ + // check to see if last Google layer in this map │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + if (cache.count <= 1) { │ │ │ │ │ + this.removeGMapElements(); │ │ │ │ │ + delete OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ } else { │ │ │ │ │ - // failure (service exception) │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - response.error = result; │ │ │ │ │ + // decrement the layer count │ │ │ │ │ + --cache.count; │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - // failure │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ } │ │ │ │ │ - options.callback.call(options.scope, response); │ │ │ │ │ + // remove references to gmap elements │ │ │ │ │ + delete this.termsOfUse; │ │ │ │ │ + delete this.poweredBy; │ │ │ │ │ + delete this.mapObject; │ │ │ │ │ + delete this.dragObject; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.removeMap.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ + // │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getOLBoundsFromMapObjectBounds │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moBounds - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} An <OpenLayers.Bounds>, translated from the │ │ │ │ │ + * passed-in MapObject Bounds. │ │ │ │ │ + * Returns null if null value is passed in. │ │ │ │ │ + */ │ │ │ │ │ + getOLBoundsFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + var olBounds = null; │ │ │ │ │ + if (moBounds != null) { │ │ │ │ │ + var sw = moBounds.getSouthWest(); │ │ │ │ │ + var ne = moBounds.getNorthEast(); │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + sw = this.forwardMercator(sw.lng(), sw.lat()); │ │ │ │ │ + ne = this.forwardMercator(ne.lng(), ne.lat()); │ │ │ │ │ + } else { │ │ │ │ │ + sw = new OpenLayers.LonLat(sw.lng(), sw.lat()); │ │ │ │ │ + ne = new OpenLayers.LonLat(ne.lng(), ne.lat()); │ │ │ │ │ + } │ │ │ │ │ + olBounds = new OpenLayers.Bounds(sw.lon, │ │ │ │ │ + sw.lat, │ │ │ │ │ + ne.lon, │ │ │ │ │ + ne.lat); │ │ │ │ │ + } │ │ │ │ │ + return olBounds; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getWarningHTML │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} String with information on why layer is broken, how to get │ │ │ │ │ + * it working. │ │ │ │ │ + */ │ │ │ │ │ + getWarningHTML: function() { │ │ │ │ │ + return OpenLayers.i18n("googleWarning"); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Interface Controls * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + // Get&Set Center, Zoom │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectCenter │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The mapObject's current center in Map Object format │ │ │ │ │ + */ │ │ │ │ │ + getMapObjectCenter: function() { │ │ │ │ │ + return this.mapObject.getCenter(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectZoom │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} The mapObject's current zoom, in Map Object format │ │ │ │ │ + */ │ │ │ │ │ + getMapObjectZoom: function() { │ │ │ │ │ + return this.mapObject.getZoom(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Primitives * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + // LonLat │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getLongitudeFromMapObjectLonLat │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} Longitude of the given MapObject LonLat │ │ │ │ │ + */ │ │ │ │ │ + getLongitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + return this.sphericalMercator ? │ │ │ │ │ + this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lon : │ │ │ │ │ + moLonLat.lng(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getLatitudeFromMapObjectLonLat │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} Latitude of the given MapObject LonLat │ │ │ │ │ + */ │ │ │ │ │ + getLatitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + var lat = this.sphericalMercator ? │ │ │ │ │ + this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lat : │ │ │ │ │ + moLonLat.lat(); │ │ │ │ │ + return lat; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + // Pixel │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getXFromMapObjectPixel │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moPixel - {Object} MapObject Pixel format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} X value of the MapObject Pixel │ │ │ │ │ + */ │ │ │ │ │ + getXFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return moPixel.x; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getYFromMapObjectPixel │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moPixel - {Object} MapObject Pixel format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} Y value of the MapObject Pixel │ │ │ │ │ + */ │ │ │ │ │ + getYFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return moPixel.y; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Google" │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Property: OpenLayers.Layer.Google.cache │ │ │ │ │ + * {Object} Cache for elements that should only be created once per map. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Google.cache = {}; │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Layer.Google.v2 │ │ │ │ │ + * │ │ │ │ │ + * Mixin providing functionality specific to the Google Maps API v2. │ │ │ │ │ + * │ │ │ │ │ + * This API has been deprecated by Google. │ │ │ │ │ + * Developers are encouraged to migrate to v3 of the API; support for this │ │ │ │ │ + * is provided by <OpenLayers.Layer.Google.v3> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Google.v2 = { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: termsOfUse │ │ │ │ │ + * {DOMElement} Div for Google's copyright and terms of use link │ │ │ │ │ + */ │ │ │ │ │ + termsOfUse: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: poweredBy │ │ │ │ │ + * {DOMElement} Div for Google's powered by logo and link │ │ │ │ │ + */ │ │ │ │ │ + poweredBy: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: dragObject │ │ │ │ │ + * {GDraggableObject} Since 2.93, Google has exposed the ability to get │ │ │ │ │ + * the maps GDraggableObject. We can now use this for smooth panning │ │ │ │ │ + */ │ │ │ │ │ + dragObject: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: loadMapObject │ │ │ │ │ + * Load the GMap and register appropriate event listeners. If we can't │ │ │ │ │ + * load GMap2, then display a warning message. │ │ │ │ │ + */ │ │ │ │ │ + loadMapObject: function() { │ │ │ │ │ + if (!this.type) { │ │ │ │ │ + this.type = G_NORMAL_MAP; │ │ │ │ │ } │ │ │ │ │ + var mapObject, termsOfUse, poweredBy; │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + // there are already Google layers added to this map │ │ │ │ │ + mapObject = cache.mapObject; │ │ │ │ │ + termsOfUse = cache.termsOfUse; │ │ │ │ │ + poweredBy = cache.poweredBy; │ │ │ │ │ + // increment the layer count │ │ │ │ │ + ++cache.count; │ │ │ │ │ + } else { │ │ │ │ │ + // this is the first Google layer for this map │ │ │ │ │ + │ │ │ │ │ + var container = this.map.viewPortDiv; │ │ │ │ │ + var div = document.createElement("div"); │ │ │ │ │ + div.id = this.map.id + "_GMap2Container"; │ │ │ │ │ + div.style.position = "absolute"; │ │ │ │ │ + div.style.width = "100%"; │ │ │ │ │ + div.style.height = "100%"; │ │ │ │ │ + container.appendChild(div); │ │ │ │ │ + │ │ │ │ │ + // create GMap and shuffle elements │ │ │ │ │ + try { │ │ │ │ │ + mapObject = new GMap2(div); │ │ │ │ │ + │ │ │ │ │ + // move the ToS and branding stuff up to the container div │ │ │ │ │ + termsOfUse = div.lastChild; │ │ │ │ │ + container.appendChild(termsOfUse); │ │ │ │ │ + termsOfUse.style.zIndex = "1100"; │ │ │ │ │ + termsOfUse.style.right = ""; │ │ │ │ │ + termsOfUse.style.bottom = ""; │ │ │ │ │ + termsOfUse.className = "olLayerGoogleCopyright"; │ │ │ │ │ + │ │ │ │ │ + poweredBy = div.lastChild; │ │ │ │ │ + container.appendChild(poweredBy); │ │ │ │ │ + poweredBy.style.zIndex = "1100"; │ │ │ │ │ + poweredBy.style.right = ""; │ │ │ │ │ + poweredBy.style.bottom = ""; │ │ │ │ │ + poweredBy.className = "olLayerGooglePoweredBy gmnoprint"; │ │ │ │ │ + │ │ │ │ │ + } catch (e) { │ │ │ │ │ + throw (e); │ │ │ │ │ + } │ │ │ │ │ + // cache elements for use by any other google layers added to │ │ │ │ │ + // this same map │ │ │ │ │ + OpenLayers.Layer.Google.cache[this.map.id] = { │ │ │ │ │ + mapObject: mapObject, │ │ │ │ │ + termsOfUse: termsOfUse, │ │ │ │ │ + poweredBy: poweredBy, │ │ │ │ │ + count: 1 │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.mapObject = mapObject; │ │ │ │ │ + this.termsOfUse = termsOfUse; │ │ │ │ │ + this.poweredBy = poweredBy; │ │ │ │ │ + │ │ │ │ │ + // ensure this layer type is one of the mapObject types │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.mapObject.getMapTypes(), │ │ │ │ │ + this.type) === -1) { │ │ │ │ │ + this.mapObject.addMapType(this.type); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //since v 2.93 getDragObject is now available. │ │ │ │ │ + if (typeof mapObject.getDragObject == "function") { │ │ │ │ │ + this.dragObject = mapObject.getDragObject(); │ │ │ │ │ + } else { │ │ │ │ │ + this.dragPanMapObject = null; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.isBaseLayer === false) { │ │ │ │ │ + this.setGMapVisibility(this.div.style.display !== "none"); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseResponse │ │ │ │ │ - * Read HTTP response body and return features │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * request - {XMLHttpRequest} The request object │ │ │ │ │ - * options - {Object} Optional object to pass to format's read │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} or {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * An object with a features property, an array of features or a single │ │ │ │ │ - * feature. │ │ │ │ │ + * APIMethod: onMapResize │ │ │ │ │ */ │ │ │ │ │ - parseResponse: function(request, options) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null; │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + // workaround for resizing of invisible or not yet fully loaded layers │ │ │ │ │ + // where GMap2.checkResize() does not work. We need to load the GMap │ │ │ │ │ + // for the old div size, then checkResize(), and then call │ │ │ │ │ + // layer.moveTo() to trigger GMap.setCenter() (which will finish │ │ │ │ │ + // the GMap initialization). │ │ │ │ │ + if (this.visibility && this.mapObject.isLoaded()) { │ │ │ │ │ + this.mapObject.checkResize(); │ │ │ │ │ + } else { │ │ │ │ │ + if (!this._resized) { │ │ │ │ │ + var layer = this; │ │ │ │ │ + var handle = GEvent.addListener(this.mapObject, "load", function() { │ │ │ │ │ + GEvent.removeListener(handle); │ │ │ │ │ + delete layer._resized; │ │ │ │ │ + layer.mapObject.checkResize(); │ │ │ │ │ + layer.moveTo(layer.map.getCenter(), layer.map.getZoom()); │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + this._resized = true; │ │ │ │ │ } │ │ │ │ │ - var result = (this.readFormat !== null) ? this.readFormat.read(doc) : │ │ │ │ │ - this.format.read(doc, options); │ │ │ │ │ - if (!this.featureNS) { │ │ │ │ │ - var format = this.readFormat || this.format; │ │ │ │ │ - this.featureNS = format.featureNS; │ │ │ │ │ - // no need to auto-configure again on subsequent reads │ │ │ │ │ - format.autoConfig = false; │ │ │ │ │ - if (!this.geometryName) { │ │ │ │ │ - this.setGeometryName(format.geometryName); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setGMapVisibility │ │ │ │ │ + * Display the GMap container and associated elements. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * visible - {Boolean} Display the GMap elements. │ │ │ │ │ + */ │ │ │ │ │ + setGMapVisibility: function(visible) { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + var container = this.mapObject.getContainer(); │ │ │ │ │ + if (visible === true) { │ │ │ │ │ + this.mapObject.setMapType(this.type); │ │ │ │ │ + container.style.display = ""; │ │ │ │ │ + this.termsOfUse.style.left = ""; │ │ │ │ │ + this.termsOfUse.style.display = ""; │ │ │ │ │ + this.poweredBy.style.display = ""; │ │ │ │ │ + cache.displayed = this.id; │ │ │ │ │ + } else { │ │ │ │ │ + if (cache.displayed === this.id) { │ │ │ │ │ + delete cache.displayed; │ │ │ │ │ + } │ │ │ │ │ + if (!cache.displayed) { │ │ │ │ │ + container.style.display = "none"; │ │ │ │ │ + this.termsOfUse.style.display = "none"; │ │ │ │ │ + // move ToU far to the left in addition to setting display │ │ │ │ │ + // to "none", because at the end of the GMap2 load │ │ │ │ │ + // sequence, display: none will be unset and ToU would be │ │ │ │ │ + // visible after loading a map with a google layer that is │ │ │ │ │ + // initially hidden. │ │ │ │ │ + this.termsOfUse.style.left = "-9999px"; │ │ │ │ │ + this.poweredBy.style.display = "none"; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return result; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: commit │ │ │ │ │ - * Given a list of feature, assemble a batch request for update, create, │ │ │ │ │ - * and delete transactions. A commit call on the prototype amounts │ │ │ │ │ - * to writing a WFS transaction - so the write method on the format │ │ │ │ │ - * is used. │ │ │ │ │ - * │ │ │ │ │ + * Method: getMapContainer │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} the GMap container's div │ │ │ │ │ + */ │ │ │ │ │ + getMapContainer: function() { │ │ │ │ │ + return this.mapObject.getContainer(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ + // │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectBoundsFromOLBounds │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ - * options - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * nativeElements - {Array({Object})} Array of objects with information for writing │ │ │ │ │ - * out <Native> elements, these objects have vendorId, safeToIgnore and │ │ │ │ │ - * value properties. The <Native> element is intended to allow access to │ │ │ │ │ - * vendor specific capabilities of any particular web feature server or │ │ │ │ │ - * datastore. │ │ │ │ │ - * │ │ │ │ │ + * olBounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} A response object with a features │ │ │ │ │ - * property containing any insertIds and a priv property referencing │ │ │ │ │ - * the XMLHttpRequest object. │ │ │ │ │ + * {Object} A MapObject Bounds, translated from olBounds │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ */ │ │ │ │ │ - commit: function(features, options) { │ │ │ │ │ + getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ + var moBounds = null; │ │ │ │ │ + if (olBounds != null) { │ │ │ │ │ + var sw = this.sphericalMercator ? │ │ │ │ │ + this.inverseMercator(olBounds.bottom, olBounds.left) : │ │ │ │ │ + new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ + var ne = this.sphericalMercator ? │ │ │ │ │ + this.inverseMercator(olBounds.top, olBounds.right) : │ │ │ │ │ + new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ + moBounds = new GLatLngBounds(new GLatLng(sw.lat, sw.lon), │ │ │ │ │ + new GLatLng(ne.lat, ne.lon)); │ │ │ │ │ + } │ │ │ │ │ + return moBounds; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "commit", │ │ │ │ │ - reqFeatures: features │ │ │ │ │ - }); │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(features, options), │ │ │ │ │ - callback: this.createCallback(this.handleCommit, response, options) │ │ │ │ │ - }); │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Interface Controls * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ │ │ │ │ │ - return response; │ │ │ │ │ + │ │ │ │ │ + // Get&Set Center, Zoom │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setMapObjectCenter │ │ │ │ │ + * Set the mapObject to the specified center and zoom │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * center - {Object} MapObject LonLat format │ │ │ │ │ + * zoom - {int} MapObject zoom format │ │ │ │ │ + */ │ │ │ │ │ + setMapObjectCenter: function(center, zoom) { │ │ │ │ │ + this.mapObject.setCenter(center, zoom); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleCommit │ │ │ │ │ - * Called when the commit request returns. │ │ │ │ │ + * APIMethod: dragPanMapObject │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ - * to the user callback. │ │ │ │ │ - * options - {Object} The user options passed to the commit call. │ │ │ │ │ + * dX - {Integer} │ │ │ │ │ + * dY - {Integer} │ │ │ │ │ */ │ │ │ │ │ - handleCommit: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ + dragPanMapObject: function(dX, dY) { │ │ │ │ │ + this.dragObject.moveBy(new GSize(-dX, dY)); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // ensure that we have an xml doc │ │ │ │ │ - var data = request.responseXML; │ │ │ │ │ - if (!data || !data.documentElement) { │ │ │ │ │ - data = request.responseText; │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - var obj = this.format.read(data) || {}; │ │ │ │ │ + // LonLat - Pixel Translation │ │ │ │ │ │ │ │ │ │ - response.insertIds = obj.insertIds || []; │ │ │ │ │ - if (obj.success) { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - response.error = obj; │ │ │ │ │ - } │ │ │ │ │ - options.callback.call(options.scope, response); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectLonLatFromMapObjectPixel │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moPixel - {Object} MapObject Pixel format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject LonLat translated from MapObject Pixel │ │ │ │ │ + */ │ │ │ │ │ + getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return this.mapObject.fromContainerPixelToLatLng(moPixel); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: filterDelete │ │ │ │ │ - * Send a request that deletes all features by their filter. │ │ │ │ │ + * APIMethod: getMapObjectPixelFromMapObjectLonLat │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} filter │ │ │ │ │ + * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject Pixel transtlated from MapObject LonLat │ │ │ │ │ */ │ │ │ │ │ - filterDelete: function(filter, options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "commit" │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - var root = this.format.createElementNSPlus("wfs:Transaction", { │ │ │ │ │ - attributes: { │ │ │ │ │ - service: "WFS", │ │ │ │ │ - version: this.version │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - var deleteNode = this.format.createElementNSPlus("wfs:Delete", { │ │ │ │ │ - attributes: { │ │ │ │ │ - typeName: (options.featureNS ? this.featurePrefix + ":" : "") + │ │ │ │ │ - options.featureType │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ + getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + return this.mapObject.fromLatLngToContainerPixel(moLonLat); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (options.featureNS) { │ │ │ │ │ - deleteNode.setAttribute("xmlns:" + this.featurePrefix, options.featureNS); │ │ │ │ │ - } │ │ │ │ │ - var filterNode = this.format.writeNode("ogc:Filter", filter); │ │ │ │ │ │ │ │ │ │ - deleteNode.appendChild(filterNode); │ │ │ │ │ + // Bounds │ │ │ │ │ │ │ │ │ │ - root.appendChild(deleteNode); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectZoomFromMapObjectBounds │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moBounds - {Object} MapObject Bounds format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject Zoom for specified MapObject Bounds │ │ │ │ │ + */ │ │ │ │ │ + getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + return this.mapObject.getBoundsZoomLevel(moBounds); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var data = OpenLayers.Format.XML.prototype.write.apply( │ │ │ │ │ - this.format, [root] │ │ │ │ │ - ); │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Primitives * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ │ │ │ │ │ - return OpenLayers.Request.POST({ │ │ │ │ │ - url: this.url, │ │ │ │ │ - callback: options.callback || function() {}, │ │ │ │ │ - data: data │ │ │ │ │ - }); │ │ │ │ │ │ │ │ │ │ - }, │ │ │ │ │ + // LonLat │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: abort │ │ │ │ │ - * Abort an ongoing request, the response object passed to │ │ │ │ │ - * this method must come from this protocol (as a result │ │ │ │ │ - * of a read, or commit operation). │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getMapObjectLonLatFromLonLat │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ + * lon - {Float} │ │ │ │ │ + * lat - {Float} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject LonLat built from lon and lat params │ │ │ │ │ */ │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - response.priv.abort(); │ │ │ │ │ + getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ + var gLatLng; │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ + gLatLng = new GLatLng(lonlat.lat, lonlat.lon); │ │ │ │ │ + } else { │ │ │ │ │ + gLatLng = new GLatLng(lat, lon); │ │ │ │ │ } │ │ │ │ │ + return gLatLng; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.WFS.v1" │ │ │ │ │ -}); │ │ │ │ │ + // Pixel │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectPixelFromXY │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * x - {Integer} │ │ │ │ │ + * y - {Integer} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject Pixel from x and y parameters │ │ │ │ │ + */ │ │ │ │ │ + getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ + return new GPoint(x, y); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/WFS/v1_0_0.js │ │ │ │ │ + OpenLayers/Layer/GeoRSS.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Protocol/WFS/v1.js │ │ │ │ │ - * @requires OpenLayers/Format/WFST/v1_0_0.js │ │ │ │ │ + * @requires OpenLayers/Layer/Markers.js │ │ │ │ │ + * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Protocol.WFS.v1_0_0 │ │ │ │ │ - * A WFS v1.0.0 protocol for vector layers. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Protocol.WFS.v1_0_0> constructor. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Layer.GeoRSS │ │ │ │ │ + * Add GeoRSS Point features to your map. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol.WFS.v1> │ │ │ │ │ + * - <OpenLayers.Layer.Markers> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.WFS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ +OpenLayers.Layer.GeoRSS = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: location │ │ │ │ │ + * {String} store url of text file │ │ │ │ │ + */ │ │ │ │ │ + location: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Array(<OpenLayers.Feature>)} │ │ │ │ │ + */ │ │ │ │ │ + features: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} WFS version number. │ │ │ │ │ + * APIProperty: formatOptions │ │ │ │ │ + * {Object} Hash of options which should be passed to the format when it is │ │ │ │ │ + * created. Must be passed in the constructor. │ │ │ │ │ */ │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: selectedFeature │ │ │ │ │ + * {<OpenLayers.Feature>} │ │ │ │ │ + */ │ │ │ │ │ + selectedFeature: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: icon │ │ │ │ │ + * {<OpenLayers.Icon>}. This determines the Icon to be used on the map │ │ │ │ │ + * for this GeoRSS layer. │ │ │ │ │ + */ │ │ │ │ │ + icon: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.WFS.v1_0_0 │ │ │ │ │ - * A class for giving layers WFS v1.0.0 protocol. │ │ │ │ │ + * APIProperty: popupSize │ │ │ │ │ + * {<OpenLayers.Size>} This determines the size of GeoRSS popups. If │ │ │ │ │ + * not provided, defaults to 250px by 120px. │ │ │ │ │ + */ │ │ │ │ │ + popupSize: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: useFeedTitle │ │ │ │ │ + * {Boolean} Set layer.name to the first <title> element in the feed. Default is true. │ │ │ │ │ + */ │ │ │ │ │ + useFeedTitle: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.GeoRSS │ │ │ │ │ + * Create a GeoRSS Layer. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * name - {String} │ │ │ │ │ + * location - {String} │ │ │ │ │ + * options - {Object} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, location, options) { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ + this.location = location; │ │ │ │ │ + this.features = []; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + // Warning: Layer.Markers.destroy() must be called prior to calling │ │ │ │ │ + // clearFeatures() here, otherwise we leak memory. Indeed, if │ │ │ │ │ + // Layer.Markers.destroy() is called after clearFeatures(), it won't be │ │ │ │ │ + // able to remove the marker image elements from the layer's div since │ │ │ │ │ + // the markers will have been destroyed by clearFeatures(). │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.destroy.apply(this, arguments); │ │ │ │ │ + this.clearFeatures(); │ │ │ │ │ + this.features = null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: loadRSS │ │ │ │ │ + * Start the load of the RSS data. Don't do this when we first add the layer, │ │ │ │ │ + * since we may not be visible at any point, and it would therefore be a waste. │ │ │ │ │ + */ │ │ │ │ │ + loadRSS: function() { │ │ │ │ │ + if (!this.loaded) { │ │ │ │ │ + this.events.triggerEvent("loadstart"); │ │ │ │ │ + OpenLayers.Request.GET({ │ │ │ │ │ + url: this.location, │ │ │ │ │ + success: this.parseData, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.loaded = true; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * If layer is visible and RSS has not been loaded, load RSS. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {Object} │ │ │ │ │ + * zoomChanged - {Object} │ │ │ │ │ + * minor - {Object} │ │ │ │ │ + */ │ │ │ │ │ + moveTo: function(bounds, zoomChanged, minor) { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + if (this.visibility && !this.loaded) { │ │ │ │ │ + this.loadRSS(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseData │ │ │ │ │ + * Parse the data returned from the Events call. │ │ │ │ │ * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ - * featureNS - {String} Feature namespace (optional). │ │ │ │ │ - * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ - * if featureNS is provided). Default is 'feature'. │ │ │ │ │ - * geometryName - {String} Name of geometry attribute. Default is 'the_geom'. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * ajaxRequest - {<OpenLayers.Request.XMLHttpRequest>} │ │ │ │ │ */ │ │ │ │ │ + parseData: function(ajaxRequest) { │ │ │ │ │ + var doc = ajaxRequest.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = OpenLayers.Format.XML.prototype.read(ajaxRequest.responseText); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.WFS.v1_0_0" │ │ │ │ │ + if (this.useFeedTitle) { │ │ │ │ │ + var name = null; │ │ │ │ │ + try { │ │ │ │ │ + name = doc.getElementsByTagNameNS('*', 'title')[0].firstChild.nodeValue; │ │ │ │ │ + } catch (e) { │ │ │ │ │ + try { │ │ │ │ │ + name = doc.getElementsByTagName('title')[0].firstChild.nodeValue; │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + } │ │ │ │ │ + if (name) { │ │ │ │ │ + this.setName(name); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + 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 format = new OpenLayers.Format.GeoRSS(options); │ │ │ │ │ + var features = format.read(doc); │ │ │ │ │ + │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + var data = {}; │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + │ │ │ │ │ + // we don't support features with no geometry in the GeoRSS │ │ │ │ │ + // layer at this time. │ │ │ │ │ + if (!feature.geometry) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var title = feature.attributes.title ? │ │ │ │ │ + feature.attributes.title : "Untitled"; │ │ │ │ │ + │ │ │ │ │ + var description = feature.attributes.description ? │ │ │ │ │ + feature.attributes.description : "No description."; │ │ │ │ │ + │ │ │ │ │ + var link = feature.attributes.link ? feature.attributes.link : ""; │ │ │ │ │ + │ │ │ │ │ + var location = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + data.icon = this.icon == null ? │ │ │ │ │ + OpenLayers.Marker.defaultIcon() : │ │ │ │ │ + this.icon.clone(); │ │ │ │ │ + │ │ │ │ │ + data.popupSize = this.popupSize ? │ │ │ │ │ + this.popupSize.clone() : │ │ │ │ │ + new OpenLayers.Size(250, 120); │ │ │ │ │ + │ │ │ │ │ + if (title || description) { │ │ │ │ │ + // we have supplemental data, store them. │ │ │ │ │ + data.title = title; │ │ │ │ │ + data.description = description; │ │ │ │ │ + │ │ │ │ │ + var contentHTML = '<div class="olLayerGeoRSSClose">[x]</div>'; │ │ │ │ │ + contentHTML += '<div class="olLayerGeoRSSTitle">'; │ │ │ │ │ + if (link) { │ │ │ │ │ + contentHTML += '<a class="link" href="' + link + '" target="_blank">'; │ │ │ │ │ + } │ │ │ │ │ + contentHTML += title; │ │ │ │ │ + if (link) { │ │ │ │ │ + contentHTML += '</a>'; │ │ │ │ │ + } │ │ │ │ │ + contentHTML += '</div>'; │ │ │ │ │ + contentHTML += '<div style="" class="olLayerGeoRSSDescription">'; │ │ │ │ │ + contentHTML += description; │ │ │ │ │ + contentHTML += '</div>'; │ │ │ │ │ + data['popupContentHTML'] = contentHTML; │ │ │ │ │ + } │ │ │ │ │ + var feature = new OpenLayers.Feature(this, location, data); │ │ │ │ │ + this.features.push(feature); │ │ │ │ │ + var marker = feature.createMarker(); │ │ │ │ │ + marker.events.register('click', feature, this.markerClick); │ │ │ │ │ + this.addMarker(marker); │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: markerClick │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + markerClick: function(evt) { │ │ │ │ │ + var sameMarkerClicked = (this == this.layer.selectedFeature); │ │ │ │ │ + this.layer.selectedFeature = (!sameMarkerClicked) ? this : null; │ │ │ │ │ + for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ + this.layer.map.removePopup(this.layer.map.popups[i]); │ │ │ │ │ + } │ │ │ │ │ + if (!sameMarkerClicked) { │ │ │ │ │ + var popup = this.createPopup(); │ │ │ │ │ + OpenLayers.Event.observe(popup.div, "click", │ │ │ │ │ + OpenLayers.Function.bind(function() { │ │ │ │ │ + for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ + this.layer.map.removePopup(this.layer.map.popups[i]); │ │ │ │ │ + } │ │ │ │ │ + }, this) │ │ │ │ │ + ); │ │ │ │ │ + this.layer.map.addPopup(popup); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clearFeatures │ │ │ │ │ + * Destroy all features in this layer. │ │ │ │ │ + */ │ │ │ │ │ + clearFeatures: function() { │ │ │ │ │ + if (this.features != null) { │ │ │ │ │ + while (this.features.length > 0) { │ │ │ │ │ + var feature = this.features[0]; │ │ │ │ │ + OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ + feature.destroy(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.GeoRSS" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/WFS/v1_1_0.js │ │ │ │ │ + OpenLayers/Layer/MapServer.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Protocol/WFS/v1.js │ │ │ │ │ - * @requires OpenLayers/Format/WFST/v1_1_0.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Protocol.WFS.v1_1_0 │ │ │ │ │ - * A WFS v1.1.0 protocol for vector layers. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Protocol.WFS.v1_1_0> constructor. │ │ │ │ │ + * Class: OpenLayers.Layer.MapServer │ │ │ │ │ + * Instances of OpenLayers.Layer.MapServer are used to display │ │ │ │ │ + * data from a MapServer CGI instance. │ │ │ │ │ * │ │ │ │ │ - * Differences from the v1.0.0 protocol: │ │ │ │ │ - * - uses Filter Encoding 1.1.0 instead of 1.0.0 │ │ │ │ │ - * - uses GML 3 instead of 2 if no format is provided │ │ │ │ │ - * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol.WFS.v1> │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.WFS.v1_1_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ +OpenLayers.Layer.MapServer = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} WFS version number. │ │ │ │ │ + * Constant: DEFAULT_PARAMS │ │ │ │ │ + * {Object} Hashtable of default parameter key/value pairs │ │ │ │ │ */ │ │ │ │ │ - version: "1.1.0", │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + mode: "map", │ │ │ │ │ + map_imagetype: "png" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.WFS.v1_1_0 │ │ │ │ │ - * A class for giving layers WFS v1.1.0 protocol. │ │ │ │ │ + * Constructor: OpenLayers.Layer.MapServer │ │ │ │ │ + * Create a new MapServer layer object │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * name - {String} A name for the layer │ │ │ │ │ + * url - {String} Base url for the MapServer CGI │ │ │ │ │ + * (e.g. http://www2.dmsolutions.ca/cgi-bin/mapserv) │ │ │ │ │ + * params - {Object} An object with key/value pairs representing the │ │ │ │ │ + * GetMap query string parameters and parameter values. │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + this.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, this.DEFAULT_PARAMS │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + // unless explicitly set in options, if the layer is transparent, │ │ │ │ │ + // it will be an overlay │ │ │ │ │ + if (options == null || options.isBaseLayer == null) { │ │ │ │ │ + this.isBaseLayer = ((this.params.transparent != "true") && │ │ │ │ │ + (this.params.transparent != true)); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ - * featureNS - {String} Feature namespace (optional). │ │ │ │ │ - * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ - * if featureNS is provided). Default is 'feature'. │ │ │ │ │ - * geometryName - {String} Name of geometry attribute. Default is 'the_geom'. │ │ │ │ │ - * outputFormat - {String} Optional output format to use for WFS GetFeature │ │ │ │ │ - * requests. This can be any format advertized by the WFS's │ │ │ │ │ - * GetCapabilities response. If set, an appropriate readFormat also │ │ │ │ │ - * has to be provided, unless outputFormat is GML3, GML2 or JSON. │ │ │ │ │ - * readFormat - {<OpenLayers.Format>} An appropriate format parser if │ │ │ │ │ - * outputFormat is none of GML3, GML2 or JSON. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.MapServer>} An exact clone of this layer │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.WFS.v1.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (this.outputFormat && !this.readFormat) { │ │ │ │ │ - if (this.outputFormat.toLowerCase() == "gml2") { │ │ │ │ │ - this.readFormat = new OpenLayers.Format.GML.v2({ │ │ │ │ │ - featureType: this.featureType, │ │ │ │ │ - featureNS: this.featureNS, │ │ │ │ │ - geometryName: this.geometryName │ │ │ │ │ - }); │ │ │ │ │ - } else if (this.outputFormat.toLowerCase() == "json") { │ │ │ │ │ - this.readFormat = new OpenLayers.Format.GeoJSON(); │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.MapServer(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.params, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * Return a query string for this layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox │ │ │ │ │ + * for the request │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the layer's url and parameters and also │ │ │ │ │ + * the passed-in bounds and appropriate tile size specified │ │ │ │ │ + * as parameters. │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + // Make a list, so that getFullRequestString uses literal "," │ │ │ │ │ + var extent = [bounds.left, bounds.bottom, bounds.right, bounds.top]; │ │ │ │ │ + │ │ │ │ │ + var imageSize = this.getImageSize(); │ │ │ │ │ + │ │ │ │ │ + // make lists, so that literal ','s are used │ │ │ │ │ + var url = this.getFullRequestString({ │ │ │ │ │ + mapext: extent, │ │ │ │ │ + imgext: extent, │ │ │ │ │ + map_size: [imageSize.w, imageSize.h], │ │ │ │ │ + imgx: imageSize.w / 2, │ │ │ │ │ + imgy: imageSize.h / 2, │ │ │ │ │ + imgxy: [imageSize.w, imageSize.h] │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + return url; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getFullRequestString │ │ │ │ │ + * combine the layer's url with its params and these newParams. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newParams - {Object} New parameters that should be added to the │ │ │ │ │ + * request string. │ │ │ │ │ + * altUrl - {String} (optional) Replace the URL in the full request │ │ │ │ │ + * string with the provided URL. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the layer's url and parameters embedded in it. │ │ │ │ │ + */ │ │ │ │ │ + getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ + // use layer's url unless altUrl passed in │ │ │ │ │ + var url = (altUrl == null) ? this.url : altUrl; │ │ │ │ │ + │ │ │ │ │ + // create a new params hashtable with all the layer params and the │ │ │ │ │ + // new params together. then convert to string │ │ │ │ │ + var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ + allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ + var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + │ │ │ │ │ + // if url is not a string, it should be an array of strings, │ │ │ │ │ + // in which case we will deterministically select one of them in │ │ │ │ │ + // order to evenly distribute requests to different urls. │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(paramsString, url); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // ignore parameters that are already in the url search string │ │ │ │ │ + var urlParams = OpenLayers.Util.upperCaseObject( │ │ │ │ │ + OpenLayers.Util.getParameters(url)); │ │ │ │ │ + for (var key in allParams) { │ │ │ │ │ + if (key.toUpperCase() in urlParams) { │ │ │ │ │ + delete allParams[key]; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + │ │ │ │ │ + // requestString always starts with url │ │ │ │ │ + var requestString = url; │ │ │ │ │ + │ │ │ │ │ + // MapServer needs '+' seperating things like bounds/height/width. │ │ │ │ │ + // Since typically this is URL encoded, we use a slight hack: we │ │ │ │ │ + // depend on the list-like functionality of getParameterString to │ │ │ │ │ + // leave ',' only in the case of list items (since otherwise it is │ │ │ │ │ + // encoded) then do a regular expression replace on the , characters │ │ │ │ │ + // to '+' │ │ │ │ │ + // │ │ │ │ │ + paramsString = paramsString.replace(/,/g, "+"); │ │ │ │ │ + │ │ │ │ │ + if (paramsString != "") { │ │ │ │ │ + var lastServerChar = url.charAt(url.length - 1); │ │ │ │ │ + if ((lastServerChar == "&") || (lastServerChar == "?")) { │ │ │ │ │ + requestString += paramsString; │ │ │ │ │ + } else { │ │ │ │ │ + if (url.indexOf('?') == -1) { │ │ │ │ │ + //serverPath has no ? -- add one │ │ │ │ │ + requestString += '?' + paramsString; │ │ │ │ │ + } else { │ │ │ │ │ + //serverPath contains ?, so must already have paramsString at the end │ │ │ │ │ + requestString += '&' + paramsString; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return requestString; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.WFS.v1_1_0" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.MapServer" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/CSW/v2_0_2.js │ │ │ │ │ + OpenLayers/Layer/UTFGrid.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Protocol/CSW.js │ │ │ │ │ - * @requires OpenLayers/Format/CSWGetRecords/v2_0_2.js │ │ │ │ │ + * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ + * @requires OpenLayers/Tile/UTFGrid.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Protocol.CSW.v2_0_2 │ │ │ │ │ - * CS-W (Catalogue services for the Web) version 2.0.2 protocol. │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.UTFGrid │ │ │ │ │ + * This Layer reads from UTFGrid tiled data sources. Since UTFGrids are │ │ │ │ │ + * essentially JSON-based ASCII art with attached attributes, they are not │ │ │ │ │ + * visibly rendered. In order to use them in the map, you must add a │ │ │ │ │ + * <OpenLayers.Control.UTFGrid> control as well. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * │ │ │ │ │ + * (start code) │ │ │ │ │ + * var world_utfgrid = new OpenLayers.Layer.UTFGrid({ │ │ │ │ │ + * url: "/tiles/world_utfgrid/${z}/${x}/${y}.json", │ │ │ │ │ + * utfgridResolution: 4, │ │ │ │ │ + * displayInLayerSwitcher: false │ │ │ │ │ + * ); │ │ │ │ │ + * map.addLayer(world_utfgrid); │ │ │ │ │ + * │ │ │ │ │ + * var control = new OpenLayers.Control.UTFGrid({ │ │ │ │ │ + * layers: [world_utfgrid], │ │ │ │ │ + * handlerMode: 'move', │ │ │ │ │ + * callback: function(dataLookup) { │ │ │ │ │ + * // do something with returned data │ │ │ │ │ + * } │ │ │ │ │ + * }) │ │ │ │ │ + * (end code) │ │ │ │ │ * │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol> │ │ │ │ │ + * - <OpenLayers.Layer.XYZ> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.CSW.v2_0_2 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ +OpenLayers.Layer.UTFGrid = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: formatOptions │ │ │ │ │ - * {Object} Optional options for the format. If a format is not provided, │ │ │ │ │ - * this property can be used to extend the default format options. │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * Default is false, as UTFGrids are designed to be a transparent overlay layer. │ │ │ │ │ */ │ │ │ │ │ - formatOptions: null, │ │ │ │ │ + isBaseLayer: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.CSW.v2_0_2 │ │ │ │ │ - * A class for CSW version 2.0.2 protocol management. │ │ │ │ │ + * APIProperty: projection │ │ │ │ │ + * {<OpenLayers.Projection>} │ │ │ │ │ + * Source projection for the UTFGrids. Default is "EPSG:900913". │ │ │ │ │ + */ │ │ │ │ │ + projection: new OpenLayers.Projection("EPSG:900913"), │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: useJSONP │ │ │ │ │ + * {Boolean} │ │ │ │ │ + * Should we use a JSONP script approach instead of a standard AJAX call? │ │ │ │ │ + * │ │ │ │ │ + * Set to true for using utfgrids from another server. │ │ │ │ │ + * Avoids same-domain policy restrictions. │ │ │ │ │ + * Note that this only works if the server accepts │ │ │ │ │ + * the callback GET parameter and dynamically │ │ │ │ │ + * wraps the returned json in a function call. │ │ │ │ │ + * │ │ │ │ │ + * Default is false │ │ │ │ │ + */ │ │ │ │ │ + useJSONP: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String} │ │ │ │ │ + * URL tempate for UTFGrid tiles. Include x, y, and z parameters. │ │ │ │ │ + * E.g. "/tiles/${z}/${x}/${y}.json" │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: utfgridResolution │ │ │ │ │ + * {Number} │ │ │ │ │ + * Ratio of the pixel width to the width of a UTFGrid data point. If an │ │ │ │ │ + * entry in the grid represents a 4x4 block of pixels, the │ │ │ │ │ + * utfgridResolution would be 4. Default is 2 (specified in │ │ │ │ │ + * <OpenLayers.Tile.UTFGrid>). │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: tileClass │ │ │ │ │ + * {<OpenLayers.Tile>} The tile class to use for this layer. │ │ │ │ │ + * Defaults is <OpenLayers.Tile.UTFGrid>. │ │ │ │ │ + */ │ │ │ │ │ + tileClass: OpenLayers.Tile.UTFGrid, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.UTFGrid │ │ │ │ │ + * Create a new UTFGrid layer. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * config - {Object} Configuration properties for the layer. │ │ │ │ │ + * │ │ │ │ │ + * Required configuration properties: │ │ │ │ │ + * url - {String} The url template for UTFGrid tiles. See the <url> property. │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!options.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.CSWGetRecords.v2_0_2(OpenLayers.Util.extend({}, this.formatOptions)); │ │ │ │ │ - } │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply( │ │ │ │ │ + this, [options.name, options.url, {}, options] │ │ │ │ │ + ); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + utfgridResolution: this.utfgridResolution │ │ │ │ │ + }, this.tileOptions); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up the protocol. │ │ │ │ │ + * Method: createBackBuffer │ │ │ │ │ + * The UTFGrid cannot create a back buffer, so this method is overriden. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.options && !this.options.format) { │ │ │ │ │ - this.format.destroy(); │ │ │ │ │ - } │ │ │ │ │ - this.format = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ - }, │ │ │ │ │ + createBackBuffer: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read │ │ │ │ │ - * Construct a request for reading new records from the Catalogue. │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} Only used by a subclass of this layer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.UTFGrid>} An exact clone of this OpenLayers.Layer.UTFGrid │ │ │ │ │ */ │ │ │ │ │ - read: function(options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - var data = this.format.write(options.params || options); │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.UTFGrid(this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: data │ │ │ │ │ - }); │ │ │ │ │ + // get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ │ │ │ │ │ - return response; │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleRead │ │ │ │ │ - * Deal with response from the read request. │ │ │ │ │ + * APIProperty: getFeatureInfo │ │ │ │ │ + * Get details about a feature associated with a map location. The object │ │ │ │ │ + * returned will have id and data properties. If the given location │ │ │ │ │ + * doesn't correspond to a feature, null will be returned. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ - * to the user callback. │ │ │ │ │ - * This response is given a code property, and optionally a data property. │ │ │ │ │ - * The latter represents the CSW records as returned by the call to │ │ │ │ │ - * the CSW format read method. │ │ │ │ │ - * options - {Object} The user options passed to the read call. │ │ │ │ │ + * location - {<OpenLayers.LonLat>} map location │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Object representing the feature id and UTFGrid data │ │ │ │ │ + * corresponding to the given map location. Returns null if the given │ │ │ │ │ + * location doesn't hit a feature. │ │ │ │ │ */ │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - // success │ │ │ │ │ - response.data = this.parseData(request); │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ - } else { │ │ │ │ │ - // failure │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - } │ │ │ │ │ - options.callback.call(options.scope, response); │ │ │ │ │ + getFeatureInfo: function(location) { │ │ │ │ │ + var info = null; │ │ │ │ │ + var tileInfo = this.getTileData(location); │ │ │ │ │ + if (tileInfo && tileInfo.tile) { │ │ │ │ │ + info = tileInfo.tile.getFeatureInfo(tileInfo.i, tileInfo.j); │ │ │ │ │ } │ │ │ │ │ + return info; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseData │ │ │ │ │ - * Read HTTP response body and return records │ │ │ │ │ + * APIMethod: getFeatureId │ │ │ │ │ + * Get the identifier for the feature associated with a map location. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * request - {XMLHttpRequest} The request object │ │ │ │ │ + * location - {<OpenLayers.LonLat>} map location │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} The CSW records as returned by the call to the format read method. │ │ │ │ │ + * {String} The feature identifier corresponding to the given map location. │ │ │ │ │ + * Returns null if the location doesn't hit a feature. │ │ │ │ │ */ │ │ │ │ │ - parseData: function(request) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null; │ │ │ │ │ + getFeatureId: function(location) { │ │ │ │ │ + var id = null; │ │ │ │ │ + var info = this.getTileData(location); │ │ │ │ │ + if (info.tile) { │ │ │ │ │ + id = info.tile.getFeatureId(info.i, info.j); │ │ │ │ │ } │ │ │ │ │ - return this.format.read(doc); │ │ │ │ │ + return id; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.CSW.v2_0_2" │ │ │ │ │ - │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.UTFGrid" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/SOS/v1_0_0.js │ │ │ │ │ + OpenLayers/Layer/Google/v3.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Protocol/SOS.js │ │ │ │ │ - * @requires OpenLayers/Format/SOSGetFeatureOfInterest.js │ │ │ │ │ + * @requires OpenLayers/Layer/Google.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Protocol.SOS.v1_0_0 │ │ │ │ │ - * An SOS v1.0.0 Protocol for vector layers. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Protocol.SOS.v1_0_0> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol> │ │ │ │ │ + * Constant: OpenLayers.Layer.Google.v3 │ │ │ │ │ + * │ │ │ │ │ + * Mixin providing functionality specific to the Google Maps API v3. │ │ │ │ │ + * │ │ │ │ │ + * To use this layer, you must include the GMaps v3 API in your html. │ │ │ │ │ + * │ │ │ │ │ + * Note that this layer configures the google.maps.map object with the │ │ │ │ │ + * "disableDefaultUI" option set to true. Using UI controls that the Google │ │ │ │ │ + * Maps API provides is not supported by the OpenLayers API. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.SOS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ +OpenLayers.Layer.Google.v3 = { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: fois │ │ │ │ │ - * {Array(String)} Array of features of interest (foi) │ │ │ │ │ + * Constant: DEFAULTS │ │ │ │ │ + * {Object} It is not recommended to change the properties set here. Note │ │ │ │ │ + * that Google.v3 layers only work when sphericalMercator is set to true. │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * { │ │ │ │ │ + * sphericalMercator: true, │ │ │ │ │ + * projection: "EPSG:900913" │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ */ │ │ │ │ │ - fois: null, │ │ │ │ │ + DEFAULTS: { │ │ │ │ │ + sphericalMercator: true, │ │ │ │ │ + projection: "EPSG:900913" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: formatOptions │ │ │ │ │ - * {Object} Optional options for the format. If a format is not provided, │ │ │ │ │ - * this property can be used to extend the default format options. │ │ │ │ │ + * APIProperty: animationEnabled │ │ │ │ │ + * {Boolean} If set to true, the transition between zoom levels will be │ │ │ │ │ + * animated (if supported by the GMaps API for the device used). Set to │ │ │ │ │ + * false to match the zooming experience of other layer types. Default │ │ │ │ │ + * is true. Note that the GMaps API does not give us control over zoom │ │ │ │ │ + * animation, so if set to false, when zooming, this will make the │ │ │ │ │ + * layer temporarily invisible, wait until GMaps reports the map being │ │ │ │ │ + * idle, and make it visible again. The result will be a blank layer │ │ │ │ │ + * for a few moments while zooming. │ │ │ │ │ */ │ │ │ │ │ - formatOptions: null, │ │ │ │ │ + animationEnabled: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: loadMapObject │ │ │ │ │ + * Load the GMap and register appropriate event listeners. │ │ │ │ │ + */ │ │ │ │ │ + loadMapObject: function() { │ │ │ │ │ + if (!this.type) { │ │ │ │ │ + this.type = google.maps.MapTypeId.ROADMAP; │ │ │ │ │ + } │ │ │ │ │ + var mapObject; │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + // there are already Google layers added to this map │ │ │ │ │ + mapObject = cache.mapObject; │ │ │ │ │ + // increment the layer count │ │ │ │ │ + ++cache.count; │ │ │ │ │ + } else { │ │ │ │ │ + // this is the first Google layer for this map │ │ │ │ │ + // create GMap │ │ │ │ │ + var center = this.map.getCenter(); │ │ │ │ │ + var container = document.createElement('div'); │ │ │ │ │ + container.className = "olForeignContainer"; │ │ │ │ │ + container.style.width = '100%'; │ │ │ │ │ + container.style.height = '100%'; │ │ │ │ │ + mapObject = new google.maps.Map(container, { │ │ │ │ │ + center: center ? │ │ │ │ │ + new google.maps.LatLng(center.lat, center.lon) : new google.maps.LatLng(0, 0), │ │ │ │ │ + zoom: this.map.getZoom() || 0, │ │ │ │ │ + mapTypeId: this.type, │ │ │ │ │ + disableDefaultUI: true, │ │ │ │ │ + keyboardShortcuts: false, │ │ │ │ │ + draggable: false, │ │ │ │ │ + disableDoubleClickZoom: true, │ │ │ │ │ + scrollwheel: false, │ │ │ │ │ + streetViewControl: false │ │ │ │ │ + }); │ │ │ │ │ + var googleControl = document.createElement('div'); │ │ │ │ │ + googleControl.style.width = '100%'; │ │ │ │ │ + googleControl.style.height = '100%'; │ │ │ │ │ + mapObject.controls[google.maps.ControlPosition.TOP_LEFT].push(googleControl); │ │ │ │ │ + │ │ │ │ │ + // cache elements for use by any other google layers added to │ │ │ │ │ + // this same map │ │ │ │ │ + cache = { │ │ │ │ │ + googleControl: googleControl, │ │ │ │ │ + mapObject: mapObject, │ │ │ │ │ + count: 1 │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Layer.Google.cache[this.map.id] = cache; │ │ │ │ │ + } │ │ │ │ │ + this.mapObject = mapObject; │ │ │ │ │ + this.setGMapVisibility(this.visibility); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.SOS │ │ │ │ │ - * A class for giving layers an SOS protocol. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: onMapResize │ │ │ │ │ + */ │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + if (this.visibility) { │ │ │ │ │ + google.maps.event.trigger(this.mapObject, "resize"); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setGMapVisibility │ │ │ │ │ + * Display the GMap container and associated elements. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * url - {String} URL to send requests to (required). │ │ │ │ │ - * fois - {Array} The features of interest (required). │ │ │ │ │ + * visible - {Boolean} Display the GMap elements. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!options.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.SOSGetFeatureOfInterest( │ │ │ │ │ - this.formatOptions); │ │ │ │ │ + setGMapVisibility: function(visible) { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + var map = this.map; │ │ │ │ │ + if (cache) { │ │ │ │ │ + var type = this.type; │ │ │ │ │ + var layers = map.layers; │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = layers[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Google && │ │ │ │ │ + layer.visibility === true && layer.inRange === true) { │ │ │ │ │ + type = layer.type; │ │ │ │ │ + visible = true; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var container = this.mapObject.getDiv(); │ │ │ │ │ + if (visible === true) { │ │ │ │ │ + if (container.parentNode !== map.div) { │ │ │ │ │ + if (!cache.rendered) { │ │ │ │ │ + var me = this; │ │ │ │ │ + google.maps.event.addListenerOnce(this.mapObject, 'tilesloaded', function() { │ │ │ │ │ + cache.rendered = true; │ │ │ │ │ + me.setGMapVisibility(me.getVisibility()); │ │ │ │ │ + me.moveTo(me.map.getCenter()); │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + map.div.appendChild(container); │ │ │ │ │ + cache.googleControl.appendChild(map.viewPortDiv); │ │ │ │ │ + google.maps.event.trigger(this.mapObject, 'resize'); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.mapObject.setMapTypeId(type); │ │ │ │ │ + } else if (cache.googleControl.hasChildNodes()) { │ │ │ │ │ + map.div.appendChild(map.viewPortDiv); │ │ │ │ │ + map.div.removeChild(container); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up the protocol. │ │ │ │ │ + * Method: getMapContainer │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} the GMap container's div │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.options && !this.options.format) { │ │ │ │ │ - this.format.destroy(); │ │ │ │ │ + getMapContainer: function() { │ │ │ │ │ + return this.mapObject.getDiv(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ + // │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectBoundsFromOLBounds │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * olBounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} A MapObject Bounds, translated from olBounds │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ + */ │ │ │ │ │ + getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ + var moBounds = null; │ │ │ │ │ + if (olBounds != null) { │ │ │ │ │ + var sw = this.sphericalMercator ? │ │ │ │ │ + this.inverseMercator(olBounds.bottom, olBounds.left) : │ │ │ │ │ + new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ + var ne = this.sphericalMercator ? │ │ │ │ │ + this.inverseMercator(olBounds.top, olBounds.right) : │ │ │ │ │ + new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ + moBounds = new google.maps.LatLngBounds( │ │ │ │ │ + new google.maps.LatLng(sw.lat, sw.lon), │ │ │ │ │ + new google.maps.LatLng(ne.lat, ne.lon) │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - this.format = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ + return moBounds; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Interface Controls * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + // LonLat - Pixel Translation │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Construct a request for reading new sensor positions. This is done by │ │ │ │ │ - * issuing one GetFeatureOfInterest request. │ │ │ │ │ + * APIMethod: getMapObjectLonLatFromMapObjectPixel │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moPixel - {Object} MapObject Pixel format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject LonLat translated from MapObject Pixel │ │ │ │ │ */ │ │ │ │ │ - read: function(options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - var format = this.format; │ │ │ │ │ - var data = OpenLayers.Format.XML.prototype.write.apply(format, │ │ │ │ │ - [format.writeNode("sos:GetFeatureOfInterest", { │ │ │ │ │ - fois: this.fois │ │ │ │ │ - })] │ │ │ │ │ + getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + var size = this.map.getSize(); │ │ │ │ │ + var lon = this.getLongitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ + var lat = this.getLatitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ + var res = this.map.getResolution(); │ │ │ │ │ + │ │ │ │ │ + var delta_x = moPixel.x - (size.w / 2); │ │ │ │ │ + var delta_y = moPixel.y - (size.h / 2); │ │ │ │ │ + │ │ │ │ │ + var lonlat = new OpenLayers.LonLat( │ │ │ │ │ + lon + delta_x * res, │ │ │ │ │ + lat - delta_y * res │ │ │ │ │ ); │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ - data: data │ │ │ │ │ - }); │ │ │ │ │ - return response; │ │ │ │ │ + │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + lonlat = lonlat.wrapDateLine(this.maxExtent); │ │ │ │ │ + } │ │ │ │ │ + return this.getMapObjectLonLatFromLonLat(lonlat.lon, lonlat.lat); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleRead │ │ │ │ │ - * Deal with response from the read request. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getMapObjectPixelFromMapObjectLonLat │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ - * to the user callback. │ │ │ │ │ - * options - {Object} The user options passed to the read call. │ │ │ │ │ + * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject Pixel transtlated from MapObject LonLat │ │ │ │ │ */ │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - // success │ │ │ │ │ - response.features = this.parseFeatures(request); │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ - } else { │ │ │ │ │ - // failure │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - } │ │ │ │ │ - options.callback.call(options.scope, response); │ │ │ │ │ + getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + var res = this.map.getResolution(); │ │ │ │ │ + var extent = this.map.getExtent(); │ │ │ │ │ + return this.getMapObjectPixelFromXY((1 / res * (lon - extent.left)), │ │ │ │ │ + (1 / res * (extent.top - lat))); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setMapObjectCenter │ │ │ │ │ + * Set the mapObject to the specified center and zoom │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * center - {Object} MapObject LonLat format │ │ │ │ │ + * zoom - {int} MapObject zoom format │ │ │ │ │ + */ │ │ │ │ │ + setMapObjectCenter: function(center, zoom) { │ │ │ │ │ + if (this.animationEnabled === false && zoom != this.mapObject.zoom) { │ │ │ │ │ + var mapContainer = this.getMapContainer(); │ │ │ │ │ + google.maps.event.addListenerOnce( │ │ │ │ │ + this.mapObject, │ │ │ │ │ + "idle", │ │ │ │ │ + function() { │ │ │ │ │ + mapContainer.style.visibility = ""; │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + mapContainer.style.visibility = "hidden"; │ │ │ │ │ } │ │ │ │ │ + this.mapObject.setOptions({ │ │ │ │ │ + center: center, │ │ │ │ │ + zoom: zoom │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ + // Bounds │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectZoomFromMapObjectBounds │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moBounds - {Object} MapObject Bounds format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject Zoom for specified MapObject Bounds │ │ │ │ │ + */ │ │ │ │ │ + getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + return this.mapObject.getBoundsZoomLevel(moBounds); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Primitives * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + // LonLat │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: parseFeatures │ │ │ │ │ - * Read HTTP response body and return features │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getMapObjectLonLatFromLonLat │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * request - {XMLHttpRequest} The request object │ │ │ │ │ - * │ │ │ │ │ + * lon - {Float} │ │ │ │ │ + * lat - {Float} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array({<OpenLayers.Feature.Vector>})} Array of features │ │ │ │ │ + * {Object} MapObject LonLat built from lon and lat params │ │ │ │ │ */ │ │ │ │ │ - parseFeatures: function(request) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null; │ │ │ │ │ + getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ + var gLatLng; │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ + gLatLng = new google.maps.LatLng(lonlat.lat, lonlat.lon); │ │ │ │ │ + } else { │ │ │ │ │ + gLatLng = new google.maps.LatLng(lat, lon); │ │ │ │ │ } │ │ │ │ │ - return this.format.read(doc); │ │ │ │ │ + return gLatLng; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.SOS.v1_0_0" │ │ │ │ │ -}); │ │ │ │ │ + // Pixel │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectPixelFromXY │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * x - {Integer} │ │ │ │ │ + * y - {Integer} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject Pixel from x and y parameters │ │ │ │ │ + */ │ │ │ │ │ + getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ + return new google.maps.Point(x, y); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Popup/Anchored.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Popup.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Popup.Anchored │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Popup> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Popup.Anchored = │ │ │ │ │ + OpenLayers.Class(OpenLayers.Popup, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: relativePosition │ │ │ │ │ + * {String} Relative position of the popup ("br", "tr", "tl" or "bl"). │ │ │ │ │ + */ │ │ │ │ │ + relativePosition: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: keepInMap │ │ │ │ │ + * {Boolean} If panMapIfOutOfView is false, and this property is true, │ │ │ │ │ + * contrain the popup such that it always fits in the available map │ │ │ │ │ + * space. By default, this is set. If you are creating popups that are │ │ │ │ │ + * near map edges and not allowing pannning, and especially if you have │ │ │ │ │ + * a popup which has a fixedRelativePosition, setting this to false may │ │ │ │ │ + * be a smart thing to do. │ │ │ │ │ + * │ │ │ │ │ + * For anchored popups, default is true, since subclasses will │ │ │ │ │ + * usually want this functionality. │ │ │ │ │ + */ │ │ │ │ │ + keepInMap: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: anchor │ │ │ │ │ + * {Object} Object to which we'll anchor the popup. Must expose a │ │ │ │ │ + * 'size' (<OpenLayers.Size>) and 'offset' (<OpenLayers.Pixel>). │ │ │ │ │ + */ │ │ │ │ │ + anchor: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Popup.Anchored │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ + * contentSize - {<OpenLayers.Size>} │ │ │ │ │ + * contentHTML - {String} │ │ │ │ │ + * anchor - {Object} Object which must expose a 'size' <OpenLayers.Size> │ │ │ │ │ + * and 'offset' <OpenLayers.Pixel> (generally an <OpenLayers.Icon>). │ │ │ │ │ + * closeBox - {Boolean} │ │ │ │ │ + * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, │ │ │ │ │ + closeBoxCallback) { │ │ │ │ │ + var newArguments = [ │ │ │ │ │ + id, lonlat, contentSize, contentHTML, closeBox, closeBoxCallback │ │ │ │ │ + ]; │ │ │ │ │ + OpenLayers.Popup.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + │ │ │ │ │ + this.anchor = (anchor != null) ? anchor : │ │ │ │ │ + { │ │ │ │ │ + size: new OpenLayers.Size(0, 0), │ │ │ │ │ + offset: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.anchor = null; │ │ │ │ │ + this.relativePosition = null; │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Popup.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: show │ │ │ │ │ + * Overridden from Popup since user might hide popup and then show() it │ │ │ │ │ + * in a new location (meaning we might want to update the relative │ │ │ │ │ + * position on the show) │ │ │ │ │ + */ │ │ │ │ │ + show: function() { │ │ │ │ │ + this.updatePosition(); │ │ │ │ │ + OpenLayers.Popup.prototype.show.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * Since the popup is moving to a new px, it might need also to be moved │ │ │ │ │ + * relative to where the marker is. We first calculate the new │ │ │ │ │ + * relativePosition, and then we calculate the new px where we will │ │ │ │ │ + * put the popup, based on the new relative position. │ │ │ │ │ + * │ │ │ │ │ + * If the relativePosition has changed, we must also call │ │ │ │ │ + * updateRelativePosition() to make any visual changes to the popup │ │ │ │ │ + * which are associated with putting it in a new relativePosition. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ + */ │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + var oldRelativePosition = this.relativePosition; │ │ │ │ │ + this.relativePosition = this.calculateRelativePosition(px); │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Popup.prototype.moveTo.call(this, this.calculateNewPx(px)); │ │ │ │ │ + │ │ │ │ │ + //if this move has caused the popup to change its relative position, │ │ │ │ │ + // we need to make the appropriate cosmetic changes. │ │ │ │ │ + if (this.relativePosition != oldRelativePosition) { │ │ │ │ │ + this.updateRelativePosition(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setSize │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * contentSize - {<OpenLayers.Size>} the new size for the popup's │ │ │ │ │ + * contents div (in pixels). │ │ │ │ │ + */ │ │ │ │ │ + setSize: function(contentSize) { │ │ │ │ │ + OpenLayers.Popup.prototype.setSize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + if ((this.lonlat) && (this.map)) { │ │ │ │ │ + var px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ + this.moveTo(px); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateRelativePosition │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The relative position ("br" "tr" "tl" "bl") at which the popup │ │ │ │ │ + * should be placed. │ │ │ │ │ + */ │ │ │ │ │ + calculateRelativePosition: function(px) { │ │ │ │ │ + var lonlat = this.map.getLonLatFromLayerPx(px); │ │ │ │ │ + │ │ │ │ │ + var extent = this.map.getExtent(); │ │ │ │ │ + var quadrant = extent.determineQuadrant(lonlat); │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Bounds.oppositeQuadrant(quadrant); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateRelativePosition │ │ │ │ │ + * The popup has been moved to a new relative location, so we may want to │ │ │ │ │ + * make some cosmetic adjustments to it. │ │ │ │ │ + * │ │ │ │ │ + * Note that in the classic Anchored popup, there is nothing to do │ │ │ │ │ + * here, since the popup looks exactly the same in all four positions. │ │ │ │ │ + * Subclasses such as Framed, however, will want to do something │ │ │ │ │ + * special here. │ │ │ │ │ + */ │ │ │ │ │ + updateRelativePosition: function() { │ │ │ │ │ + //to be overridden by subclasses │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateNewPx │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Pixel>} The the new px position of the popup on the screen │ │ │ │ │ + * relative to the passed-in px. │ │ │ │ │ + */ │ │ │ │ │ + calculateNewPx: function(px) { │ │ │ │ │ + var newPx = px.offset(this.anchor.offset); │ │ │ │ │ + │ │ │ │ │ + //use contentSize if size is not already set │ │ │ │ │ + var size = this.size || this.contentSize; │ │ │ │ │ + │ │ │ │ │ + var top = (this.relativePosition.charAt(0) == 't'); │ │ │ │ │ + newPx.y += (top) ? -size.h : this.anchor.size.h; │ │ │ │ │ + │ │ │ │ │ + var left = (this.relativePosition.charAt(1) == 'l'); │ │ │ │ │ + newPx.x += (left) ? -size.w : this.anchor.size.w; │ │ │ │ │ + │ │ │ │ │ + return newPx; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup.Anchored" │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Popup/Framed.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Popup/Anchored.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Popup.Framed │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Popup.Anchored> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Popup.Framed = │ │ │ │ │ + OpenLayers.Class(OpenLayers.Popup.Anchored, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: imageSrc │ │ │ │ │ + * {String} location of the image to be used as the popup frame │ │ │ │ │ + */ │ │ │ │ │ + imageSrc: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: imageSize │ │ │ │ │ + * {<OpenLayers.Size>} Size (measured in pixels) of the image located │ │ │ │ │ + * by the 'imageSrc' property. │ │ │ │ │ + */ │ │ │ │ │ + imageSize: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isAlphaImage │ │ │ │ │ + * {Boolean} The image has some alpha and thus needs to use the alpha │ │ │ │ │ + * image hack. Note that setting this to true will have no noticeable │ │ │ │ │ + * effect in FF or IE7 browsers, but will all but crush the ie6 │ │ │ │ │ + * browser. │ │ │ │ │ + * Default is false. │ │ │ │ │ + */ │ │ │ │ │ + isAlphaImage: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: positionBlocks │ │ │ │ │ + * {Object} Hash of different position blocks (Object/Hashs). Each block │ │ │ │ │ + * will be keyed by a two-character 'relativePosition' │ │ │ │ │ + * code string (ie "tl", "tr", "bl", "br"). Block properties are │ │ │ │ │ + * 'offset', 'padding' (self-explanatory), and finally the 'blocks' │ │ │ │ │ + * parameter, which is an array of the block objects. │ │ │ │ │ + * │ │ │ │ │ + * Each block object must have 'size', 'anchor', and 'position' │ │ │ │ │ + * properties. │ │ │ │ │ + * │ │ │ │ │ + * Note that positionBlocks should never be modified at runtime. │ │ │ │ │ + */ │ │ │ │ │ + positionBlocks: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: blocks │ │ │ │ │ + * {Array[Object]} Array of objects, each of which is one "block" of the │ │ │ │ │ + * popup. Each block has a 'div' and an 'image' property, both of │ │ │ │ │ + * which are DOMElements, and the latter of which is appended to the │ │ │ │ │ + * former. These are reused as the popup goes changing positions for │ │ │ │ │ + * great economy and elegance. │ │ │ │ │ + */ │ │ │ │ │ + blocks: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: fixedRelativePosition │ │ │ │ │ + * {Boolean} We want the framed popup to work dynamically placed relative │ │ │ │ │ + * to its anchor but also in just one fixed position. A well designed │ │ │ │ │ + * framed popup will have the pixels and logic to display itself in │ │ │ │ │ + * any of the four relative positions, but (understandably), this will │ │ │ │ │ + * not be the case for all of them. By setting this property to 'true', │ │ │ │ │ + * framed popup will not recalculate for the best placement each time │ │ │ │ │ + * it's open, but will always open the same way. │ │ │ │ │ + * Note that if this is set to true, it is generally advisable to also │ │ │ │ │ + * set the 'panIntoView' property to true so that the popup can be │ │ │ │ │ + * scrolled into view (since it will often be offscreen on open) │ │ │ │ │ + * Default is false. │ │ │ │ │ + */ │ │ │ │ │ + fixedRelativePosition: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Popup.Framed │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ + * contentSize - {<OpenLayers.Size>} │ │ │ │ │ + * contentHTML - {String} │ │ │ │ │ + * anchor - {Object} Object to which we'll anchor the popup. Must expose │ │ │ │ │ + * a 'size' (<OpenLayers.Size>) and 'offset' (<OpenLayers.Pixel>) │ │ │ │ │ + * (Note that this is generally an <OpenLayers.Icon>). │ │ │ │ │ + * closeBox - {Boolean} │ │ │ │ │ + * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, │ │ │ │ │ + closeBoxCallback) { │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.initialize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + if (this.fixedRelativePosition) { │ │ │ │ │ + //based on our decided relativePostion, set the current padding │ │ │ │ │ + // this keeps us from getting into trouble │ │ │ │ │ + this.updateRelativePosition(); │ │ │ │ │ + │ │ │ │ │ + //make calculateRelativePosition always return the specified │ │ │ │ │ + // fixed position. │ │ │ │ │ + this.calculateRelativePosition = function(px) { │ │ │ │ │ + return this.relativePosition; │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.contentDiv.style.position = "absolute"; │ │ │ │ │ + this.contentDiv.style.zIndex = 1; │ │ │ │ │ + │ │ │ │ │ + if (closeBox) { │ │ │ │ │ + this.closeDiv.style.zIndex = 1; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.groupDiv.style.position = "absolute"; │ │ │ │ │ + this.groupDiv.style.top = "0px"; │ │ │ │ │ + this.groupDiv.style.left = "0px"; │ │ │ │ │ + this.groupDiv.style.height = "100%"; │ │ │ │ │ + this.groupDiv.style.width = "100%"; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.imageSrc = null; │ │ │ │ │ + this.imageSize = null; │ │ │ │ │ + this.isAlphaImage = null; │ │ │ │ │ + │ │ │ │ │ + this.fixedRelativePosition = false; │ │ │ │ │ + this.positionBlocks = null; │ │ │ │ │ + │ │ │ │ │ + //remove our blocks │ │ │ │ │ + for (var i = 0; i < this.blocks.length; i++) { │ │ │ │ │ + var block = this.blocks[i]; │ │ │ │ │ + │ │ │ │ │ + if (block.image) { │ │ │ │ │ + block.div.removeChild(block.image); │ │ │ │ │ + } │ │ │ │ │ + block.image = null; │ │ │ │ │ + │ │ │ │ │ + if (block.div) { │ │ │ │ │ + this.groupDiv.removeChild(block.div); │ │ │ │ │ + } │ │ │ │ │ + block.div = null; │ │ │ │ │ + } │ │ │ │ │ + this.blocks = null; │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setBackgroundColor │ │ │ │ │ + */ │ │ │ │ │ + setBackgroundColor: function(color) { │ │ │ │ │ + //does nothing since the framed popup's entire scheme is based on a │ │ │ │ │ + // an image -- changing the background color makes no sense. │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setBorder │ │ │ │ │ + */ │ │ │ │ │ + setBorder: function() { │ │ │ │ │ + //does nothing since the framed popup's entire scheme is based on a │ │ │ │ │ + // an image -- changing the popup's border makes no sense. │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setOpacity │ │ │ │ │ + * Sets the opacity of the popup. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * opacity - {float} A value between 0.0 (transparent) and 1.0 (solid). │ │ │ │ │ + */ │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + //does nothing since we suppose that we'll never apply an opacity │ │ │ │ │ + // to a framed popup │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setSize │ │ │ │ │ + * Overridden here, because we need to update the blocks whenever the size │ │ │ │ │ + * of the popup has changed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * contentSize - {<OpenLayers.Size>} the new size for the popup's │ │ │ │ │ + * contents div (in pixels). │ │ │ │ │ + */ │ │ │ │ │ + setSize: function(contentSize) { │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.setSize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + this.updateBlocks(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateRelativePosition │ │ │ │ │ + * When the relative position changes, we need to set the new padding │ │ │ │ │ + * BBOX on the popup, reposition the close div, and update the blocks. │ │ │ │ │ + */ │ │ │ │ │ + updateRelativePosition: function() { │ │ │ │ │ + │ │ │ │ │ + //update the padding │ │ │ │ │ + this.padding = this.positionBlocks[this.relativePosition].padding; │ │ │ │ │ + │ │ │ │ │ + //update the position of our close box to new padding │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + // use the content div's css padding to determine if we should │ │ │ │ │ + // padd the close div │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ + │ │ │ │ │ + this.closeDiv.style.right = contentDivPadding.right + │ │ │ │ │ + this.padding.right + "px"; │ │ │ │ │ + this.closeDiv.style.top = contentDivPadding.top + │ │ │ │ │ + this.padding.top + "px"; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.updateBlocks(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateNewPx │ │ │ │ │ + * Besides the standard offset as determined by the Anchored class, our │ │ │ │ │ + * Framed popups have a special 'offset' property for each of their │ │ │ │ │ + * positions, which is used to offset the popup relative to its anchor. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Pixel>} The the new px position of the popup on the screen │ │ │ │ │ + * relative to the passed-in px. │ │ │ │ │ + */ │ │ │ │ │ + calculateNewPx: function(px) { │ │ │ │ │ + var newPx = OpenLayers.Popup.Anchored.prototype.calculateNewPx.apply( │ │ │ │ │ + this, arguments │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + newPx = newPx.offset(this.positionBlocks[this.relativePosition].offset); │ │ │ │ │ + │ │ │ │ │ + return newPx; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createBlocks │ │ │ │ │ + */ │ │ │ │ │ + createBlocks: function() { │ │ │ │ │ + this.blocks = []; │ │ │ │ │ + │ │ │ │ │ + //since all positions contain the same number of blocks, we can │ │ │ │ │ + // just pick the first position and use its blocks array to create │ │ │ │ │ + // our blocks array │ │ │ │ │ + var firstPosition = null; │ │ │ │ │ + for (var key in this.positionBlocks) { │ │ │ │ │ + firstPosition = key; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var position = this.positionBlocks[firstPosition]; │ │ │ │ │ + for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ + │ │ │ │ │ + var block = {}; │ │ │ │ │ + this.blocks.push(block); │ │ │ │ │ + │ │ │ │ │ + var divId = this.id + '_FrameDecorationDiv_' + i; │ │ │ │ │ + block.div = OpenLayers.Util.createDiv(divId, │ │ │ │ │ + null, null, null, "absolute", null, "hidden", null │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + var imgId = this.id + '_FrameDecorationImg_' + i; │ │ │ │ │ + var imageCreator = │ │ │ │ │ + (this.isAlphaImage) ? OpenLayers.Util.createAlphaImageDiv : │ │ │ │ │ + OpenLayers.Util.createImage; │ │ │ │ │ + │ │ │ │ │ + block.image = imageCreator(imgId, │ │ │ │ │ + null, this.imageSize, this.imageSrc, │ │ │ │ │ + "absolute", null, null, null │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + block.div.appendChild(block.image); │ │ │ │ │ + this.groupDiv.appendChild(block.div); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateBlocks │ │ │ │ │ + * Internal method, called on initialize and when the popup's relative │ │ │ │ │ + * position has changed. This function takes care of re-positioning │ │ │ │ │ + * the popup's blocks in their appropropriate places. │ │ │ │ │ + */ │ │ │ │ │ + updateBlocks: function() { │ │ │ │ │ + if (!this.blocks) { │ │ │ │ │ + this.createBlocks(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.size && this.relativePosition) { │ │ │ │ │ + var position = this.positionBlocks[this.relativePosition]; │ │ │ │ │ + for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ + │ │ │ │ │ + var positionBlock = position.blocks[i]; │ │ │ │ │ + var block = this.blocks[i]; │ │ │ │ │ + │ │ │ │ │ + // adjust sizes │ │ │ │ │ + var l = positionBlock.anchor.left; │ │ │ │ │ + var b = positionBlock.anchor.bottom; │ │ │ │ │ + var r = positionBlock.anchor.right; │ │ │ │ │ + var t = positionBlock.anchor.top; │ │ │ │ │ + │ │ │ │ │ + //note that we use the isNaN() test here because if the │ │ │ │ │ + // size object is initialized with a "auto" parameter, the │ │ │ │ │ + // size constructor calls parseFloat() on the string, │ │ │ │ │ + // which will turn it into NaN │ │ │ │ │ + // │ │ │ │ │ + var w = (isNaN(positionBlock.size.w)) ? this.size.w - (r + l) : │ │ │ │ │ + positionBlock.size.w; │ │ │ │ │ + │ │ │ │ │ + var h = (isNaN(positionBlock.size.h)) ? this.size.h - (b + t) : │ │ │ │ │ + positionBlock.size.h; │ │ │ │ │ + │ │ │ │ │ + block.div.style.width = (w < 0 ? 0 : w) + 'px'; │ │ │ │ │ + block.div.style.height = (h < 0 ? 0 : h) + 'px'; │ │ │ │ │ + │ │ │ │ │ + block.div.style.left = (l != null) ? l + 'px' : ''; │ │ │ │ │ + block.div.style.bottom = (b != null) ? b + 'px' : ''; │ │ │ │ │ + block.div.style.right = (r != null) ? r + 'px' : ''; │ │ │ │ │ + block.div.style.top = (t != null) ? t + 'px' : ''; │ │ │ │ │ + │ │ │ │ │ + block.image.style.left = positionBlock.position.x + 'px'; │ │ │ │ │ + block.image.style.top = positionBlock.position.y + 'px'; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.contentDiv.style.left = this.padding.left + "px"; │ │ │ │ │ + this.contentDiv.style.top = this.padding.top + "px"; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup.Framed" │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Popup/FramedCloud.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Popup/Framed.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Bounds.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Pixel.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Size.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Popup.FramedCloud │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Popup.Framed> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Popup.FramedCloud = │ │ │ │ │ + OpenLayers.Class(OpenLayers.Popup.Framed, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: contentDisplayClass │ │ │ │ │ + * {String} The CSS class of the popup content div. │ │ │ │ │ + */ │ │ │ │ │ + contentDisplayClass: "olFramedCloudPopupContent", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: autoSize │ │ │ │ │ + * {Boolean} Framed Cloud is autosizing by default. │ │ │ │ │ + */ │ │ │ │ │ + autoSize: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: panMapIfOutOfView │ │ │ │ │ + * {Boolean} Framed Cloud does pan into view by default. │ │ │ │ │ + */ │ │ │ │ │ + panMapIfOutOfView: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: imageSize │ │ │ │ │ + * {<OpenLayers.Size>} │ │ │ │ │ + */ │ │ │ │ │ + imageSize: new OpenLayers.Size(1276, 736), │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isAlphaImage │ │ │ │ │ + * {Boolean} The FramedCloud does not use an alpha image (in honor of the │ │ │ │ │ + * good ie6 folk out there) │ │ │ │ │ + */ │ │ │ │ │ + isAlphaImage: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: fixedRelativePosition │ │ │ │ │ + * {Boolean} The Framed Cloud popup works in just one fixed position. │ │ │ │ │ + */ │ │ │ │ │ + fixedRelativePosition: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: positionBlocks │ │ │ │ │ + * {Object} Hash of differen position blocks, keyed by relativePosition │ │ │ │ │ + * two-character code string (ie "tl", "tr", "bl", "br") │ │ │ │ │ + */ │ │ │ │ │ + positionBlocks: { │ │ │ │ │ + "tl": { │ │ │ │ │ + 'offset': new OpenLayers.Pixel(44, 0), │ │ │ │ │ + 'padding': new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ + 'blocks': [{ // top-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { //top-right │ │ │ │ │ + size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { //bottom-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ + }, { //bottom-right │ │ │ │ │ + size: new OpenLayers.Size(22, 18), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -632) │ │ │ │ │ + }, { // stem │ │ │ │ │ + size: new OpenLayers.Size(81, 35), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -688) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + "tr": { │ │ │ │ │ + 'offset': new OpenLayers.Pixel(-45, 0), │ │ │ │ │ + 'padding': new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ + 'blocks': [{ // top-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { //top-right │ │ │ │ │ + size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { //bottom-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ + }, { //bottom-right │ │ │ │ │ + size: new OpenLayers.Size(22, 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -631) │ │ │ │ │ + }, { // stem │ │ │ │ │ + size: new OpenLayers.Size(81, 35), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, null, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-215, -687) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + "bl": { │ │ │ │ │ + 'offset': new OpenLayers.Pixel(45, 0), │ │ │ │ │ + 'padding': new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ + 'blocks': [{ // top-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { //top-right │ │ │ │ │ + size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { //bottom-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ + }, { //bottom-right │ │ │ │ │ + size: new OpenLayers.Size(22, 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ + }, { // stem │ │ │ │ │ + size: new OpenLayers.Size(81, 33), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, null, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-101, -674) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + "br": { │ │ │ │ │ + 'offset': new OpenLayers.Pixel(-44, 0), │ │ │ │ │ + 'padding': new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ + 'blocks': [{ // top-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { //top-right │ │ │ │ │ + size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { //bottom-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ + }, { //bottom-right │ │ │ │ │ + size: new OpenLayers.Size(22, 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ + }, { // stem │ │ │ │ │ + size: new OpenLayers.Size(81, 33), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, null, null, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-311, -674) │ │ │ │ │ + }] │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: minSize │ │ │ │ │ + * {<OpenLayers.Size>} │ │ │ │ │ + */ │ │ │ │ │ + minSize: new OpenLayers.Size(105, 10), │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxSize │ │ │ │ │ + * {<OpenLayers.Size>} │ │ │ │ │ + */ │ │ │ │ │ + maxSize: new OpenLayers.Size(1200, 660), │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Popup.FramedCloud │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ + * contentSize - {<OpenLayers.Size>} │ │ │ │ │ + * contentHTML - {String} │ │ │ │ │ + * anchor - {Object} Object to which we'll anchor the popup. Must expose │ │ │ │ │ + * a 'size' (<OpenLayers.Size>) and 'offset' (<OpenLayers.Pixel>) │ │ │ │ │ + * (Note that this is generally an <OpenLayers.Icon>). │ │ │ │ │ + * closeBox - {Boolean} │ │ │ │ │ + * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, │ │ │ │ │ + closeBoxCallback) { │ │ │ │ │ + │ │ │ │ │ + this.imageSrc = OpenLayers.Util.getImageLocation('cloud-popup-relative.png'); │ │ │ │ │ + OpenLayers.Popup.Framed.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.contentDiv.className = this.contentDisplayClass; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup.FramedCloud" │ │ │ │ │ + }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ Rico/license.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @license Apache 2 │ │ │ │ │ * │ │ │ ├── ./usr/share/javascript/openlayers/OpenLayers.light.js │ │ │ │ ├── js-beautify {} │ │ │ │ │ @@ -263,449 +263,14 @@ │ │ │ │ │ source.hasOwnProperty && source.hasOwnProperty("toString")) { │ │ │ │ │ destination.toString = source.toString; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ return destination; │ │ │ │ │ }; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Util/vendorPrefix.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/SingleFile.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Util = OpenLayers.Util || {}; │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Util.vendorPrefix │ │ │ │ │ - * A collection of utility functions to detect vendor prefixed features │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Util.vendorPrefix = (function() { │ │ │ │ │ - "use strict"; │ │ │ │ │ - │ │ │ │ │ - var VENDOR_PREFIXES = ["", "O", "ms", "Moz", "Webkit"], │ │ │ │ │ - divStyle = document.createElement("div").style, │ │ │ │ │ - cssCache = {}, │ │ │ │ │ - jsCache = {}; │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Function: domToCss │ │ │ │ │ - * Converts a upper camel case DOM style property name to a CSS property │ │ │ │ │ - * i.e. transformOrigin -> transform-origin │ │ │ │ │ - * or WebkitTransformOrigin -> -webkit-transform-origin │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * prefixedDom - {String} The property to convert │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The CSS property │ │ │ │ │ - */ │ │ │ │ │ - function domToCss(prefixedDom) { │ │ │ │ │ - if (!prefixedDom) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - return prefixedDom. │ │ │ │ │ - replace(/([A-Z])/g, function(c) { │ │ │ │ │ - return "-" + c.toLowerCase(); │ │ │ │ │ - }). │ │ │ │ │ - replace(/^ms-/, "-ms-"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: css │ │ │ │ │ - * Detect which property is used for a CSS property │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * property - {String} The standard (unprefixed) CSS property name │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The standard CSS property, prefixed property or null if not │ │ │ │ │ - * supported │ │ │ │ │ - */ │ │ │ │ │ - function css(property) { │ │ │ │ │ - if (cssCache[property] === undefined) { │ │ │ │ │ - var domProperty = property. │ │ │ │ │ - replace(/(-[\s\S])/g, function(c) { │ │ │ │ │ - return c.charAt(1).toUpperCase(); │ │ │ │ │ - }); │ │ │ │ │ - var prefixedDom = style(domProperty); │ │ │ │ │ - cssCache[property] = domToCss(prefixedDom); │ │ │ │ │ - } │ │ │ │ │ - return cssCache[property]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: js │ │ │ │ │ - * Detect which property is used for a JS property/method │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} The object to test on │ │ │ │ │ - * property - {String} The standard (unprefixed) JS property name │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The standard JS property, prefixed property or null if not │ │ │ │ │ - * supported │ │ │ │ │ - */ │ │ │ │ │ - function js(obj, property) { │ │ │ │ │ - if (jsCache[property] === undefined) { │ │ │ │ │ - var tmpProp, │ │ │ │ │ - i = 0, │ │ │ │ │ - l = VENDOR_PREFIXES.length, │ │ │ │ │ - prefix, │ │ │ │ │ - isStyleObj = (typeof obj.cssText !== "undefined"); │ │ │ │ │ - │ │ │ │ │ - jsCache[property] = null; │ │ │ │ │ - for (; i < l; i++) { │ │ │ │ │ - prefix = VENDOR_PREFIXES[i]; │ │ │ │ │ - if (prefix) { │ │ │ │ │ - if (!isStyleObj) { │ │ │ │ │ - // js prefix should be lower-case, while style │ │ │ │ │ - // properties have upper case on first character │ │ │ │ │ - prefix = prefix.toLowerCase(); │ │ │ │ │ - } │ │ │ │ │ - tmpProp = prefix + property.charAt(0).toUpperCase() + property.slice(1); │ │ │ │ │ - } else { │ │ │ │ │ - tmpProp = property; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (obj[tmpProp] !== undefined) { │ │ │ │ │ - jsCache[property] = tmpProp; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return jsCache[property]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: style │ │ │ │ │ - * Detect which property is used for a DOM style property │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * property - {String} The standard (unprefixed) style property name │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The standard style property, prefixed property or null if not │ │ │ │ │ - * supported │ │ │ │ │ - */ │ │ │ │ │ - function style(property) { │ │ │ │ │ - return js(divStyle, property); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return { │ │ │ │ │ - css: css, │ │ │ │ │ - js: js, │ │ │ │ │ - style: style, │ │ │ │ │ - │ │ │ │ │ - // used for testing │ │ │ │ │ - cssCache: cssCache, │ │ │ │ │ - jsCache: jsCache │ │ │ │ │ - }; │ │ │ │ │ -}()); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Animation.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/SingleFile.js │ │ │ │ │ - * @requires OpenLayers/Util/vendorPrefix.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Animation │ │ │ │ │ - * A collection of utility functions for executing methods that repaint a │ │ │ │ │ - * portion of the browser window. These methods take advantage of the │ │ │ │ │ - * browser's scheduled repaints where requestAnimationFrame is available. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Animation = (function(window) { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: isNative │ │ │ │ │ - * {Boolean} true if a native requestAnimationFrame function is available │ │ │ │ │ - */ │ │ │ │ │ - var requestAnimationFrame = OpenLayers.Util.vendorPrefix.js(window, "requestAnimationFrame"); │ │ │ │ │ - var isNative = !!(requestAnimationFrame); │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Function: requestFrame │ │ │ │ │ - * Schedule a function to be called at the next available animation frame. │ │ │ │ │ - * Uses the native method where available. Where requestAnimationFrame is │ │ │ │ │ - * not available, setTimeout will be called with a 16ms delay. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * callback - {Function} The function to be called at the next animation frame. │ │ │ │ │ - * element - {DOMElement} Optional element that visually bounds the animation. │ │ │ │ │ - */ │ │ │ │ │ - var requestFrame = (function() { │ │ │ │ │ - var request = window[requestAnimationFrame] || │ │ │ │ │ - function(callback, element) { │ │ │ │ │ - window.setTimeout(callback, 16); │ │ │ │ │ - }; │ │ │ │ │ - // bind to window to avoid illegal invocation of native function │ │ │ │ │ - return function(callback, element) { │ │ │ │ │ - request.apply(window, [callback, element]); │ │ │ │ │ - }; │ │ │ │ │ - })(); │ │ │ │ │ - │ │ │ │ │ - // private variables for animation loops │ │ │ │ │ - var counter = 0; │ │ │ │ │ - var loops = {}; │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Function: start │ │ │ │ │ - * Executes a method with <requestFrame> in series for some │ │ │ │ │ - * duration. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * callback - {Function} The function to be called at the next animation frame. │ │ │ │ │ - * duration - {Number} Optional duration for the loop. If not provided, the │ │ │ │ │ - * animation loop will execute indefinitely. │ │ │ │ │ - * element - {DOMElement} Optional element that visually bounds the animation. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number} Identifier for the animation loop. Used to stop animations with │ │ │ │ │ - * <stop>. │ │ │ │ │ - */ │ │ │ │ │ - function start(callback, duration, element) { │ │ │ │ │ - duration = duration > 0 ? duration : Number.POSITIVE_INFINITY; │ │ │ │ │ - var id = ++counter; │ │ │ │ │ - var start = +new Date; │ │ │ │ │ - loops[id] = function() { │ │ │ │ │ - if (loops[id] && +new Date - start <= duration) { │ │ │ │ │ - callback(); │ │ │ │ │ - if (loops[id]) { │ │ │ │ │ - requestFrame(loops[id], element); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - delete loops[id]; │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - requestFrame(loops[id], element); │ │ │ │ │ - return id; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Function: stop │ │ │ │ │ - * Terminates an animation loop started with <start>. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {Number} Identifier returned from <start>. │ │ │ │ │ - */ │ │ │ │ │ - function stop(id) { │ │ │ │ │ - delete loops[id]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return { │ │ │ │ │ - isNative: isNative, │ │ │ │ │ - requestFrame: requestFrame, │ │ │ │ │ - start: start, │ │ │ │ │ - stop: stop │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ -})(window); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Kinetic.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Animation.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Kinetic = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: threshold │ │ │ │ │ - * In most cases changing the threshold isn't needed. │ │ │ │ │ - * In px/ms, default to 0. │ │ │ │ │ - */ │ │ │ │ │ - threshold: 0, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: deceleration │ │ │ │ │ - * {Float} the deseleration in px/ms², default to 0.0035. │ │ │ │ │ - */ │ │ │ │ │ - deceleration: 0.0035, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: nbPoints │ │ │ │ │ - * {Integer} the number of points we use to calculate the kinetic │ │ │ │ │ - * initial values. │ │ │ │ │ - */ │ │ │ │ │ - nbPoints: 100, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: delay │ │ │ │ │ - * {Float} time to consider to calculate the kinetic initial values. │ │ │ │ │ - * In ms, default to 200. │ │ │ │ │ - */ │ │ │ │ │ - delay: 200, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: points │ │ │ │ │ - * List of points use to calculate the kinetic initial values. │ │ │ │ │ - */ │ │ │ │ │ - points: undefined, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: timerId │ │ │ │ │ - * ID of the timer. │ │ │ │ │ - */ │ │ │ │ │ - timerId: undefined, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Kinetic │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: begin │ │ │ │ │ - * Begins the dragging. │ │ │ │ │ - */ │ │ │ │ │ - begin: function() { │ │ │ │ │ - OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ - this.timerId = undefined; │ │ │ │ │ - this.points = []; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: update │ │ │ │ │ - * Updates during the dragging. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} The new position. │ │ │ │ │ - */ │ │ │ │ │ - update: function(xy) { │ │ │ │ │ - this.points.unshift({ │ │ │ │ │ - xy: xy, │ │ │ │ │ - tick: new Date().getTime() │ │ │ │ │ - }); │ │ │ │ │ - if (this.points.length > this.nbPoints) { │ │ │ │ │ - this.points.pop(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: end │ │ │ │ │ - * Ends the dragging, start the kinetic. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} The last position. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object with two properties: "speed", and "theta". The │ │ │ │ │ - * "speed" and "theta" values are to be passed to the move │ │ │ │ │ - * function when starting the animation. │ │ │ │ │ - */ │ │ │ │ │ - end: function(xy) { │ │ │ │ │ - var last, now = new Date().getTime(); │ │ │ │ │ - for (var i = 0, l = this.points.length, point; i < l; i++) { │ │ │ │ │ - point = this.points[i]; │ │ │ │ │ - if (now - point.tick > this.delay) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - last = point; │ │ │ │ │ - } │ │ │ │ │ - if (!last) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var time = new Date().getTime() - last.tick; │ │ │ │ │ - var dist = Math.sqrt(Math.pow(xy.x - last.xy.x, 2) + │ │ │ │ │ - Math.pow(xy.y - last.xy.y, 2)); │ │ │ │ │ - var speed = dist / time; │ │ │ │ │ - if (speed == 0 || speed < this.threshold) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var theta = Math.asin((xy.y - last.xy.y) / dist); │ │ │ │ │ - if (last.xy.x <= xy.x) { │ │ │ │ │ - theta = Math.PI - theta; │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - speed: speed, │ │ │ │ │ - theta: theta │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * Launch the kinetic move pan. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * info - {Object} An object with two properties, "speed", and "theta". │ │ │ │ │ - * These values are those returned from the "end" call. │ │ │ │ │ - * callback - {Function} Function called on every step of the animation, │ │ │ │ │ - * receives x, y (values to pan), end (is the last point). │ │ │ │ │ - */ │ │ │ │ │ - move: function(info, callback) { │ │ │ │ │ - var v0 = info.speed; │ │ │ │ │ - var fx = Math.cos(info.theta); │ │ │ │ │ - var fy = -Math.sin(info.theta); │ │ │ │ │ - │ │ │ │ │ - var initialTime = new Date().getTime(); │ │ │ │ │ - │ │ │ │ │ - var lastX = 0; │ │ │ │ │ - var lastY = 0; │ │ │ │ │ - │ │ │ │ │ - var timerCallback = function() { │ │ │ │ │ - if (this.timerId == null) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var t = new Date().getTime() - initialTime; │ │ │ │ │ - │ │ │ │ │ - var p = (-this.deceleration * Math.pow(t, 2)) / 2.0 + v0 * t; │ │ │ │ │ - var x = p * fx; │ │ │ │ │ - var y = p * fy; │ │ │ │ │ - │ │ │ │ │ - var args = {}; │ │ │ │ │ - args.end = false; │ │ │ │ │ - var v = -this.deceleration * t + v0; │ │ │ │ │ - │ │ │ │ │ - if (v <= 0) { │ │ │ │ │ - OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - args.end = true; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - args.x = x - lastX; │ │ │ │ │ - args.y = y - lastY; │ │ │ │ │ - lastX = x; │ │ │ │ │ - lastY = y; │ │ │ │ │ - callback(args.x, args.y, args.end); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - this.timerId = OpenLayers.Animation.start( │ │ │ │ │ - OpenLayers.Function.bind(timerCallback, this) │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Kinetic" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ OpenLayers/BaseTypes.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -4861,14 +4426,155 @@ │ │ │ │ │ } else { │ │ │ │ │ str += coordinate < 0 ? OpenLayers.i18n("S") : OpenLayers.i18n("N"); │ │ │ │ │ } │ │ │ │ │ return str; │ │ │ │ │ }; │ │ │ │ │ │ │ │ │ │ /* ====================================================================== │ │ │ │ │ + OpenLayers/Util/vendorPrefix.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/SingleFile.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Util = OpenLayers.Util || {}; │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Util.vendorPrefix │ │ │ │ │ + * A collection of utility functions to detect vendor prefixed features │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Util.vendorPrefix = (function() { │ │ │ │ │ + "use strict"; │ │ │ │ │ + │ │ │ │ │ + var VENDOR_PREFIXES = ["", "O", "ms", "Moz", "Webkit"], │ │ │ │ │ + divStyle = document.createElement("div").style, │ │ │ │ │ + cssCache = {}, │ │ │ │ │ + jsCache = {}; │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Function: domToCss │ │ │ │ │ + * Converts a upper camel case DOM style property name to a CSS property │ │ │ │ │ + * i.e. transformOrigin -> transform-origin │ │ │ │ │ + * or WebkitTransformOrigin -> -webkit-transform-origin │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * prefixedDom - {String} The property to convert │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The CSS property │ │ │ │ │ + */ │ │ │ │ │ + function domToCss(prefixedDom) { │ │ │ │ │ + if (!prefixedDom) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + return prefixedDom. │ │ │ │ │ + replace(/([A-Z])/g, function(c) { │ │ │ │ │ + return "-" + c.toLowerCase(); │ │ │ │ │ + }). │ │ │ │ │ + replace(/^ms-/, "-ms-"); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: css │ │ │ │ │ + * Detect which property is used for a CSS property │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * property - {String} The standard (unprefixed) CSS property name │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The standard CSS property, prefixed property or null if not │ │ │ │ │ + * supported │ │ │ │ │ + */ │ │ │ │ │ + function css(property) { │ │ │ │ │ + if (cssCache[property] === undefined) { │ │ │ │ │ + var domProperty = property. │ │ │ │ │ + replace(/(-[\s\S])/g, function(c) { │ │ │ │ │ + return c.charAt(1).toUpperCase(); │ │ │ │ │ + }); │ │ │ │ │ + var prefixedDom = style(domProperty); │ │ │ │ │ + cssCache[property] = domToCss(prefixedDom); │ │ │ │ │ + } │ │ │ │ │ + return cssCache[property]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: js │ │ │ │ │ + * Detect which property is used for a JS property/method │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} The object to test on │ │ │ │ │ + * property - {String} The standard (unprefixed) JS property name │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The standard JS property, prefixed property or null if not │ │ │ │ │ + * supported │ │ │ │ │ + */ │ │ │ │ │ + function js(obj, property) { │ │ │ │ │ + if (jsCache[property] === undefined) { │ │ │ │ │ + var tmpProp, │ │ │ │ │ + i = 0, │ │ │ │ │ + l = VENDOR_PREFIXES.length, │ │ │ │ │ + prefix, │ │ │ │ │ + isStyleObj = (typeof obj.cssText !== "undefined"); │ │ │ │ │ + │ │ │ │ │ + jsCache[property] = null; │ │ │ │ │ + for (; i < l; i++) { │ │ │ │ │ + prefix = VENDOR_PREFIXES[i]; │ │ │ │ │ + if (prefix) { │ │ │ │ │ + if (!isStyleObj) { │ │ │ │ │ + // js prefix should be lower-case, while style │ │ │ │ │ + // properties have upper case on first character │ │ │ │ │ + prefix = prefix.toLowerCase(); │ │ │ │ │ + } │ │ │ │ │ + tmpProp = prefix + property.charAt(0).toUpperCase() + property.slice(1); │ │ │ │ │ + } else { │ │ │ │ │ + tmpProp = property; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (obj[tmpProp] !== undefined) { │ │ │ │ │ + jsCache[property] = tmpProp; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return jsCache[property]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: style │ │ │ │ │ + * Detect which property is used for a DOM style property │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * property - {String} The standard (unprefixed) style property name │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The standard style property, prefixed property or null if not │ │ │ │ │ + * supported │ │ │ │ │ + */ │ │ │ │ │ + function style(property) { │ │ │ │ │ + return js(divStyle, property); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return { │ │ │ │ │ + css: css, │ │ │ │ │ + js: js, │ │ │ │ │ + style: style, │ │ │ │ │ + │ │ │ │ │ + // used for testing │ │ │ │ │ + cssCache: cssCache, │ │ │ │ │ + jsCache: jsCache │ │ │ │ │ + }; │ │ │ │ │ +}()); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ OpenLayers/Events.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -6037,14 +5743,120 @@ │ │ │ │ │ │ │ │ │ │ OpenLayers.Event.observe(element, 'MSPointerUp', cb); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Events" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ + OpenLayers/Animation.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/SingleFile.js │ │ │ │ │ + * @requires OpenLayers/Util/vendorPrefix.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Animation │ │ │ │ │ + * A collection of utility functions for executing methods that repaint a │ │ │ │ │ + * portion of the browser window. These methods take advantage of the │ │ │ │ │ + * browser's scheduled repaints where requestAnimationFrame is available. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Animation = (function(window) { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: isNative │ │ │ │ │ + * {Boolean} true if a native requestAnimationFrame function is available │ │ │ │ │ + */ │ │ │ │ │ + var requestAnimationFrame = OpenLayers.Util.vendorPrefix.js(window, "requestAnimationFrame"); │ │ │ │ │ + var isNative = !!(requestAnimationFrame); │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Function: requestFrame │ │ │ │ │ + * Schedule a function to be called at the next available animation frame. │ │ │ │ │ + * Uses the native method where available. Where requestAnimationFrame is │ │ │ │ │ + * not available, setTimeout will be called with a 16ms delay. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * callback - {Function} The function to be called at the next animation frame. │ │ │ │ │ + * element - {DOMElement} Optional element that visually bounds the animation. │ │ │ │ │ + */ │ │ │ │ │ + var requestFrame = (function() { │ │ │ │ │ + var request = window[requestAnimationFrame] || │ │ │ │ │ + function(callback, element) { │ │ │ │ │ + window.setTimeout(callback, 16); │ │ │ │ │ + }; │ │ │ │ │ + // bind to window to avoid illegal invocation of native function │ │ │ │ │ + return function(callback, element) { │ │ │ │ │ + request.apply(window, [callback, element]); │ │ │ │ │ + }; │ │ │ │ │ + })(); │ │ │ │ │ + │ │ │ │ │ + // private variables for animation loops │ │ │ │ │ + var counter = 0; │ │ │ │ │ + var loops = {}; │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Function: start │ │ │ │ │ + * Executes a method with <requestFrame> in series for some │ │ │ │ │ + * duration. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * callback - {Function} The function to be called at the next animation frame. │ │ │ │ │ + * duration - {Number} Optional duration for the loop. If not provided, the │ │ │ │ │ + * animation loop will execute indefinitely. │ │ │ │ │ + * element - {DOMElement} Optional element that visually bounds the animation. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number} Identifier for the animation loop. Used to stop animations with │ │ │ │ │ + * <stop>. │ │ │ │ │ + */ │ │ │ │ │ + function start(callback, duration, element) { │ │ │ │ │ + duration = duration > 0 ? duration : Number.POSITIVE_INFINITY; │ │ │ │ │ + var id = ++counter; │ │ │ │ │ + var start = +new Date; │ │ │ │ │ + loops[id] = function() { │ │ │ │ │ + if (loops[id] && +new Date - start <= duration) { │ │ │ │ │ + callback(); │ │ │ │ │ + if (loops[id]) { │ │ │ │ │ + requestFrame(loops[id], element); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + delete loops[id]; │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + requestFrame(loops[id], element); │ │ │ │ │ + return id; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Function: stop │ │ │ │ │ + * Terminates an animation loop started with <start>. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {Number} Identifier returned from <start>. │ │ │ │ │ + */ │ │ │ │ │ + function stop(id) { │ │ │ │ │ + delete loops[id]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return { │ │ │ │ │ + isNative: isNative, │ │ │ │ │ + requestFrame: requestFrame, │ │ │ │ │ + start: start, │ │ │ │ │ + stop: stop │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ +})(window); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ OpenLayers/Tween.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -10841,14 +10653,367 @@ │ │ │ │ │ * {Array} prefixes of the sld symbolizers. These are the │ │ │ │ │ * same as the main geometry types │ │ │ │ │ */ │ │ │ │ │ OpenLayers.Style.SYMBOLIZER_PREFIXES = ['Point', 'Line', 'Polygon', 'Text', │ │ │ │ │ 'Raster' │ │ │ │ │ ]; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ + OpenLayers/StyleMap.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Style.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.StyleMap │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.StyleMap = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: styles │ │ │ │ │ + * {Object} Hash of {<OpenLayers.Style>}, keyed by names of well known │ │ │ │ │ + * rendering intents (e.g. "default", "temporary", "select", "delete"). │ │ │ │ │ + */ │ │ │ │ │ + styles: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: extendDefault │ │ │ │ │ + * {Boolean} if true, every render intent will extend the symbolizers │ │ │ │ │ + * specified for the "default" intent at rendering time. Otherwise, every │ │ │ │ │ + * rendering intent will be treated as a completely independent style. │ │ │ │ │ + */ │ │ │ │ │ + extendDefault: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.StyleMap │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * style - {Object} Optional. Either a style hash, or a style object, or │ │ │ │ │ + * a hash of style objects (style hashes) keyed by rendering │ │ │ │ │ + * intent. If just one style hash or style object is passed, │ │ │ │ │ + * this will be used for all known render intents (default, │ │ │ │ │ + * select, temporary) │ │ │ │ │ + * options - {Object} optional hash of additional options for this │ │ │ │ │ + * instance │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(style, options) { │ │ │ │ │ + this.styles = { │ │ │ │ │ + "default": new OpenLayers.Style( │ │ │ │ │ + OpenLayers.Feature.Vector.style["default"]), │ │ │ │ │ + "select": new OpenLayers.Style( │ │ │ │ │ + OpenLayers.Feature.Vector.style["select"]), │ │ │ │ │ + "temporary": new OpenLayers.Style( │ │ │ │ │ + OpenLayers.Feature.Vector.style["temporary"]), │ │ │ │ │ + "delete": new OpenLayers.Style( │ │ │ │ │ + OpenLayers.Feature.Vector.style["delete"]) │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // take whatever the user passed as style parameter and convert it │ │ │ │ │ + // into parts of stylemap. │ │ │ │ │ + if (style instanceof OpenLayers.Style) { │ │ │ │ │ + // user passed a style object │ │ │ │ │ + this.styles["default"] = style; │ │ │ │ │ + this.styles["select"] = style; │ │ │ │ │ + this.styles["temporary"] = style; │ │ │ │ │ + this.styles["delete"] = style; │ │ │ │ │ + } else if (typeof style == "object") { │ │ │ │ │ + for (var key in style) { │ │ │ │ │ + if (style[key] instanceof OpenLayers.Style) { │ │ │ │ │ + // user passed a hash of style objects │ │ │ │ │ + this.styles[key] = style[key]; │ │ │ │ │ + } else if (typeof style[key] == "object") { │ │ │ │ │ + // user passsed a hash of style hashes │ │ │ │ │ + this.styles[key] = new OpenLayers.Style(style[key]); │ │ │ │ │ + } else { │ │ │ │ │ + // user passed a style hash (i.e. symbolizer) │ │ │ │ │ + this.styles["default"] = new OpenLayers.Style(style); │ │ │ │ │ + this.styles["select"] = new OpenLayers.Style(style); │ │ │ │ │ + this.styles["temporary"] = new OpenLayers.Style(style); │ │ │ │ │ + this.styles["delete"] = new OpenLayers.Style(style); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var key in this.styles) { │ │ │ │ │ + this.styles[key].destroy(); │ │ │ │ │ + } │ │ │ │ │ + this.styles = null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createSymbolizer │ │ │ │ │ + * Creates the symbolizer for a feature for a render intent. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature>} The feature to evaluate the rules │ │ │ │ │ + * of the intended style against. │ │ │ │ │ + * intent - {String} The intent determines the symbolizer that will be │ │ │ │ │ + * used to draw the feature. Well known intents are "default" │ │ │ │ │ + * (for just drawing the features), "select" (for selected │ │ │ │ │ + * features) and "temporary" (for drawing features). │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} symbolizer hash │ │ │ │ │ + */ │ │ │ │ │ + createSymbolizer: function(feature, intent) { │ │ │ │ │ + if (!feature) { │ │ │ │ │ + feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ + } │ │ │ │ │ + if (!this.styles[intent]) { │ │ │ │ │ + intent = "default"; │ │ │ │ │ + } │ │ │ │ │ + feature.renderIntent = intent; │ │ │ │ │ + var defaultSymbolizer = {}; │ │ │ │ │ + if (this.extendDefault && intent != "default") { │ │ │ │ │ + defaultSymbolizer = this.styles["default"].createSymbolizer(feature); │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Util.extend(defaultSymbolizer, │ │ │ │ │ + this.styles[intent].createSymbolizer(feature)); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: addUniqueValueRules │ │ │ │ │ + * Convenience method to create comparison rules for unique values of a │ │ │ │ │ + * property. The rules will be added to the style object for a specified │ │ │ │ │ + * rendering intent. This method is a shortcut for creating something like │ │ │ │ │ + * the "unique value legends" familiar from well known desktop GIS systems │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * renderIntent - {String} rendering intent to add the rules to │ │ │ │ │ + * property - {String} values of feature attributes to create the │ │ │ │ │ + * rules for │ │ │ │ │ + * symbolizers - {Object} Hash of symbolizers, keyed by the desired │ │ │ │ │ + * property values │ │ │ │ │ + * context - {Object} An optional object with properties that │ │ │ │ │ + * symbolizers' property values should be evaluated │ │ │ │ │ + * against. If no context is specified, feature.attributes │ │ │ │ │ + * will be used │ │ │ │ │ + */ │ │ │ │ │ + addUniqueValueRules: function(renderIntent, property, symbolizers, context) { │ │ │ │ │ + var rules = []; │ │ │ │ │ + for (var value in symbolizers) { │ │ │ │ │ + rules.push(new OpenLayers.Rule({ │ │ │ │ │ + symbolizer: symbolizers[value], │ │ │ │ │ + context: context, │ │ │ │ │ + filter: new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.EQUAL_TO, │ │ │ │ │ + property: property, │ │ │ │ │ + value: value │ │ │ │ │ + }) │ │ │ │ │ + })); │ │ │ │ │ + } │ │ │ │ │ + this.styles[renderIntent].addRules(rules); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.StyleMap" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Kinetic.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Animation.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Kinetic = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: threshold │ │ │ │ │ + * In most cases changing the threshold isn't needed. │ │ │ │ │ + * In px/ms, default to 0. │ │ │ │ │ + */ │ │ │ │ │ + threshold: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: deceleration │ │ │ │ │ + * {Float} the deseleration in px/ms², default to 0.0035. │ │ │ │ │ + */ │ │ │ │ │ + deceleration: 0.0035, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: nbPoints │ │ │ │ │ + * {Integer} the number of points we use to calculate the kinetic │ │ │ │ │ + * initial values. │ │ │ │ │ + */ │ │ │ │ │ + nbPoints: 100, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: delay │ │ │ │ │ + * {Float} time to consider to calculate the kinetic initial values. │ │ │ │ │ + * In ms, default to 200. │ │ │ │ │ + */ │ │ │ │ │ + delay: 200, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: points │ │ │ │ │ + * List of points use to calculate the kinetic initial values. │ │ │ │ │ + */ │ │ │ │ │ + points: undefined, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: timerId │ │ │ │ │ + * ID of the timer. │ │ │ │ │ + */ │ │ │ │ │ + timerId: undefined, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Kinetic │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: begin │ │ │ │ │ + * Begins the dragging. │ │ │ │ │ + */ │ │ │ │ │ + begin: function() { │ │ │ │ │ + OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ + this.timerId = undefined; │ │ │ │ │ + this.points = []; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: update │ │ │ │ │ + * Updates during the dragging. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} The new position. │ │ │ │ │ + */ │ │ │ │ │ + update: function(xy) { │ │ │ │ │ + this.points.unshift({ │ │ │ │ │ + xy: xy, │ │ │ │ │ + tick: new Date().getTime() │ │ │ │ │ + }); │ │ │ │ │ + if (this.points.length > this.nbPoints) { │ │ │ │ │ + this.points.pop(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: end │ │ │ │ │ + * Ends the dragging, start the kinetic. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} The last position. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object with two properties: "speed", and "theta". The │ │ │ │ │ + * "speed" and "theta" values are to be passed to the move │ │ │ │ │ + * function when starting the animation. │ │ │ │ │ + */ │ │ │ │ │ + end: function(xy) { │ │ │ │ │ + var last, now = new Date().getTime(); │ │ │ │ │ + for (var i = 0, l = this.points.length, point; i < l; i++) { │ │ │ │ │ + point = this.points[i]; │ │ │ │ │ + if (now - point.tick > this.delay) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + last = point; │ │ │ │ │ + } │ │ │ │ │ + if (!last) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var time = new Date().getTime() - last.tick; │ │ │ │ │ + var dist = Math.sqrt(Math.pow(xy.x - last.xy.x, 2) + │ │ │ │ │ + Math.pow(xy.y - last.xy.y, 2)); │ │ │ │ │ + var speed = dist / time; │ │ │ │ │ + if (speed == 0 || speed < this.threshold) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var theta = Math.asin((xy.y - last.xy.y) / dist); │ │ │ │ │ + if (last.xy.x <= xy.x) { │ │ │ │ │ + theta = Math.PI - theta; │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + speed: speed, │ │ │ │ │ + theta: theta │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: move │ │ │ │ │ + * Launch the kinetic move pan. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * info - {Object} An object with two properties, "speed", and "theta". │ │ │ │ │ + * These values are those returned from the "end" call. │ │ │ │ │ + * callback - {Function} Function called on every step of the animation, │ │ │ │ │ + * receives x, y (values to pan), end (is the last point). │ │ │ │ │ + */ │ │ │ │ │ + move: function(info, callback) { │ │ │ │ │ + var v0 = info.speed; │ │ │ │ │ + var fx = Math.cos(info.theta); │ │ │ │ │ + var fy = -Math.sin(info.theta); │ │ │ │ │ + │ │ │ │ │ + var initialTime = new Date().getTime(); │ │ │ │ │ + │ │ │ │ │ + var lastX = 0; │ │ │ │ │ + var lastY = 0; │ │ │ │ │ + │ │ │ │ │ + var timerCallback = function() { │ │ │ │ │ + if (this.timerId == null) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var t = new Date().getTime() - initialTime; │ │ │ │ │ + │ │ │ │ │ + var p = (-this.deceleration * Math.pow(t, 2)) / 2.0 + v0 * t; │ │ │ │ │ + var x = p * fx; │ │ │ │ │ + var y = p * fy; │ │ │ │ │ + │ │ │ │ │ + var args = {}; │ │ │ │ │ + args.end = false; │ │ │ │ │ + var v = -this.deceleration * t + v0; │ │ │ │ │ + │ │ │ │ │ + if (v <= 0) { │ │ │ │ │ + OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + args.end = true; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + args.x = x - lastX; │ │ │ │ │ + args.y = y - lastY; │ │ │ │ │ + lastX = x; │ │ │ │ │ + lastY = y; │ │ │ │ │ + callback(args.x, args.y, args.end); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + this.timerId = OpenLayers.Animation.start( │ │ │ │ │ + OpenLayers.Function.bind(timerCallback, this) │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Kinetic" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ OpenLayers/Rule.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -11081,8449 +11246,9792 @@ │ │ │ │ │ options.context = this.context && OpenLayers.Util.extend({}, this.context); │ │ │ │ │ return new OpenLayers.Rule(options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Rule" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/StyleMap.js │ │ │ │ │ + OpenLayers/Renderer.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Style.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.StyleMap │ │ │ │ │ + * Class: OpenLayers.Renderer │ │ │ │ │ + * This is the base class for all renderers. │ │ │ │ │ + * │ │ │ │ │ + * This is based on a merger code written by Paul Spencer and Bertil Chapuis. │ │ │ │ │ + * It is largely composed of virtual functions that are to be implemented │ │ │ │ │ + * in technology-specific subclasses, but there is some generic code too. │ │ │ │ │ + * │ │ │ │ │ + * The functions that *are* implemented here merely deal with the maintenance │ │ │ │ │ + * of the size and extent variables, as well as the cached 'resolution' │ │ │ │ │ + * value. │ │ │ │ │ + * │ │ │ │ │ + * A note to the user that all subclasses should use getResolution() instead │ │ │ │ │ + * of directly accessing this.resolution in order to correctly use the │ │ │ │ │ + * cacheing system. │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.StyleMap = OpenLayers.Class({ │ │ │ │ │ +OpenLayers.Renderer = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: styles │ │ │ │ │ - * {Object} Hash of {<OpenLayers.Style>}, keyed by names of well known │ │ │ │ │ - * rendering intents (e.g. "default", "temporary", "select", "delete"). │ │ │ │ │ + /** │ │ │ │ │ + * Property: container │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - styles: null, │ │ │ │ │ + container: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: extendDefault │ │ │ │ │ - * {Boolean} if true, every render intent will extend the symbolizers │ │ │ │ │ - * specified for the "default" intent at rendering time. Otherwise, every │ │ │ │ │ - * rendering intent will be treated as a completely independent style. │ │ │ │ │ + * Property: root │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - extendDefault: true, │ │ │ │ │ + root: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.StyleMap │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * style - {Object} Optional. Either a style hash, or a style object, or │ │ │ │ │ - * a hash of style objects (style hashes) keyed by rendering │ │ │ │ │ - * intent. If just one style hash or style object is passed, │ │ │ │ │ - * this will be used for all known render intents (default, │ │ │ │ │ - * select, temporary) │ │ │ │ │ - * options - {Object} optional hash of additional options for this │ │ │ │ │ - * instance │ │ │ │ │ + /** │ │ │ │ │ + * Property: extent │ │ │ │ │ + * {<OpenLayers.Bounds>} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(style, options) { │ │ │ │ │ - this.styles = { │ │ │ │ │ - "default": new OpenLayers.Style( │ │ │ │ │ - OpenLayers.Feature.Vector.style["default"]), │ │ │ │ │ - "select": new OpenLayers.Style( │ │ │ │ │ - OpenLayers.Feature.Vector.style["select"]), │ │ │ │ │ - "temporary": new OpenLayers.Style( │ │ │ │ │ - OpenLayers.Feature.Vector.style["temporary"]), │ │ │ │ │ - "delete": new OpenLayers.Style( │ │ │ │ │ - OpenLayers.Feature.Vector.style["delete"]) │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // take whatever the user passed as style parameter and convert it │ │ │ │ │ - // into parts of stylemap. │ │ │ │ │ - if (style instanceof OpenLayers.Style) { │ │ │ │ │ - // user passed a style object │ │ │ │ │ - this.styles["default"] = style; │ │ │ │ │ - this.styles["select"] = style; │ │ │ │ │ - this.styles["temporary"] = style; │ │ │ │ │ - this.styles["delete"] = style; │ │ │ │ │ - } else if (typeof style == "object") { │ │ │ │ │ - for (var key in style) { │ │ │ │ │ - if (style[key] instanceof OpenLayers.Style) { │ │ │ │ │ - // user passed a hash of style objects │ │ │ │ │ - this.styles[key] = style[key]; │ │ │ │ │ - } else if (typeof style[key] == "object") { │ │ │ │ │ - // user passsed a hash of style hashes │ │ │ │ │ - this.styles[key] = new OpenLayers.Style(style[key]); │ │ │ │ │ - } else { │ │ │ │ │ - // user passed a style hash (i.e. symbolizer) │ │ │ │ │ - this.styles["default"] = new OpenLayers.Style(style); │ │ │ │ │ - this.styles["select"] = new OpenLayers.Style(style); │ │ │ │ │ - this.styles["temporary"] = new OpenLayers.Style(style); │ │ │ │ │ - this.styles["delete"] = new OpenLayers.Style(style); │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - }, │ │ │ │ │ + extent: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * Property: locked │ │ │ │ │ + * {Boolean} If the renderer is currently in a state where many things │ │ │ │ │ + * are changing, the 'locked' property is set to true. This means │ │ │ │ │ + * that renderers can expect at least one more drawFeature event to be │ │ │ │ │ + * called with the 'locked' property set to 'true': In some renderers, │ │ │ │ │ + * this might make sense to use as a 'only update local information' │ │ │ │ │ + * flag. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var key in this.styles) { │ │ │ │ │ - this.styles[key].destroy(); │ │ │ │ │ - } │ │ │ │ │ - this.styles = null; │ │ │ │ │ - }, │ │ │ │ │ + locked: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createSymbolizer │ │ │ │ │ - * Creates the symbolizer for a feature for a render intent. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature>} The feature to evaluate the rules │ │ │ │ │ - * of the intended style against. │ │ │ │ │ - * intent - {String} The intent determines the symbolizer that will be │ │ │ │ │ - * used to draw the feature. Well known intents are "default" │ │ │ │ │ - * (for just drawing the features), "select" (for selected │ │ │ │ │ - * features) and "temporary" (for drawing features). │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} symbolizer hash │ │ │ │ │ + /** │ │ │ │ │ + * Property: size │ │ │ │ │ + * {<OpenLayers.Size>} │ │ │ │ │ */ │ │ │ │ │ - createSymbolizer: function(feature, intent) { │ │ │ │ │ - if (!feature) { │ │ │ │ │ - feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ - } │ │ │ │ │ - if (!this.styles[intent]) { │ │ │ │ │ - intent = "default"; │ │ │ │ │ - } │ │ │ │ │ - feature.renderIntent = intent; │ │ │ │ │ - var defaultSymbolizer = {}; │ │ │ │ │ - if (this.extendDefault && intent != "default") { │ │ │ │ │ - defaultSymbolizer = this.styles["default"].createSymbolizer(feature); │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Util.extend(defaultSymbolizer, │ │ │ │ │ - this.styles[intent].createSymbolizer(feature)); │ │ │ │ │ - }, │ │ │ │ │ + size: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: addUniqueValueRules │ │ │ │ │ - * Convenience method to create comparison rules for unique values of a │ │ │ │ │ - * property. The rules will be added to the style object for a specified │ │ │ │ │ - * rendering intent. This method is a shortcut for creating something like │ │ │ │ │ - * the "unique value legends" familiar from well known desktop GIS systems │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * renderIntent - {String} rendering intent to add the rules to │ │ │ │ │ - * property - {String} values of feature attributes to create the │ │ │ │ │ - * rules for │ │ │ │ │ - * symbolizers - {Object} Hash of symbolizers, keyed by the desired │ │ │ │ │ - * property values │ │ │ │ │ - * context - {Object} An optional object with properties that │ │ │ │ │ - * symbolizers' property values should be evaluated │ │ │ │ │ - * against. If no context is specified, feature.attributes │ │ │ │ │ - * will be used │ │ │ │ │ + * Property: resolution │ │ │ │ │ + * {Float} cache of current map resolution │ │ │ │ │ */ │ │ │ │ │ - addUniqueValueRules: function(renderIntent, property, symbolizers, context) { │ │ │ │ │ - var rules = []; │ │ │ │ │ - for (var value in symbolizers) { │ │ │ │ │ - rules.push(new OpenLayers.Rule({ │ │ │ │ │ - symbolizer: symbolizers[value], │ │ │ │ │ - context: context, │ │ │ │ │ - filter: new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.EQUAL_TO, │ │ │ │ │ - property: property, │ │ │ │ │ - value: value │ │ │ │ │ - }) │ │ │ │ │ - })); │ │ │ │ │ - } │ │ │ │ │ - this.styles[renderIntent].addRules(rules); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.StyleMap" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Map.js │ │ │ │ │ - * @requires OpenLayers/Projection.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer = OpenLayers.Class({ │ │ │ │ │ + resolution: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: id │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - id: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: name │ │ │ │ │ - * {String} │ │ │ │ │ + * Property: map │ │ │ │ │ + * {<OpenLayers.Map>} Reference to the map -- this is set in Vector's setMap() │ │ │ │ │ */ │ │ │ │ │ - name: null, │ │ │ │ │ + map: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: div │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + /** │ │ │ │ │ + * Property: featureDx │ │ │ │ │ + * {Number} Feature offset in x direction. Will be calculated for and │ │ │ │ │ + * applied to the current feature while rendering (see │ │ │ │ │ + * <calculateFeatureDx>). │ │ │ │ │ */ │ │ │ │ │ - div: null, │ │ │ │ │ + featureDx: 0, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: opacity │ │ │ │ │ - * {Float} The layer's opacity. Float number between 0.0 and 1.0. Default │ │ │ │ │ - * is 1. │ │ │ │ │ + * Constructor: OpenLayers.Renderer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * containerID - {<String>} │ │ │ │ │ + * options - {Object} options for this renderer. See sublcasses for │ │ │ │ │ + * supported options. │ │ │ │ │ */ │ │ │ │ │ - opacity: 1, │ │ │ │ │ + initialize: function(containerID, options) { │ │ │ │ │ + this.container = OpenLayers.Util.getElement(containerID); │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: alwaysInRange │ │ │ │ │ - * {Boolean} If a layer's display should not be scale-based, this should │ │ │ │ │ - * be set to true. This will cause the layer, as an overlay, to always │ │ │ │ │ - * be 'active', by always returning true from the calculateInRange() │ │ │ │ │ - * function. │ │ │ │ │ - * │ │ │ │ │ - * If not explicitly specified for a layer, its value will be │ │ │ │ │ - * determined on startup in initResolutions() based on whether or not │ │ │ │ │ - * any scale-specific properties have been set as options on the │ │ │ │ │ - * layer. If no scale-specific options have been set on the layer, we │ │ │ │ │ - * assume that it should always be in range. │ │ │ │ │ - * │ │ │ │ │ - * See #987 for more info. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ */ │ │ │ │ │ - alwaysInRange: null, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.container = null; │ │ │ │ │ + this.extent = null; │ │ │ │ │ + this.size = null; │ │ │ │ │ + this.resolution = null; │ │ │ │ │ + this.map = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: RESOLUTION_PROPERTIES │ │ │ │ │ - * {Array} The properties that are used for calculating resolutions │ │ │ │ │ - * information. │ │ │ │ │ + * APIMethod: supported │ │ │ │ │ + * This should be overridden by specific subclasses │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the browser supports the renderer class │ │ │ │ │ */ │ │ │ │ │ - RESOLUTION_PROPERTIES: [ │ │ │ │ │ - 'scales', 'resolutions', │ │ │ │ │ - 'maxScale', 'minScale', │ │ │ │ │ - 'maxResolution', 'minResolution', │ │ │ │ │ - 'numZoomLevels', 'maxZoomLevel' │ │ │ │ │ - ], │ │ │ │ │ + supported: function() { │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * layer.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ + * Method: setExtent │ │ │ │ │ + * Set the visible part of the layer. │ │ │ │ │ * │ │ │ │ │ - * Listeners will be called with a reference to an event object. The │ │ │ │ │ - * properties of this event depends on exactly what happened. │ │ │ │ │ + * Resolution has probably changed, so we nullify the resolution │ │ │ │ │ + * cache (this.resolution) -- this way it will be re-computed when │ │ │ │ │ + * next it is needed. │ │ │ │ │ + * We nullify the resolution cache (this.resolution) if resolutionChanged │ │ │ │ │ + * is set to true - this way it will be re-computed on the next │ │ │ │ │ + * getResolution() request. │ │ │ │ │ * │ │ │ │ │ - * All event objects have at least the following properties: │ │ │ │ │ - * object - {Object} A reference to layer.events.object. │ │ │ │ │ - * element - {DOMElement} A reference to layer.events.element. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} │ │ │ │ │ + * resolutionChanged - {Boolean} │ │ │ │ │ * │ │ │ │ │ - * Supported map event types: │ │ │ │ │ - * loadstart - Triggered when layer loading starts. When using a Vector │ │ │ │ │ - * layer with a Fixed or BBOX strategy, the event object includes │ │ │ │ │ - * a *filter* property holding the OpenLayers.Filter used when │ │ │ │ │ - * calling read on the protocol. │ │ │ │ │ - * loadend - Triggered when layer loading ends. When using a Vector layer │ │ │ │ │ - * with a Fixed or BBOX strategy, the event object includes a │ │ │ │ │ - * *response* property holding an OpenLayers.Protocol.Response object. │ │ │ │ │ - * visibilitychanged - Triggered when the layer's visibility property is │ │ │ │ │ - * changed, e.g. by turning the layer on or off in the layer switcher. │ │ │ │ │ - * Note that the actual visibility of the layer can also change if it │ │ │ │ │ - * gets out of range (see <calculateInRange>). If you also want to catch │ │ │ │ │ - * these cases, register for the map's 'changelayer' event instead. │ │ │ │ │ - * move - Triggered when layer moves (triggered with every mousemove │ │ │ │ │ - * during a drag). │ │ │ │ │ - * moveend - Triggered when layer is done moving, object passed as │ │ │ │ │ - * argument has a zoomChanged boolean property which tells that the │ │ │ │ │ - * zoom has changed. │ │ │ │ │ - * added - Triggered after the layer is added to a map. Listeners will │ │ │ │ │ - * receive an object with a *map* property referencing the map and a │ │ │ │ │ - * *layer* property referencing the layer. │ │ │ │ │ - * removed - Triggered after the layer is removed from the map. Listeners │ │ │ │ │ - * will receive an object with a *map* property referencing the map and │ │ │ │ │ - * a *layer* property referencing the layer. │ │ │ │ │ - */ │ │ │ │ │ - events: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: map │ │ │ │ │ - * {<OpenLayers.Map>} This variable is set when the layer is added to │ │ │ │ │ - * the map, via the accessor function setMap(). │ │ │ │ │ - */ │ │ │ │ │ - map: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} Whether or not the layer is a base layer. This should be set │ │ │ │ │ - * individually by all subclasses. Default is false │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ + * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ + * False otherwise. │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: false, │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + this.extent = extent.clone(); │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + var ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ + extent = extent.scale(1 / ratio); │ │ │ │ │ + this.extent = extent.wrapDateLine(this.map.getMaxExtent()).scale(ratio); │ │ │ │ │ + } │ │ │ │ │ + if (resolutionChanged) { │ │ │ │ │ + this.resolution = null; │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: alpha │ │ │ │ │ - * {Boolean} The layer's images have an alpha channel. Default is false. │ │ │ │ │ + * Method: setSize │ │ │ │ │ + * Sets the size of the drawing surface. │ │ │ │ │ + * │ │ │ │ │ + * Resolution has probably changed, so we nullify the resolution │ │ │ │ │ + * cache (this.resolution) -- this way it will be re-computed when │ │ │ │ │ + * next it is needed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * size - {<OpenLayers.Size>} │ │ │ │ │ */ │ │ │ │ │ - alpha: false, │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + this.size = size.clone(); │ │ │ │ │ + this.resolution = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: displayInLayerSwitcher │ │ │ │ │ - * {Boolean} Display the layer's name in the layer switcher. Default is │ │ │ │ │ - * true. │ │ │ │ │ + * Method: getResolution │ │ │ │ │ + * Uses cached copy of resolution if available to minimize computing │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The current map's resolution │ │ │ │ │ */ │ │ │ │ │ - displayInLayerSwitcher: true, │ │ │ │ │ + getResolution: function() { │ │ │ │ │ + this.resolution = this.resolution || this.map.getResolution(); │ │ │ │ │ + return this.resolution; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: visibility │ │ │ │ │ - * {Boolean} The layer should be displayed in the map. Default is true. │ │ │ │ │ + * Method: drawFeature │ │ │ │ │ + * Draw the feature. The optional style argument can be used │ │ │ │ │ + * to override the feature's own style. This method should only │ │ │ │ │ + * be called from layer.drawFeature(). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * style - {<Object>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the feature has been drawn completely, false if not, │ │ │ │ │ + * undefined if the feature had no geometry │ │ │ │ │ */ │ │ │ │ │ - visibility: true, │ │ │ │ │ + drawFeature: function(feature, style) { │ │ │ │ │ + if (style == null) { │ │ │ │ │ + style = feature.style; │ │ │ │ │ + } │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + var bounds = feature.geometry.getBounds(); │ │ │ │ │ + if (bounds) { │ │ │ │ │ + var worldBounds; │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + worldBounds = this.map.getMaxExtent(); │ │ │ │ │ + } │ │ │ │ │ + if (!bounds.intersectsBounds(this.extent, { │ │ │ │ │ + worldBounds: worldBounds │ │ │ │ │ + })) { │ │ │ │ │ + style = { │ │ │ │ │ + display: "none" │ │ │ │ │ + }; │ │ │ │ │ + } else { │ │ │ │ │ + this.calculateFeatureDx(bounds, worldBounds); │ │ │ │ │ + } │ │ │ │ │ + var rendered = this.drawGeometry(feature.geometry, style, feature.id); │ │ │ │ │ + if (style.display != "none" && style.label && rendered !== false) { │ │ │ │ │ + │ │ │ │ │ + var location = feature.geometry.getCentroid(); │ │ │ │ │ + if (style.labelXOffset || style.labelYOffset) { │ │ │ │ │ + var xOffset = isNaN(style.labelXOffset) ? 0 : style.labelXOffset; │ │ │ │ │ + var yOffset = isNaN(style.labelYOffset) ? 0 : style.labelYOffset; │ │ │ │ │ + var res = this.getResolution(); │ │ │ │ │ + location.move(xOffset * res, yOffset * res); │ │ │ │ │ + } │ │ │ │ │ + this.drawText(feature.id, style, location); │ │ │ │ │ + } else { │ │ │ │ │ + this.removeText(feature.id); │ │ │ │ │ + } │ │ │ │ │ + return rendered; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: attribution │ │ │ │ │ - * {String} Attribution string, displayed when an │ │ │ │ │ - * <OpenLayers.Control.Attribution> has been added to the map. │ │ │ │ │ + * Method: calculateFeatureDx │ │ │ │ │ + * {Number} Calculates the feature offset in x direction. Looking at the │ │ │ │ │ + * center of the feature bounds and the renderer extent, we calculate how │ │ │ │ │ + * many world widths the two are away from each other. This distance is │ │ │ │ │ + * used to shift the feature as close as possible to the center of the │ │ │ │ │ + * current enderer extent, which ensures that the feature is visible in the │ │ │ │ │ + * current viewport. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} Bounds of the feature │ │ │ │ │ + * worldBounds - {<OpenLayers.Bounds>} Bounds of the world │ │ │ │ │ */ │ │ │ │ │ - attribution: null, │ │ │ │ │ + calculateFeatureDx: function(bounds, worldBounds) { │ │ │ │ │ + this.featureDx = 0; │ │ │ │ │ + if (worldBounds) { │ │ │ │ │ + var worldWidth = worldBounds.getWidth(), │ │ │ │ │ + rendererCenterX = (this.extent.left + this.extent.right) / 2, │ │ │ │ │ + featureCenterX = (bounds.left + bounds.right) / 2, │ │ │ │ │ + worldsAway = Math.round((featureCenterX - rendererCenterX) / worldWidth); │ │ │ │ │ + this.featureDx = worldsAway * worldWidth; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: inRange │ │ │ │ │ - * {Boolean} The current map resolution is within the layer's min/max │ │ │ │ │ - * range. This is set in <OpenLayers.Map.setCenter> whenever the zoom │ │ │ │ │ - * changes. │ │ │ │ │ + * Method: drawGeometry │ │ │ │ │ + * │ │ │ │ │ + * Draw a geometry. This should only be called from the renderer itself. │ │ │ │ │ + * Use layer.drawFeature() from outside the renderer. │ │ │ │ │ + * virtual function │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {<String>} │ │ │ │ │ */ │ │ │ │ │ - inRange: false, │ │ │ │ │ + drawGeometry: function(geometry, style, featureId) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Propery: imageSize │ │ │ │ │ - * {<OpenLayers.Size>} For layers with a gutter, the image is larger than │ │ │ │ │ - * the tile by twice the gutter in each dimension. │ │ │ │ │ - */ │ │ │ │ │ - imageSize: null, │ │ │ │ │ - │ │ │ │ │ - // OPTIONS │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: options │ │ │ │ │ - * {Object} An optional object whose properties will be set on the layer. │ │ │ │ │ - * Any of the layer properties can be set as a property of the options │ │ │ │ │ - * object and sent to the constructor when the layer is created. │ │ │ │ │ + * Method: drawText │ │ │ │ │ + * Function for drawing text labels. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * style - │ │ │ │ │ + * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ */ │ │ │ │ │ - options: null, │ │ │ │ │ + drawText: function(featureId, style, location) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: eventListeners │ │ │ │ │ - * {Object} If set as an option at construction, the eventListeners │ │ │ │ │ - * object will be registered with <OpenLayers.Events.on>. Object │ │ │ │ │ - * structure must be a listeners object as shown in the example for │ │ │ │ │ - * the events.on method. │ │ │ │ │ + * Method: removeText │ │ │ │ │ + * Function for removing text labels. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureId - {String} │ │ │ │ │ */ │ │ │ │ │ - eventListeners: null, │ │ │ │ │ + removeText: function(featureId) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: gutter │ │ │ │ │ - * {Integer} Determines the width (in pixels) of the gutter around image │ │ │ │ │ - * tiles to ignore. By setting this property to a non-zero value, │ │ │ │ │ - * images will be requested that are wider and taller than the tile │ │ │ │ │ - * size by a value of 2 x gutter. This allows artifacts of rendering │ │ │ │ │ - * at tile edges to be ignored. Set a gutter value that is equal to │ │ │ │ │ - * half the size of the widest symbol that needs to be displayed. │ │ │ │ │ - * Defaults to zero. Non-tiled layers always have zero gutter. │ │ │ │ │ + * Method: clear │ │ │ │ │ + * Clear all vectors from the renderer. │ │ │ │ │ + * virtual function. │ │ │ │ │ */ │ │ │ │ │ - gutter: 0, │ │ │ │ │ + clear: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: projection │ │ │ │ │ - * {<OpenLayers.Projection>} or {<String>} Specifies the projection of the layer. │ │ │ │ │ - * Can be set in the layer options. If not specified in the layer options, │ │ │ │ │ - * it is set to the default projection specified in the map, │ │ │ │ │ - * when the layer is added to the map. │ │ │ │ │ - * Projection along with default maxExtent and resolutions │ │ │ │ │ - * are set automatically with commercial baselayers in EPSG:3857, │ │ │ │ │ - * such as Google, Bing and OpenStreetMap, and do not need to be specified. │ │ │ │ │ - * Otherwise, if specifying projection, also set maxExtent, │ │ │ │ │ - * maxResolution or resolutions as appropriate. │ │ │ │ │ - * When using vector layers with strategies, layer projection should be set │ │ │ │ │ - * to the projection of the source data if that is different from the map default. │ │ │ │ │ - * │ │ │ │ │ - * Can be either a string or an <OpenLayers.Projection> object; │ │ │ │ │ - * if a string is passed, will be converted to an object when │ │ │ │ │ - * the layer is added to the map. │ │ │ │ │ + * Method: getFeatureIdFromEvent │ │ │ │ │ + * Returns a feature id from an event on the renderer. │ │ │ │ │ + * How this happens is specific to the renderer. This should be │ │ │ │ │ + * called from layer.getFeatureFromEvent(). │ │ │ │ │ + * Virtual function. │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A feature id or undefined. │ │ │ │ │ */ │ │ │ │ │ - projection: null, │ │ │ │ │ + getFeatureIdFromEvent: function(evt) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: units │ │ │ │ │ - * {String} The layer map units. Defaults to null. Possible values │ │ │ │ │ - * are 'degrees' (or 'dd'), 'm', 'ft', 'km', 'mi', 'inches'. │ │ │ │ │ - * Normally taken from the projection. │ │ │ │ │ - * Only required if both map and layers do not define a projection, │ │ │ │ │ - * or if they define a projection which does not define units. │ │ │ │ │ + * Method: eraseFeatures │ │ │ │ │ + * This is called by the layer to erase features │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ */ │ │ │ │ │ - units: null, │ │ │ │ │ + eraseFeatures: function(features) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + this.eraseGeometry(feature.geometry, feature.id); │ │ │ │ │ + this.removeText(feature.id); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: scales │ │ │ │ │ - * {Array} An array of map scales in descending order. The values in the │ │ │ │ │ - * array correspond to the map scale denominator. Note that these │ │ │ │ │ - * values only make sense if the display (monitor) resolution of the │ │ │ │ │ - * client is correctly guessed by whomever is configuring the │ │ │ │ │ - * application. In addition, the units property must also be set. │ │ │ │ │ - * Use <resolutions> instead wherever possible. │ │ │ │ │ + * Method: eraseGeometry │ │ │ │ │ + * Remove a geometry from the renderer (by id). │ │ │ │ │ + * virtual function. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ */ │ │ │ │ │ - scales: null, │ │ │ │ │ + eraseGeometry: function(geometry, featureId) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: resolutions │ │ │ │ │ - * {Array} A list of map resolutions (map units per pixel) in descending │ │ │ │ │ - * order. If this is not set in the layer constructor, it will be set │ │ │ │ │ - * based on other resolution related properties (maxExtent, │ │ │ │ │ - * maxResolution, maxScale, etc.). │ │ │ │ │ + * Method: moveRoot │ │ │ │ │ + * moves this renderer's root to a (different) renderer. │ │ │ │ │ + * To be implemented by subclasses that require a common renderer root for │ │ │ │ │ + * feature selection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * renderer - {<OpenLayers.Renderer>} target renderer for the moved root │ │ │ │ │ */ │ │ │ │ │ - resolutions: null, │ │ │ │ │ + moveRoot: function(renderer) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: maxExtent │ │ │ │ │ - * {<OpenLayers.Bounds>|Array} If provided as an array, the array │ │ │ │ │ - * should consist of four values (left, bottom, right, top). │ │ │ │ │ - * The maximum extent for the layer. Defaults to null. │ │ │ │ │ + * Method: getRenderLayerId │ │ │ │ │ + * Gets the layer that this renderer's output appears on. If moveRoot was │ │ │ │ │ + * used, this will be different from the id of the layer containing the │ │ │ │ │ + * features rendered by this renderer. │ │ │ │ │ * │ │ │ │ │ - * The center of these bounds will not stray outside │ │ │ │ │ - * of the viewport extent during panning. In addition, if │ │ │ │ │ - * <displayOutsideMaxExtent> is set to false, data will not be │ │ │ │ │ - * requested that falls completely outside of these bounds. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} the id of the output layer. │ │ │ │ │ */ │ │ │ │ │ - maxExtent: null, │ │ │ │ │ + getRenderLayerId: function() { │ │ │ │ │ + return this.container.id; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: minExtent │ │ │ │ │ - * {<OpenLayers.Bounds>|Array} If provided as an array, the array │ │ │ │ │ - * should consist of four values (left, bottom, right, top). │ │ │ │ │ - * The minimum extent for the layer. Defaults to null. │ │ │ │ │ + * Method: applyDefaultSymbolizer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * symbolizer - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} │ │ │ │ │ */ │ │ │ │ │ - minExtent: null, │ │ │ │ │ + applyDefaultSymbolizer: function(symbolizer) { │ │ │ │ │ + var result = OpenLayers.Util.extend({}, │ │ │ │ │ + OpenLayers.Renderer.defaultSymbolizer); │ │ │ │ │ + if (symbolizer.stroke === false) { │ │ │ │ │ + delete result.strokeWidth; │ │ │ │ │ + delete result.strokeColor; │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.fill === false) { │ │ │ │ │ + delete result.fillColor; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.extend(result, symbolizer); │ │ │ │ │ + return result; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: maxResolution │ │ │ │ │ - * {Float} Default max is 360 deg / 256 px, which corresponds to │ │ │ │ │ - * zoom level 0 on gmaps. Specify a different value in the layer │ │ │ │ │ - * options if you are not using the default <OpenLayers.Map.tileSize> │ │ │ │ │ - * and displaying the whole world. │ │ │ │ │ - */ │ │ │ │ │ - maxResolution: null, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: minResolution │ │ │ │ │ - * {Float} │ │ │ │ │ - */ │ │ │ │ │ - minResolution: null, │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.defaultSymbolizer │ │ │ │ │ + * {Object} Properties from this symbolizer will be applied to symbolizers │ │ │ │ │ + * with missing properties. This can also be used to set a global │ │ │ │ │ + * symbolizer default in OpenLayers. To be SLD 1.x compliant, add the │ │ │ │ │ + * following code before rendering any vector features: │ │ │ │ │ + * (code) │ │ │ │ │ + * OpenLayers.Renderer.defaultSymbolizer = { │ │ │ │ │ + * fillColor: "#808080", │ │ │ │ │ + * fillOpacity: 1, │ │ │ │ │ + * strokeColor: "#000000", │ │ │ │ │ + * strokeOpacity: 1, │ │ │ │ │ + * strokeWidth: 1, │ │ │ │ │ + * pointRadius: 3, │ │ │ │ │ + * graphicName: "square" │ │ │ │ │ + * }; │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.defaultSymbolizer = { │ │ │ │ │ + fillColor: "#000000", │ │ │ │ │ + strokeColor: "#000000", │ │ │ │ │ + strokeWidth: 2, │ │ │ │ │ + fillOpacity: 1, │ │ │ │ │ + strokeOpacity: 1, │ │ │ │ │ + pointRadius: 0, │ │ │ │ │ + labelAlign: 'cm' │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: numZoomLevels │ │ │ │ │ - * {Integer} │ │ │ │ │ - */ │ │ │ │ │ - numZoomLevels: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: minScale │ │ │ │ │ - * {Float} │ │ │ │ │ - */ │ │ │ │ │ - minScale: null, │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.symbol │ │ │ │ │ + * Coordinate arrays for well known (named) symbols. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.symbol = { │ │ │ │ │ + "star": [350, 75, 379, 161, 469, 161, 397, 215, 423, 301, 350, 250, 277, 301, │ │ │ │ │ + 303, 215, 231, 161, 321, 161, 350, 75 │ │ │ │ │ + ], │ │ │ │ │ + "cross": [4, 0, 6, 0, 6, 4, 10, 4, 10, 6, 6, 6, 6, 10, 4, 10, 4, 6, 0, 6, 0, 4, 4, 4, │ │ │ │ │ + 4, 0 │ │ │ │ │ + ], │ │ │ │ │ + "x": [0, 0, 25, 0, 50, 35, 75, 0, 100, 0, 65, 50, 100, 100, 75, 100, 50, 65, 25, 100, 0, 100, 35, 50, 0, 0], │ │ │ │ │ + "square": [0, 0, 0, 1, 1, 1, 1, 0, 0, 0], │ │ │ │ │ + "triangle": [0, 10, 10, 10, 5, 0, 0, 10] │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Renderer/Elements.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Renderer.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.ElementsIndexer │ │ │ │ │ + * This class takes care of figuring out which order elements should be │ │ │ │ │ + * placed in the DOM based on given indexing methods. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.ElementsIndexer = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: maxScale │ │ │ │ │ - * {Float} │ │ │ │ │ + * Property: maxZIndex │ │ │ │ │ + * {Integer} This is the largest-most z-index value for a node │ │ │ │ │ + * contained within the indexer. │ │ │ │ │ */ │ │ │ │ │ - maxScale: null, │ │ │ │ │ + maxZIndex: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: displayOutsideMaxExtent │ │ │ │ │ - * {Boolean} Request map tiles that are completely outside of the max │ │ │ │ │ - * extent for this layer. Defaults to false. │ │ │ │ │ + * Property: order │ │ │ │ │ + * {Array<String>} This is an array of node id's stored in the │ │ │ │ │ + * order that they should show up on screen. Id's higher up in the │ │ │ │ │ + * array (higher array index) represent nodes with higher z-indeces. │ │ │ │ │ */ │ │ │ │ │ - displayOutsideMaxExtent: false, │ │ │ │ │ + order: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: wrapDateLine │ │ │ │ │ - * {Boolean} Wraps the world at the international dateline, so the map can │ │ │ │ │ - * be panned infinitely in longitudinal direction. Only use this on the │ │ │ │ │ - * base layer, and only if the layer's maxExtent equals the world bounds. │ │ │ │ │ - * #487 for more info. │ │ │ │ │ + * Property: indices │ │ │ │ │ + * {Object} This is a hash that maps node ids to their z-index value │ │ │ │ │ + * stored in the indexer. This is done to make finding a nodes z-index │ │ │ │ │ + * value O(1). │ │ │ │ │ */ │ │ │ │ │ - wrapDateLine: false, │ │ │ │ │ + indices: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: metadata │ │ │ │ │ - * {Object} This object can be used to store additional information on a │ │ │ │ │ - * layer object. │ │ │ │ │ + * Property: compare │ │ │ │ │ + * {Function} This is the function used to determine placement of │ │ │ │ │ + * of a new node within the indexer. If null, this defaults to to │ │ │ │ │ + * the Z_ORDER_DRAWING_ORDER comparison method. │ │ │ │ │ */ │ │ │ │ │ - metadata: null, │ │ │ │ │ + compare: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: initialize │ │ │ │ │ + * Create a new indexer with │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} The layer name │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + * yOrdering - {Boolean} Whether to use y-ordering. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ + initialize: function(yOrdering) { │ │ │ │ │ │ │ │ │ │ - this.metadata = {}; │ │ │ │ │ + this.compare = yOrdering ? │ │ │ │ │ + OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_Y_ORDER : │ │ │ │ │ + OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_DRAWING_ORDER; │ │ │ │ │ │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - // make sure we respect alwaysInRange if set on the prototype │ │ │ │ │ - if (this.alwaysInRange != null) { │ │ │ │ │ - options.alwaysInRange = this.alwaysInRange; │ │ │ │ │ + this.clear(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: insert │ │ │ │ │ + * Insert a new node into the indexer. In order to find the correct │ │ │ │ │ + * positioning for the node to be inserted, this method uses a binary │ │ │ │ │ + * search. This makes inserting O(log(n)). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newNode - {DOMElement} The new node to be inserted. │ │ │ │ │ + * │ │ │ │ │ + * Returns │ │ │ │ │ + * {DOMElement} the node before which we should insert our newNode, or │ │ │ │ │ + * null if newNode can just be appended. │ │ │ │ │ + */ │ │ │ │ │ + insert: function(newNode) { │ │ │ │ │ + // If the node is known to the indexer, remove it so we can │ │ │ │ │ + // recalculate where it should go. │ │ │ │ │ + if (this.exists(newNode)) { │ │ │ │ │ + this.remove(newNode); │ │ │ │ │ } │ │ │ │ │ - this.addOptions(options); │ │ │ │ │ │ │ │ │ │ - this.name = name; │ │ │ │ │ + var nodeId = newNode.id; │ │ │ │ │ │ │ │ │ │ - if (this.id == null) { │ │ │ │ │ + this.determineZIndex(newNode); │ │ │ │ │ │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ + var leftIndex = -1; │ │ │ │ │ + var rightIndex = this.order.length; │ │ │ │ │ + var middle; │ │ │ │ │ │ │ │ │ │ - this.div = OpenLayers.Util.createDiv(this.id); │ │ │ │ │ - this.div.style.width = "100%"; │ │ │ │ │ - this.div.style.height = "100%"; │ │ │ │ │ - this.div.dir = "ltr"; │ │ │ │ │ + while (rightIndex - leftIndex > 1) { │ │ │ │ │ + middle = parseInt((leftIndex + rightIndex) / 2); │ │ │ │ │ │ │ │ │ │ - this.events = new OpenLayers.Events(this, this.div); │ │ │ │ │ - if (this.eventListeners instanceof Object) { │ │ │ │ │ - this.events.on(this.eventListeners); │ │ │ │ │ - } │ │ │ │ │ + var placement = this.compare(this, newNode, │ │ │ │ │ + OpenLayers.Util.getElement(this.order[middle])); │ │ │ │ │ │ │ │ │ │ + if (placement > 0) { │ │ │ │ │ + leftIndex = middle; │ │ │ │ │ + } else { │ │ │ │ │ + rightIndex = middle; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + this.order.splice(rightIndex, 0, nodeId); │ │ │ │ │ + this.indices[nodeId] = this.getZIndex(newNode); │ │ │ │ │ + │ │ │ │ │ + // If the new node should be before another in the index │ │ │ │ │ + // order, return the node before which we have to insert the new one; │ │ │ │ │ + // else, return null to indicate that the new node can be appended. │ │ │ │ │ + return this.getNextElement(rightIndex); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Destroy is a destructor: this is to alleviate cyclic references which │ │ │ │ │ - * the Javascript garbage cleaner can not take care of on its own. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: remove │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * setNewBaseLayer - {Boolean} Set a new base layer when this layer has │ │ │ │ │ - * been destroyed. Default is true. │ │ │ │ │ + * node - {DOMElement} The node to be removed. │ │ │ │ │ */ │ │ │ │ │ - destroy: function(setNewBaseLayer) { │ │ │ │ │ - if (setNewBaseLayer == null) { │ │ │ │ │ - setNewBaseLayer = true; │ │ │ │ │ - } │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.removeLayer(this, setNewBaseLayer); │ │ │ │ │ - } │ │ │ │ │ - this.projection = null; │ │ │ │ │ - this.map = null; │ │ │ │ │ - this.name = null; │ │ │ │ │ - this.div = null; │ │ │ │ │ - this.options = null; │ │ │ │ │ + remove: function(node) { │ │ │ │ │ + var nodeId = node.id; │ │ │ │ │ + var arrayIndex = OpenLayers.Util.indexOf(this.order, nodeId); │ │ │ │ │ + if (arrayIndex >= 0) { │ │ │ │ │ + // Remove it from the order array, as well as deleting the node │ │ │ │ │ + // from the indeces hash. │ │ │ │ │ + this.order.splice(arrayIndex, 1); │ │ │ │ │ + delete this.indices[nodeId]; │ │ │ │ │ │ │ │ │ │ - if (this.events) { │ │ │ │ │ - if (this.eventListeners) { │ │ │ │ │ - this.events.un(this.eventListeners); │ │ │ │ │ + // Reset the maxium z-index based on the last item in the │ │ │ │ │ + // order array. │ │ │ │ │ + if (this.order.length > 0) { │ │ │ │ │ + var lastId = this.order[this.order.length - 1]; │ │ │ │ │ + this.maxZIndex = this.indices[lastId]; │ │ │ │ │ + } else { │ │ │ │ │ + this.maxZIndex = 0; │ │ │ │ │ } │ │ │ │ │ - this.events.destroy(); │ │ │ │ │ } │ │ │ │ │ - this.eventListeners = null; │ │ │ │ │ - this.events = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clone │ │ │ │ │ + * APIMethod: clear │ │ │ │ │ + */ │ │ │ │ │ + clear: function() { │ │ │ │ │ + this.order = []; │ │ │ │ │ + this.indices = {}; │ │ │ │ │ + this.maxZIndex = 0; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: exists │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {<OpenLayers.Layer>} The layer to be cloned │ │ │ │ │ + * node - {DOMElement} The node to test for existence. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Layer>} An exact clone of this <OpenLayers.Layer> │ │ │ │ │ + * {Boolean} Whether or not the node exists in the indexer? │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer(this.name, this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // catch any randomly tagged-on properties │ │ │ │ │ - OpenLayers.Util.applyDefaults(obj, this); │ │ │ │ │ - │ │ │ │ │ - // a cloned layer should never have its map property set │ │ │ │ │ - // because it has not been added to a map yet. │ │ │ │ │ - obj.map = null; │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ + exists: function(node) { │ │ │ │ │ + return (this.indices[node.id] != null); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getOptions │ │ │ │ │ - * Extracts an object from the layer with the properties that were set as │ │ │ │ │ - * options, but updates them with the values currently set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * APIMethod: getZIndex │ │ │ │ │ + * Get the z-index value for the current node from the node data itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The node whose z-index to get. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} the <options> of the layer, representing the current state. │ │ │ │ │ + * {Integer} The z-index value for the specified node (from the node │ │ │ │ │ + * data itself). │ │ │ │ │ */ │ │ │ │ │ - getOptions: function() { │ │ │ │ │ - var options = {}; │ │ │ │ │ - for (var o in this.options) { │ │ │ │ │ - options[o] = this[o]; │ │ │ │ │ - } │ │ │ │ │ - return options; │ │ │ │ │ + getZIndex: function(node) { │ │ │ │ │ + return node._style.graphicZIndex; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setName │ │ │ │ │ - * Sets the new layer name for this layer. Can trigger a changelayer event │ │ │ │ │ - * on the map. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: determineZIndex │ │ │ │ │ + * Determine the z-index for the current node if there isn't one, │ │ │ │ │ + * and set the maximum value if we've found a new maximum. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * newName - {String} The new name. │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - setName: function(newName) { │ │ │ │ │ - if (newName != this.name) { │ │ │ │ │ - this.name = newName; │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "name" │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ + determineZIndex: function(node) { │ │ │ │ │ + var zIndex = node._style.graphicZIndex; │ │ │ │ │ + │ │ │ │ │ + // Everything must have a zIndex. If none is specified, │ │ │ │ │ + // this means the user *must* (hint: assumption) want this │ │ │ │ │ + // node to succomb to drawing order. To enforce drawing order │ │ │ │ │ + // over all indexing methods, we'll create a new z-index that's │ │ │ │ │ + // greater than any currently in the indexer. │ │ │ │ │ + if (zIndex == null) { │ │ │ │ │ + zIndex = this.maxZIndex; │ │ │ │ │ + node._style.graphicZIndex = zIndex; │ │ │ │ │ + } else if (zIndex > this.maxZIndex) { │ │ │ │ │ + this.maxZIndex = zIndex; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: addOptions │ │ │ │ │ + * APIMethod: getNextElement │ │ │ │ │ + * Get the next element in the order stack. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * newOptions - {Object} │ │ │ │ │ - * reinitialize - {Boolean} If set to true, and if resolution options of the │ │ │ │ │ - * current baseLayer were changed, the map will be recentered to make │ │ │ │ │ - * sure that it is displayed with a valid resolution, and a │ │ │ │ │ - * changebaselayer event will be triggered. │ │ │ │ │ + * index - {Integer} The index of the current node in this.order. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} the node following the index passed in, or │ │ │ │ │ + * null. │ │ │ │ │ */ │ │ │ │ │ - addOptions: function(newOptions, reinitialize) { │ │ │ │ │ - │ │ │ │ │ - if (this.options == null) { │ │ │ │ │ - this.options = {}; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (newOptions) { │ │ │ │ │ - // make sure this.projection references a projection object │ │ │ │ │ - if (typeof newOptions.projection == "string") { │ │ │ │ │ - newOptions.projection = new OpenLayers.Projection(newOptions.projection); │ │ │ │ │ - } │ │ │ │ │ - if (newOptions.projection) { │ │ │ │ │ - // get maxResolution, units and maxExtent from projection defaults if │ │ │ │ │ - // they are not defined already │ │ │ │ │ - OpenLayers.Util.applyDefaults(newOptions, │ │ │ │ │ - OpenLayers.Projection.defaults[newOptions.projection.getCode()]); │ │ │ │ │ - } │ │ │ │ │ - // allow array for extents │ │ │ │ │ - if (newOptions.maxExtent && !(newOptions.maxExtent instanceof OpenLayers.Bounds)) { │ │ │ │ │ - newOptions.maxExtent = new OpenLayers.Bounds(newOptions.maxExtent); │ │ │ │ │ - } │ │ │ │ │ - if (newOptions.minExtent && !(newOptions.minExtent instanceof OpenLayers.Bounds)) { │ │ │ │ │ - newOptions.minExtent = new OpenLayers.Bounds(newOptions.minExtent); │ │ │ │ │ + getNextElement: function(index) { │ │ │ │ │ + var nextIndex = index + 1; │ │ │ │ │ + if (nextIndex < this.order.length) { │ │ │ │ │ + var nextElement = OpenLayers.Util.getElement(this.order[nextIndex]); │ │ │ │ │ + if (nextElement == undefined) { │ │ │ │ │ + nextElement = this.getNextElement(nextIndex); │ │ │ │ │ } │ │ │ │ │ + return nextElement; │ │ │ │ │ + } else { │ │ │ │ │ + return null; │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // update our copy for clone │ │ │ │ │ - OpenLayers.Util.extend(this.options, newOptions); │ │ │ │ │ - │ │ │ │ │ - // add new options to this │ │ │ │ │ - OpenLayers.Util.extend(this, newOptions); │ │ │ │ │ + CLASS_NAME: "OpenLayers.ElementsIndexer" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ - // get the units from the projection, if we have a projection │ │ │ │ │ - // and it it has units │ │ │ │ │ - if (this.projection && this.projection.getUnits()) { │ │ │ │ │ - this.units = this.projection.getUnits(); │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.ElementsIndexer.IndexingMethods │ │ │ │ │ + * These are the compare methods for figuring out where a new node should be │ │ │ │ │ + * placed within the indexer. These methods are very similar to general │ │ │ │ │ + * sorting methods in that they return -1, 0, and 1 to specify the │ │ │ │ │ + * direction in which new nodes fall in the ordering. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.ElementsIndexer.IndexingMethods = { │ │ │ │ │ │ │ │ │ │ - // re-initialize resolutions if necessary, i.e. if any of the │ │ │ │ │ - // properties of the "properties" array defined below is set │ │ │ │ │ - // in the new options │ │ │ │ │ - if (this.map) { │ │ │ │ │ - // store current resolution so we can try to restore it later │ │ │ │ │ - var resolution = this.map.getResolution(); │ │ │ │ │ - var properties = this.RESOLUTION_PROPERTIES.concat( │ │ │ │ │ - ["projection", "units", "minExtent", "maxExtent"] │ │ │ │ │ - ); │ │ │ │ │ - for (var o in newOptions) { │ │ │ │ │ - if (newOptions.hasOwnProperty(o) && │ │ │ │ │ - OpenLayers.Util.indexOf(properties, o) >= 0) { │ │ │ │ │ + /** │ │ │ │ │ + * Method: Z_ORDER │ │ │ │ │ + * This compare method is used by other comparison methods. │ │ │ │ │ + * It can be used individually for ordering, but is not recommended, │ │ │ │ │ + * because it doesn't subscribe to drawing order. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ + * newNode - {DOMElement} │ │ │ │ │ + * nextNode - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} │ │ │ │ │ + */ │ │ │ │ │ + Z_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var newZIndex = indexer.getZIndex(newNode); │ │ │ │ │ │ │ │ │ │ - this.initResolutions(); │ │ │ │ │ - if (reinitialize && this.map.baseLayer === this) { │ │ │ │ │ - // update map position, and restore previous resolution │ │ │ │ │ - this.map.setCenter(this.map.getCenter(), │ │ │ │ │ - this.map.getZoomForResolution(resolution), │ │ │ │ │ - false, true │ │ │ │ │ - ); │ │ │ │ │ - // trigger a changebaselayer event to make sure that │ │ │ │ │ - // all controls (especially │ │ │ │ │ - // OpenLayers.Control.PanZoomBar) get notified of the │ │ │ │ │ - // new options │ │ │ │ │ - this.map.events.triggerEvent("changebaselayer", { │ │ │ │ │ - layer: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + var returnVal = 0; │ │ │ │ │ + if (nextNode) { │ │ │ │ │ + var nextZIndex = indexer.getZIndex(nextNode); │ │ │ │ │ + returnVal = newZIndex - nextZIndex; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: onMapResize │ │ │ │ │ - * This function can be implemented by subclasses │ │ │ │ │ - */ │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - //this function can be implemented by subclasses │ │ │ │ │ + return returnVal; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: redraw │ │ │ │ │ - * Redraws the layer. Returns true if the layer was redrawn, false if not. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: Z_ORDER_DRAWING_ORDER │ │ │ │ │ + * This method orders nodes by their z-index, but does so in a way │ │ │ │ │ + * that, if there are other nodes with the same z-index, the newest │ │ │ │ │ + * drawn will be the front most within that z-index. This is the │ │ │ │ │ + * default indexing method. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ + * newNode - {DOMElement} │ │ │ │ │ + * nextNode - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The layer was redrawn. │ │ │ │ │ + * {Integer} │ │ │ │ │ */ │ │ │ │ │ - redraw: function() { │ │ │ │ │ - var redrawn = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - │ │ │ │ │ - // min/max Range may have changed │ │ │ │ │ - this.inRange = this.calculateInRange(); │ │ │ │ │ - │ │ │ │ │ - // map's center might not yet be set │ │ │ │ │ - var extent = this.getExtent(); │ │ │ │ │ + Z_ORDER_DRAWING_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER( │ │ │ │ │ + indexer, │ │ │ │ │ + newNode, │ │ │ │ │ + nextNode │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - if (extent && this.inRange && this.visibility) { │ │ │ │ │ - var zoomChanged = true; │ │ │ │ │ - this.moveTo(extent, zoomChanged, false); │ │ │ │ │ - this.events.triggerEvent("moveend", { │ │ │ │ │ - "zoomChanged": zoomChanged │ │ │ │ │ - }); │ │ │ │ │ - redrawn = true; │ │ │ │ │ - } │ │ │ │ │ + // Make Z_ORDER subscribe to drawing order by pushing it above │ │ │ │ │ + // all of the other nodes with the same z-index. │ │ │ │ │ + if (nextNode && returnVal == 0) { │ │ │ │ │ + returnVal = 1; │ │ │ │ │ } │ │ │ │ │ - return redrawn; │ │ │ │ │ + │ │ │ │ │ + return returnVal; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ + * APIMethod: Z_ORDER_Y_ORDER │ │ │ │ │ + * This one should really be called Z_ORDER_Y_ORDER_DRAWING_ORDER, as it │ │ │ │ │ + * best describes which ordering methods have precedence (though, the │ │ │ │ │ + * name would be too long). This method orders nodes by their z-index, │ │ │ │ │ + * but does so in a way that, if there are other nodes with the same │ │ │ │ │ + * z-index, the nodes with the lower y position will be "closer" than │ │ │ │ │ + * those with a higher y position. If two nodes have the exact same y │ │ │ │ │ + * position, however, then this method will revert to using drawing │ │ │ │ │ + * order to decide placement. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * zoomChanged - {Boolean} Tells when zoom has changed, as layers have to │ │ │ │ │ - * do some init work in that case. │ │ │ │ │ - * dragging - {Boolean} │ │ │ │ │ + * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ + * newNode - {DOMElement} │ │ │ │ │ + * nextNode - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} │ │ │ │ │ */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - var display = this.visibility; │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - display = display && this.inRange; │ │ │ │ │ + Z_ORDER_Y_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER( │ │ │ │ │ + indexer, │ │ │ │ │ + newNode, │ │ │ │ │ + nextNode │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + if (nextNode && returnVal === 0) { │ │ │ │ │ + var result = nextNode._boundsBottom - newNode._boundsBottom; │ │ │ │ │ + returnVal = (result === 0) ? 1 : result; │ │ │ │ │ } │ │ │ │ │ - this.display(display); │ │ │ │ │ - }, │ │ │ │ │ + │ │ │ │ │ + return returnVal; │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Renderer.Elements │ │ │ │ │ + * This is another virtual class in that it should never be instantiated by │ │ │ │ │ + * itself as a Renderer. It exists because there is *tons* of shared │ │ │ │ │ + * functionality between different vector libraries which use nodes/elements │ │ │ │ │ + * as a base for rendering vectors. │ │ │ │ │ + * │ │ │ │ │ + * The highlevel bits of code that are implemented here are the adding and │ │ │ │ │ + * removing of geometries, which is essentially the same for any │ │ │ │ │ + * element-based renderer. The details of creating each node and drawing the │ │ │ │ │ + * paths are of course different, but the machinery is the same. │ │ │ │ │ + * │ │ │ │ │ + * Inherits: │ │ │ │ │ + * - <OpenLayers.Renderer> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.Elements = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveByPx │ │ │ │ │ - * Move the layer based on pixel vector. To be implemented by subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * dx - {Number} The x coord of the displacement vector. │ │ │ │ │ - * dy - {Number} The y coord of the displacement vector. │ │ │ │ │ + * Property: rendererRoot │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - moveByPx: function(dx, dy) {}, │ │ │ │ │ + rendererRoot: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the layer. This is done through an accessor │ │ │ │ │ - * so that subclasses can override this and take special action once │ │ │ │ │ - * they have their map variable set. │ │ │ │ │ - * │ │ │ │ │ - * Here we take care to bring over any of the necessary default │ │ │ │ │ - * properties from the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * Property: root │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - if (this.map == null) { │ │ │ │ │ - │ │ │ │ │ - this.map = map; │ │ │ │ │ + root: null, │ │ │ │ │ │ │ │ │ │ - // grab some essential layer data from the map if it hasn't already │ │ │ │ │ - // been set │ │ │ │ │ - this.maxExtent = this.maxExtent || this.map.maxExtent; │ │ │ │ │ - this.minExtent = this.minExtent || this.map.minExtent; │ │ │ │ │ + /** │ │ │ │ │ + * Property: vectorRoot │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + vectorRoot: null, │ │ │ │ │ │ │ │ │ │ - this.projection = this.projection || this.map.projection; │ │ │ │ │ - if (typeof this.projection == "string") { │ │ │ │ │ - this.projection = new OpenLayers.Projection(this.projection); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: textRoot │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + textRoot: null, │ │ │ │ │ │ │ │ │ │ - // Check the projection to see if we can get units -- if not, refer │ │ │ │ │ - // to properties. │ │ │ │ │ - this.units = this.projection.getUnits() || │ │ │ │ │ - this.units || this.map.units; │ │ │ │ │ + /** │ │ │ │ │ + * Property: xmlns │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + xmlns: null, │ │ │ │ │ │ │ │ │ │ - this.initResolutions(); │ │ │ │ │ + /** │ │ │ │ │ + * Property: xOffset │ │ │ │ │ + * {Number} Offset to apply to the renderer viewport translation in x │ │ │ │ │ + * direction. If the renderer extent's center is on the right of the │ │ │ │ │ + * dateline (i.e. exceeds the world bounds), we shift the viewport to the │ │ │ │ │ + * left by one world width. This avoids that features disappear from the │ │ │ │ │ + * map viewport. Because our dateline handling logic in other places │ │ │ │ │ + * ensures that extents crossing the dateline always have a center │ │ │ │ │ + * exceeding the world bounds on the left, we need this offset to make sure │ │ │ │ │ + * that the same is true for the renderer extent in pixel space as well. │ │ │ │ │ + */ │ │ │ │ │ + xOffset: 0, │ │ │ │ │ │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - this.inRange = this.calculateInRange(); │ │ │ │ │ - var show = ((this.visibility) && (this.inRange)); │ │ │ │ │ - this.div.style.display = show ? "" : "none"; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: rightOfDateLine │ │ │ │ │ + * {Boolean} Keeps track of the location of the map extent relative to the │ │ │ │ │ + * date line. The <setExtent> method compares this value (which is the one │ │ │ │ │ + * from the previous <setExtent> call) with the current position of the map │ │ │ │ │ + * extent relative to the date line and updates the xOffset when the extent │ │ │ │ │ + * has moved from one side of the date line to the other. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // deal with gutters │ │ │ │ │ - this.setTileSize(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: Indexer │ │ │ │ │ + * {<OpenLayers.ElementIndexer>} An instance of OpenLayers.ElementsIndexer │ │ │ │ │ + * created upon initialization if the zIndexing or yOrdering options │ │ │ │ │ + * passed to this renderer's constructor are set to true. │ │ │ │ │ + */ │ │ │ │ │ + indexer: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: afterAdd │ │ │ │ │ - * Called at the end of the map.addLayer sequence. At this point, the map │ │ │ │ │ - * will have a base layer. To be overridden by subclasses. │ │ │ │ │ + * Constant: BACKGROUND_ID_SUFFIX │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - afterAdd: function() {}, │ │ │ │ │ + BACKGROUND_ID_SUFFIX: "_background", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: removeMap │ │ │ │ │ - * Just as setMap() allows each layer the possibility to take a │ │ │ │ │ - * personalized action on being added to the map, removeMap() allows │ │ │ │ │ - * each layer to take a personalized action on being removed from it. │ │ │ │ │ - * For now, this will be mostly unused, except for the EventPane layer, │ │ │ │ │ - * which needs this hook so that it can remove the special invisible │ │ │ │ │ - * pane. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * Constant: LABEL_ID_SUFFIX │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - //to be overridden by subclasses │ │ │ │ │ - }, │ │ │ │ │ + LABEL_ID_SUFFIX: "_label", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getImageSize │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} optional tile bounds, can be used │ │ │ │ │ - * by subclasses that have to deal with different tile sizes at the │ │ │ │ │ - * layer extent edges (e.g. Zoomify) │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Size>} The size that the image should be, taking into │ │ │ │ │ - * account gutters. │ │ │ │ │ + * Constant: LABEL_OUTLINE_SUFFIX │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - getImageSize: function(bounds) { │ │ │ │ │ - return (this.imageSize || this.tileSize); │ │ │ │ │ - }, │ │ │ │ │ + LABEL_OUTLINE_SUFFIX: "_outline", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setTileSize │ │ │ │ │ - * Set the tile size based on the map size. This also sets layer.imageSize │ │ │ │ │ - * or use by Tile.Image. │ │ │ │ │ + * Constructor: OpenLayers.Renderer.Elements │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>} │ │ │ │ │ + * containerID - {String} │ │ │ │ │ + * options - {Object} options for this renderer. │ │ │ │ │ + * │ │ │ │ │ + * Supported options are: │ │ │ │ │ + * yOrdering - {Boolean} Whether to use y-ordering │ │ │ │ │ + * zIndexing - {Boolean} Whether to use z-indexing. Will be ignored │ │ │ │ │ + * if yOrdering is set to true. │ │ │ │ │ */ │ │ │ │ │ - setTileSize: function(size) { │ │ │ │ │ - var tileSize = (size) ? size : │ │ │ │ │ - ((this.tileSize) ? this.tileSize : │ │ │ │ │ - this.map.getTileSize()); │ │ │ │ │ - this.tileSize = tileSize; │ │ │ │ │ - if (this.gutter) { │ │ │ │ │ - // layers with gutters need non-null tile sizes │ │ │ │ │ - //if(tileSize == null) { │ │ │ │ │ - // OpenLayers.console.error("Error in layer.setMap() for " + │ │ │ │ │ - // this.name + ": layers with " + │ │ │ │ │ - // "gutters need non-null tile sizes"); │ │ │ │ │ - //} │ │ │ │ │ - this.imageSize = new OpenLayers.Size(tileSize.w + (2 * this.gutter), │ │ │ │ │ - tileSize.h + (2 * this.gutter)); │ │ │ │ │ + initialize: function(containerID, options) { │ │ │ │ │ + OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + this.rendererRoot = this.createRenderRoot(); │ │ │ │ │ + this.root = this.createRoot("_root"); │ │ │ │ │ + this.vectorRoot = this.createRoot("_vroot"); │ │ │ │ │ + this.textRoot = this.createRoot("_troot"); │ │ │ │ │ + │ │ │ │ │ + this.root.appendChild(this.vectorRoot); │ │ │ │ │ + this.root.appendChild(this.textRoot); │ │ │ │ │ + │ │ │ │ │ + this.rendererRoot.appendChild(this.root); │ │ │ │ │ + this.container.appendChild(this.rendererRoot); │ │ │ │ │ + │ │ │ │ │ + if (options && (options.zIndexing || options.yOrdering)) { │ │ │ │ │ + this.indexer = new OpenLayers.ElementsIndexer(options.yOrdering); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getVisibility │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The layer should be displayed (if in range). │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ - getVisibility: function() { │ │ │ │ │ - return this.visibility; │ │ │ │ │ + destroy: function() { │ │ │ │ │ + │ │ │ │ │ + this.clear(); │ │ │ │ │ + │ │ │ │ │ + this.rendererRoot = null; │ │ │ │ │ + this.root = null; │ │ │ │ │ + this.xmlns = null; │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Renderer.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setVisibility │ │ │ │ │ - * Set the visibility flag for the layer and hide/show & redraw │ │ │ │ │ - * accordingly. Fire event unless otherwise specified │ │ │ │ │ - * │ │ │ │ │ - * Note that visibility is no longer simply whether or not the layer's │ │ │ │ │ - * style.display is set to "block". Now we store a 'visibility' state │ │ │ │ │ - * property on the layer class, this allows us to remember whether or │ │ │ │ │ - * not we *desire* for a layer to be visible. In the case where the │ │ │ │ │ - * map's resolution is out of the layer's range, this desire may be │ │ │ │ │ - * subverted. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * visibility - {Boolean} Whether or not to display the layer (if in range) │ │ │ │ │ + /** │ │ │ │ │ + * Method: clear │ │ │ │ │ + * Remove all the elements from the root │ │ │ │ │ */ │ │ │ │ │ - setVisibility: function(visibility) { │ │ │ │ │ - if (visibility != this.visibility) { │ │ │ │ │ - this.visibility = visibility; │ │ │ │ │ - this.display(visibility); │ │ │ │ │ - this.redraw(); │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "visibility" │ │ │ │ │ - }); │ │ │ │ │ + clear: function() { │ │ │ │ │ + var child; │ │ │ │ │ + var root = this.vectorRoot; │ │ │ │ │ + if (root) { │ │ │ │ │ + while (child = root.firstChild) { │ │ │ │ │ + root.removeChild(child); │ │ │ │ │ } │ │ │ │ │ - this.events.triggerEvent("visibilitychanged"); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: display │ │ │ │ │ - * Hide or show the Layer. This is designed to be used internally, and │ │ │ │ │ - * is not generally the way to enable or disable the layer. For that, │ │ │ │ │ - * use the setVisibility function instead.. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * display - {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - display: function(display) { │ │ │ │ │ - if (display != (this.div.style.display != "none")) { │ │ │ │ │ - this.div.style.display = (display && this.calculateInRange()) ? "block" : "none"; │ │ │ │ │ + root = this.textRoot; │ │ │ │ │ + if (root) { │ │ │ │ │ + while (child = root.firstChild) { │ │ │ │ │ + root.removeChild(child); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + this.indexer.clear(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: calculateInRange │ │ │ │ │ - * │ │ │ │ │ + * Method: setExtent │ │ │ │ │ + * Set the visible part of the layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} │ │ │ │ │ + * resolutionChanged - {Boolean} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The layer is displayable at the current map's current │ │ │ │ │ - * resolution. Note that if 'alwaysInRange' is true for the layer, │ │ │ │ │ - * this function will always return true. │ │ │ │ │ + * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ + * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ + * False otherwise. │ │ │ │ │ */ │ │ │ │ │ - calculateInRange: function() { │ │ │ │ │ - var inRange = false; │ │ │ │ │ - │ │ │ │ │ - if (this.alwaysInRange) { │ │ │ │ │ - inRange = true; │ │ │ │ │ - } else { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - var resolution = this.map.getResolution(); │ │ │ │ │ - inRange = ((resolution >= this.minResolution) && │ │ │ │ │ - (resolution <= this.maxResolution)); │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + var rightOfDateLine, │ │ │ │ │ + ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ + extent = extent.scale(1 / ratio), │ │ │ │ │ + world = this.map.getMaxExtent(); │ │ │ │ │ + if (world.right > extent.left && world.right < extent.right) { │ │ │ │ │ + rightOfDateLine = true; │ │ │ │ │ + } else if (world.left > extent.left && world.left < extent.right) { │ │ │ │ │ + rightOfDateLine = false; │ │ │ │ │ + } │ │ │ │ │ + if (rightOfDateLine !== this.rightOfDateLine || resolutionChanged) { │ │ │ │ │ + coordSysUnchanged = false; │ │ │ │ │ + this.xOffset = rightOfDateLine === true ? │ │ │ │ │ + world.getWidth() / resolution : 0; │ │ │ │ │ } │ │ │ │ │ + this.rightOfDateLine = rightOfDateLine; │ │ │ │ │ } │ │ │ │ │ - return inRange; │ │ │ │ │ + return coordSysUnchanged; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setIsBaseLayer │ │ │ │ │ - * │ │ │ │ │ + * Method: getNodeType │ │ │ │ │ + * This function is in charge of asking the specific renderer which type │ │ │ │ │ + * of node to create for the given geometry and style. All geometries │ │ │ │ │ + * in an Elements-based renderer consist of one node and some │ │ │ │ │ + * attributes. We have the nodeFactory() function which creates a node │ │ │ │ │ + * for us, but it takes a 'type' as input, and that is precisely what │ │ │ │ │ + * this function tells us. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * isBaseLayer - {Boolean} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The corresponding node type for the specified geometry │ │ │ │ │ */ │ │ │ │ │ - setIsBaseLayer: function(isBaseLayer) { │ │ │ │ │ - if (isBaseLayer != this.isBaseLayer) { │ │ │ │ │ - this.isBaseLayer = isBaseLayer; │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changebaselayer", { │ │ │ │ │ - layer: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /********************************************************/ │ │ │ │ │ - /* */ │ │ │ │ │ - /* Baselayer Functions */ │ │ │ │ │ - /* */ │ │ │ │ │ - /********************************************************/ │ │ │ │ │ + getNodeType: function(geometry, style) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: initResolutions │ │ │ │ │ - * This method's responsibility is to set up the 'resolutions' array │ │ │ │ │ - * for the layer -- this array is what the layer will use to interface │ │ │ │ │ - * between the zoom levels of the map and the resolution display │ │ │ │ │ - * of the layer. │ │ │ │ │ + * Method: drawGeometry │ │ │ │ │ + * Draw the geometry, creating new nodes, setting paths, setting style, │ │ │ │ │ + * setting featureId on the node. This method should only be called │ │ │ │ │ + * by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ * │ │ │ │ │ - * The user has several options that determine how the array is set up. │ │ │ │ │ - * │ │ │ │ │ - * For a detailed explanation, see the following wiki from the │ │ │ │ │ - * openlayers.org homepage: │ │ │ │ │ - * http://trac.openlayers.org/wiki/SettingZoomLevels │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the geometry has been drawn completely; null if │ │ │ │ │ + * incomplete; false otherwise │ │ │ │ │ */ │ │ │ │ │ - initResolutions: function() { │ │ │ │ │ - │ │ │ │ │ - // ok we want resolutions, here's our strategy: │ │ │ │ │ - // │ │ │ │ │ - // 1. if resolutions are defined in the layer config, use them │ │ │ │ │ - // 2. else, if scales are defined in the layer config then derive │ │ │ │ │ - // resolutions from these scales │ │ │ │ │ - // 3. else, attempt to calculate resolutions from maxResolution, │ │ │ │ │ - // minResolution, numZoomLevels, maxZoomLevel set in the │ │ │ │ │ - // layer config │ │ │ │ │ - // 4. if we still don't have resolutions, and if resolutions │ │ │ │ │ - // are defined in the same, use them │ │ │ │ │ - // 5. else, if scales are defined in the map then derive │ │ │ │ │ - // resolutions from these scales │ │ │ │ │ - // 6. else, attempt to calculate resolutions from maxResolution, │ │ │ │ │ - // minResolution, numZoomLevels, maxZoomLevel set in the │ │ │ │ │ - // map │ │ │ │ │ - // 7. hope for the best! │ │ │ │ │ - │ │ │ │ │ - var i, len, p; │ │ │ │ │ - var props = {}, │ │ │ │ │ - alwaysInRange = true; │ │ │ │ │ - │ │ │ │ │ - // get resolution data from layer config │ │ │ │ │ - // (we also set alwaysInRange in the layer as appropriate) │ │ │ │ │ - for (i = 0, len = this.RESOLUTION_PROPERTIES.length; i < len; i++) { │ │ │ │ │ - p = this.RESOLUTION_PROPERTIES[i]; │ │ │ │ │ - props[p] = this.options[p]; │ │ │ │ │ - if (alwaysInRange && this.options[p]) { │ │ │ │ │ - alwaysInRange = false; │ │ │ │ │ + drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ + var className = geometry.CLASS_NAME; │ │ │ │ │ + var rendered = true; │ │ │ │ │ + if ((className == "OpenLayers.Geometry.Collection") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiPolygon")) { │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + rendered = this.drawGeometry( │ │ │ │ │ + geometry.components[i], style, featureId) && rendered; │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - if (this.options.alwaysInRange == null) { │ │ │ │ │ - this.alwaysInRange = alwaysInRange; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // if we don't have resolutions then attempt to derive them from scales │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - props.resolutions = this.resolutionsFromScales(props.scales); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // if we still don't have resolutions then attempt to calculate them │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - props.resolutions = this.calculateResolutions(props); │ │ │ │ │ + return rendered; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // if we couldn't calculate resolutions then we look at we have │ │ │ │ │ - // in the map │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - for (i = 0, len = this.RESOLUTION_PROPERTIES.length; i < len; i++) { │ │ │ │ │ - p = this.RESOLUTION_PROPERTIES[i]; │ │ │ │ │ - props[p] = this.options[p] != null ? │ │ │ │ │ - this.options[p] : this.map[p]; │ │ │ │ │ + rendered = false; │ │ │ │ │ + var removeBackground = false; │ │ │ │ │ + if (style.display != "none") { │ │ │ │ │ + if (style.backgroundGraphic) { │ │ │ │ │ + this.redrawBackgroundNode(geometry.id, geometry, style, │ │ │ │ │ + featureId); │ │ │ │ │ + } else { │ │ │ │ │ + removeBackground = true; │ │ │ │ │ } │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - props.resolutions = this.resolutionsFromScales(props.scales); │ │ │ │ │ + rendered = this.redrawNode(geometry.id, geometry, style, │ │ │ │ │ + featureId); │ │ │ │ │ + } │ │ │ │ │ + if (rendered == false) { │ │ │ │ │ + var node = document.getElementById(geometry.id); │ │ │ │ │ + if (node) { │ │ │ │ │ + if (node._style.backgroundGraphic) { │ │ │ │ │ + removeBackground = true; │ │ │ │ │ + } │ │ │ │ │ + node.parentNode.removeChild(node); │ │ │ │ │ } │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - props.resolutions = this.calculateResolutions(props); │ │ │ │ │ + } │ │ │ │ │ + if (removeBackground) { │ │ │ │ │ + var node = document.getElementById( │ │ │ │ │ + geometry.id + this.BACKGROUND_ID_SUFFIX); │ │ │ │ │ + if (node) { │ │ │ │ │ + node.parentNode.removeChild(node); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return rendered; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // ok, we new need to set properties in the instance │ │ │ │ │ + /** │ │ │ │ │ + * Method: redrawNode │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the complete geometry could be drawn, null if parts of │ │ │ │ │ + * the geometry could not be drawn, false otherwise │ │ │ │ │ + */ │ │ │ │ │ + redrawNode: function(id, geometry, style, featureId) { │ │ │ │ │ + style = this.applyDefaultSymbolizer(style); │ │ │ │ │ + // Get the node if it's already on the map. │ │ │ │ │ + var node = this.nodeFactory(id, this.getNodeType(geometry, style)); │ │ │ │ │ │ │ │ │ │ - // get maxResolution from the config if it's defined there │ │ │ │ │ - var maxResolution; │ │ │ │ │ - if (this.options.maxResolution && │ │ │ │ │ - this.options.maxResolution !== "auto") { │ │ │ │ │ - maxResolution = this.options.maxResolution; │ │ │ │ │ - } │ │ │ │ │ - if (this.options.minScale) { │ │ │ │ │ - maxResolution = OpenLayers.Util.getResolutionFromScale( │ │ │ │ │ - this.options.minScale, this.units); │ │ │ │ │ - } │ │ │ │ │ + // Set the data for the node, then draw it. │ │ │ │ │ + node._featureId = featureId; │ │ │ │ │ + node._boundsBottom = geometry.getBounds().bottom; │ │ │ │ │ + node._geometryClass = geometry.CLASS_NAME; │ │ │ │ │ + node._style = style; │ │ │ │ │ │ │ │ │ │ - // get minResolution from the config if it's defined there │ │ │ │ │ - var minResolution; │ │ │ │ │ - if (this.options.minResolution && │ │ │ │ │ - this.options.minResolution !== "auto") { │ │ │ │ │ - minResolution = this.options.minResolution; │ │ │ │ │ - } │ │ │ │ │ - if (this.options.maxScale) { │ │ │ │ │ - minResolution = OpenLayers.Util.getResolutionFromScale( │ │ │ │ │ - this.options.maxScale, this.units); │ │ │ │ │ + var drawResult = this.drawGeometryNode(node, geometry, style); │ │ │ │ │ + if (drawResult === false) { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - if (props.resolutions) { │ │ │ │ │ - │ │ │ │ │ - //sort resolutions array descendingly │ │ │ │ │ - props.resolutions.sort(function(a, b) { │ │ │ │ │ - return (b - a); │ │ │ │ │ - }); │ │ │ │ │ + node = drawResult.node; │ │ │ │ │ │ │ │ │ │ - // if we still don't have a maxResolution get it from the │ │ │ │ │ - // resolutions array │ │ │ │ │ - if (!maxResolution) { │ │ │ │ │ - maxResolution = props.resolutions[0]; │ │ │ │ │ + // Insert the node into the indexer so it can show us where to │ │ │ │ │ + // place it. Note that this operation is O(log(n)). If there's a │ │ │ │ │ + // performance problem (when dragging, for instance) this is │ │ │ │ │ + // likely where it would be. │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + var insert = this.indexer.insert(node); │ │ │ │ │ + if (insert) { │ │ │ │ │ + this.vectorRoot.insertBefore(node, insert); │ │ │ │ │ + } else { │ │ │ │ │ + this.vectorRoot.appendChild(node); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // if we still don't have a minResolution get it from the │ │ │ │ │ - // resolutions array │ │ │ │ │ - if (!minResolution) { │ │ │ │ │ - var lastIdx = props.resolutions.length - 1; │ │ │ │ │ - minResolution = props.resolutions[lastIdx]; │ │ │ │ │ + } else { │ │ │ │ │ + // if there's no indexer, simply append the node to root, │ │ │ │ │ + // but only if the node is a new one │ │ │ │ │ + if (node.parentNode !== this.vectorRoot) { │ │ │ │ │ + this.vectorRoot.appendChild(node); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - this.resolutions = props.resolutions; │ │ │ │ │ - if (this.resolutions) { │ │ │ │ │ - len = this.resolutions.length; │ │ │ │ │ - this.scales = new Array(len); │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - this.scales[i] = OpenLayers.Util.getScaleFromResolution( │ │ │ │ │ - this.resolutions[i], this.units); │ │ │ │ │ - } │ │ │ │ │ - this.numZoomLevels = len; │ │ │ │ │ - } │ │ │ │ │ - this.minResolution = minResolution; │ │ │ │ │ - if (minResolution) { │ │ │ │ │ - this.maxScale = OpenLayers.Util.getScaleFromResolution( │ │ │ │ │ - minResolution, this.units); │ │ │ │ │ - } │ │ │ │ │ - this.maxResolution = maxResolution; │ │ │ │ │ - if (maxResolution) { │ │ │ │ │ - this.minScale = OpenLayers.Util.getScaleFromResolution( │ │ │ │ │ - maxResolution, this.units); │ │ │ │ │ - } │ │ │ │ │ + this.postDraw(node); │ │ │ │ │ + │ │ │ │ │ + return drawResult.complete; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: resolutionsFromScales │ │ │ │ │ - * Derive resolutions from scales. │ │ │ │ │ - * │ │ │ │ │ + * Method: redrawBackgroundNode │ │ │ │ │ + * Redraws the node using special 'background' style properties. Basically │ │ │ │ │ + * just calls redrawNode(), but instead of directly using the │ │ │ │ │ + * 'externalGraphic', 'graphicXOffset', 'graphicYOffset', and │ │ │ │ │ + * 'graphicZIndex' properties directly from the specified 'style' │ │ │ │ │ + * parameter, we create a new style object and set those properties │ │ │ │ │ + * from the corresponding 'background'-prefixed properties from │ │ │ │ │ + * specified 'style' parameter. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * scales - {Array(Number)} Scales │ │ │ │ │ - * │ │ │ │ │ - * Returns │ │ │ │ │ - * {Array(Number)} Resolutions │ │ │ │ │ + * id - {String} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the complete geometry could be drawn, null if parts of │ │ │ │ │ + * the geometry could not be drawn, false otherwise │ │ │ │ │ */ │ │ │ │ │ - resolutionsFromScales: function(scales) { │ │ │ │ │ - if (scales == null) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var resolutions, i, len; │ │ │ │ │ - len = scales.length; │ │ │ │ │ - resolutions = new Array(len); │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - resolutions[i] = OpenLayers.Util.getResolutionFromScale( │ │ │ │ │ - scales[i], this.units); │ │ │ │ │ - } │ │ │ │ │ - return resolutions; │ │ │ │ │ + redrawBackgroundNode: function(id, geometry, style, featureId) { │ │ │ │ │ + var backgroundStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ + │ │ │ │ │ + // Set regular style attributes to apply to the background styles. │ │ │ │ │ + backgroundStyle.externalGraphic = backgroundStyle.backgroundGraphic; │ │ │ │ │ + backgroundStyle.graphicXOffset = backgroundStyle.backgroundXOffset; │ │ │ │ │ + backgroundStyle.graphicYOffset = backgroundStyle.backgroundYOffset; │ │ │ │ │ + backgroundStyle.graphicZIndex = backgroundStyle.backgroundGraphicZIndex; │ │ │ │ │ + backgroundStyle.graphicWidth = backgroundStyle.backgroundWidth || backgroundStyle.graphicWidth; │ │ │ │ │ + backgroundStyle.graphicHeight = backgroundStyle.backgroundHeight || backgroundStyle.graphicHeight; │ │ │ │ │ + │ │ │ │ │ + // Erase background styles. │ │ │ │ │ + backgroundStyle.backgroundGraphic = null; │ │ │ │ │ + backgroundStyle.backgroundXOffset = null; │ │ │ │ │ + backgroundStyle.backgroundYOffset = null; │ │ │ │ │ + backgroundStyle.backgroundGraphicZIndex = null; │ │ │ │ │ + │ │ │ │ │ + return this.redrawNode( │ │ │ │ │ + id + this.BACKGROUND_ID_SUFFIX, │ │ │ │ │ + geometry, │ │ │ │ │ + backgroundStyle, │ │ │ │ │ + null │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: calculateResolutions │ │ │ │ │ - * Calculate resolutions based on the provided properties. │ │ │ │ │ + * Method: drawGeometryNode │ │ │ │ │ + * Given a node, draw a geometry on the specified layer. │ │ │ │ │ + * node and geometry are required arguments, style is optional. │ │ │ │ │ + * This method is only called by the render itself. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * props - {Object} Properties │ │ │ │ │ - * │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array({Number})} Array of resolutions. │ │ │ │ │ + * {Object} a hash with properties "node" (the drawn node) and "complete" │ │ │ │ │ + * (null if parts of the geometry could not be drawn, false if nothing │ │ │ │ │ + * could be drawn) │ │ │ │ │ */ │ │ │ │ │ - calculateResolutions: function(props) { │ │ │ │ │ - │ │ │ │ │ - var viewSize, wRes, hRes; │ │ │ │ │ - │ │ │ │ │ - // determine maxResolution │ │ │ │ │ - var maxResolution = props.maxResolution; │ │ │ │ │ - if (props.minScale != null) { │ │ │ │ │ - maxResolution = │ │ │ │ │ - OpenLayers.Util.getResolutionFromScale(props.minScale, │ │ │ │ │ - this.units); │ │ │ │ │ - } else if (maxResolution == "auto" && this.maxExtent != null) { │ │ │ │ │ - viewSize = this.map.getSize(); │ │ │ │ │ - wRes = this.maxExtent.getWidth() / viewSize.w; │ │ │ │ │ - hRes = this.maxExtent.getHeight() / viewSize.h; │ │ │ │ │ - maxResolution = Math.max(wRes, hRes); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // determine minResolution │ │ │ │ │ - var minResolution = props.minResolution; │ │ │ │ │ - if (props.maxScale != null) { │ │ │ │ │ - minResolution = │ │ │ │ │ - OpenLayers.Util.getResolutionFromScale(props.maxScale, │ │ │ │ │ - this.units); │ │ │ │ │ - } else if (props.minResolution == "auto" && this.minExtent != null) { │ │ │ │ │ - viewSize = this.map.getSize(); │ │ │ │ │ - wRes = this.minExtent.getWidth() / viewSize.w; │ │ │ │ │ - hRes = this.minExtent.getHeight() / viewSize.h; │ │ │ │ │ - minResolution = Math.max(wRes, hRes); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (typeof maxResolution !== "number" && │ │ │ │ │ - typeof minResolution !== "number" && │ │ │ │ │ - this.maxExtent != null) { │ │ │ │ │ - // maxResolution for default grid sets assumes that at zoom │ │ │ │ │ - // level zero, the whole world fits on one tile. │ │ │ │ │ - var tileSize = this.map.getTileSize(); │ │ │ │ │ - maxResolution = Math.max( │ │ │ │ │ - this.maxExtent.getWidth() / tileSize.w, │ │ │ │ │ - this.maxExtent.getHeight() / tileSize.h │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // determine numZoomLevels │ │ │ │ │ - var maxZoomLevel = props.maxZoomLevel; │ │ │ │ │ - var numZoomLevels = props.numZoomLevels; │ │ │ │ │ - if (typeof minResolution === "number" && │ │ │ │ │ - typeof maxResolution === "number" && numZoomLevels === undefined) { │ │ │ │ │ - var ratio = maxResolution / minResolution; │ │ │ │ │ - numZoomLevels = Math.floor(Math.log(ratio) / Math.log(2)) + 1; │ │ │ │ │ - } else if (numZoomLevels === undefined && maxZoomLevel != null) { │ │ │ │ │ - numZoomLevels = maxZoomLevel + 1; │ │ │ │ │ - } │ │ │ │ │ + drawGeometryNode: function(node, geometry, style) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ │ │ │ │ │ - // are we able to calculate resolutions? │ │ │ │ │ - if (typeof numZoomLevels !== "number" || numZoomLevels <= 0 || │ │ │ │ │ - (typeof maxResolution !== "number" && │ │ │ │ │ - typeof minResolution !== "number")) { │ │ │ │ │ - return; │ │ │ │ │ + var options = { │ │ │ │ │ + 'isFilled': style.fill === undefined ? │ │ │ │ │ + true : style.fill, │ │ │ │ │ + 'isStroked': style.stroke === undefined ? │ │ │ │ │ + !!style.strokeWidth : style.stroke │ │ │ │ │ + }; │ │ │ │ │ + var drawn; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.graphic === false) { │ │ │ │ │ + options.isFilled = false; │ │ │ │ │ + options.isStroked = false; │ │ │ │ │ + } │ │ │ │ │ + drawn = this.drawPoint(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + options.isFilled = false; │ │ │ │ │ + drawn = this.drawLineString(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + drawn = this.drawLinearRing(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + drawn = this.drawPolygon(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + drawn = this.drawRectangle(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // now we have numZoomLevels and at least one of maxResolution │ │ │ │ │ - // or minResolution, we can populate the resolutions array │ │ │ │ │ - │ │ │ │ │ - var resolutions = new Array(numZoomLevels); │ │ │ │ │ - var base = 2; │ │ │ │ │ - if (typeof minResolution == "number" && │ │ │ │ │ - typeof maxResolution == "number") { │ │ │ │ │ - // if maxResolution and minResolution are set, we calculate │ │ │ │ │ - // the base for exponential scaling that starts at │ │ │ │ │ - // maxResolution and ends at minResolution in numZoomLevels │ │ │ │ │ - // steps. │ │ │ │ │ - base = Math.pow( │ │ │ │ │ - (maxResolution / minResolution), │ │ │ │ │ - (1 / (numZoomLevels - 1)) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + node._options = options; │ │ │ │ │ │ │ │ │ │ - var i; │ │ │ │ │ - if (typeof maxResolution === "number") { │ │ │ │ │ - for (i = 0; i < numZoomLevels; i++) { │ │ │ │ │ - resolutions[i] = maxResolution / Math.pow(base, i); │ │ │ │ │ - } │ │ │ │ │ + //set style │ │ │ │ │ + //TBD simplify this │ │ │ │ │ + if (drawn != false) { │ │ │ │ │ + return { │ │ │ │ │ + node: this.setStyle(node, style, options, geometry), │ │ │ │ │ + complete: drawn │ │ │ │ │ + }; │ │ │ │ │ } else { │ │ │ │ │ - for (i = 0; i < numZoomLevels; i++) { │ │ │ │ │ - resolutions[numZoomLevels - 1 - i] = │ │ │ │ │ - minResolution * Math.pow(base, i); │ │ │ │ │ - } │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return resolutions; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getResolution │ │ │ │ │ + * Method: postDraw │ │ │ │ │ + * Things that have do be done after the geometry node is appended │ │ │ │ │ + * to its parent node. To be overridden by subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + postDraw: function(node) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawPoint │ │ │ │ │ + * Virtual function for drawing Point Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Float} The currently selected resolution of the map, taken from the │ │ │ │ │ - * resolutions array, indexed by current zoom level. │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the point │ │ │ │ │ */ │ │ │ │ │ - getResolution: function() { │ │ │ │ │ - var zoom = this.map.getZoom(); │ │ │ │ │ - return this.getResolutionForZoom(zoom); │ │ │ │ │ - }, │ │ │ │ │ + drawPoint: function(node, geometry) {}, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getExtent │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawLineString │ │ │ │ │ + * Virtual function for drawing LineString Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} A Bounds object which represents the lon/lat │ │ │ │ │ - * bounds of the current viewPort. │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components of │ │ │ │ │ + * the linestring, or false if nothing could be drawn │ │ │ │ │ */ │ │ │ │ │ - getExtent: function() { │ │ │ │ │ - // just use stock map calculateBounds function -- passing no arguments │ │ │ │ │ - // means it will user map's current center & resolution │ │ │ │ │ - // │ │ │ │ │ - return this.map.calculateBounds(); │ │ │ │ │ - }, │ │ │ │ │ + drawLineString: function(node, geometry) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getZoomForExtent │ │ │ │ │ + * Method: drawLinearRing │ │ │ │ │ + * Virtual function for drawing LinearRing Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} │ │ │ │ │ - * closest - {Boolean} Find the zoom level that most closely fits the │ │ │ │ │ - * specified bounds. Note that this may result in a zoom that does │ │ │ │ │ - * not exactly contain the entire extent. │ │ │ │ │ - * Default is false. │ │ │ │ │ - * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Integer} The index of the zoomLevel (entry in the resolutions array) │ │ │ │ │ - * for the passed-in extent. We do this by calculating the ideal │ │ │ │ │ - * resolution for the given extent (based on the map size) and then │ │ │ │ │ - * calling getZoomForResolution(), passing along the 'closest' │ │ │ │ │ - * parameter. │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ + * of the linear ring, or false if nothing could be drawn │ │ │ │ │ */ │ │ │ │ │ - getZoomForExtent: function(extent, closest) { │ │ │ │ │ - var viewSize = this.map.getSize(); │ │ │ │ │ - var idealResolution = Math.max(extent.getWidth() / viewSize.w, │ │ │ │ │ - extent.getHeight() / viewSize.h); │ │ │ │ │ + drawLinearRing: function(node, geometry) {}, │ │ │ │ │ │ │ │ │ │ - return this.getZoomForResolution(idealResolution, closest); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawPolygon │ │ │ │ │ + * Virtual function for drawing Polygon Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ + * of the polygon, or false if nothing could be drawn │ │ │ │ │ + */ │ │ │ │ │ + drawPolygon: function(node, geometry) {}, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getDataExtent │ │ │ │ │ - * Calculates the max extent which includes all of the data for the layer. │ │ │ │ │ - * This function is to be implemented by subclasses. │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawRectangle │ │ │ │ │ + * Virtual function for drawing Rectangle Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the rectangle │ │ │ │ │ */ │ │ │ │ │ - getDataExtent: function() { │ │ │ │ │ - //to be implemented by subclasses │ │ │ │ │ - }, │ │ │ │ │ + drawRectangle: function(node, geometry) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getResolutionForZoom │ │ │ │ │ + * Method: drawCircle │ │ │ │ │ + * Virtual function for drawing Circle Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * zoom - {Float} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Float} A suitable resolution for the specified zoom. │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the circle │ │ │ │ │ */ │ │ │ │ │ - getResolutionForZoom: function(zoom) { │ │ │ │ │ - zoom = Math.max(0, Math.min(zoom, this.resolutions.length - 1)); │ │ │ │ │ - var resolution; │ │ │ │ │ - if (this.map.fractionalZoom) { │ │ │ │ │ - var low = Math.floor(zoom); │ │ │ │ │ - var high = Math.ceil(zoom); │ │ │ │ │ - resolution = this.resolutions[low] - │ │ │ │ │ - ((zoom - low) * (this.resolutions[low] - this.resolutions[high])); │ │ │ │ │ - } else { │ │ │ │ │ - resolution = this.resolutions[Math.round(zoom)]; │ │ │ │ │ + drawCircle: function(node, geometry) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeText │ │ │ │ │ + * Removes a label │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + removeText: function(featureId) { │ │ │ │ │ + var label = document.getElementById(featureId + this.LABEL_ID_SUFFIX); │ │ │ │ │ + if (label) { │ │ │ │ │ + this.textRoot.removeChild(label); │ │ │ │ │ + } │ │ │ │ │ + var outline = document.getElementById(featureId + this.LABEL_OUTLINE_SUFFIX); │ │ │ │ │ + if (outline) { │ │ │ │ │ + this.textRoot.removeChild(outline); │ │ │ │ │ } │ │ │ │ │ - return resolution; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getZoomForResolution │ │ │ │ │ + * Method: getFeatureIdFromEvent │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resolution - {Float} │ │ │ │ │ - * closest - {Boolean} Find the zoom level that corresponds to the absolute │ │ │ │ │ - * closest resolution, which may result in a zoom whose corresponding │ │ │ │ │ - * resolution is actually smaller than we would have desired (if this │ │ │ │ │ - * is being called from a getZoomForExtent() call, then this means that │ │ │ │ │ - * the returned zoom index might not actually contain the entire │ │ │ │ │ - * extent specified... but it'll be close). │ │ │ │ │ - * Default is false. │ │ │ │ │ - * │ │ │ │ │ + * evt - {Object} An <OpenLayers.Event> object │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Integer} The index of the zoomLevel (entry in the resolutions array) │ │ │ │ │ - * that corresponds to the best fit resolution given the passed in │ │ │ │ │ - * value and the 'closest' specification. │ │ │ │ │ + * {String} A feature id or undefined. │ │ │ │ │ */ │ │ │ │ │ - getZoomForResolution: function(resolution, closest) { │ │ │ │ │ - var zoom, i, len; │ │ │ │ │ - if (this.map.fractionalZoom) { │ │ │ │ │ - var lowZoom = 0; │ │ │ │ │ - var highZoom = this.resolutions.length - 1; │ │ │ │ │ - var highRes = this.resolutions[lowZoom]; │ │ │ │ │ - var lowRes = this.resolutions[highZoom]; │ │ │ │ │ - var res; │ │ │ │ │ - for (i = 0, len = this.resolutions.length; i < len; ++i) { │ │ │ │ │ - res = this.resolutions[i]; │ │ │ │ │ - if (res >= resolution) { │ │ │ │ │ - highRes = res; │ │ │ │ │ - lowZoom = i; │ │ │ │ │ - } │ │ │ │ │ - if (res <= resolution) { │ │ │ │ │ - lowRes = res; │ │ │ │ │ - highZoom = i; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var dRes = highRes - lowRes; │ │ │ │ │ - if (dRes > 0) { │ │ │ │ │ - zoom = lowZoom + ((highRes - resolution) / dRes); │ │ │ │ │ - } else { │ │ │ │ │ - zoom = lowZoom; │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var target = evt.target; │ │ │ │ │ + var useElement = target && target.correspondingUseElement; │ │ │ │ │ + var node = useElement ? useElement : (target || evt.srcElement); │ │ │ │ │ + return node._featureId; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: eraseGeometry │ │ │ │ │ + * Erase a geometry from the renderer. In the case of a multi-geometry, │ │ │ │ │ + * we cycle through and recurse on ourselves. Otherwise, we look for a │ │ │ │ │ + * node with the geometry.id, destroy its geometry, and remove it from │ │ │ │ │ + * the DOM. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + eraseGeometry: function(geometry, featureId) { │ │ │ │ │ + if ((geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ + (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ + (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPolygon") || │ │ │ │ │ + (geometry.CLASS_NAME == "OpenLayers.Geometry.Collection")) { │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + this.eraseGeometry(geometry.components[i], featureId); │ │ │ │ │ } │ │ │ │ │ } else { │ │ │ │ │ - var diff; │ │ │ │ │ - var minDiff = Number.POSITIVE_INFINITY; │ │ │ │ │ - for (i = 0, len = this.resolutions.length; i < len; i++) { │ │ │ │ │ - if (closest) { │ │ │ │ │ - diff = Math.abs(this.resolutions[i] - resolution); │ │ │ │ │ - if (diff > minDiff) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - minDiff = diff; │ │ │ │ │ - } else { │ │ │ │ │ - if (this.resolutions[i] < resolution) { │ │ │ │ │ - break; │ │ │ │ │ + var element = OpenLayers.Util.getElement(geometry.id); │ │ │ │ │ + if (element && element.parentNode) { │ │ │ │ │ + if (element.geometry) { │ │ │ │ │ + element.geometry.destroy(); │ │ │ │ │ + element.geometry = null; │ │ │ │ │ + } │ │ │ │ │ + element.parentNode.removeChild(element); │ │ │ │ │ + │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + this.indexer.remove(element); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (element._style.backgroundGraphic) { │ │ │ │ │ + var backgroundId = geometry.id + this.BACKGROUND_ID_SUFFIX; │ │ │ │ │ + var bElem = OpenLayers.Util.getElement(backgroundId); │ │ │ │ │ + if (bElem && bElem.parentNode) { │ │ │ │ │ + // No need to destroy the geometry since the element and the background │ │ │ │ │ + // node share the same geometry. │ │ │ │ │ + bElem.parentNode.removeChild(bElem); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - zoom = Math.max(0, i - 1); │ │ │ │ │ } │ │ │ │ │ - return zoom; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getLonLatFromViewPortPx │ │ │ │ │ + /** │ │ │ │ │ + * Method: nodeFactory │ │ │ │ │ + * Create new node of the specified type, with the (optional) specified id. │ │ │ │ │ + * │ │ │ │ │ + * If node already exists with same ID and a different type, we remove it │ │ │ │ │ + * and then call ourselves again to recreate it. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * viewPortPx - {<OpenLayers.Pixel>|Object} An OpenLayers.Pixel or │ │ │ │ │ - * an object with a 'x' │ │ │ │ │ - * and 'y' properties. │ │ │ │ │ - * │ │ │ │ │ + * id - {String} │ │ │ │ │ + * type - {String} type Kind of node to draw. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.LonLat>} An OpenLayers.LonLat which is the passed-in │ │ │ │ │ - * view port <OpenLayers.Pixel>, translated into lon/lat by the layer. │ │ │ │ │ + * {DOMElement} A new node of the given type and id. │ │ │ │ │ */ │ │ │ │ │ - getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ - var lonlat = null; │ │ │ │ │ - var map = this.map; │ │ │ │ │ - if (viewPortPx != null && map.minPx) { │ │ │ │ │ - var res = map.getResolution(); │ │ │ │ │ - var maxExtent = map.getMaxExtent({ │ │ │ │ │ - restricted: true │ │ │ │ │ - }); │ │ │ │ │ - var lon = (viewPortPx.x - map.minPx.x) * res + maxExtent.left; │ │ │ │ │ - var lat = (map.minPx.y - viewPortPx.y) * res + maxExtent.top; │ │ │ │ │ - lonlat = new OpenLayers.LonLat(lon, lat); │ │ │ │ │ - │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - lonlat = lonlat.wrapDateLine(this.maxExtent); │ │ │ │ │ + nodeFactory: function(id, type) { │ │ │ │ │ + var node = OpenLayers.Util.getElement(id); │ │ │ │ │ + if (node) { │ │ │ │ │ + if (!this.nodeTypeCompare(node, type)) { │ │ │ │ │ + node.parentNode.removeChild(node); │ │ │ │ │ + node = this.nodeFactory(id, type); │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + node = this.createNode(type, id); │ │ │ │ │ } │ │ │ │ │ - return lonlat; │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getViewPortPxFromLonLat │ │ │ │ │ - * Returns a pixel location given a map location. This method will return │ │ │ │ │ - * fractional pixel values. │ │ │ │ │ + /** │ │ │ │ │ + * Method: nodeTypeCompare │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>|Object} An OpenLayers.LonLat or │ │ │ │ │ - * an object with a 'lon' │ │ │ │ │ - * and 'lat' properties. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Pixel>} An <OpenLayers.Pixel> which is the passed-in │ │ │ │ │ - * lonlat translated into view port pixels. │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * type - {String} Kind of node │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ + * This function must be overridden by subclasses. │ │ │ │ │ */ │ │ │ │ │ - getViewPortPxFromLonLat: function(lonlat, resolution) { │ │ │ │ │ - var px = null; │ │ │ │ │ - if (lonlat != null) { │ │ │ │ │ - resolution = resolution || this.map.getResolution(); │ │ │ │ │ - var extent = this.map.calculateBounds(null, resolution); │ │ │ │ │ - px = new OpenLayers.Pixel( │ │ │ │ │ - (1 / resolution * (lonlat.lon - extent.left)), │ │ │ │ │ - (1 / resolution * (extent.top - lonlat.lat)) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return px; │ │ │ │ │ - }, │ │ │ │ │ + nodeTypeCompare: function(node, type) {}, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setOpacity │ │ │ │ │ - * Sets the opacity for the entire layer (all images) │ │ │ │ │ + /** │ │ │ │ │ + * Method: createNode │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * opacity - {Float} │ │ │ │ │ + * type - {String} Kind of node to draw. │ │ │ │ │ + * id - {String} Id for node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A new node of the given type and id. │ │ │ │ │ + * This function must be overridden by subclasses. │ │ │ │ │ */ │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - if (opacity != this.opacity) { │ │ │ │ │ - this.opacity = opacity; │ │ │ │ │ - var childNodes = this.div.childNodes; │ │ │ │ │ - for (var i = 0, len = childNodes.length; i < len; ++i) { │ │ │ │ │ - var element = childNodes[i].firstChild || childNodes[i]; │ │ │ │ │ - var lastChild = childNodes[i].lastChild; │ │ │ │ │ - //TODO de-uglify this │ │ │ │ │ - if (lastChild && lastChild.nodeName.toLowerCase() === "iframe") { │ │ │ │ │ - element = lastChild.parentNode; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.modifyDOMElement(element, null, null, null, │ │ │ │ │ - null, null, null, opacity); │ │ │ │ │ - } │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "opacity" │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + createNode: function(type, id) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getZIndex │ │ │ │ │ + * Method: moveRoot │ │ │ │ │ + * moves this renderer's root to a different renderer. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} the z-index of this layer │ │ │ │ │ + * Parameters: │ │ │ │ │ + * renderer - {<OpenLayers.Renderer>} target renderer for the moved root │ │ │ │ │ */ │ │ │ │ │ - getZIndex: function() { │ │ │ │ │ - return this.div.style.zIndex; │ │ │ │ │ + moveRoot: function(renderer) { │ │ │ │ │ + var root = this.root; │ │ │ │ │ + if (renderer.root.parentNode == this.rendererRoot) { │ │ │ │ │ + root = renderer.root; │ │ │ │ │ + } │ │ │ │ │ + root.parentNode.removeChild(root); │ │ │ │ │ + renderer.rendererRoot.appendChild(root); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setZIndex │ │ │ │ │ + * Method: getRenderLayerId │ │ │ │ │ + * Gets the layer that this renderer's output appears on. If moveRoot was │ │ │ │ │ + * used, this will be different from the id of the layer containing the │ │ │ │ │ + * features rendered by this renderer. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * zIndex - {Integer} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} the id of the output layer. │ │ │ │ │ */ │ │ │ │ │ - setZIndex: function(zIndex) { │ │ │ │ │ - this.div.style.zIndex = zIndex; │ │ │ │ │ + getRenderLayerId: function() { │ │ │ │ │ + return this.root.parentNode.parentNode.id; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: adjustBounds │ │ │ │ │ - * This function will take a bounds, and if wrapDateLine option is set │ │ │ │ │ - * on the layer, it will return a bounds which is wrapped around the │ │ │ │ │ - * world. We do not wrap for bounds which *cross* the │ │ │ │ │ - * maxExtent.left/right, only bounds which are entirely to the left │ │ │ │ │ - * or entirely to the right. │ │ │ │ │ + * Method: isComplexSymbol │ │ │ │ │ + * Determines if a symbol cannot be rendered using drawCircle │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * graphicName - {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns │ │ │ │ │ + * {Boolean} true if the symbol is complex, false if not │ │ │ │ │ */ │ │ │ │ │ - adjustBounds: function(bounds) { │ │ │ │ │ - │ │ │ │ │ - if (this.gutter) { │ │ │ │ │ - // Adjust the extent of a bounds in map units by the │ │ │ │ │ - // layer's gutter in pixels. │ │ │ │ │ - var mapGutter = this.gutter * this.map.getResolution(); │ │ │ │ │ - bounds = new OpenLayers.Bounds(bounds.left - mapGutter, │ │ │ │ │ - bounds.bottom - mapGutter, │ │ │ │ │ - bounds.right + mapGutter, │ │ │ │ │ - bounds.top + mapGutter); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - // wrap around the date line, within the limits of rounding error │ │ │ │ │ - var wrappingOptions = { │ │ │ │ │ - 'rightTolerance': this.getResolution(), │ │ │ │ │ - 'leftTolerance': this.getResolution() │ │ │ │ │ - }; │ │ │ │ │ - bounds = bounds.wrapDateLine(this.maxExtent, wrappingOptions); │ │ │ │ │ - │ │ │ │ │ - } │ │ │ │ │ - return bounds; │ │ │ │ │ + isComplexSymbol: function(graphicName) { │ │ │ │ │ + return (graphicName != "circle") && !!graphicName; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.Elements" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/HTTPRequest.js │ │ │ │ │ + OpenLayers/Renderer/VML.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer.js │ │ │ │ │ + * @requires OpenLayers/Renderer/Elements.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.HTTPRequest │ │ │ │ │ + * Class: OpenLayers.Renderer.VML │ │ │ │ │ + * Render vector features in browsers with VML capability. Construct a new │ │ │ │ │ + * VML renderer with the <OpenLayers.Renderer.VML> constructor. │ │ │ │ │ * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer> │ │ │ │ │ + * Note that for all calculations in this class, we use (num | 0) to truncate a │ │ │ │ │ + * float value to an integer. This is done because it seems that VML doesn't │ │ │ │ │ + * support float values. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Renderer.Elements> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.HTTPRequest = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: URL_HASH_FACTOR │ │ │ │ │ - * {Float} Used to hash URL param strings for multi-WMS server selection. │ │ │ │ │ - * Set to the Golden Ratio per Knuth's recommendation. │ │ │ │ │ - */ │ │ │ │ │ - URL_HASH_FACTOR: (Math.sqrt(5) - 1) / 2, │ │ │ │ │ +OpenLayers.Renderer.VML = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: url │ │ │ │ │ - * {Array(String) or String} This is either an array of url strings or │ │ │ │ │ - * a single url string. │ │ │ │ │ + /** │ │ │ │ │ + * Property: xmlns │ │ │ │ │ + * {String} XML Namespace URN │ │ │ │ │ */ │ │ │ │ │ - url: null, │ │ │ │ │ + xmlns: "urn:schemas-microsoft-com:vml", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: params │ │ │ │ │ - * {Object} Hashtable of key/value parameters │ │ │ │ │ + /** │ │ │ │ │ + * Property: symbolCache │ │ │ │ │ + * {DOMElement} node holding symbols. This hash is keyed by symbol name, │ │ │ │ │ + * and each value is a hash with a "path" and an "extent" property. │ │ │ │ │ */ │ │ │ │ │ - params: null, │ │ │ │ │ + symbolCache: {}, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: reproject │ │ │ │ │ - * *Deprecated*. See http://docs.openlayers.org/library/spherical_mercator.html │ │ │ │ │ - * for information on the replacement for this functionality. │ │ │ │ │ - * {Boolean} Whether layer should reproject itself based on base layer │ │ │ │ │ - * locations. This allows reprojection onto commercial layers. │ │ │ │ │ - * Default is false: Most layers can't reproject, but layers │ │ │ │ │ - * which can create non-square geographic pixels can, like WMS. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Property: offset │ │ │ │ │ + * {Object} Hash with "x" and "y" properties │ │ │ │ │ */ │ │ │ │ │ - reproject: false, │ │ │ │ │ + offset: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.HTTPRequest │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Renderer.VML │ │ │ │ │ + * Create a new VML renderer. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * url - {Array(String) or String} │ │ │ │ │ - * params - {Object} │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + * containerID - {String} The id for the element that contains the renderer │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ - this.url = url; │ │ │ │ │ - if (!this.params) { │ │ │ │ │ - this.params = OpenLayers.Util.extend({}, params); │ │ │ │ │ + initialize: function(containerID) { │ │ │ │ │ + if (!this.supported()) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (!document.namespaces.olv) { │ │ │ │ │ + document.namespaces.add("olv", this.xmlns); │ │ │ │ │ + var style = document.createStyleSheet(); │ │ │ │ │ + var shapes = ['shape', 'rect', 'oval', 'fill', 'stroke', 'imagedata', 'group', 'textbox']; │ │ │ │ │ + for (var i = 0, len = shapes.length; i < len; i++) { │ │ │ │ │ + │ │ │ │ │ + style.addRule('olv\\:' + shapes[i], "behavior: url(#default#VML); " + │ │ │ │ │ + "position: absolute; display: inline-block;"); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.initialize.apply(this, │ │ │ │ │ + arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ + * APIMethod: supported │ │ │ │ │ + * Determine whether a browser supports this renderer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The browser supports the VML renderer │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.url = null; │ │ │ │ │ - this.params = null; │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ + supported: function() { │ │ │ │ │ + return !!(document.namespaces); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * │ │ │ │ │ + * Method: setExtent │ │ │ │ │ + * Set the renderer's extent │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {Object} │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} │ │ │ │ │ + * resolutionChanged - {Boolean} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.HTTPRequest>} An exact clone of this │ │ │ │ │ - * <OpenLayers.Layer.HTTPRequest> │ │ │ │ │ + * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ + * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.HTTPRequest(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.params, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ + var left = (extent.left / resolution) | 0; │ │ │ │ │ + var top = (extent.top / resolution - this.size.h) | 0; │ │ │ │ │ + if (resolutionChanged || !this.offset) { │ │ │ │ │ + this.offset = { │ │ │ │ │ + x: left, │ │ │ │ │ + y: top │ │ │ │ │ + }; │ │ │ │ │ + left = 0; │ │ │ │ │ + top = 0; │ │ │ │ │ + } else { │ │ │ │ │ + left = left - this.offset.x; │ │ │ │ │ + top = top - this.offset.y; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ + var org = (left - this.xOffset) + " " + top; │ │ │ │ │ + this.root.coordorigin = org; │ │ │ │ │ + var roots = [this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ + var root; │ │ │ │ │ + for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ + root = roots[i]; │ │ │ │ │ │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ + var size = this.size.w + " " + this.size.h; │ │ │ │ │ + root.coordsize = size; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setUrl │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newUrl - {String} │ │ │ │ │ - */ │ │ │ │ │ - setUrl: function(newUrl) { │ │ │ │ │ - this.url = newUrl; │ │ │ │ │ + } │ │ │ │ │ + // flip the VML display Y axis upside down so it │ │ │ │ │ + // matches the display Y axis of the map │ │ │ │ │ + this.root.style.flip = "y"; │ │ │ │ │ + │ │ │ │ │ + return coordSysUnchanged; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: mergeNewParams │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newParams - {Object} │ │ │ │ │ + * Method: setSize │ │ │ │ │ + * Set the size of the drawing surface │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * redrawn: {Boolean} whether the layer was actually redrawn. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * size - {<OpenLayers.Size>} the size of the drawing surface │ │ │ │ │ */ │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - this.params = OpenLayers.Util.extend(this.params, newParams); │ │ │ │ │ - var ret = this.redraw(); │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "params" │ │ │ │ │ - }); │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + // setting width and height on all roots to avoid flicker which we │ │ │ │ │ + // would get with 100% width and height on child roots │ │ │ │ │ + var roots = [ │ │ │ │ │ + this.rendererRoot, │ │ │ │ │ + this.root, │ │ │ │ │ + this.vectorRoot, │ │ │ │ │ + this.textRoot │ │ │ │ │ + ]; │ │ │ │ │ + var w = this.size.w + "px"; │ │ │ │ │ + var h = this.size.h + "px"; │ │ │ │ │ + var root; │ │ │ │ │ + for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ + root = roots[i]; │ │ │ │ │ + root.style.width = w; │ │ │ │ │ + root.style.height = h; │ │ │ │ │ } │ │ │ │ │ - return ret; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: redraw │ │ │ │ │ - * Redraws the layer. Returns true if the layer was redrawn, false if not. │ │ │ │ │ + * Method: getNodeType │ │ │ │ │ + * Get the node type for a geometry and style │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * force - {Boolean} Force redraw by adding random parameter. │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The layer was redrawn. │ │ │ │ │ + * {String} The corresponding node type for the specified geometry │ │ │ │ │ */ │ │ │ │ │ - redraw: function(force) { │ │ │ │ │ - if (force) { │ │ │ │ │ - return this.mergeNewParams({ │ │ │ │ │ - "_olSalt": Math.random() │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - return OpenLayers.Layer.prototype.redraw.apply(this, []); │ │ │ │ │ + getNodeType: function(geometry, style) { │ │ │ │ │ + var nodeType = null; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + nodeType = "olv:rect"; │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + nodeType = "olv:shape"; │ │ │ │ │ + } else { │ │ │ │ │ + nodeType = "olv:oval"; │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + nodeType = "olv:rect"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + case "OpenLayers.Geometry.Curve": │ │ │ │ │ + nodeType = "olv:shape"; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ + return nodeType; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: selectUrl │ │ │ │ │ - * selectUrl() implements the standard floating-point multiplicative │ │ │ │ │ - * hash function described by Knuth, and hashes the contents of the │ │ │ │ │ - * given param string into a float between 0 and 1. This float is then │ │ │ │ │ - * scaled to the size of the provided urls array, and used to select │ │ │ │ │ - * a URL. │ │ │ │ │ + * Method: setStyle │ │ │ │ │ + * Use to set all the style attributes to a VML node. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * paramString - {String} │ │ │ │ │ - * urls - {Array(String)} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} An entry from the urls array, deterministically selected based │ │ │ │ │ - * on the paramString. │ │ │ │ │ + * node - {DOMElement} An VML element to decorate │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * options - {Object} Currently supported options include │ │ │ │ │ + * 'isFilled' {Boolean} and │ │ │ │ │ + * 'isStroked' {Boolean} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ */ │ │ │ │ │ - selectUrl: function(paramString, urls) { │ │ │ │ │ - var product = 1; │ │ │ │ │ - for (var i = 0, len = paramString.length; i < len; i++) { │ │ │ │ │ - product *= paramString.charCodeAt(i) * this.URL_HASH_FACTOR; │ │ │ │ │ - product -= Math.floor(product); │ │ │ │ │ + setStyle: function(node, style, options, geometry) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + options = options || node._options; │ │ │ │ │ + var fillColor = style.fillColor; │ │ │ │ │ + │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + node.title = title; │ │ │ │ │ } │ │ │ │ │ - return urls[Math.floor(product * urls.length)]; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getFullRequestString │ │ │ │ │ - * Combine url with layer's params and these newParams. │ │ │ │ │ - * │ │ │ │ │ - * does checking on the serverPath variable, allowing for cases when it │ │ │ │ │ - * is supplied with trailing ? or &, as well as cases where not. │ │ │ │ │ - * │ │ │ │ │ - * return in formatted string like this: │ │ │ │ │ - * "server?key1=value1&key2=value2&key3=value3" │ │ │ │ │ - * │ │ │ │ │ - * WARNING: The altUrl parameter is deprecated and will be removed in 3.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newParams - {Object} │ │ │ │ │ - * altUrl - {String} Use this as the url instead of the layer's url │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ + if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + options.isFilled = true; │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ │ │ │ │ │ - // if not altUrl passed in, use layer's url │ │ │ │ │ - var url = altUrl || this.url; │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ + style.graphicXOffset : -(0.5 * width); │ │ │ │ │ + var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ + style.graphicYOffset : -(0.5 * height); │ │ │ │ │ │ │ │ │ │ - // create a new params hashtable with all the layer params and the │ │ │ │ │ - // new params together. then convert to string │ │ │ │ │ - var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ - allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ - var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + node.style.left = ((((geometry.x - this.featureDx) / resolution - this.offset.x) + xOffset) | 0) + "px"; │ │ │ │ │ + node.style.top = (((geometry.y / resolution - this.offset.y) - (yOffset + height)) | 0) + "px"; │ │ │ │ │ + node.style.width = width + "px"; │ │ │ │ │ + node.style.height = height + "px"; │ │ │ │ │ + node.style.flip = "y"; │ │ │ │ │ │ │ │ │ │ - // if url is not a string, it should be an array of strings, │ │ │ │ │ - // in which case we will deterministically select one of them in │ │ │ │ │ - // order to evenly distribute requests to different urls. │ │ │ │ │ - // │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(paramsString, url); │ │ │ │ │ + // modify fillColor and options for stroke styling below │ │ │ │ │ + fillColor = "none"; │ │ │ │ │ + options.isStroked = false; │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + var cache = this.importSymbol(style.graphicName); │ │ │ │ │ + node.path = cache.path; │ │ │ │ │ + node.coordorigin = cache.left + "," + cache.bottom; │ │ │ │ │ + var size = cache.size; │ │ │ │ │ + node.coordsize = size + "," + size; │ │ │ │ │ + this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ + node.style.flip = "y"; │ │ │ │ │ + } else { │ │ │ │ │ + this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // ignore parameters that are already in the url search string │ │ │ │ │ - var urlParams = │ │ │ │ │ - OpenLayers.Util.upperCaseObject(OpenLayers.Util.getParameters(url)); │ │ │ │ │ - for (var key in allParams) { │ │ │ │ │ - if (key.toUpperCase() in urlParams) { │ │ │ │ │ - delete allParams[key]; │ │ │ │ │ - } │ │ │ │ │ + // fill │ │ │ │ │ + if (options.isFilled) { │ │ │ │ │ + node.fillcolor = fillColor; │ │ │ │ │ + } else { │ │ │ │ │ + node.filled = "false"; │ │ │ │ │ } │ │ │ │ │ - paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + var fills = node.getElementsByTagName("fill"); │ │ │ │ │ + var fill = (fills.length == 0) ? null : fills[0]; │ │ │ │ │ + if (!options.isFilled) { │ │ │ │ │ + if (fill) { │ │ │ │ │ + node.removeChild(fill); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + if (!fill) { │ │ │ │ │ + fill = this.createNode('olv:fill', node.id + "_fill"); │ │ │ │ │ + } │ │ │ │ │ + fill.opacity = style.fillOpacity; │ │ │ │ │ │ │ │ │ │ - return OpenLayers.Util.urlAppend(url, paramsString); │ │ │ │ │ - }, │ │ │ │ │ + if (node._geometryClass === "OpenLayers.Geometry.Point" && │ │ │ │ │ + style.externalGraphic) { │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.HTTPRequest" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Tile.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + // override fillOpacity │ │ │ │ │ + if (style.graphicOpacity) { │ │ │ │ │ + fill.opacity = style.graphicOpacity; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + fill.src = style.externalGraphic; │ │ │ │ │ + fill.type = "frame"; │ │ │ │ │ │ │ │ │ │ + if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ + fill.aspect = "atmost"; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (fill.parentNode != node) { │ │ │ │ │ + node.appendChild(fill); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - */ │ │ │ │ │ + // additional rendering for rotated graphics or symbols │ │ │ │ │ + var rotation = style.rotation; │ │ │ │ │ + if ((rotation !== undefined || node._rotation !== undefined)) { │ │ │ │ │ + node._rotation = rotation; │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ + // make the fill fully transparent, because we now have │ │ │ │ │ + // the graphic as imagedata element. We cannot just remove │ │ │ │ │ + // the fill, because this is part of the hack described │ │ │ │ │ + // in graphicRotate │ │ │ │ │ + fill.opacity = 0; │ │ │ │ │ + } else if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ + node.style.rotation = rotation || 0; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Tile │ │ │ │ │ - * This is a class designed to designate a single tile, however │ │ │ │ │ - * it is explicitly designed to do relatively little. Tiles store │ │ │ │ │ - * information about themselves -- such as the URL that they are related │ │ │ │ │ - * to, and their size - but do not add themselves to the layer div │ │ │ │ │ - * automatically, for example. Create a new tile with the │ │ │ │ │ - * <OpenLayers.Tile> constructor, or a subclass. │ │ │ │ │ - * │ │ │ │ │ - * TBD 3.0 - remove reference to url in above paragraph │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Tile = OpenLayers.Class({ │ │ │ │ │ + // stroke │ │ │ │ │ + var strokes = node.getElementsByTagName("stroke"); │ │ │ │ │ + var stroke = (strokes.length == 0) ? null : strokes[0]; │ │ │ │ │ + if (!options.isStroked) { │ │ │ │ │ + node.stroked = false; │ │ │ │ │ + if (stroke) { │ │ │ │ │ + stroke.on = false; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + if (!stroke) { │ │ │ │ │ + stroke = this.createNode('olv:stroke', node.id + "_stroke"); │ │ │ │ │ + node.appendChild(stroke); │ │ │ │ │ + } │ │ │ │ │ + stroke.on = true; │ │ │ │ │ + stroke.color = style.strokeColor; │ │ │ │ │ + stroke.weight = style.strokeWidth + "px"; │ │ │ │ │ + stroke.opacity = style.strokeOpacity; │ │ │ │ │ + stroke.endcap = style.strokeLinecap == 'butt' ? 'flat' : │ │ │ │ │ + (style.strokeLinecap || 'round'); │ │ │ │ │ + if (style.strokeDashstyle) { │ │ │ │ │ + stroke.dashstyle = this.dashStyle(style); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} An events object that handles all │ │ │ │ │ - * events on the tile. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * tile.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Supported event types: │ │ │ │ │ - * beforedraw - Triggered before the tile is drawn. Used to defer │ │ │ │ │ - * drawing to an animation queue. To defer drawing, listeners need │ │ │ │ │ - * to return false, which will abort drawing. The queue handler needs │ │ │ │ │ - * to call <draw>(true) to actually draw the tile. │ │ │ │ │ - * loadstart - Triggered when tile loading starts. │ │ │ │ │ - * loadend - Triggered when tile loading ends. │ │ │ │ │ - * loaderror - Triggered before the loadend event (i.e. when the tile is │ │ │ │ │ - * still hidden) if the tile could not be loaded. │ │ │ │ │ - * reload - Triggered when an already loading tile is reloaded. │ │ │ │ │ - * unload - Triggered before a tile is unloaded. │ │ │ │ │ - */ │ │ │ │ │ - events: null, │ │ │ │ │ + if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ + node.style.cursor = style.cursor; │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: eventListeners │ │ │ │ │ - * {Object} If set as an option at construction, the eventListeners │ │ │ │ │ - * object will be registered with <OpenLayers.Events.on>. Object │ │ │ │ │ - * structure must be a listeners object as shown in the example for │ │ │ │ │ - * the events.on method. │ │ │ │ │ - * │ │ │ │ │ - * This options can be set in the ``tileOptions`` option from │ │ │ │ │ - * <OpenLayers.Layer.Grid>. For example, to be notified of the │ │ │ │ │ - * ``loadend`` event of each tiles: │ │ │ │ │ - * (code) │ │ │ │ │ - * new OpenLayers.Layer.OSM('osm', 'http://tile.openstreetmap.org/${z}/${x}/${y}.png', { │ │ │ │ │ - * tileOptions: { │ │ │ │ │ - * eventListeners: { │ │ │ │ │ - * 'loadend': function(evt) { │ │ │ │ │ - * // do something on loadend │ │ │ │ │ - * } │ │ │ │ │ - * } │ │ │ │ │ - * } │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ + * Method: graphicRotate │ │ │ │ │ + * If a point is to be styled with externalGraphic and rotation, VML fills │ │ │ │ │ + * cannot be used to display the graphic, because rotation of graphic │ │ │ │ │ + * fills is not supported by the VML implementation of Internet Explorer. │ │ │ │ │ + * This method creates a olv:imagedata element inside the VML node, │ │ │ │ │ + * DXImageTransform.Matrix and BasicImage filters for rotation and │ │ │ │ │ + * opacity, and a 3-step hack to remove rendering artefacts from the │ │ │ │ │ + * graphic and preserve the ability of graphics to trigger events. │ │ │ │ │ + * Finally, OpenLayers methods are used to determine the correct │ │ │ │ │ + * insertion point of the rotated image, because DXImageTransform.Matrix │ │ │ │ │ + * does the rotation without the ability to specify a rotation center │ │ │ │ │ + * point. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * xOffset - {Number} rotation center relative to image, x coordinate │ │ │ │ │ + * yOffset - {Number} rotation center relative to image, y coordinate │ │ │ │ │ + * style - {Object} │ │ │ │ │ */ │ │ │ │ │ - eventListeners: null, │ │ │ │ │ + graphicRotate: function(node, xOffset, yOffset, style) { │ │ │ │ │ + var style = style || node._style; │ │ │ │ │ + var rotation = style.rotation || 0; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: id │ │ │ │ │ - * {String} null │ │ │ │ │ - */ │ │ │ │ │ - id: null, │ │ │ │ │ + var aspectRatio, size; │ │ │ │ │ + if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ + // load the image to determine its size │ │ │ │ │ + var img = new Image(); │ │ │ │ │ + img.onreadystatechange = OpenLayers.Function.bind(function() { │ │ │ │ │ + if (img.readyState == "complete" || │ │ │ │ │ + img.readyState == "interactive") { │ │ │ │ │ + aspectRatio = img.width / img.height; │ │ │ │ │ + size = Math.max(style.pointRadius * 2, │ │ │ │ │ + style.graphicWidth || 0, │ │ │ │ │ + style.graphicHeight || 0); │ │ │ │ │ + xOffset = xOffset * aspectRatio; │ │ │ │ │ + style.graphicWidth = size * aspectRatio; │ │ │ │ │ + style.graphicHeight = size; │ │ │ │ │ + this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ + } │ │ │ │ │ + }, this); │ │ │ │ │ + img.src = style.externalGraphic; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer>} layer the tile is attached to │ │ │ │ │ - */ │ │ │ │ │ - layer: null, │ │ │ │ │ + // will be called again by the onreadystate handler │ │ │ │ │ + return; │ │ │ │ │ + } else { │ │ │ │ │ + size = Math.max(style.graphicWidth, style.graphicHeight); │ │ │ │ │ + aspectRatio = style.graphicWidth / style.graphicHeight; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: url │ │ │ │ │ - * {String} url of the request. │ │ │ │ │ - * │ │ │ │ │ - * TBD 3.0 │ │ │ │ │ - * Deprecated. The base tile class does not need an url. This should be │ │ │ │ │ - * handled in subclasses. Does not belong here. │ │ │ │ │ - */ │ │ │ │ │ - url: null, │ │ │ │ │ + var width = Math.round(style.graphicWidth || size * aspectRatio); │ │ │ │ │ + var height = Math.round(style.graphicHeight || size); │ │ │ │ │ + node.style.width = width + "px"; │ │ │ │ │ + node.style.height = height + "px"; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: bounds │ │ │ │ │ - * {<OpenLayers.Bounds>} null │ │ │ │ │ - */ │ │ │ │ │ - bounds: null, │ │ │ │ │ + // Three steps are required to remove artefacts for images with │ │ │ │ │ + // transparent backgrounds (resulting from using DXImageTransform │ │ │ │ │ + // filters on svg objects), while preserving awareness for browser │ │ │ │ │ + // events on images: │ │ │ │ │ + // - Use the fill as usual (like for unrotated images) to handle │ │ │ │ │ + // events │ │ │ │ │ + // - specify an imagedata element with the same src as the fill │ │ │ │ │ + // - style the imagedata element with an AlphaImageLoader filter │ │ │ │ │ + // with empty src │ │ │ │ │ + var image = document.getElementById(node.id + "_image"); │ │ │ │ │ + if (!image) { │ │ │ │ │ + image = this.createNode("olv:imagedata", node.id + "_image"); │ │ │ │ │ + node.appendChild(image); │ │ │ │ │ + } │ │ │ │ │ + image.style.width = width + "px"; │ │ │ │ │ + image.style.height = height + "px"; │ │ │ │ │ + image.src = style.externalGraphic; │ │ │ │ │ + image.style.filter = │ │ │ │ │ + "progid:DXImageTransform.Microsoft.AlphaImageLoader(" + │ │ │ │ │ + "src='', sizingMethod='scale')"; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: size │ │ │ │ │ - * {<OpenLayers.Size>} null │ │ │ │ │ - */ │ │ │ │ │ - size: null, │ │ │ │ │ + var rot = rotation * Math.PI / 180; │ │ │ │ │ + var sintheta = Math.sin(rot); │ │ │ │ │ + var costheta = Math.cos(rot); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: position │ │ │ │ │ - * {<OpenLayers.Pixel>} Top Left pixel of the tile │ │ │ │ │ - */ │ │ │ │ │ - position: null, │ │ │ │ │ + // do the rotation on the image │ │ │ │ │ + var filter = │ │ │ │ │ + "progid:DXImageTransform.Microsoft.Matrix(M11=" + costheta + │ │ │ │ │ + ",M12=" + (-sintheta) + ",M21=" + sintheta + ",M22=" + costheta + │ │ │ │ │ + ",SizingMethod='auto expand')\n"; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: isLoading │ │ │ │ │ - * {Boolean} Is the tile loading? │ │ │ │ │ - */ │ │ │ │ │ - isLoading: false, │ │ │ │ │ + // set the opacity (needed for the imagedata) │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ + if (opacity && opacity != 1) { │ │ │ │ │ + filter += │ │ │ │ │ + "progid:DXImageTransform.Microsoft.BasicImage(opacity=" + │ │ │ │ │ + opacity + ")\n"; │ │ │ │ │ + } │ │ │ │ │ + node.style.filter = filter; │ │ │ │ │ │ │ │ │ │ - /** TBD 3.0 -- remove 'url' from the list of parameters to the constructor. │ │ │ │ │ - * there is no need for the base tile class to have a url. │ │ │ │ │ - */ │ │ │ │ │ + // do the rotation again on a box, so we know the insertion point │ │ │ │ │ + var centerPoint = new OpenLayers.Geometry.Point(-xOffset, -yOffset); │ │ │ │ │ + var imgBox = new OpenLayers.Bounds(0, 0, width, height).toGeometry(); │ │ │ │ │ + imgBox.rotate(style.rotation, centerPoint); │ │ │ │ │ + var imgBounds = imgBox.getBounds(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Tile │ │ │ │ │ - * Constructor for a new <OpenLayers.Tile> instance. │ │ │ │ │ + node.style.left = Math.round( │ │ │ │ │ + parseInt(node.style.left) + imgBounds.left) + "px"; │ │ │ │ │ + node.style.top = Math.round( │ │ │ │ │ + parseInt(node.style.top) - imgBounds.bottom) + "px"; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: postDraw │ │ │ │ │ + * Does some node postprocessing to work around browser issues: │ │ │ │ │ + * - Some versions of Internet Explorer seem to be unable to set fillcolor │ │ │ │ │ + * and strokecolor to "none" correctly before the fill node is appended │ │ │ │ │ + * to a visible vml node. This method takes care of that and sets │ │ │ │ │ + * fillcolor and strokecolor again if needed. │ │ │ │ │ + * - In some cases, a node won't become visible after being drawn. Setting │ │ │ │ │ + * style.visibility to "visible" works around that. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer>} layer that the tile will go in. │ │ │ │ │ - * position - {<OpenLayers.Pixel>} │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * url - {<String>} │ │ │ │ │ - * size - {<OpenLayers.Size>} │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(layer, position, bounds, url, size, options) { │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - this.position = position.clone(); │ │ │ │ │ - this.setBounds(bounds); │ │ │ │ │ - this.url = url; │ │ │ │ │ - if (size) { │ │ │ │ │ - this.size = size.clone(); │ │ │ │ │ + postDraw: function(node) { │ │ │ │ │ + node.style.visibility = "visible"; │ │ │ │ │ + var fillColor = node._style.fillColor; │ │ │ │ │ + var strokeColor = node._style.strokeColor; │ │ │ │ │ + if (fillColor == "none" && │ │ │ │ │ + node.fillcolor != fillColor) { │ │ │ │ │ + node.fillcolor = fillColor; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - //give the tile a unique id based on its BBOX. │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID("Tile_"); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - │ │ │ │ │ - this.events = new OpenLayers.Events(this); │ │ │ │ │ - if (this.eventListeners instanceof Object) { │ │ │ │ │ - this.events.on(this.eventListeners); │ │ │ │ │ + if (strokeColor == "none" && │ │ │ │ │ + node.strokecolor != strokeColor) { │ │ │ │ │ + node.strokecolor = strokeColor; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: unload │ │ │ │ │ - * Call immediately before destroying if you are listening to tile │ │ │ │ │ - * events, so that counters are properly handled if tile is still │ │ │ │ │ - * loading at destroy-time. Will only fire an event if the tile is │ │ │ │ │ - * still loading. │ │ │ │ │ + * Method: setNodeDimension │ │ │ │ │ + * Get the geometry's bounds, convert it to our vml coordinate system, │ │ │ │ │ + * then set the node's position, size, and local coordinate system. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ */ │ │ │ │ │ - unload: function() { │ │ │ │ │ - if (this.isLoading) { │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - this.events.triggerEvent("unload"); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + setNodeDimension: function(node, geometry) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Nullify references to prevent circular references and memory leaks. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.layer = null; │ │ │ │ │ - this.bounds = null; │ │ │ │ │ - this.size = null; │ │ │ │ │ - this.position = null; │ │ │ │ │ + var bbox = geometry.getBounds(); │ │ │ │ │ + if (bbox) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ │ │ │ │ │ - if (this.eventListeners) { │ │ │ │ │ - this.events.un(this.eventListeners); │ │ │ │ │ + var scaledBox = │ │ │ │ │ + new OpenLayers.Bounds(((bbox.left - this.featureDx) / resolution - this.offset.x) | 0, │ │ │ │ │ + (bbox.bottom / resolution - this.offset.y) | 0, │ │ │ │ │ + ((bbox.right - this.featureDx) / resolution - this.offset.x) | 0, │ │ │ │ │ + (bbox.top / resolution - this.offset.y) | 0); │ │ │ │ │ + │ │ │ │ │ + // Set the internal coordinate system to draw the path │ │ │ │ │ + node.style.left = scaledBox.left + "px"; │ │ │ │ │ + node.style.top = scaledBox.top + "px"; │ │ │ │ │ + node.style.width = scaledBox.getWidth() + "px"; │ │ │ │ │ + node.style.height = scaledBox.getHeight() + "px"; │ │ │ │ │ + │ │ │ │ │ + node.coordorigin = scaledBox.left + " " + scaledBox.top; │ │ │ │ │ + node.coordsize = scaledBox.getWidth() + " " + scaledBox.getHeight(); │ │ │ │ │ } │ │ │ │ │ - this.events.destroy(); │ │ │ │ │ - this.eventListeners = null; │ │ │ │ │ - this.events = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Clear whatever is currently in the tile, then return whether or not │ │ │ │ │ - * it should actually be re-drawn. This is an example implementation │ │ │ │ │ - * that can be overridden by subclasses. The minimum thing to do here │ │ │ │ │ - * is to call <clear> and return the result from <shouldDraw>. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: dashStyle │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * force - {Boolean} If true, the tile will not be cleared and no beforedraw │ │ │ │ │ - * event will be fired. This is used for drawing tiles asynchronously │ │ │ │ │ - * after drawing has been cancelled by returning false from a beforedraw │ │ │ │ │ - * listener. │ │ │ │ │ + * style - {Object} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Whether or not the tile should actually be drawn. Returns null │ │ │ │ │ - * if a beforedraw listener returned false. │ │ │ │ │ + * {String} A VML compliant 'stroke-dasharray' value │ │ │ │ │ */ │ │ │ │ │ - draw: function(force) { │ │ │ │ │ - if (!force) { │ │ │ │ │ - //clear tile's contents and mark as not drawn │ │ │ │ │ - this.clear(); │ │ │ │ │ - } │ │ │ │ │ - var draw = this.shouldDraw(); │ │ │ │ │ - if (draw && !force && this.events.triggerEvent("beforedraw") === false) { │ │ │ │ │ - draw = null; │ │ │ │ │ + dashStyle: function(style) { │ │ │ │ │ + var dash = style.strokeDashstyle; │ │ │ │ │ + switch (dash) { │ │ │ │ │ + case 'solid': │ │ │ │ │ + case 'dot': │ │ │ │ │ + case 'dash': │ │ │ │ │ + case 'dashdot': │ │ │ │ │ + case 'longdash': │ │ │ │ │ + case 'longdashdot': │ │ │ │ │ + return dash; │ │ │ │ │ + default: │ │ │ │ │ + // very basic guessing of dash style patterns │ │ │ │ │ + var parts = dash.split(/[ ,]/); │ │ │ │ │ + if (parts.length == 2) { │ │ │ │ │ + if (1 * parts[0] >= 2 * parts[1]) { │ │ │ │ │ + return "longdash"; │ │ │ │ │ + } │ │ │ │ │ + return (parts[0] == 1 || parts[1] == 1) ? "dot" : "dash"; │ │ │ │ │ + } else if (parts.length == 4) { │ │ │ │ │ + return (1 * parts[0] >= 2 * parts[1]) ? "longdashdot" : │ │ │ │ │ + "dashdot"; │ │ │ │ │ + } │ │ │ │ │ + return "solid"; │ │ │ │ │ } │ │ │ │ │ - return draw; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: shouldDraw │ │ │ │ │ - * Return whether or not the tile should actually be (re-)drawn. The only │ │ │ │ │ - * case where we *wouldn't* want to draw the tile is if the tile is outside │ │ │ │ │ - * its layer's maxExtent │ │ │ │ │ - * │ │ │ │ │ + * Method: createNode │ │ │ │ │ + * Create a new node │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * type - {String} Kind of node to draw │ │ │ │ │ + * id - {String} Id for node │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Whether or not the tile should actually be drawn. │ │ │ │ │ + * {DOMElement} A new node of the given type and id │ │ │ │ │ */ │ │ │ │ │ - shouldDraw: function() { │ │ │ │ │ - var withinMaxExtent = false, │ │ │ │ │ - maxExtent = this.layer.maxExtent; │ │ │ │ │ - if (maxExtent) { │ │ │ │ │ - var map = this.layer.map; │ │ │ │ │ - var worldBounds = map.baseLayer.wrapDateLine && map.getMaxExtent(); │ │ │ │ │ - if (this.bounds.intersectsBounds(maxExtent, { │ │ │ │ │ - inclusive: false, │ │ │ │ │ - worldBounds: worldBounds │ │ │ │ │ - })) { │ │ │ │ │ - withinMaxExtent = true; │ │ │ │ │ - } │ │ │ │ │ + createNode: function(type, id) { │ │ │ │ │ + var node = document.createElement(type); │ │ │ │ │ + if (id) { │ │ │ │ │ + node.id = id; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return withinMaxExtent || this.layer.displayOutsideMaxExtent; │ │ │ │ │ + // IE hack to make elements unselectable, to prevent 'blue flash' │ │ │ │ │ + // while dragging vectors; #1410 │ │ │ │ │ + node.unselectable = 'on'; │ │ │ │ │ + node.onselectstart = OpenLayers.Function.False; │ │ │ │ │ + │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setBounds │ │ │ │ │ - * Sets the bounds on this instance │ │ │ │ │ + * Method: nodeTypeCompare │ │ │ │ │ + * Determine whether a node is of a given type │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds {<OpenLayers.Bounds>} │ │ │ │ │ - */ │ │ │ │ │ - setBounds: function(bounds) { │ │ │ │ │ - bounds = bounds.clone(); │ │ │ │ │ - if (this.layer.map.baseLayer.wrapDateLine) { │ │ │ │ │ - var worldExtent = this.layer.map.getMaxExtent(), │ │ │ │ │ - tolerance = this.layer.map.getResolution(); │ │ │ │ │ - bounds = bounds.wrapDateLine(worldExtent, { │ │ │ │ │ - leftTolerance: tolerance, │ │ │ │ │ - rightTolerance: tolerance │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - this.bounds = bounds; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * Reposition the tile. │ │ │ │ │ + * node - {DOMElement} An VML element │ │ │ │ │ + * type - {String} Kind of node │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * position - {<OpenLayers.Pixel>} │ │ │ │ │ - * redraw - {Boolean} Call draw method on tile after moving. │ │ │ │ │ - * Default is true │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ */ │ │ │ │ │ - moveTo: function(bounds, position, redraw) { │ │ │ │ │ - if (redraw == null) { │ │ │ │ │ - redraw = true; │ │ │ │ │ + nodeTypeCompare: function(node, type) { │ │ │ │ │ + │ │ │ │ │ + //split type │ │ │ │ │ + var subType = type; │ │ │ │ │ + var splitIndex = subType.indexOf(":"); │ │ │ │ │ + if (splitIndex != -1) { │ │ │ │ │ + subType = subType.substr(splitIndex + 1); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - this.setBounds(bounds); │ │ │ │ │ - this.position = position.clone(); │ │ │ │ │ - if (redraw) { │ │ │ │ │ - this.draw(); │ │ │ │ │ + //split nodeName │ │ │ │ │ + var nodeName = node.nodeName; │ │ │ │ │ + splitIndex = nodeName.indexOf(":"); │ │ │ │ │ + if (splitIndex != -1) { │ │ │ │ │ + nodeName = nodeName.substr(splitIndex + 1); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clear │ │ │ │ │ - * Clear the tile of any bounds/position-related data so that it can │ │ │ │ │ - * be reused in a new location. │ │ │ │ │ - */ │ │ │ │ │ - clear: function(draw) { │ │ │ │ │ - // to be extended by subclasses │ │ │ │ │ + return (subType == nodeName); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Tile" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Tile/Image.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Tile.js │ │ │ │ │ - * @requires OpenLayers/Animation.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Tile.Image │ │ │ │ │ - * Instances of OpenLayers.Tile.Image are used to manage the image tiles │ │ │ │ │ - * used by various layers. Create a new image tile with the │ │ │ │ │ - * <OpenLayers.Tile.Image> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Tile> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Tile.Image = OpenLayers.Class(OpenLayers.Tile, { │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} An events object that handles all │ │ │ │ │ - * events on the tile. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * tile.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ + * Method: createRenderRoot │ │ │ │ │ + * Create the renderer root │ │ │ │ │ * │ │ │ │ │ - * Supported event types (in addition to the <OpenLayers.Tile> events): │ │ │ │ │ - * beforeload - Triggered before an image is prepared for loading, when the │ │ │ │ │ - * url for the image is known already. Listeners may call <setImage> on │ │ │ │ │ - * the tile instance. If they do so, that image will be used and no new │ │ │ │ │ - * one will be created. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String} The URL of the image being requested. No default. Filled in by │ │ │ │ │ - * layer.getURL() function. May be modified by loadstart listeners. │ │ │ │ │ - */ │ │ │ │ │ - url: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: imgDiv │ │ │ │ │ - * {HTMLImageElement} The image for this tile. │ │ │ │ │ - */ │ │ │ │ │ - imgDiv: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: frame │ │ │ │ │ - * {DOMElement} The image element is appended to the frame. Any gutter on │ │ │ │ │ - * the image will be hidden behind the frame. If no gutter is set, │ │ │ │ │ - * this will be null. │ │ │ │ │ - */ │ │ │ │ │ - frame: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: imageReloadAttempts │ │ │ │ │ - * {Integer} Attempts to load the image. │ │ │ │ │ - */ │ │ │ │ │ - imageReloadAttempts: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layerAlphaHack │ │ │ │ │ - * {Boolean} True if the png alpha hack needs to be applied on the layer's div. │ │ │ │ │ - */ │ │ │ │ │ - layerAlphaHack: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: asyncRequestId │ │ │ │ │ - * {Integer} ID of an request to see if request is still valid. This is a │ │ │ │ │ - * number which increments by 1 for each asynchronous request. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The specific render engine's root element │ │ │ │ │ */ │ │ │ │ │ - asyncRequestId: null, │ │ │ │ │ + createRenderRoot: function() { │ │ │ │ │ + return this.nodeFactory(this.container.id + "_vmlRoot", "div"); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: maxGetUrlLength │ │ │ │ │ - * {Number} If set, requests that would result in GET urls with more │ │ │ │ │ - * characters than the number provided will be made using form-encoded │ │ │ │ │ - * HTTP POST. It is good practice to avoid urls that are longer than 2048 │ │ │ │ │ - * characters. │ │ │ │ │ + * Method: createRoot │ │ │ │ │ + * Create the main root element │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * suffix - {String} suffix to append to the id │ │ │ │ │ * │ │ │ │ │ - * Caution: │ │ │ │ │ - * Older versions of Gecko based browsers (e.g. Firefox < 3.5) and most │ │ │ │ │ - * Opera versions do not fully support this option. On all browsers, │ │ │ │ │ - * transition effects are not supported if POST requests are used. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - maxGetUrlLength: null, │ │ │ │ │ + createRoot: function(suffix) { │ │ │ │ │ + return this.nodeFactory(this.container.id + suffix, "olv:group"); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: canvasContext │ │ │ │ │ - * {CanvasRenderingContext2D} A canvas context associated with │ │ │ │ │ - * the tile image. │ │ │ │ │ - */ │ │ │ │ │ - canvasContext: null, │ │ │ │ │ + /************************************** │ │ │ │ │ + * * │ │ │ │ │ + * GEOMETRY DRAWING FUNCTIONS * │ │ │ │ │ + * * │ │ │ │ │ + **************************************/ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: crossOriginKeyword │ │ │ │ │ - * The value of the crossorigin keyword to use when loading images. This is │ │ │ │ │ - * only relevant when using <getCanvasContext> for tiles from remote │ │ │ │ │ - * origins and should be set to either 'anonymous' or 'use-credentials' │ │ │ │ │ - * for servers that send Access-Control-Allow-Origin headers with their │ │ │ │ │ - * tiles. │ │ │ │ │ - */ │ │ │ │ │ - crossOriginKeyword: null, │ │ │ │ │ - │ │ │ │ │ - /** TBD 3.0 - reorder the parameters to the init function to remove │ │ │ │ │ - * URL. the getUrl() function on the layer gets called on │ │ │ │ │ - * each draw(), so no need to specify it here. │ │ │ │ │ + * Method: drawPoint │ │ │ │ │ + * Render a point │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the point could not be drawn │ │ │ │ │ */ │ │ │ │ │ + drawPoint: function(node, geometry) { │ │ │ │ │ + return this.drawCircle(node, geometry, 1); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Tile.Image │ │ │ │ │ - * Constructor for a new <OpenLayers.Tile.Image> instance. │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawCircle │ │ │ │ │ + * Render a circle. │ │ │ │ │ + * Size and Center a circle given geometry (x,y center) and radius │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer>} layer that the tile will go in. │ │ │ │ │ - * position - {<OpenLayers.Pixel>} │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * url - {<String>} Deprecated. Remove me in 3.0. │ │ │ │ │ - * size - {<OpenLayers.Size>} │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * radius - {float} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the circle could not ne drawn │ │ │ │ │ */ │ │ │ │ │ - initialize: function(layer, position, bounds, url, size, options) { │ │ │ │ │ - OpenLayers.Tile.prototype.initialize.apply(this, arguments); │ │ │ │ │ + drawCircle: function(node, geometry, radius) { │ │ │ │ │ + if (!isNaN(geometry.x) && !isNaN(geometry.y)) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ │ │ │ │ │ - this.url = url; //deprecated remove me │ │ │ │ │ + node.style.left = ((((geometry.x - this.featureDx) / resolution - this.offset.x) | 0) - radius) + "px"; │ │ │ │ │ + node.style.top = (((geometry.y / resolution - this.offset.y) | 0) - radius) + "px"; │ │ │ │ │ │ │ │ │ │ - this.layerAlphaHack = this.layer.alpha && OpenLayers.Util.alphaHack(); │ │ │ │ │ + var diameter = radius * 2; │ │ │ │ │ │ │ │ │ │ - if (this.maxGetUrlLength != null || this.layer.gutter || this.layerAlphaHack) { │ │ │ │ │ - // only create frame if it's needed │ │ │ │ │ - this.frame = document.createElement("div"); │ │ │ │ │ - this.frame.style.position = "absolute"; │ │ │ │ │ - this.frame.style.overflow = "hidden"; │ │ │ │ │ - } │ │ │ │ │ - if (this.maxGetUrlLength != null) { │ │ │ │ │ - OpenLayers.Util.extend(this, OpenLayers.Tile.Image.IFrame); │ │ │ │ │ + node.style.width = diameter + "px"; │ │ │ │ │ + node.style.height = diameter + "px"; │ │ │ │ │ + return node; │ │ │ │ │ } │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * nullify references to prevent circular references and memory leaks │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.imgDiv) { │ │ │ │ │ - this.clear(); │ │ │ │ │ - this.imgDiv = null; │ │ │ │ │ - this.frame = null; │ │ │ │ │ - } │ │ │ │ │ - // don't handle async requests any more │ │ │ │ │ - this.asyncRequestId = null; │ │ │ │ │ - OpenLayers.Tile.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Check that a tile should be drawn, and draw it. │ │ │ │ │ + * Method: drawLineString │ │ │ │ │ + * Render a linestring. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Was a tile drawn? Or null if a beforedraw listener returned │ │ │ │ │ - * false. │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - var shouldDraw = OpenLayers.Tile.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (shouldDraw) { │ │ │ │ │ - // The layer's reproject option is deprecated. │ │ │ │ │ - if (this.layer != this.layer.map.baseLayer && this.layer.reproject) { │ │ │ │ │ - // getBoundsFromBaseLayer is defined in deprecated.js. │ │ │ │ │ - this.bounds = this.getBoundsFromBaseLayer(this.position); │ │ │ │ │ - } │ │ │ │ │ - if (this.isLoading) { │ │ │ │ │ - //if we're already loading, send 'reload' instead of 'loadstart'. │ │ │ │ │ - this._loadEvent = "reload"; │ │ │ │ │ - } else { │ │ │ │ │ - this.isLoading = true; │ │ │ │ │ - this._loadEvent = "loadstart"; │ │ │ │ │ - } │ │ │ │ │ - this.renderTile(); │ │ │ │ │ - this.positionTile(); │ │ │ │ │ - } else if (shouldDraw === false) { │ │ │ │ │ - this.unload(); │ │ │ │ │ - } │ │ │ │ │ - return shouldDraw; │ │ │ │ │ + drawLineString: function(node, geometry) { │ │ │ │ │ + return this.drawLine(node, geometry, false); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: renderTile │ │ │ │ │ - * Internal function to actually initialize the image tile, │ │ │ │ │ - * position it correctly, and set its url. │ │ │ │ │ + * Method: drawLinearRing │ │ │ │ │ + * Render a linearring │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - renderTile: function() { │ │ │ │ │ - if (this.layer.async) { │ │ │ │ │ - // Asynchronous image requests call the asynchronous getURL method │ │ │ │ │ - // on the layer to fetch an image that covers 'this.bounds'. │ │ │ │ │ - var id = this.asyncRequestId = (this.asyncRequestId || 0) + 1; │ │ │ │ │ - this.layer.getURLasync(this.bounds, function(url) { │ │ │ │ │ - if (id == this.asyncRequestId) { │ │ │ │ │ - this.url = url; │ │ │ │ │ - this.initImage(); │ │ │ │ │ - } │ │ │ │ │ - }, this); │ │ │ │ │ - } else { │ │ │ │ │ - // synchronous image requests get the url immediately. │ │ │ │ │ - this.url = this.layer.getURL(this.bounds); │ │ │ │ │ - this.initImage(); │ │ │ │ │ - } │ │ │ │ │ + drawLinearRing: function(node, geometry) { │ │ │ │ │ + return this.drawLine(node, geometry, true); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: positionTile │ │ │ │ │ - * Using the properties currenty set on the layer, position the tile correctly. │ │ │ │ │ - * This method is used both by the async and non-async versions of the Tile.Image │ │ │ │ │ - * code. │ │ │ │ │ + * Method: DrawLine │ │ │ │ │ + * Render a line. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * closeLine - {Boolean} Close the line? (make it a ring?) │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - positionTile: function() { │ │ │ │ │ - var style = this.getTile().style, │ │ │ │ │ - size = this.frame ? this.size : │ │ │ │ │ - this.layer.getImageSize(this.bounds), │ │ │ │ │ - ratio = 1; │ │ │ │ │ - if (this.layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ - ratio = this.layer.getServerResolution() / this.layer.map.getResolution(); │ │ │ │ │ - } │ │ │ │ │ - style.left = this.position.x + "px"; │ │ │ │ │ - style.top = this.position.y + "px"; │ │ │ │ │ - style.width = Math.round(ratio * size.w) + "px"; │ │ │ │ │ - style.height = Math.round(ratio * size.h) + "px"; │ │ │ │ │ - }, │ │ │ │ │ + drawLine: function(node, geometry, closeLine) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clear │ │ │ │ │ - * Remove the tile from the DOM, clear it of any image related data so that │ │ │ │ │ - * it can be reused in a new location. │ │ │ │ │ - */ │ │ │ │ │ - clear: function() { │ │ │ │ │ - OpenLayers.Tile.prototype.clear.apply(this, arguments); │ │ │ │ │ - var img = this.imgDiv; │ │ │ │ │ - if (img) { │ │ │ │ │ - var tile = this.getTile(); │ │ │ │ │ - if (tile.parentNode === this.layer.div) { │ │ │ │ │ - this.layer.div.removeChild(tile); │ │ │ │ │ - } │ │ │ │ │ - this.setImgSrc(); │ │ │ │ │ - if (this.layerAlphaHack === true) { │ │ │ │ │ - img.style.filter = ""; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Element.removeClass(img, "olImageLoadError"); │ │ │ │ │ + this.setNodeDimension(node, geometry); │ │ │ │ │ + │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var numComponents = geometry.components.length; │ │ │ │ │ + var parts = new Array(numComponents); │ │ │ │ │ + │ │ │ │ │ + var comp, x, y; │ │ │ │ │ + for (var i = 0; i < numComponents; i++) { │ │ │ │ │ + comp = geometry.components[i]; │ │ │ │ │ + x = ((comp.x - this.featureDx) / resolution - this.offset.x) | 0; │ │ │ │ │ + y = (comp.y / resolution - this.offset.y) | 0; │ │ │ │ │ + parts[i] = " " + x + "," + y + " l "; │ │ │ │ │ } │ │ │ │ │ - this.canvasContext = null; │ │ │ │ │ + var end = (closeLine) ? " x e" : " e"; │ │ │ │ │ + node.path = "m" + parts.join("") + end; │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getImage │ │ │ │ │ - * Returns or creates and returns the tile image. │ │ │ │ │ + * Method: drawPolygon │ │ │ │ │ + * Render a polygon │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - getImage: function() { │ │ │ │ │ - if (!this.imgDiv) { │ │ │ │ │ - this.imgDiv = OpenLayers.Tile.Image.IMAGE.cloneNode(false); │ │ │ │ │ + drawPolygon: function(node, geometry) { │ │ │ │ │ + this.setNodeDimension(node, geometry); │ │ │ │ │ │ │ │ │ │ - var style = this.imgDiv.style; │ │ │ │ │ - if (this.frame) { │ │ │ │ │ - var left = 0, │ │ │ │ │ - top = 0; │ │ │ │ │ - if (this.layer.gutter) { │ │ │ │ │ - left = this.layer.gutter / this.layer.tileSize.w * 100; │ │ │ │ │ - top = this.layer.gutter / this.layer.tileSize.h * 100; │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + │ │ │ │ │ + var path = []; │ │ │ │ │ + var j, jj, points, area, first, second, i, ii, comp, pathComp, x, y; │ │ │ │ │ + for (j = 0, jj = geometry.components.length; j < jj; j++) { │ │ │ │ │ + path.push("m"); │ │ │ │ │ + points = geometry.components[j].components; │ │ │ │ │ + // we only close paths of interior rings with area │ │ │ │ │ + area = (j === 0); │ │ │ │ │ + first = null; │ │ │ │ │ + second = null; │ │ │ │ │ + for (i = 0, ii = points.length; i < ii; i++) { │ │ │ │ │ + comp = points[i]; │ │ │ │ │ + x = ((comp.x - this.featureDx) / resolution - this.offset.x) | 0; │ │ │ │ │ + y = (comp.y / resolution - this.offset.y) | 0; │ │ │ │ │ + pathComp = " " + x + "," + y; │ │ │ │ │ + path.push(pathComp); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + path.push(" l"); │ │ │ │ │ + } │ │ │ │ │ + if (!area) { │ │ │ │ │ + // IE improperly renders sub-paths that have no area. │ │ │ │ │ + // Instead of checking the area of every ring, we confirm │ │ │ │ │ + // the ring has at least three distinct points. This does │ │ │ │ │ + // not catch all non-zero area cases, but it greatly improves │ │ │ │ │ + // interior ring digitizing and is a minor performance hit │ │ │ │ │ + // when rendering rings with many points. │ │ │ │ │ + if (!first) { │ │ │ │ │ + first = pathComp; │ │ │ │ │ + } else if (first != pathComp) { │ │ │ │ │ + if (!second) { │ │ │ │ │ + second = pathComp; │ │ │ │ │ + } else if (second != pathComp) { │ │ │ │ │ + // stop looking │ │ │ │ │ + area = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - style.left = -left + "%"; │ │ │ │ │ - style.top = -top + "%"; │ │ │ │ │ - style.width = (2 * left + 100) + "%"; │ │ │ │ │ - style.height = (2 * top + 100) + "%"; │ │ │ │ │ - } │ │ │ │ │ - style.visibility = "hidden"; │ │ │ │ │ - style.opacity = 0; │ │ │ │ │ - if (this.layer.opacity < 1) { │ │ │ │ │ - style.filter = 'alpha(opacity=' + │ │ │ │ │ - (this.layer.opacity * 100) + │ │ │ │ │ - ')'; │ │ │ │ │ - } │ │ │ │ │ - style.position = "absolute"; │ │ │ │ │ - if (this.layerAlphaHack) { │ │ │ │ │ - // move the image out of sight │ │ │ │ │ - style.paddingTop = style.height; │ │ │ │ │ - style.height = "0"; │ │ │ │ │ - style.width = "100%"; │ │ │ │ │ - } │ │ │ │ │ - if (this.frame) { │ │ │ │ │ - this.frame.appendChild(this.imgDiv); │ │ │ │ │ } │ │ │ │ │ + path.push(area ? " x " : " "); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return this.imgDiv; │ │ │ │ │ + path.push("e"); │ │ │ │ │ + node.path = path.join(""); │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setImage │ │ │ │ │ - * Sets the image element for this tile. This method should only be called │ │ │ │ │ - * from beforeload listeners. │ │ │ │ │ - * │ │ │ │ │ - * Parameters │ │ │ │ │ - * img - {HTMLImageElement} The image to use for this tile. │ │ │ │ │ + * Method: drawRectangle │ │ │ │ │ + * Render a rectangle │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - setImage: function(img) { │ │ │ │ │ - this.imgDiv = img; │ │ │ │ │ + drawRectangle: function(node, geometry) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + │ │ │ │ │ + node.style.left = (((geometry.x - this.featureDx) / resolution - this.offset.x) | 0) + "px"; │ │ │ │ │ + node.style.top = ((geometry.y / resolution - this.offset.y) | 0) + "px"; │ │ │ │ │ + node.style.width = ((geometry.width / resolution) | 0) + "px"; │ │ │ │ │ + node.style.height = ((geometry.height / resolution) | 0) + "px"; │ │ │ │ │ + │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: initImage │ │ │ │ │ - * Creates the content for the frame on the tile. │ │ │ │ │ + * Method: drawText │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * style - │ │ │ │ │ + * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ */ │ │ │ │ │ - initImage: function() { │ │ │ │ │ - if (!this.url && !this.imgDiv) { │ │ │ │ │ - // fast path out - if there is no tile url and no previous image │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - return; │ │ │ │ │ + drawText: function(featureId, style, location) { │ │ │ │ │ + var label = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX, "olv:rect"); │ │ │ │ │ + var textbox = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX + "_textbox", "olv:textbox"); │ │ │ │ │ + │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + label.style.left = (((location.x - this.featureDx) / resolution - this.offset.x) | 0) + "px"; │ │ │ │ │ + label.style.top = ((location.y / resolution - this.offset.y) | 0) + "px"; │ │ │ │ │ + label.style.flip = "y"; │ │ │ │ │ + │ │ │ │ │ + textbox.innerText = style.label; │ │ │ │ │ + │ │ │ │ │ + if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ + textbox.style.cursor = style.cursor; │ │ │ │ │ } │ │ │ │ │ - this.events.triggerEvent('beforeload'); │ │ │ │ │ - this.layer.div.appendChild(this.getTile()); │ │ │ │ │ - this.events.triggerEvent(this._loadEvent); │ │ │ │ │ - var img = this.getImage(); │ │ │ │ │ - var src = img.getAttribute('src') || ''; │ │ │ │ │ - if (this.url && OpenLayers.Util.isEquivalentUrl(src, this.url)) { │ │ │ │ │ - this._loadTimeout = window.setTimeout( │ │ │ │ │ - OpenLayers.Function.bind(this.onImageLoad, this), 0 │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - this.stopLoading(); │ │ │ │ │ - if (this.crossOriginKeyword) { │ │ │ │ │ - img.removeAttribute("crossorigin"); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.observe(img, "load", │ │ │ │ │ - OpenLayers.Function.bind(this.onImageLoad, this) │ │ │ │ │ - ); │ │ │ │ │ - OpenLayers.Event.observe(img, "error", │ │ │ │ │ - OpenLayers.Function.bind(this.onImageError, this) │ │ │ │ │ - ); │ │ │ │ │ - this.imageReloadAttempts = 0; │ │ │ │ │ - this.setImgSrc(this.url); │ │ │ │ │ + if (style.fontColor) { │ │ │ │ │ + textbox.style.color = style.fontColor; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ + if (style.fontOpacity) { │ │ │ │ │ + textbox.style.filter = 'alpha(opacity=' + (style.fontOpacity * 100) + ')'; │ │ │ │ │ + } │ │ │ │ │ + if (style.fontFamily) { │ │ │ │ │ + textbox.style.fontFamily = style.fontFamily; │ │ │ │ │ + } │ │ │ │ │ + if (style.fontSize) { │ │ │ │ │ + textbox.style.fontSize = style.fontSize; │ │ │ │ │ + } │ │ │ │ │ + if (style.fontWeight) { │ │ │ │ │ + textbox.style.fontWeight = style.fontWeight; │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStyle) { │ │ │ │ │ + textbox.style.fontStyle = style.fontStyle; │ │ │ │ │ + } │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + label._featureId = featureId; │ │ │ │ │ + textbox._featureId = featureId; │ │ │ │ │ + textbox._geometry = location; │ │ │ │ │ + textbox._geometryClass = location.CLASS_NAME; │ │ │ │ │ + } │ │ │ │ │ + textbox.style.whiteSpace = "nowrap"; │ │ │ │ │ + // fun with IE: IE7 in standards compliant mode does not display any │ │ │ │ │ + // text with a left inset of 0. So we set this to 1px and subtract one │ │ │ │ │ + // pixel later when we set label.style.left │ │ │ │ │ + textbox.inset = "1px,0px,0px,0px"; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setImgSrc │ │ │ │ │ - * Sets the source for the tile image │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * url - {String} or undefined to hide the image │ │ │ │ │ - */ │ │ │ │ │ - setImgSrc: function(url) { │ │ │ │ │ - var img = this.imgDiv; │ │ │ │ │ - if (url) { │ │ │ │ │ - img.style.visibility = 'hidden'; │ │ │ │ │ - img.style.opacity = 0; │ │ │ │ │ - // don't set crossOrigin if the url is a data URL │ │ │ │ │ - if (this.crossOriginKeyword) { │ │ │ │ │ - if (url.substr(0, 5) !== 'data:') { │ │ │ │ │ - img.setAttribute("crossorigin", this.crossOriginKeyword); │ │ │ │ │ - } else { │ │ │ │ │ - img.removeAttribute("crossorigin"); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - img.src = url; │ │ │ │ │ - } else { │ │ │ │ │ - // Remove reference to the image, and leave it to the browser's │ │ │ │ │ - // caching and garbage collection. │ │ │ │ │ - this.stopLoading(); │ │ │ │ │ - this.imgDiv = null; │ │ │ │ │ - if (img.parentNode) { │ │ │ │ │ - img.parentNode.removeChild(img); │ │ │ │ │ - } │ │ │ │ │ + if (!label.parentNode) { │ │ │ │ │ + label.appendChild(textbox); │ │ │ │ │ + this.textRoot.appendChild(label); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getTile │ │ │ │ │ - * Get the tile's markup. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The tile's markup │ │ │ │ │ - */ │ │ │ │ │ - getTile: function() { │ │ │ │ │ - return this.frame ? this.frame : this.getImage(); │ │ │ │ │ + var align = style.labelAlign || "cm"; │ │ │ │ │ + if (align.length == 1) { │ │ │ │ │ + align += "m"; │ │ │ │ │ + } │ │ │ │ │ + var xshift = textbox.clientWidth * │ │ │ │ │ + (OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(0, 1)]); │ │ │ │ │ + var yshift = textbox.clientHeight * │ │ │ │ │ + (OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(1, 1)]); │ │ │ │ │ + label.style.left = parseInt(label.style.left) - xshift - 1 + "px"; │ │ │ │ │ + label.style.top = parseInt(label.style.top) + yshift + "px"; │ │ │ │ │ + │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createBackBuffer │ │ │ │ │ - * Create a backbuffer for this tile. A backbuffer isn't exactly a clone │ │ │ │ │ - * of the tile's markup, because we want to avoid the reloading of the │ │ │ │ │ - * image. So we clone the frame, and steal the image from the tile. │ │ │ │ │ - * │ │ │ │ │ + * Method: moveRoot │ │ │ │ │ + * moves this renderer's root to a different renderer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * renderer - {<OpenLayers.Renderer>} target renderer for the moved root │ │ │ │ │ + * root - {DOMElement} optional root node. To be used when this renderer │ │ │ │ │ + * holds roots from multiple layers to tell this method which one to │ │ │ │ │ + * detach │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} The markup, or undefined if the tile has no image │ │ │ │ │ - * or if it's currently loading. │ │ │ │ │ + * {Boolean} true if successful, false otherwise │ │ │ │ │ */ │ │ │ │ │ - createBackBuffer: function() { │ │ │ │ │ - if (!this.imgDiv || this.isLoading) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var backBuffer; │ │ │ │ │ - if (this.frame) { │ │ │ │ │ - backBuffer = this.frame.cloneNode(false); │ │ │ │ │ - backBuffer.appendChild(this.imgDiv); │ │ │ │ │ - } else { │ │ │ │ │ - backBuffer = this.imgDiv; │ │ │ │ │ + moveRoot: function(renderer) { │ │ │ │ │ + var layer = this.map.getLayer(renderer.container.id); │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Vector.RootContainer) { │ │ │ │ │ + layer = this.map.getLayer(this.container.id); │ │ │ │ │ } │ │ │ │ │ - this.imgDiv = null; │ │ │ │ │ - return backBuffer; │ │ │ │ │ + layer && layer.renderer.clear(); │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.moveRoot.apply(this, arguments); │ │ │ │ │ + layer && layer.redraw(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onImageLoad │ │ │ │ │ - * Handler for the image onload event │ │ │ │ │ + * Method: importSymbol │ │ │ │ │ + * add a new symbol definition from the rendererer's symbol hash │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * graphicName - {String} name of the symbol to import │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} - hash of {DOMElement} "symbol" and {Number} "size" │ │ │ │ │ */ │ │ │ │ │ - onImageLoad: function() { │ │ │ │ │ - var img = this.imgDiv; │ │ │ │ │ - this.stopLoading(); │ │ │ │ │ - img.style.visibility = 'inherit'; │ │ │ │ │ - img.style.opacity = this.layer.opacity; │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - this.canvasContext = null; │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ + importSymbol: function(graphicName) { │ │ │ │ │ + var id = this.container.id + "-" + graphicName; │ │ │ │ │ │ │ │ │ │ - if (this.layerAlphaHack === true) { │ │ │ │ │ - img.style.filter = │ │ │ │ │ - "progid:DXImageTransform.Microsoft.AlphaImageLoader(src='" + │ │ │ │ │ - img.src + "', sizingMethod='scale')"; │ │ │ │ │ + // check if symbol already exists in the cache │ │ │ │ │ + var cache = this.symbolCache[id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + return cache; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: onImageError │ │ │ │ │ - * Handler for the image onerror event │ │ │ │ │ - */ │ │ │ │ │ - onImageError: function() { │ │ │ │ │ - var img = this.imgDiv; │ │ │ │ │ - if (img.src != null) { │ │ │ │ │ - this.imageReloadAttempts++; │ │ │ │ │ - if (this.imageReloadAttempts <= OpenLayers.IMAGE_RELOAD_ATTEMPTS) { │ │ │ │ │ - this.setImgSrc(this.layer.getURL(this.bounds)); │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Element.addClass(img, "olImageLoadError"); │ │ │ │ │ - this.events.triggerEvent("loaderror"); │ │ │ │ │ - this.onImageLoad(); │ │ │ │ │ - } │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(graphicName + ' is not a valid symbol name'); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: stopLoading │ │ │ │ │ - * Stops a loading sequence so <onImageLoad> won't be executed. │ │ │ │ │ - */ │ │ │ │ │ - stopLoading: function() { │ │ │ │ │ - OpenLayers.Event.stopObservingElement(this.imgDiv); │ │ │ │ │ - window.clearTimeout(this._loadTimeout); │ │ │ │ │ - delete this._loadTimeout; │ │ │ │ │ - }, │ │ │ │ │ + var symbolExtent = new OpenLayers.Bounds( │ │ │ │ │ + Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getCanvasContext │ │ │ │ │ - * Returns a canvas context associated with the tile image (with │ │ │ │ │ - * the image drawn on it). │ │ │ │ │ - * Returns undefined if the browser does not support canvas, if │ │ │ │ │ - * the tile has no image or if it's currently loading. │ │ │ │ │ - * │ │ │ │ │ - * The function returns a canvas context instance but the │ │ │ │ │ - * underlying canvas is still available in the 'canvas' property: │ │ │ │ │ - * (code) │ │ │ │ │ - * var context = tile.getCanvasContext(); │ │ │ │ │ - * if (context) { │ │ │ │ │ - * var data = context.canvas.toDataURL('image/jpeg'); │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - getCanvasContext: function() { │ │ │ │ │ - if (OpenLayers.CANVAS_SUPPORTED && this.imgDiv && !this.isLoading) { │ │ │ │ │ - if (!this.canvasContext) { │ │ │ │ │ - var canvas = document.createElement("canvas"); │ │ │ │ │ - canvas.width = this.size.w; │ │ │ │ │ - canvas.height = this.size.h; │ │ │ │ │ - this.canvasContext = canvas.getContext("2d"); │ │ │ │ │ - this.canvasContext.drawImage(this.imgDiv, 0, 0); │ │ │ │ │ + var pathitems = ["m"]; │ │ │ │ │ + for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + var x = symbol[i]; │ │ │ │ │ + var y = symbol[i + 1]; │ │ │ │ │ + symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ + symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ + symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ + symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ + │ │ │ │ │ + pathitems.push(x); │ │ │ │ │ + pathitems.push(y); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + pathitems.push("l"); │ │ │ │ │ } │ │ │ │ │ - return this.canvasContext; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ + pathitems.push("x e"); │ │ │ │ │ + var path = pathitems.join(" "); │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Tile.Image" │ │ │ │ │ + var diff = (symbolExtent.getWidth() - symbolExtent.getHeight()) / 2; │ │ │ │ │ + if (diff > 0) { │ │ │ │ │ + symbolExtent.bottom = symbolExtent.bottom - diff; │ │ │ │ │ + symbolExtent.top = symbolExtent.top + diff; │ │ │ │ │ + } else { │ │ │ │ │ + symbolExtent.left = symbolExtent.left + diff; │ │ │ │ │ + symbolExtent.right = symbolExtent.right - diff; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + cache = { │ │ │ │ │ + path: path, │ │ │ │ │ + size: symbolExtent.getWidth(), // equals getHeight() now │ │ │ │ │ + left: symbolExtent.left, │ │ │ │ │ + bottom: symbolExtent.bottom │ │ │ │ │ + }; │ │ │ │ │ + this.symbolCache[id] = cache; │ │ │ │ │ │ │ │ │ │ + return cache; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.VML" │ │ │ │ │ }); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Tile.Image.IMAGE │ │ │ │ │ - * {HTMLImageElement} The image for a tile. │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.VML.LABEL_SHIFT │ │ │ │ │ + * {Object} │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Tile.Image.IMAGE = (function() { │ │ │ │ │ - var img = new Image(); │ │ │ │ │ - img.className = "olTileImage"; │ │ │ │ │ - // avoid image gallery menu in IE6 │ │ │ │ │ - img.galleryImg = "no"; │ │ │ │ │ - return img; │ │ │ │ │ -}()); │ │ │ │ │ - │ │ │ │ │ +OpenLayers.Renderer.VML.LABEL_SHIFT = { │ │ │ │ │ + "l": 0, │ │ │ │ │ + "c": .5, │ │ │ │ │ + "r": 1, │ │ │ │ │ + "t": 0, │ │ │ │ │ + "m": .5, │ │ │ │ │ + "b": 1 │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Grid.js │ │ │ │ │ + OpenLayers/Renderer/SVG.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/HTTPRequest.js │ │ │ │ │ - * @requires OpenLayers/Tile/Image.js │ │ │ │ │ + * @requires OpenLayers/Renderer/Elements.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.Grid │ │ │ │ │ - * Base class for layers that use a lattice of tiles. Create a new grid │ │ │ │ │ - * layer with the <OpenLayers.Layer.Grid> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.HTTPRequest> │ │ │ │ │ + * Class: OpenLayers.Renderer.SVG │ │ │ │ │ + * │ │ │ │ │ + * Inherits: │ │ │ │ │ + * - <OpenLayers.Renderer.Elements> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Grid = OpenLayers.Class(OpenLayers.Layer.HTTPRequest, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: tileSize │ │ │ │ │ - * {<OpenLayers.Size>} │ │ │ │ │ - */ │ │ │ │ │ - tileSize: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: tileOriginCorner │ │ │ │ │ - * {String} If the <tileOrigin> property is not provided, the tile origin │ │ │ │ │ - * will be derived from the layer's <maxExtent>. The corner of the │ │ │ │ │ - * <maxExtent> used is determined by this property. Acceptable values │ │ │ │ │ - * are "tl" (top left), "tr" (top right), "bl" (bottom left), and "br" │ │ │ │ │ - * (bottom right). Default is "bl". │ │ │ │ │ - */ │ │ │ │ │ - tileOriginCorner: "bl", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: tileOrigin │ │ │ │ │ - * {<OpenLayers.LonLat>} Optional origin for aligning the grid of tiles. │ │ │ │ │ - * If provided, requests for tiles at all resolutions will be aligned │ │ │ │ │ - * with this location (no tiles shall overlap this location). If │ │ │ │ │ - * not provided, the grid of tiles will be aligned with the layer's │ │ │ │ │ - * <maxExtent>. Default is ``null``. │ │ │ │ │ - */ │ │ │ │ │ - tileOrigin: null, │ │ │ │ │ - │ │ │ │ │ - /** APIProperty: tileOptions │ │ │ │ │ - * {Object} optional configuration options for <OpenLayers.Tile> instances │ │ │ │ │ - * created by this Layer, if supported by the tile class. │ │ │ │ │ - */ │ │ │ │ │ - tileOptions: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: tileClass │ │ │ │ │ - * {<OpenLayers.Tile>} The tile class to use for this layer. │ │ │ │ │ - * Defaults is OpenLayers.Tile.Image. │ │ │ │ │ - */ │ │ │ │ │ - tileClass: OpenLayers.Tile.Image, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: grid │ │ │ │ │ - * {Array(Array(<OpenLayers.Tile>))} This is an array of rows, each row is │ │ │ │ │ - * an array of tiles. │ │ │ │ │ - */ │ │ │ │ │ - grid: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: singleTile │ │ │ │ │ - * {Boolean} Moves the layer into single-tile mode, meaning that one tile │ │ │ │ │ - * will be loaded. The tile's size will be determined by the 'ratio' │ │ │ │ │ - * property. When the tile is dragged such that it does not cover the │ │ │ │ │ - * entire viewport, it is reloaded. │ │ │ │ │ - */ │ │ │ │ │ - singleTile: false, │ │ │ │ │ - │ │ │ │ │ - /** APIProperty: ratio │ │ │ │ │ - * {Float} Used only when in single-tile mode, this specifies the │ │ │ │ │ - * ratio of the size of the single tile to the size of the map. │ │ │ │ │ - * Default value is 1.5. │ │ │ │ │ - */ │ │ │ │ │ - ratio: 1.5, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: buffer │ │ │ │ │ - * {Integer} Used only when in gridded mode, this specifies the number of │ │ │ │ │ - * extra rows and colums of tiles on each side which will │ │ │ │ │ - * surround the minimum grid tiles to cover the map. │ │ │ │ │ - * For very slow loading layers, a larger value may increase │ │ │ │ │ - * performance somewhat when dragging, but will increase bandwidth │ │ │ │ │ - * use significantly. │ │ │ │ │ - */ │ │ │ │ │ - buffer: 0, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: transitionEffect │ │ │ │ │ - * {String} The transition effect to use when the map is zoomed. │ │ │ │ │ - * Two posible values: │ │ │ │ │ - * │ │ │ │ │ - * "resize" - Existing tiles are resized on zoom to provide a visual │ │ │ │ │ - * effect of the zoom having taken place immediately. As the │ │ │ │ │ - * new tiles become available, they are drawn on top of the │ │ │ │ │ - * resized tiles (this is the default setting). │ │ │ │ │ - * "map-resize" - Existing tiles are resized on zoom and placed below the │ │ │ │ │ - * base layer. New tiles for the base layer will cover existing tiles. │ │ │ │ │ - * This setting is recommended when having an overlay duplicated during │ │ │ │ │ - * the transition is undesirable (e.g. street labels or big transparent │ │ │ │ │ - * fills). │ │ │ │ │ - * null - No transition effect. │ │ │ │ │ - * │ │ │ │ │ - * Using "resize" on non-opaque layers can cause undesired visual │ │ │ │ │ - * effects. Set transitionEffect to null in this case. │ │ │ │ │ - */ │ │ │ │ │ - transitionEffect: "resize", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: numLoadingTiles │ │ │ │ │ - * {Integer} How many tiles are still loading? │ │ │ │ │ - */ │ │ │ │ │ - numLoadingTiles: 0, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: serverResolutions │ │ │ │ │ - * {Array(Number}} This property is documented in subclasses as │ │ │ │ │ - * an API property. │ │ │ │ │ - */ │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: loading │ │ │ │ │ - * {Boolean} Indicates if tiles are being loaded. │ │ │ │ │ - */ │ │ │ │ │ - loading: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: backBuffer │ │ │ │ │ - * {DOMElement} The back buffer. │ │ │ │ │ - */ │ │ │ │ │ - backBuffer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: gridResolution │ │ │ │ │ - * {Number} The resolution of the current grid. Used for backbuffer and │ │ │ │ │ - * client zoom. This property is updated every time the grid is │ │ │ │ │ - * initialized. │ │ │ │ │ - */ │ │ │ │ │ - gridResolution: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: backBufferResolution │ │ │ │ │ - * {Number} The resolution of the current back buffer. This property is │ │ │ │ │ - * updated each time a back buffer is created. │ │ │ │ │ - */ │ │ │ │ │ - backBufferResolution: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: backBufferLonLat │ │ │ │ │ - * {Object} The top-left corner of the current back buffer. Includes lon │ │ │ │ │ - * and lat properties. This object is updated each time a back buffer │ │ │ │ │ - * is created. │ │ │ │ │ - */ │ │ │ │ │ - backBufferLonLat: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: backBufferTimerId │ │ │ │ │ - * {Number} The id of the back buffer timer. This timer is used to │ │ │ │ │ - * delay the removal of the back buffer, thereby preventing │ │ │ │ │ - * flash effects caused by tile animation. │ │ │ │ │ - */ │ │ │ │ │ - backBufferTimerId: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: removeBackBufferDelay │ │ │ │ │ - * {Number} Delay for removing the backbuffer when all tiles have finished │ │ │ │ │ - * loading. Can be set to 0 when no css opacity transitions for the │ │ │ │ │ - * olTileImage class are used. Default is 0 for <singleTile> layers, │ │ │ │ │ - * 2500 for tiled layers. See <className> for more information on │ │ │ │ │ - * tile animation. │ │ │ │ │ - */ │ │ │ │ │ - removeBackBufferDelay: null, │ │ │ │ │ +OpenLayers.Renderer.SVG = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: className │ │ │ │ │ - * {String} Name of the class added to the layer div. If not set in the │ │ │ │ │ - * options passed to the constructor then className defaults to │ │ │ │ │ - * "olLayerGridSingleTile" for single tile layers (see <singleTile>), │ │ │ │ │ - * and "olLayerGrid" for non single tile layers. │ │ │ │ │ - * │ │ │ │ │ - * Note: │ │ │ │ │ - * │ │ │ │ │ - * The displaying of tiles is not animated by default for single tile │ │ │ │ │ - * layers - OpenLayers' default theme (style.css) includes this: │ │ │ │ │ - * (code) │ │ │ │ │ - * .olLayerGrid .olTileImage { │ │ │ │ │ - * -webkit-transition: opacity 0.2s linear; │ │ │ │ │ - * -moz-transition: opacity 0.2s linear; │ │ │ │ │ - * -o-transition: opacity 0.2s linear; │ │ │ │ │ - * transition: opacity 0.2s linear; │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ - * To animate tile displaying for any grid layer the following │ │ │ │ │ - * CSS rule can be used: │ │ │ │ │ - * (code) │ │ │ │ │ - * .olTileImage { │ │ │ │ │ - * -webkit-transition: opacity 0.2s linear; │ │ │ │ │ - * -moz-transition: opacity 0.2s linear; │ │ │ │ │ - * -o-transition: opacity 0.2s linear; │ │ │ │ │ - * transition: opacity 0.2s linear; │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ - * In that case, to avoid flash effects, <removeBackBufferDelay> │ │ │ │ │ - * should not be zero. │ │ │ │ │ + /** │ │ │ │ │ + * Property: xmlns │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - className: null, │ │ │ │ │ + xmlns: "http://www.w3.org/2000/svg", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * layer.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Listeners will be called with a reference to an event object. The │ │ │ │ │ - * properties of this event depends on exactly what happened. │ │ │ │ │ - * │ │ │ │ │ - * All event objects have at least the following properties: │ │ │ │ │ - * object - {Object} A reference to layer.events.object. │ │ │ │ │ - * element - {DOMElement} A reference to layer.events.element. │ │ │ │ │ - * │ │ │ │ │ - * Supported event types: │ │ │ │ │ - * addtile - Triggered when a tile is added to this layer. Listeners receive │ │ │ │ │ - * an object as first argument, which has a tile property that │ │ │ │ │ - * references the tile that has been added. │ │ │ │ │ - * tileloadstart - Triggered when a tile starts loading. Listeners receive │ │ │ │ │ - * an object as first argument, which has a tile property that │ │ │ │ │ - * references the tile that starts loading. │ │ │ │ │ - * tileloaded - Triggered when each new tile is │ │ │ │ │ - * loaded, as a means of progress update to listeners. │ │ │ │ │ - * listeners can access 'numLoadingTiles' if they wish to keep │ │ │ │ │ - * track of the loading progress. Listeners are called with an object │ │ │ │ │ - * with a 'tile' property as first argument, making the loaded tile │ │ │ │ │ - * available to the listener, and an 'aborted' property, which will be │ │ │ │ │ - * true when loading was aborted and no tile data is available. │ │ │ │ │ - * tileerror - Triggered before the tileloaded event (i.e. when the tile is │ │ │ │ │ - * still hidden) if a tile failed to load. Listeners receive an object │ │ │ │ │ - * as first argument, which has a tile property that references the │ │ │ │ │ - * tile that could not be loaded. │ │ │ │ │ - * retile - Triggered when the layer recreates its tile grid. │ │ │ │ │ + * Property: xlinkns │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ + xlinkns: "http://www.w3.org/1999/xlink", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: gridLayout │ │ │ │ │ - * {Object} Object containing properties tilelon, tilelat, startcol, │ │ │ │ │ - * startrow │ │ │ │ │ + * Constant: MAX_PIXEL │ │ │ │ │ + * {Integer} Firefox has a limitation where values larger or smaller than │ │ │ │ │ + * about 15000 in an SVG document lock the browser up. This │ │ │ │ │ + * works around it. │ │ │ │ │ */ │ │ │ │ │ - gridLayout: null, │ │ │ │ │ + MAX_PIXEL: 15000, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: rowSign │ │ │ │ │ - * {Number} 1 for grids starting at the top, -1 for grids starting at the │ │ │ │ │ - * bottom. This is used for several grid index and offset calculations. │ │ │ │ │ + * Property: translationParameters │ │ │ │ │ + * {Object} Hash with "x" and "y" properties │ │ │ │ │ */ │ │ │ │ │ - rowSign: null, │ │ │ │ │ + translationParameters: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: transitionendEvents │ │ │ │ │ - * {Array} Event names for transitionend │ │ │ │ │ + * Property: symbolMetrics │ │ │ │ │ + * {Object} Cache for symbol metrics according to their svg coordinate │ │ │ │ │ + * space. This is an object keyed by the symbol's id, and values are │ │ │ │ │ + * an array of [width, centerX, centerY]. │ │ │ │ │ */ │ │ │ │ │ - transitionendEvents: [ │ │ │ │ │ - 'transitionend', 'webkitTransitionEnd', 'otransitionend', │ │ │ │ │ - 'oTransitionEnd' │ │ │ │ │ - ], │ │ │ │ │ + symbolMetrics: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Grid │ │ │ │ │ - * Create a new grid layer │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Renderer.SVG │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * url - {String} │ │ │ │ │ - * params - {Object} │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + * containerID - {String} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.initialize.apply(this, │ │ │ │ │ + initialize: function(containerID) { │ │ │ │ │ + if (!this.supported()) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.initialize.apply(this, │ │ │ │ │ arguments); │ │ │ │ │ - this.grid = []; │ │ │ │ │ - this._removeBackBuffer = OpenLayers.Function.bind(this.removeBackBuffer, this); │ │ │ │ │ - │ │ │ │ │ - this.initProperties(); │ │ │ │ │ + this.translationParameters = { │ │ │ │ │ + x: 0, │ │ │ │ │ + y: 0 │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - this.rowSign = this.tileOriginCorner.substr(0, 1) === "t" ? 1 : -1; │ │ │ │ │ + this.symbolMetrics = {}; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: initProperties │ │ │ │ │ - * Set any properties that depend on the value of singleTile. │ │ │ │ │ - * Currently sets removeBackBufferDelay and className │ │ │ │ │ + * APIMethod: supported │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the browser supports the SVG renderer │ │ │ │ │ */ │ │ │ │ │ - initProperties: function() { │ │ │ │ │ - if (this.options.removeBackBufferDelay === undefined) { │ │ │ │ │ - this.removeBackBufferDelay = this.singleTile ? 0 : 2500; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.options.className === undefined) { │ │ │ │ │ - this.className = this.singleTile ? 'olLayerGridSingleTile' : │ │ │ │ │ - 'olLayerGrid'; │ │ │ │ │ - } │ │ │ │ │ + supported: function() { │ │ │ │ │ + var svgFeature = "http://www.w3.org/TR/SVG11/feature#"; │ │ │ │ │ + return (document.implementation && │ │ │ │ │ + (document.implementation.hasFeature("org.w3c.svg", "1.0") || │ │ │ │ │ + document.implementation.hasFeature(svgFeature + "SVG", "1.1") || │ │ │ │ │ + document.implementation.hasFeature(svgFeature + "BasicStructure", "1.1"))); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ + * Method: inValidRange │ │ │ │ │ + * See #669 for more information │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} The map. │ │ │ │ │ + * x - {Integer} │ │ │ │ │ + * y - {Integer} │ │ │ │ │ + * xyOnly - {Boolean} whether or not to just check for x and y, which means │ │ │ │ │ + * to not take the current translation parameters into account if true. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the 'x' and 'y' coordinates are in the │ │ │ │ │ + * valid range. │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.setMap.call(this, map); │ │ │ │ │ - OpenLayers.Element.addClass(this.div, this.className); │ │ │ │ │ + inValidRange: function(x, y, xyOnly) { │ │ │ │ │ + var left = x + (xyOnly ? 0 : this.translationParameters.x); │ │ │ │ │ + var top = y + (xyOnly ? 0 : this.translationParameters.y); │ │ │ │ │ + return (left >= -this.MAX_PIXEL && left <= this.MAX_PIXEL && │ │ │ │ │ + top >= -this.MAX_PIXEL && top <= this.MAX_PIXEL); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removeMap │ │ │ │ │ - * Called when the layer is removed from the map. │ │ │ │ │ - * │ │ │ │ │ + * Method: setExtent │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} The map. │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} │ │ │ │ │ + * resolutionChanged - {Boolean} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ + * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ + * False otherwise. │ │ │ │ │ */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - this.removeBackBuffer(); │ │ │ │ │ - }, │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Deconstruct the layer and clear the grid. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.removeBackBuffer(); │ │ │ │ │ - this.clearGrid(); │ │ │ │ │ + var resolution = this.getResolution(), │ │ │ │ │ + left = -extent.left / resolution, │ │ │ │ │ + top = extent.top / resolution; │ │ │ │ │ │ │ │ │ │ - this.grid = null; │ │ │ │ │ - this.tileSize = null; │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.destroy.apply(this, arguments); │ │ │ │ │ + // If the resolution has changed, start over changing the corner, because │ │ │ │ │ + // the features will redraw. │ │ │ │ │ + if (resolutionChanged) { │ │ │ │ │ + this.left = left; │ │ │ │ │ + this.top = top; │ │ │ │ │ + // Set the viewbox │ │ │ │ │ + var extentString = "0 0 " + this.size.w + " " + this.size.h; │ │ │ │ │ + │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "viewBox", extentString); │ │ │ │ │ + this.translate(this.xOffset, 0); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + var inRange = this.translate(left - this.left + this.xOffset, top - this.top); │ │ │ │ │ + if (!inRange) { │ │ │ │ │ + // recenter the coordinate system │ │ │ │ │ + this.setExtent(extent, true); │ │ │ │ │ + } │ │ │ │ │ + return coordSysUnchanged && inRange; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: mergeNewParams │ │ │ │ │ - * Refetches tiles with new params merged, keeping a backbuffer. Each │ │ │ │ │ - * loading new tile will have a css class of '.olTileReplacing'. If a │ │ │ │ │ - * stylesheet applies a 'display: none' style to that class, any fade-in │ │ │ │ │ - * transition will not apply, and backbuffers for each tile will be removed │ │ │ │ │ - * as soon as the tile is loaded. │ │ │ │ │ + * Method: translate │ │ │ │ │ + * Transforms the SVG coordinate system │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * newParams - {Object} │ │ │ │ │ - * │ │ │ │ │ + * x - {Float} │ │ │ │ │ + * y - {Float} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * redrawn: {Boolean} whether the layer was actually redrawn. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clearGrid │ │ │ │ │ - * Go through and remove all tiles from the grid, calling │ │ │ │ │ - * destroy() on each of them to kill circular references │ │ │ │ │ + * {Boolean} true if the translation parameters are in the valid coordinates │ │ │ │ │ + * range, false otherwise. │ │ │ │ │ */ │ │ │ │ │ - clearGrid: function() { │ │ │ │ │ - if (this.grid) { │ │ │ │ │ - for (var iRow = 0, len = this.grid.length; iRow < len; iRow++) { │ │ │ │ │ - var row = this.grid[iRow]; │ │ │ │ │ - for (var iCol = 0, clen = row.length; iCol < clen; iCol++) { │ │ │ │ │ - var tile = row[iCol]; │ │ │ │ │ - this.destroyTile(tile); │ │ │ │ │ - } │ │ │ │ │ + translate: function(x, y) { │ │ │ │ │ + if (!this.inValidRange(x, y, true)) { │ │ │ │ │ + return false; │ │ │ │ │ + } else { │ │ │ │ │ + var transformString = ""; │ │ │ │ │ + if (x || y) { │ │ │ │ │ + transformString = "translate(" + x + "," + y + ")"; │ │ │ │ │ } │ │ │ │ │ - this.grid = []; │ │ │ │ │ - this.gridResolution = null; │ │ │ │ │ - this.gridLayout = null; │ │ │ │ │ + this.root.setAttributeNS(null, "transform", transformString); │ │ │ │ │ + this.translationParameters = { │ │ │ │ │ + x: x, │ │ │ │ │ + y: y │ │ │ │ │ + }; │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: addOptions │ │ │ │ │ + * Method: setSize │ │ │ │ │ + * Sets the size of the drawing surface. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * newOptions - {Object} │ │ │ │ │ - * reinitialize - {Boolean} If set to true, and if resolution options of the │ │ │ │ │ - * current baseLayer were changed, the map will be recentered to make │ │ │ │ │ - * sure that it is displayed with a valid resolution, and a │ │ │ │ │ - * changebaselayer event will be triggered. │ │ │ │ │ + * size - {<OpenLayers.Size>} The size of the drawing surface │ │ │ │ │ */ │ │ │ │ │ - addOptions: function(newOptions, reinitialize) { │ │ │ │ │ - var singleTileChanged = newOptions.singleTile !== undefined && │ │ │ │ │ - newOptions.singleTile !== this.singleTile; │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.addOptions.apply(this, arguments); │ │ │ │ │ - if (this.map && singleTileChanged) { │ │ │ │ │ - this.initProperties(); │ │ │ │ │ - this.clearGrid(); │ │ │ │ │ - this.tileSize = this.options.tileSize; │ │ │ │ │ - this.setTileSize(); │ │ │ │ │ - this.moveTo(null, true); │ │ │ │ │ - } │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "width", this.size.w); │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "height", this.size.h); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: getNodeType │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {Object} Is this ever used? │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Grid>} An exact clone of this OpenLayers.Layer.Grid │ │ │ │ │ + * {String} The corresponding node type for the specified geometry │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Grid(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.params, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.HTTPRequest.prototype.clone.apply(this, [obj]); │ │ │ │ │ - │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - if (this.tileSize != null) { │ │ │ │ │ - obj.tileSize = this.tileSize.clone(); │ │ │ │ │ + getNodeType: function(geometry, style) { │ │ │ │ │ + var nodeType = null; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + nodeType = "image"; │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + nodeType = "svg"; │ │ │ │ │ + } else { │ │ │ │ │ + nodeType = "circle"; │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + nodeType = "rect"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + nodeType = "polyline"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + nodeType = "polygon"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + case "OpenLayers.Geometry.Curve": │ │ │ │ │ + nodeType = "path"; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // we do not want to copy reference to grid, so we make a new array │ │ │ │ │ - obj.grid = []; │ │ │ │ │ - obj.gridResolution = null; │ │ │ │ │ - // same for backbuffer │ │ │ │ │ - obj.backBuffer = null; │ │ │ │ │ - obj.backBufferTimerId = null; │ │ │ │ │ - obj.loading = false; │ │ │ │ │ - obj.numLoadingTiles = 0; │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ + return nodeType; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * This function is called whenever the map is moved. All the moving │ │ │ │ │ - * of actual 'tiles' is done by the map, but moveTo's role is to accept │ │ │ │ │ - * a bounds and make sure the data that that bounds requires is pre-loaded. │ │ │ │ │ + /** │ │ │ │ │ + * Method: setStyle │ │ │ │ │ + * Use to set all the style attributes to a SVG node. │ │ │ │ │ + * │ │ │ │ │ + * Takes care to adjust stroke width and point radius to be │ │ │ │ │ + * resolution-relative │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * zoomChanged - {Boolean} │ │ │ │ │ - * dragging - {Boolean} │ │ │ │ │ + * node - {SVGDomElement} An SVG element to decorate │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * options - {Object} Currently supported options include │ │ │ │ │ + * 'isFilled' {Boolean} and │ │ │ │ │ + * 'isStroked' {Boolean} │ │ │ │ │ */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - bounds = bounds || this.map.getExtent(); │ │ │ │ │ - │ │ │ │ │ - if (bounds != null) { │ │ │ │ │ - │ │ │ │ │ - // if grid is empty or zoom has changed, we *must* re-tile │ │ │ │ │ - var forceReTile = !this.grid.length || zoomChanged; │ │ │ │ │ - │ │ │ │ │ - // total bounds of the tiles │ │ │ │ │ - var tilesBounds = this.getTilesBounds(); │ │ │ │ │ - │ │ │ │ │ - // the new map resolution │ │ │ │ │ - var resolution = this.map.getResolution(); │ │ │ │ │ + setStyle: function(node, style, options) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + options = options || node._options; │ │ │ │ │ │ │ │ │ │ - // the server-supported resolution for the new map resolution │ │ │ │ │ - var serverResolution = this.getServerResolution(resolution); │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + node.setAttributeNS(null, "title", title); │ │ │ │ │ + //Standards-conformant SVG │ │ │ │ │ + // Prevent duplicate nodes. See issue https://github.com/openlayers/openlayers/issues/92 │ │ │ │ │ + var titleNode = node.getElementsByTagName("title"); │ │ │ │ │ + if (titleNode.length > 0) { │ │ │ │ │ + titleNode[0].firstChild.textContent = title; │ │ │ │ │ + } else { │ │ │ │ │ + var label = this.nodeFactory(null, "title"); │ │ │ │ │ + label.textContent = title; │ │ │ │ │ + node.appendChild(label); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ + var r = parseFloat(node.getAttributeNS(null, "r")); │ │ │ │ │ + var widthFactor = 1; │ │ │ │ │ + var pos; │ │ │ │ │ + if (node._geometryClass == "OpenLayers.Geometry.Point" && r) { │ │ │ │ │ + node.style.visibility = ""; │ │ │ │ │ + if (style.graphic === false) { │ │ │ │ │ + node.style.visibility = "hidden"; │ │ │ │ │ + } else if (style.externalGraphic) { │ │ │ │ │ + pos = this.getPosition(node); │ │ │ │ │ + if (style.graphicWidth && style.graphicHeight) { │ │ │ │ │ + node.setAttributeNS(null, "preserveAspectRatio", "none"); │ │ │ │ │ + } │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ + var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ + style.graphicXOffset : -(0.5 * width); │ │ │ │ │ + var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ + style.graphicYOffset : -(0.5 * height); │ │ │ │ │ │ │ │ │ │ - // We want to redraw whenever even the slightest part of the │ │ │ │ │ - // current bounds is not contained by our tile. │ │ │ │ │ - // (thus, we do not specify partial -- its default is false) │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ │ │ │ │ │ - if (forceReTile || │ │ │ │ │ - (!dragging && !tilesBounds.containsBounds(bounds))) { │ │ │ │ │ + node.setAttributeNS(null, "x", (pos.x + xOffset).toFixed()); │ │ │ │ │ + node.setAttributeNS(null, "y", (pos.y + yOffset).toFixed()); │ │ │ │ │ + node.setAttributeNS(null, "width", width); │ │ │ │ │ + node.setAttributeNS(null, "height", height); │ │ │ │ │ + node.setAttributeNS(this.xlinkns, "xlink:href", style.externalGraphic); │ │ │ │ │ + node.setAttributeNS(null, "style", "opacity: " + opacity); │ │ │ │ │ + node.onclick = OpenLayers.Event.preventDefault; │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + // the symbol viewBox is three times as large as the symbol │ │ │ │ │ + var offset = style.pointRadius * 3; │ │ │ │ │ + var size = offset * 2; │ │ │ │ │ + var src = this.importSymbol(style.graphicName); │ │ │ │ │ + pos = this.getPosition(node); │ │ │ │ │ + widthFactor = this.symbolMetrics[src.id][0] * 3 / size; │ │ │ │ │ │ │ │ │ │ - // In single tile mode with no transition effect, we insert │ │ │ │ │ - // a non-scaled backbuffer when the layer is moved. But if │ │ │ │ │ - // a zoom occurs right after a move, i.e. before the new │ │ │ │ │ - // image is received, we need to remove the backbuffer, or │ │ │ │ │ - // an ill-positioned image will be visible during the zoom │ │ │ │ │ - // transition. │ │ │ │ │ + // remove the node from the dom before we modify it. This │ │ │ │ │ + // prevents various rendering issues in Safari and FF │ │ │ │ │ + var parent = node.parentNode; │ │ │ │ │ + var nextSibling = node.nextSibling; │ │ │ │ │ + if (parent) { │ │ │ │ │ + parent.removeChild(node); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (zoomChanged && this.transitionEffect !== 'resize') { │ │ │ │ │ - this.removeBackBuffer(); │ │ │ │ │ - } │ │ │ │ │ + // The more appropriate way to implement this would be use/defs, │ │ │ │ │ + // but due to various issues in several browsers, it is safer to │ │ │ │ │ + // copy the symbols instead of referencing them. │ │ │ │ │ + // See e.g. ticket http://trac.osgeo.org/openlayers/ticket/2985 │ │ │ │ │ + // and this email thread │ │ │ │ │ + // http://osgeo-org.1803224.n2.nabble.com/Select-Control-Ctrl-click-on-Feature-with-a-graphicName-opens-new-browser-window-tc5846039.html │ │ │ │ │ + node.firstChild && node.removeChild(node.firstChild); │ │ │ │ │ + node.appendChild(src.firstChild.cloneNode(true)); │ │ │ │ │ + node.setAttributeNS(null, "viewBox", src.getAttributeNS(null, "viewBox")); │ │ │ │ │ │ │ │ │ │ - if (!zoomChanged || this.transitionEffect === 'resize') { │ │ │ │ │ - this.applyBackBuffer(resolution); │ │ │ │ │ - } │ │ │ │ │ + node.setAttributeNS(null, "width", size); │ │ │ │ │ + node.setAttributeNS(null, "height", size); │ │ │ │ │ + node.setAttributeNS(null, "x", pos.x - offset); │ │ │ │ │ + node.setAttributeNS(null, "y", pos.y - offset); │ │ │ │ │ │ │ │ │ │ - this.initSingleTile(bounds); │ │ │ │ │ + // now that the node has all its new properties, insert it │ │ │ │ │ + // back into the dom where it was │ │ │ │ │ + if (nextSibling) { │ │ │ │ │ + parent.insertBefore(node, nextSibling); │ │ │ │ │ + } else if (parent) { │ │ │ │ │ + parent.appendChild(node); │ │ │ │ │ } │ │ │ │ │ } else { │ │ │ │ │ + node.setAttributeNS(null, "r", style.pointRadius); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // if the bounds have changed such that they are not even │ │ │ │ │ - // *partially* contained by our tiles (e.g. when user has │ │ │ │ │ - // programmatically panned to the other side of the earth on │ │ │ │ │ - // zoom level 18), then moveGriddedTiles could potentially have │ │ │ │ │ - // to run through thousands of cycles, so we want to reTile │ │ │ │ │ - // instead (thus, partial true). │ │ │ │ │ - forceReTile = forceReTile || │ │ │ │ │ - !tilesBounds.intersectsBounds(bounds, { │ │ │ │ │ - worldBounds: this.map.baseLayer.wrapDateLine && │ │ │ │ │ - this.map.getMaxExtent() │ │ │ │ │ - }); │ │ │ │ │ + var rotation = style.rotation; │ │ │ │ │ │ │ │ │ │ - if (forceReTile) { │ │ │ │ │ - if (zoomChanged && (this.transitionEffect === 'resize' || │ │ │ │ │ - this.gridResolution === resolution)) { │ │ │ │ │ - this.applyBackBuffer(resolution); │ │ │ │ │ - } │ │ │ │ │ - this.initGriddedTiles(bounds); │ │ │ │ │ + if ((rotation !== undefined || node._rotation !== undefined) && pos) { │ │ │ │ │ + node._rotation = rotation; │ │ │ │ │ + rotation |= 0; │ │ │ │ │ + if (node.nodeName !== "svg") { │ │ │ │ │ + node.setAttributeNS(null, "transform", │ │ │ │ │ + "rotate(" + rotation + " " + pos.x + " " + │ │ │ │ │ + pos.y + ")"); │ │ │ │ │ } else { │ │ │ │ │ - this.moveGriddedTiles(); │ │ │ │ │ + var metrics = this.symbolMetrics[src.id]; │ │ │ │ │ + node.firstChild.setAttributeNS(null, "transform", "rotate(" + │ │ │ │ │ + rotation + " " + │ │ │ │ │ + metrics[1] + " " + │ │ │ │ │ + metrics[2] + ")"); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getTileData │ │ │ │ │ - * Given a map location, retrieve a tile and the pixel offset within that │ │ │ │ │ - * tile corresponding to the location. If there is not an existing │ │ │ │ │ - * tile in the grid that covers the given location, null will be │ │ │ │ │ - * returned. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * loc - {<OpenLayers.LonLat>} map location │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Object with the following properties: tile ({<OpenLayers.Tile>}), │ │ │ │ │ - * i ({Number} x-pixel offset from top left), and j ({Integer} y-pixel │ │ │ │ │ - * offset from top left). │ │ │ │ │ - */ │ │ │ │ │ - getTileData: function(loc) { │ │ │ │ │ - var data = null, │ │ │ │ │ - x = loc.lon, │ │ │ │ │ - y = loc.lat, │ │ │ │ │ - numRows = this.grid.length; │ │ │ │ │ + if (options.isFilled) { │ │ │ │ │ + node.setAttributeNS(null, "fill", style.fillColor); │ │ │ │ │ + node.setAttributeNS(null, "fill-opacity", style.fillOpacity); │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "fill", "none"); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (this.map && numRows) { │ │ │ │ │ - var res = this.map.getResolution(), │ │ │ │ │ - tileWidth = this.tileSize.w, │ │ │ │ │ - tileHeight = this.tileSize.h, │ │ │ │ │ - bounds = this.grid[0][0].bounds, │ │ │ │ │ - left = bounds.left, │ │ │ │ │ - top = bounds.top; │ │ │ │ │ + if (options.isStroked) { │ │ │ │ │ + node.setAttributeNS(null, "stroke", style.strokeColor); │ │ │ │ │ + node.setAttributeNS(null, "stroke-opacity", style.strokeOpacity); │ │ │ │ │ + node.setAttributeNS(null, "stroke-width", style.strokeWidth * widthFactor); │ │ │ │ │ + node.setAttributeNS(null, "stroke-linecap", style.strokeLinecap || "round"); │ │ │ │ │ + // Hard-coded linejoin for now, to make it look the same as in VML. │ │ │ │ │ + // There is no strokeLinejoin property yet for symbolizers. │ │ │ │ │ + node.setAttributeNS(null, "stroke-linejoin", "round"); │ │ │ │ │ + style.strokeDashstyle && node.setAttributeNS(null, │ │ │ │ │ + "stroke-dasharray", this.dashStyle(style, widthFactor)); │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "stroke", "none"); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (x < left) { │ │ │ │ │ - // deal with multiple worlds │ │ │ │ │ - if (this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - var worldWidth = this.map.getMaxExtent().getWidth(); │ │ │ │ │ - var worldsAway = Math.ceil((left - x) / worldWidth); │ │ │ │ │ - x += worldWidth * worldsAway; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // tile distance to location (fractional number of tiles); │ │ │ │ │ - var dtx = (x - left) / (res * tileWidth); │ │ │ │ │ - var dty = (top - y) / (res * tileHeight); │ │ │ │ │ - // index of tile in grid │ │ │ │ │ - var col = Math.floor(dtx); │ │ │ │ │ - var row = Math.floor(dty); │ │ │ │ │ - if (row >= 0 && row < numRows) { │ │ │ │ │ - var tile = this.grid[row][col]; │ │ │ │ │ - if (tile) { │ │ │ │ │ - data = { │ │ │ │ │ - tile: tile, │ │ │ │ │ - // pixel index within tile │ │ │ │ │ - i: Math.floor((dtx - col) * tileWidth), │ │ │ │ │ - j: Math.floor((dty - row) * tileHeight) │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + if (style.pointerEvents) { │ │ │ │ │ + node.setAttributeNS(null, "pointer-events", style.pointerEvents); │ │ │ │ │ } │ │ │ │ │ - return data; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroyTile │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * tile - {<OpenLayers.Tile>} │ │ │ │ │ - */ │ │ │ │ │ - destroyTile: function(tile) { │ │ │ │ │ - this.removeTileMonitoringHooks(tile); │ │ │ │ │ - tile.destroy(); │ │ │ │ │ + if (style.cursor != null) { │ │ │ │ │ + node.setAttributeNS(null, "cursor", style.cursor); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getServerResolution │ │ │ │ │ - * Return the closest server-supported resolution. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: dashStyle │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resolution - {Number} The base resolution. If undefined the │ │ │ │ │ - * map resolution is used. │ │ │ │ │ - * │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * widthFactor - {Number} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Number} The closest server resolution value. │ │ │ │ │ + * {String} A SVG compliant 'stroke-dasharray' value │ │ │ │ │ */ │ │ │ │ │ - getServerResolution: function(resolution) { │ │ │ │ │ - var distance = Number.POSITIVE_INFINITY; │ │ │ │ │ - resolution = resolution || this.map.getResolution(); │ │ │ │ │ - if (this.serverResolutions && │ │ │ │ │ - OpenLayers.Util.indexOf(this.serverResolutions, resolution) === -1) { │ │ │ │ │ - var i, newDistance, newResolution, serverResolution; │ │ │ │ │ - for (i = this.serverResolutions.length - 1; i >= 0; i--) { │ │ │ │ │ - newResolution = this.serverResolutions[i]; │ │ │ │ │ - newDistance = Math.abs(newResolution - resolution); │ │ │ │ │ - if (newDistance > distance) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - distance = newDistance; │ │ │ │ │ - serverResolution = newResolution; │ │ │ │ │ - } │ │ │ │ │ - resolution = serverResolution; │ │ │ │ │ + dashStyle: function(style, widthFactor) { │ │ │ │ │ + var w = style.strokeWidth * widthFactor; │ │ │ │ │ + var str = style.strokeDashstyle; │ │ │ │ │ + switch (str) { │ │ │ │ │ + case 'solid': │ │ │ │ │ + return 'none'; │ │ │ │ │ + case 'dot': │ │ │ │ │ + return [1, 4 * w].join(); │ │ │ │ │ + case 'dash': │ │ │ │ │ + return [4 * w, 4 * w].join(); │ │ │ │ │ + case 'dashdot': │ │ │ │ │ + return [4 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ + case 'longdash': │ │ │ │ │ + return [8 * w, 4 * w].join(); │ │ │ │ │ + case 'longdashdot': │ │ │ │ │ + return [8 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ + default: │ │ │ │ │ + return OpenLayers.String.trim(str).replace(/\s+/g, ","); │ │ │ │ │ } │ │ │ │ │ - return resolution; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getServerZoom │ │ │ │ │ - * Return the zoom value corresponding to the best matching server │ │ │ │ │ - * resolution, taking into account <serverResolutions> and <zoomOffset>. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: createNode │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * type - {String} Kind of node to draw │ │ │ │ │ + * id - {String} Id for node │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Number} The closest server supported zoom. This is not the map zoom │ │ │ │ │ - * level, but an index of the server's resolutions array. │ │ │ │ │ + * {DOMElement} A new node of the given type and id │ │ │ │ │ */ │ │ │ │ │ - getServerZoom: function() { │ │ │ │ │ - var resolution = this.getServerResolution(); │ │ │ │ │ - return this.serverResolutions ? │ │ │ │ │ - OpenLayers.Util.indexOf(this.serverResolutions, resolution) : │ │ │ │ │ - this.map.getZoomForResolution(resolution) + (this.zoomOffset || 0); │ │ │ │ │ + createNode: function(type, id) { │ │ │ │ │ + var node = document.createElementNS(this.xmlns, type); │ │ │ │ │ + if (id) { │ │ │ │ │ + node.setAttributeNS(null, "id", id); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: applyBackBuffer │ │ │ │ │ - * Create, insert, scale and position a back buffer for the layer. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: nodeTypeCompare │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resolution - {Number} The resolution to transition to. │ │ │ │ │ + * node - {SVGDomElement} An SVG element │ │ │ │ │ + * type - {String} Kind of node │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ */ │ │ │ │ │ - applyBackBuffer: function(resolution) { │ │ │ │ │ - if (this.backBufferTimerId !== null) { │ │ │ │ │ - this.removeBackBuffer(); │ │ │ │ │ - } │ │ │ │ │ - var backBuffer = this.backBuffer; │ │ │ │ │ - if (!backBuffer) { │ │ │ │ │ - backBuffer = this.createBackBuffer(); │ │ │ │ │ - if (!backBuffer) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (resolution === this.gridResolution) { │ │ │ │ │ - this.div.insertBefore(backBuffer, this.div.firstChild); │ │ │ │ │ - } else { │ │ │ │ │ - this.map.baseLayer.div.parentNode.insertBefore(backBuffer, this.map.baseLayer.div); │ │ │ │ │ - } │ │ │ │ │ - this.backBuffer = backBuffer; │ │ │ │ │ - │ │ │ │ │ - // set some information in the instance for subsequent │ │ │ │ │ - // calls to applyBackBuffer where the same back buffer │ │ │ │ │ - // is reused │ │ │ │ │ - var topLeftTileBounds = this.grid[0][0].bounds; │ │ │ │ │ - this.backBufferLonLat = { │ │ │ │ │ - lon: topLeftTileBounds.left, │ │ │ │ │ - lat: topLeftTileBounds.top │ │ │ │ │ - }; │ │ │ │ │ - this.backBufferResolution = this.gridResolution; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var ratio = this.backBufferResolution / resolution; │ │ │ │ │ - │ │ │ │ │ - // scale the tiles inside the back buffer │ │ │ │ │ - var tiles = backBuffer.childNodes, │ │ │ │ │ - tile; │ │ │ │ │ - for (var i = tiles.length - 1; i >= 0; --i) { │ │ │ │ │ - tile = tiles[i]; │ │ │ │ │ - tile.style.top = ((ratio * tile._i * tile._h) | 0) + 'px'; │ │ │ │ │ - tile.style.left = ((ratio * tile._j * tile._w) | 0) + 'px'; │ │ │ │ │ - tile.style.width = Math.round(ratio * tile._w) + 'px'; │ │ │ │ │ - tile.style.height = Math.round(ratio * tile._h) + 'px'; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // and position it (based on the grid's top-left corner) │ │ │ │ │ - var position = this.getViewPortPxFromLonLat( │ │ │ │ │ - this.backBufferLonLat, resolution); │ │ │ │ │ - var leftOffset = this.map.layerContainerOriginPx.x; │ │ │ │ │ - var topOffset = this.map.layerContainerOriginPx.y; │ │ │ │ │ - backBuffer.style.left = Math.round(position.x - leftOffset) + 'px'; │ │ │ │ │ - backBuffer.style.top = Math.round(position.y - topOffset) + 'px'; │ │ │ │ │ + nodeTypeCompare: function(node, type) { │ │ │ │ │ + return (type == node.nodeName); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createBackBuffer │ │ │ │ │ - * Create a back buffer. │ │ │ │ │ - * │ │ │ │ │ + * Method: createRenderRoot │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} The DOM element for the back buffer, undefined if the │ │ │ │ │ - * grid isn't initialized yet. │ │ │ │ │ + * {DOMElement} The specific render engine's root element │ │ │ │ │ */ │ │ │ │ │ - createBackBuffer: function() { │ │ │ │ │ - var backBuffer; │ │ │ │ │ - if (this.grid.length > 0) { │ │ │ │ │ - backBuffer = document.createElement('div'); │ │ │ │ │ - backBuffer.id = this.div.id + '_bb'; │ │ │ │ │ - backBuffer.className = 'olBackBuffer'; │ │ │ │ │ - backBuffer.style.position = 'absolute'; │ │ │ │ │ - var map = this.map; │ │ │ │ │ - backBuffer.style.zIndex = this.transitionEffect === 'resize' ? │ │ │ │ │ - this.getZIndex() - 1 : │ │ │ │ │ - // 'map-resize': │ │ │ │ │ - map.Z_INDEX_BASE.BaseLayer - │ │ │ │ │ - (map.getNumLayers() - map.getLayerIndex(this)); │ │ │ │ │ - for (var i = 0, lenI = this.grid.length; i < lenI; i++) { │ │ │ │ │ - for (var j = 0, lenJ = this.grid[i].length; j < lenJ; j++) { │ │ │ │ │ - var tile = this.grid[i][j], │ │ │ │ │ - markup = this.grid[i][j].createBackBuffer(); │ │ │ │ │ - if (markup) { │ │ │ │ │ - markup._i = i; │ │ │ │ │ - markup._j = j; │ │ │ │ │ - markup._w = tile.size.w; │ │ │ │ │ - markup._h = tile.size.h; │ │ │ │ │ - markup.id = tile.id + '_bb'; │ │ │ │ │ - backBuffer.appendChild(markup); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return backBuffer; │ │ │ │ │ + createRenderRoot: function() { │ │ │ │ │ + var svg = this.nodeFactory(this.container.id + "_svgRoot", "svg"); │ │ │ │ │ + svg.style.display = "block"; │ │ │ │ │ + return svg; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removeBackBuffer │ │ │ │ │ - * Remove back buffer from DOM. │ │ │ │ │ + * Method: createRoot │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * suffix - {String} suffix to append to the id │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - removeBackBuffer: function() { │ │ │ │ │ - if (this._transitionElement) { │ │ │ │ │ - for (var i = this.transitionendEvents.length - 1; i >= 0; --i) { │ │ │ │ │ - OpenLayers.Event.stopObserving(this._transitionElement, │ │ │ │ │ - this.transitionendEvents[i], this._removeBackBuffer); │ │ │ │ │ - } │ │ │ │ │ - delete this._transitionElement; │ │ │ │ │ - } │ │ │ │ │ - if (this.backBuffer) { │ │ │ │ │ - if (this.backBuffer.parentNode) { │ │ │ │ │ - this.backBuffer.parentNode.removeChild(this.backBuffer); │ │ │ │ │ - } │ │ │ │ │ - this.backBuffer = null; │ │ │ │ │ - this.backBufferResolution = null; │ │ │ │ │ - if (this.backBufferTimerId !== null) { │ │ │ │ │ - window.clearTimeout(this.backBufferTimerId); │ │ │ │ │ - this.backBufferTimerId = null; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + createRoot: function(suffix) { │ │ │ │ │ + return this.nodeFactory(this.container.id + suffix, "g"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveByPx │ │ │ │ │ - * Move the layer based on pixel vector. │ │ │ │ │ + * Method: createDefs │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * dx - {Number} │ │ │ │ │ - * dy - {Number} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The element to which we'll add the symbol definitions │ │ │ │ │ */ │ │ │ │ │ - moveByPx: function(dx, dy) { │ │ │ │ │ - if (!this.singleTile) { │ │ │ │ │ - this.moveGriddedTiles(); │ │ │ │ │ - } │ │ │ │ │ + createDefs: function() { │ │ │ │ │ + var defs = this.nodeFactory(this.container.id + "_defs", "defs"); │ │ │ │ │ + this.rendererRoot.appendChild(defs); │ │ │ │ │ + return defs; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + /************************************** │ │ │ │ │ + * * │ │ │ │ │ + * GEOMETRY DRAWING FUNCTIONS * │ │ │ │ │ + * * │ │ │ │ │ + **************************************/ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setTileSize │ │ │ │ │ - * Check if we are in singleTile mode and if so, set the size as a ratio │ │ │ │ │ - * of the map size (as specified by the layer's 'ratio' property). │ │ │ │ │ + * Method: drawPoint │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the point │ │ │ │ │ */ │ │ │ │ │ - setTileSize: function(size) { │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - size.h = parseInt(size.h * this.ratio, 10); │ │ │ │ │ - size.w = parseInt(size.w * this.ratio, 10); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.setTileSize.apply(this, [size]); │ │ │ │ │ + drawPoint: function(node, geometry) { │ │ │ │ │ + return this.drawCircle(node, geometry, 1); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getTilesBounds │ │ │ │ │ - * Return the bounds of the tile grid. │ │ │ │ │ - * │ │ │ │ │ + * Method: drawCircle │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * radius - {Float} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} A Bounds object representing the bounds of all the │ │ │ │ │ - * currently loaded tiles (including those partially or not at all seen │ │ │ │ │ - * onscreen). │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the circle │ │ │ │ │ */ │ │ │ │ │ - getTilesBounds: function() { │ │ │ │ │ - var bounds = null; │ │ │ │ │ - │ │ │ │ │ - var length = this.grid.length; │ │ │ │ │ - if (length) { │ │ │ │ │ - var bottomLeftTileBounds = this.grid[length - 1][0].bounds, │ │ │ │ │ - width = this.grid[0].length * bottomLeftTileBounds.getWidth(), │ │ │ │ │ - height = this.grid.length * bottomLeftTileBounds.getHeight(); │ │ │ │ │ + drawCircle: function(node, geometry, radius) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = ((geometry.x - this.featureDx) / resolution + this.left); │ │ │ │ │ + var y = (this.top - geometry.y / resolution); │ │ │ │ │ │ │ │ │ │ - bounds = new OpenLayers.Bounds(bottomLeftTileBounds.left, │ │ │ │ │ - bottomLeftTileBounds.bottom, │ │ │ │ │ - bottomLeftTileBounds.left + width, │ │ │ │ │ - bottomLeftTileBounds.bottom + height); │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + node.setAttributeNS(null, "cx", x); │ │ │ │ │ + node.setAttributeNS(null, "cy", y); │ │ │ │ │ + node.setAttributeNS(null, "r", radius); │ │ │ │ │ + return node; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - return bounds; │ │ │ │ │ + │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: initSingleTile │ │ │ │ │ + * Method: drawLineString │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components of │ │ │ │ │ + * the linestring, or false if nothing could be drawn │ │ │ │ │ */ │ │ │ │ │ - initSingleTile: function(bounds) { │ │ │ │ │ - this.events.triggerEvent("retile"); │ │ │ │ │ - │ │ │ │ │ - //determine new tile bounds │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var tileWidth = bounds.getWidth() * this.ratio; │ │ │ │ │ - var tileHeight = bounds.getHeight() * this.ratio; │ │ │ │ │ - │ │ │ │ │ - var tileBounds = │ │ │ │ │ - new OpenLayers.Bounds(center.lon - (tileWidth / 2), │ │ │ │ │ - center.lat - (tileHeight / 2), │ │ │ │ │ - center.lon + (tileWidth / 2), │ │ │ │ │ - center.lat + (tileHeight / 2)); │ │ │ │ │ - │ │ │ │ │ - var px = this.map.getLayerPxFromLonLat({ │ │ │ │ │ - lon: tileBounds.left, │ │ │ │ │ - lat: tileBounds.top │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - if (!this.grid.length) { │ │ │ │ │ - this.grid[0] = []; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var tile = this.grid[0][0]; │ │ │ │ │ - if (!tile) { │ │ │ │ │ - tile = this.addTile(tileBounds, px); │ │ │ │ │ - │ │ │ │ │ - this.addTileMonitoringHooks(tile); │ │ │ │ │ - tile.draw(); │ │ │ │ │ - this.grid[0][0] = tile; │ │ │ │ │ + drawLineString: function(node, geometry) { │ │ │ │ │ + var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ + if (componentsResult.path) { │ │ │ │ │ + node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ + return (componentsResult.complete ? node : null); │ │ │ │ │ } else { │ │ │ │ │ - tile.moveTo(tileBounds, px); │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - //remove all but our single tile │ │ │ │ │ - this.removeExcessTiles(1, 1); │ │ │ │ │ - │ │ │ │ │ - // store the resolution of the grid │ │ │ │ │ - this.gridResolution = this.getServerResolution(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateGridLayout │ │ │ │ │ - * Generate parameters for the grid layout. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bound>|Object} OpenLayers.Bounds or an │ │ │ │ │ - * object with a 'left' and 'top' properties. │ │ │ │ │ - * origin - {<OpenLayers.LonLat>|Object} OpenLayers.LonLat or an │ │ │ │ │ - * object with a 'lon' and 'lat' properties. │ │ │ │ │ - * resolution - {Number} │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawLinearRing │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} Object containing properties tilelon, tilelat, startcol, │ │ │ │ │ - * startrow │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ + * of the linear ring, or false if nothing could be drawn │ │ │ │ │ */ │ │ │ │ │ - calculateGridLayout: function(bounds, origin, resolution) { │ │ │ │ │ - var tilelon = resolution * this.tileSize.w; │ │ │ │ │ - var tilelat = resolution * this.tileSize.h; │ │ │ │ │ - │ │ │ │ │ - var offsetlon = bounds.left - origin.lon; │ │ │ │ │ - var tilecol = Math.floor(offsetlon / tilelon) - this.buffer; │ │ │ │ │ - │ │ │ │ │ - var rowSign = this.rowSign; │ │ │ │ │ - │ │ │ │ │ - var offsetlat = rowSign * (origin.lat - bounds.top + tilelat); │ │ │ │ │ - var tilerow = Math[~rowSign ? 'floor' : 'ceil'](offsetlat / tilelat) - this.buffer * rowSign; │ │ │ │ │ - │ │ │ │ │ - return { │ │ │ │ │ - tilelon: tilelon, │ │ │ │ │ - tilelat: tilelat, │ │ │ │ │ - startcol: tilecol, │ │ │ │ │ - startrow: tilerow │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ + drawLinearRing: function(node, geometry) { │ │ │ │ │ + var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ + if (componentsResult.path) { │ │ │ │ │ + node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ + return (componentsResult.complete ? node : null); │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getTileOrigin │ │ │ │ │ - * Determine the origin for aligning the grid of tiles. If a <tileOrigin> │ │ │ │ │ - * property is supplied, that will be returned. Otherwise, the origin │ │ │ │ │ - * will be derived from the layer's <maxExtent> property. In this case, │ │ │ │ │ - * the tile origin will be the corner of the <maxExtent> given by the │ │ │ │ │ - * <tileOriginCorner> property. │ │ │ │ │ - * │ │ │ │ │ + * Method: drawPolygon │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.LonLat>} The tile origin. │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ + * of the polygon, or false if nothing could be drawn │ │ │ │ │ */ │ │ │ │ │ - getTileOrigin: function() { │ │ │ │ │ - var origin = this.tileOrigin; │ │ │ │ │ - if (!origin) { │ │ │ │ │ - var extent = this.getMaxExtent(); │ │ │ │ │ - var edges = ({ │ │ │ │ │ - "tl": ["left", "top"], │ │ │ │ │ - "tr": ["right", "top"], │ │ │ │ │ - "bl": ["left", "bottom"], │ │ │ │ │ - "br": ["right", "bottom"] │ │ │ │ │ - })[this.tileOriginCorner]; │ │ │ │ │ - origin = new OpenLayers.LonLat(extent[edges[0]], extent[edges[1]]); │ │ │ │ │ + drawPolygon: function(node, geometry) { │ │ │ │ │ + var d = ""; │ │ │ │ │ + var draw = true; │ │ │ │ │ + var complete = true; │ │ │ │ │ + var linearRingResult, path; │ │ │ │ │ + for (var j = 0, len = geometry.components.length; j < len; j++) { │ │ │ │ │ + d += " M"; │ │ │ │ │ + linearRingResult = this.getComponentsString( │ │ │ │ │ + geometry.components[j].components, " "); │ │ │ │ │ + path = linearRingResult.path; │ │ │ │ │ + if (path) { │ │ │ │ │ + d += " " + path; │ │ │ │ │ + complete = linearRingResult.complete && complete; │ │ │ │ │ + } else { │ │ │ │ │ + draw = false; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + d += " z"; │ │ │ │ │ + if (draw) { │ │ │ │ │ + node.setAttributeNS(null, "d", d); │ │ │ │ │ + node.setAttributeNS(null, "fill-rule", "evenodd"); │ │ │ │ │ + return complete ? node : null; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - return origin; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getTileBoundsForGridIndex │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * row - {Number} The row of the grid │ │ │ │ │ - * col - {Number} The column of the grid │ │ │ │ │ - * │ │ │ │ │ + * Method: drawRectangle │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} The bounds for the tile at (row, col) │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the rectangle │ │ │ │ │ */ │ │ │ │ │ - getTileBoundsForGridIndex: function(row, col) { │ │ │ │ │ - var origin = this.getTileOrigin(); │ │ │ │ │ - var tileLayout = this.gridLayout; │ │ │ │ │ - var tilelon = tileLayout.tilelon; │ │ │ │ │ - var tilelat = tileLayout.tilelat; │ │ │ │ │ - var startcol = tileLayout.startcol; │ │ │ │ │ - var startrow = tileLayout.startrow; │ │ │ │ │ - var rowSign = this.rowSign; │ │ │ │ │ - return new OpenLayers.Bounds( │ │ │ │ │ - origin.lon + (startcol + col) * tilelon, │ │ │ │ │ - origin.lat - (startrow + row * rowSign) * tilelat * rowSign, │ │ │ │ │ - origin.lon + (startcol + col + 1) * tilelon, │ │ │ │ │ - origin.lat - (startrow + (row - 1) * rowSign) * tilelat * rowSign │ │ │ │ │ - ); │ │ │ │ │ + drawRectangle: function(node, geometry) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = ((geometry.x - this.featureDx) / resolution + this.left); │ │ │ │ │ + var y = (this.top - geometry.y / resolution); │ │ │ │ │ + │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + node.setAttributeNS(null, "x", x); │ │ │ │ │ + node.setAttributeNS(null, "y", y); │ │ │ │ │ + node.setAttributeNS(null, "width", geometry.width / resolution); │ │ │ │ │ + node.setAttributeNS(null, "height", geometry.height / resolution); │ │ │ │ │ + return node; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: initGriddedTiles │ │ │ │ │ - * │ │ │ │ │ + * Method: drawText │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * style - │ │ │ │ │ + * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ */ │ │ │ │ │ - initGriddedTiles: function(bounds) { │ │ │ │ │ - this.events.triggerEvent("retile"); │ │ │ │ │ - │ │ │ │ │ - // work out mininum number of rows and columns; this is the number of │ │ │ │ │ - // tiles required to cover the viewport plus at least one for panning │ │ │ │ │ - │ │ │ │ │ - var viewSize = this.map.getSize(); │ │ │ │ │ - │ │ │ │ │ - var origin = this.getTileOrigin(); │ │ │ │ │ - var resolution = this.map.getResolution(), │ │ │ │ │ - serverResolution = this.getServerResolution(), │ │ │ │ │ - ratio = resolution / serverResolution, │ │ │ │ │ - tileSize = { │ │ │ │ │ - w: this.tileSize.w / ratio, │ │ │ │ │ - h: this.tileSize.h / ratio │ │ │ │ │ - }; │ │ │ │ │ + drawText: function(featureId, style, location) { │ │ │ │ │ + var drawOutline = (!!style.labelOutlineWidth); │ │ │ │ │ + // First draw text in halo color and size and overlay the │ │ │ │ │ + // normal text afterwards │ │ │ │ │ + if (drawOutline) { │ │ │ │ │ + var outlineStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ + outlineStyle.fontColor = outlineStyle.labelOutlineColor; │ │ │ │ │ + outlineStyle.fontStrokeColor = outlineStyle.labelOutlineColor; │ │ │ │ │ + outlineStyle.fontStrokeWidth = style.labelOutlineWidth; │ │ │ │ │ + if (style.labelOutlineOpacity) { │ │ │ │ │ + outlineStyle.fontOpacity = style.labelOutlineOpacity; │ │ │ │ │ + } │ │ │ │ │ + delete outlineStyle.labelOutlineWidth; │ │ │ │ │ + this.drawText(featureId, outlineStyle, location); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var minRows = Math.ceil(viewSize.h / tileSize.h) + │ │ │ │ │ - 2 * this.buffer + 1; │ │ │ │ │ - var minCols = Math.ceil(viewSize.w / tileSize.w) + │ │ │ │ │ - 2 * this.buffer + 1; │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ │ │ │ │ │ - var tileLayout = this.calculateGridLayout(bounds, origin, serverResolution); │ │ │ │ │ - this.gridLayout = tileLayout; │ │ │ │ │ + var x = ((location.x - this.featureDx) / resolution + this.left); │ │ │ │ │ + var y = (location.y / resolution - this.top); │ │ │ │ │ │ │ │ │ │ - var tilelon = tileLayout.tilelon; │ │ │ │ │ - var tilelat = tileLayout.tilelat; │ │ │ │ │ + var suffix = (drawOutline) ? this.LABEL_OUTLINE_SUFFIX : this.LABEL_ID_SUFFIX; │ │ │ │ │ + var label = this.nodeFactory(featureId + suffix, "text"); │ │ │ │ │ │ │ │ │ │ - var layerContainerDivLeft = this.map.layerContainerOriginPx.x; │ │ │ │ │ - var layerContainerDivTop = this.map.layerContainerOriginPx.y; │ │ │ │ │ + label.setAttributeNS(null, "x", x); │ │ │ │ │ + label.setAttributeNS(null, "y", -y); │ │ │ │ │ │ │ │ │ │ - var tileBounds = this.getTileBoundsForGridIndex(0, 0); │ │ │ │ │ - var startPx = this.map.getViewPortPxFromLonLat( │ │ │ │ │ - new OpenLayers.LonLat(tileBounds.left, tileBounds.top) │ │ │ │ │ - ); │ │ │ │ │ - startPx.x = Math.round(startPx.x) - layerContainerDivLeft; │ │ │ │ │ - startPx.y = Math.round(startPx.y) - layerContainerDivTop; │ │ │ │ │ + if (style.fontColor) { │ │ │ │ │ + label.setAttributeNS(null, "fill", style.fontColor); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStrokeColor) { │ │ │ │ │ + label.setAttributeNS(null, "stroke", style.fontStrokeColor); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStrokeWidth) { │ │ │ │ │ + label.setAttributeNS(null, "stroke-width", style.fontStrokeWidth); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontOpacity) { │ │ │ │ │ + label.setAttributeNS(null, "opacity", style.fontOpacity); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontFamily) { │ │ │ │ │ + label.setAttributeNS(null, "font-family", style.fontFamily); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontSize) { │ │ │ │ │ + label.setAttributeNS(null, "font-size", style.fontSize); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontWeight) { │ │ │ │ │ + label.setAttributeNS(null, "font-weight", style.fontWeight); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStyle) { │ │ │ │ │ + label.setAttributeNS(null, "font-style", style.fontStyle); │ │ │ │ │ + } │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + label.setAttributeNS(null, "pointer-events", "visible"); │ │ │ │ │ + label._featureId = featureId; │ │ │ │ │ + } else { │ │ │ │ │ + label.setAttributeNS(null, "pointer-events", "none"); │ │ │ │ │ + } │ │ │ │ │ + var align = style.labelAlign || OpenLayers.Renderer.defaultSymbolizer.labelAlign; │ │ │ │ │ + label.setAttributeNS(null, "text-anchor", │ │ │ │ │ + OpenLayers.Renderer.SVG.LABEL_ALIGN[align[0]] || "middle"); │ │ │ │ │ │ │ │ │ │ - var tileData = [], │ │ │ │ │ - center = this.map.getCenter(); │ │ │ │ │ + if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ + label.setAttributeNS(null, "dominant-baseline", │ │ │ │ │ + OpenLayers.Renderer.SVG.LABEL_ALIGN[align[1]] || "central"); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var rowidx = 0; │ │ │ │ │ - do { │ │ │ │ │ - var row = this.grid[rowidx]; │ │ │ │ │ - if (!row) { │ │ │ │ │ - row = []; │ │ │ │ │ - this.grid.push(row); │ │ │ │ │ + var labelRows = style.label.split('\n'); │ │ │ │ │ + var numRows = labelRows.length; │ │ │ │ │ + while (label.childNodes.length > numRows) { │ │ │ │ │ + label.removeChild(label.lastChild); │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + var tspan = this.nodeFactory(featureId + suffix + "_tspan_" + i, "tspan"); │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + tspan._featureId = featureId; │ │ │ │ │ + tspan._geometry = location; │ │ │ │ │ + tspan._geometryClass = location.CLASS_NAME; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var colidx = 0; │ │ │ │ │ - do { │ │ │ │ │ - tileBounds = this.getTileBoundsForGridIndex(rowidx, colidx); │ │ │ │ │ - var px = startPx.clone(); │ │ │ │ │ - px.x = px.x + colidx * Math.round(tileSize.w); │ │ │ │ │ - px.y = px.y + rowidx * Math.round(tileSize.h); │ │ │ │ │ - var tile = row[colidx]; │ │ │ │ │ - if (!tile) { │ │ │ │ │ - tile = this.addTile(tileBounds, px); │ │ │ │ │ - this.addTileMonitoringHooks(tile); │ │ │ │ │ - row.push(tile); │ │ │ │ │ - } else { │ │ │ │ │ - tile.moveTo(tileBounds, px, false); │ │ │ │ │ + if (OpenLayers.IS_GECKO === false) { │ │ │ │ │ + tspan.setAttributeNS(null, "baseline-shift", │ │ │ │ │ + OpenLayers.Renderer.SVG.LABEL_VSHIFT[align[1]] || "-35%"); │ │ │ │ │ + } │ │ │ │ │ + tspan.setAttribute("x", x); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + var vfactor = OpenLayers.Renderer.SVG.LABEL_VFACTOR[align[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5; │ │ │ │ │ } │ │ │ │ │ - var tileCenter = tileBounds.getCenterLonLat(); │ │ │ │ │ - tileData.push({ │ │ │ │ │ - tile: tile, │ │ │ │ │ - distance: Math.pow(tileCenter.lon - center.lon, 2) + │ │ │ │ │ - Math.pow(tileCenter.lat - center.lat, 2) │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - colidx += 1; │ │ │ │ │ - } while ((tileBounds.right <= bounds.right + tilelon * this.buffer) || │ │ │ │ │ - colidx < minCols); │ │ │ │ │ - │ │ │ │ │ - rowidx += 1; │ │ │ │ │ - } while ((tileBounds.bottom >= bounds.bottom - tilelat * this.buffer) || │ │ │ │ │ - rowidx < minRows); │ │ │ │ │ - │ │ │ │ │ - //shave off exceess rows and colums │ │ │ │ │ - this.removeExcessTiles(rowidx, colidx); │ │ │ │ │ - │ │ │ │ │ - var resolution = this.getServerResolution(); │ │ │ │ │ - // store the resolution of the grid │ │ │ │ │ - this.gridResolution = resolution; │ │ │ │ │ - │ │ │ │ │ - //now actually draw the tiles │ │ │ │ │ - tileData.sort(function(a, b) { │ │ │ │ │ - return a.distance - b.distance; │ │ │ │ │ - }); │ │ │ │ │ - for (var i = 0, ii = tileData.length; i < ii; ++i) { │ │ │ │ │ - tileData[i].tile.draw(); │ │ │ │ │ + tspan.setAttribute("dy", (vfactor * (numRows - 1)) + "em"); │ │ │ │ │ + } else { │ │ │ │ │ + tspan.setAttribute("dy", "1em"); │ │ │ │ │ + } │ │ │ │ │ + tspan.textContent = (labelRows[i] === '') ? ' ' : labelRows[i]; │ │ │ │ │ + if (!tspan.parentNode) { │ │ │ │ │ + label.appendChild(tspan); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getMaxExtent │ │ │ │ │ - * Get this layer's maximum extent. (Implemented as a getter for │ │ │ │ │ - * potential specific implementations in sub-classes.) │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} │ │ │ │ │ - */ │ │ │ │ │ - getMaxExtent: function() { │ │ │ │ │ - return this.maxExtent; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: addTile │ │ │ │ │ - * Create a tile, initialize it, and add it to the layer div. │ │ │ │ │ - * │ │ │ │ │ - * Parameters │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * position - {<OpenLayers.Pixel>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Tile>} The added OpenLayers.Tile │ │ │ │ │ - */ │ │ │ │ │ - addTile: function(bounds, position) { │ │ │ │ │ - var tile = new this.tileClass( │ │ │ │ │ - this, position, bounds, null, this.tileSize, this.tileOptions │ │ │ │ │ - ); │ │ │ │ │ - this.events.triggerEvent("addtile", { │ │ │ │ │ - tile: tile │ │ │ │ │ - }); │ │ │ │ │ - return tile; │ │ │ │ │ + if (!label.parentNode) { │ │ │ │ │ + this.textRoot.appendChild(label); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: addTileMonitoringHooks │ │ │ │ │ - * This function takes a tile as input and adds the appropriate hooks to │ │ │ │ │ - * the tile so that the layer can keep track of the loading tiles. │ │ │ │ │ + * Method: getComponentString │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * tile - {<OpenLayers.Tile>} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * components - {Array(<OpenLayers.Geometry.Point>)} Array of points │ │ │ │ │ + * separator - {String} character between coordinate pairs. Defaults to "," │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} hash with properties "path" (the string created from the │ │ │ │ │ + * components and "complete" (false if the renderer was unable to │ │ │ │ │ + * draw all components) │ │ │ │ │ */ │ │ │ │ │ - addTileMonitoringHooks: function(tile) { │ │ │ │ │ - │ │ │ │ │ - var replacingCls = 'olTileReplacing'; │ │ │ │ │ - │ │ │ │ │ - tile.onLoadStart = function() { │ │ │ │ │ - //if that was first tile then trigger a 'loadstart' on the layer │ │ │ │ │ - if (this.loading === false) { │ │ │ │ │ - this.loading = true; │ │ │ │ │ - this.events.triggerEvent("loadstart"); │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("tileloadstart", { │ │ │ │ │ - tile: tile │ │ │ │ │ - }); │ │ │ │ │ - this.numLoadingTiles++; │ │ │ │ │ - if (!this.singleTile && this.backBuffer && this.gridResolution === this.backBufferResolution) { │ │ │ │ │ - OpenLayers.Element.addClass(tile.getTile(), replacingCls); │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - tile.onLoadEnd = function(evt) { │ │ │ │ │ - this.numLoadingTiles--; │ │ │ │ │ - var aborted = evt.type === 'unload'; │ │ │ │ │ - this.events.triggerEvent("tileloaded", { │ │ │ │ │ - tile: tile, │ │ │ │ │ - aborted: aborted │ │ │ │ │ - }); │ │ │ │ │ - if (!this.singleTile && !aborted && this.backBuffer && this.gridResolution === this.backBufferResolution) { │ │ │ │ │ - var tileDiv = tile.getTile(); │ │ │ │ │ - if (OpenLayers.Element.getStyle(tileDiv, 'display') === 'none') { │ │ │ │ │ - var bufferTile = document.getElementById(tile.id + '_bb'); │ │ │ │ │ - if (bufferTile) { │ │ │ │ │ - bufferTile.parentNode.removeChild(bufferTile); │ │ │ │ │ + getComponentsString: function(components, separator) { │ │ │ │ │ + var renderCmp = []; │ │ │ │ │ + var complete = true; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + var strings = []; │ │ │ │ │ + var str, component; │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + component = components[i]; │ │ │ │ │ + renderCmp.push(component); │ │ │ │ │ + str = this.getShortString(component); │ │ │ │ │ + if (str) { │ │ │ │ │ + strings.push(str); │ │ │ │ │ + } else { │ │ │ │ │ + // The current component is outside the valid range. Let's │ │ │ │ │ + // see if the previous or next component is inside the range. │ │ │ │ │ + // If so, add the coordinate of the intersection with the │ │ │ │ │ + // valid range bounds. │ │ │ │ │ + if (i > 0) { │ │ │ │ │ + if (this.getShortString(components[i - 1])) { │ │ │ │ │ + strings.push(this.clipLine(components[i], │ │ │ │ │ + components[i - 1])); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Element.removeClass(tileDiv, replacingCls); │ │ │ │ │ - } │ │ │ │ │ - //if that was the last tile, then trigger a 'loadend' on the layer │ │ │ │ │ - if (this.numLoadingTiles === 0) { │ │ │ │ │ - if (this.backBuffer) { │ │ │ │ │ - if (this.backBuffer.childNodes.length === 0) { │ │ │ │ │ - // no tiles transitioning, remove immediately │ │ │ │ │ - this.removeBackBuffer(); │ │ │ │ │ - } else { │ │ │ │ │ - // wait until transition has ended or delay has passed │ │ │ │ │ - this._transitionElement = aborted ? │ │ │ │ │ - this.div.lastChild : tile.imgDiv; │ │ │ │ │ - var transitionendEvents = this.transitionendEvents; │ │ │ │ │ - for (var i = transitionendEvents.length - 1; i >= 0; --i) { │ │ │ │ │ - OpenLayers.Event.observe(this._transitionElement, │ │ │ │ │ - transitionendEvents[i], │ │ │ │ │ - this._removeBackBuffer); │ │ │ │ │ - } │ │ │ │ │ - // the removal of the back buffer is delayed to prevent │ │ │ │ │ - // flash effects due to the animation of tile displaying │ │ │ │ │ - this.backBufferTimerId = window.setTimeout( │ │ │ │ │ - this._removeBackBuffer, this.removeBackBufferDelay │ │ │ │ │ - ); │ │ │ │ │ + if (i < len - 1) { │ │ │ │ │ + if (this.getShortString(components[i + 1])) { │ │ │ │ │ + strings.push(this.clipLine(components[i], │ │ │ │ │ + components[i + 1])); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.loading = false; │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ + complete = false; │ │ │ │ │ } │ │ │ │ │ - }; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - tile.onLoadError = function() { │ │ │ │ │ - this.events.triggerEvent("tileerror", { │ │ │ │ │ - tile: tile │ │ │ │ │ - }); │ │ │ │ │ + return { │ │ │ │ │ + path: strings.join(separator || ","), │ │ │ │ │ + complete: complete │ │ │ │ │ }; │ │ │ │ │ - │ │ │ │ │ - tile.events.on({ │ │ │ │ │ - "loadstart": tile.onLoadStart, │ │ │ │ │ - "loadend": tile.onLoadEnd, │ │ │ │ │ - "unload": tile.onLoadEnd, │ │ │ │ │ - "loaderror": tile.onLoadError, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeTileMonitoringHooks │ │ │ │ │ - * This function takes a tile as input and removes the tile hooks │ │ │ │ │ - * that were added in addTileMonitoringHooks() │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * tile - {<OpenLayers.Tile>} │ │ │ │ │ - */ │ │ │ │ │ - removeTileMonitoringHooks: function(tile) { │ │ │ │ │ - tile.unload(); │ │ │ │ │ - tile.events.un({ │ │ │ │ │ - "loadstart": tile.onLoadStart, │ │ │ │ │ - "loadend": tile.onLoadEnd, │ │ │ │ │ - "unload": tile.onLoadEnd, │ │ │ │ │ - "loaderror": tile.onLoadError, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveGriddedTiles │ │ │ │ │ + * Method: clipLine │ │ │ │ │ + * Given two points (one inside the valid range, and one outside), │ │ │ │ │ + * clips the line betweeen the two points so that the new points are both │ │ │ │ │ + * inside the valid range. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * badComponent - {<OpenLayers.Geometry.Point>} original geometry of the │ │ │ │ │ + * invalid point │ │ │ │ │ + * goodComponent - {<OpenLayers.Geometry.Point>} original geometry of the │ │ │ │ │ + * valid point │ │ │ │ │ + * Returns │ │ │ │ │ + * {String} the SVG coordinate pair of the clipped point (like │ │ │ │ │ + * getShortString), or an empty string if both passed componets are at │ │ │ │ │ + * the same point. │ │ │ │ │ */ │ │ │ │ │ - moveGriddedTiles: function() { │ │ │ │ │ - var buffer = this.buffer + 1; │ │ │ │ │ - while (true) { │ │ │ │ │ - var tlTile = this.grid[0][0]; │ │ │ │ │ - var tlViewPort = { │ │ │ │ │ - x: tlTile.position.x + │ │ │ │ │ - this.map.layerContainerOriginPx.x, │ │ │ │ │ - y: tlTile.position.y + │ │ │ │ │ - this.map.layerContainerOriginPx.y │ │ │ │ │ - }; │ │ │ │ │ - var ratio = this.getServerResolution() / this.map.getResolution(); │ │ │ │ │ - var tileSize = { │ │ │ │ │ - w: Math.round(this.tileSize.w * ratio), │ │ │ │ │ - h: Math.round(this.tileSize.h * ratio) │ │ │ │ │ - }; │ │ │ │ │ - if (tlViewPort.x > -tileSize.w * (buffer - 1)) { │ │ │ │ │ - this.shiftColumn(true, tileSize); │ │ │ │ │ - } else if (tlViewPort.x < -tileSize.w * buffer) { │ │ │ │ │ - this.shiftColumn(false, tileSize); │ │ │ │ │ - } else if (tlViewPort.y > -tileSize.h * (buffer - 1)) { │ │ │ │ │ - this.shiftRow(true, tileSize); │ │ │ │ │ - } else if (tlViewPort.y < -tileSize.h * buffer) { │ │ │ │ │ - this.shiftRow(false, tileSize); │ │ │ │ │ - } else { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ + clipLine: function(badComponent, goodComponent) { │ │ │ │ │ + if (goodComponent.equals(badComponent)) { │ │ │ │ │ + return ""; │ │ │ │ │ + } │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var maxX = this.MAX_PIXEL - this.translationParameters.x; │ │ │ │ │ + var maxY = this.MAX_PIXEL - this.translationParameters.y; │ │ │ │ │ + var x1 = (goodComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y1 = this.top - goodComponent.y / resolution; │ │ │ │ │ + var x2 = (badComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y2 = this.top - badComponent.y / resolution; │ │ │ │ │ + var k; │ │ │ │ │ + if (x2 < -maxX || x2 > maxX) { │ │ │ │ │ + k = (y2 - y1) / (x2 - x1); │ │ │ │ │ + x2 = x2 < 0 ? -maxX : maxX; │ │ │ │ │ + y2 = y1 + (x2 - x1) * k; │ │ │ │ │ + } │ │ │ │ │ + if (y2 < -maxY || y2 > maxY) { │ │ │ │ │ + k = (x2 - x1) / (y2 - y1); │ │ │ │ │ + y2 = y2 < 0 ? -maxY : maxY; │ │ │ │ │ + x2 = x1 + (y2 - y1) * k; │ │ │ │ │ } │ │ │ │ │ + return x2 + "," + y2; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: shiftRow │ │ │ │ │ - * Shifty grid work │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: getShortString │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * prepend - {Boolean} if true, prepend to beginning. │ │ │ │ │ - * if false, then append to end │ │ │ │ │ - * tileSize - {Object} rendered tile size; object with w and h properties │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} or false if point is outside the valid range │ │ │ │ │ */ │ │ │ │ │ - shiftRow: function(prepend, tileSize) { │ │ │ │ │ - var grid = this.grid; │ │ │ │ │ - var rowIndex = prepend ? 0 : (grid.length - 1); │ │ │ │ │ - var sign = prepend ? -1 : 1; │ │ │ │ │ - var rowSign = this.rowSign; │ │ │ │ │ - var tileLayout = this.gridLayout; │ │ │ │ │ - tileLayout.startrow += sign * rowSign; │ │ │ │ │ + getShortString: function(point) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = ((point.x - this.featureDx) / resolution + this.left); │ │ │ │ │ + var y = (this.top - point.y / resolution); │ │ │ │ │ │ │ │ │ │ - var modelRow = grid[rowIndex]; │ │ │ │ │ - var row = grid[prepend ? 'pop' : 'shift'](); │ │ │ │ │ - for (var i = 0, len = row.length; i < len; i++) { │ │ │ │ │ - var tile = row[i]; │ │ │ │ │ - var position = modelRow[i].position.clone(); │ │ │ │ │ - position.y += tileSize.h * sign; │ │ │ │ │ - tile.moveTo(this.getTileBoundsForGridIndex(rowIndex, i), position); │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + return x + "," + y; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - grid[prepend ? 'unshift' : 'push'](row); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: shiftColumn │ │ │ │ │ - * Shift grid work in the other dimension │ │ │ │ │ - * │ │ │ │ │ + * Method: getPosition │ │ │ │ │ + * Finds the position of an svg node. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * prepend - {Boolean} if true, prepend to beginning. │ │ │ │ │ - * if false, then append to end │ │ │ │ │ - * tileSize - {Object} rendered tile size; object with w and h properties │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} hash with x and y properties, representing the coordinates │ │ │ │ │ + * within the svg coordinate system │ │ │ │ │ */ │ │ │ │ │ - shiftColumn: function(prepend, tileSize) { │ │ │ │ │ - var grid = this.grid; │ │ │ │ │ - var colIndex = prepend ? 0 : (grid[0].length - 1); │ │ │ │ │ - var sign = prepend ? -1 : 1; │ │ │ │ │ - var tileLayout = this.gridLayout; │ │ │ │ │ - tileLayout.startcol += sign; │ │ │ │ │ - │ │ │ │ │ - for (var i = 0, len = grid.length; i < len; i++) { │ │ │ │ │ - var row = grid[i]; │ │ │ │ │ - var position = row[colIndex].position.clone(); │ │ │ │ │ - var tile = row[prepend ? 'pop' : 'shift'](); │ │ │ │ │ - position.x += tileSize.w * sign; │ │ │ │ │ - tile.moveTo(this.getTileBoundsForGridIndex(i, colIndex), position); │ │ │ │ │ - row[prepend ? 'unshift' : 'push'](tile); │ │ │ │ │ - } │ │ │ │ │ + getPosition: function(node) { │ │ │ │ │ + return ({ │ │ │ │ │ + x: parseFloat(node.getAttributeNS(null, "cx")), │ │ │ │ │ + y: parseFloat(node.getAttributeNS(null, "cy")) │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removeExcessTiles │ │ │ │ │ - * When the size of the map or the buffer changes, we may need to │ │ │ │ │ - * remove some excess rows and columns. │ │ │ │ │ + * Method: importSymbol │ │ │ │ │ + * add a new symbol definition from the rendererer's symbol hash │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * rows - {Integer} Maximum number of rows we want our grid to have. │ │ │ │ │ - * columns - {Integer} Maximum number of columns we want our grid to have. │ │ │ │ │ + * graphicName - {String} name of the symbol to import │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} - the imported symbol │ │ │ │ │ */ │ │ │ │ │ - removeExcessTiles: function(rows, columns) { │ │ │ │ │ - var i, l; │ │ │ │ │ + importSymbol: function(graphicName) { │ │ │ │ │ + if (!this.defs) { │ │ │ │ │ + // create svg defs tag │ │ │ │ │ + this.defs = this.createDefs(); │ │ │ │ │ + } │ │ │ │ │ + var id = this.container.id + "-" + graphicName; │ │ │ │ │ │ │ │ │ │ - // remove extra rows │ │ │ │ │ - while (this.grid.length > rows) { │ │ │ │ │ - var row = this.grid.pop(); │ │ │ │ │ - for (i = 0, l = row.length; i < l; i++) { │ │ │ │ │ - var tile = row[i]; │ │ │ │ │ - this.destroyTile(tile); │ │ │ │ │ - } │ │ │ │ │ + // check if symbol already exists in the defs │ │ │ │ │ + var existing = document.getElementById(id); │ │ │ │ │ + if (existing != null) { │ │ │ │ │ + return existing; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // remove extra columns │ │ │ │ │ - for (i = 0, l = this.grid.length; i < l; i++) { │ │ │ │ │ - while (this.grid[i].length > columns) { │ │ │ │ │ - var row = this.grid[i]; │ │ │ │ │ - var tile = row.pop(); │ │ │ │ │ - this.destroyTile(tile); │ │ │ │ │ - } │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(graphicName + ' is not a valid symbol name'); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: onMapResize │ │ │ │ │ - * For singleTile layers, this will set a new tile size according to the │ │ │ │ │ - * dimensions of the map pane. │ │ │ │ │ - */ │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - this.clearGrid(); │ │ │ │ │ - this.setTileSize(); │ │ │ │ │ + var symbolNode = this.nodeFactory(id, "symbol"); │ │ │ │ │ + var node = this.nodeFactory(null, "polygon"); │ │ │ │ │ + symbolNode.appendChild(node); │ │ │ │ │ + var symbolExtent = new OpenLayers.Bounds( │ │ │ │ │ + Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ + │ │ │ │ │ + var points = []; │ │ │ │ │ + var x, y; │ │ │ │ │ + for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ + symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ + symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ + symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ + points.push(x, ",", y); │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + node.setAttributeNS(null, "points", points.join(" ")); │ │ │ │ │ + │ │ │ │ │ + var width = symbolExtent.getWidth(); │ │ │ │ │ + var height = symbolExtent.getHeight(); │ │ │ │ │ + // create a viewBox three times as large as the symbol itself, │ │ │ │ │ + // to allow for strokeWidth being displayed correctly at the corners. │ │ │ │ │ + var viewBox = [symbolExtent.left - width, │ │ │ │ │ + symbolExtent.bottom - height, width * 3, height * 3 │ │ │ │ │ + ]; │ │ │ │ │ + symbolNode.setAttributeNS(null, "viewBox", viewBox.join(" ")); │ │ │ │ │ + this.symbolMetrics[id] = [ │ │ │ │ │ + Math.max(width, height), │ │ │ │ │ + symbolExtent.getCenterLonLat().lon, │ │ │ │ │ + symbolExtent.getCenterLonLat().lat │ │ │ │ │ + ]; │ │ │ │ │ + │ │ │ │ │ + this.defs.appendChild(symbolNode); │ │ │ │ │ + return symbolNode; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getTileBounds │ │ │ │ │ - * Returns The tile bounds for a layer given a pixel location. │ │ │ │ │ - * │ │ │ │ │ + * Method: getFeatureIdFromEvent │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * viewPortPx - {<OpenLayers.Pixel>} The location in the viewport. │ │ │ │ │ + * evt - {Object} An <OpenLayers.Event> object │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} Bounds of the tile at the given pixel location. │ │ │ │ │ + * {String} A feature id or undefined. │ │ │ │ │ */ │ │ │ │ │ - getTileBounds: function(viewPortPx) { │ │ │ │ │ - var maxExtent = this.maxExtent; │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var tileMapWidth = resolution * this.tileSize.w; │ │ │ │ │ - var tileMapHeight = resolution * this.tileSize.h; │ │ │ │ │ - var mapPoint = this.getLonLatFromViewPortPx(viewPortPx); │ │ │ │ │ - var tileLeft = maxExtent.left + (tileMapWidth * │ │ │ │ │ - Math.floor((mapPoint.lon - │ │ │ │ │ - maxExtent.left) / │ │ │ │ │ - tileMapWidth)); │ │ │ │ │ - var tileBottom = maxExtent.bottom + (tileMapHeight * │ │ │ │ │ - Math.floor((mapPoint.lat - │ │ │ │ │ - maxExtent.bottom) / │ │ │ │ │ - tileMapHeight)); │ │ │ │ │ - return new OpenLayers.Bounds(tileLeft, tileBottom, │ │ │ │ │ - tileLeft + tileMapWidth, │ │ │ │ │ - tileBottom + tileMapHeight); │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var featureId = OpenLayers.Renderer.Elements.prototype.getFeatureIdFromEvent.apply(this, arguments); │ │ │ │ │ + if (!featureId) { │ │ │ │ │ + var target = evt.target; │ │ │ │ │ + featureId = target.parentNode && target != this.rendererRoot ? │ │ │ │ │ + target.parentNode._featureId : undefined; │ │ │ │ │ + } │ │ │ │ │ + return featureId; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Grid" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.SVG" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.SVG.LABEL_ALIGN │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_ALIGN = { │ │ │ │ │ + "l": "start", │ │ │ │ │ + "r": "end", │ │ │ │ │ + "b": "bottom", │ │ │ │ │ + "t": "hanging" │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.SVG.LABEL_VSHIFT │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_VSHIFT = { │ │ │ │ │ + // according to │ │ │ │ │ + // http://www.w3.org/Graphics/SVG/Test/20061213/htmlObjectHarness/full-text-align-02-b.html │ │ │ │ │ + // a baseline-shift of -70% shifts the text exactly from the │ │ │ │ │ + // bottom to the top of the baseline, so -35% moves the text to │ │ │ │ │ + // the center of the baseline. │ │ │ │ │ + "t": "-70%", │ │ │ │ │ + "b": "0" │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.SVG.LABEL_VFACTOR │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_VFACTOR = { │ │ │ │ │ + "t": 0, │ │ │ │ │ + "b": -1 │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Function: OpenLayers.Renderer.SVG.preventDefault │ │ │ │ │ + * *Deprecated*. Use <OpenLayers.Event.preventDefault> method instead. │ │ │ │ │ + * Used to prevent default events (especially opening images in a new tab on │ │ │ │ │ + * ctrl-click) from being executed for externalGraphic symbols │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.SVG.preventDefault = function(e) { │ │ │ │ │ + OpenLayers.Event.preventDefault(e); │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/XYZ.js │ │ │ │ │ + OpenLayers/Protocol.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.XYZ │ │ │ │ │ - * The XYZ class is designed to make it easier for people who have tiles │ │ │ │ │ - * arranged by a standard XYZ grid. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol │ │ │ │ │ + * Abstract vector layer protocol class. Not to be instantiated directly. Use │ │ │ │ │ + * one of the protocol subclasses instead. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.XYZ = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ +OpenLayers.Protocol = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * Default is true, as this is designed to be a base tile source. │ │ │ │ │ + * Property: format │ │ │ │ │ + * {<OpenLayers.Format>} The format used by this protocol. │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + format: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: sphericalMercator │ │ │ │ │ - * Whether the tile extents should be set to the defaults for │ │ │ │ │ - * spherical mercator. Useful for things like OpenStreetMap. │ │ │ │ │ - * Default is false, except for the OSM subclass. │ │ │ │ │ + * Property: options │ │ │ │ │ + * {Object} Any options sent to the constructor. │ │ │ │ │ */ │ │ │ │ │ - sphericalMercator: false, │ │ │ │ │ + options: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomOffset │ │ │ │ │ - * {Number} If your cache has more zoom levels than you want to provide │ │ │ │ │ - * access to with this layer, supply a zoomOffset. This zoom offset │ │ │ │ │ - * is added to the current map zoom level to determine the level │ │ │ │ │ - * for a requested tile. For example, if you supply a zoomOffset │ │ │ │ │ - * of 3, when the map is at the zoom 0, tiles will be requested from │ │ │ │ │ - * level 3 of your cache. Default is 0 (assumes cache level and map │ │ │ │ │ - * zoom are equivalent). Using <zoomOffset> is an alternative to │ │ │ │ │ - * setting <serverResolutions> if you only want to expose a subset │ │ │ │ │ - * of the server resolutions. │ │ │ │ │ + * Property: autoDestroy │ │ │ │ │ + * {Boolean} The creator of the protocol can set autoDestroy to false │ │ │ │ │ + * to fully control when the protocol is destroyed. Defaults to │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ - zoomOffset: 0, │ │ │ │ │ + autoDestroy: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: serverResolutions │ │ │ │ │ - * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ - * property if the map resolutions differ from the server. This │ │ │ │ │ - * property serves two purposes. (a) <serverResolutions> can include │ │ │ │ │ - * resolutions that the server supports and that you don't want to │ │ │ │ │ - * provide with this layer; you can also look at <zoomOffset>, which is │ │ │ │ │ - * an alternative to <serverResolutions> for that specific purpose. │ │ │ │ │ - * (b) The map can work with resolutions that aren't supported by │ │ │ │ │ - * the server, i.e. that aren't in <serverResolutions>. When the │ │ │ │ │ - * map is displayed in such a resolution data for the closest │ │ │ │ │ - * server-supported resolution is loaded and the layer div is │ │ │ │ │ - * stretched as necessary. │ │ │ │ │ + * Property: defaultFilter │ │ │ │ │ + * {<OpenLayers.Filter>} Optional default filter to read requests │ │ │ │ │ */ │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ + defaultFilter: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.XYZ │ │ │ │ │ + * Constructor: OpenLayers.Protocol │ │ │ │ │ + * Abstract class for vector protocols. Create instances of a subclass. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * url - {String} │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - if (options && options.sphericalMercator || this.sphericalMercator) { │ │ │ │ │ - options = OpenLayers.Util.extend({ │ │ │ │ │ - projection: "EPSG:900913", │ │ │ │ │ - numZoomLevels: 19 │ │ │ │ │ - }, options); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [ │ │ │ │ │ - name || this.name, url || this.url, {}, │ │ │ │ │ - options │ │ │ │ │ - ]); │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.options = options; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ + * Method: mergeWithDefaultFilter │ │ │ │ │ + * Merge filter passed to the read method with the default one │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {Object} Is this ever used? │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.XYZ>} An exact clone of this OpenLayers.Layer.XYZ │ │ │ │ │ + * filter - {<OpenLayers.Filter>} │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.XYZ(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ + mergeWithDefaultFilter: function(filter) { │ │ │ │ │ + var merged; │ │ │ │ │ + if (filter && this.defaultFilter) { │ │ │ │ │ + merged = new OpenLayers.Filter.Logical({ │ │ │ │ │ + type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ + filters: [this.defaultFilter, filter] │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + merged = filter || this.defaultFilter || undefined; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ + return merged; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as │ │ │ │ │ - * parameters │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up the protocol. │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var xyz = this.getXYZ(bounds); │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - var s = '' + xyz.x + xyz.y + xyz.z; │ │ │ │ │ - url = this.selectUrl(s, url); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return OpenLayers.String.format(url, xyz); │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.options = null; │ │ │ │ │ + this.format = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getXYZ │ │ │ │ │ - * Calculates x, y and z for the given bounds. │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Construct a request for reading new features. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} - an object with x, y and z properties. │ │ │ │ │ - */ │ │ │ │ │ - getXYZ: function(bounds) { │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.maxExtent.left) / │ │ │ │ │ - (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((this.maxExtent.top - bounds.top) / │ │ │ │ │ - (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getServerZoom(); │ │ │ │ │ - │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - var limit = Math.pow(2, z); │ │ │ │ │ - x = ((x % limit) + limit) % limit; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return { │ │ │ │ │ - 'x': x, │ │ │ │ │ - 'y': y, │ │ │ │ │ - 'z': z │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /* APIMethod: setMap │ │ │ │ │ - * When the layer is added to a map, then we can fetch our origin │ │ │ │ │ - * (if we don't have one.) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, the same object will be passed to the callback function passed │ │ │ │ │ + * if one exists in the options object. │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (!this.tileOrigin) { │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, │ │ │ │ │ - this.maxExtent.bottom); │ │ │ │ │ - } │ │ │ │ │ + read: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.filter = this.mergeWithDefaultFilter(options.filter); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.XYZ" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/OSM.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.OSM │ │ │ │ │ - * This layer allows accessing OpenStreetMap tiles. By default the OpenStreetMap │ │ │ │ │ - * hosted tile.openstreetmap.org Mapnik tileset is used. If you wish to use │ │ │ │ │ - * a different layer instead, you need to provide a different │ │ │ │ │ - * URL to the constructor. Here's an example for using OpenCycleMap: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * new OpenLayers.Layer.OSM("OpenCycleMap", │ │ │ │ │ - * ["http://a.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ - * "http://b.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ - * "http://c.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png"]); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.XYZ> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.OSM = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: name │ │ │ │ │ - * {String} The layer name. Defaults to "OpenStreetMap" if the first │ │ │ │ │ - * argument to the constructor is null or undefined. │ │ │ │ │ - */ │ │ │ │ │ - name: "OpenStreetMap", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String} The tileset URL scheme. Defaults to │ │ │ │ │ - * : http://[a|b|c].tile.openstreetmap.org/${z}/${x}/${y}.png │ │ │ │ │ - * (the official OSM tileset) if the second argument to the constructor │ │ │ │ │ - * is null or undefined. To use another tileset you can have something │ │ │ │ │ - * like this: │ │ │ │ │ - * (code) │ │ │ │ │ - * new OpenLayers.Layer.OSM("OpenCycleMap", │ │ │ │ │ - * ["http://a.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ - * "http://b.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ - * "http://c.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png"]); │ │ │ │ │ - * (end) │ │ │ │ │ + * APIMethod: create │ │ │ │ │ + * Construct a request for writing newly created features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, the same object will be passed to the callback function passed │ │ │ │ │ + * if one exists in the options object. │ │ │ │ │ */ │ │ │ │ │ - url: [ │ │ │ │ │ - 'http://a.tile.openstreetmap.org/${z}/${x}/${y}.png', │ │ │ │ │ - 'http://b.tile.openstreetmap.org/${z}/${x}/${y}.png', │ │ │ │ │ - 'http://c.tile.openstreetmap.org/${z}/${x}/${y}.png' │ │ │ │ │ - ], │ │ │ │ │ + create: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: attribution │ │ │ │ │ - * {String} The layer attribution. │ │ │ │ │ + * APIMethod: update │ │ │ │ │ + * Construct a request updating modified features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, the same object will be passed to the callback function passed │ │ │ │ │ + * if one exists in the options object. │ │ │ │ │ */ │ │ │ │ │ - attribution: "© <a href='http://www.openstreetmap.org/copyright'>OpenStreetMap</a> contributors", │ │ │ │ │ + update: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: sphericalMercator │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * APIMethod: delete │ │ │ │ │ + * Construct a request deleting a removed feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, the same object will be passed to the callback function passed │ │ │ │ │ + * if one exists in the options object. │ │ │ │ │ */ │ │ │ │ │ - sphericalMercator: true, │ │ │ │ │ + "delete": function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: wrapDateLine │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - wrapDateLine: true, │ │ │ │ │ - │ │ │ │ │ - /** APIProperty: tileOptions │ │ │ │ │ - * {Object} optional configuration options for <OpenLayers.Tile> instances │ │ │ │ │ - * created by this Layer. Default is │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * {crossOriginKeyword: 'anonymous'} │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * When using OSM tilesets other than the default ones, it may be │ │ │ │ │ - * necessary to set this to │ │ │ │ │ + * APIMethod: commit │ │ │ │ │ + * Go over the features and for each take action │ │ │ │ │ + * based on the feature state. Possible actions are create, │ │ │ │ │ + * update and delete. │ │ │ │ │ * │ │ │ │ │ - * (code) │ │ │ │ │ - * {crossOriginKeyword: null} │ │ │ │ │ - * (end) │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array({<OpenLayers.Feature.Vector>})} │ │ │ │ │ + * options - {Object} Object whose possible keys are "create", "update", │ │ │ │ │ + * "delete", "callback" and "scope", the values referenced by the │ │ │ │ │ + * first three are objects as passed to the "create", "update", and │ │ │ │ │ + * "delete" methods, the value referenced by the "callback" key is │ │ │ │ │ + * a function which is called when the commit operation is complete │ │ │ │ │ + * using the scope referenced by the "scope" key. │ │ │ │ │ * │ │ │ │ │ - * if the server does not send Access-Control-Allow-Origin headers. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array({<OpenLayers.Protocol.Response>})} An array of │ │ │ │ │ + * <OpenLayers.Protocol.Response> objects. │ │ │ │ │ */ │ │ │ │ │ - tileOptions: null, │ │ │ │ │ + commit: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.OSM │ │ │ │ │ + * Method: abort │ │ │ │ │ + * Abort an ongoing request. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} The layer name. │ │ │ │ │ - * url - {String} The tileset URL scheme. │ │ │ │ │ - * options - {Object} Configuration options for the layer. Any inherited │ │ │ │ │ - * layer option can be set in this object (e.g. │ │ │ │ │ - * <OpenLayers.Layer.Grid.buffer>). │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - crossOriginKeyword: 'anonymous' │ │ │ │ │ - }, this.options && this.options.tileOptions); │ │ │ │ │ - }, │ │ │ │ │ + abort: function(response) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clone │ │ │ │ │ + * Method: createCallback │ │ │ │ │ + * Returns a function that applies the given public method with resp and │ │ │ │ │ + * options arguments. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * method - {Function} The method to be applied by the callback. │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The protocol response object. │ │ │ │ │ + * options - {Object} Options sent to the protocol method │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.OSM( │ │ │ │ │ - this.name, this.url, this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj; │ │ │ │ │ + createCallback: function(method, response, options) { │ │ │ │ │ + return OpenLayers.Function.bind(function() { │ │ │ │ │ + method.apply(this, [response, options]); │ │ │ │ │ + }, this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.OSM" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol" │ │ │ │ │ }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Bing.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.Bing │ │ │ │ │ - * Bing layer using direct tile access as provided by Bing Maps REST Services. │ │ │ │ │ - * See http://msdn.microsoft.com/en-us/library/ff701713.aspx for more │ │ │ │ │ - * information. Note: Terms of Service compliant use requires the map to be │ │ │ │ │ - * configured with an <OpenLayers.Control.Attribution> control and the │ │ │ │ │ - * attribution placed on or near the map. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.XYZ> │ │ │ │ │ + * Class: OpenLayers.Protocol.Response │ │ │ │ │ + * Protocols return Response objects to their users. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Bing = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ - │ │ │ │ │ +OpenLayers.Protocol.Response = OpenLayers.Class({ │ │ │ │ │ /** │ │ │ │ │ - * Property: key │ │ │ │ │ - * {String} API key for Bing maps, get your own key │ │ │ │ │ - * at http://bingmapsportal.com/ . │ │ │ │ │ + * Property: code │ │ │ │ │ + * {Number} - OpenLayers.Protocol.Response.SUCCESS or │ │ │ │ │ + * OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ */ │ │ │ │ │ - key: null, │ │ │ │ │ + code: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: serverResolutions │ │ │ │ │ - * {Array} the resolutions provided by the Bing servers. │ │ │ │ │ + * Property: requestType │ │ │ │ │ + * {String} The type of request this response corresponds to. Either │ │ │ │ │ + * "create", "read", "update" or "delete". │ │ │ │ │ */ │ │ │ │ │ - serverResolutions: [ │ │ │ │ │ - 156543.03390625, 78271.516953125, 39135.7584765625, │ │ │ │ │ - 19567.87923828125, 9783.939619140625, 4891.9698095703125, │ │ │ │ │ - 2445.9849047851562, 1222.9924523925781, 611.4962261962891, │ │ │ │ │ - 305.74811309814453, 152.87405654907226, 76.43702827453613, │ │ │ │ │ - 38.218514137268066, 19.109257068634033, 9.554628534317017, │ │ │ │ │ - 4.777314267158508, 2.388657133579254, 1.194328566789627, │ │ │ │ │ - 0.5971642833948135, 0.29858214169740677, 0.14929107084870338, │ │ │ │ │ - 0.07464553542435169 │ │ │ │ │ - ], │ │ │ │ │ + requestType: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: attributionTemplate │ │ │ │ │ - * {String} │ │ │ │ │ + * Property: last │ │ │ │ │ + * {Boolean} - true if this is the last response expected in a commit, │ │ │ │ │ + * false otherwise, defaults to true. │ │ │ │ │ */ │ │ │ │ │ - attributionTemplate: '<span class="olBingAttribution ${type}">' + │ │ │ │ │ - '<div><a target="_blank" href="http://www.bing.com/maps/">' + │ │ │ │ │ - '<img src="${logo}" /></a></div>${copyrights}' + │ │ │ │ │ - '<a style="white-space: nowrap" target="_blank" ' + │ │ │ │ │ - 'href="http://www.microsoft.com/maps/product/terms.html">' + │ │ │ │ │ - 'Terms of Use</a></span>', │ │ │ │ │ + last: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: metadata │ │ │ │ │ - * {Object} Metadata for this layer, as returned by the callback script │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Array({<OpenLayers.Feature.Vector>})} or {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * The features returned in the response by the server. Depending on the │ │ │ │ │ + * protocol's read payload, either features or data will be populated. │ │ │ │ │ */ │ │ │ │ │ - metadata: null, │ │ │ │ │ + features: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: protocolRegex │ │ │ │ │ - * {RegExp} Regular expression to match and replace http: in bing urls │ │ │ │ │ + * Property: data │ │ │ │ │ + * {Object} │ │ │ │ │ + * The data returned in the response by the server. Depending on the │ │ │ │ │ + * protocol's read payload, either features or data will be populated. │ │ │ │ │ */ │ │ │ │ │ - protocolRegex: /^http:/i, │ │ │ │ │ + data: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: type │ │ │ │ │ - * {String} The layer identifier. Any non-birdseye imageryType │ │ │ │ │ - * from http://msdn.microsoft.com/en-us/library/ff701716.aspx can be │ │ │ │ │ - * used. Default is "Road". │ │ │ │ │ + * Property: reqFeatures │ │ │ │ │ + * {Array({<OpenLayers.Feature.Vector>})} or {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * The features provided by the user and placed in the request by the │ │ │ │ │ + * protocol. │ │ │ │ │ */ │ │ │ │ │ - type: "Road", │ │ │ │ │ + reqFeatures: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: culture │ │ │ │ │ - * {String} The culture identifier. See http://msdn.microsoft.com/en-us/library/ff701709.aspx │ │ │ │ │ - * for the definition and the possible values. Default is "en-US". │ │ │ │ │ + * Property: priv │ │ │ │ │ */ │ │ │ │ │ - culture: "en-US", │ │ │ │ │ + priv: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: metadataParams │ │ │ │ │ - * {Object} Optional url parameters for the Get Imagery Metadata request │ │ │ │ │ - * as described here: http://msdn.microsoft.com/en-us/library/ff701716.aspx │ │ │ │ │ - */ │ │ │ │ │ - metadataParams: null, │ │ │ │ │ - │ │ │ │ │ - /** APIProperty: tileOptions │ │ │ │ │ - * {Object} optional configuration options for <OpenLayers.Tile> instances │ │ │ │ │ - * created by this Layer. Default is │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * {crossOriginKeyword: 'anonymous'} │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ - tileOptions: null, │ │ │ │ │ - │ │ │ │ │ - /** APIProperty: protocol │ │ │ │ │ - * {String} Protocol to use to fetch Imagery Metadata, tiles and bing logo │ │ │ │ │ - * Can be 'http:' 'https:' or '' │ │ │ │ │ - * │ │ │ │ │ - * Warning: tiles may not be available under both HTTP and HTTPS protocols. │ │ │ │ │ - * Microsoft approved use of both HTTP and HTTPS urls for tiles. However │ │ │ │ │ - * this is undocumented and the Imagery Metadata API always returns HTTP │ │ │ │ │ - * urls. │ │ │ │ │ - * │ │ │ │ │ - * Default is '', unless when executed from a file:/// uri, in which case │ │ │ │ │ - * it is 'http:'. │ │ │ │ │ + * Property: error │ │ │ │ │ + * {Object} The error object in case a service exception was encountered. │ │ │ │ │ */ │ │ │ │ │ - protocol: ~window.location.href.indexOf('http') ? '' : 'http:', │ │ │ │ │ + error: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Bing │ │ │ │ │ - * Create a new Bing layer. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var road = new OpenLayers.Layer.Bing({ │ │ │ │ │ - * name: "My Bing Aerial Layer", │ │ │ │ │ - * type: "Aerial", │ │ │ │ │ - * key: "my-api-key-here", │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ + * Constructor: OpenLayers.Protocol.Response │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Configuration properties for the layer. │ │ │ │ │ - * │ │ │ │ │ - * Required configuration properties: │ │ │ │ │ - * key - {String} Bing Maps API key for your application. Get one at │ │ │ │ │ - * http://bingmapsportal.com/. │ │ │ │ │ - * type - {String} The layer identifier. Any non-birdseye imageryType │ │ │ │ │ - * from http://msdn.microsoft.com/en-us/library/ff701716.aspx can be │ │ │ │ │ - * used. │ │ │ │ │ - * │ │ │ │ │ - * Any other documented layer properties can be provided in the config object. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - sphericalMercator: true │ │ │ │ │ - }, options); │ │ │ │ │ - var name = options.name || "Bing " + (options.type || this.type); │ │ │ │ │ - │ │ │ │ │ - var newArgs = [name, null, options]; │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, newArgs); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - crossOriginKeyword: 'anonymous' │ │ │ │ │ - }, this.options.tileOptions); │ │ │ │ │ - this.loadMetadata(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: loadMetadata │ │ │ │ │ - */ │ │ │ │ │ - loadMetadata: function() { │ │ │ │ │ - this._callbackId = "_callback_" + this.id.replace(/\./g, "_"); │ │ │ │ │ - // link the processMetadata method to the global scope and bind it │ │ │ │ │ - // to this instance │ │ │ │ │ - window[this._callbackId] = OpenLayers.Function.bind( │ │ │ │ │ - OpenLayers.Layer.Bing.processMetadata, this │ │ │ │ │ - ); │ │ │ │ │ - var params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - key: this.key, │ │ │ │ │ - jsonp: this._callbackId, │ │ │ │ │ - include: "ImageryProviders" │ │ │ │ │ - }, this.metadataParams); │ │ │ │ │ - var url = this.protocol + "//dev.virtualearth.net/REST/v1/Imagery/Metadata/" + │ │ │ │ │ - this.type + "?" + OpenLayers.Util.getParameterString(params); │ │ │ │ │ - var script = document.createElement("script"); │ │ │ │ │ - script.type = "text/javascript"; │ │ │ │ │ - script.src = url; │ │ │ │ │ - script.id = this._callbackId; │ │ │ │ │ - document.getElementsByTagName("head")[0].appendChild(script); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: initLayer │ │ │ │ │ - * │ │ │ │ │ - * Sets layer properties according to the metadata provided by the API │ │ │ │ │ - */ │ │ │ │ │ - initLayer: function() { │ │ │ │ │ - var res = this.metadata.resourceSets[0].resources[0]; │ │ │ │ │ - var url = res.imageUrl.replace("{quadkey}", "${quadkey}"); │ │ │ │ │ - url = url.replace("{culture}", this.culture); │ │ │ │ │ - url = url.replace(this.protocolRegex, this.protocol); │ │ │ │ │ - this.url = []; │ │ │ │ │ - for (var i = 0; i < res.imageUrlSubdomains.length; ++i) { │ │ │ │ │ - this.url.push(url.replace("{subdomain}", res.imageUrlSubdomains[i])); │ │ │ │ │ - } │ │ │ │ │ - this.addOptions({ │ │ │ │ │ - maxResolution: Math.min( │ │ │ │ │ - this.serverResolutions[res.zoomMin], │ │ │ │ │ - this.maxResolution || Number.POSITIVE_INFINITY │ │ │ │ │ - ), │ │ │ │ │ - numZoomLevels: Math.min( │ │ │ │ │ - res.zoomMax + 1 - res.zoomMin, this.numZoomLevels │ │ │ │ │ - ) │ │ │ │ │ - }, true); │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - this.redraw(); │ │ │ │ │ - } │ │ │ │ │ - this.updateAttribution(); │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ + * Method: success │ │ │ │ │ * │ │ │ │ │ - * Paramters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - if (!this.url) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var xyz = this.getXYZ(bounds), │ │ │ │ │ - x = xyz.x, │ │ │ │ │ - y = xyz.y, │ │ │ │ │ - z = xyz.z; │ │ │ │ │ - var quadDigits = []; │ │ │ │ │ - for (var i = z; i > 0; --i) { │ │ │ │ │ - var digit = '0'; │ │ │ │ │ - var mask = 1 << (i - 1); │ │ │ │ │ - if ((x & mask) != 0) { │ │ │ │ │ - digit++; │ │ │ │ │ - } │ │ │ │ │ - if ((y & mask) != 0) { │ │ │ │ │ - digit++; │ │ │ │ │ - digit++; │ │ │ │ │ - } │ │ │ │ │ - quadDigits.push(digit); │ │ │ │ │ - } │ │ │ │ │ - var quadKey = quadDigits.join(""); │ │ │ │ │ - var url = this.selectUrl('' + x + y + z, this.url); │ │ │ │ │ - │ │ │ │ │ - return OpenLayers.String.format(url, { │ │ │ │ │ - 'quadkey': quadKey │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateAttribution │ │ │ │ │ - * Updates the attribution according to the requirements outlined in │ │ │ │ │ - * http://gis.638310.n2.nabble.com/Bing-imagery-td5789168.html │ │ │ │ │ - */ │ │ │ │ │ - updateAttribution: function() { │ │ │ │ │ - var metadata = this.metadata; │ │ │ │ │ - if (!metadata.resourceSets || !this.map || !this.map.center) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var res = metadata.resourceSets[0].resources[0]; │ │ │ │ │ - var extent = this.map.getExtent().transform( │ │ │ │ │ - this.map.getProjectionObject(), │ │ │ │ │ - new OpenLayers.Projection("EPSG:4326") │ │ │ │ │ - ); │ │ │ │ │ - var providers = res.imageryProviders || [], │ │ │ │ │ - zoom = OpenLayers.Util.indexOf(this.serverResolutions, │ │ │ │ │ - this.getServerResolution()), │ │ │ │ │ - copyrights = "", │ │ │ │ │ - provider, i, ii, j, jj, bbox, coverage; │ │ │ │ │ - for (i = 0, ii = providers.length; i < ii; ++i) { │ │ │ │ │ - provider = providers[i]; │ │ │ │ │ - for (j = 0, jj = provider.coverageAreas.length; j < jj; ++j) { │ │ │ │ │ - coverage = provider.coverageAreas[j]; │ │ │ │ │ - // axis order provided is Y,X │ │ │ │ │ - bbox = OpenLayers.Bounds.fromArray(coverage.bbox, true); │ │ │ │ │ - if (extent.intersectsBounds(bbox) && │ │ │ │ │ - zoom <= coverage.zoomMax && zoom >= coverage.zoomMin) { │ │ │ │ │ - copyrights += provider.attribution + " "; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var logo = metadata.brandLogoUri.replace(this.protocolRegex, this.protocol); │ │ │ │ │ - this.attribution = OpenLayers.String.format(this.attributionTemplate, { │ │ │ │ │ - type: this.type.toLowerCase(), │ │ │ │ │ - logo: logo, │ │ │ │ │ - copyrights: copyrights │ │ │ │ │ - }); │ │ │ │ │ - this.map && this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "attribution" │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - */ │ │ │ │ │ - setMap: function() { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.register("moveend", this, this.updateAttribution); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} │ │ │ │ │ - * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Bing>} An exact clone of this <OpenLayers.Layer.Bing> │ │ │ │ │ - */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Bing(this.options); │ │ │ │ │ - } │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * {Boolean} - true on success, false otherwise │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.map && │ │ │ │ │ - this.map.events.unregister("moveend", this, this.updateAttribution); │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.destroy.apply(this, arguments); │ │ │ │ │ + success: function() { │ │ │ │ │ + return this.code > 0; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Bing" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.Response" │ │ │ │ │ }); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Function: OpenLayers.Layer.Bing.processMetadata │ │ │ │ │ - * This function will be bound to an instance, linked to the global scope with │ │ │ │ │ - * an id, and called by the JSONP script returned by the API. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * metadata - {Object} metadata as returned by the API │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.Bing.processMetadata = function(metadata) { │ │ │ │ │ - this.metadata = metadata; │ │ │ │ │ - this.initLayer(); │ │ │ │ │ - var script = document.getElementById(this._callbackId); │ │ │ │ │ - script.parentNode.removeChild(script); │ │ │ │ │ - window[this._callbackId] = undefined; // cannot delete from window in IE │ │ │ │ │ - delete this._callbackId; │ │ │ │ │ -}; │ │ │ │ │ +OpenLayers.Protocol.Response.SUCCESS = 1; │ │ │ │ │ +OpenLayers.Protocol.Response.FAILURE = 0; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Renderer.js │ │ │ │ │ + OpenLayers/Request.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ + * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Renderer │ │ │ │ │ - * This is the base class for all renderers. │ │ │ │ │ - * │ │ │ │ │ - * This is based on a merger code written by Paul Spencer and Bertil Chapuis. │ │ │ │ │ - * It is largely composed of virtual functions that are to be implemented │ │ │ │ │ - * in technology-specific subclasses, but there is some generic code too. │ │ │ │ │ - * │ │ │ │ │ - * The functions that *are* implemented here merely deal with the maintenance │ │ │ │ │ - * of the size and extent variables, as well as the cached 'resolution' │ │ │ │ │ - * value. │ │ │ │ │ - * │ │ │ │ │ - * A note to the user that all subclasses should use getResolution() instead │ │ │ │ │ - * of directly accessing this.resolution in order to correctly use the │ │ │ │ │ - * cacheing system. │ │ │ │ │ - * │ │ │ │ │ + * TODO: deprecate me │ │ │ │ │ + * Use OpenLayers.Request.proxy instead. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Renderer = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: container │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - container: null, │ │ │ │ │ +OpenLayers.ProxyHost = ""; │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Request │ │ │ │ │ + * The OpenLayers.Request namespace contains convenience methods for working │ │ │ │ │ + * with XMLHttpRequests. These methods work with a cross-browser │ │ │ │ │ + * W3C compliant <OpenLayers.Request.XMLHttpRequest> class. │ │ │ │ │ + */ │ │ │ │ │ +if (!OpenLayers.Request) { │ │ │ │ │ /** │ │ │ │ │ - * Property: root │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - root: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: extent │ │ │ │ │ - * {<OpenLayers.Bounds>} │ │ │ │ │ + * This allows for OpenLayers/Request/XMLHttpRequest.js to be included │ │ │ │ │ + * before or after this script. │ │ │ │ │ */ │ │ │ │ │ - extent: null, │ │ │ │ │ + OpenLayers.Request = {}; │ │ │ │ │ +} │ │ │ │ │ +OpenLayers.Util.extend(OpenLayers.Request, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: locked │ │ │ │ │ - * {Boolean} If the renderer is currently in a state where many things │ │ │ │ │ - * are changing, the 'locked' property is set to true. This means │ │ │ │ │ - * that renderers can expect at least one more drawFeature event to be │ │ │ │ │ - * called with the 'locked' property set to 'true': In some renderers, │ │ │ │ │ - * this might make sense to use as a 'only update local information' │ │ │ │ │ - * flag. │ │ │ │ │ - */ │ │ │ │ │ - locked: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: size │ │ │ │ │ - * {<OpenLayers.Size>} │ │ │ │ │ + * Constant: DEFAULT_CONFIG │ │ │ │ │ + * {Object} Default configuration for all requests. │ │ │ │ │ */ │ │ │ │ │ - size: null, │ │ │ │ │ + DEFAULT_CONFIG: { │ │ │ │ │ + method: "GET", │ │ │ │ │ + url: window.location.href, │ │ │ │ │ + async: true, │ │ │ │ │ + user: undefined, │ │ │ │ │ + password: undefined, │ │ │ │ │ + params: null, │ │ │ │ │ + proxy: OpenLayers.ProxyHost, │ │ │ │ │ + headers: {}, │ │ │ │ │ + data: null, │ │ │ │ │ + callback: function() {}, │ │ │ │ │ + success: null, │ │ │ │ │ + failure: null, │ │ │ │ │ + scope: null │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: resolution │ │ │ │ │ - * {Float} cache of current map resolution │ │ │ │ │ + * Constant: URL_SPLIT_REGEX │ │ │ │ │ */ │ │ │ │ │ - resolution: null, │ │ │ │ │ + URL_SPLIT_REGEX: /([^:]*:)\/\/([^:]*:?[^@]*@)?([^:\/\?]*):?([^\/\?]*)/, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: map │ │ │ │ │ - * {<OpenLayers.Map>} Reference to the map -- this is set in Vector's setMap() │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} An events object that handles all │ │ │ │ │ + * events on the {<OpenLayers.Request>} object. │ │ │ │ │ + * │ │ │ │ │ + * All event listeners will receive an event object with three properties: │ │ │ │ │ + * request - {<OpenLayers.Request.XMLHttpRequest>} The request object. │ │ │ │ │ + * config - {Object} The config object sent to the specific request method. │ │ │ │ │ + * requestUrl - {String} The request url. │ │ │ │ │ + * │ │ │ │ │ + * Supported event types: │ │ │ │ │ + * complete - Triggered when we have a response from the request, if a │ │ │ │ │ + * listener returns false, no further response processing will take │ │ │ │ │ + * place. │ │ │ │ │ + * success - Triggered when the HTTP response has a success code (200-299). │ │ │ │ │ + * failure - Triggered when the HTTP response does not have a success code. │ │ │ │ │ */ │ │ │ │ │ - map: null, │ │ │ │ │ + events: new OpenLayers.Events(this), │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: featureDx │ │ │ │ │ - * {Number} Feature offset in x direction. Will be calculated for and │ │ │ │ │ - * applied to the current feature while rendering (see │ │ │ │ │ - * <calculateFeatureDx>). │ │ │ │ │ + * Method: makeSameOrigin │ │ │ │ │ + * Using the specified proxy, returns a same origin url of the provided url. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * url - {String} An arbitrary url │ │ │ │ │ + * proxy {String|Function} The proxy to use to make the provided url a │ │ │ │ │ + * same origin url. │ │ │ │ │ + * │ │ │ │ │ + * Returns │ │ │ │ │ + * {String} the same origin url. If no proxy is provided, the returned url │ │ │ │ │ + * will be the same as the provided url. │ │ │ │ │ */ │ │ │ │ │ - featureDx: 0, │ │ │ │ │ + makeSameOrigin: function(url, proxy) { │ │ │ │ │ + var sameOrigin = url.indexOf("http") !== 0; │ │ │ │ │ + var urlParts = !sameOrigin && url.match(this.URL_SPLIT_REGEX); │ │ │ │ │ + if (urlParts) { │ │ │ │ │ + var location = window.location; │ │ │ │ │ + sameOrigin = │ │ │ │ │ + urlParts[1] == location.protocol && │ │ │ │ │ + urlParts[3] == location.hostname; │ │ │ │ │ + var uPort = urlParts[4], │ │ │ │ │ + lPort = location.port; │ │ │ │ │ + if (uPort != 80 && uPort != "" || lPort != "80" && lPort != "") { │ │ │ │ │ + sameOrigin = sameOrigin && uPort == lPort; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!sameOrigin) { │ │ │ │ │ + if (proxy) { │ │ │ │ │ + if (typeof proxy == "function") { │ │ │ │ │ + url = proxy(url); │ │ │ │ │ + } else { │ │ │ │ │ + url = proxy + encodeURIComponent(url); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return url; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Renderer │ │ │ │ │ + * APIMethod: issue │ │ │ │ │ + * Create a new XMLHttpRequest object, open it, set any headers, bind │ │ │ │ │ + * a callback to done state, and send any data. It is recommended that │ │ │ │ │ + * you use one <GET>, <POST>, <PUT>, <DELETE>, <OPTIONS>, or <HEAD>. │ │ │ │ │ + * This method is only documented to provide detail on the configuration │ │ │ │ │ + * options available to all request methods. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * containerID - {<String>} │ │ │ │ │ - * options - {Object} options for this renderer. See sublcasses for │ │ │ │ │ - * supported options. │ │ │ │ │ + * config - {Object} Object containing properties for configuring the │ │ │ │ │ + * request. Allowed configuration properties are described below. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Allowed config properties: │ │ │ │ │ + * method - {String} One of GET, POST, PUT, DELETE, HEAD, or │ │ │ │ │ + * OPTIONS. Default is GET. │ │ │ │ │ + * url - {String} URL for the request. │ │ │ │ │ + * async - {Boolean} Open an asynchronous request. Default is true. │ │ │ │ │ + * user - {String} User for relevant authentication scheme. Set │ │ │ │ │ + * to null to clear current user. │ │ │ │ │ + * password - {String} Password for relevant authentication scheme. │ │ │ │ │ + * Set to null to clear current password. │ │ │ │ │ + * proxy - {String} Optional proxy. Defaults to │ │ │ │ │ + * <OpenLayers.ProxyHost>. │ │ │ │ │ + * params - {Object} Any key:value pairs to be appended to the │ │ │ │ │ + * url as a query string. Assumes url doesn't already include a query │ │ │ │ │ + * string or hash. Typically, this is only appropriate for <GET> │ │ │ │ │ + * requests where the query string will be appended to the url. │ │ │ │ │ + * Parameter values that are arrays will be │ │ │ │ │ + * concatenated with a comma (note that this goes against form-encoding) │ │ │ │ │ + * as is done with <OpenLayers.Util.getParameterString>. │ │ │ │ │ + * headers - {Object} Object with header:value pairs to be set on │ │ │ │ │ + * the request. │ │ │ │ │ + * data - {String | Document} Optional data to send with the request. │ │ │ │ │ + * Typically, this is only used with <POST> and <PUT> requests. │ │ │ │ │ + * Make sure to provide the appropriate "Content-Type" header for your │ │ │ │ │ + * data. For <POST> and <PUT> requests, the content type defaults to │ │ │ │ │ + * "application-xml". If your data is a different content type, or │ │ │ │ │ + * if you are using a different HTTP method, set the "Content-Type" │ │ │ │ │ + * header to match your data type. │ │ │ │ │ + * callback - {Function} Function to call when request is done. │ │ │ │ │ + * To determine if the request failed, check request.status (200 │ │ │ │ │ + * indicates success). │ │ │ │ │ + * success - {Function} Optional function to call if request status is in │ │ │ │ │ + * the 200s. This will be called in addition to callback above and │ │ │ │ │ + * would typically only be used as an alternative. │ │ │ │ │ + * failure - {Function} Optional function to call if request status is not │ │ │ │ │ + * in the 200s. This will be called in addition to callback above and │ │ │ │ │ + * would typically only be used as an alternative. │ │ │ │ │ + * scope - {Object} If callback is a public method on some object, │ │ │ │ │ + * set the scope to that object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLHttpRequest} Request object. To abort the request before a response │ │ │ │ │ + * is received, call abort() on the request object. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(containerID, options) { │ │ │ │ │ - this.container = OpenLayers.Util.getElement(containerID); │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ + issue: function(config) { │ │ │ │ │ + // apply default config - proxy host may have changed │ │ │ │ │ + var defaultConfig = OpenLayers.Util.extend( │ │ │ │ │ + this.DEFAULT_CONFIG, { │ │ │ │ │ + proxy: OpenLayers.ProxyHost │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + config = config || {}; │ │ │ │ │ + config.headers = config.headers || {}; │ │ │ │ │ + config = OpenLayers.Util.applyDefaults(config, defaultConfig); │ │ │ │ │ + config.headers = OpenLayers.Util.applyDefaults(config.headers, defaultConfig.headers); │ │ │ │ │ + // Always set the "X-Requested-With" header to signal that this request │ │ │ │ │ + // was issued through the XHR-object. Since header keys are case │ │ │ │ │ + // insensitive and we want to allow overriding of the "X-Requested-With" │ │ │ │ │ + // header through the user we cannot use applyDefaults, but have to │ │ │ │ │ + // check manually whether we were called with a "X-Requested-With" │ │ │ │ │ + // header. │ │ │ │ │ + var customRequestedWithHeader = false, │ │ │ │ │ + headerKey; │ │ │ │ │ + for (headerKey in config.headers) { │ │ │ │ │ + if (config.headers.hasOwnProperty(headerKey)) { │ │ │ │ │ + if (headerKey.toLowerCase() === 'x-requested-with') { │ │ │ │ │ + customRequestedWithHeader = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (customRequestedWithHeader === false) { │ │ │ │ │ + // we did not have a custom "X-Requested-With" header │ │ │ │ │ + config.headers['X-Requested-With'] = 'XMLHttpRequest'; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // create request, open, and set headers │ │ │ │ │ + var request = new OpenLayers.Request.XMLHttpRequest(); │ │ │ │ │ + var url = OpenLayers.Util.urlAppend(config.url, │ │ │ │ │ + OpenLayers.Util.getParameterString(config.params || {})); │ │ │ │ │ + url = OpenLayers.Request.makeSameOrigin(url, config.proxy); │ │ │ │ │ + request.open( │ │ │ │ │ + config.method, url, config.async, config.user, config.password │ │ │ │ │ + ); │ │ │ │ │ + for (var header in config.headers) { │ │ │ │ │ + request.setRequestHeader(header, config.headers[header]); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var events = this.events; │ │ │ │ │ + │ │ │ │ │ + // we want to execute runCallbacks with "this" as the │ │ │ │ │ + // execution scope │ │ │ │ │ + var self = this; │ │ │ │ │ + │ │ │ │ │ + request.onreadystatechange = function() { │ │ │ │ │ + if (request.readyState == OpenLayers.Request.XMLHttpRequest.DONE) { │ │ │ │ │ + var proceed = events.triggerEvent( │ │ │ │ │ + "complete", { │ │ │ │ │ + request: request, │ │ │ │ │ + config: config, │ │ │ │ │ + requestUrl: url │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + if (proceed !== false) { │ │ │ │ │ + self.runCallbacks({ │ │ │ │ │ + request: request, │ │ │ │ │ + config: config, │ │ │ │ │ + requestUrl: url │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // send request (optionally with data) and return │ │ │ │ │ + // call in a timeout for asynchronous requests so the return is │ │ │ │ │ + // available before readyState == 4 for cached docs │ │ │ │ │ + if (config.async === false) { │ │ │ │ │ + request.send(config.data); │ │ │ │ │ + } else { │ │ │ │ │ + window.setTimeout(function() { │ │ │ │ │ + if (request.readyState !== 0) { // W3C: 0-UNSENT │ │ │ │ │ + request.send(config.data); │ │ │ │ │ + } │ │ │ │ │ + }, 0); │ │ │ │ │ + } │ │ │ │ │ + return request; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ + * Method: runCallbacks │ │ │ │ │ + * Calls the complete, success and failure callbacks. Application │ │ │ │ │ + * can listen to the "complete" event, have the listener │ │ │ │ │ + * display a confirm window and always return false, and │ │ │ │ │ + * execute OpenLayers.Request.runCallbacks if the user │ │ │ │ │ + * hits "yes" in the confirm window. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Hash containing request, config and requestUrl keys │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.container = null; │ │ │ │ │ - this.extent = null; │ │ │ │ │ - this.size = null; │ │ │ │ │ - this.resolution = null; │ │ │ │ │ - this.map = null; │ │ │ │ │ + runCallbacks: function(options) { │ │ │ │ │ + var request = options.request; │ │ │ │ │ + var config = options.config; │ │ │ │ │ + │ │ │ │ │ + // bind callbacks to readyState 4 (done) │ │ │ │ │ + var complete = (config.scope) ? │ │ │ │ │ + OpenLayers.Function.bind(config.callback, config.scope) : │ │ │ │ │ + config.callback; │ │ │ │ │ + │ │ │ │ │ + // optional success callback │ │ │ │ │ + var success; │ │ │ │ │ + if (config.success) { │ │ │ │ │ + success = (config.scope) ? │ │ │ │ │ + OpenLayers.Function.bind(config.success, config.scope) : │ │ │ │ │ + config.success; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // optional failure callback │ │ │ │ │ + var failure; │ │ │ │ │ + if (config.failure) { │ │ │ │ │ + failure = (config.scope) ? │ │ │ │ │ + OpenLayers.Function.bind(config.failure, config.scope) : │ │ │ │ │ + config.failure; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (OpenLayers.Util.createUrlObject(config.url).protocol == "file:" && │ │ │ │ │ + request.responseText) { │ │ │ │ │ + request.status = 200; │ │ │ │ │ + } │ │ │ │ │ + complete(request); │ │ │ │ │ + │ │ │ │ │ + if (!request.status || (request.status >= 200 && request.status < 300)) { │ │ │ │ │ + this.events.triggerEvent("success", options); │ │ │ │ │ + if (success) { │ │ │ │ │ + success(request); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ + this.events.triggerEvent("failure", options); │ │ │ │ │ + if (failure) { │ │ │ │ │ + failure(request); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: supported │ │ │ │ │ - * This should be overridden by specific subclasses │ │ │ │ │ + * APIMethod: GET │ │ │ │ │ + * Send an HTTP GET request. Additional configuration properties are │ │ │ │ │ + * documented in the <issue> method, with the method property set │ │ │ │ │ + * to GET. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the <issue> method for documentation of allowed properties. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Whether or not the browser supports the renderer class │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ */ │ │ │ │ │ - supported: function() { │ │ │ │ │ - return false; │ │ │ │ │ + GET: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "GET" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setExtent │ │ │ │ │ - * Set the visible part of the layer. │ │ │ │ │ - * │ │ │ │ │ - * Resolution has probably changed, so we nullify the resolution │ │ │ │ │ - * cache (this.resolution) -- this way it will be re-computed when │ │ │ │ │ - * next it is needed. │ │ │ │ │ - * We nullify the resolution cache (this.resolution) if resolutionChanged │ │ │ │ │ - * is set to true - this way it will be re-computed on the next │ │ │ │ │ - * getResolution() request. │ │ │ │ │ + * APIMethod: POST │ │ │ │ │ + * Send a POST request. Additional configuration properties are │ │ │ │ │ + * documented in the <issue> method, with the method property set │ │ │ │ │ + * to POST and "Content-Type" header set to "application/xml". │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} │ │ │ │ │ - * resolutionChanged - {Boolean} │ │ │ │ │ - * │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the <issue> method for documentation of allowed properties. The │ │ │ │ │ + * default "Content-Type" header will be set to "application-xml" if │ │ │ │ │ + * none is provided. This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ - * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ - * False otherwise. │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ */ │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - this.extent = extent.clone(); │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - var ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ - extent = extent.scale(1 / ratio); │ │ │ │ │ - this.extent = extent.wrapDateLine(this.map.getMaxExtent()).scale(ratio); │ │ │ │ │ - } │ │ │ │ │ - if (resolutionChanged) { │ │ │ │ │ - this.resolution = null; │ │ │ │ │ + POST: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "POST" │ │ │ │ │ + }); │ │ │ │ │ + // set content type to application/xml if it isn't already set │ │ │ │ │ + config.headers = config.headers ? config.headers : {}; │ │ │ │ │ + if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ + config.headers["Content-Type"] = "application/xml"; │ │ │ │ │ } │ │ │ │ │ - return true; │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setSize │ │ │ │ │ - * Sets the size of the drawing surface. │ │ │ │ │ - * │ │ │ │ │ - * Resolution has probably changed, so we nullify the resolution │ │ │ │ │ - * cache (this.resolution) -- this way it will be re-computed when │ │ │ │ │ - * next it is needed. │ │ │ │ │ + * APIMethod: PUT │ │ │ │ │ + * Send an HTTP PUT request. Additional configuration properties are │ │ │ │ │ + * documented in the <issue> method, with the method property set │ │ │ │ │ + * to PUT and "Content-Type" header set to "application/xml". │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>} │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the <issue> method for documentation of allowed properties. The │ │ │ │ │ + * default "Content-Type" header will be set to "application-xml" if │ │ │ │ │ + * none is provided. This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ */ │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - this.size = size.clone(); │ │ │ │ │ - this.resolution = null; │ │ │ │ │ + PUT: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "PUT" │ │ │ │ │ + }); │ │ │ │ │ + // set content type to application/xml if it isn't already set │ │ │ │ │ + config.headers = config.headers ? config.headers : {}; │ │ │ │ │ + if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ + config.headers["Content-Type"] = "application/xml"; │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getResolution │ │ │ │ │ - * Uses cached copy of resolution if available to minimize computing │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: DELETE │ │ │ │ │ + * Send an HTTP DELETE request. Additional configuration properties are │ │ │ │ │ + * documented in the <issue> method, with the method property set │ │ │ │ │ + * to DELETE. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the <issue> method for documentation of allowed properties. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Float} The current map's resolution │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ */ │ │ │ │ │ - getResolution: function() { │ │ │ │ │ - this.resolution = this.resolution || this.map.getResolution(); │ │ │ │ │ - return this.resolution; │ │ │ │ │ + DELETE: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "DELETE" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawFeature │ │ │ │ │ - * Draw the feature. The optional style argument can be used │ │ │ │ │ - * to override the feature's own style. This method should only │ │ │ │ │ - * be called from layer.drawFeature(). │ │ │ │ │ + * APIMethod: HEAD │ │ │ │ │ + * Send an HTTP HEAD request. Additional configuration properties are │ │ │ │ │ + * documented in the <issue> method, with the method property set │ │ │ │ │ + * to HEAD. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * style - {<Object>} │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the <issue> method for documentation of allowed properties. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} true if the feature has been drawn completely, false if not, │ │ │ │ │ - * undefined if the feature had no geometry │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ */ │ │ │ │ │ - drawFeature: function(feature, style) { │ │ │ │ │ - if (style == null) { │ │ │ │ │ - style = feature.style; │ │ │ │ │ - } │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - var bounds = feature.geometry.getBounds(); │ │ │ │ │ - if (bounds) { │ │ │ │ │ - var worldBounds; │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - worldBounds = this.map.getMaxExtent(); │ │ │ │ │ - } │ │ │ │ │ - if (!bounds.intersectsBounds(this.extent, { │ │ │ │ │ - worldBounds: worldBounds │ │ │ │ │ - })) { │ │ │ │ │ - style = { │ │ │ │ │ - display: "none" │ │ │ │ │ - }; │ │ │ │ │ - } else { │ │ │ │ │ - this.calculateFeatureDx(bounds, worldBounds); │ │ │ │ │ - } │ │ │ │ │ - var rendered = this.drawGeometry(feature.geometry, style, feature.id); │ │ │ │ │ - if (style.display != "none" && style.label && rendered !== false) { │ │ │ │ │ - │ │ │ │ │ - var location = feature.geometry.getCentroid(); │ │ │ │ │ - if (style.labelXOffset || style.labelYOffset) { │ │ │ │ │ - var xOffset = isNaN(style.labelXOffset) ? 0 : style.labelXOffset; │ │ │ │ │ - var yOffset = isNaN(style.labelYOffset) ? 0 : style.labelYOffset; │ │ │ │ │ - var res = this.getResolution(); │ │ │ │ │ - location.move(xOffset * res, yOffset * res); │ │ │ │ │ - } │ │ │ │ │ - this.drawText(feature.id, style, location); │ │ │ │ │ - } else { │ │ │ │ │ - this.removeText(feature.id); │ │ │ │ │ - } │ │ │ │ │ - return rendered; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + HEAD: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "HEAD" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: calculateFeatureDx │ │ │ │ │ - * {Number} Calculates the feature offset in x direction. Looking at the │ │ │ │ │ - * center of the feature bounds and the renderer extent, we calculate how │ │ │ │ │ - * many world widths the two are away from each other. This distance is │ │ │ │ │ - * used to shift the feature as close as possible to the center of the │ │ │ │ │ - * current enderer extent, which ensures that the feature is visible in the │ │ │ │ │ - * current viewport. │ │ │ │ │ + * APIMethod: OPTIONS │ │ │ │ │ + * Send an HTTP OPTIONS request. Additional configuration properties are │ │ │ │ │ + * documented in the <issue> method, with the method property set │ │ │ │ │ + * to OPTIONS. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} Bounds of the feature │ │ │ │ │ - * worldBounds - {<OpenLayers.Bounds>} Bounds of the world │ │ │ │ │ - */ │ │ │ │ │ - calculateFeatureDx: function(bounds, worldBounds) { │ │ │ │ │ - this.featureDx = 0; │ │ │ │ │ - if (worldBounds) { │ │ │ │ │ - var worldWidth = worldBounds.getWidth(), │ │ │ │ │ - rendererCenterX = (this.extent.left + this.extent.right) / 2, │ │ │ │ │ - featureCenterX = (bounds.left + bounds.right) / 2, │ │ │ │ │ - worldsAway = Math.round((featureCenterX - rendererCenterX) / worldWidth); │ │ │ │ │ - this.featureDx = worldsAway * worldWidth; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawGeometry │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the <issue> method for documentation of allowed properties. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ * │ │ │ │ │ - * Draw a geometry. This should only be called from the renderer itself. │ │ │ │ │ - * Use layer.drawFeature() from outside the renderer. │ │ │ │ │ - * virtual function │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {<String>} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ */ │ │ │ │ │ - drawGeometry: function(geometry, style, featureId) {}, │ │ │ │ │ + OPTIONS: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "OPTIONS" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawText │ │ │ │ │ - * Function for drawing text labels. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * style - │ │ │ │ │ - * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ - */ │ │ │ │ │ - drawText: function(featureId, style, location) {}, │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeText │ │ │ │ │ - * Function for removing text labels. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - */ │ │ │ │ │ - removeText: function(featureId) {}, │ │ │ │ │ +// XMLHttpRequest.js Copyright (C) 2010 Sergey Ilinsky (http://www.ilinsky.com) │ │ │ │ │ +// │ │ │ │ │ +// Licensed under the Apache License, Version 2.0 (the "License"); │ │ │ │ │ +// you may not use this file except in compliance with the License. │ │ │ │ │ +// You may obtain a copy of the License at │ │ │ │ │ +// │ │ │ │ │ +// http://www.apache.org/licenses/LICENSE-2.0 │ │ │ │ │ +// │ │ │ │ │ +// Unless required by applicable law or agreed to in writing, software │ │ │ │ │ +// distributed under the License is distributed on an "AS IS" BASIS, │ │ │ │ │ +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. │ │ │ │ │ +// See the License for the specific language governing permissions and │ │ │ │ │ +// limitations under the License. │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clear │ │ │ │ │ - * Clear all vectors from the renderer. │ │ │ │ │ - * virtual function. │ │ │ │ │ - */ │ │ │ │ │ - clear: function() {}, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Request.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getFeatureIdFromEvent │ │ │ │ │ - * Returns a feature id from an event on the renderer. │ │ │ │ │ - * How this happens is specific to the renderer. This should be │ │ │ │ │ - * called from layer.getFeatureFromEvent(). │ │ │ │ │ - * Virtual function. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A feature id or undefined. │ │ │ │ │ - */ │ │ │ │ │ - getFeatureIdFromEvent: function(evt) {}, │ │ │ │ │ +(function() { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: eraseFeatures │ │ │ │ │ - * This is called by the layer to erase features │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ - */ │ │ │ │ │ - eraseFeatures: function(features) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ + // Save reference to earlier defined object implementation (if any) │ │ │ │ │ + var oXMLHttpRequest = window.XMLHttpRequest; │ │ │ │ │ + │ │ │ │ │ + // Define on browser type │ │ │ │ │ + var bGecko = !!window.controllers, │ │ │ │ │ + bIE = window.document.all && !window.opera, │ │ │ │ │ + bIE7 = bIE && window.navigator.userAgent.match(/MSIE 7.0/); │ │ │ │ │ + │ │ │ │ │ + // Enables "XMLHttpRequest()" call next to "new XMLHttpReques()" │ │ │ │ │ + function fXMLHttpRequest() { │ │ │ │ │ + this._object = oXMLHttpRequest && !bIE7 ? new oXMLHttpRequest : new window.ActiveXObject("Microsoft.XMLHTTP"); │ │ │ │ │ + this._listeners = []; │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // Constructor │ │ │ │ │ + function cXMLHttpRequest() { │ │ │ │ │ + return new fXMLHttpRequest; │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype = fXMLHttpRequest.prototype; │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: Firefox with Firebug installed would break pages if not executed │ │ │ │ │ + if (bGecko && oXMLHttpRequest.wrapped) │ │ │ │ │ + cXMLHttpRequest.wrapped = oXMLHttpRequest.wrapped; │ │ │ │ │ + │ │ │ │ │ + // Constants │ │ │ │ │ + cXMLHttpRequest.UNSENT = 0; │ │ │ │ │ + cXMLHttpRequest.OPENED = 1; │ │ │ │ │ + cXMLHttpRequest.HEADERS_RECEIVED = 2; │ │ │ │ │ + cXMLHttpRequest.LOADING = 3; │ │ │ │ │ + cXMLHttpRequest.DONE = 4; │ │ │ │ │ + │ │ │ │ │ + // Public Properties │ │ │ │ │ + cXMLHttpRequest.prototype.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + cXMLHttpRequest.prototype.responseText = ''; │ │ │ │ │ + cXMLHttpRequest.prototype.responseXML = null; │ │ │ │ │ + cXMLHttpRequest.prototype.status = 0; │ │ │ │ │ + cXMLHttpRequest.prototype.statusText = ''; │ │ │ │ │ + │ │ │ │ │ + // Priority proposal │ │ │ │ │ + cXMLHttpRequest.prototype.priority = "NORMAL"; │ │ │ │ │ + │ │ │ │ │ + // Instance-level Events Handlers │ │ │ │ │ + cXMLHttpRequest.prototype.onreadystatechange = null; │ │ │ │ │ + │ │ │ │ │ + // Class-level Events Handlers │ │ │ │ │ + cXMLHttpRequest.onreadystatechange = null; │ │ │ │ │ + cXMLHttpRequest.onopen = null; │ │ │ │ │ + cXMLHttpRequest.onsend = null; │ │ │ │ │ + cXMLHttpRequest.onabort = null; │ │ │ │ │ + │ │ │ │ │ + // Public Methods │ │ │ │ │ + cXMLHttpRequest.prototype.open = function(sMethod, sUrl, bAsync, sUser, sPassword) { │ │ │ │ │ + // Delete headers, required when object is reused │ │ │ │ │ + delete this._headers; │ │ │ │ │ + │ │ │ │ │ + // When bAsync parameter value is omitted, use true as default │ │ │ │ │ + if (arguments.length < 3) │ │ │ │ │ + bAsync = true; │ │ │ │ │ + │ │ │ │ │ + // Save async parameter for fixing Gecko bug with missing readystatechange in synchronous requests │ │ │ │ │ + this._async = bAsync; │ │ │ │ │ + │ │ │ │ │ + // Set the onreadystatechange handler │ │ │ │ │ + var oRequest = this, │ │ │ │ │ + nState = this.readyState, │ │ │ │ │ + fOnUnload; │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: IE - memory leak on page unload (inter-page leak) │ │ │ │ │ + if (bIE && bAsync) { │ │ │ │ │ + fOnUnload = function() { │ │ │ │ │ + if (nState != cXMLHttpRequest.DONE) { │ │ │ │ │ + fCleanTransport(oRequest); │ │ │ │ │ + // Safe to abort here since onreadystatechange handler removed │ │ │ │ │ + oRequest.abort(); │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + window.attachEvent("onunload", fOnUnload); │ │ │ │ │ } │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - this.eraseGeometry(feature.geometry, feature.id); │ │ │ │ │ - this.removeText(feature.id); │ │ │ │ │ + │ │ │ │ │ + // Add method sniffer │ │ │ │ │ + if (cXMLHttpRequest.onopen) │ │ │ │ │ + cXMLHttpRequest.onopen.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + if (arguments.length > 4) │ │ │ │ │ + this._object.open(sMethod, sUrl, bAsync, sUser, sPassword); │ │ │ │ │ + else │ │ │ │ │ + if (arguments.length > 3) │ │ │ │ │ + this._object.open(sMethod, sUrl, bAsync, sUser); │ │ │ │ │ + else │ │ │ │ │ + this._object.open(sMethod, sUrl, bAsync); │ │ │ │ │ + │ │ │ │ │ + this.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ + fReadyStateChange(this); │ │ │ │ │ + │ │ │ │ │ + this._object.onreadystatechange = function() { │ │ │ │ │ + if (bGecko && !bAsync) │ │ │ │ │ + return; │ │ │ │ │ + │ │ │ │ │ + // Synchronize state │ │ │ │ │ + oRequest.readyState = oRequest._object.readyState; │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + fSynchronizeValues(oRequest); │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: Firefox fires unnecessary DONE when aborting │ │ │ │ │ + if (oRequest._aborted) { │ │ │ │ │ + // Reset readyState to UNSENT │ │ │ │ │ + oRequest.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + │ │ │ │ │ + // Return now │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (oRequest.readyState == cXMLHttpRequest.DONE) { │ │ │ │ │ + // Free up queue │ │ │ │ │ + delete oRequest._data; │ │ │ │ │ + /* if (bAsync) │ │ │ │ │ + fQueue_remove(oRequest);*/ │ │ │ │ │ + // │ │ │ │ │ + fCleanTransport(oRequest); │ │ │ │ │ + // Uncomment this block if you need a fix for IE cache │ │ │ │ │ + /* │ │ │ │ │ + // BUGFIX: IE - cache issue │ │ │ │ │ + if (!oRequest._object.getResponseHeader("Date")) { │ │ │ │ │ + // Save object to cache │ │ │ │ │ + oRequest._cached = oRequest._object; │ │ │ │ │ + │ │ │ │ │ + // Instantiate a new transport object │ │ │ │ │ + cXMLHttpRequest.call(oRequest); │ │ │ │ │ + │ │ │ │ │ + // Re-send request │ │ │ │ │ + if (sUser) { │ │ │ │ │ + if (sPassword) │ │ │ │ │ + oRequest._object.open(sMethod, sUrl, bAsync, sUser, sPassword); │ │ │ │ │ + else │ │ │ │ │ + oRequest._object.open(sMethod, sUrl, bAsync, sUser); │ │ │ │ │ + } │ │ │ │ │ + else │ │ │ │ │ + oRequest._object.open(sMethod, sUrl, bAsync); │ │ │ │ │ + oRequest._object.setRequestHeader("If-Modified-Since", oRequest._cached.getResponseHeader("Last-Modified") || new window.Date(0)); │ │ │ │ │ + // Copy headers set │ │ │ │ │ + if (oRequest._headers) │ │ │ │ │ + for (var sHeader in oRequest._headers) │ │ │ │ │ + if (typeof oRequest._headers[sHeader] == "string") // Some frameworks prototype objects with functions │ │ │ │ │ + oRequest._object.setRequestHeader(sHeader, oRequest._headers[sHeader]); │ │ │ │ │ + │ │ │ │ │ + oRequest._object.onreadystatechange = function() { │ │ │ │ │ + // Synchronize state │ │ │ │ │ + oRequest.readyState = oRequest._object.readyState; │ │ │ │ │ + │ │ │ │ │ + if (oRequest._aborted) { │ │ │ │ │ + // │ │ │ │ │ + oRequest.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + │ │ │ │ │ + // Return │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (oRequest.readyState == cXMLHttpRequest.DONE) { │ │ │ │ │ + // Clean Object │ │ │ │ │ + fCleanTransport(oRequest); │ │ │ │ │ + │ │ │ │ │ + // get cached request │ │ │ │ │ + if (oRequest.status == 304) │ │ │ │ │ + oRequest._object = oRequest._cached; │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + delete oRequest._cached; │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + fSynchronizeValues(oRequest); │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + fReadyStateChange(oRequest); │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: IE - memory leak in interrupted │ │ │ │ │ + if (bIE && bAsync) │ │ │ │ │ + window.detachEvent("onunload", fOnUnload); │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + oRequest._object.send(null); │ │ │ │ │ + │ │ │ │ │ + // Return now - wait until re-sent request is finished │ │ │ │ │ + return; │ │ │ │ │ + }; │ │ │ │ │ + */ │ │ │ │ │ + // BUGFIX: IE - memory leak in interrupted │ │ │ │ │ + if (bIE && bAsync) │ │ │ │ │ + window.detachEvent("onunload", fOnUnload); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: Some browsers (Internet Explorer, Gecko) fire OPEN readystate twice │ │ │ │ │ + if (nState != oRequest.readyState) │ │ │ │ │ + fReadyStateChange(oRequest); │ │ │ │ │ + │ │ │ │ │ + nState = oRequest.readyState; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: eraseGeometry │ │ │ │ │ - * Remove a geometry from the renderer (by id). │ │ │ │ │ - * virtual function. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - */ │ │ │ │ │ - eraseGeometry: function(geometry, featureId) {}, │ │ │ │ │ + function fXMLHttpRequest_send(oRequest) { │ │ │ │ │ + oRequest._object.send(oRequest._data); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveRoot │ │ │ │ │ - * moves this renderer's root to a (different) renderer. │ │ │ │ │ - * To be implemented by subclasses that require a common renderer root for │ │ │ │ │ - * feature selection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * renderer - {<OpenLayers.Renderer>} target renderer for the moved root │ │ │ │ │ - */ │ │ │ │ │ - moveRoot: function(renderer) {}, │ │ │ │ │ + // BUGFIX: Gecko - missing readystatechange calls in synchronous requests │ │ │ │ │ + if (bGecko && !oRequest._async) { │ │ │ │ │ + oRequest.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getRenderLayerId │ │ │ │ │ - * Gets the layer that this renderer's output appears on. If moveRoot was │ │ │ │ │ - * used, this will be different from the id of the layer containing the │ │ │ │ │ - * features rendered by this renderer. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} the id of the output layer. │ │ │ │ │ - */ │ │ │ │ │ - getRenderLayerId: function() { │ │ │ │ │ - return this.container.id; │ │ │ │ │ - }, │ │ │ │ │ + // Synchronize state │ │ │ │ │ + fSynchronizeValues(oRequest); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: applyDefaultSymbolizer │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * symbolizer - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ - applyDefaultSymbolizer: function(symbolizer) { │ │ │ │ │ - var result = OpenLayers.Util.extend({}, │ │ │ │ │ - OpenLayers.Renderer.defaultSymbolizer); │ │ │ │ │ - if (symbolizer.stroke === false) { │ │ │ │ │ - delete result.strokeWidth; │ │ │ │ │ - delete result.strokeColor; │ │ │ │ │ + // Simulate missing states │ │ │ │ │ + while (oRequest.readyState < cXMLHttpRequest.DONE) { │ │ │ │ │ + oRequest.readyState++; │ │ │ │ │ + fReadyStateChange(oRequest); │ │ │ │ │ + // Check if we are aborted │ │ │ │ │ + if (oRequest._aborted) │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (symbolizer.fill === false) { │ │ │ │ │ - delete result.fillColor; │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.send = function(vData) { │ │ │ │ │ + // Add method sniffer │ │ │ │ │ + if (cXMLHttpRequest.onsend) │ │ │ │ │ + cXMLHttpRequest.onsend.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + if (!arguments.length) │ │ │ │ │ + vData = null; │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: Safari - fails sending documents created/modified dynamically, so an explicit serialization required │ │ │ │ │ + // BUGFIX: IE - rewrites any custom mime-type to "text/xml" in case an XMLNode is sent │ │ │ │ │ + // BUGFIX: Gecko - fails sending Element (this is up to the implementation either to standard) │ │ │ │ │ + if (vData && vData.nodeType) { │ │ │ │ │ + vData = window.XMLSerializer ? new window.XMLSerializer().serializeToString(vData) : vData.xml; │ │ │ │ │ + if (!this._headers["Content-Type"]) │ │ │ │ │ + this._object.setRequestHeader("Content-Type", "application/xml"); │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Util.extend(result, symbolizer); │ │ │ │ │ - return result; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer" │ │ │ │ │ -}); │ │ │ │ │ + this._data = vData; │ │ │ │ │ + /* │ │ │ │ │ + // Add to queue │ │ │ │ │ + if (this._async) │ │ │ │ │ + fQueue_add(this); │ │ │ │ │ + else*/ │ │ │ │ │ + fXMLHttpRequest_send(this); │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.abort = function() { │ │ │ │ │ + // Add method sniffer │ │ │ │ │ + if (cXMLHttpRequest.onabort) │ │ │ │ │ + cXMLHttpRequest.onabort.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.defaultSymbolizer │ │ │ │ │ - * {Object} Properties from this symbolizer will be applied to symbolizers │ │ │ │ │ - * with missing properties. This can also be used to set a global │ │ │ │ │ - * symbolizer default in OpenLayers. To be SLD 1.x compliant, add the │ │ │ │ │ - * following code before rendering any vector features: │ │ │ │ │ - * (code) │ │ │ │ │ - * OpenLayers.Renderer.defaultSymbolizer = { │ │ │ │ │ - * fillColor: "#808080", │ │ │ │ │ - * fillOpacity: 1, │ │ │ │ │ - * strokeColor: "#000000", │ │ │ │ │ - * strokeOpacity: 1, │ │ │ │ │ - * strokeWidth: 1, │ │ │ │ │ - * pointRadius: 3, │ │ │ │ │ - * graphicName: "square" │ │ │ │ │ - * }; │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.defaultSymbolizer = { │ │ │ │ │ - fillColor: "#000000", │ │ │ │ │ - strokeColor: "#000000", │ │ │ │ │ - strokeWidth: 2, │ │ │ │ │ - fillOpacity: 1, │ │ │ │ │ - strokeOpacity: 1, │ │ │ │ │ - pointRadius: 0, │ │ │ │ │ - labelAlign: 'cm' │ │ │ │ │ -}; │ │ │ │ │ + // BUGFIX: Gecko - unnecessary DONE when aborting │ │ │ │ │ + if (this.readyState > cXMLHttpRequest.UNSENT) │ │ │ │ │ + this._aborted = true; │ │ │ │ │ │ │ │ │ │ + this._object.abort(); │ │ │ │ │ │ │ │ │ │ + // BUGFIX: IE - memory leak │ │ │ │ │ + fCleanTransport(this); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.symbol │ │ │ │ │ - * Coordinate arrays for well known (named) symbols. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.symbol = { │ │ │ │ │ - "star": [350, 75, 379, 161, 469, 161, 397, 215, 423, 301, 350, 250, 277, 301, │ │ │ │ │ - 303, 215, 231, 161, 321, 161, 350, 75 │ │ │ │ │ - ], │ │ │ │ │ - "cross": [4, 0, 6, 0, 6, 4, 10, 4, 10, 6, 6, 6, 6, 10, 4, 10, 4, 6, 0, 6, 0, 4, 4, 4, │ │ │ │ │ - 4, 0 │ │ │ │ │ - ], │ │ │ │ │ - "x": [0, 0, 25, 0, 50, 35, 75, 0, 100, 0, 65, 50, 100, 100, 75, 100, 50, 65, 25, 100, 0, 100, 35, 50, 0, 0], │ │ │ │ │ - "square": [0, 0, 0, 1, 1, 1, 1, 0, 0, 0], │ │ │ │ │ - "triangle": [0, 10, 10, 10, 5, 0, 0, 10] │ │ │ │ │ -}; │ │ │ │ │ + this.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + │ │ │ │ │ + delete this._data; │ │ │ │ │ + /* if (this._async) │ │ │ │ │ + fQueue_remove(this);*/ │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.getAllResponseHeaders = function() { │ │ │ │ │ + return this._object.getAllResponseHeaders(); │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.getResponseHeader = function(sName) { │ │ │ │ │ + return this._object.getResponseHeader(sName); │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.setRequestHeader = function(sName, sValue) { │ │ │ │ │ + // BUGFIX: IE - cache issue │ │ │ │ │ + if (!this._headers) │ │ │ │ │ + this._headers = {}; │ │ │ │ │ + this._headers[sName] = sValue; │ │ │ │ │ + │ │ │ │ │ + return this._object.setRequestHeader(sName, sValue); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // EventTarget interface implementation │ │ │ │ │ + cXMLHttpRequest.prototype.addEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ + for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ + if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) │ │ │ │ │ + return; │ │ │ │ │ + // Add listener │ │ │ │ │ + this._listeners.push([sName, fHandler, bUseCapture]); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + cXMLHttpRequest.prototype.removeEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ + for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ + if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) │ │ │ │ │ + break; │ │ │ │ │ + // Remove listener │ │ │ │ │ + if (oListener) │ │ │ │ │ + this._listeners.splice(nIndex, 1); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + cXMLHttpRequest.prototype.dispatchEvent = function(oEvent) { │ │ │ │ │ + var oEventPseudo = { │ │ │ │ │ + 'type': oEvent.type, │ │ │ │ │ + 'target': this, │ │ │ │ │ + 'currentTarget': this, │ │ │ │ │ + 'eventPhase': 2, │ │ │ │ │ + 'bubbles': oEvent.bubbles, │ │ │ │ │ + 'cancelable': oEvent.cancelable, │ │ │ │ │ + 'timeStamp': oEvent.timeStamp, │ │ │ │ │ + 'stopPropagation': function() {}, // There is no flow │ │ │ │ │ + 'preventDefault': function() {}, // There is no default action │ │ │ │ │ + 'initEvent': function() {} // Original event object should be initialized │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // Execute onreadystatechange │ │ │ │ │ + if (oEventPseudo.type == "readystatechange" && this.onreadystatechange) │ │ │ │ │ + (this.onreadystatechange.handleEvent || this.onreadystatechange).apply(this, [oEventPseudo]); │ │ │ │ │ + │ │ │ │ │ + // Execute listeners │ │ │ │ │ + for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ + if (oListener[0] == oEventPseudo.type && !oListener[2]) │ │ │ │ │ + (oListener[1].handleEvent || oListener[1]).apply(this, [oEventPseudo]); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + cXMLHttpRequest.prototype.toString = function() { │ │ │ │ │ + return '[' + "object" + ' ' + "XMLHttpRequest" + ']'; │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + cXMLHttpRequest.toString = function() { │ │ │ │ │ + return '[' + "XMLHttpRequest" + ']'; │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // Helper function │ │ │ │ │ + function fReadyStateChange(oRequest) { │ │ │ │ │ + // Sniffing code │ │ │ │ │ + if (cXMLHttpRequest.onreadystatechange) │ │ │ │ │ + cXMLHttpRequest.onreadystatechange.apply(oRequest); │ │ │ │ │ + │ │ │ │ │ + // Fake event │ │ │ │ │ + oRequest.dispatchEvent({ │ │ │ │ │ + 'type': "readystatechange", │ │ │ │ │ + 'bubbles': false, │ │ │ │ │ + 'cancelable': false, │ │ │ │ │ + 'timeStamp': new Date + 0 │ │ │ │ │ + }); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fGetDocument(oRequest) { │ │ │ │ │ + var oDocument = oRequest.responseXML, │ │ │ │ │ + sResponse = oRequest.responseText; │ │ │ │ │ + // Try parsing responseText │ │ │ │ │ + if (bIE && sResponse && oDocument && !oDocument.documentElement && oRequest.getResponseHeader("Content-Type").match(/[^\/]+\/[^\+]+\+xml/)) { │ │ │ │ │ + oDocument = new window.ActiveXObject("Microsoft.XMLDOM"); │ │ │ │ │ + oDocument.async = false; │ │ │ │ │ + oDocument.validateOnParse = false; │ │ │ │ │ + oDocument.loadXML(sResponse); │ │ │ │ │ + } │ │ │ │ │ + // Check if there is no error in document │ │ │ │ │ + if (oDocument) │ │ │ │ │ + if ((bIE && oDocument.parseError != 0) || !oDocument.documentElement || (oDocument.documentElement && oDocument.documentElement.tagName == "parsererror")) │ │ │ │ │ + return null; │ │ │ │ │ + return oDocument; │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fSynchronizeValues(oRequest) { │ │ │ │ │ + try { │ │ │ │ │ + oRequest.responseText = oRequest._object.responseText; │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + try { │ │ │ │ │ + oRequest.responseXML = fGetDocument(oRequest._object); │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + try { │ │ │ │ │ + oRequest.status = oRequest._object.status; │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + try { │ │ │ │ │ + oRequest.statusText = oRequest._object.statusText; │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fCleanTransport(oRequest) { │ │ │ │ │ + // BUGFIX: IE - memory leak (on-page leak) │ │ │ │ │ + oRequest._object.onreadystatechange = new window.Function; │ │ │ │ │ + }; │ │ │ │ │ + /* │ │ │ │ │ + // Queue manager │ │ │ │ │ + var oQueuePending = {"CRITICAL":[],"HIGH":[],"NORMAL":[],"LOW":[],"LOWEST":[]}, │ │ │ │ │ + aQueueRunning = []; │ │ │ │ │ + function fQueue_add(oRequest) { │ │ │ │ │ + oQueuePending[oRequest.priority in oQueuePending ? oRequest.priority : "NORMAL"].push(oRequest); │ │ │ │ │ + // │ │ │ │ │ + setTimeout(fQueue_process); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fQueue_remove(oRequest) { │ │ │ │ │ + for (var nIndex = 0, bFound = false; nIndex < aQueueRunning.length; nIndex++) │ │ │ │ │ + if (bFound) │ │ │ │ │ + aQueueRunning[nIndex - 1] = aQueueRunning[nIndex]; │ │ │ │ │ + else │ │ │ │ │ + if (aQueueRunning[nIndex] == oRequest) │ │ │ │ │ + bFound = true; │ │ │ │ │ + if (bFound) │ │ │ │ │ + aQueueRunning.length--; │ │ │ │ │ + // │ │ │ │ │ + setTimeout(fQueue_process); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fQueue_process() { │ │ │ │ │ + if (aQueueRunning.length < 6) { │ │ │ │ │ + for (var sPriority in oQueuePending) { │ │ │ │ │ + if (oQueuePending[sPriority].length) { │ │ │ │ │ + var oRequest = oQueuePending[sPriority][0]; │ │ │ │ │ + oQueuePending[sPriority] = oQueuePending[sPriority].slice(1); │ │ │ │ │ + // │ │ │ │ │ + aQueueRunning.push(oRequest); │ │ │ │ │ + // Send request │ │ │ │ │ + fXMLHttpRequest_send(oRequest); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + */ │ │ │ │ │ + // Internet Explorer 5.0 (missing apply) │ │ │ │ │ + if (!window.Function.prototype.apply) { │ │ │ │ │ + window.Function.prototype.apply = function(oRequest, oArguments) { │ │ │ │ │ + if (!oArguments) │ │ │ │ │ + oArguments = []; │ │ │ │ │ + oRequest.__func = this; │ │ │ │ │ + oRequest.__func(oArguments[0], oArguments[1], oArguments[2], oArguments[3], oArguments[4]); │ │ │ │ │ + delete oRequest.__func; │ │ │ │ │ + }; │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // Register new object with window │ │ │ │ │ + /** │ │ │ │ │ + * Class: OpenLayers.Request.XMLHttpRequest │ │ │ │ │ + * Standard-compliant (W3C) cross-browser implementation of the │ │ │ │ │ + * XMLHttpRequest object. From │ │ │ │ │ + * http://code.google.com/p/xmlhttprequest/. │ │ │ │ │ + */ │ │ │ │ │ + if (!OpenLayers.Request) { │ │ │ │ │ + /** │ │ │ │ │ + * This allows for OpenLayers/Request.js to be included │ │ │ │ │ + * before or after this script. │ │ │ │ │ + */ │ │ │ │ │ + OpenLayers.Request = {}; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Request.XMLHttpRequest = cXMLHttpRequest; │ │ │ │ │ +})(); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Vector.js │ │ │ │ │ + OpenLayers/Protocol/HTTP.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer.js │ │ │ │ │ - * @requires OpenLayers/Renderer.js │ │ │ │ │ - * @requires OpenLayers/StyleMap.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Console.js │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Protocol.js │ │ │ │ │ + * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.Vector │ │ │ │ │ - * Instances of OpenLayers.Layer.Vector are used to render vector data from │ │ │ │ │ - * a variety of sources. Create a new vector layer with the │ │ │ │ │ - * <OpenLayers.Layer.Vector> constructor. │ │ │ │ │ + * if application uses the query string, for example, for BBOX parameters, │ │ │ │ │ + * OpenLayers/Format/QueryStringFilter.js should be included in the build config file │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.HTTP │ │ │ │ │ + * A basic HTTP protocol for vector layers. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Protocol.HTTP> constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer> │ │ │ │ │ + * - <OpenLayers.Protocol> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Vector = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ +OpenLayers.Protocol.HTTP = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * layer.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Listeners will be called with a reference to an event object. The │ │ │ │ │ - * properties of this event depends on exactly what happened. │ │ │ │ │ - * │ │ │ │ │ - * All event objects have at least the following properties: │ │ │ │ │ - * object - {Object} A reference to layer.events.object. │ │ │ │ │ - * element - {DOMElement} A reference to layer.events.element. │ │ │ │ │ - * │ │ │ │ │ - * Supported map event types (in addition to those from <OpenLayers.Layer.events>): │ │ │ │ │ - * beforefeatureadded - Triggered before a feature is added. Listeners │ │ │ │ │ - * will receive an object with a *feature* property referencing the │ │ │ │ │ - * feature to be added. To stop the feature from being added, a │ │ │ │ │ - * listener should return false. │ │ │ │ │ - * beforefeaturesadded - Triggered before an array of features is added. │ │ │ │ │ - * Listeners will receive an object with a *features* property │ │ │ │ │ - * referencing the feature to be added. To stop the features from │ │ │ │ │ - * being added, a listener should return false. │ │ │ │ │ - * featureadded - Triggered after a feature is added. The event │ │ │ │ │ - * object passed to listeners will have a *feature* property with a │ │ │ │ │ - * reference to the added feature. │ │ │ │ │ - * featuresadded - Triggered after features are added. The event │ │ │ │ │ - * object passed to listeners will have a *features* property with a │ │ │ │ │ - * reference to an array of added features. │ │ │ │ │ - * beforefeatureremoved - Triggered before a feature is removed. Listeners │ │ │ │ │ - * will receive an object with a *feature* property referencing the │ │ │ │ │ - * feature to be removed. │ │ │ │ │ - * beforefeaturesremoved - Triggered before multiple features are removed. │ │ │ │ │ - * Listeners will receive an object with a *features* property │ │ │ │ │ - * referencing the features to be removed. │ │ │ │ │ - * featureremoved - Triggerd after a feature is removed. The event │ │ │ │ │ - * object passed to listeners will have a *feature* property with a │ │ │ │ │ - * reference to the removed feature. │ │ │ │ │ - * featuresremoved - Triggered after features are removed. The event │ │ │ │ │ - * object passed to listeners will have a *features* property with a │ │ │ │ │ - * reference to an array of removed features. │ │ │ │ │ - * beforefeatureselected - Triggered before a feature is selected. Listeners │ │ │ │ │ - * will receive an object with a *feature* property referencing the │ │ │ │ │ - * feature to be selected. To stop the feature from being selectd, a │ │ │ │ │ - * listener should return false. │ │ │ │ │ - * featureselected - Triggered after a feature is selected. Listeners │ │ │ │ │ - * will receive an object with a *feature* property referencing the │ │ │ │ │ - * selected feature. │ │ │ │ │ - * featureunselected - Triggered after a feature is unselected. │ │ │ │ │ - * Listeners will receive an object with a *feature* property │ │ │ │ │ - * referencing the unselected feature. │ │ │ │ │ - * beforefeaturemodified - Triggered when a feature is selected to │ │ │ │ │ - * be modified. Listeners will receive an object with a *feature* │ │ │ │ │ - * property referencing the selected feature. │ │ │ │ │ - * featuremodified - Triggered when a feature has been modified. │ │ │ │ │ - * Listeners will receive an object with a *feature* property referencing │ │ │ │ │ - * the modified feature. │ │ │ │ │ - * afterfeaturemodified - Triggered when a feature is finished being modified. │ │ │ │ │ - * Listeners will receive an object with a *feature* property referencing │ │ │ │ │ - * the modified feature. │ │ │ │ │ - * vertexmodified - Triggered when a vertex within any feature geometry │ │ │ │ │ - * has been modified. Listeners will receive an object with a │ │ │ │ │ - * *feature* property referencing the modified feature, a *vertex* │ │ │ │ │ - * property referencing the vertex modified (always a point geometry), │ │ │ │ │ - * and a *pixel* property referencing the pixel location of the │ │ │ │ │ - * modification. │ │ │ │ │ - * vertexremoved - Triggered when a vertex within any feature geometry │ │ │ │ │ - * has been deleted. Listeners will receive an object with a │ │ │ │ │ - * *feature* property referencing the modified feature, a *vertex* │ │ │ │ │ - * property referencing the vertex modified (always a point geometry), │ │ │ │ │ - * and a *pixel* property referencing the pixel location of the │ │ │ │ │ - * removal. │ │ │ │ │ - * sketchstarted - Triggered when a feature sketch bound for this layer │ │ │ │ │ - * is started. Listeners will receive an object with a *feature* │ │ │ │ │ - * property referencing the new sketch feature and a *vertex* property │ │ │ │ │ - * referencing the creation point. │ │ │ │ │ - * sketchmodified - Triggered when a feature sketch bound for this layer │ │ │ │ │ - * is modified. Listeners will receive an object with a *vertex* │ │ │ │ │ - * property referencing the modified vertex and a *feature* property │ │ │ │ │ - * referencing the sketch feature. │ │ │ │ │ - * sketchcomplete - Triggered when a feature sketch bound for this layer │ │ │ │ │ - * is complete. Listeners will receive an object with a *feature* │ │ │ │ │ - * property referencing the sketch feature. By returning false, a │ │ │ │ │ - * listener can stop the sketch feature from being added to the layer. │ │ │ │ │ - * refresh - Triggered when something wants a strategy to ask the protocol │ │ │ │ │ - * for a new set of features. │ │ │ │ │ + * Property: url │ │ │ │ │ + * {String} Service URL, read-only, set through the options │ │ │ │ │ + * passed to constructor. │ │ │ │ │ */ │ │ │ │ │ + url: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} The layer is a base layer. Default is false. Set this property │ │ │ │ │ - * in the layer options. │ │ │ │ │ - */ │ │ │ │ │ - isBaseLayer: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isFixed │ │ │ │ │ - * {Boolean} Whether the layer remains in one place while dragging the │ │ │ │ │ - * map. Note that setting this to true will move the layer to the bottom │ │ │ │ │ - * of the layer stack. │ │ │ │ │ - */ │ │ │ │ │ - isFixed: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: features │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ - */ │ │ │ │ │ - features: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: filter │ │ │ │ │ - * {<OpenLayers.Filter>} The filter set in this layer, │ │ │ │ │ - * a strategy launching read requests can combined │ │ │ │ │ - * this filter with its own filter. │ │ │ │ │ - */ │ │ │ │ │ - filter: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: selectedFeatures │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * Property: headers │ │ │ │ │ + * {Object} HTTP request headers, read-only, set through the options │ │ │ │ │ + * passed to the constructor, │ │ │ │ │ + * Example: {'Content-Type': 'plain/text'} │ │ │ │ │ */ │ │ │ │ │ - selectedFeatures: null, │ │ │ │ │ + headers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: unrenderedFeatures │ │ │ │ │ - * {Object} hash of features, keyed by feature.id, that the renderer │ │ │ │ │ - * failed to draw │ │ │ │ │ + * Property: params │ │ │ │ │ + * {Object} Parameters of GET requests, read-only, set through the options │ │ │ │ │ + * passed to the constructor, │ │ │ │ │ + * Example: {'bbox': '5,5,5,5'} │ │ │ │ │ */ │ │ │ │ │ - unrenderedFeatures: null, │ │ │ │ │ + params: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: reportError │ │ │ │ │ - * {Boolean} report friendly error message when loading of renderer │ │ │ │ │ - * fails. │ │ │ │ │ + * Property: callback │ │ │ │ │ + * {Object} Function to be called when the <read>, <create>, │ │ │ │ │ + * <update>, <delete> or <commit> operation completes, read-only, │ │ │ │ │ + * set through the options passed to the constructor. │ │ │ │ │ */ │ │ │ │ │ - reportError: true, │ │ │ │ │ + callback: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: style │ │ │ │ │ - * {Object} Default style for the layer │ │ │ │ │ + /** │ │ │ │ │ + * Property: scope │ │ │ │ │ + * {Object} Callback execution scope, read-only, set through the │ │ │ │ │ + * options passed to the constructor. │ │ │ │ │ */ │ │ │ │ │ - style: null, │ │ │ │ │ + scope: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: styleMap │ │ │ │ │ - * {<OpenLayers.StyleMap>} │ │ │ │ │ + * APIProperty: readWithPOST │ │ │ │ │ + * {Boolean} true if read operations are done with POST requests │ │ │ │ │ + * instead of GET, defaults to false. │ │ │ │ │ */ │ │ │ │ │ - styleMap: null, │ │ │ │ │ + readWithPOST: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: strategies │ │ │ │ │ - * {Array(<OpenLayers.Strategy>})} Optional list of strategies for the layer. │ │ │ │ │ + * APIProperty: updateWithPOST │ │ │ │ │ + * {Boolean} true if update operations are done with POST requests │ │ │ │ │ + * defaults to false. │ │ │ │ │ */ │ │ │ │ │ - strategies: null, │ │ │ │ │ + updateWithPOST: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: protocol │ │ │ │ │ - * {<OpenLayers.Protocol>} Optional protocol for the layer. │ │ │ │ │ + * APIProperty: deleteWithPOST │ │ │ │ │ + * {Boolean} true if delete operations are done with POST requests │ │ │ │ │ + * defaults to false. │ │ │ │ │ + * if true, POST data is set to output of format.write(). │ │ │ │ │ */ │ │ │ │ │ - protocol: null, │ │ │ │ │ + deleteWithPOST: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: renderers │ │ │ │ │ - * {Array(String)} List of supported Renderer classes. Add to this list to │ │ │ │ │ - * add support for additional renderers. This list is ordered: │ │ │ │ │ - * the first renderer which returns true for the 'supported()' │ │ │ │ │ - * method will be used, if not defined in the 'renderer' option. │ │ │ │ │ + * Property: wildcarded. │ │ │ │ │ + * {Boolean} If true percent signs are added around values │ │ │ │ │ + * read from LIKE filters, for example if the protocol │ │ │ │ │ + * read method is passed a LIKE filter whose property │ │ │ │ │ + * is "foo" and whose value is "bar" the string │ │ │ │ │ + * "foo__ilike=%bar%" will be sent in the query string; │ │ │ │ │ + * defaults to false. │ │ │ │ │ */ │ │ │ │ │ - renderers: ['SVG', 'VML', 'Canvas'], │ │ │ │ │ + wildcarded: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: renderer │ │ │ │ │ - * {<OpenLayers.Renderer>} │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: srsInBBOX │ │ │ │ │ + * {Boolean} Include the SRS identifier in BBOX query string parameter. │ │ │ │ │ + * Default is false. If true and the layer has a projection object set, │ │ │ │ │ + * any BBOX filter will be serialized with a fifth item identifying the │ │ │ │ │ + * projection. E.g. bbox=-1000,-1000,1000,1000,EPSG:900913 │ │ │ │ │ */ │ │ │ │ │ - renderer: null, │ │ │ │ │ + srsInBBOX: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: rendererOptions │ │ │ │ │ - * {Object} Options for the renderer. See {<OpenLayers.Renderer>} for │ │ │ │ │ - * supported options. │ │ │ │ │ + * Constructor: OpenLayers.Protocol.HTTP │ │ │ │ │ + * A class for giving layers generic HTTP protocol. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + * │ │ │ │ │ + * Valid options include: │ │ │ │ │ + * url - {String} │ │ │ │ │ + * headers - {Object} │ │ │ │ │ + * params - {Object} URL parameters for GET requests │ │ │ │ │ + * format - {<OpenLayers.Format>} │ │ │ │ │ + * callback - {Function} │ │ │ │ │ + * scope - {Object} │ │ │ │ │ */ │ │ │ │ │ - rendererOptions: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + this.params = {}; │ │ │ │ │ + this.headers = {}; │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geometryType │ │ │ │ │ - * {String} geometryType allows you to limit the types of geometries this │ │ │ │ │ - * layer supports. This should be set to something like │ │ │ │ │ - * "OpenLayers.Geometry.Point" to limit types. │ │ │ │ │ - */ │ │ │ │ │ - geometryType: null, │ │ │ │ │ + if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ + var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ + wildcarded: this.wildcarded, │ │ │ │ │ + srsInBBOX: this.srsInBBOX │ │ │ │ │ + }); │ │ │ │ │ + this.filterToParams = function(filter, params) { │ │ │ │ │ + return format.write(filter, params); │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: drawn │ │ │ │ │ - * {Boolean} Whether the Vector Layer features have been drawn yet. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up the protocol. │ │ │ │ │ */ │ │ │ │ │ - drawn: false, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.params = null; │ │ │ │ │ + this.headers = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: ratio │ │ │ │ │ - * {Float} This specifies the ratio of the size of the visiblity of the Vector Layer features to the size of the map. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: filterToParams │ │ │ │ │ + * Optional method to translate an <OpenLayers.Filter> object into an object │ │ │ │ │ + * that can be serialized as request query string provided. If a custom │ │ │ │ │ + * method is not provided, the filter will be serialized using the │ │ │ │ │ + * <OpenLayers.Format.QueryStringFilter> class. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {<OpenLayers.Filter>} filter to convert. │ │ │ │ │ + * params - {Object} The parameters object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The resulting parameters object. │ │ │ │ │ */ │ │ │ │ │ - ratio: 1, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Vector │ │ │ │ │ - * Create a new vector layer │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Construct a request for reading new features. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} A name for the layer │ │ │ │ │ - * options - {Object} Optional object with non-default properties to set on │ │ │ │ │ - * the layer. │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * url - {String} Url for the request. │ │ │ │ │ + * params - {Object} Parameters to get serialized as a query string. │ │ │ │ │ + * headers - {Object} Headers to be set on the request. │ │ │ │ │ + * filter - {<OpenLayers.Filter>} Filter to get serialized as a │ │ │ │ │ + * query string. │ │ │ │ │ + * readWithPOST - {Boolean} If the request should be done with POST. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} A new vector layer │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} A response object, whose "priv" property │ │ │ │ │ + * references the HTTP request, this object is also passed to the │ │ │ │ │ + * callback function when the request completes, its "features" property │ │ │ │ │ + * is then populated with the features received from the server. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - // allow user-set renderer, otherwise assign one │ │ │ │ │ - if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ - this.assignRenderer(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // if no valid renderer found, display error │ │ │ │ │ - if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ - this.renderer = null; │ │ │ │ │ - this.displayError(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (!this.styleMap) { │ │ │ │ │ - this.styleMap = new OpenLayers.StyleMap(); │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options.params, this.options.params); │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + if (options.filter && this.filterToParams) { │ │ │ │ │ + options.params = this.filterToParams( │ │ │ │ │ + options.filter, options.params │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - this.features = []; │ │ │ │ │ - this.selectedFeatures = []; │ │ │ │ │ - this.unrenderedFeatures = {}; │ │ │ │ │ - │ │ │ │ │ - // Allow for custom layer behavior │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - for (var i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - this.strategies[i].setLayer(this); │ │ │ │ │ - } │ │ │ │ │ + var readWithPOST = (options.readWithPOST !== undefined) ? │ │ │ │ │ + options.readWithPOST : this.readWithPOST; │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + if (readWithPOST) { │ │ │ │ │ + var headers = options.headers || {}; │ │ │ │ │ + headers["Content-Type"] = "application/x-www-form-urlencoded"; │ │ │ │ │ + resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ + data: OpenLayers.Util.getParameterString(options.params), │ │ │ │ │ + headers: headers │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + resp.priv = OpenLayers.Request.GET({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ + return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Destroy this layer │ │ │ │ │ + * Method: handleRead │ │ │ │ │ + * Individual callbacks are created for read, create and update, should │ │ │ │ │ + * a subclass need to override each one separately. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * the user callback. │ │ │ │ │ + * options - {Object} The user options passed to the read call. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoDestroy) { │ │ │ │ │ - strategy.destroy(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.strategies = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.protocol) { │ │ │ │ │ - if (this.protocol.autoDestroy) { │ │ │ │ │ - this.protocol.destroy(); │ │ │ │ │ - } │ │ │ │ │ - this.protocol = null; │ │ │ │ │ - } │ │ │ │ │ - this.destroyFeatures(); │ │ │ │ │ - this.features = null; │ │ │ │ │ - this.selectedFeatures = null; │ │ │ │ │ - this.unrenderedFeatures = null; │ │ │ │ │ - if (this.renderer) { │ │ │ │ │ - this.renderer.destroy(); │ │ │ │ │ - } │ │ │ │ │ - this.renderer = null; │ │ │ │ │ - this.geometryType = null; │ │ │ │ │ - this.drawn = null; │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ + handleRead: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer. │ │ │ │ │ - * │ │ │ │ │ - * Note: Features of the layer are also cloned. │ │ │ │ │ + * APIMethod: create │ │ │ │ │ + * Construct a request for writing newly created features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} An exact clone of this layer │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, whose "priv" property references the HTTP request, this │ │ │ │ │ + * object is also passed to the callback function when the request │ │ │ │ │ + * completes, its "features" property is then populated with the │ │ │ │ │ + * the features received from the server. │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Vector(this.name, this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ + create: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: features, │ │ │ │ │ + requestType: "create" │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - var features = this.features; │ │ │ │ │ - var len = features.length; │ │ │ │ │ - var clonedFeatures = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - clonedFeatures[i] = features[i].clone(); │ │ │ │ │ - } │ │ │ │ │ - obj.features = clonedFeatures; │ │ │ │ │ + resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleCreate, resp, options), │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(features) │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - return obj; │ │ │ │ │ + return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: refresh │ │ │ │ │ - * Ask the layer to request features again and redraw them. Triggers │ │ │ │ │ - * the refresh event if the layer is in range and visible. │ │ │ │ │ + * Method: handleCreate │ │ │ │ │ + * Called the the request issued by <create> is complete. May be overridden │ │ │ │ │ + * by subclasses. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {Object} Optional object with properties for any listener of │ │ │ │ │ - * the refresh event. │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the create call. │ │ │ │ │ */ │ │ │ │ │ - refresh: function(obj) { │ │ │ │ │ - if (this.calculateInRange() && this.visibility) { │ │ │ │ │ - this.events.triggerEvent("refresh", obj); │ │ │ │ │ - } │ │ │ │ │ + handleCreate: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: assignRenderer │ │ │ │ │ - * Iterates through the available renderer implementations and selects │ │ │ │ │ - * and assigns the first one whose "supported()" function returns true. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: update │ │ │ │ │ + * Construct a request updating modified feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, whose "priv" property references the HTTP request, this │ │ │ │ │ + * object is also passed to the callback function when the request │ │ │ │ │ + * completes, its "features" property is then populated with the │ │ │ │ │ + * the feature received from the server. │ │ │ │ │ */ │ │ │ │ │ - assignRenderer: function() { │ │ │ │ │ - for (var i = 0, len = this.renderers.length; i < len; i++) { │ │ │ │ │ - var rendererClass = this.renderers[i]; │ │ │ │ │ - var renderer = (typeof rendererClass == "function") ? │ │ │ │ │ - rendererClass : │ │ │ │ │ - OpenLayers.Renderer[rendererClass]; │ │ │ │ │ - if (renderer && renderer.prototype.supported()) { │ │ │ │ │ - this.renderer = new renderer(this.div, this.rendererOptions); │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + update: function(feature, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var url = options.url || │ │ │ │ │ + feature.url || │ │ │ │ │ + this.options.url + "/" + feature.fid; │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: feature, │ │ │ │ │ + requestType: "update" │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + var method = this.updateWithPOST ? "POST" : "PUT"; │ │ │ │ │ + resp.priv = OpenLayers.Request[method]({ │ │ │ │ │ + url: url, │ │ │ │ │ + callback: this.createCallback(this.handleUpdate, resp, options), │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(feature) │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: displayError │ │ │ │ │ - * Let the user know their browser isn't supported. │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleUpdate │ │ │ │ │ + * Called the the request issued by <update> is complete. May be overridden │ │ │ │ │ + * by subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the update call. │ │ │ │ │ */ │ │ │ │ │ - displayError: function() { │ │ │ │ │ - if (this.reportError) { │ │ │ │ │ - OpenLayers.Console.userError(OpenLayers.i18n("browserNotSupported", { │ │ │ │ │ - renderers: this.renderers.join('\n') │ │ │ │ │ - })); │ │ │ │ │ - } │ │ │ │ │ + handleUpdate: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * The layer has been added to the map. │ │ │ │ │ - * │ │ │ │ │ - * If there is no renderer set, the layer can't be used. Remove it. │ │ │ │ │ - * Otherwise, give the renderer a reference to the map and set its size. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: delete │ │ │ │ │ + * Construct a request deleting a removed feature. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, whose "priv" property references the HTTP request, this │ │ │ │ │ + * object is also passed to the callback function when the request │ │ │ │ │ + * completes. │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ + "delete": function(feature, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var url = options.url || │ │ │ │ │ + feature.url || │ │ │ │ │ + this.options.url + "/" + feature.fid; │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ │ │ │ │ │ - if (!this.renderer) { │ │ │ │ │ - this.map.removeLayer(this); │ │ │ │ │ - } else { │ │ │ │ │ - this.renderer.map = this.map; │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: feature, │ │ │ │ │ + requestType: "delete" │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - var newSize = this.map.getSize(); │ │ │ │ │ - newSize.w = newSize.w * this.ratio; │ │ │ │ │ - newSize.h = newSize.h * this.ratio; │ │ │ │ │ - this.renderer.setSize(newSize); │ │ │ │ │ + var method = this.deleteWithPOST ? "POST" : "DELETE"; │ │ │ │ │ + var requestOptions = { │ │ │ │ │ + url: url, │ │ │ │ │ + callback: this.createCallback(this.handleDelete, resp, options), │ │ │ │ │ + headers: options.headers │ │ │ │ │ + }; │ │ │ │ │ + if (this.deleteWithPOST) { │ │ │ │ │ + requestOptions.data = this.format.write(feature); │ │ │ │ │ } │ │ │ │ │ + resp.priv = OpenLayers.Request[method](requestOptions); │ │ │ │ │ + │ │ │ │ │ + return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: afterAdd │ │ │ │ │ - * Called at the end of the map.addLayer sequence. At this point, the map │ │ │ │ │ - * will have a base layer. Any autoActivate strategies will be │ │ │ │ │ - * activated here. │ │ │ │ │ + * Method: handleDelete │ │ │ │ │ + * Called the the request issued by <delete> is complete. May be overridden │ │ │ │ │ + * by subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the delete call. │ │ │ │ │ */ │ │ │ │ │ - afterAdd: function() { │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoActivate) { │ │ │ │ │ - strategy.activate(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + handleDelete: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removeMap │ │ │ │ │ - * The layer has been removed from the map. │ │ │ │ │ + * Method: handleResponse │ │ │ │ │ + * Called by CRUD specific handlers. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the create, read, update, │ │ │ │ │ + * or delete call. │ │ │ │ │ */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - this.drawn = false; │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoActivate) { │ │ │ │ │ - strategy.deactivate(); │ │ │ │ │ + handleResponse: function(resp, options) { │ │ │ │ │ + var request = resp.priv; │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + // success │ │ │ │ │ + if (resp.requestType != "delete") { │ │ │ │ │ + resp.features = this.parseFeatures(request); │ │ │ │ │ } │ │ │ │ │ + resp.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + } else { │ │ │ │ │ + // failure │ │ │ │ │ + resp.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ } │ │ │ │ │ + options.callback.call(options.scope, resp); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onMapResize │ │ │ │ │ - * Notify the renderer of the change in size. │ │ │ │ │ - * │ │ │ │ │ + * Method: parseFeatures │ │ │ │ │ + * Read HTTP response body and return features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * request - {XMLHttpRequest} The request object │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} Array of features or a single feature. │ │ │ │ │ */ │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - OpenLayers.Layer.prototype.onMapResize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - var newSize = this.map.getSize(); │ │ │ │ │ - newSize.w = newSize.w * this.ratio; │ │ │ │ │ - newSize.h = newSize.h * this.ratio; │ │ │ │ │ - this.renderer.setSize(newSize); │ │ │ │ │ + parseFeatures: function(request) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ + } │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + return this.format.read(doc); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * Reset the vector layer's div so that it once again is lined up with │ │ │ │ │ - * the map. Notify the renderer of the change of extent, and in the │ │ │ │ │ - * case of a change of zoom level (resolution), have the │ │ │ │ │ - * renderer redraw features. │ │ │ │ │ - * │ │ │ │ │ - * If the layer has not yet been drawn, cycle through the layer's │ │ │ │ │ - * features and draw each one. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: commit │ │ │ │ │ + * Iterate over each feature and take action based on the feature state. │ │ │ │ │ + * Possible actions are create, update and delete. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * zoomChanged - {Boolean} │ │ │ │ │ - * dragging - {Boolean} │ │ │ │ │ + * features - {Array({<OpenLayers.Feature.Vector>})} │ │ │ │ │ + * options - {Object} Optional object for setting up intermediate commit │ │ │ │ │ + * callbacks. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * create - {Object} Optional object to be passed to the <create> method. │ │ │ │ │ + * update - {Object} Optional object to be passed to the <update> method. │ │ │ │ │ + * delete - {Object} Optional object to be passed to the <delete> method. │ │ │ │ │ + * callback - {Function} Optional function to be called when the commit │ │ │ │ │ + * is complete. │ │ │ │ │ + * scope - {Object} Optional object to be set as the scope of the callback. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Protocol.Response>)} An array of response objects, │ │ │ │ │ + * one per request made to the server, each object's "priv" property │ │ │ │ │ + * references the corresponding HTTP request. │ │ │ │ │ */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - var coordSysUnchanged = true; │ │ │ │ │ - if (!dragging) { │ │ │ │ │ - this.renderer.root.style.visibility = 'hidden'; │ │ │ │ │ - │ │ │ │ │ - var viewSize = this.map.getSize(), │ │ │ │ │ - viewWidth = viewSize.w, │ │ │ │ │ - viewHeight = viewSize.h, │ │ │ │ │ - offsetLeft = (viewWidth / 2 * this.ratio) - viewWidth / 2, │ │ │ │ │ - offsetTop = (viewHeight / 2 * this.ratio) - viewHeight / 2; │ │ │ │ │ - offsetLeft += this.map.layerContainerOriginPx.x; │ │ │ │ │ - offsetLeft = -Math.round(offsetLeft); │ │ │ │ │ - offsetTop += this.map.layerContainerOriginPx.y; │ │ │ │ │ - offsetTop = -Math.round(offsetTop); │ │ │ │ │ - │ │ │ │ │ - this.div.style.left = offsetLeft + 'px'; │ │ │ │ │ - this.div.style.top = offsetTop + 'px'; │ │ │ │ │ + commit: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = [], │ │ │ │ │ + nResponses = 0; │ │ │ │ │ │ │ │ │ │ - var extent = this.map.getExtent().scale(this.ratio); │ │ │ │ │ - coordSysUnchanged = this.renderer.setExtent(extent, zoomChanged); │ │ │ │ │ + // Divide up features before issuing any requests. This properly │ │ │ │ │ + // counts requests in the event that any responses come in before │ │ │ │ │ + // all requests have been issued. │ │ │ │ │ + var types = {}; │ │ │ │ │ + types[OpenLayers.State.INSERT] = []; │ │ │ │ │ + types[OpenLayers.State.UPDATE] = []; │ │ │ │ │ + types[OpenLayers.State.DELETE] = []; │ │ │ │ │ + var feature, list, requestFeatures = []; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + list = types[feature.state]; │ │ │ │ │ + if (list) { │ │ │ │ │ + list.push(feature); │ │ │ │ │ + requestFeatures.push(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // tally up number of requests │ │ │ │ │ + var nRequests = (types[OpenLayers.State.INSERT].length > 0 ? 1 : 0) + │ │ │ │ │ + types[OpenLayers.State.UPDATE].length + │ │ │ │ │ + types[OpenLayers.State.DELETE].length; │ │ │ │ │ │ │ │ │ │ - this.renderer.root.style.visibility = 'visible'; │ │ │ │ │ + // This response will be sent to the final callback after all the others │ │ │ │ │ + // have been fired. │ │ │ │ │ + var success = true; │ │ │ │ │ + var finalResponse = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: requestFeatures │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - // Force a reflow on gecko based browsers to prevent jump/flicker. │ │ │ │ │ - // This seems to happen on only certain configurations; it was originally │ │ │ │ │ - // noticed in FF 2.0 and Linux. │ │ │ │ │ - if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ - this.div.scrollLeft = this.div.scrollLeft; │ │ │ │ │ + function insertCallback(response) { │ │ │ │ │ + var len = response.features ? response.features.length : 0; │ │ │ │ │ + var fids = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + fids[i] = response.features[i].fid; │ │ │ │ │ } │ │ │ │ │ + finalResponse.insertIds = fids; │ │ │ │ │ + callback.apply(this, [response]); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (!zoomChanged && coordSysUnchanged) { │ │ │ │ │ - for (var i in this.unrenderedFeatures) { │ │ │ │ │ - var feature = this.unrenderedFeatures[i]; │ │ │ │ │ - this.drawFeature(feature); │ │ │ │ │ + function callback(response) { │ │ │ │ │ + this.callUserCallback(response, options); │ │ │ │ │ + success = success && response.success(); │ │ │ │ │ + nResponses++; │ │ │ │ │ + if (nResponses >= nRequests) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + finalResponse.code = success ? │ │ │ │ │ + OpenLayers.Protocol.Response.SUCCESS : │ │ │ │ │ + OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + options.callback.apply(options.scope, [finalResponse]); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (!this.drawn || zoomChanged || !coordSysUnchanged) { │ │ │ │ │ - this.drawn = true; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0, len = this.features.length; i < len; i++) { │ │ │ │ │ - this.renderer.locked = (i !== (len - 1)); │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - this.drawFeature(feature); │ │ │ │ │ - } │ │ │ │ │ + │ │ │ │ │ + // start issuing requests │ │ │ │ │ + var queue = types[OpenLayers.State.INSERT]; │ │ │ │ │ + if (queue.length > 0) { │ │ │ │ │ + resp.push(this.create( │ │ │ │ │ + queue, OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: insertCallback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options.create) │ │ │ │ │ + )); │ │ │ │ │ + } │ │ │ │ │ + queue = types[OpenLayers.State.UPDATE]; │ │ │ │ │ + for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ + resp.push(this.update( │ │ │ │ │ + queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: callback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options.update))); │ │ │ │ │ + } │ │ │ │ │ + queue = types[OpenLayers.State.DELETE]; │ │ │ │ │ + for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ + resp.push(this["delete"]( │ │ │ │ │ + queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: callback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options["delete"]))); │ │ │ │ │ } │ │ │ │ │ + return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: display │ │ │ │ │ - * Hide or show the Layer │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: abort │ │ │ │ │ + * Abort an ongoing request, the response object passed to │ │ │ │ │ + * this method must come from this HTTP protocol (as a result │ │ │ │ │ + * of a create, read, update, delete or commit operation). │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * display - {Boolean} │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ */ │ │ │ │ │ - display: function(display) { │ │ │ │ │ - OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ - // we need to set the display style of the root in case it is attached │ │ │ │ │ - // to a foreign layer │ │ │ │ │ - var currentDisplay = this.div.style.display; │ │ │ │ │ - if (currentDisplay != this.renderer.root.style.display) { │ │ │ │ │ - this.renderer.root.style.display = currentDisplay; │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + response.priv.abort(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: addFeatures │ │ │ │ │ - * Add Features to the layer. │ │ │ │ │ + * Method: callUserCallback │ │ │ │ │ + * This method is used from within the commit method each time an │ │ │ │ │ + * an HTTP response is received from the server, it is responsible │ │ │ │ │ + * for calling the user-supplied callbacks. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} │ │ │ │ │ + * options - {Object} The map of options passed to the commit call. │ │ │ │ │ */ │ │ │ │ │ - addFeatures: function(features, options) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ - if (notify) { │ │ │ │ │ - var event = { │ │ │ │ │ - features: features │ │ │ │ │ - }; │ │ │ │ │ - var ret = this.events.triggerEvent("beforefeaturesadded", event); │ │ │ │ │ - if (ret === false) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - features = event.features; │ │ │ │ │ + callUserCallback: function(resp, options) { │ │ │ │ │ + var opt = options[resp.requestType]; │ │ │ │ │ + if (opt && opt.callback) { │ │ │ │ │ + opt.callback.call(opt.scope, resp); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Track successfully added features for featuresadded event, since │ │ │ │ │ - // beforefeatureadded can veto single features. │ │ │ │ │ - var featuresAdded = []; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - if (i != (features.length - 1)) { │ │ │ │ │ - this.renderer.locked = true; │ │ │ │ │ - } else { │ │ │ │ │ - this.renderer.locked = false; │ │ │ │ │ - } │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.HTTP" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - if (this.geometryType && │ │ │ │ │ - !(feature.geometry instanceof this.geometryType)) { │ │ │ │ │ - throw new TypeError('addFeatures: component should be an ' + │ │ │ │ │ - this.geometryType.prototype.CLASS_NAME); │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - //give feature reference to its layer │ │ │ │ │ - feature.layer = this; │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (!feature.style && this.style) { │ │ │ │ │ - feature.style = OpenLayers.Util.extend({}, this.style); │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format │ │ │ │ │ + * Base class for format reading/writing a variety of formats. Subclasses │ │ │ │ │ + * of OpenLayers.Format are expected to have read and write methods. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ - if (notify) { │ │ │ │ │ - if (this.events.triggerEvent("beforefeatureadded", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) === false) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - this.preFeatureInsert(feature); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: options │ │ │ │ │ + * {Object} A reference to options passed to the constructor. │ │ │ │ │ + */ │ │ │ │ │ + options: null, │ │ │ │ │ │ │ │ │ │ - featuresAdded.push(feature); │ │ │ │ │ - this.features.push(feature); │ │ │ │ │ - this.drawFeature(feature); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: externalProjection │ │ │ │ │ + * {<OpenLayers.Projection>} When passed a externalProjection and │ │ │ │ │ + * internalProjection, the format will reproject the geometries it │ │ │ │ │ + * reads or writes. The externalProjection is the projection used by │ │ │ │ │ + * the content which is passed into read or which comes out of write. │ │ │ │ │ + * In order to reproject, a projection transformation function for the │ │ │ │ │ + * specified projections must be available. This support may be │ │ │ │ │ + * provided via proj4js or via a custom transformation function. See │ │ │ │ │ + * {<OpenLayers.Projection.addTransform>} for more information on │ │ │ │ │ + * custom transformations. │ │ │ │ │ + */ │ │ │ │ │ + externalProjection: null, │ │ │ │ │ │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureadded", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - this.onFeatureInsert(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: internalProjection │ │ │ │ │ + * {<OpenLayers.Projection>} When passed a externalProjection and │ │ │ │ │ + * internalProjection, the format will reproject the geometries it │ │ │ │ │ + * reads or writes. The internalProjection is the projection used by │ │ │ │ │ + * the geometries which are returned by read or which are passed into │ │ │ │ │ + * write. In order to reproject, a projection transformation function │ │ │ │ │ + * for the specified projections must be available. This support may be │ │ │ │ │ + * provided via proj4js or via a custom transformation function. See │ │ │ │ │ + * {<OpenLayers.Projection.addTransform>} for more information on │ │ │ │ │ + * custom transformations. │ │ │ │ │ + */ │ │ │ │ │ + internalProjection: null, │ │ │ │ │ │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresadded", { │ │ │ │ │ - features: featuresAdded │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: data │ │ │ │ │ + * {Object} When <keepData> is true, this is the parsed string sent to │ │ │ │ │ + * <read>. │ │ │ │ │ + */ │ │ │ │ │ + data: null, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: keepData │ │ │ │ │ + * {Object} Maintain a reference (<data>) to the most recently read data. │ │ │ │ │ + * Default is false. │ │ │ │ │ + */ │ │ │ │ │ + keepData: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: removeFeatures │ │ │ │ │ - * Remove features from the layer. This erases any drawn features and │ │ │ │ │ - * removes them from the layer's control. The beforefeatureremoved │ │ │ │ │ - * and featureremoved events will be triggered for each feature. The │ │ │ │ │ - * featuresremoved event will be triggered after all features have │ │ │ │ │ - * been removed. To supress event triggering, use the silent option. │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Format │ │ │ │ │ + * Instances of this class are not useful. See one of the subclasses. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} List of features to be │ │ │ │ │ - * removed. │ │ │ │ │ - * options - {Object} Optional properties for changing behavior of the │ │ │ │ │ - * removal. │ │ │ │ │ + * options - {Object} An optional object with properties to set on the │ │ │ │ │ + * format │ │ │ │ │ * │ │ │ │ │ * Valid options: │ │ │ │ │ - * silent - {Boolean} Supress event triggering. Default is false. │ │ │ │ │ + * keepData - {Boolean} If true, upon <read>, the data property will be │ │ │ │ │ + * set to the parsed object (e.g. the json or xml object). │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * An instance of OpenLayers.Format │ │ │ │ │ */ │ │ │ │ │ - removeFeatures: function(features, options) { │ │ │ │ │ - if (!features || features.length === 0) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (features === this.features) { │ │ │ │ │ - return this.removeAllFeatures(options); │ │ │ │ │ - } │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ - if (features === this.selectedFeatures) { │ │ │ │ │ - features = features.slice(); │ │ │ │ │ - } │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.options = options; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() {}, │ │ │ │ │ │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent( │ │ │ │ │ - "beforefeaturesremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: read │ │ │ │ │ + * Read data from a string, and return an object whose type depends on the │ │ │ │ │ + * subclass. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {string} Data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * Depends on the subclass │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + throw new Error('Read not implemented.'); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - // We remain locked so long as we're not at 0 │ │ │ │ │ - // and the 'next' feature has a geometry. We do the geometry check │ │ │ │ │ - // because if all the features after the current one are 'null', we │ │ │ │ │ - // won't call eraseGeometry, so we break the 'renderer functions │ │ │ │ │ - // will always be called with locked=false *last*' rule. The end result │ │ │ │ │ - // is a possible gratiutious unlocking to save a loop through the rest │ │ │ │ │ - // of the list checking the remaining features every time. So long as │ │ │ │ │ - // null geoms are rare, this is probably okay. │ │ │ │ │ - if (i != 0 && features[i - 1].geometry) { │ │ │ │ │ - this.renderer.locked = true; │ │ │ │ │ - } else { │ │ │ │ │ - this.renderer.locked = false; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: write │ │ │ │ │ + * Accept an object, and return a string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * object - {Object} Object to be serialized │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string representation of the object. │ │ │ │ │ + */ │ │ │ │ │ + write: function(object) { │ │ │ │ │ + throw new Error('Write not implemented.'); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/JSON.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - this.features = OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ - // feature has no layer at this point │ │ │ │ │ - feature.layer = null; │ │ │ │ │ +/** │ │ │ │ │ + * Note: │ │ │ │ │ + * This work draws heavily from the public domain JSON serializer/deserializer │ │ │ │ │ + * at http://www.json.org/json.js. Rewritten so that it doesn't modify │ │ │ │ │ + * basic data prototypes. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - this.renderer.eraseFeatures(feature); │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - //in the case that this feature is one of the selected features, │ │ │ │ │ - // remove it from that array as well. │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.selectedFeatures, feature) != -1) { │ │ │ │ │ - OpenLayers.Util.removeItem(this.selectedFeatures, feature); │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.JSON │ │ │ │ │ + * A parser to read/write JSON safely. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Format.JSON> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.JSON = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: indent │ │ │ │ │ + * {String} For "pretty" printing, the indent string will be used once for │ │ │ │ │ + * each indentation level. │ │ │ │ │ + */ │ │ │ │ │ + indent: " ", │ │ │ │ │ │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: space │ │ │ │ │ + * {String} For "pretty" printing, the space string will be used after │ │ │ │ │ + * the ":" separating a name/value pair. │ │ │ │ │ + */ │ │ │ │ │ + space: " ", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: removeAllFeatures │ │ │ │ │ - * Remove all features from the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional properties for changing behavior of the │ │ │ │ │ - * removal. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * silent - {Boolean} Supress event triggering. Default is false. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: newline │ │ │ │ │ + * {String} For "pretty" printing, the newline string will be used at the │ │ │ │ │ + * end of each name/value pair or array item. │ │ │ │ │ */ │ │ │ │ │ - removeAllFeatures: function(options) { │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ - var features = this.features; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent( │ │ │ │ │ - "beforefeaturesremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - feature.layer = null; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.renderer.clear(); │ │ │ │ │ - this.features = []; │ │ │ │ │ - this.unrenderedFeatures = {}; │ │ │ │ │ - this.selectedFeatures = []; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + newline: "\n", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroyFeatures │ │ │ │ │ - * Erase and destroy features on the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} An optional array of │ │ │ │ │ - * features to destroy. If not supplied, all features on the layer │ │ │ │ │ - * will be destroyed. │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * Property: level │ │ │ │ │ + * {Integer} For "pretty" printing, this is incremented/decremented during │ │ │ │ │ + * serialization. │ │ │ │ │ */ │ │ │ │ │ - destroyFeatures: function(features, options) { │ │ │ │ │ - var all = (features == undefined); // evaluates to true if │ │ │ │ │ - // features is null │ │ │ │ │ - if (all) { │ │ │ │ │ - features = this.features; │ │ │ │ │ - } │ │ │ │ │ - if (features) { │ │ │ │ │ - this.removeFeatures(features, options); │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - features[i].destroy(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + level: 0, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: drawFeature │ │ │ │ │ - * Draw (or redraw) a feature on the layer. If the optional style argument │ │ │ │ │ - * is included, this style will be used. If no style is included, the │ │ │ │ │ - * feature's style will be used. If the feature doesn't have a style, │ │ │ │ │ - * the layer's style will be used. │ │ │ │ │ - * │ │ │ │ │ - * This function is not designed to be used when adding features to │ │ │ │ │ - * the layer (use addFeatures instead). It is meant to be used when │ │ │ │ │ - * the style of a feature has changed, or in some other way needs to │ │ │ │ │ - * visually updated *after* it has already been added to a layer. You │ │ │ │ │ - * must add the feature to the layer for most layer-related events to │ │ │ │ │ - * happen. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * style - {String | Object} Named render intent or full symbolizer object. │ │ │ │ │ + * Property: pretty │ │ │ │ │ + * {Boolean} Serialize with extra whitespace for structure. This is set │ │ │ │ │ + * by the <write> method. │ │ │ │ │ */ │ │ │ │ │ - drawFeature: function(feature, style) { │ │ │ │ │ - // don't try to draw the feature with the renderer if the layer is not │ │ │ │ │ - // drawn itself │ │ │ │ │ - if (!this.drawn) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (typeof style != "object") { │ │ │ │ │ - if (!style && feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ - style = "delete"; │ │ │ │ │ - } │ │ │ │ │ - var renderIntent = style || feature.renderIntent; │ │ │ │ │ - style = feature.style || this.style; │ │ │ │ │ - if (!style) { │ │ │ │ │ - style = this.styleMap.createSymbolizer(feature, renderIntent); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + pretty: false, │ │ │ │ │ │ │ │ │ │ - var drawn = this.renderer.drawFeature(feature, style); │ │ │ │ │ - //TODO remove the check for null when we get rid of Renderer.SVG │ │ │ │ │ - if (drawn === false || drawn === null) { │ │ │ │ │ - this.unrenderedFeatures[feature.id] = feature; │ │ │ │ │ - } else { │ │ │ │ │ - delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: nativeJSON │ │ │ │ │ + * {Boolean} Does the browser support native json? │ │ │ │ │ + */ │ │ │ │ │ + nativeJSON: (function() { │ │ │ │ │ + return !!(window.JSON && typeof JSON.parse == "function" && typeof JSON.stringify == "function"); │ │ │ │ │ + })(), │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: eraseFeatures │ │ │ │ │ - * Erase features from the layer. │ │ │ │ │ + * Constructor: OpenLayers.Format.JSON │ │ │ │ │ + * Create a new parser for JSON. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - eraseFeatures: function(features) { │ │ │ │ │ - this.renderer.eraseFeatures(features); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureFromEvent │ │ │ │ │ - * Given an event, return a feature if the event occurred over one. │ │ │ │ │ - * Otherwise, return null. │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Deserialize a json string. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ + * json - {String} A JSON string │ │ │ │ │ + * filter - {Function} A function which will be called for every key and │ │ │ │ │ + * value at every level of the final result. Each value will be │ │ │ │ │ + * replaced by the result of the filter function. This can be used to │ │ │ │ │ + * reform generic objects into instances of classes, or to transform │ │ │ │ │ + * date strings into Date objects. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A feature if one was under the event. │ │ │ │ │ + * {Object} An object, array, string, or number . │ │ │ │ │ */ │ │ │ │ │ - getFeatureFromEvent: function(evt) { │ │ │ │ │ - if (!this.renderer) { │ │ │ │ │ - throw new Error('getFeatureFromEvent called on layer with no ' + │ │ │ │ │ - 'renderer. This usually means you destroyed a ' + │ │ │ │ │ - 'layer, but not some handler which is associated ' + │ │ │ │ │ - 'with it.'); │ │ │ │ │ - } │ │ │ │ │ - var feature = null; │ │ │ │ │ - var featureId = this.renderer.getFeatureIdFromEvent(evt); │ │ │ │ │ - if (featureId) { │ │ │ │ │ - if (typeof featureId === "string") { │ │ │ │ │ - feature = this.getFeatureById(featureId); │ │ │ │ │ - } else { │ │ │ │ │ - feature = featureId; │ │ │ │ │ + read: function(json, filter) { │ │ │ │ │ + var object; │ │ │ │ │ + if (this.nativeJSON) { │ │ │ │ │ + object = JSON.parse(json, filter); │ │ │ │ │ + } else try { │ │ │ │ │ + /** │ │ │ │ │ + * Parsing happens in three stages. In the first stage, we run the │ │ │ │ │ + * text against a regular expression which looks for non-JSON │ │ │ │ │ + * characters. We are especially concerned with '()' and 'new' │ │ │ │ │ + * because they can cause invocation, and '=' because it can │ │ │ │ │ + * cause mutation. But just to be safe, we will reject all │ │ │ │ │ + * unexpected characters. │ │ │ │ │ + */ │ │ │ │ │ + if (/^[\],:{}\s]*$/.test(json.replace(/\\["\\\/bfnrtu]/g, '@').replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, ']').replace(/(?:^|:|,)(?:\s*\[)+/g, ''))) { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * In the second stage we use the eval function to compile the │ │ │ │ │ + * text into a JavaScript structure. The '{' operator is │ │ │ │ │ + * subject to a syntactic ambiguity in JavaScript - it can │ │ │ │ │ + * begin a block or an object literal. We wrap the text in │ │ │ │ │ + * parens to eliminate the ambiguity. │ │ │ │ │ + */ │ │ │ │ │ + object = eval('(' + json + ')'); │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * In the optional third stage, we recursively walk the new │ │ │ │ │ + * structure, passing each name/value pair to a filter │ │ │ │ │ + * function for possible transformation. │ │ │ │ │ + */ │ │ │ │ │ + if (typeof filter === 'function') { │ │ │ │ │ + function walk(k, v) { │ │ │ │ │ + if (v && typeof v === 'object') { │ │ │ │ │ + for (var i in v) { │ │ │ │ │ + if (v.hasOwnProperty(i)) { │ │ │ │ │ + v[i] = walk(i, v[i]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return filter(k, v); │ │ │ │ │ + } │ │ │ │ │ + object = walk('', object); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + } catch (e) { │ │ │ │ │ + // Fall through if the regexp test fails. │ │ │ │ │ } │ │ │ │ │ - return feature; │ │ │ │ │ + │ │ │ │ │ + if (this.keepData) { │ │ │ │ │ + this.data = object; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return object; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getFeatureBy │ │ │ │ │ - * Given a property value, return the feature if it exists in the features array │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Serialize an object into a JSON string. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * property - {String} │ │ │ │ │ - * value - {String} │ │ │ │ │ + * value - {String} The object, array, string, number, boolean or date │ │ │ │ │ + * to be serialized. │ │ │ │ │ + * pretty - {Boolean} Structure the output with newlines and indentation. │ │ │ │ │ + * Default is false. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A feature corresponding to the given │ │ │ │ │ - * property value or null if there is no such feature. │ │ │ │ │ + * {String} The JSON string representation of the input value. │ │ │ │ │ */ │ │ │ │ │ - getFeatureBy: function(property, value) { │ │ │ │ │ - //TBD - would it be more efficient to use a hash for this.features? │ │ │ │ │ - var feature = null; │ │ │ │ │ - for (var i = 0, len = this.features.length; i < len; ++i) { │ │ │ │ │ - if (this.features[i][property] == value) { │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - break; │ │ │ │ │ + write: function(value, pretty) { │ │ │ │ │ + this.pretty = !!pretty; │ │ │ │ │ + var json = null; │ │ │ │ │ + var type = typeof value; │ │ │ │ │ + if (this.serialize[type]) { │ │ │ │ │ + try { │ │ │ │ │ + json = (!this.pretty && this.nativeJSON) ? │ │ │ │ │ + JSON.stringify(value) : │ │ │ │ │ + this.serialize[type].apply(this, [value]); │ │ │ │ │ + } catch (err) { │ │ │ │ │ + OpenLayers.Console.error("Trouble serializing: " + err); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return feature; │ │ │ │ │ + return json; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getFeatureById │ │ │ │ │ - * Given a feature id, return the feature if it exists in the features array │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * Method: writeIndent │ │ │ │ │ + * Output an indentation string depending on the indentation level. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A feature corresponding to the given │ │ │ │ │ - * featureId or null if there is no such feature. │ │ │ │ │ + * {String} An appropriate indentation string. │ │ │ │ │ */ │ │ │ │ │ - getFeatureById: function(featureId) { │ │ │ │ │ - return this.getFeatureBy('id', featureId); │ │ │ │ │ + writeIndent: function() { │ │ │ │ │ + var pieces = []; │ │ │ │ │ + if (this.pretty) { │ │ │ │ │ + for (var i = 0; i < this.level; ++i) { │ │ │ │ │ + pieces.push(this.indent); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return pieces.join(''); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getFeatureByFid │ │ │ │ │ - * Given a feature fid, return the feature if it exists in the features array │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureFid - {String} │ │ │ │ │ + * Method: writeNewline │ │ │ │ │ + * Output a string representing a newline if in pretty printing mode. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A feature corresponding to the given │ │ │ │ │ - * featureFid or null if there is no such feature. │ │ │ │ │ + * {String} A string representing a new line. │ │ │ │ │ */ │ │ │ │ │ - getFeatureByFid: function(featureFid) { │ │ │ │ │ - return this.getFeatureBy('fid', featureFid); │ │ │ │ │ + writeNewline: function() { │ │ │ │ │ + return (this.pretty) ? this.newline : ''; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getFeaturesByAttribute │ │ │ │ │ - * Returns an array of features that have the given attribute key set to the │ │ │ │ │ - * given value. Comparison of attribute values takes care of datatypes, e.g. │ │ │ │ │ - * the string '1234' is not equal to the number 1234. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * attrName - {String} │ │ │ │ │ - * attrValue - {Mixed} │ │ │ │ │ + * Method: writeSpace │ │ │ │ │ + * Output a string representing a space if in pretty printing mode. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * Array({<OpenLayers.Feature.Vector>}) An array of features that have the │ │ │ │ │ - * passed named attribute set to the given value. │ │ │ │ │ + * {String} A space. │ │ │ │ │ */ │ │ │ │ │ - getFeaturesByAttribute: function(attrName, attrValue) { │ │ │ │ │ - var i, │ │ │ │ │ - feature, │ │ │ │ │ - len = this.features.length, │ │ │ │ │ - foundFeatures = []; │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - if (feature && feature.attributes) { │ │ │ │ │ - if (feature.attributes[attrName] === attrValue) { │ │ │ │ │ - foundFeatures.push(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return foundFeatures; │ │ │ │ │ + writeSpace: function() { │ │ │ │ │ + return (this.pretty) ? this.space : ''; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Unselect the selected features │ │ │ │ │ - * i.e. clears the featureSelection array │ │ │ │ │ - * change the style back │ │ │ │ │ - clearSelection: function() { │ │ │ │ │ - │ │ │ │ │ - var vectorLayer = this.map.vectorLayer; │ │ │ │ │ - for (var i = 0; i < this.map.featureSelection.length; i++) { │ │ │ │ │ - var featureSelection = this.map.featureSelection[i]; │ │ │ │ │ - vectorLayer.drawFeature(featureSelection, vectorLayer.style); │ │ │ │ │ - } │ │ │ │ │ - this.map.featureSelection = []; │ │ │ │ │ - }, │ │ │ │ │ + * Property: serialize │ │ │ │ │ + * Object with properties corresponding to the serializable data types. │ │ │ │ │ + * Property values are functions that do the actual serializing. │ │ │ │ │ */ │ │ │ │ │ + serialize: { │ │ │ │ │ + /** │ │ │ │ │ + * Method: serialize.object │ │ │ │ │ + * Transform an object into a JSON string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * object - {Object} The object to be serialized. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A JSON string representing the object. │ │ │ │ │ + */ │ │ │ │ │ + 'object': function(object) { │ │ │ │ │ + // three special objects that we want to treat differently │ │ │ │ │ + if (object == null) { │ │ │ │ │ + return "null"; │ │ │ │ │ + } │ │ │ │ │ + if (object.constructor == Date) { │ │ │ │ │ + return this.serialize.date.apply(this, [object]); │ │ │ │ │ + } │ │ │ │ │ + if (object.constructor == Array) { │ │ │ │ │ + return this.serialize.array.apply(this, [object]); │ │ │ │ │ + } │ │ │ │ │ + var pieces = ['{']; │ │ │ │ │ + this.level += 1; │ │ │ │ │ + var key, keyJSON, valueJSON; │ │ │ │ │ │ │ │ │ │ + var addComma = false; │ │ │ │ │ + for (key in object) { │ │ │ │ │ + if (object.hasOwnProperty(key)) { │ │ │ │ │ + // recursive calls need to allow for sub-classing │ │ │ │ │ + keyJSON = OpenLayers.Format.JSON.prototype.write.apply(this, │ │ │ │ │ + [key, this.pretty]); │ │ │ │ │ + valueJSON = OpenLayers.Format.JSON.prototype.write.apply(this, │ │ │ │ │ + [object[key], this.pretty]); │ │ │ │ │ + if (keyJSON != null && valueJSON != null) { │ │ │ │ │ + if (addComma) { │ │ │ │ │ + pieces.push(','); │ │ │ │ │ + } │ │ │ │ │ + pieces.push(this.writeNewline(), this.writeIndent(), │ │ │ │ │ + keyJSON, ':', this.writeSpace(), valueJSON); │ │ │ │ │ + addComma = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: onFeatureInsert │ │ │ │ │ - * method called after a feature is inserted. │ │ │ │ │ - * Does nothing by default. Override this if you │ │ │ │ │ - * need to do something on feature updates. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - onFeatureInsert: function(feature) {}, │ │ │ │ │ + this.level -= 1; │ │ │ │ │ + pieces.push(this.writeNewline(), this.writeIndent(), '}'); │ │ │ │ │ + return pieces.join(''); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: preFeatureInsert │ │ │ │ │ - * method called before a feature is inserted. │ │ │ │ │ - * Does nothing by default. Override this if you │ │ │ │ │ - * need to do something when features are first added to the │ │ │ │ │ - * layer, but before they are drawn, such as adjust the style. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - preFeatureInsert: function(feature) {}, │ │ │ │ │ + /** │ │ │ │ │ + * Method: serialize.array │ │ │ │ │ + * Transform an array into a JSON string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * array - {Array} The array to be serialized │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A JSON string representing the array. │ │ │ │ │ + */ │ │ │ │ │ + 'array': function(array) { │ │ │ │ │ + var json; │ │ │ │ │ + var pieces = ['[']; │ │ │ │ │ + this.level += 1; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getDataExtent │ │ │ │ │ - * Calculates the max extent which includes all of the features. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} or null if the layer has no features with │ │ │ │ │ - * geometries. │ │ │ │ │ - */ │ │ │ │ │ - getDataExtent: function() { │ │ │ │ │ - var maxExtent = null; │ │ │ │ │ - var features = this.features; │ │ │ │ │ - if (features && (features.length > 0)) { │ │ │ │ │ - var geometry = null; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - geometry = features[i].geometry; │ │ │ │ │ - if (geometry) { │ │ │ │ │ - if (maxExtent === null) { │ │ │ │ │ - maxExtent = new OpenLayers.Bounds(); │ │ │ │ │ + for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ + // recursive calls need to allow for sub-classing │ │ │ │ │ + json = OpenLayers.Format.JSON.prototype.write.apply(this, │ │ │ │ │ + [array[i], this.pretty]); │ │ │ │ │ + if (json != null) { │ │ │ │ │ + if (i > 0) { │ │ │ │ │ + pieces.push(','); │ │ │ │ │ } │ │ │ │ │ - maxExtent.extend(geometry.getBounds()); │ │ │ │ │ + pieces.push(this.writeNewline(), this.writeIndent(), json); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + this.level -= 1; │ │ │ │ │ + pieces.push(this.writeNewline(), this.writeIndent(), ']'); │ │ │ │ │ + return pieces.join(''); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: serialize.string │ │ │ │ │ + * Transform a string into a JSON string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * string - {String} The string to be serialized │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A JSON string representing the string. │ │ │ │ │ + */ │ │ │ │ │ + 'string': function(string) { │ │ │ │ │ + // If the string contains no control characters, no quote characters, and no │ │ │ │ │ + // backslash characters, then we can simply slap some quotes around it. │ │ │ │ │ + // Otherwise we must also replace the offending characters with safe │ │ │ │ │ + // sequences. │ │ │ │ │ + var m = { │ │ │ │ │ + '\b': '\\b', │ │ │ │ │ + '\t': '\\t', │ │ │ │ │ + '\n': '\\n', │ │ │ │ │ + '\f': '\\f', │ │ │ │ │ + '\r': '\\r', │ │ │ │ │ + '"': '\\"', │ │ │ │ │ + '\\': '\\\\' │ │ │ │ │ + }; │ │ │ │ │ + if (/["\\\x00-\x1f]/.test(string)) { │ │ │ │ │ + return '"' + string.replace(/([\x00-\x1f\\"])/g, function(a, b) { │ │ │ │ │ + var c = m[b]; │ │ │ │ │ + if (c) { │ │ │ │ │ + return c; │ │ │ │ │ + } │ │ │ │ │ + c = b.charCodeAt(); │ │ │ │ │ + return '\\u00' + │ │ │ │ │ + Math.floor(c / 16).toString(16) + │ │ │ │ │ + (c % 16).toString(16); │ │ │ │ │ + }) + '"'; │ │ │ │ │ + } │ │ │ │ │ + return '"' + string + '"'; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: serialize.number │ │ │ │ │ + * Transform a number into a JSON string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * number - {Number} The number to be serialized. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A JSON string representing the number. │ │ │ │ │ + */ │ │ │ │ │ + 'number': function(number) { │ │ │ │ │ + return isFinite(number) ? String(number) : "null"; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: serialize.boolean │ │ │ │ │ + * Transform a boolean into a JSON string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bool - {Boolean} The boolean to be serialized. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A JSON string representing the boolean. │ │ │ │ │ + */ │ │ │ │ │ + 'boolean': function(bool) { │ │ │ │ │ + return String(bool); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: serialize.object │ │ │ │ │ + * Transform a date into a JSON string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * date - {Date} The date to be serialized. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A JSON string representing the date. │ │ │ │ │ + */ │ │ │ │ │ + 'date': function(date) { │ │ │ │ │ + function format(number) { │ │ │ │ │ + // Format integers to have at least two digits. │ │ │ │ │ + return (number < 10) ? '0' + number : number; │ │ │ │ │ + } │ │ │ │ │ + return '"' + date.getFullYear() + '-' + │ │ │ │ │ + format(date.getMonth() + 1) + '-' + │ │ │ │ │ + format(date.getDate()) + 'T' + │ │ │ │ │ + format(date.getHours()) + ':' + │ │ │ │ │ + format(date.getMinutes()) + ':' + │ │ │ │ │ + format(date.getSeconds()) + '"'; │ │ │ │ │ } │ │ │ │ │ - return maxExtent; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Vector" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.JSON" │ │ │ │ │ + │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/WMS.js │ │ │ │ │ + OpenLayers/Geometry.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.WMS │ │ │ │ │ - * Instances of OpenLayers.Layer.WMS are used to display data from OGC Web │ │ │ │ │ - * Mapping Services. Create a new WMS layer with the <OpenLayers.Layer.WMS> │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ + * Class: OpenLayers.Geometry │ │ │ │ │ + * A Geometry is a description of a geographic object. Create an instance of │ │ │ │ │ + * this class with the <OpenLayers.Geometry> constructor. This is a base class, │ │ │ │ │ + * typical geometry types are described by subclasses of this class. │ │ │ │ │ + * │ │ │ │ │ + * Note that if you use the <OpenLayers.Geometry.fromWKT> method, you must │ │ │ │ │ + * explicitly include the OpenLayers.Format.WKT in your build. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.WMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ +OpenLayers.Geometry = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: DEFAULT_PARAMS │ │ │ │ │ - * {Object} Hashtable of default parameter key/value pairs │ │ │ │ │ + * Property: id │ │ │ │ │ + * {String} A unique identifier for this geometry. │ │ │ │ │ */ │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - service: "WMS", │ │ │ │ │ - version: "1.1.1", │ │ │ │ │ - request: "GetMap", │ │ │ │ │ - styles: "", │ │ │ │ │ - format: "image/jpeg" │ │ │ │ │ - }, │ │ │ │ │ + id: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} Default is true for WMS layer │ │ │ │ │ + * Property: parent │ │ │ │ │ + * {<OpenLayers.Geometry>}This is set when a Geometry is added as component │ │ │ │ │ + * of another geometry │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + parent: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: encodeBBOX │ │ │ │ │ - * {Boolean} Should the BBOX commas be encoded? The WMS spec says 'no', │ │ │ │ │ - * but some services want it that way. Default false. │ │ │ │ │ + * Property: bounds │ │ │ │ │ + * {<OpenLayers.Bounds>} The bounds of this geometry │ │ │ │ │ */ │ │ │ │ │ - encodeBBOX: false, │ │ │ │ │ + bounds: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: noMagic │ │ │ │ │ - * {Boolean} If true, the image format will not be automagicaly switched │ │ │ │ │ - * from image/jpeg to image/png or image/gif when using │ │ │ │ │ - * TRANSPARENT=TRUE. Also isBaseLayer will not changed by the │ │ │ │ │ - * constructor. Default false. │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Geometry │ │ │ │ │ + * Creates a geometry object. │ │ │ │ │ */ │ │ │ │ │ - noMagic: false, │ │ │ │ │ + initialize: function() { │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: yx │ │ │ │ │ - * {Object} Keys in this object are EPSG codes for which the axis order │ │ │ │ │ - * is to be reversed (yx instead of xy, LatLon instead of LonLat), with │ │ │ │ │ - * true as value. This is only relevant for WMS versions >= 1.3.0, and │ │ │ │ │ - * only if yx is not set in <OpenLayers.Projection.defaults> for the │ │ │ │ │ - * used projection. │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Destroy this geometry. │ │ │ │ │ */ │ │ │ │ │ - yx: {}, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.id = null; │ │ │ │ │ + this.bounds = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.WMS │ │ │ │ │ - * Create a new WMS layer object │ │ │ │ │ - * │ │ │ │ │ - * Examples: │ │ │ │ │ - * │ │ │ │ │ - * The code below creates a simple WMS layer using the image/jpeg format. │ │ │ │ │ - * (code) │ │ │ │ │ - * var wms = new OpenLayers.Layer.WMS("NASA Global Mosaic", │ │ │ │ │ - * "http://wms.jpl.nasa.gov/wms.cgi", │ │ │ │ │ - * {layers: "modis,global_mosaic"}); │ │ │ │ │ - * (end) │ │ │ │ │ - * Note the 3rd argument (params). Properties added to this object will be │ │ │ │ │ - * added to the WMS GetMap requests used for this layer's tiles. The only │ │ │ │ │ - * mandatory parameter is "layers". Other common WMS params include │ │ │ │ │ - * "transparent", "styles" and "format". Note that the "srs" param will │ │ │ │ │ - * always be ignored. Instead, it will be derived from the baseLayer's or │ │ │ │ │ - * map's projection. │ │ │ │ │ - * │ │ │ │ │ - * The code below creates a transparent WMS layer with additional options. │ │ │ │ │ - * (code) │ │ │ │ │ - * var wms = new OpenLayers.Layer.WMS("NASA Global Mosaic", │ │ │ │ │ - * "http://wms.jpl.nasa.gov/wms.cgi", │ │ │ │ │ - * { │ │ │ │ │ - * layers: "modis,global_mosaic", │ │ │ │ │ - * transparent: true │ │ │ │ │ - * }, { │ │ │ │ │ - * opacity: 0.5, │ │ │ │ │ - * singleTile: true │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * Note that by default, a WMS layer is configured as baseLayer. Setting │ │ │ │ │ - * the "transparent" param to true will apply some magic (see <noMagic>). │ │ │ │ │ - * The default image format changes from image/jpeg to image/png, and the │ │ │ │ │ - * layer is not configured as baseLayer. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Create a clone of this geometry. Does not set any non-standard │ │ │ │ │ + * properties of the cloned geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry>} An exact clone of this geometry. │ │ │ │ │ + */ │ │ │ │ │ + clone: function() { │ │ │ │ │ + return new OpenLayers.Geometry(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setBounds │ │ │ │ │ + * Set the bounds for this Geometry. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} A name for the layer │ │ │ │ │ - * url - {String} Base url for the WMS │ │ │ │ │ - * (e.g. http://wms.jpl.nasa.gov/wms.cgi) │ │ │ │ │ - * params - {Object} An object with key/value pairs representing the │ │ │ │ │ - * GetMap query string parameters and parameter values. │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer. │ │ │ │ │ - * These options include all properties listed above, plus the ones │ │ │ │ │ - * inherited from superclasses. │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - //uppercase params │ │ │ │ │ - params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ - if (parseFloat(params.VERSION) >= 1.3 && !params.EXCEPTIONS) { │ │ │ │ │ - params.EXCEPTIONS = "INIMAGE"; │ │ │ │ │ + setBounds: function(bounds) { │ │ │ │ │ + if (bounds) { │ │ │ │ │ + this.bounds = bounds.clone(); │ │ │ │ │ } │ │ │ │ │ - newArguments.push(name, url, params, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, │ │ │ │ │ - OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS) │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - //layer is transparent │ │ │ │ │ - if (!this.noMagic && this.params.TRANSPARENT && │ │ │ │ │ - this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ - │ │ │ │ │ - // unless explicitly set in options, make layer an overlay │ │ │ │ │ - if ((options == null) || (!options.isBaseLayer)) { │ │ │ │ │ - this.isBaseLayer = false; │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // jpegs can never be transparent, so intelligently switch the │ │ │ │ │ - // format, depending on the browser's capabilities │ │ │ │ │ - if (this.params.FORMAT == "image/jpeg") { │ │ │ │ │ - this.params.FORMAT = OpenLayers.Util.alphaHack() ? "image/gif" : │ │ │ │ │ - "image/png"; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: clearBounds │ │ │ │ │ + * Nullify this components bounds and that of its parent as well. │ │ │ │ │ + */ │ │ │ │ │ + clearBounds: function() { │ │ │ │ │ + this.bounds = null; │ │ │ │ │ + if (this.parent) { │ │ │ │ │ + this.parent.clearBounds(); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Method: extendBounds │ │ │ │ │ + * Extend the existing bounds to include the new bounds. │ │ │ │ │ + * If geometry's bounds is not yet set, then set a new Bounds. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newBounds - {<OpenLayers.Bounds>} │ │ │ │ │ + */ │ │ │ │ │ + extendBounds: function(newBounds) { │ │ │ │ │ + var bounds = this.getBounds(); │ │ │ │ │ + if (!bounds) { │ │ │ │ │ + this.setBounds(newBounds); │ │ │ │ │ + } else { │ │ │ │ │ + this.bounds.extend(newBounds); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getBounds │ │ │ │ │ + * Get the bounds for this Geometry. If bounds is not set, it │ │ │ │ │ + * is calculated again, this makes queries faster. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.WMS>} An exact clone of this layer │ │ │ │ │ + * {<OpenLayers.Bounds>} │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.WMS(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.params, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ + getBounds: function() { │ │ │ │ │ + if (this.bounds == null) { │ │ │ │ │ + this.calculateBounds(); │ │ │ │ │ } │ │ │ │ │ + return this.bounds; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: calculateBounds │ │ │ │ │ + * Recalculate the bounds for the geometry. │ │ │ │ │ + */ │ │ │ │ │ + calculateBounds: function() { │ │ │ │ │ + // │ │ │ │ │ + // This should be overridden by subclasses. │ │ │ │ │ + // │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: reverseAxisOrder │ │ │ │ │ - * Returns true if the axis order is reversed for the WMS version and │ │ │ │ │ - * projection of the layer. │ │ │ │ │ + * APIMethod: distanceTo │ │ │ │ │ + * Calculate the closest distance between two geometries (on the x-y plane). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ + * options - {Object} Optional properties for configuring the distance │ │ │ │ │ + * calculation. │ │ │ │ │ + * │ │ │ │ │ + * Valid options depend on the specific geometry type. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} true if the axis order is reversed, false otherwise. │ │ │ │ │ + * {Number | Object} The distance between this geometry and the target. │ │ │ │ │ + * If details is true, the return will be an object with distance, │ │ │ │ │ + * x0, y0, x1, and x2 properties. The x0 and y0 properties represent │ │ │ │ │ + * the coordinates of the closest point on this geometry. The x1 and y1 │ │ │ │ │ + * properties represent the coordinates of the closest point on the │ │ │ │ │ + * target geometry. │ │ │ │ │ */ │ │ │ │ │ - reverseAxisOrder: function() { │ │ │ │ │ - var projCode = this.projection.getCode(); │ │ │ │ │ - return parseFloat(this.params.VERSION) >= 1.3 && │ │ │ │ │ - !!(this.yx[projCode] || (OpenLayers.Projection.defaults[projCode] && │ │ │ │ │ - OpenLayers.Projection.defaults[projCode].yx)); │ │ │ │ │ - }, │ │ │ │ │ + distanceTo: function(geometry, options) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * Return a GetMap query string for this layer │ │ │ │ │ + * APIMethod: getVertices │ │ │ │ │ + * Return a list of all points in this geometry. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox for the │ │ │ │ │ - * request. │ │ │ │ │ + * nodes - {Boolean} For lines, only return vertices that are │ │ │ │ │ + * endpoints. If false, for lines, only vertices that are not │ │ │ │ │ + * endpoints will be returned. If not provided, all vertices will │ │ │ │ │ + * be returned. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as │ │ │ │ │ - * parameters. │ │ │ │ │ + * {Array} A list of all vertices in the geometry. │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ + getVertices: function(nodes) {}, │ │ │ │ │ │ │ │ │ │ - var imageSize = this.getImageSize(); │ │ │ │ │ - var newParams = {}; │ │ │ │ │ - // WMS 1.3 introduced axis order │ │ │ │ │ - var reverseAxisOrder = this.reverseAxisOrder(); │ │ │ │ │ - newParams.BBOX = this.encodeBBOX ? │ │ │ │ │ - bounds.toBBOX(null, reverseAxisOrder) : │ │ │ │ │ - bounds.toArray(reverseAxisOrder); │ │ │ │ │ - newParams.WIDTH = imageSize.w; │ │ │ │ │ - newParams.HEIGHT = imageSize.h; │ │ │ │ │ - var requestString = this.getFullRequestString(newParams); │ │ │ │ │ - return requestString; │ │ │ │ │ + /** │ │ │ │ │ + * Method: atPoint │ │ │ │ │ + * Note - This is only an approximation based on the bounds of the │ │ │ │ │ + * geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>|Object} OpenLayers.LonLat or an │ │ │ │ │ + * object with a 'lon' and 'lat' properties. │ │ │ │ │ + * toleranceLon - {float} Optional tolerance in Geometric Coords │ │ │ │ │ + * toleranceLat - {float} Optional tolerance in Geographic Coords │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the geometry is at the specified location │ │ │ │ │ + */ │ │ │ │ │ + atPoint: function(lonlat, toleranceLon, toleranceLat) { │ │ │ │ │ + var atPoint = false; │ │ │ │ │ + var bounds = this.getBounds(); │ │ │ │ │ + if ((bounds != null) && (lonlat != null)) { │ │ │ │ │ + │ │ │ │ │ + var dX = (toleranceLon != null) ? toleranceLon : 0; │ │ │ │ │ + var dY = (toleranceLat != null) ? toleranceLat : 0; │ │ │ │ │ + │ │ │ │ │ + var toleranceBounds = │ │ │ │ │ + new OpenLayers.Bounds(this.bounds.left - dX, │ │ │ │ │ + this.bounds.bottom - dY, │ │ │ │ │ + this.bounds.right + dX, │ │ │ │ │ + this.bounds.top + dY); │ │ │ │ │ + │ │ │ │ │ + atPoint = toleranceBounds.containsLonLat(lonlat); │ │ │ │ │ + } │ │ │ │ │ + return atPoint; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: mergeNewParams │ │ │ │ │ - * Catch changeParams and uppercase the new params to be merged in │ │ │ │ │ - * before calling changeParams on the super class. │ │ │ │ │ + * Method: getLength │ │ │ │ │ + * Calculate the length of this geometry. This method is defined in │ │ │ │ │ + * subclasses. │ │ │ │ │ * │ │ │ │ │ - * Once params have been changed, the tiles will be reloaded with │ │ │ │ │ - * the new parameters. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The length of the collection by summing its parts │ │ │ │ │ + */ │ │ │ │ │ + getLength: function() { │ │ │ │ │ + //to be overridden by geometries that actually have a length │ │ │ │ │ + // │ │ │ │ │ + return 0.0; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getArea │ │ │ │ │ + * Calculate the area of this geometry. This method is defined in subclasses. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newParams - {Object} Hashtable of new params to use │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The area of the collection by summing its parts │ │ │ │ │ */ │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ - var newArguments = [upperParams]; │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, │ │ │ │ │ - newArguments); │ │ │ │ │ + getArea: function() { │ │ │ │ │ + //to be overridden by geometries that actually have an area │ │ │ │ │ + // │ │ │ │ │ + return 0.0; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getFullRequestString │ │ │ │ │ - * Combine the layer's url with its params and these newParams. │ │ │ │ │ - * │ │ │ │ │ - * Add the SRS parameter from projection -- this is probably │ │ │ │ │ - * more eloquently done via a setProjection() method, but this │ │ │ │ │ - * works for now and always. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getCentroid │ │ │ │ │ + * Calculate the centroid of this geometry. This method is defined in subclasses. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newParams - {Object} │ │ │ │ │ - * altUrl - {String} Use this as the url instead of the layer's url │ │ │ │ │ - * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} │ │ │ │ │ + * {<OpenLayers.Geometry.Point>} The centroid of the collection │ │ │ │ │ */ │ │ │ │ │ - getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ - var mapProjection = this.map.getProjectionObject(); │ │ │ │ │ - var projectionCode = this.projection && this.projection.equals(mapProjection) ? │ │ │ │ │ - this.projection.getCode() : │ │ │ │ │ - mapProjection.getCode(); │ │ │ │ │ - var value = (projectionCode == "none") ? null : projectionCode; │ │ │ │ │ - if (parseFloat(this.params.VERSION) >= 1.3) { │ │ │ │ │ - this.params.CRS = value; │ │ │ │ │ + getCentroid: function() { │ │ │ │ │ + return null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: toString │ │ │ │ │ + * Returns a text representation of the geometry. If the WKT format is │ │ │ │ │ + * included in a build, this will be the Well-Known Text │ │ │ │ │ + * representation. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} String representation of this geometry. │ │ │ │ │ + */ │ │ │ │ │ + toString: function() { │ │ │ │ │ + var string; │ │ │ │ │ + if (OpenLayers.Format && OpenLayers.Format.WKT) { │ │ │ │ │ + string = OpenLayers.Format.WKT.prototype.write( │ │ │ │ │ + new OpenLayers.Feature.Vector(this) │ │ │ │ │ + ); │ │ │ │ │ } else { │ │ │ │ │ - this.params.SRS = value; │ │ │ │ │ + string = Object.prototype.toString.call(this); │ │ │ │ │ } │ │ │ │ │ + return string; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (typeof this.params.TRANSPARENT == "boolean") { │ │ │ │ │ - newParams.TRANSPARENT = this.params.TRANSPARENT ? "TRUE" : "FALSE"; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Geometry" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Function: OpenLayers.Geometry.fromWKT │ │ │ │ │ + * Generate a geometry given a Well-Known Text string. For this method to │ │ │ │ │ + * work, you must include the OpenLayers.Format.WKT in your build │ │ │ │ │ + * explicitly. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * wkt - {String} A string representing the geometry in Well-Known Text. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry>} A geometry of the appropriate class. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Geometry.fromWKT = function(wkt) { │ │ │ │ │ + var geom; │ │ │ │ │ + if (OpenLayers.Format && OpenLayers.Format.WKT) { │ │ │ │ │ + var format = OpenLayers.Geometry.fromWKT.format; │ │ │ │ │ + if (!format) { │ │ │ │ │ + format = new OpenLayers.Format.WKT(); │ │ │ │ │ + OpenLayers.Geometry.fromWKT.format = format; │ │ │ │ │ + } │ │ │ │ │ + var result = format.read(wkt); │ │ │ │ │ + if (result instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ + geom = result.geometry; │ │ │ │ │ + } else if (OpenLayers.Util.isArray(result)) { │ │ │ │ │ + var len = result.length; │ │ │ │ │ + var components = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + components[i] = result[i].geometry; │ │ │ │ │ + } │ │ │ │ │ + geom = new OpenLayers.Geometry.Collection(components); │ │ │ │ │ } │ │ │ │ │ + } │ │ │ │ │ + return geom; │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.getFullRequestString.apply( │ │ │ │ │ - this, arguments); │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Method: OpenLayers.Geometry.segmentsIntersect │ │ │ │ │ + * Determine whether two line segments intersect. Optionally calculates │ │ │ │ │ + * and returns the intersection point. This function is optimized for │ │ │ │ │ + * cases where seg1.x2 >= seg2.x1 || seg2.x2 >= seg1.x1. In those │ │ │ │ │ + * obvious cases where there is no intersection, the function should │ │ │ │ │ + * not be called. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * seg1 - {Object} Object representing a segment with properties x1, y1, x2, │ │ │ │ │ + * and y2. The start point is represented by x1 and y1. The end point │ │ │ │ │ + * is represented by x2 and y2. Start and end are ordered so that x1 < x2. │ │ │ │ │ + * seg2 - {Object} Object representing a segment with properties x1, y1, x2, │ │ │ │ │ + * and y2. The start point is represented by x1 and y1. The end point │ │ │ │ │ + * is represented by x2 and y2. Start and end are ordered so that x1 < x2. │ │ │ │ │ + * options - {Object} Optional properties for calculating the intersection. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * point - {Boolean} Return the intersection point. If false, the actual │ │ │ │ │ + * intersection point will not be calculated. If true and the segments │ │ │ │ │ + * intersect, the intersection point will be returned. If true and │ │ │ │ │ + * the segments do not intersect, false will be returned. If true and │ │ │ │ │ + * the segments are coincident, true will be returned. │ │ │ │ │ + * tolerance - {Number} If a non-null value is provided, if the segments are │ │ │ │ │ + * within the tolerance distance, this will be considered an intersection. │ │ │ │ │ + * In addition, if the point option is true and the calculated intersection │ │ │ │ │ + * is within the tolerance distance of an end point, the endpoint will be │ │ │ │ │ + * returned instead of the calculated intersection. Further, if the │ │ │ │ │ + * intersection is within the tolerance of endpoints on both segments, or │ │ │ │ │ + * if two segment endpoints are within the tolerance distance of eachother │ │ │ │ │ + * (but no intersection is otherwise calculated), an endpoint on the │ │ │ │ │ + * first segment provided will be returned. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean | <OpenLayers.Geometry.Point>} The two segments intersect. │ │ │ │ │ + * If the point argument is true, the return will be the intersection │ │ │ │ │ + * point or false if none exists. If point is true and the segments │ │ │ │ │ + * are coincident, return will be true (and the instersection is equal │ │ │ │ │ + * to the shorter segment). │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Geometry.segmentsIntersect = function(seg1, seg2, options) { │ │ │ │ │ + var point = options && options.point; │ │ │ │ │ + var tolerance = options && options.tolerance; │ │ │ │ │ + var intersection = false; │ │ │ │ │ + var x11_21 = seg1.x1 - seg2.x1; │ │ │ │ │ + var y11_21 = seg1.y1 - seg2.y1; │ │ │ │ │ + var x12_11 = seg1.x2 - seg1.x1; │ │ │ │ │ + var y12_11 = seg1.y2 - seg1.y1; │ │ │ │ │ + var y22_21 = seg2.y2 - seg2.y1; │ │ │ │ │ + var x22_21 = seg2.x2 - seg2.x1; │ │ │ │ │ + var d = (y22_21 * x12_11) - (x22_21 * y12_11); │ │ │ │ │ + var n1 = (x22_21 * y11_21) - (y22_21 * x11_21); │ │ │ │ │ + var n2 = (x12_11 * y11_21) - (y12_11 * x11_21); │ │ │ │ │ + if (d == 0) { │ │ │ │ │ + // parallel │ │ │ │ │ + if (n1 == 0 && n2 == 0) { │ │ │ │ │ + // coincident │ │ │ │ │ + intersection = true; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + var along1 = n1 / d; │ │ │ │ │ + var along2 = n2 / d; │ │ │ │ │ + if (along1 >= 0 && along1 <= 1 && along2 >= 0 && along2 <= 1) { │ │ │ │ │ + // intersect │ │ │ │ │ + if (!point) { │ │ │ │ │ + intersection = true; │ │ │ │ │ + } else { │ │ │ │ │ + // calculate the intersection point │ │ │ │ │ + var x = seg1.x1 + (along1 * x12_11); │ │ │ │ │ + var y = seg1.y1 + (along1 * y12_11); │ │ │ │ │ + intersection = new OpenLayers.Geometry.Point(x, y); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (tolerance) { │ │ │ │ │ + var dist; │ │ │ │ │ + if (intersection) { │ │ │ │ │ + if (point) { │ │ │ │ │ + var segs = [seg1, seg2]; │ │ │ │ │ + var seg, x, y; │ │ │ │ │ + // check segment endpoints for proximity to intersection │ │ │ │ │ + // set intersection to first endpoint within the tolerance │ │ │ │ │ + outer: for (var i = 0; i < 2; ++i) { │ │ │ │ │ + seg = segs[i]; │ │ │ │ │ + for (var j = 1; j < 3; ++j) { │ │ │ │ │ + x = seg["x" + j]; │ │ │ │ │ + y = seg["y" + j]; │ │ │ │ │ + dist = Math.sqrt( │ │ │ │ │ + Math.pow(x - intersection.x, 2) + │ │ │ │ │ + Math.pow(y - intersection.y, 2) │ │ │ │ │ + ); │ │ │ │ │ + if (dist < tolerance) { │ │ │ │ │ + intersection.x = x; │ │ │ │ │ + intersection.y = y; │ │ │ │ │ + break outer; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.WMS" │ │ │ │ │ -}); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // no calculated intersection, but segments could be within │ │ │ │ │ + // the tolerance of one another │ │ │ │ │ + var segs = [seg1, seg2]; │ │ │ │ │ + var source, target, x, y, p, result; │ │ │ │ │ + // check segment endpoints for proximity to intersection │ │ │ │ │ + // set intersection to first endpoint within the tolerance │ │ │ │ │ + outer: for (var i = 0; i < 2; ++i) { │ │ │ │ │ + source = segs[i]; │ │ │ │ │ + target = segs[(i + 1) % 2]; │ │ │ │ │ + for (var j = 1; j < 3; ++j) { │ │ │ │ │ + p = { │ │ │ │ │ + x: source["x" + j], │ │ │ │ │ + y: source["y" + j] │ │ │ │ │ + }; │ │ │ │ │ + result = OpenLayers.Geometry.distanceToSegment(p, target); │ │ │ │ │ + if (result.distance < tolerance) { │ │ │ │ │ + if (point) { │ │ │ │ │ + intersection = new OpenLayers.Geometry.Point(p.x, p.y); │ │ │ │ │ + } else { │ │ │ │ │ + intersection = true; │ │ │ │ │ + } │ │ │ │ │ + break outer; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return intersection; │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Function: OpenLayers.Geometry.distanceToSegment │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {Object} An object with x and y properties representing the │ │ │ │ │ + * point coordinates. │ │ │ │ │ + * segment - {Object} An object with x1, y1, x2, and y2 properties │ │ │ │ │ + * representing endpoint coordinates. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object with distance, along, x, and y properties. The distance │ │ │ │ │ + * will be the shortest distance between the input point and segment. │ │ │ │ │ + * The x and y properties represent the coordinates along the segment │ │ │ │ │ + * where the shortest distance meets the segment. The along attribute │ │ │ │ │ + * describes how far between the two segment points the given point is. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Geometry.distanceToSegment = function(point, segment) { │ │ │ │ │ + var result = OpenLayers.Geometry.distanceSquaredToSegment(point, segment); │ │ │ │ │ + result.distance = Math.sqrt(result.distance); │ │ │ │ │ + return result; │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Function: OpenLayers.Geometry.distanceSquaredToSegment │ │ │ │ │ + * │ │ │ │ │ + * Usually the distanceToSegment function should be used. This variant however │ │ │ │ │ + * can be used for comparisons where the exact distance is not important. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {Object} An object with x and y properties representing the │ │ │ │ │ + * point coordinates. │ │ │ │ │ + * segment - {Object} An object with x1, y1, x2, and y2 properties │ │ │ │ │ + * representing endpoint coordinates. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object with squared distance, along, x, and y properties. │ │ │ │ │ + * The distance will be the shortest distance between the input point and │ │ │ │ │ + * segment. The x and y properties represent the coordinates along the │ │ │ │ │ + * segment where the shortest distance meets the segment. The along │ │ │ │ │ + * attribute describes how far between the two segment points the given │ │ │ │ │ + * point is. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Geometry.distanceSquaredToSegment = function(point, segment) { │ │ │ │ │ + var x0 = point.x; │ │ │ │ │ + var y0 = point.y; │ │ │ │ │ + var x1 = segment.x1; │ │ │ │ │ + var y1 = segment.y1; │ │ │ │ │ + var x2 = segment.x2; │ │ │ │ │ + var y2 = segment.y2; │ │ │ │ │ + var dx = x2 - x1; │ │ │ │ │ + var dy = y2 - y1; │ │ │ │ │ + var along = ((dx * (x0 - x1)) + (dy * (y0 - y1))) / │ │ │ │ │ + (Math.pow(dx, 2) + Math.pow(dy, 2)); │ │ │ │ │ + var x, y; │ │ │ │ │ + if (along <= 0.0) { │ │ │ │ │ + x = x1; │ │ │ │ │ + y = y1; │ │ │ │ │ + } else if (along >= 1.0) { │ │ │ │ │ + x = x2; │ │ │ │ │ + y = y2; │ │ │ │ │ + } else { │ │ │ │ │ + x = x1 + along * dx; │ │ │ │ │ + y = y1 + along * dy; │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + distance: Math.pow(x - x0, 2) + Math.pow(y - y0, 2), │ │ │ │ │ + x: x, │ │ │ │ │ + y: y, │ │ │ │ │ + along: along │ │ │ │ │ + }; │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/SphericalMercator.js │ │ │ │ │ + OpenLayers/Geometry/Point.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer.js │ │ │ │ │ - * @requires OpenLayers/Projection.js │ │ │ │ │ + * @requires OpenLayers/Geometry.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.SphericalMercator │ │ │ │ │ - * A mixin for layers that wraps up the pieces neccesary to have a coordinate │ │ │ │ │ - * conversion for working with commercial APIs which use a spherical │ │ │ │ │ - * mercator projection. Using this layer as a base layer, additional │ │ │ │ │ - * layers can be used as overlays if they are in the same projection. │ │ │ │ │ - * │ │ │ │ │ - * A layer is given properties of this object by setting the sphericalMercator │ │ │ │ │ - * property to true. │ │ │ │ │ - * │ │ │ │ │ - * More projection information: │ │ │ │ │ - * - http://spatialreference.org/ref/user/google-projection/ │ │ │ │ │ - * │ │ │ │ │ - * Proj4 Text: │ │ │ │ │ - * +proj=merc +a=6378137 +b=6378137 +lat_ts=0.0 +lon_0=0.0 +x_0=0.0 +y_0=0 │ │ │ │ │ - * +k=1.0 +units=m +nadgrids=@null +no_defs │ │ │ │ │ - * │ │ │ │ │ - * WKT: │ │ │ │ │ - * 900913=PROJCS["WGS84 / Simple Mercator", GEOGCS["WGS 84", │ │ │ │ │ - * DATUM["WGS_1984", SPHEROID["WGS_1984", 6378137.0, 298.257223563]], │ │ │ │ │ - * PRIMEM["Greenwich", 0.0], UNIT["degree", 0.017453292519943295], │ │ │ │ │ - * AXIS["Longitude", EAST], AXIS["Latitude", NORTH]], │ │ │ │ │ - * PROJECTION["Mercator_1SP_Google"], │ │ │ │ │ - * PARAMETER["latitude_of_origin", 0.0], PARAMETER["central_meridian", 0.0], │ │ │ │ │ - * PARAMETER["scale_factor", 1.0], PARAMETER["false_easting", 0.0], │ │ │ │ │ - * PARAMETER["false_northing", 0.0], UNIT["m", 1.0], AXIS["x", EAST], │ │ │ │ │ - * AXIS["y", NORTH], AUTHORITY["EPSG","900913"]] │ │ │ │ │ + * Class: OpenLayers.Geometry.Point │ │ │ │ │ + * Point geometry class. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Geometry> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.SphericalMercator = { │ │ │ │ │ +OpenLayers.Geometry.Point = OpenLayers.Class(OpenLayers.Geometry, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: x │ │ │ │ │ + * {float} │ │ │ │ │ + */ │ │ │ │ │ + x: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: y │ │ │ │ │ + * {float} │ │ │ │ │ + */ │ │ │ │ │ + y: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getExtent │ │ │ │ │ - * Get the map's extent. │ │ │ │ │ + * Constructor: OpenLayers.Geometry.Point │ │ │ │ │ + * Construct a point geometry. │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * x - {float} │ │ │ │ │ + * y - {float} │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(x, y) { │ │ │ │ │ + OpenLayers.Geometry.prototype.initialize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + this.x = parseFloat(x); │ │ │ │ │ + this.y = parseFloat(y); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} The map extent. │ │ │ │ │ + * {<OpenLayers.Geometry.Point>} An exact clone of this OpenLayers.Geometry.Point │ │ │ │ │ */ │ │ │ │ │ - getExtent: function() { │ │ │ │ │ - var extent = null; │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - extent = this.map.calculateBounds(); │ │ │ │ │ - } else { │ │ │ │ │ - extent = OpenLayers.Layer.FixedZoomLevels.prototype.getExtent.apply(this); │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Geometry.Point(this.x, this.y); │ │ │ │ │ } │ │ │ │ │ - return extent; │ │ │ │ │ + │ │ │ │ │ + // catch any randomly tagged-on properties │ │ │ │ │ + OpenLayers.Util.applyDefaults(obj, this); │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateBounds │ │ │ │ │ + * Create a new Bounds based on the lon/lat │ │ │ │ │ + */ │ │ │ │ │ + calculateBounds: function() { │ │ │ │ │ + this.bounds = new OpenLayers.Bounds(this.x, this.y, │ │ │ │ │ + this.x, this.y); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getLonLatFromViewPortPx │ │ │ │ │ - * Get a map location from a pixel location │ │ │ │ │ + * APIMethod: distanceTo │ │ │ │ │ + * Calculate the closest distance between two geometries (on the x-y plane). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ + * options - {Object} Optional properties for configuring the distance │ │ │ │ │ + * calculation. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * details - {Boolean} Return details from the distance calculation. │ │ │ │ │ + * Default is false. │ │ │ │ │ + * edge - {Boolean} Calculate the distance from this geometry to the │ │ │ │ │ + * nearest edge of the target geometry. Default is true. If true, │ │ │ │ │ + * calling distanceTo from a geometry that is wholly contained within │ │ │ │ │ + * the target will result in a non-zero distance. If false, whenever │ │ │ │ │ + * geometries intersect, calling distanceTo will return 0. If false, │ │ │ │ │ + * details cannot be returned. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number | Object} The distance between this geometry and the target. │ │ │ │ │ + * If details is true, the return will be an object with distance, │ │ │ │ │ + * x0, y0, x1, and x2 properties. The x0 and y0 properties represent │ │ │ │ │ + * the coordinates of the closest point on this geometry. The x1 and y1 │ │ │ │ │ + * properties represent the coordinates of the closest point on the │ │ │ │ │ + * target geometry. │ │ │ │ │ + */ │ │ │ │ │ + distanceTo: function(geometry, options) { │ │ │ │ │ + var edge = !(options && options.edge === false); │ │ │ │ │ + var details = edge && options && options.details; │ │ │ │ │ + var distance, x0, y0, x1, y1, result; │ │ │ │ │ + if (geometry instanceof OpenLayers.Geometry.Point) { │ │ │ │ │ + x0 = this.x; │ │ │ │ │ + y0 = this.y; │ │ │ │ │ + x1 = geometry.x; │ │ │ │ │ + y1 = geometry.y; │ │ │ │ │ + distance = Math.sqrt(Math.pow(x0 - x1, 2) + Math.pow(y0 - y1, 2)); │ │ │ │ │ + result = !details ? │ │ │ │ │ + distance : { │ │ │ │ │ + x0: x0, │ │ │ │ │ + y0: y0, │ │ │ │ │ + x1: x1, │ │ │ │ │ + y1: y1, │ │ │ │ │ + distance: distance │ │ │ │ │ + }; │ │ │ │ │ + } else { │ │ │ │ │ + result = geometry.distanceTo(this, options); │ │ │ │ │ + if (details) { │ │ │ │ │ + // switch coord order since this geom is target │ │ │ │ │ + result = { │ │ │ │ │ + x0: result.x1, │ │ │ │ │ + y0: result.y1, │ │ │ │ │ + x1: result.x0, │ │ │ │ │ + y1: result.y0, │ │ │ │ │ + distance: result.distance │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return result; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: equals │ │ │ │ │ + * Determine whether another geometry is equivalent to this one. Geometries │ │ │ │ │ + * are considered equivalent if all components have the same coordinates. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * viewPortPx - {<OpenLayers.Pixel>} │ │ │ │ │ + * geom - {<OpenLayers.Geometry.Point>} The geometry to test. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.LonLat>} An OpenLayers.LonLat which is the passed-in view │ │ │ │ │ - * port OpenLayers.Pixel, translated into lon/lat by map lib │ │ │ │ │ - * If the map lib is not loaded or not centered, returns null │ │ │ │ │ + * {Boolean} The supplied geometry is equivalent to this geometry. │ │ │ │ │ */ │ │ │ │ │ - getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ - return OpenLayers.Layer.prototype.getLonLatFromViewPortPx.apply(this, arguments); │ │ │ │ │ + equals: function(geom) { │ │ │ │ │ + var equals = false; │ │ │ │ │ + if (geom != null) { │ │ │ │ │ + equals = ((this.x == geom.x && this.y == geom.y) || │ │ │ │ │ + (isNaN(this.x) && isNaN(this.y) && isNaN(geom.x) && isNaN(geom.y))); │ │ │ │ │ + } │ │ │ │ │ + return equals; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getViewPortPxFromLonLat │ │ │ │ │ - * Get a pixel location from a map location │ │ │ │ │ + * Method: toShortString │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} Shortened String representation of Point object. │ │ │ │ │ + * (ex. <i>"5, 42"</i>) │ │ │ │ │ + */ │ │ │ │ │ + toShortString: function() { │ │ │ │ │ + return (this.x + ", " + this.y); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: move │ │ │ │ │ + * Moves a geometry by the given displacement along positive x and y axes. │ │ │ │ │ + * This modifies the position of the geometry and clears the cached │ │ │ │ │ + * bounds. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ + * x - {Float} Distance to move geometry in positive x direction. │ │ │ │ │ + * y - {Float} Distance to move geometry in positive y direction. │ │ │ │ │ + */ │ │ │ │ │ + move: function(x, y) { │ │ │ │ │ + this.x = this.x + x; │ │ │ │ │ + this.y = this.y + y; │ │ │ │ │ + this.clearBounds(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: rotate │ │ │ │ │ + * Rotate a point around another. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Pixel>} An OpenLayers.Pixel which is the passed-in │ │ │ │ │ - * OpenLayers.LonLat, translated into view port pixels by map lib │ │ │ │ │ - * If map lib is not loaded or not centered, returns null │ │ │ │ │ + * Parameters: │ │ │ │ │ + * angle - {Float} Rotation angle in degrees (measured counterclockwise │ │ │ │ │ + * from the positive x-axis) │ │ │ │ │ + * origin - {<OpenLayers.Geometry.Point>} Center point for the rotation │ │ │ │ │ */ │ │ │ │ │ - getViewPortPxFromLonLat: function(lonlat) { │ │ │ │ │ - return OpenLayers.Layer.prototype.getViewPortPxFromLonLat.apply(this, arguments); │ │ │ │ │ + rotate: function(angle, origin) { │ │ │ │ │ + angle *= Math.PI / 180; │ │ │ │ │ + var radius = this.distanceTo(origin); │ │ │ │ │ + var theta = angle + Math.atan2(this.y - origin.y, this.x - origin.x); │ │ │ │ │ + this.x = origin.x + (radius * Math.cos(theta)); │ │ │ │ │ + this.y = origin.y + (radius * Math.sin(theta)); │ │ │ │ │ + this.clearBounds(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: initMercatorParameters │ │ │ │ │ - * Set up the mercator parameters on the layer: resolutions, │ │ │ │ │ - * projection, units. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getCentroid │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Point>} The centroid of the collection │ │ │ │ │ */ │ │ │ │ │ - initMercatorParameters: function() { │ │ │ │ │ - // set up properties for Mercator - assume EPSG:900913 │ │ │ │ │ - this.RESOLUTIONS = []; │ │ │ │ │ - var maxResolution = 156543.03390625; │ │ │ │ │ - for (var zoom = 0; zoom <= this.MAX_ZOOM_LEVEL; ++zoom) { │ │ │ │ │ - this.RESOLUTIONS[zoom] = maxResolution / Math.pow(2, zoom); │ │ │ │ │ - } │ │ │ │ │ - this.units = "m"; │ │ │ │ │ - this.projection = this.projection || "EPSG:900913"; │ │ │ │ │ + getCentroid: function() { │ │ │ │ │ + return new OpenLayers.Geometry.Point(this.x, this.y); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: forwardMercator │ │ │ │ │ - * Given a lon,lat in EPSG:4326, return a point in Spherical Mercator. │ │ │ │ │ + * APIMethod: resize │ │ │ │ │ + * Resize a point relative to some origin. For points, this has the effect │ │ │ │ │ + * of scaling a vector (from the origin to the point). This method is │ │ │ │ │ + * more useful on geometry collection subclasses. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * lon - {float} │ │ │ │ │ - * lat - {float} │ │ │ │ │ + * scale - {Float} Ratio of the new distance from the origin to the old │ │ │ │ │ + * distance from the origin. A scale of 2 doubles the │ │ │ │ │ + * distance between the point and origin. │ │ │ │ │ + * origin - {<OpenLayers.Geometry.Point>} Point of origin for resizing │ │ │ │ │ + * ratio - {Float} Optional x:y ratio for resizing. Default ratio is 1. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.LonLat>} The coordinates transformed to Mercator. │ │ │ │ │ + * {<OpenLayers.Geometry>} - The current geometry. │ │ │ │ │ */ │ │ │ │ │ - forwardMercator: (function() { │ │ │ │ │ - var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - var sm = new OpenLayers.Projection("EPSG:900913"); │ │ │ │ │ - return function(lon, lat) { │ │ │ │ │ - var point = OpenLayers.Projection.transform({ │ │ │ │ │ - x: lon, │ │ │ │ │ - y: lat │ │ │ │ │ - }, gg, sm); │ │ │ │ │ - return new OpenLayers.LonLat(point.x, point.y); │ │ │ │ │ - }; │ │ │ │ │ - })(), │ │ │ │ │ + resize: function(scale, origin, ratio) { │ │ │ │ │ + ratio = (ratio == undefined) ? 1 : ratio; │ │ │ │ │ + this.x = origin.x + (scale * ratio * (this.x - origin.x)); │ │ │ │ │ + this.y = origin.y + (scale * (this.y - origin.y)); │ │ │ │ │ + this.clearBounds(); │ │ │ │ │ + return this; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: inverseMercator │ │ │ │ │ - * Given a x,y in Spherical Mercator, return a point in EPSG:4326. │ │ │ │ │ + * APIMethod: intersects │ │ │ │ │ + * Determine if the input geometry intersects this one. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * x - {float} A map x in Spherical Mercator. │ │ │ │ │ - * y - {float} A map y in Spherical Mercator. │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} Any type of geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The input geometry intersects this one. │ │ │ │ │ + */ │ │ │ │ │ + intersects: function(geometry) { │ │ │ │ │ + var intersect = false; │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + intersect = this.equals(geometry); │ │ │ │ │ + } else { │ │ │ │ │ + intersect = geometry.intersects(this); │ │ │ │ │ + } │ │ │ │ │ + return intersect; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: transform │ │ │ │ │ + * Translate the x,y properties of the point from source to dest. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * source - {<OpenLayers.Projection>} │ │ │ │ │ + * dest - {<OpenLayers.Projection>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.LonLat>} The coordinates transformed to EPSG:4326. │ │ │ │ │ + * {<OpenLayers.Geometry>} │ │ │ │ │ */ │ │ │ │ │ - inverseMercator: (function() { │ │ │ │ │ - var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - var sm = new OpenLayers.Projection("EPSG:900913"); │ │ │ │ │ - return function(x, y) { │ │ │ │ │ - var point = OpenLayers.Projection.transform({ │ │ │ │ │ - x: x, │ │ │ │ │ - y: y │ │ │ │ │ - }, sm, gg); │ │ │ │ │ - return new OpenLayers.LonLat(point.x, point.y); │ │ │ │ │ - }; │ │ │ │ │ - })() │ │ │ │ │ + transform: function(source, dest) { │ │ │ │ │ + if ((source && dest)) { │ │ │ │ │ + OpenLayers.Projection.transform( │ │ │ │ │ + this, source, dest); │ │ │ │ │ + this.bounds = null; │ │ │ │ │ + } │ │ │ │ │ + return this; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -}; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getVertices │ │ │ │ │ + * Return a list of all points in this geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * nodes - {Boolean} For lines, only return vertices that are │ │ │ │ │ + * endpoints. If false, for lines, only vertices that are not │ │ │ │ │ + * endpoints will be returned. If not provided, all vertices will │ │ │ │ │ + * be returned. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} A list of all vertices in the geometry. │ │ │ │ │ + */ │ │ │ │ │ + getVertices: function(nodes) { │ │ │ │ │ + return [this]; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Geometry.Point" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/EventPane.js │ │ │ │ │ + OpenLayers/Geometry/Collection.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ + * @requires OpenLayers/Geometry.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.EventPane │ │ │ │ │ - * Base class for 3rd party layers, providing a DOM element which isolates │ │ │ │ │ - * the 3rd-party layer from mouse events. │ │ │ │ │ - * Only used by Google layers. │ │ │ │ │ + * Class: OpenLayers.Geometry.Collection │ │ │ │ │ + * A Collection is exactly what it sounds like: A collection of different │ │ │ │ │ + * Geometries. These are stored in the local parameter <components> (which │ │ │ │ │ + * can be passed as a parameter to the constructor). │ │ │ │ │ + * │ │ │ │ │ + * As new geometries are added to the collection, they are NOT cloned. │ │ │ │ │ + * When removing geometries, they need to be specified by reference (ie you │ │ │ │ │ + * have to pass in the *exact* geometry to be removed). │ │ │ │ │ + * │ │ │ │ │ + * The <getArea> and <getLength> functions here merely iterate through │ │ │ │ │ + * the components, summing their respective areas and lengths. │ │ │ │ │ * │ │ │ │ │ - * Automatically instantiated by the Google constructor, and not usually instantiated directly. │ │ │ │ │ + * Create a new instance with the <OpenLayers.Geometry.Collection> constructor. │ │ │ │ │ * │ │ │ │ │ - * Create a new event pane layer with the │ │ │ │ │ - * <OpenLayers.Layer.EventPane> constructor. │ │ │ │ │ - * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer> │ │ │ │ │ + * - <OpenLayers.Geometry> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.EventPane = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ +OpenLayers.Geometry.Collection = OpenLayers.Class(OpenLayers.Geometry, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: smoothDragPan │ │ │ │ │ - * {Boolean} smoothDragPan determines whether non-public/internal API │ │ │ │ │ - * methods are used for better performance while dragging EventPane │ │ │ │ │ - * layers. When not in sphericalMercator mode, the smoother dragging │ │ │ │ │ - * doesn't actually move north/south directly with the number of │ │ │ │ │ - * pixels moved, resulting in a slight offset when you drag your mouse │ │ │ │ │ - * north south with this option on. If this visual disparity bothers │ │ │ │ │ - * you, you should turn this option off, or use spherical mercator. │ │ │ │ │ - * Default is on. │ │ │ │ │ + * APIProperty: components │ │ │ │ │ + * {Array(<OpenLayers.Geometry>)} The component parts of this geometry │ │ │ │ │ */ │ │ │ │ │ - smoothDragPan: true, │ │ │ │ │ + components: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: isBaseLayer │ │ │ │ │ - * {Boolean} EventPaned layers are always base layers, by necessity. │ │ │ │ │ + * Property: componentTypes │ │ │ │ │ + * {Array(String)} An array of class names representing the types of │ │ │ │ │ + * components that the collection can include. A null value means the │ │ │ │ │ + * component types are not restricted. │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + componentTypes: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isFixed │ │ │ │ │ - * {Boolean} EventPaned layers are fixed by default. │ │ │ │ │ + * Constructor: OpenLayers.Geometry.Collection │ │ │ │ │ + * Creates a Geometry Collection -- a list of geoms. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * components - {Array(<OpenLayers.Geometry>)} Optional array of geometries │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - isFixed: true, │ │ │ │ │ + initialize: function(components) { │ │ │ │ │ + OpenLayers.Geometry.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.components = []; │ │ │ │ │ + if (components != null) { │ │ │ │ │ + this.addComponents(components); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: pane │ │ │ │ │ - * {DOMElement} A reference to the element that controls the events. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Destroy this geometry. │ │ │ │ │ */ │ │ │ │ │ - pane: null, │ │ │ │ │ - │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.components.length = 0; │ │ │ │ │ + this.components = null; │ │ │ │ │ + OpenLayers.Geometry.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: mapObject │ │ │ │ │ - * {Object} This is the object which will be used to load the 3rd party library │ │ │ │ │ - * in the case of the google layer, this will be of type GMap, │ │ │ │ │ - * in the case of the ve layer, this will be of type VEMap │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Clone this geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Collection>} An exact clone of this collection │ │ │ │ │ */ │ │ │ │ │ - mapObject: null, │ │ │ │ │ + clone: function() { │ │ │ │ │ + var geometry = eval("new " + this.CLASS_NAME + "()"); │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ + geometry.addComponent(this.components[i].clone()); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // catch any randomly tagged-on properties │ │ │ │ │ + OpenLayers.Util.applyDefaults(geometry, this); │ │ │ │ │ │ │ │ │ │ + return geometry; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.EventPane │ │ │ │ │ - * Create a new event pane layer │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + * Method: getComponentsString │ │ │ │ │ + * Get a string representing the components for this collection │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string representation of the components of this geometry │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (this.pane == null) { │ │ │ │ │ - this.pane = OpenLayers.Util.createDiv(this.div.id + "_EventPane"); │ │ │ │ │ + getComponentsString: function() { │ │ │ │ │ + var strings = []; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ + strings.push(this.components[i].toShortString()); │ │ │ │ │ } │ │ │ │ │ + return strings.join(","); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Deconstruct this layer. │ │ │ │ │ + * APIMethod: calculateBounds │ │ │ │ │ + * Recalculate the bounds by iterating through the components and │ │ │ │ │ + * calling calling extendBounds() on each item. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.mapObject = null; │ │ │ │ │ - this.pane = null; │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ + calculateBounds: function() { │ │ │ │ │ + this.bounds = null; │ │ │ │ │ + var bounds = new OpenLayers.Bounds(); │ │ │ │ │ + var components = this.components; │ │ │ │ │ + if (components) { │ │ │ │ │ + for (var i = 0, len = components.length; i < len; i++) { │ │ │ │ │ + bounds.extend(components[i].getBounds()); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // to preserve old behavior, we only set bounds if non-null │ │ │ │ │ + // in the future, we could add bounds.isEmpty() │ │ │ │ │ + if (bounds.left != null && bounds.bottom != null && │ │ │ │ │ + bounds.right != null && bounds.top != null) { │ │ │ │ │ + this.setBounds(bounds); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the layer. This is done through an accessor │ │ │ │ │ - * so that subclasses can override this and take special action once │ │ │ │ │ - * they have their map variable set. │ │ │ │ │ + * APIMethod: addComponents │ │ │ │ │ + * Add components to this geometry. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * components - {Array(<OpenLayers.Geometry>)} An array of geometries to add │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - this.pane.style.zIndex = parseInt(this.div.style.zIndex) + 1; │ │ │ │ │ - this.pane.style.display = this.div.style.display; │ │ │ │ │ - this.pane.style.width = "100%"; │ │ │ │ │ - this.pane.style.height = "100%"; │ │ │ │ │ - if (OpenLayers.BROWSER_NAME == "msie") { │ │ │ │ │ - this.pane.style.background = │ │ │ │ │ - "url(" + OpenLayers.Util.getImageLocation("blank.gif") + ")"; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.isFixed) { │ │ │ │ │ - this.map.viewPortDiv.appendChild(this.pane); │ │ │ │ │ - } else { │ │ │ │ │ - this.map.layerContainerDiv.appendChild(this.pane); │ │ │ │ │ + addComponents: function(components) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(components))) { │ │ │ │ │ + components = [components]; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // once our layer has been added to the map, we can load it │ │ │ │ │ - this.loadMapObject(); │ │ │ │ │ - │ │ │ │ │ - // if map didn't load, display warning │ │ │ │ │ - if (this.mapObject == null) { │ │ │ │ │ - this.loadWarningMessage(); │ │ │ │ │ + for (var i = 0, len = components.length; i < len; i++) { │ │ │ │ │ + this.addComponent(components[i]); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: removeMap │ │ │ │ │ - * On being removed from the map, we'll like to remove the invisible 'pane' │ │ │ │ │ - * div that we added to it on creation. │ │ │ │ │ + * Method: addComponent │ │ │ │ │ + * Add a new component (geometry) to the collection. If this.componentTypes │ │ │ │ │ + * is set, then the component class name must be in the componentTypes array. │ │ │ │ │ + * │ │ │ │ │ + * The bounds cache is reset. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * component - {<OpenLayers.Geometry>} A geometry to add │ │ │ │ │ + * index - {int} Optional index into the array to insert the component │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The component geometry was successfully added │ │ │ │ │ */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - if (this.pane && this.pane.parentNode) { │ │ │ │ │ - this.pane.parentNode.removeChild(this.pane); │ │ │ │ │ + addComponent: function(component, index) { │ │ │ │ │ + var added = false; │ │ │ │ │ + if (component) { │ │ │ │ │ + if (this.componentTypes == null || │ │ │ │ │ + (OpenLayers.Util.indexOf(this.componentTypes, │ │ │ │ │ + component.CLASS_NAME) > -1)) { │ │ │ │ │ + │ │ │ │ │ + if (index != null && (index < this.components.length)) { │ │ │ │ │ + var components1 = this.components.slice(0, index); │ │ │ │ │ + var components2 = this.components.slice(index, │ │ │ │ │ + this.components.length); │ │ │ │ │ + components1.push(component); │ │ │ │ │ + this.components = components1.concat(components2); │ │ │ │ │ + } else { │ │ │ │ │ + this.components.push(component); │ │ │ │ │ + } │ │ │ │ │ + component.parent = this; │ │ │ │ │ + this.clearBounds(); │ │ │ │ │ + added = true; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.prototype.removeMap.apply(this, arguments); │ │ │ │ │ + return added; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: loadWarningMessage │ │ │ │ │ - * If we can't load the map lib, then display an error message to the │ │ │ │ │ - * user and tell them where to go for help. │ │ │ │ │ - * │ │ │ │ │ - * This function sets up the layout for the warning message. Each 3rd │ │ │ │ │ - * party layer must implement its own getWarningHTML() function to │ │ │ │ │ - * provide the actual warning message. │ │ │ │ │ + * APIMethod: removeComponents │ │ │ │ │ + * Remove components from this geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * components - {Array(<OpenLayers.Geometry>)} The components to be removed │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} A component was removed. │ │ │ │ │ */ │ │ │ │ │ - loadWarningMessage: function() { │ │ │ │ │ - │ │ │ │ │ - this.div.style.backgroundColor = "darkblue"; │ │ │ │ │ - │ │ │ │ │ - var viewSize = this.map.getSize(); │ │ │ │ │ + removeComponents: function(components) { │ │ │ │ │ + var removed = false; │ │ │ │ │ │ │ │ │ │ - var msgW = Math.min(viewSize.w, 300); │ │ │ │ │ - var msgH = Math.min(viewSize.h, 200); │ │ │ │ │ - var size = new OpenLayers.Size(msgW, msgH); │ │ │ │ │ + if (!(OpenLayers.Util.isArray(components))) { │ │ │ │ │ + components = [components]; │ │ │ │ │ + } │ │ │ │ │ + for (var i = components.length - 1; i >= 0; --i) { │ │ │ │ │ + removed = this.removeComponent(components[i]) || removed; │ │ │ │ │ + } │ │ │ │ │ + return removed; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var centerPx = new OpenLayers.Pixel(viewSize.w / 2, viewSize.h / 2); │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeComponent │ │ │ │ │ + * Remove a component from this geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * component - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The component was removed. │ │ │ │ │ + */ │ │ │ │ │ + removeComponent: function(component) { │ │ │ │ │ │ │ │ │ │ - var topLeft = centerPx.add(-size.w / 2, -size.h / 2); │ │ │ │ │ + OpenLayers.Util.removeItem(this.components, component); │ │ │ │ │ │ │ │ │ │ - var div = OpenLayers.Util.createDiv(this.name + "_warning", │ │ │ │ │ - topLeft, │ │ │ │ │ - size, │ │ │ │ │ - null, │ │ │ │ │ - null, │ │ │ │ │ - null, │ │ │ │ │ - "auto"); │ │ │ │ │ + // clearBounds() so that it gets recalculated on the next call │ │ │ │ │ + // to this.getBounds(); │ │ │ │ │ + this.clearBounds(); │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - div.style.padding = "7px"; │ │ │ │ │ - div.style.backgroundColor = "yellow"; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getLength │ │ │ │ │ + * Calculate the length of this geometry │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The length of the geometry │ │ │ │ │ + */ │ │ │ │ │ + getLength: function() { │ │ │ │ │ + var length = 0.0; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ + length += this.components[i].getLength(); │ │ │ │ │ + } │ │ │ │ │ + return length; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - div.innerHTML = this.getWarningHTML(); │ │ │ │ │ - this.div.appendChild(div); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getArea │ │ │ │ │ + * Calculate the area of this geometry. Note how this function is overridden │ │ │ │ │ + * in <OpenLayers.Geometry.Polygon>. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The area of the collection by summing its parts │ │ │ │ │ + */ │ │ │ │ │ + getArea: function() { │ │ │ │ │ + var area = 0.0; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ + area += this.components[i].getArea(); │ │ │ │ │ + } │ │ │ │ │ + return area; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getWarningHTML │ │ │ │ │ - * To be implemented by subclasses. │ │ │ │ │ + * APIMethod: getGeodesicArea │ │ │ │ │ + * Calculate the approximate area of the polygon were it projected onto │ │ │ │ │ + * the earth. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * projection - {<OpenLayers.Projection>} The spatial reference system │ │ │ │ │ + * for the geometry coordinates. If not provided, Geographic/WGS84 is │ │ │ │ │ + * assumed. │ │ │ │ │ * │ │ │ │ │ + * Reference: │ │ │ │ │ + * Robert. G. Chamberlain and William H. Duquette, "Some Algorithms for │ │ │ │ │ + * Polygons on a Sphere", JPL Publication 07-03, Jet Propulsion │ │ │ │ │ + * Laboratory, Pasadena, CA, June 2007 http://trs-new.jpl.nasa.gov/dspace/handle/2014/40409 │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} String with information on why layer is broken, how to get │ │ │ │ │ - * it working. │ │ │ │ │ + * {float} The approximate geodesic area of the geometry in square meters. │ │ │ │ │ */ │ │ │ │ │ - getWarningHTML: function() { │ │ │ │ │ - //should be implemented by subclasses │ │ │ │ │ - return ""; │ │ │ │ │ + getGeodesicArea: function(projection) { │ │ │ │ │ + var area = 0.0; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ + area += this.components[i].getGeodesicArea(projection); │ │ │ │ │ + } │ │ │ │ │ + return area; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: display │ │ │ │ │ - * Set the display on the pane │ │ │ │ │ + * APIMethod: getCentroid │ │ │ │ │ + * │ │ │ │ │ + * Compute the centroid for this geometry collection. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * display - {Boolean} │ │ │ │ │ + * weighted - {Boolean} Perform the getCentroid computation recursively, │ │ │ │ │ + * returning an area weighted average of all geometries in this collection. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Point>} The centroid of the collection │ │ │ │ │ */ │ │ │ │ │ - display: function(display) { │ │ │ │ │ - OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ - this.pane.style.display = this.div.style.display; │ │ │ │ │ + getCentroid: function(weighted) { │ │ │ │ │ + if (!weighted) { │ │ │ │ │ + return this.components.length && this.components[0].getCentroid(); │ │ │ │ │ + } │ │ │ │ │ + var len = this.components.length; │ │ │ │ │ + if (!len) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var areas = []; │ │ │ │ │ + var centroids = []; │ │ │ │ │ + var areaSum = 0; │ │ │ │ │ + var minArea = Number.MAX_VALUE; │ │ │ │ │ + var component; │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + component = this.components[i]; │ │ │ │ │ + var area = component.getArea(); │ │ │ │ │ + var centroid = component.getCentroid(true); │ │ │ │ │ + if (isNaN(area) || isNaN(centroid.x) || isNaN(centroid.y)) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + areas.push(area); │ │ │ │ │ + areaSum += area; │ │ │ │ │ + minArea = (area < minArea && area > 0) ? area : minArea; │ │ │ │ │ + centroids.push(centroid); │ │ │ │ │ + } │ │ │ │ │ + len = areas.length; │ │ │ │ │ + if (areaSum === 0) { │ │ │ │ │ + // all the components in this collection have 0 area │ │ │ │ │ + // probably a collection of points -- weight all the points the same │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + areas[i] = 1; │ │ │ │ │ + } │ │ │ │ │ + areaSum = areas.length; │ │ │ │ │ + } else { │ │ │ │ │ + // normalize all the areas where the smallest area will get │ │ │ │ │ + // a value of 1 │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + areas[i] /= minArea; │ │ │ │ │ + } │ │ │ │ │ + areaSum /= minArea; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var xSum = 0, │ │ │ │ │ + ySum = 0, │ │ │ │ │ + centroid, area; │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + centroid = centroids[i]; │ │ │ │ │ + area = areas[i]; │ │ │ │ │ + xSum += centroid.x * area; │ │ │ │ │ + ySum += centroid.y * area; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return new OpenLayers.Geometry.Point(xSum / areaSum, ySum / areaSum); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setZIndex │ │ │ │ │ - * Set the z-index order for the pane. │ │ │ │ │ + * APIMethod: getGeodesicLength │ │ │ │ │ + * Calculate the approximate length of the geometry were it projected onto │ │ │ │ │ + * the earth. │ │ │ │ │ + * │ │ │ │ │ + * projection - {<OpenLayers.Projection>} The spatial reference system │ │ │ │ │ + * for the geometry coordinates. If not provided, Geographic/WGS84 is │ │ │ │ │ + * assumed. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * zIndex - {int} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The appoximate geodesic length of the geometry in meters. │ │ │ │ │ */ │ │ │ │ │ - setZIndex: function(zIndex) { │ │ │ │ │ - OpenLayers.Layer.prototype.setZIndex.apply(this, arguments); │ │ │ │ │ - this.pane.style.zIndex = parseInt(this.div.style.zIndex) + 1; │ │ │ │ │ + getGeodesicLength: function(projection) { │ │ │ │ │ + var length = 0.0; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ + length += this.components[i].getGeodesicLength(projection); │ │ │ │ │ + } │ │ │ │ │ + return length; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveByPx │ │ │ │ │ - * Move the layer based on pixel vector. To be implemented by subclasses. │ │ │ │ │ + * APIMethod: move │ │ │ │ │ + * Moves a geometry by the given displacement along positive x and y axes. │ │ │ │ │ + * This modifies the position of the geometry and clears the cached │ │ │ │ │ + * bounds. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * dx - {Number} The x coord of the displacement vector. │ │ │ │ │ - * dy - {Number} The y coord of the displacement vector. │ │ │ │ │ + * x - {Float} Distance to move geometry in positive x direction. │ │ │ │ │ + * y - {Float} Distance to move geometry in positive y direction. │ │ │ │ │ */ │ │ │ │ │ - moveByPx: function(dx, dy) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveByPx.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - if (this.dragPanMapObject) { │ │ │ │ │ - this.dragPanMapObject(dx, -dy); │ │ │ │ │ - } else { │ │ │ │ │ - this.moveTo(this.map.getCachedCenter()); │ │ │ │ │ + move: function(x, y) { │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ + this.components[i].move(x, y); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * Handle calls to move the layer. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: rotate │ │ │ │ │ + * Rotate a geometry around some origin │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * zoomChanged - {Boolean} │ │ │ │ │ - * dragging - {Boolean} │ │ │ │ │ + * angle - {Float} Rotation angle in degrees (measured counterclockwise │ │ │ │ │ + * from the positive x-axis) │ │ │ │ │ + * origin - {<OpenLayers.Geometry.Point>} Center point for the rotation │ │ │ │ │ */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - if (this.mapObject != null) { │ │ │ │ │ - │ │ │ │ │ - var newCenter = this.map.getCenter(); │ │ │ │ │ - var newZoom = this.map.getZoom(); │ │ │ │ │ - │ │ │ │ │ - if (newCenter != null) { │ │ │ │ │ - │ │ │ │ │ - var moOldCenter = this.getMapObjectCenter(); │ │ │ │ │ - var oldCenter = this.getOLLonLatFromMapObjectLonLat(moOldCenter); │ │ │ │ │ - │ │ │ │ │ - var moOldZoom = this.getMapObjectZoom(); │ │ │ │ │ - var oldZoom = this.getOLZoomFromMapObjectZoom(moOldZoom); │ │ │ │ │ - │ │ │ │ │ - if (!(newCenter.equals(oldCenter)) || newZoom != oldZoom) { │ │ │ │ │ - │ │ │ │ │ - if (!zoomChanged && oldCenter && this.dragPanMapObject && │ │ │ │ │ - this.smoothDragPan) { │ │ │ │ │ - var oldPx = this.map.getViewPortPxFromLonLat(oldCenter); │ │ │ │ │ - var newPx = this.map.getViewPortPxFromLonLat(newCenter); │ │ │ │ │ - this.dragPanMapObject(newPx.x - oldPx.x, oldPx.y - newPx.y); │ │ │ │ │ - } else { │ │ │ │ │ - var center = this.getMapObjectLonLatFromOLLonLat(newCenter); │ │ │ │ │ - var zoom = this.getMapObjectZoomFromOLZoom(newZoom); │ │ │ │ │ - this.setMapObjectCenter(center, zoom, dragging); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + rotate: function(angle, origin) { │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ + this.components[i].rotate(angle, origin); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ - /********************************************************/ │ │ │ │ │ - /* */ │ │ │ │ │ - /* Baselayer Functions */ │ │ │ │ │ - /* */ │ │ │ │ │ - /********************************************************/ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: getLonLatFromViewPortPx │ │ │ │ │ - * Get a map location from a pixel location │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * viewPortPx - {<OpenLayers.Pixel>} │ │ │ │ │ + * APIMethod: resize │ │ │ │ │ + * Resize a geometry relative to some origin. Use this method to apply │ │ │ │ │ + * a uniform scaling to a geometry. │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * scale - {Float} Factor by which to scale the geometry. A scale of 2 │ │ │ │ │ + * doubles the size of the geometry in each dimension │ │ │ │ │ + * (lines, for example, will be twice as long, and polygons │ │ │ │ │ + * will have four times the area). │ │ │ │ │ + * origin - {<OpenLayers.Geometry.Point>} Point of origin for resizing │ │ │ │ │ + * ratio - {Float} Optional x:y ratio for resizing. Default ratio is 1. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.LonLat>} An OpenLayers.LonLat which is the passed-in view │ │ │ │ │ - * port OpenLayers.Pixel, translated into lon/lat by map lib │ │ │ │ │ - * If the map lib is not loaded or not centered, returns null │ │ │ │ │ + * {<OpenLayers.Geometry>} - The current geometry. │ │ │ │ │ */ │ │ │ │ │ - getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ - var lonlat = null; │ │ │ │ │ - if ((this.mapObject != null) && │ │ │ │ │ - (this.getMapObjectCenter() != null)) { │ │ │ │ │ - var moPixel = this.getMapObjectPixelFromOLPixel(viewPortPx); │ │ │ │ │ - var moLonLat = this.getMapObjectLonLatFromMapObjectPixel(moPixel); │ │ │ │ │ - lonlat = this.getOLLonLatFromMapObjectLonLat(moLonLat); │ │ │ │ │ + resize: function(scale, origin, ratio) { │ │ │ │ │ + for (var i = 0; i < this.components.length; ++i) { │ │ │ │ │ + this.components[i].resize(scale, origin, ratio); │ │ │ │ │ } │ │ │ │ │ - return lonlat; │ │ │ │ │ + return this; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: getViewPortPxFromLonLat │ │ │ │ │ - * Get a pixel location from a map location │ │ │ │ │ + * APIMethod: distanceTo │ │ │ │ │ + * Calculate the closest distance between two geometries (on the x-y plane). │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ + * options - {Object} Optional properties for configuring the distance │ │ │ │ │ + * calculation. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * details - {Boolean} Return details from the distance calculation. │ │ │ │ │ + * Default is false. │ │ │ │ │ + * edge - {Boolean} Calculate the distance from this geometry to the │ │ │ │ │ + * nearest edge of the target geometry. Default is true. If true, │ │ │ │ │ + * calling distanceTo from a geometry that is wholly contained within │ │ │ │ │ + * the target will result in a non-zero distance. If false, whenever │ │ │ │ │ + * geometries intersect, calling distanceTo will return 0. If false, │ │ │ │ │ + * details cannot be returned. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Pixel>} An OpenLayers.Pixel which is the passed-in │ │ │ │ │ - * OpenLayers.LonLat, translated into view port pixels by map lib │ │ │ │ │ - * If map lib is not loaded or not centered, returns null │ │ │ │ │ + * {Number | Object} The distance between this geometry and the target. │ │ │ │ │ + * If details is true, the return will be an object with distance, │ │ │ │ │ + * x0, y0, x1, and y1 properties. The x0 and y0 properties represent │ │ │ │ │ + * the coordinates of the closest point on this geometry. The x1 and y1 │ │ │ │ │ + * properties represent the coordinates of the closest point on the │ │ │ │ │ + * target geometry. │ │ │ │ │ */ │ │ │ │ │ - getViewPortPxFromLonLat: function(lonlat) { │ │ │ │ │ - var viewPortPx = null; │ │ │ │ │ - if ((this.mapObject != null) && │ │ │ │ │ - (this.getMapObjectCenter() != null)) { │ │ │ │ │ - │ │ │ │ │ - var moLonLat = this.getMapObjectLonLatFromOLLonLat(lonlat); │ │ │ │ │ - var moPixel = this.getMapObjectPixelFromMapObjectLonLat(moLonLat); │ │ │ │ │ - │ │ │ │ │ - viewPortPx = this.getOLPixelFromMapObjectPixel(moPixel); │ │ │ │ │ + distanceTo: function(geometry, options) { │ │ │ │ │ + var edge = !(options && options.edge === false); │ │ │ │ │ + var details = edge && options && options.details; │ │ │ │ │ + var result, best, distance; │ │ │ │ │ + var min = Number.POSITIVE_INFINITY; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ + result = this.components[i].distanceTo(geometry, options); │ │ │ │ │ + distance = details ? result.distance : result; │ │ │ │ │ + if (distance < min) { │ │ │ │ │ + min = distance; │ │ │ │ │ + best = result; │ │ │ │ │ + if (min == 0) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return viewPortPx; │ │ │ │ │ + return best; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /********************************************************/ │ │ │ │ │ - /* */ │ │ │ │ │ - /* Translation Functions */ │ │ │ │ │ - /* */ │ │ │ │ │ - /* The following functions translate Map Object and */ │ │ │ │ │ - /* OL formats for Pixel, LonLat */ │ │ │ │ │ - /* */ │ │ │ │ │ - /********************************************************/ │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - // TRANSLATION: MapObject LatLng <-> OpenLayers.LonLat │ │ │ │ │ - // │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getOLLonLatFromMapObjectLonLat │ │ │ │ │ - * Get an OL style map location from a 3rd party style map location │ │ │ │ │ - * │ │ │ │ │ - * Parameters │ │ │ │ │ - * moLonLat - {Object} │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: equals │ │ │ │ │ + * Determine whether another geometry is equivalent to this one. Geometries │ │ │ │ │ + * are considered equivalent if all components have the same coordinates. │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} The geometry to test. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.LonLat>} An OpenLayers.LonLat, translated from the passed in │ │ │ │ │ - * MapObject LonLat │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ + * {Boolean} The supplied geometry is equivalent to this geometry. │ │ │ │ │ */ │ │ │ │ │ - getOLLonLatFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - var olLonLat = null; │ │ │ │ │ - if (moLonLat != null) { │ │ │ │ │ - var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - olLonLat = new OpenLayers.LonLat(lon, lat); │ │ │ │ │ + equals: function(geometry) { │ │ │ │ │ + var equivalent = true; │ │ │ │ │ + if (!geometry || !geometry.CLASS_NAME || │ │ │ │ │ + (this.CLASS_NAME != geometry.CLASS_NAME)) { │ │ │ │ │ + equivalent = false; │ │ │ │ │ + } else if (!(OpenLayers.Util.isArray(geometry.components)) || │ │ │ │ │ + (geometry.components.length != this.components.length)) { │ │ │ │ │ + equivalent = false; │ │ │ │ │ + } else { │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ + if (!this.components[i].equals(geometry.components[i])) { │ │ │ │ │ + equivalent = false; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return olLonLat; │ │ │ │ │ + return equivalent; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getMapObjectLonLatFromOLLonLat │ │ │ │ │ - * Get a 3rd party map location from an OL map location. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: transform │ │ │ │ │ + * Reproject the components geometry from source to dest. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * olLonLat - {<OpenLayers.LonLat>} │ │ │ │ │ + * source - {<OpenLayers.Projection>} │ │ │ │ │ + * dest - {<OpenLayers.Projection>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} A MapObject LonLat, translated from the passed in │ │ │ │ │ - * OpenLayers.LonLat │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ + * {<OpenLayers.Geometry>} │ │ │ │ │ */ │ │ │ │ │ - getMapObjectLonLatFromOLLonLat: function(olLonLat) { │ │ │ │ │ - var moLatLng = null; │ │ │ │ │ - if (olLonLat != null) { │ │ │ │ │ - moLatLng = this.getMapObjectLonLatFromLonLat(olLonLat.lon, │ │ │ │ │ - olLonLat.lat); │ │ │ │ │ + transform: function(source, dest) { │ │ │ │ │ + if (source && dest) { │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ + var component = this.components[i]; │ │ │ │ │ + component.transform(source, dest); │ │ │ │ │ + } │ │ │ │ │ + this.bounds = null; │ │ │ │ │ } │ │ │ │ │ - return moLatLng; │ │ │ │ │ + return this; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - // TRANSLATION: MapObject Pixel <-> OpenLayers.Pixel │ │ │ │ │ - // │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: getOLPixelFromMapObjectPixel │ │ │ │ │ - * Get an OL pixel location from a 3rd party pixel location. │ │ │ │ │ + * APIMethod: intersects │ │ │ │ │ + * Determine if the input geometry intersects this one. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * moPixel - {Object} │ │ │ │ │ - * │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} Any type of geometry. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Pixel>} An OpenLayers.Pixel, translated from the passed in │ │ │ │ │ - * MapObject Pixel │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ + * {Boolean} The input geometry intersects this one. │ │ │ │ │ */ │ │ │ │ │ - getOLPixelFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - var olPixel = null; │ │ │ │ │ - if (moPixel != null) { │ │ │ │ │ - var x = this.getXFromMapObjectPixel(moPixel); │ │ │ │ │ - var y = this.getYFromMapObjectPixel(moPixel); │ │ │ │ │ - olPixel = new OpenLayers.Pixel(x, y); │ │ │ │ │ + intersects: function(geometry) { │ │ │ │ │ + var intersect = false; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ + intersect = geometry.intersects(this.components[i]); │ │ │ │ │ + if (intersect) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return olPixel; │ │ │ │ │ + return intersect; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getMapObjectPixelFromOLPixel │ │ │ │ │ - * Get a 3rd party pixel location from an OL pixel location │ │ │ │ │ + * APIMethod: getVertices │ │ │ │ │ + * Return a list of all points in this geometry. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * olPixel - {<OpenLayers.Pixel>} │ │ │ │ │ - * │ │ │ │ │ + * nodes - {Boolean} For lines, only return vertices that are │ │ │ │ │ + * endpoints. If false, for lines, only vertices that are not │ │ │ │ │ + * endpoints will be returned. If not provided, all vertices will │ │ │ │ │ + * be returned. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} A MapObject Pixel, translated from the passed in │ │ │ │ │ - * OpenLayers.Pixel │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ + * {Array} A list of all vertices in the geometry. │ │ │ │ │ */ │ │ │ │ │ - getMapObjectPixelFromOLPixel: function(olPixel) { │ │ │ │ │ - var moPixel = null; │ │ │ │ │ - if (olPixel != null) { │ │ │ │ │ - moPixel = this.getMapObjectPixelFromXY(olPixel.x, olPixel.y); │ │ │ │ │ + getVertices: function(nodes) { │ │ │ │ │ + var vertices = []; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ + Array.prototype.push.apply( │ │ │ │ │ + vertices, this.components[i].getVertices(nodes) │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - return moPixel; │ │ │ │ │ + return vertices; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.EventPane" │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Geometry.Collection" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/FixedZoomLevels.js │ │ │ │ │ + OpenLayers/Geometry/MultiPoint.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Collection.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.FixedZoomLevels │ │ │ │ │ - * Some Layers will already have established zoom levels (like google │ │ │ │ │ - * or ve). Instead of trying to determine them and populate a resolutions[] │ │ │ │ │ - * Array with those values, we will hijack the resolution functionality │ │ │ │ │ - * here. │ │ │ │ │ - * │ │ │ │ │ - * When you subclass FixedZoomLevels: │ │ │ │ │ - * │ │ │ │ │ - * The initResolutions() call gets nullified, meaning no resolutions[] array │ │ │ │ │ - * is set up. Which would be a big problem getResolution() in Layer, since │ │ │ │ │ - * it merely takes map.zoom and indexes into resolutions[]... but.... │ │ │ │ │ - * │ │ │ │ │ - * The getResolution() call is also overridden. Instead of using the │ │ │ │ │ - * resolutions[] array, we simply calculate the current resolution based │ │ │ │ │ - * on the current extent and the current map size. But how will we be able │ │ │ │ │ - * to calculate the current extent without knowing the resolution...? │ │ │ │ │ - * │ │ │ │ │ - * The getExtent() function is also overridden. Instead of calculating extent │ │ │ │ │ - * based on the center point and the current resolution, we instead │ │ │ │ │ - * calculate the extent by getting the lonlats at the top-left and │ │ │ │ │ - * bottom-right by using the getLonLatFromViewPortPx() translation function, │ │ │ │ │ - * taken from the pixel locations (0,0) and the size of the map. But how │ │ │ │ │ - * will we be able to do lonlat-px translation without resolution....? │ │ │ │ │ - * │ │ │ │ │ - * The getZoomForResolution() method is overridden. Instead of indexing into │ │ │ │ │ - * the resolutions[] array, we call OpenLayers.Layer.getExent(), passing in │ │ │ │ │ - * the desired resolution. With this extent, we then call getZoomForExtent() │ │ │ │ │ - * │ │ │ │ │ - * │ │ │ │ │ - * Whenever you implement a layer using OpenLayers.Layer.FixedZoomLevels, │ │ │ │ │ - * it is your responsibility to provide the following three functions: │ │ │ │ │ - * │ │ │ │ │ - * - getLonLatFromViewPortPx │ │ │ │ │ - * - getViewPortPxFromLonLat │ │ │ │ │ - * - getZoomForExtent │ │ │ │ │ - * │ │ │ │ │ - * ...those three functions should generally be provided by any reasonable │ │ │ │ │ - * API that you might be working from. │ │ │ │ │ + * Class: OpenLayers.Geometry.MultiPoint │ │ │ │ │ + * MultiPoint is a collection of Points. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Geometry.MultiPoint> constructor. │ │ │ │ │ * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Geometry.Collection> │ │ │ │ │ + * - <OpenLayers.Geometry> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.FixedZoomLevels = OpenLayers.Class({ │ │ │ │ │ +OpenLayers.Geometry.MultiPoint = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Geometry.Collection, { │ │ │ │ │ │ │ │ │ │ - /********************************************************/ │ │ │ │ │ - /* */ │ │ │ │ │ - /* Baselayer Functions */ │ │ │ │ │ - /* */ │ │ │ │ │ - /* The following functions must all be implemented */ │ │ │ │ │ - /* by all base layers */ │ │ │ │ │ - /* */ │ │ │ │ │ - /********************************************************/ │ │ │ │ │ + /** │ │ │ │ │ + * Property: componentTypes │ │ │ │ │ + * {Array(String)} An array of class names representing the types of │ │ │ │ │ + * components that the collection can include. A null value means the │ │ │ │ │ + * component types are not restricted. │ │ │ │ │ + */ │ │ │ │ │ + componentTypes: ["OpenLayers.Geometry.Point"], │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.FixedZoomLevels │ │ │ │ │ - * Create a new fixed zoom levels layer. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function() { │ │ │ │ │ - //this class is only just to add the following functions... │ │ │ │ │ - // nothing to actually do here... but it is probably a good │ │ │ │ │ - // idea to have layers that use these functions call this │ │ │ │ │ - // inititalize() anyways, in case at some point we decide we │ │ │ │ │ - // do want to put some functionality or state in here. │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Geometry.MultiPoint │ │ │ │ │ + * Create a new MultiPoint Geometry │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * components - {Array(<OpenLayers.Geometry.Point>)} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.MultiPoint>} │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: addPoint │ │ │ │ │ + * Wrapper for <OpenLayers.Geometry.Collection.addComponent> │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} Point to be added │ │ │ │ │ + * index - {Integer} Optional index │ │ │ │ │ + */ │ │ │ │ │ + addPoint: function(point, index) { │ │ │ │ │ + this.addComponent(point, index); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: removePoint │ │ │ │ │ + * Wrapper for <OpenLayers.Geometry.Collection.removeComponent> │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} Point to be removed │ │ │ │ │ + */ │ │ │ │ │ + removePoint: function(point) { │ │ │ │ │ + this.removeComponent(point); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Geometry.MultiPoint" │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Geometry/Curve.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Geometry/MultiPoint.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Geometry.Curve │ │ │ │ │ + * A Curve is a MultiPoint, whose points are assumed to be connected. To │ │ │ │ │ + * this end, we provide a "getLength()" function, which iterates through │ │ │ │ │ + * the points, summing the distances between them. │ │ │ │ │ + * │ │ │ │ │ + * Inherits: │ │ │ │ │ + * - <OpenLayers.Geometry.MultiPoint> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Geometry.Curve = OpenLayers.Class(OpenLayers.Geometry.MultiPoint, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: initResolutions │ │ │ │ │ - * Populate the resolutions array │ │ │ │ │ + * Property: componentTypes │ │ │ │ │ + * {Array(String)} An array of class names representing the types of │ │ │ │ │ + * components that the collection can include. A null │ │ │ │ │ + * value means the component types are not restricted. │ │ │ │ │ */ │ │ │ │ │ - initResolutions: function() { │ │ │ │ │ + componentTypes: ["OpenLayers.Geometry.Point"], │ │ │ │ │ │ │ │ │ │ - var props = ['minZoomLevel', 'maxZoomLevel', 'numZoomLevels']; │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Geometry.Curve │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {Array(<OpenLayers.Geometry.Point>)} │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - for (var i = 0, len = props.length; i < len; i++) { │ │ │ │ │ - var property = props[i]; │ │ │ │ │ - this[property] = (this.options[property] != null) ? │ │ │ │ │ - this.options[property] : │ │ │ │ │ - this.map[property]; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getLength │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The length of the curve │ │ │ │ │ + */ │ │ │ │ │ + getLength: function() { │ │ │ │ │ + var length = 0.0; │ │ │ │ │ + if (this.components && (this.components.length > 1)) { │ │ │ │ │ + for (var i = 1, len = this.components.length; i < len; i++) { │ │ │ │ │ + length += this.components[i - 1].distanceTo(this.components[i]); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return length; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if ((this.minZoomLevel == null) || │ │ │ │ │ - (this.minZoomLevel < this.MIN_ZOOM_LEVEL)) { │ │ │ │ │ - this.minZoomLevel = this.MIN_ZOOM_LEVEL; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getGeodesicLength │ │ │ │ │ + * Calculate the approximate length of the geometry were it projected onto │ │ │ │ │ + * the earth. │ │ │ │ │ + * │ │ │ │ │ + * projection - {<OpenLayers.Projection>} The spatial reference system │ │ │ │ │ + * for the geometry coordinates. If not provided, Geographic/WGS84 is │ │ │ │ │ + * assumed. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The appoximate geodesic length of the geometry in meters. │ │ │ │ │ + */ │ │ │ │ │ + getGeodesicLength: function(projection) { │ │ │ │ │ + var geom = this; // so we can work with a clone if needed │ │ │ │ │ + if (projection) { │ │ │ │ │ + var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + if (!gg.equals(projection)) { │ │ │ │ │ + geom = this.clone().transform(projection, gg); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var length = 0.0; │ │ │ │ │ + if (geom.components && (geom.components.length > 1)) { │ │ │ │ │ + var p1, p2; │ │ │ │ │ + for (var i = 1, len = geom.components.length; i < len; i++) { │ │ │ │ │ + p1 = geom.components[i - 1]; │ │ │ │ │ + p2 = geom.components[i]; │ │ │ │ │ + // this returns km and requires lon/lat properties │ │ │ │ │ + length += OpenLayers.Util.distVincenty({ │ │ │ │ │ + lon: p1.x, │ │ │ │ │ + lat: p1.y │ │ │ │ │ + }, { │ │ │ │ │ + lon: p2.x, │ │ │ │ │ + lat: p2.y │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + // convert to m │ │ │ │ │ + return length * 1000; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // │ │ │ │ │ - // At this point, we know what the minimum desired zoom level is, and │ │ │ │ │ - // we must calculate the total number of zoom levels. │ │ │ │ │ - // │ │ │ │ │ - // Because we allow for the setting of either the 'numZoomLevels' │ │ │ │ │ - // or the 'maxZoomLevel' properties... on either the layer or the │ │ │ │ │ - // map, we have to define some rules to see which we take into │ │ │ │ │ - // account first in this calculation. │ │ │ │ │ - // │ │ │ │ │ - // The following is the precedence list for these properties: │ │ │ │ │ - // │ │ │ │ │ - // (1) numZoomLevels set on layer │ │ │ │ │ - // (2) maxZoomLevel set on layer │ │ │ │ │ - // (3) numZoomLevels set on map │ │ │ │ │ - // (4) maxZoomLevel set on map* │ │ │ │ │ - // (5) none of the above* │ │ │ │ │ - // │ │ │ │ │ - // *Note that options (4) and (5) are only possible if the user │ │ │ │ │ - // _explicitly_ sets the 'numZoomLevels' property on the map to │ │ │ │ │ - // null, since it is set by default to 16. │ │ │ │ │ - // │ │ │ │ │ + CLASS_NAME: "OpenLayers.Geometry.Curve" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Geometry/LineString.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - // │ │ │ │ │ - // Note to future: In 3.0, I think we should remove the default │ │ │ │ │ - // value of 16 for map.numZoomLevels. Rather, I think that value │ │ │ │ │ - // should be set as a default on the Layer.WMS class. If someone │ │ │ │ │ - // creates a 3rd party layer and does not specify any 'minZoomLevel', │ │ │ │ │ - // 'maxZoomLevel', or 'numZoomLevels', and has not explicitly │ │ │ │ │ - // specified any of those on the map object either.. then I think │ │ │ │ │ - // it is fair to say that s/he wants all the zoom levels available. │ │ │ │ │ - // │ │ │ │ │ - // By making map.numZoomLevels *null* by default, that will be the │ │ │ │ │ - // case. As it is, I don't feel comfortable changing that right now │ │ │ │ │ - // as it would be a glaring API change and actually would probably │ │ │ │ │ - // break many peoples' codes. │ │ │ │ │ - // │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - //the number of zoom levels we'd like to have. │ │ │ │ │ - var desiredZoomLevels; │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Geometry/Curve.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - //this is the maximum number of zoom levels the layer will allow, │ │ │ │ │ - // given the specified starting minimum zoom level. │ │ │ │ │ - var limitZoomLevels = this.MAX_ZOOM_LEVEL - this.minZoomLevel + 1; │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Geometry.LineString │ │ │ │ │ + * A LineString is a Curve which, once two points have been added to it, can │ │ │ │ │ + * never be less than two points long. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Geometry.Curve> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Geometry.LineString = OpenLayers.Class(OpenLayers.Geometry.Curve, { │ │ │ │ │ │ │ │ │ │ - if (((this.options.numZoomLevels == null) && │ │ │ │ │ - (this.options.maxZoomLevel != null)) // (2) │ │ │ │ │ - || │ │ │ │ │ - ((this.numZoomLevels == null) && │ │ │ │ │ - (this.maxZoomLevel != null)) // (4) │ │ │ │ │ - ) { │ │ │ │ │ - //calculate based on specified maxZoomLevel (on layer or map) │ │ │ │ │ - desiredZoomLevels = this.maxZoomLevel - this.minZoomLevel + 1; │ │ │ │ │ - } else { │ │ │ │ │ - //calculate based on specified numZoomLevels (on layer or map) │ │ │ │ │ - // this covers cases (1) and (3) │ │ │ │ │ - desiredZoomLevels = this.numZoomLevels; │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Geometry.LineString │ │ │ │ │ + * Create a new LineString geometry │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * points - {Array(<OpenLayers.Geometry.Point>)} An array of points used to │ │ │ │ │ + * generate the linestring │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: removeComponent │ │ │ │ │ + * Only allows removal of a point if there are three or more points in │ │ │ │ │ + * the linestring. (otherwise the result would be just a single point) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} The point to be removed │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The component was removed. │ │ │ │ │ + */ │ │ │ │ │ + removeComponent: function(point) { │ │ │ │ │ + var removed = this.components && (this.components.length > 2); │ │ │ │ │ + if (removed) { │ │ │ │ │ + OpenLayers.Geometry.Collection.prototype.removeComponent.apply(this, │ │ │ │ │ + arguments); │ │ │ │ │ } │ │ │ │ │ + return removed; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (desiredZoomLevels != null) { │ │ │ │ │ - //Now that we know what we would *like* the number of zoom levels │ │ │ │ │ - // to be, based on layer or map options, we have to make sure that │ │ │ │ │ - // it does not conflict with the actual limit, as specified by │ │ │ │ │ - // the constants on the layer itself (and calculated into the │ │ │ │ │ - // 'limitZoomLevels' variable). │ │ │ │ │ - this.numZoomLevels = Math.min(desiredZoomLevels, limitZoomLevels); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: intersects │ │ │ │ │ + * Test for instersection between two geometries. This is a cheapo │ │ │ │ │ + * implementation of the Bently-Ottmann algorigithm. It doesn't │ │ │ │ │ + * really keep track of a sweep line data structure. It is closer │ │ │ │ │ + * to the brute force method, except that segments are sorted and │ │ │ │ │ + * potential intersections are only calculated when bounding boxes │ │ │ │ │ + * intersect. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The input geometry intersects this geometry. │ │ │ │ │ + */ │ │ │ │ │ + intersects: function(geometry) { │ │ │ │ │ + var intersect = false; │ │ │ │ │ + var type = geometry.CLASS_NAME; │ │ │ │ │ + if (type == "OpenLayers.Geometry.LineString" || │ │ │ │ │ + type == "OpenLayers.Geometry.LinearRing" || │ │ │ │ │ + type == "OpenLayers.Geometry.Point") { │ │ │ │ │ + var segs1 = this.getSortedSegments(); │ │ │ │ │ + var segs2; │ │ │ │ │ + if (type == "OpenLayers.Geometry.Point") { │ │ │ │ │ + segs2 = [{ │ │ │ │ │ + x1: geometry.x, │ │ │ │ │ + y1: geometry.y, │ │ │ │ │ + x2: geometry.x, │ │ │ │ │ + y2: geometry.y │ │ │ │ │ + }]; │ │ │ │ │ + } else { │ │ │ │ │ + segs2 = geometry.getSortedSegments(); │ │ │ │ │ + } │ │ │ │ │ + var seg1, seg1x1, seg1x2, seg1y1, seg1y2, │ │ │ │ │ + seg2, seg2y1, seg2y2; │ │ │ │ │ + // sweep right │ │ │ │ │ + outer: for (var i = 0, len = segs1.length; i < len; ++i) { │ │ │ │ │ + seg1 = segs1[i]; │ │ │ │ │ + seg1x1 = seg1.x1; │ │ │ │ │ + seg1x2 = seg1.x2; │ │ │ │ │ + seg1y1 = seg1.y1; │ │ │ │ │ + seg1y2 = seg1.y2; │ │ │ │ │ + inner: for (var j = 0, jlen = segs2.length; j < jlen; ++j) { │ │ │ │ │ + seg2 = segs2[j]; │ │ │ │ │ + if (seg2.x1 > seg1x2) { │ │ │ │ │ + // seg1 still left of seg2 │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + if (seg2.x2 < seg1x1) { │ │ │ │ │ + // seg2 still left of seg1 │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + seg2y1 = seg2.y1; │ │ │ │ │ + seg2y2 = seg2.y2; │ │ │ │ │ + if (Math.min(seg2y1, seg2y2) > Math.max(seg1y1, seg1y2)) { │ │ │ │ │ + // seg2 above seg1 │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + if (Math.max(seg2y1, seg2y2) < Math.min(seg1y1, seg1y2)) { │ │ │ │ │ + // seg2 below seg1 │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.Geometry.segmentsIntersect(seg1, seg2)) { │ │ │ │ │ + intersect = true; │ │ │ │ │ + break outer; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } else { │ │ │ │ │ - // case (5) -- neither 'numZoomLevels' not 'maxZoomLevel' was │ │ │ │ │ - // set on either the layer or the map. So we just use the │ │ │ │ │ - // maximum limit as calculated by the layer's constants. │ │ │ │ │ - this.numZoomLevels = limitZoomLevels; │ │ │ │ │ + intersect = geometry.intersects(this); │ │ │ │ │ } │ │ │ │ │ + return intersect; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - //now that the 'numZoomLevels' is appropriately, safely set, │ │ │ │ │ - // we go back and re-calculate the 'maxZoomLevel'. │ │ │ │ │ - this.maxZoomLevel = this.minZoomLevel + this.numZoomLevels - 1; │ │ │ │ │ + /** │ │ │ │ │ + * Method: getSortedSegments │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} An array of segment objects. Segment objects have properties │ │ │ │ │ + * x1, y1, x2, and y2. The start point is represented by x1 and y1. │ │ │ │ │ + * The end point is represented by x2 and y2. Start and end are │ │ │ │ │ + * ordered so that x1 < x2. │ │ │ │ │ + */ │ │ │ │ │ + getSortedSegments: function() { │ │ │ │ │ + var numSeg = this.components.length - 1; │ │ │ │ │ + var segments = new Array(numSeg), │ │ │ │ │ + point1, point2; │ │ │ │ │ + for (var i = 0; i < numSeg; ++i) { │ │ │ │ │ + point1 = this.components[i]; │ │ │ │ │ + point2 = this.components[i + 1]; │ │ │ │ │ + if (point1.x < point2.x) { │ │ │ │ │ + segments[i] = { │ │ │ │ │ + x1: point1.x, │ │ │ │ │ + y1: point1.y, │ │ │ │ │ + x2: point2.x, │ │ │ │ │ + y2: point2.y │ │ │ │ │ + }; │ │ │ │ │ + } else { │ │ │ │ │ + segments[i] = { │ │ │ │ │ + x1: point2.x, │ │ │ │ │ + y1: point2.y, │ │ │ │ │ + x2: point1.x, │ │ │ │ │ + y2: point1.y │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // more efficient to define this somewhere static │ │ │ │ │ + function byX1(seg1, seg2) { │ │ │ │ │ + return seg1.x1 - seg2.x1; │ │ │ │ │ + } │ │ │ │ │ + return segments.sort(byX1); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (this.RESOLUTIONS != null) { │ │ │ │ │ - var resolutionsIndex = 0; │ │ │ │ │ - this.resolutions = []; │ │ │ │ │ - for (var i = this.minZoomLevel; i <= this.maxZoomLevel; i++) { │ │ │ │ │ - this.resolutions[resolutionsIndex++] = this.RESOLUTIONS[i]; │ │ │ │ │ + /** │ │ │ │ │ + * Method: splitWithSegment │ │ │ │ │ + * Split this geometry with the given segment. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * seg - {Object} An object with x1, y1, x2, and y2 properties referencing │ │ │ │ │ + * segment endpoint coordinates. │ │ │ │ │ + * options - {Object} Properties of this object will be used to determine │ │ │ │ │ + * how the split is conducted. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * edge - {Boolean} Allow splitting when only edges intersect. Default is │ │ │ │ │ + * true. If false, a vertex on the source segment must be within the │ │ │ │ │ + * tolerance distance of the intersection to be considered a split. │ │ │ │ │ + * tolerance - {Number} If a non-null value is provided, intersections │ │ │ │ │ + * within the tolerance distance of one of the source segment's │ │ │ │ │ + * endpoints will be assumed to occur at the endpoint. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object with *lines* and *points* properties. If the given │ │ │ │ │ + * segment intersects this linestring, the lines array will reference │ │ │ │ │ + * geometries that result from the split. The points array will contain │ │ │ │ │ + * all intersection points. Intersection points are sorted along the │ │ │ │ │ + * segment (in order from x1,y1 to x2,y2). │ │ │ │ │ + */ │ │ │ │ │ + splitWithSegment: function(seg, options) { │ │ │ │ │ + var edge = !(options && options.edge === false); │ │ │ │ │ + var tolerance = options && options.tolerance; │ │ │ │ │ + var lines = []; │ │ │ │ │ + var verts = this.getVertices(); │ │ │ │ │ + var points = []; │ │ │ │ │ + var intersections = []; │ │ │ │ │ + var split = false; │ │ │ │ │ + var vert1, vert2, point; │ │ │ │ │ + var node, vertex, target; │ │ │ │ │ + var interOptions = { │ │ │ │ │ + point: true, │ │ │ │ │ + tolerance: tolerance │ │ │ │ │ + }; │ │ │ │ │ + var result = null; │ │ │ │ │ + for (var i = 0, stop = verts.length - 2; i <= stop; ++i) { │ │ │ │ │ + vert1 = verts[i]; │ │ │ │ │ + points.push(vert1.clone()); │ │ │ │ │ + vert2 = verts[i + 1]; │ │ │ │ │ + target = { │ │ │ │ │ + x1: vert1.x, │ │ │ │ │ + y1: vert1.y, │ │ │ │ │ + x2: vert2.x, │ │ │ │ │ + y2: vert2.y │ │ │ │ │ + }; │ │ │ │ │ + point = OpenLayers.Geometry.segmentsIntersect( │ │ │ │ │ + seg, target, interOptions │ │ │ │ │ + ); │ │ │ │ │ + if (point instanceof OpenLayers.Geometry.Point) { │ │ │ │ │ + if ((point.x === seg.x1 && point.y === seg.y1) || │ │ │ │ │ + (point.x === seg.x2 && point.y === seg.y2) || │ │ │ │ │ + point.equals(vert1) || point.equals(vert2)) { │ │ │ │ │ + vertex = true; │ │ │ │ │ + } else { │ │ │ │ │ + vertex = false; │ │ │ │ │ + } │ │ │ │ │ + if (vertex || edge) { │ │ │ │ │ + // push intersections different than the previous │ │ │ │ │ + if (!point.equals(intersections[intersections.length - 1])) { │ │ │ │ │ + intersections.push(point.clone()); │ │ │ │ │ + } │ │ │ │ │ + if (i === 0) { │ │ │ │ │ + if (point.equals(vert1)) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (point.equals(vert2)) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + split = true; │ │ │ │ │ + if (!point.equals(vert1)) { │ │ │ │ │ + points.push(point); │ │ │ │ │ + } │ │ │ │ │ + lines.push(new OpenLayers.Geometry.LineString(points)); │ │ │ │ │ + points = [point.clone()]; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.maxResolution = this.resolutions[0]; │ │ │ │ │ - this.minResolution = this.resolutions[this.resolutions.length - 1]; │ │ │ │ │ } │ │ │ │ │ + if (split) { │ │ │ │ │ + points.push(vert2.clone()); │ │ │ │ │ + lines.push(new OpenLayers.Geometry.LineString(points)); │ │ │ │ │ + } │ │ │ │ │ + if (intersections.length > 0) { │ │ │ │ │ + // sort intersections along segment │ │ │ │ │ + var xDir = seg.x1 < seg.x2 ? 1 : -1; │ │ │ │ │ + var yDir = seg.y1 < seg.y2 ? 1 : -1; │ │ │ │ │ + result = { │ │ │ │ │ + lines: lines, │ │ │ │ │ + points: intersections.sort(function(p1, p2) { │ │ │ │ │ + return (xDir * p1.x - xDir * p2.x) || (yDir * p1.y - yDir * p2.y); │ │ │ │ │ + }) │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + return result; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getResolution │ │ │ │ │ - * Get the current map resolution │ │ │ │ │ + * Method: split │ │ │ │ │ + * Use this geometry (the source) to attempt to split a target geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * target - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ + * options - {Object} Properties of this object will be used to determine │ │ │ │ │ + * how the split is conducted. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * mutual - {Boolean} Split the source geometry in addition to the target │ │ │ │ │ + * geometry. Default is false. │ │ │ │ │ + * edge - {Boolean} Allow splitting when only edges intersect. Default is │ │ │ │ │ + * true. If false, a vertex on the source must be within the tolerance │ │ │ │ │ + * distance of the intersection to be considered a split. │ │ │ │ │ + * tolerance - {Number} If a non-null value is provided, intersections │ │ │ │ │ + * within the tolerance distance of an existing vertex on the source │ │ │ │ │ + * will be assumed to occur at the vertex. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Float} Map units per Pixel │ │ │ │ │ + * {Array} A list of geometries (of this same type as the target) that │ │ │ │ │ + * result from splitting the target with the source geometry. The │ │ │ │ │ + * source and target geometry will remain unmodified. If no split │ │ │ │ │ + * results, null will be returned. If mutual is true and a split │ │ │ │ │ + * results, return will be an array of two arrays - the first will be │ │ │ │ │ + * all geometries that result from splitting the source geometry and │ │ │ │ │ + * the second will be all geometries that result from splitting the │ │ │ │ │ + * target geometry. │ │ │ │ │ */ │ │ │ │ │ - getResolution: function() { │ │ │ │ │ - │ │ │ │ │ - if (this.resolutions != null) { │ │ │ │ │ - return OpenLayers.Layer.prototype.getResolution.apply(this, arguments); │ │ │ │ │ + split: function(target, options) { │ │ │ │ │ + var results = null; │ │ │ │ │ + var mutual = options && options.mutual; │ │ │ │ │ + var sourceSplit, targetSplit, sourceParts, targetParts; │ │ │ │ │ + if (target instanceof OpenLayers.Geometry.LineString) { │ │ │ │ │ + var verts = this.getVertices(); │ │ │ │ │ + var vert1, vert2, seg, splits, lines, point; │ │ │ │ │ + var points = []; │ │ │ │ │ + sourceParts = []; │ │ │ │ │ + for (var i = 0, stop = verts.length - 2; i <= stop; ++i) { │ │ │ │ │ + vert1 = verts[i]; │ │ │ │ │ + vert2 = verts[i + 1]; │ │ │ │ │ + seg = { │ │ │ │ │ + x1: vert1.x, │ │ │ │ │ + y1: vert1.y, │ │ │ │ │ + x2: vert2.x, │ │ │ │ │ + y2: vert2.y │ │ │ │ │ + }; │ │ │ │ │ + targetParts = targetParts || [target]; │ │ │ │ │ + if (mutual) { │ │ │ │ │ + points.push(vert1.clone()); │ │ │ │ │ + } │ │ │ │ │ + for (var j = 0; j < targetParts.length; ++j) { │ │ │ │ │ + splits = targetParts[j].splitWithSegment(seg, options); │ │ │ │ │ + if (splits) { │ │ │ │ │ + // splice in new features │ │ │ │ │ + lines = splits.lines; │ │ │ │ │ + if (lines.length > 0) { │ │ │ │ │ + lines.unshift(j, 1); │ │ │ │ │ + Array.prototype.splice.apply(targetParts, lines); │ │ │ │ │ + j += lines.length - 2; │ │ │ │ │ + } │ │ │ │ │ + if (mutual) { │ │ │ │ │ + for (var k = 0, len = splits.points.length; k < len; ++k) { │ │ │ │ │ + point = splits.points[k]; │ │ │ │ │ + if (!point.equals(vert1)) { │ │ │ │ │ + points.push(point); │ │ │ │ │ + sourceParts.push(new OpenLayers.Geometry.LineString(points)); │ │ │ │ │ + if (point.equals(vert2)) { │ │ │ │ │ + points = []; │ │ │ │ │ + } else { │ │ │ │ │ + points = [point.clone()]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (mutual && sourceParts.length > 0 && points.length > 0) { │ │ │ │ │ + points.push(vert2.clone()); │ │ │ │ │ + sourceParts.push(new OpenLayers.Geometry.LineString(points)); │ │ │ │ │ + } │ │ │ │ │ } else { │ │ │ │ │ - var resolution = null; │ │ │ │ │ - │ │ │ │ │ - var viewSize = this.map.getSize(); │ │ │ │ │ - var extent = this.getExtent(); │ │ │ │ │ - │ │ │ │ │ - if ((viewSize != null) && (extent != null)) { │ │ │ │ │ - resolution = Math.max(extent.getWidth() / viewSize.w, │ │ │ │ │ - extent.getHeight() / viewSize.h); │ │ │ │ │ + results = target.splitWith(this, options); │ │ │ │ │ + } │ │ │ │ │ + if (targetParts && targetParts.length > 1) { │ │ │ │ │ + targetSplit = true; │ │ │ │ │ + } else { │ │ │ │ │ + targetParts = []; │ │ │ │ │ + } │ │ │ │ │ + if (sourceParts && sourceParts.length > 1) { │ │ │ │ │ + sourceSplit = true; │ │ │ │ │ + } else { │ │ │ │ │ + sourceParts = []; │ │ │ │ │ + } │ │ │ │ │ + if (targetSplit || sourceSplit) { │ │ │ │ │ + if (mutual) { │ │ │ │ │ + results = [sourceParts, targetParts]; │ │ │ │ │ + } else { │ │ │ │ │ + results = targetParts; │ │ │ │ │ } │ │ │ │ │ - return resolution; │ │ │ │ │ } │ │ │ │ │ + return results; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getExtent │ │ │ │ │ - * Calculates using px-> lonlat translation functions on tl and br │ │ │ │ │ - * corners of viewport │ │ │ │ │ + * Method: splitWith │ │ │ │ │ + * Split this geometry (the target) with the given geometry (the source). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} A geometry used to split this │ │ │ │ │ + * geometry (the source). │ │ │ │ │ + * options - {Object} Properties of this object will be used to determine │ │ │ │ │ + * how the split is conducted. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * mutual - {Boolean} Split the source geometry in addition to the target │ │ │ │ │ + * geometry. Default is false. │ │ │ │ │ + * edge - {Boolean} Allow splitting when only edges intersect. Default is │ │ │ │ │ + * true. If false, a vertex on the source must be within the tolerance │ │ │ │ │ + * distance of the intersection to be considered a split. │ │ │ │ │ + * tolerance - {Number} If a non-null value is provided, intersections │ │ │ │ │ + * within the tolerance distance of an existing vertex on the source │ │ │ │ │ + * will be assumed to occur at the vertex. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} A Bounds object which represents the lon/lat │ │ │ │ │ - * bounds of the current viewPort. │ │ │ │ │ + * {Array} A list of geometries (of this same type as the target) that │ │ │ │ │ + * result from splitting the target with the source geometry. The │ │ │ │ │ + * source and target geometry will remain unmodified. If no split │ │ │ │ │ + * results, null will be returned. If mutual is true and a split │ │ │ │ │ + * results, return will be an array of two arrays - the first will be │ │ │ │ │ + * all geometries that result from splitting the source geometry and │ │ │ │ │ + * the second will be all geometries that result from splitting the │ │ │ │ │ + * target geometry. │ │ │ │ │ */ │ │ │ │ │ - getExtent: function() { │ │ │ │ │ - var size = this.map.getSize(); │ │ │ │ │ - var tl = this.getLonLatFromViewPortPx({ │ │ │ │ │ - x: 0, │ │ │ │ │ - y: 0 │ │ │ │ │ - }); │ │ │ │ │ - var br = this.getLonLatFromViewPortPx({ │ │ │ │ │ - x: size.w, │ │ │ │ │ - y: size.h │ │ │ │ │ - }); │ │ │ │ │ + splitWith: function(geometry, options) { │ │ │ │ │ + return geometry.split(this, options); │ │ │ │ │ │ │ │ │ │ - if ((tl != null) && (br != null)) { │ │ │ │ │ - return new OpenLayers.Bounds(tl.lon, br.lat, br.lon, tl.lat); │ │ │ │ │ - } else { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getZoomForResolution │ │ │ │ │ - * Get the zoom level for a given resolution │ │ │ │ │ + * APIMethod: getVertices │ │ │ │ │ + * Return a list of all points in this geometry. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resolution - {Float} │ │ │ │ │ + * nodes - {Boolean} For lines, only return vertices that are │ │ │ │ │ + * endpoints. If false, for lines, only vertices that are not │ │ │ │ │ + * endpoints will be returned. If not provided, all vertices will │ │ │ │ │ + * be returned. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Integer} A suitable zoom level for the specified resolution. │ │ │ │ │ - * If no baselayer is set, returns null. │ │ │ │ │ + * {Array} A list of all vertices in the geometry. │ │ │ │ │ */ │ │ │ │ │ - getZoomForResolution: function(resolution) { │ │ │ │ │ - │ │ │ │ │ - if (this.resolutions != null) { │ │ │ │ │ - return OpenLayers.Layer.prototype.getZoomForResolution.apply(this, arguments); │ │ │ │ │ + getVertices: function(nodes) { │ │ │ │ │ + var vertices; │ │ │ │ │ + if (nodes === true) { │ │ │ │ │ + vertices = [ │ │ │ │ │ + this.components[0], │ │ │ │ │ + this.components[this.components.length - 1] │ │ │ │ │ + ]; │ │ │ │ │ + } else if (nodes === false) { │ │ │ │ │ + vertices = this.components.slice(1, this.components.length - 1); │ │ │ │ │ } else { │ │ │ │ │ - var extent = OpenLayers.Layer.prototype.getExtent.apply(this, []); │ │ │ │ │ - return this.getZoomForExtent(extent); │ │ │ │ │ + vertices = this.components.slice(); │ │ │ │ │ } │ │ │ │ │ + return vertices; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /********************************************************/ │ │ │ │ │ - /* */ │ │ │ │ │ - /* Translation Functions */ │ │ │ │ │ - /* */ │ │ │ │ │ - /* The following functions translate GMaps and OL */ │ │ │ │ │ - /* formats for Pixel, LonLat, Bounds, and Zoom */ │ │ │ │ │ - /* */ │ │ │ │ │ - /********************************************************/ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - // TRANSLATION: MapObject Zoom <-> OpenLayers Zoom │ │ │ │ │ - // │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: getOLZoomFromMapObjectZoom │ │ │ │ │ - * Get the OL zoom index from the map object zoom level │ │ │ │ │ + * APIMethod: distanceTo │ │ │ │ │ + * Calculate the closest distance between two geometries (on the x-y plane). │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * moZoom - {Integer} │ │ │ │ │ - * │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ + * options - {Object} Optional properties for configuring the distance │ │ │ │ │ + * calculation. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * details - {Boolean} Return details from the distance calculation. │ │ │ │ │ + * Default is false. │ │ │ │ │ + * edge - {Boolean} Calculate the distance from this geometry to the │ │ │ │ │ + * nearest edge of the target geometry. Default is true. If true, │ │ │ │ │ + * calling distanceTo from a geometry that is wholly contained within │ │ │ │ │ + * the target will result in a non-zero distance. If false, whenever │ │ │ │ │ + * geometries intersect, calling distanceTo will return 0. If false, │ │ │ │ │ + * details cannot be returned. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Integer} An OpenLayers Zoom level, translated from the passed in zoom │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ + * {Number | Object} The distance between this geometry and the target. │ │ │ │ │ + * If details is true, the return will be an object with distance, │ │ │ │ │ + * x0, y0, x1, and x2 properties. The x0 and y0 properties represent │ │ │ │ │ + * the coordinates of the closest point on this geometry. The x1 and y1 │ │ │ │ │ + * properties represent the coordinates of the closest point on the │ │ │ │ │ + * target geometry. │ │ │ │ │ */ │ │ │ │ │ - getOLZoomFromMapObjectZoom: function(moZoom) { │ │ │ │ │ - var zoom = null; │ │ │ │ │ - if (moZoom != null) { │ │ │ │ │ - zoom = moZoom - this.minZoomLevel; │ │ │ │ │ - if (this.map.baseLayer !== this) { │ │ │ │ │ - zoom = this.map.baseLayer.getZoomForResolution( │ │ │ │ │ - this.getResolutionForZoom(zoom) │ │ │ │ │ - ); │ │ │ │ │ + distanceTo: function(geometry, options) { │ │ │ │ │ + var edge = !(options && options.edge === false); │ │ │ │ │ + var details = edge && options && options.details; │ │ │ │ │ + var result, best = {}; │ │ │ │ │ + var min = Number.POSITIVE_INFINITY; │ │ │ │ │ + if (geometry instanceof OpenLayers.Geometry.Point) { │ │ │ │ │ + var segs = this.getSortedSegments(); │ │ │ │ │ + var x = geometry.x; │ │ │ │ │ + var y = geometry.y; │ │ │ │ │ + var seg; │ │ │ │ │ + for (var i = 0, len = segs.length; i < len; ++i) { │ │ │ │ │ + seg = segs[i]; │ │ │ │ │ + result = OpenLayers.Geometry.distanceToSegment(geometry, seg); │ │ │ │ │ + if (result.distance < min) { │ │ │ │ │ + min = result.distance; │ │ │ │ │ + best = result; │ │ │ │ │ + if (min === 0) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // if distance increases and we cross y0 to the right of x0, no need to keep looking. │ │ │ │ │ + if (seg.x2 > x && ((y > seg.y1 && y < seg.y2) || (y < seg.y1 && y > seg.y2))) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (details) { │ │ │ │ │ + best = { │ │ │ │ │ + distance: best.distance, │ │ │ │ │ + x0: best.x, │ │ │ │ │ + y0: best.y, │ │ │ │ │ + x1: x, │ │ │ │ │ + y1: y │ │ │ │ │ + }; │ │ │ │ │ + } else { │ │ │ │ │ + best = best.distance; │ │ │ │ │ + } │ │ │ │ │ + } else if (geometry instanceof OpenLayers.Geometry.LineString) { │ │ │ │ │ + var segs0 = this.getSortedSegments(); │ │ │ │ │ + var segs1 = geometry.getSortedSegments(); │ │ │ │ │ + var seg0, seg1, intersection, x0, y0; │ │ │ │ │ + var len1 = segs1.length; │ │ │ │ │ + var interOptions = { │ │ │ │ │ + point: true │ │ │ │ │ + }; │ │ │ │ │ + outer: for (var i = 0, len = segs0.length; i < len; ++i) { │ │ │ │ │ + seg0 = segs0[i]; │ │ │ │ │ + x0 = seg0.x1; │ │ │ │ │ + y0 = seg0.y1; │ │ │ │ │ + for (var j = 0; j < len1; ++j) { │ │ │ │ │ + seg1 = segs1[j]; │ │ │ │ │ + intersection = OpenLayers.Geometry.segmentsIntersect(seg0, seg1, interOptions); │ │ │ │ │ + if (intersection) { │ │ │ │ │ + min = 0; │ │ │ │ │ + best = { │ │ │ │ │ + distance: 0, │ │ │ │ │ + x0: intersection.x, │ │ │ │ │ + y0: intersection.y, │ │ │ │ │ + x1: intersection.x, │ │ │ │ │ + y1: intersection.y │ │ │ │ │ + }; │ │ │ │ │ + break outer; │ │ │ │ │ + } else { │ │ │ │ │ + result = OpenLayers.Geometry.distanceToSegment({ │ │ │ │ │ + x: x0, │ │ │ │ │ + y: y0 │ │ │ │ │ + }, seg1); │ │ │ │ │ + if (result.distance < min) { │ │ │ │ │ + min = result.distance; │ │ │ │ │ + best = { │ │ │ │ │ + distance: min, │ │ │ │ │ + x0: x0, │ │ │ │ │ + y0: y0, │ │ │ │ │ + x1: result.x, │ │ │ │ │ + y1: result.y │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!details) { │ │ │ │ │ + best = best.distance; │ │ │ │ │ + } │ │ │ │ │ + if (min !== 0) { │ │ │ │ │ + // check the final vertex in this line's sorted segments │ │ │ │ │ + if (seg0) { │ │ │ │ │ + result = geometry.distanceTo( │ │ │ │ │ + new OpenLayers.Geometry.Point(seg0.x2, seg0.y2), │ │ │ │ │ + options │ │ │ │ │ + ); │ │ │ │ │ + var dist = details ? result.distance : result; │ │ │ │ │ + if (dist < min) { │ │ │ │ │ + if (details) { │ │ │ │ │ + best = { │ │ │ │ │ + distance: min, │ │ │ │ │ + x0: result.x1, │ │ │ │ │ + y0: result.y1, │ │ │ │ │ + x1: result.x0, │ │ │ │ │ + y1: result.y0 │ │ │ │ │ + }; │ │ │ │ │ + } else { │ │ │ │ │ + best = dist; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + best = geometry.distanceTo(this, options); │ │ │ │ │ + // swap since target comes from this line │ │ │ │ │ + if (details) { │ │ │ │ │ + best = { │ │ │ │ │ + distance: best.distance, │ │ │ │ │ + x0: best.x1, │ │ │ │ │ + y0: best.y1, │ │ │ │ │ + x1: best.x0, │ │ │ │ │ + y1: best.y0 │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return zoom; │ │ │ │ │ + return best; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getMapObjectZoomFromOLZoom │ │ │ │ │ - * Get the map object zoom level from the OL zoom level │ │ │ │ │ + * APIMethod: simplify │ │ │ │ │ + * This function will return a simplified LineString. │ │ │ │ │ + * Simplification is based on the Douglas-Peucker algorithm. │ │ │ │ │ + * │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * olZoom - {Integer} │ │ │ │ │ - * │ │ │ │ │ + * tolerance - {number} threshhold for simplification in map units │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Integer} A MapObject level, translated from the passed in olZoom │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ + * {OpenLayers.Geometry.LineString} the simplified LineString │ │ │ │ │ */ │ │ │ │ │ - getMapObjectZoomFromOLZoom: function(olZoom) { │ │ │ │ │ - var zoom = null; │ │ │ │ │ - if (olZoom != null) { │ │ │ │ │ - zoom = olZoom + this.minZoomLevel; │ │ │ │ │ - if (this.map.baseLayer !== this) { │ │ │ │ │ - zoom = this.getZoomForResolution( │ │ │ │ │ - this.map.baseLayer.getResolutionForZoom(zoom) │ │ │ │ │ - ); │ │ │ │ │ + simplify: function(tolerance) { │ │ │ │ │ + if (this && this !== null) { │ │ │ │ │ + var points = this.getVertices(); │ │ │ │ │ + if (points.length < 3) { │ │ │ │ │ + return this; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var compareNumbers = function(a, b) { │ │ │ │ │ + return (a - b); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Private function doing the Douglas-Peucker reduction │ │ │ │ │ + */ │ │ │ │ │ + var douglasPeuckerReduction = function(points, firstPoint, lastPoint, tolerance) { │ │ │ │ │ + var maxDistance = 0; │ │ │ │ │ + var indexFarthest = 0; │ │ │ │ │ + │ │ │ │ │ + for (var index = firstPoint, distance; index < lastPoint; index++) { │ │ │ │ │ + distance = perpendicularDistance(points[firstPoint], points[lastPoint], points[index]); │ │ │ │ │ + if (distance > maxDistance) { │ │ │ │ │ + maxDistance = distance; │ │ │ │ │ + indexFarthest = index; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (maxDistance > tolerance && indexFarthest != firstPoint) { │ │ │ │ │ + //Add the largest point that exceeds the tolerance │ │ │ │ │ + pointIndexsToKeep.push(indexFarthest); │ │ │ │ │ + douglasPeuckerReduction(points, firstPoint, indexFarthest, tolerance); │ │ │ │ │ + douglasPeuckerReduction(points, indexFarthest, lastPoint, tolerance); │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Private function calculating the perpendicular distance │ │ │ │ │ + * TODO: check whether OpenLayers.Geometry.LineString::distanceTo() is faster or slower │ │ │ │ │ + */ │ │ │ │ │ + var perpendicularDistance = function(point1, point2, point) { │ │ │ │ │ + //Area = |(1/2)(x1y2 + x2y3 + x3y1 - x2y1 - x3y2 - x1y3)| *Area of triangle │ │ │ │ │ + //Base = v((x1-x2)²+(x1-x2)²) *Base of Triangle* │ │ │ │ │ + //Area = .5*Base*H *Solve for height │ │ │ │ │ + //Height = Area/.5/Base │ │ │ │ │ + │ │ │ │ │ + var area = Math.abs(0.5 * (point1.x * point2.y + point2.x * point.y + point.x * point1.y - point2.x * point1.y - point.x * point2.y - point1.x * point.y)); │ │ │ │ │ + var bottom = Math.sqrt(Math.pow(point1.x - point2.x, 2) + Math.pow(point1.y - point2.y, 2)); │ │ │ │ │ + var height = area / bottom * 2; │ │ │ │ │ + │ │ │ │ │ + return height; │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + var firstPoint = 0; │ │ │ │ │ + var lastPoint = points.length - 1; │ │ │ │ │ + var pointIndexsToKeep = []; │ │ │ │ │ + │ │ │ │ │ + //Add the first and last index to the keepers │ │ │ │ │ + pointIndexsToKeep.push(firstPoint); │ │ │ │ │ + pointIndexsToKeep.push(lastPoint); │ │ │ │ │ + │ │ │ │ │ + //The first and the last point cannot be the same │ │ │ │ │ + while (points[firstPoint].equals(points[lastPoint])) { │ │ │ │ │ + lastPoint--; │ │ │ │ │ + //Addition: the first point not equal to first point in the LineString is kept as well │ │ │ │ │ + pointIndexsToKeep.push(lastPoint); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + douglasPeuckerReduction(points, firstPoint, lastPoint, tolerance); │ │ │ │ │ + var returnPoints = []; │ │ │ │ │ + pointIndexsToKeep.sort(compareNumbers); │ │ │ │ │ + for (var index = 0; index < pointIndexsToKeep.length; index++) { │ │ │ │ │ + returnPoints.push(points[pointIndexsToKeep[index]]); │ │ │ │ │ } │ │ │ │ │ + return new OpenLayers.Geometry.LineString(returnPoints); │ │ │ │ │ + │ │ │ │ │ + } else { │ │ │ │ │ + return this; │ │ │ │ │ } │ │ │ │ │ - return zoom; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.FixedZoomLevels" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Geometry.LineString" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Google.js │ │ │ │ │ + OpenLayers/Geometry/MultiLineString.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/SphericalMercator.js │ │ │ │ │ - * @requires OpenLayers/Layer/EventPane.js │ │ │ │ │ - * @requires OpenLayers/Layer/FixedZoomLevels.js │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Collection.js │ │ │ │ │ + * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.Google │ │ │ │ │ - * │ │ │ │ │ - * Provides a wrapper for Google's Maps API │ │ │ │ │ - * Normally the Terms of Use for this API do not allow wrapping, but Google │ │ │ │ │ - * have provided written consent to OpenLayers for this - see email in │ │ │ │ │ - * http://osgeo-org.1560.n6.nabble.com/Google-Maps-API-Terms-of-Use-changes-tp4910013p4911981.html │ │ │ │ │ + * Class: OpenLayers.Geometry.MultiLineString │ │ │ │ │ + * A MultiLineString is a geometry with multiple <OpenLayers.Geometry.LineString> │ │ │ │ │ + * components. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.SphericalMercator> │ │ │ │ │ - * - <OpenLayers.Layer.EventPane> │ │ │ │ │ - * - <OpenLayers.Layer.FixedZoomLevels> │ │ │ │ │ + * - <OpenLayers.Geometry.Collection> │ │ │ │ │ + * - <OpenLayers.Geometry> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Google = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Layer.EventPane, │ │ │ │ │ - OpenLayers.Layer.FixedZoomLevels, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: MIN_ZOOM_LEVEL │ │ │ │ │ - * {Integer} 0 │ │ │ │ │ - */ │ │ │ │ │ - MIN_ZOOM_LEVEL: 0, │ │ │ │ │ +OpenLayers.Geometry.MultiLineString = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Geometry.Collection, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: MAX_ZOOM_LEVEL │ │ │ │ │ - * {Integer} 21 │ │ │ │ │ + /** │ │ │ │ │ + * Property: componentTypes │ │ │ │ │ + * {Array(String)} An array of class names representing the types of │ │ │ │ │ + * components that the collection can include. A null value means the │ │ │ │ │ + * component types are not restricted. │ │ │ │ │ */ │ │ │ │ │ - MAX_ZOOM_LEVEL: 21, │ │ │ │ │ + componentTypes: ["OpenLayers.Geometry.LineString"], │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: RESOLUTIONS │ │ │ │ │ - * {Array(Float)} Hardcode these resolutions so that they are more closely │ │ │ │ │ - * tied with the standard wms projection │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Geometry.MultiLineString │ │ │ │ │ + * Constructor for a MultiLineString Geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * components - {Array(<OpenLayers.Geometry.LineString>)} │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - RESOLUTIONS: [ │ │ │ │ │ - 1.40625, │ │ │ │ │ - 0.703125, │ │ │ │ │ - 0.3515625, │ │ │ │ │ - 0.17578125, │ │ │ │ │ - 0.087890625, │ │ │ │ │ - 0.0439453125, │ │ │ │ │ - 0.02197265625, │ │ │ │ │ - 0.010986328125, │ │ │ │ │ - 0.0054931640625, │ │ │ │ │ - 0.00274658203125, │ │ │ │ │ - 0.001373291015625, │ │ │ │ │ - 0.0006866455078125, │ │ │ │ │ - 0.00034332275390625, │ │ │ │ │ - 0.000171661376953125, │ │ │ │ │ - 0.0000858306884765625, │ │ │ │ │ - 0.00004291534423828125, │ │ │ │ │ - 0.00002145767211914062, │ │ │ │ │ - 0.00001072883605957031, │ │ │ │ │ - 0.00000536441802978515, │ │ │ │ │ - 0.00000268220901489257, │ │ │ │ │ - 0.0000013411045074462891, │ │ │ │ │ - 0.00000067055225372314453 │ │ │ │ │ - ], │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: type │ │ │ │ │ - * {GMapType} │ │ │ │ │ + * Method: split │ │ │ │ │ + * Use this geometry (the source) to attempt to split a target geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ + * options - {Object} Properties of this object will be used to determine │ │ │ │ │ + * how the split is conducted. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * mutual - {Boolean} Split the source geometry in addition to the target │ │ │ │ │ + * geometry. Default is false. │ │ │ │ │ + * edge - {Boolean} Allow splitting when only edges intersect. Default is │ │ │ │ │ + * true. If false, a vertex on the source must be within the tolerance │ │ │ │ │ + * distance of the intersection to be considered a split. │ │ │ │ │ + * tolerance - {Number} If a non-null value is provided, intersections │ │ │ │ │ + * within the tolerance distance of an existing vertex on the source │ │ │ │ │ + * will be assumed to occur at the vertex. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} A list of geometries (of this same type as the target) that │ │ │ │ │ + * result from splitting the target with the source geometry. The │ │ │ │ │ + * source and target geometry will remain unmodified. If no split │ │ │ │ │ + * results, null will be returned. If mutual is true and a split │ │ │ │ │ + * results, return will be an array of two arrays - the first will be │ │ │ │ │ + * all geometries that result from splitting the source geometry and │ │ │ │ │ + * the second will be all geometries that result from splitting the │ │ │ │ │ + * target geometry. │ │ │ │ │ */ │ │ │ │ │ - type: null, │ │ │ │ │ + split: function(geometry, options) { │ │ │ │ │ + var results = null; │ │ │ │ │ + var mutual = options && options.mutual; │ │ │ │ │ + var splits, sourceLine, sourceLines, sourceSplit, targetSplit; │ │ │ │ │ + var sourceParts = []; │ │ │ │ │ + var targetParts = [geometry]; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ + sourceLine = this.components[i]; │ │ │ │ │ + sourceSplit = false; │ │ │ │ │ + for (var j = 0; j < targetParts.length; ++j) { │ │ │ │ │ + splits = sourceLine.split(targetParts[j], options); │ │ │ │ │ + if (splits) { │ │ │ │ │ + if (mutual) { │ │ │ │ │ + sourceLines = splits[0]; │ │ │ │ │ + for (var k = 0, klen = sourceLines.length; k < klen; ++k) { │ │ │ │ │ + if (k === 0 && sourceParts.length) { │ │ │ │ │ + sourceParts[sourceParts.length - 1].addComponent( │ │ │ │ │ + sourceLines[k] │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + sourceParts.push( │ │ │ │ │ + new OpenLayers.Geometry.MultiLineString([ │ │ │ │ │ + sourceLines[k] │ │ │ │ │ + ]) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + sourceSplit = true; │ │ │ │ │ + splits = splits[1]; │ │ │ │ │ + } │ │ │ │ │ + if (splits.length) { │ │ │ │ │ + // splice in new target parts │ │ │ │ │ + splits.unshift(j, 1); │ │ │ │ │ + Array.prototype.splice.apply(targetParts, splits); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!sourceSplit) { │ │ │ │ │ + // source line was not hit │ │ │ │ │ + if (sourceParts.length) { │ │ │ │ │ + // add line to existing multi │ │ │ │ │ + sourceParts[sourceParts.length - 1].addComponent( │ │ │ │ │ + sourceLine.clone() │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + // create a fresh multi │ │ │ │ │ + sourceParts = [ │ │ │ │ │ + new OpenLayers.Geometry.MultiLineString( │ │ │ │ │ + sourceLine.clone() │ │ │ │ │ + ) │ │ │ │ │ + ]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (sourceParts && sourceParts.length > 1) { │ │ │ │ │ + sourceSplit = true; │ │ │ │ │ + } else { │ │ │ │ │ + sourceParts = []; │ │ │ │ │ + } │ │ │ │ │ + if (targetParts && targetParts.length > 1) { │ │ │ │ │ + targetSplit = true; │ │ │ │ │ + } else { │ │ │ │ │ + targetParts = []; │ │ │ │ │ + } │ │ │ │ │ + if (sourceSplit || targetSplit) { │ │ │ │ │ + if (mutual) { │ │ │ │ │ + results = [sourceParts, targetParts]; │ │ │ │ │ + } else { │ │ │ │ │ + results = targetParts; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return results; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: wrapDateLine │ │ │ │ │ - * {Boolean} Allow user to pan forever east/west. Default is true. │ │ │ │ │ - * Setting this to false only restricts panning if │ │ │ │ │ - * <sphericalMercator> is true. │ │ │ │ │ + * Method: splitWith │ │ │ │ │ + * Split this geometry (the target) with the given geometry (the source). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} A geometry used to split this │ │ │ │ │ + * geometry (the source). │ │ │ │ │ + * options - {Object} Properties of this object will be used to determine │ │ │ │ │ + * how the split is conducted. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * mutual - {Boolean} Split the source geometry in addition to the target │ │ │ │ │ + * geometry. Default is false. │ │ │ │ │ + * edge - {Boolean} Allow splitting when only edges intersect. Default is │ │ │ │ │ + * true. If false, a vertex on the source must be within the tolerance │ │ │ │ │ + * distance of the intersection to be considered a split. │ │ │ │ │ + * tolerance - {Number} If a non-null value is provided, intersections │ │ │ │ │ + * within the tolerance distance of an existing vertex on the source │ │ │ │ │ + * will be assumed to occur at the vertex. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} A list of geometries (of this same type as the target) that │ │ │ │ │ + * result from splitting the target with the source geometry. The │ │ │ │ │ + * source and target geometry will remain unmodified. If no split │ │ │ │ │ + * results, null will be returned. If mutual is true and a split │ │ │ │ │ + * results, return will be an array of two arrays - the first will be │ │ │ │ │ + * all geometries that result from splitting the source geometry and │ │ │ │ │ + * the second will be all geometries that result from splitting the │ │ │ │ │ + * target geometry. │ │ │ │ │ */ │ │ │ │ │ - wrapDateLine: true, │ │ │ │ │ + splitWith: function(geometry, options) { │ │ │ │ │ + var results = null; │ │ │ │ │ + var mutual = options && options.mutual; │ │ │ │ │ + var splits, targetLine, sourceLines, sourceSplit, targetSplit, sourceParts, targetParts; │ │ │ │ │ + if (geometry instanceof OpenLayers.Geometry.LineString) { │ │ │ │ │ + targetParts = []; │ │ │ │ │ + sourceParts = [geometry]; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ + targetSplit = false; │ │ │ │ │ + targetLine = this.components[i]; │ │ │ │ │ + for (var j = 0; j < sourceParts.length; ++j) { │ │ │ │ │ + splits = sourceParts[j].split(targetLine, options); │ │ │ │ │ + if (splits) { │ │ │ │ │ + if (mutual) { │ │ │ │ │ + sourceLines = splits[0]; │ │ │ │ │ + if (sourceLines.length) { │ │ │ │ │ + // splice in new source parts │ │ │ │ │ + sourceLines.unshift(j, 1); │ │ │ │ │ + Array.prototype.splice.apply(sourceParts, sourceLines); │ │ │ │ │ + j += sourceLines.length - 2; │ │ │ │ │ + } │ │ │ │ │ + splits = splits[1]; │ │ │ │ │ + if (splits.length === 0) { │ │ │ │ │ + splits = [targetLine.clone()]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + for (var k = 0, klen = splits.length; k < klen; ++k) { │ │ │ │ │ + if (k === 0 && targetParts.length) { │ │ │ │ │ + targetParts[targetParts.length - 1].addComponent( │ │ │ │ │ + splits[k] │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + targetParts.push( │ │ │ │ │ + new OpenLayers.Geometry.MultiLineString([ │ │ │ │ │ + splits[k] │ │ │ │ │ + ]) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + targetSplit = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!targetSplit) { │ │ │ │ │ + // target component was not hit │ │ │ │ │ + if (targetParts.length) { │ │ │ │ │ + // add it to any existing multi-line │ │ │ │ │ + targetParts[targetParts.length - 1].addComponent( │ │ │ │ │ + targetLine.clone() │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + // or start with a fresh multi-line │ │ │ │ │ + targetParts = [ │ │ │ │ │ + new OpenLayers.Geometry.MultiLineString([ │ │ │ │ │ + targetLine.clone() │ │ │ │ │ + ]) │ │ │ │ │ + ]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + results = geometry.split(this); │ │ │ │ │ + } │ │ │ │ │ + if (sourceParts && sourceParts.length > 1) { │ │ │ │ │ + sourceSplit = true; │ │ │ │ │ + } else { │ │ │ │ │ + sourceParts = []; │ │ │ │ │ + } │ │ │ │ │ + if (targetParts && targetParts.length > 1) { │ │ │ │ │ + targetSplit = true; │ │ │ │ │ + } else { │ │ │ │ │ + targetParts = []; │ │ │ │ │ + } │ │ │ │ │ + if (sourceSplit || targetSplit) { │ │ │ │ │ + if (mutual) { │ │ │ │ │ + results = [sourceParts, targetParts]; │ │ │ │ │ + } else { │ │ │ │ │ + results = targetParts; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return results; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Geometry.MultiLineString" │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Geometry/LinearRing.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Geometry.LinearRing │ │ │ │ │ + * │ │ │ │ │ + * A Linear Ring is a special LineString which is closed. It closes itself │ │ │ │ │ + * automatically on every addPoint/removePoint by adding a copy of the first │ │ │ │ │ + * point as the last point. │ │ │ │ │ + * │ │ │ │ │ + * Also, as it is the first in the line family to close itself, a getArea() │ │ │ │ │ + * function is defined to calculate the enclosed area of the linearRing │ │ │ │ │ + * │ │ │ │ │ + * Inherits: │ │ │ │ │ + * - <OpenLayers.Geometry.LineString> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Geometry.LinearRing = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Geometry.LineString, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: sphericalMercator │ │ │ │ │ - * {Boolean} Should the map act as a mercator-projected map? This will │ │ │ │ │ - * cause all interactions with the map to be in the actual map │ │ │ │ │ - * projection, which allows support for vector drawing, overlaying │ │ │ │ │ - * other maps, etc. │ │ │ │ │ + * Property: componentTypes │ │ │ │ │ + * {Array(String)} An array of class names representing the types of │ │ │ │ │ + * components that the collection can include. A null │ │ │ │ │ + * value means the component types are not restricted. │ │ │ │ │ */ │ │ │ │ │ - sphericalMercator: false, │ │ │ │ │ + componentTypes: ["OpenLayers.Geometry.Point"], │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {Number} The version of the Google Maps API │ │ │ │ │ + * Constructor: OpenLayers.Geometry.LinearRing │ │ │ │ │ + * Linear rings are constructed with an array of points. This array │ │ │ │ │ + * can represent a closed or open ring. If the ring is open (the last │ │ │ │ │ + * point does not equal the first point), the constructor will close │ │ │ │ │ + * the ring. If the ring is already closed (the last point does equal │ │ │ │ │ + * the first point), it will be left closed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * points - {Array(<OpenLayers.Geometry.Point>)} points │ │ │ │ │ */ │ │ │ │ │ - version: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Google │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: addComponent │ │ │ │ │ + * Adds a point to geometry components. If the point is to be added to │ │ │ │ │ + * the end of the components array and it is the same as the last point │ │ │ │ │ + * already in that array, the duplicate point is not added. This has │ │ │ │ │ + * the effect of closing the ring if it is not already closed, and │ │ │ │ │ + * doing the right thing if it is already closed. This behavior can │ │ │ │ │ + * be overridden by calling the method with a non-null index as the │ │ │ │ │ + * second argument. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} A name for the layer. │ │ │ │ │ - * options - {Object} An optional object whose properties will be set │ │ │ │ │ - * on the layer. │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * index - {Integer} Index into the array to insert the component │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Was the Point successfully added? │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - if (!options.version) { │ │ │ │ │ - options.version = typeof GMap2 === "function" ? "2" : "3"; │ │ │ │ │ - } │ │ │ │ │ - var mixin = OpenLayers.Layer.Google["v" + │ │ │ │ │ - options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (mixin) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, mixin); │ │ │ │ │ - } else { │ │ │ │ │ - throw "Unsupported Google Maps API version: " + options.version; │ │ │ │ │ - } │ │ │ │ │ + addComponent: function(point, index) { │ │ │ │ │ + var added = false; │ │ │ │ │ │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, mixin.DEFAULTS); │ │ │ │ │ - if (options.maxExtent) { │ │ │ │ │ - options.maxExtent = options.maxExtent.clone(); │ │ │ │ │ + //remove last point │ │ │ │ │ + var lastPoint = this.components.pop(); │ │ │ │ │ + │ │ │ │ │ + // given an index, add the point │ │ │ │ │ + // without an index only add non-duplicate points │ │ │ │ │ + if (index != null || !point.equals(lastPoint)) { │ │ │ │ │ + added = OpenLayers.Geometry.Collection.prototype.addComponent.apply(this, │ │ │ │ │ + arguments); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.initialize.apply(this, │ │ │ │ │ - [name, options]); │ │ │ │ │ - OpenLayers.Layer.FixedZoomLevels.prototype.initialize.apply(this, │ │ │ │ │ - [name, options]); │ │ │ │ │ + //append copy of first point │ │ │ │ │ + var firstPoint = this.components[0]; │ │ │ │ │ + OpenLayers.Geometry.Collection.prototype.addComponent.apply(this, │ │ │ │ │ + [firstPoint]); │ │ │ │ │ │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - OpenLayers.Util.extend(this, OpenLayers.Layer.SphericalMercator); │ │ │ │ │ - this.initMercatorParameters(); │ │ │ │ │ - } │ │ │ │ │ + return added; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ + * APIMethod: removeComponent │ │ │ │ │ + * Removes a point from geometry components. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Google>} An exact clone of this layer │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The component was removed. │ │ │ │ │ */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - /** │ │ │ │ │ - * This method isn't intended to be called by a subclass and it │ │ │ │ │ - * doesn't call the same method on the superclass. We don't call │ │ │ │ │ - * the super's clone because we don't want properties that are set │ │ │ │ │ - * on this layer after initialize (i.e. this.mapObject etc.). │ │ │ │ │ - */ │ │ │ │ │ - return new OpenLayers.Layer.Google( │ │ │ │ │ - this.name, this.getOptions() │ │ │ │ │ - ); │ │ │ │ │ + removeComponent: function(point) { │ │ │ │ │ + var removed = this.components && (this.components.length > 3); │ │ │ │ │ + if (removed) { │ │ │ │ │ + //remove last point │ │ │ │ │ + this.components.pop(); │ │ │ │ │ + │ │ │ │ │ + //remove our point │ │ │ │ │ + OpenLayers.Geometry.Collection.prototype.removeComponent.apply(this, │ │ │ │ │ + arguments); │ │ │ │ │ + //append copy of first point │ │ │ │ │ + var firstPoint = this.components[0]; │ │ │ │ │ + OpenLayers.Geometry.Collection.prototype.addComponent.apply(this, │ │ │ │ │ + [firstPoint]); │ │ │ │ │ + } │ │ │ │ │ + return removed; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setVisibility │ │ │ │ │ - * Set the visibility flag for the layer and hide/show & redraw │ │ │ │ │ - * accordingly. Fire event unless otherwise specified │ │ │ │ │ - * │ │ │ │ │ - * Note that visibility is no longer simply whether or not the layer's │ │ │ │ │ - * style.display is set to "block". Now we store a 'visibility' state │ │ │ │ │ - * property on the layer class, this allows us to remember whether or │ │ │ │ │ - * not we *desire* for a layer to be visible. In the case where the │ │ │ │ │ - * map's resolution is out of the layer's range, this desire may be │ │ │ │ │ - * subverted. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: move │ │ │ │ │ + * Moves a geometry by the given displacement along positive x and y axes. │ │ │ │ │ + * This modifies the position of the geometry and clears the cached │ │ │ │ │ + * bounds. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * visible - {Boolean} Display the layer (if in range) │ │ │ │ │ + * x - {Float} Distance to move geometry in positive x direction. │ │ │ │ │ + * y - {Float} Distance to move geometry in positive y direction. │ │ │ │ │ */ │ │ │ │ │ - setVisibility: function(visible) { │ │ │ │ │ - // sharing a map container, opacity has to be set per layer │ │ │ │ │ - var opacity = this.opacity == null ? 1 : this.opacity; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.setVisibility.apply(this, arguments); │ │ │ │ │ - this.setOpacity(opacity); │ │ │ │ │ + move: function(x, y) { │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len - 1; i++) { │ │ │ │ │ + this.components[i].move(x, y); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: display │ │ │ │ │ - * Hide or show the Layer │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: rotate │ │ │ │ │ + * Rotate a geometry around some origin │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * visible - {Boolean} │ │ │ │ │ + * angle - {Float} Rotation angle in degrees (measured counterclockwise │ │ │ │ │ + * from the positive x-axis) │ │ │ │ │ + * origin - {<OpenLayers.Geometry.Point>} Center point for the rotation │ │ │ │ │ */ │ │ │ │ │ - display: function(visible) { │ │ │ │ │ - if (!this._dragging) { │ │ │ │ │ - this.setGMapVisibility(visible); │ │ │ │ │ + rotate: function(angle, origin) { │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len - 1; ++i) { │ │ │ │ │ + this.components[i].rotate(angle, origin); │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.display.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: resize │ │ │ │ │ + * Resize a geometry relative to some origin. Use this method to apply │ │ │ │ │ + * a uniform scaling to a geometry. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * zoomChanged - {Boolean} Tells when zoom has changed, as layers have to │ │ │ │ │ - * do some init work in that case. │ │ │ │ │ - * dragging - {Boolean} │ │ │ │ │ + * scale - {Float} Factor by which to scale the geometry. A scale of 2 │ │ │ │ │ + * doubles the size of the geometry in each dimension │ │ │ │ │ + * (lines, for example, will be twice as long, and polygons │ │ │ │ │ + * will have four times the area). │ │ │ │ │ + * origin - {<OpenLayers.Geometry.Point>} Point of origin for resizing │ │ │ │ │ + * ratio - {Float} Optional x:y ratio for resizing. Default ratio is 1. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry>} - The current geometry. │ │ │ │ │ */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - this._dragging = dragging; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - delete this._dragging; │ │ │ │ │ + resize: function(scale, origin, ratio) { │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len - 1; ++i) { │ │ │ │ │ + this.components[i].resize(scale, origin, ratio); │ │ │ │ │ + } │ │ │ │ │ + return this; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setOpacity │ │ │ │ │ - * Sets the opacity for the entire layer (all images) │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: transform │ │ │ │ │ + * Reproject the components geometry from source to dest. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * opacity - {Float} │ │ │ │ │ + * source - {<OpenLayers.Projection>} │ │ │ │ │ + * dest - {<OpenLayers.Projection>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry>} │ │ │ │ │ */ │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - if (opacity !== this.opacity) { │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "opacity" │ │ │ │ │ - }); │ │ │ │ │ + transform: function(source, dest) { │ │ │ │ │ + if (source && dest) { │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len - 1; i++) { │ │ │ │ │ + var component = this.components[i]; │ │ │ │ │ + component.transform(source, dest); │ │ │ │ │ } │ │ │ │ │ - this.opacity = opacity; │ │ │ │ │ - } │ │ │ │ │ - // Though this layer's opacity may not change, we're sharing a container │ │ │ │ │ - // and need to update the opacity for the entire container. │ │ │ │ │ - if (this.getVisibility()) { │ │ │ │ │ - var container = this.getMapContainer(); │ │ │ │ │ - OpenLayers.Util.modifyDOMElement( │ │ │ │ │ - container, null, null, null, null, null, null, opacity │ │ │ │ │ - ); │ │ │ │ │ + this.bounds = null; │ │ │ │ │ } │ │ │ │ │ + return this; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up this layer. │ │ │ │ │ + * APIMethod: getCentroid │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Point>} The centroid of the collection │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - /** │ │ │ │ │ - * We have to override this method because the event pane destroy │ │ │ │ │ - * deletes the mapObject reference before removing this layer from │ │ │ │ │ - * the map. │ │ │ │ │ - */ │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.setGMapVisibility(false); │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache && cache.count <= 1) { │ │ │ │ │ - this.removeGMapElements(); │ │ │ │ │ + getCentroid: function() { │ │ │ │ │ + if (this.components) { │ │ │ │ │ + var len = this.components.length; │ │ │ │ │ + if (len > 0 && len <= 2) { │ │ │ │ │ + return this.components[0].clone(); │ │ │ │ │ + } else if (len > 2) { │ │ │ │ │ + var sumX = 0.0; │ │ │ │ │ + var sumY = 0.0; │ │ │ │ │ + var x0 = this.components[0].x; │ │ │ │ │ + var y0 = this.components[0].y; │ │ │ │ │ + var area = -1 * this.getArea(); │ │ │ │ │ + if (area != 0) { │ │ │ │ │ + for (var i = 0; i < len - 1; i++) { │ │ │ │ │ + var b = this.components[i]; │ │ │ │ │ + var c = this.components[i + 1]; │ │ │ │ │ + sumX += (b.x + c.x - 2 * x0) * ((b.x - x0) * (c.y - y0) - (c.x - x0) * (b.y - y0)); │ │ │ │ │ + sumY += (b.y + c.y - 2 * y0) * ((b.x - x0) * (c.y - y0) - (c.x - x0) * (b.y - y0)); │ │ │ │ │ + } │ │ │ │ │ + var x = x0 + sumX / (6 * area); │ │ │ │ │ + var y = y0 + sumY / (6 * area); │ │ │ │ │ + } else { │ │ │ │ │ + for (var i = 0; i < len - 1; i++) { │ │ │ │ │ + sumX += this.components[i].x; │ │ │ │ │ + sumY += this.components[i].y; │ │ │ │ │ + } │ │ │ │ │ + var x = sumX / (len - 1); │ │ │ │ │ + var y = sumY / (len - 1); │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.Point(x, y); │ │ │ │ │ + } else { │ │ │ │ │ + return null; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removeGMapElements │ │ │ │ │ - * Remove all elements added to the dom. This should only be called if │ │ │ │ │ - * this is the last of the Google layers for the given map. │ │ │ │ │ + * APIMethod: getArea │ │ │ │ │ + * Note - The area is positive if the ring is oriented CW, otherwise │ │ │ │ │ + * it will be negative. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The signed area for a ring. │ │ │ │ │ */ │ │ │ │ │ - removeGMapElements: function() { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - // remove shared elements from dom │ │ │ │ │ - var container = this.mapObject && this.getMapContainer(); │ │ │ │ │ - if (container && container.parentNode) { │ │ │ │ │ - container.parentNode.removeChild(container); │ │ │ │ │ - } │ │ │ │ │ - var termsOfUse = cache.termsOfUse; │ │ │ │ │ - if (termsOfUse && termsOfUse.parentNode) { │ │ │ │ │ - termsOfUse.parentNode.removeChild(termsOfUse); │ │ │ │ │ + getArea: function() { │ │ │ │ │ + var area = 0.0; │ │ │ │ │ + if (this.components && (this.components.length > 2)) { │ │ │ │ │ + var sum = 0.0; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len - 1; i++) { │ │ │ │ │ + var b = this.components[i]; │ │ │ │ │ + var c = this.components[i + 1]; │ │ │ │ │ + sum += (b.x + c.x) * (c.y - b.y); │ │ │ │ │ } │ │ │ │ │ - var poweredBy = cache.poweredBy; │ │ │ │ │ - if (poweredBy && poweredBy.parentNode) { │ │ │ │ │ - poweredBy.parentNode.removeChild(poweredBy); │ │ │ │ │ + area = -sum / 2.0; │ │ │ │ │ + } │ │ │ │ │ + return area; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getGeodesicArea │ │ │ │ │ + * Calculate the approximate area of the polygon were it projected onto │ │ │ │ │ + * the earth. Note that this area will be positive if ring is oriented │ │ │ │ │ + * clockwise, otherwise it will be negative. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * projection - {<OpenLayers.Projection>} The spatial reference system │ │ │ │ │ + * for the geometry coordinates. If not provided, Geographic/WGS84 is │ │ │ │ │ + * assumed. │ │ │ │ │ + * │ │ │ │ │ + * Reference: │ │ │ │ │ + * Robert. G. Chamberlain and William H. Duquette, "Some Algorithms for │ │ │ │ │ + * Polygons on a Sphere", JPL Publication 07-03, Jet Propulsion │ │ │ │ │ + * Laboratory, Pasadena, CA, June 2007 http://trs-new.jpl.nasa.gov/dspace/handle/2014/40409 │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {float} The approximate signed geodesic area of the polygon in square │ │ │ │ │ + * meters. │ │ │ │ │ + */ │ │ │ │ │ + getGeodesicArea: function(projection) { │ │ │ │ │ + var ring = this; // so we can work with a clone if needed │ │ │ │ │ + if (projection) { │ │ │ │ │ + var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + if (!gg.equals(projection)) { │ │ │ │ │ + ring = this.clone().transform(projection, gg); │ │ │ │ │ } │ │ │ │ │ - if (this.mapObject && window.google && google.maps && │ │ │ │ │ - google.maps.event && google.maps.event.clearListeners) { │ │ │ │ │ - google.maps.event.clearListeners(this.mapObject, 'tilesloaded'); │ │ │ │ │ + } │ │ │ │ │ + var area = 0.0; │ │ │ │ │ + var len = ring.components && ring.components.length; │ │ │ │ │ + if (len > 2) { │ │ │ │ │ + var p1, p2; │ │ │ │ │ + for (var i = 0; i < len - 1; i++) { │ │ │ │ │ + p1 = ring.components[i]; │ │ │ │ │ + p2 = ring.components[i + 1]; │ │ │ │ │ + area += OpenLayers.Util.rad(p2.x - p1.x) * │ │ │ │ │ + (2 + Math.sin(OpenLayers.Util.rad(p1.y)) + │ │ │ │ │ + Math.sin(OpenLayers.Util.rad(p2.y))); │ │ │ │ │ } │ │ │ │ │ + area = area * 6378137.0 * 6378137.0 / 2.0; │ │ │ │ │ } │ │ │ │ │ + return area; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: removeMap │ │ │ │ │ - * On being removed from the map, also remove termsOfUse and poweredBy divs │ │ │ │ │ - * │ │ │ │ │ + * Method: containsPoint │ │ │ │ │ + * Test if a point is inside a linear ring. For the case where a point │ │ │ │ │ + * is coincident with a linear ring edge, returns 1. Otherwise, │ │ │ │ │ + * returns boolean. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean | Number} The point is inside the linear ring. Returns 1 if │ │ │ │ │ + * the point is coincident with an edge. Returns boolean otherwise. │ │ │ │ │ */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - // hide layer before removing │ │ │ │ │ - if (this.visibility && this.mapObject) { │ │ │ │ │ - this.setGMapVisibility(false); │ │ │ │ │ + containsPoint: function(point) { │ │ │ │ │ + var approx = OpenLayers.Number.limitSigDigs; │ │ │ │ │ + var digs = 14; │ │ │ │ │ + var px = approx(point.x, digs); │ │ │ │ │ + var py = approx(point.y, digs); │ │ │ │ │ + │ │ │ │ │ + function getX(y, x1, y1, x2, y2) { │ │ │ │ │ + return (y - y2) * ((x2 - x1) / (y2 - y1)) + x2; │ │ │ │ │ } │ │ │ │ │ - // check to see if last Google layer in this map │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - if (cache.count <= 1) { │ │ │ │ │ - this.removeGMapElements(); │ │ │ │ │ - delete OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ - } else { │ │ │ │ │ - // decrement the layer count │ │ │ │ │ - --cache.count; │ │ │ │ │ + var numSeg = this.components.length - 1; │ │ │ │ │ + var start, end, x1, y1, x2, y2, cx, cy; │ │ │ │ │ + var crosses = 0; │ │ │ │ │ + for (var i = 0; i < numSeg; ++i) { │ │ │ │ │ + start = this.components[i]; │ │ │ │ │ + x1 = approx(start.x, digs); │ │ │ │ │ + y1 = approx(start.y, digs); │ │ │ │ │ + end = this.components[i + 1]; │ │ │ │ │ + x2 = approx(end.x, digs); │ │ │ │ │ + y2 = approx(end.y, digs); │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * The following conditions enforce five edge-crossing rules: │ │ │ │ │ + * 1. points coincident with edges are considered contained; │ │ │ │ │ + * 2. an upward edge includes its starting endpoint, and │ │ │ │ │ + * excludes its final endpoint; │ │ │ │ │ + * 3. a downward edge excludes its starting endpoint, and │ │ │ │ │ + * includes its final endpoint; │ │ │ │ │ + * 4. horizontal edges are excluded; and │ │ │ │ │ + * 5. the edge-ray intersection point must be strictly right │ │ │ │ │ + * of the point P. │ │ │ │ │ + */ │ │ │ │ │ + if (y1 == y2) { │ │ │ │ │ + // horizontal edge │ │ │ │ │ + if (py == y1) { │ │ │ │ │ + // point on horizontal line │ │ │ │ │ + if (x1 <= x2 && (px >= x1 && px <= x2) || // right or vert │ │ │ │ │ + x1 >= x2 && (px <= x1 && px >= x2)) { // left or vert │ │ │ │ │ + // point on edge │ │ │ │ │ + crosses = -1; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // ignore other horizontal edges │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + cx = approx(getX(py, x1, y1, x2, y2), digs); │ │ │ │ │ + if (cx == px) { │ │ │ │ │ + // point on line │ │ │ │ │ + if (y1 < y2 && (py >= y1 && py <= y2) || // upward │ │ │ │ │ + y1 > y2 && (py <= y1 && py >= y2)) { // downward │ │ │ │ │ + // point on edge │ │ │ │ │ + crosses = -1; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (cx <= px) { │ │ │ │ │ + // no crossing to the right │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + if (x1 != x2 && (cx < Math.min(x1, x2) || cx > Math.max(x1, x2))) { │ │ │ │ │ + // no crossing │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + if (y1 < y2 && (py >= y1 && py < y2) || // upward │ │ │ │ │ + y1 > y2 && (py < y1 && py >= y2)) { // downward │ │ │ │ │ + ++crosses; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - // remove references to gmap elements │ │ │ │ │ - delete this.termsOfUse; │ │ │ │ │ - delete this.poweredBy; │ │ │ │ │ - delete this.mapObject; │ │ │ │ │ - delete this.dragObject; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.removeMap.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + var contained = (crosses == -1) ? │ │ │ │ │ + // on edge │ │ │ │ │ + 1 : │ │ │ │ │ + // even (out) or odd (in) │ │ │ │ │ + !!(crosses & 1); │ │ │ │ │ │ │ │ │ │ - // │ │ │ │ │ - // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ - // │ │ │ │ │ + return contained; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getOLBoundsFromMapObjectBounds │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: intersects │ │ │ │ │ + * Determine if the input geometry intersects this one. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * moBounds - {Object} │ │ │ │ │ - * │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} Any type of geometry. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} An <OpenLayers.Bounds>, translated from the │ │ │ │ │ - * passed-in MapObject Bounds. │ │ │ │ │ - * Returns null if null value is passed in. │ │ │ │ │ + * {Boolean} The input geometry intersects this one. │ │ │ │ │ */ │ │ │ │ │ - getOLBoundsFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - var olBounds = null; │ │ │ │ │ - if (moBounds != null) { │ │ │ │ │ - var sw = moBounds.getSouthWest(); │ │ │ │ │ - var ne = moBounds.getNorthEast(); │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - sw = this.forwardMercator(sw.lng(), sw.lat()); │ │ │ │ │ - ne = this.forwardMercator(ne.lng(), ne.lat()); │ │ │ │ │ - } else { │ │ │ │ │ - sw = new OpenLayers.LonLat(sw.lng(), sw.lat()); │ │ │ │ │ - ne = new OpenLayers.LonLat(ne.lng(), ne.lat()); │ │ │ │ │ + intersects: function(geometry) { │ │ │ │ │ + var intersect = false; │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + intersect = this.containsPoint(geometry); │ │ │ │ │ + } else if (geometry.CLASS_NAME == "OpenLayers.Geometry.LineString") { │ │ │ │ │ + intersect = geometry.intersects(this); │ │ │ │ │ + } else if (geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ + intersect = OpenLayers.Geometry.LineString.prototype.intersects.apply( │ │ │ │ │ + this, [geometry] │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + // check for component intersections │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; ++i) { │ │ │ │ │ + intersect = geometry.components[i].intersects(this); │ │ │ │ │ + if (intersect) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - olBounds = new OpenLayers.Bounds(sw.lon, │ │ │ │ │ - sw.lat, │ │ │ │ │ - ne.lon, │ │ │ │ │ - ne.lat); │ │ │ │ │ } │ │ │ │ │ - return olBounds; │ │ │ │ │ + return intersect; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getWarningHTML │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} String with information on why layer is broken, how to get │ │ │ │ │ - * it working. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getVertices │ │ │ │ │ + * Return a list of all points in this geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * nodes - {Boolean} For lines, only return vertices that are │ │ │ │ │ + * endpoints. If false, for lines, only vertices that are not │ │ │ │ │ + * endpoints will be returned. If not provided, all vertices will │ │ │ │ │ + * be returned. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} A list of all vertices in the geometry. │ │ │ │ │ */ │ │ │ │ │ - getWarningHTML: function() { │ │ │ │ │ - return OpenLayers.i18n("googleWarning"); │ │ │ │ │ + getVertices: function(nodes) { │ │ │ │ │ + return (nodes === true) ? [] : this.components.slice(0, this.components.length - 1); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Geometry.LinearRing" │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Geometry/Polygon.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Interface Controls * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Geometry/Collection.js │ │ │ │ │ + * @requires OpenLayers/Geometry/LinearRing.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // Get&Set Center, Zoom │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Geometry.Polygon │ │ │ │ │ + * Polygon is a collection of Geometry.LinearRings. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Geometry.Collection> │ │ │ │ │ + * - <OpenLayers.Geometry> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Geometry.Polygon = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Geometry.Collection, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectCenter │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The mapObject's current center in Map Object format │ │ │ │ │ + * Property: componentTypes │ │ │ │ │ + * {Array(String)} An array of class names representing the types of │ │ │ │ │ + * components that the collection can include. A null value means the │ │ │ │ │ + * component types are not restricted. │ │ │ │ │ + */ │ │ │ │ │ + componentTypes: ["OpenLayers.Geometry.LinearRing"], │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Geometry.Polygon │ │ │ │ │ + * Constructor for a Polygon geometry. │ │ │ │ │ + * The first ring (this.component[0])is the outer bounds of the polygon and │ │ │ │ │ + * all subsequent rings (this.component[1-n]) are internal holes. │ │ │ │ │ + * │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * components - {Array(<OpenLayers.Geometry.LinearRing>)} │ │ │ │ │ */ │ │ │ │ │ - getMapObjectCenter: function() { │ │ │ │ │ - return this.mapObject.getCenter(); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectZoom │ │ │ │ │ + * APIMethod: getArea │ │ │ │ │ + * Calculated by subtracting the areas of the internal holes from the │ │ │ │ │ + * area of the outer hole. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Integer} The mapObject's current zoom, in Map Object format │ │ │ │ │ + * {float} The area of the geometry │ │ │ │ │ */ │ │ │ │ │ - getMapObjectZoom: function() { │ │ │ │ │ - return this.mapObject.getZoom(); │ │ │ │ │ + getArea: function() { │ │ │ │ │ + var area = 0.0; │ │ │ │ │ + if (this.components && (this.components.length > 0)) { │ │ │ │ │ + area += Math.abs(this.components[0].getArea()); │ │ │ │ │ + for (var i = 1, len = this.components.length; i < len; i++) { │ │ │ │ │ + area -= Math.abs(this.components[i].getArea()); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return area; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Primitives * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - // LonLat │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getLongitudeFromMapObjectLonLat │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getGeodesicArea │ │ │ │ │ + * Calculate the approximate area of the polygon were it projected onto │ │ │ │ │ + * the earth. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ + * projection - {<OpenLayers.Projection>} The spatial reference system │ │ │ │ │ + * for the geometry coordinates. If not provided, Geographic/WGS84 is │ │ │ │ │ + * assumed. │ │ │ │ │ * │ │ │ │ │ + * Reference: │ │ │ │ │ + * Robert. G. Chamberlain and William H. Duquette, "Some Algorithms for │ │ │ │ │ + * Polygons on a Sphere", JPL Publication 07-03, Jet Propulsion │ │ │ │ │ + * Laboratory, Pasadena, CA, June 2007 http://trs-new.jpl.nasa.gov/dspace/handle/2014/40409 │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Float} Longitude of the given MapObject LonLat │ │ │ │ │ + * {float} The approximate geodesic area of the polygon in square meters. │ │ │ │ │ */ │ │ │ │ │ - getLongitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - return this.sphericalMercator ? │ │ │ │ │ - this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lon : │ │ │ │ │ - moLonLat.lng(); │ │ │ │ │ + getGeodesicArea: function(projection) { │ │ │ │ │ + var area = 0.0; │ │ │ │ │ + if (this.components && (this.components.length > 0)) { │ │ │ │ │ + area += Math.abs(this.components[0].getGeodesicArea(projection)); │ │ │ │ │ + for (var i = 1, len = this.components.length; i < len; i++) { │ │ │ │ │ + area -= Math.abs(this.components[i].getGeodesicArea(projection)); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return area; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getLatitudeFromMapObjectLonLat │ │ │ │ │ - * │ │ │ │ │ + * Method: containsPoint │ │ │ │ │ + * Test if a point is inside a polygon. Points on a polygon edge are │ │ │ │ │ + * considered inside. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ - * │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Float} Latitude of the given MapObject LonLat │ │ │ │ │ + * {Boolean | Number} The point is inside the polygon. Returns 1 if the │ │ │ │ │ + * point is on an edge. Returns boolean otherwise. │ │ │ │ │ */ │ │ │ │ │ - getLatitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - var lat = this.sphericalMercator ? │ │ │ │ │ - this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lat : │ │ │ │ │ - moLonLat.lat(); │ │ │ │ │ - return lat; │ │ │ │ │ + containsPoint: function(point) { │ │ │ │ │ + var numRings = this.components.length; │ │ │ │ │ + var contained = false; │ │ │ │ │ + if (numRings > 0) { │ │ │ │ │ + // check exterior ring - 1 means on edge, boolean otherwise │ │ │ │ │ + contained = this.components[0].containsPoint(point); │ │ │ │ │ + if (contained !== 1) { │ │ │ │ │ + if (contained && numRings > 1) { │ │ │ │ │ + // check interior rings │ │ │ │ │ + var hole; │ │ │ │ │ + for (var i = 1; i < numRings; ++i) { │ │ │ │ │ + hole = this.components[i].containsPoint(point); │ │ │ │ │ + if (hole) { │ │ │ │ │ + if (hole === 1) { │ │ │ │ │ + // on edge │ │ │ │ │ + contained = 1; │ │ │ │ │ + } else { │ │ │ │ │ + // in hole │ │ │ │ │ + contained = false; │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return contained; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - // Pixel │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getXFromMapObjectPixel │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: intersects │ │ │ │ │ + * Determine if the input geometry intersects this one. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * moPixel - {Object} MapObject Pixel format │ │ │ │ │ - * │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} Any type of geometry. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Integer} X value of the MapObject Pixel │ │ │ │ │ + * {Boolean} The input geometry intersects this one. │ │ │ │ │ */ │ │ │ │ │ - getXFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return moPixel.x; │ │ │ │ │ + intersects: function(geometry) { │ │ │ │ │ + var intersect = false; │ │ │ │ │ + var i, len; │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + intersect = this.containsPoint(geometry); │ │ │ │ │ + } else if (geometry.CLASS_NAME == "OpenLayers.Geometry.LineString" || │ │ │ │ │ + geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ + // check if rings/linestrings intersect │ │ │ │ │ + for (i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ + intersect = geometry.intersects(this.components[i]); │ │ │ │ │ + if (intersect) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!intersect) { │ │ │ │ │ + // check if this poly contains points of the ring/linestring │ │ │ │ │ + for (i = 0, len = geometry.components.length; i < len; ++i) { │ │ │ │ │ + intersect = this.containsPoint(geometry.components[i]); │ │ │ │ │ + if (intersect) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + for (i = 0, len = geometry.components.length; i < len; ++i) { │ │ │ │ │ + intersect = this.intersects(geometry.components[i]); │ │ │ │ │ + if (intersect) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // check case where this poly is wholly contained by another │ │ │ │ │ + if (!intersect && geometry.CLASS_NAME == "OpenLayers.Geometry.Polygon") { │ │ │ │ │ + // exterior ring points will be contained in the other geometry │ │ │ │ │ + var ring = this.components[0]; │ │ │ │ │ + for (i = 0, len = ring.components.length; i < len; ++i) { │ │ │ │ │ + intersect = geometry.containsPoint(ring.components[i]); │ │ │ │ │ + if (intersect) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return intersect; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getYFromMapObjectPixel │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: distanceTo │ │ │ │ │ + * Calculate the closest distance between two geometries (on the x-y plane). │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * moPixel - {Object} MapObject Pixel format │ │ │ │ │ - * │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ + * options - {Object} Optional properties for configuring the distance │ │ │ │ │ + * calculation. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * details - {Boolean} Return details from the distance calculation. │ │ │ │ │ + * Default is false. │ │ │ │ │ + * edge - {Boolean} Calculate the distance from this geometry to the │ │ │ │ │ + * nearest edge of the target geometry. Default is true. If true, │ │ │ │ │ + * calling distanceTo from a geometry that is wholly contained within │ │ │ │ │ + * the target will result in a non-zero distance. If false, whenever │ │ │ │ │ + * geometries intersect, calling distanceTo will return 0. If false, │ │ │ │ │ + * details cannot be returned. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Integer} Y value of the MapObject Pixel │ │ │ │ │ + * {Number | Object} The distance between this geometry and the target. │ │ │ │ │ + * If details is true, the return will be an object with distance, │ │ │ │ │ + * x0, y0, x1, and y1 properties. The x0 and y0 properties represent │ │ │ │ │ + * the coordinates of the closest point on this geometry. The x1 and y1 │ │ │ │ │ + * properties represent the coordinates of the closest point on the │ │ │ │ │ + * target geometry. │ │ │ │ │ */ │ │ │ │ │ - getYFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return moPixel.y; │ │ │ │ │ + distanceTo: function(geometry, options) { │ │ │ │ │ + var edge = !(options && options.edge === false); │ │ │ │ │ + var result; │ │ │ │ │ + // this is the case where we might not be looking for distance to edge │ │ │ │ │ + if (!edge && this.intersects(geometry)) { │ │ │ │ │ + result = 0; │ │ │ │ │ + } else { │ │ │ │ │ + result = OpenLayers.Geometry.Collection.prototype.distanceTo.apply( │ │ │ │ │ + this, [geometry, options] │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return result; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Google" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Geometry.Polygon" │ │ │ │ │ }); │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: OpenLayers.Layer.Google.cache │ │ │ │ │ - * {Object} Cache for elements that should only be created once per map. │ │ │ │ │ + * APIMethod: createRegularPolygon │ │ │ │ │ + * Create a regular polygon around a radius. Useful for creating circles │ │ │ │ │ + * and the like. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * origin - {<OpenLayers.Geometry.Point>} center of polygon. │ │ │ │ │ + * radius - {Float} distance to vertex, in map units. │ │ │ │ │ + * sides - {Integer} Number of sides. 20 approximates a circle. │ │ │ │ │ + * rotation - {Float} original angle of rotation, in degrees. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Google.cache = {}; │ │ │ │ │ +OpenLayers.Geometry.Polygon.createRegularPolygon = function(origin, radius, sides, rotation) { │ │ │ │ │ + var angle = Math.PI * ((1 / sides) - (1 / 2)); │ │ │ │ │ + if (rotation) { │ │ │ │ │ + angle += (rotation / 180) * Math.PI; │ │ │ │ │ + } │ │ │ │ │ + var rotatedAngle, x, y; │ │ │ │ │ + var points = []; │ │ │ │ │ + for (var i = 0; i < sides; ++i) { │ │ │ │ │ + rotatedAngle = angle + (i * 2 * Math.PI / sides); │ │ │ │ │ + x = origin.x + (radius * Math.cos(rotatedAngle)); │ │ │ │ │ + y = origin.y + (radius * Math.sin(rotatedAngle)); │ │ │ │ │ + points.push(new OpenLayers.Geometry.Point(x, y)); │ │ │ │ │ + } │ │ │ │ │ + var ring = new OpenLayers.Geometry.LinearRing(points); │ │ │ │ │ + return new OpenLayers.Geometry.Polygon([ring]); │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Geometry/MultiPolygon.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: OpenLayers.Layer.Google.v2 │ │ │ │ │ - * │ │ │ │ │ - * Mixin providing functionality specific to the Google Maps API v2. │ │ │ │ │ - * │ │ │ │ │ - * This API has been deprecated by Google. │ │ │ │ │ - * Developers are encouraged to migrate to v3 of the API; support for this │ │ │ │ │ - * is provided by <OpenLayers.Layer.Google.v3> │ │ │ │ │ + * @requires OpenLayers/Geometry/Collection.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Google.v2 = { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: termsOfUse │ │ │ │ │ - * {DOMElement} Div for Google's copyright and terms of use link │ │ │ │ │ - */ │ │ │ │ │ - termsOfUse: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: poweredBy │ │ │ │ │ - * {DOMElement} Div for Google's powered by logo and link │ │ │ │ │ - */ │ │ │ │ │ - poweredBy: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: dragObject │ │ │ │ │ - * {GDraggableObject} Since 2.93, Google has exposed the ability to get │ │ │ │ │ - * the maps GDraggableObject. We can now use this for smooth panning │ │ │ │ │ - */ │ │ │ │ │ - dragObject: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: loadMapObject │ │ │ │ │ - * Load the GMap and register appropriate event listeners. If we can't │ │ │ │ │ - * load GMap2, then display a warning message. │ │ │ │ │ - */ │ │ │ │ │ - loadMapObject: function() { │ │ │ │ │ - if (!this.type) { │ │ │ │ │ - this.type = G_NORMAL_MAP; │ │ │ │ │ - } │ │ │ │ │ - var mapObject, termsOfUse, poweredBy; │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - // there are already Google layers added to this map │ │ │ │ │ - mapObject = cache.mapObject; │ │ │ │ │ - termsOfUse = cache.termsOfUse; │ │ │ │ │ - poweredBy = cache.poweredBy; │ │ │ │ │ - // increment the layer count │ │ │ │ │ - ++cache.count; │ │ │ │ │ - } else { │ │ │ │ │ - // this is the first Google layer for this map │ │ │ │ │ - │ │ │ │ │ - var container = this.map.viewPortDiv; │ │ │ │ │ - var div = document.createElement("div"); │ │ │ │ │ - div.id = this.map.id + "_GMap2Container"; │ │ │ │ │ - div.style.position = "absolute"; │ │ │ │ │ - div.style.width = "100%"; │ │ │ │ │ - div.style.height = "100%"; │ │ │ │ │ - container.appendChild(div); │ │ │ │ │ - │ │ │ │ │ - // create GMap and shuffle elements │ │ │ │ │ - try { │ │ │ │ │ - mapObject = new GMap2(div); │ │ │ │ │ - │ │ │ │ │ - // move the ToS and branding stuff up to the container div │ │ │ │ │ - termsOfUse = div.lastChild; │ │ │ │ │ - container.appendChild(termsOfUse); │ │ │ │ │ - termsOfUse.style.zIndex = "1100"; │ │ │ │ │ - termsOfUse.style.right = ""; │ │ │ │ │ - termsOfUse.style.bottom = ""; │ │ │ │ │ - termsOfUse.className = "olLayerGoogleCopyright"; │ │ │ │ │ │ │ │ │ │ - poweredBy = div.lastChild; │ │ │ │ │ - container.appendChild(poweredBy); │ │ │ │ │ - poweredBy.style.zIndex = "1100"; │ │ │ │ │ - poweredBy.style.right = ""; │ │ │ │ │ - poweredBy.style.bottom = ""; │ │ │ │ │ - poweredBy.className = "olLayerGooglePoweredBy gmnoprint"; │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Geometry.MultiPolygon │ │ │ │ │ + * MultiPolygon is a geometry with multiple <OpenLayers.Geometry.Polygon> │ │ │ │ │ + * components. Create a new instance with the <OpenLayers.Geometry.MultiPolygon> │ │ │ │ │ + * constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Geometry.Collection> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Geometry.MultiPolygon = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Geometry.Collection, { │ │ │ │ │ │ │ │ │ │ - } catch (e) { │ │ │ │ │ - throw (e); │ │ │ │ │ - } │ │ │ │ │ - // cache elements for use by any other google layers added to │ │ │ │ │ - // this same map │ │ │ │ │ - OpenLayers.Layer.Google.cache[this.map.id] = { │ │ │ │ │ - mapObject: mapObject, │ │ │ │ │ - termsOfUse: termsOfUse, │ │ │ │ │ - poweredBy: poweredBy, │ │ │ │ │ - count: 1 │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: componentTypes │ │ │ │ │ + * {Array(String)} An array of class names representing the types of │ │ │ │ │ + * components that the collection can include. A null value means the │ │ │ │ │ + * component types are not restricted. │ │ │ │ │ + */ │ │ │ │ │ + componentTypes: ["OpenLayers.Geometry.Polygon"], │ │ │ │ │ │ │ │ │ │ - this.mapObject = mapObject; │ │ │ │ │ - this.termsOfUse = termsOfUse; │ │ │ │ │ - this.poweredBy = poweredBy; │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Geometry.MultiPolygon │ │ │ │ │ + * Create a new MultiPolygon geometry │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * components - {Array(<OpenLayers.Geometry.Polygon>)} An array of polygons │ │ │ │ │ + * used to generate the MultiPolygon │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // ensure this layer type is one of the mapObject types │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.mapObject.getMapTypes(), │ │ │ │ │ - this.type) === -1) { │ │ │ │ │ - this.mapObject.addMapType(this.type); │ │ │ │ │ - } │ │ │ │ │ + CLASS_NAME: "OpenLayers.Geometry.MultiPolygon" │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/GeoJSON.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - //since v 2.93 getDragObject is now available. │ │ │ │ │ - if (typeof mapObject.getDragObject == "function") { │ │ │ │ │ - this.dragObject = mapObject.getDragObject(); │ │ │ │ │ - } else { │ │ │ │ │ - this.dragPanMapObject = null; │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - if (this.isBaseLayer === false) { │ │ │ │ │ - this.setGMapVisibility(this.div.style.display !== "none"); │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/JSON.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Geometry/MultiPoint.js │ │ │ │ │ + * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ + * @requires OpenLayers/Geometry/MultiLineString.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ + * @requires OpenLayers/Geometry/MultiPolygon.js │ │ │ │ │ + * @requires OpenLayers/Console.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.GeoJSON │ │ │ │ │ + * Read and write GeoJSON. Create a new parser with the │ │ │ │ │ + * <OpenLayers.Format.GeoJSON> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.JSON> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.GeoJSON = OpenLayers.Class(OpenLayers.Format.JSON, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: onMapResize │ │ │ │ │ + * APIProperty: ignoreExtraDims │ │ │ │ │ + * {Boolean} Ignore dimensions higher than 2 when reading geometry │ │ │ │ │ + * coordinates. │ │ │ │ │ */ │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - // workaround for resizing of invisible or not yet fully loaded layers │ │ │ │ │ - // where GMap2.checkResize() does not work. We need to load the GMap │ │ │ │ │ - // for the old div size, then checkResize(), and then call │ │ │ │ │ - // layer.moveTo() to trigger GMap.setCenter() (which will finish │ │ │ │ │ - // the GMap initialization). │ │ │ │ │ - if (this.visibility && this.mapObject.isLoaded()) { │ │ │ │ │ - this.mapObject.checkResize(); │ │ │ │ │ - } else { │ │ │ │ │ - if (!this._resized) { │ │ │ │ │ - var layer = this; │ │ │ │ │ - var handle = GEvent.addListener(this.mapObject, "load", function() { │ │ │ │ │ - GEvent.removeListener(handle); │ │ │ │ │ - delete layer._resized; │ │ │ │ │ - layer.mapObject.checkResize(); │ │ │ │ │ - layer.moveTo(layer.map.getCenter(), layer.map.getZoom()); │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - this._resized = true; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + ignoreExtraDims: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setGMapVisibility │ │ │ │ │ - * Display the GMap container and associated elements. │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Format.GeoJSON │ │ │ │ │ + * Create a new parser for GeoJSON. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * visible - {Boolean} Display the GMap elements. │ │ │ │ │ - */ │ │ │ │ │ - setGMapVisibility: function(visible) { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - var container = this.mapObject.getContainer(); │ │ │ │ │ - if (visible === true) { │ │ │ │ │ - this.mapObject.setMapType(this.type); │ │ │ │ │ - container.style.display = ""; │ │ │ │ │ - this.termsOfUse.style.left = ""; │ │ │ │ │ - this.termsOfUse.style.display = ""; │ │ │ │ │ - this.poweredBy.style.display = ""; │ │ │ │ │ - cache.displayed = this.id; │ │ │ │ │ - } else { │ │ │ │ │ - if (cache.displayed === this.id) { │ │ │ │ │ - delete cache.displayed; │ │ │ │ │ - } │ │ │ │ │ - if (!cache.displayed) { │ │ │ │ │ - container.style.display = "none"; │ │ │ │ │ - this.termsOfUse.style.display = "none"; │ │ │ │ │ - // move ToU far to the left in addition to setting display │ │ │ │ │ - // to "none", because at the end of the GMap2 load │ │ │ │ │ - // sequence, display: none will be unset and ToU would be │ │ │ │ │ - // visible after loading a map with a google layer that is │ │ │ │ │ - // initially hidden. │ │ │ │ │ - this.termsOfUse.style.left = "-9999px"; │ │ │ │ │ - this.poweredBy.style.display = "none"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getMapContainer │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} the GMap container's div │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - getMapContainer: function() { │ │ │ │ │ - return this.mapObject.getContainer(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ - // │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectBoundsFromOLBounds │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Deserialize a GeoJSON string. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * olBounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} A MapObject Bounds, translated from olBounds │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ + * json - {String} A GeoJSON string │ │ │ │ │ + * type - {String} Optional string that determines the structure of │ │ │ │ │ + * the output. Supported values are "Geometry", "Feature", and │ │ │ │ │ + * "FeatureCollection". If absent or null, a default of │ │ │ │ │ + * "FeatureCollection" is assumed. │ │ │ │ │ + * filter - {Function} A function which will be called for every key and │ │ │ │ │ + * value at every level of the final result. Each value will be │ │ │ │ │ + * replaced by the result of the filter function. This can be used to │ │ │ │ │ + * reform generic objects into instances of classes, or to transform │ │ │ │ │ + * date strings into Date objects. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The return depends on the value of the type argument. If type │ │ │ │ │ + * is "FeatureCollection" (the default), the return will be an array │ │ │ │ │ + * of <OpenLayers.Feature.Vector>. If type is "Geometry", the input json │ │ │ │ │ + * must represent a single geometry, and the return will be an │ │ │ │ │ + * <OpenLayers.Geometry>. If type is "Feature", the input json must │ │ │ │ │ + * represent a single feature, and the return will be an │ │ │ │ │ + * <OpenLayers.Feature.Vector>. │ │ │ │ │ */ │ │ │ │ │ - getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ - var moBounds = null; │ │ │ │ │ - if (olBounds != null) { │ │ │ │ │ - var sw = this.sphericalMercator ? │ │ │ │ │ - this.inverseMercator(olBounds.bottom, olBounds.left) : │ │ │ │ │ - new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ - var ne = this.sphericalMercator ? │ │ │ │ │ - this.inverseMercator(olBounds.top, olBounds.right) : │ │ │ │ │ - new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ - moBounds = new GLatLngBounds(new GLatLng(sw.lat, sw.lon), │ │ │ │ │ - new GLatLng(ne.lat, ne.lon)); │ │ │ │ │ + read: function(json, type, filter) { │ │ │ │ │ + type = (type) ? type : "FeatureCollection"; │ │ │ │ │ + var results = null; │ │ │ │ │ + var obj = null; │ │ │ │ │ + if (typeof json == "string") { │ │ │ │ │ + obj = OpenLayers.Format.JSON.prototype.read.apply(this, │ │ │ │ │ + [json, filter]); │ │ │ │ │ + } else { │ │ │ │ │ + obj = json; │ │ │ │ │ } │ │ │ │ │ - return moBounds; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Interface Controls * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - // Get&Set Center, Zoom │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setMapObjectCenter │ │ │ │ │ - * Set the mapObject to the specified center and zoom │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * center - {Object} MapObject LonLat format │ │ │ │ │ - * zoom - {int} MapObject zoom format │ │ │ │ │ - */ │ │ │ │ │ - setMapObjectCenter: function(center, zoom) { │ │ │ │ │ - this.mapObject.setCenter(center, zoom); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: dragPanMapObject │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * dX - {Integer} │ │ │ │ │ - * dY - {Integer} │ │ │ │ │ - */ │ │ │ │ │ - dragPanMapObject: function(dX, dY) { │ │ │ │ │ - this.dragObject.moveBy(new GSize(-dX, dY)); │ │ │ │ │ + if (!obj) { │ │ │ │ │ + OpenLayers.Console.error("Bad JSON: " + json); │ │ │ │ │ + } else if (typeof(obj.type) != "string") { │ │ │ │ │ + OpenLayers.Console.error("Bad GeoJSON - no type: " + json); │ │ │ │ │ + } else if (this.isValidType(obj, type)) { │ │ │ │ │ + switch (type) { │ │ │ │ │ + case "Geometry": │ │ │ │ │ + try { │ │ │ │ │ + results = this.parseGeometry(obj); │ │ │ │ │ + } catch (err) { │ │ │ │ │ + OpenLayers.Console.error(err); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "Feature": │ │ │ │ │ + try { │ │ │ │ │ + results = this.parseFeature(obj); │ │ │ │ │ + results.type = "Feature"; │ │ │ │ │ + } catch (err) { │ │ │ │ │ + OpenLayers.Console.error(err); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "FeatureCollection": │ │ │ │ │ + // for type FeatureCollection, we allow input to be any type │ │ │ │ │ + results = []; │ │ │ │ │ + switch (obj.type) { │ │ │ │ │ + case "Feature": │ │ │ │ │ + try { │ │ │ │ │ + results.push(this.parseFeature(obj)); │ │ │ │ │ + } catch (err) { │ │ │ │ │ + results = null; │ │ │ │ │ + OpenLayers.Console.error(err); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "FeatureCollection": │ │ │ │ │ + for (var i = 0, len = obj.features.length; i < len; ++i) { │ │ │ │ │ + try { │ │ │ │ │ + results.push(this.parseFeature(obj.features[i])); │ │ │ │ │ + } catch (err) { │ │ │ │ │ + results = null; │ │ │ │ │ + OpenLayers.Console.error(err); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + try { │ │ │ │ │ + var geom = this.parseGeometry(obj); │ │ │ │ │ + results.push(new OpenLayers.Feature.Vector(geom)); │ │ │ │ │ + } catch (err) { │ │ │ │ │ + results = null; │ │ │ │ │ + OpenLayers.Console.error(err); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return results; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ - // LonLat - Pixel Translation │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectLonLatFromMapObjectPixel │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moPixel - {Object} MapObject Pixel format │ │ │ │ │ - * │ │ │ │ │ + * Method: isValidType │ │ │ │ │ + * Check if a GeoJSON object is a valid representative of the given type. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} MapObject LonLat translated from MapObject Pixel │ │ │ │ │ + * {Boolean} The object is valid GeoJSON object of the given type. │ │ │ │ │ */ │ │ │ │ │ - getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return this.mapObject.fromContainerPixelToLatLng(moPixel); │ │ │ │ │ + isValidType: function(obj, type) { │ │ │ │ │ + var valid = false; │ │ │ │ │ + switch (type) { │ │ │ │ │ + case "Geometry": │ │ │ │ │ + if (OpenLayers.Util.indexOf( │ │ │ │ │ + ["Point", "MultiPoint", "LineString", "MultiLineString", │ │ │ │ │ + "Polygon", "MultiPolygon", "Box", "GeometryCollection" │ │ │ │ │ + ], │ │ │ │ │ + obj.type) == -1) { │ │ │ │ │ + // unsupported geometry type │ │ │ │ │ + OpenLayers.Console.error("Unsupported geometry type: " + │ │ │ │ │ + obj.type); │ │ │ │ │ + } else { │ │ │ │ │ + valid = true; │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "FeatureCollection": │ │ │ │ │ + // allow for any type to be converted to a feature collection │ │ │ │ │ + valid = true; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + // for Feature types must match │ │ │ │ │ + if (obj.type == type) { │ │ │ │ │ + valid = true; │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Console.error("Cannot convert types from " + │ │ │ │ │ + obj.type + " to " + type); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return valid; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectPixelFromMapObjectLonLat │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject Pixel transtlated from MapObject LonLat │ │ │ │ │ - */ │ │ │ │ │ - getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - return this.mapObject.fromLatLngToContainerPixel(moLonLat); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - // Bounds │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectZoomFromMapObjectBounds │ │ │ │ │ - * │ │ │ │ │ + * Method: parseFeature │ │ │ │ │ + * Convert a feature object from GeoJSON into an │ │ │ │ │ + * <OpenLayers.Feature.Vector>. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * moBounds - {Object} MapObject Bounds format │ │ │ │ │ - * │ │ │ │ │ + * obj - {Object} An object created from a GeoJSON object │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} MapObject Zoom for specified MapObject Bounds │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A feature. │ │ │ │ │ */ │ │ │ │ │ - getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - return this.mapObject.getBoundsZoomLevel(moBounds); │ │ │ │ │ + parseFeature: function(obj) { │ │ │ │ │ + var feature, geometry, attributes, bbox; │ │ │ │ │ + attributes = (obj.properties) ? obj.properties : {}; │ │ │ │ │ + bbox = (obj.geometry && obj.geometry.bbox) || obj.bbox; │ │ │ │ │ + try { │ │ │ │ │ + geometry = this.parseGeometry(obj.geometry); │ │ │ │ │ + } catch (err) { │ │ │ │ │ + // deal with bad geometries │ │ │ │ │ + throw err; │ │ │ │ │ + } │ │ │ │ │ + feature = new OpenLayers.Feature.Vector(geometry, attributes); │ │ │ │ │ + if (bbox) { │ │ │ │ │ + feature.bounds = OpenLayers.Bounds.fromArray(bbox); │ │ │ │ │ + } │ │ │ │ │ + if (obj.id) { │ │ │ │ │ + feature.fid = obj.id; │ │ │ │ │ + } │ │ │ │ │ + return feature; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Primitives * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - // LonLat │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectLonLatFromLonLat │ │ │ │ │ - * │ │ │ │ │ + * Method: parseGeometry │ │ │ │ │ + * Convert a geometry object from GeoJSON into an <OpenLayers.Geometry>. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * lon - {Float} │ │ │ │ │ - * lat - {Float} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject LonLat built from lon and lat params │ │ │ │ │ + * obj - {Object} An object created from a GeoJSON object │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry>} A geometry. │ │ │ │ │ */ │ │ │ │ │ - getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ - var gLatLng; │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ - gLatLng = new GLatLng(lonlat.lat, lonlat.lon); │ │ │ │ │ + parseGeometry: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + var geometry, collection = false; │ │ │ │ │ + if (obj.type == "GeometryCollection") { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(obj.geometries))) { │ │ │ │ │ + throw "GeometryCollection must have geometries array: " + obj; │ │ │ │ │ + } │ │ │ │ │ + var numGeom = obj.geometries.length; │ │ │ │ │ + var components = new Array(numGeom); │ │ │ │ │ + for (var i = 0; i < numGeom; ++i) { │ │ │ │ │ + components[i] = this.parseGeometry.apply( │ │ │ │ │ + this, [obj.geometries[i]] │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + geometry = new OpenLayers.Geometry.Collection(components); │ │ │ │ │ + collection = true; │ │ │ │ │ } else { │ │ │ │ │ - gLatLng = new GLatLng(lat, lon); │ │ │ │ │ + if (!(OpenLayers.Util.isArray(obj.coordinates))) { │ │ │ │ │ + throw "Geometry must have coordinates array: " + obj; │ │ │ │ │ + } │ │ │ │ │ + if (!this.parseCoords[obj.type.toLowerCase()]) { │ │ │ │ │ + throw "Unsupported geometry type: " + obj.type; │ │ │ │ │ + } │ │ │ │ │ + try { │ │ │ │ │ + geometry = this.parseCoords[obj.type.toLowerCase()].apply( │ │ │ │ │ + this, [obj.coordinates] │ │ │ │ │ + ); │ │ │ │ │ + } catch (err) { │ │ │ │ │ + // deal with bad coordinates │ │ │ │ │ + throw err; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return gLatLng; │ │ │ │ │ + // We don't reproject collections because the children are reprojected │ │ │ │ │ + // for us when they are created. │ │ │ │ │ + if (this.internalProjection && this.externalProjection && !collection) { │ │ │ │ │ + geometry.transform(this.externalProjection, │ │ │ │ │ + this.internalProjection); │ │ │ │ │ + } │ │ │ │ │ + return geometry; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - // Pixel │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectPixelFromXY │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {Integer} │ │ │ │ │ - * y - {Integer} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject Pixel from x and y parameters │ │ │ │ │ + * Property: parseCoords │ │ │ │ │ + * Object with properties corresponding to the GeoJSON geometry types. │ │ │ │ │ + * Property values are functions that do the actual parsing. │ │ │ │ │ */ │ │ │ │ │ - getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ - return new GPoint(x, y); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Google/v3.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ + parseCoords: { │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseCoords.point │ │ │ │ │ + * Convert a coordinate array from GeoJSON into an │ │ │ │ │ + * <OpenLayers.Geometry>. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * array - {Object} The coordinates array from the GeoJSON fragment. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry>} A geometry. │ │ │ │ │ + */ │ │ │ │ │ + "point": function(array) { │ │ │ │ │ + if (this.ignoreExtraDims == false && │ │ │ │ │ + array.length != 2) { │ │ │ │ │ + throw "Only 2D points are supported: " + array; │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.Point(array[0], array[1]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/Google.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseCoords.multipoint │ │ │ │ │ + * Convert a coordinate array from GeoJSON into an │ │ │ │ │ + * <OpenLayers.Geometry>. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * array - {Object} The coordinates array from the GeoJSON fragment. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry>} A geometry. │ │ │ │ │ + */ │ │ │ │ │ + "multipoint": function(array) { │ │ │ │ │ + var points = []; │ │ │ │ │ + var p = null; │ │ │ │ │ + for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ + try { │ │ │ │ │ + p = this.parseCoords["point"].apply(this, [array[i]]); │ │ │ │ │ + } catch (err) { │ │ │ │ │ + throw err; │ │ │ │ │ + } │ │ │ │ │ + points.push(p); │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.MultiPoint(points); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Layer.Google.v3 │ │ │ │ │ - * │ │ │ │ │ - * Mixin providing functionality specific to the Google Maps API v3. │ │ │ │ │ - * │ │ │ │ │ - * To use this layer, you must include the GMaps v3 API in your html. │ │ │ │ │ - * │ │ │ │ │ - * Note that this layer configures the google.maps.map object with the │ │ │ │ │ - * "disableDefaultUI" option set to true. Using UI controls that the Google │ │ │ │ │ - * Maps API provides is not supported by the OpenLayers API. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.Google.v3 = { │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseCoords.linestring │ │ │ │ │ + * Convert a coordinate array from GeoJSON into an │ │ │ │ │ + * <OpenLayers.Geometry>. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * array - {Object} The coordinates array from the GeoJSON fragment. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry>} A geometry. │ │ │ │ │ + */ │ │ │ │ │ + "linestring": function(array) { │ │ │ │ │ + var points = []; │ │ │ │ │ + var p = null; │ │ │ │ │ + for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ + try { │ │ │ │ │ + p = this.parseCoords["point"].apply(this, [array[i]]); │ │ │ │ │ + } catch (err) { │ │ │ │ │ + throw err; │ │ │ │ │ + } │ │ │ │ │ + points.push(p); │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.LineString(points); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: DEFAULTS │ │ │ │ │ - * {Object} It is not recommended to change the properties set here. Note │ │ │ │ │ - * that Google.v3 layers only work when sphericalMercator is set to true. │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * { │ │ │ │ │ - * sphericalMercator: true, │ │ │ │ │ - * projection: "EPSG:900913" │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ - DEFAULTS: { │ │ │ │ │ - sphericalMercator: true, │ │ │ │ │ - projection: "EPSG:900913" │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseCoords.multilinestring │ │ │ │ │ + * Convert a coordinate array from GeoJSON into an │ │ │ │ │ + * <OpenLayers.Geometry>. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * array - {Object} The coordinates array from the GeoJSON fragment. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry>} A geometry. │ │ │ │ │ + */ │ │ │ │ │ + "multilinestring": function(array) { │ │ │ │ │ + var lines = []; │ │ │ │ │ + var l = null; │ │ │ │ │ + for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ + try { │ │ │ │ │ + l = this.parseCoords["linestring"].apply(this, [array[i]]); │ │ │ │ │ + } catch (err) { │ │ │ │ │ + throw err; │ │ │ │ │ + } │ │ │ │ │ + lines.push(l); │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.MultiLineString(lines); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: animationEnabled │ │ │ │ │ - * {Boolean} If set to true, the transition between zoom levels will be │ │ │ │ │ - * animated (if supported by the GMaps API for the device used). Set to │ │ │ │ │ - * false to match the zooming experience of other layer types. Default │ │ │ │ │ - * is true. Note that the GMaps API does not give us control over zoom │ │ │ │ │ - * animation, so if set to false, when zooming, this will make the │ │ │ │ │ - * layer temporarily invisible, wait until GMaps reports the map being │ │ │ │ │ - * idle, and make it visible again. The result will be a blank layer │ │ │ │ │ - * for a few moments while zooming. │ │ │ │ │ - */ │ │ │ │ │ - animationEnabled: true, │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseCoords.polygon │ │ │ │ │ + * Convert a coordinate array from GeoJSON into an │ │ │ │ │ + * <OpenLayers.Geometry>. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry>} A geometry. │ │ │ │ │ + */ │ │ │ │ │ + "polygon": function(array) { │ │ │ │ │ + var rings = []; │ │ │ │ │ + var r, l; │ │ │ │ │ + for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ + try { │ │ │ │ │ + l = this.parseCoords["linestring"].apply(this, [array[i]]); │ │ │ │ │ + } catch (err) { │ │ │ │ │ + throw err; │ │ │ │ │ + } │ │ │ │ │ + r = new OpenLayers.Geometry.LinearRing(l.components); │ │ │ │ │ + rings.push(r); │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.Polygon(rings); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: loadMapObject │ │ │ │ │ - * Load the GMap and register appropriate event listeners. │ │ │ │ │ - */ │ │ │ │ │ - loadMapObject: function() { │ │ │ │ │ - if (!this.type) { │ │ │ │ │ - this.type = google.maps.MapTypeId.ROADMAP; │ │ │ │ │ - } │ │ │ │ │ - var mapObject; │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - // there are already Google layers added to this map │ │ │ │ │ - mapObject = cache.mapObject; │ │ │ │ │ - // increment the layer count │ │ │ │ │ - ++cache.count; │ │ │ │ │ - } else { │ │ │ │ │ - // this is the first Google layer for this map │ │ │ │ │ - // create GMap │ │ │ │ │ - var center = this.map.getCenter(); │ │ │ │ │ - var container = document.createElement('div'); │ │ │ │ │ - container.className = "olForeignContainer"; │ │ │ │ │ - container.style.width = '100%'; │ │ │ │ │ - container.style.height = '100%'; │ │ │ │ │ - mapObject = new google.maps.Map(container, { │ │ │ │ │ - center: center ? │ │ │ │ │ - new google.maps.LatLng(center.lat, center.lon) : new google.maps.LatLng(0, 0), │ │ │ │ │ - zoom: this.map.getZoom() || 0, │ │ │ │ │ - mapTypeId: this.type, │ │ │ │ │ - disableDefaultUI: true, │ │ │ │ │ - keyboardShortcuts: false, │ │ │ │ │ - draggable: false, │ │ │ │ │ - disableDoubleClickZoom: true, │ │ │ │ │ - scrollwheel: false, │ │ │ │ │ - streetViewControl: false │ │ │ │ │ - }); │ │ │ │ │ - var googleControl = document.createElement('div'); │ │ │ │ │ - googleControl.style.width = '100%'; │ │ │ │ │ - googleControl.style.height = '100%'; │ │ │ │ │ - mapObject.controls[google.maps.ControlPosition.TOP_LEFT].push(googleControl); │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseCoords.multipolygon │ │ │ │ │ + * Convert a coordinate array from GeoJSON into an │ │ │ │ │ + * <OpenLayers.Geometry>. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * array - {Object} The coordinates array from the GeoJSON fragment. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry>} A geometry. │ │ │ │ │ + */ │ │ │ │ │ + "multipolygon": function(array) { │ │ │ │ │ + var polys = []; │ │ │ │ │ + var p = null; │ │ │ │ │ + for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ + try { │ │ │ │ │ + p = this.parseCoords["polygon"].apply(this, [array[i]]); │ │ │ │ │ + } catch (err) { │ │ │ │ │ + throw err; │ │ │ │ │ + } │ │ │ │ │ + polys.push(p); │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.MultiPolygon(polys); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // cache elements for use by any other google layers added to │ │ │ │ │ - // this same map │ │ │ │ │ - cache = { │ │ │ │ │ - googleControl: googleControl, │ │ │ │ │ - mapObject: mapObject, │ │ │ │ │ - count: 1 │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Layer.Google.cache[this.map.id] = cache; │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseCoords.box │ │ │ │ │ + * Convert a coordinate array from GeoJSON into an │ │ │ │ │ + * <OpenLayers.Geometry>. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * array - {Object} The coordinates array from the GeoJSON fragment. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry>} A geometry. │ │ │ │ │ + */ │ │ │ │ │ + "box": function(array) { │ │ │ │ │ + if (array.length != 2) { │ │ │ │ │ + throw "GeoJSON box coordinates must have 2 elements"; │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.Polygon([ │ │ │ │ │ + new OpenLayers.Geometry.LinearRing([ │ │ │ │ │ + new OpenLayers.Geometry.Point(array[0][0], array[0][1]), │ │ │ │ │ + new OpenLayers.Geometry.Point(array[1][0], array[0][1]), │ │ │ │ │ + new OpenLayers.Geometry.Point(array[1][0], array[1][1]), │ │ │ │ │ + new OpenLayers.Geometry.Point(array[0][0], array[1][1]), │ │ │ │ │ + new OpenLayers.Geometry.Point(array[0][0], array[0][1]) │ │ │ │ │ + ]) │ │ │ │ │ + ]); │ │ │ │ │ } │ │ │ │ │ - this.mapObject = mapObject; │ │ │ │ │ - this.setGMapVisibility(this.visibility); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: onMapResize │ │ │ │ │ - */ │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - if (this.visibility) { │ │ │ │ │ - google.maps.event.trigger(this.mapObject, "resize"); │ │ │ │ │ - } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setGMapVisibility │ │ │ │ │ - * Display the GMap container and associated elements. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Serialize a feature, geometry, array of features into a GeoJSON string. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * visible - {Boolean} Display the GMap elements. │ │ │ │ │ + * obj - {Object} An <OpenLayers.Feature.Vector>, <OpenLayers.Geometry>, │ │ │ │ │ + * or an array of features. │ │ │ │ │ + * pretty - {Boolean} Structure the output with newlines and indentation. │ │ │ │ │ + * Default is false. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The GeoJSON string representation of the input geometry, │ │ │ │ │ + * features, or array of features. │ │ │ │ │ */ │ │ │ │ │ - setGMapVisibility: function(visible) { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - var map = this.map; │ │ │ │ │ - if (cache) { │ │ │ │ │ - var type = this.type; │ │ │ │ │ - var layers = map.layers; │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = layers[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Google && │ │ │ │ │ - layer.visibility === true && layer.inRange === true) { │ │ │ │ │ - type = layer.type; │ │ │ │ │ - visible = true; │ │ │ │ │ - break; │ │ │ │ │ + write: function(obj, pretty) { │ │ │ │ │ + var geojson = { │ │ │ │ │ + "type": null │ │ │ │ │ + }; │ │ │ │ │ + if (OpenLayers.Util.isArray(obj)) { │ │ │ │ │ + geojson.type = "FeatureCollection"; │ │ │ │ │ + var numFeatures = obj.length; │ │ │ │ │ + geojson.features = new Array(numFeatures); │ │ │ │ │ + for (var i = 0; i < numFeatures; ++i) { │ │ │ │ │ + var element = obj[i]; │ │ │ │ │ + if (!element instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ + var msg = "FeatureCollection only supports collections " + │ │ │ │ │ + "of features: " + element; │ │ │ │ │ + throw msg; │ │ │ │ │ } │ │ │ │ │ + geojson.features[i] = this.extract.feature.apply( │ │ │ │ │ + this, [element] │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - var container = this.mapObject.getDiv(); │ │ │ │ │ - if (visible === true) { │ │ │ │ │ - if (container.parentNode !== map.div) { │ │ │ │ │ - if (!cache.rendered) { │ │ │ │ │ - var me = this; │ │ │ │ │ - google.maps.event.addListenerOnce(this.mapObject, 'tilesloaded', function() { │ │ │ │ │ - cache.rendered = true; │ │ │ │ │ - me.setGMapVisibility(me.getVisibility()); │ │ │ │ │ - me.moveTo(me.map.getCenter()); │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - map.div.appendChild(container); │ │ │ │ │ - cache.googleControl.appendChild(map.viewPortDiv); │ │ │ │ │ - google.maps.event.trigger(this.mapObject, 'resize'); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.mapObject.setMapTypeId(type); │ │ │ │ │ - } else if (cache.googleControl.hasChildNodes()) { │ │ │ │ │ - map.div.appendChild(map.viewPortDiv); │ │ │ │ │ - map.div.removeChild(container); │ │ │ │ │ + } else if (obj.CLASS_NAME.indexOf("OpenLayers.Geometry") == 0) { │ │ │ │ │ + geojson = this.extract.geometry.apply(this, [obj]); │ │ │ │ │ + } else if (obj instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ + geojson = this.extract.feature.apply(this, [obj]); │ │ │ │ │ + if (obj.layer && obj.layer.projection) { │ │ │ │ │ + geojson.crs = this.createCRSObject(obj); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return OpenLayers.Format.JSON.prototype.write.apply(this, │ │ │ │ │ + [geojson, pretty]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getMapContainer │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} the GMap container's div │ │ │ │ │ - */ │ │ │ │ │ - getMapContainer: function() { │ │ │ │ │ - return this.mapObject.getDiv(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ - // │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectBoundsFromOLBounds │ │ │ │ │ - * │ │ │ │ │ + * Method: createCRSObject │ │ │ │ │ + * Create the CRS object for an object. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * olBounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * │ │ │ │ │ + * object - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} A MapObject Bounds, translated from olBounds │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ + * {Object} An object which can be assigned to the crs property │ │ │ │ │ + * of a GeoJSON object. │ │ │ │ │ */ │ │ │ │ │ - getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ - var moBounds = null; │ │ │ │ │ - if (olBounds != null) { │ │ │ │ │ - var sw = this.sphericalMercator ? │ │ │ │ │ - this.inverseMercator(olBounds.bottom, olBounds.left) : │ │ │ │ │ - new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ - var ne = this.sphericalMercator ? │ │ │ │ │ - this.inverseMercator(olBounds.top, olBounds.right) : │ │ │ │ │ - new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ - moBounds = new google.maps.LatLngBounds( │ │ │ │ │ - new google.maps.LatLng(sw.lat, sw.lon), │ │ │ │ │ - new google.maps.LatLng(ne.lat, ne.lon) │ │ │ │ │ - ); │ │ │ │ │ + createCRSObject: function(object) { │ │ │ │ │ + var proj = object.layer.projection.toString(); │ │ │ │ │ + var crs = {}; │ │ │ │ │ + if (proj.match(/epsg:/i)) { │ │ │ │ │ + var code = parseInt(proj.substring(proj.indexOf(":") + 1)); │ │ │ │ │ + if (code == 4326) { │ │ │ │ │ + crs = { │ │ │ │ │ + "type": "name", │ │ │ │ │ + "properties": { │ │ │ │ │ + "name": "urn:ogc:def:crs:OGC:1.3:CRS84" │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + } else { │ │ │ │ │ + crs = { │ │ │ │ │ + "type": "name", │ │ │ │ │ + "properties": { │ │ │ │ │ + "name": "EPSG:" + code │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return moBounds; │ │ │ │ │ + return crs; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Interface Controls * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - // LonLat - Pixel Translation │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectLonLatFromMapObjectPixel │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moPixel - {Object} MapObject Pixel format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject LonLat translated from MapObject Pixel │ │ │ │ │ + * Property: extract │ │ │ │ │ + * Object with properties corresponding to the GeoJSON types. │ │ │ │ │ + * Property values are functions that do the actual value extraction. │ │ │ │ │ */ │ │ │ │ │ - getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - var size = this.map.getSize(); │ │ │ │ │ - var lon = this.getLongitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ - var lat = this.getLatitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ - var res = this.map.getResolution(); │ │ │ │ │ - │ │ │ │ │ - var delta_x = moPixel.x - (size.w / 2); │ │ │ │ │ - var delta_y = moPixel.y - (size.h / 2); │ │ │ │ │ - │ │ │ │ │ - var lonlat = new OpenLayers.LonLat( │ │ │ │ │ - lon + delta_x * res, │ │ │ │ │ - lat - delta_y * res │ │ │ │ │ - ); │ │ │ │ │ + extract: { │ │ │ │ │ + /** │ │ │ │ │ + * Method: extract.feature │ │ │ │ │ + * Return a partial GeoJSON object representing a single feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object representing the point. │ │ │ │ │ + */ │ │ │ │ │ + 'feature': function(feature) { │ │ │ │ │ + var geom = this.extract.geometry.apply(this, [feature.geometry]); │ │ │ │ │ + var json = { │ │ │ │ │ + "type": "Feature", │ │ │ │ │ + "properties": feature.attributes, │ │ │ │ │ + "geometry": geom │ │ │ │ │ + }; │ │ │ │ │ + if (feature.fid != null) { │ │ │ │ │ + json.id = feature.fid; │ │ │ │ │ + } │ │ │ │ │ + return json; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - lonlat = lonlat.wrapDateLine(this.maxExtent); │ │ │ │ │ - } │ │ │ │ │ - return this.getMapObjectLonLatFromLonLat(lonlat.lon, lonlat.lat); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: extract.geometry │ │ │ │ │ + * Return a GeoJSON object representing a single geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object representing the geometry. │ │ │ │ │ + */ │ │ │ │ │ + 'geometry': function(geometry) { │ │ │ │ │ + if (geometry == null) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry = geometry.clone(); │ │ │ │ │ + geometry.transform(this.internalProjection, │ │ │ │ │ + this.externalProjection); │ │ │ │ │ + } │ │ │ │ │ + var geometryType = geometry.CLASS_NAME.split('.')[2]; │ │ │ │ │ + var data = this.extract[geometryType.toLowerCase()].apply(this, [geometry]); │ │ │ │ │ + var json; │ │ │ │ │ + if (geometryType == "Collection") { │ │ │ │ │ + json = { │ │ │ │ │ + "type": "GeometryCollection", │ │ │ │ │ + "geometries": data │ │ │ │ │ + }; │ │ │ │ │ + } else { │ │ │ │ │ + json = { │ │ │ │ │ + "type": geometryType, │ │ │ │ │ + "coordinates": data │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectPixelFromMapObjectLonLat │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject Pixel transtlated from MapObject LonLat │ │ │ │ │ - */ │ │ │ │ │ - getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - var res = this.map.getResolution(); │ │ │ │ │ - var extent = this.map.getExtent(); │ │ │ │ │ - return this.getMapObjectPixelFromXY((1 / res * (lon - extent.left)), │ │ │ │ │ - (1 / res * (extent.top - lat))); │ │ │ │ │ - }, │ │ │ │ │ + return json; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Method: extract.point │ │ │ │ │ + * Return an array of coordinates from a point. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} An array of coordinates representing the point. │ │ │ │ │ + */ │ │ │ │ │ + 'point': function(point) { │ │ │ │ │ + return [point.x, point.y]; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setMapObjectCenter │ │ │ │ │ - * Set the mapObject to the specified center and zoom │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * center - {Object} MapObject LonLat format │ │ │ │ │ - * zoom - {int} MapObject zoom format │ │ │ │ │ - */ │ │ │ │ │ - setMapObjectCenter: function(center, zoom) { │ │ │ │ │ - if (this.animationEnabled === false && zoom != this.mapObject.zoom) { │ │ │ │ │ - var mapContainer = this.getMapContainer(); │ │ │ │ │ - google.maps.event.addListenerOnce( │ │ │ │ │ - this.mapObject, │ │ │ │ │ - "idle", │ │ │ │ │ - function() { │ │ │ │ │ - mapContainer.style.visibility = ""; │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - mapContainer.style.visibility = "hidden"; │ │ │ │ │ - } │ │ │ │ │ - this.mapObject.setOptions({ │ │ │ │ │ - center: center, │ │ │ │ │ - zoom: zoom │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: extract.multipoint │ │ │ │ │ + * Return an array of point coordinates from a multipoint. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * multipoint - {<OpenLayers.Geometry.MultiPoint>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} An array of point coordinate arrays representing │ │ │ │ │ + * the multipoint. │ │ │ │ │ + */ │ │ │ │ │ + 'multipoint': function(multipoint) { │ │ │ │ │ + var array = []; │ │ │ │ │ + for (var i = 0, len = multipoint.components.length; i < len; ++i) { │ │ │ │ │ + array.push(this.extract.point.apply(this, [multipoint.components[i]])); │ │ │ │ │ + } │ │ │ │ │ + return array; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Method: extract.linestring │ │ │ │ │ + * Return an array of coordinate arrays from a linestring. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * linestring - {<OpenLayers.Geometry.LineString>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} An array of coordinate arrays representing │ │ │ │ │ + * the linestring. │ │ │ │ │ + */ │ │ │ │ │ + 'linestring': function(linestring) { │ │ │ │ │ + var array = []; │ │ │ │ │ + for (var i = 0, len = linestring.components.length; i < len; ++i) { │ │ │ │ │ + array.push(this.extract.point.apply(this, [linestring.components[i]])); │ │ │ │ │ + } │ │ │ │ │ + return array; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Bounds │ │ │ │ │ + /** │ │ │ │ │ + * Method: extract.multilinestring │ │ │ │ │ + * Return an array of linestring arrays from a linestring. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * multilinestring - {<OpenLayers.Geometry.MultiLineString>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} An array of linestring arrays representing │ │ │ │ │ + * the multilinestring. │ │ │ │ │ + */ │ │ │ │ │ + 'multilinestring': function(multilinestring) { │ │ │ │ │ + var array = []; │ │ │ │ │ + for (var i = 0, len = multilinestring.components.length; i < len; ++i) { │ │ │ │ │ + array.push(this.extract.linestring.apply(this, [multilinestring.components[i]])); │ │ │ │ │ + } │ │ │ │ │ + return array; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectZoomFromMapObjectBounds │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moBounds - {Object} MapObject Bounds format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject Zoom for specified MapObject Bounds │ │ │ │ │ - */ │ │ │ │ │ - getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - return this.mapObject.getBoundsZoomLevel(moBounds); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: extract.polygon │ │ │ │ │ + * Return an array of linear ring arrays from a polygon. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * polygon - {<OpenLayers.Geometry.Polygon>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} An array of linear ring arrays representing the polygon. │ │ │ │ │ + */ │ │ │ │ │ + 'polygon': function(polygon) { │ │ │ │ │ + var array = []; │ │ │ │ │ + for (var i = 0, len = polygon.components.length; i < len; ++i) { │ │ │ │ │ + array.push(this.extract.linestring.apply(this, [polygon.components[i]])); │ │ │ │ │ + } │ │ │ │ │ + return array; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Primitives * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ + /** │ │ │ │ │ + * Method: extract.multipolygon │ │ │ │ │ + * Return an array of polygon arrays from a multipolygon. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * multipolygon - {<OpenLayers.Geometry.MultiPolygon>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} An array of polygon arrays representing │ │ │ │ │ + * the multipolygon │ │ │ │ │ + */ │ │ │ │ │ + 'multipolygon': function(multipolygon) { │ │ │ │ │ + var array = []; │ │ │ │ │ + for (var i = 0, len = multipolygon.components.length; i < len; ++i) { │ │ │ │ │ + array.push(this.extract.polygon.apply(this, [multipolygon.components[i]])); │ │ │ │ │ + } │ │ │ │ │ + return array; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Method: extract.collection │ │ │ │ │ + * Return an array of geometries from a geometry collection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * collection - {<OpenLayers.Geometry.Collection>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} An array of geometry objects representing the geometry │ │ │ │ │ + * collection. │ │ │ │ │ + */ │ │ │ │ │ + 'collection': function(collection) { │ │ │ │ │ + var len = collection.components.length; │ │ │ │ │ + var array = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + array[i] = this.extract.geometry.apply( │ │ │ │ │ + this, [collection.components[i]] │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return array; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // LonLat │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectLonLatFromLonLat │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * lon - {Float} │ │ │ │ │ - * lat - {Float} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject LonLat built from lon and lat params │ │ │ │ │ - */ │ │ │ │ │ - getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ - var gLatLng; │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ - gLatLng = new google.maps.LatLng(lonlat.lat, lonlat.lon); │ │ │ │ │ - } else { │ │ │ │ │ - gLatLng = new google.maps.LatLng(lat, lon); │ │ │ │ │ - } │ │ │ │ │ - return gLatLng; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - // Pixel │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectPixelFromXY │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {Integer} │ │ │ │ │ - * y - {Integer} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject Pixel from x and y parameters │ │ │ │ │ - */ │ │ │ │ │ - getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ - return new google.maps.Point(x, y); │ │ │ │ │ - } │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.GeoJSON" │ │ │ │ │ │ │ │ │ │ -}; │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Filter.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ @@ -20003,8582 +21511,2913 @@ │ │ │ │ │ OpenLayers.Filter.Comparison.GREATER_THAN = ">"; │ │ │ │ │ OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO = "<="; │ │ │ │ │ OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO = ">="; │ │ │ │ │ OpenLayers.Filter.Comparison.BETWEEN = ".."; │ │ │ │ │ OpenLayers.Filter.Comparison.LIKE = "~"; │ │ │ │ │ OpenLayers.Filter.Comparison.IS_NULL = "NULL"; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Popup.js │ │ │ │ │ + OpenLayers/Control.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Popup │ │ │ │ │ - * A popup is a small div that can opened and closed on the map. │ │ │ │ │ - * Typically opened in response to clicking on a marker. │ │ │ │ │ - * See <OpenLayers.Marker>. Popup's don't require their own │ │ │ │ │ - * layer and are added the the map using the <OpenLayers.Map.addPopup> │ │ │ │ │ - * method. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Control │ │ │ │ │ + * Controls affect the display or behavior of the map. They allow everything │ │ │ │ │ + * from panning and zooming to displaying a scale indicator. Controls by │ │ │ │ │ + * default are added to the map they are contained within however it is │ │ │ │ │ + * possible to add a control to an external div by passing the div in the │ │ │ │ │ + * options parameter. │ │ │ │ │ + * │ │ │ │ │ * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * popup = new OpenLayers.Popup("chicken", │ │ │ │ │ - * new OpenLayers.LonLat(5,40), │ │ │ │ │ - * new OpenLayers.Size(200,200), │ │ │ │ │ - * "example popup", │ │ │ │ │ - * true); │ │ │ │ │ - * │ │ │ │ │ - * map.addPopup(popup); │ │ │ │ │ - * (end) │ │ │ │ │ + * The following example shows how to add many of the common controls │ │ │ │ │ + * to a map. │ │ │ │ │ + * │ │ │ │ │ + * > var map = new OpenLayers.Map('map', { controls: [] }); │ │ │ │ │ + * > │ │ │ │ │ + * > map.addControl(new OpenLayers.Control.PanZoomBar()); │ │ │ │ │ + * > map.addControl(new OpenLayers.Control.LayerSwitcher({'ascending':false})); │ │ │ │ │ + * > map.addControl(new OpenLayers.Control.Permalink()); │ │ │ │ │ + * > map.addControl(new OpenLayers.Control.Permalink('permalink')); │ │ │ │ │ + * > map.addControl(new OpenLayers.Control.MousePosition()); │ │ │ │ │ + * > map.addControl(new OpenLayers.Control.OverviewMap()); │ │ │ │ │ + * > map.addControl(new OpenLayers.Control.KeyboardDefaults()); │ │ │ │ │ + * │ │ │ │ │ + * The next code fragment is a quick example of how to intercept │ │ │ │ │ + * shift-mouse click to display the extent of the bounding box │ │ │ │ │ + * dragged out by the user. Usually controls are not created │ │ │ │ │ + * in exactly this manner. See the source for a more complete │ │ │ │ │ + * example: │ │ │ │ │ + * │ │ │ │ │ + * > var control = new OpenLayers.Control(); │ │ │ │ │ + * > OpenLayers.Util.extend(control, { │ │ │ │ │ + * > draw: function () { │ │ │ │ │ + * > // this Handler.Box will intercept the shift-mousedown │ │ │ │ │ + * > // before Control.MouseDefault gets to see it │ │ │ │ │ + * > this.box = new OpenLayers.Handler.Box( control, │ │ │ │ │ + * > {"done": this.notice}, │ │ │ │ │ + * > {keyMask: OpenLayers.Handler.MOD_SHIFT}); │ │ │ │ │ + * > this.box.activate(); │ │ │ │ │ + * > }, │ │ │ │ │ + * > │ │ │ │ │ + * > notice: function (bounds) { │ │ │ │ │ + * > OpenLayers.Console.userError(bounds); │ │ │ │ │ + * > } │ │ │ │ │ + * > }); │ │ │ │ │ + * > map.addControl(control); │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Popup = OpenLayers.Class({ │ │ │ │ │ +OpenLayers.Control = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: events │ │ │ │ │ - * {<OpenLayers.Events>} custom event manager │ │ │ │ │ - */ │ │ │ │ │ - events: null, │ │ │ │ │ - │ │ │ │ │ - /** Property: id │ │ │ │ │ - * {String} the unique identifier assigned to this popup. │ │ │ │ │ + * Property: id │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - id: "", │ │ │ │ │ + id: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: lonlat │ │ │ │ │ - * {<OpenLayers.LonLat>} the position of this popup on the map │ │ │ │ │ + * Property: map │ │ │ │ │ + * {<OpenLayers.Map>} this gets set in the addControl() function in │ │ │ │ │ + * OpenLayers.Map │ │ │ │ │ */ │ │ │ │ │ - lonlat: null, │ │ │ │ │ + map: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: div │ │ │ │ │ - * {DOMElement} the div that contains this popup. │ │ │ │ │ + * APIProperty: div │ │ │ │ │ + * {DOMElement} The element that contains the control, if not present the │ │ │ │ │ + * control is placed inside the map. │ │ │ │ │ */ │ │ │ │ │ div: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: contentSize │ │ │ │ │ - * {<OpenLayers.Size>} the width and height of the content. │ │ │ │ │ - */ │ │ │ │ │ - contentSize: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: size │ │ │ │ │ - * {<OpenLayers.Size>} the width and height of the popup. │ │ │ │ │ - */ │ │ │ │ │ - size: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: contentHTML │ │ │ │ │ - * {String} An HTML string for this popup to display. │ │ │ │ │ - */ │ │ │ │ │ - contentHTML: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: backgroundColor │ │ │ │ │ - * {String} the background color used by the popup. │ │ │ │ │ - */ │ │ │ │ │ - backgroundColor: "", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: opacity │ │ │ │ │ - * {float} the opacity of this popup (between 0.0 and 1.0) │ │ │ │ │ - */ │ │ │ │ │ - opacity: "", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: border │ │ │ │ │ - * {String} the border size of the popup. (eg 2px) │ │ │ │ │ - */ │ │ │ │ │ - border: "", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: contentDiv │ │ │ │ │ - * {DOMElement} a reference to the element that holds the content of │ │ │ │ │ - * the div. │ │ │ │ │ - */ │ │ │ │ │ - contentDiv: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: groupDiv │ │ │ │ │ - * {DOMElement} First and only child of 'div'. The group Div contains the │ │ │ │ │ - * 'contentDiv' and the 'closeDiv'. │ │ │ │ │ + * APIProperty: type │ │ │ │ │ + * {Number} Controls can have a 'type'. The type determines the type of │ │ │ │ │ + * interactions which are possible with them when they are placed in an │ │ │ │ │ + * <OpenLayers.Control.Panel>. │ │ │ │ │ */ │ │ │ │ │ - groupDiv: null, │ │ │ │ │ + type: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: closeDiv │ │ │ │ │ - * {DOMElement} the optional closer image │ │ │ │ │ + * Property: allowSelection │ │ │ │ │ + * {Boolean} By default, controls do not allow selection, because │ │ │ │ │ + * it may interfere with map dragging. If this is true, OpenLayers │ │ │ │ │ + * will not prevent selection of the control. │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - closeDiv: null, │ │ │ │ │ + allowSelection: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: autoSize │ │ │ │ │ - * {Boolean} Resize the popup to auto-fit the contents. │ │ │ │ │ - * Default is false. │ │ │ │ │ + * Property: displayClass │ │ │ │ │ + * {string} This property is used for CSS related to the drawing of the │ │ │ │ │ + * Control. │ │ │ │ │ */ │ │ │ │ │ - autoSize: false, │ │ │ │ │ + displayClass: "", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: minSize │ │ │ │ │ - * {<OpenLayers.Size>} Minimum size allowed for the popup's contents. │ │ │ │ │ + * APIProperty: title │ │ │ │ │ + * {string} This property is used for showing a tooltip over the │ │ │ │ │ + * Control. │ │ │ │ │ */ │ │ │ │ │ - minSize: null, │ │ │ │ │ + title: "", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: maxSize │ │ │ │ │ - * {<OpenLayers.Size>} Maximum size allowed for the popup's contents. │ │ │ │ │ - */ │ │ │ │ │ - maxSize: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: displayClass │ │ │ │ │ - * {String} The CSS class of the popup. │ │ │ │ │ - */ │ │ │ │ │ - displayClass: "olPopup", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: contentDisplayClass │ │ │ │ │ - * {String} The CSS class of the popup content div. │ │ │ │ │ - */ │ │ │ │ │ - contentDisplayClass: "olPopupContent", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: padding │ │ │ │ │ - * {int or <OpenLayers.Bounds>} An extra opportunity to specify internal │ │ │ │ │ - * padding of the content div inside the popup. This was originally │ │ │ │ │ - * confused with the css padding as specified in style.css's │ │ │ │ │ - * 'olPopupContent' class. We would like to get rid of this altogether, │ │ │ │ │ - * except that it does come in handy for the framed and anchoredbubble │ │ │ │ │ - * popups, who need to maintain yet another barrier between their │ │ │ │ │ - * content and the outer border of the popup itself. │ │ │ │ │ - * │ │ │ │ │ - * Note that in order to not break API, we must continue to support │ │ │ │ │ - * this property being set as an integer. Really, though, we'd like to │ │ │ │ │ - * have this specified as a Bounds object so that user can specify │ │ │ │ │ - * distinct left, top, right, bottom paddings. With the 3.0 release │ │ │ │ │ - * we can make this only a bounds. │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * false. │ │ │ │ │ */ │ │ │ │ │ - padding: 0, │ │ │ │ │ + autoActivate: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: disableFirefoxOverflowHack │ │ │ │ │ - * {Boolean} The hack for overflow in Firefox causes all elements │ │ │ │ │ - * to be re-drawn, which causes Flash elements to be │ │ │ │ │ - * re-initialized, which is troublesome. │ │ │ │ │ - * With this property the hack can be disabled. │ │ │ │ │ - */ │ │ │ │ │ - disableFirefoxOverflowHack: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: fixPadding │ │ │ │ │ - * To be removed in 3.0, this function merely helps us to deal with the │ │ │ │ │ - * case where the user may have set an integer value for padding, │ │ │ │ │ - * instead of an <OpenLayers.Bounds> object. │ │ │ │ │ + * APIProperty: active │ │ │ │ │ + * {Boolean} The control is active (read-only). Use <activate> and │ │ │ │ │ + * <deactivate> to change control state. │ │ │ │ │ */ │ │ │ │ │ - fixPadding: function() { │ │ │ │ │ - if (typeof this.padding == "number") { │ │ │ │ │ - this.padding = new OpenLayers.Bounds( │ │ │ │ │ - this.padding, this.padding, this.padding, this.padding │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + active: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: panMapIfOutOfView │ │ │ │ │ - * {Boolean} When drawn, pan map such that the entire popup is visible in │ │ │ │ │ - * the current viewport (if necessary). │ │ │ │ │ - * Default is false. │ │ │ │ │ + * Property: handlerOptions │ │ │ │ │ + * {Object} Used to set non-default properties on the control's handler │ │ │ │ │ */ │ │ │ │ │ - panMapIfOutOfView: false, │ │ │ │ │ + handlerOptions: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: keepInMap │ │ │ │ │ - * {Boolean} If panMapIfOutOfView is false, and this property is true, │ │ │ │ │ - * contrain the popup such that it always fits in the available map │ │ │ │ │ - * space. By default, this is not set on the base class. If you are │ │ │ │ │ - * creating popups that are near map edges and not allowing pannning, │ │ │ │ │ - * and especially if you have a popup which has a │ │ │ │ │ - * fixedRelativePosition, setting this to false may be a smart thing to │ │ │ │ │ - * do. Subclasses may want to override this setting. │ │ │ │ │ - * │ │ │ │ │ - * Default is false. │ │ │ │ │ + /** │ │ │ │ │ + * Property: handler │ │ │ │ │ + * {<OpenLayers.Handler>} null │ │ │ │ │ */ │ │ │ │ │ - keepInMap: false, │ │ │ │ │ + handler: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: closeOnMove │ │ │ │ │ - * {Boolean} When map pans, close the popup. │ │ │ │ │ - * Default is false. │ │ │ │ │ + * APIProperty: eventListeners │ │ │ │ │ + * {Object} If set as an option at construction, the eventListeners │ │ │ │ │ + * object will be registered with <OpenLayers.Events.on>. Object │ │ │ │ │ + * structure must be a listeners object as shown in the example for │ │ │ │ │ + * the events.on method. │ │ │ │ │ */ │ │ │ │ │ - closeOnMove: false, │ │ │ │ │ + eventListeners: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: map │ │ │ │ │ - * {<OpenLayers.Map>} this gets set in Map.js when the popup is added to the map │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Listeners will be called with a reference to an event object. The │ │ │ │ │ + * properties of this event depends on exactly what happened. │ │ │ │ │ + * │ │ │ │ │ + * All event objects have at least the following properties: │ │ │ │ │ + * object - {Object} A reference to control.events.object (a reference │ │ │ │ │ + * to the control). │ │ │ │ │ + * element - {DOMElement} A reference to control.events.element (which │ │ │ │ │ + * will be null unless documented otherwise). │ │ │ │ │ + * │ │ │ │ │ + * Supported map event types: │ │ │ │ │ + * activate - Triggered when activated. │ │ │ │ │ + * deactivate - Triggered when deactivated. │ │ │ │ │ */ │ │ │ │ │ - map: null, │ │ │ │ │ + events: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Popup │ │ │ │ │ - * Create a popup. │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control │ │ │ │ │ + * Create an OpenLayers Control. The options passed as a parameter │ │ │ │ │ + * directly extend the control. For example passing the following: │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} a unqiue identifier for this popup. If null is passed │ │ │ │ │ - * an identifier will be automatically generated. │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>} The position on the map the popup will │ │ │ │ │ - * be shown. │ │ │ │ │ - * contentSize - {<OpenLayers.Size>} The size of the content. │ │ │ │ │ - * contentHTML - {String} An HTML string to display inside the │ │ │ │ │ - * popup. │ │ │ │ │ - * closeBox - {Boolean} Whether to display a close box inside │ │ │ │ │ - * the popup. │ │ │ │ │ - * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ + * > var control = new OpenLayers.Control({div: myDiv}); │ │ │ │ │ + * │ │ │ │ │ + * Overrides the default div attribute value of null. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, closeBox, closeBoxCallback) { │ │ │ │ │ - if (id == null) { │ │ │ │ │ - id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ - } │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + // We do this before the extend so that instances can override │ │ │ │ │ + // className in options. │ │ │ │ │ + this.displayClass = │ │ │ │ │ + this.CLASS_NAME.replace("OpenLayers.", "ol").replace(/\./g, ""); │ │ │ │ │ │ │ │ │ │ - this.id = id; │ │ │ │ │ - this.lonlat = lonlat; │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ │ │ │ │ │ - this.contentSize = (contentSize != null) ? contentSize : │ │ │ │ │ - new OpenLayers.Size( │ │ │ │ │ - OpenLayers.Popup.WIDTH, │ │ │ │ │ - OpenLayers.Popup.HEIGHT); │ │ │ │ │ - if (contentHTML != null) { │ │ │ │ │ - this.contentHTML = contentHTML; │ │ │ │ │ + this.events = new OpenLayers.Events(this); │ │ │ │ │ + if (this.eventListeners instanceof Object) { │ │ │ │ │ + this.events.on(this.eventListeners); │ │ │ │ │ } │ │ │ │ │ - this.backgroundColor = OpenLayers.Popup.COLOR; │ │ │ │ │ - this.opacity = OpenLayers.Popup.OPACITY; │ │ │ │ │ - this.border = OpenLayers.Popup.BORDER; │ │ │ │ │ - │ │ │ │ │ - this.div = OpenLayers.Util.createDiv(this.id, null, null, │ │ │ │ │ - null, null, null, "hidden"); │ │ │ │ │ - this.div.className = this.displayClass; │ │ │ │ │ - │ │ │ │ │ - var groupDivId = this.id + "_GroupDiv"; │ │ │ │ │ - this.groupDiv = OpenLayers.Util.createDiv(groupDivId, null, null, │ │ │ │ │ - null, "relative", null, │ │ │ │ │ - "hidden"); │ │ │ │ │ - │ │ │ │ │ - var id = this.div.id + "_contentDiv"; │ │ │ │ │ - this.contentDiv = OpenLayers.Util.createDiv(id, null, this.contentSize.clone(), │ │ │ │ │ - null, "relative"); │ │ │ │ │ - this.contentDiv.className = this.contentDisplayClass; │ │ │ │ │ - this.groupDiv.appendChild(this.contentDiv); │ │ │ │ │ - this.div.appendChild(this.groupDiv); │ │ │ │ │ - │ │ │ │ │ - if (closeBox) { │ │ │ │ │ - this.addCloseBox(closeBoxCallback); │ │ │ │ │ + if (this.id == null) { │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - this.registerEvents(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ + /** │ │ │ │ │ * Method: destroy │ │ │ │ │ - * nullify references to prevent circular references and memory leaks │ │ │ │ │ + * The destroy method is used to perform any clean up before the control │ │ │ │ │ + * is dereferenced. Typically this is where event listeners are removed │ │ │ │ │ + * to prevent memory leaks. │ │ │ │ │ */ │ │ │ │ │ destroy: function() { │ │ │ │ │ - │ │ │ │ │ - this.id = null; │ │ │ │ │ - this.lonlat = null; │ │ │ │ │ - this.size = null; │ │ │ │ │ - this.contentHTML = null; │ │ │ │ │ - │ │ │ │ │ - this.backgroundColor = null; │ │ │ │ │ - this.opacity = null; │ │ │ │ │ - this.border = null; │ │ │ │ │ - │ │ │ │ │ - if (this.closeOnMove && this.map) { │ │ │ │ │ - this.map.events.unregister("movestart", this, this.hide); │ │ │ │ │ + if (this.events) { │ │ │ │ │ + if (this.eventListeners) { │ │ │ │ │ + this.events.un(this.eventListeners); │ │ │ │ │ + } │ │ │ │ │ + this.events.destroy(); │ │ │ │ │ + this.events = null; │ │ │ │ │ } │ │ │ │ │ + this.eventListeners = null; │ │ │ │ │ │ │ │ │ │ - this.events.destroy(); │ │ │ │ │ - this.events = null; │ │ │ │ │ - │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - OpenLayers.Event.stopObservingElement(this.closeDiv); │ │ │ │ │ - this.groupDiv.removeChild(this.closeDiv); │ │ │ │ │ + // eliminate circular references │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.destroy(); │ │ │ │ │ + this.handler = null; │ │ │ │ │ } │ │ │ │ │ - this.closeDiv = null; │ │ │ │ │ - │ │ │ │ │ - this.div.removeChild(this.groupDiv); │ │ │ │ │ - this.groupDiv = null; │ │ │ │ │ - │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.removePopup(this); │ │ │ │ │ + if (this.handlers) { │ │ │ │ │ + for (var key in this.handlers) { │ │ │ │ │ + if (this.handlers.hasOwnProperty(key) && │ │ │ │ │ + typeof this.handlers[key].destroy == "function") { │ │ │ │ │ + this.handlers[key].destroy(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.handlers = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.removeControl(this); │ │ │ │ │ + this.map = null; │ │ │ │ │ } │ │ │ │ │ - this.map = null; │ │ │ │ │ this.div = null; │ │ │ │ │ - │ │ │ │ │ - this.autoSize = null; │ │ │ │ │ - this.minSize = null; │ │ │ │ │ - this.maxSize = null; │ │ │ │ │ - this.padding = null; │ │ │ │ │ - this.panMapIfOutOfView = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the control. This is done through an accessor │ │ │ │ │ + * so that subclasses can override this and take special action once │ │ │ │ │ + * they have their map variable set. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.map = map; │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.setMap(map); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ * Method: draw │ │ │ │ │ - * Constructs the elements that make up the popup. │ │ │ │ │ + * The draw method is called when the control is ready to be displayed │ │ │ │ │ + * on the page. If a div has not been created one is created. Controls │ │ │ │ │ + * with a visual component will almost always want to override this method │ │ │ │ │ + * to customize the look of control. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} the position the popup in pixels. │ │ │ │ │ - * │ │ │ │ │ + * px - {<OpenLayers.Pixel>} The top-left pixel position of the control │ │ │ │ │ + * or null. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} Reference to a div that contains the drawn popup │ │ │ │ │ + * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ */ │ │ │ │ │ draw: function(px) { │ │ │ │ │ - if (px == null) { │ │ │ │ │ - if ((this.lonlat != null) && (this.map != null)) { │ │ │ │ │ - px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ + if (this.div == null) { │ │ │ │ │ + this.div = OpenLayers.Util.createDiv(this.id); │ │ │ │ │ + this.div.className = this.displayClass; │ │ │ │ │ + if (!this.allowSelection) { │ │ │ │ │ + this.div.className += " olControlNoSelect"; │ │ │ │ │ + this.div.setAttribute("unselectable", "on", 0); │ │ │ │ │ + this.div.onselectstart = OpenLayers.Function.False; │ │ │ │ │ + } │ │ │ │ │ + if (this.title != "") { │ │ │ │ │ + this.div.title = this.title; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // this assumes that this.map already exists, which is okay because │ │ │ │ │ - // this.draw is only called once the popup has been added to the map. │ │ │ │ │ - if (this.closeOnMove) { │ │ │ │ │ - this.map.events.register("movestart", this, this.hide); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //listen to movestart, moveend to disable overflow (FF bug) │ │ │ │ │ - if (!this.disableFirefoxOverflowHack && OpenLayers.BROWSER_NAME == 'firefox') { │ │ │ │ │ - this.map.events.register("movestart", this, function() { │ │ │ │ │ - var style = document.defaultView.getComputedStyle( │ │ │ │ │ - this.contentDiv, null │ │ │ │ │ - ); │ │ │ │ │ - var currentOverflow = style.getPropertyValue("overflow"); │ │ │ │ │ - if (currentOverflow != "hidden") { │ │ │ │ │ - this.contentDiv._oldOverflow = currentOverflow; │ │ │ │ │ - this.contentDiv.style.overflow = "hidden"; │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - this.map.events.register("moveend", this, function() { │ │ │ │ │ - var oldOverflow = this.contentDiv._oldOverflow; │ │ │ │ │ - if (oldOverflow) { │ │ │ │ │ - this.contentDiv.style.overflow = oldOverflow; │ │ │ │ │ - this.contentDiv._oldOverflow = null; │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.moveTo(px); │ │ │ │ │ - if (!this.autoSize && !this.size) { │ │ │ │ │ - this.setSize(this.contentSize); │ │ │ │ │ - } │ │ │ │ │ - this.setBackgroundColor(); │ │ │ │ │ - this.setOpacity(); │ │ │ │ │ - this.setBorder(); │ │ │ │ │ - this.setContentHTML(); │ │ │ │ │ - │ │ │ │ │ - if (this.panMapIfOutOfView) { │ │ │ │ │ - this.panIntoView(); │ │ │ │ │ + if (px != null) { │ │ │ │ │ + this.position = px.clone(); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ + this.moveTo(this.position); │ │ │ │ │ return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: updatePosition │ │ │ │ │ - * if the popup has a lonlat and its map members set, │ │ │ │ │ - * then have it move itself to its proper position │ │ │ │ │ - */ │ │ │ │ │ - updatePosition: function() { │ │ │ │ │ - if ((this.lonlat) && (this.map)) { │ │ │ │ │ - var px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ - if (px) { │ │ │ │ │ - this.moveTo(px); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ * Method: moveTo │ │ │ │ │ - * │ │ │ │ │ + * Sets the left and top style attributes to the passed in pixel │ │ │ │ │ + * coordinates. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} the top and left position of the popup div. │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ */ │ │ │ │ │ moveTo: function(px) { │ │ │ │ │ if ((px != null) && (this.div != null)) { │ │ │ │ │ this.div.style.left = px.x + "px"; │ │ │ │ │ this.div.style.top = px.y + "px"; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: visible │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Boolean indicating whether or not the popup is visible │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Explicitly activates a control and it's associated │ │ │ │ │ + * handler if one has been set. Controls can be │ │ │ │ │ + * deactivated by calling the deactivate() method. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} True if the control was successfully activated or │ │ │ │ │ + * false if the control was already active. │ │ │ │ │ */ │ │ │ │ │ - visible: function() { │ │ │ │ │ - return OpenLayers.Element.visible(this.div); │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.activate(); │ │ │ │ │ + } │ │ │ │ │ + this.active = true; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + OpenLayers.Element.addClass( │ │ │ │ │ + this.map.viewPortDiv, │ │ │ │ │ + this.displayClass.replace(/ /g, "") + "Active" │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("activate"); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: toggle │ │ │ │ │ - * Toggles visibility of the popup. │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivates a control and it's associated handler if any. The exact │ │ │ │ │ + * effect of this depends on the control itself. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} True if the control was effectively deactivated or false │ │ │ │ │ + * if the control was already inactive. │ │ │ │ │ */ │ │ │ │ │ - toggle: function() { │ │ │ │ │ - if (this.visible()) { │ │ │ │ │ - this.hide(); │ │ │ │ │ - } else { │ │ │ │ │ - this.show(); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.deactivate(); │ │ │ │ │ + } │ │ │ │ │ + this.active = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + OpenLayers.Element.removeClass( │ │ │ │ │ + this.map.viewPortDiv, │ │ │ │ │ + this.displayClass.replace(/ /g, "") + "Active" │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("deactivate"); │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Control.TYPE_BUTTON │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.TYPE_BUTTON = 1; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Control.TYPE_TOGGLE │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.TYPE_TOGGLE = 2; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Control.TYPE_TOOL │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.TYPE_TOOL = 3; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Events/buttonclick.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Events.buttonclick │ │ │ │ │ + * Extension event type for handling buttons on top of a dom element. This │ │ │ │ │ + * event type fires "buttonclick" on its <target> when a button was │ │ │ │ │ + * clicked. Buttons are detected by the "olButton" class. │ │ │ │ │ + * │ │ │ │ │ + * This event type makes sure that button clicks do not interfere with other │ │ │ │ │ + * events that are registered on the same <element>. │ │ │ │ │ + * │ │ │ │ │ + * Event types provided by this extension: │ │ │ │ │ + * - *buttonclick* Triggered when a button is clicked. Listeners receive an │ │ │ │ │ + * object with a *buttonElement* property referencing the dom element of │ │ │ │ │ + * the clicked button, and an *buttonXY* property with the click position │ │ │ │ │ + * relative to the button. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Events.buttonclick = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: show │ │ │ │ │ - * Makes the popup visible. │ │ │ │ │ + * Property: target │ │ │ │ │ + * {<OpenLayers.Events>} The events instance that the buttonclick event will │ │ │ │ │ + * be triggered on. │ │ │ │ │ */ │ │ │ │ │ - show: function() { │ │ │ │ │ - this.div.style.display = ''; │ │ │ │ │ + target: null, │ │ │ │ │ │ │ │ │ │ - if (this.panMapIfOutOfView) { │ │ │ │ │ - this.panIntoView(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: events │ │ │ │ │ + * {Array} Events to observe and conditionally stop from propagating when │ │ │ │ │ + * an element with the olButton class (or its olAlphaImg child) is │ │ │ │ │ + * clicked. │ │ │ │ │ + */ │ │ │ │ │ + events: [ │ │ │ │ │ + 'mousedown', 'mouseup', 'click', 'dblclick', │ │ │ │ │ + 'touchstart', 'touchmove', 'touchend', 'keydown' │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: hide │ │ │ │ │ - * Makes the popup invisible. │ │ │ │ │ + * Property: startRegEx │ │ │ │ │ + * {RegExp} Regular expression to test Event.type for events that start │ │ │ │ │ + * a buttonclick sequence. │ │ │ │ │ */ │ │ │ │ │ - hide: function() { │ │ │ │ │ - this.div.style.display = 'none'; │ │ │ │ │ - }, │ │ │ │ │ + startRegEx: /^mousedown|touchstart$/, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setSize │ │ │ │ │ - * Used to adjust the size of the popup. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * contentSize - {<OpenLayers.Size>} the new size for the popup's │ │ │ │ │ - * contents div (in pixels). │ │ │ │ │ + * Property: cancelRegEx │ │ │ │ │ + * {RegExp} Regular expression to test Event.type for events that cancel │ │ │ │ │ + * a buttonclick sequence. │ │ │ │ │ */ │ │ │ │ │ - setSize: function(contentSize) { │ │ │ │ │ - this.size = contentSize.clone(); │ │ │ │ │ + cancelRegEx: /^touchmove$/, │ │ │ │ │ │ │ │ │ │ - // if our contentDiv has a css 'padding' set on it by a stylesheet, we │ │ │ │ │ - // must add that to the desired "size". │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ - var wPadding = contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ - var hPadding = contentDivPadding.top + contentDivPadding.bottom; │ │ │ │ │ + /** │ │ │ │ │ + * Property: completeRegEx │ │ │ │ │ + * {RegExp} Regular expression to test Event.type for events that complete │ │ │ │ │ + * a buttonclick sequence. │ │ │ │ │ + */ │ │ │ │ │ + completeRegEx: /^mouseup|touchend$/, │ │ │ │ │ │ │ │ │ │ - // take into account the popup's 'padding' property │ │ │ │ │ - this.fixPadding(); │ │ │ │ │ - wPadding += this.padding.left + this.padding.right; │ │ │ │ │ - hPadding += this.padding.top + this.padding.bottom; │ │ │ │ │ + /** │ │ │ │ │ + * Property: startEvt │ │ │ │ │ + * {Event} The event that started the click sequence │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // make extra space for the close div │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - var closeDivWidth = parseInt(this.closeDiv.style.width); │ │ │ │ │ - wPadding += closeDivWidth + contentDivPadding.right; │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Events.buttonclick │ │ │ │ │ + * Construct a buttonclick event type. Applications are not supposed to │ │ │ │ │ + * create instances of this class - they are created on demand by │ │ │ │ │ + * <OpenLayers.Events> instances. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * target - {<OpenLayers.Events>} The events instance that the buttonclick │ │ │ │ │ + * event will be triggered on. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(target) { │ │ │ │ │ + this.target = target; │ │ │ │ │ + for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ + this.target.register(this.events[i], this, this.buttonClick, { │ │ │ │ │ + extension: true │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - //increase size of the main popup div to take into account the │ │ │ │ │ - // users's desired padding and close div. │ │ │ │ │ - this.size.w += wPadding; │ │ │ │ │ - this.size.h += hPadding; │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ + this.target.unregister(this.events[i], this, this.buttonClick); │ │ │ │ │ + } │ │ │ │ │ + delete this.target; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - //now if our browser is IE, we need to actually make the contents │ │ │ │ │ - // div itself bigger to take its own padding into effect. this makes │ │ │ │ │ - // me want to shoot someone, but so it goes. │ │ │ │ │ - if (OpenLayers.BROWSER_NAME == "msie") { │ │ │ │ │ - this.contentSize.w += │ │ │ │ │ - contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ - this.contentSize.h += │ │ │ │ │ - contentDivPadding.bottom + contentDivPadding.top; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.div.style.width = this.size.w + "px"; │ │ │ │ │ - this.div.style.height = this.size.h + "px"; │ │ │ │ │ - } │ │ │ │ │ - if (this.contentDiv != null) { │ │ │ │ │ - this.contentDiv.style.width = contentSize.w + "px"; │ │ │ │ │ - this.contentDiv.style.height = contentSize.h + "px"; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: getPressedButton │ │ │ │ │ + * Get the pressed button, if any. Returns undefined if no button │ │ │ │ │ + * was pressed. │ │ │ │ │ + * │ │ │ │ │ + * Arguments: │ │ │ │ │ + * element - {DOMElement} The event target. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The button element, or undefined. │ │ │ │ │ + */ │ │ │ │ │ + getPressedButton: function(element) { │ │ │ │ │ + var depth = 3, // limit the search depth │ │ │ │ │ + button; │ │ │ │ │ + do { │ │ │ │ │ + if (OpenLayers.Element.hasClass(element, "olButton")) { │ │ │ │ │ + // hit! │ │ │ │ │ + button = element; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + element = element.parentNode; │ │ │ │ │ + } while (--depth > 0 && element); │ │ │ │ │ + return button; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: updateSize │ │ │ │ │ - * Auto size the popup so that it precisely fits its contents (as │ │ │ │ │ - * determined by this.contentDiv.innerHTML). Popup size will, of │ │ │ │ │ - * course, be limited by the available space on the current map │ │ │ │ │ + * Method: ignore │ │ │ │ │ + * Check for event target elements that should be ignored by OpenLayers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * element - {DOMElement} The event target. │ │ │ │ │ */ │ │ │ │ │ - updateSize: function() { │ │ │ │ │ + ignore: function(element) { │ │ │ │ │ + var depth = 3, │ │ │ │ │ + ignore = false; │ │ │ │ │ + do { │ │ │ │ │ + if (element.nodeName.toLowerCase() === 'a') { │ │ │ │ │ + ignore = true; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + element = element.parentNode; │ │ │ │ │ + } while (--depth > 0 && element); │ │ │ │ │ + return ignore; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // determine actual render dimensions of the contents by putting its │ │ │ │ │ - // contents into a fake contentDiv (for the CSS) and then measuring it │ │ │ │ │ - var preparedHTML = "<div class='" + this.contentDisplayClass + "'>" + │ │ │ │ │ - this.contentDiv.innerHTML + │ │ │ │ │ - "</div>"; │ │ │ │ │ + /** │ │ │ │ │ + * Method: buttonClick │ │ │ │ │ + * Check if a button was clicked, and fire the buttonclick event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + buttonClick: function(evt) { │ │ │ │ │ + var propagate = true, │ │ │ │ │ + element = OpenLayers.Event.element(evt); │ │ │ │ │ + if (element && (OpenLayers.Event.isLeftClick(evt) || !~evt.type.indexOf("mouse"))) { │ │ │ │ │ + // was a button pressed? │ │ │ │ │ + var button = this.getPressedButton(element); │ │ │ │ │ + if (button) { │ │ │ │ │ + if (evt.type === "keydown") { │ │ │ │ │ + switch (evt.keyCode) { │ │ │ │ │ + case OpenLayers.Event.KEY_RETURN: │ │ │ │ │ + case OpenLayers.Event.KEY_SPACE: │ │ │ │ │ + this.target.triggerEvent("buttonclick", { │ │ │ │ │ + buttonElement: button │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } else if (this.startEvt) { │ │ │ │ │ + if (this.completeRegEx.test(evt.type)) { │ │ │ │ │ + var pos = OpenLayers.Util.pagePosition(button); │ │ │ │ │ + var viewportElement = OpenLayers.Util.getViewportElement(); │ │ │ │ │ + var scrollTop = window.pageYOffset || viewportElement.scrollTop; │ │ │ │ │ + var scrollLeft = window.pageXOffset || viewportElement.scrollLeft; │ │ │ │ │ + pos[0] = pos[0] - scrollLeft; │ │ │ │ │ + pos[1] = pos[1] - scrollTop; │ │ │ │ │ │ │ │ │ │ - var containerElement = (this.map) ? this.map.div : document.body; │ │ │ │ │ - var realSize = OpenLayers.Util.getRenderedDimensions( │ │ │ │ │ - preparedHTML, null, { │ │ │ │ │ - displayClass: this.displayClass, │ │ │ │ │ - containerElement: containerElement │ │ │ │ │ + this.target.triggerEvent("buttonclick", { │ │ │ │ │ + buttonElement: button, │ │ │ │ │ + buttonXY: { │ │ │ │ │ + x: this.startEvt.clientX - pos[0], │ │ │ │ │ + y: this.startEvt.clientY - pos[1] │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + if (this.cancelRegEx.test(evt.type)) { │ │ │ │ │ + delete this.startEvt; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false; │ │ │ │ │ + } │ │ │ │ │ + if (this.startRegEx.test(evt.type)) { │ │ │ │ │ + this.startEvt = evt; │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + propagate = !this.ignore(OpenLayers.Event.element(evt)); │ │ │ │ │ + delete this.startEvt; │ │ │ │ │ } │ │ │ │ │ - ); │ │ │ │ │ + } │ │ │ │ │ + return propagate; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // is the "real" size of the div is safe to display in our map? │ │ │ │ │ - var safeSize = this.getSafeContentSize(realSize); │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/Zoom.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - var newSize = null; │ │ │ │ │ - if (safeSize.equals(realSize)) { │ │ │ │ │ - //real size of content is small enough to fit on the map, │ │ │ │ │ - // so we use real size. │ │ │ │ │ - newSize = realSize; │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - } else { │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // make a new 'size' object with the clipped dimensions │ │ │ │ │ - // set or null if not clipped. │ │ │ │ │ - var fixedSize = { │ │ │ │ │ - w: (safeSize.w < realSize.w) ? safeSize.w : null, │ │ │ │ │ - h: (safeSize.h < realSize.h) ? safeSize.h : null │ │ │ │ │ - }; │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.Zoom │ │ │ │ │ + * The Zoom control is a pair of +/- links for zooming in and out. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.Zoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - if (fixedSize.w && fixedSize.h) { │ │ │ │ │ - //content is too big in both directions, so we will use │ │ │ │ │ - // max popup size (safeSize), knowing well that it will │ │ │ │ │ - // overflow both ways. │ │ │ │ │ - newSize = safeSize; │ │ │ │ │ - } else { │ │ │ │ │ - //content is clipped in only one direction, so we need to │ │ │ │ │ - // run getRenderedDimensions() again with a fixed dimension │ │ │ │ │ - var clippedSize = OpenLayers.Util.getRenderedDimensions( │ │ │ │ │ - preparedHTML, fixedSize, { │ │ │ │ │ - displayClass: this.contentDisplayClass, │ │ │ │ │ - containerElement: containerElement │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomInText │ │ │ │ │ + * {String} │ │ │ │ │ + * Text for zoom-in link. Default is "+". │ │ │ │ │ + */ │ │ │ │ │ + zoomInText: "+", │ │ │ │ │ │ │ │ │ │ - //if the clipped size is still the same as the safeSize, │ │ │ │ │ - // that means that our content must be fixed in the │ │ │ │ │ - // offending direction. If overflow is 'auto', this means │ │ │ │ │ - // we are going to have a scrollbar for sure, so we must │ │ │ │ │ - // adjust for that. │ │ │ │ │ - // │ │ │ │ │ - var currentOverflow = OpenLayers.Element.getStyle( │ │ │ │ │ - this.contentDiv, "overflow" │ │ │ │ │ - ); │ │ │ │ │ - if ((currentOverflow != "hidden") && │ │ │ │ │ - (clippedSize.equals(safeSize))) { │ │ │ │ │ - var scrollBar = OpenLayers.Util.getScrollbarWidth(); │ │ │ │ │ - if (fixedSize.w) { │ │ │ │ │ - clippedSize.h += scrollBar; │ │ │ │ │ - } else { │ │ │ │ │ - clippedSize.w += scrollBar; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomInId │ │ │ │ │ + * {String} │ │ │ │ │ + * Instead of having the control create a zoom in link, you can provide │ │ │ │ │ + * the identifier for an anchor element already added to the document. │ │ │ │ │ + * By default, an element with id "olZoomInLink" will be searched for │ │ │ │ │ + * and used if it exists. │ │ │ │ │ + */ │ │ │ │ │ + zoomInId: "olZoomInLink", │ │ │ │ │ │ │ │ │ │ - newSize = this.getSafeContentSize(clippedSize); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.setSize(newSize); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomOutText │ │ │ │ │ + * {String} │ │ │ │ │ + * Text for zoom-out link. Default is "\u2212". │ │ │ │ │ + */ │ │ │ │ │ + zoomOutText: "\u2212", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setBackgroundColor │ │ │ │ │ - * Sets the background color of the popup. │ │ │ │ │ + * APIProperty: zoomOutId │ │ │ │ │ + * {String} │ │ │ │ │ + * Instead of having the control create a zoom out link, you can provide │ │ │ │ │ + * the identifier for an anchor element already added to the document. │ │ │ │ │ + * By default, an element with id "olZoomOutLink" will be searched for │ │ │ │ │ + * and used if it exists. │ │ │ │ │ + */ │ │ │ │ │ + zoomOutId: "olZoomOutLink", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * color - {String} the background color. eg "#FFBBBB" │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A reference to the DOMElement containing the zoom links. │ │ │ │ │ */ │ │ │ │ │ - setBackgroundColor: function(color) { │ │ │ │ │ - if (color != undefined) { │ │ │ │ │ - this.backgroundColor = color; │ │ │ │ │ - } │ │ │ │ │ + draw: function() { │ │ │ │ │ + var div = OpenLayers.Control.prototype.draw.apply(this), │ │ │ │ │ + links = this.getOrCreateLinks(div), │ │ │ │ │ + zoomIn = links.zoomIn, │ │ │ │ │ + zoomOut = links.zoomOut, │ │ │ │ │ + eventsInstance = this.map.events; │ │ │ │ │ │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.div.style.backgroundColor = this.backgroundColor; │ │ │ │ │ + if (zoomOut.parentNode !== div) { │ │ │ │ │ + eventsInstance = this.events; │ │ │ │ │ + eventsInstance.attachToElement(zoomOut.parentNode); │ │ │ │ │ } │ │ │ │ │ + eventsInstance.register("buttonclick", this, this.onZoomClick); │ │ │ │ │ + │ │ │ │ │ + this.zoomInLink = zoomIn; │ │ │ │ │ + this.zoomOutLink = zoomOut; │ │ │ │ │ + return div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setOpacity │ │ │ │ │ - * Sets the opacity of the popup. │ │ │ │ │ + * Method: getOrCreateLinks │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * opacity - {float} A value between 0.0 (transparent) and 1.0 (solid). │ │ │ │ │ + * el - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Return: │ │ │ │ │ + * {Object} Object with zoomIn and zoomOut properties referencing links. │ │ │ │ │ */ │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - if (opacity != undefined) { │ │ │ │ │ - this.opacity = opacity; │ │ │ │ │ + getOrCreateLinks: function(el) { │ │ │ │ │ + var zoomIn = document.getElementById(this.zoomInId), │ │ │ │ │ + zoomOut = document.getElementById(this.zoomOutId); │ │ │ │ │ + if (!zoomIn) { │ │ │ │ │ + zoomIn = document.createElement("a"); │ │ │ │ │ + zoomIn.href = "#zoomIn"; │ │ │ │ │ + zoomIn.appendChild(document.createTextNode(this.zoomInText)); │ │ │ │ │ + zoomIn.className = "olControlZoomIn"; │ │ │ │ │ + el.appendChild(zoomIn); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - // for Mozilla and Safari │ │ │ │ │ - this.div.style.opacity = this.opacity; │ │ │ │ │ - │ │ │ │ │ - // for IE │ │ │ │ │ - this.div.style.filter = 'alpha(opacity=' + this.opacity * 100 + ')'; │ │ │ │ │ + OpenLayers.Element.addClass(zoomIn, "olButton"); │ │ │ │ │ + if (!zoomOut) { │ │ │ │ │ + zoomOut = document.createElement("a"); │ │ │ │ │ + zoomOut.href = "#zoomOut"; │ │ │ │ │ + zoomOut.appendChild(document.createTextNode(this.zoomOutText)); │ │ │ │ │ + zoomOut.className = "olControlZoomOut"; │ │ │ │ │ + el.appendChild(zoomOut); │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Element.addClass(zoomOut, "olButton"); │ │ │ │ │ + return { │ │ │ │ │ + zoomIn: zoomIn, │ │ │ │ │ + zoomOut: zoomOut │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setBorder │ │ │ │ │ - * Sets the border style of the popup. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * border - {String} The border style value. eg 2px │ │ │ │ │ + * Method: onZoomClick │ │ │ │ │ + * Called when zoomin/out link is clicked. │ │ │ │ │ */ │ │ │ │ │ - setBorder: function(border) { │ │ │ │ │ - if (border != undefined) { │ │ │ │ │ - this.border = border; │ │ │ │ │ + onZoomClick: function(evt) { │ │ │ │ │ + var button = evt.buttonElement; │ │ │ │ │ + if (button === this.zoomInLink) { │ │ │ │ │ + this.map.zoomIn(); │ │ │ │ │ + } else if (button === this.zoomOutLink) { │ │ │ │ │ + this.map.zoomOut(); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.div.style.border = this.border; │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Clean up. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onZoomClick); │ │ │ │ │ } │ │ │ │ │ + delete this.zoomInLink; │ │ │ │ │ + delete this.zoomOutLink; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Zoom" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/Panel.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.Panel │ │ │ │ │ + * The Panel control is a container for other controls. With it toolbars │ │ │ │ │ + * may be composed. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.Panel = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ /** │ │ │ │ │ - * Method: setContentHTML │ │ │ │ │ - * Allows the user to set the HTML content of the popup. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * contentHTML - {String} HTML for the div. │ │ │ │ │ + * Property: controls │ │ │ │ │ + * {Array(<OpenLayers.Control>)} │ │ │ │ │ */ │ │ │ │ │ - setContentHTML: function(contentHTML) { │ │ │ │ │ + controls: null, │ │ │ │ │ │ │ │ │ │ - if (contentHTML != null) { │ │ │ │ │ - this.contentHTML = contentHTML; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ + */ │ │ │ │ │ + autoActivate: true, │ │ │ │ │ │ │ │ │ │ - if ((this.contentDiv != null) && │ │ │ │ │ - (this.contentHTML != null) && │ │ │ │ │ - (this.contentHTML != this.contentDiv.innerHTML)) { │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultControl │ │ │ │ │ + * {<OpenLayers.Control>} The control which is activated when the control is │ │ │ │ │ + * activated (turned on), which also happens at instantiation. │ │ │ │ │ + * If <saveState> is true, <defaultControl> will be nullified after the │ │ │ │ │ + * first activation of the panel. │ │ │ │ │ + */ │ │ │ │ │ + defaultControl: null, │ │ │ │ │ │ │ │ │ │ - this.contentDiv.innerHTML = this.contentHTML; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: saveState │ │ │ │ │ + * {Boolean} If set to true, the active state of this panel's controls will │ │ │ │ │ + * be stored on panel deactivation, and restored on reactivation. Default │ │ │ │ │ + * is false. │ │ │ │ │ + */ │ │ │ │ │ + saveState: false, │ │ │ │ │ │ │ │ │ │ - if (this.autoSize) { │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: allowDepress │ │ │ │ │ + * {Boolean} If is true the <OpenLayers.Control.TYPE_TOOL> controls can │ │ │ │ │ + * be deactivated by clicking the icon that represents them. Default │ │ │ │ │ + * is false. │ │ │ │ │ + */ │ │ │ │ │ + allowDepress: false, │ │ │ │ │ │ │ │ │ │ - //if popup has images, listen for when they finish │ │ │ │ │ - // loading and resize accordingly │ │ │ │ │ - this.registerImageListeners(); │ │ │ │ │ + /** │ │ │ │ │ + * Property: activeState │ │ │ │ │ + * {Object} stores the active state of this panel's controls. │ │ │ │ │ + */ │ │ │ │ │ + activeState: null, │ │ │ │ │ │ │ │ │ │ - //auto size the popup to its current contents │ │ │ │ │ - this.updateSize(); │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.Panel │ │ │ │ │ + * Create a new control panel. │ │ │ │ │ + * │ │ │ │ │ + * Each control in the panel is represented by an icon. When clicking │ │ │ │ │ + * on an icon, the <activateControl> method is called. │ │ │ │ │ + * │ │ │ │ │ + * Specific properties for controls on a panel: │ │ │ │ │ + * type - {Number} One of <OpenLayers.Control.TYPE_TOOL>, │ │ │ │ │ + * <OpenLayers.Control.TYPE_TOGGLE>, <OpenLayers.Control.TYPE_BUTTON>. │ │ │ │ │ + * If not provided, <OpenLayers.Control.TYPE_TOOL> is assumed. │ │ │ │ │ + * title - {string} Text displayed when mouse is over the icon that │ │ │ │ │ + * represents the control. │ │ │ │ │ + * │ │ │ │ │ + * The <OpenLayers.Control.type> of a control determines the behavior when │ │ │ │ │ + * clicking its icon: │ │ │ │ │ + * <OpenLayers.Control.TYPE_TOOL> - The control is activated and other │ │ │ │ │ + * controls of this type in the same panel are deactivated. This is │ │ │ │ │ + * the default type. │ │ │ │ │ + * <OpenLayers.Control.TYPE_TOGGLE> - The active state of the control is │ │ │ │ │ + * toggled. │ │ │ │ │ + * <OpenLayers.Control.TYPE_BUTTON> - The │ │ │ │ │ + * <OpenLayers.Control.Button.trigger> method of the control is called, │ │ │ │ │ + * but its active state is not changed. │ │ │ │ │ + * │ │ │ │ │ + * If a control is <OpenLayers.Control.active>, it will be drawn with the │ │ │ │ │ + * olControl[Name]ItemActive class, otherwise with the │ │ │ │ │ + * olControl[Name]ItemInactive class. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be used │ │ │ │ │ + * to extend the control. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.controls = []; │ │ │ │ │ + this.activeState = {}; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + for (var ctl, i = this.controls.length - 1; i >= 0; i--) { │ │ │ │ │ + ctl = this.controls[i]; │ │ │ │ │ + if (ctl.events) { │ │ │ │ │ + ctl.events.un({ │ │ │ │ │ + activate: this.iconOn, │ │ │ │ │ + deactivate: this.iconOff │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ + ctl.panel_div = null; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ + this.activeState = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: registerImageListeners │ │ │ │ │ - * Called when an image contained by the popup loaded. this function │ │ │ │ │ - * updates the popup size, then unregisters the image load listener. │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ */ │ │ │ │ │ - registerImageListeners: function() { │ │ │ │ │ - │ │ │ │ │ - // As the images load, this function will call updateSize() to │ │ │ │ │ - // resize the popup to fit the content div (which presumably is now │ │ │ │ │ - // bigger than when the image was not loaded). │ │ │ │ │ - // │ │ │ │ │ - // If the 'panMapIfOutOfView' property is set, we will pan the newly │ │ │ │ │ - // resized popup back into view. │ │ │ │ │ - // │ │ │ │ │ - // Note that this function, when called, will have 'popup' and │ │ │ │ │ - // 'img' properties in the context. │ │ │ │ │ - // │ │ │ │ │ - var onImgLoad = function() { │ │ │ │ │ - if (this.popup.id === null) { // this.popup has been destroyed! │ │ │ │ │ - return; │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + control = this.controls[i]; │ │ │ │ │ + if (control === this.defaultControl || │ │ │ │ │ + (this.saveState && this.activeState[control.id])) { │ │ │ │ │ + control.activate(); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.popup.updateSize(); │ │ │ │ │ - │ │ │ │ │ - if (this.popup.visible() && this.popup.panMapIfOutOfView) { │ │ │ │ │ - this.popup.panIntoView(); │ │ │ │ │ + if (this.saveState === true) { │ │ │ │ │ + this.defaultControl = null; │ │ │ │ │ } │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - OpenLayers.Event.stopObserving( │ │ │ │ │ - this.img, "load", this.img._onImgLoad │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - //cycle through the images and if their size is 0x0, that means that │ │ │ │ │ - // they haven't been loaded yet, so we attach the listener, which │ │ │ │ │ - // will fire when the images finish loading and will resize the │ │ │ │ │ - // popup accordingly to its new size. │ │ │ │ │ - var images = this.contentDiv.getElementsByTagName("img"); │ │ │ │ │ - for (var i = 0, len = images.length; i < len; i++) { │ │ │ │ │ - var img = images[i]; │ │ │ │ │ - if (img.width == 0 || img.height == 0) { │ │ │ │ │ - │ │ │ │ │ - var context = { │ │ │ │ │ - 'popup': this, │ │ │ │ │ - 'img': img │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - //expando this function to the image itself before registering │ │ │ │ │ - // it. This way we can easily and properly unregister it. │ │ │ │ │ - img._onImgLoad = OpenLayers.Function.bind(onImgLoad, context); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Event.observe(img, 'load', img._onImgLoad); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + control = this.controls[i]; │ │ │ │ │ + this.activeState[control.id] = control.deactivate(); │ │ │ │ │ } │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getSafeContentSize │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>} Desired size to make the popup. │ │ │ │ │ - * │ │ │ │ │ + * Method: draw │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Size>} A size to make the popup which is neither smaller │ │ │ │ │ - * than the specified minimum size, nor bigger than the maximum │ │ │ │ │ - * size (which is calculated relative to the size of the viewport). │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - getSafeContentSize: function(size) { │ │ │ │ │ - │ │ │ │ │ - var safeContentSize = size.clone(); │ │ │ │ │ - │ │ │ │ │ - // if our contentDiv has a css 'padding' set on it by a stylesheet, we │ │ │ │ │ - // must add that to the desired "size". │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ - var wPadding = contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ - var hPadding = contentDivPadding.top + contentDivPadding.bottom; │ │ │ │ │ - │ │ │ │ │ - // take into account the popup's 'padding' property │ │ │ │ │ - this.fixPadding(); │ │ │ │ │ - wPadding += this.padding.left + this.padding.right; │ │ │ │ │ - hPadding += this.padding.top + this.padding.bottom; │ │ │ │ │ - │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - var closeDivWidth = parseInt(this.closeDiv.style.width); │ │ │ │ │ - wPadding += closeDivWidth + contentDivPadding.right; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // prevent the popup from being smaller than a specified minimal size │ │ │ │ │ - if (this.minSize) { │ │ │ │ │ - safeContentSize.w = Math.max(safeContentSize.w, │ │ │ │ │ - (this.minSize.w - wPadding)); │ │ │ │ │ - safeContentSize.h = Math.max(safeContentSize.h, │ │ │ │ │ - (this.minSize.h - hPadding)); │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (this.outsideViewport) { │ │ │ │ │ + this.events.attachToElement(this.div); │ │ │ │ │ + this.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ + } else { │ │ │ │ │ + this.map.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ } │ │ │ │ │ + this.addControlsToMap(this.controls); │ │ │ │ │ + return this.div; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // prevent the popup from being bigger than a specified maximum size │ │ │ │ │ - if (this.maxSize) { │ │ │ │ │ - safeContentSize.w = Math.min(safeContentSize.w, │ │ │ │ │ - (this.maxSize.w - wPadding)); │ │ │ │ │ - safeContentSize.h = Math.min(safeContentSize.h, │ │ │ │ │ - (this.maxSize.h - hPadding)); │ │ │ │ │ + /** │ │ │ │ │ + * Method: redraw │ │ │ │ │ + */ │ │ │ │ │ + redraw: function() { │ │ │ │ │ + for (var l = this.div.childNodes.length, i = l - 1; i >= 0; i--) { │ │ │ │ │ + this.div.removeChild(this.div.childNodes[i]); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - //make sure the desired size to set doesn't result in a popup that │ │ │ │ │ - // is bigger than the map's viewport. │ │ │ │ │ - // │ │ │ │ │ - if (this.map && this.map.size) { │ │ │ │ │ - │ │ │ │ │ - var extraX = 0, │ │ │ │ │ - extraY = 0; │ │ │ │ │ - if (this.keepInMap && !this.panMapIfOutOfView) { │ │ │ │ │ - var px = this.map.getPixelFromLonLat(this.lonlat); │ │ │ │ │ - switch (this.relativePosition) { │ │ │ │ │ - case "tr": │ │ │ │ │ - extraX = px.x; │ │ │ │ │ - extraY = this.map.size.h - px.y; │ │ │ │ │ - break; │ │ │ │ │ - case "tl": │ │ │ │ │ - extraX = this.map.size.w - px.x; │ │ │ │ │ - extraY = this.map.size.h - px.y; │ │ │ │ │ - break; │ │ │ │ │ - case "bl": │ │ │ │ │ - extraX = this.map.size.w - px.x; │ │ │ │ │ - extraY = px.y; │ │ │ │ │ - break; │ │ │ │ │ - case "br": │ │ │ │ │ - extraX = px.x; │ │ │ │ │ - extraY = px.y; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - extraX = px.x; │ │ │ │ │ - extraY = this.map.size.h - px.y; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ + this.div.innerHTML = ""; │ │ │ │ │ + if (this.active) { │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + this.div.appendChild(this.controls[i].panel_div); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var maxY = this.map.size.h - │ │ │ │ │ - this.map.paddingForPopups.top - │ │ │ │ │ - this.map.paddingForPopups.bottom - │ │ │ │ │ - hPadding - extraY; │ │ │ │ │ - │ │ │ │ │ - var maxX = this.map.size.w - │ │ │ │ │ - this.map.paddingForPopups.left - │ │ │ │ │ - this.map.paddingForPopups.right - │ │ │ │ │ - wPadding - extraX; │ │ │ │ │ - │ │ │ │ │ - safeContentSize.w = Math.min(safeContentSize.w, maxX); │ │ │ │ │ - safeContentSize.h = Math.min(safeContentSize.h, maxY); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return safeContentSize; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getContentDivPadding │ │ │ │ │ - * Glorious, oh glorious hack in order to determine the css 'padding' of │ │ │ │ │ - * the contentDiv. IE/Opera return null here unless we actually add the │ │ │ │ │ - * popup's main 'div' element (which contains contentDiv) to the DOM. │ │ │ │ │ - * So we make it invisible and then add it to the document temporarily. │ │ │ │ │ - * │ │ │ │ │ - * Once we've taken the padding readings we need, we then remove it │ │ │ │ │ - * from the DOM (it will actually get added to the DOM in │ │ │ │ │ - * Map.js's addPopup) │ │ │ │ │ + * APIMethod: activateControl │ │ │ │ │ + * This method is called when the user click on the icon representing a │ │ │ │ │ + * control in the panel. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * control - {<OpenLayers.Control>} │ │ │ │ │ */ │ │ │ │ │ - getContentDivPadding: function() { │ │ │ │ │ - │ │ │ │ │ - //use cached value if we have it │ │ │ │ │ - var contentDivPadding = this._contentDivPadding; │ │ │ │ │ - if (!contentDivPadding) { │ │ │ │ │ - │ │ │ │ │ - if (this.div.parentNode == null) { │ │ │ │ │ - //make the div invisible and add it to the page │ │ │ │ │ - this.div.style.display = "none"; │ │ │ │ │ - document.body.appendChild(this.div); │ │ │ │ │ + activateControl: function(control) { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + if (control.type == OpenLayers.Control.TYPE_BUTTON) { │ │ │ │ │ + control.trigger(); │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (control.type == OpenLayers.Control.TYPE_TOGGLE) { │ │ │ │ │ + if (control.active) { │ │ │ │ │ + control.deactivate(); │ │ │ │ │ + } else { │ │ │ │ │ + control.activate(); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - //read the padding settings from css, put them in an OL.Bounds │ │ │ │ │ - contentDivPadding = new OpenLayers.Bounds( │ │ │ │ │ - OpenLayers.Element.getStyle(this.contentDiv, "padding-left"), │ │ │ │ │ - OpenLayers.Element.getStyle(this.contentDiv, "padding-bottom"), │ │ │ │ │ - OpenLayers.Element.getStyle(this.contentDiv, "padding-right"), │ │ │ │ │ - OpenLayers.Element.getStyle(this.contentDiv, "padding-top") │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - //cache the value │ │ │ │ │ - this._contentDivPadding = contentDivPadding; │ │ │ │ │ - │ │ │ │ │ - if (this.div.parentNode == document.body) { │ │ │ │ │ - //remove the div from the page and make it visible again │ │ │ │ │ - document.body.removeChild(this.div); │ │ │ │ │ - this.div.style.display = ""; │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (this.allowDepress && control.active) { │ │ │ │ │ + control.deactivate(); │ │ │ │ │ + } else { │ │ │ │ │ + var c; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + c = this.controls[i]; │ │ │ │ │ + if (c != control && │ │ │ │ │ + (c.type === OpenLayers.Control.TYPE_TOOL || c.type == null)) { │ │ │ │ │ + c.deactivate(); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + control.activate(); │ │ │ │ │ } │ │ │ │ │ - return contentDivPadding; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: addCloseBox │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: addControls │ │ │ │ │ + * To build a toolbar, you add a set of controls to it. addControls │ │ │ │ │ + * lets you add a single control or a list of controls to the │ │ │ │ │ + * Control Panel. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * callback - {Function} The callback to be called when the close button │ │ │ │ │ - * is clicked. │ │ │ │ │ + * controls - {<OpenLayers.Control>} Controls to add in the panel. │ │ │ │ │ */ │ │ │ │ │ - addCloseBox: function(callback) { │ │ │ │ │ + addControls: function(controls) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(controls))) { │ │ │ │ │ + controls = [controls]; │ │ │ │ │ + } │ │ │ │ │ + this.controls = this.controls.concat(controls); │ │ │ │ │ │ │ │ │ │ - this.closeDiv = OpenLayers.Util.createDiv( │ │ │ │ │ - this.id + "_close", null, { │ │ │ │ │ - w: 17, │ │ │ │ │ - h: 17 │ │ │ │ │ + for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ + var control = controls[i], │ │ │ │ │ + element = this.createControlMarkup(control); │ │ │ │ │ + OpenLayers.Element.addClass(element, │ │ │ │ │ + control.displayClass + "ItemInactive"); │ │ │ │ │ + OpenLayers.Element.addClass(element, "olButton"); │ │ │ │ │ + if (control.title != "" && !element.title) { │ │ │ │ │ + element.title = control.title; │ │ │ │ │ } │ │ │ │ │ - ); │ │ │ │ │ - this.closeDiv.className = "olPopupCloseBox"; │ │ │ │ │ - │ │ │ │ │ - // use the content div's css padding to determine if we should │ │ │ │ │ - // padd the close div │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ - │ │ │ │ │ - this.closeDiv.style.right = contentDivPadding.right + "px"; │ │ │ │ │ - this.closeDiv.style.top = contentDivPadding.top + "px"; │ │ │ │ │ - this.groupDiv.appendChild(this.closeDiv); │ │ │ │ │ + control.panel_div = element; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var closePopup = callback || function(e) { │ │ │ │ │ - this.hide(); │ │ │ │ │ - OpenLayers.Event.stop(e); │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Event.observe(this.closeDiv, "touchend", │ │ │ │ │ - OpenLayers.Function.bindAsEventListener(closePopup, this)); │ │ │ │ │ - OpenLayers.Event.observe(this.closeDiv, "click", │ │ │ │ │ - OpenLayers.Function.bindAsEventListener(closePopup, this)); │ │ │ │ │ + if (this.map) { // map.addControl() has already been called on the panel │ │ │ │ │ + this.addControlsToMap(controls); │ │ │ │ │ + this.redraw(); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: panIntoView │ │ │ │ │ - * Pans the map such that the popup is totaly viewable (if necessary) │ │ │ │ │ + * APIMethod: createControlMarkup │ │ │ │ │ + * This function just creates a div for the control. If specific HTML │ │ │ │ │ + * markup is needed this function can be overridden in specific classes, │ │ │ │ │ + * or at panel instantiation time: │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var panel = new OpenLayers.Control.Panel({ │ │ │ │ │ + * defaultControl: control, │ │ │ │ │ + * // ovverride createControlMarkup to create actual buttons │ │ │ │ │ + * // including texts wrapped into span elements. │ │ │ │ │ + * createControlMarkup: function(control) { │ │ │ │ │ + * var button = document.createElement('button'), │ │ │ │ │ + * span = document.createElement('span'); │ │ │ │ │ + * if (control.text) { │ │ │ │ │ + * span.innerHTML = control.text; │ │ │ │ │ + * } │ │ │ │ │ + * return button; │ │ │ │ │ + * } │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * control - {<OpenLayers.Control>} The control to create the HTML │ │ │ │ │ + * markup for. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The markup. │ │ │ │ │ */ │ │ │ │ │ - panIntoView: function() { │ │ │ │ │ - │ │ │ │ │ - var mapSize = this.map.getSize(); │ │ │ │ │ - │ │ │ │ │ - //start with the top left corner of the popup, in px, │ │ │ │ │ - // relative to the viewport │ │ │ │ │ - var origTL = this.map.getViewPortPxFromLayerPx(new OpenLayers.Pixel( │ │ │ │ │ - parseInt(this.div.style.left), │ │ │ │ │ - parseInt(this.div.style.top) │ │ │ │ │ - )); │ │ │ │ │ - var newTL = origTL.clone(); │ │ │ │ │ - │ │ │ │ │ - //new left (compare to margins, using this.size to calculate right) │ │ │ │ │ - if (origTL.x < this.map.paddingForPopups.left) { │ │ │ │ │ - newTL.x = this.map.paddingForPopups.left; │ │ │ │ │ - } else │ │ │ │ │ - if ((origTL.x + this.size.w) > (mapSize.w - this.map.paddingForPopups.right)) { │ │ │ │ │ - newTL.x = mapSize.w - this.map.paddingForPopups.right - this.size.w; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //new top (compare to margins, using this.size to calculate bottom) │ │ │ │ │ - if (origTL.y < this.map.paddingForPopups.top) { │ │ │ │ │ - newTL.y = this.map.paddingForPopups.top; │ │ │ │ │ - } else │ │ │ │ │ - if ((origTL.y + this.size.h) > (mapSize.h - this.map.paddingForPopups.bottom)) { │ │ │ │ │ - newTL.y = mapSize.h - this.map.paddingForPopups.bottom - this.size.h; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var dx = origTL.x - newTL.x; │ │ │ │ │ - var dy = origTL.y - newTL.y; │ │ │ │ │ - │ │ │ │ │ - this.map.pan(dx, dy); │ │ │ │ │ + createControlMarkup: function(control) { │ │ │ │ │ + return document.createElement("div"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: registerEvents │ │ │ │ │ - * Registers events on the popup. │ │ │ │ │ + /** │ │ │ │ │ + * Method: addControlsToMap │ │ │ │ │ + * Only for internal use in draw() and addControls() methods. │ │ │ │ │ * │ │ │ │ │ - * Do this in a separate function so that subclasses can │ │ │ │ │ - * choose to override it if they wish to deal differently │ │ │ │ │ - * with mouse events │ │ │ │ │ - * │ │ │ │ │ - * Note in the following handler functions that some special │ │ │ │ │ - * care is needed to deal correctly with mousing and popups. │ │ │ │ │ - * │ │ │ │ │ - * Because the user might select the zoom-rectangle option and │ │ │ │ │ - * then drag it over a popup, we need a safe way to allow the │ │ │ │ │ - * mousemove and mouseup events to pass through the popup when │ │ │ │ │ - * they are initiated from outside. The same procedure is needed for │ │ │ │ │ - * touchmove and touchend events. │ │ │ │ │ - * │ │ │ │ │ - * Otherwise, we want to essentially kill the event propagation │ │ │ │ │ - * for all other events, though we have to do so carefully, │ │ │ │ │ - * without disabling basic html functionality, like clicking on │ │ │ │ │ - * hyperlinks or drag-selecting text. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * controls - {Array(<OpenLayers.Control>)} Controls to add into map. │ │ │ │ │ */ │ │ │ │ │ - registerEvents: function() { │ │ │ │ │ - this.events = new OpenLayers.Events(this, this.div, null, true); │ │ │ │ │ - │ │ │ │ │ - function onTouchstart(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt, true); │ │ │ │ │ + addControlsToMap: function(controls) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ + control = controls[i]; │ │ │ │ │ + if (control.autoActivate === true) { │ │ │ │ │ + control.autoActivate = false; │ │ │ │ │ + this.map.addControl(control); │ │ │ │ │ + control.autoActivate = true; │ │ │ │ │ + } else { │ │ │ │ │ + this.map.addControl(control); │ │ │ │ │ + control.deactivate(); │ │ │ │ │ + } │ │ │ │ │ + control.events.on({ │ │ │ │ │ + activate: this.iconOn, │ │ │ │ │ + deactivate: this.iconOff │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - this.events.on({ │ │ │ │ │ - "mousedown": this.onmousedown, │ │ │ │ │ - "mousemove": this.onmousemove, │ │ │ │ │ - "mouseup": this.onmouseup, │ │ │ │ │ - "click": this.onclick, │ │ │ │ │ - "mouseout": this.onmouseout, │ │ │ │ │ - "dblclick": this.ondblclick, │ │ │ │ │ - "touchstart": onTouchstart, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: onmousedown │ │ │ │ │ - * When mouse goes down within the popup, make a note of │ │ │ │ │ - * it locally, and then do not propagate the mousedown │ │ │ │ │ - * (but do so safely so that user can select text inside) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + /** │ │ │ │ │ + * Method: iconOn │ │ │ │ │ + * Internal use, for use only with "controls[i].events.on/un". │ │ │ │ │ */ │ │ │ │ │ - onmousedown: function(evt) { │ │ │ │ │ - this.mousedown = true; │ │ │ │ │ - OpenLayers.Event.stop(evt, true); │ │ │ │ │ + iconOn: function() { │ │ │ │ │ + var d = this.panel_div; // "this" refers to a control on panel! │ │ │ │ │ + var re = new RegExp("\\b(" + this.displayClass + "Item)Inactive\\b"); │ │ │ │ │ + d.className = d.className.replace(re, "$1Active"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: onmousemove │ │ │ │ │ - * If the drag was started within the popup, then │ │ │ │ │ - * do not propagate the mousemove (but do so safely │ │ │ │ │ - * so that user can select text inside) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + /** │ │ │ │ │ + * Method: iconOff │ │ │ │ │ + * Internal use, for use only with "controls[i].events.on/un". │ │ │ │ │ */ │ │ │ │ │ - onmousemove: function(evt) { │ │ │ │ │ - if (this.mousedown) { │ │ │ │ │ - OpenLayers.Event.stop(evt, true); │ │ │ │ │ - } │ │ │ │ │ + iconOff: function() { │ │ │ │ │ + var d = this.panel_div; // "this" refers to a control on panel! │ │ │ │ │ + var re = new RegExp("\\b(" + this.displayClass + "Item)Active\\b"); │ │ │ │ │ + d.className = d.className.replace(re, "$1Inactive"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: onmouseup │ │ │ │ │ - * When mouse comes up within the popup, after going down │ │ │ │ │ - * in it, reset the flag, and then (once again) do not │ │ │ │ │ - * propagate the event, but do so safely so that user can │ │ │ │ │ - * select text inside │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: onButtonClick │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - onmouseup: function(evt) { │ │ │ │ │ - if (this.mousedown) { │ │ │ │ │ - this.mousedown = false; │ │ │ │ │ - OpenLayers.Event.stop(evt, true); │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + var controls = this.controls, │ │ │ │ │ + button = evt.buttonElement; │ │ │ │ │ + for (var i = controls.length - 1; i >= 0; --i) { │ │ │ │ │ + if (controls[i].panel_div === button) { │ │ │ │ │ + this.activateControl(controls[i]); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onclick │ │ │ │ │ - * Ignore clicks, but allowing default browser handling │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getControlsBy │ │ │ │ │ + * Get a list of controls with properties matching the given criteria. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * property - {String} A control property to be matched. │ │ │ │ │ + * match - {String | Object} A string to match. Can also be a regular │ │ │ │ │ + * expression literal or object. In addition, it can be any object │ │ │ │ │ + * with a method named test. For reqular expressions or other, if │ │ │ │ │ + * match.test(control[property]) evaluates to true, the control will be │ │ │ │ │ + * included in the array returned. If no controls are found, an empty │ │ │ │ │ + * array is returned. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Control>)} A list of controls matching the given criteria. │ │ │ │ │ + * An empty array is returned if no matches are found. │ │ │ │ │ */ │ │ │ │ │ - onclick: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt, true); │ │ │ │ │ + getControlsBy: function(property, match) { │ │ │ │ │ + var test = (typeof match.test == "function"); │ │ │ │ │ + var found = OpenLayers.Array.filter(this.controls, function(item) { │ │ │ │ │ + return item[property] == match || (test && match.test(item[property])); │ │ │ │ │ + }); │ │ │ │ │ + return found; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: onmouseout │ │ │ │ │ - * When mouse goes out of the popup set the flag to false so that │ │ │ │ │ - * if they let go and then drag back in, we won't be confused. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getControlsByName │ │ │ │ │ + * Get a list of contorls with names matching the given name. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * match - {String | Object} A control name. The name can also be a regular │ │ │ │ │ + * expression literal or object. In addition, it can be any object │ │ │ │ │ + * with a method named test. For reqular expressions or other, if │ │ │ │ │ + * name.test(control.name) evaluates to true, the control will be included │ │ │ │ │ + * in the list of controls returned. If no controls are found, an empty │ │ │ │ │ + * array is returned. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Control>)} A list of controls matching the given name. │ │ │ │ │ + * An empty array is returned if no matches are found. │ │ │ │ │ */ │ │ │ │ │ - onmouseout: function(evt) { │ │ │ │ │ - this.mousedown = false; │ │ │ │ │ + getControlsByName: function(match) { │ │ │ │ │ + return this.getControlsBy("name", match); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: ondblclick │ │ │ │ │ - * Ignore double-clicks, but allowing default browser handling │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getControlsByClass │ │ │ │ │ + * Get a list of controls of a given type (CLASS_NAME). │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * match - {String | Object} A control class name. The type can also be a │ │ │ │ │ + * regular expression literal or object. In addition, it can be any │ │ │ │ │ + * object with a method named test. For reqular expressions or other, │ │ │ │ │ + * if type.test(control.CLASS_NAME) evaluates to true, the control will │ │ │ │ │ + * be included in the list of controls returned. If no controls are │ │ │ │ │ + * found, an empty array is returned. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Control>)} A list of controls matching the given type. │ │ │ │ │ + * An empty array is returned if no matches are found. │ │ │ │ │ */ │ │ │ │ │ - ondblclick: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt, true); │ │ │ │ │ + getControlsByClass: function(match) { │ │ │ │ │ + return this.getControlsBy("CLASS_NAME", match); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Panel" │ │ │ │ │ }); │ │ │ │ │ │ │ │ │ │ -OpenLayers.Popup.WIDTH = 200; │ │ │ │ │ -OpenLayers.Popup.HEIGHT = 200; │ │ │ │ │ -OpenLayers.Popup.COLOR = "white"; │ │ │ │ │ -OpenLayers.Popup.OPACITY = 1; │ │ │ │ │ -OpenLayers.Popup.BORDER = "0px"; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Popup/Anchored.js │ │ │ │ │ + OpenLayers/Control/LayerSwitcher.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Popup.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Popup.Anchored │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Control.LayerSwitcher │ │ │ │ │ + * The LayerSwitcher control displays a table of contents for the map. This │ │ │ │ │ + * allows the user interface to switch between BaseLasyers and to show or hide │ │ │ │ │ + * Overlays. By default the switcher is shown minimized on the right edge of │ │ │ │ │ + * the map, the user may expand it by clicking on the handle. │ │ │ │ │ + * │ │ │ │ │ + * To create the LayerSwitcher outside of the map, pass the Id of a html div │ │ │ │ │ + * as the first argument to the constructor. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Popup> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Popup.Anchored = │ │ │ │ │ - OpenLayers.Class(OpenLayers.Popup, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: relativePosition │ │ │ │ │ - * {String} Relative position of the popup ("br", "tr", "tl" or "bl"). │ │ │ │ │ - */ │ │ │ │ │ - relativePosition: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: keepInMap │ │ │ │ │ - * {Boolean} If panMapIfOutOfView is false, and this property is true, │ │ │ │ │ - * contrain the popup such that it always fits in the available map │ │ │ │ │ - * space. By default, this is set. If you are creating popups that are │ │ │ │ │ - * near map edges and not allowing pannning, and especially if you have │ │ │ │ │ - * a popup which has a fixedRelativePosition, setting this to false may │ │ │ │ │ - * be a smart thing to do. │ │ │ │ │ - * │ │ │ │ │ - * For anchored popups, default is true, since subclasses will │ │ │ │ │ - * usually want this functionality. │ │ │ │ │ - */ │ │ │ │ │ - keepInMap: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: anchor │ │ │ │ │ - * {Object} Object to which we'll anchor the popup. Must expose a │ │ │ │ │ - * 'size' (<OpenLayers.Size>) and 'offset' (<OpenLayers.Pixel>). │ │ │ │ │ - */ │ │ │ │ │ - anchor: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Popup.Anchored │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ - * contentSize - {<OpenLayers.Size>} │ │ │ │ │ - * contentHTML - {String} │ │ │ │ │ - * anchor - {Object} Object which must expose a 'size' <OpenLayers.Size> │ │ │ │ │ - * and 'offset' <OpenLayers.Pixel> (generally an <OpenLayers.Icon>). │ │ │ │ │ - * closeBox - {Boolean} │ │ │ │ │ - * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, │ │ │ │ │ - closeBoxCallback) { │ │ │ │ │ - var newArguments = [ │ │ │ │ │ - id, lonlat, contentSize, contentHTML, closeBox, closeBoxCallback │ │ │ │ │ - ]; │ │ │ │ │ - OpenLayers.Popup.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - │ │ │ │ │ - this.anchor = (anchor != null) ? anchor : │ │ │ │ │ - { │ │ │ │ │ - size: new OpenLayers.Size(0, 0), │ │ │ │ │ - offset: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.anchor = null; │ │ │ │ │ - this.relativePosition = null; │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Popup.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: show │ │ │ │ │ - * Overridden from Popup since user might hide popup and then show() it │ │ │ │ │ - * in a new location (meaning we might want to update the relative │ │ │ │ │ - * position on the show) │ │ │ │ │ - */ │ │ │ │ │ - show: function() { │ │ │ │ │ - this.updatePosition(); │ │ │ │ │ - OpenLayers.Popup.prototype.show.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * Since the popup is moving to a new px, it might need also to be moved │ │ │ │ │ - * relative to where the marker is. We first calculate the new │ │ │ │ │ - * relativePosition, and then we calculate the new px where we will │ │ │ │ │ - * put the popup, based on the new relative position. │ │ │ │ │ - * │ │ │ │ │ - * If the relativePosition has changed, we must also call │ │ │ │ │ - * updateRelativePosition() to make any visual changes to the popup │ │ │ │ │ - * which are associated with putting it in a new relativePosition. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - */ │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - var oldRelativePosition = this.relativePosition; │ │ │ │ │ - this.relativePosition = this.calculateRelativePosition(px); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Popup.prototype.moveTo.call(this, this.calculateNewPx(px)); │ │ │ │ │ +OpenLayers.Control.LayerSwitcher = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - //if this move has caused the popup to change its relative position, │ │ │ │ │ - // we need to make the appropriate cosmetic changes. │ │ │ │ │ - if (this.relativePosition != oldRelativePosition) { │ │ │ │ │ - this.updateRelativePosition(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: layerStates │ │ │ │ │ + * {Array(Object)} Basically a copy of the "state" of the map's layers │ │ │ │ │ + * the last time the control was drawn. We have this in order to avoid │ │ │ │ │ + * unnecessarily redrawing the control. │ │ │ │ │ + */ │ │ │ │ │ + layerStates: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setSize │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * contentSize - {<OpenLayers.Size>} the new size for the popup's │ │ │ │ │ - * contents div (in pixels). │ │ │ │ │ - */ │ │ │ │ │ - setSize: function(contentSize) { │ │ │ │ │ - OpenLayers.Popup.prototype.setSize.apply(this, arguments); │ │ │ │ │ + // DOM Elements │ │ │ │ │ │ │ │ │ │ - if ((this.lonlat) && (this.map)) { │ │ │ │ │ - var px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ - this.moveTo(px); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: layersDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + layersDiv: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateRelativePosition │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The relative position ("br" "tr" "tl" "bl") at which the popup │ │ │ │ │ - * should be placed. │ │ │ │ │ - */ │ │ │ │ │ - calculateRelativePosition: function(px) { │ │ │ │ │ - var lonlat = this.map.getLonLatFromLayerPx(px); │ │ │ │ │ + /** │ │ │ │ │ + * Property: baseLayersDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + baseLayersDiv: null, │ │ │ │ │ │ │ │ │ │ - var extent = this.map.getExtent(); │ │ │ │ │ - var quadrant = extent.determineQuadrant(lonlat); │ │ │ │ │ + /** │ │ │ │ │ + * Property: baseLayers │ │ │ │ │ + * {Array(Object)} │ │ │ │ │ + */ │ │ │ │ │ + baseLayers: null, │ │ │ │ │ │ │ │ │ │ - return OpenLayers.Bounds.oppositeQuadrant(quadrant); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateRelativePosition │ │ │ │ │ - * The popup has been moved to a new relative location, so we may want to │ │ │ │ │ - * make some cosmetic adjustments to it. │ │ │ │ │ - * │ │ │ │ │ - * Note that in the classic Anchored popup, there is nothing to do │ │ │ │ │ - * here, since the popup looks exactly the same in all four positions. │ │ │ │ │ - * Subclasses such as Framed, however, will want to do something │ │ │ │ │ - * special here. │ │ │ │ │ - */ │ │ │ │ │ - updateRelativePosition: function() { │ │ │ │ │ - //to be overridden by subclasses │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: dataLbl │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + dataLbl: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateNewPx │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Pixel>} The the new px position of the popup on the screen │ │ │ │ │ - * relative to the passed-in px. │ │ │ │ │ - */ │ │ │ │ │ - calculateNewPx: function(px) { │ │ │ │ │ - var newPx = px.offset(this.anchor.offset); │ │ │ │ │ + /** │ │ │ │ │ + * Property: dataLayersDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + dataLayersDiv: null, │ │ │ │ │ │ │ │ │ │ - //use contentSize if size is not already set │ │ │ │ │ - var size = this.size || this.contentSize; │ │ │ │ │ + /** │ │ │ │ │ + * Property: dataLayers │ │ │ │ │ + * {Array(Object)} │ │ │ │ │ + */ │ │ │ │ │ + dataLayers: null, │ │ │ │ │ │ │ │ │ │ - var top = (this.relativePosition.charAt(0) == 't'); │ │ │ │ │ - newPx.y += (top) ? -size.h : this.anchor.size.h; │ │ │ │ │ │ │ │ │ │ - var left = (this.relativePosition.charAt(1) == 'l'); │ │ │ │ │ - newPx.x += (left) ? -size.w : this.anchor.size.w; │ │ │ │ │ + /** │ │ │ │ │ + * Property: minimizeDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + minimizeDiv: null, │ │ │ │ │ │ │ │ │ │ - return newPx; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: maximizeDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + maximizeDiv: null, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup.Anchored" │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Popup/Framed.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: ascending │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + ascending: true, │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.LayerSwitcher │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.layerStates = []; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Popup/Anchored.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Popup.Framed │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Popup.Anchored> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Popup.Framed = │ │ │ │ │ - OpenLayers.Class(OpenLayers.Popup.Anchored, { │ │ │ │ │ + //clear out layers info and unregister their events │ │ │ │ │ + this.clearLayersArray("base"); │ │ │ │ │ + this.clearLayersArray("data"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: imageSrc │ │ │ │ │ - * {String} location of the image to be used as the popup frame │ │ │ │ │ - */ │ │ │ │ │ - imageSrc: null, │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + buttonclick: this.onButtonClick, │ │ │ │ │ + addlayer: this.redraw, │ │ │ │ │ + changelayer: this.redraw, │ │ │ │ │ + removelayer: this.redraw, │ │ │ │ │ + changebaselayer: this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: imageSize │ │ │ │ │ - * {<OpenLayers.Size>} Size (measured in pixels) of the image located │ │ │ │ │ - * by the 'imageSrc' property. │ │ │ │ │ - */ │ │ │ │ │ - imageSize: null, │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isAlphaImage │ │ │ │ │ - * {Boolean} The image has some alpha and thus needs to use the alpha │ │ │ │ │ - * image hack. Note that setting this to true will have no noticeable │ │ │ │ │ - * effect in FF or IE7 browsers, but will all but crush the ie6 │ │ │ │ │ - * browser. │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - isAlphaImage: false, │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * │ │ │ │ │ + * Properties: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: positionBlocks │ │ │ │ │ - * {Object} Hash of different position blocks (Object/Hashs). Each block │ │ │ │ │ - * will be keyed by a two-character 'relativePosition' │ │ │ │ │ - * code string (ie "tl", "tr", "bl", "br"). Block properties are │ │ │ │ │ - * 'offset', 'padding' (self-explanatory), and finally the 'blocks' │ │ │ │ │ - * parameter, which is an array of the block objects. │ │ │ │ │ - * │ │ │ │ │ - * Each block object must have 'size', 'anchor', and 'position' │ │ │ │ │ - * properties. │ │ │ │ │ - * │ │ │ │ │ - * Note that positionBlocks should never be modified at runtime. │ │ │ │ │ - */ │ │ │ │ │ - positionBlocks: null, │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + addlayer: this.redraw, │ │ │ │ │ + changelayer: this.redraw, │ │ │ │ │ + removelayer: this.redraw, │ │ │ │ │ + changebaselayer: this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + if (this.outsideViewport) { │ │ │ │ │ + this.events.attachToElement(this.div); │ │ │ │ │ + this.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ + } else { │ │ │ │ │ + this.map.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: blocks │ │ │ │ │ - * {Array[Object]} Array of objects, each of which is one "block" of the │ │ │ │ │ - * popup. Each block has a 'div' and an 'image' property, both of │ │ │ │ │ - * which are DOMElements, and the latter of which is appended to the │ │ │ │ │ - * former. These are reused as the popup goes changing positions for │ │ │ │ │ - * great economy and elegance. │ │ │ │ │ - */ │ │ │ │ │ - blocks: null, │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A reference to the DIV DOMElement containing the │ │ │ │ │ + * switcher tabs. │ │ │ │ │ + */ │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: fixedRelativePosition │ │ │ │ │ - * {Boolean} We want the framed popup to work dynamically placed relative │ │ │ │ │ - * to its anchor but also in just one fixed position. A well designed │ │ │ │ │ - * framed popup will have the pixels and logic to display itself in │ │ │ │ │ - * any of the four relative positions, but (understandably), this will │ │ │ │ │ - * not be the case for all of them. By setting this property to 'true', │ │ │ │ │ - * framed popup will not recalculate for the best placement each time │ │ │ │ │ - * it's open, but will always open the same way. │ │ │ │ │ - * Note that if this is set to true, it is generally advisable to also │ │ │ │ │ - * set the 'panIntoView' property to true so that the popup can be │ │ │ │ │ - * scrolled into view (since it will often be offscreen on open) │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - fixedRelativePosition: false, │ │ │ │ │ + // create layout divs │ │ │ │ │ + this.loadContents(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Popup.Framed │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ - * contentSize - {<OpenLayers.Size>} │ │ │ │ │ - * contentHTML - {String} │ │ │ │ │ - * anchor - {Object} Object to which we'll anchor the popup. Must expose │ │ │ │ │ - * a 'size' (<OpenLayers.Size>) and 'offset' (<OpenLayers.Pixel>) │ │ │ │ │ - * (Note that this is generally an <OpenLayers.Icon>). │ │ │ │ │ - * closeBox - {Boolean} │ │ │ │ │ - * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, │ │ │ │ │ - closeBoxCallback) { │ │ │ │ │ + // set mode to minimize │ │ │ │ │ + if (!this.outsideViewport) { │ │ │ │ │ + this.minimizeControl(); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.initialize.apply(this, arguments); │ │ │ │ │ + // populate div with current info │ │ │ │ │ + this.redraw(); │ │ │ │ │ │ │ │ │ │ - if (this.fixedRelativePosition) { │ │ │ │ │ - //based on our decided relativePostion, set the current padding │ │ │ │ │ - // this keeps us from getting into trouble │ │ │ │ │ - this.updateRelativePosition(); │ │ │ │ │ + return this.div; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - //make calculateRelativePosition always return the specified │ │ │ │ │ - // fixed position. │ │ │ │ │ - this.calculateRelativePosition = function(px) { │ │ │ │ │ - return this.relativePosition; │ │ │ │ │ - }; │ │ │ │ │ + /** │ │ │ │ │ + * Method: onButtonClick │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + var button = evt.buttonElement; │ │ │ │ │ + if (button === this.minimizeDiv) { │ │ │ │ │ + this.minimizeControl(); │ │ │ │ │ + } else if (button === this.maximizeDiv) { │ │ │ │ │ + this.maximizeControl(); │ │ │ │ │ + } else if (button._layerSwitcher === this.id) { │ │ │ │ │ + if (button["for"]) { │ │ │ │ │ + button = document.getElementById(button["for"]); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - this.contentDiv.style.position = "absolute"; │ │ │ │ │ - this.contentDiv.style.zIndex = 1; │ │ │ │ │ - │ │ │ │ │ - if (closeBox) { │ │ │ │ │ - this.closeDiv.style.zIndex = 1; │ │ │ │ │ + if (!button.disabled) { │ │ │ │ │ + if (button.type == "radio") { │ │ │ │ │ + button.checked = true; │ │ │ │ │ + this.map.setBaseLayer(this.map.getLayer(button._layer)); │ │ │ │ │ + } else { │ │ │ │ │ + button.checked = !button.checked; │ │ │ │ │ + this.updateMap(); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.groupDiv.style.position = "absolute"; │ │ │ │ │ - this.groupDiv.style.top = "0px"; │ │ │ │ │ - this.groupDiv.style.left = "0px"; │ │ │ │ │ - this.groupDiv.style.height = "100%"; │ │ │ │ │ - this.groupDiv.style.width = "100%"; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.imageSrc = null; │ │ │ │ │ - this.imageSize = null; │ │ │ │ │ - this.isAlphaImage = null; │ │ │ │ │ - │ │ │ │ │ - this.fixedRelativePosition = false; │ │ │ │ │ - this.positionBlocks = null; │ │ │ │ │ + /** │ │ │ │ │ + * Method: clearLayersArray │ │ │ │ │ + * User specifies either "base" or "data". we then clear all the │ │ │ │ │ + * corresponding listeners, the div, and reinitialize a new array. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layersType - {String} │ │ │ │ │ + */ │ │ │ │ │ + clearLayersArray: function(layersType) { │ │ │ │ │ + this[layersType + "LayersDiv"].innerHTML = ""; │ │ │ │ │ + this[layersType + "Layers"] = []; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - //remove our blocks │ │ │ │ │ - for (var i = 0; i < this.blocks.length; i++) { │ │ │ │ │ - var block = this.blocks[i]; │ │ │ │ │ │ │ │ │ │ - if (block.image) { │ │ │ │ │ - block.div.removeChild(block.image); │ │ │ │ │ - } │ │ │ │ │ - block.image = null; │ │ │ │ │ + /** │ │ │ │ │ + * Method: checkRedraw │ │ │ │ │ + * Checks if the layer state has changed since the last redraw() call. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The layer state changed since the last redraw() call. │ │ │ │ │ + */ │ │ │ │ │ + checkRedraw: function() { │ │ │ │ │ + if (!this.layerStates.length || │ │ │ │ │ + (this.map.layers.length != this.layerStates.length)) { │ │ │ │ │ + return true; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (block.div) { │ │ │ │ │ - this.groupDiv.removeChild(block.div); │ │ │ │ │ - } │ │ │ │ │ - block.div = null; │ │ │ │ │ + for (var i = 0, len = this.layerStates.length; i < len; i++) { │ │ │ │ │ + var layerState = this.layerStates[i]; │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if ((layerState.name != layer.name) || │ │ │ │ │ + (layerState.inRange != layer.inRange) || │ │ │ │ │ + (layerState.id != layer.id) || │ │ │ │ │ + (layerState.visibility != layer.visibility)) { │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ - this.blocks = null; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setBackgroundColor │ │ │ │ │ - */ │ │ │ │ │ - setBackgroundColor: function(color) { │ │ │ │ │ - //does nothing since the framed popup's entire scheme is based on a │ │ │ │ │ - // an image -- changing the background color makes no sense. │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: redraw │ │ │ │ │ + * Goes through and takes the current state of the Map and rebuilds the │ │ │ │ │ + * control to display that state. Groups base layers into a │ │ │ │ │ + * radio-button group and lists each data layer with a checkbox. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ + */ │ │ │ │ │ + redraw: function() { │ │ │ │ │ + //if the state hasn't changed since last redraw, no need │ │ │ │ │ + // to do anything. Just return the existing div. │ │ │ │ │ + if (!this.checkRedraw()) { │ │ │ │ │ + return this.div; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setBorder │ │ │ │ │ - */ │ │ │ │ │ - setBorder: function() { │ │ │ │ │ - //does nothing since the framed popup's entire scheme is based on a │ │ │ │ │ - // an image -- changing the popup's border makes no sense. │ │ │ │ │ - }, │ │ │ │ │ + //clear out previous layers │ │ │ │ │ + this.clearLayersArray("base"); │ │ │ │ │ + this.clearLayersArray("data"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setOpacity │ │ │ │ │ - * Sets the opacity of the popup. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * opacity - {float} A value between 0.0 (transparent) and 1.0 (solid). │ │ │ │ │ - */ │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - //does nothing since we suppose that we'll never apply an opacity │ │ │ │ │ - // to a framed popup │ │ │ │ │ - }, │ │ │ │ │ + var containsOverlays = false; │ │ │ │ │ + var containsBaseLayers = false; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setSize │ │ │ │ │ - * Overridden here, because we need to update the blocks whenever the size │ │ │ │ │ - * of the popup has changed. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * contentSize - {<OpenLayers.Size>} the new size for the popup's │ │ │ │ │ - * contents div (in pixels). │ │ │ │ │ - */ │ │ │ │ │ - setSize: function(contentSize) { │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.setSize.apply(this, arguments); │ │ │ │ │ + // Save state -- for checking layer if the map state changed. │ │ │ │ │ + // We save this before redrawing, because in the process of redrawing │ │ │ │ │ + // we will trigger more visibility changes, and we want to not redraw │ │ │ │ │ + // and enter an infinite loop. │ │ │ │ │ + var len = this.map.layers.length; │ │ │ │ │ + this.layerStates = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + this.layerStates[i] = { │ │ │ │ │ + 'name': layer.name, │ │ │ │ │ + 'visibility': layer.visibility, │ │ │ │ │ + 'inRange': layer.inRange, │ │ │ │ │ + 'id': layer.id │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - this.updateBlocks(); │ │ │ │ │ - }, │ │ │ │ │ + var layers = this.map.layers.slice(); │ │ │ │ │ + if (!this.ascending) { │ │ │ │ │ + layers.reverse(); │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + var layer = layers[i]; │ │ │ │ │ + var baseLayer = layer.isBaseLayer; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateRelativePosition │ │ │ │ │ - * When the relative position changes, we need to set the new padding │ │ │ │ │ - * BBOX on the popup, reposition the close div, and update the blocks. │ │ │ │ │ - */ │ │ │ │ │ - updateRelativePosition: function() { │ │ │ │ │ + if (layer.displayInLayerSwitcher) { │ │ │ │ │ │ │ │ │ │ - //update the padding │ │ │ │ │ - this.padding = this.positionBlocks[this.relativePosition].padding; │ │ │ │ │ + if (baseLayer) { │ │ │ │ │ + containsBaseLayers = true; │ │ │ │ │ + } else { │ │ │ │ │ + containsOverlays = true; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - //update the position of our close box to new padding │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - // use the content div's css padding to determine if we should │ │ │ │ │ - // padd the close div │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ + // only check a baselayer if it is *the* baselayer, check data │ │ │ │ │ + // layers if they are visible │ │ │ │ │ + var checked = (baseLayer) ? (layer == this.map.baseLayer) : │ │ │ │ │ + layer.getVisibility(); │ │ │ │ │ │ │ │ │ │ - this.closeDiv.style.right = contentDivPadding.right + │ │ │ │ │ - this.padding.right + "px"; │ │ │ │ │ - this.closeDiv.style.top = contentDivPadding.top + │ │ │ │ │ - this.padding.top + "px"; │ │ │ │ │ - } │ │ │ │ │ + // create input element │ │ │ │ │ + var inputElem = document.createElement("input"), │ │ │ │ │ + // The input shall have an id attribute so we can use │ │ │ │ │ + // labels to interact with them. │ │ │ │ │ + inputId = OpenLayers.Util.createUniqueID( │ │ │ │ │ + this.id + "_input_" │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - this.updateBlocks(); │ │ │ │ │ - }, │ │ │ │ │ + inputElem.id = inputId; │ │ │ │ │ + inputElem.name = (baseLayer) ? this.id + "_baseLayers" : layer.name; │ │ │ │ │ + inputElem.type = (baseLayer) ? "radio" : "checkbox"; │ │ │ │ │ + inputElem.value = layer.name; │ │ │ │ │ + inputElem.checked = checked; │ │ │ │ │ + inputElem.defaultChecked = checked; │ │ │ │ │ + inputElem.className = "olButton"; │ │ │ │ │ + inputElem._layer = layer.id; │ │ │ │ │ + inputElem._layerSwitcher = this.id; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateNewPx │ │ │ │ │ - * Besides the standard offset as determined by the Anchored class, our │ │ │ │ │ - * Framed popups have a special 'offset' property for each of their │ │ │ │ │ - * positions, which is used to offset the popup relative to its anchor. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Pixel>} The the new px position of the popup on the screen │ │ │ │ │ - * relative to the passed-in px. │ │ │ │ │ - */ │ │ │ │ │ - calculateNewPx: function(px) { │ │ │ │ │ - var newPx = OpenLayers.Popup.Anchored.prototype.calculateNewPx.apply( │ │ │ │ │ - this, arguments │ │ │ │ │ - ); │ │ │ │ │ + if (!baseLayer && !layer.inRange) { │ │ │ │ │ + inputElem.disabled = true; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - newPx = newPx.offset(this.positionBlocks[this.relativePosition].offset); │ │ │ │ │ + // create span │ │ │ │ │ + var labelSpan = document.createElement("label"); │ │ │ │ │ + // this isn't the DOM attribute 'for', but an arbitrary name we │ │ │ │ │ + // use to find the appropriate input element in <onButtonClick> │ │ │ │ │ + labelSpan["for"] = inputElem.id; │ │ │ │ │ + OpenLayers.Element.addClass(labelSpan, "labelSpan olButton"); │ │ │ │ │ + labelSpan._layer = layer.id; │ │ │ │ │ + labelSpan._layerSwitcher = this.id; │ │ │ │ │ + if (!baseLayer && !layer.inRange) { │ │ │ │ │ + labelSpan.style.color = "gray"; │ │ │ │ │ + } │ │ │ │ │ + labelSpan.innerHTML = layer.name; │ │ │ │ │ + labelSpan.style.verticalAlign = (baseLayer) ? "bottom" : │ │ │ │ │ + "baseline"; │ │ │ │ │ + // create line break │ │ │ │ │ + var br = document.createElement("br"); │ │ │ │ │ │ │ │ │ │ - return newPx; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createBlocks │ │ │ │ │ - */ │ │ │ │ │ - createBlocks: function() { │ │ │ │ │ - this.blocks = []; │ │ │ │ │ + var groupArray = (baseLayer) ? this.baseLayers : │ │ │ │ │ + this.dataLayers; │ │ │ │ │ + groupArray.push({ │ │ │ │ │ + 'layer': layer, │ │ │ │ │ + 'inputElem': inputElem, │ │ │ │ │ + 'labelSpan': labelSpan │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - //since all positions contain the same number of blocks, we can │ │ │ │ │ - // just pick the first position and use its blocks array to create │ │ │ │ │ - // our blocks array │ │ │ │ │ - var firstPosition = null; │ │ │ │ │ - for (var key in this.positionBlocks) { │ │ │ │ │ - firstPosition = key; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - var position = this.positionBlocks[firstPosition]; │ │ │ │ │ - for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ + var groupDiv = (baseLayer) ? this.baseLayersDiv : │ │ │ │ │ + this.dataLayersDiv; │ │ │ │ │ + groupDiv.appendChild(inputElem); │ │ │ │ │ + groupDiv.appendChild(labelSpan); │ │ │ │ │ + groupDiv.appendChild(br); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var block = {}; │ │ │ │ │ - this.blocks.push(block); │ │ │ │ │ + // if no overlays, dont display the overlay label │ │ │ │ │ + this.dataLbl.style.display = (containsOverlays) ? "" : "none"; │ │ │ │ │ │ │ │ │ │ - var divId = this.id + '_FrameDecorationDiv_' + i; │ │ │ │ │ - block.div = OpenLayers.Util.createDiv(divId, │ │ │ │ │ - null, null, null, "absolute", null, "hidden", null │ │ │ │ │ - ); │ │ │ │ │ + // if no baselayers, dont display the baselayer label │ │ │ │ │ + this.baseLbl.style.display = (containsBaseLayers) ? "" : "none"; │ │ │ │ │ │ │ │ │ │ - var imgId = this.id + '_FrameDecorationImg_' + i; │ │ │ │ │ - var imageCreator = │ │ │ │ │ - (this.isAlphaImage) ? OpenLayers.Util.createAlphaImageDiv : │ │ │ │ │ - OpenLayers.Util.createImage; │ │ │ │ │ + return this.div; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - block.image = imageCreator(imgId, │ │ │ │ │ - null, this.imageSize, this.imageSrc, │ │ │ │ │ - "absolute", null, null, null │ │ │ │ │ - ); │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateMap │ │ │ │ │ + * Cycles through the loaded data and base layer input arrays and makes │ │ │ │ │ + * the necessary calls to the Map object such that that the map's │ │ │ │ │ + * visual state corresponds to what the user has selected in │ │ │ │ │ + * the control. │ │ │ │ │ + */ │ │ │ │ │ + updateMap: function() { │ │ │ │ │ │ │ │ │ │ - block.div.appendChild(block.image); │ │ │ │ │ - this.groupDiv.appendChild(block.div); │ │ │ │ │ + // set the newly selected base layer │ │ │ │ │ + for (var i = 0, len = this.baseLayers.length; i < len; i++) { │ │ │ │ │ + var layerEntry = this.baseLayers[i]; │ │ │ │ │ + if (layerEntry.inputElem.checked) { │ │ │ │ │ + this.map.setBaseLayer(layerEntry.layer, false); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateBlocks │ │ │ │ │ - * Internal method, called on initialize and when the popup's relative │ │ │ │ │ - * position has changed. This function takes care of re-positioning │ │ │ │ │ - * the popup's blocks in their appropropriate places. │ │ │ │ │ - */ │ │ │ │ │ - updateBlocks: function() { │ │ │ │ │ - if (!this.blocks) { │ │ │ │ │ - this.createBlocks(); │ │ │ │ │ - } │ │ │ │ │ + // set the correct visibilities for the overlays │ │ │ │ │ + for (var i = 0, len = this.dataLayers.length; i < len; i++) { │ │ │ │ │ + var layerEntry = this.dataLayers[i]; │ │ │ │ │ + layerEntry.layer.setVisibility(layerEntry.inputElem.checked); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (this.size && this.relativePosition) { │ │ │ │ │ - var position = this.positionBlocks[this.relativePosition]; │ │ │ │ │ - for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var positionBlock = position.blocks[i]; │ │ │ │ │ - var block = this.blocks[i]; │ │ │ │ │ + /** │ │ │ │ │ + * Method: maximizeControl │ │ │ │ │ + * Set up the labels and divs for the control │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * e - {Event} │ │ │ │ │ + */ │ │ │ │ │ + maximizeControl: function(e) { │ │ │ │ │ │ │ │ │ │ - // adjust sizes │ │ │ │ │ - var l = positionBlock.anchor.left; │ │ │ │ │ - var b = positionBlock.anchor.bottom; │ │ │ │ │ - var r = positionBlock.anchor.right; │ │ │ │ │ - var t = positionBlock.anchor.top; │ │ │ │ │ + // set the div's width and height to empty values, so │ │ │ │ │ + // the div dimensions can be controlled by CSS │ │ │ │ │ + this.div.style.width = ""; │ │ │ │ │ + this.div.style.height = ""; │ │ │ │ │ │ │ │ │ │ - //note that we use the isNaN() test here because if the │ │ │ │ │ - // size object is initialized with a "auto" parameter, the │ │ │ │ │ - // size constructor calls parseFloat() on the string, │ │ │ │ │ - // which will turn it into NaN │ │ │ │ │ - // │ │ │ │ │ - var w = (isNaN(positionBlock.size.w)) ? this.size.w - (r + l) : │ │ │ │ │ - positionBlock.size.w; │ │ │ │ │ + this.showControls(false); │ │ │ │ │ │ │ │ │ │ - var h = (isNaN(positionBlock.size.h)) ? this.size.h - (b + t) : │ │ │ │ │ - positionBlock.size.h; │ │ │ │ │ + if (e != null) { │ │ │ │ │ + OpenLayers.Event.stop(e); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - block.div.style.width = (w < 0 ? 0 : w) + 'px'; │ │ │ │ │ - block.div.style.height = (h < 0 ? 0 : h) + 'px'; │ │ │ │ │ + /** │ │ │ │ │ + * Method: minimizeControl │ │ │ │ │ + * Hide all the contents of the control, shrink the size, │ │ │ │ │ + * add the maximize icon │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * e - {Event} │ │ │ │ │ + */ │ │ │ │ │ + minimizeControl: function(e) { │ │ │ │ │ │ │ │ │ │ - block.div.style.left = (l != null) ? l + 'px' : ''; │ │ │ │ │ - block.div.style.bottom = (b != null) ? b + 'px' : ''; │ │ │ │ │ - block.div.style.right = (r != null) ? r + 'px' : ''; │ │ │ │ │ - block.div.style.top = (t != null) ? t + 'px' : ''; │ │ │ │ │ + // to minimize the control we set its div's width │ │ │ │ │ + // and height to 0px, we cannot just set "display" │ │ │ │ │ + // to "none" because it would hide the maximize │ │ │ │ │ + // div │ │ │ │ │ + this.div.style.width = "0px"; │ │ │ │ │ + this.div.style.height = "0px"; │ │ │ │ │ │ │ │ │ │ - block.image.style.left = positionBlock.position.x + 'px'; │ │ │ │ │ - block.image.style.top = positionBlock.position.y + 'px'; │ │ │ │ │ - } │ │ │ │ │ + this.showControls(true); │ │ │ │ │ │ │ │ │ │ - this.contentDiv.style.left = this.padding.left + "px"; │ │ │ │ │ - this.contentDiv.style.top = this.padding.top + "px"; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + if (e != null) { │ │ │ │ │ + OpenLayers.Event.stop(e); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup.Framed" │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Popup/FramedCloud.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: showControls │ │ │ │ │ + * Hide/Show all LayerSwitcher controls depending on whether we are │ │ │ │ │ + * minimized or not │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * minimize - {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + showControls: function(minimize) { │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + this.maximizeDiv.style.display = minimize ? "" : "none"; │ │ │ │ │ + this.minimizeDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Popup/Framed.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Bounds.js │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Pixel.js │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Size.js │ │ │ │ │ - */ │ │ │ │ │ + this.layersDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Popup.FramedCloud │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Popup.Framed> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Popup.FramedCloud = │ │ │ │ │ - OpenLayers.Class(OpenLayers.Popup.Framed, { │ │ │ │ │ + /** │ │ │ │ │ + * Method: loadContents │ │ │ │ │ + * Set up the labels and divs for the control │ │ │ │ │ + */ │ │ │ │ │ + loadContents: function() { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: contentDisplayClass │ │ │ │ │ - * {String} The CSS class of the popup content div. │ │ │ │ │ - */ │ │ │ │ │ - contentDisplayClass: "olFramedCloudPopupContent", │ │ │ │ │ + // layers list div │ │ │ │ │ + this.layersDiv = document.createElement("div"); │ │ │ │ │ + this.layersDiv.id = this.id + "_layersDiv"; │ │ │ │ │ + OpenLayers.Element.addClass(this.layersDiv, "layersDiv"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoSize │ │ │ │ │ - * {Boolean} Framed Cloud is autosizing by default. │ │ │ │ │ - */ │ │ │ │ │ - autoSize: true, │ │ │ │ │ + this.baseLbl = document.createElement("div"); │ │ │ │ │ + this.baseLbl.innerHTML = OpenLayers.i18n("Base Layer"); │ │ │ │ │ + OpenLayers.Element.addClass(this.baseLbl, "baseLbl"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: panMapIfOutOfView │ │ │ │ │ - * {Boolean} Framed Cloud does pan into view by default. │ │ │ │ │ - */ │ │ │ │ │ - panMapIfOutOfView: true, │ │ │ │ │ + this.baseLayersDiv = document.createElement("div"); │ │ │ │ │ + OpenLayers.Element.addClass(this.baseLayersDiv, "baseLayersDiv"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: imageSize │ │ │ │ │ - * {<OpenLayers.Size>} │ │ │ │ │ - */ │ │ │ │ │ - imageSize: new OpenLayers.Size(1276, 736), │ │ │ │ │ + this.dataLbl = document.createElement("div"); │ │ │ │ │ + this.dataLbl.innerHTML = OpenLayers.i18n("Overlays"); │ │ │ │ │ + OpenLayers.Element.addClass(this.dataLbl, "dataLbl"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isAlphaImage │ │ │ │ │ - * {Boolean} The FramedCloud does not use an alpha image (in honor of the │ │ │ │ │ - * good ie6 folk out there) │ │ │ │ │ - */ │ │ │ │ │ - isAlphaImage: false, │ │ │ │ │ + this.dataLayersDiv = document.createElement("div"); │ │ │ │ │ + OpenLayers.Element.addClass(this.dataLayersDiv, "dataLayersDiv"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: fixedRelativePosition │ │ │ │ │ - * {Boolean} The Framed Cloud popup works in just one fixed position. │ │ │ │ │ - */ │ │ │ │ │ - fixedRelativePosition: false, │ │ │ │ │ + if (this.ascending) { │ │ │ │ │ + this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ + } else { │ │ │ │ │ + this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: positionBlocks │ │ │ │ │ - * {Object} Hash of differen position blocks, keyed by relativePosition │ │ │ │ │ - * two-character code string (ie "tl", "tr", "bl", "br") │ │ │ │ │ - */ │ │ │ │ │ - positionBlocks: { │ │ │ │ │ - "tl": { │ │ │ │ │ - 'offset': new OpenLayers.Pixel(44, 0), │ │ │ │ │ - 'padding': new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ - 'blocks': [{ // top-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { //top-right │ │ │ │ │ - size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { //bottom-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ - }, { //bottom-right │ │ │ │ │ - size: new OpenLayers.Size(22, 18), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -632) │ │ │ │ │ - }, { // stem │ │ │ │ │ - size: new OpenLayers.Size(81, 35), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -688) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - "tr": { │ │ │ │ │ - 'offset': new OpenLayers.Pixel(-45, 0), │ │ │ │ │ - 'padding': new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ - 'blocks': [{ // top-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { //top-right │ │ │ │ │ - size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { //bottom-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ - }, { //bottom-right │ │ │ │ │ - size: new OpenLayers.Size(22, 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -631) │ │ │ │ │ - }, { // stem │ │ │ │ │ - size: new OpenLayers.Size(81, 35), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, null, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-215, -687) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - "bl": { │ │ │ │ │ - 'offset': new OpenLayers.Pixel(45, 0), │ │ │ │ │ - 'padding': new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ - 'blocks': [{ // top-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { //top-right │ │ │ │ │ - size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { //bottom-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ - }, { //bottom-right │ │ │ │ │ - size: new OpenLayers.Size(22, 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ - }, { // stem │ │ │ │ │ - size: new OpenLayers.Size(81, 33), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, null, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-101, -674) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - "br": { │ │ │ │ │ - 'offset': new OpenLayers.Pixel(-44, 0), │ │ │ │ │ - 'padding': new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ - 'blocks': [{ // top-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { //top-right │ │ │ │ │ - size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { //bottom-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ - }, { //bottom-right │ │ │ │ │ - size: new OpenLayers.Size(22, 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ - }, { // stem │ │ │ │ │ - size: new OpenLayers.Size(81, 33), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, null, null, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-311, -674) │ │ │ │ │ - }] │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + this.div.appendChild(this.layersDiv); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: minSize │ │ │ │ │ - * {<OpenLayers.Size>} │ │ │ │ │ - */ │ │ │ │ │ - minSize: new OpenLayers.Size(105, 10), │ │ │ │ │ + // maximize button div │ │ │ │ │ + var img = OpenLayers.Util.getImageLocation('layer-switcher-maximize.png'); │ │ │ │ │ + this.maximizeDiv = OpenLayers.Util.createAlphaImageDiv( │ │ │ │ │ + "OpenLayers_Control_MaximizeDiv", │ │ │ │ │ + null, │ │ │ │ │ + null, │ │ │ │ │ + img, │ │ │ │ │ + "absolute"); │ │ │ │ │ + OpenLayers.Element.addClass(this.maximizeDiv, "maximizeDiv olButton"); │ │ │ │ │ + this.maximizeDiv.style.display = "none"; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: maxSize │ │ │ │ │ - * {<OpenLayers.Size>} │ │ │ │ │ - */ │ │ │ │ │ - maxSize: new OpenLayers.Size(1200, 660), │ │ │ │ │ + this.div.appendChild(this.maximizeDiv); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Popup.FramedCloud │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ - * contentSize - {<OpenLayers.Size>} │ │ │ │ │ - * contentHTML - {String} │ │ │ │ │ - * anchor - {Object} Object to which we'll anchor the popup. Must expose │ │ │ │ │ - * a 'size' (<OpenLayers.Size>) and 'offset' (<OpenLayers.Pixel>) │ │ │ │ │ - * (Note that this is generally an <OpenLayers.Icon>). │ │ │ │ │ - * closeBox - {Boolean} │ │ │ │ │ - * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, │ │ │ │ │ - closeBoxCallback) { │ │ │ │ │ + // minimize button div │ │ │ │ │ + var img = OpenLayers.Util.getImageLocation('layer-switcher-minimize.png'); │ │ │ │ │ + this.minimizeDiv = OpenLayers.Util.createAlphaImageDiv( │ │ │ │ │ + "OpenLayers_Control_MinimizeDiv", │ │ │ │ │ + null, │ │ │ │ │ + null, │ │ │ │ │ + img, │ │ │ │ │ + "absolute"); │ │ │ │ │ + OpenLayers.Element.addClass(this.minimizeDiv, "minimizeDiv olButton"); │ │ │ │ │ + this.minimizeDiv.style.display = "none"; │ │ │ │ │ │ │ │ │ │ - this.imageSrc = OpenLayers.Util.getImageLocation('cloud-popup-relative.png'); │ │ │ │ │ - OpenLayers.Popup.Framed.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.contentDiv.className = this.contentDisplayClass; │ │ │ │ │ - }, │ │ │ │ │ + this.div.appendChild(this.minimizeDiv); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup.FramedCloud" │ │ │ │ │ - }); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.LayerSwitcher" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format.js │ │ │ │ │ + OpenLayers/Handler.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format │ │ │ │ │ - * Base class for format reading/writing a variety of formats. Subclasses │ │ │ │ │ - * of OpenLayers.Format are expected to have read and write methods. │ │ │ │ │ + * Class: OpenLayers.Handler │ │ │ │ │ + * Base class to construct a higher-level handler for event sequences. All │ │ │ │ │ + * handlers have activate and deactivate methods. In addition, they have │ │ │ │ │ + * methods named like browser events. When a handler is activated, any │ │ │ │ │ + * additional methods named like a browser event is registered as a │ │ │ │ │ + * listener for the corresponding event. When a handler is deactivated, │ │ │ │ │ + * those same methods are unregistered as event listeners. │ │ │ │ │ + * │ │ │ │ │ + * Handlers also typically have a callbacks object with keys named like │ │ │ │ │ + * the abstracted events or event sequences that they are in charge of │ │ │ │ │ + * handling. The controls that wrap handlers define the methods that │ │ │ │ │ + * correspond to these abstract events - so instead of listening for │ │ │ │ │ + * individual browser events, they only listen for the abstract events │ │ │ │ │ + * defined by the handler. │ │ │ │ │ + * │ │ │ │ │ + * Handlers are created by controls, which ultimately have the responsibility │ │ │ │ │ + * of making changes to the the state of the application. Handlers │ │ │ │ │ + * themselves may make temporary changes, but in general are expected to │ │ │ │ │ + * return the application in the same state that they found it. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format = OpenLayers.Class({ │ │ │ │ │ +OpenLayers.Handler = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: options │ │ │ │ │ - * {Object} A reference to options passed to the constructor. │ │ │ │ │ + * Property: id │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - options: null, │ │ │ │ │ + id: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: externalProjection │ │ │ │ │ - * {<OpenLayers.Projection>} When passed a externalProjection and │ │ │ │ │ - * internalProjection, the format will reproject the geometries it │ │ │ │ │ - * reads or writes. The externalProjection is the projection used by │ │ │ │ │ - * the content which is passed into read or which comes out of write. │ │ │ │ │ - * In order to reproject, a projection transformation function for the │ │ │ │ │ - * specified projections must be available. This support may be │ │ │ │ │ - * provided via proj4js or via a custom transformation function. See │ │ │ │ │ - * {<OpenLayers.Projection.addTransform>} for more information on │ │ │ │ │ - * custom transformations. │ │ │ │ │ + * APIProperty: control │ │ │ │ │ + * {<OpenLayers.Control>}. The control that initialized this handler. The │ │ │ │ │ + * control is assumed to have a valid map property - that map is used │ │ │ │ │ + * in the handler's own setMap method. │ │ │ │ │ */ │ │ │ │ │ - externalProjection: null, │ │ │ │ │ + control: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: internalProjection │ │ │ │ │ - * {<OpenLayers.Projection>} When passed a externalProjection and │ │ │ │ │ - * internalProjection, the format will reproject the geometries it │ │ │ │ │ - * reads or writes. The internalProjection is the projection used by │ │ │ │ │ - * the geometries which are returned by read or which are passed into │ │ │ │ │ - * write. In order to reproject, a projection transformation function │ │ │ │ │ - * for the specified projections must be available. This support may be │ │ │ │ │ - * provided via proj4js or via a custom transformation function. See │ │ │ │ │ - * {<OpenLayers.Projection.addTransform>} for more information on │ │ │ │ │ - * custom transformations. │ │ │ │ │ + * Property: map │ │ │ │ │ + * {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - internalProjection: null, │ │ │ │ │ + map: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: data │ │ │ │ │ - * {Object} When <keepData> is true, this is the parsed string sent to │ │ │ │ │ - * <read>. │ │ │ │ │ + * APIProperty: keyMask │ │ │ │ │ + * {Integer} Use bitwise operators and one or more of the OpenLayers.Handler │ │ │ │ │ + * constants to construct a keyMask. The keyMask is used by │ │ │ │ │ + * <checkModifiers>. If the keyMask matches the combination of keys │ │ │ │ │ + * down on an event, checkModifiers returns true. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * // handler only responds if the Shift key is down │ │ │ │ │ + * handler.keyMask = OpenLayers.Handler.MOD_SHIFT; │ │ │ │ │ + * │ │ │ │ │ + * // handler only responds if Ctrl-Shift is down │ │ │ │ │ + * handler.keyMask = OpenLayers.Handler.MOD_SHIFT | │ │ │ │ │ + * OpenLayers.Handler.MOD_CTRL; │ │ │ │ │ + * (end) │ │ │ │ │ */ │ │ │ │ │ - data: null, │ │ │ │ │ + keyMask: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: keepData │ │ │ │ │ - * {Object} Maintain a reference (<data>) to the most recently read data. │ │ │ │ │ - * Default is false. │ │ │ │ │ + * Property: active │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - keepData: false, │ │ │ │ │ + active: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format │ │ │ │ │ - * Instances of this class are not useful. See one of the subclasses. │ │ │ │ │ + * Property: evt │ │ │ │ │ + * {Event} This property references the last event handled by the handler. │ │ │ │ │ + * Note that this property is not part of the stable API. Use of the │ │ │ │ │ + * evt property should be restricted to controls in the library │ │ │ │ │ + * or other applications that are willing to update with changes to │ │ │ │ │ + * the OpenLayers code. │ │ │ │ │ + */ │ │ │ │ │ + evt: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: touch │ │ │ │ │ + * {Boolean} Indicates the support of touch events. When touch events are │ │ │ │ │ + * started touch will be true and all mouse related listeners will do │ │ │ │ │ + * nothing. │ │ │ │ │ + */ │ │ │ │ │ + touch: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Handler │ │ │ │ │ + * Construct a handler. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object with properties to set on the │ │ │ │ │ - * format │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * keepData - {Boolean} If true, upon <read>, the data property will be │ │ │ │ │ - * set to the parsed object (e.g. the json or xml object). │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that initialized this │ │ │ │ │ + * handler. The control is assumed to have a valid map property; that │ │ │ │ │ + * map is used in the handler's own setMap method. If a map property │ │ │ │ │ + * is present in the options argument it will be used instead. │ │ │ │ │ + * callbacks - {Object} An object whose properties correspond to abstracted │ │ │ │ │ + * events or sequences of browser events. The values for these │ │ │ │ │ + * properties are functions defined by the control that get called by │ │ │ │ │ + * the handler. │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * the handler. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.control = control; │ │ │ │ │ + this.callbacks = callbacks; │ │ │ │ │ + │ │ │ │ │ + var map = this.map || control.map; │ │ │ │ │ + if (map) { │ │ │ │ │ + this.setMap(map); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.map = map; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: checkModifiers │ │ │ │ │ + * Check the keyMask on the handler. If no <keyMask> is set, this always │ │ │ │ │ + * returns true. If a <keyMask> is set and it matches the combination │ │ │ │ │ + * of keys down on an event, this returns true. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * An instance of OpenLayers.Format │ │ │ │ │ + * {Boolean} The keyMask matches the keys down on an event. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.options = options; │ │ │ │ │ + checkModifiers: function(evt) { │ │ │ │ │ + if (this.keyMask == null) { │ │ │ │ │ + return true; │ │ │ │ │ + } │ │ │ │ │ + /* calculate the keyboard modifier mask for this event */ │ │ │ │ │ + var keyModifiers = │ │ │ │ │ + (evt.shiftKey ? OpenLayers.Handler.MOD_SHIFT : 0) | │ │ │ │ │ + (evt.ctrlKey ? OpenLayers.Handler.MOD_CTRL : 0) | │ │ │ │ │ + (evt.altKey ? OpenLayers.Handler.MOD_ALT : 0) | │ │ │ │ │ + (evt.metaKey ? OpenLayers.Handler.MOD_META : 0); │ │ │ │ │ + │ │ │ │ │ + /* if it differs from the handler object's key mask, │ │ │ │ │ + bail out of the event handler */ │ │ │ │ │ + return (keyModifiers == this.keyMask); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up. │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Turn on the handler. Returns false if the handler was already active. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was activated. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() {}, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + // register for event handlers defined on this class. │ │ │ │ │ + var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.register(events[i], this[events[i]]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.active = true; │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read │ │ │ │ │ - * Read data from a string, and return an object whose type depends on the │ │ │ │ │ - * subclass. │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Turn off the handler. Returns false if the handler was already inactive. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {string} Data to read/parse. │ │ │ │ │ - * │ │ │ │ │ * Returns: │ │ │ │ │ - * Depends on the subclass │ │ │ │ │ + * {Boolean} The handler was deactivated. │ │ │ │ │ */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - throw new Error('Read not implemented.'); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + // unregister event handlers defined on this class. │ │ │ │ │ + var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.unregister(events[i], this[events[i]]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.touch = false; │ │ │ │ │ + this.active = false; │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: write │ │ │ │ │ - * Accept an object, and return a string. │ │ │ │ │ + * Method: startTouch │ │ │ │ │ + * Start touch events, this method must be called by subclasses in │ │ │ │ │ + * "touchstart" method. When touch events are started <touch> will be │ │ │ │ │ + * true and all mouse related listeners will do nothing. │ │ │ │ │ + */ │ │ │ │ │ + startTouch: function() { │ │ │ │ │ + if (!this.touch) { │ │ │ │ │ + this.touch = true; │ │ │ │ │ + var events = [ │ │ │ │ │ + "mousedown", "mouseup", "mousemove", "click", "dblclick", │ │ │ │ │ + "mouseout" │ │ │ │ │ + ]; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.unregister(events[i], this[events[i]]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: callback │ │ │ │ │ + * Trigger the control's named callback with the given arguments │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * object - {Object} Object to be serialized │ │ │ │ │ + * name - {String} The key for the callback that is one of the properties │ │ │ │ │ + * of the handler's callbacks object. │ │ │ │ │ + * args - {Array(*)} An array of arguments (any type) with which to call │ │ │ │ │ + * the callback (defined by the control). │ │ │ │ │ + */ │ │ │ │ │ + callback: function(name, args) { │ │ │ │ │ + if (name && this.callbacks[name]) { │ │ │ │ │ + this.callbacks[name].apply(this.control, args); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: register │ │ │ │ │ + * register an event on the map │ │ │ │ │ + */ │ │ │ │ │ + register: function(name, method) { │ │ │ │ │ + // TODO: deal with registerPriority in 3.0 │ │ │ │ │ + this.map.events.registerPriority(name, this, method); │ │ │ │ │ + this.map.events.registerPriority(name, this, this.setEvent); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: unregister │ │ │ │ │ + * unregister an event from the map │ │ │ │ │ + */ │ │ │ │ │ + unregister: function(name, method) { │ │ │ │ │ + this.map.events.unregister(name, this, method); │ │ │ │ │ + this.map.events.unregister(name, this, this.setEvent); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setEvent │ │ │ │ │ + * With each registered browser event, the handler sets its own evt │ │ │ │ │ + * property. This property can be accessed by controls if needed │ │ │ │ │ + * to get more information about the event that the handler is │ │ │ │ │ + * processing. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string representation of the object. │ │ │ │ │ + * This allows modifier keys on the event to be checked (alt, shift, ctrl, │ │ │ │ │ + * and meta cannot be checked with the keyboard handler). For a │ │ │ │ │ + * control to determine which modifier keys are associated with the │ │ │ │ │ + * event that a handler is currently processing, it should access │ │ │ │ │ + * (code)handler.evt.altKey || handler.evt.shiftKey || │ │ │ │ │ + * handler.evt.ctrlKey || handler.evt.metaKey(end). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event. │ │ │ │ │ */ │ │ │ │ │ - write: function(object) { │ │ │ │ │ - throw new Error('Write not implemented.'); │ │ │ │ │ + setEvent: function(evt) { │ │ │ │ │ + this.evt = evt; │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format" │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Deconstruct the handler. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + // unregister event listeners │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + // eliminate circular references │ │ │ │ │ + this.control = this.map = null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_NONE │ │ │ │ │ + * If set as the <keyMask>, <checkModifiers> returns false if any key is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_NONE = 0; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_SHIFT │ │ │ │ │ + * If set as the <keyMask>, <checkModifiers> returns false if Shift is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_SHIFT = 1; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_CTRL │ │ │ │ │ + * If set as the <keyMask>, <checkModifiers> returns false if Ctrl is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_CTRL = 2; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_ALT │ │ │ │ │ + * If set as the <keyMask>, <checkModifiers> returns false if Alt is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_ALT = 4; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_META │ │ │ │ │ + * If set as the <keyMask>, <checkModifiers> returns false if Cmd is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_META = 8; │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/JSON.js │ │ │ │ │ + OpenLayers/Handler/Drag.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Note: │ │ │ │ │ - * This work draws heavily from the public domain JSON serializer/deserializer │ │ │ │ │ - * at http://www.json.org/json.js. Rewritten so that it doesn't modify │ │ │ │ │ - * basic data prototypes. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.JSON │ │ │ │ │ - * A parser to read/write JSON safely. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Format.JSON> constructor. │ │ │ │ │ + * Class: OpenLayers.Handler.Drag │ │ │ │ │ + * The drag handler is used to deal with sequences of browser events related │ │ │ │ │ + * to dragging. The handler is used by controls that want to know when │ │ │ │ │ + * a drag sequence begins, when a drag is happening, and when it has │ │ │ │ │ + * finished. │ │ │ │ │ + * │ │ │ │ │ + * Controls that use the drag handler typically construct it with callbacks │ │ │ │ │ + * for 'down', 'move', and 'done'. Callbacks for these keys are called │ │ │ │ │ + * when the drag begins, with each move, and when the drag is done. In │ │ │ │ │ + * addition, controls can have callbacks keyed to 'up' and 'out' if they │ │ │ │ │ + * care to differentiate between the types of events that correspond with │ │ │ │ │ + * the end of a drag sequence. If no drag actually occurs (no mouse move) │ │ │ │ │ + * the 'down' and 'up' callbacks will be called, but not the 'done' │ │ │ │ │ + * callback. │ │ │ │ │ + * │ │ │ │ │ + * Create a new drag handler with the <OpenLayers.Handler.Drag> constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.JSON = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ +OpenLayers.Handler.Drag = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: started │ │ │ │ │ + * {Boolean} When a mousedown or touchstart event is received, we want to │ │ │ │ │ + * record it, but not set 'dragging' until the mouse moves after starting. │ │ │ │ │ + */ │ │ │ │ │ + started: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: indent │ │ │ │ │ - * {String} For "pretty" printing, the indent string will be used once for │ │ │ │ │ - * each indentation level. │ │ │ │ │ + * Property: stopDown │ │ │ │ │ + * {Boolean} Stop propagation of mousedown events from getting to listeners │ │ │ │ │ + * on the same element. Default is true. │ │ │ │ │ */ │ │ │ │ │ - indent: " ", │ │ │ │ │ + stopDown: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: dragging │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + dragging: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: last │ │ │ │ │ + * {<OpenLayers.Pixel>} The last pixel location of the drag. │ │ │ │ │ + */ │ │ │ │ │ + last: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: start │ │ │ │ │ + * {<OpenLayers.Pixel>} The first pixel location of the drag. │ │ │ │ │ + */ │ │ │ │ │ + start: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: space │ │ │ │ │ - * {String} For "pretty" printing, the space string will be used after │ │ │ │ │ - * the ":" separating a name/value pair. │ │ │ │ │ + * Property: lastMoveEvt │ │ │ │ │ + * {Object} The last mousemove event that occurred. Used to │ │ │ │ │ + * position the map correctly when our "delay drag" │ │ │ │ │ + * timeout expired. │ │ │ │ │ */ │ │ │ │ │ - space: " ", │ │ │ │ │ + lastMoveEvt: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: newline │ │ │ │ │ - * {String} For "pretty" printing, the newline string will be used at the │ │ │ │ │ - * end of each name/value pair or array item. │ │ │ │ │ + * Property: oldOnselectstart │ │ │ │ │ + * {Function} │ │ │ │ │ */ │ │ │ │ │ - newline: "\n", │ │ │ │ │ + oldOnselectstart: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: level │ │ │ │ │ - * {Integer} For "pretty" printing, this is incremented/decremented during │ │ │ │ │ - * serialization. │ │ │ │ │ + * Property: interval │ │ │ │ │ + * {Integer} In order to increase performance, an interval (in │ │ │ │ │ + * milliseconds) can be set to reduce the number of drag events │ │ │ │ │ + * called. If set, a new drag event will not be set until the │ │ │ │ │ + * interval has passed. │ │ │ │ │ + * Defaults to 0, meaning no interval. │ │ │ │ │ */ │ │ │ │ │ - level: 0, │ │ │ │ │ + interval: 0, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: pretty │ │ │ │ │ - * {Boolean} Serialize with extra whitespace for structure. This is set │ │ │ │ │ - * by the <write> method. │ │ │ │ │ + * Property: timeoutId │ │ │ │ │ + * {String} The id of the timeout used for the mousedown interval. │ │ │ │ │ + * This is "private", and should be left alone. │ │ │ │ │ */ │ │ │ │ │ - pretty: false, │ │ │ │ │ + timeoutId: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: nativeJSON │ │ │ │ │ - * {Boolean} Does the browser support native json? │ │ │ │ │ + * APIProperty: documentDrag │ │ │ │ │ + * {Boolean} If set to true, the handler will also handle mouse moves when │ │ │ │ │ + * the cursor has moved out of the map viewport. Default is false. │ │ │ │ │ */ │ │ │ │ │ - nativeJSON: (function() { │ │ │ │ │ - return !!(window.JSON && typeof JSON.parse == "function" && typeof JSON.stringify == "function"); │ │ │ │ │ - })(), │ │ │ │ │ + documentDrag: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.JSON │ │ │ │ │ - * Create a new parser for JSON. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * Property: documentEvents │ │ │ │ │ + * {Boolean} Are we currently observing document events? │ │ │ │ │ */ │ │ │ │ │ + documentEvents: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Deserialize a json string. │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Handler.Drag │ │ │ │ │ + * Returns OpenLayers.Handler.Drag │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * json - {String} A JSON string │ │ │ │ │ - * filter - {Function} A function which will be called for every key and │ │ │ │ │ - * value at every level of the final result. Each value will be │ │ │ │ │ - * replaced by the result of the filter function. This can be used to │ │ │ │ │ - * reform generic objects into instances of classes, or to transform │ │ │ │ │ - * date strings into Date objects. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object, array, string, or number . │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ + * this handler. If a handler is being used without a control, the │ │ │ │ │ + * handlers setMap method must be overridden to deal properly with │ │ │ │ │ + * the map. │ │ │ │ │ + * callbacks - {Object} An object containing a single function to be │ │ │ │ │ + * called when the drag operation is finished. The callback should │ │ │ │ │ + * expect to recieve a single argument, the pixel location of the event. │ │ │ │ │ + * Callbacks for 'move' and 'done' are supported. You can also speficy │ │ │ │ │ + * callbacks for 'down', 'up', and 'out' to respond to those events. │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - read: function(json, filter) { │ │ │ │ │ - var object; │ │ │ │ │ - if (this.nativeJSON) { │ │ │ │ │ - object = JSON.parse(json, filter); │ │ │ │ │ - } else try { │ │ │ │ │ - /** │ │ │ │ │ - * Parsing happens in three stages. In the first stage, we run the │ │ │ │ │ - * text against a regular expression which looks for non-JSON │ │ │ │ │ - * characters. We are especially concerned with '()' and 'new' │ │ │ │ │ - * because they can cause invocation, and '=' because it can │ │ │ │ │ - * cause mutation. But just to be safe, we will reject all │ │ │ │ │ - * unexpected characters. │ │ │ │ │ - */ │ │ │ │ │ - if (/^[\],:{}\s]*$/.test(json.replace(/\\["\\\/bfnrtu]/g, '@').replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, ']').replace(/(?:^|:|,)(?:\s*\[)+/g, ''))) { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * In the second stage we use the eval function to compile the │ │ │ │ │ - * text into a JavaScript structure. The '{' operator is │ │ │ │ │ - * subject to a syntactic ambiguity in JavaScript - it can │ │ │ │ │ - * begin a block or an object literal. We wrap the text in │ │ │ │ │ - * parens to eliminate the ambiguity. │ │ │ │ │ - */ │ │ │ │ │ - object = eval('(' + json + ')'); │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * In the optional third stage, we recursively walk the new │ │ │ │ │ - * structure, passing each name/value pair to a filter │ │ │ │ │ - * function for possible transformation. │ │ │ │ │ - */ │ │ │ │ │ - if (typeof filter === 'function') { │ │ │ │ │ - function walk(k, v) { │ │ │ │ │ - if (v && typeof v === 'object') { │ │ │ │ │ - for (var i in v) { │ │ │ │ │ - if (v.hasOwnProperty(i)) { │ │ │ │ │ - v[i] = walk(i, v[i]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return filter(k, v); │ │ │ │ │ + if (this.documentDrag === true) { │ │ │ │ │ + var me = this; │ │ │ │ │ + this._docMove = function(evt) { │ │ │ │ │ + me.mousemove({ │ │ │ │ │ + xy: { │ │ │ │ │ + x: evt.clientX, │ │ │ │ │ + y: evt.clientY │ │ │ │ │ + }, │ │ │ │ │ + element: document │ │ │ │ │ + }); │ │ │ │ │ + }; │ │ │ │ │ + this._docUp = function(evt) { │ │ │ │ │ + me.mouseup({ │ │ │ │ │ + xy: { │ │ │ │ │ + x: evt.clientX, │ │ │ │ │ + y: evt.clientY │ │ │ │ │ } │ │ │ │ │ - object = walk('', object); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } catch (e) { │ │ │ │ │ - // Fall through if the regexp test fails. │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.keepData) { │ │ │ │ │ - this.data = object; │ │ │ │ │ + }); │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return object; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Serialize an object into a JSON string. │ │ │ │ │ + * Method: dragstart │ │ │ │ │ + * This private method is factorized from mousedown and touchstart methods │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * value - {String} The object, array, string, number, boolean or date │ │ │ │ │ - * to be serialized. │ │ │ │ │ - * pretty - {Boolean} Structure the output with newlines and indentation. │ │ │ │ │ - * Default is false. │ │ │ │ │ + * evt - {Event} The event │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} The JSON string representation of the input value. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - write: function(value, pretty) { │ │ │ │ │ - this.pretty = !!pretty; │ │ │ │ │ - var json = null; │ │ │ │ │ - var type = typeof value; │ │ │ │ │ - if (this.serialize[type]) { │ │ │ │ │ - try { │ │ │ │ │ - json = (!this.pretty && this.nativeJSON) ? │ │ │ │ │ - JSON.stringify(value) : │ │ │ │ │ - this.serialize[type].apply(this, [value]); │ │ │ │ │ - } catch (err) { │ │ │ │ │ - OpenLayers.Console.error("Trouble serializing: " + err); │ │ │ │ │ + dragstart: function(evt) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + if (this.checkModifiers(evt) && │ │ │ │ │ + (OpenLayers.Event.isLeftClick(evt) || │ │ │ │ │ + OpenLayers.Event.isSingleTouch(evt))) { │ │ │ │ │ + this.started = true; │ │ │ │ │ + this.start = evt.xy; │ │ │ │ │ + this.last = evt.xy; │ │ │ │ │ + OpenLayers.Element.addClass( │ │ │ │ │ + this.map.viewPortDiv, "olDragDown" │ │ │ │ │ + ); │ │ │ │ │ + this.down(evt); │ │ │ │ │ + this.callback("down", [evt.xy]); │ │ │ │ │ + │ │ │ │ │ + // prevent document dragging │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + │ │ │ │ │ + if (!this.oldOnselectstart) { │ │ │ │ │ + this.oldOnselectstart = document.onselectstart ? │ │ │ │ │ + document.onselectstart : OpenLayers.Function.True; │ │ │ │ │ } │ │ │ │ │ + document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ + │ │ │ │ │ + propagate = !this.stopDown; │ │ │ │ │ + } else { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ } │ │ │ │ │ - return json; │ │ │ │ │ + return propagate; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: writeIndent │ │ │ │ │ - * Output an indentation string depending on the indentation level. │ │ │ │ │ + * Method: dragmove │ │ │ │ │ + * This private method is factorized from mousemove and touchmove methods │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The event │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} An appropriate indentation string. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - writeIndent: function() { │ │ │ │ │ - var pieces = []; │ │ │ │ │ - if (this.pretty) { │ │ │ │ │ - for (var i = 0; i < this.level; ++i) { │ │ │ │ │ - pieces.push(this.indent); │ │ │ │ │ + dragmove: function(evt) { │ │ │ │ │ + this.lastMoveEvt = evt; │ │ │ │ │ + if (this.started && !this.timeoutId && (evt.xy.x != this.last.x || │ │ │ │ │ + evt.xy.y != this.last.y)) { │ │ │ │ │ + if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ + if (evt.element === document) { │ │ │ │ │ + this.adjustXY(evt); │ │ │ │ │ + // do setEvent manually because the documentEvents are not │ │ │ │ │ + // registered with the map │ │ │ │ │ + this.setEvent(evt); │ │ │ │ │ + } else { │ │ │ │ │ + this.removeDocumentEvents(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.interval > 0) { │ │ │ │ │ + this.timeoutId = setTimeout( │ │ │ │ │ + OpenLayers.Function.bind(this.removeTimeout, this), │ │ │ │ │ + this.interval); │ │ │ │ │ + } │ │ │ │ │ + this.dragging = true; │ │ │ │ │ + │ │ │ │ │ + this.move(evt); │ │ │ │ │ + this.callback("move", [evt.xy]); │ │ │ │ │ + if (!this.oldOnselectstart) { │ │ │ │ │ + this.oldOnselectstart = document.onselectstart; │ │ │ │ │ + document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ } │ │ │ │ │ + this.last = evt.xy; │ │ │ │ │ } │ │ │ │ │ - return pieces.join(''); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: writeNewline │ │ │ │ │ - * Output a string representing a newline if in pretty printing mode. │ │ │ │ │ + * Method: dragend │ │ │ │ │ + * This private method is factorized from mouseup and touchend methods │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string representing a new line. │ │ │ │ │ - */ │ │ │ │ │ - writeNewline: function() { │ │ │ │ │ - return (this.pretty) ? this.newline : ''; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: writeSpace │ │ │ │ │ - * Output a string representing a space if in pretty printing mode. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The event │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A space. │ │ │ │ │ - */ │ │ │ │ │ - writeSpace: function() { │ │ │ │ │ - return (this.pretty) ? this.space : ''; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: serialize │ │ │ │ │ - * Object with properties corresponding to the serializable data types. │ │ │ │ │ - * Property values are functions that do the actual serializing. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - serialize: { │ │ │ │ │ - /** │ │ │ │ │ - * Method: serialize.object │ │ │ │ │ - * Transform an object into a JSON string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * object - {Object} The object to be serialized. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A JSON string representing the object. │ │ │ │ │ - */ │ │ │ │ │ - 'object': function(object) { │ │ │ │ │ - // three special objects that we want to treat differently │ │ │ │ │ - if (object == null) { │ │ │ │ │ - return "null"; │ │ │ │ │ - } │ │ │ │ │ - if (object.constructor == Date) { │ │ │ │ │ - return this.serialize.date.apply(this, [object]); │ │ │ │ │ - } │ │ │ │ │ - if (object.constructor == Array) { │ │ │ │ │ - return this.serialize.array.apply(this, [object]); │ │ │ │ │ - } │ │ │ │ │ - var pieces = ['{']; │ │ │ │ │ - this.level += 1; │ │ │ │ │ - var key, keyJSON, valueJSON; │ │ │ │ │ - │ │ │ │ │ - var addComma = false; │ │ │ │ │ - for (key in object) { │ │ │ │ │ - if (object.hasOwnProperty(key)) { │ │ │ │ │ - // recursive calls need to allow for sub-classing │ │ │ │ │ - keyJSON = OpenLayers.Format.JSON.prototype.write.apply(this, │ │ │ │ │ - [key, this.pretty]); │ │ │ │ │ - valueJSON = OpenLayers.Format.JSON.prototype.write.apply(this, │ │ │ │ │ - [object[key], this.pretty]); │ │ │ │ │ - if (keyJSON != null && valueJSON != null) { │ │ │ │ │ - if (addComma) { │ │ │ │ │ - pieces.push(','); │ │ │ │ │ - } │ │ │ │ │ - pieces.push(this.writeNewline(), this.writeIndent(), │ │ │ │ │ - keyJSON, ':', this.writeSpace(), valueJSON); │ │ │ │ │ - addComma = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.level -= 1; │ │ │ │ │ - pieces.push(this.writeNewline(), this.writeIndent(), '}'); │ │ │ │ │ - return pieces.join(''); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: serialize.array │ │ │ │ │ - * Transform an array into a JSON string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * array - {Array} The array to be serialized │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A JSON string representing the array. │ │ │ │ │ - */ │ │ │ │ │ - 'array': function(array) { │ │ │ │ │ - var json; │ │ │ │ │ - var pieces = ['[']; │ │ │ │ │ - this.level += 1; │ │ │ │ │ - │ │ │ │ │ - for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ - // recursive calls need to allow for sub-classing │ │ │ │ │ - json = OpenLayers.Format.JSON.prototype.write.apply(this, │ │ │ │ │ - [array[i], this.pretty]); │ │ │ │ │ - if (json != null) { │ │ │ │ │ - if (i > 0) { │ │ │ │ │ - pieces.push(','); │ │ │ │ │ - } │ │ │ │ │ - pieces.push(this.writeNewline(), this.writeIndent(), json); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.level -= 1; │ │ │ │ │ - pieces.push(this.writeNewline(), this.writeIndent(), ']'); │ │ │ │ │ - return pieces.join(''); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: serialize.string │ │ │ │ │ - * Transform a string into a JSON string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * string - {String} The string to be serialized │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A JSON string representing the string. │ │ │ │ │ - */ │ │ │ │ │ - 'string': function(string) { │ │ │ │ │ - // If the string contains no control characters, no quote characters, and no │ │ │ │ │ - // backslash characters, then we can simply slap some quotes around it. │ │ │ │ │ - // Otherwise we must also replace the offending characters with safe │ │ │ │ │ - // sequences. │ │ │ │ │ - var m = { │ │ │ │ │ - '\b': '\\b', │ │ │ │ │ - '\t': '\\t', │ │ │ │ │ - '\n': '\\n', │ │ │ │ │ - '\f': '\\f', │ │ │ │ │ - '\r': '\\r', │ │ │ │ │ - '"': '\\"', │ │ │ │ │ - '\\': '\\\\' │ │ │ │ │ - }; │ │ │ │ │ - if (/["\\\x00-\x1f]/.test(string)) { │ │ │ │ │ - return '"' + string.replace(/([\x00-\x1f\\"])/g, function(a, b) { │ │ │ │ │ - var c = m[b]; │ │ │ │ │ - if (c) { │ │ │ │ │ - return c; │ │ │ │ │ - } │ │ │ │ │ - c = b.charCodeAt(); │ │ │ │ │ - return '\\u00' + │ │ │ │ │ - Math.floor(c / 16).toString(16) + │ │ │ │ │ - (c % 16).toString(16); │ │ │ │ │ - }) + '"'; │ │ │ │ │ + dragend: function(evt) { │ │ │ │ │ + if (this.started) { │ │ │ │ │ + if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ + this.adjustXY(evt); │ │ │ │ │ + this.removeDocumentEvents(); │ │ │ │ │ } │ │ │ │ │ - return '"' + string + '"'; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: serialize.number │ │ │ │ │ - * Transform a number into a JSON string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * number - {Number} The number to be serialized. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A JSON string representing the number. │ │ │ │ │ - */ │ │ │ │ │ - 'number': function(number) { │ │ │ │ │ - return isFinite(number) ? String(number) : "null"; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: serialize.boolean │ │ │ │ │ - * Transform a boolean into a JSON string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bool - {Boolean} The boolean to be serialized. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A JSON string representing the boolean. │ │ │ │ │ - */ │ │ │ │ │ - 'boolean': function(bool) { │ │ │ │ │ - return String(bool); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: serialize.object │ │ │ │ │ - * Transform a date into a JSON string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * date - {Date} The date to be serialized. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A JSON string representing the date. │ │ │ │ │ - */ │ │ │ │ │ - 'date': function(date) { │ │ │ │ │ - function format(number) { │ │ │ │ │ - // Format integers to have at least two digits. │ │ │ │ │ - return (number < 10) ? '0' + number : number; │ │ │ │ │ + var dragged = (this.start != this.last); │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + OpenLayers.Element.removeClass( │ │ │ │ │ + this.map.viewPortDiv, "olDragDown" │ │ │ │ │ + ); │ │ │ │ │ + this.up(evt); │ │ │ │ │ + this.callback("up", [evt.xy]); │ │ │ │ │ + if (dragged) { │ │ │ │ │ + this.callback("done", [evt.xy]); │ │ │ │ │ } │ │ │ │ │ - return '"' + date.getFullYear() + '-' + │ │ │ │ │ - format(date.getMonth() + 1) + '-' + │ │ │ │ │ - format(date.getDate()) + 'T' + │ │ │ │ │ - format(date.getHours()) + ':' + │ │ │ │ │ - format(date.getMinutes()) + ':' + │ │ │ │ │ - format(date.getSeconds()) + '"'; │ │ │ │ │ + document.onselectstart = this.oldOnselectstart; │ │ │ │ │ } │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.JSON" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Geometry.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Geometry │ │ │ │ │ - * A Geometry is a description of a geographic object. Create an instance of │ │ │ │ │ - * this class with the <OpenLayers.Geometry> constructor. This is a base class, │ │ │ │ │ - * typical geometry types are described by subclasses of this class. │ │ │ │ │ - * │ │ │ │ │ - * Note that if you use the <OpenLayers.Geometry.fromWKT> method, you must │ │ │ │ │ - * explicitly include the OpenLayers.Format.WKT in your build. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry = OpenLayers.Class({ │ │ │ │ │ + /** │ │ │ │ │ + * The four methods below (down, move, up, and out) are used by subclasses │ │ │ │ │ + * to do their own processing related to these mouse events. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: id │ │ │ │ │ - * {String} A unique identifier for this geometry. │ │ │ │ │ + * Method: down │ │ │ │ │ + * This method is called during the handling of the mouse down event. │ │ │ │ │ + * Subclasses can do their own processing here. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The mouse down event │ │ │ │ │ */ │ │ │ │ │ - id: null, │ │ │ │ │ + down: function(evt) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: parent │ │ │ │ │ - * {<OpenLayers.Geometry>}This is set when a Geometry is added as component │ │ │ │ │ - * of another geometry │ │ │ │ │ + * Method: move │ │ │ │ │ + * This method is called during the handling of the mouse move event. │ │ │ │ │ + * Subclasses can do their own processing here. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The mouse move event │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - parent: null, │ │ │ │ │ + move: function(evt) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: bounds │ │ │ │ │ - * {<OpenLayers.Bounds>} The bounds of this geometry │ │ │ │ │ + * Method: up │ │ │ │ │ + * This method is called during the handling of the mouse up event. │ │ │ │ │ + * Subclasses can do their own processing here. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The mouse up event │ │ │ │ │ */ │ │ │ │ │ - bounds: null, │ │ │ │ │ + up: function(evt) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Geometry │ │ │ │ │ - * Creates a geometry object. │ │ │ │ │ + * Method: out │ │ │ │ │ + * This method is called during the handling of the mouse out event. │ │ │ │ │ + * Subclasses can do their own processing here. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The mouse out event │ │ │ │ │ */ │ │ │ │ │ - initialize: function() { │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ - }, │ │ │ │ │ + out: function(evt) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Destroy this geometry. │ │ │ │ │ + * The methods below are part of the magic of event handling. Because │ │ │ │ │ + * they are named like browser events, they are registered as listeners │ │ │ │ │ + * for the events they represent. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.id = null; │ │ │ │ │ - this.bounds = null; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Create a clone of this geometry. Does not set any non-standard │ │ │ │ │ - * properties of the cloned geometry. │ │ │ │ │ - * │ │ │ │ │ + * Method: mousedown │ │ │ │ │ + * Handle mousedown events │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} An exact clone of this geometry. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - return new OpenLayers.Geometry(); │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + return this.dragstart(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setBounds │ │ │ │ │ - * Set the bounds for this Geometry. │ │ │ │ │ - * │ │ │ │ │ + * Method: touchstart │ │ │ │ │ + * Handle touchstart events │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - setBounds: function(bounds) { │ │ │ │ │ - if (bounds) { │ │ │ │ │ - this.bounds = bounds.clone(); │ │ │ │ │ - } │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + return this.dragstart(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clearBounds │ │ │ │ │ - * Nullify this components bounds and that of its parent as well. │ │ │ │ │ + * Method: mousemove │ │ │ │ │ + * Handle mousemove events │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - clearBounds: function() { │ │ │ │ │ - this.bounds = null; │ │ │ │ │ - if (this.parent) { │ │ │ │ │ - this.parent.clearBounds(); │ │ │ │ │ - } │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + return this.dragmove(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: extendBounds │ │ │ │ │ - * Extend the existing bounds to include the new bounds. │ │ │ │ │ - * If geometry's bounds is not yet set, then set a new Bounds. │ │ │ │ │ - * │ │ │ │ │ + * Method: touchmove │ │ │ │ │ + * Handle touchmove events │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * newBounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - extendBounds: function(newBounds) { │ │ │ │ │ - var bounds = this.getBounds(); │ │ │ │ │ - if (!bounds) { │ │ │ │ │ - this.setBounds(newBounds); │ │ │ │ │ - } else { │ │ │ │ │ - this.bounds.extend(newBounds); │ │ │ │ │ - } │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + return this.dragmove(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getBounds │ │ │ │ │ - * Get the bounds for this Geometry. If bounds is not set, it │ │ │ │ │ - * is calculated again, this makes queries faster. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} │ │ │ │ │ + * Method: removeTimeout │ │ │ │ │ + * Private. Called by mousemove() to remove the drag timeout. │ │ │ │ │ */ │ │ │ │ │ - getBounds: function() { │ │ │ │ │ - if (this.bounds == null) { │ │ │ │ │ - this.calculateBounds(); │ │ │ │ │ + removeTimeout: function() { │ │ │ │ │ + this.timeoutId = null; │ │ │ │ │ + // if timeout expires while we're still dragging (mouseup │ │ │ │ │ + // hasn't occurred) then call mousemove to move to the │ │ │ │ │ + // correct position │ │ │ │ │ + if (this.dragging) { │ │ │ │ │ + this.mousemove(this.lastMoveEvt); │ │ │ │ │ } │ │ │ │ │ - return this.bounds; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: calculateBounds │ │ │ │ │ - * Recalculate the bounds for the geometry. │ │ │ │ │ + /** │ │ │ │ │ + * Method: mouseup │ │ │ │ │ + * Handle mouseup events │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - calculateBounds: function() { │ │ │ │ │ - // │ │ │ │ │ - // This should be overridden by subclasses. │ │ │ │ │ - // │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + return this.dragend(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: distanceTo │ │ │ │ │ - * Calculate the closest distance between two geometries (on the x-y plane). │ │ │ │ │ + * Method: touchend │ │ │ │ │ + * Handle touchend events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ - * options - {Object} Optional properties for configuring the distance │ │ │ │ │ - * calculation. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ - * Valid options depend on the specific geometry type. │ │ │ │ │ - * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Number | Object} The distance between this geometry and the target. │ │ │ │ │ - * If details is true, the return will be an object with distance, │ │ │ │ │ - * x0, y0, x1, and x2 properties. The x0 and y0 properties represent │ │ │ │ │ - * the coordinates of the closest point on this geometry. The x1 and y1 │ │ │ │ │ - * properties represent the coordinates of the closest point on the │ │ │ │ │ - * target geometry. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - distanceTo: function(geometry, options) {}, │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + // override evt.xy with last position since touchend does not have │ │ │ │ │ + // any touch position │ │ │ │ │ + evt.xy = this.last; │ │ │ │ │ + return this.dragend(evt); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getVertices │ │ │ │ │ - * Return a list of all points in this geometry. │ │ │ │ │ + * Method: mouseout │ │ │ │ │ + * Handle mouseout events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * nodes - {Boolean} For lines, only return vertices that are │ │ │ │ │ - * endpoints. If false, for lines, only vertices that are not │ │ │ │ │ - * endpoints will be returned. If not provided, all vertices will │ │ │ │ │ - * be returned. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array} A list of all vertices in the geometry. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - getVertices: function(nodes) {}, │ │ │ │ │ + mouseout: function(evt) { │ │ │ │ │ + if (this.started && OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ + if (this.documentDrag === true) { │ │ │ │ │ + this.addDocumentEvents(); │ │ │ │ │ + } else { │ │ │ │ │ + var dragged = (this.start != this.last); │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + OpenLayers.Element.removeClass( │ │ │ │ │ + this.map.viewPortDiv, "olDragDown" │ │ │ │ │ + ); │ │ │ │ │ + this.out(evt); │ │ │ │ │ + this.callback("out", []); │ │ │ │ │ + if (dragged) { │ │ │ │ │ + this.callback("done", [evt.xy]); │ │ │ │ │ + } │ │ │ │ │ + if (document.onselectstart) { │ │ │ │ │ + document.onselectstart = this.oldOnselectstart; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: atPoint │ │ │ │ │ - * Note - This is only an approximation based on the bounds of the │ │ │ │ │ - * geometry. │ │ │ │ │ + * Method: click │ │ │ │ │ + * The drag handler captures the click event. If something else registers │ │ │ │ │ + * for clicks on the same element, its listener will not be called │ │ │ │ │ + * after a drag. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>|Object} OpenLayers.LonLat or an │ │ │ │ │ - * object with a 'lon' and 'lat' properties. │ │ │ │ │ - * toleranceLon - {float} Optional tolerance in Geometric Coords │ │ │ │ │ - * toleranceLat - {float} Optional tolerance in Geographic Coords │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Whether or not the geometry is at the specified location │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - atPoint: function(lonlat, toleranceLon, toleranceLat) { │ │ │ │ │ - var atPoint = false; │ │ │ │ │ - var bounds = this.getBounds(); │ │ │ │ │ - if ((bounds != null) && (lonlat != null)) { │ │ │ │ │ - │ │ │ │ │ - var dX = (toleranceLon != null) ? toleranceLon : 0; │ │ │ │ │ - var dY = (toleranceLat != null) ? toleranceLat : 0; │ │ │ │ │ - │ │ │ │ │ - var toleranceBounds = │ │ │ │ │ - new OpenLayers.Bounds(this.bounds.left - dX, │ │ │ │ │ - this.bounds.bottom - dY, │ │ │ │ │ - this.bounds.right + dX, │ │ │ │ │ - this.bounds.top + dY); │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + // let the click event propagate only if the mouse moved │ │ │ │ │ + return (this.start == this.last); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - atPoint = toleranceBounds.containsLonLat(lonlat); │ │ │ │ │ + /** │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activate the handler. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was successfully activated. │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + activated = true; │ │ │ │ │ } │ │ │ │ │ - return atPoint; │ │ │ │ │ + return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getLength │ │ │ │ │ - * Calculate the length of this geometry. This method is defined in │ │ │ │ │ - * subclasses. │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivate the handler. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Float} The length of the collection by summing its parts │ │ │ │ │ + * {Boolean} The handler was successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ - getLength: function() { │ │ │ │ │ - //to be overridden by geometries that actually have a length │ │ │ │ │ - // │ │ │ │ │ - return 0.0; │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + deactivated = true; │ │ │ │ │ + OpenLayers.Element.removeClass( │ │ │ │ │ + this.map.viewPortDiv, "olDragDown" │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getArea │ │ │ │ │ - * Calculate the area of this geometry. This method is defined in subclasses. │ │ │ │ │ + * Method: adjustXY │ │ │ │ │ + * Converts event coordinates that are relative to the document body to │ │ │ │ │ + * ones that are relative to the map viewport. The latter is the default in │ │ │ │ │ + * OpenLayers. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The area of the collection by summing its parts │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} │ │ │ │ │ */ │ │ │ │ │ - getArea: function() { │ │ │ │ │ - //to be overridden by geometries that actually have an area │ │ │ │ │ - // │ │ │ │ │ - return 0.0; │ │ │ │ │ + adjustXY: function(evt) { │ │ │ │ │ + var pos = OpenLayers.Util.pagePosition(this.map.viewPortDiv); │ │ │ │ │ + evt.xy.x -= pos[0]; │ │ │ │ │ + evt.xy.y -= pos[1]; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getCentroid │ │ │ │ │ - * Calculate the centroid of this geometry. This method is defined in subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} The centroid of the collection │ │ │ │ │ + * Method: addDocumentEvents │ │ │ │ │ + * Start observing document events when documentDrag is true and the mouse │ │ │ │ │ + * cursor leaves the map viewport while dragging. │ │ │ │ │ */ │ │ │ │ │ - getCentroid: function() { │ │ │ │ │ - return null; │ │ │ │ │ + addDocumentEvents: function() { │ │ │ │ │ + OpenLayers.Element.addClass(document.body, "olDragDown"); │ │ │ │ │ + this.documentEvents = true; │ │ │ │ │ + OpenLayers.Event.observe(document, "mousemove", this._docMove); │ │ │ │ │ + OpenLayers.Event.observe(document, "mouseup", this._docUp); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: toString │ │ │ │ │ - * Returns a text representation of the geometry. If the WKT format is │ │ │ │ │ - * included in a build, this will be the Well-Known Text │ │ │ │ │ - * representation. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} String representation of this geometry. │ │ │ │ │ + * Method: removeDocumentEvents │ │ │ │ │ + * Stops observing document events when documentDrag is true and the mouse │ │ │ │ │ + * cursor re-enters the map viewport while dragging. │ │ │ │ │ */ │ │ │ │ │ - toString: function() { │ │ │ │ │ - var string; │ │ │ │ │ - if (OpenLayers.Format && OpenLayers.Format.WKT) { │ │ │ │ │ - string = OpenLayers.Format.WKT.prototype.write( │ │ │ │ │ - new OpenLayers.Feature.Vector(this) │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - string = Object.prototype.toString.call(this); │ │ │ │ │ - } │ │ │ │ │ - return string; │ │ │ │ │ + removeDocumentEvents: function() { │ │ │ │ │ + OpenLayers.Element.removeClass(document.body, "olDragDown"); │ │ │ │ │ + this.documentEvents = false; │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mousemove", this._docMove); │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mouseup", this._docUp); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Geometry" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Drag" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Function: OpenLayers.Geometry.fromWKT │ │ │ │ │ - * Generate a geometry given a Well-Known Text string. For this method to │ │ │ │ │ - * work, you must include the OpenLayers.Format.WKT in your build │ │ │ │ │ - * explicitly. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * wkt - {String} A string representing the geometry in Well-Known Text. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} A geometry of the appropriate class. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.fromWKT = function(wkt) { │ │ │ │ │ - var geom; │ │ │ │ │ - if (OpenLayers.Format && OpenLayers.Format.WKT) { │ │ │ │ │ - var format = OpenLayers.Geometry.fromWKT.format; │ │ │ │ │ - if (!format) { │ │ │ │ │ - format = new OpenLayers.Format.WKT(); │ │ │ │ │ - OpenLayers.Geometry.fromWKT.format = format; │ │ │ │ │ - } │ │ │ │ │ - var result = format.read(wkt); │ │ │ │ │ - if (result instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ - geom = result.geometry; │ │ │ │ │ - } else if (OpenLayers.Util.isArray(result)) { │ │ │ │ │ - var len = result.length; │ │ │ │ │ - var components = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - components[i] = result[i].geometry; │ │ │ │ │ - } │ │ │ │ │ - geom = new OpenLayers.Geometry.Collection(components); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return geom; │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Method: OpenLayers.Geometry.segmentsIntersect │ │ │ │ │ - * Determine whether two line segments intersect. Optionally calculates │ │ │ │ │ - * and returns the intersection point. This function is optimized for │ │ │ │ │ - * cases where seg1.x2 >= seg2.x1 || seg2.x2 >= seg1.x1. In those │ │ │ │ │ - * obvious cases where there is no intersection, the function should │ │ │ │ │ - * not be called. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * seg1 - {Object} Object representing a segment with properties x1, y1, x2, │ │ │ │ │ - * and y2. The start point is represented by x1 and y1. The end point │ │ │ │ │ - * is represented by x2 and y2. Start and end are ordered so that x1 < x2. │ │ │ │ │ - * seg2 - {Object} Object representing a segment with properties x1, y1, x2, │ │ │ │ │ - * and y2. The start point is represented by x1 and y1. The end point │ │ │ │ │ - * is represented by x2 and y2. Start and end are ordered so that x1 < x2. │ │ │ │ │ - * options - {Object} Optional properties for calculating the intersection. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * point - {Boolean} Return the intersection point. If false, the actual │ │ │ │ │ - * intersection point will not be calculated. If true and the segments │ │ │ │ │ - * intersect, the intersection point will be returned. If true and │ │ │ │ │ - * the segments do not intersect, false will be returned. If true and │ │ │ │ │ - * the segments are coincident, true will be returned. │ │ │ │ │ - * tolerance - {Number} If a non-null value is provided, if the segments are │ │ │ │ │ - * within the tolerance distance, this will be considered an intersection. │ │ │ │ │ - * In addition, if the point option is true and the calculated intersection │ │ │ │ │ - * is within the tolerance distance of an end point, the endpoint will be │ │ │ │ │ - * returned instead of the calculated intersection. Further, if the │ │ │ │ │ - * intersection is within the tolerance of endpoints on both segments, or │ │ │ │ │ - * if two segment endpoints are within the tolerance distance of eachother │ │ │ │ │ - * (but no intersection is otherwise calculated), an endpoint on the │ │ │ │ │ - * first segment provided will be returned. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean | <OpenLayers.Geometry.Point>} The two segments intersect. │ │ │ │ │ - * If the point argument is true, the return will be the intersection │ │ │ │ │ - * point or false if none exists. If point is true and the segments │ │ │ │ │ - * are coincident, return will be true (and the instersection is equal │ │ │ │ │ - * to the shorter segment). │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.segmentsIntersect = function(seg1, seg2, options) { │ │ │ │ │ - var point = options && options.point; │ │ │ │ │ - var tolerance = options && options.tolerance; │ │ │ │ │ - var intersection = false; │ │ │ │ │ - var x11_21 = seg1.x1 - seg2.x1; │ │ │ │ │ - var y11_21 = seg1.y1 - seg2.y1; │ │ │ │ │ - var x12_11 = seg1.x2 - seg1.x1; │ │ │ │ │ - var y12_11 = seg1.y2 - seg1.y1; │ │ │ │ │ - var y22_21 = seg2.y2 - seg2.y1; │ │ │ │ │ - var x22_21 = seg2.x2 - seg2.x1; │ │ │ │ │ - var d = (y22_21 * x12_11) - (x22_21 * y12_11); │ │ │ │ │ - var n1 = (x22_21 * y11_21) - (y22_21 * x11_21); │ │ │ │ │ - var n2 = (x12_11 * y11_21) - (y12_11 * x11_21); │ │ │ │ │ - if (d == 0) { │ │ │ │ │ - // parallel │ │ │ │ │ - if (n1 == 0 && n2 == 0) { │ │ │ │ │ - // coincident │ │ │ │ │ - intersection = true; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - var along1 = n1 / d; │ │ │ │ │ - var along2 = n2 / d; │ │ │ │ │ - if (along1 >= 0 && along1 <= 1 && along2 >= 0 && along2 <= 1) { │ │ │ │ │ - // intersect │ │ │ │ │ - if (!point) { │ │ │ │ │ - intersection = true; │ │ │ │ │ - } else { │ │ │ │ │ - // calculate the intersection point │ │ │ │ │ - var x = seg1.x1 + (along1 * x12_11); │ │ │ │ │ - var y = seg1.y1 + (along1 * y12_11); │ │ │ │ │ - intersection = new OpenLayers.Geometry.Point(x, y); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (tolerance) { │ │ │ │ │ - var dist; │ │ │ │ │ - if (intersection) { │ │ │ │ │ - if (point) { │ │ │ │ │ - var segs = [seg1, seg2]; │ │ │ │ │ - var seg, x, y; │ │ │ │ │ - // check segment endpoints for proximity to intersection │ │ │ │ │ - // set intersection to first endpoint within the tolerance │ │ │ │ │ - outer: for (var i = 0; i < 2; ++i) { │ │ │ │ │ - seg = segs[i]; │ │ │ │ │ - for (var j = 1; j < 3; ++j) { │ │ │ │ │ - x = seg["x" + j]; │ │ │ │ │ - y = seg["y" + j]; │ │ │ │ │ - dist = Math.sqrt( │ │ │ │ │ - Math.pow(x - intersection.x, 2) + │ │ │ │ │ - Math.pow(y - intersection.y, 2) │ │ │ │ │ - ); │ │ │ │ │ - if (dist < tolerance) { │ │ │ │ │ - intersection.x = x; │ │ │ │ │ - intersection.y = y; │ │ │ │ │ - break outer; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - // no calculated intersection, but segments could be within │ │ │ │ │ - // the tolerance of one another │ │ │ │ │ - var segs = [seg1, seg2]; │ │ │ │ │ - var source, target, x, y, p, result; │ │ │ │ │ - // check segment endpoints for proximity to intersection │ │ │ │ │ - // set intersection to first endpoint within the tolerance │ │ │ │ │ - outer: for (var i = 0; i < 2; ++i) { │ │ │ │ │ - source = segs[i]; │ │ │ │ │ - target = segs[(i + 1) % 2]; │ │ │ │ │ - for (var j = 1; j < 3; ++j) { │ │ │ │ │ - p = { │ │ │ │ │ - x: source["x" + j], │ │ │ │ │ - y: source["y" + j] │ │ │ │ │ - }; │ │ │ │ │ - result = OpenLayers.Geometry.distanceToSegment(p, target); │ │ │ │ │ - if (result.distance < tolerance) { │ │ │ │ │ - if (point) { │ │ │ │ │ - intersection = new OpenLayers.Geometry.Point(p.x, p.y); │ │ │ │ │ - } else { │ │ │ │ │ - intersection = true; │ │ │ │ │ - } │ │ │ │ │ - break outer; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return intersection; │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Function: OpenLayers.Geometry.distanceToSegment │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {Object} An object with x and y properties representing the │ │ │ │ │ - * point coordinates. │ │ │ │ │ - * segment - {Object} An object with x1, y1, x2, and y2 properties │ │ │ │ │ - * representing endpoint coordinates. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object with distance, along, x, and y properties. The distance │ │ │ │ │ - * will be the shortest distance between the input point and segment. │ │ │ │ │ - * The x and y properties represent the coordinates along the segment │ │ │ │ │ - * where the shortest distance meets the segment. The along attribute │ │ │ │ │ - * describes how far between the two segment points the given point is. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.distanceToSegment = function(point, segment) { │ │ │ │ │ - var result = OpenLayers.Geometry.distanceSquaredToSegment(point, segment); │ │ │ │ │ - result.distance = Math.sqrt(result.distance); │ │ │ │ │ - return result; │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Function: OpenLayers.Geometry.distanceSquaredToSegment │ │ │ │ │ - * │ │ │ │ │ - * Usually the distanceToSegment function should be used. This variant however │ │ │ │ │ - * can be used for comparisons where the exact distance is not important. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {Object} An object with x and y properties representing the │ │ │ │ │ - * point coordinates. │ │ │ │ │ - * segment - {Object} An object with x1, y1, x2, and y2 properties │ │ │ │ │ - * representing endpoint coordinates. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object with squared distance, along, x, and y properties. │ │ │ │ │ - * The distance will be the shortest distance between the input point and │ │ │ │ │ - * segment. The x and y properties represent the coordinates along the │ │ │ │ │ - * segment where the shortest distance meets the segment. The along │ │ │ │ │ - * attribute describes how far between the two segment points the given │ │ │ │ │ - * point is. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.distanceSquaredToSegment = function(point, segment) { │ │ │ │ │ - var x0 = point.x; │ │ │ │ │ - var y0 = point.y; │ │ │ │ │ - var x1 = segment.x1; │ │ │ │ │ - var y1 = segment.y1; │ │ │ │ │ - var x2 = segment.x2; │ │ │ │ │ - var y2 = segment.y2; │ │ │ │ │ - var dx = x2 - x1; │ │ │ │ │ - var dy = y2 - y1; │ │ │ │ │ - var along = ((dx * (x0 - x1)) + (dy * (y0 - y1))) / │ │ │ │ │ - (Math.pow(dx, 2) + Math.pow(dy, 2)); │ │ │ │ │ - var x, y; │ │ │ │ │ - if (along <= 0.0) { │ │ │ │ │ - x = x1; │ │ │ │ │ - y = y1; │ │ │ │ │ - } else if (along >= 1.0) { │ │ │ │ │ - x = x2; │ │ │ │ │ - y = y2; │ │ │ │ │ - } else { │ │ │ │ │ - x = x1 + along * dx; │ │ │ │ │ - y = y1 + along * dy; │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - distance: Math.pow(x - x0, 2) + Math.pow(y - y0, 2), │ │ │ │ │ - x: x, │ │ │ │ │ - y: y, │ │ │ │ │ - along: along │ │ │ │ │ - }; │ │ │ │ │ -}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Geometry/Point.js │ │ │ │ │ + OpenLayers/Handler/Box.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Geometry.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Handler/Drag.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Geometry.Point │ │ │ │ │ - * Point geometry class. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Handler.Box │ │ │ │ │ + * Handler for dragging a rectangle across the map. Box is displayed │ │ │ │ │ + * on mouse down, moves on mouse move, and is finished on mouse up. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Geometry> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Geometry.Point = OpenLayers.Class(OpenLayers.Geometry, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: x │ │ │ │ │ - * {float} │ │ │ │ │ - */ │ │ │ │ │ - x: null, │ │ │ │ │ +OpenLayers.Handler.Box = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: y │ │ │ │ │ - * {float} │ │ │ │ │ + * Property: dragHandler │ │ │ │ │ + * {<OpenLayers.Handler.Drag>} │ │ │ │ │ */ │ │ │ │ │ - y: null, │ │ │ │ │ + dragHandler: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Geometry.Point │ │ │ │ │ - * Construct a point geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {float} │ │ │ │ │ - * y - {float} │ │ │ │ │ - * │ │ │ │ │ + * APIProperty: boxDivClassName │ │ │ │ │ + * {String} The CSS class to use for drawing the box. Default is │ │ │ │ │ + * olHandlerBoxZoomBox │ │ │ │ │ */ │ │ │ │ │ - initialize: function(x, y) { │ │ │ │ │ - OpenLayers.Geometry.prototype.initialize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - this.x = parseFloat(x); │ │ │ │ │ - this.y = parseFloat(y); │ │ │ │ │ - }, │ │ │ │ │ + boxDivClassName: 'olHandlerBoxZoomBox', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} An exact clone of this OpenLayers.Geometry.Point │ │ │ │ │ - */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Geometry.Point(this.x, this.y); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // catch any randomly tagged-on properties │ │ │ │ │ - OpenLayers.Util.applyDefaults(obj, this); │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateBounds │ │ │ │ │ - * Create a new Bounds based on the lon/lat │ │ │ │ │ + * Property: boxOffsets │ │ │ │ │ + * {Object} Caches box offsets from css. This is used by the getBoxOffsets │ │ │ │ │ + * method. │ │ │ │ │ */ │ │ │ │ │ - calculateBounds: function() { │ │ │ │ │ - this.bounds = new OpenLayers.Bounds(this.x, this.y, │ │ │ │ │ - this.x, this.y); │ │ │ │ │ - }, │ │ │ │ │ + boxOffsets: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: distanceTo │ │ │ │ │ - * Calculate the closest distance between two geometries (on the x-y plane). │ │ │ │ │ + * Constructor: OpenLayers.Handler.Box │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ - * options - {Object} Optional properties for configuring the distance │ │ │ │ │ - * calculation. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * details - {Boolean} Return details from the distance calculation. │ │ │ │ │ - * Default is false. │ │ │ │ │ - * edge - {Boolean} Calculate the distance from this geometry to the │ │ │ │ │ - * nearest edge of the target geometry. Default is true. If true, │ │ │ │ │ - * calling distanceTo from a geometry that is wholly contained within │ │ │ │ │ - * the target will result in a non-zero distance. If false, whenever │ │ │ │ │ - * geometries intersect, calling distanceTo will return 0. If false, │ │ │ │ │ - * details cannot be returned. │ │ │ │ │ + * control - {<OpenLayers.Control>} │ │ │ │ │ + * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ + * functions. Various callbacks described below. │ │ │ │ │ + * options - {Object} │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number | Object} The distance between this geometry and the target. │ │ │ │ │ - * If details is true, the return will be an object with distance, │ │ │ │ │ - * x0, y0, x1, and x2 properties. The x0 and y0 properties represent │ │ │ │ │ - * the coordinates of the closest point on this geometry. The x1 and y1 │ │ │ │ │ - * properties represent the coordinates of the closest point on the │ │ │ │ │ - * target geometry. │ │ │ │ │ + * Named callbacks: │ │ │ │ │ + * start - Called when the box drag operation starts. │ │ │ │ │ + * done - Called when the box drag operation is finished. │ │ │ │ │ + * The callback should expect to receive a single argument, the box │ │ │ │ │ + * bounds or a pixel. If the box dragging didn't span more than a 5 │ │ │ │ │ + * pixel distance, a pixel will be returned instead of a bounds object. │ │ │ │ │ */ │ │ │ │ │ - distanceTo: function(geometry, options) { │ │ │ │ │ - var edge = !(options && options.edge === false); │ │ │ │ │ - var details = edge && options && options.details; │ │ │ │ │ - var distance, x0, y0, x1, y1, result; │ │ │ │ │ - if (geometry instanceof OpenLayers.Geometry.Point) { │ │ │ │ │ - x0 = this.x; │ │ │ │ │ - y0 = this.y; │ │ │ │ │ - x1 = geometry.x; │ │ │ │ │ - y1 = geometry.y; │ │ │ │ │ - distance = Math.sqrt(Math.pow(x0 - x1, 2) + Math.pow(y0 - y1, 2)); │ │ │ │ │ - result = !details ? │ │ │ │ │ - distance : { │ │ │ │ │ - x0: x0, │ │ │ │ │ - y0: y0, │ │ │ │ │ - x1: x1, │ │ │ │ │ - y1: y1, │ │ │ │ │ - distance: distance │ │ │ │ │ - }; │ │ │ │ │ - } else { │ │ │ │ │ - result = geometry.distanceTo(this, options); │ │ │ │ │ - if (details) { │ │ │ │ │ - // switch coord order since this geom is target │ │ │ │ │ - result = { │ │ │ │ │ - x0: result.x1, │ │ │ │ │ - y0: result.y1, │ │ │ │ │ - x1: result.x0, │ │ │ │ │ - y1: result.y0, │ │ │ │ │ - distance: result.distance │ │ │ │ │ - }; │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.dragHandler = new OpenLayers.Handler.Drag( │ │ │ │ │ + this, { │ │ │ │ │ + down: this.startBox, │ │ │ │ │ + move: this.moveBox, │ │ │ │ │ + out: this.removeBox, │ │ │ │ │ + up: this.endBox │ │ │ │ │ + }, { │ │ │ │ │ + keyMask: this.keyMask │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return result; │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: equals │ │ │ │ │ - * Determine whether another geometry is equivalent to this one. Geometries │ │ │ │ │ - * are considered equivalent if all components have the same coordinates. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geom - {<OpenLayers.Geometry.Point>} The geometry to test. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The supplied geometry is equivalent to this geometry. │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ - equals: function(geom) { │ │ │ │ │ - var equals = false; │ │ │ │ │ - if (geom != null) { │ │ │ │ │ - equals = ((this.x == geom.x && this.y == geom.y) || │ │ │ │ │ - (isNaN(this.x) && isNaN(this.y) && isNaN(geom.x) && isNaN(geom.y))); │ │ │ │ │ + destroy: function() { │ │ │ │ │ + OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ + if (this.dragHandler) { │ │ │ │ │ + this.dragHandler.destroy(); │ │ │ │ │ + this.dragHandler = null; │ │ │ │ │ } │ │ │ │ │ - return equals; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: toShortString │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} Shortened String representation of Point object. │ │ │ │ │ - * (ex. <i>"5, 42"</i>) │ │ │ │ │ + * Method: setMap │ │ │ │ │ */ │ │ │ │ │ - toShortString: function() { │ │ │ │ │ - return (this.x + ", " + this.y); │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Handler.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (this.dragHandler) { │ │ │ │ │ + this.dragHandler.setMap(map); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: move │ │ │ │ │ - * Moves a geometry by the given displacement along positive x and y axes. │ │ │ │ │ - * This modifies the position of the geometry and clears the cached │ │ │ │ │ - * bounds. │ │ │ │ │ + * Method: startBox │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * x - {Float} Distance to move geometry in positive x direction. │ │ │ │ │ - * y - {Float} Distance to move geometry in positive y direction. │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} │ │ │ │ │ */ │ │ │ │ │ - move: function(x, y) { │ │ │ │ │ - this.x = this.x + x; │ │ │ │ │ - this.y = this.y + y; │ │ │ │ │ - this.clearBounds(); │ │ │ │ │ + startBox: function(xy) { │ │ │ │ │ + this.callback("start", []); │ │ │ │ │ + this.zoomBox = OpenLayers.Util.createDiv('zoomBox', { │ │ │ │ │ + x: -9999, │ │ │ │ │ + y: -9999 │ │ │ │ │ + }); │ │ │ │ │ + this.zoomBox.className = this.boxDivClassName; │ │ │ │ │ + this.zoomBox.style.zIndex = this.map.Z_INDEX_BASE["Popup"] - 1; │ │ │ │ │ + │ │ │ │ │ + this.map.viewPortDiv.appendChild(this.zoomBox); │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Element.addClass( │ │ │ │ │ + this.map.viewPortDiv, "olDrawBox" │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: rotate │ │ │ │ │ - * Rotate a point around another. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * angle - {Float} Rotation angle in degrees (measured counterclockwise │ │ │ │ │ - * from the positive x-axis) │ │ │ │ │ - * origin - {<OpenLayers.Geometry.Point>} Center point for the rotation │ │ │ │ │ + * Method: moveBox │ │ │ │ │ */ │ │ │ │ │ - rotate: function(angle, origin) { │ │ │ │ │ - angle *= Math.PI / 180; │ │ │ │ │ - var radius = this.distanceTo(origin); │ │ │ │ │ - var theta = angle + Math.atan2(this.y - origin.y, this.x - origin.x); │ │ │ │ │ - this.x = origin.x + (radius * Math.cos(theta)); │ │ │ │ │ - this.y = origin.y + (radius * Math.sin(theta)); │ │ │ │ │ - this.clearBounds(); │ │ │ │ │ + moveBox: function(xy) { │ │ │ │ │ + var startX = this.dragHandler.start.x; │ │ │ │ │ + var startY = this.dragHandler.start.y; │ │ │ │ │ + var deltaX = Math.abs(startX - xy.x); │ │ │ │ │ + var deltaY = Math.abs(startY - xy.y); │ │ │ │ │ + │ │ │ │ │ + var offset = this.getBoxOffsets(); │ │ │ │ │ + this.zoomBox.style.width = (deltaX + offset.width + 1) + "px"; │ │ │ │ │ + this.zoomBox.style.height = (deltaY + offset.height + 1) + "px"; │ │ │ │ │ + this.zoomBox.style.left = (xy.x < startX ? │ │ │ │ │ + startX - deltaX - offset.left : startX - offset.left) + "px"; │ │ │ │ │ + this.zoomBox.style.top = (xy.y < startY ? │ │ │ │ │ + startY - deltaY - offset.top : startY - offset.top) + "px"; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getCentroid │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} The centroid of the collection │ │ │ │ │ + * Method: endBox │ │ │ │ │ */ │ │ │ │ │ - getCentroid: function() { │ │ │ │ │ - return new OpenLayers.Geometry.Point(this.x, this.y); │ │ │ │ │ + endBox: function(end) { │ │ │ │ │ + var result; │ │ │ │ │ + if (Math.abs(this.dragHandler.start.x - end.x) > 5 || │ │ │ │ │ + Math.abs(this.dragHandler.start.y - end.y) > 5) { │ │ │ │ │ + var start = this.dragHandler.start; │ │ │ │ │ + var top = Math.min(start.y, end.y); │ │ │ │ │ + var bottom = Math.max(start.y, end.y); │ │ │ │ │ + var left = Math.min(start.x, end.x); │ │ │ │ │ + var right = Math.max(start.x, end.x); │ │ │ │ │ + result = new OpenLayers.Bounds(left, bottom, right, top); │ │ │ │ │ + } else { │ │ │ │ │ + result = this.dragHandler.start.clone(); // i.e. OL.Pixel │ │ │ │ │ + } │ │ │ │ │ + this.removeBox(); │ │ │ │ │ + │ │ │ │ │ + this.callback("done", [result]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: resize │ │ │ │ │ - * Resize a point relative to some origin. For points, this has the effect │ │ │ │ │ - * of scaling a vector (from the origin to the point). This method is │ │ │ │ │ - * more useful on geometry collection subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * scale - {Float} Ratio of the new distance from the origin to the old │ │ │ │ │ - * distance from the origin. A scale of 2 doubles the │ │ │ │ │ - * distance between the point and origin. │ │ │ │ │ - * origin - {<OpenLayers.Geometry.Point>} Point of origin for resizing │ │ │ │ │ - * ratio - {Float} Optional x:y ratio for resizing. Default ratio is 1. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} - The current geometry. │ │ │ │ │ + * Method: removeBox │ │ │ │ │ + * Remove the zoombox from the screen and nullify our reference to it. │ │ │ │ │ */ │ │ │ │ │ - resize: function(scale, origin, ratio) { │ │ │ │ │ - ratio = (ratio == undefined) ? 1 : ratio; │ │ │ │ │ - this.x = origin.x + (scale * ratio * (this.x - origin.x)); │ │ │ │ │ - this.y = origin.y + (scale * (this.y - origin.y)); │ │ │ │ │ - this.clearBounds(); │ │ │ │ │ - return this; │ │ │ │ │ + removeBox: function() { │ │ │ │ │ + this.map.viewPortDiv.removeChild(this.zoomBox); │ │ │ │ │ + this.zoomBox = null; │ │ │ │ │ + this.boxOffsets = null; │ │ │ │ │ + OpenLayers.Element.removeClass( │ │ │ │ │ + this.map.viewPortDiv, "olDrawBox" │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: intersects │ │ │ │ │ - * Determine if the input geometry intersects this one. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} Any type of geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The input geometry intersects this one. │ │ │ │ │ + * Method: activate │ │ │ │ │ */ │ │ │ │ │ - intersects: function(geometry) { │ │ │ │ │ - var intersect = false; │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - intersect = this.equals(geometry); │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.dragHandler.activate(); │ │ │ │ │ + return true; │ │ │ │ │ } else { │ │ │ │ │ - intersect = geometry.intersects(this); │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - return intersect; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: transform │ │ │ │ │ - * Translate the x,y properties of the point from source to dest. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * source - {<OpenLayers.Projection>} │ │ │ │ │ - * dest - {<OpenLayers.Projection>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} │ │ │ │ │ + * Method: deactivate │ │ │ │ │ */ │ │ │ │ │ - transform: function(source, dest) { │ │ │ │ │ - if ((source && dest)) { │ │ │ │ │ - OpenLayers.Projection.transform( │ │ │ │ │ - this, source, dest); │ │ │ │ │ - this.bounds = null; │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + if (this.dragHandler.deactivate()) { │ │ │ │ │ + if (this.zoomBox) { │ │ │ │ │ + this.removeBox(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - return this; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getVertices │ │ │ │ │ - * Return a list of all points in this geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * nodes - {Boolean} For lines, only return vertices that are │ │ │ │ │ - * endpoints. If false, for lines, only vertices that are not │ │ │ │ │ - * endpoints will be returned. If not provided, all vertices will │ │ │ │ │ - * be returned. │ │ │ │ │ - * │ │ │ │ │ + * Method: getBoxOffsets │ │ │ │ │ + * Determines border offsets for a box, according to the box model. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array} A list of all vertices in the geometry. │ │ │ │ │ + * {Object} an object with the following offsets: │ │ │ │ │ + * - left │ │ │ │ │ + * - right │ │ │ │ │ + * - top │ │ │ │ │ + * - bottom │ │ │ │ │ + * - width │ │ │ │ │ + * - height │ │ │ │ │ */ │ │ │ │ │ - getVertices: function(nodes) { │ │ │ │ │ - return [this]; │ │ │ │ │ + getBoxOffsets: function() { │ │ │ │ │ + if (!this.boxOffsets) { │ │ │ │ │ + // Determine the box model. If the testDiv's clientWidth is 3, then │ │ │ │ │ + // the borders are outside and we are dealing with the w3c box │ │ │ │ │ + // model. Otherwise, the browser uses the traditional box model and │ │ │ │ │ + // the borders are inside the box bounds, leaving us with a │ │ │ │ │ + // clientWidth of 1. │ │ │ │ │ + var testDiv = document.createElement("div"); │ │ │ │ │ + //testDiv.style.visibility = "hidden"; │ │ │ │ │ + testDiv.style.position = "absolute"; │ │ │ │ │ + testDiv.style.border = "1px solid black"; │ │ │ │ │ + testDiv.style.width = "3px"; │ │ │ │ │ + document.body.appendChild(testDiv); │ │ │ │ │ + var w3cBoxModel = testDiv.clientWidth == 3; │ │ │ │ │ + document.body.removeChild(testDiv); │ │ │ │ │ + │ │ │ │ │ + var left = parseInt(OpenLayers.Element.getStyle(this.zoomBox, │ │ │ │ │ + "border-left-width")); │ │ │ │ │ + var right = parseInt(OpenLayers.Element.getStyle( │ │ │ │ │ + this.zoomBox, "border-right-width")); │ │ │ │ │ + var top = parseInt(OpenLayers.Element.getStyle(this.zoomBox, │ │ │ │ │ + "border-top-width")); │ │ │ │ │ + var bottom = parseInt(OpenLayers.Element.getStyle( │ │ │ │ │ + this.zoomBox, "border-bottom-width")); │ │ │ │ │ + this.boxOffsets = { │ │ │ │ │ + left: left, │ │ │ │ │ + right: right, │ │ │ │ │ + top: top, │ │ │ │ │ + bottom: bottom, │ │ │ │ │ + width: w3cBoxModel === false ? left + right : 0, │ │ │ │ │ + height: w3cBoxModel === false ? top + bottom : 0 │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + return this.boxOffsets; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Geometry.Point" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Box" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Geometry/Collection.js │ │ │ │ │ + OpenLayers/Control/ZoomBox.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Geometry.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Handler/Box.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Geometry.Collection │ │ │ │ │ - * A Collection is exactly what it sounds like: A collection of different │ │ │ │ │ - * Geometries. These are stored in the local parameter <components> (which │ │ │ │ │ - * can be passed as a parameter to the constructor). │ │ │ │ │ - * │ │ │ │ │ - * As new geometries are added to the collection, they are NOT cloned. │ │ │ │ │ - * When removing geometries, they need to be specified by reference (ie you │ │ │ │ │ - * have to pass in the *exact* geometry to be removed). │ │ │ │ │ - * │ │ │ │ │ - * The <getArea> and <getLength> functions here merely iterate through │ │ │ │ │ - * the components, summing their respective areas and lengths. │ │ │ │ │ - * │ │ │ │ │ - * Create a new instance with the <OpenLayers.Geometry.Collection> constructor. │ │ │ │ │ + * Class: OpenLayers.Control.ZoomBox │ │ │ │ │ + * The ZoomBox control enables zooming directly to a given extent, by drawing │ │ │ │ │ + * a box on the map. The box is drawn by holding down shift, whilst dragging │ │ │ │ │ + * the mouse. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Geometry> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Geometry.Collection = OpenLayers.Class(OpenLayers.Geometry, { │ │ │ │ │ +OpenLayers.Control.ZoomBox = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + /** │ │ │ │ │ + * Property: type │ │ │ │ │ + * {OpenLayers.Control.TYPE} │ │ │ │ │ + */ │ │ │ │ │ + type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: components │ │ │ │ │ - * {Array(<OpenLayers.Geometry>)} The component parts of this geometry │ │ │ │ │ + * Property: out │ │ │ │ │ + * {Boolean} Should the control be used for zooming out? │ │ │ │ │ */ │ │ │ │ │ - components: null, │ │ │ │ │ + out: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: componentTypes │ │ │ │ │ - * {Array(String)} An array of class names representing the types of │ │ │ │ │ - * components that the collection can include. A null value means the │ │ │ │ │ - * component types are not restricted. │ │ │ │ │ + * APIProperty: keyMask │ │ │ │ │ + * {Integer} Zoom only occurs if the keyMask matches the combination of │ │ │ │ │ + * keys down. Use bitwise operators and one or more of the │ │ │ │ │ + * <OpenLayers.Handler> constants to construct a keyMask. Leave null if │ │ │ │ │ + * not used mask. Default is null. │ │ │ │ │ */ │ │ │ │ │ - componentTypes: null, │ │ │ │ │ + keyMask: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Geometry.Collection │ │ │ │ │ - * Creates a Geometry Collection -- a list of geoms. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * components - {Array(<OpenLayers.Geometry>)} Optional array of geometries │ │ │ │ │ - * │ │ │ │ │ + * APIProperty: alwaysZoom │ │ │ │ │ + * {Boolean} Always zoom in/out when box drawn, even if the zoom level does │ │ │ │ │ + * not change. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(components) { │ │ │ │ │ - OpenLayers.Geometry.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.components = []; │ │ │ │ │ - if (components != null) { │ │ │ │ │ - this.addComponents(components); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + alwaysZoom: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Destroy this geometry. │ │ │ │ │ + * APIProperty: zoomOnClick │ │ │ │ │ + * {Boolean} Should we zoom when no box was dragged, i.e. the user only │ │ │ │ │ + * clicked? Default is true. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.components.length = 0; │ │ │ │ │ - this.components = null; │ │ │ │ │ - OpenLayers.Geometry.prototype.destroy.apply(this, arguments); │ │ │ │ │ + zoomOnClick: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + */ │ │ │ │ │ + draw: function() { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Box(this, { │ │ │ │ │ + done: this.zoomBox │ │ │ │ │ + }, { │ │ │ │ │ + keyMask: this.keyMask │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Clone this geometry. │ │ │ │ │ + * Method: zoomBox │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Collection>} An exact clone of this collection │ │ │ │ │ - */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - var geometry = eval("new " + this.CLASS_NAME + "()"); │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ - geometry.addComponent(this.components[i].clone()); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // catch any randomly tagged-on properties │ │ │ │ │ - OpenLayers.Util.applyDefaults(geometry, this); │ │ │ │ │ - │ │ │ │ │ - return geometry; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getComponentsString │ │ │ │ │ - * Get a string representing the components for this collection │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string representation of the components of this geometry │ │ │ │ │ - */ │ │ │ │ │ - getComponentsString: function() { │ │ │ │ │ - var strings = []; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ - strings.push(this.components[i].toShortString()); │ │ │ │ │ - } │ │ │ │ │ - return strings.join(","); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: calculateBounds │ │ │ │ │ - * Recalculate the bounds by iterating through the components and │ │ │ │ │ - * calling calling extendBounds() on each item. │ │ │ │ │ - */ │ │ │ │ │ - calculateBounds: function() { │ │ │ │ │ - this.bounds = null; │ │ │ │ │ - var bounds = new OpenLayers.Bounds(); │ │ │ │ │ - var components = this.components; │ │ │ │ │ - if (components) { │ │ │ │ │ - for (var i = 0, len = components.length; i < len; i++) { │ │ │ │ │ - bounds.extend(components[i].getBounds()); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // to preserve old behavior, we only set bounds if non-null │ │ │ │ │ - // in the future, we could add bounds.isEmpty() │ │ │ │ │ - if (bounds.left != null && bounds.bottom != null && │ │ │ │ │ - bounds.right != null && bounds.top != null) { │ │ │ │ │ - this.setBounds(bounds); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: addComponents │ │ │ │ │ - * Add components to this geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * components - {Array(<OpenLayers.Geometry>)} An array of geometries to add │ │ │ │ │ - */ │ │ │ │ │ - addComponents: function(components) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(components))) { │ │ │ │ │ - components = [components]; │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = components.length; i < len; i++) { │ │ │ │ │ - this.addComponent(components[i]); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addComponent │ │ │ │ │ - * Add a new component (geometry) to the collection. If this.componentTypes │ │ │ │ │ - * is set, then the component class name must be in the componentTypes array. │ │ │ │ │ - * │ │ │ │ │ - * The bounds cache is reset. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * component - {<OpenLayers.Geometry>} A geometry to add │ │ │ │ │ - * index - {int} Optional index into the array to insert the component │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The component geometry was successfully added │ │ │ │ │ - */ │ │ │ │ │ - addComponent: function(component, index) { │ │ │ │ │ - var added = false; │ │ │ │ │ - if (component) { │ │ │ │ │ - if (this.componentTypes == null || │ │ │ │ │ - (OpenLayers.Util.indexOf(this.componentTypes, │ │ │ │ │ - component.CLASS_NAME) > -1)) { │ │ │ │ │ - │ │ │ │ │ - if (index != null && (index < this.components.length)) { │ │ │ │ │ - var components1 = this.components.slice(0, index); │ │ │ │ │ - var components2 = this.components.slice(index, │ │ │ │ │ - this.components.length); │ │ │ │ │ - components1.push(component); │ │ │ │ │ - this.components = components1.concat(components2); │ │ │ │ │ - } else { │ │ │ │ │ - this.components.push(component); │ │ │ │ │ - } │ │ │ │ │ - component.parent = this; │ │ │ │ │ - this.clearBounds(); │ │ │ │ │ - added = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return added; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: removeComponents │ │ │ │ │ - * Remove components from this geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * components - {Array(<OpenLayers.Geometry>)} The components to be removed │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} A component was removed. │ │ │ │ │ - */ │ │ │ │ │ - removeComponents: function(components) { │ │ │ │ │ - var removed = false; │ │ │ │ │ - │ │ │ │ │ - if (!(OpenLayers.Util.isArray(components))) { │ │ │ │ │ - components = [components]; │ │ │ │ │ - } │ │ │ │ │ - for (var i = components.length - 1; i >= 0; --i) { │ │ │ │ │ - removed = this.removeComponent(components[i]) || removed; │ │ │ │ │ - } │ │ │ │ │ - return removed; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeComponent │ │ │ │ │ - * Remove a component from this geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * component - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The component was removed. │ │ │ │ │ - */ │ │ │ │ │ - removeComponent: function(component) { │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Util.removeItem(this.components, component); │ │ │ │ │ - │ │ │ │ │ - // clearBounds() so that it gets recalculated on the next call │ │ │ │ │ - // to this.getBounds(); │ │ │ │ │ - this.clearBounds(); │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getLength │ │ │ │ │ - * Calculate the length of this geometry │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The length of the geometry │ │ │ │ │ - */ │ │ │ │ │ - getLength: function() { │ │ │ │ │ - var length = 0.0; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ - length += this.components[i].getLength(); │ │ │ │ │ - } │ │ │ │ │ - return length; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getArea │ │ │ │ │ - * Calculate the area of this geometry. Note how this function is overridden │ │ │ │ │ - * in <OpenLayers.Geometry.Polygon>. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The area of the collection by summing its parts │ │ │ │ │ - */ │ │ │ │ │ - getArea: function() { │ │ │ │ │ - var area = 0.0; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ - area += this.components[i].getArea(); │ │ │ │ │ - } │ │ │ │ │ - return area; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getGeodesicArea │ │ │ │ │ - * Calculate the approximate area of the polygon were it projected onto │ │ │ │ │ - * the earth. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * projection - {<OpenLayers.Projection>} The spatial reference system │ │ │ │ │ - * for the geometry coordinates. If not provided, Geographic/WGS84 is │ │ │ │ │ - * assumed. │ │ │ │ │ - * │ │ │ │ │ - * Reference: │ │ │ │ │ - * Robert. G. Chamberlain and William H. Duquette, "Some Algorithms for │ │ │ │ │ - * Polygons on a Sphere", JPL Publication 07-03, Jet Propulsion │ │ │ │ │ - * Laboratory, Pasadena, CA, June 2007 http://trs-new.jpl.nasa.gov/dspace/handle/2014/40409 │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {float} The approximate geodesic area of the geometry in square meters. │ │ │ │ │ - */ │ │ │ │ │ - getGeodesicArea: function(projection) { │ │ │ │ │ - var area = 0.0; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ - area += this.components[i].getGeodesicArea(projection); │ │ │ │ │ - } │ │ │ │ │ - return area; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getCentroid │ │ │ │ │ - * │ │ │ │ │ - * Compute the centroid for this geometry collection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * weighted - {Boolean} Perform the getCentroid computation recursively, │ │ │ │ │ - * returning an area weighted average of all geometries in this collection. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} The centroid of the collection │ │ │ │ │ - */ │ │ │ │ │ - getCentroid: function(weighted) { │ │ │ │ │ - if (!weighted) { │ │ │ │ │ - return this.components.length && this.components[0].getCentroid(); │ │ │ │ │ - } │ │ │ │ │ - var len = this.components.length; │ │ │ │ │ - if (!len) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var areas = []; │ │ │ │ │ - var centroids = []; │ │ │ │ │ - var areaSum = 0; │ │ │ │ │ - var minArea = Number.MAX_VALUE; │ │ │ │ │ - var component; │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - component = this.components[i]; │ │ │ │ │ - var area = component.getArea(); │ │ │ │ │ - var centroid = component.getCentroid(true); │ │ │ │ │ - if (isNaN(area) || isNaN(centroid.x) || isNaN(centroid.y)) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - areas.push(area); │ │ │ │ │ - areaSum += area; │ │ │ │ │ - minArea = (area < minArea && area > 0) ? area : minArea; │ │ │ │ │ - centroids.push(centroid); │ │ │ │ │ - } │ │ │ │ │ - len = areas.length; │ │ │ │ │ - if (areaSum === 0) { │ │ │ │ │ - // all the components in this collection have 0 area │ │ │ │ │ - // probably a collection of points -- weight all the points the same │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - areas[i] = 1; │ │ │ │ │ - } │ │ │ │ │ - areaSum = areas.length; │ │ │ │ │ - } else { │ │ │ │ │ - // normalize all the areas where the smallest area will get │ │ │ │ │ - // a value of 1 │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - areas[i] /= minArea; │ │ │ │ │ - } │ │ │ │ │ - areaSum /= minArea; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var xSum = 0, │ │ │ │ │ - ySum = 0, │ │ │ │ │ - centroid, area; │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - centroid = centroids[i]; │ │ │ │ │ - area = areas[i]; │ │ │ │ │ - xSum += centroid.x * area; │ │ │ │ │ - ySum += centroid.y * area; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return new OpenLayers.Geometry.Point(xSum / areaSum, ySum / areaSum); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getGeodesicLength │ │ │ │ │ - * Calculate the approximate length of the geometry were it projected onto │ │ │ │ │ - * the earth. │ │ │ │ │ - * │ │ │ │ │ - * projection - {<OpenLayers.Projection>} The spatial reference system │ │ │ │ │ - * for the geometry coordinates. If not provided, Geographic/WGS84 is │ │ │ │ │ - * assumed. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The appoximate geodesic length of the geometry in meters. │ │ │ │ │ - */ │ │ │ │ │ - getGeodesicLength: function(projection) { │ │ │ │ │ - var length = 0.0; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ - length += this.components[i].getGeodesicLength(projection); │ │ │ │ │ - } │ │ │ │ │ - return length; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: move │ │ │ │ │ - * Moves a geometry by the given displacement along positive x and y axes. │ │ │ │ │ - * This modifies the position of the geometry and clears the cached │ │ │ │ │ - * bounds. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {Float} Distance to move geometry in positive x direction. │ │ │ │ │ - * y - {Float} Distance to move geometry in positive y direction. │ │ │ │ │ - */ │ │ │ │ │ - move: function(x, y) { │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ - this.components[i].move(x, y); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: rotate │ │ │ │ │ - * Rotate a geometry around some origin │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * angle - {Float} Rotation angle in degrees (measured counterclockwise │ │ │ │ │ - * from the positive x-axis) │ │ │ │ │ - * origin - {<OpenLayers.Geometry.Point>} Center point for the rotation │ │ │ │ │ - */ │ │ │ │ │ - rotate: function(angle, origin) { │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ - this.components[i].rotate(angle, origin); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: resize │ │ │ │ │ - * Resize a geometry relative to some origin. Use this method to apply │ │ │ │ │ - * a uniform scaling to a geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * scale - {Float} Factor by which to scale the geometry. A scale of 2 │ │ │ │ │ - * doubles the size of the geometry in each dimension │ │ │ │ │ - * (lines, for example, will be twice as long, and polygons │ │ │ │ │ - * will have four times the area). │ │ │ │ │ - * origin - {<OpenLayers.Geometry.Point>} Point of origin for resizing │ │ │ │ │ - * ratio - {Float} Optional x:y ratio for resizing. Default ratio is 1. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} - The current geometry. │ │ │ │ │ - */ │ │ │ │ │ - resize: function(scale, origin, ratio) { │ │ │ │ │ - for (var i = 0; i < this.components.length; ++i) { │ │ │ │ │ - this.components[i].resize(scale, origin, ratio); │ │ │ │ │ - } │ │ │ │ │ - return this; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: distanceTo │ │ │ │ │ - * Calculate the closest distance between two geometries (on the x-y plane). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ - * options - {Object} Optional properties for configuring the distance │ │ │ │ │ - * calculation. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * details - {Boolean} Return details from the distance calculation. │ │ │ │ │ - * Default is false. │ │ │ │ │ - * edge - {Boolean} Calculate the distance from this geometry to the │ │ │ │ │ - * nearest edge of the target geometry. Default is true. If true, │ │ │ │ │ - * calling distanceTo from a geometry that is wholly contained within │ │ │ │ │ - * the target will result in a non-zero distance. If false, whenever │ │ │ │ │ - * geometries intersect, calling distanceTo will return 0. If false, │ │ │ │ │ - * details cannot be returned. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number | Object} The distance between this geometry and the target. │ │ │ │ │ - * If details is true, the return will be an object with distance, │ │ │ │ │ - * x0, y0, x1, and y1 properties. The x0 and y0 properties represent │ │ │ │ │ - * the coordinates of the closest point on this geometry. The x1 and y1 │ │ │ │ │ - * properties represent the coordinates of the closest point on the │ │ │ │ │ - * target geometry. │ │ │ │ │ - */ │ │ │ │ │ - distanceTo: function(geometry, options) { │ │ │ │ │ - var edge = !(options && options.edge === false); │ │ │ │ │ - var details = edge && options && options.details; │ │ │ │ │ - var result, best, distance; │ │ │ │ │ - var min = Number.POSITIVE_INFINITY; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ - result = this.components[i].distanceTo(geometry, options); │ │ │ │ │ - distance = details ? result.distance : result; │ │ │ │ │ - if (distance < min) { │ │ │ │ │ - min = distance; │ │ │ │ │ - best = result; │ │ │ │ │ - if (min == 0) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return best; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: equals │ │ │ │ │ - * Determine whether another geometry is equivalent to this one. Geometries │ │ │ │ │ - * are considered equivalent if all components have the same coordinates. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} The geometry to test. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The supplied geometry is equivalent to this geometry. │ │ │ │ │ - */ │ │ │ │ │ - equals: function(geometry) { │ │ │ │ │ - var equivalent = true; │ │ │ │ │ - if (!geometry || !geometry.CLASS_NAME || │ │ │ │ │ - (this.CLASS_NAME != geometry.CLASS_NAME)) { │ │ │ │ │ - equivalent = false; │ │ │ │ │ - } else if (!(OpenLayers.Util.isArray(geometry.components)) || │ │ │ │ │ - (geometry.components.length != this.components.length)) { │ │ │ │ │ - equivalent = false; │ │ │ │ │ - } else { │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ - if (!this.components[i].equals(geometry.components[i])) { │ │ │ │ │ - equivalent = false; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return equivalent; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: transform │ │ │ │ │ - * Reproject the components geometry from source to dest. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * source - {<OpenLayers.Projection>} │ │ │ │ │ - * dest - {<OpenLayers.Projection>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} │ │ │ │ │ - */ │ │ │ │ │ - transform: function(source, dest) { │ │ │ │ │ - if (source && dest) { │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ - var component = this.components[i]; │ │ │ │ │ - component.transform(source, dest); │ │ │ │ │ - } │ │ │ │ │ - this.bounds = null; │ │ │ │ │ - } │ │ │ │ │ - return this; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: intersects │ │ │ │ │ - * Determine if the input geometry intersects this one. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} Any type of geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The input geometry intersects this one. │ │ │ │ │ - */ │ │ │ │ │ - intersects: function(geometry) { │ │ │ │ │ - var intersect = false; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ - intersect = geometry.intersects(this.components[i]); │ │ │ │ │ - if (intersect) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return intersect; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getVertices │ │ │ │ │ - * Return a list of all points in this geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * nodes - {Boolean} For lines, only return vertices that are │ │ │ │ │ - * endpoints. If false, for lines, only vertices that are not │ │ │ │ │ - * endpoints will be returned. If not provided, all vertices will │ │ │ │ │ - * be returned. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} A list of all vertices in the geometry. │ │ │ │ │ - */ │ │ │ │ │ - getVertices: function(nodes) { │ │ │ │ │ - var vertices = []; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ - Array.prototype.push.apply( │ │ │ │ │ - vertices, this.components[i].getVertices(nodes) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return vertices; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Geometry.Collection" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Geometry/MultiPoint.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Geometry/Collection.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Geometry.MultiPoint │ │ │ │ │ - * MultiPoint is a collection of Points. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Geometry.MultiPoint> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Geometry.Collection> │ │ │ │ │ - * - <OpenLayers.Geometry> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.MultiPoint = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Geometry.Collection, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: componentTypes │ │ │ │ │ - * {Array(String)} An array of class names representing the types of │ │ │ │ │ - * components that the collection can include. A null value means the │ │ │ │ │ - * component types are not restricted. │ │ │ │ │ - */ │ │ │ │ │ - componentTypes: ["OpenLayers.Geometry.Point"], │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Geometry.MultiPoint │ │ │ │ │ - * Create a new MultiPoint Geometry │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * components - {Array(<OpenLayers.Geometry.Point>)} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.MultiPoint>} │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: addPoint │ │ │ │ │ - * Wrapper for <OpenLayers.Geometry.Collection.addComponent> │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} Point to be added │ │ │ │ │ - * index - {Integer} Optional index │ │ │ │ │ - */ │ │ │ │ │ - addPoint: function(point, index) { │ │ │ │ │ - this.addComponent(point, index); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: removePoint │ │ │ │ │ - * Wrapper for <OpenLayers.Geometry.Collection.removeComponent> │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} Point to be removed │ │ │ │ │ - */ │ │ │ │ │ - removePoint: function(point) { │ │ │ │ │ - this.removeComponent(point); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Geometry.MultiPoint" │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Geometry/Curve.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Geometry/MultiPoint.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Geometry.Curve │ │ │ │ │ - * A Curve is a MultiPoint, whose points are assumed to be connected. To │ │ │ │ │ - * this end, we provide a "getLength()" function, which iterates through │ │ │ │ │ - * the points, summing the distances between them. │ │ │ │ │ - * │ │ │ │ │ - * Inherits: │ │ │ │ │ - * - <OpenLayers.Geometry.MultiPoint> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.Curve = OpenLayers.Class(OpenLayers.Geometry.MultiPoint, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: componentTypes │ │ │ │ │ - * {Array(String)} An array of class names representing the types of │ │ │ │ │ - * components that the collection can include. A null │ │ │ │ │ - * value means the component types are not restricted. │ │ │ │ │ - */ │ │ │ │ │ - componentTypes: ["OpenLayers.Geometry.Point"], │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Geometry.Curve │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {Array(<OpenLayers.Geometry.Point>)} │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getLength │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The length of the curve │ │ │ │ │ - */ │ │ │ │ │ - getLength: function() { │ │ │ │ │ - var length = 0.0; │ │ │ │ │ - if (this.components && (this.components.length > 1)) { │ │ │ │ │ - for (var i = 1, len = this.components.length; i < len; i++) { │ │ │ │ │ - length += this.components[i - 1].distanceTo(this.components[i]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return length; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getGeodesicLength │ │ │ │ │ - * Calculate the approximate length of the geometry were it projected onto │ │ │ │ │ - * the earth. │ │ │ │ │ - * │ │ │ │ │ - * projection - {<OpenLayers.Projection>} The spatial reference system │ │ │ │ │ - * for the geometry coordinates. If not provided, Geographic/WGS84 is │ │ │ │ │ - * assumed. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The appoximate geodesic length of the geometry in meters. │ │ │ │ │ - */ │ │ │ │ │ - getGeodesicLength: function(projection) { │ │ │ │ │ - var geom = this; // so we can work with a clone if needed │ │ │ │ │ - if (projection) { │ │ │ │ │ - var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - if (!gg.equals(projection)) { │ │ │ │ │ - geom = this.clone().transform(projection, gg); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var length = 0.0; │ │ │ │ │ - if (geom.components && (geom.components.length > 1)) { │ │ │ │ │ - var p1, p2; │ │ │ │ │ - for (var i = 1, len = geom.components.length; i < len; i++) { │ │ │ │ │ - p1 = geom.components[i - 1]; │ │ │ │ │ - p2 = geom.components[i]; │ │ │ │ │ - // this returns km and requires lon/lat properties │ │ │ │ │ - length += OpenLayers.Util.distVincenty({ │ │ │ │ │ - lon: p1.x, │ │ │ │ │ - lat: p1.y │ │ │ │ │ - }, { │ │ │ │ │ - lon: p2.x, │ │ │ │ │ - lat: p2.y │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // convert to m │ │ │ │ │ - return length * 1000; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Geometry.Curve" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Geometry/LineString.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Geometry/Curve.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Geometry.LineString │ │ │ │ │ - * A LineString is a Curve which, once two points have been added to it, can │ │ │ │ │ - * never be less than two points long. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Geometry.Curve> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.LineString = OpenLayers.Class(OpenLayers.Geometry.Curve, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Geometry.LineString │ │ │ │ │ - * Create a new LineString geometry │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * points - {Array(<OpenLayers.Geometry.Point>)} An array of points used to │ │ │ │ │ - * generate the linestring │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: removeComponent │ │ │ │ │ - * Only allows removal of a point if there are three or more points in │ │ │ │ │ - * the linestring. (otherwise the result would be just a single point) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} The point to be removed │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The component was removed. │ │ │ │ │ - */ │ │ │ │ │ - removeComponent: function(point) { │ │ │ │ │ - var removed = this.components && (this.components.length > 2); │ │ │ │ │ - if (removed) { │ │ │ │ │ - OpenLayers.Geometry.Collection.prototype.removeComponent.apply(this, │ │ │ │ │ - arguments); │ │ │ │ │ - } │ │ │ │ │ - return removed; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: intersects │ │ │ │ │ - * Test for instersection between two geometries. This is a cheapo │ │ │ │ │ - * implementation of the Bently-Ottmann algorigithm. It doesn't │ │ │ │ │ - * really keep track of a sweep line data structure. It is closer │ │ │ │ │ - * to the brute force method, except that segments are sorted and │ │ │ │ │ - * potential intersections are only calculated when bounding boxes │ │ │ │ │ - * intersect. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The input geometry intersects this geometry. │ │ │ │ │ - */ │ │ │ │ │ - intersects: function(geometry) { │ │ │ │ │ - var intersect = false; │ │ │ │ │ - var type = geometry.CLASS_NAME; │ │ │ │ │ - if (type == "OpenLayers.Geometry.LineString" || │ │ │ │ │ - type == "OpenLayers.Geometry.LinearRing" || │ │ │ │ │ - type == "OpenLayers.Geometry.Point") { │ │ │ │ │ - var segs1 = this.getSortedSegments(); │ │ │ │ │ - var segs2; │ │ │ │ │ - if (type == "OpenLayers.Geometry.Point") { │ │ │ │ │ - segs2 = [{ │ │ │ │ │ - x1: geometry.x, │ │ │ │ │ - y1: geometry.y, │ │ │ │ │ - x2: geometry.x, │ │ │ │ │ - y2: geometry.y │ │ │ │ │ - }]; │ │ │ │ │ - } else { │ │ │ │ │ - segs2 = geometry.getSortedSegments(); │ │ │ │ │ - } │ │ │ │ │ - var seg1, seg1x1, seg1x2, seg1y1, seg1y2, │ │ │ │ │ - seg2, seg2y1, seg2y2; │ │ │ │ │ - // sweep right │ │ │ │ │ - outer: for (var i = 0, len = segs1.length; i < len; ++i) { │ │ │ │ │ - seg1 = segs1[i]; │ │ │ │ │ - seg1x1 = seg1.x1; │ │ │ │ │ - seg1x2 = seg1.x2; │ │ │ │ │ - seg1y1 = seg1.y1; │ │ │ │ │ - seg1y2 = seg1.y2; │ │ │ │ │ - inner: for (var j = 0, jlen = segs2.length; j < jlen; ++j) { │ │ │ │ │ - seg2 = segs2[j]; │ │ │ │ │ - if (seg2.x1 > seg1x2) { │ │ │ │ │ - // seg1 still left of seg2 │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - if (seg2.x2 < seg1x1) { │ │ │ │ │ - // seg2 still left of seg1 │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - seg2y1 = seg2.y1; │ │ │ │ │ - seg2y2 = seg2.y2; │ │ │ │ │ - if (Math.min(seg2y1, seg2y2) > Math.max(seg1y1, seg1y2)) { │ │ │ │ │ - // seg2 above seg1 │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - if (Math.max(seg2y1, seg2y2) < Math.min(seg1y1, seg1y2)) { │ │ │ │ │ - // seg2 below seg1 │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.Geometry.segmentsIntersect(seg1, seg2)) { │ │ │ │ │ - intersect = true; │ │ │ │ │ - break outer; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - intersect = geometry.intersects(this); │ │ │ │ │ - } │ │ │ │ │ - return intersect; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getSortedSegments │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} An array of segment objects. Segment objects have properties │ │ │ │ │ - * x1, y1, x2, and y2. The start point is represented by x1 and y1. │ │ │ │ │ - * The end point is represented by x2 and y2. Start and end are │ │ │ │ │ - * ordered so that x1 < x2. │ │ │ │ │ - */ │ │ │ │ │ - getSortedSegments: function() { │ │ │ │ │ - var numSeg = this.components.length - 1; │ │ │ │ │ - var segments = new Array(numSeg), │ │ │ │ │ - point1, point2; │ │ │ │ │ - for (var i = 0; i < numSeg; ++i) { │ │ │ │ │ - point1 = this.components[i]; │ │ │ │ │ - point2 = this.components[i + 1]; │ │ │ │ │ - if (point1.x < point2.x) { │ │ │ │ │ - segments[i] = { │ │ │ │ │ - x1: point1.x, │ │ │ │ │ - y1: point1.y, │ │ │ │ │ - x2: point2.x, │ │ │ │ │ - y2: point2.y │ │ │ │ │ - }; │ │ │ │ │ - } else { │ │ │ │ │ - segments[i] = { │ │ │ │ │ - x1: point2.x, │ │ │ │ │ - y1: point2.y, │ │ │ │ │ - x2: point1.x, │ │ │ │ │ - y2: point1.y │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // more efficient to define this somewhere static │ │ │ │ │ - function byX1(seg1, seg2) { │ │ │ │ │ - return seg1.x1 - seg2.x1; │ │ │ │ │ - } │ │ │ │ │ - return segments.sort(byX1); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: splitWithSegment │ │ │ │ │ - * Split this geometry with the given segment. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * seg - {Object} An object with x1, y1, x2, and y2 properties referencing │ │ │ │ │ - * segment endpoint coordinates. │ │ │ │ │ - * options - {Object} Properties of this object will be used to determine │ │ │ │ │ - * how the split is conducted. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * edge - {Boolean} Allow splitting when only edges intersect. Default is │ │ │ │ │ - * true. If false, a vertex on the source segment must be within the │ │ │ │ │ - * tolerance distance of the intersection to be considered a split. │ │ │ │ │ - * tolerance - {Number} If a non-null value is provided, intersections │ │ │ │ │ - * within the tolerance distance of one of the source segment's │ │ │ │ │ - * endpoints will be assumed to occur at the endpoint. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object with *lines* and *points* properties. If the given │ │ │ │ │ - * segment intersects this linestring, the lines array will reference │ │ │ │ │ - * geometries that result from the split. The points array will contain │ │ │ │ │ - * all intersection points. Intersection points are sorted along the │ │ │ │ │ - * segment (in order from x1,y1 to x2,y2). │ │ │ │ │ - */ │ │ │ │ │ - splitWithSegment: function(seg, options) { │ │ │ │ │ - var edge = !(options && options.edge === false); │ │ │ │ │ - var tolerance = options && options.tolerance; │ │ │ │ │ - var lines = []; │ │ │ │ │ - var verts = this.getVertices(); │ │ │ │ │ - var points = []; │ │ │ │ │ - var intersections = []; │ │ │ │ │ - var split = false; │ │ │ │ │ - var vert1, vert2, point; │ │ │ │ │ - var node, vertex, target; │ │ │ │ │ - var interOptions = { │ │ │ │ │ - point: true, │ │ │ │ │ - tolerance: tolerance │ │ │ │ │ - }; │ │ │ │ │ - var result = null; │ │ │ │ │ - for (var i = 0, stop = verts.length - 2; i <= stop; ++i) { │ │ │ │ │ - vert1 = verts[i]; │ │ │ │ │ - points.push(vert1.clone()); │ │ │ │ │ - vert2 = verts[i + 1]; │ │ │ │ │ - target = { │ │ │ │ │ - x1: vert1.x, │ │ │ │ │ - y1: vert1.y, │ │ │ │ │ - x2: vert2.x, │ │ │ │ │ - y2: vert2.y │ │ │ │ │ - }; │ │ │ │ │ - point = OpenLayers.Geometry.segmentsIntersect( │ │ │ │ │ - seg, target, interOptions │ │ │ │ │ - ); │ │ │ │ │ - if (point instanceof OpenLayers.Geometry.Point) { │ │ │ │ │ - if ((point.x === seg.x1 && point.y === seg.y1) || │ │ │ │ │ - (point.x === seg.x2 && point.y === seg.y2) || │ │ │ │ │ - point.equals(vert1) || point.equals(vert2)) { │ │ │ │ │ - vertex = true; │ │ │ │ │ - } else { │ │ │ │ │ - vertex = false; │ │ │ │ │ - } │ │ │ │ │ - if (vertex || edge) { │ │ │ │ │ - // push intersections different than the previous │ │ │ │ │ - if (!point.equals(intersections[intersections.length - 1])) { │ │ │ │ │ - intersections.push(point.clone()); │ │ │ │ │ - } │ │ │ │ │ - if (i === 0) { │ │ │ │ │ - if (point.equals(vert1)) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (point.equals(vert2)) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - split = true; │ │ │ │ │ - if (!point.equals(vert1)) { │ │ │ │ │ - points.push(point); │ │ │ │ │ - } │ │ │ │ │ - lines.push(new OpenLayers.Geometry.LineString(points)); │ │ │ │ │ - points = [point.clone()]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (split) { │ │ │ │ │ - points.push(vert2.clone()); │ │ │ │ │ - lines.push(new OpenLayers.Geometry.LineString(points)); │ │ │ │ │ - } │ │ │ │ │ - if (intersections.length > 0) { │ │ │ │ │ - // sort intersections along segment │ │ │ │ │ - var xDir = seg.x1 < seg.x2 ? 1 : -1; │ │ │ │ │ - var yDir = seg.y1 < seg.y2 ? 1 : -1; │ │ │ │ │ - result = { │ │ │ │ │ - lines: lines, │ │ │ │ │ - points: intersections.sort(function(p1, p2) { │ │ │ │ │ - return (xDir * p1.x - xDir * p2.x) || (yDir * p1.y - yDir * p2.y); │ │ │ │ │ - }) │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - return result; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: split │ │ │ │ │ - * Use this geometry (the source) to attempt to split a target geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * target - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ - * options - {Object} Properties of this object will be used to determine │ │ │ │ │ - * how the split is conducted. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * mutual - {Boolean} Split the source geometry in addition to the target │ │ │ │ │ - * geometry. Default is false. │ │ │ │ │ - * edge - {Boolean} Allow splitting when only edges intersect. Default is │ │ │ │ │ - * true. If false, a vertex on the source must be within the tolerance │ │ │ │ │ - * distance of the intersection to be considered a split. │ │ │ │ │ - * tolerance - {Number} If a non-null value is provided, intersections │ │ │ │ │ - * within the tolerance distance of an existing vertex on the source │ │ │ │ │ - * will be assumed to occur at the vertex. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} A list of geometries (of this same type as the target) that │ │ │ │ │ - * result from splitting the target with the source geometry. The │ │ │ │ │ - * source and target geometry will remain unmodified. If no split │ │ │ │ │ - * results, null will be returned. If mutual is true and a split │ │ │ │ │ - * results, return will be an array of two arrays - the first will be │ │ │ │ │ - * all geometries that result from splitting the source geometry and │ │ │ │ │ - * the second will be all geometries that result from splitting the │ │ │ │ │ - * target geometry. │ │ │ │ │ - */ │ │ │ │ │ - split: function(target, options) { │ │ │ │ │ - var results = null; │ │ │ │ │ - var mutual = options && options.mutual; │ │ │ │ │ - var sourceSplit, targetSplit, sourceParts, targetParts; │ │ │ │ │ - if (target instanceof OpenLayers.Geometry.LineString) { │ │ │ │ │ - var verts = this.getVertices(); │ │ │ │ │ - var vert1, vert2, seg, splits, lines, point; │ │ │ │ │ - var points = []; │ │ │ │ │ - sourceParts = []; │ │ │ │ │ - for (var i = 0, stop = verts.length - 2; i <= stop; ++i) { │ │ │ │ │ - vert1 = verts[i]; │ │ │ │ │ - vert2 = verts[i + 1]; │ │ │ │ │ - seg = { │ │ │ │ │ - x1: vert1.x, │ │ │ │ │ - y1: vert1.y, │ │ │ │ │ - x2: vert2.x, │ │ │ │ │ - y2: vert2.y │ │ │ │ │ - }; │ │ │ │ │ - targetParts = targetParts || [target]; │ │ │ │ │ - if (mutual) { │ │ │ │ │ - points.push(vert1.clone()); │ │ │ │ │ - } │ │ │ │ │ - for (var j = 0; j < targetParts.length; ++j) { │ │ │ │ │ - splits = targetParts[j].splitWithSegment(seg, options); │ │ │ │ │ - if (splits) { │ │ │ │ │ - // splice in new features │ │ │ │ │ - lines = splits.lines; │ │ │ │ │ - if (lines.length > 0) { │ │ │ │ │ - lines.unshift(j, 1); │ │ │ │ │ - Array.prototype.splice.apply(targetParts, lines); │ │ │ │ │ - j += lines.length - 2; │ │ │ │ │ - } │ │ │ │ │ - if (mutual) { │ │ │ │ │ - for (var k = 0, len = splits.points.length; k < len; ++k) { │ │ │ │ │ - point = splits.points[k]; │ │ │ │ │ - if (!point.equals(vert1)) { │ │ │ │ │ - points.push(point); │ │ │ │ │ - sourceParts.push(new OpenLayers.Geometry.LineString(points)); │ │ │ │ │ - if (point.equals(vert2)) { │ │ │ │ │ - points = []; │ │ │ │ │ - } else { │ │ │ │ │ - points = [point.clone()]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (mutual && sourceParts.length > 0 && points.length > 0) { │ │ │ │ │ - points.push(vert2.clone()); │ │ │ │ │ - sourceParts.push(new OpenLayers.Geometry.LineString(points)); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - results = target.splitWith(this, options); │ │ │ │ │ - } │ │ │ │ │ - if (targetParts && targetParts.length > 1) { │ │ │ │ │ - targetSplit = true; │ │ │ │ │ - } else { │ │ │ │ │ - targetParts = []; │ │ │ │ │ - } │ │ │ │ │ - if (sourceParts && sourceParts.length > 1) { │ │ │ │ │ - sourceSplit = true; │ │ │ │ │ - } else { │ │ │ │ │ - sourceParts = []; │ │ │ │ │ - } │ │ │ │ │ - if (targetSplit || sourceSplit) { │ │ │ │ │ - if (mutual) { │ │ │ │ │ - results = [sourceParts, targetParts]; │ │ │ │ │ - } else { │ │ │ │ │ - results = targetParts; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return results; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: splitWith │ │ │ │ │ - * Split this geometry (the target) with the given geometry (the source). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} A geometry used to split this │ │ │ │ │ - * geometry (the source). │ │ │ │ │ - * options - {Object} Properties of this object will be used to determine │ │ │ │ │ - * how the split is conducted. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * mutual - {Boolean} Split the source geometry in addition to the target │ │ │ │ │ - * geometry. Default is false. │ │ │ │ │ - * edge - {Boolean} Allow splitting when only edges intersect. Default is │ │ │ │ │ - * true. If false, a vertex on the source must be within the tolerance │ │ │ │ │ - * distance of the intersection to be considered a split. │ │ │ │ │ - * tolerance - {Number} If a non-null value is provided, intersections │ │ │ │ │ - * within the tolerance distance of an existing vertex on the source │ │ │ │ │ - * will be assumed to occur at the vertex. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} A list of geometries (of this same type as the target) that │ │ │ │ │ - * result from splitting the target with the source geometry. The │ │ │ │ │ - * source and target geometry will remain unmodified. If no split │ │ │ │ │ - * results, null will be returned. If mutual is true and a split │ │ │ │ │ - * results, return will be an array of two arrays - the first will be │ │ │ │ │ - * all geometries that result from splitting the source geometry and │ │ │ │ │ - * the second will be all geometries that result from splitting the │ │ │ │ │ - * target geometry. │ │ │ │ │ - */ │ │ │ │ │ - splitWith: function(geometry, options) { │ │ │ │ │ - return geometry.split(this, options); │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getVertices │ │ │ │ │ - * Return a list of all points in this geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * nodes - {Boolean} For lines, only return vertices that are │ │ │ │ │ - * endpoints. If false, for lines, only vertices that are not │ │ │ │ │ - * endpoints will be returned. If not provided, all vertices will │ │ │ │ │ - * be returned. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} A list of all vertices in the geometry. │ │ │ │ │ - */ │ │ │ │ │ - getVertices: function(nodes) { │ │ │ │ │ - var vertices; │ │ │ │ │ - if (nodes === true) { │ │ │ │ │ - vertices = [ │ │ │ │ │ - this.components[0], │ │ │ │ │ - this.components[this.components.length - 1] │ │ │ │ │ - ]; │ │ │ │ │ - } else if (nodes === false) { │ │ │ │ │ - vertices = this.components.slice(1, this.components.length - 1); │ │ │ │ │ - } else { │ │ │ │ │ - vertices = this.components.slice(); │ │ │ │ │ - } │ │ │ │ │ - return vertices; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: distanceTo │ │ │ │ │ - * Calculate the closest distance between two geometries (on the x-y plane). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ - * options - {Object} Optional properties for configuring the distance │ │ │ │ │ - * calculation. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * details - {Boolean} Return details from the distance calculation. │ │ │ │ │ - * Default is false. │ │ │ │ │ - * edge - {Boolean} Calculate the distance from this geometry to the │ │ │ │ │ - * nearest edge of the target geometry. Default is true. If true, │ │ │ │ │ - * calling distanceTo from a geometry that is wholly contained within │ │ │ │ │ - * the target will result in a non-zero distance. If false, whenever │ │ │ │ │ - * geometries intersect, calling distanceTo will return 0. If false, │ │ │ │ │ - * details cannot be returned. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number | Object} The distance between this geometry and the target. │ │ │ │ │ - * If details is true, the return will be an object with distance, │ │ │ │ │ - * x0, y0, x1, and x2 properties. The x0 and y0 properties represent │ │ │ │ │ - * the coordinates of the closest point on this geometry. The x1 and y1 │ │ │ │ │ - * properties represent the coordinates of the closest point on the │ │ │ │ │ - * target geometry. │ │ │ │ │ - */ │ │ │ │ │ - distanceTo: function(geometry, options) { │ │ │ │ │ - var edge = !(options && options.edge === false); │ │ │ │ │ - var details = edge && options && options.details; │ │ │ │ │ - var result, best = {}; │ │ │ │ │ - var min = Number.POSITIVE_INFINITY; │ │ │ │ │ - if (geometry instanceof OpenLayers.Geometry.Point) { │ │ │ │ │ - var segs = this.getSortedSegments(); │ │ │ │ │ - var x = geometry.x; │ │ │ │ │ - var y = geometry.y; │ │ │ │ │ - var seg; │ │ │ │ │ - for (var i = 0, len = segs.length; i < len; ++i) { │ │ │ │ │ - seg = segs[i]; │ │ │ │ │ - result = OpenLayers.Geometry.distanceToSegment(geometry, seg); │ │ │ │ │ - if (result.distance < min) { │ │ │ │ │ - min = result.distance; │ │ │ │ │ - best = result; │ │ │ │ │ - if (min === 0) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - // if distance increases and we cross y0 to the right of x0, no need to keep looking. │ │ │ │ │ - if (seg.x2 > x && ((y > seg.y1 && y < seg.y2) || (y < seg.y1 && y > seg.y2))) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (details) { │ │ │ │ │ - best = { │ │ │ │ │ - distance: best.distance, │ │ │ │ │ - x0: best.x, │ │ │ │ │ - y0: best.y, │ │ │ │ │ - x1: x, │ │ │ │ │ - y1: y │ │ │ │ │ - }; │ │ │ │ │ - } else { │ │ │ │ │ - best = best.distance; │ │ │ │ │ - } │ │ │ │ │ - } else if (geometry instanceof OpenLayers.Geometry.LineString) { │ │ │ │ │ - var segs0 = this.getSortedSegments(); │ │ │ │ │ - var segs1 = geometry.getSortedSegments(); │ │ │ │ │ - var seg0, seg1, intersection, x0, y0; │ │ │ │ │ - var len1 = segs1.length; │ │ │ │ │ - var interOptions = { │ │ │ │ │ - point: true │ │ │ │ │ - }; │ │ │ │ │ - outer: for (var i = 0, len = segs0.length; i < len; ++i) { │ │ │ │ │ - seg0 = segs0[i]; │ │ │ │ │ - x0 = seg0.x1; │ │ │ │ │ - y0 = seg0.y1; │ │ │ │ │ - for (var j = 0; j < len1; ++j) { │ │ │ │ │ - seg1 = segs1[j]; │ │ │ │ │ - intersection = OpenLayers.Geometry.segmentsIntersect(seg0, seg1, interOptions); │ │ │ │ │ - if (intersection) { │ │ │ │ │ - min = 0; │ │ │ │ │ - best = { │ │ │ │ │ - distance: 0, │ │ │ │ │ - x0: intersection.x, │ │ │ │ │ - y0: intersection.y, │ │ │ │ │ - x1: intersection.x, │ │ │ │ │ - y1: intersection.y │ │ │ │ │ - }; │ │ │ │ │ - break outer; │ │ │ │ │ - } else { │ │ │ │ │ - result = OpenLayers.Geometry.distanceToSegment({ │ │ │ │ │ - x: x0, │ │ │ │ │ - y: y0 │ │ │ │ │ - }, seg1); │ │ │ │ │ - if (result.distance < min) { │ │ │ │ │ - min = result.distance; │ │ │ │ │ - best = { │ │ │ │ │ - distance: min, │ │ │ │ │ - x0: x0, │ │ │ │ │ - y0: y0, │ │ │ │ │ - x1: result.x, │ │ │ │ │ - y1: result.y │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!details) { │ │ │ │ │ - best = best.distance; │ │ │ │ │ - } │ │ │ │ │ - if (min !== 0) { │ │ │ │ │ - // check the final vertex in this line's sorted segments │ │ │ │ │ - if (seg0) { │ │ │ │ │ - result = geometry.distanceTo( │ │ │ │ │ - new OpenLayers.Geometry.Point(seg0.x2, seg0.y2), │ │ │ │ │ - options │ │ │ │ │ - ); │ │ │ │ │ - var dist = details ? result.distance : result; │ │ │ │ │ - if (dist < min) { │ │ │ │ │ - if (details) { │ │ │ │ │ - best = { │ │ │ │ │ - distance: min, │ │ │ │ │ - x0: result.x1, │ │ │ │ │ - y0: result.y1, │ │ │ │ │ - x1: result.x0, │ │ │ │ │ - y1: result.y0 │ │ │ │ │ - }; │ │ │ │ │ - } else { │ │ │ │ │ - best = dist; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - best = geometry.distanceTo(this, options); │ │ │ │ │ - // swap since target comes from this line │ │ │ │ │ - if (details) { │ │ │ │ │ - best = { │ │ │ │ │ - distance: best.distance, │ │ │ │ │ - x0: best.x1, │ │ │ │ │ - y0: best.y1, │ │ │ │ │ - x1: best.x0, │ │ │ │ │ - y1: best.y0 │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return best; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: simplify │ │ │ │ │ - * This function will return a simplified LineString. │ │ │ │ │ - * Simplification is based on the Douglas-Peucker algorithm. │ │ │ │ │ - * │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * tolerance - {number} threshhold for simplification in map units │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {OpenLayers.Geometry.LineString} the simplified LineString │ │ │ │ │ - */ │ │ │ │ │ - simplify: function(tolerance) { │ │ │ │ │ - if (this && this !== null) { │ │ │ │ │ - var points = this.getVertices(); │ │ │ │ │ - if (points.length < 3) { │ │ │ │ │ - return this; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var compareNumbers = function(a, b) { │ │ │ │ │ - return (a - b); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Private function doing the Douglas-Peucker reduction │ │ │ │ │ - */ │ │ │ │ │ - var douglasPeuckerReduction = function(points, firstPoint, lastPoint, tolerance) { │ │ │ │ │ - var maxDistance = 0; │ │ │ │ │ - var indexFarthest = 0; │ │ │ │ │ - │ │ │ │ │ - for (var index = firstPoint, distance; index < lastPoint; index++) { │ │ │ │ │ - distance = perpendicularDistance(points[firstPoint], points[lastPoint], points[index]); │ │ │ │ │ - if (distance > maxDistance) { │ │ │ │ │ - maxDistance = distance; │ │ │ │ │ - indexFarthest = index; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (maxDistance > tolerance && indexFarthest != firstPoint) { │ │ │ │ │ - //Add the largest point that exceeds the tolerance │ │ │ │ │ - pointIndexsToKeep.push(indexFarthest); │ │ │ │ │ - douglasPeuckerReduction(points, firstPoint, indexFarthest, tolerance); │ │ │ │ │ - douglasPeuckerReduction(points, indexFarthest, lastPoint, tolerance); │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Private function calculating the perpendicular distance │ │ │ │ │ - * TODO: check whether OpenLayers.Geometry.LineString::distanceTo() is faster or slower │ │ │ │ │ - */ │ │ │ │ │ - var perpendicularDistance = function(point1, point2, point) { │ │ │ │ │ - //Area = |(1/2)(x1y2 + x2y3 + x3y1 - x2y1 - x3y2 - x1y3)| *Area of triangle │ │ │ │ │ - //Base = v((x1-x2)²+(x1-x2)²) *Base of Triangle* │ │ │ │ │ - //Area = .5*Base*H *Solve for height │ │ │ │ │ - //Height = Area/.5/Base │ │ │ │ │ - │ │ │ │ │ - var area = Math.abs(0.5 * (point1.x * point2.y + point2.x * point.y + point.x * point1.y - point2.x * point1.y - point.x * point2.y - point1.x * point.y)); │ │ │ │ │ - var bottom = Math.sqrt(Math.pow(point1.x - point2.x, 2) + Math.pow(point1.y - point2.y, 2)); │ │ │ │ │ - var height = area / bottom * 2; │ │ │ │ │ - │ │ │ │ │ - return height; │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - var firstPoint = 0; │ │ │ │ │ - var lastPoint = points.length - 1; │ │ │ │ │ - var pointIndexsToKeep = []; │ │ │ │ │ - │ │ │ │ │ - //Add the first and last index to the keepers │ │ │ │ │ - pointIndexsToKeep.push(firstPoint); │ │ │ │ │ - pointIndexsToKeep.push(lastPoint); │ │ │ │ │ - │ │ │ │ │ - //The first and the last point cannot be the same │ │ │ │ │ - while (points[firstPoint].equals(points[lastPoint])) { │ │ │ │ │ - lastPoint--; │ │ │ │ │ - //Addition: the first point not equal to first point in the LineString is kept as well │ │ │ │ │ - pointIndexsToKeep.push(lastPoint); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - douglasPeuckerReduction(points, firstPoint, lastPoint, tolerance); │ │ │ │ │ - var returnPoints = []; │ │ │ │ │ - pointIndexsToKeep.sort(compareNumbers); │ │ │ │ │ - for (var index = 0; index < pointIndexsToKeep.length; index++) { │ │ │ │ │ - returnPoints.push(points[pointIndexsToKeep[index]]); │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Geometry.LineString(returnPoints); │ │ │ │ │ - │ │ │ │ │ - } else { │ │ │ │ │ - return this; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Geometry.LineString" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Geometry/MultiLineString.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Geometry/Collection.js │ │ │ │ │ - * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Geometry.MultiLineString │ │ │ │ │ - * A MultiLineString is a geometry with multiple <OpenLayers.Geometry.LineString> │ │ │ │ │ - * components. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Geometry.Collection> │ │ │ │ │ - * - <OpenLayers.Geometry> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.MultiLineString = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Geometry.Collection, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: componentTypes │ │ │ │ │ - * {Array(String)} An array of class names representing the types of │ │ │ │ │ - * components that the collection can include. A null value means the │ │ │ │ │ - * component types are not restricted. │ │ │ │ │ - */ │ │ │ │ │ - componentTypes: ["OpenLayers.Geometry.LineString"], │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Geometry.MultiLineString │ │ │ │ │ - * Constructor for a MultiLineString Geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * components - {Array(<OpenLayers.Geometry.LineString>)} │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: split │ │ │ │ │ - * Use this geometry (the source) to attempt to split a target geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ - * options - {Object} Properties of this object will be used to determine │ │ │ │ │ - * how the split is conducted. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * mutual - {Boolean} Split the source geometry in addition to the target │ │ │ │ │ - * geometry. Default is false. │ │ │ │ │ - * edge - {Boolean} Allow splitting when only edges intersect. Default is │ │ │ │ │ - * true. If false, a vertex on the source must be within the tolerance │ │ │ │ │ - * distance of the intersection to be considered a split. │ │ │ │ │ - * tolerance - {Number} If a non-null value is provided, intersections │ │ │ │ │ - * within the tolerance distance of an existing vertex on the source │ │ │ │ │ - * will be assumed to occur at the vertex. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} A list of geometries (of this same type as the target) that │ │ │ │ │ - * result from splitting the target with the source geometry. The │ │ │ │ │ - * source and target geometry will remain unmodified. If no split │ │ │ │ │ - * results, null will be returned. If mutual is true and a split │ │ │ │ │ - * results, return will be an array of two arrays - the first will be │ │ │ │ │ - * all geometries that result from splitting the source geometry and │ │ │ │ │ - * the second will be all geometries that result from splitting the │ │ │ │ │ - * target geometry. │ │ │ │ │ - */ │ │ │ │ │ - split: function(geometry, options) { │ │ │ │ │ - var results = null; │ │ │ │ │ - var mutual = options && options.mutual; │ │ │ │ │ - var splits, sourceLine, sourceLines, sourceSplit, targetSplit; │ │ │ │ │ - var sourceParts = []; │ │ │ │ │ - var targetParts = [geometry]; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ - sourceLine = this.components[i]; │ │ │ │ │ - sourceSplit = false; │ │ │ │ │ - for (var j = 0; j < targetParts.length; ++j) { │ │ │ │ │ - splits = sourceLine.split(targetParts[j], options); │ │ │ │ │ - if (splits) { │ │ │ │ │ - if (mutual) { │ │ │ │ │ - sourceLines = splits[0]; │ │ │ │ │ - for (var k = 0, klen = sourceLines.length; k < klen; ++k) { │ │ │ │ │ - if (k === 0 && sourceParts.length) { │ │ │ │ │ - sourceParts[sourceParts.length - 1].addComponent( │ │ │ │ │ - sourceLines[k] │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - sourceParts.push( │ │ │ │ │ - new OpenLayers.Geometry.MultiLineString([ │ │ │ │ │ - sourceLines[k] │ │ │ │ │ - ]) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - sourceSplit = true; │ │ │ │ │ - splits = splits[1]; │ │ │ │ │ - } │ │ │ │ │ - if (splits.length) { │ │ │ │ │ - // splice in new target parts │ │ │ │ │ - splits.unshift(j, 1); │ │ │ │ │ - Array.prototype.splice.apply(targetParts, splits); │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!sourceSplit) { │ │ │ │ │ - // source line was not hit │ │ │ │ │ - if (sourceParts.length) { │ │ │ │ │ - // add line to existing multi │ │ │ │ │ - sourceParts[sourceParts.length - 1].addComponent( │ │ │ │ │ - sourceLine.clone() │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - // create a fresh multi │ │ │ │ │ - sourceParts = [ │ │ │ │ │ - new OpenLayers.Geometry.MultiLineString( │ │ │ │ │ - sourceLine.clone() │ │ │ │ │ - ) │ │ │ │ │ - ]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (sourceParts && sourceParts.length > 1) { │ │ │ │ │ - sourceSplit = true; │ │ │ │ │ - } else { │ │ │ │ │ - sourceParts = []; │ │ │ │ │ - } │ │ │ │ │ - if (targetParts && targetParts.length > 1) { │ │ │ │ │ - targetSplit = true; │ │ │ │ │ - } else { │ │ │ │ │ - targetParts = []; │ │ │ │ │ - } │ │ │ │ │ - if (sourceSplit || targetSplit) { │ │ │ │ │ - if (mutual) { │ │ │ │ │ - results = [sourceParts, targetParts]; │ │ │ │ │ - } else { │ │ │ │ │ - results = targetParts; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return results; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: splitWith │ │ │ │ │ - * Split this geometry (the target) with the given geometry (the source). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} A geometry used to split this │ │ │ │ │ - * geometry (the source). │ │ │ │ │ - * options - {Object} Properties of this object will be used to determine │ │ │ │ │ - * how the split is conducted. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * mutual - {Boolean} Split the source geometry in addition to the target │ │ │ │ │ - * geometry. Default is false. │ │ │ │ │ - * edge - {Boolean} Allow splitting when only edges intersect. Default is │ │ │ │ │ - * true. If false, a vertex on the source must be within the tolerance │ │ │ │ │ - * distance of the intersection to be considered a split. │ │ │ │ │ - * tolerance - {Number} If a non-null value is provided, intersections │ │ │ │ │ - * within the tolerance distance of an existing vertex on the source │ │ │ │ │ - * will be assumed to occur at the vertex. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} A list of geometries (of this same type as the target) that │ │ │ │ │ - * result from splitting the target with the source geometry. The │ │ │ │ │ - * source and target geometry will remain unmodified. If no split │ │ │ │ │ - * results, null will be returned. If mutual is true and a split │ │ │ │ │ - * results, return will be an array of two arrays - the first will be │ │ │ │ │ - * all geometries that result from splitting the source geometry and │ │ │ │ │ - * the second will be all geometries that result from splitting the │ │ │ │ │ - * target geometry. │ │ │ │ │ - */ │ │ │ │ │ - splitWith: function(geometry, options) { │ │ │ │ │ - var results = null; │ │ │ │ │ - var mutual = options && options.mutual; │ │ │ │ │ - var splits, targetLine, sourceLines, sourceSplit, targetSplit, sourceParts, targetParts; │ │ │ │ │ - if (geometry instanceof OpenLayers.Geometry.LineString) { │ │ │ │ │ - targetParts = []; │ │ │ │ │ - sourceParts = [geometry]; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ - targetSplit = false; │ │ │ │ │ - targetLine = this.components[i]; │ │ │ │ │ - for (var j = 0; j < sourceParts.length; ++j) { │ │ │ │ │ - splits = sourceParts[j].split(targetLine, options); │ │ │ │ │ - if (splits) { │ │ │ │ │ - if (mutual) { │ │ │ │ │ - sourceLines = splits[0]; │ │ │ │ │ - if (sourceLines.length) { │ │ │ │ │ - // splice in new source parts │ │ │ │ │ - sourceLines.unshift(j, 1); │ │ │ │ │ - Array.prototype.splice.apply(sourceParts, sourceLines); │ │ │ │ │ - j += sourceLines.length - 2; │ │ │ │ │ - } │ │ │ │ │ - splits = splits[1]; │ │ │ │ │ - if (splits.length === 0) { │ │ │ │ │ - splits = [targetLine.clone()]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - for (var k = 0, klen = splits.length; k < klen; ++k) { │ │ │ │ │ - if (k === 0 && targetParts.length) { │ │ │ │ │ - targetParts[targetParts.length - 1].addComponent( │ │ │ │ │ - splits[k] │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - targetParts.push( │ │ │ │ │ - new OpenLayers.Geometry.MultiLineString([ │ │ │ │ │ - splits[k] │ │ │ │ │ - ]) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - targetSplit = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!targetSplit) { │ │ │ │ │ - // target component was not hit │ │ │ │ │ - if (targetParts.length) { │ │ │ │ │ - // add it to any existing multi-line │ │ │ │ │ - targetParts[targetParts.length - 1].addComponent( │ │ │ │ │ - targetLine.clone() │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - // or start with a fresh multi-line │ │ │ │ │ - targetParts = [ │ │ │ │ │ - new OpenLayers.Geometry.MultiLineString([ │ │ │ │ │ - targetLine.clone() │ │ │ │ │ - ]) │ │ │ │ │ - ]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - results = geometry.split(this); │ │ │ │ │ - } │ │ │ │ │ - if (sourceParts && sourceParts.length > 1) { │ │ │ │ │ - sourceSplit = true; │ │ │ │ │ - } else { │ │ │ │ │ - sourceParts = []; │ │ │ │ │ - } │ │ │ │ │ - if (targetParts && targetParts.length > 1) { │ │ │ │ │ - targetSplit = true; │ │ │ │ │ - } else { │ │ │ │ │ - targetParts = []; │ │ │ │ │ - } │ │ │ │ │ - if (sourceSplit || targetSplit) { │ │ │ │ │ - if (mutual) { │ │ │ │ │ - results = [sourceParts, targetParts]; │ │ │ │ │ - } else { │ │ │ │ │ - results = targetParts; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return results; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Geometry.MultiLineString" │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Geometry/LinearRing.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Geometry.LinearRing │ │ │ │ │ - * │ │ │ │ │ - * A Linear Ring is a special LineString which is closed. It closes itself │ │ │ │ │ - * automatically on every addPoint/removePoint by adding a copy of the first │ │ │ │ │ - * point as the last point. │ │ │ │ │ - * │ │ │ │ │ - * Also, as it is the first in the line family to close itself, a getArea() │ │ │ │ │ - * function is defined to calculate the enclosed area of the linearRing │ │ │ │ │ - * │ │ │ │ │ - * Inherits: │ │ │ │ │ - * - <OpenLayers.Geometry.LineString> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.LinearRing = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Geometry.LineString, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: componentTypes │ │ │ │ │ - * {Array(String)} An array of class names representing the types of │ │ │ │ │ - * components that the collection can include. A null │ │ │ │ │ - * value means the component types are not restricted. │ │ │ │ │ - */ │ │ │ │ │ - componentTypes: ["OpenLayers.Geometry.Point"], │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Geometry.LinearRing │ │ │ │ │ - * Linear rings are constructed with an array of points. This array │ │ │ │ │ - * can represent a closed or open ring. If the ring is open (the last │ │ │ │ │ - * point does not equal the first point), the constructor will close │ │ │ │ │ - * the ring. If the ring is already closed (the last point does equal │ │ │ │ │ - * the first point), it will be left closed. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * points - {Array(<OpenLayers.Geometry.Point>)} points │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: addComponent │ │ │ │ │ - * Adds a point to geometry components. If the point is to be added to │ │ │ │ │ - * the end of the components array and it is the same as the last point │ │ │ │ │ - * already in that array, the duplicate point is not added. This has │ │ │ │ │ - * the effect of closing the ring if it is not already closed, and │ │ │ │ │ - * doing the right thing if it is already closed. This behavior can │ │ │ │ │ - * be overridden by calling the method with a non-null index as the │ │ │ │ │ - * second argument. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ - * index - {Integer} Index into the array to insert the component │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Was the Point successfully added? │ │ │ │ │ - */ │ │ │ │ │ - addComponent: function(point, index) { │ │ │ │ │ - var added = false; │ │ │ │ │ - │ │ │ │ │ - //remove last point │ │ │ │ │ - var lastPoint = this.components.pop(); │ │ │ │ │ - │ │ │ │ │ - // given an index, add the point │ │ │ │ │ - // without an index only add non-duplicate points │ │ │ │ │ - if (index != null || !point.equals(lastPoint)) { │ │ │ │ │ - added = OpenLayers.Geometry.Collection.prototype.addComponent.apply(this, │ │ │ │ │ - arguments); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //append copy of first point │ │ │ │ │ - var firstPoint = this.components[0]; │ │ │ │ │ - OpenLayers.Geometry.Collection.prototype.addComponent.apply(this, │ │ │ │ │ - [firstPoint]); │ │ │ │ │ - │ │ │ │ │ - return added; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: removeComponent │ │ │ │ │ - * Removes a point from geometry components. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The component was removed. │ │ │ │ │ - */ │ │ │ │ │ - removeComponent: function(point) { │ │ │ │ │ - var removed = this.components && (this.components.length > 3); │ │ │ │ │ - if (removed) { │ │ │ │ │ - //remove last point │ │ │ │ │ - this.components.pop(); │ │ │ │ │ - │ │ │ │ │ - //remove our point │ │ │ │ │ - OpenLayers.Geometry.Collection.prototype.removeComponent.apply(this, │ │ │ │ │ - arguments); │ │ │ │ │ - //append copy of first point │ │ │ │ │ - var firstPoint = this.components[0]; │ │ │ │ │ - OpenLayers.Geometry.Collection.prototype.addComponent.apply(this, │ │ │ │ │ - [firstPoint]); │ │ │ │ │ - } │ │ │ │ │ - return removed; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: move │ │ │ │ │ - * Moves a geometry by the given displacement along positive x and y axes. │ │ │ │ │ - * This modifies the position of the geometry and clears the cached │ │ │ │ │ - * bounds. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {Float} Distance to move geometry in positive x direction. │ │ │ │ │ - * y - {Float} Distance to move geometry in positive y direction. │ │ │ │ │ - */ │ │ │ │ │ - move: function(x, y) { │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len - 1; i++) { │ │ │ │ │ - this.components[i].move(x, y); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: rotate │ │ │ │ │ - * Rotate a geometry around some origin │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * angle - {Float} Rotation angle in degrees (measured counterclockwise │ │ │ │ │ - * from the positive x-axis) │ │ │ │ │ - * origin - {<OpenLayers.Geometry.Point>} Center point for the rotation │ │ │ │ │ - */ │ │ │ │ │ - rotate: function(angle, origin) { │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len - 1; ++i) { │ │ │ │ │ - this.components[i].rotate(angle, origin); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: resize │ │ │ │ │ - * Resize a geometry relative to some origin. Use this method to apply │ │ │ │ │ - * a uniform scaling to a geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * scale - {Float} Factor by which to scale the geometry. A scale of 2 │ │ │ │ │ - * doubles the size of the geometry in each dimension │ │ │ │ │ - * (lines, for example, will be twice as long, and polygons │ │ │ │ │ - * will have four times the area). │ │ │ │ │ - * origin - {<OpenLayers.Geometry.Point>} Point of origin for resizing │ │ │ │ │ - * ratio - {Float} Optional x:y ratio for resizing. Default ratio is 1. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} - The current geometry. │ │ │ │ │ - */ │ │ │ │ │ - resize: function(scale, origin, ratio) { │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len - 1; ++i) { │ │ │ │ │ - this.components[i].resize(scale, origin, ratio); │ │ │ │ │ - } │ │ │ │ │ - return this; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: transform │ │ │ │ │ - * Reproject the components geometry from source to dest. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * source - {<OpenLayers.Projection>} │ │ │ │ │ - * dest - {<OpenLayers.Projection>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} │ │ │ │ │ - */ │ │ │ │ │ - transform: function(source, dest) { │ │ │ │ │ - if (source && dest) { │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len - 1; i++) { │ │ │ │ │ - var component = this.components[i]; │ │ │ │ │ - component.transform(source, dest); │ │ │ │ │ - } │ │ │ │ │ - this.bounds = null; │ │ │ │ │ - } │ │ │ │ │ - return this; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getCentroid │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} The centroid of the collection │ │ │ │ │ - */ │ │ │ │ │ - getCentroid: function() { │ │ │ │ │ - if (this.components) { │ │ │ │ │ - var len = this.components.length; │ │ │ │ │ - if (len > 0 && len <= 2) { │ │ │ │ │ - return this.components[0].clone(); │ │ │ │ │ - } else if (len > 2) { │ │ │ │ │ - var sumX = 0.0; │ │ │ │ │ - var sumY = 0.0; │ │ │ │ │ - var x0 = this.components[0].x; │ │ │ │ │ - var y0 = this.components[0].y; │ │ │ │ │ - var area = -1 * this.getArea(); │ │ │ │ │ - if (area != 0) { │ │ │ │ │ - for (var i = 0; i < len - 1; i++) { │ │ │ │ │ - var b = this.components[i]; │ │ │ │ │ - var c = this.components[i + 1]; │ │ │ │ │ - sumX += (b.x + c.x - 2 * x0) * ((b.x - x0) * (c.y - y0) - (c.x - x0) * (b.y - y0)); │ │ │ │ │ - sumY += (b.y + c.y - 2 * y0) * ((b.x - x0) * (c.y - y0) - (c.x - x0) * (b.y - y0)); │ │ │ │ │ - } │ │ │ │ │ - var x = x0 + sumX / (6 * area); │ │ │ │ │ - var y = y0 + sumY / (6 * area); │ │ │ │ │ - } else { │ │ │ │ │ - for (var i = 0; i < len - 1; i++) { │ │ │ │ │ - sumX += this.components[i].x; │ │ │ │ │ - sumY += this.components[i].y; │ │ │ │ │ - } │ │ │ │ │ - var x = sumX / (len - 1); │ │ │ │ │ - var y = sumY / (len - 1); │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Geometry.Point(x, y); │ │ │ │ │ - } else { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getArea │ │ │ │ │ - * Note - The area is positive if the ring is oriented CW, otherwise │ │ │ │ │ - * it will be negative. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The signed area for a ring. │ │ │ │ │ - */ │ │ │ │ │ - getArea: function() { │ │ │ │ │ - var area = 0.0; │ │ │ │ │ - if (this.components && (this.components.length > 2)) { │ │ │ │ │ - var sum = 0.0; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len - 1; i++) { │ │ │ │ │ - var b = this.components[i]; │ │ │ │ │ - var c = this.components[i + 1]; │ │ │ │ │ - sum += (b.x + c.x) * (c.y - b.y); │ │ │ │ │ - } │ │ │ │ │ - area = -sum / 2.0; │ │ │ │ │ - } │ │ │ │ │ - return area; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getGeodesicArea │ │ │ │ │ - * Calculate the approximate area of the polygon were it projected onto │ │ │ │ │ - * the earth. Note that this area will be positive if ring is oriented │ │ │ │ │ - * clockwise, otherwise it will be negative. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * projection - {<OpenLayers.Projection>} The spatial reference system │ │ │ │ │ - * for the geometry coordinates. If not provided, Geographic/WGS84 is │ │ │ │ │ - * assumed. │ │ │ │ │ - * │ │ │ │ │ - * Reference: │ │ │ │ │ - * Robert. G. Chamberlain and William H. Duquette, "Some Algorithms for │ │ │ │ │ - * Polygons on a Sphere", JPL Publication 07-03, Jet Propulsion │ │ │ │ │ - * Laboratory, Pasadena, CA, June 2007 http://trs-new.jpl.nasa.gov/dspace/handle/2014/40409 │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {float} The approximate signed geodesic area of the polygon in square │ │ │ │ │ - * meters. │ │ │ │ │ - */ │ │ │ │ │ - getGeodesicArea: function(projection) { │ │ │ │ │ - var ring = this; // so we can work with a clone if needed │ │ │ │ │ - if (projection) { │ │ │ │ │ - var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - if (!gg.equals(projection)) { │ │ │ │ │ - ring = this.clone().transform(projection, gg); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var area = 0.0; │ │ │ │ │ - var len = ring.components && ring.components.length; │ │ │ │ │ - if (len > 2) { │ │ │ │ │ - var p1, p2; │ │ │ │ │ - for (var i = 0; i < len - 1; i++) { │ │ │ │ │ - p1 = ring.components[i]; │ │ │ │ │ - p2 = ring.components[i + 1]; │ │ │ │ │ - area += OpenLayers.Util.rad(p2.x - p1.x) * │ │ │ │ │ - (2 + Math.sin(OpenLayers.Util.rad(p1.y)) + │ │ │ │ │ - Math.sin(OpenLayers.Util.rad(p2.y))); │ │ │ │ │ - } │ │ │ │ │ - area = area * 6378137.0 * 6378137.0 / 2.0; │ │ │ │ │ - } │ │ │ │ │ - return area; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: containsPoint │ │ │ │ │ - * Test if a point is inside a linear ring. For the case where a point │ │ │ │ │ - * is coincident with a linear ring edge, returns 1. Otherwise, │ │ │ │ │ - * returns boolean. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean | Number} The point is inside the linear ring. Returns 1 if │ │ │ │ │ - * the point is coincident with an edge. Returns boolean otherwise. │ │ │ │ │ - */ │ │ │ │ │ - containsPoint: function(point) { │ │ │ │ │ - var approx = OpenLayers.Number.limitSigDigs; │ │ │ │ │ - var digs = 14; │ │ │ │ │ - var px = approx(point.x, digs); │ │ │ │ │ - var py = approx(point.y, digs); │ │ │ │ │ - │ │ │ │ │ - function getX(y, x1, y1, x2, y2) { │ │ │ │ │ - return (y - y2) * ((x2 - x1) / (y2 - y1)) + x2; │ │ │ │ │ - } │ │ │ │ │ - var numSeg = this.components.length - 1; │ │ │ │ │ - var start, end, x1, y1, x2, y2, cx, cy; │ │ │ │ │ - var crosses = 0; │ │ │ │ │ - for (var i = 0; i < numSeg; ++i) { │ │ │ │ │ - start = this.components[i]; │ │ │ │ │ - x1 = approx(start.x, digs); │ │ │ │ │ - y1 = approx(start.y, digs); │ │ │ │ │ - end = this.components[i + 1]; │ │ │ │ │ - x2 = approx(end.x, digs); │ │ │ │ │ - y2 = approx(end.y, digs); │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * The following conditions enforce five edge-crossing rules: │ │ │ │ │ - * 1. points coincident with edges are considered contained; │ │ │ │ │ - * 2. an upward edge includes its starting endpoint, and │ │ │ │ │ - * excludes its final endpoint; │ │ │ │ │ - * 3. a downward edge excludes its starting endpoint, and │ │ │ │ │ - * includes its final endpoint; │ │ │ │ │ - * 4. horizontal edges are excluded; and │ │ │ │ │ - * 5. the edge-ray intersection point must be strictly right │ │ │ │ │ - * of the point P. │ │ │ │ │ - */ │ │ │ │ │ - if (y1 == y2) { │ │ │ │ │ - // horizontal edge │ │ │ │ │ - if (py == y1) { │ │ │ │ │ - // point on horizontal line │ │ │ │ │ - if (x1 <= x2 && (px >= x1 && px <= x2) || // right or vert │ │ │ │ │ - x1 >= x2 && (px <= x1 && px >= x2)) { // left or vert │ │ │ │ │ - // point on edge │ │ │ │ │ - crosses = -1; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // ignore other horizontal edges │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - cx = approx(getX(py, x1, y1, x2, y2), digs); │ │ │ │ │ - if (cx == px) { │ │ │ │ │ - // point on line │ │ │ │ │ - if (y1 < y2 && (py >= y1 && py <= y2) || // upward │ │ │ │ │ - y1 > y2 && (py <= y1 && py >= y2)) { // downward │ │ │ │ │ - // point on edge │ │ │ │ │ - crosses = -1; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (cx <= px) { │ │ │ │ │ - // no crossing to the right │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - if (x1 != x2 && (cx < Math.min(x1, x2) || cx > Math.max(x1, x2))) { │ │ │ │ │ - // no crossing │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - if (y1 < y2 && (py >= y1 && py < y2) || // upward │ │ │ │ │ - y1 > y2 && (py < y1 && py >= y2)) { // downward │ │ │ │ │ - ++crosses; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var contained = (crosses == -1) ? │ │ │ │ │ - // on edge │ │ │ │ │ - 1 : │ │ │ │ │ - // even (out) or odd (in) │ │ │ │ │ - !!(crosses & 1); │ │ │ │ │ - │ │ │ │ │ - return contained; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: intersects │ │ │ │ │ - * Determine if the input geometry intersects this one. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} Any type of geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The input geometry intersects this one. │ │ │ │ │ - */ │ │ │ │ │ - intersects: function(geometry) { │ │ │ │ │ - var intersect = false; │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - intersect = this.containsPoint(geometry); │ │ │ │ │ - } else if (geometry.CLASS_NAME == "OpenLayers.Geometry.LineString") { │ │ │ │ │ - intersect = geometry.intersects(this); │ │ │ │ │ - } else if (geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ - intersect = OpenLayers.Geometry.LineString.prototype.intersects.apply( │ │ │ │ │ - this, [geometry] │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - // check for component intersections │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; ++i) { │ │ │ │ │ - intersect = geometry.components[i].intersects(this); │ │ │ │ │ - if (intersect) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return intersect; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getVertices │ │ │ │ │ - * Return a list of all points in this geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * nodes - {Boolean} For lines, only return vertices that are │ │ │ │ │ - * endpoints. If false, for lines, only vertices that are not │ │ │ │ │ - * endpoints will be returned. If not provided, all vertices will │ │ │ │ │ - * be returned. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} A list of all vertices in the geometry. │ │ │ │ │ - */ │ │ │ │ │ - getVertices: function(nodes) { │ │ │ │ │ - return (nodes === true) ? [] : this.components.slice(0, this.components.length - 1); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Geometry.LinearRing" │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Geometry/Polygon.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Geometry/Collection.js │ │ │ │ │ - * @requires OpenLayers/Geometry/LinearRing.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Geometry.Polygon │ │ │ │ │ - * Polygon is a collection of Geometry.LinearRings. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Geometry.Collection> │ │ │ │ │ - * - <OpenLayers.Geometry> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.Polygon = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Geometry.Collection, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: componentTypes │ │ │ │ │ - * {Array(String)} An array of class names representing the types of │ │ │ │ │ - * components that the collection can include. A null value means the │ │ │ │ │ - * component types are not restricted. │ │ │ │ │ - */ │ │ │ │ │ - componentTypes: ["OpenLayers.Geometry.LinearRing"], │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Geometry.Polygon │ │ │ │ │ - * Constructor for a Polygon geometry. │ │ │ │ │ - * The first ring (this.component[0])is the outer bounds of the polygon and │ │ │ │ │ - * all subsequent rings (this.component[1-n]) are internal holes. │ │ │ │ │ - * │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * components - {Array(<OpenLayers.Geometry.LinearRing>)} │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getArea │ │ │ │ │ - * Calculated by subtracting the areas of the internal holes from the │ │ │ │ │ - * area of the outer hole. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {float} The area of the geometry │ │ │ │ │ - */ │ │ │ │ │ - getArea: function() { │ │ │ │ │ - var area = 0.0; │ │ │ │ │ - if (this.components && (this.components.length > 0)) { │ │ │ │ │ - area += Math.abs(this.components[0].getArea()); │ │ │ │ │ - for (var i = 1, len = this.components.length; i < len; i++) { │ │ │ │ │ - area -= Math.abs(this.components[i].getArea()); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return area; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getGeodesicArea │ │ │ │ │ - * Calculate the approximate area of the polygon were it projected onto │ │ │ │ │ - * the earth. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * projection - {<OpenLayers.Projection>} The spatial reference system │ │ │ │ │ - * for the geometry coordinates. If not provided, Geographic/WGS84 is │ │ │ │ │ - * assumed. │ │ │ │ │ - * │ │ │ │ │ - * Reference: │ │ │ │ │ - * Robert. G. Chamberlain and William H. Duquette, "Some Algorithms for │ │ │ │ │ - * Polygons on a Sphere", JPL Publication 07-03, Jet Propulsion │ │ │ │ │ - * Laboratory, Pasadena, CA, June 2007 http://trs-new.jpl.nasa.gov/dspace/handle/2014/40409 │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {float} The approximate geodesic area of the polygon in square meters. │ │ │ │ │ - */ │ │ │ │ │ - getGeodesicArea: function(projection) { │ │ │ │ │ - var area = 0.0; │ │ │ │ │ - if (this.components && (this.components.length > 0)) { │ │ │ │ │ - area += Math.abs(this.components[0].getGeodesicArea(projection)); │ │ │ │ │ - for (var i = 1, len = this.components.length; i < len; i++) { │ │ │ │ │ - area -= Math.abs(this.components[i].getGeodesicArea(projection)); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return area; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: containsPoint │ │ │ │ │ - * Test if a point is inside a polygon. Points on a polygon edge are │ │ │ │ │ - * considered inside. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean | Number} The point is inside the polygon. Returns 1 if the │ │ │ │ │ - * point is on an edge. Returns boolean otherwise. │ │ │ │ │ - */ │ │ │ │ │ - containsPoint: function(point) { │ │ │ │ │ - var numRings = this.components.length; │ │ │ │ │ - var contained = false; │ │ │ │ │ - if (numRings > 0) { │ │ │ │ │ - // check exterior ring - 1 means on edge, boolean otherwise │ │ │ │ │ - contained = this.components[0].containsPoint(point); │ │ │ │ │ - if (contained !== 1) { │ │ │ │ │ - if (contained && numRings > 1) { │ │ │ │ │ - // check interior rings │ │ │ │ │ - var hole; │ │ │ │ │ - for (var i = 1; i < numRings; ++i) { │ │ │ │ │ - hole = this.components[i].containsPoint(point); │ │ │ │ │ - if (hole) { │ │ │ │ │ - if (hole === 1) { │ │ │ │ │ - // on edge │ │ │ │ │ - contained = 1; │ │ │ │ │ - } else { │ │ │ │ │ - // in hole │ │ │ │ │ - contained = false; │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return contained; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: intersects │ │ │ │ │ - * Determine if the input geometry intersects this one. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} Any type of geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The input geometry intersects this one. │ │ │ │ │ - */ │ │ │ │ │ - intersects: function(geometry) { │ │ │ │ │ - var intersect = false; │ │ │ │ │ - var i, len; │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - intersect = this.containsPoint(geometry); │ │ │ │ │ - } else if (geometry.CLASS_NAME == "OpenLayers.Geometry.LineString" || │ │ │ │ │ - geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ - // check if rings/linestrings intersect │ │ │ │ │ - for (i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ - intersect = geometry.intersects(this.components[i]); │ │ │ │ │ - if (intersect) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!intersect) { │ │ │ │ │ - // check if this poly contains points of the ring/linestring │ │ │ │ │ - for (i = 0, len = geometry.components.length; i < len; ++i) { │ │ │ │ │ - intersect = this.containsPoint(geometry.components[i]); │ │ │ │ │ - if (intersect) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - for (i = 0, len = geometry.components.length; i < len; ++i) { │ │ │ │ │ - intersect = this.intersects(geometry.components[i]); │ │ │ │ │ - if (intersect) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // check case where this poly is wholly contained by another │ │ │ │ │ - if (!intersect && geometry.CLASS_NAME == "OpenLayers.Geometry.Polygon") { │ │ │ │ │ - // exterior ring points will be contained in the other geometry │ │ │ │ │ - var ring = this.components[0]; │ │ │ │ │ - for (i = 0, len = ring.components.length; i < len; ++i) { │ │ │ │ │ - intersect = geometry.containsPoint(ring.components[i]); │ │ │ │ │ - if (intersect) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return intersect; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: distanceTo │ │ │ │ │ - * Calculate the closest distance between two geometries (on the x-y plane). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ - * options - {Object} Optional properties for configuring the distance │ │ │ │ │ - * calculation. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * details - {Boolean} Return details from the distance calculation. │ │ │ │ │ - * Default is false. │ │ │ │ │ - * edge - {Boolean} Calculate the distance from this geometry to the │ │ │ │ │ - * nearest edge of the target geometry. Default is true. If true, │ │ │ │ │ - * calling distanceTo from a geometry that is wholly contained within │ │ │ │ │ - * the target will result in a non-zero distance. If false, whenever │ │ │ │ │ - * geometries intersect, calling distanceTo will return 0. If false, │ │ │ │ │ - * details cannot be returned. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number | Object} The distance between this geometry and the target. │ │ │ │ │ - * If details is true, the return will be an object with distance, │ │ │ │ │ - * x0, y0, x1, and y1 properties. The x0 and y0 properties represent │ │ │ │ │ - * the coordinates of the closest point on this geometry. The x1 and y1 │ │ │ │ │ - * properties represent the coordinates of the closest point on the │ │ │ │ │ - * target geometry. │ │ │ │ │ - */ │ │ │ │ │ - distanceTo: function(geometry, options) { │ │ │ │ │ - var edge = !(options && options.edge === false); │ │ │ │ │ - var result; │ │ │ │ │ - // this is the case where we might not be looking for distance to edge │ │ │ │ │ - if (!edge && this.intersects(geometry)) { │ │ │ │ │ - result = 0; │ │ │ │ │ - } else { │ │ │ │ │ - result = OpenLayers.Geometry.Collection.prototype.distanceTo.apply( │ │ │ │ │ - this, [geometry, options] │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return result; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Geometry.Polygon" │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * APIMethod: createRegularPolygon │ │ │ │ │ - * Create a regular polygon around a radius. Useful for creating circles │ │ │ │ │ - * and the like. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * origin - {<OpenLayers.Geometry.Point>} center of polygon. │ │ │ │ │ - * radius - {Float} distance to vertex, in map units. │ │ │ │ │ - * sides - {Integer} Number of sides. 20 approximates a circle. │ │ │ │ │ - * rotation - {Float} original angle of rotation, in degrees. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.Polygon.createRegularPolygon = function(origin, radius, sides, rotation) { │ │ │ │ │ - var angle = Math.PI * ((1 / sides) - (1 / 2)); │ │ │ │ │ - if (rotation) { │ │ │ │ │ - angle += (rotation / 180) * Math.PI; │ │ │ │ │ - } │ │ │ │ │ - var rotatedAngle, x, y; │ │ │ │ │ - var points = []; │ │ │ │ │ - for (var i = 0; i < sides; ++i) { │ │ │ │ │ - rotatedAngle = angle + (i * 2 * Math.PI / sides); │ │ │ │ │ - x = origin.x + (radius * Math.cos(rotatedAngle)); │ │ │ │ │ - y = origin.y + (radius * Math.sin(rotatedAngle)); │ │ │ │ │ - points.push(new OpenLayers.Geometry.Point(x, y)); │ │ │ │ │ - } │ │ │ │ │ - var ring = new OpenLayers.Geometry.LinearRing(points); │ │ │ │ │ - return new OpenLayers.Geometry.Polygon([ring]); │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Geometry/MultiPolygon.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Geometry/Collection.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Geometry.MultiPolygon │ │ │ │ │ - * MultiPolygon is a geometry with multiple <OpenLayers.Geometry.Polygon> │ │ │ │ │ - * components. Create a new instance with the <OpenLayers.Geometry.MultiPolygon> │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Geometry.Collection> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.MultiPolygon = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Geometry.Collection, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: componentTypes │ │ │ │ │ - * {Array(String)} An array of class names representing the types of │ │ │ │ │ - * components that the collection can include. A null value means the │ │ │ │ │ - * component types are not restricted. │ │ │ │ │ - */ │ │ │ │ │ - componentTypes: ["OpenLayers.Geometry.Polygon"], │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Geometry.MultiPolygon │ │ │ │ │ - * Create a new MultiPolygon geometry │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * components - {Array(<OpenLayers.Geometry.Polygon>)} An array of polygons │ │ │ │ │ - * used to generate the MultiPolygon │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Geometry.MultiPolygon" │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/GeoJSON.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/JSON.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/MultiPoint.js │ │ │ │ │ - * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ - * @requires OpenLayers/Geometry/MultiLineString.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ - * @requires OpenLayers/Geometry/MultiPolygon.js │ │ │ │ │ - * @requires OpenLayers/Console.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.GeoJSON │ │ │ │ │ - * Read and write GeoJSON. Create a new parser with the │ │ │ │ │ - * <OpenLayers.Format.GeoJSON> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.JSON> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.GeoJSON = OpenLayers.Class(OpenLayers.Format.JSON, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: ignoreExtraDims │ │ │ │ │ - * {Boolean} Ignore dimensions higher than 2 when reading geometry │ │ │ │ │ - * coordinates. │ │ │ │ │ - */ │ │ │ │ │ - ignoreExtraDims: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.GeoJSON │ │ │ │ │ - * Create a new parser for GeoJSON. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Deserialize a GeoJSON string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * json - {String} A GeoJSON string │ │ │ │ │ - * type - {String} Optional string that determines the structure of │ │ │ │ │ - * the output. Supported values are "Geometry", "Feature", and │ │ │ │ │ - * "FeatureCollection". If absent or null, a default of │ │ │ │ │ - * "FeatureCollection" is assumed. │ │ │ │ │ - * filter - {Function} A function which will be called for every key and │ │ │ │ │ - * value at every level of the final result. Each value will be │ │ │ │ │ - * replaced by the result of the filter function. This can be used to │ │ │ │ │ - * reform generic objects into instances of classes, or to transform │ │ │ │ │ - * date strings into Date objects. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The return depends on the value of the type argument. If type │ │ │ │ │ - * is "FeatureCollection" (the default), the return will be an array │ │ │ │ │ - * of <OpenLayers.Feature.Vector>. If type is "Geometry", the input json │ │ │ │ │ - * must represent a single geometry, and the return will be an │ │ │ │ │ - * <OpenLayers.Geometry>. If type is "Feature", the input json must │ │ │ │ │ - * represent a single feature, and the return will be an │ │ │ │ │ - * <OpenLayers.Feature.Vector>. │ │ │ │ │ - */ │ │ │ │ │ - read: function(json, type, filter) { │ │ │ │ │ - type = (type) ? type : "FeatureCollection"; │ │ │ │ │ - var results = null; │ │ │ │ │ - var obj = null; │ │ │ │ │ - if (typeof json == "string") { │ │ │ │ │ - obj = OpenLayers.Format.JSON.prototype.read.apply(this, │ │ │ │ │ - [json, filter]); │ │ │ │ │ - } else { │ │ │ │ │ - obj = json; │ │ │ │ │ - } │ │ │ │ │ - if (!obj) { │ │ │ │ │ - OpenLayers.Console.error("Bad JSON: " + json); │ │ │ │ │ - } else if (typeof(obj.type) != "string") { │ │ │ │ │ - OpenLayers.Console.error("Bad GeoJSON - no type: " + json); │ │ │ │ │ - } else if (this.isValidType(obj, type)) { │ │ │ │ │ - switch (type) { │ │ │ │ │ - case "Geometry": │ │ │ │ │ - try { │ │ │ │ │ - results = this.parseGeometry(obj); │ │ │ │ │ - } catch (err) { │ │ │ │ │ - OpenLayers.Console.error(err); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "Feature": │ │ │ │ │ - try { │ │ │ │ │ - results = this.parseFeature(obj); │ │ │ │ │ - results.type = "Feature"; │ │ │ │ │ - } catch (err) { │ │ │ │ │ - OpenLayers.Console.error(err); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "FeatureCollection": │ │ │ │ │ - // for type FeatureCollection, we allow input to be any type │ │ │ │ │ - results = []; │ │ │ │ │ - switch (obj.type) { │ │ │ │ │ - case "Feature": │ │ │ │ │ - try { │ │ │ │ │ - results.push(this.parseFeature(obj)); │ │ │ │ │ - } catch (err) { │ │ │ │ │ - results = null; │ │ │ │ │ - OpenLayers.Console.error(err); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "FeatureCollection": │ │ │ │ │ - for (var i = 0, len = obj.features.length; i < len; ++i) { │ │ │ │ │ - try { │ │ │ │ │ - results.push(this.parseFeature(obj.features[i])); │ │ │ │ │ - } catch (err) { │ │ │ │ │ - results = null; │ │ │ │ │ - OpenLayers.Console.error(err); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - try { │ │ │ │ │ - var geom = this.parseGeometry(obj); │ │ │ │ │ - results.push(new OpenLayers.Feature.Vector(geom)); │ │ │ │ │ - } catch (err) { │ │ │ │ │ - results = null; │ │ │ │ │ - OpenLayers.Console.error(err); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return results; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: isValidType │ │ │ │ │ - * Check if a GeoJSON object is a valid representative of the given type. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The object is valid GeoJSON object of the given type. │ │ │ │ │ - */ │ │ │ │ │ - isValidType: function(obj, type) { │ │ │ │ │ - var valid = false; │ │ │ │ │ - switch (type) { │ │ │ │ │ - case "Geometry": │ │ │ │ │ - if (OpenLayers.Util.indexOf( │ │ │ │ │ - ["Point", "MultiPoint", "LineString", "MultiLineString", │ │ │ │ │ - "Polygon", "MultiPolygon", "Box", "GeometryCollection" │ │ │ │ │ - ], │ │ │ │ │ - obj.type) == -1) { │ │ │ │ │ - // unsupported geometry type │ │ │ │ │ - OpenLayers.Console.error("Unsupported geometry type: " + │ │ │ │ │ - obj.type); │ │ │ │ │ - } else { │ │ │ │ │ - valid = true; │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "FeatureCollection": │ │ │ │ │ - // allow for any type to be converted to a feature collection │ │ │ │ │ - valid = true; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - // for Feature types must match │ │ │ │ │ - if (obj.type == type) { │ │ │ │ │ - valid = true; │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Console.error("Cannot convert types from " + │ │ │ │ │ - obj.type + " to " + type); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return valid; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseFeature │ │ │ │ │ - * Convert a feature object from GeoJSON into an │ │ │ │ │ - * <OpenLayers.Feature.Vector>. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} An object created from a GeoJSON object │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A feature. │ │ │ │ │ - */ │ │ │ │ │ - parseFeature: function(obj) { │ │ │ │ │ - var feature, geometry, attributes, bbox; │ │ │ │ │ - attributes = (obj.properties) ? obj.properties : {}; │ │ │ │ │ - bbox = (obj.geometry && obj.geometry.bbox) || obj.bbox; │ │ │ │ │ - try { │ │ │ │ │ - geometry = this.parseGeometry(obj.geometry); │ │ │ │ │ - } catch (err) { │ │ │ │ │ - // deal with bad geometries │ │ │ │ │ - throw err; │ │ │ │ │ - } │ │ │ │ │ - feature = new OpenLayers.Feature.Vector(geometry, attributes); │ │ │ │ │ - if (bbox) { │ │ │ │ │ - feature.bounds = OpenLayers.Bounds.fromArray(bbox); │ │ │ │ │ - } │ │ │ │ │ - if (obj.id) { │ │ │ │ │ - feature.fid = obj.id; │ │ │ │ │ - } │ │ │ │ │ - return feature; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseGeometry │ │ │ │ │ - * Convert a geometry object from GeoJSON into an <OpenLayers.Geometry>. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} An object created from a GeoJSON object │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} A geometry. │ │ │ │ │ - */ │ │ │ │ │ - parseGeometry: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - var geometry, collection = false; │ │ │ │ │ - if (obj.type == "GeometryCollection") { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(obj.geometries))) { │ │ │ │ │ - throw "GeometryCollection must have geometries array: " + obj; │ │ │ │ │ - } │ │ │ │ │ - var numGeom = obj.geometries.length; │ │ │ │ │ - var components = new Array(numGeom); │ │ │ │ │ - for (var i = 0; i < numGeom; ++i) { │ │ │ │ │ - components[i] = this.parseGeometry.apply( │ │ │ │ │ - this, [obj.geometries[i]] │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - geometry = new OpenLayers.Geometry.Collection(components); │ │ │ │ │ - collection = true; │ │ │ │ │ - } else { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(obj.coordinates))) { │ │ │ │ │ - throw "Geometry must have coordinates array: " + obj; │ │ │ │ │ - } │ │ │ │ │ - if (!this.parseCoords[obj.type.toLowerCase()]) { │ │ │ │ │ - throw "Unsupported geometry type: " + obj.type; │ │ │ │ │ - } │ │ │ │ │ - try { │ │ │ │ │ - geometry = this.parseCoords[obj.type.toLowerCase()].apply( │ │ │ │ │ - this, [obj.coordinates] │ │ │ │ │ - ); │ │ │ │ │ - } catch (err) { │ │ │ │ │ - // deal with bad coordinates │ │ │ │ │ - throw err; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // We don't reproject collections because the children are reprojected │ │ │ │ │ - // for us when they are created. │ │ │ │ │ - if (this.internalProjection && this.externalProjection && !collection) { │ │ │ │ │ - geometry.transform(this.externalProjection, │ │ │ │ │ - this.internalProjection); │ │ │ │ │ - } │ │ │ │ │ - return geometry; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: parseCoords │ │ │ │ │ - * Object with properties corresponding to the GeoJSON geometry types. │ │ │ │ │ - * Property values are functions that do the actual parsing. │ │ │ │ │ - */ │ │ │ │ │ - parseCoords: { │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseCoords.point │ │ │ │ │ - * Convert a coordinate array from GeoJSON into an │ │ │ │ │ - * <OpenLayers.Geometry>. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * array - {Object} The coordinates array from the GeoJSON fragment. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} A geometry. │ │ │ │ │ - */ │ │ │ │ │ - "point": function(array) { │ │ │ │ │ - if (this.ignoreExtraDims == false && │ │ │ │ │ - array.length != 2) { │ │ │ │ │ - throw "Only 2D points are supported: " + array; │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Geometry.Point(array[0], array[1]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseCoords.multipoint │ │ │ │ │ - * Convert a coordinate array from GeoJSON into an │ │ │ │ │ - * <OpenLayers.Geometry>. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * array - {Object} The coordinates array from the GeoJSON fragment. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} A geometry. │ │ │ │ │ - */ │ │ │ │ │ - "multipoint": function(array) { │ │ │ │ │ - var points = []; │ │ │ │ │ - var p = null; │ │ │ │ │ - for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ - try { │ │ │ │ │ - p = this.parseCoords["point"].apply(this, [array[i]]); │ │ │ │ │ - } catch (err) { │ │ │ │ │ - throw err; │ │ │ │ │ - } │ │ │ │ │ - points.push(p); │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Geometry.MultiPoint(points); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseCoords.linestring │ │ │ │ │ - * Convert a coordinate array from GeoJSON into an │ │ │ │ │ - * <OpenLayers.Geometry>. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * array - {Object} The coordinates array from the GeoJSON fragment. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} A geometry. │ │ │ │ │ - */ │ │ │ │ │ - "linestring": function(array) { │ │ │ │ │ - var points = []; │ │ │ │ │ - var p = null; │ │ │ │ │ - for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ - try { │ │ │ │ │ - p = this.parseCoords["point"].apply(this, [array[i]]); │ │ │ │ │ - } catch (err) { │ │ │ │ │ - throw err; │ │ │ │ │ - } │ │ │ │ │ - points.push(p); │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Geometry.LineString(points); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseCoords.multilinestring │ │ │ │ │ - * Convert a coordinate array from GeoJSON into an │ │ │ │ │ - * <OpenLayers.Geometry>. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * array - {Object} The coordinates array from the GeoJSON fragment. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} A geometry. │ │ │ │ │ - */ │ │ │ │ │ - "multilinestring": function(array) { │ │ │ │ │ - var lines = []; │ │ │ │ │ - var l = null; │ │ │ │ │ - for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ - try { │ │ │ │ │ - l = this.parseCoords["linestring"].apply(this, [array[i]]); │ │ │ │ │ - } catch (err) { │ │ │ │ │ - throw err; │ │ │ │ │ - } │ │ │ │ │ - lines.push(l); │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Geometry.MultiLineString(lines); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseCoords.polygon │ │ │ │ │ - * Convert a coordinate array from GeoJSON into an │ │ │ │ │ - * <OpenLayers.Geometry>. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} A geometry. │ │ │ │ │ - */ │ │ │ │ │ - "polygon": function(array) { │ │ │ │ │ - var rings = []; │ │ │ │ │ - var r, l; │ │ │ │ │ - for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ - try { │ │ │ │ │ - l = this.parseCoords["linestring"].apply(this, [array[i]]); │ │ │ │ │ - } catch (err) { │ │ │ │ │ - throw err; │ │ │ │ │ - } │ │ │ │ │ - r = new OpenLayers.Geometry.LinearRing(l.components); │ │ │ │ │ - rings.push(r); │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Geometry.Polygon(rings); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseCoords.multipolygon │ │ │ │ │ - * Convert a coordinate array from GeoJSON into an │ │ │ │ │ - * <OpenLayers.Geometry>. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * array - {Object} The coordinates array from the GeoJSON fragment. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} A geometry. │ │ │ │ │ - */ │ │ │ │ │ - "multipolygon": function(array) { │ │ │ │ │ - var polys = []; │ │ │ │ │ - var p = null; │ │ │ │ │ - for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ - try { │ │ │ │ │ - p = this.parseCoords["polygon"].apply(this, [array[i]]); │ │ │ │ │ - } catch (err) { │ │ │ │ │ - throw err; │ │ │ │ │ - } │ │ │ │ │ - polys.push(p); │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Geometry.MultiPolygon(polys); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseCoords.box │ │ │ │ │ - * Convert a coordinate array from GeoJSON into an │ │ │ │ │ - * <OpenLayers.Geometry>. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * array - {Object} The coordinates array from the GeoJSON fragment. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} A geometry. │ │ │ │ │ - */ │ │ │ │ │ - "box": function(array) { │ │ │ │ │ - if (array.length != 2) { │ │ │ │ │ - throw "GeoJSON box coordinates must have 2 elements"; │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Geometry.Polygon([ │ │ │ │ │ - new OpenLayers.Geometry.LinearRing([ │ │ │ │ │ - new OpenLayers.Geometry.Point(array[0][0], array[0][1]), │ │ │ │ │ - new OpenLayers.Geometry.Point(array[1][0], array[0][1]), │ │ │ │ │ - new OpenLayers.Geometry.Point(array[1][0], array[1][1]), │ │ │ │ │ - new OpenLayers.Geometry.Point(array[0][0], array[1][1]), │ │ │ │ │ - new OpenLayers.Geometry.Point(array[0][0], array[0][1]) │ │ │ │ │ - ]) │ │ │ │ │ - ]); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Serialize a feature, geometry, array of features into a GeoJSON string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} An <OpenLayers.Feature.Vector>, <OpenLayers.Geometry>, │ │ │ │ │ - * or an array of features. │ │ │ │ │ - * pretty - {Boolean} Structure the output with newlines and indentation. │ │ │ │ │ - * Default is false. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The GeoJSON string representation of the input geometry, │ │ │ │ │ - * features, or array of features. │ │ │ │ │ - */ │ │ │ │ │ - write: function(obj, pretty) { │ │ │ │ │ - var geojson = { │ │ │ │ │ - "type": null │ │ │ │ │ - }; │ │ │ │ │ - if (OpenLayers.Util.isArray(obj)) { │ │ │ │ │ - geojson.type = "FeatureCollection"; │ │ │ │ │ - var numFeatures = obj.length; │ │ │ │ │ - geojson.features = new Array(numFeatures); │ │ │ │ │ - for (var i = 0; i < numFeatures; ++i) { │ │ │ │ │ - var element = obj[i]; │ │ │ │ │ - if (!element instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ - var msg = "FeatureCollection only supports collections " + │ │ │ │ │ - "of features: " + element; │ │ │ │ │ - throw msg; │ │ │ │ │ - } │ │ │ │ │ - geojson.features[i] = this.extract.feature.apply( │ │ │ │ │ - this, [element] │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } else if (obj.CLASS_NAME.indexOf("OpenLayers.Geometry") == 0) { │ │ │ │ │ - geojson = this.extract.geometry.apply(this, [obj]); │ │ │ │ │ - } else if (obj instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ - geojson = this.extract.feature.apply(this, [obj]); │ │ │ │ │ - if (obj.layer && obj.layer.projection) { │ │ │ │ │ - geojson.crs = this.createCRSObject(obj); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Format.JSON.prototype.write.apply(this, │ │ │ │ │ - [geojson, pretty]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createCRSObject │ │ │ │ │ - * Create the CRS object for an object. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * object - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object which can be assigned to the crs property │ │ │ │ │ - * of a GeoJSON object. │ │ │ │ │ - */ │ │ │ │ │ - createCRSObject: function(object) { │ │ │ │ │ - var proj = object.layer.projection.toString(); │ │ │ │ │ - var crs = {}; │ │ │ │ │ - if (proj.match(/epsg:/i)) { │ │ │ │ │ - var code = parseInt(proj.substring(proj.indexOf(":") + 1)); │ │ │ │ │ - if (code == 4326) { │ │ │ │ │ - crs = { │ │ │ │ │ - "type": "name", │ │ │ │ │ - "properties": { │ │ │ │ │ - "name": "urn:ogc:def:crs:OGC:1.3:CRS84" │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - } else { │ │ │ │ │ - crs = { │ │ │ │ │ - "type": "name", │ │ │ │ │ - "properties": { │ │ │ │ │ - "name": "EPSG:" + code │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return crs; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: extract │ │ │ │ │ - * Object with properties corresponding to the GeoJSON types. │ │ │ │ │ - * Property values are functions that do the actual value extraction. │ │ │ │ │ - */ │ │ │ │ │ - extract: { │ │ │ │ │ - /** │ │ │ │ │ - * Method: extract.feature │ │ │ │ │ - * Return a partial GeoJSON object representing a single feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object representing the point. │ │ │ │ │ - */ │ │ │ │ │ - 'feature': function(feature) { │ │ │ │ │ - var geom = this.extract.geometry.apply(this, [feature.geometry]); │ │ │ │ │ - var json = { │ │ │ │ │ - "type": "Feature", │ │ │ │ │ - "properties": feature.attributes, │ │ │ │ │ - "geometry": geom │ │ │ │ │ - }; │ │ │ │ │ - if (feature.fid != null) { │ │ │ │ │ - json.id = feature.fid; │ │ │ │ │ - } │ │ │ │ │ - return json; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: extract.geometry │ │ │ │ │ - * Return a GeoJSON object representing a single geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object representing the geometry. │ │ │ │ │ - */ │ │ │ │ │ - 'geometry': function(geometry) { │ │ │ │ │ - if (geometry == null) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry = geometry.clone(); │ │ │ │ │ - geometry.transform(this.internalProjection, │ │ │ │ │ - this.externalProjection); │ │ │ │ │ - } │ │ │ │ │ - var geometryType = geometry.CLASS_NAME.split('.')[2]; │ │ │ │ │ - var data = this.extract[geometryType.toLowerCase()].apply(this, [geometry]); │ │ │ │ │ - var json; │ │ │ │ │ - if (geometryType == "Collection") { │ │ │ │ │ - json = { │ │ │ │ │ - "type": "GeometryCollection", │ │ │ │ │ - "geometries": data │ │ │ │ │ - }; │ │ │ │ │ - } else { │ │ │ │ │ - json = { │ │ │ │ │ - "type": geometryType, │ │ │ │ │ - "coordinates": data │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return json; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: extract.point │ │ │ │ │ - * Return an array of coordinates from a point. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} An array of coordinates representing the point. │ │ │ │ │ - */ │ │ │ │ │ - 'point': function(point) { │ │ │ │ │ - return [point.x, point.y]; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: extract.multipoint │ │ │ │ │ - * Return an array of point coordinates from a multipoint. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * multipoint - {<OpenLayers.Geometry.MultiPoint>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} An array of point coordinate arrays representing │ │ │ │ │ - * the multipoint. │ │ │ │ │ - */ │ │ │ │ │ - 'multipoint': function(multipoint) { │ │ │ │ │ - var array = []; │ │ │ │ │ - for (var i = 0, len = multipoint.components.length; i < len; ++i) { │ │ │ │ │ - array.push(this.extract.point.apply(this, [multipoint.components[i]])); │ │ │ │ │ - } │ │ │ │ │ - return array; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: extract.linestring │ │ │ │ │ - * Return an array of coordinate arrays from a linestring. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * linestring - {<OpenLayers.Geometry.LineString>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} An array of coordinate arrays representing │ │ │ │ │ - * the linestring. │ │ │ │ │ - */ │ │ │ │ │ - 'linestring': function(linestring) { │ │ │ │ │ - var array = []; │ │ │ │ │ - for (var i = 0, len = linestring.components.length; i < len; ++i) { │ │ │ │ │ - array.push(this.extract.point.apply(this, [linestring.components[i]])); │ │ │ │ │ - } │ │ │ │ │ - return array; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: extract.multilinestring │ │ │ │ │ - * Return an array of linestring arrays from a linestring. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * multilinestring - {<OpenLayers.Geometry.MultiLineString>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} An array of linestring arrays representing │ │ │ │ │ - * the multilinestring. │ │ │ │ │ - */ │ │ │ │ │ - 'multilinestring': function(multilinestring) { │ │ │ │ │ - var array = []; │ │ │ │ │ - for (var i = 0, len = multilinestring.components.length; i < len; ++i) { │ │ │ │ │ - array.push(this.extract.linestring.apply(this, [multilinestring.components[i]])); │ │ │ │ │ - } │ │ │ │ │ - return array; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: extract.polygon │ │ │ │ │ - * Return an array of linear ring arrays from a polygon. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * polygon - {<OpenLayers.Geometry.Polygon>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} An array of linear ring arrays representing the polygon. │ │ │ │ │ - */ │ │ │ │ │ - 'polygon': function(polygon) { │ │ │ │ │ - var array = []; │ │ │ │ │ - for (var i = 0, len = polygon.components.length; i < len; ++i) { │ │ │ │ │ - array.push(this.extract.linestring.apply(this, [polygon.components[i]])); │ │ │ │ │ - } │ │ │ │ │ - return array; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: extract.multipolygon │ │ │ │ │ - * Return an array of polygon arrays from a multipolygon. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * multipolygon - {<OpenLayers.Geometry.MultiPolygon>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} An array of polygon arrays representing │ │ │ │ │ - * the multipolygon │ │ │ │ │ - */ │ │ │ │ │ - 'multipolygon': function(multipolygon) { │ │ │ │ │ - var array = []; │ │ │ │ │ - for (var i = 0, len = multipolygon.components.length; i < len; ++i) { │ │ │ │ │ - array.push(this.extract.polygon.apply(this, [multipolygon.components[i]])); │ │ │ │ │ - } │ │ │ │ │ - return array; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: extract.collection │ │ │ │ │ - * Return an array of geometries from a geometry collection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * collection - {<OpenLayers.Geometry.Collection>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} An array of geometry objects representing the geometry │ │ │ │ │ - * collection. │ │ │ │ │ - */ │ │ │ │ │ - 'collection': function(collection) { │ │ │ │ │ - var len = collection.components.length; │ │ │ │ │ - var array = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - array[i] = this.extract.geometry.apply( │ │ │ │ │ - this, [collection.components[i]] │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return array; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.GeoJSON" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Strategy │ │ │ │ │ - * Abstract vector layer strategy class. Not to be instantiated directly. Use │ │ │ │ │ - * one of the strategy subclasses instead. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Strategy = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} The layer this strategy belongs to. │ │ │ │ │ - */ │ │ │ │ │ - layer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: options │ │ │ │ │ - * {Object} Any options sent to the constructor. │ │ │ │ │ - */ │ │ │ │ │ - options: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: active │ │ │ │ │ - * {Boolean} The control is active. │ │ │ │ │ - */ │ │ │ │ │ - active: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: autoActivate │ │ │ │ │ - * {Boolean} The creator of the strategy can set autoActivate to false │ │ │ │ │ - * to fully control when the protocol is activated and deactivated. │ │ │ │ │ - * Defaults to true. │ │ │ │ │ - */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: autoDestroy │ │ │ │ │ - * {Boolean} The creator of the strategy can set autoDestroy to false │ │ │ │ │ - * to fully control when the strategy is destroyed. Defaults to │ │ │ │ │ - * true. │ │ │ │ │ - */ │ │ │ │ │ - autoDestroy: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy │ │ │ │ │ - * Abstract class for vector strategies. Create instances of a subclass. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.options = options; │ │ │ │ │ - // set the active property here, so that user cannot override it │ │ │ │ │ - this.active = false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up the strategy. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.layer = null; │ │ │ │ │ - this.options = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setLayer │ │ │ │ │ - * Called to set the <layer> property. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - setLayer: function(layer) { │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} True if the strategy was successfully activated or false if │ │ │ │ │ - * the strategy was already active. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - this.active = true; │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ - * tear-down. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} True if the strategy was successfully deactivated or false if │ │ │ │ │ - * the strategy was already inactive. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.active = false; │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy/Fixed.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Strategy.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Strategy.Fixed │ │ │ │ │ - * A simple strategy that requests features once and never requests new data. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Strategy> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Strategy.Fixed = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: preload │ │ │ │ │ - * {Boolean} Load data before layer made visible. Enabling this may result │ │ │ │ │ - * in considerable overhead if your application loads many data layers │ │ │ │ │ - * that are not visible by default. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - preload: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy.Fixed │ │ │ │ │ - * Create a new Fixed strategy. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Activate the strategy: load data or add listener to load when visible │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} True if the strategy was successfully activated or false if │ │ │ │ │ - * the strategy was already active. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "refresh": this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - if (this.layer.visibility == true || this.preload) { │ │ │ │ │ - this.load(); │ │ │ │ │ - } else { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "visibilitychanged": this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Deactivate the strategy. Undo what is done in <activate>. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - "refresh": this.load, │ │ │ │ │ - "visibilitychanged": this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: load │ │ │ │ │ - * Tells protocol to load data and unhooks the visibilitychanged event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} options to pass to protocol read. │ │ │ │ │ - */ │ │ │ │ │ - load: function(options) { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - layer.events.triggerEvent("loadstart", { │ │ │ │ │ - filter: layer.filter │ │ │ │ │ - }); │ │ │ │ │ - layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: this.merge, │ │ │ │ │ - filter: layer.filter, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options)); │ │ │ │ │ - layer.events.un({ │ │ │ │ │ - "visibilitychanged": this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: merge │ │ │ │ │ - * Add all features to the layer. │ │ │ │ │ - * If the layer projection differs from the map projection, features │ │ │ │ │ - * will be transformed from the layer projection to the map projection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object passed │ │ │ │ │ - * by the protocol. │ │ │ │ │ - */ │ │ │ │ │ - merge: function(resp) { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - layer.destroyFeatures(); │ │ │ │ │ - var features = resp.features; │ │ │ │ │ - if (features && features.length > 0) { │ │ │ │ │ - var remote = layer.projection; │ │ │ │ │ - var local = layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var geom; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - geom = features[i].geometry; │ │ │ │ │ - if (geom) { │ │ │ │ │ - geom.transform(remote, local); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - layer.addFeatures(features); │ │ │ │ │ - } │ │ │ │ │ - layer.events.triggerEvent("loadend", { │ │ │ │ │ - response: resp │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Fixed" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Filter/Spatial.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Filter.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Filter.Spatial │ │ │ │ │ - * This class represents a spatial filter. │ │ │ │ │ - * Currently implemented: BBOX, DWithin and Intersects │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Filter> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Filter.Spatial = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: type │ │ │ │ │ - * {String} Type of spatial filter. │ │ │ │ │ - * │ │ │ │ │ - * The type should be one of: │ │ │ │ │ - * - OpenLayers.Filter.Spatial.BBOX │ │ │ │ │ - * - OpenLayers.Filter.Spatial.INTERSECTS │ │ │ │ │ - * - OpenLayers.Filter.Spatial.DWITHIN │ │ │ │ │ - * - OpenLayers.Filter.Spatial.WITHIN │ │ │ │ │ - * - OpenLayers.Filter.Spatial.CONTAINS │ │ │ │ │ - */ │ │ │ │ │ - type: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: property │ │ │ │ │ - * {String} Name of the context property to compare. │ │ │ │ │ - */ │ │ │ │ │ - property: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: value │ │ │ │ │ - * {<OpenLayers.Bounds> || <OpenLayers.Geometry>} The bounds or geometry │ │ │ │ │ - * to be used by the filter. Use bounds for BBOX filters and geometry │ │ │ │ │ - * for INTERSECTS or DWITHIN filters. │ │ │ │ │ - */ │ │ │ │ │ - value: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: distance │ │ │ │ │ - * {Number} The distance to use in a DWithin spatial filter. │ │ │ │ │ - */ │ │ │ │ │ - distance: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: distanceUnits │ │ │ │ │ - * {String} The units to use for the distance, e.g. 'm'. │ │ │ │ │ - */ │ │ │ │ │ - distanceUnits: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Filter.Spatial │ │ │ │ │ - * Creates a spatial filter. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object with properties to set on the │ │ │ │ │ - * filter. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Filter.Spatial>} │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: evaluate │ │ │ │ │ - * Evaluates this filter for a specific feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} feature to apply the filter to. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The feature meets filter criteria. │ │ │ │ │ - */ │ │ │ │ │ - evaluate: function(feature) { │ │ │ │ │ - var intersect = false; │ │ │ │ │ - switch (this.type) { │ │ │ │ │ - case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ - case OpenLayers.Filter.Spatial.INTERSECTS: │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - var geom = this.value; │ │ │ │ │ - if (this.value.CLASS_NAME == "OpenLayers.Bounds") { │ │ │ │ │ - geom = this.value.toGeometry(); │ │ │ │ │ - } │ │ │ │ │ - if (feature.geometry.intersects(geom)) { │ │ │ │ │ - intersect = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - throw new Error('evaluate is not implemented for this filter type.'); │ │ │ │ │ - } │ │ │ │ │ - return intersect; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Clones this filter. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Filter.Spatial>} Clone of this filter. │ │ │ │ │ - */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - var options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - value: this.value && this.value.clone && this.value.clone() │ │ │ │ │ - }, this); │ │ │ │ │ - return new OpenLayers.Filter.Spatial(options); │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Filter.Spatial" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Filter.Spatial.BBOX = "BBOX"; │ │ │ │ │ -OpenLayers.Filter.Spatial.INTERSECTS = "INTERSECTS"; │ │ │ │ │ -OpenLayers.Filter.Spatial.DWITHIN = "DWITHIN"; │ │ │ │ │ -OpenLayers.Filter.Spatial.WITHIN = "WITHIN"; │ │ │ │ │ -OpenLayers.Filter.Spatial.CONTAINS = "CONTAINS"; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy/BBOX.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Strategy.js │ │ │ │ │ - * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Strategy.BBOX │ │ │ │ │ - * A simple strategy that reads new features when the viewport invalidates │ │ │ │ │ - * some bounds. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Strategy> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Strategy.BBOX = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: bounds │ │ │ │ │ - * {<OpenLayers.Bounds>} The current data bounds (in the same projection │ │ │ │ │ - * as the layer - not always the same projection as the map). │ │ │ │ │ - */ │ │ │ │ │ - bounds: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: resolution │ │ │ │ │ - * {Float} The current data resolution. │ │ │ │ │ - */ │ │ │ │ │ - resolution: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: ratio │ │ │ │ │ - * {Float} The ratio of the data bounds to the viewport bounds (in each │ │ │ │ │ - * dimension). Default is 2. │ │ │ │ │ - */ │ │ │ │ │ - ratio: 2, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: resFactor │ │ │ │ │ - * {Float} Optional factor used to determine when previously requested │ │ │ │ │ - * features are invalid. If set, the resFactor will be compared to the │ │ │ │ │ - * resolution of the previous request to the current map resolution. │ │ │ │ │ - * If resFactor > (old / new) and 1/resFactor < (old / new). If you │ │ │ │ │ - * set a resFactor of 1, data will be requested every time the │ │ │ │ │ - * resolution changes. If you set a resFactor of 3, data will be │ │ │ │ │ - * requested if the old resolution is 3 times the new, or if the new is │ │ │ │ │ - * 3 times the old. If the old bounds do not contain the new bounds │ │ │ │ │ - * new data will always be requested (with or without considering │ │ │ │ │ - * resFactor). │ │ │ │ │ - */ │ │ │ │ │ - resFactor: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: response │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} The protocol response object returned │ │ │ │ │ - * by the layer protocol. │ │ │ │ │ - */ │ │ │ │ │ - response: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy.BBOX │ │ │ │ │ - * Create a new BBOX strategy. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Set up strategy with regard to reading new batches of remote data. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "moveend": this.update, │ │ │ │ │ - "refresh": this.update, │ │ │ │ │ - "visibilitychanged": this.update, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.update(); │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Tear down strategy with regard to reading new batches of remote data. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - "moveend": this.update, │ │ │ │ │ - "refresh": this.update, │ │ │ │ │ - "visibilitychanged": this.update, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: update │ │ │ │ │ - * Callback function called on "moveend" or "refresh" layer events. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will determine │ │ │ │ │ - * the behaviour of this Strategy │ │ │ │ │ - * │ │ │ │ │ - * Valid options include: │ │ │ │ │ - * force - {Boolean} if true, new data must be unconditionally read. │ │ │ │ │ - * noAbort - {Boolean} if true, do not abort previous requests. │ │ │ │ │ - */ │ │ │ │ │ - update: function(options) { │ │ │ │ │ - var mapBounds = this.getMapBounds(); │ │ │ │ │ - if (mapBounds !== null && ((options && options.force) || │ │ │ │ │ - (this.layer.visibility && this.layer.calculateInRange() && this.invalidBounds(mapBounds)))) { │ │ │ │ │ - this.calculateBounds(mapBounds); │ │ │ │ │ - this.resolution = this.layer.map.getResolution(); │ │ │ │ │ - this.triggerRead(options); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getMapBounds │ │ │ │ │ - * Get the map bounds expressed in the same projection as this layer. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} Map bounds in the projection of the layer. │ │ │ │ │ - */ │ │ │ │ │ - getMapBounds: function() { │ │ │ │ │ - if (this.layer.map === null) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - var bounds = this.layer.map.getExtent(); │ │ │ │ │ - if (bounds && !this.layer.projection.equals( │ │ │ │ │ - this.layer.map.getProjectionObject())) { │ │ │ │ │ - bounds = bounds.clone().transform( │ │ │ │ │ - this.layer.map.getProjectionObject(), this.layer.projection │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return bounds; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: invalidBounds │ │ │ │ │ - * Determine whether the previously requested set of features is invalid. │ │ │ │ │ - * This occurs when the new map bounds do not contain the previously │ │ │ │ │ - * requested bounds. In addition, if <resFactor> is set, it will be │ │ │ │ │ - * considered. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * mapBounds - {<OpenLayers.Bounds>} the current map extent, will be │ │ │ │ │ - * retrieved from the map object if not provided │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - invalidBounds: function(mapBounds) { │ │ │ │ │ - if (!mapBounds) { │ │ │ │ │ - mapBounds = this.getMapBounds(); │ │ │ │ │ - } │ │ │ │ │ - var invalid = !this.bounds || !this.bounds.containsBounds(mapBounds); │ │ │ │ │ - if (!invalid && this.resFactor) { │ │ │ │ │ - var ratio = this.resolution / this.layer.map.getResolution(); │ │ │ │ │ - invalid = (ratio >= this.resFactor || ratio <= (1 / this.resFactor)); │ │ │ │ │ - } │ │ │ │ │ - return invalid; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateBounds │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * mapBounds - {<OpenLayers.Bounds>} the current map extent, will be │ │ │ │ │ - * retrieved from the map object if not provided │ │ │ │ │ - */ │ │ │ │ │ - calculateBounds: function(mapBounds) { │ │ │ │ │ - if (!mapBounds) { │ │ │ │ │ - mapBounds = this.getMapBounds(); │ │ │ │ │ - } │ │ │ │ │ - var center = mapBounds.getCenterLonLat(); │ │ │ │ │ - var dataWidth = mapBounds.getWidth() * this.ratio; │ │ │ │ │ - var dataHeight = mapBounds.getHeight() * this.ratio; │ │ │ │ │ - this.bounds = new OpenLayers.Bounds( │ │ │ │ │ - center.lon - (dataWidth / 2), │ │ │ │ │ - center.lat - (dataHeight / 2), │ │ │ │ │ - center.lon + (dataWidth / 2), │ │ │ │ │ - center.lat + (dataHeight / 2) │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: triggerRead │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Additional options for the protocol's read method │ │ │ │ │ - * (optional) │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} The protocol response object │ │ │ │ │ - * returned by the layer protocol. │ │ │ │ │ - */ │ │ │ │ │ - triggerRead: function(options) { │ │ │ │ │ - if (this.response && !(options && options.noAbort === true)) { │ │ │ │ │ - this.layer.protocol.abort(this.response); │ │ │ │ │ - this.layer.events.triggerEvent("loadend"); │ │ │ │ │ - } │ │ │ │ │ - var evt = { │ │ │ │ │ - filter: this.createFilter() │ │ │ │ │ - }; │ │ │ │ │ - this.layer.events.triggerEvent("loadstart", evt); │ │ │ │ │ - this.response = this.layer.protocol.read( │ │ │ │ │ - OpenLayers.Util.applyDefaults({ │ │ │ │ │ - filter: evt.filter, │ │ │ │ │ - callback: this.merge, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options)); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createFilter │ │ │ │ │ - * Creates a spatial BBOX filter. If the layer that this strategy belongs │ │ │ │ │ - * to has a filter property, this filter will be combined with the BBOX │ │ │ │ │ - * filter. │ │ │ │ │ - * │ │ │ │ │ - * Returns │ │ │ │ │ - * {<OpenLayers.Filter>} The filter object. │ │ │ │ │ - */ │ │ │ │ │ - createFilter: function() { │ │ │ │ │ - var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ - value: this.bounds, │ │ │ │ │ - projection: this.layer.projection │ │ │ │ │ - }); │ │ │ │ │ - if (this.layer.filter) { │ │ │ │ │ - filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ - type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ - filters: [this.layer.filter, filter] │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return filter; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: merge │ │ │ │ │ - * Given a list of features, determine which ones to add to the layer. │ │ │ │ │ - * If the layer projection differs from the map projection, features │ │ │ │ │ - * will be transformed from the layer projection to the map projection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object passed │ │ │ │ │ - * by the protocol. │ │ │ │ │ - */ │ │ │ │ │ - merge: function(resp) { │ │ │ │ │ - this.layer.destroyFeatures(); │ │ │ │ │ - if (resp.success()) { │ │ │ │ │ - var features = resp.features; │ │ │ │ │ - if (features && features.length > 0) { │ │ │ │ │ - var remote = this.layer.projection; │ │ │ │ │ - var local = this.layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var geom; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - geom = features[i].geometry; │ │ │ │ │ - if (geom) { │ │ │ │ │ - geom.transform(remote, local); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.layer.addFeatures(features); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.bounds = null; │ │ │ │ │ - } │ │ │ │ │ - this.response = null; │ │ │ │ │ - this.layer.events.triggerEvent("loadend", { │ │ │ │ │ - response: resp │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.BBOX" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control │ │ │ │ │ - * Controls affect the display or behavior of the map. They allow everything │ │ │ │ │ - * from panning and zooming to displaying a scale indicator. Controls by │ │ │ │ │ - * default are added to the map they are contained within however it is │ │ │ │ │ - * possible to add a control to an external div by passing the div in the │ │ │ │ │ - * options parameter. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * The following example shows how to add many of the common controls │ │ │ │ │ - * to a map. │ │ │ │ │ - * │ │ │ │ │ - * > var map = new OpenLayers.Map('map', { controls: [] }); │ │ │ │ │ - * > │ │ │ │ │ - * > map.addControl(new OpenLayers.Control.PanZoomBar()); │ │ │ │ │ - * > map.addControl(new OpenLayers.Control.LayerSwitcher({'ascending':false})); │ │ │ │ │ - * > map.addControl(new OpenLayers.Control.Permalink()); │ │ │ │ │ - * > map.addControl(new OpenLayers.Control.Permalink('permalink')); │ │ │ │ │ - * > map.addControl(new OpenLayers.Control.MousePosition()); │ │ │ │ │ - * > map.addControl(new OpenLayers.Control.OverviewMap()); │ │ │ │ │ - * > map.addControl(new OpenLayers.Control.KeyboardDefaults()); │ │ │ │ │ - * │ │ │ │ │ - * The next code fragment is a quick example of how to intercept │ │ │ │ │ - * shift-mouse click to display the extent of the bounding box │ │ │ │ │ - * dragged out by the user. Usually controls are not created │ │ │ │ │ - * in exactly this manner. See the source for a more complete │ │ │ │ │ - * example: │ │ │ │ │ - * │ │ │ │ │ - * > var control = new OpenLayers.Control(); │ │ │ │ │ - * > OpenLayers.Util.extend(control, { │ │ │ │ │ - * > draw: function () { │ │ │ │ │ - * > // this Handler.Box will intercept the shift-mousedown │ │ │ │ │ - * > // before Control.MouseDefault gets to see it │ │ │ │ │ - * > this.box = new OpenLayers.Handler.Box( control, │ │ │ │ │ - * > {"done": this.notice}, │ │ │ │ │ - * > {keyMask: OpenLayers.Handler.MOD_SHIFT}); │ │ │ │ │ - * > this.box.activate(); │ │ │ │ │ - * > }, │ │ │ │ │ - * > │ │ │ │ │ - * > notice: function (bounds) { │ │ │ │ │ - * > OpenLayers.Console.userError(bounds); │ │ │ │ │ - * > } │ │ │ │ │ - * > }); │ │ │ │ │ - * > map.addControl(control); │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: id │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - id: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: map │ │ │ │ │ - * {<OpenLayers.Map>} this gets set in the addControl() function in │ │ │ │ │ - * OpenLayers.Map │ │ │ │ │ - */ │ │ │ │ │ - map: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: div │ │ │ │ │ - * {DOMElement} The element that contains the control, if not present the │ │ │ │ │ - * control is placed inside the map. │ │ │ │ │ - */ │ │ │ │ │ - div: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: type │ │ │ │ │ - * {Number} Controls can have a 'type'. The type determines the type of │ │ │ │ │ - * interactions which are possible with them when they are placed in an │ │ │ │ │ - * <OpenLayers.Control.Panel>. │ │ │ │ │ - */ │ │ │ │ │ - type: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: allowSelection │ │ │ │ │ - * {Boolean} By default, controls do not allow selection, because │ │ │ │ │ - * it may interfere with map dragging. If this is true, OpenLayers │ │ │ │ │ - * will not prevent selection of the control. │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - allowSelection: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: displayClass │ │ │ │ │ - * {string} This property is used for CSS related to the drawing of the │ │ │ │ │ - * Control. │ │ │ │ │ - */ │ │ │ │ │ - displayClass: "", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: title │ │ │ │ │ - * {string} This property is used for showing a tooltip over the │ │ │ │ │ - * Control. │ │ │ │ │ - */ │ │ │ │ │ - title: "", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * false. │ │ │ │ │ - */ │ │ │ │ │ - autoActivate: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: active │ │ │ │ │ - * {Boolean} The control is active (read-only). Use <activate> and │ │ │ │ │ - * <deactivate> to change control state. │ │ │ │ │ - */ │ │ │ │ │ - active: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: handlerOptions │ │ │ │ │ - * {Object} Used to set non-default properties on the control's handler │ │ │ │ │ - */ │ │ │ │ │ - handlerOptions: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: handler │ │ │ │ │ - * {<OpenLayers.Handler>} null │ │ │ │ │ - */ │ │ │ │ │ - handler: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: eventListeners │ │ │ │ │ - * {Object} If set as an option at construction, the eventListeners │ │ │ │ │ - * object will be registered with <OpenLayers.Events.on>. Object │ │ │ │ │ - * structure must be a listeners object as shown in the example for │ │ │ │ │ - * the events.on method. │ │ │ │ │ - */ │ │ │ │ │ - eventListeners: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Listeners will be called with a reference to an event object. The │ │ │ │ │ - * properties of this event depends on exactly what happened. │ │ │ │ │ - * │ │ │ │ │ - * All event objects have at least the following properties: │ │ │ │ │ - * object - {Object} A reference to control.events.object (a reference │ │ │ │ │ - * to the control). │ │ │ │ │ - * element - {DOMElement} A reference to control.events.element (which │ │ │ │ │ - * will be null unless documented otherwise). │ │ │ │ │ - * │ │ │ │ │ - * Supported map event types: │ │ │ │ │ - * activate - Triggered when activated. │ │ │ │ │ - * deactivate - Triggered when deactivated. │ │ │ │ │ - */ │ │ │ │ │ - events: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control │ │ │ │ │ - * Create an OpenLayers Control. The options passed as a parameter │ │ │ │ │ - * directly extend the control. For example passing the following: │ │ │ │ │ - * │ │ │ │ │ - * > var control = new OpenLayers.Control({div: myDiv}); │ │ │ │ │ - * │ │ │ │ │ - * Overrides the default div attribute value of null. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - // We do this before the extend so that instances can override │ │ │ │ │ - // className in options. │ │ │ │ │ - this.displayClass = │ │ │ │ │ - this.CLASS_NAME.replace("OpenLayers.", "ol").replace(/\./g, ""); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - │ │ │ │ │ - this.events = new OpenLayers.Events(this); │ │ │ │ │ - if (this.eventListeners instanceof Object) { │ │ │ │ │ - this.events.on(this.eventListeners); │ │ │ │ │ - } │ │ │ │ │ - if (this.id == null) { │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * The destroy method is used to perform any clean up before the control │ │ │ │ │ - * is dereferenced. Typically this is where event listeners are removed │ │ │ │ │ - * to prevent memory leaks. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.events) { │ │ │ │ │ - if (this.eventListeners) { │ │ │ │ │ - this.events.un(this.eventListeners); │ │ │ │ │ - } │ │ │ │ │ - this.events.destroy(); │ │ │ │ │ - this.events = null; │ │ │ │ │ - } │ │ │ │ │ - this.eventListeners = null; │ │ │ │ │ - │ │ │ │ │ - // eliminate circular references │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.destroy(); │ │ │ │ │ - this.handler = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.handlers) { │ │ │ │ │ - for (var key in this.handlers) { │ │ │ │ │ - if (this.handlers.hasOwnProperty(key) && │ │ │ │ │ - typeof this.handlers[key].destroy == "function") { │ │ │ │ │ - this.handlers[key].destroy(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.handlers = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.removeControl(this); │ │ │ │ │ - this.map = null; │ │ │ │ │ - } │ │ │ │ │ - this.div = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the control. This is done through an accessor │ │ │ │ │ - * so that subclasses can override this and take special action once │ │ │ │ │ - * they have their map variable set. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.map = map; │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.setMap(map); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * The draw method is called when the control is ready to be displayed │ │ │ │ │ - * on the page. If a div has not been created one is created. Controls │ │ │ │ │ - * with a visual component will almost always want to override this method │ │ │ │ │ - * to customize the look of control. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} The top-left pixel position of the control │ │ │ │ │ - * or null. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ - */ │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - if (this.div == null) { │ │ │ │ │ - this.div = OpenLayers.Util.createDiv(this.id); │ │ │ │ │ - this.div.className = this.displayClass; │ │ │ │ │ - if (!this.allowSelection) { │ │ │ │ │ - this.div.className += " olControlNoSelect"; │ │ │ │ │ - this.div.setAttribute("unselectable", "on", 0); │ │ │ │ │ - this.div.onselectstart = OpenLayers.Function.False; │ │ │ │ │ - } │ │ │ │ │ - if (this.title != "") { │ │ │ │ │ - this.div.title = this.title; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (px != null) { │ │ │ │ │ - this.position = px.clone(); │ │ │ │ │ - } │ │ │ │ │ - this.moveTo(this.position); │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * Sets the left and top style attributes to the passed in pixel │ │ │ │ │ - * coordinates. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - */ │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - if ((px != null) && (this.div != null)) { │ │ │ │ │ - this.div.style.left = px.x + "px"; │ │ │ │ │ - this.div.style.top = px.y + "px"; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Explicitly activates a control and it's associated │ │ │ │ │ - * handler if one has been set. Controls can be │ │ │ │ │ - * deactivated by calling the deactivate() method. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} True if the control was successfully activated or │ │ │ │ │ - * false if the control was already active. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.activate(); │ │ │ │ │ - } │ │ │ │ │ - this.active = true; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - OpenLayers.Element.addClass( │ │ │ │ │ - this.map.viewPortDiv, │ │ │ │ │ - this.displayClass.replace(/ /g, "") + "Active" │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("activate"); │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivates a control and it's associated handler if any. The exact │ │ │ │ │ - * effect of this depends on the control itself. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} True if the control was effectively deactivated or false │ │ │ │ │ - * if the control was already inactive. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.deactivate(); │ │ │ │ │ - } │ │ │ │ │ - this.active = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - OpenLayers.Element.removeClass( │ │ │ │ │ - this.map.viewPortDiv, │ │ │ │ │ - this.displayClass.replace(/ /g, "") + "Active" │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("deactivate"); │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Control.TYPE_BUTTON │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.TYPE_BUTTON = 1; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Control.TYPE_TOGGLE │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.TYPE_TOGGLE = 2; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Control.TYPE_TOOL │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.TYPE_TOOL = 3; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Handler │ │ │ │ │ - * Base class to construct a higher-level handler for event sequences. All │ │ │ │ │ - * handlers have activate and deactivate methods. In addition, they have │ │ │ │ │ - * methods named like browser events. When a handler is activated, any │ │ │ │ │ - * additional methods named like a browser event is registered as a │ │ │ │ │ - * listener for the corresponding event. When a handler is deactivated, │ │ │ │ │ - * those same methods are unregistered as event listeners. │ │ │ │ │ - * │ │ │ │ │ - * Handlers also typically have a callbacks object with keys named like │ │ │ │ │ - * the abstracted events or event sequences that they are in charge of │ │ │ │ │ - * handling. The controls that wrap handlers define the methods that │ │ │ │ │ - * correspond to these abstract events - so instead of listening for │ │ │ │ │ - * individual browser events, they only listen for the abstract events │ │ │ │ │ - * defined by the handler. │ │ │ │ │ - * │ │ │ │ │ - * Handlers are created by controls, which ultimately have the responsibility │ │ │ │ │ - * of making changes to the the state of the application. Handlers │ │ │ │ │ - * themselves may make temporary changes, but in general are expected to │ │ │ │ │ - * return the application in the same state that they found it. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: id │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - id: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: control │ │ │ │ │ - * {<OpenLayers.Control>}. The control that initialized this handler. The │ │ │ │ │ - * control is assumed to have a valid map property - that map is used │ │ │ │ │ - * in the handler's own setMap method. │ │ │ │ │ - */ │ │ │ │ │ - control: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: map │ │ │ │ │ - * {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - map: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: keyMask │ │ │ │ │ - * {Integer} Use bitwise operators and one or more of the OpenLayers.Handler │ │ │ │ │ - * constants to construct a keyMask. The keyMask is used by │ │ │ │ │ - * <checkModifiers>. If the keyMask matches the combination of keys │ │ │ │ │ - * down on an event, checkModifiers returns true. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * // handler only responds if the Shift key is down │ │ │ │ │ - * handler.keyMask = OpenLayers.Handler.MOD_SHIFT; │ │ │ │ │ - * │ │ │ │ │ - * // handler only responds if Ctrl-Shift is down │ │ │ │ │ - * handler.keyMask = OpenLayers.Handler.MOD_SHIFT | │ │ │ │ │ - * OpenLayers.Handler.MOD_CTRL; │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ - keyMask: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: active │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - active: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: evt │ │ │ │ │ - * {Event} This property references the last event handled by the handler. │ │ │ │ │ - * Note that this property is not part of the stable API. Use of the │ │ │ │ │ - * evt property should be restricted to controls in the library │ │ │ │ │ - * or other applications that are willing to update with changes to │ │ │ │ │ - * the OpenLayers code. │ │ │ │ │ - */ │ │ │ │ │ - evt: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: touch │ │ │ │ │ - * {Boolean} Indicates the support of touch events. When touch events are │ │ │ │ │ - * started touch will be true and all mouse related listeners will do │ │ │ │ │ - * nothing. │ │ │ │ │ - */ │ │ │ │ │ - touch: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Handler │ │ │ │ │ - * Construct a handler. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that initialized this │ │ │ │ │ - * handler. The control is assumed to have a valid map property; that │ │ │ │ │ - * map is used in the handler's own setMap method. If a map property │ │ │ │ │ - * is present in the options argument it will be used instead. │ │ │ │ │ - * callbacks - {Object} An object whose properties correspond to abstracted │ │ │ │ │ - * events or sequences of browser events. The values for these │ │ │ │ │ - * properties are functions defined by the control that get called by │ │ │ │ │ - * the handler. │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * the handler. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.control = control; │ │ │ │ │ - this.callbacks = callbacks; │ │ │ │ │ - │ │ │ │ │ - var map = this.map || control.map; │ │ │ │ │ - if (map) { │ │ │ │ │ - this.setMap(map); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.map = map; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: checkModifiers │ │ │ │ │ - * Check the keyMask on the handler. If no <keyMask> is set, this always │ │ │ │ │ - * returns true. If a <keyMask> is set and it matches the combination │ │ │ │ │ - * of keys down on an event, this returns true. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The keyMask matches the keys down on an event. │ │ │ │ │ - */ │ │ │ │ │ - checkModifiers: function(evt) { │ │ │ │ │ - if (this.keyMask == null) { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - /* calculate the keyboard modifier mask for this event */ │ │ │ │ │ - var keyModifiers = │ │ │ │ │ - (evt.shiftKey ? OpenLayers.Handler.MOD_SHIFT : 0) | │ │ │ │ │ - (evt.ctrlKey ? OpenLayers.Handler.MOD_CTRL : 0) | │ │ │ │ │ - (evt.altKey ? OpenLayers.Handler.MOD_ALT : 0) | │ │ │ │ │ - (evt.metaKey ? OpenLayers.Handler.MOD_META : 0); │ │ │ │ │ - │ │ │ │ │ - /* if it differs from the handler object's key mask, │ │ │ │ │ - bail out of the event handler */ │ │ │ │ │ - return (keyModifiers == this.keyMask); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Turn on the handler. Returns false if the handler was already active. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - // register for event handlers defined on this class. │ │ │ │ │ - var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.register(events[i], this[events[i]]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.active = true; │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Turn off the handler. Returns false if the handler was already inactive. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - // unregister event handlers defined on this class. │ │ │ │ │ - var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.unregister(events[i], this[events[i]]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.touch = false; │ │ │ │ │ - this.active = false; │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: startTouch │ │ │ │ │ - * Start touch events, this method must be called by subclasses in │ │ │ │ │ - * "touchstart" method. When touch events are started <touch> will be │ │ │ │ │ - * true and all mouse related listeners will do nothing. │ │ │ │ │ - */ │ │ │ │ │ - startTouch: function() { │ │ │ │ │ - if (!this.touch) { │ │ │ │ │ - this.touch = true; │ │ │ │ │ - var events = [ │ │ │ │ │ - "mousedown", "mouseup", "mousemove", "click", "dblclick", │ │ │ │ │ - "mouseout" │ │ │ │ │ - ]; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.unregister(events[i], this[events[i]]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: callback │ │ │ │ │ - * Trigger the control's named callback with the given arguments │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} The key for the callback that is one of the properties │ │ │ │ │ - * of the handler's callbacks object. │ │ │ │ │ - * args - {Array(*)} An array of arguments (any type) with which to call │ │ │ │ │ - * the callback (defined by the control). │ │ │ │ │ - */ │ │ │ │ │ - callback: function(name, args) { │ │ │ │ │ - if (name && this.callbacks[name]) { │ │ │ │ │ - this.callbacks[name].apply(this.control, args); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: register │ │ │ │ │ - * register an event on the map │ │ │ │ │ - */ │ │ │ │ │ - register: function(name, method) { │ │ │ │ │ - // TODO: deal with registerPriority in 3.0 │ │ │ │ │ - this.map.events.registerPriority(name, this, method); │ │ │ │ │ - this.map.events.registerPriority(name, this, this.setEvent); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: unregister │ │ │ │ │ - * unregister an event from the map │ │ │ │ │ - */ │ │ │ │ │ - unregister: function(name, method) { │ │ │ │ │ - this.map.events.unregister(name, this, method); │ │ │ │ │ - this.map.events.unregister(name, this, this.setEvent); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setEvent │ │ │ │ │ - * With each registered browser event, the handler sets its own evt │ │ │ │ │ - * property. This property can be accessed by controls if needed │ │ │ │ │ - * to get more information about the event that the handler is │ │ │ │ │ - * processing. │ │ │ │ │ - * │ │ │ │ │ - * This allows modifier keys on the event to be checked (alt, shift, ctrl, │ │ │ │ │ - * and meta cannot be checked with the keyboard handler). For a │ │ │ │ │ - * control to determine which modifier keys are associated with the │ │ │ │ │ - * event that a handler is currently processing, it should access │ │ │ │ │ - * (code)handler.evt.altKey || handler.evt.shiftKey || │ │ │ │ │ - * handler.evt.ctrlKey || handler.evt.metaKey(end). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event. │ │ │ │ │ - */ │ │ │ │ │ - setEvent: function(evt) { │ │ │ │ │ - this.evt = evt; │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Deconstruct the handler. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - // unregister event listeners │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - // eliminate circular references │ │ │ │ │ - this.control = this.map = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_NONE │ │ │ │ │ - * If set as the <keyMask>, <checkModifiers> returns false if any key is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_NONE = 0; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_SHIFT │ │ │ │ │ - * If set as the <keyMask>, <checkModifiers> returns false if Shift is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_SHIFT = 1; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_CTRL │ │ │ │ │ - * If set as the <keyMask>, <checkModifiers> returns false if Ctrl is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_CTRL = 2; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_ALT │ │ │ │ │ - * If set as the <keyMask>, <checkModifiers> returns false if Alt is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_ALT = 4; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_META │ │ │ │ │ - * If set as the <keyMask>, <checkModifiers> returns false if Cmd is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_META = 8; │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Drag.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Handler.Drag │ │ │ │ │ - * The drag handler is used to deal with sequences of browser events related │ │ │ │ │ - * to dragging. The handler is used by controls that want to know when │ │ │ │ │ - * a drag sequence begins, when a drag is happening, and when it has │ │ │ │ │ - * finished. │ │ │ │ │ - * │ │ │ │ │ - * Controls that use the drag handler typically construct it with callbacks │ │ │ │ │ - * for 'down', 'move', and 'done'. Callbacks for these keys are called │ │ │ │ │ - * when the drag begins, with each move, and when the drag is done. In │ │ │ │ │ - * addition, controls can have callbacks keyed to 'up' and 'out' if they │ │ │ │ │ - * care to differentiate between the types of events that correspond with │ │ │ │ │ - * the end of a drag sequence. If no drag actually occurs (no mouse move) │ │ │ │ │ - * the 'down' and 'up' callbacks will be called, but not the 'done' │ │ │ │ │ - * callback. │ │ │ │ │ - * │ │ │ │ │ - * Create a new drag handler with the <OpenLayers.Handler.Drag> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.Drag = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: started │ │ │ │ │ - * {Boolean} When a mousedown or touchstart event is received, we want to │ │ │ │ │ - * record it, but not set 'dragging' until the mouse moves after starting. │ │ │ │ │ - */ │ │ │ │ │ - started: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: stopDown │ │ │ │ │ - * {Boolean} Stop propagation of mousedown events from getting to listeners │ │ │ │ │ - * on the same element. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - stopDown: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: dragging │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - dragging: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: last │ │ │ │ │ - * {<OpenLayers.Pixel>} The last pixel location of the drag. │ │ │ │ │ - */ │ │ │ │ │ - last: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: start │ │ │ │ │ - * {<OpenLayers.Pixel>} The first pixel location of the drag. │ │ │ │ │ - */ │ │ │ │ │ - start: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: lastMoveEvt │ │ │ │ │ - * {Object} The last mousemove event that occurred. Used to │ │ │ │ │ - * position the map correctly when our "delay drag" │ │ │ │ │ - * timeout expired. │ │ │ │ │ - */ │ │ │ │ │ - lastMoveEvt: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: oldOnselectstart │ │ │ │ │ - * {Function} │ │ │ │ │ - */ │ │ │ │ │ - oldOnselectstart: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: interval │ │ │ │ │ - * {Integer} In order to increase performance, an interval (in │ │ │ │ │ - * milliseconds) can be set to reduce the number of drag events │ │ │ │ │ - * called. If set, a new drag event will not be set until the │ │ │ │ │ - * interval has passed. │ │ │ │ │ - * Defaults to 0, meaning no interval. │ │ │ │ │ - */ │ │ │ │ │ - interval: 0, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: timeoutId │ │ │ │ │ - * {String} The id of the timeout used for the mousedown interval. │ │ │ │ │ - * This is "private", and should be left alone. │ │ │ │ │ - */ │ │ │ │ │ - timeoutId: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: documentDrag │ │ │ │ │ - * {Boolean} If set to true, the handler will also handle mouse moves when │ │ │ │ │ - * the cursor has moved out of the map viewport. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - documentDrag: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: documentEvents │ │ │ │ │ - * {Boolean} Are we currently observing document events? │ │ │ │ │ - */ │ │ │ │ │ - documentEvents: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Drag │ │ │ │ │ - * Returns OpenLayers.Handler.Drag │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ - * this handler. If a handler is being used without a control, the │ │ │ │ │ - * handlers setMap method must be overridden to deal properly with │ │ │ │ │ - * the map. │ │ │ │ │ - * callbacks - {Object} An object containing a single function to be │ │ │ │ │ - * called when the drag operation is finished. The callback should │ │ │ │ │ - * expect to recieve a single argument, the pixel location of the event. │ │ │ │ │ - * Callbacks for 'move' and 'done' are supported. You can also speficy │ │ │ │ │ - * callbacks for 'down', 'up', and 'out' to respond to those events. │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - if (this.documentDrag === true) { │ │ │ │ │ - var me = this; │ │ │ │ │ - this._docMove = function(evt) { │ │ │ │ │ - me.mousemove({ │ │ │ │ │ - xy: { │ │ │ │ │ - x: evt.clientX, │ │ │ │ │ - y: evt.clientY │ │ │ │ │ - }, │ │ │ │ │ - element: document │ │ │ │ │ - }); │ │ │ │ │ - }; │ │ │ │ │ - this._docUp = function(evt) { │ │ │ │ │ - me.mouseup({ │ │ │ │ │ - xy: { │ │ │ │ │ - x: evt.clientX, │ │ │ │ │ - y: evt.clientY │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: dragstart │ │ │ │ │ - * This private method is factorized from mousedown and touchstart methods │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - dragstart: function(evt) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - if (this.checkModifiers(evt) && │ │ │ │ │ - (OpenLayers.Event.isLeftClick(evt) || │ │ │ │ │ - OpenLayers.Event.isSingleTouch(evt))) { │ │ │ │ │ - this.started = true; │ │ │ │ │ - this.start = evt.xy; │ │ │ │ │ - this.last = evt.xy; │ │ │ │ │ - OpenLayers.Element.addClass( │ │ │ │ │ - this.map.viewPortDiv, "olDragDown" │ │ │ │ │ - ); │ │ │ │ │ - this.down(evt); │ │ │ │ │ - this.callback("down", [evt.xy]); │ │ │ │ │ - │ │ │ │ │ - // prevent document dragging │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ - │ │ │ │ │ - if (!this.oldOnselectstart) { │ │ │ │ │ - this.oldOnselectstart = document.onselectstart ? │ │ │ │ │ - document.onselectstart : OpenLayers.Function.True; │ │ │ │ │ - } │ │ │ │ │ - document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ - │ │ │ │ │ - propagate = !this.stopDown; │ │ │ │ │ - } else { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - } │ │ │ │ │ - return propagate; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: dragmove │ │ │ │ │ - * This private method is factorized from mousemove and touchmove methods │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - dragmove: function(evt) { │ │ │ │ │ - this.lastMoveEvt = evt; │ │ │ │ │ - if (this.started && !this.timeoutId && (evt.xy.x != this.last.x || │ │ │ │ │ - evt.xy.y != this.last.y)) { │ │ │ │ │ - if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ - if (evt.element === document) { │ │ │ │ │ - this.adjustXY(evt); │ │ │ │ │ - // do setEvent manually because the documentEvents are not │ │ │ │ │ - // registered with the map │ │ │ │ │ - this.setEvent(evt); │ │ │ │ │ - } else { │ │ │ │ │ - this.removeDocumentEvents(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.interval > 0) { │ │ │ │ │ - this.timeoutId = setTimeout( │ │ │ │ │ - OpenLayers.Function.bind(this.removeTimeout, this), │ │ │ │ │ - this.interval); │ │ │ │ │ - } │ │ │ │ │ - this.dragging = true; │ │ │ │ │ - │ │ │ │ │ - this.move(evt); │ │ │ │ │ - this.callback("move", [evt.xy]); │ │ │ │ │ - if (!this.oldOnselectstart) { │ │ │ │ │ - this.oldOnselectstart = document.onselectstart; │ │ │ │ │ - document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ - } │ │ │ │ │ - this.last = evt.xy; │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: dragend │ │ │ │ │ - * This private method is factorized from mouseup and touchend methods │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - dragend: function(evt) { │ │ │ │ │ - if (this.started) { │ │ │ │ │ - if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ - this.adjustXY(evt); │ │ │ │ │ - this.removeDocumentEvents(); │ │ │ │ │ - } │ │ │ │ │ - var dragged = (this.start != this.last); │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - OpenLayers.Element.removeClass( │ │ │ │ │ - this.map.viewPortDiv, "olDragDown" │ │ │ │ │ - ); │ │ │ │ │ - this.up(evt); │ │ │ │ │ - this.callback("up", [evt.xy]); │ │ │ │ │ - if (dragged) { │ │ │ │ │ - this.callback("done", [evt.xy]); │ │ │ │ │ - } │ │ │ │ │ - document.onselectstart = this.oldOnselectstart; │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * The four methods below (down, move, up, and out) are used by subclasses │ │ │ │ │ - * to do their own processing related to these mouse events. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: down │ │ │ │ │ - * This method is called during the handling of the mouse down event. │ │ │ │ │ - * Subclasses can do their own processing here. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The mouse down event │ │ │ │ │ - */ │ │ │ │ │ - down: function(evt) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * This method is called during the handling of the mouse move event. │ │ │ │ │ - * Subclasses can do their own processing here. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The mouse move event │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - move: function(evt) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: up │ │ │ │ │ - * This method is called during the handling of the mouse up event. │ │ │ │ │ - * Subclasses can do their own processing here. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The mouse up event │ │ │ │ │ - */ │ │ │ │ │ - up: function(evt) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: out │ │ │ │ │ - * This method is called during the handling of the mouse out event. │ │ │ │ │ - * Subclasses can do their own processing here. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The mouse out event │ │ │ │ │ - */ │ │ │ │ │ - out: function(evt) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * The methods below are part of the magic of event handling. Because │ │ │ │ │ - * they are named like browser events, they are registered as listeners │ │ │ │ │ - * for the events they represent. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mousedown │ │ │ │ │ - * Handle mousedown events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - return this.dragstart(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchstart │ │ │ │ │ - * Handle touchstart events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - return this.dragstart(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mousemove │ │ │ │ │ - * Handle mousemove events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - return this.dragmove(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchmove │ │ │ │ │ - * Handle touchmove events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - return this.dragmove(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeTimeout │ │ │ │ │ - * Private. Called by mousemove() to remove the drag timeout. │ │ │ │ │ - */ │ │ │ │ │ - removeTimeout: function() { │ │ │ │ │ - this.timeoutId = null; │ │ │ │ │ - // if timeout expires while we're still dragging (mouseup │ │ │ │ │ - // hasn't occurred) then call mousemove to move to the │ │ │ │ │ - // correct position │ │ │ │ │ - if (this.dragging) { │ │ │ │ │ - this.mousemove(this.lastMoveEvt); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mouseup │ │ │ │ │ - * Handle mouseup events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - return this.dragend(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchend │ │ │ │ │ - * Handle touchend events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - // override evt.xy with last position since touchend does not have │ │ │ │ │ - // any touch position │ │ │ │ │ - evt.xy = this.last; │ │ │ │ │ - return this.dragend(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mouseout │ │ │ │ │ - * Handle mouseout events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - mouseout: function(evt) { │ │ │ │ │ - if (this.started && OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ - if (this.documentDrag === true) { │ │ │ │ │ - this.addDocumentEvents(); │ │ │ │ │ - } else { │ │ │ │ │ - var dragged = (this.start != this.last); │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - OpenLayers.Element.removeClass( │ │ │ │ │ - this.map.viewPortDiv, "olDragDown" │ │ │ │ │ - ); │ │ │ │ │ - this.out(evt); │ │ │ │ │ - this.callback("out", []); │ │ │ │ │ - if (dragged) { │ │ │ │ │ - this.callback("done", [evt.xy]); │ │ │ │ │ - } │ │ │ │ │ - if (document.onselectstart) { │ │ │ │ │ - document.onselectstart = this.oldOnselectstart; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: click │ │ │ │ │ - * The drag handler captures the click event. If something else registers │ │ │ │ │ - * for clicks on the same element, its listener will not be called │ │ │ │ │ - * after a drag. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - // let the click event propagate only if the mouse moved │ │ │ │ │ - return (this.start == this.last); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Activate the handler. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - activated = true; │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Deactivate the handler. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - deactivated = true; │ │ │ │ │ - OpenLayers.Element.removeClass( │ │ │ │ │ - this.map.viewPortDiv, "olDragDown" │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: adjustXY │ │ │ │ │ - * Converts event coordinates that are relative to the document body to │ │ │ │ │ - * ones that are relative to the map viewport. The latter is the default in │ │ │ │ │ - * OpenLayers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ - */ │ │ │ │ │ - adjustXY: function(evt) { │ │ │ │ │ - var pos = OpenLayers.Util.pagePosition(this.map.viewPortDiv); │ │ │ │ │ - evt.xy.x -= pos[0]; │ │ │ │ │ - evt.xy.y -= pos[1]; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addDocumentEvents │ │ │ │ │ - * Start observing document events when documentDrag is true and the mouse │ │ │ │ │ - * cursor leaves the map viewport while dragging. │ │ │ │ │ - */ │ │ │ │ │ - addDocumentEvents: function() { │ │ │ │ │ - OpenLayers.Element.addClass(document.body, "olDragDown"); │ │ │ │ │ - this.documentEvents = true; │ │ │ │ │ - OpenLayers.Event.observe(document, "mousemove", this._docMove); │ │ │ │ │ - OpenLayers.Event.observe(document, "mouseup", this._docUp); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeDocumentEvents │ │ │ │ │ - * Stops observing document events when documentDrag is true and the mouse │ │ │ │ │ - * cursor re-enters the map viewport while dragging. │ │ │ │ │ - */ │ │ │ │ │ - removeDocumentEvents: function() { │ │ │ │ │ - OpenLayers.Element.removeClass(document.body, "olDragDown"); │ │ │ │ │ - this.documentEvents = false; │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mousemove", this._docMove); │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mouseup", this._docUp); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Drag" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Box.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ - * @requires OpenLayers/Handler/Drag.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Handler.Box │ │ │ │ │ - * Handler for dragging a rectangle across the map. Box is displayed │ │ │ │ │ - * on mouse down, moves on mouse move, and is finished on mouse up. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.Box = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: dragHandler │ │ │ │ │ - * {<OpenLayers.Handler.Drag>} │ │ │ │ │ - */ │ │ │ │ │ - dragHandler: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: boxDivClassName │ │ │ │ │ - * {String} The CSS class to use for drawing the box. Default is │ │ │ │ │ - * olHandlerBoxZoomBox │ │ │ │ │ - */ │ │ │ │ │ - boxDivClassName: 'olHandlerBoxZoomBox', │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: boxOffsets │ │ │ │ │ - * {Object} Caches box offsets from css. This is used by the getBoxOffsets │ │ │ │ │ - * method. │ │ │ │ │ - */ │ │ │ │ │ - boxOffsets: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Box │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} │ │ │ │ │ - * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ - * functions. Various callbacks described below. │ │ │ │ │ - * options - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Named callbacks: │ │ │ │ │ - * start - Called when the box drag operation starts. │ │ │ │ │ - * done - Called when the box drag operation is finished. │ │ │ │ │ - * The callback should expect to receive a single argument, the box │ │ │ │ │ - * bounds or a pixel. If the box dragging didn't span more than a 5 │ │ │ │ │ - * pixel distance, a pixel will be returned instead of a bounds object. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.dragHandler = new OpenLayers.Handler.Drag( │ │ │ │ │ - this, { │ │ │ │ │ - down: this.startBox, │ │ │ │ │ - move: this.moveBox, │ │ │ │ │ - out: this.removeBox, │ │ │ │ │ - up: this.endBox │ │ │ │ │ - }, { │ │ │ │ │ - keyMask: this.keyMask │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ - if (this.dragHandler) { │ │ │ │ │ - this.dragHandler.destroy(); │ │ │ │ │ - this.dragHandler = null; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Handler.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (this.dragHandler) { │ │ │ │ │ - this.dragHandler.setMap(map); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: startBox │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} │ │ │ │ │ - */ │ │ │ │ │ - startBox: function(xy) { │ │ │ │ │ - this.callback("start", []); │ │ │ │ │ - this.zoomBox = OpenLayers.Util.createDiv('zoomBox', { │ │ │ │ │ - x: -9999, │ │ │ │ │ - y: -9999 │ │ │ │ │ - }); │ │ │ │ │ - this.zoomBox.className = this.boxDivClassName; │ │ │ │ │ - this.zoomBox.style.zIndex = this.map.Z_INDEX_BASE["Popup"] - 1; │ │ │ │ │ - │ │ │ │ │ - this.map.viewPortDiv.appendChild(this.zoomBox); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Element.addClass( │ │ │ │ │ - this.map.viewPortDiv, "olDrawBox" │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveBox │ │ │ │ │ - */ │ │ │ │ │ - moveBox: function(xy) { │ │ │ │ │ - var startX = this.dragHandler.start.x; │ │ │ │ │ - var startY = this.dragHandler.start.y; │ │ │ │ │ - var deltaX = Math.abs(startX - xy.x); │ │ │ │ │ - var deltaY = Math.abs(startY - xy.y); │ │ │ │ │ - │ │ │ │ │ - var offset = this.getBoxOffsets(); │ │ │ │ │ - this.zoomBox.style.width = (deltaX + offset.width + 1) + "px"; │ │ │ │ │ - this.zoomBox.style.height = (deltaY + offset.height + 1) + "px"; │ │ │ │ │ - this.zoomBox.style.left = (xy.x < startX ? │ │ │ │ │ - startX - deltaX - offset.left : startX - offset.left) + "px"; │ │ │ │ │ - this.zoomBox.style.top = (xy.y < startY ? │ │ │ │ │ - startY - deltaY - offset.top : startY - offset.top) + "px"; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: endBox │ │ │ │ │ - */ │ │ │ │ │ - endBox: function(end) { │ │ │ │ │ - var result; │ │ │ │ │ - if (Math.abs(this.dragHandler.start.x - end.x) > 5 || │ │ │ │ │ - Math.abs(this.dragHandler.start.y - end.y) > 5) { │ │ │ │ │ - var start = this.dragHandler.start; │ │ │ │ │ - var top = Math.min(start.y, end.y); │ │ │ │ │ - var bottom = Math.max(start.y, end.y); │ │ │ │ │ - var left = Math.min(start.x, end.x); │ │ │ │ │ - var right = Math.max(start.x, end.x); │ │ │ │ │ - result = new OpenLayers.Bounds(left, bottom, right, top); │ │ │ │ │ - } else { │ │ │ │ │ - result = this.dragHandler.start.clone(); // i.e. OL.Pixel │ │ │ │ │ - } │ │ │ │ │ - this.removeBox(); │ │ │ │ │ - │ │ │ │ │ - this.callback("done", [result]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeBox │ │ │ │ │ - * Remove the zoombox from the screen and nullify our reference to it. │ │ │ │ │ - */ │ │ │ │ │ - removeBox: function() { │ │ │ │ │ - this.map.viewPortDiv.removeChild(this.zoomBox); │ │ │ │ │ - this.zoomBox = null; │ │ │ │ │ - this.boxOffsets = null; │ │ │ │ │ - OpenLayers.Element.removeClass( │ │ │ │ │ - this.map.viewPortDiv, "olDrawBox" │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.dragHandler.activate(); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - if (this.dragHandler.deactivate()) { │ │ │ │ │ - if (this.zoomBox) { │ │ │ │ │ - this.removeBox(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getBoxOffsets │ │ │ │ │ - * Determines border offsets for a box, according to the box model. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} an object with the following offsets: │ │ │ │ │ - * - left │ │ │ │ │ - * - right │ │ │ │ │ - * - top │ │ │ │ │ - * - bottom │ │ │ │ │ - * - width │ │ │ │ │ - * - height │ │ │ │ │ - */ │ │ │ │ │ - getBoxOffsets: function() { │ │ │ │ │ - if (!this.boxOffsets) { │ │ │ │ │ - // Determine the box model. If the testDiv's clientWidth is 3, then │ │ │ │ │ - // the borders are outside and we are dealing with the w3c box │ │ │ │ │ - // model. Otherwise, the browser uses the traditional box model and │ │ │ │ │ - // the borders are inside the box bounds, leaving us with a │ │ │ │ │ - // clientWidth of 1. │ │ │ │ │ - var testDiv = document.createElement("div"); │ │ │ │ │ - //testDiv.style.visibility = "hidden"; │ │ │ │ │ - testDiv.style.position = "absolute"; │ │ │ │ │ - testDiv.style.border = "1px solid black"; │ │ │ │ │ - testDiv.style.width = "3px"; │ │ │ │ │ - document.body.appendChild(testDiv); │ │ │ │ │ - var w3cBoxModel = testDiv.clientWidth == 3; │ │ │ │ │ - document.body.removeChild(testDiv); │ │ │ │ │ - │ │ │ │ │ - var left = parseInt(OpenLayers.Element.getStyle(this.zoomBox, │ │ │ │ │ - "border-left-width")); │ │ │ │ │ - var right = parseInt(OpenLayers.Element.getStyle( │ │ │ │ │ - this.zoomBox, "border-right-width")); │ │ │ │ │ - var top = parseInt(OpenLayers.Element.getStyle(this.zoomBox, │ │ │ │ │ - "border-top-width")); │ │ │ │ │ - var bottom = parseInt(OpenLayers.Element.getStyle( │ │ │ │ │ - this.zoomBox, "border-bottom-width")); │ │ │ │ │ - this.boxOffsets = { │ │ │ │ │ - left: left, │ │ │ │ │ - right: right, │ │ │ │ │ - top: top, │ │ │ │ │ - bottom: bottom, │ │ │ │ │ - width: w3cBoxModel === false ? left + right : 0, │ │ │ │ │ - height: w3cBoxModel === false ? top + bottom : 0 │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - return this.boxOffsets; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Box" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/ZoomBox.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Handler/Box.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.ZoomBox │ │ │ │ │ - * The ZoomBox control enables zooming directly to a given extent, by drawing │ │ │ │ │ - * a box on the map. The box is drawn by holding down shift, whilst dragging │ │ │ │ │ - * the mouse. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ZoomBox = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - /** │ │ │ │ │ - * Property: type │ │ │ │ │ - * {OpenLayers.Control.TYPE} │ │ │ │ │ - */ │ │ │ │ │ - type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: out │ │ │ │ │ - * {Boolean} Should the control be used for zooming out? │ │ │ │ │ - */ │ │ │ │ │ - out: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: keyMask │ │ │ │ │ - * {Integer} Zoom only occurs if the keyMask matches the combination of │ │ │ │ │ - * keys down. Use bitwise operators and one or more of the │ │ │ │ │ - * <OpenLayers.Handler> constants to construct a keyMask. Leave null if │ │ │ │ │ - * not used mask. Default is null. │ │ │ │ │ - */ │ │ │ │ │ - keyMask: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: alwaysZoom │ │ │ │ │ - * {Boolean} Always zoom in/out when box drawn, even if the zoom level does │ │ │ │ │ - * not change. │ │ │ │ │ - */ │ │ │ │ │ - alwaysZoom: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomOnClick │ │ │ │ │ - * {Boolean} Should we zoom when no box was dragged, i.e. the user only │ │ │ │ │ - * clicked? Default is true. │ │ │ │ │ - */ │ │ │ │ │ - zoomOnClick: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Box(this, { │ │ │ │ │ - done: this.zoomBox │ │ │ │ │ - }, { │ │ │ │ │ - keyMask: this.keyMask │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: zoomBox │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * position - {<OpenLayers.Bounds>} or {<OpenLayers.Pixel>} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * position - {<OpenLayers.Bounds>} or {<OpenLayers.Pixel>} │ │ │ │ │ */ │ │ │ │ │ zoomBox: function(position) { │ │ │ │ │ if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ var bounds, │ │ │ │ │ targetCenterPx = position.getCenterPixel(); │ │ │ │ │ if (!this.out) { │ │ │ │ │ var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ @@ -29936,1366 +25775,2874 @@ │ │ │ │ │ this.handlers.wheel.activate(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.Navigation" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Events/buttonclick.js │ │ │ │ │ + OpenLayers/Handler/Feature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Events.buttonclick │ │ │ │ │ - * Extension event type for handling buttons on top of a dom element. This │ │ │ │ │ - * event type fires "buttonclick" on its <target> when a button was │ │ │ │ │ - * clicked. Buttons are detected by the "olButton" class. │ │ │ │ │ + * Class: OpenLayers.Handler.Feature │ │ │ │ │ + * Handler to respond to mouse events related to a drawn feature. Callbacks │ │ │ │ │ + * with the following keys will be notified of the following events │ │ │ │ │ + * associated with features: click, clickout, over, out, and dblclick. │ │ │ │ │ * │ │ │ │ │ - * This event type makes sure that button clicks do not interfere with other │ │ │ │ │ - * events that are registered on the same <element>. │ │ │ │ │ + * This handler stops event propagation for mousedown and mouseup if those │ │ │ │ │ + * browser events target features that can be selected. │ │ │ │ │ * │ │ │ │ │ - * Event types provided by this extension: │ │ │ │ │ - * - *buttonclick* Triggered when a button is clicked. Listeners receive an │ │ │ │ │ - * object with a *buttonElement* property referencing the dom element of │ │ │ │ │ - * the clicked button, and an *buttonXY* property with the click position │ │ │ │ │ - * relative to the button. │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Events.buttonclick = OpenLayers.Class({ │ │ │ │ │ +OpenLayers.Handler.Feature = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: target │ │ │ │ │ - * {<OpenLayers.Events>} The events instance that the buttonclick event will │ │ │ │ │ - * be triggered on. │ │ │ │ │ + * Property: EVENTMAP │ │ │ │ │ + * {Object} A object mapping the browser events to objects with callback │ │ │ │ │ + * keys for in and out. │ │ │ │ │ */ │ │ │ │ │ - target: null, │ │ │ │ │ + EVENTMAP: { │ │ │ │ │ + 'click': { │ │ │ │ │ + 'in': 'click', │ │ │ │ │ + 'out': 'clickout' │ │ │ │ │ + }, │ │ │ │ │ + 'mousemove': { │ │ │ │ │ + 'in': 'over', │ │ │ │ │ + 'out': 'out' │ │ │ │ │ + }, │ │ │ │ │ + 'dblclick': { │ │ │ │ │ + 'in': 'dblclick', │ │ │ │ │ + 'out': null │ │ │ │ │ + }, │ │ │ │ │ + 'mousedown': { │ │ │ │ │ + 'in': null, │ │ │ │ │ + 'out': null │ │ │ │ │ + }, │ │ │ │ │ + 'mouseup': { │ │ │ │ │ + 'in': null, │ │ │ │ │ + 'out': null │ │ │ │ │ + }, │ │ │ │ │ + 'touchstart': { │ │ │ │ │ + 'in': 'click', │ │ │ │ │ + 'out': 'clickout' │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: events │ │ │ │ │ - * {Array} Events to observe and conditionally stop from propagating when │ │ │ │ │ - * an element with the olButton class (or its olAlphaImg child) is │ │ │ │ │ - * clicked. │ │ │ │ │ + * Property: feature │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} The last feature that was hovered. │ │ │ │ │ */ │ │ │ │ │ - events: [ │ │ │ │ │ - 'mousedown', 'mouseup', 'click', 'dblclick', │ │ │ │ │ - 'touchstart', 'touchmove', 'touchend', 'keydown' │ │ │ │ │ - ], │ │ │ │ │ + feature: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: startRegEx │ │ │ │ │ - * {RegExp} Regular expression to test Event.type for events that start │ │ │ │ │ - * a buttonclick sequence. │ │ │ │ │ + * Property: lastFeature │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} The last feature that was handled. │ │ │ │ │ */ │ │ │ │ │ - startRegEx: /^mousedown|touchstart$/, │ │ │ │ │ + lastFeature: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: cancelRegEx │ │ │ │ │ - * {RegExp} Regular expression to test Event.type for events that cancel │ │ │ │ │ - * a buttonclick sequence. │ │ │ │ │ + * Property: down │ │ │ │ │ + * {<OpenLayers.Pixel>} The location of the last mousedown. │ │ │ │ │ */ │ │ │ │ │ - cancelRegEx: /^touchmove$/, │ │ │ │ │ + down: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: completeRegEx │ │ │ │ │ - * {RegExp} Regular expression to test Event.type for events that complete │ │ │ │ │ - * a buttonclick sequence. │ │ │ │ │ + * Property: up │ │ │ │ │ + * {<OpenLayers.Pixel>} The location of the last mouseup. │ │ │ │ │ */ │ │ │ │ │ - completeRegEx: /^mouseup|touchend$/, │ │ │ │ │ + up: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: startEvt │ │ │ │ │ - * {Event} The event that started the click sequence │ │ │ │ │ + * Property: clickTolerance │ │ │ │ │ + * {Number} The number of pixels the mouse can move between mousedown │ │ │ │ │ + * and mouseup for the event to still be considered a click. │ │ │ │ │ + * Dragging the map should not trigger the click and clickout callbacks │ │ │ │ │ + * unless the map is moved by less than this tolerance. Defaults to 4. │ │ │ │ │ */ │ │ │ │ │ + clickTolerance: 4, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Events.buttonclick │ │ │ │ │ - * Construct a buttonclick event type. Applications are not supposed to │ │ │ │ │ - * create instances of this class - they are created on demand by │ │ │ │ │ - * <OpenLayers.Events> instances. │ │ │ │ │ + * Property: geometryTypes │ │ │ │ │ + * To restrict dragging to a limited set of geometry types, send a list │ │ │ │ │ + * of strings corresponding to the geometry class names. │ │ │ │ │ + * │ │ │ │ │ + * @type Array(String) │ │ │ │ │ + */ │ │ │ │ │ + geometryTypes: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: stopClick │ │ │ │ │ + * {Boolean} If stopClick is set to true, handled clicks do not │ │ │ │ │ + * propagate to other click listeners. Otherwise, handled clicks │ │ │ │ │ + * do propagate. Unhandled clicks always propagate, whatever the │ │ │ │ │ + * value of stopClick. Defaults to true. │ │ │ │ │ + */ │ │ │ │ │ + stopClick: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: stopDown │ │ │ │ │ + * {Boolean} If stopDown is set to true, handled mousedowns do not │ │ │ │ │ + * propagate to other mousedown listeners. Otherwise, handled │ │ │ │ │ + * mousedowns do propagate. Unhandled mousedowns always propagate, │ │ │ │ │ + * whatever the value of stopDown. Defaults to true. │ │ │ │ │ + */ │ │ │ │ │ + stopDown: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: stopUp │ │ │ │ │ + * {Boolean} If stopUp is set to true, handled mouseups do not │ │ │ │ │ + * propagate to other mouseup listeners. Otherwise, handled mouseups │ │ │ │ │ + * do propagate. Unhandled mouseups always propagate, whatever the │ │ │ │ │ + * value of stopUp. Defaults to false. │ │ │ │ │ + */ │ │ │ │ │ + stopUp: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Handler.Feature │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * target - {<OpenLayers.Events>} The events instance that the buttonclick │ │ │ │ │ - * event will be triggered on. │ │ │ │ │ + * control - {<OpenLayers.Control>} │ │ │ │ │ + * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ + * callbacks - {Object} An object with a 'over' property whos value is │ │ │ │ │ + * a function to be called when the mouse is over a feature. The │ │ │ │ │ + * callback should expect to recieve a single argument, the feature. │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(target) { │ │ │ │ │ - this.target = target; │ │ │ │ │ - for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ - this.target.register(this.events[i], this, this.buttonClick, { │ │ │ │ │ - extension: true │ │ │ │ │ - }); │ │ │ │ │ + initialize: function(control, layer, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: touchstart │ │ │ │ │ + * Handle touchstart events │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ + */ │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + return OpenLayers.Event.isMultiTouch(evt) ? │ │ │ │ │ + true : this.mousedown(evt); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: touchmove │ │ │ │ │ + * Handle touchmove events. We just prevent the browser default behavior, │ │ │ │ │ + * for Android Webkit not to select text when moving the finger after │ │ │ │ │ + * selecting a feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: mousedown │ │ │ │ │ + * Handle mouse down. Stop propagation if a feature is targeted by this │ │ │ │ │ + * event (stops map dragging during feature selection). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + // Feature selection is only done with a left click. Other handlers may stop the │ │ │ │ │ + // propagation of left-click mousedown events but not right-click mousedown events. │ │ │ │ │ + // This mismatch causes problems when comparing the location of the down and up │ │ │ │ │ + // events in the click function so it is important ignore right-clicks. │ │ │ │ │ + if (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ + this.down = evt.xy; │ │ │ │ │ } │ │ │ │ │ + return this.handle(evt) ? !this.stopDown : true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * Method: mouseup │ │ │ │ │ + * Handle mouse up. Stop propagation if a feature is targeted by this │ │ │ │ │ + * event. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ - this.target.unregister(this.events[i], this, this.buttonClick); │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + this.up = evt.xy; │ │ │ │ │ + return this.handle(evt) ? !this.stopUp : true; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: click │ │ │ │ │ + * Handle click. Call the "click" callback if click on a feature, │ │ │ │ │ + * or the "clickout" callback if click outside any feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + return this.handle(evt) ? !this.stopClick : true; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: mousemove │ │ │ │ │ + * Handle mouse moves. Call the "over" callback if moving in to a feature, │ │ │ │ │ + * or the "out" callback if moving out of a feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + if (!this.callbacks['over'] && !this.callbacks['out']) { │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ - delete this.target; │ │ │ │ │ + this.handle(evt); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getPressedButton │ │ │ │ │ - * Get the pressed button, if any. Returns undefined if no button │ │ │ │ │ - * was pressed. │ │ │ │ │ + * Method: dblclick │ │ │ │ │ + * Handle dblclick. Call the "dblclick" callback if dblclick on a feature. │ │ │ │ │ * │ │ │ │ │ - * Arguments: │ │ │ │ │ - * element - {DOMElement} The event target. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} The button element, or undefined. │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - getPressedButton: function(element) { │ │ │ │ │ - var depth = 3, // limit the search depth │ │ │ │ │ - button; │ │ │ │ │ - do { │ │ │ │ │ - if (OpenLayers.Element.hasClass(element, "olButton")) { │ │ │ │ │ - // hit! │ │ │ │ │ - button = element; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - element = element.parentNode; │ │ │ │ │ - } while (--depth > 0 && element); │ │ │ │ │ - return button; │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + return !this.handle(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: ignore │ │ │ │ │ - * Check for event target elements that should be ignored by OpenLayers. │ │ │ │ │ + * Method: geometryTypeMatches │ │ │ │ │ + * Return true if the geometry type of the passed feature matches │ │ │ │ │ + * one of the geometry types in the geometryTypes array. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * element - {DOMElement} The event target. │ │ │ │ │ + * feature - {<OpenLayers.Vector.Feature>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - ignore: function(element) { │ │ │ │ │ - var depth = 3, │ │ │ │ │ - ignore = false; │ │ │ │ │ - do { │ │ │ │ │ - if (element.nodeName.toLowerCase() === 'a') { │ │ │ │ │ - ignore = true; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - element = element.parentNode; │ │ │ │ │ - } while (--depth > 0 && element); │ │ │ │ │ - return ignore; │ │ │ │ │ + geometryTypeMatches: function(feature) { │ │ │ │ │ + return this.geometryTypes == null || │ │ │ │ │ + OpenLayers.Util.indexOf(this.geometryTypes, │ │ │ │ │ + feature.geometry.CLASS_NAME) > -1; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buttonClick │ │ │ │ │ - * Check if a button was clicked, and fire the buttonclick event │ │ │ │ │ + * Method: handle │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The event occurred over a relevant feature. │ │ │ │ │ */ │ │ │ │ │ - buttonClick: function(evt) { │ │ │ │ │ - var propagate = true, │ │ │ │ │ - element = OpenLayers.Event.element(evt); │ │ │ │ │ - if (element && (OpenLayers.Event.isLeftClick(evt) || !~evt.type.indexOf("mouse"))) { │ │ │ │ │ - // was a button pressed? │ │ │ │ │ - var button = this.getPressedButton(element); │ │ │ │ │ - if (button) { │ │ │ │ │ - if (evt.type === "keydown") { │ │ │ │ │ - switch (evt.keyCode) { │ │ │ │ │ - case OpenLayers.Event.KEY_RETURN: │ │ │ │ │ - case OpenLayers.Event.KEY_SPACE: │ │ │ │ │ - this.target.triggerEvent("buttonclick", { │ │ │ │ │ - buttonElement: button │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } else if (this.startEvt) { │ │ │ │ │ - if (this.completeRegEx.test(evt.type)) { │ │ │ │ │ - var pos = OpenLayers.Util.pagePosition(button); │ │ │ │ │ - var viewportElement = OpenLayers.Util.getViewportElement(); │ │ │ │ │ - var scrollTop = window.pageYOffset || viewportElement.scrollTop; │ │ │ │ │ - var scrollLeft = window.pageXOffset || viewportElement.scrollLeft; │ │ │ │ │ - pos[0] = pos[0] - scrollLeft; │ │ │ │ │ - pos[1] = pos[1] - scrollTop; │ │ │ │ │ - │ │ │ │ │ - this.target.triggerEvent("buttonclick", { │ │ │ │ │ - buttonElement: button, │ │ │ │ │ - buttonXY: { │ │ │ │ │ - x: this.startEvt.clientX - pos[0], │ │ │ │ │ - y: this.startEvt.clientY - pos[1] │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - if (this.cancelRegEx.test(evt.type)) { │ │ │ │ │ - delete this.startEvt; │ │ │ │ │ + handle: function(evt) { │ │ │ │ │ + if (this.feature && !this.feature.layer) { │ │ │ │ │ + // feature has been destroyed │ │ │ │ │ + this.feature = null; │ │ │ │ │ + } │ │ │ │ │ + var type = evt.type; │ │ │ │ │ + var handled = false; │ │ │ │ │ + var previouslyIn = !!(this.feature); // previously in a feature │ │ │ │ │ + var click = (type == "click" || type == "dblclick" || type == "touchstart"); │ │ │ │ │ + this.feature = this.layer.getFeatureFromEvent(evt); │ │ │ │ │ + if (this.feature && !this.feature.layer) { │ │ │ │ │ + // feature has been destroyed │ │ │ │ │ + this.feature = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.lastFeature && !this.lastFeature.layer) { │ │ │ │ │ + // last feature has been destroyed │ │ │ │ │ + this.lastFeature = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.feature) { │ │ │ │ │ + if (type === "touchstart") { │ │ │ │ │ + // stop the event to prevent Android Webkit from │ │ │ │ │ + // "flashing" the map div │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + } │ │ │ │ │ + var inNew = (this.feature != this.lastFeature); │ │ │ │ │ + if (this.geometryTypeMatches(this.feature)) { │ │ │ │ │ + // in to a feature │ │ │ │ │ + if (previouslyIn && inNew) { │ │ │ │ │ + // out of last feature and in to another │ │ │ │ │ + if (this.lastFeature) { │ │ │ │ │ + this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false; │ │ │ │ │ + this.triggerCallback(type, 'in', [this.feature]); │ │ │ │ │ + } else if (!previouslyIn || click) { │ │ │ │ │ + // in feature for the first time │ │ │ │ │ + this.triggerCallback(type, 'in', [this.feature]); │ │ │ │ │ } │ │ │ │ │ - if (this.startRegEx.test(evt.type)) { │ │ │ │ │ - this.startEvt = evt; │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false; │ │ │ │ │ + this.lastFeature = this.feature; │ │ │ │ │ + handled = true; │ │ │ │ │ + } else { │ │ │ │ │ + // not in to a feature │ │ │ │ │ + if (this.lastFeature && (previouslyIn && inNew || click)) { │ │ │ │ │ + // out of last feature for the first time │ │ │ │ │ + this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ + } │ │ │ │ │ + // next time the mouse goes in a feature whose geometry type │ │ │ │ │ + // doesn't match we don't want to call the 'out' callback │ │ │ │ │ + // again, so let's set this.feature to null so that │ │ │ │ │ + // previouslyIn will evaluate to false the next time │ │ │ │ │ + // we enter handle. Yes, a bit hackish... │ │ │ │ │ + this.feature = null; │ │ │ │ │ + } │ │ │ │ │ + } else if (this.lastFeature && (previouslyIn || click)) { │ │ │ │ │ + this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ + } │ │ │ │ │ + return handled; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: triggerCallback │ │ │ │ │ + * Call the callback keyed in the event map with the supplied arguments. │ │ │ │ │ + * For click and clickout, the <clickTolerance> is checked first. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * type - {String} │ │ │ │ │ + */ │ │ │ │ │ + triggerCallback: function(type, mode, args) { │ │ │ │ │ + var key = this.EVENTMAP[type][mode]; │ │ │ │ │ + if (key) { │ │ │ │ │ + if (type == 'click' && this.up && this.down) { │ │ │ │ │ + // for click/clickout, only trigger callback if tolerance is met │ │ │ │ │ + var dpx = Math.sqrt( │ │ │ │ │ + Math.pow(this.up.x - this.down.x, 2) + │ │ │ │ │ + Math.pow(this.up.y - this.down.y, 2) │ │ │ │ │ + ); │ │ │ │ │ + if (dpx <= this.clickTolerance) { │ │ │ │ │ + this.callback(key, args); │ │ │ │ │ } │ │ │ │ │ + // we're done with this set of events now: clear the cached │ │ │ │ │ + // positions so we can't trip over them later (this can occur │ │ │ │ │ + // if one of the up/down events gets eaten before it gets to us │ │ │ │ │ + // but we still get the click) │ │ │ │ │ + this.up = this.down = null; │ │ │ │ │ } else { │ │ │ │ │ - propagate = !this.ignore(OpenLayers.Event.element(evt)); │ │ │ │ │ - delete this.startEvt; │ │ │ │ │ + this.callback(key, args); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return propagate; │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Turn on the handler. Returns false if the handler was already active. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerToTop(); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + "removelayer": this.handleMapEvents, │ │ │ │ │ + "changelayer": this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + activated = true; │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Turn off the handler. Returns false if the handler was already active. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerBack(); │ │ │ │ │ + this.feature = null; │ │ │ │ │ + this.lastFeature = null; │ │ │ │ │ + this.down = null; │ │ │ │ │ + this.up = null; │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + "removelayer": this.handleMapEvents, │ │ │ │ │ + "changelayer": this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + deactivated = true; │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleMapEvents │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} │ │ │ │ │ + */ │ │ │ │ │ + handleMapEvents: function(evt) { │ │ │ │ │ + if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ + this.moveLayerToTop(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveLayerToTop │ │ │ │ │ + * Moves the layer for this handler to the top, so mouse events can reach │ │ │ │ │ + * it. │ │ │ │ │ + */ │ │ │ │ │ + moveLayerToTop: function() { │ │ │ │ │ + var index = Math.max(this.map.Z_INDEX_BASE['Feature'] - 1, │ │ │ │ │ + this.layer.getZIndex()) + 1; │ │ │ │ │ + this.layer.setZIndex(index); │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveLayerBack │ │ │ │ │ + * Moves the layer back to the position determined by the map's layers │ │ │ │ │ + * array. │ │ │ │ │ + */ │ │ │ │ │ + moveLayerBack: function() { │ │ │ │ │ + var index = this.layer.getZIndex() - 1; │ │ │ │ │ + if (index >= this.map.Z_INDEX_BASE['Feature']) { │ │ │ │ │ + this.layer.setZIndex(index); │ │ │ │ │ + } else { │ │ │ │ │ + this.map.setLayerZIndex(this.layer, │ │ │ │ │ + this.map.getLayerIndex(this.layer)); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Feature" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Panel.js │ │ │ │ │ + OpenLayers/Layer.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Map.js │ │ │ │ │ + * @requires OpenLayers/Projection.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.Panel │ │ │ │ │ - * The Panel control is a container for other controls. With it toolbars │ │ │ │ │ - * may be composed. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ + * Class: OpenLayers.Layer │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.Panel = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Layer = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Property: controls │ │ │ │ │ - * {Array(<OpenLayers.Control>)} │ │ │ │ │ + * APIProperty: id │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - controls: null, │ │ │ │ │ + id: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: name │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ + name: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: defaultControl │ │ │ │ │ - * {<OpenLayers.Control>} The control which is activated when the control is │ │ │ │ │ - * activated (turned on), which also happens at instantiation. │ │ │ │ │ - * If <saveState> is true, <defaultControl> will be nullified after the │ │ │ │ │ - * first activation of the panel. │ │ │ │ │ + * APIProperty: div │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - defaultControl: null, │ │ │ │ │ + div: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: saveState │ │ │ │ │ - * {Boolean} If set to true, the active state of this panel's controls will │ │ │ │ │ - * be stored on panel deactivation, and restored on reactivation. Default │ │ │ │ │ - * is false. │ │ │ │ │ + * APIProperty: opacity │ │ │ │ │ + * {Float} The layer's opacity. Float number between 0.0 and 1.0. Default │ │ │ │ │ + * is 1. │ │ │ │ │ */ │ │ │ │ │ - saveState: false, │ │ │ │ │ + opacity: 1, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: allowDepress │ │ │ │ │ - * {Boolean} If is true the <OpenLayers.Control.TYPE_TOOL> controls can │ │ │ │ │ - * be deactivated by clicking the icon that represents them. Default │ │ │ │ │ - * is false. │ │ │ │ │ + * APIProperty: alwaysInRange │ │ │ │ │ + * {Boolean} If a layer's display should not be scale-based, this should │ │ │ │ │ + * be set to true. This will cause the layer, as an overlay, to always │ │ │ │ │ + * be 'active', by always returning true from the calculateInRange() │ │ │ │ │ + * function. │ │ │ │ │ + * │ │ │ │ │ + * If not explicitly specified for a layer, its value will be │ │ │ │ │ + * determined on startup in initResolutions() based on whether or not │ │ │ │ │ + * any scale-specific properties have been set as options on the │ │ │ │ │ + * layer. If no scale-specific options have been set on the layer, we │ │ │ │ │ + * assume that it should always be in range. │ │ │ │ │ + * │ │ │ │ │ + * See #987 for more info. │ │ │ │ │ */ │ │ │ │ │ - allowDepress: false, │ │ │ │ │ + alwaysInRange: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: activeState │ │ │ │ │ - * {Object} stores the active state of this panel's controls. │ │ │ │ │ + * Constant: RESOLUTION_PROPERTIES │ │ │ │ │ + * {Array} The properties that are used for calculating resolutions │ │ │ │ │ + * information. │ │ │ │ │ */ │ │ │ │ │ - activeState: null, │ │ │ │ │ + RESOLUTION_PROPERTIES: [ │ │ │ │ │ + 'scales', 'resolutions', │ │ │ │ │ + 'maxScale', 'minScale', │ │ │ │ │ + 'maxResolution', 'minResolution', │ │ │ │ │ + 'numZoomLevels', 'maxZoomLevel' │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Panel │ │ │ │ │ - * Create a new control panel. │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} │ │ │ │ │ * │ │ │ │ │ - * Each control in the panel is represented by an icon. When clicking │ │ │ │ │ - * on an icon, the <activateControl> method is called. │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * layer.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ * │ │ │ │ │ - * Specific properties for controls on a panel: │ │ │ │ │ - * type - {Number} One of <OpenLayers.Control.TYPE_TOOL>, │ │ │ │ │ - * <OpenLayers.Control.TYPE_TOGGLE>, <OpenLayers.Control.TYPE_BUTTON>. │ │ │ │ │ - * If not provided, <OpenLayers.Control.TYPE_TOOL> is assumed. │ │ │ │ │ - * title - {string} Text displayed when mouse is over the icon that │ │ │ │ │ - * represents the control. │ │ │ │ │ + * Listeners will be called with a reference to an event object. The │ │ │ │ │ + * properties of this event depends on exactly what happened. │ │ │ │ │ * │ │ │ │ │ - * The <OpenLayers.Control.type> of a control determines the behavior when │ │ │ │ │ - * clicking its icon: │ │ │ │ │ - * <OpenLayers.Control.TYPE_TOOL> - The control is activated and other │ │ │ │ │ - * controls of this type in the same panel are deactivated. This is │ │ │ │ │ - * the default type. │ │ │ │ │ - * <OpenLayers.Control.TYPE_TOGGLE> - The active state of the control is │ │ │ │ │ - * toggled. │ │ │ │ │ - * <OpenLayers.Control.TYPE_BUTTON> - The │ │ │ │ │ - * <OpenLayers.Control.Button.trigger> method of the control is called, │ │ │ │ │ - * but its active state is not changed. │ │ │ │ │ + * All event objects have at least the following properties: │ │ │ │ │ + * object - {Object} A reference to layer.events.object. │ │ │ │ │ + * element - {DOMElement} A reference to layer.events.element. │ │ │ │ │ * │ │ │ │ │ - * If a control is <OpenLayers.Control.active>, it will be drawn with the │ │ │ │ │ - * olControl[Name]ItemActive class, otherwise with the │ │ │ │ │ - * olControl[Name]ItemInactive class. │ │ │ │ │ + * Supported map event types: │ │ │ │ │ + * loadstart - Triggered when layer loading starts. When using a Vector │ │ │ │ │ + * layer with a Fixed or BBOX strategy, the event object includes │ │ │ │ │ + * a *filter* property holding the OpenLayers.Filter used when │ │ │ │ │ + * calling read on the protocol. │ │ │ │ │ + * loadend - Triggered when layer loading ends. When using a Vector layer │ │ │ │ │ + * with a Fixed or BBOX strategy, the event object includes a │ │ │ │ │ + * *response* property holding an OpenLayers.Protocol.Response object. │ │ │ │ │ + * visibilitychanged - Triggered when the layer's visibility property is │ │ │ │ │ + * changed, e.g. by turning the layer on or off in the layer switcher. │ │ │ │ │ + * Note that the actual visibility of the layer can also change if it │ │ │ │ │ + * gets out of range (see <calculateInRange>). If you also want to catch │ │ │ │ │ + * these cases, register for the map's 'changelayer' event instead. │ │ │ │ │ + * move - Triggered when layer moves (triggered with every mousemove │ │ │ │ │ + * during a drag). │ │ │ │ │ + * moveend - Triggered when layer is done moving, object passed as │ │ │ │ │ + * argument has a zoomChanged boolean property which tells that the │ │ │ │ │ + * zoom has changed. │ │ │ │ │ + * added - Triggered after the layer is added to a map. Listeners will │ │ │ │ │ + * receive an object with a *map* property referencing the map and a │ │ │ │ │ + * *layer* property referencing the layer. │ │ │ │ │ + * removed - Triggered after the layer is removed from the map. Listeners │ │ │ │ │ + * will receive an object with a *map* property referencing the map and │ │ │ │ │ + * a *layer* property referencing the layer. │ │ │ │ │ + */ │ │ │ │ │ + events: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: map │ │ │ │ │ + * {<OpenLayers.Map>} This variable is set when the layer is added to │ │ │ │ │ + * the map, via the accessor function setMap(). │ │ │ │ │ + */ │ │ │ │ │ + map: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} Whether or not the layer is a base layer. This should be set │ │ │ │ │ + * individually by all subclasses. Default is false │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: alpha │ │ │ │ │ + * {Boolean} The layer's images have an alpha channel. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + alpha: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: displayInLayerSwitcher │ │ │ │ │ + * {Boolean} Display the layer's name in the layer switcher. Default is │ │ │ │ │ + * true. │ │ │ │ │ + */ │ │ │ │ │ + displayInLayerSwitcher: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: visibility │ │ │ │ │ + * {Boolean} The layer should be displayed in the map. Default is true. │ │ │ │ │ + */ │ │ │ │ │ + visibility: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: attribution │ │ │ │ │ + * {String} Attribution string, displayed when an │ │ │ │ │ + * <OpenLayers.Control.Attribution> has been added to the map. │ │ │ │ │ + */ │ │ │ │ │ + attribution: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: inRange │ │ │ │ │ + * {Boolean} The current map resolution is within the layer's min/max │ │ │ │ │ + * range. This is set in <OpenLayers.Map.setCenter> whenever the zoom │ │ │ │ │ + * changes. │ │ │ │ │ + */ │ │ │ │ │ + inRange: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Propery: imageSize │ │ │ │ │ + * {<OpenLayers.Size>} For layers with a gutter, the image is larger than │ │ │ │ │ + * the tile by twice the gutter in each dimension. │ │ │ │ │ + */ │ │ │ │ │ + imageSize: null, │ │ │ │ │ + │ │ │ │ │ + // OPTIONS │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: options │ │ │ │ │ + * {Object} An optional object whose properties will be set on the layer. │ │ │ │ │ + * Any of the layer properties can be set as a property of the options │ │ │ │ │ + * object and sent to the constructor when the layer is created. │ │ │ │ │ + */ │ │ │ │ │ + options: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: eventListeners │ │ │ │ │ + * {Object} If set as an option at construction, the eventListeners │ │ │ │ │ + * object will be registered with <OpenLayers.Events.on>. Object │ │ │ │ │ + * structure must be a listeners object as shown in the example for │ │ │ │ │ + * the events.on method. │ │ │ │ │ + */ │ │ │ │ │ + eventListeners: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: gutter │ │ │ │ │ + * {Integer} Determines the width (in pixels) of the gutter around image │ │ │ │ │ + * tiles to ignore. By setting this property to a non-zero value, │ │ │ │ │ + * images will be requested that are wider and taller than the tile │ │ │ │ │ + * size by a value of 2 x gutter. This allows artifacts of rendering │ │ │ │ │ + * at tile edges to be ignored. Set a gutter value that is equal to │ │ │ │ │ + * half the size of the widest symbol that needs to be displayed. │ │ │ │ │ + * Defaults to zero. Non-tiled layers always have zero gutter. │ │ │ │ │ + */ │ │ │ │ │ + gutter: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: projection │ │ │ │ │ + * {<OpenLayers.Projection>} or {<String>} Specifies the projection of the layer. │ │ │ │ │ + * Can be set in the layer options. If not specified in the layer options, │ │ │ │ │ + * it is set to the default projection specified in the map, │ │ │ │ │ + * when the layer is added to the map. │ │ │ │ │ + * Projection along with default maxExtent and resolutions │ │ │ │ │ + * are set automatically with commercial baselayers in EPSG:3857, │ │ │ │ │ + * such as Google, Bing and OpenStreetMap, and do not need to be specified. │ │ │ │ │ + * Otherwise, if specifying projection, also set maxExtent, │ │ │ │ │ + * maxResolution or resolutions as appropriate. │ │ │ │ │ + * When using vector layers with strategies, layer projection should be set │ │ │ │ │ + * to the projection of the source data if that is different from the map default. │ │ │ │ │ + * │ │ │ │ │ + * Can be either a string or an <OpenLayers.Projection> object; │ │ │ │ │ + * if a string is passed, will be converted to an object when │ │ │ │ │ + * the layer is added to the map. │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + projection: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: units │ │ │ │ │ + * {String} The layer map units. Defaults to null. Possible values │ │ │ │ │ + * are 'degrees' (or 'dd'), 'm', 'ft', 'km', 'mi', 'inches'. │ │ │ │ │ + * Normally taken from the projection. │ │ │ │ │ + * Only required if both map and layers do not define a projection, │ │ │ │ │ + * or if they define a projection which does not define units. │ │ │ │ │ + */ │ │ │ │ │ + units: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: scales │ │ │ │ │ + * {Array} An array of map scales in descending order. The values in the │ │ │ │ │ + * array correspond to the map scale denominator. Note that these │ │ │ │ │ + * values only make sense if the display (monitor) resolution of the │ │ │ │ │ + * client is correctly guessed by whomever is configuring the │ │ │ │ │ + * application. In addition, the units property must also be set. │ │ │ │ │ + * Use <resolutions> instead wherever possible. │ │ │ │ │ + */ │ │ │ │ │ + scales: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: resolutions │ │ │ │ │ + * {Array} A list of map resolutions (map units per pixel) in descending │ │ │ │ │ + * order. If this is not set in the layer constructor, it will be set │ │ │ │ │ + * based on other resolution related properties (maxExtent, │ │ │ │ │ + * maxResolution, maxScale, etc.). │ │ │ │ │ + */ │ │ │ │ │ + resolutions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxExtent │ │ │ │ │ + * {<OpenLayers.Bounds>|Array} If provided as an array, the array │ │ │ │ │ + * should consist of four values (left, bottom, right, top). │ │ │ │ │ + * The maximum extent for the layer. Defaults to null. │ │ │ │ │ + * │ │ │ │ │ + * The center of these bounds will not stray outside │ │ │ │ │ + * of the viewport extent during panning. In addition, if │ │ │ │ │ + * <displayOutsideMaxExtent> is set to false, data will not be │ │ │ │ │ + * requested that falls completely outside of these bounds. │ │ │ │ │ + */ │ │ │ │ │ + maxExtent: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: minExtent │ │ │ │ │ + * {<OpenLayers.Bounds>|Array} If provided as an array, the array │ │ │ │ │ + * should consist of four values (left, bottom, right, top). │ │ │ │ │ + * The minimum extent for the layer. Defaults to null. │ │ │ │ │ + */ │ │ │ │ │ + minExtent: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxResolution │ │ │ │ │ + * {Float} Default max is 360 deg / 256 px, which corresponds to │ │ │ │ │ + * zoom level 0 on gmaps. Specify a different value in the layer │ │ │ │ │ + * options if you are not using the default <OpenLayers.Map.tileSize> │ │ │ │ │ + * and displaying the whole world. │ │ │ │ │ + */ │ │ │ │ │ + maxResolution: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: minResolution │ │ │ │ │ + * {Float} │ │ │ │ │ + */ │ │ │ │ │ + minResolution: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: numZoomLevels │ │ │ │ │ + * {Integer} │ │ │ │ │ + */ │ │ │ │ │ + numZoomLevels: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: minScale │ │ │ │ │ + * {Float} │ │ │ │ │ + */ │ │ │ │ │ + minScale: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxScale │ │ │ │ │ + * {Float} │ │ │ │ │ + */ │ │ │ │ │ + maxScale: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: displayOutsideMaxExtent │ │ │ │ │ + * {Boolean} Request map tiles that are completely outside of the max │ │ │ │ │ + * extent for this layer. Defaults to false. │ │ │ │ │ + */ │ │ │ │ │ + displayOutsideMaxExtent: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: wrapDateLine │ │ │ │ │ + * {Boolean} Wraps the world at the international dateline, so the map can │ │ │ │ │ + * be panned infinitely in longitudinal direction. Only use this on the │ │ │ │ │ + * base layer, and only if the layer's maxExtent equals the world bounds. │ │ │ │ │ + * #487 for more info. │ │ │ │ │ + */ │ │ │ │ │ + wrapDateLine: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: metadata │ │ │ │ │ + * {Object} This object can be used to store additional information on a │ │ │ │ │ + * layer object. │ │ │ │ │ + */ │ │ │ │ │ + metadata: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be used │ │ │ │ │ - * to extend the control. │ │ │ │ │ + * name - {String} The layer name │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.controls = []; │ │ │ │ │ - this.activeState = {}; │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + │ │ │ │ │ + this.metadata = {}; │ │ │ │ │ + │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + // make sure we respect alwaysInRange if set on the prototype │ │ │ │ │ + if (this.alwaysInRange != null) { │ │ │ │ │ + options.alwaysInRange = this.alwaysInRange; │ │ │ │ │ + } │ │ │ │ │ + this.addOptions(options); │ │ │ │ │ + │ │ │ │ │ + this.name = name; │ │ │ │ │ + │ │ │ │ │ + if (this.id == null) { │ │ │ │ │ + │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ + │ │ │ │ │ + this.div = OpenLayers.Util.createDiv(this.id); │ │ │ │ │ + this.div.style.width = "100%"; │ │ │ │ │ + this.div.style.height = "100%"; │ │ │ │ │ + this.div.dir = "ltr"; │ │ │ │ │ + │ │ │ │ │ + this.events = new OpenLayers.Events(this, this.div); │ │ │ │ │ + if (this.eventListeners instanceof Object) { │ │ │ │ │ + this.events.on(this.eventListeners); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Destroy is a destructor: this is to alleviate cyclic references which │ │ │ │ │ + * the Javascript garbage cleaner can not take care of on its own. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * setNewBaseLayer - {Boolean} Set a new base layer when this layer has │ │ │ │ │ + * been destroyed. Default is true. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ + destroy: function(setNewBaseLayer) { │ │ │ │ │ + if (setNewBaseLayer == null) { │ │ │ │ │ + setNewBaseLayer = true; │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - for (var ctl, i = this.controls.length - 1; i >= 0; i--) { │ │ │ │ │ - ctl = this.controls[i]; │ │ │ │ │ - if (ctl.events) { │ │ │ │ │ - ctl.events.un({ │ │ │ │ │ - activate: this.iconOn, │ │ │ │ │ - deactivate: this.iconOff │ │ │ │ │ - }); │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.removeLayer(this, setNewBaseLayer); │ │ │ │ │ + } │ │ │ │ │ + this.projection = null; │ │ │ │ │ + this.map = null; │ │ │ │ │ + this.name = null; │ │ │ │ │ + this.div = null; │ │ │ │ │ + this.options = null; │ │ │ │ │ + │ │ │ │ │ + if (this.events) { │ │ │ │ │ + if (this.eventListeners) { │ │ │ │ │ + this.events.un(this.eventListeners); │ │ │ │ │ } │ │ │ │ │ - ctl.panel_div = null; │ │ │ │ │ + this.events.destroy(); │ │ │ │ │ } │ │ │ │ │ - this.activeState = null; │ │ │ │ │ + this.eventListeners = null; │ │ │ │ │ + this.events = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ + * Method: clone │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {<OpenLayers.Layer>} The layer to be cloned │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer>} An exact clone of this <OpenLayers.Layer> │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - control = this.controls[i]; │ │ │ │ │ - if (control === this.defaultControl || │ │ │ │ │ - (this.saveState && this.activeState[control.id])) { │ │ │ │ │ - control.activate(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.saveState === true) { │ │ │ │ │ - this.defaultControl = null; │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer(this.name, this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // catch any randomly tagged-on properties │ │ │ │ │ + OpenLayers.Util.applyDefaults(obj, this); │ │ │ │ │ + │ │ │ │ │ + // a cloned layer should never have its map property set │ │ │ │ │ + // because it has not been added to a map yet. │ │ │ │ │ + obj.map = null; │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getOptions │ │ │ │ │ + * Extracts an object from the layer with the properties that were set as │ │ │ │ │ + * options, but updates them with the values currently set on the │ │ │ │ │ + * instance. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} the <options> of the layer, representing the current state. │ │ │ │ │ + */ │ │ │ │ │ + getOptions: function() { │ │ │ │ │ + var options = {}; │ │ │ │ │ + for (var o in this.options) { │ │ │ │ │ + options[o] = this[o]; │ │ │ │ │ + } │ │ │ │ │ + return options; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setName │ │ │ │ │ + * Sets the new layer name for this layer. Can trigger a changelayer event │ │ │ │ │ + * on the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newName - {String} The new name. │ │ │ │ │ + */ │ │ │ │ │ + setName: function(newName) { │ │ │ │ │ + if (newName != this.name) { │ │ │ │ │ + this.name = newName; │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "name" │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ + * APIMethod: addOptions │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newOptions - {Object} │ │ │ │ │ + * reinitialize - {Boolean} If set to true, and if resolution options of the │ │ │ │ │ + * current baseLayer were changed, the map will be recentered to make │ │ │ │ │ + * sure that it is displayed with a valid resolution, and a │ │ │ │ │ + * changebaselayer event will be triggered. │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - control = this.controls[i]; │ │ │ │ │ - this.activeState[control.id] = control.deactivate(); │ │ │ │ │ + addOptions: function(newOptions, reinitialize) { │ │ │ │ │ + │ │ │ │ │ + if (this.options == null) { │ │ │ │ │ + this.options = {}; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (newOptions) { │ │ │ │ │ + // make sure this.projection references a projection object │ │ │ │ │ + if (typeof newOptions.projection == "string") { │ │ │ │ │ + newOptions.projection = new OpenLayers.Projection(newOptions.projection); │ │ │ │ │ + } │ │ │ │ │ + if (newOptions.projection) { │ │ │ │ │ + // get maxResolution, units and maxExtent from projection defaults if │ │ │ │ │ + // they are not defined already │ │ │ │ │ + OpenLayers.Util.applyDefaults(newOptions, │ │ │ │ │ + OpenLayers.Projection.defaults[newOptions.projection.getCode()]); │ │ │ │ │ + } │ │ │ │ │ + // allow array for extents │ │ │ │ │ + if (newOptions.maxExtent && !(newOptions.maxExtent instanceof OpenLayers.Bounds)) { │ │ │ │ │ + newOptions.maxExtent = new OpenLayers.Bounds(newOptions.maxExtent); │ │ │ │ │ + } │ │ │ │ │ + if (newOptions.minExtent && !(newOptions.minExtent instanceof OpenLayers.Bounds)) { │ │ │ │ │ + newOptions.minExtent = new OpenLayers.Bounds(newOptions.minExtent); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // update our copy for clone │ │ │ │ │ + OpenLayers.Util.extend(this.options, newOptions); │ │ │ │ │ + │ │ │ │ │ + // add new options to this │ │ │ │ │ + OpenLayers.Util.extend(this, newOptions); │ │ │ │ │ + │ │ │ │ │ + // get the units from the projection, if we have a projection │ │ │ │ │ + // and it it has units │ │ │ │ │ + if (this.projection && this.projection.getUnits()) { │ │ │ │ │ + this.units = this.projection.getUnits(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // re-initialize resolutions if necessary, i.e. if any of the │ │ │ │ │ + // properties of the "properties" array defined below is set │ │ │ │ │ + // in the new options │ │ │ │ │ + if (this.map) { │ │ │ │ │ + // store current resolution so we can try to restore it later │ │ │ │ │ + var resolution = this.map.getResolution(); │ │ │ │ │ + var properties = this.RESOLUTION_PROPERTIES.concat( │ │ │ │ │ + ["projection", "units", "minExtent", "maxExtent"] │ │ │ │ │ + ); │ │ │ │ │ + for (var o in newOptions) { │ │ │ │ │ + if (newOptions.hasOwnProperty(o) && │ │ │ │ │ + OpenLayers.Util.indexOf(properties, o) >= 0) { │ │ │ │ │ + │ │ │ │ │ + this.initResolutions(); │ │ │ │ │ + if (reinitialize && this.map.baseLayer === this) { │ │ │ │ │ + // update map position, and restore previous resolution │ │ │ │ │ + this.map.setCenter(this.map.getCenter(), │ │ │ │ │ + this.map.getZoomForResolution(resolution), │ │ │ │ │ + false, true │ │ │ │ │ + ); │ │ │ │ │ + // trigger a changebaselayer event to make sure that │ │ │ │ │ + // all controls (especially │ │ │ │ │ + // OpenLayers.Control.PanZoomBar) get notified of the │ │ │ │ │ + // new options │ │ │ │ │ + this.map.events.triggerEvent("changebaselayer", { │ │ │ │ │ + layer: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ + * APIMethod: onMapResize │ │ │ │ │ + * This function can be implemented by subclasses │ │ │ │ │ + */ │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + //this function can be implemented by subclasses │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: redraw │ │ │ │ │ + * Redraws the layer. Returns true if the layer was redrawn, false if not. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * {Boolean} The layer was redrawn. │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (this.outsideViewport) { │ │ │ │ │ - this.events.attachToElement(this.div); │ │ │ │ │ - this.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ - } else { │ │ │ │ │ - this.map.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ + redraw: function() { │ │ │ │ │ + var redrawn = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + │ │ │ │ │ + // min/max Range may have changed │ │ │ │ │ + this.inRange = this.calculateInRange(); │ │ │ │ │ + │ │ │ │ │ + // map's center might not yet be set │ │ │ │ │ + var extent = this.getExtent(); │ │ │ │ │ + │ │ │ │ │ + if (extent && this.inRange && this.visibility) { │ │ │ │ │ + var zoomChanged = true; │ │ │ │ │ + this.moveTo(extent, zoomChanged, false); │ │ │ │ │ + this.events.triggerEvent("moveend", { │ │ │ │ │ + "zoomChanged": zoomChanged │ │ │ │ │ + }); │ │ │ │ │ + redrawn = true; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.addControlsToMap(this.controls); │ │ │ │ │ - return this.div; │ │ │ │ │ + return redrawn; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: redraw │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * zoomChanged - {Boolean} Tells when zoom has changed, as layers have to │ │ │ │ │ + * do some init work in that case. │ │ │ │ │ + * dragging - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - redraw: function() { │ │ │ │ │ - for (var l = this.div.childNodes.length, i = l - 1; i >= 0; i--) { │ │ │ │ │ - this.div.removeChild(this.div.childNodes[i]); │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + var display = this.visibility; │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + display = display && this.inRange; │ │ │ │ │ } │ │ │ │ │ - this.div.innerHTML = ""; │ │ │ │ │ - if (this.active) { │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - this.div.appendChild(this.controls[i].panel_div); │ │ │ │ │ + this.display(display); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveByPx │ │ │ │ │ + * Move the layer based on pixel vector. To be implemented by subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * dx - {Number} The x coord of the displacement vector. │ │ │ │ │ + * dy - {Number} The y coord of the displacement vector. │ │ │ │ │ + */ │ │ │ │ │ + moveByPx: function(dx, dy) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the layer. This is done through an accessor │ │ │ │ │ + * so that subclasses can override this and take special action once │ │ │ │ │ + * they have their map variable set. │ │ │ │ │ + * │ │ │ │ │ + * Here we take care to bring over any of the necessary default │ │ │ │ │ + * properties from the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + if (this.map == null) { │ │ │ │ │ + │ │ │ │ │ + this.map = map; │ │ │ │ │ + │ │ │ │ │ + // grab some essential layer data from the map if it hasn't already │ │ │ │ │ + // been set │ │ │ │ │ + this.maxExtent = this.maxExtent || this.map.maxExtent; │ │ │ │ │ + this.minExtent = this.minExtent || this.map.minExtent; │ │ │ │ │ + │ │ │ │ │ + this.projection = this.projection || this.map.projection; │ │ │ │ │ + if (typeof this.projection == "string") { │ │ │ │ │ + this.projection = new OpenLayers.Projection(this.projection); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Check the projection to see if we can get units -- if not, refer │ │ │ │ │ + // to properties. │ │ │ │ │ + this.units = this.projection.getUnits() || │ │ │ │ │ + this.units || this.map.units; │ │ │ │ │ + │ │ │ │ │ + this.initResolutions(); │ │ │ │ │ + │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + this.inRange = this.calculateInRange(); │ │ │ │ │ + var show = ((this.visibility) && (this.inRange)); │ │ │ │ │ + this.div.style.display = show ? "" : "none"; │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + // deal with gutters │ │ │ │ │ + this.setTileSize(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: activateControl │ │ │ │ │ - * This method is called when the user click on the icon representing a │ │ │ │ │ - * control in the panel. │ │ │ │ │ + * Method: afterAdd │ │ │ │ │ + * Called at the end of the map.addLayer sequence. At this point, the map │ │ │ │ │ + * will have a base layer. To be overridden by subclasses. │ │ │ │ │ + */ │ │ │ │ │ + afterAdd: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: removeMap │ │ │ │ │ + * Just as setMap() allows each layer the possibility to take a │ │ │ │ │ + * personalized action on being added to the map, removeMap() allows │ │ │ │ │ + * each layer to take a personalized action on being removed from it. │ │ │ │ │ + * For now, this will be mostly unused, except for the EventPane layer, │ │ │ │ │ + * which needs this hook so that it can remove the special invisible │ │ │ │ │ + * pane. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + //to be overridden by subclasses │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getImageSize │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} optional tile bounds, can be used │ │ │ │ │ + * by subclasses that have to deal with different tile sizes at the │ │ │ │ │ + * layer extent edges (e.g. Zoomify) │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Size>} The size that the image should be, taking into │ │ │ │ │ + * account gutters. │ │ │ │ │ */ │ │ │ │ │ - activateControl: function(control) { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - if (control.type == OpenLayers.Control.TYPE_BUTTON) { │ │ │ │ │ - control.trigger(); │ │ │ │ │ - return; │ │ │ │ │ + getImageSize: function(bounds) { │ │ │ │ │ + return (this.imageSize || this.tileSize); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setTileSize │ │ │ │ │ + * Set the tile size based on the map size. This also sets layer.imageSize │ │ │ │ │ + * or use by Tile.Image. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * size - {<OpenLayers.Size>} │ │ │ │ │ + */ │ │ │ │ │ + setTileSize: function(size) { │ │ │ │ │ + var tileSize = (size) ? size : │ │ │ │ │ + ((this.tileSize) ? this.tileSize : │ │ │ │ │ + this.map.getTileSize()); │ │ │ │ │ + this.tileSize = tileSize; │ │ │ │ │ + if (this.gutter) { │ │ │ │ │ + // layers with gutters need non-null tile sizes │ │ │ │ │ + //if(tileSize == null) { │ │ │ │ │ + // OpenLayers.console.error("Error in layer.setMap() for " + │ │ │ │ │ + // this.name + ": layers with " + │ │ │ │ │ + // "gutters need non-null tile sizes"); │ │ │ │ │ + //} │ │ │ │ │ + this.imageSize = new OpenLayers.Size(tileSize.w + (2 * this.gutter), │ │ │ │ │ + tileSize.h + (2 * this.gutter)); │ │ │ │ │ } │ │ │ │ │ - if (control.type == OpenLayers.Control.TYPE_TOGGLE) { │ │ │ │ │ - if (control.active) { │ │ │ │ │ - control.deactivate(); │ │ │ │ │ - } else { │ │ │ │ │ - control.activate(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getVisibility │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The layer should be displayed (if in range). │ │ │ │ │ + */ │ │ │ │ │ + getVisibility: function() { │ │ │ │ │ + return this.visibility; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setVisibility │ │ │ │ │ + * Set the visibility flag for the layer and hide/show & redraw │ │ │ │ │ + * accordingly. Fire event unless otherwise specified │ │ │ │ │ + * │ │ │ │ │ + * Note that visibility is no longer simply whether or not the layer's │ │ │ │ │ + * style.display is set to "block". Now we store a 'visibility' state │ │ │ │ │ + * property on the layer class, this allows us to remember whether or │ │ │ │ │ + * not we *desire* for a layer to be visible. In the case where the │ │ │ │ │ + * map's resolution is out of the layer's range, this desire may be │ │ │ │ │ + * subverted. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * visibility - {Boolean} Whether or not to display the layer (if in range) │ │ │ │ │ + */ │ │ │ │ │ + setVisibility: function(visibility) { │ │ │ │ │ + if (visibility != this.visibility) { │ │ │ │ │ + this.visibility = visibility; │ │ │ │ │ + this.display(visibility); │ │ │ │ │ + this.redraw(); │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "visibility" │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - return; │ │ │ │ │ + this.events.triggerEvent("visibilitychanged"); │ │ │ │ │ } │ │ │ │ │ - if (this.allowDepress && control.active) { │ │ │ │ │ - control.deactivate(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: display │ │ │ │ │ + * Hide or show the Layer. This is designed to be used internally, and │ │ │ │ │ + * is not generally the way to enable or disable the layer. For that, │ │ │ │ │ + * use the setVisibility function instead.. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * display - {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + display: function(display) { │ │ │ │ │ + if (display != (this.div.style.display != "none")) { │ │ │ │ │ + this.div.style.display = (display && this.calculateInRange()) ? "block" : "none"; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: calculateInRange │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The layer is displayable at the current map's current │ │ │ │ │ + * resolution. Note that if 'alwaysInRange' is true for the layer, │ │ │ │ │ + * this function will always return true. │ │ │ │ │ + */ │ │ │ │ │ + calculateInRange: function() { │ │ │ │ │ + var inRange = false; │ │ │ │ │ + │ │ │ │ │ + if (this.alwaysInRange) { │ │ │ │ │ + inRange = true; │ │ │ │ │ } else { │ │ │ │ │ - var c; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - c = this.controls[i]; │ │ │ │ │ - if (c != control && │ │ │ │ │ - (c.type === OpenLayers.Control.TYPE_TOOL || c.type == null)) { │ │ │ │ │ - c.deactivate(); │ │ │ │ │ - } │ │ │ │ │ + if (this.map) { │ │ │ │ │ + var resolution = this.map.getResolution(); │ │ │ │ │ + inRange = ((resolution >= this.minResolution) && │ │ │ │ │ + (resolution <= this.maxResolution)); │ │ │ │ │ } │ │ │ │ │ - control.activate(); │ │ │ │ │ } │ │ │ │ │ + return inRange; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: addControls │ │ │ │ │ - * To build a toolbar, you add a set of controls to it. addControls │ │ │ │ │ - * lets you add a single control or a list of controls to the │ │ │ │ │ - * Control Panel. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setIsBaseLayer │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * controls - {<OpenLayers.Control>} Controls to add in the panel. │ │ │ │ │ + * isBaseLayer - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - addControls: function(controls) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(controls))) { │ │ │ │ │ - controls = [controls]; │ │ │ │ │ + setIsBaseLayer: function(isBaseLayer) { │ │ │ │ │ + if (isBaseLayer != this.isBaseLayer) { │ │ │ │ │ + this.isBaseLayer = isBaseLayer; │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changebaselayer", { │ │ │ │ │ + layer: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.controls = this.controls.concat(controls); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ - var control = controls[i], │ │ │ │ │ - element = this.createControlMarkup(control); │ │ │ │ │ - OpenLayers.Element.addClass(element, │ │ │ │ │ - control.displayClass + "ItemInactive"); │ │ │ │ │ - OpenLayers.Element.addClass(element, "olButton"); │ │ │ │ │ - if (control.title != "" && !element.title) { │ │ │ │ │ - element.title = control.title; │ │ │ │ │ + /********************************************************/ │ │ │ │ │ + /* */ │ │ │ │ │ + /* Baselayer Functions */ │ │ │ │ │ + /* */ │ │ │ │ │ + /********************************************************/ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: initResolutions │ │ │ │ │ + * This method's responsibility is to set up the 'resolutions' array │ │ │ │ │ + * for the layer -- this array is what the layer will use to interface │ │ │ │ │ + * between the zoom levels of the map and the resolution display │ │ │ │ │ + * of the layer. │ │ │ │ │ + * │ │ │ │ │ + * The user has several options that determine how the array is set up. │ │ │ │ │ + * │ │ │ │ │ + * For a detailed explanation, see the following wiki from the │ │ │ │ │ + * openlayers.org homepage: │ │ │ │ │ + * http://trac.openlayers.org/wiki/SettingZoomLevels │ │ │ │ │ + */ │ │ │ │ │ + initResolutions: function() { │ │ │ │ │ + │ │ │ │ │ + // ok we want resolutions, here's our strategy: │ │ │ │ │ + // │ │ │ │ │ + // 1. if resolutions are defined in the layer config, use them │ │ │ │ │ + // 2. else, if scales are defined in the layer config then derive │ │ │ │ │ + // resolutions from these scales │ │ │ │ │ + // 3. else, attempt to calculate resolutions from maxResolution, │ │ │ │ │ + // minResolution, numZoomLevels, maxZoomLevel set in the │ │ │ │ │ + // layer config │ │ │ │ │ + // 4. if we still don't have resolutions, and if resolutions │ │ │ │ │ + // are defined in the same, use them │ │ │ │ │ + // 5. else, if scales are defined in the map then derive │ │ │ │ │ + // resolutions from these scales │ │ │ │ │ + // 6. else, attempt to calculate resolutions from maxResolution, │ │ │ │ │ + // minResolution, numZoomLevels, maxZoomLevel set in the │ │ │ │ │ + // map │ │ │ │ │ + // 7. hope for the best! │ │ │ │ │ + │ │ │ │ │ + var i, len, p; │ │ │ │ │ + var props = {}, │ │ │ │ │ + alwaysInRange = true; │ │ │ │ │ + │ │ │ │ │ + // get resolution data from layer config │ │ │ │ │ + // (we also set alwaysInRange in the layer as appropriate) │ │ │ │ │ + for (i = 0, len = this.RESOLUTION_PROPERTIES.length; i < len; i++) { │ │ │ │ │ + p = this.RESOLUTION_PROPERTIES[i]; │ │ │ │ │ + props[p] = this.options[p]; │ │ │ │ │ + if (alwaysInRange && this.options[p]) { │ │ │ │ │ + alwaysInRange = false; │ │ │ │ │ } │ │ │ │ │ - control.panel_div = element; │ │ │ │ │ + } │ │ │ │ │ + if (this.options.alwaysInRange == null) { │ │ │ │ │ + this.alwaysInRange = alwaysInRange; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - if (this.map) { // map.addControl() has already been called on the panel │ │ │ │ │ - this.addControlsToMap(controls); │ │ │ │ │ - this.redraw(); │ │ │ │ │ + // if we don't have resolutions then attempt to derive them from scales │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + props.resolutions = this.resolutionsFromScales(props.scales); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // if we still don't have resolutions then attempt to calculate them │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + props.resolutions = this.calculateResolutions(props); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // if we couldn't calculate resolutions then we look at we have │ │ │ │ │ + // in the map │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + for (i = 0, len = this.RESOLUTION_PROPERTIES.length; i < len; i++) { │ │ │ │ │ + p = this.RESOLUTION_PROPERTIES[i]; │ │ │ │ │ + props[p] = this.options[p] != null ? │ │ │ │ │ + this.options[p] : this.map[p]; │ │ │ │ │ + } │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + props.resolutions = this.resolutionsFromScales(props.scales); │ │ │ │ │ + } │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + props.resolutions = this.calculateResolutions(props); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // ok, we new need to set properties in the instance │ │ │ │ │ + │ │ │ │ │ + // get maxResolution from the config if it's defined there │ │ │ │ │ + var maxResolution; │ │ │ │ │ + if (this.options.maxResolution && │ │ │ │ │ + this.options.maxResolution !== "auto") { │ │ │ │ │ + maxResolution = this.options.maxResolution; │ │ │ │ │ + } │ │ │ │ │ + if (this.options.minScale) { │ │ │ │ │ + maxResolution = OpenLayers.Util.getResolutionFromScale( │ │ │ │ │ + this.options.minScale, this.units); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // get minResolution from the config if it's defined there │ │ │ │ │ + var minResolution; │ │ │ │ │ + if (this.options.minResolution && │ │ │ │ │ + this.options.minResolution !== "auto") { │ │ │ │ │ + minResolution = this.options.minResolution; │ │ │ │ │ + } │ │ │ │ │ + if (this.options.maxScale) { │ │ │ │ │ + minResolution = OpenLayers.Util.getResolutionFromScale( │ │ │ │ │ + this.options.maxScale, this.units); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (props.resolutions) { │ │ │ │ │ + │ │ │ │ │ + //sort resolutions array descendingly │ │ │ │ │ + props.resolutions.sort(function(a, b) { │ │ │ │ │ + return (b - a); │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + // if we still don't have a maxResolution get it from the │ │ │ │ │ + // resolutions array │ │ │ │ │ + if (!maxResolution) { │ │ │ │ │ + maxResolution = props.resolutions[0]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // if we still don't have a minResolution get it from the │ │ │ │ │ + // resolutions array │ │ │ │ │ + if (!minResolution) { │ │ │ │ │ + var lastIdx = props.resolutions.length - 1; │ │ │ │ │ + minResolution = props.resolutions[lastIdx]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.resolutions = props.resolutions; │ │ │ │ │ + if (this.resolutions) { │ │ │ │ │ + len = this.resolutions.length; │ │ │ │ │ + this.scales = new Array(len); │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + this.scales[i] = OpenLayers.Util.getScaleFromResolution( │ │ │ │ │ + this.resolutions[i], this.units); │ │ │ │ │ + } │ │ │ │ │ + this.numZoomLevels = len; │ │ │ │ │ + } │ │ │ │ │ + this.minResolution = minResolution; │ │ │ │ │ + if (minResolution) { │ │ │ │ │ + this.maxScale = OpenLayers.Util.getScaleFromResolution( │ │ │ │ │ + minResolution, this.units); │ │ │ │ │ + } │ │ │ │ │ + this.maxResolution = maxResolution; │ │ │ │ │ + if (maxResolution) { │ │ │ │ │ + this.minScale = OpenLayers.Util.getScaleFromResolution( │ │ │ │ │ + maxResolution, this.units); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: createControlMarkup │ │ │ │ │ - * This function just creates a div for the control. If specific HTML │ │ │ │ │ - * markup is needed this function can be overridden in specific classes, │ │ │ │ │ - * or at panel instantiation time: │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var panel = new OpenLayers.Control.Panel({ │ │ │ │ │ - * defaultControl: control, │ │ │ │ │ - * // ovverride createControlMarkup to create actual buttons │ │ │ │ │ - * // including texts wrapped into span elements. │ │ │ │ │ - * createControlMarkup: function(control) { │ │ │ │ │ - * var button = document.createElement('button'), │ │ │ │ │ - * span = document.createElement('span'); │ │ │ │ │ - * if (control.text) { │ │ │ │ │ - * span.innerHTML = control.text; │ │ │ │ │ - * } │ │ │ │ │ - * return button; │ │ │ │ │ - * } │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ + * Method: resolutionsFromScales │ │ │ │ │ + * Derive resolutions from scales. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control to create the HTML │ │ │ │ │ - * markup for. │ │ │ │ │ + * scales - {Array(Number)} Scales │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The markup. │ │ │ │ │ + * Returns │ │ │ │ │ + * {Array(Number)} Resolutions │ │ │ │ │ */ │ │ │ │ │ - createControlMarkup: function(control) { │ │ │ │ │ - return document.createElement("div"); │ │ │ │ │ + resolutionsFromScales: function(scales) { │ │ │ │ │ + if (scales == null) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var resolutions, i, len; │ │ │ │ │ + len = scales.length; │ │ │ │ │ + resolutions = new Array(len); │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + resolutions[i] = OpenLayers.Util.getResolutionFromScale( │ │ │ │ │ + scales[i], this.units); │ │ │ │ │ + } │ │ │ │ │ + return resolutions; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: addControlsToMap │ │ │ │ │ - * Only for internal use in draw() and addControls() methods. │ │ │ │ │ + * Method: calculateResolutions │ │ │ │ │ + * Calculate resolutions based on the provided properties. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * controls - {Array(<OpenLayers.Control>)} Controls to add into map. │ │ │ │ │ + * props - {Object} Properties │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array({Number})} Array of resolutions. │ │ │ │ │ */ │ │ │ │ │ - addControlsToMap: function(controls) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ - control = controls[i]; │ │ │ │ │ - if (control.autoActivate === true) { │ │ │ │ │ - control.autoActivate = false; │ │ │ │ │ - this.map.addControl(control); │ │ │ │ │ - control.autoActivate = true; │ │ │ │ │ - } else { │ │ │ │ │ - this.map.addControl(control); │ │ │ │ │ - control.deactivate(); │ │ │ │ │ + calculateResolutions: function(props) { │ │ │ │ │ + │ │ │ │ │ + var viewSize, wRes, hRes; │ │ │ │ │ + │ │ │ │ │ + // determine maxResolution │ │ │ │ │ + var maxResolution = props.maxResolution; │ │ │ │ │ + if (props.minScale != null) { │ │ │ │ │ + maxResolution = │ │ │ │ │ + OpenLayers.Util.getResolutionFromScale(props.minScale, │ │ │ │ │ + this.units); │ │ │ │ │ + } else if (maxResolution == "auto" && this.maxExtent != null) { │ │ │ │ │ + viewSize = this.map.getSize(); │ │ │ │ │ + wRes = this.maxExtent.getWidth() / viewSize.w; │ │ │ │ │ + hRes = this.maxExtent.getHeight() / viewSize.h; │ │ │ │ │ + maxResolution = Math.max(wRes, hRes); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // determine minResolution │ │ │ │ │ + var minResolution = props.minResolution; │ │ │ │ │ + if (props.maxScale != null) { │ │ │ │ │ + minResolution = │ │ │ │ │ + OpenLayers.Util.getResolutionFromScale(props.maxScale, │ │ │ │ │ + this.units); │ │ │ │ │ + } else if (props.minResolution == "auto" && this.minExtent != null) { │ │ │ │ │ + viewSize = this.map.getSize(); │ │ │ │ │ + wRes = this.minExtent.getWidth() / viewSize.w; │ │ │ │ │ + hRes = this.minExtent.getHeight() / viewSize.h; │ │ │ │ │ + minResolution = Math.max(wRes, hRes); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (typeof maxResolution !== "number" && │ │ │ │ │ + typeof minResolution !== "number" && │ │ │ │ │ + this.maxExtent != null) { │ │ │ │ │ + // maxResolution for default grid sets assumes that at zoom │ │ │ │ │ + // level zero, the whole world fits on one tile. │ │ │ │ │ + var tileSize = this.map.getTileSize(); │ │ │ │ │ + maxResolution = Math.max( │ │ │ │ │ + this.maxExtent.getWidth() / tileSize.w, │ │ │ │ │ + this.maxExtent.getHeight() / tileSize.h │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // determine numZoomLevels │ │ │ │ │ + var maxZoomLevel = props.maxZoomLevel; │ │ │ │ │ + var numZoomLevels = props.numZoomLevels; │ │ │ │ │ + if (typeof minResolution === "number" && │ │ │ │ │ + typeof maxResolution === "number" && numZoomLevels === undefined) { │ │ │ │ │ + var ratio = maxResolution / minResolution; │ │ │ │ │ + numZoomLevels = Math.floor(Math.log(ratio) / Math.log(2)) + 1; │ │ │ │ │ + } else if (numZoomLevels === undefined && maxZoomLevel != null) { │ │ │ │ │ + numZoomLevels = maxZoomLevel + 1; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // are we able to calculate resolutions? │ │ │ │ │ + if (typeof numZoomLevels !== "number" || numZoomLevels <= 0 || │ │ │ │ │ + (typeof maxResolution !== "number" && │ │ │ │ │ + typeof minResolution !== "number")) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // now we have numZoomLevels and at least one of maxResolution │ │ │ │ │ + // or minResolution, we can populate the resolutions array │ │ │ │ │ + │ │ │ │ │ + var resolutions = new Array(numZoomLevels); │ │ │ │ │ + var base = 2; │ │ │ │ │ + if (typeof minResolution == "number" && │ │ │ │ │ + typeof maxResolution == "number") { │ │ │ │ │ + // if maxResolution and minResolution are set, we calculate │ │ │ │ │ + // the base for exponential scaling that starts at │ │ │ │ │ + // maxResolution and ends at minResolution in numZoomLevels │ │ │ │ │ + // steps. │ │ │ │ │ + base = Math.pow( │ │ │ │ │ + (maxResolution / minResolution), │ │ │ │ │ + (1 / (numZoomLevels - 1)) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var i; │ │ │ │ │ + if (typeof maxResolution === "number") { │ │ │ │ │ + for (i = 0; i < numZoomLevels; i++) { │ │ │ │ │ + resolutions[i] = maxResolution / Math.pow(base, i); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + for (i = 0; i < numZoomLevels; i++) { │ │ │ │ │ + resolutions[numZoomLevels - 1 - i] = │ │ │ │ │ + minResolution * Math.pow(base, i); │ │ │ │ │ } │ │ │ │ │ - control.events.on({ │ │ │ │ │ - activate: this.iconOn, │ │ │ │ │ - deactivate: this.iconOff │ │ │ │ │ - }); │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + return resolutions; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: iconOn │ │ │ │ │ - * Internal use, for use only with "controls[i].events.on/un". │ │ │ │ │ + * APIMethod: getResolution │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The currently selected resolution of the map, taken from the │ │ │ │ │ + * resolutions array, indexed by current zoom level. │ │ │ │ │ */ │ │ │ │ │ - iconOn: function() { │ │ │ │ │ - var d = this.panel_div; // "this" refers to a control on panel! │ │ │ │ │ - var re = new RegExp("\\b(" + this.displayClass + "Item)Inactive\\b"); │ │ │ │ │ - d.className = d.className.replace(re, "$1Active"); │ │ │ │ │ + getResolution: function() { │ │ │ │ │ + var zoom = this.map.getZoom(); │ │ │ │ │ + return this.getResolutionForZoom(zoom); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: iconOff │ │ │ │ │ - * Internal use, for use only with "controls[i].events.on/un". │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getExtent │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} A Bounds object which represents the lon/lat │ │ │ │ │ + * bounds of the current viewPort. │ │ │ │ │ */ │ │ │ │ │ - iconOff: function() { │ │ │ │ │ - var d = this.panel_div; // "this" refers to a control on panel! │ │ │ │ │ - var re = new RegExp("\\b(" + this.displayClass + "Item)Active\\b"); │ │ │ │ │ - d.className = d.className.replace(re, "$1Inactive"); │ │ │ │ │ + getExtent: function() { │ │ │ │ │ + // just use stock map calculateBounds function -- passing no arguments │ │ │ │ │ + // means it will user map's current center & resolution │ │ │ │ │ + // │ │ │ │ │ + return this.map.calculateBounds(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onButtonClick │ │ │ │ │ + * APIMethod: getZoomForExtent │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} │ │ │ │ │ + * closest - {Boolean} Find the zoom level that most closely fits the │ │ │ │ │ + * specified bounds. Note that this may result in a zoom that does │ │ │ │ │ + * not exactly contain the entire extent. │ │ │ │ │ + * Default is false. │ │ │ │ │ * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} The index of the zoomLevel (entry in the resolutions array) │ │ │ │ │ + * for the passed-in extent. We do this by calculating the ideal │ │ │ │ │ + * resolution for the given extent (based on the map size) and then │ │ │ │ │ + * calling getZoomForResolution(), passing along the 'closest' │ │ │ │ │ + * parameter. │ │ │ │ │ + */ │ │ │ │ │ + getZoomForExtent: function(extent, closest) { │ │ │ │ │ + var viewSize = this.map.getSize(); │ │ │ │ │ + var idealResolution = Math.max(extent.getWidth() / viewSize.w, │ │ │ │ │ + extent.getHeight() / viewSize.h); │ │ │ │ │ + │ │ │ │ │ + return this.getZoomForResolution(idealResolution, closest); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getDataExtent │ │ │ │ │ + * Calculates the max extent which includes all of the data for the layer. │ │ │ │ │ + * This function is to be implemented by subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} │ │ │ │ │ + */ │ │ │ │ │ + getDataExtent: function() { │ │ │ │ │ + //to be implemented by subclasses │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getResolutionForZoom │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * zoom - {Float} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} A suitable resolution for the specified zoom. │ │ │ │ │ */ │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - var controls = this.controls, │ │ │ │ │ - button = evt.buttonElement; │ │ │ │ │ - for (var i = controls.length - 1; i >= 0; --i) { │ │ │ │ │ - if (controls[i].panel_div === button) { │ │ │ │ │ - this.activateControl(controls[i]); │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ + getResolutionForZoom: function(zoom) { │ │ │ │ │ + zoom = Math.max(0, Math.min(zoom, this.resolutions.length - 1)); │ │ │ │ │ + var resolution; │ │ │ │ │ + if (this.map.fractionalZoom) { │ │ │ │ │ + var low = Math.floor(zoom); │ │ │ │ │ + var high = Math.ceil(zoom); │ │ │ │ │ + resolution = this.resolutions[low] - │ │ │ │ │ + ((zoom - low) * (this.resolutions[low] - this.resolutions[high])); │ │ │ │ │ + } else { │ │ │ │ │ + resolution = this.resolutions[Math.round(zoom)]; │ │ │ │ │ } │ │ │ │ │ + return resolution; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getControlsBy │ │ │ │ │ - * Get a list of controls with properties matching the given criteria. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getZoomForResolution │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * property - {String} A control property to be matched. │ │ │ │ │ - * match - {String | Object} A string to match. Can also be a regular │ │ │ │ │ - * expression literal or object. In addition, it can be any object │ │ │ │ │ - * with a method named test. For reqular expressions or other, if │ │ │ │ │ - * match.test(control[property]) evaluates to true, the control will be │ │ │ │ │ - * included in the array returned. If no controls are found, an empty │ │ │ │ │ - * array is returned. │ │ │ │ │ - * │ │ │ │ │ + * resolution - {Float} │ │ │ │ │ + * closest - {Boolean} Find the zoom level that corresponds to the absolute │ │ │ │ │ + * closest resolution, which may result in a zoom whose corresponding │ │ │ │ │ + * resolution is actually smaller than we would have desired (if this │ │ │ │ │ + * is being called from a getZoomForExtent() call, then this means that │ │ │ │ │ + * the returned zoom index might not actually contain the entire │ │ │ │ │ + * extent specified... but it'll be close). │ │ │ │ │ + * Default is false. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Control>)} A list of controls matching the given criteria. │ │ │ │ │ - * An empty array is returned if no matches are found. │ │ │ │ │ + * {Integer} The index of the zoomLevel (entry in the resolutions array) │ │ │ │ │ + * that corresponds to the best fit resolution given the passed in │ │ │ │ │ + * value and the 'closest' specification. │ │ │ │ │ */ │ │ │ │ │ - getControlsBy: function(property, match) { │ │ │ │ │ - var test = (typeof match.test == "function"); │ │ │ │ │ - var found = OpenLayers.Array.filter(this.controls, function(item) { │ │ │ │ │ - return item[property] == match || (test && match.test(item[property])); │ │ │ │ │ - }); │ │ │ │ │ - return found; │ │ │ │ │ + getZoomForResolution: function(resolution, closest) { │ │ │ │ │ + var zoom, i, len; │ │ │ │ │ + if (this.map.fractionalZoom) { │ │ │ │ │ + var lowZoom = 0; │ │ │ │ │ + var highZoom = this.resolutions.length - 1; │ │ │ │ │ + var highRes = this.resolutions[lowZoom]; │ │ │ │ │ + var lowRes = this.resolutions[highZoom]; │ │ │ │ │ + var res; │ │ │ │ │ + for (i = 0, len = this.resolutions.length; i < len; ++i) { │ │ │ │ │ + res = this.resolutions[i]; │ │ │ │ │ + if (res >= resolution) { │ │ │ │ │ + highRes = res; │ │ │ │ │ + lowZoom = i; │ │ │ │ │ + } │ │ │ │ │ + if (res <= resolution) { │ │ │ │ │ + lowRes = res; │ │ │ │ │ + highZoom = i; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var dRes = highRes - lowRes; │ │ │ │ │ + if (dRes > 0) { │ │ │ │ │ + zoom = lowZoom + ((highRes - resolution) / dRes); │ │ │ │ │ + } else { │ │ │ │ │ + zoom = lowZoom; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + var diff; │ │ │ │ │ + var minDiff = Number.POSITIVE_INFINITY; │ │ │ │ │ + for (i = 0, len = this.resolutions.length; i < len; i++) { │ │ │ │ │ + if (closest) { │ │ │ │ │ + diff = Math.abs(this.resolutions[i] - resolution); │ │ │ │ │ + if (diff > minDiff) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + minDiff = diff; │ │ │ │ │ + } else { │ │ │ │ │ + if (this.resolutions[i] < resolution) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + zoom = Math.max(0, i - 1); │ │ │ │ │ + } │ │ │ │ │ + return zoom; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getControlsByName │ │ │ │ │ - * Get a list of contorls with names matching the given name. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getLonLatFromViewPortPx │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * match - {String | Object} A control name. The name can also be a regular │ │ │ │ │ - * expression literal or object. In addition, it can be any object │ │ │ │ │ - * with a method named test. For reqular expressions or other, if │ │ │ │ │ - * name.test(control.name) evaluates to true, the control will be included │ │ │ │ │ - * in the list of controls returned. If no controls are found, an empty │ │ │ │ │ - * array is returned. │ │ │ │ │ + * viewPortPx - {<OpenLayers.Pixel>|Object} An OpenLayers.Pixel or │ │ │ │ │ + * an object with a 'x' │ │ │ │ │ + * and 'y' properties. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Control>)} A list of controls matching the given name. │ │ │ │ │ - * An empty array is returned if no matches are found. │ │ │ │ │ + * {<OpenLayers.LonLat>} An OpenLayers.LonLat which is the passed-in │ │ │ │ │ + * view port <OpenLayers.Pixel>, translated into lon/lat by the layer. │ │ │ │ │ */ │ │ │ │ │ - getControlsByName: function(match) { │ │ │ │ │ - return this.getControlsBy("name", match); │ │ │ │ │ + getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ + var lonlat = null; │ │ │ │ │ + var map = this.map; │ │ │ │ │ + if (viewPortPx != null && map.minPx) { │ │ │ │ │ + var res = map.getResolution(); │ │ │ │ │ + var maxExtent = map.getMaxExtent({ │ │ │ │ │ + restricted: true │ │ │ │ │ + }); │ │ │ │ │ + var lon = (viewPortPx.x - map.minPx.x) * res + maxExtent.left; │ │ │ │ │ + var lat = (map.minPx.y - viewPortPx.y) * res + maxExtent.top; │ │ │ │ │ + lonlat = new OpenLayers.LonLat(lon, lat); │ │ │ │ │ + │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + lonlat = lonlat.wrapDateLine(this.maxExtent); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return lonlat; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getControlsByClass │ │ │ │ │ - * Get a list of controls of a given type (CLASS_NAME). │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getViewPortPxFromLonLat │ │ │ │ │ + * Returns a pixel location given a map location. This method will return │ │ │ │ │ + * fractional pixel values. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * match - {String | Object} A control class name. The type can also be a │ │ │ │ │ - * regular expression literal or object. In addition, it can be any │ │ │ │ │ - * object with a method named test. For reqular expressions or other, │ │ │ │ │ - * if type.test(control.CLASS_NAME) evaluates to true, the control will │ │ │ │ │ - * be included in the list of controls returned. If no controls are │ │ │ │ │ - * found, an empty array is returned. │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>|Object} An OpenLayers.LonLat or │ │ │ │ │ + * an object with a 'lon' │ │ │ │ │ + * and 'lat' properties. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Control>)} A list of controls matching the given type. │ │ │ │ │ - * An empty array is returned if no matches are found. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Pixel>} An <OpenLayers.Pixel> which is the passed-in │ │ │ │ │ + * lonlat translated into view port pixels. │ │ │ │ │ */ │ │ │ │ │ - getControlsByClass: function(match) { │ │ │ │ │ - return this.getControlsBy("CLASS_NAME", match); │ │ │ │ │ + getViewPortPxFromLonLat: function(lonlat, resolution) { │ │ │ │ │ + var px = null; │ │ │ │ │ + if (lonlat != null) { │ │ │ │ │ + resolution = resolution || this.map.getResolution(); │ │ │ │ │ + var extent = this.map.calculateBounds(null, resolution); │ │ │ │ │ + px = new OpenLayers.Pixel( │ │ │ │ │ + (1 / resolution * (lonlat.lon - extent.left)), │ │ │ │ │ + (1 / resolution * (extent.top - lonlat.lat)) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return px; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Panel" │ │ │ │ │ -}); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setOpacity │ │ │ │ │ + * Sets the opacity for the entire layer (all images) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * opacity - {Float} │ │ │ │ │ + */ │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + if (opacity != this.opacity) { │ │ │ │ │ + this.opacity = opacity; │ │ │ │ │ + var childNodes = this.div.childNodes; │ │ │ │ │ + for (var i = 0, len = childNodes.length; i < len; ++i) { │ │ │ │ │ + var element = childNodes[i].firstChild || childNodes[i]; │ │ │ │ │ + var lastChild = childNodes[i].lastChild; │ │ │ │ │ + //TODO de-uglify this │ │ │ │ │ + if (lastChild && lastChild.nodeName.toLowerCase() === "iframe") { │ │ │ │ │ + element = lastChild.parentNode; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.modifyDOMElement(element, null, null, null, │ │ │ │ │ + null, null, null, opacity); │ │ │ │ │ + } │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "opacity" │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Method: getZIndex │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} the z-index of this layer │ │ │ │ │ + */ │ │ │ │ │ + getZIndex: function() { │ │ │ │ │ + return this.div.style.zIndex; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setZIndex │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * zIndex - {Integer} │ │ │ │ │ + */ │ │ │ │ │ + setZIndex: function(zIndex) { │ │ │ │ │ + this.div.style.zIndex = zIndex; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: adjustBounds │ │ │ │ │ + * This function will take a bounds, and if wrapDateLine option is set │ │ │ │ │ + * on the layer, it will return a bounds which is wrapped around the │ │ │ │ │ + * world. We do not wrap for bounds which *cross* the │ │ │ │ │ + * maxExtent.left/right, only bounds which are entirely to the left │ │ │ │ │ + * or entirely to the right. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + */ │ │ │ │ │ + adjustBounds: function(bounds) { │ │ │ │ │ + │ │ │ │ │ + if (this.gutter) { │ │ │ │ │ + // Adjust the extent of a bounds in map units by the │ │ │ │ │ + // layer's gutter in pixels. │ │ │ │ │ + var mapGutter = this.gutter * this.map.getResolution(); │ │ │ │ │ + bounds = new OpenLayers.Bounds(bounds.left - mapGutter, │ │ │ │ │ + bounds.bottom - mapGutter, │ │ │ │ │ + bounds.right + mapGutter, │ │ │ │ │ + bounds.top + mapGutter); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + // wrap around the date line, within the limits of rounding error │ │ │ │ │ + var wrappingOptions = { │ │ │ │ │ + 'rightTolerance': this.getResolution(), │ │ │ │ │ + 'leftTolerance': this.getResolution() │ │ │ │ │ + }; │ │ │ │ │ + bounds = bounds.wrapDateLine(this.maxExtent, wrappingOptions); │ │ │ │ │ + │ │ │ │ │ + } │ │ │ │ │ + return bounds; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Attribution.js │ │ │ │ │ + OpenLayers/Layer/Vector.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Layer.js │ │ │ │ │ + * @requires OpenLayers/Renderer.js │ │ │ │ │ + * @requires OpenLayers/StyleMap.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Console.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.Attribution │ │ │ │ │ - * The attribution control adds attribution from layers to the map display. │ │ │ │ │ - * It uses 'attribution' property of each layer. │ │ │ │ │ + * Class: OpenLayers.Layer.Vector │ │ │ │ │ + * Instances of OpenLayers.Layer.Vector are used to render vector data from │ │ │ │ │ + * a variety of sources. Create a new vector layer with the │ │ │ │ │ + * <OpenLayers.Layer.Vector> constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ + * - <OpenLayers.Layer> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.Attribution = │ │ │ │ │ - OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: separator │ │ │ │ │ - * {String} String used to separate layers. │ │ │ │ │ - */ │ │ │ │ │ - separator: ", ", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: template │ │ │ │ │ - * {String} Template for the attribution. This has to include the substring │ │ │ │ │ - * "${layers}", which will be replaced by the layer specific │ │ │ │ │ - * attributions, separated by <separator>. The default is "${layers}". │ │ │ │ │ - */ │ │ │ │ │ - template: "${layers}", │ │ │ │ │ +OpenLayers.Layer.Vector = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Attribution │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Options for control. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * layer.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Listeners will be called with a reference to an event object. The │ │ │ │ │ + * properties of this event depends on exactly what happened. │ │ │ │ │ + * │ │ │ │ │ + * All event objects have at least the following properties: │ │ │ │ │ + * object - {Object} A reference to layer.events.object. │ │ │ │ │ + * element - {DOMElement} A reference to layer.events.element. │ │ │ │ │ + * │ │ │ │ │ + * Supported map event types (in addition to those from <OpenLayers.Layer.events>): │ │ │ │ │ + * beforefeatureadded - Triggered before a feature is added. Listeners │ │ │ │ │ + * will receive an object with a *feature* property referencing the │ │ │ │ │ + * feature to be added. To stop the feature from being added, a │ │ │ │ │ + * listener should return false. │ │ │ │ │ + * beforefeaturesadded - Triggered before an array of features is added. │ │ │ │ │ + * Listeners will receive an object with a *features* property │ │ │ │ │ + * referencing the feature to be added. To stop the features from │ │ │ │ │ + * being added, a listener should return false. │ │ │ │ │ + * featureadded - Triggered after a feature is added. The event │ │ │ │ │ + * object passed to listeners will have a *feature* property with a │ │ │ │ │ + * reference to the added feature. │ │ │ │ │ + * featuresadded - Triggered after features are added. The event │ │ │ │ │ + * object passed to listeners will have a *features* property with a │ │ │ │ │ + * reference to an array of added features. │ │ │ │ │ + * beforefeatureremoved - Triggered before a feature is removed. Listeners │ │ │ │ │ + * will receive an object with a *feature* property referencing the │ │ │ │ │ + * feature to be removed. │ │ │ │ │ + * beforefeaturesremoved - Triggered before multiple features are removed. │ │ │ │ │ + * Listeners will receive an object with a *features* property │ │ │ │ │ + * referencing the features to be removed. │ │ │ │ │ + * featureremoved - Triggerd after a feature is removed. The event │ │ │ │ │ + * object passed to listeners will have a *feature* property with a │ │ │ │ │ + * reference to the removed feature. │ │ │ │ │ + * featuresremoved - Triggered after features are removed. The event │ │ │ │ │ + * object passed to listeners will have a *features* property with a │ │ │ │ │ + * reference to an array of removed features. │ │ │ │ │ + * beforefeatureselected - Triggered before a feature is selected. Listeners │ │ │ │ │ + * will receive an object with a *feature* property referencing the │ │ │ │ │ + * feature to be selected. To stop the feature from being selectd, a │ │ │ │ │ + * listener should return false. │ │ │ │ │ + * featureselected - Triggered after a feature is selected. Listeners │ │ │ │ │ + * will receive an object with a *feature* property referencing the │ │ │ │ │ + * selected feature. │ │ │ │ │ + * featureunselected - Triggered after a feature is unselected. │ │ │ │ │ + * Listeners will receive an object with a *feature* property │ │ │ │ │ + * referencing the unselected feature. │ │ │ │ │ + * beforefeaturemodified - Triggered when a feature is selected to │ │ │ │ │ + * be modified. Listeners will receive an object with a *feature* │ │ │ │ │ + * property referencing the selected feature. │ │ │ │ │ + * featuremodified - Triggered when a feature has been modified. │ │ │ │ │ + * Listeners will receive an object with a *feature* property referencing │ │ │ │ │ + * the modified feature. │ │ │ │ │ + * afterfeaturemodified - Triggered when a feature is finished being modified. │ │ │ │ │ + * Listeners will receive an object with a *feature* property referencing │ │ │ │ │ + * the modified feature. │ │ │ │ │ + * vertexmodified - Triggered when a vertex within any feature geometry │ │ │ │ │ + * has been modified. Listeners will receive an object with a │ │ │ │ │ + * *feature* property referencing the modified feature, a *vertex* │ │ │ │ │ + * property referencing the vertex modified (always a point geometry), │ │ │ │ │ + * and a *pixel* property referencing the pixel location of the │ │ │ │ │ + * modification. │ │ │ │ │ + * vertexremoved - Triggered when a vertex within any feature geometry │ │ │ │ │ + * has been deleted. Listeners will receive an object with a │ │ │ │ │ + * *feature* property referencing the modified feature, a *vertex* │ │ │ │ │ + * property referencing the vertex modified (always a point geometry), │ │ │ │ │ + * and a *pixel* property referencing the pixel location of the │ │ │ │ │ + * removal. │ │ │ │ │ + * sketchstarted - Triggered when a feature sketch bound for this layer │ │ │ │ │ + * is started. Listeners will receive an object with a *feature* │ │ │ │ │ + * property referencing the new sketch feature and a *vertex* property │ │ │ │ │ + * referencing the creation point. │ │ │ │ │ + * sketchmodified - Triggered when a feature sketch bound for this layer │ │ │ │ │ + * is modified. Listeners will receive an object with a *vertex* │ │ │ │ │ + * property referencing the modified vertex and a *feature* property │ │ │ │ │ + * referencing the sketch feature. │ │ │ │ │ + * sketchcomplete - Triggered when a feature sketch bound for this layer │ │ │ │ │ + * is complete. Listeners will receive an object with a *feature* │ │ │ │ │ + * property referencing the sketch feature. By returning false, a │ │ │ │ │ + * listener can stop the sketch feature from being added to the layer. │ │ │ │ │ + * refresh - Triggered when something wants a strategy to ask the protocol │ │ │ │ │ + * for a new set of features. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Destroy control. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - "removelayer": this.updateAttribution, │ │ │ │ │ - "addlayer": this.updateAttribution, │ │ │ │ │ - "changelayer": this.updateAttribution, │ │ │ │ │ - "changebaselayer": this.updateAttribution, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} The layer is a base layer. Default is false. Set this property │ │ │ │ │ + * in the layer options. │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: false, │ │ │ │ │ │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isFixed │ │ │ │ │ + * {Boolean} Whether the layer remains in one place while dragging the │ │ │ │ │ + * map. Note that setting this to true will move the layer to the bottom │ │ │ │ │ + * of the layer stack. │ │ │ │ │ + */ │ │ │ │ │ + isFixed: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Initialize control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: features │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + */ │ │ │ │ │ + features: null, │ │ │ │ │ │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - 'changebaselayer': this.updateAttribution, │ │ │ │ │ - 'changelayer': this.updateAttribution, │ │ │ │ │ - 'addlayer': this.updateAttribution, │ │ │ │ │ - 'removelayer': this.updateAttribution, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.updateAttribution(); │ │ │ │ │ + /** │ │ │ │ │ + * Property: filter │ │ │ │ │ + * {<OpenLayers.Filter>} The filter set in this layer, │ │ │ │ │ + * a strategy launching read requests can combined │ │ │ │ │ + * this filter with its own filter. │ │ │ │ │ + */ │ │ │ │ │ + filter: null, │ │ │ │ │ │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: selectedFeatures │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + */ │ │ │ │ │ + selectedFeatures: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateAttribution │ │ │ │ │ - * Update attribution string. │ │ │ │ │ - */ │ │ │ │ │ - updateAttribution: function() { │ │ │ │ │ - var attributions = []; │ │ │ │ │ - if (this.map && this.map.layers) { │ │ │ │ │ - for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if (layer.attribution && layer.getVisibility()) { │ │ │ │ │ - // add attribution only if attribution text is unique │ │ │ │ │ - if (OpenLayers.Util.indexOf( │ │ │ │ │ - attributions, layer.attribution) === -1) { │ │ │ │ │ - attributions.push(layer.attribution); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.div.innerHTML = OpenLayers.String.format(this.template, { │ │ │ │ │ - layers: attributions.join(this.separator) │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: unrenderedFeatures │ │ │ │ │ + * {Object} hash of features, keyed by feature.id, that the renderer │ │ │ │ │ + * failed to draw │ │ │ │ │ + */ │ │ │ │ │ + unrenderedFeatures: null, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Attribution" │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Zoom.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: reportError │ │ │ │ │ + * {Boolean} report friendly error message when loading of renderer │ │ │ │ │ + * fails. │ │ │ │ │ + */ │ │ │ │ │ + reportError: true, │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: style │ │ │ │ │ + * {Object} Default style for the layer │ │ │ │ │ + */ │ │ │ │ │ + style: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: styleMap │ │ │ │ │ + * {<OpenLayers.StyleMap>} │ │ │ │ │ + */ │ │ │ │ │ + styleMap: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Zoom │ │ │ │ │ - * The Zoom control is a pair of +/- links for zooming in and out. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Zoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + /** │ │ │ │ │ + * Property: strategies │ │ │ │ │ + * {Array(<OpenLayers.Strategy>})} Optional list of strategies for the layer. │ │ │ │ │ + */ │ │ │ │ │ + strategies: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomInText │ │ │ │ │ - * {String} │ │ │ │ │ - * Text for zoom-in link. Default is "+". │ │ │ │ │ + * Property: protocol │ │ │ │ │ + * {<OpenLayers.Protocol>} Optional protocol for the layer. │ │ │ │ │ */ │ │ │ │ │ - zoomInText: "+", │ │ │ │ │ + protocol: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomInId │ │ │ │ │ - * {String} │ │ │ │ │ - * Instead of having the control create a zoom in link, you can provide │ │ │ │ │ - * the identifier for an anchor element already added to the document. │ │ │ │ │ - * By default, an element with id "olZoomInLink" will be searched for │ │ │ │ │ - * and used if it exists. │ │ │ │ │ + * Property: renderers │ │ │ │ │ + * {Array(String)} List of supported Renderer classes. Add to this list to │ │ │ │ │ + * add support for additional renderers. This list is ordered: │ │ │ │ │ + * the first renderer which returns true for the 'supported()' │ │ │ │ │ + * method will be used, if not defined in the 'renderer' option. │ │ │ │ │ */ │ │ │ │ │ - zoomInId: "olZoomInLink", │ │ │ │ │ + renderers: ['SVG', 'VML', 'Canvas'], │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: renderer │ │ │ │ │ + * {<OpenLayers.Renderer>} │ │ │ │ │ + */ │ │ │ │ │ + renderer: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomOutText │ │ │ │ │ - * {String} │ │ │ │ │ - * Text for zoom-out link. Default is "\u2212". │ │ │ │ │ + * APIProperty: rendererOptions │ │ │ │ │ + * {Object} Options for the renderer. See {<OpenLayers.Renderer>} for │ │ │ │ │ + * supported options. │ │ │ │ │ + */ │ │ │ │ │ + rendererOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: geometryType │ │ │ │ │ + * {String} geometryType allows you to limit the types of geometries this │ │ │ │ │ + * layer supports. This should be set to something like │ │ │ │ │ + * "OpenLayers.Geometry.Point" to limit types. │ │ │ │ │ + */ │ │ │ │ │ + geometryType: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: drawn │ │ │ │ │ + * {Boolean} Whether the Vector Layer features have been drawn yet. │ │ │ │ │ */ │ │ │ │ │ - zoomOutText: "\u2212", │ │ │ │ │ + drawn: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomOutId │ │ │ │ │ - * {String} │ │ │ │ │ - * Instead of having the control create a zoom out link, you can provide │ │ │ │ │ - * the identifier for an anchor element already added to the document. │ │ │ │ │ - * By default, an element with id "olZoomOutLink" will be searched for │ │ │ │ │ - * and used if it exists. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: ratio │ │ │ │ │ + * {Float} This specifies the ratio of the size of the visiblity of the Vector Layer features to the size of the map. │ │ │ │ │ */ │ │ │ │ │ - zoomOutId: "olZoomOutLink", │ │ │ │ │ + ratio: 1, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ + * Constructor: OpenLayers.Layer.Vector │ │ │ │ │ + * Create a new vector layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} A name for the layer │ │ │ │ │ + * options - {Object} Optional object with non-default properties to set on │ │ │ │ │ + * the layer. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} A reference to the DOMElement containing the zoom links. │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} A new vector layer │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - var div = OpenLayers.Control.prototype.draw.apply(this), │ │ │ │ │ - links = this.getOrCreateLinks(div), │ │ │ │ │ - zoomIn = links.zoomIn, │ │ │ │ │ - zoomOut = links.zoomOut, │ │ │ │ │ - eventsInstance = this.map.events; │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - if (zoomOut.parentNode !== div) { │ │ │ │ │ - eventsInstance = this.events; │ │ │ │ │ - eventsInstance.attachToElement(zoomOut.parentNode); │ │ │ │ │ + // allow user-set renderer, otherwise assign one │ │ │ │ │ + if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ + this.assignRenderer(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // if no valid renderer found, display error │ │ │ │ │ + if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ + this.renderer = null; │ │ │ │ │ + this.displayError(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (!this.styleMap) { │ │ │ │ │ + this.styleMap = new OpenLayers.StyleMap(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.features = []; │ │ │ │ │ + this.selectedFeatures = []; │ │ │ │ │ + this.unrenderedFeatures = {}; │ │ │ │ │ + │ │ │ │ │ + // Allow for custom layer behavior │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + for (var i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + this.strategies[i].setLayer(this); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - eventsInstance.register("buttonclick", this, this.onZoomClick); │ │ │ │ │ │ │ │ │ │ - this.zoomInLink = zoomIn; │ │ │ │ │ - this.zoomOutLink = zoomOut; │ │ │ │ │ - return div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getOrCreateLinks │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Destroy this layer │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoDestroy) { │ │ │ │ │ + strategy.destroy(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.strategies = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.protocol) { │ │ │ │ │ + if (this.protocol.autoDestroy) { │ │ │ │ │ + this.protocol.destroy(); │ │ │ │ │ + } │ │ │ │ │ + this.protocol = null; │ │ │ │ │ + } │ │ │ │ │ + this.destroyFeatures(); │ │ │ │ │ + this.features = null; │ │ │ │ │ + this.selectedFeatures = null; │ │ │ │ │ + this.unrenderedFeatures = null; │ │ │ │ │ + if (this.renderer) { │ │ │ │ │ + this.renderer.destroy(); │ │ │ │ │ + } │ │ │ │ │ + this.renderer = null; │ │ │ │ │ + this.geometryType = null; │ │ │ │ │ + this.drawn = null; │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * el - {DOMElement} │ │ │ │ │ + * Note: Features of the layer are also cloned. │ │ │ │ │ * │ │ │ │ │ - * Return: │ │ │ │ │ - * {Object} Object with zoomIn and zoomOut properties referencing links. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} An exact clone of this layer │ │ │ │ │ */ │ │ │ │ │ - getOrCreateLinks: function(el) { │ │ │ │ │ - var zoomIn = document.getElementById(this.zoomInId), │ │ │ │ │ - zoomOut = document.getElementById(this.zoomOutId); │ │ │ │ │ - if (!zoomIn) { │ │ │ │ │ - zoomIn = document.createElement("a"); │ │ │ │ │ - zoomIn.href = "#zoomIn"; │ │ │ │ │ - zoomIn.appendChild(document.createTextNode(this.zoomInText)); │ │ │ │ │ - zoomIn.className = "olControlZoomIn"; │ │ │ │ │ - el.appendChild(zoomIn); │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Vector(this.name, this.getOptions()); │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Element.addClass(zoomIn, "olButton"); │ │ │ │ │ - if (!zoomOut) { │ │ │ │ │ - zoomOut = document.createElement("a"); │ │ │ │ │ - zoomOut.href = "#zoomOut"; │ │ │ │ │ - zoomOut.appendChild(document.createTextNode(this.zoomOutText)); │ │ │ │ │ - zoomOut.className = "olControlZoomOut"; │ │ │ │ │ - el.appendChild(zoomOut); │ │ │ │ │ + │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ + │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + var features = this.features; │ │ │ │ │ + var len = features.length; │ │ │ │ │ + var clonedFeatures = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + clonedFeatures[i] = features[i].clone(); │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Element.addClass(zoomOut, "olButton"); │ │ │ │ │ - return { │ │ │ │ │ - zoomIn: zoomIn, │ │ │ │ │ - zoomOut: zoomOut │ │ │ │ │ - }; │ │ │ │ │ + obj.features = clonedFeatures; │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onZoomClick │ │ │ │ │ - * Called when zoomin/out link is clicked. │ │ │ │ │ + * Method: refresh │ │ │ │ │ + * Ask the layer to request features again and redraw them. Triggers │ │ │ │ │ + * the refresh event if the layer is in range and visible. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} Optional object with properties for any listener of │ │ │ │ │ + * the refresh event. │ │ │ │ │ */ │ │ │ │ │ - onZoomClick: function(evt) { │ │ │ │ │ - var button = evt.buttonElement; │ │ │ │ │ - if (button === this.zoomInLink) { │ │ │ │ │ - this.map.zoomIn(); │ │ │ │ │ - } else if (button === this.zoomOutLink) { │ │ │ │ │ - this.map.zoomOut(); │ │ │ │ │ + refresh: function(obj) { │ │ │ │ │ + if (this.calculateInRange() && this.visibility) { │ │ │ │ │ + this.events.triggerEvent("refresh", obj); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Clean up. │ │ │ │ │ + * Method: assignRenderer │ │ │ │ │ + * Iterates through the available renderer implementations and selects │ │ │ │ │ + * and assigns the first one whose "supported()" function returns true. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onZoomClick); │ │ │ │ │ + assignRenderer: function() { │ │ │ │ │ + for (var i = 0, len = this.renderers.length; i < len; i++) { │ │ │ │ │ + var rendererClass = this.renderers[i]; │ │ │ │ │ + var renderer = (typeof rendererClass == "function") ? │ │ │ │ │ + rendererClass : │ │ │ │ │ + OpenLayers.Renderer[rendererClass]; │ │ │ │ │ + if (renderer && renderer.prototype.supported()) { │ │ │ │ │ + this.renderer = new renderer(this.div, this.rendererOptions); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - delete this.zoomInLink; │ │ │ │ │ - delete this.zoomOutLink; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Zoom" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Feature.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: displayError │ │ │ │ │ + * Let the user know their browser isn't supported. │ │ │ │ │ + */ │ │ │ │ │ + displayError: function() { │ │ │ │ │ + if (this.reportError) { │ │ │ │ │ + OpenLayers.Console.userError(OpenLayers.i18n("browserNotSupported", { │ │ │ │ │ + renderers: this.renderers.join('\n') │ │ │ │ │ + })); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * The layer has been added to the map. │ │ │ │ │ + * │ │ │ │ │ + * If there is no renderer set, the layer can't be used. Remove it. │ │ │ │ │ + * Otherwise, give the renderer a reference to the map and set its size. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ - */ │ │ │ │ │ + if (!this.renderer) { │ │ │ │ │ + this.map.removeLayer(this); │ │ │ │ │ + } else { │ │ │ │ │ + this.renderer.map = this.map; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Handler.Feature │ │ │ │ │ - * Handler to respond to mouse events related to a drawn feature. Callbacks │ │ │ │ │ - * with the following keys will be notified of the following events │ │ │ │ │ - * associated with features: click, clickout, over, out, and dblclick. │ │ │ │ │ - * │ │ │ │ │ - * This handler stops event propagation for mousedown and mouseup if those │ │ │ │ │ - * browser events target features that can be selected. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.Feature = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + var newSize = this.map.getSize(); │ │ │ │ │ + newSize.w = newSize.w * this.ratio; │ │ │ │ │ + newSize.h = newSize.h * this.ratio; │ │ │ │ │ + this.renderer.setSize(newSize); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: EVENTMAP │ │ │ │ │ - * {Object} A object mapping the browser events to objects with callback │ │ │ │ │ - * keys for in and out. │ │ │ │ │ + * Method: afterAdd │ │ │ │ │ + * Called at the end of the map.addLayer sequence. At this point, the map │ │ │ │ │ + * will have a base layer. Any autoActivate strategies will be │ │ │ │ │ + * activated here. │ │ │ │ │ */ │ │ │ │ │ - EVENTMAP: { │ │ │ │ │ - 'click': { │ │ │ │ │ - 'in': 'click', │ │ │ │ │ - 'out': 'clickout' │ │ │ │ │ - }, │ │ │ │ │ - 'mousemove': { │ │ │ │ │ - 'in': 'over', │ │ │ │ │ - 'out': 'out' │ │ │ │ │ - }, │ │ │ │ │ - 'dblclick': { │ │ │ │ │ - 'in': 'dblclick', │ │ │ │ │ - 'out': null │ │ │ │ │ - }, │ │ │ │ │ - 'mousedown': { │ │ │ │ │ - 'in': null, │ │ │ │ │ - 'out': null │ │ │ │ │ - }, │ │ │ │ │ - 'mouseup': { │ │ │ │ │ - 'in': null, │ │ │ │ │ - 'out': null │ │ │ │ │ - }, │ │ │ │ │ - 'touchstart': { │ │ │ │ │ - 'in': 'click', │ │ │ │ │ - 'out': 'clickout' │ │ │ │ │ + afterAdd: function() { │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoActivate) { │ │ │ │ │ + strategy.activate(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: feature │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} The last feature that was hovered. │ │ │ │ │ + * Method: removeMap │ │ │ │ │ + * The layer has been removed from the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - feature: null, │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + this.drawn = false; │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoActivate) { │ │ │ │ │ + strategy.deactivate(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: lastFeature │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} The last feature that was handled. │ │ │ │ │ + * Method: onMapResize │ │ │ │ │ + * Notify the renderer of the change in size. │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - lastFeature: null, │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + OpenLayers.Layer.prototype.onMapResize.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: down │ │ │ │ │ - * {<OpenLayers.Pixel>} The location of the last mousedown. │ │ │ │ │ - */ │ │ │ │ │ - down: null, │ │ │ │ │ + var newSize = this.map.getSize(); │ │ │ │ │ + newSize.w = newSize.w * this.ratio; │ │ │ │ │ + newSize.h = newSize.h * this.ratio; │ │ │ │ │ + this.renderer.setSize(newSize); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: up │ │ │ │ │ - * {<OpenLayers.Pixel>} The location of the last mouseup. │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * Reset the vector layer's div so that it once again is lined up with │ │ │ │ │ + * the map. Notify the renderer of the change of extent, and in the │ │ │ │ │ + * case of a change of zoom level (resolution), have the │ │ │ │ │ + * renderer redraw features. │ │ │ │ │ + * │ │ │ │ │ + * If the layer has not yet been drawn, cycle through the layer's │ │ │ │ │ + * features and draw each one. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * zoomChanged - {Boolean} │ │ │ │ │ + * dragging - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - up: null, │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: clickTolerance │ │ │ │ │ - * {Number} The number of pixels the mouse can move between mousedown │ │ │ │ │ - * and mouseup for the event to still be considered a click. │ │ │ │ │ - * Dragging the map should not trigger the click and clickout callbacks │ │ │ │ │ - * unless the map is moved by less than this tolerance. Defaults to 4. │ │ │ │ │ - */ │ │ │ │ │ - clickTolerance: 4, │ │ │ │ │ + var coordSysUnchanged = true; │ │ │ │ │ + if (!dragging) { │ │ │ │ │ + this.renderer.root.style.visibility = 'hidden'; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: geometryTypes │ │ │ │ │ - * To restrict dragging to a limited set of geometry types, send a list │ │ │ │ │ - * of strings corresponding to the geometry class names. │ │ │ │ │ - * │ │ │ │ │ - * @type Array(String) │ │ │ │ │ - */ │ │ │ │ │ - geometryTypes: null, │ │ │ │ │ + var viewSize = this.map.getSize(), │ │ │ │ │ + viewWidth = viewSize.w, │ │ │ │ │ + viewHeight = viewSize.h, │ │ │ │ │ + offsetLeft = (viewWidth / 2 * this.ratio) - viewWidth / 2, │ │ │ │ │ + offsetTop = (viewHeight / 2 * this.ratio) - viewHeight / 2; │ │ │ │ │ + offsetLeft += this.map.layerContainerOriginPx.x; │ │ │ │ │ + offsetLeft = -Math.round(offsetLeft); │ │ │ │ │ + offsetTop += this.map.layerContainerOriginPx.y; │ │ │ │ │ + offsetTop = -Math.round(offsetTop); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: stopClick │ │ │ │ │ - * {Boolean} If stopClick is set to true, handled clicks do not │ │ │ │ │ - * propagate to other click listeners. Otherwise, handled clicks │ │ │ │ │ - * do propagate. Unhandled clicks always propagate, whatever the │ │ │ │ │ - * value of stopClick. Defaults to true. │ │ │ │ │ - */ │ │ │ │ │ - stopClick: true, │ │ │ │ │ + this.div.style.left = offsetLeft + 'px'; │ │ │ │ │ + this.div.style.top = offsetTop + 'px'; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: stopDown │ │ │ │ │ - * {Boolean} If stopDown is set to true, handled mousedowns do not │ │ │ │ │ - * propagate to other mousedown listeners. Otherwise, handled │ │ │ │ │ - * mousedowns do propagate. Unhandled mousedowns always propagate, │ │ │ │ │ - * whatever the value of stopDown. Defaults to true. │ │ │ │ │ - */ │ │ │ │ │ - stopDown: true, │ │ │ │ │ + var extent = this.map.getExtent().scale(this.ratio); │ │ │ │ │ + coordSysUnchanged = this.renderer.setExtent(extent, zoomChanged); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: stopUp │ │ │ │ │ - * {Boolean} If stopUp is set to true, handled mouseups do not │ │ │ │ │ - * propagate to other mouseup listeners. Otherwise, handled mouseups │ │ │ │ │ - * do propagate. Unhandled mouseups always propagate, whatever the │ │ │ │ │ - * value of stopUp. Defaults to false. │ │ │ │ │ + this.renderer.root.style.visibility = 'visible'; │ │ │ │ │ + │ │ │ │ │ + // Force a reflow on gecko based browsers to prevent jump/flicker. │ │ │ │ │ + // This seems to happen on only certain configurations; it was originally │ │ │ │ │ + // noticed in FF 2.0 and Linux. │ │ │ │ │ + if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ + this.div.scrollLeft = this.div.scrollLeft; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (!zoomChanged && coordSysUnchanged) { │ │ │ │ │ + for (var i in this.unrenderedFeatures) { │ │ │ │ │ + var feature = this.unrenderedFeatures[i]; │ │ │ │ │ + this.drawFeature(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!this.drawn || zoomChanged || !coordSysUnchanged) { │ │ │ │ │ + this.drawn = true; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0, len = this.features.length; i < len; i++) { │ │ │ │ │ + this.renderer.locked = (i !== (len - 1)); │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + this.drawFeature(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: display │ │ │ │ │ + * Hide or show the Layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * display - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - stopUp: false, │ │ │ │ │ + display: function(display) { │ │ │ │ │ + OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ + // we need to set the display style of the root in case it is attached │ │ │ │ │ + // to a foreign layer │ │ │ │ │ + var currentDisplay = this.div.style.display; │ │ │ │ │ + if (currentDisplay != this.renderer.root.style.display) { │ │ │ │ │ + this.renderer.root.style.display = currentDisplay; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Feature │ │ │ │ │ + * APIMethod: addFeatures │ │ │ │ │ + * Add Features to the layer. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} │ │ │ │ │ - * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ - * callbacks - {Object} An object with a 'over' property whos value is │ │ │ │ │ - * a function to be called when the mouse is over a feature. The │ │ │ │ │ - * callback should expect to recieve a single argument, the feature. │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(control, layer, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ - this.layer = layer; │ │ │ │ │ + addFeatures: function(features, options) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ + if (notify) { │ │ │ │ │ + var event = { │ │ │ │ │ + features: features │ │ │ │ │ + }; │ │ │ │ │ + var ret = this.events.triggerEvent("beforefeaturesadded", event); │ │ │ │ │ + if (ret === false) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + features = event.features; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Track successfully added features for featuresadded event, since │ │ │ │ │ + // beforefeatureadded can veto single features. │ │ │ │ │ + var featuresAdded = []; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + if (i != (features.length - 1)) { │ │ │ │ │ + this.renderer.locked = true; │ │ │ │ │ + } else { │ │ │ │ │ + this.renderer.locked = false; │ │ │ │ │ + } │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + │ │ │ │ │ + if (this.geometryType && │ │ │ │ │ + !(feature.geometry instanceof this.geometryType)) { │ │ │ │ │ + throw new TypeError('addFeatures: component should be an ' + │ │ │ │ │ + this.geometryType.prototype.CLASS_NAME); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //give feature reference to its layer │ │ │ │ │ + feature.layer = this; │ │ │ │ │ + │ │ │ │ │ + if (!feature.style && this.style) { │ │ │ │ │ + feature.style = OpenLayers.Util.extend({}, this.style); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (notify) { │ │ │ │ │ + if (this.events.triggerEvent("beforefeatureadded", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) === false) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + this.preFeatureInsert(feature); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + featuresAdded.push(feature); │ │ │ │ │ + this.features.push(feature); │ │ │ │ │ + this.drawFeature(feature); │ │ │ │ │ + │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureadded", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.onFeatureInsert(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresadded", { │ │ │ │ │ + features: featuresAdded │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: touchstart │ │ │ │ │ - * Handle touchstart events │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: removeFeatures │ │ │ │ │ + * Remove features from the layer. This erases any drawn features and │ │ │ │ │ + * removes them from the layer's control. The beforefeatureremoved │ │ │ │ │ + * and featureremoved events will be triggered for each feature. The │ │ │ │ │ + * featuresremoved event will be triggered after all features have │ │ │ │ │ + * been removed. To supress event triggering, use the silent option. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} List of features to be │ │ │ │ │ + * removed. │ │ │ │ │ + * options - {Object} Optional properties for changing behavior of the │ │ │ │ │ + * removal. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * Valid options: │ │ │ │ │ + * silent - {Boolean} Supress event triggering. Default is false. │ │ │ │ │ */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - return OpenLayers.Event.isMultiTouch(evt) ? │ │ │ │ │ - true : this.mousedown(evt); │ │ │ │ │ + removeFeatures: function(features, options) { │ │ │ │ │ + if (!features || features.length === 0) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (features === this.features) { │ │ │ │ │ + return this.removeAllFeatures(options); │ │ │ │ │ + } │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ + } │ │ │ │ │ + if (features === this.selectedFeatures) { │ │ │ │ │ + features = features.slice(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ + │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent( │ │ │ │ │ + "beforefeaturesremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + // We remain locked so long as we're not at 0 │ │ │ │ │ + // and the 'next' feature has a geometry. We do the geometry check │ │ │ │ │ + // because if all the features after the current one are 'null', we │ │ │ │ │ + // won't call eraseGeometry, so we break the 'renderer functions │ │ │ │ │ + // will always be called with locked=false *last*' rule. The end result │ │ │ │ │ + // is a possible gratiutious unlocking to save a loop through the rest │ │ │ │ │ + // of the list checking the remaining features every time. So long as │ │ │ │ │ + // null geoms are rare, this is probably okay. │ │ │ │ │ + if (i != 0 && features[i - 1].geometry) { │ │ │ │ │ + this.renderer.locked = true; │ │ │ │ │ + } else { │ │ │ │ │ + this.renderer.locked = false; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ + │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.features = OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ + // feature has no layer at this point │ │ │ │ │ + feature.layer = null; │ │ │ │ │ + │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + this.renderer.eraseFeatures(feature); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //in the case that this feature is one of the selected features, │ │ │ │ │ + // remove it from that array as well. │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.selectedFeatures, feature) != -1) { │ │ │ │ │ + OpenLayers.Util.removeItem(this.selectedFeatures, feature); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchmove │ │ │ │ │ - * Handle touchmove events. We just prevent the browser default behavior, │ │ │ │ │ - * for Android Webkit not to select text when moving the finger after │ │ │ │ │ - * selecting a feature. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: removeAllFeatures │ │ │ │ │ + * Remove all features from the layer. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * options - {Object} Optional properties for changing behavior of the │ │ │ │ │ + * removal. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * silent - {Boolean} Supress event triggering. Default is false. │ │ │ │ │ */ │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + removeAllFeatures: function(options) { │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ + var features = this.features; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent( │ │ │ │ │ + "beforefeaturesremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + feature.layer = null; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.renderer.clear(); │ │ │ │ │ + this.features = []; │ │ │ │ │ + this.unrenderedFeatures = {}; │ │ │ │ │ + this.selectedFeatures = []; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mousedown │ │ │ │ │ - * Handle mouse down. Stop propagation if a feature is targeted by this │ │ │ │ │ - * event (stops map dragging during feature selection). │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: destroyFeatures │ │ │ │ │ + * Erase and destroy features on the layer. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} An optional array of │ │ │ │ │ + * features to destroy. If not supplied, all features on the layer │ │ │ │ │ + * will be destroyed. │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - // Feature selection is only done with a left click. Other handlers may stop the │ │ │ │ │ - // propagation of left-click mousedown events but not right-click mousedown events. │ │ │ │ │ - // This mismatch causes problems when comparing the location of the down and up │ │ │ │ │ - // events in the click function so it is important ignore right-clicks. │ │ │ │ │ - if (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ - this.down = evt.xy; │ │ │ │ │ + destroyFeatures: function(features, options) { │ │ │ │ │ + var all = (features == undefined); // evaluates to true if │ │ │ │ │ + // features is null │ │ │ │ │ + if (all) { │ │ │ │ │ + features = this.features; │ │ │ │ │ + } │ │ │ │ │ + if (features) { │ │ │ │ │ + this.removeFeatures(features, options); │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + features[i].destroy(); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return this.handle(evt) ? !this.stopDown : true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mouseup │ │ │ │ │ - * Handle mouse up. Stop propagation if a feature is targeted by this │ │ │ │ │ - * event. │ │ │ │ │ + * APIMethod: drawFeature │ │ │ │ │ + * Draw (or redraw) a feature on the layer. If the optional style argument │ │ │ │ │ + * is included, this style will be used. If no style is included, the │ │ │ │ │ + * feature's style will be used. If the feature doesn't have a style, │ │ │ │ │ + * the layer's style will be used. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * This function is not designed to be used when adding features to │ │ │ │ │ + * the layer (use addFeatures instead). It is meant to be used when │ │ │ │ │ + * the style of a feature has changed, or in some other way needs to │ │ │ │ │ + * visually updated *after* it has already been added to a layer. You │ │ │ │ │ + * must add the feature to the layer for most layer-related events to │ │ │ │ │ + * happen. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * style - {String | Object} Named render intent or full symbolizer object. │ │ │ │ │ */ │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - this.up = evt.xy; │ │ │ │ │ - return this.handle(evt) ? !this.stopUp : true; │ │ │ │ │ + drawFeature: function(feature, style) { │ │ │ │ │ + // don't try to draw the feature with the renderer if the layer is not │ │ │ │ │ + // drawn itself │ │ │ │ │ + if (!this.drawn) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (typeof style != "object") { │ │ │ │ │ + if (!style && feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ + style = "delete"; │ │ │ │ │ + } │ │ │ │ │ + var renderIntent = style || feature.renderIntent; │ │ │ │ │ + style = feature.style || this.style; │ │ │ │ │ + if (!style) { │ │ │ │ │ + style = this.styleMap.createSymbolizer(feature, renderIntent); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var drawn = this.renderer.drawFeature(feature, style); │ │ │ │ │ + //TODO remove the check for null when we get rid of Renderer.SVG │ │ │ │ │ + if (drawn === false || drawn === null) { │ │ │ │ │ + this.unrenderedFeatures[feature.id] = feature; │ │ │ │ │ + } else { │ │ │ │ │ + delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: click │ │ │ │ │ - * Handle click. Call the "click" callback if click on a feature, │ │ │ │ │ - * or the "clickout" callback if click outside any feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * Method: eraseFeatures │ │ │ │ │ + * Erase features from the layer. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ */ │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - return this.handle(evt) ? !this.stopClick : true; │ │ │ │ │ + eraseFeatures: function(features) { │ │ │ │ │ + this.renderer.eraseFeatures(features); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mousemove │ │ │ │ │ - * Handle mouse moves. Call the "over" callback if moving in to a feature, │ │ │ │ │ - * or the "out" callback if moving out of a feature. │ │ │ │ │ - * │ │ │ │ │ + * Method: getFeatureFromEvent │ │ │ │ │ + * Given an event, return a feature if the event occurred over one. │ │ │ │ │ + * Otherwise, return null. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A feature if one was under the event. │ │ │ │ │ */ │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - if (!this.callbacks['over'] && !this.callbacks['out']) { │ │ │ │ │ - return true; │ │ │ │ │ + getFeatureFromEvent: function(evt) { │ │ │ │ │ + if (!this.renderer) { │ │ │ │ │ + throw new Error('getFeatureFromEvent called on layer with no ' + │ │ │ │ │ + 'renderer. This usually means you destroyed a ' + │ │ │ │ │ + 'layer, but not some handler which is associated ' + │ │ │ │ │ + 'with it.'); │ │ │ │ │ } │ │ │ │ │ - this.handle(evt); │ │ │ │ │ - return true; │ │ │ │ │ + var feature = null; │ │ │ │ │ + var featureId = this.renderer.getFeatureIdFromEvent(evt); │ │ │ │ │ + if (featureId) { │ │ │ │ │ + if (typeof featureId === "string") { │ │ │ │ │ + feature = this.getFeatureById(featureId); │ │ │ │ │ + } else { │ │ │ │ │ + feature = featureId; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return feature; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: dblclick │ │ │ │ │ - * Handle dblclick. Call the "dblclick" callback if dblclick on a feature. │ │ │ │ │ + * APIMethod: getFeatureBy │ │ │ │ │ + * Given a property value, return the feature if it exists in the features array │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * property - {String} │ │ │ │ │ + * value - {String} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A feature corresponding to the given │ │ │ │ │ + * property value or null if there is no such feature. │ │ │ │ │ */ │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - return !this.handle(evt); │ │ │ │ │ + getFeatureBy: function(property, value) { │ │ │ │ │ + //TBD - would it be more efficient to use a hash for this.features? │ │ │ │ │ + var feature = null; │ │ │ │ │ + for (var i = 0, len = this.features.length; i < len; ++i) { │ │ │ │ │ + if (this.features[i][property] == value) { │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return feature; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: geometryTypeMatches │ │ │ │ │ - * Return true if the geometry type of the passed feature matches │ │ │ │ │ - * one of the geometry types in the geometryTypes array. │ │ │ │ │ + * APIMethod: getFeatureById │ │ │ │ │ + * Given a feature id, return the feature if it exists in the features array │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Vector.Feature>} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A feature corresponding to the given │ │ │ │ │ + * featureId or null if there is no such feature. │ │ │ │ │ */ │ │ │ │ │ - geometryTypeMatches: function(feature) { │ │ │ │ │ - return this.geometryTypes == null || │ │ │ │ │ - OpenLayers.Util.indexOf(this.geometryTypes, │ │ │ │ │ - feature.geometry.CLASS_NAME) > -1; │ │ │ │ │ + getFeatureById: function(featureId) { │ │ │ │ │ + return this.getFeatureBy('id', featureId); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handle │ │ │ │ │ + * APIMethod: getFeatureByFid │ │ │ │ │ + * Given a feature fid, return the feature if it exists in the features array │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * featureFid - {String} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The event occurred over a relevant feature. │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A feature corresponding to the given │ │ │ │ │ + * featureFid or null if there is no such feature. │ │ │ │ │ */ │ │ │ │ │ - handle: function(evt) { │ │ │ │ │ - if (this.feature && !this.feature.layer) { │ │ │ │ │ - // feature has been destroyed │ │ │ │ │ - this.feature = null; │ │ │ │ │ - } │ │ │ │ │ - var type = evt.type; │ │ │ │ │ - var handled = false; │ │ │ │ │ - var previouslyIn = !!(this.feature); // previously in a feature │ │ │ │ │ - var click = (type == "click" || type == "dblclick" || type == "touchstart"); │ │ │ │ │ - this.feature = this.layer.getFeatureFromEvent(evt); │ │ │ │ │ - if (this.feature && !this.feature.layer) { │ │ │ │ │ - // feature has been destroyed │ │ │ │ │ - this.feature = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.lastFeature && !this.lastFeature.layer) { │ │ │ │ │ - // last feature has been destroyed │ │ │ │ │ - this.lastFeature = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.feature) { │ │ │ │ │ - if (type === "touchstart") { │ │ │ │ │ - // stop the event to prevent Android Webkit from │ │ │ │ │ - // "flashing" the map div │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ - } │ │ │ │ │ - var inNew = (this.feature != this.lastFeature); │ │ │ │ │ - if (this.geometryTypeMatches(this.feature)) { │ │ │ │ │ - // in to a feature │ │ │ │ │ - if (previouslyIn && inNew) { │ │ │ │ │ - // out of last feature and in to another │ │ │ │ │ - if (this.lastFeature) { │ │ │ │ │ - this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ - } │ │ │ │ │ - this.triggerCallback(type, 'in', [this.feature]); │ │ │ │ │ - } else if (!previouslyIn || click) { │ │ │ │ │ - // in feature for the first time │ │ │ │ │ - this.triggerCallback(type, 'in', [this.feature]); │ │ │ │ │ - } │ │ │ │ │ - this.lastFeature = this.feature; │ │ │ │ │ - handled = true; │ │ │ │ │ - } else { │ │ │ │ │ - // not in to a feature │ │ │ │ │ - if (this.lastFeature && (previouslyIn && inNew || click)) { │ │ │ │ │ - // out of last feature for the first time │ │ │ │ │ - this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ - } │ │ │ │ │ - // next time the mouse goes in a feature whose geometry type │ │ │ │ │ - // doesn't match we don't want to call the 'out' callback │ │ │ │ │ - // again, so let's set this.feature to null so that │ │ │ │ │ - // previouslyIn will evaluate to false the next time │ │ │ │ │ - // we enter handle. Yes, a bit hackish... │ │ │ │ │ - this.feature = null; │ │ │ │ │ - } │ │ │ │ │ - } else if (this.lastFeature && (previouslyIn || click)) { │ │ │ │ │ - this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ - } │ │ │ │ │ - return handled; │ │ │ │ │ + getFeatureByFid: function(featureFid) { │ │ │ │ │ + return this.getFeatureBy('fid', featureFid); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: triggerCallback │ │ │ │ │ - * Call the callback keyed in the event map with the supplied arguments. │ │ │ │ │ - * For click and clickout, the <clickTolerance> is checked first. │ │ │ │ │ + * APIMethod: getFeaturesByAttribute │ │ │ │ │ + * Returns an array of features that have the given attribute key set to the │ │ │ │ │ + * given value. Comparison of attribute values takes care of datatypes, e.g. │ │ │ │ │ + * the string '1234' is not equal to the number 1234. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * type - {String} │ │ │ │ │ + * attrName - {String} │ │ │ │ │ + * attrValue - {Mixed} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * Array({<OpenLayers.Feature.Vector>}) An array of features that have the │ │ │ │ │ + * passed named attribute set to the given value. │ │ │ │ │ */ │ │ │ │ │ - triggerCallback: function(type, mode, args) { │ │ │ │ │ - var key = this.EVENTMAP[type][mode]; │ │ │ │ │ - if (key) { │ │ │ │ │ - if (type == 'click' && this.up && this.down) { │ │ │ │ │ - // for click/clickout, only trigger callback if tolerance is met │ │ │ │ │ - var dpx = Math.sqrt( │ │ │ │ │ - Math.pow(this.up.x - this.down.x, 2) + │ │ │ │ │ - Math.pow(this.up.y - this.down.y, 2) │ │ │ │ │ - ); │ │ │ │ │ - if (dpx <= this.clickTolerance) { │ │ │ │ │ - this.callback(key, args); │ │ │ │ │ + getFeaturesByAttribute: function(attrName, attrValue) { │ │ │ │ │ + var i, │ │ │ │ │ + feature, │ │ │ │ │ + len = this.features.length, │ │ │ │ │ + foundFeatures = []; │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + if (feature && feature.attributes) { │ │ │ │ │ + if (feature.attributes[attrName] === attrValue) { │ │ │ │ │ + foundFeatures.push(feature); │ │ │ │ │ } │ │ │ │ │ - // we're done with this set of events now: clear the cached │ │ │ │ │ - // positions so we can't trip over them later (this can occur │ │ │ │ │ - // if one of the up/down events gets eaten before it gets to us │ │ │ │ │ - // but we still get the click) │ │ │ │ │ - this.up = this.down = null; │ │ │ │ │ - } else { │ │ │ │ │ - this.callback(key, args); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return foundFeatures; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Turn on the handler. Returns false if the handler was already active. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerToTop(); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - "removelayer": this.handleMapEvents, │ │ │ │ │ - "changelayer": this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - activated = true; │ │ │ │ │ + * Unselect the selected features │ │ │ │ │ + * i.e. clears the featureSelection array │ │ │ │ │ + * change the style back │ │ │ │ │ + clearSelection: function() { │ │ │ │ │ + │ │ │ │ │ + var vectorLayer = this.map.vectorLayer; │ │ │ │ │ + for (var i = 0; i < this.map.featureSelection.length; i++) { │ │ │ │ │ + var featureSelection = this.map.featureSelection[i]; │ │ │ │ │ + vectorLayer.drawFeature(featureSelection, vectorLayer.style); │ │ │ │ │ } │ │ │ │ │ - return activated; │ │ │ │ │ + this.map.featureSelection = []; │ │ │ │ │ }, │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Turn off the handler. Returns false if the handler was already active. │ │ │ │ │ + * APIMethod: onFeatureInsert │ │ │ │ │ + * method called after a feature is inserted. │ │ │ │ │ + * Does nothing by default. Override this if you │ │ │ │ │ + * need to do something on feature updates. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerBack(); │ │ │ │ │ - this.feature = null; │ │ │ │ │ - this.lastFeature = null; │ │ │ │ │ - this.down = null; │ │ │ │ │ - this.up = null; │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - "removelayer": this.handleMapEvents, │ │ │ │ │ - "changelayer": this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - deactivated = true; │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ + onFeatureInsert: function(feature) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleMapEvents │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: preFeatureInsert │ │ │ │ │ + * method called before a feature is inserted. │ │ │ │ │ + * Does nothing by default. Override this if you │ │ │ │ │ + * need to do something when features are first added to the │ │ │ │ │ + * layer, but before they are drawn, such as adjust the style. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ - */ │ │ │ │ │ - handleMapEvents: function(evt) { │ │ │ │ │ - if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ - this.moveLayerToTop(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveLayerToTop │ │ │ │ │ - * Moves the layer for this handler to the top, so mouse events can reach │ │ │ │ │ - * it. │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - moveLayerToTop: function() { │ │ │ │ │ - var index = Math.max(this.map.Z_INDEX_BASE['Feature'] - 1, │ │ │ │ │ - this.layer.getZIndex()) + 1; │ │ │ │ │ - this.layer.setZIndex(index); │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ + preFeatureInsert: function(feature) {}, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveLayerBack │ │ │ │ │ - * Moves the layer back to the position determined by the map's layers │ │ │ │ │ - * array. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getDataExtent │ │ │ │ │ + * Calculates the max extent which includes all of the features. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} or null if the layer has no features with │ │ │ │ │ + * geometries. │ │ │ │ │ */ │ │ │ │ │ - moveLayerBack: function() { │ │ │ │ │ - var index = this.layer.getZIndex() - 1; │ │ │ │ │ - if (index >= this.map.Z_INDEX_BASE['Feature']) { │ │ │ │ │ - this.layer.setZIndex(index); │ │ │ │ │ - } else { │ │ │ │ │ - this.map.setLayerZIndex(this.layer, │ │ │ │ │ - this.map.getLayerIndex(this.layer)); │ │ │ │ │ + getDataExtent: function() { │ │ │ │ │ + var maxExtent = null; │ │ │ │ │ + var features = this.features; │ │ │ │ │ + if (features && (features.length > 0)) { │ │ │ │ │ + var geometry = null; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + geometry = features[i].geometry; │ │ │ │ │ + if (geometry) { │ │ │ │ │ + if (maxExtent === null) { │ │ │ │ │ + maxExtent = new OpenLayers.Bounds(); │ │ │ │ │ + } │ │ │ │ │ + maxExtent.extend(geometry.getBounds()); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return maxExtent; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Feature" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Vector" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Layer/Vector/RootContainer.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ @@ -32115,5404 +29462,8057 @@ │ │ │ │ │ this.activate(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.SelectFeature" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/LayerSwitcher.js │ │ │ │ │ + OpenLayers/Control/Attribution.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.LayerSwitcher │ │ │ │ │ - * The LayerSwitcher control displays a table of contents for the map. This │ │ │ │ │ - * allows the user interface to switch between BaseLasyers and to show or hide │ │ │ │ │ - * Overlays. By default the switcher is shown minimized on the right edge of │ │ │ │ │ - * the map, the user may expand it by clicking on the handle. │ │ │ │ │ - * │ │ │ │ │ - * To create the LayerSwitcher outside of the map, pass the Id of a html div │ │ │ │ │ - * as the first argument to the constructor. │ │ │ │ │ + * Class: OpenLayers.Control.Attribution │ │ │ │ │ + * The attribution control adds attribution from layers to the map display. │ │ │ │ │ + * It uses 'attribution' property of each layer. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.LayerSwitcher = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Control.Attribution = │ │ │ │ │ + OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: layerStates │ │ │ │ │ - * {Array(Object)} Basically a copy of the "state" of the map's layers │ │ │ │ │ - * the last time the control was drawn. We have this in order to avoid │ │ │ │ │ - * unnecessarily redrawing the control. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: separator │ │ │ │ │ + * {String} String used to separate layers. │ │ │ │ │ + */ │ │ │ │ │ + separator: ", ", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: template │ │ │ │ │ + * {String} Template for the attribution. This has to include the substring │ │ │ │ │ + * "${layers}", which will be replaced by the layer specific │ │ │ │ │ + * attributions, separated by <separator>. The default is "${layers}". │ │ │ │ │ + */ │ │ │ │ │ + template: "${layers}", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.Attribution │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Options for control. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Destroy control. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + "removelayer": this.updateAttribution, │ │ │ │ │ + "addlayer": this.updateAttribution, │ │ │ │ │ + "changelayer": this.updateAttribution, │ │ │ │ │ + "changebaselayer": this.updateAttribution, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Initialize control. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ + */ │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + 'changebaselayer': this.updateAttribution, │ │ │ │ │ + 'changelayer': this.updateAttribution, │ │ │ │ │ + 'addlayer': this.updateAttribution, │ │ │ │ │ + 'removelayer': this.updateAttribution, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.updateAttribution(); │ │ │ │ │ + │ │ │ │ │ + return this.div; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateAttribution │ │ │ │ │ + * Update attribution string. │ │ │ │ │ + */ │ │ │ │ │ + updateAttribution: function() { │ │ │ │ │ + var attributions = []; │ │ │ │ │ + if (this.map && this.map.layers) { │ │ │ │ │ + for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if (layer.attribution && layer.getVisibility()) { │ │ │ │ │ + // add attribution only if attribution text is unique │ │ │ │ │ + if (OpenLayers.Util.indexOf( │ │ │ │ │ + attributions, layer.attribution) === -1) { │ │ │ │ │ + attributions.push(layer.attribution); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.div.innerHTML = OpenLayers.String.format(this.template, { │ │ │ │ │ + layers: attributions.join(this.separator) │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Attribution" │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Strategy.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Strategy │ │ │ │ │ + * Abstract vector layer strategy class. Not to be instantiated directly. Use │ │ │ │ │ + * one of the strategy subclasses instead. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Strategy = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} The layer this strategy belongs to. │ │ │ │ │ */ │ │ │ │ │ - layerStates: null, │ │ │ │ │ + layer: null, │ │ │ │ │ │ │ │ │ │ - // DOM Elements │ │ │ │ │ + /** │ │ │ │ │ + * Property: options │ │ │ │ │ + * {Object} Any options sent to the constructor. │ │ │ │ │ + */ │ │ │ │ │ + options: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: active │ │ │ │ │ + * {Boolean} The control is active. │ │ │ │ │ + */ │ │ │ │ │ + active: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: layersDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Property: autoActivate │ │ │ │ │ + * {Boolean} The creator of the strategy can set autoActivate to false │ │ │ │ │ + * to fully control when the protocol is activated and deactivated. │ │ │ │ │ + * Defaults to true. │ │ │ │ │ */ │ │ │ │ │ - layersDiv: null, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: baseLayersDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Property: autoDestroy │ │ │ │ │ + * {Boolean} The creator of the strategy can set autoDestroy to false │ │ │ │ │ + * to fully control when the strategy is destroyed. Defaults to │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ - baseLayersDiv: null, │ │ │ │ │ + autoDestroy: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: baseLayers │ │ │ │ │ - * {Array(Object)} │ │ │ │ │ + * Constructor: OpenLayers.Strategy │ │ │ │ │ + * Abstract class for vector strategies. Create instances of a subclass. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ */ │ │ │ │ │ - baseLayers: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.options = options; │ │ │ │ │ + // set the active property here, so that user cannot override it │ │ │ │ │ + this.active = false; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up the strategy. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.layer = null; │ │ │ │ │ + this.options = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: dataLbl │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Method: setLayer │ │ │ │ │ + * Called to set the <layer> property. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ */ │ │ │ │ │ - dataLbl: null, │ │ │ │ │ + setLayer: function(layer) { │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: dataLayersDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} True if the strategy was successfully activated or false if │ │ │ │ │ + * the strategy was already active. │ │ │ │ │ */ │ │ │ │ │ - dataLayersDiv: null, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + this.active = true; │ │ │ │ │ + return true; │ │ │ │ │ + } │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: dataLayers │ │ │ │ │ - * {Array(Object)} │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ + * tear-down. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} True if the strategy was successfully deactivated or false if │ │ │ │ │ + * the strategy was already inactive. │ │ │ │ │ */ │ │ │ │ │ - dataLayers: null, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.active = false; │ │ │ │ │ + return true; │ │ │ │ │ + } │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Filter/Spatial.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Filter.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Filter.Spatial │ │ │ │ │ + * This class represents a spatial filter. │ │ │ │ │ + * Currently implemented: BBOX, DWithin and Intersects │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Filter> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Filter.Spatial = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: minimizeDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * APIProperty: type │ │ │ │ │ + * {String} Type of spatial filter. │ │ │ │ │ + * │ │ │ │ │ + * The type should be one of: │ │ │ │ │ + * - OpenLayers.Filter.Spatial.BBOX │ │ │ │ │ + * - OpenLayers.Filter.Spatial.INTERSECTS │ │ │ │ │ + * - OpenLayers.Filter.Spatial.DWITHIN │ │ │ │ │ + * - OpenLayers.Filter.Spatial.WITHIN │ │ │ │ │ + * - OpenLayers.Filter.Spatial.CONTAINS │ │ │ │ │ */ │ │ │ │ │ - minimizeDiv: null, │ │ │ │ │ + type: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: maximizeDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * APIProperty: property │ │ │ │ │ + * {String} Name of the context property to compare. │ │ │ │ │ */ │ │ │ │ │ - maximizeDiv: null, │ │ │ │ │ + property: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: ascending │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * APIProperty: value │ │ │ │ │ + * {<OpenLayers.Bounds> || <OpenLayers.Geometry>} The bounds or geometry │ │ │ │ │ + * to be used by the filter. Use bounds for BBOX filters and geometry │ │ │ │ │ + * for INTERSECTS or DWITHIN filters. │ │ │ │ │ */ │ │ │ │ │ - ascending: true, │ │ │ │ │ + value: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.LayerSwitcher │ │ │ │ │ + * APIProperty: distance │ │ │ │ │ + * {Number} The distance to use in a DWithin spatial filter. │ │ │ │ │ + */ │ │ │ │ │ + distance: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: distanceUnits │ │ │ │ │ + * {String} The units to use for the distance, e.g. 'm'. │ │ │ │ │ + */ │ │ │ │ │ + distanceUnits: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Filter.Spatial │ │ │ │ │ + * Creates a spatial filter. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * options - {Object} An optional object with properties to set on the │ │ │ │ │ + * filter. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter.Spatial>} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.layerStates = []; │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: evaluate │ │ │ │ │ + * Evaluates this filter for a specific feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} feature to apply the filter to. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The feature meets filter criteria. │ │ │ │ │ + */ │ │ │ │ │ + evaluate: function(feature) { │ │ │ │ │ + var intersect = false; │ │ │ │ │ + switch (this.type) { │ │ │ │ │ + case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ + case OpenLayers.Filter.Spatial.INTERSECTS: │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + var geom = this.value; │ │ │ │ │ + if (this.value.CLASS_NAME == "OpenLayers.Bounds") { │ │ │ │ │ + geom = this.value.toGeometry(); │ │ │ │ │ + } │ │ │ │ │ + if (feature.geometry.intersects(geom)) { │ │ │ │ │ + intersect = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + throw new Error('evaluate is not implemented for this filter type.'); │ │ │ │ │ + } │ │ │ │ │ + return intersect; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Clones this filter. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter.Spatial>} Clone of this filter. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ + clone: function() { │ │ │ │ │ + var options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + value: this.value && this.value.clone && this.value.clone() │ │ │ │ │ + }, this); │ │ │ │ │ + return new OpenLayers.Filter.Spatial(options); │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Filter.Spatial" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ - //clear out layers info and unregister their events │ │ │ │ │ - this.clearLayersArray("base"); │ │ │ │ │ - this.clearLayersArray("data"); │ │ │ │ │ +OpenLayers.Filter.Spatial.BBOX = "BBOX"; │ │ │ │ │ +OpenLayers.Filter.Spatial.INTERSECTS = "INTERSECTS"; │ │ │ │ │ +OpenLayers.Filter.Spatial.DWITHIN = "DWITHIN"; │ │ │ │ │ +OpenLayers.Filter.Spatial.WITHIN = "WITHIN"; │ │ │ │ │ +OpenLayers.Filter.Spatial.CONTAINS = "CONTAINS"; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Strategy/BBOX.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - buttonclick: this.onButtonClick, │ │ │ │ │ - addlayer: this.redraw, │ │ │ │ │ - changelayer: this.redraw, │ │ │ │ │ - removelayer: this.redraw, │ │ │ │ │ - changebaselayer: this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Strategy.js │ │ │ │ │ + * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Strategy.BBOX │ │ │ │ │ + * A simple strategy that reads new features when the viewport invalidates │ │ │ │ │ + * some bounds. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Strategy> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Strategy.BBOX = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: bounds │ │ │ │ │ + * {<OpenLayers.Bounds>} The current data bounds (in the same projection │ │ │ │ │ + * as the layer - not always the same projection as the map). │ │ │ │ │ + */ │ │ │ │ │ + bounds: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: resolution │ │ │ │ │ + * {Float} The current data resolution. │ │ │ │ │ + */ │ │ │ │ │ + resolution: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: ratio │ │ │ │ │ + * {Float} The ratio of the data bounds to the viewport bounds (in each │ │ │ │ │ + * dimension). Default is 2. │ │ │ │ │ + */ │ │ │ │ │ + ratio: 2, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: resFactor │ │ │ │ │ + * {Float} Optional factor used to determine when previously requested │ │ │ │ │ + * features are invalid. If set, the resFactor will be compared to the │ │ │ │ │ + * resolution of the previous request to the current map resolution. │ │ │ │ │ + * If resFactor > (old / new) and 1/resFactor < (old / new). If you │ │ │ │ │ + * set a resFactor of 1, data will be requested every time the │ │ │ │ │ + * resolution changes. If you set a resFactor of 3, data will be │ │ │ │ │ + * requested if the old resolution is 3 times the new, or if the new is │ │ │ │ │ + * 3 times the old. If the old bounds do not contain the new bounds │ │ │ │ │ + * new data will always be requested (with or without considering │ │ │ │ │ + * resFactor). │ │ │ │ │ + */ │ │ │ │ │ + resFactor: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: response │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} The protocol response object returned │ │ │ │ │ + * by the layer protocol. │ │ │ │ │ + */ │ │ │ │ │ + response: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Strategy.BBOX │ │ │ │ │ + * Create a new BBOX strategy. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Set up strategy with regard to reading new batches of remote data. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The strategy was successfully activated. │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "moveend": this.update, │ │ │ │ │ + "refresh": this.update, │ │ │ │ │ + "visibilitychanged": this.update, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.update(); │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Tear down strategy with regard to reading new batches of remote data. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + "moveend": this.update, │ │ │ │ │ + "refresh": this.update, │ │ │ │ │ + "visibilitychanged": this.update, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: update │ │ │ │ │ + * Callback function called on "moveend" or "refresh" layer events. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will determine │ │ │ │ │ + * the behaviour of this Strategy │ │ │ │ │ + * │ │ │ │ │ + * Valid options include: │ │ │ │ │ + * force - {Boolean} if true, new data must be unconditionally read. │ │ │ │ │ + * noAbort - {Boolean} if true, do not abort previous requests. │ │ │ │ │ + */ │ │ │ │ │ + update: function(options) { │ │ │ │ │ + var mapBounds = this.getMapBounds(); │ │ │ │ │ + if (mapBounds !== null && ((options && options.force) || │ │ │ │ │ + (this.layer.visibility && this.layer.calculateInRange() && this.invalidBounds(mapBounds)))) { │ │ │ │ │ + this.calculateBounds(mapBounds); │ │ │ │ │ + this.resolution = this.layer.map.getResolution(); │ │ │ │ │ + this.triggerRead(options); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getMapBounds │ │ │ │ │ + * Get the map bounds expressed in the same projection as this layer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} Map bounds in the projection of the layer. │ │ │ │ │ + */ │ │ │ │ │ + getMapBounds: function() { │ │ │ │ │ + if (this.layer.map === null) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + var bounds = this.layer.map.getExtent(); │ │ │ │ │ + if (bounds && !this.layer.projection.equals( │ │ │ │ │ + this.layer.map.getProjectionObject())) { │ │ │ │ │ + bounds = bounds.clone().transform( │ │ │ │ │ + this.layer.map.getProjectionObject(), this.layer.projection │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return bounds; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: invalidBounds │ │ │ │ │ + * Determine whether the previously requested set of features is invalid. │ │ │ │ │ + * This occurs when the new map bounds do not contain the previously │ │ │ │ │ + * requested bounds. In addition, if <resFactor> is set, it will be │ │ │ │ │ + * considered. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * mapBounds - {<OpenLayers.Bounds>} the current map extent, will be │ │ │ │ │ + * retrieved from the map object if not provided │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + invalidBounds: function(mapBounds) { │ │ │ │ │ + if (!mapBounds) { │ │ │ │ │ + mapBounds = this.getMapBounds(); │ │ │ │ │ + } │ │ │ │ │ + var invalid = !this.bounds || !this.bounds.containsBounds(mapBounds); │ │ │ │ │ + if (!invalid && this.resFactor) { │ │ │ │ │ + var ratio = this.resolution / this.layer.map.getResolution(); │ │ │ │ │ + invalid = (ratio >= this.resFactor || ratio <= (1 / this.resFactor)); │ │ │ │ │ + } │ │ │ │ │ + return invalid; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateBounds │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * mapBounds - {<OpenLayers.Bounds>} the current map extent, will be │ │ │ │ │ + * retrieved from the map object if not provided │ │ │ │ │ + */ │ │ │ │ │ + calculateBounds: function(mapBounds) { │ │ │ │ │ + if (!mapBounds) { │ │ │ │ │ + mapBounds = this.getMapBounds(); │ │ │ │ │ + } │ │ │ │ │ + var center = mapBounds.getCenterLonLat(); │ │ │ │ │ + var dataWidth = mapBounds.getWidth() * this.ratio; │ │ │ │ │ + var dataHeight = mapBounds.getHeight() * this.ratio; │ │ │ │ │ + this.bounds = new OpenLayers.Bounds( │ │ │ │ │ + center.lon - (dataWidth / 2), │ │ │ │ │ + center.lat - (dataHeight / 2), │ │ │ │ │ + center.lon + (dataWidth / 2), │ │ │ │ │ + center.lat + (dataHeight / 2) │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: triggerRead │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Additional options for the protocol's read method │ │ │ │ │ + * (optional) │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} The protocol response object │ │ │ │ │ + * returned by the layer protocol. │ │ │ │ │ + */ │ │ │ │ │ + triggerRead: function(options) { │ │ │ │ │ + if (this.response && !(options && options.noAbort === true)) { │ │ │ │ │ + this.layer.protocol.abort(this.response); │ │ │ │ │ + this.layer.events.triggerEvent("loadend"); │ │ │ │ │ + } │ │ │ │ │ + var evt = { │ │ │ │ │ + filter: this.createFilter() │ │ │ │ │ + }; │ │ │ │ │ + this.layer.events.triggerEvent("loadstart", evt); │ │ │ │ │ + this.response = this.layer.protocol.read( │ │ │ │ │ + OpenLayers.Util.applyDefaults({ │ │ │ │ │ + filter: evt.filter, │ │ │ │ │ + callback: this.merge, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options)); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createFilter │ │ │ │ │ + * Creates a spatial BBOX filter. If the layer that this strategy belongs │ │ │ │ │ + * to has a filter property, this filter will be combined with the BBOX │ │ │ │ │ + * filter. │ │ │ │ │ + * │ │ │ │ │ + * Returns │ │ │ │ │ + * {<OpenLayers.Filter>} The filter object. │ │ │ │ │ + */ │ │ │ │ │ + createFilter: function() { │ │ │ │ │ + var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ + value: this.bounds, │ │ │ │ │ + projection: this.layer.projection │ │ │ │ │ }); │ │ │ │ │ - this.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ + if (this.layer.filter) { │ │ │ │ │ + filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ + type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ + filters: [this.layer.filter, filter] │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return filter; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + /** │ │ │ │ │ + * Method: merge │ │ │ │ │ + * Given a list of features, determine which ones to add to the layer. │ │ │ │ │ + * If the layer projection differs from the map projection, features │ │ │ │ │ + * will be transformed from the layer projection to the map projection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object passed │ │ │ │ │ + * by the protocol. │ │ │ │ │ + */ │ │ │ │ │ + merge: function(resp) { │ │ │ │ │ + this.layer.destroyFeatures(); │ │ │ │ │ + if (resp.success()) { │ │ │ │ │ + var features = resp.features; │ │ │ │ │ + if (features && features.length > 0) { │ │ │ │ │ + var remote = this.layer.projection; │ │ │ │ │ + var local = this.layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var geom; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + geom = features[i].geometry; │ │ │ │ │ + if (geom) { │ │ │ │ │ + geom.transform(remote, local); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.layer.addFeatures(features); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this.bounds = null; │ │ │ │ │ + } │ │ │ │ │ + this.response = null; │ │ │ │ │ + this.layer.events.triggerEvent("loadend", { │ │ │ │ │ + response: resp │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.BBOX" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Strategy/Fixed.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Strategy.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Strategy.Fixed │ │ │ │ │ + * A simple strategy that requests features once and never requests new data. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Strategy> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Strategy.Fixed = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ + * APIProperty: preload │ │ │ │ │ + * {Boolean} Load data before layer made visible. Enabling this may result │ │ │ │ │ + * in considerable overhead if your application loads many data layers │ │ │ │ │ + * that are not visible by default. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + preload: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Strategy.Fixed │ │ │ │ │ + * Create a new Fixed strategy. │ │ │ │ │ * │ │ │ │ │ - * Properties: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - addlayer: this.redraw, │ │ │ │ │ - changelayer: this.redraw, │ │ │ │ │ - removelayer: this.redraw, │ │ │ │ │ - changebaselayer: this.redraw, │ │ │ │ │ + /** │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activate the strategy: load data or add listener to load when visible │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} True if the strategy was successfully activated or false if │ │ │ │ │ + * the strategy was already active. │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "refresh": this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + if (this.layer.visibility == true || this.preload) { │ │ │ │ │ + this.load(); │ │ │ │ │ + } else { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "visibilitychanged": this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivate the strategy. Undo what is done in <activate>. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + "refresh": this.load, │ │ │ │ │ + "visibilitychanged": this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: load │ │ │ │ │ + * Tells protocol to load data and unhooks the visibilitychanged event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} options to pass to protocol read. │ │ │ │ │ + */ │ │ │ │ │ + load: function(options) { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + layer.events.triggerEvent("loadstart", { │ │ │ │ │ + filter: layer.filter │ │ │ │ │ + }); │ │ │ │ │ + layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: this.merge, │ │ │ │ │ + filter: layer.filter, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options)); │ │ │ │ │ + layer.events.un({ │ │ │ │ │ + "visibilitychanged": this.load, │ │ │ │ │ scope: this │ │ │ │ │ }); │ │ │ │ │ - if (this.outsideViewport) { │ │ │ │ │ - this.events.attachToElement(this.div); │ │ │ │ │ - this.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: merge │ │ │ │ │ + * Add all features to the layer. │ │ │ │ │ + * If the layer projection differs from the map projection, features │ │ │ │ │ + * will be transformed from the layer projection to the map projection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object passed │ │ │ │ │ + * by the protocol. │ │ │ │ │ + */ │ │ │ │ │ + merge: function(resp) { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + layer.destroyFeatures(); │ │ │ │ │ + var features = resp.features; │ │ │ │ │ + if (features && features.length > 0) { │ │ │ │ │ + var remote = layer.projection; │ │ │ │ │ + var local = layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var geom; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + geom = features[i].geometry; │ │ │ │ │ + if (geom) { │ │ │ │ │ + geom.transform(remote, local); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + layer.addFeatures(features); │ │ │ │ │ + } │ │ │ │ │ + layer.events.triggerEvent("loadend", { │ │ │ │ │ + response: resp │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Fixed" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/HTTPRequest.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.HTTPRequest │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.HTTPRequest = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: URL_HASH_FACTOR │ │ │ │ │ + * {Float} Used to hash URL param strings for multi-WMS server selection. │ │ │ │ │ + * Set to the Golden Ratio per Knuth's recommendation. │ │ │ │ │ + */ │ │ │ │ │ + URL_HASH_FACTOR: (Math.sqrt(5) - 1) / 2, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: url │ │ │ │ │ + * {Array(String) or String} This is either an array of url strings or │ │ │ │ │ + * a single url string. │ │ │ │ │ + */ │ │ │ │ │ + url: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: params │ │ │ │ │ + * {Object} Hashtable of key/value parameters │ │ │ │ │ + */ │ │ │ │ │ + params: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: reproject │ │ │ │ │ + * *Deprecated*. See http://docs.openlayers.org/library/spherical_mercator.html │ │ │ │ │ + * for information on the replacement for this functionality. │ │ │ │ │ + * {Boolean} Whether layer should reproject itself based on base layer │ │ │ │ │ + * locations. This allows reprojection onto commercial layers. │ │ │ │ │ + * Default is false: Most layers can't reproject, but layers │ │ │ │ │ + * which can create non-square geographic pixels can, like WMS. │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + reproject: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.HTTPRequest │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} │ │ │ │ │ + * url - {Array(String) or String} │ │ │ │ │ + * params - {Object} │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ + this.url = url; │ │ │ │ │ + if (!this.params) { │ │ │ │ │ + this.params = OpenLayers.Util.extend({}, params); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.url = null; │ │ │ │ │ + this.params = null; │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.HTTPRequest>} An exact clone of this │ │ │ │ │ + * <OpenLayers.Layer.HTTPRequest> │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.HTTPRequest(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.params, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ + │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setUrl │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newUrl - {String} │ │ │ │ │ + */ │ │ │ │ │ + setUrl: function(newUrl) { │ │ │ │ │ + this.url = newUrl; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: mergeNewParams │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newParams - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * redrawn: {Boolean} whether the layer was actually redrawn. │ │ │ │ │ + */ │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + this.params = OpenLayers.Util.extend(this.params, newParams); │ │ │ │ │ + var ret = this.redraw(); │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "params" │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return ret; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: redraw │ │ │ │ │ + * Redraws the layer. Returns true if the layer was redrawn, false if not. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * force - {Boolean} Force redraw by adding random parameter. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The layer was redrawn. │ │ │ │ │ + */ │ │ │ │ │ + redraw: function(force) { │ │ │ │ │ + if (force) { │ │ │ │ │ + return this.mergeNewParams({ │ │ │ │ │ + "_olSalt": Math.random() │ │ │ │ │ + }); │ │ │ │ │ } else { │ │ │ │ │ - this.map.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ + return OpenLayers.Layer.prototype.redraw.apply(this, []); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: selectUrl │ │ │ │ │ + * selectUrl() implements the standard floating-point multiplicative │ │ │ │ │ + * hash function described by Knuth, and hashes the contents of the │ │ │ │ │ + * given param string into a float between 0 and 1. This float is then │ │ │ │ │ + * scaled to the size of the provided urls array, and used to select │ │ │ │ │ + * a URL. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * paramString - {String} │ │ │ │ │ + * urls - {Array(String)} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} An entry from the urls array, deterministically selected based │ │ │ │ │ + * on the paramString. │ │ │ │ │ + */ │ │ │ │ │ + selectUrl: function(paramString, urls) { │ │ │ │ │ + var product = 1; │ │ │ │ │ + for (var i = 0, len = paramString.length; i < len; i++) { │ │ │ │ │ + product *= paramString.charCodeAt(i) * this.URL_HASH_FACTOR; │ │ │ │ │ + product -= Math.floor(product); │ │ │ │ │ + } │ │ │ │ │ + return urls[Math.floor(product * urls.length)]; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getFullRequestString │ │ │ │ │ + * Combine url with layer's params and these newParams. │ │ │ │ │ + * │ │ │ │ │ + * does checking on the serverPath variable, allowing for cases when it │ │ │ │ │ + * is supplied with trailing ? or &, as well as cases where not. │ │ │ │ │ + * │ │ │ │ │ + * return in formatted string like this: │ │ │ │ │ + * "server?key1=value1&key2=value2&key3=value3" │ │ │ │ │ + * │ │ │ │ │ + * WARNING: The altUrl parameter is deprecated and will be removed in 3.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newParams - {Object} │ │ │ │ │ + * altUrl - {String} Use this as the url instead of the layer's url │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ + │ │ │ │ │ + // if not altUrl passed in, use layer's url │ │ │ │ │ + var url = altUrl || this.url; │ │ │ │ │ + │ │ │ │ │ + // create a new params hashtable with all the layer params and the │ │ │ │ │ + // new params together. then convert to string │ │ │ │ │ + var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ + allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ + var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + │ │ │ │ │ + // if url is not a string, it should be an array of strings, │ │ │ │ │ + // in which case we will deterministically select one of them in │ │ │ │ │ + // order to evenly distribute requests to different urls. │ │ │ │ │ + // │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(paramsString, url); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // ignore parameters that are already in the url search string │ │ │ │ │ + var urlParams = │ │ │ │ │ + OpenLayers.Util.upperCaseObject(OpenLayers.Util.getParameters(url)); │ │ │ │ │ + for (var key in allParams) { │ │ │ │ │ + if (key.toUpperCase() in urlParams) { │ │ │ │ │ + delete allParams[key]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Util.urlAppend(url, paramsString); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.HTTPRequest" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Tile.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Tile │ │ │ │ │ + * This is a class designed to designate a single tile, however │ │ │ │ │ + * it is explicitly designed to do relatively little. Tiles store │ │ │ │ │ + * information about themselves -- such as the URL that they are related │ │ │ │ │ + * to, and their size - but do not add themselves to the layer div │ │ │ │ │ + * automatically, for example. Create a new tile with the │ │ │ │ │ + * <OpenLayers.Tile> constructor, or a subclass. │ │ │ │ │ + * │ │ │ │ │ + * TBD 3.0 - remove reference to url in above paragraph │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Tile = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} An events object that handles all │ │ │ │ │ + * events on the tile. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * tile.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types: │ │ │ │ │ + * beforedraw - Triggered before the tile is drawn. Used to defer │ │ │ │ │ + * drawing to an animation queue. To defer drawing, listeners need │ │ │ │ │ + * to return false, which will abort drawing. The queue handler needs │ │ │ │ │ + * to call <draw>(true) to actually draw the tile. │ │ │ │ │ + * loadstart - Triggered when tile loading starts. │ │ │ │ │ + * loadend - Triggered when tile loading ends. │ │ │ │ │ + * loaderror - Triggered before the loadend event (i.e. when the tile is │ │ │ │ │ + * still hidden) if the tile could not be loaded. │ │ │ │ │ + * reload - Triggered when an already loading tile is reloaded. │ │ │ │ │ + * unload - Triggered before a tile is unloaded. │ │ │ │ │ + */ │ │ │ │ │ + events: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: eventListeners │ │ │ │ │ + * {Object} If set as an option at construction, the eventListeners │ │ │ │ │ + * object will be registered with <OpenLayers.Events.on>. Object │ │ │ │ │ + * structure must be a listeners object as shown in the example for │ │ │ │ │ + * the events.on method. │ │ │ │ │ + * │ │ │ │ │ + * This options can be set in the ``tileOptions`` option from │ │ │ │ │ + * <OpenLayers.Layer.Grid>. For example, to be notified of the │ │ │ │ │ + * ``loadend`` event of each tiles: │ │ │ │ │ + * (code) │ │ │ │ │ + * new OpenLayers.Layer.OSM('osm', 'http://tile.openstreetmap.org/${z}/${x}/${y}.png', { │ │ │ │ │ + * tileOptions: { │ │ │ │ │ + * eventListeners: { │ │ │ │ │ + * 'loadend': function(evt) { │ │ │ │ │ + * // do something on loadend │ │ │ │ │ + * } │ │ │ │ │ + * } │ │ │ │ │ + * } │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ + eventListeners: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: id │ │ │ │ │ + * {String} null │ │ │ │ │ + */ │ │ │ │ │ + id: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {<OpenLayers.Layer>} layer the tile is attached to │ │ │ │ │ + */ │ │ │ │ │ + layer: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: url │ │ │ │ │ + * {String} url of the request. │ │ │ │ │ + * │ │ │ │ │ + * TBD 3.0 │ │ │ │ │ + * Deprecated. The base tile class does not need an url. This should be │ │ │ │ │ + * handled in subclasses. Does not belong here. │ │ │ │ │ + */ │ │ │ │ │ + url: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: bounds │ │ │ │ │ + * {<OpenLayers.Bounds>} null │ │ │ │ │ + */ │ │ │ │ │ + bounds: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: size │ │ │ │ │ + * {<OpenLayers.Size>} null │ │ │ │ │ + */ │ │ │ │ │ + size: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: position │ │ │ │ │ + * {<OpenLayers.Pixel>} Top Left pixel of the tile │ │ │ │ │ + */ │ │ │ │ │ + position: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: isLoading │ │ │ │ │ + * {Boolean} Is the tile loading? │ │ │ │ │ + */ │ │ │ │ │ + isLoading: false, │ │ │ │ │ + │ │ │ │ │ + /** TBD 3.0 -- remove 'url' from the list of parameters to the constructor. │ │ │ │ │ + * there is no need for the base tile class to have a url. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Tile │ │ │ │ │ + * Constructor for a new <OpenLayers.Tile> instance. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer>} layer that the tile will go in. │ │ │ │ │ + * position - {<OpenLayers.Pixel>} │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * url - {<String>} │ │ │ │ │ + * size - {<OpenLayers.Size>} │ │ │ │ │ + * options - {Object} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(layer, position, bounds, url, size, options) { │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + this.position = position.clone(); │ │ │ │ │ + this.setBounds(bounds); │ │ │ │ │ + this.url = url; │ │ │ │ │ + if (size) { │ │ │ │ │ + this.size = size.clone(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //give the tile a unique id based on its BBOX. │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID("Tile_"); │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + │ │ │ │ │ + this.events = new OpenLayers.Events(this); │ │ │ │ │ + if (this.eventListeners instanceof Object) { │ │ │ │ │ + this.events.on(this.eventListeners); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: unload │ │ │ │ │ + * Call immediately before destroying if you are listening to tile │ │ │ │ │ + * events, so that counters are properly handled if tile is still │ │ │ │ │ + * loading at destroy-time. Will only fire an event if the tile is │ │ │ │ │ + * still loading. │ │ │ │ │ + */ │ │ │ │ │ + unload: function() { │ │ │ │ │ + if (this.isLoading) { │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + this.events.triggerEvent("unload"); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Nullify references to prevent circular references and memory leaks. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.layer = null; │ │ │ │ │ + this.bounds = null; │ │ │ │ │ + this.size = null; │ │ │ │ │ + this.position = null; │ │ │ │ │ + │ │ │ │ │ + if (this.eventListeners) { │ │ │ │ │ + this.events.un(this.eventListeners); │ │ │ │ │ } │ │ │ │ │ + this.events.destroy(); │ │ │ │ │ + this.eventListeners = null; │ │ │ │ │ + this.events = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Method: draw │ │ │ │ │ + * Clear whatever is currently in the tile, then return whether or not │ │ │ │ │ + * it should actually be re-drawn. This is an example implementation │ │ │ │ │ + * that can be overridden by subclasses. The minimum thing to do here │ │ │ │ │ + * is to call <clear> and return the result from <shouldDraw>. │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * force - {Boolean} If true, the tile will not be cleared and no beforedraw │ │ │ │ │ + * event will be fired. This is used for drawing tiles asynchronously │ │ │ │ │ + * after drawing has been cancelled by returning false from a beforedraw │ │ │ │ │ + * listener. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} A reference to the DIV DOMElement containing the │ │ │ │ │ - * switcher tabs. │ │ │ │ │ + * {Boolean} Whether or not the tile should actually be drawn. Returns null │ │ │ │ │ + * if a beforedraw listener returned false. │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this); │ │ │ │ │ + draw: function(force) { │ │ │ │ │ + if (!force) { │ │ │ │ │ + //clear tile's contents and mark as not drawn │ │ │ │ │ + this.clear(); │ │ │ │ │ + } │ │ │ │ │ + var draw = this.shouldDraw(); │ │ │ │ │ + if (draw && !force && this.events.triggerEvent("beforedraw") === false) { │ │ │ │ │ + draw = null; │ │ │ │ │ + } │ │ │ │ │ + return draw; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // create layout divs │ │ │ │ │ - this.loadContents(); │ │ │ │ │ + /** │ │ │ │ │ + * Method: shouldDraw │ │ │ │ │ + * Return whether or not the tile should actually be (re-)drawn. The only │ │ │ │ │ + * case where we *wouldn't* want to draw the tile is if the tile is outside │ │ │ │ │ + * its layer's maxExtent │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the tile should actually be drawn. │ │ │ │ │ + */ │ │ │ │ │ + shouldDraw: function() { │ │ │ │ │ + var withinMaxExtent = false, │ │ │ │ │ + maxExtent = this.layer.maxExtent; │ │ │ │ │ + if (maxExtent) { │ │ │ │ │ + var map = this.layer.map; │ │ │ │ │ + var worldBounds = map.baseLayer.wrapDateLine && map.getMaxExtent(); │ │ │ │ │ + if (this.bounds.intersectsBounds(maxExtent, { │ │ │ │ │ + inclusive: false, │ │ │ │ │ + worldBounds: worldBounds │ │ │ │ │ + })) { │ │ │ │ │ + withinMaxExtent = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // set mode to minimize │ │ │ │ │ - if (!this.outsideViewport) { │ │ │ │ │ - this.minimizeControl(); │ │ │ │ │ + return withinMaxExtent || this.layer.displayOutsideMaxExtent; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setBounds │ │ │ │ │ + * Sets the bounds on this instance │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds {<OpenLayers.Bounds>} │ │ │ │ │ + */ │ │ │ │ │ + setBounds: function(bounds) { │ │ │ │ │ + bounds = bounds.clone(); │ │ │ │ │ + if (this.layer.map.baseLayer.wrapDateLine) { │ │ │ │ │ + var worldExtent = this.layer.map.getMaxExtent(), │ │ │ │ │ + tolerance = this.layer.map.getResolution(); │ │ │ │ │ + bounds = bounds.wrapDateLine(worldExtent, { │ │ │ │ │ + leftTolerance: tolerance, │ │ │ │ │ + rightTolerance: tolerance │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ + this.bounds = bounds; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // populate div with current info │ │ │ │ │ - this.redraw(); │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * Reposition the tile. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * position - {<OpenLayers.Pixel>} │ │ │ │ │ + * redraw - {Boolean} Call draw method on tile after moving. │ │ │ │ │ + * Default is true │ │ │ │ │ + */ │ │ │ │ │ + moveTo: function(bounds, position, redraw) { │ │ │ │ │ + if (redraw == null) { │ │ │ │ │ + redraw = true; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - return this.div; │ │ │ │ │ + this.setBounds(bounds); │ │ │ │ │ + this.position = position.clone(); │ │ │ │ │ + if (redraw) { │ │ │ │ │ + this.draw(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clear │ │ │ │ │ + * Clear the tile of any bounds/position-related data so that it can │ │ │ │ │ + * be reused in a new location. │ │ │ │ │ + */ │ │ │ │ │ + clear: function(draw) { │ │ │ │ │ + // to be extended by subclasses │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Tile" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Tile/Image.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Tile.js │ │ │ │ │ + * @requires OpenLayers/Animation.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Tile.Image │ │ │ │ │ + * Instances of OpenLayers.Tile.Image are used to manage the image tiles │ │ │ │ │ + * used by various layers. Create a new image tile with the │ │ │ │ │ + * <OpenLayers.Tile.Image> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Tile> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Tile.Image = OpenLayers.Class(OpenLayers.Tile, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: onButtonClick │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} An events object that handles all │ │ │ │ │ + * events on the tile. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * tile.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to the <OpenLayers.Tile> events): │ │ │ │ │ + * beforeload - Triggered before an image is prepared for loading, when the │ │ │ │ │ + * url for the image is known already. Listeners may call <setImage> on │ │ │ │ │ + * the tile instance. If they do so, that image will be used and no new │ │ │ │ │ + * one will be created. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String} The URL of the image being requested. No default. Filled in by │ │ │ │ │ + * layer.getURL() function. May be modified by loadstart listeners. │ │ │ │ │ + */ │ │ │ │ │ + url: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: imgDiv │ │ │ │ │ + * {HTMLImageElement} The image for this tile. │ │ │ │ │ + */ │ │ │ │ │ + imgDiv: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: frame │ │ │ │ │ + * {DOMElement} The image element is appended to the frame. Any gutter on │ │ │ │ │ + * the image will be hidden behind the frame. If no gutter is set, │ │ │ │ │ + * this will be null. │ │ │ │ │ + */ │ │ │ │ │ + frame: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: imageReloadAttempts │ │ │ │ │ + * {Integer} Attempts to load the image. │ │ │ │ │ + */ │ │ │ │ │ + imageReloadAttempts: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: layerAlphaHack │ │ │ │ │ + * {Boolean} True if the png alpha hack needs to be applied on the layer's div. │ │ │ │ │ + */ │ │ │ │ │ + layerAlphaHack: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: asyncRequestId │ │ │ │ │ + * {Integer} ID of an request to see if request is still valid. This is a │ │ │ │ │ + * number which increments by 1 for each asynchronous request. │ │ │ │ │ + */ │ │ │ │ │ + asyncRequestId: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxGetUrlLength │ │ │ │ │ + * {Number} If set, requests that would result in GET urls with more │ │ │ │ │ + * characters than the number provided will be made using form-encoded │ │ │ │ │ + * HTTP POST. It is good practice to avoid urls that are longer than 2048 │ │ │ │ │ + * characters. │ │ │ │ │ * │ │ │ │ │ + * Caution: │ │ │ │ │ + * Older versions of Gecko based browsers (e.g. Firefox < 3.5) and most │ │ │ │ │ + * Opera versions do not fully support this option. On all browsers, │ │ │ │ │ + * transition effects are not supported if POST requests are used. │ │ │ │ │ + */ │ │ │ │ │ + maxGetUrlLength: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: canvasContext │ │ │ │ │ + * {CanvasRenderingContext2D} A canvas context associated with │ │ │ │ │ + * the tile image. │ │ │ │ │ + */ │ │ │ │ │ + canvasContext: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: crossOriginKeyword │ │ │ │ │ + * The value of the crossorigin keyword to use when loading images. This is │ │ │ │ │ + * only relevant when using <getCanvasContext> for tiles from remote │ │ │ │ │ + * origins and should be set to either 'anonymous' or 'use-credentials' │ │ │ │ │ + * for servers that send Access-Control-Allow-Origin headers with their │ │ │ │ │ + * tiles. │ │ │ │ │ + */ │ │ │ │ │ + crossOriginKeyword: null, │ │ │ │ │ + │ │ │ │ │ + /** TBD 3.0 - reorder the parameters to the init function to remove │ │ │ │ │ + * URL. the getUrl() function on the layer gets called on │ │ │ │ │ + * each draw(), so no need to specify it here. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Tile.Image │ │ │ │ │ + * Constructor for a new <OpenLayers.Tile.Image> instance. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * layer - {<OpenLayers.Layer>} layer that the tile will go in. │ │ │ │ │ + * position - {<OpenLayers.Pixel>} │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * url - {<String>} Deprecated. Remove me in 3.0. │ │ │ │ │ + * size - {<OpenLayers.Size>} │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - var button = evt.buttonElement; │ │ │ │ │ - if (button === this.minimizeDiv) { │ │ │ │ │ - this.minimizeControl(); │ │ │ │ │ - } else if (button === this.maximizeDiv) { │ │ │ │ │ - this.maximizeControl(); │ │ │ │ │ - } else if (button._layerSwitcher === this.id) { │ │ │ │ │ - if (button["for"]) { │ │ │ │ │ - button = document.getElementById(button["for"]); │ │ │ │ │ + initialize: function(layer, position, bounds, url, size, options) { │ │ │ │ │ + OpenLayers.Tile.prototype.initialize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + this.url = url; //deprecated remove me │ │ │ │ │ + │ │ │ │ │ + this.layerAlphaHack = this.layer.alpha && OpenLayers.Util.alphaHack(); │ │ │ │ │ + │ │ │ │ │ + if (this.maxGetUrlLength != null || this.layer.gutter || this.layerAlphaHack) { │ │ │ │ │ + // only create frame if it's needed │ │ │ │ │ + this.frame = document.createElement("div"); │ │ │ │ │ + this.frame.style.position = "absolute"; │ │ │ │ │ + this.frame.style.overflow = "hidden"; │ │ │ │ │ + } │ │ │ │ │ + if (this.maxGetUrlLength != null) { │ │ │ │ │ + OpenLayers.Util.extend(this, OpenLayers.Tile.Image.IFrame); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * nullify references to prevent circular references and memory leaks │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.imgDiv) { │ │ │ │ │ + this.clear(); │ │ │ │ │ + this.imgDiv = null; │ │ │ │ │ + this.frame = null; │ │ │ │ │ + } │ │ │ │ │ + // don't handle async requests any more │ │ │ │ │ + this.asyncRequestId = null; │ │ │ │ │ + OpenLayers.Tile.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Check that a tile should be drawn, and draw it. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Was a tile drawn? Or null if a beforedraw listener returned │ │ │ │ │ + * false. │ │ │ │ │ + */ │ │ │ │ │ + draw: function() { │ │ │ │ │ + var shouldDraw = OpenLayers.Tile.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (shouldDraw) { │ │ │ │ │ + // The layer's reproject option is deprecated. │ │ │ │ │ + if (this.layer != this.layer.map.baseLayer && this.layer.reproject) { │ │ │ │ │ + // getBoundsFromBaseLayer is defined in deprecated.js. │ │ │ │ │ + this.bounds = this.getBoundsFromBaseLayer(this.position); │ │ │ │ │ } │ │ │ │ │ - if (!button.disabled) { │ │ │ │ │ - if (button.type == "radio") { │ │ │ │ │ - button.checked = true; │ │ │ │ │ - this.map.setBaseLayer(this.map.getLayer(button._layer)); │ │ │ │ │ + if (this.isLoading) { │ │ │ │ │ + //if we're already loading, send 'reload' instead of 'loadstart'. │ │ │ │ │ + this._loadEvent = "reload"; │ │ │ │ │ + } else { │ │ │ │ │ + this.isLoading = true; │ │ │ │ │ + this._loadEvent = "loadstart"; │ │ │ │ │ + } │ │ │ │ │ + this.renderTile(); │ │ │ │ │ + this.positionTile(); │ │ │ │ │ + } else if (shouldDraw === false) { │ │ │ │ │ + this.unload(); │ │ │ │ │ + } │ │ │ │ │ + return shouldDraw; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: renderTile │ │ │ │ │ + * Internal function to actually initialize the image tile, │ │ │ │ │ + * position it correctly, and set its url. │ │ │ │ │ + */ │ │ │ │ │ + renderTile: function() { │ │ │ │ │ + if (this.layer.async) { │ │ │ │ │ + // Asynchronous image requests call the asynchronous getURL method │ │ │ │ │ + // on the layer to fetch an image that covers 'this.bounds'. │ │ │ │ │ + var id = this.asyncRequestId = (this.asyncRequestId || 0) + 1; │ │ │ │ │ + this.layer.getURLasync(this.bounds, function(url) { │ │ │ │ │ + if (id == this.asyncRequestId) { │ │ │ │ │ + this.url = url; │ │ │ │ │ + this.initImage(); │ │ │ │ │ + } │ │ │ │ │ + }, this); │ │ │ │ │ + } else { │ │ │ │ │ + // synchronous image requests get the url immediately. │ │ │ │ │ + this.url = this.layer.getURL(this.bounds); │ │ │ │ │ + this.initImage(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: positionTile │ │ │ │ │ + * Using the properties currenty set on the layer, position the tile correctly. │ │ │ │ │ + * This method is used both by the async and non-async versions of the Tile.Image │ │ │ │ │ + * code. │ │ │ │ │ + */ │ │ │ │ │ + positionTile: function() { │ │ │ │ │ + var style = this.getTile().style, │ │ │ │ │ + size = this.frame ? this.size : │ │ │ │ │ + this.layer.getImageSize(this.bounds), │ │ │ │ │ + ratio = 1; │ │ │ │ │ + if (this.layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ + ratio = this.layer.getServerResolution() / this.layer.map.getResolution(); │ │ │ │ │ + } │ │ │ │ │ + style.left = this.position.x + "px"; │ │ │ │ │ + style.top = this.position.y + "px"; │ │ │ │ │ + style.width = Math.round(ratio * size.w) + "px"; │ │ │ │ │ + style.height = Math.round(ratio * size.h) + "px"; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clear │ │ │ │ │ + * Remove the tile from the DOM, clear it of any image related data so that │ │ │ │ │ + * it can be reused in a new location. │ │ │ │ │ + */ │ │ │ │ │ + clear: function() { │ │ │ │ │ + OpenLayers.Tile.prototype.clear.apply(this, arguments); │ │ │ │ │ + var img = this.imgDiv; │ │ │ │ │ + if (img) { │ │ │ │ │ + var tile = this.getTile(); │ │ │ │ │ + if (tile.parentNode === this.layer.div) { │ │ │ │ │ + this.layer.div.removeChild(tile); │ │ │ │ │ + } │ │ │ │ │ + this.setImgSrc(); │ │ │ │ │ + if (this.layerAlphaHack === true) { │ │ │ │ │ + img.style.filter = ""; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.removeClass(img, "olImageLoadError"); │ │ │ │ │ + } │ │ │ │ │ + this.canvasContext = null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getImage │ │ │ │ │ + * Returns or creates and returns the tile image. │ │ │ │ │ + */ │ │ │ │ │ + getImage: function() { │ │ │ │ │ + if (!this.imgDiv) { │ │ │ │ │ + this.imgDiv = OpenLayers.Tile.Image.IMAGE.cloneNode(false); │ │ │ │ │ + │ │ │ │ │ + var style = this.imgDiv.style; │ │ │ │ │ + if (this.frame) { │ │ │ │ │ + var left = 0, │ │ │ │ │ + top = 0; │ │ │ │ │ + if (this.layer.gutter) { │ │ │ │ │ + left = this.layer.gutter / this.layer.tileSize.w * 100; │ │ │ │ │ + top = this.layer.gutter / this.layer.tileSize.h * 100; │ │ │ │ │ + } │ │ │ │ │ + style.left = -left + "%"; │ │ │ │ │ + style.top = -top + "%"; │ │ │ │ │ + style.width = (2 * left + 100) + "%"; │ │ │ │ │ + style.height = (2 * top + 100) + "%"; │ │ │ │ │ + } │ │ │ │ │ + style.visibility = "hidden"; │ │ │ │ │ + style.opacity = 0; │ │ │ │ │ + if (this.layer.opacity < 1) { │ │ │ │ │ + style.filter = 'alpha(opacity=' + │ │ │ │ │ + (this.layer.opacity * 100) + │ │ │ │ │ + ')'; │ │ │ │ │ + } │ │ │ │ │ + style.position = "absolute"; │ │ │ │ │ + if (this.layerAlphaHack) { │ │ │ │ │ + // move the image out of sight │ │ │ │ │ + style.paddingTop = style.height; │ │ │ │ │ + style.height = "0"; │ │ │ │ │ + style.width = "100%"; │ │ │ │ │ + } │ │ │ │ │ + if (this.frame) { │ │ │ │ │ + this.frame.appendChild(this.imgDiv); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return this.imgDiv; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setImage │ │ │ │ │ + * Sets the image element for this tile. This method should only be called │ │ │ │ │ + * from beforeload listeners. │ │ │ │ │ + * │ │ │ │ │ + * Parameters │ │ │ │ │ + * img - {HTMLImageElement} The image to use for this tile. │ │ │ │ │ + */ │ │ │ │ │ + setImage: function(img) { │ │ │ │ │ + this.imgDiv = img; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: initImage │ │ │ │ │ + * Creates the content for the frame on the tile. │ │ │ │ │ + */ │ │ │ │ │ + initImage: function() { │ │ │ │ │ + if (!this.url && !this.imgDiv) { │ │ │ │ │ + // fast path out - if there is no tile url and no previous image │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent('beforeload'); │ │ │ │ │ + this.layer.div.appendChild(this.getTile()); │ │ │ │ │ + this.events.triggerEvent(this._loadEvent); │ │ │ │ │ + var img = this.getImage(); │ │ │ │ │ + var src = img.getAttribute('src') || ''; │ │ │ │ │ + if (this.url && OpenLayers.Util.isEquivalentUrl(src, this.url)) { │ │ │ │ │ + this._loadTimeout = window.setTimeout( │ │ │ │ │ + OpenLayers.Function.bind(this.onImageLoad, this), 0 │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + this.stopLoading(); │ │ │ │ │ + if (this.crossOriginKeyword) { │ │ │ │ │ + img.removeAttribute("crossorigin"); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.observe(img, "load", │ │ │ │ │ + OpenLayers.Function.bind(this.onImageLoad, this) │ │ │ │ │ + ); │ │ │ │ │ + OpenLayers.Event.observe(img, "error", │ │ │ │ │ + OpenLayers.Function.bind(this.onImageError, this) │ │ │ │ │ + ); │ │ │ │ │ + this.imageReloadAttempts = 0; │ │ │ │ │ + this.setImgSrc(this.url); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setImgSrc │ │ │ │ │ + * Sets the source for the tile image │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * url - {String} or undefined to hide the image │ │ │ │ │ + */ │ │ │ │ │ + setImgSrc: function(url) { │ │ │ │ │ + var img = this.imgDiv; │ │ │ │ │ + if (url) { │ │ │ │ │ + img.style.visibility = 'hidden'; │ │ │ │ │ + img.style.opacity = 0; │ │ │ │ │ + // don't set crossOrigin if the url is a data URL │ │ │ │ │ + if (this.crossOriginKeyword) { │ │ │ │ │ + if (url.substr(0, 5) !== 'data:') { │ │ │ │ │ + img.setAttribute("crossorigin", this.crossOriginKeyword); │ │ │ │ │ } else { │ │ │ │ │ - button.checked = !button.checked; │ │ │ │ │ - this.updateMap(); │ │ │ │ │ + img.removeAttribute("crossorigin"); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + img.src = url; │ │ │ │ │ + } else { │ │ │ │ │ + // Remove reference to the image, and leave it to the browser's │ │ │ │ │ + // caching and garbage collection. │ │ │ │ │ + this.stopLoading(); │ │ │ │ │ + this.imgDiv = null; │ │ │ │ │ + if (img.parentNode) { │ │ │ │ │ + img.parentNode.removeChild(img); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clearLayersArray │ │ │ │ │ - * User specifies either "base" or "data". we then clear all the │ │ │ │ │ - * corresponding listeners, the div, and reinitialize a new array. │ │ │ │ │ + * Method: getTile │ │ │ │ │ + * Get the tile's markup. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The tile's markup │ │ │ │ │ + */ │ │ │ │ │ + getTile: function() { │ │ │ │ │ + return this.frame ? this.frame : this.getImage(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createBackBuffer │ │ │ │ │ + * Create a backbuffer for this tile. A backbuffer isn't exactly a clone │ │ │ │ │ + * of the tile's markup, because we want to avoid the reloading of the │ │ │ │ │ + * image. So we clone the frame, and steal the image from the tile. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The markup, or undefined if the tile has no image │ │ │ │ │ + * or if it's currently loading. │ │ │ │ │ + */ │ │ │ │ │ + createBackBuffer: function() { │ │ │ │ │ + if (!this.imgDiv || this.isLoading) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var backBuffer; │ │ │ │ │ + if (this.frame) { │ │ │ │ │ + backBuffer = this.frame.cloneNode(false); │ │ │ │ │ + backBuffer.appendChild(this.imgDiv); │ │ │ │ │ + } else { │ │ │ │ │ + backBuffer = this.imgDiv; │ │ │ │ │ + } │ │ │ │ │ + this.imgDiv = null; │ │ │ │ │ + return backBuffer; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: onImageLoad │ │ │ │ │ + * Handler for the image onload event │ │ │ │ │ + */ │ │ │ │ │ + onImageLoad: function() { │ │ │ │ │ + var img = this.imgDiv; │ │ │ │ │ + this.stopLoading(); │ │ │ │ │ + img.style.visibility = 'inherit'; │ │ │ │ │ + img.style.opacity = this.layer.opacity; │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + this.canvasContext = null; │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + │ │ │ │ │ + if (this.layerAlphaHack === true) { │ │ │ │ │ + img.style.filter = │ │ │ │ │ + "progid:DXImageTransform.Microsoft.AlphaImageLoader(src='" + │ │ │ │ │ + img.src + "', sizingMethod='scale')"; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: onImageError │ │ │ │ │ + * Handler for the image onerror event │ │ │ │ │ + */ │ │ │ │ │ + onImageError: function() { │ │ │ │ │ + var img = this.imgDiv; │ │ │ │ │ + if (img.src != null) { │ │ │ │ │ + this.imageReloadAttempts++; │ │ │ │ │ + if (this.imageReloadAttempts <= OpenLayers.IMAGE_RELOAD_ATTEMPTS) { │ │ │ │ │ + this.setImgSrc(this.layer.getURL(this.bounds)); │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Element.addClass(img, "olImageLoadError"); │ │ │ │ │ + this.events.triggerEvent("loaderror"); │ │ │ │ │ + this.onImageLoad(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: stopLoading │ │ │ │ │ + * Stops a loading sequence so <onImageLoad> won't be executed. │ │ │ │ │ + */ │ │ │ │ │ + stopLoading: function() { │ │ │ │ │ + OpenLayers.Event.stopObservingElement(this.imgDiv); │ │ │ │ │ + window.clearTimeout(this._loadTimeout); │ │ │ │ │ + delete this._loadTimeout; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getCanvasContext │ │ │ │ │ + * Returns a canvas context associated with the tile image (with │ │ │ │ │ + * the image drawn on it). │ │ │ │ │ + * Returns undefined if the browser does not support canvas, if │ │ │ │ │ + * the tile has no image or if it's currently loading. │ │ │ │ │ + * │ │ │ │ │ + * The function returns a canvas context instance but the │ │ │ │ │ + * underlying canvas is still available in the 'canvas' property: │ │ │ │ │ + * (code) │ │ │ │ │ + * var context = tile.getCanvasContext(); │ │ │ │ │ + * if (context) { │ │ │ │ │ + * var data = context.canvas.toDataURL('image/jpeg'); │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + getCanvasContext: function() { │ │ │ │ │ + if (OpenLayers.CANVAS_SUPPORTED && this.imgDiv && !this.isLoading) { │ │ │ │ │ + if (!this.canvasContext) { │ │ │ │ │ + var canvas = document.createElement("canvas"); │ │ │ │ │ + canvas.width = this.size.w; │ │ │ │ │ + canvas.height = this.size.h; │ │ │ │ │ + this.canvasContext = canvas.getContext("2d"); │ │ │ │ │ + this.canvasContext.drawImage(this.imgDiv, 0, 0); │ │ │ │ │ + } │ │ │ │ │ + return this.canvasContext; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Tile.Image" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Tile.Image.IMAGE │ │ │ │ │ + * {HTMLImageElement} The image for a tile. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Tile.Image.IMAGE = (function() { │ │ │ │ │ + var img = new Image(); │ │ │ │ │ + img.className = "olTileImage"; │ │ │ │ │ + // avoid image gallery menu in IE6 │ │ │ │ │ + img.galleryImg = "no"; │ │ │ │ │ + return img; │ │ │ │ │ +}()); │ │ │ │ │ + │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/Grid.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/HTTPRequest.js │ │ │ │ │ + * @requires OpenLayers/Tile/Image.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.Grid │ │ │ │ │ + * Base class for layers that use a lattice of tiles. Create a new grid │ │ │ │ │ + * layer with the <OpenLayers.Layer.Grid> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.HTTPRequest> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Grid = OpenLayers.Class(OpenLayers.Layer.HTTPRequest, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: tileSize │ │ │ │ │ + * {<OpenLayers.Size>} │ │ │ │ │ + */ │ │ │ │ │ + tileSize: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: tileOriginCorner │ │ │ │ │ + * {String} If the <tileOrigin> property is not provided, the tile origin │ │ │ │ │ + * will be derived from the layer's <maxExtent>. The corner of the │ │ │ │ │ + * <maxExtent> used is determined by this property. Acceptable values │ │ │ │ │ + * are "tl" (top left), "tr" (top right), "bl" (bottom left), and "br" │ │ │ │ │ + * (bottom right). Default is "bl". │ │ │ │ │ + */ │ │ │ │ │ + tileOriginCorner: "bl", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: tileOrigin │ │ │ │ │ + * {<OpenLayers.LonLat>} Optional origin for aligning the grid of tiles. │ │ │ │ │ + * If provided, requests for tiles at all resolutions will be aligned │ │ │ │ │ + * with this location (no tiles shall overlap this location). If │ │ │ │ │ + * not provided, the grid of tiles will be aligned with the layer's │ │ │ │ │ + * <maxExtent>. Default is ``null``. │ │ │ │ │ + */ │ │ │ │ │ + tileOrigin: null, │ │ │ │ │ + │ │ │ │ │ + /** APIProperty: tileOptions │ │ │ │ │ + * {Object} optional configuration options for <OpenLayers.Tile> instances │ │ │ │ │ + * created by this Layer, if supported by the tile class. │ │ │ │ │ + */ │ │ │ │ │ + tileOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: tileClass │ │ │ │ │ + * {<OpenLayers.Tile>} The tile class to use for this layer. │ │ │ │ │ + * Defaults is OpenLayers.Tile.Image. │ │ │ │ │ + */ │ │ │ │ │ + tileClass: OpenLayers.Tile.Image, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: grid │ │ │ │ │ + * {Array(Array(<OpenLayers.Tile>))} This is an array of rows, each row is │ │ │ │ │ + * an array of tiles. │ │ │ │ │ + */ │ │ │ │ │ + grid: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: singleTile │ │ │ │ │ + * {Boolean} Moves the layer into single-tile mode, meaning that one tile │ │ │ │ │ + * will be loaded. The tile's size will be determined by the 'ratio' │ │ │ │ │ + * property. When the tile is dragged such that it does not cover the │ │ │ │ │ + * entire viewport, it is reloaded. │ │ │ │ │ + */ │ │ │ │ │ + singleTile: false, │ │ │ │ │ + │ │ │ │ │ + /** APIProperty: ratio │ │ │ │ │ + * {Float} Used only when in single-tile mode, this specifies the │ │ │ │ │ + * ratio of the size of the single tile to the size of the map. │ │ │ │ │ + * Default value is 1.5. │ │ │ │ │ + */ │ │ │ │ │ + ratio: 1.5, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: buffer │ │ │ │ │ + * {Integer} Used only when in gridded mode, this specifies the number of │ │ │ │ │ + * extra rows and colums of tiles on each side which will │ │ │ │ │ + * surround the minimum grid tiles to cover the map. │ │ │ │ │ + * For very slow loading layers, a larger value may increase │ │ │ │ │ + * performance somewhat when dragging, but will increase bandwidth │ │ │ │ │ + * use significantly. │ │ │ │ │ + */ │ │ │ │ │ + buffer: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: transitionEffect │ │ │ │ │ + * {String} The transition effect to use when the map is zoomed. │ │ │ │ │ + * Two posible values: │ │ │ │ │ + * │ │ │ │ │ + * "resize" - Existing tiles are resized on zoom to provide a visual │ │ │ │ │ + * effect of the zoom having taken place immediately. As the │ │ │ │ │ + * new tiles become available, they are drawn on top of the │ │ │ │ │ + * resized tiles (this is the default setting). │ │ │ │ │ + * "map-resize" - Existing tiles are resized on zoom and placed below the │ │ │ │ │ + * base layer. New tiles for the base layer will cover existing tiles. │ │ │ │ │ + * This setting is recommended when having an overlay duplicated during │ │ │ │ │ + * the transition is undesirable (e.g. street labels or big transparent │ │ │ │ │ + * fills). │ │ │ │ │ + * null - No transition effect. │ │ │ │ │ + * │ │ │ │ │ + * Using "resize" on non-opaque layers can cause undesired visual │ │ │ │ │ + * effects. Set transitionEffect to null in this case. │ │ │ │ │ + */ │ │ │ │ │ + transitionEffect: "resize", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: numLoadingTiles │ │ │ │ │ + * {Integer} How many tiles are still loading? │ │ │ │ │ + */ │ │ │ │ │ + numLoadingTiles: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: serverResolutions │ │ │ │ │ + * {Array(Number}} This property is documented in subclasses as │ │ │ │ │ + * an API property. │ │ │ │ │ + */ │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: loading │ │ │ │ │ + * {Boolean} Indicates if tiles are being loaded. │ │ │ │ │ + */ │ │ │ │ │ + loading: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: backBuffer │ │ │ │ │ + * {DOMElement} The back buffer. │ │ │ │ │ + */ │ │ │ │ │ + backBuffer: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: gridResolution │ │ │ │ │ + * {Number} The resolution of the current grid. Used for backbuffer and │ │ │ │ │ + * client zoom. This property is updated every time the grid is │ │ │ │ │ + * initialized. │ │ │ │ │ + */ │ │ │ │ │ + gridResolution: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: backBufferResolution │ │ │ │ │ + * {Number} The resolution of the current back buffer. This property is │ │ │ │ │ + * updated each time a back buffer is created. │ │ │ │ │ + */ │ │ │ │ │ + backBufferResolution: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: backBufferLonLat │ │ │ │ │ + * {Object} The top-left corner of the current back buffer. Includes lon │ │ │ │ │ + * and lat properties. This object is updated each time a back buffer │ │ │ │ │ + * is created. │ │ │ │ │ + */ │ │ │ │ │ + backBufferLonLat: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: backBufferTimerId │ │ │ │ │ + * {Number} The id of the back buffer timer. This timer is used to │ │ │ │ │ + * delay the removal of the back buffer, thereby preventing │ │ │ │ │ + * flash effects caused by tile animation. │ │ │ │ │ + */ │ │ │ │ │ + backBufferTimerId: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: removeBackBufferDelay │ │ │ │ │ + * {Number} Delay for removing the backbuffer when all tiles have finished │ │ │ │ │ + * loading. Can be set to 0 when no css opacity transitions for the │ │ │ │ │ + * olTileImage class are used. Default is 0 for <singleTile> layers, │ │ │ │ │ + * 2500 for tiled layers. See <className> for more information on │ │ │ │ │ + * tile animation. │ │ │ │ │ + */ │ │ │ │ │ + removeBackBufferDelay: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: className │ │ │ │ │ + * {String} Name of the class added to the layer div. If not set in the │ │ │ │ │ + * options passed to the constructor then className defaults to │ │ │ │ │ + * "olLayerGridSingleTile" for single tile layers (see <singleTile>), │ │ │ │ │ + * and "olLayerGrid" for non single tile layers. │ │ │ │ │ + * │ │ │ │ │ + * Note: │ │ │ │ │ + * │ │ │ │ │ + * The displaying of tiles is not animated by default for single tile │ │ │ │ │ + * layers - OpenLayers' default theme (style.css) includes this: │ │ │ │ │ + * (code) │ │ │ │ │ + * .olLayerGrid .olTileImage { │ │ │ │ │ + * -webkit-transition: opacity 0.2s linear; │ │ │ │ │ + * -moz-transition: opacity 0.2s linear; │ │ │ │ │ + * -o-transition: opacity 0.2s linear; │ │ │ │ │ + * transition: opacity 0.2s linear; │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ + * To animate tile displaying for any grid layer the following │ │ │ │ │ + * CSS rule can be used: │ │ │ │ │ + * (code) │ │ │ │ │ + * .olTileImage { │ │ │ │ │ + * -webkit-transition: opacity 0.2s linear; │ │ │ │ │ + * -moz-transition: opacity 0.2s linear; │ │ │ │ │ + * -o-transition: opacity 0.2s linear; │ │ │ │ │ + * transition: opacity 0.2s linear; │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ + * In that case, to avoid flash effects, <removeBackBufferDelay> │ │ │ │ │ + * should not be zero. │ │ │ │ │ + */ │ │ │ │ │ + className: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * layer.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Listeners will be called with a reference to an event object. The │ │ │ │ │ + * properties of this event depends on exactly what happened. │ │ │ │ │ + * │ │ │ │ │ + * All event objects have at least the following properties: │ │ │ │ │ + * object - {Object} A reference to layer.events.object. │ │ │ │ │ + * element - {DOMElement} A reference to layer.events.element. │ │ │ │ │ + * │ │ │ │ │ + * Supported event types: │ │ │ │ │ + * addtile - Triggered when a tile is added to this layer. Listeners receive │ │ │ │ │ + * an object as first argument, which has a tile property that │ │ │ │ │ + * references the tile that has been added. │ │ │ │ │ + * tileloadstart - Triggered when a tile starts loading. Listeners receive │ │ │ │ │ + * an object as first argument, which has a tile property that │ │ │ │ │ + * references the tile that starts loading. │ │ │ │ │ + * tileloaded - Triggered when each new tile is │ │ │ │ │ + * loaded, as a means of progress update to listeners. │ │ │ │ │ + * listeners can access 'numLoadingTiles' if they wish to keep │ │ │ │ │ + * track of the loading progress. Listeners are called with an object │ │ │ │ │ + * with a 'tile' property as first argument, making the loaded tile │ │ │ │ │ + * available to the listener, and an 'aborted' property, which will be │ │ │ │ │ + * true when loading was aborted and no tile data is available. │ │ │ │ │ + * tileerror - Triggered before the tileloaded event (i.e. when the tile is │ │ │ │ │ + * still hidden) if a tile failed to load. Listeners receive an object │ │ │ │ │ + * as first argument, which has a tile property that references the │ │ │ │ │ + * tile that could not be loaded. │ │ │ │ │ + * retile - Triggered when the layer recreates its tile grid. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: gridLayout │ │ │ │ │ + * {Object} Object containing properties tilelon, tilelat, startcol, │ │ │ │ │ + * startrow │ │ │ │ │ + */ │ │ │ │ │ + gridLayout: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: rowSign │ │ │ │ │ + * {Number} 1 for grids starting at the top, -1 for grids starting at the │ │ │ │ │ + * bottom. This is used for several grid index and offset calculations. │ │ │ │ │ + */ │ │ │ │ │ + rowSign: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: transitionendEvents │ │ │ │ │ + * {Array} Event names for transitionend │ │ │ │ │ + */ │ │ │ │ │ + transitionendEvents: [ │ │ │ │ │ + 'transitionend', 'webkitTransitionEnd', 'otransitionend', │ │ │ │ │ + 'oTransitionEnd' │ │ │ │ │ + ], │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.Grid │ │ │ │ │ + * Create a new grid layer │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layersType - {String} │ │ │ │ │ + * name - {String} │ │ │ │ │ + * url - {String} │ │ │ │ │ + * params - {Object} │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ */ │ │ │ │ │ - clearLayersArray: function(layersType) { │ │ │ │ │ - this[layersType + "LayersDiv"].innerHTML = ""; │ │ │ │ │ - this[layersType + "Layers"] = []; │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.initialize.apply(this, │ │ │ │ │ + arguments); │ │ │ │ │ + this.grid = []; │ │ │ │ │ + this._removeBackBuffer = OpenLayers.Function.bind(this.removeBackBuffer, this); │ │ │ │ │ + │ │ │ │ │ + this.initProperties(); │ │ │ │ │ + │ │ │ │ │ + this.rowSign = this.tileOriginCorner.substr(0, 1) === "t" ? 1 : -1; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: initProperties │ │ │ │ │ + * Set any properties that depend on the value of singleTile. │ │ │ │ │ + * Currently sets removeBackBufferDelay and className │ │ │ │ │ + */ │ │ │ │ │ + initProperties: function() { │ │ │ │ │ + if (this.options.removeBackBufferDelay === undefined) { │ │ │ │ │ + this.removeBackBufferDelay = this.singleTile ? 0 : 2500; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.options.className === undefined) { │ │ │ │ │ + this.className = this.singleTile ? 'olLayerGridSingleTile' : │ │ │ │ │ + 'olLayerGrid'; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} The map. │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.setMap.call(this, map); │ │ │ │ │ + OpenLayers.Element.addClass(this.div, this.className); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: checkRedraw │ │ │ │ │ - * Checks if the layer state has changed since the last redraw() call. │ │ │ │ │ + * Method: removeMap │ │ │ │ │ + * Called when the layer is removed from the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} The map. │ │ │ │ │ + */ │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + this.removeBackBuffer(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Deconstruct the layer and clear the grid. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.removeBackBuffer(); │ │ │ │ │ + this.clearGrid(); │ │ │ │ │ + │ │ │ │ │ + this.grid = null; │ │ │ │ │ + this.tileSize = null; │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: mergeNewParams │ │ │ │ │ + * Refetches tiles with new params merged, keeping a backbuffer. Each │ │ │ │ │ + * loading new tile will have a css class of '.olTileReplacing'. If a │ │ │ │ │ + * stylesheet applies a 'display: none' style to that class, any fade-in │ │ │ │ │ + * transition will not apply, and backbuffers for each tile will be removed │ │ │ │ │ + * as soon as the tile is loaded. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newParams - {Object} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The layer state changed since the last redraw() call. │ │ │ │ │ + * redrawn: {Boolean} whether the layer was actually redrawn. │ │ │ │ │ */ │ │ │ │ │ - checkRedraw: function() { │ │ │ │ │ - if (!this.layerStates.length || │ │ │ │ │ - (this.map.layers.length != this.layerStates.length)) { │ │ │ │ │ - return true; │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clearGrid │ │ │ │ │ + * Go through and remove all tiles from the grid, calling │ │ │ │ │ + * destroy() on each of them to kill circular references │ │ │ │ │ + */ │ │ │ │ │ + clearGrid: function() { │ │ │ │ │ + if (this.grid) { │ │ │ │ │ + for (var iRow = 0, len = this.grid.length; iRow < len; iRow++) { │ │ │ │ │ + var row = this.grid[iRow]; │ │ │ │ │ + for (var iCol = 0, clen = row.length; iCol < clen; iCol++) { │ │ │ │ │ + var tile = row[iCol]; │ │ │ │ │ + this.destroyTile(tile); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.grid = []; │ │ │ │ │ + this.gridResolution = null; │ │ │ │ │ + this.gridLayout = null; │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - for (var i = 0, len = this.layerStates.length; i < len; i++) { │ │ │ │ │ - var layerState = this.layerStates[i]; │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if ((layerState.name != layer.name) || │ │ │ │ │ - (layerState.inRange != layer.inRange) || │ │ │ │ │ - (layerState.id != layer.id) || │ │ │ │ │ - (layerState.visibility != layer.visibility)) { │ │ │ │ │ - return true; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: addOptions │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newOptions - {Object} │ │ │ │ │ + * reinitialize - {Boolean} If set to true, and if resolution options of the │ │ │ │ │ + * current baseLayer were changed, the map will be recentered to make │ │ │ │ │ + * sure that it is displayed with a valid resolution, and a │ │ │ │ │ + * changebaselayer event will be triggered. │ │ │ │ │ + */ │ │ │ │ │ + addOptions: function(newOptions, reinitialize) { │ │ │ │ │ + var singleTileChanged = newOptions.singleTile !== undefined && │ │ │ │ │ + newOptions.singleTile !== this.singleTile; │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.addOptions.apply(this, arguments); │ │ │ │ │ + if (this.map && singleTileChanged) { │ │ │ │ │ + this.initProperties(); │ │ │ │ │ + this.clearGrid(); │ │ │ │ │ + this.tileSize = this.options.tileSize; │ │ │ │ │ + this.setTileSize(); │ │ │ │ │ + this.moveTo(null, true); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} Is this ever used? │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Grid>} An exact clone of this OpenLayers.Layer.Grid │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Grid(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.params, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.HTTPRequest.prototype.clone.apply(this, [obj]); │ │ │ │ │ + │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + if (this.tileSize != null) { │ │ │ │ │ + obj.tileSize = this.tileSize.clone(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // we do not want to copy reference to grid, so we make a new array │ │ │ │ │ + obj.grid = []; │ │ │ │ │ + obj.gridResolution = null; │ │ │ │ │ + // same for backbuffer │ │ │ │ │ + obj.backBuffer = null; │ │ │ │ │ + obj.backBufferTimerId = null; │ │ │ │ │ + obj.loading = false; │ │ │ │ │ + obj.numLoadingTiles = 0; │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * This function is called whenever the map is moved. All the moving │ │ │ │ │ + * of actual 'tiles' is done by the map, but moveTo's role is to accept │ │ │ │ │ + * a bounds and make sure the data that that bounds requires is pre-loaded. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * zoomChanged - {Boolean} │ │ │ │ │ + * dragging - {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + bounds = bounds || this.map.getExtent(); │ │ │ │ │ + │ │ │ │ │ + if (bounds != null) { │ │ │ │ │ + │ │ │ │ │ + // if grid is empty or zoom has changed, we *must* re-tile │ │ │ │ │ + var forceReTile = !this.grid.length || zoomChanged; │ │ │ │ │ + │ │ │ │ │ + // total bounds of the tiles │ │ │ │ │ + var tilesBounds = this.getTilesBounds(); │ │ │ │ │ + │ │ │ │ │ + // the new map resolution │ │ │ │ │ + var resolution = this.map.getResolution(); │ │ │ │ │ + │ │ │ │ │ + // the server-supported resolution for the new map resolution │ │ │ │ │ + var serverResolution = this.getServerResolution(resolution); │ │ │ │ │ + │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + │ │ │ │ │ + // We want to redraw whenever even the slightest part of the │ │ │ │ │ + // current bounds is not contained by our tile. │ │ │ │ │ + // (thus, we do not specify partial -- its default is false) │ │ │ │ │ + │ │ │ │ │ + if (forceReTile || │ │ │ │ │ + (!dragging && !tilesBounds.containsBounds(bounds))) { │ │ │ │ │ + │ │ │ │ │ + // In single tile mode with no transition effect, we insert │ │ │ │ │ + // a non-scaled backbuffer when the layer is moved. But if │ │ │ │ │ + // a zoom occurs right after a move, i.e. before the new │ │ │ │ │ + // image is received, we need to remove the backbuffer, or │ │ │ │ │ + // an ill-positioned image will be visible during the zoom │ │ │ │ │ + // transition. │ │ │ │ │ + │ │ │ │ │ + if (zoomChanged && this.transitionEffect !== 'resize') { │ │ │ │ │ + this.removeBackBuffer(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (!zoomChanged || this.transitionEffect === 'resize') { │ │ │ │ │ + this.applyBackBuffer(resolution); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.initSingleTile(bounds); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + │ │ │ │ │ + // if the bounds have changed such that they are not even │ │ │ │ │ + // *partially* contained by our tiles (e.g. when user has │ │ │ │ │ + // programmatically panned to the other side of the earth on │ │ │ │ │ + // zoom level 18), then moveGriddedTiles could potentially have │ │ │ │ │ + // to run through thousands of cycles, so we want to reTile │ │ │ │ │ + // instead (thus, partial true). │ │ │ │ │ + forceReTile = forceReTile || │ │ │ │ │ + !tilesBounds.intersectsBounds(bounds, { │ │ │ │ │ + worldBounds: this.map.baseLayer.wrapDateLine && │ │ │ │ │ + this.map.getMaxExtent() │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + if (forceReTile) { │ │ │ │ │ + if (zoomChanged && (this.transitionEffect === 'resize' || │ │ │ │ │ + this.gridResolution === resolution)) { │ │ │ │ │ + this.applyBackBuffer(resolution); │ │ │ │ │ + } │ │ │ │ │ + this.initGriddedTiles(bounds); │ │ │ │ │ + } else { │ │ │ │ │ + this.moveGriddedTiles(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getTileData │ │ │ │ │ + * Given a map location, retrieve a tile and the pixel offset within that │ │ │ │ │ + * tile corresponding to the location. If there is not an existing │ │ │ │ │ + * tile in the grid that covers the given location, null will be │ │ │ │ │ + * returned. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * loc - {<OpenLayers.LonLat>} map location │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Object with the following properties: tile ({<OpenLayers.Tile>}), │ │ │ │ │ + * i ({Number} x-pixel offset from top left), and j ({Integer} y-pixel │ │ │ │ │ + * offset from top left). │ │ │ │ │ + */ │ │ │ │ │ + getTileData: function(loc) { │ │ │ │ │ + var data = null, │ │ │ │ │ + x = loc.lon, │ │ │ │ │ + y = loc.lat, │ │ │ │ │ + numRows = this.grid.length; │ │ │ │ │ + │ │ │ │ │ + if (this.map && numRows) { │ │ │ │ │ + var res = this.map.getResolution(), │ │ │ │ │ + tileWidth = this.tileSize.w, │ │ │ │ │ + tileHeight = this.tileSize.h, │ │ │ │ │ + bounds = this.grid[0][0].bounds, │ │ │ │ │ + left = bounds.left, │ │ │ │ │ + top = bounds.top; │ │ │ │ │ + │ │ │ │ │ + if (x < left) { │ │ │ │ │ + // deal with multiple worlds │ │ │ │ │ + if (this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + var worldWidth = this.map.getMaxExtent().getWidth(); │ │ │ │ │ + var worldsAway = Math.ceil((left - x) / worldWidth); │ │ │ │ │ + x += worldWidth * worldsAway; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // tile distance to location (fractional number of tiles); │ │ │ │ │ + var dtx = (x - left) / (res * tileWidth); │ │ │ │ │ + var dty = (top - y) / (res * tileHeight); │ │ │ │ │ + // index of tile in grid │ │ │ │ │ + var col = Math.floor(dtx); │ │ │ │ │ + var row = Math.floor(dty); │ │ │ │ │ + if (row >= 0 && row < numRows) { │ │ │ │ │ + var tile = this.grid[row][col]; │ │ │ │ │ + if (tile) { │ │ │ │ │ + data = { │ │ │ │ │ + tile: tile, │ │ │ │ │ + // pixel index within tile │ │ │ │ │ + i: Math.floor((dtx - col) * tileWidth), │ │ │ │ │ + j: Math.floor((dty - row) * tileHeight) │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return data; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return false; │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroyTile │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * tile - {<OpenLayers.Tile>} │ │ │ │ │ + */ │ │ │ │ │ + destroyTile: function(tile) { │ │ │ │ │ + this.removeTileMonitoringHooks(tile); │ │ │ │ │ + tile.destroy(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: redraw │ │ │ │ │ - * Goes through and takes the current state of the Map and rebuilds the │ │ │ │ │ - * control to display that state. Groups base layers into a │ │ │ │ │ - * radio-button group and lists each data layer with a checkbox. │ │ │ │ │ + * Method: getServerResolution │ │ │ │ │ + * Return the closest server-supported resolution. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resolution - {Number} The base resolution. If undefined the │ │ │ │ │ + * map resolution is used. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ + * {Number} The closest server resolution value. │ │ │ │ │ */ │ │ │ │ │ - redraw: function() { │ │ │ │ │ - //if the state hasn't changed since last redraw, no need │ │ │ │ │ - // to do anything. Just return the existing div. │ │ │ │ │ - if (!this.checkRedraw()) { │ │ │ │ │ - return this.div; │ │ │ │ │ + getServerResolution: function(resolution) { │ │ │ │ │ + var distance = Number.POSITIVE_INFINITY; │ │ │ │ │ + resolution = resolution || this.map.getResolution(); │ │ │ │ │ + if (this.serverResolutions && │ │ │ │ │ + OpenLayers.Util.indexOf(this.serverResolutions, resolution) === -1) { │ │ │ │ │ + var i, newDistance, newResolution, serverResolution; │ │ │ │ │ + for (i = this.serverResolutions.length - 1; i >= 0; i--) { │ │ │ │ │ + newResolution = this.serverResolutions[i]; │ │ │ │ │ + newDistance = Math.abs(newResolution - resolution); │ │ │ │ │ + if (newDistance > distance) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + distance = newDistance; │ │ │ │ │ + serverResolution = newResolution; │ │ │ │ │ + } │ │ │ │ │ + resolution = serverResolution; │ │ │ │ │ } │ │ │ │ │ + return resolution; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - //clear out previous layers │ │ │ │ │ - this.clearLayersArray("base"); │ │ │ │ │ - this.clearLayersArray("data"); │ │ │ │ │ + /** │ │ │ │ │ + * Method: getServerZoom │ │ │ │ │ + * Return the zoom value corresponding to the best matching server │ │ │ │ │ + * resolution, taking into account <serverResolutions> and <zoomOffset>. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number} The closest server supported zoom. This is not the map zoom │ │ │ │ │ + * level, but an index of the server's resolutions array. │ │ │ │ │ + */ │ │ │ │ │ + getServerZoom: function() { │ │ │ │ │ + var resolution = this.getServerResolution(); │ │ │ │ │ + return this.serverResolutions ? │ │ │ │ │ + OpenLayers.Util.indexOf(this.serverResolutions, resolution) : │ │ │ │ │ + this.map.getZoomForResolution(resolution) + (this.zoomOffset || 0); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var containsOverlays = false; │ │ │ │ │ - var containsBaseLayers = false; │ │ │ │ │ + /** │ │ │ │ │ + * Method: applyBackBuffer │ │ │ │ │ + * Create, insert, scale and position a back buffer for the layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resolution - {Number} The resolution to transition to. │ │ │ │ │ + */ │ │ │ │ │ + applyBackBuffer: function(resolution) { │ │ │ │ │ + if (this.backBufferTimerId !== null) { │ │ │ │ │ + this.removeBackBuffer(); │ │ │ │ │ + } │ │ │ │ │ + var backBuffer = this.backBuffer; │ │ │ │ │ + if (!backBuffer) { │ │ │ │ │ + backBuffer = this.createBackBuffer(); │ │ │ │ │ + if (!backBuffer) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (resolution === this.gridResolution) { │ │ │ │ │ + this.div.insertBefore(backBuffer, this.div.firstChild); │ │ │ │ │ + } else { │ │ │ │ │ + this.map.baseLayer.div.parentNode.insertBefore(backBuffer, this.map.baseLayer.div); │ │ │ │ │ + } │ │ │ │ │ + this.backBuffer = backBuffer; │ │ │ │ │ │ │ │ │ │ - // Save state -- for checking layer if the map state changed. │ │ │ │ │ - // We save this before redrawing, because in the process of redrawing │ │ │ │ │ - // we will trigger more visibility changes, and we want to not redraw │ │ │ │ │ - // and enter an infinite loop. │ │ │ │ │ - var len = this.map.layers.length; │ │ │ │ │ - this.layerStates = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - this.layerStates[i] = { │ │ │ │ │ - 'name': layer.name, │ │ │ │ │ - 'visibility': layer.visibility, │ │ │ │ │ - 'inRange': layer.inRange, │ │ │ │ │ - 'id': layer.id │ │ │ │ │ + // set some information in the instance for subsequent │ │ │ │ │ + // calls to applyBackBuffer where the same back buffer │ │ │ │ │ + // is reused │ │ │ │ │ + var topLeftTileBounds = this.grid[0][0].bounds; │ │ │ │ │ + this.backBufferLonLat = { │ │ │ │ │ + lon: topLeftTileBounds.left, │ │ │ │ │ + lat: topLeftTileBounds.top │ │ │ │ │ }; │ │ │ │ │ + this.backBufferResolution = this.gridResolution; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - var layers = this.map.layers.slice(); │ │ │ │ │ - if (!this.ascending) { │ │ │ │ │ - layers.reverse(); │ │ │ │ │ + var ratio = this.backBufferResolution / resolution; │ │ │ │ │ + │ │ │ │ │ + // scale the tiles inside the back buffer │ │ │ │ │ + var tiles = backBuffer.childNodes, │ │ │ │ │ + tile; │ │ │ │ │ + for (var i = tiles.length - 1; i >= 0; --i) { │ │ │ │ │ + tile = tiles[i]; │ │ │ │ │ + tile.style.top = ((ratio * tile._i * tile._h) | 0) + 'px'; │ │ │ │ │ + tile.style.left = ((ratio * tile._j * tile._w) | 0) + 'px'; │ │ │ │ │ + tile.style.width = Math.round(ratio * tile._w) + 'px'; │ │ │ │ │ + tile.style.height = Math.round(ratio * tile._h) + 'px'; │ │ │ │ │ } │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - var layer = layers[i]; │ │ │ │ │ - var baseLayer = layer.isBaseLayer; │ │ │ │ │ │ │ │ │ │ - if (layer.displayInLayerSwitcher) { │ │ │ │ │ + // and position it (based on the grid's top-left corner) │ │ │ │ │ + var position = this.getViewPortPxFromLonLat( │ │ │ │ │ + this.backBufferLonLat, resolution); │ │ │ │ │ + var leftOffset = this.map.layerContainerOriginPx.x; │ │ │ │ │ + var topOffset = this.map.layerContainerOriginPx.y; │ │ │ │ │ + backBuffer.style.left = Math.round(position.x - leftOffset) + 'px'; │ │ │ │ │ + backBuffer.style.top = Math.round(position.y - topOffset) + 'px'; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (baseLayer) { │ │ │ │ │ - containsBaseLayers = true; │ │ │ │ │ + /** │ │ │ │ │ + * Method: createBackBuffer │ │ │ │ │ + * Create a back buffer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The DOM element for the back buffer, undefined if the │ │ │ │ │ + * grid isn't initialized yet. │ │ │ │ │ + */ │ │ │ │ │ + createBackBuffer: function() { │ │ │ │ │ + var backBuffer; │ │ │ │ │ + if (this.grid.length > 0) { │ │ │ │ │ + backBuffer = document.createElement('div'); │ │ │ │ │ + backBuffer.id = this.div.id + '_bb'; │ │ │ │ │ + backBuffer.className = 'olBackBuffer'; │ │ │ │ │ + backBuffer.style.position = 'absolute'; │ │ │ │ │ + var map = this.map; │ │ │ │ │ + backBuffer.style.zIndex = this.transitionEffect === 'resize' ? │ │ │ │ │ + this.getZIndex() - 1 : │ │ │ │ │ + // 'map-resize': │ │ │ │ │ + map.Z_INDEX_BASE.BaseLayer - │ │ │ │ │ + (map.getNumLayers() - map.getLayerIndex(this)); │ │ │ │ │ + for (var i = 0, lenI = this.grid.length; i < lenI; i++) { │ │ │ │ │ + for (var j = 0, lenJ = this.grid[i].length; j < lenJ; j++) { │ │ │ │ │ + var tile = this.grid[i][j], │ │ │ │ │ + markup = this.grid[i][j].createBackBuffer(); │ │ │ │ │ + if (markup) { │ │ │ │ │ + markup._i = i; │ │ │ │ │ + markup._j = j; │ │ │ │ │ + markup._w = tile.size.w; │ │ │ │ │ + markup._h = tile.size.h; │ │ │ │ │ + markup.id = tile.id + '_bb'; │ │ │ │ │ + backBuffer.appendChild(markup); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return backBuffer; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeBackBuffer │ │ │ │ │ + * Remove back buffer from DOM. │ │ │ │ │ + */ │ │ │ │ │ + removeBackBuffer: function() { │ │ │ │ │ + if (this._transitionElement) { │ │ │ │ │ + for (var i = this.transitionendEvents.length - 1; i >= 0; --i) { │ │ │ │ │ + OpenLayers.Event.stopObserving(this._transitionElement, │ │ │ │ │ + this.transitionendEvents[i], this._removeBackBuffer); │ │ │ │ │ + } │ │ │ │ │ + delete this._transitionElement; │ │ │ │ │ + } │ │ │ │ │ + if (this.backBuffer) { │ │ │ │ │ + if (this.backBuffer.parentNode) { │ │ │ │ │ + this.backBuffer.parentNode.removeChild(this.backBuffer); │ │ │ │ │ + } │ │ │ │ │ + this.backBuffer = null; │ │ │ │ │ + this.backBufferResolution = null; │ │ │ │ │ + if (this.backBufferTimerId !== null) { │ │ │ │ │ + window.clearTimeout(this.backBufferTimerId); │ │ │ │ │ + this.backBufferTimerId = null; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveByPx │ │ │ │ │ + * Move the layer based on pixel vector. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * dx - {Number} │ │ │ │ │ + * dy - {Number} │ │ │ │ │ + */ │ │ │ │ │ + moveByPx: function(dx, dy) { │ │ │ │ │ + if (!this.singleTile) { │ │ │ │ │ + this.moveGriddedTiles(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setTileSize │ │ │ │ │ + * Check if we are in singleTile mode and if so, set the size as a ratio │ │ │ │ │ + * of the map size (as specified by the layer's 'ratio' property). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * size - {<OpenLayers.Size>} │ │ │ │ │ + */ │ │ │ │ │ + setTileSize: function(size) { │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + size.h = parseInt(size.h * this.ratio, 10); │ │ │ │ │ + size.w = parseInt(size.w * this.ratio, 10); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.setTileSize.apply(this, [size]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getTilesBounds │ │ │ │ │ + * Return the bounds of the tile grid. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} A Bounds object representing the bounds of all the │ │ │ │ │ + * currently loaded tiles (including those partially or not at all seen │ │ │ │ │ + * onscreen). │ │ │ │ │ + */ │ │ │ │ │ + getTilesBounds: function() { │ │ │ │ │ + var bounds = null; │ │ │ │ │ + │ │ │ │ │ + var length = this.grid.length; │ │ │ │ │ + if (length) { │ │ │ │ │ + var bottomLeftTileBounds = this.grid[length - 1][0].bounds, │ │ │ │ │ + width = this.grid[0].length * bottomLeftTileBounds.getWidth(), │ │ │ │ │ + height = this.grid.length * bottomLeftTileBounds.getHeight(); │ │ │ │ │ + │ │ │ │ │ + bounds = new OpenLayers.Bounds(bottomLeftTileBounds.left, │ │ │ │ │ + bottomLeftTileBounds.bottom, │ │ │ │ │ + bottomLeftTileBounds.left + width, │ │ │ │ │ + bottomLeftTileBounds.bottom + height); │ │ │ │ │ + } │ │ │ │ │ + return bounds; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: initSingleTile │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + */ │ │ │ │ │ + initSingleTile: function(bounds) { │ │ │ │ │ + this.events.triggerEvent("retile"); │ │ │ │ │ + │ │ │ │ │ + //determine new tile bounds │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var tileWidth = bounds.getWidth() * this.ratio; │ │ │ │ │ + var tileHeight = bounds.getHeight() * this.ratio; │ │ │ │ │ + │ │ │ │ │ + var tileBounds = │ │ │ │ │ + new OpenLayers.Bounds(center.lon - (tileWidth / 2), │ │ │ │ │ + center.lat - (tileHeight / 2), │ │ │ │ │ + center.lon + (tileWidth / 2), │ │ │ │ │ + center.lat + (tileHeight / 2)); │ │ │ │ │ + │ │ │ │ │ + var px = this.map.getLayerPxFromLonLat({ │ │ │ │ │ + lon: tileBounds.left, │ │ │ │ │ + lat: tileBounds.top │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + if (!this.grid.length) { │ │ │ │ │ + this.grid[0] = []; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var tile = this.grid[0][0]; │ │ │ │ │ + if (!tile) { │ │ │ │ │ + tile = this.addTile(tileBounds, px); │ │ │ │ │ + │ │ │ │ │ + this.addTileMonitoringHooks(tile); │ │ │ │ │ + tile.draw(); │ │ │ │ │ + this.grid[0][0] = tile; │ │ │ │ │ + } else { │ │ │ │ │ + tile.moveTo(tileBounds, px); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //remove all but our single tile │ │ │ │ │ + this.removeExcessTiles(1, 1); │ │ │ │ │ + │ │ │ │ │ + // store the resolution of the grid │ │ │ │ │ + this.gridResolution = this.getServerResolution(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateGridLayout │ │ │ │ │ + * Generate parameters for the grid layout. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bound>|Object} OpenLayers.Bounds or an │ │ │ │ │ + * object with a 'left' and 'top' properties. │ │ │ │ │ + * origin - {<OpenLayers.LonLat>|Object} OpenLayers.LonLat or an │ │ │ │ │ + * object with a 'lon' and 'lat' properties. │ │ │ │ │ + * resolution - {Number} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Object containing properties tilelon, tilelat, startcol, │ │ │ │ │ + * startrow │ │ │ │ │ + */ │ │ │ │ │ + calculateGridLayout: function(bounds, origin, resolution) { │ │ │ │ │ + var tilelon = resolution * this.tileSize.w; │ │ │ │ │ + var tilelat = resolution * this.tileSize.h; │ │ │ │ │ + │ │ │ │ │ + var offsetlon = bounds.left - origin.lon; │ │ │ │ │ + var tilecol = Math.floor(offsetlon / tilelon) - this.buffer; │ │ │ │ │ + │ │ │ │ │ + var rowSign = this.rowSign; │ │ │ │ │ + │ │ │ │ │ + var offsetlat = rowSign * (origin.lat - bounds.top + tilelat); │ │ │ │ │ + var tilerow = Math[~rowSign ? 'floor' : 'ceil'](offsetlat / tilelat) - this.buffer * rowSign; │ │ │ │ │ + │ │ │ │ │ + return { │ │ │ │ │ + tilelon: tilelon, │ │ │ │ │ + tilelat: tilelat, │ │ │ │ │ + startcol: tilecol, │ │ │ │ │ + startrow: tilerow │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getTileOrigin │ │ │ │ │ + * Determine the origin for aligning the grid of tiles. If a <tileOrigin> │ │ │ │ │ + * property is supplied, that will be returned. Otherwise, the origin │ │ │ │ │ + * will be derived from the layer's <maxExtent> property. In this case, │ │ │ │ │ + * the tile origin will be the corner of the <maxExtent> given by the │ │ │ │ │ + * <tileOriginCorner> property. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.LonLat>} The tile origin. │ │ │ │ │ + */ │ │ │ │ │ + getTileOrigin: function() { │ │ │ │ │ + var origin = this.tileOrigin; │ │ │ │ │ + if (!origin) { │ │ │ │ │ + var extent = this.getMaxExtent(); │ │ │ │ │ + var edges = ({ │ │ │ │ │ + "tl": ["left", "top"], │ │ │ │ │ + "tr": ["right", "top"], │ │ │ │ │ + "bl": ["left", "bottom"], │ │ │ │ │ + "br": ["right", "bottom"] │ │ │ │ │ + })[this.tileOriginCorner]; │ │ │ │ │ + origin = new OpenLayers.LonLat(extent[edges[0]], extent[edges[1]]); │ │ │ │ │ + } │ │ │ │ │ + return origin; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getTileBoundsForGridIndex │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * row - {Number} The row of the grid │ │ │ │ │ + * col - {Number} The column of the grid │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} The bounds for the tile at (row, col) │ │ │ │ │ + */ │ │ │ │ │ + getTileBoundsForGridIndex: function(row, col) { │ │ │ │ │ + var origin = this.getTileOrigin(); │ │ │ │ │ + var tileLayout = this.gridLayout; │ │ │ │ │ + var tilelon = tileLayout.tilelon; │ │ │ │ │ + var tilelat = tileLayout.tilelat; │ │ │ │ │ + var startcol = tileLayout.startcol; │ │ │ │ │ + var startrow = tileLayout.startrow; │ │ │ │ │ + var rowSign = this.rowSign; │ │ │ │ │ + return new OpenLayers.Bounds( │ │ │ │ │ + origin.lon + (startcol + col) * tilelon, │ │ │ │ │ + origin.lat - (startrow + row * rowSign) * tilelat * rowSign, │ │ │ │ │ + origin.lon + (startcol + col + 1) * tilelon, │ │ │ │ │ + origin.lat - (startrow + (row - 1) * rowSign) * tilelat * rowSign │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: initGriddedTiles │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + */ │ │ │ │ │ + initGriddedTiles: function(bounds) { │ │ │ │ │ + this.events.triggerEvent("retile"); │ │ │ │ │ + │ │ │ │ │ + // work out mininum number of rows and columns; this is the number of │ │ │ │ │ + // tiles required to cover the viewport plus at least one for panning │ │ │ │ │ + │ │ │ │ │ + var viewSize = this.map.getSize(); │ │ │ │ │ + │ │ │ │ │ + var origin = this.getTileOrigin(); │ │ │ │ │ + var resolution = this.map.getResolution(), │ │ │ │ │ + serverResolution = this.getServerResolution(), │ │ │ │ │ + ratio = resolution / serverResolution, │ │ │ │ │ + tileSize = { │ │ │ │ │ + w: this.tileSize.w / ratio, │ │ │ │ │ + h: this.tileSize.h / ratio │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + var minRows = Math.ceil(viewSize.h / tileSize.h) + │ │ │ │ │ + 2 * this.buffer + 1; │ │ │ │ │ + var minCols = Math.ceil(viewSize.w / tileSize.w) + │ │ │ │ │ + 2 * this.buffer + 1; │ │ │ │ │ + │ │ │ │ │ + var tileLayout = this.calculateGridLayout(bounds, origin, serverResolution); │ │ │ │ │ + this.gridLayout = tileLayout; │ │ │ │ │ + │ │ │ │ │ + var tilelon = tileLayout.tilelon; │ │ │ │ │ + var tilelat = tileLayout.tilelat; │ │ │ │ │ + │ │ │ │ │ + var layerContainerDivLeft = this.map.layerContainerOriginPx.x; │ │ │ │ │ + var layerContainerDivTop = this.map.layerContainerOriginPx.y; │ │ │ │ │ + │ │ │ │ │ + var tileBounds = this.getTileBoundsForGridIndex(0, 0); │ │ │ │ │ + var startPx = this.map.getViewPortPxFromLonLat( │ │ │ │ │ + new OpenLayers.LonLat(tileBounds.left, tileBounds.top) │ │ │ │ │ + ); │ │ │ │ │ + startPx.x = Math.round(startPx.x) - layerContainerDivLeft; │ │ │ │ │ + startPx.y = Math.round(startPx.y) - layerContainerDivTop; │ │ │ │ │ + │ │ │ │ │ + var tileData = [], │ │ │ │ │ + center = this.map.getCenter(); │ │ │ │ │ + │ │ │ │ │ + var rowidx = 0; │ │ │ │ │ + do { │ │ │ │ │ + var row = this.grid[rowidx]; │ │ │ │ │ + if (!row) { │ │ │ │ │ + row = []; │ │ │ │ │ + this.grid.push(row); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var colidx = 0; │ │ │ │ │ + do { │ │ │ │ │ + tileBounds = this.getTileBoundsForGridIndex(rowidx, colidx); │ │ │ │ │ + var px = startPx.clone(); │ │ │ │ │ + px.x = px.x + colidx * Math.round(tileSize.w); │ │ │ │ │ + px.y = px.y + rowidx * Math.round(tileSize.h); │ │ │ │ │ + var tile = row[colidx]; │ │ │ │ │ + if (!tile) { │ │ │ │ │ + tile = this.addTile(tileBounds, px); │ │ │ │ │ + this.addTileMonitoringHooks(tile); │ │ │ │ │ + row.push(tile); │ │ │ │ │ } else { │ │ │ │ │ - containsOverlays = true; │ │ │ │ │ + tile.moveTo(tileBounds, px, false); │ │ │ │ │ } │ │ │ │ │ + var tileCenter = tileBounds.getCenterLonLat(); │ │ │ │ │ + tileData.push({ │ │ │ │ │ + tile: tile, │ │ │ │ │ + distance: Math.pow(tileCenter.lon - center.lon, 2) + │ │ │ │ │ + Math.pow(tileCenter.lat - center.lat, 2) │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - // only check a baselayer if it is *the* baselayer, check data │ │ │ │ │ - // layers if they are visible │ │ │ │ │ - var checked = (baseLayer) ? (layer == this.map.baseLayer) : │ │ │ │ │ - layer.getVisibility(); │ │ │ │ │ + colidx += 1; │ │ │ │ │ + } while ((tileBounds.right <= bounds.right + tilelon * this.buffer) || │ │ │ │ │ + colidx < minCols); │ │ │ │ │ │ │ │ │ │ - // create input element │ │ │ │ │ - var inputElem = document.createElement("input"), │ │ │ │ │ - // The input shall have an id attribute so we can use │ │ │ │ │ - // labels to interact with them. │ │ │ │ │ - inputId = OpenLayers.Util.createUniqueID( │ │ │ │ │ - this.id + "_input_" │ │ │ │ │ - ); │ │ │ │ │ + rowidx += 1; │ │ │ │ │ + } while ((tileBounds.bottom >= bounds.bottom - tilelat * this.buffer) || │ │ │ │ │ + rowidx < minRows); │ │ │ │ │ │ │ │ │ │ - inputElem.id = inputId; │ │ │ │ │ - inputElem.name = (baseLayer) ? this.id + "_baseLayers" : layer.name; │ │ │ │ │ - inputElem.type = (baseLayer) ? "radio" : "checkbox"; │ │ │ │ │ - inputElem.value = layer.name; │ │ │ │ │ - inputElem.checked = checked; │ │ │ │ │ - inputElem.defaultChecked = checked; │ │ │ │ │ - inputElem.className = "olButton"; │ │ │ │ │ - inputElem._layer = layer.id; │ │ │ │ │ - inputElem._layerSwitcher = this.id; │ │ │ │ │ + //shave off exceess rows and colums │ │ │ │ │ + this.removeExcessTiles(rowidx, colidx); │ │ │ │ │ │ │ │ │ │ - if (!baseLayer && !layer.inRange) { │ │ │ │ │ - inputElem.disabled = true; │ │ │ │ │ + var resolution = this.getServerResolution(); │ │ │ │ │ + // store the resolution of the grid │ │ │ │ │ + this.gridResolution = resolution; │ │ │ │ │ + │ │ │ │ │ + //now actually draw the tiles │ │ │ │ │ + tileData.sort(function(a, b) { │ │ │ │ │ + return a.distance - b.distance; │ │ │ │ │ + }); │ │ │ │ │ + for (var i = 0, ii = tileData.length; i < ii; ++i) { │ │ │ │ │ + tileData[i].tile.draw(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getMaxExtent │ │ │ │ │ + * Get this layer's maximum extent. (Implemented as a getter for │ │ │ │ │ + * potential specific implementations in sub-classes.) │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} │ │ │ │ │ + */ │ │ │ │ │ + getMaxExtent: function() { │ │ │ │ │ + return this.maxExtent; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: addTile │ │ │ │ │ + * Create a tile, initialize it, and add it to the layer div. │ │ │ │ │ + * │ │ │ │ │ + * Parameters │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * position - {<OpenLayers.Pixel>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Tile>} The added OpenLayers.Tile │ │ │ │ │ + */ │ │ │ │ │ + addTile: function(bounds, position) { │ │ │ │ │ + var tile = new this.tileClass( │ │ │ │ │ + this, position, bounds, null, this.tileSize, this.tileOptions │ │ │ │ │ + ); │ │ │ │ │ + this.events.triggerEvent("addtile", { │ │ │ │ │ + tile: tile │ │ │ │ │ + }); │ │ │ │ │ + return tile; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: addTileMonitoringHooks │ │ │ │ │ + * This function takes a tile as input and adds the appropriate hooks to │ │ │ │ │ + * the tile so that the layer can keep track of the loading tiles. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * tile - {<OpenLayers.Tile>} │ │ │ │ │ + */ │ │ │ │ │ + addTileMonitoringHooks: function(tile) { │ │ │ │ │ + │ │ │ │ │ + var replacingCls = 'olTileReplacing'; │ │ │ │ │ + │ │ │ │ │ + tile.onLoadStart = function() { │ │ │ │ │ + //if that was first tile then trigger a 'loadstart' on the layer │ │ │ │ │ + if (this.loading === false) { │ │ │ │ │ + this.loading = true; │ │ │ │ │ + this.events.triggerEvent("loadstart"); │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("tileloadstart", { │ │ │ │ │ + tile: tile │ │ │ │ │ + }); │ │ │ │ │ + this.numLoadingTiles++; │ │ │ │ │ + if (!this.singleTile && this.backBuffer && this.gridResolution === this.backBufferResolution) { │ │ │ │ │ + OpenLayers.Element.addClass(tile.getTile(), replacingCls); │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + tile.onLoadEnd = function(evt) { │ │ │ │ │ + this.numLoadingTiles--; │ │ │ │ │ + var aborted = evt.type === 'unload'; │ │ │ │ │ + this.events.triggerEvent("tileloaded", { │ │ │ │ │ + tile: tile, │ │ │ │ │ + aborted: aborted │ │ │ │ │ + }); │ │ │ │ │ + if (!this.singleTile && !aborted && this.backBuffer && this.gridResolution === this.backBufferResolution) { │ │ │ │ │ + var tileDiv = tile.getTile(); │ │ │ │ │ + if (OpenLayers.Element.getStyle(tileDiv, 'display') === 'none') { │ │ │ │ │ + var bufferTile = document.getElementById(tile.id + '_bb'); │ │ │ │ │ + if (bufferTile) { │ │ │ │ │ + bufferTile.parentNode.removeChild(bufferTile); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // create span │ │ │ │ │ - var labelSpan = document.createElement("label"); │ │ │ │ │ - // this isn't the DOM attribute 'for', but an arbitrary name we │ │ │ │ │ - // use to find the appropriate input element in <onButtonClick> │ │ │ │ │ - labelSpan["for"] = inputElem.id; │ │ │ │ │ - OpenLayers.Element.addClass(labelSpan, "labelSpan olButton"); │ │ │ │ │ - labelSpan._layer = layer.id; │ │ │ │ │ - labelSpan._layerSwitcher = this.id; │ │ │ │ │ - if (!baseLayer && !layer.inRange) { │ │ │ │ │ - labelSpan.style.color = "gray"; │ │ │ │ │ + OpenLayers.Element.removeClass(tileDiv, replacingCls); │ │ │ │ │ + } │ │ │ │ │ + //if that was the last tile, then trigger a 'loadend' on the layer │ │ │ │ │ + if (this.numLoadingTiles === 0) { │ │ │ │ │ + if (this.backBuffer) { │ │ │ │ │ + if (this.backBuffer.childNodes.length === 0) { │ │ │ │ │ + // no tiles transitioning, remove immediately │ │ │ │ │ + this.removeBackBuffer(); │ │ │ │ │ + } else { │ │ │ │ │ + // wait until transition has ended or delay has passed │ │ │ │ │ + this._transitionElement = aborted ? │ │ │ │ │ + this.div.lastChild : tile.imgDiv; │ │ │ │ │ + var transitionendEvents = this.transitionendEvents; │ │ │ │ │ + for (var i = transitionendEvents.length - 1; i >= 0; --i) { │ │ │ │ │ + OpenLayers.Event.observe(this._transitionElement, │ │ │ │ │ + transitionendEvents[i], │ │ │ │ │ + this._removeBackBuffer); │ │ │ │ │ + } │ │ │ │ │ + // the removal of the back buffer is delayed to prevent │ │ │ │ │ + // flash effects due to the animation of tile displaying │ │ │ │ │ + this.backBufferTimerId = window.setTimeout( │ │ │ │ │ + this._removeBackBuffer, this.removeBackBufferDelay │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - labelSpan.innerHTML = layer.name; │ │ │ │ │ - labelSpan.style.verticalAlign = (baseLayer) ? "bottom" : │ │ │ │ │ - "baseline"; │ │ │ │ │ - // create line break │ │ │ │ │ - var br = document.createElement("br"); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - var groupArray = (baseLayer) ? this.baseLayers : │ │ │ │ │ - this.dataLayers; │ │ │ │ │ - groupArray.push({ │ │ │ │ │ - 'layer': layer, │ │ │ │ │ - 'inputElem': inputElem, │ │ │ │ │ - 'labelSpan': labelSpan │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - var groupDiv = (baseLayer) ? this.baseLayersDiv : │ │ │ │ │ - this.dataLayersDiv; │ │ │ │ │ - groupDiv.appendChild(inputElem); │ │ │ │ │ - groupDiv.appendChild(labelSpan); │ │ │ │ │ - groupDiv.appendChild(br); │ │ │ │ │ + this.loading = false; │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - // if no overlays, dont display the overlay label │ │ │ │ │ - this.dataLbl.style.display = (containsOverlays) ? "" : "none"; │ │ │ │ │ + tile.onLoadError = function() { │ │ │ │ │ + this.events.triggerEvent("tileerror", { │ │ │ │ │ + tile: tile │ │ │ │ │ + }); │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - // if no baselayers, dont display the baselayer label │ │ │ │ │ - this.baseLbl.style.display = (containsBaseLayers) ? "" : "none"; │ │ │ │ │ + tile.events.on({ │ │ │ │ │ + "loadstart": tile.onLoadStart, │ │ │ │ │ + "loadend": tile.onLoadEnd, │ │ │ │ │ + "unload": tile.onLoadEnd, │ │ │ │ │ + "loaderror": tile.onLoadError, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return this.div; │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeTileMonitoringHooks │ │ │ │ │ + * This function takes a tile as input and removes the tile hooks │ │ │ │ │ + * that were added in addTileMonitoringHooks() │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * tile - {<OpenLayers.Tile>} │ │ │ │ │ + */ │ │ │ │ │ + removeTileMonitoringHooks: function(tile) { │ │ │ │ │ + tile.unload(); │ │ │ │ │ + tile.events.un({ │ │ │ │ │ + "loadstart": tile.onLoadStart, │ │ │ │ │ + "loadend": tile.onLoadEnd, │ │ │ │ │ + "unload": tile.onLoadEnd, │ │ │ │ │ + "loaderror": tile.onLoadError, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: updateMap │ │ │ │ │ - * Cycles through the loaded data and base layer input arrays and makes │ │ │ │ │ - * the necessary calls to the Map object such that that the map's │ │ │ │ │ - * visual state corresponds to what the user has selected in │ │ │ │ │ - * the control. │ │ │ │ │ + * Method: moveGriddedTiles │ │ │ │ │ */ │ │ │ │ │ - updateMap: function() { │ │ │ │ │ - │ │ │ │ │ - // set the newly selected base layer │ │ │ │ │ - for (var i = 0, len = this.baseLayers.length; i < len; i++) { │ │ │ │ │ - var layerEntry = this.baseLayers[i]; │ │ │ │ │ - if (layerEntry.inputElem.checked) { │ │ │ │ │ - this.map.setBaseLayer(layerEntry.layer, false); │ │ │ │ │ + moveGriddedTiles: function() { │ │ │ │ │ + var buffer = this.buffer + 1; │ │ │ │ │ + while (true) { │ │ │ │ │ + var tlTile = this.grid[0][0]; │ │ │ │ │ + var tlViewPort = { │ │ │ │ │ + x: tlTile.position.x + │ │ │ │ │ + this.map.layerContainerOriginPx.x, │ │ │ │ │ + y: tlTile.position.y + │ │ │ │ │ + this.map.layerContainerOriginPx.y │ │ │ │ │ + }; │ │ │ │ │ + var ratio = this.getServerResolution() / this.map.getResolution(); │ │ │ │ │ + var tileSize = { │ │ │ │ │ + w: Math.round(this.tileSize.w * ratio), │ │ │ │ │ + h: Math.round(this.tileSize.h * ratio) │ │ │ │ │ + }; │ │ │ │ │ + if (tlViewPort.x > -tileSize.w * (buffer - 1)) { │ │ │ │ │ + this.shiftColumn(true, tileSize); │ │ │ │ │ + } else if (tlViewPort.x < -tileSize.w * buffer) { │ │ │ │ │ + this.shiftColumn(false, tileSize); │ │ │ │ │ + } else if (tlViewPort.y > -tileSize.h * (buffer - 1)) { │ │ │ │ │ + this.shiftRow(true, tileSize); │ │ │ │ │ + } else if (tlViewPort.y < -tileSize.h * buffer) { │ │ │ │ │ + this.shiftRow(false, tileSize); │ │ │ │ │ + } else { │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // set the correct visibilities for the overlays │ │ │ │ │ - for (var i = 0, len = this.dataLayers.length; i < len; i++) { │ │ │ │ │ - var layerEntry = this.dataLayers[i]; │ │ │ │ │ - layerEntry.layer.setVisibility(layerEntry.inputElem.checked); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: maximizeControl │ │ │ │ │ - * Set up the labels and divs for the control │ │ │ │ │ + * Method: shiftRow │ │ │ │ │ + * Shifty grid work │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * e - {Event} │ │ │ │ │ + * prepend - {Boolean} if true, prepend to beginning. │ │ │ │ │ + * if false, then append to end │ │ │ │ │ + * tileSize - {Object} rendered tile size; object with w and h properties │ │ │ │ │ */ │ │ │ │ │ - maximizeControl: function(e) { │ │ │ │ │ - │ │ │ │ │ - // set the div's width and height to empty values, so │ │ │ │ │ - // the div dimensions can be controlled by CSS │ │ │ │ │ - this.div.style.width = ""; │ │ │ │ │ - this.div.style.height = ""; │ │ │ │ │ - │ │ │ │ │ - this.showControls(false); │ │ │ │ │ + shiftRow: function(prepend, tileSize) { │ │ │ │ │ + var grid = this.grid; │ │ │ │ │ + var rowIndex = prepend ? 0 : (grid.length - 1); │ │ │ │ │ + var sign = prepend ? -1 : 1; │ │ │ │ │ + var rowSign = this.rowSign; │ │ │ │ │ + var tileLayout = this.gridLayout; │ │ │ │ │ + tileLayout.startrow += sign * rowSign; │ │ │ │ │ │ │ │ │ │ - if (e != null) { │ │ │ │ │ - OpenLayers.Event.stop(e); │ │ │ │ │ + var modelRow = grid[rowIndex]; │ │ │ │ │ + var row = grid[prepend ? 'pop' : 'shift'](); │ │ │ │ │ + for (var i = 0, len = row.length; i < len; i++) { │ │ │ │ │ + var tile = row[i]; │ │ │ │ │ + var position = modelRow[i].position.clone(); │ │ │ │ │ + position.y += tileSize.h * sign; │ │ │ │ │ + tile.moveTo(this.getTileBoundsForGridIndex(rowIndex, i), position); │ │ │ │ │ } │ │ │ │ │ + grid[prepend ? 'unshift' : 'push'](row); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: minimizeControl │ │ │ │ │ - * Hide all the contents of the control, shrink the size, │ │ │ │ │ - * add the maximize icon │ │ │ │ │ + * Method: shiftColumn │ │ │ │ │ + * Shift grid work in the other dimension │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * e - {Event} │ │ │ │ │ + * prepend - {Boolean} if true, prepend to beginning. │ │ │ │ │ + * if false, then append to end │ │ │ │ │ + * tileSize - {Object} rendered tile size; object with w and h properties │ │ │ │ │ */ │ │ │ │ │ - minimizeControl: function(e) { │ │ │ │ │ - │ │ │ │ │ - // to minimize the control we set its div's width │ │ │ │ │ - // and height to 0px, we cannot just set "display" │ │ │ │ │ - // to "none" because it would hide the maximize │ │ │ │ │ - // div │ │ │ │ │ - this.div.style.width = "0px"; │ │ │ │ │ - this.div.style.height = "0px"; │ │ │ │ │ - │ │ │ │ │ - this.showControls(true); │ │ │ │ │ + shiftColumn: function(prepend, tileSize) { │ │ │ │ │ + var grid = this.grid; │ │ │ │ │ + var colIndex = prepend ? 0 : (grid[0].length - 1); │ │ │ │ │ + var sign = prepend ? -1 : 1; │ │ │ │ │ + var tileLayout = this.gridLayout; │ │ │ │ │ + tileLayout.startcol += sign; │ │ │ │ │ │ │ │ │ │ - if (e != null) { │ │ │ │ │ - OpenLayers.Event.stop(e); │ │ │ │ │ + for (var i = 0, len = grid.length; i < len; i++) { │ │ │ │ │ + var row = grid[i]; │ │ │ │ │ + var position = row[colIndex].position.clone(); │ │ │ │ │ + var tile = row[prepend ? 'pop' : 'shift'](); │ │ │ │ │ + position.x += tileSize.w * sign; │ │ │ │ │ + tile.moveTo(this.getTileBoundsForGridIndex(i, colIndex), position); │ │ │ │ │ + row[prepend ? 'unshift' : 'push'](tile); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: showControls │ │ │ │ │ - * Hide/Show all LayerSwitcher controls depending on whether we are │ │ │ │ │ - * minimized or not │ │ │ │ │ - * │ │ │ │ │ + * Method: removeExcessTiles │ │ │ │ │ + * When the size of the map or the buffer changes, we may need to │ │ │ │ │ + * remove some excess rows and columns. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * minimize - {Boolean} │ │ │ │ │ + * rows - {Integer} Maximum number of rows we want our grid to have. │ │ │ │ │ + * columns - {Integer} Maximum number of columns we want our grid to have. │ │ │ │ │ */ │ │ │ │ │ - showControls: function(minimize) { │ │ │ │ │ + removeExcessTiles: function(rows, columns) { │ │ │ │ │ + var i, l; │ │ │ │ │ │ │ │ │ │ - this.maximizeDiv.style.display = minimize ? "" : "none"; │ │ │ │ │ - this.minimizeDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ + // remove extra rows │ │ │ │ │ + while (this.grid.length > rows) { │ │ │ │ │ + var row = this.grid.pop(); │ │ │ │ │ + for (i = 0, l = row.length; i < l; i++) { │ │ │ │ │ + var tile = row[i]; │ │ │ │ │ + this.destroyTile(tile); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - this.layersDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ + // remove extra columns │ │ │ │ │ + for (i = 0, l = this.grid.length; i < l; i++) { │ │ │ │ │ + while (this.grid[i].length > columns) { │ │ │ │ │ + var row = this.grid[i]; │ │ │ │ │ + var tile = row.pop(); │ │ │ │ │ + this.destroyTile(tile); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: loadContents │ │ │ │ │ - * Set up the labels and divs for the control │ │ │ │ │ + * Method: onMapResize │ │ │ │ │ + * For singleTile layers, this will set a new tile size according to the │ │ │ │ │ + * dimensions of the map pane. │ │ │ │ │ */ │ │ │ │ │ - loadContents: function() { │ │ │ │ │ - │ │ │ │ │ - // layers list div │ │ │ │ │ - this.layersDiv = document.createElement("div"); │ │ │ │ │ - this.layersDiv.id = this.id + "_layersDiv"; │ │ │ │ │ - OpenLayers.Element.addClass(this.layersDiv, "layersDiv"); │ │ │ │ │ - │ │ │ │ │ - this.baseLbl = document.createElement("div"); │ │ │ │ │ - this.baseLbl.innerHTML = OpenLayers.i18n("Base Layer"); │ │ │ │ │ - OpenLayers.Element.addClass(this.baseLbl, "baseLbl"); │ │ │ │ │ - │ │ │ │ │ - this.baseLayersDiv = document.createElement("div"); │ │ │ │ │ - OpenLayers.Element.addClass(this.baseLayersDiv, "baseLayersDiv"); │ │ │ │ │ - │ │ │ │ │ - this.dataLbl = document.createElement("div"); │ │ │ │ │ - this.dataLbl.innerHTML = OpenLayers.i18n("Overlays"); │ │ │ │ │ - OpenLayers.Element.addClass(this.dataLbl, "dataLbl"); │ │ │ │ │ - │ │ │ │ │ - this.dataLayersDiv = document.createElement("div"); │ │ │ │ │ - OpenLayers.Element.addClass(this.dataLayersDiv, "dataLayersDiv"); │ │ │ │ │ - │ │ │ │ │ - if (this.ascending) { │ │ │ │ │ - this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ - } else { │ │ │ │ │ - this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + this.clearGrid(); │ │ │ │ │ + this.setTileSize(); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.div.appendChild(this.layersDiv); │ │ │ │ │ - │ │ │ │ │ - // maximize button div │ │ │ │ │ - var img = OpenLayers.Util.getImageLocation('layer-switcher-maximize.png'); │ │ │ │ │ - this.maximizeDiv = OpenLayers.Util.createAlphaImageDiv( │ │ │ │ │ - "OpenLayers_Control_MaximizeDiv", │ │ │ │ │ - null, │ │ │ │ │ - null, │ │ │ │ │ - img, │ │ │ │ │ - "absolute"); │ │ │ │ │ - OpenLayers.Element.addClass(this.maximizeDiv, "maximizeDiv olButton"); │ │ │ │ │ - this.maximizeDiv.style.display = "none"; │ │ │ │ │ - │ │ │ │ │ - this.div.appendChild(this.maximizeDiv); │ │ │ │ │ - │ │ │ │ │ - // minimize button div │ │ │ │ │ - var img = OpenLayers.Util.getImageLocation('layer-switcher-minimize.png'); │ │ │ │ │ - this.minimizeDiv = OpenLayers.Util.createAlphaImageDiv( │ │ │ │ │ - "OpenLayers_Control_MinimizeDiv", │ │ │ │ │ - null, │ │ │ │ │ - null, │ │ │ │ │ - img, │ │ │ │ │ - "absolute"); │ │ │ │ │ - OpenLayers.Element.addClass(this.minimizeDiv, "minimizeDiv olButton"); │ │ │ │ │ - this.minimizeDiv.style.display = "none"; │ │ │ │ │ - │ │ │ │ │ - this.div.appendChild(this.minimizeDiv); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getTileBounds │ │ │ │ │ + * Returns The tile bounds for a layer given a pixel location. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * viewPortPx - {<OpenLayers.Pixel>} The location in the viewport. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} Bounds of the tile at the given pixel location. │ │ │ │ │ + */ │ │ │ │ │ + getTileBounds: function(viewPortPx) { │ │ │ │ │ + var maxExtent = this.maxExtent; │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var tileMapWidth = resolution * this.tileSize.w; │ │ │ │ │ + var tileMapHeight = resolution * this.tileSize.h; │ │ │ │ │ + var mapPoint = this.getLonLatFromViewPortPx(viewPortPx); │ │ │ │ │ + var tileLeft = maxExtent.left + (tileMapWidth * │ │ │ │ │ + Math.floor((mapPoint.lon - │ │ │ │ │ + maxExtent.left) / │ │ │ │ │ + tileMapWidth)); │ │ │ │ │ + var tileBottom = maxExtent.bottom + (tileMapHeight * │ │ │ │ │ + Math.floor((mapPoint.lat - │ │ │ │ │ + maxExtent.bottom) / │ │ │ │ │ + tileMapHeight)); │ │ │ │ │ + return new OpenLayers.Bounds(tileLeft, tileBottom, │ │ │ │ │ + tileLeft + tileMapWidth, │ │ │ │ │ + tileBottom + tileMapHeight); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.LayerSwitcher" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Grid" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Renderer/Elements.js │ │ │ │ │ + OpenLayers/Layer/WMS.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Renderer.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.ElementsIndexer │ │ │ │ │ - * This class takes care of figuring out which order elements should be │ │ │ │ │ - * placed in the DOM based on given indexing methods. │ │ │ │ │ + * Class: OpenLayers.Layer.WMS │ │ │ │ │ + * Instances of OpenLayers.Layer.WMS are used to display data from OGC Web │ │ │ │ │ + * Mapping Services. Create a new WMS layer with the <OpenLayers.Layer.WMS> │ │ │ │ │ + * constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.ElementsIndexer = OpenLayers.Class({ │ │ │ │ │ +OpenLayers.Layer.WMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: maxZIndex │ │ │ │ │ - * {Integer} This is the largest-most z-index value for a node │ │ │ │ │ - * contained within the indexer. │ │ │ │ │ + * Constant: DEFAULT_PARAMS │ │ │ │ │ + * {Object} Hashtable of default parameter key/value pairs │ │ │ │ │ */ │ │ │ │ │ - maxZIndex: null, │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + service: "WMS", │ │ │ │ │ + version: "1.1.1", │ │ │ │ │ + request: "GetMap", │ │ │ │ │ + styles: "", │ │ │ │ │ + format: "image/jpeg" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: order │ │ │ │ │ - * {Array<String>} This is an array of node id's stored in the │ │ │ │ │ - * order that they should show up on screen. Id's higher up in the │ │ │ │ │ - * array (higher array index) represent nodes with higher z-indeces. │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} Default is true for WMS layer │ │ │ │ │ */ │ │ │ │ │ - order: null, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: indices │ │ │ │ │ - * {Object} This is a hash that maps node ids to their z-index value │ │ │ │ │ - * stored in the indexer. This is done to make finding a nodes z-index │ │ │ │ │ - * value O(1). │ │ │ │ │ + * APIProperty: encodeBBOX │ │ │ │ │ + * {Boolean} Should the BBOX commas be encoded? The WMS spec says 'no', │ │ │ │ │ + * but some services want it that way. Default false. │ │ │ │ │ */ │ │ │ │ │ - indices: null, │ │ │ │ │ + encodeBBOX: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: compare │ │ │ │ │ - * {Function} This is the function used to determine placement of │ │ │ │ │ - * of a new node within the indexer. If null, this defaults to to │ │ │ │ │ - * the Z_ORDER_DRAWING_ORDER comparison method. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: noMagic │ │ │ │ │ + * {Boolean} If true, the image format will not be automagicaly switched │ │ │ │ │ + * from image/jpeg to image/png or image/gif when using │ │ │ │ │ + * TRANSPARENT=TRUE. Also isBaseLayer will not changed by the │ │ │ │ │ + * constructor. Default false. │ │ │ │ │ */ │ │ │ │ │ - compare: null, │ │ │ │ │ + noMagic: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: initialize │ │ │ │ │ - * Create a new indexer with │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * yOrdering - {Boolean} Whether to use y-ordering. │ │ │ │ │ + * Property: yx │ │ │ │ │ + * {Object} Keys in this object are EPSG codes for which the axis order │ │ │ │ │ + * is to be reversed (yx instead of xy, LatLon instead of LonLat), with │ │ │ │ │ + * true as value. This is only relevant for WMS versions >= 1.3.0, and │ │ │ │ │ + * only if yx is not set in <OpenLayers.Projection.defaults> for the │ │ │ │ │ + * used projection. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(yOrdering) { │ │ │ │ │ - │ │ │ │ │ - this.compare = yOrdering ? │ │ │ │ │ - OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_Y_ORDER : │ │ │ │ │ - OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_DRAWING_ORDER; │ │ │ │ │ - │ │ │ │ │ - this.clear(); │ │ │ │ │ - }, │ │ │ │ │ + yx: {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: insert │ │ │ │ │ - * Insert a new node into the indexer. In order to find the correct │ │ │ │ │ - * positioning for the node to be inserted, this method uses a binary │ │ │ │ │ - * search. This makes inserting O(log(n)). │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Layer.WMS │ │ │ │ │ + * Create a new WMS layer object │ │ │ │ │ + * │ │ │ │ │ + * Examples: │ │ │ │ │ + * │ │ │ │ │ + * The code below creates a simple WMS layer using the image/jpeg format. │ │ │ │ │ + * (code) │ │ │ │ │ + * var wms = new OpenLayers.Layer.WMS("NASA Global Mosaic", │ │ │ │ │ + * "http://wms.jpl.nasa.gov/wms.cgi", │ │ │ │ │ + * {layers: "modis,global_mosaic"}); │ │ │ │ │ + * (end) │ │ │ │ │ + * Note the 3rd argument (params). Properties added to this object will be │ │ │ │ │ + * added to the WMS GetMap requests used for this layer's tiles. The only │ │ │ │ │ + * mandatory parameter is "layers". Other common WMS params include │ │ │ │ │ + * "transparent", "styles" and "format". Note that the "srs" param will │ │ │ │ │ + * always be ignored. Instead, it will be derived from the baseLayer's or │ │ │ │ │ + * map's projection. │ │ │ │ │ + * │ │ │ │ │ + * The code below creates a transparent WMS layer with additional options. │ │ │ │ │ + * (code) │ │ │ │ │ + * var wms = new OpenLayers.Layer.WMS("NASA Global Mosaic", │ │ │ │ │ + * "http://wms.jpl.nasa.gov/wms.cgi", │ │ │ │ │ + * { │ │ │ │ │ + * layers: "modis,global_mosaic", │ │ │ │ │ + * transparent: true │ │ │ │ │ + * }, { │ │ │ │ │ + * opacity: 0.5, │ │ │ │ │ + * singleTile: true │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * Note that by default, a WMS layer is configured as baseLayer. Setting │ │ │ │ │ + * the "transparent" param to true will apply some magic (see <noMagic>). │ │ │ │ │ + * The default image format changes from image/jpeg to image/png, and the │ │ │ │ │ + * layer is not configured as baseLayer. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * newNode - {DOMElement} The new node to be inserted. │ │ │ │ │ - * │ │ │ │ │ - * Returns │ │ │ │ │ - * {DOMElement} the node before which we should insert our newNode, or │ │ │ │ │ - * null if newNode can just be appended. │ │ │ │ │ + * name - {String} A name for the layer │ │ │ │ │ + * url - {String} Base url for the WMS │ │ │ │ │ + * (e.g. http://wms.jpl.nasa.gov/wms.cgi) │ │ │ │ │ + * params - {Object} An object with key/value pairs representing the │ │ │ │ │ + * GetMap query string parameters and parameter values. │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer. │ │ │ │ │ + * These options include all properties listed above, plus the ones │ │ │ │ │ + * inherited from superclasses. │ │ │ │ │ */ │ │ │ │ │ - insert: function(newNode) { │ │ │ │ │ - // If the node is known to the indexer, remove it so we can │ │ │ │ │ - // recalculate where it should go. │ │ │ │ │ - if (this.exists(newNode)) { │ │ │ │ │ - this.remove(newNode); │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + //uppercase params │ │ │ │ │ + params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ + if (parseFloat(params.VERSION) >= 1.3 && !params.EXCEPTIONS) { │ │ │ │ │ + params.EXCEPTIONS = "INIMAGE"; │ │ │ │ │ } │ │ │ │ │ + newArguments.push(name, url, params, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, │ │ │ │ │ + OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS) │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - var nodeId = newNode.id; │ │ │ │ │ - │ │ │ │ │ - this.determineZIndex(newNode); │ │ │ │ │ - │ │ │ │ │ - var leftIndex = -1; │ │ │ │ │ - var rightIndex = this.order.length; │ │ │ │ │ - var middle; │ │ │ │ │ │ │ │ │ │ - while (rightIndex - leftIndex > 1) { │ │ │ │ │ - middle = parseInt((leftIndex + rightIndex) / 2); │ │ │ │ │ + //layer is transparent │ │ │ │ │ + if (!this.noMagic && this.params.TRANSPARENT && │ │ │ │ │ + this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ │ │ │ │ │ - var placement = this.compare(this, newNode, │ │ │ │ │ - OpenLayers.Util.getElement(this.order[middle])); │ │ │ │ │ + // unless explicitly set in options, make layer an overlay │ │ │ │ │ + if ((options == null) || (!options.isBaseLayer)) { │ │ │ │ │ + this.isBaseLayer = false; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (placement > 0) { │ │ │ │ │ - leftIndex = middle; │ │ │ │ │ - } else { │ │ │ │ │ - rightIndex = middle; │ │ │ │ │ + // jpegs can never be transparent, so intelligently switch the │ │ │ │ │ + // format, depending on the browser's capabilities │ │ │ │ │ + if (this.params.FORMAT == "image/jpeg") { │ │ │ │ │ + this.params.FORMAT = OpenLayers.Util.alphaHack() ? "image/gif" : │ │ │ │ │ + "image/png"; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - this.order.splice(rightIndex, 0, nodeId); │ │ │ │ │ - this.indices[nodeId] = this.getZIndex(newNode); │ │ │ │ │ - │ │ │ │ │ - // If the new node should be before another in the index │ │ │ │ │ - // order, return the node before which we have to insert the new one; │ │ │ │ │ - // else, return null to indicate that the new node can be appended. │ │ │ │ │ - return this.getNextElement(rightIndex); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: remove │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} The node to be removed. │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.WMS>} An exact clone of this layer │ │ │ │ │ */ │ │ │ │ │ - remove: function(node) { │ │ │ │ │ - var nodeId = node.id; │ │ │ │ │ - var arrayIndex = OpenLayers.Util.indexOf(this.order, nodeId); │ │ │ │ │ - if (arrayIndex >= 0) { │ │ │ │ │ - // Remove it from the order array, as well as deleting the node │ │ │ │ │ - // from the indeces hash. │ │ │ │ │ - this.order.splice(arrayIndex, 1); │ │ │ │ │ - delete this.indices[nodeId]; │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ │ │ │ │ │ - // Reset the maxium z-index based on the last item in the │ │ │ │ │ - // order array. │ │ │ │ │ - if (this.order.length > 0) { │ │ │ │ │ - var lastId = this.order[this.order.length - 1]; │ │ │ │ │ - this.maxZIndex = this.indices[lastId]; │ │ │ │ │ - } else { │ │ │ │ │ - this.maxZIndex = 0; │ │ │ │ │ - } │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.WMS(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.params, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clear │ │ │ │ │ + * APIMethod: reverseAxisOrder │ │ │ │ │ + * Returns true if the axis order is reversed for the WMS version and │ │ │ │ │ + * projection of the layer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the axis order is reversed, false otherwise. │ │ │ │ │ */ │ │ │ │ │ - clear: function() { │ │ │ │ │ - this.order = []; │ │ │ │ │ - this.indices = {}; │ │ │ │ │ - this.maxZIndex = 0; │ │ │ │ │ + reverseAxisOrder: function() { │ │ │ │ │ + var projCode = this.projection.getCode(); │ │ │ │ │ + return parseFloat(this.params.VERSION) >= 1.3 && │ │ │ │ │ + !!(this.yx[projCode] || (OpenLayers.Projection.defaults[projCode] && │ │ │ │ │ + OpenLayers.Projection.defaults[projCode].yx)); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: exists │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * Return a GetMap query string for this layer │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} The node to test for existence. │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox for the │ │ │ │ │ + * request. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Whether or not the node exists in the indexer? │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as │ │ │ │ │ + * parameters. │ │ │ │ │ */ │ │ │ │ │ - exists: function(node) { │ │ │ │ │ - return (this.indices[node.id] != null); │ │ │ │ │ - }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getZIndex │ │ │ │ │ - * Get the z-index value for the current node from the node data itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} The node whose z-index to get. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} The z-index value for the specified node (from the node │ │ │ │ │ - * data itself). │ │ │ │ │ - */ │ │ │ │ │ - getZIndex: function(node) { │ │ │ │ │ - return node._style.graphicZIndex; │ │ │ │ │ + var imageSize = this.getImageSize(); │ │ │ │ │ + var newParams = {}; │ │ │ │ │ + // WMS 1.3 introduced axis order │ │ │ │ │ + var reverseAxisOrder = this.reverseAxisOrder(); │ │ │ │ │ + newParams.BBOX = this.encodeBBOX ? │ │ │ │ │ + bounds.toBBOX(null, reverseAxisOrder) : │ │ │ │ │ + bounds.toArray(reverseAxisOrder); │ │ │ │ │ + newParams.WIDTH = imageSize.w; │ │ │ │ │ + newParams.HEIGHT = imageSize.h; │ │ │ │ │ + var requestString = this.getFullRequestString(newParams); │ │ │ │ │ + return requestString; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: determineZIndex │ │ │ │ │ - * Determine the z-index for the current node if there isn't one, │ │ │ │ │ - * and set the maximum value if we've found a new maximum. │ │ │ │ │ + * APIMethod: mergeNewParams │ │ │ │ │ + * Catch changeParams and uppercase the new params to be merged in │ │ │ │ │ + * before calling changeParams on the super class. │ │ │ │ │ + * │ │ │ │ │ + * Once params have been changed, the tiles will be reloaded with │ │ │ │ │ + * the new parameters. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ + * newParams - {Object} Hashtable of new params to use │ │ │ │ │ */ │ │ │ │ │ - determineZIndex: function(node) { │ │ │ │ │ - var zIndex = node._style.graphicZIndex; │ │ │ │ │ - │ │ │ │ │ - // Everything must have a zIndex. If none is specified, │ │ │ │ │ - // this means the user *must* (hint: assumption) want this │ │ │ │ │ - // node to succomb to drawing order. To enforce drawing order │ │ │ │ │ - // over all indexing methods, we'll create a new z-index that's │ │ │ │ │ - // greater than any currently in the indexer. │ │ │ │ │ - if (zIndex == null) { │ │ │ │ │ - zIndex = this.maxZIndex; │ │ │ │ │ - node._style.graphicZIndex = zIndex; │ │ │ │ │ - } else if (zIndex > this.maxZIndex) { │ │ │ │ │ - this.maxZIndex = zIndex; │ │ │ │ │ - } │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ + var newArguments = [upperParams]; │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, │ │ │ │ │ + newArguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getNextElement │ │ │ │ │ - * Get the next element in the order stack. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getFullRequestString │ │ │ │ │ + * Combine the layer's url with its params and these newParams. │ │ │ │ │ + * │ │ │ │ │ + * Add the SRS parameter from projection -- this is probably │ │ │ │ │ + * more eloquently done via a setProjection() method, but this │ │ │ │ │ + * works for now and always. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * index - {Integer} The index of the current node in this.order. │ │ │ │ │ + * newParams - {Object} │ │ │ │ │ + * altUrl - {String} Use this as the url instead of the layer's url │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} the node following the index passed in, or │ │ │ │ │ - * null. │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - getNextElement: function(index) { │ │ │ │ │ - var nextIndex = index + 1; │ │ │ │ │ - if (nextIndex < this.order.length) { │ │ │ │ │ - var nextElement = OpenLayers.Util.getElement(this.order[nextIndex]); │ │ │ │ │ - if (nextElement == undefined) { │ │ │ │ │ - nextElement = this.getNextElement(nextIndex); │ │ │ │ │ - } │ │ │ │ │ - return nextElement; │ │ │ │ │ + getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ + var mapProjection = this.map.getProjectionObject(); │ │ │ │ │ + var projectionCode = this.projection && this.projection.equals(mapProjection) ? │ │ │ │ │ + this.projection.getCode() : │ │ │ │ │ + mapProjection.getCode(); │ │ │ │ │ + var value = (projectionCode == "none") ? null : projectionCode; │ │ │ │ │ + if (parseFloat(this.params.VERSION) >= 1.3) { │ │ │ │ │ + this.params.CRS = value; │ │ │ │ │ } else { │ │ │ │ │ - return null; │ │ │ │ │ + this.params.SRS = value; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (typeof this.params.TRANSPARENT == "boolean") { │ │ │ │ │ + newParams.TRANSPARENT = this.params.TRANSPARENT ? "TRUE" : "FALSE"; │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.getFullRequestString.apply( │ │ │ │ │ + this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.ElementsIndexer" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.WMS" │ │ │ │ │ }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/XYZ.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.ElementsIndexer.IndexingMethods │ │ │ │ │ - * These are the compare methods for figuring out where a new node should be │ │ │ │ │ - * placed within the indexer. These methods are very similar to general │ │ │ │ │ - * sorting methods in that they return -1, 0, and 1 to specify the │ │ │ │ │ - * direction in which new nodes fall in the ordering. │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.ElementsIndexer.IndexingMethods = { │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.XYZ │ │ │ │ │ + * The XYZ class is designed to make it easier for people who have tiles │ │ │ │ │ + * arranged by a standard XYZ grid. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.XYZ = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: Z_ORDER │ │ │ │ │ - * This compare method is used by other comparison methods. │ │ │ │ │ - * It can be used individually for ordering, but is not recommended, │ │ │ │ │ - * because it doesn't subscribe to drawing order. │ │ │ │ │ - * │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * Default is true, as this is designed to be a base tile source. │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: sphericalMercator │ │ │ │ │ + * Whether the tile extents should be set to the defaults for │ │ │ │ │ + * spherical mercator. Useful for things like OpenStreetMap. │ │ │ │ │ + * Default is false, except for the OSM subclass. │ │ │ │ │ + */ │ │ │ │ │ + sphericalMercator: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomOffset │ │ │ │ │ + * {Number} If your cache has more zoom levels than you want to provide │ │ │ │ │ + * access to with this layer, supply a zoomOffset. This zoom offset │ │ │ │ │ + * is added to the current map zoom level to determine the level │ │ │ │ │ + * for a requested tile. For example, if you supply a zoomOffset │ │ │ │ │ + * of 3, when the map is at the zoom 0, tiles will be requested from │ │ │ │ │ + * level 3 of your cache. Default is 0 (assumes cache level and map │ │ │ │ │ + * zoom are equivalent). Using <zoomOffset> is an alternative to │ │ │ │ │ + * setting <serverResolutions> if you only want to expose a subset │ │ │ │ │ + * of the server resolutions. │ │ │ │ │ + */ │ │ │ │ │ + zoomOffset: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: serverResolutions │ │ │ │ │ + * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ + * property if the map resolutions differ from the server. This │ │ │ │ │ + * property serves two purposes. (a) <serverResolutions> can include │ │ │ │ │ + * resolutions that the server supports and that you don't want to │ │ │ │ │ + * provide with this layer; you can also look at <zoomOffset>, which is │ │ │ │ │ + * an alternative to <serverResolutions> for that specific purpose. │ │ │ │ │ + * (b) The map can work with resolutions that aren't supported by │ │ │ │ │ + * the server, i.e. that aren't in <serverResolutions>. When the │ │ │ │ │ + * map is displayed in such a resolution data for the closest │ │ │ │ │ + * server-supported resolution is loaded and the layer div is │ │ │ │ │ + * stretched as necessary. │ │ │ │ │ + */ │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.XYZ │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ - * newNode - {DOMElement} │ │ │ │ │ - * nextNode - {DOMElement} │ │ │ │ │ + * name - {String} │ │ │ │ │ + * url - {String} │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + if (options && options.sphericalMercator || this.sphericalMercator) { │ │ │ │ │ + options = OpenLayers.Util.extend({ │ │ │ │ │ + projection: "EPSG:900913", │ │ │ │ │ + numZoomLevels: 19 │ │ │ │ │ + }, options); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [ │ │ │ │ │ + name || this.name, url || this.url, {}, │ │ │ │ │ + options │ │ │ │ │ + ]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} Is this ever used? │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Integer} │ │ │ │ │ + * {<OpenLayers.Layer.XYZ>} An exact clone of this OpenLayers.Layer.XYZ │ │ │ │ │ */ │ │ │ │ │ - Z_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var newZIndex = indexer.getZIndex(newNode); │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ │ │ │ │ │ - var returnVal = 0; │ │ │ │ │ - if (nextNode) { │ │ │ │ │ - var nextZIndex = indexer.getZIndex(nextNode); │ │ │ │ │ - returnVal = newZIndex - nextZIndex; │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.XYZ(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return returnVal; │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: Z_ORDER_DRAWING_ORDER │ │ │ │ │ - * This method orders nodes by their z-index, but does so in a way │ │ │ │ │ - * that, if there are other nodes with the same z-index, the newest │ │ │ │ │ - * drawn will be the front most within that z-index. This is the │ │ │ │ │ - * default indexing method. │ │ │ │ │ - * │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ - * newNode - {DOMElement} │ │ │ │ │ - * nextNode - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Integer} │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as │ │ │ │ │ + * parameters │ │ │ │ │ */ │ │ │ │ │ - Z_ORDER_DRAWING_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER( │ │ │ │ │ - indexer, │ │ │ │ │ - newNode, │ │ │ │ │ - nextNode │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - // Make Z_ORDER subscribe to drawing order by pushing it above │ │ │ │ │ - // all of the other nodes with the same z-index. │ │ │ │ │ - if (nextNode && returnVal == 0) { │ │ │ │ │ - returnVal = 1; │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var xyz = this.getXYZ(bounds); │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + var s = '' + xyz.x + xyz.y + xyz.z; │ │ │ │ │ + url = this.selectUrl(s, url); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return returnVal; │ │ │ │ │ + return OpenLayers.String.format(url, xyz); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: Z_ORDER_Y_ORDER │ │ │ │ │ - * This one should really be called Z_ORDER_Y_ORDER_DRAWING_ORDER, as it │ │ │ │ │ - * best describes which ordering methods have precedence (though, the │ │ │ │ │ - * name would be too long). This method orders nodes by their z-index, │ │ │ │ │ - * but does so in a way that, if there are other nodes with the same │ │ │ │ │ - * z-index, the nodes with the lower y position will be "closer" than │ │ │ │ │ - * those with a higher y position. If two nodes have the exact same y │ │ │ │ │ - * position, however, then this method will revert to using drawing │ │ │ │ │ - * order to decide placement. │ │ │ │ │ - * │ │ │ │ │ + * Method: getXYZ │ │ │ │ │ + * Calculates x, y and z for the given bounds. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ - * newNode - {DOMElement} │ │ │ │ │ - * nextNode - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Integer} │ │ │ │ │ + * {Object} - an object with x, y and z properties. │ │ │ │ │ */ │ │ │ │ │ - Z_ORDER_Y_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER( │ │ │ │ │ - indexer, │ │ │ │ │ - newNode, │ │ │ │ │ - nextNode │ │ │ │ │ - ); │ │ │ │ │ + getXYZ: function(bounds) { │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.maxExtent.left) / │ │ │ │ │ + (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((this.maxExtent.top - bounds.top) / │ │ │ │ │ + (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getServerZoom(); │ │ │ │ │ │ │ │ │ │ - if (nextNode && returnVal === 0) { │ │ │ │ │ - var result = nextNode._boundsBottom - newNode._boundsBottom; │ │ │ │ │ - returnVal = (result === 0) ? 1 : result; │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + var limit = Math.pow(2, z); │ │ │ │ │ + x = ((x % limit) + limit) % limit; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return returnVal; │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ + return { │ │ │ │ │ + 'x': x, │ │ │ │ │ + 'y': y, │ │ │ │ │ + 'z': z │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /* APIMethod: setMap │ │ │ │ │ + * When the layer is added to a map, then we can fetch our origin │ │ │ │ │ + * (if we don't have one.) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (!this.tileOrigin) { │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, │ │ │ │ │ + this.maxExtent.bottom); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.XYZ" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/Bing.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Renderer.Elements │ │ │ │ │ - * This is another virtual class in that it should never be instantiated by │ │ │ │ │ - * itself as a Renderer. It exists because there is *tons* of shared │ │ │ │ │ - * functionality between different vector libraries which use nodes/elements │ │ │ │ │ - * as a base for rendering vectors. │ │ │ │ │ - * │ │ │ │ │ - * The highlevel bits of code that are implemented here are the adding and │ │ │ │ │ - * removing of geometries, which is essentially the same for any │ │ │ │ │ - * element-based renderer. The details of creating each node and drawing the │ │ │ │ │ - * paths are of course different, but the machinery is the same. │ │ │ │ │ - * │ │ │ │ │ - * Inherits: │ │ │ │ │ - * - <OpenLayers.Renderer> │ │ │ │ │ + * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Renderer.Elements = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: rendererRoot │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - rendererRoot: null, │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.Bing │ │ │ │ │ + * Bing layer using direct tile access as provided by Bing Maps REST Services. │ │ │ │ │ + * See http://msdn.microsoft.com/en-us/library/ff701713.aspx for more │ │ │ │ │ + * information. Note: Terms of Service compliant use requires the map to be │ │ │ │ │ + * configured with an <OpenLayers.Control.Attribution> control and the │ │ │ │ │ + * attribution placed on or near the map. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.XYZ> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Bing = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: root │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Property: key │ │ │ │ │ + * {String} API key for Bing maps, get your own key │ │ │ │ │ + * at http://bingmapsportal.com/ . │ │ │ │ │ */ │ │ │ │ │ - root: null, │ │ │ │ │ + key: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: vectorRoot │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Property: serverResolutions │ │ │ │ │ + * {Array} the resolutions provided by the Bing servers. │ │ │ │ │ */ │ │ │ │ │ - vectorRoot: null, │ │ │ │ │ + serverResolutions: [ │ │ │ │ │ + 156543.03390625, 78271.516953125, 39135.7584765625, │ │ │ │ │ + 19567.87923828125, 9783.939619140625, 4891.9698095703125, │ │ │ │ │ + 2445.9849047851562, 1222.9924523925781, 611.4962261962891, │ │ │ │ │ + 305.74811309814453, 152.87405654907226, 76.43702827453613, │ │ │ │ │ + 38.218514137268066, 19.109257068634033, 9.554628534317017, │ │ │ │ │ + 4.777314267158508, 2.388657133579254, 1.194328566789627, │ │ │ │ │ + 0.5971642833948135, 0.29858214169740677, 0.14929107084870338, │ │ │ │ │ + 0.07464553542435169 │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: textRoot │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Property: attributionTemplate │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - textRoot: null, │ │ │ │ │ + attributionTemplate: '<span class="olBingAttribution ${type}">' + │ │ │ │ │ + '<div><a target="_blank" href="http://www.bing.com/maps/">' + │ │ │ │ │ + '<img src="${logo}" /></a></div>${copyrights}' + │ │ │ │ │ + '<a style="white-space: nowrap" target="_blank" ' + │ │ │ │ │ + 'href="http://www.microsoft.com/maps/product/terms.html">' + │ │ │ │ │ + 'Terms of Use</a></span>', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: xmlns │ │ │ │ │ - * {String} │ │ │ │ │ + * Property: metadata │ │ │ │ │ + * {Object} Metadata for this layer, as returned by the callback script │ │ │ │ │ */ │ │ │ │ │ - xmlns: null, │ │ │ │ │ + metadata: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: xOffset │ │ │ │ │ - * {Number} Offset to apply to the renderer viewport translation in x │ │ │ │ │ - * direction. If the renderer extent's center is on the right of the │ │ │ │ │ - * dateline (i.e. exceeds the world bounds), we shift the viewport to the │ │ │ │ │ - * left by one world width. This avoids that features disappear from the │ │ │ │ │ - * map viewport. Because our dateline handling logic in other places │ │ │ │ │ - * ensures that extents crossing the dateline always have a center │ │ │ │ │ - * exceeding the world bounds on the left, we need this offset to make sure │ │ │ │ │ - * that the same is true for the renderer extent in pixel space as well. │ │ │ │ │ + * Property: protocolRegex │ │ │ │ │ + * {RegExp} Regular expression to match and replace http: in bing urls │ │ │ │ │ */ │ │ │ │ │ - xOffset: 0, │ │ │ │ │ + protocolRegex: /^http:/i, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: rightOfDateLine │ │ │ │ │ - * {Boolean} Keeps track of the location of the map extent relative to the │ │ │ │ │ - * date line. The <setExtent> method compares this value (which is the one │ │ │ │ │ - * from the previous <setExtent> call) with the current position of the map │ │ │ │ │ - * extent relative to the date line and updates the xOffset when the extent │ │ │ │ │ - * has moved from one side of the date line to the other. │ │ │ │ │ + * APIProperty: type │ │ │ │ │ + * {String} The layer identifier. Any non-birdseye imageryType │ │ │ │ │ + * from http://msdn.microsoft.com/en-us/library/ff701716.aspx can be │ │ │ │ │ + * used. Default is "Road". │ │ │ │ │ */ │ │ │ │ │ + type: "Road", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: Indexer │ │ │ │ │ - * {<OpenLayers.ElementIndexer>} An instance of OpenLayers.ElementsIndexer │ │ │ │ │ - * created upon initialization if the zIndexing or yOrdering options │ │ │ │ │ - * passed to this renderer's constructor are set to true. │ │ │ │ │ + * APIProperty: culture │ │ │ │ │ + * {String} The culture identifier. See http://msdn.microsoft.com/en-us/library/ff701709.aspx │ │ │ │ │ + * for the definition and the possible values. Default is "en-US". │ │ │ │ │ */ │ │ │ │ │ - indexer: null, │ │ │ │ │ + culture: "en-US", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: BACKGROUND_ID_SUFFIX │ │ │ │ │ - * {String} │ │ │ │ │ + * APIProperty: metadataParams │ │ │ │ │ + * {Object} Optional url parameters for the Get Imagery Metadata request │ │ │ │ │ + * as described here: http://msdn.microsoft.com/en-us/library/ff701716.aspx │ │ │ │ │ */ │ │ │ │ │ - BACKGROUND_ID_SUFFIX: "_background", │ │ │ │ │ + metadataParams: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: LABEL_ID_SUFFIX │ │ │ │ │ - * {String} │ │ │ │ │ + /** APIProperty: tileOptions │ │ │ │ │ + * {Object} optional configuration options for <OpenLayers.Tile> instances │ │ │ │ │ + * created by this Layer. Default is │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * {crossOriginKeyword: 'anonymous'} │ │ │ │ │ + * (end) │ │ │ │ │ */ │ │ │ │ │ - LABEL_ID_SUFFIX: "_label", │ │ │ │ │ + tileOptions: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: LABEL_OUTLINE_SUFFIX │ │ │ │ │ - * {String} │ │ │ │ │ + /** APIProperty: protocol │ │ │ │ │ + * {String} Protocol to use to fetch Imagery Metadata, tiles and bing logo │ │ │ │ │ + * Can be 'http:' 'https:' or '' │ │ │ │ │ + * │ │ │ │ │ + * Warning: tiles may not be available under both HTTP and HTTPS protocols. │ │ │ │ │ + * Microsoft approved use of both HTTP and HTTPS urls for tiles. However │ │ │ │ │ + * this is undocumented and the Imagery Metadata API always returns HTTP │ │ │ │ │ + * urls. │ │ │ │ │ + * │ │ │ │ │ + * Default is '', unless when executed from a file:/// uri, in which case │ │ │ │ │ + * it is 'http:'. │ │ │ │ │ */ │ │ │ │ │ - LABEL_OUTLINE_SUFFIX: "_outline", │ │ │ │ │ + protocol: ~window.location.href.indexOf('http') ? '' : 'http:', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Renderer.Elements │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Layer.Bing │ │ │ │ │ + * Create a new Bing layer. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var road = new OpenLayers.Layer.Bing({ │ │ │ │ │ + * name: "My Bing Aerial Layer", │ │ │ │ │ + * type: "Aerial", │ │ │ │ │ + * key: "my-api-key-here", │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * containerID - {String} │ │ │ │ │ - * options - {Object} options for this renderer. │ │ │ │ │ + * options - {Object} Configuration properties for the layer. │ │ │ │ │ * │ │ │ │ │ - * Supported options are: │ │ │ │ │ - * yOrdering - {Boolean} Whether to use y-ordering │ │ │ │ │ - * zIndexing - {Boolean} Whether to use z-indexing. Will be ignored │ │ │ │ │ - * if yOrdering is set to true. │ │ │ │ │ + * Required configuration properties: │ │ │ │ │ + * key - {String} Bing Maps API key for your application. Get one at │ │ │ │ │ + * http://bingmapsportal.com/. │ │ │ │ │ + * type - {String} The layer identifier. Any non-birdseye imageryType │ │ │ │ │ + * from http://msdn.microsoft.com/en-us/library/ff701716.aspx can be │ │ │ │ │ + * used. │ │ │ │ │ + * │ │ │ │ │ + * Any other documented layer properties can be provided in the config object. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(containerID, options) { │ │ │ │ │ - OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - this.rendererRoot = this.createRenderRoot(); │ │ │ │ │ - this.root = this.createRoot("_root"); │ │ │ │ │ - this.vectorRoot = this.createRoot("_vroot"); │ │ │ │ │ - this.textRoot = this.createRoot("_troot"); │ │ │ │ │ - │ │ │ │ │ - this.root.appendChild(this.vectorRoot); │ │ │ │ │ - this.root.appendChild(this.textRoot); │ │ │ │ │ - │ │ │ │ │ - this.rendererRoot.appendChild(this.root); │ │ │ │ │ - this.container.appendChild(this.rendererRoot); │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + sphericalMercator: true │ │ │ │ │ + }, options); │ │ │ │ │ + var name = options.name || "Bing " + (options.type || this.type); │ │ │ │ │ │ │ │ │ │ - if (options && (options.zIndexing || options.yOrdering)) { │ │ │ │ │ - this.indexer = new OpenLayers.ElementsIndexer(options.yOrdering); │ │ │ │ │ - } │ │ │ │ │ + var newArgs = [name, null, options]; │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, newArgs); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + crossOriginKeyword: 'anonymous' │ │ │ │ │ + }, this.options.tileOptions); │ │ │ │ │ + this.loadMetadata(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * Method: loadMetadata │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - │ │ │ │ │ - this.clear(); │ │ │ │ │ - │ │ │ │ │ - this.rendererRoot = null; │ │ │ │ │ - this.root = null; │ │ │ │ │ - this.xmlns = null; │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Renderer.prototype.destroy.apply(this, arguments); │ │ │ │ │ + loadMetadata: function() { │ │ │ │ │ + this._callbackId = "_callback_" + this.id.replace(/\./g, "_"); │ │ │ │ │ + // link the processMetadata method to the global scope and bind it │ │ │ │ │ + // to this instance │ │ │ │ │ + window[this._callbackId] = OpenLayers.Function.bind( │ │ │ │ │ + OpenLayers.Layer.Bing.processMetadata, this │ │ │ │ │ + ); │ │ │ │ │ + var params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + key: this.key, │ │ │ │ │ + jsonp: this._callbackId, │ │ │ │ │ + include: "ImageryProviders" │ │ │ │ │ + }, this.metadataParams); │ │ │ │ │ + var url = this.protocol + "//dev.virtualearth.net/REST/v1/Imagery/Metadata/" + │ │ │ │ │ + this.type + "?" + OpenLayers.Util.getParameterString(params); │ │ │ │ │ + var script = document.createElement("script"); │ │ │ │ │ + script.type = "text/javascript"; │ │ │ │ │ + script.src = url; │ │ │ │ │ + script.id = this._callbackId; │ │ │ │ │ + document.getElementsByTagName("head")[0].appendChild(script); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clear │ │ │ │ │ - * Remove all the elements from the root │ │ │ │ │ + * Method: initLayer │ │ │ │ │ + * │ │ │ │ │ + * Sets layer properties according to the metadata provided by the API │ │ │ │ │ */ │ │ │ │ │ - clear: function() { │ │ │ │ │ - var child; │ │ │ │ │ - var root = this.vectorRoot; │ │ │ │ │ - if (root) { │ │ │ │ │ - while (child = root.firstChild) { │ │ │ │ │ - root.removeChild(child); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - root = this.textRoot; │ │ │ │ │ - if (root) { │ │ │ │ │ - while (child = root.firstChild) { │ │ │ │ │ - root.removeChild(child); │ │ │ │ │ - } │ │ │ │ │ + initLayer: function() { │ │ │ │ │ + var res = this.metadata.resourceSets[0].resources[0]; │ │ │ │ │ + var url = res.imageUrl.replace("{quadkey}", "${quadkey}"); │ │ │ │ │ + url = url.replace("{culture}", this.culture); │ │ │ │ │ + url = url.replace(this.protocolRegex, this.protocol); │ │ │ │ │ + this.url = []; │ │ │ │ │ + for (var i = 0; i < res.imageUrlSubdomains.length; ++i) { │ │ │ │ │ + this.url.push(url.replace("{subdomain}", res.imageUrlSubdomains[i])); │ │ │ │ │ } │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - this.indexer.clear(); │ │ │ │ │ + this.addOptions({ │ │ │ │ │ + maxResolution: Math.min( │ │ │ │ │ + this.serverResolutions[res.zoomMin], │ │ │ │ │ + this.maxResolution || Number.POSITIVE_INFINITY │ │ │ │ │ + ), │ │ │ │ │ + numZoomLevels: Math.min( │ │ │ │ │ + res.zoomMax + 1 - res.zoomMin, this.numZoomLevels │ │ │ │ │ + ) │ │ │ │ │ + }, true); │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + this.redraw(); │ │ │ │ │ } │ │ │ │ │ + this.updateAttribution(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setExtent │ │ │ │ │ - * Set the visible part of the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} │ │ │ │ │ - * resolutionChanged - {Boolean} │ │ │ │ │ + * Method: getURL │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ - * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ - * False otherwise. │ │ │ │ │ + * Paramters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ */ │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - var rightOfDateLine, │ │ │ │ │ - ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ - extent = extent.scale(1 / ratio), │ │ │ │ │ - world = this.map.getMaxExtent(); │ │ │ │ │ - if (world.right > extent.left && world.right < extent.right) { │ │ │ │ │ - rightOfDateLine = true; │ │ │ │ │ - } else if (world.left > extent.left && world.left < extent.right) { │ │ │ │ │ - rightOfDateLine = false; │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + if (!this.url) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var xyz = this.getXYZ(bounds), │ │ │ │ │ + x = xyz.x, │ │ │ │ │ + y = xyz.y, │ │ │ │ │ + z = xyz.z; │ │ │ │ │ + var quadDigits = []; │ │ │ │ │ + for (var i = z; i > 0; --i) { │ │ │ │ │ + var digit = '0'; │ │ │ │ │ + var mask = 1 << (i - 1); │ │ │ │ │ + if ((x & mask) != 0) { │ │ │ │ │ + digit++; │ │ │ │ │ } │ │ │ │ │ - if (rightOfDateLine !== this.rightOfDateLine || resolutionChanged) { │ │ │ │ │ - coordSysUnchanged = false; │ │ │ │ │ - this.xOffset = rightOfDateLine === true ? │ │ │ │ │ - world.getWidth() / resolution : 0; │ │ │ │ │ + if ((y & mask) != 0) { │ │ │ │ │ + digit++; │ │ │ │ │ + digit++; │ │ │ │ │ } │ │ │ │ │ - this.rightOfDateLine = rightOfDateLine; │ │ │ │ │ + quadDigits.push(digit); │ │ │ │ │ } │ │ │ │ │ - return coordSysUnchanged; │ │ │ │ │ - }, │ │ │ │ │ + var quadKey = quadDigits.join(""); │ │ │ │ │ + var url = this.selectUrl('' + x + y + z, this.url); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getNodeType │ │ │ │ │ - * This function is in charge of asking the specific renderer which type │ │ │ │ │ - * of node to create for the given geometry and style. All geometries │ │ │ │ │ - * in an Elements-based renderer consist of one node and some │ │ │ │ │ - * attributes. We have the nodeFactory() function which creates a node │ │ │ │ │ - * for us, but it takes a 'type' as input, and that is precisely what │ │ │ │ │ - * this function tells us. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The corresponding node type for the specified geometry │ │ │ │ │ - */ │ │ │ │ │ - getNodeType: function(geometry, style) {}, │ │ │ │ │ + return OpenLayers.String.format(url, { │ │ │ │ │ + 'quadkey': quadKey │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawGeometry │ │ │ │ │ - * Draw the geometry, creating new nodes, setting paths, setting style, │ │ │ │ │ - * setting featureId on the node. This method should only be called │ │ │ │ │ - * by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true if the geometry has been drawn completely; null if │ │ │ │ │ - * incomplete; false otherwise │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateAttribution │ │ │ │ │ + * Updates the attribution according to the requirements outlined in │ │ │ │ │ + * http://gis.638310.n2.nabble.com/Bing-imagery-td5789168.html │ │ │ │ │ */ │ │ │ │ │ - drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ - var className = geometry.CLASS_NAME; │ │ │ │ │ - var rendered = true; │ │ │ │ │ - if ((className == "OpenLayers.Geometry.Collection") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiPolygon")) { │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - rendered = this.drawGeometry( │ │ │ │ │ - geometry.components[i], style, featureId) && rendered; │ │ │ │ │ - } │ │ │ │ │ - return rendered; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - rendered = false; │ │ │ │ │ - var removeBackground = false; │ │ │ │ │ - if (style.display != "none") { │ │ │ │ │ - if (style.backgroundGraphic) { │ │ │ │ │ - this.redrawBackgroundNode(geometry.id, geometry, style, │ │ │ │ │ - featureId); │ │ │ │ │ - } else { │ │ │ │ │ - removeBackground = true; │ │ │ │ │ - } │ │ │ │ │ - rendered = this.redrawNode(geometry.id, geometry, style, │ │ │ │ │ - featureId); │ │ │ │ │ + updateAttribution: function() { │ │ │ │ │ + var metadata = this.metadata; │ │ │ │ │ + if (!metadata.resourceSets || !this.map || !this.map.center) { │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ - if (rendered == false) { │ │ │ │ │ - var node = document.getElementById(geometry.id); │ │ │ │ │ - if (node) { │ │ │ │ │ - if (node._style.backgroundGraphic) { │ │ │ │ │ - removeBackground = true; │ │ │ │ │ + var res = metadata.resourceSets[0].resources[0]; │ │ │ │ │ + var extent = this.map.getExtent().transform( │ │ │ │ │ + this.map.getProjectionObject(), │ │ │ │ │ + new OpenLayers.Projection("EPSG:4326") │ │ │ │ │ + ); │ │ │ │ │ + var providers = res.imageryProviders || [], │ │ │ │ │ + zoom = OpenLayers.Util.indexOf(this.serverResolutions, │ │ │ │ │ + this.getServerResolution()), │ │ │ │ │ + copyrights = "", │ │ │ │ │ + provider, i, ii, j, jj, bbox, coverage; │ │ │ │ │ + for (i = 0, ii = providers.length; i < ii; ++i) { │ │ │ │ │ + provider = providers[i]; │ │ │ │ │ + for (j = 0, jj = provider.coverageAreas.length; j < jj; ++j) { │ │ │ │ │ + coverage = provider.coverageAreas[j]; │ │ │ │ │ + // axis order provided is Y,X │ │ │ │ │ + bbox = OpenLayers.Bounds.fromArray(coverage.bbox, true); │ │ │ │ │ + if (extent.intersectsBounds(bbox) && │ │ │ │ │ + zoom <= coverage.zoomMax && zoom >= coverage.zoomMin) { │ │ │ │ │ + copyrights += provider.attribution + " "; │ │ │ │ │ } │ │ │ │ │ - node.parentNode.removeChild(node); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (removeBackground) { │ │ │ │ │ - var node = document.getElementById( │ │ │ │ │ - geometry.id + this.BACKGROUND_ID_SUFFIX); │ │ │ │ │ - if (node) { │ │ │ │ │ - node.parentNode.removeChild(node); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return rendered; │ │ │ │ │ + var logo = metadata.brandLogoUri.replace(this.protocolRegex, this.protocol); │ │ │ │ │ + this.attribution = OpenLayers.String.format(this.attributionTemplate, { │ │ │ │ │ + type: this.type.toLowerCase(), │ │ │ │ │ + logo: logo, │ │ │ │ │ + copyrights: copyrights │ │ │ │ │ + }); │ │ │ │ │ + this.map && this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "attribution" │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: redrawNode │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true if the complete geometry could be drawn, null if parts of │ │ │ │ │ - * the geometry could not be drawn, false otherwise │ │ │ │ │ + * Method: setMap │ │ │ │ │ */ │ │ │ │ │ - redrawNode: function(id, geometry, style, featureId) { │ │ │ │ │ - style = this.applyDefaultSymbolizer(style); │ │ │ │ │ - // Get the node if it's already on the map. │ │ │ │ │ - var node = this.nodeFactory(id, this.getNodeType(geometry, style)); │ │ │ │ │ - │ │ │ │ │ - // Set the data for the node, then draw it. │ │ │ │ │ - node._featureId = featureId; │ │ │ │ │ - node._boundsBottom = geometry.getBounds().bottom; │ │ │ │ │ - node._geometryClass = geometry.CLASS_NAME; │ │ │ │ │ - node._style = style; │ │ │ │ │ - │ │ │ │ │ - var drawResult = this.drawGeometryNode(node, geometry, style); │ │ │ │ │ - if (drawResult === false) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - node = drawResult.node; │ │ │ │ │ - │ │ │ │ │ - // Insert the node into the indexer so it can show us where to │ │ │ │ │ - // place it. Note that this operation is O(log(n)). If there's a │ │ │ │ │ - // performance problem (when dragging, for instance) this is │ │ │ │ │ - // likely where it would be. │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - var insert = this.indexer.insert(node); │ │ │ │ │ - if (insert) { │ │ │ │ │ - this.vectorRoot.insertBefore(node, insert); │ │ │ │ │ - } else { │ │ │ │ │ - this.vectorRoot.appendChild(node); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - // if there's no indexer, simply append the node to root, │ │ │ │ │ - // but only if the node is a new one │ │ │ │ │ - if (node.parentNode !== this.vectorRoot) { │ │ │ │ │ - this.vectorRoot.appendChild(node); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.postDraw(node); │ │ │ │ │ - │ │ │ │ │ - return drawResult.complete; │ │ │ │ │ + setMap: function() { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.register("moveend", this, this.updateAttribution); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: redrawBackgroundNode │ │ │ │ │ - * Redraws the node using special 'background' style properties. Basically │ │ │ │ │ - * just calls redrawNode(), but instead of directly using the │ │ │ │ │ - * 'externalGraphic', 'graphicXOffset', 'graphicYOffset', and │ │ │ │ │ - * 'graphicZIndex' properties directly from the specified 'style' │ │ │ │ │ - * parameter, we create a new style object and set those properties │ │ │ │ │ - * from the corresponding 'background'-prefixed properties from │ │ │ │ │ - * specified 'style' parameter. │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * obj - {Object} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} true if the complete geometry could be drawn, null if parts of │ │ │ │ │ - * the geometry could not be drawn, false otherwise │ │ │ │ │ + * {<OpenLayers.Layer.Bing>} An exact clone of this <OpenLayers.Layer.Bing> │ │ │ │ │ */ │ │ │ │ │ - redrawBackgroundNode: function(id, geometry, style, featureId) { │ │ │ │ │ - var backgroundStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ - │ │ │ │ │ - // Set regular style attributes to apply to the background styles. │ │ │ │ │ - backgroundStyle.externalGraphic = backgroundStyle.backgroundGraphic; │ │ │ │ │ - backgroundStyle.graphicXOffset = backgroundStyle.backgroundXOffset; │ │ │ │ │ - backgroundStyle.graphicYOffset = backgroundStyle.backgroundYOffset; │ │ │ │ │ - backgroundStyle.graphicZIndex = backgroundStyle.backgroundGraphicZIndex; │ │ │ │ │ - backgroundStyle.graphicWidth = backgroundStyle.backgroundWidth || backgroundStyle.graphicWidth; │ │ │ │ │ - backgroundStyle.graphicHeight = backgroundStyle.backgroundHeight || backgroundStyle.graphicHeight; │ │ │ │ │ - │ │ │ │ │ - // Erase background styles. │ │ │ │ │ - backgroundStyle.backgroundGraphic = null; │ │ │ │ │ - backgroundStyle.backgroundXOffset = null; │ │ │ │ │ - backgroundStyle.backgroundYOffset = null; │ │ │ │ │ - backgroundStyle.backgroundGraphicZIndex = null; │ │ │ │ │ - │ │ │ │ │ - return this.redrawNode( │ │ │ │ │ - id + this.BACKGROUND_ID_SUFFIX, │ │ │ │ │ - geometry, │ │ │ │ │ - backgroundStyle, │ │ │ │ │ - null │ │ │ │ │ - ); │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Bing(this.options); │ │ │ │ │ + } │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawGeometryNode │ │ │ │ │ - * Given a node, draw a geometry on the specified layer. │ │ │ │ │ - * node and geometry are required arguments, style is optional. │ │ │ │ │ - * This method is only called by the render itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} a hash with properties "node" (the drawn node) and "complete" │ │ │ │ │ - * (null if parts of the geometry could not be drawn, false if nothing │ │ │ │ │ - * could be drawn) │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ - drawGeometryNode: function(node, geometry, style) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.map && │ │ │ │ │ + this.map.events.unregister("moveend", this, this.updateAttribution); │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var options = { │ │ │ │ │ - 'isFilled': style.fill === undefined ? │ │ │ │ │ - true : style.fill, │ │ │ │ │ - 'isStroked': style.stroke === undefined ? │ │ │ │ │ - !!style.strokeWidth : style.stroke │ │ │ │ │ - }; │ │ │ │ │ - var drawn; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.graphic === false) { │ │ │ │ │ - options.isFilled = false; │ │ │ │ │ - options.isStroked = false; │ │ │ │ │ - } │ │ │ │ │ - drawn = this.drawPoint(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - options.isFilled = false; │ │ │ │ │ - drawn = this.drawLineString(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - drawn = this.drawLinearRing(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - drawn = this.drawPolygon(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - drawn = this.drawRectangle(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Bing" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ - node._options = options; │ │ │ │ │ +/** │ │ │ │ │ + * Function: OpenLayers.Layer.Bing.processMetadata │ │ │ │ │ + * This function will be bound to an instance, linked to the global scope with │ │ │ │ │ + * an id, and called by the JSONP script returned by the API. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * metadata - {Object} metadata as returned by the API │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Bing.processMetadata = function(metadata) { │ │ │ │ │ + this.metadata = metadata; │ │ │ │ │ + this.initLayer(); │ │ │ │ │ + var script = document.getElementById(this._callbackId); │ │ │ │ │ + script.parentNode.removeChild(script); │ │ │ │ │ + window[this._callbackId] = undefined; // cannot delete from window in IE │ │ │ │ │ + delete this._callbackId; │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/OSM.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - //set style │ │ │ │ │ - //TBD simplify this │ │ │ │ │ - if (drawn != false) { │ │ │ │ │ - return { │ │ │ │ │ - node: this.setStyle(node, style, options, geometry), │ │ │ │ │ - complete: drawn │ │ │ │ │ - }; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: postDraw │ │ │ │ │ - * Things that have do be done after the geometry node is appended │ │ │ │ │ - * to its parent node. To be overridden by subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - postDraw: function(node) {}, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.OSM │ │ │ │ │ + * This layer allows accessing OpenStreetMap tiles. By default the OpenStreetMap │ │ │ │ │ + * hosted tile.openstreetmap.org Mapnik tileset is used. If you wish to use │ │ │ │ │ + * a different layer instead, you need to provide a different │ │ │ │ │ + * URL to the constructor. Here's an example for using OpenCycleMap: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * new OpenLayers.Layer.OSM("OpenCycleMap", │ │ │ │ │ + * ["http://a.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ + * "http://b.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ + * "http://c.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png"]); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.XYZ> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.OSM = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPoint │ │ │ │ │ - * Virtual function for drawing Point Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the point │ │ │ │ │ + * APIProperty: name │ │ │ │ │ + * {String} The layer name. Defaults to "OpenStreetMap" if the first │ │ │ │ │ + * argument to the constructor is null or undefined. │ │ │ │ │ */ │ │ │ │ │ - drawPoint: function(node, geometry) {}, │ │ │ │ │ + name: "OpenStreetMap", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLineString │ │ │ │ │ - * Virtual function for drawing LineString Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components of │ │ │ │ │ - * the linestring, or false if nothing could be drawn │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String} The tileset URL scheme. Defaults to │ │ │ │ │ + * : http://[a|b|c].tile.openstreetmap.org/${z}/${x}/${y}.png │ │ │ │ │ + * (the official OSM tileset) if the second argument to the constructor │ │ │ │ │ + * is null or undefined. To use another tileset you can have something │ │ │ │ │ + * like this: │ │ │ │ │ + * (code) │ │ │ │ │ + * new OpenLayers.Layer.OSM("OpenCycleMap", │ │ │ │ │ + * ["http://a.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ + * "http://b.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ + * "http://c.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png"]); │ │ │ │ │ + * (end) │ │ │ │ │ */ │ │ │ │ │ - drawLineString: function(node, geometry) {}, │ │ │ │ │ + url: [ │ │ │ │ │ + 'http://a.tile.openstreetmap.org/${z}/${x}/${y}.png', │ │ │ │ │ + 'http://b.tile.openstreetmap.org/${z}/${x}/${y}.png', │ │ │ │ │ + 'http://c.tile.openstreetmap.org/${z}/${x}/${y}.png' │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLinearRing │ │ │ │ │ - * Virtual function for drawing LinearRing Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ - * of the linear ring, or false if nothing could be drawn │ │ │ │ │ + * Property: attribution │ │ │ │ │ + * {String} The layer attribution. │ │ │ │ │ */ │ │ │ │ │ - drawLinearRing: function(node, geometry) {}, │ │ │ │ │ + attribution: "© <a href='http://www.openstreetmap.org/copyright'>OpenStreetMap</a> contributors", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPolygon │ │ │ │ │ - * Virtual function for drawing Polygon Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ - * of the polygon, or false if nothing could be drawn │ │ │ │ │ + * Property: sphericalMercator │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - drawPolygon: function(node, geometry) {}, │ │ │ │ │ + sphericalMercator: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawRectangle │ │ │ │ │ - * Virtual function for drawing Rectangle Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the rectangle │ │ │ │ │ + * Property: wrapDateLine │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - drawRectangle: function(node, geometry) {}, │ │ │ │ │ + wrapDateLine: true, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawCircle │ │ │ │ │ - * Virtual function for drawing Circle Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the circle │ │ │ │ │ + /** APIProperty: tileOptions │ │ │ │ │ + * {Object} optional configuration options for <OpenLayers.Tile> instances │ │ │ │ │ + * created by this Layer. Default is │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * {crossOriginKeyword: 'anonymous'} │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * When using OSM tilesets other than the default ones, it may be │ │ │ │ │ + * necessary to set this to │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * {crossOriginKeyword: null} │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * if the server does not send Access-Control-Allow-Origin headers. │ │ │ │ │ */ │ │ │ │ │ - drawCircle: function(node, geometry) {}, │ │ │ │ │ + tileOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removeText │ │ │ │ │ - * Removes a label │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Layer.OSM │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * name - {String} The layer name. │ │ │ │ │ + * url - {String} The tileset URL scheme. │ │ │ │ │ + * options - {Object} Configuration options for the layer. Any inherited │ │ │ │ │ + * layer option can be set in this object (e.g. │ │ │ │ │ + * <OpenLayers.Layer.Grid.buffer>). │ │ │ │ │ */ │ │ │ │ │ - removeText: function(featureId) { │ │ │ │ │ - var label = document.getElementById(featureId + this.LABEL_ID_SUFFIX); │ │ │ │ │ - if (label) { │ │ │ │ │ - this.textRoot.removeChild(label); │ │ │ │ │ - } │ │ │ │ │ - var outline = document.getElementById(featureId + this.LABEL_OUTLINE_SUFFIX); │ │ │ │ │ - if (outline) { │ │ │ │ │ - this.textRoot.removeChild(outline); │ │ │ │ │ - } │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + crossOriginKeyword: 'anonymous' │ │ │ │ │ + }, this.options && this.options.tileOptions); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureIdFromEvent │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} An <OpenLayers.Event> object │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A feature id or undefined. │ │ │ │ │ + * Method: clone │ │ │ │ │ */ │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var target = evt.target; │ │ │ │ │ - var useElement = target && target.correspondingUseElement; │ │ │ │ │ - var node = useElement ? useElement : (target || evt.srcElement); │ │ │ │ │ - return node._featureId; │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.OSM( │ │ │ │ │ + this.name, this.url, this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: eraseGeometry │ │ │ │ │ - * Erase a geometry from the renderer. In the case of a multi-geometry, │ │ │ │ │ - * we cycle through and recurse on ourselves. Otherwise, we look for a │ │ │ │ │ - * node with the geometry.id, destroy its geometry, and remove it from │ │ │ │ │ - * the DOM. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - */ │ │ │ │ │ - eraseGeometry: function(geometry, featureId) { │ │ │ │ │ - if ((geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ - (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ - (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPolygon") || │ │ │ │ │ - (geometry.CLASS_NAME == "OpenLayers.Geometry.Collection")) { │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - this.eraseGeometry(geometry.components[i], featureId); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - var element = OpenLayers.Util.getElement(geometry.id); │ │ │ │ │ - if (element && element.parentNode) { │ │ │ │ │ - if (element.geometry) { │ │ │ │ │ - element.geometry.destroy(); │ │ │ │ │ - element.geometry = null; │ │ │ │ │ - } │ │ │ │ │ - element.parentNode.removeChild(element); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.OSM" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/SphericalMercator.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - this.indexer.remove(element); │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - if (element._style.backgroundGraphic) { │ │ │ │ │ - var backgroundId = geometry.id + this.BACKGROUND_ID_SUFFIX; │ │ │ │ │ - var bElem = OpenLayers.Util.getElement(backgroundId); │ │ │ │ │ - if (bElem && bElem.parentNode) { │ │ │ │ │ - // No need to destroy the geometry since the element and the background │ │ │ │ │ - // node share the same geometry. │ │ │ │ │ - bElem.parentNode.removeChild(bElem); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer.js │ │ │ │ │ + * @requires OpenLayers/Projection.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: nodeFactory │ │ │ │ │ - * Create new node of the specified type, with the (optional) specified id. │ │ │ │ │ - * │ │ │ │ │ - * If node already exists with same ID and a different type, we remove it │ │ │ │ │ - * and then call ourselves again to recreate it. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * type - {String} type Kind of node to draw. │ │ │ │ │ - * │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.SphericalMercator │ │ │ │ │ + * A mixin for layers that wraps up the pieces neccesary to have a coordinate │ │ │ │ │ + * conversion for working with commercial APIs which use a spherical │ │ │ │ │ + * mercator projection. Using this layer as a base layer, additional │ │ │ │ │ + * layers can be used as overlays if they are in the same projection. │ │ │ │ │ + * │ │ │ │ │ + * A layer is given properties of this object by setting the sphericalMercator │ │ │ │ │ + * property to true. │ │ │ │ │ + * │ │ │ │ │ + * More projection information: │ │ │ │ │ + * - http://spatialreference.org/ref/user/google-projection/ │ │ │ │ │ + * │ │ │ │ │ + * Proj4 Text: │ │ │ │ │ + * +proj=merc +a=6378137 +b=6378137 +lat_ts=0.0 +lon_0=0.0 +x_0=0.0 +y_0=0 │ │ │ │ │ + * +k=1.0 +units=m +nadgrids=@null +no_defs │ │ │ │ │ + * │ │ │ │ │ + * WKT: │ │ │ │ │ + * 900913=PROJCS["WGS84 / Simple Mercator", GEOGCS["WGS 84", │ │ │ │ │ + * DATUM["WGS_1984", SPHEROID["WGS_1984", 6378137.0, 298.257223563]], │ │ │ │ │ + * PRIMEM["Greenwich", 0.0], UNIT["degree", 0.017453292519943295], │ │ │ │ │ + * AXIS["Longitude", EAST], AXIS["Latitude", NORTH]], │ │ │ │ │ + * PROJECTION["Mercator_1SP_Google"], │ │ │ │ │ + * PARAMETER["latitude_of_origin", 0.0], PARAMETER["central_meridian", 0.0], │ │ │ │ │ + * PARAMETER["scale_factor", 1.0], PARAMETER["false_easting", 0.0], │ │ │ │ │ + * PARAMETER["false_northing", 0.0], UNIT["m", 1.0], AXIS["x", EAST], │ │ │ │ │ + * AXIS["y", NORTH], AUTHORITY["EPSG","900913"]] │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.SphericalMercator = { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getExtent │ │ │ │ │ + * Get the map's extent. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} A new node of the given type and id. │ │ │ │ │ + * {<OpenLayers.Bounds>} The map extent. │ │ │ │ │ */ │ │ │ │ │ - nodeFactory: function(id, type) { │ │ │ │ │ - var node = OpenLayers.Util.getElement(id); │ │ │ │ │ - if (node) { │ │ │ │ │ - if (!this.nodeTypeCompare(node, type)) { │ │ │ │ │ - node.parentNode.removeChild(node); │ │ │ │ │ - node = this.nodeFactory(id, type); │ │ │ │ │ - } │ │ │ │ │ + getExtent: function() { │ │ │ │ │ + var extent = null; │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + extent = this.map.calculateBounds(); │ │ │ │ │ } else { │ │ │ │ │ - node = this.createNode(type, id); │ │ │ │ │ + extent = OpenLayers.Layer.FixedZoomLevels.prototype.getExtent.apply(this); │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ + return extent; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: nodeTypeCompare │ │ │ │ │ + /** │ │ │ │ │ + * Method: getLonLatFromViewPortPx │ │ │ │ │ + * Get a map location from a pixel location │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * type - {String} Kind of node │ │ │ │ │ - * │ │ │ │ │ + * viewPortPx - {<OpenLayers.Pixel>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ - * This function must be overridden by subclasses. │ │ │ │ │ + * {<OpenLayers.LonLat>} An OpenLayers.LonLat which is the passed-in view │ │ │ │ │ + * port OpenLayers.Pixel, translated into lon/lat by map lib │ │ │ │ │ + * If the map lib is not loaded or not centered, returns null │ │ │ │ │ */ │ │ │ │ │ - nodeTypeCompare: function(node, type) {}, │ │ │ │ │ + getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ + return OpenLayers.Layer.prototype.getLonLatFromViewPortPx.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createNode │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: getViewPortPxFromLonLat │ │ │ │ │ + * Get a pixel location from a map location │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * type - {String} Kind of node to draw. │ │ │ │ │ - * id - {String} Id for node. │ │ │ │ │ - * │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} A new node of the given type and id. │ │ │ │ │ - * This function must be overridden by subclasses. │ │ │ │ │ + * {<OpenLayers.Pixel>} An OpenLayers.Pixel which is the passed-in │ │ │ │ │ + * OpenLayers.LonLat, translated into view port pixels by map lib │ │ │ │ │ + * If map lib is not loaded or not centered, returns null │ │ │ │ │ */ │ │ │ │ │ - createNode: function(type, id) {}, │ │ │ │ │ + getViewPortPxFromLonLat: function(lonlat) { │ │ │ │ │ + return OpenLayers.Layer.prototype.getViewPortPxFromLonLat.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveRoot │ │ │ │ │ - * moves this renderer's root to a different renderer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * renderer - {<OpenLayers.Renderer>} target renderer for the moved root │ │ │ │ │ + /** │ │ │ │ │ + * Method: initMercatorParameters │ │ │ │ │ + * Set up the mercator parameters on the layer: resolutions, │ │ │ │ │ + * projection, units. │ │ │ │ │ */ │ │ │ │ │ - moveRoot: function(renderer) { │ │ │ │ │ - var root = this.root; │ │ │ │ │ - if (renderer.root.parentNode == this.rendererRoot) { │ │ │ │ │ - root = renderer.root; │ │ │ │ │ + initMercatorParameters: function() { │ │ │ │ │ + // set up properties for Mercator - assume EPSG:900913 │ │ │ │ │ + this.RESOLUTIONS = []; │ │ │ │ │ + var maxResolution = 156543.03390625; │ │ │ │ │ + for (var zoom = 0; zoom <= this.MAX_ZOOM_LEVEL; ++zoom) { │ │ │ │ │ + this.RESOLUTIONS[zoom] = maxResolution / Math.pow(2, zoom); │ │ │ │ │ } │ │ │ │ │ - root.parentNode.removeChild(root); │ │ │ │ │ - renderer.rendererRoot.appendChild(root); │ │ │ │ │ + this.units = "m"; │ │ │ │ │ + this.projection = this.projection || "EPSG:900913"; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getRenderLayerId │ │ │ │ │ - * Gets the layer that this renderer's output appears on. If moveRoot was │ │ │ │ │ - * used, this will be different from the id of the layer containing the │ │ │ │ │ - * features rendered by this renderer. │ │ │ │ │ + * APIMethod: forwardMercator │ │ │ │ │ + * Given a lon,lat in EPSG:4326, return a point in Spherical Mercator. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * lon - {float} │ │ │ │ │ + * lat - {float} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} the id of the output layer. │ │ │ │ │ + * {<OpenLayers.LonLat>} The coordinates transformed to Mercator. │ │ │ │ │ */ │ │ │ │ │ - getRenderLayerId: function() { │ │ │ │ │ - return this.root.parentNode.parentNode.id; │ │ │ │ │ - }, │ │ │ │ │ + forwardMercator: (function() { │ │ │ │ │ + var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + var sm = new OpenLayers.Projection("EPSG:900913"); │ │ │ │ │ + return function(lon, lat) { │ │ │ │ │ + var point = OpenLayers.Projection.transform({ │ │ │ │ │ + x: lon, │ │ │ │ │ + y: lat │ │ │ │ │ + }, gg, sm); │ │ │ │ │ + return new OpenLayers.LonLat(point.x, point.y); │ │ │ │ │ + }; │ │ │ │ │ + })(), │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: isComplexSymbol │ │ │ │ │ - * Determines if a symbol cannot be rendered using drawCircle │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: inverseMercator │ │ │ │ │ + * Given a x,y in Spherical Mercator, return a point in EPSG:4326. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * graphicName - {String} │ │ │ │ │ + * x - {float} A map x in Spherical Mercator. │ │ │ │ │ + * y - {float} A map y in Spherical Mercator. │ │ │ │ │ * │ │ │ │ │ - * Returns │ │ │ │ │ - * {Boolean} true if the symbol is complex, false if not │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.LonLat>} The coordinates transformed to EPSG:4326. │ │ │ │ │ */ │ │ │ │ │ - isComplexSymbol: function(graphicName) { │ │ │ │ │ - return (graphicName != "circle") && !!graphicName; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.Elements" │ │ │ │ │ -}); │ │ │ │ │ + inverseMercator: (function() { │ │ │ │ │ + var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + var sm = new OpenLayers.Projection("EPSG:900913"); │ │ │ │ │ + return function(x, y) { │ │ │ │ │ + var point = OpenLayers.Projection.transform({ │ │ │ │ │ + x: x, │ │ │ │ │ + y: y │ │ │ │ │ + }, sm, gg); │ │ │ │ │ + return new OpenLayers.LonLat(point.x, point.y); │ │ │ │ │ + }; │ │ │ │ │ + })() │ │ │ │ │ │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Renderer/VML.js │ │ │ │ │ + OpenLayers/Layer/EventPane.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Renderer/Elements.js │ │ │ │ │ + * @requires OpenLayers/Layer.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Renderer.VML │ │ │ │ │ - * Render vector features in browsers with VML capability. Construct a new │ │ │ │ │ - * VML renderer with the <OpenLayers.Renderer.VML> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Note that for all calculations in this class, we use (num | 0) to truncate a │ │ │ │ │ - * float value to an integer. This is done because it seems that VML doesn't │ │ │ │ │ - * support float values. │ │ │ │ │ + * Class: OpenLayers.Layer.EventPane │ │ │ │ │ + * Base class for 3rd party layers, providing a DOM element which isolates │ │ │ │ │ + * the 3rd-party layer from mouse events. │ │ │ │ │ + * Only used by Google layers. │ │ │ │ │ + * │ │ │ │ │ + * Automatically instantiated by the Google constructor, and not usually instantiated directly. │ │ │ │ │ * │ │ │ │ │ + * Create a new event pane layer with the │ │ │ │ │ + * <OpenLayers.Layer.EventPane> constructor. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Renderer.Elements> │ │ │ │ │ + * - <OpenLayers.Layer> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Renderer.VML = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ +OpenLayers.Layer.EventPane = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: xmlns │ │ │ │ │ - * {String} XML Namespace URN │ │ │ │ │ + * APIProperty: smoothDragPan │ │ │ │ │ + * {Boolean} smoothDragPan determines whether non-public/internal API │ │ │ │ │ + * methods are used for better performance while dragging EventPane │ │ │ │ │ + * layers. When not in sphericalMercator mode, the smoother dragging │ │ │ │ │ + * doesn't actually move north/south directly with the number of │ │ │ │ │ + * pixels moved, resulting in a slight offset when you drag your mouse │ │ │ │ │ + * north south with this option on. If this visual disparity bothers │ │ │ │ │ + * you, you should turn this option off, or use spherical mercator. │ │ │ │ │ + * Default is on. │ │ │ │ │ */ │ │ │ │ │ - xmlns: "urn:schemas-microsoft-com:vml", │ │ │ │ │ + smoothDragPan: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: symbolCache │ │ │ │ │ - * {DOMElement} node holding symbols. This hash is keyed by symbol name, │ │ │ │ │ - * and each value is a hash with a "path" and an "extent" property. │ │ │ │ │ + * Property: isBaseLayer │ │ │ │ │ + * {Boolean} EventPaned layers are always base layers, by necessity. │ │ │ │ │ */ │ │ │ │ │ - symbolCache: {}, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: offset │ │ │ │ │ - * {Object} Hash with "x" and "y" properties │ │ │ │ │ + * APIProperty: isFixed │ │ │ │ │ + * {Boolean} EventPaned layers are fixed by default. │ │ │ │ │ */ │ │ │ │ │ - offset: null, │ │ │ │ │ + isFixed: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Renderer.VML │ │ │ │ │ - * Create a new VML renderer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * containerID - {String} The id for the element that contains the renderer │ │ │ │ │ + * Property: pane │ │ │ │ │ + * {DOMElement} A reference to the element that controls the events. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(containerID) { │ │ │ │ │ - if (!this.supported()) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (!document.namespaces.olv) { │ │ │ │ │ - document.namespaces.add("olv", this.xmlns); │ │ │ │ │ - var style = document.createStyleSheet(); │ │ │ │ │ - var shapes = ['shape', 'rect', 'oval', 'fill', 'stroke', 'imagedata', 'group', 'textbox']; │ │ │ │ │ - for (var i = 0, len = shapes.length; i < len; i++) { │ │ │ │ │ - │ │ │ │ │ - style.addRule('olv\\:' + shapes[i], "behavior: url(#default#VML); " + │ │ │ │ │ - "position: absolute; display: inline-block;"); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.initialize.apply(this, │ │ │ │ │ - arguments); │ │ │ │ │ - }, │ │ │ │ │ + pane: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: supported │ │ │ │ │ - * Determine whether a browser supports this renderer. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The browser supports the VML renderer │ │ │ │ │ - */ │ │ │ │ │ - supported: function() { │ │ │ │ │ - return !!(document.namespaces); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setExtent │ │ │ │ │ - * Set the renderer's extent │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} │ │ │ │ │ - * resolutionChanged - {Boolean} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ - * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ + * Property: mapObject │ │ │ │ │ + * {Object} This is the object which will be used to load the 3rd party library │ │ │ │ │ + * in the case of the google layer, this will be of type GMap, │ │ │ │ │ + * in the case of the ve layer, this will be of type VEMap │ │ │ │ │ */ │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - │ │ │ │ │ - var left = (extent.left / resolution) | 0; │ │ │ │ │ - var top = (extent.top / resolution - this.size.h) | 0; │ │ │ │ │ - if (resolutionChanged || !this.offset) { │ │ │ │ │ - this.offset = { │ │ │ │ │ - x: left, │ │ │ │ │ - y: top │ │ │ │ │ - }; │ │ │ │ │ - left = 0; │ │ │ │ │ - top = 0; │ │ │ │ │ - } else { │ │ │ │ │ - left = left - this.offset.x; │ │ │ │ │ - top = top - this.offset.y; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - var org = (left - this.xOffset) + " " + top; │ │ │ │ │ - this.root.coordorigin = org; │ │ │ │ │ - var roots = [this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ - var root; │ │ │ │ │ - for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ - root = roots[i]; │ │ │ │ │ - │ │ │ │ │ - var size = this.size.w + " " + this.size.h; │ │ │ │ │ - root.coordsize = size; │ │ │ │ │ - │ │ │ │ │ - } │ │ │ │ │ - // flip the VML display Y axis upside down so it │ │ │ │ │ - // matches the display Y axis of the map │ │ │ │ │ - this.root.style.flip = "y"; │ │ │ │ │ - │ │ │ │ │ - return coordSysUnchanged; │ │ │ │ │ - }, │ │ │ │ │ + mapObject: null, │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setSize │ │ │ │ │ - * Set the size of the drawing surface │ │ │ │ │ + * Constructor: OpenLayers.Layer.EventPane │ │ │ │ │ + * Create a new event pane layer │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>} the size of the drawing surface │ │ │ │ │ + * name - {String} │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ */ │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - // setting width and height on all roots to avoid flicker which we │ │ │ │ │ - // would get with 100% width and height on child roots │ │ │ │ │ - var roots = [ │ │ │ │ │ - this.rendererRoot, │ │ │ │ │ - this.root, │ │ │ │ │ - this.vectorRoot, │ │ │ │ │ - this.textRoot │ │ │ │ │ - ]; │ │ │ │ │ - var w = this.size.w + "px"; │ │ │ │ │ - var h = this.size.h + "px"; │ │ │ │ │ - var root; │ │ │ │ │ - for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ - root = roots[i]; │ │ │ │ │ - root.style.width = w; │ │ │ │ │ - root.style.height = h; │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (this.pane == null) { │ │ │ │ │ + this.pane = OpenLayers.Util.createDiv(this.div.id + "_EventPane"); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getNodeType │ │ │ │ │ - * Get the node type for a geometry and style │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The corresponding node type for the specified geometry │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Deconstruct this layer. │ │ │ │ │ */ │ │ │ │ │ - getNodeType: function(geometry, style) { │ │ │ │ │ - var nodeType = null; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - nodeType = "olv:rect"; │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - nodeType = "olv:shape"; │ │ │ │ │ - } else { │ │ │ │ │ - nodeType = "olv:oval"; │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - nodeType = "olv:rect"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - case "OpenLayers.Geometry.Curve": │ │ │ │ │ - nodeType = "olv:shape"; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - return nodeType; │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.mapObject = null; │ │ │ │ │ + this.pane = null; │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: setStyle │ │ │ │ │ - * Use to set all the style attributes to a VML node. │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the layer. This is done through an accessor │ │ │ │ │ + * so that subclasses can override this and take special action once │ │ │ │ │ + * they have their map variable set. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} An VML element to decorate │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * options - {Object} Currently supported options include │ │ │ │ │ - * 'isFilled' {Boolean} and │ │ │ │ │ - * 'isStroked' {Boolean} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - setStyle: function(node, style, options, geometry) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - options = options || node._options; │ │ │ │ │ - var fillColor = style.fillColor; │ │ │ │ │ - │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - node.title = title; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - options.isFilled = true; │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ - style.graphicXOffset : -(0.5 * width); │ │ │ │ │ - var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ - style.graphicYOffset : -(0.5 * height); │ │ │ │ │ - │ │ │ │ │ - node.style.left = ((((geometry.x - this.featureDx) / resolution - this.offset.x) + xOffset) | 0) + "px"; │ │ │ │ │ - node.style.top = (((geometry.y / resolution - this.offset.y) - (yOffset + height)) | 0) + "px"; │ │ │ │ │ - node.style.width = width + "px"; │ │ │ │ │ - node.style.height = height + "px"; │ │ │ │ │ - node.style.flip = "y"; │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - // modify fillColor and options for stroke styling below │ │ │ │ │ - fillColor = "none"; │ │ │ │ │ - options.isStroked = false; │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - var cache = this.importSymbol(style.graphicName); │ │ │ │ │ - node.path = cache.path; │ │ │ │ │ - node.coordorigin = cache.left + "," + cache.bottom; │ │ │ │ │ - var size = cache.size; │ │ │ │ │ - node.coordsize = size + "," + size; │ │ │ │ │ - this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ - node.style.flip = "y"; │ │ │ │ │ - } else { │ │ │ │ │ - this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ - } │ │ │ │ │ + this.pane.style.zIndex = parseInt(this.div.style.zIndex) + 1; │ │ │ │ │ + this.pane.style.display = this.div.style.display; │ │ │ │ │ + this.pane.style.width = "100%"; │ │ │ │ │ + this.pane.style.height = "100%"; │ │ │ │ │ + if (OpenLayers.BROWSER_NAME == "msie") { │ │ │ │ │ + this.pane.style.background = │ │ │ │ │ + "url(" + OpenLayers.Util.getImageLocation("blank.gif") + ")"; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // fill │ │ │ │ │ - if (options.isFilled) { │ │ │ │ │ - node.fillcolor = fillColor; │ │ │ │ │ - } else { │ │ │ │ │ - node.filled = "false"; │ │ │ │ │ - } │ │ │ │ │ - var fills = node.getElementsByTagName("fill"); │ │ │ │ │ - var fill = (fills.length == 0) ? null : fills[0]; │ │ │ │ │ - if (!options.isFilled) { │ │ │ │ │ - if (fill) { │ │ │ │ │ - node.removeChild(fill); │ │ │ │ │ - } │ │ │ │ │ + if (this.isFixed) { │ │ │ │ │ + this.map.viewPortDiv.appendChild(this.pane); │ │ │ │ │ } else { │ │ │ │ │ - if (!fill) { │ │ │ │ │ - fill = this.createNode('olv:fill', node.id + "_fill"); │ │ │ │ │ - } │ │ │ │ │ - fill.opacity = style.fillOpacity; │ │ │ │ │ - │ │ │ │ │ - if (node._geometryClass === "OpenLayers.Geometry.Point" && │ │ │ │ │ - style.externalGraphic) { │ │ │ │ │ - │ │ │ │ │ - // override fillOpacity │ │ │ │ │ - if (style.graphicOpacity) { │ │ │ │ │ - fill.opacity = style.graphicOpacity; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - fill.src = style.externalGraphic; │ │ │ │ │ - fill.type = "frame"; │ │ │ │ │ - │ │ │ │ │ - if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ - fill.aspect = "atmost"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (fill.parentNode != node) { │ │ │ │ │ - node.appendChild(fill); │ │ │ │ │ - } │ │ │ │ │ + this.map.layerContainerDiv.appendChild(this.pane); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // additional rendering for rotated graphics or symbols │ │ │ │ │ - var rotation = style.rotation; │ │ │ │ │ - if ((rotation !== undefined || node._rotation !== undefined)) { │ │ │ │ │ - node._rotation = rotation; │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ - // make the fill fully transparent, because we now have │ │ │ │ │ - // the graphic as imagedata element. We cannot just remove │ │ │ │ │ - // the fill, because this is part of the hack described │ │ │ │ │ - // in graphicRotate │ │ │ │ │ - fill.opacity = 0; │ │ │ │ │ - } else if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ - node.style.rotation = rotation || 0; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + // once our layer has been added to the map, we can load it │ │ │ │ │ + this.loadMapObject(); │ │ │ │ │ │ │ │ │ │ - // stroke │ │ │ │ │ - var strokes = node.getElementsByTagName("stroke"); │ │ │ │ │ - var stroke = (strokes.length == 0) ? null : strokes[0]; │ │ │ │ │ - if (!options.isStroked) { │ │ │ │ │ - node.stroked = false; │ │ │ │ │ - if (stroke) { │ │ │ │ │ - stroke.on = false; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - if (!stroke) { │ │ │ │ │ - stroke = this.createNode('olv:stroke', node.id + "_stroke"); │ │ │ │ │ - node.appendChild(stroke); │ │ │ │ │ - } │ │ │ │ │ - stroke.on = true; │ │ │ │ │ - stroke.color = style.strokeColor; │ │ │ │ │ - stroke.weight = style.strokeWidth + "px"; │ │ │ │ │ - stroke.opacity = style.strokeOpacity; │ │ │ │ │ - stroke.endcap = style.strokeLinecap == 'butt' ? 'flat' : │ │ │ │ │ - (style.strokeLinecap || 'round'); │ │ │ │ │ - if (style.strokeDashstyle) { │ │ │ │ │ - stroke.dashstyle = this.dashStyle(style); │ │ │ │ │ - } │ │ │ │ │ + // if map didn't load, display warning │ │ │ │ │ + if (this.mapObject == null) { │ │ │ │ │ + this.loadWarningMessage(); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ - node.style.cursor = style.cursor; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: removeMap │ │ │ │ │ + * On being removed from the map, we'll like to remove the invisible 'pane' │ │ │ │ │ + * div that we added to it on creation. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + if (this.pane && this.pane.parentNode) { │ │ │ │ │ + this.pane.parentNode.removeChild(this.pane); │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ + OpenLayers.Layer.prototype.removeMap.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: graphicRotate │ │ │ │ │ - * If a point is to be styled with externalGraphic and rotation, VML fills │ │ │ │ │ - * cannot be used to display the graphic, because rotation of graphic │ │ │ │ │ - * fills is not supported by the VML implementation of Internet Explorer. │ │ │ │ │ - * This method creates a olv:imagedata element inside the VML node, │ │ │ │ │ - * DXImageTransform.Matrix and BasicImage filters for rotation and │ │ │ │ │ - * opacity, and a 3-step hack to remove rendering artefacts from the │ │ │ │ │ - * graphic and preserve the ability of graphics to trigger events. │ │ │ │ │ - * Finally, OpenLayers methods are used to determine the correct │ │ │ │ │ - * insertion point of the rotated image, because DXImageTransform.Matrix │ │ │ │ │ - * does the rotation without the ability to specify a rotation center │ │ │ │ │ - * point. │ │ │ │ │ + * Method: loadWarningMessage │ │ │ │ │ + * If we can't load the map lib, then display an error message to the │ │ │ │ │ + * user and tell them where to go for help. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * xOffset - {Number} rotation center relative to image, x coordinate │ │ │ │ │ - * yOffset - {Number} rotation center relative to image, y coordinate │ │ │ │ │ - * style - {Object} │ │ │ │ │ + * This function sets up the layout for the warning message. Each 3rd │ │ │ │ │ + * party layer must implement its own getWarningHTML() function to │ │ │ │ │ + * provide the actual warning message. │ │ │ │ │ */ │ │ │ │ │ - graphicRotate: function(node, xOffset, yOffset, style) { │ │ │ │ │ - var style = style || node._style; │ │ │ │ │ - var rotation = style.rotation || 0; │ │ │ │ │ + loadWarningMessage: function() { │ │ │ │ │ │ │ │ │ │ - var aspectRatio, size; │ │ │ │ │ - if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ - // load the image to determine its size │ │ │ │ │ - var img = new Image(); │ │ │ │ │ - img.onreadystatechange = OpenLayers.Function.bind(function() { │ │ │ │ │ - if (img.readyState == "complete" || │ │ │ │ │ - img.readyState == "interactive") { │ │ │ │ │ - aspectRatio = img.width / img.height; │ │ │ │ │ - size = Math.max(style.pointRadius * 2, │ │ │ │ │ - style.graphicWidth || 0, │ │ │ │ │ - style.graphicHeight || 0); │ │ │ │ │ - xOffset = xOffset * aspectRatio; │ │ │ │ │ - style.graphicWidth = size * aspectRatio; │ │ │ │ │ - style.graphicHeight = size; │ │ │ │ │ - this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ - } │ │ │ │ │ - }, this); │ │ │ │ │ - img.src = style.externalGraphic; │ │ │ │ │ + this.div.style.backgroundColor = "darkblue"; │ │ │ │ │ │ │ │ │ │ - // will be called again by the onreadystate handler │ │ │ │ │ - return; │ │ │ │ │ - } else { │ │ │ │ │ - size = Math.max(style.graphicWidth, style.graphicHeight); │ │ │ │ │ - aspectRatio = style.graphicWidth / style.graphicHeight; │ │ │ │ │ - } │ │ │ │ │ + var viewSize = this.map.getSize(); │ │ │ │ │ │ │ │ │ │ - var width = Math.round(style.graphicWidth || size * aspectRatio); │ │ │ │ │ - var height = Math.round(style.graphicHeight || size); │ │ │ │ │ - node.style.width = width + "px"; │ │ │ │ │ - node.style.height = height + "px"; │ │ │ │ │ + var msgW = Math.min(viewSize.w, 300); │ │ │ │ │ + var msgH = Math.min(viewSize.h, 200); │ │ │ │ │ + var size = new OpenLayers.Size(msgW, msgH); │ │ │ │ │ │ │ │ │ │ - // Three steps are required to remove artefacts for images with │ │ │ │ │ - // transparent backgrounds (resulting from using DXImageTransform │ │ │ │ │ - // filters on svg objects), while preserving awareness for browser │ │ │ │ │ - // events on images: │ │ │ │ │ - // - Use the fill as usual (like for unrotated images) to handle │ │ │ │ │ - // events │ │ │ │ │ - // - specify an imagedata element with the same src as the fill │ │ │ │ │ - // - style the imagedata element with an AlphaImageLoader filter │ │ │ │ │ - // with empty src │ │ │ │ │ - var image = document.getElementById(node.id + "_image"); │ │ │ │ │ - if (!image) { │ │ │ │ │ - image = this.createNode("olv:imagedata", node.id + "_image"); │ │ │ │ │ - node.appendChild(image); │ │ │ │ │ - } │ │ │ │ │ - image.style.width = width + "px"; │ │ │ │ │ - image.style.height = height + "px"; │ │ │ │ │ - image.src = style.externalGraphic; │ │ │ │ │ - image.style.filter = │ │ │ │ │ - "progid:DXImageTransform.Microsoft.AlphaImageLoader(" + │ │ │ │ │ - "src='', sizingMethod='scale')"; │ │ │ │ │ + var centerPx = new OpenLayers.Pixel(viewSize.w / 2, viewSize.h / 2); │ │ │ │ │ │ │ │ │ │ - var rot = rotation * Math.PI / 180; │ │ │ │ │ - var sintheta = Math.sin(rot); │ │ │ │ │ - var costheta = Math.cos(rot); │ │ │ │ │ + var topLeft = centerPx.add(-size.w / 2, -size.h / 2); │ │ │ │ │ │ │ │ │ │ - // do the rotation on the image │ │ │ │ │ - var filter = │ │ │ │ │ - "progid:DXImageTransform.Microsoft.Matrix(M11=" + costheta + │ │ │ │ │ - ",M12=" + (-sintheta) + ",M21=" + sintheta + ",M22=" + costheta + │ │ │ │ │ - ",SizingMethod='auto expand')\n"; │ │ │ │ │ + var div = OpenLayers.Util.createDiv(this.name + "_warning", │ │ │ │ │ + topLeft, │ │ │ │ │ + size, │ │ │ │ │ + null, │ │ │ │ │ + null, │ │ │ │ │ + null, │ │ │ │ │ + "auto"); │ │ │ │ │ │ │ │ │ │ - // set the opacity (needed for the imagedata) │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ - if (opacity && opacity != 1) { │ │ │ │ │ - filter += │ │ │ │ │ - "progid:DXImageTransform.Microsoft.BasicImage(opacity=" + │ │ │ │ │ - opacity + ")\n"; │ │ │ │ │ - } │ │ │ │ │ - node.style.filter = filter; │ │ │ │ │ + div.style.padding = "7px"; │ │ │ │ │ + div.style.backgroundColor = "yellow"; │ │ │ │ │ │ │ │ │ │ - // do the rotation again on a box, so we know the insertion point │ │ │ │ │ - var centerPoint = new OpenLayers.Geometry.Point(-xOffset, -yOffset); │ │ │ │ │ - var imgBox = new OpenLayers.Bounds(0, 0, width, height).toGeometry(); │ │ │ │ │ - imgBox.rotate(style.rotation, centerPoint); │ │ │ │ │ - var imgBounds = imgBox.getBounds(); │ │ │ │ │ + div.innerHTML = this.getWarningHTML(); │ │ │ │ │ + this.div.appendChild(div); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - node.style.left = Math.round( │ │ │ │ │ - parseInt(node.style.left) + imgBounds.left) + "px"; │ │ │ │ │ - node.style.top = Math.round( │ │ │ │ │ - parseInt(node.style.top) - imgBounds.bottom) + "px"; │ │ │ │ │ + /** │ │ │ │ │ + * Method: getWarningHTML │ │ │ │ │ + * To be implemented by subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} String with information on why layer is broken, how to get │ │ │ │ │ + * it working. │ │ │ │ │ + */ │ │ │ │ │ + getWarningHTML: function() { │ │ │ │ │ + //should be implemented by subclasses │ │ │ │ │ + return ""; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: postDraw │ │ │ │ │ - * Does some node postprocessing to work around browser issues: │ │ │ │ │ - * - Some versions of Internet Explorer seem to be unable to set fillcolor │ │ │ │ │ - * and strokecolor to "none" correctly before the fill node is appended │ │ │ │ │ - * to a visible vml node. This method takes care of that and sets │ │ │ │ │ - * fillcolor and strokecolor again if needed. │ │ │ │ │ - * - In some cases, a node won't become visible after being drawn. Setting │ │ │ │ │ - * style.visibility to "visible" works around that. │ │ │ │ │ - * │ │ │ │ │ + * Method: display │ │ │ │ │ + * Set the display on the pane │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ + * display - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - postDraw: function(node) { │ │ │ │ │ - node.style.visibility = "visible"; │ │ │ │ │ - var fillColor = node._style.fillColor; │ │ │ │ │ - var strokeColor = node._style.strokeColor; │ │ │ │ │ - if (fillColor == "none" && │ │ │ │ │ - node.fillcolor != fillColor) { │ │ │ │ │ - node.fillcolor = fillColor; │ │ │ │ │ - } │ │ │ │ │ - if (strokeColor == "none" && │ │ │ │ │ - node.strokecolor != strokeColor) { │ │ │ │ │ - node.strokecolor = strokeColor; │ │ │ │ │ - } │ │ │ │ │ + display: function(display) { │ │ │ │ │ + OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ + this.pane.style.display = this.div.style.display; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: setNodeDimension │ │ │ │ │ - * Get the geometry's bounds, convert it to our vml coordinate system, │ │ │ │ │ - * then set the node's position, size, and local coordinate system. │ │ │ │ │ - * │ │ │ │ │ + * Method: setZIndex │ │ │ │ │ + * Set the z-index order for the pane. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * zIndex - {int} │ │ │ │ │ */ │ │ │ │ │ - setNodeDimension: function(node, geometry) { │ │ │ │ │ - │ │ │ │ │ - var bbox = geometry.getBounds(); │ │ │ │ │ - if (bbox) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - │ │ │ │ │ - var scaledBox = │ │ │ │ │ - new OpenLayers.Bounds(((bbox.left - this.featureDx) / resolution - this.offset.x) | 0, │ │ │ │ │ - (bbox.bottom / resolution - this.offset.y) | 0, │ │ │ │ │ - ((bbox.right - this.featureDx) / resolution - this.offset.x) | 0, │ │ │ │ │ - (bbox.top / resolution - this.offset.y) | 0); │ │ │ │ │ + setZIndex: function(zIndex) { │ │ │ │ │ + OpenLayers.Layer.prototype.setZIndex.apply(this, arguments); │ │ │ │ │ + this.pane.style.zIndex = parseInt(this.div.style.zIndex) + 1; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Set the internal coordinate system to draw the path │ │ │ │ │ - node.style.left = scaledBox.left + "px"; │ │ │ │ │ - node.style.top = scaledBox.top + "px"; │ │ │ │ │ - node.style.width = scaledBox.getWidth() + "px"; │ │ │ │ │ - node.style.height = scaledBox.getHeight() + "px"; │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveByPx │ │ │ │ │ + * Move the layer based on pixel vector. To be implemented by subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * dx - {Number} The x coord of the displacement vector. │ │ │ │ │ + * dy - {Number} The y coord of the displacement vector. │ │ │ │ │ + */ │ │ │ │ │ + moveByPx: function(dx, dy) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveByPx.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - node.coordorigin = scaledBox.left + " " + scaledBox.top; │ │ │ │ │ - node.coordsize = scaledBox.getWidth() + " " + scaledBox.getHeight(); │ │ │ │ │ + if (this.dragPanMapObject) { │ │ │ │ │ + this.dragPanMapObject(dx, -dy); │ │ │ │ │ + } else { │ │ │ │ │ + this.moveTo(this.map.getCachedCenter()); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: dashStyle │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * Handle calls to move the layer. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A VML compliant 'stroke-dasharray' value │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * zoomChanged - {Boolean} │ │ │ │ │ + * dragging - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - dashStyle: function(style) { │ │ │ │ │ - var dash = style.strokeDashstyle; │ │ │ │ │ - switch (dash) { │ │ │ │ │ - case 'solid': │ │ │ │ │ - case 'dot': │ │ │ │ │ - case 'dash': │ │ │ │ │ - case 'dashdot': │ │ │ │ │ - case 'longdash': │ │ │ │ │ - case 'longdashdot': │ │ │ │ │ - return dash; │ │ │ │ │ - default: │ │ │ │ │ - // very basic guessing of dash style patterns │ │ │ │ │ - var parts = dash.split(/[ ,]/); │ │ │ │ │ - if (parts.length == 2) { │ │ │ │ │ - if (1 * parts[0] >= 2 * parts[1]) { │ │ │ │ │ - return "longdash"; │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + if (this.mapObject != null) { │ │ │ │ │ + │ │ │ │ │ + var newCenter = this.map.getCenter(); │ │ │ │ │ + var newZoom = this.map.getZoom(); │ │ │ │ │ + │ │ │ │ │ + if (newCenter != null) { │ │ │ │ │ + │ │ │ │ │ + var moOldCenter = this.getMapObjectCenter(); │ │ │ │ │ + var oldCenter = this.getOLLonLatFromMapObjectLonLat(moOldCenter); │ │ │ │ │ + │ │ │ │ │ + var moOldZoom = this.getMapObjectZoom(); │ │ │ │ │ + var oldZoom = this.getOLZoomFromMapObjectZoom(moOldZoom); │ │ │ │ │ + │ │ │ │ │ + if (!(newCenter.equals(oldCenter)) || newZoom != oldZoom) { │ │ │ │ │ + │ │ │ │ │ + if (!zoomChanged && oldCenter && this.dragPanMapObject && │ │ │ │ │ + this.smoothDragPan) { │ │ │ │ │ + var oldPx = this.map.getViewPortPxFromLonLat(oldCenter); │ │ │ │ │ + var newPx = this.map.getViewPortPxFromLonLat(newCenter); │ │ │ │ │ + this.dragPanMapObject(newPx.x - oldPx.x, oldPx.y - newPx.y); │ │ │ │ │ + } else { │ │ │ │ │ + var center = this.getMapObjectLonLatFromOLLonLat(newCenter); │ │ │ │ │ + var zoom = this.getMapObjectZoomFromOLZoom(newZoom); │ │ │ │ │ + this.setMapObjectCenter(center, zoom, dragging); │ │ │ │ │ } │ │ │ │ │ - return (parts[0] == 1 || parts[1] == 1) ? "dot" : "dash"; │ │ │ │ │ - } else if (parts.length == 4) { │ │ │ │ │ - return (1 * parts[0] >= 2 * parts[1]) ? "longdashdot" : │ │ │ │ │ - "dashdot"; │ │ │ │ │ } │ │ │ │ │ - return "solid"; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ + /********************************************************/ │ │ │ │ │ + /* */ │ │ │ │ │ + /* Baselayer Functions */ │ │ │ │ │ + /* */ │ │ │ │ │ + /********************************************************/ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: createNode │ │ │ │ │ - * Create a new node │ │ │ │ │ - * │ │ │ │ │ + * Method: getLonLatFromViewPortPx │ │ │ │ │ + * Get a map location from a pixel location │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * type - {String} Kind of node to draw │ │ │ │ │ - * id - {String} Id for node │ │ │ │ │ + * viewPortPx - {<OpenLayers.Pixel>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} A new node of the given type and id │ │ │ │ │ + * {<OpenLayers.LonLat>} An OpenLayers.LonLat which is the passed-in view │ │ │ │ │ + * port OpenLayers.Pixel, translated into lon/lat by map lib │ │ │ │ │ + * If the map lib is not loaded or not centered, returns null │ │ │ │ │ */ │ │ │ │ │ - createNode: function(type, id) { │ │ │ │ │ - var node = document.createElement(type); │ │ │ │ │ - if (id) { │ │ │ │ │ - node.id = id; │ │ │ │ │ + getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ + var lonlat = null; │ │ │ │ │ + if ((this.mapObject != null) && │ │ │ │ │ + (this.getMapObjectCenter() != null)) { │ │ │ │ │ + var moPixel = this.getMapObjectPixelFromOLPixel(viewPortPx); │ │ │ │ │ + var moLonLat = this.getMapObjectLonLatFromMapObjectPixel(moPixel); │ │ │ │ │ + lonlat = this.getOLLonLatFromMapObjectLonLat(moLonLat); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // IE hack to make elements unselectable, to prevent 'blue flash' │ │ │ │ │ - // while dragging vectors; #1410 │ │ │ │ │ - node.unselectable = 'on'; │ │ │ │ │ - node.onselectstart = OpenLayers.Function.False; │ │ │ │ │ - │ │ │ │ │ - return node; │ │ │ │ │ + return lonlat; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: nodeTypeCompare │ │ │ │ │ - * Determine whether a node is of a given type │ │ │ │ │ + * Method: getViewPortPxFromLonLat │ │ │ │ │ + * Get a pixel location from a map location │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} An VML element │ │ │ │ │ - * type - {String} Kind of node │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ + * {<OpenLayers.Pixel>} An OpenLayers.Pixel which is the passed-in │ │ │ │ │ + * OpenLayers.LonLat, translated into view port pixels by map lib │ │ │ │ │ + * If map lib is not loaded or not centered, returns null │ │ │ │ │ */ │ │ │ │ │ - nodeTypeCompare: function(node, type) { │ │ │ │ │ + getViewPortPxFromLonLat: function(lonlat) { │ │ │ │ │ + var viewPortPx = null; │ │ │ │ │ + if ((this.mapObject != null) && │ │ │ │ │ + (this.getMapObjectCenter() != null)) { │ │ │ │ │ │ │ │ │ │ - //split type │ │ │ │ │ - var subType = type; │ │ │ │ │ - var splitIndex = subType.indexOf(":"); │ │ │ │ │ - if (splitIndex != -1) { │ │ │ │ │ - subType = subType.substr(splitIndex + 1); │ │ │ │ │ - } │ │ │ │ │ + var moLonLat = this.getMapObjectLonLatFromOLLonLat(lonlat); │ │ │ │ │ + var moPixel = this.getMapObjectPixelFromMapObjectLonLat(moLonLat); │ │ │ │ │ │ │ │ │ │ - //split nodeName │ │ │ │ │ - var nodeName = node.nodeName; │ │ │ │ │ - splitIndex = nodeName.indexOf(":"); │ │ │ │ │ - if (splitIndex != -1) { │ │ │ │ │ - nodeName = nodeName.substr(splitIndex + 1); │ │ │ │ │ + viewPortPx = this.getOLPixelFromMapObjectPixel(moPixel); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return (subType == nodeName); │ │ │ │ │ + return viewPortPx; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + /********************************************************/ │ │ │ │ │ + /* */ │ │ │ │ │ + /* Translation Functions */ │ │ │ │ │ + /* */ │ │ │ │ │ + /* The following functions translate Map Object and */ │ │ │ │ │ + /* OL formats for Pixel, LonLat */ │ │ │ │ │ + /* */ │ │ │ │ │ + /********************************************************/ │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + // TRANSLATION: MapObject LatLng <-> OpenLayers.LonLat │ │ │ │ │ + // │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: createRenderRoot │ │ │ │ │ - * Create the renderer root │ │ │ │ │ + * Method: getOLLonLatFromMapObjectLonLat │ │ │ │ │ + * Get an OL style map location from a 3rd party style map location │ │ │ │ │ * │ │ │ │ │ + * Parameters │ │ │ │ │ + * moLonLat - {Object} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} The specific render engine's root element │ │ │ │ │ + * {<OpenLayers.LonLat>} An OpenLayers.LonLat, translated from the passed in │ │ │ │ │ + * MapObject LonLat │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ */ │ │ │ │ │ - createRenderRoot: function() { │ │ │ │ │ - return this.nodeFactory(this.container.id + "_vmlRoot", "div"); │ │ │ │ │ + getOLLonLatFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + var olLonLat = null; │ │ │ │ │ + if (moLonLat != null) { │ │ │ │ │ + var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + olLonLat = new OpenLayers.LonLat(lon, lat); │ │ │ │ │ + } │ │ │ │ │ + return olLonLat; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createRoot │ │ │ │ │ - * Create the main root element │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * suffix - {String} suffix to append to the id │ │ │ │ │ + * Method: getMapObjectLonLatFromOLLonLat │ │ │ │ │ + * Get a 3rd party map location from an OL map location. │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * olLonLat - {<OpenLayers.LonLat>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * {Object} A MapObject LonLat, translated from the passed in │ │ │ │ │ + * OpenLayers.LonLat │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ */ │ │ │ │ │ - createRoot: function(suffix) { │ │ │ │ │ - return this.nodeFactory(this.container.id + suffix, "olv:group"); │ │ │ │ │ + getMapObjectLonLatFromOLLonLat: function(olLonLat) { │ │ │ │ │ + var moLatLng = null; │ │ │ │ │ + if (olLonLat != null) { │ │ │ │ │ + moLatLng = this.getMapObjectLonLatFromLonLat(olLonLat.lon, │ │ │ │ │ + olLonLat.lat); │ │ │ │ │ + } │ │ │ │ │ + return moLatLng; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /************************************** │ │ │ │ │ - * * │ │ │ │ │ - * GEOMETRY DRAWING FUNCTIONS * │ │ │ │ │ - * * │ │ │ │ │ - **************************************/ │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + // TRANSLATION: MapObject Pixel <-> OpenLayers.Pixel │ │ │ │ │ + // │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPoint │ │ │ │ │ - * Render a point │ │ │ │ │ - * │ │ │ │ │ + * Method: getOLPixelFromMapObjectPixel │ │ │ │ │ + * Get an OL pixel location from a 3rd party pixel location. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * moPixel - {Object} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or false if the point could not be drawn │ │ │ │ │ + * {<OpenLayers.Pixel>} An OpenLayers.Pixel, translated from the passed in │ │ │ │ │ + * MapObject Pixel │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ */ │ │ │ │ │ - drawPoint: function(node, geometry) { │ │ │ │ │ - return this.drawCircle(node, geometry, 1); │ │ │ │ │ + getOLPixelFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + var olPixel = null; │ │ │ │ │ + if (moPixel != null) { │ │ │ │ │ + var x = this.getXFromMapObjectPixel(moPixel); │ │ │ │ │ + var y = this.getYFromMapObjectPixel(moPixel); │ │ │ │ │ + olPixel = new OpenLayers.Pixel(x, y); │ │ │ │ │ + } │ │ │ │ │ + return olPixel; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawCircle │ │ │ │ │ - * Render a circle. │ │ │ │ │ - * Size and Center a circle given geometry (x,y center) and radius │ │ │ │ │ - * │ │ │ │ │ + * Method: getMapObjectPixelFromOLPixel │ │ │ │ │ + * Get a 3rd party pixel location from an OL pixel location │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * radius - {float} │ │ │ │ │ + * olPixel - {<OpenLayers.Pixel>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or false if the circle could not ne drawn │ │ │ │ │ + * {Object} A MapObject Pixel, translated from the passed in │ │ │ │ │ + * OpenLayers.Pixel │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ */ │ │ │ │ │ - drawCircle: function(node, geometry, radius) { │ │ │ │ │ - if (!isNaN(geometry.x) && !isNaN(geometry.y)) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ + getMapObjectPixelFromOLPixel: function(olPixel) { │ │ │ │ │ + var moPixel = null; │ │ │ │ │ + if (olPixel != null) { │ │ │ │ │ + moPixel = this.getMapObjectPixelFromXY(olPixel.x, olPixel.y); │ │ │ │ │ + } │ │ │ │ │ + return moPixel; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - node.style.left = ((((geometry.x - this.featureDx) / resolution - this.offset.x) | 0) - radius) + "px"; │ │ │ │ │ - node.style.top = (((geometry.y / resolution - this.offset.y) | 0) - radius) + "px"; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.EventPane" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/FixedZoomLevels.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - var diameter = radius * 2; │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - node.style.width = diameter + "px"; │ │ │ │ │ - node.style.height = diameter + "px"; │ │ │ │ │ - return node; │ │ │ │ │ - } │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.FixedZoomLevels │ │ │ │ │ + * Some Layers will already have established zoom levels (like google │ │ │ │ │ + * or ve). Instead of trying to determine them and populate a resolutions[] │ │ │ │ │ + * Array with those values, we will hijack the resolution functionality │ │ │ │ │ + * here. │ │ │ │ │ + * │ │ │ │ │ + * When you subclass FixedZoomLevels: │ │ │ │ │ + * │ │ │ │ │ + * The initResolutions() call gets nullified, meaning no resolutions[] array │ │ │ │ │ + * is set up. Which would be a big problem getResolution() in Layer, since │ │ │ │ │ + * it merely takes map.zoom and indexes into resolutions[]... but.... │ │ │ │ │ + * │ │ │ │ │ + * The getResolution() call is also overridden. Instead of using the │ │ │ │ │ + * resolutions[] array, we simply calculate the current resolution based │ │ │ │ │ + * on the current extent and the current map size. But how will we be able │ │ │ │ │ + * to calculate the current extent without knowing the resolution...? │ │ │ │ │ + * │ │ │ │ │ + * The getExtent() function is also overridden. Instead of calculating extent │ │ │ │ │ + * based on the center point and the current resolution, we instead │ │ │ │ │ + * calculate the extent by getting the lonlats at the top-left and │ │ │ │ │ + * bottom-right by using the getLonLatFromViewPortPx() translation function, │ │ │ │ │ + * taken from the pixel locations (0,0) and the size of the map. But how │ │ │ │ │ + * will we be able to do lonlat-px translation without resolution....? │ │ │ │ │ + * │ │ │ │ │ + * The getZoomForResolution() method is overridden. Instead of indexing into │ │ │ │ │ + * the resolutions[] array, we call OpenLayers.Layer.getExent(), passing in │ │ │ │ │ + * the desired resolution. With this extent, we then call getZoomForExtent() │ │ │ │ │ + * │ │ │ │ │ + * │ │ │ │ │ + * Whenever you implement a layer using OpenLayers.Layer.FixedZoomLevels, │ │ │ │ │ + * it is your responsibility to provide the following three functions: │ │ │ │ │ + * │ │ │ │ │ + * - getLonLatFromViewPortPx │ │ │ │ │ + * - getViewPortPxFromLonLat │ │ │ │ │ + * - getZoomForExtent │ │ │ │ │ + * │ │ │ │ │ + * ...those three functions should generally be provided by any reasonable │ │ │ │ │ + * API that you might be working from. │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.FixedZoomLevels = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawLineString │ │ │ │ │ - * Render a linestring. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - drawLineString: function(node, geometry) { │ │ │ │ │ - return this.drawLine(node, geometry, false); │ │ │ │ │ - }, │ │ │ │ │ + /********************************************************/ │ │ │ │ │ + /* */ │ │ │ │ │ + /* Baselayer Functions */ │ │ │ │ │ + /* */ │ │ │ │ │ + /* The following functions must all be implemented */ │ │ │ │ │ + /* by all base layers */ │ │ │ │ │ + /* */ │ │ │ │ │ + /********************************************************/ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLinearRing │ │ │ │ │ - * Render a linearring │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Constructor: OpenLayers.Layer.FixedZoomLevels │ │ │ │ │ + * Create a new fixed zoom levels layer. │ │ │ │ │ */ │ │ │ │ │ - drawLinearRing: function(node, geometry) { │ │ │ │ │ - return this.drawLine(node, geometry, true); │ │ │ │ │ + initialize: function() { │ │ │ │ │ + //this class is only just to add the following functions... │ │ │ │ │ + // nothing to actually do here... but it is probably a good │ │ │ │ │ + // idea to have layers that use these functions call this │ │ │ │ │ + // inititalize() anyways, in case at some point we decide we │ │ │ │ │ + // do want to put some functionality or state in here. │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: DrawLine │ │ │ │ │ - * Render a line. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * closeLine - {Boolean} Close the line? (make it a ring?) │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Method: initResolutions │ │ │ │ │ + * Populate the resolutions array │ │ │ │ │ */ │ │ │ │ │ - drawLine: function(node, geometry, closeLine) { │ │ │ │ │ + initResolutions: function() { │ │ │ │ │ │ │ │ │ │ - this.setNodeDimension(node, geometry); │ │ │ │ │ + var props = ['minZoomLevel', 'maxZoomLevel', 'numZoomLevels']; │ │ │ │ │ │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var numComponents = geometry.components.length; │ │ │ │ │ - var parts = new Array(numComponents); │ │ │ │ │ + for (var i = 0, len = props.length; i < len; i++) { │ │ │ │ │ + var property = props[i]; │ │ │ │ │ + this[property] = (this.options[property] != null) ? │ │ │ │ │ + this.options[property] : │ │ │ │ │ + this.map[property]; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var comp, x, y; │ │ │ │ │ - for (var i = 0; i < numComponents; i++) { │ │ │ │ │ - comp = geometry.components[i]; │ │ │ │ │ - x = ((comp.x - this.featureDx) / resolution - this.offset.x) | 0; │ │ │ │ │ - y = (comp.y / resolution - this.offset.y) | 0; │ │ │ │ │ - parts[i] = " " + x + "," + y + " l "; │ │ │ │ │ + if ((this.minZoomLevel == null) || │ │ │ │ │ + (this.minZoomLevel < this.MIN_ZOOM_LEVEL)) { │ │ │ │ │ + this.minZoomLevel = this.MIN_ZOOM_LEVEL; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + // At this point, we know what the minimum desired zoom level is, and │ │ │ │ │ + // we must calculate the total number of zoom levels. │ │ │ │ │ + // │ │ │ │ │ + // Because we allow for the setting of either the 'numZoomLevels' │ │ │ │ │ + // or the 'maxZoomLevel' properties... on either the layer or the │ │ │ │ │ + // map, we have to define some rules to see which we take into │ │ │ │ │ + // account first in this calculation. │ │ │ │ │ + // │ │ │ │ │ + // The following is the precedence list for these properties: │ │ │ │ │ + // │ │ │ │ │ + // (1) numZoomLevels set on layer │ │ │ │ │ + // (2) maxZoomLevel set on layer │ │ │ │ │ + // (3) numZoomLevels set on map │ │ │ │ │ + // (4) maxZoomLevel set on map* │ │ │ │ │ + // (5) none of the above* │ │ │ │ │ + // │ │ │ │ │ + // *Note that options (4) and (5) are only possible if the user │ │ │ │ │ + // _explicitly_ sets the 'numZoomLevels' property on the map to │ │ │ │ │ + // null, since it is set by default to 16. │ │ │ │ │ + // │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + // Note to future: In 3.0, I think we should remove the default │ │ │ │ │ + // value of 16 for map.numZoomLevels. Rather, I think that value │ │ │ │ │ + // should be set as a default on the Layer.WMS class. If someone │ │ │ │ │ + // creates a 3rd party layer and does not specify any 'minZoomLevel', │ │ │ │ │ + // 'maxZoomLevel', or 'numZoomLevels', and has not explicitly │ │ │ │ │ + // specified any of those on the map object either.. then I think │ │ │ │ │ + // it is fair to say that s/he wants all the zoom levels available. │ │ │ │ │ + // │ │ │ │ │ + // By making map.numZoomLevels *null* by default, that will be the │ │ │ │ │ + // case. As it is, I don't feel comfortable changing that right now │ │ │ │ │ + // as it would be a glaring API change and actually would probably │ │ │ │ │ + // break many peoples' codes. │ │ │ │ │ + // │ │ │ │ │ + │ │ │ │ │ + //the number of zoom levels we'd like to have. │ │ │ │ │ + var desiredZoomLevels; │ │ │ │ │ + │ │ │ │ │ + //this is the maximum number of zoom levels the layer will allow, │ │ │ │ │ + // given the specified starting minimum zoom level. │ │ │ │ │ + var limitZoomLevels = this.MAX_ZOOM_LEVEL - this.minZoomLevel + 1; │ │ │ │ │ + │ │ │ │ │ + if (((this.options.numZoomLevels == null) && │ │ │ │ │ + (this.options.maxZoomLevel != null)) // (2) │ │ │ │ │ + || │ │ │ │ │ + ((this.numZoomLevels == null) && │ │ │ │ │ + (this.maxZoomLevel != null)) // (4) │ │ │ │ │ + ) { │ │ │ │ │ + //calculate based on specified maxZoomLevel (on layer or map) │ │ │ │ │ + desiredZoomLevels = this.maxZoomLevel - this.minZoomLevel + 1; │ │ │ │ │ + } else { │ │ │ │ │ + //calculate based on specified numZoomLevels (on layer or map) │ │ │ │ │ + // this covers cases (1) and (3) │ │ │ │ │ + desiredZoomLevels = this.numZoomLevels; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (desiredZoomLevels != null) { │ │ │ │ │ + //Now that we know what we would *like* the number of zoom levels │ │ │ │ │ + // to be, based on layer or map options, we have to make sure that │ │ │ │ │ + // it does not conflict with the actual limit, as specified by │ │ │ │ │ + // the constants on the layer itself (and calculated into the │ │ │ │ │ + // 'limitZoomLevels' variable). │ │ │ │ │ + this.numZoomLevels = Math.min(desiredZoomLevels, limitZoomLevels); │ │ │ │ │ + } else { │ │ │ │ │ + // case (5) -- neither 'numZoomLevels' not 'maxZoomLevel' was │ │ │ │ │ + // set on either the layer or the map. So we just use the │ │ │ │ │ + // maximum limit as calculated by the layer's constants. │ │ │ │ │ + this.numZoomLevels = limitZoomLevels; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //now that the 'numZoomLevels' is appropriately, safely set, │ │ │ │ │ + // we go back and re-calculate the 'maxZoomLevel'. │ │ │ │ │ + this.maxZoomLevel = this.minZoomLevel + this.numZoomLevels - 1; │ │ │ │ │ + │ │ │ │ │ + if (this.RESOLUTIONS != null) { │ │ │ │ │ + var resolutionsIndex = 0; │ │ │ │ │ + this.resolutions = []; │ │ │ │ │ + for (var i = this.minZoomLevel; i <= this.maxZoomLevel; i++) { │ │ │ │ │ + this.resolutions[resolutionsIndex++] = this.RESOLUTIONS[i]; │ │ │ │ │ + } │ │ │ │ │ + this.maxResolution = this.resolutions[0]; │ │ │ │ │ + this.minResolution = this.resolutions[this.resolutions.length - 1]; │ │ │ │ │ } │ │ │ │ │ - var end = (closeLine) ? " x e" : " e"; │ │ │ │ │ - node.path = "m" + parts.join("") + end; │ │ │ │ │ - return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPolygon │ │ │ │ │ - * Render a polygon │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * APIMethod: getResolution │ │ │ │ │ + * Get the current map resolution │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * {Float} Map units per Pixel │ │ │ │ │ */ │ │ │ │ │ - drawPolygon: function(node, geometry) { │ │ │ │ │ - this.setNodeDimension(node, geometry); │ │ │ │ │ + getResolution: function() { │ │ │ │ │ │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ + if (this.resolutions != null) { │ │ │ │ │ + return OpenLayers.Layer.prototype.getResolution.apply(this, arguments); │ │ │ │ │ + } else { │ │ │ │ │ + var resolution = null; │ │ │ │ │ │ │ │ │ │ - var path = []; │ │ │ │ │ - var j, jj, points, area, first, second, i, ii, comp, pathComp, x, y; │ │ │ │ │ - for (j = 0, jj = geometry.components.length; j < jj; j++) { │ │ │ │ │ - path.push("m"); │ │ │ │ │ - points = geometry.components[j].components; │ │ │ │ │ - // we only close paths of interior rings with area │ │ │ │ │ - area = (j === 0); │ │ │ │ │ - first = null; │ │ │ │ │ - second = null; │ │ │ │ │ - for (i = 0, ii = points.length; i < ii; i++) { │ │ │ │ │ - comp = points[i]; │ │ │ │ │ - x = ((comp.x - this.featureDx) / resolution - this.offset.x) | 0; │ │ │ │ │ - y = (comp.y / resolution - this.offset.y) | 0; │ │ │ │ │ - pathComp = " " + x + "," + y; │ │ │ │ │ - path.push(pathComp); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - path.push(" l"); │ │ │ │ │ - } │ │ │ │ │ - if (!area) { │ │ │ │ │ - // IE improperly renders sub-paths that have no area. │ │ │ │ │ - // Instead of checking the area of every ring, we confirm │ │ │ │ │ - // the ring has at least three distinct points. This does │ │ │ │ │ - // not catch all non-zero area cases, but it greatly improves │ │ │ │ │ - // interior ring digitizing and is a minor performance hit │ │ │ │ │ - // when rendering rings with many points. │ │ │ │ │ - if (!first) { │ │ │ │ │ - first = pathComp; │ │ │ │ │ - } else if (first != pathComp) { │ │ │ │ │ - if (!second) { │ │ │ │ │ - second = pathComp; │ │ │ │ │ - } else if (second != pathComp) { │ │ │ │ │ - // stop looking │ │ │ │ │ - area = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + var viewSize = this.map.getSize(); │ │ │ │ │ + var extent = this.getExtent(); │ │ │ │ │ + │ │ │ │ │ + if ((viewSize != null) && (extent != null)) { │ │ │ │ │ + resolution = Math.max(extent.getWidth() / viewSize.w, │ │ │ │ │ + extent.getHeight() / viewSize.h); │ │ │ │ │ } │ │ │ │ │ - path.push(area ? " x " : " "); │ │ │ │ │ + return resolution; │ │ │ │ │ } │ │ │ │ │ - path.push("e"); │ │ │ │ │ - node.path = path.join(""); │ │ │ │ │ - return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawRectangle │ │ │ │ │ - * Render a rectangle │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * APIMethod: getExtent │ │ │ │ │ + * Calculates using px-> lonlat translation functions on tl and br │ │ │ │ │ + * corners of viewport │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * {<OpenLayers.Bounds>} A Bounds object which represents the lon/lat │ │ │ │ │ + * bounds of the current viewPort. │ │ │ │ │ */ │ │ │ │ │ - drawRectangle: function(node, geometry) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - │ │ │ │ │ - node.style.left = (((geometry.x - this.featureDx) / resolution - this.offset.x) | 0) + "px"; │ │ │ │ │ - node.style.top = ((geometry.y / resolution - this.offset.y) | 0) + "px"; │ │ │ │ │ - node.style.width = ((geometry.width / resolution) | 0) + "px"; │ │ │ │ │ - node.style.height = ((geometry.height / resolution) | 0) + "px"; │ │ │ │ │ + getExtent: function() { │ │ │ │ │ + var size = this.map.getSize(); │ │ │ │ │ + var tl = this.getLonLatFromViewPortPx({ │ │ │ │ │ + x: 0, │ │ │ │ │ + y: 0 │ │ │ │ │ + }); │ │ │ │ │ + var br = this.getLonLatFromViewPortPx({ │ │ │ │ │ + x: size.w, │ │ │ │ │ + y: size.h │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - return node; │ │ │ │ │ + if ((tl != null) && (br != null)) { │ │ │ │ │ + return new OpenLayers.Bounds(tl.lon, br.lat, br.lon, tl.lat); │ │ │ │ │ + } else { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawText │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * style - │ │ │ │ │ - * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * Method: getZoomForResolution │ │ │ │ │ + * Get the zoom level for a given resolution │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resolution - {Float} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} A suitable zoom level for the specified resolution. │ │ │ │ │ + * If no baselayer is set, returns null. │ │ │ │ │ */ │ │ │ │ │ - drawText: function(featureId, style, location) { │ │ │ │ │ - var label = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX, "olv:rect"); │ │ │ │ │ - var textbox = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX + "_textbox", "olv:textbox"); │ │ │ │ │ + getZoomForResolution: function(resolution) { │ │ │ │ │ │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - label.style.left = (((location.x - this.featureDx) / resolution - this.offset.x) | 0) + "px"; │ │ │ │ │ - label.style.top = ((location.y / resolution - this.offset.y) | 0) + "px"; │ │ │ │ │ - label.style.flip = "y"; │ │ │ │ │ + if (this.resolutions != null) { │ │ │ │ │ + return OpenLayers.Layer.prototype.getZoomForResolution.apply(this, arguments); │ │ │ │ │ + } else { │ │ │ │ │ + var extent = OpenLayers.Layer.prototype.getExtent.apply(this, []); │ │ │ │ │ + return this.getZoomForExtent(extent); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - textbox.innerText = style.label; │ │ │ │ │ │ │ │ │ │ - if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ - textbox.style.cursor = style.cursor; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontColor) { │ │ │ │ │ - textbox.style.color = style.fontColor; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontOpacity) { │ │ │ │ │ - textbox.style.filter = 'alpha(opacity=' + (style.fontOpacity * 100) + ')'; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontFamily) { │ │ │ │ │ - textbox.style.fontFamily = style.fontFamily; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontSize) { │ │ │ │ │ - textbox.style.fontSize = style.fontSize; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontWeight) { │ │ │ │ │ - textbox.style.fontWeight = style.fontWeight; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStyle) { │ │ │ │ │ - textbox.style.fontStyle = style.fontStyle; │ │ │ │ │ - } │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - label._featureId = featureId; │ │ │ │ │ - textbox._featureId = featureId; │ │ │ │ │ - textbox._geometry = location; │ │ │ │ │ - textbox._geometryClass = location.CLASS_NAME; │ │ │ │ │ - } │ │ │ │ │ - textbox.style.whiteSpace = "nowrap"; │ │ │ │ │ - // fun with IE: IE7 in standards compliant mode does not display any │ │ │ │ │ - // text with a left inset of 0. So we set this to 1px and subtract one │ │ │ │ │ - // pixel later when we set label.style.left │ │ │ │ │ - textbox.inset = "1px,0px,0px,0px"; │ │ │ │ │ │ │ │ │ │ - if (!label.parentNode) { │ │ │ │ │ - label.appendChild(textbox); │ │ │ │ │ - this.textRoot.appendChild(label); │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - var align = style.labelAlign || "cm"; │ │ │ │ │ - if (align.length == 1) { │ │ │ │ │ - align += "m"; │ │ │ │ │ - } │ │ │ │ │ - var xshift = textbox.clientWidth * │ │ │ │ │ - (OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(0, 1)]); │ │ │ │ │ - var yshift = textbox.clientHeight * │ │ │ │ │ - (OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(1, 1)]); │ │ │ │ │ - label.style.left = parseInt(label.style.left) - xshift - 1 + "px"; │ │ │ │ │ - label.style.top = parseInt(label.style.top) + yshift + "px"; │ │ │ │ │ + /********************************************************/ │ │ │ │ │ + /* */ │ │ │ │ │ + /* Translation Functions */ │ │ │ │ │ + /* */ │ │ │ │ │ + /* The following functions translate GMaps and OL */ │ │ │ │ │ + /* formats for Pixel, LonLat, Bounds, and Zoom */ │ │ │ │ │ + /* */ │ │ │ │ │ + /********************************************************/ │ │ │ │ │ │ │ │ │ │ - }, │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + // TRANSLATION: MapObject Zoom <-> OpenLayers Zoom │ │ │ │ │ + // │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveRoot │ │ │ │ │ - * moves this renderer's root to a different renderer. │ │ │ │ │ - * │ │ │ │ │ + * Method: getOLZoomFromMapObjectZoom │ │ │ │ │ + * Get the OL zoom index from the map object zoom level │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * renderer - {<OpenLayers.Renderer>} target renderer for the moved root │ │ │ │ │ - * root - {DOMElement} optional root node. To be used when this renderer │ │ │ │ │ - * holds roots from multiple layers to tell this method which one to │ │ │ │ │ - * detach │ │ │ │ │ + * moZoom - {Integer} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} true if successful, false otherwise │ │ │ │ │ + * {Integer} An OpenLayers Zoom level, translated from the passed in zoom │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ */ │ │ │ │ │ - moveRoot: function(renderer) { │ │ │ │ │ - var layer = this.map.getLayer(renderer.container.id); │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Vector.RootContainer) { │ │ │ │ │ - layer = this.map.getLayer(this.container.id); │ │ │ │ │ + getOLZoomFromMapObjectZoom: function(moZoom) { │ │ │ │ │ + var zoom = null; │ │ │ │ │ + if (moZoom != null) { │ │ │ │ │ + zoom = moZoom - this.minZoomLevel; │ │ │ │ │ + if (this.map.baseLayer !== this) { │ │ │ │ │ + zoom = this.map.baseLayer.getZoomForResolution( │ │ │ │ │ + this.getResolutionForZoom(zoom) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - layer && layer.renderer.clear(); │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.moveRoot.apply(this, arguments); │ │ │ │ │ - layer && layer.redraw(); │ │ │ │ │ + return zoom; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: importSymbol │ │ │ │ │ - * add a new symbol definition from the rendererer's symbol hash │ │ │ │ │ - * │ │ │ │ │ + * Method: getMapObjectZoomFromOLZoom │ │ │ │ │ + * Get the map object zoom level from the OL zoom level │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * graphicName - {String} name of the symbol to import │ │ │ │ │ + * olZoom - {Integer} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} - hash of {DOMElement} "symbol" and {Number} "size" │ │ │ │ │ + * {Integer} A MapObject level, translated from the passed in olZoom │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ */ │ │ │ │ │ - importSymbol: function(graphicName) { │ │ │ │ │ - var id = this.container.id + "-" + graphicName; │ │ │ │ │ - │ │ │ │ │ - // check if symbol already exists in the cache │ │ │ │ │ - var cache = this.symbolCache[id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - return cache; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(graphicName + ' is not a valid symbol name'); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var symbolExtent = new OpenLayers.Bounds( │ │ │ │ │ - Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ - │ │ │ │ │ - var pathitems = ["m"]; │ │ │ │ │ - for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - var x = symbol[i]; │ │ │ │ │ - var y = symbol[i + 1]; │ │ │ │ │ - symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ - symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ - symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ - symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ - │ │ │ │ │ - pathitems.push(x); │ │ │ │ │ - pathitems.push(y); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - pathitems.push("l"); │ │ │ │ │ + getMapObjectZoomFromOLZoom: function(olZoom) { │ │ │ │ │ + var zoom = null; │ │ │ │ │ + if (olZoom != null) { │ │ │ │ │ + zoom = olZoom + this.minZoomLevel; │ │ │ │ │ + if (this.map.baseLayer !== this) { │ │ │ │ │ + zoom = this.getZoomForResolution( │ │ │ │ │ + this.map.baseLayer.getResolutionForZoom(zoom) │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - pathitems.push("x e"); │ │ │ │ │ - var path = pathitems.join(" "); │ │ │ │ │ - │ │ │ │ │ - var diff = (symbolExtent.getWidth() - symbolExtent.getHeight()) / 2; │ │ │ │ │ - if (diff > 0) { │ │ │ │ │ - symbolExtent.bottom = symbolExtent.bottom - diff; │ │ │ │ │ - symbolExtent.top = symbolExtent.top + diff; │ │ │ │ │ - } else { │ │ │ │ │ - symbolExtent.left = symbolExtent.left + diff; │ │ │ │ │ - symbolExtent.right = symbolExtent.right - diff; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - cache = { │ │ │ │ │ - path: path, │ │ │ │ │ - size: symbolExtent.getWidth(), // equals getHeight() now │ │ │ │ │ - left: symbolExtent.left, │ │ │ │ │ - bottom: symbolExtent.bottom │ │ │ │ │ - }; │ │ │ │ │ - this.symbolCache[id] = cache; │ │ │ │ │ - │ │ │ │ │ - return cache; │ │ │ │ │ + return zoom; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.VML" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.FixedZoomLevels" │ │ │ │ │ }); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.VML.LABEL_SHIFT │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.VML.LABEL_SHIFT = { │ │ │ │ │ - "l": 0, │ │ │ │ │ - "c": .5, │ │ │ │ │ - "r": 1, │ │ │ │ │ - "t": 0, │ │ │ │ │ - "m": .5, │ │ │ │ │ - "b": 1 │ │ │ │ │ -}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Renderer/SVG.js │ │ │ │ │ + OpenLayers/Layer/Google.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Renderer/Elements.js │ │ │ │ │ + * @requires OpenLayers/Layer/SphericalMercator.js │ │ │ │ │ + * @requires OpenLayers/Layer/EventPane.js │ │ │ │ │ + * @requires OpenLayers/Layer/FixedZoomLevels.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Renderer.SVG │ │ │ │ │ + * Class: OpenLayers.Layer.Google │ │ │ │ │ * │ │ │ │ │ - * Inherits: │ │ │ │ │ - * - <OpenLayers.Renderer.Elements> │ │ │ │ │ + * Provides a wrapper for Google's Maps API │ │ │ │ │ + * Normally the Terms of Use for this API do not allow wrapping, but Google │ │ │ │ │ + * have provided written consent to OpenLayers for this - see email in │ │ │ │ │ + * http://osgeo-org.1560.n6.nabble.com/Google-Maps-API-Terms-of-Use-changes-tp4910013p4911981.html │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.SphericalMercator> │ │ │ │ │ + * - <OpenLayers.Layer.EventPane> │ │ │ │ │ + * - <OpenLayers.Layer.FixedZoomLevels> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Renderer.SVG = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ +OpenLayers.Layer.Google = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Layer.EventPane, │ │ │ │ │ + OpenLayers.Layer.FixedZoomLevels, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: xmlns │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - xmlns: "http://www.w3.org/2000/svg", │ │ │ │ │ + /** │ │ │ │ │ + * Constant: MIN_ZOOM_LEVEL │ │ │ │ │ + * {Integer} 0 │ │ │ │ │ + */ │ │ │ │ │ + MIN_ZOOM_LEVEL: 0, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: xlinkns │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - xlinkns: "http://www.w3.org/1999/xlink", │ │ │ │ │ + /** │ │ │ │ │ + * Constant: MAX_ZOOM_LEVEL │ │ │ │ │ + * {Integer} 21 │ │ │ │ │ + */ │ │ │ │ │ + MAX_ZOOM_LEVEL: 21, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: MAX_PIXEL │ │ │ │ │ - * {Integer} Firefox has a limitation where values larger or smaller than │ │ │ │ │ - * about 15000 in an SVG document lock the browser up. This │ │ │ │ │ - * works around it. │ │ │ │ │ - */ │ │ │ │ │ - MAX_PIXEL: 15000, │ │ │ │ │ + /** │ │ │ │ │ + * Constant: RESOLUTIONS │ │ │ │ │ + * {Array(Float)} Hardcode these resolutions so that they are more closely │ │ │ │ │ + * tied with the standard wms projection │ │ │ │ │ + */ │ │ │ │ │ + RESOLUTIONS: [ │ │ │ │ │ + 1.40625, │ │ │ │ │ + 0.703125, │ │ │ │ │ + 0.3515625, │ │ │ │ │ + 0.17578125, │ │ │ │ │ + 0.087890625, │ │ │ │ │ + 0.0439453125, │ │ │ │ │ + 0.02197265625, │ │ │ │ │ + 0.010986328125, │ │ │ │ │ + 0.0054931640625, │ │ │ │ │ + 0.00274658203125, │ │ │ │ │ + 0.001373291015625, │ │ │ │ │ + 0.0006866455078125, │ │ │ │ │ + 0.00034332275390625, │ │ │ │ │ + 0.000171661376953125, │ │ │ │ │ + 0.0000858306884765625, │ │ │ │ │ + 0.00004291534423828125, │ │ │ │ │ + 0.00002145767211914062, │ │ │ │ │ + 0.00001072883605957031, │ │ │ │ │ + 0.00000536441802978515, │ │ │ │ │ + 0.00000268220901489257, │ │ │ │ │ + 0.0000013411045074462891, │ │ │ │ │ + 0.00000067055225372314453 │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: translationParameters │ │ │ │ │ - * {Object} Hash with "x" and "y" properties │ │ │ │ │ - */ │ │ │ │ │ - translationParameters: null, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: type │ │ │ │ │ + * {GMapType} │ │ │ │ │ + */ │ │ │ │ │ + type: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: symbolMetrics │ │ │ │ │ - * {Object} Cache for symbol metrics according to their svg coordinate │ │ │ │ │ - * space. This is an object keyed by the symbol's id, and values are │ │ │ │ │ - * an array of [width, centerX, centerY]. │ │ │ │ │ - */ │ │ │ │ │ - symbolMetrics: null, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: wrapDateLine │ │ │ │ │ + * {Boolean} Allow user to pan forever east/west. Default is true. │ │ │ │ │ + * Setting this to false only restricts panning if │ │ │ │ │ + * <sphericalMercator> is true. │ │ │ │ │ + */ │ │ │ │ │ + wrapDateLine: true, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Renderer.SVG │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * containerID - {String} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(containerID) { │ │ │ │ │ - if (!this.supported()) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.initialize.apply(this, │ │ │ │ │ - arguments); │ │ │ │ │ - this.translationParameters = { │ │ │ │ │ - x: 0, │ │ │ │ │ - y: 0 │ │ │ │ │ - }; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: sphericalMercator │ │ │ │ │ + * {Boolean} Should the map act as a mercator-projected map? This will │ │ │ │ │ + * cause all interactions with the map to be in the actual map │ │ │ │ │ + * projection, which allows support for vector drawing, overlaying │ │ │ │ │ + * other maps, etc. │ │ │ │ │ + */ │ │ │ │ │ + sphericalMercator: false, │ │ │ │ │ │ │ │ │ │ - this.symbolMetrics = {}; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {Number} The version of the Google Maps API │ │ │ │ │ + */ │ │ │ │ │ + version: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: supported │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the browser supports the SVG renderer │ │ │ │ │ - */ │ │ │ │ │ - supported: function() { │ │ │ │ │ - var svgFeature = "http://www.w3.org/TR/SVG11/feature#"; │ │ │ │ │ - return (document.implementation && │ │ │ │ │ - (document.implementation.hasFeature("org.w3c.svg", "1.0") || │ │ │ │ │ - document.implementation.hasFeature(svgFeature + "SVG", "1.1") || │ │ │ │ │ - document.implementation.hasFeature(svgFeature + "BasicStructure", "1.1"))); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.Google │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} A name for the layer. │ │ │ │ │ + * options - {Object} An optional object whose properties will be set │ │ │ │ │ + * on the layer. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + if (!options.version) { │ │ │ │ │ + options.version = typeof GMap2 === "function" ? "2" : "3"; │ │ │ │ │ + } │ │ │ │ │ + var mixin = OpenLayers.Layer.Google["v" + │ │ │ │ │ + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (mixin) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, mixin); │ │ │ │ │ + } else { │ │ │ │ │ + throw "Unsupported Google Maps API version: " + options.version; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: inValidRange │ │ │ │ │ - * See #669 for more information │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {Integer} │ │ │ │ │ - * y - {Integer} │ │ │ │ │ - * xyOnly - {Boolean} whether or not to just check for x and y, which means │ │ │ │ │ - * to not take the current translation parameters into account if true. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the 'x' and 'y' coordinates are in the │ │ │ │ │ - * valid range. │ │ │ │ │ - */ │ │ │ │ │ - inValidRange: function(x, y, xyOnly) { │ │ │ │ │ - var left = x + (xyOnly ? 0 : this.translationParameters.x); │ │ │ │ │ - var top = y + (xyOnly ? 0 : this.translationParameters.y); │ │ │ │ │ - return (left >= -this.MAX_PIXEL && left <= this.MAX_PIXEL && │ │ │ │ │ - top >= -this.MAX_PIXEL && top <= this.MAX_PIXEL); │ │ │ │ │ - }, │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, mixin.DEFAULTS); │ │ │ │ │ + if (options.maxExtent) { │ │ │ │ │ + options.maxExtent = options.maxExtent.clone(); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setExtent │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} │ │ │ │ │ - * resolutionChanged - {Boolean} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ - * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ - * False otherwise. │ │ │ │ │ - */ │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.initialize.apply(this, │ │ │ │ │ + [name, options]); │ │ │ │ │ + OpenLayers.Layer.FixedZoomLevels.prototype.initialize.apply(this, │ │ │ │ │ + [name, options]); │ │ │ │ │ │ │ │ │ │ - var resolution = this.getResolution(), │ │ │ │ │ - left = -extent.left / resolution, │ │ │ │ │ - top = extent.top / resolution; │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + OpenLayers.Util.extend(this, OpenLayers.Layer.SphericalMercator); │ │ │ │ │ + this.initMercatorParameters(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // If the resolution has changed, start over changing the corner, because │ │ │ │ │ - // the features will redraw. │ │ │ │ │ - if (resolutionChanged) { │ │ │ │ │ - this.left = left; │ │ │ │ │ - this.top = top; │ │ │ │ │ - // Set the viewbox │ │ │ │ │ - var extentString = "0 0 " + this.size.w + " " + this.size.h; │ │ │ │ │ + /** │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Google>} An exact clone of this layer │ │ │ │ │ + */ │ │ │ │ │ + clone: function() { │ │ │ │ │ + /** │ │ │ │ │ + * This method isn't intended to be called by a subclass and it │ │ │ │ │ + * doesn't call the same method on the superclass. We don't call │ │ │ │ │ + * the super's clone because we don't want properties that are set │ │ │ │ │ + * on this layer after initialize (i.e. this.mapObject etc.). │ │ │ │ │ + */ │ │ │ │ │ + return new OpenLayers.Layer.Google( │ │ │ │ │ + this.name, this.getOptions() │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "viewBox", extentString); │ │ │ │ │ - this.translate(this.xOffset, 0); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - var inRange = this.translate(left - this.left + this.xOffset, top - this.top); │ │ │ │ │ - if (!inRange) { │ │ │ │ │ - // recenter the coordinate system │ │ │ │ │ - this.setExtent(extent, true); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setVisibility │ │ │ │ │ + * Set the visibility flag for the layer and hide/show & redraw │ │ │ │ │ + * accordingly. Fire event unless otherwise specified │ │ │ │ │ + * │ │ │ │ │ + * Note that visibility is no longer simply whether or not the layer's │ │ │ │ │ + * style.display is set to "block". Now we store a 'visibility' state │ │ │ │ │ + * property on the layer class, this allows us to remember whether or │ │ │ │ │ + * not we *desire* for a layer to be visible. In the case where the │ │ │ │ │ + * map's resolution is out of the layer's range, this desire may be │ │ │ │ │ + * subverted. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * visible - {Boolean} Display the layer (if in range) │ │ │ │ │ + */ │ │ │ │ │ + setVisibility: function(visible) { │ │ │ │ │ + // sharing a map container, opacity has to be set per layer │ │ │ │ │ + var opacity = this.opacity == null ? 1 : this.opacity; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.setVisibility.apply(this, arguments); │ │ │ │ │ + this.setOpacity(opacity); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: display │ │ │ │ │ + * Hide or show the Layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * visible - {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + display: function(visible) { │ │ │ │ │ + if (!this._dragging) { │ │ │ │ │ + this.setGMapVisibility(visible); │ │ │ │ │ } │ │ │ │ │ - return coordSysUnchanged && inRange; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.display.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: translate │ │ │ │ │ - * Transforms the SVG coordinate system │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {Float} │ │ │ │ │ - * y - {Float} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true if the translation parameters are in the valid coordinates │ │ │ │ │ - * range, false otherwise. │ │ │ │ │ - */ │ │ │ │ │ - translate: function(x, y) { │ │ │ │ │ - if (!this.inValidRange(x, y, true)) { │ │ │ │ │ - return false; │ │ │ │ │ - } else { │ │ │ │ │ - var transformString = ""; │ │ │ │ │ - if (x || y) { │ │ │ │ │ - transformString = "translate(" + x + "," + y + ")"; │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * zoomChanged - {Boolean} Tells when zoom has changed, as layers have to │ │ │ │ │ + * do some init work in that case. │ │ │ │ │ + * dragging - {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + this._dragging = dragging; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + delete this._dragging; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setOpacity │ │ │ │ │ + * Sets the opacity for the entire layer (all images) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * opacity - {Float} │ │ │ │ │ + */ │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + if (opacity !== this.opacity) { │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "opacity" │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + this.opacity = opacity; │ │ │ │ │ } │ │ │ │ │ - this.root.setAttributeNS(null, "transform", transformString); │ │ │ │ │ - this.translationParameters = { │ │ │ │ │ - x: x, │ │ │ │ │ - y: y │ │ │ │ │ - }; │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + // Though this layer's opacity may not change, we're sharing a container │ │ │ │ │ + // and need to update the opacity for the entire container. │ │ │ │ │ + if (this.getVisibility()) { │ │ │ │ │ + var container = this.getMapContainer(); │ │ │ │ │ + OpenLayers.Util.modifyDOMElement( │ │ │ │ │ + container, null, null, null, null, null, null, opacity │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setSize │ │ │ │ │ - * Sets the size of the drawing surface. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>} The size of the drawing surface │ │ │ │ │ - */ │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up this layer. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + /** │ │ │ │ │ + * We have to override this method because the event pane destroy │ │ │ │ │ + * deletes the mapObject reference before removing this layer from │ │ │ │ │ + * the map. │ │ │ │ │ + */ │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.setGMapVisibility(false); │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache && cache.count <= 1) { │ │ │ │ │ + this.removeGMapElements(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "width", this.size.w); │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "height", this.size.h); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeGMapElements │ │ │ │ │ + * Remove all elements added to the dom. This should only be called if │ │ │ │ │ + * this is the last of the Google layers for the given map. │ │ │ │ │ + */ │ │ │ │ │ + removeGMapElements: function() { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + // remove shared elements from dom │ │ │ │ │ + var container = this.mapObject && this.getMapContainer(); │ │ │ │ │ + if (container && container.parentNode) { │ │ │ │ │ + container.parentNode.removeChild(container); │ │ │ │ │ + } │ │ │ │ │ + var termsOfUse = cache.termsOfUse; │ │ │ │ │ + if (termsOfUse && termsOfUse.parentNode) { │ │ │ │ │ + termsOfUse.parentNode.removeChild(termsOfUse); │ │ │ │ │ + } │ │ │ │ │ + var poweredBy = cache.poweredBy; │ │ │ │ │ + if (poweredBy && poweredBy.parentNode) { │ │ │ │ │ + poweredBy.parentNode.removeChild(poweredBy); │ │ │ │ │ + } │ │ │ │ │ + if (this.mapObject && window.google && google.maps && │ │ │ │ │ + google.maps.event && google.maps.event.clearListeners) { │ │ │ │ │ + google.maps.event.clearListeners(this.mapObject, 'tilesloaded'); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getNodeType │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The corresponding node type for the specified geometry │ │ │ │ │ - */ │ │ │ │ │ - getNodeType: function(geometry, style) { │ │ │ │ │ - var nodeType = null; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - nodeType = "image"; │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - nodeType = "svg"; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: removeMap │ │ │ │ │ + * On being removed from the map, also remove termsOfUse and poweredBy divs │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + // hide layer before removing │ │ │ │ │ + if (this.visibility && this.mapObject) { │ │ │ │ │ + this.setGMapVisibility(false); │ │ │ │ │ + } │ │ │ │ │ + // check to see if last Google layer in this map │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + if (cache.count <= 1) { │ │ │ │ │ + this.removeGMapElements(); │ │ │ │ │ + delete OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ } else { │ │ │ │ │ - nodeType = "circle"; │ │ │ │ │ + // decrement the layer count │ │ │ │ │ + --cache.count; │ │ │ │ │ } │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - nodeType = "rect"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - nodeType = "polyline"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - nodeType = "polygon"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - case "OpenLayers.Geometry.Curve": │ │ │ │ │ - nodeType = "path"; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - return nodeType; │ │ │ │ │ - }, │ │ │ │ │ + } │ │ │ │ │ + // remove references to gmap elements │ │ │ │ │ + delete this.termsOfUse; │ │ │ │ │ + delete this.poweredBy; │ │ │ │ │ + delete this.mapObject; │ │ │ │ │ + delete this.dragObject; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.removeMap.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setStyle │ │ │ │ │ - * Use to set all the style attributes to a SVG node. │ │ │ │ │ - * │ │ │ │ │ - * Takes care to adjust stroke width and point radius to be │ │ │ │ │ - * resolution-relative │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {SVGDomElement} An SVG element to decorate │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * options - {Object} Currently supported options include │ │ │ │ │ - * 'isFilled' {Boolean} and │ │ │ │ │ - * 'isStroked' {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - setStyle: function(node, style, options) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - options = options || node._options; │ │ │ │ │ + // │ │ │ │ │ + // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ + // │ │ │ │ │ │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - node.setAttributeNS(null, "title", title); │ │ │ │ │ - //Standards-conformant SVG │ │ │ │ │ - // Prevent duplicate nodes. See issue https://github.com/openlayers/openlayers/issues/92 │ │ │ │ │ - var titleNode = node.getElementsByTagName("title"); │ │ │ │ │ - if (titleNode.length > 0) { │ │ │ │ │ - titleNode[0].firstChild.textContent = title; │ │ │ │ │ - } else { │ │ │ │ │ - var label = this.nodeFactory(null, "title"); │ │ │ │ │ - label.textContent = title; │ │ │ │ │ - node.appendChild(label); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getOLBoundsFromMapObjectBounds │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moBounds - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} An <OpenLayers.Bounds>, translated from the │ │ │ │ │ + * passed-in MapObject Bounds. │ │ │ │ │ + * Returns null if null value is passed in. │ │ │ │ │ + */ │ │ │ │ │ + getOLBoundsFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + var olBounds = null; │ │ │ │ │ + if (moBounds != null) { │ │ │ │ │ + var sw = moBounds.getSouthWest(); │ │ │ │ │ + var ne = moBounds.getNorthEast(); │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + sw = this.forwardMercator(sw.lng(), sw.lat()); │ │ │ │ │ + ne = this.forwardMercator(ne.lng(), ne.lat()); │ │ │ │ │ + } else { │ │ │ │ │ + sw = new OpenLayers.LonLat(sw.lng(), sw.lat()); │ │ │ │ │ + ne = new OpenLayers.LonLat(ne.lng(), ne.lat()); │ │ │ │ │ + } │ │ │ │ │ + olBounds = new OpenLayers.Bounds(sw.lon, │ │ │ │ │ + sw.lat, │ │ │ │ │ + ne.lon, │ │ │ │ │ + ne.lat); │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ + return olBounds; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var r = parseFloat(node.getAttributeNS(null, "r")); │ │ │ │ │ - var widthFactor = 1; │ │ │ │ │ - var pos; │ │ │ │ │ - if (node._geometryClass == "OpenLayers.Geometry.Point" && r) { │ │ │ │ │ - node.style.visibility = ""; │ │ │ │ │ - if (style.graphic === false) { │ │ │ │ │ - node.style.visibility = "hidden"; │ │ │ │ │ - } else if (style.externalGraphic) { │ │ │ │ │ - pos = this.getPosition(node); │ │ │ │ │ - if (style.graphicWidth && style.graphicHeight) { │ │ │ │ │ - node.setAttributeNS(null, "preserveAspectRatio", "none"); │ │ │ │ │ - } │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ - style.graphicXOffset : -(0.5 * width); │ │ │ │ │ - var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ - style.graphicYOffset : -(0.5 * height); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getWarningHTML │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} String with information on why layer is broken, how to get │ │ │ │ │ + * it working. │ │ │ │ │ + */ │ │ │ │ │ + getWarningHTML: function() { │ │ │ │ │ + return OpenLayers.i18n("googleWarning"); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ │ │ │ │ │ - node.setAttributeNS(null, "x", (pos.x + xOffset).toFixed()); │ │ │ │ │ - node.setAttributeNS(null, "y", (pos.y + yOffset).toFixed()); │ │ │ │ │ - node.setAttributeNS(null, "width", width); │ │ │ │ │ - node.setAttributeNS(null, "height", height); │ │ │ │ │ - node.setAttributeNS(this.xlinkns, "xlink:href", style.externalGraphic); │ │ │ │ │ - node.setAttributeNS(null, "style", "opacity: " + opacity); │ │ │ │ │ - node.onclick = OpenLayers.Event.preventDefault; │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - // the symbol viewBox is three times as large as the symbol │ │ │ │ │ - var offset = style.pointRadius * 3; │ │ │ │ │ - var size = offset * 2; │ │ │ │ │ - var src = this.importSymbol(style.graphicName); │ │ │ │ │ - pos = this.getPosition(node); │ │ │ │ │ - widthFactor = this.symbolMetrics[src.id][0] * 3 / size; │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Interface Controls * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ │ │ │ │ │ - // remove the node from the dom before we modify it. This │ │ │ │ │ - // prevents various rendering issues in Safari and FF │ │ │ │ │ - var parent = node.parentNode; │ │ │ │ │ - var nextSibling = node.nextSibling; │ │ │ │ │ - if (parent) { │ │ │ │ │ - parent.removeChild(node); │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - // The more appropriate way to implement this would be use/defs, │ │ │ │ │ - // but due to various issues in several browsers, it is safer to │ │ │ │ │ - // copy the symbols instead of referencing them. │ │ │ │ │ - // See e.g. ticket http://trac.osgeo.org/openlayers/ticket/2985 │ │ │ │ │ - // and this email thread │ │ │ │ │ - // http://osgeo-org.1803224.n2.nabble.com/Select-Control-Ctrl-click-on-Feature-with-a-graphicName-opens-new-browser-window-tc5846039.html │ │ │ │ │ - node.firstChild && node.removeChild(node.firstChild); │ │ │ │ │ - node.appendChild(src.firstChild.cloneNode(true)); │ │ │ │ │ - node.setAttributeNS(null, "viewBox", src.getAttributeNS(null, "viewBox")); │ │ │ │ │ + // Get&Set Center, Zoom │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectCenter │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The mapObject's current center in Map Object format │ │ │ │ │ + */ │ │ │ │ │ + getMapObjectCenter: function() { │ │ │ │ │ + return this.mapObject.getCenter(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectZoom │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} The mapObject's current zoom, in Map Object format │ │ │ │ │ + */ │ │ │ │ │ + getMapObjectZoom: function() { │ │ │ │ │ + return this.mapObject.getZoom(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Primitives * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + // LonLat │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getLongitudeFromMapObjectLonLat │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} Longitude of the given MapObject LonLat │ │ │ │ │ + */ │ │ │ │ │ + getLongitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + return this.sphericalMercator ? │ │ │ │ │ + this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lon : │ │ │ │ │ + moLonLat.lng(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getLatitudeFromMapObjectLonLat │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} Latitude of the given MapObject LonLat │ │ │ │ │ + */ │ │ │ │ │ + getLatitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + var lat = this.sphericalMercator ? │ │ │ │ │ + this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lat : │ │ │ │ │ + moLonLat.lat(); │ │ │ │ │ + return lat; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + // Pixel │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getXFromMapObjectPixel │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moPixel - {Object} MapObject Pixel format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} X value of the MapObject Pixel │ │ │ │ │ + */ │ │ │ │ │ + getXFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return moPixel.x; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getYFromMapObjectPixel │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moPixel - {Object} MapObject Pixel format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} Y value of the MapObject Pixel │ │ │ │ │ + */ │ │ │ │ │ + getYFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return moPixel.y; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Google" │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Property: OpenLayers.Layer.Google.cache │ │ │ │ │ + * {Object} Cache for elements that should only be created once per map. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Google.cache = {}; │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Layer.Google.v2 │ │ │ │ │ + * │ │ │ │ │ + * Mixin providing functionality specific to the Google Maps API v2. │ │ │ │ │ + * │ │ │ │ │ + * This API has been deprecated by Google. │ │ │ │ │ + * Developers are encouraged to migrate to v3 of the API; support for this │ │ │ │ │ + * is provided by <OpenLayers.Layer.Google.v3> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Google.v2 = { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: termsOfUse │ │ │ │ │ + * {DOMElement} Div for Google's copyright and terms of use link │ │ │ │ │ + */ │ │ │ │ │ + termsOfUse: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: poweredBy │ │ │ │ │ + * {DOMElement} Div for Google's powered by logo and link │ │ │ │ │ + */ │ │ │ │ │ + poweredBy: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: dragObject │ │ │ │ │ + * {GDraggableObject} Since 2.93, Google has exposed the ability to get │ │ │ │ │ + * the maps GDraggableObject. We can now use this for smooth panning │ │ │ │ │ + */ │ │ │ │ │ + dragObject: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: loadMapObject │ │ │ │ │ + * Load the GMap and register appropriate event listeners. If we can't │ │ │ │ │ + * load GMap2, then display a warning message. │ │ │ │ │ + */ │ │ │ │ │ + loadMapObject: function() { │ │ │ │ │ + if (!this.type) { │ │ │ │ │ + this.type = G_NORMAL_MAP; │ │ │ │ │ + } │ │ │ │ │ + var mapObject, termsOfUse, poweredBy; │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + // there are already Google layers added to this map │ │ │ │ │ + mapObject = cache.mapObject; │ │ │ │ │ + termsOfUse = cache.termsOfUse; │ │ │ │ │ + poweredBy = cache.poweredBy; │ │ │ │ │ + // increment the layer count │ │ │ │ │ + ++cache.count; │ │ │ │ │ + } else { │ │ │ │ │ + // this is the first Google layer for this map │ │ │ │ │ │ │ │ │ │ - node.setAttributeNS(null, "width", size); │ │ │ │ │ - node.setAttributeNS(null, "height", size); │ │ │ │ │ - node.setAttributeNS(null, "x", pos.x - offset); │ │ │ │ │ - node.setAttributeNS(null, "y", pos.y - offset); │ │ │ │ │ + var container = this.map.viewPortDiv; │ │ │ │ │ + var div = document.createElement("div"); │ │ │ │ │ + div.id = this.map.id + "_GMap2Container"; │ │ │ │ │ + div.style.position = "absolute"; │ │ │ │ │ + div.style.width = "100%"; │ │ │ │ │ + div.style.height = "100%"; │ │ │ │ │ + container.appendChild(div); │ │ │ │ │ │ │ │ │ │ - // now that the node has all its new properties, insert it │ │ │ │ │ - // back into the dom where it was │ │ │ │ │ - if (nextSibling) { │ │ │ │ │ - parent.insertBefore(node, nextSibling); │ │ │ │ │ - } else if (parent) { │ │ │ │ │ - parent.appendChild(node); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "r", style.pointRadius); │ │ │ │ │ - } │ │ │ │ │ + // create GMap and shuffle elements │ │ │ │ │ + try { │ │ │ │ │ + mapObject = new GMap2(div); │ │ │ │ │ │ │ │ │ │ - var rotation = style.rotation; │ │ │ │ │ + // move the ToS and branding stuff up to the container div │ │ │ │ │ + termsOfUse = div.lastChild; │ │ │ │ │ + container.appendChild(termsOfUse); │ │ │ │ │ + termsOfUse.style.zIndex = "1100"; │ │ │ │ │ + termsOfUse.style.right = ""; │ │ │ │ │ + termsOfUse.style.bottom = ""; │ │ │ │ │ + termsOfUse.className = "olLayerGoogleCopyright"; │ │ │ │ │ │ │ │ │ │ - if ((rotation !== undefined || node._rotation !== undefined) && pos) { │ │ │ │ │ - node._rotation = rotation; │ │ │ │ │ - rotation |= 0; │ │ │ │ │ - if (node.nodeName !== "svg") { │ │ │ │ │ - node.setAttributeNS(null, "transform", │ │ │ │ │ - "rotate(" + rotation + " " + pos.x + " " + │ │ │ │ │ - pos.y + ")"); │ │ │ │ │ - } else { │ │ │ │ │ - var metrics = this.symbolMetrics[src.id]; │ │ │ │ │ - node.firstChild.setAttributeNS(null, "transform", "rotate(" + │ │ │ │ │ - rotation + " " + │ │ │ │ │ - metrics[1] + " " + │ │ │ │ │ - metrics[2] + ")"); │ │ │ │ │ - } │ │ │ │ │ + poweredBy = div.lastChild; │ │ │ │ │ + container.appendChild(poweredBy); │ │ │ │ │ + poweredBy.style.zIndex = "1100"; │ │ │ │ │ + poweredBy.style.right = ""; │ │ │ │ │ + poweredBy.style.bottom = ""; │ │ │ │ │ + poweredBy.className = "olLayerGooglePoweredBy gmnoprint"; │ │ │ │ │ + │ │ │ │ │ + } catch (e) { │ │ │ │ │ + throw (e); │ │ │ │ │ } │ │ │ │ │ + // cache elements for use by any other google layers added to │ │ │ │ │ + // this same map │ │ │ │ │ + OpenLayers.Layer.Google.cache[this.map.id] = { │ │ │ │ │ + mapObject: mapObject, │ │ │ │ │ + termsOfUse: termsOfUse, │ │ │ │ │ + poweredBy: poweredBy, │ │ │ │ │ + count: 1 │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - if (options.isFilled) { │ │ │ │ │ - node.setAttributeNS(null, "fill", style.fillColor); │ │ │ │ │ - node.setAttributeNS(null, "fill-opacity", style.fillOpacity); │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "fill", "none"); │ │ │ │ │ - } │ │ │ │ │ + this.mapObject = mapObject; │ │ │ │ │ + this.termsOfUse = termsOfUse; │ │ │ │ │ + this.poweredBy = poweredBy; │ │ │ │ │ │ │ │ │ │ - if (options.isStroked) { │ │ │ │ │ - node.setAttributeNS(null, "stroke", style.strokeColor); │ │ │ │ │ - node.setAttributeNS(null, "stroke-opacity", style.strokeOpacity); │ │ │ │ │ - node.setAttributeNS(null, "stroke-width", style.strokeWidth * widthFactor); │ │ │ │ │ - node.setAttributeNS(null, "stroke-linecap", style.strokeLinecap || "round"); │ │ │ │ │ - // Hard-coded linejoin for now, to make it look the same as in VML. │ │ │ │ │ - // There is no strokeLinejoin property yet for symbolizers. │ │ │ │ │ - node.setAttributeNS(null, "stroke-linejoin", "round"); │ │ │ │ │ - style.strokeDashstyle && node.setAttributeNS(null, │ │ │ │ │ - "stroke-dasharray", this.dashStyle(style, widthFactor)); │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "stroke", "none"); │ │ │ │ │ + // ensure this layer type is one of the mapObject types │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.mapObject.getMapTypes(), │ │ │ │ │ + this.type) === -1) { │ │ │ │ │ + this.mapObject.addMapType(this.type); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - if (style.pointerEvents) { │ │ │ │ │ - node.setAttributeNS(null, "pointer-events", style.pointerEvents); │ │ │ │ │ + //since v 2.93 getDragObject is now available. │ │ │ │ │ + if (typeof mapObject.getDragObject == "function") { │ │ │ │ │ + this.dragObject = mapObject.getDragObject(); │ │ │ │ │ + } else { │ │ │ │ │ + this.dragPanMapObject = null; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - if (style.cursor != null) { │ │ │ │ │ - node.setAttributeNS(null, "cursor", style.cursor); │ │ │ │ │ + if (this.isBaseLayer === false) { │ │ │ │ │ + this.setGMapVisibility(this.div.style.display !== "none"); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: dashStyle │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * widthFactor - {Number} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A SVG compliant 'stroke-dasharray' value │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: onMapResize │ │ │ │ │ */ │ │ │ │ │ - dashStyle: function(style, widthFactor) { │ │ │ │ │ - var w = style.strokeWidth * widthFactor; │ │ │ │ │ - var str = style.strokeDashstyle; │ │ │ │ │ - switch (str) { │ │ │ │ │ - case 'solid': │ │ │ │ │ - return 'none'; │ │ │ │ │ - case 'dot': │ │ │ │ │ - return [1, 4 * w].join(); │ │ │ │ │ - case 'dash': │ │ │ │ │ - return [4 * w, 4 * w].join(); │ │ │ │ │ - case 'dashdot': │ │ │ │ │ - return [4 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ - case 'longdash': │ │ │ │ │ - return [8 * w, 4 * w].join(); │ │ │ │ │ - case 'longdashdot': │ │ │ │ │ - return [8 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ - default: │ │ │ │ │ - return OpenLayers.String.trim(str).replace(/\s+/g, ","); │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + // workaround for resizing of invisible or not yet fully loaded layers │ │ │ │ │ + // where GMap2.checkResize() does not work. We need to load the GMap │ │ │ │ │ + // for the old div size, then checkResize(), and then call │ │ │ │ │ + // layer.moveTo() to trigger GMap.setCenter() (which will finish │ │ │ │ │ + // the GMap initialization). │ │ │ │ │ + if (this.visibility && this.mapObject.isLoaded()) { │ │ │ │ │ + this.mapObject.checkResize(); │ │ │ │ │ + } else { │ │ │ │ │ + if (!this._resized) { │ │ │ │ │ + var layer = this; │ │ │ │ │ + var handle = GEvent.addListener(this.mapObject, "load", function() { │ │ │ │ │ + GEvent.removeListener(handle); │ │ │ │ │ + delete layer._resized; │ │ │ │ │ + layer.mapObject.checkResize(); │ │ │ │ │ + layer.moveTo(layer.map.getCenter(), layer.map.getZoom()); │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + this._resized = true; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createNode │ │ │ │ │ + /** │ │ │ │ │ + * Method: setGMapVisibility │ │ │ │ │ + * Display the GMap container and associated elements. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * type - {String} Kind of node to draw │ │ │ │ │ - * id - {String} Id for node │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A new node of the given type and id │ │ │ │ │ + * visible - {Boolean} Display the GMap elements. │ │ │ │ │ */ │ │ │ │ │ - createNode: function(type, id) { │ │ │ │ │ - var node = document.createElementNS(this.xmlns, type); │ │ │ │ │ - if (id) { │ │ │ │ │ - node.setAttributeNS(null, "id", id); │ │ │ │ │ + setGMapVisibility: function(visible) { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + var container = this.mapObject.getContainer(); │ │ │ │ │ + if (visible === true) { │ │ │ │ │ + this.mapObject.setMapType(this.type); │ │ │ │ │ + container.style.display = ""; │ │ │ │ │ + this.termsOfUse.style.left = ""; │ │ │ │ │ + this.termsOfUse.style.display = ""; │ │ │ │ │ + this.poweredBy.style.display = ""; │ │ │ │ │ + cache.displayed = this.id; │ │ │ │ │ + } else { │ │ │ │ │ + if (cache.displayed === this.id) { │ │ │ │ │ + delete cache.displayed; │ │ │ │ │ + } │ │ │ │ │ + if (!cache.displayed) { │ │ │ │ │ + container.style.display = "none"; │ │ │ │ │ + this.termsOfUse.style.display = "none"; │ │ │ │ │ + // move ToU far to the left in addition to setting display │ │ │ │ │ + // to "none", because at the end of the GMap2 load │ │ │ │ │ + // sequence, display: none will be unset and ToU would be │ │ │ │ │ + // visible after loading a map with a google layer that is │ │ │ │ │ + // initially hidden. │ │ │ │ │ + this.termsOfUse.style.left = "-9999px"; │ │ │ │ │ + this.poweredBy.style.display = "none"; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: nodeTypeCompare │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {SVGDomElement} An SVG element │ │ │ │ │ - * type - {String} Kind of node │ │ │ │ │ + /** │ │ │ │ │ + * Method: getMapContainer │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ + * {DOMElement} the GMap container's div │ │ │ │ │ */ │ │ │ │ │ - nodeTypeCompare: function(node, type) { │ │ │ │ │ - return (type == node.nodeName); │ │ │ │ │ + getMapContainer: function() { │ │ │ │ │ + return this.mapObject.getContainer(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + // │ │ │ │ │ + // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ + // │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: createRenderRoot │ │ │ │ │ + * APIMethod: getMapObjectBoundsFromOLBounds │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * olBounds - {<OpenLayers.Bounds>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} The specific render engine's root element │ │ │ │ │ + * {Object} A MapObject Bounds, translated from olBounds │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ */ │ │ │ │ │ - createRenderRoot: function() { │ │ │ │ │ - var svg = this.nodeFactory(this.container.id + "_svgRoot", "svg"); │ │ │ │ │ - svg.style.display = "block"; │ │ │ │ │ - return svg; │ │ │ │ │ + getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ + var moBounds = null; │ │ │ │ │ + if (olBounds != null) { │ │ │ │ │ + var sw = this.sphericalMercator ? │ │ │ │ │ + this.inverseMercator(olBounds.bottom, olBounds.left) : │ │ │ │ │ + new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ + var ne = this.sphericalMercator ? │ │ │ │ │ + this.inverseMercator(olBounds.top, olBounds.right) : │ │ │ │ │ + new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ + moBounds = new GLatLngBounds(new GLatLng(sw.lat, sw.lon), │ │ │ │ │ + new GLatLng(ne.lat, ne.lon)); │ │ │ │ │ + } │ │ │ │ │ + return moBounds; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createRoot │ │ │ │ │ + │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Interface Controls * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + // Get&Set Center, Zoom │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setMapObjectCenter │ │ │ │ │ + * Set the mapObject to the specified center and zoom │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * suffix - {String} suffix to append to the id │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * center - {Object} MapObject LonLat format │ │ │ │ │ + * zoom - {int} MapObject zoom format │ │ │ │ │ */ │ │ │ │ │ - createRoot: function(suffix) { │ │ │ │ │ - return this.nodeFactory(this.container.id + suffix, "g"); │ │ │ │ │ + setMapObjectCenter: function(center, zoom) { │ │ │ │ │ + this.mapObject.setCenter(center, zoom); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createDefs │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The element to which we'll add the symbol definitions │ │ │ │ │ + * APIMethod: dragPanMapObject │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * dX - {Integer} │ │ │ │ │ + * dY - {Integer} │ │ │ │ │ */ │ │ │ │ │ - createDefs: function() { │ │ │ │ │ - var defs = this.nodeFactory(this.container.id + "_defs", "defs"); │ │ │ │ │ - this.rendererRoot.appendChild(defs); │ │ │ │ │ - return defs; │ │ │ │ │ + dragPanMapObject: function(dX, dY) { │ │ │ │ │ + this.dragObject.moveBy(new GSize(-dX, dY)); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /************************************** │ │ │ │ │ - * * │ │ │ │ │ - * GEOMETRY DRAWING FUNCTIONS * │ │ │ │ │ - * * │ │ │ │ │ - **************************************/ │ │ │ │ │ + │ │ │ │ │ + // LonLat - Pixel Translation │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPoint │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ + * APIMethod: getMapObjectLonLatFromMapObjectPixel │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moPixel - {Object} MapObject Pixel format │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the point │ │ │ │ │ + * {Object} MapObject LonLat translated from MapObject Pixel │ │ │ │ │ */ │ │ │ │ │ - drawPoint: function(node, geometry) { │ │ │ │ │ - return this.drawCircle(node, geometry, 1); │ │ │ │ │ + getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return this.mapObject.fromContainerPixelToLatLng(moPixel); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawCircle │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ + * APIMethod: getMapObjectPixelFromMapObjectLonLat │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * radius - {Float} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the circle │ │ │ │ │ + * {Object} MapObject Pixel transtlated from MapObject LonLat │ │ │ │ │ */ │ │ │ │ │ - drawCircle: function(node, geometry, radius) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = ((geometry.x - this.featureDx) / resolution + this.left); │ │ │ │ │ - var y = (this.top - geometry.y / resolution); │ │ │ │ │ + getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + return this.mapObject.fromLatLngToContainerPixel(moLonLat); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - node.setAttributeNS(null, "cx", x); │ │ │ │ │ - node.setAttributeNS(null, "cy", y); │ │ │ │ │ - node.setAttributeNS(null, "r", radius); │ │ │ │ │ - return node; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - }, │ │ │ │ │ + // Bounds │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawLineString │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectZoomFromMapObjectBounds │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moBounds - {Object} MapObject Bounds format │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components of │ │ │ │ │ - * the linestring, or false if nothing could be drawn │ │ │ │ │ + * {Object} MapObject Zoom for specified MapObject Bounds │ │ │ │ │ */ │ │ │ │ │ - drawLineString: function(node, geometry) { │ │ │ │ │ - var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ - if (componentsResult.path) { │ │ │ │ │ - node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ - return (componentsResult.complete ? node : null); │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ + getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + return this.mapObject.getBoundsZoomLevel(moBounds); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Primitives * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + // LonLat │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLinearRing │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ + * APIMethod: getMapObjectLonLatFromLonLat │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * lon - {Float} │ │ │ │ │ + * lat - {Float} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ - * of the linear ring, or false if nothing could be drawn │ │ │ │ │ + * {Object} MapObject LonLat built from lon and lat params │ │ │ │ │ */ │ │ │ │ │ - drawLinearRing: function(node, geometry) { │ │ │ │ │ - var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ - if (componentsResult.path) { │ │ │ │ │ - node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ - return (componentsResult.complete ? node : null); │ │ │ │ │ + getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ + var gLatLng; │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ + gLatLng = new GLatLng(lonlat.lat, lonlat.lon); │ │ │ │ │ } else { │ │ │ │ │ - return false; │ │ │ │ │ + gLatLng = new GLatLng(lat, lon); │ │ │ │ │ } │ │ │ │ │ + return gLatLng; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + // Pixel │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPolygon │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ + * APIMethod: getMapObjectPixelFromXY │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * x - {Integer} │ │ │ │ │ + * y - {Integer} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ - * of the polygon, or false if nothing could be drawn │ │ │ │ │ + * {Object} MapObject Pixel from x and y parameters │ │ │ │ │ */ │ │ │ │ │ - drawPolygon: function(node, geometry) { │ │ │ │ │ - var d = ""; │ │ │ │ │ - var draw = true; │ │ │ │ │ - var complete = true; │ │ │ │ │ - var linearRingResult, path; │ │ │ │ │ - for (var j = 0, len = geometry.components.length; j < len; j++) { │ │ │ │ │ - d += " M"; │ │ │ │ │ - linearRingResult = this.getComponentsString( │ │ │ │ │ - geometry.components[j].components, " "); │ │ │ │ │ - path = linearRingResult.path; │ │ │ │ │ - if (path) { │ │ │ │ │ - d += " " + path; │ │ │ │ │ - complete = linearRingResult.complete && complete; │ │ │ │ │ - } else { │ │ │ │ │ - draw = false; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - d += " z"; │ │ │ │ │ - if (draw) { │ │ │ │ │ - node.setAttributeNS(null, "d", d); │ │ │ │ │ - node.setAttributeNS(null, "fill-rule", "evenodd"); │ │ │ │ │ - return complete ? node : null; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ + return new GPoint(x, y); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/Google/v3.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/Google.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Layer.Google.v3 │ │ │ │ │ + * │ │ │ │ │ + * Mixin providing functionality specific to the Google Maps API v3. │ │ │ │ │ + * │ │ │ │ │ + * To use this layer, you must include the GMaps v3 API in your html. │ │ │ │ │ + * │ │ │ │ │ + * Note that this layer configures the google.maps.map object with the │ │ │ │ │ + * "disableDefaultUI" option set to true. Using UI controls that the Google │ │ │ │ │ + * Maps API provides is not supported by the OpenLayers API. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Google.v3 = { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawRectangle │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * Constant: DEFAULTS │ │ │ │ │ + * {Object} It is not recommended to change the properties set here. Note │ │ │ │ │ + * that Google.v3 layers only work when sphericalMercator is set to true. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the rectangle │ │ │ │ │ + * (code) │ │ │ │ │ + * { │ │ │ │ │ + * sphericalMercator: true, │ │ │ │ │ + * projection: "EPSG:900913" │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ */ │ │ │ │ │ - drawRectangle: function(node, geometry) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = ((geometry.x - this.featureDx) / resolution + this.left); │ │ │ │ │ - var y = (this.top - geometry.y / resolution); │ │ │ │ │ - │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - node.setAttributeNS(null, "x", x); │ │ │ │ │ - node.setAttributeNS(null, "y", y); │ │ │ │ │ - node.setAttributeNS(null, "width", geometry.width / resolution); │ │ │ │ │ - node.setAttributeNS(null, "height", geometry.height / resolution); │ │ │ │ │ - return node; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ + DEFAULTS: { │ │ │ │ │ + sphericalMercator: true, │ │ │ │ │ + projection: "EPSG:900913" │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawText │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * style - │ │ │ │ │ - * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * APIProperty: animationEnabled │ │ │ │ │ + * {Boolean} If set to true, the transition between zoom levels will be │ │ │ │ │ + * animated (if supported by the GMaps API for the device used). Set to │ │ │ │ │ + * false to match the zooming experience of other layer types. Default │ │ │ │ │ + * is true. Note that the GMaps API does not give us control over zoom │ │ │ │ │ + * animation, so if set to false, when zooming, this will make the │ │ │ │ │ + * layer temporarily invisible, wait until GMaps reports the map being │ │ │ │ │ + * idle, and make it visible again. The result will be a blank layer │ │ │ │ │ + * for a few moments while zooming. │ │ │ │ │ */ │ │ │ │ │ - drawText: function(featureId, style, location) { │ │ │ │ │ - var drawOutline = (!!style.labelOutlineWidth); │ │ │ │ │ - // First draw text in halo color and size and overlay the │ │ │ │ │ - // normal text afterwards │ │ │ │ │ - if (drawOutline) { │ │ │ │ │ - var outlineStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ - outlineStyle.fontColor = outlineStyle.labelOutlineColor; │ │ │ │ │ - outlineStyle.fontStrokeColor = outlineStyle.labelOutlineColor; │ │ │ │ │ - outlineStyle.fontStrokeWidth = style.labelOutlineWidth; │ │ │ │ │ - if (style.labelOutlineOpacity) { │ │ │ │ │ - outlineStyle.fontOpacity = style.labelOutlineOpacity; │ │ │ │ │ - } │ │ │ │ │ - delete outlineStyle.labelOutlineWidth; │ │ │ │ │ - this.drawText(featureId, outlineStyle, location); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - │ │ │ │ │ - var x = ((location.x - this.featureDx) / resolution + this.left); │ │ │ │ │ - var y = (location.y / resolution - this.top); │ │ │ │ │ - │ │ │ │ │ - var suffix = (drawOutline) ? this.LABEL_OUTLINE_SUFFIX : this.LABEL_ID_SUFFIX; │ │ │ │ │ - var label = this.nodeFactory(featureId + suffix, "text"); │ │ │ │ │ - │ │ │ │ │ - label.setAttributeNS(null, "x", x); │ │ │ │ │ - label.setAttributeNS(null, "y", -y); │ │ │ │ │ + animationEnabled: true, │ │ │ │ │ │ │ │ │ │ - if (style.fontColor) { │ │ │ │ │ - label.setAttributeNS(null, "fill", style.fontColor); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStrokeColor) { │ │ │ │ │ - label.setAttributeNS(null, "stroke", style.fontStrokeColor); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStrokeWidth) { │ │ │ │ │ - label.setAttributeNS(null, "stroke-width", style.fontStrokeWidth); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontOpacity) { │ │ │ │ │ - label.setAttributeNS(null, "opacity", style.fontOpacity); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontFamily) { │ │ │ │ │ - label.setAttributeNS(null, "font-family", style.fontFamily); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontSize) { │ │ │ │ │ - label.setAttributeNS(null, "font-size", style.fontSize); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontWeight) { │ │ │ │ │ - label.setAttributeNS(null, "font-weight", style.fontWeight); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStyle) { │ │ │ │ │ - label.setAttributeNS(null, "font-style", style.fontStyle); │ │ │ │ │ + /** │ │ │ │ │ + * Method: loadMapObject │ │ │ │ │ + * Load the GMap and register appropriate event listeners. │ │ │ │ │ + */ │ │ │ │ │ + loadMapObject: function() { │ │ │ │ │ + if (!this.type) { │ │ │ │ │ + this.type = google.maps.MapTypeId.ROADMAP; │ │ │ │ │ } │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - label.setAttributeNS(null, "pointer-events", "visible"); │ │ │ │ │ - label._featureId = featureId; │ │ │ │ │ + var mapObject; │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + // there are already Google layers added to this map │ │ │ │ │ + mapObject = cache.mapObject; │ │ │ │ │ + // increment the layer count │ │ │ │ │ + ++cache.count; │ │ │ │ │ } else { │ │ │ │ │ - label.setAttributeNS(null, "pointer-events", "none"); │ │ │ │ │ - } │ │ │ │ │ - var align = style.labelAlign || OpenLayers.Renderer.defaultSymbolizer.labelAlign; │ │ │ │ │ - label.setAttributeNS(null, "text-anchor", │ │ │ │ │ - OpenLayers.Renderer.SVG.LABEL_ALIGN[align[0]] || "middle"); │ │ │ │ │ - │ │ │ │ │ - if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ - label.setAttributeNS(null, "dominant-baseline", │ │ │ │ │ - OpenLayers.Renderer.SVG.LABEL_ALIGN[align[1]] || "central"); │ │ │ │ │ - } │ │ │ │ │ + // this is the first Google layer for this map │ │ │ │ │ + // create GMap │ │ │ │ │ + var center = this.map.getCenter(); │ │ │ │ │ + var container = document.createElement('div'); │ │ │ │ │ + container.className = "olForeignContainer"; │ │ │ │ │ + container.style.width = '100%'; │ │ │ │ │ + container.style.height = '100%'; │ │ │ │ │ + mapObject = new google.maps.Map(container, { │ │ │ │ │ + center: center ? │ │ │ │ │ + new google.maps.LatLng(center.lat, center.lon) : new google.maps.LatLng(0, 0), │ │ │ │ │ + zoom: this.map.getZoom() || 0, │ │ │ │ │ + mapTypeId: this.type, │ │ │ │ │ + disableDefaultUI: true, │ │ │ │ │ + keyboardShortcuts: false, │ │ │ │ │ + draggable: false, │ │ │ │ │ + disableDoubleClickZoom: true, │ │ │ │ │ + scrollwheel: false, │ │ │ │ │ + streetViewControl: false │ │ │ │ │ + }); │ │ │ │ │ + var googleControl = document.createElement('div'); │ │ │ │ │ + googleControl.style.width = '100%'; │ │ │ │ │ + googleControl.style.height = '100%'; │ │ │ │ │ + mapObject.controls[google.maps.ControlPosition.TOP_LEFT].push(googleControl); │ │ │ │ │ │ │ │ │ │ - var labelRows = style.label.split('\n'); │ │ │ │ │ - var numRows = labelRows.length; │ │ │ │ │ - while (label.childNodes.length > numRows) { │ │ │ │ │ - label.removeChild(label.lastChild); │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - var tspan = this.nodeFactory(featureId + suffix + "_tspan_" + i, "tspan"); │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - tspan._featureId = featureId; │ │ │ │ │ - tspan._geometry = location; │ │ │ │ │ - tspan._geometryClass = location.CLASS_NAME; │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.IS_GECKO === false) { │ │ │ │ │ - tspan.setAttributeNS(null, "baseline-shift", │ │ │ │ │ - OpenLayers.Renderer.SVG.LABEL_VSHIFT[align[1]] || "-35%"); │ │ │ │ │ - } │ │ │ │ │ - tspan.setAttribute("x", x); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - var vfactor = OpenLayers.Renderer.SVG.LABEL_VFACTOR[align[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5; │ │ │ │ │ - } │ │ │ │ │ - tspan.setAttribute("dy", (vfactor * (numRows - 1)) + "em"); │ │ │ │ │ - } else { │ │ │ │ │ - tspan.setAttribute("dy", "1em"); │ │ │ │ │ - } │ │ │ │ │ - tspan.textContent = (labelRows[i] === '') ? ' ' : labelRows[i]; │ │ │ │ │ - if (!tspan.parentNode) { │ │ │ │ │ - label.appendChild(tspan); │ │ │ │ │ - } │ │ │ │ │ + // cache elements for use by any other google layers added to │ │ │ │ │ + // this same map │ │ │ │ │ + cache = { │ │ │ │ │ + googleControl: googleControl, │ │ │ │ │ + mapObject: mapObject, │ │ │ │ │ + count: 1 │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Layer.Google.cache[this.map.id] = cache; │ │ │ │ │ } │ │ │ │ │ + this.mapObject = mapObject; │ │ │ │ │ + this.setGMapVisibility(this.visibility); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (!label.parentNode) { │ │ │ │ │ - this.textRoot.appendChild(label); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: onMapResize │ │ │ │ │ + */ │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + if (this.visibility) { │ │ │ │ │ + google.maps.event.trigger(this.mapObject, "resize"); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getComponentString │ │ │ │ │ + /** │ │ │ │ │ + * Method: setGMapVisibility │ │ │ │ │ + * Display the GMap container and associated elements. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * components - {Array(<OpenLayers.Geometry.Point>)} Array of points │ │ │ │ │ - * separator - {String} character between coordinate pairs. Defaults to "," │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} hash with properties "path" (the string created from the │ │ │ │ │ - * components and "complete" (false if the renderer was unable to │ │ │ │ │ - * draw all components) │ │ │ │ │ + * visible - {Boolean} Display the GMap elements. │ │ │ │ │ */ │ │ │ │ │ - getComponentsString: function(components, separator) { │ │ │ │ │ - var renderCmp = []; │ │ │ │ │ - var complete = true; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - var strings = []; │ │ │ │ │ - var str, component; │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - component = components[i]; │ │ │ │ │ - renderCmp.push(component); │ │ │ │ │ - str = this.getShortString(component); │ │ │ │ │ - if (str) { │ │ │ │ │ - strings.push(str); │ │ │ │ │ - } else { │ │ │ │ │ - // The current component is outside the valid range. Let's │ │ │ │ │ - // see if the previous or next component is inside the range. │ │ │ │ │ - // If so, add the coordinate of the intersection with the │ │ │ │ │ - // valid range bounds. │ │ │ │ │ - if (i > 0) { │ │ │ │ │ - if (this.getShortString(components[i - 1])) { │ │ │ │ │ - strings.push(this.clipLine(components[i], │ │ │ │ │ - components[i - 1])); │ │ │ │ │ - } │ │ │ │ │ + setGMapVisibility: function(visible) { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + var map = this.map; │ │ │ │ │ + if (cache) { │ │ │ │ │ + var type = this.type; │ │ │ │ │ + var layers = map.layers; │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = layers[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Google && │ │ │ │ │ + layer.visibility === true && layer.inRange === true) { │ │ │ │ │ + type = layer.type; │ │ │ │ │ + visible = true; │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ - if (i < len - 1) { │ │ │ │ │ - if (this.getShortString(components[i + 1])) { │ │ │ │ │ - strings.push(this.clipLine(components[i], │ │ │ │ │ - components[i + 1])); │ │ │ │ │ + } │ │ │ │ │ + var container = this.mapObject.getDiv(); │ │ │ │ │ + if (visible === true) { │ │ │ │ │ + if (container.parentNode !== map.div) { │ │ │ │ │ + if (!cache.rendered) { │ │ │ │ │ + var me = this; │ │ │ │ │ + google.maps.event.addListenerOnce(this.mapObject, 'tilesloaded', function() { │ │ │ │ │ + cache.rendered = true; │ │ │ │ │ + me.setGMapVisibility(me.getVisibility()); │ │ │ │ │ + me.moveTo(me.map.getCenter()); │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + map.div.appendChild(container); │ │ │ │ │ + cache.googleControl.appendChild(map.viewPortDiv); │ │ │ │ │ + google.maps.event.trigger(this.mapObject, 'resize'); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - complete = false; │ │ │ │ │ + this.mapObject.setMapTypeId(type); │ │ │ │ │ + } else if (cache.googleControl.hasChildNodes()) { │ │ │ │ │ + map.div.appendChild(map.viewPortDiv); │ │ │ │ │ + map.div.removeChild(container); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return { │ │ │ │ │ - path: strings.join(separator || ","), │ │ │ │ │ - complete: complete │ │ │ │ │ - }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clipLine │ │ │ │ │ - * Given two points (one inside the valid range, and one outside), │ │ │ │ │ - * clips the line betweeen the two points so that the new points are both │ │ │ │ │ - * inside the valid range. │ │ │ │ │ + * Method: getMapContainer │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * badComponent - {<OpenLayers.Geometry.Point>} original geometry of the │ │ │ │ │ - * invalid point │ │ │ │ │ - * goodComponent - {<OpenLayers.Geometry.Point>} original geometry of the │ │ │ │ │ - * valid point │ │ │ │ │ - * Returns │ │ │ │ │ - * {String} the SVG coordinate pair of the clipped point (like │ │ │ │ │ - * getShortString), or an empty string if both passed componets are at │ │ │ │ │ - * the same point. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} the GMap container's div │ │ │ │ │ */ │ │ │ │ │ - clipLine: function(badComponent, goodComponent) { │ │ │ │ │ - if (goodComponent.equals(badComponent)) { │ │ │ │ │ - return ""; │ │ │ │ │ - } │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var maxX = this.MAX_PIXEL - this.translationParameters.x; │ │ │ │ │ - var maxY = this.MAX_PIXEL - this.translationParameters.y; │ │ │ │ │ - var x1 = (goodComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y1 = this.top - goodComponent.y / resolution; │ │ │ │ │ - var x2 = (badComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y2 = this.top - badComponent.y / resolution; │ │ │ │ │ - var k; │ │ │ │ │ - if (x2 < -maxX || x2 > maxX) { │ │ │ │ │ - k = (y2 - y1) / (x2 - x1); │ │ │ │ │ - x2 = x2 < 0 ? -maxX : maxX; │ │ │ │ │ - y2 = y1 + (x2 - x1) * k; │ │ │ │ │ - } │ │ │ │ │ - if (y2 < -maxY || y2 > maxY) { │ │ │ │ │ - k = (x2 - x1) / (y2 - y1); │ │ │ │ │ - y2 = y2 < 0 ? -maxY : maxY; │ │ │ │ │ - x2 = x1 + (y2 - y1) * k; │ │ │ │ │ - } │ │ │ │ │ - return x2 + "," + y2; │ │ │ │ │ + getMapContainer: function() { │ │ │ │ │ + return this.mapObject.getDiv(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getShortString │ │ │ │ │ + // │ │ │ │ │ + // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ + // │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectBoundsFromOLBounds │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * olBounds - {<OpenLayers.Bounds>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} or false if point is outside the valid range │ │ │ │ │ + * {Object} A MapObject Bounds, translated from olBounds │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ */ │ │ │ │ │ - getShortString: function(point) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = ((point.x - this.featureDx) / resolution + this.left); │ │ │ │ │ - var y = (this.top - point.y / resolution); │ │ │ │ │ - │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - return x + "," + y; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ + getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ + var moBounds = null; │ │ │ │ │ + if (olBounds != null) { │ │ │ │ │ + var sw = this.sphericalMercator ? │ │ │ │ │ + this.inverseMercator(olBounds.bottom, olBounds.left) : │ │ │ │ │ + new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ + var ne = this.sphericalMercator ? │ │ │ │ │ + this.inverseMercator(olBounds.top, olBounds.right) : │ │ │ │ │ + new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ + moBounds = new google.maps.LatLngBounds( │ │ │ │ │ + new google.maps.LatLng(sw.lat, sw.lon), │ │ │ │ │ + new google.maps.LatLng(ne.lat, ne.lon) │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ + return moBounds; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Interface Controls * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + // LonLat - Pixel Translation │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: getPosition │ │ │ │ │ - * Finds the position of an svg node. │ │ │ │ │ + * APIMethod: getMapObjectLonLatFromMapObjectPixel │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ + * moPixel - {Object} MapObject Pixel format │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} hash with x and y properties, representing the coordinates │ │ │ │ │ - * within the svg coordinate system │ │ │ │ │ + * {Object} MapObject LonLat translated from MapObject Pixel │ │ │ │ │ */ │ │ │ │ │ - getPosition: function(node) { │ │ │ │ │ - return ({ │ │ │ │ │ - x: parseFloat(node.getAttributeNS(null, "cx")), │ │ │ │ │ - y: parseFloat(node.getAttributeNS(null, "cy")) │ │ │ │ │ - }); │ │ │ │ │ + getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + var size = this.map.getSize(); │ │ │ │ │ + var lon = this.getLongitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ + var lat = this.getLatitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ + var res = this.map.getResolution(); │ │ │ │ │ + │ │ │ │ │ + var delta_x = moPixel.x - (size.w / 2); │ │ │ │ │ + var delta_y = moPixel.y - (size.h / 2); │ │ │ │ │ + │ │ │ │ │ + var lonlat = new OpenLayers.LonLat( │ │ │ │ │ + lon + delta_x * res, │ │ │ │ │ + lat - delta_y * res │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + lonlat = lonlat.wrapDateLine(this.maxExtent); │ │ │ │ │ + } │ │ │ │ │ + return this.getMapObjectLonLatFromLonLat(lonlat.lon, lonlat.lat); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: importSymbol │ │ │ │ │ - * add a new symbol definition from the rendererer's symbol hash │ │ │ │ │ + * APIMethod: getMapObjectPixelFromMapObjectLonLat │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * graphicName - {String} name of the symbol to import │ │ │ │ │ + * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} - the imported symbol │ │ │ │ │ + * {Object} MapObject Pixel transtlated from MapObject LonLat │ │ │ │ │ */ │ │ │ │ │ - importSymbol: function(graphicName) { │ │ │ │ │ - if (!this.defs) { │ │ │ │ │ - // create svg defs tag │ │ │ │ │ - this.defs = this.createDefs(); │ │ │ │ │ - } │ │ │ │ │ - var id = this.container.id + "-" + graphicName; │ │ │ │ │ + getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + var res = this.map.getResolution(); │ │ │ │ │ + var extent = this.map.getExtent(); │ │ │ │ │ + return this.getMapObjectPixelFromXY((1 / res * (lon - extent.left)), │ │ │ │ │ + (1 / res * (extent.top - lat))); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // check if symbol already exists in the defs │ │ │ │ │ - var existing = document.getElementById(id); │ │ │ │ │ - if (existing != null) { │ │ │ │ │ - return existing; │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(graphicName + ' is not a valid symbol name'); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setMapObjectCenter │ │ │ │ │ + * Set the mapObject to the specified center and zoom │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * center - {Object} MapObject LonLat format │ │ │ │ │ + * zoom - {int} MapObject zoom format │ │ │ │ │ + */ │ │ │ │ │ + setMapObjectCenter: function(center, zoom) { │ │ │ │ │ + if (this.animationEnabled === false && zoom != this.mapObject.zoom) { │ │ │ │ │ + var mapContainer = this.getMapContainer(); │ │ │ │ │ + google.maps.event.addListenerOnce( │ │ │ │ │ + this.mapObject, │ │ │ │ │ + "idle", │ │ │ │ │ + function() { │ │ │ │ │ + mapContainer.style.visibility = ""; │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + mapContainer.style.visibility = "hidden"; │ │ │ │ │ } │ │ │ │ │ + this.mapObject.setOptions({ │ │ │ │ │ + center: center, │ │ │ │ │ + zoom: zoom │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var symbolNode = this.nodeFactory(id, "symbol"); │ │ │ │ │ - var node = this.nodeFactory(null, "polygon"); │ │ │ │ │ - symbolNode.appendChild(node); │ │ │ │ │ - var symbolExtent = new OpenLayers.Bounds( │ │ │ │ │ - Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ │ │ │ │ │ - var points = []; │ │ │ │ │ - var x, y; │ │ │ │ │ - for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ - symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ - symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ - symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ - points.push(x, ",", y); │ │ │ │ │ - } │ │ │ │ │ + // Bounds │ │ │ │ │ │ │ │ │ │ - node.setAttributeNS(null, "points", points.join(" ")); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectZoomFromMapObjectBounds │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moBounds - {Object} MapObject Bounds format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject Zoom for specified MapObject Bounds │ │ │ │ │ + */ │ │ │ │ │ + getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + return this.mapObject.getBoundsZoomLevel(moBounds); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var width = symbolExtent.getWidth(); │ │ │ │ │ - var height = symbolExtent.getHeight(); │ │ │ │ │ - // create a viewBox three times as large as the symbol itself, │ │ │ │ │ - // to allow for strokeWidth being displayed correctly at the corners. │ │ │ │ │ - var viewBox = [symbolExtent.left - width, │ │ │ │ │ - symbolExtent.bottom - height, width * 3, height * 3 │ │ │ │ │ - ]; │ │ │ │ │ - symbolNode.setAttributeNS(null, "viewBox", viewBox.join(" ")); │ │ │ │ │ - this.symbolMetrics[id] = [ │ │ │ │ │ - Math.max(width, height), │ │ │ │ │ - symbolExtent.getCenterLonLat().lon, │ │ │ │ │ - symbolExtent.getCenterLonLat().lat │ │ │ │ │ - ]; │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Primitives * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ │ │ │ │ │ - this.defs.appendChild(symbolNode); │ │ │ │ │ - return symbolNode; │ │ │ │ │ - }, │ │ │ │ │ + │ │ │ │ │ + // LonLat │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureIdFromEvent │ │ │ │ │ + * APIMethod: getMapObjectLonLatFromLonLat │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} An <OpenLayers.Event> object │ │ │ │ │ - * │ │ │ │ │ + * lon - {Float} │ │ │ │ │ + * lat - {Float} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A feature id or undefined. │ │ │ │ │ + * {Object} MapObject LonLat built from lon and lat params │ │ │ │ │ */ │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var featureId = OpenLayers.Renderer.Elements.prototype.getFeatureIdFromEvent.apply(this, arguments); │ │ │ │ │ - if (!featureId) { │ │ │ │ │ - var target = evt.target; │ │ │ │ │ - featureId = target.parentNode && target != this.rendererRoot ? │ │ │ │ │ - target.parentNode._featureId : undefined; │ │ │ │ │ + getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ + var gLatLng; │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ + gLatLng = new google.maps.LatLng(lonlat.lat, lonlat.lon); │ │ │ │ │ + } else { │ │ │ │ │ + gLatLng = new google.maps.LatLng(lat, lon); │ │ │ │ │ } │ │ │ │ │ - return featureId; │ │ │ │ │ + return gLatLng; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.SVG" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.SVG.LABEL_ALIGN │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_ALIGN = { │ │ │ │ │ - "l": "start", │ │ │ │ │ - "r": "end", │ │ │ │ │ - "b": "bottom", │ │ │ │ │ - "t": "hanging" │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.SVG.LABEL_VSHIFT │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_VSHIFT = { │ │ │ │ │ - // according to │ │ │ │ │ - // http://www.w3.org/Graphics/SVG/Test/20061213/htmlObjectHarness/full-text-align-02-b.html │ │ │ │ │ - // a baseline-shift of -70% shifts the text exactly from the │ │ │ │ │ - // bottom to the top of the baseline, so -35% moves the text to │ │ │ │ │ - // the center of the baseline. │ │ │ │ │ - "t": "-70%", │ │ │ │ │ - "b": "0" │ │ │ │ │ -}; │ │ │ │ │ + // Pixel │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.SVG.LABEL_VFACTOR │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_VFACTOR = { │ │ │ │ │ - "t": 0, │ │ │ │ │ - "b": -1 │ │ │ │ │ -}; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectPixelFromXY │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * x - {Integer} │ │ │ │ │ + * y - {Integer} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject Pixel from x and y parameters │ │ │ │ │ + */ │ │ │ │ │ + getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ + return new google.maps.Point(x, y); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Function: OpenLayers.Renderer.SVG.preventDefault │ │ │ │ │ - * *Deprecated*. Use <OpenLayers.Event.preventDefault> method instead. │ │ │ │ │ - * Used to prevent default events (especially opening images in a new tab on │ │ │ │ │ - * ctrl-click) from being executed for externalGraphic symbols │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.SVG.preventDefault = function(e) { │ │ │ │ │ - OpenLayers.Event.preventDefault(e); │ │ │ │ │ }; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol.js │ │ │ │ │ + OpenLayers/Popup.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Protocol │ │ │ │ │ - * Abstract vector layer protocol class. Not to be instantiated directly. Use │ │ │ │ │ - * one of the protocol subclasses instead. │ │ │ │ │ + * Class: OpenLayers.Popup │ │ │ │ │ + * A popup is a small div that can opened and closed on the map. │ │ │ │ │ + * Typically opened in response to clicking on a marker. │ │ │ │ │ + * See <OpenLayers.Marker>. Popup's don't require their own │ │ │ │ │ + * layer and are added the the map using the <OpenLayers.Map.addPopup> │ │ │ │ │ + * method. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * popup = new OpenLayers.Popup("chicken", │ │ │ │ │ + * new OpenLayers.LonLat(5,40), │ │ │ │ │ + * new OpenLayers.Size(200,200), │ │ │ │ │ + * "example popup", │ │ │ │ │ + * true); │ │ │ │ │ + * │ │ │ │ │ + * map.addPopup(popup); │ │ │ │ │ + * (end) │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol = OpenLayers.Class({ │ │ │ │ │ +OpenLayers.Popup = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: format │ │ │ │ │ - * {<OpenLayers.Format>} The format used by this protocol. │ │ │ │ │ + /** │ │ │ │ │ + * Property: events │ │ │ │ │ + * {<OpenLayers.Events>} custom event manager │ │ │ │ │ */ │ │ │ │ │ - format: null, │ │ │ │ │ + events: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: options │ │ │ │ │ - * {Object} Any options sent to the constructor. │ │ │ │ │ + /** Property: id │ │ │ │ │ + * {String} the unique identifier assigned to this popup. │ │ │ │ │ */ │ │ │ │ │ - options: null, │ │ │ │ │ + id: "", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: autoDestroy │ │ │ │ │ - * {Boolean} The creator of the protocol can set autoDestroy to false │ │ │ │ │ - * to fully control when the protocol is destroyed. Defaults to │ │ │ │ │ - * true. │ │ │ │ │ + /** │ │ │ │ │ + * Property: lonlat │ │ │ │ │ + * {<OpenLayers.LonLat>} the position of this popup on the map │ │ │ │ │ */ │ │ │ │ │ - autoDestroy: true, │ │ │ │ │ + lonlat: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultFilter │ │ │ │ │ - * {<OpenLayers.Filter>} Optional default filter to read requests │ │ │ │ │ + /** │ │ │ │ │ + * Property: div │ │ │ │ │ + * {DOMElement} the div that contains this popup. │ │ │ │ │ */ │ │ │ │ │ - defaultFilter: null, │ │ │ │ │ + div: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol │ │ │ │ │ - * Abstract class for vector protocols. Create instances of a subclass. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + /** │ │ │ │ │ + * Property: contentSize │ │ │ │ │ + * {<OpenLayers.Size>} the width and height of the content. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.options = options; │ │ │ │ │ - }, │ │ │ │ │ + contentSize: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: mergeWithDefaultFilter │ │ │ │ │ - * Merge filter passed to the read method with the default one │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} │ │ │ │ │ + /** │ │ │ │ │ + * Property: size │ │ │ │ │ + * {<OpenLayers.Size>} the width and height of the popup. │ │ │ │ │ */ │ │ │ │ │ - mergeWithDefaultFilter: function(filter) { │ │ │ │ │ - var merged; │ │ │ │ │ - if (filter && this.defaultFilter) { │ │ │ │ │ - merged = new OpenLayers.Filter.Logical({ │ │ │ │ │ - type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ - filters: [this.defaultFilter, filter] │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - merged = filter || this.defaultFilter || undefined; │ │ │ │ │ - } │ │ │ │ │ - return merged; │ │ │ │ │ - }, │ │ │ │ │ + size: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up the protocol. │ │ │ │ │ + /** │ │ │ │ │ + * Property: contentHTML │ │ │ │ │ + * {String} An HTML string for this popup to display. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.options = null; │ │ │ │ │ - this.format = null; │ │ │ │ │ - }, │ │ │ │ │ + contentHTML: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Construct a request for reading new features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, the same object will be passed to the callback function passed │ │ │ │ │ - * if one exists in the options object. │ │ │ │ │ + /** │ │ │ │ │ + * Property: backgroundColor │ │ │ │ │ + * {String} the background color used by the popup. │ │ │ │ │ */ │ │ │ │ │ - read: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.filter = this.mergeWithDefaultFilter(options.filter); │ │ │ │ │ - }, │ │ │ │ │ + backgroundColor: "", │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: opacity │ │ │ │ │ + * {float} the opacity of this popup (between 0.0 and 1.0) │ │ │ │ │ + */ │ │ │ │ │ + opacity: "", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: create │ │ │ │ │ - * Construct a request for writing newly created features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, the same object will be passed to the callback function passed │ │ │ │ │ - * if one exists in the options object. │ │ │ │ │ + /** │ │ │ │ │ + * Property: border │ │ │ │ │ + * {String} the border size of the popup. (eg 2px) │ │ │ │ │ */ │ │ │ │ │ - create: function() {}, │ │ │ │ │ + border: "", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: update │ │ │ │ │ - * Construct a request updating modified features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, the same object will be passed to the callback function passed │ │ │ │ │ - * if one exists in the options object. │ │ │ │ │ + /** │ │ │ │ │ + * Property: contentDiv │ │ │ │ │ + * {DOMElement} a reference to the element that holds the content of │ │ │ │ │ + * the div. │ │ │ │ │ */ │ │ │ │ │ - update: function() {}, │ │ │ │ │ + contentDiv: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: delete │ │ │ │ │ - * Construct a request deleting a removed feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, the same object will be passed to the callback function passed │ │ │ │ │ - * if one exists in the options object. │ │ │ │ │ + /** │ │ │ │ │ + * Property: groupDiv │ │ │ │ │ + * {DOMElement} First and only child of 'div'. The group Div contains the │ │ │ │ │ + * 'contentDiv' and the 'closeDiv'. │ │ │ │ │ */ │ │ │ │ │ - "delete": function() {}, │ │ │ │ │ + groupDiv: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: commit │ │ │ │ │ - * Go over the features and for each take action │ │ │ │ │ - * based on the feature state. Possible actions are create, │ │ │ │ │ - * update and delete. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array({<OpenLayers.Feature.Vector>})} │ │ │ │ │ - * options - {Object} Object whose possible keys are "create", "update", │ │ │ │ │ - * "delete", "callback" and "scope", the values referenced by the │ │ │ │ │ - * first three are objects as passed to the "create", "update", and │ │ │ │ │ - * "delete" methods, the value referenced by the "callback" key is │ │ │ │ │ - * a function which is called when the commit operation is complete │ │ │ │ │ - * using the scope referenced by the "scope" key. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array({<OpenLayers.Protocol.Response>})} An array of │ │ │ │ │ - * <OpenLayers.Protocol.Response> objects. │ │ │ │ │ + /** │ │ │ │ │ + * Property: closeDiv │ │ │ │ │ + * {DOMElement} the optional closer image │ │ │ │ │ */ │ │ │ │ │ - commit: function() {}, │ │ │ │ │ + closeDiv: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: abort │ │ │ │ │ - * Abort an ongoing request. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: autoSize │ │ │ │ │ + * {Boolean} Resize the popup to auto-fit the contents. │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - abort: function(response) {}, │ │ │ │ │ + autoSize: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createCallback │ │ │ │ │ - * Returns a function that applies the given public method with resp and │ │ │ │ │ - * options arguments. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * method - {Function} The method to be applied by the callback. │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The protocol response object. │ │ │ │ │ - * options - {Object} Options sent to the protocol method │ │ │ │ │ + * APIProperty: minSize │ │ │ │ │ + * {<OpenLayers.Size>} Minimum size allowed for the popup's contents. │ │ │ │ │ */ │ │ │ │ │ - createCallback: function(method, response, options) { │ │ │ │ │ - return OpenLayers.Function.bind(function() { │ │ │ │ │ - method.apply(this, [response, options]); │ │ │ │ │ - }, this); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol" │ │ │ │ │ -}); │ │ │ │ │ + minSize: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Protocol.Response │ │ │ │ │ - * Protocols return Response objects to their users. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol.Response = OpenLayers.Class({ │ │ │ │ │ /** │ │ │ │ │ - * Property: code │ │ │ │ │ - * {Number} - OpenLayers.Protocol.Response.SUCCESS or │ │ │ │ │ - * OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ + * APIProperty: maxSize │ │ │ │ │ + * {<OpenLayers.Size>} Maximum size allowed for the popup's contents. │ │ │ │ │ */ │ │ │ │ │ - code: null, │ │ │ │ │ + maxSize: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: requestType │ │ │ │ │ - * {String} The type of request this response corresponds to. Either │ │ │ │ │ - * "create", "read", "update" or "delete". │ │ │ │ │ + /** │ │ │ │ │ + * Property: displayClass │ │ │ │ │ + * {String} The CSS class of the popup. │ │ │ │ │ */ │ │ │ │ │ - requestType: null, │ │ │ │ │ + displayClass: "olPopup", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: last │ │ │ │ │ - * {Boolean} - true if this is the last response expected in a commit, │ │ │ │ │ - * false otherwise, defaults to true. │ │ │ │ │ + /** │ │ │ │ │ + * Property: contentDisplayClass │ │ │ │ │ + * {String} The CSS class of the popup content div. │ │ │ │ │ */ │ │ │ │ │ - last: true, │ │ │ │ │ + contentDisplayClass: "olPopupContent", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Array({<OpenLayers.Feature.Vector>})} or {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * The features returned in the response by the server. Depending on the │ │ │ │ │ - * protocol's read payload, either features or data will be populated. │ │ │ │ │ + /** │ │ │ │ │ + * Property: padding │ │ │ │ │ + * {int or <OpenLayers.Bounds>} An extra opportunity to specify internal │ │ │ │ │ + * padding of the content div inside the popup. This was originally │ │ │ │ │ + * confused with the css padding as specified in style.css's │ │ │ │ │ + * 'olPopupContent' class. We would like to get rid of this altogether, │ │ │ │ │ + * except that it does come in handy for the framed and anchoredbubble │ │ │ │ │ + * popups, who need to maintain yet another barrier between their │ │ │ │ │ + * content and the outer border of the popup itself. │ │ │ │ │ + * │ │ │ │ │ + * Note that in order to not break API, we must continue to support │ │ │ │ │ + * this property being set as an integer. Really, though, we'd like to │ │ │ │ │ + * have this specified as a Bounds object so that user can specify │ │ │ │ │ + * distinct left, top, right, bottom paddings. With the 3.0 release │ │ │ │ │ + * we can make this only a bounds. │ │ │ │ │ */ │ │ │ │ │ - features: null, │ │ │ │ │ + padding: 0, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: data │ │ │ │ │ - * {Object} │ │ │ │ │ - * The data returned in the response by the server. Depending on the │ │ │ │ │ - * protocol's read payload, either features or data will be populated. │ │ │ │ │ + /** │ │ │ │ │ + * Property: disableFirefoxOverflowHack │ │ │ │ │ + * {Boolean} The hack for overflow in Firefox causes all elements │ │ │ │ │ + * to be re-drawn, which causes Flash elements to be │ │ │ │ │ + * re-initialized, which is troublesome. │ │ │ │ │ + * With this property the hack can be disabled. │ │ │ │ │ */ │ │ │ │ │ - data: null, │ │ │ │ │ + disableFirefoxOverflowHack: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: reqFeatures │ │ │ │ │ - * {Array({<OpenLayers.Feature.Vector>})} or {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * The features provided by the user and placed in the request by the │ │ │ │ │ - * protocol. │ │ │ │ │ + * Method: fixPadding │ │ │ │ │ + * To be removed in 3.0, this function merely helps us to deal with the │ │ │ │ │ + * case where the user may have set an integer value for padding, │ │ │ │ │ + * instead of an <OpenLayers.Bounds> object. │ │ │ │ │ */ │ │ │ │ │ - reqFeatures: null, │ │ │ │ │ + fixPadding: function() { │ │ │ │ │ + if (typeof this.padding == "number") { │ │ │ │ │ + this.padding = new OpenLayers.Bounds( │ │ │ │ │ + this.padding, this.padding, this.padding, this.padding │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: priv │ │ │ │ │ + * APIProperty: panMapIfOutOfView │ │ │ │ │ + * {Boolean} When drawn, pan map such that the entire popup is visible in │ │ │ │ │ + * the current viewport (if necessary). │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - priv: null, │ │ │ │ │ + panMapIfOutOfView: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: error │ │ │ │ │ - * {Object} The error object in case a service exception was encountered. │ │ │ │ │ + * APIProperty: keepInMap │ │ │ │ │ + * {Boolean} If panMapIfOutOfView is false, and this property is true, │ │ │ │ │ + * contrain the popup such that it always fits in the available map │ │ │ │ │ + * space. By default, this is not set on the base class. If you are │ │ │ │ │ + * creating popups that are near map edges and not allowing pannning, │ │ │ │ │ + * and especially if you have a popup which has a │ │ │ │ │ + * fixedRelativePosition, setting this to false may be a smart thing to │ │ │ │ │ + * do. Subclasses may want to override this setting. │ │ │ │ │ + * │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - error: null, │ │ │ │ │ + keepInMap: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.Response │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * APIProperty: closeOnMove │ │ │ │ │ + * {Boolean} When map pans, close the popup. │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - }, │ │ │ │ │ + closeOnMove: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: success │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} - true on success, false otherwise │ │ │ │ │ + /** │ │ │ │ │ + * Property: map │ │ │ │ │ + * {<OpenLayers.Map>} this gets set in Map.js when the popup is added to the map │ │ │ │ │ */ │ │ │ │ │ - success: function() { │ │ │ │ │ - return this.code > 0; │ │ │ │ │ - }, │ │ │ │ │ + map: null, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.Response" │ │ │ │ │ -}); │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Popup │ │ │ │ │ + * Create a popup. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} a unqiue identifier for this popup. If null is passed │ │ │ │ │ + * an identifier will be automatically generated. │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>} The position on the map the popup will │ │ │ │ │ + * be shown. │ │ │ │ │ + * contentSize - {<OpenLayers.Size>} The size of the content. │ │ │ │ │ + * contentHTML - {String} An HTML string to display inside the │ │ │ │ │ + * popup. │ │ │ │ │ + * closeBox - {Boolean} Whether to display a close box inside │ │ │ │ │ + * the popup. │ │ │ │ │ + * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, closeBox, closeBoxCallback) { │ │ │ │ │ + if (id == null) { │ │ │ │ │ + id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -OpenLayers.Protocol.Response.SUCCESS = 1; │ │ │ │ │ -OpenLayers.Protocol.Response.FAILURE = 0; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Request.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + this.id = id; │ │ │ │ │ + this.lonlat = lonlat; │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + this.contentSize = (contentSize != null) ? contentSize : │ │ │ │ │ + new OpenLayers.Size( │ │ │ │ │ + OpenLayers.Popup.WIDTH, │ │ │ │ │ + OpenLayers.Popup.HEIGHT); │ │ │ │ │ + if (contentHTML != null) { │ │ │ │ │ + this.contentHTML = contentHTML; │ │ │ │ │ + } │ │ │ │ │ + this.backgroundColor = OpenLayers.Popup.COLOR; │ │ │ │ │ + this.opacity = OpenLayers.Popup.OPACITY; │ │ │ │ │ + this.border = OpenLayers.Popup.BORDER; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ - * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ - */ │ │ │ │ │ + this.div = OpenLayers.Util.createDiv(this.id, null, null, │ │ │ │ │ + null, null, null, "hidden"); │ │ │ │ │ + this.div.className = this.displayClass; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * TODO: deprecate me │ │ │ │ │ - * Use OpenLayers.Request.proxy instead. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.ProxyHost = ""; │ │ │ │ │ + var groupDivId = this.id + "_GroupDiv"; │ │ │ │ │ + this.groupDiv = OpenLayers.Util.createDiv(groupDivId, null, null, │ │ │ │ │ + null, "relative", null, │ │ │ │ │ + "hidden"); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Request │ │ │ │ │ - * The OpenLayers.Request namespace contains convenience methods for working │ │ │ │ │ - * with XMLHttpRequests. These methods work with a cross-browser │ │ │ │ │ - * W3C compliant <OpenLayers.Request.XMLHttpRequest> class. │ │ │ │ │ - */ │ │ │ │ │ -if (!OpenLayers.Request) { │ │ │ │ │ - /** │ │ │ │ │ - * This allows for OpenLayers/Request/XMLHttpRequest.js to be included │ │ │ │ │ - * before or after this script. │ │ │ │ │ - */ │ │ │ │ │ - OpenLayers.Request = {}; │ │ │ │ │ -} │ │ │ │ │ -OpenLayers.Util.extend(OpenLayers.Request, { │ │ │ │ │ + var id = this.div.id + "_contentDiv"; │ │ │ │ │ + this.contentDiv = OpenLayers.Util.createDiv(id, null, this.contentSize.clone(), │ │ │ │ │ + null, "relative"); │ │ │ │ │ + this.contentDiv.className = this.contentDisplayClass; │ │ │ │ │ + this.groupDiv.appendChild(this.contentDiv); │ │ │ │ │ + this.div.appendChild(this.groupDiv); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: DEFAULT_CONFIG │ │ │ │ │ - * {Object} Default configuration for all requests. │ │ │ │ │ - */ │ │ │ │ │ - DEFAULT_CONFIG: { │ │ │ │ │ - method: "GET", │ │ │ │ │ - url: window.location.href, │ │ │ │ │ - async: true, │ │ │ │ │ - user: undefined, │ │ │ │ │ - password: undefined, │ │ │ │ │ - params: null, │ │ │ │ │ - proxy: OpenLayers.ProxyHost, │ │ │ │ │ - headers: {}, │ │ │ │ │ - data: null, │ │ │ │ │ - callback: function() {}, │ │ │ │ │ - success: null, │ │ │ │ │ - failure: null, │ │ │ │ │ - scope: null │ │ │ │ │ + if (closeBox) { │ │ │ │ │ + this.addCloseBox(closeBoxCallback); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.registerEvents(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: URL_SPLIT_REGEX │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * nullify references to prevent circular references and memory leaks │ │ │ │ │ */ │ │ │ │ │ - URL_SPLIT_REGEX: /([^:]*:)\/\/([^:]*:?[^@]*@)?([^:\/\?]*):?([^\/\?]*)/, │ │ │ │ │ + destroy: function() { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} An events object that handles all │ │ │ │ │ - * events on the {<OpenLayers.Request>} object. │ │ │ │ │ - * │ │ │ │ │ - * All event listeners will receive an event object with three properties: │ │ │ │ │ - * request - {<OpenLayers.Request.XMLHttpRequest>} The request object. │ │ │ │ │ - * config - {Object} The config object sent to the specific request method. │ │ │ │ │ - * requestUrl - {String} The request url. │ │ │ │ │ - * │ │ │ │ │ - * Supported event types: │ │ │ │ │ - * complete - Triggered when we have a response from the request, if a │ │ │ │ │ - * listener returns false, no further response processing will take │ │ │ │ │ - * place. │ │ │ │ │ - * success - Triggered when the HTTP response has a success code (200-299). │ │ │ │ │ - * failure - Triggered when the HTTP response does not have a success code. │ │ │ │ │ - */ │ │ │ │ │ - events: new OpenLayers.Events(this), │ │ │ │ │ + this.id = null; │ │ │ │ │ + this.lonlat = null; │ │ │ │ │ + this.size = null; │ │ │ │ │ + this.contentHTML = null; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: makeSameOrigin │ │ │ │ │ - * Using the specified proxy, returns a same origin url of the provided url. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * url - {String} An arbitrary url │ │ │ │ │ - * proxy {String|Function} The proxy to use to make the provided url a │ │ │ │ │ - * same origin url. │ │ │ │ │ - * │ │ │ │ │ - * Returns │ │ │ │ │ - * {String} the same origin url. If no proxy is provided, the returned url │ │ │ │ │ - * will be the same as the provided url. │ │ │ │ │ - */ │ │ │ │ │ - makeSameOrigin: function(url, proxy) { │ │ │ │ │ - var sameOrigin = url.indexOf("http") !== 0; │ │ │ │ │ - var urlParts = !sameOrigin && url.match(this.URL_SPLIT_REGEX); │ │ │ │ │ - if (urlParts) { │ │ │ │ │ - var location = window.location; │ │ │ │ │ - sameOrigin = │ │ │ │ │ - urlParts[1] == location.protocol && │ │ │ │ │ - urlParts[3] == location.hostname; │ │ │ │ │ - var uPort = urlParts[4], │ │ │ │ │ - lPort = location.port; │ │ │ │ │ - if (uPort != 80 && uPort != "" || lPort != "80" && lPort != "") { │ │ │ │ │ - sameOrigin = sameOrigin && uPort == lPort; │ │ │ │ │ - } │ │ │ │ │ + this.backgroundColor = null; │ │ │ │ │ + this.opacity = null; │ │ │ │ │ + this.border = null; │ │ │ │ │ + │ │ │ │ │ + if (this.closeOnMove && this.map) { │ │ │ │ │ + this.map.events.unregister("movestart", this, this.hide); │ │ │ │ │ } │ │ │ │ │ - if (!sameOrigin) { │ │ │ │ │ - if (proxy) { │ │ │ │ │ - if (typeof proxy == "function") { │ │ │ │ │ - url = proxy(url); │ │ │ │ │ - } else { │ │ │ │ │ - url = proxy + encodeURIComponent(url); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + │ │ │ │ │ + this.events.destroy(); │ │ │ │ │ + this.events = null; │ │ │ │ │ + │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + OpenLayers.Event.stopObservingElement(this.closeDiv); │ │ │ │ │ + this.groupDiv.removeChild(this.closeDiv); │ │ │ │ │ } │ │ │ │ │ - return url; │ │ │ │ │ + this.closeDiv = null; │ │ │ │ │ + │ │ │ │ │ + this.div.removeChild(this.groupDiv); │ │ │ │ │ + this.groupDiv = null; │ │ │ │ │ + │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.removePopup(this); │ │ │ │ │ + } │ │ │ │ │ + this.map = null; │ │ │ │ │ + this.div = null; │ │ │ │ │ + │ │ │ │ │ + this.autoSize = null; │ │ │ │ │ + this.minSize = null; │ │ │ │ │ + this.maxSize = null; │ │ │ │ │ + this.padding = null; │ │ │ │ │ + this.panMapIfOutOfView = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: issue │ │ │ │ │ - * Create a new XMLHttpRequest object, open it, set any headers, bind │ │ │ │ │ - * a callback to done state, and send any data. It is recommended that │ │ │ │ │ - * you use one <GET>, <POST>, <PUT>, <DELETE>, <OPTIONS>, or <HEAD>. │ │ │ │ │ - * This method is only documented to provide detail on the configuration │ │ │ │ │ - * options available to all request methods. │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Constructs the elements that make up the popup. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * config - {Object} Object containing properties for configuring the │ │ │ │ │ - * request. Allowed configuration properties are described below. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Allowed config properties: │ │ │ │ │ - * method - {String} One of GET, POST, PUT, DELETE, HEAD, or │ │ │ │ │ - * OPTIONS. Default is GET. │ │ │ │ │ - * url - {String} URL for the request. │ │ │ │ │ - * async - {Boolean} Open an asynchronous request. Default is true. │ │ │ │ │ - * user - {String} User for relevant authentication scheme. Set │ │ │ │ │ - * to null to clear current user. │ │ │ │ │ - * password - {String} Password for relevant authentication scheme. │ │ │ │ │ - * Set to null to clear current password. │ │ │ │ │ - * proxy - {String} Optional proxy. Defaults to │ │ │ │ │ - * <OpenLayers.ProxyHost>. │ │ │ │ │ - * params - {Object} Any key:value pairs to be appended to the │ │ │ │ │ - * url as a query string. Assumes url doesn't already include a query │ │ │ │ │ - * string or hash. Typically, this is only appropriate for <GET> │ │ │ │ │ - * requests where the query string will be appended to the url. │ │ │ │ │ - * Parameter values that are arrays will be │ │ │ │ │ - * concatenated with a comma (note that this goes against form-encoding) │ │ │ │ │ - * as is done with <OpenLayers.Util.getParameterString>. │ │ │ │ │ - * headers - {Object} Object with header:value pairs to be set on │ │ │ │ │ - * the request. │ │ │ │ │ - * data - {String | Document} Optional data to send with the request. │ │ │ │ │ - * Typically, this is only used with <POST> and <PUT> requests. │ │ │ │ │ - * Make sure to provide the appropriate "Content-Type" header for your │ │ │ │ │ - * data. For <POST> and <PUT> requests, the content type defaults to │ │ │ │ │ - * "application-xml". If your data is a different content type, or │ │ │ │ │ - * if you are using a different HTTP method, set the "Content-Type" │ │ │ │ │ - * header to match your data type. │ │ │ │ │ - * callback - {Function} Function to call when request is done. │ │ │ │ │ - * To determine if the request failed, check request.status (200 │ │ │ │ │ - * indicates success). │ │ │ │ │ - * success - {Function} Optional function to call if request status is in │ │ │ │ │ - * the 200s. This will be called in addition to callback above and │ │ │ │ │ - * would typically only be used as an alternative. │ │ │ │ │ - * failure - {Function} Optional function to call if request status is not │ │ │ │ │ - * in the 200s. This will be called in addition to callback above and │ │ │ │ │ - * would typically only be used as an alternative. │ │ │ │ │ - * scope - {Object} If callback is a public method on some object, │ │ │ │ │ - * set the scope to that object. │ │ │ │ │ - * │ │ │ │ │ + * px - {<OpenLayers.Pixel>} the position the popup in pixels. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. To abort the request before a response │ │ │ │ │ - * is received, call abort() on the request object. │ │ │ │ │ + * {DOMElement} Reference to a div that contains the drawn popup │ │ │ │ │ */ │ │ │ │ │ - issue: function(config) { │ │ │ │ │ - // apply default config - proxy host may have changed │ │ │ │ │ - var defaultConfig = OpenLayers.Util.extend( │ │ │ │ │ - this.DEFAULT_CONFIG, { │ │ │ │ │ - proxy: OpenLayers.ProxyHost │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - config = config || {}; │ │ │ │ │ - config.headers = config.headers || {}; │ │ │ │ │ - config = OpenLayers.Util.applyDefaults(config, defaultConfig); │ │ │ │ │ - config.headers = OpenLayers.Util.applyDefaults(config.headers, defaultConfig.headers); │ │ │ │ │ - // Always set the "X-Requested-With" header to signal that this request │ │ │ │ │ - // was issued through the XHR-object. Since header keys are case │ │ │ │ │ - // insensitive and we want to allow overriding of the "X-Requested-With" │ │ │ │ │ - // header through the user we cannot use applyDefaults, but have to │ │ │ │ │ - // check manually whether we were called with a "X-Requested-With" │ │ │ │ │ - // header. │ │ │ │ │ - var customRequestedWithHeader = false, │ │ │ │ │ - headerKey; │ │ │ │ │ - for (headerKey in config.headers) { │ │ │ │ │ - if (config.headers.hasOwnProperty(headerKey)) { │ │ │ │ │ - if (headerKey.toLowerCase() === 'x-requested-with') { │ │ │ │ │ - customRequestedWithHeader = true; │ │ │ │ │ - } │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + if (px == null) { │ │ │ │ │ + if ((this.lonlat != null) && (this.map != null)) { │ │ │ │ │ + px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (customRequestedWithHeader === false) { │ │ │ │ │ - // we did not have a custom "X-Requested-With" header │ │ │ │ │ - config.headers['X-Requested-With'] = 'XMLHttpRequest'; │ │ │ │ │ + │ │ │ │ │ + // this assumes that this.map already exists, which is okay because │ │ │ │ │ + // this.draw is only called once the popup has been added to the map. │ │ │ │ │ + if (this.closeOnMove) { │ │ │ │ │ + this.map.events.register("movestart", this, this.hide); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // create request, open, and set headers │ │ │ │ │ - var request = new OpenLayers.Request.XMLHttpRequest(); │ │ │ │ │ - var url = OpenLayers.Util.urlAppend(config.url, │ │ │ │ │ - OpenLayers.Util.getParameterString(config.params || {})); │ │ │ │ │ - url = OpenLayers.Request.makeSameOrigin(url, config.proxy); │ │ │ │ │ - request.open( │ │ │ │ │ - config.method, url, config.async, config.user, config.password │ │ │ │ │ - ); │ │ │ │ │ - for (var header in config.headers) { │ │ │ │ │ - request.setRequestHeader(header, config.headers[header]); │ │ │ │ │ + //listen to movestart, moveend to disable overflow (FF bug) │ │ │ │ │ + if (!this.disableFirefoxOverflowHack && OpenLayers.BROWSER_NAME == 'firefox') { │ │ │ │ │ + this.map.events.register("movestart", this, function() { │ │ │ │ │ + var style = document.defaultView.getComputedStyle( │ │ │ │ │ + this.contentDiv, null │ │ │ │ │ + ); │ │ │ │ │ + var currentOverflow = style.getPropertyValue("overflow"); │ │ │ │ │ + if (currentOverflow != "hidden") { │ │ │ │ │ + this.contentDiv._oldOverflow = currentOverflow; │ │ │ │ │ + this.contentDiv.style.overflow = "hidden"; │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + this.map.events.register("moveend", this, function() { │ │ │ │ │ + var oldOverflow = this.contentDiv._oldOverflow; │ │ │ │ │ + if (oldOverflow) { │ │ │ │ │ + this.contentDiv.style.overflow = oldOverflow; │ │ │ │ │ + this.contentDiv._oldOverflow = null; │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - var events = this.events; │ │ │ │ │ + this.moveTo(px); │ │ │ │ │ + if (!this.autoSize && !this.size) { │ │ │ │ │ + this.setSize(this.contentSize); │ │ │ │ │ + } │ │ │ │ │ + this.setBackgroundColor(); │ │ │ │ │ + this.setOpacity(); │ │ │ │ │ + this.setBorder(); │ │ │ │ │ + this.setContentHTML(); │ │ │ │ │ │ │ │ │ │ - // we want to execute runCallbacks with "this" as the │ │ │ │ │ - // execution scope │ │ │ │ │ - var self = this; │ │ │ │ │ + if (this.panMapIfOutOfView) { │ │ │ │ │ + this.panIntoView(); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - request.onreadystatechange = function() { │ │ │ │ │ - if (request.readyState == OpenLayers.Request.XMLHttpRequest.DONE) { │ │ │ │ │ - var proceed = events.triggerEvent( │ │ │ │ │ - "complete", { │ │ │ │ │ - request: request, │ │ │ │ │ - config: config, │ │ │ │ │ - requestUrl: url │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - if (proceed !== false) { │ │ │ │ │ - self.runCallbacks({ │ │ │ │ │ - request: request, │ │ │ │ │ - config: config, │ │ │ │ │ - requestUrl: url │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ + return this.div; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // send request (optionally with data) and return │ │ │ │ │ - // call in a timeout for asynchronous requests so the return is │ │ │ │ │ - // available before readyState == 4 for cached docs │ │ │ │ │ - if (config.async === false) { │ │ │ │ │ - request.send(config.data); │ │ │ │ │ - } else { │ │ │ │ │ - window.setTimeout(function() { │ │ │ │ │ - if (request.readyState !== 0) { // W3C: 0-UNSENT │ │ │ │ │ - request.send(config.data); │ │ │ │ │ - } │ │ │ │ │ - }, 0); │ │ │ │ │ + /** │ │ │ │ │ + * Method: updatePosition │ │ │ │ │ + * if the popup has a lonlat and its map members set, │ │ │ │ │ + * then have it move itself to its proper position │ │ │ │ │ + */ │ │ │ │ │ + updatePosition: function() { │ │ │ │ │ + if ((this.lonlat) && (this.map)) { │ │ │ │ │ + var px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ + if (px) { │ │ │ │ │ + this.moveTo(px); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return request; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: runCallbacks │ │ │ │ │ - * Calls the complete, success and failure callbacks. Application │ │ │ │ │ - * can listen to the "complete" event, have the listener │ │ │ │ │ - * display a confirm window and always return false, and │ │ │ │ │ - * execute OpenLayers.Request.runCallbacks if the user │ │ │ │ │ - * hits "yes" in the confirm window. │ │ │ │ │ - * │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Hash containing request, config and requestUrl keys │ │ │ │ │ + * px - {<OpenLayers.Pixel>} the top and left position of the popup div. │ │ │ │ │ */ │ │ │ │ │ - runCallbacks: function(options) { │ │ │ │ │ - var request = options.request; │ │ │ │ │ - var config = options.config; │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + if ((px != null) && (this.div != null)) { │ │ │ │ │ + this.div.style.left = px.x + "px"; │ │ │ │ │ + this.div.style.top = px.y + "px"; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // bind callbacks to readyState 4 (done) │ │ │ │ │ - var complete = (config.scope) ? │ │ │ │ │ - OpenLayers.Function.bind(config.callback, config.scope) : │ │ │ │ │ - config.callback; │ │ │ │ │ + /** │ │ │ │ │ + * Method: visible │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Boolean indicating whether or not the popup is visible │ │ │ │ │ + */ │ │ │ │ │ + visible: function() { │ │ │ │ │ + return OpenLayers.Element.visible(this.div); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // optional success callback │ │ │ │ │ - var success; │ │ │ │ │ - if (config.success) { │ │ │ │ │ - success = (config.scope) ? │ │ │ │ │ - OpenLayers.Function.bind(config.success, config.scope) : │ │ │ │ │ - config.success; │ │ │ │ │ + /** │ │ │ │ │ + * Method: toggle │ │ │ │ │ + * Toggles visibility of the popup. │ │ │ │ │ + */ │ │ │ │ │ + toggle: function() { │ │ │ │ │ + if (this.visible()) { │ │ │ │ │ + this.hide(); │ │ │ │ │ + } else { │ │ │ │ │ + this.show(); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // optional failure callback │ │ │ │ │ - var failure; │ │ │ │ │ - if (config.failure) { │ │ │ │ │ - failure = (config.scope) ? │ │ │ │ │ - OpenLayers.Function.bind(config.failure, config.scope) : │ │ │ │ │ - config.failure; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: show │ │ │ │ │ + * Makes the popup visible. │ │ │ │ │ + */ │ │ │ │ │ + show: function() { │ │ │ │ │ + this.div.style.display = ''; │ │ │ │ │ │ │ │ │ │ - if (OpenLayers.Util.createUrlObject(config.url).protocol == "file:" && │ │ │ │ │ - request.responseText) { │ │ │ │ │ - request.status = 200; │ │ │ │ │ + if (this.panMapIfOutOfView) { │ │ │ │ │ + this.panIntoView(); │ │ │ │ │ } │ │ │ │ │ - complete(request); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (!request.status || (request.status >= 200 && request.status < 300)) { │ │ │ │ │ - this.events.triggerEvent("success", options); │ │ │ │ │ - if (success) { │ │ │ │ │ - success(request); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ - this.events.triggerEvent("failure", options); │ │ │ │ │ - if (failure) { │ │ │ │ │ - failure(request); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: hide │ │ │ │ │ + * Makes the popup invisible. │ │ │ │ │ + */ │ │ │ │ │ + hide: function() { │ │ │ │ │ + this.div.style.display = 'none'; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: GET │ │ │ │ │ - * Send an HTTP GET request. Additional configuration properties are │ │ │ │ │ - * documented in the <issue> method, with the method property set │ │ │ │ │ - * to GET. │ │ │ │ │ + * Method: setSize │ │ │ │ │ + * Used to adjust the size of the popup. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the <issue> method for documentation of allowed properties. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ + * contentSize - {<OpenLayers.Size>} the new size for the popup's │ │ │ │ │ + * contents div (in pixels). │ │ │ │ │ */ │ │ │ │ │ - GET: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "GET" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ + setSize: function(contentSize) { │ │ │ │ │ + this.size = contentSize.clone(); │ │ │ │ │ + │ │ │ │ │ + // if our contentDiv has a css 'padding' set on it by a stylesheet, we │ │ │ │ │ + // must add that to the desired "size". │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ + var wPadding = contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ + var hPadding = contentDivPadding.top + contentDivPadding.bottom; │ │ │ │ │ + │ │ │ │ │ + // take into account the popup's 'padding' property │ │ │ │ │ + this.fixPadding(); │ │ │ │ │ + wPadding += this.padding.left + this.padding.right; │ │ │ │ │ + hPadding += this.padding.top + this.padding.bottom; │ │ │ │ │ + │ │ │ │ │ + // make extra space for the close div │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + var closeDivWidth = parseInt(this.closeDiv.style.width); │ │ │ │ │ + wPadding += closeDivWidth + contentDivPadding.right; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //increase size of the main popup div to take into account the │ │ │ │ │ + // users's desired padding and close div. │ │ │ │ │ + this.size.w += wPadding; │ │ │ │ │ + this.size.h += hPadding; │ │ │ │ │ + │ │ │ │ │ + //now if our browser is IE, we need to actually make the contents │ │ │ │ │ + // div itself bigger to take its own padding into effect. this makes │ │ │ │ │ + // me want to shoot someone, but so it goes. │ │ │ │ │ + if (OpenLayers.BROWSER_NAME == "msie") { │ │ │ │ │ + this.contentSize.w += │ │ │ │ │ + contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ + this.contentSize.h += │ │ │ │ │ + contentDivPadding.bottom + contentDivPadding.top; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.div.style.width = this.size.w + "px"; │ │ │ │ │ + this.div.style.height = this.size.h + "px"; │ │ │ │ │ + } │ │ │ │ │ + if (this.contentDiv != null) { │ │ │ │ │ + this.contentDiv.style.width = contentSize.w + "px"; │ │ │ │ │ + this.contentDiv.style.height = contentSize.h + "px"; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: POST │ │ │ │ │ - * Send a POST request. Additional configuration properties are │ │ │ │ │ - * documented in the <issue> method, with the method property set │ │ │ │ │ - * to POST and "Content-Type" header set to "application/xml". │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the <issue> method for documentation of allowed properties. The │ │ │ │ │ - * default "Content-Type" header will be set to "application-xml" if │ │ │ │ │ - * none is provided. This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ + * APIMethod: updateSize │ │ │ │ │ + * Auto size the popup so that it precisely fits its contents (as │ │ │ │ │ + * determined by this.contentDiv.innerHTML). Popup size will, of │ │ │ │ │ + * course, be limited by the available space on the current map │ │ │ │ │ */ │ │ │ │ │ - POST: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "POST" │ │ │ │ │ - }); │ │ │ │ │ - // set content type to application/xml if it isn't already set │ │ │ │ │ - config.headers = config.headers ? config.headers : {}; │ │ │ │ │ - if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ - config.headers["Content-Type"] = "application/xml"; │ │ │ │ │ + updateSize: function() { │ │ │ │ │ + │ │ │ │ │ + // determine actual render dimensions of the contents by putting its │ │ │ │ │ + // contents into a fake contentDiv (for the CSS) and then measuring it │ │ │ │ │ + var preparedHTML = "<div class='" + this.contentDisplayClass + "'>" + │ │ │ │ │ + this.contentDiv.innerHTML + │ │ │ │ │ + "</div>"; │ │ │ │ │ + │ │ │ │ │ + var containerElement = (this.map) ? this.map.div : document.body; │ │ │ │ │ + var realSize = OpenLayers.Util.getRenderedDimensions( │ │ │ │ │ + preparedHTML, null, { │ │ │ │ │ + displayClass: this.displayClass, │ │ │ │ │ + containerElement: containerElement │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + // is the "real" size of the div is safe to display in our map? │ │ │ │ │ + var safeSize = this.getSafeContentSize(realSize); │ │ │ │ │ + │ │ │ │ │ + var newSize = null; │ │ │ │ │ + if (safeSize.equals(realSize)) { │ │ │ │ │ + //real size of content is small enough to fit on the map, │ │ │ │ │ + // so we use real size. │ │ │ │ │ + newSize = realSize; │ │ │ │ │ + │ │ │ │ │ + } else { │ │ │ │ │ + │ │ │ │ │ + // make a new 'size' object with the clipped dimensions │ │ │ │ │ + // set or null if not clipped. │ │ │ │ │ + var fixedSize = { │ │ │ │ │ + w: (safeSize.w < realSize.w) ? safeSize.w : null, │ │ │ │ │ + h: (safeSize.h < realSize.h) ? safeSize.h : null │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + if (fixedSize.w && fixedSize.h) { │ │ │ │ │ + //content is too big in both directions, so we will use │ │ │ │ │ + // max popup size (safeSize), knowing well that it will │ │ │ │ │ + // overflow both ways. │ │ │ │ │ + newSize = safeSize; │ │ │ │ │ + } else { │ │ │ │ │ + //content is clipped in only one direction, so we need to │ │ │ │ │ + // run getRenderedDimensions() again with a fixed dimension │ │ │ │ │ + var clippedSize = OpenLayers.Util.getRenderedDimensions( │ │ │ │ │ + preparedHTML, fixedSize, { │ │ │ │ │ + displayClass: this.contentDisplayClass, │ │ │ │ │ + containerElement: containerElement │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + //if the clipped size is still the same as the safeSize, │ │ │ │ │ + // that means that our content must be fixed in the │ │ │ │ │ + // offending direction. If overflow is 'auto', this means │ │ │ │ │ + // we are going to have a scrollbar for sure, so we must │ │ │ │ │ + // adjust for that. │ │ │ │ │ + // │ │ │ │ │ + var currentOverflow = OpenLayers.Element.getStyle( │ │ │ │ │ + this.contentDiv, "overflow" │ │ │ │ │ + ); │ │ │ │ │ + if ((currentOverflow != "hidden") && │ │ │ │ │ + (clippedSize.equals(safeSize))) { │ │ │ │ │ + var scrollBar = OpenLayers.Util.getScrollbarWidth(); │ │ │ │ │ + if (fixedSize.w) { │ │ │ │ │ + clippedSize.h += scrollBar; │ │ │ │ │ + } else { │ │ │ │ │ + clippedSize.w += scrollBar; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + newSize = this.getSafeContentSize(clippedSize); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ + this.setSize(newSize); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: PUT │ │ │ │ │ - * Send an HTTP PUT request. Additional configuration properties are │ │ │ │ │ - * documented in the <issue> method, with the method property set │ │ │ │ │ - * to PUT and "Content-Type" header set to "application/xml". │ │ │ │ │ + * Method: setBackgroundColor │ │ │ │ │ + * Sets the background color of the popup. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the <issue> method for documentation of allowed properties. The │ │ │ │ │ - * default "Content-Type" header will be set to "application-xml" if │ │ │ │ │ - * none is provided. This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ + * color - {String} the background color. eg "#FFBBBB" │ │ │ │ │ */ │ │ │ │ │ - PUT: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "PUT" │ │ │ │ │ - }); │ │ │ │ │ - // set content type to application/xml if it isn't already set │ │ │ │ │ - config.headers = config.headers ? config.headers : {}; │ │ │ │ │ - if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ - config.headers["Content-Type"] = "application/xml"; │ │ │ │ │ + setBackgroundColor: function(color) { │ │ │ │ │ + if (color != undefined) { │ │ │ │ │ + this.backgroundColor = color; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.div.style.backgroundColor = this.backgroundColor; │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: DELETE │ │ │ │ │ - * Send an HTTP DELETE request. Additional configuration properties are │ │ │ │ │ - * documented in the <issue> method, with the method property set │ │ │ │ │ - * to DELETE. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the <issue> method for documentation of allowed properties. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ + * Method: setOpacity │ │ │ │ │ + * Sets the opacity of the popup. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * opacity - {float} A value between 0.0 (transparent) and 1.0 (solid). │ │ │ │ │ */ │ │ │ │ │ - DELETE: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "DELETE" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + if (opacity != undefined) { │ │ │ │ │ + this.opacity = opacity; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + // for Mozilla and Safari │ │ │ │ │ + this.div.style.opacity = this.opacity; │ │ │ │ │ + │ │ │ │ │ + // for IE │ │ │ │ │ + this.div.style.filter = 'alpha(opacity=' + this.opacity * 100 + ')'; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: HEAD │ │ │ │ │ - * Send an HTTP HEAD request. Additional configuration properties are │ │ │ │ │ - * documented in the <issue> method, with the method property set │ │ │ │ │ - * to HEAD. │ │ │ │ │ + * Method: setBorder │ │ │ │ │ + * Sets the border style of the popup. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the <issue> method for documentation of allowed properties. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ + * border - {String} The border style value. eg 2px │ │ │ │ │ */ │ │ │ │ │ - HEAD: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "HEAD" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ + setBorder: function(border) { │ │ │ │ │ + if (border != undefined) { │ │ │ │ │ + this.border = border; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.div.style.border = this.border; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: OPTIONS │ │ │ │ │ - * Send an HTTP OPTIONS request. Additional configuration properties are │ │ │ │ │ - * documented in the <issue> method, with the method property set │ │ │ │ │ - * to OPTIONS. │ │ │ │ │ + * Method: setContentHTML │ │ │ │ │ + * Allows the user to set the HTML content of the popup. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the <issue> method for documentation of allowed properties. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ + * contentHTML - {String} HTML for the div. │ │ │ │ │ */ │ │ │ │ │ - OPTIONS: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "OPTIONS" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ - } │ │ │ │ │ + setContentHTML: function(contentHTML) { │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + if (contentHTML != null) { │ │ │ │ │ + this.contentHTML = contentHTML; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -// XMLHttpRequest.js Copyright (C) 2010 Sergey Ilinsky (http://www.ilinsky.com) │ │ │ │ │ -// │ │ │ │ │ -// Licensed under the Apache License, Version 2.0 (the "License"); │ │ │ │ │ -// you may not use this file except in compliance with the License. │ │ │ │ │ -// You may obtain a copy of the License at │ │ │ │ │ -// │ │ │ │ │ -// http://www.apache.org/licenses/LICENSE-2.0 │ │ │ │ │ -// │ │ │ │ │ -// Unless required by applicable law or agreed to in writing, software │ │ │ │ │ -// distributed under the License is distributed on an "AS IS" BASIS, │ │ │ │ │ -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. │ │ │ │ │ -// See the License for the specific language governing permissions and │ │ │ │ │ -// limitations under the License. │ │ │ │ │ + if ((this.contentDiv != null) && │ │ │ │ │ + (this.contentHTML != null) && │ │ │ │ │ + (this.contentHTML != this.contentDiv.innerHTML)) { │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Request.js │ │ │ │ │ - */ │ │ │ │ │ + this.contentDiv.innerHTML = this.contentHTML; │ │ │ │ │ │ │ │ │ │ -(function() { │ │ │ │ │ + if (this.autoSize) { │ │ │ │ │ │ │ │ │ │ - // Save reference to earlier defined object implementation (if any) │ │ │ │ │ - var oXMLHttpRequest = window.XMLHttpRequest; │ │ │ │ │ + //if popup has images, listen for when they finish │ │ │ │ │ + // loading and resize accordingly │ │ │ │ │ + this.registerImageListeners(); │ │ │ │ │ │ │ │ │ │ - // Define on browser type │ │ │ │ │ - var bGecko = !!window.controllers, │ │ │ │ │ - bIE = window.document.all && !window.opera, │ │ │ │ │ - bIE7 = bIE && window.navigator.userAgent.match(/MSIE 7.0/); │ │ │ │ │ + //auto size the popup to its current contents │ │ │ │ │ + this.updateSize(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // Enables "XMLHttpRequest()" call next to "new XMLHttpReques()" │ │ │ │ │ - function fXMLHttpRequest() { │ │ │ │ │ - this._object = oXMLHttpRequest && !bIE7 ? new oXMLHttpRequest : new window.ActiveXObject("Microsoft.XMLHTTP"); │ │ │ │ │ - this._listeners = []; │ │ │ │ │ - }; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Constructor │ │ │ │ │ - function cXMLHttpRequest() { │ │ │ │ │ - return new fXMLHttpRequest; │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype = fXMLHttpRequest.prototype; │ │ │ │ │ + /** │ │ │ │ │ + * Method: registerImageListeners │ │ │ │ │ + * Called when an image contained by the popup loaded. this function │ │ │ │ │ + * updates the popup size, then unregisters the image load listener. │ │ │ │ │ + */ │ │ │ │ │ + registerImageListeners: function() { │ │ │ │ │ │ │ │ │ │ - // BUGFIX: Firefox with Firebug installed would break pages if not executed │ │ │ │ │ - if (bGecko && oXMLHttpRequest.wrapped) │ │ │ │ │ - cXMLHttpRequest.wrapped = oXMLHttpRequest.wrapped; │ │ │ │ │ + // As the images load, this function will call updateSize() to │ │ │ │ │ + // resize the popup to fit the content div (which presumably is now │ │ │ │ │ + // bigger than when the image was not loaded). │ │ │ │ │ + // │ │ │ │ │ + // If the 'panMapIfOutOfView' property is set, we will pan the newly │ │ │ │ │ + // resized popup back into view. │ │ │ │ │ + // │ │ │ │ │ + // Note that this function, when called, will have 'popup' and │ │ │ │ │ + // 'img' properties in the context. │ │ │ │ │ + // │ │ │ │ │ + var onImgLoad = function() { │ │ │ │ │ + if (this.popup.id === null) { // this.popup has been destroyed! │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + this.popup.updateSize(); │ │ │ │ │ │ │ │ │ │ - // Constants │ │ │ │ │ - cXMLHttpRequest.UNSENT = 0; │ │ │ │ │ - cXMLHttpRequest.OPENED = 1; │ │ │ │ │ - cXMLHttpRequest.HEADERS_RECEIVED = 2; │ │ │ │ │ - cXMLHttpRequest.LOADING = 3; │ │ │ │ │ - cXMLHttpRequest.DONE = 4; │ │ │ │ │ + if (this.popup.visible() && this.popup.panMapIfOutOfView) { │ │ │ │ │ + this.popup.panIntoView(); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // Public Properties │ │ │ │ │ - cXMLHttpRequest.prototype.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ - cXMLHttpRequest.prototype.responseText = ''; │ │ │ │ │ - cXMLHttpRequest.prototype.responseXML = null; │ │ │ │ │ - cXMLHttpRequest.prototype.status = 0; │ │ │ │ │ - cXMLHttpRequest.prototype.statusText = ''; │ │ │ │ │ + OpenLayers.Event.stopObserving( │ │ │ │ │ + this.img, "load", this.img._onImgLoad │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - // Priority proposal │ │ │ │ │ - cXMLHttpRequest.prototype.priority = "NORMAL"; │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - // Instance-level Events Handlers │ │ │ │ │ - cXMLHttpRequest.prototype.onreadystatechange = null; │ │ │ │ │ + //cycle through the images and if their size is 0x0, that means that │ │ │ │ │ + // they haven't been loaded yet, so we attach the listener, which │ │ │ │ │ + // will fire when the images finish loading and will resize the │ │ │ │ │ + // popup accordingly to its new size. │ │ │ │ │ + var images = this.contentDiv.getElementsByTagName("img"); │ │ │ │ │ + for (var i = 0, len = images.length; i < len; i++) { │ │ │ │ │ + var img = images[i]; │ │ │ │ │ + if (img.width == 0 || img.height == 0) { │ │ │ │ │ │ │ │ │ │ - // Class-level Events Handlers │ │ │ │ │ - cXMLHttpRequest.onreadystatechange = null; │ │ │ │ │ - cXMLHttpRequest.onopen = null; │ │ │ │ │ - cXMLHttpRequest.onsend = null; │ │ │ │ │ - cXMLHttpRequest.onabort = null; │ │ │ │ │ + var context = { │ │ │ │ │ + 'popup': this, │ │ │ │ │ + 'img': img │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - // Public Methods │ │ │ │ │ - cXMLHttpRequest.prototype.open = function(sMethod, sUrl, bAsync, sUser, sPassword) { │ │ │ │ │ - // Delete headers, required when object is reused │ │ │ │ │ - delete this._headers; │ │ │ │ │ + //expando this function to the image itself before registering │ │ │ │ │ + // it. This way we can easily and properly unregister it. │ │ │ │ │ + img._onImgLoad = OpenLayers.Function.bind(onImgLoad, context); │ │ │ │ │ │ │ │ │ │ - // When bAsync parameter value is omitted, use true as default │ │ │ │ │ - if (arguments.length < 3) │ │ │ │ │ - bAsync = true; │ │ │ │ │ + OpenLayers.Event.observe(img, 'load', img._onImgLoad); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Save async parameter for fixing Gecko bug with missing readystatechange in synchronous requests │ │ │ │ │ - this._async = bAsync; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getSafeContentSize │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * size - {<OpenLayers.Size>} Desired size to make the popup. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Size>} A size to make the popup which is neither smaller │ │ │ │ │ + * than the specified minimum size, nor bigger than the maximum │ │ │ │ │ + * size (which is calculated relative to the size of the viewport). │ │ │ │ │ + */ │ │ │ │ │ + getSafeContentSize: function(size) { │ │ │ │ │ │ │ │ │ │ - // Set the onreadystatechange handler │ │ │ │ │ - var oRequest = this, │ │ │ │ │ - nState = this.readyState, │ │ │ │ │ - fOnUnload; │ │ │ │ │ + var safeContentSize = size.clone(); │ │ │ │ │ │ │ │ │ │ - // BUGFIX: IE - memory leak on page unload (inter-page leak) │ │ │ │ │ - if (bIE && bAsync) { │ │ │ │ │ - fOnUnload = function() { │ │ │ │ │ - if (nState != cXMLHttpRequest.DONE) { │ │ │ │ │ - fCleanTransport(oRequest); │ │ │ │ │ - // Safe to abort here since onreadystatechange handler removed │ │ │ │ │ - oRequest.abort(); │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - window.attachEvent("onunload", fOnUnload); │ │ │ │ │ + // if our contentDiv has a css 'padding' set on it by a stylesheet, we │ │ │ │ │ + // must add that to the desired "size". │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ + var wPadding = contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ + var hPadding = contentDivPadding.top + contentDivPadding.bottom; │ │ │ │ │ + │ │ │ │ │ + // take into account the popup's 'padding' property │ │ │ │ │ + this.fixPadding(); │ │ │ │ │ + wPadding += this.padding.left + this.padding.right; │ │ │ │ │ + hPadding += this.padding.top + this.padding.bottom; │ │ │ │ │ + │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + var closeDivWidth = parseInt(this.closeDiv.style.width); │ │ │ │ │ + wPadding += closeDivWidth + contentDivPadding.right; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // Add method sniffer │ │ │ │ │ - if (cXMLHttpRequest.onopen) │ │ │ │ │ - cXMLHttpRequest.onopen.apply(this, arguments); │ │ │ │ │ + // prevent the popup from being smaller than a specified minimal size │ │ │ │ │ + if (this.minSize) { │ │ │ │ │ + safeContentSize.w = Math.max(safeContentSize.w, │ │ │ │ │ + (this.minSize.w - wPadding)); │ │ │ │ │ + safeContentSize.h = Math.max(safeContentSize.h, │ │ │ │ │ + (this.minSize.h - hPadding)); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (arguments.length > 4) │ │ │ │ │ - this._object.open(sMethod, sUrl, bAsync, sUser, sPassword); │ │ │ │ │ - else │ │ │ │ │ - if (arguments.length > 3) │ │ │ │ │ - this._object.open(sMethod, sUrl, bAsync, sUser); │ │ │ │ │ - else │ │ │ │ │ - this._object.open(sMethod, sUrl, bAsync); │ │ │ │ │ + // prevent the popup from being bigger than a specified maximum size │ │ │ │ │ + if (this.maxSize) { │ │ │ │ │ + safeContentSize.w = Math.min(safeContentSize.w, │ │ │ │ │ + (this.maxSize.w - wPadding)); │ │ │ │ │ + safeContentSize.h = Math.min(safeContentSize.h, │ │ │ │ │ + (this.maxSize.h - hPadding)); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - this.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ - fReadyStateChange(this); │ │ │ │ │ + //make sure the desired size to set doesn't result in a popup that │ │ │ │ │ + // is bigger than the map's viewport. │ │ │ │ │ + // │ │ │ │ │ + if (this.map && this.map.size) { │ │ │ │ │ │ │ │ │ │ - this._object.onreadystatechange = function() { │ │ │ │ │ - if (bGecko && !bAsync) │ │ │ │ │ - return; │ │ │ │ │ + var extraX = 0, │ │ │ │ │ + extraY = 0; │ │ │ │ │ + if (this.keepInMap && !this.panMapIfOutOfView) { │ │ │ │ │ + var px = this.map.getPixelFromLonLat(this.lonlat); │ │ │ │ │ + switch (this.relativePosition) { │ │ │ │ │ + case "tr": │ │ │ │ │ + extraX = px.x; │ │ │ │ │ + extraY = this.map.size.h - px.y; │ │ │ │ │ + break; │ │ │ │ │ + case "tl": │ │ │ │ │ + extraX = this.map.size.w - px.x; │ │ │ │ │ + extraY = this.map.size.h - px.y; │ │ │ │ │ + break; │ │ │ │ │ + case "bl": │ │ │ │ │ + extraX = this.map.size.w - px.x; │ │ │ │ │ + extraY = px.y; │ │ │ │ │ + break; │ │ │ │ │ + case "br": │ │ │ │ │ + extraX = px.x; │ │ │ │ │ + extraY = px.y; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + extraX = px.x; │ │ │ │ │ + extraY = this.map.size.h - px.y; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // Synchronize state │ │ │ │ │ - oRequest.readyState = oRequest._object.readyState; │ │ │ │ │ + var maxY = this.map.size.h - │ │ │ │ │ + this.map.paddingForPopups.top - │ │ │ │ │ + this.map.paddingForPopups.bottom - │ │ │ │ │ + hPadding - extraY; │ │ │ │ │ │ │ │ │ │ - // │ │ │ │ │ - fSynchronizeValues(oRequest); │ │ │ │ │ + var maxX = this.map.size.w - │ │ │ │ │ + this.map.paddingForPopups.left - │ │ │ │ │ + this.map.paddingForPopups.right - │ │ │ │ │ + wPadding - extraX; │ │ │ │ │ │ │ │ │ │ - // BUGFIX: Firefox fires unnecessary DONE when aborting │ │ │ │ │ - if (oRequest._aborted) { │ │ │ │ │ - // Reset readyState to UNSENT │ │ │ │ │ - oRequest.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + safeContentSize.w = Math.min(safeContentSize.w, maxX); │ │ │ │ │ + safeContentSize.h = Math.min(safeContentSize.h, maxY); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // Return now │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ + return safeContentSize; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (oRequest.readyState == cXMLHttpRequest.DONE) { │ │ │ │ │ - // Free up queue │ │ │ │ │ - delete oRequest._data; │ │ │ │ │ - /* if (bAsync) │ │ │ │ │ - fQueue_remove(oRequest);*/ │ │ │ │ │ - // │ │ │ │ │ - fCleanTransport(oRequest); │ │ │ │ │ - // Uncomment this block if you need a fix for IE cache │ │ │ │ │ - /* │ │ │ │ │ - // BUGFIX: IE - cache issue │ │ │ │ │ - if (!oRequest._object.getResponseHeader("Date")) { │ │ │ │ │ - // Save object to cache │ │ │ │ │ - oRequest._cached = oRequest._object; │ │ │ │ │ + /** │ │ │ │ │ + * Method: getContentDivPadding │ │ │ │ │ + * Glorious, oh glorious hack in order to determine the css 'padding' of │ │ │ │ │ + * the contentDiv. IE/Opera return null here unless we actually add the │ │ │ │ │ + * popup's main 'div' element (which contains contentDiv) to the DOM. │ │ │ │ │ + * So we make it invisible and then add it to the document temporarily. │ │ │ │ │ + * │ │ │ │ │ + * Once we've taken the padding readings we need, we then remove it │ │ │ │ │ + * from the DOM (it will actually get added to the DOM in │ │ │ │ │ + * Map.js's addPopup) │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} │ │ │ │ │ + */ │ │ │ │ │ + getContentDivPadding: function() { │ │ │ │ │ │ │ │ │ │ - // Instantiate a new transport object │ │ │ │ │ - cXMLHttpRequest.call(oRequest); │ │ │ │ │ + //use cached value if we have it │ │ │ │ │ + var contentDivPadding = this._contentDivPadding; │ │ │ │ │ + if (!contentDivPadding) { │ │ │ │ │ │ │ │ │ │ - // Re-send request │ │ │ │ │ - if (sUser) { │ │ │ │ │ - if (sPassword) │ │ │ │ │ - oRequest._object.open(sMethod, sUrl, bAsync, sUser, sPassword); │ │ │ │ │ - else │ │ │ │ │ - oRequest._object.open(sMethod, sUrl, bAsync, sUser); │ │ │ │ │ - } │ │ │ │ │ - else │ │ │ │ │ - oRequest._object.open(sMethod, sUrl, bAsync); │ │ │ │ │ - oRequest._object.setRequestHeader("If-Modified-Since", oRequest._cached.getResponseHeader("Last-Modified") || new window.Date(0)); │ │ │ │ │ - // Copy headers set │ │ │ │ │ - if (oRequest._headers) │ │ │ │ │ - for (var sHeader in oRequest._headers) │ │ │ │ │ - if (typeof oRequest._headers[sHeader] == "string") // Some frameworks prototype objects with functions │ │ │ │ │ - oRequest._object.setRequestHeader(sHeader, oRequest._headers[sHeader]); │ │ │ │ │ + if (this.div.parentNode == null) { │ │ │ │ │ + //make the div invisible and add it to the page │ │ │ │ │ + this.div.style.display = "none"; │ │ │ │ │ + document.body.appendChild(this.div); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - oRequest._object.onreadystatechange = function() { │ │ │ │ │ - // Synchronize state │ │ │ │ │ - oRequest.readyState = oRequest._object.readyState; │ │ │ │ │ + //read the padding settings from css, put them in an OL.Bounds │ │ │ │ │ + contentDivPadding = new OpenLayers.Bounds( │ │ │ │ │ + OpenLayers.Element.getStyle(this.contentDiv, "padding-left"), │ │ │ │ │ + OpenLayers.Element.getStyle(this.contentDiv, "padding-bottom"), │ │ │ │ │ + OpenLayers.Element.getStyle(this.contentDiv, "padding-right"), │ │ │ │ │ + OpenLayers.Element.getStyle(this.contentDiv, "padding-top") │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - if (oRequest._aborted) { │ │ │ │ │ - // │ │ │ │ │ - oRequest.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + //cache the value │ │ │ │ │ + this._contentDivPadding = contentDivPadding; │ │ │ │ │ │ │ │ │ │ - // Return │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ + if (this.div.parentNode == document.body) { │ │ │ │ │ + //remove the div from the page and make it visible again │ │ │ │ │ + document.body.removeChild(this.div); │ │ │ │ │ + this.div.style.display = ""; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return contentDivPadding; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (oRequest.readyState == cXMLHttpRequest.DONE) { │ │ │ │ │ - // Clean Object │ │ │ │ │ - fCleanTransport(oRequest); │ │ │ │ │ + /** │ │ │ │ │ + * Method: addCloseBox │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * callback - {Function} The callback to be called when the close button │ │ │ │ │ + * is clicked. │ │ │ │ │ + */ │ │ │ │ │ + addCloseBox: function(callback) { │ │ │ │ │ │ │ │ │ │ - // get cached request │ │ │ │ │ - if (oRequest.status == 304) │ │ │ │ │ - oRequest._object = oRequest._cached; │ │ │ │ │ + this.closeDiv = OpenLayers.Util.createDiv( │ │ │ │ │ + this.id + "_close", null, { │ │ │ │ │ + w: 17, │ │ │ │ │ + h: 17 │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + this.closeDiv.className = "olPopupCloseBox"; │ │ │ │ │ │ │ │ │ │ - // │ │ │ │ │ - delete oRequest._cached; │ │ │ │ │ + // use the content div's css padding to determine if we should │ │ │ │ │ + // padd the close div │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ │ │ │ │ │ - // │ │ │ │ │ - fSynchronizeValues(oRequest); │ │ │ │ │ + this.closeDiv.style.right = contentDivPadding.right + "px"; │ │ │ │ │ + this.closeDiv.style.top = contentDivPadding.top + "px"; │ │ │ │ │ + this.groupDiv.appendChild(this.closeDiv); │ │ │ │ │ │ │ │ │ │ - // │ │ │ │ │ - fReadyStateChange(oRequest); │ │ │ │ │ + var closePopup = callback || function(e) { │ │ │ │ │ + this.hide(); │ │ │ │ │ + OpenLayers.Event.stop(e); │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Event.observe(this.closeDiv, "touchend", │ │ │ │ │ + OpenLayers.Function.bindAsEventListener(closePopup, this)); │ │ │ │ │ + OpenLayers.Event.observe(this.closeDiv, "click", │ │ │ │ │ + OpenLayers.Function.bindAsEventListener(closePopup, this)); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // BUGFIX: IE - memory leak in interrupted │ │ │ │ │ - if (bIE && bAsync) │ │ │ │ │ - window.detachEvent("onunload", fOnUnload); │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - oRequest._object.send(null); │ │ │ │ │ + /** │ │ │ │ │ + * Method: panIntoView │ │ │ │ │ + * Pans the map such that the popup is totaly viewable (if necessary) │ │ │ │ │ + */ │ │ │ │ │ + panIntoView: function() { │ │ │ │ │ │ │ │ │ │ - // Return now - wait until re-sent request is finished │ │ │ │ │ - return; │ │ │ │ │ - }; │ │ │ │ │ - */ │ │ │ │ │ - // BUGFIX: IE - memory leak in interrupted │ │ │ │ │ - if (bIE && bAsync) │ │ │ │ │ - window.detachEvent("onunload", fOnUnload); │ │ │ │ │ - } │ │ │ │ │ + var mapSize = this.map.getSize(); │ │ │ │ │ │ │ │ │ │ - // BUGFIX: Some browsers (Internet Explorer, Gecko) fire OPEN readystate twice │ │ │ │ │ - if (nState != oRequest.readyState) │ │ │ │ │ - fReadyStateChange(oRequest); │ │ │ │ │ + //start with the top left corner of the popup, in px, │ │ │ │ │ + // relative to the viewport │ │ │ │ │ + var origTL = this.map.getViewPortPxFromLayerPx(new OpenLayers.Pixel( │ │ │ │ │ + parseInt(this.div.style.left), │ │ │ │ │ + parseInt(this.div.style.top) │ │ │ │ │ + )); │ │ │ │ │ + var newTL = origTL.clone(); │ │ │ │ │ │ │ │ │ │ - nState = oRequest.readyState; │ │ │ │ │ + //new left (compare to margins, using this.size to calculate right) │ │ │ │ │ + if (origTL.x < this.map.paddingForPopups.left) { │ │ │ │ │ + newTL.x = this.map.paddingForPopups.left; │ │ │ │ │ + } else │ │ │ │ │ + if ((origTL.x + this.size.w) > (mapSize.w - this.map.paddingForPopups.right)) { │ │ │ │ │ + newTL.x = mapSize.w - this.map.paddingForPopups.right - this.size.w; │ │ │ │ │ } │ │ │ │ │ - }; │ │ │ │ │ │ │ │ │ │ - function fXMLHttpRequest_send(oRequest) { │ │ │ │ │ - oRequest._object.send(oRequest._data); │ │ │ │ │ + //new top (compare to margins, using this.size to calculate bottom) │ │ │ │ │ + if (origTL.y < this.map.paddingForPopups.top) { │ │ │ │ │ + newTL.y = this.map.paddingForPopups.top; │ │ │ │ │ + } else │ │ │ │ │ + if ((origTL.y + this.size.h) > (mapSize.h - this.map.paddingForPopups.bottom)) { │ │ │ │ │ + newTL.y = mapSize.h - this.map.paddingForPopups.bottom - this.size.h; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // BUGFIX: Gecko - missing readystatechange calls in synchronous requests │ │ │ │ │ - if (bGecko && !oRequest._async) { │ │ │ │ │ - oRequest.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ + var dx = origTL.x - newTL.x; │ │ │ │ │ + var dy = origTL.y - newTL.y; │ │ │ │ │ │ │ │ │ │ - // Synchronize state │ │ │ │ │ - fSynchronizeValues(oRequest); │ │ │ │ │ + this.map.pan(dx, dy); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Simulate missing states │ │ │ │ │ - while (oRequest.readyState < cXMLHttpRequest.DONE) { │ │ │ │ │ - oRequest.readyState++; │ │ │ │ │ - fReadyStateChange(oRequest); │ │ │ │ │ - // Check if we are aborted │ │ │ │ │ - if (oRequest._aborted) │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: registerEvents │ │ │ │ │ + * Registers events on the popup. │ │ │ │ │ + * │ │ │ │ │ + * Do this in a separate function so that subclasses can │ │ │ │ │ + * choose to override it if they wish to deal differently │ │ │ │ │ + * with mouse events │ │ │ │ │ + * │ │ │ │ │ + * Note in the following handler functions that some special │ │ │ │ │ + * care is needed to deal correctly with mousing and popups. │ │ │ │ │ + * │ │ │ │ │ + * Because the user might select the zoom-rectangle option and │ │ │ │ │ + * then drag it over a popup, we need a safe way to allow the │ │ │ │ │ + * mousemove and mouseup events to pass through the popup when │ │ │ │ │ + * they are initiated from outside. The same procedure is needed for │ │ │ │ │ + * touchmove and touchend events. │ │ │ │ │ + * │ │ │ │ │ + * Otherwise, we want to essentially kill the event propagation │ │ │ │ │ + * for all other events, though we have to do so carefully, │ │ │ │ │ + * without disabling basic html functionality, like clicking on │ │ │ │ │ + * hyperlinks or drag-selecting text. │ │ │ │ │ + */ │ │ │ │ │ + registerEvents: function() { │ │ │ │ │ + this.events = new OpenLayers.Events(this, this.div, null, true); │ │ │ │ │ + │ │ │ │ │ + function onTouchstart(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt, true); │ │ │ │ │ } │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.send = function(vData) { │ │ │ │ │ - // Add method sniffer │ │ │ │ │ - if (cXMLHttpRequest.onsend) │ │ │ │ │ - cXMLHttpRequest.onsend.apply(this, arguments); │ │ │ │ │ + this.events.on({ │ │ │ │ │ + "mousedown": this.onmousedown, │ │ │ │ │ + "mousemove": this.onmousemove, │ │ │ │ │ + "mouseup": this.onmouseup, │ │ │ │ │ + "click": this.onclick, │ │ │ │ │ + "mouseout": this.onmouseout, │ │ │ │ │ + "dblclick": this.ondblclick, │ │ │ │ │ + "touchstart": onTouchstart, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - if (!arguments.length) │ │ │ │ │ - vData = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // BUGFIX: Safari - fails sending documents created/modified dynamically, so an explicit serialization required │ │ │ │ │ - // BUGFIX: IE - rewrites any custom mime-type to "text/xml" in case an XMLNode is sent │ │ │ │ │ - // BUGFIX: Gecko - fails sending Element (this is up to the implementation either to standard) │ │ │ │ │ - if (vData && vData.nodeType) { │ │ │ │ │ - vData = window.XMLSerializer ? new window.XMLSerializer().serializeToString(vData) : vData.xml; │ │ │ │ │ - if (!this._headers["Content-Type"]) │ │ │ │ │ - this._object.setRequestHeader("Content-Type", "application/xml"); │ │ │ │ │ + /** │ │ │ │ │ + * Method: onmousedown │ │ │ │ │ + * When mouse goes down within the popup, make a note of │ │ │ │ │ + * it locally, and then do not propagate the mousedown │ │ │ │ │ + * (but do so safely so that user can select text inside) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + onmousedown: function(evt) { │ │ │ │ │ + this.mousedown = true; │ │ │ │ │ + OpenLayers.Event.stop(evt, true); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: onmousemove │ │ │ │ │ + * If the drag was started within the popup, then │ │ │ │ │ + * do not propagate the mousemove (but do so safely │ │ │ │ │ + * so that user can select text inside) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + onmousemove: function(evt) { │ │ │ │ │ + if (this.mousedown) { │ │ │ │ │ + OpenLayers.Event.stop(evt, true); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this._data = vData; │ │ │ │ │ - /* │ │ │ │ │ - // Add to queue │ │ │ │ │ - if (this._async) │ │ │ │ │ - fQueue_add(this); │ │ │ │ │ - else*/ │ │ │ │ │ - fXMLHttpRequest_send(this); │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.abort = function() { │ │ │ │ │ - // Add method sniffer │ │ │ │ │ - if (cXMLHttpRequest.onabort) │ │ │ │ │ - cXMLHttpRequest.onabort.apply(this, arguments); │ │ │ │ │ + /** │ │ │ │ │ + * Method: onmouseup │ │ │ │ │ + * When mouse comes up within the popup, after going down │ │ │ │ │ + * in it, reset the flag, and then (once again) do not │ │ │ │ │ + * propagate the event, but do so safely so that user can │ │ │ │ │ + * select text inside │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + onmouseup: function(evt) { │ │ │ │ │ + if (this.mousedown) { │ │ │ │ │ + this.mousedown = false; │ │ │ │ │ + OpenLayers.Event.stop(evt, true); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // BUGFIX: Gecko - unnecessary DONE when aborting │ │ │ │ │ - if (this.readyState > cXMLHttpRequest.UNSENT) │ │ │ │ │ - this._aborted = true; │ │ │ │ │ + /** │ │ │ │ │ + * Method: onclick │ │ │ │ │ + * Ignore clicks, but allowing default browser handling │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + onclick: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt, true); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this._object.abort(); │ │ │ │ │ + /** │ │ │ │ │ + * Method: onmouseout │ │ │ │ │ + * When mouse goes out of the popup set the flag to false so that │ │ │ │ │ + * if they let go and then drag back in, we won't be confused. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + onmouseout: function(evt) { │ │ │ │ │ + this.mousedown = false; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // BUGFIX: IE - memory leak │ │ │ │ │ - fCleanTransport(this); │ │ │ │ │ + /** │ │ │ │ │ + * Method: ondblclick │ │ │ │ │ + * Ignore double-clicks, but allowing default browser handling │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + ondblclick: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt, true); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ - delete this._data; │ │ │ │ │ - /* if (this._async) │ │ │ │ │ - fQueue_remove(this);*/ │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.getAllResponseHeaders = function() { │ │ │ │ │ - return this._object.getAllResponseHeaders(); │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.getResponseHeader = function(sName) { │ │ │ │ │ - return this._object.getResponseHeader(sName); │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.setRequestHeader = function(sName, sValue) { │ │ │ │ │ - // BUGFIX: IE - cache issue │ │ │ │ │ - if (!this._headers) │ │ │ │ │ - this._headers = {}; │ │ │ │ │ - this._headers[sName] = sValue; │ │ │ │ │ +OpenLayers.Popup.WIDTH = 200; │ │ │ │ │ +OpenLayers.Popup.HEIGHT = 200; │ │ │ │ │ +OpenLayers.Popup.COLOR = "white"; │ │ │ │ │ +OpenLayers.Popup.OPACITY = 1; │ │ │ │ │ +OpenLayers.Popup.BORDER = "0px"; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Popup/Anchored.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - return this._object.setRequestHeader(sName, sValue); │ │ │ │ │ - }; │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - // EventTarget interface implementation │ │ │ │ │ - cXMLHttpRequest.prototype.addEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ - for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ - if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) │ │ │ │ │ - return; │ │ │ │ │ - // Add listener │ │ │ │ │ - this._listeners.push([sName, fHandler, bUseCapture]); │ │ │ │ │ - }; │ │ │ │ │ │ │ │ │ │ - cXMLHttpRequest.prototype.removeEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ - for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ - if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) │ │ │ │ │ - break; │ │ │ │ │ - // Remove listener │ │ │ │ │ - if (oListener) │ │ │ │ │ - this._listeners.splice(nIndex, 1); │ │ │ │ │ - }; │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Popup.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - cXMLHttpRequest.prototype.dispatchEvent = function(oEvent) { │ │ │ │ │ - var oEventPseudo = { │ │ │ │ │ - 'type': oEvent.type, │ │ │ │ │ - 'target': this, │ │ │ │ │ - 'currentTarget': this, │ │ │ │ │ - 'eventPhase': 2, │ │ │ │ │ - 'bubbles': oEvent.bubbles, │ │ │ │ │ - 'cancelable': oEvent.cancelable, │ │ │ │ │ - 'timeStamp': oEvent.timeStamp, │ │ │ │ │ - 'stopPropagation': function() {}, // There is no flow │ │ │ │ │ - 'preventDefault': function() {}, // There is no default action │ │ │ │ │ - 'initEvent': function() {} // Original event object should be initialized │ │ │ │ │ - }; │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Popup.Anchored │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Popup> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Popup.Anchored = │ │ │ │ │ + OpenLayers.Class(OpenLayers.Popup, { │ │ │ │ │ │ │ │ │ │ - // Execute onreadystatechange │ │ │ │ │ - if (oEventPseudo.type == "readystatechange" && this.onreadystatechange) │ │ │ │ │ - (this.onreadystatechange.handleEvent || this.onreadystatechange).apply(this, [oEventPseudo]); │ │ │ │ │ + /** │ │ │ │ │ + * Property: relativePosition │ │ │ │ │ + * {String} Relative position of the popup ("br", "tr", "tl" or "bl"). │ │ │ │ │ + */ │ │ │ │ │ + relativePosition: null, │ │ │ │ │ │ │ │ │ │ - // Execute listeners │ │ │ │ │ - for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ - if (oListener[0] == oEventPseudo.type && !oListener[2]) │ │ │ │ │ - (oListener[1].handleEvent || oListener[1]).apply(this, [oEventPseudo]); │ │ │ │ │ - }; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: keepInMap │ │ │ │ │ + * {Boolean} If panMapIfOutOfView is false, and this property is true, │ │ │ │ │ + * contrain the popup such that it always fits in the available map │ │ │ │ │ + * space. By default, this is set. If you are creating popups that are │ │ │ │ │ + * near map edges and not allowing pannning, and especially if you have │ │ │ │ │ + * a popup which has a fixedRelativePosition, setting this to false may │ │ │ │ │ + * be a smart thing to do. │ │ │ │ │ + * │ │ │ │ │ + * For anchored popups, default is true, since subclasses will │ │ │ │ │ + * usually want this functionality. │ │ │ │ │ + */ │ │ │ │ │ + keepInMap: true, │ │ │ │ │ │ │ │ │ │ - // │ │ │ │ │ - cXMLHttpRequest.prototype.toString = function() { │ │ │ │ │ - return '[' + "object" + ' ' + "XMLHttpRequest" + ']'; │ │ │ │ │ - }; │ │ │ │ │ + /** │ │ │ │ │ + * Property: anchor │ │ │ │ │ + * {Object} Object to which we'll anchor the popup. Must expose a │ │ │ │ │ + * 'size' (<OpenLayers.Size>) and 'offset' (<OpenLayers.Pixel>). │ │ │ │ │ + */ │ │ │ │ │ + anchor: null, │ │ │ │ │ │ │ │ │ │ - cXMLHttpRequest.toString = function() { │ │ │ │ │ - return '[' + "XMLHttpRequest" + ']'; │ │ │ │ │ - }; │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Popup.Anchored │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ + * contentSize - {<OpenLayers.Size>} │ │ │ │ │ + * contentHTML - {String} │ │ │ │ │ + * anchor - {Object} Object which must expose a 'size' <OpenLayers.Size> │ │ │ │ │ + * and 'offset' <OpenLayers.Pixel> (generally an <OpenLayers.Icon>). │ │ │ │ │ + * closeBox - {Boolean} │ │ │ │ │ + * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, │ │ │ │ │ + closeBoxCallback) { │ │ │ │ │ + var newArguments = [ │ │ │ │ │ + id, lonlat, contentSize, contentHTML, closeBox, closeBoxCallback │ │ │ │ │ + ]; │ │ │ │ │ + OpenLayers.Popup.prototype.initialize.apply(this, newArguments); │ │ │ │ │ │ │ │ │ │ - // Helper function │ │ │ │ │ - function fReadyStateChange(oRequest) { │ │ │ │ │ - // Sniffing code │ │ │ │ │ - if (cXMLHttpRequest.onreadystatechange) │ │ │ │ │ - cXMLHttpRequest.onreadystatechange.apply(oRequest); │ │ │ │ │ + this.anchor = (anchor != null) ? anchor : │ │ │ │ │ + { │ │ │ │ │ + size: new OpenLayers.Size(0, 0), │ │ │ │ │ + offset: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Fake event │ │ │ │ │ - oRequest.dispatchEvent({ │ │ │ │ │ - 'type': "readystatechange", │ │ │ │ │ - 'bubbles': false, │ │ │ │ │ - 'cancelable': false, │ │ │ │ │ - 'timeStamp': new Date + 0 │ │ │ │ │ - }); │ │ │ │ │ - }; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.anchor = null; │ │ │ │ │ + this.relativePosition = null; │ │ │ │ │ │ │ │ │ │ - function fGetDocument(oRequest) { │ │ │ │ │ - var oDocument = oRequest.responseXML, │ │ │ │ │ - sResponse = oRequest.responseText; │ │ │ │ │ - // Try parsing responseText │ │ │ │ │ - if (bIE && sResponse && oDocument && !oDocument.documentElement && oRequest.getResponseHeader("Content-Type").match(/[^\/]+\/[^\+]+\+xml/)) { │ │ │ │ │ - oDocument = new window.ActiveXObject("Microsoft.XMLDOM"); │ │ │ │ │ - oDocument.async = false; │ │ │ │ │ - oDocument.validateOnParse = false; │ │ │ │ │ - oDocument.loadXML(sResponse); │ │ │ │ │ - } │ │ │ │ │ - // Check if there is no error in document │ │ │ │ │ - if (oDocument) │ │ │ │ │ - if ((bIE && oDocument.parseError != 0) || !oDocument.documentElement || (oDocument.documentElement && oDocument.documentElement.tagName == "parsererror")) │ │ │ │ │ - return null; │ │ │ │ │ - return oDocument; │ │ │ │ │ - }; │ │ │ │ │ + OpenLayers.Popup.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - function fSynchronizeValues(oRequest) { │ │ │ │ │ - try { │ │ │ │ │ - oRequest.responseText = oRequest._object.responseText; │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - try { │ │ │ │ │ - oRequest.responseXML = fGetDocument(oRequest._object); │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - try { │ │ │ │ │ - oRequest.status = oRequest._object.status; │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - try { │ │ │ │ │ - oRequest.statusText = oRequest._object.statusText; │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - }; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: show │ │ │ │ │ + * Overridden from Popup since user might hide popup and then show() it │ │ │ │ │ + * in a new location (meaning we might want to update the relative │ │ │ │ │ + * position on the show) │ │ │ │ │ + */ │ │ │ │ │ + show: function() { │ │ │ │ │ + this.updatePosition(); │ │ │ │ │ + OpenLayers.Popup.prototype.show.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - function fCleanTransport(oRequest) { │ │ │ │ │ - // BUGFIX: IE - memory leak (on-page leak) │ │ │ │ │ - oRequest._object.onreadystatechange = new window.Function; │ │ │ │ │ - }; │ │ │ │ │ - /* │ │ │ │ │ - // Queue manager │ │ │ │ │ - var oQueuePending = {"CRITICAL":[],"HIGH":[],"NORMAL":[],"LOW":[],"LOWEST":[]}, │ │ │ │ │ - aQueueRunning = []; │ │ │ │ │ - function fQueue_add(oRequest) { │ │ │ │ │ - oQueuePending[oRequest.priority in oQueuePending ? oRequest.priority : "NORMAL"].push(oRequest); │ │ │ │ │ - // │ │ │ │ │ - setTimeout(fQueue_process); │ │ │ │ │ - }; │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * Since the popup is moving to a new px, it might need also to be moved │ │ │ │ │ + * relative to where the marker is. We first calculate the new │ │ │ │ │ + * relativePosition, and then we calculate the new px where we will │ │ │ │ │ + * put the popup, based on the new relative position. │ │ │ │ │ + * │ │ │ │ │ + * If the relativePosition has changed, we must also call │ │ │ │ │ + * updateRelativePosition() to make any visual changes to the popup │ │ │ │ │ + * which are associated with putting it in a new relativePosition. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ + */ │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + var oldRelativePosition = this.relativePosition; │ │ │ │ │ + this.relativePosition = this.calculateRelativePosition(px); │ │ │ │ │ │ │ │ │ │ - function fQueue_remove(oRequest) { │ │ │ │ │ - for (var nIndex = 0, bFound = false; nIndex < aQueueRunning.length; nIndex++) │ │ │ │ │ - if (bFound) │ │ │ │ │ - aQueueRunning[nIndex - 1] = aQueueRunning[nIndex]; │ │ │ │ │ - else │ │ │ │ │ - if (aQueueRunning[nIndex] == oRequest) │ │ │ │ │ - bFound = true; │ │ │ │ │ - if (bFound) │ │ │ │ │ - aQueueRunning.length--; │ │ │ │ │ - // │ │ │ │ │ - setTimeout(fQueue_process); │ │ │ │ │ - }; │ │ │ │ │ + OpenLayers.Popup.prototype.moveTo.call(this, this.calculateNewPx(px)); │ │ │ │ │ │ │ │ │ │ - function fQueue_process() { │ │ │ │ │ - if (aQueueRunning.length < 6) { │ │ │ │ │ - for (var sPriority in oQueuePending) { │ │ │ │ │ - if (oQueuePending[sPriority].length) { │ │ │ │ │ - var oRequest = oQueuePending[sPriority][0]; │ │ │ │ │ - oQueuePending[sPriority] = oQueuePending[sPriority].slice(1); │ │ │ │ │ - // │ │ │ │ │ - aQueueRunning.push(oRequest); │ │ │ │ │ - // Send request │ │ │ │ │ - fXMLHttpRequest_send(oRequest); │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + //if this move has caused the popup to change its relative position, │ │ │ │ │ + // we need to make the appropriate cosmetic changes. │ │ │ │ │ + if (this.relativePosition != oldRelativePosition) { │ │ │ │ │ + this.updateRelativePosition(); │ │ │ │ │ } │ │ │ │ │ - }; │ │ │ │ │ - */ │ │ │ │ │ - // Internet Explorer 5.0 (missing apply) │ │ │ │ │ - if (!window.Function.prototype.apply) { │ │ │ │ │ - window.Function.prototype.apply = function(oRequest, oArguments) { │ │ │ │ │ - if (!oArguments) │ │ │ │ │ - oArguments = []; │ │ │ │ │ - oRequest.__func = this; │ │ │ │ │ - oRequest.__func(oArguments[0], oArguments[1], oArguments[2], oArguments[3], oArguments[4]); │ │ │ │ │ - delete oRequest.__func; │ │ │ │ │ - }; │ │ │ │ │ - }; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Register new object with window │ │ │ │ │ - /** │ │ │ │ │ - * Class: OpenLayers.Request.XMLHttpRequest │ │ │ │ │ - * Standard-compliant (W3C) cross-browser implementation of the │ │ │ │ │ - * XMLHttpRequest object. From │ │ │ │ │ - * http://code.google.com/p/xmlhttprequest/. │ │ │ │ │ - */ │ │ │ │ │ - if (!OpenLayers.Request) { │ │ │ │ │ /** │ │ │ │ │ - * This allows for OpenLayers/Request.js to be included │ │ │ │ │ - * before or after this script. │ │ │ │ │ + * APIMethod: setSize │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * contentSize - {<OpenLayers.Size>} the new size for the popup's │ │ │ │ │ + * contents div (in pixels). │ │ │ │ │ */ │ │ │ │ │ - OpenLayers.Request = {}; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Request.XMLHttpRequest = cXMLHttpRequest; │ │ │ │ │ -})(); │ │ │ │ │ + setSize: function(contentSize) { │ │ │ │ │ + OpenLayers.Popup.prototype.setSize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + if ((this.lonlat) && (this.map)) { │ │ │ │ │ + var px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ + this.moveTo(px); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateRelativePosition │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The relative position ("br" "tr" "tl" "bl") at which the popup │ │ │ │ │ + * should be placed. │ │ │ │ │ + */ │ │ │ │ │ + calculateRelativePosition: function(px) { │ │ │ │ │ + var lonlat = this.map.getLonLatFromLayerPx(px); │ │ │ │ │ + │ │ │ │ │ + var extent = this.map.getExtent(); │ │ │ │ │ + var quadrant = extent.determineQuadrant(lonlat); │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Bounds.oppositeQuadrant(quadrant); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateRelativePosition │ │ │ │ │ + * The popup has been moved to a new relative location, so we may want to │ │ │ │ │ + * make some cosmetic adjustments to it. │ │ │ │ │ + * │ │ │ │ │ + * Note that in the classic Anchored popup, there is nothing to do │ │ │ │ │ + * here, since the popup looks exactly the same in all four positions. │ │ │ │ │ + * Subclasses such as Framed, however, will want to do something │ │ │ │ │ + * special here. │ │ │ │ │ + */ │ │ │ │ │ + updateRelativePosition: function() { │ │ │ │ │ + //to be overridden by subclasses │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateNewPx │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Pixel>} The the new px position of the popup on the screen │ │ │ │ │ + * relative to the passed-in px. │ │ │ │ │ + */ │ │ │ │ │ + calculateNewPx: function(px) { │ │ │ │ │ + var newPx = px.offset(this.anchor.offset); │ │ │ │ │ + │ │ │ │ │ + //use contentSize if size is not already set │ │ │ │ │ + var size = this.size || this.contentSize; │ │ │ │ │ + │ │ │ │ │ + var top = (this.relativePosition.charAt(0) == 't'); │ │ │ │ │ + newPx.y += (top) ? -size.h : this.anchor.size.h; │ │ │ │ │ + │ │ │ │ │ + var left = (this.relativePosition.charAt(1) == 'l'); │ │ │ │ │ + newPx.x += (left) ? -size.w : this.anchor.size.w; │ │ │ │ │ + │ │ │ │ │ + return newPx; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup.Anchored" │ │ │ │ │ + }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/HTTP.js │ │ │ │ │ + OpenLayers/Popup/Framed.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Protocol.js │ │ │ │ │ - * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * if application uses the query string, for example, for BBOX parameters, │ │ │ │ │ - * OpenLayers/Format/QueryStringFilter.js should be included in the build config file │ │ │ │ │ + * @requires OpenLayers/Popup/Anchored.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Protocol.HTTP │ │ │ │ │ - * A basic HTTP protocol for vector layers. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Protocol.HTTP> constructor. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Popup.Framed │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol> │ │ │ │ │ + * - <OpenLayers.Popup.Anchored> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.HTTP = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ +OpenLayers.Popup.Framed = │ │ │ │ │ + OpenLayers.Class(OpenLayers.Popup.Anchored, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: url │ │ │ │ │ - * {String} Service URL, read-only, set through the options │ │ │ │ │ - * passed to constructor. │ │ │ │ │ - */ │ │ │ │ │ - url: null, │ │ │ │ │ + /** │ │ │ │ │ + * Property: imageSrc │ │ │ │ │ + * {String} location of the image to be used as the popup frame │ │ │ │ │ + */ │ │ │ │ │ + imageSrc: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: headers │ │ │ │ │ - * {Object} HTTP request headers, read-only, set through the options │ │ │ │ │ - * passed to the constructor, │ │ │ │ │ - * Example: {'Content-Type': 'plain/text'} │ │ │ │ │ - */ │ │ │ │ │ - headers: null, │ │ │ │ │ + /** │ │ │ │ │ + * Property: imageSize │ │ │ │ │ + * {<OpenLayers.Size>} Size (measured in pixels) of the image located │ │ │ │ │ + * by the 'imageSrc' property. │ │ │ │ │ + */ │ │ │ │ │ + imageSize: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: params │ │ │ │ │ - * {Object} Parameters of GET requests, read-only, set through the options │ │ │ │ │ - * passed to the constructor, │ │ │ │ │ - * Example: {'bbox': '5,5,5,5'} │ │ │ │ │ - */ │ │ │ │ │ - params: null, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isAlphaImage │ │ │ │ │ + * {Boolean} The image has some alpha and thus needs to use the alpha │ │ │ │ │ + * image hack. Note that setting this to true will have no noticeable │ │ │ │ │ + * effect in FF or IE7 browsers, but will all but crush the ie6 │ │ │ │ │ + * browser. │ │ │ │ │ + * Default is false. │ │ │ │ │ + */ │ │ │ │ │ + isAlphaImage: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: callback │ │ │ │ │ - * {Object} Function to be called when the <read>, <create>, │ │ │ │ │ - * <update>, <delete> or <commit> operation completes, read-only, │ │ │ │ │ - * set through the options passed to the constructor. │ │ │ │ │ - */ │ │ │ │ │ - callback: null, │ │ │ │ │ + /** │ │ │ │ │ + * Property: positionBlocks │ │ │ │ │ + * {Object} Hash of different position blocks (Object/Hashs). Each block │ │ │ │ │ + * will be keyed by a two-character 'relativePosition' │ │ │ │ │ + * code string (ie "tl", "tr", "bl", "br"). Block properties are │ │ │ │ │ + * 'offset', 'padding' (self-explanatory), and finally the 'blocks' │ │ │ │ │ + * parameter, which is an array of the block objects. │ │ │ │ │ + * │ │ │ │ │ + * Each block object must have 'size', 'anchor', and 'position' │ │ │ │ │ + * properties. │ │ │ │ │ + * │ │ │ │ │ + * Note that positionBlocks should never be modified at runtime. │ │ │ │ │ + */ │ │ │ │ │ + positionBlocks: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: scope │ │ │ │ │ - * {Object} Callback execution scope, read-only, set through the │ │ │ │ │ - * options passed to the constructor. │ │ │ │ │ - */ │ │ │ │ │ - scope: null, │ │ │ │ │ + /** │ │ │ │ │ + * Property: blocks │ │ │ │ │ + * {Array[Object]} Array of objects, each of which is one "block" of the │ │ │ │ │ + * popup. Each block has a 'div' and an 'image' property, both of │ │ │ │ │ + * which are DOMElements, and the latter of which is appended to the │ │ │ │ │ + * former. These are reused as the popup goes changing positions for │ │ │ │ │ + * great economy and elegance. │ │ │ │ │ + */ │ │ │ │ │ + blocks: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: readWithPOST │ │ │ │ │ - * {Boolean} true if read operations are done with POST requests │ │ │ │ │ - * instead of GET, defaults to false. │ │ │ │ │ - */ │ │ │ │ │ - readWithPOST: false, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: fixedRelativePosition │ │ │ │ │ + * {Boolean} We want the framed popup to work dynamically placed relative │ │ │ │ │ + * to its anchor but also in just one fixed position. A well designed │ │ │ │ │ + * framed popup will have the pixels and logic to display itself in │ │ │ │ │ + * any of the four relative positions, but (understandably), this will │ │ │ │ │ + * not be the case for all of them. By setting this property to 'true', │ │ │ │ │ + * framed popup will not recalculate for the best placement each time │ │ │ │ │ + * it's open, but will always open the same way. │ │ │ │ │ + * Note that if this is set to true, it is generally advisable to also │ │ │ │ │ + * set the 'panIntoView' property to true so that the popup can be │ │ │ │ │ + * scrolled into view (since it will often be offscreen on open) │ │ │ │ │ + * Default is false. │ │ │ │ │ + */ │ │ │ │ │ + fixedRelativePosition: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: updateWithPOST │ │ │ │ │ - * {Boolean} true if update operations are done with POST requests │ │ │ │ │ - * defaults to false. │ │ │ │ │ - */ │ │ │ │ │ - updateWithPOST: false, │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Popup.Framed │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ + * contentSize - {<OpenLayers.Size>} │ │ │ │ │ + * contentHTML - {String} │ │ │ │ │ + * anchor - {Object} Object to which we'll anchor the popup. Must expose │ │ │ │ │ + * a 'size' (<OpenLayers.Size>) and 'offset' (<OpenLayers.Pixel>) │ │ │ │ │ + * (Note that this is generally an <OpenLayers.Icon>). │ │ │ │ │ + * closeBox - {Boolean} │ │ │ │ │ + * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, │ │ │ │ │ + closeBoxCallback) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: deleteWithPOST │ │ │ │ │ - * {Boolean} true if delete operations are done with POST requests │ │ │ │ │ - * defaults to false. │ │ │ │ │ - * if true, POST data is set to output of format.write(). │ │ │ │ │ - */ │ │ │ │ │ - deleteWithPOST: false, │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.initialize.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: wildcarded. │ │ │ │ │ - * {Boolean} If true percent signs are added around values │ │ │ │ │ - * read from LIKE filters, for example if the protocol │ │ │ │ │ - * read method is passed a LIKE filter whose property │ │ │ │ │ - * is "foo" and whose value is "bar" the string │ │ │ │ │ - * "foo__ilike=%bar%" will be sent in the query string; │ │ │ │ │ - * defaults to false. │ │ │ │ │ - */ │ │ │ │ │ - wildcarded: false, │ │ │ │ │ + if (this.fixedRelativePosition) { │ │ │ │ │ + //based on our decided relativePostion, set the current padding │ │ │ │ │ + // this keeps us from getting into trouble │ │ │ │ │ + this.updateRelativePosition(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: srsInBBOX │ │ │ │ │ - * {Boolean} Include the SRS identifier in BBOX query string parameter. │ │ │ │ │ - * Default is false. If true and the layer has a projection object set, │ │ │ │ │ - * any BBOX filter will be serialized with a fifth item identifying the │ │ │ │ │ - * projection. E.g. bbox=-1000,-1000,1000,1000,EPSG:900913 │ │ │ │ │ - */ │ │ │ │ │ - srsInBBOX: false, │ │ │ │ │ + //make calculateRelativePosition always return the specified │ │ │ │ │ + // fixed position. │ │ │ │ │ + this.calculateRelativePosition = function(px) { │ │ │ │ │ + return this.relativePosition; │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.HTTP │ │ │ │ │ - * A class for giving layers generic HTTP protocol. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - * │ │ │ │ │ - * Valid options include: │ │ │ │ │ - * url - {String} │ │ │ │ │ - * headers - {Object} │ │ │ │ │ - * params - {Object} URL parameters for GET requests │ │ │ │ │ - * format - {<OpenLayers.Format>} │ │ │ │ │ - * callback - {Function} │ │ │ │ │ - * scope - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - this.params = {}; │ │ │ │ │ - this.headers = {}; │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.contentDiv.style.position = "absolute"; │ │ │ │ │ + this.contentDiv.style.zIndex = 1; │ │ │ │ │ │ │ │ │ │ - if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ - var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ - wildcarded: this.wildcarded, │ │ │ │ │ - srsInBBOX: this.srsInBBOX │ │ │ │ │ - }); │ │ │ │ │ - this.filterToParams = function(filter, params) { │ │ │ │ │ - return format.write(filter, params); │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + if (closeBox) { │ │ │ │ │ + this.closeDiv.style.zIndex = 1; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up the protocol. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.params = null; │ │ │ │ │ - this.headers = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ - }, │ │ │ │ │ + this.groupDiv.style.position = "absolute"; │ │ │ │ │ + this.groupDiv.style.top = "0px"; │ │ │ │ │ + this.groupDiv.style.left = "0px"; │ │ │ │ │ + this.groupDiv.style.height = "100%"; │ │ │ │ │ + this.groupDiv.style.width = "100%"; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: filterToParams │ │ │ │ │ - * Optional method to translate an <OpenLayers.Filter> object into an object │ │ │ │ │ - * that can be serialized as request query string provided. If a custom │ │ │ │ │ - * method is not provided, the filter will be serialized using the │ │ │ │ │ - * <OpenLayers.Format.QueryStringFilter> class. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} filter to convert. │ │ │ │ │ - * params - {Object} The parameters object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The resulting parameters object. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.imageSrc = null; │ │ │ │ │ + this.imageSize = null; │ │ │ │ │ + this.isAlphaImage = null; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Construct a request for reading new features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * url - {String} Url for the request. │ │ │ │ │ - * params - {Object} Parameters to get serialized as a query string. │ │ │ │ │ - * headers - {Object} Headers to be set on the request. │ │ │ │ │ - * filter - {<OpenLayers.Filter>} Filter to get serialized as a │ │ │ │ │ - * query string. │ │ │ │ │ - * readWithPOST - {Boolean} If the request should be done with POST. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} A response object, whose "priv" property │ │ │ │ │ - * references the HTTP request, this object is also passed to the │ │ │ │ │ - * callback function when the request completes, its "features" property │ │ │ │ │ - * is then populated with the features received from the server. │ │ │ │ │ - */ │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options.params, this.options.params); │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - if (options.filter && this.filterToParams) { │ │ │ │ │ - options.params = this.filterToParams( │ │ │ │ │ - options.filter, options.params │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - var readWithPOST = (options.readWithPOST !== undefined) ? │ │ │ │ │ - options.readWithPOST : this.readWithPOST; │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - if (readWithPOST) { │ │ │ │ │ - var headers = options.headers || {}; │ │ │ │ │ - headers["Content-Type"] = "application/x-www-form-urlencoded"; │ │ │ │ │ - resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ - data: OpenLayers.Util.getParameterString(options.params), │ │ │ │ │ - headers: headers │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - resp.priv = OpenLayers.Request.GET({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return resp; │ │ │ │ │ - }, │ │ │ │ │ + this.fixedRelativePosition = false; │ │ │ │ │ + this.positionBlocks = null; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleRead │ │ │ │ │ - * Individual callbacks are created for read, create and update, should │ │ │ │ │ - * a subclass need to override each one separately. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * the user callback. │ │ │ │ │ - * options - {Object} The user options passed to the read call. │ │ │ │ │ - */ │ │ │ │ │ - handleRead: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options); │ │ │ │ │ - }, │ │ │ │ │ + //remove our blocks │ │ │ │ │ + for (var i = 0; i < this.blocks.length; i++) { │ │ │ │ │ + var block = this.blocks[i]; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: create │ │ │ │ │ - * Construct a request for writing newly created features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, whose "priv" property references the HTTP request, this │ │ │ │ │ - * object is also passed to the callback function when the request │ │ │ │ │ - * completes, its "features" property is then populated with the │ │ │ │ │ - * the features received from the server. │ │ │ │ │ - */ │ │ │ │ │ - create: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + if (block.image) { │ │ │ │ │ + block.div.removeChild(block.image); │ │ │ │ │ + } │ │ │ │ │ + block.image = null; │ │ │ │ │ │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: features, │ │ │ │ │ - requestType: "create" │ │ │ │ │ - }); │ │ │ │ │ + if (block.div) { │ │ │ │ │ + this.groupDiv.removeChild(block.div); │ │ │ │ │ + } │ │ │ │ │ + block.div = null; │ │ │ │ │ + } │ │ │ │ │ + this.blocks = null; │ │ │ │ │ │ │ │ │ │ - resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleCreate, resp, options), │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(features) │ │ │ │ │ - }); │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return resp; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setBackgroundColor │ │ │ │ │ + */ │ │ │ │ │ + setBackgroundColor: function(color) { │ │ │ │ │ + //does nothing since the framed popup's entire scheme is based on a │ │ │ │ │ + // an image -- changing the background color makes no sense. │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleCreate │ │ │ │ │ - * Called the the request issued by <create> is complete. May be overridden │ │ │ │ │ - * by subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the create call. │ │ │ │ │ - */ │ │ │ │ │ - handleCreate: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setBorder │ │ │ │ │ + */ │ │ │ │ │ + setBorder: function() { │ │ │ │ │ + //does nothing since the framed popup's entire scheme is based on a │ │ │ │ │ + // an image -- changing the popup's border makes no sense. │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: update │ │ │ │ │ - * Construct a request updating modified feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, whose "priv" property references the HTTP request, this │ │ │ │ │ - * object is also passed to the callback function when the request │ │ │ │ │ - * completes, its "features" property is then populated with the │ │ │ │ │ - * the feature received from the server. │ │ │ │ │ - */ │ │ │ │ │ - update: function(feature, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var url = options.url || │ │ │ │ │ - feature.url || │ │ │ │ │ - this.options.url + "/" + feature.fid; │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + /** │ │ │ │ │ + * Method: setOpacity │ │ │ │ │ + * Sets the opacity of the popup. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * opacity - {float} A value between 0.0 (transparent) and 1.0 (solid). │ │ │ │ │ + */ │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + //does nothing since we suppose that we'll never apply an opacity │ │ │ │ │ + // to a framed popup │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: feature, │ │ │ │ │ - requestType: "update" │ │ │ │ │ - }); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setSize │ │ │ │ │ + * Overridden here, because we need to update the blocks whenever the size │ │ │ │ │ + * of the popup has changed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * contentSize - {<OpenLayers.Size>} the new size for the popup's │ │ │ │ │ + * contents div (in pixels). │ │ │ │ │ + */ │ │ │ │ │ + setSize: function(contentSize) { │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.setSize.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - var method = this.updateWithPOST ? "POST" : "PUT"; │ │ │ │ │ - resp.priv = OpenLayers.Request[method]({ │ │ │ │ │ - url: url, │ │ │ │ │ - callback: this.createCallback(this.handleUpdate, resp, options), │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(feature) │ │ │ │ │ - }); │ │ │ │ │ + this.updateBlocks(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return resp; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateRelativePosition │ │ │ │ │ + * When the relative position changes, we need to set the new padding │ │ │ │ │ + * BBOX on the popup, reposition the close div, and update the blocks. │ │ │ │ │ + */ │ │ │ │ │ + updateRelativePosition: function() { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleUpdate │ │ │ │ │ - * Called the the request issued by <update> is complete. May be overridden │ │ │ │ │ - * by subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the update call. │ │ │ │ │ - */ │ │ │ │ │ - handleUpdate: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options); │ │ │ │ │ - }, │ │ │ │ │ + //update the padding │ │ │ │ │ + this.padding = this.positionBlocks[this.relativePosition].padding; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: delete │ │ │ │ │ - * Construct a request deleting a removed feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, whose "priv" property references the HTTP request, this │ │ │ │ │ - * object is also passed to the callback function when the request │ │ │ │ │ - * completes. │ │ │ │ │ - */ │ │ │ │ │ - "delete": function(feature, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var url = options.url || │ │ │ │ │ - feature.url || │ │ │ │ │ - this.options.url + "/" + feature.fid; │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + //update the position of our close box to new padding │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + // use the content div's css padding to determine if we should │ │ │ │ │ + // padd the close div │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: feature, │ │ │ │ │ - requestType: "delete" │ │ │ │ │ - }); │ │ │ │ │ + this.closeDiv.style.right = contentDivPadding.right + │ │ │ │ │ + this.padding.right + "px"; │ │ │ │ │ + this.closeDiv.style.top = contentDivPadding.top + │ │ │ │ │ + this.padding.top + "px"; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var method = this.deleteWithPOST ? "POST" : "DELETE"; │ │ │ │ │ - var requestOptions = { │ │ │ │ │ - url: url, │ │ │ │ │ - callback: this.createCallback(this.handleDelete, resp, options), │ │ │ │ │ - headers: options.headers │ │ │ │ │ - }; │ │ │ │ │ - if (this.deleteWithPOST) { │ │ │ │ │ - requestOptions.data = this.format.write(feature); │ │ │ │ │ - } │ │ │ │ │ - resp.priv = OpenLayers.Request[method](requestOptions); │ │ │ │ │ + this.updateBlocks(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return resp; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateNewPx │ │ │ │ │ + * Besides the standard offset as determined by the Anchored class, our │ │ │ │ │ + * Framed popups have a special 'offset' property for each of their │ │ │ │ │ + * positions, which is used to offset the popup relative to its anchor. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Pixel>} The the new px position of the popup on the screen │ │ │ │ │ + * relative to the passed-in px. │ │ │ │ │ + */ │ │ │ │ │ + calculateNewPx: function(px) { │ │ │ │ │ + var newPx = OpenLayers.Popup.Anchored.prototype.calculateNewPx.apply( │ │ │ │ │ + this, arguments │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleDelete │ │ │ │ │ - * Called the the request issued by <delete> is complete. May be overridden │ │ │ │ │ - * by subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the delete call. │ │ │ │ │ - */ │ │ │ │ │ - handleDelete: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options); │ │ │ │ │ - }, │ │ │ │ │ + newPx = newPx.offset(this.positionBlocks[this.relativePosition].offset); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleResponse │ │ │ │ │ - * Called by CRUD specific handlers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the create, read, update, │ │ │ │ │ - * or delete call. │ │ │ │ │ - */ │ │ │ │ │ - handleResponse: function(resp, options) { │ │ │ │ │ - var request = resp.priv; │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - // success │ │ │ │ │ - if (resp.requestType != "delete") { │ │ │ │ │ - resp.features = this.parseFeatures(request); │ │ │ │ │ - } │ │ │ │ │ - resp.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ - } else { │ │ │ │ │ - // failure │ │ │ │ │ - resp.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + return newPx; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createBlocks │ │ │ │ │ + */ │ │ │ │ │ + createBlocks: function() { │ │ │ │ │ + this.blocks = []; │ │ │ │ │ + │ │ │ │ │ + //since all positions contain the same number of blocks, we can │ │ │ │ │ + // just pick the first position and use its blocks array to create │ │ │ │ │ + // our blocks array │ │ │ │ │ + var firstPosition = null; │ │ │ │ │ + for (var key in this.positionBlocks) { │ │ │ │ │ + firstPosition = key; │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ - options.callback.call(options.scope, resp); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseFeatures │ │ │ │ │ - * Read HTTP response body and return features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * request - {XMLHttpRequest} The request object │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} Array of features or a single feature. │ │ │ │ │ - */ │ │ │ │ │ - parseFeatures: function(request) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - return this.format.read(doc); │ │ │ │ │ - }, │ │ │ │ │ + var position = this.positionBlocks[firstPosition]; │ │ │ │ │ + for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: commit │ │ │ │ │ - * Iterate over each feature and take action based on the feature state. │ │ │ │ │ - * Possible actions are create, update and delete. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array({<OpenLayers.Feature.Vector>})} │ │ │ │ │ - * options - {Object} Optional object for setting up intermediate commit │ │ │ │ │ - * callbacks. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * create - {Object} Optional object to be passed to the <create> method. │ │ │ │ │ - * update - {Object} Optional object to be passed to the <update> method. │ │ │ │ │ - * delete - {Object} Optional object to be passed to the <delete> method. │ │ │ │ │ - * callback - {Function} Optional function to be called when the commit │ │ │ │ │ - * is complete. │ │ │ │ │ - * scope - {Object} Optional object to be set as the scope of the callback. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Protocol.Response>)} An array of response objects, │ │ │ │ │ - * one per request made to the server, each object's "priv" property │ │ │ │ │ - * references the corresponding HTTP request. │ │ │ │ │ - */ │ │ │ │ │ - commit: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = [], │ │ │ │ │ - nResponses = 0; │ │ │ │ │ + var block = {}; │ │ │ │ │ + this.blocks.push(block); │ │ │ │ │ │ │ │ │ │ - // Divide up features before issuing any requests. This properly │ │ │ │ │ - // counts requests in the event that any responses come in before │ │ │ │ │ - // all requests have been issued. │ │ │ │ │ - var types = {}; │ │ │ │ │ - types[OpenLayers.State.INSERT] = []; │ │ │ │ │ - types[OpenLayers.State.UPDATE] = []; │ │ │ │ │ - types[OpenLayers.State.DELETE] = []; │ │ │ │ │ - var feature, list, requestFeatures = []; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - list = types[feature.state]; │ │ │ │ │ - if (list) { │ │ │ │ │ - list.push(feature); │ │ │ │ │ - requestFeatures.push(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // tally up number of requests │ │ │ │ │ - var nRequests = (types[OpenLayers.State.INSERT].length > 0 ? 1 : 0) + │ │ │ │ │ - types[OpenLayers.State.UPDATE].length + │ │ │ │ │ - types[OpenLayers.State.DELETE].length; │ │ │ │ │ + var divId = this.id + '_FrameDecorationDiv_' + i; │ │ │ │ │ + block.div = OpenLayers.Util.createDiv(divId, │ │ │ │ │ + null, null, null, "absolute", null, "hidden", null │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - // This response will be sent to the final callback after all the others │ │ │ │ │ - // have been fired. │ │ │ │ │ - var success = true; │ │ │ │ │ - var finalResponse = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: requestFeatures │ │ │ │ │ - }); │ │ │ │ │ + var imgId = this.id + '_FrameDecorationImg_' + i; │ │ │ │ │ + var imageCreator = │ │ │ │ │ + (this.isAlphaImage) ? OpenLayers.Util.createAlphaImageDiv : │ │ │ │ │ + OpenLayers.Util.createImage; │ │ │ │ │ │ │ │ │ │ - function insertCallback(response) { │ │ │ │ │ - var len = response.features ? response.features.length : 0; │ │ │ │ │ - var fids = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - fids[i] = response.features[i].fid; │ │ │ │ │ + block.image = imageCreator(imgId, │ │ │ │ │ + null, this.imageSize, this.imageSrc, │ │ │ │ │ + "absolute", null, null, null │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + block.div.appendChild(block.image); │ │ │ │ │ + this.groupDiv.appendChild(block.div); │ │ │ │ │ } │ │ │ │ │ - finalResponse.insertIds = fids; │ │ │ │ │ - callback.apply(this, [response]); │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - function callback(response) { │ │ │ │ │ - this.callUserCallback(response, options); │ │ │ │ │ - success = success && response.success(); │ │ │ │ │ - nResponses++; │ │ │ │ │ - if (nResponses >= nRequests) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - finalResponse.code = success ? │ │ │ │ │ - OpenLayers.Protocol.Response.SUCCESS : │ │ │ │ │ - OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - options.callback.apply(options.scope, [finalResponse]); │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateBlocks │ │ │ │ │ + * Internal method, called on initialize and when the popup's relative │ │ │ │ │ + * position has changed. This function takes care of re-positioning │ │ │ │ │ + * the popup's blocks in their appropropriate places. │ │ │ │ │ + */ │ │ │ │ │ + updateBlocks: function() { │ │ │ │ │ + if (!this.blocks) { │ │ │ │ │ + this.createBlocks(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.size && this.relativePosition) { │ │ │ │ │ + var position = this.positionBlocks[this.relativePosition]; │ │ │ │ │ + for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ + │ │ │ │ │ + var positionBlock = position.blocks[i]; │ │ │ │ │ + var block = this.blocks[i]; │ │ │ │ │ + │ │ │ │ │ + // adjust sizes │ │ │ │ │ + var l = positionBlock.anchor.left; │ │ │ │ │ + var b = positionBlock.anchor.bottom; │ │ │ │ │ + var r = positionBlock.anchor.right; │ │ │ │ │ + var t = positionBlock.anchor.top; │ │ │ │ │ + │ │ │ │ │ + //note that we use the isNaN() test here because if the │ │ │ │ │ + // size object is initialized with a "auto" parameter, the │ │ │ │ │ + // size constructor calls parseFloat() on the string, │ │ │ │ │ + // which will turn it into NaN │ │ │ │ │ + // │ │ │ │ │ + var w = (isNaN(positionBlock.size.w)) ? this.size.w - (r + l) : │ │ │ │ │ + positionBlock.size.w; │ │ │ │ │ + │ │ │ │ │ + var h = (isNaN(positionBlock.size.h)) ? this.size.h - (b + t) : │ │ │ │ │ + positionBlock.size.h; │ │ │ │ │ + │ │ │ │ │ + block.div.style.width = (w < 0 ? 0 : w) + 'px'; │ │ │ │ │ + block.div.style.height = (h < 0 ? 0 : h) + 'px'; │ │ │ │ │ + │ │ │ │ │ + block.div.style.left = (l != null) ? l + 'px' : ''; │ │ │ │ │ + block.div.style.bottom = (b != null) ? b + 'px' : ''; │ │ │ │ │ + block.div.style.right = (r != null) ? r + 'px' : ''; │ │ │ │ │ + block.div.style.top = (t != null) ? t + 'px' : ''; │ │ │ │ │ + │ │ │ │ │ + block.image.style.left = positionBlock.position.x + 'px'; │ │ │ │ │ + block.image.style.top = positionBlock.position.y + 'px'; │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + this.contentDiv.style.left = this.padding.left + "px"; │ │ │ │ │ + this.contentDiv.style.top = this.padding.top + "px"; │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // start issuing requests │ │ │ │ │ - var queue = types[OpenLayers.State.INSERT]; │ │ │ │ │ - if (queue.length > 0) { │ │ │ │ │ - resp.push(this.create( │ │ │ │ │ - queue, OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: insertCallback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options.create) │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - queue = types[OpenLayers.State.UPDATE]; │ │ │ │ │ - for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ - resp.push(this.update( │ │ │ │ │ - queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: callback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options.update))); │ │ │ │ │ - } │ │ │ │ │ - queue = types[OpenLayers.State.DELETE]; │ │ │ │ │ - for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ - resp.push(this["delete"]( │ │ │ │ │ - queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: callback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options["delete"]))); │ │ │ │ │ - } │ │ │ │ │ - return resp; │ │ │ │ │ - }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup.Framed" │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Popup/FramedCloud.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: abort │ │ │ │ │ - * Abort an ongoing request, the response object passed to │ │ │ │ │ - * this method must come from this HTTP protocol (as a result │ │ │ │ │ - * of a create, read, update, delete or commit operation). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ - */ │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - response.priv.abort(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: callUserCallback │ │ │ │ │ - * This method is used from within the commit method each time an │ │ │ │ │ - * an HTTP response is received from the server, it is responsible │ │ │ │ │ - * for calling the user-supplied callbacks. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} │ │ │ │ │ - * options - {Object} The map of options passed to the commit call. │ │ │ │ │ - */ │ │ │ │ │ - callUserCallback: function(resp, options) { │ │ │ │ │ - var opt = options[resp.requestType]; │ │ │ │ │ - if (opt && opt.callback) { │ │ │ │ │ - opt.callback.call(opt.scope, resp); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Popup/Framed.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Bounds.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Pixel.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Size.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.HTTP" │ │ │ │ │ -}); │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Popup.FramedCloud │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Popup.Framed> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Popup.FramedCloud = │ │ │ │ │ + OpenLayers.Class(OpenLayers.Popup.Framed, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: contentDisplayClass │ │ │ │ │ + * {String} The CSS class of the popup content div. │ │ │ │ │ + */ │ │ │ │ │ + contentDisplayClass: "olFramedCloudPopupContent", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: autoSize │ │ │ │ │ + * {Boolean} Framed Cloud is autosizing by default. │ │ │ │ │ + */ │ │ │ │ │ + autoSize: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: panMapIfOutOfView │ │ │ │ │ + * {Boolean} Framed Cloud does pan into view by default. │ │ │ │ │ + */ │ │ │ │ │ + panMapIfOutOfView: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: imageSize │ │ │ │ │ + * {<OpenLayers.Size>} │ │ │ │ │ + */ │ │ │ │ │ + imageSize: new OpenLayers.Size(1276, 736), │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isAlphaImage │ │ │ │ │ + * {Boolean} The FramedCloud does not use an alpha image (in honor of the │ │ │ │ │ + * good ie6 folk out there) │ │ │ │ │ + */ │ │ │ │ │ + isAlphaImage: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: fixedRelativePosition │ │ │ │ │ + * {Boolean} The Framed Cloud popup works in just one fixed position. │ │ │ │ │ + */ │ │ │ │ │ + fixedRelativePosition: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: positionBlocks │ │ │ │ │ + * {Object} Hash of differen position blocks, keyed by relativePosition │ │ │ │ │ + * two-character code string (ie "tl", "tr", "bl", "br") │ │ │ │ │ + */ │ │ │ │ │ + positionBlocks: { │ │ │ │ │ + "tl": { │ │ │ │ │ + 'offset': new OpenLayers.Pixel(44, 0), │ │ │ │ │ + 'padding': new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ + 'blocks': [{ // top-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { //top-right │ │ │ │ │ + size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { //bottom-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ + }, { //bottom-right │ │ │ │ │ + size: new OpenLayers.Size(22, 18), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -632) │ │ │ │ │ + }, { // stem │ │ │ │ │ + size: new OpenLayers.Size(81, 35), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -688) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + "tr": { │ │ │ │ │ + 'offset': new OpenLayers.Pixel(-45, 0), │ │ │ │ │ + 'padding': new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ + 'blocks': [{ // top-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { //top-right │ │ │ │ │ + size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { //bottom-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ + }, { //bottom-right │ │ │ │ │ + size: new OpenLayers.Size(22, 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -631) │ │ │ │ │ + }, { // stem │ │ │ │ │ + size: new OpenLayers.Size(81, 35), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, null, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-215, -687) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + "bl": { │ │ │ │ │ + 'offset': new OpenLayers.Pixel(45, 0), │ │ │ │ │ + 'padding': new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ + 'blocks': [{ // top-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { //top-right │ │ │ │ │ + size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { //bottom-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ + }, { //bottom-right │ │ │ │ │ + size: new OpenLayers.Size(22, 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ + }, { // stem │ │ │ │ │ + size: new OpenLayers.Size(81, 33), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, null, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-101, -674) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + "br": { │ │ │ │ │ + 'offset': new OpenLayers.Pixel(-44, 0), │ │ │ │ │ + 'padding': new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ + 'blocks': [{ // top-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { //top-right │ │ │ │ │ + size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { //bottom-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ + }, { //bottom-right │ │ │ │ │ + size: new OpenLayers.Size(22, 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ + }, { // stem │ │ │ │ │ + size: new OpenLayers.Size(81, 33), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, null, null, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-311, -674) │ │ │ │ │ + }] │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: minSize │ │ │ │ │ + * {<OpenLayers.Size>} │ │ │ │ │ + */ │ │ │ │ │ + minSize: new OpenLayers.Size(105, 10), │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxSize │ │ │ │ │ + * {<OpenLayers.Size>} │ │ │ │ │ + */ │ │ │ │ │ + maxSize: new OpenLayers.Size(1200, 660), │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Popup.FramedCloud │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ + * contentSize - {<OpenLayers.Size>} │ │ │ │ │ + * contentHTML - {String} │ │ │ │ │ + * anchor - {Object} Object to which we'll anchor the popup. Must expose │ │ │ │ │ + * a 'size' (<OpenLayers.Size>) and 'offset' (<OpenLayers.Pixel>) │ │ │ │ │ + * (Note that this is generally an <OpenLayers.Icon>). │ │ │ │ │ + * closeBox - {Boolean} │ │ │ │ │ + * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, │ │ │ │ │ + closeBoxCallback) { │ │ │ │ │ + │ │ │ │ │ + this.imageSrc = OpenLayers.Util.getImageLocation('cloud-popup-relative.png'); │ │ │ │ │ + OpenLayers.Popup.Framed.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.contentDiv.className = this.contentDisplayClass; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup.FramedCloud" │ │ │ │ │ + }); │ │ │ ├── ./usr/share/javascript/openlayers/OpenLayers.light.min.js │ │ │ │ ├── js-beautify {} │ │ │ │ │ @@ -62,204 +62,14 @@ │ │ │ │ │ var sourceIsEvt = typeof window.Event == "function" && source instanceof window.Event; │ │ │ │ │ if (!sourceIsEvt && source.hasOwnProperty && source.hasOwnProperty("toString")) { │ │ │ │ │ destination.toString = source.toString │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ return destination │ │ │ │ │ }; │ │ │ │ │ -OpenLayers.Util = OpenLayers.Util || {}; │ │ │ │ │ -OpenLayers.Util.vendorPrefix = function() { │ │ │ │ │ - "use strict"; │ │ │ │ │ - var VENDOR_PREFIXES = ["", "O", "ms", "Moz", "Webkit"], │ │ │ │ │ - divStyle = document.createElement("div").style, │ │ │ │ │ - cssCache = {}, │ │ │ │ │ - jsCache = {}; │ │ │ │ │ - │ │ │ │ │ - function domToCss(prefixedDom) { │ │ │ │ │ - if (!prefixedDom) { │ │ │ │ │ - return null │ │ │ │ │ - } │ │ │ │ │ - return prefixedDom.replace(/([A-Z])/g, function(c) { │ │ │ │ │ - return "-" + c.toLowerCase() │ │ │ │ │ - }).replace(/^ms-/, "-ms-") │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function css(property) { │ │ │ │ │ - if (cssCache[property] === undefined) { │ │ │ │ │ - var domProperty = property.replace(/(-[\s\S])/g, function(c) { │ │ │ │ │ - return c.charAt(1).toUpperCase() │ │ │ │ │ - }); │ │ │ │ │ - var prefixedDom = style(domProperty); │ │ │ │ │ - cssCache[property] = domToCss(prefixedDom) │ │ │ │ │ - } │ │ │ │ │ - return cssCache[property] │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function js(obj, property) { │ │ │ │ │ - if (jsCache[property] === undefined) { │ │ │ │ │ - var tmpProp, i = 0, │ │ │ │ │ - l = VENDOR_PREFIXES.length, │ │ │ │ │ - prefix, isStyleObj = typeof obj.cssText !== "undefined"; │ │ │ │ │ - jsCache[property] = null; │ │ │ │ │ - for (; i < l; i++) { │ │ │ │ │ - prefix = VENDOR_PREFIXES[i]; │ │ │ │ │ - if (prefix) { │ │ │ │ │ - if (!isStyleObj) { │ │ │ │ │ - prefix = prefix.toLowerCase() │ │ │ │ │ - } │ │ │ │ │ - tmpProp = prefix + property.charAt(0).toUpperCase() + property.slice(1) │ │ │ │ │ - } else { │ │ │ │ │ - tmpProp = property │ │ │ │ │ - } │ │ │ │ │ - if (obj[tmpProp] !== undefined) { │ │ │ │ │ - jsCache[property] = tmpProp; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return jsCache[property] │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function style(property) { │ │ │ │ │ - return js(divStyle, property) │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - css: css, │ │ │ │ │ - js: js, │ │ │ │ │ - style: style, │ │ │ │ │ - cssCache: cssCache, │ │ │ │ │ - jsCache: jsCache │ │ │ │ │ - } │ │ │ │ │ -}(); │ │ │ │ │ -OpenLayers.Animation = function(window) { │ │ │ │ │ - var requestAnimationFrame = OpenLayers.Util.vendorPrefix.js(window, "requestAnimationFrame"); │ │ │ │ │ - var isNative = !!requestAnimationFrame; │ │ │ │ │ - var requestFrame = function() { │ │ │ │ │ - var request = window[requestAnimationFrame] || function(callback, element) { │ │ │ │ │ - window.setTimeout(callback, 16) │ │ │ │ │ - }; │ │ │ │ │ - return function(callback, element) { │ │ │ │ │ - request.apply(window, [callback, element]) │ │ │ │ │ - } │ │ │ │ │ - }(); │ │ │ │ │ - var counter = 0; │ │ │ │ │ - var loops = {}; │ │ │ │ │ - │ │ │ │ │ - function start(callback, duration, element) { │ │ │ │ │ - duration = duration > 0 ? duration : Number.POSITIVE_INFINITY; │ │ │ │ │ - var id = ++counter; │ │ │ │ │ - var start = +new Date; │ │ │ │ │ - loops[id] = function() { │ │ │ │ │ - if (loops[id] && +new Date - start <= duration) { │ │ │ │ │ - callback(); │ │ │ │ │ - if (loops[id]) { │ │ │ │ │ - requestFrame(loops[id], element) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - delete loops[id] │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - requestFrame(loops[id], element); │ │ │ │ │ - return id │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function stop(id) { │ │ │ │ │ - delete loops[id] │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - isNative: isNative, │ │ │ │ │ - requestFrame: requestFrame, │ │ │ │ │ - start: start, │ │ │ │ │ - stop: stop │ │ │ │ │ - } │ │ │ │ │ -}(window); │ │ │ │ │ -OpenLayers.Kinetic = OpenLayers.Class({ │ │ │ │ │ - threshold: 0, │ │ │ │ │ - deceleration: .0035, │ │ │ │ │ - nbPoints: 100, │ │ │ │ │ - delay: 200, │ │ │ │ │ - points: undefined, │ │ │ │ │ - timerId: undefined, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options) │ │ │ │ │ - }, │ │ │ │ │ - begin: function() { │ │ │ │ │ - OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ - this.timerId = undefined; │ │ │ │ │ - this.points = [] │ │ │ │ │ - }, │ │ │ │ │ - update: function(xy) { │ │ │ │ │ - this.points.unshift({ │ │ │ │ │ - xy: xy, │ │ │ │ │ - tick: (new Date).getTime() │ │ │ │ │ - }); │ │ │ │ │ - if (this.points.length > this.nbPoints) { │ │ │ │ │ - this.points.pop() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - end: function(xy) { │ │ │ │ │ - var last, now = (new Date).getTime(); │ │ │ │ │ - for (var i = 0, l = this.points.length, point; i < l; i++) { │ │ │ │ │ - point = this.points[i]; │ │ │ │ │ - if (now - point.tick > this.delay) { │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - last = point │ │ │ │ │ - } │ │ │ │ │ - if (!last) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var time = (new Date).getTime() - last.tick; │ │ │ │ │ - var dist = Math.sqrt(Math.pow(xy.x - last.xy.x, 2) + Math.pow(xy.y - last.xy.y, 2)); │ │ │ │ │ - var speed = dist / time; │ │ │ │ │ - if (speed == 0 || speed < this.threshold) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var theta = Math.asin((xy.y - last.xy.y) / dist); │ │ │ │ │ - if (last.xy.x <= xy.x) { │ │ │ │ │ - theta = Math.PI - theta │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - speed: speed, │ │ │ │ │ - theta: theta │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - move: function(info, callback) { │ │ │ │ │ - var v0 = info.speed; │ │ │ │ │ - var fx = Math.cos(info.theta); │ │ │ │ │ - var fy = -Math.sin(info.theta); │ │ │ │ │ - var initialTime = (new Date).getTime(); │ │ │ │ │ - var lastX = 0; │ │ │ │ │ - var lastY = 0; │ │ │ │ │ - var timerCallback = function() { │ │ │ │ │ - if (this.timerId == null) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var t = (new Date).getTime() - initialTime; │ │ │ │ │ - var p = -this.deceleration * Math.pow(t, 2) / 2 + v0 * t; │ │ │ │ │ - var x = p * fx; │ │ │ │ │ - var y = p * fy; │ │ │ │ │ - var args = {}; │ │ │ │ │ - args.end = false; │ │ │ │ │ - var v = -this.deceleration * t + v0; │ │ │ │ │ - if (v <= 0) { │ │ │ │ │ - OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - args.end = true │ │ │ │ │ - } │ │ │ │ │ - args.x = x - lastX; │ │ │ │ │ - args.y = y - lastY; │ │ │ │ │ - lastX = x; │ │ │ │ │ - lastY = y; │ │ │ │ │ - callback(args.x, args.y, args.end) │ │ │ │ │ - }; │ │ │ │ │ - this.timerId = OpenLayers.Animation.start(OpenLayers.Function.bind(timerCallback, this)) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Kinetic" │ │ │ │ │ -}); │ │ │ │ │ OpenLayers.String = { │ │ │ │ │ startsWith: function(str, sub) { │ │ │ │ │ return str.indexOf(sub) == 0 │ │ │ │ │ }, │ │ │ │ │ contains: function(str, sub) { │ │ │ │ │ return str.indexOf(sub) != -1 │ │ │ │ │ }, │ │ │ │ │ @@ -1795,14 +1605,78 @@ │ │ │ │ │ if (axis == "lon") { │ │ │ │ │ str += coordinate < 0 ? OpenLayers.i18n("W") : OpenLayers.i18n("E") │ │ │ │ │ } else { │ │ │ │ │ str += coordinate < 0 ? OpenLayers.i18n("S") : OpenLayers.i18n("N") │ │ │ │ │ } │ │ │ │ │ return str │ │ │ │ │ }; │ │ │ │ │ +OpenLayers.Util = OpenLayers.Util || {}; │ │ │ │ │ +OpenLayers.Util.vendorPrefix = function() { │ │ │ │ │ + "use strict"; │ │ │ │ │ + var VENDOR_PREFIXES = ["", "O", "ms", "Moz", "Webkit"], │ │ │ │ │ + divStyle = document.createElement("div").style, │ │ │ │ │ + cssCache = {}, │ │ │ │ │ + jsCache = {}; │ │ │ │ │ + │ │ │ │ │ + function domToCss(prefixedDom) { │ │ │ │ │ + if (!prefixedDom) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ + return prefixedDom.replace(/([A-Z])/g, function(c) { │ │ │ │ │ + return "-" + c.toLowerCase() │ │ │ │ │ + }).replace(/^ms-/, "-ms-") │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function css(property) { │ │ │ │ │ + if (cssCache[property] === undefined) { │ │ │ │ │ + var domProperty = property.replace(/(-[\s\S])/g, function(c) { │ │ │ │ │ + return c.charAt(1).toUpperCase() │ │ │ │ │ + }); │ │ │ │ │ + var prefixedDom = style(domProperty); │ │ │ │ │ + cssCache[property] = domToCss(prefixedDom) │ │ │ │ │ + } │ │ │ │ │ + return cssCache[property] │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function js(obj, property) { │ │ │ │ │ + if (jsCache[property] === undefined) { │ │ │ │ │ + var tmpProp, i = 0, │ │ │ │ │ + l = VENDOR_PREFIXES.length, │ │ │ │ │ + prefix, isStyleObj = typeof obj.cssText !== "undefined"; │ │ │ │ │ + jsCache[property] = null; │ │ │ │ │ + for (; i < l; i++) { │ │ │ │ │ + prefix = VENDOR_PREFIXES[i]; │ │ │ │ │ + if (prefix) { │ │ │ │ │ + if (!isStyleObj) { │ │ │ │ │ + prefix = prefix.toLowerCase() │ │ │ │ │ + } │ │ │ │ │ + tmpProp = prefix + property.charAt(0).toUpperCase() + property.slice(1) │ │ │ │ │ + } else { │ │ │ │ │ + tmpProp = property │ │ │ │ │ + } │ │ │ │ │ + if (obj[tmpProp] !== undefined) { │ │ │ │ │ + jsCache[property] = tmpProp; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return jsCache[property] │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function style(property) { │ │ │ │ │ + return js(divStyle, property) │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + css: css, │ │ │ │ │ + js: js, │ │ │ │ │ + style: style, │ │ │ │ │ + cssCache: cssCache, │ │ │ │ │ + jsCache: jsCache │ │ │ │ │ + } │ │ │ │ │ +}(); │ │ │ │ │ OpenLayers.Event = { │ │ │ │ │ observers: false, │ │ │ │ │ KEY_SPACE: 32, │ │ │ │ │ KEY_BACKSPACE: 8, │ │ │ │ │ KEY_TAB: 9, │ │ │ │ │ KEY_RETURN: 13, │ │ │ │ │ KEY_ESC: 27, │ │ │ │ │ @@ -2253,14 +2127,56 @@ │ │ │ │ │ e.touches = touches.slice(); │ │ │ │ │ handler(e) │ │ │ │ │ }; │ │ │ │ │ OpenLayers.Event.observe(element, "MSPointerUp", cb) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Events" │ │ │ │ │ }); │ │ │ │ │ +OpenLayers.Animation = function(window) { │ │ │ │ │ + var requestAnimationFrame = OpenLayers.Util.vendorPrefix.js(window, "requestAnimationFrame"); │ │ │ │ │ + var isNative = !!requestAnimationFrame; │ │ │ │ │ + var requestFrame = function() { │ │ │ │ │ + var request = window[requestAnimationFrame] || function(callback, element) { │ │ │ │ │ + window.setTimeout(callback, 16) │ │ │ │ │ + }; │ │ │ │ │ + return function(callback, element) { │ │ │ │ │ + request.apply(window, [callback, element]) │ │ │ │ │ + } │ │ │ │ │ + }(); │ │ │ │ │ + var counter = 0; │ │ │ │ │ + var loops = {}; │ │ │ │ │ + │ │ │ │ │ + function start(callback, duration, element) { │ │ │ │ │ + duration = duration > 0 ? duration : Number.POSITIVE_INFINITY; │ │ │ │ │ + var id = ++counter; │ │ │ │ │ + var start = +new Date; │ │ │ │ │ + loops[id] = function() { │ │ │ │ │ + if (loops[id] && +new Date - start <= duration) { │ │ │ │ │ + callback(); │ │ │ │ │ + if (loops[id]) { │ │ │ │ │ + requestFrame(loops[id], element) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + delete loops[id] │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + requestFrame(loops[id], element); │ │ │ │ │ + return id │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function stop(id) { │ │ │ │ │ + delete loops[id] │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + isNative: isNative, │ │ │ │ │ + requestFrame: requestFrame, │ │ │ │ │ + start: start, │ │ │ │ │ + stop: stop │ │ │ │ │ + } │ │ │ │ │ +}(window); │ │ │ │ │ OpenLayers.Tween = OpenLayers.Class({ │ │ │ │ │ easing: null, │ │ │ │ │ begin: null, │ │ │ │ │ finish: null, │ │ │ │ │ duration: null, │ │ │ │ │ callbacks: null, │ │ │ │ │ time: null, │ │ │ │ │ @@ -4129,99 +4045,14 @@ │ │ │ │ │ if (typeof value == "string" && value.indexOf("${") != -1) { │ │ │ │ │ value = OpenLayers.String.format(value, context, [feature, property]); │ │ │ │ │ value = isNaN(value) || !value ? value : parseFloat(value) │ │ │ │ │ } │ │ │ │ │ return value │ │ │ │ │ }; │ │ │ │ │ OpenLayers.Style.SYMBOLIZER_PREFIXES = ["Point", "Line", "Polygon", "Text", "Raster"]; │ │ │ │ │ -OpenLayers.Rule = OpenLayers.Class({ │ │ │ │ │ - id: null, │ │ │ │ │ - name: null, │ │ │ │ │ - title: null, │ │ │ │ │ - description: null, │ │ │ │ │ - context: null, │ │ │ │ │ - filter: null, │ │ │ │ │ - elseFilter: false, │ │ │ │ │ - symbolizer: null, │ │ │ │ │ - symbolizers: null, │ │ │ │ │ - minScaleDenominator: null, │ │ │ │ │ - maxScaleDenominator: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.symbolizer = {}; │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - if (this.symbolizers) { │ │ │ │ │ - delete this.symbolizer │ │ │ │ │ - } │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i in this.symbolizer) { │ │ │ │ │ - this.symbolizer[i] = null │ │ │ │ │ - } │ │ │ │ │ - this.symbolizer = null; │ │ │ │ │ - delete this.symbolizers │ │ │ │ │ - }, │ │ │ │ │ - evaluate: function(feature) { │ │ │ │ │ - var context = this.getContext(feature); │ │ │ │ │ - var applies = true; │ │ │ │ │ - if (this.minScaleDenominator || this.maxScaleDenominator) { │ │ │ │ │ - var scale = feature.layer.map.getScale() │ │ │ │ │ - } │ │ │ │ │ - if (this.minScaleDenominator) { │ │ │ │ │ - applies = scale >= OpenLayers.Style.createLiteral(this.minScaleDenominator, context) │ │ │ │ │ - } │ │ │ │ │ - if (applies && this.maxScaleDenominator) { │ │ │ │ │ - applies = scale < OpenLayers.Style.createLiteral(this.maxScaleDenominator, context) │ │ │ │ │ - } │ │ │ │ │ - if (applies && this.filter) { │ │ │ │ │ - if (this.filter.CLASS_NAME == "OpenLayers.Filter.FeatureId") { │ │ │ │ │ - applies = this.filter.evaluate(feature) │ │ │ │ │ - } else { │ │ │ │ │ - applies = this.filter.evaluate(context) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return applies │ │ │ │ │ - }, │ │ │ │ │ - getContext: function(feature) { │ │ │ │ │ - var context = this.context; │ │ │ │ │ - if (!context) { │ │ │ │ │ - context = feature.attributes || feature.data │ │ │ │ │ - } │ │ │ │ │ - if (typeof this.context == "function") { │ │ │ │ │ - context = this.context(feature) │ │ │ │ │ - } │ │ │ │ │ - return context │ │ │ │ │ - }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - var options = OpenLayers.Util.extend({}, this); │ │ │ │ │ - if (this.symbolizers) { │ │ │ │ │ - var len = this.symbolizers.length; │ │ │ │ │ - options.symbolizers = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - options.symbolizers[i] = this.symbolizers[i].clone() │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - options.symbolizer = {}; │ │ │ │ │ - var value, type; │ │ │ │ │ - for (var key in this.symbolizer) { │ │ │ │ │ - value = this.symbolizer[key]; │ │ │ │ │ - type = typeof value; │ │ │ │ │ - if (type === "object") { │ │ │ │ │ - options.symbolizer[key] = OpenLayers.Util.extend({}, value) │ │ │ │ │ - } else if (type === "string") { │ │ │ │ │ - options.symbolizer[key] = value │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - options.filter = this.filter && this.filter.clone(); │ │ │ │ │ - options.context = this.context && OpenLayers.Util.extend({}, this.context); │ │ │ │ │ - return new OpenLayers.Rule(options) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Rule" │ │ │ │ │ -}); │ │ │ │ │ OpenLayers.StyleMap = OpenLayers.Class({ │ │ │ │ │ styles: null, │ │ │ │ │ extendDefault: true, │ │ │ │ │ initialize: function(style, options) { │ │ │ │ │ this.styles = { │ │ │ │ │ default: new OpenLayers.Style(OpenLayers.Feature.Vector.style["default"]), │ │ │ │ │ select: new OpenLayers.Style(OpenLayers.Feature.Vector.style["select"]), │ │ │ │ │ @@ -4283,1756 +4114,183 @@ │ │ │ │ │ }) │ │ │ │ │ })) │ │ │ │ │ } │ │ │ │ │ this.styles[renderIntent].addRules(rules) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.StyleMap" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer = OpenLayers.Class({ │ │ │ │ │ - id: null, │ │ │ │ │ - name: null, │ │ │ │ │ - div: null, │ │ │ │ │ - opacity: 1, │ │ │ │ │ - alwaysInRange: null, │ │ │ │ │ - RESOLUTION_PROPERTIES: ["scales", "resolutions", "maxScale", "minScale", "maxResolution", "minResolution", "numZoomLevels", "maxZoomLevel"], │ │ │ │ │ - events: null, │ │ │ │ │ - map: null, │ │ │ │ │ - isBaseLayer: false, │ │ │ │ │ - alpha: false, │ │ │ │ │ - displayInLayerSwitcher: true, │ │ │ │ │ - visibility: true, │ │ │ │ │ - attribution: null, │ │ │ │ │ - inRange: false, │ │ │ │ │ - imageSize: null, │ │ │ │ │ - options: null, │ │ │ │ │ - eventListeners: null, │ │ │ │ │ - gutter: 0, │ │ │ │ │ - projection: null, │ │ │ │ │ - units: null, │ │ │ │ │ - scales: null, │ │ │ │ │ - resolutions: null, │ │ │ │ │ - maxExtent: null, │ │ │ │ │ - minExtent: null, │ │ │ │ │ - maxResolution: null, │ │ │ │ │ - minResolution: null, │ │ │ │ │ - numZoomLevels: null, │ │ │ │ │ - minScale: null, │ │ │ │ │ - maxScale: null, │ │ │ │ │ - displayOutsideMaxExtent: false, │ │ │ │ │ - wrapDateLine: false, │ │ │ │ │ - metadata: null, │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - this.metadata = {}; │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - if (this.alwaysInRange != null) { │ │ │ │ │ - options.alwaysInRange = this.alwaysInRange │ │ │ │ │ - } │ │ │ │ │ - this.addOptions(options); │ │ │ │ │ - this.name = name; │ │ │ │ │ - if (this.id == null) { │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ - this.div = OpenLayers.Util.createDiv(this.id); │ │ │ │ │ - this.div.style.width = "100%"; │ │ │ │ │ - this.div.style.height = "100%"; │ │ │ │ │ - this.div.dir = "ltr"; │ │ │ │ │ - this.events = new OpenLayers.Events(this, this.div); │ │ │ │ │ - if (this.eventListeners instanceof Object) { │ │ │ │ │ - this.events.on(this.eventListeners) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function(setNewBaseLayer) { │ │ │ │ │ - if (setNewBaseLayer == null) { │ │ │ │ │ - setNewBaseLayer = true │ │ │ │ │ - } │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.removeLayer(this, setNewBaseLayer) │ │ │ │ │ - } │ │ │ │ │ - this.projection = null; │ │ │ │ │ - this.map = null; │ │ │ │ │ - this.name = null; │ │ │ │ │ - this.div = null; │ │ │ │ │ - this.options = null; │ │ │ │ │ - if (this.events) { │ │ │ │ │ - if (this.eventListeners) { │ │ │ │ │ - this.events.un(this.eventListeners) │ │ │ │ │ - } │ │ │ │ │ - this.events.destroy() │ │ │ │ │ - } │ │ │ │ │ - this.eventListeners = null; │ │ │ │ │ - this.events = null │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer(this.name, this.getOptions()) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.applyDefaults(obj, this); │ │ │ │ │ - obj.map = null; │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - getOptions: function() { │ │ │ │ │ - var options = {}; │ │ │ │ │ - for (var o in this.options) { │ │ │ │ │ - options[o] = this[o] │ │ │ │ │ - } │ │ │ │ │ - return options │ │ │ │ │ - }, │ │ │ │ │ - setName: function(newName) { │ │ │ │ │ - if (newName != this.name) { │ │ │ │ │ - this.name = newName; │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "name" │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - addOptions: function(newOptions, reinitialize) { │ │ │ │ │ - if (this.options == null) { │ │ │ │ │ - this.options = {} │ │ │ │ │ - } │ │ │ │ │ - if (newOptions) { │ │ │ │ │ - if (typeof newOptions.projection == "string") { │ │ │ │ │ - newOptions.projection = new OpenLayers.Projection(newOptions.projection) │ │ │ │ │ - } │ │ │ │ │ - if (newOptions.projection) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(newOptions, OpenLayers.Projection.defaults[newOptions.projection.getCode()]) │ │ │ │ │ - } │ │ │ │ │ - if (newOptions.maxExtent && !(newOptions.maxExtent instanceof OpenLayers.Bounds)) { │ │ │ │ │ - newOptions.maxExtent = new OpenLayers.Bounds(newOptions.maxExtent) │ │ │ │ │ - } │ │ │ │ │ - if (newOptions.minExtent && !(newOptions.minExtent instanceof OpenLayers.Bounds)) { │ │ │ │ │ - newOptions.minExtent = new OpenLayers.Bounds(newOptions.minExtent) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.extend(this.options, newOptions); │ │ │ │ │ - OpenLayers.Util.extend(this, newOptions); │ │ │ │ │ - if (this.projection && this.projection.getUnits()) { │ │ │ │ │ - this.units = this.projection.getUnits() │ │ │ │ │ - } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - var resolution = this.map.getResolution(); │ │ │ │ │ - var properties = this.RESOLUTION_PROPERTIES.concat(["projection", "units", "minExtent", "maxExtent"]); │ │ │ │ │ - for (var o in newOptions) { │ │ │ │ │ - if (newOptions.hasOwnProperty(o) && OpenLayers.Util.indexOf(properties, o) >= 0) { │ │ │ │ │ - this.initResolutions(); │ │ │ │ │ - if (reinitialize && this.map.baseLayer === this) { │ │ │ │ │ - this.map.setCenter(this.map.getCenter(), this.map.getZoomForResolution(resolution), false, true); │ │ │ │ │ - this.map.events.triggerEvent("changebaselayer", { │ │ │ │ │ - layer: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - onMapResize: function() {}, │ │ │ │ │ - redraw: function() { │ │ │ │ │ - var redrawn = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.inRange = this.calculateInRange(); │ │ │ │ │ - var extent = this.getExtent(); │ │ │ │ │ - if (extent && this.inRange && this.visibility) { │ │ │ │ │ - var zoomChanged = true; │ │ │ │ │ - this.moveTo(extent, zoomChanged, false); │ │ │ │ │ - this.events.triggerEvent("moveend", { │ │ │ │ │ - zoomChanged: zoomChanged │ │ │ │ │ - }); │ │ │ │ │ - redrawn = true │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return redrawn │ │ │ │ │ - }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - var display = this.visibility; │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - display = display && this.inRange │ │ │ │ │ - } │ │ │ │ │ - this.display(display) │ │ │ │ │ - }, │ │ │ │ │ - moveByPx: function(dx, dy) {}, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - if (this.map == null) { │ │ │ │ │ - this.map = map; │ │ │ │ │ - this.maxExtent = this.maxExtent || this.map.maxExtent; │ │ │ │ │ - this.minExtent = this.minExtent || this.map.minExtent; │ │ │ │ │ - this.projection = this.projection || this.map.projection; │ │ │ │ │ - if (typeof this.projection == "string") { │ │ │ │ │ - this.projection = new OpenLayers.Projection(this.projection) │ │ │ │ │ - } │ │ │ │ │ - this.units = this.projection.getUnits() || this.units || this.map.units; │ │ │ │ │ - this.initResolutions(); │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - this.inRange = this.calculateInRange(); │ │ │ │ │ - var show = this.visibility && this.inRange; │ │ │ │ │ - this.div.style.display = show ? "" : "none" │ │ │ │ │ - } │ │ │ │ │ - this.setTileSize() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - afterAdd: function() {}, │ │ │ │ │ - removeMap: function(map) {}, │ │ │ │ │ - getImageSize: function(bounds) { │ │ │ │ │ - return this.imageSize || this.tileSize │ │ │ │ │ - }, │ │ │ │ │ - setTileSize: function(size) { │ │ │ │ │ - var tileSize = size ? size : this.tileSize ? this.tileSize : this.map.getTileSize(); │ │ │ │ │ - this.tileSize = tileSize; │ │ │ │ │ - if (this.gutter) { │ │ │ │ │ - this.imageSize = new OpenLayers.Size(tileSize.w + 2 * this.gutter, tileSize.h + 2 * this.gutter) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getVisibility: function() { │ │ │ │ │ - return this.visibility │ │ │ │ │ - }, │ │ │ │ │ - setVisibility: function(visibility) { │ │ │ │ │ - if (visibility != this.visibility) { │ │ │ │ │ - this.visibility = visibility; │ │ │ │ │ - this.display(visibility); │ │ │ │ │ - this.redraw(); │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "visibility" │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("visibilitychanged") │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - display: function(display) { │ │ │ │ │ - if (display != (this.div.style.display != "none")) { │ │ │ │ │ - this.div.style.display = display && this.calculateInRange() ? "block" : "none" │ │ │ │ │ - } │ │ │ │ │ +OpenLayers.Kinetic = OpenLayers.Class({ │ │ │ │ │ + threshold: 0, │ │ │ │ │ + deceleration: .0035, │ │ │ │ │ + nbPoints: 100, │ │ │ │ │ + delay: 200, │ │ │ │ │ + points: undefined, │ │ │ │ │ + timerId: undefined, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options) │ │ │ │ │ }, │ │ │ │ │ - calculateInRange: function() { │ │ │ │ │ - var inRange = false; │ │ │ │ │ - if (this.alwaysInRange) { │ │ │ │ │ - inRange = true │ │ │ │ │ - } else { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - var resolution = this.map.getResolution(); │ │ │ │ │ - inRange = resolution >= this.minResolution && resolution <= this.maxResolution │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return inRange │ │ │ │ │ + begin: function() { │ │ │ │ │ + OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ + this.timerId = undefined; │ │ │ │ │ + this.points = [] │ │ │ │ │ }, │ │ │ │ │ - setIsBaseLayer: function(isBaseLayer) { │ │ │ │ │ - if (isBaseLayer != this.isBaseLayer) { │ │ │ │ │ - this.isBaseLayer = isBaseLayer; │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changebaselayer", { │ │ │ │ │ - layer: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ + update: function(xy) { │ │ │ │ │ + this.points.unshift({ │ │ │ │ │ + xy: xy, │ │ │ │ │ + tick: (new Date).getTime() │ │ │ │ │ + }); │ │ │ │ │ + if (this.points.length > this.nbPoints) { │ │ │ │ │ + this.points.pop() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - initResolutions: function() { │ │ │ │ │ - var i, len, p; │ │ │ │ │ - var props = {}, │ │ │ │ │ - alwaysInRange = true; │ │ │ │ │ - for (i = 0, len = this.RESOLUTION_PROPERTIES.length; i < len; i++) { │ │ │ │ │ - p = this.RESOLUTION_PROPERTIES[i]; │ │ │ │ │ - props[p] = this.options[p]; │ │ │ │ │ - if (alwaysInRange && this.options[p]) { │ │ │ │ │ - alwaysInRange = false │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.options.alwaysInRange == null) { │ │ │ │ │ - this.alwaysInRange = alwaysInRange │ │ │ │ │ - } │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - props.resolutions = this.resolutionsFromScales(props.scales) │ │ │ │ │ - } │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - props.resolutions = this.calculateResolutions(props) │ │ │ │ │ - } │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - for (i = 0, len = this.RESOLUTION_PROPERTIES.length; i < len; i++) { │ │ │ │ │ - p = this.RESOLUTION_PROPERTIES[i]; │ │ │ │ │ - props[p] = this.options[p] != null ? this.options[p] : this.map[p] │ │ │ │ │ - } │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - props.resolutions = this.resolutionsFromScales(props.scales) │ │ │ │ │ - } │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - props.resolutions = this.calculateResolutions(props) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var maxResolution; │ │ │ │ │ - if (this.options.maxResolution && this.options.maxResolution !== "auto") { │ │ │ │ │ - maxResolution = this.options.maxResolution │ │ │ │ │ - } │ │ │ │ │ - if (this.options.minScale) { │ │ │ │ │ - maxResolution = OpenLayers.Util.getResolutionFromScale(this.options.minScale, this.units) │ │ │ │ │ - } │ │ │ │ │ - var minResolution; │ │ │ │ │ - if (this.options.minResolution && this.options.minResolution !== "auto") { │ │ │ │ │ - minResolution = this.options.minResolution │ │ │ │ │ - } │ │ │ │ │ - if (this.options.maxScale) { │ │ │ │ │ - minResolution = OpenLayers.Util.getResolutionFromScale(this.options.maxScale, this.units) │ │ │ │ │ - } │ │ │ │ │ - if (props.resolutions) { │ │ │ │ │ - props.resolutions.sort(function(a, b) { │ │ │ │ │ - return b - a │ │ │ │ │ - }); │ │ │ │ │ - if (!maxResolution) { │ │ │ │ │ - maxResolution = props.resolutions[0] │ │ │ │ │ - } │ │ │ │ │ - if (!minResolution) { │ │ │ │ │ - var lastIdx = props.resolutions.length - 1; │ │ │ │ │ - minResolution = props.resolutions[lastIdx] │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.resolutions = props.resolutions; │ │ │ │ │ - if (this.resolutions) { │ │ │ │ │ - len = this.resolutions.length; │ │ │ │ │ - this.scales = new Array(len); │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - this.scales[i] = OpenLayers.Util.getScaleFromResolution(this.resolutions[i], this.units) │ │ │ │ │ + end: function(xy) { │ │ │ │ │ + var last, now = (new Date).getTime(); │ │ │ │ │ + for (var i = 0, l = this.points.length, point; i < l; i++) { │ │ │ │ │ + point = this.points[i]; │ │ │ │ │ + if (now - point.tick > this.delay) { │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - this.numZoomLevels = len │ │ │ │ │ - } │ │ │ │ │ - this.minResolution = minResolution; │ │ │ │ │ - if (minResolution) { │ │ │ │ │ - this.maxScale = OpenLayers.Util.getScaleFromResolution(minResolution, this.units) │ │ │ │ │ - } │ │ │ │ │ - this.maxResolution = maxResolution; │ │ │ │ │ - if (maxResolution) { │ │ │ │ │ - this.minScale = OpenLayers.Util.getScaleFromResolution(maxResolution, this.units) │ │ │ │ │ + last = point │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - resolutionsFromScales: function(scales) { │ │ │ │ │ - if (scales == null) { │ │ │ │ │ + if (!last) { │ │ │ │ │ return │ │ │ │ │ } │ │ │ │ │ - var resolutions, i, len; │ │ │ │ │ - len = scales.length; │ │ │ │ │ - resolutions = new Array(len); │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - resolutions[i] = OpenLayers.Util.getResolutionFromScale(scales[i], this.units) │ │ │ │ │ - } │ │ │ │ │ - return resolutions │ │ │ │ │ - }, │ │ │ │ │ - calculateResolutions: function(props) { │ │ │ │ │ - var viewSize, wRes, hRes; │ │ │ │ │ - var maxResolution = props.maxResolution; │ │ │ │ │ - if (props.minScale != null) { │ │ │ │ │ - maxResolution = OpenLayers.Util.getResolutionFromScale(props.minScale, this.units) │ │ │ │ │ - } else if (maxResolution == "auto" && this.maxExtent != null) { │ │ │ │ │ - viewSize = this.map.getSize(); │ │ │ │ │ - wRes = this.maxExtent.getWidth() / viewSize.w; │ │ │ │ │ - hRes = this.maxExtent.getHeight() / viewSize.h; │ │ │ │ │ - maxResolution = Math.max(wRes, hRes) │ │ │ │ │ - } │ │ │ │ │ - var minResolution = props.minResolution; │ │ │ │ │ - if (props.maxScale != null) { │ │ │ │ │ - minResolution = OpenLayers.Util.getResolutionFromScale(props.maxScale, this.units) │ │ │ │ │ - } else if (props.minResolution == "auto" && this.minExtent != null) { │ │ │ │ │ - viewSize = this.map.getSize(); │ │ │ │ │ - wRes = this.minExtent.getWidth() / viewSize.w; │ │ │ │ │ - hRes = this.minExtent.getHeight() / viewSize.h; │ │ │ │ │ - minResolution = Math.max(wRes, hRes) │ │ │ │ │ - } │ │ │ │ │ - if (typeof maxResolution !== "number" && typeof minResolution !== "number" && this.maxExtent != null) { │ │ │ │ │ - var tileSize = this.map.getTileSize(); │ │ │ │ │ - maxResolution = Math.max(this.maxExtent.getWidth() / tileSize.w, this.maxExtent.getHeight() / tileSize.h) │ │ │ │ │ - } │ │ │ │ │ - var maxZoomLevel = props.maxZoomLevel; │ │ │ │ │ - var numZoomLevels = props.numZoomLevels; │ │ │ │ │ - if (typeof minResolution === "number" && typeof maxResolution === "number" && numZoomLevels === undefined) { │ │ │ │ │ - var ratio = maxResolution / minResolution; │ │ │ │ │ - numZoomLevels = Math.floor(Math.log(ratio) / Math.log(2)) + 1 │ │ │ │ │ - } else if (numZoomLevels === undefined && maxZoomLevel != null) { │ │ │ │ │ - numZoomLevels = maxZoomLevel + 1 │ │ │ │ │ - } │ │ │ │ │ - if (typeof numZoomLevels !== "number" || numZoomLevels <= 0 || typeof maxResolution !== "number" && typeof minResolution !== "number") { │ │ │ │ │ + var time = (new Date).getTime() - last.tick; │ │ │ │ │ + var dist = Math.sqrt(Math.pow(xy.x - last.xy.x, 2) + Math.pow(xy.y - last.xy.y, 2)); │ │ │ │ │ + var speed = dist / time; │ │ │ │ │ + if (speed == 0 || speed < this.threshold) { │ │ │ │ │ return │ │ │ │ │ } │ │ │ │ │ - var resolutions = new Array(numZoomLevels); │ │ │ │ │ - var base = 2; │ │ │ │ │ - if (typeof minResolution == "number" && typeof maxResolution == "number") { │ │ │ │ │ - base = Math.pow(maxResolution / minResolution, 1 / (numZoomLevels - 1)) │ │ │ │ │ - } │ │ │ │ │ - var i; │ │ │ │ │ - if (typeof maxResolution === "number") { │ │ │ │ │ - for (i = 0; i < numZoomLevels; i++) { │ │ │ │ │ - resolutions[i] = maxResolution / Math.pow(base, i) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - for (i = 0; i < numZoomLevels; i++) { │ │ │ │ │ - resolutions[numZoomLevels - 1 - i] = minResolution * Math.pow(base, i) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return resolutions │ │ │ │ │ - }, │ │ │ │ │ - getResolution: function() { │ │ │ │ │ - var zoom = this.map.getZoom(); │ │ │ │ │ - return this.getResolutionForZoom(zoom) │ │ │ │ │ - }, │ │ │ │ │ - getExtent: function() { │ │ │ │ │ - return this.map.calculateBounds() │ │ │ │ │ - }, │ │ │ │ │ - getZoomForExtent: function(extent, closest) { │ │ │ │ │ - var viewSize = this.map.getSize(); │ │ │ │ │ - var idealResolution = Math.max(extent.getWidth() / viewSize.w, extent.getHeight() / viewSize.h); │ │ │ │ │ - return this.getZoomForResolution(idealResolution, closest) │ │ │ │ │ - }, │ │ │ │ │ - getDataExtent: function() {}, │ │ │ │ │ - getResolutionForZoom: function(zoom) { │ │ │ │ │ - zoom = Math.max(0, Math.min(zoom, this.resolutions.length - 1)); │ │ │ │ │ - var resolution; │ │ │ │ │ - if (this.map.fractionalZoom) { │ │ │ │ │ - var low = Math.floor(zoom); │ │ │ │ │ - var high = Math.ceil(zoom); │ │ │ │ │ - resolution = this.resolutions[low] - (zoom - low) * (this.resolutions[low] - this.resolutions[high]) │ │ │ │ │ - } else { │ │ │ │ │ - resolution = this.resolutions[Math.round(zoom)] │ │ │ │ │ - } │ │ │ │ │ - return resolution │ │ │ │ │ - }, │ │ │ │ │ - getZoomForResolution: function(resolution, closest) { │ │ │ │ │ - var zoom, i, len; │ │ │ │ │ - if (this.map.fractionalZoom) { │ │ │ │ │ - var lowZoom = 0; │ │ │ │ │ - var highZoom = this.resolutions.length - 1; │ │ │ │ │ - var highRes = this.resolutions[lowZoom]; │ │ │ │ │ - var lowRes = this.resolutions[highZoom]; │ │ │ │ │ - var res; │ │ │ │ │ - for (i = 0, len = this.resolutions.length; i < len; ++i) { │ │ │ │ │ - res = this.resolutions[i]; │ │ │ │ │ - if (res >= resolution) { │ │ │ │ │ - highRes = res; │ │ │ │ │ - lowZoom = i │ │ │ │ │ - } │ │ │ │ │ - if (res <= resolution) { │ │ │ │ │ - lowRes = res; │ │ │ │ │ - highZoom = i; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var dRes = highRes - lowRes; │ │ │ │ │ - if (dRes > 0) { │ │ │ │ │ - zoom = lowZoom + (highRes - resolution) / dRes │ │ │ │ │ - } else { │ │ │ │ │ - zoom = lowZoom │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - var diff; │ │ │ │ │ - var minDiff = Number.POSITIVE_INFINITY; │ │ │ │ │ - for (i = 0, len = this.resolutions.length; i < len; i++) { │ │ │ │ │ - if (closest) { │ │ │ │ │ - diff = Math.abs(this.resolutions[i] - resolution); │ │ │ │ │ - if (diff > minDiff) { │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - minDiff = diff │ │ │ │ │ - } else { │ │ │ │ │ - if (this.resolutions[i] < resolution) { │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - zoom = Math.max(0, i - 1) │ │ │ │ │ - } │ │ │ │ │ - return zoom │ │ │ │ │ - }, │ │ │ │ │ - getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ - var lonlat = null; │ │ │ │ │ - var map = this.map; │ │ │ │ │ - if (viewPortPx != null && map.minPx) { │ │ │ │ │ - var res = map.getResolution(); │ │ │ │ │ - var maxExtent = map.getMaxExtent({ │ │ │ │ │ - restricted: true │ │ │ │ │ - }); │ │ │ │ │ - var lon = (viewPortPx.x - map.minPx.x) * res + maxExtent.left; │ │ │ │ │ - var lat = (map.minPx.y - viewPortPx.y) * res + maxExtent.top; │ │ │ │ │ - lonlat = new OpenLayers.LonLat(lon, lat); │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - lonlat = lonlat.wrapDateLine(this.maxExtent) │ │ │ │ │ - } │ │ │ │ │ + var theta = Math.asin((xy.y - last.xy.y) / dist); │ │ │ │ │ + if (last.xy.x <= xy.x) { │ │ │ │ │ + theta = Math.PI - theta │ │ │ │ │ } │ │ │ │ │ - return lonlat │ │ │ │ │ - }, │ │ │ │ │ - getViewPortPxFromLonLat: function(lonlat, resolution) { │ │ │ │ │ - var px = null; │ │ │ │ │ - if (lonlat != null) { │ │ │ │ │ - resolution = resolution || this.map.getResolution(); │ │ │ │ │ - var extent = this.map.calculateBounds(null, resolution); │ │ │ │ │ - px = new OpenLayers.Pixel(1 / resolution * (lonlat.lon - extent.left), 1 / resolution * (extent.top - lonlat.lat)) │ │ │ │ │ + return { │ │ │ │ │ + speed: speed, │ │ │ │ │ + theta: theta │ │ │ │ │ } │ │ │ │ │ - return px │ │ │ │ │ }, │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - if (opacity != this.opacity) { │ │ │ │ │ - this.opacity = opacity; │ │ │ │ │ - var childNodes = this.div.childNodes; │ │ │ │ │ - for (var i = 0, len = childNodes.length; i < len; ++i) { │ │ │ │ │ - var element = childNodes[i].firstChild || childNodes[i]; │ │ │ │ │ - var lastChild = childNodes[i].lastChild; │ │ │ │ │ - if (lastChild && lastChild.nodeName.toLowerCase() === "iframe") { │ │ │ │ │ - element = lastChild.parentNode │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.modifyDOMElement(element, null, null, null, null, null, null, opacity) │ │ │ │ │ - } │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "opacity" │ │ │ │ │ - }) │ │ │ │ │ + move: function(info, callback) { │ │ │ │ │ + var v0 = info.speed; │ │ │ │ │ + var fx = Math.cos(info.theta); │ │ │ │ │ + var fy = -Math.sin(info.theta); │ │ │ │ │ + var initialTime = (new Date).getTime(); │ │ │ │ │ + var lastX = 0; │ │ │ │ │ + var lastY = 0; │ │ │ │ │ + var timerCallback = function() { │ │ │ │ │ + if (this.timerId == null) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getZIndex: function() { │ │ │ │ │ - return this.div.style.zIndex │ │ │ │ │ - }, │ │ │ │ │ - setZIndex: function(zIndex) { │ │ │ │ │ - this.div.style.zIndex = zIndex │ │ │ │ │ - }, │ │ │ │ │ - adjustBounds: function(bounds) { │ │ │ │ │ - if (this.gutter) { │ │ │ │ │ - var mapGutter = this.gutter * this.map.getResolution(); │ │ │ │ │ - bounds = new OpenLayers.Bounds(bounds.left - mapGutter, bounds.bottom - mapGutter, bounds.right + mapGutter, bounds.top + mapGutter) │ │ │ │ │ - } │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - var wrappingOptions = { │ │ │ │ │ - rightTolerance: this.getResolution(), │ │ │ │ │ - leftTolerance: this.getResolution() │ │ │ │ │ - }; │ │ │ │ │ - bounds = bounds.wrapDateLine(this.maxExtent, wrappingOptions) │ │ │ │ │ - } │ │ │ │ │ - return bounds │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.HTTPRequest = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ - URL_HASH_FACTOR: (Math.sqrt(5) - 1) / 2, │ │ │ │ │ - url: null, │ │ │ │ │ - params: null, │ │ │ │ │ - reproject: false, │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ - this.url = url; │ │ │ │ │ - if (!this.params) { │ │ │ │ │ - this.params = OpenLayers.Util.extend({}, params) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.url = null; │ │ │ │ │ - this.params = null; │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.HTTPRequest(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - setUrl: function(newUrl) { │ │ │ │ │ - this.url = newUrl │ │ │ │ │ - }, │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - this.params = OpenLayers.Util.extend(this.params, newParams); │ │ │ │ │ - var ret = this.redraw(); │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "params" │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return ret │ │ │ │ │ - }, │ │ │ │ │ - redraw: function(force) { │ │ │ │ │ - if (force) { │ │ │ │ │ - return this.mergeNewParams({ │ │ │ │ │ - _olSalt: Math.random() │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - return OpenLayers.Layer.prototype.redraw.apply(this, []) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - selectUrl: function(paramString, urls) { │ │ │ │ │ - var product = 1; │ │ │ │ │ - for (var i = 0, len = paramString.length; i < len; i++) { │ │ │ │ │ - product *= paramString.charCodeAt(i) * this.URL_HASH_FACTOR; │ │ │ │ │ - product -= Math.floor(product) │ │ │ │ │ - } │ │ │ │ │ - return urls[Math.floor(product * urls.length)] │ │ │ │ │ - }, │ │ │ │ │ - getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ - var url = altUrl || this.url; │ │ │ │ │ - var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ - allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ - var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(paramsString, url) │ │ │ │ │ - } │ │ │ │ │ - var urlParams = OpenLayers.Util.upperCaseObject(OpenLayers.Util.getParameters(url)); │ │ │ │ │ - for (var key in allParams) { │ │ │ │ │ - if (key.toUpperCase() in urlParams) { │ │ │ │ │ - delete allParams[key] │ │ │ │ │ + var t = (new Date).getTime() - initialTime; │ │ │ │ │ + var p = -this.deceleration * Math.pow(t, 2) / 2 + v0 * t; │ │ │ │ │ + var x = p * fx; │ │ │ │ │ + var y = p * fy; │ │ │ │ │ + var args = {}; │ │ │ │ │ + args.end = false; │ │ │ │ │ + var v = -this.deceleration * t + v0; │ │ │ │ │ + if (v <= 0) { │ │ │ │ │ + OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + args.end = true │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ - return OpenLayers.Util.urlAppend(url, paramsString) │ │ │ │ │ + args.x = x - lastX; │ │ │ │ │ + args.y = y - lastY; │ │ │ │ │ + lastX = x; │ │ │ │ │ + lastY = y; │ │ │ │ │ + callback(args.x, args.y, args.end) │ │ │ │ │ + }; │ │ │ │ │ + this.timerId = OpenLayers.Animation.start(OpenLayers.Function.bind(timerCallback, this)) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.HTTPRequest" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Kinetic" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Tile = OpenLayers.Class({ │ │ │ │ │ - events: null, │ │ │ │ │ - eventListeners: null, │ │ │ │ │ +OpenLayers.Rule = OpenLayers.Class({ │ │ │ │ │ id: null, │ │ │ │ │ - layer: null, │ │ │ │ │ - url: null, │ │ │ │ │ - bounds: null, │ │ │ │ │ - size: null, │ │ │ │ │ - position: null, │ │ │ │ │ - isLoading: false, │ │ │ │ │ - initialize: function(layer, position, bounds, url, size, options) { │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - this.position = position.clone(); │ │ │ │ │ - this.setBounds(bounds); │ │ │ │ │ - this.url = url; │ │ │ │ │ - if (size) { │ │ │ │ │ - this.size = size.clone() │ │ │ │ │ - } │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID("Tile_"); │ │ │ │ │ + name: null, │ │ │ │ │ + title: null, │ │ │ │ │ + description: null, │ │ │ │ │ + context: null, │ │ │ │ │ + filter: null, │ │ │ │ │ + elseFilter: false, │ │ │ │ │ + symbolizer: null, │ │ │ │ │ + symbolizers: null, │ │ │ │ │ + minScaleDenominator: null, │ │ │ │ │ + maxScaleDenominator: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.symbolizer = {}; │ │ │ │ │ OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.events = new OpenLayers.Events(this); │ │ │ │ │ - if (this.eventListeners instanceof Object) { │ │ │ │ │ - this.events.on(this.eventListeners) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - unload: function() { │ │ │ │ │ - if (this.isLoading) { │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - this.events.triggerEvent("unload") │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.layer = null; │ │ │ │ │ - this.bounds = null; │ │ │ │ │ - this.size = null; │ │ │ │ │ - this.position = null; │ │ │ │ │ - if (this.eventListeners) { │ │ │ │ │ - this.events.un(this.eventListeners) │ │ │ │ │ - } │ │ │ │ │ - this.events.destroy(); │ │ │ │ │ - this.eventListeners = null; │ │ │ │ │ - this.events = null │ │ │ │ │ - }, │ │ │ │ │ - draw: function(force) { │ │ │ │ │ - if (!force) { │ │ │ │ │ - this.clear() │ │ │ │ │ - } │ │ │ │ │ - var draw = this.shouldDraw(); │ │ │ │ │ - if (draw && !force && this.events.triggerEvent("beforedraw") === false) { │ │ │ │ │ - draw = null │ │ │ │ │ - } │ │ │ │ │ - return draw │ │ │ │ │ - }, │ │ │ │ │ - shouldDraw: function() { │ │ │ │ │ - var withinMaxExtent = false, │ │ │ │ │ - maxExtent = this.layer.maxExtent; │ │ │ │ │ - if (maxExtent) { │ │ │ │ │ - var map = this.layer.map; │ │ │ │ │ - var worldBounds = map.baseLayer.wrapDateLine && map.getMaxExtent(); │ │ │ │ │ - if (this.bounds.intersectsBounds(maxExtent, { │ │ │ │ │ - inclusive: false, │ │ │ │ │ - worldBounds: worldBounds │ │ │ │ │ - })) { │ │ │ │ │ - withinMaxExtent = true │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return withinMaxExtent || this.layer.displayOutsideMaxExtent │ │ │ │ │ - }, │ │ │ │ │ - setBounds: function(bounds) { │ │ │ │ │ - bounds = bounds.clone(); │ │ │ │ │ - if (this.layer.map.baseLayer.wrapDateLine) { │ │ │ │ │ - var worldExtent = this.layer.map.getMaxExtent(), │ │ │ │ │ - tolerance = this.layer.map.getResolution(); │ │ │ │ │ - bounds = bounds.wrapDateLine(worldExtent, { │ │ │ │ │ - leftTolerance: tolerance, │ │ │ │ │ - rightTolerance: tolerance │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.bounds = bounds │ │ │ │ │ - }, │ │ │ │ │ - moveTo: function(bounds, position, redraw) { │ │ │ │ │ - if (redraw == null) { │ │ │ │ │ - redraw = true │ │ │ │ │ - } │ │ │ │ │ - this.setBounds(bounds); │ │ │ │ │ - this.position = position.clone(); │ │ │ │ │ - if (redraw) { │ │ │ │ │ - this.draw() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - clear: function(draw) {}, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Tile" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Tile.Image = OpenLayers.Class(OpenLayers.Tile, { │ │ │ │ │ - url: null, │ │ │ │ │ - imgDiv: null, │ │ │ │ │ - frame: null, │ │ │ │ │ - imageReloadAttempts: null, │ │ │ │ │ - layerAlphaHack: null, │ │ │ │ │ - asyncRequestId: null, │ │ │ │ │ - maxGetUrlLength: null, │ │ │ │ │ - canvasContext: null, │ │ │ │ │ - crossOriginKeyword: null, │ │ │ │ │ - initialize: function(layer, position, bounds, url, size, options) { │ │ │ │ │ - OpenLayers.Tile.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.url = url; │ │ │ │ │ - this.layerAlphaHack = this.layer.alpha && OpenLayers.Util.alphaHack(); │ │ │ │ │ - if (this.maxGetUrlLength != null || this.layer.gutter || this.layerAlphaHack) { │ │ │ │ │ - this.frame = document.createElement("div"); │ │ │ │ │ - this.frame.style.position = "absolute"; │ │ │ │ │ - this.frame.style.overflow = "hidden" │ │ │ │ │ - } │ │ │ │ │ - if (this.maxGetUrlLength != null) { │ │ │ │ │ - OpenLayers.Util.extend(this, OpenLayers.Tile.Image.IFrame) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.imgDiv) { │ │ │ │ │ - this.clear(); │ │ │ │ │ - this.imgDiv = null; │ │ │ │ │ - this.frame = null │ │ │ │ │ - } │ │ │ │ │ - this.asyncRequestId = null; │ │ │ │ │ - OpenLayers.Tile.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - var shouldDraw = OpenLayers.Tile.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (shouldDraw) { │ │ │ │ │ - if (this.layer != this.layer.map.baseLayer && this.layer.reproject) { │ │ │ │ │ - this.bounds = this.getBoundsFromBaseLayer(this.position) │ │ │ │ │ - } │ │ │ │ │ - if (this.isLoading) { │ │ │ │ │ - this._loadEvent = "reload" │ │ │ │ │ - } else { │ │ │ │ │ - this.isLoading = true; │ │ │ │ │ - this._loadEvent = "loadstart" │ │ │ │ │ - } │ │ │ │ │ - this.renderTile(); │ │ │ │ │ - this.positionTile() │ │ │ │ │ - } else if (shouldDraw === false) { │ │ │ │ │ - this.unload() │ │ │ │ │ - } │ │ │ │ │ - return shouldDraw │ │ │ │ │ - }, │ │ │ │ │ - renderTile: function() { │ │ │ │ │ - if (this.layer.async) { │ │ │ │ │ - var id = this.asyncRequestId = (this.asyncRequestId || 0) + 1; │ │ │ │ │ - this.layer.getURLasync(this.bounds, function(url) { │ │ │ │ │ - if (id == this.asyncRequestId) { │ │ │ │ │ - this.url = url; │ │ │ │ │ - this.initImage() │ │ │ │ │ - } │ │ │ │ │ - }, this) │ │ │ │ │ - } else { │ │ │ │ │ - this.url = this.layer.getURL(this.bounds); │ │ │ │ │ - this.initImage() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - positionTile: function() { │ │ │ │ │ - var style = this.getTile().style, │ │ │ │ │ - size = this.frame ? this.size : this.layer.getImageSize(this.bounds), │ │ │ │ │ - ratio = 1; │ │ │ │ │ - if (this.layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ - ratio = this.layer.getServerResolution() / this.layer.map.getResolution() │ │ │ │ │ - } │ │ │ │ │ - style.left = this.position.x + "px"; │ │ │ │ │ - style.top = this.position.y + "px"; │ │ │ │ │ - style.width = Math.round(ratio * size.w) + "px"; │ │ │ │ │ - style.height = Math.round(ratio * size.h) + "px" │ │ │ │ │ - }, │ │ │ │ │ - clear: function() { │ │ │ │ │ - OpenLayers.Tile.prototype.clear.apply(this, arguments); │ │ │ │ │ - var img = this.imgDiv; │ │ │ │ │ - if (img) { │ │ │ │ │ - var tile = this.getTile(); │ │ │ │ │ - if (tile.parentNode === this.layer.div) { │ │ │ │ │ - this.layer.div.removeChild(tile) │ │ │ │ │ - } │ │ │ │ │ - this.setImgSrc(); │ │ │ │ │ - if (this.layerAlphaHack === true) { │ │ │ │ │ - img.style.filter = "" │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Element.removeClass(img, "olImageLoadError") │ │ │ │ │ - } │ │ │ │ │ - this.canvasContext = null │ │ │ │ │ - }, │ │ │ │ │ - getImage: function() { │ │ │ │ │ - if (!this.imgDiv) { │ │ │ │ │ - this.imgDiv = OpenLayers.Tile.Image.IMAGE.cloneNode(false); │ │ │ │ │ - var style = this.imgDiv.style; │ │ │ │ │ - if (this.frame) { │ │ │ │ │ - var left = 0, │ │ │ │ │ - top = 0; │ │ │ │ │ - if (this.layer.gutter) { │ │ │ │ │ - left = this.layer.gutter / this.layer.tileSize.w * 100; │ │ │ │ │ - top = this.layer.gutter / this.layer.tileSize.h * 100 │ │ │ │ │ - } │ │ │ │ │ - style.left = -left + "%"; │ │ │ │ │ - style.top = -top + "%"; │ │ │ │ │ - style.width = 2 * left + 100 + "%"; │ │ │ │ │ - style.height = 2 * top + 100 + "%" │ │ │ │ │ - } │ │ │ │ │ - style.visibility = "hidden"; │ │ │ │ │ - style.opacity = 0; │ │ │ │ │ - if (this.layer.opacity < 1) { │ │ │ │ │ - style.filter = "alpha(opacity=" + this.layer.opacity * 100 + ")" │ │ │ │ │ - } │ │ │ │ │ - style.position = "absolute"; │ │ │ │ │ - if (this.layerAlphaHack) { │ │ │ │ │ - style.paddingTop = style.height; │ │ │ │ │ - style.height = "0"; │ │ │ │ │ - style.width = "100%" │ │ │ │ │ - } │ │ │ │ │ - if (this.frame) { │ │ │ │ │ - this.frame.appendChild(this.imgDiv) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return this.imgDiv │ │ │ │ │ - }, │ │ │ │ │ - setImage: function(img) { │ │ │ │ │ - this.imgDiv = img │ │ │ │ │ - }, │ │ │ │ │ - initImage: function() { │ │ │ │ │ - if (!this.url && !this.imgDiv) { │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("beforeload"); │ │ │ │ │ - this.layer.div.appendChild(this.getTile()); │ │ │ │ │ - this.events.triggerEvent(this._loadEvent); │ │ │ │ │ - var img = this.getImage(); │ │ │ │ │ - var src = img.getAttribute("src") || ""; │ │ │ │ │ - if (this.url && OpenLayers.Util.isEquivalentUrl(src, this.url)) { │ │ │ │ │ - this._loadTimeout = window.setTimeout(OpenLayers.Function.bind(this.onImageLoad, this), 0) │ │ │ │ │ - } else { │ │ │ │ │ - this.stopLoading(); │ │ │ │ │ - if (this.crossOriginKeyword) { │ │ │ │ │ - img.removeAttribute("crossorigin") │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.observe(img, "load", OpenLayers.Function.bind(this.onImageLoad, this)); │ │ │ │ │ - OpenLayers.Event.observe(img, "error", OpenLayers.Function.bind(this.onImageError, this)); │ │ │ │ │ - this.imageReloadAttempts = 0; │ │ │ │ │ - this.setImgSrc(this.url) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setImgSrc: function(url) { │ │ │ │ │ - var img = this.imgDiv; │ │ │ │ │ - if (url) { │ │ │ │ │ - img.style.visibility = "hidden"; │ │ │ │ │ - img.style.opacity = 0; │ │ │ │ │ - if (this.crossOriginKeyword) { │ │ │ │ │ - if (url.substr(0, 5) !== "data:") { │ │ │ │ │ - img.setAttribute("crossorigin", this.crossOriginKeyword) │ │ │ │ │ - } else { │ │ │ │ │ - img.removeAttribute("crossorigin") │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - img.src = url │ │ │ │ │ - } else { │ │ │ │ │ - this.stopLoading(); │ │ │ │ │ - this.imgDiv = null; │ │ │ │ │ - if (img.parentNode) { │ │ │ │ │ - img.parentNode.removeChild(img) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getTile: function() { │ │ │ │ │ - return this.frame ? this.frame : this.getImage() │ │ │ │ │ - }, │ │ │ │ │ - createBackBuffer: function() { │ │ │ │ │ - if (!this.imgDiv || this.isLoading) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var backBuffer; │ │ │ │ │ - if (this.frame) { │ │ │ │ │ - backBuffer = this.frame.cloneNode(false); │ │ │ │ │ - backBuffer.appendChild(this.imgDiv) │ │ │ │ │ - } else { │ │ │ │ │ - backBuffer = this.imgDiv │ │ │ │ │ - } │ │ │ │ │ - this.imgDiv = null; │ │ │ │ │ - return backBuffer │ │ │ │ │ - }, │ │ │ │ │ - onImageLoad: function() { │ │ │ │ │ - var img = this.imgDiv; │ │ │ │ │ - this.stopLoading(); │ │ │ │ │ - img.style.visibility = "inherit"; │ │ │ │ │ - img.style.opacity = this.layer.opacity; │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - this.canvasContext = null; │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ - if (this.layerAlphaHack === true) { │ │ │ │ │ - img.style.filter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(src='" + img.src + "', sizingMethod='scale')" │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - onImageError: function() { │ │ │ │ │ - var img = this.imgDiv; │ │ │ │ │ - if (img.src != null) { │ │ │ │ │ - this.imageReloadAttempts++; │ │ │ │ │ - if (this.imageReloadAttempts <= OpenLayers.IMAGE_RELOAD_ATTEMPTS) { │ │ │ │ │ - this.setImgSrc(this.layer.getURL(this.bounds)) │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Element.addClass(img, "olImageLoadError"); │ │ │ │ │ - this.events.triggerEvent("loaderror"); │ │ │ │ │ - this.onImageLoad() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - stopLoading: function() { │ │ │ │ │ - OpenLayers.Event.stopObservingElement(this.imgDiv); │ │ │ │ │ - window.clearTimeout(this._loadTimeout); │ │ │ │ │ - delete this._loadTimeout │ │ │ │ │ - }, │ │ │ │ │ - getCanvasContext: function() { │ │ │ │ │ - if (OpenLayers.CANVAS_SUPPORTED && this.imgDiv && !this.isLoading) { │ │ │ │ │ - if (!this.canvasContext) { │ │ │ │ │ - var canvas = document.createElement("canvas"); │ │ │ │ │ - canvas.width = this.size.w; │ │ │ │ │ - canvas.height = this.size.h; │ │ │ │ │ - this.canvasContext = canvas.getContext("2d"); │ │ │ │ │ - this.canvasContext.drawImage(this.imgDiv, 0, 0) │ │ │ │ │ - } │ │ │ │ │ - return this.canvasContext │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Tile.Image" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Tile.Image.IMAGE = function() { │ │ │ │ │ - var img = new Image; │ │ │ │ │ - img.className = "olTileImage"; │ │ │ │ │ - img.galleryImg = "no"; │ │ │ │ │ - return img │ │ │ │ │ -}(); │ │ │ │ │ -OpenLayers.Layer.Grid = OpenLayers.Class(OpenLayers.Layer.HTTPRequest, { │ │ │ │ │ - tileSize: null, │ │ │ │ │ - tileOriginCorner: "bl", │ │ │ │ │ - tileOrigin: null, │ │ │ │ │ - tileOptions: null, │ │ │ │ │ - tileClass: OpenLayers.Tile.Image, │ │ │ │ │ - grid: null, │ │ │ │ │ - singleTile: false, │ │ │ │ │ - ratio: 1.5, │ │ │ │ │ - buffer: 0, │ │ │ │ │ - transitionEffect: "resize", │ │ │ │ │ - numLoadingTiles: 0, │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ - loading: false, │ │ │ │ │ - backBuffer: null, │ │ │ │ │ - gridResolution: null, │ │ │ │ │ - backBufferResolution: null, │ │ │ │ │ - backBufferLonLat: null, │ │ │ │ │ - backBufferTimerId: null, │ │ │ │ │ - removeBackBufferDelay: null, │ │ │ │ │ - className: null, │ │ │ │ │ - gridLayout: null, │ │ │ │ │ - rowSign: null, │ │ │ │ │ - transitionendEvents: ["transitionend", "webkitTransitionEnd", "otransitionend", "oTransitionEnd"], │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.grid = []; │ │ │ │ │ - this._removeBackBuffer = OpenLayers.Function.bind(this.removeBackBuffer, this); │ │ │ │ │ - this.initProperties(); │ │ │ │ │ - this.rowSign = this.tileOriginCorner.substr(0, 1) === "t" ? 1 : -1 │ │ │ │ │ - }, │ │ │ │ │ - initProperties: function() { │ │ │ │ │ - if (this.options.removeBackBufferDelay === undefined) { │ │ │ │ │ - this.removeBackBufferDelay = this.singleTile ? 0 : 2500 │ │ │ │ │ - } │ │ │ │ │ - if (this.options.className === undefined) { │ │ │ │ │ - this.className = this.singleTile ? "olLayerGridSingleTile" : "olLayerGrid" │ │ │ │ │ + if (this.symbolizers) { │ │ │ │ │ + delete this.symbolizer │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.setMap.call(this, map); │ │ │ │ │ - OpenLayers.Element.addClass(this.div, this.className) │ │ │ │ │ - }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - this.removeBackBuffer() │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - this.removeBackBuffer(); │ │ │ │ │ - this.clearGrid(); │ │ │ │ │ - this.grid = null; │ │ │ │ │ - this.tileSize = null; │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - clearGrid: function() { │ │ │ │ │ - if (this.grid) { │ │ │ │ │ - for (var iRow = 0, len = this.grid.length; iRow < len; iRow++) { │ │ │ │ │ - var row = this.grid[iRow]; │ │ │ │ │ - for (var iCol = 0, clen = row.length; iCol < clen; iCol++) { │ │ │ │ │ - var tile = row[iCol]; │ │ │ │ │ - this.destroyTile(tile) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.grid = []; │ │ │ │ │ - this.gridResolution = null; │ │ │ │ │ - this.gridLayout = null │ │ │ │ │ + for (var i in this.symbolizer) { │ │ │ │ │ + this.symbolizer[i] = null │ │ │ │ │ } │ │ │ │ │ + this.symbolizer = null; │ │ │ │ │ + delete this.symbolizers │ │ │ │ │ }, │ │ │ │ │ - addOptions: function(newOptions, reinitialize) { │ │ │ │ │ - var singleTileChanged = newOptions.singleTile !== undefined && newOptions.singleTile !== this.singleTile; │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.addOptions.apply(this, arguments); │ │ │ │ │ - if (this.map && singleTileChanged) { │ │ │ │ │ - this.initProperties(); │ │ │ │ │ - this.clearGrid(); │ │ │ │ │ - this.tileSize = this.options.tileSize; │ │ │ │ │ - this.setTileSize(); │ │ │ │ │ - this.moveTo(null, true) │ │ │ │ │ + evaluate: function(feature) { │ │ │ │ │ + var context = this.getContext(feature); │ │ │ │ │ + var applies = true; │ │ │ │ │ + if (this.minScaleDenominator || this.maxScaleDenominator) { │ │ │ │ │ + var scale = feature.layer.map.getScale() │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Grid(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ + if (this.minScaleDenominator) { │ │ │ │ │ + applies = scale >= OpenLayers.Style.createLiteral(this.minScaleDenominator, context) │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.HTTPRequest.prototype.clone.apply(this, [obj]); │ │ │ │ │ - if (this.tileSize != null) { │ │ │ │ │ - obj.tileSize = this.tileSize.clone() │ │ │ │ │ + if (applies && this.maxScaleDenominator) { │ │ │ │ │ + applies = scale < OpenLayers.Style.createLiteral(this.maxScaleDenominator, context) │ │ │ │ │ } │ │ │ │ │ - obj.grid = []; │ │ │ │ │ - obj.gridResolution = null; │ │ │ │ │ - obj.backBuffer = null; │ │ │ │ │ - obj.backBufferTimerId = null; │ │ │ │ │ - obj.loading = false; │ │ │ │ │ - obj.numLoadingTiles = 0; │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - bounds = bounds || this.map.getExtent(); │ │ │ │ │ - if (bounds != null) { │ │ │ │ │ - var forceReTile = !this.grid.length || zoomChanged; │ │ │ │ │ - var tilesBounds = this.getTilesBounds(); │ │ │ │ │ - var resolution = this.map.getResolution(); │ │ │ │ │ - var serverResolution = this.getServerResolution(resolution); │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - if (forceReTile || !dragging && !tilesBounds.containsBounds(bounds)) { │ │ │ │ │ - if (zoomChanged && this.transitionEffect !== "resize") { │ │ │ │ │ - this.removeBackBuffer() │ │ │ │ │ - } │ │ │ │ │ - if (!zoomChanged || this.transitionEffect === "resize") { │ │ │ │ │ - this.applyBackBuffer(resolution) │ │ │ │ │ - } │ │ │ │ │ - this.initSingleTile(bounds) │ │ │ │ │ - } │ │ │ │ │ + if (applies && this.filter) { │ │ │ │ │ + if (this.filter.CLASS_NAME == "OpenLayers.Filter.FeatureId") { │ │ │ │ │ + applies = this.filter.evaluate(feature) │ │ │ │ │ } else { │ │ │ │ │ - forceReTile = forceReTile || !tilesBounds.intersectsBounds(bounds, { │ │ │ │ │ - worldBounds: this.map.baseLayer.wrapDateLine && this.map.getMaxExtent() │ │ │ │ │ - }); │ │ │ │ │ - if (forceReTile) { │ │ │ │ │ - if (zoomChanged && (this.transitionEffect === "resize" || this.gridResolution === resolution)) { │ │ │ │ │ - this.applyBackBuffer(resolution) │ │ │ │ │ - } │ │ │ │ │ - this.initGriddedTiles(bounds) │ │ │ │ │ - } else { │ │ │ │ │ - this.moveGriddedTiles() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getTileData: function(loc) { │ │ │ │ │ - var data = null, │ │ │ │ │ - x = loc.lon, │ │ │ │ │ - y = loc.lat, │ │ │ │ │ - numRows = this.grid.length; │ │ │ │ │ - if (this.map && numRows) { │ │ │ │ │ - var res = this.map.getResolution(), │ │ │ │ │ - tileWidth = this.tileSize.w, │ │ │ │ │ - tileHeight = this.tileSize.h, │ │ │ │ │ - bounds = this.grid[0][0].bounds, │ │ │ │ │ - left = bounds.left, │ │ │ │ │ - top = bounds.top; │ │ │ │ │ - if (x < left) { │ │ │ │ │ - if (this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - var worldWidth = this.map.getMaxExtent().getWidth(); │ │ │ │ │ - var worldsAway = Math.ceil((left - x) / worldWidth); │ │ │ │ │ - x += worldWidth * worldsAway │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var dtx = (x - left) / (res * tileWidth); │ │ │ │ │ - var dty = (top - y) / (res * tileHeight); │ │ │ │ │ - var col = Math.floor(dtx); │ │ │ │ │ - var row = Math.floor(dty); │ │ │ │ │ - if (row >= 0 && row < numRows) { │ │ │ │ │ - var tile = this.grid[row][col]; │ │ │ │ │ - if (tile) { │ │ │ │ │ - data = { │ │ │ │ │ - tile: tile, │ │ │ │ │ - i: Math.floor((dtx - col) * tileWidth), │ │ │ │ │ - j: Math.floor((dty - row) * tileHeight) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return data │ │ │ │ │ - }, │ │ │ │ │ - destroyTile: function(tile) { │ │ │ │ │ - this.removeTileMonitoringHooks(tile); │ │ │ │ │ - tile.destroy() │ │ │ │ │ - }, │ │ │ │ │ - getServerResolution: function(resolution) { │ │ │ │ │ - var distance = Number.POSITIVE_INFINITY; │ │ │ │ │ - resolution = resolution || this.map.getResolution(); │ │ │ │ │ - if (this.serverResolutions && OpenLayers.Util.indexOf(this.serverResolutions, resolution) === -1) { │ │ │ │ │ - var i, newDistance, newResolution, serverResolution; │ │ │ │ │ - for (i = this.serverResolutions.length - 1; i >= 0; i--) { │ │ │ │ │ - newResolution = this.serverResolutions[i]; │ │ │ │ │ - newDistance = Math.abs(newResolution - resolution); │ │ │ │ │ - if (newDistance > distance) { │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - distance = newDistance; │ │ │ │ │ - serverResolution = newResolution │ │ │ │ │ + applies = this.filter.evaluate(context) │ │ │ │ │ } │ │ │ │ │ - resolution = serverResolution │ │ │ │ │ } │ │ │ │ │ - return resolution │ │ │ │ │ - }, │ │ │ │ │ - getServerZoom: function() { │ │ │ │ │ - var resolution = this.getServerResolution(); │ │ │ │ │ - return this.serverResolutions ? OpenLayers.Util.indexOf(this.serverResolutions, resolution) : this.map.getZoomForResolution(resolution) + (this.zoomOffset || 0) │ │ │ │ │ + return applies │ │ │ │ │ }, │ │ │ │ │ - applyBackBuffer: function(resolution) { │ │ │ │ │ - if (this.backBufferTimerId !== null) { │ │ │ │ │ - this.removeBackBuffer() │ │ │ │ │ - } │ │ │ │ │ - var backBuffer = this.backBuffer; │ │ │ │ │ - if (!backBuffer) { │ │ │ │ │ - backBuffer = this.createBackBuffer(); │ │ │ │ │ - if (!backBuffer) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (resolution === this.gridResolution) { │ │ │ │ │ - this.div.insertBefore(backBuffer, this.div.firstChild) │ │ │ │ │ - } else { │ │ │ │ │ - this.map.baseLayer.div.parentNode.insertBefore(backBuffer, this.map.baseLayer.div) │ │ │ │ │ - } │ │ │ │ │ - this.backBuffer = backBuffer; │ │ │ │ │ - var topLeftTileBounds = this.grid[0][0].bounds; │ │ │ │ │ - this.backBufferLonLat = { │ │ │ │ │ - lon: topLeftTileBounds.left, │ │ │ │ │ - lat: topLeftTileBounds.top │ │ │ │ │ - }; │ │ │ │ │ - this.backBufferResolution = this.gridResolution │ │ │ │ │ - } │ │ │ │ │ - var ratio = this.backBufferResolution / resolution; │ │ │ │ │ - var tiles = backBuffer.childNodes, │ │ │ │ │ - tile; │ │ │ │ │ - for (var i = tiles.length - 1; i >= 0; --i) { │ │ │ │ │ - tile = tiles[i]; │ │ │ │ │ - tile.style.top = (ratio * tile._i * tile._h | 0) + "px"; │ │ │ │ │ - tile.style.left = (ratio * tile._j * tile._w | 0) + "px"; │ │ │ │ │ - tile.style.width = Math.round(ratio * tile._w) + "px"; │ │ │ │ │ - tile.style.height = Math.round(ratio * tile._h) + "px" │ │ │ │ │ + getContext: function(feature) { │ │ │ │ │ + var context = this.context; │ │ │ │ │ + if (!context) { │ │ │ │ │ + context = feature.attributes || feature.data │ │ │ │ │ } │ │ │ │ │ - var position = this.getViewPortPxFromLonLat(this.backBufferLonLat, resolution); │ │ │ │ │ - var leftOffset = this.map.layerContainerOriginPx.x; │ │ │ │ │ - var topOffset = this.map.layerContainerOriginPx.y; │ │ │ │ │ - backBuffer.style.left = Math.round(position.x - leftOffset) + "px"; │ │ │ │ │ - backBuffer.style.top = Math.round(position.y - topOffset) + "px" │ │ │ │ │ - }, │ │ │ │ │ - createBackBuffer: function() { │ │ │ │ │ - var backBuffer; │ │ │ │ │ - if (this.grid.length > 0) { │ │ │ │ │ - backBuffer = document.createElement("div"); │ │ │ │ │ - backBuffer.id = this.div.id + "_bb"; │ │ │ │ │ - backBuffer.className = "olBackBuffer"; │ │ │ │ │ - backBuffer.style.position = "absolute"; │ │ │ │ │ - var map = this.map; │ │ │ │ │ - backBuffer.style.zIndex = this.transitionEffect === "resize" ? this.getZIndex() - 1 : map.Z_INDEX_BASE.BaseLayer - (map.getNumLayers() - map.getLayerIndex(this)); │ │ │ │ │ - for (var i = 0, lenI = this.grid.length; i < lenI; i++) { │ │ │ │ │ - for (var j = 0, lenJ = this.grid[i].length; j < lenJ; j++) { │ │ │ │ │ - var tile = this.grid[i][j], │ │ │ │ │ - markup = this.grid[i][j].createBackBuffer(); │ │ │ │ │ - if (markup) { │ │ │ │ │ - markup._i = i; │ │ │ │ │ - markup._j = j; │ │ │ │ │ - markup._w = tile.size.w; │ │ │ │ │ - markup._h = tile.size.h; │ │ │ │ │ - markup.id = tile.id + "_bb"; │ │ │ │ │ - backBuffer.appendChild(markup) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + if (typeof this.context == "function") { │ │ │ │ │ + context = this.context(feature) │ │ │ │ │ } │ │ │ │ │ - return backBuffer │ │ │ │ │ + return context │ │ │ │ │ }, │ │ │ │ │ - removeBackBuffer: function() { │ │ │ │ │ - if (this._transitionElement) { │ │ │ │ │ - for (var i = this.transitionendEvents.length - 1; i >= 0; --i) { │ │ │ │ │ - OpenLayers.Event.stopObserving(this._transitionElement, this.transitionendEvents[i], this._removeBackBuffer) │ │ │ │ │ - } │ │ │ │ │ - delete this._transitionElement │ │ │ │ │ - } │ │ │ │ │ - if (this.backBuffer) { │ │ │ │ │ - if (this.backBuffer.parentNode) { │ │ │ │ │ - this.backBuffer.parentNode.removeChild(this.backBuffer) │ │ │ │ │ - } │ │ │ │ │ - this.backBuffer = null; │ │ │ │ │ - this.backBufferResolution = null; │ │ │ │ │ - if (this.backBufferTimerId !== null) { │ │ │ │ │ - window.clearTimeout(this.backBufferTimerId); │ │ │ │ │ - this.backBufferTimerId = null │ │ │ │ │ + clone: function() { │ │ │ │ │ + var options = OpenLayers.Util.extend({}, this); │ │ │ │ │ + if (this.symbolizers) { │ │ │ │ │ + var len = this.symbolizers.length; │ │ │ │ │ + options.symbolizers = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + options.symbolizers[i] = this.symbolizers[i].clone() │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - moveByPx: function(dx, dy) { │ │ │ │ │ - if (!this.singleTile) { │ │ │ │ │ - this.moveGriddedTiles() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setTileSize: function(size) { │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - size.h = parseInt(size.h * this.ratio, 10); │ │ │ │ │ - size.w = parseInt(size.w * this.ratio, 10) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.setTileSize.apply(this, [size]) │ │ │ │ │ - }, │ │ │ │ │ - getTilesBounds: function() { │ │ │ │ │ - var bounds = null; │ │ │ │ │ - var length = this.grid.length; │ │ │ │ │ - if (length) { │ │ │ │ │ - var bottomLeftTileBounds = this.grid[length - 1][0].bounds, │ │ │ │ │ - width = this.grid[0].length * bottomLeftTileBounds.getWidth(), │ │ │ │ │ - height = this.grid.length * bottomLeftTileBounds.getHeight(); │ │ │ │ │ - bounds = new OpenLayers.Bounds(bottomLeftTileBounds.left, bottomLeftTileBounds.bottom, bottomLeftTileBounds.left + width, bottomLeftTileBounds.bottom + height) │ │ │ │ │ - } │ │ │ │ │ - return bounds │ │ │ │ │ - }, │ │ │ │ │ - initSingleTile: function(bounds) { │ │ │ │ │ - this.events.triggerEvent("retile"); │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var tileWidth = bounds.getWidth() * this.ratio; │ │ │ │ │ - var tileHeight = bounds.getHeight() * this.ratio; │ │ │ │ │ - var tileBounds = new OpenLayers.Bounds(center.lon - tileWidth / 2, center.lat - tileHeight / 2, center.lon + tileWidth / 2, center.lat + tileHeight / 2); │ │ │ │ │ - var px = this.map.getLayerPxFromLonLat({ │ │ │ │ │ - lon: tileBounds.left, │ │ │ │ │ - lat: tileBounds.top │ │ │ │ │ - }); │ │ │ │ │ - if (!this.grid.length) { │ │ │ │ │ - this.grid[0] = [] │ │ │ │ │ - } │ │ │ │ │ - var tile = this.grid[0][0]; │ │ │ │ │ - if (!tile) { │ │ │ │ │ - tile = this.addTile(tileBounds, px); │ │ │ │ │ - this.addTileMonitoringHooks(tile); │ │ │ │ │ - tile.draw(); │ │ │ │ │ - this.grid[0][0] = tile │ │ │ │ │ } else { │ │ │ │ │ - tile.moveTo(tileBounds, px) │ │ │ │ │ - } │ │ │ │ │ - this.removeExcessTiles(1, 1); │ │ │ │ │ - this.gridResolution = this.getServerResolution() │ │ │ │ │ - }, │ │ │ │ │ - calculateGridLayout: function(bounds, origin, resolution) { │ │ │ │ │ - var tilelon = resolution * this.tileSize.w; │ │ │ │ │ - var tilelat = resolution * this.tileSize.h; │ │ │ │ │ - var offsetlon = bounds.left - origin.lon; │ │ │ │ │ - var tilecol = Math.floor(offsetlon / tilelon) - this.buffer; │ │ │ │ │ - var rowSign = this.rowSign; │ │ │ │ │ - var offsetlat = rowSign * (origin.lat - bounds.top + tilelat); │ │ │ │ │ - var tilerow = Math[~rowSign ? "floor" : "ceil"](offsetlat / tilelat) - this.buffer * rowSign; │ │ │ │ │ - return { │ │ │ │ │ - tilelon: tilelon, │ │ │ │ │ - tilelat: tilelat, │ │ │ │ │ - startcol: tilecol, │ │ │ │ │ - startrow: tilerow │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getTileOrigin: function() { │ │ │ │ │ - var origin = this.tileOrigin; │ │ │ │ │ - if (!origin) { │ │ │ │ │ - var extent = this.getMaxExtent(); │ │ │ │ │ - var edges = { │ │ │ │ │ - tl: ["left", "top"], │ │ │ │ │ - tr: ["right", "top"], │ │ │ │ │ - bl: ["left", "bottom"], │ │ │ │ │ - br: ["right", "bottom"] │ │ │ │ │ - } [this.tileOriginCorner]; │ │ │ │ │ - origin = new OpenLayers.LonLat(extent[edges[0]], extent[edges[1]]) │ │ │ │ │ - } │ │ │ │ │ - return origin │ │ │ │ │ - }, │ │ │ │ │ - getTileBoundsForGridIndex: function(row, col) { │ │ │ │ │ - var origin = this.getTileOrigin(); │ │ │ │ │ - var tileLayout = this.gridLayout; │ │ │ │ │ - var tilelon = tileLayout.tilelon; │ │ │ │ │ - var tilelat = tileLayout.tilelat; │ │ │ │ │ - var startcol = tileLayout.startcol; │ │ │ │ │ - var startrow = tileLayout.startrow; │ │ │ │ │ - var rowSign = this.rowSign; │ │ │ │ │ - return new OpenLayers.Bounds(origin.lon + (startcol + col) * tilelon, origin.lat - (startrow + row * rowSign) * tilelat * rowSign, origin.lon + (startcol + col + 1) * tilelon, origin.lat - (startrow + (row - 1) * rowSign) * tilelat * rowSign) │ │ │ │ │ - }, │ │ │ │ │ - initGriddedTiles: function(bounds) { │ │ │ │ │ - this.events.triggerEvent("retile"); │ │ │ │ │ - var viewSize = this.map.getSize(); │ │ │ │ │ - var origin = this.getTileOrigin(); │ │ │ │ │ - var resolution = this.map.getResolution(), │ │ │ │ │ - serverResolution = this.getServerResolution(), │ │ │ │ │ - ratio = resolution / serverResolution, │ │ │ │ │ - tileSize = { │ │ │ │ │ - w: this.tileSize.w / ratio, │ │ │ │ │ - h: this.tileSize.h / ratio │ │ │ │ │ - }; │ │ │ │ │ - var minRows = Math.ceil(viewSize.h / tileSize.h) + 2 * this.buffer + 1; │ │ │ │ │ - var minCols = Math.ceil(viewSize.w / tileSize.w) + 2 * this.buffer + 1; │ │ │ │ │ - var tileLayout = this.calculateGridLayout(bounds, origin, serverResolution); │ │ │ │ │ - this.gridLayout = tileLayout; │ │ │ │ │ - var tilelon = tileLayout.tilelon; │ │ │ │ │ - var tilelat = tileLayout.tilelat; │ │ │ │ │ - var layerContainerDivLeft = this.map.layerContainerOriginPx.x; │ │ │ │ │ - var layerContainerDivTop = this.map.layerContainerOriginPx.y; │ │ │ │ │ - var tileBounds = this.getTileBoundsForGridIndex(0, 0); │ │ │ │ │ - var startPx = this.map.getViewPortPxFromLonLat(new OpenLayers.LonLat(tileBounds.left, tileBounds.top)); │ │ │ │ │ - startPx.x = Math.round(startPx.x) - layerContainerDivLeft; │ │ │ │ │ - startPx.y = Math.round(startPx.y) - layerContainerDivTop; │ │ │ │ │ - var tileData = [], │ │ │ │ │ - center = this.map.getCenter(); │ │ │ │ │ - var rowidx = 0; │ │ │ │ │ - do { │ │ │ │ │ - var row = this.grid[rowidx]; │ │ │ │ │ - if (!row) { │ │ │ │ │ - row = []; │ │ │ │ │ - this.grid.push(row) │ │ │ │ │ - } │ │ │ │ │ - var colidx = 0; │ │ │ │ │ - do { │ │ │ │ │ - tileBounds = this.getTileBoundsForGridIndex(rowidx, colidx); │ │ │ │ │ - var px = startPx.clone(); │ │ │ │ │ - px.x = px.x + colidx * Math.round(tileSize.w); │ │ │ │ │ - px.y = px.y + rowidx * Math.round(tileSize.h); │ │ │ │ │ - var tile = row[colidx]; │ │ │ │ │ - if (!tile) { │ │ │ │ │ - tile = this.addTile(tileBounds, px); │ │ │ │ │ - this.addTileMonitoringHooks(tile); │ │ │ │ │ - row.push(tile) │ │ │ │ │ - } else { │ │ │ │ │ - tile.moveTo(tileBounds, px, false) │ │ │ │ │ - } │ │ │ │ │ - var tileCenter = tileBounds.getCenterLonLat(); │ │ │ │ │ - tileData.push({ │ │ │ │ │ - tile: tile, │ │ │ │ │ - distance: Math.pow(tileCenter.lon - center.lon, 2) + Math.pow(tileCenter.lat - center.lat, 2) │ │ │ │ │ - }); │ │ │ │ │ - colidx += 1 │ │ │ │ │ - } while (tileBounds.right <= bounds.right + tilelon * this.buffer || colidx < minCols); │ │ │ │ │ - rowidx += 1 │ │ │ │ │ - } while (tileBounds.bottom >= bounds.bottom - tilelat * this.buffer || rowidx < minRows); │ │ │ │ │ - this.removeExcessTiles(rowidx, colidx); │ │ │ │ │ - var resolution = this.getServerResolution(); │ │ │ │ │ - this.gridResolution = resolution; │ │ │ │ │ - tileData.sort(function(a, b) { │ │ │ │ │ - return a.distance - b.distance │ │ │ │ │ - }); │ │ │ │ │ - for (var i = 0, ii = tileData.length; i < ii; ++i) { │ │ │ │ │ - tileData[i].tile.draw() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getMaxExtent: function() { │ │ │ │ │ - return this.maxExtent │ │ │ │ │ - }, │ │ │ │ │ - addTile: function(bounds, position) { │ │ │ │ │ - var tile = new this.tileClass(this, position, bounds, null, this.tileSize, this.tileOptions); │ │ │ │ │ - this.events.triggerEvent("addtile", { │ │ │ │ │ - tile: tile │ │ │ │ │ - }); │ │ │ │ │ - return tile │ │ │ │ │ - }, │ │ │ │ │ - addTileMonitoringHooks: function(tile) { │ │ │ │ │ - var replacingCls = "olTileReplacing"; │ │ │ │ │ - tile.onLoadStart = function() { │ │ │ │ │ - if (this.loading === false) { │ │ │ │ │ - this.loading = true; │ │ │ │ │ - this.events.triggerEvent("loadstart") │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("tileloadstart", { │ │ │ │ │ - tile: tile │ │ │ │ │ - }); │ │ │ │ │ - this.numLoadingTiles++; │ │ │ │ │ - if (!this.singleTile && this.backBuffer && this.gridResolution === this.backBufferResolution) { │ │ │ │ │ - OpenLayers.Element.addClass(tile.getTile(), replacingCls) │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - tile.onLoadEnd = function(evt) { │ │ │ │ │ - this.numLoadingTiles--; │ │ │ │ │ - var aborted = evt.type === "unload"; │ │ │ │ │ - this.events.triggerEvent("tileloaded", { │ │ │ │ │ - tile: tile, │ │ │ │ │ - aborted: aborted │ │ │ │ │ - }); │ │ │ │ │ - if (!this.singleTile && !aborted && this.backBuffer && this.gridResolution === this.backBufferResolution) { │ │ │ │ │ - var tileDiv = tile.getTile(); │ │ │ │ │ - if (OpenLayers.Element.getStyle(tileDiv, "display") === "none") { │ │ │ │ │ - var bufferTile = document.getElementById(tile.id + "_bb"); │ │ │ │ │ - if (bufferTile) { │ │ │ │ │ - bufferTile.parentNode.removeChild(bufferTile) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Element.removeClass(tileDiv, replacingCls) │ │ │ │ │ - } │ │ │ │ │ - if (this.numLoadingTiles === 0) { │ │ │ │ │ - if (this.backBuffer) { │ │ │ │ │ - if (this.backBuffer.childNodes.length === 0) { │ │ │ │ │ - this.removeBackBuffer() │ │ │ │ │ - } else { │ │ │ │ │ - this._transitionElement = aborted ? this.div.lastChild : tile.imgDiv; │ │ │ │ │ - var transitionendEvents = this.transitionendEvents; │ │ │ │ │ - for (var i = transitionendEvents.length - 1; i >= 0; --i) { │ │ │ │ │ - OpenLayers.Event.observe(this._transitionElement, transitionendEvents[i], this._removeBackBuffer) │ │ │ │ │ - } │ │ │ │ │ - this.backBufferTimerId = window.setTimeout(this._removeBackBuffer, this.removeBackBufferDelay) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.loading = false; │ │ │ │ │ - this.events.triggerEvent("loadend") │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - tile.onLoadError = function() { │ │ │ │ │ - this.events.triggerEvent("tileerror", { │ │ │ │ │ - tile: tile │ │ │ │ │ - }) │ │ │ │ │ - }; │ │ │ │ │ - tile.events.on({ │ │ │ │ │ - loadstart: tile.onLoadStart, │ │ │ │ │ - loadend: tile.onLoadEnd, │ │ │ │ │ - unload: tile.onLoadEnd, │ │ │ │ │ - loaderror: tile.onLoadError, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - removeTileMonitoringHooks: function(tile) { │ │ │ │ │ - tile.unload(); │ │ │ │ │ - tile.events.un({ │ │ │ │ │ - loadstart: tile.onLoadStart, │ │ │ │ │ - loadend: tile.onLoadEnd, │ │ │ │ │ - unload: tile.onLoadEnd, │ │ │ │ │ - loaderror: tile.onLoadError, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - moveGriddedTiles: function() { │ │ │ │ │ - var buffer = this.buffer + 1; │ │ │ │ │ - while (true) { │ │ │ │ │ - var tlTile = this.grid[0][0]; │ │ │ │ │ - var tlViewPort = { │ │ │ │ │ - x: tlTile.position.x + this.map.layerContainerOriginPx.x, │ │ │ │ │ - y: tlTile.position.y + this.map.layerContainerOriginPx.y │ │ │ │ │ - }; │ │ │ │ │ - var ratio = this.getServerResolution() / this.map.getResolution(); │ │ │ │ │ - var tileSize = { │ │ │ │ │ - w: Math.round(this.tileSize.w * ratio), │ │ │ │ │ - h: Math.round(this.tileSize.h * ratio) │ │ │ │ │ - }; │ │ │ │ │ - if (tlViewPort.x > -tileSize.w * (buffer - 1)) { │ │ │ │ │ - this.shiftColumn(true, tileSize) │ │ │ │ │ - } else if (tlViewPort.x < -tileSize.w * buffer) { │ │ │ │ │ - this.shiftColumn(false, tileSize) │ │ │ │ │ - } else if (tlViewPort.y > -tileSize.h * (buffer - 1)) { │ │ │ │ │ - this.shiftRow(true, tileSize) │ │ │ │ │ - } else if (tlViewPort.y < -tileSize.h * buffer) { │ │ │ │ │ - this.shiftRow(false, tileSize) │ │ │ │ │ - } else { │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - shiftRow: function(prepend, tileSize) { │ │ │ │ │ - var grid = this.grid; │ │ │ │ │ - var rowIndex = prepend ? 0 : grid.length - 1; │ │ │ │ │ - var sign = prepend ? -1 : 1; │ │ │ │ │ - var rowSign = this.rowSign; │ │ │ │ │ - var tileLayout = this.gridLayout; │ │ │ │ │ - tileLayout.startrow += sign * rowSign; │ │ │ │ │ - var modelRow = grid[rowIndex]; │ │ │ │ │ - var row = grid[prepend ? "pop" : "shift"](); │ │ │ │ │ - for (var i = 0, len = row.length; i < len; i++) { │ │ │ │ │ - var tile = row[i]; │ │ │ │ │ - var position = modelRow[i].position.clone(); │ │ │ │ │ - position.y += tileSize.h * sign; │ │ │ │ │ - tile.moveTo(this.getTileBoundsForGridIndex(rowIndex, i), position) │ │ │ │ │ - } │ │ │ │ │ - grid[prepend ? "unshift" : "push"](row) │ │ │ │ │ - }, │ │ │ │ │ - shiftColumn: function(prepend, tileSize) { │ │ │ │ │ - var grid = this.grid; │ │ │ │ │ - var colIndex = prepend ? 0 : grid[0].length - 1; │ │ │ │ │ - var sign = prepend ? -1 : 1; │ │ │ │ │ - var tileLayout = this.gridLayout; │ │ │ │ │ - tileLayout.startcol += sign; │ │ │ │ │ - for (var i = 0, len = grid.length; i < len; i++) { │ │ │ │ │ - var row = grid[i]; │ │ │ │ │ - var position = row[colIndex].position.clone(); │ │ │ │ │ - var tile = row[prepend ? "pop" : "shift"](); │ │ │ │ │ - position.x += tileSize.w * sign; │ │ │ │ │ - tile.moveTo(this.getTileBoundsForGridIndex(i, colIndex), position); │ │ │ │ │ - row[prepend ? "unshift" : "push"](tile) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - removeExcessTiles: function(rows, columns) { │ │ │ │ │ - var i, l; │ │ │ │ │ - while (this.grid.length > rows) { │ │ │ │ │ - var row = this.grid.pop(); │ │ │ │ │ - for (i = 0, l = row.length; i < l; i++) { │ │ │ │ │ - var tile = row[i]; │ │ │ │ │ - this.destroyTile(tile) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - for (i = 0, l = this.grid.length; i < l; i++) { │ │ │ │ │ - while (this.grid[i].length > columns) { │ │ │ │ │ - var row = this.grid[i]; │ │ │ │ │ - var tile = row.pop(); │ │ │ │ │ - this.destroyTile(tile) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - this.clearGrid(); │ │ │ │ │ - this.setTileSize() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getTileBounds: function(viewPortPx) { │ │ │ │ │ - var maxExtent = this.maxExtent; │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var tileMapWidth = resolution * this.tileSize.w; │ │ │ │ │ - var tileMapHeight = resolution * this.tileSize.h; │ │ │ │ │ - var mapPoint = this.getLonLatFromViewPortPx(viewPortPx); │ │ │ │ │ - var tileLeft = maxExtent.left + tileMapWidth * Math.floor((mapPoint.lon - maxExtent.left) / tileMapWidth); │ │ │ │ │ - var tileBottom = maxExtent.bottom + tileMapHeight * Math.floor((mapPoint.lat - maxExtent.bottom) / tileMapHeight); │ │ │ │ │ - return new OpenLayers.Bounds(tileLeft, tileBottom, tileLeft + tileMapWidth, tileBottom + tileMapHeight) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Grid" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.XYZ = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - sphericalMercator: false, │ │ │ │ │ - zoomOffset: 0, │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - if (options && options.sphericalMercator || this.sphericalMercator) { │ │ │ │ │ - options = OpenLayers.Util.extend({ │ │ │ │ │ - projection: "EPSG:900913", │ │ │ │ │ - numZoomLevels: 19 │ │ │ │ │ - }, options) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name || this.name, url || this.url, {}, options]) │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.XYZ(this.name, this.url, this.getOptions()) │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var xyz = this.getXYZ(bounds); │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - var s = "" + xyz.x + xyz.y + xyz.z; │ │ │ │ │ - url = this.selectUrl(s, url) │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.String.format(url, xyz) │ │ │ │ │ - }, │ │ │ │ │ - getXYZ: function(bounds) { │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.maxExtent.left) / (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((this.maxExtent.top - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getServerZoom(); │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - var limit = Math.pow(2, z); │ │ │ │ │ - x = (x % limit + limit) % limit │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - x: x, │ │ │ │ │ - y: y, │ │ │ │ │ - z: z │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (!this.tileOrigin) { │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, this.maxExtent.bottom) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.XYZ" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.OSM = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ - name: "OpenStreetMap", │ │ │ │ │ - url: ["http://a.tile.openstreetmap.org/${z}/${x}/${y}.png", "http://b.tile.openstreetmap.org/${z}/${x}/${y}.png", "http://c.tile.openstreetmap.org/${z}/${x}/${y}.png"], │ │ │ │ │ - attribution: "© <a href='http://www.openstreetmap.org/copyright'>OpenStreetMap</a> contributors", │ │ │ │ │ - sphericalMercator: true, │ │ │ │ │ - wrapDateLine: true, │ │ │ │ │ - tileOptions: null, │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - crossOriginKeyword: "anonymous" │ │ │ │ │ - }, this.options && this.options.tileOptions) │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.OSM(this.name, this.url, this.getOptions()) │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.OSM" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Bing = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ - key: null, │ │ │ │ │ - serverResolutions: [156543.03390625, 78271.516953125, 39135.7584765625, 19567.87923828125, 9783.939619140625, 4891.9698095703125, 2445.9849047851562, 1222.9924523925781, 611.4962261962891, 305.74811309814453, 152.87405654907226, 76.43702827453613, 38.218514137268066, 19.109257068634033, 9.554628534317017, 4.777314267158508, 2.388657133579254, 1.194328566789627, .5971642833948135, .29858214169740677, .14929107084870338, .07464553542435169], │ │ │ │ │ - attributionTemplate: '<span class="olBingAttribution ${type}">' + '<div><a target="_blank" href="http://www.bing.com/maps/">' + '<img src="${logo}" /></a></div>${copyrights}' + '<a style="white-space: nowrap" target="_blank" ' + 'href="http://www.microsoft.com/maps/product/terms.html">' + "Terms of Use</a></span>", │ │ │ │ │ - metadata: null, │ │ │ │ │ - protocolRegex: /^http:/i, │ │ │ │ │ - type: "Road", │ │ │ │ │ - culture: "en-US", │ │ │ │ │ - metadataParams: null, │ │ │ │ │ - tileOptions: null, │ │ │ │ │ - protocol: ~window.location.href.indexOf("http") ? "" : "http:", │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - sphericalMercator: true │ │ │ │ │ - }, options); │ │ │ │ │ - var name = options.name || "Bing " + (options.type || this.type); │ │ │ │ │ - var newArgs = [name, null, options]; │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, newArgs); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - crossOriginKeyword: "anonymous" │ │ │ │ │ - }, this.options.tileOptions); │ │ │ │ │ - this.loadMetadata() │ │ │ │ │ - }, │ │ │ │ │ - loadMetadata: function() { │ │ │ │ │ - this._callbackId = "_callback_" + this.id.replace(/\./g, "_"); │ │ │ │ │ - window[this._callbackId] = OpenLayers.Function.bind(OpenLayers.Layer.Bing.processMetadata, this); │ │ │ │ │ - var params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - key: this.key, │ │ │ │ │ - jsonp: this._callbackId, │ │ │ │ │ - include: "ImageryProviders" │ │ │ │ │ - }, this.metadataParams); │ │ │ │ │ - var url = this.protocol + "//dev.virtualearth.net/REST/v1/Imagery/Metadata/" + this.type + "?" + OpenLayers.Util.getParameterString(params); │ │ │ │ │ - var script = document.createElement("script"); │ │ │ │ │ - script.type = "text/javascript"; │ │ │ │ │ - script.src = url; │ │ │ │ │ - script.id = this._callbackId; │ │ │ │ │ - document.getElementsByTagName("head")[0].appendChild(script) │ │ │ │ │ - }, │ │ │ │ │ - initLayer: function() { │ │ │ │ │ - var res = this.metadata.resourceSets[0].resources[0]; │ │ │ │ │ - var url = res.imageUrl.replace("{quadkey}", "${quadkey}"); │ │ │ │ │ - url = url.replace("{culture}", this.culture); │ │ │ │ │ - url = url.replace(this.protocolRegex, this.protocol); │ │ │ │ │ - this.url = []; │ │ │ │ │ - for (var i = 0; i < res.imageUrlSubdomains.length; ++i) { │ │ │ │ │ - this.url.push(url.replace("{subdomain}", res.imageUrlSubdomains[i])) │ │ │ │ │ - } │ │ │ │ │ - this.addOptions({ │ │ │ │ │ - maxResolution: Math.min(this.serverResolutions[res.zoomMin], this.maxResolution || Number.POSITIVE_INFINITY), │ │ │ │ │ - numZoomLevels: Math.min(res.zoomMax + 1 - res.zoomMin, this.numZoomLevels) │ │ │ │ │ - }, true); │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - this.redraw() │ │ │ │ │ - } │ │ │ │ │ - this.updateAttribution() │ │ │ │ │ - }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - if (!this.url) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var xyz = this.getXYZ(bounds), │ │ │ │ │ - x = xyz.x, │ │ │ │ │ - y = xyz.y, │ │ │ │ │ - z = xyz.z; │ │ │ │ │ - var quadDigits = []; │ │ │ │ │ - for (var i = z; i > 0; --i) { │ │ │ │ │ - var digit = "0"; │ │ │ │ │ - var mask = 1 << i - 1; │ │ │ │ │ - if ((x & mask) != 0) { │ │ │ │ │ - digit++ │ │ │ │ │ - } │ │ │ │ │ - if ((y & mask) != 0) { │ │ │ │ │ - digit++; │ │ │ │ │ - digit++ │ │ │ │ │ - } │ │ │ │ │ - quadDigits.push(digit) │ │ │ │ │ - } │ │ │ │ │ - var quadKey = quadDigits.join(""); │ │ │ │ │ - var url = this.selectUrl("" + x + y + z, this.url); │ │ │ │ │ - return OpenLayers.String.format(url, { │ │ │ │ │ - quadkey: quadKey │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - updateAttribution: function() { │ │ │ │ │ - var metadata = this.metadata; │ │ │ │ │ - if (!metadata.resourceSets || !this.map || !this.map.center) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var res = metadata.resourceSets[0].resources[0]; │ │ │ │ │ - var extent = this.map.getExtent().transform(this.map.getProjectionObject(), new OpenLayers.Projection("EPSG:4326")); │ │ │ │ │ - var providers = res.imageryProviders || [], │ │ │ │ │ - zoom = OpenLayers.Util.indexOf(this.serverResolutions, this.getServerResolution()), │ │ │ │ │ - copyrights = "", │ │ │ │ │ - provider, i, ii, j, jj, bbox, coverage; │ │ │ │ │ - for (i = 0, ii = providers.length; i < ii; ++i) { │ │ │ │ │ - provider = providers[i]; │ │ │ │ │ - for (j = 0, jj = provider.coverageAreas.length; j < jj; ++j) { │ │ │ │ │ - coverage = provider.coverageAreas[j]; │ │ │ │ │ - bbox = OpenLayers.Bounds.fromArray(coverage.bbox, true); │ │ │ │ │ - if (extent.intersectsBounds(bbox) && zoom <= coverage.zoomMax && zoom >= coverage.zoomMin) { │ │ │ │ │ - copyrights += provider.attribution + " " │ │ │ │ │ + options.symbolizer = {}; │ │ │ │ │ + var value, type; │ │ │ │ │ + for (var key in this.symbolizer) { │ │ │ │ │ + value = this.symbolizer[key]; │ │ │ │ │ + type = typeof value; │ │ │ │ │ + if (type === "object") { │ │ │ │ │ + options.symbolizer[key] = OpenLayers.Util.extend({}, value) │ │ │ │ │ + } else if (type === "string") { │ │ │ │ │ + options.symbolizer[key] = value │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var logo = metadata.brandLogoUri.replace(this.protocolRegex, this.protocol); │ │ │ │ │ - this.attribution = OpenLayers.String.format(this.attributionTemplate, { │ │ │ │ │ - type: this.type.toLowerCase(), │ │ │ │ │ - logo: logo, │ │ │ │ │ - copyrights: copyrights │ │ │ │ │ - }); │ │ │ │ │ - this.map && this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "attribution" │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - setMap: function() { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.register("moveend", this, this.updateAttribution) │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Bing(this.options) │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.map && this.map.events.unregister("moveend", this, this.updateAttribution); │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.destroy.apply(this, arguments) │ │ │ │ │ + options.filter = this.filter && this.filter.clone(); │ │ │ │ │ + options.context = this.context && OpenLayers.Util.extend({}, this.context); │ │ │ │ │ + return new OpenLayers.Rule(options) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Bing" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Rule" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.Bing.processMetadata = function(metadata) { │ │ │ │ │ - this.metadata = metadata; │ │ │ │ │ - this.initLayer(); │ │ │ │ │ - var script = document.getElementById(this._callbackId); │ │ │ │ │ - script.parentNode.removeChild(script); │ │ │ │ │ - window[this._callbackId] = undefined; │ │ │ │ │ - delete this._callbackId │ │ │ │ │ -}; │ │ │ │ │ OpenLayers.Renderer = OpenLayers.Class({ │ │ │ │ │ container: null, │ │ │ │ │ root: null, │ │ │ │ │ extent: null, │ │ │ │ │ locked: false, │ │ │ │ │ size: null, │ │ │ │ │ resolution: null, │ │ │ │ │ @@ -6165,2159 +4423,2105 @@ │ │ │ │ │ OpenLayers.Renderer.symbol = { │ │ │ │ │ star: [350, 75, 379, 161, 469, 161, 397, 215, 423, 301, 350, 250, 277, 301, 303, 215, 231, 161, 321, 161, 350, 75], │ │ │ │ │ cross: [4, 0, 6, 0, 6, 4, 10, 4, 10, 6, 6, 6, 6, 10, 4, 10, 4, 6, 0, 6, 0, 4, 4, 4, 4, 0], │ │ │ │ │ x: [0, 0, 25, 0, 50, 35, 75, 0, 100, 0, 65, 50, 100, 100, 75, 100, 50, 65, 25, 100, 0, 100, 35, 50, 0, 0], │ │ │ │ │ square: [0, 0, 0, 1, 1, 1, 1, 0, 0, 0], │ │ │ │ │ triangle: [0, 10, 10, 10, 5, 0, 0, 10] │ │ │ │ │ }; │ │ │ │ │ -OpenLayers.Layer.Vector = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ - isBaseLayer: false, │ │ │ │ │ - isFixed: false, │ │ │ │ │ - features: null, │ │ │ │ │ - filter: null, │ │ │ │ │ - selectedFeatures: null, │ │ │ │ │ - unrenderedFeatures: null, │ │ │ │ │ - reportError: true, │ │ │ │ │ - style: null, │ │ │ │ │ - styleMap: null, │ │ │ │ │ - strategies: null, │ │ │ │ │ - protocol: null, │ │ │ │ │ - renderers: ["SVG", "VML", "Canvas"], │ │ │ │ │ - renderer: null, │ │ │ │ │ - rendererOptions: null, │ │ │ │ │ - geometryType: null, │ │ │ │ │ - drawn: false, │ │ │ │ │ - ratio: 1, │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ - this.assignRenderer() │ │ │ │ │ - } │ │ │ │ │ - if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ - this.renderer = null; │ │ │ │ │ - this.displayError() │ │ │ │ │ - } │ │ │ │ │ - if (!this.styleMap) { │ │ │ │ │ - this.styleMap = new OpenLayers.StyleMap │ │ │ │ │ +OpenLayers.ElementsIndexer = OpenLayers.Class({ │ │ │ │ │ + maxZIndex: null, │ │ │ │ │ + order: null, │ │ │ │ │ + indices: null, │ │ │ │ │ + compare: null, │ │ │ │ │ + initialize: function(yOrdering) { │ │ │ │ │ + this.compare = yOrdering ? OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_Y_ORDER : OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_DRAWING_ORDER; │ │ │ │ │ + this.clear() │ │ │ │ │ + }, │ │ │ │ │ + insert: function(newNode) { │ │ │ │ │ + if (this.exists(newNode)) { │ │ │ │ │ + this.remove(newNode) │ │ │ │ │ } │ │ │ │ │ - this.features = []; │ │ │ │ │ - this.selectedFeatures = []; │ │ │ │ │ - this.unrenderedFeatures = {}; │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - for (var i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - this.strategies[i].setLayer(this) │ │ │ │ │ + var nodeId = newNode.id; │ │ │ │ │ + this.determineZIndex(newNode); │ │ │ │ │ + var leftIndex = -1; │ │ │ │ │ + var rightIndex = this.order.length; │ │ │ │ │ + var middle; │ │ │ │ │ + while (rightIndex - leftIndex > 1) { │ │ │ │ │ + middle = parseInt((leftIndex + rightIndex) / 2); │ │ │ │ │ + var placement = this.compare(this, newNode, OpenLayers.Util.getElement(this.order[middle])); │ │ │ │ │ + if (placement > 0) { │ │ │ │ │ + leftIndex = middle │ │ │ │ │ + } else { │ │ │ │ │ + rightIndex = middle │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + this.order.splice(rightIndex, 0, nodeId); │ │ │ │ │ + this.indices[nodeId] = this.getZIndex(newNode); │ │ │ │ │ + return this.getNextElement(rightIndex) │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoDestroy) { │ │ │ │ │ - strategy.destroy() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.strategies = null │ │ │ │ │ - } │ │ │ │ │ - if (this.protocol) { │ │ │ │ │ - if (this.protocol.autoDestroy) { │ │ │ │ │ - this.protocol.destroy() │ │ │ │ │ + remove: function(node) { │ │ │ │ │ + var nodeId = node.id; │ │ │ │ │ + var arrayIndex = OpenLayers.Util.indexOf(this.order, nodeId); │ │ │ │ │ + if (arrayIndex >= 0) { │ │ │ │ │ + this.order.splice(arrayIndex, 1); │ │ │ │ │ + delete this.indices[nodeId]; │ │ │ │ │ + if (this.order.length > 0) { │ │ │ │ │ + var lastId = this.order[this.order.length - 1]; │ │ │ │ │ + this.maxZIndex = this.indices[lastId] │ │ │ │ │ + } else { │ │ │ │ │ + this.maxZIndex = 0 │ │ │ │ │ } │ │ │ │ │ - this.protocol = null │ │ │ │ │ - } │ │ │ │ │ - this.destroyFeatures(); │ │ │ │ │ - this.features = null; │ │ │ │ │ - this.selectedFeatures = null; │ │ │ │ │ - this.unrenderedFeatures = null; │ │ │ │ │ - if (this.renderer) { │ │ │ │ │ - this.renderer.destroy() │ │ │ │ │ } │ │ │ │ │ - this.renderer = null; │ │ │ │ │ - this.geometryType = null; │ │ │ │ │ - this.drawn = null; │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Vector(this.name, this.getOptions()) │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ - var features = this.features; │ │ │ │ │ - var len = features.length; │ │ │ │ │ - var clonedFeatures = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - clonedFeatures[i] = features[i].clone() │ │ │ │ │ - } │ │ │ │ │ - obj.features = clonedFeatures; │ │ │ │ │ - return obj │ │ │ │ │ + clear: function() { │ │ │ │ │ + this.order = []; │ │ │ │ │ + this.indices = {}; │ │ │ │ │ + this.maxZIndex = 0 │ │ │ │ │ }, │ │ │ │ │ - refresh: function(obj) { │ │ │ │ │ - if (this.calculateInRange() && this.visibility) { │ │ │ │ │ - this.events.triggerEvent("refresh", obj) │ │ │ │ │ + exists: function(node) { │ │ │ │ │ + return this.indices[node.id] != null │ │ │ │ │ + }, │ │ │ │ │ + getZIndex: function(node) { │ │ │ │ │ + return node._style.graphicZIndex │ │ │ │ │ + }, │ │ │ │ │ + determineZIndex: function(node) { │ │ │ │ │ + var zIndex = node._style.graphicZIndex; │ │ │ │ │ + if (zIndex == null) { │ │ │ │ │ + zIndex = this.maxZIndex; │ │ │ │ │ + node._style.graphicZIndex = zIndex │ │ │ │ │ + } else if (zIndex > this.maxZIndex) { │ │ │ │ │ + this.maxZIndex = zIndex │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - assignRenderer: function() { │ │ │ │ │ - for (var i = 0, len = this.renderers.length; i < len; i++) { │ │ │ │ │ - var rendererClass = this.renderers[i]; │ │ │ │ │ - var renderer = typeof rendererClass == "function" ? rendererClass : OpenLayers.Renderer[rendererClass]; │ │ │ │ │ - if (renderer && renderer.prototype.supported()) { │ │ │ │ │ - this.renderer = new renderer(this.div, this.rendererOptions); │ │ │ │ │ - break │ │ │ │ │ + getNextElement: function(index) { │ │ │ │ │ + var nextIndex = index + 1; │ │ │ │ │ + if (nextIndex < this.order.length) { │ │ │ │ │ + var nextElement = OpenLayers.Util.getElement(this.order[nextIndex]); │ │ │ │ │ + if (nextElement == undefined) { │ │ │ │ │ + nextElement = this.getNextElement(nextIndex) │ │ │ │ │ } │ │ │ │ │ + return nextElement │ │ │ │ │ + } else { │ │ │ │ │ + return null │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - displayError: function() { │ │ │ │ │ - if (this.reportError) { │ │ │ │ │ - OpenLayers.Console.userError(OpenLayers.i18n("browserNotSupported", { │ │ │ │ │ - renderers: this.renderers.join("\n") │ │ │ │ │ - })) │ │ │ │ │ + CLASS_NAME: "OpenLayers.ElementsIndexer" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.ElementsIndexer.IndexingMethods = { │ │ │ │ │ + Z_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var newZIndex = indexer.getZIndex(newNode); │ │ │ │ │ + var returnVal = 0; │ │ │ │ │ + if (nextNode) { │ │ │ │ │ + var nextZIndex = indexer.getZIndex(nextNode); │ │ │ │ │ + returnVal = newZIndex - nextZIndex │ │ │ │ │ } │ │ │ │ │ + return returnVal │ │ │ │ │ }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (!this.renderer) { │ │ │ │ │ - this.map.removeLayer(this) │ │ │ │ │ - } else { │ │ │ │ │ - this.renderer.map = this.map; │ │ │ │ │ - var newSize = this.map.getSize(); │ │ │ │ │ - newSize.w = newSize.w * this.ratio; │ │ │ │ │ - newSize.h = newSize.h * this.ratio; │ │ │ │ │ - this.renderer.setSize(newSize) │ │ │ │ │ + Z_ORDER_DRAWING_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER(indexer, newNode, nextNode); │ │ │ │ │ + if (nextNode && returnVal == 0) { │ │ │ │ │ + returnVal = 1 │ │ │ │ │ } │ │ │ │ │ + return returnVal │ │ │ │ │ }, │ │ │ │ │ - afterAdd: function() { │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoActivate) { │ │ │ │ │ - strategy.activate() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + Z_ORDER_Y_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER(indexer, newNode, nextNode); │ │ │ │ │ + if (nextNode && returnVal === 0) { │ │ │ │ │ + var result = nextNode._boundsBottom - newNode._boundsBottom; │ │ │ │ │ + returnVal = result === 0 ? 1 : result │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - this.drawn = false; │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoActivate) { │ │ │ │ │ - strategy.deactivate() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + return returnVal │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.Elements = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ + rendererRoot: null, │ │ │ │ │ + root: null, │ │ │ │ │ + vectorRoot: null, │ │ │ │ │ + textRoot: null, │ │ │ │ │ + xmlns: null, │ │ │ │ │ + xOffset: 0, │ │ │ │ │ + indexer: null, │ │ │ │ │ + BACKGROUND_ID_SUFFIX: "_background", │ │ │ │ │ + LABEL_ID_SUFFIX: "_label", │ │ │ │ │ + LABEL_OUTLINE_SUFFIX: "_outline", │ │ │ │ │ + initialize: function(containerID, options) { │ │ │ │ │ + OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.rendererRoot = this.createRenderRoot(); │ │ │ │ │ + this.root = this.createRoot("_root"); │ │ │ │ │ + this.vectorRoot = this.createRoot("_vroot"); │ │ │ │ │ + this.textRoot = this.createRoot("_troot"); │ │ │ │ │ + this.root.appendChild(this.vectorRoot); │ │ │ │ │ + this.root.appendChild(this.textRoot); │ │ │ │ │ + this.rendererRoot.appendChild(this.root); │ │ │ │ │ + this.container.appendChild(this.rendererRoot); │ │ │ │ │ + if (options && (options.zIndexing || options.yOrdering)) { │ │ │ │ │ + this.indexer = new OpenLayers.ElementsIndexer(options.yOrdering) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - OpenLayers.Layer.prototype.onMapResize.apply(this, arguments); │ │ │ │ │ - var newSize = this.map.getSize(); │ │ │ │ │ - newSize.w = newSize.w * this.ratio; │ │ │ │ │ - newSize.h = newSize.h * this.ratio; │ │ │ │ │ - this.renderer.setSize(newSize) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.clear(); │ │ │ │ │ + this.rendererRoot = null; │ │ │ │ │ + this.root = null; │ │ │ │ │ + this.xmlns = null; │ │ │ │ │ + OpenLayers.Renderer.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - var coordSysUnchanged = true; │ │ │ │ │ - if (!dragging) { │ │ │ │ │ - this.renderer.root.style.visibility = "hidden"; │ │ │ │ │ - var viewSize = this.map.getSize(), │ │ │ │ │ - viewWidth = viewSize.w, │ │ │ │ │ - viewHeight = viewSize.h, │ │ │ │ │ - offsetLeft = viewWidth / 2 * this.ratio - viewWidth / 2, │ │ │ │ │ - offsetTop = viewHeight / 2 * this.ratio - viewHeight / 2; │ │ │ │ │ - offsetLeft += this.map.layerContainerOriginPx.x; │ │ │ │ │ - offsetLeft = -Math.round(offsetLeft); │ │ │ │ │ - offsetTop += this.map.layerContainerOriginPx.y; │ │ │ │ │ - offsetTop = -Math.round(offsetTop); │ │ │ │ │ - this.div.style.left = offsetLeft + "px"; │ │ │ │ │ - this.div.style.top = offsetTop + "px"; │ │ │ │ │ - var extent = this.map.getExtent().scale(this.ratio); │ │ │ │ │ - coordSysUnchanged = this.renderer.setExtent(extent, zoomChanged); │ │ │ │ │ - this.renderer.root.style.visibility = "visible"; │ │ │ │ │ - if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ - this.div.scrollLeft = this.div.scrollLeft │ │ │ │ │ - } │ │ │ │ │ - if (!zoomChanged && coordSysUnchanged) { │ │ │ │ │ - for (var i in this.unrenderedFeatures) { │ │ │ │ │ - var feature = this.unrenderedFeatures[i]; │ │ │ │ │ - this.drawFeature(feature) │ │ │ │ │ - } │ │ │ │ │ + clear: function() { │ │ │ │ │ + var child; │ │ │ │ │ + var root = this.vectorRoot; │ │ │ │ │ + if (root) { │ │ │ │ │ + while (child = root.firstChild) { │ │ │ │ │ + root.removeChild(child) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (!this.drawn || zoomChanged || !coordSysUnchanged) { │ │ │ │ │ - this.drawn = true; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0, len = this.features.length; i < len; i++) { │ │ │ │ │ - this.renderer.locked = i !== len - 1; │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - this.drawFeature(feature) │ │ │ │ │ + root = this.textRoot; │ │ │ │ │ + if (root) { │ │ │ │ │ + while (child = root.firstChild) { │ │ │ │ │ + root.removeChild(child) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - display: function(display) { │ │ │ │ │ - OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ - var currentDisplay = this.div.style.display; │ │ │ │ │ - if (currentDisplay != this.renderer.root.style.display) { │ │ │ │ │ - this.renderer.root.style.display = currentDisplay │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + this.indexer.clear() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - addFeatures: function(features, options) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ - features = [features] │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + var rightOfDateLine, ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ + extent = extent.scale(1 / ratio), │ │ │ │ │ + world = this.map.getMaxExtent(); │ │ │ │ │ + if (world.right > extent.left && world.right < extent.right) { │ │ │ │ │ + rightOfDateLine = true │ │ │ │ │ + } else if (world.left > extent.left && world.left < extent.right) { │ │ │ │ │ + rightOfDateLine = false │ │ │ │ │ + } │ │ │ │ │ + if (rightOfDateLine !== this.rightOfDateLine || resolutionChanged) { │ │ │ │ │ + coordSysUnchanged = false; │ │ │ │ │ + this.xOffset = rightOfDateLine === true ? world.getWidth() / resolution : 0 │ │ │ │ │ + } │ │ │ │ │ + this.rightOfDateLine = rightOfDateLine │ │ │ │ │ } │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ - if (notify) { │ │ │ │ │ - var event = { │ │ │ │ │ - features: features │ │ │ │ │ - }; │ │ │ │ │ - var ret = this.events.triggerEvent("beforefeaturesadded", event); │ │ │ │ │ - if (ret === false) { │ │ │ │ │ - return │ │ │ │ │ + return coordSysUnchanged │ │ │ │ │ + }, │ │ │ │ │ + getNodeType: function(geometry, style) {}, │ │ │ │ │ + drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ + var className = geometry.CLASS_NAME; │ │ │ │ │ + var rendered = true; │ │ │ │ │ + if (className == "OpenLayers.Geometry.Collection" || className == "OpenLayers.Geometry.MultiPoint" || className == "OpenLayers.Geometry.MultiLineString" || className == "OpenLayers.Geometry.MultiPolygon") { │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + rendered = this.drawGeometry(geometry.components[i], style, featureId) && rendered │ │ │ │ │ } │ │ │ │ │ - features = event.features │ │ │ │ │ + return rendered │ │ │ │ │ } │ │ │ │ │ - var featuresAdded = []; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - if (i != features.length - 1) { │ │ │ │ │ - this.renderer.locked = true │ │ │ │ │ + rendered = false; │ │ │ │ │ + var removeBackground = false; │ │ │ │ │ + if (style.display != "none") { │ │ │ │ │ + if (style.backgroundGraphic) { │ │ │ │ │ + this.redrawBackgroundNode(geometry.id, geometry, style, featureId) │ │ │ │ │ } else { │ │ │ │ │ - this.renderer.locked = false │ │ │ │ │ - } │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - if (this.geometryType && !(feature.geometry instanceof this.geometryType)) { │ │ │ │ │ - throw new TypeError("addFeatures: component should be an " + this.geometryType.prototype.CLASS_NAME) │ │ │ │ │ - } │ │ │ │ │ - feature.layer = this; │ │ │ │ │ - if (!feature.style && this.style) { │ │ │ │ │ - feature.style = OpenLayers.Util.extend({}, this.style) │ │ │ │ │ + removeBackground = true │ │ │ │ │ } │ │ │ │ │ - if (notify) { │ │ │ │ │ - if (this.events.triggerEvent("beforefeatureadded", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) === false) { │ │ │ │ │ - continue │ │ │ │ │ + rendered = this.redrawNode(geometry.id, geometry, style, featureId) │ │ │ │ │ + } │ │ │ │ │ + if (rendered == false) { │ │ │ │ │ + var node = document.getElementById(geometry.id); │ │ │ │ │ + if (node) { │ │ │ │ │ + if (node._style.backgroundGraphic) { │ │ │ │ │ + removeBackground = true │ │ │ │ │ } │ │ │ │ │ - this.preFeatureInsert(feature) │ │ │ │ │ - } │ │ │ │ │ - featuresAdded.push(feature); │ │ │ │ │ - this.features.push(feature); │ │ │ │ │ - this.drawFeature(feature); │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureadded", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - this.onFeatureInsert(feature) │ │ │ │ │ + node.parentNode.removeChild(node) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresadded", { │ │ │ │ │ - features: featuresAdded │ │ │ │ │ - }) │ │ │ │ │ + if (removeBackground) { │ │ │ │ │ + var node = document.getElementById(geometry.id + this.BACKGROUND_ID_SUFFIX); │ │ │ │ │ + if (node) { │ │ │ │ │ + node.parentNode.removeChild(node) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return rendered │ │ │ │ │ }, │ │ │ │ │ - removeFeatures: function(features, options) { │ │ │ │ │ - if (!features || features.length === 0) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (features === this.features) { │ │ │ │ │ - return this.removeAllFeatures(options) │ │ │ │ │ - } │ │ │ │ │ - if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ - features = [features] │ │ │ │ │ - } │ │ │ │ │ - if (features === this.selectedFeatures) { │ │ │ │ │ - features = features.slice() │ │ │ │ │ - } │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeaturesremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }) │ │ │ │ │ + redrawNode: function(id, geometry, style, featureId) { │ │ │ │ │ + style = this.applyDefaultSymbolizer(style); │ │ │ │ │ + var node = this.nodeFactory(id, this.getNodeType(geometry, style)); │ │ │ │ │ + node._featureId = featureId; │ │ │ │ │ + node._boundsBottom = geometry.getBounds().bottom; │ │ │ │ │ + node._geometryClass = geometry.CLASS_NAME; │ │ │ │ │ + node._style = style; │ │ │ │ │ + var drawResult = this.drawGeometryNode(node, geometry, style); │ │ │ │ │ + if (drawResult === false) { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - if (i != 0 && features[i - 1].geometry) { │ │ │ │ │ - this.renderer.locked = true │ │ │ │ │ + node = drawResult.node; │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + var insert = this.indexer.insert(node); │ │ │ │ │ + if (insert) { │ │ │ │ │ + this.vectorRoot.insertBefore(node, insert) │ │ │ │ │ } else { │ │ │ │ │ - this.renderer.locked = false │ │ │ │ │ - } │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.features = OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ - feature.layer = null; │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - this.renderer.eraseFeatures(feature) │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.selectedFeatures, feature) != -1) { │ │ │ │ │ - OpenLayers.Util.removeItem(this.selectedFeatures, feature) │ │ │ │ │ + this.vectorRoot.appendChild(node) │ │ │ │ │ } │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ + } else { │ │ │ │ │ + if (node.parentNode !== this.vectorRoot) { │ │ │ │ │ + this.vectorRoot.appendChild(node) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ + this.postDraw(node); │ │ │ │ │ + return drawResult.complete │ │ │ │ │ }, │ │ │ │ │ - removeAllFeatures: function(options) { │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ - var features = this.features; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeaturesremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }) │ │ │ │ │ + redrawBackgroundNode: function(id, geometry, style, featureId) { │ │ │ │ │ + var backgroundStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ + backgroundStyle.externalGraphic = backgroundStyle.backgroundGraphic; │ │ │ │ │ + backgroundStyle.graphicXOffset = backgroundStyle.backgroundXOffset; │ │ │ │ │ + backgroundStyle.graphicYOffset = backgroundStyle.backgroundYOffset; │ │ │ │ │ + backgroundStyle.graphicZIndex = backgroundStyle.backgroundGraphicZIndex; │ │ │ │ │ + backgroundStyle.graphicWidth = backgroundStyle.backgroundWidth || backgroundStyle.graphicWidth; │ │ │ │ │ + backgroundStyle.graphicHeight = backgroundStyle.backgroundHeight || backgroundStyle.graphicHeight; │ │ │ │ │ + backgroundStyle.backgroundGraphic = null; │ │ │ │ │ + backgroundStyle.backgroundXOffset = null; │ │ │ │ │ + backgroundStyle.backgroundYOffset = null; │ │ │ │ │ + backgroundStyle.backgroundGraphicZIndex = null; │ │ │ │ │ + return this.redrawNode(id + this.BACKGROUND_ID_SUFFIX, geometry, backgroundStyle, null) │ │ │ │ │ + }, │ │ │ │ │ + drawGeometryNode: function(node, geometry, style) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + var options = { │ │ │ │ │ + isFilled: style.fill === undefined ? true : style.fill, │ │ │ │ │ + isStroked: style.stroke === undefined ? !!style.strokeWidth : style.stroke │ │ │ │ │ + }; │ │ │ │ │ + var drawn; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.graphic === false) { │ │ │ │ │ + options.isFilled = false; │ │ │ │ │ + options.isStroked = false │ │ │ │ │ + } │ │ │ │ │ + drawn = this.drawPoint(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + options.isFilled = false; │ │ │ │ │ + drawn = this.drawLineString(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + drawn = this.drawLinearRing(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + drawn = this.drawPolygon(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + drawn = this.drawRectangle(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - feature.layer = null; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ + node._options = options; │ │ │ │ │ + if (drawn != false) { │ │ │ │ │ + return { │ │ │ │ │ + node: this.setStyle(node, style, options, geometry), │ │ │ │ │ + complete: drawn │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - this.renderer.clear(); │ │ │ │ │ - this.features = []; │ │ │ │ │ - this.unrenderedFeatures = {}; │ │ │ │ │ - this.selectedFeatures = []; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }) │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - destroyFeatures: function(features, options) { │ │ │ │ │ - var all = features == undefined; │ │ │ │ │ - if (all) { │ │ │ │ │ - features = this.features │ │ │ │ │ + postDraw: function(node) {}, │ │ │ │ │ + drawPoint: function(node, geometry) {}, │ │ │ │ │ + drawLineString: function(node, geometry) {}, │ │ │ │ │ + drawLinearRing: function(node, geometry) {}, │ │ │ │ │ + drawPolygon: function(node, geometry) {}, │ │ │ │ │ + drawRectangle: function(node, geometry) {}, │ │ │ │ │ + drawCircle: function(node, geometry) {}, │ │ │ │ │ + removeText: function(featureId) { │ │ │ │ │ + var label = document.getElementById(featureId + this.LABEL_ID_SUFFIX); │ │ │ │ │ + if (label) { │ │ │ │ │ + this.textRoot.removeChild(label) │ │ │ │ │ } │ │ │ │ │ - if (features) { │ │ │ │ │ - this.removeFeatures(features, options); │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - features[i].destroy() │ │ │ │ │ - } │ │ │ │ │ + var outline = document.getElementById(featureId + this.LABEL_OUTLINE_SUFFIX); │ │ │ │ │ + if (outline) { │ │ │ │ │ + this.textRoot.removeChild(outline) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - drawFeature: function(feature, style) { │ │ │ │ │ - if (!this.drawn) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (typeof style != "object") { │ │ │ │ │ - if (!style && feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ - style = "delete" │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var target = evt.target; │ │ │ │ │ + var useElement = target && target.correspondingUseElement; │ │ │ │ │ + var node = useElement ? useElement : target || evt.srcElement; │ │ │ │ │ + return node._featureId │ │ │ │ │ + }, │ │ │ │ │ + eraseGeometry: function(geometry, featureId) { │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPoint" || geometry.CLASS_NAME == "OpenLayers.Geometry.MultiLineString" || geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPolygon" || geometry.CLASS_NAME == "OpenLayers.Geometry.Collection") { │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + this.eraseGeometry(geometry.components[i], featureId) │ │ │ │ │ } │ │ │ │ │ - var renderIntent = style || feature.renderIntent; │ │ │ │ │ - style = feature.style || this.style; │ │ │ │ │ - if (!style) { │ │ │ │ │ - style = this.styleMap.createSymbolizer(feature, renderIntent) │ │ │ │ │ + } else { │ │ │ │ │ + var element = OpenLayers.Util.getElement(geometry.id); │ │ │ │ │ + if (element && element.parentNode) { │ │ │ │ │ + if (element.geometry) { │ │ │ │ │ + element.geometry.destroy(); │ │ │ │ │ + element.geometry = null │ │ │ │ │ + } │ │ │ │ │ + element.parentNode.removeChild(element); │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + this.indexer.remove(element) │ │ │ │ │ + } │ │ │ │ │ + if (element._style.backgroundGraphic) { │ │ │ │ │ + var backgroundId = geometry.id + this.BACKGROUND_ID_SUFFIX; │ │ │ │ │ + var bElem = OpenLayers.Util.getElement(backgroundId); │ │ │ │ │ + if (bElem && bElem.parentNode) { │ │ │ │ │ + bElem.parentNode.removeChild(bElem) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var drawn = this.renderer.drawFeature(feature, style); │ │ │ │ │ - if (drawn === false || drawn === null) { │ │ │ │ │ - this.unrenderedFeatures[feature.id] = feature │ │ │ │ │ + }, │ │ │ │ │ + nodeFactory: function(id, type) { │ │ │ │ │ + var node = OpenLayers.Util.getElement(id); │ │ │ │ │ + if (node) { │ │ │ │ │ + if (!this.nodeTypeCompare(node, type)) { │ │ │ │ │ + node.parentNode.removeChild(node); │ │ │ │ │ + node = this.nodeFactory(id, type) │ │ │ │ │ + } │ │ │ │ │ } else { │ │ │ │ │ - delete this.unrenderedFeatures[feature.id] │ │ │ │ │ + node = this.createNode(type, id) │ │ │ │ │ } │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - eraseFeatures: function(features) { │ │ │ │ │ - this.renderer.eraseFeatures(features) │ │ │ │ │ + nodeTypeCompare: function(node, type) {}, │ │ │ │ │ + createNode: function(type, id) {}, │ │ │ │ │ + moveRoot: function(renderer) { │ │ │ │ │ + var root = this.root; │ │ │ │ │ + if (renderer.root.parentNode == this.rendererRoot) { │ │ │ │ │ + root = renderer.root │ │ │ │ │ + } │ │ │ │ │ + root.parentNode.removeChild(root); │ │ │ │ │ + renderer.rendererRoot.appendChild(root) │ │ │ │ │ }, │ │ │ │ │ - getFeatureFromEvent: function(evt) { │ │ │ │ │ - if (!this.renderer) { │ │ │ │ │ - throw new Error("getFeatureFromEvent called on layer with no " + "renderer. This usually means you destroyed a " + "layer, but not some handler which is associated " + "with it.") │ │ │ │ │ + getRenderLayerId: function() { │ │ │ │ │ + return this.root.parentNode.parentNode.id │ │ │ │ │ + }, │ │ │ │ │ + isComplexSymbol: function(graphicName) { │ │ │ │ │ + return graphicName != "circle" && !!graphicName │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.Elements" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Renderer.VML = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ + xmlns: "urn:schemas-microsoft-com:vml", │ │ │ │ │ + symbolCache: {}, │ │ │ │ │ + offset: null, │ │ │ │ │ + initialize: function(containerID) { │ │ │ │ │ + if (!this.supported()) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - var feature = null; │ │ │ │ │ - var featureId = this.renderer.getFeatureIdFromEvent(evt); │ │ │ │ │ - if (featureId) { │ │ │ │ │ - if (typeof featureId === "string") { │ │ │ │ │ - feature = this.getFeatureById(featureId) │ │ │ │ │ - } else { │ │ │ │ │ - feature = featureId │ │ │ │ │ + if (!document.namespaces.olv) { │ │ │ │ │ + document.namespaces.add("olv", this.xmlns); │ │ │ │ │ + var style = document.createStyleSheet(); │ │ │ │ │ + var shapes = ["shape", "rect", "oval", "fill", "stroke", "imagedata", "group", "textbox"]; │ │ │ │ │ + for (var i = 0, len = shapes.length; i < len; i++) { │ │ │ │ │ + style.addRule("olv\\:" + shapes[i], "behavior: url(#default#VML); " + "position: absolute; display: inline-block;") │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return feature │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.initialize.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - getFeatureBy: function(property, value) { │ │ │ │ │ - var feature = null; │ │ │ │ │ - for (var i = 0, len = this.features.length; i < len; ++i) { │ │ │ │ │ - if (this.features[i][property] == value) { │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return feature │ │ │ │ │ + supported: function() { │ │ │ │ │ + return !!document.namespaces │ │ │ │ │ }, │ │ │ │ │ - getFeatureById: function(featureId) { │ │ │ │ │ - return this.getFeatureBy("id", featureId) │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var left = extent.left / resolution | 0; │ │ │ │ │ + var top = extent.top / resolution - this.size.h | 0; │ │ │ │ │ + if (resolutionChanged || !this.offset) { │ │ │ │ │ + this.offset = { │ │ │ │ │ + x: left, │ │ │ │ │ + y: top │ │ │ │ │ + }; │ │ │ │ │ + left = 0; │ │ │ │ │ + top = 0 │ │ │ │ │ + } else { │ │ │ │ │ + left = left - this.offset.x; │ │ │ │ │ + top = top - this.offset.y │ │ │ │ │ + } │ │ │ │ │ + var org = left - this.xOffset + " " + top; │ │ │ │ │ + this.root.coordorigin = org; │ │ │ │ │ + var roots = [this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ + var root; │ │ │ │ │ + for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ + root = roots[i]; │ │ │ │ │ + var size = this.size.w + " " + this.size.h; │ │ │ │ │ + root.coordsize = size │ │ │ │ │ + } │ │ │ │ │ + this.root.style.flip = "y"; │ │ │ │ │ + return coordSysUnchanged │ │ │ │ │ }, │ │ │ │ │ - getFeatureByFid: function(featureFid) { │ │ │ │ │ - return this.getFeatureBy("fid", featureFid) │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ + var roots = [this.rendererRoot, this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ + var w = this.size.w + "px"; │ │ │ │ │ + var h = this.size.h + "px"; │ │ │ │ │ + var root; │ │ │ │ │ + for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ + root = roots[i]; │ │ │ │ │ + root.style.width = w; │ │ │ │ │ + root.style.height = h │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - getFeaturesByAttribute: function(attrName, attrValue) { │ │ │ │ │ - var i, feature, len = this.features.length, │ │ │ │ │ - foundFeatures = []; │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - if (feature && feature.attributes) { │ │ │ │ │ - if (feature.attributes[attrName] === attrValue) { │ │ │ │ │ - foundFeatures.push(feature) │ │ │ │ │ + getNodeType: function(geometry, style) { │ │ │ │ │ + var nodeType = null; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + nodeType = "olv:rect" │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + nodeType = "olv:shape" │ │ │ │ │ + } else { │ │ │ │ │ + nodeType = "olv:oval" │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + nodeType = "olv:rect"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + case "OpenLayers.Geometry.Curve": │ │ │ │ │ + nodeType = "olv:shape"; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - return foundFeatures │ │ │ │ │ + return nodeType │ │ │ │ │ }, │ │ │ │ │ - onFeatureInsert: function(feature) {}, │ │ │ │ │ - preFeatureInsert: function(feature) {}, │ │ │ │ │ - getDataExtent: function() { │ │ │ │ │ - var maxExtent = null; │ │ │ │ │ - var features = this.features; │ │ │ │ │ - if (features && features.length > 0) { │ │ │ │ │ - var geometry = null; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - geometry = features[i].geometry; │ │ │ │ │ - if (geometry) { │ │ │ │ │ - if (maxExtent === null) { │ │ │ │ │ - maxExtent = new OpenLayers.Bounds │ │ │ │ │ - } │ │ │ │ │ - maxExtent.extend(geometry.getBounds()) │ │ │ │ │ - } │ │ │ │ │ + setStyle: function(node, style, options, geometry) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + options = options || node._options; │ │ │ │ │ + var fillColor = style.fillColor; │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + node.title = title │ │ │ │ │ + } │ │ │ │ │ + if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + options.isFilled = true; │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ + var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ + node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x + xOffset | 0) + "px"; │ │ │ │ │ + node.style.top = (geometry.y / resolution - this.offset.y - (yOffset + height) | 0) + "px"; │ │ │ │ │ + node.style.width = width + "px"; │ │ │ │ │ + node.style.height = height + "px"; │ │ │ │ │ + node.style.flip = "y"; │ │ │ │ │ + fillColor = "none"; │ │ │ │ │ + options.isStroked = false │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + var cache = this.importSymbol(style.graphicName); │ │ │ │ │ + node.path = cache.path; │ │ │ │ │ + node.coordorigin = cache.left + "," + cache.bottom; │ │ │ │ │ + var size = cache.size; │ │ │ │ │ + node.coordsize = size + "," + size; │ │ │ │ │ + this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ + node.style.flip = "y" │ │ │ │ │ + } else { │ │ │ │ │ + this.drawCircle(node, geometry, style.pointRadius) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return maxExtent │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Vector" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.WMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - service: "WMS", │ │ │ │ │ - version: "1.1.1", │ │ │ │ │ - request: "GetMap", │ │ │ │ │ - styles: "", │ │ │ │ │ - format: "image/jpeg" │ │ │ │ │ - }, │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - encodeBBOX: false, │ │ │ │ │ - noMagic: false, │ │ │ │ │ - yx: {}, │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ - if (parseFloat(params.VERSION) >= 1.3 && !params.EXCEPTIONS) { │ │ │ │ │ - params.EXCEPTIONS = "INIMAGE" │ │ │ │ │ + if (options.isFilled) { │ │ │ │ │ + node.fillcolor = fillColor │ │ │ │ │ + } else { │ │ │ │ │ + node.filled = "false" │ │ │ │ │ } │ │ │ │ │ - newArguments.push(name, url, params, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - OpenLayers.Util.applyDefaults(this.params, OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS)); │ │ │ │ │ - if (!this.noMagic && this.params.TRANSPARENT && this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ - if (options == null || !options.isBaseLayer) { │ │ │ │ │ - this.isBaseLayer = false │ │ │ │ │ + var fills = node.getElementsByTagName("fill"); │ │ │ │ │ + var fill = fills.length == 0 ? null : fills[0]; │ │ │ │ │ + if (!options.isFilled) { │ │ │ │ │ + if (fill) { │ │ │ │ │ + node.removeChild(fill) │ │ │ │ │ } │ │ │ │ │ - if (this.params.FORMAT == "image/jpeg") { │ │ │ │ │ - this.params.FORMAT = OpenLayers.Util.alphaHack() ? "image/gif" : "image/png" │ │ │ │ │ + } else { │ │ │ │ │ + if (!fill) { │ │ │ │ │ + fill = this.createNode("olv:fill", node.id + "_fill") │ │ │ │ │ + } │ │ │ │ │ + fill.opacity = style.fillOpacity; │ │ │ │ │ + if (node._geometryClass === "OpenLayers.Geometry.Point" && style.externalGraphic) { │ │ │ │ │ + if (style.graphicOpacity) { │ │ │ │ │ + fill.opacity = style.graphicOpacity │ │ │ │ │ + } │ │ │ │ │ + fill.src = style.externalGraphic; │ │ │ │ │ + fill.type = "frame"; │ │ │ │ │ + if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ + fill.aspect = "atmost" │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (fill.parentNode != node) { │ │ │ │ │ + node.appendChild(fill) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.WMS(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ + var rotation = style.rotation; │ │ │ │ │ + if (rotation !== undefined || node._rotation !== undefined) { │ │ │ │ │ + node._rotation = rotation; │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ + fill.opacity = 0 │ │ │ │ │ + } else if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ + node.style.rotation = rotation || 0 │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - reverseAxisOrder: function() { │ │ │ │ │ - var projCode = this.projection.getCode(); │ │ │ │ │ - return parseFloat(this.params.VERSION) >= 1.3 && !!(this.yx[projCode] || OpenLayers.Projection.defaults[projCode] && OpenLayers.Projection.defaults[projCode].yx) │ │ │ │ │ - }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var imageSize = this.getImageSize(); │ │ │ │ │ - var newParams = {}; │ │ │ │ │ - var reverseAxisOrder = this.reverseAxisOrder(); │ │ │ │ │ - newParams.BBOX = this.encodeBBOX ? bounds.toBBOX(null, reverseAxisOrder) : bounds.toArray(reverseAxisOrder); │ │ │ │ │ - newParams.WIDTH = imageSize.w; │ │ │ │ │ - newParams.HEIGHT = imageSize.h; │ │ │ │ │ - var requestString = this.getFullRequestString(newParams); │ │ │ │ │ - return requestString │ │ │ │ │ - }, │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ - var newArguments = [upperParams]; │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, newArguments) │ │ │ │ │ - }, │ │ │ │ │ - getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ - var mapProjection = this.map.getProjectionObject(); │ │ │ │ │ - var projectionCode = this.projection && this.projection.equals(mapProjection) ? this.projection.getCode() : mapProjection.getCode(); │ │ │ │ │ - var value = projectionCode == "none" ? null : projectionCode; │ │ │ │ │ - if (parseFloat(this.params.VERSION) >= 1.3) { │ │ │ │ │ - this.params.CRS = value │ │ │ │ │ + var strokes = node.getElementsByTagName("stroke"); │ │ │ │ │ + var stroke = strokes.length == 0 ? null : strokes[0]; │ │ │ │ │ + if (!options.isStroked) { │ │ │ │ │ + node.stroked = false; │ │ │ │ │ + if (stroke) { │ │ │ │ │ + stroke.on = false │ │ │ │ │ + } │ │ │ │ │ } else { │ │ │ │ │ - this.params.SRS = value │ │ │ │ │ + if (!stroke) { │ │ │ │ │ + stroke = this.createNode("olv:stroke", node.id + "_stroke"); │ │ │ │ │ + node.appendChild(stroke) │ │ │ │ │ + } │ │ │ │ │ + stroke.on = true; │ │ │ │ │ + stroke.color = style.strokeColor; │ │ │ │ │ + stroke.weight = style.strokeWidth + "px"; │ │ │ │ │ + stroke.opacity = style.strokeOpacity; │ │ │ │ │ + stroke.endcap = style.strokeLinecap == "butt" ? "flat" : style.strokeLinecap || "round"; │ │ │ │ │ + if (style.strokeDashstyle) { │ │ │ │ │ + stroke.dashstyle = this.dashStyle(style) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (typeof this.params.TRANSPARENT == "boolean") { │ │ │ │ │ - newParams.TRANSPARENT = this.params.TRANSPARENT ? "TRUE" : "FALSE" │ │ │ │ │ + if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ + node.style.cursor = style.cursor │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.getFullRequestString.apply(this, arguments) │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.WMS" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.SphericalMercator = { │ │ │ │ │ - getExtent: function() { │ │ │ │ │ - var extent = null; │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - extent = this.map.calculateBounds() │ │ │ │ │ + graphicRotate: function(node, xOffset, yOffset, style) { │ │ │ │ │ + var style = style || node._style; │ │ │ │ │ + var rotation = style.rotation || 0; │ │ │ │ │ + var aspectRatio, size; │ │ │ │ │ + if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ + var img = new Image; │ │ │ │ │ + img.onreadystatechange = OpenLayers.Function.bind(function() { │ │ │ │ │ + if (img.readyState == "complete" || img.readyState == "interactive") { │ │ │ │ │ + aspectRatio = img.width / img.height; │ │ │ │ │ + size = Math.max(style.pointRadius * 2, style.graphicWidth || 0, style.graphicHeight || 0); │ │ │ │ │ + xOffset = xOffset * aspectRatio; │ │ │ │ │ + style.graphicWidth = size * aspectRatio; │ │ │ │ │ + style.graphicHeight = size; │ │ │ │ │ + this.graphicRotate(node, xOffset, yOffset, style) │ │ │ │ │ + } │ │ │ │ │ + }, this); │ │ │ │ │ + img.src = style.externalGraphic; │ │ │ │ │ + return │ │ │ │ │ } else { │ │ │ │ │ - extent = OpenLayers.Layer.FixedZoomLevels.prototype.getExtent.apply(this) │ │ │ │ │ + size = Math.max(style.graphicWidth, style.graphicHeight); │ │ │ │ │ + aspectRatio = style.graphicWidth / style.graphicHeight │ │ │ │ │ } │ │ │ │ │ - return extent │ │ │ │ │ - }, │ │ │ │ │ - getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ - return OpenLayers.Layer.prototype.getLonLatFromViewPortPx.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - getViewPortPxFromLonLat: function(lonlat) { │ │ │ │ │ - return OpenLayers.Layer.prototype.getViewPortPxFromLonLat.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - initMercatorParameters: function() { │ │ │ │ │ - this.RESOLUTIONS = []; │ │ │ │ │ - var maxResolution = 156543.03390625; │ │ │ │ │ - for (var zoom = 0; zoom <= this.MAX_ZOOM_LEVEL; ++zoom) { │ │ │ │ │ - this.RESOLUTIONS[zoom] = maxResolution / Math.pow(2, zoom) │ │ │ │ │ + var width = Math.round(style.graphicWidth || size * aspectRatio); │ │ │ │ │ + var height = Math.round(style.graphicHeight || size); │ │ │ │ │ + node.style.width = width + "px"; │ │ │ │ │ + node.style.height = height + "px"; │ │ │ │ │ + var image = document.getElementById(node.id + "_image"); │ │ │ │ │ + if (!image) { │ │ │ │ │ + image = this.createNode("olv:imagedata", node.id + "_image"); │ │ │ │ │ + node.appendChild(image) │ │ │ │ │ } │ │ │ │ │ - this.units = "m"; │ │ │ │ │ - this.projection = this.projection || "EPSG:900913" │ │ │ │ │ + image.style.width = width + "px"; │ │ │ │ │ + image.style.height = height + "px"; │ │ │ │ │ + image.src = style.externalGraphic; │ │ │ │ │ + image.style.filter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(" + "src='', sizingMethod='scale')"; │ │ │ │ │ + var rot = rotation * Math.PI / 180; │ │ │ │ │ + var sintheta = Math.sin(rot); │ │ │ │ │ + var costheta = Math.cos(rot); │ │ │ │ │ + var filter = "progid:DXImageTransform.Microsoft.Matrix(M11=" + costheta + ",M12=" + -sintheta + ",M21=" + sintheta + ",M22=" + costheta + ",SizingMethod='auto expand')\n"; │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ + if (opacity && opacity != 1) { │ │ │ │ │ + filter += "progid:DXImageTransform.Microsoft.BasicImage(opacity=" + opacity + ")\n" │ │ │ │ │ + } │ │ │ │ │ + node.style.filter = filter; │ │ │ │ │ + var centerPoint = new OpenLayers.Geometry.Point(-xOffset, -yOffset); │ │ │ │ │ + var imgBox = new OpenLayers.Bounds(0, 0, width, height).toGeometry(); │ │ │ │ │ + imgBox.rotate(style.rotation, centerPoint); │ │ │ │ │ + var imgBounds = imgBox.getBounds(); │ │ │ │ │ + node.style.left = Math.round(parseInt(node.style.left) + imgBounds.left) + "px"; │ │ │ │ │ + node.style.top = Math.round(parseInt(node.style.top) - imgBounds.bottom) + "px" │ │ │ │ │ }, │ │ │ │ │ - forwardMercator: function() { │ │ │ │ │ - var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - var sm = new OpenLayers.Projection("EPSG:900913"); │ │ │ │ │ - return function(lon, lat) { │ │ │ │ │ - var point = OpenLayers.Projection.transform({ │ │ │ │ │ - x: lon, │ │ │ │ │ - y: lat │ │ │ │ │ - }, gg, sm); │ │ │ │ │ - return new OpenLayers.LonLat(point.x, point.y) │ │ │ │ │ + postDraw: function(node) { │ │ │ │ │ + node.style.visibility = "visible"; │ │ │ │ │ + var fillColor = node._style.fillColor; │ │ │ │ │ + var strokeColor = node._style.strokeColor; │ │ │ │ │ + if (fillColor == "none" && node.fillcolor != fillColor) { │ │ │ │ │ + node.fillcolor = fillColor │ │ │ │ │ } │ │ │ │ │ - }(), │ │ │ │ │ - inverseMercator: function() { │ │ │ │ │ - var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - var sm = new OpenLayers.Projection("EPSG:900913"); │ │ │ │ │ - return function(x, y) { │ │ │ │ │ - var point = OpenLayers.Projection.transform({ │ │ │ │ │ - x: x, │ │ │ │ │ - y: y │ │ │ │ │ - }, sm, gg); │ │ │ │ │ - return new OpenLayers.LonLat(point.x, point.y) │ │ │ │ │ + if (strokeColor == "none" && node.strokecolor != strokeColor) { │ │ │ │ │ + node.strokecolor = strokeColor │ │ │ │ │ } │ │ │ │ │ - }() │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Layer.EventPane = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ - smoothDragPan: true, │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - isFixed: true, │ │ │ │ │ - pane: null, │ │ │ │ │ - mapObject: null, │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (this.pane == null) { │ │ │ │ │ - this.pane = OpenLayers.Util.createDiv(this.div.id + "_EventPane") │ │ │ │ │ + }, │ │ │ │ │ + setNodeDimension: function(node, geometry) { │ │ │ │ │ + var bbox = geometry.getBounds(); │ │ │ │ │ + if (bbox) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var scaledBox = new OpenLayers.Bounds((bbox.left - this.featureDx) / resolution - this.offset.x | 0, bbox.bottom / resolution - this.offset.y | 0, (bbox.right - this.featureDx) / resolution - this.offset.x | 0, bbox.top / resolution - this.offset.y | 0); │ │ │ │ │ + node.style.left = scaledBox.left + "px"; │ │ │ │ │ + node.style.top = scaledBox.top + "px"; │ │ │ │ │ + node.style.width = scaledBox.getWidth() + "px"; │ │ │ │ │ + node.style.height = scaledBox.getHeight() + "px"; │ │ │ │ │ + node.coordorigin = scaledBox.left + " " + scaledBox.top; │ │ │ │ │ + node.coordsize = scaledBox.getWidth() + " " + scaledBox.getHeight() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.mapObject = null; │ │ │ │ │ - this.pane = null; │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ + dashStyle: function(style) { │ │ │ │ │ + var dash = style.strokeDashstyle; │ │ │ │ │ + switch (dash) { │ │ │ │ │ + case "solid": │ │ │ │ │ + case "dot": │ │ │ │ │ + case "dash": │ │ │ │ │ + case "dashdot": │ │ │ │ │ + case "longdash": │ │ │ │ │ + case "longdashdot": │ │ │ │ │ + return dash; │ │ │ │ │ + default: │ │ │ │ │ + var parts = dash.split(/[ ,]/); │ │ │ │ │ + if (parts.length == 2) { │ │ │ │ │ + if (1 * parts[0] >= 2 * parts[1]) { │ │ │ │ │ + return "longdash" │ │ │ │ │ + } │ │ │ │ │ + return parts[0] == 1 || parts[1] == 1 ? "dot" : "dash" │ │ │ │ │ + } else if (parts.length == 4) { │ │ │ │ │ + return 1 * parts[0] >= 2 * parts[1] ? "longdashdot" : "dashdot" │ │ │ │ │ + } │ │ │ │ │ + return "solid" │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.pane.style.zIndex = parseInt(this.div.style.zIndex) + 1; │ │ │ │ │ - this.pane.style.display = this.div.style.display; │ │ │ │ │ - this.pane.style.width = "100%"; │ │ │ │ │ - this.pane.style.height = "100%"; │ │ │ │ │ - if (OpenLayers.BROWSER_NAME == "msie") { │ │ │ │ │ - this.pane.style.background = "url(" + OpenLayers.Util.getImageLocation("blank.gif") + ")" │ │ │ │ │ + createNode: function(type, id) { │ │ │ │ │ + var node = document.createElement(type); │ │ │ │ │ + if (id) { │ │ │ │ │ + node.id = id │ │ │ │ │ } │ │ │ │ │ - if (this.isFixed) { │ │ │ │ │ - this.map.viewPortDiv.appendChild(this.pane) │ │ │ │ │ - } else { │ │ │ │ │ - this.map.layerContainerDiv.appendChild(this.pane) │ │ │ │ │ + node.unselectable = "on"; │ │ │ │ │ + node.onselectstart = OpenLayers.Function.False; │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + nodeTypeCompare: function(node, type) { │ │ │ │ │ + var subType = type; │ │ │ │ │ + var splitIndex = subType.indexOf(":"); │ │ │ │ │ + if (splitIndex != -1) { │ │ │ │ │ + subType = subType.substr(splitIndex + 1) │ │ │ │ │ } │ │ │ │ │ - this.loadMapObject(); │ │ │ │ │ - if (this.mapObject == null) { │ │ │ │ │ - this.loadWarningMessage() │ │ │ │ │ + var nodeName = node.nodeName; │ │ │ │ │ + splitIndex = nodeName.indexOf(":"); │ │ │ │ │ + if (splitIndex != -1) { │ │ │ │ │ + nodeName = nodeName.substr(splitIndex + 1) │ │ │ │ │ } │ │ │ │ │ + return subType == nodeName │ │ │ │ │ }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - if (this.pane && this.pane.parentNode) { │ │ │ │ │ - this.pane.parentNode.removeChild(this.pane) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.prototype.removeMap.apply(this, arguments) │ │ │ │ │ + createRenderRoot: function() { │ │ │ │ │ + return this.nodeFactory(this.container.id + "_vmlRoot", "div") │ │ │ │ │ }, │ │ │ │ │ - loadWarningMessage: function() { │ │ │ │ │ - this.div.style.backgroundColor = "darkblue"; │ │ │ │ │ - var viewSize = this.map.getSize(); │ │ │ │ │ - var msgW = Math.min(viewSize.w, 300); │ │ │ │ │ - var msgH = Math.min(viewSize.h, 200); │ │ │ │ │ - var size = new OpenLayers.Size(msgW, msgH); │ │ │ │ │ - var centerPx = new OpenLayers.Pixel(viewSize.w / 2, viewSize.h / 2); │ │ │ │ │ - var topLeft = centerPx.add(-size.w / 2, -size.h / 2); │ │ │ │ │ - var div = OpenLayers.Util.createDiv(this.name + "_warning", topLeft, size, null, null, null, "auto"); │ │ │ │ │ - div.style.padding = "7px"; │ │ │ │ │ - div.style.backgroundColor = "yellow"; │ │ │ │ │ - div.innerHTML = this.getWarningHTML(); │ │ │ │ │ - this.div.appendChild(div) │ │ │ │ │ + createRoot: function(suffix) { │ │ │ │ │ + return this.nodeFactory(this.container.id + suffix, "olv:group") │ │ │ │ │ }, │ │ │ │ │ - getWarningHTML: function() { │ │ │ │ │ - return "" │ │ │ │ │ + drawPoint: function(node, geometry) { │ │ │ │ │ + return this.drawCircle(node, geometry, 1) │ │ │ │ │ }, │ │ │ │ │ - display: function(display) { │ │ │ │ │ - OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ - this.pane.style.display = this.div.style.display │ │ │ │ │ + drawCircle: function(node, geometry, radius) { │ │ │ │ │ + if (!isNaN(geometry.x) && !isNaN(geometry.y)) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x | 0) - radius + "px"; │ │ │ │ │ + node.style.top = (geometry.y / resolution - this.offset.y | 0) - radius + "px"; │ │ │ │ │ + var diameter = radius * 2; │ │ │ │ │ + node.style.width = diameter + "px"; │ │ │ │ │ + node.style.height = diameter + "px"; │ │ │ │ │ + return node │ │ │ │ │ + } │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - setZIndex: function(zIndex) { │ │ │ │ │ - OpenLayers.Layer.prototype.setZIndex.apply(this, arguments); │ │ │ │ │ - this.pane.style.zIndex = parseInt(this.div.style.zIndex) + 1 │ │ │ │ │ + drawLineString: function(node, geometry) { │ │ │ │ │ + return this.drawLine(node, geometry, false) │ │ │ │ │ }, │ │ │ │ │ - moveByPx: function(dx, dy) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveByPx.apply(this, arguments); │ │ │ │ │ - if (this.dragPanMapObject) { │ │ │ │ │ - this.dragPanMapObject(dx, -dy) │ │ │ │ │ - } else { │ │ │ │ │ - this.moveTo(this.map.getCachedCenter()) │ │ │ │ │ + drawLinearRing: function(node, geometry) { │ │ │ │ │ + return this.drawLine(node, geometry, true) │ │ │ │ │ + }, │ │ │ │ │ + drawLine: function(node, geometry, closeLine) { │ │ │ │ │ + this.setNodeDimension(node, geometry); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var numComponents = geometry.components.length; │ │ │ │ │ + var parts = new Array(numComponents); │ │ │ │ │ + var comp, x, y; │ │ │ │ │ + for (var i = 0; i < numComponents; i++) { │ │ │ │ │ + comp = geometry.components[i]; │ │ │ │ │ + x = (comp.x - this.featureDx) / resolution - this.offset.x | 0; │ │ │ │ │ + y = comp.y / resolution - this.offset.y | 0; │ │ │ │ │ + parts[i] = " " + x + "," + y + " l " │ │ │ │ │ } │ │ │ │ │ + var end = closeLine ? " x e" : " e"; │ │ │ │ │ + node.path = "m" + parts.join("") + end; │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - if (this.mapObject != null) { │ │ │ │ │ - var newCenter = this.map.getCenter(); │ │ │ │ │ - var newZoom = this.map.getZoom(); │ │ │ │ │ - if (newCenter != null) { │ │ │ │ │ - var moOldCenter = this.getMapObjectCenter(); │ │ │ │ │ - var oldCenter = this.getOLLonLatFromMapObjectLonLat(moOldCenter); │ │ │ │ │ - var moOldZoom = this.getMapObjectZoom(); │ │ │ │ │ - var oldZoom = this.getOLZoomFromMapObjectZoom(moOldZoom); │ │ │ │ │ - if (!newCenter.equals(oldCenter) || newZoom != oldZoom) { │ │ │ │ │ - if (!zoomChanged && oldCenter && this.dragPanMapObject && this.smoothDragPan) { │ │ │ │ │ - var oldPx = this.map.getViewPortPxFromLonLat(oldCenter); │ │ │ │ │ - var newPx = this.map.getViewPortPxFromLonLat(newCenter); │ │ │ │ │ - this.dragPanMapObject(newPx.x - oldPx.x, oldPx.y - newPx.y) │ │ │ │ │ - } else { │ │ │ │ │ - var center = this.getMapObjectLonLatFromOLLonLat(newCenter); │ │ │ │ │ - var zoom = this.getMapObjectZoomFromOLZoom(newZoom); │ │ │ │ │ - this.setMapObjectCenter(center, zoom, dragging) │ │ │ │ │ + drawPolygon: function(node, geometry) { │ │ │ │ │ + this.setNodeDimension(node, geometry); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var path = []; │ │ │ │ │ + var j, jj, points, area, first, second, i, ii, comp, pathComp, x, y; │ │ │ │ │ + for (j = 0, jj = geometry.components.length; j < jj; j++) { │ │ │ │ │ + path.push("m"); │ │ │ │ │ + points = geometry.components[j].components; │ │ │ │ │ + area = j === 0; │ │ │ │ │ + first = null; │ │ │ │ │ + second = null; │ │ │ │ │ + for (i = 0, ii = points.length; i < ii; i++) { │ │ │ │ │ + comp = points[i]; │ │ │ │ │ + x = (comp.x - this.featureDx) / resolution - this.offset.x | 0; │ │ │ │ │ + y = comp.y / resolution - this.offset.y | 0; │ │ │ │ │ + pathComp = " " + x + "," + y; │ │ │ │ │ + path.push(pathComp); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + path.push(" l") │ │ │ │ │ + } │ │ │ │ │ + if (!area) { │ │ │ │ │ + if (!first) { │ │ │ │ │ + first = pathComp │ │ │ │ │ + } else if (first != pathComp) { │ │ │ │ │ + if (!second) { │ │ │ │ │ + second = pathComp │ │ │ │ │ + } else if (second != pathComp) { │ │ │ │ │ + area = true │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + path.push(area ? " x " : " ") │ │ │ │ │ } │ │ │ │ │ + path.push("e"); │ │ │ │ │ + node.path = path.join(""); │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ - var lonlat = null; │ │ │ │ │ - if (this.mapObject != null && this.getMapObjectCenter() != null) { │ │ │ │ │ - var moPixel = this.getMapObjectPixelFromOLPixel(viewPortPx); │ │ │ │ │ - var moLonLat = this.getMapObjectLonLatFromMapObjectPixel(moPixel); │ │ │ │ │ - lonlat = this.getOLLonLatFromMapObjectLonLat(moLonLat) │ │ │ │ │ - } │ │ │ │ │ - return lonlat │ │ │ │ │ + drawRectangle: function(node, geometry) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x | 0) + "px"; │ │ │ │ │ + node.style.top = (geometry.y / resolution - this.offset.y | 0) + "px"; │ │ │ │ │ + node.style.width = (geometry.width / resolution | 0) + "px"; │ │ │ │ │ + node.style.height = (geometry.height / resolution | 0) + "px"; │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - getViewPortPxFromLonLat: function(lonlat) { │ │ │ │ │ - var viewPortPx = null; │ │ │ │ │ - if (this.mapObject != null && this.getMapObjectCenter() != null) { │ │ │ │ │ - var moLonLat = this.getMapObjectLonLatFromOLLonLat(lonlat); │ │ │ │ │ - var moPixel = this.getMapObjectPixelFromMapObjectLonLat(moLonLat); │ │ │ │ │ - viewPortPx = this.getOLPixelFromMapObjectPixel(moPixel) │ │ │ │ │ + drawText: function(featureId, style, location) { │ │ │ │ │ + var label = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX, "olv:rect"); │ │ │ │ │ + var textbox = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX + "_textbox", "olv:textbox"); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + label.style.left = ((location.x - this.featureDx) / resolution - this.offset.x | 0) + "px"; │ │ │ │ │ + label.style.top = (location.y / resolution - this.offset.y | 0) + "px"; │ │ │ │ │ + label.style.flip = "y"; │ │ │ │ │ + textbox.innerText = style.label; │ │ │ │ │ + if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ + textbox.style.cursor = style.cursor │ │ │ │ │ } │ │ │ │ │ - return viewPortPx │ │ │ │ │ - }, │ │ │ │ │ - getOLLonLatFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - var olLonLat = null; │ │ │ │ │ - if (moLonLat != null) { │ │ │ │ │ - var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - olLonLat = new OpenLayers.LonLat(lon, lat) │ │ │ │ │ + if (style.fontColor) { │ │ │ │ │ + textbox.style.color = style.fontColor │ │ │ │ │ } │ │ │ │ │ - return olLonLat │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectLonLatFromOLLonLat: function(olLonLat) { │ │ │ │ │ - var moLatLng = null; │ │ │ │ │ - if (olLonLat != null) { │ │ │ │ │ - moLatLng = this.getMapObjectLonLatFromLonLat(olLonLat.lon, olLonLat.lat) │ │ │ │ │ + if (style.fontOpacity) { │ │ │ │ │ + textbox.style.filter = "alpha(opacity=" + style.fontOpacity * 100 + ")" │ │ │ │ │ } │ │ │ │ │ - return moLatLng │ │ │ │ │ - }, │ │ │ │ │ - getOLPixelFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - var olPixel = null; │ │ │ │ │ - if (moPixel != null) { │ │ │ │ │ - var x = this.getXFromMapObjectPixel(moPixel); │ │ │ │ │ - var y = this.getYFromMapObjectPixel(moPixel); │ │ │ │ │ - olPixel = new OpenLayers.Pixel(x, y) │ │ │ │ │ + if (style.fontFamily) { │ │ │ │ │ + textbox.style.fontFamily = style.fontFamily │ │ │ │ │ } │ │ │ │ │ - return olPixel │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectPixelFromOLPixel: function(olPixel) { │ │ │ │ │ - var moPixel = null; │ │ │ │ │ - if (olPixel != null) { │ │ │ │ │ - moPixel = this.getMapObjectPixelFromXY(olPixel.x, olPixel.y) │ │ │ │ │ + if (style.fontSize) { │ │ │ │ │ + textbox.style.fontSize = style.fontSize │ │ │ │ │ } │ │ │ │ │ - return moPixel │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.EventPane" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.FixedZoomLevels = OpenLayers.Class({ │ │ │ │ │ - initialize: function() {}, │ │ │ │ │ - initResolutions: function() { │ │ │ │ │ - var props = ["minZoomLevel", "maxZoomLevel", "numZoomLevels"]; │ │ │ │ │ - for (var i = 0, len = props.length; i < len; i++) { │ │ │ │ │ - var property = props[i]; │ │ │ │ │ - this[property] = this.options[property] != null ? this.options[property] : this.map[property] │ │ │ │ │ + if (style.fontWeight) { │ │ │ │ │ + textbox.style.fontWeight = style.fontWeight │ │ │ │ │ } │ │ │ │ │ - if (this.minZoomLevel == null || this.minZoomLevel < this.MIN_ZOOM_LEVEL) { │ │ │ │ │ - this.minZoomLevel = this.MIN_ZOOM_LEVEL │ │ │ │ │ + if (style.fontStyle) { │ │ │ │ │ + textbox.style.fontStyle = style.fontStyle │ │ │ │ │ } │ │ │ │ │ - var desiredZoomLevels; │ │ │ │ │ - var limitZoomLevels = this.MAX_ZOOM_LEVEL - this.minZoomLevel + 1; │ │ │ │ │ - if (this.options.numZoomLevels == null && this.options.maxZoomLevel != null || this.numZoomLevels == null && this.maxZoomLevel != null) { │ │ │ │ │ - desiredZoomLevels = this.maxZoomLevel - this.minZoomLevel + 1 │ │ │ │ │ - } else { │ │ │ │ │ - desiredZoomLevels = this.numZoomLevels │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + label._featureId = featureId; │ │ │ │ │ + textbox._featureId = featureId; │ │ │ │ │ + textbox._geometry = location; │ │ │ │ │ + textbox._geometryClass = location.CLASS_NAME │ │ │ │ │ } │ │ │ │ │ - if (desiredZoomLevels != null) { │ │ │ │ │ - this.numZoomLevels = Math.min(desiredZoomLevels, limitZoomLevels) │ │ │ │ │ - } else { │ │ │ │ │ - this.numZoomLevels = limitZoomLevels │ │ │ │ │ + textbox.style.whiteSpace = "nowrap"; │ │ │ │ │ + textbox.inset = "1px,0px,0px,0px"; │ │ │ │ │ + if (!label.parentNode) { │ │ │ │ │ + label.appendChild(textbox); │ │ │ │ │ + this.textRoot.appendChild(label) │ │ │ │ │ } │ │ │ │ │ - this.maxZoomLevel = this.minZoomLevel + this.numZoomLevels - 1; │ │ │ │ │ - if (this.RESOLUTIONS != null) { │ │ │ │ │ - var resolutionsIndex = 0; │ │ │ │ │ - this.resolutions = []; │ │ │ │ │ - for (var i = this.minZoomLevel; i <= this.maxZoomLevel; i++) { │ │ │ │ │ - this.resolutions[resolutionsIndex++] = this.RESOLUTIONS[i] │ │ │ │ │ - } │ │ │ │ │ - this.maxResolution = this.resolutions[0]; │ │ │ │ │ - this.minResolution = this.resolutions[this.resolutions.length - 1] │ │ │ │ │ + var align = style.labelAlign || "cm"; │ │ │ │ │ + if (align.length == 1) { │ │ │ │ │ + align += "m" │ │ │ │ │ } │ │ │ │ │ + var xshift = textbox.clientWidth * OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(0, 1)]; │ │ │ │ │ + var yshift = textbox.clientHeight * OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(1, 1)]; │ │ │ │ │ + label.style.left = parseInt(label.style.left) - xshift - 1 + "px"; │ │ │ │ │ + label.style.top = parseInt(label.style.top) + yshift + "px" │ │ │ │ │ }, │ │ │ │ │ - getResolution: function() { │ │ │ │ │ - if (this.resolutions != null) { │ │ │ │ │ - return OpenLayers.Layer.prototype.getResolution.apply(this, arguments) │ │ │ │ │ - } else { │ │ │ │ │ - var resolution = null; │ │ │ │ │ - var viewSize = this.map.getSize(); │ │ │ │ │ - var extent = this.getExtent(); │ │ │ │ │ - if (viewSize != null && extent != null) { │ │ │ │ │ - resolution = Math.max(extent.getWidth() / viewSize.w, extent.getHeight() / viewSize.h) │ │ │ │ │ - } │ │ │ │ │ - return resolution │ │ │ │ │ + moveRoot: function(renderer) { │ │ │ │ │ + var layer = this.map.getLayer(renderer.container.id); │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Vector.RootContainer) { │ │ │ │ │ + layer = this.map.getLayer(this.container.id) │ │ │ │ │ } │ │ │ │ │ + layer && layer.renderer.clear(); │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.moveRoot.apply(this, arguments); │ │ │ │ │ + layer && layer.redraw() │ │ │ │ │ }, │ │ │ │ │ - getExtent: function() { │ │ │ │ │ - var size = this.map.getSize(); │ │ │ │ │ - var tl = this.getLonLatFromViewPortPx({ │ │ │ │ │ - x: 0, │ │ │ │ │ - y: 0 │ │ │ │ │ - }); │ │ │ │ │ - var br = this.getLonLatFromViewPortPx({ │ │ │ │ │ - x: size.w, │ │ │ │ │ - y: size.h │ │ │ │ │ - }); │ │ │ │ │ - if (tl != null && br != null) { │ │ │ │ │ - return new OpenLayers.Bounds(tl.lon, br.lat, br.lon, tl.lat) │ │ │ │ │ - } else { │ │ │ │ │ - return null │ │ │ │ │ + importSymbol: function(graphicName) { │ │ │ │ │ + var id = this.container.id + "-" + graphicName; │ │ │ │ │ + var cache = this.symbolCache[id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + return cache │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - getZoomForResolution: function(resolution) { │ │ │ │ │ - if (this.resolutions != null) { │ │ │ │ │ - return OpenLayers.Layer.prototype.getZoomForResolution.apply(this, arguments) │ │ │ │ │ - } else { │ │ │ │ │ - var extent = OpenLayers.Layer.prototype.getExtent.apply(this, []); │ │ │ │ │ - return this.getZoomForExtent(extent) │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(graphicName + " is not a valid symbol name") │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - getOLZoomFromMapObjectZoom: function(moZoom) { │ │ │ │ │ - var zoom = null; │ │ │ │ │ - if (moZoom != null) { │ │ │ │ │ - zoom = moZoom - this.minZoomLevel; │ │ │ │ │ - if (this.map.baseLayer !== this) { │ │ │ │ │ - zoom = this.map.baseLayer.getZoomForResolution(this.getResolutionForZoom(zoom)) │ │ │ │ │ + var symbolExtent = new OpenLayers.Bounds(Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ + var pathitems = ["m"]; │ │ │ │ │ + for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + var x = symbol[i]; │ │ │ │ │ + var y = symbol[i + 1]; │ │ │ │ │ + symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ + symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ + symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ + symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ + pathitems.push(x); │ │ │ │ │ + pathitems.push(y); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + pathitems.push("l") │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return zoom │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectZoomFromOLZoom: function(olZoom) { │ │ │ │ │ - var zoom = null; │ │ │ │ │ - if (olZoom != null) { │ │ │ │ │ - zoom = olZoom + this.minZoomLevel; │ │ │ │ │ - if (this.map.baseLayer !== this) { │ │ │ │ │ - zoom = this.getZoomForResolution(this.map.baseLayer.getResolutionForZoom(zoom)) │ │ │ │ │ - } │ │ │ │ │ + pathitems.push("x e"); │ │ │ │ │ + var path = pathitems.join(" "); │ │ │ │ │ + var diff = (symbolExtent.getWidth() - symbolExtent.getHeight()) / 2; │ │ │ │ │ + if (diff > 0) { │ │ │ │ │ + symbolExtent.bottom = symbolExtent.bottom - diff; │ │ │ │ │ + symbolExtent.top = symbolExtent.top + diff │ │ │ │ │ + } else { │ │ │ │ │ + symbolExtent.left = symbolExtent.left + diff; │ │ │ │ │ + symbolExtent.right = symbolExtent.right - diff │ │ │ │ │ } │ │ │ │ │ - return zoom │ │ │ │ │ + cache = { │ │ │ │ │ + path: path, │ │ │ │ │ + size: symbolExtent.getWidth(), │ │ │ │ │ + left: symbolExtent.left, │ │ │ │ │ + bottom: symbolExtent.bottom │ │ │ │ │ + }; │ │ │ │ │ + this.symbolCache[id] = cache; │ │ │ │ │ + return cache │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.FixedZoomLevels" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.VML" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.Google = OpenLayers.Class(OpenLayers.Layer.EventPane, OpenLayers.Layer.FixedZoomLevels, { │ │ │ │ │ - MIN_ZOOM_LEVEL: 0, │ │ │ │ │ - MAX_ZOOM_LEVEL: 21, │ │ │ │ │ - RESOLUTIONS: [1.40625, .703125, .3515625, .17578125, .087890625, .0439453125, .02197265625, .010986328125, .0054931640625, .00274658203125, .001373291015625, .0006866455078125, .00034332275390625, .000171661376953125, 858306884765625e-19, 4291534423828125e-20, 2145767211914062e-20, 1072883605957031e-20, 536441802978515e-20, 268220901489257e-20, 1341104507446289e-21, 6.705522537231445e-7], │ │ │ │ │ - type: null, │ │ │ │ │ - wrapDateLine: true, │ │ │ │ │ - sphericalMercator: false, │ │ │ │ │ - version: null, │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - if (!options.version) { │ │ │ │ │ - options.version = typeof GMap2 === "function" ? "2" : "3" │ │ │ │ │ - } │ │ │ │ │ - var mixin = OpenLayers.Layer.Google["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (mixin) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, mixin) │ │ │ │ │ - } else { │ │ │ │ │ - throw "Unsupported Google Maps API version: " + options.version │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, mixin.DEFAULTS); │ │ │ │ │ - if (options.maxExtent) { │ │ │ │ │ - options.maxExtent = options.maxExtent.clone() │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ - OpenLayers.Layer.FixedZoomLevels.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - OpenLayers.Util.extend(this, OpenLayers.Layer.SphericalMercator); │ │ │ │ │ - this.initMercatorParameters() │ │ │ │ │ +OpenLayers.Renderer.VML.LABEL_SHIFT = { │ │ │ │ │ + l: 0, │ │ │ │ │ + c: .5, │ │ │ │ │ + r: 1, │ │ │ │ │ + t: 0, │ │ │ │ │ + m: .5, │ │ │ │ │ + b: 1 │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.SVG = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ + xmlns: "http://www.w3.org/2000/svg", │ │ │ │ │ + xlinkns: "http://www.w3.org/1999/xlink", │ │ │ │ │ + MAX_PIXEL: 15e3, │ │ │ │ │ + translationParameters: null, │ │ │ │ │ + symbolMetrics: null, │ │ │ │ │ + initialize: function(containerID) { │ │ │ │ │ + if (!this.supported()) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.translationParameters = { │ │ │ │ │ + x: 0, │ │ │ │ │ + y: 0 │ │ │ │ │ + }; │ │ │ │ │ + this.symbolMetrics = {} │ │ │ │ │ }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - return new OpenLayers.Layer.Google(this.name, this.getOptions()) │ │ │ │ │ - }, │ │ │ │ │ - setVisibility: function(visible) { │ │ │ │ │ - var opacity = this.opacity == null ? 1 : this.opacity; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.setVisibility.apply(this, arguments); │ │ │ │ │ - this.setOpacity(opacity) │ │ │ │ │ - }, │ │ │ │ │ - display: function(visible) { │ │ │ │ │ - if (!this._dragging) { │ │ │ │ │ - this.setGMapVisibility(visible) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.display.apply(this, arguments) │ │ │ │ │ + supported: function() { │ │ │ │ │ + var svgFeature = "http://www.w3.org/TR/SVG11/feature#"; │ │ │ │ │ + return document.implementation && (document.implementation.hasFeature("org.w3c.svg", "1.0") || document.implementation.hasFeature(svgFeature + "SVG", "1.1") || document.implementation.hasFeature(svgFeature + "BasicStructure", "1.1")) │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - this._dragging = dragging; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - delete this._dragging │ │ │ │ │ + inValidRange: function(x, y, xyOnly) { │ │ │ │ │ + var left = x + (xyOnly ? 0 : this.translationParameters.x); │ │ │ │ │ + var top = y + (xyOnly ? 0 : this.translationParameters.y); │ │ │ │ │ + return left >= -this.MAX_PIXEL && left <= this.MAX_PIXEL && top >= -this.MAX_PIXEL && top <= this.MAX_PIXEL │ │ │ │ │ }, │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - if (opacity !== this.opacity) { │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "opacity" │ │ │ │ │ - }) │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + var resolution = this.getResolution(), │ │ │ │ │ + left = -extent.left / resolution, │ │ │ │ │ + top = extent.top / resolution; │ │ │ │ │ + if (resolutionChanged) { │ │ │ │ │ + this.left = left; │ │ │ │ │ + this.top = top; │ │ │ │ │ + var extentString = "0 0 " + this.size.w + " " + this.size.h; │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "viewBox", extentString); │ │ │ │ │ + this.translate(this.xOffset, 0); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + var inRange = this.translate(left - this.left + this.xOffset, top - this.top); │ │ │ │ │ + if (!inRange) { │ │ │ │ │ + this.setExtent(extent, true) │ │ │ │ │ } │ │ │ │ │ - this.opacity = opacity │ │ │ │ │ - } │ │ │ │ │ - if (this.getVisibility()) { │ │ │ │ │ - var container = this.getMapContainer(); │ │ │ │ │ - OpenLayers.Util.modifyDOMElement(container, null, null, null, null, null, null, opacity) │ │ │ │ │ + return coordSysUnchanged && inRange │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.setGMapVisibility(false); │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache && cache.count <= 1) { │ │ │ │ │ - this.removeGMapElements() │ │ │ │ │ + translate: function(x, y) { │ │ │ │ │ + if (!this.inValidRange(x, y, true)) { │ │ │ │ │ + return false │ │ │ │ │ + } else { │ │ │ │ │ + var transformString = ""; │ │ │ │ │ + if (x || y) { │ │ │ │ │ + transformString = "translate(" + x + "," + y + ")" │ │ │ │ │ } │ │ │ │ │ + this.root.setAttributeNS(null, "transform", transformString); │ │ │ │ │ + this.translationParameters = { │ │ │ │ │ + x: x, │ │ │ │ │ + y: y │ │ │ │ │ + }; │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - removeGMapElements: function() { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - var container = this.mapObject && this.getMapContainer(); │ │ │ │ │ - if (container && container.parentNode) { │ │ │ │ │ - container.parentNode.removeChild(container) │ │ │ │ │ - } │ │ │ │ │ - var termsOfUse = cache.termsOfUse; │ │ │ │ │ - if (termsOfUse && termsOfUse.parentNode) { │ │ │ │ │ - termsOfUse.parentNode.removeChild(termsOfUse) │ │ │ │ │ - } │ │ │ │ │ - var poweredBy = cache.poweredBy; │ │ │ │ │ - if (poweredBy && poweredBy.parentNode) { │ │ │ │ │ - poweredBy.parentNode.removeChild(poweredBy) │ │ │ │ │ - } │ │ │ │ │ - if (this.mapObject && window.google && google.maps && google.maps.event && google.maps.event.clearListeners) { │ │ │ │ │ - google.maps.event.clearListeners(this.mapObject, "tilesloaded") │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "width", this.size.w); │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "height", this.size.h) │ │ │ │ │ }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - if (this.visibility && this.mapObject) { │ │ │ │ │ - this.setGMapVisibility(false) │ │ │ │ │ + getNodeType: function(geometry, style) { │ │ │ │ │ + var nodeType = null; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + nodeType = "image" │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + nodeType = "svg" │ │ │ │ │ + } else { │ │ │ │ │ + nodeType = "circle" │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + nodeType = "rect"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + nodeType = "polyline"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + nodeType = "polygon"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + case "OpenLayers.Geometry.Curve": │ │ │ │ │ + nodeType = "path"; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - if (cache.count <= 1) { │ │ │ │ │ - this.removeGMapElements(); │ │ │ │ │ - delete OpenLayers.Layer.Google.cache[map.id] │ │ │ │ │ + return nodeType │ │ │ │ │ + }, │ │ │ │ │ + setStyle: function(node, style, options) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + options = options || node._options; │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + node.setAttributeNS(null, "title", title); │ │ │ │ │ + var titleNode = node.getElementsByTagName("title"); │ │ │ │ │ + if (titleNode.length > 0) { │ │ │ │ │ + titleNode[0].firstChild.textContent = title │ │ │ │ │ } else { │ │ │ │ │ - --cache.count │ │ │ │ │ + var label = this.nodeFactory(null, "title"); │ │ │ │ │ + label.textContent = title; │ │ │ │ │ + node.appendChild(label) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - delete this.termsOfUse; │ │ │ │ │ - delete this.poweredBy; │ │ │ │ │ - delete this.mapObject; │ │ │ │ │ - delete this.dragObject; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.removeMap.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - getOLBoundsFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - var olBounds = null; │ │ │ │ │ - if (moBounds != null) { │ │ │ │ │ - var sw = moBounds.getSouthWest(); │ │ │ │ │ - var ne = moBounds.getNorthEast(); │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - sw = this.forwardMercator(sw.lng(), sw.lat()); │ │ │ │ │ - ne = this.forwardMercator(ne.lng(), ne.lat()) │ │ │ │ │ + var r = parseFloat(node.getAttributeNS(null, "r")); │ │ │ │ │ + var widthFactor = 1; │ │ │ │ │ + var pos; │ │ │ │ │ + if (node._geometryClass == "OpenLayers.Geometry.Point" && r) { │ │ │ │ │ + node.style.visibility = ""; │ │ │ │ │ + if (style.graphic === false) { │ │ │ │ │ + node.style.visibility = "hidden" │ │ │ │ │ + } else if (style.externalGraphic) { │ │ │ │ │ + pos = this.getPosition(node); │ │ │ │ │ + if (style.graphicWidth && style.graphicHeight) { │ │ │ │ │ + node.setAttributeNS(null, "preserveAspectRatio", "none") │ │ │ │ │ + } │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ + var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ + var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ + node.setAttributeNS(null, "x", (pos.x + xOffset).toFixed()); │ │ │ │ │ + node.setAttributeNS(null, "y", (pos.y + yOffset).toFixed()); │ │ │ │ │ + node.setAttributeNS(null, "width", width); │ │ │ │ │ + node.setAttributeNS(null, "height", height); │ │ │ │ │ + node.setAttributeNS(this.xlinkns, "xlink:href", style.externalGraphic); │ │ │ │ │ + node.setAttributeNS(null, "style", "opacity: " + opacity); │ │ │ │ │ + node.onclick = OpenLayers.Event.preventDefault │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + var offset = style.pointRadius * 3; │ │ │ │ │ + var size = offset * 2; │ │ │ │ │ + var src = this.importSymbol(style.graphicName); │ │ │ │ │ + pos = this.getPosition(node); │ │ │ │ │ + widthFactor = this.symbolMetrics[src.id][0] * 3 / size; │ │ │ │ │ + var parent = node.parentNode; │ │ │ │ │ + var nextSibling = node.nextSibling; │ │ │ │ │ + if (parent) { │ │ │ │ │ + parent.removeChild(node) │ │ │ │ │ + } │ │ │ │ │ + node.firstChild && node.removeChild(node.firstChild); │ │ │ │ │ + node.appendChild(src.firstChild.cloneNode(true)); │ │ │ │ │ + node.setAttributeNS(null, "viewBox", src.getAttributeNS(null, "viewBox")); │ │ │ │ │ + node.setAttributeNS(null, "width", size); │ │ │ │ │ + node.setAttributeNS(null, "height", size); │ │ │ │ │ + node.setAttributeNS(null, "x", pos.x - offset); │ │ │ │ │ + node.setAttributeNS(null, "y", pos.y - offset); │ │ │ │ │ + if (nextSibling) { │ │ │ │ │ + parent.insertBefore(node, nextSibling) │ │ │ │ │ + } else if (parent) { │ │ │ │ │ + parent.appendChild(node) │ │ │ │ │ + } │ │ │ │ │ } else { │ │ │ │ │ - sw = new OpenLayers.LonLat(sw.lng(), sw.lat()); │ │ │ │ │ - ne = new OpenLayers.LonLat(ne.lng(), ne.lat()) │ │ │ │ │ + node.setAttributeNS(null, "r", style.pointRadius) │ │ │ │ │ + } │ │ │ │ │ + var rotation = style.rotation; │ │ │ │ │ + if ((rotation !== undefined || node._rotation !== undefined) && pos) { │ │ │ │ │ + node._rotation = rotation; │ │ │ │ │ + rotation |= 0; │ │ │ │ │ + if (node.nodeName !== "svg") { │ │ │ │ │ + node.setAttributeNS(null, "transform", "rotate(" + rotation + " " + pos.x + " " + pos.y + ")") │ │ │ │ │ + } else { │ │ │ │ │ + var metrics = this.symbolMetrics[src.id]; │ │ │ │ │ + node.firstChild.setAttributeNS(null, "transform", "rotate(" + rotation + " " + metrics[1] + " " + metrics[2] + ")") │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - olBounds = new OpenLayers.Bounds(sw.lon, sw.lat, ne.lon, ne.lat) │ │ │ │ │ } │ │ │ │ │ - return olBounds │ │ │ │ │ + if (options.isFilled) { │ │ │ │ │ + node.setAttributeNS(null, "fill", style.fillColor); │ │ │ │ │ + node.setAttributeNS(null, "fill-opacity", style.fillOpacity) │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "fill", "none") │ │ │ │ │ + } │ │ │ │ │ + if (options.isStroked) { │ │ │ │ │ + node.setAttributeNS(null, "stroke", style.strokeColor); │ │ │ │ │ + node.setAttributeNS(null, "stroke-opacity", style.strokeOpacity); │ │ │ │ │ + node.setAttributeNS(null, "stroke-width", style.strokeWidth * widthFactor); │ │ │ │ │ + node.setAttributeNS(null, "stroke-linecap", style.strokeLinecap || "round"); │ │ │ │ │ + node.setAttributeNS(null, "stroke-linejoin", "round"); │ │ │ │ │ + style.strokeDashstyle && node.setAttributeNS(null, "stroke-dasharray", this.dashStyle(style, widthFactor)) │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "stroke", "none") │ │ │ │ │ + } │ │ │ │ │ + if (style.pointerEvents) { │ │ │ │ │ + node.setAttributeNS(null, "pointer-events", style.pointerEvents) │ │ │ │ │ + } │ │ │ │ │ + if (style.cursor != null) { │ │ │ │ │ + node.setAttributeNS(null, "cursor", style.cursor) │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - getWarningHTML: function() { │ │ │ │ │ - return OpenLayers.i18n("googleWarning") │ │ │ │ │ + dashStyle: function(style, widthFactor) { │ │ │ │ │ + var w = style.strokeWidth * widthFactor; │ │ │ │ │ + var str = style.strokeDashstyle; │ │ │ │ │ + switch (str) { │ │ │ │ │ + case "solid": │ │ │ │ │ + return "none"; │ │ │ │ │ + case "dot": │ │ │ │ │ + return [1, 4 * w].join(); │ │ │ │ │ + case "dash": │ │ │ │ │ + return [4 * w, 4 * w].join(); │ │ │ │ │ + case "dashdot": │ │ │ │ │ + return [4 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ + case "longdash": │ │ │ │ │ + return [8 * w, 4 * w].join(); │ │ │ │ │ + case "longdashdot": │ │ │ │ │ + return [8 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ + default: │ │ │ │ │ + return OpenLayers.String.trim(str).replace(/\s+/g, ",") │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - getMapObjectCenter: function() { │ │ │ │ │ - return this.mapObject.getCenter() │ │ │ │ │ + createNode: function(type, id) { │ │ │ │ │ + var node = document.createElementNS(this.xmlns, type); │ │ │ │ │ + if (id) { │ │ │ │ │ + node.setAttributeNS(null, "id", id) │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - getMapObjectZoom: function() { │ │ │ │ │ - return this.mapObject.getZoom() │ │ │ │ │ + nodeTypeCompare: function(node, type) { │ │ │ │ │ + return type == node.nodeName │ │ │ │ │ }, │ │ │ │ │ - getLongitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - return this.sphericalMercator ? this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lon : moLonLat.lng() │ │ │ │ │ + createRenderRoot: function() { │ │ │ │ │ + var svg = this.nodeFactory(this.container.id + "_svgRoot", "svg"); │ │ │ │ │ + svg.style.display = "block"; │ │ │ │ │ + return svg │ │ │ │ │ }, │ │ │ │ │ - getLatitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - var lat = this.sphericalMercator ? this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lat : moLonLat.lat(); │ │ │ │ │ - return lat │ │ │ │ │ + createRoot: function(suffix) { │ │ │ │ │ + return this.nodeFactory(this.container.id + suffix, "g") │ │ │ │ │ }, │ │ │ │ │ - getXFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return moPixel.x │ │ │ │ │ + createDefs: function() { │ │ │ │ │ + var defs = this.nodeFactory(this.container.id + "_defs", "defs"); │ │ │ │ │ + this.rendererRoot.appendChild(defs); │ │ │ │ │ + return defs │ │ │ │ │ }, │ │ │ │ │ - getYFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return moPixel.y │ │ │ │ │ + drawPoint: function(node, geometry) { │ │ │ │ │ + return this.drawCircle(node, geometry, 1) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Google" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Google.cache = {}; │ │ │ │ │ -OpenLayers.Layer.Google.v2 = { │ │ │ │ │ - termsOfUse: null, │ │ │ │ │ - poweredBy: null, │ │ │ │ │ - dragObject: null, │ │ │ │ │ - loadMapObject: function() { │ │ │ │ │ - if (!this.type) { │ │ │ │ │ - this.type = G_NORMAL_MAP │ │ │ │ │ - } │ │ │ │ │ - var mapObject, termsOfUse, poweredBy; │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - mapObject = cache.mapObject; │ │ │ │ │ - termsOfUse = cache.termsOfUse; │ │ │ │ │ - poweredBy = cache.poweredBy; │ │ │ │ │ - ++cache.count │ │ │ │ │ + drawCircle: function(node, geometry, radius) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = (geometry.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y = this.top - geometry.y / resolution; │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + node.setAttributeNS(null, "cx", x); │ │ │ │ │ + node.setAttributeNS(null, "cy", y); │ │ │ │ │ + node.setAttributeNS(null, "r", radius); │ │ │ │ │ + return node │ │ │ │ │ } else { │ │ │ │ │ - var container = this.map.viewPortDiv; │ │ │ │ │ - var div = document.createElement("div"); │ │ │ │ │ - div.id = this.map.id + "_GMap2Container"; │ │ │ │ │ - div.style.position = "absolute"; │ │ │ │ │ - div.style.width = "100%"; │ │ │ │ │ - div.style.height = "100%"; │ │ │ │ │ - container.appendChild(div); │ │ │ │ │ - try { │ │ │ │ │ - mapObject = new GMap2(div); │ │ │ │ │ - termsOfUse = div.lastChild; │ │ │ │ │ - container.appendChild(termsOfUse); │ │ │ │ │ - termsOfUse.style.zIndex = "1100"; │ │ │ │ │ - termsOfUse.style.right = ""; │ │ │ │ │ - termsOfUse.style.bottom = ""; │ │ │ │ │ - termsOfUse.className = "olLayerGoogleCopyright"; │ │ │ │ │ - poweredBy = div.lastChild; │ │ │ │ │ - container.appendChild(poweredBy); │ │ │ │ │ - poweredBy.style.zIndex = "1100"; │ │ │ │ │ - poweredBy.style.right = ""; │ │ │ │ │ - poweredBy.style.bottom = ""; │ │ │ │ │ - poweredBy.className = "olLayerGooglePoweredBy gmnoprint" │ │ │ │ │ - } catch (e) { │ │ │ │ │ - throw e │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.Google.cache[this.map.id] = { │ │ │ │ │ - mapObject: mapObject, │ │ │ │ │ - termsOfUse: termsOfUse, │ │ │ │ │ - poweredBy: poweredBy, │ │ │ │ │ - count: 1 │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.mapObject = mapObject; │ │ │ │ │ - this.termsOfUse = termsOfUse; │ │ │ │ │ - this.poweredBy = poweredBy; │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.mapObject.getMapTypes(), this.type) === -1) { │ │ │ │ │ - this.mapObject.addMapType(this.type) │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - if (typeof mapObject.getDragObject == "function") { │ │ │ │ │ - this.dragObject = mapObject.getDragObject() │ │ │ │ │ + }, │ │ │ │ │ + drawLineString: function(node, geometry) { │ │ │ │ │ + var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ + if (componentsResult.path) { │ │ │ │ │ + node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ + return componentsResult.complete ? node : null │ │ │ │ │ } else { │ │ │ │ │ - this.dragPanMapObject = null │ │ │ │ │ - } │ │ │ │ │ - if (this.isBaseLayer === false) { │ │ │ │ │ - this.setGMapVisibility(this.div.style.display !== "none") │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - if (this.visibility && this.mapObject.isLoaded()) { │ │ │ │ │ - this.mapObject.checkResize() │ │ │ │ │ + drawLinearRing: function(node, geometry) { │ │ │ │ │ + var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ + if (componentsResult.path) { │ │ │ │ │ + node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ + return componentsResult.complete ? node : null │ │ │ │ │ } else { │ │ │ │ │ - if (!this._resized) { │ │ │ │ │ - var layer = this; │ │ │ │ │ - var handle = GEvent.addListener(this.mapObject, "load", function() { │ │ │ │ │ - GEvent.removeListener(handle); │ │ │ │ │ - delete layer._resized; │ │ │ │ │ - layer.mapObject.checkResize(); │ │ │ │ │ - layer.moveTo(layer.map.getCenter(), layer.map.getZoom()) │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this._resized = true │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - setGMapVisibility: function(visible) { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - var container = this.mapObject.getContainer(); │ │ │ │ │ - if (visible === true) { │ │ │ │ │ - this.mapObject.setMapType(this.type); │ │ │ │ │ - container.style.display = ""; │ │ │ │ │ - this.termsOfUse.style.left = ""; │ │ │ │ │ - this.termsOfUse.style.display = ""; │ │ │ │ │ - this.poweredBy.style.display = ""; │ │ │ │ │ - cache.displayed = this.id │ │ │ │ │ + drawPolygon: function(node, geometry) { │ │ │ │ │ + var d = ""; │ │ │ │ │ + var draw = true; │ │ │ │ │ + var complete = true; │ │ │ │ │ + var linearRingResult, path; │ │ │ │ │ + for (var j = 0, len = geometry.components.length; j < len; j++) { │ │ │ │ │ + d += " M"; │ │ │ │ │ + linearRingResult = this.getComponentsString(geometry.components[j].components, " "); │ │ │ │ │ + path = linearRingResult.path; │ │ │ │ │ + if (path) { │ │ │ │ │ + d += " " + path; │ │ │ │ │ + complete = linearRingResult.complete && complete │ │ │ │ │ } else { │ │ │ │ │ - if (cache.displayed === this.id) { │ │ │ │ │ - delete cache.displayed │ │ │ │ │ - } │ │ │ │ │ - if (!cache.displayed) { │ │ │ │ │ - container.style.display = "none"; │ │ │ │ │ - this.termsOfUse.style.display = "none"; │ │ │ │ │ - this.termsOfUse.style.left = "-9999px"; │ │ │ │ │ - this.poweredBy.style.display = "none" │ │ │ │ │ - } │ │ │ │ │ + draw = false │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - getMapContainer: function() { │ │ │ │ │ - return this.mapObject.getContainer() │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ - var moBounds = null; │ │ │ │ │ - if (olBounds != null) { │ │ │ │ │ - var sw = this.sphericalMercator ? this.inverseMercator(olBounds.bottom, olBounds.left) : new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ - var ne = this.sphericalMercator ? this.inverseMercator(olBounds.top, olBounds.right) : new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ - moBounds = new GLatLngBounds(new GLatLng(sw.lat, sw.lon), new GLatLng(ne.lat, ne.lon)) │ │ │ │ │ + d += " z"; │ │ │ │ │ + if (draw) { │ │ │ │ │ + node.setAttributeNS(null, "d", d); │ │ │ │ │ + node.setAttributeNS(null, "fill-rule", "evenodd"); │ │ │ │ │ + return complete ? node : null │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return moBounds │ │ │ │ │ - }, │ │ │ │ │ - setMapObjectCenter: function(center, zoom) { │ │ │ │ │ - this.mapObject.setCenter(center, zoom) │ │ │ │ │ - }, │ │ │ │ │ - dragPanMapObject: function(dX, dY) { │ │ │ │ │ - this.dragObject.moveBy(new GSize(-dX, dY)) │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return this.mapObject.fromContainerPixelToLatLng(moPixel) │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - return this.mapObject.fromLatLngToContainerPixel(moLonLat) │ │ │ │ │ }, │ │ │ │ │ - getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - return this.mapObject.getBoundsZoomLevel(moBounds) │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ - var gLatLng; │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ - gLatLng = new GLatLng(lonlat.lat, lonlat.lon) │ │ │ │ │ + drawRectangle: function(node, geometry) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = (geometry.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y = this.top - geometry.y / resolution; │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + node.setAttributeNS(null, "x", x); │ │ │ │ │ + node.setAttributeNS(null, "y", y); │ │ │ │ │ + node.setAttributeNS(null, "width", geometry.width / resolution); │ │ │ │ │ + node.setAttributeNS(null, "height", geometry.height / resolution); │ │ │ │ │ + return node │ │ │ │ │ } else { │ │ │ │ │ - gLatLng = new GLatLng(lat, lon) │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return gLatLng │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ - return new GPoint(x, y) │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Layer.Google.v3 = { │ │ │ │ │ - DEFAULTS: { │ │ │ │ │ - sphericalMercator: true, │ │ │ │ │ - projection: "EPSG:900913" │ │ │ │ │ }, │ │ │ │ │ - animationEnabled: true, │ │ │ │ │ - loadMapObject: function() { │ │ │ │ │ - if (!this.type) { │ │ │ │ │ - this.type = google.maps.MapTypeId.ROADMAP │ │ │ │ │ + drawText: function(featureId, style, location) { │ │ │ │ │ + var drawOutline = !!style.labelOutlineWidth; │ │ │ │ │ + if (drawOutline) { │ │ │ │ │ + var outlineStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ + outlineStyle.fontColor = outlineStyle.labelOutlineColor; │ │ │ │ │ + outlineStyle.fontStrokeColor = outlineStyle.labelOutlineColor; │ │ │ │ │ + outlineStyle.fontStrokeWidth = style.labelOutlineWidth; │ │ │ │ │ + if (style.labelOutlineOpacity) { │ │ │ │ │ + outlineStyle.fontOpacity = style.labelOutlineOpacity │ │ │ │ │ + } │ │ │ │ │ + delete outlineStyle.labelOutlineWidth; │ │ │ │ │ + this.drawText(featureId, outlineStyle, location) │ │ │ │ │ } │ │ │ │ │ - var mapObject; │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - mapObject = cache.mapObject; │ │ │ │ │ - ++cache.count │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = (location.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y = location.y / resolution - this.top; │ │ │ │ │ + var suffix = drawOutline ? this.LABEL_OUTLINE_SUFFIX : this.LABEL_ID_SUFFIX; │ │ │ │ │ + var label = this.nodeFactory(featureId + suffix, "text"); │ │ │ │ │ + label.setAttributeNS(null, "x", x); │ │ │ │ │ + label.setAttributeNS(null, "y", -y); │ │ │ │ │ + if (style.fontColor) { │ │ │ │ │ + label.setAttributeNS(null, "fill", style.fontColor) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStrokeColor) { │ │ │ │ │ + label.setAttributeNS(null, "stroke", style.fontStrokeColor) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStrokeWidth) { │ │ │ │ │ + label.setAttributeNS(null, "stroke-width", style.fontStrokeWidth) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontOpacity) { │ │ │ │ │ + label.setAttributeNS(null, "opacity", style.fontOpacity) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontFamily) { │ │ │ │ │ + label.setAttributeNS(null, "font-family", style.fontFamily) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontSize) { │ │ │ │ │ + label.setAttributeNS(null, "font-size", style.fontSize) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontWeight) { │ │ │ │ │ + label.setAttributeNS(null, "font-weight", style.fontWeight) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStyle) { │ │ │ │ │ + label.setAttributeNS(null, "font-style", style.fontStyle) │ │ │ │ │ + } │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + label.setAttributeNS(null, "pointer-events", "visible"); │ │ │ │ │ + label._featureId = featureId │ │ │ │ │ } else { │ │ │ │ │ - var center = this.map.getCenter(); │ │ │ │ │ - var container = document.createElement("div"); │ │ │ │ │ - container.className = "olForeignContainer"; │ │ │ │ │ - container.style.width = "100%"; │ │ │ │ │ - container.style.height = "100%"; │ │ │ │ │ - mapObject = new google.maps.Map(container, { │ │ │ │ │ - center: center ? new google.maps.LatLng(center.lat, center.lon) : new google.maps.LatLng(0, 0), │ │ │ │ │ - zoom: this.map.getZoom() || 0, │ │ │ │ │ - mapTypeId: this.type, │ │ │ │ │ - disableDefaultUI: true, │ │ │ │ │ - keyboardShortcuts: false, │ │ │ │ │ - draggable: false, │ │ │ │ │ - disableDoubleClickZoom: true, │ │ │ │ │ - scrollwheel: false, │ │ │ │ │ - streetViewControl: false │ │ │ │ │ - }); │ │ │ │ │ - var googleControl = document.createElement("div"); │ │ │ │ │ - googleControl.style.width = "100%"; │ │ │ │ │ - googleControl.style.height = "100%"; │ │ │ │ │ - mapObject.controls[google.maps.ControlPosition.TOP_LEFT].push(googleControl); │ │ │ │ │ - cache = { │ │ │ │ │ - googleControl: googleControl, │ │ │ │ │ - mapObject: mapObject, │ │ │ │ │ - count: 1 │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Layer.Google.cache[this.map.id] = cache │ │ │ │ │ + label.setAttributeNS(null, "pointer-events", "none") │ │ │ │ │ } │ │ │ │ │ - this.mapObject = mapObject; │ │ │ │ │ - this.setGMapVisibility(this.visibility) │ │ │ │ │ - }, │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - if (this.visibility) { │ │ │ │ │ - google.maps.event.trigger(this.mapObject, "resize") │ │ │ │ │ + var align = style.labelAlign || OpenLayers.Renderer.defaultSymbolizer.labelAlign; │ │ │ │ │ + label.setAttributeNS(null, "text-anchor", OpenLayers.Renderer.SVG.LABEL_ALIGN[align[0]] || "middle"); │ │ │ │ │ + if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ + label.setAttributeNS(null, "dominant-baseline", OpenLayers.Renderer.SVG.LABEL_ALIGN[align[1]] || "central") │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - setGMapVisibility: function(visible) { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - var map = this.map; │ │ │ │ │ - if (cache) { │ │ │ │ │ - var type = this.type; │ │ │ │ │ - var layers = map.layers; │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = layers[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Google && layer.visibility === true && layer.inRange === true) { │ │ │ │ │ - type = layer.type; │ │ │ │ │ - visible = true; │ │ │ │ │ - break │ │ │ │ │ + var labelRows = style.label.split("\n"); │ │ │ │ │ + var numRows = labelRows.length; │ │ │ │ │ + while (label.childNodes.length > numRows) { │ │ │ │ │ + label.removeChild(label.lastChild) │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + var tspan = this.nodeFactory(featureId + suffix + "_tspan_" + i, "tspan"); │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + tspan._featureId = featureId; │ │ │ │ │ + tspan._geometry = location; │ │ │ │ │ + tspan._geometryClass = location.CLASS_NAME │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.IS_GECKO === false) { │ │ │ │ │ + tspan.setAttributeNS(null, "baseline-shift", OpenLayers.Renderer.SVG.LABEL_VSHIFT[align[1]] || "-35%") │ │ │ │ │ + } │ │ │ │ │ + tspan.setAttribute("x", x); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + var vfactor = OpenLayers.Renderer.SVG.LABEL_VFACTOR[align[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5 │ │ │ │ │ } │ │ │ │ │ + tspan.setAttribute("dy", vfactor * (numRows - 1) + "em") │ │ │ │ │ + } else { │ │ │ │ │ + tspan.setAttribute("dy", "1em") │ │ │ │ │ } │ │ │ │ │ - var container = this.mapObject.getDiv(); │ │ │ │ │ - if (visible === true) { │ │ │ │ │ - if (container.parentNode !== map.div) { │ │ │ │ │ - if (!cache.rendered) { │ │ │ │ │ - var me = this; │ │ │ │ │ - google.maps.event.addListenerOnce(this.mapObject, "tilesloaded", function() { │ │ │ │ │ - cache.rendered = true; │ │ │ │ │ - me.setGMapVisibility(me.getVisibility()); │ │ │ │ │ - me.moveTo(me.map.getCenter()) │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - map.div.appendChild(container); │ │ │ │ │ - cache.googleControl.appendChild(map.viewPortDiv); │ │ │ │ │ - google.maps.event.trigger(this.mapObject, "resize") │ │ │ │ │ + tspan.textContent = labelRows[i] === "" ? " " : labelRows[i]; │ │ │ │ │ + if (!tspan.parentNode) { │ │ │ │ │ + label.appendChild(tspan) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!label.parentNode) { │ │ │ │ │ + this.textRoot.appendChild(label) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getComponentsString: function(components, separator) { │ │ │ │ │ + var renderCmp = []; │ │ │ │ │ + var complete = true; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + var strings = []; │ │ │ │ │ + var str, component; │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + component = components[i]; │ │ │ │ │ + renderCmp.push(component); │ │ │ │ │ + str = this.getShortString(component); │ │ │ │ │ + if (str) { │ │ │ │ │ + strings.push(str) │ │ │ │ │ + } else { │ │ │ │ │ + if (i > 0) { │ │ │ │ │ + if (this.getShortString(components[i - 1])) { │ │ │ │ │ + strings.push(this.clipLine(components[i], components[i - 1])) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.mapObject.setMapTypeId(type) │ │ │ │ │ - } else if (cache.googleControl.hasChildNodes()) { │ │ │ │ │ - map.div.appendChild(map.viewPortDiv); │ │ │ │ │ - map.div.removeChild(container) │ │ │ │ │ + if (i < len - 1) { │ │ │ │ │ + if (this.getShortString(components[i + 1])) { │ │ │ │ │ + strings.push(this.clipLine(components[i], components[i + 1])) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + complete = false │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - getMapContainer: function() { │ │ │ │ │ - return this.mapObject.getDiv() │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ - var moBounds = null; │ │ │ │ │ - if (olBounds != null) { │ │ │ │ │ - var sw = this.sphericalMercator ? this.inverseMercator(olBounds.bottom, olBounds.left) : new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ - var ne = this.sphericalMercator ? this.inverseMercator(olBounds.top, olBounds.right) : new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ - moBounds = new google.maps.LatLngBounds(new google.maps.LatLng(sw.lat, sw.lon), new google.maps.LatLng(ne.lat, ne.lon)) │ │ │ │ │ + return { │ │ │ │ │ + path: strings.join(separator || ","), │ │ │ │ │ + complete: complete │ │ │ │ │ } │ │ │ │ │ - return moBounds │ │ │ │ │ }, │ │ │ │ │ - getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - var size = this.map.getSize(); │ │ │ │ │ - var lon = this.getLongitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ - var lat = this.getLatitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ - var res = this.map.getResolution(); │ │ │ │ │ - var delta_x = moPixel.x - size.w / 2; │ │ │ │ │ - var delta_y = moPixel.y - size.h / 2; │ │ │ │ │ - var lonlat = new OpenLayers.LonLat(lon + delta_x * res, lat - delta_y * res); │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - lonlat = lonlat.wrapDateLine(this.maxExtent) │ │ │ │ │ + clipLine: function(badComponent, goodComponent) { │ │ │ │ │ + if (goodComponent.equals(badComponent)) { │ │ │ │ │ + return "" │ │ │ │ │ } │ │ │ │ │ - return this.getMapObjectLonLatFromLonLat(lonlat.lon, lonlat.lat) │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var maxX = this.MAX_PIXEL - this.translationParameters.x; │ │ │ │ │ + var maxY = this.MAX_PIXEL - this.translationParameters.y; │ │ │ │ │ + var x1 = (goodComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y1 = this.top - goodComponent.y / resolution; │ │ │ │ │ + var x2 = (badComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y2 = this.top - badComponent.y / resolution; │ │ │ │ │ + var k; │ │ │ │ │ + if (x2 < -maxX || x2 > maxX) { │ │ │ │ │ + k = (y2 - y1) / (x2 - x1); │ │ │ │ │ + x2 = x2 < 0 ? -maxX : maxX; │ │ │ │ │ + y2 = y1 + (x2 - x1) * k │ │ │ │ │ + } │ │ │ │ │ + if (y2 < -maxY || y2 > maxY) { │ │ │ │ │ + k = (x2 - x1) / (y2 - y1); │ │ │ │ │ + y2 = y2 < 0 ? -maxY : maxY; │ │ │ │ │ + x2 = x1 + (y2 - y1) * k │ │ │ │ │ + } │ │ │ │ │ + return x2 + "," + y2 │ │ │ │ │ }, │ │ │ │ │ - getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - var res = this.map.getResolution(); │ │ │ │ │ - var extent = this.map.getExtent(); │ │ │ │ │ - return this.getMapObjectPixelFromXY(1 / res * (lon - extent.left), 1 / res * (extent.top - lat)) │ │ │ │ │ + getShortString: function(point) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = (point.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y = this.top - point.y / resolution; │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + return x + "," + y │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - setMapObjectCenter: function(center, zoom) { │ │ │ │ │ - if (this.animationEnabled === false && zoom != this.mapObject.zoom) { │ │ │ │ │ - var mapContainer = this.getMapContainer(); │ │ │ │ │ - google.maps.event.addListenerOnce(this.mapObject, "idle", function() { │ │ │ │ │ - mapContainer.style.visibility = "" │ │ │ │ │ - }); │ │ │ │ │ - mapContainer.style.visibility = "hidden" │ │ │ │ │ + getPosition: function(node) { │ │ │ │ │ + return { │ │ │ │ │ + x: parseFloat(node.getAttributeNS(null, "cx")), │ │ │ │ │ + y: parseFloat(node.getAttributeNS(null, "cy")) │ │ │ │ │ } │ │ │ │ │ - this.mapObject.setOptions({ │ │ │ │ │ - center: center, │ │ │ │ │ - zoom: zoom │ │ │ │ │ - }) │ │ │ │ │ }, │ │ │ │ │ - getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - return this.mapObject.getBoundsZoomLevel(moBounds) │ │ │ │ │ + importSymbol: function(graphicName) { │ │ │ │ │ + if (!this.defs) { │ │ │ │ │ + this.defs = this.createDefs() │ │ │ │ │ + } │ │ │ │ │ + var id = this.container.id + "-" + graphicName; │ │ │ │ │ + var existing = document.getElementById(id); │ │ │ │ │ + if (existing != null) { │ │ │ │ │ + return existing │ │ │ │ │ + } │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(graphicName + " is not a valid symbol name") │ │ │ │ │ + } │ │ │ │ │ + var symbolNode = this.nodeFactory(id, "symbol"); │ │ │ │ │ + var node = this.nodeFactory(null, "polygon"); │ │ │ │ │ + symbolNode.appendChild(node); │ │ │ │ │ + var symbolExtent = new OpenLayers.Bounds(Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ + var points = []; │ │ │ │ │ + var x, y; │ │ │ │ │ + for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ + symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ + symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ + symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ + points.push(x, ",", y) │ │ │ │ │ + } │ │ │ │ │ + node.setAttributeNS(null, "points", points.join(" ")); │ │ │ │ │ + var width = symbolExtent.getWidth(); │ │ │ │ │ + var height = symbolExtent.getHeight(); │ │ │ │ │ + var viewBox = [symbolExtent.left - width, symbolExtent.bottom - height, width * 3, height * 3]; │ │ │ │ │ + symbolNode.setAttributeNS(null, "viewBox", viewBox.join(" ")); │ │ │ │ │ + this.symbolMetrics[id] = [Math.max(width, height), symbolExtent.getCenterLonLat().lon, symbolExtent.getCenterLonLat().lat]; │ │ │ │ │ + this.defs.appendChild(symbolNode); │ │ │ │ │ + return symbolNode │ │ │ │ │ }, │ │ │ │ │ - getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ - var gLatLng; │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ - gLatLng = new google.maps.LatLng(lonlat.lat, lonlat.lon) │ │ │ │ │ - } else { │ │ │ │ │ - gLatLng = new google.maps.LatLng(lat, lon) │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var featureId = OpenLayers.Renderer.Elements.prototype.getFeatureIdFromEvent.apply(this, arguments); │ │ │ │ │ + if (!featureId) { │ │ │ │ │ + var target = evt.target; │ │ │ │ │ + featureId = target.parentNode && target != this.rendererRoot ? target.parentNode._featureId : undefined │ │ │ │ │ } │ │ │ │ │ - return gLatLng │ │ │ │ │ + return featureId │ │ │ │ │ }, │ │ │ │ │ - getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ - return new google.maps.Point(x, y) │ │ │ │ │ - } │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.SVG" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_ALIGN = { │ │ │ │ │ + l: "start", │ │ │ │ │ + r: "end", │ │ │ │ │ + b: "bottom", │ │ │ │ │ + t: "hanging" │ │ │ │ │ }; │ │ │ │ │ -OpenLayers.Filter = OpenLayers.Class({ │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_VSHIFT = { │ │ │ │ │ + t: "-70%", │ │ │ │ │ + b: "0" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_VFACTOR = { │ │ │ │ │ + t: 0, │ │ │ │ │ + b: -1 │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.SVG.preventDefault = function(e) { │ │ │ │ │ + OpenLayers.Event.preventDefault(e) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol = OpenLayers.Class({ │ │ │ │ │ + format: null, │ │ │ │ │ + options: null, │ │ │ │ │ + autoDestroy: true, │ │ │ │ │ + defaultFilter: null, │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() {}, │ │ │ │ │ - evaluate: function(context) { │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - return null │ │ │ │ │ + options = options || {}; │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.options = options │ │ │ │ │ }, │ │ │ │ │ - toString: function() { │ │ │ │ │ - var string; │ │ │ │ │ - if (OpenLayers.Format && OpenLayers.Format.CQL) { │ │ │ │ │ - string = OpenLayers.Format.CQL.prototype.write(this) │ │ │ │ │ + mergeWithDefaultFilter: function(filter) { │ │ │ │ │ + var merged; │ │ │ │ │ + if (filter && this.defaultFilter) { │ │ │ │ │ + merged = new OpenLayers.Filter.Logical({ │ │ │ │ │ + type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ + filters: [this.defaultFilter, filter] │ │ │ │ │ + }) │ │ │ │ │ } else { │ │ │ │ │ - string = Object.prototype.toString.call(this) │ │ │ │ │ + merged = filter || this.defaultFilter || undefined │ │ │ │ │ } │ │ │ │ │ - return string │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Filter" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Filter.Logical = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ - filters: null, │ │ │ │ │ - type: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.filters = []; │ │ │ │ │ - OpenLayers.Filter.prototype.initialize.apply(this, [options]) │ │ │ │ │ + return merged │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - this.filters = null; │ │ │ │ │ - OpenLayers.Filter.prototype.destroy.apply(this) │ │ │ │ │ + this.options = null; │ │ │ │ │ + this.format = null │ │ │ │ │ }, │ │ │ │ │ - evaluate: function(context) { │ │ │ │ │ - var i, len; │ │ │ │ │ - switch (this.type) { │ │ │ │ │ - case OpenLayers.Filter.Logical.AND: │ │ │ │ │ - for (i = 0, len = this.filters.length; i < len; i++) { │ │ │ │ │ - if (this.filters[i].evaluate(context) == false) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - case OpenLayers.Filter.Logical.OR: │ │ │ │ │ - for (i = 0, len = this.filters.length; i < len; i++) { │ │ │ │ │ - if (this.filters[i].evaluate(context) == true) { │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return false; │ │ │ │ │ - case OpenLayers.Filter.Logical.NOT: │ │ │ │ │ - return !this.filters[0].evaluate(context) │ │ │ │ │ - } │ │ │ │ │ - return undefined │ │ │ │ │ + read: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.filter = this.mergeWithDefaultFilter(options.filter) │ │ │ │ │ }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - var filters = []; │ │ │ │ │ - for (var i = 0, len = this.filters.length; i < len; ++i) { │ │ │ │ │ - filters.push(this.filters[i].clone()) │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Filter.Logical({ │ │ │ │ │ - type: this.type, │ │ │ │ │ - filters: filters │ │ │ │ │ - }) │ │ │ │ │ + create: function() {}, │ │ │ │ │ + update: function() {}, │ │ │ │ │ + delete: function() {}, │ │ │ │ │ + commit: function() {}, │ │ │ │ │ + abort: function(response) {}, │ │ │ │ │ + createCallback: function(method, response, options) { │ │ │ │ │ + return OpenLayers.Function.bind(function() { │ │ │ │ │ + method.apply(this, [response, options]) │ │ │ │ │ + }, this) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Filter.Logical" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Filter.Logical.AND = "&&"; │ │ │ │ │ -OpenLayers.Filter.Logical.OR = "||"; │ │ │ │ │ -OpenLayers.Filter.Logical.NOT = "!"; │ │ │ │ │ -OpenLayers.Filter.Comparison = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ - type: null, │ │ │ │ │ - property: null, │ │ │ │ │ - value: null, │ │ │ │ │ - matchCase: true, │ │ │ │ │ - lowerBoundary: null, │ │ │ │ │ - upperBoundary: null, │ │ │ │ │ +OpenLayers.Protocol.Response = OpenLayers.Class({ │ │ │ │ │ + code: null, │ │ │ │ │ + requestType: null, │ │ │ │ │ + last: true, │ │ │ │ │ + features: null, │ │ │ │ │ + data: null, │ │ │ │ │ + reqFeatures: null, │ │ │ │ │ + priv: null, │ │ │ │ │ + error: null, │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - OpenLayers.Filter.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (this.type === OpenLayers.Filter.Comparison.LIKE && options.matchCase === undefined) { │ │ │ │ │ - this.matchCase = null │ │ │ │ │ - } │ │ │ │ │ + OpenLayers.Util.extend(this, options) │ │ │ │ │ }, │ │ │ │ │ - evaluate: function(context) { │ │ │ │ │ - if (context instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ - context = context.attributes │ │ │ │ │ + success: function() { │ │ │ │ │ + return this.code > 0 │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.Response" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Protocol.Response.SUCCESS = 1; │ │ │ │ │ +OpenLayers.Protocol.Response.FAILURE = 0; │ │ │ │ │ +OpenLayers.ProxyHost = ""; │ │ │ │ │ +if (!OpenLayers.Request) { │ │ │ │ │ + OpenLayers.Request = {} │ │ │ │ │ +} │ │ │ │ │ +OpenLayers.Util.extend(OpenLayers.Request, { │ │ │ │ │ + DEFAULT_CONFIG: { │ │ │ │ │ + method: "GET", │ │ │ │ │ + url: window.location.href, │ │ │ │ │ + async: true, │ │ │ │ │ + user: undefined, │ │ │ │ │ + password: undefined, │ │ │ │ │ + params: null, │ │ │ │ │ + proxy: OpenLayers.ProxyHost, │ │ │ │ │ + headers: {}, │ │ │ │ │ + data: null, │ │ │ │ │ + callback: function() {}, │ │ │ │ │ + success: null, │ │ │ │ │ + failure: null, │ │ │ │ │ + scope: null │ │ │ │ │ + }, │ │ │ │ │ + URL_SPLIT_REGEX: /([^:]*:)\/\/([^:]*:?[^@]*@)?([^:\/\?]*):?([^\/\?]*)/, │ │ │ │ │ + events: new OpenLayers.Events(this), │ │ │ │ │ + makeSameOrigin: function(url, proxy) { │ │ │ │ │ + var sameOrigin = url.indexOf("http") !== 0; │ │ │ │ │ + var urlParts = !sameOrigin && url.match(this.URL_SPLIT_REGEX); │ │ │ │ │ + if (urlParts) { │ │ │ │ │ + var location = window.location; │ │ │ │ │ + sameOrigin = urlParts[1] == location.protocol && urlParts[3] == location.hostname; │ │ │ │ │ + var uPort = urlParts[4], │ │ │ │ │ + lPort = location.port; │ │ │ │ │ + if (uPort != 80 && uPort != "" || lPort != "80" && lPort != "") { │ │ │ │ │ + sameOrigin = sameOrigin && uPort == lPort │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var result = false; │ │ │ │ │ - var got = context[this.property]; │ │ │ │ │ - var exp; │ │ │ │ │ - switch (this.type) { │ │ │ │ │ - case OpenLayers.Filter.Comparison.EQUAL_TO: │ │ │ │ │ - exp = this.value; │ │ │ │ │ - if (!this.matchCase && typeof got == "string" && typeof exp == "string") { │ │ │ │ │ - result = got.toUpperCase() == exp.toUpperCase() │ │ │ │ │ - } else { │ │ │ │ │ - result = got == exp │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Comparison.NOT_EQUAL_TO: │ │ │ │ │ - exp = this.value; │ │ │ │ │ - if (!this.matchCase && typeof got == "string" && typeof exp == "string") { │ │ │ │ │ - result = got.toUpperCase() != exp.toUpperCase() │ │ │ │ │ + if (!sameOrigin) { │ │ │ │ │ + if (proxy) { │ │ │ │ │ + if (typeof proxy == "function") { │ │ │ │ │ + url = proxy(url) │ │ │ │ │ } else { │ │ │ │ │ - result = got != exp │ │ │ │ │ + url = proxy + encodeURIComponent(url) │ │ │ │ │ } │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Comparison.LESS_THAN: │ │ │ │ │ - result = got < this.value; │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Comparison.GREATER_THAN: │ │ │ │ │ - result = got > this.value; │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO: │ │ │ │ │ - result = got <= this.value; │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO: │ │ │ │ │ - result = got >= this.value; │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Comparison.BETWEEN: │ │ │ │ │ - result = got >= this.lowerBoundary && got <= this.upperBoundary; │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Comparison.LIKE: │ │ │ │ │ - var regexp = new RegExp(this.value, "gi"); │ │ │ │ │ - result = regexp.test(got); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Comparison.IS_NULL: │ │ │ │ │ - result = got === null; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - return result │ │ │ │ │ - }, │ │ │ │ │ - value2regex: function(wildCard, singleChar, escapeChar) { │ │ │ │ │ - if (wildCard == ".") { │ │ │ │ │ - throw new Error("'.' is an unsupported wildCard character for " + "OpenLayers.Filter.Comparison") │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - wildCard = wildCard ? wildCard : "*"; │ │ │ │ │ - singleChar = singleChar ? singleChar : "."; │ │ │ │ │ - escapeChar = escapeChar ? escapeChar : "!"; │ │ │ │ │ - this.value = this.value.replace(new RegExp("\\" + escapeChar + "(.|$)", "g"), "\\$1"); │ │ │ │ │ - this.value = this.value.replace(new RegExp("\\" + singleChar, "g"), "."); │ │ │ │ │ - this.value = this.value.replace(new RegExp("\\" + wildCard, "g"), ".*"); │ │ │ │ │ - this.value = this.value.replace(new RegExp("\\\\.\\*", "g"), "\\" + wildCard); │ │ │ │ │ - this.value = this.value.replace(new RegExp("\\\\\\.", "g"), "\\" + singleChar); │ │ │ │ │ - return this.value │ │ │ │ │ + return url │ │ │ │ │ }, │ │ │ │ │ - regex2value: function() { │ │ │ │ │ - var value = this.value; │ │ │ │ │ - value = value.replace(/!/g, "!!"); │ │ │ │ │ - value = value.replace(/(\\)?\\\./g, function($0, $1) { │ │ │ │ │ - return $1 ? $0 : "!." │ │ │ │ │ - }); │ │ │ │ │ - value = value.replace(/(\\)?\\\*/g, function($0, $1) { │ │ │ │ │ - return $1 ? $0 : "!*" │ │ │ │ │ + issue: function(config) { │ │ │ │ │ + var defaultConfig = OpenLayers.Util.extend(this.DEFAULT_CONFIG, { │ │ │ │ │ + proxy: OpenLayers.ProxyHost │ │ │ │ │ }); │ │ │ │ │ - value = value.replace(/\\\\/g, "\\"); │ │ │ │ │ - value = value.replace(/\.\*/g, "*"); │ │ │ │ │ - return value │ │ │ │ │ - }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - return OpenLayers.Util.extend(new OpenLayers.Filter.Comparison, this) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Filter.Comparison" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Filter.Comparison.EQUAL_TO = "=="; │ │ │ │ │ -OpenLayers.Filter.Comparison.NOT_EQUAL_TO = "!="; │ │ │ │ │ -OpenLayers.Filter.Comparison.LESS_THAN = "<"; │ │ │ │ │ -OpenLayers.Filter.Comparison.GREATER_THAN = ">"; │ │ │ │ │ -OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO = "<="; │ │ │ │ │ -OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO = ">="; │ │ │ │ │ -OpenLayers.Filter.Comparison.BETWEEN = ".."; │ │ │ │ │ -OpenLayers.Filter.Comparison.LIKE = "~"; │ │ │ │ │ -OpenLayers.Filter.Comparison.IS_NULL = "NULL"; │ │ │ │ │ -OpenLayers.Popup = OpenLayers.Class({ │ │ │ │ │ - events: null, │ │ │ │ │ - id: "", │ │ │ │ │ - lonlat: null, │ │ │ │ │ - div: null, │ │ │ │ │ - contentSize: null, │ │ │ │ │ - size: null, │ │ │ │ │ - contentHTML: null, │ │ │ │ │ - backgroundColor: "", │ │ │ │ │ - opacity: "", │ │ │ │ │ - border: "", │ │ │ │ │ - contentDiv: null, │ │ │ │ │ - groupDiv: null, │ │ │ │ │ - closeDiv: null, │ │ │ │ │ - autoSize: false, │ │ │ │ │ - minSize: null, │ │ │ │ │ - maxSize: null, │ │ │ │ │ - displayClass: "olPopup", │ │ │ │ │ - contentDisplayClass: "olPopupContent", │ │ │ │ │ - padding: 0, │ │ │ │ │ - disableFirefoxOverflowHack: false, │ │ │ │ │ - fixPadding: function() { │ │ │ │ │ - if (typeof this.padding == "number") { │ │ │ │ │ - this.padding = new OpenLayers.Bounds(this.padding, this.padding, this.padding, this.padding) │ │ │ │ │ + config = config || {}; │ │ │ │ │ + config.headers = config.headers || {}; │ │ │ │ │ + config = OpenLayers.Util.applyDefaults(config, defaultConfig); │ │ │ │ │ + config.headers = OpenLayers.Util.applyDefaults(config.headers, defaultConfig.headers); │ │ │ │ │ + var customRequestedWithHeader = false, │ │ │ │ │ + headerKey; │ │ │ │ │ + for (headerKey in config.headers) { │ │ │ │ │ + if (config.headers.hasOwnProperty(headerKey)) { │ │ │ │ │ + if (headerKey.toLowerCase() === "x-requested-with") { │ │ │ │ │ + customRequestedWithHeader = true │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - panMapIfOutOfView: false, │ │ │ │ │ - keepInMap: false, │ │ │ │ │ - closeOnMove: false, │ │ │ │ │ - map: null, │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, closeBox, closeBoxCallback) { │ │ │ │ │ - if (id == null) { │ │ │ │ │ - id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ + if (customRequestedWithHeader === false) { │ │ │ │ │ + config.headers["X-Requested-With"] = "XMLHttpRequest" │ │ │ │ │ } │ │ │ │ │ - this.id = id; │ │ │ │ │ - this.lonlat = lonlat; │ │ │ │ │ - this.contentSize = contentSize != null ? contentSize : new OpenLayers.Size(OpenLayers.Popup.WIDTH, OpenLayers.Popup.HEIGHT); │ │ │ │ │ - if (contentHTML != null) { │ │ │ │ │ - this.contentHTML = contentHTML │ │ │ │ │ + var request = new OpenLayers.Request.XMLHttpRequest; │ │ │ │ │ + var url = OpenLayers.Util.urlAppend(config.url, OpenLayers.Util.getParameterString(config.params || {})); │ │ │ │ │ + url = OpenLayers.Request.makeSameOrigin(url, config.proxy); │ │ │ │ │ + request.open(config.method, url, config.async, config.user, config.password); │ │ │ │ │ + for (var header in config.headers) { │ │ │ │ │ + request.setRequestHeader(header, config.headers[header]) │ │ │ │ │ } │ │ │ │ │ - this.backgroundColor = OpenLayers.Popup.COLOR; │ │ │ │ │ - this.opacity = OpenLayers.Popup.OPACITY; │ │ │ │ │ - this.border = OpenLayers.Popup.BORDER; │ │ │ │ │ - this.div = OpenLayers.Util.createDiv(this.id, null, null, null, null, null, "hidden"); │ │ │ │ │ - this.div.className = this.displayClass; │ │ │ │ │ - var groupDivId = this.id + "_GroupDiv"; │ │ │ │ │ - this.groupDiv = OpenLayers.Util.createDiv(groupDivId, null, null, null, "relative", null, "hidden"); │ │ │ │ │ - var id = this.div.id + "_contentDiv"; │ │ │ │ │ - this.contentDiv = OpenLayers.Util.createDiv(id, null, this.contentSize.clone(), null, "relative"); │ │ │ │ │ - this.contentDiv.className = this.contentDisplayClass; │ │ │ │ │ - this.groupDiv.appendChild(this.contentDiv); │ │ │ │ │ - this.div.appendChild(this.groupDiv); │ │ │ │ │ - if (closeBox) { │ │ │ │ │ - this.addCloseBox(closeBoxCallback) │ │ │ │ │ + var events = this.events; │ │ │ │ │ + var self = this; │ │ │ │ │ + request.onreadystatechange = function() { │ │ │ │ │ + if (request.readyState == OpenLayers.Request.XMLHttpRequest.DONE) { │ │ │ │ │ + var proceed = events.triggerEvent("complete", { │ │ │ │ │ + request: request, │ │ │ │ │ + config: config, │ │ │ │ │ + requestUrl: url │ │ │ │ │ + }); │ │ │ │ │ + if (proceed !== false) { │ │ │ │ │ + self.runCallbacks({ │ │ │ │ │ + request: request, │ │ │ │ │ + config: config, │ │ │ │ │ + requestUrl: url │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + if (config.async === false) { │ │ │ │ │ + request.send(config.data) │ │ │ │ │ + } else { │ │ │ │ │ + window.setTimeout(function() { │ │ │ │ │ + if (request.readyState !== 0) { │ │ │ │ │ + request.send(config.data) │ │ │ │ │ + } │ │ │ │ │ + }, 0) │ │ │ │ │ } │ │ │ │ │ - this.registerEvents() │ │ │ │ │ + return request │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.id = null; │ │ │ │ │ - this.lonlat = null; │ │ │ │ │ - this.size = null; │ │ │ │ │ - this.contentHTML = null; │ │ │ │ │ - this.backgroundColor = null; │ │ │ │ │ - this.opacity = null; │ │ │ │ │ - this.border = null; │ │ │ │ │ - if (this.closeOnMove && this.map) { │ │ │ │ │ - this.map.events.unregister("movestart", this, this.hide) │ │ │ │ │ + runCallbacks: function(options) { │ │ │ │ │ + var request = options.request; │ │ │ │ │ + var config = options.config; │ │ │ │ │ + var complete = config.scope ? OpenLayers.Function.bind(config.callback, config.scope) : config.callback; │ │ │ │ │ + var success; │ │ │ │ │ + if (config.success) { │ │ │ │ │ + success = config.scope ? OpenLayers.Function.bind(config.success, config.scope) : config.success │ │ │ │ │ } │ │ │ │ │ - this.events.destroy(); │ │ │ │ │ - this.events = null; │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - OpenLayers.Event.stopObservingElement(this.closeDiv); │ │ │ │ │ - this.groupDiv.removeChild(this.closeDiv) │ │ │ │ │ + var failure; │ │ │ │ │ + if (config.failure) { │ │ │ │ │ + failure = config.scope ? OpenLayers.Function.bind(config.failure, config.scope) : config.failure │ │ │ │ │ } │ │ │ │ │ - this.closeDiv = null; │ │ │ │ │ - this.div.removeChild(this.groupDiv); │ │ │ │ │ - this.groupDiv = null; │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.removePopup(this) │ │ │ │ │ + if (OpenLayers.Util.createUrlObject(config.url).protocol == "file:" && request.responseText) { │ │ │ │ │ + request.status = 200 │ │ │ │ │ } │ │ │ │ │ - this.map = null; │ │ │ │ │ - this.div = null; │ │ │ │ │ - this.autoSize = null; │ │ │ │ │ - this.minSize = null; │ │ │ │ │ - this.maxSize = null; │ │ │ │ │ - this.padding = null; │ │ │ │ │ - this.panMapIfOutOfView = null │ │ │ │ │ - }, │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - if (px == null) { │ │ │ │ │ - if (this.lonlat != null && this.map != null) { │ │ │ │ │ - px = this.map.getLayerPxFromLonLat(this.lonlat) │ │ │ │ │ + complete(request); │ │ │ │ │ + if (!request.status || request.status >= 200 && request.status < 300) { │ │ │ │ │ + this.events.triggerEvent("success", options); │ │ │ │ │ + if (success) { │ │ │ │ │ + success(request) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (this.closeOnMove) { │ │ │ │ │ - this.map.events.register("movestart", this, this.hide) │ │ │ │ │ - } │ │ │ │ │ - if (!this.disableFirefoxOverflowHack && OpenLayers.BROWSER_NAME == "firefox") { │ │ │ │ │ - this.map.events.register("movestart", this, function() { │ │ │ │ │ - var style = document.defaultView.getComputedStyle(this.contentDiv, null); │ │ │ │ │ - var currentOverflow = style.getPropertyValue("overflow"); │ │ │ │ │ - if (currentOverflow != "hidden") { │ │ │ │ │ - this.contentDiv._oldOverflow = currentOverflow; │ │ │ │ │ - this.contentDiv.style.overflow = "hidden" │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - this.map.events.register("moveend", this, function() { │ │ │ │ │ - var oldOverflow = this.contentDiv._oldOverflow; │ │ │ │ │ - if (oldOverflow) { │ │ │ │ │ - this.contentDiv.style.overflow = oldOverflow; │ │ │ │ │ - this.contentDiv._oldOverflow = null │ │ │ │ │ - } │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.moveTo(px); │ │ │ │ │ - if (!this.autoSize && !this.size) { │ │ │ │ │ - this.setSize(this.contentSize) │ │ │ │ │ - } │ │ │ │ │ - this.setBackgroundColor(); │ │ │ │ │ - this.setOpacity(); │ │ │ │ │ - this.setBorder(); │ │ │ │ │ - this.setContentHTML(); │ │ │ │ │ - if (this.panMapIfOutOfView) { │ │ │ │ │ - this.panIntoView() │ │ │ │ │ - } │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - updatePosition: function() { │ │ │ │ │ - if (this.lonlat && this.map) { │ │ │ │ │ - var px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ - if (px) { │ │ │ │ │ - this.moveTo(px) │ │ │ │ │ + if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ + this.events.triggerEvent("failure", options); │ │ │ │ │ + if (failure) { │ │ │ │ │ + failure(request) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - if (px != null && this.div != null) { │ │ │ │ │ - this.div.style.left = px.x + "px"; │ │ │ │ │ - this.div.style.top = px.y + "px" │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - visible: function() { │ │ │ │ │ - return OpenLayers.Element.visible(this.div) │ │ │ │ │ + GET: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "GET" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config) │ │ │ │ │ }, │ │ │ │ │ - toggle: function() { │ │ │ │ │ - if (this.visible()) { │ │ │ │ │ - this.hide() │ │ │ │ │ - } else { │ │ │ │ │ - this.show() │ │ │ │ │ + POST: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "POST" │ │ │ │ │ + }); │ │ │ │ │ + config.headers = config.headers ? config.headers : {}; │ │ │ │ │ + if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ + config.headers["Content-Type"] = "application/xml" │ │ │ │ │ } │ │ │ │ │ + return OpenLayers.Request.issue(config) │ │ │ │ │ }, │ │ │ │ │ - show: function() { │ │ │ │ │ - this.div.style.display = ""; │ │ │ │ │ - if (this.panMapIfOutOfView) { │ │ │ │ │ - this.panIntoView() │ │ │ │ │ + PUT: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "PUT" │ │ │ │ │ + }); │ │ │ │ │ + config.headers = config.headers ? config.headers : {}; │ │ │ │ │ + if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ + config.headers["Content-Type"] = "application/xml" │ │ │ │ │ } │ │ │ │ │ + return OpenLayers.Request.issue(config) │ │ │ │ │ }, │ │ │ │ │ - hide: function() { │ │ │ │ │ - this.div.style.display = "none" │ │ │ │ │ + DELETE: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "DELETE" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config) │ │ │ │ │ }, │ │ │ │ │ - setSize: function(contentSize) { │ │ │ │ │ - this.size = contentSize.clone(); │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ - var wPadding = contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ - var hPadding = contentDivPadding.top + contentDivPadding.bottom; │ │ │ │ │ - this.fixPadding(); │ │ │ │ │ - wPadding += this.padding.left + this.padding.right; │ │ │ │ │ - hPadding += this.padding.top + this.padding.bottom; │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - var closeDivWidth = parseInt(this.closeDiv.style.width); │ │ │ │ │ - wPadding += closeDivWidth + contentDivPadding.right │ │ │ │ │ - } │ │ │ │ │ - this.size.w += wPadding; │ │ │ │ │ - this.size.h += hPadding; │ │ │ │ │ - if (OpenLayers.BROWSER_NAME == "msie") { │ │ │ │ │ - this.contentSize.w += contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ - this.contentSize.h += contentDivPadding.bottom + contentDivPadding.top │ │ │ │ │ - } │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.div.style.width = this.size.w + "px"; │ │ │ │ │ - this.div.style.height = this.size.h + "px" │ │ │ │ │ - } │ │ │ │ │ - if (this.contentDiv != null) { │ │ │ │ │ - this.contentDiv.style.width = contentSize.w + "px"; │ │ │ │ │ - this.contentDiv.style.height = contentSize.h + "px" │ │ │ │ │ - } │ │ │ │ │ + HEAD: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "HEAD" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config) │ │ │ │ │ }, │ │ │ │ │ - updateSize: function() { │ │ │ │ │ - var preparedHTML = "<div class='" + this.contentDisplayClass + "'>" + this.contentDiv.innerHTML + "</div>"; │ │ │ │ │ - var containerElement = this.map ? this.map.div : document.body; │ │ │ │ │ - var realSize = OpenLayers.Util.getRenderedDimensions(preparedHTML, null, { │ │ │ │ │ - displayClass: this.displayClass, │ │ │ │ │ - containerElement: containerElement │ │ │ │ │ + OPTIONS: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "OPTIONS" │ │ │ │ │ }); │ │ │ │ │ - var safeSize = this.getSafeContentSize(realSize); │ │ │ │ │ - var newSize = null; │ │ │ │ │ - if (safeSize.equals(realSize)) { │ │ │ │ │ - newSize = realSize │ │ │ │ │ - } else { │ │ │ │ │ - var fixedSize = { │ │ │ │ │ - w: safeSize.w < realSize.w ? safeSize.w : null, │ │ │ │ │ - h: safeSize.h < realSize.h ? safeSize.h : null │ │ │ │ │ - }; │ │ │ │ │ - if (fixedSize.w && fixedSize.h) { │ │ │ │ │ - newSize = safeSize │ │ │ │ │ - } else { │ │ │ │ │ - var clippedSize = OpenLayers.Util.getRenderedDimensions(preparedHTML, fixedSize, { │ │ │ │ │ - displayClass: this.contentDisplayClass, │ │ │ │ │ - containerElement: containerElement │ │ │ │ │ - }); │ │ │ │ │ - var currentOverflow = OpenLayers.Element.getStyle(this.contentDiv, "overflow"); │ │ │ │ │ - if (currentOverflow != "hidden" && clippedSize.equals(safeSize)) { │ │ │ │ │ - var scrollBar = OpenLayers.Util.getScrollbarWidth(); │ │ │ │ │ - if (fixedSize.w) { │ │ │ │ │ - clippedSize.h += scrollBar │ │ │ │ │ - } else { │ │ │ │ │ - clippedSize.w += scrollBar │ │ │ │ │ - } │ │ │ │ │ + return OpenLayers.Request.issue(config) │ │ │ │ │ + } │ │ │ │ │ +}); │ │ │ │ │ +(function() { │ │ │ │ │ + var oXMLHttpRequest = window.XMLHttpRequest; │ │ │ │ │ + var bGecko = !!window.controllers, │ │ │ │ │ + bIE = window.document.all && !window.opera, │ │ │ │ │ + bIE7 = bIE && window.navigator.userAgent.match(/MSIE 7.0/); │ │ │ │ │ + │ │ │ │ │ + function fXMLHttpRequest() { │ │ │ │ │ + this._object = oXMLHttpRequest && !bIE7 ? new oXMLHttpRequest : new window.ActiveXObject("Microsoft.XMLHTTP"); │ │ │ │ │ + this._listeners = [] │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function cXMLHttpRequest() { │ │ │ │ │ + return new fXMLHttpRequest │ │ │ │ │ + } │ │ │ │ │ + cXMLHttpRequest.prototype = fXMLHttpRequest.prototype; │ │ │ │ │ + if (bGecko && oXMLHttpRequest.wrapped) cXMLHttpRequest.wrapped = oXMLHttpRequest.wrapped; │ │ │ │ │ + cXMLHttpRequest.UNSENT = 0; │ │ │ │ │ + cXMLHttpRequest.OPENED = 1; │ │ │ │ │ + cXMLHttpRequest.HEADERS_RECEIVED = 2; │ │ │ │ │ + cXMLHttpRequest.LOADING = 3; │ │ │ │ │ + cXMLHttpRequest.DONE = 4; │ │ │ │ │ + cXMLHttpRequest.prototype.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + cXMLHttpRequest.prototype.responseText = ""; │ │ │ │ │ + cXMLHttpRequest.prototype.responseXML = null; │ │ │ │ │ + cXMLHttpRequest.prototype.status = 0; │ │ │ │ │ + cXMLHttpRequest.prototype.statusText = ""; │ │ │ │ │ + cXMLHttpRequest.prototype.priority = "NORMAL"; │ │ │ │ │ + cXMLHttpRequest.prototype.onreadystatechange = null; │ │ │ │ │ + cXMLHttpRequest.onreadystatechange = null; │ │ │ │ │ + cXMLHttpRequest.onopen = null; │ │ │ │ │ + cXMLHttpRequest.onsend = null; │ │ │ │ │ + cXMLHttpRequest.onabort = null; │ │ │ │ │ + cXMLHttpRequest.prototype.open = function(sMethod, sUrl, bAsync, sUser, sPassword) { │ │ │ │ │ + delete this._headers; │ │ │ │ │ + if (arguments.length < 3) bAsync = true; │ │ │ │ │ + this._async = bAsync; │ │ │ │ │ + var oRequest = this, │ │ │ │ │ + nState = this.readyState, │ │ │ │ │ + fOnUnload; │ │ │ │ │ + if (bIE && bAsync) { │ │ │ │ │ + fOnUnload = function() { │ │ │ │ │ + if (nState != cXMLHttpRequest.DONE) { │ │ │ │ │ + fCleanTransport(oRequest); │ │ │ │ │ + oRequest.abort() │ │ │ │ │ } │ │ │ │ │ - newSize = this.getSafeContentSize(clippedSize) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.setSize(newSize) │ │ │ │ │ - }, │ │ │ │ │ - setBackgroundColor: function(color) { │ │ │ │ │ - if (color != undefined) { │ │ │ │ │ - this.backgroundColor = color │ │ │ │ │ - } │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.div.style.backgroundColor = this.backgroundColor │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - if (opacity != undefined) { │ │ │ │ │ - this.opacity = opacity │ │ │ │ │ - } │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.div.style.opacity = this.opacity; │ │ │ │ │ - this.div.style.filter = "alpha(opacity=" + this.opacity * 100 + ")" │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setBorder: function(border) { │ │ │ │ │ - if (border != undefined) { │ │ │ │ │ - this.border = border │ │ │ │ │ - } │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.div.style.border = this.border │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setContentHTML: function(contentHTML) { │ │ │ │ │ - if (contentHTML != null) { │ │ │ │ │ - this.contentHTML = contentHTML │ │ │ │ │ - } │ │ │ │ │ - if (this.contentDiv != null && this.contentHTML != null && this.contentHTML != this.contentDiv.innerHTML) { │ │ │ │ │ - this.contentDiv.innerHTML = this.contentHTML; │ │ │ │ │ - if (this.autoSize) { │ │ │ │ │ - this.registerImageListeners(); │ │ │ │ │ - this.updateSize() │ │ │ │ │ - } │ │ │ │ │ + }; │ │ │ │ │ + window.attachEvent("onunload", fOnUnload) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - registerImageListeners: function() { │ │ │ │ │ - var onImgLoad = function() { │ │ │ │ │ - if (this.popup.id === null) { │ │ │ │ │ + if (cXMLHttpRequest.onopen) cXMLHttpRequest.onopen.apply(this, arguments); │ │ │ │ │ + if (arguments.length > 4) this._object.open(sMethod, sUrl, bAsync, sUser, sPassword); │ │ │ │ │ + else if (arguments.length > 3) this._object.open(sMethod, sUrl, bAsync, sUser); │ │ │ │ │ + else this._object.open(sMethod, sUrl, bAsync); │ │ │ │ │ + this.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ + fReadyStateChange(this); │ │ │ │ │ + this._object.onreadystatechange = function() { │ │ │ │ │ + if (bGecko && !bAsync) return; │ │ │ │ │ + oRequest.readyState = oRequest._object.readyState; │ │ │ │ │ + fSynchronizeValues(oRequest); │ │ │ │ │ + if (oRequest._aborted) { │ │ │ │ │ + oRequest.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ return │ │ │ │ │ } │ │ │ │ │ - this.popup.updateSize(); │ │ │ │ │ - if (this.popup.visible() && this.popup.panMapIfOutOfView) { │ │ │ │ │ - this.popup.panIntoView() │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.stopObserving(this.img, "load", this.img._onImgLoad) │ │ │ │ │ - }; │ │ │ │ │ - var images = this.contentDiv.getElementsByTagName("img"); │ │ │ │ │ - for (var i = 0, len = images.length; i < len; i++) { │ │ │ │ │ - var img = images[i]; │ │ │ │ │ - if (img.width == 0 || img.height == 0) { │ │ │ │ │ - var context = { │ │ │ │ │ - popup: this, │ │ │ │ │ - img: img │ │ │ │ │ - }; │ │ │ │ │ - img._onImgLoad = OpenLayers.Function.bind(onImgLoad, context); │ │ │ │ │ - OpenLayers.Event.observe(img, "load", img._onImgLoad) │ │ │ │ │ + if (oRequest.readyState == cXMLHttpRequest.DONE) { │ │ │ │ │ + delete oRequest._data; │ │ │ │ │ + fCleanTransport(oRequest); │ │ │ │ │ + if (bIE && bAsync) window.detachEvent("onunload", fOnUnload) │ │ │ │ │ } │ │ │ │ │ + if (nState != oRequest.readyState) fReadyStateChange(oRequest); │ │ │ │ │ + nState = oRequest.readyState │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - getSafeContentSize: function(size) { │ │ │ │ │ - var safeContentSize = size.clone(); │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ - var wPadding = contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ - var hPadding = contentDivPadding.top + contentDivPadding.bottom; │ │ │ │ │ - this.fixPadding(); │ │ │ │ │ - wPadding += this.padding.left + this.padding.right; │ │ │ │ │ - hPadding += this.padding.top + this.padding.bottom; │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - var closeDivWidth = parseInt(this.closeDiv.style.width); │ │ │ │ │ - wPadding += closeDivWidth + contentDivPadding.right │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fXMLHttpRequest_send(oRequest) { │ │ │ │ │ + oRequest._object.send(oRequest._data); │ │ │ │ │ + if (bGecko && !oRequest._async) { │ │ │ │ │ + oRequest.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ + fSynchronizeValues(oRequest); │ │ │ │ │ + while (oRequest.readyState < cXMLHttpRequest.DONE) { │ │ │ │ │ + oRequest.readyState++; │ │ │ │ │ + fReadyStateChange(oRequest); │ │ │ │ │ + if (oRequest._aborted) return │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (this.minSize) { │ │ │ │ │ - safeContentSize.w = Math.max(safeContentSize.w, this.minSize.w - wPadding); │ │ │ │ │ - safeContentSize.h = Math.max(safeContentSize.h, this.minSize.h - hPadding) │ │ │ │ │ + } │ │ │ │ │ + cXMLHttpRequest.prototype.send = function(vData) { │ │ │ │ │ + if (cXMLHttpRequest.onsend) cXMLHttpRequest.onsend.apply(this, arguments); │ │ │ │ │ + if (!arguments.length) vData = null; │ │ │ │ │ + if (vData && vData.nodeType) { │ │ │ │ │ + vData = window.XMLSerializer ? (new window.XMLSerializer).serializeToString(vData) : vData.xml; │ │ │ │ │ + if (!this._headers["Content-Type"]) this._object.setRequestHeader("Content-Type", "application/xml") │ │ │ │ │ } │ │ │ │ │ - if (this.maxSize) { │ │ │ │ │ - safeContentSize.w = Math.min(safeContentSize.w, this.maxSize.w - wPadding); │ │ │ │ │ - safeContentSize.h = Math.min(safeContentSize.h, this.maxSize.h - hPadding) │ │ │ │ │ + this._data = vData; │ │ │ │ │ + fXMLHttpRequest_send(this) │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.abort = function() { │ │ │ │ │ + if (cXMLHttpRequest.onabort) cXMLHttpRequest.onabort.apply(this, arguments); │ │ │ │ │ + if (this.readyState > cXMLHttpRequest.UNSENT) this._aborted = true; │ │ │ │ │ + this._object.abort(); │ │ │ │ │ + fCleanTransport(this); │ │ │ │ │ + this.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + delete this._data │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.getAllResponseHeaders = function() { │ │ │ │ │ + return this._object.getAllResponseHeaders() │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.getResponseHeader = function(sName) { │ │ │ │ │ + return this._object.getResponseHeader(sName) │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.setRequestHeader = function(sName, sValue) { │ │ │ │ │ + if (!this._headers) this._headers = {}; │ │ │ │ │ + this._headers[sName] = sValue; │ │ │ │ │ + return this._object.setRequestHeader(sName, sValue) │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.addEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ + for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ + if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) return; │ │ │ │ │ + this._listeners.push([sName, fHandler, bUseCapture]) │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.removeEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ + for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ + if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) break; │ │ │ │ │ + if (oListener) this._listeners.splice(nIndex, 1) │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.dispatchEvent = function(oEvent) { │ │ │ │ │ + var oEventPseudo = { │ │ │ │ │ + type: oEvent.type, │ │ │ │ │ + target: this, │ │ │ │ │ + currentTarget: this, │ │ │ │ │ + eventPhase: 2, │ │ │ │ │ + bubbles: oEvent.bubbles, │ │ │ │ │ + cancelable: oEvent.cancelable, │ │ │ │ │ + timeStamp: oEvent.timeStamp, │ │ │ │ │ + stopPropagation: function() {}, │ │ │ │ │ + preventDefault: function() {}, │ │ │ │ │ + initEvent: function() {} │ │ │ │ │ + }; │ │ │ │ │ + if (oEventPseudo.type == "readystatechange" && this.onreadystatechange)(this.onreadystatechange.handleEvent || this.onreadystatechange).apply(this, [oEventPseudo]); │ │ │ │ │ + for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ + if (oListener[0] == oEventPseudo.type && !oListener[2])(oListener[1].handleEvent || oListener[1]).apply(this, [oEventPseudo]) │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.toString = function() { │ │ │ │ │ + return "[" + "object" + " " + "XMLHttpRequest" + "]" │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.toString = function() { │ │ │ │ │ + return "[" + "XMLHttpRequest" + "]" │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fReadyStateChange(oRequest) { │ │ │ │ │ + if (cXMLHttpRequest.onreadystatechange) cXMLHttpRequest.onreadystatechange.apply(oRequest); │ │ │ │ │ + oRequest.dispatchEvent({ │ │ │ │ │ + type: "readystatechange", │ │ │ │ │ + bubbles: false, │ │ │ │ │ + cancelable: false, │ │ │ │ │ + timeStamp: new Date + 0 │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function fGetDocument(oRequest) { │ │ │ │ │ + var oDocument = oRequest.responseXML, │ │ │ │ │ + sResponse = oRequest.responseText; │ │ │ │ │ + if (bIE && sResponse && oDocument && !oDocument.documentElement && oRequest.getResponseHeader("Content-Type").match(/[^\/]+\/[^\+]+\+xml/)) { │ │ │ │ │ + oDocument = new window.ActiveXObject("Microsoft.XMLDOM"); │ │ │ │ │ + oDocument.async = false; │ │ │ │ │ + oDocument.validateOnParse = false; │ │ │ │ │ + oDocument.loadXML(sResponse) │ │ │ │ │ } │ │ │ │ │ - if (this.map && this.map.size) { │ │ │ │ │ - var extraX = 0, │ │ │ │ │ - extraY = 0; │ │ │ │ │ - if (this.keepInMap && !this.panMapIfOutOfView) { │ │ │ │ │ - var px = this.map.getPixelFromLonLat(this.lonlat); │ │ │ │ │ - switch (this.relativePosition) { │ │ │ │ │ - case "tr": │ │ │ │ │ - extraX = px.x; │ │ │ │ │ - extraY = this.map.size.h - px.y; │ │ │ │ │ - break; │ │ │ │ │ - case "tl": │ │ │ │ │ - extraX = this.map.size.w - px.x; │ │ │ │ │ - extraY = this.map.size.h - px.y; │ │ │ │ │ - break; │ │ │ │ │ - case "bl": │ │ │ │ │ - extraX = this.map.size.w - px.x; │ │ │ │ │ - extraY = px.y; │ │ │ │ │ - break; │ │ │ │ │ - case "br": │ │ │ │ │ - extraX = px.x; │ │ │ │ │ - extraY = px.y; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - extraX = px.x; │ │ │ │ │ - extraY = this.map.size.h - px.y; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var maxY = this.map.size.h - this.map.paddingForPopups.top - this.map.paddingForPopups.bottom - hPadding - extraY; │ │ │ │ │ - var maxX = this.map.size.w - this.map.paddingForPopups.left - this.map.paddingForPopups.right - wPadding - extraX; │ │ │ │ │ - safeContentSize.w = Math.min(safeContentSize.w, maxX); │ │ │ │ │ - safeContentSize.h = Math.min(safeContentSize.h, maxY) │ │ │ │ │ + if (oDocument) │ │ │ │ │ + if (bIE && oDocument.parseError != 0 || !oDocument.documentElement || oDocument.documentElement && oDocument.documentElement.tagName == "parsererror") return null; │ │ │ │ │ + return oDocument │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function fSynchronizeValues(oRequest) { │ │ │ │ │ + try { │ │ │ │ │ + oRequest.responseText = oRequest._object.responseText │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + try { │ │ │ │ │ + oRequest.responseXML = fGetDocument(oRequest._object) │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + try { │ │ │ │ │ + oRequest.status = oRequest._object.status │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + try { │ │ │ │ │ + oRequest.statusText = oRequest._object.statusText │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function fCleanTransport(oRequest) { │ │ │ │ │ + oRequest._object.onreadystatechange = new window.Function │ │ │ │ │ + } │ │ │ │ │ + if (!window.Function.prototype.apply) { │ │ │ │ │ + window.Function.prototype.apply = function(oRequest, oArguments) { │ │ │ │ │ + if (!oArguments) oArguments = []; │ │ │ │ │ + oRequest.__func = this; │ │ │ │ │ + oRequest.__func(oArguments[0], oArguments[1], oArguments[2], oArguments[3], oArguments[4]); │ │ │ │ │ + delete oRequest.__func │ │ │ │ │ } │ │ │ │ │ - return safeContentSize │ │ │ │ │ - }, │ │ │ │ │ - getContentDivPadding: function() { │ │ │ │ │ - var contentDivPadding = this._contentDivPadding; │ │ │ │ │ - if (!contentDivPadding) { │ │ │ │ │ - if (this.div.parentNode == null) { │ │ │ │ │ - this.div.style.display = "none"; │ │ │ │ │ - document.body.appendChild(this.div) │ │ │ │ │ - } │ │ │ │ │ - contentDivPadding = new OpenLayers.Bounds(OpenLayers.Element.getStyle(this.contentDiv, "padding-left"), OpenLayers.Element.getStyle(this.contentDiv, "padding-bottom"), OpenLayers.Element.getStyle(this.contentDiv, "padding-right"), OpenLayers.Element.getStyle(this.contentDiv, "padding-top")); │ │ │ │ │ - this._contentDivPadding = contentDivPadding; │ │ │ │ │ - if (this.div.parentNode == document.body) { │ │ │ │ │ - document.body.removeChild(this.div); │ │ │ │ │ - this.div.style.display = "" │ │ │ │ │ + } │ │ │ │ │ + if (!OpenLayers.Request) { │ │ │ │ │ + OpenLayers.Request = {} │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Request.XMLHttpRequest = cXMLHttpRequest │ │ │ │ │ +})(); │ │ │ │ │ +OpenLayers.Protocol.HTTP = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + url: null, │ │ │ │ │ + headers: null, │ │ │ │ │ + params: null, │ │ │ │ │ + callback: null, │ │ │ │ │ + scope: null, │ │ │ │ │ + readWithPOST: false, │ │ │ │ │ + updateWithPOST: false, │ │ │ │ │ + deleteWithPOST: false, │ │ │ │ │ + wildcarded: false, │ │ │ │ │ + srsInBBOX: false, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + this.params = {}; │ │ │ │ │ + this.headers = {}; │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ + var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ + wildcarded: this.wildcarded, │ │ │ │ │ + srsInBBOX: this.srsInBBOX │ │ │ │ │ + }); │ │ │ │ │ + this.filterToParams = function(filter, params) { │ │ │ │ │ + return format.write(filter, params) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return contentDivPadding │ │ │ │ │ }, │ │ │ │ │ - addCloseBox: function(callback) { │ │ │ │ │ - this.closeDiv = OpenLayers.Util.createDiv(this.id + "_close", null, { │ │ │ │ │ - w: 17, │ │ │ │ │ - h: 17 │ │ │ │ │ - }); │ │ │ │ │ - this.closeDiv.className = "olPopupCloseBox"; │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ - this.closeDiv.style.right = contentDivPadding.right + "px"; │ │ │ │ │ - this.closeDiv.style.top = contentDivPadding.top + "px"; │ │ │ │ │ - this.groupDiv.appendChild(this.closeDiv); │ │ │ │ │ - var closePopup = callback || function(e) { │ │ │ │ │ - this.hide(); │ │ │ │ │ - OpenLayers.Event.stop(e) │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Event.observe(this.closeDiv, "touchend", OpenLayers.Function.bindAsEventListener(closePopup, this)); │ │ │ │ │ - OpenLayers.Event.observe(this.closeDiv, "click", OpenLayers.Function.bindAsEventListener(closePopup, this)) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.params = null; │ │ │ │ │ + this.headers = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ }, │ │ │ │ │ - panIntoView: function() { │ │ │ │ │ - var mapSize = this.map.getSize(); │ │ │ │ │ - var origTL = this.map.getViewPortPxFromLayerPx(new OpenLayers.Pixel(parseInt(this.div.style.left), parseInt(this.div.style.top))); │ │ │ │ │ - var newTL = origTL.clone(); │ │ │ │ │ - if (origTL.x < this.map.paddingForPopups.left) { │ │ │ │ │ - newTL.x = this.map.paddingForPopups.left │ │ │ │ │ - } else if (origTL.x + this.size.w > mapSize.w - this.map.paddingForPopups.right) { │ │ │ │ │ - newTL.x = mapSize.w - this.map.paddingForPopups.right - this.size.w │ │ │ │ │ - } │ │ │ │ │ - if (origTL.y < this.map.paddingForPopups.top) { │ │ │ │ │ - newTL.y = this.map.paddingForPopups.top │ │ │ │ │ - } else if (origTL.y + this.size.h > mapSize.h - this.map.paddingForPopups.bottom) { │ │ │ │ │ - newTL.y = mapSize.h - this.map.paddingForPopups.bottom - this.size.h │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.params = OpenLayers.Util.applyDefaults(options.params, this.options.params); │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + if (options.filter && this.filterToParams) { │ │ │ │ │ + options.params = this.filterToParams(options.filter, options.params) │ │ │ │ │ } │ │ │ │ │ - var dx = origTL.x - newTL.x; │ │ │ │ │ - var dy = origTL.y - newTL.y; │ │ │ │ │ - this.map.pan(dx, dy) │ │ │ │ │ - }, │ │ │ │ │ - registerEvents: function() { │ │ │ │ │ - this.events = new OpenLayers.Events(this, this.div, null, true); │ │ │ │ │ - │ │ │ │ │ - function onTouchstart(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt, true) │ │ │ │ │ + var readWithPOST = options.readWithPOST !== undefined ? options.readWithPOST : this.readWithPOST; │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + if (readWithPOST) { │ │ │ │ │ + var headers = options.headers || {}; │ │ │ │ │ + headers["Content-Type"] = "application/x-www-form-urlencoded"; │ │ │ │ │ + resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ + data: OpenLayers.Util.getParameterString(options.params), │ │ │ │ │ + headers: headers │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + resp.priv = OpenLayers.Request.GET({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - this.events.on({ │ │ │ │ │ - mousedown: this.onmousedown, │ │ │ │ │ - mousemove: this.onmousemove, │ │ │ │ │ - mouseup: this.onmouseup, │ │ │ │ │ - click: this.onclick, │ │ │ │ │ - mouseout: this.onmouseout, │ │ │ │ │ - dblclick: this.ondblclick, │ │ │ │ │ - touchstart: onTouchstart, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - onmousedown: function(evt) { │ │ │ │ │ - this.mousedown = true; │ │ │ │ │ - OpenLayers.Event.stop(evt, true) │ │ │ │ │ + return resp │ │ │ │ │ }, │ │ │ │ │ - onmousemove: function(evt) { │ │ │ │ │ - if (this.mousedown) { │ │ │ │ │ - OpenLayers.Event.stop(evt, true) │ │ │ │ │ - } │ │ │ │ │ + handleRead: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options) │ │ │ │ │ }, │ │ │ │ │ - onmouseup: function(evt) { │ │ │ │ │ - if (this.mousedown) { │ │ │ │ │ - this.mousedown = false; │ │ │ │ │ - OpenLayers.Event.stop(evt, true) │ │ │ │ │ - } │ │ │ │ │ + create: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: features, │ │ │ │ │ + requestType: "create" │ │ │ │ │ + }); │ │ │ │ │ + resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleCreate, resp, options), │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(features) │ │ │ │ │ + }); │ │ │ │ │ + return resp │ │ │ │ │ }, │ │ │ │ │ - onclick: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt, true) │ │ │ │ │ + handleCreate: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options) │ │ │ │ │ }, │ │ │ │ │ - onmouseout: function(evt) { │ │ │ │ │ - this.mousedown = false │ │ │ │ │ + update: function(feature, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var url = options.url || feature.url || this.options.url + "/" + feature.fid; │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: feature, │ │ │ │ │ + requestType: "update" │ │ │ │ │ + }); │ │ │ │ │ + var method = this.updateWithPOST ? "POST" : "PUT"; │ │ │ │ │ + resp.priv = OpenLayers.Request[method]({ │ │ │ │ │ + url: url, │ │ │ │ │ + callback: this.createCallback(this.handleUpdate, resp, options), │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(feature) │ │ │ │ │ + }); │ │ │ │ │ + return resp │ │ │ │ │ }, │ │ │ │ │ - ondblclick: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt, true) │ │ │ │ │ + handleUpdate: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Popup.WIDTH = 200; │ │ │ │ │ -OpenLayers.Popup.HEIGHT = 200; │ │ │ │ │ -OpenLayers.Popup.COLOR = "white"; │ │ │ │ │ -OpenLayers.Popup.OPACITY = 1; │ │ │ │ │ -OpenLayers.Popup.BORDER = "0px"; │ │ │ │ │ -OpenLayers.Popup.Anchored = OpenLayers.Class(OpenLayers.Popup, { │ │ │ │ │ - relativePosition: null, │ │ │ │ │ - keepInMap: true, │ │ │ │ │ - anchor: null, │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, closeBoxCallback) { │ │ │ │ │ - var newArguments = [id, lonlat, contentSize, contentHTML, closeBox, closeBoxCallback]; │ │ │ │ │ - OpenLayers.Popup.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - this.anchor = anchor != null ? anchor : { │ │ │ │ │ - size: new OpenLayers.Size(0, 0), │ │ │ │ │ - offset: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + delete: function(feature, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var url = options.url || feature.url || this.options.url + "/" + feature.fid; │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: feature, │ │ │ │ │ + requestType: "delete" │ │ │ │ │ + }); │ │ │ │ │ + var method = this.deleteWithPOST ? "POST" : "DELETE"; │ │ │ │ │ + var requestOptions = { │ │ │ │ │ + url: url, │ │ │ │ │ + callback: this.createCallback(this.handleDelete, resp, options), │ │ │ │ │ + headers: options.headers │ │ │ │ │ + }; │ │ │ │ │ + if (this.deleteWithPOST) { │ │ │ │ │ + requestOptions.data = this.format.write(feature) │ │ │ │ │ } │ │ │ │ │ + resp.priv = OpenLayers.Request[method](requestOptions); │ │ │ │ │ + return resp │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.anchor = null; │ │ │ │ │ - this.relativePosition = null; │ │ │ │ │ - OpenLayers.Popup.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - show: function() { │ │ │ │ │ - this.updatePosition(); │ │ │ │ │ - OpenLayers.Popup.prototype.show.apply(this, arguments) │ │ │ │ │ + handleDelete: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options) │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - var oldRelativePosition = this.relativePosition; │ │ │ │ │ - this.relativePosition = this.calculateRelativePosition(px); │ │ │ │ │ - OpenLayers.Popup.prototype.moveTo.call(this, this.calculateNewPx(px)); │ │ │ │ │ - if (this.relativePosition != oldRelativePosition) { │ │ │ │ │ - this.updateRelativePosition() │ │ │ │ │ + handleResponse: function(resp, options) { │ │ │ │ │ + var request = resp.priv; │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + if (resp.requestType != "delete") { │ │ │ │ │ + resp.features = this.parseFeatures(request) │ │ │ │ │ + } │ │ │ │ │ + resp.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ + } else { │ │ │ │ │ + resp.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ + } │ │ │ │ │ + options.callback.call(options.scope, resp) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - setSize: function(contentSize) { │ │ │ │ │ - OpenLayers.Popup.prototype.setSize.apply(this, arguments); │ │ │ │ │ - if (this.lonlat && this.map) { │ │ │ │ │ - var px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ - this.moveTo(px) │ │ │ │ │ + parseFeatures: function(request) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ } │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ + return this.format.read(doc) │ │ │ │ │ }, │ │ │ │ │ - calculateRelativePosition: function(px) { │ │ │ │ │ - var lonlat = this.map.getLonLatFromLayerPx(px); │ │ │ │ │ - var extent = this.map.getExtent(); │ │ │ │ │ - var quadrant = extent.determineQuadrant(lonlat); │ │ │ │ │ - return OpenLayers.Bounds.oppositeQuadrant(quadrant) │ │ │ │ │ - }, │ │ │ │ │ - updateRelativePosition: function() {}, │ │ │ │ │ - calculateNewPx: function(px) { │ │ │ │ │ - var newPx = px.offset(this.anchor.offset); │ │ │ │ │ - var size = this.size || this.contentSize; │ │ │ │ │ - var top = this.relativePosition.charAt(0) == "t"; │ │ │ │ │ - newPx.y += top ? -size.h : this.anchor.size.h; │ │ │ │ │ - var left = this.relativePosition.charAt(1) == "l"; │ │ │ │ │ - newPx.x += left ? -size.w : this.anchor.size.w; │ │ │ │ │ - return newPx │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup.Anchored" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Popup.Framed = OpenLayers.Class(OpenLayers.Popup.Anchored, { │ │ │ │ │ - imageSrc: null, │ │ │ │ │ - imageSize: null, │ │ │ │ │ - isAlphaImage: false, │ │ │ │ │ - positionBlocks: null, │ │ │ │ │ - blocks: null, │ │ │ │ │ - fixedRelativePosition: false, │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, closeBoxCallback) { │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (this.fixedRelativePosition) { │ │ │ │ │ - this.updateRelativePosition(); │ │ │ │ │ - this.calculateRelativePosition = function(px) { │ │ │ │ │ - return this.relativePosition │ │ │ │ │ + commit: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = [], │ │ │ │ │ + nResponses = 0; │ │ │ │ │ + var types = {}; │ │ │ │ │ + types[OpenLayers.State.INSERT] = []; │ │ │ │ │ + types[OpenLayers.State.UPDATE] = []; │ │ │ │ │ + types[OpenLayers.State.DELETE] = []; │ │ │ │ │ + var feature, list, requestFeatures = []; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + list = types[feature.state]; │ │ │ │ │ + if (list) { │ │ │ │ │ + list.push(feature); │ │ │ │ │ + requestFeatures.push(feature) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.contentDiv.style.position = "absolute"; │ │ │ │ │ - this.contentDiv.style.zIndex = 1; │ │ │ │ │ - if (closeBox) { │ │ │ │ │ - this.closeDiv.style.zIndex = 1 │ │ │ │ │ - } │ │ │ │ │ - this.groupDiv.style.position = "absolute"; │ │ │ │ │ - this.groupDiv.style.top = "0px"; │ │ │ │ │ - this.groupDiv.style.left = "0px"; │ │ │ │ │ - this.groupDiv.style.height = "100%"; │ │ │ │ │ - this.groupDiv.style.width = "100%" │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.imageSrc = null; │ │ │ │ │ - this.imageSize = null; │ │ │ │ │ - this.isAlphaImage = null; │ │ │ │ │ - this.fixedRelativePosition = false; │ │ │ │ │ - this.positionBlocks = null; │ │ │ │ │ - for (var i = 0; i < this.blocks.length; i++) { │ │ │ │ │ - var block = this.blocks[i]; │ │ │ │ │ - if (block.image) { │ │ │ │ │ - block.div.removeChild(block.image) │ │ │ │ │ + var nRequests = (types[OpenLayers.State.INSERT].length > 0 ? 1 : 0) + types[OpenLayers.State.UPDATE].length + types[OpenLayers.State.DELETE].length; │ │ │ │ │ + var success = true; │ │ │ │ │ + var finalResponse = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: requestFeatures │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + function insertCallback(response) { │ │ │ │ │ + var len = response.features ? response.features.length : 0; │ │ │ │ │ + var fids = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + fids[i] = response.features[i].fid │ │ │ │ │ } │ │ │ │ │ - block.image = null; │ │ │ │ │ - if (block.div) { │ │ │ │ │ - this.groupDiv.removeChild(block.div) │ │ │ │ │ + finalResponse.insertIds = fids; │ │ │ │ │ + callback.apply(this, [response]) │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function callback(response) { │ │ │ │ │ + this.callUserCallback(response, options); │ │ │ │ │ + success = success && response.success(); │ │ │ │ │ + nResponses++; │ │ │ │ │ + if (nResponses >= nRequests) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + finalResponse.code = success ? OpenLayers.Protocol.Response.SUCCESS : OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + options.callback.apply(options.scope, [finalResponse]) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - block.div = null │ │ │ │ │ } │ │ │ │ │ - this.blocks = null; │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - setBackgroundColor: function(color) {}, │ │ │ │ │ - setBorder: function() {}, │ │ │ │ │ - setOpacity: function(opacity) {}, │ │ │ │ │ - setSize: function(contentSize) { │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.setSize.apply(this, arguments); │ │ │ │ │ - this.updateBlocks() │ │ │ │ │ - }, │ │ │ │ │ - updateRelativePosition: function() { │ │ │ │ │ - this.padding = this.positionBlocks[this.relativePosition].padding; │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ - this.closeDiv.style.right = contentDivPadding.right + this.padding.right + "px"; │ │ │ │ │ - this.closeDiv.style.top = contentDivPadding.top + this.padding.top + "px" │ │ │ │ │ + var queue = types[OpenLayers.State.INSERT]; │ │ │ │ │ + if (queue.length > 0) { │ │ │ │ │ + resp.push(this.create(queue, OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: insertCallback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options.create))) │ │ │ │ │ } │ │ │ │ │ - this.updateBlocks() │ │ │ │ │ - }, │ │ │ │ │ - calculateNewPx: function(px) { │ │ │ │ │ - var newPx = OpenLayers.Popup.Anchored.prototype.calculateNewPx.apply(this, arguments); │ │ │ │ │ - newPx = newPx.offset(this.positionBlocks[this.relativePosition].offset); │ │ │ │ │ - return newPx │ │ │ │ │ - }, │ │ │ │ │ - createBlocks: function() { │ │ │ │ │ - this.blocks = []; │ │ │ │ │ - var firstPosition = null; │ │ │ │ │ - for (var key in this.positionBlocks) { │ │ │ │ │ - firstPosition = key; │ │ │ │ │ - break │ │ │ │ │ + queue = types[OpenLayers.State.UPDATE]; │ │ │ │ │ + for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ + resp.push(this.update(queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: callback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options.update))) │ │ │ │ │ } │ │ │ │ │ - var position = this.positionBlocks[firstPosition]; │ │ │ │ │ - for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ - var block = {}; │ │ │ │ │ - this.blocks.push(block); │ │ │ │ │ - var divId = this.id + "_FrameDecorationDiv_" + i; │ │ │ │ │ - block.div = OpenLayers.Util.createDiv(divId, null, null, null, "absolute", null, "hidden", null); │ │ │ │ │ - var imgId = this.id + "_FrameDecorationImg_" + i; │ │ │ │ │ - var imageCreator = this.isAlphaImage ? OpenLayers.Util.createAlphaImageDiv : OpenLayers.Util.createImage; │ │ │ │ │ - block.image = imageCreator(imgId, null, this.imageSize, this.imageSrc, "absolute", null, null, null); │ │ │ │ │ - block.div.appendChild(block.image); │ │ │ │ │ - this.groupDiv.appendChild(block.div) │ │ │ │ │ + queue = types[OpenLayers.State.DELETE]; │ │ │ │ │ + for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ + resp.push(this["delete"](queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: callback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options["delete"]))) │ │ │ │ │ } │ │ │ │ │ + return resp │ │ │ │ │ }, │ │ │ │ │ - updateBlocks: function() { │ │ │ │ │ - if (!this.blocks) { │ │ │ │ │ - this.createBlocks() │ │ │ │ │ - } │ │ │ │ │ - if (this.size && this.relativePosition) { │ │ │ │ │ - var position = this.positionBlocks[this.relativePosition]; │ │ │ │ │ - for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ - var positionBlock = position.blocks[i]; │ │ │ │ │ - var block = this.blocks[i]; │ │ │ │ │ - var l = positionBlock.anchor.left; │ │ │ │ │ - var b = positionBlock.anchor.bottom; │ │ │ │ │ - var r = positionBlock.anchor.right; │ │ │ │ │ - var t = positionBlock.anchor.top; │ │ │ │ │ - var w = isNaN(positionBlock.size.w) ? this.size.w - (r + l) : positionBlock.size.w; │ │ │ │ │ - var h = isNaN(positionBlock.size.h) ? this.size.h - (b + t) : positionBlock.size.h; │ │ │ │ │ - block.div.style.width = (w < 0 ? 0 : w) + "px"; │ │ │ │ │ - block.div.style.height = (h < 0 ? 0 : h) + "px"; │ │ │ │ │ - block.div.style.left = l != null ? l + "px" : ""; │ │ │ │ │ - block.div.style.bottom = b != null ? b + "px" : ""; │ │ │ │ │ - block.div.style.right = r != null ? r + "px" : ""; │ │ │ │ │ - block.div.style.top = t != null ? t + "px" : ""; │ │ │ │ │ - block.image.style.left = positionBlock.position.x + "px"; │ │ │ │ │ - block.image.style.top = positionBlock.position.y + "px" │ │ │ │ │ - } │ │ │ │ │ - this.contentDiv.style.left = this.padding.left + "px"; │ │ │ │ │ - this.contentDiv.style.top = this.padding.top + "px" │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + response.priv.abort() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup.Framed" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Popup.FramedCloud = OpenLayers.Class(OpenLayers.Popup.Framed, { │ │ │ │ │ - contentDisplayClass: "olFramedCloudPopupContent", │ │ │ │ │ - autoSize: true, │ │ │ │ │ - panMapIfOutOfView: true, │ │ │ │ │ - imageSize: new OpenLayers.Size(1276, 736), │ │ │ │ │ - isAlphaImage: false, │ │ │ │ │ - fixedRelativePosition: false, │ │ │ │ │ - positionBlocks: { │ │ │ │ │ - tl: { │ │ │ │ │ - offset: new OpenLayers.Pixel(44, 0), │ │ │ │ │ - padding: new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ - blocks: [{ │ │ │ │ │ - size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size("auto", 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, 18), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -632) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(81, 35), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -688) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - tr: { │ │ │ │ │ - offset: new OpenLayers.Pixel(-45, 0), │ │ │ │ │ - padding: new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ - blocks: [{ │ │ │ │ │ - size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size("auto", 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -631) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(81, 35), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, null, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-215, -687) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - bl: { │ │ │ │ │ - offset: new OpenLayers.Pixel(45, 0), │ │ │ │ │ - padding: new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ - blocks: [{ │ │ │ │ │ - size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size("auto", 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(81, 33), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, null, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-101, -674) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - br: { │ │ │ │ │ - offset: new OpenLayers.Pixel(-44, 0), │ │ │ │ │ - padding: new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ - blocks: [{ │ │ │ │ │ - size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size("auto", 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(81, 33), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, null, null, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-311, -674) │ │ │ │ │ - }] │ │ │ │ │ + callUserCallback: function(resp, options) { │ │ │ │ │ + var opt = options[resp.requestType]; │ │ │ │ │ + if (opt && opt.callback) { │ │ │ │ │ + opt.callback.call(opt.scope, resp) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - minSize: new OpenLayers.Size(105, 10), │ │ │ │ │ - maxSize: new OpenLayers.Size(1200, 660), │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, closeBoxCallback) { │ │ │ │ │ - this.imageSrc = OpenLayers.Util.getImageLocation("cloud-popup-relative.png"); │ │ │ │ │ - OpenLayers.Popup.Framed.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.contentDiv.className = this.contentDisplayClass │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup.FramedCloud" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.HTTP" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Format = OpenLayers.Class({ │ │ │ │ │ options: null, │ │ │ │ │ externalProjection: null, │ │ │ │ │ internalProjection: null, │ │ │ │ │ data: null, │ │ │ │ │ keepData: false, │ │ │ │ │ @@ -10274,284 +8478,187 @@ │ │ │ │ │ array[i] = this.extract.geometry.apply(this, [collection.components[i]]) │ │ │ │ │ } │ │ │ │ │ return array │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.GeoJSON" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Strategy = OpenLayers.Class({ │ │ │ │ │ - layer: null, │ │ │ │ │ - options: null, │ │ │ │ │ - active: null, │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - autoDestroy: true, │ │ │ │ │ +OpenLayers.Filter = OpenLayers.Class({ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.options = options; │ │ │ │ │ - this.active = false │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.layer = null; │ │ │ │ │ - this.options = null │ │ │ │ │ + OpenLayers.Util.extend(this, options) │ │ │ │ │ }, │ │ │ │ │ - setLayer: function(layer) { │ │ │ │ │ - this.layer = layer │ │ │ │ │ + destroy: function() {}, │ │ │ │ │ + evaluate: function(context) { │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - this.active = true; │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - return false │ │ │ │ │ + clone: function() { │ │ │ │ │ + return null │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.active = false; │ │ │ │ │ - return true │ │ │ │ │ + toString: function() { │ │ │ │ │ + var string; │ │ │ │ │ + if (OpenLayers.Format && OpenLayers.Format.CQL) { │ │ │ │ │ + string = OpenLayers.Format.CQL.prototype.write(this) │ │ │ │ │ + } else { │ │ │ │ │ + string = Object.prototype.toString.call(this) │ │ │ │ │ } │ │ │ │ │ - return false │ │ │ │ │ + return string │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Filter" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Strategy.Fixed = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - preload: false, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - refresh: this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - if (this.layer.visibility == true || this.preload) { │ │ │ │ │ - this.load() │ │ │ │ │ - } else { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - visibilitychanged: this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - refresh: this.load, │ │ │ │ │ - visibilitychanged: this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - load: function(options) { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - layer.events.triggerEvent("loadstart", { │ │ │ │ │ - filter: layer.filter │ │ │ │ │ - }); │ │ │ │ │ - layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: this.merge, │ │ │ │ │ - filter: layer.filter, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options)); │ │ │ │ │ - layer.events.un({ │ │ │ │ │ - visibilitychanged: this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ +OpenLayers.Filter.Logical = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ + filters: null, │ │ │ │ │ + type: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.filters = []; │ │ │ │ │ + OpenLayers.Filter.prototype.initialize.apply(this, [options]) │ │ │ │ │ }, │ │ │ │ │ - merge: function(resp) { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - layer.destroyFeatures(); │ │ │ │ │ - var features = resp.features; │ │ │ │ │ - if (features && features.length > 0) { │ │ │ │ │ - var remote = layer.projection; │ │ │ │ │ - var local = layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var geom; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - geom = features[i].geometry; │ │ │ │ │ - if (geom) { │ │ │ │ │ - geom.transform(remote, local) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - layer.addFeatures(features) │ │ │ │ │ - } │ │ │ │ │ - layer.events.triggerEvent("loadend", { │ │ │ │ │ - response: resp │ │ │ │ │ - }) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.filters = null; │ │ │ │ │ + OpenLayers.Filter.prototype.destroy.apply(this) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Fixed" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Filter.Spatial = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ - type: null, │ │ │ │ │ - property: null, │ │ │ │ │ - value: null, │ │ │ │ │ - distance: null, │ │ │ │ │ - distanceUnits: null, │ │ │ │ │ - evaluate: function(feature) { │ │ │ │ │ - var intersect = false; │ │ │ │ │ + evaluate: function(context) { │ │ │ │ │ + var i, len; │ │ │ │ │ switch (this.type) { │ │ │ │ │ - case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ - case OpenLayers.Filter.Spatial.INTERSECTS: │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - var geom = this.value; │ │ │ │ │ - if (this.value.CLASS_NAME == "OpenLayers.Bounds") { │ │ │ │ │ - geom = this.value.toGeometry() │ │ │ │ │ + case OpenLayers.Filter.Logical.AND: │ │ │ │ │ + for (i = 0, len = this.filters.length; i < len; i++) { │ │ │ │ │ + if (this.filters[i].evaluate(context) == false) { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - if (feature.geometry.intersects(geom)) { │ │ │ │ │ - intersect = true │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ + case OpenLayers.Filter.Logical.OR: │ │ │ │ │ + for (i = 0, len = this.filters.length; i < len; i++) { │ │ │ │ │ + if (this.filters[i].evaluate(context) == true) { │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - throw new Error("evaluate is not implemented for this filter type.") │ │ │ │ │ + return false; │ │ │ │ │ + case OpenLayers.Filter.Logical.NOT: │ │ │ │ │ + return !this.filters[0].evaluate(context) │ │ │ │ │ } │ │ │ │ │ - return intersect │ │ │ │ │ + return undefined │ │ │ │ │ }, │ │ │ │ │ clone: function() { │ │ │ │ │ - var options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - value: this.value && this.value.clone && this.value.clone() │ │ │ │ │ - }, this); │ │ │ │ │ - return new OpenLayers.Filter.Spatial(options) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Filter.Spatial" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Filter.Spatial.BBOX = "BBOX"; │ │ │ │ │ -OpenLayers.Filter.Spatial.INTERSECTS = "INTERSECTS"; │ │ │ │ │ -OpenLayers.Filter.Spatial.DWITHIN = "DWITHIN"; │ │ │ │ │ -OpenLayers.Filter.Spatial.WITHIN = "WITHIN"; │ │ │ │ │ -OpenLayers.Filter.Spatial.CONTAINS = "CONTAINS"; │ │ │ │ │ -OpenLayers.Strategy.BBOX = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - bounds: null, │ │ │ │ │ - resolution: null, │ │ │ │ │ - ratio: 2, │ │ │ │ │ - resFactor: null, │ │ │ │ │ - response: null, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - moveend: this.update, │ │ │ │ │ - refresh: this.update, │ │ │ │ │ - visibilitychanged: this.update, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.update() │ │ │ │ │ - } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - moveend: this.update, │ │ │ │ │ - refresh: this.update, │ │ │ │ │ - visibilitychanged: this.update, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - update: function(options) { │ │ │ │ │ - var mapBounds = this.getMapBounds(); │ │ │ │ │ - if (mapBounds !== null && (options && options.force || this.layer.visibility && this.layer.calculateInRange() && this.invalidBounds(mapBounds))) { │ │ │ │ │ - this.calculateBounds(mapBounds); │ │ │ │ │ - this.resolution = this.layer.map.getResolution(); │ │ │ │ │ - this.triggerRead(options) │ │ │ │ │ + var filters = []; │ │ │ │ │ + for (var i = 0, len = this.filters.length; i < len; ++i) { │ │ │ │ │ + filters.push(this.filters[i].clone()) │ │ │ │ │ } │ │ │ │ │ + return new OpenLayers.Filter.Logical({ │ │ │ │ │ + type: this.type, │ │ │ │ │ + filters: filters │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - getMapBounds: function() { │ │ │ │ │ - if (this.layer.map === null) { │ │ │ │ │ - return null │ │ │ │ │ - } │ │ │ │ │ - var bounds = this.layer.map.getExtent(); │ │ │ │ │ - if (bounds && !this.layer.projection.equals(this.layer.map.getProjectionObject())) { │ │ │ │ │ - bounds = bounds.clone().transform(this.layer.map.getProjectionObject(), this.layer.projection) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Filter.Logical" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Filter.Logical.AND = "&&"; │ │ │ │ │ +OpenLayers.Filter.Logical.OR = "||"; │ │ │ │ │ +OpenLayers.Filter.Logical.NOT = "!"; │ │ │ │ │ +OpenLayers.Filter.Comparison = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ + type: null, │ │ │ │ │ + property: null, │ │ │ │ │ + value: null, │ │ │ │ │ + matchCase: true, │ │ │ │ │ + lowerBoundary: null, │ │ │ │ │ + upperBoundary: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Filter.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (this.type === OpenLayers.Filter.Comparison.LIKE && options.matchCase === undefined) { │ │ │ │ │ + this.matchCase = null │ │ │ │ │ } │ │ │ │ │ - return bounds │ │ │ │ │ }, │ │ │ │ │ - invalidBounds: function(mapBounds) { │ │ │ │ │ - if (!mapBounds) { │ │ │ │ │ - mapBounds = this.getMapBounds() │ │ │ │ │ - } │ │ │ │ │ - var invalid = !this.bounds || !this.bounds.containsBounds(mapBounds); │ │ │ │ │ - if (!invalid && this.resFactor) { │ │ │ │ │ - var ratio = this.resolution / this.layer.map.getResolution(); │ │ │ │ │ - invalid = ratio >= this.resFactor || ratio <= 1 / this.resFactor │ │ │ │ │ + evaluate: function(context) { │ │ │ │ │ + if (context instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ + context = context.attributes │ │ │ │ │ } │ │ │ │ │ - return invalid │ │ │ │ │ - }, │ │ │ │ │ - calculateBounds: function(mapBounds) { │ │ │ │ │ - if (!mapBounds) { │ │ │ │ │ - mapBounds = this.getMapBounds() │ │ │ │ │ + var result = false; │ │ │ │ │ + var got = context[this.property]; │ │ │ │ │ + var exp; │ │ │ │ │ + switch (this.type) { │ │ │ │ │ + case OpenLayers.Filter.Comparison.EQUAL_TO: │ │ │ │ │ + exp = this.value; │ │ │ │ │ + if (!this.matchCase && typeof got == "string" && typeof exp == "string") { │ │ │ │ │ + result = got.toUpperCase() == exp.toUpperCase() │ │ │ │ │ + } else { │ │ │ │ │ + result = got == exp │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Comparison.NOT_EQUAL_TO: │ │ │ │ │ + exp = this.value; │ │ │ │ │ + if (!this.matchCase && typeof got == "string" && typeof exp == "string") { │ │ │ │ │ + result = got.toUpperCase() != exp.toUpperCase() │ │ │ │ │ + } else { │ │ │ │ │ + result = got != exp │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Comparison.LESS_THAN: │ │ │ │ │ + result = got < this.value; │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Comparison.GREATER_THAN: │ │ │ │ │ + result = got > this.value; │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO: │ │ │ │ │ + result = got <= this.value; │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO: │ │ │ │ │ + result = got >= this.value; │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Comparison.BETWEEN: │ │ │ │ │ + result = got >= this.lowerBoundary && got <= this.upperBoundary; │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Comparison.LIKE: │ │ │ │ │ + var regexp = new RegExp(this.value, "gi"); │ │ │ │ │ + result = regexp.test(got); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Comparison.IS_NULL: │ │ │ │ │ + result = got === null; │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - var center = mapBounds.getCenterLonLat(); │ │ │ │ │ - var dataWidth = mapBounds.getWidth() * this.ratio; │ │ │ │ │ - var dataHeight = mapBounds.getHeight() * this.ratio; │ │ │ │ │ - this.bounds = new OpenLayers.Bounds(center.lon - dataWidth / 2, center.lat - dataHeight / 2, center.lon + dataWidth / 2, center.lat + dataHeight / 2) │ │ │ │ │ + return result │ │ │ │ │ }, │ │ │ │ │ - triggerRead: function(options) { │ │ │ │ │ - if (this.response && !(options && options.noAbort === true)) { │ │ │ │ │ - this.layer.protocol.abort(this.response); │ │ │ │ │ - this.layer.events.triggerEvent("loadend") │ │ │ │ │ + value2regex: function(wildCard, singleChar, escapeChar) { │ │ │ │ │ + if (wildCard == ".") { │ │ │ │ │ + throw new Error("'.' is an unsupported wildCard character for " + "OpenLayers.Filter.Comparison") │ │ │ │ │ } │ │ │ │ │ - var evt = { │ │ │ │ │ - filter: this.createFilter() │ │ │ │ │ - }; │ │ │ │ │ - this.layer.events.triggerEvent("loadstart", evt); │ │ │ │ │ - this.response = this.layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ - filter: evt.filter, │ │ │ │ │ - callback: this.merge, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options)) │ │ │ │ │ + wildCard = wildCard ? wildCard : "*"; │ │ │ │ │ + singleChar = singleChar ? singleChar : "."; │ │ │ │ │ + escapeChar = escapeChar ? escapeChar : "!"; │ │ │ │ │ + this.value = this.value.replace(new RegExp("\\" + escapeChar + "(.|$)", "g"), "\\$1"); │ │ │ │ │ + this.value = this.value.replace(new RegExp("\\" + singleChar, "g"), "."); │ │ │ │ │ + this.value = this.value.replace(new RegExp("\\" + wildCard, "g"), ".*"); │ │ │ │ │ + this.value = this.value.replace(new RegExp("\\\\.\\*", "g"), "\\" + wildCard); │ │ │ │ │ + this.value = this.value.replace(new RegExp("\\\\\\.", "g"), "\\" + singleChar); │ │ │ │ │ + return this.value │ │ │ │ │ }, │ │ │ │ │ - createFilter: function() { │ │ │ │ │ - var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ - value: this.bounds, │ │ │ │ │ - projection: this.layer.projection │ │ │ │ │ + regex2value: function() { │ │ │ │ │ + var value = this.value; │ │ │ │ │ + value = value.replace(/!/g, "!!"); │ │ │ │ │ + value = value.replace(/(\\)?\\\./g, function($0, $1) { │ │ │ │ │ + return $1 ? $0 : "!." │ │ │ │ │ }); │ │ │ │ │ - if (this.layer.filter) { │ │ │ │ │ - filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ - type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ - filters: [this.layer.filter, filter] │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return filter │ │ │ │ │ + value = value.replace(/(\\)?\\\*/g, function($0, $1) { │ │ │ │ │ + return $1 ? $0 : "!*" │ │ │ │ │ + }); │ │ │ │ │ + value = value.replace(/\\\\/g, "\\"); │ │ │ │ │ + value = value.replace(/\.\*/g, "*"); │ │ │ │ │ + return value │ │ │ │ │ }, │ │ │ │ │ - merge: function(resp) { │ │ │ │ │ - this.layer.destroyFeatures(); │ │ │ │ │ - if (resp.success()) { │ │ │ │ │ - var features = resp.features; │ │ │ │ │ - if (features && features.length > 0) { │ │ │ │ │ - var remote = this.layer.projection; │ │ │ │ │ - var local = this.layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var geom; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - geom = features[i].geometry; │ │ │ │ │ - if (geom) { │ │ │ │ │ - geom.transform(remote, local) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.layer.addFeatures(features) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.bounds = null │ │ │ │ │ - } │ │ │ │ │ - this.response = null; │ │ │ │ │ - this.layer.events.triggerEvent("loadend", { │ │ │ │ │ - response: resp │ │ │ │ │ - }) │ │ │ │ │ + clone: function() { │ │ │ │ │ + return OpenLayers.Util.extend(new OpenLayers.Filter.Comparison, this) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.BBOX" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Filter.Comparison" │ │ │ │ │ }); │ │ │ │ │ +OpenLayers.Filter.Comparison.EQUAL_TO = "=="; │ │ │ │ │ +OpenLayers.Filter.Comparison.NOT_EQUAL_TO = "!="; │ │ │ │ │ +OpenLayers.Filter.Comparison.LESS_THAN = "<"; │ │ │ │ │ +OpenLayers.Filter.Comparison.GREATER_THAN = ">"; │ │ │ │ │ +OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO = "<="; │ │ │ │ │ +OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO = ">="; │ │ │ │ │ +OpenLayers.Filter.Comparison.BETWEEN = ".."; │ │ │ │ │ +OpenLayers.Filter.Comparison.LIKE = "~"; │ │ │ │ │ +OpenLayers.Filter.Comparison.IS_NULL = "NULL"; │ │ │ │ │ OpenLayers.Control = OpenLayers.Class({ │ │ │ │ │ id: null, │ │ │ │ │ map: null, │ │ │ │ │ div: null, │ │ │ │ │ type: null, │ │ │ │ │ allowSelection: false, │ │ │ │ │ displayClass: "", │ │ │ │ │ @@ -10660,14 +8767,597 @@ │ │ │ │ │ return false │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Control.TYPE_BUTTON = 1; │ │ │ │ │ OpenLayers.Control.TYPE_TOGGLE = 2; │ │ │ │ │ OpenLayers.Control.TYPE_TOOL = 3; │ │ │ │ │ +OpenLayers.Events.buttonclick = OpenLayers.Class({ │ │ │ │ │ + target: null, │ │ │ │ │ + events: ["mousedown", "mouseup", "click", "dblclick", "touchstart", "touchmove", "touchend", "keydown"], │ │ │ │ │ + startRegEx: /^mousedown|touchstart$/, │ │ │ │ │ + cancelRegEx: /^touchmove$/, │ │ │ │ │ + completeRegEx: /^mouseup|touchend$/, │ │ │ │ │ + initialize: function(target) { │ │ │ │ │ + this.target = target; │ │ │ │ │ + for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ + this.target.register(this.events[i], this, this.buttonClick, { │ │ │ │ │ + extension: true │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ + this.target.unregister(this.events[i], this, this.buttonClick) │ │ │ │ │ + } │ │ │ │ │ + delete this.target │ │ │ │ │ + }, │ │ │ │ │ + getPressedButton: function(element) { │ │ │ │ │ + var depth = 3, │ │ │ │ │ + button; │ │ │ │ │ + do { │ │ │ │ │ + if (OpenLayers.Element.hasClass(element, "olButton")) { │ │ │ │ │ + button = element; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + element = element.parentNode │ │ │ │ │ + } while (--depth > 0 && element); │ │ │ │ │ + return button │ │ │ │ │ + }, │ │ │ │ │ + ignore: function(element) { │ │ │ │ │ + var depth = 3, │ │ │ │ │ + ignore = false; │ │ │ │ │ + do { │ │ │ │ │ + if (element.nodeName.toLowerCase() === "a") { │ │ │ │ │ + ignore = true; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + element = element.parentNode │ │ │ │ │ + } while (--depth > 0 && element); │ │ │ │ │ + return ignore │ │ │ │ │ + }, │ │ │ │ │ + buttonClick: function(evt) { │ │ │ │ │ + var propagate = true, │ │ │ │ │ + element = OpenLayers.Event.element(evt); │ │ │ │ │ + if (element && (OpenLayers.Event.isLeftClick(evt) || !~evt.type.indexOf("mouse"))) { │ │ │ │ │ + var button = this.getPressedButton(element); │ │ │ │ │ + if (button) { │ │ │ │ │ + if (evt.type === "keydown") { │ │ │ │ │ + switch (evt.keyCode) { │ │ │ │ │ + case OpenLayers.Event.KEY_RETURN: │ │ │ │ │ + case OpenLayers.Event.KEY_SPACE: │ │ │ │ │ + this.target.triggerEvent("buttonclick", { │ │ │ │ │ + buttonElement: button │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } else if (this.startEvt) { │ │ │ │ │ + if (this.completeRegEx.test(evt.type)) { │ │ │ │ │ + var pos = OpenLayers.Util.pagePosition(button); │ │ │ │ │ + var viewportElement = OpenLayers.Util.getViewportElement(); │ │ │ │ │ + var scrollTop = window.pageYOffset || viewportElement.scrollTop; │ │ │ │ │ + var scrollLeft = window.pageXOffset || viewportElement.scrollLeft; │ │ │ │ │ + pos[0] = pos[0] - scrollLeft; │ │ │ │ │ + pos[1] = pos[1] - scrollTop; │ │ │ │ │ + this.target.triggerEvent("buttonclick", { │ │ │ │ │ + buttonElement: button, │ │ │ │ │ + buttonXY: { │ │ │ │ │ + x: this.startEvt.clientX - pos[0], │ │ │ │ │ + y: this.startEvt.clientY - pos[1] │ │ │ │ │ + } │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + if (this.cancelRegEx.test(evt.type)) { │ │ │ │ │ + delete this.startEvt │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false │ │ │ │ │ + } │ │ │ │ │ + if (this.startRegEx.test(evt.type)) { │ │ │ │ │ + this.startEvt = evt; │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + propagate = !this.ignore(OpenLayers.Event.element(evt)); │ │ │ │ │ + delete this.startEvt │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return propagate │ │ │ │ │ + } │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Zoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + zoomInText: "+", │ │ │ │ │ + zoomInId: "olZoomInLink", │ │ │ │ │ + zoomOutText: "−", │ │ │ │ │ + zoomOutId: "olZoomOutLink", │ │ │ │ │ + draw: function() { │ │ │ │ │ + var div = OpenLayers.Control.prototype.draw.apply(this), │ │ │ │ │ + links = this.getOrCreateLinks(div), │ │ │ │ │ + zoomIn = links.zoomIn, │ │ │ │ │ + zoomOut = links.zoomOut, │ │ │ │ │ + eventsInstance = this.map.events; │ │ │ │ │ + if (zoomOut.parentNode !== div) { │ │ │ │ │ + eventsInstance = this.events; │ │ │ │ │ + eventsInstance.attachToElement(zoomOut.parentNode) │ │ │ │ │ + } │ │ │ │ │ + eventsInstance.register("buttonclick", this, this.onZoomClick); │ │ │ │ │ + this.zoomInLink = zoomIn; │ │ │ │ │ + this.zoomOutLink = zoomOut; │ │ │ │ │ + return div │ │ │ │ │ + }, │ │ │ │ │ + getOrCreateLinks: function(el) { │ │ │ │ │ + var zoomIn = document.getElementById(this.zoomInId), │ │ │ │ │ + zoomOut = document.getElementById(this.zoomOutId); │ │ │ │ │ + if (!zoomIn) { │ │ │ │ │ + zoomIn = document.createElement("a"); │ │ │ │ │ + zoomIn.href = "#zoomIn"; │ │ │ │ │ + zoomIn.appendChild(document.createTextNode(this.zoomInText)); │ │ │ │ │ + zoomIn.className = "olControlZoomIn"; │ │ │ │ │ + el.appendChild(zoomIn) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.addClass(zoomIn, "olButton"); │ │ │ │ │ + if (!zoomOut) { │ │ │ │ │ + zoomOut = document.createElement("a"); │ │ │ │ │ + zoomOut.href = "#zoomOut"; │ │ │ │ │ + zoomOut.appendChild(document.createTextNode(this.zoomOutText)); │ │ │ │ │ + zoomOut.className = "olControlZoomOut"; │ │ │ │ │ + el.appendChild(zoomOut) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.addClass(zoomOut, "olButton"); │ │ │ │ │ + return { │ │ │ │ │ + zoomIn: zoomIn, │ │ │ │ │ + zoomOut: zoomOut │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + onZoomClick: function(evt) { │ │ │ │ │ + var button = evt.buttonElement; │ │ │ │ │ + if (button === this.zoomInLink) { │ │ │ │ │ + this.map.zoomIn() │ │ │ │ │ + } else if (button === this.zoomOutLink) { │ │ │ │ │ + this.map.zoomOut() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onZoomClick) │ │ │ │ │ + } │ │ │ │ │ + delete this.zoomInLink; │ │ │ │ │ + delete this.zoomOutLink; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Zoom" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Panel = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + controls: null, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + defaultControl: null, │ │ │ │ │ + saveState: false, │ │ │ │ │ + allowDepress: false, │ │ │ │ │ + activeState: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.controls = []; │ │ │ │ │ + this.activeState = {} │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onButtonClick) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + for (var ctl, i = this.controls.length - 1; i >= 0; i--) { │ │ │ │ │ + ctl = this.controls[i]; │ │ │ │ │ + if (ctl.events) { │ │ │ │ │ + ctl.events.un({ │ │ │ │ │ + activate: this.iconOn, │ │ │ │ │ + deactivate: this.iconOff │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + ctl.panel_div = null │ │ │ │ │ + } │ │ │ │ │ + this.activeState = null │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + control = this.controls[i]; │ │ │ │ │ + if (control === this.defaultControl || this.saveState && this.activeState[control.id]) { │ │ │ │ │ + control.activate() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.saveState === true) { │ │ │ │ │ + this.defaultControl = null │ │ │ │ │ + } │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + control = this.controls[i]; │ │ │ │ │ + this.activeState[control.id] = control.deactivate() │ │ │ │ │ + } │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (this.outsideViewport) { │ │ │ │ │ + this.events.attachToElement(this.div); │ │ │ │ │ + this.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ + } else { │ │ │ │ │ + this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ + } │ │ │ │ │ + this.addControlsToMap(this.controls); │ │ │ │ │ + return this.div │ │ │ │ │ + }, │ │ │ │ │ + redraw: function() { │ │ │ │ │ + for (var l = this.div.childNodes.length, i = l - 1; i >= 0; i--) { │ │ │ │ │ + this.div.removeChild(this.div.childNodes[i]) │ │ │ │ │ + } │ │ │ │ │ + this.div.innerHTML = ""; │ │ │ │ │ + if (this.active) { │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + this.div.appendChild(this.controls[i].panel_div) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + activateControl: function(control) { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + if (control.type == OpenLayers.Control.TYPE_BUTTON) { │ │ │ │ │ + control.trigger(); │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (control.type == OpenLayers.Control.TYPE_TOGGLE) { │ │ │ │ │ + if (control.active) { │ │ │ │ │ + control.deactivate() │ │ │ │ │ + } else { │ │ │ │ │ + control.activate() │ │ │ │ │ + } │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (this.allowDepress && control.active) { │ │ │ │ │ + control.deactivate() │ │ │ │ │ + } else { │ │ │ │ │ + var c; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + c = this.controls[i]; │ │ │ │ │ + if (c != control && (c.type === OpenLayers.Control.TYPE_TOOL || c.type == null)) { │ │ │ │ │ + c.deactivate() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + control.activate() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addControls: function(controls) { │ │ │ │ │ + if (!OpenLayers.Util.isArray(controls)) { │ │ │ │ │ + controls = [controls] │ │ │ │ │ + } │ │ │ │ │ + this.controls = this.controls.concat(controls); │ │ │ │ │ + for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ + var control = controls[i], │ │ │ │ │ + element = this.createControlMarkup(control); │ │ │ │ │ + OpenLayers.Element.addClass(element, control.displayClass + "ItemInactive"); │ │ │ │ │ + OpenLayers.Element.addClass(element, "olButton"); │ │ │ │ │ + if (control.title != "" && !element.title) { │ │ │ │ │ + element.title = control.title │ │ │ │ │ + } │ │ │ │ │ + control.panel_div = element │ │ │ │ │ + } │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.addControlsToMap(controls); │ │ │ │ │ + this.redraw() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + createControlMarkup: function(control) { │ │ │ │ │ + return document.createElement("div") │ │ │ │ │ + }, │ │ │ │ │ + addControlsToMap: function(controls) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ + control = controls[i]; │ │ │ │ │ + if (control.autoActivate === true) { │ │ │ │ │ + control.autoActivate = false; │ │ │ │ │ + this.map.addControl(control); │ │ │ │ │ + control.autoActivate = true │ │ │ │ │ + } else { │ │ │ │ │ + this.map.addControl(control); │ │ │ │ │ + control.deactivate() │ │ │ │ │ + } │ │ │ │ │ + control.events.on({ │ │ │ │ │ + activate: this.iconOn, │ │ │ │ │ + deactivate: this.iconOff │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + iconOn: function() { │ │ │ │ │ + var d = this.panel_div; │ │ │ │ │ + var re = new RegExp("\\b(" + this.displayClass + "Item)Inactive\\b"); │ │ │ │ │ + d.className = d.className.replace(re, "$1Active") │ │ │ │ │ + }, │ │ │ │ │ + iconOff: function() { │ │ │ │ │ + var d = this.panel_div; │ │ │ │ │ + var re = new RegExp("\\b(" + this.displayClass + "Item)Active\\b"); │ │ │ │ │ + d.className = d.className.replace(re, "$1Inactive") │ │ │ │ │ + }, │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + var controls = this.controls, │ │ │ │ │ + button = evt.buttonElement; │ │ │ │ │ + for (var i = controls.length - 1; i >= 0; --i) { │ │ │ │ │ + if (controls[i].panel_div === button) { │ │ │ │ │ + this.activateControl(controls[i]); │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getControlsBy: function(property, match) { │ │ │ │ │ + var test = typeof match.test == "function"; │ │ │ │ │ + var found = OpenLayers.Array.filter(this.controls, function(item) { │ │ │ │ │ + return item[property] == match || test && match.test(item[property]) │ │ │ │ │ + }); │ │ │ │ │ + return found │ │ │ │ │ + }, │ │ │ │ │ + getControlsByName: function(match) { │ │ │ │ │ + return this.getControlsBy("name", match) │ │ │ │ │ + }, │ │ │ │ │ + getControlsByClass: function(match) { │ │ │ │ │ + return this.getControlsBy("CLASS_NAME", match) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Panel" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.LayerSwitcher = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + layerStates: null, │ │ │ │ │ + layersDiv: null, │ │ │ │ │ + baseLayersDiv: null, │ │ │ │ │ + baseLayers: null, │ │ │ │ │ + dataLbl: null, │ │ │ │ │ + dataLayersDiv: null, │ │ │ │ │ + dataLayers: null, │ │ │ │ │ + minimizeDiv: null, │ │ │ │ │ + maximizeDiv: null, │ │ │ │ │ + ascending: true, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.layerStates = [] │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.clearLayersArray("base"); │ │ │ │ │ + this.clearLayersArray("data"); │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + buttonclick: this.onButtonClick, │ │ │ │ │ + addlayer: this.redraw, │ │ │ │ │ + changelayer: this.redraw, │ │ │ │ │ + removelayer: this.redraw, │ │ │ │ │ + changebaselayer: this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + addlayer: this.redraw, │ │ │ │ │ + changelayer: this.redraw, │ │ │ │ │ + removelayer: this.redraw, │ │ │ │ │ + changebaselayer: this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + if (this.outsideViewport) { │ │ │ │ │ + this.events.attachToElement(this.div); │ │ │ │ │ + this.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ + } else { │ │ │ │ │ + this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this); │ │ │ │ │ + this.loadContents(); │ │ │ │ │ + if (!this.outsideViewport) { │ │ │ │ │ + this.minimizeControl() │ │ │ │ │ + } │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return this.div │ │ │ │ │ + }, │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + var button = evt.buttonElement; │ │ │ │ │ + if (button === this.minimizeDiv) { │ │ │ │ │ + this.minimizeControl() │ │ │ │ │ + } else if (button === this.maximizeDiv) { │ │ │ │ │ + this.maximizeControl() │ │ │ │ │ + } else if (button._layerSwitcher === this.id) { │ │ │ │ │ + if (button["for"]) { │ │ │ │ │ + button = document.getElementById(button["for"]) │ │ │ │ │ + } │ │ │ │ │ + if (!button.disabled) { │ │ │ │ │ + if (button.type == "radio") { │ │ │ │ │ + button.checked = true; │ │ │ │ │ + this.map.setBaseLayer(this.map.getLayer(button._layer)) │ │ │ │ │ + } else { │ │ │ │ │ + button.checked = !button.checked; │ │ │ │ │ + this.updateMap() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + clearLayersArray: function(layersType) { │ │ │ │ │ + this[layersType + "LayersDiv"].innerHTML = ""; │ │ │ │ │ + this[layersType + "Layers"] = [] │ │ │ │ │ + }, │ │ │ │ │ + checkRedraw: function() { │ │ │ │ │ + if (!this.layerStates.length || this.map.layers.length != this.layerStates.length) { │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, len = this.layerStates.length; i < len; i++) { │ │ │ │ │ + var layerState = this.layerStates[i]; │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if (layerState.name != layer.name || layerState.inRange != layer.inRange || layerState.id != layer.id || layerState.visibility != layer.visibility) { │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + redraw: function() { │ │ │ │ │ + if (!this.checkRedraw()) { │ │ │ │ │ + return this.div │ │ │ │ │ + } │ │ │ │ │ + this.clearLayersArray("base"); │ │ │ │ │ + this.clearLayersArray("data"); │ │ │ │ │ + var containsOverlays = false; │ │ │ │ │ + var containsBaseLayers = false; │ │ │ │ │ + var len = this.map.layers.length; │ │ │ │ │ + this.layerStates = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + this.layerStates[i] = { │ │ │ │ │ + name: layer.name, │ │ │ │ │ + visibility: layer.visibility, │ │ │ │ │ + inRange: layer.inRange, │ │ │ │ │ + id: layer.id │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var layers = this.map.layers.slice(); │ │ │ │ │ + if (!this.ascending) { │ │ │ │ │ + layers.reverse() │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + var layer = layers[i]; │ │ │ │ │ + var baseLayer = layer.isBaseLayer; │ │ │ │ │ + if (layer.displayInLayerSwitcher) { │ │ │ │ │ + if (baseLayer) { │ │ │ │ │ + containsBaseLayers = true │ │ │ │ │ + } else { │ │ │ │ │ + containsOverlays = true │ │ │ │ │ + } │ │ │ │ │ + var checked = baseLayer ? layer == this.map.baseLayer : layer.getVisibility(); │ │ │ │ │ + var inputElem = document.createElement("input"), │ │ │ │ │ + inputId = OpenLayers.Util.createUniqueID(this.id + "_input_"); │ │ │ │ │ + inputElem.id = inputId; │ │ │ │ │ + inputElem.name = baseLayer ? this.id + "_baseLayers" : layer.name; │ │ │ │ │ + inputElem.type = baseLayer ? "radio" : "checkbox"; │ │ │ │ │ + inputElem.value = layer.name; │ │ │ │ │ + inputElem.checked = checked; │ │ │ │ │ + inputElem.defaultChecked = checked; │ │ │ │ │ + inputElem.className = "olButton"; │ │ │ │ │ + inputElem._layer = layer.id; │ │ │ │ │ + inputElem._layerSwitcher = this.id; │ │ │ │ │ + if (!baseLayer && !layer.inRange) { │ │ │ │ │ + inputElem.disabled = true │ │ │ │ │ + } │ │ │ │ │ + var labelSpan = document.createElement("label"); │ │ │ │ │ + labelSpan["for"] = inputElem.id; │ │ │ │ │ + OpenLayers.Element.addClass(labelSpan, "labelSpan olButton"); │ │ │ │ │ + labelSpan._layer = layer.id; │ │ │ │ │ + labelSpan._layerSwitcher = this.id; │ │ │ │ │ + if (!baseLayer && !layer.inRange) { │ │ │ │ │ + labelSpan.style.color = "gray" │ │ │ │ │ + } │ │ │ │ │ + labelSpan.innerHTML = layer.name; │ │ │ │ │ + labelSpan.style.verticalAlign = baseLayer ? "bottom" : "baseline"; │ │ │ │ │ + var br = document.createElement("br"); │ │ │ │ │ + var groupArray = baseLayer ? this.baseLayers : this.dataLayers; │ │ │ │ │ + groupArray.push({ │ │ │ │ │ + layer: layer, │ │ │ │ │ + inputElem: inputElem, │ │ │ │ │ + labelSpan: labelSpan │ │ │ │ │ + }); │ │ │ │ │ + var groupDiv = baseLayer ? this.baseLayersDiv : this.dataLayersDiv; │ │ │ │ │ + groupDiv.appendChild(inputElem); │ │ │ │ │ + groupDiv.appendChild(labelSpan); │ │ │ │ │ + groupDiv.appendChild(br) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.dataLbl.style.display = containsOverlays ? "" : "none"; │ │ │ │ │ + this.baseLbl.style.display = containsBaseLayers ? "" : "none"; │ │ │ │ │ + return this.div │ │ │ │ │ + }, │ │ │ │ │ + updateMap: function() { │ │ │ │ │ + for (var i = 0, len = this.baseLayers.length; i < len; i++) { │ │ │ │ │ + var layerEntry = this.baseLayers[i]; │ │ │ │ │ + if (layerEntry.inputElem.checked) { │ │ │ │ │ + this.map.setBaseLayer(layerEntry.layer, false) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, len = this.dataLayers.length; i < len; i++) { │ │ │ │ │ + var layerEntry = this.dataLayers[i]; │ │ │ │ │ + layerEntry.layer.setVisibility(layerEntry.inputElem.checked) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + maximizeControl: function(e) { │ │ │ │ │ + this.div.style.width = ""; │ │ │ │ │ + this.div.style.height = ""; │ │ │ │ │ + this.showControls(false); │ │ │ │ │ + if (e != null) { │ │ │ │ │ + OpenLayers.Event.stop(e) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + minimizeControl: function(e) { │ │ │ │ │ + this.div.style.width = "0px"; │ │ │ │ │ + this.div.style.height = "0px"; │ │ │ │ │ + this.showControls(true); │ │ │ │ │ + if (e != null) { │ │ │ │ │ + OpenLayers.Event.stop(e) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + showControls: function(minimize) { │ │ │ │ │ + this.maximizeDiv.style.display = minimize ? "" : "none"; │ │ │ │ │ + this.minimizeDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ + this.layersDiv.style.display = minimize ? "none" : "" │ │ │ │ │ + }, │ │ │ │ │ + loadContents: function() { │ │ │ │ │ + this.layersDiv = document.createElement("div"); │ │ │ │ │ + this.layersDiv.id = this.id + "_layersDiv"; │ │ │ │ │ + OpenLayers.Element.addClass(this.layersDiv, "layersDiv"); │ │ │ │ │ + this.baseLbl = document.createElement("div"); │ │ │ │ │ + this.baseLbl.innerHTML = OpenLayers.i18n("Base Layer"); │ │ │ │ │ + OpenLayers.Element.addClass(this.baseLbl, "baseLbl"); │ │ │ │ │ + this.baseLayersDiv = document.createElement("div"); │ │ │ │ │ + OpenLayers.Element.addClass(this.baseLayersDiv, "baseLayersDiv"); │ │ │ │ │ + this.dataLbl = document.createElement("div"); │ │ │ │ │ + this.dataLbl.innerHTML = OpenLayers.i18n("Overlays"); │ │ │ │ │ + OpenLayers.Element.addClass(this.dataLbl, "dataLbl"); │ │ │ │ │ + this.dataLayersDiv = document.createElement("div"); │ │ │ │ │ + OpenLayers.Element.addClass(this.dataLayersDiv, "dataLayersDiv"); │ │ │ │ │ + if (this.ascending) { │ │ │ │ │ + this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLayersDiv) │ │ │ │ │ + } else { │ │ │ │ │ + this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLayersDiv) │ │ │ │ │ + } │ │ │ │ │ + this.div.appendChild(this.layersDiv); │ │ │ │ │ + var img = OpenLayers.Util.getImageLocation("layer-switcher-maximize.png"); │ │ │ │ │ + this.maximizeDiv = OpenLayers.Util.createAlphaImageDiv("OpenLayers_Control_MaximizeDiv", null, null, img, "absolute"); │ │ │ │ │ + OpenLayers.Element.addClass(this.maximizeDiv, "maximizeDiv olButton"); │ │ │ │ │ + this.maximizeDiv.style.display = "none"; │ │ │ │ │ + this.div.appendChild(this.maximizeDiv); │ │ │ │ │ + var img = OpenLayers.Util.getImageLocation("layer-switcher-minimize.png"); │ │ │ │ │ + this.minimizeDiv = OpenLayers.Util.createAlphaImageDiv("OpenLayers_Control_MinimizeDiv", null, null, img, "absolute"); │ │ │ │ │ + OpenLayers.Element.addClass(this.minimizeDiv, "minimizeDiv olButton"); │ │ │ │ │ + this.minimizeDiv.style.display = "none"; │ │ │ │ │ + this.div.appendChild(this.minimizeDiv) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.LayerSwitcher" │ │ │ │ │ +}); │ │ │ │ │ OpenLayers.Handler = OpenLayers.Class({ │ │ │ │ │ id: null, │ │ │ │ │ control: null, │ │ │ │ │ map: null, │ │ │ │ │ keyMask: null, │ │ │ │ │ active: false, │ │ │ │ │ evt: null, │ │ │ │ │ @@ -11639,402 +10329,14 @@ │ │ │ │ │ this.zoomWheelEnabled = true; │ │ │ │ │ if (this.active) { │ │ │ │ │ this.handlers.wheel.activate() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.Navigation" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Events.buttonclick = OpenLayers.Class({ │ │ │ │ │ - target: null, │ │ │ │ │ - events: ["mousedown", "mouseup", "click", "dblclick", "touchstart", "touchmove", "touchend", "keydown"], │ │ │ │ │ - startRegEx: /^mousedown|touchstart$/, │ │ │ │ │ - cancelRegEx: /^touchmove$/, │ │ │ │ │ - completeRegEx: /^mouseup|touchend$/, │ │ │ │ │ - initialize: function(target) { │ │ │ │ │ - this.target = target; │ │ │ │ │ - for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ - this.target.register(this.events[i], this, this.buttonClick, { │ │ │ │ │ - extension: true │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ - this.target.unregister(this.events[i], this, this.buttonClick) │ │ │ │ │ - } │ │ │ │ │ - delete this.target │ │ │ │ │ - }, │ │ │ │ │ - getPressedButton: function(element) { │ │ │ │ │ - var depth = 3, │ │ │ │ │ - button; │ │ │ │ │ - do { │ │ │ │ │ - if (OpenLayers.Element.hasClass(element, "olButton")) { │ │ │ │ │ - button = element; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - element = element.parentNode │ │ │ │ │ - } while (--depth > 0 && element); │ │ │ │ │ - return button │ │ │ │ │ - }, │ │ │ │ │ - ignore: function(element) { │ │ │ │ │ - var depth = 3, │ │ │ │ │ - ignore = false; │ │ │ │ │ - do { │ │ │ │ │ - if (element.nodeName.toLowerCase() === "a") { │ │ │ │ │ - ignore = true; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - element = element.parentNode │ │ │ │ │ - } while (--depth > 0 && element); │ │ │ │ │ - return ignore │ │ │ │ │ - }, │ │ │ │ │ - buttonClick: function(evt) { │ │ │ │ │ - var propagate = true, │ │ │ │ │ - element = OpenLayers.Event.element(evt); │ │ │ │ │ - if (element && (OpenLayers.Event.isLeftClick(evt) || !~evt.type.indexOf("mouse"))) { │ │ │ │ │ - var button = this.getPressedButton(element); │ │ │ │ │ - if (button) { │ │ │ │ │ - if (evt.type === "keydown") { │ │ │ │ │ - switch (evt.keyCode) { │ │ │ │ │ - case OpenLayers.Event.KEY_RETURN: │ │ │ │ │ - case OpenLayers.Event.KEY_SPACE: │ │ │ │ │ - this.target.triggerEvent("buttonclick", { │ │ │ │ │ - buttonElement: button │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } else if (this.startEvt) { │ │ │ │ │ - if (this.completeRegEx.test(evt.type)) { │ │ │ │ │ - var pos = OpenLayers.Util.pagePosition(button); │ │ │ │ │ - var viewportElement = OpenLayers.Util.getViewportElement(); │ │ │ │ │ - var scrollTop = window.pageYOffset || viewportElement.scrollTop; │ │ │ │ │ - var scrollLeft = window.pageXOffset || viewportElement.scrollLeft; │ │ │ │ │ - pos[0] = pos[0] - scrollLeft; │ │ │ │ │ - pos[1] = pos[1] - scrollTop; │ │ │ │ │ - this.target.triggerEvent("buttonclick", { │ │ │ │ │ - buttonElement: button, │ │ │ │ │ - buttonXY: { │ │ │ │ │ - x: this.startEvt.clientX - pos[0], │ │ │ │ │ - y: this.startEvt.clientY - pos[1] │ │ │ │ │ - } │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - if (this.cancelRegEx.test(evt.type)) { │ │ │ │ │ - delete this.startEvt │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false │ │ │ │ │ - } │ │ │ │ │ - if (this.startRegEx.test(evt.type)) { │ │ │ │ │ - this.startEvt = evt; │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - propagate = !this.ignore(OpenLayers.Event.element(evt)); │ │ │ │ │ - delete this.startEvt │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return propagate │ │ │ │ │ - } │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.Panel = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - controls: null, │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - defaultControl: null, │ │ │ │ │ - saveState: false, │ │ │ │ │ - allowDepress: false, │ │ │ │ │ - activeState: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.controls = []; │ │ │ │ │ - this.activeState = {} │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onButtonClick) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - for (var ctl, i = this.controls.length - 1; i >= 0; i--) { │ │ │ │ │ - ctl = this.controls[i]; │ │ │ │ │ - if (ctl.events) { │ │ │ │ │ - ctl.events.un({ │ │ │ │ │ - activate: this.iconOn, │ │ │ │ │ - deactivate: this.iconOff │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - ctl.panel_div = null │ │ │ │ │ - } │ │ │ │ │ - this.activeState = null │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - control = this.controls[i]; │ │ │ │ │ - if (control === this.defaultControl || this.saveState && this.activeState[control.id]) { │ │ │ │ │ - control.activate() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.saveState === true) { │ │ │ │ │ - this.defaultControl = null │ │ │ │ │ - } │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - control = this.controls[i]; │ │ │ │ │ - this.activeState[control.id] = control.deactivate() │ │ │ │ │ - } │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (this.outsideViewport) { │ │ │ │ │ - this.events.attachToElement(this.div); │ │ │ │ │ - this.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ - } else { │ │ │ │ │ - this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ - } │ │ │ │ │ - this.addControlsToMap(this.controls); │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - redraw: function() { │ │ │ │ │ - for (var l = this.div.childNodes.length, i = l - 1; i >= 0; i--) { │ │ │ │ │ - this.div.removeChild(this.div.childNodes[i]) │ │ │ │ │ - } │ │ │ │ │ - this.div.innerHTML = ""; │ │ │ │ │ - if (this.active) { │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - this.div.appendChild(this.controls[i].panel_div) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - activateControl: function(control) { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - if (control.type == OpenLayers.Control.TYPE_BUTTON) { │ │ │ │ │ - control.trigger(); │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (control.type == OpenLayers.Control.TYPE_TOGGLE) { │ │ │ │ │ - if (control.active) { │ │ │ │ │ - control.deactivate() │ │ │ │ │ - } else { │ │ │ │ │ - control.activate() │ │ │ │ │ - } │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (this.allowDepress && control.active) { │ │ │ │ │ - control.deactivate() │ │ │ │ │ - } else { │ │ │ │ │ - var c; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - c = this.controls[i]; │ │ │ │ │ - if (c != control && (c.type === OpenLayers.Control.TYPE_TOOL || c.type == null)) { │ │ │ │ │ - c.deactivate() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - control.activate() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - addControls: function(controls) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(controls)) { │ │ │ │ │ - controls = [controls] │ │ │ │ │ - } │ │ │ │ │ - this.controls = this.controls.concat(controls); │ │ │ │ │ - for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ - var control = controls[i], │ │ │ │ │ - element = this.createControlMarkup(control); │ │ │ │ │ - OpenLayers.Element.addClass(element, control.displayClass + "ItemInactive"); │ │ │ │ │ - OpenLayers.Element.addClass(element, "olButton"); │ │ │ │ │ - if (control.title != "" && !element.title) { │ │ │ │ │ - element.title = control.title │ │ │ │ │ - } │ │ │ │ │ - control.panel_div = element │ │ │ │ │ - } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.addControlsToMap(controls); │ │ │ │ │ - this.redraw() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - createControlMarkup: function(control) { │ │ │ │ │ - return document.createElement("div") │ │ │ │ │ - }, │ │ │ │ │ - addControlsToMap: function(controls) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ - control = controls[i]; │ │ │ │ │ - if (control.autoActivate === true) { │ │ │ │ │ - control.autoActivate = false; │ │ │ │ │ - this.map.addControl(control); │ │ │ │ │ - control.autoActivate = true │ │ │ │ │ - } else { │ │ │ │ │ - this.map.addControl(control); │ │ │ │ │ - control.deactivate() │ │ │ │ │ - } │ │ │ │ │ - control.events.on({ │ │ │ │ │ - activate: this.iconOn, │ │ │ │ │ - deactivate: this.iconOff │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - iconOn: function() { │ │ │ │ │ - var d = this.panel_div; │ │ │ │ │ - var re = new RegExp("\\b(" + this.displayClass + "Item)Inactive\\b"); │ │ │ │ │ - d.className = d.className.replace(re, "$1Active") │ │ │ │ │ - }, │ │ │ │ │ - iconOff: function() { │ │ │ │ │ - var d = this.panel_div; │ │ │ │ │ - var re = new RegExp("\\b(" + this.displayClass + "Item)Active\\b"); │ │ │ │ │ - d.className = d.className.replace(re, "$1Inactive") │ │ │ │ │ - }, │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - var controls = this.controls, │ │ │ │ │ - button = evt.buttonElement; │ │ │ │ │ - for (var i = controls.length - 1; i >= 0; --i) { │ │ │ │ │ - if (controls[i].panel_div === button) { │ │ │ │ │ - this.activateControl(controls[i]); │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getControlsBy: function(property, match) { │ │ │ │ │ - var test = typeof match.test == "function"; │ │ │ │ │ - var found = OpenLayers.Array.filter(this.controls, function(item) { │ │ │ │ │ - return item[property] == match || test && match.test(item[property]) │ │ │ │ │ - }); │ │ │ │ │ - return found │ │ │ │ │ - }, │ │ │ │ │ - getControlsByName: function(match) { │ │ │ │ │ - return this.getControlsBy("name", match) │ │ │ │ │ - }, │ │ │ │ │ - getControlsByClass: function(match) { │ │ │ │ │ - return this.getControlsBy("CLASS_NAME", match) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Panel" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.Attribution = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - separator: ", ", │ │ │ │ │ - template: "${layers}", │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - removelayer: this.updateAttribution, │ │ │ │ │ - addlayer: this.updateAttribution, │ │ │ │ │ - changelayer: this.updateAttribution, │ │ │ │ │ - changebaselayer: this.updateAttribution, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - changebaselayer: this.updateAttribution, │ │ │ │ │ - changelayer: this.updateAttribution, │ │ │ │ │ - addlayer: this.updateAttribution, │ │ │ │ │ - removelayer: this.updateAttribution, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.updateAttribution(); │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - updateAttribution: function() { │ │ │ │ │ - var attributions = []; │ │ │ │ │ - if (this.map && this.map.layers) { │ │ │ │ │ - for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if (layer.attribution && layer.getVisibility()) { │ │ │ │ │ - if (OpenLayers.Util.indexOf(attributions, layer.attribution) === -1) { │ │ │ │ │ - attributions.push(layer.attribution) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.div.innerHTML = OpenLayers.String.format(this.template, { │ │ │ │ │ - layers: attributions.join(this.separator) │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Attribution" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.Zoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - zoomInText: "+", │ │ │ │ │ - zoomInId: "olZoomInLink", │ │ │ │ │ - zoomOutText: "−", │ │ │ │ │ - zoomOutId: "olZoomOutLink", │ │ │ │ │ - draw: function() { │ │ │ │ │ - var div = OpenLayers.Control.prototype.draw.apply(this), │ │ │ │ │ - links = this.getOrCreateLinks(div), │ │ │ │ │ - zoomIn = links.zoomIn, │ │ │ │ │ - zoomOut = links.zoomOut, │ │ │ │ │ - eventsInstance = this.map.events; │ │ │ │ │ - if (zoomOut.parentNode !== div) { │ │ │ │ │ - eventsInstance = this.events; │ │ │ │ │ - eventsInstance.attachToElement(zoomOut.parentNode) │ │ │ │ │ - } │ │ │ │ │ - eventsInstance.register("buttonclick", this, this.onZoomClick); │ │ │ │ │ - this.zoomInLink = zoomIn; │ │ │ │ │ - this.zoomOutLink = zoomOut; │ │ │ │ │ - return div │ │ │ │ │ - }, │ │ │ │ │ - getOrCreateLinks: function(el) { │ │ │ │ │ - var zoomIn = document.getElementById(this.zoomInId), │ │ │ │ │ - zoomOut = document.getElementById(this.zoomOutId); │ │ │ │ │ - if (!zoomIn) { │ │ │ │ │ - zoomIn = document.createElement("a"); │ │ │ │ │ - zoomIn.href = "#zoomIn"; │ │ │ │ │ - zoomIn.appendChild(document.createTextNode(this.zoomInText)); │ │ │ │ │ - zoomIn.className = "olControlZoomIn"; │ │ │ │ │ - el.appendChild(zoomIn) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Element.addClass(zoomIn, "olButton"); │ │ │ │ │ - if (!zoomOut) { │ │ │ │ │ - zoomOut = document.createElement("a"); │ │ │ │ │ - zoomOut.href = "#zoomOut"; │ │ │ │ │ - zoomOut.appendChild(document.createTextNode(this.zoomOutText)); │ │ │ │ │ - zoomOut.className = "olControlZoomOut"; │ │ │ │ │ - el.appendChild(zoomOut) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Element.addClass(zoomOut, "olButton"); │ │ │ │ │ - return { │ │ │ │ │ - zoomIn: zoomIn, │ │ │ │ │ - zoomOut: zoomOut │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - onZoomClick: function(evt) { │ │ │ │ │ - var button = evt.buttonElement; │ │ │ │ │ - if (button === this.zoomInLink) { │ │ │ │ │ - this.map.zoomIn() │ │ │ │ │ - } else if (button === this.zoomOutLink) { │ │ │ │ │ - this.map.zoomOut() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onZoomClick) │ │ │ │ │ - } │ │ │ │ │ - delete this.zoomInLink; │ │ │ │ │ - delete this.zoomOutLink; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Zoom" │ │ │ │ │ -}); │ │ │ │ │ OpenLayers.Handler.Feature = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ EVENTMAP: { │ │ │ │ │ click: { │ │ │ │ │ in: "click", │ │ │ │ │ out: "clickout" │ │ │ │ │ }, │ │ │ │ │ mousemove: { │ │ │ │ │ @@ -12205,2683 +10507,4381 @@ │ │ │ │ │ this.layer.setZIndex(index) │ │ │ │ │ } else { │ │ │ │ │ this.map.setLayerZIndex(this.layer, this.map.getLayerIndex(this.layer)) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Handler.Feature" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.Vector.RootContainer = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - layers: null, │ │ │ │ │ - display: function() {}, │ │ │ │ │ - getFeatureFromEvent: function(evt) { │ │ │ │ │ - var layers = this.layers; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0; i < layers.length; i++) { │ │ │ │ │ - feature = layers[i].getFeatureFromEvent(evt); │ │ │ │ │ - if (feature) { │ │ │ │ │ - return feature │ │ │ │ │ - } │ │ │ │ │ +OpenLayers.Layer = OpenLayers.Class({ │ │ │ │ │ + id: null, │ │ │ │ │ + name: null, │ │ │ │ │ + div: null, │ │ │ │ │ + opacity: 1, │ │ │ │ │ + alwaysInRange: null, │ │ │ │ │ + RESOLUTION_PROPERTIES: ["scales", "resolutions", "maxScale", "minScale", "maxResolution", "minResolution", "numZoomLevels", "maxZoomLevel"], │ │ │ │ │ + events: null, │ │ │ │ │ + map: null, │ │ │ │ │ + isBaseLayer: false, │ │ │ │ │ + alpha: false, │ │ │ │ │ + displayInLayerSwitcher: true, │ │ │ │ │ + visibility: true, │ │ │ │ │ + attribution: null, │ │ │ │ │ + inRange: false, │ │ │ │ │ + imageSize: null, │ │ │ │ │ + options: null, │ │ │ │ │ + eventListeners: null, │ │ │ │ │ + gutter: 0, │ │ │ │ │ + projection: null, │ │ │ │ │ + units: null, │ │ │ │ │ + scales: null, │ │ │ │ │ + resolutions: null, │ │ │ │ │ + maxExtent: null, │ │ │ │ │ + minExtent: null, │ │ │ │ │ + maxResolution: null, │ │ │ │ │ + minResolution: null, │ │ │ │ │ + numZoomLevels: null, │ │ │ │ │ + minScale: null, │ │ │ │ │ + maxScale: null, │ │ │ │ │ + displayOutsideMaxExtent: false, │ │ │ │ │ + wrapDateLine: false, │ │ │ │ │ + metadata: null, │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + this.metadata = {}; │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + if (this.alwaysInRange != null) { │ │ │ │ │ + options.alwaysInRange = this.alwaysInRange │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.collectRoots(); │ │ │ │ │ - map.events.register("changelayer", this, this.handleChangeLayer) │ │ │ │ │ - }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - map.events.unregister("changelayer", this, this.handleChangeLayer); │ │ │ │ │ - this.resetRoots(); │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - collectRoots: function() { │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = 0; i < this.map.layers.length; ++i) { │ │ │ │ │ - layer = this.map.layers[i]; │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ - layer.renderer.moveRoot(this.renderer) │ │ │ │ │ + this.addOptions(options); │ │ │ │ │ + this.name = name; │ │ │ │ │ + if (this.id == null) { │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ + this.div = OpenLayers.Util.createDiv(this.id); │ │ │ │ │ + this.div.style.width = "100%"; │ │ │ │ │ + this.div.style.height = "100%"; │ │ │ │ │ + this.div.dir = "ltr"; │ │ │ │ │ + this.events = new OpenLayers.Events(this, this.div); │ │ │ │ │ + if (this.eventListeners instanceof Object) { │ │ │ │ │ + this.events.on(this.eventListeners) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - resetRoots: function() { │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = 0; i < this.layers.length; ++i) { │ │ │ │ │ - layer = this.layers[i]; │ │ │ │ │ - if (this.renderer && layer.renderer.getRenderLayerId() == this.id) { │ │ │ │ │ - this.renderer.moveRoot(layer.renderer) │ │ │ │ │ + destroy: function(setNewBaseLayer) { │ │ │ │ │ + if (setNewBaseLayer == null) { │ │ │ │ │ + setNewBaseLayer = true │ │ │ │ │ + } │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.removeLayer(this, setNewBaseLayer) │ │ │ │ │ + } │ │ │ │ │ + this.projection = null; │ │ │ │ │ + this.map = null; │ │ │ │ │ + this.name = null; │ │ │ │ │ + this.div = null; │ │ │ │ │ + this.options = null; │ │ │ │ │ + if (this.events) { │ │ │ │ │ + if (this.eventListeners) { │ │ │ │ │ + this.events.un(this.eventListeners) │ │ │ │ │ } │ │ │ │ │ + this.events.destroy() │ │ │ │ │ } │ │ │ │ │ + this.eventListeners = null; │ │ │ │ │ + this.events = null │ │ │ │ │ }, │ │ │ │ │ - handleChangeLayer: function(evt) { │ │ │ │ │ - var layer = evt.layer; │ │ │ │ │ - if (evt.property == "order" && OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ - this.resetRoots(); │ │ │ │ │ - this.collectRoots() │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer(this.name, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Util.applyDefaults(obj, this); │ │ │ │ │ + obj.map = null; │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Vector.RootContainer" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.SelectFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - multipleKey: null, │ │ │ │ │ - toggleKey: null, │ │ │ │ │ - multiple: false, │ │ │ │ │ - clickout: true, │ │ │ │ │ - toggle: false, │ │ │ │ │ - hover: false, │ │ │ │ │ - highlightOnly: false, │ │ │ │ │ - box: false, │ │ │ │ │ - onBeforeSelect: function() {}, │ │ │ │ │ - onSelect: function() {}, │ │ │ │ │ - onUnselect: function() {}, │ │ │ │ │ - scope: null, │ │ │ │ │ - geometryTypes: null, │ │ │ │ │ - layer: null, │ │ │ │ │ - layers: null, │ │ │ │ │ - callbacks: null, │ │ │ │ │ - selectStyle: null, │ │ │ │ │ - renderIntent: "select", │ │ │ │ │ - handlers: null, │ │ │ │ │ - initialize: function(layers, options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (this.scope === null) { │ │ │ │ │ - this.scope = this │ │ │ │ │ - } │ │ │ │ │ - this.initLayer(layers); │ │ │ │ │ - var callbacks = { │ │ │ │ │ - click: this.clickFeature, │ │ │ │ │ - clickout: this.clickoutFeature │ │ │ │ │ - }; │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - callbacks.over = this.overFeature; │ │ │ │ │ - callbacks.out = this.outFeature │ │ │ │ │ - } │ │ │ │ │ - this.callbacks = OpenLayers.Util.extend(callbacks, this.callbacks); │ │ │ │ │ - this.handlers = { │ │ │ │ │ - feature: new OpenLayers.Handler.Feature(this, this.layer, this.callbacks, { │ │ │ │ │ - geometryTypes: this.geometryTypes │ │ │ │ │ - }) │ │ │ │ │ - }; │ │ │ │ │ - if (this.box) { │ │ │ │ │ - this.handlers.box = new OpenLayers.Handler.Box(this, { │ │ │ │ │ - done: this.selectBox │ │ │ │ │ - }, { │ │ │ │ │ - boxDivClassName: "olHandlerBoxSelectFeature" │ │ │ │ │ - }) │ │ │ │ │ + getOptions: function() { │ │ │ │ │ + var options = {}; │ │ │ │ │ + for (var o in this.options) { │ │ │ │ │ + options[o] = this[o] │ │ │ │ │ } │ │ │ │ │ + return options │ │ │ │ │ }, │ │ │ │ │ - initLayer: function(layers) { │ │ │ │ │ - if (OpenLayers.Util.isArray(layers)) { │ │ │ │ │ - this.layers = layers; │ │ │ │ │ - this.layer = new OpenLayers.Layer.Vector.RootContainer(this.id + "_container", { │ │ │ │ │ - layers: layers │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - this.layer = layers │ │ │ │ │ + setName: function(newName) { │ │ │ │ │ + if (newName != this.name) { │ │ │ │ │ + this.name = newName; │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "name" │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.active && this.layers) { │ │ │ │ │ - this.map.removeLayer(this.layer) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - if (this.layers) { │ │ │ │ │ - this.layer.destroy() │ │ │ │ │ + addOptions: function(newOptions, reinitialize) { │ │ │ │ │ + if (this.options == null) { │ │ │ │ │ + this.options = {} │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - if (this.layers) { │ │ │ │ │ - this.map.addLayer(this.layer) │ │ │ │ │ + if (newOptions) { │ │ │ │ │ + if (typeof newOptions.projection == "string") { │ │ │ │ │ + newOptions.projection = new OpenLayers.Projection(newOptions.projection) │ │ │ │ │ } │ │ │ │ │ - this.handlers.feature.activate(); │ │ │ │ │ - if (this.box && this.handlers.box) { │ │ │ │ │ - this.handlers.box.activate() │ │ │ │ │ + if (newOptions.projection) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(newOptions, OpenLayers.Projection.defaults[newOptions.projection.getCode()]) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Control.prototype.activate.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.handlers.feature.deactivate(); │ │ │ │ │ - if (this.handlers.box) { │ │ │ │ │ - this.handlers.box.deactivate() │ │ │ │ │ + if (newOptions.maxExtent && !(newOptions.maxExtent instanceof OpenLayers.Bounds)) { │ │ │ │ │ + newOptions.maxExtent = new OpenLayers.Bounds(newOptions.maxExtent) │ │ │ │ │ } │ │ │ │ │ - if (this.layers) { │ │ │ │ │ - this.map.removeLayer(this.layer) │ │ │ │ │ + if (newOptions.minExtent && !(newOptions.minExtent instanceof OpenLayers.Bounds)) { │ │ │ │ │ + newOptions.minExtent = new OpenLayers.Bounds(newOptions.minExtent) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Control.prototype.deactivate.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - unselectAll: function(options) { │ │ │ │ │ - var layers = this.layers || [this.layer], │ │ │ │ │ - layer, feature, l, numExcept; │ │ │ │ │ - for (l = 0; l < layers.length; ++l) { │ │ │ │ │ - layer = layers[l]; │ │ │ │ │ - numExcept = 0; │ │ │ │ │ - if (layer.selectedFeatures != null) { │ │ │ │ │ - while (layer.selectedFeatures.length > numExcept) { │ │ │ │ │ - feature = layer.selectedFeatures[numExcept]; │ │ │ │ │ - if (!options || options.except != feature) { │ │ │ │ │ - this.unselect(feature) │ │ │ │ │ - } else { │ │ │ │ │ - ++numExcept │ │ │ │ │ + OpenLayers.Util.extend(this.options, newOptions); │ │ │ │ │ + OpenLayers.Util.extend(this, newOptions); │ │ │ │ │ + if (this.projection && this.projection.getUnits()) { │ │ │ │ │ + this.units = this.projection.getUnits() │ │ │ │ │ + } │ │ │ │ │ + if (this.map) { │ │ │ │ │ + var resolution = this.map.getResolution(); │ │ │ │ │ + var properties = this.RESOLUTION_PROPERTIES.concat(["projection", "units", "minExtent", "maxExtent"]); │ │ │ │ │ + for (var o in newOptions) { │ │ │ │ │ + if (newOptions.hasOwnProperty(o) && OpenLayers.Util.indexOf(properties, o) >= 0) { │ │ │ │ │ + this.initResolutions(); │ │ │ │ │ + if (reinitialize && this.map.baseLayer === this) { │ │ │ │ │ + this.map.setCenter(this.map.getCenter(), this.map.getZoomForResolution(resolution), false, true); │ │ │ │ │ + this.map.events.triggerEvent("changebaselayer", { │ │ │ │ │ + layer: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - clickFeature: function(feature) { │ │ │ │ │ - if (!this.hover) { │ │ │ │ │ - var selected = OpenLayers.Util.indexOf(feature.layer.selectedFeatures, feature) > -1; │ │ │ │ │ - if (selected) { │ │ │ │ │ - if (this.toggleSelect()) { │ │ │ │ │ - this.unselect(feature) │ │ │ │ │ - } else if (!this.multipleSelect()) { │ │ │ │ │ - this.unselectAll({ │ │ │ │ │ - except: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - if (!this.multipleSelect()) { │ │ │ │ │ - this.unselectAll({ │ │ │ │ │ - except: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.select(feature) │ │ │ │ │ + onMapResize: function() {}, │ │ │ │ │ + redraw: function() { │ │ │ │ │ + var redrawn = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.inRange = this.calculateInRange(); │ │ │ │ │ + var extent = this.getExtent(); │ │ │ │ │ + if (extent && this.inRange && this.visibility) { │ │ │ │ │ + var zoomChanged = true; │ │ │ │ │ + this.moveTo(extent, zoomChanged, false); │ │ │ │ │ + this.events.triggerEvent("moveend", { │ │ │ │ │ + zoomChanged: zoomChanged │ │ │ │ │ + }); │ │ │ │ │ + redrawn = true │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return redrawn │ │ │ │ │ }, │ │ │ │ │ - multipleSelect: function() { │ │ │ │ │ - return this.multiple || this.handlers.feature.evt && this.handlers.feature.evt[this.multipleKey] │ │ │ │ │ - }, │ │ │ │ │ - toggleSelect: function() { │ │ │ │ │ - return this.toggle || this.handlers.feature.evt && this.handlers.feature.evt[this.toggleKey] │ │ │ │ │ - }, │ │ │ │ │ - clickoutFeature: function(feature) { │ │ │ │ │ - if (!this.hover && this.clickout) { │ │ │ │ │ - this.unselectAll() │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + var display = this.visibility; │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + display = display && this.inRange │ │ │ │ │ } │ │ │ │ │ + this.display(display) │ │ │ │ │ }, │ │ │ │ │ - overFeature: function(feature) { │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - if (this.highlightOnly) { │ │ │ │ │ - this.highlight(feature) │ │ │ │ │ - } else if (OpenLayers.Util.indexOf(layer.selectedFeatures, feature) == -1) { │ │ │ │ │ - this.select(feature) │ │ │ │ │ + moveByPx: function(dx, dy) {}, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + if (this.map == null) { │ │ │ │ │ + this.map = map; │ │ │ │ │ + this.maxExtent = this.maxExtent || this.map.maxExtent; │ │ │ │ │ + this.minExtent = this.minExtent || this.map.minExtent; │ │ │ │ │ + this.projection = this.projection || this.map.projection; │ │ │ │ │ + if (typeof this.projection == "string") { │ │ │ │ │ + this.projection = new OpenLayers.Projection(this.projection) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - outFeature: function(feature) { │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - if (this.highlightOnly) { │ │ │ │ │ - if (feature._lastHighlighter == this.id) { │ │ │ │ │ - if (feature._prevHighlighter && feature._prevHighlighter != this.id) { │ │ │ │ │ - delete feature._lastHighlighter; │ │ │ │ │ - var control = this.map.getControl(feature._prevHighlighter); │ │ │ │ │ - if (control) { │ │ │ │ │ - control.highlight(feature) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.unhighlight(feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.unselect(feature) │ │ │ │ │ + this.units = this.projection.getUnits() || this.units || this.map.units; │ │ │ │ │ + this.initResolutions(); │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + this.inRange = this.calculateInRange(); │ │ │ │ │ + var show = this.visibility && this.inRange; │ │ │ │ │ + this.div.style.display = show ? "" : "none" │ │ │ │ │ } │ │ │ │ │ + this.setTileSize() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - highlight: function(feature) { │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - var cont = this.events.triggerEvent("beforefeaturehighlighted", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - feature._prevHighlighter = feature._lastHighlighter; │ │ │ │ │ - feature._lastHighlighter = this.id; │ │ │ │ │ - var style = this.selectStyle || this.renderIntent; │ │ │ │ │ - layer.drawFeature(feature, style); │ │ │ │ │ - this.events.triggerEvent("featurehighlighted", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ + afterAdd: function() {}, │ │ │ │ │ + removeMap: function(map) {}, │ │ │ │ │ + getImageSize: function(bounds) { │ │ │ │ │ + return this.imageSize || this.tileSize │ │ │ │ │ }, │ │ │ │ │ - unhighlight: function(feature) { │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - if (feature._prevHighlighter == undefined) { │ │ │ │ │ - delete feature._lastHighlighter │ │ │ │ │ - } else if (feature._prevHighlighter == this.id) { │ │ │ │ │ - delete feature._prevHighlighter │ │ │ │ │ - } else { │ │ │ │ │ - feature._lastHighlighter = feature._prevHighlighter; │ │ │ │ │ - delete feature._prevHighlighter │ │ │ │ │ + setTileSize: function(size) { │ │ │ │ │ + var tileSize = size ? size : this.tileSize ? this.tileSize : this.map.getTileSize(); │ │ │ │ │ + this.tileSize = tileSize; │ │ │ │ │ + if (this.gutter) { │ │ │ │ │ + this.imageSize = new OpenLayers.Size(tileSize.w + 2 * this.gutter, tileSize.h + 2 * this.gutter) │ │ │ │ │ } │ │ │ │ │ - layer.drawFeature(feature, feature.style || feature.layer.style || "default"); │ │ │ │ │ - this.events.triggerEvent("featureunhighlighted", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ }, │ │ │ │ │ - select: function(feature) { │ │ │ │ │ - var cont = this.onBeforeSelect.call(this.scope, feature); │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - cont = layer.events.triggerEvent("beforefeatureselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - layer.selectedFeatures.push(feature); │ │ │ │ │ - this.highlight(feature); │ │ │ │ │ - if (!this.handlers.feature.lastFeature) { │ │ │ │ │ - this.handlers.feature.lastFeature = layer.selectedFeatures[0] │ │ │ │ │ - } │ │ │ │ │ - layer.events.triggerEvent("featureselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - this.onSelect.call(this.scope, feature) │ │ │ │ │ + getVisibility: function() { │ │ │ │ │ + return this.visibility │ │ │ │ │ + }, │ │ │ │ │ + setVisibility: function(visibility) { │ │ │ │ │ + if (visibility != this.visibility) { │ │ │ │ │ + this.visibility = visibility; │ │ │ │ │ + this.display(visibility); │ │ │ │ │ + this.redraw(); │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "visibility" │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ + this.events.triggerEvent("visibilitychanged") │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - unselect: function(feature) { │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - this.unhighlight(feature); │ │ │ │ │ - OpenLayers.Util.removeItem(layer.selectedFeatures, feature); │ │ │ │ │ - layer.events.triggerEvent("featureunselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - this.onUnselect.call(this.scope, feature) │ │ │ │ │ + display: function(display) { │ │ │ │ │ + if (display != (this.div.style.display != "none")) { │ │ │ │ │ + this.div.style.display = display && this.calculateInRange() ? "block" : "none" │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - selectBox: function(position) { │ │ │ │ │ - if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ - var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.left, │ │ │ │ │ - y: position.bottom │ │ │ │ │ - }); │ │ │ │ │ - var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.right, │ │ │ │ │ - y: position.top │ │ │ │ │ - }); │ │ │ │ │ - var bounds = new OpenLayers.Bounds(minXY.lon, minXY.lat, maxXY.lon, maxXY.lat); │ │ │ │ │ - if (!this.multipleSelect()) { │ │ │ │ │ - this.unselectAll() │ │ │ │ │ - } │ │ │ │ │ - var prevMultiple = this.multiple; │ │ │ │ │ - this.multiple = true; │ │ │ │ │ - var layers = this.layers || [this.layer]; │ │ │ │ │ - this.events.triggerEvent("boxselectionstart", { │ │ │ │ │ - layers: layers │ │ │ │ │ - }); │ │ │ │ │ - var layer; │ │ │ │ │ - for (var l = 0; l < layers.length; ++l) { │ │ │ │ │ - layer = layers[l]; │ │ │ │ │ - for (var i = 0, len = layer.features.length; i < len; ++i) { │ │ │ │ │ - var feature = layer.features[i]; │ │ │ │ │ - if (!feature.getVisibility()) { │ │ │ │ │ - continue │ │ │ │ │ - } │ │ │ │ │ - if (this.geometryTypes == null || OpenLayers.Util.indexOf(this.geometryTypes, feature.geometry.CLASS_NAME) > -1) { │ │ │ │ │ - if (bounds.toGeometry().intersects(feature.geometry)) { │ │ │ │ │ - if (OpenLayers.Util.indexOf(layer.selectedFeatures, feature) == -1) { │ │ │ │ │ - this.select(feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + calculateInRange: function() { │ │ │ │ │ + var inRange = false; │ │ │ │ │ + if (this.alwaysInRange) { │ │ │ │ │ + inRange = true │ │ │ │ │ + } else { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + var resolution = this.map.getResolution(); │ │ │ │ │ + inRange = resolution >= this.minResolution && resolution <= this.maxResolution │ │ │ │ │ } │ │ │ │ │ - this.multiple = prevMultiple; │ │ │ │ │ - this.events.triggerEvent("boxselectionend", { │ │ │ │ │ - layers: layers │ │ │ │ │ - }) │ │ │ │ │ } │ │ │ │ │ + return inRange │ │ │ │ │ }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.handlers.feature.setMap(map); │ │ │ │ │ - if (this.box) { │ │ │ │ │ - this.handlers.box.setMap(map) │ │ │ │ │ + setIsBaseLayer: function(isBaseLayer) { │ │ │ │ │ + if (isBaseLayer != this.isBaseLayer) { │ │ │ │ │ + this.isBaseLayer = isBaseLayer; │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changebaselayer", { │ │ │ │ │ + layer: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - setLayer: function(layers) { │ │ │ │ │ - var isActive = this.active; │ │ │ │ │ - this.unselectAll(); │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - if (this.layers) { │ │ │ │ │ - this.layer.destroy(); │ │ │ │ │ - this.layers = null │ │ │ │ │ + initResolutions: function() { │ │ │ │ │ + var i, len, p; │ │ │ │ │ + var props = {}, │ │ │ │ │ + alwaysInRange = true; │ │ │ │ │ + for (i = 0, len = this.RESOLUTION_PROPERTIES.length; i < len; i++) { │ │ │ │ │ + p = this.RESOLUTION_PROPERTIES[i]; │ │ │ │ │ + props[p] = this.options[p]; │ │ │ │ │ + if (alwaysInRange && this.options[p]) { │ │ │ │ │ + alwaysInRange = false │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.initLayer(layers); │ │ │ │ │ - this.handlers.feature.layer = this.layer; │ │ │ │ │ - if (isActive) { │ │ │ │ │ - this.activate() │ │ │ │ │ + if (this.options.alwaysInRange == null) { │ │ │ │ │ + this.alwaysInRange = alwaysInRange │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.SelectFeature" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.LayerSwitcher = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - layerStates: null, │ │ │ │ │ - layersDiv: null, │ │ │ │ │ - baseLayersDiv: null, │ │ │ │ │ - baseLayers: null, │ │ │ │ │ - dataLbl: null, │ │ │ │ │ - dataLayersDiv: null, │ │ │ │ │ - dataLayers: null, │ │ │ │ │ - minimizeDiv: null, │ │ │ │ │ - maximizeDiv: null, │ │ │ │ │ - ascending: true, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.layerStates = [] │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.clearLayersArray("base"); │ │ │ │ │ - this.clearLayersArray("data"); │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - buttonclick: this.onButtonClick, │ │ │ │ │ - addlayer: this.redraw, │ │ │ │ │ - changelayer: this.redraw, │ │ │ │ │ - removelayer: this.redraw, │ │ │ │ │ - changebaselayer: this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - addlayer: this.redraw, │ │ │ │ │ - changelayer: this.redraw, │ │ │ │ │ - removelayer: this.redraw, │ │ │ │ │ - changebaselayer: this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - if (this.outsideViewport) { │ │ │ │ │ - this.events.attachToElement(this.div); │ │ │ │ │ - this.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ - } else { │ │ │ │ │ - this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + props.resolutions = this.resolutionsFromScales(props.scales) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this); │ │ │ │ │ - this.loadContents(); │ │ │ │ │ - if (!this.outsideViewport) { │ │ │ │ │ - this.minimizeControl() │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + props.resolutions = this.calculateResolutions(props) │ │ │ │ │ } │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - var button = evt.buttonElement; │ │ │ │ │ - if (button === this.minimizeDiv) { │ │ │ │ │ - this.minimizeControl() │ │ │ │ │ - } else if (button === this.maximizeDiv) { │ │ │ │ │ - this.maximizeControl() │ │ │ │ │ - } else if (button._layerSwitcher === this.id) { │ │ │ │ │ - if (button["for"]) { │ │ │ │ │ - button = document.getElementById(button["for"]) │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + for (i = 0, len = this.RESOLUTION_PROPERTIES.length; i < len; i++) { │ │ │ │ │ + p = this.RESOLUTION_PROPERTIES[i]; │ │ │ │ │ + props[p] = this.options[p] != null ? this.options[p] : this.map[p] │ │ │ │ │ } │ │ │ │ │ - if (!button.disabled) { │ │ │ │ │ - if (button.type == "radio") { │ │ │ │ │ - button.checked = true; │ │ │ │ │ - this.map.setBaseLayer(this.map.getLayer(button._layer)) │ │ │ │ │ - } else { │ │ │ │ │ - button.checked = !button.checked; │ │ │ │ │ - this.updateMap() │ │ │ │ │ - } │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + props.resolutions = this.resolutionsFromScales(props.scales) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - clearLayersArray: function(layersType) { │ │ │ │ │ - this[layersType + "LayersDiv"].innerHTML = ""; │ │ │ │ │ - this[layersType + "Layers"] = [] │ │ │ │ │ - }, │ │ │ │ │ - checkRedraw: function() { │ │ │ │ │ - if (!this.layerStates.length || this.map.layers.length != this.layerStates.length) { │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = this.layerStates.length; i < len; i++) { │ │ │ │ │ - var layerState = this.layerStates[i]; │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if (layerState.name != layer.name || layerState.inRange != layer.inRange || layerState.id != layer.id || layerState.visibility != layer.visibility) { │ │ │ │ │ - return true │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + props.resolutions = this.calculateResolutions(props) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - redraw: function() { │ │ │ │ │ - if (!this.checkRedraw()) { │ │ │ │ │ - return this.div │ │ │ │ │ + var maxResolution; │ │ │ │ │ + if (this.options.maxResolution && this.options.maxResolution !== "auto") { │ │ │ │ │ + maxResolution = this.options.maxResolution │ │ │ │ │ } │ │ │ │ │ - this.clearLayersArray("base"); │ │ │ │ │ - this.clearLayersArray("data"); │ │ │ │ │ - var containsOverlays = false; │ │ │ │ │ - var containsBaseLayers = false; │ │ │ │ │ - var len = this.map.layers.length; │ │ │ │ │ - this.layerStates = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - this.layerStates[i] = { │ │ │ │ │ - name: layer.name, │ │ │ │ │ - visibility: layer.visibility, │ │ │ │ │ - inRange: layer.inRange, │ │ │ │ │ - id: layer.id │ │ │ │ │ - } │ │ │ │ │ + if (this.options.minScale) { │ │ │ │ │ + maxResolution = OpenLayers.Util.getResolutionFromScale(this.options.minScale, this.units) │ │ │ │ │ } │ │ │ │ │ - var layers = this.map.layers.slice(); │ │ │ │ │ - if (!this.ascending) { │ │ │ │ │ - layers.reverse() │ │ │ │ │ + var minResolution; │ │ │ │ │ + if (this.options.minResolution && this.options.minResolution !== "auto") { │ │ │ │ │ + minResolution = this.options.minResolution │ │ │ │ │ } │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - var layer = layers[i]; │ │ │ │ │ - var baseLayer = layer.isBaseLayer; │ │ │ │ │ - if (layer.displayInLayerSwitcher) { │ │ │ │ │ - if (baseLayer) { │ │ │ │ │ - containsBaseLayers = true │ │ │ │ │ - } else { │ │ │ │ │ - containsOverlays = true │ │ │ │ │ - } │ │ │ │ │ - var checked = baseLayer ? layer == this.map.baseLayer : layer.getVisibility(); │ │ │ │ │ - var inputElem = document.createElement("input"), │ │ │ │ │ - inputId = OpenLayers.Util.createUniqueID(this.id + "_input_"); │ │ │ │ │ - inputElem.id = inputId; │ │ │ │ │ - inputElem.name = baseLayer ? this.id + "_baseLayers" : layer.name; │ │ │ │ │ - inputElem.type = baseLayer ? "radio" : "checkbox"; │ │ │ │ │ - inputElem.value = layer.name; │ │ │ │ │ - inputElem.checked = checked; │ │ │ │ │ - inputElem.defaultChecked = checked; │ │ │ │ │ - inputElem.className = "olButton"; │ │ │ │ │ - inputElem._layer = layer.id; │ │ │ │ │ - inputElem._layerSwitcher = this.id; │ │ │ │ │ - if (!baseLayer && !layer.inRange) { │ │ │ │ │ - inputElem.disabled = true │ │ │ │ │ - } │ │ │ │ │ - var labelSpan = document.createElement("label"); │ │ │ │ │ - labelSpan["for"] = inputElem.id; │ │ │ │ │ - OpenLayers.Element.addClass(labelSpan, "labelSpan olButton"); │ │ │ │ │ - labelSpan._layer = layer.id; │ │ │ │ │ - labelSpan._layerSwitcher = this.id; │ │ │ │ │ - if (!baseLayer && !layer.inRange) { │ │ │ │ │ - labelSpan.style.color = "gray" │ │ │ │ │ - } │ │ │ │ │ - labelSpan.innerHTML = layer.name; │ │ │ │ │ - labelSpan.style.verticalAlign = baseLayer ? "bottom" : "baseline"; │ │ │ │ │ - var br = document.createElement("br"); │ │ │ │ │ - var groupArray = baseLayer ? this.baseLayers : this.dataLayers; │ │ │ │ │ - groupArray.push({ │ │ │ │ │ - layer: layer, │ │ │ │ │ - inputElem: inputElem, │ │ │ │ │ - labelSpan: labelSpan │ │ │ │ │ - }); │ │ │ │ │ - var groupDiv = baseLayer ? this.baseLayersDiv : this.dataLayersDiv; │ │ │ │ │ - groupDiv.appendChild(inputElem); │ │ │ │ │ - groupDiv.appendChild(labelSpan); │ │ │ │ │ - groupDiv.appendChild(br) │ │ │ │ │ + if (this.options.maxScale) { │ │ │ │ │ + minResolution = OpenLayers.Util.getResolutionFromScale(this.options.maxScale, this.units) │ │ │ │ │ + } │ │ │ │ │ + if (props.resolutions) { │ │ │ │ │ + props.resolutions.sort(function(a, b) { │ │ │ │ │ + return b - a │ │ │ │ │ + }); │ │ │ │ │ + if (!maxResolution) { │ │ │ │ │ + maxResolution = props.resolutions[0] │ │ │ │ │ + } │ │ │ │ │ + if (!minResolution) { │ │ │ │ │ + var lastIdx = props.resolutions.length - 1; │ │ │ │ │ + minResolution = props.resolutions[lastIdx] │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.dataLbl.style.display = containsOverlays ? "" : "none"; │ │ │ │ │ - this.baseLbl.style.display = containsBaseLayers ? "" : "none"; │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - updateMap: function() { │ │ │ │ │ - for (var i = 0, len = this.baseLayers.length; i < len; i++) { │ │ │ │ │ - var layerEntry = this.baseLayers[i]; │ │ │ │ │ - if (layerEntry.inputElem.checked) { │ │ │ │ │ - this.map.setBaseLayer(layerEntry.layer, false) │ │ │ │ │ + this.resolutions = props.resolutions; │ │ │ │ │ + if (this.resolutions) { │ │ │ │ │ + len = this.resolutions.length; │ │ │ │ │ + this.scales = new Array(len); │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + this.scales[i] = OpenLayers.Util.getScaleFromResolution(this.resolutions[i], this.units) │ │ │ │ │ } │ │ │ │ │ + this.numZoomLevels = len │ │ │ │ │ } │ │ │ │ │ - for (var i = 0, len = this.dataLayers.length; i < len; i++) { │ │ │ │ │ - var layerEntry = this.dataLayers[i]; │ │ │ │ │ - layerEntry.layer.setVisibility(layerEntry.inputElem.checked) │ │ │ │ │ + this.minResolution = minResolution; │ │ │ │ │ + if (minResolution) { │ │ │ │ │ + this.maxScale = OpenLayers.Util.getScaleFromResolution(minResolution, this.units) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - maximizeControl: function(e) { │ │ │ │ │ - this.div.style.width = ""; │ │ │ │ │ - this.div.style.height = ""; │ │ │ │ │ - this.showControls(false); │ │ │ │ │ - if (e != null) { │ │ │ │ │ - OpenLayers.Event.stop(e) │ │ │ │ │ + this.maxResolution = maxResolution; │ │ │ │ │ + if (maxResolution) { │ │ │ │ │ + this.minScale = OpenLayers.Util.getScaleFromResolution(maxResolution, this.units) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - minimizeControl: function(e) { │ │ │ │ │ - this.div.style.width = "0px"; │ │ │ │ │ - this.div.style.height = "0px"; │ │ │ │ │ - this.showControls(true); │ │ │ │ │ - if (e != null) { │ │ │ │ │ - OpenLayers.Event.stop(e) │ │ │ │ │ + resolutionsFromScales: function(scales) { │ │ │ │ │ + if (scales == null) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - showControls: function(minimize) { │ │ │ │ │ - this.maximizeDiv.style.display = minimize ? "" : "none"; │ │ │ │ │ - this.minimizeDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ - this.layersDiv.style.display = minimize ? "none" : "" │ │ │ │ │ - }, │ │ │ │ │ - loadContents: function() { │ │ │ │ │ - this.layersDiv = document.createElement("div"); │ │ │ │ │ - this.layersDiv.id = this.id + "_layersDiv"; │ │ │ │ │ - OpenLayers.Element.addClass(this.layersDiv, "layersDiv"); │ │ │ │ │ - this.baseLbl = document.createElement("div"); │ │ │ │ │ - this.baseLbl.innerHTML = OpenLayers.i18n("Base Layer"); │ │ │ │ │ - OpenLayers.Element.addClass(this.baseLbl, "baseLbl"); │ │ │ │ │ - this.baseLayersDiv = document.createElement("div"); │ │ │ │ │ - OpenLayers.Element.addClass(this.baseLayersDiv, "baseLayersDiv"); │ │ │ │ │ - this.dataLbl = document.createElement("div"); │ │ │ │ │ - this.dataLbl.innerHTML = OpenLayers.i18n("Overlays"); │ │ │ │ │ - OpenLayers.Element.addClass(this.dataLbl, "dataLbl"); │ │ │ │ │ - this.dataLayersDiv = document.createElement("div"); │ │ │ │ │ - OpenLayers.Element.addClass(this.dataLayersDiv, "dataLayersDiv"); │ │ │ │ │ - if (this.ascending) { │ │ │ │ │ - this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLayersDiv) │ │ │ │ │ - } else { │ │ │ │ │ - this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLayersDiv) │ │ │ │ │ + var resolutions, i, len; │ │ │ │ │ + len = scales.length; │ │ │ │ │ + resolutions = new Array(len); │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + resolutions[i] = OpenLayers.Util.getResolutionFromScale(scales[i], this.units) │ │ │ │ │ } │ │ │ │ │ - this.div.appendChild(this.layersDiv); │ │ │ │ │ - var img = OpenLayers.Util.getImageLocation("layer-switcher-maximize.png"); │ │ │ │ │ - this.maximizeDiv = OpenLayers.Util.createAlphaImageDiv("OpenLayers_Control_MaximizeDiv", null, null, img, "absolute"); │ │ │ │ │ - OpenLayers.Element.addClass(this.maximizeDiv, "maximizeDiv olButton"); │ │ │ │ │ - this.maximizeDiv.style.display = "none"; │ │ │ │ │ - this.div.appendChild(this.maximizeDiv); │ │ │ │ │ - var img = OpenLayers.Util.getImageLocation("layer-switcher-minimize.png"); │ │ │ │ │ - this.minimizeDiv = OpenLayers.Util.createAlphaImageDiv("OpenLayers_Control_MinimizeDiv", null, null, img, "absolute"); │ │ │ │ │ - OpenLayers.Element.addClass(this.minimizeDiv, "minimizeDiv olButton"); │ │ │ │ │ - this.minimizeDiv.style.display = "none"; │ │ │ │ │ - this.div.appendChild(this.minimizeDiv) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.LayerSwitcher" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.ElementsIndexer = OpenLayers.Class({ │ │ │ │ │ - maxZIndex: null, │ │ │ │ │ - order: null, │ │ │ │ │ - indices: null, │ │ │ │ │ - compare: null, │ │ │ │ │ - initialize: function(yOrdering) { │ │ │ │ │ - this.compare = yOrdering ? OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_Y_ORDER : OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_DRAWING_ORDER; │ │ │ │ │ - this.clear() │ │ │ │ │ + return resolutions │ │ │ │ │ }, │ │ │ │ │ - insert: function(newNode) { │ │ │ │ │ - if (this.exists(newNode)) { │ │ │ │ │ - this.remove(newNode) │ │ │ │ │ + calculateResolutions: function(props) { │ │ │ │ │ + var viewSize, wRes, hRes; │ │ │ │ │ + var maxResolution = props.maxResolution; │ │ │ │ │ + if (props.minScale != null) { │ │ │ │ │ + maxResolution = OpenLayers.Util.getResolutionFromScale(props.minScale, this.units) │ │ │ │ │ + } else if (maxResolution == "auto" && this.maxExtent != null) { │ │ │ │ │ + viewSize = this.map.getSize(); │ │ │ │ │ + wRes = this.maxExtent.getWidth() / viewSize.w; │ │ │ │ │ + hRes = this.maxExtent.getHeight() / viewSize.h; │ │ │ │ │ + maxResolution = Math.max(wRes, hRes) │ │ │ │ │ } │ │ │ │ │ - var nodeId = newNode.id; │ │ │ │ │ - this.determineZIndex(newNode); │ │ │ │ │ - var leftIndex = -1; │ │ │ │ │ - var rightIndex = this.order.length; │ │ │ │ │ - var middle; │ │ │ │ │ - while (rightIndex - leftIndex > 1) { │ │ │ │ │ - middle = parseInt((leftIndex + rightIndex) / 2); │ │ │ │ │ - var placement = this.compare(this, newNode, OpenLayers.Util.getElement(this.order[middle])); │ │ │ │ │ - if (placement > 0) { │ │ │ │ │ - leftIndex = middle │ │ │ │ │ - } else { │ │ │ │ │ - rightIndex = middle │ │ │ │ │ - } │ │ │ │ │ + var minResolution = props.minResolution; │ │ │ │ │ + if (props.maxScale != null) { │ │ │ │ │ + minResolution = OpenLayers.Util.getResolutionFromScale(props.maxScale, this.units) │ │ │ │ │ + } else if (props.minResolution == "auto" && this.minExtent != null) { │ │ │ │ │ + viewSize = this.map.getSize(); │ │ │ │ │ + wRes = this.minExtent.getWidth() / viewSize.w; │ │ │ │ │ + hRes = this.minExtent.getHeight() / viewSize.h; │ │ │ │ │ + minResolution = Math.max(wRes, hRes) │ │ │ │ │ } │ │ │ │ │ - this.order.splice(rightIndex, 0, nodeId); │ │ │ │ │ - this.indices[nodeId] = this.getZIndex(newNode); │ │ │ │ │ - return this.getNextElement(rightIndex) │ │ │ │ │ - }, │ │ │ │ │ - remove: function(node) { │ │ │ │ │ - var nodeId = node.id; │ │ │ │ │ - var arrayIndex = OpenLayers.Util.indexOf(this.order, nodeId); │ │ │ │ │ - if (arrayIndex >= 0) { │ │ │ │ │ - this.order.splice(arrayIndex, 1); │ │ │ │ │ - delete this.indices[nodeId]; │ │ │ │ │ - if (this.order.length > 0) { │ │ │ │ │ - var lastId = this.order[this.order.length - 1]; │ │ │ │ │ - this.maxZIndex = this.indices[lastId] │ │ │ │ │ - } else { │ │ │ │ │ - this.maxZIndex = 0 │ │ │ │ │ + if (typeof maxResolution !== "number" && typeof minResolution !== "number" && this.maxExtent != null) { │ │ │ │ │ + var tileSize = this.map.getTileSize(); │ │ │ │ │ + maxResolution = Math.max(this.maxExtent.getWidth() / tileSize.w, this.maxExtent.getHeight() / tileSize.h) │ │ │ │ │ + } │ │ │ │ │ + var maxZoomLevel = props.maxZoomLevel; │ │ │ │ │ + var numZoomLevels = props.numZoomLevels; │ │ │ │ │ + if (typeof minResolution === "number" && typeof maxResolution === "number" && numZoomLevels === undefined) { │ │ │ │ │ + var ratio = maxResolution / minResolution; │ │ │ │ │ + numZoomLevels = Math.floor(Math.log(ratio) / Math.log(2)) + 1 │ │ │ │ │ + } else if (numZoomLevels === undefined && maxZoomLevel != null) { │ │ │ │ │ + numZoomLevels = maxZoomLevel + 1 │ │ │ │ │ + } │ │ │ │ │ + if (typeof numZoomLevels !== "number" || numZoomLevels <= 0 || typeof maxResolution !== "number" && typeof minResolution !== "number") { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var resolutions = new Array(numZoomLevels); │ │ │ │ │ + var base = 2; │ │ │ │ │ + if (typeof minResolution == "number" && typeof maxResolution == "number") { │ │ │ │ │ + base = Math.pow(maxResolution / minResolution, 1 / (numZoomLevels - 1)) │ │ │ │ │ + } │ │ │ │ │ + var i; │ │ │ │ │ + if (typeof maxResolution === "number") { │ │ │ │ │ + for (i = 0; i < numZoomLevels; i++) { │ │ │ │ │ + resolutions[i] = maxResolution / Math.pow(base, i) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + for (i = 0; i < numZoomLevels; i++) { │ │ │ │ │ + resolutions[numZoomLevels - 1 - i] = minResolution * Math.pow(base, i) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return resolutions │ │ │ │ │ }, │ │ │ │ │ - clear: function() { │ │ │ │ │ - this.order = []; │ │ │ │ │ - this.indices = {}; │ │ │ │ │ - this.maxZIndex = 0 │ │ │ │ │ - }, │ │ │ │ │ - exists: function(node) { │ │ │ │ │ - return this.indices[node.id] != null │ │ │ │ │ + getResolution: function() { │ │ │ │ │ + var zoom = this.map.getZoom(); │ │ │ │ │ + return this.getResolutionForZoom(zoom) │ │ │ │ │ }, │ │ │ │ │ - getZIndex: function(node) { │ │ │ │ │ - return node._style.graphicZIndex │ │ │ │ │ + getExtent: function() { │ │ │ │ │ + return this.map.calculateBounds() │ │ │ │ │ }, │ │ │ │ │ - determineZIndex: function(node) { │ │ │ │ │ - var zIndex = node._style.graphicZIndex; │ │ │ │ │ - if (zIndex == null) { │ │ │ │ │ - zIndex = this.maxZIndex; │ │ │ │ │ - node._style.graphicZIndex = zIndex │ │ │ │ │ - } else if (zIndex > this.maxZIndex) { │ │ │ │ │ - this.maxZIndex = zIndex │ │ │ │ │ - } │ │ │ │ │ + getZoomForExtent: function(extent, closest) { │ │ │ │ │ + var viewSize = this.map.getSize(); │ │ │ │ │ + var idealResolution = Math.max(extent.getWidth() / viewSize.w, extent.getHeight() / viewSize.h); │ │ │ │ │ + return this.getZoomForResolution(idealResolution, closest) │ │ │ │ │ }, │ │ │ │ │ - getNextElement: function(index) { │ │ │ │ │ - var nextIndex = index + 1; │ │ │ │ │ - if (nextIndex < this.order.length) { │ │ │ │ │ - var nextElement = OpenLayers.Util.getElement(this.order[nextIndex]); │ │ │ │ │ - if (nextElement == undefined) { │ │ │ │ │ - nextElement = this.getNextElement(nextIndex) │ │ │ │ │ - } │ │ │ │ │ - return nextElement │ │ │ │ │ + getDataExtent: function() {}, │ │ │ │ │ + getResolutionForZoom: function(zoom) { │ │ │ │ │ + zoom = Math.max(0, Math.min(zoom, this.resolutions.length - 1)); │ │ │ │ │ + var resolution; │ │ │ │ │ + if (this.map.fractionalZoom) { │ │ │ │ │ + var low = Math.floor(zoom); │ │ │ │ │ + var high = Math.ceil(zoom); │ │ │ │ │ + resolution = this.resolutions[low] - (zoom - low) * (this.resolutions[low] - this.resolutions[high]) │ │ │ │ │ } else { │ │ │ │ │ - return null │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.ElementsIndexer" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.ElementsIndexer.IndexingMethods = { │ │ │ │ │ - Z_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var newZIndex = indexer.getZIndex(newNode); │ │ │ │ │ - var returnVal = 0; │ │ │ │ │ - if (nextNode) { │ │ │ │ │ - var nextZIndex = indexer.getZIndex(nextNode); │ │ │ │ │ - returnVal = newZIndex - nextZIndex │ │ │ │ │ + resolution = this.resolutions[Math.round(zoom)] │ │ │ │ │ } │ │ │ │ │ - return returnVal │ │ │ │ │ + return resolution │ │ │ │ │ }, │ │ │ │ │ - Z_ORDER_DRAWING_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER(indexer, newNode, nextNode); │ │ │ │ │ - if (nextNode && returnVal == 0) { │ │ │ │ │ - returnVal = 1 │ │ │ │ │ + getZoomForResolution: function(resolution, closest) { │ │ │ │ │ + var zoom, i, len; │ │ │ │ │ + if (this.map.fractionalZoom) { │ │ │ │ │ + var lowZoom = 0; │ │ │ │ │ + var highZoom = this.resolutions.length - 1; │ │ │ │ │ + var highRes = this.resolutions[lowZoom]; │ │ │ │ │ + var lowRes = this.resolutions[highZoom]; │ │ │ │ │ + var res; │ │ │ │ │ + for (i = 0, len = this.resolutions.length; i < len; ++i) { │ │ │ │ │ + res = this.resolutions[i]; │ │ │ │ │ + if (res >= resolution) { │ │ │ │ │ + highRes = res; │ │ │ │ │ + lowZoom = i │ │ │ │ │ + } │ │ │ │ │ + if (res <= resolution) { │ │ │ │ │ + lowRes = res; │ │ │ │ │ + highZoom = i; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var dRes = highRes - lowRes; │ │ │ │ │ + if (dRes > 0) { │ │ │ │ │ + zoom = lowZoom + (highRes - resolution) / dRes │ │ │ │ │ + } else { │ │ │ │ │ + zoom = lowZoom │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + var diff; │ │ │ │ │ + var minDiff = Number.POSITIVE_INFINITY; │ │ │ │ │ + for (i = 0, len = this.resolutions.length; i < len; i++) { │ │ │ │ │ + if (closest) { │ │ │ │ │ + diff = Math.abs(this.resolutions[i] - resolution); │ │ │ │ │ + if (diff > minDiff) { │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + minDiff = diff │ │ │ │ │ + } else { │ │ │ │ │ + if (this.resolutions[i] < resolution) { │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + zoom = Math.max(0, i - 1) │ │ │ │ │ } │ │ │ │ │ - return returnVal │ │ │ │ │ + return zoom │ │ │ │ │ }, │ │ │ │ │ - Z_ORDER_Y_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER(indexer, newNode, nextNode); │ │ │ │ │ - if (nextNode && returnVal === 0) { │ │ │ │ │ - var result = nextNode._boundsBottom - newNode._boundsBottom; │ │ │ │ │ - returnVal = result === 0 ? 1 : result │ │ │ │ │ + getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ + var lonlat = null; │ │ │ │ │ + var map = this.map; │ │ │ │ │ + if (viewPortPx != null && map.minPx) { │ │ │ │ │ + var res = map.getResolution(); │ │ │ │ │ + var maxExtent = map.getMaxExtent({ │ │ │ │ │ + restricted: true │ │ │ │ │ + }); │ │ │ │ │ + var lon = (viewPortPx.x - map.minPx.x) * res + maxExtent.left; │ │ │ │ │ + var lat = (map.minPx.y - viewPortPx.y) * res + maxExtent.top; │ │ │ │ │ + lonlat = new OpenLayers.LonLat(lon, lat); │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + lonlat = lonlat.wrapDateLine(this.maxExtent) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return returnVal │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.Elements = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ - rendererRoot: null, │ │ │ │ │ - root: null, │ │ │ │ │ - vectorRoot: null, │ │ │ │ │ - textRoot: null, │ │ │ │ │ - xmlns: null, │ │ │ │ │ - xOffset: 0, │ │ │ │ │ - indexer: null, │ │ │ │ │ - BACKGROUND_ID_SUFFIX: "_background", │ │ │ │ │ - LABEL_ID_SUFFIX: "_label", │ │ │ │ │ - LABEL_OUTLINE_SUFFIX: "_outline", │ │ │ │ │ - initialize: function(containerID, options) { │ │ │ │ │ - OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.rendererRoot = this.createRenderRoot(); │ │ │ │ │ - this.root = this.createRoot("_root"); │ │ │ │ │ - this.vectorRoot = this.createRoot("_vroot"); │ │ │ │ │ - this.textRoot = this.createRoot("_troot"); │ │ │ │ │ - this.root.appendChild(this.vectorRoot); │ │ │ │ │ - this.root.appendChild(this.textRoot); │ │ │ │ │ - this.rendererRoot.appendChild(this.root); │ │ │ │ │ - this.container.appendChild(this.rendererRoot); │ │ │ │ │ - if (options && (options.zIndexing || options.yOrdering)) { │ │ │ │ │ - this.indexer = new OpenLayers.ElementsIndexer(options.yOrdering) │ │ │ │ │ + return lonlat │ │ │ │ │ + }, │ │ │ │ │ + getViewPortPxFromLonLat: function(lonlat, resolution) { │ │ │ │ │ + var px = null; │ │ │ │ │ + if (lonlat != null) { │ │ │ │ │ + resolution = resolution || this.map.getResolution(); │ │ │ │ │ + var extent = this.map.calculateBounds(null, resolution); │ │ │ │ │ + px = new OpenLayers.Pixel(1 / resolution * (lonlat.lon - extent.left), 1 / resolution * (extent.top - lonlat.lat)) │ │ │ │ │ + } │ │ │ │ │ + return px │ │ │ │ │ + }, │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + if (opacity != this.opacity) { │ │ │ │ │ + this.opacity = opacity; │ │ │ │ │ + var childNodes = this.div.childNodes; │ │ │ │ │ + for (var i = 0, len = childNodes.length; i < len; ++i) { │ │ │ │ │ + var element = childNodes[i].firstChild || childNodes[i]; │ │ │ │ │ + var lastChild = childNodes[i].lastChild; │ │ │ │ │ + if (lastChild && lastChild.nodeName.toLowerCase() === "iframe") { │ │ │ │ │ + element = lastChild.parentNode │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.modifyDOMElement(element, null, null, null, null, null, null, opacity) │ │ │ │ │ + } │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "opacity" │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getZIndex: function() { │ │ │ │ │ + return this.div.style.zIndex │ │ │ │ │ + }, │ │ │ │ │ + setZIndex: function(zIndex) { │ │ │ │ │ + this.div.style.zIndex = zIndex │ │ │ │ │ + }, │ │ │ │ │ + adjustBounds: function(bounds) { │ │ │ │ │ + if (this.gutter) { │ │ │ │ │ + var mapGutter = this.gutter * this.map.getResolution(); │ │ │ │ │ + bounds = new OpenLayers.Bounds(bounds.left - mapGutter, bounds.bottom - mapGutter, bounds.right + mapGutter, bounds.top + mapGutter) │ │ │ │ │ + } │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + var wrappingOptions = { │ │ │ │ │ + rightTolerance: this.getResolution(), │ │ │ │ │ + leftTolerance: this.getResolution() │ │ │ │ │ + }; │ │ │ │ │ + bounds = bounds.wrapDateLine(this.maxExtent, wrappingOptions) │ │ │ │ │ + } │ │ │ │ │ + return bounds │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Vector = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ + isBaseLayer: false, │ │ │ │ │ + isFixed: false, │ │ │ │ │ + features: null, │ │ │ │ │ + filter: null, │ │ │ │ │ + selectedFeatures: null, │ │ │ │ │ + unrenderedFeatures: null, │ │ │ │ │ + reportError: true, │ │ │ │ │ + style: null, │ │ │ │ │ + styleMap: null, │ │ │ │ │ + strategies: null, │ │ │ │ │ + protocol: null, │ │ │ │ │ + renderers: ["SVG", "VML", "Canvas"], │ │ │ │ │ + renderer: null, │ │ │ │ │ + rendererOptions: null, │ │ │ │ │ + geometryType: null, │ │ │ │ │ + drawn: false, │ │ │ │ │ + ratio: 1, │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ + this.assignRenderer() │ │ │ │ │ + } │ │ │ │ │ + if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ + this.renderer = null; │ │ │ │ │ + this.displayError() │ │ │ │ │ + } │ │ │ │ │ + if (!this.styleMap) { │ │ │ │ │ + this.styleMap = new OpenLayers.StyleMap │ │ │ │ │ + } │ │ │ │ │ + this.features = []; │ │ │ │ │ + this.selectedFeatures = []; │ │ │ │ │ + this.unrenderedFeatures = {}; │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + for (var i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + this.strategies[i].setLayer(this) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - this.clear(); │ │ │ │ │ - this.rendererRoot = null; │ │ │ │ │ - this.root = null; │ │ │ │ │ - this.xmlns = null; │ │ │ │ │ - OpenLayers.Renderer.prototype.destroy.apply(this, arguments) │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoDestroy) { │ │ │ │ │ + strategy.destroy() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.strategies = null │ │ │ │ │ + } │ │ │ │ │ + if (this.protocol) { │ │ │ │ │ + if (this.protocol.autoDestroy) { │ │ │ │ │ + this.protocol.destroy() │ │ │ │ │ + } │ │ │ │ │ + this.protocol = null │ │ │ │ │ + } │ │ │ │ │ + this.destroyFeatures(); │ │ │ │ │ + this.features = null; │ │ │ │ │ + this.selectedFeatures = null; │ │ │ │ │ + this.unrenderedFeatures = null; │ │ │ │ │ + if (this.renderer) { │ │ │ │ │ + this.renderer.destroy() │ │ │ │ │ + } │ │ │ │ │ + this.renderer = null; │ │ │ │ │ + this.geometryType = null; │ │ │ │ │ + this.drawn = null; │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - clear: function() { │ │ │ │ │ - var child; │ │ │ │ │ - var root = this.vectorRoot; │ │ │ │ │ - if (root) { │ │ │ │ │ - while (child = root.firstChild) { │ │ │ │ │ - root.removeChild(child) │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Vector(this.name, this.getOptions()) │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ + var features = this.features; │ │ │ │ │ + var len = features.length; │ │ │ │ │ + var clonedFeatures = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + clonedFeatures[i] = features[i].clone() │ │ │ │ │ + } │ │ │ │ │ + obj.features = clonedFeatures; │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + refresh: function(obj) { │ │ │ │ │ + if (this.calculateInRange() && this.visibility) { │ │ │ │ │ + this.events.triggerEvent("refresh", obj) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + assignRenderer: function() { │ │ │ │ │ + for (var i = 0, len = this.renderers.length; i < len; i++) { │ │ │ │ │ + var rendererClass = this.renderers[i]; │ │ │ │ │ + var renderer = typeof rendererClass == "function" ? rendererClass : OpenLayers.Renderer[rendererClass]; │ │ │ │ │ + if (renderer && renderer.prototype.supported()) { │ │ │ │ │ + this.renderer = new renderer(this.div, this.rendererOptions); │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - root = this.textRoot; │ │ │ │ │ - if (root) { │ │ │ │ │ - while (child = root.firstChild) { │ │ │ │ │ - root.removeChild(child) │ │ │ │ │ + }, │ │ │ │ │ + displayError: function() { │ │ │ │ │ + if (this.reportError) { │ │ │ │ │ + OpenLayers.Console.userError(OpenLayers.i18n("browserNotSupported", { │ │ │ │ │ + renderers: this.renderers.join("\n") │ │ │ │ │ + })) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (!this.renderer) { │ │ │ │ │ + this.map.removeLayer(this) │ │ │ │ │ + } else { │ │ │ │ │ + this.renderer.map = this.map; │ │ │ │ │ + var newSize = this.map.getSize(); │ │ │ │ │ + newSize.w = newSize.w * this.ratio; │ │ │ │ │ + newSize.h = newSize.h * this.ratio; │ │ │ │ │ + this.renderer.setSize(newSize) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + afterAdd: function() { │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoActivate) { │ │ │ │ │ + strategy.activate() │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - this.indexer.clear() │ │ │ │ │ + }, │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + this.drawn = false; │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoActivate) { │ │ │ │ │ + strategy.deactivate() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - var rightOfDateLine, ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ - extent = extent.scale(1 / ratio), │ │ │ │ │ - world = this.map.getMaxExtent(); │ │ │ │ │ - if (world.right > extent.left && world.right < extent.right) { │ │ │ │ │ - rightOfDateLine = true │ │ │ │ │ - } else if (world.left > extent.left && world.left < extent.right) { │ │ │ │ │ - rightOfDateLine = false │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + OpenLayers.Layer.prototype.onMapResize.apply(this, arguments); │ │ │ │ │ + var newSize = this.map.getSize(); │ │ │ │ │ + newSize.w = newSize.w * this.ratio; │ │ │ │ │ + newSize.h = newSize.h * this.ratio; │ │ │ │ │ + this.renderer.setSize(newSize) │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + var coordSysUnchanged = true; │ │ │ │ │ + if (!dragging) { │ │ │ │ │ + this.renderer.root.style.visibility = "hidden"; │ │ │ │ │ + var viewSize = this.map.getSize(), │ │ │ │ │ + viewWidth = viewSize.w, │ │ │ │ │ + viewHeight = viewSize.h, │ │ │ │ │ + offsetLeft = viewWidth / 2 * this.ratio - viewWidth / 2, │ │ │ │ │ + offsetTop = viewHeight / 2 * this.ratio - viewHeight / 2; │ │ │ │ │ + offsetLeft += this.map.layerContainerOriginPx.x; │ │ │ │ │ + offsetLeft = -Math.round(offsetLeft); │ │ │ │ │ + offsetTop += this.map.layerContainerOriginPx.y; │ │ │ │ │ + offsetTop = -Math.round(offsetTop); │ │ │ │ │ + this.div.style.left = offsetLeft + "px"; │ │ │ │ │ + this.div.style.top = offsetTop + "px"; │ │ │ │ │ + var extent = this.map.getExtent().scale(this.ratio); │ │ │ │ │ + coordSysUnchanged = this.renderer.setExtent(extent, zoomChanged); │ │ │ │ │ + this.renderer.root.style.visibility = "visible"; │ │ │ │ │ + if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ + this.div.scrollLeft = this.div.scrollLeft │ │ │ │ │ } │ │ │ │ │ - if (rightOfDateLine !== this.rightOfDateLine || resolutionChanged) { │ │ │ │ │ - coordSysUnchanged = false; │ │ │ │ │ - this.xOffset = rightOfDateLine === true ? world.getWidth() / resolution : 0 │ │ │ │ │ + if (!zoomChanged && coordSysUnchanged) { │ │ │ │ │ + for (var i in this.unrenderedFeatures) { │ │ │ │ │ + var feature = this.unrenderedFeatures[i]; │ │ │ │ │ + this.drawFeature(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!this.drawn || zoomChanged || !coordSysUnchanged) { │ │ │ │ │ + this.drawn = true; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0, len = this.features.length; i < len; i++) { │ │ │ │ │ + this.renderer.locked = i !== len - 1; │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + this.drawFeature(feature) │ │ │ │ │ } │ │ │ │ │ - this.rightOfDateLine = rightOfDateLine │ │ │ │ │ } │ │ │ │ │ - return coordSysUnchanged │ │ │ │ │ }, │ │ │ │ │ - getNodeType: function(geometry, style) {}, │ │ │ │ │ - drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ - var className = geometry.CLASS_NAME; │ │ │ │ │ - var rendered = true; │ │ │ │ │ - if (className == "OpenLayers.Geometry.Collection" || className == "OpenLayers.Geometry.MultiPoint" || className == "OpenLayers.Geometry.MultiLineString" || className == "OpenLayers.Geometry.MultiPolygon") { │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - rendered = this.drawGeometry(geometry.components[i], style, featureId) && rendered │ │ │ │ │ + display: function(display) { │ │ │ │ │ + OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ + var currentDisplay = this.div.style.display; │ │ │ │ │ + if (currentDisplay != this.renderer.root.style.display) { │ │ │ │ │ + this.renderer.root.style.display = currentDisplay │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addFeatures: function(features, options) { │ │ │ │ │ + if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ + features = [features] │ │ │ │ │ + } │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ + if (notify) { │ │ │ │ │ + var event = { │ │ │ │ │ + features: features │ │ │ │ │ + }; │ │ │ │ │ + var ret = this.events.triggerEvent("beforefeaturesadded", event); │ │ │ │ │ + if (ret === false) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - return rendered │ │ │ │ │ + features = event.features │ │ │ │ │ } │ │ │ │ │ - rendered = false; │ │ │ │ │ - var removeBackground = false; │ │ │ │ │ - if (style.display != "none") { │ │ │ │ │ - if (style.backgroundGraphic) { │ │ │ │ │ - this.redrawBackgroundNode(geometry.id, geometry, style, featureId) │ │ │ │ │ + var featuresAdded = []; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + if (i != features.length - 1) { │ │ │ │ │ + this.renderer.locked = true │ │ │ │ │ } else { │ │ │ │ │ - removeBackground = true │ │ │ │ │ + this.renderer.locked = false │ │ │ │ │ } │ │ │ │ │ - rendered = this.redrawNode(geometry.id, geometry, style, featureId) │ │ │ │ │ - } │ │ │ │ │ - if (rendered == false) { │ │ │ │ │ - var node = document.getElementById(geometry.id); │ │ │ │ │ - if (node) { │ │ │ │ │ - if (node._style.backgroundGraphic) { │ │ │ │ │ - removeBackground = true │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + if (this.geometryType && !(feature.geometry instanceof this.geometryType)) { │ │ │ │ │ + throw new TypeError("addFeatures: component should be an " + this.geometryType.prototype.CLASS_NAME) │ │ │ │ │ + } │ │ │ │ │ + feature.layer = this; │ │ │ │ │ + if (!feature.style && this.style) { │ │ │ │ │ + feature.style = OpenLayers.Util.extend({}, this.style) │ │ │ │ │ + } │ │ │ │ │ + if (notify) { │ │ │ │ │ + if (this.events.triggerEvent("beforefeatureadded", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) === false) { │ │ │ │ │ + continue │ │ │ │ │ } │ │ │ │ │ - node.parentNode.removeChild(node) │ │ │ │ │ + this.preFeatureInsert(feature) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - if (removeBackground) { │ │ │ │ │ - var node = document.getElementById(geometry.id + this.BACKGROUND_ID_SUFFIX); │ │ │ │ │ - if (node) { │ │ │ │ │ - node.parentNode.removeChild(node) │ │ │ │ │ + featuresAdded.push(feature); │ │ │ │ │ + this.features.push(feature); │ │ │ │ │ + this.drawFeature(feature); │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureadded", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.onFeatureInsert(feature) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return rendered │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresadded", { │ │ │ │ │ + features: featuresAdded │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - redrawNode: function(id, geometry, style, featureId) { │ │ │ │ │ - style = this.applyDefaultSymbolizer(style); │ │ │ │ │ - var node = this.nodeFactory(id, this.getNodeType(geometry, style)); │ │ │ │ │ - node._featureId = featureId; │ │ │ │ │ - node._boundsBottom = geometry.getBounds().bottom; │ │ │ │ │ - node._geometryClass = geometry.CLASS_NAME; │ │ │ │ │ - node._style = style; │ │ │ │ │ - var drawResult = this.drawGeometryNode(node, geometry, style); │ │ │ │ │ - if (drawResult === false) { │ │ │ │ │ - return false │ │ │ │ │ + removeFeatures: function(features, options) { │ │ │ │ │ + if (!features || features.length === 0) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - node = drawResult.node; │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - var insert = this.indexer.insert(node); │ │ │ │ │ - if (insert) { │ │ │ │ │ - this.vectorRoot.insertBefore(node, insert) │ │ │ │ │ + if (features === this.features) { │ │ │ │ │ + return this.removeAllFeatures(options) │ │ │ │ │ + } │ │ │ │ │ + if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ + features = [features] │ │ │ │ │ + } │ │ │ │ │ + if (features === this.selectedFeatures) { │ │ │ │ │ + features = features.slice() │ │ │ │ │ + } │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeaturesremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + if (i != 0 && features[i - 1].geometry) { │ │ │ │ │ + this.renderer.locked = true │ │ │ │ │ } else { │ │ │ │ │ - this.vectorRoot.appendChild(node) │ │ │ │ │ + this.renderer.locked = false │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - if (node.parentNode !== this.vectorRoot) { │ │ │ │ │ - this.vectorRoot.appendChild(node) │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + this.features = OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ + feature.layer = null; │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + this.renderer.eraseFeatures(feature) │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.selectedFeatures, feature) != -1) { │ │ │ │ │ + OpenLayers.Util.removeItem(this.selectedFeatures, feature) │ │ │ │ │ + } │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.postDraw(node); │ │ │ │ │ - return drawResult.complete │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - redrawBackgroundNode: function(id, geometry, style, featureId) { │ │ │ │ │ - var backgroundStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ - backgroundStyle.externalGraphic = backgroundStyle.backgroundGraphic; │ │ │ │ │ - backgroundStyle.graphicXOffset = backgroundStyle.backgroundXOffset; │ │ │ │ │ - backgroundStyle.graphicYOffset = backgroundStyle.backgroundYOffset; │ │ │ │ │ - backgroundStyle.graphicZIndex = backgroundStyle.backgroundGraphicZIndex; │ │ │ │ │ - backgroundStyle.graphicWidth = backgroundStyle.backgroundWidth || backgroundStyle.graphicWidth; │ │ │ │ │ - backgroundStyle.graphicHeight = backgroundStyle.backgroundHeight || backgroundStyle.graphicHeight; │ │ │ │ │ - backgroundStyle.backgroundGraphic = null; │ │ │ │ │ - backgroundStyle.backgroundXOffset = null; │ │ │ │ │ - backgroundStyle.backgroundYOffset = null; │ │ │ │ │ - backgroundStyle.backgroundGraphicZIndex = null; │ │ │ │ │ - return this.redrawNode(id + this.BACKGROUND_ID_SUFFIX, geometry, backgroundStyle, null) │ │ │ │ │ + removeAllFeatures: function(options) { │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ + var features = this.features; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeaturesremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + feature.layer = null; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.renderer.clear(); │ │ │ │ │ + this.features = []; │ │ │ │ │ + this.unrenderedFeatures = {}; │ │ │ │ │ + this.selectedFeatures = []; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - drawGeometryNode: function(node, geometry, style) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - var options = { │ │ │ │ │ - isFilled: style.fill === undefined ? true : style.fill, │ │ │ │ │ - isStroked: style.stroke === undefined ? !!style.strokeWidth : style.stroke │ │ │ │ │ - }; │ │ │ │ │ - var drawn; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.graphic === false) { │ │ │ │ │ - options.isFilled = false; │ │ │ │ │ - options.isStroked = false │ │ │ │ │ - } │ │ │ │ │ - drawn = this.drawPoint(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - options.isFilled = false; │ │ │ │ │ - drawn = this.drawLineString(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - drawn = this.drawLinearRing(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - drawn = this.drawPolygon(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - drawn = this.drawRectangle(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break │ │ │ │ │ + destroyFeatures: function(features, options) { │ │ │ │ │ + var all = features == undefined; │ │ │ │ │ + if (all) { │ │ │ │ │ + features = this.features │ │ │ │ │ } │ │ │ │ │ - node._options = options; │ │ │ │ │ - if (drawn != false) { │ │ │ │ │ - return { │ │ │ │ │ - node: this.setStyle(node, style, options, geometry), │ │ │ │ │ - complete: drawn │ │ │ │ │ + if (features) { │ │ │ │ │ + this.removeFeatures(features, options); │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + features[i].destroy() │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - postDraw: function(node) {}, │ │ │ │ │ - drawPoint: function(node, geometry) {}, │ │ │ │ │ - drawLineString: function(node, geometry) {}, │ │ │ │ │ - drawLinearRing: function(node, geometry) {}, │ │ │ │ │ - drawPolygon: function(node, geometry) {}, │ │ │ │ │ - drawRectangle: function(node, geometry) {}, │ │ │ │ │ - drawCircle: function(node, geometry) {}, │ │ │ │ │ - removeText: function(featureId) { │ │ │ │ │ - var label = document.getElementById(featureId + this.LABEL_ID_SUFFIX); │ │ │ │ │ - if (label) { │ │ │ │ │ - this.textRoot.removeChild(label) │ │ │ │ │ + drawFeature: function(feature, style) { │ │ │ │ │ + if (!this.drawn) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - var outline = document.getElementById(featureId + this.LABEL_OUTLINE_SUFFIX); │ │ │ │ │ - if (outline) { │ │ │ │ │ - this.textRoot.removeChild(outline) │ │ │ │ │ + if (typeof style != "object") { │ │ │ │ │ + if (!style && feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ + style = "delete" │ │ │ │ │ + } │ │ │ │ │ + var renderIntent = style || feature.renderIntent; │ │ │ │ │ + style = feature.style || this.style; │ │ │ │ │ + if (!style) { │ │ │ │ │ + style = this.styleMap.createSymbolizer(feature, renderIntent) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var drawn = this.renderer.drawFeature(feature, style); │ │ │ │ │ + if (drawn === false || drawn === null) { │ │ │ │ │ + this.unrenderedFeatures[feature.id] = feature │ │ │ │ │ + } else { │ │ │ │ │ + delete this.unrenderedFeatures[feature.id] │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var target = evt.target; │ │ │ │ │ - var useElement = target && target.correspondingUseElement; │ │ │ │ │ - var node = useElement ? useElement : target || evt.srcElement; │ │ │ │ │ - return node._featureId │ │ │ │ │ + eraseFeatures: function(features) { │ │ │ │ │ + this.renderer.eraseFeatures(features) │ │ │ │ │ }, │ │ │ │ │ - eraseGeometry: function(geometry, featureId) { │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPoint" || geometry.CLASS_NAME == "OpenLayers.Geometry.MultiLineString" || geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPolygon" || geometry.CLASS_NAME == "OpenLayers.Geometry.Collection") { │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - this.eraseGeometry(geometry.components[i], featureId) │ │ │ │ │ + getFeatureFromEvent: function(evt) { │ │ │ │ │ + if (!this.renderer) { │ │ │ │ │ + throw new Error("getFeatureFromEvent called on layer with no " + "renderer. This usually means you destroyed a " + "layer, but not some handler which is associated " + "with it.") │ │ │ │ │ + } │ │ │ │ │ + var feature = null; │ │ │ │ │ + var featureId = this.renderer.getFeatureIdFromEvent(evt); │ │ │ │ │ + if (featureId) { │ │ │ │ │ + if (typeof featureId === "string") { │ │ │ │ │ + feature = this.getFeatureById(featureId) │ │ │ │ │ + } else { │ │ │ │ │ + feature = featureId │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - var element = OpenLayers.Util.getElement(geometry.id); │ │ │ │ │ - if (element && element.parentNode) { │ │ │ │ │ - if (element.geometry) { │ │ │ │ │ - element.geometry.destroy(); │ │ │ │ │ - element.geometry = null │ │ │ │ │ - } │ │ │ │ │ - element.parentNode.removeChild(element); │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - this.indexer.remove(element) │ │ │ │ │ + } │ │ │ │ │ + return feature │ │ │ │ │ + }, │ │ │ │ │ + getFeatureBy: function(property, value) { │ │ │ │ │ + var feature = null; │ │ │ │ │ + for (var i = 0, len = this.features.length; i < len; ++i) { │ │ │ │ │ + if (this.features[i][property] == value) { │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return feature │ │ │ │ │ + }, │ │ │ │ │ + getFeatureById: function(featureId) { │ │ │ │ │ + return this.getFeatureBy("id", featureId) │ │ │ │ │ + }, │ │ │ │ │ + getFeatureByFid: function(featureFid) { │ │ │ │ │ + return this.getFeatureBy("fid", featureFid) │ │ │ │ │ + }, │ │ │ │ │ + getFeaturesByAttribute: function(attrName, attrValue) { │ │ │ │ │ + var i, feature, len = this.features.length, │ │ │ │ │ + foundFeatures = []; │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + if (feature && feature.attributes) { │ │ │ │ │ + if (feature.attributes[attrName] === attrValue) { │ │ │ │ │ + foundFeatures.push(feature) │ │ │ │ │ } │ │ │ │ │ - if (element._style.backgroundGraphic) { │ │ │ │ │ - var backgroundId = geometry.id + this.BACKGROUND_ID_SUFFIX; │ │ │ │ │ - var bElem = OpenLayers.Util.getElement(backgroundId); │ │ │ │ │ - if (bElem && bElem.parentNode) { │ │ │ │ │ - bElem.parentNode.removeChild(bElem) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return foundFeatures │ │ │ │ │ + }, │ │ │ │ │ + onFeatureInsert: function(feature) {}, │ │ │ │ │ + preFeatureInsert: function(feature) {}, │ │ │ │ │ + getDataExtent: function() { │ │ │ │ │ + var maxExtent = null; │ │ │ │ │ + var features = this.features; │ │ │ │ │ + if (features && features.length > 0) { │ │ │ │ │ + var geometry = null; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + geometry = features[i].geometry; │ │ │ │ │ + if (geometry) { │ │ │ │ │ + if (maxExtent === null) { │ │ │ │ │ + maxExtent = new OpenLayers.Bounds │ │ │ │ │ } │ │ │ │ │ + maxExtent.extend(geometry.getBounds()) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return maxExtent │ │ │ │ │ }, │ │ │ │ │ - nodeFactory: function(id, type) { │ │ │ │ │ - var node = OpenLayers.Util.getElement(id); │ │ │ │ │ - if (node) { │ │ │ │ │ - if (!this.nodeTypeCompare(node, type)) { │ │ │ │ │ - node.parentNode.removeChild(node); │ │ │ │ │ - node = this.nodeFactory(id, type) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Vector" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Vector.RootContainer = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + layers: null, │ │ │ │ │ + display: function() {}, │ │ │ │ │ + getFeatureFromEvent: function(evt) { │ │ │ │ │ + var layers = this.layers; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0; i < layers.length; i++) { │ │ │ │ │ + feature = layers[i].getFeatureFromEvent(evt); │ │ │ │ │ + if (feature) { │ │ │ │ │ + return feature │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - node = this.createNode(type, id) │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ }, │ │ │ │ │ - nodeTypeCompare: function(node, type) {}, │ │ │ │ │ - createNode: function(type, id) {}, │ │ │ │ │ - moveRoot: function(renderer) { │ │ │ │ │ - var root = this.root; │ │ │ │ │ - if (renderer.root.parentNode == this.rendererRoot) { │ │ │ │ │ - root = renderer.root │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.collectRoots(); │ │ │ │ │ + map.events.register("changelayer", this, this.handleChangeLayer) │ │ │ │ │ + }, │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + map.events.unregister("changelayer", this, this.handleChangeLayer); │ │ │ │ │ + this.resetRoots(); │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + collectRoots: function() { │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = 0; i < this.map.layers.length; ++i) { │ │ │ │ │ + layer = this.map.layers[i]; │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ + layer.renderer.moveRoot(this.renderer) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - root.parentNode.removeChild(root); │ │ │ │ │ - renderer.rendererRoot.appendChild(root) │ │ │ │ │ }, │ │ │ │ │ - getRenderLayerId: function() { │ │ │ │ │ - return this.root.parentNode.parentNode.id │ │ │ │ │ + resetRoots: function() { │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = 0; i < this.layers.length; ++i) { │ │ │ │ │ + layer = this.layers[i]; │ │ │ │ │ + if (this.renderer && layer.renderer.getRenderLayerId() == this.id) { │ │ │ │ │ + this.renderer.moveRoot(layer.renderer) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - isComplexSymbol: function(graphicName) { │ │ │ │ │ - return graphicName != "circle" && !!graphicName │ │ │ │ │ + handleChangeLayer: function(evt) { │ │ │ │ │ + var layer = evt.layer; │ │ │ │ │ + if (evt.property == "order" && OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ + this.resetRoots(); │ │ │ │ │ + this.collectRoots() │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.Elements" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Vector.RootContainer" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Renderer.VML = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ - xmlns: "urn:schemas-microsoft-com:vml", │ │ │ │ │ - symbolCache: {}, │ │ │ │ │ - offset: null, │ │ │ │ │ - initialize: function(containerID) { │ │ │ │ │ - if (!this.supported()) { │ │ │ │ │ - return │ │ │ │ │ +OpenLayers.Control.SelectFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + multipleKey: null, │ │ │ │ │ + toggleKey: null, │ │ │ │ │ + multiple: false, │ │ │ │ │ + clickout: true, │ │ │ │ │ + toggle: false, │ │ │ │ │ + hover: false, │ │ │ │ │ + highlightOnly: false, │ │ │ │ │ + box: false, │ │ │ │ │ + onBeforeSelect: function() {}, │ │ │ │ │ + onSelect: function() {}, │ │ │ │ │ + onUnselect: function() {}, │ │ │ │ │ + scope: null, │ │ │ │ │ + geometryTypes: null, │ │ │ │ │ + layer: null, │ │ │ │ │ + layers: null, │ │ │ │ │ + callbacks: null, │ │ │ │ │ + selectStyle: null, │ │ │ │ │ + renderIntent: "select", │ │ │ │ │ + handlers: null, │ │ │ │ │ + initialize: function(layers, options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (this.scope === null) { │ │ │ │ │ + this.scope = this │ │ │ │ │ } │ │ │ │ │ - if (!document.namespaces.olv) { │ │ │ │ │ - document.namespaces.add("olv", this.xmlns); │ │ │ │ │ - var style = document.createStyleSheet(); │ │ │ │ │ - var shapes = ["shape", "rect", "oval", "fill", "stroke", "imagedata", "group", "textbox"]; │ │ │ │ │ - for (var i = 0, len = shapes.length; i < len; i++) { │ │ │ │ │ - style.addRule("olv\\:" + shapes[i], "behavior: url(#default#VML); " + "position: absolute; display: inline-block;") │ │ │ │ │ - } │ │ │ │ │ + this.initLayer(layers); │ │ │ │ │ + var callbacks = { │ │ │ │ │ + click: this.clickFeature, │ │ │ │ │ + clickout: this.clickoutFeature │ │ │ │ │ + }; │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + callbacks.over = this.overFeature; │ │ │ │ │ + callbacks.out = this.outFeature │ │ │ │ │ + } │ │ │ │ │ + this.callbacks = OpenLayers.Util.extend(callbacks, this.callbacks); │ │ │ │ │ + this.handlers = { │ │ │ │ │ + feature: new OpenLayers.Handler.Feature(this, this.layer, this.callbacks, { │ │ │ │ │ + geometryTypes: this.geometryTypes │ │ │ │ │ + }) │ │ │ │ │ + }; │ │ │ │ │ + if (this.box) { │ │ │ │ │ + this.handlers.box = new OpenLayers.Handler.Box(this, { │ │ │ │ │ + done: this.selectBox │ │ │ │ │ + }, { │ │ │ │ │ + boxDivClassName: "olHandlerBoxSelectFeature" │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.initialize.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - supported: function() { │ │ │ │ │ - return !!document.namespaces │ │ │ │ │ }, │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var left = extent.left / resolution | 0; │ │ │ │ │ - var top = extent.top / resolution - this.size.h | 0; │ │ │ │ │ - if (resolutionChanged || !this.offset) { │ │ │ │ │ - this.offset = { │ │ │ │ │ - x: left, │ │ │ │ │ - y: top │ │ │ │ │ - }; │ │ │ │ │ - left = 0; │ │ │ │ │ - top = 0 │ │ │ │ │ + initLayer: function(layers) { │ │ │ │ │ + if (OpenLayers.Util.isArray(layers)) { │ │ │ │ │ + this.layers = layers; │ │ │ │ │ + this.layer = new OpenLayers.Layer.Vector.RootContainer(this.id + "_container", { │ │ │ │ │ + layers: layers │ │ │ │ │ + }) │ │ │ │ │ } else { │ │ │ │ │ - left = left - this.offset.x; │ │ │ │ │ - top = top - this.offset.y │ │ │ │ │ + this.layer = layers │ │ │ │ │ } │ │ │ │ │ - var org = left - this.xOffset + " " + top; │ │ │ │ │ - this.root.coordorigin = org; │ │ │ │ │ - var roots = [this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ - var root; │ │ │ │ │ - for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ - root = roots[i]; │ │ │ │ │ - var size = this.size.w + " " + this.size.h; │ │ │ │ │ - root.coordsize = size │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.active && this.layers) { │ │ │ │ │ + this.map.removeLayer(this.layer) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + if (this.layers) { │ │ │ │ │ + this.layer.destroy() │ │ │ │ │ } │ │ │ │ │ - this.root.style.flip = "y"; │ │ │ │ │ - return coordSysUnchanged │ │ │ │ │ }, │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ - var roots = [this.rendererRoot, this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ - var w = this.size.w + "px"; │ │ │ │ │ - var h = this.size.h + "px"; │ │ │ │ │ - var root; │ │ │ │ │ - for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ - root = roots[i]; │ │ │ │ │ - root.style.width = w; │ │ │ │ │ - root.style.height = h │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + if (this.layers) { │ │ │ │ │ + this.map.addLayer(this.layer) │ │ │ │ │ + } │ │ │ │ │ + this.handlers.feature.activate(); │ │ │ │ │ + if (this.box && this.handlers.box) { │ │ │ │ │ + this.handlers.box.activate() │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return OpenLayers.Control.prototype.activate.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - getNodeType: function(geometry, style) { │ │ │ │ │ - var nodeType = null; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - nodeType = "olv:rect" │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - nodeType = "olv:shape" │ │ │ │ │ - } else { │ │ │ │ │ - nodeType = "olv:oval" │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - nodeType = "olv:rect"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - case "OpenLayers.Geometry.Curve": │ │ │ │ │ - nodeType = "olv:shape"; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.handlers.feature.deactivate(); │ │ │ │ │ + if (this.handlers.box) { │ │ │ │ │ + this.handlers.box.deactivate() │ │ │ │ │ + } │ │ │ │ │ + if (this.layers) { │ │ │ │ │ + this.map.removeLayer(this.layer) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return nodeType │ │ │ │ │ + return OpenLayers.Control.prototype.deactivate.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - setStyle: function(node, style, options, geometry) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - options = options || node._options; │ │ │ │ │ - var fillColor = style.fillColor; │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - node.title = title │ │ │ │ │ + unselectAll: function(options) { │ │ │ │ │ + var layers = this.layers || [this.layer], │ │ │ │ │ + layer, feature, l, numExcept; │ │ │ │ │ + for (l = 0; l < layers.length; ++l) { │ │ │ │ │ + layer = layers[l]; │ │ │ │ │ + numExcept = 0; │ │ │ │ │ + if (layer.selectedFeatures != null) { │ │ │ │ │ + while (layer.selectedFeatures.length > numExcept) { │ │ │ │ │ + feature = layer.selectedFeatures[numExcept]; │ │ │ │ │ + if (!options || options.except != feature) { │ │ │ │ │ + this.unselect(feature) │ │ │ │ │ + } else { │ │ │ │ │ + ++numExcept │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - options.isFilled = true; │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ - var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ - node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x + xOffset | 0) + "px"; │ │ │ │ │ - node.style.top = (geometry.y / resolution - this.offset.y - (yOffset + height) | 0) + "px"; │ │ │ │ │ - node.style.width = width + "px"; │ │ │ │ │ - node.style.height = height + "px"; │ │ │ │ │ - node.style.flip = "y"; │ │ │ │ │ - fillColor = "none"; │ │ │ │ │ - options.isStroked = false │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - var cache = this.importSymbol(style.graphicName); │ │ │ │ │ - node.path = cache.path; │ │ │ │ │ - node.coordorigin = cache.left + "," + cache.bottom; │ │ │ │ │ - var size = cache.size; │ │ │ │ │ - node.coordsize = size + "," + size; │ │ │ │ │ - this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ - node.style.flip = "y" │ │ │ │ │ + }, │ │ │ │ │ + clickFeature: function(feature) { │ │ │ │ │ + if (!this.hover) { │ │ │ │ │ + var selected = OpenLayers.Util.indexOf(feature.layer.selectedFeatures, feature) > -1; │ │ │ │ │ + if (selected) { │ │ │ │ │ + if (this.toggleSelect()) { │ │ │ │ │ + this.unselect(feature) │ │ │ │ │ + } else if (!this.multipleSelect()) { │ │ │ │ │ + this.unselectAll({ │ │ │ │ │ + except: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ } else { │ │ │ │ │ - this.drawCircle(node, geometry, style.pointRadius) │ │ │ │ │ + if (!this.multipleSelect()) { │ │ │ │ │ + this.unselectAll({ │ │ │ │ │ + except: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + this.select(feature) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (options.isFilled) { │ │ │ │ │ - node.fillcolor = fillColor │ │ │ │ │ - } else { │ │ │ │ │ - node.filled = "false" │ │ │ │ │ + }, │ │ │ │ │ + multipleSelect: function() { │ │ │ │ │ + return this.multiple || this.handlers.feature.evt && this.handlers.feature.evt[this.multipleKey] │ │ │ │ │ + }, │ │ │ │ │ + toggleSelect: function() { │ │ │ │ │ + return this.toggle || this.handlers.feature.evt && this.handlers.feature.evt[this.toggleKey] │ │ │ │ │ + }, │ │ │ │ │ + clickoutFeature: function(feature) { │ │ │ │ │ + if (!this.hover && this.clickout) { │ │ │ │ │ + this.unselectAll() │ │ │ │ │ } │ │ │ │ │ - var fills = node.getElementsByTagName("fill"); │ │ │ │ │ - var fill = fills.length == 0 ? null : fills[0]; │ │ │ │ │ - if (!options.isFilled) { │ │ │ │ │ - if (fill) { │ │ │ │ │ - node.removeChild(fill) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - if (!fill) { │ │ │ │ │ - fill = this.createNode("olv:fill", node.id + "_fill") │ │ │ │ │ + }, │ │ │ │ │ + overFeature: function(feature) { │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + if (this.highlightOnly) { │ │ │ │ │ + this.highlight(feature) │ │ │ │ │ + } else if (OpenLayers.Util.indexOf(layer.selectedFeatures, feature) == -1) { │ │ │ │ │ + this.select(feature) │ │ │ │ │ } │ │ │ │ │ - fill.opacity = style.fillOpacity; │ │ │ │ │ - if (node._geometryClass === "OpenLayers.Geometry.Point" && style.externalGraphic) { │ │ │ │ │ - if (style.graphicOpacity) { │ │ │ │ │ - fill.opacity = style.graphicOpacity │ │ │ │ │ - } │ │ │ │ │ - fill.src = style.externalGraphic; │ │ │ │ │ - fill.type = "frame"; │ │ │ │ │ - if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ - fill.aspect = "atmost" │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + outFeature: function(feature) { │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + if (this.highlightOnly) { │ │ │ │ │ + if (feature._lastHighlighter == this.id) { │ │ │ │ │ + if (feature._prevHighlighter && feature._prevHighlighter != this.id) { │ │ │ │ │ + delete feature._lastHighlighter; │ │ │ │ │ + var control = this.map.getControl(feature._prevHighlighter); │ │ │ │ │ + if (control) { │ │ │ │ │ + control.highlight(feature) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this.unhighlight(feature) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - if (fill.parentNode != node) { │ │ │ │ │ - node.appendChild(fill) │ │ │ │ │ + } else { │ │ │ │ │ + this.unselect(feature) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var rotation = style.rotation; │ │ │ │ │ - if (rotation !== undefined || node._rotation !== undefined) { │ │ │ │ │ - node._rotation = rotation; │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ - fill.opacity = 0 │ │ │ │ │ - } else if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ - node.style.rotation = rotation || 0 │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ + highlight: function(feature) { │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + var cont = this.events.triggerEvent("beforefeaturehighlighted", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + feature._prevHighlighter = feature._lastHighlighter; │ │ │ │ │ + feature._lastHighlighter = this.id; │ │ │ │ │ + var style = this.selectStyle || this.renderIntent; │ │ │ │ │ + layer.drawFeature(feature, style); │ │ │ │ │ + this.events.triggerEvent("featurehighlighted", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - var strokes = node.getElementsByTagName("stroke"); │ │ │ │ │ - var stroke = strokes.length == 0 ? null : strokes[0]; │ │ │ │ │ - if (!options.isStroked) { │ │ │ │ │ - node.stroked = false; │ │ │ │ │ - if (stroke) { │ │ │ │ │ - stroke.on = false │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ + unhighlight: function(feature) { │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + if (feature._prevHighlighter == undefined) { │ │ │ │ │ + delete feature._lastHighlighter │ │ │ │ │ + } else if (feature._prevHighlighter == this.id) { │ │ │ │ │ + delete feature._prevHighlighter │ │ │ │ │ } else { │ │ │ │ │ - if (!stroke) { │ │ │ │ │ - stroke = this.createNode("olv:stroke", node.id + "_stroke"); │ │ │ │ │ - node.appendChild(stroke) │ │ │ │ │ - } │ │ │ │ │ - stroke.on = true; │ │ │ │ │ - stroke.color = style.strokeColor; │ │ │ │ │ - stroke.weight = style.strokeWidth + "px"; │ │ │ │ │ - stroke.opacity = style.strokeOpacity; │ │ │ │ │ - stroke.endcap = style.strokeLinecap == "butt" ? "flat" : style.strokeLinecap || "round"; │ │ │ │ │ - if (style.strokeDashstyle) { │ │ │ │ │ - stroke.dashstyle = this.dashStyle(style) │ │ │ │ │ - } │ │ │ │ │ + feature._lastHighlighter = feature._prevHighlighter; │ │ │ │ │ + delete feature._prevHighlighter │ │ │ │ │ } │ │ │ │ │ - if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ - node.style.cursor = style.cursor │ │ │ │ │ + layer.drawFeature(feature, feature.style || feature.layer.style || "default"); │ │ │ │ │ + this.events.triggerEvent("featureunhighlighted", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + select: function(feature) { │ │ │ │ │ + var cont = this.onBeforeSelect.call(this.scope, feature); │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + cont = layer.events.triggerEvent("beforefeatureselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + layer.selectedFeatures.push(feature); │ │ │ │ │ + this.highlight(feature); │ │ │ │ │ + if (!this.handlers.feature.lastFeature) { │ │ │ │ │ + this.handlers.feature.lastFeature = layer.selectedFeatures[0] │ │ │ │ │ + } │ │ │ │ │ + layer.events.triggerEvent("featureselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.onSelect.call(this.scope, feature) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ }, │ │ │ │ │ - graphicRotate: function(node, xOffset, yOffset, style) { │ │ │ │ │ - var style = style || node._style; │ │ │ │ │ - var rotation = style.rotation || 0; │ │ │ │ │ - var aspectRatio, size; │ │ │ │ │ - if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ - var img = new Image; │ │ │ │ │ - img.onreadystatechange = OpenLayers.Function.bind(function() { │ │ │ │ │ - if (img.readyState == "complete" || img.readyState == "interactive") { │ │ │ │ │ - aspectRatio = img.width / img.height; │ │ │ │ │ - size = Math.max(style.pointRadius * 2, style.graphicWidth || 0, style.graphicHeight || 0); │ │ │ │ │ - xOffset = xOffset * aspectRatio; │ │ │ │ │ - style.graphicWidth = size * aspectRatio; │ │ │ │ │ - style.graphicHeight = size; │ │ │ │ │ - this.graphicRotate(node, xOffset, yOffset, style) │ │ │ │ │ + unselect: function(feature) { │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + this.unhighlight(feature); │ │ │ │ │ + OpenLayers.Util.removeItem(layer.selectedFeatures, feature); │ │ │ │ │ + layer.events.triggerEvent("featureunselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.onUnselect.call(this.scope, feature) │ │ │ │ │ + }, │ │ │ │ │ + selectBox: function(position) { │ │ │ │ │ + if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ + var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.left, │ │ │ │ │ + y: position.bottom │ │ │ │ │ + }); │ │ │ │ │ + var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.right, │ │ │ │ │ + y: position.top │ │ │ │ │ + }); │ │ │ │ │ + var bounds = new OpenLayers.Bounds(minXY.lon, minXY.lat, maxXY.lon, maxXY.lat); │ │ │ │ │ + if (!this.multipleSelect()) { │ │ │ │ │ + this.unselectAll() │ │ │ │ │ + } │ │ │ │ │ + var prevMultiple = this.multiple; │ │ │ │ │ + this.multiple = true; │ │ │ │ │ + var layers = this.layers || [this.layer]; │ │ │ │ │ + this.events.triggerEvent("boxselectionstart", { │ │ │ │ │ + layers: layers │ │ │ │ │ + }); │ │ │ │ │ + var layer; │ │ │ │ │ + for (var l = 0; l < layers.length; ++l) { │ │ │ │ │ + layer = layers[l]; │ │ │ │ │ + for (var i = 0, len = layer.features.length; i < len; ++i) { │ │ │ │ │ + var feature = layer.features[i]; │ │ │ │ │ + if (!feature.getVisibility()) { │ │ │ │ │ + continue │ │ │ │ │ + } │ │ │ │ │ + if (this.geometryTypes == null || OpenLayers.Util.indexOf(this.geometryTypes, feature.geometry.CLASS_NAME) > -1) { │ │ │ │ │ + if (bounds.toGeometry().intersects(feature.geometry)) { │ │ │ │ │ + if (OpenLayers.Util.indexOf(layer.selectedFeatures, feature) == -1) { │ │ │ │ │ + this.select(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - }, this); │ │ │ │ │ - img.src = style.externalGraphic; │ │ │ │ │ - return │ │ │ │ │ - } else { │ │ │ │ │ - size = Math.max(style.graphicWidth, style.graphicHeight); │ │ │ │ │ - aspectRatio = style.graphicWidth / style.graphicHeight │ │ │ │ │ + } │ │ │ │ │ + this.multiple = prevMultiple; │ │ │ │ │ + this.events.triggerEvent("boxselectionend", { │ │ │ │ │ + layers: layers │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - var width = Math.round(style.graphicWidth || size * aspectRatio); │ │ │ │ │ - var height = Math.round(style.graphicHeight || size); │ │ │ │ │ - node.style.width = width + "px"; │ │ │ │ │ - node.style.height = height + "px"; │ │ │ │ │ - var image = document.getElementById(node.id + "_image"); │ │ │ │ │ - if (!image) { │ │ │ │ │ - image = this.createNode("olv:imagedata", node.id + "_image"); │ │ │ │ │ - node.appendChild(image) │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.handlers.feature.setMap(map); │ │ │ │ │ + if (this.box) { │ │ │ │ │ + this.handlers.box.setMap(map) │ │ │ │ │ } │ │ │ │ │ - image.style.width = width + "px"; │ │ │ │ │ - image.style.height = height + "px"; │ │ │ │ │ - image.src = style.externalGraphic; │ │ │ │ │ - image.style.filter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(" + "src='', sizingMethod='scale')"; │ │ │ │ │ - var rot = rotation * Math.PI / 180; │ │ │ │ │ - var sintheta = Math.sin(rot); │ │ │ │ │ - var costheta = Math.cos(rot); │ │ │ │ │ - var filter = "progid:DXImageTransform.Microsoft.Matrix(M11=" + costheta + ",M12=" + -sintheta + ",M21=" + sintheta + ",M22=" + costheta + ",SizingMethod='auto expand')\n"; │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ - if (opacity && opacity != 1) { │ │ │ │ │ - filter += "progid:DXImageTransform.Microsoft.BasicImage(opacity=" + opacity + ")\n" │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + setLayer: function(layers) { │ │ │ │ │ + var isActive = this.active; │ │ │ │ │ + this.unselectAll(); │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + if (this.layers) { │ │ │ │ │ + this.layer.destroy(); │ │ │ │ │ + this.layers = null │ │ │ │ │ + } │ │ │ │ │ + this.initLayer(layers); │ │ │ │ │ + this.handlers.feature.layer = this.layer; │ │ │ │ │ + if (isActive) { │ │ │ │ │ + this.activate() │ │ │ │ │ } │ │ │ │ │ - node.style.filter = filter; │ │ │ │ │ - var centerPoint = new OpenLayers.Geometry.Point(-xOffset, -yOffset); │ │ │ │ │ - var imgBox = new OpenLayers.Bounds(0, 0, width, height).toGeometry(); │ │ │ │ │ - imgBox.rotate(style.rotation, centerPoint); │ │ │ │ │ - var imgBounds = imgBox.getBounds(); │ │ │ │ │ - node.style.left = Math.round(parseInt(node.style.left) + imgBounds.left) + "px"; │ │ │ │ │ - node.style.top = Math.round(parseInt(node.style.top) - imgBounds.bottom) + "px" │ │ │ │ │ }, │ │ │ │ │ - postDraw: function(node) { │ │ │ │ │ - node.style.visibility = "visible"; │ │ │ │ │ - var fillColor = node._style.fillColor; │ │ │ │ │ - var strokeColor = node._style.strokeColor; │ │ │ │ │ - if (fillColor == "none" && node.fillcolor != fillColor) { │ │ │ │ │ - node.fillcolor = fillColor │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.SelectFeature" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Attribution = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + separator: ", ", │ │ │ │ │ + template: "${layers}", │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + removelayer: this.updateAttribution, │ │ │ │ │ + addlayer: this.updateAttribution, │ │ │ │ │ + changelayer: this.updateAttribution, │ │ │ │ │ + changebaselayer: this.updateAttribution, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + changebaselayer: this.updateAttribution, │ │ │ │ │ + changelayer: this.updateAttribution, │ │ │ │ │ + addlayer: this.updateAttribution, │ │ │ │ │ + removelayer: this.updateAttribution, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.updateAttribution(); │ │ │ │ │ + return this.div │ │ │ │ │ + }, │ │ │ │ │ + updateAttribution: function() { │ │ │ │ │ + var attributions = []; │ │ │ │ │ + if (this.map && this.map.layers) { │ │ │ │ │ + for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if (layer.attribution && layer.getVisibility()) { │ │ │ │ │ + if (OpenLayers.Util.indexOf(attributions, layer.attribution) === -1) { │ │ │ │ │ + attributions.push(layer.attribution) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.div.innerHTML = OpenLayers.String.format(this.template, { │ │ │ │ │ + layers: attributions.join(this.separator) │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - if (strokeColor == "none" && node.strokecolor != strokeColor) { │ │ │ │ │ - node.strokecolor = strokeColor │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Attribution" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Strategy = OpenLayers.Class({ │ │ │ │ │ + layer: null, │ │ │ │ │ + options: null, │ │ │ │ │ + active: null, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + autoDestroy: true, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.options = options; │ │ │ │ │ + this.active = false │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.layer = null; │ │ │ │ │ + this.options = null │ │ │ │ │ + }, │ │ │ │ │ + setLayer: function(layer) { │ │ │ │ │ + this.layer = layer │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + this.active = true; │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - setNodeDimension: function(node, geometry) { │ │ │ │ │ - var bbox = geometry.getBounds(); │ │ │ │ │ - if (bbox) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var scaledBox = new OpenLayers.Bounds((bbox.left - this.featureDx) / resolution - this.offset.x | 0, bbox.bottom / resolution - this.offset.y | 0, (bbox.right - this.featureDx) / resolution - this.offset.x | 0, bbox.top / resolution - this.offset.y | 0); │ │ │ │ │ - node.style.left = scaledBox.left + "px"; │ │ │ │ │ - node.style.top = scaledBox.top + "px"; │ │ │ │ │ - node.style.width = scaledBox.getWidth() + "px"; │ │ │ │ │ - node.style.height = scaledBox.getHeight() + "px"; │ │ │ │ │ - node.coordorigin = scaledBox.left + " " + scaledBox.top; │ │ │ │ │ - node.coordsize = scaledBox.getWidth() + " " + scaledBox.getHeight() │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.active = false; │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - dashStyle: function(style) { │ │ │ │ │ - var dash = style.strokeDashstyle; │ │ │ │ │ - switch (dash) { │ │ │ │ │ - case "solid": │ │ │ │ │ - case "dot": │ │ │ │ │ - case "dash": │ │ │ │ │ - case "dashdot": │ │ │ │ │ - case "longdash": │ │ │ │ │ - case "longdashdot": │ │ │ │ │ - return dash; │ │ │ │ │ - default: │ │ │ │ │ - var parts = dash.split(/[ ,]/); │ │ │ │ │ - if (parts.length == 2) { │ │ │ │ │ - if (1 * parts[0] >= 2 * parts[1]) { │ │ │ │ │ - return "longdash" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Filter.Spatial = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ + type: null, │ │ │ │ │ + property: null, │ │ │ │ │ + value: null, │ │ │ │ │ + distance: null, │ │ │ │ │ + distanceUnits: null, │ │ │ │ │ + evaluate: function(feature) { │ │ │ │ │ + var intersect = false; │ │ │ │ │ + switch (this.type) { │ │ │ │ │ + case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ + case OpenLayers.Filter.Spatial.INTERSECTS: │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + var geom = this.value; │ │ │ │ │ + if (this.value.CLASS_NAME == "OpenLayers.Bounds") { │ │ │ │ │ + geom = this.value.toGeometry() │ │ │ │ │ + } │ │ │ │ │ + if (feature.geometry.intersects(geom)) { │ │ │ │ │ + intersect = true │ │ │ │ │ } │ │ │ │ │ - return parts[0] == 1 || parts[1] == 1 ? "dot" : "dash" │ │ │ │ │ - } else if (parts.length == 4) { │ │ │ │ │ - return 1 * parts[0] >= 2 * parts[1] ? "longdashdot" : "dashdot" │ │ │ │ │ } │ │ │ │ │ - return "solid" │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + throw new Error("evaluate is not implemented for this filter type.") │ │ │ │ │ } │ │ │ │ │ + return intersect │ │ │ │ │ }, │ │ │ │ │ - createNode: function(type, id) { │ │ │ │ │ - var node = document.createElement(type); │ │ │ │ │ - if (id) { │ │ │ │ │ - node.id = id │ │ │ │ │ + clone: function() { │ │ │ │ │ + var options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + value: this.value && this.value.clone && this.value.clone() │ │ │ │ │ + }, this); │ │ │ │ │ + return new OpenLayers.Filter.Spatial(options) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Filter.Spatial" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Filter.Spatial.BBOX = "BBOX"; │ │ │ │ │ +OpenLayers.Filter.Spatial.INTERSECTS = "INTERSECTS"; │ │ │ │ │ +OpenLayers.Filter.Spatial.DWITHIN = "DWITHIN"; │ │ │ │ │ +OpenLayers.Filter.Spatial.WITHIN = "WITHIN"; │ │ │ │ │ +OpenLayers.Filter.Spatial.CONTAINS = "CONTAINS"; │ │ │ │ │ +OpenLayers.Strategy.BBOX = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + bounds: null, │ │ │ │ │ + resolution: null, │ │ │ │ │ + ratio: 2, │ │ │ │ │ + resFactor: null, │ │ │ │ │ + response: null, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + moveend: this.update, │ │ │ │ │ + refresh: this.update, │ │ │ │ │ + visibilitychanged: this.update, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.update() │ │ │ │ │ } │ │ │ │ │ - node.unselectable = "on"; │ │ │ │ │ - node.onselectstart = OpenLayers.Function.False; │ │ │ │ │ - return node │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - nodeTypeCompare: function(node, type) { │ │ │ │ │ - var subType = type; │ │ │ │ │ - var splitIndex = subType.indexOf(":"); │ │ │ │ │ - if (splitIndex != -1) { │ │ │ │ │ - subType = subType.substr(splitIndex + 1) │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + moveend: this.update, │ │ │ │ │ + refresh: this.update, │ │ │ │ │ + visibilitychanged: this.update, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - var nodeName = node.nodeName; │ │ │ │ │ - splitIndex = nodeName.indexOf(":"); │ │ │ │ │ - if (splitIndex != -1) { │ │ │ │ │ - nodeName = nodeName.substr(splitIndex + 1) │ │ │ │ │ + return deactivated │ │ │ │ │ + }, │ │ │ │ │ + update: function(options) { │ │ │ │ │ + var mapBounds = this.getMapBounds(); │ │ │ │ │ + if (mapBounds !== null && (options && options.force || this.layer.visibility && this.layer.calculateInRange() && this.invalidBounds(mapBounds))) { │ │ │ │ │ + this.calculateBounds(mapBounds); │ │ │ │ │ + this.resolution = this.layer.map.getResolution(); │ │ │ │ │ + this.triggerRead(options) │ │ │ │ │ } │ │ │ │ │ - return subType == nodeName │ │ │ │ │ }, │ │ │ │ │ - createRenderRoot: function() { │ │ │ │ │ - return this.nodeFactory(this.container.id + "_vmlRoot", "div") │ │ │ │ │ + getMapBounds: function() { │ │ │ │ │ + if (this.layer.map === null) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ + var bounds = this.layer.map.getExtent(); │ │ │ │ │ + if (bounds && !this.layer.projection.equals(this.layer.map.getProjectionObject())) { │ │ │ │ │ + bounds = bounds.clone().transform(this.layer.map.getProjectionObject(), this.layer.projection) │ │ │ │ │ + } │ │ │ │ │ + return bounds │ │ │ │ │ }, │ │ │ │ │ - createRoot: function(suffix) { │ │ │ │ │ - return this.nodeFactory(this.container.id + suffix, "olv:group") │ │ │ │ │ + invalidBounds: function(mapBounds) { │ │ │ │ │ + if (!mapBounds) { │ │ │ │ │ + mapBounds = this.getMapBounds() │ │ │ │ │ + } │ │ │ │ │ + var invalid = !this.bounds || !this.bounds.containsBounds(mapBounds); │ │ │ │ │ + if (!invalid && this.resFactor) { │ │ │ │ │ + var ratio = this.resolution / this.layer.map.getResolution(); │ │ │ │ │ + invalid = ratio >= this.resFactor || ratio <= 1 / this.resFactor │ │ │ │ │ + } │ │ │ │ │ + return invalid │ │ │ │ │ }, │ │ │ │ │ - drawPoint: function(node, geometry) { │ │ │ │ │ - return this.drawCircle(node, geometry, 1) │ │ │ │ │ + calculateBounds: function(mapBounds) { │ │ │ │ │ + if (!mapBounds) { │ │ │ │ │ + mapBounds = this.getMapBounds() │ │ │ │ │ + } │ │ │ │ │ + var center = mapBounds.getCenterLonLat(); │ │ │ │ │ + var dataWidth = mapBounds.getWidth() * this.ratio; │ │ │ │ │ + var dataHeight = mapBounds.getHeight() * this.ratio; │ │ │ │ │ + this.bounds = new OpenLayers.Bounds(center.lon - dataWidth / 2, center.lat - dataHeight / 2, center.lon + dataWidth / 2, center.lat + dataHeight / 2) │ │ │ │ │ }, │ │ │ │ │ - drawCircle: function(node, geometry, radius) { │ │ │ │ │ - if (!isNaN(geometry.x) && !isNaN(geometry.y)) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x | 0) - radius + "px"; │ │ │ │ │ - node.style.top = (geometry.y / resolution - this.offset.y | 0) - radius + "px"; │ │ │ │ │ - var diameter = radius * 2; │ │ │ │ │ - node.style.width = diameter + "px"; │ │ │ │ │ - node.style.height = diameter + "px"; │ │ │ │ │ - return node │ │ │ │ │ + triggerRead: function(options) { │ │ │ │ │ + if (this.response && !(options && options.noAbort === true)) { │ │ │ │ │ + this.layer.protocol.abort(this.response); │ │ │ │ │ + this.layer.events.triggerEvent("loadend") │ │ │ │ │ } │ │ │ │ │ - return false │ │ │ │ │ + var evt = { │ │ │ │ │ + filter: this.createFilter() │ │ │ │ │ + }; │ │ │ │ │ + this.layer.events.triggerEvent("loadstart", evt); │ │ │ │ │ + this.response = this.layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ + filter: evt.filter, │ │ │ │ │ + callback: this.merge, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options)) │ │ │ │ │ }, │ │ │ │ │ - drawLineString: function(node, geometry) { │ │ │ │ │ - return this.drawLine(node, geometry, false) │ │ │ │ │ + createFilter: function() { │ │ │ │ │ + var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ + value: this.bounds, │ │ │ │ │ + projection: this.layer.projection │ │ │ │ │ + }); │ │ │ │ │ + if (this.layer.filter) { │ │ │ │ │ + filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ + type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ + filters: [this.layer.filter, filter] │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + return filter │ │ │ │ │ }, │ │ │ │ │ - drawLinearRing: function(node, geometry) { │ │ │ │ │ - return this.drawLine(node, geometry, true) │ │ │ │ │ + merge: function(resp) { │ │ │ │ │ + this.layer.destroyFeatures(); │ │ │ │ │ + if (resp.success()) { │ │ │ │ │ + var features = resp.features; │ │ │ │ │ + if (features && features.length > 0) { │ │ │ │ │ + var remote = this.layer.projection; │ │ │ │ │ + var local = this.layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var geom; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + geom = features[i].geometry; │ │ │ │ │ + if (geom) { │ │ │ │ │ + geom.transform(remote, local) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.layer.addFeatures(features) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this.bounds = null │ │ │ │ │ + } │ │ │ │ │ + this.response = null; │ │ │ │ │ + this.layer.events.triggerEvent("loadend", { │ │ │ │ │ + response: resp │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - drawLine: function(node, geometry, closeLine) { │ │ │ │ │ - this.setNodeDimension(node, geometry); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var numComponents = geometry.components.length; │ │ │ │ │ - var parts = new Array(numComponents); │ │ │ │ │ - var comp, x, y; │ │ │ │ │ - for (var i = 0; i < numComponents; i++) { │ │ │ │ │ - comp = geometry.components[i]; │ │ │ │ │ - x = (comp.x - this.featureDx) / resolution - this.offset.x | 0; │ │ │ │ │ - y = comp.y / resolution - this.offset.y | 0; │ │ │ │ │ - parts[i] = " " + x + "," + y + " l " │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.BBOX" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Strategy.Fixed = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + preload: false, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + refresh: this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + if (this.layer.visibility == true || this.preload) { │ │ │ │ │ + this.load() │ │ │ │ │ + } else { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + visibilitychanged: this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var end = closeLine ? " x e" : " e"; │ │ │ │ │ - node.path = "m" + parts.join("") + end; │ │ │ │ │ - return node │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - drawPolygon: function(node, geometry) { │ │ │ │ │ - this.setNodeDimension(node, geometry); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var path = []; │ │ │ │ │ - var j, jj, points, area, first, second, i, ii, comp, pathComp, x, y; │ │ │ │ │ - for (j = 0, jj = geometry.components.length; j < jj; j++) { │ │ │ │ │ - path.push("m"); │ │ │ │ │ - points = geometry.components[j].components; │ │ │ │ │ - area = j === 0; │ │ │ │ │ - first = null; │ │ │ │ │ - second = null; │ │ │ │ │ - for (i = 0, ii = points.length; i < ii; i++) { │ │ │ │ │ - comp = points[i]; │ │ │ │ │ - x = (comp.x - this.featureDx) / resolution - this.offset.x | 0; │ │ │ │ │ - y = comp.y / resolution - this.offset.y | 0; │ │ │ │ │ - pathComp = " " + x + "," + y; │ │ │ │ │ - path.push(pathComp); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - path.push(" l") │ │ │ │ │ - } │ │ │ │ │ - if (!area) { │ │ │ │ │ - if (!first) { │ │ │ │ │ - first = pathComp │ │ │ │ │ - } else if (first != pathComp) { │ │ │ │ │ - if (!second) { │ │ │ │ │ - second = pathComp │ │ │ │ │ - } else if (second != pathComp) { │ │ │ │ │ - area = true │ │ │ │ │ - } │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + refresh: this.load, │ │ │ │ │ + visibilitychanged: this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + return deactivated │ │ │ │ │ + }, │ │ │ │ │ + load: function(options) { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + layer.events.triggerEvent("loadstart", { │ │ │ │ │ + filter: layer.filter │ │ │ │ │ + }); │ │ │ │ │ + layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: this.merge, │ │ │ │ │ + filter: layer.filter, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options)); │ │ │ │ │ + layer.events.un({ │ │ │ │ │ + visibilitychanged: this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + merge: function(resp) { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + layer.destroyFeatures(); │ │ │ │ │ + var features = resp.features; │ │ │ │ │ + if (features && features.length > 0) { │ │ │ │ │ + var remote = layer.projection; │ │ │ │ │ + var local = layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var geom; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + geom = features[i].geometry; │ │ │ │ │ + if (geom) { │ │ │ │ │ + geom.transform(remote, local) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - path.push(area ? " x " : " ") │ │ │ │ │ + layer.addFeatures(features) │ │ │ │ │ } │ │ │ │ │ - path.push("e"); │ │ │ │ │ - node.path = path.join(""); │ │ │ │ │ - return node │ │ │ │ │ + layer.events.triggerEvent("loadend", { │ │ │ │ │ + response: resp │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - drawRectangle: function(node, geometry) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x | 0) + "px"; │ │ │ │ │ - node.style.top = (geometry.y / resolution - this.offset.y | 0) + "px"; │ │ │ │ │ - node.style.width = (geometry.width / resolution | 0) + "px"; │ │ │ │ │ - node.style.height = (geometry.height / resolution | 0) + "px"; │ │ │ │ │ - return node │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Fixed" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.HTTPRequest = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ + URL_HASH_FACTOR: (Math.sqrt(5) - 1) / 2, │ │ │ │ │ + url: null, │ │ │ │ │ + params: null, │ │ │ │ │ + reproject: false, │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ + this.url = url; │ │ │ │ │ + if (!this.params) { │ │ │ │ │ + this.params = OpenLayers.Util.extend({}, params) │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - drawText: function(featureId, style, location) { │ │ │ │ │ - var label = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX, "olv:rect"); │ │ │ │ │ - var textbox = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX + "_textbox", "olv:textbox"); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - label.style.left = ((location.x - this.featureDx) / resolution - this.offset.x | 0) + "px"; │ │ │ │ │ - label.style.top = (location.y / resolution - this.offset.y | 0) + "px"; │ │ │ │ │ - label.style.flip = "y"; │ │ │ │ │ - textbox.innerText = style.label; │ │ │ │ │ - if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ - textbox.style.cursor = style.cursor │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.url = null; │ │ │ │ │ + this.params = null; │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.HTTPRequest(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - if (style.fontColor) { │ │ │ │ │ - textbox.style.color = style.fontColor │ │ │ │ │ + obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + setUrl: function(newUrl) { │ │ │ │ │ + this.url = newUrl │ │ │ │ │ + }, │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + this.params = OpenLayers.Util.extend(this.params, newParams); │ │ │ │ │ + var ret = this.redraw(); │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "params" │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - if (style.fontOpacity) { │ │ │ │ │ - textbox.style.filter = "alpha(opacity=" + style.fontOpacity * 100 + ")" │ │ │ │ │ + return ret │ │ │ │ │ + }, │ │ │ │ │ + redraw: function(force) { │ │ │ │ │ + if (force) { │ │ │ │ │ + return this.mergeNewParams({ │ │ │ │ │ + _olSalt: Math.random() │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + return OpenLayers.Layer.prototype.redraw.apply(this, []) │ │ │ │ │ } │ │ │ │ │ - if (style.fontFamily) { │ │ │ │ │ - textbox.style.fontFamily = style.fontFamily │ │ │ │ │ + }, │ │ │ │ │ + selectUrl: function(paramString, urls) { │ │ │ │ │ + var product = 1; │ │ │ │ │ + for (var i = 0, len = paramString.length; i < len; i++) { │ │ │ │ │ + product *= paramString.charCodeAt(i) * this.URL_HASH_FACTOR; │ │ │ │ │ + product -= Math.floor(product) │ │ │ │ │ } │ │ │ │ │ - if (style.fontSize) { │ │ │ │ │ - textbox.style.fontSize = style.fontSize │ │ │ │ │ + return urls[Math.floor(product * urls.length)] │ │ │ │ │ + }, │ │ │ │ │ + getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ + var url = altUrl || this.url; │ │ │ │ │ + var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ + allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ + var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(paramsString, url) │ │ │ │ │ } │ │ │ │ │ - if (style.fontWeight) { │ │ │ │ │ - textbox.style.fontWeight = style.fontWeight │ │ │ │ │ + var urlParams = OpenLayers.Util.upperCaseObject(OpenLayers.Util.getParameters(url)); │ │ │ │ │ + for (var key in allParams) { │ │ │ │ │ + if (key.toUpperCase() in urlParams) { │ │ │ │ │ + delete allParams[key] │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (style.fontStyle) { │ │ │ │ │ - textbox.style.fontStyle = style.fontStyle │ │ │ │ │ + paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + return OpenLayers.Util.urlAppend(url, paramsString) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.HTTPRequest" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Tile = OpenLayers.Class({ │ │ │ │ │ + events: null, │ │ │ │ │ + eventListeners: null, │ │ │ │ │ + id: null, │ │ │ │ │ + layer: null, │ │ │ │ │ + url: null, │ │ │ │ │ + bounds: null, │ │ │ │ │ + size: null, │ │ │ │ │ + position: null, │ │ │ │ │ + isLoading: false, │ │ │ │ │ + initialize: function(layer, position, bounds, url, size, options) { │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + this.position = position.clone(); │ │ │ │ │ + this.setBounds(bounds); │ │ │ │ │ + this.url = url; │ │ │ │ │ + if (size) { │ │ │ │ │ + this.size = size.clone() │ │ │ │ │ } │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - label._featureId = featureId; │ │ │ │ │ - textbox._featureId = featureId; │ │ │ │ │ - textbox._geometry = location; │ │ │ │ │ - textbox._geometryClass = location.CLASS_NAME │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID("Tile_"); │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.events = new OpenLayers.Events(this); │ │ │ │ │ + if (this.eventListeners instanceof Object) { │ │ │ │ │ + this.events.on(this.eventListeners) │ │ │ │ │ } │ │ │ │ │ - textbox.style.whiteSpace = "nowrap"; │ │ │ │ │ - textbox.inset = "1px,0px,0px,0px"; │ │ │ │ │ - if (!label.parentNode) { │ │ │ │ │ - label.appendChild(textbox); │ │ │ │ │ - this.textRoot.appendChild(label) │ │ │ │ │ + }, │ │ │ │ │ + unload: function() { │ │ │ │ │ + if (this.isLoading) { │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + this.events.triggerEvent("unload") │ │ │ │ │ } │ │ │ │ │ - var align = style.labelAlign || "cm"; │ │ │ │ │ - if (align.length == 1) { │ │ │ │ │ - align += "m" │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.layer = null; │ │ │ │ │ + this.bounds = null; │ │ │ │ │ + this.size = null; │ │ │ │ │ + this.position = null; │ │ │ │ │ + if (this.eventListeners) { │ │ │ │ │ + this.events.un(this.eventListeners) │ │ │ │ │ } │ │ │ │ │ - var xshift = textbox.clientWidth * OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(0, 1)]; │ │ │ │ │ - var yshift = textbox.clientHeight * OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(1, 1)]; │ │ │ │ │ - label.style.left = parseInt(label.style.left) - xshift - 1 + "px"; │ │ │ │ │ - label.style.top = parseInt(label.style.top) + yshift + "px" │ │ │ │ │ + this.events.destroy(); │ │ │ │ │ + this.eventListeners = null; │ │ │ │ │ + this.events = null │ │ │ │ │ }, │ │ │ │ │ - moveRoot: function(renderer) { │ │ │ │ │ - var layer = this.map.getLayer(renderer.container.id); │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Vector.RootContainer) { │ │ │ │ │ - layer = this.map.getLayer(this.container.id) │ │ │ │ │ + draw: function(force) { │ │ │ │ │ + if (!force) { │ │ │ │ │ + this.clear() │ │ │ │ │ } │ │ │ │ │ - layer && layer.renderer.clear(); │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.moveRoot.apply(this, arguments); │ │ │ │ │ - layer && layer.redraw() │ │ │ │ │ + var draw = this.shouldDraw(); │ │ │ │ │ + if (draw && !force && this.events.triggerEvent("beforedraw") === false) { │ │ │ │ │ + draw = null │ │ │ │ │ + } │ │ │ │ │ + return draw │ │ │ │ │ }, │ │ │ │ │ - importSymbol: function(graphicName) { │ │ │ │ │ - var id = this.container.id + "-" + graphicName; │ │ │ │ │ - var cache = this.symbolCache[id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - return cache │ │ │ │ │ + shouldDraw: function() { │ │ │ │ │ + var withinMaxExtent = false, │ │ │ │ │ + maxExtent = this.layer.maxExtent; │ │ │ │ │ + if (maxExtent) { │ │ │ │ │ + var map = this.layer.map; │ │ │ │ │ + var worldBounds = map.baseLayer.wrapDateLine && map.getMaxExtent(); │ │ │ │ │ + if (this.bounds.intersectsBounds(maxExtent, { │ │ │ │ │ + inclusive: false, │ │ │ │ │ + worldBounds: worldBounds │ │ │ │ │ + })) { │ │ │ │ │ + withinMaxExtent = true │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(graphicName + " is not a valid symbol name") │ │ │ │ │ + return withinMaxExtent || this.layer.displayOutsideMaxExtent │ │ │ │ │ + }, │ │ │ │ │ + setBounds: function(bounds) { │ │ │ │ │ + bounds = bounds.clone(); │ │ │ │ │ + if (this.layer.map.baseLayer.wrapDateLine) { │ │ │ │ │ + var worldExtent = this.layer.map.getMaxExtent(), │ │ │ │ │ + tolerance = this.layer.map.getResolution(); │ │ │ │ │ + bounds = bounds.wrapDateLine(worldExtent, { │ │ │ │ │ + leftTolerance: tolerance, │ │ │ │ │ + rightTolerance: tolerance │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - var symbolExtent = new OpenLayers.Bounds(Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ - var pathitems = ["m"]; │ │ │ │ │ - for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - var x = symbol[i]; │ │ │ │ │ - var y = symbol[i + 1]; │ │ │ │ │ - symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ - symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ - symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ - symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ - pathitems.push(x); │ │ │ │ │ - pathitems.push(y); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - pathitems.push("l") │ │ │ │ │ + this.bounds = bounds │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(bounds, position, redraw) { │ │ │ │ │ + if (redraw == null) { │ │ │ │ │ + redraw = true │ │ │ │ │ + } │ │ │ │ │ + this.setBounds(bounds); │ │ │ │ │ + this.position = position.clone(); │ │ │ │ │ + if (redraw) { │ │ │ │ │ + this.draw() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + clear: function(draw) {}, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Tile" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Tile.Image = OpenLayers.Class(OpenLayers.Tile, { │ │ │ │ │ + url: null, │ │ │ │ │ + imgDiv: null, │ │ │ │ │ + frame: null, │ │ │ │ │ + imageReloadAttempts: null, │ │ │ │ │ + layerAlphaHack: null, │ │ │ │ │ + asyncRequestId: null, │ │ │ │ │ + maxGetUrlLength: null, │ │ │ │ │ + canvasContext: null, │ │ │ │ │ + crossOriginKeyword: null, │ │ │ │ │ + initialize: function(layer, position, bounds, url, size, options) { │ │ │ │ │ + OpenLayers.Tile.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.url = url; │ │ │ │ │ + this.layerAlphaHack = this.layer.alpha && OpenLayers.Util.alphaHack(); │ │ │ │ │ + if (this.maxGetUrlLength != null || this.layer.gutter || this.layerAlphaHack) { │ │ │ │ │ + this.frame = document.createElement("div"); │ │ │ │ │ + this.frame.style.position = "absolute"; │ │ │ │ │ + this.frame.style.overflow = "hidden" │ │ │ │ │ + } │ │ │ │ │ + if (this.maxGetUrlLength != null) { │ │ │ │ │ + OpenLayers.Util.extend(this, OpenLayers.Tile.Image.IFrame) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.imgDiv) { │ │ │ │ │ + this.clear(); │ │ │ │ │ + this.imgDiv = null; │ │ │ │ │ + this.frame = null │ │ │ │ │ + } │ │ │ │ │ + this.asyncRequestId = null; │ │ │ │ │ + OpenLayers.Tile.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + var shouldDraw = OpenLayers.Tile.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (shouldDraw) { │ │ │ │ │ + if (this.layer != this.layer.map.baseLayer && this.layer.reproject) { │ │ │ │ │ + this.bounds = this.getBoundsFromBaseLayer(this.position) │ │ │ │ │ + } │ │ │ │ │ + if (this.isLoading) { │ │ │ │ │ + this._loadEvent = "reload" │ │ │ │ │ + } else { │ │ │ │ │ + this.isLoading = true; │ │ │ │ │ + this._loadEvent = "loadstart" │ │ │ │ │ } │ │ │ │ │ + this.renderTile(); │ │ │ │ │ + this.positionTile() │ │ │ │ │ + } else if (shouldDraw === false) { │ │ │ │ │ + this.unload() │ │ │ │ │ } │ │ │ │ │ - pathitems.push("x e"); │ │ │ │ │ - var path = pathitems.join(" "); │ │ │ │ │ - var diff = (symbolExtent.getWidth() - symbolExtent.getHeight()) / 2; │ │ │ │ │ - if (diff > 0) { │ │ │ │ │ - symbolExtent.bottom = symbolExtent.bottom - diff; │ │ │ │ │ - symbolExtent.top = symbolExtent.top + diff │ │ │ │ │ + return shouldDraw │ │ │ │ │ + }, │ │ │ │ │ + renderTile: function() { │ │ │ │ │ + if (this.layer.async) { │ │ │ │ │ + var id = this.asyncRequestId = (this.asyncRequestId || 0) + 1; │ │ │ │ │ + this.layer.getURLasync(this.bounds, function(url) { │ │ │ │ │ + if (id == this.asyncRequestId) { │ │ │ │ │ + this.url = url; │ │ │ │ │ + this.initImage() │ │ │ │ │ + } │ │ │ │ │ + }, this) │ │ │ │ │ } else { │ │ │ │ │ - symbolExtent.left = symbolExtent.left + diff; │ │ │ │ │ - symbolExtent.right = symbolExtent.right - diff │ │ │ │ │ + this.url = this.layer.getURL(this.bounds); │ │ │ │ │ + this.initImage() │ │ │ │ │ } │ │ │ │ │ - cache = { │ │ │ │ │ - path: path, │ │ │ │ │ - size: symbolExtent.getWidth(), │ │ │ │ │ - left: symbolExtent.left, │ │ │ │ │ - bottom: symbolExtent.bottom │ │ │ │ │ - }; │ │ │ │ │ - this.symbolCache[id] = cache; │ │ │ │ │ - return cache │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.VML" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Renderer.VML.LABEL_SHIFT = { │ │ │ │ │ - l: 0, │ │ │ │ │ - c: .5, │ │ │ │ │ - r: 1, │ │ │ │ │ - t: 0, │ │ │ │ │ - m: .5, │ │ │ │ │ - b: 1 │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.SVG = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ - xmlns: "http://www.w3.org/2000/svg", │ │ │ │ │ - xlinkns: "http://www.w3.org/1999/xlink", │ │ │ │ │ - MAX_PIXEL: 15e3, │ │ │ │ │ - translationParameters: null, │ │ │ │ │ - symbolMetrics: null, │ │ │ │ │ - initialize: function(containerID) { │ │ │ │ │ - if (!this.supported()) { │ │ │ │ │ - return │ │ │ │ │ + positionTile: function() { │ │ │ │ │ + var style = this.getTile().style, │ │ │ │ │ + size = this.frame ? this.size : this.layer.getImageSize(this.bounds), │ │ │ │ │ + ratio = 1; │ │ │ │ │ + if (this.layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ + ratio = this.layer.getServerResolution() / this.layer.map.getResolution() │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.translationParameters = { │ │ │ │ │ - x: 0, │ │ │ │ │ - y: 0 │ │ │ │ │ - }; │ │ │ │ │ - this.symbolMetrics = {} │ │ │ │ │ + style.left = this.position.x + "px"; │ │ │ │ │ + style.top = this.position.y + "px"; │ │ │ │ │ + style.width = Math.round(ratio * size.w) + "px"; │ │ │ │ │ + style.height = Math.round(ratio * size.h) + "px" │ │ │ │ │ }, │ │ │ │ │ - supported: function() { │ │ │ │ │ - var svgFeature = "http://www.w3.org/TR/SVG11/feature#"; │ │ │ │ │ - return document.implementation && (document.implementation.hasFeature("org.w3c.svg", "1.0") || document.implementation.hasFeature(svgFeature + "SVG", "1.1") || document.implementation.hasFeature(svgFeature + "BasicStructure", "1.1")) │ │ │ │ │ + clear: function() { │ │ │ │ │ + OpenLayers.Tile.prototype.clear.apply(this, arguments); │ │ │ │ │ + var img = this.imgDiv; │ │ │ │ │ + if (img) { │ │ │ │ │ + var tile = this.getTile(); │ │ │ │ │ + if (tile.parentNode === this.layer.div) { │ │ │ │ │ + this.layer.div.removeChild(tile) │ │ │ │ │ + } │ │ │ │ │ + this.setImgSrc(); │ │ │ │ │ + if (this.layerAlphaHack === true) { │ │ │ │ │ + img.style.filter = "" │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.removeClass(img, "olImageLoadError") │ │ │ │ │ + } │ │ │ │ │ + this.canvasContext = null │ │ │ │ │ }, │ │ │ │ │ - inValidRange: function(x, y, xyOnly) { │ │ │ │ │ - var left = x + (xyOnly ? 0 : this.translationParameters.x); │ │ │ │ │ - var top = y + (xyOnly ? 0 : this.translationParameters.y); │ │ │ │ │ - return left >= -this.MAX_PIXEL && left <= this.MAX_PIXEL && top >= -this.MAX_PIXEL && top <= this.MAX_PIXEL │ │ │ │ │ + getImage: function() { │ │ │ │ │ + if (!this.imgDiv) { │ │ │ │ │ + this.imgDiv = OpenLayers.Tile.Image.IMAGE.cloneNode(false); │ │ │ │ │ + var style = this.imgDiv.style; │ │ │ │ │ + if (this.frame) { │ │ │ │ │ + var left = 0, │ │ │ │ │ + top = 0; │ │ │ │ │ + if (this.layer.gutter) { │ │ │ │ │ + left = this.layer.gutter / this.layer.tileSize.w * 100; │ │ │ │ │ + top = this.layer.gutter / this.layer.tileSize.h * 100 │ │ │ │ │ + } │ │ │ │ │ + style.left = -left + "%"; │ │ │ │ │ + style.top = -top + "%"; │ │ │ │ │ + style.width = 2 * left + 100 + "%"; │ │ │ │ │ + style.height = 2 * top + 100 + "%" │ │ │ │ │ + } │ │ │ │ │ + style.visibility = "hidden"; │ │ │ │ │ + style.opacity = 0; │ │ │ │ │ + if (this.layer.opacity < 1) { │ │ │ │ │ + style.filter = "alpha(opacity=" + this.layer.opacity * 100 + ")" │ │ │ │ │ + } │ │ │ │ │ + style.position = "absolute"; │ │ │ │ │ + if (this.layerAlphaHack) { │ │ │ │ │ + style.paddingTop = style.height; │ │ │ │ │ + style.height = "0"; │ │ │ │ │ + style.width = "100%" │ │ │ │ │ + } │ │ │ │ │ + if (this.frame) { │ │ │ │ │ + this.frame.appendChild(this.imgDiv) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return this.imgDiv │ │ │ │ │ }, │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - var resolution = this.getResolution(), │ │ │ │ │ - left = -extent.left / resolution, │ │ │ │ │ - top = extent.top / resolution; │ │ │ │ │ - if (resolutionChanged) { │ │ │ │ │ - this.left = left; │ │ │ │ │ - this.top = top; │ │ │ │ │ - var extentString = "0 0 " + this.size.w + " " + this.size.h; │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "viewBox", extentString); │ │ │ │ │ - this.translate(this.xOffset, 0); │ │ │ │ │ - return true │ │ │ │ │ + setImage: function(img) { │ │ │ │ │ + this.imgDiv = img │ │ │ │ │ + }, │ │ │ │ │ + initImage: function() { │ │ │ │ │ + if (!this.url && !this.imgDiv) { │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("beforeload"); │ │ │ │ │ + this.layer.div.appendChild(this.getTile()); │ │ │ │ │ + this.events.triggerEvent(this._loadEvent); │ │ │ │ │ + var img = this.getImage(); │ │ │ │ │ + var src = img.getAttribute("src") || ""; │ │ │ │ │ + if (this.url && OpenLayers.Util.isEquivalentUrl(src, this.url)) { │ │ │ │ │ + this._loadTimeout = window.setTimeout(OpenLayers.Function.bind(this.onImageLoad, this), 0) │ │ │ │ │ } else { │ │ │ │ │ - var inRange = this.translate(left - this.left + this.xOffset, top - this.top); │ │ │ │ │ - if (!inRange) { │ │ │ │ │ - this.setExtent(extent, true) │ │ │ │ │ + this.stopLoading(); │ │ │ │ │ + if (this.crossOriginKeyword) { │ │ │ │ │ + img.removeAttribute("crossorigin") │ │ │ │ │ } │ │ │ │ │ - return coordSysUnchanged && inRange │ │ │ │ │ + OpenLayers.Event.observe(img, "load", OpenLayers.Function.bind(this.onImageLoad, this)); │ │ │ │ │ + OpenLayers.Event.observe(img, "error", OpenLayers.Function.bind(this.onImageError, this)); │ │ │ │ │ + this.imageReloadAttempts = 0; │ │ │ │ │ + this.setImgSrc(this.url) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - translate: function(x, y) { │ │ │ │ │ - if (!this.inValidRange(x, y, true)) { │ │ │ │ │ - return false │ │ │ │ │ + setImgSrc: function(url) { │ │ │ │ │ + var img = this.imgDiv; │ │ │ │ │ + if (url) { │ │ │ │ │ + img.style.visibility = "hidden"; │ │ │ │ │ + img.style.opacity = 0; │ │ │ │ │ + if (this.crossOriginKeyword) { │ │ │ │ │ + if (url.substr(0, 5) !== "data:") { │ │ │ │ │ + img.setAttribute("crossorigin", this.crossOriginKeyword) │ │ │ │ │ + } else { │ │ │ │ │ + img.removeAttribute("crossorigin") │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + img.src = url │ │ │ │ │ } else { │ │ │ │ │ - var transformString = ""; │ │ │ │ │ - if (x || y) { │ │ │ │ │ - transformString = "translate(" + x + "," + y + ")" │ │ │ │ │ + this.stopLoading(); │ │ │ │ │ + this.imgDiv = null; │ │ │ │ │ + if (img.parentNode) { │ │ │ │ │ + img.parentNode.removeChild(img) │ │ │ │ │ } │ │ │ │ │ - this.root.setAttributeNS(null, "transform", transformString); │ │ │ │ │ - this.translationParameters = { │ │ │ │ │ - x: x, │ │ │ │ │ - y: y │ │ │ │ │ - }; │ │ │ │ │ - return true │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "width", this.size.w); │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "height", this.size.h) │ │ │ │ │ + getTile: function() { │ │ │ │ │ + return this.frame ? this.frame : this.getImage() │ │ │ │ │ }, │ │ │ │ │ - getNodeType: function(geometry, style) { │ │ │ │ │ - var nodeType = null; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - nodeType = "image" │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - nodeType = "svg" │ │ │ │ │ - } else { │ │ │ │ │ - nodeType = "circle" │ │ │ │ │ + createBackBuffer: function() { │ │ │ │ │ + if (!this.imgDiv || this.isLoading) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var backBuffer; │ │ │ │ │ + if (this.frame) { │ │ │ │ │ + backBuffer = this.frame.cloneNode(false); │ │ │ │ │ + backBuffer.appendChild(this.imgDiv) │ │ │ │ │ + } else { │ │ │ │ │ + backBuffer = this.imgDiv │ │ │ │ │ + } │ │ │ │ │ + this.imgDiv = null; │ │ │ │ │ + return backBuffer │ │ │ │ │ + }, │ │ │ │ │ + onImageLoad: function() { │ │ │ │ │ + var img = this.imgDiv; │ │ │ │ │ + this.stopLoading(); │ │ │ │ │ + img.style.visibility = "inherit"; │ │ │ │ │ + img.style.opacity = this.layer.opacity; │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + this.canvasContext = null; │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + if (this.layerAlphaHack === true) { │ │ │ │ │ + img.style.filter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(src='" + img.src + "', sizingMethod='scale')" │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + onImageError: function() { │ │ │ │ │ + var img = this.imgDiv; │ │ │ │ │ + if (img.src != null) { │ │ │ │ │ + this.imageReloadAttempts++; │ │ │ │ │ + if (this.imageReloadAttempts <= OpenLayers.IMAGE_RELOAD_ATTEMPTS) { │ │ │ │ │ + this.setImgSrc(this.layer.getURL(this.bounds)) │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Element.addClass(img, "olImageLoadError"); │ │ │ │ │ + this.events.triggerEvent("loaderror"); │ │ │ │ │ + this.onImageLoad() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + stopLoading: function() { │ │ │ │ │ + OpenLayers.Event.stopObservingElement(this.imgDiv); │ │ │ │ │ + window.clearTimeout(this._loadTimeout); │ │ │ │ │ + delete this._loadTimeout │ │ │ │ │ + }, │ │ │ │ │ + getCanvasContext: function() { │ │ │ │ │ + if (OpenLayers.CANVAS_SUPPORTED && this.imgDiv && !this.isLoading) { │ │ │ │ │ + if (!this.canvasContext) { │ │ │ │ │ + var canvas = document.createElement("canvas"); │ │ │ │ │ + canvas.width = this.size.w; │ │ │ │ │ + canvas.height = this.size.h; │ │ │ │ │ + this.canvasContext = canvas.getContext("2d"); │ │ │ │ │ + this.canvasContext.drawImage(this.imgDiv, 0, 0) │ │ │ │ │ + } │ │ │ │ │ + return this.canvasContext │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Tile.Image" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Tile.Image.IMAGE = function() { │ │ │ │ │ + var img = new Image; │ │ │ │ │ + img.className = "olTileImage"; │ │ │ │ │ + img.galleryImg = "no"; │ │ │ │ │ + return img │ │ │ │ │ +}(); │ │ │ │ │ +OpenLayers.Layer.Grid = OpenLayers.Class(OpenLayers.Layer.HTTPRequest, { │ │ │ │ │ + tileSize: null, │ │ │ │ │ + tileOriginCorner: "bl", │ │ │ │ │ + tileOrigin: null, │ │ │ │ │ + tileOptions: null, │ │ │ │ │ + tileClass: OpenLayers.Tile.Image, │ │ │ │ │ + grid: null, │ │ │ │ │ + singleTile: false, │ │ │ │ │ + ratio: 1.5, │ │ │ │ │ + buffer: 0, │ │ │ │ │ + transitionEffect: "resize", │ │ │ │ │ + numLoadingTiles: 0, │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + loading: false, │ │ │ │ │ + backBuffer: null, │ │ │ │ │ + gridResolution: null, │ │ │ │ │ + backBufferResolution: null, │ │ │ │ │ + backBufferLonLat: null, │ │ │ │ │ + backBufferTimerId: null, │ │ │ │ │ + removeBackBufferDelay: null, │ │ │ │ │ + className: null, │ │ │ │ │ + gridLayout: null, │ │ │ │ │ + rowSign: null, │ │ │ │ │ + transitionendEvents: ["transitionend", "webkitTransitionEnd", "otransitionend", "oTransitionEnd"], │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.grid = []; │ │ │ │ │ + this._removeBackBuffer = OpenLayers.Function.bind(this.removeBackBuffer, this); │ │ │ │ │ + this.initProperties(); │ │ │ │ │ + this.rowSign = this.tileOriginCorner.substr(0, 1) === "t" ? 1 : -1 │ │ │ │ │ + }, │ │ │ │ │ + initProperties: function() { │ │ │ │ │ + if (this.options.removeBackBufferDelay === undefined) { │ │ │ │ │ + this.removeBackBufferDelay = this.singleTile ? 0 : 2500 │ │ │ │ │ + } │ │ │ │ │ + if (this.options.className === undefined) { │ │ │ │ │ + this.className = this.singleTile ? "olLayerGridSingleTile" : "olLayerGrid" │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.setMap.call(this, map); │ │ │ │ │ + OpenLayers.Element.addClass(this.div, this.className) │ │ │ │ │ + }, │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + this.removeBackBuffer() │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.removeBackBuffer(); │ │ │ │ │ + this.clearGrid(); │ │ │ │ │ + this.grid = null; │ │ │ │ │ + this.tileSize = null; │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + clearGrid: function() { │ │ │ │ │ + if (this.grid) { │ │ │ │ │ + for (var iRow = 0, len = this.grid.length; iRow < len; iRow++) { │ │ │ │ │ + var row = this.grid[iRow]; │ │ │ │ │ + for (var iCol = 0, clen = row.length; iCol < clen; iCol++) { │ │ │ │ │ + var tile = row[iCol]; │ │ │ │ │ + this.destroyTile(tile) │ │ │ │ │ } │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - nodeType = "rect"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - nodeType = "polyline"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - nodeType = "polygon"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - case "OpenLayers.Geometry.Curve": │ │ │ │ │ - nodeType = "path"; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break │ │ │ │ │ + } │ │ │ │ │ + this.grid = []; │ │ │ │ │ + this.gridResolution = null; │ │ │ │ │ + this.gridLayout = null │ │ │ │ │ } │ │ │ │ │ - return nodeType │ │ │ │ │ }, │ │ │ │ │ - setStyle: function(node, style, options) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - options = options || node._options; │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - node.setAttributeNS(null, "title", title); │ │ │ │ │ - var titleNode = node.getElementsByTagName("title"); │ │ │ │ │ - if (titleNode.length > 0) { │ │ │ │ │ - titleNode[0].firstChild.textContent = title │ │ │ │ │ + addOptions: function(newOptions, reinitialize) { │ │ │ │ │ + var singleTileChanged = newOptions.singleTile !== undefined && newOptions.singleTile !== this.singleTile; │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.addOptions.apply(this, arguments); │ │ │ │ │ + if (this.map && singleTileChanged) { │ │ │ │ │ + this.initProperties(); │ │ │ │ │ + this.clearGrid(); │ │ │ │ │ + this.tileSize = this.options.tileSize; │ │ │ │ │ + this.setTileSize(); │ │ │ │ │ + this.moveTo(null, true) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Grid(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.HTTPRequest.prototype.clone.apply(this, [obj]); │ │ │ │ │ + if (this.tileSize != null) { │ │ │ │ │ + obj.tileSize = this.tileSize.clone() │ │ │ │ │ + } │ │ │ │ │ + obj.grid = []; │ │ │ │ │ + obj.gridResolution = null; │ │ │ │ │ + obj.backBuffer = null; │ │ │ │ │ + obj.backBufferTimerId = null; │ │ │ │ │ + obj.loading = false; │ │ │ │ │ + obj.numLoadingTiles = 0; │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + bounds = bounds || this.map.getExtent(); │ │ │ │ │ + if (bounds != null) { │ │ │ │ │ + var forceReTile = !this.grid.length || zoomChanged; │ │ │ │ │ + var tilesBounds = this.getTilesBounds(); │ │ │ │ │ + var resolution = this.map.getResolution(); │ │ │ │ │ + var serverResolution = this.getServerResolution(resolution); │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + if (forceReTile || !dragging && !tilesBounds.containsBounds(bounds)) { │ │ │ │ │ + if (zoomChanged && this.transitionEffect !== "resize") { │ │ │ │ │ + this.removeBackBuffer() │ │ │ │ │ + } │ │ │ │ │ + if (!zoomChanged || this.transitionEffect === "resize") { │ │ │ │ │ + this.applyBackBuffer(resolution) │ │ │ │ │ + } │ │ │ │ │ + this.initSingleTile(bounds) │ │ │ │ │ + } │ │ │ │ │ } else { │ │ │ │ │ - var label = this.nodeFactory(null, "title"); │ │ │ │ │ - label.textContent = title; │ │ │ │ │ - node.appendChild(label) │ │ │ │ │ + forceReTile = forceReTile || !tilesBounds.intersectsBounds(bounds, { │ │ │ │ │ + worldBounds: this.map.baseLayer.wrapDateLine && this.map.getMaxExtent() │ │ │ │ │ + }); │ │ │ │ │ + if (forceReTile) { │ │ │ │ │ + if (zoomChanged && (this.transitionEffect === "resize" || this.gridResolution === resolution)) { │ │ │ │ │ + this.applyBackBuffer(resolution) │ │ │ │ │ + } │ │ │ │ │ + this.initGriddedTiles(bounds) │ │ │ │ │ + } else { │ │ │ │ │ + this.moveGriddedTiles() │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var r = parseFloat(node.getAttributeNS(null, "r")); │ │ │ │ │ - var widthFactor = 1; │ │ │ │ │ - var pos; │ │ │ │ │ - if (node._geometryClass == "OpenLayers.Geometry.Point" && r) { │ │ │ │ │ - node.style.visibility = ""; │ │ │ │ │ - if (style.graphic === false) { │ │ │ │ │ - node.style.visibility = "hidden" │ │ │ │ │ - } else if (style.externalGraphic) { │ │ │ │ │ - pos = this.getPosition(node); │ │ │ │ │ - if (style.graphicWidth && style.graphicHeight) { │ │ │ │ │ - node.setAttributeNS(null, "preserveAspectRatio", "none") │ │ │ │ │ + }, │ │ │ │ │ + getTileData: function(loc) { │ │ │ │ │ + var data = null, │ │ │ │ │ + x = loc.lon, │ │ │ │ │ + y = loc.lat, │ │ │ │ │ + numRows = this.grid.length; │ │ │ │ │ + if (this.map && numRows) { │ │ │ │ │ + var res = this.map.getResolution(), │ │ │ │ │ + tileWidth = this.tileSize.w, │ │ │ │ │ + tileHeight = this.tileSize.h, │ │ │ │ │ + bounds = this.grid[0][0].bounds, │ │ │ │ │ + left = bounds.left, │ │ │ │ │ + top = bounds.top; │ │ │ │ │ + if (x < left) { │ │ │ │ │ + if (this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + var worldWidth = this.map.getMaxExtent().getWidth(); │ │ │ │ │ + var worldsAway = Math.ceil((left - x) / worldWidth); │ │ │ │ │ + x += worldWidth * worldsAway │ │ │ │ │ } │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ - var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ - node.setAttributeNS(null, "x", (pos.x + xOffset).toFixed()); │ │ │ │ │ - node.setAttributeNS(null, "y", (pos.y + yOffset).toFixed()); │ │ │ │ │ - node.setAttributeNS(null, "width", width); │ │ │ │ │ - node.setAttributeNS(null, "height", height); │ │ │ │ │ - node.setAttributeNS(this.xlinkns, "xlink:href", style.externalGraphic); │ │ │ │ │ - node.setAttributeNS(null, "style", "opacity: " + opacity); │ │ │ │ │ - node.onclick = OpenLayers.Event.preventDefault │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - var offset = style.pointRadius * 3; │ │ │ │ │ - var size = offset * 2; │ │ │ │ │ - var src = this.importSymbol(style.graphicName); │ │ │ │ │ - pos = this.getPosition(node); │ │ │ │ │ - widthFactor = this.symbolMetrics[src.id][0] * 3 / size; │ │ │ │ │ - var parent = node.parentNode; │ │ │ │ │ - var nextSibling = node.nextSibling; │ │ │ │ │ - if (parent) { │ │ │ │ │ - parent.removeChild(node) │ │ │ │ │ + } │ │ │ │ │ + var dtx = (x - left) / (res * tileWidth); │ │ │ │ │ + var dty = (top - y) / (res * tileHeight); │ │ │ │ │ + var col = Math.floor(dtx); │ │ │ │ │ + var row = Math.floor(dty); │ │ │ │ │ + if (row >= 0 && row < numRows) { │ │ │ │ │ + var tile = this.grid[row][col]; │ │ │ │ │ + if (tile) { │ │ │ │ │ + data = { │ │ │ │ │ + tile: tile, │ │ │ │ │ + i: Math.floor((dtx - col) * tileWidth), │ │ │ │ │ + j: Math.floor((dty - row) * tileHeight) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - node.firstChild && node.removeChild(node.firstChild); │ │ │ │ │ - node.appendChild(src.firstChild.cloneNode(true)); │ │ │ │ │ - node.setAttributeNS(null, "viewBox", src.getAttributeNS(null, "viewBox")); │ │ │ │ │ - node.setAttributeNS(null, "width", size); │ │ │ │ │ - node.setAttributeNS(null, "height", size); │ │ │ │ │ - node.setAttributeNS(null, "x", pos.x - offset); │ │ │ │ │ - node.setAttributeNS(null, "y", pos.y - offset); │ │ │ │ │ - if (nextSibling) { │ │ │ │ │ - parent.insertBefore(node, nextSibling) │ │ │ │ │ - } else if (parent) { │ │ │ │ │ - parent.appendChild(node) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return data │ │ │ │ │ + }, │ │ │ │ │ + destroyTile: function(tile) { │ │ │ │ │ + this.removeTileMonitoringHooks(tile); │ │ │ │ │ + tile.destroy() │ │ │ │ │ + }, │ │ │ │ │ + getServerResolution: function(resolution) { │ │ │ │ │ + var distance = Number.POSITIVE_INFINITY; │ │ │ │ │ + resolution = resolution || this.map.getResolution(); │ │ │ │ │ + if (this.serverResolutions && OpenLayers.Util.indexOf(this.serverResolutions, resolution) === -1) { │ │ │ │ │ + var i, newDistance, newResolution, serverResolution; │ │ │ │ │ + for (i = this.serverResolutions.length - 1; i >= 0; i--) { │ │ │ │ │ + newResolution = this.serverResolutions[i]; │ │ │ │ │ + newDistance = Math.abs(newResolution - resolution); │ │ │ │ │ + if (newDistance > distance) { │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ + distance = newDistance; │ │ │ │ │ + serverResolution = newResolution │ │ │ │ │ + } │ │ │ │ │ + resolution = serverResolution │ │ │ │ │ + } │ │ │ │ │ + return resolution │ │ │ │ │ + }, │ │ │ │ │ + getServerZoom: function() { │ │ │ │ │ + var resolution = this.getServerResolution(); │ │ │ │ │ + return this.serverResolutions ? OpenLayers.Util.indexOf(this.serverResolutions, resolution) : this.map.getZoomForResolution(resolution) + (this.zoomOffset || 0) │ │ │ │ │ + }, │ │ │ │ │ + applyBackBuffer: function(resolution) { │ │ │ │ │ + if (this.backBufferTimerId !== null) { │ │ │ │ │ + this.removeBackBuffer() │ │ │ │ │ + } │ │ │ │ │ + var backBuffer = this.backBuffer; │ │ │ │ │ + if (!backBuffer) { │ │ │ │ │ + backBuffer = this.createBackBuffer(); │ │ │ │ │ + if (!backBuffer) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (resolution === this.gridResolution) { │ │ │ │ │ + this.div.insertBefore(backBuffer, this.div.firstChild) │ │ │ │ │ } else { │ │ │ │ │ - node.setAttributeNS(null, "r", style.pointRadius) │ │ │ │ │ + this.map.baseLayer.div.parentNode.insertBefore(backBuffer, this.map.baseLayer.div) │ │ │ │ │ } │ │ │ │ │ - var rotation = style.rotation; │ │ │ │ │ - if ((rotation !== undefined || node._rotation !== undefined) && pos) { │ │ │ │ │ - node._rotation = rotation; │ │ │ │ │ - rotation |= 0; │ │ │ │ │ - if (node.nodeName !== "svg") { │ │ │ │ │ - node.setAttributeNS(null, "transform", "rotate(" + rotation + " " + pos.x + " " + pos.y + ")") │ │ │ │ │ - } else { │ │ │ │ │ - var metrics = this.symbolMetrics[src.id]; │ │ │ │ │ - node.firstChild.setAttributeNS(null, "transform", "rotate(" + rotation + " " + metrics[1] + " " + metrics[2] + ")") │ │ │ │ │ + this.backBuffer = backBuffer; │ │ │ │ │ + var topLeftTileBounds = this.grid[0][0].bounds; │ │ │ │ │ + this.backBufferLonLat = { │ │ │ │ │ + lon: topLeftTileBounds.left, │ │ │ │ │ + lat: topLeftTileBounds.top │ │ │ │ │ + }; │ │ │ │ │ + this.backBufferResolution = this.gridResolution │ │ │ │ │ + } │ │ │ │ │ + var ratio = this.backBufferResolution / resolution; │ │ │ │ │ + var tiles = backBuffer.childNodes, │ │ │ │ │ + tile; │ │ │ │ │ + for (var i = tiles.length - 1; i >= 0; --i) { │ │ │ │ │ + tile = tiles[i]; │ │ │ │ │ + tile.style.top = (ratio * tile._i * tile._h | 0) + "px"; │ │ │ │ │ + tile.style.left = (ratio * tile._j * tile._w | 0) + "px"; │ │ │ │ │ + tile.style.width = Math.round(ratio * tile._w) + "px"; │ │ │ │ │ + tile.style.height = Math.round(ratio * tile._h) + "px" │ │ │ │ │ + } │ │ │ │ │ + var position = this.getViewPortPxFromLonLat(this.backBufferLonLat, resolution); │ │ │ │ │ + var leftOffset = this.map.layerContainerOriginPx.x; │ │ │ │ │ + var topOffset = this.map.layerContainerOriginPx.y; │ │ │ │ │ + backBuffer.style.left = Math.round(position.x - leftOffset) + "px"; │ │ │ │ │ + backBuffer.style.top = Math.round(position.y - topOffset) + "px" │ │ │ │ │ + }, │ │ │ │ │ + createBackBuffer: function() { │ │ │ │ │ + var backBuffer; │ │ │ │ │ + if (this.grid.length > 0) { │ │ │ │ │ + backBuffer = document.createElement("div"); │ │ │ │ │ + backBuffer.id = this.div.id + "_bb"; │ │ │ │ │ + backBuffer.className = "olBackBuffer"; │ │ │ │ │ + backBuffer.style.position = "absolute"; │ │ │ │ │ + var map = this.map; │ │ │ │ │ + backBuffer.style.zIndex = this.transitionEffect === "resize" ? this.getZIndex() - 1 : map.Z_INDEX_BASE.BaseLayer - (map.getNumLayers() - map.getLayerIndex(this)); │ │ │ │ │ + for (var i = 0, lenI = this.grid.length; i < lenI; i++) { │ │ │ │ │ + for (var j = 0, lenJ = this.grid[i].length; j < lenJ; j++) { │ │ │ │ │ + var tile = this.grid[i][j], │ │ │ │ │ + markup = this.grid[i][j].createBackBuffer(); │ │ │ │ │ + if (markup) { │ │ │ │ │ + markup._i = i; │ │ │ │ │ + markup._j = j; │ │ │ │ │ + markup._w = tile.size.w; │ │ │ │ │ + markup._h = tile.size.h; │ │ │ │ │ + markup.id = tile.id + "_bb"; │ │ │ │ │ + backBuffer.appendChild(markup) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (options.isFilled) { │ │ │ │ │ - node.setAttributeNS(null, "fill", style.fillColor); │ │ │ │ │ - node.setAttributeNS(null, "fill-opacity", style.fillOpacity) │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "fill", "none") │ │ │ │ │ + return backBuffer │ │ │ │ │ + }, │ │ │ │ │ + removeBackBuffer: function() { │ │ │ │ │ + if (this._transitionElement) { │ │ │ │ │ + for (var i = this.transitionendEvents.length - 1; i >= 0; --i) { │ │ │ │ │ + OpenLayers.Event.stopObserving(this._transitionElement, this.transitionendEvents[i], this._removeBackBuffer) │ │ │ │ │ + } │ │ │ │ │ + delete this._transitionElement │ │ │ │ │ } │ │ │ │ │ - if (options.isStroked) { │ │ │ │ │ - node.setAttributeNS(null, "stroke", style.strokeColor); │ │ │ │ │ - node.setAttributeNS(null, "stroke-opacity", style.strokeOpacity); │ │ │ │ │ - node.setAttributeNS(null, "stroke-width", style.strokeWidth * widthFactor); │ │ │ │ │ - node.setAttributeNS(null, "stroke-linecap", style.strokeLinecap || "round"); │ │ │ │ │ - node.setAttributeNS(null, "stroke-linejoin", "round"); │ │ │ │ │ - style.strokeDashstyle && node.setAttributeNS(null, "stroke-dasharray", this.dashStyle(style, widthFactor)) │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "stroke", "none") │ │ │ │ │ + if (this.backBuffer) { │ │ │ │ │ + if (this.backBuffer.parentNode) { │ │ │ │ │ + this.backBuffer.parentNode.removeChild(this.backBuffer) │ │ │ │ │ + } │ │ │ │ │ + this.backBuffer = null; │ │ │ │ │ + this.backBufferResolution = null; │ │ │ │ │ + if (this.backBufferTimerId !== null) { │ │ │ │ │ + window.clearTimeout(this.backBufferTimerId); │ │ │ │ │ + this.backBufferTimerId = null │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (style.pointerEvents) { │ │ │ │ │ - node.setAttributeNS(null, "pointer-events", style.pointerEvents) │ │ │ │ │ + }, │ │ │ │ │ + moveByPx: function(dx, dy) { │ │ │ │ │ + if (!this.singleTile) { │ │ │ │ │ + this.moveGriddedTiles() │ │ │ │ │ } │ │ │ │ │ - if (style.cursor != null) { │ │ │ │ │ - node.setAttributeNS(null, "cursor", style.cursor) │ │ │ │ │ + }, │ │ │ │ │ + setTileSize: function(size) { │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + size.h = parseInt(size.h * this.ratio, 10); │ │ │ │ │ + size.w = parseInt(size.w * this.ratio, 10) │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.setTileSize.apply(this, [size]) │ │ │ │ │ }, │ │ │ │ │ - dashStyle: function(style, widthFactor) { │ │ │ │ │ - var w = style.strokeWidth * widthFactor; │ │ │ │ │ - var str = style.strokeDashstyle; │ │ │ │ │ - switch (str) { │ │ │ │ │ - case "solid": │ │ │ │ │ - return "none"; │ │ │ │ │ - case "dot": │ │ │ │ │ - return [1, 4 * w].join(); │ │ │ │ │ - case "dash": │ │ │ │ │ - return [4 * w, 4 * w].join(); │ │ │ │ │ - case "dashdot": │ │ │ │ │ - return [4 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ - case "longdash": │ │ │ │ │ - return [8 * w, 4 * w].join(); │ │ │ │ │ - case "longdashdot": │ │ │ │ │ - return [8 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ - default: │ │ │ │ │ - return OpenLayers.String.trim(str).replace(/\s+/g, ",") │ │ │ │ │ + getTilesBounds: function() { │ │ │ │ │ + var bounds = null; │ │ │ │ │ + var length = this.grid.length; │ │ │ │ │ + if (length) { │ │ │ │ │ + var bottomLeftTileBounds = this.grid[length - 1][0].bounds, │ │ │ │ │ + width = this.grid[0].length * bottomLeftTileBounds.getWidth(), │ │ │ │ │ + height = this.grid.length * bottomLeftTileBounds.getHeight(); │ │ │ │ │ + bounds = new OpenLayers.Bounds(bottomLeftTileBounds.left, bottomLeftTileBounds.bottom, bottomLeftTileBounds.left + width, bottomLeftTileBounds.bottom + height) │ │ │ │ │ } │ │ │ │ │ + return bounds │ │ │ │ │ }, │ │ │ │ │ - createNode: function(type, id) { │ │ │ │ │ - var node = document.createElementNS(this.xmlns, type); │ │ │ │ │ - if (id) { │ │ │ │ │ - node.setAttributeNS(null, "id", id) │ │ │ │ │ + initSingleTile: function(bounds) { │ │ │ │ │ + this.events.triggerEvent("retile"); │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var tileWidth = bounds.getWidth() * this.ratio; │ │ │ │ │ + var tileHeight = bounds.getHeight() * this.ratio; │ │ │ │ │ + var tileBounds = new OpenLayers.Bounds(center.lon - tileWidth / 2, center.lat - tileHeight / 2, center.lon + tileWidth / 2, center.lat + tileHeight / 2); │ │ │ │ │ + var px = this.map.getLayerPxFromLonLat({ │ │ │ │ │ + lon: tileBounds.left, │ │ │ │ │ + lat: tileBounds.top │ │ │ │ │ + }); │ │ │ │ │ + if (!this.grid.length) { │ │ │ │ │ + this.grid[0] = [] │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ + var tile = this.grid[0][0]; │ │ │ │ │ + if (!tile) { │ │ │ │ │ + tile = this.addTile(tileBounds, px); │ │ │ │ │ + this.addTileMonitoringHooks(tile); │ │ │ │ │ + tile.draw(); │ │ │ │ │ + this.grid[0][0] = tile │ │ │ │ │ + } else { │ │ │ │ │ + tile.moveTo(tileBounds, px) │ │ │ │ │ + } │ │ │ │ │ + this.removeExcessTiles(1, 1); │ │ │ │ │ + this.gridResolution = this.getServerResolution() │ │ │ │ │ }, │ │ │ │ │ - nodeTypeCompare: function(node, type) { │ │ │ │ │ - return type == node.nodeName │ │ │ │ │ + calculateGridLayout: function(bounds, origin, resolution) { │ │ │ │ │ + var tilelon = resolution * this.tileSize.w; │ │ │ │ │ + var tilelat = resolution * this.tileSize.h; │ │ │ │ │ + var offsetlon = bounds.left - origin.lon; │ │ │ │ │ + var tilecol = Math.floor(offsetlon / tilelon) - this.buffer; │ │ │ │ │ + var rowSign = this.rowSign; │ │ │ │ │ + var offsetlat = rowSign * (origin.lat - bounds.top + tilelat); │ │ │ │ │ + var tilerow = Math[~rowSign ? "floor" : "ceil"](offsetlat / tilelat) - this.buffer * rowSign; │ │ │ │ │ + return { │ │ │ │ │ + tilelon: tilelon, │ │ │ │ │ + tilelat: tilelat, │ │ │ │ │ + startcol: tilecol, │ │ │ │ │ + startrow: tilerow │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - createRenderRoot: function() { │ │ │ │ │ - var svg = this.nodeFactory(this.container.id + "_svgRoot", "svg"); │ │ │ │ │ - svg.style.display = "block"; │ │ │ │ │ - return svg │ │ │ │ │ + getTileOrigin: function() { │ │ │ │ │ + var origin = this.tileOrigin; │ │ │ │ │ + if (!origin) { │ │ │ │ │ + var extent = this.getMaxExtent(); │ │ │ │ │ + var edges = { │ │ │ │ │ + tl: ["left", "top"], │ │ │ │ │ + tr: ["right", "top"], │ │ │ │ │ + bl: ["left", "bottom"], │ │ │ │ │ + br: ["right", "bottom"] │ │ │ │ │ + } [this.tileOriginCorner]; │ │ │ │ │ + origin = new OpenLayers.LonLat(extent[edges[0]], extent[edges[1]]) │ │ │ │ │ + } │ │ │ │ │ + return origin │ │ │ │ │ }, │ │ │ │ │ - createRoot: function(suffix) { │ │ │ │ │ - return this.nodeFactory(this.container.id + suffix, "g") │ │ │ │ │ + getTileBoundsForGridIndex: function(row, col) { │ │ │ │ │ + var origin = this.getTileOrigin(); │ │ │ │ │ + var tileLayout = this.gridLayout; │ │ │ │ │ + var tilelon = tileLayout.tilelon; │ │ │ │ │ + var tilelat = tileLayout.tilelat; │ │ │ │ │ + var startcol = tileLayout.startcol; │ │ │ │ │ + var startrow = tileLayout.startrow; │ │ │ │ │ + var rowSign = this.rowSign; │ │ │ │ │ + return new OpenLayers.Bounds(origin.lon + (startcol + col) * tilelon, origin.lat - (startrow + row * rowSign) * tilelat * rowSign, origin.lon + (startcol + col + 1) * tilelon, origin.lat - (startrow + (row - 1) * rowSign) * tilelat * rowSign) │ │ │ │ │ }, │ │ │ │ │ - createDefs: function() { │ │ │ │ │ - var defs = this.nodeFactory(this.container.id + "_defs", "defs"); │ │ │ │ │ - this.rendererRoot.appendChild(defs); │ │ │ │ │ - return defs │ │ │ │ │ + initGriddedTiles: function(bounds) { │ │ │ │ │ + this.events.triggerEvent("retile"); │ │ │ │ │ + var viewSize = this.map.getSize(); │ │ │ │ │ + var origin = this.getTileOrigin(); │ │ │ │ │ + var resolution = this.map.getResolution(), │ │ │ │ │ + serverResolution = this.getServerResolution(), │ │ │ │ │ + ratio = resolution / serverResolution, │ │ │ │ │ + tileSize = { │ │ │ │ │ + w: this.tileSize.w / ratio, │ │ │ │ │ + h: this.tileSize.h / ratio │ │ │ │ │ + }; │ │ │ │ │ + var minRows = Math.ceil(viewSize.h / tileSize.h) + 2 * this.buffer + 1; │ │ │ │ │ + var minCols = Math.ceil(viewSize.w / tileSize.w) + 2 * this.buffer + 1; │ │ │ │ │ + var tileLayout = this.calculateGridLayout(bounds, origin, serverResolution); │ │ │ │ │ + this.gridLayout = tileLayout; │ │ │ │ │ + var tilelon = tileLayout.tilelon; │ │ │ │ │ + var tilelat = tileLayout.tilelat; │ │ │ │ │ + var layerContainerDivLeft = this.map.layerContainerOriginPx.x; │ │ │ │ │ + var layerContainerDivTop = this.map.layerContainerOriginPx.y; │ │ │ │ │ + var tileBounds = this.getTileBoundsForGridIndex(0, 0); │ │ │ │ │ + var startPx = this.map.getViewPortPxFromLonLat(new OpenLayers.LonLat(tileBounds.left, tileBounds.top)); │ │ │ │ │ + startPx.x = Math.round(startPx.x) - layerContainerDivLeft; │ │ │ │ │ + startPx.y = Math.round(startPx.y) - layerContainerDivTop; │ │ │ │ │ + var tileData = [], │ │ │ │ │ + center = this.map.getCenter(); │ │ │ │ │ + var rowidx = 0; │ │ │ │ │ + do { │ │ │ │ │ + var row = this.grid[rowidx]; │ │ │ │ │ + if (!row) { │ │ │ │ │ + row = []; │ │ │ │ │ + this.grid.push(row) │ │ │ │ │ + } │ │ │ │ │ + var colidx = 0; │ │ │ │ │ + do { │ │ │ │ │ + tileBounds = this.getTileBoundsForGridIndex(rowidx, colidx); │ │ │ │ │ + var px = startPx.clone(); │ │ │ │ │ + px.x = px.x + colidx * Math.round(tileSize.w); │ │ │ │ │ + px.y = px.y + rowidx * Math.round(tileSize.h); │ │ │ │ │ + var tile = row[colidx]; │ │ │ │ │ + if (!tile) { │ │ │ │ │ + tile = this.addTile(tileBounds, px); │ │ │ │ │ + this.addTileMonitoringHooks(tile); │ │ │ │ │ + row.push(tile) │ │ │ │ │ + } else { │ │ │ │ │ + tile.moveTo(tileBounds, px, false) │ │ │ │ │ + } │ │ │ │ │ + var tileCenter = tileBounds.getCenterLonLat(); │ │ │ │ │ + tileData.push({ │ │ │ │ │ + tile: tile, │ │ │ │ │ + distance: Math.pow(tileCenter.lon - center.lon, 2) + Math.pow(tileCenter.lat - center.lat, 2) │ │ │ │ │ + }); │ │ │ │ │ + colidx += 1 │ │ │ │ │ + } while (tileBounds.right <= bounds.right + tilelon * this.buffer || colidx < minCols); │ │ │ │ │ + rowidx += 1 │ │ │ │ │ + } while (tileBounds.bottom >= bounds.bottom - tilelat * this.buffer || rowidx < minRows); │ │ │ │ │ + this.removeExcessTiles(rowidx, colidx); │ │ │ │ │ + var resolution = this.getServerResolution(); │ │ │ │ │ + this.gridResolution = resolution; │ │ │ │ │ + tileData.sort(function(a, b) { │ │ │ │ │ + return a.distance - b.distance │ │ │ │ │ + }); │ │ │ │ │ + for (var i = 0, ii = tileData.length; i < ii; ++i) { │ │ │ │ │ + tileData[i].tile.draw() │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - drawPoint: function(node, geometry) { │ │ │ │ │ - return this.drawCircle(node, geometry, 1) │ │ │ │ │ + getMaxExtent: function() { │ │ │ │ │ + return this.maxExtent │ │ │ │ │ }, │ │ │ │ │ - drawCircle: function(node, geometry, radius) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = (geometry.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y = this.top - geometry.y / resolution; │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - node.setAttributeNS(null, "cx", x); │ │ │ │ │ - node.setAttributeNS(null, "cy", y); │ │ │ │ │ - node.setAttributeNS(null, "r", radius); │ │ │ │ │ - return node │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + addTile: function(bounds, position) { │ │ │ │ │ + var tile = new this.tileClass(this, position, bounds, null, this.tileSize, this.tileOptions); │ │ │ │ │ + this.events.triggerEvent("addtile", { │ │ │ │ │ + tile: tile │ │ │ │ │ + }); │ │ │ │ │ + return tile │ │ │ │ │ + }, │ │ │ │ │ + addTileMonitoringHooks: function(tile) { │ │ │ │ │ + var replacingCls = "olTileReplacing"; │ │ │ │ │ + tile.onLoadStart = function() { │ │ │ │ │ + if (this.loading === false) { │ │ │ │ │ + this.loading = true; │ │ │ │ │ + this.events.triggerEvent("loadstart") │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("tileloadstart", { │ │ │ │ │ + tile: tile │ │ │ │ │ + }); │ │ │ │ │ + this.numLoadingTiles++; │ │ │ │ │ + if (!this.singleTile && this.backBuffer && this.gridResolution === this.backBufferResolution) { │ │ │ │ │ + OpenLayers.Element.addClass(tile.getTile(), replacingCls) │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + tile.onLoadEnd = function(evt) { │ │ │ │ │ + this.numLoadingTiles--; │ │ │ │ │ + var aborted = evt.type === "unload"; │ │ │ │ │ + this.events.triggerEvent("tileloaded", { │ │ │ │ │ + tile: tile, │ │ │ │ │ + aborted: aborted │ │ │ │ │ + }); │ │ │ │ │ + if (!this.singleTile && !aborted && this.backBuffer && this.gridResolution === this.backBufferResolution) { │ │ │ │ │ + var tileDiv = tile.getTile(); │ │ │ │ │ + if (OpenLayers.Element.getStyle(tileDiv, "display") === "none") { │ │ │ │ │ + var bufferTile = document.getElementById(tile.id + "_bb"); │ │ │ │ │ + if (bufferTile) { │ │ │ │ │ + bufferTile.parentNode.removeChild(bufferTile) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.removeClass(tileDiv, replacingCls) │ │ │ │ │ + } │ │ │ │ │ + if (this.numLoadingTiles === 0) { │ │ │ │ │ + if (this.backBuffer) { │ │ │ │ │ + if (this.backBuffer.childNodes.length === 0) { │ │ │ │ │ + this.removeBackBuffer() │ │ │ │ │ + } else { │ │ │ │ │ + this._transitionElement = aborted ? this.div.lastChild : tile.imgDiv; │ │ │ │ │ + var transitionendEvents = this.transitionendEvents; │ │ │ │ │ + for (var i = transitionendEvents.length - 1; i >= 0; --i) { │ │ │ │ │ + OpenLayers.Event.observe(this._transitionElement, transitionendEvents[i], this._removeBackBuffer) │ │ │ │ │ + } │ │ │ │ │ + this.backBufferTimerId = window.setTimeout(this._removeBackBuffer, this.removeBackBufferDelay) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.loading = false; │ │ │ │ │ + this.events.triggerEvent("loadend") │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + tile.onLoadError = function() { │ │ │ │ │ + this.events.triggerEvent("tileerror", { │ │ │ │ │ + tile: tile │ │ │ │ │ + }) │ │ │ │ │ + }; │ │ │ │ │ + tile.events.on({ │ │ │ │ │ + loadstart: tile.onLoadStart, │ │ │ │ │ + loadend: tile.onLoadEnd, │ │ │ │ │ + unload: tile.onLoadEnd, │ │ │ │ │ + loaderror: tile.onLoadError, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + removeTileMonitoringHooks: function(tile) { │ │ │ │ │ + tile.unload(); │ │ │ │ │ + tile.events.un({ │ │ │ │ │ + loadstart: tile.onLoadStart, │ │ │ │ │ + loadend: tile.onLoadEnd, │ │ │ │ │ + unload: tile.onLoadEnd, │ │ │ │ │ + loaderror: tile.onLoadError, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + moveGriddedTiles: function() { │ │ │ │ │ + var buffer = this.buffer + 1; │ │ │ │ │ + while (true) { │ │ │ │ │ + var tlTile = this.grid[0][0]; │ │ │ │ │ + var tlViewPort = { │ │ │ │ │ + x: tlTile.position.x + this.map.layerContainerOriginPx.x, │ │ │ │ │ + y: tlTile.position.y + this.map.layerContainerOriginPx.y │ │ │ │ │ + }; │ │ │ │ │ + var ratio = this.getServerResolution() / this.map.getResolution(); │ │ │ │ │ + var tileSize = { │ │ │ │ │ + w: Math.round(this.tileSize.w * ratio), │ │ │ │ │ + h: Math.round(this.tileSize.h * ratio) │ │ │ │ │ + }; │ │ │ │ │ + if (tlViewPort.x > -tileSize.w * (buffer - 1)) { │ │ │ │ │ + this.shiftColumn(true, tileSize) │ │ │ │ │ + } else if (tlViewPort.x < -tileSize.w * buffer) { │ │ │ │ │ + this.shiftColumn(false, tileSize) │ │ │ │ │ + } else if (tlViewPort.y > -tileSize.h * (buffer - 1)) { │ │ │ │ │ + this.shiftRow(true, tileSize) │ │ │ │ │ + } else if (tlViewPort.y < -tileSize.h * buffer) { │ │ │ │ │ + this.shiftRow(false, tileSize) │ │ │ │ │ + } else { │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - drawLineString: function(node, geometry) { │ │ │ │ │ - var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ - if (componentsResult.path) { │ │ │ │ │ - node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ - return componentsResult.complete ? node : null │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + shiftRow: function(prepend, tileSize) { │ │ │ │ │ + var grid = this.grid; │ │ │ │ │ + var rowIndex = prepend ? 0 : grid.length - 1; │ │ │ │ │ + var sign = prepend ? -1 : 1; │ │ │ │ │ + var rowSign = this.rowSign; │ │ │ │ │ + var tileLayout = this.gridLayout; │ │ │ │ │ + tileLayout.startrow += sign * rowSign; │ │ │ │ │ + var modelRow = grid[rowIndex]; │ │ │ │ │ + var row = grid[prepend ? "pop" : "shift"](); │ │ │ │ │ + for (var i = 0, len = row.length; i < len; i++) { │ │ │ │ │ + var tile = row[i]; │ │ │ │ │ + var position = modelRow[i].position.clone(); │ │ │ │ │ + position.y += tileSize.h * sign; │ │ │ │ │ + tile.moveTo(this.getTileBoundsForGridIndex(rowIndex, i), position) │ │ │ │ │ } │ │ │ │ │ + grid[prepend ? "unshift" : "push"](row) │ │ │ │ │ }, │ │ │ │ │ - drawLinearRing: function(node, geometry) { │ │ │ │ │ - var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ - if (componentsResult.path) { │ │ │ │ │ - node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ - return componentsResult.complete ? node : null │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + shiftColumn: function(prepend, tileSize) { │ │ │ │ │ + var grid = this.grid; │ │ │ │ │ + var colIndex = prepend ? 0 : grid[0].length - 1; │ │ │ │ │ + var sign = prepend ? -1 : 1; │ │ │ │ │ + var tileLayout = this.gridLayout; │ │ │ │ │ + tileLayout.startcol += sign; │ │ │ │ │ + for (var i = 0, len = grid.length; i < len; i++) { │ │ │ │ │ + var row = grid[i]; │ │ │ │ │ + var position = row[colIndex].position.clone(); │ │ │ │ │ + var tile = row[prepend ? "pop" : "shift"](); │ │ │ │ │ + position.x += tileSize.w * sign; │ │ │ │ │ + tile.moveTo(this.getTileBoundsForGridIndex(i, colIndex), position); │ │ │ │ │ + row[prepend ? "unshift" : "push"](tile) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - drawPolygon: function(node, geometry) { │ │ │ │ │ - var d = ""; │ │ │ │ │ - var draw = true; │ │ │ │ │ - var complete = true; │ │ │ │ │ - var linearRingResult, path; │ │ │ │ │ - for (var j = 0, len = geometry.components.length; j < len; j++) { │ │ │ │ │ - d += " M"; │ │ │ │ │ - linearRingResult = this.getComponentsString(geometry.components[j].components, " "); │ │ │ │ │ - path = linearRingResult.path; │ │ │ │ │ - if (path) { │ │ │ │ │ - d += " " + path; │ │ │ │ │ - complete = linearRingResult.complete && complete │ │ │ │ │ - } else { │ │ │ │ │ - draw = false │ │ │ │ │ + removeExcessTiles: function(rows, columns) { │ │ │ │ │ + var i, l; │ │ │ │ │ + while (this.grid.length > rows) { │ │ │ │ │ + var row = this.grid.pop(); │ │ │ │ │ + for (i = 0, l = row.length; i < l; i++) { │ │ │ │ │ + var tile = row[i]; │ │ │ │ │ + this.destroyTile(tile) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - d += " z"; │ │ │ │ │ - if (draw) { │ │ │ │ │ - node.setAttributeNS(null, "d", d); │ │ │ │ │ - node.setAttributeNS(null, "fill-rule", "evenodd"); │ │ │ │ │ - return complete ? node : null │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + for (i = 0, l = this.grid.length; i < l; i++) { │ │ │ │ │ + while (this.grid[i].length > columns) { │ │ │ │ │ + var row = this.grid[i]; │ │ │ │ │ + var tile = row.pop(); │ │ │ │ │ + this.destroyTile(tile) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - drawRectangle: function(node, geometry) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = (geometry.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y = this.top - geometry.y / resolution; │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - node.setAttributeNS(null, "x", x); │ │ │ │ │ - node.setAttributeNS(null, "y", y); │ │ │ │ │ - node.setAttributeNS(null, "width", geometry.width / resolution); │ │ │ │ │ - node.setAttributeNS(null, "height", geometry.height / resolution); │ │ │ │ │ - return node │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + this.clearGrid(); │ │ │ │ │ + this.setTileSize() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - drawText: function(featureId, style, location) { │ │ │ │ │ - var drawOutline = !!style.labelOutlineWidth; │ │ │ │ │ - if (drawOutline) { │ │ │ │ │ - var outlineStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ - outlineStyle.fontColor = outlineStyle.labelOutlineColor; │ │ │ │ │ - outlineStyle.fontStrokeColor = outlineStyle.labelOutlineColor; │ │ │ │ │ - outlineStyle.fontStrokeWidth = style.labelOutlineWidth; │ │ │ │ │ - if (style.labelOutlineOpacity) { │ │ │ │ │ - outlineStyle.fontOpacity = style.labelOutlineOpacity │ │ │ │ │ + getTileBounds: function(viewPortPx) { │ │ │ │ │ + var maxExtent = this.maxExtent; │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var tileMapWidth = resolution * this.tileSize.w; │ │ │ │ │ + var tileMapHeight = resolution * this.tileSize.h; │ │ │ │ │ + var mapPoint = this.getLonLatFromViewPortPx(viewPortPx); │ │ │ │ │ + var tileLeft = maxExtent.left + tileMapWidth * Math.floor((mapPoint.lon - maxExtent.left) / tileMapWidth); │ │ │ │ │ + var tileBottom = maxExtent.bottom + tileMapHeight * Math.floor((mapPoint.lat - maxExtent.bottom) / tileMapHeight); │ │ │ │ │ + return new OpenLayers.Bounds(tileLeft, tileBottom, tileLeft + tileMapWidth, tileBottom + tileMapHeight) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Grid" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.WMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + service: "WMS", │ │ │ │ │ + version: "1.1.1", │ │ │ │ │ + request: "GetMap", │ │ │ │ │ + styles: "", │ │ │ │ │ + format: "image/jpeg" │ │ │ │ │ + }, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + encodeBBOX: false, │ │ │ │ │ + noMagic: false, │ │ │ │ │ + yx: {}, │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ + if (parseFloat(params.VERSION) >= 1.3 && !params.EXCEPTIONS) { │ │ │ │ │ + params.EXCEPTIONS = "INIMAGE" │ │ │ │ │ + } │ │ │ │ │ + newArguments.push(name, url, params, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + OpenLayers.Util.applyDefaults(this.params, OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS)); │ │ │ │ │ + if (!this.noMagic && this.params.TRANSPARENT && this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ + if (options == null || !options.isBaseLayer) { │ │ │ │ │ + this.isBaseLayer = false │ │ │ │ │ + } │ │ │ │ │ + if (this.params.FORMAT == "image/jpeg") { │ │ │ │ │ + this.params.FORMAT = OpenLayers.Util.alphaHack() ? "image/gif" : "image/png" │ │ │ │ │ } │ │ │ │ │ - delete outlineStyle.labelOutlineWidth; │ │ │ │ │ - this.drawText(featureId, outlineStyle, location) │ │ │ │ │ } │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = (location.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y = location.y / resolution - this.top; │ │ │ │ │ - var suffix = drawOutline ? this.LABEL_OUTLINE_SUFFIX : this.LABEL_ID_SUFFIX; │ │ │ │ │ - var label = this.nodeFactory(featureId + suffix, "text"); │ │ │ │ │ - label.setAttributeNS(null, "x", x); │ │ │ │ │ - label.setAttributeNS(null, "y", -y); │ │ │ │ │ - if (style.fontColor) { │ │ │ │ │ - label.setAttributeNS(null, "fill", style.fontColor) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.WMS(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - if (style.fontStrokeColor) { │ │ │ │ │ - label.setAttributeNS(null, "stroke", style.fontStrokeColor) │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + reverseAxisOrder: function() { │ │ │ │ │ + var projCode = this.projection.getCode(); │ │ │ │ │ + return parseFloat(this.params.VERSION) >= 1.3 && !!(this.yx[projCode] || OpenLayers.Projection.defaults[projCode] && OpenLayers.Projection.defaults[projCode].yx) │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var imageSize = this.getImageSize(); │ │ │ │ │ + var newParams = {}; │ │ │ │ │ + var reverseAxisOrder = this.reverseAxisOrder(); │ │ │ │ │ + newParams.BBOX = this.encodeBBOX ? bounds.toBBOX(null, reverseAxisOrder) : bounds.toArray(reverseAxisOrder); │ │ │ │ │ + newParams.WIDTH = imageSize.w; │ │ │ │ │ + newParams.HEIGHT = imageSize.h; │ │ │ │ │ + var requestString = this.getFullRequestString(newParams); │ │ │ │ │ + return requestString │ │ │ │ │ + }, │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ + var newArguments = [upperParams]; │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, newArguments) │ │ │ │ │ + }, │ │ │ │ │ + getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ + var mapProjection = this.map.getProjectionObject(); │ │ │ │ │ + var projectionCode = this.projection && this.projection.equals(mapProjection) ? this.projection.getCode() : mapProjection.getCode(); │ │ │ │ │ + var value = projectionCode == "none" ? null : projectionCode; │ │ │ │ │ + if (parseFloat(this.params.VERSION) >= 1.3) { │ │ │ │ │ + this.params.CRS = value │ │ │ │ │ + } else { │ │ │ │ │ + this.params.SRS = value │ │ │ │ │ } │ │ │ │ │ - if (style.fontStrokeWidth) { │ │ │ │ │ - label.setAttributeNS(null, "stroke-width", style.fontStrokeWidth) │ │ │ │ │ + if (typeof this.params.TRANSPARENT == "boolean") { │ │ │ │ │ + newParams.TRANSPARENT = this.params.TRANSPARENT ? "TRUE" : "FALSE" │ │ │ │ │ } │ │ │ │ │ - if (style.fontOpacity) { │ │ │ │ │ - label.setAttributeNS(null, "opacity", style.fontOpacity) │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.getFullRequestString.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.WMS" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.XYZ = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + sphericalMercator: false, │ │ │ │ │ + zoomOffset: 0, │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + if (options && options.sphericalMercator || this.sphericalMercator) { │ │ │ │ │ + options = OpenLayers.Util.extend({ │ │ │ │ │ + projection: "EPSG:900913", │ │ │ │ │ + numZoomLevels: 19 │ │ │ │ │ + }, options) │ │ │ │ │ } │ │ │ │ │ - if (style.fontFamily) { │ │ │ │ │ - label.setAttributeNS(null, "font-family", style.fontFamily) │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name || this.name, url || this.url, {}, options]) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.XYZ(this.name, this.url, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - if (style.fontSize) { │ │ │ │ │ - label.setAttributeNS(null, "font-size", style.fontSize) │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var xyz = this.getXYZ(bounds); │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + var s = "" + xyz.x + xyz.y + xyz.z; │ │ │ │ │ + url = this.selectUrl(s, url) │ │ │ │ │ } │ │ │ │ │ - if (style.fontWeight) { │ │ │ │ │ - label.setAttributeNS(null, "font-weight", style.fontWeight) │ │ │ │ │ + return OpenLayers.String.format(url, xyz) │ │ │ │ │ + }, │ │ │ │ │ + getXYZ: function(bounds) { │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.maxExtent.left) / (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((this.maxExtent.top - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getServerZoom(); │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + var limit = Math.pow(2, z); │ │ │ │ │ + x = (x % limit + limit) % limit │ │ │ │ │ } │ │ │ │ │ - if (style.fontStyle) { │ │ │ │ │ - label.setAttributeNS(null, "font-style", style.fontStyle) │ │ │ │ │ + return { │ │ │ │ │ + x: x, │ │ │ │ │ + y: y, │ │ │ │ │ + z: z │ │ │ │ │ } │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - label.setAttributeNS(null, "pointer-events", "visible"); │ │ │ │ │ - label._featureId = featureId │ │ │ │ │ - } else { │ │ │ │ │ - label.setAttributeNS(null, "pointer-events", "none") │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (!this.tileOrigin) { │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, this.maxExtent.bottom) │ │ │ │ │ } │ │ │ │ │ - var align = style.labelAlign || OpenLayers.Renderer.defaultSymbolizer.labelAlign; │ │ │ │ │ - label.setAttributeNS(null, "text-anchor", OpenLayers.Renderer.SVG.LABEL_ALIGN[align[0]] || "middle"); │ │ │ │ │ - if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ - label.setAttributeNS(null, "dominant-baseline", OpenLayers.Renderer.SVG.LABEL_ALIGN[align[1]] || "central") │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.XYZ" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Bing = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ + key: null, │ │ │ │ │ + serverResolutions: [156543.03390625, 78271.516953125, 39135.7584765625, 19567.87923828125, 9783.939619140625, 4891.9698095703125, 2445.9849047851562, 1222.9924523925781, 611.4962261962891, 305.74811309814453, 152.87405654907226, 76.43702827453613, 38.218514137268066, 19.109257068634033, 9.554628534317017, 4.777314267158508, 2.388657133579254, 1.194328566789627, .5971642833948135, .29858214169740677, .14929107084870338, .07464553542435169], │ │ │ │ │ + attributionTemplate: '<span class="olBingAttribution ${type}">' + '<div><a target="_blank" href="http://www.bing.com/maps/">' + '<img src="${logo}" /></a></div>${copyrights}' + '<a style="white-space: nowrap" target="_blank" ' + 'href="http://www.microsoft.com/maps/product/terms.html">' + "Terms of Use</a></span>", │ │ │ │ │ + metadata: null, │ │ │ │ │ + protocolRegex: /^http:/i, │ │ │ │ │ + type: "Road", │ │ │ │ │ + culture: "en-US", │ │ │ │ │ + metadataParams: null, │ │ │ │ │ + tileOptions: null, │ │ │ │ │ + protocol: ~window.location.href.indexOf("http") ? "" : "http:", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + sphericalMercator: true │ │ │ │ │ + }, options); │ │ │ │ │ + var name = options.name || "Bing " + (options.type || this.type); │ │ │ │ │ + var newArgs = [name, null, options]; │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, newArgs); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + crossOriginKeyword: "anonymous" │ │ │ │ │ + }, this.options.tileOptions); │ │ │ │ │ + this.loadMetadata() │ │ │ │ │ + }, │ │ │ │ │ + loadMetadata: function() { │ │ │ │ │ + this._callbackId = "_callback_" + this.id.replace(/\./g, "_"); │ │ │ │ │ + window[this._callbackId] = OpenLayers.Function.bind(OpenLayers.Layer.Bing.processMetadata, this); │ │ │ │ │ + var params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + key: this.key, │ │ │ │ │ + jsonp: this._callbackId, │ │ │ │ │ + include: "ImageryProviders" │ │ │ │ │ + }, this.metadataParams); │ │ │ │ │ + var url = this.protocol + "//dev.virtualearth.net/REST/v1/Imagery/Metadata/" + this.type + "?" + OpenLayers.Util.getParameterString(params); │ │ │ │ │ + var script = document.createElement("script"); │ │ │ │ │ + script.type = "text/javascript"; │ │ │ │ │ + script.src = url; │ │ │ │ │ + script.id = this._callbackId; │ │ │ │ │ + document.getElementsByTagName("head")[0].appendChild(script) │ │ │ │ │ + }, │ │ │ │ │ + initLayer: function() { │ │ │ │ │ + var res = this.metadata.resourceSets[0].resources[0]; │ │ │ │ │ + var url = res.imageUrl.replace("{quadkey}", "${quadkey}"); │ │ │ │ │ + url = url.replace("{culture}", this.culture); │ │ │ │ │ + url = url.replace(this.protocolRegex, this.protocol); │ │ │ │ │ + this.url = []; │ │ │ │ │ + for (var i = 0; i < res.imageUrlSubdomains.length; ++i) { │ │ │ │ │ + this.url.push(url.replace("{subdomain}", res.imageUrlSubdomains[i])) │ │ │ │ │ } │ │ │ │ │ - var labelRows = style.label.split("\n"); │ │ │ │ │ - var numRows = labelRows.length; │ │ │ │ │ - while (label.childNodes.length > numRows) { │ │ │ │ │ - label.removeChild(label.lastChild) │ │ │ │ │ + this.addOptions({ │ │ │ │ │ + maxResolution: Math.min(this.serverResolutions[res.zoomMin], this.maxResolution || Number.POSITIVE_INFINITY), │ │ │ │ │ + numZoomLevels: Math.min(res.zoomMax + 1 - res.zoomMin, this.numZoomLevels) │ │ │ │ │ + }, true); │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + this.redraw() │ │ │ │ │ } │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - var tspan = this.nodeFactory(featureId + suffix + "_tspan_" + i, "tspan"); │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - tspan._featureId = featureId; │ │ │ │ │ - tspan._geometry = location; │ │ │ │ │ - tspan._geometryClass = location.CLASS_NAME │ │ │ │ │ + this.updateAttribution() │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + if (!this.url) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var xyz = this.getXYZ(bounds), │ │ │ │ │ + x = xyz.x, │ │ │ │ │ + y = xyz.y, │ │ │ │ │ + z = xyz.z; │ │ │ │ │ + var quadDigits = []; │ │ │ │ │ + for (var i = z; i > 0; --i) { │ │ │ │ │ + var digit = "0"; │ │ │ │ │ + var mask = 1 << i - 1; │ │ │ │ │ + if ((x & mask) != 0) { │ │ │ │ │ + digit++ │ │ │ │ │ } │ │ │ │ │ - if (OpenLayers.IS_GECKO === false) { │ │ │ │ │ - tspan.setAttributeNS(null, "baseline-shift", OpenLayers.Renderer.SVG.LABEL_VSHIFT[align[1]] || "-35%") │ │ │ │ │ + if ((y & mask) != 0) { │ │ │ │ │ + digit++; │ │ │ │ │ + digit++ │ │ │ │ │ } │ │ │ │ │ - tspan.setAttribute("x", x); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - var vfactor = OpenLayers.Renderer.SVG.LABEL_VFACTOR[align[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5 │ │ │ │ │ + quadDigits.push(digit) │ │ │ │ │ + } │ │ │ │ │ + var quadKey = quadDigits.join(""); │ │ │ │ │ + var url = this.selectUrl("" + x + y + z, this.url); │ │ │ │ │ + return OpenLayers.String.format(url, { │ │ │ │ │ + quadkey: quadKey │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + updateAttribution: function() { │ │ │ │ │ + var metadata = this.metadata; │ │ │ │ │ + if (!metadata.resourceSets || !this.map || !this.map.center) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var res = metadata.resourceSets[0].resources[0]; │ │ │ │ │ + var extent = this.map.getExtent().transform(this.map.getProjectionObject(), new OpenLayers.Projection("EPSG:4326")); │ │ │ │ │ + var providers = res.imageryProviders || [], │ │ │ │ │ + zoom = OpenLayers.Util.indexOf(this.serverResolutions, this.getServerResolution()), │ │ │ │ │ + copyrights = "", │ │ │ │ │ + provider, i, ii, j, jj, bbox, coverage; │ │ │ │ │ + for (i = 0, ii = providers.length; i < ii; ++i) { │ │ │ │ │ + provider = providers[i]; │ │ │ │ │ + for (j = 0, jj = provider.coverageAreas.length; j < jj; ++j) { │ │ │ │ │ + coverage = provider.coverageAreas[j]; │ │ │ │ │ + bbox = OpenLayers.Bounds.fromArray(coverage.bbox, true); │ │ │ │ │ + if (extent.intersectsBounds(bbox) && zoom <= coverage.zoomMax && zoom >= coverage.zoomMin) { │ │ │ │ │ + copyrights += provider.attribution + " " │ │ │ │ │ } │ │ │ │ │ - tspan.setAttribute("dy", vfactor * (numRows - 1) + "em") │ │ │ │ │ - } else { │ │ │ │ │ - tspan.setAttribute("dy", "1em") │ │ │ │ │ - } │ │ │ │ │ - tspan.textContent = labelRows[i] === "" ? " " : labelRows[i]; │ │ │ │ │ - if (!tspan.parentNode) { │ │ │ │ │ - label.appendChild(tspan) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (!label.parentNode) { │ │ │ │ │ - this.textRoot.appendChild(label) │ │ │ │ │ + var logo = metadata.brandLogoUri.replace(this.protocolRegex, this.protocol); │ │ │ │ │ + this.attribution = OpenLayers.String.format(this.attributionTemplate, { │ │ │ │ │ + type: this.type.toLowerCase(), │ │ │ │ │ + logo: logo, │ │ │ │ │ + copyrights: copyrights │ │ │ │ │ + }); │ │ │ │ │ + this.map && this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "attribution" │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + setMap: function() { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.register("moveend", this, this.updateAttribution) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Bing(this.options) │ │ │ │ │ } │ │ │ │ │ + obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - getComponentsString: function(components, separator) { │ │ │ │ │ - var renderCmp = []; │ │ │ │ │ - var complete = true; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - var strings = []; │ │ │ │ │ - var str, component; │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - component = components[i]; │ │ │ │ │ - renderCmp.push(component); │ │ │ │ │ - str = this.getShortString(component); │ │ │ │ │ - if (str) { │ │ │ │ │ - strings.push(str) │ │ │ │ │ - } else { │ │ │ │ │ - if (i > 0) { │ │ │ │ │ - if (this.getShortString(components[i - 1])) { │ │ │ │ │ - strings.push(this.clipLine(components[i], components[i - 1])) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (i < len - 1) { │ │ │ │ │ - if (this.getShortString(components[i + 1])) { │ │ │ │ │ - strings.push(this.clipLine(components[i], components[i + 1])) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.map && this.map.events.unregister("moveend", this, this.updateAttribution); │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Bing" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Bing.processMetadata = function(metadata) { │ │ │ │ │ + this.metadata = metadata; │ │ │ │ │ + this.initLayer(); │ │ │ │ │ + var script = document.getElementById(this._callbackId); │ │ │ │ │ + script.parentNode.removeChild(script); │ │ │ │ │ + window[this._callbackId] = undefined; │ │ │ │ │ + delete this._callbackId │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Layer.OSM = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ + name: "OpenStreetMap", │ │ │ │ │ + url: ["http://a.tile.openstreetmap.org/${z}/${x}/${y}.png", "http://b.tile.openstreetmap.org/${z}/${x}/${y}.png", "http://c.tile.openstreetmap.org/${z}/${x}/${y}.png"], │ │ │ │ │ + attribution: "© <a href='http://www.openstreetmap.org/copyright'>OpenStreetMap</a> contributors", │ │ │ │ │ + sphericalMercator: true, │ │ │ │ │ + wrapDateLine: true, │ │ │ │ │ + tileOptions: null, │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + crossOriginKeyword: "anonymous" │ │ │ │ │ + }, this.options && this.options.tileOptions) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.OSM(this.name, this.url, this.getOptions()) │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.OSM" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.SphericalMercator = { │ │ │ │ │ + getExtent: function() { │ │ │ │ │ + var extent = null; │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + extent = this.map.calculateBounds() │ │ │ │ │ + } else { │ │ │ │ │ + extent = OpenLayers.Layer.FixedZoomLevels.prototype.getExtent.apply(this) │ │ │ │ │ + } │ │ │ │ │ + return extent │ │ │ │ │ + }, │ │ │ │ │ + getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ + return OpenLayers.Layer.prototype.getLonLatFromViewPortPx.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + getViewPortPxFromLonLat: function(lonlat) { │ │ │ │ │ + return OpenLayers.Layer.prototype.getViewPortPxFromLonLat.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + initMercatorParameters: function() { │ │ │ │ │ + this.RESOLUTIONS = []; │ │ │ │ │ + var maxResolution = 156543.03390625; │ │ │ │ │ + for (var zoom = 0; zoom <= this.MAX_ZOOM_LEVEL; ++zoom) { │ │ │ │ │ + this.RESOLUTIONS[zoom] = maxResolution / Math.pow(2, zoom) │ │ │ │ │ + } │ │ │ │ │ + this.units = "m"; │ │ │ │ │ + this.projection = this.projection || "EPSG:900913" │ │ │ │ │ + }, │ │ │ │ │ + forwardMercator: function() { │ │ │ │ │ + var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + var sm = new OpenLayers.Projection("EPSG:900913"); │ │ │ │ │ + return function(lon, lat) { │ │ │ │ │ + var point = OpenLayers.Projection.transform({ │ │ │ │ │ + x: lon, │ │ │ │ │ + y: lat │ │ │ │ │ + }, gg, sm); │ │ │ │ │ + return new OpenLayers.LonLat(point.x, point.y) │ │ │ │ │ + } │ │ │ │ │ + }(), │ │ │ │ │ + inverseMercator: function() { │ │ │ │ │ + var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + var sm = new OpenLayers.Projection("EPSG:900913"); │ │ │ │ │ + return function(x, y) { │ │ │ │ │ + var point = OpenLayers.Projection.transform({ │ │ │ │ │ + x: x, │ │ │ │ │ + y: y │ │ │ │ │ + }, sm, gg); │ │ │ │ │ + return new OpenLayers.LonLat(point.x, point.y) │ │ │ │ │ + } │ │ │ │ │ + }() │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Layer.EventPane = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ + smoothDragPan: true, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + isFixed: true, │ │ │ │ │ + pane: null, │ │ │ │ │ + mapObject: null, │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (this.pane == null) { │ │ │ │ │ + this.pane = OpenLayers.Util.createDiv(this.div.id + "_EventPane") │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.mapObject = null; │ │ │ │ │ + this.pane = null; │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.pane.style.zIndex = parseInt(this.div.style.zIndex) + 1; │ │ │ │ │ + this.pane.style.display = this.div.style.display; │ │ │ │ │ + this.pane.style.width = "100%"; │ │ │ │ │ + this.pane.style.height = "100%"; │ │ │ │ │ + if (OpenLayers.BROWSER_NAME == "msie") { │ │ │ │ │ + this.pane.style.background = "url(" + OpenLayers.Util.getImageLocation("blank.gif") + ")" │ │ │ │ │ + } │ │ │ │ │ + if (this.isFixed) { │ │ │ │ │ + this.map.viewPortDiv.appendChild(this.pane) │ │ │ │ │ + } else { │ │ │ │ │ + this.map.layerContainerDiv.appendChild(this.pane) │ │ │ │ │ + } │ │ │ │ │ + this.loadMapObject(); │ │ │ │ │ + if (this.mapObject == null) { │ │ │ │ │ + this.loadWarningMessage() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + if (this.pane && this.pane.parentNode) { │ │ │ │ │ + this.pane.parentNode.removeChild(this.pane) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.prototype.removeMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + loadWarningMessage: function() { │ │ │ │ │ + this.div.style.backgroundColor = "darkblue"; │ │ │ │ │ + var viewSize = this.map.getSize(); │ │ │ │ │ + var msgW = Math.min(viewSize.w, 300); │ │ │ │ │ + var msgH = Math.min(viewSize.h, 200); │ │ │ │ │ + var size = new OpenLayers.Size(msgW, msgH); │ │ │ │ │ + var centerPx = new OpenLayers.Pixel(viewSize.w / 2, viewSize.h / 2); │ │ │ │ │ + var topLeft = centerPx.add(-size.w / 2, -size.h / 2); │ │ │ │ │ + var div = OpenLayers.Util.createDiv(this.name + "_warning", topLeft, size, null, null, null, "auto"); │ │ │ │ │ + div.style.padding = "7px"; │ │ │ │ │ + div.style.backgroundColor = "yellow"; │ │ │ │ │ + div.innerHTML = this.getWarningHTML(); │ │ │ │ │ + this.div.appendChild(div) │ │ │ │ │ + }, │ │ │ │ │ + getWarningHTML: function() { │ │ │ │ │ + return "" │ │ │ │ │ + }, │ │ │ │ │ + display: function(display) { │ │ │ │ │ + OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ + this.pane.style.display = this.div.style.display │ │ │ │ │ + }, │ │ │ │ │ + setZIndex: function(zIndex) { │ │ │ │ │ + OpenLayers.Layer.prototype.setZIndex.apply(this, arguments); │ │ │ │ │ + this.pane.style.zIndex = parseInt(this.div.style.zIndex) + 1 │ │ │ │ │ + }, │ │ │ │ │ + moveByPx: function(dx, dy) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveByPx.apply(this, arguments); │ │ │ │ │ + if (this.dragPanMapObject) { │ │ │ │ │ + this.dragPanMapObject(dx, -dy) │ │ │ │ │ + } else { │ │ │ │ │ + this.moveTo(this.map.getCachedCenter()) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + if (this.mapObject != null) { │ │ │ │ │ + var newCenter = this.map.getCenter(); │ │ │ │ │ + var newZoom = this.map.getZoom(); │ │ │ │ │ + if (newCenter != null) { │ │ │ │ │ + var moOldCenter = this.getMapObjectCenter(); │ │ │ │ │ + var oldCenter = this.getOLLonLatFromMapObjectLonLat(moOldCenter); │ │ │ │ │ + var moOldZoom = this.getMapObjectZoom(); │ │ │ │ │ + var oldZoom = this.getOLZoomFromMapObjectZoom(moOldZoom); │ │ │ │ │ + if (!newCenter.equals(oldCenter) || newZoom != oldZoom) { │ │ │ │ │ + if (!zoomChanged && oldCenter && this.dragPanMapObject && this.smoothDragPan) { │ │ │ │ │ + var oldPx = this.map.getViewPortPxFromLonLat(oldCenter); │ │ │ │ │ + var newPx = this.map.getViewPortPxFromLonLat(newCenter); │ │ │ │ │ + this.dragPanMapObject(newPx.x - oldPx.x, oldPx.y - newPx.y) │ │ │ │ │ + } else { │ │ │ │ │ + var center = this.getMapObjectLonLatFromOLLonLat(newCenter); │ │ │ │ │ + var zoom = this.getMapObjectZoomFromOLZoom(newZoom); │ │ │ │ │ + this.setMapObjectCenter(center, zoom, dragging) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - complete = false │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return { │ │ │ │ │ - path: strings.join(separator || ","), │ │ │ │ │ - complete: complete │ │ │ │ │ + }, │ │ │ │ │ + getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ + var lonlat = null; │ │ │ │ │ + if (this.mapObject != null && this.getMapObjectCenter() != null) { │ │ │ │ │ + var moPixel = this.getMapObjectPixelFromOLPixel(viewPortPx); │ │ │ │ │ + var moLonLat = this.getMapObjectLonLatFromMapObjectPixel(moPixel); │ │ │ │ │ + lonlat = this.getOLLonLatFromMapObjectLonLat(moLonLat) │ │ │ │ │ } │ │ │ │ │ + return lonlat │ │ │ │ │ }, │ │ │ │ │ - clipLine: function(badComponent, goodComponent) { │ │ │ │ │ - if (goodComponent.equals(badComponent)) { │ │ │ │ │ - return "" │ │ │ │ │ + getViewPortPxFromLonLat: function(lonlat) { │ │ │ │ │ + var viewPortPx = null; │ │ │ │ │ + if (this.mapObject != null && this.getMapObjectCenter() != null) { │ │ │ │ │ + var moLonLat = this.getMapObjectLonLatFromOLLonLat(lonlat); │ │ │ │ │ + var moPixel = this.getMapObjectPixelFromMapObjectLonLat(moLonLat); │ │ │ │ │ + viewPortPx = this.getOLPixelFromMapObjectPixel(moPixel) │ │ │ │ │ } │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var maxX = this.MAX_PIXEL - this.translationParameters.x; │ │ │ │ │ - var maxY = this.MAX_PIXEL - this.translationParameters.y; │ │ │ │ │ - var x1 = (goodComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y1 = this.top - goodComponent.y / resolution; │ │ │ │ │ - var x2 = (badComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y2 = this.top - badComponent.y / resolution; │ │ │ │ │ - var k; │ │ │ │ │ - if (x2 < -maxX || x2 > maxX) { │ │ │ │ │ - k = (y2 - y1) / (x2 - x1); │ │ │ │ │ - x2 = x2 < 0 ? -maxX : maxX; │ │ │ │ │ - y2 = y1 + (x2 - x1) * k │ │ │ │ │ + return viewPortPx │ │ │ │ │ + }, │ │ │ │ │ + getOLLonLatFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + var olLonLat = null; │ │ │ │ │ + if (moLonLat != null) { │ │ │ │ │ + var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + olLonLat = new OpenLayers.LonLat(lon, lat) │ │ │ │ │ } │ │ │ │ │ - if (y2 < -maxY || y2 > maxY) { │ │ │ │ │ - k = (x2 - x1) / (y2 - y1); │ │ │ │ │ - y2 = y2 < 0 ? -maxY : maxY; │ │ │ │ │ - x2 = x1 + (y2 - y1) * k │ │ │ │ │ + return olLonLat │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectLonLatFromOLLonLat: function(olLonLat) { │ │ │ │ │ + var moLatLng = null; │ │ │ │ │ + if (olLonLat != null) { │ │ │ │ │ + moLatLng = this.getMapObjectLonLatFromLonLat(olLonLat.lon, olLonLat.lat) │ │ │ │ │ } │ │ │ │ │ - return x2 + "," + y2 │ │ │ │ │ + return moLatLng │ │ │ │ │ }, │ │ │ │ │ - getShortString: function(point) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = (point.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y = this.top - point.y / resolution; │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - return x + "," + y │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + getOLPixelFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + var olPixel = null; │ │ │ │ │ + if (moPixel != null) { │ │ │ │ │ + var x = this.getXFromMapObjectPixel(moPixel); │ │ │ │ │ + var y = this.getYFromMapObjectPixel(moPixel); │ │ │ │ │ + olPixel = new OpenLayers.Pixel(x, y) │ │ │ │ │ } │ │ │ │ │ + return olPixel │ │ │ │ │ }, │ │ │ │ │ - getPosition: function(node) { │ │ │ │ │ - return { │ │ │ │ │ - x: parseFloat(node.getAttributeNS(null, "cx")), │ │ │ │ │ - y: parseFloat(node.getAttributeNS(null, "cy")) │ │ │ │ │ + getMapObjectPixelFromOLPixel: function(olPixel) { │ │ │ │ │ + var moPixel = null; │ │ │ │ │ + if (olPixel != null) { │ │ │ │ │ + moPixel = this.getMapObjectPixelFromXY(olPixel.x, olPixel.y) │ │ │ │ │ } │ │ │ │ │ + return moPixel │ │ │ │ │ }, │ │ │ │ │ - importSymbol: function(graphicName) { │ │ │ │ │ - if (!this.defs) { │ │ │ │ │ - this.defs = this.createDefs() │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.EventPane" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.FixedZoomLevels = OpenLayers.Class({ │ │ │ │ │ + initialize: function() {}, │ │ │ │ │ + initResolutions: function() { │ │ │ │ │ + var props = ["minZoomLevel", "maxZoomLevel", "numZoomLevels"]; │ │ │ │ │ + for (var i = 0, len = props.length; i < len; i++) { │ │ │ │ │ + var property = props[i]; │ │ │ │ │ + this[property] = this.options[property] != null ? this.options[property] : this.map[property] │ │ │ │ │ } │ │ │ │ │ - var id = this.container.id + "-" + graphicName; │ │ │ │ │ - var existing = document.getElementById(id); │ │ │ │ │ - if (existing != null) { │ │ │ │ │ - return existing │ │ │ │ │ + if (this.minZoomLevel == null || this.minZoomLevel < this.MIN_ZOOM_LEVEL) { │ │ │ │ │ + this.minZoomLevel = this.MIN_ZOOM_LEVEL │ │ │ │ │ } │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(graphicName + " is not a valid symbol name") │ │ │ │ │ + var desiredZoomLevels; │ │ │ │ │ + var limitZoomLevels = this.MAX_ZOOM_LEVEL - this.minZoomLevel + 1; │ │ │ │ │ + if (this.options.numZoomLevels == null && this.options.maxZoomLevel != null || this.numZoomLevels == null && this.maxZoomLevel != null) { │ │ │ │ │ + desiredZoomLevels = this.maxZoomLevel - this.minZoomLevel + 1 │ │ │ │ │ + } else { │ │ │ │ │ + desiredZoomLevels = this.numZoomLevels │ │ │ │ │ } │ │ │ │ │ - var symbolNode = this.nodeFactory(id, "symbol"); │ │ │ │ │ - var node = this.nodeFactory(null, "polygon"); │ │ │ │ │ - symbolNode.appendChild(node); │ │ │ │ │ - var symbolExtent = new OpenLayers.Bounds(Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ - var points = []; │ │ │ │ │ - var x, y; │ │ │ │ │ - for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ - symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ - symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ - symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ - points.push(x, ",", y) │ │ │ │ │ + if (desiredZoomLevels != null) { │ │ │ │ │ + this.numZoomLevels = Math.min(desiredZoomLevels, limitZoomLevels) │ │ │ │ │ + } else { │ │ │ │ │ + this.numZoomLevels = limitZoomLevels │ │ │ │ │ + } │ │ │ │ │ + this.maxZoomLevel = this.minZoomLevel + this.numZoomLevels - 1; │ │ │ │ │ + if (this.RESOLUTIONS != null) { │ │ │ │ │ + var resolutionsIndex = 0; │ │ │ │ │ + this.resolutions = []; │ │ │ │ │ + for (var i = this.minZoomLevel; i <= this.maxZoomLevel; i++) { │ │ │ │ │ + this.resolutions[resolutionsIndex++] = this.RESOLUTIONS[i] │ │ │ │ │ + } │ │ │ │ │ + this.maxResolution = this.resolutions[0]; │ │ │ │ │ + this.minResolution = this.resolutions[this.resolutions.length - 1] │ │ │ │ │ } │ │ │ │ │ - node.setAttributeNS(null, "points", points.join(" ")); │ │ │ │ │ - var width = symbolExtent.getWidth(); │ │ │ │ │ - var height = symbolExtent.getHeight(); │ │ │ │ │ - var viewBox = [symbolExtent.left - width, symbolExtent.bottom - height, width * 3, height * 3]; │ │ │ │ │ - symbolNode.setAttributeNS(null, "viewBox", viewBox.join(" ")); │ │ │ │ │ - this.symbolMetrics[id] = [Math.max(width, height), symbolExtent.getCenterLonLat().lon, symbolExtent.getCenterLonLat().lat]; │ │ │ │ │ - this.defs.appendChild(symbolNode); │ │ │ │ │ - return symbolNode │ │ │ │ │ }, │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var featureId = OpenLayers.Renderer.Elements.prototype.getFeatureIdFromEvent.apply(this, arguments); │ │ │ │ │ - if (!featureId) { │ │ │ │ │ - var target = evt.target; │ │ │ │ │ - featureId = target.parentNode && target != this.rendererRoot ? target.parentNode._featureId : undefined │ │ │ │ │ + getResolution: function() { │ │ │ │ │ + if (this.resolutions != null) { │ │ │ │ │ + return OpenLayers.Layer.prototype.getResolution.apply(this, arguments) │ │ │ │ │ + } else { │ │ │ │ │ + var resolution = null; │ │ │ │ │ + var viewSize = this.map.getSize(); │ │ │ │ │ + var extent = this.getExtent(); │ │ │ │ │ + if (viewSize != null && extent != null) { │ │ │ │ │ + resolution = Math.max(extent.getWidth() / viewSize.w, extent.getHeight() / viewSize.h) │ │ │ │ │ + } │ │ │ │ │ + return resolution │ │ │ │ │ } │ │ │ │ │ - return featureId │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.SVG" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_ALIGN = { │ │ │ │ │ - l: "start", │ │ │ │ │ - r: "end", │ │ │ │ │ - b: "bottom", │ │ │ │ │ - t: "hanging" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_VSHIFT = { │ │ │ │ │ - t: "-70%", │ │ │ │ │ - b: "0" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_VFACTOR = { │ │ │ │ │ - t: 0, │ │ │ │ │ - b: -1 │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.SVG.preventDefault = function(e) { │ │ │ │ │ - OpenLayers.Event.preventDefault(e) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol = OpenLayers.Class({ │ │ │ │ │ - format: null, │ │ │ │ │ - options: null, │ │ │ │ │ - autoDestroy: true, │ │ │ │ │ - defaultFilter: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.options = options │ │ │ │ │ + getExtent: function() { │ │ │ │ │ + var size = this.map.getSize(); │ │ │ │ │ + var tl = this.getLonLatFromViewPortPx({ │ │ │ │ │ + x: 0, │ │ │ │ │ + y: 0 │ │ │ │ │ + }); │ │ │ │ │ + var br = this.getLonLatFromViewPortPx({ │ │ │ │ │ + x: size.w, │ │ │ │ │ + y: size.h │ │ │ │ │ + }); │ │ │ │ │ + if (tl != null && br != null) { │ │ │ │ │ + return new OpenLayers.Bounds(tl.lon, br.lat, br.lon, tl.lat) │ │ │ │ │ + } else { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - mergeWithDefaultFilter: function(filter) { │ │ │ │ │ - var merged; │ │ │ │ │ - if (filter && this.defaultFilter) { │ │ │ │ │ - merged = new OpenLayers.Filter.Logical({ │ │ │ │ │ - type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ - filters: [this.defaultFilter, filter] │ │ │ │ │ - }) │ │ │ │ │ + getZoomForResolution: function(resolution) { │ │ │ │ │ + if (this.resolutions != null) { │ │ │ │ │ + return OpenLayers.Layer.prototype.getZoomForResolution.apply(this, arguments) │ │ │ │ │ } else { │ │ │ │ │ - merged = filter || this.defaultFilter || undefined │ │ │ │ │ + var extent = OpenLayers.Layer.prototype.getExtent.apply(this, []); │ │ │ │ │ + return this.getZoomForExtent(extent) │ │ │ │ │ } │ │ │ │ │ - return merged │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.options = null; │ │ │ │ │ - this.format = null │ │ │ │ │ + getOLZoomFromMapObjectZoom: function(moZoom) { │ │ │ │ │ + var zoom = null; │ │ │ │ │ + if (moZoom != null) { │ │ │ │ │ + zoom = moZoom - this.minZoomLevel; │ │ │ │ │ + if (this.map.baseLayer !== this) { │ │ │ │ │ + zoom = this.map.baseLayer.getZoomForResolution(this.getResolutionForZoom(zoom)) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return zoom │ │ │ │ │ }, │ │ │ │ │ - read: function(options) { │ │ │ │ │ + getMapObjectZoomFromOLZoom: function(olZoom) { │ │ │ │ │ + var zoom = null; │ │ │ │ │ + if (olZoom != null) { │ │ │ │ │ + zoom = olZoom + this.minZoomLevel; │ │ │ │ │ + if (this.map.baseLayer !== this) { │ │ │ │ │ + zoom = this.getZoomForResolution(this.map.baseLayer.getResolutionForZoom(zoom)) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return zoom │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.FixedZoomLevels" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Google = OpenLayers.Class(OpenLayers.Layer.EventPane, OpenLayers.Layer.FixedZoomLevels, { │ │ │ │ │ + MIN_ZOOM_LEVEL: 0, │ │ │ │ │ + MAX_ZOOM_LEVEL: 21, │ │ │ │ │ + RESOLUTIONS: [1.40625, .703125, .3515625, .17578125, .087890625, .0439453125, .02197265625, .010986328125, .0054931640625, .00274658203125, .001373291015625, .0006866455078125, .00034332275390625, .000171661376953125, 858306884765625e-19, 4291534423828125e-20, 2145767211914062e-20, 1072883605957031e-20, 536441802978515e-20, 268220901489257e-20, 1341104507446289e-21, 6.705522537231445e-7], │ │ │ │ │ + type: null, │ │ │ │ │ + wrapDateLine: true, │ │ │ │ │ + sphericalMercator: false, │ │ │ │ │ + version: null, │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ options = options || {}; │ │ │ │ │ - options.filter = this.mergeWithDefaultFilter(options.filter) │ │ │ │ │ + if (!options.version) { │ │ │ │ │ + options.version = typeof GMap2 === "function" ? "2" : "3" │ │ │ │ │ + } │ │ │ │ │ + var mixin = OpenLayers.Layer.Google["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (mixin) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, mixin) │ │ │ │ │ + } else { │ │ │ │ │ + throw "Unsupported Google Maps API version: " + options.version │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, mixin.DEFAULTS); │ │ │ │ │ + if (options.maxExtent) { │ │ │ │ │ + options.maxExtent = options.maxExtent.clone() │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ + OpenLayers.Layer.FixedZoomLevels.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + OpenLayers.Util.extend(this, OpenLayers.Layer.SphericalMercator); │ │ │ │ │ + this.initMercatorParameters() │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - create: function() {}, │ │ │ │ │ - update: function() {}, │ │ │ │ │ - delete: function() {}, │ │ │ │ │ - commit: function() {}, │ │ │ │ │ - abort: function(response) {}, │ │ │ │ │ - createCallback: function(method, response, options) { │ │ │ │ │ - return OpenLayers.Function.bind(function() { │ │ │ │ │ - method.apply(this, [response, options]) │ │ │ │ │ - }, this) │ │ │ │ │ + clone: function() { │ │ │ │ │ + return new OpenLayers.Layer.Google(this.name, this.getOptions()) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Protocol.Response = OpenLayers.Class({ │ │ │ │ │ - code: null, │ │ │ │ │ - requestType: null, │ │ │ │ │ - last: true, │ │ │ │ │ - features: null, │ │ │ │ │ - data: null, │ │ │ │ │ - reqFeatures: null, │ │ │ │ │ - priv: null, │ │ │ │ │ - error: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options) │ │ │ │ │ + setVisibility: function(visible) { │ │ │ │ │ + var opacity = this.opacity == null ? 1 : this.opacity; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.setVisibility.apply(this, arguments); │ │ │ │ │ + this.setOpacity(opacity) │ │ │ │ │ }, │ │ │ │ │ - success: function() { │ │ │ │ │ - return this.code > 0 │ │ │ │ │ + display: function(visible) { │ │ │ │ │ + if (!this._dragging) { │ │ │ │ │ + this.setGMapVisibility(visible) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.display.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.Response" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Protocol.Response.SUCCESS = 1; │ │ │ │ │ -OpenLayers.Protocol.Response.FAILURE = 0; │ │ │ │ │ -OpenLayers.ProxyHost = ""; │ │ │ │ │ -if (!OpenLayers.Request) { │ │ │ │ │ - OpenLayers.Request = {} │ │ │ │ │ -} │ │ │ │ │ -OpenLayers.Util.extend(OpenLayers.Request, { │ │ │ │ │ - DEFAULT_CONFIG: { │ │ │ │ │ - method: "GET", │ │ │ │ │ - url: window.location.href, │ │ │ │ │ - async: true, │ │ │ │ │ - user: undefined, │ │ │ │ │ - password: undefined, │ │ │ │ │ - params: null, │ │ │ │ │ - proxy: OpenLayers.ProxyHost, │ │ │ │ │ - headers: {}, │ │ │ │ │ - data: null, │ │ │ │ │ - callback: function() {}, │ │ │ │ │ - success: null, │ │ │ │ │ - failure: null, │ │ │ │ │ - scope: null │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + this._dragging = dragging; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + delete this._dragging │ │ │ │ │ }, │ │ │ │ │ - URL_SPLIT_REGEX: /([^:]*:)\/\/([^:]*:?[^@]*@)?([^:\/\?]*):?([^\/\?]*)/, │ │ │ │ │ - events: new OpenLayers.Events(this), │ │ │ │ │ - makeSameOrigin: function(url, proxy) { │ │ │ │ │ - var sameOrigin = url.indexOf("http") !== 0; │ │ │ │ │ - var urlParts = !sameOrigin && url.match(this.URL_SPLIT_REGEX); │ │ │ │ │ - if (urlParts) { │ │ │ │ │ - var location = window.location; │ │ │ │ │ - sameOrigin = urlParts[1] == location.protocol && urlParts[3] == location.hostname; │ │ │ │ │ - var uPort = urlParts[4], │ │ │ │ │ - lPort = location.port; │ │ │ │ │ - if (uPort != 80 && uPort != "" || lPort != "80" && lPort != "") { │ │ │ │ │ - sameOrigin = sameOrigin && uPort == lPort │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + if (opacity !== this.opacity) { │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "opacity" │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ + this.opacity = opacity │ │ │ │ │ } │ │ │ │ │ - if (!sameOrigin) { │ │ │ │ │ - if (proxy) { │ │ │ │ │ - if (typeof proxy == "function") { │ │ │ │ │ - url = proxy(url) │ │ │ │ │ - } else { │ │ │ │ │ - url = proxy + encodeURIComponent(url) │ │ │ │ │ - } │ │ │ │ │ + if (this.getVisibility()) { │ │ │ │ │ + var container = this.getMapContainer(); │ │ │ │ │ + OpenLayers.Util.modifyDOMElement(container, null, null, null, null, null, null, opacity) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.setGMapVisibility(false); │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache && cache.count <= 1) { │ │ │ │ │ + this.removeGMapElements() │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return url │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - issue: function(config) { │ │ │ │ │ - var defaultConfig = OpenLayers.Util.extend(this.DEFAULT_CONFIG, { │ │ │ │ │ - proxy: OpenLayers.ProxyHost │ │ │ │ │ - }); │ │ │ │ │ - config = config || {}; │ │ │ │ │ - config.headers = config.headers || {}; │ │ │ │ │ - config = OpenLayers.Util.applyDefaults(config, defaultConfig); │ │ │ │ │ - config.headers = OpenLayers.Util.applyDefaults(config.headers, defaultConfig.headers); │ │ │ │ │ - var customRequestedWithHeader = false, │ │ │ │ │ - headerKey; │ │ │ │ │ - for (headerKey in config.headers) { │ │ │ │ │ - if (config.headers.hasOwnProperty(headerKey)) { │ │ │ │ │ - if (headerKey.toLowerCase() === "x-requested-with") { │ │ │ │ │ - customRequestedWithHeader = true │ │ │ │ │ - } │ │ │ │ │ + removeGMapElements: function() { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + var container = this.mapObject && this.getMapContainer(); │ │ │ │ │ + if (container && container.parentNode) { │ │ │ │ │ + container.parentNode.removeChild(container) │ │ │ │ │ + } │ │ │ │ │ + var termsOfUse = cache.termsOfUse; │ │ │ │ │ + if (termsOfUse && termsOfUse.parentNode) { │ │ │ │ │ + termsOfUse.parentNode.removeChild(termsOfUse) │ │ │ │ │ + } │ │ │ │ │ + var poweredBy = cache.poweredBy; │ │ │ │ │ + if (poweredBy && poweredBy.parentNode) { │ │ │ │ │ + poweredBy.parentNode.removeChild(poweredBy) │ │ │ │ │ + } │ │ │ │ │ + if (this.mapObject && window.google && google.maps && google.maps.event && google.maps.event.clearListeners) { │ │ │ │ │ + google.maps.event.clearListeners(this.mapObject, "tilesloaded") │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (customRequestedWithHeader === false) { │ │ │ │ │ - config.headers["X-Requested-With"] = "XMLHttpRequest" │ │ │ │ │ + }, │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + if (this.visibility && this.mapObject) { │ │ │ │ │ + this.setGMapVisibility(false) │ │ │ │ │ } │ │ │ │ │ - var request = new OpenLayers.Request.XMLHttpRequest; │ │ │ │ │ - var url = OpenLayers.Util.urlAppend(config.url, OpenLayers.Util.getParameterString(config.params || {})); │ │ │ │ │ - url = OpenLayers.Request.makeSameOrigin(url, config.proxy); │ │ │ │ │ - request.open(config.method, url, config.async, config.user, config.password); │ │ │ │ │ - for (var header in config.headers) { │ │ │ │ │ - request.setRequestHeader(header, config.headers[header]) │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + if (cache.count <= 1) { │ │ │ │ │ + this.removeGMapElements(); │ │ │ │ │ + delete OpenLayers.Layer.Google.cache[map.id] │ │ │ │ │ + } else { │ │ │ │ │ + --cache.count │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var events = this.events; │ │ │ │ │ - var self = this; │ │ │ │ │ - request.onreadystatechange = function() { │ │ │ │ │ - if (request.readyState == OpenLayers.Request.XMLHttpRequest.DONE) { │ │ │ │ │ - var proceed = events.triggerEvent("complete", { │ │ │ │ │ - request: request, │ │ │ │ │ - config: config, │ │ │ │ │ - requestUrl: url │ │ │ │ │ - }); │ │ │ │ │ - if (proceed !== false) { │ │ │ │ │ - self.runCallbacks({ │ │ │ │ │ - request: request, │ │ │ │ │ - config: config, │ │ │ │ │ - requestUrl: url │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ + delete this.termsOfUse; │ │ │ │ │ + delete this.poweredBy; │ │ │ │ │ + delete this.mapObject; │ │ │ │ │ + delete this.dragObject; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.removeMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + getOLBoundsFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + var olBounds = null; │ │ │ │ │ + if (moBounds != null) { │ │ │ │ │ + var sw = moBounds.getSouthWest(); │ │ │ │ │ + var ne = moBounds.getNorthEast(); │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + sw = this.forwardMercator(sw.lng(), sw.lat()); │ │ │ │ │ + ne = this.forwardMercator(ne.lng(), ne.lat()) │ │ │ │ │ + } else { │ │ │ │ │ + sw = new OpenLayers.LonLat(sw.lng(), sw.lat()); │ │ │ │ │ + ne = new OpenLayers.LonLat(ne.lng(), ne.lat()) │ │ │ │ │ } │ │ │ │ │ - }; │ │ │ │ │ - if (config.async === false) { │ │ │ │ │ - request.send(config.data) │ │ │ │ │ - } else { │ │ │ │ │ - window.setTimeout(function() { │ │ │ │ │ - if (request.readyState !== 0) { │ │ │ │ │ - request.send(config.data) │ │ │ │ │ - } │ │ │ │ │ - }, 0) │ │ │ │ │ + olBounds = new OpenLayers.Bounds(sw.lon, sw.lat, ne.lon, ne.lat) │ │ │ │ │ } │ │ │ │ │ - return request │ │ │ │ │ + return olBounds │ │ │ │ │ }, │ │ │ │ │ - runCallbacks: function(options) { │ │ │ │ │ - var request = options.request; │ │ │ │ │ - var config = options.config; │ │ │ │ │ - var complete = config.scope ? OpenLayers.Function.bind(config.callback, config.scope) : config.callback; │ │ │ │ │ - var success; │ │ │ │ │ - if (config.success) { │ │ │ │ │ - success = config.scope ? OpenLayers.Function.bind(config.success, config.scope) : config.success │ │ │ │ │ + getWarningHTML: function() { │ │ │ │ │ + return OpenLayers.i18n("googleWarning") │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectCenter: function() { │ │ │ │ │ + return this.mapObject.getCenter() │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectZoom: function() { │ │ │ │ │ + return this.mapObject.getZoom() │ │ │ │ │ + }, │ │ │ │ │ + getLongitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + return this.sphericalMercator ? this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lon : moLonLat.lng() │ │ │ │ │ + }, │ │ │ │ │ + getLatitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + var lat = this.sphericalMercator ? this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lat : moLonLat.lat(); │ │ │ │ │ + return lat │ │ │ │ │ + }, │ │ │ │ │ + getXFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return moPixel.x │ │ │ │ │ + }, │ │ │ │ │ + getYFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return moPixel.y │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Google" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Google.cache = {}; │ │ │ │ │ +OpenLayers.Layer.Google.v2 = { │ │ │ │ │ + termsOfUse: null, │ │ │ │ │ + poweredBy: null, │ │ │ │ │ + dragObject: null, │ │ │ │ │ + loadMapObject: function() { │ │ │ │ │ + if (!this.type) { │ │ │ │ │ + this.type = G_NORMAL_MAP │ │ │ │ │ } │ │ │ │ │ - var failure; │ │ │ │ │ - if (config.failure) { │ │ │ │ │ - failure = config.scope ? OpenLayers.Function.bind(config.failure, config.scope) : config.failure │ │ │ │ │ + var mapObject, termsOfUse, poweredBy; │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + mapObject = cache.mapObject; │ │ │ │ │ + termsOfUse = cache.termsOfUse; │ │ │ │ │ + poweredBy = cache.poweredBy; │ │ │ │ │ + ++cache.count │ │ │ │ │ + } else { │ │ │ │ │ + var container = this.map.viewPortDiv; │ │ │ │ │ + var div = document.createElement("div"); │ │ │ │ │ + div.id = this.map.id + "_GMap2Container"; │ │ │ │ │ + div.style.position = "absolute"; │ │ │ │ │ + div.style.width = "100%"; │ │ │ │ │ + div.style.height = "100%"; │ │ │ │ │ + container.appendChild(div); │ │ │ │ │ + try { │ │ │ │ │ + mapObject = new GMap2(div); │ │ │ │ │ + termsOfUse = div.lastChild; │ │ │ │ │ + container.appendChild(termsOfUse); │ │ │ │ │ + termsOfUse.style.zIndex = "1100"; │ │ │ │ │ + termsOfUse.style.right = ""; │ │ │ │ │ + termsOfUse.style.bottom = ""; │ │ │ │ │ + termsOfUse.className = "olLayerGoogleCopyright"; │ │ │ │ │ + poweredBy = div.lastChild; │ │ │ │ │ + container.appendChild(poweredBy); │ │ │ │ │ + poweredBy.style.zIndex = "1100"; │ │ │ │ │ + poweredBy.style.right = ""; │ │ │ │ │ + poweredBy.style.bottom = ""; │ │ │ │ │ + poweredBy.className = "olLayerGooglePoweredBy gmnoprint" │ │ │ │ │ + } catch (e) { │ │ │ │ │ + throw e │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.Google.cache[this.map.id] = { │ │ │ │ │ + mapObject: mapObject, │ │ │ │ │ + termsOfUse: termsOfUse, │ │ │ │ │ + poweredBy: poweredBy, │ │ │ │ │ + count: 1 │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (OpenLayers.Util.createUrlObject(config.url).protocol == "file:" && request.responseText) { │ │ │ │ │ - request.status = 200 │ │ │ │ │ + this.mapObject = mapObject; │ │ │ │ │ + this.termsOfUse = termsOfUse; │ │ │ │ │ + this.poweredBy = poweredBy; │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.mapObject.getMapTypes(), this.type) === -1) { │ │ │ │ │ + this.mapObject.addMapType(this.type) │ │ │ │ │ } │ │ │ │ │ - complete(request); │ │ │ │ │ - if (!request.status || request.status >= 200 && request.status < 300) { │ │ │ │ │ - this.events.triggerEvent("success", options); │ │ │ │ │ - if (success) { │ │ │ │ │ - success(request) │ │ │ │ │ + if (typeof mapObject.getDragObject == "function") { │ │ │ │ │ + this.dragObject = mapObject.getDragObject() │ │ │ │ │ + } else { │ │ │ │ │ + this.dragPanMapObject = null │ │ │ │ │ + } │ │ │ │ │ + if (this.isBaseLayer === false) { │ │ │ │ │ + this.setGMapVisibility(this.div.style.display !== "none") │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + if (this.visibility && this.mapObject.isLoaded()) { │ │ │ │ │ + this.mapObject.checkResize() │ │ │ │ │ + } else { │ │ │ │ │ + if (!this._resized) { │ │ │ │ │ + var layer = this; │ │ │ │ │ + var handle = GEvent.addListener(this.mapObject, "load", function() { │ │ │ │ │ + GEvent.removeListener(handle); │ │ │ │ │ + delete layer._resized; │ │ │ │ │ + layer.mapObject.checkResize(); │ │ │ │ │ + layer.moveTo(layer.map.getCenter(), layer.map.getZoom()) │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ + this._resized = true │ │ │ │ │ } │ │ │ │ │ - if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ - this.events.triggerEvent("failure", options); │ │ │ │ │ - if (failure) { │ │ │ │ │ - failure(request) │ │ │ │ │ + }, │ │ │ │ │ + setGMapVisibility: function(visible) { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + var container = this.mapObject.getContainer(); │ │ │ │ │ + if (visible === true) { │ │ │ │ │ + this.mapObject.setMapType(this.type); │ │ │ │ │ + container.style.display = ""; │ │ │ │ │ + this.termsOfUse.style.left = ""; │ │ │ │ │ + this.termsOfUse.style.display = ""; │ │ │ │ │ + this.poweredBy.style.display = ""; │ │ │ │ │ + cache.displayed = this.id │ │ │ │ │ + } else { │ │ │ │ │ + if (cache.displayed === this.id) { │ │ │ │ │ + delete cache.displayed │ │ │ │ │ + } │ │ │ │ │ + if (!cache.displayed) { │ │ │ │ │ + container.style.display = "none"; │ │ │ │ │ + this.termsOfUse.style.display = "none"; │ │ │ │ │ + this.termsOfUse.style.left = "-9999px"; │ │ │ │ │ + this.poweredBy.style.display = "none" │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - GET: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "GET" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config) │ │ │ │ │ + getMapContainer: function() { │ │ │ │ │ + return this.mapObject.getContainer() │ │ │ │ │ }, │ │ │ │ │ - POST: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "POST" │ │ │ │ │ - }); │ │ │ │ │ - config.headers = config.headers ? config.headers : {}; │ │ │ │ │ - if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ - config.headers["Content-Type"] = "application/xml" │ │ │ │ │ + getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ + var moBounds = null; │ │ │ │ │ + if (olBounds != null) { │ │ │ │ │ + var sw = this.sphericalMercator ? this.inverseMercator(olBounds.bottom, olBounds.left) : new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ + var ne = this.sphericalMercator ? this.inverseMercator(olBounds.top, olBounds.right) : new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ + moBounds = new GLatLngBounds(new GLatLng(sw.lat, sw.lon), new GLatLng(ne.lat, ne.lon)) │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Request.issue(config) │ │ │ │ │ + return moBounds │ │ │ │ │ }, │ │ │ │ │ - PUT: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "PUT" │ │ │ │ │ - }); │ │ │ │ │ - config.headers = config.headers ? config.headers : {}; │ │ │ │ │ - if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ - config.headers["Content-Type"] = "application/xml" │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Request.issue(config) │ │ │ │ │ + setMapObjectCenter: function(center, zoom) { │ │ │ │ │ + this.mapObject.setCenter(center, zoom) │ │ │ │ │ }, │ │ │ │ │ - DELETE: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "DELETE" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config) │ │ │ │ │ + dragPanMapObject: function(dX, dY) { │ │ │ │ │ + this.dragObject.moveBy(new GSize(-dX, dY)) │ │ │ │ │ }, │ │ │ │ │ - HEAD: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "HEAD" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config) │ │ │ │ │ + getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return this.mapObject.fromContainerPixelToLatLng(moPixel) │ │ │ │ │ }, │ │ │ │ │ - OPTIONS: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "OPTIONS" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config) │ │ │ │ │ - } │ │ │ │ │ -}); │ │ │ │ │ -(function() { │ │ │ │ │ - var oXMLHttpRequest = window.XMLHttpRequest; │ │ │ │ │ - var bGecko = !!window.controllers, │ │ │ │ │ - bIE = window.document.all && !window.opera, │ │ │ │ │ - bIE7 = bIE && window.navigator.userAgent.match(/MSIE 7.0/); │ │ │ │ │ - │ │ │ │ │ - function fXMLHttpRequest() { │ │ │ │ │ - this._object = oXMLHttpRequest && !bIE7 ? new oXMLHttpRequest : new window.ActiveXObject("Microsoft.XMLHTTP"); │ │ │ │ │ - this._listeners = [] │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function cXMLHttpRequest() { │ │ │ │ │ - return new fXMLHttpRequest │ │ │ │ │ + getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + return this.mapObject.fromLatLngToContainerPixel(moLonLat) │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + return this.mapObject.getBoundsZoomLevel(moBounds) │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ + var gLatLng; │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ + gLatLng = new GLatLng(lonlat.lat, lonlat.lon) │ │ │ │ │ + } else { │ │ │ │ │ + gLatLng = new GLatLng(lat, lon) │ │ │ │ │ + } │ │ │ │ │ + return gLatLng │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ + return new GPoint(x, y) │ │ │ │ │ } │ │ │ │ │ - cXMLHttpRequest.prototype = fXMLHttpRequest.prototype; │ │ │ │ │ - if (bGecko && oXMLHttpRequest.wrapped) cXMLHttpRequest.wrapped = oXMLHttpRequest.wrapped; │ │ │ │ │ - cXMLHttpRequest.UNSENT = 0; │ │ │ │ │ - cXMLHttpRequest.OPENED = 1; │ │ │ │ │ - cXMLHttpRequest.HEADERS_RECEIVED = 2; │ │ │ │ │ - cXMLHttpRequest.LOADING = 3; │ │ │ │ │ - cXMLHttpRequest.DONE = 4; │ │ │ │ │ - cXMLHttpRequest.prototype.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ - cXMLHttpRequest.prototype.responseText = ""; │ │ │ │ │ - cXMLHttpRequest.prototype.responseXML = null; │ │ │ │ │ - cXMLHttpRequest.prototype.status = 0; │ │ │ │ │ - cXMLHttpRequest.prototype.statusText = ""; │ │ │ │ │ - cXMLHttpRequest.prototype.priority = "NORMAL"; │ │ │ │ │ - cXMLHttpRequest.prototype.onreadystatechange = null; │ │ │ │ │ - cXMLHttpRequest.onreadystatechange = null; │ │ │ │ │ - cXMLHttpRequest.onopen = null; │ │ │ │ │ - cXMLHttpRequest.onsend = null; │ │ │ │ │ - cXMLHttpRequest.onabort = null; │ │ │ │ │ - cXMLHttpRequest.prototype.open = function(sMethod, sUrl, bAsync, sUser, sPassword) { │ │ │ │ │ - delete this._headers; │ │ │ │ │ - if (arguments.length < 3) bAsync = true; │ │ │ │ │ - this._async = bAsync; │ │ │ │ │ - var oRequest = this, │ │ │ │ │ - nState = this.readyState, │ │ │ │ │ - fOnUnload; │ │ │ │ │ - if (bIE && bAsync) { │ │ │ │ │ - fOnUnload = function() { │ │ │ │ │ - if (nState != cXMLHttpRequest.DONE) { │ │ │ │ │ - fCleanTransport(oRequest); │ │ │ │ │ - oRequest.abort() │ │ │ │ │ - } │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Layer.Google.v3 = { │ │ │ │ │ + DEFAULTS: { │ │ │ │ │ + sphericalMercator: true, │ │ │ │ │ + projection: "EPSG:900913" │ │ │ │ │ + }, │ │ │ │ │ + animationEnabled: true, │ │ │ │ │ + loadMapObject: function() { │ │ │ │ │ + if (!this.type) { │ │ │ │ │ + this.type = google.maps.MapTypeId.ROADMAP │ │ │ │ │ + } │ │ │ │ │ + var mapObject; │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + mapObject = cache.mapObject; │ │ │ │ │ + ++cache.count │ │ │ │ │ + } else { │ │ │ │ │ + var center = this.map.getCenter(); │ │ │ │ │ + var container = document.createElement("div"); │ │ │ │ │ + container.className = "olForeignContainer"; │ │ │ │ │ + container.style.width = "100%"; │ │ │ │ │ + container.style.height = "100%"; │ │ │ │ │ + mapObject = new google.maps.Map(container, { │ │ │ │ │ + center: center ? new google.maps.LatLng(center.lat, center.lon) : new google.maps.LatLng(0, 0), │ │ │ │ │ + zoom: this.map.getZoom() || 0, │ │ │ │ │ + mapTypeId: this.type, │ │ │ │ │ + disableDefaultUI: true, │ │ │ │ │ + keyboardShortcuts: false, │ │ │ │ │ + draggable: false, │ │ │ │ │ + disableDoubleClickZoom: true, │ │ │ │ │ + scrollwheel: false, │ │ │ │ │ + streetViewControl: false │ │ │ │ │ + }); │ │ │ │ │ + var googleControl = document.createElement("div"); │ │ │ │ │ + googleControl.style.width = "100%"; │ │ │ │ │ + googleControl.style.height = "100%"; │ │ │ │ │ + mapObject.controls[google.maps.ControlPosition.TOP_LEFT].push(googleControl); │ │ │ │ │ + cache = { │ │ │ │ │ + googleControl: googleControl, │ │ │ │ │ + mapObject: mapObject, │ │ │ │ │ + count: 1 │ │ │ │ │ }; │ │ │ │ │ - window.attachEvent("onunload", fOnUnload) │ │ │ │ │ + OpenLayers.Layer.Google.cache[this.map.id] = cache │ │ │ │ │ } │ │ │ │ │ - if (cXMLHttpRequest.onopen) cXMLHttpRequest.onopen.apply(this, arguments); │ │ │ │ │ - if (arguments.length > 4) this._object.open(sMethod, sUrl, bAsync, sUser, sPassword); │ │ │ │ │ - else if (arguments.length > 3) this._object.open(sMethod, sUrl, bAsync, sUser); │ │ │ │ │ - else this._object.open(sMethod, sUrl, bAsync); │ │ │ │ │ - this.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ - fReadyStateChange(this); │ │ │ │ │ - this._object.onreadystatechange = function() { │ │ │ │ │ - if (bGecko && !bAsync) return; │ │ │ │ │ - oRequest.readyState = oRequest._object.readyState; │ │ │ │ │ - fSynchronizeValues(oRequest); │ │ │ │ │ - if (oRequest._aborted) { │ │ │ │ │ - oRequest.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ - return │ │ │ │ │ + this.mapObject = mapObject; │ │ │ │ │ + this.setGMapVisibility(this.visibility) │ │ │ │ │ + }, │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + if (this.visibility) { │ │ │ │ │ + google.maps.event.trigger(this.mapObject, "resize") │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setGMapVisibility: function(visible) { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + var map = this.map; │ │ │ │ │ + if (cache) { │ │ │ │ │ + var type = this.type; │ │ │ │ │ + var layers = map.layers; │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = layers[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Google && layer.visibility === true && layer.inRange === true) { │ │ │ │ │ + type = layer.type; │ │ │ │ │ + visible = true; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (oRequest.readyState == cXMLHttpRequest.DONE) { │ │ │ │ │ - delete oRequest._data; │ │ │ │ │ - fCleanTransport(oRequest); │ │ │ │ │ - if (bIE && bAsync) window.detachEvent("onunload", fOnUnload) │ │ │ │ │ + var container = this.mapObject.getDiv(); │ │ │ │ │ + if (visible === true) { │ │ │ │ │ + if (container.parentNode !== map.div) { │ │ │ │ │ + if (!cache.rendered) { │ │ │ │ │ + var me = this; │ │ │ │ │ + google.maps.event.addListenerOnce(this.mapObject, "tilesloaded", function() { │ │ │ │ │ + cache.rendered = true; │ │ │ │ │ + me.setGMapVisibility(me.getVisibility()); │ │ │ │ │ + me.moveTo(me.map.getCenter()) │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + map.div.appendChild(container); │ │ │ │ │ + cache.googleControl.appendChild(map.viewPortDiv); │ │ │ │ │ + google.maps.event.trigger(this.mapObject, "resize") │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.mapObject.setMapTypeId(type) │ │ │ │ │ + } else if (cache.googleControl.hasChildNodes()) { │ │ │ │ │ + map.div.appendChild(map.viewPortDiv); │ │ │ │ │ + map.div.removeChild(container) │ │ │ │ │ } │ │ │ │ │ - if (nState != oRequest.readyState) fReadyStateChange(oRequest); │ │ │ │ │ - nState = oRequest.readyState │ │ │ │ │ } │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - function fXMLHttpRequest_send(oRequest) { │ │ │ │ │ - oRequest._object.send(oRequest._data); │ │ │ │ │ - if (bGecko && !oRequest._async) { │ │ │ │ │ - oRequest.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ - fSynchronizeValues(oRequest); │ │ │ │ │ - while (oRequest.readyState < cXMLHttpRequest.DONE) { │ │ │ │ │ - oRequest.readyState++; │ │ │ │ │ - fReadyStateChange(oRequest); │ │ │ │ │ - if (oRequest._aborted) return │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ + getMapContainer: function() { │ │ │ │ │ + return this.mapObject.getDiv() │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ + var moBounds = null; │ │ │ │ │ + if (olBounds != null) { │ │ │ │ │ + var sw = this.sphericalMercator ? this.inverseMercator(olBounds.bottom, olBounds.left) : new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ + var ne = this.sphericalMercator ? this.inverseMercator(olBounds.top, olBounds.right) : new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ + moBounds = new google.maps.LatLngBounds(new google.maps.LatLng(sw.lat, sw.lon), new google.maps.LatLng(ne.lat, ne.lon)) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - cXMLHttpRequest.prototype.send = function(vData) { │ │ │ │ │ - if (cXMLHttpRequest.onsend) cXMLHttpRequest.onsend.apply(this, arguments); │ │ │ │ │ - if (!arguments.length) vData = null; │ │ │ │ │ - if (vData && vData.nodeType) { │ │ │ │ │ - vData = window.XMLSerializer ? (new window.XMLSerializer).serializeToString(vData) : vData.xml; │ │ │ │ │ - if (!this._headers["Content-Type"]) this._object.setRequestHeader("Content-Type", "application/xml") │ │ │ │ │ + return moBounds │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + var size = this.map.getSize(); │ │ │ │ │ + var lon = this.getLongitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ + var lat = this.getLatitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ + var res = this.map.getResolution(); │ │ │ │ │ + var delta_x = moPixel.x - size.w / 2; │ │ │ │ │ + var delta_y = moPixel.y - size.h / 2; │ │ │ │ │ + var lonlat = new OpenLayers.LonLat(lon + delta_x * res, lat - delta_y * res); │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + lonlat = lonlat.wrapDateLine(this.maxExtent) │ │ │ │ │ } │ │ │ │ │ - this._data = vData; │ │ │ │ │ - fXMLHttpRequest_send(this) │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.abort = function() { │ │ │ │ │ - if (cXMLHttpRequest.onabort) cXMLHttpRequest.onabort.apply(this, arguments); │ │ │ │ │ - if (this.readyState > cXMLHttpRequest.UNSENT) this._aborted = true; │ │ │ │ │ - this._object.abort(); │ │ │ │ │ - fCleanTransport(this); │ │ │ │ │ - this.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ - delete this._data │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.getAllResponseHeaders = function() { │ │ │ │ │ - return this._object.getAllResponseHeaders() │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.getResponseHeader = function(sName) { │ │ │ │ │ - return this._object.getResponseHeader(sName) │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.setRequestHeader = function(sName, sValue) { │ │ │ │ │ - if (!this._headers) this._headers = {}; │ │ │ │ │ - this._headers[sName] = sValue; │ │ │ │ │ - return this._object.setRequestHeader(sName, sValue) │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.addEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ - for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ - if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) return; │ │ │ │ │ - this._listeners.push([sName, fHandler, bUseCapture]) │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.removeEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ - for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ - if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) break; │ │ │ │ │ - if (oListener) this._listeners.splice(nIndex, 1) │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.dispatchEvent = function(oEvent) { │ │ │ │ │ - var oEventPseudo = { │ │ │ │ │ - type: oEvent.type, │ │ │ │ │ - target: this, │ │ │ │ │ - currentTarget: this, │ │ │ │ │ - eventPhase: 2, │ │ │ │ │ - bubbles: oEvent.bubbles, │ │ │ │ │ - cancelable: oEvent.cancelable, │ │ │ │ │ - timeStamp: oEvent.timeStamp, │ │ │ │ │ - stopPropagation: function() {}, │ │ │ │ │ - preventDefault: function() {}, │ │ │ │ │ - initEvent: function() {} │ │ │ │ │ - }; │ │ │ │ │ - if (oEventPseudo.type == "readystatechange" && this.onreadystatechange)(this.onreadystatechange.handleEvent || this.onreadystatechange).apply(this, [oEventPseudo]); │ │ │ │ │ - for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ - if (oListener[0] == oEventPseudo.type && !oListener[2])(oListener[1].handleEvent || oListener[1]).apply(this, [oEventPseudo]) │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.toString = function() { │ │ │ │ │ - return "[" + "object" + " " + "XMLHttpRequest" + "]" │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.toString = function() { │ │ │ │ │ - return "[" + "XMLHttpRequest" + "]" │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - function fReadyStateChange(oRequest) { │ │ │ │ │ - if (cXMLHttpRequest.onreadystatechange) cXMLHttpRequest.onreadystatechange.apply(oRequest); │ │ │ │ │ - oRequest.dispatchEvent({ │ │ │ │ │ - type: "readystatechange", │ │ │ │ │ - bubbles: false, │ │ │ │ │ - cancelable: false, │ │ │ │ │ - timeStamp: new Date + 0 │ │ │ │ │ + return this.getMapObjectLonLatFromLonLat(lonlat.lon, lonlat.lat) │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + var res = this.map.getResolution(); │ │ │ │ │ + var extent = this.map.getExtent(); │ │ │ │ │ + return this.getMapObjectPixelFromXY(1 / res * (lon - extent.left), 1 / res * (extent.top - lat)) │ │ │ │ │ + }, │ │ │ │ │ + setMapObjectCenter: function(center, zoom) { │ │ │ │ │ + if (this.animationEnabled === false && zoom != this.mapObject.zoom) { │ │ │ │ │ + var mapContainer = this.getMapContainer(); │ │ │ │ │ + google.maps.event.addListenerOnce(this.mapObject, "idle", function() { │ │ │ │ │ + mapContainer.style.visibility = "" │ │ │ │ │ + }); │ │ │ │ │ + mapContainer.style.visibility = "hidden" │ │ │ │ │ + } │ │ │ │ │ + this.mapObject.setOptions({ │ │ │ │ │ + center: center, │ │ │ │ │ + zoom: zoom │ │ │ │ │ }) │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function fGetDocument(oRequest) { │ │ │ │ │ - var oDocument = oRequest.responseXML, │ │ │ │ │ - sResponse = oRequest.responseText; │ │ │ │ │ - if (bIE && sResponse && oDocument && !oDocument.documentElement && oRequest.getResponseHeader("Content-Type").match(/[^\/]+\/[^\+]+\+xml/)) { │ │ │ │ │ - oDocument = new window.ActiveXObject("Microsoft.XMLDOM"); │ │ │ │ │ - oDocument.async = false; │ │ │ │ │ - oDocument.validateOnParse = false; │ │ │ │ │ - oDocument.loadXML(sResponse) │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + return this.mapObject.getBoundsZoomLevel(moBounds) │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ + var gLatLng; │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ + gLatLng = new google.maps.LatLng(lonlat.lat, lonlat.lon) │ │ │ │ │ + } else { │ │ │ │ │ + gLatLng = new google.maps.LatLng(lat, lon) │ │ │ │ │ } │ │ │ │ │ - if (oDocument) │ │ │ │ │ - if (bIE && oDocument.parseError != 0 || !oDocument.documentElement || oDocument.documentElement && oDocument.documentElement.tagName == "parsererror") return null; │ │ │ │ │ - return oDocument │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function fSynchronizeValues(oRequest) { │ │ │ │ │ - try { │ │ │ │ │ - oRequest.responseText = oRequest._object.responseText │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - try { │ │ │ │ │ - oRequest.responseXML = fGetDocument(oRequest._object) │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - try { │ │ │ │ │ - oRequest.status = oRequest._object.status │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - try { │ │ │ │ │ - oRequest.statusText = oRequest._object.statusText │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function fCleanTransport(oRequest) { │ │ │ │ │ - oRequest._object.onreadystatechange = new window.Function │ │ │ │ │ + return gLatLng │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ + return new google.maps.Point(x, y) │ │ │ │ │ } │ │ │ │ │ - if (!window.Function.prototype.apply) { │ │ │ │ │ - window.Function.prototype.apply = function(oRequest, oArguments) { │ │ │ │ │ - if (!oArguments) oArguments = []; │ │ │ │ │ - oRequest.__func = this; │ │ │ │ │ - oRequest.__func(oArguments[0], oArguments[1], oArguments[2], oArguments[3], oArguments[4]); │ │ │ │ │ - delete oRequest.__func │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Popup = OpenLayers.Class({ │ │ │ │ │ + events: null, │ │ │ │ │ + id: "", │ │ │ │ │ + lonlat: null, │ │ │ │ │ + div: null, │ │ │ │ │ + contentSize: null, │ │ │ │ │ + size: null, │ │ │ │ │ + contentHTML: null, │ │ │ │ │ + backgroundColor: "", │ │ │ │ │ + opacity: "", │ │ │ │ │ + border: "", │ │ │ │ │ + contentDiv: null, │ │ │ │ │ + groupDiv: null, │ │ │ │ │ + closeDiv: null, │ │ │ │ │ + autoSize: false, │ │ │ │ │ + minSize: null, │ │ │ │ │ + maxSize: null, │ │ │ │ │ + displayClass: "olPopup", │ │ │ │ │ + contentDisplayClass: "olPopupContent", │ │ │ │ │ + padding: 0, │ │ │ │ │ + disableFirefoxOverflowHack: false, │ │ │ │ │ + fixPadding: function() { │ │ │ │ │ + if (typeof this.padding == "number") { │ │ │ │ │ + this.padding = new OpenLayers.Bounds(this.padding, this.padding, this.padding, this.padding) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - if (!OpenLayers.Request) { │ │ │ │ │ - OpenLayers.Request = {} │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Request.XMLHttpRequest = cXMLHttpRequest │ │ │ │ │ -})(); │ │ │ │ │ -OpenLayers.Protocol.HTTP = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ - url: null, │ │ │ │ │ - headers: null, │ │ │ │ │ - params: null, │ │ │ │ │ - callback: null, │ │ │ │ │ - scope: null, │ │ │ │ │ - readWithPOST: false, │ │ │ │ │ - updateWithPOST: false, │ │ │ │ │ - deleteWithPOST: false, │ │ │ │ │ - wildcarded: false, │ │ │ │ │ - srsInBBOX: false, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - this.params = {}; │ │ │ │ │ - this.headers = {}; │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ - var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ - wildcarded: this.wildcarded, │ │ │ │ │ - srsInBBOX: this.srsInBBOX │ │ │ │ │ - }); │ │ │ │ │ - this.filterToParams = function(filter, params) { │ │ │ │ │ - return format.write(filter, params) │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ + panMapIfOutOfView: false, │ │ │ │ │ + keepInMap: false, │ │ │ │ │ + closeOnMove: false, │ │ │ │ │ + map: null, │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, closeBox, closeBoxCallback) { │ │ │ │ │ + if (id == null) { │ │ │ │ │ + id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ + } │ │ │ │ │ + this.id = id; │ │ │ │ │ + this.lonlat = lonlat; │ │ │ │ │ + this.contentSize = contentSize != null ? contentSize : new OpenLayers.Size(OpenLayers.Popup.WIDTH, OpenLayers.Popup.HEIGHT); │ │ │ │ │ + if (contentHTML != null) { │ │ │ │ │ + this.contentHTML = contentHTML │ │ │ │ │ + } │ │ │ │ │ + this.backgroundColor = OpenLayers.Popup.COLOR; │ │ │ │ │ + this.opacity = OpenLayers.Popup.OPACITY; │ │ │ │ │ + this.border = OpenLayers.Popup.BORDER; │ │ │ │ │ + this.div = OpenLayers.Util.createDiv(this.id, null, null, null, null, null, "hidden"); │ │ │ │ │ + this.div.className = this.displayClass; │ │ │ │ │ + var groupDivId = this.id + "_GroupDiv"; │ │ │ │ │ + this.groupDiv = OpenLayers.Util.createDiv(groupDivId, null, null, null, "relative", null, "hidden"); │ │ │ │ │ + var id = this.div.id + "_contentDiv"; │ │ │ │ │ + this.contentDiv = OpenLayers.Util.createDiv(id, null, this.contentSize.clone(), null, "relative"); │ │ │ │ │ + this.contentDiv.className = this.contentDisplayClass; │ │ │ │ │ + this.groupDiv.appendChild(this.contentDiv); │ │ │ │ │ + this.div.appendChild(this.groupDiv); │ │ │ │ │ + if (closeBox) { │ │ │ │ │ + this.addCloseBox(closeBoxCallback) │ │ │ │ │ } │ │ │ │ │ + this.registerEvents() │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - this.params = null; │ │ │ │ │ - this.headers = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ + this.id = null; │ │ │ │ │ + this.lonlat = null; │ │ │ │ │ + this.size = null; │ │ │ │ │ + this.contentHTML = null; │ │ │ │ │ + this.backgroundColor = null; │ │ │ │ │ + this.opacity = null; │ │ │ │ │ + this.border = null; │ │ │ │ │ + if (this.closeOnMove && this.map) { │ │ │ │ │ + this.map.events.unregister("movestart", this, this.hide) │ │ │ │ │ + } │ │ │ │ │ + this.events.destroy(); │ │ │ │ │ + this.events = null; │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + OpenLayers.Event.stopObservingElement(this.closeDiv); │ │ │ │ │ + this.groupDiv.removeChild(this.closeDiv) │ │ │ │ │ + } │ │ │ │ │ + this.closeDiv = null; │ │ │ │ │ + this.div.removeChild(this.groupDiv); │ │ │ │ │ + this.groupDiv = null; │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.removePopup(this) │ │ │ │ │ + } │ │ │ │ │ + this.map = null; │ │ │ │ │ + this.div = null; │ │ │ │ │ + this.autoSize = null; │ │ │ │ │ + this.minSize = null; │ │ │ │ │ + this.maxSize = null; │ │ │ │ │ + this.padding = null; │ │ │ │ │ + this.panMapIfOutOfView = null │ │ │ │ │ }, │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.params = OpenLayers.Util.applyDefaults(options.params, this.options.params); │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - if (options.filter && this.filterToParams) { │ │ │ │ │ - options.params = this.filterToParams(options.filter, options.params) │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + if (px == null) { │ │ │ │ │ + if (this.lonlat != null && this.map != null) { │ │ │ │ │ + px = this.map.getLayerPxFromLonLat(this.lonlat) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var readWithPOST = options.readWithPOST !== undefined ? options.readWithPOST : this.readWithPOST; │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - if (readWithPOST) { │ │ │ │ │ - var headers = options.headers || {}; │ │ │ │ │ - headers["Content-Type"] = "application/x-www-form-urlencoded"; │ │ │ │ │ - resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ - data: OpenLayers.Util.getParameterString(options.params), │ │ │ │ │ - headers: headers │ │ │ │ │ + if (this.closeOnMove) { │ │ │ │ │ + this.map.events.register("movestart", this, this.hide) │ │ │ │ │ + } │ │ │ │ │ + if (!this.disableFirefoxOverflowHack && OpenLayers.BROWSER_NAME == "firefox") { │ │ │ │ │ + this.map.events.register("movestart", this, function() { │ │ │ │ │ + var style = document.defaultView.getComputedStyle(this.contentDiv, null); │ │ │ │ │ + var currentOverflow = style.getPropertyValue("overflow"); │ │ │ │ │ + if (currentOverflow != "hidden") { │ │ │ │ │ + this.contentDiv._oldOverflow = currentOverflow; │ │ │ │ │ + this.contentDiv.style.overflow = "hidden" │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + this.map.events.register("moveend", this, function() { │ │ │ │ │ + var oldOverflow = this.contentDiv._oldOverflow; │ │ │ │ │ + if (oldOverflow) { │ │ │ │ │ + this.contentDiv.style.overflow = oldOverflow; │ │ │ │ │ + this.contentDiv._oldOverflow = null │ │ │ │ │ + } │ │ │ │ │ }) │ │ │ │ │ + } │ │ │ │ │ + this.moveTo(px); │ │ │ │ │ + if (!this.autoSize && !this.size) { │ │ │ │ │ + this.setSize(this.contentSize) │ │ │ │ │ + } │ │ │ │ │ + this.setBackgroundColor(); │ │ │ │ │ + this.setOpacity(); │ │ │ │ │ + this.setBorder(); │ │ │ │ │ + this.setContentHTML(); │ │ │ │ │ + if (this.panMapIfOutOfView) { │ │ │ │ │ + this.panIntoView() │ │ │ │ │ + } │ │ │ │ │ + return this.div │ │ │ │ │ + }, │ │ │ │ │ + updatePosition: function() { │ │ │ │ │ + if (this.lonlat && this.map) { │ │ │ │ │ + var px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ + if (px) { │ │ │ │ │ + this.moveTo(px) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + if (px != null && this.div != null) { │ │ │ │ │ + this.div.style.left = px.x + "px"; │ │ │ │ │ + this.div.style.top = px.y + "px" │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + visible: function() { │ │ │ │ │ + return OpenLayers.Element.visible(this.div) │ │ │ │ │ + }, │ │ │ │ │ + toggle: function() { │ │ │ │ │ + if (this.visible()) { │ │ │ │ │ + this.hide() │ │ │ │ │ } else { │ │ │ │ │ - resp.priv = OpenLayers.Request.GET({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers │ │ │ │ │ - }) │ │ │ │ │ + this.show() │ │ │ │ │ } │ │ │ │ │ - return resp │ │ │ │ │ }, │ │ │ │ │ - handleRead: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options) │ │ │ │ │ + show: function() { │ │ │ │ │ + this.div.style.display = ""; │ │ │ │ │ + if (this.panMapIfOutOfView) { │ │ │ │ │ + this.panIntoView() │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - create: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: features, │ │ │ │ │ - requestType: "create" │ │ │ │ │ - }); │ │ │ │ │ - resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleCreate, resp, options), │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(features) │ │ │ │ │ - }); │ │ │ │ │ - return resp │ │ │ │ │ + hide: function() { │ │ │ │ │ + this.div.style.display = "none" │ │ │ │ │ }, │ │ │ │ │ - handleCreate: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options) │ │ │ │ │ + setSize: function(contentSize) { │ │ │ │ │ + this.size = contentSize.clone(); │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ + var wPadding = contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ + var hPadding = contentDivPadding.top + contentDivPadding.bottom; │ │ │ │ │ + this.fixPadding(); │ │ │ │ │ + wPadding += this.padding.left + this.padding.right; │ │ │ │ │ + hPadding += this.padding.top + this.padding.bottom; │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + var closeDivWidth = parseInt(this.closeDiv.style.width); │ │ │ │ │ + wPadding += closeDivWidth + contentDivPadding.right │ │ │ │ │ + } │ │ │ │ │ + this.size.w += wPadding; │ │ │ │ │ + this.size.h += hPadding; │ │ │ │ │ + if (OpenLayers.BROWSER_NAME == "msie") { │ │ │ │ │ + this.contentSize.w += contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ + this.contentSize.h += contentDivPadding.bottom + contentDivPadding.top │ │ │ │ │ + } │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.div.style.width = this.size.w + "px"; │ │ │ │ │ + this.div.style.height = this.size.h + "px" │ │ │ │ │ + } │ │ │ │ │ + if (this.contentDiv != null) { │ │ │ │ │ + this.contentDiv.style.width = contentSize.w + "px"; │ │ │ │ │ + this.contentDiv.style.height = contentSize.h + "px" │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - update: function(feature, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var url = options.url || feature.url || this.options.url + "/" + feature.fid; │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: feature, │ │ │ │ │ - requestType: "update" │ │ │ │ │ - }); │ │ │ │ │ - var method = this.updateWithPOST ? "POST" : "PUT"; │ │ │ │ │ - resp.priv = OpenLayers.Request[method]({ │ │ │ │ │ - url: url, │ │ │ │ │ - callback: this.createCallback(this.handleUpdate, resp, options), │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(feature) │ │ │ │ │ + updateSize: function() { │ │ │ │ │ + var preparedHTML = "<div class='" + this.contentDisplayClass + "'>" + this.contentDiv.innerHTML + "</div>"; │ │ │ │ │ + var containerElement = this.map ? this.map.div : document.body; │ │ │ │ │ + var realSize = OpenLayers.Util.getRenderedDimensions(preparedHTML, null, { │ │ │ │ │ + displayClass: this.displayClass, │ │ │ │ │ + containerElement: containerElement │ │ │ │ │ }); │ │ │ │ │ - return resp │ │ │ │ │ + var safeSize = this.getSafeContentSize(realSize); │ │ │ │ │ + var newSize = null; │ │ │ │ │ + if (safeSize.equals(realSize)) { │ │ │ │ │ + newSize = realSize │ │ │ │ │ + } else { │ │ │ │ │ + var fixedSize = { │ │ │ │ │ + w: safeSize.w < realSize.w ? safeSize.w : null, │ │ │ │ │ + h: safeSize.h < realSize.h ? safeSize.h : null │ │ │ │ │ + }; │ │ │ │ │ + if (fixedSize.w && fixedSize.h) { │ │ │ │ │ + newSize = safeSize │ │ │ │ │ + } else { │ │ │ │ │ + var clippedSize = OpenLayers.Util.getRenderedDimensions(preparedHTML, fixedSize, { │ │ │ │ │ + displayClass: this.contentDisplayClass, │ │ │ │ │ + containerElement: containerElement │ │ │ │ │ + }); │ │ │ │ │ + var currentOverflow = OpenLayers.Element.getStyle(this.contentDiv, "overflow"); │ │ │ │ │ + if (currentOverflow != "hidden" && clippedSize.equals(safeSize)) { │ │ │ │ │ + var scrollBar = OpenLayers.Util.getScrollbarWidth(); │ │ │ │ │ + if (fixedSize.w) { │ │ │ │ │ + clippedSize.h += scrollBar │ │ │ │ │ + } else { │ │ │ │ │ + clippedSize.w += scrollBar │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + newSize = this.getSafeContentSize(clippedSize) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.setSize(newSize) │ │ │ │ │ }, │ │ │ │ │ - handleUpdate: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options) │ │ │ │ │ + setBackgroundColor: function(color) { │ │ │ │ │ + if (color != undefined) { │ │ │ │ │ + this.backgroundColor = color │ │ │ │ │ + } │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.div.style.backgroundColor = this.backgroundColor │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - delete: function(feature, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var url = options.url || feature.url || this.options.url + "/" + feature.fid; │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: feature, │ │ │ │ │ - requestType: "delete" │ │ │ │ │ - }); │ │ │ │ │ - var method = this.deleteWithPOST ? "POST" : "DELETE"; │ │ │ │ │ - var requestOptions = { │ │ │ │ │ - url: url, │ │ │ │ │ - callback: this.createCallback(this.handleDelete, resp, options), │ │ │ │ │ - headers: options.headers │ │ │ │ │ - }; │ │ │ │ │ - if (this.deleteWithPOST) { │ │ │ │ │ - requestOptions.data = this.format.write(feature) │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + if (opacity != undefined) { │ │ │ │ │ + this.opacity = opacity │ │ │ │ │ + } │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.div.style.opacity = this.opacity; │ │ │ │ │ + this.div.style.filter = "alpha(opacity=" + this.opacity * 100 + ")" │ │ │ │ │ } │ │ │ │ │ - resp.priv = OpenLayers.Request[method](requestOptions); │ │ │ │ │ - return resp │ │ │ │ │ }, │ │ │ │ │ - handleDelete: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options) │ │ │ │ │ + setBorder: function(border) { │ │ │ │ │ + if (border != undefined) { │ │ │ │ │ + this.border = border │ │ │ │ │ + } │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.div.style.border = this.border │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - handleResponse: function(resp, options) { │ │ │ │ │ - var request = resp.priv; │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - if (resp.requestType != "delete") { │ │ │ │ │ - resp.features = this.parseFeatures(request) │ │ │ │ │ - } │ │ │ │ │ - resp.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ - } else { │ │ │ │ │ - resp.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ + setContentHTML: function(contentHTML) { │ │ │ │ │ + if (contentHTML != null) { │ │ │ │ │ + this.contentHTML = contentHTML │ │ │ │ │ + } │ │ │ │ │ + if (this.contentDiv != null && this.contentHTML != null && this.contentHTML != this.contentDiv.innerHTML) { │ │ │ │ │ + this.contentDiv.innerHTML = this.contentHTML; │ │ │ │ │ + if (this.autoSize) { │ │ │ │ │ + this.registerImageListeners(); │ │ │ │ │ + this.updateSize() │ │ │ │ │ } │ │ │ │ │ - options.callback.call(options.scope, resp) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - parseFeatures: function(request) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ + registerImageListeners: function() { │ │ │ │ │ + var onImgLoad = function() { │ │ │ │ │ + if (this.popup.id === null) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + this.popup.updateSize(); │ │ │ │ │ + if (this.popup.visible() && this.popup.panMapIfOutOfView) { │ │ │ │ │ + this.popup.panIntoView() │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stopObserving(this.img, "load", this.img._onImgLoad) │ │ │ │ │ + }; │ │ │ │ │ + var images = this.contentDiv.getElementsByTagName("img"); │ │ │ │ │ + for (var i = 0, len = images.length; i < len; i++) { │ │ │ │ │ + var img = images[i]; │ │ │ │ │ + if (img.width == 0 || img.height == 0) { │ │ │ │ │ + var context = { │ │ │ │ │ + popup: this, │ │ │ │ │ + img: img │ │ │ │ │ + }; │ │ │ │ │ + img._onImgLoad = OpenLayers.Function.bind(onImgLoad, context); │ │ │ │ │ + OpenLayers.Event.observe(img, "load", img._onImgLoad) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null │ │ │ │ │ + }, │ │ │ │ │ + getSafeContentSize: function(size) { │ │ │ │ │ + var safeContentSize = size.clone(); │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ + var wPadding = contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ + var hPadding = contentDivPadding.top + contentDivPadding.bottom; │ │ │ │ │ + this.fixPadding(); │ │ │ │ │ + wPadding += this.padding.left + this.padding.right; │ │ │ │ │ + hPadding += this.padding.top + this.padding.bottom; │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + var closeDivWidth = parseInt(this.closeDiv.style.width); │ │ │ │ │ + wPadding += closeDivWidth + contentDivPadding.right │ │ │ │ │ } │ │ │ │ │ - return this.format.read(doc) │ │ │ │ │ + if (this.minSize) { │ │ │ │ │ + safeContentSize.w = Math.max(safeContentSize.w, this.minSize.w - wPadding); │ │ │ │ │ + safeContentSize.h = Math.max(safeContentSize.h, this.minSize.h - hPadding) │ │ │ │ │ + } │ │ │ │ │ + if (this.maxSize) { │ │ │ │ │ + safeContentSize.w = Math.min(safeContentSize.w, this.maxSize.w - wPadding); │ │ │ │ │ + safeContentSize.h = Math.min(safeContentSize.h, this.maxSize.h - hPadding) │ │ │ │ │ + } │ │ │ │ │ + if (this.map && this.map.size) { │ │ │ │ │ + var extraX = 0, │ │ │ │ │ + extraY = 0; │ │ │ │ │ + if (this.keepInMap && !this.panMapIfOutOfView) { │ │ │ │ │ + var px = this.map.getPixelFromLonLat(this.lonlat); │ │ │ │ │ + switch (this.relativePosition) { │ │ │ │ │ + case "tr": │ │ │ │ │ + extraX = px.x; │ │ │ │ │ + extraY = this.map.size.h - px.y; │ │ │ │ │ + break; │ │ │ │ │ + case "tl": │ │ │ │ │ + extraX = this.map.size.w - px.x; │ │ │ │ │ + extraY = this.map.size.h - px.y; │ │ │ │ │ + break; │ │ │ │ │ + case "bl": │ │ │ │ │ + extraX = this.map.size.w - px.x; │ │ │ │ │ + extraY = px.y; │ │ │ │ │ + break; │ │ │ │ │ + case "br": │ │ │ │ │ + extraX = px.x; │ │ │ │ │ + extraY = px.y; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + extraX = px.x; │ │ │ │ │ + extraY = this.map.size.h - px.y; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var maxY = this.map.size.h - this.map.paddingForPopups.top - this.map.paddingForPopups.bottom - hPadding - extraY; │ │ │ │ │ + var maxX = this.map.size.w - this.map.paddingForPopups.left - this.map.paddingForPopups.right - wPadding - extraX; │ │ │ │ │ + safeContentSize.w = Math.min(safeContentSize.w, maxX); │ │ │ │ │ + safeContentSize.h = Math.min(safeContentSize.h, maxY) │ │ │ │ │ + } │ │ │ │ │ + return safeContentSize │ │ │ │ │ }, │ │ │ │ │ - commit: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = [], │ │ │ │ │ - nResponses = 0; │ │ │ │ │ - var types = {}; │ │ │ │ │ - types[OpenLayers.State.INSERT] = []; │ │ │ │ │ - types[OpenLayers.State.UPDATE] = []; │ │ │ │ │ - types[OpenLayers.State.DELETE] = []; │ │ │ │ │ - var feature, list, requestFeatures = []; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - list = types[feature.state]; │ │ │ │ │ - if (list) { │ │ │ │ │ - list.push(feature); │ │ │ │ │ - requestFeatures.push(feature) │ │ │ │ │ + getContentDivPadding: function() { │ │ │ │ │ + var contentDivPadding = this._contentDivPadding; │ │ │ │ │ + if (!contentDivPadding) { │ │ │ │ │ + if (this.div.parentNode == null) { │ │ │ │ │ + this.div.style.display = "none"; │ │ │ │ │ + document.body.appendChild(this.div) │ │ │ │ │ + } │ │ │ │ │ + contentDivPadding = new OpenLayers.Bounds(OpenLayers.Element.getStyle(this.contentDiv, "padding-left"), OpenLayers.Element.getStyle(this.contentDiv, "padding-bottom"), OpenLayers.Element.getStyle(this.contentDiv, "padding-right"), OpenLayers.Element.getStyle(this.contentDiv, "padding-top")); │ │ │ │ │ + this._contentDivPadding = contentDivPadding; │ │ │ │ │ + if (this.div.parentNode == document.body) { │ │ │ │ │ + document.body.removeChild(this.div); │ │ │ │ │ + this.div.style.display = "" │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var nRequests = (types[OpenLayers.State.INSERT].length > 0 ? 1 : 0) + types[OpenLayers.State.UPDATE].length + types[OpenLayers.State.DELETE].length; │ │ │ │ │ - var success = true; │ │ │ │ │ - var finalResponse = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: requestFeatures │ │ │ │ │ + return contentDivPadding │ │ │ │ │ + }, │ │ │ │ │ + addCloseBox: function(callback) { │ │ │ │ │ + this.closeDiv = OpenLayers.Util.createDiv(this.id + "_close", null, { │ │ │ │ │ + w: 17, │ │ │ │ │ + h: 17 │ │ │ │ │ }); │ │ │ │ │ + this.closeDiv.className = "olPopupCloseBox"; │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ + this.closeDiv.style.right = contentDivPadding.right + "px"; │ │ │ │ │ + this.closeDiv.style.top = contentDivPadding.top + "px"; │ │ │ │ │ + this.groupDiv.appendChild(this.closeDiv); │ │ │ │ │ + var closePopup = callback || function(e) { │ │ │ │ │ + this.hide(); │ │ │ │ │ + OpenLayers.Event.stop(e) │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Event.observe(this.closeDiv, "touchend", OpenLayers.Function.bindAsEventListener(closePopup, this)); │ │ │ │ │ + OpenLayers.Event.observe(this.closeDiv, "click", OpenLayers.Function.bindAsEventListener(closePopup, this)) │ │ │ │ │ + }, │ │ │ │ │ + panIntoView: function() { │ │ │ │ │ + var mapSize = this.map.getSize(); │ │ │ │ │ + var origTL = this.map.getViewPortPxFromLayerPx(new OpenLayers.Pixel(parseInt(this.div.style.left), parseInt(this.div.style.top))); │ │ │ │ │ + var newTL = origTL.clone(); │ │ │ │ │ + if (origTL.x < this.map.paddingForPopups.left) { │ │ │ │ │ + newTL.x = this.map.paddingForPopups.left │ │ │ │ │ + } else if (origTL.x + this.size.w > mapSize.w - this.map.paddingForPopups.right) { │ │ │ │ │ + newTL.x = mapSize.w - this.map.paddingForPopups.right - this.size.w │ │ │ │ │ + } │ │ │ │ │ + if (origTL.y < this.map.paddingForPopups.top) { │ │ │ │ │ + newTL.y = this.map.paddingForPopups.top │ │ │ │ │ + } else if (origTL.y + this.size.h > mapSize.h - this.map.paddingForPopups.bottom) { │ │ │ │ │ + newTL.y = mapSize.h - this.map.paddingForPopups.bottom - this.size.h │ │ │ │ │ + } │ │ │ │ │ + var dx = origTL.x - newTL.x; │ │ │ │ │ + var dy = origTL.y - newTL.y; │ │ │ │ │ + this.map.pan(dx, dy) │ │ │ │ │ + }, │ │ │ │ │ + registerEvents: function() { │ │ │ │ │ + this.events = new OpenLayers.Events(this, this.div, null, true); │ │ │ │ │ │ │ │ │ │ - function insertCallback(response) { │ │ │ │ │ - var len = response.features ? response.features.length : 0; │ │ │ │ │ - var fids = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - fids[i] = response.features[i].fid │ │ │ │ │ + function onTouchstart(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt, true) │ │ │ │ │ + } │ │ │ │ │ + this.events.on({ │ │ │ │ │ + mousedown: this.onmousedown, │ │ │ │ │ + mousemove: this.onmousemove, │ │ │ │ │ + mouseup: this.onmouseup, │ │ │ │ │ + click: this.onclick, │ │ │ │ │ + mouseout: this.onmouseout, │ │ │ │ │ + dblclick: this.ondblclick, │ │ │ │ │ + touchstart: onTouchstart, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + onmousedown: function(evt) { │ │ │ │ │ + this.mousedown = true; │ │ │ │ │ + OpenLayers.Event.stop(evt, true) │ │ │ │ │ + }, │ │ │ │ │ + onmousemove: function(evt) { │ │ │ │ │ + if (this.mousedown) { │ │ │ │ │ + OpenLayers.Event.stop(evt, true) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + onmouseup: function(evt) { │ │ │ │ │ + if (this.mousedown) { │ │ │ │ │ + this.mousedown = false; │ │ │ │ │ + OpenLayers.Event.stop(evt, true) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + onclick: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt, true) │ │ │ │ │ + }, │ │ │ │ │ + onmouseout: function(evt) { │ │ │ │ │ + this.mousedown = false │ │ │ │ │ + }, │ │ │ │ │ + ondblclick: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt, true) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Popup.WIDTH = 200; │ │ │ │ │ +OpenLayers.Popup.HEIGHT = 200; │ │ │ │ │ +OpenLayers.Popup.COLOR = "white"; │ │ │ │ │ +OpenLayers.Popup.OPACITY = 1; │ │ │ │ │ +OpenLayers.Popup.BORDER = "0px"; │ │ │ │ │ +OpenLayers.Popup.Anchored = OpenLayers.Class(OpenLayers.Popup, { │ │ │ │ │ + relativePosition: null, │ │ │ │ │ + keepInMap: true, │ │ │ │ │ + anchor: null, │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, closeBoxCallback) { │ │ │ │ │ + var newArguments = [id, lonlat, contentSize, contentHTML, closeBox, closeBoxCallback]; │ │ │ │ │ + OpenLayers.Popup.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + this.anchor = anchor != null ? anchor : { │ │ │ │ │ + size: new OpenLayers.Size(0, 0), │ │ │ │ │ + offset: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.anchor = null; │ │ │ │ │ + this.relativePosition = null; │ │ │ │ │ + OpenLayers.Popup.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + show: function() { │ │ │ │ │ + this.updatePosition(); │ │ │ │ │ + OpenLayers.Popup.prototype.show.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + var oldRelativePosition = this.relativePosition; │ │ │ │ │ + this.relativePosition = this.calculateRelativePosition(px); │ │ │ │ │ + OpenLayers.Popup.prototype.moveTo.call(this, this.calculateNewPx(px)); │ │ │ │ │ + if (this.relativePosition != oldRelativePosition) { │ │ │ │ │ + this.updateRelativePosition() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setSize: function(contentSize) { │ │ │ │ │ + OpenLayers.Popup.prototype.setSize.apply(this, arguments); │ │ │ │ │ + if (this.lonlat && this.map) { │ │ │ │ │ + var px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ + this.moveTo(px) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + calculateRelativePosition: function(px) { │ │ │ │ │ + var lonlat = this.map.getLonLatFromLayerPx(px); │ │ │ │ │ + var extent = this.map.getExtent(); │ │ │ │ │ + var quadrant = extent.determineQuadrant(lonlat); │ │ │ │ │ + return OpenLayers.Bounds.oppositeQuadrant(quadrant) │ │ │ │ │ + }, │ │ │ │ │ + updateRelativePosition: function() {}, │ │ │ │ │ + calculateNewPx: function(px) { │ │ │ │ │ + var newPx = px.offset(this.anchor.offset); │ │ │ │ │ + var size = this.size || this.contentSize; │ │ │ │ │ + var top = this.relativePosition.charAt(0) == "t"; │ │ │ │ │ + newPx.y += top ? -size.h : this.anchor.size.h; │ │ │ │ │ + var left = this.relativePosition.charAt(1) == "l"; │ │ │ │ │ + newPx.x += left ? -size.w : this.anchor.size.w; │ │ │ │ │ + return newPx │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup.Anchored" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Popup.Framed = OpenLayers.Class(OpenLayers.Popup.Anchored, { │ │ │ │ │ + imageSrc: null, │ │ │ │ │ + imageSize: null, │ │ │ │ │ + isAlphaImage: false, │ │ │ │ │ + positionBlocks: null, │ │ │ │ │ + blocks: null, │ │ │ │ │ + fixedRelativePosition: false, │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, closeBoxCallback) { │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (this.fixedRelativePosition) { │ │ │ │ │ + this.updateRelativePosition(); │ │ │ │ │ + this.calculateRelativePosition = function(px) { │ │ │ │ │ + return this.relativePosition │ │ │ │ │ } │ │ │ │ │ - finalResponse.insertIds = fids; │ │ │ │ │ - callback.apply(this, [response]) │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - function callback(response) { │ │ │ │ │ - this.callUserCallback(response, options); │ │ │ │ │ - success = success && response.success(); │ │ │ │ │ - nResponses++; │ │ │ │ │ - if (nResponses >= nRequests) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - finalResponse.code = success ? OpenLayers.Protocol.Response.SUCCESS : OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - options.callback.apply(options.scope, [finalResponse]) │ │ │ │ │ - } │ │ │ │ │ + this.contentDiv.style.position = "absolute"; │ │ │ │ │ + this.contentDiv.style.zIndex = 1; │ │ │ │ │ + if (closeBox) { │ │ │ │ │ + this.closeDiv.style.zIndex = 1 │ │ │ │ │ + } │ │ │ │ │ + this.groupDiv.style.position = "absolute"; │ │ │ │ │ + this.groupDiv.style.top = "0px"; │ │ │ │ │ + this.groupDiv.style.left = "0px"; │ │ │ │ │ + this.groupDiv.style.height = "100%"; │ │ │ │ │ + this.groupDiv.style.width = "100%" │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.imageSrc = null; │ │ │ │ │ + this.imageSize = null; │ │ │ │ │ + this.isAlphaImage = null; │ │ │ │ │ + this.fixedRelativePosition = false; │ │ │ │ │ + this.positionBlocks = null; │ │ │ │ │ + for (var i = 0; i < this.blocks.length; i++) { │ │ │ │ │ + var block = this.blocks[i]; │ │ │ │ │ + if (block.image) { │ │ │ │ │ + block.div.removeChild(block.image) │ │ │ │ │ + } │ │ │ │ │ + block.image = null; │ │ │ │ │ + if (block.div) { │ │ │ │ │ + this.groupDiv.removeChild(block.div) │ │ │ │ │ } │ │ │ │ │ + block.div = null │ │ │ │ │ } │ │ │ │ │ - var queue = types[OpenLayers.State.INSERT]; │ │ │ │ │ - if (queue.length > 0) { │ │ │ │ │ - resp.push(this.create(queue, OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: insertCallback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options.create))) │ │ │ │ │ + this.blocks = null; │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + setBackgroundColor: function(color) {}, │ │ │ │ │ + setBorder: function() {}, │ │ │ │ │ + setOpacity: function(opacity) {}, │ │ │ │ │ + setSize: function(contentSize) { │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.setSize.apply(this, arguments); │ │ │ │ │ + this.updateBlocks() │ │ │ │ │ + }, │ │ │ │ │ + updateRelativePosition: function() { │ │ │ │ │ + this.padding = this.positionBlocks[this.relativePosition].padding; │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ + this.closeDiv.style.right = contentDivPadding.right + this.padding.right + "px"; │ │ │ │ │ + this.closeDiv.style.top = contentDivPadding.top + this.padding.top + "px" │ │ │ │ │ } │ │ │ │ │ - queue = types[OpenLayers.State.UPDATE]; │ │ │ │ │ - for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ - resp.push(this.update(queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: callback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options.update))) │ │ │ │ │ + this.updateBlocks() │ │ │ │ │ + }, │ │ │ │ │ + calculateNewPx: function(px) { │ │ │ │ │ + var newPx = OpenLayers.Popup.Anchored.prototype.calculateNewPx.apply(this, arguments); │ │ │ │ │ + newPx = newPx.offset(this.positionBlocks[this.relativePosition].offset); │ │ │ │ │ + return newPx │ │ │ │ │ + }, │ │ │ │ │ + createBlocks: function() { │ │ │ │ │ + this.blocks = []; │ │ │ │ │ + var firstPosition = null; │ │ │ │ │ + for (var key in this.positionBlocks) { │ │ │ │ │ + firstPosition = key; │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - queue = types[OpenLayers.State.DELETE]; │ │ │ │ │ - for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ - resp.push(this["delete"](queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: callback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options["delete"]))) │ │ │ │ │ + var position = this.positionBlocks[firstPosition]; │ │ │ │ │ + for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ + var block = {}; │ │ │ │ │ + this.blocks.push(block); │ │ │ │ │ + var divId = this.id + "_FrameDecorationDiv_" + i; │ │ │ │ │ + block.div = OpenLayers.Util.createDiv(divId, null, null, null, "absolute", null, "hidden", null); │ │ │ │ │ + var imgId = this.id + "_FrameDecorationImg_" + i; │ │ │ │ │ + var imageCreator = this.isAlphaImage ? OpenLayers.Util.createAlphaImageDiv : OpenLayers.Util.createImage; │ │ │ │ │ + block.image = imageCreator(imgId, null, this.imageSize, this.imageSrc, "absolute", null, null, null); │ │ │ │ │ + block.div.appendChild(block.image); │ │ │ │ │ + this.groupDiv.appendChild(block.div) │ │ │ │ │ } │ │ │ │ │ - return resp │ │ │ │ │ }, │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - response.priv.abort() │ │ │ │ │ + updateBlocks: function() { │ │ │ │ │ + if (!this.blocks) { │ │ │ │ │ + this.createBlocks() │ │ │ │ │ + } │ │ │ │ │ + if (this.size && this.relativePosition) { │ │ │ │ │ + var position = this.positionBlocks[this.relativePosition]; │ │ │ │ │ + for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ + var positionBlock = position.blocks[i]; │ │ │ │ │ + var block = this.blocks[i]; │ │ │ │ │ + var l = positionBlock.anchor.left; │ │ │ │ │ + var b = positionBlock.anchor.bottom; │ │ │ │ │ + var r = positionBlock.anchor.right; │ │ │ │ │ + var t = positionBlock.anchor.top; │ │ │ │ │ + var w = isNaN(positionBlock.size.w) ? this.size.w - (r + l) : positionBlock.size.w; │ │ │ │ │ + var h = isNaN(positionBlock.size.h) ? this.size.h - (b + t) : positionBlock.size.h; │ │ │ │ │ + block.div.style.width = (w < 0 ? 0 : w) + "px"; │ │ │ │ │ + block.div.style.height = (h < 0 ? 0 : h) + "px"; │ │ │ │ │ + block.div.style.left = l != null ? l + "px" : ""; │ │ │ │ │ + block.div.style.bottom = b != null ? b + "px" : ""; │ │ │ │ │ + block.div.style.right = r != null ? r + "px" : ""; │ │ │ │ │ + block.div.style.top = t != null ? t + "px" : ""; │ │ │ │ │ + block.image.style.left = positionBlock.position.x + "px"; │ │ │ │ │ + block.image.style.top = positionBlock.position.y + "px" │ │ │ │ │ + } │ │ │ │ │ + this.contentDiv.style.left = this.padding.left + "px"; │ │ │ │ │ + this.contentDiv.style.top = this.padding.top + "px" │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - callUserCallback: function(resp, options) { │ │ │ │ │ - var opt = options[resp.requestType]; │ │ │ │ │ - if (opt && opt.callback) { │ │ │ │ │ - opt.callback.call(opt.scope, resp) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup.Framed" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Popup.FramedCloud = OpenLayers.Class(OpenLayers.Popup.Framed, { │ │ │ │ │ + contentDisplayClass: "olFramedCloudPopupContent", │ │ │ │ │ + autoSize: true, │ │ │ │ │ + panMapIfOutOfView: true, │ │ │ │ │ + imageSize: new OpenLayers.Size(1276, 736), │ │ │ │ │ + isAlphaImage: false, │ │ │ │ │ + fixedRelativePosition: false, │ │ │ │ │ + positionBlocks: { │ │ │ │ │ + tl: { │ │ │ │ │ + offset: new OpenLayers.Pixel(44, 0), │ │ │ │ │ + padding: new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ + blocks: [{ │ │ │ │ │ + size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size("auto", 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, 18), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -632) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(81, 35), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -688) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + tr: { │ │ │ │ │ + offset: new OpenLayers.Pixel(-45, 0), │ │ │ │ │ + padding: new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ + blocks: [{ │ │ │ │ │ + size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size("auto", 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -631) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(81, 35), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, null, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-215, -687) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + bl: { │ │ │ │ │ + offset: new OpenLayers.Pixel(45, 0), │ │ │ │ │ + padding: new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ + blocks: [{ │ │ │ │ │ + size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size("auto", 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(81, 33), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, null, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-101, -674) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + br: { │ │ │ │ │ + offset: new OpenLayers.Pixel(-44, 0), │ │ │ │ │ + padding: new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ + blocks: [{ │ │ │ │ │ + size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size("auto", 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(81, 33), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, null, null, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-311, -674) │ │ │ │ │ + }] │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.HTTP" │ │ │ │ │ + minSize: new OpenLayers.Size(105, 10), │ │ │ │ │ + maxSize: new OpenLayers.Size(1200, 660), │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, closeBoxCallback) { │ │ │ │ │ + this.imageSrc = OpenLayers.Util.getImageLocation("cloud-popup-relative.png"); │ │ │ │ │ + OpenLayers.Popup.Framed.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.contentDiv.className = this.contentDisplayClass │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup.FramedCloud" │ │ │ │ │ }); │ │ │ ├── ./usr/share/javascript/openlayers/OpenLayers.min.js │ │ │ │ ├── js-beautify {} │ │ │ │ │ @@ -62,676 +62,14 @@ │ │ │ │ │ var sourceIsEvt = typeof window.Event == "function" && source instanceof window.Event; │ │ │ │ │ if (!sourceIsEvt && source.hasOwnProperty && source.hasOwnProperty("toString")) { │ │ │ │ │ destination.toString = source.toString │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ return destination │ │ │ │ │ }; │ │ │ │ │ -OpenLayers.Console = { │ │ │ │ │ - log: function() {}, │ │ │ │ │ - debug: function() {}, │ │ │ │ │ - info: function() {}, │ │ │ │ │ - warn: function() {}, │ │ │ │ │ - error: function() {}, │ │ │ │ │ - userError: function(error) { │ │ │ │ │ - alert(error) │ │ │ │ │ - }, │ │ │ │ │ - assert: function() {}, │ │ │ │ │ - dir: function() {}, │ │ │ │ │ - dirxml: function() {}, │ │ │ │ │ - trace: function() {}, │ │ │ │ │ - group: function() {}, │ │ │ │ │ - groupEnd: function() {}, │ │ │ │ │ - time: function() {}, │ │ │ │ │ - timeEnd: function() {}, │ │ │ │ │ - profile: function() {}, │ │ │ │ │ - profileEnd: function() {}, │ │ │ │ │ - count: function() {}, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Console" │ │ │ │ │ -}; │ │ │ │ │ -(function() { │ │ │ │ │ - var scripts = document.getElementsByTagName("script"); │ │ │ │ │ - for (var i = 0, len = scripts.length; i < len; ++i) { │ │ │ │ │ - if (scripts[i].src.indexOf("firebug.js") != -1) { │ │ │ │ │ - if (console) { │ │ │ │ │ - OpenLayers.Util.extend(OpenLayers.Console, console); │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ -})(); │ │ │ │ │ -OpenLayers.Popup = OpenLayers.Class({ │ │ │ │ │ - events: null, │ │ │ │ │ - id: "", │ │ │ │ │ - lonlat: null, │ │ │ │ │ - div: null, │ │ │ │ │ - contentSize: null, │ │ │ │ │ - size: null, │ │ │ │ │ - contentHTML: null, │ │ │ │ │ - backgroundColor: "", │ │ │ │ │ - opacity: "", │ │ │ │ │ - border: "", │ │ │ │ │ - contentDiv: null, │ │ │ │ │ - groupDiv: null, │ │ │ │ │ - closeDiv: null, │ │ │ │ │ - autoSize: false, │ │ │ │ │ - minSize: null, │ │ │ │ │ - maxSize: null, │ │ │ │ │ - displayClass: "olPopup", │ │ │ │ │ - contentDisplayClass: "olPopupContent", │ │ │ │ │ - padding: 0, │ │ │ │ │ - disableFirefoxOverflowHack: false, │ │ │ │ │ - fixPadding: function() { │ │ │ │ │ - if (typeof this.padding == "number") { │ │ │ │ │ - this.padding = new OpenLayers.Bounds(this.padding, this.padding, this.padding, this.padding) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - panMapIfOutOfView: false, │ │ │ │ │ - keepInMap: false, │ │ │ │ │ - closeOnMove: false, │ │ │ │ │ - map: null, │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, closeBox, closeBoxCallback) { │ │ │ │ │ - if (id == null) { │ │ │ │ │ - id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ - } │ │ │ │ │ - this.id = id; │ │ │ │ │ - this.lonlat = lonlat; │ │ │ │ │ - this.contentSize = contentSize != null ? contentSize : new OpenLayers.Size(OpenLayers.Popup.WIDTH, OpenLayers.Popup.HEIGHT); │ │ │ │ │ - if (contentHTML != null) { │ │ │ │ │ - this.contentHTML = contentHTML │ │ │ │ │ - } │ │ │ │ │ - this.backgroundColor = OpenLayers.Popup.COLOR; │ │ │ │ │ - this.opacity = OpenLayers.Popup.OPACITY; │ │ │ │ │ - this.border = OpenLayers.Popup.BORDER; │ │ │ │ │ - this.div = OpenLayers.Util.createDiv(this.id, null, null, null, null, null, "hidden"); │ │ │ │ │ - this.div.className = this.displayClass; │ │ │ │ │ - var groupDivId = this.id + "_GroupDiv"; │ │ │ │ │ - this.groupDiv = OpenLayers.Util.createDiv(groupDivId, null, null, null, "relative", null, "hidden"); │ │ │ │ │ - var id = this.div.id + "_contentDiv"; │ │ │ │ │ - this.contentDiv = OpenLayers.Util.createDiv(id, null, this.contentSize.clone(), null, "relative"); │ │ │ │ │ - this.contentDiv.className = this.contentDisplayClass; │ │ │ │ │ - this.groupDiv.appendChild(this.contentDiv); │ │ │ │ │ - this.div.appendChild(this.groupDiv); │ │ │ │ │ - if (closeBox) { │ │ │ │ │ - this.addCloseBox(closeBoxCallback) │ │ │ │ │ - } │ │ │ │ │ - this.registerEvents() │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.id = null; │ │ │ │ │ - this.lonlat = null; │ │ │ │ │ - this.size = null; │ │ │ │ │ - this.contentHTML = null; │ │ │ │ │ - this.backgroundColor = null; │ │ │ │ │ - this.opacity = null; │ │ │ │ │ - this.border = null; │ │ │ │ │ - if (this.closeOnMove && this.map) { │ │ │ │ │ - this.map.events.unregister("movestart", this, this.hide) │ │ │ │ │ - } │ │ │ │ │ - this.events.destroy(); │ │ │ │ │ - this.events = null; │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - OpenLayers.Event.stopObservingElement(this.closeDiv); │ │ │ │ │ - this.groupDiv.removeChild(this.closeDiv) │ │ │ │ │ - } │ │ │ │ │ - this.closeDiv = null; │ │ │ │ │ - this.div.removeChild(this.groupDiv); │ │ │ │ │ - this.groupDiv = null; │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.removePopup(this) │ │ │ │ │ - } │ │ │ │ │ - this.map = null; │ │ │ │ │ - this.div = null; │ │ │ │ │ - this.autoSize = null; │ │ │ │ │ - this.minSize = null; │ │ │ │ │ - this.maxSize = null; │ │ │ │ │ - this.padding = null; │ │ │ │ │ - this.panMapIfOutOfView = null │ │ │ │ │ - }, │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - if (px == null) { │ │ │ │ │ - if (this.lonlat != null && this.map != null) { │ │ │ │ │ - px = this.map.getLayerPxFromLonLat(this.lonlat) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.closeOnMove) { │ │ │ │ │ - this.map.events.register("movestart", this, this.hide) │ │ │ │ │ - } │ │ │ │ │ - if (!this.disableFirefoxOverflowHack && OpenLayers.BROWSER_NAME == "firefox") { │ │ │ │ │ - this.map.events.register("movestart", this, function() { │ │ │ │ │ - var style = document.defaultView.getComputedStyle(this.contentDiv, null); │ │ │ │ │ - var currentOverflow = style.getPropertyValue("overflow"); │ │ │ │ │ - if (currentOverflow != "hidden") { │ │ │ │ │ - this.contentDiv._oldOverflow = currentOverflow; │ │ │ │ │ - this.contentDiv.style.overflow = "hidden" │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - this.map.events.register("moveend", this, function() { │ │ │ │ │ - var oldOverflow = this.contentDiv._oldOverflow; │ │ │ │ │ - if (oldOverflow) { │ │ │ │ │ - this.contentDiv.style.overflow = oldOverflow; │ │ │ │ │ - this.contentDiv._oldOverflow = null │ │ │ │ │ - } │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.moveTo(px); │ │ │ │ │ - if (!this.autoSize && !this.size) { │ │ │ │ │ - this.setSize(this.contentSize) │ │ │ │ │ - } │ │ │ │ │ - this.setBackgroundColor(); │ │ │ │ │ - this.setOpacity(); │ │ │ │ │ - this.setBorder(); │ │ │ │ │ - this.setContentHTML(); │ │ │ │ │ - if (this.panMapIfOutOfView) { │ │ │ │ │ - this.panIntoView() │ │ │ │ │ - } │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - updatePosition: function() { │ │ │ │ │ - if (this.lonlat && this.map) { │ │ │ │ │ - var px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ - if (px) { │ │ │ │ │ - this.moveTo(px) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - if (px != null && this.div != null) { │ │ │ │ │ - this.div.style.left = px.x + "px"; │ │ │ │ │ - this.div.style.top = px.y + "px" │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - visible: function() { │ │ │ │ │ - return OpenLayers.Element.visible(this.div) │ │ │ │ │ - }, │ │ │ │ │ - toggle: function() { │ │ │ │ │ - if (this.visible()) { │ │ │ │ │ - this.hide() │ │ │ │ │ - } else { │ │ │ │ │ - this.show() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - show: function() { │ │ │ │ │ - this.div.style.display = ""; │ │ │ │ │ - if (this.panMapIfOutOfView) { │ │ │ │ │ - this.panIntoView() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - hide: function() { │ │ │ │ │ - this.div.style.display = "none" │ │ │ │ │ - }, │ │ │ │ │ - setSize: function(contentSize) { │ │ │ │ │ - this.size = contentSize.clone(); │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ - var wPadding = contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ - var hPadding = contentDivPadding.top + contentDivPadding.bottom; │ │ │ │ │ - this.fixPadding(); │ │ │ │ │ - wPadding += this.padding.left + this.padding.right; │ │ │ │ │ - hPadding += this.padding.top + this.padding.bottom; │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - var closeDivWidth = parseInt(this.closeDiv.style.width); │ │ │ │ │ - wPadding += closeDivWidth + contentDivPadding.right │ │ │ │ │ - } │ │ │ │ │ - this.size.w += wPadding; │ │ │ │ │ - this.size.h += hPadding; │ │ │ │ │ - if (OpenLayers.BROWSER_NAME == "msie") { │ │ │ │ │ - this.contentSize.w += contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ - this.contentSize.h += contentDivPadding.bottom + contentDivPadding.top │ │ │ │ │ - } │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.div.style.width = this.size.w + "px"; │ │ │ │ │ - this.div.style.height = this.size.h + "px" │ │ │ │ │ - } │ │ │ │ │ - if (this.contentDiv != null) { │ │ │ │ │ - this.contentDiv.style.width = contentSize.w + "px"; │ │ │ │ │ - this.contentDiv.style.height = contentSize.h + "px" │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - updateSize: function() { │ │ │ │ │ - var preparedHTML = "<div class='" + this.contentDisplayClass + "'>" + this.contentDiv.innerHTML + "</div>"; │ │ │ │ │ - var containerElement = this.map ? this.map.div : document.body; │ │ │ │ │ - var realSize = OpenLayers.Util.getRenderedDimensions(preparedHTML, null, { │ │ │ │ │ - displayClass: this.displayClass, │ │ │ │ │ - containerElement: containerElement │ │ │ │ │ - }); │ │ │ │ │ - var safeSize = this.getSafeContentSize(realSize); │ │ │ │ │ - var newSize = null; │ │ │ │ │ - if (safeSize.equals(realSize)) { │ │ │ │ │ - newSize = realSize │ │ │ │ │ - } else { │ │ │ │ │ - var fixedSize = { │ │ │ │ │ - w: safeSize.w < realSize.w ? safeSize.w : null, │ │ │ │ │ - h: safeSize.h < realSize.h ? safeSize.h : null │ │ │ │ │ - }; │ │ │ │ │ - if (fixedSize.w && fixedSize.h) { │ │ │ │ │ - newSize = safeSize │ │ │ │ │ - } else { │ │ │ │ │ - var clippedSize = OpenLayers.Util.getRenderedDimensions(preparedHTML, fixedSize, { │ │ │ │ │ - displayClass: this.contentDisplayClass, │ │ │ │ │ - containerElement: containerElement │ │ │ │ │ - }); │ │ │ │ │ - var currentOverflow = OpenLayers.Element.getStyle(this.contentDiv, "overflow"); │ │ │ │ │ - if (currentOverflow != "hidden" && clippedSize.equals(safeSize)) { │ │ │ │ │ - var scrollBar = OpenLayers.Util.getScrollbarWidth(); │ │ │ │ │ - if (fixedSize.w) { │ │ │ │ │ - clippedSize.h += scrollBar │ │ │ │ │ - } else { │ │ │ │ │ - clippedSize.w += scrollBar │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - newSize = this.getSafeContentSize(clippedSize) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.setSize(newSize) │ │ │ │ │ - }, │ │ │ │ │ - setBackgroundColor: function(color) { │ │ │ │ │ - if (color != undefined) { │ │ │ │ │ - this.backgroundColor = color │ │ │ │ │ - } │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.div.style.backgroundColor = this.backgroundColor │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - if (opacity != undefined) { │ │ │ │ │ - this.opacity = opacity │ │ │ │ │ - } │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.div.style.opacity = this.opacity; │ │ │ │ │ - this.div.style.filter = "alpha(opacity=" + this.opacity * 100 + ")" │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setBorder: function(border) { │ │ │ │ │ - if (border != undefined) { │ │ │ │ │ - this.border = border │ │ │ │ │ - } │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.div.style.border = this.border │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setContentHTML: function(contentHTML) { │ │ │ │ │ - if (contentHTML != null) { │ │ │ │ │ - this.contentHTML = contentHTML │ │ │ │ │ - } │ │ │ │ │ - if (this.contentDiv != null && this.contentHTML != null && this.contentHTML != this.contentDiv.innerHTML) { │ │ │ │ │ - this.contentDiv.innerHTML = this.contentHTML; │ │ │ │ │ - if (this.autoSize) { │ │ │ │ │ - this.registerImageListeners(); │ │ │ │ │ - this.updateSize() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - registerImageListeners: function() { │ │ │ │ │ - var onImgLoad = function() { │ │ │ │ │ - if (this.popup.id === null) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - this.popup.updateSize(); │ │ │ │ │ - if (this.popup.visible() && this.popup.panMapIfOutOfView) { │ │ │ │ │ - this.popup.panIntoView() │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.stopObserving(this.img, "load", this.img._onImgLoad) │ │ │ │ │ - }; │ │ │ │ │ - var images = this.contentDiv.getElementsByTagName("img"); │ │ │ │ │ - for (var i = 0, len = images.length; i < len; i++) { │ │ │ │ │ - var img = images[i]; │ │ │ │ │ - if (img.width == 0 || img.height == 0) { │ │ │ │ │ - var context = { │ │ │ │ │ - popup: this, │ │ │ │ │ - img: img │ │ │ │ │ - }; │ │ │ │ │ - img._onImgLoad = OpenLayers.Function.bind(onImgLoad, context); │ │ │ │ │ - OpenLayers.Event.observe(img, "load", img._onImgLoad) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getSafeContentSize: function(size) { │ │ │ │ │ - var safeContentSize = size.clone(); │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ - var wPadding = contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ - var hPadding = contentDivPadding.top + contentDivPadding.bottom; │ │ │ │ │ - this.fixPadding(); │ │ │ │ │ - wPadding += this.padding.left + this.padding.right; │ │ │ │ │ - hPadding += this.padding.top + this.padding.bottom; │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - var closeDivWidth = parseInt(this.closeDiv.style.width); │ │ │ │ │ - wPadding += closeDivWidth + contentDivPadding.right │ │ │ │ │ - } │ │ │ │ │ - if (this.minSize) { │ │ │ │ │ - safeContentSize.w = Math.max(safeContentSize.w, this.minSize.w - wPadding); │ │ │ │ │ - safeContentSize.h = Math.max(safeContentSize.h, this.minSize.h - hPadding) │ │ │ │ │ - } │ │ │ │ │ - if (this.maxSize) { │ │ │ │ │ - safeContentSize.w = Math.min(safeContentSize.w, this.maxSize.w - wPadding); │ │ │ │ │ - safeContentSize.h = Math.min(safeContentSize.h, this.maxSize.h - hPadding) │ │ │ │ │ - } │ │ │ │ │ - if (this.map && this.map.size) { │ │ │ │ │ - var extraX = 0, │ │ │ │ │ - extraY = 0; │ │ │ │ │ - if (this.keepInMap && !this.panMapIfOutOfView) { │ │ │ │ │ - var px = this.map.getPixelFromLonLat(this.lonlat); │ │ │ │ │ - switch (this.relativePosition) { │ │ │ │ │ - case "tr": │ │ │ │ │ - extraX = px.x; │ │ │ │ │ - extraY = this.map.size.h - px.y; │ │ │ │ │ - break; │ │ │ │ │ - case "tl": │ │ │ │ │ - extraX = this.map.size.w - px.x; │ │ │ │ │ - extraY = this.map.size.h - px.y; │ │ │ │ │ - break; │ │ │ │ │ - case "bl": │ │ │ │ │ - extraX = this.map.size.w - px.x; │ │ │ │ │ - extraY = px.y; │ │ │ │ │ - break; │ │ │ │ │ - case "br": │ │ │ │ │ - extraX = px.x; │ │ │ │ │ - extraY = px.y; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - extraX = px.x; │ │ │ │ │ - extraY = this.map.size.h - px.y; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var maxY = this.map.size.h - this.map.paddingForPopups.top - this.map.paddingForPopups.bottom - hPadding - extraY; │ │ │ │ │ - var maxX = this.map.size.w - this.map.paddingForPopups.left - this.map.paddingForPopups.right - wPadding - extraX; │ │ │ │ │ - safeContentSize.w = Math.min(safeContentSize.w, maxX); │ │ │ │ │ - safeContentSize.h = Math.min(safeContentSize.h, maxY) │ │ │ │ │ - } │ │ │ │ │ - return safeContentSize │ │ │ │ │ - }, │ │ │ │ │ - getContentDivPadding: function() { │ │ │ │ │ - var contentDivPadding = this._contentDivPadding; │ │ │ │ │ - if (!contentDivPadding) { │ │ │ │ │ - if (this.div.parentNode == null) { │ │ │ │ │ - this.div.style.display = "none"; │ │ │ │ │ - document.body.appendChild(this.div) │ │ │ │ │ - } │ │ │ │ │ - contentDivPadding = new OpenLayers.Bounds(OpenLayers.Element.getStyle(this.contentDiv, "padding-left"), OpenLayers.Element.getStyle(this.contentDiv, "padding-bottom"), OpenLayers.Element.getStyle(this.contentDiv, "padding-right"), OpenLayers.Element.getStyle(this.contentDiv, "padding-top")); │ │ │ │ │ - this._contentDivPadding = contentDivPadding; │ │ │ │ │ - if (this.div.parentNode == document.body) { │ │ │ │ │ - document.body.removeChild(this.div); │ │ │ │ │ - this.div.style.display = "" │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return contentDivPadding │ │ │ │ │ - }, │ │ │ │ │ - addCloseBox: function(callback) { │ │ │ │ │ - this.closeDiv = OpenLayers.Util.createDiv(this.id + "_close", null, { │ │ │ │ │ - w: 17, │ │ │ │ │ - h: 17 │ │ │ │ │ - }); │ │ │ │ │ - this.closeDiv.className = "olPopupCloseBox"; │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ - this.closeDiv.style.right = contentDivPadding.right + "px"; │ │ │ │ │ - this.closeDiv.style.top = contentDivPadding.top + "px"; │ │ │ │ │ - this.groupDiv.appendChild(this.closeDiv); │ │ │ │ │ - var closePopup = callback || function(e) { │ │ │ │ │ - this.hide(); │ │ │ │ │ - OpenLayers.Event.stop(e) │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Event.observe(this.closeDiv, "touchend", OpenLayers.Function.bindAsEventListener(closePopup, this)); │ │ │ │ │ - OpenLayers.Event.observe(this.closeDiv, "click", OpenLayers.Function.bindAsEventListener(closePopup, this)) │ │ │ │ │ - }, │ │ │ │ │ - panIntoView: function() { │ │ │ │ │ - var mapSize = this.map.getSize(); │ │ │ │ │ - var origTL = this.map.getViewPortPxFromLayerPx(new OpenLayers.Pixel(parseInt(this.div.style.left), parseInt(this.div.style.top))); │ │ │ │ │ - var newTL = origTL.clone(); │ │ │ │ │ - if (origTL.x < this.map.paddingForPopups.left) { │ │ │ │ │ - newTL.x = this.map.paddingForPopups.left │ │ │ │ │ - } else if (origTL.x + this.size.w > mapSize.w - this.map.paddingForPopups.right) { │ │ │ │ │ - newTL.x = mapSize.w - this.map.paddingForPopups.right - this.size.w │ │ │ │ │ - } │ │ │ │ │ - if (origTL.y < this.map.paddingForPopups.top) { │ │ │ │ │ - newTL.y = this.map.paddingForPopups.top │ │ │ │ │ - } else if (origTL.y + this.size.h > mapSize.h - this.map.paddingForPopups.bottom) { │ │ │ │ │ - newTL.y = mapSize.h - this.map.paddingForPopups.bottom - this.size.h │ │ │ │ │ - } │ │ │ │ │ - var dx = origTL.x - newTL.x; │ │ │ │ │ - var dy = origTL.y - newTL.y; │ │ │ │ │ - this.map.pan(dx, dy) │ │ │ │ │ - }, │ │ │ │ │ - registerEvents: function() { │ │ │ │ │ - this.events = new OpenLayers.Events(this, this.div, null, true); │ │ │ │ │ - │ │ │ │ │ - function onTouchstart(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt, true) │ │ │ │ │ - } │ │ │ │ │ - this.events.on({ │ │ │ │ │ - mousedown: this.onmousedown, │ │ │ │ │ - mousemove: this.onmousemove, │ │ │ │ │ - mouseup: this.onmouseup, │ │ │ │ │ - click: this.onclick, │ │ │ │ │ - mouseout: this.onmouseout, │ │ │ │ │ - dblclick: this.ondblclick, │ │ │ │ │ - touchstart: onTouchstart, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - onmousedown: function(evt) { │ │ │ │ │ - this.mousedown = true; │ │ │ │ │ - OpenLayers.Event.stop(evt, true) │ │ │ │ │ - }, │ │ │ │ │ - onmousemove: function(evt) { │ │ │ │ │ - if (this.mousedown) { │ │ │ │ │ - OpenLayers.Event.stop(evt, true) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - onmouseup: function(evt) { │ │ │ │ │ - if (this.mousedown) { │ │ │ │ │ - this.mousedown = false; │ │ │ │ │ - OpenLayers.Event.stop(evt, true) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - onclick: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt, true) │ │ │ │ │ - }, │ │ │ │ │ - onmouseout: function(evt) { │ │ │ │ │ - this.mousedown = false │ │ │ │ │ - }, │ │ │ │ │ - ondblclick: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt, true) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Popup.WIDTH = 200; │ │ │ │ │ -OpenLayers.Popup.HEIGHT = 200; │ │ │ │ │ -OpenLayers.Popup.COLOR = "white"; │ │ │ │ │ -OpenLayers.Popup.OPACITY = 1; │ │ │ │ │ -OpenLayers.Popup.BORDER = "0px"; │ │ │ │ │ -OpenLayers.Util = OpenLayers.Util || {}; │ │ │ │ │ -OpenLayers.Util.vendorPrefix = function() { │ │ │ │ │ - "use strict"; │ │ │ │ │ - var VENDOR_PREFIXES = ["", "O", "ms", "Moz", "Webkit"], │ │ │ │ │ - divStyle = document.createElement("div").style, │ │ │ │ │ - cssCache = {}, │ │ │ │ │ - jsCache = {}; │ │ │ │ │ - │ │ │ │ │ - function domToCss(prefixedDom) { │ │ │ │ │ - if (!prefixedDom) { │ │ │ │ │ - return null │ │ │ │ │ - } │ │ │ │ │ - return prefixedDom.replace(/([A-Z])/g, function(c) { │ │ │ │ │ - return "-" + c.toLowerCase() │ │ │ │ │ - }).replace(/^ms-/, "-ms-") │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function css(property) { │ │ │ │ │ - if (cssCache[property] === undefined) { │ │ │ │ │ - var domProperty = property.replace(/(-[\s\S])/g, function(c) { │ │ │ │ │ - return c.charAt(1).toUpperCase() │ │ │ │ │ - }); │ │ │ │ │ - var prefixedDom = style(domProperty); │ │ │ │ │ - cssCache[property] = domToCss(prefixedDom) │ │ │ │ │ - } │ │ │ │ │ - return cssCache[property] │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function js(obj, property) { │ │ │ │ │ - if (jsCache[property] === undefined) { │ │ │ │ │ - var tmpProp, i = 0, │ │ │ │ │ - l = VENDOR_PREFIXES.length, │ │ │ │ │ - prefix, isStyleObj = typeof obj.cssText !== "undefined"; │ │ │ │ │ - jsCache[property] = null; │ │ │ │ │ - for (; i < l; i++) { │ │ │ │ │ - prefix = VENDOR_PREFIXES[i]; │ │ │ │ │ - if (prefix) { │ │ │ │ │ - if (!isStyleObj) { │ │ │ │ │ - prefix = prefix.toLowerCase() │ │ │ │ │ - } │ │ │ │ │ - tmpProp = prefix + property.charAt(0).toUpperCase() + property.slice(1) │ │ │ │ │ - } else { │ │ │ │ │ - tmpProp = property │ │ │ │ │ - } │ │ │ │ │ - if (obj[tmpProp] !== undefined) { │ │ │ │ │ - jsCache[property] = tmpProp; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return jsCache[property] │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function style(property) { │ │ │ │ │ - return js(divStyle, property) │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - css: css, │ │ │ │ │ - js: js, │ │ │ │ │ - style: style, │ │ │ │ │ - cssCache: cssCache, │ │ │ │ │ - jsCache: jsCache │ │ │ │ │ - } │ │ │ │ │ -}(); │ │ │ │ │ -OpenLayers.Animation = function(window) { │ │ │ │ │ - var requestAnimationFrame = OpenLayers.Util.vendorPrefix.js(window, "requestAnimationFrame"); │ │ │ │ │ - var isNative = !!requestAnimationFrame; │ │ │ │ │ - var requestFrame = function() { │ │ │ │ │ - var request = window[requestAnimationFrame] || function(callback, element) { │ │ │ │ │ - window.setTimeout(callback, 16) │ │ │ │ │ - }; │ │ │ │ │ - return function(callback, element) { │ │ │ │ │ - request.apply(window, [callback, element]) │ │ │ │ │ - } │ │ │ │ │ - }(); │ │ │ │ │ - var counter = 0; │ │ │ │ │ - var loops = {}; │ │ │ │ │ - │ │ │ │ │ - function start(callback, duration, element) { │ │ │ │ │ - duration = duration > 0 ? duration : Number.POSITIVE_INFINITY; │ │ │ │ │ - var id = ++counter; │ │ │ │ │ - var start = +new Date; │ │ │ │ │ - loops[id] = function() { │ │ │ │ │ - if (loops[id] && +new Date - start <= duration) { │ │ │ │ │ - callback(); │ │ │ │ │ - if (loops[id]) { │ │ │ │ │ - requestFrame(loops[id], element) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - delete loops[id] │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - requestFrame(loops[id], element); │ │ │ │ │ - return id │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function stop(id) { │ │ │ │ │ - delete loops[id] │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - isNative: isNative, │ │ │ │ │ - requestFrame: requestFrame, │ │ │ │ │ - start: start, │ │ │ │ │ - stop: stop │ │ │ │ │ - } │ │ │ │ │ -}(window); │ │ │ │ │ -OpenLayers.Kinetic = OpenLayers.Class({ │ │ │ │ │ - threshold: 0, │ │ │ │ │ - deceleration: .0035, │ │ │ │ │ - nbPoints: 100, │ │ │ │ │ - delay: 200, │ │ │ │ │ - points: undefined, │ │ │ │ │ - timerId: undefined, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options) │ │ │ │ │ - }, │ │ │ │ │ - begin: function() { │ │ │ │ │ - OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ - this.timerId = undefined; │ │ │ │ │ - this.points = [] │ │ │ │ │ - }, │ │ │ │ │ - update: function(xy) { │ │ │ │ │ - this.points.unshift({ │ │ │ │ │ - xy: xy, │ │ │ │ │ - tick: (new Date).getTime() │ │ │ │ │ - }); │ │ │ │ │ - if (this.points.length > this.nbPoints) { │ │ │ │ │ - this.points.pop() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - end: function(xy) { │ │ │ │ │ - var last, now = (new Date).getTime(); │ │ │ │ │ - for (var i = 0, l = this.points.length, point; i < l; i++) { │ │ │ │ │ - point = this.points[i]; │ │ │ │ │ - if (now - point.tick > this.delay) { │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - last = point │ │ │ │ │ - } │ │ │ │ │ - if (!last) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var time = (new Date).getTime() - last.tick; │ │ │ │ │ - var dist = Math.sqrt(Math.pow(xy.x - last.xy.x, 2) + Math.pow(xy.y - last.xy.y, 2)); │ │ │ │ │ - var speed = dist / time; │ │ │ │ │ - if (speed == 0 || speed < this.threshold) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var theta = Math.asin((xy.y - last.xy.y) / dist); │ │ │ │ │ - if (last.xy.x <= xy.x) { │ │ │ │ │ - theta = Math.PI - theta │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - speed: speed, │ │ │ │ │ - theta: theta │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - move: function(info, callback) { │ │ │ │ │ - var v0 = info.speed; │ │ │ │ │ - var fx = Math.cos(info.theta); │ │ │ │ │ - var fy = -Math.sin(info.theta); │ │ │ │ │ - var initialTime = (new Date).getTime(); │ │ │ │ │ - var lastX = 0; │ │ │ │ │ - var lastY = 0; │ │ │ │ │ - var timerCallback = function() { │ │ │ │ │ - if (this.timerId == null) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var t = (new Date).getTime() - initialTime; │ │ │ │ │ - var p = -this.deceleration * Math.pow(t, 2) / 2 + v0 * t; │ │ │ │ │ - var x = p * fx; │ │ │ │ │ - var y = p * fy; │ │ │ │ │ - var args = {}; │ │ │ │ │ - args.end = false; │ │ │ │ │ - var v = -this.deceleration * t + v0; │ │ │ │ │ - if (v <= 0) { │ │ │ │ │ - OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - args.end = true │ │ │ │ │ - } │ │ │ │ │ - args.x = x - lastX; │ │ │ │ │ - args.y = y - lastY; │ │ │ │ │ - lastX = x; │ │ │ │ │ - lastY = y; │ │ │ │ │ - callback(args.x, args.y, args.end) │ │ │ │ │ - }; │ │ │ │ │ - this.timerId = OpenLayers.Animation.start(OpenLayers.Function.bind(timerCallback, this)) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Kinetic" │ │ │ │ │ -}); │ │ │ │ │ OpenLayers.String = { │ │ │ │ │ startsWith: function(str, sub) { │ │ │ │ │ return str.indexOf(sub) == 0 │ │ │ │ │ }, │ │ │ │ │ contains: function(str, sub) { │ │ │ │ │ return str.indexOf(sub) != -1 │ │ │ │ │ }, │ │ │ │ │ @@ -1381,14 +719,47 @@ │ │ │ │ │ if (sz != null) { │ │ │ │ │ equals = this.w == sz.w && this.h == sz.h || isNaN(this.w) && isNaN(this.h) && isNaN(sz.w) && isNaN(sz.h) │ │ │ │ │ } │ │ │ │ │ return equals │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Size" │ │ │ │ │ }); │ │ │ │ │ +OpenLayers.Console = { │ │ │ │ │ + log: function() {}, │ │ │ │ │ + debug: function() {}, │ │ │ │ │ + info: function() {}, │ │ │ │ │ + warn: function() {}, │ │ │ │ │ + error: function() {}, │ │ │ │ │ + userError: function(error) { │ │ │ │ │ + alert(error) │ │ │ │ │ + }, │ │ │ │ │ + assert: function() {}, │ │ │ │ │ + dir: function() {}, │ │ │ │ │ + dirxml: function() {}, │ │ │ │ │ + trace: function() {}, │ │ │ │ │ + group: function() {}, │ │ │ │ │ + groupEnd: function() {}, │ │ │ │ │ + time: function() {}, │ │ │ │ │ + timeEnd: function() {}, │ │ │ │ │ + profile: function() {}, │ │ │ │ │ + profileEnd: function() {}, │ │ │ │ │ + count: function() {}, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Console" │ │ │ │ │ +}; │ │ │ │ │ +(function() { │ │ │ │ │ + var scripts = document.getElementsByTagName("script"); │ │ │ │ │ + for (var i = 0, len = scripts.length; i < len; ++i) { │ │ │ │ │ + if (scripts[i].src.indexOf("firebug.js") != -1) { │ │ │ │ │ + if (console) { │ │ │ │ │ + OpenLayers.Util.extend(OpenLayers.Console, console); │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ +})(); │ │ │ │ │ OpenLayers.Lang = { │ │ │ │ │ code: null, │ │ │ │ │ defaultCode: "en", │ │ │ │ │ getCode: function() { │ │ │ │ │ if (!OpenLayers.Lang.code) { │ │ │ │ │ OpenLayers.Lang.setCode() │ │ │ │ │ } │ │ │ │ │ @@ -2234,14 +1605,78 @@ │ │ │ │ │ if (axis == "lon") { │ │ │ │ │ str += coordinate < 0 ? OpenLayers.i18n("W") : OpenLayers.i18n("E") │ │ │ │ │ } else { │ │ │ │ │ str += coordinate < 0 ? OpenLayers.i18n("S") : OpenLayers.i18n("N") │ │ │ │ │ } │ │ │ │ │ return str │ │ │ │ │ }; │ │ │ │ │ +OpenLayers.Util = OpenLayers.Util || {}; │ │ │ │ │ +OpenLayers.Util.vendorPrefix = function() { │ │ │ │ │ + "use strict"; │ │ │ │ │ + var VENDOR_PREFIXES = ["", "O", "ms", "Moz", "Webkit"], │ │ │ │ │ + divStyle = document.createElement("div").style, │ │ │ │ │ + cssCache = {}, │ │ │ │ │ + jsCache = {}; │ │ │ │ │ + │ │ │ │ │ + function domToCss(prefixedDom) { │ │ │ │ │ + if (!prefixedDom) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ + return prefixedDom.replace(/([A-Z])/g, function(c) { │ │ │ │ │ + return "-" + c.toLowerCase() │ │ │ │ │ + }).replace(/^ms-/, "-ms-") │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function css(property) { │ │ │ │ │ + if (cssCache[property] === undefined) { │ │ │ │ │ + var domProperty = property.replace(/(-[\s\S])/g, function(c) { │ │ │ │ │ + return c.charAt(1).toUpperCase() │ │ │ │ │ + }); │ │ │ │ │ + var prefixedDom = style(domProperty); │ │ │ │ │ + cssCache[property] = domToCss(prefixedDom) │ │ │ │ │ + } │ │ │ │ │ + return cssCache[property] │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function js(obj, property) { │ │ │ │ │ + if (jsCache[property] === undefined) { │ │ │ │ │ + var tmpProp, i = 0, │ │ │ │ │ + l = VENDOR_PREFIXES.length, │ │ │ │ │ + prefix, isStyleObj = typeof obj.cssText !== "undefined"; │ │ │ │ │ + jsCache[property] = null; │ │ │ │ │ + for (; i < l; i++) { │ │ │ │ │ + prefix = VENDOR_PREFIXES[i]; │ │ │ │ │ + if (prefix) { │ │ │ │ │ + if (!isStyleObj) { │ │ │ │ │ + prefix = prefix.toLowerCase() │ │ │ │ │ + } │ │ │ │ │ + tmpProp = prefix + property.charAt(0).toUpperCase() + property.slice(1) │ │ │ │ │ + } else { │ │ │ │ │ + tmpProp = property │ │ │ │ │ + } │ │ │ │ │ + if (obj[tmpProp] !== undefined) { │ │ │ │ │ + jsCache[property] = tmpProp; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return jsCache[property] │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function style(property) { │ │ │ │ │ + return js(divStyle, property) │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + css: css, │ │ │ │ │ + js: js, │ │ │ │ │ + style: style, │ │ │ │ │ + cssCache: cssCache, │ │ │ │ │ + jsCache: jsCache │ │ │ │ │ + } │ │ │ │ │ +}(); │ │ │ │ │ OpenLayers.Event = { │ │ │ │ │ observers: false, │ │ │ │ │ KEY_SPACE: 32, │ │ │ │ │ KEY_BACKSPACE: 8, │ │ │ │ │ KEY_TAB: 9, │ │ │ │ │ KEY_RETURN: 13, │ │ │ │ │ KEY_ESC: 27, │ │ │ │ │ @@ -2692,14 +2127,56 @@ │ │ │ │ │ e.touches = touches.slice(); │ │ │ │ │ handler(e) │ │ │ │ │ }; │ │ │ │ │ OpenLayers.Event.observe(element, "MSPointerUp", cb) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Events" │ │ │ │ │ }); │ │ │ │ │ +OpenLayers.Animation = function(window) { │ │ │ │ │ + var requestAnimationFrame = OpenLayers.Util.vendorPrefix.js(window, "requestAnimationFrame"); │ │ │ │ │ + var isNative = !!requestAnimationFrame; │ │ │ │ │ + var requestFrame = function() { │ │ │ │ │ + var request = window[requestAnimationFrame] || function(callback, element) { │ │ │ │ │ + window.setTimeout(callback, 16) │ │ │ │ │ + }; │ │ │ │ │ + return function(callback, element) { │ │ │ │ │ + request.apply(window, [callback, element]) │ │ │ │ │ + } │ │ │ │ │ + }(); │ │ │ │ │ + var counter = 0; │ │ │ │ │ + var loops = {}; │ │ │ │ │ + │ │ │ │ │ + function start(callback, duration, element) { │ │ │ │ │ + duration = duration > 0 ? duration : Number.POSITIVE_INFINITY; │ │ │ │ │ + var id = ++counter; │ │ │ │ │ + var start = +new Date; │ │ │ │ │ + loops[id] = function() { │ │ │ │ │ + if (loops[id] && +new Date - start <= duration) { │ │ │ │ │ + callback(); │ │ │ │ │ + if (loops[id]) { │ │ │ │ │ + requestFrame(loops[id], element) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + delete loops[id] │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + requestFrame(loops[id], element); │ │ │ │ │ + return id │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function stop(id) { │ │ │ │ │ + delete loops[id] │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + isNative: isNative, │ │ │ │ │ + requestFrame: requestFrame, │ │ │ │ │ + start: start, │ │ │ │ │ + stop: stop │ │ │ │ │ + } │ │ │ │ │ +}(window); │ │ │ │ │ OpenLayers.Tween = OpenLayers.Class({ │ │ │ │ │ easing: null, │ │ │ │ │ begin: null, │ │ │ │ │ finish: null, │ │ │ │ │ duration: null, │ │ │ │ │ callbacks: null, │ │ │ │ │ time: null, │ │ │ │ │ @@ -4120,1168 +3597,154 @@ │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Map" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Map.TILE_WIDTH = 256; │ │ │ │ │ OpenLayers.Map.TILE_HEIGHT = 256; │ │ │ │ │ -OpenLayers.Layer = OpenLayers.Class({ │ │ │ │ │ - id: null, │ │ │ │ │ - name: null, │ │ │ │ │ - div: null, │ │ │ │ │ - opacity: 1, │ │ │ │ │ - alwaysInRange: null, │ │ │ │ │ - RESOLUTION_PROPERTIES: ["scales", "resolutions", "maxScale", "minScale", "maxResolution", "minResolution", "numZoomLevels", "maxZoomLevel"], │ │ │ │ │ - events: null, │ │ │ │ │ +OpenLayers.Renderer = OpenLayers.Class({ │ │ │ │ │ + container: null, │ │ │ │ │ + root: null, │ │ │ │ │ + extent: null, │ │ │ │ │ + locked: false, │ │ │ │ │ + size: null, │ │ │ │ │ + resolution: null, │ │ │ │ │ map: null, │ │ │ │ │ - isBaseLayer: false, │ │ │ │ │ - alpha: false, │ │ │ │ │ - displayInLayerSwitcher: true, │ │ │ │ │ - visibility: true, │ │ │ │ │ - attribution: null, │ │ │ │ │ - inRange: false, │ │ │ │ │ - imageSize: null, │ │ │ │ │ - options: null, │ │ │ │ │ - eventListeners: null, │ │ │ │ │ - gutter: 0, │ │ │ │ │ - projection: null, │ │ │ │ │ - units: null, │ │ │ │ │ - scales: null, │ │ │ │ │ - resolutions: null, │ │ │ │ │ - maxExtent: null, │ │ │ │ │ - minExtent: null, │ │ │ │ │ - maxResolution: null, │ │ │ │ │ - minResolution: null, │ │ │ │ │ - numZoomLevels: null, │ │ │ │ │ - minScale: null, │ │ │ │ │ - maxScale: null, │ │ │ │ │ - displayOutsideMaxExtent: false, │ │ │ │ │ - wrapDateLine: false, │ │ │ │ │ - metadata: null, │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - this.metadata = {}; │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - if (this.alwaysInRange != null) { │ │ │ │ │ - options.alwaysInRange = this.alwaysInRange │ │ │ │ │ - } │ │ │ │ │ - this.addOptions(options); │ │ │ │ │ - this.name = name; │ │ │ │ │ - if (this.id == null) { │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ - this.div = OpenLayers.Util.createDiv(this.id); │ │ │ │ │ - this.div.style.width = "100%"; │ │ │ │ │ - this.div.style.height = "100%"; │ │ │ │ │ - this.div.dir = "ltr"; │ │ │ │ │ - this.events = new OpenLayers.Events(this, this.div); │ │ │ │ │ - if (this.eventListeners instanceof Object) { │ │ │ │ │ - this.events.on(this.eventListeners) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function(setNewBaseLayer) { │ │ │ │ │ - if (setNewBaseLayer == null) { │ │ │ │ │ - setNewBaseLayer = true │ │ │ │ │ - } │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.removeLayer(this, setNewBaseLayer) │ │ │ │ │ - } │ │ │ │ │ - this.projection = null; │ │ │ │ │ - this.map = null; │ │ │ │ │ - this.name = null; │ │ │ │ │ - this.div = null; │ │ │ │ │ - this.options = null; │ │ │ │ │ - if (this.events) { │ │ │ │ │ - if (this.eventListeners) { │ │ │ │ │ - this.events.un(this.eventListeners) │ │ │ │ │ - } │ │ │ │ │ - this.events.destroy() │ │ │ │ │ - } │ │ │ │ │ - this.eventListeners = null; │ │ │ │ │ - this.events = null │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer(this.name, this.getOptions()) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.applyDefaults(obj, this); │ │ │ │ │ - obj.map = null; │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - getOptions: function() { │ │ │ │ │ - var options = {}; │ │ │ │ │ - for (var o in this.options) { │ │ │ │ │ - options[o] = this[o] │ │ │ │ │ - } │ │ │ │ │ - return options │ │ │ │ │ - }, │ │ │ │ │ - setName: function(newName) { │ │ │ │ │ - if (newName != this.name) { │ │ │ │ │ - this.name = newName; │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "name" │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - addOptions: function(newOptions, reinitialize) { │ │ │ │ │ - if (this.options == null) { │ │ │ │ │ - this.options = {} │ │ │ │ │ - } │ │ │ │ │ - if (newOptions) { │ │ │ │ │ - if (typeof newOptions.projection == "string") { │ │ │ │ │ - newOptions.projection = new OpenLayers.Projection(newOptions.projection) │ │ │ │ │ - } │ │ │ │ │ - if (newOptions.projection) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(newOptions, OpenLayers.Projection.defaults[newOptions.projection.getCode()]) │ │ │ │ │ - } │ │ │ │ │ - if (newOptions.maxExtent && !(newOptions.maxExtent instanceof OpenLayers.Bounds)) { │ │ │ │ │ - newOptions.maxExtent = new OpenLayers.Bounds(newOptions.maxExtent) │ │ │ │ │ - } │ │ │ │ │ - if (newOptions.minExtent && !(newOptions.minExtent instanceof OpenLayers.Bounds)) { │ │ │ │ │ - newOptions.minExtent = new OpenLayers.Bounds(newOptions.minExtent) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.extend(this.options, newOptions); │ │ │ │ │ - OpenLayers.Util.extend(this, newOptions); │ │ │ │ │ - if (this.projection && this.projection.getUnits()) { │ │ │ │ │ - this.units = this.projection.getUnits() │ │ │ │ │ - } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - var resolution = this.map.getResolution(); │ │ │ │ │ - var properties = this.RESOLUTION_PROPERTIES.concat(["projection", "units", "minExtent", "maxExtent"]); │ │ │ │ │ - for (var o in newOptions) { │ │ │ │ │ - if (newOptions.hasOwnProperty(o) && OpenLayers.Util.indexOf(properties, o) >= 0) { │ │ │ │ │ - this.initResolutions(); │ │ │ │ │ - if (reinitialize && this.map.baseLayer === this) { │ │ │ │ │ - this.map.setCenter(this.map.getCenter(), this.map.getZoomForResolution(resolution), false, true); │ │ │ │ │ - this.map.events.triggerEvent("changebaselayer", { │ │ │ │ │ - layer: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - onMapResize: function() {}, │ │ │ │ │ - redraw: function() { │ │ │ │ │ - var redrawn = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.inRange = this.calculateInRange(); │ │ │ │ │ - var extent = this.getExtent(); │ │ │ │ │ - if (extent && this.inRange && this.visibility) { │ │ │ │ │ - var zoomChanged = true; │ │ │ │ │ - this.moveTo(extent, zoomChanged, false); │ │ │ │ │ - this.events.triggerEvent("moveend", { │ │ │ │ │ - zoomChanged: zoomChanged │ │ │ │ │ - }); │ │ │ │ │ - redrawn = true │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return redrawn │ │ │ │ │ - }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - var display = this.visibility; │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - display = display && this.inRange │ │ │ │ │ - } │ │ │ │ │ - this.display(display) │ │ │ │ │ - }, │ │ │ │ │ - moveByPx: function(dx, dy) {}, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - if (this.map == null) { │ │ │ │ │ - this.map = map; │ │ │ │ │ - this.maxExtent = this.maxExtent || this.map.maxExtent; │ │ │ │ │ - this.minExtent = this.minExtent || this.map.minExtent; │ │ │ │ │ - this.projection = this.projection || this.map.projection; │ │ │ │ │ - if (typeof this.projection == "string") { │ │ │ │ │ - this.projection = new OpenLayers.Projection(this.projection) │ │ │ │ │ - } │ │ │ │ │ - this.units = this.projection.getUnits() || this.units || this.map.units; │ │ │ │ │ - this.initResolutions(); │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - this.inRange = this.calculateInRange(); │ │ │ │ │ - var show = this.visibility && this.inRange; │ │ │ │ │ - this.div.style.display = show ? "" : "none" │ │ │ │ │ - } │ │ │ │ │ - this.setTileSize() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - afterAdd: function() {}, │ │ │ │ │ - removeMap: function(map) {}, │ │ │ │ │ - getImageSize: function(bounds) { │ │ │ │ │ - return this.imageSize || this.tileSize │ │ │ │ │ - }, │ │ │ │ │ - setTileSize: function(size) { │ │ │ │ │ - var tileSize = size ? size : this.tileSize ? this.tileSize : this.map.getTileSize(); │ │ │ │ │ - this.tileSize = tileSize; │ │ │ │ │ - if (this.gutter) { │ │ │ │ │ - this.imageSize = new OpenLayers.Size(tileSize.w + 2 * this.gutter, tileSize.h + 2 * this.gutter) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getVisibility: function() { │ │ │ │ │ - return this.visibility │ │ │ │ │ - }, │ │ │ │ │ - setVisibility: function(visibility) { │ │ │ │ │ - if (visibility != this.visibility) { │ │ │ │ │ - this.visibility = visibility; │ │ │ │ │ - this.display(visibility); │ │ │ │ │ - this.redraw(); │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "visibility" │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("visibilitychanged") │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - display: function(display) { │ │ │ │ │ - if (display != (this.div.style.display != "none")) { │ │ │ │ │ - this.div.style.display = display && this.calculateInRange() ? "block" : "none" │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - calculateInRange: function() { │ │ │ │ │ - var inRange = false; │ │ │ │ │ - if (this.alwaysInRange) { │ │ │ │ │ - inRange = true │ │ │ │ │ - } else { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - var resolution = this.map.getResolution(); │ │ │ │ │ - inRange = resolution >= this.minResolution && resolution <= this.maxResolution │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return inRange │ │ │ │ │ + featureDx: 0, │ │ │ │ │ + initialize: function(containerID, options) { │ │ │ │ │ + this.container = OpenLayers.Util.getElement(containerID); │ │ │ │ │ + OpenLayers.Util.extend(this, options) │ │ │ │ │ }, │ │ │ │ │ - setIsBaseLayer: function(isBaseLayer) { │ │ │ │ │ - if (isBaseLayer != this.isBaseLayer) { │ │ │ │ │ - this.isBaseLayer = isBaseLayer; │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changebaselayer", { │ │ │ │ │ - layer: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.container = null; │ │ │ │ │ + this.extent = null; │ │ │ │ │ + this.size = null; │ │ │ │ │ + this.resolution = null; │ │ │ │ │ + this.map = null │ │ │ │ │ }, │ │ │ │ │ - initResolutions: function() { │ │ │ │ │ - var i, len, p; │ │ │ │ │ - var props = {}, │ │ │ │ │ - alwaysInRange = true; │ │ │ │ │ - for (i = 0, len = this.RESOLUTION_PROPERTIES.length; i < len; i++) { │ │ │ │ │ - p = this.RESOLUTION_PROPERTIES[i]; │ │ │ │ │ - props[p] = this.options[p]; │ │ │ │ │ - if (alwaysInRange && this.options[p]) { │ │ │ │ │ - alwaysInRange = false │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.options.alwaysInRange == null) { │ │ │ │ │ - this.alwaysInRange = alwaysInRange │ │ │ │ │ - } │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - props.resolutions = this.resolutionsFromScales(props.scales) │ │ │ │ │ - } │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - props.resolutions = this.calculateResolutions(props) │ │ │ │ │ - } │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - for (i = 0, len = this.RESOLUTION_PROPERTIES.length; i < len; i++) { │ │ │ │ │ - p = this.RESOLUTION_PROPERTIES[i]; │ │ │ │ │ - props[p] = this.options[p] != null ? this.options[p] : this.map[p] │ │ │ │ │ - } │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - props.resolutions = this.resolutionsFromScales(props.scales) │ │ │ │ │ - } │ │ │ │ │ - if (props.resolutions == null) { │ │ │ │ │ - props.resolutions = this.calculateResolutions(props) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var maxResolution; │ │ │ │ │ - if (this.options.maxResolution && this.options.maxResolution !== "auto") { │ │ │ │ │ - maxResolution = this.options.maxResolution │ │ │ │ │ - } │ │ │ │ │ - if (this.options.minScale) { │ │ │ │ │ - maxResolution = OpenLayers.Util.getResolutionFromScale(this.options.minScale, this.units) │ │ │ │ │ - } │ │ │ │ │ - var minResolution; │ │ │ │ │ - if (this.options.minResolution && this.options.minResolution !== "auto") { │ │ │ │ │ - minResolution = this.options.minResolution │ │ │ │ │ - } │ │ │ │ │ - if (this.options.maxScale) { │ │ │ │ │ - minResolution = OpenLayers.Util.getResolutionFromScale(this.options.maxScale, this.units) │ │ │ │ │ - } │ │ │ │ │ - if (props.resolutions) { │ │ │ │ │ - props.resolutions.sort(function(a, b) { │ │ │ │ │ - return b - a │ │ │ │ │ - }); │ │ │ │ │ - if (!maxResolution) { │ │ │ │ │ - maxResolution = props.resolutions[0] │ │ │ │ │ - } │ │ │ │ │ - if (!minResolution) { │ │ │ │ │ - var lastIdx = props.resolutions.length - 1; │ │ │ │ │ - minResolution = props.resolutions[lastIdx] │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.resolutions = props.resolutions; │ │ │ │ │ - if (this.resolutions) { │ │ │ │ │ - len = this.resolutions.length; │ │ │ │ │ - this.scales = new Array(len); │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - this.scales[i] = OpenLayers.Util.getScaleFromResolution(this.resolutions[i], this.units) │ │ │ │ │ - } │ │ │ │ │ - this.numZoomLevels = len │ │ │ │ │ - } │ │ │ │ │ - this.minResolution = minResolution; │ │ │ │ │ - if (minResolution) { │ │ │ │ │ - this.maxScale = OpenLayers.Util.getScaleFromResolution(minResolution, this.units) │ │ │ │ │ - } │ │ │ │ │ - this.maxResolution = maxResolution; │ │ │ │ │ - if (maxResolution) { │ │ │ │ │ - this.minScale = OpenLayers.Util.getScaleFromResolution(maxResolution, this.units) │ │ │ │ │ - } │ │ │ │ │ + supported: function() { │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - resolutionsFromScales: function(scales) { │ │ │ │ │ - if (scales == null) { │ │ │ │ │ - return │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + this.extent = extent.clone(); │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + var ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ + extent = extent.scale(1 / ratio); │ │ │ │ │ + this.extent = extent.wrapDateLine(this.map.getMaxExtent()).scale(ratio) │ │ │ │ │ } │ │ │ │ │ - var resolutions, i, len; │ │ │ │ │ - len = scales.length; │ │ │ │ │ - resolutions = new Array(len); │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - resolutions[i] = OpenLayers.Util.getResolutionFromScale(scales[i], this.units) │ │ │ │ │ + if (resolutionChanged) { │ │ │ │ │ + this.resolution = null │ │ │ │ │ } │ │ │ │ │ - return resolutions │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - calculateResolutions: function(props) { │ │ │ │ │ - var viewSize, wRes, hRes; │ │ │ │ │ - var maxResolution = props.maxResolution; │ │ │ │ │ - if (props.minScale != null) { │ │ │ │ │ - maxResolution = OpenLayers.Util.getResolutionFromScale(props.minScale, this.units) │ │ │ │ │ - } else if (maxResolution == "auto" && this.maxExtent != null) { │ │ │ │ │ - viewSize = this.map.getSize(); │ │ │ │ │ - wRes = this.maxExtent.getWidth() / viewSize.w; │ │ │ │ │ - hRes = this.maxExtent.getHeight() / viewSize.h; │ │ │ │ │ - maxResolution = Math.max(wRes, hRes) │ │ │ │ │ - } │ │ │ │ │ - var minResolution = props.minResolution; │ │ │ │ │ - if (props.maxScale != null) { │ │ │ │ │ - minResolution = OpenLayers.Util.getResolutionFromScale(props.maxScale, this.units) │ │ │ │ │ - } else if (props.minResolution == "auto" && this.minExtent != null) { │ │ │ │ │ - viewSize = this.map.getSize(); │ │ │ │ │ - wRes = this.minExtent.getWidth() / viewSize.w; │ │ │ │ │ - hRes = this.minExtent.getHeight() / viewSize.h; │ │ │ │ │ - minResolution = Math.max(wRes, hRes) │ │ │ │ │ - } │ │ │ │ │ - if (typeof maxResolution !== "number" && typeof minResolution !== "number" && this.maxExtent != null) { │ │ │ │ │ - var tileSize = this.map.getTileSize(); │ │ │ │ │ - maxResolution = Math.max(this.maxExtent.getWidth() / tileSize.w, this.maxExtent.getHeight() / tileSize.h) │ │ │ │ │ - } │ │ │ │ │ - var maxZoomLevel = props.maxZoomLevel; │ │ │ │ │ - var numZoomLevels = props.numZoomLevels; │ │ │ │ │ - if (typeof minResolution === "number" && typeof maxResolution === "number" && numZoomLevels === undefined) { │ │ │ │ │ - var ratio = maxResolution / minResolution; │ │ │ │ │ - numZoomLevels = Math.floor(Math.log(ratio) / Math.log(2)) + 1 │ │ │ │ │ - } else if (numZoomLevels === undefined && maxZoomLevel != null) { │ │ │ │ │ - numZoomLevels = maxZoomLevel + 1 │ │ │ │ │ - } │ │ │ │ │ - if (typeof numZoomLevels !== "number" || numZoomLevels <= 0 || typeof maxResolution !== "number" && typeof minResolution !== "number") { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var resolutions = new Array(numZoomLevels); │ │ │ │ │ - var base = 2; │ │ │ │ │ - if (typeof minResolution == "number" && typeof maxResolution == "number") { │ │ │ │ │ - base = Math.pow(maxResolution / minResolution, 1 / (numZoomLevels - 1)) │ │ │ │ │ - } │ │ │ │ │ - var i; │ │ │ │ │ - if (typeof maxResolution === "number") { │ │ │ │ │ - for (i = 0; i < numZoomLevels; i++) { │ │ │ │ │ - resolutions[i] = maxResolution / Math.pow(base, i) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - for (i = 0; i < numZoomLevels; i++) { │ │ │ │ │ - resolutions[numZoomLevels - 1 - i] = minResolution * Math.pow(base, i) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return resolutions │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + this.size = size.clone(); │ │ │ │ │ + this.resolution = null │ │ │ │ │ }, │ │ │ │ │ getResolution: function() { │ │ │ │ │ - var zoom = this.map.getZoom(); │ │ │ │ │ - return this.getResolutionForZoom(zoom) │ │ │ │ │ - }, │ │ │ │ │ - getExtent: function() { │ │ │ │ │ - return this.map.calculateBounds() │ │ │ │ │ - }, │ │ │ │ │ - getZoomForExtent: function(extent, closest) { │ │ │ │ │ - var viewSize = this.map.getSize(); │ │ │ │ │ - var idealResolution = Math.max(extent.getWidth() / viewSize.w, extent.getHeight() / viewSize.h); │ │ │ │ │ - return this.getZoomForResolution(idealResolution, closest) │ │ │ │ │ + this.resolution = this.resolution || this.map.getResolution(); │ │ │ │ │ + return this.resolution │ │ │ │ │ }, │ │ │ │ │ - getDataExtent: function() {}, │ │ │ │ │ - getResolutionForZoom: function(zoom) { │ │ │ │ │ - zoom = Math.max(0, Math.min(zoom, this.resolutions.length - 1)); │ │ │ │ │ - var resolution; │ │ │ │ │ - if (this.map.fractionalZoom) { │ │ │ │ │ - var low = Math.floor(zoom); │ │ │ │ │ - var high = Math.ceil(zoom); │ │ │ │ │ - resolution = this.resolutions[low] - (zoom - low) * (this.resolutions[low] - this.resolutions[high]) │ │ │ │ │ - } else { │ │ │ │ │ - resolution = this.resolutions[Math.round(zoom)] │ │ │ │ │ + drawFeature: function(feature, style) { │ │ │ │ │ + if (style == null) { │ │ │ │ │ + style = feature.style │ │ │ │ │ } │ │ │ │ │ - return resolution │ │ │ │ │ - }, │ │ │ │ │ - getZoomForResolution: function(resolution, closest) { │ │ │ │ │ - var zoom, i, len; │ │ │ │ │ - if (this.map.fractionalZoom) { │ │ │ │ │ - var lowZoom = 0; │ │ │ │ │ - var highZoom = this.resolutions.length - 1; │ │ │ │ │ - var highRes = this.resolutions[lowZoom]; │ │ │ │ │ - var lowRes = this.resolutions[highZoom]; │ │ │ │ │ - var res; │ │ │ │ │ - for (i = 0, len = this.resolutions.length; i < len; ++i) { │ │ │ │ │ - res = this.resolutions[i]; │ │ │ │ │ - if (res >= resolution) { │ │ │ │ │ - highRes = res; │ │ │ │ │ - lowZoom = i │ │ │ │ │ - } │ │ │ │ │ - if (res <= resolution) { │ │ │ │ │ - lowRes = res; │ │ │ │ │ - highZoom = i; │ │ │ │ │ - break │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + var bounds = feature.geometry.getBounds(); │ │ │ │ │ + if (bounds) { │ │ │ │ │ + var worldBounds; │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + worldBounds = this.map.getMaxExtent() │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - var dRes = highRes - lowRes; │ │ │ │ │ - if (dRes > 0) { │ │ │ │ │ - zoom = lowZoom + (highRes - resolution) / dRes │ │ │ │ │ - } else { │ │ │ │ │ - zoom = lowZoom │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - var diff; │ │ │ │ │ - var minDiff = Number.POSITIVE_INFINITY; │ │ │ │ │ - for (i = 0, len = this.resolutions.length; i < len; i++) { │ │ │ │ │ - if (closest) { │ │ │ │ │ - diff = Math.abs(this.resolutions[i] - resolution); │ │ │ │ │ - if (diff > minDiff) { │ │ │ │ │ - break │ │ │ │ │ + if (!bounds.intersectsBounds(this.extent, { │ │ │ │ │ + worldBounds: worldBounds │ │ │ │ │ + })) { │ │ │ │ │ + style = { │ │ │ │ │ + display: "none" │ │ │ │ │ } │ │ │ │ │ - minDiff = diff │ │ │ │ │ } else { │ │ │ │ │ - if (this.resolutions[i] < resolution) { │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - zoom = Math.max(0, i - 1) │ │ │ │ │ - } │ │ │ │ │ - return zoom │ │ │ │ │ - }, │ │ │ │ │ - getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ - var lonlat = null; │ │ │ │ │ - var map = this.map; │ │ │ │ │ - if (viewPortPx != null && map.minPx) { │ │ │ │ │ - var res = map.getResolution(); │ │ │ │ │ - var maxExtent = map.getMaxExtent({ │ │ │ │ │ - restricted: true │ │ │ │ │ - }); │ │ │ │ │ - var lon = (viewPortPx.x - map.minPx.x) * res + maxExtent.left; │ │ │ │ │ - var lat = (map.minPx.y - viewPortPx.y) * res + maxExtent.top; │ │ │ │ │ - lonlat = new OpenLayers.LonLat(lon, lat); │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - lonlat = lonlat.wrapDateLine(this.maxExtent) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return lonlat │ │ │ │ │ - }, │ │ │ │ │ - getViewPortPxFromLonLat: function(lonlat, resolution) { │ │ │ │ │ - var px = null; │ │ │ │ │ - if (lonlat != null) { │ │ │ │ │ - resolution = resolution || this.map.getResolution(); │ │ │ │ │ - var extent = this.map.calculateBounds(null, resolution); │ │ │ │ │ - px = new OpenLayers.Pixel(1 / resolution * (lonlat.lon - extent.left), 1 / resolution * (extent.top - lonlat.lat)) │ │ │ │ │ - } │ │ │ │ │ - return px │ │ │ │ │ - }, │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - if (opacity != this.opacity) { │ │ │ │ │ - this.opacity = opacity; │ │ │ │ │ - var childNodes = this.div.childNodes; │ │ │ │ │ - for (var i = 0, len = childNodes.length; i < len; ++i) { │ │ │ │ │ - var element = childNodes[i].firstChild || childNodes[i]; │ │ │ │ │ - var lastChild = childNodes[i].lastChild; │ │ │ │ │ - if (lastChild && lastChild.nodeName.toLowerCase() === "iframe") { │ │ │ │ │ - element = lastChild.parentNode │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.modifyDOMElement(element, null, null, null, null, null, null, opacity) │ │ │ │ │ - } │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "opacity" │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getZIndex: function() { │ │ │ │ │ - return this.div.style.zIndex │ │ │ │ │ - }, │ │ │ │ │ - setZIndex: function(zIndex) { │ │ │ │ │ - this.div.style.zIndex = zIndex │ │ │ │ │ - }, │ │ │ │ │ - adjustBounds: function(bounds) { │ │ │ │ │ - if (this.gutter) { │ │ │ │ │ - var mapGutter = this.gutter * this.map.getResolution(); │ │ │ │ │ - bounds = new OpenLayers.Bounds(bounds.left - mapGutter, bounds.bottom - mapGutter, bounds.right + mapGutter, bounds.top + mapGutter) │ │ │ │ │ - } │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - var wrappingOptions = { │ │ │ │ │ - rightTolerance: this.getResolution(), │ │ │ │ │ - leftTolerance: this.getResolution() │ │ │ │ │ - }; │ │ │ │ │ - bounds = bounds.wrapDateLine(this.maxExtent, wrappingOptions) │ │ │ │ │ - } │ │ │ │ │ - return bounds │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer" │ │ │ │ │ -}); │ │ │ │ │ -(function() { │ │ │ │ │ - var oXMLHttpRequest = window.XMLHttpRequest; │ │ │ │ │ - var bGecko = !!window.controllers, │ │ │ │ │ - bIE = window.document.all && !window.opera, │ │ │ │ │ - bIE7 = bIE && window.navigator.userAgent.match(/MSIE 7.0/); │ │ │ │ │ - │ │ │ │ │ - function fXMLHttpRequest() { │ │ │ │ │ - this._object = oXMLHttpRequest && !bIE7 ? new oXMLHttpRequest : new window.ActiveXObject("Microsoft.XMLHTTP"); │ │ │ │ │ - this._listeners = [] │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function cXMLHttpRequest() { │ │ │ │ │ - return new fXMLHttpRequest │ │ │ │ │ - } │ │ │ │ │ - cXMLHttpRequest.prototype = fXMLHttpRequest.prototype; │ │ │ │ │ - if (bGecko && oXMLHttpRequest.wrapped) cXMLHttpRequest.wrapped = oXMLHttpRequest.wrapped; │ │ │ │ │ - cXMLHttpRequest.UNSENT = 0; │ │ │ │ │ - cXMLHttpRequest.OPENED = 1; │ │ │ │ │ - cXMLHttpRequest.HEADERS_RECEIVED = 2; │ │ │ │ │ - cXMLHttpRequest.LOADING = 3; │ │ │ │ │ - cXMLHttpRequest.DONE = 4; │ │ │ │ │ - cXMLHttpRequest.prototype.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ - cXMLHttpRequest.prototype.responseText = ""; │ │ │ │ │ - cXMLHttpRequest.prototype.responseXML = null; │ │ │ │ │ - cXMLHttpRequest.prototype.status = 0; │ │ │ │ │ - cXMLHttpRequest.prototype.statusText = ""; │ │ │ │ │ - cXMLHttpRequest.prototype.priority = "NORMAL"; │ │ │ │ │ - cXMLHttpRequest.prototype.onreadystatechange = null; │ │ │ │ │ - cXMLHttpRequest.onreadystatechange = null; │ │ │ │ │ - cXMLHttpRequest.onopen = null; │ │ │ │ │ - cXMLHttpRequest.onsend = null; │ │ │ │ │ - cXMLHttpRequest.onabort = null; │ │ │ │ │ - cXMLHttpRequest.prototype.open = function(sMethod, sUrl, bAsync, sUser, sPassword) { │ │ │ │ │ - delete this._headers; │ │ │ │ │ - if (arguments.length < 3) bAsync = true; │ │ │ │ │ - this._async = bAsync; │ │ │ │ │ - var oRequest = this, │ │ │ │ │ - nState = this.readyState, │ │ │ │ │ - fOnUnload; │ │ │ │ │ - if (bIE && bAsync) { │ │ │ │ │ - fOnUnload = function() { │ │ │ │ │ - if (nState != cXMLHttpRequest.DONE) { │ │ │ │ │ - fCleanTransport(oRequest); │ │ │ │ │ - oRequest.abort() │ │ │ │ │ + this.calculateFeatureDx(bounds, worldBounds) │ │ │ │ │ } │ │ │ │ │ - }; │ │ │ │ │ - window.attachEvent("onunload", fOnUnload) │ │ │ │ │ - } │ │ │ │ │ - if (cXMLHttpRequest.onopen) cXMLHttpRequest.onopen.apply(this, arguments); │ │ │ │ │ - if (arguments.length > 4) this._object.open(sMethod, sUrl, bAsync, sUser, sPassword); │ │ │ │ │ - else if (arguments.length > 3) this._object.open(sMethod, sUrl, bAsync, sUser); │ │ │ │ │ - else this._object.open(sMethod, sUrl, bAsync); │ │ │ │ │ - this.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ - fReadyStateChange(this); │ │ │ │ │ - this._object.onreadystatechange = function() { │ │ │ │ │ - if (bGecko && !bAsync) return; │ │ │ │ │ - oRequest.readyState = oRequest._object.readyState; │ │ │ │ │ - fSynchronizeValues(oRequest); │ │ │ │ │ - if (oRequest._aborted) { │ │ │ │ │ - oRequest.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (oRequest.readyState == cXMLHttpRequest.DONE) { │ │ │ │ │ - delete oRequest._data; │ │ │ │ │ - fCleanTransport(oRequest); │ │ │ │ │ - if (bIE && bAsync) window.detachEvent("onunload", fOnUnload) │ │ │ │ │ - } │ │ │ │ │ - if (nState != oRequest.readyState) fReadyStateChange(oRequest); │ │ │ │ │ - nState = oRequest.readyState │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - function fXMLHttpRequest_send(oRequest) { │ │ │ │ │ - oRequest._object.send(oRequest._data); │ │ │ │ │ - if (bGecko && !oRequest._async) { │ │ │ │ │ - oRequest.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ - fSynchronizeValues(oRequest); │ │ │ │ │ - while (oRequest.readyState < cXMLHttpRequest.DONE) { │ │ │ │ │ - oRequest.readyState++; │ │ │ │ │ - fReadyStateChange(oRequest); │ │ │ │ │ - if (oRequest._aborted) return │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - cXMLHttpRequest.prototype.send = function(vData) { │ │ │ │ │ - if (cXMLHttpRequest.onsend) cXMLHttpRequest.onsend.apply(this, arguments); │ │ │ │ │ - if (!arguments.length) vData = null; │ │ │ │ │ - if (vData && vData.nodeType) { │ │ │ │ │ - vData = window.XMLSerializer ? (new window.XMLSerializer).serializeToString(vData) : vData.xml; │ │ │ │ │ - if (!this._headers["Content-Type"]) this._object.setRequestHeader("Content-Type", "application/xml") │ │ │ │ │ - } │ │ │ │ │ - this._data = vData; │ │ │ │ │ - fXMLHttpRequest_send(this) │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.abort = function() { │ │ │ │ │ - if (cXMLHttpRequest.onabort) cXMLHttpRequest.onabort.apply(this, arguments); │ │ │ │ │ - if (this.readyState > cXMLHttpRequest.UNSENT) this._aborted = true; │ │ │ │ │ - this._object.abort(); │ │ │ │ │ - fCleanTransport(this); │ │ │ │ │ - this.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ - delete this._data │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.getAllResponseHeaders = function() { │ │ │ │ │ - return this._object.getAllResponseHeaders() │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.getResponseHeader = function(sName) { │ │ │ │ │ - return this._object.getResponseHeader(sName) │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.setRequestHeader = function(sName, sValue) { │ │ │ │ │ - if (!this._headers) this._headers = {}; │ │ │ │ │ - this._headers[sName] = sValue; │ │ │ │ │ - return this._object.setRequestHeader(sName, sValue) │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.addEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ - for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ - if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) return; │ │ │ │ │ - this._listeners.push([sName, fHandler, bUseCapture]) │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.removeEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ - for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ - if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) break; │ │ │ │ │ - if (oListener) this._listeners.splice(nIndex, 1) │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.dispatchEvent = function(oEvent) { │ │ │ │ │ - var oEventPseudo = { │ │ │ │ │ - type: oEvent.type, │ │ │ │ │ - target: this, │ │ │ │ │ - currentTarget: this, │ │ │ │ │ - eventPhase: 2, │ │ │ │ │ - bubbles: oEvent.bubbles, │ │ │ │ │ - cancelable: oEvent.cancelable, │ │ │ │ │ - timeStamp: oEvent.timeStamp, │ │ │ │ │ - stopPropagation: function() {}, │ │ │ │ │ - preventDefault: function() {}, │ │ │ │ │ - initEvent: function() {} │ │ │ │ │ - }; │ │ │ │ │ - if (oEventPseudo.type == "readystatechange" && this.onreadystatechange)(this.onreadystatechange.handleEvent || this.onreadystatechange).apply(this, [oEventPseudo]); │ │ │ │ │ - for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ - if (oListener[0] == oEventPseudo.type && !oListener[2])(oListener[1].handleEvent || oListener[1]).apply(this, [oEventPseudo]) │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.toString = function() { │ │ │ │ │ - return "[" + "object" + " " + "XMLHttpRequest" + "]" │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.toString = function() { │ │ │ │ │ - return "[" + "XMLHttpRequest" + "]" │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - function fReadyStateChange(oRequest) { │ │ │ │ │ - if (cXMLHttpRequest.onreadystatechange) cXMLHttpRequest.onreadystatechange.apply(oRequest); │ │ │ │ │ - oRequest.dispatchEvent({ │ │ │ │ │ - type: "readystatechange", │ │ │ │ │ - bubbles: false, │ │ │ │ │ - cancelable: false, │ │ │ │ │ - timeStamp: new Date + 0 │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function fGetDocument(oRequest) { │ │ │ │ │ - var oDocument = oRequest.responseXML, │ │ │ │ │ - sResponse = oRequest.responseText; │ │ │ │ │ - if (bIE && sResponse && oDocument && !oDocument.documentElement && oRequest.getResponseHeader("Content-Type").match(/[^\/]+\/[^\+]+\+xml/)) { │ │ │ │ │ - oDocument = new window.ActiveXObject("Microsoft.XMLDOM"); │ │ │ │ │ - oDocument.async = false; │ │ │ │ │ - oDocument.validateOnParse = false; │ │ │ │ │ - oDocument.loadXML(sResponse) │ │ │ │ │ - } │ │ │ │ │ - if (oDocument) │ │ │ │ │ - if (bIE && oDocument.parseError != 0 || !oDocument.documentElement || oDocument.documentElement && oDocument.documentElement.tagName == "parsererror") return null; │ │ │ │ │ - return oDocument │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function fSynchronizeValues(oRequest) { │ │ │ │ │ - try { │ │ │ │ │ - oRequest.responseText = oRequest._object.responseText │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - try { │ │ │ │ │ - oRequest.responseXML = fGetDocument(oRequest._object) │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - try { │ │ │ │ │ - oRequest.status = oRequest._object.status │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - try { │ │ │ │ │ - oRequest.statusText = oRequest._object.statusText │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function fCleanTransport(oRequest) { │ │ │ │ │ - oRequest._object.onreadystatechange = new window.Function │ │ │ │ │ - } │ │ │ │ │ - if (!window.Function.prototype.apply) { │ │ │ │ │ - window.Function.prototype.apply = function(oRequest, oArguments) { │ │ │ │ │ - if (!oArguments) oArguments = []; │ │ │ │ │ - oRequest.__func = this; │ │ │ │ │ - oRequest.__func(oArguments[0], oArguments[1], oArguments[2], oArguments[3], oArguments[4]); │ │ │ │ │ - delete oRequest.__func │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!OpenLayers.Request) { │ │ │ │ │ - OpenLayers.Request = {} │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Request.XMLHttpRequest = cXMLHttpRequest │ │ │ │ │ -})(); │ │ │ │ │ -OpenLayers.ProxyHost = ""; │ │ │ │ │ -if (!OpenLayers.Request) { │ │ │ │ │ - OpenLayers.Request = {} │ │ │ │ │ -} │ │ │ │ │ -OpenLayers.Util.extend(OpenLayers.Request, { │ │ │ │ │ - DEFAULT_CONFIG: { │ │ │ │ │ - method: "GET", │ │ │ │ │ - url: window.location.href, │ │ │ │ │ - async: true, │ │ │ │ │ - user: undefined, │ │ │ │ │ - password: undefined, │ │ │ │ │ - params: null, │ │ │ │ │ - proxy: OpenLayers.ProxyHost, │ │ │ │ │ - headers: {}, │ │ │ │ │ - data: null, │ │ │ │ │ - callback: function() {}, │ │ │ │ │ - success: null, │ │ │ │ │ - failure: null, │ │ │ │ │ - scope: null │ │ │ │ │ - }, │ │ │ │ │ - URL_SPLIT_REGEX: /([^:]*:)\/\/([^:]*:?[^@]*@)?([^:\/\?]*):?([^\/\?]*)/, │ │ │ │ │ - events: new OpenLayers.Events(this), │ │ │ │ │ - makeSameOrigin: function(url, proxy) { │ │ │ │ │ - var sameOrigin = url.indexOf("http") !== 0; │ │ │ │ │ - var urlParts = !sameOrigin && url.match(this.URL_SPLIT_REGEX); │ │ │ │ │ - if (urlParts) { │ │ │ │ │ - var location = window.location; │ │ │ │ │ - sameOrigin = urlParts[1] == location.protocol && urlParts[3] == location.hostname; │ │ │ │ │ - var uPort = urlParts[4], │ │ │ │ │ - lPort = location.port; │ │ │ │ │ - if (uPort != 80 && uPort != "" || lPort != "80" && lPort != "") { │ │ │ │ │ - sameOrigin = sameOrigin && uPort == lPort │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!sameOrigin) { │ │ │ │ │ - if (proxy) { │ │ │ │ │ - if (typeof proxy == "function") { │ │ │ │ │ - url = proxy(url) │ │ │ │ │ + var rendered = this.drawGeometry(feature.geometry, style, feature.id); │ │ │ │ │ + if (style.display != "none" && style.label && rendered !== false) { │ │ │ │ │ + var location = feature.geometry.getCentroid(); │ │ │ │ │ + if (style.labelXOffset || style.labelYOffset) { │ │ │ │ │ + var xOffset = isNaN(style.labelXOffset) ? 0 : style.labelXOffset; │ │ │ │ │ + var yOffset = isNaN(style.labelYOffset) ? 0 : style.labelYOffset; │ │ │ │ │ + var res = this.getResolution(); │ │ │ │ │ + location.move(xOffset * res, yOffset * res) │ │ │ │ │ + } │ │ │ │ │ + this.drawText(feature.id, style, location) │ │ │ │ │ } else { │ │ │ │ │ - url = proxy + encodeURIComponent(url) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return url │ │ │ │ │ - }, │ │ │ │ │ - issue: function(config) { │ │ │ │ │ - var defaultConfig = OpenLayers.Util.extend(this.DEFAULT_CONFIG, { │ │ │ │ │ - proxy: OpenLayers.ProxyHost │ │ │ │ │ - }); │ │ │ │ │ - config = config || {}; │ │ │ │ │ - config.headers = config.headers || {}; │ │ │ │ │ - config = OpenLayers.Util.applyDefaults(config, defaultConfig); │ │ │ │ │ - config.headers = OpenLayers.Util.applyDefaults(config.headers, defaultConfig.headers); │ │ │ │ │ - var customRequestedWithHeader = false, │ │ │ │ │ - headerKey; │ │ │ │ │ - for (headerKey in config.headers) { │ │ │ │ │ - if (config.headers.hasOwnProperty(headerKey)) { │ │ │ │ │ - if (headerKey.toLowerCase() === "x-requested-with") { │ │ │ │ │ - customRequestedWithHeader = true │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (customRequestedWithHeader === false) { │ │ │ │ │ - config.headers["X-Requested-With"] = "XMLHttpRequest" │ │ │ │ │ - } │ │ │ │ │ - var request = new OpenLayers.Request.XMLHttpRequest; │ │ │ │ │ - var url = OpenLayers.Util.urlAppend(config.url, OpenLayers.Util.getParameterString(config.params || {})); │ │ │ │ │ - url = OpenLayers.Request.makeSameOrigin(url, config.proxy); │ │ │ │ │ - request.open(config.method, url, config.async, config.user, config.password); │ │ │ │ │ - for (var header in config.headers) { │ │ │ │ │ - request.setRequestHeader(header, config.headers[header]) │ │ │ │ │ - } │ │ │ │ │ - var events = this.events; │ │ │ │ │ - var self = this; │ │ │ │ │ - request.onreadystatechange = function() { │ │ │ │ │ - if (request.readyState == OpenLayers.Request.XMLHttpRequest.DONE) { │ │ │ │ │ - var proceed = events.triggerEvent("complete", { │ │ │ │ │ - request: request, │ │ │ │ │ - config: config, │ │ │ │ │ - requestUrl: url │ │ │ │ │ - }); │ │ │ │ │ - if (proceed !== false) { │ │ │ │ │ - self.runCallbacks({ │ │ │ │ │ - request: request, │ │ │ │ │ - config: config, │ │ │ │ │ - requestUrl: url │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - if (config.async === false) { │ │ │ │ │ - request.send(config.data) │ │ │ │ │ - } else { │ │ │ │ │ - window.setTimeout(function() { │ │ │ │ │ - if (request.readyState !== 0) { │ │ │ │ │ - request.send(config.data) │ │ │ │ │ - } │ │ │ │ │ - }, 0) │ │ │ │ │ - } │ │ │ │ │ - return request │ │ │ │ │ - }, │ │ │ │ │ - runCallbacks: function(options) { │ │ │ │ │ - var request = options.request; │ │ │ │ │ - var config = options.config; │ │ │ │ │ - var complete = config.scope ? OpenLayers.Function.bind(config.callback, config.scope) : config.callback; │ │ │ │ │ - var success; │ │ │ │ │ - if (config.success) { │ │ │ │ │ - success = config.scope ? OpenLayers.Function.bind(config.success, config.scope) : config.success │ │ │ │ │ - } │ │ │ │ │ - var failure; │ │ │ │ │ - if (config.failure) { │ │ │ │ │ - failure = config.scope ? OpenLayers.Function.bind(config.failure, config.scope) : config.failure │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.Util.createUrlObject(config.url).protocol == "file:" && request.responseText) { │ │ │ │ │ - request.status = 200 │ │ │ │ │ - } │ │ │ │ │ - complete(request); │ │ │ │ │ - if (!request.status || request.status >= 200 && request.status < 300) { │ │ │ │ │ - this.events.triggerEvent("success", options); │ │ │ │ │ - if (success) { │ │ │ │ │ - success(request) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ - this.events.triggerEvent("failure", options); │ │ │ │ │ - if (failure) { │ │ │ │ │ - failure(request) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - GET: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "GET" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config) │ │ │ │ │ - }, │ │ │ │ │ - POST: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "POST" │ │ │ │ │ - }); │ │ │ │ │ - config.headers = config.headers ? config.headers : {}; │ │ │ │ │ - if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ - config.headers["Content-Type"] = "application/xml" │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Request.issue(config) │ │ │ │ │ - }, │ │ │ │ │ - PUT: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "PUT" │ │ │ │ │ - }); │ │ │ │ │ - config.headers = config.headers ? config.headers : {}; │ │ │ │ │ - if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ - config.headers["Content-Type"] = "application/xml" │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Request.issue(config) │ │ │ │ │ - }, │ │ │ │ │ - DELETE: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "DELETE" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config) │ │ │ │ │ - }, │ │ │ │ │ - HEAD: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "HEAD" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config) │ │ │ │ │ - }, │ │ │ │ │ - OPTIONS: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "OPTIONS" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config) │ │ │ │ │ - } │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Icon = OpenLayers.Class({ │ │ │ │ │ - url: null, │ │ │ │ │ - size: null, │ │ │ │ │ - offset: null, │ │ │ │ │ - calculateOffset: null, │ │ │ │ │ - imageDiv: null, │ │ │ │ │ - px: null, │ │ │ │ │ - initialize: function(url, size, offset, calculateOffset) { │ │ │ │ │ - this.url = url; │ │ │ │ │ - this.size = size || { │ │ │ │ │ - w: 20, │ │ │ │ │ - h: 20 │ │ │ │ │ - }; │ │ │ │ │ - this.offset = offset || { │ │ │ │ │ - x: -(this.size.w / 2), │ │ │ │ │ - y: -(this.size.h / 2) │ │ │ │ │ - }; │ │ │ │ │ - this.calculateOffset = calculateOffset; │ │ │ │ │ - var id = OpenLayers.Util.createUniqueID("OL_Icon_"); │ │ │ │ │ - this.imageDiv = OpenLayers.Util.createAlphaImageDiv(id) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.erase(); │ │ │ │ │ - OpenLayers.Event.stopObservingElement(this.imageDiv.firstChild); │ │ │ │ │ - this.imageDiv.innerHTML = ""; │ │ │ │ │ - this.imageDiv = null │ │ │ │ │ - }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - return new OpenLayers.Icon(this.url, this.size, this.offset, this.calculateOffset) │ │ │ │ │ - }, │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - if (size != null) { │ │ │ │ │ - this.size = size │ │ │ │ │ - } │ │ │ │ │ - this.draw() │ │ │ │ │ - }, │ │ │ │ │ - setUrl: function(url) { │ │ │ │ │ - if (url != null) { │ │ │ │ │ - this.url = url │ │ │ │ │ - } │ │ │ │ │ - this.draw() │ │ │ │ │ - }, │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, this.size, this.url, "absolute"); │ │ │ │ │ - this.moveTo(px); │ │ │ │ │ - return this.imageDiv │ │ │ │ │ - }, │ │ │ │ │ - erase: function() { │ │ │ │ │ - if (this.imageDiv != null && this.imageDiv.parentNode != null) { │ │ │ │ │ - OpenLayers.Element.remove(this.imageDiv) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, null, null, null, null, null, opacity) │ │ │ │ │ - }, │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - if (px != null) { │ │ │ │ │ - this.px = px │ │ │ │ │ - } │ │ │ │ │ - if (this.imageDiv != null) { │ │ │ │ │ - if (this.px == null) { │ │ │ │ │ - this.display(false) │ │ │ │ │ - } else { │ │ │ │ │ - if (this.calculateOffset) { │ │ │ │ │ - this.offset = this.calculateOffset(this.size) │ │ │ │ │ + this.removeText(feature.id) │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, { │ │ │ │ │ - x: this.px.x + this.offset.x, │ │ │ │ │ - y: this.px.y + this.offset.y │ │ │ │ │ - }) │ │ │ │ │ + return rendered │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - display: function(display) { │ │ │ │ │ - this.imageDiv.style.display = display ? "" : "none" │ │ │ │ │ - }, │ │ │ │ │ - isDrawn: function() { │ │ │ │ │ - var isDrawn = this.imageDiv && this.imageDiv.parentNode && this.imageDiv.parentNode.nodeType != 11; │ │ │ │ │ - return isDrawn │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Icon" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Marker = OpenLayers.Class({ │ │ │ │ │ - icon: null, │ │ │ │ │ - lonlat: null, │ │ │ │ │ - events: null, │ │ │ │ │ - map: null, │ │ │ │ │ - initialize: function(lonlat, icon) { │ │ │ │ │ - this.lonlat = lonlat; │ │ │ │ │ - var newIcon = icon ? icon : OpenLayers.Marker.defaultIcon(); │ │ │ │ │ - if (this.icon == null) { │ │ │ │ │ - this.icon = newIcon │ │ │ │ │ - } else { │ │ │ │ │ - this.icon.url = newIcon.url; │ │ │ │ │ - this.icon.size = newIcon.size; │ │ │ │ │ - this.icon.offset = newIcon.offset; │ │ │ │ │ - this.icon.calculateOffset = newIcon.calculateOffset │ │ │ │ │ - } │ │ │ │ │ - this.events = new OpenLayers.Events(this, this.icon.imageDiv) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.erase(); │ │ │ │ │ - this.map = null; │ │ │ │ │ - this.events.destroy(); │ │ │ │ │ - this.events = null; │ │ │ │ │ - if (this.icon != null) { │ │ │ │ │ - this.icon.destroy(); │ │ │ │ │ - this.icon = null │ │ │ │ │ + calculateFeatureDx: function(bounds, worldBounds) { │ │ │ │ │ + this.featureDx = 0; │ │ │ │ │ + if (worldBounds) { │ │ │ │ │ + var worldWidth = worldBounds.getWidth(), │ │ │ │ │ + rendererCenterX = (this.extent.left + this.extent.right) / 2, │ │ │ │ │ + featureCenterX = (bounds.left + bounds.right) / 2, │ │ │ │ │ + worldsAway = Math.round((featureCenterX - rendererCenterX) / worldWidth); │ │ │ │ │ + this.featureDx = worldsAway * worldWidth │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - return this.icon.draw(px) │ │ │ │ │ - }, │ │ │ │ │ - erase: function() { │ │ │ │ │ - if (this.icon != null) { │ │ │ │ │ - this.icon.erase() │ │ │ │ │ + drawGeometry: function(geometry, style, featureId) {}, │ │ │ │ │ + drawText: function(featureId, style, location) {}, │ │ │ │ │ + removeText: function(featureId) {}, │ │ │ │ │ + clear: function() {}, │ │ │ │ │ + getFeatureIdFromEvent: function(evt) {}, │ │ │ │ │ + eraseFeatures: function(features) { │ │ │ │ │ + if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ + features = [features] │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - if (px != null && this.icon != null) { │ │ │ │ │ - this.icon.moveTo(px) │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + this.eraseGeometry(feature.geometry, feature.id); │ │ │ │ │ + this.removeText(feature.id) │ │ │ │ │ } │ │ │ │ │ - this.lonlat = this.map.getLonLatFromLayerPx(px) │ │ │ │ │ }, │ │ │ │ │ - isDrawn: function() { │ │ │ │ │ - var isDrawn = this.icon && this.icon.isDrawn(); │ │ │ │ │ - return isDrawn │ │ │ │ │ + eraseGeometry: function(geometry, featureId) {}, │ │ │ │ │ + moveRoot: function(renderer) {}, │ │ │ │ │ + getRenderLayerId: function() { │ │ │ │ │ + return this.container.id │ │ │ │ │ }, │ │ │ │ │ - onScreen: function() { │ │ │ │ │ - var onScreen = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - var screenBounds = this.map.getExtent(); │ │ │ │ │ - onScreen = screenBounds.containsLonLat(this.lonlat) │ │ │ │ │ + applyDefaultSymbolizer: function(symbolizer) { │ │ │ │ │ + var result = OpenLayers.Util.extend({}, OpenLayers.Renderer.defaultSymbolizer); │ │ │ │ │ + if (symbolizer.stroke === false) { │ │ │ │ │ + delete result.strokeWidth; │ │ │ │ │ + delete result.strokeColor │ │ │ │ │ } │ │ │ │ │ - return onScreen │ │ │ │ │ - }, │ │ │ │ │ - inflate: function(inflate) { │ │ │ │ │ - if (this.icon) { │ │ │ │ │ - this.icon.setSize({ │ │ │ │ │ - w: this.icon.size.w * inflate, │ │ │ │ │ - h: this.icon.size.h * inflate │ │ │ │ │ - }) │ │ │ │ │ + if (symbolizer.fill === false) { │ │ │ │ │ + delete result.fillColor │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Util.extend(result, symbolizer); │ │ │ │ │ + return result │ │ │ │ │ }, │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - this.icon.setOpacity(opacity) │ │ │ │ │ - }, │ │ │ │ │ - setUrl: function(url) { │ │ │ │ │ - this.icon.setUrl(url) │ │ │ │ │ - }, │ │ │ │ │ - display: function(display) { │ │ │ │ │ - this.icon.display(display) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Marker" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Marker.defaultIcon = function() { │ │ │ │ │ - return new OpenLayers.Icon(OpenLayers.Util.getImageLocation("marker.png"), { │ │ │ │ │ - w: 21, │ │ │ │ │ - h: 25 │ │ │ │ │ - }, { │ │ │ │ │ - x: -10.5, │ │ │ │ │ - y: -25 │ │ │ │ │ - }) │ │ │ │ │ +OpenLayers.Renderer.defaultSymbolizer = { │ │ │ │ │ + fillColor: "#000000", │ │ │ │ │ + strokeColor: "#000000", │ │ │ │ │ + strokeWidth: 2, │ │ │ │ │ + fillOpacity: 1, │ │ │ │ │ + strokeOpacity: 1, │ │ │ │ │ + pointRadius: 0, │ │ │ │ │ + labelAlign: "cm" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.symbol = { │ │ │ │ │ + star: [350, 75, 379, 161, 469, 161, 397, 215, 423, 301, 350, 250, 277, 301, 303, 215, 231, 161, 321, 161, 350, 75], │ │ │ │ │ + cross: [4, 0, 6, 0, 6, 4, 10, 4, 10, 6, 6, 6, 6, 10, 4, 10, 4, 6, 0, 6, 0, 4, 4, 4, 4, 0], │ │ │ │ │ + x: [0, 0, 25, 0, 50, 35, 75, 0, 100, 0, 65, 50, 100, 100, 75, 100, 50, 65, 25, 100, 0, 100, 35, 50, 0, 0], │ │ │ │ │ + square: [0, 0, 0, 1, 1, 1, 1, 0, 0, 0], │ │ │ │ │ + triangle: [0, 10, 10, 10, 5, 0, 0, 10] │ │ │ │ │ }; │ │ │ │ │ -OpenLayers.Handler = OpenLayers.Class({ │ │ │ │ │ - id: null, │ │ │ │ │ - control: null, │ │ │ │ │ - map: null, │ │ │ │ │ - keyMask: null, │ │ │ │ │ - active: false, │ │ │ │ │ - evt: null, │ │ │ │ │ - touch: false, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.control = control; │ │ │ │ │ - this.callbacks = callbacks; │ │ │ │ │ - var map = this.map || control.map; │ │ │ │ │ - if (map) { │ │ │ │ │ - this.setMap(map) │ │ │ │ │ - } │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.map = map │ │ │ │ │ - }, │ │ │ │ │ - checkModifiers: function(evt) { │ │ │ │ │ - if (this.keyMask == null) { │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - var keyModifiers = (evt.shiftKey ? OpenLayers.Handler.MOD_SHIFT : 0) | (evt.ctrlKey ? OpenLayers.Handler.MOD_CTRL : 0) | (evt.altKey ? OpenLayers.Handler.MOD_ALT : 0) | (evt.metaKey ? OpenLayers.Handler.MOD_META : 0); │ │ │ │ │ - return keyModifiers == this.keyMask │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.register(events[i], this[events[i]]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.active = true; │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.unregister(events[i], this[events[i]]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.touch = false; │ │ │ │ │ - this.active = false; │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - startTouch: function() { │ │ │ │ │ - if (!this.touch) { │ │ │ │ │ - this.touch = true; │ │ │ │ │ - var events = ["mousedown", "mouseup", "mousemove", "click", "dblclick", "mouseout"]; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.unregister(events[i], this[events[i]]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - callback: function(name, args) { │ │ │ │ │ - if (name && this.callbacks[name]) { │ │ │ │ │ - this.callbacks[name].apply(this.control, args) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - register: function(name, method) { │ │ │ │ │ - this.map.events.registerPriority(name, this, method); │ │ │ │ │ - this.map.events.registerPriority(name, this, this.setEvent) │ │ │ │ │ - }, │ │ │ │ │ - unregister: function(name, method) { │ │ │ │ │ - this.map.events.unregister(name, this, method); │ │ │ │ │ - this.map.events.unregister(name, this, this.setEvent) │ │ │ │ │ - }, │ │ │ │ │ - setEvent: function(evt) { │ │ │ │ │ - this.evt = evt; │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.control = this.map = null │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.MOD_NONE = 0; │ │ │ │ │ -OpenLayers.Handler.MOD_SHIFT = 1; │ │ │ │ │ -OpenLayers.Handler.MOD_CTRL = 2; │ │ │ │ │ -OpenLayers.Handler.MOD_ALT = 4; │ │ │ │ │ -OpenLayers.Handler.MOD_META = 8; │ │ │ │ │ OpenLayers.Feature = OpenLayers.Class({ │ │ │ │ │ layer: null, │ │ │ │ │ id: null, │ │ │ │ │ lonlat: null, │ │ │ │ │ data: null, │ │ │ │ │ marker: null, │ │ │ │ │ popupClass: null, │ │ │ │ │ @@ -5722,128 +4185,82 @@ │ │ │ │ │ if (typeof value == "string" && value.indexOf("${") != -1) { │ │ │ │ │ value = OpenLayers.String.format(value, context, [feature, property]); │ │ │ │ │ value = isNaN(value) || !value ? value : parseFloat(value) │ │ │ │ │ } │ │ │ │ │ return value │ │ │ │ │ }; │ │ │ │ │ OpenLayers.Style.SYMBOLIZER_PREFIXES = ["Point", "Line", "Polygon", "Text", "Raster"]; │ │ │ │ │ -OpenLayers.Filter = OpenLayers.Class({ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() {}, │ │ │ │ │ - evaluate: function(context) { │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - return null │ │ │ │ │ - }, │ │ │ │ │ - toString: function() { │ │ │ │ │ - var string; │ │ │ │ │ - if (OpenLayers.Format && OpenLayers.Format.CQL) { │ │ │ │ │ - string = OpenLayers.Format.CQL.prototype.write(this) │ │ │ │ │ - } else { │ │ │ │ │ - string = Object.prototype.toString.call(this) │ │ │ │ │ - } │ │ │ │ │ - return string │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Filter" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format = OpenLayers.Class({ │ │ │ │ │ - options: null, │ │ │ │ │ - externalProjection: null, │ │ │ │ │ - internalProjection: null, │ │ │ │ │ - data: null, │ │ │ │ │ - keepData: false, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.options = options │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() {}, │ │ │ │ │ - read: function(data) { │ │ │ │ │ - throw new Error("Read not implemented.") │ │ │ │ │ - }, │ │ │ │ │ - write: function(object) { │ │ │ │ │ - throw new Error("Write not implemented.") │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.HTTPRequest = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ - URL_HASH_FACTOR: (Math.sqrt(5) - 1) / 2, │ │ │ │ │ - url: null, │ │ │ │ │ - params: null, │ │ │ │ │ - reproject: false, │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ - this.url = url; │ │ │ │ │ - if (!this.params) { │ │ │ │ │ - this.params = OpenLayers.Util.extend({}, params) │ │ │ │ │ +OpenLayers.StyleMap = OpenLayers.Class({ │ │ │ │ │ + styles: null, │ │ │ │ │ + extendDefault: true, │ │ │ │ │ + initialize: function(style, options) { │ │ │ │ │ + this.styles = { │ │ │ │ │ + default: new OpenLayers.Style(OpenLayers.Feature.Vector.style["default"]), │ │ │ │ │ + select: new OpenLayers.Style(OpenLayers.Feature.Vector.style["select"]), │ │ │ │ │ + temporary: new OpenLayers.Style(OpenLayers.Feature.Vector.style["temporary"]), │ │ │ │ │ + delete: new OpenLayers.Style(OpenLayers.Feature.Vector.style["delete"]) │ │ │ │ │ + }; │ │ │ │ │ + if (style instanceof OpenLayers.Style) { │ │ │ │ │ + this.styles["default"] = style; │ │ │ │ │ + this.styles["select"] = style; │ │ │ │ │ + this.styles["temporary"] = style; │ │ │ │ │ + this.styles["delete"] = style │ │ │ │ │ + } else if (typeof style == "object") { │ │ │ │ │ + for (var key in style) { │ │ │ │ │ + if (style[key] instanceof OpenLayers.Style) { │ │ │ │ │ + this.styles[key] = style[key] │ │ │ │ │ + } else if (typeof style[key] == "object") { │ │ │ │ │ + this.styles[key] = new OpenLayers.Style(style[key]) │ │ │ │ │ + } else { │ │ │ │ │ + this.styles["default"] = new OpenLayers.Style(style); │ │ │ │ │ + this.styles["select"] = new OpenLayers.Style(style); │ │ │ │ │ + this.styles["temporary"] = new OpenLayers.Style(style); │ │ │ │ │ + this.styles["delete"] = new OpenLayers.Style(style); │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Util.extend(this, options) │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - this.url = null; │ │ │ │ │ - this.params = null; │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.HTTPRequest(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ + for (var key in this.styles) { │ │ │ │ │ + this.styles[key].destroy() │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - setUrl: function(newUrl) { │ │ │ │ │ - this.url = newUrl │ │ │ │ │ + this.styles = null │ │ │ │ │ }, │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - this.params = OpenLayers.Util.extend(this.params, newParams); │ │ │ │ │ - var ret = this.redraw(); │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "params" │ │ │ │ │ - }) │ │ │ │ │ + createSymbolizer: function(feature, intent) { │ │ │ │ │ + if (!feature) { │ │ │ │ │ + feature = new OpenLayers.Feature.Vector │ │ │ │ │ } │ │ │ │ │ - return ret │ │ │ │ │ - }, │ │ │ │ │ - redraw: function(force) { │ │ │ │ │ - if (force) { │ │ │ │ │ - return this.mergeNewParams({ │ │ │ │ │ - _olSalt: Math.random() │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - return OpenLayers.Layer.prototype.redraw.apply(this, []) │ │ │ │ │ + if (!this.styles[intent]) { │ │ │ │ │ + intent = "default" │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - selectUrl: function(paramString, urls) { │ │ │ │ │ - var product = 1; │ │ │ │ │ - for (var i = 0, len = paramString.length; i < len; i++) { │ │ │ │ │ - product *= paramString.charCodeAt(i) * this.URL_HASH_FACTOR; │ │ │ │ │ - product -= Math.floor(product) │ │ │ │ │ + feature.renderIntent = intent; │ │ │ │ │ + var defaultSymbolizer = {}; │ │ │ │ │ + if (this.extendDefault && intent != "default") { │ │ │ │ │ + defaultSymbolizer = this.styles["default"].createSymbolizer(feature) │ │ │ │ │ } │ │ │ │ │ - return urls[Math.floor(product * urls.length)] │ │ │ │ │ + return OpenLayers.Util.extend(defaultSymbolizer, this.styles[intent].createSymbolizer(feature)) │ │ │ │ │ }, │ │ │ │ │ - getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ - var url = altUrl || this.url; │ │ │ │ │ - var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ - allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ - var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(paramsString, url) │ │ │ │ │ - } │ │ │ │ │ - var urlParams = OpenLayers.Util.upperCaseObject(OpenLayers.Util.getParameters(url)); │ │ │ │ │ - for (var key in allParams) { │ │ │ │ │ - if (key.toUpperCase() in urlParams) { │ │ │ │ │ - delete allParams[key] │ │ │ │ │ - } │ │ │ │ │ + addUniqueValueRules: function(renderIntent, property, symbolizers, context) { │ │ │ │ │ + var rules = []; │ │ │ │ │ + for (var value in symbolizers) { │ │ │ │ │ + rules.push(new OpenLayers.Rule({ │ │ │ │ │ + symbolizer: symbolizers[value], │ │ │ │ │ + context: context, │ │ │ │ │ + filter: new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.EQUAL_TO, │ │ │ │ │ + property: property, │ │ │ │ │ + value: value │ │ │ │ │ + }) │ │ │ │ │ + })) │ │ │ │ │ } │ │ │ │ │ - paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ - return OpenLayers.Util.urlAppend(url, paramsString) │ │ │ │ │ + this.styles[renderIntent].addRules(rules) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.HTTPRequest" │ │ │ │ │ + CLASS_NAME: "OpenLayers.StyleMap" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Tile = OpenLayers.Class({ │ │ │ │ │ events: null, │ │ │ │ │ eventListeners: null, │ │ │ │ │ id: null, │ │ │ │ │ layer: null, │ │ │ │ │ url: null, │ │ │ │ │ @@ -5930,1371 +4347,493 @@ │ │ │ │ │ if (redraw) { │ │ │ │ │ this.draw() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ clear: function(draw) {}, │ │ │ │ │ CLASS_NAME: "OpenLayers.Tile" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Tile.Image = OpenLayers.Class(OpenLayers.Tile, { │ │ │ │ │ - url: null, │ │ │ │ │ - imgDiv: null, │ │ │ │ │ - frame: null, │ │ │ │ │ - imageReloadAttempts: null, │ │ │ │ │ - layerAlphaHack: null, │ │ │ │ │ - asyncRequestId: null, │ │ │ │ │ - maxGetUrlLength: null, │ │ │ │ │ - canvasContext: null, │ │ │ │ │ - crossOriginKeyword: null, │ │ │ │ │ - initialize: function(layer, position, bounds, url, size, options) { │ │ │ │ │ - OpenLayers.Tile.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.url = url; │ │ │ │ │ - this.layerAlphaHack = this.layer.alpha && OpenLayers.Util.alphaHack(); │ │ │ │ │ - if (this.maxGetUrlLength != null || this.layer.gutter || this.layerAlphaHack) { │ │ │ │ │ - this.frame = document.createElement("div"); │ │ │ │ │ - this.frame.style.position = "absolute"; │ │ │ │ │ - this.frame.style.overflow = "hidden" │ │ │ │ │ - } │ │ │ │ │ - if (this.maxGetUrlLength != null) { │ │ │ │ │ - OpenLayers.Util.extend(this, OpenLayers.Tile.Image.IFrame) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.imgDiv) { │ │ │ │ │ - this.clear(); │ │ │ │ │ - this.imgDiv = null; │ │ │ │ │ - this.frame = null │ │ │ │ │ - } │ │ │ │ │ - this.asyncRequestId = null; │ │ │ │ │ - OpenLayers.Tile.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - var shouldDraw = OpenLayers.Tile.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (shouldDraw) { │ │ │ │ │ - if (this.layer != this.layer.map.baseLayer && this.layer.reproject) { │ │ │ │ │ - this.bounds = this.getBoundsFromBaseLayer(this.position) │ │ │ │ │ - } │ │ │ │ │ - if (this.isLoading) { │ │ │ │ │ - this._loadEvent = "reload" │ │ │ │ │ - } else { │ │ │ │ │ - this.isLoading = true; │ │ │ │ │ - this._loadEvent = "loadstart" │ │ │ │ │ - } │ │ │ │ │ - this.renderTile(); │ │ │ │ │ - this.positionTile() │ │ │ │ │ - } else if (shouldDraw === false) { │ │ │ │ │ - this.unload() │ │ │ │ │ - } │ │ │ │ │ - return shouldDraw │ │ │ │ │ - }, │ │ │ │ │ - renderTile: function() { │ │ │ │ │ - if (this.layer.async) { │ │ │ │ │ - var id = this.asyncRequestId = (this.asyncRequestId || 0) + 1; │ │ │ │ │ - this.layer.getURLasync(this.bounds, function(url) { │ │ │ │ │ - if (id == this.asyncRequestId) { │ │ │ │ │ - this.url = url; │ │ │ │ │ - this.initImage() │ │ │ │ │ - } │ │ │ │ │ - }, this) │ │ │ │ │ - } else { │ │ │ │ │ - this.url = this.layer.getURL(this.bounds); │ │ │ │ │ - this.initImage() │ │ │ │ │ - } │ │ │ │ │ +OpenLayers.Format = OpenLayers.Class({ │ │ │ │ │ + options: null, │ │ │ │ │ + externalProjection: null, │ │ │ │ │ + internalProjection: null, │ │ │ │ │ + data: null, │ │ │ │ │ + keepData: false, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.options = options │ │ │ │ │ }, │ │ │ │ │ - positionTile: function() { │ │ │ │ │ - var style = this.getTile().style, │ │ │ │ │ - size = this.frame ? this.size : this.layer.getImageSize(this.bounds), │ │ │ │ │ - ratio = 1; │ │ │ │ │ - if (this.layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ - ratio = this.layer.getServerResolution() / this.layer.map.getResolution() │ │ │ │ │ - } │ │ │ │ │ - style.left = this.position.x + "px"; │ │ │ │ │ - style.top = this.position.y + "px"; │ │ │ │ │ - style.width = Math.round(ratio * size.w) + "px"; │ │ │ │ │ - style.height = Math.round(ratio * size.h) + "px" │ │ │ │ │ + destroy: function() {}, │ │ │ │ │ + read: function(data) { │ │ │ │ │ + throw new Error("Read not implemented.") │ │ │ │ │ }, │ │ │ │ │ - clear: function() { │ │ │ │ │ - OpenLayers.Tile.prototype.clear.apply(this, arguments); │ │ │ │ │ - var img = this.imgDiv; │ │ │ │ │ - if (img) { │ │ │ │ │ - var tile = this.getTile(); │ │ │ │ │ - if (tile.parentNode === this.layer.div) { │ │ │ │ │ - this.layer.div.removeChild(tile) │ │ │ │ │ - } │ │ │ │ │ - this.setImgSrc(); │ │ │ │ │ - if (this.layerAlphaHack === true) { │ │ │ │ │ - img.style.filter = "" │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Element.removeClass(img, "olImageLoadError") │ │ │ │ │ - } │ │ │ │ │ - this.canvasContext = null │ │ │ │ │ + write: function(object) { │ │ │ │ │ + throw new Error("Write not implemented.") │ │ │ │ │ }, │ │ │ │ │ - getImage: function() { │ │ │ │ │ - if (!this.imgDiv) { │ │ │ │ │ - this.imgDiv = OpenLayers.Tile.Image.IMAGE.cloneNode(false); │ │ │ │ │ - var style = this.imgDiv.style; │ │ │ │ │ - if (this.frame) { │ │ │ │ │ - var left = 0, │ │ │ │ │ - top = 0; │ │ │ │ │ - if (this.layer.gutter) { │ │ │ │ │ - left = this.layer.gutter / this.layer.tileSize.w * 100; │ │ │ │ │ - top = this.layer.gutter / this.layer.tileSize.h * 100 │ │ │ │ │ - } │ │ │ │ │ - style.left = -left + "%"; │ │ │ │ │ - style.top = -top + "%"; │ │ │ │ │ - style.width = 2 * left + 100 + "%"; │ │ │ │ │ - style.height = 2 * top + 100 + "%" │ │ │ │ │ - } │ │ │ │ │ - style.visibility = "hidden"; │ │ │ │ │ - style.opacity = 0; │ │ │ │ │ - if (this.layer.opacity < 1) { │ │ │ │ │ - style.filter = "alpha(opacity=" + this.layer.opacity * 100 + ")" │ │ │ │ │ - } │ │ │ │ │ - style.position = "absolute"; │ │ │ │ │ - if (this.layerAlphaHack) { │ │ │ │ │ - style.paddingTop = style.height; │ │ │ │ │ - style.height = "0"; │ │ │ │ │ - style.width = "100%" │ │ │ │ │ - } │ │ │ │ │ - if (this.frame) { │ │ │ │ │ - this.frame.appendChild(this.imgDiv) │ │ │ │ │ - } │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Popup = OpenLayers.Class({ │ │ │ │ │ + events: null, │ │ │ │ │ + id: "", │ │ │ │ │ + lonlat: null, │ │ │ │ │ + div: null, │ │ │ │ │ + contentSize: null, │ │ │ │ │ + size: null, │ │ │ │ │ + contentHTML: null, │ │ │ │ │ + backgroundColor: "", │ │ │ │ │ + opacity: "", │ │ │ │ │ + border: "", │ │ │ │ │ + contentDiv: null, │ │ │ │ │ + groupDiv: null, │ │ │ │ │ + closeDiv: null, │ │ │ │ │ + autoSize: false, │ │ │ │ │ + minSize: null, │ │ │ │ │ + maxSize: null, │ │ │ │ │ + displayClass: "olPopup", │ │ │ │ │ + contentDisplayClass: "olPopupContent", │ │ │ │ │ + padding: 0, │ │ │ │ │ + disableFirefoxOverflowHack: false, │ │ │ │ │ + fixPadding: function() { │ │ │ │ │ + if (typeof this.padding == "number") { │ │ │ │ │ + this.padding = new OpenLayers.Bounds(this.padding, this.padding, this.padding, this.padding) │ │ │ │ │ } │ │ │ │ │ - return this.imgDiv │ │ │ │ │ - }, │ │ │ │ │ - setImage: function(img) { │ │ │ │ │ - this.imgDiv = img │ │ │ │ │ }, │ │ │ │ │ - initImage: function() { │ │ │ │ │ - if (!this.url && !this.imgDiv) { │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - return │ │ │ │ │ + panMapIfOutOfView: false, │ │ │ │ │ + keepInMap: false, │ │ │ │ │ + closeOnMove: false, │ │ │ │ │ + map: null, │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, closeBox, closeBoxCallback) { │ │ │ │ │ + if (id == null) { │ │ │ │ │ + id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ } │ │ │ │ │ - this.events.triggerEvent("beforeload"); │ │ │ │ │ - this.layer.div.appendChild(this.getTile()); │ │ │ │ │ - this.events.triggerEvent(this._loadEvent); │ │ │ │ │ - var img = this.getImage(); │ │ │ │ │ - var src = img.getAttribute("src") || ""; │ │ │ │ │ - if (this.url && OpenLayers.Util.isEquivalentUrl(src, this.url)) { │ │ │ │ │ - this._loadTimeout = window.setTimeout(OpenLayers.Function.bind(this.onImageLoad, this), 0) │ │ │ │ │ - } else { │ │ │ │ │ - this.stopLoading(); │ │ │ │ │ - if (this.crossOriginKeyword) { │ │ │ │ │ - img.removeAttribute("crossorigin") │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.observe(img, "load", OpenLayers.Function.bind(this.onImageLoad, this)); │ │ │ │ │ - OpenLayers.Event.observe(img, "error", OpenLayers.Function.bind(this.onImageError, this)); │ │ │ │ │ - this.imageReloadAttempts = 0; │ │ │ │ │ - this.setImgSrc(this.url) │ │ │ │ │ + this.id = id; │ │ │ │ │ + this.lonlat = lonlat; │ │ │ │ │ + this.contentSize = contentSize != null ? contentSize : new OpenLayers.Size(OpenLayers.Popup.WIDTH, OpenLayers.Popup.HEIGHT); │ │ │ │ │ + if (contentHTML != null) { │ │ │ │ │ + this.contentHTML = contentHTML │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - setImgSrc: function(url) { │ │ │ │ │ - var img = this.imgDiv; │ │ │ │ │ - if (url) { │ │ │ │ │ - img.style.visibility = "hidden"; │ │ │ │ │ - img.style.opacity = 0; │ │ │ │ │ - if (this.crossOriginKeyword) { │ │ │ │ │ - if (url.substr(0, 5) !== "data:") { │ │ │ │ │ - img.setAttribute("crossorigin", this.crossOriginKeyword) │ │ │ │ │ - } else { │ │ │ │ │ - img.removeAttribute("crossorigin") │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - img.src = url │ │ │ │ │ - } else { │ │ │ │ │ - this.stopLoading(); │ │ │ │ │ - this.imgDiv = null; │ │ │ │ │ - if (img.parentNode) { │ │ │ │ │ - img.parentNode.removeChild(img) │ │ │ │ │ - } │ │ │ │ │ + this.backgroundColor = OpenLayers.Popup.COLOR; │ │ │ │ │ + this.opacity = OpenLayers.Popup.OPACITY; │ │ │ │ │ + this.border = OpenLayers.Popup.BORDER; │ │ │ │ │ + this.div = OpenLayers.Util.createDiv(this.id, null, null, null, null, null, "hidden"); │ │ │ │ │ + this.div.className = this.displayClass; │ │ │ │ │ + var groupDivId = this.id + "_GroupDiv"; │ │ │ │ │ + this.groupDiv = OpenLayers.Util.createDiv(groupDivId, null, null, null, "relative", null, "hidden"); │ │ │ │ │ + var id = this.div.id + "_contentDiv"; │ │ │ │ │ + this.contentDiv = OpenLayers.Util.createDiv(id, null, this.contentSize.clone(), null, "relative"); │ │ │ │ │ + this.contentDiv.className = this.contentDisplayClass; │ │ │ │ │ + this.groupDiv.appendChild(this.contentDiv); │ │ │ │ │ + this.div.appendChild(this.groupDiv); │ │ │ │ │ + if (closeBox) { │ │ │ │ │ + this.addCloseBox(closeBoxCallback) │ │ │ │ │ } │ │ │ │ │ + this.registerEvents() │ │ │ │ │ }, │ │ │ │ │ - getTile: function() { │ │ │ │ │ - return this.frame ? this.frame : this.getImage() │ │ │ │ │ - }, │ │ │ │ │ - createBackBuffer: function() { │ │ │ │ │ - if (!this.imgDiv || this.isLoading) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var backBuffer; │ │ │ │ │ - if (this.frame) { │ │ │ │ │ - backBuffer = this.frame.cloneNode(false); │ │ │ │ │ - backBuffer.appendChild(this.imgDiv) │ │ │ │ │ - } else { │ │ │ │ │ - backBuffer = this.imgDiv │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.id = null; │ │ │ │ │ + this.lonlat = null; │ │ │ │ │ + this.size = null; │ │ │ │ │ + this.contentHTML = null; │ │ │ │ │ + this.backgroundColor = null; │ │ │ │ │ + this.opacity = null; │ │ │ │ │ + this.border = null; │ │ │ │ │ + if (this.closeOnMove && this.map) { │ │ │ │ │ + this.map.events.unregister("movestart", this, this.hide) │ │ │ │ │ } │ │ │ │ │ - this.imgDiv = null; │ │ │ │ │ - return backBuffer │ │ │ │ │ - }, │ │ │ │ │ - onImageLoad: function() { │ │ │ │ │ - var img = this.imgDiv; │ │ │ │ │ - this.stopLoading(); │ │ │ │ │ - img.style.visibility = "inherit"; │ │ │ │ │ - img.style.opacity = this.layer.opacity; │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - this.canvasContext = null; │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ - if (this.layerAlphaHack === true) { │ │ │ │ │ - img.style.filter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(src='" + img.src + "', sizingMethod='scale')" │ │ │ │ │ + this.events.destroy(); │ │ │ │ │ + this.events = null; │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + OpenLayers.Event.stopObservingElement(this.closeDiv); │ │ │ │ │ + this.groupDiv.removeChild(this.closeDiv) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - onImageError: function() { │ │ │ │ │ - var img = this.imgDiv; │ │ │ │ │ - if (img.src != null) { │ │ │ │ │ - this.imageReloadAttempts++; │ │ │ │ │ - if (this.imageReloadAttempts <= OpenLayers.IMAGE_RELOAD_ATTEMPTS) { │ │ │ │ │ - this.setImgSrc(this.layer.getURL(this.bounds)) │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Element.addClass(img, "olImageLoadError"); │ │ │ │ │ - this.events.triggerEvent("loaderror"); │ │ │ │ │ - this.onImageLoad() │ │ │ │ │ - } │ │ │ │ │ + this.closeDiv = null; │ │ │ │ │ + this.div.removeChild(this.groupDiv); │ │ │ │ │ + this.groupDiv = null; │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.removePopup(this) │ │ │ │ │ } │ │ │ │ │ + this.map = null; │ │ │ │ │ + this.div = null; │ │ │ │ │ + this.autoSize = null; │ │ │ │ │ + this.minSize = null; │ │ │ │ │ + this.maxSize = null; │ │ │ │ │ + this.padding = null; │ │ │ │ │ + this.panMapIfOutOfView = null │ │ │ │ │ }, │ │ │ │ │ - stopLoading: function() { │ │ │ │ │ - OpenLayers.Event.stopObservingElement(this.imgDiv); │ │ │ │ │ - window.clearTimeout(this._loadTimeout); │ │ │ │ │ - delete this._loadTimeout │ │ │ │ │ - }, │ │ │ │ │ - getCanvasContext: function() { │ │ │ │ │ - if (OpenLayers.CANVAS_SUPPORTED && this.imgDiv && !this.isLoading) { │ │ │ │ │ - if (!this.canvasContext) { │ │ │ │ │ - var canvas = document.createElement("canvas"); │ │ │ │ │ - canvas.width = this.size.w; │ │ │ │ │ - canvas.height = this.size.h; │ │ │ │ │ - this.canvasContext = canvas.getContext("2d"); │ │ │ │ │ - this.canvasContext.drawImage(this.imgDiv, 0, 0) │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + if (px == null) { │ │ │ │ │ + if (this.lonlat != null && this.map != null) { │ │ │ │ │ + px = this.map.getLayerPxFromLonLat(this.lonlat) │ │ │ │ │ } │ │ │ │ │ - return this.canvasContext │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Tile.Image" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Tile.Image.IMAGE = function() { │ │ │ │ │ - var img = new Image; │ │ │ │ │ - img.className = "olTileImage"; │ │ │ │ │ - img.galleryImg = "no"; │ │ │ │ │ - return img │ │ │ │ │ -}(); │ │ │ │ │ -OpenLayers.Layer.Grid = OpenLayers.Class(OpenLayers.Layer.HTTPRequest, { │ │ │ │ │ - tileSize: null, │ │ │ │ │ - tileOriginCorner: "bl", │ │ │ │ │ - tileOrigin: null, │ │ │ │ │ - tileOptions: null, │ │ │ │ │ - tileClass: OpenLayers.Tile.Image, │ │ │ │ │ - grid: null, │ │ │ │ │ - singleTile: false, │ │ │ │ │ - ratio: 1.5, │ │ │ │ │ - buffer: 0, │ │ │ │ │ - transitionEffect: "resize", │ │ │ │ │ - numLoadingTiles: 0, │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ - loading: false, │ │ │ │ │ - backBuffer: null, │ │ │ │ │ - gridResolution: null, │ │ │ │ │ - backBufferResolution: null, │ │ │ │ │ - backBufferLonLat: null, │ │ │ │ │ - backBufferTimerId: null, │ │ │ │ │ - removeBackBufferDelay: null, │ │ │ │ │ - className: null, │ │ │ │ │ - gridLayout: null, │ │ │ │ │ - rowSign: null, │ │ │ │ │ - transitionendEvents: ["transitionend", "webkitTransitionEnd", "otransitionend", "oTransitionEnd"], │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.grid = []; │ │ │ │ │ - this._removeBackBuffer = OpenLayers.Function.bind(this.removeBackBuffer, this); │ │ │ │ │ - this.initProperties(); │ │ │ │ │ - this.rowSign = this.tileOriginCorner.substr(0, 1) === "t" ? 1 : -1 │ │ │ │ │ - }, │ │ │ │ │ - initProperties: function() { │ │ │ │ │ - if (this.options.removeBackBufferDelay === undefined) { │ │ │ │ │ - this.removeBackBufferDelay = this.singleTile ? 0 : 2500 │ │ │ │ │ } │ │ │ │ │ - if (this.options.className === undefined) { │ │ │ │ │ - this.className = this.singleTile ? "olLayerGridSingleTile" : "olLayerGrid" │ │ │ │ │ + if (this.closeOnMove) { │ │ │ │ │ + this.map.events.register("movestart", this, this.hide) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.setMap.call(this, map); │ │ │ │ │ - OpenLayers.Element.addClass(this.div, this.className) │ │ │ │ │ - }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - this.removeBackBuffer() │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.removeBackBuffer(); │ │ │ │ │ - this.clearGrid(); │ │ │ │ │ - this.grid = null; │ │ │ │ │ - this.tileSize = null; │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - clearGrid: function() { │ │ │ │ │ - if (this.grid) { │ │ │ │ │ - for (var iRow = 0, len = this.grid.length; iRow < len; iRow++) { │ │ │ │ │ - var row = this.grid[iRow]; │ │ │ │ │ - for (var iCol = 0, clen = row.length; iCol < clen; iCol++) { │ │ │ │ │ - var tile = row[iCol]; │ │ │ │ │ - this.destroyTile(tile) │ │ │ │ │ + if (!this.disableFirefoxOverflowHack && OpenLayers.BROWSER_NAME == "firefox") { │ │ │ │ │ + this.map.events.register("movestart", this, function() { │ │ │ │ │ + var style = document.defaultView.getComputedStyle(this.contentDiv, null); │ │ │ │ │ + var currentOverflow = style.getPropertyValue("overflow"); │ │ │ │ │ + if (currentOverflow != "hidden") { │ │ │ │ │ + this.contentDiv._oldOverflow = currentOverflow; │ │ │ │ │ + this.contentDiv.style.overflow = "hidden" │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - this.grid = []; │ │ │ │ │ - this.gridResolution = null; │ │ │ │ │ - this.gridLayout = null │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - addOptions: function(newOptions, reinitialize) { │ │ │ │ │ - var singleTileChanged = newOptions.singleTile !== undefined && newOptions.singleTile !== this.singleTile; │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.addOptions.apply(this, arguments); │ │ │ │ │ - if (this.map && singleTileChanged) { │ │ │ │ │ - this.initProperties(); │ │ │ │ │ - this.clearGrid(); │ │ │ │ │ - this.tileSize = this.options.tileSize; │ │ │ │ │ - this.setTileSize(); │ │ │ │ │ - this.moveTo(null, true) │ │ │ │ │ + }); │ │ │ │ │ + this.map.events.register("moveend", this, function() { │ │ │ │ │ + var oldOverflow = this.contentDiv._oldOverflow; │ │ │ │ │ + if (oldOverflow) { │ │ │ │ │ + this.contentDiv.style.overflow = oldOverflow; │ │ │ │ │ + this.contentDiv._oldOverflow = null │ │ │ │ │ + } │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Grid(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ + this.moveTo(px); │ │ │ │ │ + if (!this.autoSize && !this.size) { │ │ │ │ │ + this.setSize(this.contentSize) │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.HTTPRequest.prototype.clone.apply(this, [obj]); │ │ │ │ │ - if (this.tileSize != null) { │ │ │ │ │ - obj.tileSize = this.tileSize.clone() │ │ │ │ │ + this.setBackgroundColor(); │ │ │ │ │ + this.setOpacity(); │ │ │ │ │ + this.setBorder(); │ │ │ │ │ + this.setContentHTML(); │ │ │ │ │ + if (this.panMapIfOutOfView) { │ │ │ │ │ + this.panIntoView() │ │ │ │ │ } │ │ │ │ │ - obj.grid = []; │ │ │ │ │ - obj.gridResolution = null; │ │ │ │ │ - obj.backBuffer = null; │ │ │ │ │ - obj.backBufferTimerId = null; │ │ │ │ │ - obj.loading = false; │ │ │ │ │ - obj.numLoadingTiles = 0; │ │ │ │ │ - return obj │ │ │ │ │ + return this.div │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - bounds = bounds || this.map.getExtent(); │ │ │ │ │ - if (bounds != null) { │ │ │ │ │ - var forceReTile = !this.grid.length || zoomChanged; │ │ │ │ │ - var tilesBounds = this.getTilesBounds(); │ │ │ │ │ - var resolution = this.map.getResolution(); │ │ │ │ │ - var serverResolution = this.getServerResolution(resolution); │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - if (forceReTile || !dragging && !tilesBounds.containsBounds(bounds)) { │ │ │ │ │ - if (zoomChanged && this.transitionEffect !== "resize") { │ │ │ │ │ - this.removeBackBuffer() │ │ │ │ │ - } │ │ │ │ │ - if (!zoomChanged || this.transitionEffect === "resize") { │ │ │ │ │ - this.applyBackBuffer(resolution) │ │ │ │ │ - } │ │ │ │ │ - this.initSingleTile(bounds) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - forceReTile = forceReTile || !tilesBounds.intersectsBounds(bounds, { │ │ │ │ │ - worldBounds: this.map.baseLayer.wrapDateLine && this.map.getMaxExtent() │ │ │ │ │ - }); │ │ │ │ │ - if (forceReTile) { │ │ │ │ │ - if (zoomChanged && (this.transitionEffect === "resize" || this.gridResolution === resolution)) { │ │ │ │ │ - this.applyBackBuffer(resolution) │ │ │ │ │ - } │ │ │ │ │ - this.initGriddedTiles(bounds) │ │ │ │ │ - } else { │ │ │ │ │ - this.moveGriddedTiles() │ │ │ │ │ - } │ │ │ │ │ + updatePosition: function() { │ │ │ │ │ + if (this.lonlat && this.map) { │ │ │ │ │ + var px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ + if (px) { │ │ │ │ │ + this.moveTo(px) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - getTileData: function(loc) { │ │ │ │ │ - var data = null, │ │ │ │ │ - x = loc.lon, │ │ │ │ │ - y = loc.lat, │ │ │ │ │ - numRows = this.grid.length; │ │ │ │ │ - if (this.map && numRows) { │ │ │ │ │ - var res = this.map.getResolution(), │ │ │ │ │ - tileWidth = this.tileSize.w, │ │ │ │ │ - tileHeight = this.tileSize.h, │ │ │ │ │ - bounds = this.grid[0][0].bounds, │ │ │ │ │ - left = bounds.left, │ │ │ │ │ - top = bounds.top; │ │ │ │ │ - if (x < left) { │ │ │ │ │ - if (this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - var worldWidth = this.map.getMaxExtent().getWidth(); │ │ │ │ │ - var worldsAway = Math.ceil((left - x) / worldWidth); │ │ │ │ │ - x += worldWidth * worldsAway │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var dtx = (x - left) / (res * tileWidth); │ │ │ │ │ - var dty = (top - y) / (res * tileHeight); │ │ │ │ │ - var col = Math.floor(dtx); │ │ │ │ │ - var row = Math.floor(dty); │ │ │ │ │ - if (row >= 0 && row < numRows) { │ │ │ │ │ - var tile = this.grid[row][col]; │ │ │ │ │ - if (tile) { │ │ │ │ │ - data = { │ │ │ │ │ - tile: tile, │ │ │ │ │ - i: Math.floor((dtx - col) * tileWidth), │ │ │ │ │ - j: Math.floor((dty - row) * tileHeight) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + if (px != null && this.div != null) { │ │ │ │ │ + this.div.style.left = px.x + "px"; │ │ │ │ │ + this.div.style.top = px.y + "px" │ │ │ │ │ } │ │ │ │ │ - return data │ │ │ │ │ }, │ │ │ │ │ - destroyTile: function(tile) { │ │ │ │ │ - this.removeTileMonitoringHooks(tile); │ │ │ │ │ - tile.destroy() │ │ │ │ │ + visible: function() { │ │ │ │ │ + return OpenLayers.Element.visible(this.div) │ │ │ │ │ }, │ │ │ │ │ - getServerResolution: function(resolution) { │ │ │ │ │ - var distance = Number.POSITIVE_INFINITY; │ │ │ │ │ - resolution = resolution || this.map.getResolution(); │ │ │ │ │ - if (this.serverResolutions && OpenLayers.Util.indexOf(this.serverResolutions, resolution) === -1) { │ │ │ │ │ - var i, newDistance, newResolution, serverResolution; │ │ │ │ │ - for (i = this.serverResolutions.length - 1; i >= 0; i--) { │ │ │ │ │ - newResolution = this.serverResolutions[i]; │ │ │ │ │ - newDistance = Math.abs(newResolution - resolution); │ │ │ │ │ - if (newDistance > distance) { │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - distance = newDistance; │ │ │ │ │ - serverResolution = newResolution │ │ │ │ │ - } │ │ │ │ │ - resolution = serverResolution │ │ │ │ │ + toggle: function() { │ │ │ │ │ + if (this.visible()) { │ │ │ │ │ + this.hide() │ │ │ │ │ + } else { │ │ │ │ │ + this.show() │ │ │ │ │ } │ │ │ │ │ - return resolution │ │ │ │ │ - }, │ │ │ │ │ - getServerZoom: function() { │ │ │ │ │ - var resolution = this.getServerResolution(); │ │ │ │ │ - return this.serverResolutions ? OpenLayers.Util.indexOf(this.serverResolutions, resolution) : this.map.getZoomForResolution(resolution) + (this.zoomOffset || 0) │ │ │ │ │ }, │ │ │ │ │ - applyBackBuffer: function(resolution) { │ │ │ │ │ - if (this.backBufferTimerId !== null) { │ │ │ │ │ - this.removeBackBuffer() │ │ │ │ │ - } │ │ │ │ │ - var backBuffer = this.backBuffer; │ │ │ │ │ - if (!backBuffer) { │ │ │ │ │ - backBuffer = this.createBackBuffer(); │ │ │ │ │ - if (!backBuffer) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (resolution === this.gridResolution) { │ │ │ │ │ - this.div.insertBefore(backBuffer, this.div.firstChild) │ │ │ │ │ - } else { │ │ │ │ │ - this.map.baseLayer.div.parentNode.insertBefore(backBuffer, this.map.baseLayer.div) │ │ │ │ │ - } │ │ │ │ │ - this.backBuffer = backBuffer; │ │ │ │ │ - var topLeftTileBounds = this.grid[0][0].bounds; │ │ │ │ │ - this.backBufferLonLat = { │ │ │ │ │ - lon: topLeftTileBounds.left, │ │ │ │ │ - lat: topLeftTileBounds.top │ │ │ │ │ - }; │ │ │ │ │ - this.backBufferResolution = this.gridResolution │ │ │ │ │ - } │ │ │ │ │ - var ratio = this.backBufferResolution / resolution; │ │ │ │ │ - var tiles = backBuffer.childNodes, │ │ │ │ │ - tile; │ │ │ │ │ - for (var i = tiles.length - 1; i >= 0; --i) { │ │ │ │ │ - tile = tiles[i]; │ │ │ │ │ - tile.style.top = (ratio * tile._i * tile._h | 0) + "px"; │ │ │ │ │ - tile.style.left = (ratio * tile._j * tile._w | 0) + "px"; │ │ │ │ │ - tile.style.width = Math.round(ratio * tile._w) + "px"; │ │ │ │ │ - tile.style.height = Math.round(ratio * tile._h) + "px" │ │ │ │ │ + show: function() { │ │ │ │ │ + this.div.style.display = ""; │ │ │ │ │ + if (this.panMapIfOutOfView) { │ │ │ │ │ + this.panIntoView() │ │ │ │ │ } │ │ │ │ │ - var position = this.getViewPortPxFromLonLat(this.backBufferLonLat, resolution); │ │ │ │ │ - var leftOffset = this.map.layerContainerOriginPx.x; │ │ │ │ │ - var topOffset = this.map.layerContainerOriginPx.y; │ │ │ │ │ - backBuffer.style.left = Math.round(position.x - leftOffset) + "px"; │ │ │ │ │ - backBuffer.style.top = Math.round(position.y - topOffset) + "px" │ │ │ │ │ }, │ │ │ │ │ - createBackBuffer: function() { │ │ │ │ │ - var backBuffer; │ │ │ │ │ - if (this.grid.length > 0) { │ │ │ │ │ - backBuffer = document.createElement("div"); │ │ │ │ │ - backBuffer.id = this.div.id + "_bb"; │ │ │ │ │ - backBuffer.className = "olBackBuffer"; │ │ │ │ │ - backBuffer.style.position = "absolute"; │ │ │ │ │ - var map = this.map; │ │ │ │ │ - backBuffer.style.zIndex = this.transitionEffect === "resize" ? this.getZIndex() - 1 : map.Z_INDEX_BASE.BaseLayer - (map.getNumLayers() - map.getLayerIndex(this)); │ │ │ │ │ - for (var i = 0, lenI = this.grid.length; i < lenI; i++) { │ │ │ │ │ - for (var j = 0, lenJ = this.grid[i].length; j < lenJ; j++) { │ │ │ │ │ - var tile = this.grid[i][j], │ │ │ │ │ - markup = this.grid[i][j].createBackBuffer(); │ │ │ │ │ - if (markup) { │ │ │ │ │ - markup._i = i; │ │ │ │ │ - markup._j = j; │ │ │ │ │ - markup._w = tile.size.w; │ │ │ │ │ - markup._h = tile.size.h; │ │ │ │ │ - markup.id = tile.id + "_bb"; │ │ │ │ │ - backBuffer.appendChild(markup) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return backBuffer │ │ │ │ │ + hide: function() { │ │ │ │ │ + this.div.style.display = "none" │ │ │ │ │ }, │ │ │ │ │ - removeBackBuffer: function() { │ │ │ │ │ - if (this._transitionElement) { │ │ │ │ │ - for (var i = this.transitionendEvents.length - 1; i >= 0; --i) { │ │ │ │ │ - OpenLayers.Event.stopObserving(this._transitionElement, this.transitionendEvents[i], this._removeBackBuffer) │ │ │ │ │ - } │ │ │ │ │ - delete this._transitionElement │ │ │ │ │ - } │ │ │ │ │ - if (this.backBuffer) { │ │ │ │ │ - if (this.backBuffer.parentNode) { │ │ │ │ │ - this.backBuffer.parentNode.removeChild(this.backBuffer) │ │ │ │ │ - } │ │ │ │ │ - this.backBuffer = null; │ │ │ │ │ - this.backBufferResolution = null; │ │ │ │ │ - if (this.backBufferTimerId !== null) { │ │ │ │ │ - window.clearTimeout(this.backBufferTimerId); │ │ │ │ │ - this.backBufferTimerId = null │ │ │ │ │ - } │ │ │ │ │ + setSize: function(contentSize) { │ │ │ │ │ + this.size = contentSize.clone(); │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ + var wPadding = contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ + var hPadding = contentDivPadding.top + contentDivPadding.bottom; │ │ │ │ │ + this.fixPadding(); │ │ │ │ │ + wPadding += this.padding.left + this.padding.right; │ │ │ │ │ + hPadding += this.padding.top + this.padding.bottom; │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + var closeDivWidth = parseInt(this.closeDiv.style.width); │ │ │ │ │ + wPadding += closeDivWidth + contentDivPadding.right │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - moveByPx: function(dx, dy) { │ │ │ │ │ - if (!this.singleTile) { │ │ │ │ │ - this.moveGriddedTiles() │ │ │ │ │ + this.size.w += wPadding; │ │ │ │ │ + this.size.h += hPadding; │ │ │ │ │ + if (OpenLayers.BROWSER_NAME == "msie") { │ │ │ │ │ + this.contentSize.w += contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ + this.contentSize.h += contentDivPadding.bottom + contentDivPadding.top │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - setTileSize: function(size) { │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - size.h = parseInt(size.h * this.ratio, 10); │ │ │ │ │ - size.w = parseInt(size.w * this.ratio, 10) │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.div.style.width = this.size.w + "px"; │ │ │ │ │ + this.div.style.height = this.size.h + "px" │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.HTTPRequest.prototype.setTileSize.apply(this, [size]) │ │ │ │ │ - }, │ │ │ │ │ - getTilesBounds: function() { │ │ │ │ │ - var bounds = null; │ │ │ │ │ - var length = this.grid.length; │ │ │ │ │ - if (length) { │ │ │ │ │ - var bottomLeftTileBounds = this.grid[length - 1][0].bounds, │ │ │ │ │ - width = this.grid[0].length * bottomLeftTileBounds.getWidth(), │ │ │ │ │ - height = this.grid.length * bottomLeftTileBounds.getHeight(); │ │ │ │ │ - bounds = new OpenLayers.Bounds(bottomLeftTileBounds.left, bottomLeftTileBounds.bottom, bottomLeftTileBounds.left + width, bottomLeftTileBounds.bottom + height) │ │ │ │ │ + if (this.contentDiv != null) { │ │ │ │ │ + this.contentDiv.style.width = contentSize.w + "px"; │ │ │ │ │ + this.contentDiv.style.height = contentSize.h + "px" │ │ │ │ │ } │ │ │ │ │ - return bounds │ │ │ │ │ }, │ │ │ │ │ - initSingleTile: function(bounds) { │ │ │ │ │ - this.events.triggerEvent("retile"); │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var tileWidth = bounds.getWidth() * this.ratio; │ │ │ │ │ - var tileHeight = bounds.getHeight() * this.ratio; │ │ │ │ │ - var tileBounds = new OpenLayers.Bounds(center.lon - tileWidth / 2, center.lat - tileHeight / 2, center.lon + tileWidth / 2, center.lat + tileHeight / 2); │ │ │ │ │ - var px = this.map.getLayerPxFromLonLat({ │ │ │ │ │ - lon: tileBounds.left, │ │ │ │ │ - lat: tileBounds.top │ │ │ │ │ + updateSize: function() { │ │ │ │ │ + var preparedHTML = "<div class='" + this.contentDisplayClass + "'>" + this.contentDiv.innerHTML + "</div>"; │ │ │ │ │ + var containerElement = this.map ? this.map.div : document.body; │ │ │ │ │ + var realSize = OpenLayers.Util.getRenderedDimensions(preparedHTML, null, { │ │ │ │ │ + displayClass: this.displayClass, │ │ │ │ │ + containerElement: containerElement │ │ │ │ │ }); │ │ │ │ │ - if (!this.grid.length) { │ │ │ │ │ - this.grid[0] = [] │ │ │ │ │ - } │ │ │ │ │ - var tile = this.grid[0][0]; │ │ │ │ │ - if (!tile) { │ │ │ │ │ - tile = this.addTile(tileBounds, px); │ │ │ │ │ - this.addTileMonitoringHooks(tile); │ │ │ │ │ - tile.draw(); │ │ │ │ │ - this.grid[0][0] = tile │ │ │ │ │ + var safeSize = this.getSafeContentSize(realSize); │ │ │ │ │ + var newSize = null; │ │ │ │ │ + if (safeSize.equals(realSize)) { │ │ │ │ │ + newSize = realSize │ │ │ │ │ } else { │ │ │ │ │ - tile.moveTo(tileBounds, px) │ │ │ │ │ - } │ │ │ │ │ - this.removeExcessTiles(1, 1); │ │ │ │ │ - this.gridResolution = this.getServerResolution() │ │ │ │ │ - }, │ │ │ │ │ - calculateGridLayout: function(bounds, origin, resolution) { │ │ │ │ │ - var tilelon = resolution * this.tileSize.w; │ │ │ │ │ - var tilelat = resolution * this.tileSize.h; │ │ │ │ │ - var offsetlon = bounds.left - origin.lon; │ │ │ │ │ - var tilecol = Math.floor(offsetlon / tilelon) - this.buffer; │ │ │ │ │ - var rowSign = this.rowSign; │ │ │ │ │ - var offsetlat = rowSign * (origin.lat - bounds.top + tilelat); │ │ │ │ │ - var tilerow = Math[~rowSign ? "floor" : "ceil"](offsetlat / tilelat) - this.buffer * rowSign; │ │ │ │ │ - return { │ │ │ │ │ - tilelon: tilelon, │ │ │ │ │ - tilelat: tilelat, │ │ │ │ │ - startcol: tilecol, │ │ │ │ │ - startrow: tilerow │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getTileOrigin: function() { │ │ │ │ │ - var origin = this.tileOrigin; │ │ │ │ │ - if (!origin) { │ │ │ │ │ - var extent = this.getMaxExtent(); │ │ │ │ │ - var edges = { │ │ │ │ │ - tl: ["left", "top"], │ │ │ │ │ - tr: ["right", "top"], │ │ │ │ │ - bl: ["left", "bottom"], │ │ │ │ │ - br: ["right", "bottom"] │ │ │ │ │ - } [this.tileOriginCorner]; │ │ │ │ │ - origin = new OpenLayers.LonLat(extent[edges[0]], extent[edges[1]]) │ │ │ │ │ - } │ │ │ │ │ - return origin │ │ │ │ │ - }, │ │ │ │ │ - getTileBoundsForGridIndex: function(row, col) { │ │ │ │ │ - var origin = this.getTileOrigin(); │ │ │ │ │ - var tileLayout = this.gridLayout; │ │ │ │ │ - var tilelon = tileLayout.tilelon; │ │ │ │ │ - var tilelat = tileLayout.tilelat; │ │ │ │ │ - var startcol = tileLayout.startcol; │ │ │ │ │ - var startrow = tileLayout.startrow; │ │ │ │ │ - var rowSign = this.rowSign; │ │ │ │ │ - return new OpenLayers.Bounds(origin.lon + (startcol + col) * tilelon, origin.lat - (startrow + row * rowSign) * tilelat * rowSign, origin.lon + (startcol + col + 1) * tilelon, origin.lat - (startrow + (row - 1) * rowSign) * tilelat * rowSign) │ │ │ │ │ - }, │ │ │ │ │ - initGriddedTiles: function(bounds) { │ │ │ │ │ - this.events.triggerEvent("retile"); │ │ │ │ │ - var viewSize = this.map.getSize(); │ │ │ │ │ - var origin = this.getTileOrigin(); │ │ │ │ │ - var resolution = this.map.getResolution(), │ │ │ │ │ - serverResolution = this.getServerResolution(), │ │ │ │ │ - ratio = resolution / serverResolution, │ │ │ │ │ - tileSize = { │ │ │ │ │ - w: this.tileSize.w / ratio, │ │ │ │ │ - h: this.tileSize.h / ratio │ │ │ │ │ + var fixedSize = { │ │ │ │ │ + w: safeSize.w < realSize.w ? safeSize.w : null, │ │ │ │ │ + h: safeSize.h < realSize.h ? safeSize.h : null │ │ │ │ │ }; │ │ │ │ │ - var minRows = Math.ceil(viewSize.h / tileSize.h) + 2 * this.buffer + 1; │ │ │ │ │ - var minCols = Math.ceil(viewSize.w / tileSize.w) + 2 * this.buffer + 1; │ │ │ │ │ - var tileLayout = this.calculateGridLayout(bounds, origin, serverResolution); │ │ │ │ │ - this.gridLayout = tileLayout; │ │ │ │ │ - var tilelon = tileLayout.tilelon; │ │ │ │ │ - var tilelat = tileLayout.tilelat; │ │ │ │ │ - var layerContainerDivLeft = this.map.layerContainerOriginPx.x; │ │ │ │ │ - var layerContainerDivTop = this.map.layerContainerOriginPx.y; │ │ │ │ │ - var tileBounds = this.getTileBoundsForGridIndex(0, 0); │ │ │ │ │ - var startPx = this.map.getViewPortPxFromLonLat(new OpenLayers.LonLat(tileBounds.left, tileBounds.top)); │ │ │ │ │ - startPx.x = Math.round(startPx.x) - layerContainerDivLeft; │ │ │ │ │ - startPx.y = Math.round(startPx.y) - layerContainerDivTop; │ │ │ │ │ - var tileData = [], │ │ │ │ │ - center = this.map.getCenter(); │ │ │ │ │ - var rowidx = 0; │ │ │ │ │ - do { │ │ │ │ │ - var row = this.grid[rowidx]; │ │ │ │ │ - if (!row) { │ │ │ │ │ - row = []; │ │ │ │ │ - this.grid.push(row) │ │ │ │ │ - } │ │ │ │ │ - var colidx = 0; │ │ │ │ │ - do { │ │ │ │ │ - tileBounds = this.getTileBoundsForGridIndex(rowidx, colidx); │ │ │ │ │ - var px = startPx.clone(); │ │ │ │ │ - px.x = px.x + colidx * Math.round(tileSize.w); │ │ │ │ │ - px.y = px.y + rowidx * Math.round(tileSize.h); │ │ │ │ │ - var tile = row[colidx]; │ │ │ │ │ - if (!tile) { │ │ │ │ │ - tile = this.addTile(tileBounds, px); │ │ │ │ │ - this.addTileMonitoringHooks(tile); │ │ │ │ │ - row.push(tile) │ │ │ │ │ - } else { │ │ │ │ │ - tile.moveTo(tileBounds, px, false) │ │ │ │ │ - } │ │ │ │ │ - var tileCenter = tileBounds.getCenterLonLat(); │ │ │ │ │ - tileData.push({ │ │ │ │ │ - tile: tile, │ │ │ │ │ - distance: Math.pow(tileCenter.lon - center.lon, 2) + Math.pow(tileCenter.lat - center.lat, 2) │ │ │ │ │ + if (fixedSize.w && fixedSize.h) { │ │ │ │ │ + newSize = safeSize │ │ │ │ │ + } else { │ │ │ │ │ + var clippedSize = OpenLayers.Util.getRenderedDimensions(preparedHTML, fixedSize, { │ │ │ │ │ + displayClass: this.contentDisplayClass, │ │ │ │ │ + containerElement: containerElement │ │ │ │ │ }); │ │ │ │ │ - colidx += 1 │ │ │ │ │ - } while (tileBounds.right <= bounds.right + tilelon * this.buffer || colidx < minCols); │ │ │ │ │ - rowidx += 1 │ │ │ │ │ - } while (tileBounds.bottom >= bounds.bottom - tilelat * this.buffer || rowidx < minRows); │ │ │ │ │ - this.removeExcessTiles(rowidx, colidx); │ │ │ │ │ - var resolution = this.getServerResolution(); │ │ │ │ │ - this.gridResolution = resolution; │ │ │ │ │ - tileData.sort(function(a, b) { │ │ │ │ │ - return a.distance - b.distance │ │ │ │ │ - }); │ │ │ │ │ - for (var i = 0, ii = tileData.length; i < ii; ++i) { │ │ │ │ │ - tileData[i].tile.draw() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getMaxExtent: function() { │ │ │ │ │ - return this.maxExtent │ │ │ │ │ - }, │ │ │ │ │ - addTile: function(bounds, position) { │ │ │ │ │ - var tile = new this.tileClass(this, position, bounds, null, this.tileSize, this.tileOptions); │ │ │ │ │ - this.events.triggerEvent("addtile", { │ │ │ │ │ - tile: tile │ │ │ │ │ - }); │ │ │ │ │ - return tile │ │ │ │ │ - }, │ │ │ │ │ - addTileMonitoringHooks: function(tile) { │ │ │ │ │ - var replacingCls = "olTileReplacing"; │ │ │ │ │ - tile.onLoadStart = function() { │ │ │ │ │ - if (this.loading === false) { │ │ │ │ │ - this.loading = true; │ │ │ │ │ - this.events.triggerEvent("loadstart") │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("tileloadstart", { │ │ │ │ │ - tile: tile │ │ │ │ │ - }); │ │ │ │ │ - this.numLoadingTiles++; │ │ │ │ │ - if (!this.singleTile && this.backBuffer && this.gridResolution === this.backBufferResolution) { │ │ │ │ │ - OpenLayers.Element.addClass(tile.getTile(), replacingCls) │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - tile.onLoadEnd = function(evt) { │ │ │ │ │ - this.numLoadingTiles--; │ │ │ │ │ - var aborted = evt.type === "unload"; │ │ │ │ │ - this.events.triggerEvent("tileloaded", { │ │ │ │ │ - tile: tile, │ │ │ │ │ - aborted: aborted │ │ │ │ │ - }); │ │ │ │ │ - if (!this.singleTile && !aborted && this.backBuffer && this.gridResolution === this.backBufferResolution) { │ │ │ │ │ - var tileDiv = tile.getTile(); │ │ │ │ │ - if (OpenLayers.Element.getStyle(tileDiv, "display") === "none") { │ │ │ │ │ - var bufferTile = document.getElementById(tile.id + "_bb"); │ │ │ │ │ - if (bufferTile) { │ │ │ │ │ - bufferTile.parentNode.removeChild(bufferTile) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Element.removeClass(tileDiv, replacingCls) │ │ │ │ │ - } │ │ │ │ │ - if (this.numLoadingTiles === 0) { │ │ │ │ │ - if (this.backBuffer) { │ │ │ │ │ - if (this.backBuffer.childNodes.length === 0) { │ │ │ │ │ - this.removeBackBuffer() │ │ │ │ │ + var currentOverflow = OpenLayers.Element.getStyle(this.contentDiv, "overflow"); │ │ │ │ │ + if (currentOverflow != "hidden" && clippedSize.equals(safeSize)) { │ │ │ │ │ + var scrollBar = OpenLayers.Util.getScrollbarWidth(); │ │ │ │ │ + if (fixedSize.w) { │ │ │ │ │ + clippedSize.h += scrollBar │ │ │ │ │ } else { │ │ │ │ │ - this._transitionElement = aborted ? this.div.lastChild : tile.imgDiv; │ │ │ │ │ - var transitionendEvents = this.transitionendEvents; │ │ │ │ │ - for (var i = transitionendEvents.length - 1; i >= 0; --i) { │ │ │ │ │ - OpenLayers.Event.observe(this._transitionElement, transitionendEvents[i], this._removeBackBuffer) │ │ │ │ │ - } │ │ │ │ │ - this.backBufferTimerId = window.setTimeout(this._removeBackBuffer, this.removeBackBufferDelay) │ │ │ │ │ + clippedSize.w += scrollBar │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.loading = false; │ │ │ │ │ - this.events.triggerEvent("loadend") │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - tile.onLoadError = function() { │ │ │ │ │ - this.events.triggerEvent("tileerror", { │ │ │ │ │ - tile: tile │ │ │ │ │ - }) │ │ │ │ │ - }; │ │ │ │ │ - tile.events.on({ │ │ │ │ │ - loadstart: tile.onLoadStart, │ │ │ │ │ - loadend: tile.onLoadEnd, │ │ │ │ │ - unload: tile.onLoadEnd, │ │ │ │ │ - loaderror: tile.onLoadError, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - removeTileMonitoringHooks: function(tile) { │ │ │ │ │ - tile.unload(); │ │ │ │ │ - tile.events.un({ │ │ │ │ │ - loadstart: tile.onLoadStart, │ │ │ │ │ - loadend: tile.onLoadEnd, │ │ │ │ │ - unload: tile.onLoadEnd, │ │ │ │ │ - loaderror: tile.onLoadError, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - moveGriddedTiles: function() { │ │ │ │ │ - var buffer = this.buffer + 1; │ │ │ │ │ - while (true) { │ │ │ │ │ - var tlTile = this.grid[0][0]; │ │ │ │ │ - var tlViewPort = { │ │ │ │ │ - x: tlTile.position.x + this.map.layerContainerOriginPx.x, │ │ │ │ │ - y: tlTile.position.y + this.map.layerContainerOriginPx.y │ │ │ │ │ - }; │ │ │ │ │ - var ratio = this.getServerResolution() / this.map.getResolution(); │ │ │ │ │ - var tileSize = { │ │ │ │ │ - w: Math.round(this.tileSize.w * ratio), │ │ │ │ │ - h: Math.round(this.tileSize.h * ratio) │ │ │ │ │ - }; │ │ │ │ │ - if (tlViewPort.x > -tileSize.w * (buffer - 1)) { │ │ │ │ │ - this.shiftColumn(true, tileSize) │ │ │ │ │ - } else if (tlViewPort.x < -tileSize.w * buffer) { │ │ │ │ │ - this.shiftColumn(false, tileSize) │ │ │ │ │ - } else if (tlViewPort.y > -tileSize.h * (buffer - 1)) { │ │ │ │ │ - this.shiftRow(true, tileSize) │ │ │ │ │ - } else if (tlViewPort.y < -tileSize.h * buffer) { │ │ │ │ │ - this.shiftRow(false, tileSize) │ │ │ │ │ - } else { │ │ │ │ │ - break │ │ │ │ │ + newSize = this.getSafeContentSize(clippedSize) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + this.setSize(newSize) │ │ │ │ │ }, │ │ │ │ │ - shiftRow: function(prepend, tileSize) { │ │ │ │ │ - var grid = this.grid; │ │ │ │ │ - var rowIndex = prepend ? 0 : grid.length - 1; │ │ │ │ │ - var sign = prepend ? -1 : 1; │ │ │ │ │ - var rowSign = this.rowSign; │ │ │ │ │ - var tileLayout = this.gridLayout; │ │ │ │ │ - tileLayout.startrow += sign * rowSign; │ │ │ │ │ - var modelRow = grid[rowIndex]; │ │ │ │ │ - var row = grid[prepend ? "pop" : "shift"](); │ │ │ │ │ - for (var i = 0, len = row.length; i < len; i++) { │ │ │ │ │ - var tile = row[i]; │ │ │ │ │ - var position = modelRow[i].position.clone(); │ │ │ │ │ - position.y += tileSize.h * sign; │ │ │ │ │ - tile.moveTo(this.getTileBoundsForGridIndex(rowIndex, i), position) │ │ │ │ │ + setBackgroundColor: function(color) { │ │ │ │ │ + if (color != undefined) { │ │ │ │ │ + this.backgroundColor = color │ │ │ │ │ } │ │ │ │ │ - grid[prepend ? "unshift" : "push"](row) │ │ │ │ │ - }, │ │ │ │ │ - shiftColumn: function(prepend, tileSize) { │ │ │ │ │ - var grid = this.grid; │ │ │ │ │ - var colIndex = prepend ? 0 : grid[0].length - 1; │ │ │ │ │ - var sign = prepend ? -1 : 1; │ │ │ │ │ - var tileLayout = this.gridLayout; │ │ │ │ │ - tileLayout.startcol += sign; │ │ │ │ │ - for (var i = 0, len = grid.length; i < len; i++) { │ │ │ │ │ - var row = grid[i]; │ │ │ │ │ - var position = row[colIndex].position.clone(); │ │ │ │ │ - var tile = row[prepend ? "pop" : "shift"](); │ │ │ │ │ - position.x += tileSize.w * sign; │ │ │ │ │ - tile.moveTo(this.getTileBoundsForGridIndex(i, colIndex), position); │ │ │ │ │ - row[prepend ? "unshift" : "push"](tile) │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.div.style.backgroundColor = this.backgroundColor │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - removeExcessTiles: function(rows, columns) { │ │ │ │ │ - var i, l; │ │ │ │ │ - while (this.grid.length > rows) { │ │ │ │ │ - var row = this.grid.pop(); │ │ │ │ │ - for (i = 0, l = row.length; i < l; i++) { │ │ │ │ │ - var tile = row[i]; │ │ │ │ │ - this.destroyTile(tile) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - for (i = 0, l = this.grid.length; i < l; i++) { │ │ │ │ │ - while (this.grid[i].length > columns) { │ │ │ │ │ - var row = this.grid[i]; │ │ │ │ │ - var tile = row.pop(); │ │ │ │ │ - this.destroyTile(tile) │ │ │ │ │ - } │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + if (opacity != undefined) { │ │ │ │ │ + this.opacity = opacity │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - this.clearGrid(); │ │ │ │ │ - this.setTileSize() │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.div.style.opacity = this.opacity; │ │ │ │ │ + this.div.style.filter = "alpha(opacity=" + this.opacity * 100 + ")" │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - getTileBounds: function(viewPortPx) { │ │ │ │ │ - var maxExtent = this.maxExtent; │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var tileMapWidth = resolution * this.tileSize.w; │ │ │ │ │ - var tileMapHeight = resolution * this.tileSize.h; │ │ │ │ │ - var mapPoint = this.getLonLatFromViewPortPx(viewPortPx); │ │ │ │ │ - var tileLeft = maxExtent.left + tileMapWidth * Math.floor((mapPoint.lon - maxExtent.left) / tileMapWidth); │ │ │ │ │ - var tileBottom = maxExtent.bottom + tileMapHeight * Math.floor((mapPoint.lat - maxExtent.bottom) / tileMapHeight); │ │ │ │ │ - return new OpenLayers.Bounds(tileLeft, tileBottom, tileLeft + tileMapWidth, tileBottom + tileMapHeight) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Grid" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.TileManager = OpenLayers.Class({ │ │ │ │ │ - cacheSize: 256, │ │ │ │ │ - tilesPerFrame: 2, │ │ │ │ │ - frameDelay: 16, │ │ │ │ │ - moveDelay: 100, │ │ │ │ │ - zoomDelay: 200, │ │ │ │ │ - maps: null, │ │ │ │ │ - tileQueueId: null, │ │ │ │ │ - tileQueue: null, │ │ │ │ │ - tileCache: null, │ │ │ │ │ - tileCacheIndex: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.maps = []; │ │ │ │ │ - this.tileQueueId = {}; │ │ │ │ │ - this.tileQueue = {}; │ │ │ │ │ - this.tileCache = {}; │ │ │ │ │ - this.tileCacheIndex = [] │ │ │ │ │ - }, │ │ │ │ │ - addMap: function(map) { │ │ │ │ │ - if (this._destroyed || !OpenLayers.Layer.Grid) { │ │ │ │ │ - return │ │ │ │ │ + setBorder: function(border) { │ │ │ │ │ + if (border != undefined) { │ │ │ │ │ + this.border = border │ │ │ │ │ } │ │ │ │ │ - this.maps.push(map); │ │ │ │ │ - this.tileQueue[map.id] = []; │ │ │ │ │ - for (var i = 0, ii = map.layers.length; i < ii; ++i) { │ │ │ │ │ - this.addLayer({ │ │ │ │ │ - layer: map.layers[i] │ │ │ │ │ - }) │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.div.style.border = this.border │ │ │ │ │ } │ │ │ │ │ - map.events.on({ │ │ │ │ │ - move: this.move, │ │ │ │ │ - zoomend: this.zoomEnd, │ │ │ │ │ - changelayer: this.changeLayer, │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - preremovelayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - if (this._destroyed || !OpenLayers.Layer.Grid) { │ │ │ │ │ - return │ │ │ │ │ + setContentHTML: function(contentHTML) { │ │ │ │ │ + if (contentHTML != null) { │ │ │ │ │ + this.contentHTML = contentHTML │ │ │ │ │ } │ │ │ │ │ - window.clearTimeout(this.tileQueueId[map.id]); │ │ │ │ │ - if (map.layers) { │ │ │ │ │ - for (var i = 0, ii = map.layers.length; i < ii; ++i) { │ │ │ │ │ - this.removeLayer({ │ │ │ │ │ - layer: map.layers[i] │ │ │ │ │ - }) │ │ │ │ │ + if (this.contentDiv != null && this.contentHTML != null && this.contentHTML != this.contentDiv.innerHTML) { │ │ │ │ │ + this.contentDiv.innerHTML = this.contentHTML; │ │ │ │ │ + if (this.autoSize) { │ │ │ │ │ + this.registerImageListeners(); │ │ │ │ │ + this.updateSize() │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (map.events) { │ │ │ │ │ - map.events.un({ │ │ │ │ │ - move: this.move, │ │ │ │ │ - zoomend: this.zoomEnd, │ │ │ │ │ - changelayer: this.changeLayer, │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - preremovelayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - delete this.tileQueue[map.id]; │ │ │ │ │ - delete this.tileQueueId[map.id]; │ │ │ │ │ - OpenLayers.Util.removeItem(this.maps, map) │ │ │ │ │ - }, │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - this.updateTimeout(evt.object, this.moveDelay, true) │ │ │ │ │ - }, │ │ │ │ │ - zoomEnd: function(evt) { │ │ │ │ │ - this.updateTimeout(evt.object, this.zoomDelay) │ │ │ │ │ - }, │ │ │ │ │ - changeLayer: function(evt) { │ │ │ │ │ - if (evt.property === "visibility" || evt.property === "params") { │ │ │ │ │ - this.updateTimeout(evt.object, 0) │ │ │ │ │ - } │ │ │ │ │ }, │ │ │ │ │ - addLayer: function(evt) { │ │ │ │ │ - var layer = evt.layer; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ - layer.events.on({ │ │ │ │ │ - addtile: this.addTile, │ │ │ │ │ - retile: this.clearTileQueue, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - var i, j, tile; │ │ │ │ │ - for (i = layer.grid.length - 1; i >= 0; --i) { │ │ │ │ │ - for (j = layer.grid[i].length - 1; j >= 0; --j) { │ │ │ │ │ - tile = layer.grid[i][j]; │ │ │ │ │ - this.addTile({ │ │ │ │ │ - tile: tile │ │ │ │ │ - }); │ │ │ │ │ - if (tile.url && !tile.imgDiv) { │ │ │ │ │ - this.manageTileCache({ │ │ │ │ │ - object: tile │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + registerImageListeners: function() { │ │ │ │ │ + var onImgLoad = function() { │ │ │ │ │ + if (this.popup.id === null) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - removeLayer: function(evt) { │ │ │ │ │ - var layer = evt.layer; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ - this.clearTileQueue({ │ │ │ │ │ - object: layer │ │ │ │ │ - }); │ │ │ │ │ - if (layer.events) { │ │ │ │ │ - layer.events.un({ │ │ │ │ │ - addtile: this.addTile, │ │ │ │ │ - retile: this.clearTileQueue, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ + this.popup.updateSize(); │ │ │ │ │ + if (this.popup.visible() && this.popup.panMapIfOutOfView) { │ │ │ │ │ + this.popup.panIntoView() │ │ │ │ │ } │ │ │ │ │ - if (layer.grid) { │ │ │ │ │ - var i, j, tile; │ │ │ │ │ - for (i = layer.grid.length - 1; i >= 0; --i) { │ │ │ │ │ - for (j = layer.grid[i].length - 1; j >= 0; --j) { │ │ │ │ │ - tile = layer.grid[i][j]; │ │ │ │ │ - this.unloadTile({ │ │ │ │ │ - object: tile │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + OpenLayers.Event.stopObserving(this.img, "load", this.img._onImgLoad) │ │ │ │ │ + }; │ │ │ │ │ + var images = this.contentDiv.getElementsByTagName("img"); │ │ │ │ │ + for (var i = 0, len = images.length; i < len; i++) { │ │ │ │ │ + var img = images[i]; │ │ │ │ │ + if (img.width == 0 || img.height == 0) { │ │ │ │ │ + var context = { │ │ │ │ │ + popup: this, │ │ │ │ │ + img: img │ │ │ │ │ + }; │ │ │ │ │ + img._onImgLoad = OpenLayers.Function.bind(onImgLoad, context); │ │ │ │ │ + OpenLayers.Event.observe(img, "load", img._onImgLoad) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - updateTimeout: function(map, delay, nice) { │ │ │ │ │ - window.clearTimeout(this.tileQueueId[map.id]); │ │ │ │ │ - var tileQueue = this.tileQueue[map.id]; │ │ │ │ │ - if (!nice || tileQueue.length) { │ │ │ │ │ - this.tileQueueId[map.id] = window.setTimeout(OpenLayers.Function.bind(function() { │ │ │ │ │ - this.drawTilesFromQueue(map); │ │ │ │ │ - if (tileQueue.length) { │ │ │ │ │ - this.updateTimeout(map, this.frameDelay) │ │ │ │ │ - } │ │ │ │ │ - }, this), delay) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - addTile: function(evt) { │ │ │ │ │ - if (evt.tile instanceof OpenLayers.Tile.Image) { │ │ │ │ │ - evt.tile.events.on({ │ │ │ │ │ - beforedraw: this.queueTileDraw, │ │ │ │ │ - beforeload: this.manageTileCache, │ │ │ │ │ - loadend: this.addToCache, │ │ │ │ │ - unload: this.unloadTile, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - this.removeLayer({ │ │ │ │ │ - layer: evt.tile.layer │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - unloadTile: function(evt) { │ │ │ │ │ - var tile = evt.object; │ │ │ │ │ - tile.events.un({ │ │ │ │ │ - beforedraw: this.queueTileDraw, │ │ │ │ │ - beforeload: this.manageTileCache, │ │ │ │ │ - loadend: this.addToCache, │ │ │ │ │ - unload: this.unloadTile, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Util.removeItem(this.tileQueue[tile.layer.map.id], tile) │ │ │ │ │ - }, │ │ │ │ │ - queueTileDraw: function(evt) { │ │ │ │ │ - var tile = evt.object; │ │ │ │ │ - var queued = false; │ │ │ │ │ - var layer = tile.layer; │ │ │ │ │ - var url = layer.getURL(tile.bounds); │ │ │ │ │ - var img = this.tileCache[url]; │ │ │ │ │ - if (img && img.className !== "olTileImage") { │ │ │ │ │ - delete this.tileCache[url]; │ │ │ │ │ - OpenLayers.Util.removeItem(this.tileCacheIndex, url); │ │ │ │ │ - img = null │ │ │ │ │ - } │ │ │ │ │ - if (layer.url && (layer.async || !img)) { │ │ │ │ │ - var tileQueue = this.tileQueue[layer.map.id]; │ │ │ │ │ - if (!~OpenLayers.Util.indexOf(tileQueue, tile)) { │ │ │ │ │ - tileQueue.push(tile) │ │ │ │ │ - } │ │ │ │ │ - queued = true │ │ │ │ │ + getSafeContentSize: function(size) { │ │ │ │ │ + var safeContentSize = size.clone(); │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ + var wPadding = contentDivPadding.left + contentDivPadding.right; │ │ │ │ │ + var hPadding = contentDivPadding.top + contentDivPadding.bottom; │ │ │ │ │ + this.fixPadding(); │ │ │ │ │ + wPadding += this.padding.left + this.padding.right; │ │ │ │ │ + hPadding += this.padding.top + this.padding.bottom; │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + var closeDivWidth = parseInt(this.closeDiv.style.width); │ │ │ │ │ + wPadding += closeDivWidth + contentDivPadding.right │ │ │ │ │ } │ │ │ │ │ - return !queued │ │ │ │ │ - }, │ │ │ │ │ - drawTilesFromQueue: function(map) { │ │ │ │ │ - var tileQueue = this.tileQueue[map.id]; │ │ │ │ │ - var limit = this.tilesPerFrame; │ │ │ │ │ - var animating = map.zoomTween && map.zoomTween.playing; │ │ │ │ │ - while (!animating && tileQueue.length && limit) { │ │ │ │ │ - tileQueue.shift().draw(true); │ │ │ │ │ - --limit │ │ │ │ │ + if (this.minSize) { │ │ │ │ │ + safeContentSize.w = Math.max(safeContentSize.w, this.minSize.w - wPadding); │ │ │ │ │ + safeContentSize.h = Math.max(safeContentSize.h, this.minSize.h - hPadding) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - manageTileCache: function(evt) { │ │ │ │ │ - var tile = evt.object; │ │ │ │ │ - var img = this.tileCache[tile.url]; │ │ │ │ │ - if (img) { │ │ │ │ │ - if (img.parentNode && OpenLayers.Element.hasClass(img.parentNode, "olBackBuffer")) { │ │ │ │ │ - img.parentNode.removeChild(img); │ │ │ │ │ - img.id = null │ │ │ │ │ - } │ │ │ │ │ - if (!img.parentNode) { │ │ │ │ │ - img.style.visibility = "hidden"; │ │ │ │ │ - img.style.opacity = 0; │ │ │ │ │ - tile.setImage(img); │ │ │ │ │ - OpenLayers.Util.removeItem(this.tileCacheIndex, tile.url); │ │ │ │ │ - this.tileCacheIndex.push(tile.url) │ │ │ │ │ - } │ │ │ │ │ + if (this.maxSize) { │ │ │ │ │ + safeContentSize.w = Math.min(safeContentSize.w, this.maxSize.w - wPadding); │ │ │ │ │ + safeContentSize.h = Math.min(safeContentSize.h, this.maxSize.h - hPadding) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - addToCache: function(evt) { │ │ │ │ │ - var tile = evt.object; │ │ │ │ │ - if (!this.tileCache[tile.url]) { │ │ │ │ │ - if (!OpenLayers.Element.hasClass(tile.imgDiv, "olImageLoadError")) { │ │ │ │ │ - if (this.tileCacheIndex.length >= this.cacheSize) { │ │ │ │ │ - delete this.tileCache[this.tileCacheIndex[0]]; │ │ │ │ │ - this.tileCacheIndex.shift() │ │ │ │ │ + if (this.map && this.map.size) { │ │ │ │ │ + var extraX = 0, │ │ │ │ │ + extraY = 0; │ │ │ │ │ + if (this.keepInMap && !this.panMapIfOutOfView) { │ │ │ │ │ + var px = this.map.getPixelFromLonLat(this.lonlat); │ │ │ │ │ + switch (this.relativePosition) { │ │ │ │ │ + case "tr": │ │ │ │ │ + extraX = px.x; │ │ │ │ │ + extraY = this.map.size.h - px.y; │ │ │ │ │ + break; │ │ │ │ │ + case "tl": │ │ │ │ │ + extraX = this.map.size.w - px.x; │ │ │ │ │ + extraY = this.map.size.h - px.y; │ │ │ │ │ + break; │ │ │ │ │ + case "bl": │ │ │ │ │ + extraX = this.map.size.w - px.x; │ │ │ │ │ + extraY = px.y; │ │ │ │ │ + break; │ │ │ │ │ + case "br": │ │ │ │ │ + extraX = px.x; │ │ │ │ │ + extraY = px.y; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + extraX = px.x; │ │ │ │ │ + extraY = this.map.size.h - px.y; │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - this.tileCache[tile.url] = tile.imgDiv; │ │ │ │ │ - this.tileCacheIndex.push(tile.url) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - clearTileQueue: function(evt) { │ │ │ │ │ - var layer = evt.object; │ │ │ │ │ - var tileQueue = this.tileQueue[layer.map.id]; │ │ │ │ │ - for (var i = tileQueue.length - 1; i >= 0; --i) { │ │ │ │ │ - if (tileQueue[i].layer === layer) { │ │ │ │ │ - tileQueue.splice(i, 1) │ │ │ │ │ } │ │ │ │ │ + var maxY = this.map.size.h - this.map.paddingForPopups.top - this.map.paddingForPopups.bottom - hPadding - extraY; │ │ │ │ │ + var maxX = this.map.size.w - this.map.paddingForPopups.left - this.map.paddingForPopups.right - wPadding - extraX; │ │ │ │ │ + safeContentSize.w = Math.min(safeContentSize.w, maxX); │ │ │ │ │ + safeContentSize.h = Math.min(safeContentSize.h, maxY) │ │ │ │ │ } │ │ │ │ │ + return safeContentSize │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i = this.maps.length - 1; i >= 0; --i) { │ │ │ │ │ - this.removeMap(this.maps[i]) │ │ │ │ │ - } │ │ │ │ │ - this.maps = null; │ │ │ │ │ - this.tileQueue = null; │ │ │ │ │ - this.tileQueueId = null; │ │ │ │ │ - this.tileCache = null; │ │ │ │ │ - this.tileCacheIndex = null; │ │ │ │ │ - this._destroyed = true │ │ │ │ │ - } │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control = OpenLayers.Class({ │ │ │ │ │ - id: null, │ │ │ │ │ - map: null, │ │ │ │ │ - div: null, │ │ │ │ │ - type: null, │ │ │ │ │ - allowSelection: false, │ │ │ │ │ - displayClass: "", │ │ │ │ │ - title: "", │ │ │ │ │ - autoActivate: false, │ │ │ │ │ - active: null, │ │ │ │ │ - handlerOptions: null, │ │ │ │ │ - handler: null, │ │ │ │ │ - eventListeners: null, │ │ │ │ │ - events: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.displayClass = this.CLASS_NAME.replace("OpenLayers.", "ol").replace(/\./g, ""); │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.events = new OpenLayers.Events(this); │ │ │ │ │ - if (this.eventListeners instanceof Object) { │ │ │ │ │ - this.events.on(this.eventListeners) │ │ │ │ │ - } │ │ │ │ │ - if (this.id == null) { │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.events) { │ │ │ │ │ - if (this.eventListeners) { │ │ │ │ │ - this.events.un(this.eventListeners) │ │ │ │ │ + getContentDivPadding: function() { │ │ │ │ │ + var contentDivPadding = this._contentDivPadding; │ │ │ │ │ + if (!contentDivPadding) { │ │ │ │ │ + if (this.div.parentNode == null) { │ │ │ │ │ + this.div.style.display = "none"; │ │ │ │ │ + document.body.appendChild(this.div) │ │ │ │ │ } │ │ │ │ │ - this.events.destroy(); │ │ │ │ │ - this.events = null │ │ │ │ │ - } │ │ │ │ │ - this.eventListeners = null; │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.destroy(); │ │ │ │ │ - this.handler = null │ │ │ │ │ - } │ │ │ │ │ - if (this.handlers) { │ │ │ │ │ - for (var key in this.handlers) { │ │ │ │ │ - if (this.handlers.hasOwnProperty(key) && typeof this.handlers[key].destroy == "function") { │ │ │ │ │ - this.handlers[key].destroy() │ │ │ │ │ - } │ │ │ │ │ + contentDivPadding = new OpenLayers.Bounds(OpenLayers.Element.getStyle(this.contentDiv, "padding-left"), OpenLayers.Element.getStyle(this.contentDiv, "padding-bottom"), OpenLayers.Element.getStyle(this.contentDiv, "padding-right"), OpenLayers.Element.getStyle(this.contentDiv, "padding-top")); │ │ │ │ │ + this._contentDivPadding = contentDivPadding; │ │ │ │ │ + if (this.div.parentNode == document.body) { │ │ │ │ │ + document.body.removeChild(this.div); │ │ │ │ │ + this.div.style.display = "" │ │ │ │ │ } │ │ │ │ │ - this.handlers = null │ │ │ │ │ - } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.removeControl(this); │ │ │ │ │ - this.map = null │ │ │ │ │ } │ │ │ │ │ - this.div = null │ │ │ │ │ + return contentDivPadding │ │ │ │ │ }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.map = map; │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.setMap(map) │ │ │ │ │ - } │ │ │ │ │ + addCloseBox: function(callback) { │ │ │ │ │ + this.closeDiv = OpenLayers.Util.createDiv(this.id + "_close", null, { │ │ │ │ │ + w: 17, │ │ │ │ │ + h: 17 │ │ │ │ │ + }); │ │ │ │ │ + this.closeDiv.className = "olPopupCloseBox"; │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ + this.closeDiv.style.right = contentDivPadding.right + "px"; │ │ │ │ │ + this.closeDiv.style.top = contentDivPadding.top + "px"; │ │ │ │ │ + this.groupDiv.appendChild(this.closeDiv); │ │ │ │ │ + var closePopup = callback || function(e) { │ │ │ │ │ + this.hide(); │ │ │ │ │ + OpenLayers.Event.stop(e) │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Event.observe(this.closeDiv, "touchend", OpenLayers.Function.bindAsEventListener(closePopup, this)); │ │ │ │ │ + OpenLayers.Event.observe(this.closeDiv, "click", OpenLayers.Function.bindAsEventListener(closePopup, this)) │ │ │ │ │ }, │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - if (this.div == null) { │ │ │ │ │ - this.div = OpenLayers.Util.createDiv(this.id); │ │ │ │ │ - this.div.className = this.displayClass; │ │ │ │ │ - if (!this.allowSelection) { │ │ │ │ │ - this.div.className += " olControlNoSelect"; │ │ │ │ │ - this.div.setAttribute("unselectable", "on", 0); │ │ │ │ │ - this.div.onselectstart = OpenLayers.Function.False │ │ │ │ │ - } │ │ │ │ │ - if (this.title != "") { │ │ │ │ │ - this.div.title = this.title │ │ │ │ │ - } │ │ │ │ │ + panIntoView: function() { │ │ │ │ │ + var mapSize = this.map.getSize(); │ │ │ │ │ + var origTL = this.map.getViewPortPxFromLayerPx(new OpenLayers.Pixel(parseInt(this.div.style.left), parseInt(this.div.style.top))); │ │ │ │ │ + var newTL = origTL.clone(); │ │ │ │ │ + if (origTL.x < this.map.paddingForPopups.left) { │ │ │ │ │ + newTL.x = this.map.paddingForPopups.left │ │ │ │ │ + } else if (origTL.x + this.size.w > mapSize.w - this.map.paddingForPopups.right) { │ │ │ │ │ + newTL.x = mapSize.w - this.map.paddingForPopups.right - this.size.w │ │ │ │ │ } │ │ │ │ │ - if (px != null) { │ │ │ │ │ - this.position = px.clone() │ │ │ │ │ + if (origTL.y < this.map.paddingForPopups.top) { │ │ │ │ │ + newTL.y = this.map.paddingForPopups.top │ │ │ │ │ + } else if (origTL.y + this.size.h > mapSize.h - this.map.paddingForPopups.bottom) { │ │ │ │ │ + newTL.y = mapSize.h - this.map.paddingForPopups.bottom - this.size.h │ │ │ │ │ } │ │ │ │ │ - this.moveTo(this.position); │ │ │ │ │ - return this.div │ │ │ │ │ + var dx = origTL.x - newTL.x; │ │ │ │ │ + var dy = origTL.y - newTL.y; │ │ │ │ │ + this.map.pan(dx, dy) │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - if (px != null && this.div != null) { │ │ │ │ │ - this.div.style.left = px.x + "px"; │ │ │ │ │ - this.div.style.top = px.y + "px" │ │ │ │ │ + registerEvents: function() { │ │ │ │ │ + this.events = new OpenLayers.Events(this, this.div, null, true); │ │ │ │ │ + │ │ │ │ │ + function onTouchstart(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt, true) │ │ │ │ │ } │ │ │ │ │ + this.events.on({ │ │ │ │ │ + mousedown: this.onmousedown, │ │ │ │ │ + mousemove: this.onmousemove, │ │ │ │ │ + mouseup: this.onmouseup, │ │ │ │ │ + click: this.onclick, │ │ │ │ │ + mouseout: this.onmouseout, │ │ │ │ │ + dblclick: this.ondblclick, │ │ │ │ │ + touchstart: onTouchstart, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.activate() │ │ │ │ │ - } │ │ │ │ │ - this.active = true; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, this.displayClass.replace(/ /g, "") + "Active") │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("activate"); │ │ │ │ │ - return true │ │ │ │ │ + onmousedown: function(evt) { │ │ │ │ │ + this.mousedown = true; │ │ │ │ │ + OpenLayers.Event.stop(evt, true) │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.deactivate() │ │ │ │ │ - } │ │ │ │ │ - this.active = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, this.displayClass.replace(/ /g, "") + "Active") │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("deactivate"); │ │ │ │ │ - return true │ │ │ │ │ + onmousemove: function(evt) { │ │ │ │ │ + if (this.mousedown) { │ │ │ │ │ + OpenLayers.Event.stop(evt, true) │ │ │ │ │ } │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.TYPE_BUTTON = 1; │ │ │ │ │ -OpenLayers.Control.TYPE_TOGGLE = 2; │ │ │ │ │ -OpenLayers.Control.TYPE_TOOL = 3; │ │ │ │ │ -OpenLayers.Protocol = OpenLayers.Class({ │ │ │ │ │ - format: null, │ │ │ │ │ - options: null, │ │ │ │ │ - autoDestroy: true, │ │ │ │ │ - defaultFilter: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.options = options │ │ │ │ │ }, │ │ │ │ │ - mergeWithDefaultFilter: function(filter) { │ │ │ │ │ - var merged; │ │ │ │ │ - if (filter && this.defaultFilter) { │ │ │ │ │ - merged = new OpenLayers.Filter.Logical({ │ │ │ │ │ - type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ - filters: [this.defaultFilter, filter] │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - merged = filter || this.defaultFilter || undefined │ │ │ │ │ + onmouseup: function(evt) { │ │ │ │ │ + if (this.mousedown) { │ │ │ │ │ + this.mousedown = false; │ │ │ │ │ + OpenLayers.Event.stop(evt, true) │ │ │ │ │ } │ │ │ │ │ - return merged │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.options = null; │ │ │ │ │ - this.format = null │ │ │ │ │ + onclick: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt, true) │ │ │ │ │ }, │ │ │ │ │ - read: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.filter = this.mergeWithDefaultFilter(options.filter) │ │ │ │ │ + onmouseout: function(evt) { │ │ │ │ │ + this.mousedown = false │ │ │ │ │ }, │ │ │ │ │ - create: function() {}, │ │ │ │ │ - update: function() {}, │ │ │ │ │ - delete: function() {}, │ │ │ │ │ - commit: function() {}, │ │ │ │ │ - abort: function(response) {}, │ │ │ │ │ - createCallback: function(method, response, options) { │ │ │ │ │ - return OpenLayers.Function.bind(function() { │ │ │ │ │ - method.apply(this, [response, options]) │ │ │ │ │ - }, this) │ │ │ │ │ + ondblclick: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt, true) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Protocol.Response = OpenLayers.Class({ │ │ │ │ │ - code: null, │ │ │ │ │ - requestType: null, │ │ │ │ │ - last: true, │ │ │ │ │ - features: null, │ │ │ │ │ - data: null, │ │ │ │ │ - reqFeatures: null, │ │ │ │ │ - priv: null, │ │ │ │ │ - error: null, │ │ │ │ │ +OpenLayers.Popup.WIDTH = 200; │ │ │ │ │ +OpenLayers.Popup.HEIGHT = 200; │ │ │ │ │ +OpenLayers.Popup.COLOR = "white"; │ │ │ │ │ +OpenLayers.Popup.OPACITY = 1; │ │ │ │ │ +OpenLayers.Popup.BORDER = "0px"; │ │ │ │ │ +OpenLayers.Filter = OpenLayers.Class({ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ OpenLayers.Util.extend(this, options) │ │ │ │ │ }, │ │ │ │ │ - success: function() { │ │ │ │ │ - return this.code > 0 │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.Response" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Protocol.Response.SUCCESS = 1; │ │ │ │ │ -OpenLayers.Protocol.Response.FAILURE = 0; │ │ │ │ │ -OpenLayers.Symbolizer = OpenLayers.Class({ │ │ │ │ │ - zIndex: 0, │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ - OpenLayers.Util.extend(this, config) │ │ │ │ │ + destroy: function() {}, │ │ │ │ │ + evaluate: function(context) { │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ clone: function() { │ │ │ │ │ - var Type = eval(this.CLASS_NAME); │ │ │ │ │ - return new Type(OpenLayers.Util.extend({}, this)) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Symbolizer" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Rule = OpenLayers.Class({ │ │ │ │ │ - id: null, │ │ │ │ │ - name: null, │ │ │ │ │ - title: null, │ │ │ │ │ - description: null, │ │ │ │ │ - context: null, │ │ │ │ │ - filter: null, │ │ │ │ │ - elseFilter: false, │ │ │ │ │ - symbolizer: null, │ │ │ │ │ - symbolizers: null, │ │ │ │ │ - minScaleDenominator: null, │ │ │ │ │ - maxScaleDenominator: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.symbolizer = {}; │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - if (this.symbolizers) { │ │ │ │ │ - delete this.symbolizer │ │ │ │ │ - } │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i in this.symbolizer) { │ │ │ │ │ - this.symbolizer[i] = null │ │ │ │ │ - } │ │ │ │ │ - this.symbolizer = null; │ │ │ │ │ - delete this.symbolizers │ │ │ │ │ - }, │ │ │ │ │ - evaluate: function(feature) { │ │ │ │ │ - var context = this.getContext(feature); │ │ │ │ │ - var applies = true; │ │ │ │ │ - if (this.minScaleDenominator || this.maxScaleDenominator) { │ │ │ │ │ - var scale = feature.layer.map.getScale() │ │ │ │ │ - } │ │ │ │ │ - if (this.minScaleDenominator) { │ │ │ │ │ - applies = scale >= OpenLayers.Style.createLiteral(this.minScaleDenominator, context) │ │ │ │ │ - } │ │ │ │ │ - if (applies && this.maxScaleDenominator) { │ │ │ │ │ - applies = scale < OpenLayers.Style.createLiteral(this.maxScaleDenominator, context) │ │ │ │ │ - } │ │ │ │ │ - if (applies && this.filter) { │ │ │ │ │ - if (this.filter.CLASS_NAME == "OpenLayers.Filter.FeatureId") { │ │ │ │ │ - applies = this.filter.evaluate(feature) │ │ │ │ │ - } else { │ │ │ │ │ - applies = this.filter.evaluate(context) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return applies │ │ │ │ │ - }, │ │ │ │ │ - getContext: function(feature) { │ │ │ │ │ - var context = this.context; │ │ │ │ │ - if (!context) { │ │ │ │ │ - context = feature.attributes || feature.data │ │ │ │ │ - } │ │ │ │ │ - if (typeof this.context == "function") { │ │ │ │ │ - context = this.context(feature) │ │ │ │ │ - } │ │ │ │ │ - return context │ │ │ │ │ + return null │ │ │ │ │ }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - var options = OpenLayers.Util.extend({}, this); │ │ │ │ │ - if (this.symbolizers) { │ │ │ │ │ - var len = this.symbolizers.length; │ │ │ │ │ - options.symbolizers = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - options.symbolizers[i] = this.symbolizers[i].clone() │ │ │ │ │ - } │ │ │ │ │ + toString: function() { │ │ │ │ │ + var string; │ │ │ │ │ + if (OpenLayers.Format && OpenLayers.Format.CQL) { │ │ │ │ │ + string = OpenLayers.Format.CQL.prototype.write(this) │ │ │ │ │ } else { │ │ │ │ │ - options.symbolizer = {}; │ │ │ │ │ - var value, type; │ │ │ │ │ - for (var key in this.symbolizer) { │ │ │ │ │ - value = this.symbolizer[key]; │ │ │ │ │ - type = typeof value; │ │ │ │ │ - if (type === "object") { │ │ │ │ │ - options.symbolizer[key] = OpenLayers.Util.extend({}, value) │ │ │ │ │ - } else if (type === "string") { │ │ │ │ │ - options.symbolizer[key] = value │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + string = Object.prototype.toString.call(this) │ │ │ │ │ } │ │ │ │ │ - options.filter = this.filter && this.filter.clone(); │ │ │ │ │ - options.context = this.context && OpenLayers.Util.extend({}, this.context); │ │ │ │ │ - return new OpenLayers.Rule(options) │ │ │ │ │ + return string │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Rule" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Filter" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Geometry = OpenLayers.Class({ │ │ │ │ │ id: null, │ │ │ │ │ parent: null, │ │ │ │ │ bounds: null, │ │ │ │ │ initialize: function() { │ │ │ │ │ this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ @@ -13109,216 +10648,757 @@ │ │ │ │ │ this.readChildNodes(node, output.boundingBoxData) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ ows: OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.WPSExecute" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.WPSProcess = OpenLayers.Class({ │ │ │ │ │ - client: null, │ │ │ │ │ - server: null, │ │ │ │ │ - identifier: null, │ │ │ │ │ - description: null, │ │ │ │ │ - localWPS: "http://geoserver/wps", │ │ │ │ │ - formats: null, │ │ │ │ │ - chained: 0, │ │ │ │ │ - executeCallbacks: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.executeCallbacks = []; │ │ │ │ │ - this.formats = { │ │ │ │ │ - "application/wkt": new OpenLayers.Format.WKT, │ │ │ │ │ - "application/json": new OpenLayers.Format.GeoJSON │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - describe: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - if (!this.description) { │ │ │ │ │ - this.client.describeProcess(this.server, this.identifier, function(description) { │ │ │ │ │ - if (!this.description) { │ │ │ │ │ - this.parseDescription(description) │ │ │ │ │ - } │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - options.callback.call(options.scope, this.description) │ │ │ │ │ - } │ │ │ │ │ - }, this) │ │ │ │ │ - } else if (options.callback) { │ │ │ │ │ - var description = this.description; │ │ │ │ │ - window.setTimeout(function() { │ │ │ │ │ - options.callback.call(options.scope, description) │ │ │ │ │ - }, 0) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - configure: function(options) { │ │ │ │ │ - this.describe({ │ │ │ │ │ - callback: function() { │ │ │ │ │ - var description = this.description, │ │ │ │ │ - inputs = options.inputs, │ │ │ │ │ - input, i, ii; │ │ │ │ │ - for (i = 0, ii = description.dataInputs.length; i < ii; ++i) { │ │ │ │ │ - input = description.dataInputs[i]; │ │ │ │ │ - this.setInputData(input, inputs[input.identifier]) │ │ │ │ │ - } │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - options.callback.call(options.scope) │ │ │ │ │ +(function() { │ │ │ │ │ + var oXMLHttpRequest = window.XMLHttpRequest; │ │ │ │ │ + var bGecko = !!window.controllers, │ │ │ │ │ + bIE = window.document.all && !window.opera, │ │ │ │ │ + bIE7 = bIE && window.navigator.userAgent.match(/MSIE 7.0/); │ │ │ │ │ + │ │ │ │ │ + function fXMLHttpRequest() { │ │ │ │ │ + this._object = oXMLHttpRequest && !bIE7 ? new oXMLHttpRequest : new window.ActiveXObject("Microsoft.XMLHTTP"); │ │ │ │ │ + this._listeners = [] │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function cXMLHttpRequest() { │ │ │ │ │ + return new fXMLHttpRequest │ │ │ │ │ + } │ │ │ │ │ + cXMLHttpRequest.prototype = fXMLHttpRequest.prototype; │ │ │ │ │ + if (bGecko && oXMLHttpRequest.wrapped) cXMLHttpRequest.wrapped = oXMLHttpRequest.wrapped; │ │ │ │ │ + cXMLHttpRequest.UNSENT = 0; │ │ │ │ │ + cXMLHttpRequest.OPENED = 1; │ │ │ │ │ + cXMLHttpRequest.HEADERS_RECEIVED = 2; │ │ │ │ │ + cXMLHttpRequest.LOADING = 3; │ │ │ │ │ + cXMLHttpRequest.DONE = 4; │ │ │ │ │ + cXMLHttpRequest.prototype.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + cXMLHttpRequest.prototype.responseText = ""; │ │ │ │ │ + cXMLHttpRequest.prototype.responseXML = null; │ │ │ │ │ + cXMLHttpRequest.prototype.status = 0; │ │ │ │ │ + cXMLHttpRequest.prototype.statusText = ""; │ │ │ │ │ + cXMLHttpRequest.prototype.priority = "NORMAL"; │ │ │ │ │ + cXMLHttpRequest.prototype.onreadystatechange = null; │ │ │ │ │ + cXMLHttpRequest.onreadystatechange = null; │ │ │ │ │ + cXMLHttpRequest.onopen = null; │ │ │ │ │ + cXMLHttpRequest.onsend = null; │ │ │ │ │ + cXMLHttpRequest.onabort = null; │ │ │ │ │ + cXMLHttpRequest.prototype.open = function(sMethod, sUrl, bAsync, sUser, sPassword) { │ │ │ │ │ + delete this._headers; │ │ │ │ │ + if (arguments.length < 3) bAsync = true; │ │ │ │ │ + this._async = bAsync; │ │ │ │ │ + var oRequest = this, │ │ │ │ │ + nState = this.readyState, │ │ │ │ │ + fOnUnload; │ │ │ │ │ + if (bIE && bAsync) { │ │ │ │ │ + fOnUnload = function() { │ │ │ │ │ + if (nState != cXMLHttpRequest.DONE) { │ │ │ │ │ + fCleanTransport(oRequest); │ │ │ │ │ + oRequest.abort() │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - return this │ │ │ │ │ - }, │ │ │ │ │ - execute: function(options) { │ │ │ │ │ - this.configure({ │ │ │ │ │ - inputs: options.inputs, │ │ │ │ │ - callback: function() { │ │ │ │ │ - var me = this; │ │ │ │ │ - var outputIndex = this.getOutputIndex(me.description.processOutputs, options.output); │ │ │ │ │ - me.setResponseForm({ │ │ │ │ │ - outputIndex: outputIndex │ │ │ │ │ - }); │ │ │ │ │ - (function callback() { │ │ │ │ │ - OpenLayers.Util.removeItem(me.executeCallbacks, callback); │ │ │ │ │ - if (me.chained !== 0) { │ │ │ │ │ - me.executeCallbacks.push(callback); │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Request.POST({ │ │ │ │ │ - url: me.client.servers[me.server].url, │ │ │ │ │ - data: (new OpenLayers.Format.WPSExecute).write(me.description), │ │ │ │ │ - success: function(response) { │ │ │ │ │ - var output = me.description.processOutputs[outputIndex]; │ │ │ │ │ - var mimeType = me.findMimeType(output.complexOutput.supported.formats); │ │ │ │ │ - var features = me.formats[mimeType].read(response.responseText); │ │ │ │ │ - if (features instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ - features = [features] │ │ │ │ │ - } │ │ │ │ │ - if (options.success) { │ │ │ │ │ - var outputs = {}; │ │ │ │ │ - outputs[options.output || "result"] = features; │ │ │ │ │ - options.success.call(options.scope, outputs) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - scope: me │ │ │ │ │ - }) │ │ │ │ │ - })() │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - output: function(identifier) { │ │ │ │ │ - return new OpenLayers.WPSProcess.ChainLink({ │ │ │ │ │ - process: this, │ │ │ │ │ - output: identifier │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - parseDescription: function(description) { │ │ │ │ │ - var server = this.client.servers[this.server]; │ │ │ │ │ - this.description = (new OpenLayers.Format.WPSDescribeProcess).read(server.processDescription[this.identifier]).processDescriptions[this.identifier] │ │ │ │ │ - }, │ │ │ │ │ - setInputData: function(input, data) { │ │ │ │ │ - delete input.data; │ │ │ │ │ - delete input.reference; │ │ │ │ │ - if (data instanceof OpenLayers.WPSProcess.ChainLink) { │ │ │ │ │ - ++this.chained; │ │ │ │ │ - input.reference = { │ │ │ │ │ - method: "POST", │ │ │ │ │ - href: data.process.server === this.server ? this.localWPS : this.client.servers[data.process.server].url │ │ │ │ │ }; │ │ │ │ │ - data.process.describe({ │ │ │ │ │ - callback: function() { │ │ │ │ │ - --this.chained; │ │ │ │ │ - this.chainProcess(input, data) │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - input.data = {}; │ │ │ │ │ - var complexData = input.complexData; │ │ │ │ │ - if (complexData) { │ │ │ │ │ - var format = this.findMimeType(complexData.supported.formats); │ │ │ │ │ - input.data.complexData = { │ │ │ │ │ - mimeType: format, │ │ │ │ │ - value: this.formats[format].write(this.toFeatures(data)) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - input.data.literalData = { │ │ │ │ │ - value: data │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + window.attachEvent("onunload", fOnUnload) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - setResponseForm: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var output = this.description.processOutputs[options.outputIndex || 0]; │ │ │ │ │ - this.description.responseForm = { │ │ │ │ │ - rawDataOutput: { │ │ │ │ │ - identifier: output.identifier, │ │ │ │ │ - mimeType: this.findMimeType(output.complexOutput.supported.formats, options.supportedFormats) │ │ │ │ │ + if (cXMLHttpRequest.onopen) cXMLHttpRequest.onopen.apply(this, arguments); │ │ │ │ │ + if (arguments.length > 4) this._object.open(sMethod, sUrl, bAsync, sUser, sPassword); │ │ │ │ │ + else if (arguments.length > 3) this._object.open(sMethod, sUrl, bAsync, sUser); │ │ │ │ │ + else this._object.open(sMethod, sUrl, bAsync); │ │ │ │ │ + this.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ + fReadyStateChange(this); │ │ │ │ │ + this._object.onreadystatechange = function() { │ │ │ │ │ + if (bGecko && !bAsync) return; │ │ │ │ │ + oRequest.readyState = oRequest._object.readyState; │ │ │ │ │ + fSynchronizeValues(oRequest); │ │ │ │ │ + if (oRequest._aborted) { │ │ │ │ │ + oRequest.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getOutputIndex: function(outputs, identifier) { │ │ │ │ │ - var output; │ │ │ │ │ - if (identifier) { │ │ │ │ │ - for (var i = outputs.length - 1; i >= 0; --i) { │ │ │ │ │ - if (outputs[i].identifier === identifier) { │ │ │ │ │ - output = i; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ + if (oRequest.readyState == cXMLHttpRequest.DONE) { │ │ │ │ │ + delete oRequest._data; │ │ │ │ │ + fCleanTransport(oRequest); │ │ │ │ │ + if (bIE && bAsync) window.detachEvent("onunload", fOnUnload) │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - output = 0 │ │ │ │ │ - } │ │ │ │ │ - return output │ │ │ │ │ - }, │ │ │ │ │ - chainProcess: function(input, chainLink) { │ │ │ │ │ - var output = this.getOutputIndex(chainLink.process.description.processOutputs, chainLink.output); │ │ │ │ │ - input.reference.mimeType = this.findMimeType(input.complexData.supported.formats, chainLink.process.description.processOutputs[output].complexOutput.supported.formats); │ │ │ │ │ - var formats = {}; │ │ │ │ │ - formats[input.reference.mimeType] = true; │ │ │ │ │ - chainLink.process.setResponseForm({ │ │ │ │ │ - outputIndex: output, │ │ │ │ │ - supportedFormats: formats │ │ │ │ │ - }); │ │ │ │ │ - input.reference.body = chainLink.process.description; │ │ │ │ │ - while (this.executeCallbacks.length > 0) { │ │ │ │ │ - this.executeCallbacks[0]() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - toFeatures: function(source) { │ │ │ │ │ - var isArray = OpenLayers.Util.isArray(source); │ │ │ │ │ - if (!isArray) { │ │ │ │ │ - source = [source] │ │ │ │ │ - } │ │ │ │ │ - var target = new Array(source.length), │ │ │ │ │ - current; │ │ │ │ │ - for (var i = 0, ii = source.length; i < ii; ++i) { │ │ │ │ │ - current = source[i]; │ │ │ │ │ - target[i] = current instanceof OpenLayers.Feature.Vector ? current : new OpenLayers.Feature.Vector(current) │ │ │ │ │ + if (nState != oRequest.readyState) fReadyStateChange(oRequest); │ │ │ │ │ + nState = oRequest.readyState │ │ │ │ │ } │ │ │ │ │ - return isArray ? target : target[0] │ │ │ │ │ - }, │ │ │ │ │ - findMimeType: function(sourceFormats, targetFormats) { │ │ │ │ │ - targetFormats = targetFormats || this.formats; │ │ │ │ │ - for (var f in sourceFormats) { │ │ │ │ │ - if (f in targetFormats) { │ │ │ │ │ - return f │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fXMLHttpRequest_send(oRequest) { │ │ │ │ │ + oRequest._object.send(oRequest._data); │ │ │ │ │ + if (bGecko && !oRequest._async) { │ │ │ │ │ + oRequest.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ + fSynchronizeValues(oRequest); │ │ │ │ │ + while (oRequest.readyState < cXMLHttpRequest.DONE) { │ │ │ │ │ + oRequest.readyState++; │ │ │ │ │ + fReadyStateChange(oRequest); │ │ │ │ │ + if (oRequest._aborted) return │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.WPSProcess" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.WPSProcess.ChainLink = OpenLayers.Class({ │ │ │ │ │ - process: null, │ │ │ │ │ - output: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options) │ │ │ │ │ - }, │ │ │ │ │ + } │ │ │ │ │ + cXMLHttpRequest.prototype.send = function(vData) { │ │ │ │ │ + if (cXMLHttpRequest.onsend) cXMLHttpRequest.onsend.apply(this, arguments); │ │ │ │ │ + if (!arguments.length) vData = null; │ │ │ │ │ + if (vData && vData.nodeType) { │ │ │ │ │ + vData = window.XMLSerializer ? (new window.XMLSerializer).serializeToString(vData) : vData.xml; │ │ │ │ │ + if (!this._headers["Content-Type"]) this._object.setRequestHeader("Content-Type", "application/xml") │ │ │ │ │ + } │ │ │ │ │ + this._data = vData; │ │ │ │ │ + fXMLHttpRequest_send(this) │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.abort = function() { │ │ │ │ │ + if (cXMLHttpRequest.onabort) cXMLHttpRequest.onabort.apply(this, arguments); │ │ │ │ │ + if (this.readyState > cXMLHttpRequest.UNSENT) this._aborted = true; │ │ │ │ │ + this._object.abort(); │ │ │ │ │ + fCleanTransport(this); │ │ │ │ │ + this.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + delete this._data │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.getAllResponseHeaders = function() { │ │ │ │ │ + return this._object.getAllResponseHeaders() │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.getResponseHeader = function(sName) { │ │ │ │ │ + return this._object.getResponseHeader(sName) │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.setRequestHeader = function(sName, sValue) { │ │ │ │ │ + if (!this._headers) this._headers = {}; │ │ │ │ │ + this._headers[sName] = sValue; │ │ │ │ │ + return this._object.setRequestHeader(sName, sValue) │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.addEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ + for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ + if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) return; │ │ │ │ │ + this._listeners.push([sName, fHandler, bUseCapture]) │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.removeEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ + for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ + if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) break; │ │ │ │ │ + if (oListener) this._listeners.splice(nIndex, 1) │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.dispatchEvent = function(oEvent) { │ │ │ │ │ + var oEventPseudo = { │ │ │ │ │ + type: oEvent.type, │ │ │ │ │ + target: this, │ │ │ │ │ + currentTarget: this, │ │ │ │ │ + eventPhase: 2, │ │ │ │ │ + bubbles: oEvent.bubbles, │ │ │ │ │ + cancelable: oEvent.cancelable, │ │ │ │ │ + timeStamp: oEvent.timeStamp, │ │ │ │ │ + stopPropagation: function() {}, │ │ │ │ │ + preventDefault: function() {}, │ │ │ │ │ + initEvent: function() {} │ │ │ │ │ + }; │ │ │ │ │ + if (oEventPseudo.type == "readystatechange" && this.onreadystatechange)(this.onreadystatechange.handleEvent || this.onreadystatechange).apply(this, [oEventPseudo]); │ │ │ │ │ + for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ + if (oListener[0] == oEventPseudo.type && !oListener[2])(oListener[1].handleEvent || oListener[1]).apply(this, [oEventPseudo]) │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.toString = function() { │ │ │ │ │ + return "[" + "object" + " " + "XMLHttpRequest" + "]" │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.toString = function() { │ │ │ │ │ + return "[" + "XMLHttpRequest" + "]" │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fReadyStateChange(oRequest) { │ │ │ │ │ + if (cXMLHttpRequest.onreadystatechange) cXMLHttpRequest.onreadystatechange.apply(oRequest); │ │ │ │ │ + oRequest.dispatchEvent({ │ │ │ │ │ + type: "readystatechange", │ │ │ │ │ + bubbles: false, │ │ │ │ │ + cancelable: false, │ │ │ │ │ + timeStamp: new Date + 0 │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function fGetDocument(oRequest) { │ │ │ │ │ + var oDocument = oRequest.responseXML, │ │ │ │ │ + sResponse = oRequest.responseText; │ │ │ │ │ + if (bIE && sResponse && oDocument && !oDocument.documentElement && oRequest.getResponseHeader("Content-Type").match(/[^\/]+\/[^\+]+\+xml/)) { │ │ │ │ │ + oDocument = new window.ActiveXObject("Microsoft.XMLDOM"); │ │ │ │ │ + oDocument.async = false; │ │ │ │ │ + oDocument.validateOnParse = false; │ │ │ │ │ + oDocument.loadXML(sResponse) │ │ │ │ │ + } │ │ │ │ │ + if (oDocument) │ │ │ │ │ + if (bIE && oDocument.parseError != 0 || !oDocument.documentElement || oDocument.documentElement && oDocument.documentElement.tagName == "parsererror") return null; │ │ │ │ │ + return oDocument │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function fSynchronizeValues(oRequest) { │ │ │ │ │ + try { │ │ │ │ │ + oRequest.responseText = oRequest._object.responseText │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + try { │ │ │ │ │ + oRequest.responseXML = fGetDocument(oRequest._object) │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + try { │ │ │ │ │ + oRequest.status = oRequest._object.status │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + try { │ │ │ │ │ + oRequest.statusText = oRequest._object.statusText │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function fCleanTransport(oRequest) { │ │ │ │ │ + oRequest._object.onreadystatechange = new window.Function │ │ │ │ │ + } │ │ │ │ │ + if (!window.Function.prototype.apply) { │ │ │ │ │ + window.Function.prototype.apply = function(oRequest, oArguments) { │ │ │ │ │ + if (!oArguments) oArguments = []; │ │ │ │ │ + oRequest.__func = this; │ │ │ │ │ + oRequest.__func(oArguments[0], oArguments[1], oArguments[2], oArguments[3], oArguments[4]); │ │ │ │ │ + delete oRequest.__func │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!OpenLayers.Request) { │ │ │ │ │ + OpenLayers.Request = {} │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Request.XMLHttpRequest = cXMLHttpRequest │ │ │ │ │ +})(); │ │ │ │ │ +OpenLayers.ProxyHost = ""; │ │ │ │ │ +if (!OpenLayers.Request) { │ │ │ │ │ + OpenLayers.Request = {} │ │ │ │ │ +} │ │ │ │ │ +OpenLayers.Util.extend(OpenLayers.Request, { │ │ │ │ │ + DEFAULT_CONFIG: { │ │ │ │ │ + method: "GET", │ │ │ │ │ + url: window.location.href, │ │ │ │ │ + async: true, │ │ │ │ │ + user: undefined, │ │ │ │ │ + password: undefined, │ │ │ │ │ + params: null, │ │ │ │ │ + proxy: OpenLayers.ProxyHost, │ │ │ │ │ + headers: {}, │ │ │ │ │ + data: null, │ │ │ │ │ + callback: function() {}, │ │ │ │ │ + success: null, │ │ │ │ │ + failure: null, │ │ │ │ │ + scope: null │ │ │ │ │ + }, │ │ │ │ │ + URL_SPLIT_REGEX: /([^:]*:)\/\/([^:]*:?[^@]*@)?([^:\/\?]*):?([^\/\?]*)/, │ │ │ │ │ + events: new OpenLayers.Events(this), │ │ │ │ │ + makeSameOrigin: function(url, proxy) { │ │ │ │ │ + var sameOrigin = url.indexOf("http") !== 0; │ │ │ │ │ + var urlParts = !sameOrigin && url.match(this.URL_SPLIT_REGEX); │ │ │ │ │ + if (urlParts) { │ │ │ │ │ + var location = window.location; │ │ │ │ │ + sameOrigin = urlParts[1] == location.protocol && urlParts[3] == location.hostname; │ │ │ │ │ + var uPort = urlParts[4], │ │ │ │ │ + lPort = location.port; │ │ │ │ │ + if (uPort != 80 && uPort != "" || lPort != "80" && lPort != "") { │ │ │ │ │ + sameOrigin = sameOrigin && uPort == lPort │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!sameOrigin) { │ │ │ │ │ + if (proxy) { │ │ │ │ │ + if (typeof proxy == "function") { │ │ │ │ │ + url = proxy(url) │ │ │ │ │ + } else { │ │ │ │ │ + url = proxy + encodeURIComponent(url) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return url │ │ │ │ │ + }, │ │ │ │ │ + issue: function(config) { │ │ │ │ │ + var defaultConfig = OpenLayers.Util.extend(this.DEFAULT_CONFIG, { │ │ │ │ │ + proxy: OpenLayers.ProxyHost │ │ │ │ │ + }); │ │ │ │ │ + config = config || {}; │ │ │ │ │ + config.headers = config.headers || {}; │ │ │ │ │ + config = OpenLayers.Util.applyDefaults(config, defaultConfig); │ │ │ │ │ + config.headers = OpenLayers.Util.applyDefaults(config.headers, defaultConfig.headers); │ │ │ │ │ + var customRequestedWithHeader = false, │ │ │ │ │ + headerKey; │ │ │ │ │ + for (headerKey in config.headers) { │ │ │ │ │ + if (config.headers.hasOwnProperty(headerKey)) { │ │ │ │ │ + if (headerKey.toLowerCase() === "x-requested-with") { │ │ │ │ │ + customRequestedWithHeader = true │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (customRequestedWithHeader === false) { │ │ │ │ │ + config.headers["X-Requested-With"] = "XMLHttpRequest" │ │ │ │ │ + } │ │ │ │ │ + var request = new OpenLayers.Request.XMLHttpRequest; │ │ │ │ │ + var url = OpenLayers.Util.urlAppend(config.url, OpenLayers.Util.getParameterString(config.params || {})); │ │ │ │ │ + url = OpenLayers.Request.makeSameOrigin(url, config.proxy); │ │ │ │ │ + request.open(config.method, url, config.async, config.user, config.password); │ │ │ │ │ + for (var header in config.headers) { │ │ │ │ │ + request.setRequestHeader(header, config.headers[header]) │ │ │ │ │ + } │ │ │ │ │ + var events = this.events; │ │ │ │ │ + var self = this; │ │ │ │ │ + request.onreadystatechange = function() { │ │ │ │ │ + if (request.readyState == OpenLayers.Request.XMLHttpRequest.DONE) { │ │ │ │ │ + var proceed = events.triggerEvent("complete", { │ │ │ │ │ + request: request, │ │ │ │ │ + config: config, │ │ │ │ │ + requestUrl: url │ │ │ │ │ + }); │ │ │ │ │ + if (proceed !== false) { │ │ │ │ │ + self.runCallbacks({ │ │ │ │ │ + request: request, │ │ │ │ │ + config: config, │ │ │ │ │ + requestUrl: url │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + if (config.async === false) { │ │ │ │ │ + request.send(config.data) │ │ │ │ │ + } else { │ │ │ │ │ + window.setTimeout(function() { │ │ │ │ │ + if (request.readyState !== 0) { │ │ │ │ │ + request.send(config.data) │ │ │ │ │ + } │ │ │ │ │ + }, 0) │ │ │ │ │ + } │ │ │ │ │ + return request │ │ │ │ │ + }, │ │ │ │ │ + runCallbacks: function(options) { │ │ │ │ │ + var request = options.request; │ │ │ │ │ + var config = options.config; │ │ │ │ │ + var complete = config.scope ? OpenLayers.Function.bind(config.callback, config.scope) : config.callback; │ │ │ │ │ + var success; │ │ │ │ │ + if (config.success) { │ │ │ │ │ + success = config.scope ? OpenLayers.Function.bind(config.success, config.scope) : config.success │ │ │ │ │ + } │ │ │ │ │ + var failure; │ │ │ │ │ + if (config.failure) { │ │ │ │ │ + failure = config.scope ? OpenLayers.Function.bind(config.failure, config.scope) : config.failure │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.Util.createUrlObject(config.url).protocol == "file:" && request.responseText) { │ │ │ │ │ + request.status = 200 │ │ │ │ │ + } │ │ │ │ │ + complete(request); │ │ │ │ │ + if (!request.status || request.status >= 200 && request.status < 300) { │ │ │ │ │ + this.events.triggerEvent("success", options); │ │ │ │ │ + if (success) { │ │ │ │ │ + success(request) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ + this.events.triggerEvent("failure", options); │ │ │ │ │ + if (failure) { │ │ │ │ │ + failure(request) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + GET: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "GET" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config) │ │ │ │ │ + }, │ │ │ │ │ + POST: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "POST" │ │ │ │ │ + }); │ │ │ │ │ + config.headers = config.headers ? config.headers : {}; │ │ │ │ │ + if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ + config.headers["Content-Type"] = "application/xml" │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Request.issue(config) │ │ │ │ │ + }, │ │ │ │ │ + PUT: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "PUT" │ │ │ │ │ + }); │ │ │ │ │ + config.headers = config.headers ? config.headers : {}; │ │ │ │ │ + if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ + config.headers["Content-Type"] = "application/xml" │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Request.issue(config) │ │ │ │ │ + }, │ │ │ │ │ + DELETE: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "DELETE" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config) │ │ │ │ │ + }, │ │ │ │ │ + HEAD: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "HEAD" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config) │ │ │ │ │ + }, │ │ │ │ │ + OPTIONS: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "OPTIONS" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config) │ │ │ │ │ + } │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.WPSProcess = OpenLayers.Class({ │ │ │ │ │ + client: null, │ │ │ │ │ + server: null, │ │ │ │ │ + identifier: null, │ │ │ │ │ + description: null, │ │ │ │ │ + localWPS: "http://geoserver/wps", │ │ │ │ │ + formats: null, │ │ │ │ │ + chained: 0, │ │ │ │ │ + executeCallbacks: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.executeCallbacks = []; │ │ │ │ │ + this.formats = { │ │ │ │ │ + "application/wkt": new OpenLayers.Format.WKT, │ │ │ │ │ + "application/json": new OpenLayers.Format.GeoJSON │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + describe: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + if (!this.description) { │ │ │ │ │ + this.client.describeProcess(this.server, this.identifier, function(description) { │ │ │ │ │ + if (!this.description) { │ │ │ │ │ + this.parseDescription(description) │ │ │ │ │ + } │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + options.callback.call(options.scope, this.description) │ │ │ │ │ + } │ │ │ │ │ + }, this) │ │ │ │ │ + } else if (options.callback) { │ │ │ │ │ + var description = this.description; │ │ │ │ │ + window.setTimeout(function() { │ │ │ │ │ + options.callback.call(options.scope, description) │ │ │ │ │ + }, 0) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + configure: function(options) { │ │ │ │ │ + this.describe({ │ │ │ │ │ + callback: function() { │ │ │ │ │ + var description = this.description, │ │ │ │ │ + inputs = options.inputs, │ │ │ │ │ + input, i, ii; │ │ │ │ │ + for (i = 0, ii = description.dataInputs.length; i < ii; ++i) { │ │ │ │ │ + input = description.dataInputs[i]; │ │ │ │ │ + this.setInputData(input, inputs[input.identifier]) │ │ │ │ │ + } │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + options.callback.call(options.scope) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + return this │ │ │ │ │ + }, │ │ │ │ │ + execute: function(options) { │ │ │ │ │ + this.configure({ │ │ │ │ │ + inputs: options.inputs, │ │ │ │ │ + callback: function() { │ │ │ │ │ + var me = this; │ │ │ │ │ + var outputIndex = this.getOutputIndex(me.description.processOutputs, options.output); │ │ │ │ │ + me.setResponseForm({ │ │ │ │ │ + outputIndex: outputIndex │ │ │ │ │ + }); │ │ │ │ │ + (function callback() { │ │ │ │ │ + OpenLayers.Util.removeItem(me.executeCallbacks, callback); │ │ │ │ │ + if (me.chained !== 0) { │ │ │ │ │ + me.executeCallbacks.push(callback); │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Request.POST({ │ │ │ │ │ + url: me.client.servers[me.server].url, │ │ │ │ │ + data: (new OpenLayers.Format.WPSExecute).write(me.description), │ │ │ │ │ + success: function(response) { │ │ │ │ │ + var output = me.description.processOutputs[outputIndex]; │ │ │ │ │ + var mimeType = me.findMimeType(output.complexOutput.supported.formats); │ │ │ │ │ + var features = me.formats[mimeType].read(response.responseText); │ │ │ │ │ + if (features instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ + features = [features] │ │ │ │ │ + } │ │ │ │ │ + if (options.success) { │ │ │ │ │ + var outputs = {}; │ │ │ │ │ + outputs[options.output || "result"] = features; │ │ │ │ │ + options.success.call(options.scope, outputs) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + scope: me │ │ │ │ │ + }) │ │ │ │ │ + })() │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + output: function(identifier) { │ │ │ │ │ + return new OpenLayers.WPSProcess.ChainLink({ │ │ │ │ │ + process: this, │ │ │ │ │ + output: identifier │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + parseDescription: function(description) { │ │ │ │ │ + var server = this.client.servers[this.server]; │ │ │ │ │ + this.description = (new OpenLayers.Format.WPSDescribeProcess).read(server.processDescription[this.identifier]).processDescriptions[this.identifier] │ │ │ │ │ + }, │ │ │ │ │ + setInputData: function(input, data) { │ │ │ │ │ + delete input.data; │ │ │ │ │ + delete input.reference; │ │ │ │ │ + if (data instanceof OpenLayers.WPSProcess.ChainLink) { │ │ │ │ │ + ++this.chained; │ │ │ │ │ + input.reference = { │ │ │ │ │ + method: "POST", │ │ │ │ │ + href: data.process.server === this.server ? this.localWPS : this.client.servers[data.process.server].url │ │ │ │ │ + }; │ │ │ │ │ + data.process.describe({ │ │ │ │ │ + callback: function() { │ │ │ │ │ + --this.chained; │ │ │ │ │ + this.chainProcess(input, data) │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + input.data = {}; │ │ │ │ │ + var complexData = input.complexData; │ │ │ │ │ + if (complexData) { │ │ │ │ │ + var format = this.findMimeType(complexData.supported.formats); │ │ │ │ │ + input.data.complexData = { │ │ │ │ │ + mimeType: format, │ │ │ │ │ + value: this.formats[format].write(this.toFeatures(data)) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + input.data.literalData = { │ │ │ │ │ + value: data │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setResponseForm: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var output = this.description.processOutputs[options.outputIndex || 0]; │ │ │ │ │ + this.description.responseForm = { │ │ │ │ │ + rawDataOutput: { │ │ │ │ │ + identifier: output.identifier, │ │ │ │ │ + mimeType: this.findMimeType(output.complexOutput.supported.formats, options.supportedFormats) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getOutputIndex: function(outputs, identifier) { │ │ │ │ │ + var output; │ │ │ │ │ + if (identifier) { │ │ │ │ │ + for (var i = outputs.length - 1; i >= 0; --i) { │ │ │ │ │ + if (outputs[i].identifier === identifier) { │ │ │ │ │ + output = i; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + output = 0 │ │ │ │ │ + } │ │ │ │ │ + return output │ │ │ │ │ + }, │ │ │ │ │ + chainProcess: function(input, chainLink) { │ │ │ │ │ + var output = this.getOutputIndex(chainLink.process.description.processOutputs, chainLink.output); │ │ │ │ │ + input.reference.mimeType = this.findMimeType(input.complexData.supported.formats, chainLink.process.description.processOutputs[output].complexOutput.supported.formats); │ │ │ │ │ + var formats = {}; │ │ │ │ │ + formats[input.reference.mimeType] = true; │ │ │ │ │ + chainLink.process.setResponseForm({ │ │ │ │ │ + outputIndex: output, │ │ │ │ │ + supportedFormats: formats │ │ │ │ │ + }); │ │ │ │ │ + input.reference.body = chainLink.process.description; │ │ │ │ │ + while (this.executeCallbacks.length > 0) { │ │ │ │ │ + this.executeCallbacks[0]() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + toFeatures: function(source) { │ │ │ │ │ + var isArray = OpenLayers.Util.isArray(source); │ │ │ │ │ + if (!isArray) { │ │ │ │ │ + source = [source] │ │ │ │ │ + } │ │ │ │ │ + var target = new Array(source.length), │ │ │ │ │ + current; │ │ │ │ │ + for (var i = 0, ii = source.length; i < ii; ++i) { │ │ │ │ │ + current = source[i]; │ │ │ │ │ + target[i] = current instanceof OpenLayers.Feature.Vector ? current : new OpenLayers.Feature.Vector(current) │ │ │ │ │ + } │ │ │ │ │ + return isArray ? target : target[0] │ │ │ │ │ + }, │ │ │ │ │ + findMimeType: function(sourceFormats, targetFormats) { │ │ │ │ │ + targetFormats = targetFormats || this.formats; │ │ │ │ │ + for (var f in sourceFormats) { │ │ │ │ │ + if (f in targetFormats) { │ │ │ │ │ + return f │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.WPSProcess" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.WPSProcess.ChainLink = OpenLayers.Class({ │ │ │ │ │ + process: null, │ │ │ │ │ + output: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options) │ │ │ │ │ + }, │ │ │ │ │ CLASS_NAME: "OpenLayers.WPSProcess.ChainLink" │ │ │ │ │ }); │ │ │ │ │ +OpenLayers.Icon = OpenLayers.Class({ │ │ │ │ │ + url: null, │ │ │ │ │ + size: null, │ │ │ │ │ + offset: null, │ │ │ │ │ + calculateOffset: null, │ │ │ │ │ + imageDiv: null, │ │ │ │ │ + px: null, │ │ │ │ │ + initialize: function(url, size, offset, calculateOffset) { │ │ │ │ │ + this.url = url; │ │ │ │ │ + this.size = size || { │ │ │ │ │ + w: 20, │ │ │ │ │ + h: 20 │ │ │ │ │ + }; │ │ │ │ │ + this.offset = offset || { │ │ │ │ │ + x: -(this.size.w / 2), │ │ │ │ │ + y: -(this.size.h / 2) │ │ │ │ │ + }; │ │ │ │ │ + this.calculateOffset = calculateOffset; │ │ │ │ │ + var id = OpenLayers.Util.createUniqueID("OL_Icon_"); │ │ │ │ │ + this.imageDiv = OpenLayers.Util.createAlphaImageDiv(id) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.erase(); │ │ │ │ │ + OpenLayers.Event.stopObservingElement(this.imageDiv.firstChild); │ │ │ │ │ + this.imageDiv.innerHTML = ""; │ │ │ │ │ + this.imageDiv = null │ │ │ │ │ + }, │ │ │ │ │ + clone: function() { │ │ │ │ │ + return new OpenLayers.Icon(this.url, this.size, this.offset, this.calculateOffset) │ │ │ │ │ + }, │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + if (size != null) { │ │ │ │ │ + this.size = size │ │ │ │ │ + } │ │ │ │ │ + this.draw() │ │ │ │ │ + }, │ │ │ │ │ + setUrl: function(url) { │ │ │ │ │ + if (url != null) { │ │ │ │ │ + this.url = url │ │ │ │ │ + } │ │ │ │ │ + this.draw() │ │ │ │ │ + }, │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, this.size, this.url, "absolute"); │ │ │ │ │ + this.moveTo(px); │ │ │ │ │ + return this.imageDiv │ │ │ │ │ + }, │ │ │ │ │ + erase: function() { │ │ │ │ │ + if (this.imageDiv != null && this.imageDiv.parentNode != null) { │ │ │ │ │ + OpenLayers.Element.remove(this.imageDiv) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, null, null, null, null, null, opacity) │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + if (px != null) { │ │ │ │ │ + this.px = px │ │ │ │ │ + } │ │ │ │ │ + if (this.imageDiv != null) { │ │ │ │ │ + if (this.px == null) { │ │ │ │ │ + this.display(false) │ │ │ │ │ + } else { │ │ │ │ │ + if (this.calculateOffset) { │ │ │ │ │ + this.offset = this.calculateOffset(this.size) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, { │ │ │ │ │ + x: this.px.x + this.offset.x, │ │ │ │ │ + y: this.px.y + this.offset.y │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + display: function(display) { │ │ │ │ │ + this.imageDiv.style.display = display ? "" : "none" │ │ │ │ │ + }, │ │ │ │ │ + isDrawn: function() { │ │ │ │ │ + var isDrawn = this.imageDiv && this.imageDiv.parentNode && this.imageDiv.parentNode.nodeType != 11; │ │ │ │ │ + return isDrawn │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Icon" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Marker = OpenLayers.Class({ │ │ │ │ │ + icon: null, │ │ │ │ │ + lonlat: null, │ │ │ │ │ + events: null, │ │ │ │ │ + map: null, │ │ │ │ │ + initialize: function(lonlat, icon) { │ │ │ │ │ + this.lonlat = lonlat; │ │ │ │ │ + var newIcon = icon ? icon : OpenLayers.Marker.defaultIcon(); │ │ │ │ │ + if (this.icon == null) { │ │ │ │ │ + this.icon = newIcon │ │ │ │ │ + } else { │ │ │ │ │ + this.icon.url = newIcon.url; │ │ │ │ │ + this.icon.size = newIcon.size; │ │ │ │ │ + this.icon.offset = newIcon.offset; │ │ │ │ │ + this.icon.calculateOffset = newIcon.calculateOffset │ │ │ │ │ + } │ │ │ │ │ + this.events = new OpenLayers.Events(this, this.icon.imageDiv) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.erase(); │ │ │ │ │ + this.map = null; │ │ │ │ │ + this.events.destroy(); │ │ │ │ │ + this.events = null; │ │ │ │ │ + if (this.icon != null) { │ │ │ │ │ + this.icon.destroy(); │ │ │ │ │ + this.icon = null │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + return this.icon.draw(px) │ │ │ │ │ + }, │ │ │ │ │ + erase: function() { │ │ │ │ │ + if (this.icon != null) { │ │ │ │ │ + this.icon.erase() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + if (px != null && this.icon != null) { │ │ │ │ │ + this.icon.moveTo(px) │ │ │ │ │ + } │ │ │ │ │ + this.lonlat = this.map.getLonLatFromLayerPx(px) │ │ │ │ │ + }, │ │ │ │ │ + isDrawn: function() { │ │ │ │ │ + var isDrawn = this.icon && this.icon.isDrawn(); │ │ │ │ │ + return isDrawn │ │ │ │ │ + }, │ │ │ │ │ + onScreen: function() { │ │ │ │ │ + var onScreen = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + var screenBounds = this.map.getExtent(); │ │ │ │ │ + onScreen = screenBounds.containsLonLat(this.lonlat) │ │ │ │ │ + } │ │ │ │ │ + return onScreen │ │ │ │ │ + }, │ │ │ │ │ + inflate: function(inflate) { │ │ │ │ │ + if (this.icon) { │ │ │ │ │ + this.icon.setSize({ │ │ │ │ │ + w: this.icon.size.w * inflate, │ │ │ │ │ + h: this.icon.size.h * inflate │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + this.icon.setOpacity(opacity) │ │ │ │ │ + }, │ │ │ │ │ + setUrl: function(url) { │ │ │ │ │ + this.icon.setUrl(url) │ │ │ │ │ + }, │ │ │ │ │ + display: function(display) { │ │ │ │ │ + this.icon.display(display) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Marker" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Marker.defaultIcon = function() { │ │ │ │ │ + return new OpenLayers.Icon(OpenLayers.Util.getImageLocation("marker.png"), { │ │ │ │ │ + w: 21, │ │ │ │ │ + h: 25 │ │ │ │ │ + }, { │ │ │ │ │ + x: -10.5, │ │ │ │ │ + y: -25 │ │ │ │ │ + }) │ │ │ │ │ +}; │ │ │ │ │ OpenLayers.Strategy = OpenLayers.Class({ │ │ │ │ │ layer: null, │ │ │ │ │ options: null, │ │ │ │ │ active: null, │ │ │ │ │ autoActivate: true, │ │ │ │ │ autoDestroy: true, │ │ │ │ │ initialize: function(options) { │ │ │ │ │ @@ -13346,82 +11426,1860 @@ │ │ │ │ │ this.active = false; │ │ │ │ │ return true │ │ │ │ │ } │ │ │ │ │ return false │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Strategy" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.StyleMap = OpenLayers.Class({ │ │ │ │ │ - styles: null, │ │ │ │ │ - extendDefault: true, │ │ │ │ │ - initialize: function(style, options) { │ │ │ │ │ - this.styles = { │ │ │ │ │ - default: new OpenLayers.Style(OpenLayers.Feature.Vector.style["default"]), │ │ │ │ │ - select: new OpenLayers.Style(OpenLayers.Feature.Vector.style["select"]), │ │ │ │ │ - temporary: new OpenLayers.Style(OpenLayers.Feature.Vector.style["temporary"]), │ │ │ │ │ - delete: new OpenLayers.Style(OpenLayers.Feature.Vector.style["delete"]) │ │ │ │ │ +OpenLayers.Kinetic = OpenLayers.Class({ │ │ │ │ │ + threshold: 0, │ │ │ │ │ + deceleration: .0035, │ │ │ │ │ + nbPoints: 100, │ │ │ │ │ + delay: 200, │ │ │ │ │ + points: undefined, │ │ │ │ │ + timerId: undefined, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options) │ │ │ │ │ + }, │ │ │ │ │ + begin: function() { │ │ │ │ │ + OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ + this.timerId = undefined; │ │ │ │ │ + this.points = [] │ │ │ │ │ + }, │ │ │ │ │ + update: function(xy) { │ │ │ │ │ + this.points.unshift({ │ │ │ │ │ + xy: xy, │ │ │ │ │ + tick: (new Date).getTime() │ │ │ │ │ + }); │ │ │ │ │ + if (this.points.length > this.nbPoints) { │ │ │ │ │ + this.points.pop() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + end: function(xy) { │ │ │ │ │ + var last, now = (new Date).getTime(); │ │ │ │ │ + for (var i = 0, l = this.points.length, point; i < l; i++) { │ │ │ │ │ + point = this.points[i]; │ │ │ │ │ + if (now - point.tick > this.delay) { │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + last = point │ │ │ │ │ + } │ │ │ │ │ + if (!last) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var time = (new Date).getTime() - last.tick; │ │ │ │ │ + var dist = Math.sqrt(Math.pow(xy.x - last.xy.x, 2) + Math.pow(xy.y - last.xy.y, 2)); │ │ │ │ │ + var speed = dist / time; │ │ │ │ │ + if (speed == 0 || speed < this.threshold) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var theta = Math.asin((xy.y - last.xy.y) / dist); │ │ │ │ │ + if (last.xy.x <= xy.x) { │ │ │ │ │ + theta = Math.PI - theta │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + speed: speed, │ │ │ │ │ + theta: theta │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + move: function(info, callback) { │ │ │ │ │ + var v0 = info.speed; │ │ │ │ │ + var fx = Math.cos(info.theta); │ │ │ │ │ + var fy = -Math.sin(info.theta); │ │ │ │ │ + var initialTime = (new Date).getTime(); │ │ │ │ │ + var lastX = 0; │ │ │ │ │ + var lastY = 0; │ │ │ │ │ + var timerCallback = function() { │ │ │ │ │ + if (this.timerId == null) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var t = (new Date).getTime() - initialTime; │ │ │ │ │ + var p = -this.deceleration * Math.pow(t, 2) / 2 + v0 * t; │ │ │ │ │ + var x = p * fx; │ │ │ │ │ + var y = p * fy; │ │ │ │ │ + var args = {}; │ │ │ │ │ + args.end = false; │ │ │ │ │ + var v = -this.deceleration * t + v0; │ │ │ │ │ + if (v <= 0) { │ │ │ │ │ + OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + args.end = true │ │ │ │ │ + } │ │ │ │ │ + args.x = x - lastX; │ │ │ │ │ + args.y = y - lastY; │ │ │ │ │ + lastX = x; │ │ │ │ │ + lastY = y; │ │ │ │ │ + callback(args.x, args.y, args.end) │ │ │ │ │ }; │ │ │ │ │ - if (style instanceof OpenLayers.Style) { │ │ │ │ │ - this.styles["default"] = style; │ │ │ │ │ - this.styles["select"] = style; │ │ │ │ │ - this.styles["temporary"] = style; │ │ │ │ │ - this.styles["delete"] = style │ │ │ │ │ - } else if (typeof style == "object") { │ │ │ │ │ - for (var key in style) { │ │ │ │ │ - if (style[key] instanceof OpenLayers.Style) { │ │ │ │ │ - this.styles[key] = style[key] │ │ │ │ │ - } else if (typeof style[key] == "object") { │ │ │ │ │ - this.styles[key] = new OpenLayers.Style(style[key]) │ │ │ │ │ - } else { │ │ │ │ │ - this.styles["default"] = new OpenLayers.Style(style); │ │ │ │ │ - this.styles["select"] = new OpenLayers.Style(style); │ │ │ │ │ - this.styles["temporary"] = new OpenLayers.Style(style); │ │ │ │ │ - this.styles["delete"] = new OpenLayers.Style(style); │ │ │ │ │ + this.timerId = OpenLayers.Animation.start(OpenLayers.Function.bind(timerCallback, this)) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Kinetic" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer = OpenLayers.Class({ │ │ │ │ │ + id: null, │ │ │ │ │ + name: null, │ │ │ │ │ + div: null, │ │ │ │ │ + opacity: 1, │ │ │ │ │ + alwaysInRange: null, │ │ │ │ │ + RESOLUTION_PROPERTIES: ["scales", "resolutions", "maxScale", "minScale", "maxResolution", "minResolution", "numZoomLevels", "maxZoomLevel"], │ │ │ │ │ + events: null, │ │ │ │ │ + map: null, │ │ │ │ │ + isBaseLayer: false, │ │ │ │ │ + alpha: false, │ │ │ │ │ + displayInLayerSwitcher: true, │ │ │ │ │ + visibility: true, │ │ │ │ │ + attribution: null, │ │ │ │ │ + inRange: false, │ │ │ │ │ + imageSize: null, │ │ │ │ │ + options: null, │ │ │ │ │ + eventListeners: null, │ │ │ │ │ + gutter: 0, │ │ │ │ │ + projection: null, │ │ │ │ │ + units: null, │ │ │ │ │ + scales: null, │ │ │ │ │ + resolutions: null, │ │ │ │ │ + maxExtent: null, │ │ │ │ │ + minExtent: null, │ │ │ │ │ + maxResolution: null, │ │ │ │ │ + minResolution: null, │ │ │ │ │ + numZoomLevels: null, │ │ │ │ │ + minScale: null, │ │ │ │ │ + maxScale: null, │ │ │ │ │ + displayOutsideMaxExtent: false, │ │ │ │ │ + wrapDateLine: false, │ │ │ │ │ + metadata: null, │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + this.metadata = {}; │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + if (this.alwaysInRange != null) { │ │ │ │ │ + options.alwaysInRange = this.alwaysInRange │ │ │ │ │ + } │ │ │ │ │ + this.addOptions(options); │ │ │ │ │ + this.name = name; │ │ │ │ │ + if (this.id == null) { │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ + this.div = OpenLayers.Util.createDiv(this.id); │ │ │ │ │ + this.div.style.width = "100%"; │ │ │ │ │ + this.div.style.height = "100%"; │ │ │ │ │ + this.div.dir = "ltr"; │ │ │ │ │ + this.events = new OpenLayers.Events(this, this.div); │ │ │ │ │ + if (this.eventListeners instanceof Object) { │ │ │ │ │ + this.events.on(this.eventListeners) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function(setNewBaseLayer) { │ │ │ │ │ + if (setNewBaseLayer == null) { │ │ │ │ │ + setNewBaseLayer = true │ │ │ │ │ + } │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.removeLayer(this, setNewBaseLayer) │ │ │ │ │ + } │ │ │ │ │ + this.projection = null; │ │ │ │ │ + this.map = null; │ │ │ │ │ + this.name = null; │ │ │ │ │ + this.div = null; │ │ │ │ │ + this.options = null; │ │ │ │ │ + if (this.events) { │ │ │ │ │ + if (this.eventListeners) { │ │ │ │ │ + this.events.un(this.eventListeners) │ │ │ │ │ + } │ │ │ │ │ + this.events.destroy() │ │ │ │ │ + } │ │ │ │ │ + this.eventListeners = null; │ │ │ │ │ + this.events = null │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer(this.name, this.getOptions()) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.applyDefaults(obj, this); │ │ │ │ │ + obj.map = null; │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + getOptions: function() { │ │ │ │ │ + var options = {}; │ │ │ │ │ + for (var o in this.options) { │ │ │ │ │ + options[o] = this[o] │ │ │ │ │ + } │ │ │ │ │ + return options │ │ │ │ │ + }, │ │ │ │ │ + setName: function(newName) { │ │ │ │ │ + if (newName != this.name) { │ │ │ │ │ + this.name = newName; │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "name" │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addOptions: function(newOptions, reinitialize) { │ │ │ │ │ + if (this.options == null) { │ │ │ │ │ + this.options = {} │ │ │ │ │ + } │ │ │ │ │ + if (newOptions) { │ │ │ │ │ + if (typeof newOptions.projection == "string") { │ │ │ │ │ + newOptions.projection = new OpenLayers.Projection(newOptions.projection) │ │ │ │ │ + } │ │ │ │ │ + if (newOptions.projection) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(newOptions, OpenLayers.Projection.defaults[newOptions.projection.getCode()]) │ │ │ │ │ + } │ │ │ │ │ + if (newOptions.maxExtent && !(newOptions.maxExtent instanceof OpenLayers.Bounds)) { │ │ │ │ │ + newOptions.maxExtent = new OpenLayers.Bounds(newOptions.maxExtent) │ │ │ │ │ + } │ │ │ │ │ + if (newOptions.minExtent && !(newOptions.minExtent instanceof OpenLayers.Bounds)) { │ │ │ │ │ + newOptions.minExtent = new OpenLayers.Bounds(newOptions.minExtent) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.extend(this.options, newOptions); │ │ │ │ │ + OpenLayers.Util.extend(this, newOptions); │ │ │ │ │ + if (this.projection && this.projection.getUnits()) { │ │ │ │ │ + this.units = this.projection.getUnits() │ │ │ │ │ + } │ │ │ │ │ + if (this.map) { │ │ │ │ │ + var resolution = this.map.getResolution(); │ │ │ │ │ + var properties = this.RESOLUTION_PROPERTIES.concat(["projection", "units", "minExtent", "maxExtent"]); │ │ │ │ │ + for (var o in newOptions) { │ │ │ │ │ + if (newOptions.hasOwnProperty(o) && OpenLayers.Util.indexOf(properties, o) >= 0) { │ │ │ │ │ + this.initResolutions(); │ │ │ │ │ + if (reinitialize && this.map.baseLayer === this) { │ │ │ │ │ + this.map.setCenter(this.map.getCenter(), this.map.getZoomForResolution(resolution), false, true); │ │ │ │ │ + this.map.events.triggerEvent("changebaselayer", { │ │ │ │ │ + layer: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ break │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Util.extend(this, options) │ │ │ │ │ + }, │ │ │ │ │ + onMapResize: function() {}, │ │ │ │ │ + redraw: function() { │ │ │ │ │ + var redrawn = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.inRange = this.calculateInRange(); │ │ │ │ │ + var extent = this.getExtent(); │ │ │ │ │ + if (extent && this.inRange && this.visibility) { │ │ │ │ │ + var zoomChanged = true; │ │ │ │ │ + this.moveTo(extent, zoomChanged, false); │ │ │ │ │ + this.events.triggerEvent("moveend", { │ │ │ │ │ + zoomChanged: zoomChanged │ │ │ │ │ + }); │ │ │ │ │ + redrawn = true │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return redrawn │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + var display = this.visibility; │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + display = display && this.inRange │ │ │ │ │ + } │ │ │ │ │ + this.display(display) │ │ │ │ │ + }, │ │ │ │ │ + moveByPx: function(dx, dy) {}, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + if (this.map == null) { │ │ │ │ │ + this.map = map; │ │ │ │ │ + this.maxExtent = this.maxExtent || this.map.maxExtent; │ │ │ │ │ + this.minExtent = this.minExtent || this.map.minExtent; │ │ │ │ │ + this.projection = this.projection || this.map.projection; │ │ │ │ │ + if (typeof this.projection == "string") { │ │ │ │ │ + this.projection = new OpenLayers.Projection(this.projection) │ │ │ │ │ + } │ │ │ │ │ + this.units = this.projection.getUnits() || this.units || this.map.units; │ │ │ │ │ + this.initResolutions(); │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + this.inRange = this.calculateInRange(); │ │ │ │ │ + var show = this.visibility && this.inRange; │ │ │ │ │ + this.div.style.display = show ? "" : "none" │ │ │ │ │ + } │ │ │ │ │ + this.setTileSize() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + afterAdd: function() {}, │ │ │ │ │ + removeMap: function(map) {}, │ │ │ │ │ + getImageSize: function(bounds) { │ │ │ │ │ + return this.imageSize || this.tileSize │ │ │ │ │ + }, │ │ │ │ │ + setTileSize: function(size) { │ │ │ │ │ + var tileSize = size ? size : this.tileSize ? this.tileSize : this.map.getTileSize(); │ │ │ │ │ + this.tileSize = tileSize; │ │ │ │ │ + if (this.gutter) { │ │ │ │ │ + this.imageSize = new OpenLayers.Size(tileSize.w + 2 * this.gutter, tileSize.h + 2 * this.gutter) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getVisibility: function() { │ │ │ │ │ + return this.visibility │ │ │ │ │ + }, │ │ │ │ │ + setVisibility: function(visibility) { │ │ │ │ │ + if (visibility != this.visibility) { │ │ │ │ │ + this.visibility = visibility; │ │ │ │ │ + this.display(visibility); │ │ │ │ │ + this.redraw(); │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "visibility" │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("visibilitychanged") │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + display: function(display) { │ │ │ │ │ + if (display != (this.div.style.display != "none")) { │ │ │ │ │ + this.div.style.display = display && this.calculateInRange() ? "block" : "none" │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + calculateInRange: function() { │ │ │ │ │ + var inRange = false; │ │ │ │ │ + if (this.alwaysInRange) { │ │ │ │ │ + inRange = true │ │ │ │ │ + } else { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + var resolution = this.map.getResolution(); │ │ │ │ │ + inRange = resolution >= this.minResolution && resolution <= this.maxResolution │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return inRange │ │ │ │ │ + }, │ │ │ │ │ + setIsBaseLayer: function(isBaseLayer) { │ │ │ │ │ + if (isBaseLayer != this.isBaseLayer) { │ │ │ │ │ + this.isBaseLayer = isBaseLayer; │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changebaselayer", { │ │ │ │ │ + layer: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + initResolutions: function() { │ │ │ │ │ + var i, len, p; │ │ │ │ │ + var props = {}, │ │ │ │ │ + alwaysInRange = true; │ │ │ │ │ + for (i = 0, len = this.RESOLUTION_PROPERTIES.length; i < len; i++) { │ │ │ │ │ + p = this.RESOLUTION_PROPERTIES[i]; │ │ │ │ │ + props[p] = this.options[p]; │ │ │ │ │ + if (alwaysInRange && this.options[p]) { │ │ │ │ │ + alwaysInRange = false │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.options.alwaysInRange == null) { │ │ │ │ │ + this.alwaysInRange = alwaysInRange │ │ │ │ │ + } │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + props.resolutions = this.resolutionsFromScales(props.scales) │ │ │ │ │ + } │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + props.resolutions = this.calculateResolutions(props) │ │ │ │ │ + } │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + for (i = 0, len = this.RESOLUTION_PROPERTIES.length; i < len; i++) { │ │ │ │ │ + p = this.RESOLUTION_PROPERTIES[i]; │ │ │ │ │ + props[p] = this.options[p] != null ? this.options[p] : this.map[p] │ │ │ │ │ + } │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + props.resolutions = this.resolutionsFromScales(props.scales) │ │ │ │ │ + } │ │ │ │ │ + if (props.resolutions == null) { │ │ │ │ │ + props.resolutions = this.calculateResolutions(props) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var maxResolution; │ │ │ │ │ + if (this.options.maxResolution && this.options.maxResolution !== "auto") { │ │ │ │ │ + maxResolution = this.options.maxResolution │ │ │ │ │ + } │ │ │ │ │ + if (this.options.minScale) { │ │ │ │ │ + maxResolution = OpenLayers.Util.getResolutionFromScale(this.options.minScale, this.units) │ │ │ │ │ + } │ │ │ │ │ + var minResolution; │ │ │ │ │ + if (this.options.minResolution && this.options.minResolution !== "auto") { │ │ │ │ │ + minResolution = this.options.minResolution │ │ │ │ │ + } │ │ │ │ │ + if (this.options.maxScale) { │ │ │ │ │ + minResolution = OpenLayers.Util.getResolutionFromScale(this.options.maxScale, this.units) │ │ │ │ │ + } │ │ │ │ │ + if (props.resolutions) { │ │ │ │ │ + props.resolutions.sort(function(a, b) { │ │ │ │ │ + return b - a │ │ │ │ │ + }); │ │ │ │ │ + if (!maxResolution) { │ │ │ │ │ + maxResolution = props.resolutions[0] │ │ │ │ │ + } │ │ │ │ │ + if (!minResolution) { │ │ │ │ │ + var lastIdx = props.resolutions.length - 1; │ │ │ │ │ + minResolution = props.resolutions[lastIdx] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.resolutions = props.resolutions; │ │ │ │ │ + if (this.resolutions) { │ │ │ │ │ + len = this.resolutions.length; │ │ │ │ │ + this.scales = new Array(len); │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + this.scales[i] = OpenLayers.Util.getScaleFromResolution(this.resolutions[i], this.units) │ │ │ │ │ + } │ │ │ │ │ + this.numZoomLevels = len │ │ │ │ │ + } │ │ │ │ │ + this.minResolution = minResolution; │ │ │ │ │ + if (minResolution) { │ │ │ │ │ + this.maxScale = OpenLayers.Util.getScaleFromResolution(minResolution, this.units) │ │ │ │ │ + } │ │ │ │ │ + this.maxResolution = maxResolution; │ │ │ │ │ + if (maxResolution) { │ │ │ │ │ + this.minScale = OpenLayers.Util.getScaleFromResolution(maxResolution, this.units) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + resolutionsFromScales: function(scales) { │ │ │ │ │ + if (scales == null) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var resolutions, i, len; │ │ │ │ │ + len = scales.length; │ │ │ │ │ + resolutions = new Array(len); │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + resolutions[i] = OpenLayers.Util.getResolutionFromScale(scales[i], this.units) │ │ │ │ │ + } │ │ │ │ │ + return resolutions │ │ │ │ │ + }, │ │ │ │ │ + calculateResolutions: function(props) { │ │ │ │ │ + var viewSize, wRes, hRes; │ │ │ │ │ + var maxResolution = props.maxResolution; │ │ │ │ │ + if (props.minScale != null) { │ │ │ │ │ + maxResolution = OpenLayers.Util.getResolutionFromScale(props.minScale, this.units) │ │ │ │ │ + } else if (maxResolution == "auto" && this.maxExtent != null) { │ │ │ │ │ + viewSize = this.map.getSize(); │ │ │ │ │ + wRes = this.maxExtent.getWidth() / viewSize.w; │ │ │ │ │ + hRes = this.maxExtent.getHeight() / viewSize.h; │ │ │ │ │ + maxResolution = Math.max(wRes, hRes) │ │ │ │ │ + } │ │ │ │ │ + var minResolution = props.minResolution; │ │ │ │ │ + if (props.maxScale != null) { │ │ │ │ │ + minResolution = OpenLayers.Util.getResolutionFromScale(props.maxScale, this.units) │ │ │ │ │ + } else if (props.minResolution == "auto" && this.minExtent != null) { │ │ │ │ │ + viewSize = this.map.getSize(); │ │ │ │ │ + wRes = this.minExtent.getWidth() / viewSize.w; │ │ │ │ │ + hRes = this.minExtent.getHeight() / viewSize.h; │ │ │ │ │ + minResolution = Math.max(wRes, hRes) │ │ │ │ │ + } │ │ │ │ │ + if (typeof maxResolution !== "number" && typeof minResolution !== "number" && this.maxExtent != null) { │ │ │ │ │ + var tileSize = this.map.getTileSize(); │ │ │ │ │ + maxResolution = Math.max(this.maxExtent.getWidth() / tileSize.w, this.maxExtent.getHeight() / tileSize.h) │ │ │ │ │ + } │ │ │ │ │ + var maxZoomLevel = props.maxZoomLevel; │ │ │ │ │ + var numZoomLevels = props.numZoomLevels; │ │ │ │ │ + if (typeof minResolution === "number" && typeof maxResolution === "number" && numZoomLevels === undefined) { │ │ │ │ │ + var ratio = maxResolution / minResolution; │ │ │ │ │ + numZoomLevels = Math.floor(Math.log(ratio) / Math.log(2)) + 1 │ │ │ │ │ + } else if (numZoomLevels === undefined && maxZoomLevel != null) { │ │ │ │ │ + numZoomLevels = maxZoomLevel + 1 │ │ │ │ │ + } │ │ │ │ │ + if (typeof numZoomLevels !== "number" || numZoomLevels <= 0 || typeof maxResolution !== "number" && typeof minResolution !== "number") { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var resolutions = new Array(numZoomLevels); │ │ │ │ │ + var base = 2; │ │ │ │ │ + if (typeof minResolution == "number" && typeof maxResolution == "number") { │ │ │ │ │ + base = Math.pow(maxResolution / minResolution, 1 / (numZoomLevels - 1)) │ │ │ │ │ + } │ │ │ │ │ + var i; │ │ │ │ │ + if (typeof maxResolution === "number") { │ │ │ │ │ + for (i = 0; i < numZoomLevels; i++) { │ │ │ │ │ + resolutions[i] = maxResolution / Math.pow(base, i) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + for (i = 0; i < numZoomLevels; i++) { │ │ │ │ │ + resolutions[numZoomLevels - 1 - i] = minResolution * Math.pow(base, i) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return resolutions │ │ │ │ │ + }, │ │ │ │ │ + getResolution: function() { │ │ │ │ │ + var zoom = this.map.getZoom(); │ │ │ │ │ + return this.getResolutionForZoom(zoom) │ │ │ │ │ + }, │ │ │ │ │ + getExtent: function() { │ │ │ │ │ + return this.map.calculateBounds() │ │ │ │ │ + }, │ │ │ │ │ + getZoomForExtent: function(extent, closest) { │ │ │ │ │ + var viewSize = this.map.getSize(); │ │ │ │ │ + var idealResolution = Math.max(extent.getWidth() / viewSize.w, extent.getHeight() / viewSize.h); │ │ │ │ │ + return this.getZoomForResolution(idealResolution, closest) │ │ │ │ │ + }, │ │ │ │ │ + getDataExtent: function() {}, │ │ │ │ │ + getResolutionForZoom: function(zoom) { │ │ │ │ │ + zoom = Math.max(0, Math.min(zoom, this.resolutions.length - 1)); │ │ │ │ │ + var resolution; │ │ │ │ │ + if (this.map.fractionalZoom) { │ │ │ │ │ + var low = Math.floor(zoom); │ │ │ │ │ + var high = Math.ceil(zoom); │ │ │ │ │ + resolution = this.resolutions[low] - (zoom - low) * (this.resolutions[low] - this.resolutions[high]) │ │ │ │ │ + } else { │ │ │ │ │ + resolution = this.resolutions[Math.round(zoom)] │ │ │ │ │ + } │ │ │ │ │ + return resolution │ │ │ │ │ + }, │ │ │ │ │ + getZoomForResolution: function(resolution, closest) { │ │ │ │ │ + var zoom, i, len; │ │ │ │ │ + if (this.map.fractionalZoom) { │ │ │ │ │ + var lowZoom = 0; │ │ │ │ │ + var highZoom = this.resolutions.length - 1; │ │ │ │ │ + var highRes = this.resolutions[lowZoom]; │ │ │ │ │ + var lowRes = this.resolutions[highZoom]; │ │ │ │ │ + var res; │ │ │ │ │ + for (i = 0, len = this.resolutions.length; i < len; ++i) { │ │ │ │ │ + res = this.resolutions[i]; │ │ │ │ │ + if (res >= resolution) { │ │ │ │ │ + highRes = res; │ │ │ │ │ + lowZoom = i │ │ │ │ │ + } │ │ │ │ │ + if (res <= resolution) { │ │ │ │ │ + lowRes = res; │ │ │ │ │ + highZoom = i; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var dRes = highRes - lowRes; │ │ │ │ │ + if (dRes > 0) { │ │ │ │ │ + zoom = lowZoom + (highRes - resolution) / dRes │ │ │ │ │ + } else { │ │ │ │ │ + zoom = lowZoom │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + var diff; │ │ │ │ │ + var minDiff = Number.POSITIVE_INFINITY; │ │ │ │ │ + for (i = 0, len = this.resolutions.length; i < len; i++) { │ │ │ │ │ + if (closest) { │ │ │ │ │ + diff = Math.abs(this.resolutions[i] - resolution); │ │ │ │ │ + if (diff > minDiff) { │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + minDiff = diff │ │ │ │ │ + } else { │ │ │ │ │ + if (this.resolutions[i] < resolution) { │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + zoom = Math.max(0, i - 1) │ │ │ │ │ + } │ │ │ │ │ + return zoom │ │ │ │ │ + }, │ │ │ │ │ + getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ + var lonlat = null; │ │ │ │ │ + var map = this.map; │ │ │ │ │ + if (viewPortPx != null && map.minPx) { │ │ │ │ │ + var res = map.getResolution(); │ │ │ │ │ + var maxExtent = map.getMaxExtent({ │ │ │ │ │ + restricted: true │ │ │ │ │ + }); │ │ │ │ │ + var lon = (viewPortPx.x - map.minPx.x) * res + maxExtent.left; │ │ │ │ │ + var lat = (map.minPx.y - viewPortPx.y) * res + maxExtent.top; │ │ │ │ │ + lonlat = new OpenLayers.LonLat(lon, lat); │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + lonlat = lonlat.wrapDateLine(this.maxExtent) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return lonlat │ │ │ │ │ + }, │ │ │ │ │ + getViewPortPxFromLonLat: function(lonlat, resolution) { │ │ │ │ │ + var px = null; │ │ │ │ │ + if (lonlat != null) { │ │ │ │ │ + resolution = resolution || this.map.getResolution(); │ │ │ │ │ + var extent = this.map.calculateBounds(null, resolution); │ │ │ │ │ + px = new OpenLayers.Pixel(1 / resolution * (lonlat.lon - extent.left), 1 / resolution * (extent.top - lonlat.lat)) │ │ │ │ │ + } │ │ │ │ │ + return px │ │ │ │ │ + }, │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + if (opacity != this.opacity) { │ │ │ │ │ + this.opacity = opacity; │ │ │ │ │ + var childNodes = this.div.childNodes; │ │ │ │ │ + for (var i = 0, len = childNodes.length; i < len; ++i) { │ │ │ │ │ + var element = childNodes[i].firstChild || childNodes[i]; │ │ │ │ │ + var lastChild = childNodes[i].lastChild; │ │ │ │ │ + if (lastChild && lastChild.nodeName.toLowerCase() === "iframe") { │ │ │ │ │ + element = lastChild.parentNode │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.modifyDOMElement(element, null, null, null, null, null, null, opacity) │ │ │ │ │ + } │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "opacity" │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getZIndex: function() { │ │ │ │ │ + return this.div.style.zIndex │ │ │ │ │ + }, │ │ │ │ │ + setZIndex: function(zIndex) { │ │ │ │ │ + this.div.style.zIndex = zIndex │ │ │ │ │ + }, │ │ │ │ │ + adjustBounds: function(bounds) { │ │ │ │ │ + if (this.gutter) { │ │ │ │ │ + var mapGutter = this.gutter * this.map.getResolution(); │ │ │ │ │ + bounds = new OpenLayers.Bounds(bounds.left - mapGutter, bounds.bottom - mapGutter, bounds.right + mapGutter, bounds.top + mapGutter) │ │ │ │ │ + } │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + var wrappingOptions = { │ │ │ │ │ + rightTolerance: this.getResolution(), │ │ │ │ │ + leftTolerance: this.getResolution() │ │ │ │ │ + }; │ │ │ │ │ + bounds = bounds.wrapDateLine(this.maxExtent, wrappingOptions) │ │ │ │ │ + } │ │ │ │ │ + return bounds │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.HTTPRequest = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ + URL_HASH_FACTOR: (Math.sqrt(5) - 1) / 2, │ │ │ │ │ + url: null, │ │ │ │ │ + params: null, │ │ │ │ │ + reproject: false, │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ + this.url = url; │ │ │ │ │ + if (!this.params) { │ │ │ │ │ + this.params = OpenLayers.Util.extend({}, params) │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - for (var key in this.styles) { │ │ │ │ │ - this.styles[key].destroy() │ │ │ │ │ + this.url = null; │ │ │ │ │ + this.params = null; │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.HTTPRequest(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - this.styles = null │ │ │ │ │ + obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - createSymbolizer: function(feature, intent) { │ │ │ │ │ - if (!feature) { │ │ │ │ │ - feature = new OpenLayers.Feature.Vector │ │ │ │ │ + setUrl: function(newUrl) { │ │ │ │ │ + this.url = newUrl │ │ │ │ │ + }, │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + this.params = OpenLayers.Util.extend(this.params, newParams); │ │ │ │ │ + var ret = this.redraw(); │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "params" │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - if (!this.styles[intent]) { │ │ │ │ │ - intent = "default" │ │ │ │ │ + return ret │ │ │ │ │ + }, │ │ │ │ │ + redraw: function(force) { │ │ │ │ │ + if (force) { │ │ │ │ │ + return this.mergeNewParams({ │ │ │ │ │ + _olSalt: Math.random() │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + return OpenLayers.Layer.prototype.redraw.apply(this, []) │ │ │ │ │ } │ │ │ │ │ - feature.renderIntent = intent; │ │ │ │ │ - var defaultSymbolizer = {}; │ │ │ │ │ - if (this.extendDefault && intent != "default") { │ │ │ │ │ - defaultSymbolizer = this.styles["default"].createSymbolizer(feature) │ │ │ │ │ + }, │ │ │ │ │ + selectUrl: function(paramString, urls) { │ │ │ │ │ + var product = 1; │ │ │ │ │ + for (var i = 0, len = paramString.length; i < len; i++) { │ │ │ │ │ + product *= paramString.charCodeAt(i) * this.URL_HASH_FACTOR; │ │ │ │ │ + product -= Math.floor(product) │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Util.extend(defaultSymbolizer, this.styles[intent].createSymbolizer(feature)) │ │ │ │ │ + return urls[Math.floor(product * urls.length)] │ │ │ │ │ }, │ │ │ │ │ - addUniqueValueRules: function(renderIntent, property, symbolizers, context) { │ │ │ │ │ - var rules = []; │ │ │ │ │ - for (var value in symbolizers) { │ │ │ │ │ - rules.push(new OpenLayers.Rule({ │ │ │ │ │ - symbolizer: symbolizers[value], │ │ │ │ │ - context: context, │ │ │ │ │ - filter: new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.EQUAL_TO, │ │ │ │ │ - property: property, │ │ │ │ │ - value: value │ │ │ │ │ + getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ + var url = altUrl || this.url; │ │ │ │ │ + var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ + allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ + var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(paramsString, url) │ │ │ │ │ + } │ │ │ │ │ + var urlParams = OpenLayers.Util.upperCaseObject(OpenLayers.Util.getParameters(url)); │ │ │ │ │ + for (var key in allParams) { │ │ │ │ │ + if (key.toUpperCase() in urlParams) { │ │ │ │ │ + delete allParams[key] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + return OpenLayers.Util.urlAppend(url, paramsString) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.HTTPRequest" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Tile.Image = OpenLayers.Class(OpenLayers.Tile, { │ │ │ │ │ + url: null, │ │ │ │ │ + imgDiv: null, │ │ │ │ │ + frame: null, │ │ │ │ │ + imageReloadAttempts: null, │ │ │ │ │ + layerAlphaHack: null, │ │ │ │ │ + asyncRequestId: null, │ │ │ │ │ + maxGetUrlLength: null, │ │ │ │ │ + canvasContext: null, │ │ │ │ │ + crossOriginKeyword: null, │ │ │ │ │ + initialize: function(layer, position, bounds, url, size, options) { │ │ │ │ │ + OpenLayers.Tile.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.url = url; │ │ │ │ │ + this.layerAlphaHack = this.layer.alpha && OpenLayers.Util.alphaHack(); │ │ │ │ │ + if (this.maxGetUrlLength != null || this.layer.gutter || this.layerAlphaHack) { │ │ │ │ │ + this.frame = document.createElement("div"); │ │ │ │ │ + this.frame.style.position = "absolute"; │ │ │ │ │ + this.frame.style.overflow = "hidden" │ │ │ │ │ + } │ │ │ │ │ + if (this.maxGetUrlLength != null) { │ │ │ │ │ + OpenLayers.Util.extend(this, OpenLayers.Tile.Image.IFrame) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.imgDiv) { │ │ │ │ │ + this.clear(); │ │ │ │ │ + this.imgDiv = null; │ │ │ │ │ + this.frame = null │ │ │ │ │ + } │ │ │ │ │ + this.asyncRequestId = null; │ │ │ │ │ + OpenLayers.Tile.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + var shouldDraw = OpenLayers.Tile.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (shouldDraw) { │ │ │ │ │ + if (this.layer != this.layer.map.baseLayer && this.layer.reproject) { │ │ │ │ │ + this.bounds = this.getBoundsFromBaseLayer(this.position) │ │ │ │ │ + } │ │ │ │ │ + if (this.isLoading) { │ │ │ │ │ + this._loadEvent = "reload" │ │ │ │ │ + } else { │ │ │ │ │ + this.isLoading = true; │ │ │ │ │ + this._loadEvent = "loadstart" │ │ │ │ │ + } │ │ │ │ │ + this.renderTile(); │ │ │ │ │ + this.positionTile() │ │ │ │ │ + } else if (shouldDraw === false) { │ │ │ │ │ + this.unload() │ │ │ │ │ + } │ │ │ │ │ + return shouldDraw │ │ │ │ │ + }, │ │ │ │ │ + renderTile: function() { │ │ │ │ │ + if (this.layer.async) { │ │ │ │ │ + var id = this.asyncRequestId = (this.asyncRequestId || 0) + 1; │ │ │ │ │ + this.layer.getURLasync(this.bounds, function(url) { │ │ │ │ │ + if (id == this.asyncRequestId) { │ │ │ │ │ + this.url = url; │ │ │ │ │ + this.initImage() │ │ │ │ │ + } │ │ │ │ │ + }, this) │ │ │ │ │ + } else { │ │ │ │ │ + this.url = this.layer.getURL(this.bounds); │ │ │ │ │ + this.initImage() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + positionTile: function() { │ │ │ │ │ + var style = this.getTile().style, │ │ │ │ │ + size = this.frame ? this.size : this.layer.getImageSize(this.bounds), │ │ │ │ │ + ratio = 1; │ │ │ │ │ + if (this.layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ + ratio = this.layer.getServerResolution() / this.layer.map.getResolution() │ │ │ │ │ + } │ │ │ │ │ + style.left = this.position.x + "px"; │ │ │ │ │ + style.top = this.position.y + "px"; │ │ │ │ │ + style.width = Math.round(ratio * size.w) + "px"; │ │ │ │ │ + style.height = Math.round(ratio * size.h) + "px" │ │ │ │ │ + }, │ │ │ │ │ + clear: function() { │ │ │ │ │ + OpenLayers.Tile.prototype.clear.apply(this, arguments); │ │ │ │ │ + var img = this.imgDiv; │ │ │ │ │ + if (img) { │ │ │ │ │ + var tile = this.getTile(); │ │ │ │ │ + if (tile.parentNode === this.layer.div) { │ │ │ │ │ + this.layer.div.removeChild(tile) │ │ │ │ │ + } │ │ │ │ │ + this.setImgSrc(); │ │ │ │ │ + if (this.layerAlphaHack === true) { │ │ │ │ │ + img.style.filter = "" │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.removeClass(img, "olImageLoadError") │ │ │ │ │ + } │ │ │ │ │ + this.canvasContext = null │ │ │ │ │ + }, │ │ │ │ │ + getImage: function() { │ │ │ │ │ + if (!this.imgDiv) { │ │ │ │ │ + this.imgDiv = OpenLayers.Tile.Image.IMAGE.cloneNode(false); │ │ │ │ │ + var style = this.imgDiv.style; │ │ │ │ │ + if (this.frame) { │ │ │ │ │ + var left = 0, │ │ │ │ │ + top = 0; │ │ │ │ │ + if (this.layer.gutter) { │ │ │ │ │ + left = this.layer.gutter / this.layer.tileSize.w * 100; │ │ │ │ │ + top = this.layer.gutter / this.layer.tileSize.h * 100 │ │ │ │ │ + } │ │ │ │ │ + style.left = -left + "%"; │ │ │ │ │ + style.top = -top + "%"; │ │ │ │ │ + style.width = 2 * left + 100 + "%"; │ │ │ │ │ + style.height = 2 * top + 100 + "%" │ │ │ │ │ + } │ │ │ │ │ + style.visibility = "hidden"; │ │ │ │ │ + style.opacity = 0; │ │ │ │ │ + if (this.layer.opacity < 1) { │ │ │ │ │ + style.filter = "alpha(opacity=" + this.layer.opacity * 100 + ")" │ │ │ │ │ + } │ │ │ │ │ + style.position = "absolute"; │ │ │ │ │ + if (this.layerAlphaHack) { │ │ │ │ │ + style.paddingTop = style.height; │ │ │ │ │ + style.height = "0"; │ │ │ │ │ + style.width = "100%" │ │ │ │ │ + } │ │ │ │ │ + if (this.frame) { │ │ │ │ │ + this.frame.appendChild(this.imgDiv) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return this.imgDiv │ │ │ │ │ + }, │ │ │ │ │ + setImage: function(img) { │ │ │ │ │ + this.imgDiv = img │ │ │ │ │ + }, │ │ │ │ │ + initImage: function() { │ │ │ │ │ + if (!this.url && !this.imgDiv) { │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("beforeload"); │ │ │ │ │ + this.layer.div.appendChild(this.getTile()); │ │ │ │ │ + this.events.triggerEvent(this._loadEvent); │ │ │ │ │ + var img = this.getImage(); │ │ │ │ │ + var src = img.getAttribute("src") || ""; │ │ │ │ │ + if (this.url && OpenLayers.Util.isEquivalentUrl(src, this.url)) { │ │ │ │ │ + this._loadTimeout = window.setTimeout(OpenLayers.Function.bind(this.onImageLoad, this), 0) │ │ │ │ │ + } else { │ │ │ │ │ + this.stopLoading(); │ │ │ │ │ + if (this.crossOriginKeyword) { │ │ │ │ │ + img.removeAttribute("crossorigin") │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.observe(img, "load", OpenLayers.Function.bind(this.onImageLoad, this)); │ │ │ │ │ + OpenLayers.Event.observe(img, "error", OpenLayers.Function.bind(this.onImageError, this)); │ │ │ │ │ + this.imageReloadAttempts = 0; │ │ │ │ │ + this.setImgSrc(this.url) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setImgSrc: function(url) { │ │ │ │ │ + var img = this.imgDiv; │ │ │ │ │ + if (url) { │ │ │ │ │ + img.style.visibility = "hidden"; │ │ │ │ │ + img.style.opacity = 0; │ │ │ │ │ + if (this.crossOriginKeyword) { │ │ │ │ │ + if (url.substr(0, 5) !== "data:") { │ │ │ │ │ + img.setAttribute("crossorigin", this.crossOriginKeyword) │ │ │ │ │ + } else { │ │ │ │ │ + img.removeAttribute("crossorigin") │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + img.src = url │ │ │ │ │ + } else { │ │ │ │ │ + this.stopLoading(); │ │ │ │ │ + this.imgDiv = null; │ │ │ │ │ + if (img.parentNode) { │ │ │ │ │ + img.parentNode.removeChild(img) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getTile: function() { │ │ │ │ │ + return this.frame ? this.frame : this.getImage() │ │ │ │ │ + }, │ │ │ │ │ + createBackBuffer: function() { │ │ │ │ │ + if (!this.imgDiv || this.isLoading) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var backBuffer; │ │ │ │ │ + if (this.frame) { │ │ │ │ │ + backBuffer = this.frame.cloneNode(false); │ │ │ │ │ + backBuffer.appendChild(this.imgDiv) │ │ │ │ │ + } else { │ │ │ │ │ + backBuffer = this.imgDiv │ │ │ │ │ + } │ │ │ │ │ + this.imgDiv = null; │ │ │ │ │ + return backBuffer │ │ │ │ │ + }, │ │ │ │ │ + onImageLoad: function() { │ │ │ │ │ + var img = this.imgDiv; │ │ │ │ │ + this.stopLoading(); │ │ │ │ │ + img.style.visibility = "inherit"; │ │ │ │ │ + img.style.opacity = this.layer.opacity; │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + this.canvasContext = null; │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + if (this.layerAlphaHack === true) { │ │ │ │ │ + img.style.filter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(src='" + img.src + "', sizingMethod='scale')" │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + onImageError: function() { │ │ │ │ │ + var img = this.imgDiv; │ │ │ │ │ + if (img.src != null) { │ │ │ │ │ + this.imageReloadAttempts++; │ │ │ │ │ + if (this.imageReloadAttempts <= OpenLayers.IMAGE_RELOAD_ATTEMPTS) { │ │ │ │ │ + this.setImgSrc(this.layer.getURL(this.bounds)) │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Element.addClass(img, "olImageLoadError"); │ │ │ │ │ + this.events.triggerEvent("loaderror"); │ │ │ │ │ + this.onImageLoad() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + stopLoading: function() { │ │ │ │ │ + OpenLayers.Event.stopObservingElement(this.imgDiv); │ │ │ │ │ + window.clearTimeout(this._loadTimeout); │ │ │ │ │ + delete this._loadTimeout │ │ │ │ │ + }, │ │ │ │ │ + getCanvasContext: function() { │ │ │ │ │ + if (OpenLayers.CANVAS_SUPPORTED && this.imgDiv && !this.isLoading) { │ │ │ │ │ + if (!this.canvasContext) { │ │ │ │ │ + var canvas = document.createElement("canvas"); │ │ │ │ │ + canvas.width = this.size.w; │ │ │ │ │ + canvas.height = this.size.h; │ │ │ │ │ + this.canvasContext = canvas.getContext("2d"); │ │ │ │ │ + this.canvasContext.drawImage(this.imgDiv, 0, 0) │ │ │ │ │ + } │ │ │ │ │ + return this.canvasContext │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Tile.Image" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Tile.Image.IMAGE = function() { │ │ │ │ │ + var img = new Image; │ │ │ │ │ + img.className = "olTileImage"; │ │ │ │ │ + img.galleryImg = "no"; │ │ │ │ │ + return img │ │ │ │ │ +}(); │ │ │ │ │ +OpenLayers.Layer.Grid = OpenLayers.Class(OpenLayers.Layer.HTTPRequest, { │ │ │ │ │ + tileSize: null, │ │ │ │ │ + tileOriginCorner: "bl", │ │ │ │ │ + tileOrigin: null, │ │ │ │ │ + tileOptions: null, │ │ │ │ │ + tileClass: OpenLayers.Tile.Image, │ │ │ │ │ + grid: null, │ │ │ │ │ + singleTile: false, │ │ │ │ │ + ratio: 1.5, │ │ │ │ │ + buffer: 0, │ │ │ │ │ + transitionEffect: "resize", │ │ │ │ │ + numLoadingTiles: 0, │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + loading: false, │ │ │ │ │ + backBuffer: null, │ │ │ │ │ + gridResolution: null, │ │ │ │ │ + backBufferResolution: null, │ │ │ │ │ + backBufferLonLat: null, │ │ │ │ │ + backBufferTimerId: null, │ │ │ │ │ + removeBackBufferDelay: null, │ │ │ │ │ + className: null, │ │ │ │ │ + gridLayout: null, │ │ │ │ │ + rowSign: null, │ │ │ │ │ + transitionendEvents: ["transitionend", "webkitTransitionEnd", "otransitionend", "oTransitionEnd"], │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.grid = []; │ │ │ │ │ + this._removeBackBuffer = OpenLayers.Function.bind(this.removeBackBuffer, this); │ │ │ │ │ + this.initProperties(); │ │ │ │ │ + this.rowSign = this.tileOriginCorner.substr(0, 1) === "t" ? 1 : -1 │ │ │ │ │ + }, │ │ │ │ │ + initProperties: function() { │ │ │ │ │ + if (this.options.removeBackBufferDelay === undefined) { │ │ │ │ │ + this.removeBackBufferDelay = this.singleTile ? 0 : 2500 │ │ │ │ │ + } │ │ │ │ │ + if (this.options.className === undefined) { │ │ │ │ │ + this.className = this.singleTile ? "olLayerGridSingleTile" : "olLayerGrid" │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.setMap.call(this, map); │ │ │ │ │ + OpenLayers.Element.addClass(this.div, this.className) │ │ │ │ │ + }, │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + this.removeBackBuffer() │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.removeBackBuffer(); │ │ │ │ │ + this.clearGrid(); │ │ │ │ │ + this.grid = null; │ │ │ │ │ + this.tileSize = null; │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + clearGrid: function() { │ │ │ │ │ + if (this.grid) { │ │ │ │ │ + for (var iRow = 0, len = this.grid.length; iRow < len; iRow++) { │ │ │ │ │ + var row = this.grid[iRow]; │ │ │ │ │ + for (var iCol = 0, clen = row.length; iCol < clen; iCol++) { │ │ │ │ │ + var tile = row[iCol]; │ │ │ │ │ + this.destroyTile(tile) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.grid = []; │ │ │ │ │ + this.gridResolution = null; │ │ │ │ │ + this.gridLayout = null │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addOptions: function(newOptions, reinitialize) { │ │ │ │ │ + var singleTileChanged = newOptions.singleTile !== undefined && newOptions.singleTile !== this.singleTile; │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.addOptions.apply(this, arguments); │ │ │ │ │ + if (this.map && singleTileChanged) { │ │ │ │ │ + this.initProperties(); │ │ │ │ │ + this.clearGrid(); │ │ │ │ │ + this.tileSize = this.options.tileSize; │ │ │ │ │ + this.setTileSize(); │ │ │ │ │ + this.moveTo(null, true) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Grid(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.HTTPRequest.prototype.clone.apply(this, [obj]); │ │ │ │ │ + if (this.tileSize != null) { │ │ │ │ │ + obj.tileSize = this.tileSize.clone() │ │ │ │ │ + } │ │ │ │ │ + obj.grid = []; │ │ │ │ │ + obj.gridResolution = null; │ │ │ │ │ + obj.backBuffer = null; │ │ │ │ │ + obj.backBufferTimerId = null; │ │ │ │ │ + obj.loading = false; │ │ │ │ │ + obj.numLoadingTiles = 0; │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + bounds = bounds || this.map.getExtent(); │ │ │ │ │ + if (bounds != null) { │ │ │ │ │ + var forceReTile = !this.grid.length || zoomChanged; │ │ │ │ │ + var tilesBounds = this.getTilesBounds(); │ │ │ │ │ + var resolution = this.map.getResolution(); │ │ │ │ │ + var serverResolution = this.getServerResolution(resolution); │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + if (forceReTile || !dragging && !tilesBounds.containsBounds(bounds)) { │ │ │ │ │ + if (zoomChanged && this.transitionEffect !== "resize") { │ │ │ │ │ + this.removeBackBuffer() │ │ │ │ │ + } │ │ │ │ │ + if (!zoomChanged || this.transitionEffect === "resize") { │ │ │ │ │ + this.applyBackBuffer(resolution) │ │ │ │ │ + } │ │ │ │ │ + this.initSingleTile(bounds) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + forceReTile = forceReTile || !tilesBounds.intersectsBounds(bounds, { │ │ │ │ │ + worldBounds: this.map.baseLayer.wrapDateLine && this.map.getMaxExtent() │ │ │ │ │ + }); │ │ │ │ │ + if (forceReTile) { │ │ │ │ │ + if (zoomChanged && (this.transitionEffect === "resize" || this.gridResolution === resolution)) { │ │ │ │ │ + this.applyBackBuffer(resolution) │ │ │ │ │ + } │ │ │ │ │ + this.initGriddedTiles(bounds) │ │ │ │ │ + } else { │ │ │ │ │ + this.moveGriddedTiles() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getTileData: function(loc) { │ │ │ │ │ + var data = null, │ │ │ │ │ + x = loc.lon, │ │ │ │ │ + y = loc.lat, │ │ │ │ │ + numRows = this.grid.length; │ │ │ │ │ + if (this.map && numRows) { │ │ │ │ │ + var res = this.map.getResolution(), │ │ │ │ │ + tileWidth = this.tileSize.w, │ │ │ │ │ + tileHeight = this.tileSize.h, │ │ │ │ │ + bounds = this.grid[0][0].bounds, │ │ │ │ │ + left = bounds.left, │ │ │ │ │ + top = bounds.top; │ │ │ │ │ + if (x < left) { │ │ │ │ │ + if (this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + var worldWidth = this.map.getMaxExtent().getWidth(); │ │ │ │ │ + var worldsAway = Math.ceil((left - x) / worldWidth); │ │ │ │ │ + x += worldWidth * worldsAway │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var dtx = (x - left) / (res * tileWidth); │ │ │ │ │ + var dty = (top - y) / (res * tileHeight); │ │ │ │ │ + var col = Math.floor(dtx); │ │ │ │ │ + var row = Math.floor(dty); │ │ │ │ │ + if (row >= 0 && row < numRows) { │ │ │ │ │ + var tile = this.grid[row][col]; │ │ │ │ │ + if (tile) { │ │ │ │ │ + data = { │ │ │ │ │ + tile: tile, │ │ │ │ │ + i: Math.floor((dtx - col) * tileWidth), │ │ │ │ │ + j: Math.floor((dty - row) * tileHeight) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return data │ │ │ │ │ + }, │ │ │ │ │ + destroyTile: function(tile) { │ │ │ │ │ + this.removeTileMonitoringHooks(tile); │ │ │ │ │ + tile.destroy() │ │ │ │ │ + }, │ │ │ │ │ + getServerResolution: function(resolution) { │ │ │ │ │ + var distance = Number.POSITIVE_INFINITY; │ │ │ │ │ + resolution = resolution || this.map.getResolution(); │ │ │ │ │ + if (this.serverResolutions && OpenLayers.Util.indexOf(this.serverResolutions, resolution) === -1) { │ │ │ │ │ + var i, newDistance, newResolution, serverResolution; │ │ │ │ │ + for (i = this.serverResolutions.length - 1; i >= 0; i--) { │ │ │ │ │ + newResolution = this.serverResolutions[i]; │ │ │ │ │ + newDistance = Math.abs(newResolution - resolution); │ │ │ │ │ + if (newDistance > distance) { │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + distance = newDistance; │ │ │ │ │ + serverResolution = newResolution │ │ │ │ │ + } │ │ │ │ │ + resolution = serverResolution │ │ │ │ │ + } │ │ │ │ │ + return resolution │ │ │ │ │ + }, │ │ │ │ │ + getServerZoom: function() { │ │ │ │ │ + var resolution = this.getServerResolution(); │ │ │ │ │ + return this.serverResolutions ? OpenLayers.Util.indexOf(this.serverResolutions, resolution) : this.map.getZoomForResolution(resolution) + (this.zoomOffset || 0) │ │ │ │ │ + }, │ │ │ │ │ + applyBackBuffer: function(resolution) { │ │ │ │ │ + if (this.backBufferTimerId !== null) { │ │ │ │ │ + this.removeBackBuffer() │ │ │ │ │ + } │ │ │ │ │ + var backBuffer = this.backBuffer; │ │ │ │ │ + if (!backBuffer) { │ │ │ │ │ + backBuffer = this.createBackBuffer(); │ │ │ │ │ + if (!backBuffer) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (resolution === this.gridResolution) { │ │ │ │ │ + this.div.insertBefore(backBuffer, this.div.firstChild) │ │ │ │ │ + } else { │ │ │ │ │ + this.map.baseLayer.div.parentNode.insertBefore(backBuffer, this.map.baseLayer.div) │ │ │ │ │ + } │ │ │ │ │ + this.backBuffer = backBuffer; │ │ │ │ │ + var topLeftTileBounds = this.grid[0][0].bounds; │ │ │ │ │ + this.backBufferLonLat = { │ │ │ │ │ + lon: topLeftTileBounds.left, │ │ │ │ │ + lat: topLeftTileBounds.top │ │ │ │ │ + }; │ │ │ │ │ + this.backBufferResolution = this.gridResolution │ │ │ │ │ + } │ │ │ │ │ + var ratio = this.backBufferResolution / resolution; │ │ │ │ │ + var tiles = backBuffer.childNodes, │ │ │ │ │ + tile; │ │ │ │ │ + for (var i = tiles.length - 1; i >= 0; --i) { │ │ │ │ │ + tile = tiles[i]; │ │ │ │ │ + tile.style.top = (ratio * tile._i * tile._h | 0) + "px"; │ │ │ │ │ + tile.style.left = (ratio * tile._j * tile._w | 0) + "px"; │ │ │ │ │ + tile.style.width = Math.round(ratio * tile._w) + "px"; │ │ │ │ │ + tile.style.height = Math.round(ratio * tile._h) + "px" │ │ │ │ │ + } │ │ │ │ │ + var position = this.getViewPortPxFromLonLat(this.backBufferLonLat, resolution); │ │ │ │ │ + var leftOffset = this.map.layerContainerOriginPx.x; │ │ │ │ │ + var topOffset = this.map.layerContainerOriginPx.y; │ │ │ │ │ + backBuffer.style.left = Math.round(position.x - leftOffset) + "px"; │ │ │ │ │ + backBuffer.style.top = Math.round(position.y - topOffset) + "px" │ │ │ │ │ + }, │ │ │ │ │ + createBackBuffer: function() { │ │ │ │ │ + var backBuffer; │ │ │ │ │ + if (this.grid.length > 0) { │ │ │ │ │ + backBuffer = document.createElement("div"); │ │ │ │ │ + backBuffer.id = this.div.id + "_bb"; │ │ │ │ │ + backBuffer.className = "olBackBuffer"; │ │ │ │ │ + backBuffer.style.position = "absolute"; │ │ │ │ │ + var map = this.map; │ │ │ │ │ + backBuffer.style.zIndex = this.transitionEffect === "resize" ? this.getZIndex() - 1 : map.Z_INDEX_BASE.BaseLayer - (map.getNumLayers() - map.getLayerIndex(this)); │ │ │ │ │ + for (var i = 0, lenI = this.grid.length; i < lenI; i++) { │ │ │ │ │ + for (var j = 0, lenJ = this.grid[i].length; j < lenJ; j++) { │ │ │ │ │ + var tile = this.grid[i][j], │ │ │ │ │ + markup = this.grid[i][j].createBackBuffer(); │ │ │ │ │ + if (markup) { │ │ │ │ │ + markup._i = i; │ │ │ │ │ + markup._j = j; │ │ │ │ │ + markup._w = tile.size.w; │ │ │ │ │ + markup._h = tile.size.h; │ │ │ │ │ + markup.id = tile.id + "_bb"; │ │ │ │ │ + backBuffer.appendChild(markup) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return backBuffer │ │ │ │ │ + }, │ │ │ │ │ + removeBackBuffer: function() { │ │ │ │ │ + if (this._transitionElement) { │ │ │ │ │ + for (var i = this.transitionendEvents.length - 1; i >= 0; --i) { │ │ │ │ │ + OpenLayers.Event.stopObserving(this._transitionElement, this.transitionendEvents[i], this._removeBackBuffer) │ │ │ │ │ + } │ │ │ │ │ + delete this._transitionElement │ │ │ │ │ + } │ │ │ │ │ + if (this.backBuffer) { │ │ │ │ │ + if (this.backBuffer.parentNode) { │ │ │ │ │ + this.backBuffer.parentNode.removeChild(this.backBuffer) │ │ │ │ │ + } │ │ │ │ │ + this.backBuffer = null; │ │ │ │ │ + this.backBufferResolution = null; │ │ │ │ │ + if (this.backBufferTimerId !== null) { │ │ │ │ │ + window.clearTimeout(this.backBufferTimerId); │ │ │ │ │ + this.backBufferTimerId = null │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + moveByPx: function(dx, dy) { │ │ │ │ │ + if (!this.singleTile) { │ │ │ │ │ + this.moveGriddedTiles() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setTileSize: function(size) { │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + size.h = parseInt(size.h * this.ratio, 10); │ │ │ │ │ + size.w = parseInt(size.w * this.ratio, 10) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.HTTPRequest.prototype.setTileSize.apply(this, [size]) │ │ │ │ │ + }, │ │ │ │ │ + getTilesBounds: function() { │ │ │ │ │ + var bounds = null; │ │ │ │ │ + var length = this.grid.length; │ │ │ │ │ + if (length) { │ │ │ │ │ + var bottomLeftTileBounds = this.grid[length - 1][0].bounds, │ │ │ │ │ + width = this.grid[0].length * bottomLeftTileBounds.getWidth(), │ │ │ │ │ + height = this.grid.length * bottomLeftTileBounds.getHeight(); │ │ │ │ │ + bounds = new OpenLayers.Bounds(bottomLeftTileBounds.left, bottomLeftTileBounds.bottom, bottomLeftTileBounds.left + width, bottomLeftTileBounds.bottom + height) │ │ │ │ │ + } │ │ │ │ │ + return bounds │ │ │ │ │ + }, │ │ │ │ │ + initSingleTile: function(bounds) { │ │ │ │ │ + this.events.triggerEvent("retile"); │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var tileWidth = bounds.getWidth() * this.ratio; │ │ │ │ │ + var tileHeight = bounds.getHeight() * this.ratio; │ │ │ │ │ + var tileBounds = new OpenLayers.Bounds(center.lon - tileWidth / 2, center.lat - tileHeight / 2, center.lon + tileWidth / 2, center.lat + tileHeight / 2); │ │ │ │ │ + var px = this.map.getLayerPxFromLonLat({ │ │ │ │ │ + lon: tileBounds.left, │ │ │ │ │ + lat: tileBounds.top │ │ │ │ │ + }); │ │ │ │ │ + if (!this.grid.length) { │ │ │ │ │ + this.grid[0] = [] │ │ │ │ │ + } │ │ │ │ │ + var tile = this.grid[0][0]; │ │ │ │ │ + if (!tile) { │ │ │ │ │ + tile = this.addTile(tileBounds, px); │ │ │ │ │ + this.addTileMonitoringHooks(tile); │ │ │ │ │ + tile.draw(); │ │ │ │ │ + this.grid[0][0] = tile │ │ │ │ │ + } else { │ │ │ │ │ + tile.moveTo(tileBounds, px) │ │ │ │ │ + } │ │ │ │ │ + this.removeExcessTiles(1, 1); │ │ │ │ │ + this.gridResolution = this.getServerResolution() │ │ │ │ │ + }, │ │ │ │ │ + calculateGridLayout: function(bounds, origin, resolution) { │ │ │ │ │ + var tilelon = resolution * this.tileSize.w; │ │ │ │ │ + var tilelat = resolution * this.tileSize.h; │ │ │ │ │ + var offsetlon = bounds.left - origin.lon; │ │ │ │ │ + var tilecol = Math.floor(offsetlon / tilelon) - this.buffer; │ │ │ │ │ + var rowSign = this.rowSign; │ │ │ │ │ + var offsetlat = rowSign * (origin.lat - bounds.top + tilelat); │ │ │ │ │ + var tilerow = Math[~rowSign ? "floor" : "ceil"](offsetlat / tilelat) - this.buffer * rowSign; │ │ │ │ │ + return { │ │ │ │ │ + tilelon: tilelon, │ │ │ │ │ + tilelat: tilelat, │ │ │ │ │ + startcol: tilecol, │ │ │ │ │ + startrow: tilerow │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getTileOrigin: function() { │ │ │ │ │ + var origin = this.tileOrigin; │ │ │ │ │ + if (!origin) { │ │ │ │ │ + var extent = this.getMaxExtent(); │ │ │ │ │ + var edges = { │ │ │ │ │ + tl: ["left", "top"], │ │ │ │ │ + tr: ["right", "top"], │ │ │ │ │ + bl: ["left", "bottom"], │ │ │ │ │ + br: ["right", "bottom"] │ │ │ │ │ + } [this.tileOriginCorner]; │ │ │ │ │ + origin = new OpenLayers.LonLat(extent[edges[0]], extent[edges[1]]) │ │ │ │ │ + } │ │ │ │ │ + return origin │ │ │ │ │ + }, │ │ │ │ │ + getTileBoundsForGridIndex: function(row, col) { │ │ │ │ │ + var origin = this.getTileOrigin(); │ │ │ │ │ + var tileLayout = this.gridLayout; │ │ │ │ │ + var tilelon = tileLayout.tilelon; │ │ │ │ │ + var tilelat = tileLayout.tilelat; │ │ │ │ │ + var startcol = tileLayout.startcol; │ │ │ │ │ + var startrow = tileLayout.startrow; │ │ │ │ │ + var rowSign = this.rowSign; │ │ │ │ │ + return new OpenLayers.Bounds(origin.lon + (startcol + col) * tilelon, origin.lat - (startrow + row * rowSign) * tilelat * rowSign, origin.lon + (startcol + col + 1) * tilelon, origin.lat - (startrow + (row - 1) * rowSign) * tilelat * rowSign) │ │ │ │ │ + }, │ │ │ │ │ + initGriddedTiles: function(bounds) { │ │ │ │ │ + this.events.triggerEvent("retile"); │ │ │ │ │ + var viewSize = this.map.getSize(); │ │ │ │ │ + var origin = this.getTileOrigin(); │ │ │ │ │ + var resolution = this.map.getResolution(), │ │ │ │ │ + serverResolution = this.getServerResolution(), │ │ │ │ │ + ratio = resolution / serverResolution, │ │ │ │ │ + tileSize = { │ │ │ │ │ + w: this.tileSize.w / ratio, │ │ │ │ │ + h: this.tileSize.h / ratio │ │ │ │ │ + }; │ │ │ │ │ + var minRows = Math.ceil(viewSize.h / tileSize.h) + 2 * this.buffer + 1; │ │ │ │ │ + var minCols = Math.ceil(viewSize.w / tileSize.w) + 2 * this.buffer + 1; │ │ │ │ │ + var tileLayout = this.calculateGridLayout(bounds, origin, serverResolution); │ │ │ │ │ + this.gridLayout = tileLayout; │ │ │ │ │ + var tilelon = tileLayout.tilelon; │ │ │ │ │ + var tilelat = tileLayout.tilelat; │ │ │ │ │ + var layerContainerDivLeft = this.map.layerContainerOriginPx.x; │ │ │ │ │ + var layerContainerDivTop = this.map.layerContainerOriginPx.y; │ │ │ │ │ + var tileBounds = this.getTileBoundsForGridIndex(0, 0); │ │ │ │ │ + var startPx = this.map.getViewPortPxFromLonLat(new OpenLayers.LonLat(tileBounds.left, tileBounds.top)); │ │ │ │ │ + startPx.x = Math.round(startPx.x) - layerContainerDivLeft; │ │ │ │ │ + startPx.y = Math.round(startPx.y) - layerContainerDivTop; │ │ │ │ │ + var tileData = [], │ │ │ │ │ + center = this.map.getCenter(); │ │ │ │ │ + var rowidx = 0; │ │ │ │ │ + do { │ │ │ │ │ + var row = this.grid[rowidx]; │ │ │ │ │ + if (!row) { │ │ │ │ │ + row = []; │ │ │ │ │ + this.grid.push(row) │ │ │ │ │ + } │ │ │ │ │ + var colidx = 0; │ │ │ │ │ + do { │ │ │ │ │ + tileBounds = this.getTileBoundsForGridIndex(rowidx, colidx); │ │ │ │ │ + var px = startPx.clone(); │ │ │ │ │ + px.x = px.x + colidx * Math.round(tileSize.w); │ │ │ │ │ + px.y = px.y + rowidx * Math.round(tileSize.h); │ │ │ │ │ + var tile = row[colidx]; │ │ │ │ │ + if (!tile) { │ │ │ │ │ + tile = this.addTile(tileBounds, px); │ │ │ │ │ + this.addTileMonitoringHooks(tile); │ │ │ │ │ + row.push(tile) │ │ │ │ │ + } else { │ │ │ │ │ + tile.moveTo(tileBounds, px, false) │ │ │ │ │ + } │ │ │ │ │ + var tileCenter = tileBounds.getCenterLonLat(); │ │ │ │ │ + tileData.push({ │ │ │ │ │ + tile: tile, │ │ │ │ │ + distance: Math.pow(tileCenter.lon - center.lon, 2) + Math.pow(tileCenter.lat - center.lat, 2) │ │ │ │ │ + }); │ │ │ │ │ + colidx += 1 │ │ │ │ │ + } while (tileBounds.right <= bounds.right + tilelon * this.buffer || colidx < minCols); │ │ │ │ │ + rowidx += 1 │ │ │ │ │ + } while (tileBounds.bottom >= bounds.bottom - tilelat * this.buffer || rowidx < minRows); │ │ │ │ │ + this.removeExcessTiles(rowidx, colidx); │ │ │ │ │ + var resolution = this.getServerResolution(); │ │ │ │ │ + this.gridResolution = resolution; │ │ │ │ │ + tileData.sort(function(a, b) { │ │ │ │ │ + return a.distance - b.distance │ │ │ │ │ + }); │ │ │ │ │ + for (var i = 0, ii = tileData.length; i < ii; ++i) { │ │ │ │ │ + tileData[i].tile.draw() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getMaxExtent: function() { │ │ │ │ │ + return this.maxExtent │ │ │ │ │ + }, │ │ │ │ │ + addTile: function(bounds, position) { │ │ │ │ │ + var tile = new this.tileClass(this, position, bounds, null, this.tileSize, this.tileOptions); │ │ │ │ │ + this.events.triggerEvent("addtile", { │ │ │ │ │ + tile: tile │ │ │ │ │ + }); │ │ │ │ │ + return tile │ │ │ │ │ + }, │ │ │ │ │ + addTileMonitoringHooks: function(tile) { │ │ │ │ │ + var replacingCls = "olTileReplacing"; │ │ │ │ │ + tile.onLoadStart = function() { │ │ │ │ │ + if (this.loading === false) { │ │ │ │ │ + this.loading = true; │ │ │ │ │ + this.events.triggerEvent("loadstart") │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("tileloadstart", { │ │ │ │ │ + tile: tile │ │ │ │ │ + }); │ │ │ │ │ + this.numLoadingTiles++; │ │ │ │ │ + if (!this.singleTile && this.backBuffer && this.gridResolution === this.backBufferResolution) { │ │ │ │ │ + OpenLayers.Element.addClass(tile.getTile(), replacingCls) │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + tile.onLoadEnd = function(evt) { │ │ │ │ │ + this.numLoadingTiles--; │ │ │ │ │ + var aborted = evt.type === "unload"; │ │ │ │ │ + this.events.triggerEvent("tileloaded", { │ │ │ │ │ + tile: tile, │ │ │ │ │ + aborted: aborted │ │ │ │ │ + }); │ │ │ │ │ + if (!this.singleTile && !aborted && this.backBuffer && this.gridResolution === this.backBufferResolution) { │ │ │ │ │ + var tileDiv = tile.getTile(); │ │ │ │ │ + if (OpenLayers.Element.getStyle(tileDiv, "display") === "none") { │ │ │ │ │ + var bufferTile = document.getElementById(tile.id + "_bb"); │ │ │ │ │ + if (bufferTile) { │ │ │ │ │ + bufferTile.parentNode.removeChild(bufferTile) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.removeClass(tileDiv, replacingCls) │ │ │ │ │ + } │ │ │ │ │ + if (this.numLoadingTiles === 0) { │ │ │ │ │ + if (this.backBuffer) { │ │ │ │ │ + if (this.backBuffer.childNodes.length === 0) { │ │ │ │ │ + this.removeBackBuffer() │ │ │ │ │ + } else { │ │ │ │ │ + this._transitionElement = aborted ? this.div.lastChild : tile.imgDiv; │ │ │ │ │ + var transitionendEvents = this.transitionendEvents; │ │ │ │ │ + for (var i = transitionendEvents.length - 1; i >= 0; --i) { │ │ │ │ │ + OpenLayers.Event.observe(this._transitionElement, transitionendEvents[i], this._removeBackBuffer) │ │ │ │ │ + } │ │ │ │ │ + this.backBufferTimerId = window.setTimeout(this._removeBackBuffer, this.removeBackBufferDelay) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.loading = false; │ │ │ │ │ + this.events.triggerEvent("loadend") │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + tile.onLoadError = function() { │ │ │ │ │ + this.events.triggerEvent("tileerror", { │ │ │ │ │ + tile: tile │ │ │ │ │ + }) │ │ │ │ │ + }; │ │ │ │ │ + tile.events.on({ │ │ │ │ │ + loadstart: tile.onLoadStart, │ │ │ │ │ + loadend: tile.onLoadEnd, │ │ │ │ │ + unload: tile.onLoadEnd, │ │ │ │ │ + loaderror: tile.onLoadError, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + removeTileMonitoringHooks: function(tile) { │ │ │ │ │ + tile.unload(); │ │ │ │ │ + tile.events.un({ │ │ │ │ │ + loadstart: tile.onLoadStart, │ │ │ │ │ + loadend: tile.onLoadEnd, │ │ │ │ │ + unload: tile.onLoadEnd, │ │ │ │ │ + loaderror: tile.onLoadError, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + moveGriddedTiles: function() { │ │ │ │ │ + var buffer = this.buffer + 1; │ │ │ │ │ + while (true) { │ │ │ │ │ + var tlTile = this.grid[0][0]; │ │ │ │ │ + var tlViewPort = { │ │ │ │ │ + x: tlTile.position.x + this.map.layerContainerOriginPx.x, │ │ │ │ │ + y: tlTile.position.y + this.map.layerContainerOriginPx.y │ │ │ │ │ + }; │ │ │ │ │ + var ratio = this.getServerResolution() / this.map.getResolution(); │ │ │ │ │ + var tileSize = { │ │ │ │ │ + w: Math.round(this.tileSize.w * ratio), │ │ │ │ │ + h: Math.round(this.tileSize.h * ratio) │ │ │ │ │ + }; │ │ │ │ │ + if (tlViewPort.x > -tileSize.w * (buffer - 1)) { │ │ │ │ │ + this.shiftColumn(true, tileSize) │ │ │ │ │ + } else if (tlViewPort.x < -tileSize.w * buffer) { │ │ │ │ │ + this.shiftColumn(false, tileSize) │ │ │ │ │ + } else if (tlViewPort.y > -tileSize.h * (buffer - 1)) { │ │ │ │ │ + this.shiftRow(true, tileSize) │ │ │ │ │ + } else if (tlViewPort.y < -tileSize.h * buffer) { │ │ │ │ │ + this.shiftRow(false, tileSize) │ │ │ │ │ + } else { │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + shiftRow: function(prepend, tileSize) { │ │ │ │ │ + var grid = this.grid; │ │ │ │ │ + var rowIndex = prepend ? 0 : grid.length - 1; │ │ │ │ │ + var sign = prepend ? -1 : 1; │ │ │ │ │ + var rowSign = this.rowSign; │ │ │ │ │ + var tileLayout = this.gridLayout; │ │ │ │ │ + tileLayout.startrow += sign * rowSign; │ │ │ │ │ + var modelRow = grid[rowIndex]; │ │ │ │ │ + var row = grid[prepend ? "pop" : "shift"](); │ │ │ │ │ + for (var i = 0, len = row.length; i < len; i++) { │ │ │ │ │ + var tile = row[i]; │ │ │ │ │ + var position = modelRow[i].position.clone(); │ │ │ │ │ + position.y += tileSize.h * sign; │ │ │ │ │ + tile.moveTo(this.getTileBoundsForGridIndex(rowIndex, i), position) │ │ │ │ │ + } │ │ │ │ │ + grid[prepend ? "unshift" : "push"](row) │ │ │ │ │ + }, │ │ │ │ │ + shiftColumn: function(prepend, tileSize) { │ │ │ │ │ + var grid = this.grid; │ │ │ │ │ + var colIndex = prepend ? 0 : grid[0].length - 1; │ │ │ │ │ + var sign = prepend ? -1 : 1; │ │ │ │ │ + var tileLayout = this.gridLayout; │ │ │ │ │ + tileLayout.startcol += sign; │ │ │ │ │ + for (var i = 0, len = grid.length; i < len; i++) { │ │ │ │ │ + var row = grid[i]; │ │ │ │ │ + var position = row[colIndex].position.clone(); │ │ │ │ │ + var tile = row[prepend ? "pop" : "shift"](); │ │ │ │ │ + position.x += tileSize.w * sign; │ │ │ │ │ + tile.moveTo(this.getTileBoundsForGridIndex(i, colIndex), position); │ │ │ │ │ + row[prepend ? "unshift" : "push"](tile) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + removeExcessTiles: function(rows, columns) { │ │ │ │ │ + var i, l; │ │ │ │ │ + while (this.grid.length > rows) { │ │ │ │ │ + var row = this.grid.pop(); │ │ │ │ │ + for (i = 0, l = row.length; i < l; i++) { │ │ │ │ │ + var tile = row[i]; │ │ │ │ │ + this.destroyTile(tile) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + for (i = 0, l = this.grid.length; i < l; i++) { │ │ │ │ │ + while (this.grid[i].length > columns) { │ │ │ │ │ + var row = this.grid[i]; │ │ │ │ │ + var tile = row.pop(); │ │ │ │ │ + this.destroyTile(tile) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + this.clearGrid(); │ │ │ │ │ + this.setTileSize() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getTileBounds: function(viewPortPx) { │ │ │ │ │ + var maxExtent = this.maxExtent; │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var tileMapWidth = resolution * this.tileSize.w; │ │ │ │ │ + var tileMapHeight = resolution * this.tileSize.h; │ │ │ │ │ + var mapPoint = this.getLonLatFromViewPortPx(viewPortPx); │ │ │ │ │ + var tileLeft = maxExtent.left + tileMapWidth * Math.floor((mapPoint.lon - maxExtent.left) / tileMapWidth); │ │ │ │ │ + var tileBottom = maxExtent.bottom + tileMapHeight * Math.floor((mapPoint.lat - maxExtent.bottom) / tileMapHeight); │ │ │ │ │ + return new OpenLayers.Bounds(tileLeft, tileBottom, tileLeft + tileMapWidth, tileBottom + tileMapHeight) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Grid" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.TileManager = OpenLayers.Class({ │ │ │ │ │ + cacheSize: 256, │ │ │ │ │ + tilesPerFrame: 2, │ │ │ │ │ + frameDelay: 16, │ │ │ │ │ + moveDelay: 100, │ │ │ │ │ + zoomDelay: 200, │ │ │ │ │ + maps: null, │ │ │ │ │ + tileQueueId: null, │ │ │ │ │ + tileQueue: null, │ │ │ │ │ + tileCache: null, │ │ │ │ │ + tileCacheIndex: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.maps = []; │ │ │ │ │ + this.tileQueueId = {}; │ │ │ │ │ + this.tileQueue = {}; │ │ │ │ │ + this.tileCache = {}; │ │ │ │ │ + this.tileCacheIndex = [] │ │ │ │ │ + }, │ │ │ │ │ + addMap: function(map) { │ │ │ │ │ + if (this._destroyed || !OpenLayers.Layer.Grid) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + this.maps.push(map); │ │ │ │ │ + this.tileQueue[map.id] = []; │ │ │ │ │ + for (var i = 0, ii = map.layers.length; i < ii; ++i) { │ │ │ │ │ + this.addLayer({ │ │ │ │ │ + layer: map.layers[i] │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + map.events.on({ │ │ │ │ │ + move: this.move, │ │ │ │ │ + zoomend: this.zoomEnd, │ │ │ │ │ + changelayer: this.changeLayer, │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + preremovelayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + if (this._destroyed || !OpenLayers.Layer.Grid) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + window.clearTimeout(this.tileQueueId[map.id]); │ │ │ │ │ + if (map.layers) { │ │ │ │ │ + for (var i = 0, ii = map.layers.length; i < ii; ++i) { │ │ │ │ │ + this.removeLayer({ │ │ │ │ │ + layer: map.layers[i] │ │ │ │ │ }) │ │ │ │ │ - })) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.styles[renderIntent].addRules(rules) │ │ │ │ │ + if (map.events) { │ │ │ │ │ + map.events.un({ │ │ │ │ │ + move: this.move, │ │ │ │ │ + zoomend: this.zoomEnd, │ │ │ │ │ + changelayer: this.changeLayer, │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + preremovelayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + delete this.tileQueue[map.id]; │ │ │ │ │ + delete this.tileQueueId[map.id]; │ │ │ │ │ + OpenLayers.Util.removeItem(this.maps, map) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.StyleMap" │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + this.updateTimeout(evt.object, this.moveDelay, true) │ │ │ │ │ + }, │ │ │ │ │ + zoomEnd: function(evt) { │ │ │ │ │ + this.updateTimeout(evt.object, this.zoomDelay) │ │ │ │ │ + }, │ │ │ │ │ + changeLayer: function(evt) { │ │ │ │ │ + if (evt.property === "visibility" || evt.property === "params") { │ │ │ │ │ + this.updateTimeout(evt.object, 0) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addLayer: function(evt) { │ │ │ │ │ + var layer = evt.layer; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ + layer.events.on({ │ │ │ │ │ + addtile: this.addTile, │ │ │ │ │ + retile: this.clearTileQueue, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + var i, j, tile; │ │ │ │ │ + for (i = layer.grid.length - 1; i >= 0; --i) { │ │ │ │ │ + for (j = layer.grid[i].length - 1; j >= 0; --j) { │ │ │ │ │ + tile = layer.grid[i][j]; │ │ │ │ │ + this.addTile({ │ │ │ │ │ + tile: tile │ │ │ │ │ + }); │ │ │ │ │ + if (tile.url && !tile.imgDiv) { │ │ │ │ │ + this.manageTileCache({ │ │ │ │ │ + object: tile │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + removeLayer: function(evt) { │ │ │ │ │ + var layer = evt.layer; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ + this.clearTileQueue({ │ │ │ │ │ + object: layer │ │ │ │ │ + }); │ │ │ │ │ + if (layer.events) { │ │ │ │ │ + layer.events.un({ │ │ │ │ │ + addtile: this.addTile, │ │ │ │ │ + retile: this.clearTileQueue, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + if (layer.grid) { │ │ │ │ │ + var i, j, tile; │ │ │ │ │ + for (i = layer.grid.length - 1; i >= 0; --i) { │ │ │ │ │ + for (j = layer.grid[i].length - 1; j >= 0; --j) { │ │ │ │ │ + tile = layer.grid[i][j]; │ │ │ │ │ + this.unloadTile({ │ │ │ │ │ + object: tile │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + updateTimeout: function(map, delay, nice) { │ │ │ │ │ + window.clearTimeout(this.tileQueueId[map.id]); │ │ │ │ │ + var tileQueue = this.tileQueue[map.id]; │ │ │ │ │ + if (!nice || tileQueue.length) { │ │ │ │ │ + this.tileQueueId[map.id] = window.setTimeout(OpenLayers.Function.bind(function() { │ │ │ │ │ + this.drawTilesFromQueue(map); │ │ │ │ │ + if (tileQueue.length) { │ │ │ │ │ + this.updateTimeout(map, this.frameDelay) │ │ │ │ │ + } │ │ │ │ │ + }, this), delay) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addTile: function(evt) { │ │ │ │ │ + if (evt.tile instanceof OpenLayers.Tile.Image) { │ │ │ │ │ + evt.tile.events.on({ │ │ │ │ │ + beforedraw: this.queueTileDraw, │ │ │ │ │ + beforeload: this.manageTileCache, │ │ │ │ │ + loadend: this.addToCache, │ │ │ │ │ + unload: this.unloadTile, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + this.removeLayer({ │ │ │ │ │ + layer: evt.tile.layer │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + unloadTile: function(evt) { │ │ │ │ │ + var tile = evt.object; │ │ │ │ │ + tile.events.un({ │ │ │ │ │ + beforedraw: this.queueTileDraw, │ │ │ │ │ + beforeload: this.manageTileCache, │ │ │ │ │ + loadend: this.addToCache, │ │ │ │ │ + unload: this.unloadTile, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Util.removeItem(this.tileQueue[tile.layer.map.id], tile) │ │ │ │ │ + }, │ │ │ │ │ + queueTileDraw: function(evt) { │ │ │ │ │ + var tile = evt.object; │ │ │ │ │ + var queued = false; │ │ │ │ │ + var layer = tile.layer; │ │ │ │ │ + var url = layer.getURL(tile.bounds); │ │ │ │ │ + var img = this.tileCache[url]; │ │ │ │ │ + if (img && img.className !== "olTileImage") { │ │ │ │ │ + delete this.tileCache[url]; │ │ │ │ │ + OpenLayers.Util.removeItem(this.tileCacheIndex, url); │ │ │ │ │ + img = null │ │ │ │ │ + } │ │ │ │ │ + if (layer.url && (layer.async || !img)) { │ │ │ │ │ + var tileQueue = this.tileQueue[layer.map.id]; │ │ │ │ │ + if (!~OpenLayers.Util.indexOf(tileQueue, tile)) { │ │ │ │ │ + tileQueue.push(tile) │ │ │ │ │ + } │ │ │ │ │ + queued = true │ │ │ │ │ + } │ │ │ │ │ + return !queued │ │ │ │ │ + }, │ │ │ │ │ + drawTilesFromQueue: function(map) { │ │ │ │ │ + var tileQueue = this.tileQueue[map.id]; │ │ │ │ │ + var limit = this.tilesPerFrame; │ │ │ │ │ + var animating = map.zoomTween && map.zoomTween.playing; │ │ │ │ │ + while (!animating && tileQueue.length && limit) { │ │ │ │ │ + tileQueue.shift().draw(true); │ │ │ │ │ + --limit │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + manageTileCache: function(evt) { │ │ │ │ │ + var tile = evt.object; │ │ │ │ │ + var img = this.tileCache[tile.url]; │ │ │ │ │ + if (img) { │ │ │ │ │ + if (img.parentNode && OpenLayers.Element.hasClass(img.parentNode, "olBackBuffer")) { │ │ │ │ │ + img.parentNode.removeChild(img); │ │ │ │ │ + img.id = null │ │ │ │ │ + } │ │ │ │ │ + if (!img.parentNode) { │ │ │ │ │ + img.style.visibility = "hidden"; │ │ │ │ │ + img.style.opacity = 0; │ │ │ │ │ + tile.setImage(img); │ │ │ │ │ + OpenLayers.Util.removeItem(this.tileCacheIndex, tile.url); │ │ │ │ │ + this.tileCacheIndex.push(tile.url) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addToCache: function(evt) { │ │ │ │ │ + var tile = evt.object; │ │ │ │ │ + if (!this.tileCache[tile.url]) { │ │ │ │ │ + if (!OpenLayers.Element.hasClass(tile.imgDiv, "olImageLoadError")) { │ │ │ │ │ + if (this.tileCacheIndex.length >= this.cacheSize) { │ │ │ │ │ + delete this.tileCache[this.tileCacheIndex[0]]; │ │ │ │ │ + this.tileCacheIndex.shift() │ │ │ │ │ + } │ │ │ │ │ + this.tileCache[tile.url] = tile.imgDiv; │ │ │ │ │ + this.tileCacheIndex.push(tile.url) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + clearTileQueue: function(evt) { │ │ │ │ │ + var layer = evt.object; │ │ │ │ │ + var tileQueue = this.tileQueue[layer.map.id]; │ │ │ │ │ + for (var i = tileQueue.length - 1; i >= 0; --i) { │ │ │ │ │ + if (tileQueue[i].layer === layer) { │ │ │ │ │ + tileQueue.splice(i, 1) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i = this.maps.length - 1; i >= 0; --i) { │ │ │ │ │ + this.removeMap(this.maps[i]) │ │ │ │ │ + } │ │ │ │ │ + this.maps = null; │ │ │ │ │ + this.tileQueue = null; │ │ │ │ │ + this.tileQueueId = null; │ │ │ │ │ + this.tileCache = null; │ │ │ │ │ + this.tileCacheIndex = null; │ │ │ │ │ + this._destroyed = true │ │ │ │ │ + } │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Rule = OpenLayers.Class({ │ │ │ │ │ + id: null, │ │ │ │ │ + name: null, │ │ │ │ │ + title: null, │ │ │ │ │ + description: null, │ │ │ │ │ + context: null, │ │ │ │ │ + filter: null, │ │ │ │ │ + elseFilter: false, │ │ │ │ │ + symbolizer: null, │ │ │ │ │ + symbolizers: null, │ │ │ │ │ + minScaleDenominator: null, │ │ │ │ │ + maxScaleDenominator: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.symbolizer = {}; │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + if (this.symbolizers) { │ │ │ │ │ + delete this.symbolizer │ │ │ │ │ + } │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i in this.symbolizer) { │ │ │ │ │ + this.symbolizer[i] = null │ │ │ │ │ + } │ │ │ │ │ + this.symbolizer = null; │ │ │ │ │ + delete this.symbolizers │ │ │ │ │ + }, │ │ │ │ │ + evaluate: function(feature) { │ │ │ │ │ + var context = this.getContext(feature); │ │ │ │ │ + var applies = true; │ │ │ │ │ + if (this.minScaleDenominator || this.maxScaleDenominator) { │ │ │ │ │ + var scale = feature.layer.map.getScale() │ │ │ │ │ + } │ │ │ │ │ + if (this.minScaleDenominator) { │ │ │ │ │ + applies = scale >= OpenLayers.Style.createLiteral(this.minScaleDenominator, context) │ │ │ │ │ + } │ │ │ │ │ + if (applies && this.maxScaleDenominator) { │ │ │ │ │ + applies = scale < OpenLayers.Style.createLiteral(this.maxScaleDenominator, context) │ │ │ │ │ + } │ │ │ │ │ + if (applies && this.filter) { │ │ │ │ │ + if (this.filter.CLASS_NAME == "OpenLayers.Filter.FeatureId") { │ │ │ │ │ + applies = this.filter.evaluate(feature) │ │ │ │ │ + } else { │ │ │ │ │ + applies = this.filter.evaluate(context) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return applies │ │ │ │ │ + }, │ │ │ │ │ + getContext: function(feature) { │ │ │ │ │ + var context = this.context; │ │ │ │ │ + if (!context) { │ │ │ │ │ + context = feature.attributes || feature.data │ │ │ │ │ + } │ │ │ │ │ + if (typeof this.context == "function") { │ │ │ │ │ + context = this.context(feature) │ │ │ │ │ + } │ │ │ │ │ + return context │ │ │ │ │ + }, │ │ │ │ │ + clone: function() { │ │ │ │ │ + var options = OpenLayers.Util.extend({}, this); │ │ │ │ │ + if (this.symbolizers) { │ │ │ │ │ + var len = this.symbolizers.length; │ │ │ │ │ + options.symbolizers = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + options.symbolizers[i] = this.symbolizers[i].clone() │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + options.symbolizer = {}; │ │ │ │ │ + var value, type; │ │ │ │ │ + for (var key in this.symbolizer) { │ │ │ │ │ + value = this.symbolizer[key]; │ │ │ │ │ + type = typeof value; │ │ │ │ │ + if (type === "object") { │ │ │ │ │ + options.symbolizer[key] = OpenLayers.Util.extend({}, value) │ │ │ │ │ + } else if (type === "string") { │ │ │ │ │ + options.symbolizer[key] = value │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + options.filter = this.filter && this.filter.clone(); │ │ │ │ │ + options.context = this.context && OpenLayers.Util.extend({}, this.context); │ │ │ │ │ + return new OpenLayers.Rule(options) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Rule" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Format.WPSDescribeProcess = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ VERSION: "1.0.0", │ │ │ │ │ namespaces: { │ │ │ │ │ wps: "http://www.opengis.net/wps/1.0.0", │ │ │ │ │ ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ @@ -13607,168 +13465,25 @@ │ │ │ │ │ destroy: function() { │ │ │ │ │ this.events.destroy(); │ │ │ │ │ this.events = null; │ │ │ │ │ this.servers = null │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.WPSClient" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Renderer = OpenLayers.Class({ │ │ │ │ │ - container: null, │ │ │ │ │ - root: null, │ │ │ │ │ - extent: null, │ │ │ │ │ - locked: false, │ │ │ │ │ - size: null, │ │ │ │ │ - resolution: null, │ │ │ │ │ - map: null, │ │ │ │ │ - featureDx: 0, │ │ │ │ │ - initialize: function(containerID, options) { │ │ │ │ │ - this.container = OpenLayers.Util.getElement(containerID); │ │ │ │ │ - OpenLayers.Util.extend(this, options) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.container = null; │ │ │ │ │ - this.extent = null; │ │ │ │ │ - this.size = null; │ │ │ │ │ - this.resolution = null; │ │ │ │ │ - this.map = null │ │ │ │ │ - }, │ │ │ │ │ - supported: function() { │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - this.extent = extent.clone(); │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - var ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ - extent = extent.scale(1 / ratio); │ │ │ │ │ - this.extent = extent.wrapDateLine(this.map.getMaxExtent()).scale(ratio) │ │ │ │ │ - } │ │ │ │ │ - if (resolutionChanged) { │ │ │ │ │ - this.resolution = null │ │ │ │ │ - } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - this.size = size.clone(); │ │ │ │ │ - this.resolution = null │ │ │ │ │ - }, │ │ │ │ │ - getResolution: function() { │ │ │ │ │ - this.resolution = this.resolution || this.map.getResolution(); │ │ │ │ │ - return this.resolution │ │ │ │ │ - }, │ │ │ │ │ - drawFeature: function(feature, style) { │ │ │ │ │ - if (style == null) { │ │ │ │ │ - style = feature.style │ │ │ │ │ - } │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - var bounds = feature.geometry.getBounds(); │ │ │ │ │ - if (bounds) { │ │ │ │ │ - var worldBounds; │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - worldBounds = this.map.getMaxExtent() │ │ │ │ │ - } │ │ │ │ │ - if (!bounds.intersectsBounds(this.extent, { │ │ │ │ │ - worldBounds: worldBounds │ │ │ │ │ - })) { │ │ │ │ │ - style = { │ │ │ │ │ - display: "none" │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.calculateFeatureDx(bounds, worldBounds) │ │ │ │ │ - } │ │ │ │ │ - var rendered = this.drawGeometry(feature.geometry, style, feature.id); │ │ │ │ │ - if (style.display != "none" && style.label && rendered !== false) { │ │ │ │ │ - var location = feature.geometry.getCentroid(); │ │ │ │ │ - if (style.labelXOffset || style.labelYOffset) { │ │ │ │ │ - var xOffset = isNaN(style.labelXOffset) ? 0 : style.labelXOffset; │ │ │ │ │ - var yOffset = isNaN(style.labelYOffset) ? 0 : style.labelYOffset; │ │ │ │ │ - var res = this.getResolution(); │ │ │ │ │ - location.move(xOffset * res, yOffset * res) │ │ │ │ │ - } │ │ │ │ │ - this.drawText(feature.id, style, location) │ │ │ │ │ - } else { │ │ │ │ │ - this.removeText(feature.id) │ │ │ │ │ - } │ │ │ │ │ - return rendered │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - calculateFeatureDx: function(bounds, worldBounds) { │ │ │ │ │ - this.featureDx = 0; │ │ │ │ │ - if (worldBounds) { │ │ │ │ │ - var worldWidth = worldBounds.getWidth(), │ │ │ │ │ - rendererCenterX = (this.extent.left + this.extent.right) / 2, │ │ │ │ │ - featureCenterX = (bounds.left + bounds.right) / 2, │ │ │ │ │ - worldsAway = Math.round((featureCenterX - rendererCenterX) / worldWidth); │ │ │ │ │ - this.featureDx = worldsAway * worldWidth │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - drawGeometry: function(geometry, style, featureId) {}, │ │ │ │ │ - drawText: function(featureId, style, location) {}, │ │ │ │ │ - removeText: function(featureId) {}, │ │ │ │ │ - clear: function() {}, │ │ │ │ │ - getFeatureIdFromEvent: function(evt) {}, │ │ │ │ │ - eraseFeatures: function(features) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ - features = [features] │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - this.eraseGeometry(feature.geometry, feature.id); │ │ │ │ │ - this.removeText(feature.id) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - eraseGeometry: function(geometry, featureId) {}, │ │ │ │ │ - moveRoot: function(renderer) {}, │ │ │ │ │ - getRenderLayerId: function() { │ │ │ │ │ - return this.container.id │ │ │ │ │ +OpenLayers.Symbolizer = OpenLayers.Class({ │ │ │ │ │ + zIndex: 0, │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ + OpenLayers.Util.extend(this, config) │ │ │ │ │ }, │ │ │ │ │ - applyDefaultSymbolizer: function(symbolizer) { │ │ │ │ │ - var result = OpenLayers.Util.extend({}, OpenLayers.Renderer.defaultSymbolizer); │ │ │ │ │ - if (symbolizer.stroke === false) { │ │ │ │ │ - delete result.strokeWidth; │ │ │ │ │ - delete result.strokeColor │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.fill === false) { │ │ │ │ │ - delete result.fillColor │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.extend(result, symbolizer); │ │ │ │ │ - return result │ │ │ │ │ + clone: function() { │ │ │ │ │ + var Type = eval(this.CLASS_NAME); │ │ │ │ │ + return new Type(OpenLayers.Util.extend({}, this)) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Symbolizer" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Renderer.defaultSymbolizer = { │ │ │ │ │ - fillColor: "#000000", │ │ │ │ │ - strokeColor: "#000000", │ │ │ │ │ - strokeWidth: 2, │ │ │ │ │ - fillOpacity: 1, │ │ │ │ │ - strokeOpacity: 1, │ │ │ │ │ - pointRadius: 0, │ │ │ │ │ - labelAlign: "cm" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.symbol = { │ │ │ │ │ - star: [350, 75, 379, 161, 469, 161, 397, 215, 423, 301, 350, 250, 277, 301, 303, 215, 231, 161, 321, 161, 350, 75], │ │ │ │ │ - cross: [4, 0, 6, 0, 6, 4, 10, 4, 10, 6, 6, 6, 6, 10, 4, 10, 4, 6, 0, 6, 0, 4, 4, 4, 4, 0], │ │ │ │ │ - x: [0, 0, 25, 0, 50, 35, 75, 0, 100, 0, 65, 50, 100, 100, 75, 100, 50, 65, 25, 100, 0, 100, 35, 50, 0, 0], │ │ │ │ │ - square: [0, 0, 0, 1, 1, 1, 1, 0, 0, 0], │ │ │ │ │ - triangle: [0, 10, 10, 10, 5, 0, 0, 10] │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Spherical = OpenLayers.Spherical || {}; │ │ │ │ │ -OpenLayers.Spherical.DEFAULT_RADIUS = 6378137; │ │ │ │ │ -OpenLayers.Spherical.computeDistanceBetween = function(from, to, radius) { │ │ │ │ │ - var R = radius || OpenLayers.Spherical.DEFAULT_RADIUS; │ │ │ │ │ - var sinHalfDeltaLon = Math.sin(Math.PI * (to.lon - from.lon) / 360); │ │ │ │ │ - var sinHalfDeltaLat = Math.sin(Math.PI * (to.lat - from.lat) / 360); │ │ │ │ │ - var a = sinHalfDeltaLat * sinHalfDeltaLat + sinHalfDeltaLon * sinHalfDeltaLon * Math.cos(Math.PI * from.lat / 180) * Math.cos(Math.PI * to.lat / 180); │ │ │ │ │ - return 2 * R * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a)) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Spherical.computeHeading = function(from, to) { │ │ │ │ │ - var y = Math.sin(Math.PI * (from.lon - to.lon) / 180) * Math.cos(Math.PI * to.lat / 180); │ │ │ │ │ - var x = Math.cos(Math.PI * from.lat / 180) * Math.sin(Math.PI * to.lat / 180) - Math.sin(Math.PI * from.lat / 180) * Math.cos(Math.PI * to.lat / 180) * Math.cos(Math.PI * (from.lon - to.lon) / 180); │ │ │ │ │ - return 180 * Math.atan2(y, x) / Math.PI │ │ │ │ │ -}; │ │ │ │ │ OpenLayers.Symbolizer.Point = OpenLayers.Class(OpenLayers.Symbolizer, { │ │ │ │ │ initialize: function(config) { │ │ │ │ │ OpenLayers.Symbolizer.prototype.initialize.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Symbolizer.Point" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Symbolizer.Line = OpenLayers.Class(OpenLayers.Symbolizer, { │ │ │ │ │ @@ -13821,4853 +13536,4833 @@ │ │ │ │ │ config.rules.push(this.rules[i].clone()) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ return new OpenLayers.Style2(config) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Style2" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.TileCache = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - format: "image/png", │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ - initialize: function(name, url, layername, options) { │ │ │ │ │ - this.layername = layername; │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name, url, {}, options]); │ │ │ │ │ - this.extension = this.format.split("/")[1].toLowerCase(); │ │ │ │ │ - this.extension = this.extension == "jpg" ? "jpeg" : this.extension │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.TileCache(this.name, this.url, this.layername, this.getOptions()) │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var bbox = this.maxExtent; │ │ │ │ │ - var size = this.tileSize; │ │ │ │ │ - var tileX = Math.round((bounds.left - bbox.left) / (res * size.w)); │ │ │ │ │ - var tileY = Math.round((bounds.bottom - bbox.bottom) / (res * size.h)); │ │ │ │ │ - var tileZ = this.serverResolutions != null ? OpenLayers.Util.indexOf(this.serverResolutions, res) : this.map.getZoom(); │ │ │ │ │ - var components = [this.layername, OpenLayers.Number.zeroPad(tileZ, 2), OpenLayers.Number.zeroPad(parseInt(tileX / 1e6), 3), OpenLayers.Number.zeroPad(parseInt(tileX / 1e3) % 1e3, 3), OpenLayers.Number.zeroPad(parseInt(tileX) % 1e3, 3), OpenLayers.Number.zeroPad(parseInt(tileY / 1e6), 3), OpenLayers.Number.zeroPad(parseInt(tileY / 1e3) % 1e3, 3), OpenLayers.Number.zeroPad(parseInt(tileY) % 1e3, 3) + "." + this.extension]; │ │ │ │ │ - var path = components.join("/"); │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(path, url) │ │ │ │ │ +OpenLayers.Control = OpenLayers.Class({ │ │ │ │ │ + id: null, │ │ │ │ │ + map: null, │ │ │ │ │ + div: null, │ │ │ │ │ + type: null, │ │ │ │ │ + allowSelection: false, │ │ │ │ │ + displayClass: "", │ │ │ │ │ + title: "", │ │ │ │ │ + autoActivate: false, │ │ │ │ │ + active: null, │ │ │ │ │ + handlerOptions: null, │ │ │ │ │ + handler: null, │ │ │ │ │ + eventListeners: null, │ │ │ │ │ + events: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.displayClass = this.CLASS_NAME.replace("OpenLayers.", "ol").replace(/\./g, ""); │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.events = new OpenLayers.Events(this); │ │ │ │ │ + if (this.eventListeners instanceof Object) { │ │ │ │ │ + this.events.on(this.eventListeners) │ │ │ │ │ } │ │ │ │ │ - url = url.charAt(url.length - 1) == "/" ? url : url + "/"; │ │ │ │ │ - return url + path │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.TileCache" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.EventPane = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ - smoothDragPan: true, │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - isFixed: true, │ │ │ │ │ - pane: null, │ │ │ │ │ - mapObject: null, │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (this.pane == null) { │ │ │ │ │ - this.pane = OpenLayers.Util.createDiv(this.div.id + "_EventPane") │ │ │ │ │ + if (this.id == null) { │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - this.mapObject = null; │ │ │ │ │ - this.pane = null; │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.pane.style.zIndex = parseInt(this.div.style.zIndex) + 1; │ │ │ │ │ - this.pane.style.display = this.div.style.display; │ │ │ │ │ - this.pane.style.width = "100%"; │ │ │ │ │ - this.pane.style.height = "100%"; │ │ │ │ │ - if (OpenLayers.BROWSER_NAME == "msie") { │ │ │ │ │ - this.pane.style.background = "url(" + OpenLayers.Util.getImageLocation("blank.gif") + ")" │ │ │ │ │ + if (this.events) { │ │ │ │ │ + if (this.eventListeners) { │ │ │ │ │ + this.events.un(this.eventListeners) │ │ │ │ │ + } │ │ │ │ │ + this.events.destroy(); │ │ │ │ │ + this.events = null │ │ │ │ │ } │ │ │ │ │ - if (this.isFixed) { │ │ │ │ │ - this.map.viewPortDiv.appendChild(this.pane) │ │ │ │ │ - } else { │ │ │ │ │ - this.map.layerContainerDiv.appendChild(this.pane) │ │ │ │ │ + this.eventListeners = null; │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.destroy(); │ │ │ │ │ + this.handler = null │ │ │ │ │ } │ │ │ │ │ - this.loadMapObject(); │ │ │ │ │ - if (this.mapObject == null) { │ │ │ │ │ - this.loadWarningMessage() │ │ │ │ │ + if (this.handlers) { │ │ │ │ │ + for (var key in this.handlers) { │ │ │ │ │ + if (this.handlers.hasOwnProperty(key) && typeof this.handlers[key].destroy == "function") { │ │ │ │ │ + this.handlers[key].destroy() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.handlers = null │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - if (this.pane && this.pane.parentNode) { │ │ │ │ │ - this.pane.parentNode.removeChild(this.pane) │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.removeControl(this); │ │ │ │ │ + this.map = null │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.prototype.removeMap.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - loadWarningMessage: function() { │ │ │ │ │ - this.div.style.backgroundColor = "darkblue"; │ │ │ │ │ - var viewSize = this.map.getSize(); │ │ │ │ │ - var msgW = Math.min(viewSize.w, 300); │ │ │ │ │ - var msgH = Math.min(viewSize.h, 200); │ │ │ │ │ - var size = new OpenLayers.Size(msgW, msgH); │ │ │ │ │ - var centerPx = new OpenLayers.Pixel(viewSize.w / 2, viewSize.h / 2); │ │ │ │ │ - var topLeft = centerPx.add(-size.w / 2, -size.h / 2); │ │ │ │ │ - var div = OpenLayers.Util.createDiv(this.name + "_warning", topLeft, size, null, null, null, "auto"); │ │ │ │ │ - div.style.padding = "7px"; │ │ │ │ │ - div.style.backgroundColor = "yellow"; │ │ │ │ │ - div.innerHTML = this.getWarningHTML(); │ │ │ │ │ - this.div.appendChild(div) │ │ │ │ │ - }, │ │ │ │ │ - getWarningHTML: function() { │ │ │ │ │ - return "" │ │ │ │ │ - }, │ │ │ │ │ - display: function(display) { │ │ │ │ │ - OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ - this.pane.style.display = this.div.style.display │ │ │ │ │ - }, │ │ │ │ │ - setZIndex: function(zIndex) { │ │ │ │ │ - OpenLayers.Layer.prototype.setZIndex.apply(this, arguments); │ │ │ │ │ - this.pane.style.zIndex = parseInt(this.div.style.zIndex) + 1 │ │ │ │ │ + this.div = null │ │ │ │ │ }, │ │ │ │ │ - moveByPx: function(dx, dy) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveByPx.apply(this, arguments); │ │ │ │ │ - if (this.dragPanMapObject) { │ │ │ │ │ - this.dragPanMapObject(dx, -dy) │ │ │ │ │ - } else { │ │ │ │ │ - this.moveTo(this.map.getCachedCenter()) │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.map = map; │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.setMap(map) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - if (this.mapObject != null) { │ │ │ │ │ - var newCenter = this.map.getCenter(); │ │ │ │ │ - var newZoom = this.map.getZoom(); │ │ │ │ │ - if (newCenter != null) { │ │ │ │ │ - var moOldCenter = this.getMapObjectCenter(); │ │ │ │ │ - var oldCenter = this.getOLLonLatFromMapObjectLonLat(moOldCenter); │ │ │ │ │ - var moOldZoom = this.getMapObjectZoom(); │ │ │ │ │ - var oldZoom = this.getOLZoomFromMapObjectZoom(moOldZoom); │ │ │ │ │ - if (!newCenter.equals(oldCenter) || newZoom != oldZoom) { │ │ │ │ │ - if (!zoomChanged && oldCenter && this.dragPanMapObject && this.smoothDragPan) { │ │ │ │ │ - var oldPx = this.map.getViewPortPxFromLonLat(oldCenter); │ │ │ │ │ - var newPx = this.map.getViewPortPxFromLonLat(newCenter); │ │ │ │ │ - this.dragPanMapObject(newPx.x - oldPx.x, oldPx.y - newPx.y) │ │ │ │ │ - } else { │ │ │ │ │ - var center = this.getMapObjectLonLatFromOLLonLat(newCenter); │ │ │ │ │ - var zoom = this.getMapObjectZoomFromOLZoom(newZoom); │ │ │ │ │ - this.setMapObjectCenter(center, zoom, dragging) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + if (this.div == null) { │ │ │ │ │ + this.div = OpenLayers.Util.createDiv(this.id); │ │ │ │ │ + this.div.className = this.displayClass; │ │ │ │ │ + if (!this.allowSelection) { │ │ │ │ │ + this.div.className += " olControlNoSelect"; │ │ │ │ │ + this.div.setAttribute("unselectable", "on", 0); │ │ │ │ │ + this.div.onselectstart = OpenLayers.Function.False │ │ │ │ │ + } │ │ │ │ │ + if (this.title != "") { │ │ │ │ │ + this.div.title = this.title │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ - var lonlat = null; │ │ │ │ │ - if (this.mapObject != null && this.getMapObjectCenter() != null) { │ │ │ │ │ - var moPixel = this.getMapObjectPixelFromOLPixel(viewPortPx); │ │ │ │ │ - var moLonLat = this.getMapObjectLonLatFromMapObjectPixel(moPixel); │ │ │ │ │ - lonlat = this.getOLLonLatFromMapObjectLonLat(moLonLat) │ │ │ │ │ + if (px != null) { │ │ │ │ │ + this.position = px.clone() │ │ │ │ │ } │ │ │ │ │ - return lonlat │ │ │ │ │ + this.moveTo(this.position); │ │ │ │ │ + return this.div │ │ │ │ │ }, │ │ │ │ │ - getViewPortPxFromLonLat: function(lonlat) { │ │ │ │ │ - var viewPortPx = null; │ │ │ │ │ - if (this.mapObject != null && this.getMapObjectCenter() != null) { │ │ │ │ │ - var moLonLat = this.getMapObjectLonLatFromOLLonLat(lonlat); │ │ │ │ │ - var moPixel = this.getMapObjectPixelFromMapObjectLonLat(moLonLat); │ │ │ │ │ - viewPortPx = this.getOLPixelFromMapObjectPixel(moPixel) │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + if (px != null && this.div != null) { │ │ │ │ │ + this.div.style.left = px.x + "px"; │ │ │ │ │ + this.div.style.top = px.y + "px" │ │ │ │ │ } │ │ │ │ │ - return viewPortPx │ │ │ │ │ }, │ │ │ │ │ - getOLLonLatFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - var olLonLat = null; │ │ │ │ │ - if (moLonLat != null) { │ │ │ │ │ - var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - olLonLat = new OpenLayers.LonLat(lon, lat) │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return olLonLat │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectLonLatFromOLLonLat: function(olLonLat) { │ │ │ │ │ - var moLatLng = null; │ │ │ │ │ - if (olLonLat != null) { │ │ │ │ │ - moLatLng = this.getMapObjectLonLatFromLonLat(olLonLat.lon, olLonLat.lat) │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.activate() │ │ │ │ │ } │ │ │ │ │ - return moLatLng │ │ │ │ │ - }, │ │ │ │ │ - getOLPixelFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - var olPixel = null; │ │ │ │ │ - if (moPixel != null) { │ │ │ │ │ - var x = this.getXFromMapObjectPixel(moPixel); │ │ │ │ │ - var y = this.getYFromMapObjectPixel(moPixel); │ │ │ │ │ - olPixel = new OpenLayers.Pixel(x, y) │ │ │ │ │ + this.active = true; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, this.displayClass.replace(/ /g, "") + "Active") │ │ │ │ │ } │ │ │ │ │ - return olPixel │ │ │ │ │ + this.events.triggerEvent("activate"); │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - getMapObjectPixelFromOLPixel: function(olPixel) { │ │ │ │ │ - var moPixel = null; │ │ │ │ │ - if (olPixel != null) { │ │ │ │ │ - moPixel = this.getMapObjectPixelFromXY(olPixel.x, olPixel.y) │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.deactivate() │ │ │ │ │ + } │ │ │ │ │ + this.active = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, this.displayClass.replace(/ /g, "") + "Active") │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("deactivate"); │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ - return moPixel │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.EventPane" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.WorldWind = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - DEFAULT_PARAMS: {}, │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - lzd: null, │ │ │ │ │ - zoomLevels: null, │ │ │ │ │ - initialize: function(name, url, lzd, zoomLevels, params, options) { │ │ │ │ │ - this.lzd = lzd; │ │ │ │ │ - this.zoomLevels = zoomLevels; │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - newArguments.push(name, url, params, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - this.params = OpenLayers.Util.applyDefaults(this.params, this.DEFAULT_PARAMS) │ │ │ │ │ - }, │ │ │ │ │ - getZoom: function() { │ │ │ │ │ - var zoom = this.map.getZoom(); │ │ │ │ │ - var extent = this.map.getMaxExtent(); │ │ │ │ │ - zoom = zoom - Math.log(this.maxResolution / (this.lzd / 512)) / Math.log(2); │ │ │ │ │ - return zoom │ │ │ │ │ +OpenLayers.Control.TYPE_BUTTON = 1; │ │ │ │ │ +OpenLayers.Control.TYPE_TOGGLE = 2; │ │ │ │ │ +OpenLayers.Control.TYPE_TOOL = 3; │ │ │ │ │ +OpenLayers.Protocol = OpenLayers.Class({ │ │ │ │ │ + format: null, │ │ │ │ │ + options: null, │ │ │ │ │ + autoDestroy: true, │ │ │ │ │ + defaultFilter: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.options = options │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var zoom = this.getZoom(); │ │ │ │ │ - var extent = this.map.getMaxExtent(); │ │ │ │ │ - var deg = this.lzd / Math.pow(2, this.getZoom()); │ │ │ │ │ - var x = Math.floor((bounds.left - extent.left) / deg); │ │ │ │ │ - var y = Math.floor((bounds.bottom - extent.bottom) / deg); │ │ │ │ │ - if (this.map.getResolution() <= this.lzd / 512 && this.getZoom() <= this.zoomLevels) { │ │ │ │ │ - return this.getFullRequestString({ │ │ │ │ │ - L: zoom, │ │ │ │ │ - X: x, │ │ │ │ │ - Y: y │ │ │ │ │ + mergeWithDefaultFilter: function(filter) { │ │ │ │ │ + var merged; │ │ │ │ │ + if (filter && this.defaultFilter) { │ │ │ │ │ + merged = new OpenLayers.Filter.Logical({ │ │ │ │ │ + type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ + filters: [this.defaultFilter, filter] │ │ │ │ │ }) │ │ │ │ │ } else { │ │ │ │ │ - return OpenLayers.Util.getImageLocation("blank.gif") │ │ │ │ │ + merged = filter || this.defaultFilter || undefined │ │ │ │ │ } │ │ │ │ │ + return merged │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.WorldWind" │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.options = null; │ │ │ │ │ + this.format = null │ │ │ │ │ + }, │ │ │ │ │ + read: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.filter = this.mergeWithDefaultFilter(options.filter) │ │ │ │ │ + }, │ │ │ │ │ + create: function() {}, │ │ │ │ │ + update: function() {}, │ │ │ │ │ + delete: function() {}, │ │ │ │ │ + commit: function() {}, │ │ │ │ │ + abort: function(response) {}, │ │ │ │ │ + createCallback: function(method, response, options) { │ │ │ │ │ + return OpenLayers.Function.bind(function() { │ │ │ │ │ + method.apply(this, [response, options]) │ │ │ │ │ + }, this) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.TMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - serviceVersion: "1.0.0", │ │ │ │ │ - layername: null, │ │ │ │ │ - type: null, │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - tileOrigin: null, │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ - zoomOffset: 0, │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - newArguments.push(name, url, {}, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments) │ │ │ │ │ +OpenLayers.Protocol.Response = OpenLayers.Class({ │ │ │ │ │ + code: null, │ │ │ │ │ + requestType: null, │ │ │ │ │ + last: true, │ │ │ │ │ + features: null, │ │ │ │ │ + data: null, │ │ │ │ │ + reqFeatures: null, │ │ │ │ │ + priv: null, │ │ │ │ │ + error: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options) │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.TMS(this.name, this.url, this.getOptions()) │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ + success: function() { │ │ │ │ │ + return this.code > 0 │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((bounds.bottom - this.tileOrigin.lat) / (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getServerZoom(); │ │ │ │ │ - var path = this.serviceVersion + "/" + this.layername + "/" + z + "/" + x + "/" + y + "." + this.type; │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(path, url) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.Response" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Protocol.Response.SUCCESS = 1; │ │ │ │ │ +OpenLayers.Protocol.Response.FAILURE = 0; │ │ │ │ │ +OpenLayers.Handler = OpenLayers.Class({ │ │ │ │ │ + id: null, │ │ │ │ │ + control: null, │ │ │ │ │ + map: null, │ │ │ │ │ + keyMask: null, │ │ │ │ │ + active: false, │ │ │ │ │ + evt: null, │ │ │ │ │ + touch: false, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.control = control; │ │ │ │ │ + this.callbacks = callbacks; │ │ │ │ │ + var map = this.map || control.map; │ │ │ │ │ + if (map) { │ │ │ │ │ + this.setMap(map) │ │ │ │ │ } │ │ │ │ │ - return url + path │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ }, │ │ │ │ │ setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (!this.tileOrigin) { │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.map.maxExtent.left, this.map.maxExtent.bottom) │ │ │ │ │ - } │ │ │ │ │ + this.map = map │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.TMS" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.XYZ = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - sphericalMercator: false, │ │ │ │ │ - zoomOffset: 0, │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - if (options && options.sphericalMercator || this.sphericalMercator) { │ │ │ │ │ - options = OpenLayers.Util.extend({ │ │ │ │ │ - projection: "EPSG:900913", │ │ │ │ │ - numZoomLevels: 19 │ │ │ │ │ - }, options) │ │ │ │ │ + checkModifiers: function(evt) { │ │ │ │ │ + if (this.keyMask == null) { │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name || this.name, url || this.url, {}, options]) │ │ │ │ │ + var keyModifiers = (evt.shiftKey ? OpenLayers.Handler.MOD_SHIFT : 0) | (evt.ctrlKey ? OpenLayers.Handler.MOD_CTRL : 0) | (evt.altKey ? OpenLayers.Handler.MOD_ALT : 0) | (evt.metaKey ? OpenLayers.Handler.MOD_META : 0); │ │ │ │ │ + return keyModifiers == this.keyMask │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.XYZ(this.name, this.url, this.getOptions()) │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var xyz = this.getXYZ(bounds); │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - var s = "" + xyz.x + xyz.y + xyz.z; │ │ │ │ │ - url = this.selectUrl(s, url) │ │ │ │ │ + var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.register(events[i], this[events[i]]) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.String.format(url, xyz) │ │ │ │ │ + this.active = true; │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - getXYZ: function(bounds) { │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.maxExtent.left) / (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((this.maxExtent.top - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getServerZoom(); │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - var limit = Math.pow(2, z); │ │ │ │ │ - x = (x % limit + limit) % limit │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return { │ │ │ │ │ - x: x, │ │ │ │ │ - y: y, │ │ │ │ │ - z: z │ │ │ │ │ + var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.unregister(events[i], this[events[i]]) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + this.touch = false; │ │ │ │ │ + this.active = false; │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (!this.tileOrigin) { │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, this.maxExtent.bottom) │ │ │ │ │ + startTouch: function() { │ │ │ │ │ + if (!this.touch) { │ │ │ │ │ + this.touch = true; │ │ │ │ │ + var events = ["mousedown", "mouseup", "mousemove", "click", "dblclick", "mouseout"]; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.unregister(events[i], this[events[i]]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.XYZ" │ │ │ │ │ + callback: function(name, args) { │ │ │ │ │ + if (name && this.callbacks[name]) { │ │ │ │ │ + this.callbacks[name].apply(this.control, args) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + register: function(name, method) { │ │ │ │ │ + this.map.events.registerPriority(name, this, method); │ │ │ │ │ + this.map.events.registerPriority(name, this, this.setEvent) │ │ │ │ │ + }, │ │ │ │ │ + unregister: function(name, method) { │ │ │ │ │ + this.map.events.unregister(name, this, method); │ │ │ │ │ + this.map.events.unregister(name, this, this.setEvent) │ │ │ │ │ + }, │ │ │ │ │ + setEvent: function(evt) { │ │ │ │ │ + this.evt = evt; │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.control = this.map = null │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.ArcGISCache = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ - url: null, │ │ │ │ │ - tileOrigin: null, │ │ │ │ │ - tileSize: new OpenLayers.Size(256, 256), │ │ │ │ │ - useArcGISServer: true, │ │ │ │ │ - type: "png", │ │ │ │ │ - useScales: false, │ │ │ │ │ - overrideDPI: false, │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (this.resolutions) { │ │ │ │ │ - this.serverResolutions = this.resolutions; │ │ │ │ │ - this.maxExtent = this.getMaxExtentForResolution(this.resolutions[0]) │ │ │ │ │ +OpenLayers.Handler.MOD_NONE = 0; │ │ │ │ │ +OpenLayers.Handler.MOD_SHIFT = 1; │ │ │ │ │ +OpenLayers.Handler.MOD_CTRL = 2; │ │ │ │ │ +OpenLayers.Handler.MOD_ALT = 4; │ │ │ │ │ +OpenLayers.Handler.MOD_META = 8; │ │ │ │ │ +OpenLayers.Spherical = OpenLayers.Spherical || {}; │ │ │ │ │ +OpenLayers.Spherical.DEFAULT_RADIUS = 6378137; │ │ │ │ │ +OpenLayers.Spherical.computeDistanceBetween = function(from, to, radius) { │ │ │ │ │ + var R = radius || OpenLayers.Spherical.DEFAULT_RADIUS; │ │ │ │ │ + var sinHalfDeltaLon = Math.sin(Math.PI * (to.lon - from.lon) / 360); │ │ │ │ │ + var sinHalfDeltaLat = Math.sin(Math.PI * (to.lat - from.lat) / 360); │ │ │ │ │ + var a = sinHalfDeltaLat * sinHalfDeltaLat + sinHalfDeltaLon * sinHalfDeltaLon * Math.cos(Math.PI * from.lat / 180) * Math.cos(Math.PI * to.lat / 180); │ │ │ │ │ + return 2 * R * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a)) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Spherical.computeHeading = function(from, to) { │ │ │ │ │ + var y = Math.sin(Math.PI * (from.lon - to.lon) / 180) * Math.cos(Math.PI * to.lat / 180); │ │ │ │ │ + var x = Math.cos(Math.PI * from.lat / 180) * Math.sin(Math.PI * to.lat / 180) - Math.sin(Math.PI * from.lat / 180) * Math.cos(Math.PI * to.lat / 180) * Math.cos(Math.PI * (from.lon - to.lon) / 180); │ │ │ │ │ + return 180 * Math.atan2(y, x) / Math.PI │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Handler.Drag = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + started: false, │ │ │ │ │ + stopDown: true, │ │ │ │ │ + dragging: false, │ │ │ │ │ + last: null, │ │ │ │ │ + start: null, │ │ │ │ │ + lastMoveEvt: null, │ │ │ │ │ + oldOnselectstart: null, │ │ │ │ │ + interval: 0, │ │ │ │ │ + timeoutId: null, │ │ │ │ │ + documentDrag: false, │ │ │ │ │ + documentEvents: null, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (this.documentDrag === true) { │ │ │ │ │ + var me = this; │ │ │ │ │ + this._docMove = function(evt) { │ │ │ │ │ + me.mousemove({ │ │ │ │ │ + xy: { │ │ │ │ │ + x: evt.clientX, │ │ │ │ │ + y: evt.clientY │ │ │ │ │ + }, │ │ │ │ │ + element: document │ │ │ │ │ + }) │ │ │ │ │ + }; │ │ │ │ │ + this._docUp = function(evt) { │ │ │ │ │ + me.mouseup({ │ │ │ │ │ + xy: { │ │ │ │ │ + x: evt.clientX, │ │ │ │ │ + y: evt.clientY │ │ │ │ │ + } │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (this.layerInfo) { │ │ │ │ │ - var info = this.layerInfo; │ │ │ │ │ - var startingTileExtent = new OpenLayers.Bounds(info.fullExtent.xmin, info.fullExtent.ymin, info.fullExtent.xmax, info.fullExtent.ymax); │ │ │ │ │ - this.projection = "EPSG:" + info.spatialReference.wkid; │ │ │ │ │ - this.sphericalMercator = info.spatialReference.wkid == 102100; │ │ │ │ │ - this.units = info.units == "esriFeet" ? "ft" : "m"; │ │ │ │ │ - if (!!info.tileInfo) { │ │ │ │ │ - this.tileSize = new OpenLayers.Size(info.tileInfo.width || info.tileInfo.cols, info.tileInfo.height || info.tileInfo.rows); │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(info.tileInfo.origin.x, info.tileInfo.origin.y); │ │ │ │ │ - var upperLeft = new OpenLayers.Geometry.Point(startingTileExtent.left, startingTileExtent.top); │ │ │ │ │ - var bottomRight = new OpenLayers.Geometry.Point(startingTileExtent.right, startingTileExtent.bottom); │ │ │ │ │ - if (this.useScales) { │ │ │ │ │ - this.scales = [] │ │ │ │ │ + }, │ │ │ │ │ + dragstart: function(evt) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + if (this.checkModifiers(evt) && (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt))) { │ │ │ │ │ + this.started = true; │ │ │ │ │ + this.start = evt.xy; │ │ │ │ │ + this.last = evt.xy; │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ + this.down(evt); │ │ │ │ │ + this.callback("down", [evt.xy]); │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + if (!this.oldOnselectstart) { │ │ │ │ │ + this.oldOnselectstart = document.onselectstart ? document.onselectstart : OpenLayers.Function.True │ │ │ │ │ + } │ │ │ │ │ + document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ + propagate = !this.stopDown │ │ │ │ │ + } else { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null │ │ │ │ │ + } │ │ │ │ │ + return propagate │ │ │ │ │ + }, │ │ │ │ │ + dragmove: function(evt) { │ │ │ │ │ + this.lastMoveEvt = evt; │ │ │ │ │ + if (this.started && !this.timeoutId && (evt.xy.x != this.last.x || evt.xy.y != this.last.y)) { │ │ │ │ │ + if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ + if (evt.element === document) { │ │ │ │ │ + this.adjustXY(evt); │ │ │ │ │ + this.setEvent(evt) │ │ │ │ │ } else { │ │ │ │ │ - this.resolutions = [] │ │ │ │ │ + this.removeDocumentEvents() │ │ │ │ │ } │ │ │ │ │ - this.lods = []; │ │ │ │ │ - for (var key in info.tileInfo.lods) { │ │ │ │ │ - if (info.tileInfo.lods.hasOwnProperty(key)) { │ │ │ │ │ - var lod = info.tileInfo.lods[key]; │ │ │ │ │ - if (this.useScales) { │ │ │ │ │ - this.scales.push(lod.scale) │ │ │ │ │ - } else { │ │ │ │ │ - this.resolutions.push(lod.resolution) │ │ │ │ │ - } │ │ │ │ │ - var start = this.getContainingTileCoords(upperLeft, lod.resolution); │ │ │ │ │ - lod.startTileCol = start.x; │ │ │ │ │ - lod.startTileRow = start.y; │ │ │ │ │ - var end = this.getContainingTileCoords(bottomRight, lod.resolution); │ │ │ │ │ - lod.endTileCol = end.x; │ │ │ │ │ - lod.endTileRow = end.y; │ │ │ │ │ - this.lods.push(lod) │ │ │ │ │ - } │ │ │ │ │ + } │ │ │ │ │ + if (this.interval > 0) { │ │ │ │ │ + this.timeoutId = setTimeout(OpenLayers.Function.bind(this.removeTimeout, this), this.interval) │ │ │ │ │ + } │ │ │ │ │ + this.dragging = true; │ │ │ │ │ + this.move(evt); │ │ │ │ │ + this.callback("move", [evt.xy]); │ │ │ │ │ + if (!this.oldOnselectstart) { │ │ │ │ │ + this.oldOnselectstart = document.onselectstart; │ │ │ │ │ + document.onselectstart = OpenLayers.Function.False │ │ │ │ │ + } │ │ │ │ │ + this.last = evt.xy │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + dragend: function(evt) { │ │ │ │ │ + if (this.started) { │ │ │ │ │ + if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ + this.adjustXY(evt); │ │ │ │ │ + this.removeDocumentEvents() │ │ │ │ │ + } │ │ │ │ │ + var dragged = this.start != this.last; │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ + this.up(evt); │ │ │ │ │ + this.callback("up", [evt.xy]); │ │ │ │ │ + if (dragged) { │ │ │ │ │ + this.callback("done", [evt.xy]) │ │ │ │ │ + } │ │ │ │ │ + document.onselectstart = this.oldOnselectstart │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + down: function(evt) {}, │ │ │ │ │ + move: function(evt) {}, │ │ │ │ │ + up: function(evt) {}, │ │ │ │ │ + out: function(evt) {}, │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + return this.dragstart(evt) │ │ │ │ │ + }, │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + return this.dragstart(evt) │ │ │ │ │ + }, │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + return this.dragmove(evt) │ │ │ │ │ + }, │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + return this.dragmove(evt) │ │ │ │ │ + }, │ │ │ │ │ + removeTimeout: function() { │ │ │ │ │ + this.timeoutId = null; │ │ │ │ │ + if (this.dragging) { │ │ │ │ │ + this.mousemove(this.lastMoveEvt) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + return this.dragend(evt) │ │ │ │ │ + }, │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + evt.xy = this.last; │ │ │ │ │ + return this.dragend(evt) │ │ │ │ │ + }, │ │ │ │ │ + mouseout: function(evt) { │ │ │ │ │ + if (this.started && OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ + if (this.documentDrag === true) { │ │ │ │ │ + this.addDocumentEvents() │ │ │ │ │ + } else { │ │ │ │ │ + var dragged = this.start != this.last; │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ + this.out(evt); │ │ │ │ │ + this.callback("out", []); │ │ │ │ │ + if (dragged) { │ │ │ │ │ + this.callback("done", [evt.xy]) │ │ │ │ │ } │ │ │ │ │ - this.maxExtent = this.calculateMaxExtentWithLOD(this.lods[0]); │ │ │ │ │ - this.serverResolutions = this.resolutions; │ │ │ │ │ - if (this.overrideDPI && info.tileInfo.dpi) { │ │ │ │ │ - OpenLayers.DOTS_PER_INCH = info.tileInfo.dpi │ │ │ │ │ + if (document.onselectstart) { │ │ │ │ │ + document.onselectstart = this.oldOnselectstart │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - getContainingTileCoords: function(point, res) { │ │ │ │ │ - return new OpenLayers.Pixel(Math.max(Math.floor((point.x - this.tileOrigin.lon) / (this.tileSize.w * res)), 0), Math.max(Math.floor((this.tileOrigin.lat - point.y) / (this.tileSize.h * res)), 0)) │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + return this.start == this.last │ │ │ │ │ }, │ │ │ │ │ - calculateMaxExtentWithLOD: function(lod) { │ │ │ │ │ - var numTileCols = lod.endTileCol - lod.startTileCol + 1; │ │ │ │ │ - var numTileRows = lod.endTileRow - lod.startTileRow + 1; │ │ │ │ │ - var minX = this.tileOrigin.lon + lod.startTileCol * this.tileSize.w * lod.resolution; │ │ │ │ │ - var maxX = minX + numTileCols * this.tileSize.w * lod.resolution; │ │ │ │ │ - var maxY = this.tileOrigin.lat - lod.startTileRow * this.tileSize.h * lod.resolution; │ │ │ │ │ - var minY = maxY - numTileRows * this.tileSize.h * lod.resolution; │ │ │ │ │ - return new OpenLayers.Bounds(minX, minY, maxX, maxY) │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + activated = true │ │ │ │ │ + } │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - calculateMaxExtentWithExtent: function(extent, res) { │ │ │ │ │ - var upperLeft = new OpenLayers.Geometry.Point(extent.left, extent.top); │ │ │ │ │ - var bottomRight = new OpenLayers.Geometry.Point(extent.right, extent.bottom); │ │ │ │ │ - var start = this.getContainingTileCoords(upperLeft, res); │ │ │ │ │ - var end = this.getContainingTileCoords(bottomRight, res); │ │ │ │ │ - var lod = { │ │ │ │ │ - resolution: res, │ │ │ │ │ - startTileCol: start.x, │ │ │ │ │ - startTileRow: start.y, │ │ │ │ │ - endTileCol: end.x, │ │ │ │ │ - endTileRow: end.y │ │ │ │ │ - }; │ │ │ │ │ - return this.calculateMaxExtentWithLOD(lod) │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + deactivated = true; │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown") │ │ │ │ │ + } │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - getUpperLeftTileCoord: function(res) { │ │ │ │ │ - var upperLeft = new OpenLayers.Geometry.Point(this.maxExtent.left, this.maxExtent.top); │ │ │ │ │ - return this.getContainingTileCoords(upperLeft, res) │ │ │ │ │ + adjustXY: function(evt) { │ │ │ │ │ + var pos = OpenLayers.Util.pagePosition(this.map.viewPortDiv); │ │ │ │ │ + evt.xy.x -= pos[0]; │ │ │ │ │ + evt.xy.y -= pos[1] │ │ │ │ │ }, │ │ │ │ │ - getLowerRightTileCoord: function(res) { │ │ │ │ │ - var bottomRight = new OpenLayers.Geometry.Point(this.maxExtent.right, this.maxExtent.bottom); │ │ │ │ │ - return this.getContainingTileCoords(bottomRight, res) │ │ │ │ │ + addDocumentEvents: function() { │ │ │ │ │ + OpenLayers.Element.addClass(document.body, "olDragDown"); │ │ │ │ │ + this.documentEvents = true; │ │ │ │ │ + OpenLayers.Event.observe(document, "mousemove", this._docMove); │ │ │ │ │ + OpenLayers.Event.observe(document, "mouseup", this._docUp) │ │ │ │ │ }, │ │ │ │ │ - getMaxExtentForResolution: function(res) { │ │ │ │ │ - var start = this.getUpperLeftTileCoord(res); │ │ │ │ │ - var end = this.getLowerRightTileCoord(res); │ │ │ │ │ - var numTileCols = end.x - start.x + 1; │ │ │ │ │ - var numTileRows = end.y - start.y + 1; │ │ │ │ │ - var minX = this.tileOrigin.lon + start.x * this.tileSize.w * res; │ │ │ │ │ - var maxX = minX + numTileCols * this.tileSize.w * res; │ │ │ │ │ - var maxY = this.tileOrigin.lat - start.y * this.tileSize.h * res; │ │ │ │ │ - var minY = maxY - numTileRows * this.tileSize.h * res; │ │ │ │ │ - return new OpenLayers.Bounds(minX, minY, maxX, maxY) │ │ │ │ │ + removeDocumentEvents: function() { │ │ │ │ │ + OpenLayers.Element.removeClass(document.body, "olDragDown"); │ │ │ │ │ + this.documentEvents = false; │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mousemove", this._docMove); │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mouseup", this._docUp) │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.ArcGISCache(this.name, this.url, this.options) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Drag" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Handler.RegularPolygon = OpenLayers.Class(OpenLayers.Handler.Drag, { │ │ │ │ │ + sides: 4, │ │ │ │ │ + radius: null, │ │ │ │ │ + snapAngle: null, │ │ │ │ │ + snapToggle: "shiftKey", │ │ │ │ │ + layerOptions: null, │ │ │ │ │ + persist: false, │ │ │ │ │ + irregular: false, │ │ │ │ │ + citeCompliant: false, │ │ │ │ │ + angle: null, │ │ │ │ │ + fixedRadius: false, │ │ │ │ │ + feature: null, │ │ │ │ │ + layer: null, │ │ │ │ │ + origin: null, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ + this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style["default"], {}) │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]) │ │ │ │ │ - }, │ │ │ │ │ - initGriddedTiles: function(bounds) { │ │ │ │ │ - delete this._tileOrigin; │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initGriddedTiles.apply(this, arguments) │ │ │ │ │ + OpenLayers.Handler.Drag.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ + this.options = options ? options : {} │ │ │ │ │ }, │ │ │ │ │ - getMaxExtent: function() { │ │ │ │ │ - var resolution = this.map.getResolution(); │ │ │ │ │ - return this.maxExtent = this.getMaxExtentForResolution(resolution) │ │ │ │ │ + setOptions: function(newOptions) { │ │ │ │ │ + OpenLayers.Util.extend(this.options, newOptions); │ │ │ │ │ + OpenLayers.Util.extend(this, newOptions) │ │ │ │ │ }, │ │ │ │ │ - getTileOrigin: function() { │ │ │ │ │ - if (!this._tileOrigin) { │ │ │ │ │ - var extent = this.getMaxExtent(); │ │ │ │ │ - this._tileOrigin = new OpenLayers.LonLat(extent.left, extent.bottom) │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.Drag.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + var options = OpenLayers.Util.extend({ │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + calculateInRange: OpenLayers.Function.True, │ │ │ │ │ + wrapDateLine: this.citeCompliant │ │ │ │ │ + }, this.layerOptions); │ │ │ │ │ + this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ + this.map.addLayer(this.layer); │ │ │ │ │ + activated = true │ │ │ │ │ } │ │ │ │ │ - return this._tileOrigin │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var res = this.getResolution(); │ │ │ │ │ - var originTileX = this.tileOrigin.lon + res * this.tileSize.w / 2; │ │ │ │ │ - var originTileY = this.tileOrigin.lat - res * this.tileSize.h / 2; │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var point = { │ │ │ │ │ - x: center.lon, │ │ │ │ │ - y: center.lat │ │ │ │ │ - }; │ │ │ │ │ - var x = Math.round(Math.abs((center.lon - originTileX) / (res * this.tileSize.w))); │ │ │ │ │ - var y = Math.round(Math.abs((originTileY - center.lat) / (res * this.tileSize.h))); │ │ │ │ │ - var z = this.map.getZoom(); │ │ │ │ │ - if (this.lods) { │ │ │ │ │ - var lod = this.lods[this.map.getZoom()]; │ │ │ │ │ - if (x < lod.startTileCol || x > lod.endTileCol || (y < lod.startTileRow || y > lod.endTileRow)) { │ │ │ │ │ - return null │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.Drag.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + if (this.dragging) { │ │ │ │ │ + this.cancel() │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - var start = this.getUpperLeftTileCoord(res); │ │ │ │ │ - var end = this.getLowerRightTileCoord(res); │ │ │ │ │ - if (x < start.x || x >= end.x || (y < start.y || y >= end.y)) { │ │ │ │ │ - return null │ │ │ │ │ + if (this.layer.map != null) { │ │ │ │ │ + this.layer.destroy(false); │ │ │ │ │ + if (this.feature) { │ │ │ │ │ + this.feature.destroy() │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + this.layer = null; │ │ │ │ │ + this.feature = null; │ │ │ │ │ + deactivated = true │ │ │ │ │ } │ │ │ │ │ - var url = this.url; │ │ │ │ │ - var s = "" + x + y + z; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(s, url) │ │ │ │ │ + return deactivated │ │ │ │ │ + }, │ │ │ │ │ + down: function(evt) { │ │ │ │ │ + this.fixedRadius = !!this.radius; │ │ │ │ │ + var maploc = this.layer.getLonLatFromViewPortPx(evt.xy); │ │ │ │ │ + this.origin = new OpenLayers.Geometry.Point(maploc.lon, maploc.lat); │ │ │ │ │ + if (!this.fixedRadius || this.irregular) { │ │ │ │ │ + this.radius = this.map.getResolution() │ │ │ │ │ } │ │ │ │ │ - if (this.useArcGISServer) { │ │ │ │ │ - url = url + "/tile/${z}/${y}/${x}" │ │ │ │ │ - } else { │ │ │ │ │ - x = "C" + OpenLayers.Number.zeroPad(x, 8, 16); │ │ │ │ │ - y = "R" + OpenLayers.Number.zeroPad(y, 8, 16); │ │ │ │ │ - z = "L" + OpenLayers.Number.zeroPad(z, 2, 10); │ │ │ │ │ - url = url + "/${z}/${y}/${x}." + this.type │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.clear() │ │ │ │ │ } │ │ │ │ │ - url = OpenLayers.String.format(url, { │ │ │ │ │ - x: x, │ │ │ │ │ - y: y, │ │ │ │ │ - z: z │ │ │ │ │ + this.feature = new OpenLayers.Feature.Vector; │ │ │ │ │ + this.createGeometry(); │ │ │ │ │ + this.callback("create", [this.origin, this.feature]); │ │ │ │ │ + this.layer.addFeatures([this.feature], { │ │ │ │ │ + silent: true │ │ │ │ │ }); │ │ │ │ │ - return OpenLayers.Util.urlAppend(url, OpenLayers.Util.getParameterString(this.params)) │ │ │ │ │ + this.layer.drawFeature(this.feature, this.style) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.ArcGISCache" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.ArcXML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - fontStyleKeys: ["antialiasing", "blockout", "font", "fontcolor", "fontsize", "fontstyle", "glowing", "interval", "outline", "printmode", "shadow", "transparency"], │ │ │ │ │ - request: null, │ │ │ │ │ - response: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.request = new OpenLayers.Format.ArcXML.Request; │ │ │ │ │ - this.response = new OpenLayers.Format.ArcXML.Response; │ │ │ │ │ - if (options) { │ │ │ │ │ - if (options.requesttype == "feature") { │ │ │ │ │ - this.request.get_image = null; │ │ │ │ │ - var qry = this.request.get_feature.query; │ │ │ │ │ - this.addCoordSys(qry.featurecoordsys, options.featureCoordSys); │ │ │ │ │ - this.addCoordSys(qry.filtercoordsys, options.filterCoordSys); │ │ │ │ │ - if (options.polygon) { │ │ │ │ │ - qry.isspatial = true; │ │ │ │ │ - qry.spatialfilter.polygon = options.polygon │ │ │ │ │ - } else if (options.envelope) { │ │ │ │ │ - qry.isspatial = true; │ │ │ │ │ - qry.spatialfilter.envelope = { │ │ │ │ │ - minx: 0, │ │ │ │ │ - miny: 0, │ │ │ │ │ - maxx: 0, │ │ │ │ │ - maxy: 0 │ │ │ │ │ - }; │ │ │ │ │ - this.parseEnvelope(qry.spatialfilter.envelope, options.envelope) │ │ │ │ │ - } │ │ │ │ │ - } else if (options.requesttype == "image") { │ │ │ │ │ - this.request.get_feature = null; │ │ │ │ │ - var props = this.request.get_image.properties; │ │ │ │ │ - this.parseEnvelope(props.envelope, options.envelope); │ │ │ │ │ - this.addLayers(props.layerlist, options.layers); │ │ │ │ │ - this.addImageSize(props.imagesize, options.tileSize); │ │ │ │ │ - this.addCoordSys(props.featurecoordsys, options.featureCoordSys); │ │ │ │ │ - this.addCoordSys(props.filtercoordsys, options.filterCoordSys) │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + var maploc = this.layer.getLonLatFromViewPortPx(evt.xy); │ │ │ │ │ + var point = new OpenLayers.Geometry.Point(maploc.lon, maploc.lat); │ │ │ │ │ + if (this.irregular) { │ │ │ │ │ + var ry = Math.sqrt(2) * Math.abs(point.y - this.origin.y) / 2; │ │ │ │ │ + this.radius = Math.max(this.map.getResolution() / 2, ry) │ │ │ │ │ + } else if (this.fixedRadius) { │ │ │ │ │ + this.origin = point │ │ │ │ │ + } else { │ │ │ │ │ + this.calculateAngle(point, evt); │ │ │ │ │ + this.radius = Math.max(this.map.getResolution() / 2, point.distanceTo(this.origin)) │ │ │ │ │ + } │ │ │ │ │ + this.modifyGeometry(); │ │ │ │ │ + if (this.irregular) { │ │ │ │ │ + var dx = point.x - this.origin.x; │ │ │ │ │ + var dy = point.y - this.origin.y; │ │ │ │ │ + var ratio; │ │ │ │ │ + if (dy == 0) { │ │ │ │ │ + ratio = dx / (this.radius * Math.sqrt(2)) │ │ │ │ │ } else { │ │ │ │ │ - this.request = null │ │ │ │ │ + ratio = dx / dy │ │ │ │ │ } │ │ │ │ │ + this.feature.geometry.resize(1, this.origin, ratio); │ │ │ │ │ + this.feature.geometry.move(dx / 2, dy / 2) │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ + this.layer.drawFeature(this.feature, this.style) │ │ │ │ │ }, │ │ │ │ │ - parseEnvelope: function(env, arr) { │ │ │ │ │ - if (arr && arr.length == 4) { │ │ │ │ │ - env.minx = arr[0]; │ │ │ │ │ - env.miny = arr[1]; │ │ │ │ │ - env.maxx = arr[2]; │ │ │ │ │ - env.maxy = arr[3] │ │ │ │ │ + up: function(evt) { │ │ │ │ │ + this.finalize(); │ │ │ │ │ + if (this.start == this.last) { │ │ │ │ │ + this.callback("done", [evt.xy]) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - addLayers: function(ll, lyrs) { │ │ │ │ │ - for (var lind = 0, len = lyrs.length; lind < len; lind++) { │ │ │ │ │ - ll.push(lyrs[lind]) │ │ │ │ │ - } │ │ │ │ │ + out: function(evt) { │ │ │ │ │ + this.finalize() │ │ │ │ │ }, │ │ │ │ │ - addImageSize: function(imsize, olsize) { │ │ │ │ │ - if (olsize !== null) { │ │ │ │ │ - imsize.width = olsize.w; │ │ │ │ │ - imsize.height = olsize.h; │ │ │ │ │ - imsize.printwidth = olsize.w; │ │ │ │ │ - imsize.printheight = olsize.h │ │ │ │ │ + createGeometry: function() { │ │ │ │ │ + this.angle = Math.PI * (1 / this.sides - 1 / 2); │ │ │ │ │ + if (this.snapAngle) { │ │ │ │ │ + this.angle += this.snapAngle * (Math.PI / 180) │ │ │ │ │ } │ │ │ │ │ + this.feature.geometry = OpenLayers.Geometry.Polygon.createRegularPolygon(this.origin, this.radius, this.sides, this.snapAngle) │ │ │ │ │ }, │ │ │ │ │ - addCoordSys: function(featOrFilt, fsys) { │ │ │ │ │ - if (typeof fsys == "string") { │ │ │ │ │ - featOrFilt.id = parseInt(fsys); │ │ │ │ │ - featOrFilt.string = fsys │ │ │ │ │ - } else if (typeof fsys == "object" && fsys.proj !== null) { │ │ │ │ │ - featOrFilt.id = fsys.proj.srsProjNumber; │ │ │ │ │ - featOrFilt.string = fsys.proj.srsCode │ │ │ │ │ - } else { │ │ │ │ │ - featOrFilt = fsys │ │ │ │ │ + modifyGeometry: function() { │ │ │ │ │ + var angle, point; │ │ │ │ │ + var ring = this.feature.geometry.components[0]; │ │ │ │ │ + if (ring.components.length != this.sides + 1) { │ │ │ │ │ + this.createGeometry(); │ │ │ │ │ + ring = this.feature.geometry.components[0] │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0; i < this.sides; ++i) { │ │ │ │ │ + point = ring.components[i]; │ │ │ │ │ + angle = this.angle + i * 2 * Math.PI / this.sides; │ │ │ │ │ + point.x = this.origin.x + this.radius * Math.cos(angle); │ │ │ │ │ + point.y = this.origin.y + this.radius * Math.sin(angle); │ │ │ │ │ + point.clearBounds() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - iserror: function(data) { │ │ │ │ │ - var ret = null; │ │ │ │ │ - if (!data) { │ │ │ │ │ - ret = this.response.error !== "" │ │ │ │ │ + calculateAngle: function(point, evt) { │ │ │ │ │ + var alpha = Math.atan2(point.y - this.origin.y, point.x - this.origin.x); │ │ │ │ │ + if (this.snapAngle && (this.snapToggle && !evt[this.snapToggle])) { │ │ │ │ │ + var snapAngleRad = Math.PI / 180 * this.snapAngle; │ │ │ │ │ + this.angle = Math.round(alpha / snapAngleRad) * snapAngleRad │ │ │ │ │ } else { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - var errorNodes = data.documentElement.getElementsByTagName("ERROR"); │ │ │ │ │ - ret = errorNodes !== null && errorNodes.length > 0 │ │ │ │ │ + this.angle = alpha │ │ │ │ │ } │ │ │ │ │ - return ret │ │ │ │ │ }, │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + cancel: function() { │ │ │ │ │ + this.callback("cancel", null); │ │ │ │ │ + this.finalize() │ │ │ │ │ + }, │ │ │ │ │ + finalize: function() { │ │ │ │ │ + this.origin = null; │ │ │ │ │ + this.radius = this.options.radius │ │ │ │ │ + }, │ │ │ │ │ + clear: function() { │ │ │ │ │ + if (this.layer) { │ │ │ │ │ + this.layer.renderer.clear(); │ │ │ │ │ + this.layer.destroyFeatures() │ │ │ │ │ } │ │ │ │ │ - var arcNode = null; │ │ │ │ │ - if (data && data.documentElement) { │ │ │ │ │ - if (data.documentElement.nodeName == "ARCXML") { │ │ │ │ │ - arcNode = data.documentElement │ │ │ │ │ - } else { │ │ │ │ │ - arcNode = data.documentElement.getElementsByTagName("ARCXML")[0] │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ + callback: function(name, args) { │ │ │ │ │ + if (this.callbacks[name]) { │ │ │ │ │ + this.callbacks[name].apply(this.control, [this.feature.geometry.clone()]) │ │ │ │ │ } │ │ │ │ │ - if (!arcNode || arcNode.firstChild.nodeName === "parsererror") { │ │ │ │ │ - var error, source; │ │ │ │ │ - try { │ │ │ │ │ - error = data.firstChild.nodeValue; │ │ │ │ │ - source = data.firstChild.childNodes[1].firstChild.nodeValue │ │ │ │ │ - } catch (err) {} │ │ │ │ │ - throw { │ │ │ │ │ - message: "Error parsing the ArcXML request", │ │ │ │ │ - error: error, │ │ │ │ │ - source: source │ │ │ │ │ - } │ │ │ │ │ + if (!this.persist && (name == "done" || name == "cancel")) { │ │ │ │ │ + this.clear() │ │ │ │ │ } │ │ │ │ │ - var response = this.parseResponse(arcNode); │ │ │ │ │ - return response │ │ │ │ │ }, │ │ │ │ │ - write: function(request) { │ │ │ │ │ - if (!request) { │ │ │ │ │ - request = this.request │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.RegularPolygon" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Handler.Point = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + point: null, │ │ │ │ │ + layer: null, │ │ │ │ │ + multi: false, │ │ │ │ │ + citeCompliant: false, │ │ │ │ │ + mouseDown: false, │ │ │ │ │ + stoppedDown: null, │ │ │ │ │ + lastDown: null, │ │ │ │ │ + lastUp: null, │ │ │ │ │ + persist: false, │ │ │ │ │ + stopDown: false, │ │ │ │ │ + stopUp: false, │ │ │ │ │ + layerOptions: null, │ │ │ │ │ + pixelTolerance: 5, │ │ │ │ │ + lastTouchPx: null, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ + this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style["default"], {}) │ │ │ │ │ } │ │ │ │ │ - var root = this.createElementNS("", "ARCXML"); │ │ │ │ │ - root.setAttribute("version", "1.1"); │ │ │ │ │ - var reqElem = this.createElementNS("", "REQUEST"); │ │ │ │ │ - if (request.get_image != null) { │ │ │ │ │ - var getElem = this.createElementNS("", "GET_IMAGE"); │ │ │ │ │ - reqElem.appendChild(getElem); │ │ │ │ │ - var propElem = this.createElementNS("", "PROPERTIES"); │ │ │ │ │ - getElem.appendChild(propElem); │ │ │ │ │ - var props = request.get_image.properties; │ │ │ │ │ - if (props.featurecoordsys != null) { │ │ │ │ │ - var feat = this.createElementNS("", "FEATURECOORDSYS"); │ │ │ │ │ - propElem.appendChild(feat); │ │ │ │ │ - if (props.featurecoordsys.id === 0) { │ │ │ │ │ - feat.setAttribute("string", props.featurecoordsys["string"]) │ │ │ │ │ - } else { │ │ │ │ │ - feat.setAttribute("id", props.featurecoordsys.id) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (props.filtercoordsys != null) { │ │ │ │ │ - var filt = this.createElementNS("", "FILTERCOORDSYS"); │ │ │ │ │ - propElem.appendChild(filt); │ │ │ │ │ - if (props.filtercoordsys.id === 0) { │ │ │ │ │ - filt.setAttribute("string", props.filtercoordsys.string) │ │ │ │ │ - } else { │ │ │ │ │ - filt.setAttribute("id", props.filtercoordsys.id) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (props.envelope != null) { │ │ │ │ │ - var env = this.createElementNS("", "ENVELOPE"); │ │ │ │ │ - propElem.appendChild(env); │ │ │ │ │ - env.setAttribute("minx", props.envelope.minx); │ │ │ │ │ - env.setAttribute("miny", props.envelope.miny); │ │ │ │ │ - env.setAttribute("maxx", props.envelope.maxx); │ │ │ │ │ - env.setAttribute("maxy", props.envelope.maxy) │ │ │ │ │ - } │ │ │ │ │ - var imagesz = this.createElementNS("", "IMAGESIZE"); │ │ │ │ │ - propElem.appendChild(imagesz); │ │ │ │ │ - imagesz.setAttribute("height", props.imagesize.height); │ │ │ │ │ - imagesz.setAttribute("width", props.imagesize.width); │ │ │ │ │ - if (props.imagesize.height != props.imagesize.printheight || props.imagesize.width != props.imagesize.printwidth) { │ │ │ │ │ - imagesz.setAttribute("printheight", props.imagesize.printheight); │ │ │ │ │ - imagesz.setArrtibute("printwidth", props.imagesize.printwidth) │ │ │ │ │ - } │ │ │ │ │ - if (props.background != null) { │ │ │ │ │ - var backgrnd = this.createElementNS("", "BACKGROUND"); │ │ │ │ │ - propElem.appendChild(backgrnd); │ │ │ │ │ - backgrnd.setAttribute("color", props.background.color.r + "," + props.background.color.g + "," + props.background.color.b); │ │ │ │ │ - if (props.background.transcolor !== null) { │ │ │ │ │ - backgrnd.setAttribute("transcolor", props.background.transcolor.r + "," + props.background.transcolor.g + "," + props.background.transcolor.b) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (props.layerlist != null && props.layerlist.length > 0) { │ │ │ │ │ - var layerlst = this.createElementNS("", "LAYERLIST"); │ │ │ │ │ - propElem.appendChild(layerlst); │ │ │ │ │ - for (var ld = 0; ld < props.layerlist.length; ld++) { │ │ │ │ │ - var ldef = this.createElementNS("", "LAYERDEF"); │ │ │ │ │ - layerlst.appendChild(ldef); │ │ │ │ │ - ldef.setAttribute("id", props.layerlist[ld].id); │ │ │ │ │ - ldef.setAttribute("visible", props.layerlist[ld].visible); │ │ │ │ │ - if (typeof props.layerlist[ld].query == "object") { │ │ │ │ │ - var query = props.layerlist[ld].query; │ │ │ │ │ - if (query.where.length < 0) { │ │ │ │ │ - continue │ │ │ │ │ - } │ │ │ │ │ - var queryElem = null; │ │ │ │ │ - if (typeof query.spatialfilter == "boolean" && query.spatialfilter) { │ │ │ │ │ - queryElem = this.createElementNS("", "SPATIALQUERY") │ │ │ │ │ - } else { │ │ │ │ │ - queryElem = this.createElementNS("", "QUERY") │ │ │ │ │ - } │ │ │ │ │ - queryElem.setAttribute("where", query.where); │ │ │ │ │ - if (typeof query.accuracy == "number" && query.accuracy > 0) { │ │ │ │ │ - queryElem.setAttribute("accuracy", query.accuracy) │ │ │ │ │ - } │ │ │ │ │ - if (typeof query.featurelimit == "number" && query.featurelimit < 2e3) { │ │ │ │ │ - queryElem.setAttribute("featurelimit", query.featurelimit) │ │ │ │ │ - } │ │ │ │ │ - if (typeof query.subfields == "string" && query.subfields != "#ALL#") { │ │ │ │ │ - queryElem.setAttribute("subfields", query.subfields) │ │ │ │ │ - } │ │ │ │ │ - if (typeof query.joinexpression == "string" && query.joinexpression.length > 0) { │ │ │ │ │ - queryElem.setAttribute("joinexpression", query.joinexpression) │ │ │ │ │ - } │ │ │ │ │ - if (typeof query.jointables == "string" && query.jointables.length > 0) { │ │ │ │ │ - queryElem.setAttribute("jointables", query.jointables) │ │ │ │ │ - } │ │ │ │ │ - ldef.appendChild(queryElem) │ │ │ │ │ - } │ │ │ │ │ - if (typeof props.layerlist[ld].renderer == "object") { │ │ │ │ │ - this.addRenderer(ldef, props.layerlist[ld].renderer) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else if (request.get_feature != null) { │ │ │ │ │ - var getElem = this.createElementNS("", "GET_FEATURES"); │ │ │ │ │ - getElem.setAttribute("outputmode", "newxml"); │ │ │ │ │ - getElem.setAttribute("checkesc", "true"); │ │ │ │ │ - if (request.get_feature.geometry) { │ │ │ │ │ - getElem.setAttribute("geometry", request.get_feature.geometry) │ │ │ │ │ - } else { │ │ │ │ │ - getElem.setAttribute("geometry", "false") │ │ │ │ │ - } │ │ │ │ │ - if (request.get_feature.compact) { │ │ │ │ │ - getElem.setAttribute("compact", request.get_feature.compact) │ │ │ │ │ - } │ │ │ │ │ - if (request.get_feature.featurelimit == "number") { │ │ │ │ │ - getElem.setAttribute("featurelimit", request.get_feature.featurelimit) │ │ │ │ │ - } │ │ │ │ │ - getElem.setAttribute("globalenvelope", "true"); │ │ │ │ │ - reqElem.appendChild(getElem); │ │ │ │ │ - if (request.get_feature.layer != null && request.get_feature.layer.length > 0) { │ │ │ │ │ - var lyrElem = this.createElementNS("", "LAYER"); │ │ │ │ │ - lyrElem.setAttribute("id", request.get_feature.layer); │ │ │ │ │ - getElem.appendChild(lyrElem) │ │ │ │ │ - } │ │ │ │ │ - var fquery = request.get_feature.query; │ │ │ │ │ - if (fquery != null) { │ │ │ │ │ - var qElem = null; │ │ │ │ │ - if (fquery.isspatial) { │ │ │ │ │ - qElem = this.createElementNS("", "SPATIALQUERY") │ │ │ │ │ - } else { │ │ │ │ │ - qElem = this.createElementNS("", "QUERY") │ │ │ │ │ - } │ │ │ │ │ - getElem.appendChild(qElem); │ │ │ │ │ - if (typeof fquery.accuracy == "number") { │ │ │ │ │ - qElem.setAttribute("accuracy", fquery.accuracy) │ │ │ │ │ - } │ │ │ │ │ - if (fquery.featurecoordsys != null) { │ │ │ │ │ - var fcsElem1 = this.createElementNS("", "FEATURECOORDSYS"); │ │ │ │ │ - if (fquery.featurecoordsys.id == 0) { │ │ │ │ │ - fcsElem1.setAttribute("string", fquery.featurecoordsys.string) │ │ │ │ │ - } else { │ │ │ │ │ - fcsElem1.setAttribute("id", fquery.featurecoordsys.id) │ │ │ │ │ - } │ │ │ │ │ - qElem.appendChild(fcsElem1) │ │ │ │ │ - } │ │ │ │ │ - if (fquery.filtercoordsys != null) { │ │ │ │ │ - var fcsElem2 = this.createElementNS("", "FILTERCOORDSYS"); │ │ │ │ │ - if (fquery.filtercoordsys.id === 0) { │ │ │ │ │ - fcsElem2.setAttribute("string", fquery.filtercoordsys.string) │ │ │ │ │ - } else { │ │ │ │ │ - fcsElem2.setAttribute("id", fquery.filtercoordsys.id) │ │ │ │ │ - } │ │ │ │ │ - qElem.appendChild(fcsElem2) │ │ │ │ │ - } │ │ │ │ │ - if (fquery.buffer > 0) { │ │ │ │ │ - var bufElem = this.createElementNS("", "BUFFER"); │ │ │ │ │ - bufElem.setAttribute("distance", fquery.buffer); │ │ │ │ │ - qElem.appendChild(bufElem) │ │ │ │ │ - } │ │ │ │ │ - if (fquery.isspatial) { │ │ │ │ │ - var spfElem = this.createElementNS("", "SPATIALFILTER"); │ │ │ │ │ - spfElem.setAttribute("relation", fquery.spatialfilter.relation); │ │ │ │ │ - qElem.appendChild(spfElem); │ │ │ │ │ - if (fquery.spatialfilter.envelope) { │ │ │ │ │ - var envElem = this.createElementNS("", "ENVELOPE"); │ │ │ │ │ - envElem.setAttribute("minx", fquery.spatialfilter.envelope.minx); │ │ │ │ │ - envElem.setAttribute("miny", fquery.spatialfilter.envelope.miny); │ │ │ │ │ - envElem.setAttribute("maxx", fquery.spatialfilter.envelope.maxx); │ │ │ │ │ - envElem.setAttribute("maxy", fquery.spatialfilter.envelope.maxy); │ │ │ │ │ - spfElem.appendChild(envElem) │ │ │ │ │ - } else if (typeof fquery.spatialfilter.polygon == "object") { │ │ │ │ │ - spfElem.appendChild(this.writePolygonGeometry(fquery.spatialfilter.polygon)) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (fquery.where != null && fquery.where.length > 0) { │ │ │ │ │ - qElem.setAttribute("where", fquery.where) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (!OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - root.appendChild(reqElem); │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [root]) │ │ │ │ │ + var options = OpenLayers.Util.extend({ │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + calculateInRange: OpenLayers.Function.True, │ │ │ │ │ + wrapDateLine: this.citeCompliant │ │ │ │ │ + }, this.layerOptions); │ │ │ │ │ + this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ + this.map.addLayer(this.layer); │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - addGroupRenderer: function(ldef, toprenderer) { │ │ │ │ │ - var topRelem = this.createElementNS("", "GROUPRENDERER"); │ │ │ │ │ - ldef.appendChild(topRelem); │ │ │ │ │ - for (var rind = 0; rind < toprenderer.length; rind++) { │ │ │ │ │ - var renderer = toprenderer[rind]; │ │ │ │ │ - this.addRenderer(topRelem, renderer) │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.point]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (!OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + this.cancel(); │ │ │ │ │ + if (this.layer.map != null) { │ │ │ │ │ + this.destroyFeature(true); │ │ │ │ │ + this.layer.destroy(false) │ │ │ │ │ } │ │ │ │ │ + this.layer = null; │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - addRenderer: function(topRelem, renderer) { │ │ │ │ │ - if (OpenLayers.Util.isArray(renderer)) { │ │ │ │ │ - this.addGroupRenderer(topRelem, renderer) │ │ │ │ │ - } else { │ │ │ │ │ - var renderElem = this.createElementNS("", renderer.type.toUpperCase() + "RENDERER"); │ │ │ │ │ - topRelem.appendChild(renderElem); │ │ │ │ │ - if (renderElem.tagName == "VALUEMAPRENDERER") { │ │ │ │ │ - this.addValueMapRenderer(renderElem, renderer) │ │ │ │ │ - } else if (renderElem.tagName == "VALUEMAPLABELRENDERER") { │ │ │ │ │ - this.addValueMapLabelRenderer(renderElem, renderer) │ │ │ │ │ - } else if (renderElem.tagName == "SIMPLELABELRENDERER") { │ │ │ │ │ - this.addSimpleLabelRenderer(renderElem, renderer) │ │ │ │ │ - } else if (renderElem.tagName == "SCALEDEPENDENTRENDERER") { │ │ │ │ │ - this.addScaleDependentRenderer(renderElem, renderer) │ │ │ │ │ - } │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + if (this.layer && (force || !this.persist)) { │ │ │ │ │ + this.layer.destroyFeatures() │ │ │ │ │ } │ │ │ │ │ + this.point = null │ │ │ │ │ }, │ │ │ │ │ - addScaleDependentRenderer: function(renderElem, renderer) { │ │ │ │ │ - if (typeof renderer.lower == "string" || typeof renderer.lower == "number") { │ │ │ │ │ - renderElem.setAttribute("lower", renderer.lower) │ │ │ │ │ + destroyPersistedFeature: function() { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + if (layer && layer.features.length > 1) { │ │ │ │ │ + this.layer.features[0].destroy() │ │ │ │ │ } │ │ │ │ │ - if (typeof renderer.upper == "string" || typeof renderer.upper == "number") { │ │ │ │ │ - renderElem.setAttribute("upper", renderer.upper) │ │ │ │ │ + }, │ │ │ │ │ + finalize: function(cancel) { │ │ │ │ │ + var key = cancel ? "cancel" : "done"; │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + this.lastDown = null; │ │ │ │ │ + this.lastUp = null; │ │ │ │ │ + this.lastTouchPx = null; │ │ │ │ │ + this.callback(key, [this.geometryClone()]); │ │ │ │ │ + this.destroyFeature(cancel) │ │ │ │ │ + }, │ │ │ │ │ + cancel: function() { │ │ │ │ │ + this.finalize(true) │ │ │ │ │ + }, │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + modifyFeature: function(pixel) { │ │ │ │ │ + if (!this.point) { │ │ │ │ │ + this.createFeature(pixel) │ │ │ │ │ } │ │ │ │ │ - this.addRenderer(renderElem, renderer.renderer) │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point.geometry.x = lonlat.lon; │ │ │ │ │ + this.point.geometry.y = lonlat.lat; │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.point, false]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.drawFeature() │ │ │ │ │ }, │ │ │ │ │ - addValueMapLabelRenderer: function(renderElem, renderer) { │ │ │ │ │ - renderElem.setAttribute("lookupfield", renderer.lookupfield); │ │ │ │ │ - renderElem.setAttribute("labelfield", renderer.labelfield); │ │ │ │ │ - if (typeof renderer.exacts == "object") { │ │ │ │ │ - for (var ext = 0, extlen = renderer.exacts.length; ext < extlen; ext++) { │ │ │ │ │ - var exact = renderer.exacts[ext]; │ │ │ │ │ - var eelem = this.createElementNS("", "EXACT"); │ │ │ │ │ - if (typeof exact.value == "string") { │ │ │ │ │ - eelem.setAttribute("value", exact.value) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.label == "string") { │ │ │ │ │ - eelem.setAttribute("label", exact.label) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.method == "string") { │ │ │ │ │ - eelem.setAttribute("method", exact.method) │ │ │ │ │ - } │ │ │ │ │ - renderElem.appendChild(eelem); │ │ │ │ │ - if (typeof exact.symbol == "object") { │ │ │ │ │ - var selem = null; │ │ │ │ │ - if (exact.symbol.type == "text") { │ │ │ │ │ - selem = this.createElementNS("", "TEXTSYMBOL") │ │ │ │ │ - } │ │ │ │ │ - if (selem != null) { │ │ │ │ │ - var keys = this.fontStyleKeys; │ │ │ │ │ - for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ - var key = keys[i]; │ │ │ │ │ - if (exact.symbol[key]) { │ │ │ │ │ - selem.setAttribute(key, exact.symbol[key]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - eelem.appendChild(selem) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.point, this.style) │ │ │ │ │ + }, │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.point && this.point.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiPoint([geometry]) │ │ │ │ │ } │ │ │ │ │ + return geometry │ │ │ │ │ }, │ │ │ │ │ - addValueMapRenderer: function(renderElem, renderer) { │ │ │ │ │ - renderElem.setAttribute("lookupfield", renderer.lookupfield); │ │ │ │ │ - if (typeof renderer.ranges == "object") { │ │ │ │ │ - for (var rng = 0, rnglen = renderer.ranges.length; rng < rnglen; rng++) { │ │ │ │ │ - var range = renderer.ranges[rng]; │ │ │ │ │ - var relem = this.createElementNS("", "RANGE"); │ │ │ │ │ - relem.setAttribute("lower", range.lower); │ │ │ │ │ - relem.setAttribute("upper", range.upper); │ │ │ │ │ - renderElem.appendChild(relem); │ │ │ │ │ - if (typeof range.symbol == "object") { │ │ │ │ │ - var selem = null; │ │ │ │ │ - if (range.symbol.type == "simplepolygon") { │ │ │ │ │ - selem = this.createElementNS("", "SIMPLEPOLYGONSYMBOL") │ │ │ │ │ - } │ │ │ │ │ - if (selem != null) { │ │ │ │ │ - if (typeof range.symbol.boundarycolor == "string") { │ │ │ │ │ - selem.setAttribute("boundarycolor", range.symbol.boundarycolor) │ │ │ │ │ - } │ │ │ │ │ - if (typeof range.symbol.fillcolor == "string") { │ │ │ │ │ - selem.setAttribute("fillcolor", range.symbol.fillcolor) │ │ │ │ │ - } │ │ │ │ │ - if (typeof range.symbol.filltransparency == "number") { │ │ │ │ │ - selem.setAttribute("filltransparency", range.symbol.filltransparency) │ │ │ │ │ - } │ │ │ │ │ - relem.appendChild(selem) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else if (typeof renderer.exacts == "object") { │ │ │ │ │ - for (var ext = 0, extlen = renderer.exacts.length; ext < extlen; ext++) { │ │ │ │ │ - var exact = renderer.exacts[ext]; │ │ │ │ │ - var eelem = this.createElementNS("", "EXACT"); │ │ │ │ │ - if (typeof exact.value == "string") { │ │ │ │ │ - eelem.setAttribute("value", exact.value) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.label == "string") { │ │ │ │ │ - eelem.setAttribute("label", exact.label) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.method == "string") { │ │ │ │ │ - eelem.setAttribute("method", exact.method) │ │ │ │ │ - } │ │ │ │ │ - renderElem.appendChild(eelem); │ │ │ │ │ - if (typeof exact.symbol == "object") { │ │ │ │ │ - var selem = null; │ │ │ │ │ - if (exact.symbol.type == "simplemarker") { │ │ │ │ │ - selem = this.createElementNS("", "SIMPLEMARKERSYMBOL") │ │ │ │ │ - } │ │ │ │ │ - if (selem != null) { │ │ │ │ │ - if (typeof exact.symbol.antialiasing == "string") { │ │ │ │ │ - selem.setAttribute("antialiasing", exact.symbol.antialiasing) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.color == "string") { │ │ │ │ │ - selem.setAttribute("color", exact.symbol.color) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.outline == "string") { │ │ │ │ │ - selem.setAttribute("outline", exact.symbol.outline) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.overlap == "string") { │ │ │ │ │ - selem.setAttribute("overlap", exact.symbol.overlap) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.shadow == "string") { │ │ │ │ │ - selem.setAttribute("shadow", exact.symbol.shadow) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.transparency == "number") { │ │ │ │ │ - selem.setAttribute("transparency", exact.symbol.transparency) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.usecentroid == "string") { │ │ │ │ │ - selem.setAttribute("usecentroid", exact.symbol.usecentroid) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.width == "number") { │ │ │ │ │ - selem.setAttribute("width", exact.symbol.width) │ │ │ │ │ - } │ │ │ │ │ - eelem.appendChild(selem) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + geometryClone: function() { │ │ │ │ │ + var geom = this.getGeometry(); │ │ │ │ │ + return geom && geom.clone() │ │ │ │ │ }, │ │ │ │ │ - addSimpleLabelRenderer: function(renderElem, renderer) { │ │ │ │ │ - renderElem.setAttribute("field", renderer.field); │ │ │ │ │ - var keys = ["featureweight", "howmanylabels", "labelbufferratio", "labelpriorities", "labelweight", "linelabelposition", "rotationalangles"]; │ │ │ │ │ - for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ - var key = keys[i]; │ │ │ │ │ - if (renderer[key]) { │ │ │ │ │ - renderElem.setAttribute(key, renderer[key]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (renderer.symbol.type == "text") { │ │ │ │ │ - var symbol = renderer.symbol; │ │ │ │ │ - var selem = this.createElementNS("", "TEXTSYMBOL"); │ │ │ │ │ - renderElem.appendChild(selem); │ │ │ │ │ - var keys = this.fontStyleKeys; │ │ │ │ │ - for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ - var key = keys[i]; │ │ │ │ │ - if (symbol[key]) { │ │ │ │ │ - selem.setAttribute(key, renderer[key]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + return this.down(evt) │ │ │ │ │ }, │ │ │ │ │ - writePolygonGeometry: function(polygon) { │ │ │ │ │ - if (!(polygon instanceof OpenLayers.Geometry.Polygon)) { │ │ │ │ │ - throw { │ │ │ │ │ - message: "Cannot write polygon geometry to ArcXML with an " + polygon.CLASS_NAME + " object.", │ │ │ │ │ - geometry: polygon │ │ │ │ │ - } │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + this.lastTouchPx = evt.xy; │ │ │ │ │ + return this.down(evt) │ │ │ │ │ + }, │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + return this.move(evt) │ │ │ │ │ + }, │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + this.lastTouchPx = evt.xy; │ │ │ │ │ + return this.move(evt) │ │ │ │ │ + }, │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + return this.up(evt) │ │ │ │ │ + }, │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + evt.xy = this.lastTouchPx; │ │ │ │ │ + return this.up(evt) │ │ │ │ │ + }, │ │ │ │ │ + down: function(evt) { │ │ │ │ │ + this.mouseDown = true; │ │ │ │ │ + this.lastDown = evt.xy; │ │ │ │ │ + if (!this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy) │ │ │ │ │ } │ │ │ │ │ - var polyElem = this.createElementNS("", "POLYGON"); │ │ │ │ │ - for (var ln = 0, lnlen = polygon.components.length; ln < lnlen; ln++) { │ │ │ │ │ - var ring = polygon.components[ln]; │ │ │ │ │ - var ringElem = this.createElementNS("", "RING"); │ │ │ │ │ - for (var rn = 0, rnlen = ring.components.length; rn < rnlen; rn++) { │ │ │ │ │ - var point = ring.components[rn]; │ │ │ │ │ - var pointElem = this.createElementNS("", "POINT"); │ │ │ │ │ - pointElem.setAttribute("x", point.x); │ │ │ │ │ - pointElem.setAttribute("y", point.y); │ │ │ │ │ - ringElem.appendChild(pointElem) │ │ │ │ │ - } │ │ │ │ │ - polyElem.appendChild(ringElem) │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + return !this.stopDown │ │ │ │ │ + }, │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ + this.modifyFeature(evt.xy) │ │ │ │ │ } │ │ │ │ │ - return polyElem │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - parseResponse: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - var newData = new OpenLayers.Format.XML; │ │ │ │ │ - data = newData.read(data) │ │ │ │ │ + up: function(evt) { │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + if (!this.checkModifiers(evt)) { │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ - var response = new OpenLayers.Format.ArcXML.Response; │ │ │ │ │ - var errorNode = data.getElementsByTagName("ERROR"); │ │ │ │ │ - if (errorNode != null && errorNode.length > 0) { │ │ │ │ │ - response.error = this.getChildValue(errorNode, "Unknown error.") │ │ │ │ │ - } else { │ │ │ │ │ - var responseNode = data.getElementsByTagName("RESPONSE"); │ │ │ │ │ - if (responseNode == null || responseNode.length == 0) { │ │ │ │ │ - response.error = "No RESPONSE tag found in ArcXML response."; │ │ │ │ │ - return response │ │ │ │ │ - } │ │ │ │ │ - var rtype = responseNode[0].firstChild.nodeName; │ │ │ │ │ - if (rtype == "#text") { │ │ │ │ │ - rtype = responseNode[0].firstChild.nextSibling.nodeName │ │ │ │ │ + if (this.lastUp && this.lastUp.equals(evt.xy)) { │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + if (this.lastDown && this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance)) { │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy) │ │ │ │ │ } │ │ │ │ │ - if (rtype == "IMAGE") { │ │ │ │ │ - var envelopeNode = data.getElementsByTagName("ENVELOPE"); │ │ │ │ │ - var outputNode = data.getElementsByTagName("OUTPUT"); │ │ │ │ │ - if (envelopeNode == null || envelopeNode.length == 0) { │ │ │ │ │ - response.error = "No ENVELOPE tag found in ArcXML response." │ │ │ │ │ - } else if (outputNode == null || outputNode.length == 0) { │ │ │ │ │ - response.error = "No OUTPUT tag found in ArcXML response." │ │ │ │ │ - } else { │ │ │ │ │ - var envAttr = this.parseAttributes(envelopeNode[0]); │ │ │ │ │ - var outputAttr = this.parseAttributes(outputNode[0]); │ │ │ │ │ - if (typeof outputAttr.type == "string") { │ │ │ │ │ - response.image = { │ │ │ │ │ - envelope: envAttr, │ │ │ │ │ - output: { │ │ │ │ │ - type: outputAttr.type, │ │ │ │ │ - data: this.getChildValue(outputNode[0]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - response.image = { │ │ │ │ │ - envelope: envAttr, │ │ │ │ │ - output: outputAttr │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else if (rtype == "FEATURES") { │ │ │ │ │ - var features = responseNode[0].getElementsByTagName("FEATURES"); │ │ │ │ │ - var featureCount = features[0].getElementsByTagName("FEATURECOUNT"); │ │ │ │ │ - response.features.featurecount = featureCount[0].getAttribute("count"); │ │ │ │ │ - if (response.features.featurecount > 0) { │ │ │ │ │ - var envelope = features[0].getElementsByTagName("ENVELOPE"); │ │ │ │ │ - response.features.envelope = this.parseAttributes(envelope[0], typeof 0); │ │ │ │ │ - var featureList = features[0].getElementsByTagName("FEATURE"); │ │ │ │ │ - for (var fn = 0; fn < featureList.length; fn++) { │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector; │ │ │ │ │ - var fields = featureList[fn].getElementsByTagName("FIELD"); │ │ │ │ │ - for (var fdn = 0; fdn < fields.length; fdn++) { │ │ │ │ │ - var fieldName = fields[fdn].getAttribute("name"); │ │ │ │ │ - var fieldValue = fields[fdn].getAttribute("value"); │ │ │ │ │ - feature.attributes[fieldName] = fieldValue │ │ │ │ │ - } │ │ │ │ │ - var geom = featureList[fn].getElementsByTagName("POLYGON"); │ │ │ │ │ - if (geom.length > 0) { │ │ │ │ │ - var ring = geom[0].getElementsByTagName("RING"); │ │ │ │ │ - var polys = []; │ │ │ │ │ - for (var rn = 0; rn < ring.length; rn++) { │ │ │ │ │ - var linearRings = []; │ │ │ │ │ - linearRings.push(this.parsePointGeometry(ring[rn])); │ │ │ │ │ - var holes = ring[rn].getElementsByTagName("HOLE"); │ │ │ │ │ - for (var hn = 0; hn < holes.length; hn++) { │ │ │ │ │ - linearRings.push(this.parsePointGeometry(holes[hn])) │ │ │ │ │ - } │ │ │ │ │ - holes = null; │ │ │ │ │ - polys.push(new OpenLayers.Geometry.Polygon(linearRings)); │ │ │ │ │ - linearRings = null │ │ │ │ │ - } │ │ │ │ │ - ring = null; │ │ │ │ │ - if (polys.length == 1) { │ │ │ │ │ - feature.geometry = polys[0] │ │ │ │ │ - } else { │ │ │ │ │ - feature.geometry = new OpenLayers.Geometry.MultiPolygon(polys) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - response.features.feature.push(feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - response.error = "Unidentified response type." │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature() │ │ │ │ │ } │ │ │ │ │ + this.lastUp = evt.xy; │ │ │ │ │ + this.finalize(); │ │ │ │ │ + return !this.stopUp │ │ │ │ │ + } else { │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ - return response │ │ │ │ │ }, │ │ │ │ │ - parseAttributes: function(node, type) { │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - for (var attr = 0; attr < node.attributes.length; attr++) { │ │ │ │ │ - if (type == "number") { │ │ │ │ │ - attributes[node.attributes[attr].nodeName] = parseFloat(node.attributes[attr].nodeValue) │ │ │ │ │ - } else { │ │ │ │ │ - attributes[node.attributes[attr].nodeName] = node.attributes[attr].nodeValue │ │ │ │ │ - } │ │ │ │ │ + mouseout: function(evt) { │ │ │ │ │ + if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + this.mouseDown = false │ │ │ │ │ } │ │ │ │ │ - return attributes │ │ │ │ │ }, │ │ │ │ │ - parsePointGeometry: function(node) { │ │ │ │ │ - var ringPoints = []; │ │ │ │ │ - var coords = node.getElementsByTagName("COORDS"); │ │ │ │ │ - if (coords.length > 0) { │ │ │ │ │ - var coordArr = this.getChildValue(coords[0]); │ │ │ │ │ - coordArr = coordArr.split(/;/); │ │ │ │ │ - for (var cn = 0; cn < coordArr.length; cn++) { │ │ │ │ │ - var coordItems = coordArr[cn].split(/ /); │ │ │ │ │ - ringPoints.push(new OpenLayers.Geometry.Point(coordItems[0], coordItems[1])) │ │ │ │ │ - } │ │ │ │ │ - coords = null │ │ │ │ │ - } else { │ │ │ │ │ - var point = node.getElementsByTagName("POINT"); │ │ │ │ │ - if (point.length > 0) { │ │ │ │ │ - for (var pn = 0; pn < point.length; pn++) { │ │ │ │ │ - ringPoints.push(new OpenLayers.Geometry.Point(parseFloat(point[pn].getAttribute("x")), parseFloat(point[pn].getAttribute("y")))) │ │ │ │ │ - } │ │ │ │ │ + passesTolerance: function(pixel1, pixel2, tolerance) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + if (tolerance != null && pixel1 && pixel2) { │ │ │ │ │ + var dist = pixel1.distanceTo(pixel2); │ │ │ │ │ + if (dist > tolerance) { │ │ │ │ │ + passes = false │ │ │ │ │ } │ │ │ │ │ - point = null │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Geometry.LinearRing(ringPoints) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.ArcXML" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.ArcXML.Request = OpenLayers.Class({ │ │ │ │ │ - initialize: function(params) { │ │ │ │ │ - var defaults = { │ │ │ │ │ - get_image: { │ │ │ │ │ - properties: { │ │ │ │ │ - background: null, │ │ │ │ │ - draw: true, │ │ │ │ │ - envelope: { │ │ │ │ │ - minx: 0, │ │ │ │ │ - miny: 0, │ │ │ │ │ - maxx: 0, │ │ │ │ │ - maxy: 0 │ │ │ │ │ - }, │ │ │ │ │ - featurecoordsys: { │ │ │ │ │ - id: 0, │ │ │ │ │ - string: "", │ │ │ │ │ - datumtransformid: 0, │ │ │ │ │ - datumtransformstring: "" │ │ │ │ │ - }, │ │ │ │ │ - filtercoordsys: { │ │ │ │ │ - id: 0, │ │ │ │ │ - string: "", │ │ │ │ │ - datumtransformid: 0, │ │ │ │ │ - datumtransformstring: "" │ │ │ │ │ - }, │ │ │ │ │ - imagesize: { │ │ │ │ │ - height: 0, │ │ │ │ │ - width: 0, │ │ │ │ │ - dpi: 96, │ │ │ │ │ - printheight: 0, │ │ │ │ │ - printwidth: 0, │ │ │ │ │ - scalesymbols: false │ │ │ │ │ - }, │ │ │ │ │ - layerlist: [], │ │ │ │ │ - output: { │ │ │ │ │ - baseurl: "", │ │ │ │ │ - legendbaseurl: "", │ │ │ │ │ - legendname: "", │ │ │ │ │ - legendpath: "", │ │ │ │ │ - legendurl: "", │ │ │ │ │ - name: "", │ │ │ │ │ - path: "", │ │ │ │ │ - type: "jpg", │ │ │ │ │ - url: "" │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - get_feature: { │ │ │ │ │ - layer: "", │ │ │ │ │ - query: { │ │ │ │ │ - isspatial: false, │ │ │ │ │ - featurecoordsys: { │ │ │ │ │ - id: 0, │ │ │ │ │ - string: "", │ │ │ │ │ - datumtransformid: 0, │ │ │ │ │ - datumtransformstring: "" │ │ │ │ │ - }, │ │ │ │ │ - filtercoordsys: { │ │ │ │ │ - id: 0, │ │ │ │ │ - string: "", │ │ │ │ │ - datumtransformid: 0, │ │ │ │ │ - datumtransformstring: "" │ │ │ │ │ - }, │ │ │ │ │ - buffer: 0, │ │ │ │ │ - where: "", │ │ │ │ │ - spatialfilter: { │ │ │ │ │ - relation: "envelope_intersection", │ │ │ │ │ - envelope: null │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - environment: { │ │ │ │ │ - separators: { │ │ │ │ │ - cs: " ", │ │ │ │ │ - ts: ";" │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - layer: [], │ │ │ │ │ - workspaces: [] │ │ │ │ │ - }; │ │ │ │ │ - return OpenLayers.Util.extend(this, defaults) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.ArcXML.Request" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.ArcXML.Response = OpenLayers.Class({ │ │ │ │ │ - initialize: function(params) { │ │ │ │ │ - var defaults = { │ │ │ │ │ - image: { │ │ │ │ │ - envelope: null, │ │ │ │ │ - output: "" │ │ │ │ │ - }, │ │ │ │ │ - features: { │ │ │ │ │ - featurecount: 0, │ │ │ │ │ - envelope: null, │ │ │ │ │ - feature: [] │ │ │ │ │ - }, │ │ │ │ │ - error: "" │ │ │ │ │ - }; │ │ │ │ │ - return OpenLayers.Util.extend(this, defaults) │ │ │ │ │ + return passes │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.ArcXML.Response" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Point" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.ArcIMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - ClientVersion: "9.2", │ │ │ │ │ - ServiceName: "" │ │ │ │ │ +OpenLayers.Handler.Path = OpenLayers.Class(OpenLayers.Handler.Point, { │ │ │ │ │ + line: null, │ │ │ │ │ + maxVertices: null, │ │ │ │ │ + doubleTouchTolerance: 20, │ │ │ │ │ + freehand: false, │ │ │ │ │ + freehandToggle: "shiftKey", │ │ │ │ │ + timerId: null, │ │ │ │ │ + redoStack: null, │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.line = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.LineString([this.point.geometry])); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.line, this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - featureCoordSys: "4326", │ │ │ │ │ - filterCoordSys: "4326", │ │ │ │ │ - layers: null, │ │ │ │ │ - async: true, │ │ │ │ │ - name: "ArcIMS", │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - DEFAULT_OPTIONS: { │ │ │ │ │ - tileSize: new OpenLayers.Size(512, 512), │ │ │ │ │ - featureCoordSys: "4326", │ │ │ │ │ - filterCoordSys: "4326", │ │ │ │ │ - layers: null, │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - async: true, │ │ │ │ │ - name: "ArcIMS" │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + OpenLayers.Handler.Point.prototype.destroyFeature.call(this, force); │ │ │ │ │ + this.line = null │ │ │ │ │ }, │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - this.tileSize = new OpenLayers.Size(512, 512); │ │ │ │ │ - this.params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - ServiceName: options.serviceName │ │ │ │ │ - }, this.DEFAULT_PARAMS); │ │ │ │ │ - this.options = OpenLayers.Util.applyDefaults(options, this.DEFAULT_OPTIONS); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name, url, this.params, options]); │ │ │ │ │ - if (this.transparent) { │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - this.isBaseLayer = false │ │ │ │ │ - } │ │ │ │ │ - if (this.format == "image/jpeg") { │ │ │ │ │ - this.format = OpenLayers.Util.alphaHack() ? "image/gif" : "image/png" │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.options.layers === null) { │ │ │ │ │ - this.options.layers = [] │ │ │ │ │ + destroyPersistedFeature: function() { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + if (layer && layer.features.length > 2) { │ │ │ │ │ + this.layer.features[0].destroy() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var url = ""; │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var axlReq = new OpenLayers.Format.ArcXML(OpenLayers.Util.extend(this.options, { │ │ │ │ │ - requesttype: "image", │ │ │ │ │ - envelope: bounds.toArray(), │ │ │ │ │ - tileSize: this.tileSize │ │ │ │ │ - })); │ │ │ │ │ - var req = new OpenLayers.Request.POST({ │ │ │ │ │ - url: this.getFullRequestString(), │ │ │ │ │ - data: axlReq.write(), │ │ │ │ │ - async: false │ │ │ │ │ - }); │ │ │ │ │ - if (req != null) { │ │ │ │ │ - var doc = req.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = req.responseText │ │ │ │ │ - } │ │ │ │ │ - var axlResp = new OpenLayers.Format.ArcXML; │ │ │ │ │ - var arcxml = axlResp.read(doc); │ │ │ │ │ - url = this.getUrlOrImage(arcxml.image.output) │ │ │ │ │ + removePoint: function() { │ │ │ │ │ + if (this.point) { │ │ │ │ │ + this.layer.removeFeatures([this.point]) │ │ │ │ │ } │ │ │ │ │ - return url │ │ │ │ │ }, │ │ │ │ │ - getURLasync: function(bounds, callback, scope) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var axlReq = new OpenLayers.Format.ArcXML(OpenLayers.Util.extend(this.options, { │ │ │ │ │ - requesttype: "image", │ │ │ │ │ - envelope: bounds.toArray(), │ │ │ │ │ - tileSize: this.tileSize │ │ │ │ │ - })); │ │ │ │ │ - OpenLayers.Request.POST({ │ │ │ │ │ - url: this.getFullRequestString(), │ │ │ │ │ - async: true, │ │ │ │ │ - data: axlReq.write(), │ │ │ │ │ - callback: function(req) { │ │ │ │ │ - var doc = req.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = req.responseText │ │ │ │ │ - } │ │ │ │ │ - var axlResp = new OpenLayers.Format.ArcXML; │ │ │ │ │ - var arcxml = axlResp.read(doc); │ │ │ │ │ - callback.call(scope, this.getUrlOrImage(arcxml.image.output)) │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ + addPoint: function(pixel) { │ │ │ │ │ + this.layer.removeFeatures([this.point]); │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat)); │ │ │ │ │ + this.line.geometry.addComponent(this.point.geometry, this.line.geometry.components.length); │ │ │ │ │ + this.layer.addFeatures([this.point]); │ │ │ │ │ + this.callback("point", [this.point.geometry, this.getGeometry()]); │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + delete this.redoStack │ │ │ │ │ }, │ │ │ │ │ - getUrlOrImage: function(output) { │ │ │ │ │ - var ret = ""; │ │ │ │ │ - if (output.url) { │ │ │ │ │ - ret = output.url │ │ │ │ │ - } else if (output.data) { │ │ │ │ │ - ret = "data:image/" + output.type + ";base64," + output.data │ │ │ │ │ - } │ │ │ │ │ - return ret │ │ │ │ │ + insertXY: function(x, y) { │ │ │ │ │ + this.line.geometry.addComponent(new OpenLayers.Geometry.Point(x, y), this.getCurrentPointIndex()); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + delete this.redoStack │ │ │ │ │ }, │ │ │ │ │ - setLayerQuery: function(id, querydef) { │ │ │ │ │ - for (var lyr = 0; lyr < this.options.layers.length; lyr++) { │ │ │ │ │ - if (id == this.options.layers[lyr].id) { │ │ │ │ │ - this.options.layers[lyr].query = querydef; │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ + insertDeltaXY: function(dx, dy) { │ │ │ │ │ + var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ + var p0 = this.line.geometry.components[previousIndex]; │ │ │ │ │ + if (p0 && !isNaN(p0.x) && !isNaN(p0.y)) { │ │ │ │ │ + this.insertXY(p0.x + dx, p0.y + dy) │ │ │ │ │ } │ │ │ │ │ - this.options.layers.push({ │ │ │ │ │ - id: id, │ │ │ │ │ - visible: true, │ │ │ │ │ - query: querydef │ │ │ │ │ - }) │ │ │ │ │ }, │ │ │ │ │ - getFeatureInfo: function(geometry, layer, options) { │ │ │ │ │ - var buffer = options.buffer || 1; │ │ │ │ │ - var callback = options.callback || function() {}; │ │ │ │ │ - var scope = options.scope || window; │ │ │ │ │ - var requestOptions = {}; │ │ │ │ │ - OpenLayers.Util.extend(requestOptions, this.options); │ │ │ │ │ - requestOptions.requesttype = "feature"; │ │ │ │ │ - if (geometry instanceof OpenLayers.LonLat) { │ │ │ │ │ - requestOptions.polygon = null; │ │ │ │ │ - requestOptions.envelope = [geometry.lon - buffer, geometry.lat - buffer, geometry.lon + buffer, geometry.lat + buffer] │ │ │ │ │ - } else if (geometry instanceof OpenLayers.Geometry.Polygon) { │ │ │ │ │ - requestOptions.envelope = null; │ │ │ │ │ - requestOptions.polygon = geometry │ │ │ │ │ - } │ │ │ │ │ - var arcxml = new OpenLayers.Format.ArcXML(requestOptions); │ │ │ │ │ - OpenLayers.Util.extend(arcxml.request.get_feature, options); │ │ │ │ │ - arcxml.request.get_feature.layer = layer.id; │ │ │ │ │ - if (typeof layer.query.accuracy == "number") { │ │ │ │ │ - arcxml.request.get_feature.query.accuracy = layer.query.accuracy │ │ │ │ │ - } else { │ │ │ │ │ - var mapCenter = this.map.getCenter(); │ │ │ │ │ - var viewPx = this.map.getViewPortPxFromLonLat(mapCenter); │ │ │ │ │ - viewPx.x++; │ │ │ │ │ - var mapOffCenter = this.map.getLonLatFromPixel(viewPx); │ │ │ │ │ - arcxml.request.get_feature.query.accuracy = mapOffCenter.lon - mapCenter.lon │ │ │ │ │ - } │ │ │ │ │ - arcxml.request.get_feature.query.where = layer.query.where; │ │ │ │ │ - arcxml.request.get_feature.query.spatialfilter.relation = "area_intersection"; │ │ │ │ │ - OpenLayers.Request.POST({ │ │ │ │ │ - url: this.getFullRequestString({ │ │ │ │ │ - CustomService: "Query" │ │ │ │ │ - }), │ │ │ │ │ - data: arcxml.write(), │ │ │ │ │ - callback: function(request) { │ │ │ │ │ - var response = arcxml.parseResponse(request.responseText); │ │ │ │ │ - if (!arcxml.iserror()) { │ │ │ │ │ - callback.call(scope, response.features) │ │ │ │ │ - } else { │ │ │ │ │ - callback.call(scope, null) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }) │ │ │ │ │ + insertDirectionLength: function(direction, length) { │ │ │ │ │ + direction *= Math.PI / 180; │ │ │ │ │ + var dx = length * Math.cos(direction); │ │ │ │ │ + var dy = length * Math.sin(direction); │ │ │ │ │ + this.insertDeltaXY(dx, dy) │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.ArcIMS(this.name, this.url, this.getOptions()) │ │ │ │ │ + insertDeflectionLength: function(deflection, length) { │ │ │ │ │ + var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ + if (previousIndex > 0) { │ │ │ │ │ + var p1 = this.line.geometry.components[previousIndex]; │ │ │ │ │ + var p0 = this.line.geometry.components[previousIndex - 1]; │ │ │ │ │ + var theta = Math.atan2(p1.y - p0.y, p1.x - p0.x); │ │ │ │ │ + this.insertDirectionLength(theta * 180 / Math.PI + deflection, length) │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.ArcIMS" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.OSM = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ - name: "OpenStreetMap", │ │ │ │ │ - url: ["http://a.tile.openstreetmap.org/${z}/${x}/${y}.png", "http://b.tile.openstreetmap.org/${z}/${x}/${y}.png", "http://c.tile.openstreetmap.org/${z}/${x}/${y}.png"], │ │ │ │ │ - attribution: "© <a href='http://www.openstreetmap.org/copyright'>OpenStreetMap</a> contributors", │ │ │ │ │ - sphericalMercator: true, │ │ │ │ │ - wrapDateLine: true, │ │ │ │ │ - tileOptions: null, │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - crossOriginKeyword: "anonymous" │ │ │ │ │ - }, this.options && this.options.tileOptions) │ │ │ │ │ + getCurrentPointIndex: function() { │ │ │ │ │ + return this.line.geometry.components.length - 1 │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.OSM(this.name, this.url, this.getOptions()) │ │ │ │ │ + undo: function() { │ │ │ │ │ + var geometry = this.line.geometry; │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var index = this.getCurrentPointIndex() - 1; │ │ │ │ │ + var target = components[index]; │ │ │ │ │ + var undone = geometry.removeComponent(target); │ │ │ │ │ + if (undone) { │ │ │ │ │ + if (this.touch && index > 0) { │ │ │ │ │ + components = geometry.components; │ │ │ │ │ + var lastpt = components[index - 1]; │ │ │ │ │ + var curptidx = this.getCurrentPointIndex(); │ │ │ │ │ + var curpt = components[curptidx]; │ │ │ │ │ + curpt.x = lastpt.x; │ │ │ │ │ + curpt.y = lastpt.y │ │ │ │ │ + } │ │ │ │ │ + if (!this.redoStack) { │ │ │ │ │ + this.redoStack = [] │ │ │ │ │ + } │ │ │ │ │ + this.redoStack.push(target); │ │ │ │ │ + this.drawFeature() │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ + return undone │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.OSM" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.MapGuide = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - useHttpTile: false, │ │ │ │ │ - singleTile: false, │ │ │ │ │ - useOverlay: false, │ │ │ │ │ - useAsyncOverlay: true, │ │ │ │ │ - TILE_PARAMS: { │ │ │ │ │ - operation: "GETTILEIMAGE", │ │ │ │ │ - version: "1.2.0" │ │ │ │ │ + redo: function() { │ │ │ │ │ + var target = this.redoStack && this.redoStack.pop(); │ │ │ │ │ + if (target) { │ │ │ │ │ + this.line.geometry.addComponent(target, this.getCurrentPointIndex()); │ │ │ │ │ + this.drawFeature() │ │ │ │ │ + } │ │ │ │ │ + return !!target │ │ │ │ │ }, │ │ │ │ │ - SINGLE_TILE_PARAMS: { │ │ │ │ │ - operation: "GETMAPIMAGE", │ │ │ │ │ - format: "PNG", │ │ │ │ │ - locale: "en", │ │ │ │ │ - clip: "1", │ │ │ │ │ - version: "1.0.0" │ │ │ │ │ + freehandMode: function(evt) { │ │ │ │ │ + return this.freehandToggle && evt[this.freehandToggle] ? !this.freehand : this.freehand │ │ │ │ │ }, │ │ │ │ │ - OVERLAY_PARAMS: { │ │ │ │ │ - operation: "GETDYNAMICMAPOVERLAYIMAGE", │ │ │ │ │ - format: "PNG", │ │ │ │ │ - locale: "en", │ │ │ │ │ - clip: "1", │ │ │ │ │ - version: "2.0.0" │ │ │ │ │ + modifyFeature: function(pixel, drawing) { │ │ │ │ │ + if (!this.line) { │ │ │ │ │ + this.createFeature(pixel) │ │ │ │ │ + } │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point.geometry.x = lonlat.lon; │ │ │ │ │ + this.point.geometry.y = lonlat.lat; │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.getSketch(), drawing]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.drawFeature() │ │ │ │ │ }, │ │ │ │ │ - FOLDER_PARAMS: { │ │ │ │ │ - tileColumnsPerFolder: 30, │ │ │ │ │ - tileRowsPerFolder: 30, │ │ │ │ │ - format: "png", │ │ │ │ │ - querystring: null │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.line, this.style); │ │ │ │ │ + this.layer.drawFeature(this.point, this.style) │ │ │ │ │ }, │ │ │ │ │ - defaultSize: new OpenLayers.Size(300, 300), │ │ │ │ │ - tileOriginCorner: "tl", │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (options == null || options.isBaseLayer == null) { │ │ │ │ │ - this.isBaseLayer = this.transparent != "true" && this.transparent != true │ │ │ │ │ - } │ │ │ │ │ - if (options && options.useOverlay != null) { │ │ │ │ │ - this.useOverlay = options.useOverlay │ │ │ │ │ + getSketch: function() { │ │ │ │ │ + return this.line │ │ │ │ │ + }, │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.line && this.line.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiLineString([geometry]) │ │ │ │ │ } │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - if (this.useOverlay) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(this.params, this.OVERLAY_PARAMS); │ │ │ │ │ - if (!this.useAsyncOverlay) { │ │ │ │ │ - this.params.version = "1.0.0" │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Util.applyDefaults(this.params, this.SINGLE_TILE_PARAMS) │ │ │ │ │ - } │ │ │ │ │ + return geometry │ │ │ │ │ + }, │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + if (this.timerId && this.passesTolerance(this.lastTouchPx, evt.xy, this.doubleTouchTolerance)) { │ │ │ │ │ + this.finishGeometry(); │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + return false │ │ │ │ │ } else { │ │ │ │ │ - if (this.useHttpTile) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(this.params, this.FOLDER_PARAMS) │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Util.applyDefaults(this.params, this.TILE_PARAMS) │ │ │ │ │ + if (this.timerId) { │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null │ │ │ │ │ } │ │ │ │ │ - this.setTileSize(this.defaultSize) │ │ │ │ │ + this.timerId = window.setTimeout(OpenLayers.Function.bind(function() { │ │ │ │ │ + this.timerId = null │ │ │ │ │ + }, this), 300); │ │ │ │ │ + return OpenLayers.Handler.Point.prototype.touchstart.call(this, evt) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.MapGuide(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ + down: function(evt) { │ │ │ │ │ + var stopDown = this.stopDown; │ │ │ │ │ + if (this.freehandMode(evt)) { │ │ │ │ │ + stopDown = true; │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ + if (!this.touch && (!this.lastDown || !this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance))) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp) │ │ │ │ │ + } │ │ │ │ │ + this.mouseDown = true; │ │ │ │ │ + this.lastDown = evt.xy; │ │ │ │ │ + this.stoppedDown = stopDown; │ │ │ │ │ + return !stopDown │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var url; │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var mapSize = this.map.getSize(); │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - var params = { │ │ │ │ │ - setdisplaydpi: OpenLayers.DOTS_PER_INCH, │ │ │ │ │ - setdisplayheight: mapSize.h * this.ratio, │ │ │ │ │ - setdisplaywidth: mapSize.w * this.ratio, │ │ │ │ │ - setviewcenterx: center.lon, │ │ │ │ │ - setviewcentery: center.lat, │ │ │ │ │ - setviewscale: this.map.getScale() │ │ │ │ │ - }; │ │ │ │ │ - if (this.useOverlay && !this.useAsyncOverlay) { │ │ │ │ │ - var getVisParams = {}; │ │ │ │ │ - getVisParams = OpenLayers.Util.extend(getVisParams, params); │ │ │ │ │ - getVisParams.operation = "GETVISIBLEMAPEXTENT"; │ │ │ │ │ - getVisParams.version = "1.0.0"; │ │ │ │ │ - getVisParams.session = this.params.session; │ │ │ │ │ - getVisParams.mapName = this.params.mapName; │ │ │ │ │ - getVisParams.format = "text/xml"; │ │ │ │ │ - url = this.getFullRequestString(getVisParams); │ │ │ │ │ - OpenLayers.Request.GET({ │ │ │ │ │ - url: url, │ │ │ │ │ - async: false │ │ │ │ │ - }) │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature() │ │ │ │ │ } │ │ │ │ │ - url = this.getFullRequestString(params) │ │ │ │ │ - } else { │ │ │ │ │ - var currentRes = this.map.getResolution(); │ │ │ │ │ - var colidx = Math.floor((bounds.left - this.maxExtent.left) / currentRes); │ │ │ │ │ - colidx = Math.round(colidx / this.tileSize.w); │ │ │ │ │ - var rowidx = Math.floor((this.maxExtent.top - bounds.top) / currentRes); │ │ │ │ │ - rowidx = Math.round(rowidx / this.tileSize.h); │ │ │ │ │ - if (this.useHttpTile) { │ │ │ │ │ - url = this.getImageFilePath({ │ │ │ │ │ - tilecol: colidx, │ │ │ │ │ - tilerow: rowidx, │ │ │ │ │ - scaleindex: this.resolutions.length - this.map.zoom - 1 │ │ │ │ │ - }) │ │ │ │ │ + if (this.maxVertices && this.line && this.line.geometry.components.length === this.maxVertices) { │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize() │ │ │ │ │ } else { │ │ │ │ │ - url = this.getFullRequestString({ │ │ │ │ │ - tilecol: colidx, │ │ │ │ │ - tilerow: rowidx, │ │ │ │ │ - scaleindex: this.resolutions.length - this.map.zoom - 1 │ │ │ │ │ - }) │ │ │ │ │ + this.addPoint(evt.xy) │ │ │ │ │ } │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return url │ │ │ │ │ - }, │ │ │ │ │ - getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ - var url = altUrl == null ? this.url : altUrl; │ │ │ │ │ - if (typeof url == "object") { │ │ │ │ │ - url = url[Math.floor(Math.random() * url.length)] │ │ │ │ │ - } │ │ │ │ │ - var requestString = url; │ │ │ │ │ - var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ - allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ - var urlParams = OpenLayers.Util.upperCaseObject(OpenLayers.Util.getParameters(url)); │ │ │ │ │ - for (var key in allParams) { │ │ │ │ │ - if (key.toUpperCase() in urlParams) { │ │ │ │ │ - delete allParams[key] │ │ │ │ │ - } │ │ │ │ │ + if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp) │ │ │ │ │ } │ │ │ │ │ - var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ - paramsString = paramsString.replace(/,/g, "+"); │ │ │ │ │ - if (paramsString != "") { │ │ │ │ │ - var lastServerChar = url.charAt(url.length - 1); │ │ │ │ │ - if (lastServerChar == "&" || lastServerChar == "?") { │ │ │ │ │ - requestString += paramsString │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + up: function(evt) { │ │ │ │ │ + if (this.mouseDown && (!this.lastUp || !this.lastUp.equals(evt.xy))) { │ │ │ │ │ + if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature() │ │ │ │ │ + } │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize() │ │ │ │ │ } else { │ │ │ │ │ - if (url.indexOf("?") == -1) { │ │ │ │ │ - requestString += "?" + paramsString │ │ │ │ │ - } else { │ │ │ │ │ - requestString += "&" + paramsString │ │ │ │ │ + if (this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance)) { │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy) │ │ │ │ │ + } │ │ │ │ │ + if (this.lastUp == null && this.persist) { │ │ │ │ │ + this.destroyPersistedFeature() │ │ │ │ │ + } │ │ │ │ │ + this.addPoint(evt.xy); │ │ │ │ │ + this.lastUp = evt.xy; │ │ │ │ │ + if (this.line.geometry.components.length === this.maxVertices + 1) { │ │ │ │ │ + this.finishGeometry() │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return requestString │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + return !this.stopUp │ │ │ │ │ }, │ │ │ │ │ - getImageFilePath: function(newParams, altUrl) { │ │ │ │ │ - var url = altUrl == null ? this.url : altUrl; │ │ │ │ │ - if (typeof url == "object") { │ │ │ │ │ - url = url[Math.floor(Math.random() * url.length)] │ │ │ │ │ - } │ │ │ │ │ - var requestString = url; │ │ │ │ │ - var tileRowGroup = ""; │ │ │ │ │ - var tileColGroup = ""; │ │ │ │ │ - if (newParams.tilerow < 0) { │ │ │ │ │ - tileRowGroup = "-" │ │ │ │ │ - } │ │ │ │ │ - if (newParams.tilerow == 0) { │ │ │ │ │ - tileRowGroup += "0" │ │ │ │ │ - } else { │ │ │ │ │ - tileRowGroup += Math.floor(Math.abs(newParams.tilerow / this.params.tileRowsPerFolder)) * this.params.tileRowsPerFolder │ │ │ │ │ - } │ │ │ │ │ - if (newParams.tilecol < 0) { │ │ │ │ │ - tileColGroup = "-" │ │ │ │ │ - } │ │ │ │ │ - if (newParams.tilecol == 0) { │ │ │ │ │ - tileColGroup += "0" │ │ │ │ │ - } else { │ │ │ │ │ - tileColGroup += Math.floor(Math.abs(newParams.tilecol / this.params.tileColumnsPerFolder)) * this.params.tileColumnsPerFolder │ │ │ │ │ - } │ │ │ │ │ - var tilePath = "/S" + Math.floor(newParams.scaleindex) + "/" + this.params.basemaplayergroupname + "/R" + tileRowGroup + "/C" + tileColGroup + "/" + newParams.tilerow % this.params.tileRowsPerFolder + "_" + newParams.tilecol % this.params.tileColumnsPerFolder + "." + this.params.format; │ │ │ │ │ - if (this.params.querystring) { │ │ │ │ │ - tilePath += "?" + this.params.querystring │ │ │ │ │ + finishGeometry: function() { │ │ │ │ │ + var index = this.line.geometry.components.length - 1; │ │ │ │ │ + this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize() │ │ │ │ │ + }, │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + if (!this.freehandMode(evt)) { │ │ │ │ │ + this.finishGeometry() │ │ │ │ │ } │ │ │ │ │ - requestString += tilePath; │ │ │ │ │ - return requestString │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.MapGuide" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Path" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.Vector = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ - isBaseLayer: false, │ │ │ │ │ - isFixed: false, │ │ │ │ │ - features: null, │ │ │ │ │ - filter: null, │ │ │ │ │ - selectedFeatures: null, │ │ │ │ │ - unrenderedFeatures: null, │ │ │ │ │ - reportError: true, │ │ │ │ │ - style: null, │ │ │ │ │ - styleMap: null, │ │ │ │ │ - strategies: null, │ │ │ │ │ - protocol: null, │ │ │ │ │ - renderers: ["SVG", "VML", "Canvas"], │ │ │ │ │ - renderer: null, │ │ │ │ │ - rendererOptions: null, │ │ │ │ │ - geometryType: null, │ │ │ │ │ - drawn: false, │ │ │ │ │ - ratio: 1, │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ - this.assignRenderer() │ │ │ │ │ - } │ │ │ │ │ - if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ - this.renderer = null; │ │ │ │ │ - this.displayError() │ │ │ │ │ - } │ │ │ │ │ - if (!this.styleMap) { │ │ │ │ │ - this.styleMap = new OpenLayers.StyleMap │ │ │ │ │ - } │ │ │ │ │ - this.features = []; │ │ │ │ │ - this.selectedFeatures = []; │ │ │ │ │ - this.unrenderedFeatures = {}; │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - for (var i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - this.strategies[i].setLayer(this) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ +OpenLayers.Handler.Polygon = OpenLayers.Class(OpenLayers.Handler.Path, { │ │ │ │ │ + holeModifier: null, │ │ │ │ │ + drawingHole: false, │ │ │ │ │ + polygon: null, │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.line = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.LinearRing([this.point.geometry])); │ │ │ │ │ + this.polygon = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Polygon([this.line.geometry])); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.polygon, this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoDestroy) { │ │ │ │ │ - strategy.destroy() │ │ │ │ │ + addPoint: function(pixel) { │ │ │ │ │ + if (!this.drawingHole && this.holeModifier && this.evt && this.evt[this.holeModifier]) { │ │ │ │ │ + var geometry = this.point.geometry; │ │ │ │ │ + var features = this.control.layer.features; │ │ │ │ │ + var candidate, polygon; │ │ │ │ │ + for (var i = features.length - 1; i >= 0; --i) { │ │ │ │ │ + candidate = features[i].geometry; │ │ │ │ │ + if ((candidate instanceof OpenLayers.Geometry.Polygon || candidate instanceof OpenLayers.Geometry.MultiPolygon) && candidate.intersects(geometry)) { │ │ │ │ │ + polygon = features[i]; │ │ │ │ │ + this.control.layer.removeFeatures([polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.control.layer.events.registerPriority("sketchcomplete", this, this.finalizeInteriorRing); │ │ │ │ │ + this.control.layer.events.registerPriority("sketchmodified", this, this.enforceTopology); │ │ │ │ │ + polygon.geometry.addComponent(this.line.geometry); │ │ │ │ │ + this.polygon = polygon; │ │ │ │ │ + this.drawingHole = true; │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.strategies = null │ │ │ │ │ - } │ │ │ │ │ - if (this.protocol) { │ │ │ │ │ - if (this.protocol.autoDestroy) { │ │ │ │ │ - this.protocol.destroy() │ │ │ │ │ - } │ │ │ │ │ - this.protocol = null │ │ │ │ │ - } │ │ │ │ │ - this.destroyFeatures(); │ │ │ │ │ - this.features = null; │ │ │ │ │ - this.selectedFeatures = null; │ │ │ │ │ - this.unrenderedFeatures = null; │ │ │ │ │ - if (this.renderer) { │ │ │ │ │ - this.renderer.destroy() │ │ │ │ │ - } │ │ │ │ │ - this.renderer = null; │ │ │ │ │ - this.geometryType = null; │ │ │ │ │ - this.drawn = null; │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Vector(this.name, this.getOptions()) │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ - var features = this.features; │ │ │ │ │ - var len = features.length; │ │ │ │ │ - var clonedFeatures = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - clonedFeatures[i] = features[i].clone() │ │ │ │ │ - } │ │ │ │ │ - obj.features = clonedFeatures; │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - refresh: function(obj) { │ │ │ │ │ - if (this.calculateInRange() && this.visibility) { │ │ │ │ │ - this.events.triggerEvent("refresh", obj) │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Handler.Path.prototype.addPoint.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - assignRenderer: function() { │ │ │ │ │ - for (var i = 0, len = this.renderers.length; i < len; i++) { │ │ │ │ │ - var rendererClass = this.renderers[i]; │ │ │ │ │ - var renderer = typeof rendererClass == "function" ? rendererClass : OpenLayers.Renderer[rendererClass]; │ │ │ │ │ - if (renderer && renderer.prototype.supported()) { │ │ │ │ │ - this.renderer = new renderer(this.div, this.rendererOptions); │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + getCurrentPointIndex: function() { │ │ │ │ │ + return this.line.geometry.components.length - 2 │ │ │ │ │ }, │ │ │ │ │ - displayError: function() { │ │ │ │ │ - if (this.reportError) { │ │ │ │ │ - OpenLayers.Console.userError(OpenLayers.i18n("browserNotSupported", { │ │ │ │ │ - renderers: this.renderers.join("\n") │ │ │ │ │ - })) │ │ │ │ │ + enforceTopology: function(event) { │ │ │ │ │ + var point = event.vertex; │ │ │ │ │ + var components = this.line.geometry.components; │ │ │ │ │ + if (!this.polygon.geometry.intersects(point)) { │ │ │ │ │ + var last = components[components.length - 3]; │ │ │ │ │ + point.x = last.x; │ │ │ │ │ + point.y = last.y │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (!this.renderer) { │ │ │ │ │ - this.map.removeLayer(this) │ │ │ │ │ - } else { │ │ │ │ │ - this.renderer.map = this.map; │ │ │ │ │ - var newSize = this.map.getSize(); │ │ │ │ │ - newSize.w = newSize.w * this.ratio; │ │ │ │ │ - newSize.h = newSize.h * this.ratio; │ │ │ │ │ - this.renderer.setSize(newSize) │ │ │ │ │ - } │ │ │ │ │ + finishGeometry: function() { │ │ │ │ │ + var index = this.line.geometry.components.length - 2; │ │ │ │ │ + this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize() │ │ │ │ │ }, │ │ │ │ │ - afterAdd: function() { │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoActivate) { │ │ │ │ │ - strategy.activate() │ │ │ │ │ + finalizeInteriorRing: function() { │ │ │ │ │ + var ring = this.line.geometry; │ │ │ │ │ + var modified = ring.getArea() !== 0; │ │ │ │ │ + if (modified) { │ │ │ │ │ + var rings = this.polygon.geometry.components; │ │ │ │ │ + for (var i = rings.length - 2; i >= 0; --i) { │ │ │ │ │ + if (ring.intersects(rings[i])) { │ │ │ │ │ + modified = false; │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - this.drawn = false; │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoActivate) { │ │ │ │ │ - strategy.deactivate() │ │ │ │ │ + if (modified) { │ │ │ │ │ + var target; │ │ │ │ │ + outer: for (var i = rings.length - 2; i > 0; --i) { │ │ │ │ │ + var points = rings[i].components; │ │ │ │ │ + for (var j = 0, jj = points.length; j < jj; ++j) { │ │ │ │ │ + if (ring.containsPoint(points[j])) { │ │ │ │ │ + modified = false; │ │ │ │ │ + break outer │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - OpenLayers.Layer.prototype.onMapResize.apply(this, arguments); │ │ │ │ │ - var newSize = this.map.getSize(); │ │ │ │ │ - newSize.w = newSize.w * this.ratio; │ │ │ │ │ - newSize.h = newSize.h * this.ratio; │ │ │ │ │ - this.renderer.setSize(newSize) │ │ │ │ │ - }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - var coordSysUnchanged = true; │ │ │ │ │ - if (!dragging) { │ │ │ │ │ - this.renderer.root.style.visibility = "hidden"; │ │ │ │ │ - var viewSize = this.map.getSize(), │ │ │ │ │ - viewWidth = viewSize.w, │ │ │ │ │ - viewHeight = viewSize.h, │ │ │ │ │ - offsetLeft = viewWidth / 2 * this.ratio - viewWidth / 2, │ │ │ │ │ - offsetTop = viewHeight / 2 * this.ratio - viewHeight / 2; │ │ │ │ │ - offsetLeft += this.map.layerContainerOriginPx.x; │ │ │ │ │ - offsetLeft = -Math.round(offsetLeft); │ │ │ │ │ - offsetTop += this.map.layerContainerOriginPx.y; │ │ │ │ │ - offsetTop = -Math.round(offsetTop); │ │ │ │ │ - this.div.style.left = offsetLeft + "px"; │ │ │ │ │ - this.div.style.top = offsetTop + "px"; │ │ │ │ │ - var extent = this.map.getExtent().scale(this.ratio); │ │ │ │ │ - coordSysUnchanged = this.renderer.setExtent(extent, zoomChanged); │ │ │ │ │ - this.renderer.root.style.visibility = "visible"; │ │ │ │ │ - if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ - this.div.scrollLeft = this.div.scrollLeft │ │ │ │ │ - } │ │ │ │ │ - if (!zoomChanged && coordSysUnchanged) { │ │ │ │ │ - for (var i in this.unrenderedFeatures) { │ │ │ │ │ - var feature = this.unrenderedFeatures[i]; │ │ │ │ │ - this.drawFeature(feature) │ │ │ │ │ - } │ │ │ │ │ + if (modified) { │ │ │ │ │ + if (this.polygon.state !== OpenLayers.State.INSERT) { │ │ │ │ │ + this.polygon.state = OpenLayers.State.UPDATE │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + this.polygon.geometry.removeComponent(ring) │ │ │ │ │ } │ │ │ │ │ - if (!this.drawn || zoomChanged || !coordSysUnchanged) { │ │ │ │ │ - this.drawn = true; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0, len = this.features.length; i < len; i++) { │ │ │ │ │ - this.renderer.locked = i !== len - 1; │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - this.drawFeature(feature) │ │ │ │ │ - } │ │ │ │ │ + this.restoreFeature(); │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + cancel: function() { │ │ │ │ │ + if (this.drawingHole) { │ │ │ │ │ + this.polygon.geometry.removeComponent(this.line.geometry); │ │ │ │ │ + this.restoreFeature(true) │ │ │ │ │ } │ │ │ │ │ + return OpenLayers.Handler.Path.prototype.cancel.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - display: function(display) { │ │ │ │ │ - OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ - var currentDisplay = this.div.style.display; │ │ │ │ │ - if (currentDisplay != this.renderer.root.style.display) { │ │ │ │ │ - this.renderer.root.style.display = currentDisplay │ │ │ │ │ + restoreFeature: function(cancel) { │ │ │ │ │ + this.control.layer.events.unregister("sketchcomplete", this, this.finalizeInteriorRing); │ │ │ │ │ + this.control.layer.events.unregister("sketchmodified", this, this.enforceTopology); │ │ │ │ │ + this.layer.removeFeatures([this.polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.control.layer.addFeatures([this.polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.drawingHole = false; │ │ │ │ │ + if (!cancel) { │ │ │ │ │ + this.control.layer.events.triggerEvent("sketchcomplete", { │ │ │ │ │ + feature: this.polygon │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - addFeatures: function(features, options) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ - features = [features] │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + OpenLayers.Handler.Path.prototype.destroyFeature.call(this, force); │ │ │ │ │ + this.polygon = null │ │ │ │ │ + }, │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.polygon, this.style); │ │ │ │ │ + this.layer.drawFeature(this.point, this.style) │ │ │ │ │ + }, │ │ │ │ │ + getSketch: function() { │ │ │ │ │ + return this.polygon │ │ │ │ │ + }, │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.polygon && this.polygon.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiPolygon([geometry]) │ │ │ │ │ } │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ - if (notify) { │ │ │ │ │ - var event = { │ │ │ │ │ - features: features │ │ │ │ │ - }; │ │ │ │ │ - var ret = this.events.triggerEvent("beforefeaturesadded", event); │ │ │ │ │ - if (ret === false) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - features = event.features │ │ │ │ │ + return geometry │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Polygon" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Handler.MouseWheel = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + wheelListener: null, │ │ │ │ │ + interval: 0, │ │ │ │ │ + maxDelta: Number.POSITIVE_INFINITY, │ │ │ │ │ + delta: 0, │ │ │ │ │ + cumulative: true, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.wheelListener = OpenLayers.Function.bindAsEventListener(this.onWheelEvent, this) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ + this.wheelListener = null │ │ │ │ │ + }, │ │ │ │ │ + onWheelEvent: function(e) { │ │ │ │ │ + if (!this.map || !this.checkModifiers(e)) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - var featuresAdded = []; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - if (i != features.length - 1) { │ │ │ │ │ - this.renderer.locked = true │ │ │ │ │ - } else { │ │ │ │ │ - this.renderer.locked = false │ │ │ │ │ - } │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - if (this.geometryType && !(feature.geometry instanceof this.geometryType)) { │ │ │ │ │ - throw new TypeError("addFeatures: component should be an " + this.geometryType.prototype.CLASS_NAME) │ │ │ │ │ - } │ │ │ │ │ - feature.layer = this; │ │ │ │ │ - if (!feature.style && this.style) { │ │ │ │ │ - feature.style = OpenLayers.Util.extend({}, this.style) │ │ │ │ │ + var overScrollableDiv = false; │ │ │ │ │ + var allowScroll = false; │ │ │ │ │ + var overMapDiv = false; │ │ │ │ │ + var elem = OpenLayers.Event.element(e); │ │ │ │ │ + while (elem != null && !overMapDiv && !overScrollableDiv) { │ │ │ │ │ + if (!overScrollableDiv) { │ │ │ │ │ + try { │ │ │ │ │ + var overflow; │ │ │ │ │ + if (elem.currentStyle) { │ │ │ │ │ + overflow = elem.currentStyle["overflow"] │ │ │ │ │ + } else { │ │ │ │ │ + var style = document.defaultView.getComputedStyle(elem, null); │ │ │ │ │ + overflow = style.getPropertyValue("overflow") │ │ │ │ │ + } │ │ │ │ │ + overScrollableDiv = overflow && overflow == "auto" || overflow == "scroll" │ │ │ │ │ + } catch (err) {} │ │ │ │ │ } │ │ │ │ │ - if (notify) { │ │ │ │ │ - if (this.events.triggerEvent("beforefeatureadded", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) === false) { │ │ │ │ │ - continue │ │ │ │ │ + if (!allowScroll) { │ │ │ │ │ + allowScroll = OpenLayers.Element.hasClass(elem, "olScrollable"); │ │ │ │ │ + if (!allowScroll) { │ │ │ │ │ + for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if (elem == layer.div || elem == layer.pane) { │ │ │ │ │ + allowScroll = true; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.preFeatureInsert(feature) │ │ │ │ │ - } │ │ │ │ │ - featuresAdded.push(feature); │ │ │ │ │ - this.features.push(feature); │ │ │ │ │ - this.drawFeature(feature); │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureadded", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - this.onFeatureInsert(feature) │ │ │ │ │ } │ │ │ │ │ + overMapDiv = elem == this.map.div; │ │ │ │ │ + elem = elem.parentNode │ │ │ │ │ } │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresadded", { │ │ │ │ │ - features: featuresAdded │ │ │ │ │ - }) │ │ │ │ │ + if (!overScrollableDiv && overMapDiv) { │ │ │ │ │ + if (allowScroll) { │ │ │ │ │ + var delta = 0; │ │ │ │ │ + if (e.wheelDelta) { │ │ │ │ │ + delta = e.wheelDelta; │ │ │ │ │ + if (delta % 160 === 0) { │ │ │ │ │ + delta = delta * .75 │ │ │ │ │ + } │ │ │ │ │ + delta = delta / 120 │ │ │ │ │ + } else if (e.detail) { │ │ │ │ │ + delta = -(e.detail / Math.abs(e.detail)) │ │ │ │ │ + } │ │ │ │ │ + this.delta += delta; │ │ │ │ │ + window.clearTimeout(this._timeoutId); │ │ │ │ │ + if (this.interval && Math.abs(this.delta) < this.maxDelta) { │ │ │ │ │ + var evt = OpenLayers.Util.extend({}, e); │ │ │ │ │ + this._timeoutId = window.setTimeout(OpenLayers.Function.bind(function() { │ │ │ │ │ + this.wheelZoom(evt) │ │ │ │ │ + }, this), this.interval) │ │ │ │ │ + } else { │ │ │ │ │ + this.wheelZoom(e) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stop(e) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - removeFeatures: function(features, options) { │ │ │ │ │ - if (!features || features.length === 0) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (features === this.features) { │ │ │ │ │ - return this.removeAllFeatures(options) │ │ │ │ │ - } │ │ │ │ │ - if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ - features = [features] │ │ │ │ │ - } │ │ │ │ │ - if (features === this.selectedFeatures) { │ │ │ │ │ - features = features.slice() │ │ │ │ │ - } │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeaturesremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - if (i != 0 && features[i - 1].geometry) { │ │ │ │ │ - this.renderer.locked = true │ │ │ │ │ + wheelZoom: function(e) { │ │ │ │ │ + var delta = this.delta; │ │ │ │ │ + this.delta = 0; │ │ │ │ │ + if (delta) { │ │ │ │ │ + e.xy = this.map.events.getMousePosition(e); │ │ │ │ │ + if (delta < 0) { │ │ │ │ │ + this.callback("down", [e, this.cumulative ? Math.max(-this.maxDelta, delta) : -1]) │ │ │ │ │ } else { │ │ │ │ │ - this.renderer.locked = false │ │ │ │ │ - } │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.features = OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ - feature.layer = null; │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - this.renderer.eraseFeatures(feature) │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.selectedFeatures, feature) != -1) { │ │ │ │ │ - OpenLayers.Util.removeItem(this.selectedFeatures, feature) │ │ │ │ │ - } │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ + this.callback("up", [e, this.cumulative ? Math.min(this.maxDelta, delta) : 1]) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ }, │ │ │ │ │ - removeAllFeatures: function(options) { │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ - var features = this.features; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeaturesremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - feature.layer = null; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.renderer.clear(); │ │ │ │ │ - this.features = []; │ │ │ │ │ - this.unrenderedFeatures = {}; │ │ │ │ │ - this.selectedFeatures = []; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }) │ │ │ │ │ + activate: function(evt) { │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + var wheelListener = this.wheelListener; │ │ │ │ │ + OpenLayers.Event.observe(window, "DOMMouseScroll", wheelListener); │ │ │ │ │ + OpenLayers.Event.observe(window, "mousewheel", wheelListener); │ │ │ │ │ + OpenLayers.Event.observe(document, "mousewheel", wheelListener); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - destroyFeatures: function(features, options) { │ │ │ │ │ - var all = features == undefined; │ │ │ │ │ - if (all) { │ │ │ │ │ - features = this.features │ │ │ │ │ - } │ │ │ │ │ - if (features) { │ │ │ │ │ - this.removeFeatures(features, options); │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - features[i].destroy() │ │ │ │ │ - } │ │ │ │ │ + deactivate: function(evt) { │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + var wheelListener = this.wheelListener; │ │ │ │ │ + OpenLayers.Event.stopObserving(window, "DOMMouseScroll", wheelListener); │ │ │ │ │ + OpenLayers.Event.stopObserving(window, "mousewheel", wheelListener); │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mousewheel", wheelListener); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - drawFeature: function(feature, style) { │ │ │ │ │ - if (!this.drawn) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (typeof style != "object") { │ │ │ │ │ - if (!style && feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ - style = "delete" │ │ │ │ │ - } │ │ │ │ │ - var renderIntent = style || feature.renderIntent; │ │ │ │ │ - style = feature.style || this.style; │ │ │ │ │ - if (!style) { │ │ │ │ │ - style = this.styleMap.createSymbolizer(feature, renderIntent) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var drawn = this.renderer.drawFeature(feature, style); │ │ │ │ │ - if (drawn === false || drawn === null) { │ │ │ │ │ - this.unrenderedFeatures[feature.id] = feature │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.MouseWheel" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Handler.Pinch = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + started: false, │ │ │ │ │ + stopDown: false, │ │ │ │ │ + pinching: false, │ │ │ │ │ + last: null, │ │ │ │ │ + start: null, │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + if (OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ + this.started = true; │ │ │ │ │ + this.last = this.start = { │ │ │ │ │ + distance: this.getDistance(evt.touches), │ │ │ │ │ + delta: 0, │ │ │ │ │ + scale: 1 │ │ │ │ │ + }; │ │ │ │ │ + this.callback("start", [evt, this.start]); │ │ │ │ │ + propagate = !this.stopDown │ │ │ │ │ + } else if (this.started) { │ │ │ │ │ + return false │ │ │ │ │ } else { │ │ │ │ │ - delete this.unrenderedFeatures[feature.id] │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + return propagate │ │ │ │ │ }, │ │ │ │ │ - eraseFeatures: function(features) { │ │ │ │ │ - this.renderer.eraseFeatures(features) │ │ │ │ │ - }, │ │ │ │ │ - getFeatureFromEvent: function(evt) { │ │ │ │ │ - if (!this.renderer) { │ │ │ │ │ - throw new Error("getFeatureFromEvent called on layer with no " + "renderer. This usually means you destroyed a " + "layer, but not some handler which is associated " + "with it.") │ │ │ │ │ - } │ │ │ │ │ - var feature = null; │ │ │ │ │ - var featureId = this.renderer.getFeatureIdFromEvent(evt); │ │ │ │ │ - if (featureId) { │ │ │ │ │ - if (typeof featureId === "string") { │ │ │ │ │ - feature = this.getFeatureById(featureId) │ │ │ │ │ - } else { │ │ │ │ │ - feature = featureId │ │ │ │ │ - } │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + if (this.started && OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ + this.pinching = true; │ │ │ │ │ + var current = this.getPinchData(evt); │ │ │ │ │ + this.callback("move", [evt, current]); │ │ │ │ │ + this.last = current; │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ + } else if (this.started) { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return feature │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - getFeatureBy: function(property, value) { │ │ │ │ │ - var feature = null; │ │ │ │ │ - for (var i = 0, len = this.features.length; i < len; ++i) { │ │ │ │ │ - if (this.features[i][property] == value) { │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + if (this.started && !OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + this.callback("done", [evt, this.start, this.last]); │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return feature │ │ │ │ │ - }, │ │ │ │ │ - getFeatureById: function(featureId) { │ │ │ │ │ - return this.getFeatureBy("id", featureId) │ │ │ │ │ - }, │ │ │ │ │ - getFeatureByFid: function(featureFid) { │ │ │ │ │ - return this.getFeatureBy("fid", featureFid) │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - getFeaturesByAttribute: function(attrName, attrValue) { │ │ │ │ │ - var i, feature, len = this.features.length, │ │ │ │ │ - foundFeatures = []; │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - if (feature && feature.attributes) { │ │ │ │ │ - if (feature.attributes[attrName] === attrValue) { │ │ │ │ │ - foundFeatures.push(feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + activated = true │ │ │ │ │ } │ │ │ │ │ - return foundFeatures │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - onFeatureInsert: function(feature) {}, │ │ │ │ │ - preFeatureInsert: function(feature) {}, │ │ │ │ │ - getDataExtent: function() { │ │ │ │ │ - var maxExtent = null; │ │ │ │ │ - var features = this.features; │ │ │ │ │ - if (features && features.length > 0) { │ │ │ │ │ - var geometry = null; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - geometry = features[i].geometry; │ │ │ │ │ - if (geometry) { │ │ │ │ │ - if (maxExtent === null) { │ │ │ │ │ - maxExtent = new OpenLayers.Bounds │ │ │ │ │ - } │ │ │ │ │ - maxExtent.extend(geometry.getBounds()) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + deactivated = true │ │ │ │ │ } │ │ │ │ │ - return maxExtent │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Vector" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.PointTrack = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ - dataFrom: null, │ │ │ │ │ - styleFrom: null, │ │ │ │ │ - addNodes: function(pointFeatures, options) { │ │ │ │ │ - if (pointFeatures.length < 2) { │ │ │ │ │ - throw new Error("At least two point features have to be added to " + "create a line from") │ │ │ │ │ - } │ │ │ │ │ - var lines = new Array(pointFeatures.length - 1); │ │ │ │ │ - var pointFeature, startPoint, endPoint; │ │ │ │ │ - for (var i = 0, len = pointFeatures.length; i < len; i++) { │ │ │ │ │ - pointFeature = pointFeatures[i]; │ │ │ │ │ - endPoint = pointFeature.geometry; │ │ │ │ │ - if (!endPoint) { │ │ │ │ │ - var lonlat = pointFeature.lonlat; │ │ │ │ │ - endPoint = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat) │ │ │ │ │ - } else if (endPoint.CLASS_NAME != "OpenLayers.Geometry.Point") { │ │ │ │ │ - throw new TypeError("Only features with point geometries are supported.") │ │ │ │ │ - } │ │ │ │ │ - if (i > 0) { │ │ │ │ │ - var attributes = this.dataFrom != null ? pointFeatures[i + this.dataFrom].data || pointFeatures[i + this.dataFrom].attributes : null; │ │ │ │ │ - var style = this.styleFrom != null ? pointFeatures[i + this.styleFrom].style : null; │ │ │ │ │ - var line = new OpenLayers.Geometry.LineString([startPoint, endPoint]); │ │ │ │ │ - lines[i - 1] = new OpenLayers.Feature.Vector(line, attributes, style) │ │ │ │ │ - } │ │ │ │ │ - startPoint = endPoint │ │ │ │ │ + getDistance: function(touches) { │ │ │ │ │ + var t0 = touches[0]; │ │ │ │ │ + var t1 = touches[1]; │ │ │ │ │ + return Math.sqrt(Math.pow(t0.olClientX - t1.olClientX, 2) + Math.pow(t0.olClientY - t1.olClientY, 2)) │ │ │ │ │ + }, │ │ │ │ │ + getPinchData: function(evt) { │ │ │ │ │ + var distance = this.getDistance(evt.touches); │ │ │ │ │ + var scale = distance / this.start.distance; │ │ │ │ │ + return { │ │ │ │ │ + distance: distance, │ │ │ │ │ + delta: this.last.distance - distance, │ │ │ │ │ + scale: scale │ │ │ │ │ } │ │ │ │ │ - this.addFeatures(lines, options) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.PointTrack" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Pinch" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.PointTrack.SOURCE_NODE = -1; │ │ │ │ │ -OpenLayers.Layer.PointTrack.TARGET_NODE = 0; │ │ │ │ │ -OpenLayers.Layer.PointTrack.dataFrom = { │ │ │ │ │ - SOURCE_NODE: -1, │ │ │ │ │ - TARGET_NODE: 0 │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Layer.Markers = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ - isBaseLayer: false, │ │ │ │ │ - markers: null, │ │ │ │ │ - drawn: false, │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.markers = [] │ │ │ │ │ +OpenLayers.Handler.Box = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + dragHandler: null, │ │ │ │ │ + boxDivClassName: "olHandlerBoxZoomBox", │ │ │ │ │ + boxOffsets: null, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.dragHandler = new OpenLayers.Handler.Drag(this, { │ │ │ │ │ + down: this.startBox, │ │ │ │ │ + move: this.moveBox, │ │ │ │ │ + out: this.removeBox, │ │ │ │ │ + up: this.endBox │ │ │ │ │ + }, { │ │ │ │ │ + keyMask: this.keyMask │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - this.clearMarkers(); │ │ │ │ │ - this.markers = null; │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - if (opacity != this.opacity) { │ │ │ │ │ - this.opacity = opacity; │ │ │ │ │ - for (var i = 0, len = this.markers.length; i < len; i++) { │ │ │ │ │ - this.markers[i].setOpacity(this.opacity) │ │ │ │ │ - } │ │ │ │ │ + OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ + if (this.dragHandler) { │ │ │ │ │ + this.dragHandler.destroy(); │ │ │ │ │ + this.dragHandler = null │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - if (zoomChanged || !this.drawn) { │ │ │ │ │ - for (var i = 0, len = this.markers.length; i < len; i++) { │ │ │ │ │ - this.drawMarker(this.markers[i]) │ │ │ │ │ - } │ │ │ │ │ - this.drawn = true │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Handler.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (this.dragHandler) { │ │ │ │ │ + this.dragHandler.setMap(map) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - addMarker: function(marker) { │ │ │ │ │ - this.markers.push(marker); │ │ │ │ │ - if (this.opacity < 1) { │ │ │ │ │ - marker.setOpacity(this.opacity) │ │ │ │ │ - } │ │ │ │ │ - if (this.map && this.map.getExtent()) { │ │ │ │ │ - marker.map = this.map; │ │ │ │ │ - this.drawMarker(marker) │ │ │ │ │ - } │ │ │ │ │ + startBox: function(xy) { │ │ │ │ │ + this.callback("start", []); │ │ │ │ │ + this.zoomBox = OpenLayers.Util.createDiv("zoomBox", { │ │ │ │ │ + x: -9999, │ │ │ │ │ + y: -9999 │ │ │ │ │ + }); │ │ │ │ │ + this.zoomBox.className = this.boxDivClassName; │ │ │ │ │ + this.zoomBox.style.zIndex = this.map.Z_INDEX_BASE["Popup"] - 1; │ │ │ │ │ + this.map.viewPortDiv.appendChild(this.zoomBox); │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, "olDrawBox") │ │ │ │ │ }, │ │ │ │ │ - removeMarker: function(marker) { │ │ │ │ │ - if (this.markers && this.markers.length) { │ │ │ │ │ - OpenLayers.Util.removeItem(this.markers, marker); │ │ │ │ │ - marker.erase() │ │ │ │ │ - } │ │ │ │ │ + moveBox: function(xy) { │ │ │ │ │ + var startX = this.dragHandler.start.x; │ │ │ │ │ + var startY = this.dragHandler.start.y; │ │ │ │ │ + var deltaX = Math.abs(startX - xy.x); │ │ │ │ │ + var deltaY = Math.abs(startY - xy.y); │ │ │ │ │ + var offset = this.getBoxOffsets(); │ │ │ │ │ + this.zoomBox.style.width = deltaX + offset.width + 1 + "px"; │ │ │ │ │ + this.zoomBox.style.height = deltaY + offset.height + 1 + "px"; │ │ │ │ │ + this.zoomBox.style.left = (xy.x < startX ? startX - deltaX - offset.left : startX - offset.left) + "px"; │ │ │ │ │ + this.zoomBox.style.top = (xy.y < startY ? startY - deltaY - offset.top : startY - offset.top) + "px" │ │ │ │ │ }, │ │ │ │ │ - clearMarkers: function() { │ │ │ │ │ - if (this.markers != null) { │ │ │ │ │ - while (this.markers.length > 0) { │ │ │ │ │ - this.removeMarker(this.markers[0]) │ │ │ │ │ - } │ │ │ │ │ + endBox: function(end) { │ │ │ │ │ + var result; │ │ │ │ │ + if (Math.abs(this.dragHandler.start.x - end.x) > 5 || Math.abs(this.dragHandler.start.y - end.y) > 5) { │ │ │ │ │ + var start = this.dragHandler.start; │ │ │ │ │ + var top = Math.min(start.y, end.y); │ │ │ │ │ + var bottom = Math.max(start.y, end.y); │ │ │ │ │ + var left = Math.min(start.x, end.x); │ │ │ │ │ + var right = Math.max(start.x, end.x); │ │ │ │ │ + result = new OpenLayers.Bounds(left, bottom, right, top) │ │ │ │ │ + } else { │ │ │ │ │ + result = this.dragHandler.start.clone() │ │ │ │ │ } │ │ │ │ │ + this.removeBox(); │ │ │ │ │ + this.callback("done", [result]) │ │ │ │ │ }, │ │ │ │ │ - drawMarker: function(marker) { │ │ │ │ │ - var px = this.map.getLayerPxFromLonLat(marker.lonlat); │ │ │ │ │ - if (px == null) { │ │ │ │ │ - marker.display(false) │ │ │ │ │ + removeBox: function() { │ │ │ │ │ + this.map.viewPortDiv.removeChild(this.zoomBox); │ │ │ │ │ + this.zoomBox = null; │ │ │ │ │ + this.boxOffsets = null; │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDrawBox") │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.dragHandler.activate(); │ │ │ │ │ + return true │ │ │ │ │ } else { │ │ │ │ │ - if (!marker.isDrawn()) { │ │ │ │ │ - var markerImg = marker.draw(px); │ │ │ │ │ - this.div.appendChild(markerImg) │ │ │ │ │ - } else if (marker.icon) { │ │ │ │ │ - marker.icon.moveTo(px) │ │ │ │ │ - } │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - getDataExtent: function() { │ │ │ │ │ - var maxExtent = null; │ │ │ │ │ - if (this.markers && this.markers.length > 0) { │ │ │ │ │ - var maxExtent = new OpenLayers.Bounds; │ │ │ │ │ - for (var i = 0, len = this.markers.length; i < len; i++) { │ │ │ │ │ - var marker = this.markers[i]; │ │ │ │ │ - maxExtent.extend(marker.lonlat) │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + if (this.dragHandler.deactivate()) { │ │ │ │ │ + if (this.zoomBox) { │ │ │ │ │ + this.removeBox() │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return maxExtent │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Markers" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Boxes = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ - drawMarker: function(marker) { │ │ │ │ │ - var topleft = this.map.getLayerPxFromLonLat({ │ │ │ │ │ - lon: marker.bounds.left, │ │ │ │ │ - lat: marker.bounds.top │ │ │ │ │ - }); │ │ │ │ │ - var botright = this.map.getLayerPxFromLonLat({ │ │ │ │ │ - lon: marker.bounds.right, │ │ │ │ │ - lat: marker.bounds.bottom │ │ │ │ │ - }); │ │ │ │ │ - if (botright == null || topleft == null) { │ │ │ │ │ - marker.display(false) │ │ │ │ │ + return true │ │ │ │ │ } else { │ │ │ │ │ - var markerDiv = marker.draw(topleft, { │ │ │ │ │ - w: Math.max(1, botright.x - topleft.x), │ │ │ │ │ - h: Math.max(1, botright.y - topleft.y) │ │ │ │ │ - }); │ │ │ │ │ - if (!marker.drawn) { │ │ │ │ │ - this.div.appendChild(markerDiv); │ │ │ │ │ - marker.drawn = true │ │ │ │ │ - } │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - removeMarker: function(marker) { │ │ │ │ │ - OpenLayers.Util.removeItem(this.markers, marker); │ │ │ │ │ - if (marker.div != null && marker.div.parentNode == this.div) { │ │ │ │ │ - this.div.removeChild(marker.div) │ │ │ │ │ + getBoxOffsets: function() { │ │ │ │ │ + if (!this.boxOffsets) { │ │ │ │ │ + var testDiv = document.createElement("div"); │ │ │ │ │ + testDiv.style.position = "absolute"; │ │ │ │ │ + testDiv.style.border = "1px solid black"; │ │ │ │ │ + testDiv.style.width = "3px"; │ │ │ │ │ + document.body.appendChild(testDiv); │ │ │ │ │ + var w3cBoxModel = testDiv.clientWidth == 3; │ │ │ │ │ + document.body.removeChild(testDiv); │ │ │ │ │ + var left = parseInt(OpenLayers.Element.getStyle(this.zoomBox, "border-left-width")); │ │ │ │ │ + var right = parseInt(OpenLayers.Element.getStyle(this.zoomBox, "border-right-width")); │ │ │ │ │ + var top = parseInt(OpenLayers.Element.getStyle(this.zoomBox, "border-top-width")); │ │ │ │ │ + var bottom = parseInt(OpenLayers.Element.getStyle(this.zoomBox, "border-bottom-width")); │ │ │ │ │ + this.boxOffsets = { │ │ │ │ │ + left: left, │ │ │ │ │ + right: right, │ │ │ │ │ + top: top, │ │ │ │ │ + bottom: bottom, │ │ │ │ │ + width: w3cBoxModel === false ? left + right : 0, │ │ │ │ │ + height: w3cBoxModel === false ? top + bottom : 0 │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return this.boxOffsets │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Boxes" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Box" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.Image = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - url: null, │ │ │ │ │ - extent: null, │ │ │ │ │ - size: null, │ │ │ │ │ - tile: null, │ │ │ │ │ - aspectRatio: null, │ │ │ │ │ - initialize: function(name, url, extent, size, options) { │ │ │ │ │ - this.url = url; │ │ │ │ │ - this.extent = extent; │ │ │ │ │ - this.maxExtent = extent; │ │ │ │ │ - this.size = size; │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ - this.aspectRatio = this.extent.getHeight() / this.size.h / (this.extent.getWidth() / this.size.w) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.tile) { │ │ │ │ │ - this.removeTileMonitoringHooks(this.tile); │ │ │ │ │ - this.tile.destroy(); │ │ │ │ │ - this.tile = null │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Image(this.name, this.url, this.extent, this.size, this.getOptions()) │ │ │ │ │ +OpenLayers.Handler.Hover = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + delay: 500, │ │ │ │ │ + pixelTolerance: null, │ │ │ │ │ + stopMove: false, │ │ │ │ │ + px: null, │ │ │ │ │ + timerId: null, │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + if (this.passesTolerance(evt.xy)) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.callback("move", [evt]); │ │ │ │ │ + this.px = evt.xy; │ │ │ │ │ + evt = OpenLayers.Util.extend({}, evt); │ │ │ │ │ + this.timerId = window.setTimeout(OpenLayers.Function.bind(this.delayedCall, this, evt), this.delay) │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ + return !this.stopMove │ │ │ │ │ }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - if (this.options.maxResolution == null) { │ │ │ │ │ - this.options.maxResolution = this.aspectRatio * this.extent.getWidth() / this.size.w │ │ │ │ │ + mouseout: function(evt) { │ │ │ │ │ + if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.callback("move", [evt]) │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.prototype.setMap.apply(this, arguments) │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - var firstRendering = this.tile == null; │ │ │ │ │ - if (zoomChanged || firstRendering) { │ │ │ │ │ - this.setTileSize(); │ │ │ │ │ - var ulPx = this.map.getLayerPxFromLonLat({ │ │ │ │ │ - lon: this.extent.left, │ │ │ │ │ - lat: this.extent.top │ │ │ │ │ - }); │ │ │ │ │ - if (firstRendering) { │ │ │ │ │ - this.tile = new OpenLayers.Tile.Image(this, ulPx, this.extent, null, this.tileSize); │ │ │ │ │ - this.addTileMonitoringHooks(this.tile) │ │ │ │ │ - } else { │ │ │ │ │ - this.tile.size = this.tileSize.clone(); │ │ │ │ │ - this.tile.position = ulPx.clone() │ │ │ │ │ + passesTolerance: function(px) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + if (this.pixelTolerance && this.px) { │ │ │ │ │ + var dpx = Math.sqrt(Math.pow(this.px.x - px.x, 2) + Math.pow(this.px.y - px.y, 2)); │ │ │ │ │ + if (dpx < this.pixelTolerance) { │ │ │ │ │ + passes = false │ │ │ │ │ } │ │ │ │ │ - this.tile.draw() │ │ │ │ │ } │ │ │ │ │ + return passes │ │ │ │ │ }, │ │ │ │ │ - setTileSize: function() { │ │ │ │ │ - var tileWidth = this.extent.getWidth() / this.map.getResolution(); │ │ │ │ │ - var tileHeight = this.extent.getHeight() / this.map.getResolution(); │ │ │ │ │ - this.tileSize = new OpenLayers.Size(tileWidth, tileHeight) │ │ │ │ │ - }, │ │ │ │ │ - addTileMonitoringHooks: function(tile) { │ │ │ │ │ - tile.onLoadStart = function() { │ │ │ │ │ - this.events.triggerEvent("loadstart") │ │ │ │ │ - }; │ │ │ │ │ - tile.events.register("loadstart", this, tile.onLoadStart); │ │ │ │ │ - tile.onLoadEnd = function() { │ │ │ │ │ - this.events.triggerEvent("loadend") │ │ │ │ │ - }; │ │ │ │ │ - tile.events.register("loadend", this, tile.onLoadEnd); │ │ │ │ │ - tile.events.register("unload", this, tile.onLoadEnd) │ │ │ │ │ - }, │ │ │ │ │ - removeTileMonitoringHooks: function(tile) { │ │ │ │ │ - tile.unload(); │ │ │ │ │ - tile.events.un({ │ │ │ │ │ - loadstart: tile.onLoadStart, │ │ │ │ │ - loadend: tile.onLoadEnd, │ │ │ │ │ - unload: tile.onLoadEnd, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ + clearTimer: function() { │ │ │ │ │ + if (this.timerId != null) { │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - setUrl: function(newUrl) { │ │ │ │ │ - this.url = newUrl; │ │ │ │ │ - this.tile.draw() │ │ │ │ │ + delayedCall: function(evt) { │ │ │ │ │ + this.callback("pause", [evt]) │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - return this.url │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + deactivated = true │ │ │ │ │ + } │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Image" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Hover" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.PointGrid = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ - dx: null, │ │ │ │ │ - dy: null, │ │ │ │ │ - ratio: 1.5, │ │ │ │ │ - maxFeatures: 250, │ │ │ │ │ - rotation: 0, │ │ │ │ │ - origin: null, │ │ │ │ │ - gridBounds: null, │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ - config = config || {}; │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.initialize.apply(this, [config.name, config]) │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ - map.events.register("moveend", this, this.onMoveEnd) │ │ │ │ │ +OpenLayers.Handler.Click = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + delay: 300, │ │ │ │ │ + single: true, │ │ │ │ │ + double: false, │ │ │ │ │ + pixelTolerance: 0, │ │ │ │ │ + dblclickTolerance: 13, │ │ │ │ │ + stopSingle: false, │ │ │ │ │ + stopDouble: false, │ │ │ │ │ + timerId: null, │ │ │ │ │ + down: null, │ │ │ │ │ + last: null, │ │ │ │ │ + first: null, │ │ │ │ │ + rightclickTimerId: null, │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + this.down = this.getEventInfo(evt); │ │ │ │ │ + this.last = this.getEventInfo(evt); │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - map.events.unregister("moveend", this, this.onMoveEnd); │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments) │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + this.last = this.getEventInfo(evt); │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - setRatio: function(ratio) { │ │ │ │ │ - this.ratio = ratio; │ │ │ │ │ - this.updateGrid(true) │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + if (this.down) { │ │ │ │ │ + evt.xy = this.last.xy; │ │ │ │ │ + evt.lastTouches = this.last.touches; │ │ │ │ │ + this.handleSingle(evt); │ │ │ │ │ + this.down = null │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - setMaxFeatures: function(maxFeatures) { │ │ │ │ │ - this.maxFeatures = maxFeatures; │ │ │ │ │ - this.updateGrid(true) │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + this.down = this.getEventInfo(evt); │ │ │ │ │ + this.last = this.getEventInfo(evt); │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - setSpacing: function(dx, dy) { │ │ │ │ │ - this.dx = dx; │ │ │ │ │ - this.dy = dy || dx; │ │ │ │ │ - this.updateGrid(true) │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + if (this.checkModifiers(evt) && this.control.handleRightClicks && OpenLayers.Event.isRightClick(evt)) { │ │ │ │ │ + propagate = this.rightclick(evt) │ │ │ │ │ + } │ │ │ │ │ + return propagate │ │ │ │ │ }, │ │ │ │ │ - setOrigin: function(origin) { │ │ │ │ │ - this.origin = origin; │ │ │ │ │ - this.updateGrid(true) │ │ │ │ │ + rightclick: function(evt) { │ │ │ │ │ + if (this.passesTolerance(evt)) { │ │ │ │ │ + if (this.rightclickTimerId != null) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.callback("dblrightclick", [evt]); │ │ │ │ │ + return !this.stopDouble │ │ │ │ │ + } else { │ │ │ │ │ + var clickEvent = this["double"] ? OpenLayers.Util.extend({}, evt) : this.callback("rightclick", [evt]); │ │ │ │ │ + var delayedRightCall = OpenLayers.Function.bind(this.delayedRightCall, this, clickEvent); │ │ │ │ │ + this.rightclickTimerId = window.setTimeout(delayedRightCall, this.delay) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return !this.stopSingle │ │ │ │ │ }, │ │ │ │ │ - getOrigin: function() { │ │ │ │ │ - if (!this.origin) { │ │ │ │ │ - this.origin = this.map.getExtent().getCenterLonLat() │ │ │ │ │ + delayedRightCall: function(evt) { │ │ │ │ │ + this.rightclickTimerId = null; │ │ │ │ │ + if (evt) { │ │ │ │ │ + this.callback("rightclick", [evt]) │ │ │ │ │ } │ │ │ │ │ - return this.origin │ │ │ │ │ }, │ │ │ │ │ - setRotation: function(rotation) { │ │ │ │ │ - this.rotation = rotation; │ │ │ │ │ - this.updateGrid(true) │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + if (!this.last) { │ │ │ │ │ + this.last = this.getEventInfo(evt) │ │ │ │ │ + } │ │ │ │ │ + this.handleSingle(evt); │ │ │ │ │ + return !this.stopSingle │ │ │ │ │ }, │ │ │ │ │ - onMoveEnd: function() { │ │ │ │ │ - this.updateGrid() │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + this.handleDouble(evt); │ │ │ │ │ + return !this.stopDouble │ │ │ │ │ }, │ │ │ │ │ - getViewBounds: function() { │ │ │ │ │ - var bounds = this.map.getExtent(); │ │ │ │ │ - if (this.rotation) { │ │ │ │ │ - var origin = this.getOrigin(); │ │ │ │ │ - var rotationOrigin = new OpenLayers.Geometry.Point(origin.lon, origin.lat); │ │ │ │ │ - var rect = bounds.toGeometry(); │ │ │ │ │ - rect.rotate(-this.rotation, rotationOrigin); │ │ │ │ │ - bounds = rect.getBounds() │ │ │ │ │ + handleDouble: function(evt) { │ │ │ │ │ + if (this.passesDblclickTolerance(evt)) { │ │ │ │ │ + if (this["double"]) { │ │ │ │ │ + this.callback("dblclick", [evt]) │ │ │ │ │ + } │ │ │ │ │ + this.clearTimer() │ │ │ │ │ } │ │ │ │ │ - return bounds │ │ │ │ │ }, │ │ │ │ │ - updateGrid: function(force) { │ │ │ │ │ - if (force || this.invalidBounds()) { │ │ │ │ │ - var viewBounds = this.getViewBounds(); │ │ │ │ │ - var origin = this.getOrigin(); │ │ │ │ │ - var rotationOrigin = new OpenLayers.Geometry.Point(origin.lon, origin.lat); │ │ │ │ │ - var viewBoundsWidth = viewBounds.getWidth(); │ │ │ │ │ - var viewBoundsHeight = viewBounds.getHeight(); │ │ │ │ │ - var aspectRatio = viewBoundsWidth / viewBoundsHeight; │ │ │ │ │ - var maxHeight = Math.sqrt(this.dx * this.dy * this.maxFeatures / aspectRatio); │ │ │ │ │ - var maxWidth = maxHeight * aspectRatio; │ │ │ │ │ - var gridWidth = Math.min(viewBoundsWidth * this.ratio, maxWidth); │ │ │ │ │ - var gridHeight = Math.min(viewBoundsHeight * this.ratio, maxHeight); │ │ │ │ │ - var center = viewBounds.getCenterLonLat(); │ │ │ │ │ - this.gridBounds = new OpenLayers.Bounds(center.lon - gridWidth / 2, center.lat - gridHeight / 2, center.lon + gridWidth / 2, center.lat + gridHeight / 2); │ │ │ │ │ - var rows = Math.floor(gridHeight / this.dy); │ │ │ │ │ - var cols = Math.floor(gridWidth / this.dx); │ │ │ │ │ - var gridLeft = origin.lon + this.dx * Math.ceil((this.gridBounds.left - origin.lon) / this.dx); │ │ │ │ │ - var gridBottom = origin.lat + this.dy * Math.ceil((this.gridBounds.bottom - origin.lat) / this.dy); │ │ │ │ │ - var features = new Array(rows * cols); │ │ │ │ │ - var x, y, point; │ │ │ │ │ - for (var i = 0; i < cols; ++i) { │ │ │ │ │ - x = gridLeft + i * this.dx; │ │ │ │ │ - for (var j = 0; j < rows; ++j) { │ │ │ │ │ - y = gridBottom + j * this.dy; │ │ │ │ │ - point = new OpenLayers.Geometry.Point(x, y); │ │ │ │ │ - if (this.rotation) { │ │ │ │ │ - point.rotate(this.rotation, rotationOrigin) │ │ │ │ │ + handleSingle: function(evt) { │ │ │ │ │ + if (this.passesTolerance(evt)) { │ │ │ │ │ + if (this.timerId != null) { │ │ │ │ │ + if (this.last.touches && this.last.touches.length === 1) { │ │ │ │ │ + if (this["double"]) { │ │ │ │ │ + OpenLayers.Event.preventDefault(evt) │ │ │ │ │ } │ │ │ │ │ - features[i * rows + j] = new OpenLayers.Feature.Vector(point) │ │ │ │ │ + this.handleDouble(evt) │ │ │ │ │ } │ │ │ │ │ + if (!this.last.touches || this.last.touches.length !== 2) { │ │ │ │ │ + this.clearTimer() │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this.first = this.getEventInfo(evt); │ │ │ │ │ + var clickEvent = this.single ? OpenLayers.Util.extend({}, evt) : null; │ │ │ │ │ + this.queuePotentialClick(clickEvent) │ │ │ │ │ } │ │ │ │ │ - this.destroyFeatures(this.features, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.addFeatures(features, { │ │ │ │ │ - silent: true │ │ │ │ │ - }) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - invalidBounds: function() { │ │ │ │ │ - return !this.gridBounds || !this.gridBounds.containsBounds(this.getViewBounds()) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.PointGrid" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.GeoRSS = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ - location: null, │ │ │ │ │ - features: null, │ │ │ │ │ - formatOptions: null, │ │ │ │ │ - selectedFeature: null, │ │ │ │ │ - icon: null, │ │ │ │ │ - popupSize: null, │ │ │ │ │ - useFeedTitle: true, │ │ │ │ │ - initialize: function(name, location, options) { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ - this.location = location; │ │ │ │ │ - this.features = [] │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.destroy.apply(this, arguments); │ │ │ │ │ - this.clearFeatures(); │ │ │ │ │ - this.features = null │ │ │ │ │ + queuePotentialClick: function(evt) { │ │ │ │ │ + this.timerId = window.setTimeout(OpenLayers.Function.bind(this.delayedCall, this, evt), this.delay) │ │ │ │ │ }, │ │ │ │ │ - loadRSS: function() { │ │ │ │ │ - if (!this.loaded) { │ │ │ │ │ - this.events.triggerEvent("loadstart"); │ │ │ │ │ - OpenLayers.Request.GET({ │ │ │ │ │ - url: this.location, │ │ │ │ │ - success: this.parseData, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.loaded = true │ │ │ │ │ + passesTolerance: function(evt) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + if (this.pixelTolerance != null && this.down && this.down.xy) { │ │ │ │ │ + passes = this.pixelTolerance >= this.down.xy.distanceTo(evt.xy); │ │ │ │ │ + if (passes && this.touch && this.down.touches.length === this.last.touches.length) { │ │ │ │ │ + for (var i = 0, ii = this.down.touches.length; i < ii; ++i) { │ │ │ │ │ + if (this.getTouchDistance(this.down.touches[i], this.last.touches[i]) > this.pixelTolerance) { │ │ │ │ │ + passes = false; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return passes │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, minor) { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - if (this.visibility && !this.loaded) { │ │ │ │ │ - this.loadRSS() │ │ │ │ │ + getTouchDistance: function(from, to) { │ │ │ │ │ + return Math.sqrt(Math.pow(from.clientX - to.clientX, 2) + Math.pow(from.clientY - to.clientY, 2)) │ │ │ │ │ + }, │ │ │ │ │ + passesDblclickTolerance: function(evt) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + if (this.down && this.first) { │ │ │ │ │ + passes = this.down.xy.distanceTo(this.first.xy) <= this.dblclickTolerance │ │ │ │ │ } │ │ │ │ │ + return passes │ │ │ │ │ }, │ │ │ │ │ - parseData: function(ajaxRequest) { │ │ │ │ │ - var doc = ajaxRequest.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = OpenLayers.Format.XML.prototype.read(ajaxRequest.responseText) │ │ │ │ │ + clearTimer: function() { │ │ │ │ │ + if (this.timerId != null) { │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null │ │ │ │ │ } │ │ │ │ │ - if (this.useFeedTitle) { │ │ │ │ │ - var name = null; │ │ │ │ │ - try { │ │ │ │ │ - name = doc.getElementsByTagNameNS("*", "title")[0].firstChild.nodeValue │ │ │ │ │ - } catch (e) { │ │ │ │ │ - try { │ │ │ │ │ - name = doc.getElementsByTagName("title")[0].firstChild.nodeValue │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - } │ │ │ │ │ - if (name) { │ │ │ │ │ - this.setName(name) │ │ │ │ │ - } │ │ │ │ │ + if (this.rightclickTimerId != null) { │ │ │ │ │ + window.clearTimeout(this.rightclickTimerId); │ │ │ │ │ + this.rightclickTimerId = null │ │ │ │ │ } │ │ │ │ │ - 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() │ │ │ │ │ + }, │ │ │ │ │ + delayedCall: function(evt) { │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + if (evt) { │ │ │ │ │ + this.callback("click", [evt]) │ │ │ │ │ } │ │ │ │ │ - var format = new OpenLayers.Format.GeoRSS(options); │ │ │ │ │ - var features = format.read(doc); │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - var data = {}; │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - if (!feature.geometry) { │ │ │ │ │ - continue │ │ │ │ │ - } │ │ │ │ │ - var title = feature.attributes.title ? feature.attributes.title : "Untitled"; │ │ │ │ │ - var description = feature.attributes.description ? feature.attributes.description : "No description."; │ │ │ │ │ - var link = feature.attributes.link ? feature.attributes.link : ""; │ │ │ │ │ - var location = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - data.icon = this.icon == null ? OpenLayers.Marker.defaultIcon() : this.icon.clone(); │ │ │ │ │ - data.popupSize = this.popupSize ? this.popupSize.clone() : new OpenLayers.Size(250, 120); │ │ │ │ │ - if (title || description) { │ │ │ │ │ - data.title = title; │ │ │ │ │ - data.description = description; │ │ │ │ │ - var contentHTML = '<div class="olLayerGeoRSSClose">[x]</div>'; │ │ │ │ │ - contentHTML += '<div class="olLayerGeoRSSTitle">'; │ │ │ │ │ - if (link) { │ │ │ │ │ - contentHTML += '<a class="link" href="' + link + '" target="_blank">' │ │ │ │ │ - } │ │ │ │ │ - contentHTML += title; │ │ │ │ │ - if (link) { │ │ │ │ │ - contentHTML += "</a>" │ │ │ │ │ + }, │ │ │ │ │ + getEventInfo: function(evt) { │ │ │ │ │ + var touches; │ │ │ │ │ + if (evt.touches) { │ │ │ │ │ + var len = evt.touches.length; │ │ │ │ │ + touches = new Array(len); │ │ │ │ │ + var touch; │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + touch = evt.touches[i]; │ │ │ │ │ + touches[i] = { │ │ │ │ │ + clientX: touch.olClientX, │ │ │ │ │ + clientY: touch.olClientY │ │ │ │ │ } │ │ │ │ │ - contentHTML += "</div>"; │ │ │ │ │ - contentHTML += '<div style="" class="olLayerGeoRSSDescription">'; │ │ │ │ │ - contentHTML += description; │ │ │ │ │ - contentHTML += "</div>"; │ │ │ │ │ - data["popupContentHTML"] = contentHTML │ │ │ │ │ } │ │ │ │ │ - var feature = new OpenLayers.Feature(this, location, data); │ │ │ │ │ - this.features.push(feature); │ │ │ │ │ - var marker = feature.createMarker(); │ │ │ │ │ - marker.events.register("click", feature, this.markerClick); │ │ │ │ │ - this.addMarker(marker) │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("loadend") │ │ │ │ │ - }, │ │ │ │ │ - markerClick: function(evt) { │ │ │ │ │ - var sameMarkerClicked = this == this.layer.selectedFeature; │ │ │ │ │ - this.layer.selectedFeature = !sameMarkerClicked ? this : null; │ │ │ │ │ - for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ - this.layer.map.removePopup(this.layer.map.popups[i]) │ │ │ │ │ } │ │ │ │ │ - if (!sameMarkerClicked) { │ │ │ │ │ - var popup = this.createPopup(); │ │ │ │ │ - OpenLayers.Event.observe(popup.div, "click", OpenLayers.Function.bind(function() { │ │ │ │ │ - for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ - this.layer.map.removePopup(this.layer.map.popups[i]) │ │ │ │ │ - } │ │ │ │ │ - }, this)); │ │ │ │ │ - this.layer.map.addPopup(popup) │ │ │ │ │ + return { │ │ │ │ │ + xy: evt.xy, │ │ │ │ │ + touches: touches │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ }, │ │ │ │ │ - clearFeatures: function() { │ │ │ │ │ - if (this.features != null) { │ │ │ │ │ - while (this.features.length > 0) { │ │ │ │ │ - var feature = this.features[0]; │ │ │ │ │ - OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ - feature.destroy() │ │ │ │ │ - } │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.down = null; │ │ │ │ │ + this.first = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + deactivated = true │ │ │ │ │ } │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.GeoRSS" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Click" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.SphericalMercator = { │ │ │ │ │ - getExtent: function() { │ │ │ │ │ - var extent = null; │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - extent = this.map.calculateBounds() │ │ │ │ │ - } else { │ │ │ │ │ - extent = OpenLayers.Layer.FixedZoomLevels.prototype.getExtent.apply(this) │ │ │ │ │ +OpenLayers.Handler.Feature = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + EVENTMAP: { │ │ │ │ │ + click: { │ │ │ │ │ + in: "click", │ │ │ │ │ + out: "clickout" │ │ │ │ │ + }, │ │ │ │ │ + mousemove: { │ │ │ │ │ + in: "over", │ │ │ │ │ + out: "out" │ │ │ │ │ + }, │ │ │ │ │ + dblclick: { │ │ │ │ │ + in: "dblclick", │ │ │ │ │ + out: null │ │ │ │ │ + }, │ │ │ │ │ + mousedown: { │ │ │ │ │ + in: null, │ │ │ │ │ + out: null │ │ │ │ │ + }, │ │ │ │ │ + mouseup: { │ │ │ │ │ + in: null, │ │ │ │ │ + out: null │ │ │ │ │ + }, │ │ │ │ │ + touchstart: { │ │ │ │ │ + in: "click", │ │ │ │ │ + out: "clickout" │ │ │ │ │ } │ │ │ │ │ - return extent │ │ │ │ │ }, │ │ │ │ │ - getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ - return OpenLayers.Layer.prototype.getLonLatFromViewPortPx.apply(this, arguments) │ │ │ │ │ + feature: null, │ │ │ │ │ + lastFeature: null, │ │ │ │ │ + down: null, │ │ │ │ │ + up: null, │ │ │ │ │ + clickTolerance: 4, │ │ │ │ │ + geometryTypes: null, │ │ │ │ │ + stopClick: true, │ │ │ │ │ + stopDown: true, │ │ │ │ │ + stopUp: false, │ │ │ │ │ + initialize: function(control, layer, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ + this.layer = layer │ │ │ │ │ }, │ │ │ │ │ - getViewPortPxFromLonLat: function(lonlat) { │ │ │ │ │ - return OpenLayers.Layer.prototype.getViewPortPxFromLonLat.apply(this, arguments) │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + return OpenLayers.Event.isMultiTouch(evt) ? true : this.mousedown(evt) │ │ │ │ │ }, │ │ │ │ │ - initMercatorParameters: function() { │ │ │ │ │ - this.RESOLUTIONS = []; │ │ │ │ │ - var maxResolution = 156543.03390625; │ │ │ │ │ - for (var zoom = 0; zoom <= this.MAX_ZOOM_LEVEL; ++zoom) { │ │ │ │ │ - this.RESOLUTIONS[zoom] = maxResolution / Math.pow(2, zoom) │ │ │ │ │ - } │ │ │ │ │ - this.units = "m"; │ │ │ │ │ - this.projection = this.projection || "EPSG:900913" │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + OpenLayers.Event.preventDefault(evt) │ │ │ │ │ }, │ │ │ │ │ - forwardMercator: function() { │ │ │ │ │ - var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - var sm = new OpenLayers.Projection("EPSG:900913"); │ │ │ │ │ - return function(lon, lat) { │ │ │ │ │ - var point = OpenLayers.Projection.transform({ │ │ │ │ │ - x: lon, │ │ │ │ │ - y: lat │ │ │ │ │ - }, gg, sm); │ │ │ │ │ - return new OpenLayers.LonLat(point.x, point.y) │ │ │ │ │ - } │ │ │ │ │ - }(), │ │ │ │ │ - inverseMercator: function() { │ │ │ │ │ - var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - var sm = new OpenLayers.Projection("EPSG:900913"); │ │ │ │ │ - return function(x, y) { │ │ │ │ │ - var point = OpenLayers.Projection.transform({ │ │ │ │ │ - x: x, │ │ │ │ │ - y: y │ │ │ │ │ - }, sm, gg); │ │ │ │ │ - return new OpenLayers.LonLat(point.x, point.y) │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + if (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ + this.down = evt.xy │ │ │ │ │ } │ │ │ │ │ - }() │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Layer.FixedZoomLevels = OpenLayers.Class({ │ │ │ │ │ - initialize: function() {}, │ │ │ │ │ - initResolutions: function() { │ │ │ │ │ - var props = ["minZoomLevel", "maxZoomLevel", "numZoomLevels"]; │ │ │ │ │ - for (var i = 0, len = props.length; i < len; i++) { │ │ │ │ │ - var property = props[i]; │ │ │ │ │ - this[property] = this.options[property] != null ? this.options[property] : this.map[property] │ │ │ │ │ + return this.handle(evt) ? !this.stopDown : true │ │ │ │ │ + }, │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + this.up = evt.xy; │ │ │ │ │ + return this.handle(evt) ? !this.stopUp : true │ │ │ │ │ + }, │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + return this.handle(evt) ? !this.stopClick : true │ │ │ │ │ + }, │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + if (!this.callbacks["over"] && !this.callbacks["out"]) { │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ - if (this.minZoomLevel == null || this.minZoomLevel < this.MIN_ZOOM_LEVEL) { │ │ │ │ │ - this.minZoomLevel = this.MIN_ZOOM_LEVEL │ │ │ │ │ + this.handle(evt); │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + return !this.handle(evt) │ │ │ │ │ + }, │ │ │ │ │ + geometryTypeMatches: function(feature) { │ │ │ │ │ + return this.geometryTypes == null || OpenLayers.Util.indexOf(this.geometryTypes, feature.geometry.CLASS_NAME) > -1 │ │ │ │ │ + }, │ │ │ │ │ + handle: function(evt) { │ │ │ │ │ + if (this.feature && !this.feature.layer) { │ │ │ │ │ + this.feature = null │ │ │ │ │ } │ │ │ │ │ - var desiredZoomLevels; │ │ │ │ │ - var limitZoomLevels = this.MAX_ZOOM_LEVEL - this.minZoomLevel + 1; │ │ │ │ │ - if (this.options.numZoomLevels == null && this.options.maxZoomLevel != null || this.numZoomLevels == null && this.maxZoomLevel != null) { │ │ │ │ │ - desiredZoomLevels = this.maxZoomLevel - this.minZoomLevel + 1 │ │ │ │ │ - } else { │ │ │ │ │ - desiredZoomLevels = this.numZoomLevels │ │ │ │ │ + var type = evt.type; │ │ │ │ │ + var handled = false; │ │ │ │ │ + var previouslyIn = !!this.feature; │ │ │ │ │ + var click = type == "click" || type == "dblclick" || type == "touchstart"; │ │ │ │ │ + this.feature = this.layer.getFeatureFromEvent(evt); │ │ │ │ │ + if (this.feature && !this.feature.layer) { │ │ │ │ │ + this.feature = null │ │ │ │ │ } │ │ │ │ │ - if (desiredZoomLevels != null) { │ │ │ │ │ - this.numZoomLevels = Math.min(desiredZoomLevels, limitZoomLevels) │ │ │ │ │ - } else { │ │ │ │ │ - this.numZoomLevels = limitZoomLevels │ │ │ │ │ + if (this.lastFeature && !this.lastFeature.layer) { │ │ │ │ │ + this.lastFeature = null │ │ │ │ │ } │ │ │ │ │ - this.maxZoomLevel = this.minZoomLevel + this.numZoomLevels - 1; │ │ │ │ │ - if (this.RESOLUTIONS != null) { │ │ │ │ │ - var resolutionsIndex = 0; │ │ │ │ │ - this.resolutions = []; │ │ │ │ │ - for (var i = this.minZoomLevel; i <= this.maxZoomLevel; i++) { │ │ │ │ │ - this.resolutions[resolutionsIndex++] = this.RESOLUTIONS[i] │ │ │ │ │ + if (this.feature) { │ │ │ │ │ + if (type === "touchstart") { │ │ │ │ │ + OpenLayers.Event.preventDefault(evt) │ │ │ │ │ } │ │ │ │ │ - this.maxResolution = this.resolutions[0]; │ │ │ │ │ - this.minResolution = this.resolutions[this.resolutions.length - 1] │ │ │ │ │ + var inNew = this.feature != this.lastFeature; │ │ │ │ │ + if (this.geometryTypeMatches(this.feature)) { │ │ │ │ │ + if (previouslyIn && inNew) { │ │ │ │ │ + if (this.lastFeature) { │ │ │ │ │ + this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ + } │ │ │ │ │ + this.triggerCallback(type, "in", [this.feature]) │ │ │ │ │ + } else if (!previouslyIn || click) { │ │ │ │ │ + this.triggerCallback(type, "in", [this.feature]) │ │ │ │ │ + } │ │ │ │ │ + this.lastFeature = this.feature; │ │ │ │ │ + handled = true │ │ │ │ │ + } else { │ │ │ │ │ + if (this.lastFeature && (previouslyIn && inNew || click)) { │ │ │ │ │ + this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ + } │ │ │ │ │ + this.feature = null │ │ │ │ │ + } │ │ │ │ │ + } else if (this.lastFeature && (previouslyIn || click)) { │ │ │ │ │ + this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ } │ │ │ │ │ + return handled │ │ │ │ │ }, │ │ │ │ │ - getResolution: function() { │ │ │ │ │ - if (this.resolutions != null) { │ │ │ │ │ - return OpenLayers.Layer.prototype.getResolution.apply(this, arguments) │ │ │ │ │ - } else { │ │ │ │ │ - var resolution = null; │ │ │ │ │ - var viewSize = this.map.getSize(); │ │ │ │ │ - var extent = this.getExtent(); │ │ │ │ │ - if (viewSize != null && extent != null) { │ │ │ │ │ - resolution = Math.max(extent.getWidth() / viewSize.w, extent.getHeight() / viewSize.h) │ │ │ │ │ + triggerCallback: function(type, mode, args) { │ │ │ │ │ + var key = this.EVENTMAP[type][mode]; │ │ │ │ │ + if (key) { │ │ │ │ │ + if (type == "click" && this.up && this.down) { │ │ │ │ │ + var dpx = Math.sqrt(Math.pow(this.up.x - this.down.x, 2) + Math.pow(this.up.y - this.down.y, 2)); │ │ │ │ │ + if (dpx <= this.clickTolerance) { │ │ │ │ │ + this.callback(key, args) │ │ │ │ │ + } │ │ │ │ │ + this.up = this.down = null │ │ │ │ │ + } else { │ │ │ │ │ + this.callback(key, args) │ │ │ │ │ } │ │ │ │ │ - return resolution │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - getExtent: function() { │ │ │ │ │ - var size = this.map.getSize(); │ │ │ │ │ - var tl = this.getLonLatFromViewPortPx({ │ │ │ │ │ - x: 0, │ │ │ │ │ - y: 0 │ │ │ │ │ - }); │ │ │ │ │ - var br = this.getLonLatFromViewPortPx({ │ │ │ │ │ - x: size.w, │ │ │ │ │ - y: size.h │ │ │ │ │ - }); │ │ │ │ │ - if (tl != null && br != null) { │ │ │ │ │ - return new OpenLayers.Bounds(tl.lon, br.lat, br.lon, tl.lat) │ │ │ │ │ - } else { │ │ │ │ │ - return null │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerToTop(); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + removelayer: this.handleMapEvents, │ │ │ │ │ + changelayer: this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + activated = true │ │ │ │ │ } │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - getZoomForResolution: function(resolution) { │ │ │ │ │ - if (this.resolutions != null) { │ │ │ │ │ - return OpenLayers.Layer.prototype.getZoomForResolution.apply(this, arguments) │ │ │ │ │ - } else { │ │ │ │ │ - var extent = OpenLayers.Layer.prototype.getExtent.apply(this, []); │ │ │ │ │ - return this.getZoomForExtent(extent) │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerBack(); │ │ │ │ │ + this.feature = null; │ │ │ │ │ + this.lastFeature = null; │ │ │ │ │ + this.down = null; │ │ │ │ │ + this.up = null; │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + removelayer: this.handleMapEvents, │ │ │ │ │ + changelayer: this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + deactivated = true │ │ │ │ │ } │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - getOLZoomFromMapObjectZoom: function(moZoom) { │ │ │ │ │ - var zoom = null; │ │ │ │ │ - if (moZoom != null) { │ │ │ │ │ - zoom = moZoom - this.minZoomLevel; │ │ │ │ │ - if (this.map.baseLayer !== this) { │ │ │ │ │ - zoom = this.map.baseLayer.getZoomForResolution(this.getResolutionForZoom(zoom)) │ │ │ │ │ - } │ │ │ │ │ + handleMapEvents: function(evt) { │ │ │ │ │ + if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ + this.moveLayerToTop() │ │ │ │ │ } │ │ │ │ │ - return zoom │ │ │ │ │ }, │ │ │ │ │ - getMapObjectZoomFromOLZoom: function(olZoom) { │ │ │ │ │ - var zoom = null; │ │ │ │ │ - if (olZoom != null) { │ │ │ │ │ - zoom = olZoom + this.minZoomLevel; │ │ │ │ │ - if (this.map.baseLayer !== this) { │ │ │ │ │ - zoom = this.getZoomForResolution(this.map.baseLayer.getResolutionForZoom(zoom)) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return zoom │ │ │ │ │ + moveLayerToTop: function() { │ │ │ │ │ + var index = Math.max(this.map.Z_INDEX_BASE["Feature"] - 1, this.layer.getZIndex()) + 1; │ │ │ │ │ + this.layer.setZIndex(index) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.FixedZoomLevels" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Google = OpenLayers.Class(OpenLayers.Layer.EventPane, OpenLayers.Layer.FixedZoomLevels, { │ │ │ │ │ - MIN_ZOOM_LEVEL: 0, │ │ │ │ │ - MAX_ZOOM_LEVEL: 21, │ │ │ │ │ - RESOLUTIONS: [1.40625, .703125, .3515625, .17578125, .087890625, .0439453125, .02197265625, .010986328125, .0054931640625, .00274658203125, .001373291015625, .0006866455078125, .00034332275390625, .000171661376953125, 858306884765625e-19, 4291534423828125e-20, 2145767211914062e-20, 1072883605957031e-20, 536441802978515e-20, 268220901489257e-20, 1341104507446289e-21, 6.705522537231445e-7], │ │ │ │ │ - type: null, │ │ │ │ │ - wrapDateLine: true, │ │ │ │ │ - sphericalMercator: false, │ │ │ │ │ - version: null, │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - if (!options.version) { │ │ │ │ │ - options.version = typeof GMap2 === "function" ? "2" : "3" │ │ │ │ │ - } │ │ │ │ │ - var mixin = OpenLayers.Layer.Google["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (mixin) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, mixin) │ │ │ │ │ + moveLayerBack: function() { │ │ │ │ │ + var index = this.layer.getZIndex() - 1; │ │ │ │ │ + if (index >= this.map.Z_INDEX_BASE["Feature"]) { │ │ │ │ │ + this.layer.setZIndex(index) │ │ │ │ │ } else { │ │ │ │ │ - throw "Unsupported Google Maps API version: " + options.version │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, mixin.DEFAULTS); │ │ │ │ │ - if (options.maxExtent) { │ │ │ │ │ - options.maxExtent = options.maxExtent.clone() │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ - OpenLayers.Layer.FixedZoomLevels.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - OpenLayers.Util.extend(this, OpenLayers.Layer.SphericalMercator); │ │ │ │ │ - this.initMercatorParameters() │ │ │ │ │ + this.map.setLayerZIndex(this.layer, this.map.getLayerIndex(this.layer)) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - return new OpenLayers.Layer.Google(this.name, this.getOptions()) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Feature" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Handler.Keyboard = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + KEY_EVENTS: ["keydown", "keyup"], │ │ │ │ │ + eventListener: null, │ │ │ │ │ + observeElement: null, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.eventListener = OpenLayers.Function.bindAsEventListener(this.handleKeyEvent, this) │ │ │ │ │ }, │ │ │ │ │ - setVisibility: function(visible) { │ │ │ │ │ - var opacity = this.opacity == null ? 1 : this.opacity; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.setVisibility.apply(this, arguments); │ │ │ │ │ - this.setOpacity(opacity) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.eventListener = null; │ │ │ │ │ + OpenLayers.Handler.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - display: function(visible) { │ │ │ │ │ - if (!this._dragging) { │ │ │ │ │ - this.setGMapVisibility(visible) │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.observeElement = this.observeElement || document; │ │ │ │ │ + for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ + OpenLayers.Event.observe(this.observeElement, this.KEY_EVENTS[i], this.eventListener) │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.display.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - this._dragging = dragging; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - delete this._dragging │ │ │ │ │ }, │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - if (opacity !== this.opacity) { │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "opacity" │ │ │ │ │ - }) │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ + OpenLayers.Event.stopObserving(this.observeElement, this.KEY_EVENTS[i], this.eventListener) │ │ │ │ │ } │ │ │ │ │ - this.opacity = opacity │ │ │ │ │ - } │ │ │ │ │ - if (this.getVisibility()) { │ │ │ │ │ - var container = this.getMapContainer(); │ │ │ │ │ - OpenLayers.Util.modifyDOMElement(container, null, null, null, null, null, null, opacity) │ │ │ │ │ + deactivated = true │ │ │ │ │ } │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.setGMapVisibility(false); │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache && cache.count <= 1) { │ │ │ │ │ - this.removeGMapElements() │ │ │ │ │ - } │ │ │ │ │ + handleKeyEvent: function(evt) { │ │ │ │ │ + if (this.checkModifiers(evt)) { │ │ │ │ │ + this.callback(evt.type, [evt]) │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - removeGMapElements: function() { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - var container = this.mapObject && this.getMapContainer(); │ │ │ │ │ - if (container && container.parentNode) { │ │ │ │ │ - container.parentNode.removeChild(container) │ │ │ │ │ - } │ │ │ │ │ - var termsOfUse = cache.termsOfUse; │ │ │ │ │ - if (termsOfUse && termsOfUse.parentNode) { │ │ │ │ │ - termsOfUse.parentNode.removeChild(termsOfUse) │ │ │ │ │ - } │ │ │ │ │ - var poweredBy = cache.poweredBy; │ │ │ │ │ - if (poweredBy && poweredBy.parentNode) { │ │ │ │ │ - poweredBy.parentNode.removeChild(poweredBy) │ │ │ │ │ - } │ │ │ │ │ - if (this.mapObject && window.google && google.maps && google.maps.event && google.maps.event.clearListeners) { │ │ │ │ │ - google.maps.event.clearListeners(this.mapObject, "tilesloaded") │ │ │ │ │ - } │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Keyboard" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Renderer.Canvas = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ + hitDetection: true, │ │ │ │ │ + hitOverflow: 0, │ │ │ │ │ + canvas: null, │ │ │ │ │ + features: null, │ │ │ │ │ + pendingRedraw: false, │ │ │ │ │ + cachedSymbolBounds: {}, │ │ │ │ │ + initialize: function(containerID, options) { │ │ │ │ │ + OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.root = document.createElement("canvas"); │ │ │ │ │ + this.container.appendChild(this.root); │ │ │ │ │ + this.canvas = this.root.getContext("2d"); │ │ │ │ │ + this.features = {}; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitCanvas = document.createElement("canvas"); │ │ │ │ │ + this.hitContext = this.hitCanvas.getContext("2d") │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - if (this.visibility && this.mapObject) { │ │ │ │ │ - this.setGMapVisibility(false) │ │ │ │ │ + setExtent: function() { │ │ │ │ │ + OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + eraseGeometry: function(geometry, featureId) { │ │ │ │ │ + this.eraseFeatures(this.features[featureId][0]) │ │ │ │ │ + }, │ │ │ │ │ + supported: function() { │ │ │ │ │ + return OpenLayers.CANVAS_SUPPORTED │ │ │ │ │ + }, │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + this.size = size.clone(); │ │ │ │ │ + var root = this.root; │ │ │ │ │ + root.style.width = size.w + "px"; │ │ │ │ │ + root.style.height = size.h + "px"; │ │ │ │ │ + root.width = size.w; │ │ │ │ │ + root.height = size.h; │ │ │ │ │ + this.resolution = null; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + var hitCanvas = this.hitCanvas; │ │ │ │ │ + hitCanvas.style.width = size.w + "px"; │ │ │ │ │ + hitCanvas.style.height = size.h + "px"; │ │ │ │ │ + hitCanvas.width = size.w; │ │ │ │ │ + hitCanvas.height = size.h │ │ │ │ │ } │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - if (cache.count <= 1) { │ │ │ │ │ - this.removeGMapElements(); │ │ │ │ │ - delete OpenLayers.Layer.Google.cache[map.id] │ │ │ │ │ + }, │ │ │ │ │ + drawFeature: function(feature, style) { │ │ │ │ │ + var rendered; │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + style = this.applyDefaultSymbolizer(style || feature.style); │ │ │ │ │ + var bounds = feature.geometry.getBounds(); │ │ │ │ │ + var worldBounds; │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + worldBounds = this.map.getMaxExtent() │ │ │ │ │ + } │ │ │ │ │ + var intersects = bounds && bounds.intersectsBounds(this.extent, { │ │ │ │ │ + worldBounds: worldBounds │ │ │ │ │ + }); │ │ │ │ │ + rendered = style.display !== "none" && !!bounds && intersects; │ │ │ │ │ + if (rendered) { │ │ │ │ │ + this.features[feature.id] = [feature, style] │ │ │ │ │ } else { │ │ │ │ │ - --cache.count │ │ │ │ │ + delete this.features[feature.id] │ │ │ │ │ } │ │ │ │ │ + this.pendingRedraw = true │ │ │ │ │ } │ │ │ │ │ - delete this.termsOfUse; │ │ │ │ │ - delete this.poweredBy; │ │ │ │ │ - delete this.mapObject; │ │ │ │ │ - delete this.dragObject; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.removeMap.apply(this, arguments) │ │ │ │ │ + if (this.pendingRedraw && !this.locked) { │ │ │ │ │ + this.redraw(); │ │ │ │ │ + this.pendingRedraw = false │ │ │ │ │ + } │ │ │ │ │ + return rendered │ │ │ │ │ }, │ │ │ │ │ - getOLBoundsFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - var olBounds = null; │ │ │ │ │ - if (moBounds != null) { │ │ │ │ │ - var sw = moBounds.getSouthWest(); │ │ │ │ │ - var ne = moBounds.getNorthEast(); │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - sw = this.forwardMercator(sw.lng(), sw.lat()); │ │ │ │ │ - ne = this.forwardMercator(ne.lng(), ne.lat()) │ │ │ │ │ - } else { │ │ │ │ │ - sw = new OpenLayers.LonLat(sw.lng(), sw.lat()); │ │ │ │ │ - ne = new OpenLayers.LonLat(ne.lng(), ne.lat()) │ │ │ │ │ + drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ + var className = geometry.CLASS_NAME; │ │ │ │ │ + if (className == "OpenLayers.Geometry.Collection" || className == "OpenLayers.Geometry.MultiPoint" || className == "OpenLayers.Geometry.MultiLineString" || className == "OpenLayers.Geometry.MultiPolygon") { │ │ │ │ │ + for (var i = 0; i < geometry.components.length; i++) { │ │ │ │ │ + this.drawGeometry(geometry.components[i], style, featureId) │ │ │ │ │ } │ │ │ │ │ - olBounds = new OpenLayers.Bounds(sw.lon, sw.lat, ne.lon, ne.lat) │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + this.drawPoint(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + this.drawLineString(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + this.drawPolygon(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - return olBounds │ │ │ │ │ - }, │ │ │ │ │ - getWarningHTML: function() { │ │ │ │ │ - return OpenLayers.i18n("googleWarning") │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectCenter: function() { │ │ │ │ │ - return this.mapObject.getCenter() │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectZoom: function() { │ │ │ │ │ - return this.mapObject.getZoom() │ │ │ │ │ - }, │ │ │ │ │ - getLongitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - return this.sphericalMercator ? this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lon : moLonLat.lng() │ │ │ │ │ - }, │ │ │ │ │ - getLatitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - var lat = this.sphericalMercator ? this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lat : moLonLat.lat(); │ │ │ │ │ - return lat │ │ │ │ │ - }, │ │ │ │ │ - getXFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return moPixel.x │ │ │ │ │ - }, │ │ │ │ │ - getYFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return moPixel.y │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Google" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Google.cache = {}; │ │ │ │ │ -OpenLayers.Layer.Google.v2 = { │ │ │ │ │ - termsOfUse: null, │ │ │ │ │ - poweredBy: null, │ │ │ │ │ - dragObject: null, │ │ │ │ │ - loadMapObject: function() { │ │ │ │ │ - if (!this.type) { │ │ │ │ │ - this.type = G_NORMAL_MAP │ │ │ │ │ + drawExternalGraphic: function(geometry, style, featureId) { │ │ │ │ │ + var img = new Image; │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + img.title = title │ │ │ │ │ } │ │ │ │ │ - var mapObject, termsOfUse, poweredBy; │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - mapObject = cache.mapObject; │ │ │ │ │ - termsOfUse = cache.termsOfUse; │ │ │ │ │ - poweredBy = cache.poweredBy; │ │ │ │ │ - ++cache.count │ │ │ │ │ - } else { │ │ │ │ │ - var container = this.map.viewPortDiv; │ │ │ │ │ - var div = document.createElement("div"); │ │ │ │ │ - div.id = this.map.id + "_GMap2Container"; │ │ │ │ │ - div.style.position = "absolute"; │ │ │ │ │ - div.style.width = "100%"; │ │ │ │ │ - div.style.height = "100%"; │ │ │ │ │ - container.appendChild(div); │ │ │ │ │ - try { │ │ │ │ │ - mapObject = new GMap2(div); │ │ │ │ │ - termsOfUse = div.lastChild; │ │ │ │ │ - container.appendChild(termsOfUse); │ │ │ │ │ - termsOfUse.style.zIndex = "1100"; │ │ │ │ │ - termsOfUse.style.right = ""; │ │ │ │ │ - termsOfUse.style.bottom = ""; │ │ │ │ │ - termsOfUse.className = "olLayerGoogleCopyright"; │ │ │ │ │ - poweredBy = div.lastChild; │ │ │ │ │ - container.appendChild(poweredBy); │ │ │ │ │ - poweredBy.style.zIndex = "1100"; │ │ │ │ │ - poweredBy.style.right = ""; │ │ │ │ │ - poweredBy.style.bottom = ""; │ │ │ │ │ - poweredBy.className = "olLayerGooglePoweredBy gmnoprint" │ │ │ │ │ - } catch (e) { │ │ │ │ │ - throw e │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ + var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ + var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ + var onLoad = function() { │ │ │ │ │ + if (!this.features[featureId]) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.Google.cache[this.map.id] = { │ │ │ │ │ - mapObject: mapObject, │ │ │ │ │ - termsOfUse: termsOfUse, │ │ │ │ │ - poweredBy: poweredBy, │ │ │ │ │ - count: 1 │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ + if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ + var x = p0 + xOffset | 0; │ │ │ │ │ + var y = p1 + yOffset | 0; │ │ │ │ │ + var canvas = this.canvas; │ │ │ │ │ + canvas.globalAlpha = opacity; │ │ │ │ │ + var factor = OpenLayers.Renderer.Canvas.drawImageScaleFactor || (OpenLayers.Renderer.Canvas.drawImageScaleFactor = /android 2.1/.test(navigator.userAgent.toLowerCase()) ? 320 / window.screen.width : 1); │ │ │ │ │ + canvas.drawImage(img, x * factor, y * factor, width * factor, height * factor); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId); │ │ │ │ │ + this.hitContext.fillRect(x, y, width, height) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + }; │ │ │ │ │ + img.onload = OpenLayers.Function.bind(onLoad, this); │ │ │ │ │ + img.src = style.externalGraphic │ │ │ │ │ + }, │ │ │ │ │ + drawNamedSymbol: function(geometry, style, featureId) { │ │ │ │ │ + var x, y, cx, cy, i, symbolBounds, scaling, angle; │ │ │ │ │ + var unscaledStrokeWidth; │ │ │ │ │ + var deg2rad = Math.PI / 180; │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[style.graphicName]; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(style.graphicName + " is not a valid symbol name") │ │ │ │ │ } │ │ │ │ │ - this.mapObject = mapObject; │ │ │ │ │ - this.termsOfUse = termsOfUse; │ │ │ │ │ - this.poweredBy = poweredBy; │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.mapObject.getMapTypes(), this.type) === -1) { │ │ │ │ │ - this.mapObject.addMapType(this.type) │ │ │ │ │ + if (!symbol.length || symbol.length < 2) return; │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ + if (isNaN(p0) || isNaN(p1)) return; │ │ │ │ │ + this.canvas.lineCap = "round"; │ │ │ │ │ + this.canvas.lineJoin = "round"; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.lineCap = "round"; │ │ │ │ │ + this.hitContext.lineJoin = "round" │ │ │ │ │ } │ │ │ │ │ - if (typeof mapObject.getDragObject == "function") { │ │ │ │ │ - this.dragObject = mapObject.getDragObject() │ │ │ │ │ + if (style.graphicName in this.cachedSymbolBounds) { │ │ │ │ │ + symbolBounds = this.cachedSymbolBounds[style.graphicName] │ │ │ │ │ } else { │ │ │ │ │ - this.dragPanMapObject = null │ │ │ │ │ + symbolBounds = new OpenLayers.Bounds; │ │ │ │ │ + for (i = 0; i < symbol.length; i += 2) { │ │ │ │ │ + symbolBounds.extend(new OpenLayers.LonLat(symbol[i], symbol[i + 1])) │ │ │ │ │ + } │ │ │ │ │ + this.cachedSymbolBounds[style.graphicName] = symbolBounds │ │ │ │ │ } │ │ │ │ │ - if (this.isBaseLayer === false) { │ │ │ │ │ - this.setGMapVisibility(this.div.style.display !== "none") │ │ │ │ │ + this.canvas.save(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.save() │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - if (this.visibility && this.mapObject.isLoaded()) { │ │ │ │ │ - this.mapObject.checkResize() │ │ │ │ │ - } else { │ │ │ │ │ - if (!this._resized) { │ │ │ │ │ - var layer = this; │ │ │ │ │ - var handle = GEvent.addListener(this.mapObject, "load", function() { │ │ │ │ │ - GEvent.removeListener(handle); │ │ │ │ │ - delete layer._resized; │ │ │ │ │ - layer.mapObject.checkResize(); │ │ │ │ │ - layer.moveTo(layer.map.getCenter(), layer.map.getZoom()) │ │ │ │ │ - }) │ │ │ │ │ + this.canvas.translate(p0, p1); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.translate(p0, p1) │ │ │ │ │ + } │ │ │ │ │ + angle = deg2rad * style.rotation; │ │ │ │ │ + if (!isNaN(angle)) { │ │ │ │ │ + this.canvas.rotate(angle); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.rotate(angle) │ │ │ │ │ } │ │ │ │ │ - this._resized = true │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - setGMapVisibility: function(visible) { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - var container = this.mapObject.getContainer(); │ │ │ │ │ - if (visible === true) { │ │ │ │ │ - this.mapObject.setMapType(this.type); │ │ │ │ │ - container.style.display = ""; │ │ │ │ │ - this.termsOfUse.style.left = ""; │ │ │ │ │ - this.termsOfUse.style.display = ""; │ │ │ │ │ - this.poweredBy.style.display = ""; │ │ │ │ │ - cache.displayed = this.id │ │ │ │ │ - } else { │ │ │ │ │ - if (cache.displayed === this.id) { │ │ │ │ │ - delete cache.displayed │ │ │ │ │ + scaling = 2 * style.pointRadius / Math.max(symbolBounds.getWidth(), symbolBounds.getHeight()); │ │ │ │ │ + this.canvas.scale(scaling, scaling); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.scale(scaling, scaling) │ │ │ │ │ + } │ │ │ │ │ + cx = symbolBounds.getCenterLonLat().lon; │ │ │ │ │ + cy = symbolBounds.getCenterLonLat().lat; │ │ │ │ │ + this.canvas.translate(-cx, -cy); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.translate(-cx, -cy) │ │ │ │ │ + } │ │ │ │ │ + unscaledStrokeWidth = style.strokeWidth; │ │ │ │ │ + style.strokeWidth = unscaledStrokeWidth / scaling; │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.canvas.lineTo(x, y) │ │ │ │ │ + } │ │ │ │ │ + this.canvas.closePath(); │ │ │ │ │ + this.canvas.fill(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.hitContext.lineTo(x, y) │ │ │ │ │ } │ │ │ │ │ - if (!cache.displayed) { │ │ │ │ │ - container.style.display = "none"; │ │ │ │ │ - this.termsOfUse.style.display = "none"; │ │ │ │ │ - this.termsOfUse.style.left = "-9999px"; │ │ │ │ │ - this.poweredBy.style.display = "none" │ │ │ │ │ + this.hitContext.closePath(); │ │ │ │ │ + this.hitContext.fill() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.canvas.lineTo(x, y) │ │ │ │ │ + } │ │ │ │ │ + this.canvas.closePath(); │ │ │ │ │ + this.canvas.stroke(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style, scaling); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.hitContext.moveTo(x, y); │ │ │ │ │ + this.hitContext.lineTo(x, y) │ │ │ │ │ } │ │ │ │ │ + this.hitContext.closePath(); │ │ │ │ │ + this.hitContext.stroke() │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - getMapContainer: function() { │ │ │ │ │ - return this.mapObject.getContainer() │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ - var moBounds = null; │ │ │ │ │ - if (olBounds != null) { │ │ │ │ │ - var sw = this.sphericalMercator ? this.inverseMercator(olBounds.bottom, olBounds.left) : new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ - var ne = this.sphericalMercator ? this.inverseMercator(olBounds.top, olBounds.right) : new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ - moBounds = new GLatLngBounds(new GLatLng(sw.lat, sw.lon), new GLatLng(ne.lat, ne.lon)) │ │ │ │ │ + style.strokeWidth = unscaledStrokeWidth; │ │ │ │ │ + this.canvas.restore(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.restore() │ │ │ │ │ } │ │ │ │ │ - return moBounds │ │ │ │ │ - }, │ │ │ │ │ - setMapObjectCenter: function(center, zoom) { │ │ │ │ │ - this.mapObject.setCenter(center, zoom) │ │ │ │ │ - }, │ │ │ │ │ - dragPanMapObject: function(dX, dY) { │ │ │ │ │ - this.dragObject.moveBy(new GSize(-dX, dY)) │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return this.mapObject.fromContainerPixelToLatLng(moPixel) │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - return this.mapObject.fromLatLngToContainerPixel(moLonLat) │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - return this.mapObject.getBoundsZoomLevel(moBounds) │ │ │ │ │ + this.setCanvasStyle("reset") │ │ │ │ │ }, │ │ │ │ │ - getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ - var gLatLng; │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ - gLatLng = new GLatLng(lonlat.lat, lonlat.lon) │ │ │ │ │ + setCanvasStyle: function(type, style) { │ │ │ │ │ + if (type === "fill") { │ │ │ │ │ + this.canvas.globalAlpha = style["fillOpacity"]; │ │ │ │ │ + this.canvas.fillStyle = style["fillColor"] │ │ │ │ │ + } else if (type === "stroke") { │ │ │ │ │ + this.canvas.globalAlpha = style["strokeOpacity"]; │ │ │ │ │ + this.canvas.strokeStyle = style["strokeColor"]; │ │ │ │ │ + this.canvas.lineWidth = style["strokeWidth"] │ │ │ │ │ } else { │ │ │ │ │ - gLatLng = new GLatLng(lat, lon) │ │ │ │ │ - } │ │ │ │ │ - return gLatLng │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ - return new GPoint(x, y) │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Layer.KaMap = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - i: "jpeg", │ │ │ │ │ - map: "" │ │ │ │ │ - }, │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.params = OpenLayers.Util.applyDefaults(this.params, this.DEFAULT_PARAMS) │ │ │ │ │ - }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var mapRes = this.map.getResolution(); │ │ │ │ │ - var scale = Math.round(this.map.getScale() * 1e4) / 1e4; │ │ │ │ │ - var pX = Math.round(bounds.left / mapRes); │ │ │ │ │ - var pY = -Math.round(bounds.top / mapRes); │ │ │ │ │ - return this.getFullRequestString({ │ │ │ │ │ - t: pY, │ │ │ │ │ - l: pX, │ │ │ │ │ - s: scale │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - calculateGridLayout: function(bounds, origin, resolution) { │ │ │ │ │ - var tilelon = resolution * this.tileSize.w; │ │ │ │ │ - var tilelat = resolution * this.tileSize.h; │ │ │ │ │ - var offsetlon = bounds.left; │ │ │ │ │ - var tilecol = Math.floor(offsetlon / tilelon) - this.buffer; │ │ │ │ │ - var offsetlat = bounds.top; │ │ │ │ │ - var tilerow = Math.floor(offsetlat / tilelat) + this.buffer; │ │ │ │ │ - return { │ │ │ │ │ - tilelon: tilelon, │ │ │ │ │ - tilelat: tilelat, │ │ │ │ │ - startcol: tilecol, │ │ │ │ │ - startrow: tilerow │ │ │ │ │ + this.canvas.globalAlpha = 0; │ │ │ │ │ + this.canvas.lineWidth = 1 │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - getTileBoundsForGridIndex: function(row, col) { │ │ │ │ │ - var origin = this.getTileOrigin(); │ │ │ │ │ - var tileLayout = this.gridLayout; │ │ │ │ │ - var tilelon = tileLayout.tilelon; │ │ │ │ │ - var tilelat = tileLayout.tilelat; │ │ │ │ │ - var minX = (tileLayout.startcol + col) * tilelon; │ │ │ │ │ - var minY = (tileLayout.startrow - row) * tilelat; │ │ │ │ │ - return new OpenLayers.Bounds(minX, minY, minX + tilelon, minY + tilelat) │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.KaMap(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - if (this.tileSize != null) { │ │ │ │ │ - obj.tileSize = this.tileSize.clone() │ │ │ │ │ + featureIdToHex: function(featureId) { │ │ │ │ │ + var id = Number(featureId.split("_").pop()) + 1; │ │ │ │ │ + if (id >= 16777216) { │ │ │ │ │ + this.hitOverflow = id - 16777215; │ │ │ │ │ + id = id % 16777216 + 1 │ │ │ │ │ } │ │ │ │ │ - obj.grid = []; │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - getTileBounds: function(viewPortPx) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var tileMapWidth = resolution * this.tileSize.w; │ │ │ │ │ - var tileMapHeight = resolution * this.tileSize.h; │ │ │ │ │ - var mapPoint = this.getLonLatFromViewPortPx(viewPortPx); │ │ │ │ │ - var tileLeft = tileMapWidth * Math.floor(mapPoint.lon / tileMapWidth); │ │ │ │ │ - var tileBottom = tileMapHeight * Math.floor(mapPoint.lat / tileMapHeight); │ │ │ │ │ - return new OpenLayers.Bounds(tileLeft, tileBottom, tileLeft + tileMapWidth, tileBottom + tileMapHeight) │ │ │ │ │ + var hex = "000000" + id.toString(16); │ │ │ │ │ + var len = hex.length; │ │ │ │ │ + hex = "#" + hex.substring(len - 6, len); │ │ │ │ │ + return hex │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.KaMap" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.Text = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ - defaultStyle: null, │ │ │ │ │ - extractStyles: true, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - if (options.extractStyles !== false) { │ │ │ │ │ - options.defaultStyle = { │ │ │ │ │ - externalGraphic: OpenLayers.Util.getImageLocation("marker.png"), │ │ │ │ │ - graphicWidth: 21, │ │ │ │ │ - graphicHeight: 25, │ │ │ │ │ - graphicXOffset: -10.5, │ │ │ │ │ - graphicYOffset: -12.5 │ │ │ │ │ + setHitContextStyle: function(type, featureId, symbolizer, strokeScaling) { │ │ │ │ │ + var hex = this.featureIdToHex(featureId); │ │ │ │ │ + if (type == "fill") { │ │ │ │ │ + this.hitContext.globalAlpha = 1; │ │ │ │ │ + this.hitContext.fillStyle = hex │ │ │ │ │ + } else if (type == "stroke") { │ │ │ │ │ + this.hitContext.globalAlpha = 1; │ │ │ │ │ + this.hitContext.strokeStyle = hex; │ │ │ │ │ + if (typeof strokeScaling === "undefined") { │ │ │ │ │ + this.hitContext.lineWidth = symbolizer.strokeWidth + 2 │ │ │ │ │ + } else { │ │ │ │ │ + if (!isNaN(strokeScaling)) { │ │ │ │ │ + this.hitContext.lineWidth = symbolizer.strokeWidth + 2 / strokeScaling │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + this.hitContext.globalAlpha = 0; │ │ │ │ │ + this.hitContext.lineWidth = 1 │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Format.prototype.initialize.apply(this, [options]) │ │ │ │ │ }, │ │ │ │ │ - read: function(text) { │ │ │ │ │ - var lines = text.split("\n"); │ │ │ │ │ - var columns; │ │ │ │ │ - var features = []; │ │ │ │ │ - for (var lcv = 0; lcv < lines.length - 1; lcv++) { │ │ │ │ │ - var currLine = lines[lcv].replace(/^\s*/, "").replace(/\s*$/, ""); │ │ │ │ │ - if (currLine.charAt(0) != "#") { │ │ │ │ │ - if (!columns) { │ │ │ │ │ - columns = currLine.split("\t") │ │ │ │ │ - } else { │ │ │ │ │ - var vals = currLine.split("\t"); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point(0, 0); │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - var style = this.defaultStyle ? OpenLayers.Util.applyDefaults({}, this.defaultStyle) : null; │ │ │ │ │ - var icon, iconSize, iconOffset, overflow; │ │ │ │ │ - var set = false; │ │ │ │ │ - for (var valIndex = 0; valIndex < vals.length; valIndex++) { │ │ │ │ │ - if (vals[valIndex]) { │ │ │ │ │ - if (columns[valIndex] == "point") { │ │ │ │ │ - var coords = vals[valIndex].split(","); │ │ │ │ │ - geometry.y = parseFloat(coords[0]); │ │ │ │ │ - geometry.x = parseFloat(coords[1]); │ │ │ │ │ - set = true │ │ │ │ │ - } else if (columns[valIndex] == "lat") { │ │ │ │ │ - geometry.y = parseFloat(vals[valIndex]); │ │ │ │ │ - set = true │ │ │ │ │ - } else if (columns[valIndex] == "lon") { │ │ │ │ │ - geometry.x = parseFloat(vals[valIndex]); │ │ │ │ │ - set = true │ │ │ │ │ - } else if (columns[valIndex] == "title") attributes["title"] = vals[valIndex]; │ │ │ │ │ - else if (columns[valIndex] == "image" || columns[valIndex] == "icon" && style) { │ │ │ │ │ - style["externalGraphic"] = vals[valIndex] │ │ │ │ │ - } else if (columns[valIndex] == "iconSize" && style) { │ │ │ │ │ - var size = vals[valIndex].split(","); │ │ │ │ │ - style["graphicWidth"] = parseFloat(size[0]); │ │ │ │ │ - style["graphicHeight"] = parseFloat(size[1]) │ │ │ │ │ - } else if (columns[valIndex] == "iconOffset" && style) { │ │ │ │ │ - var offset = vals[valIndex].split(","); │ │ │ │ │ - style["graphicXOffset"] = parseFloat(offset[0]); │ │ │ │ │ - style["graphicYOffset"] = parseFloat(offset[1]) │ │ │ │ │ - } else if (columns[valIndex] == "description") { │ │ │ │ │ - attributes["description"] = vals[valIndex] │ │ │ │ │ - } else if (columns[valIndex] == "overflow") { │ │ │ │ │ - attributes["overflow"] = vals[valIndex] │ │ │ │ │ - } else { │ │ │ │ │ - attributes[columns[valIndex]] = vals[valIndex] │ │ │ │ │ - } │ │ │ │ │ + drawPoint: function(geometry, style, featureId) { │ │ │ │ │ + if (style.graphic !== false) { │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + this.drawExternalGraphic(geometry, style, featureId) │ │ │ │ │ + } else if (style.graphicName && style.graphicName != "circle") { │ │ │ │ │ + this.drawNamedSymbol(geometry, style, featureId) │ │ │ │ │ + } else { │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ + if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ + var twoPi = Math.PI * 2; │ │ │ │ │ + var radius = style.pointRadius; │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.canvas.fill(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.hitContext.fill() │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (set) { │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.canvas.stroke(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.hitContext.stroke() │ │ │ │ │ } │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry, attributes, style); │ │ │ │ │ - features.push(feature) │ │ │ │ │ + this.setCanvasStyle("reset") │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return features │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.Text" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Text = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ - location: null, │ │ │ │ │ - features: null, │ │ │ │ │ - formatOptions: null, │ │ │ │ │ - selectedFeature: null, │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.features = [] │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.destroy.apply(this, arguments); │ │ │ │ │ - this.clearFeatures(); │ │ │ │ │ - this.features = null │ │ │ │ │ + drawLineString: function(geometry, style, featureId) { │ │ │ │ │ + style = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + fill: false │ │ │ │ │ + }, style); │ │ │ │ │ + this.drawLinearRing(geometry, style, featureId) │ │ │ │ │ }, │ │ │ │ │ - loadText: function() { │ │ │ │ │ - if (!this.loaded) { │ │ │ │ │ - if (this.location != null) { │ │ │ │ │ - var onFail = function(e) { │ │ │ │ │ - this.events.triggerEvent("loadend") │ │ │ │ │ - }; │ │ │ │ │ - this.events.triggerEvent("loadstart"); │ │ │ │ │ - OpenLayers.Request.GET({ │ │ │ │ │ - url: this.location, │ │ │ │ │ - success: this.parseData, │ │ │ │ │ - failure: onFail, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.loaded = true │ │ │ │ │ + drawLinearRing: function(geometry, style, featureId) { │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.renderPath(this.canvas, geometry, style, featureId, "fill"); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.renderPath(this.hitContext, geometry, style, featureId, "fill") │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, minor) { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - if (this.visibility && !this.loaded) { │ │ │ │ │ - this.loadText() │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.renderPath(this.canvas, geometry, style, featureId, "stroke"); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ + this.renderPath(this.hitContext, geometry, style, featureId, "stroke") │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + this.setCanvasStyle("reset") │ │ │ │ │ }, │ │ │ │ │ - parseData: function(ajaxRequest) { │ │ │ │ │ - var text = ajaxRequest.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() │ │ │ │ │ + renderPath: function(context, geometry, style, featureId, type) { │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + context.beginPath(); │ │ │ │ │ + var start = this.getLocalXY(components[0]); │ │ │ │ │ + var x = start[0]; │ │ │ │ │ + var y = start[1]; │ │ │ │ │ + if (!isNaN(x) && !isNaN(y)) { │ │ │ │ │ + context.moveTo(start[0], start[1]); │ │ │ │ │ + for (var i = 1; i < len; ++i) { │ │ │ │ │ + var pt = this.getLocalXY(components[i]); │ │ │ │ │ + context.lineTo(pt[0], pt[1]) │ │ │ │ │ + } │ │ │ │ │ + if (type === "fill") { │ │ │ │ │ + context.fill() │ │ │ │ │ + } else { │ │ │ │ │ + context.stroke() │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var parser = new OpenLayers.Format.Text(options); │ │ │ │ │ - var features = parser.read(text); │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - var data = {}; │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - var location; │ │ │ │ │ - var iconSize, iconOffset; │ │ │ │ │ - location = new OpenLayers.LonLat(feature.geometry.x, feature.geometry.y); │ │ │ │ │ - if (feature.style.graphicWidth && feature.style.graphicHeight) { │ │ │ │ │ - iconSize = new OpenLayers.Size(feature.style.graphicWidth, feature.style.graphicHeight) │ │ │ │ │ + }, │ │ │ │ │ + drawPolygon: function(geometry, style, featureId) { │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + this.drawLinearRing(components[0], style, featureId); │ │ │ │ │ + for (var i = 1; i < len; ++i) { │ │ │ │ │ + this.canvas.globalCompositeOperation = "destination-out"; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.globalCompositeOperation = "destination-out" │ │ │ │ │ } │ │ │ │ │ - if (feature.style.graphicXOffset !== undefined && feature.style.graphicYOffset !== undefined) { │ │ │ │ │ - iconOffset = new OpenLayers.Pixel(feature.style.graphicXOffset, feature.style.graphicYOffset) │ │ │ │ │ + this.drawLinearRing(components[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + stroke: false, │ │ │ │ │ + fillOpacity: 1 │ │ │ │ │ + }, style), featureId); │ │ │ │ │ + this.canvas.globalCompositeOperation = "source-over"; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.globalCompositeOperation = "source-over" │ │ │ │ │ } │ │ │ │ │ - if (feature.style.externalGraphic != null) { │ │ │ │ │ - data.icon = new OpenLayers.Icon(feature.style.externalGraphic, iconSize, iconOffset) │ │ │ │ │ - } else { │ │ │ │ │ - data.icon = OpenLayers.Marker.defaultIcon(); │ │ │ │ │ - if (iconSize != null) { │ │ │ │ │ - data.icon.setSize(iconSize) │ │ │ │ │ + this.drawLinearRing(components[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + fill: false │ │ │ │ │ + }, style), featureId) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + drawText: function(location, style) { │ │ │ │ │ + var pt = this.getLocalXY(location); │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ + this.canvas.fillStyle = style.fontColor; │ │ │ │ │ + this.canvas.globalAlpha = style.fontOpacity || 1; │ │ │ │ │ + var fontStyle = [style.fontStyle ? style.fontStyle : "normal", "normal", style.fontWeight ? style.fontWeight : "normal", style.fontSize ? style.fontSize : "1em", style.fontFamily ? style.fontFamily : "sans-serif"].join(" "); │ │ │ │ │ + var labelRows = style.label.split("\n"); │ │ │ │ │ + var numRows = labelRows.length; │ │ │ │ │ + if (this.canvas.fillText) { │ │ │ │ │ + this.canvas.font = fontStyle; │ │ │ │ │ + this.canvas.textAlign = OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[0]] || "center"; │ │ │ │ │ + this.canvas.textBaseline = OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[1]] || "middle"; │ │ │ │ │ + var vfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5 │ │ │ │ │ + } │ │ │ │ │ + var lineHeight = this.canvas.measureText("Mg").height || this.canvas.measureText("xx").width; │ │ │ │ │ + pt[1] += lineHeight * vfactor * (numRows - 1); │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + if (style.labelOutlineWidth) { │ │ │ │ │ + this.canvas.save(); │ │ │ │ │ + this.canvas.globalAlpha = style.labelOutlineOpacity || style.fontOpacity || 1; │ │ │ │ │ + this.canvas.strokeStyle = style.labelOutlineColor; │ │ │ │ │ + this.canvas.lineWidth = style.labelOutlineWidth; │ │ │ │ │ + this.canvas.strokeText(labelRows[i], pt[0], pt[1] + lineHeight * i + 1); │ │ │ │ │ + this.canvas.restore() │ │ │ │ │ } │ │ │ │ │ + this.canvas.fillText(labelRows[i], pt[0], pt[1] + lineHeight * i) │ │ │ │ │ } │ │ │ │ │ - if (feature.attributes.title != null && feature.attributes.description != null) { │ │ │ │ │ - data["popupContentHTML"] = "<h2>" + feature.attributes.title + "</h2>" + "<p>" + feature.attributes.description + "</p>" │ │ │ │ │ + } else if (this.canvas.mozDrawText) { │ │ │ │ │ + this.canvas.mozTextStyle = fontStyle; │ │ │ │ │ + var hfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[0]]; │ │ │ │ │ + if (hfactor == null) { │ │ │ │ │ + hfactor = -.5 │ │ │ │ │ } │ │ │ │ │ - data["overflow"] = feature.attributes.overflow || "auto"; │ │ │ │ │ - var markerFeature = new OpenLayers.Feature(this, location, data); │ │ │ │ │ - this.features.push(markerFeature); │ │ │ │ │ - var marker = markerFeature.createMarker(); │ │ │ │ │ - if (feature.attributes.title != null && feature.attributes.description != null) { │ │ │ │ │ - marker.events.register("click", markerFeature, this.markerClick) │ │ │ │ │ + var vfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5 │ │ │ │ │ + } │ │ │ │ │ + var lineHeight = this.canvas.mozMeasureText("xx"); │ │ │ │ │ + pt[1] += lineHeight * (1 + vfactor * numRows); │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + var x = pt[0] + hfactor * this.canvas.mozMeasureText(labelRows[i]); │ │ │ │ │ + var y = pt[1] + i * lineHeight; │ │ │ │ │ + this.canvas.translate(x, y); │ │ │ │ │ + this.canvas.mozDrawText(labelRows[i]); │ │ │ │ │ + this.canvas.translate(-x, -y) │ │ │ │ │ } │ │ │ │ │ - this.addMarker(marker) │ │ │ │ │ } │ │ │ │ │ - this.events.triggerEvent("loadend") │ │ │ │ │ + this.setCanvasStyle("reset") │ │ │ │ │ }, │ │ │ │ │ - markerClick: function(evt) { │ │ │ │ │ - var sameMarkerClicked = this == this.layer.selectedFeature; │ │ │ │ │ - this.layer.selectedFeature = !sameMarkerClicked ? this : null; │ │ │ │ │ - for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ - this.layer.map.removePopup(this.layer.map.popups[i]) │ │ │ │ │ - } │ │ │ │ │ - if (!sameMarkerClicked) { │ │ │ │ │ - this.layer.map.addPopup(this.createPopup()) │ │ │ │ │ + getLocalXY: function(point) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var extent = this.extent; │ │ │ │ │ + var x = (point.x - this.featureDx) / resolution + -extent.left / resolution; │ │ │ │ │ + var y = extent.top / resolution - point.y / resolution; │ │ │ │ │ + return [x, y] │ │ │ │ │ + }, │ │ │ │ │ + clear: function() { │ │ │ │ │ + var height = this.root.height; │ │ │ │ │ + var width = this.root.width; │ │ │ │ │ + this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ + this.features = {}; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.clearRect(0, 0, width, height) │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ }, │ │ │ │ │ - clearFeatures: function() { │ │ │ │ │ - if (this.features != null) { │ │ │ │ │ - while (this.features.length > 0) { │ │ │ │ │ - var feature = this.features[0]; │ │ │ │ │ - OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ - feature.destroy() │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var featureId, feature; │ │ │ │ │ + if (this.hitDetection && this.root.style.display !== "none") { │ │ │ │ │ + if (!this.map.dragging) { │ │ │ │ │ + var xy = evt.xy; │ │ │ │ │ + var x = xy.x | 0; │ │ │ │ │ + var y = xy.y | 0; │ │ │ │ │ + var data = this.hitContext.getImageData(x, y, 1, 1).data; │ │ │ │ │ + if (data[3] === 255) { │ │ │ │ │ + var id = data[2] + 256 * (data[1] + 256 * data[0]); │ │ │ │ │ + if (id) { │ │ │ │ │ + featureId = "OpenLayers_Feature_Vector_" + (id - 1 + this.hitOverflow); │ │ │ │ │ + try { │ │ │ │ │ + feature = this.features[featureId][0] │ │ │ │ │ + } catch (err) {} │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return feature │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Text" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Bing = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ - key: null, │ │ │ │ │ - serverResolutions: [156543.03390625, 78271.516953125, 39135.7584765625, 19567.87923828125, 9783.939619140625, 4891.9698095703125, 2445.9849047851562, 1222.9924523925781, 611.4962261962891, 305.74811309814453, 152.87405654907226, 76.43702827453613, 38.218514137268066, 19.109257068634033, 9.554628534317017, 4.777314267158508, 2.388657133579254, 1.194328566789627, .5971642833948135, .29858214169740677, .14929107084870338, .07464553542435169], │ │ │ │ │ - attributionTemplate: '<span class="olBingAttribution ${type}">' + '<div><a target="_blank" href="http://www.bing.com/maps/">' + '<img src="${logo}" /></a></div>${copyrights}' + '<a style="white-space: nowrap" target="_blank" ' + 'href="http://www.microsoft.com/maps/product/terms.html">' + "Terms of Use</a></span>", │ │ │ │ │ - metadata: null, │ │ │ │ │ - protocolRegex: /^http:/i, │ │ │ │ │ - type: "Road", │ │ │ │ │ - culture: "en-US", │ │ │ │ │ - metadataParams: null, │ │ │ │ │ - tileOptions: null, │ │ │ │ │ - protocol: ~window.location.href.indexOf("http") ? "" : "http:", │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - sphericalMercator: true │ │ │ │ │ - }, options); │ │ │ │ │ - var name = options.name || "Bing " + (options.type || this.type); │ │ │ │ │ - var newArgs = [name, null, options]; │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, newArgs); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - crossOriginKeyword: "anonymous" │ │ │ │ │ - }, this.options.tileOptions); │ │ │ │ │ - this.loadMetadata() │ │ │ │ │ - }, │ │ │ │ │ - loadMetadata: function() { │ │ │ │ │ - this._callbackId = "_callback_" + this.id.replace(/\./g, "_"); │ │ │ │ │ - window[this._callbackId] = OpenLayers.Function.bind(OpenLayers.Layer.Bing.processMetadata, this); │ │ │ │ │ - var params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - key: this.key, │ │ │ │ │ - jsonp: this._callbackId, │ │ │ │ │ - include: "ImageryProviders" │ │ │ │ │ - }, this.metadataParams); │ │ │ │ │ - var url = this.protocol + "//dev.virtualearth.net/REST/v1/Imagery/Metadata/" + this.type + "?" + OpenLayers.Util.getParameterString(params); │ │ │ │ │ - var script = document.createElement("script"); │ │ │ │ │ - script.type = "text/javascript"; │ │ │ │ │ - script.src = url; │ │ │ │ │ - script.id = this._callbackId; │ │ │ │ │ - document.getElementsByTagName("head")[0].appendChild(script) │ │ │ │ │ - }, │ │ │ │ │ - initLayer: function() { │ │ │ │ │ - var res = this.metadata.resourceSets[0].resources[0]; │ │ │ │ │ - var url = res.imageUrl.replace("{quadkey}", "${quadkey}"); │ │ │ │ │ - url = url.replace("{culture}", this.culture); │ │ │ │ │ - url = url.replace(this.protocolRegex, this.protocol); │ │ │ │ │ - this.url = []; │ │ │ │ │ - for (var i = 0; i < res.imageUrlSubdomains.length; ++i) { │ │ │ │ │ - this.url.push(url.replace("{subdomain}", res.imageUrlSubdomains[i])) │ │ │ │ │ + eraseFeatures: function(features) { │ │ │ │ │ + if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ + features = [features] │ │ │ │ │ } │ │ │ │ │ - this.addOptions({ │ │ │ │ │ - maxResolution: Math.min(this.serverResolutions[res.zoomMin], this.maxResolution || Number.POSITIVE_INFINITY), │ │ │ │ │ - numZoomLevels: Math.min(res.zoomMax + 1 - res.zoomMin, this.numZoomLevels) │ │ │ │ │ - }, true); │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - this.redraw() │ │ │ │ │ + for (var i = 0; i < features.length; ++i) { │ │ │ │ │ + delete this.features[features[i].id] │ │ │ │ │ } │ │ │ │ │ - this.updateAttribution() │ │ │ │ │ + this.redraw() │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - if (!this.url) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var xyz = this.getXYZ(bounds), │ │ │ │ │ - x = xyz.x, │ │ │ │ │ - y = xyz.y, │ │ │ │ │ - z = xyz.z; │ │ │ │ │ - var quadDigits = []; │ │ │ │ │ - for (var i = z; i > 0; --i) { │ │ │ │ │ - var digit = "0"; │ │ │ │ │ - var mask = 1 << i - 1; │ │ │ │ │ - if ((x & mask) != 0) { │ │ │ │ │ - digit++ │ │ │ │ │ + redraw: function() { │ │ │ │ │ + if (!this.locked) { │ │ │ │ │ + var height = this.root.height; │ │ │ │ │ + var width = this.root.width; │ │ │ │ │ + this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.clearRect(0, 0, width, height) │ │ │ │ │ } │ │ │ │ │ - if ((y & mask) != 0) { │ │ │ │ │ - digit++; │ │ │ │ │ - digit++ │ │ │ │ │ + var labelMap = []; │ │ │ │ │ + var feature, geometry, style; │ │ │ │ │ + var worldBounds = this.map.baseLayer && this.map.baseLayer.wrapDateLine && this.map.getMaxExtent(); │ │ │ │ │ + for (var id in this.features) { │ │ │ │ │ + if (!this.features.hasOwnProperty(id)) { │ │ │ │ │ + continue │ │ │ │ │ + } │ │ │ │ │ + feature = this.features[id][0]; │ │ │ │ │ + geometry = feature.geometry; │ │ │ │ │ + this.calculateFeatureDx(geometry.getBounds(), worldBounds); │ │ │ │ │ + style = this.features[id][1]; │ │ │ │ │ + this.drawGeometry(geometry, style, feature.id); │ │ │ │ │ + if (style.label) { │ │ │ │ │ + labelMap.push([feature, style]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var item; │ │ │ │ │ + for (var i = 0, len = labelMap.length; i < len; ++i) { │ │ │ │ │ + item = labelMap[i]; │ │ │ │ │ + this.drawText(item[0].geometry.getCentroid(), item[1]) │ │ │ │ │ } │ │ │ │ │ - quadDigits.push(digit) │ │ │ │ │ } │ │ │ │ │ - var quadKey = quadDigits.join(""); │ │ │ │ │ - var url = this.selectUrl("" + x + y + z, this.url); │ │ │ │ │ - return OpenLayers.String.format(url, { │ │ │ │ │ - quadkey: quadKey │ │ │ │ │ - }) │ │ │ │ │ }, │ │ │ │ │ - updateAttribution: function() { │ │ │ │ │ - var metadata = this.metadata; │ │ │ │ │ - if (!metadata.resourceSets || !this.map || !this.map.center) { │ │ │ │ │ - return │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.Canvas" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Renderer.Canvas.LABEL_ALIGN = { │ │ │ │ │ + l: "left", │ │ │ │ │ + r: "right", │ │ │ │ │ + t: "top", │ │ │ │ │ + b: "bottom" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.Canvas.LABEL_FACTOR = { │ │ │ │ │ + l: 0, │ │ │ │ │ + r: -1, │ │ │ │ │ + t: 0, │ │ │ │ │ + b: -1 │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.Canvas.drawImageScaleFactor = null; │ │ │ │ │ +OpenLayers.ElementsIndexer = OpenLayers.Class({ │ │ │ │ │ + maxZIndex: null, │ │ │ │ │ + order: null, │ │ │ │ │ + indices: null, │ │ │ │ │ + compare: null, │ │ │ │ │ + initialize: function(yOrdering) { │ │ │ │ │ + this.compare = yOrdering ? OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_Y_ORDER : OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_DRAWING_ORDER; │ │ │ │ │ + this.clear() │ │ │ │ │ + }, │ │ │ │ │ + insert: function(newNode) { │ │ │ │ │ + if (this.exists(newNode)) { │ │ │ │ │ + this.remove(newNode) │ │ │ │ │ } │ │ │ │ │ - var res = metadata.resourceSets[0].resources[0]; │ │ │ │ │ - var extent = this.map.getExtent().transform(this.map.getProjectionObject(), new OpenLayers.Projection("EPSG:4326")); │ │ │ │ │ - var providers = res.imageryProviders || [], │ │ │ │ │ - zoom = OpenLayers.Util.indexOf(this.serverResolutions, this.getServerResolution()), │ │ │ │ │ - copyrights = "", │ │ │ │ │ - provider, i, ii, j, jj, bbox, coverage; │ │ │ │ │ - for (i = 0, ii = providers.length; i < ii; ++i) { │ │ │ │ │ - provider = providers[i]; │ │ │ │ │ - for (j = 0, jj = provider.coverageAreas.length; j < jj; ++j) { │ │ │ │ │ - coverage = provider.coverageAreas[j]; │ │ │ │ │ - bbox = OpenLayers.Bounds.fromArray(coverage.bbox, true); │ │ │ │ │ - if (extent.intersectsBounds(bbox) && zoom <= coverage.zoomMax && zoom >= coverage.zoomMin) { │ │ │ │ │ - copyrights += provider.attribution + " " │ │ │ │ │ - } │ │ │ │ │ + var nodeId = newNode.id; │ │ │ │ │ + this.determineZIndex(newNode); │ │ │ │ │ + var leftIndex = -1; │ │ │ │ │ + var rightIndex = this.order.length; │ │ │ │ │ + var middle; │ │ │ │ │ + while (rightIndex - leftIndex > 1) { │ │ │ │ │ + middle = parseInt((leftIndex + rightIndex) / 2); │ │ │ │ │ + var placement = this.compare(this, newNode, OpenLayers.Util.getElement(this.order[middle])); │ │ │ │ │ + if (placement > 0) { │ │ │ │ │ + leftIndex = middle │ │ │ │ │ + } else { │ │ │ │ │ + rightIndex = middle │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var logo = metadata.brandLogoUri.replace(this.protocolRegex, this.protocol); │ │ │ │ │ - this.attribution = OpenLayers.String.format(this.attributionTemplate, { │ │ │ │ │ - type: this.type.toLowerCase(), │ │ │ │ │ - logo: logo, │ │ │ │ │ - copyrights: copyrights │ │ │ │ │ - }); │ │ │ │ │ - this.map && this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "attribution" │ │ │ │ │ - }) │ │ │ │ │ + this.order.splice(rightIndex, 0, nodeId); │ │ │ │ │ + this.indices[nodeId] = this.getZIndex(newNode); │ │ │ │ │ + return this.getNextElement(rightIndex) │ │ │ │ │ }, │ │ │ │ │ - setMap: function() { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.register("moveend", this, this.updateAttribution) │ │ │ │ │ + remove: function(node) { │ │ │ │ │ + var nodeId = node.id; │ │ │ │ │ + var arrayIndex = OpenLayers.Util.indexOf(this.order, nodeId); │ │ │ │ │ + if (arrayIndex >= 0) { │ │ │ │ │ + this.order.splice(arrayIndex, 1); │ │ │ │ │ + delete this.indices[nodeId]; │ │ │ │ │ + if (this.order.length > 0) { │ │ │ │ │ + var lastId = this.order[this.order.length - 1]; │ │ │ │ │ + this.maxZIndex = this.indices[lastId] │ │ │ │ │ + } else { │ │ │ │ │ + this.maxZIndex = 0 │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Bing(this.options) │ │ │ │ │ + clear: function() { │ │ │ │ │ + this.order = []; │ │ │ │ │ + this.indices = {}; │ │ │ │ │ + this.maxZIndex = 0 │ │ │ │ │ + }, │ │ │ │ │ + exists: function(node) { │ │ │ │ │ + return this.indices[node.id] != null │ │ │ │ │ + }, │ │ │ │ │ + getZIndex: function(node) { │ │ │ │ │ + return node._style.graphicZIndex │ │ │ │ │ + }, │ │ │ │ │ + determineZIndex: function(node) { │ │ │ │ │ + var zIndex = node._style.graphicZIndex; │ │ │ │ │ + if (zIndex == null) { │ │ │ │ │ + zIndex = this.maxZIndex; │ │ │ │ │ + node._style.graphicZIndex = zIndex │ │ │ │ │ + } else if (zIndex > this.maxZIndex) { │ │ │ │ │ + this.maxZIndex = zIndex │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.map && this.map.events.unregister("moveend", this, this.updateAttribution); │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.destroy.apply(this, arguments) │ │ │ │ │ + getNextElement: function(index) { │ │ │ │ │ + var nextIndex = index + 1; │ │ │ │ │ + if (nextIndex < this.order.length) { │ │ │ │ │ + var nextElement = OpenLayers.Util.getElement(this.order[nextIndex]); │ │ │ │ │ + if (nextElement == undefined) { │ │ │ │ │ + nextElement = this.getNextElement(nextIndex) │ │ │ │ │ + } │ │ │ │ │ + return nextElement │ │ │ │ │ + } else { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Bing" │ │ │ │ │ + CLASS_NAME: "OpenLayers.ElementsIndexer" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.Bing.processMetadata = function(metadata) { │ │ │ │ │ - this.metadata = metadata; │ │ │ │ │ - this.initLayer(); │ │ │ │ │ - var script = document.getElementById(this._callbackId); │ │ │ │ │ - script.parentNode.removeChild(script); │ │ │ │ │ - window[this._callbackId] = undefined; │ │ │ │ │ - delete this._callbackId │ │ │ │ │ +OpenLayers.ElementsIndexer.IndexingMethods = { │ │ │ │ │ + Z_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var newZIndex = indexer.getZIndex(newNode); │ │ │ │ │ + var returnVal = 0; │ │ │ │ │ + if (nextNode) { │ │ │ │ │ + var nextZIndex = indexer.getZIndex(nextNode); │ │ │ │ │ + returnVal = newZIndex - nextZIndex │ │ │ │ │ + } │ │ │ │ │ + return returnVal │ │ │ │ │ + }, │ │ │ │ │ + Z_ORDER_DRAWING_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER(indexer, newNode, nextNode); │ │ │ │ │ + if (nextNode && returnVal == 0) { │ │ │ │ │ + returnVal = 1 │ │ │ │ │ + } │ │ │ │ │ + return returnVal │ │ │ │ │ + }, │ │ │ │ │ + Z_ORDER_Y_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER(indexer, newNode, nextNode); │ │ │ │ │ + if (nextNode && returnVal === 0) { │ │ │ │ │ + var result = nextNode._boundsBottom - newNode._boundsBottom; │ │ │ │ │ + returnVal = result === 0 ? 1 : result │ │ │ │ │ + } │ │ │ │ │ + return returnVal │ │ │ │ │ + } │ │ │ │ │ }; │ │ │ │ │ -OpenLayers.Tile.UTFGrid = OpenLayers.Class(OpenLayers.Tile, { │ │ │ │ │ - url: null, │ │ │ │ │ - utfgridResolution: 2, │ │ │ │ │ - json: null, │ │ │ │ │ - format: null, │ │ │ │ │ +OpenLayers.Renderer.Elements = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ + rendererRoot: null, │ │ │ │ │ + root: null, │ │ │ │ │ + vectorRoot: null, │ │ │ │ │ + textRoot: null, │ │ │ │ │ + xmlns: null, │ │ │ │ │ + xOffset: 0, │ │ │ │ │ + indexer: null, │ │ │ │ │ + BACKGROUND_ID_SUFFIX: "_background", │ │ │ │ │ + LABEL_ID_SUFFIX: "_label", │ │ │ │ │ + LABEL_OUTLINE_SUFFIX: "_outline", │ │ │ │ │ + initialize: function(containerID, options) { │ │ │ │ │ + OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.rendererRoot = this.createRenderRoot(); │ │ │ │ │ + this.root = this.createRoot("_root"); │ │ │ │ │ + this.vectorRoot = this.createRoot("_vroot"); │ │ │ │ │ + this.textRoot = this.createRoot("_troot"); │ │ │ │ │ + this.root.appendChild(this.vectorRoot); │ │ │ │ │ + this.root.appendChild(this.textRoot); │ │ │ │ │ + this.rendererRoot.appendChild(this.root); │ │ │ │ │ + this.container.appendChild(this.rendererRoot); │ │ │ │ │ + if (options && (options.zIndexing || options.yOrdering)) { │ │ │ │ │ + this.indexer = new OpenLayers.ElementsIndexer(options.yOrdering) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ destroy: function() { │ │ │ │ │ this.clear(); │ │ │ │ │ - OpenLayers.Tile.prototype.destroy.apply(this, arguments) │ │ │ │ │ + this.rendererRoot = null; │ │ │ │ │ + this.root = null; │ │ │ │ │ + this.xmlns = null; │ │ │ │ │ + OpenLayers.Renderer.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - var drawn = OpenLayers.Tile.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (drawn) { │ │ │ │ │ - if (this.isLoading) { │ │ │ │ │ - this.abortLoading(); │ │ │ │ │ - this.events.triggerEvent("reload") │ │ │ │ │ - } else { │ │ │ │ │ - this.isLoading = true; │ │ │ │ │ - this.events.triggerEvent("loadstart") │ │ │ │ │ + clear: function() { │ │ │ │ │ + var child; │ │ │ │ │ + var root = this.vectorRoot; │ │ │ │ │ + if (root) { │ │ │ │ │ + while (child = root.firstChild) { │ │ │ │ │ + root.removeChild(child) │ │ │ │ │ } │ │ │ │ │ - this.url = this.layer.getURL(this.bounds); │ │ │ │ │ - if (this.layer.useJSONP) { │ │ │ │ │ - var ols = new OpenLayers.Protocol.Script({ │ │ │ │ │ - url: this.url, │ │ │ │ │ - callback: function(response) { │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ - this.json = response.data │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - ols.read(); │ │ │ │ │ - this.request = ols │ │ │ │ │ - } else { │ │ │ │ │ - this.request = OpenLayers.Request.GET({ │ │ │ │ │ - url: this.url, │ │ │ │ │ - callback: function(response) { │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ - if (response.status === 200) { │ │ │ │ │ - this.parseData(response.responseText) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ + } │ │ │ │ │ + root = this.textRoot; │ │ │ │ │ + if (root) { │ │ │ │ │ + while (child = root.firstChild) { │ │ │ │ │ + root.removeChild(child) │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - this.unload() │ │ │ │ │ } │ │ │ │ │ - return drawn │ │ │ │ │ - }, │ │ │ │ │ - abortLoading: function() { │ │ │ │ │ - if (this.request) { │ │ │ │ │ - this.request.abort(); │ │ │ │ │ - delete this.request │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + this.indexer.clear() │ │ │ │ │ } │ │ │ │ │ - this.isLoading = false │ │ │ │ │ }, │ │ │ │ │ - getFeatureInfo: function(i, j) { │ │ │ │ │ - var info = null; │ │ │ │ │ - if (this.json) { │ │ │ │ │ - var id = this.getFeatureId(i, j); │ │ │ │ │ - if (id !== null) { │ │ │ │ │ - info = { │ │ │ │ │ - id: id, │ │ │ │ │ - data: this.json.data[id] │ │ │ │ │ - } │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + var rightOfDateLine, ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ + extent = extent.scale(1 / ratio), │ │ │ │ │ + world = this.map.getMaxExtent(); │ │ │ │ │ + if (world.right > extent.left && world.right < extent.right) { │ │ │ │ │ + rightOfDateLine = true │ │ │ │ │ + } else if (world.left > extent.left && world.left < extent.right) { │ │ │ │ │ + rightOfDateLine = false │ │ │ │ │ + } │ │ │ │ │ + if (rightOfDateLine !== this.rightOfDateLine || resolutionChanged) { │ │ │ │ │ + coordSysUnchanged = false; │ │ │ │ │ + this.xOffset = rightOfDateLine === true ? world.getWidth() / resolution : 0 │ │ │ │ │ } │ │ │ │ │ + this.rightOfDateLine = rightOfDateLine │ │ │ │ │ } │ │ │ │ │ - return info │ │ │ │ │ + return coordSysUnchanged │ │ │ │ │ }, │ │ │ │ │ - getFeatureId: function(i, j) { │ │ │ │ │ - var id = null; │ │ │ │ │ - if (this.json) { │ │ │ │ │ - var resolution = this.utfgridResolution; │ │ │ │ │ - var row = Math.floor(j / resolution); │ │ │ │ │ - var col = Math.floor(i / resolution); │ │ │ │ │ - var charCode = this.json.grid[row].charCodeAt(col); │ │ │ │ │ - var index = this.indexFromCharCode(charCode); │ │ │ │ │ - var keys = this.json.keys; │ │ │ │ │ - if (!isNaN(index) && index in keys) { │ │ │ │ │ - id = keys[index] │ │ │ │ │ + getNodeType: function(geometry, style) {}, │ │ │ │ │ + drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ + var className = geometry.CLASS_NAME; │ │ │ │ │ + var rendered = true; │ │ │ │ │ + if (className == "OpenLayers.Geometry.Collection" || className == "OpenLayers.Geometry.MultiPoint" || className == "OpenLayers.Geometry.MultiLineString" || className == "OpenLayers.Geometry.MultiPolygon") { │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + rendered = this.drawGeometry(geometry.components[i], style, featureId) && rendered │ │ │ │ │ } │ │ │ │ │ + return rendered │ │ │ │ │ } │ │ │ │ │ - return id │ │ │ │ │ - }, │ │ │ │ │ - indexFromCharCode: function(charCode) { │ │ │ │ │ - if (charCode >= 93) { │ │ │ │ │ - charCode-- │ │ │ │ │ + rendered = false; │ │ │ │ │ + var removeBackground = false; │ │ │ │ │ + if (style.display != "none") { │ │ │ │ │ + if (style.backgroundGraphic) { │ │ │ │ │ + this.redrawBackgroundNode(geometry.id, geometry, style, featureId) │ │ │ │ │ + } else { │ │ │ │ │ + removeBackground = true │ │ │ │ │ + } │ │ │ │ │ + rendered = this.redrawNode(geometry.id, geometry, style, featureId) │ │ │ │ │ } │ │ │ │ │ - if (charCode >= 35) { │ │ │ │ │ - charCode-- │ │ │ │ │ + if (rendered == false) { │ │ │ │ │ + var node = document.getElementById(geometry.id); │ │ │ │ │ + if (node) { │ │ │ │ │ + if (node._style.backgroundGraphic) { │ │ │ │ │ + removeBackground = true │ │ │ │ │ + } │ │ │ │ │ + node.parentNode.removeChild(node) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return charCode - 32 │ │ │ │ │ - }, │ │ │ │ │ - parseData: function(str) { │ │ │ │ │ - if (!this.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.JSON │ │ │ │ │ + if (removeBackground) { │ │ │ │ │ + var node = document.getElementById(geometry.id + this.BACKGROUND_ID_SUFFIX); │ │ │ │ │ + if (node) { │ │ │ │ │ + node.parentNode.removeChild(node) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.json = this.format.read(str) │ │ │ │ │ + return rendered │ │ │ │ │ }, │ │ │ │ │ - clear: function() { │ │ │ │ │ - this.json = null │ │ │ │ │ + redrawNode: function(id, geometry, style, featureId) { │ │ │ │ │ + style = this.applyDefaultSymbolizer(style); │ │ │ │ │ + var node = this.nodeFactory(id, this.getNodeType(geometry, style)); │ │ │ │ │ + node._featureId = featureId; │ │ │ │ │ + node._boundsBottom = geometry.getBounds().bottom; │ │ │ │ │ + node._geometryClass = geometry.CLASS_NAME; │ │ │ │ │ + node._style = style; │ │ │ │ │ + var drawResult = this.drawGeometryNode(node, geometry, style); │ │ │ │ │ + if (drawResult === false) { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + node = drawResult.node; │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + var insert = this.indexer.insert(node); │ │ │ │ │ + if (insert) { │ │ │ │ │ + this.vectorRoot.insertBefore(node, insert) │ │ │ │ │ + } else { │ │ │ │ │ + this.vectorRoot.appendChild(node) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + if (node.parentNode !== this.vectorRoot) { │ │ │ │ │ + this.vectorRoot.appendChild(node) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.postDraw(node); │ │ │ │ │ + return drawResult.complete │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Tile.UTFGrid" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.UTFGrid = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ - isBaseLayer: false, │ │ │ │ │ - projection: new OpenLayers.Projection("EPSG:900913"), │ │ │ │ │ - useJSONP: false, │ │ │ │ │ - tileClass: OpenLayers.Tile.UTFGrid, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [options.name, options.url, {}, options]); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - utfgridResolution: this.utfgridResolution │ │ │ │ │ - }, this.tileOptions) │ │ │ │ │ + redrawBackgroundNode: function(id, geometry, style, featureId) { │ │ │ │ │ + var backgroundStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ + backgroundStyle.externalGraphic = backgroundStyle.backgroundGraphic; │ │ │ │ │ + backgroundStyle.graphicXOffset = backgroundStyle.backgroundXOffset; │ │ │ │ │ + backgroundStyle.graphicYOffset = backgroundStyle.backgroundYOffset; │ │ │ │ │ + backgroundStyle.graphicZIndex = backgroundStyle.backgroundGraphicZIndex; │ │ │ │ │ + backgroundStyle.graphicWidth = backgroundStyle.backgroundWidth || backgroundStyle.graphicWidth; │ │ │ │ │ + backgroundStyle.graphicHeight = backgroundStyle.backgroundHeight || backgroundStyle.graphicHeight; │ │ │ │ │ + backgroundStyle.backgroundGraphic = null; │ │ │ │ │ + backgroundStyle.backgroundXOffset = null; │ │ │ │ │ + backgroundStyle.backgroundYOffset = null; │ │ │ │ │ + backgroundStyle.backgroundGraphicZIndex = null; │ │ │ │ │ + return this.redrawNode(id + this.BACKGROUND_ID_SUFFIX, geometry, backgroundStyle, null) │ │ │ │ │ }, │ │ │ │ │ - createBackBuffer: function() {}, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.UTFGrid(this.getOptions()) │ │ │ │ │ + drawGeometryNode: function(node, geometry, style) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + var options = { │ │ │ │ │ + isFilled: style.fill === undefined ? true : style.fill, │ │ │ │ │ + isStroked: style.stroke === undefined ? !!style.strokeWidth : style.stroke │ │ │ │ │ + }; │ │ │ │ │ + var drawn; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.graphic === false) { │ │ │ │ │ + options.isFilled = false; │ │ │ │ │ + options.isStroked = false │ │ │ │ │ + } │ │ │ │ │ + drawn = this.drawPoint(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + options.isFilled = false; │ │ │ │ │ + drawn = this.drawLineString(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + drawn = this.drawLinearRing(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + drawn = this.drawPolygon(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + drawn = this.drawRectangle(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - getFeatureInfo: function(location) { │ │ │ │ │ - var info = null; │ │ │ │ │ - var tileInfo = this.getTileData(location); │ │ │ │ │ - if (tileInfo && tileInfo.tile) { │ │ │ │ │ - info = tileInfo.tile.getFeatureInfo(tileInfo.i, tileInfo.j) │ │ │ │ │ + node._options = options; │ │ │ │ │ + if (drawn != false) { │ │ │ │ │ + return { │ │ │ │ │ + node: this.setStyle(node, style, options, geometry), │ │ │ │ │ + complete: drawn │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return info │ │ │ │ │ }, │ │ │ │ │ - getFeatureId: function(location) { │ │ │ │ │ - var id = null; │ │ │ │ │ - var info = this.getTileData(location); │ │ │ │ │ - if (info.tile) { │ │ │ │ │ - id = info.tile.getFeatureId(info.i, info.j) │ │ │ │ │ + postDraw: function(node) {}, │ │ │ │ │ + drawPoint: function(node, geometry) {}, │ │ │ │ │ + drawLineString: function(node, geometry) {}, │ │ │ │ │ + drawLinearRing: function(node, geometry) {}, │ │ │ │ │ + drawPolygon: function(node, geometry) {}, │ │ │ │ │ + drawRectangle: function(node, geometry) {}, │ │ │ │ │ + drawCircle: function(node, geometry) {}, │ │ │ │ │ + removeText: function(featureId) { │ │ │ │ │ + var label = document.getElementById(featureId + this.LABEL_ID_SUFFIX); │ │ │ │ │ + if (label) { │ │ │ │ │ + this.textRoot.removeChild(label) │ │ │ │ │ + } │ │ │ │ │ + var outline = document.getElementById(featureId + this.LABEL_OUTLINE_SUFFIX); │ │ │ │ │ + if (outline) { │ │ │ │ │ + this.textRoot.removeChild(outline) │ │ │ │ │ } │ │ │ │ │ - return id │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.UTFGrid" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.WMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - service: "WMS", │ │ │ │ │ - version: "1.1.1", │ │ │ │ │ - request: "GetMap", │ │ │ │ │ - styles: "", │ │ │ │ │ - format: "image/jpeg" │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var target = evt.target; │ │ │ │ │ + var useElement = target && target.correspondingUseElement; │ │ │ │ │ + var node = useElement ? useElement : target || evt.srcElement; │ │ │ │ │ + return node._featureId │ │ │ │ │ }, │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - encodeBBOX: false, │ │ │ │ │ - noMagic: false, │ │ │ │ │ - yx: {}, │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ - if (parseFloat(params.VERSION) >= 1.3 && !params.EXCEPTIONS) { │ │ │ │ │ - params.EXCEPTIONS = "INIMAGE" │ │ │ │ │ - } │ │ │ │ │ - newArguments.push(name, url, params, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - OpenLayers.Util.applyDefaults(this.params, OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS)); │ │ │ │ │ - if (!this.noMagic && this.params.TRANSPARENT && this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ - if (options == null || !options.isBaseLayer) { │ │ │ │ │ - this.isBaseLayer = false │ │ │ │ │ + eraseGeometry: function(geometry, featureId) { │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPoint" || geometry.CLASS_NAME == "OpenLayers.Geometry.MultiLineString" || geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPolygon" || geometry.CLASS_NAME == "OpenLayers.Geometry.Collection") { │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + this.eraseGeometry(geometry.components[i], featureId) │ │ │ │ │ } │ │ │ │ │ - if (this.params.FORMAT == "image/jpeg") { │ │ │ │ │ - this.params.FORMAT = OpenLayers.Util.alphaHack() ? "image/gif" : "image/png" │ │ │ │ │ + } else { │ │ │ │ │ + var element = OpenLayers.Util.getElement(geometry.id); │ │ │ │ │ + if (element && element.parentNode) { │ │ │ │ │ + if (element.geometry) { │ │ │ │ │ + element.geometry.destroy(); │ │ │ │ │ + element.geometry = null │ │ │ │ │ + } │ │ │ │ │ + element.parentNode.removeChild(element); │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + this.indexer.remove(element) │ │ │ │ │ + } │ │ │ │ │ + if (element._style.backgroundGraphic) { │ │ │ │ │ + var backgroundId = geometry.id + this.BACKGROUND_ID_SUFFIX; │ │ │ │ │ + var bElem = OpenLayers.Util.getElement(backgroundId); │ │ │ │ │ + if (bElem && bElem.parentNode) { │ │ │ │ │ + bElem.parentNode.removeChild(bElem) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.WMS(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ + nodeFactory: function(id, type) { │ │ │ │ │ + var node = OpenLayers.Util.getElement(id); │ │ │ │ │ + if (node) { │ │ │ │ │ + if (!this.nodeTypeCompare(node, type)) { │ │ │ │ │ + node.parentNode.removeChild(node); │ │ │ │ │ + node = this.nodeFactory(id, type) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + node = this.createNode(type, id) │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - reverseAxisOrder: function() { │ │ │ │ │ - var projCode = this.projection.getCode(); │ │ │ │ │ - return parseFloat(this.params.VERSION) >= 1.3 && !!(this.yx[projCode] || OpenLayers.Projection.defaults[projCode] && OpenLayers.Projection.defaults[projCode].yx) │ │ │ │ │ + nodeTypeCompare: function(node, type) {}, │ │ │ │ │ + createNode: function(type, id) {}, │ │ │ │ │ + moveRoot: function(renderer) { │ │ │ │ │ + var root = this.root; │ │ │ │ │ + if (renderer.root.parentNode == this.rendererRoot) { │ │ │ │ │ + root = renderer.root │ │ │ │ │ + } │ │ │ │ │ + root.parentNode.removeChild(root); │ │ │ │ │ + renderer.rendererRoot.appendChild(root) │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var imageSize = this.getImageSize(); │ │ │ │ │ - var newParams = {}; │ │ │ │ │ - var reverseAxisOrder = this.reverseAxisOrder(); │ │ │ │ │ - newParams.BBOX = this.encodeBBOX ? bounds.toBBOX(null, reverseAxisOrder) : bounds.toArray(reverseAxisOrder); │ │ │ │ │ - newParams.WIDTH = imageSize.w; │ │ │ │ │ - newParams.HEIGHT = imageSize.h; │ │ │ │ │ - var requestString = this.getFullRequestString(newParams); │ │ │ │ │ - return requestString │ │ │ │ │ + getRenderLayerId: function() { │ │ │ │ │ + return this.root.parentNode.parentNode.id │ │ │ │ │ }, │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ - var newArguments = [upperParams]; │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, newArguments) │ │ │ │ │ + isComplexSymbol: function(graphicName) { │ │ │ │ │ + return graphicName != "circle" && !!graphicName │ │ │ │ │ }, │ │ │ │ │ - getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ - var mapProjection = this.map.getProjectionObject(); │ │ │ │ │ - var projectionCode = this.projection && this.projection.equals(mapProjection) ? this.projection.getCode() : mapProjection.getCode(); │ │ │ │ │ - var value = projectionCode == "none" ? null : projectionCode; │ │ │ │ │ - if (parseFloat(this.params.VERSION) >= 1.3) { │ │ │ │ │ - this.params.CRS = value │ │ │ │ │ - } else { │ │ │ │ │ - this.params.SRS = value │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.Elements" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Renderer.VML = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ + xmlns: "urn:schemas-microsoft-com:vml", │ │ │ │ │ + symbolCache: {}, │ │ │ │ │ + offset: null, │ │ │ │ │ + initialize: function(containerID) { │ │ │ │ │ + if (!this.supported()) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - if (typeof this.params.TRANSPARENT == "boolean") { │ │ │ │ │ - newParams.TRANSPARENT = this.params.TRANSPARENT ? "TRUE" : "FALSE" │ │ │ │ │ + if (!document.namespaces.olv) { │ │ │ │ │ + document.namespaces.add("olv", this.xmlns); │ │ │ │ │ + var style = document.createStyleSheet(); │ │ │ │ │ + var shapes = ["shape", "rect", "oval", "fill", "stroke", "imagedata", "group", "textbox"]; │ │ │ │ │ + for (var i = 0, len = shapes.length; i < len; i++) { │ │ │ │ │ + style.addRule("olv\\:" + shapes[i], "behavior: url(#default#VML); " + "position: absolute; display: inline-block;") │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.getFullRequestString.apply(this, arguments) │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.initialize.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.WMS" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.KaMapCache = OpenLayers.Class(OpenLayers.Layer.KaMap, { │ │ │ │ │ - IMAGE_EXTENSIONS: { │ │ │ │ │ - jpeg: "jpg", │ │ │ │ │ - gif: "gif", │ │ │ │ │ - png: "png", │ │ │ │ │ - png8: "png", │ │ │ │ │ - png24: "png", │ │ │ │ │ - dithered: "png" │ │ │ │ │ + supported: function() { │ │ │ │ │ + return !!document.namespaces │ │ │ │ │ }, │ │ │ │ │ - DEFAULT_FORMAT: "jpeg", │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.KaMap.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.extension = this.IMAGE_EXTENSIONS[this.params.i.toLowerCase() || this.DEFAULT_FORMAT] │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var left = extent.left / resolution | 0; │ │ │ │ │ + var top = extent.top / resolution - this.size.h | 0; │ │ │ │ │ + if (resolutionChanged || !this.offset) { │ │ │ │ │ + this.offset = { │ │ │ │ │ + x: left, │ │ │ │ │ + y: top │ │ │ │ │ + }; │ │ │ │ │ + left = 0; │ │ │ │ │ + top = 0 │ │ │ │ │ + } else { │ │ │ │ │ + left = left - this.offset.x; │ │ │ │ │ + top = top - this.offset.y │ │ │ │ │ + } │ │ │ │ │ + var org = left - this.xOffset + " " + top; │ │ │ │ │ + this.root.coordorigin = org; │ │ │ │ │ + var roots = [this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ + var root; │ │ │ │ │ + for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ + root = roots[i]; │ │ │ │ │ + var size = this.size.w + " " + this.size.h; │ │ │ │ │ + root.coordsize = size │ │ │ │ │ + } │ │ │ │ │ + this.root.style.flip = "y"; │ │ │ │ │ + return coordSysUnchanged │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var mapRes = this.map.getResolution(); │ │ │ │ │ - var scale = Math.round(this.map.getScale() * 1e4) / 1e4; │ │ │ │ │ - var pX = Math.round(bounds.left / mapRes); │ │ │ │ │ - var pY = -Math.round(bounds.top / mapRes); │ │ │ │ │ - var metaX = Math.floor(pX / this.tileSize.w / this.params.metaTileSize.w) * this.tileSize.w * this.params.metaTileSize.w; │ │ │ │ │ - var metaY = Math.floor(pY / this.tileSize.h / this.params.metaTileSize.h) * this.tileSize.h * this.params.metaTileSize.h; │ │ │ │ │ - var components = ["/", this.params.map, "/", scale, "/", this.params.g.replace(/\s/g, "_"), "/def/t", metaY, "/l", metaX, "/t", pY, "l", pX, ".", this.extension]; │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(components.join(""), url) │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ + var roots = [this.rendererRoot, this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ + var w = this.size.w + "px"; │ │ │ │ │ + var h = this.size.h + "px"; │ │ │ │ │ + var root; │ │ │ │ │ + for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ + root = roots[i]; │ │ │ │ │ + root.style.width = w; │ │ │ │ │ + root.style.height = h │ │ │ │ │ } │ │ │ │ │ - return url + components.join("") │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.KaMapCache" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.WMTS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - requestEncoding: "KVP", │ │ │ │ │ - url: null, │ │ │ │ │ - layer: null, │ │ │ │ │ - matrixSet: null, │ │ │ │ │ - style: null, │ │ │ │ │ - format: "image/jpeg", │ │ │ │ │ - tileOrigin: null, │ │ │ │ │ - tileFullExtent: null, │ │ │ │ │ - formatSuffix: null, │ │ │ │ │ - matrixIds: null, │ │ │ │ │ - dimensions: null, │ │ │ │ │ - params: null, │ │ │ │ │ - zoomOffset: 0, │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ - formatSuffixMap: { │ │ │ │ │ - "image/png": "png", │ │ │ │ │ - "image/png8": "png", │ │ │ │ │ - "image/png24": "png", │ │ │ │ │ - "image/png32": "png", │ │ │ │ │ - png: "png", │ │ │ │ │ - "image/jpeg": "jpg", │ │ │ │ │ - "image/jpg": "jpg", │ │ │ │ │ - jpeg: "jpg", │ │ │ │ │ - jpg: "jpg" │ │ │ │ │ + getNodeType: function(geometry, style) { │ │ │ │ │ + var nodeType = null; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + nodeType = "olv:rect" │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + nodeType = "olv:shape" │ │ │ │ │ + } else { │ │ │ │ │ + nodeType = "olv:oval" │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + nodeType = "olv:rect"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + case "OpenLayers.Geometry.Curve": │ │ │ │ │ + nodeType = "olv:shape"; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + return nodeType │ │ │ │ │ }, │ │ │ │ │ - matrix: null, │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ - var required = { │ │ │ │ │ - url: true, │ │ │ │ │ - layer: true, │ │ │ │ │ - style: true, │ │ │ │ │ - matrixSet: true │ │ │ │ │ - }; │ │ │ │ │ - for (var prop in required) { │ │ │ │ │ - if (!(prop in config)) { │ │ │ │ │ - throw new Error("Missing property '" + prop + "' in layer configuration.") │ │ │ │ │ + setStyle: function(node, style, options, geometry) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + options = options || node._options; │ │ │ │ │ + var fillColor = style.fillColor; │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + node.title = title │ │ │ │ │ + } │ │ │ │ │ + if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + options.isFilled = true; │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ + var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ + node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x + xOffset | 0) + "px"; │ │ │ │ │ + node.style.top = (geometry.y / resolution - this.offset.y - (yOffset + height) | 0) + "px"; │ │ │ │ │ + node.style.width = width + "px"; │ │ │ │ │ + node.style.height = height + "px"; │ │ │ │ │ + node.style.flip = "y"; │ │ │ │ │ + fillColor = "none"; │ │ │ │ │ + options.isStroked = false │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + var cache = this.importSymbol(style.graphicName); │ │ │ │ │ + node.path = cache.path; │ │ │ │ │ + node.coordorigin = cache.left + "," + cache.bottom; │ │ │ │ │ + var size = cache.size; │ │ │ │ │ + node.coordsize = size + "," + size; │ │ │ │ │ + this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ + node.style.flip = "y" │ │ │ │ │ + } else { │ │ │ │ │ + this.drawCircle(node, geometry, style.pointRadius) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - config.params = OpenLayers.Util.upperCaseObject(config.params); │ │ │ │ │ - var args = [config.name, config.url, config.params, config]; │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, args); │ │ │ │ │ - if (!this.formatSuffix) { │ │ │ │ │ - this.formatSuffix = this.formatSuffixMap[this.format] || this.format.split("/").pop() │ │ │ │ │ + if (options.isFilled) { │ │ │ │ │ + node.fillcolor = fillColor │ │ │ │ │ + } else { │ │ │ │ │ + node.filled = "false" │ │ │ │ │ } │ │ │ │ │ - if (this.matrixIds) { │ │ │ │ │ - var len = this.matrixIds.length; │ │ │ │ │ - if (len && typeof this.matrixIds[0] === "string") { │ │ │ │ │ - var ids = this.matrixIds; │ │ │ │ │ - this.matrixIds = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - this.matrixIds[i] = { │ │ │ │ │ - identifier: ids[i] │ │ │ │ │ - } │ │ │ │ │ + var fills = node.getElementsByTagName("fill"); │ │ │ │ │ + var fill = fills.length == 0 ? null : fills[0]; │ │ │ │ │ + if (!options.isFilled) { │ │ │ │ │ + if (fill) { │ │ │ │ │ + node.removeChild(fill) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + if (!fill) { │ │ │ │ │ + fill = this.createNode("olv:fill", node.id + "_fill") │ │ │ │ │ + } │ │ │ │ │ + fill.opacity = style.fillOpacity; │ │ │ │ │ + if (node._geometryClass === "OpenLayers.Geometry.Point" && style.externalGraphic) { │ │ │ │ │ + if (style.graphicOpacity) { │ │ │ │ │ + fill.opacity = style.graphicOpacity │ │ │ │ │ + } │ │ │ │ │ + fill.src = style.externalGraphic; │ │ │ │ │ + fill.type = "frame"; │ │ │ │ │ + if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ + fill.aspect = "atmost" │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + if (fill.parentNode != node) { │ │ │ │ │ + node.appendChild(fill) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - setMap: function() { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - updateMatrixProperties: function() { │ │ │ │ │ - this.matrix = this.getMatrix(); │ │ │ │ │ - if (this.matrix) { │ │ │ │ │ - if (this.matrix.topLeftCorner) { │ │ │ │ │ - this.tileOrigin = this.matrix.topLeftCorner │ │ │ │ │ + var rotation = style.rotation; │ │ │ │ │ + if (rotation !== undefined || node._rotation !== undefined) { │ │ │ │ │ + node._rotation = rotation; │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ + fill.opacity = 0 │ │ │ │ │ + } else if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ + node.style.rotation = rotation || 0 │ │ │ │ │ } │ │ │ │ │ - if (this.matrix.tileWidth && this.matrix.tileHeight) { │ │ │ │ │ - this.tileSize = new OpenLayers.Size(this.matrix.tileWidth, this.matrix.tileHeight) │ │ │ │ │ + } │ │ │ │ │ + var strokes = node.getElementsByTagName("stroke"); │ │ │ │ │ + var stroke = strokes.length == 0 ? null : strokes[0]; │ │ │ │ │ + if (!options.isStroked) { │ │ │ │ │ + node.stroked = false; │ │ │ │ │ + if (stroke) { │ │ │ │ │ + stroke.on = false │ │ │ │ │ } │ │ │ │ │ - if (!this.tileOrigin) { │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, this.maxExtent.top) │ │ │ │ │ + } else { │ │ │ │ │ + if (!stroke) { │ │ │ │ │ + stroke = this.createNode("olv:stroke", node.id + "_stroke"); │ │ │ │ │ + node.appendChild(stroke) │ │ │ │ │ } │ │ │ │ │ - if (!this.tileFullExtent) { │ │ │ │ │ - this.tileFullExtent = this.maxExtent │ │ │ │ │ + stroke.on = true; │ │ │ │ │ + stroke.color = style.strokeColor; │ │ │ │ │ + stroke.weight = style.strokeWidth + "px"; │ │ │ │ │ + stroke.opacity = style.strokeOpacity; │ │ │ │ │ + stroke.endcap = style.strokeLinecap == "butt" ? "flat" : style.strokeLinecap || "round"; │ │ │ │ │ + if (style.strokeDashstyle) { │ │ │ │ │ + stroke.dashstyle = this.dashStyle(style) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ + node.style.cursor = style.cursor │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - if (zoomChanged || !this.matrix) { │ │ │ │ │ - this.updateMatrixProperties() │ │ │ │ │ + graphicRotate: function(node, xOffset, yOffset, style) { │ │ │ │ │ + var style = style || node._style; │ │ │ │ │ + var rotation = style.rotation || 0; │ │ │ │ │ + var aspectRatio, size; │ │ │ │ │ + if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ + var img = new Image; │ │ │ │ │ + img.onreadystatechange = OpenLayers.Function.bind(function() { │ │ │ │ │ + if (img.readyState == "complete" || img.readyState == "interactive") { │ │ │ │ │ + aspectRatio = img.width / img.height; │ │ │ │ │ + size = Math.max(style.pointRadius * 2, style.graphicWidth || 0, style.graphicHeight || 0); │ │ │ │ │ + xOffset = xOffset * aspectRatio; │ │ │ │ │ + style.graphicWidth = size * aspectRatio; │ │ │ │ │ + style.graphicHeight = size; │ │ │ │ │ + this.graphicRotate(node, xOffset, yOffset, style) │ │ │ │ │ + } │ │ │ │ │ + }, this); │ │ │ │ │ + img.src = style.externalGraphic; │ │ │ │ │ + return │ │ │ │ │ + } else { │ │ │ │ │ + size = Math.max(style.graphicWidth, style.graphicHeight); │ │ │ │ │ + aspectRatio = style.graphicWidth / style.graphicHeight │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.moveTo.apply(this, arguments) │ │ │ │ │ + var width = Math.round(style.graphicWidth || size * aspectRatio); │ │ │ │ │ + var height = Math.round(style.graphicHeight || size); │ │ │ │ │ + node.style.width = width + "px"; │ │ │ │ │ + node.style.height = height + "px"; │ │ │ │ │ + var image = document.getElementById(node.id + "_image"); │ │ │ │ │ + if (!image) { │ │ │ │ │ + image = this.createNode("olv:imagedata", node.id + "_image"); │ │ │ │ │ + node.appendChild(image) │ │ │ │ │ + } │ │ │ │ │ + image.style.width = width + "px"; │ │ │ │ │ + image.style.height = height + "px"; │ │ │ │ │ + image.src = style.externalGraphic; │ │ │ │ │ + image.style.filter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(" + "src='', sizingMethod='scale')"; │ │ │ │ │ + var rot = rotation * Math.PI / 180; │ │ │ │ │ + var sintheta = Math.sin(rot); │ │ │ │ │ + var costheta = Math.cos(rot); │ │ │ │ │ + var filter = "progid:DXImageTransform.Microsoft.Matrix(M11=" + costheta + ",M12=" + -sintheta + ",M21=" + sintheta + ",M22=" + costheta + ",SizingMethod='auto expand')\n"; │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ + if (opacity && opacity != 1) { │ │ │ │ │ + filter += "progid:DXImageTransform.Microsoft.BasicImage(opacity=" + opacity + ")\n" │ │ │ │ │ + } │ │ │ │ │ + node.style.filter = filter; │ │ │ │ │ + var centerPoint = new OpenLayers.Geometry.Point(-xOffset, -yOffset); │ │ │ │ │ + var imgBox = new OpenLayers.Bounds(0, 0, width, height).toGeometry(); │ │ │ │ │ + imgBox.rotate(style.rotation, centerPoint); │ │ │ │ │ + var imgBounds = imgBox.getBounds(); │ │ │ │ │ + node.style.left = Math.round(parseInt(node.style.left) + imgBounds.left) + "px"; │ │ │ │ │ + node.style.top = Math.round(parseInt(node.style.top) - imgBounds.bottom) + "px" │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.WMTS(this.options) │ │ │ │ │ + postDraw: function(node) { │ │ │ │ │ + node.style.visibility = "visible"; │ │ │ │ │ + var fillColor = node._style.fillColor; │ │ │ │ │ + var strokeColor = node._style.strokeColor; │ │ │ │ │ + if (fillColor == "none" && node.fillcolor != fillColor) { │ │ │ │ │ + node.fillcolor = fillColor │ │ │ │ │ + } │ │ │ │ │ + if (strokeColor == "none" && node.strokecolor != strokeColor) { │ │ │ │ │ + node.strokecolor = strokeColor │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ }, │ │ │ │ │ - getIdentifier: function() { │ │ │ │ │ - return this.getServerZoom() │ │ │ │ │ + setNodeDimension: function(node, geometry) { │ │ │ │ │ + var bbox = geometry.getBounds(); │ │ │ │ │ + if (bbox) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var scaledBox = new OpenLayers.Bounds((bbox.left - this.featureDx) / resolution - this.offset.x | 0, bbox.bottom / resolution - this.offset.y | 0, (bbox.right - this.featureDx) / resolution - this.offset.x | 0, bbox.top / resolution - this.offset.y | 0); │ │ │ │ │ + node.style.left = scaledBox.left + "px"; │ │ │ │ │ + node.style.top = scaledBox.top + "px"; │ │ │ │ │ + node.style.width = scaledBox.getWidth() + "px"; │ │ │ │ │ + node.style.height = scaledBox.getHeight() + "px"; │ │ │ │ │ + node.coordorigin = scaledBox.left + " " + scaledBox.top; │ │ │ │ │ + node.coordsize = scaledBox.getWidth() + " " + scaledBox.getHeight() │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - getMatrix: function() { │ │ │ │ │ - var matrix; │ │ │ │ │ - if (!this.matrixIds || this.matrixIds.length === 0) { │ │ │ │ │ - matrix = { │ │ │ │ │ - identifier: this.getIdentifier() │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - if ("scaleDenominator" in this.matrixIds[0]) { │ │ │ │ │ - var denom = OpenLayers.METERS_PER_INCH * OpenLayers.INCHES_PER_UNIT[this.units] * this.getServerResolution() / 28e-5; │ │ │ │ │ - var diff = Number.POSITIVE_INFINITY; │ │ │ │ │ - var delta; │ │ │ │ │ - for (var i = 0, ii = this.matrixIds.length; i < ii; ++i) { │ │ │ │ │ - delta = Math.abs(1 - this.matrixIds[i].scaleDenominator / denom); │ │ │ │ │ - if (delta < diff) { │ │ │ │ │ - diff = delta; │ │ │ │ │ - matrix = this.matrixIds[i] │ │ │ │ │ + dashStyle: function(style) { │ │ │ │ │ + var dash = style.strokeDashstyle; │ │ │ │ │ + switch (dash) { │ │ │ │ │ + case "solid": │ │ │ │ │ + case "dot": │ │ │ │ │ + case "dash": │ │ │ │ │ + case "dashdot": │ │ │ │ │ + case "longdash": │ │ │ │ │ + case "longdashdot": │ │ │ │ │ + return dash; │ │ │ │ │ + default: │ │ │ │ │ + var parts = dash.split(/[ ,]/); │ │ │ │ │ + if (parts.length == 2) { │ │ │ │ │ + if (1 * parts[0] >= 2 * parts[1]) { │ │ │ │ │ + return "longdash" │ │ │ │ │ } │ │ │ │ │ + return parts[0] == 1 || parts[1] == 1 ? "dot" : "dash" │ │ │ │ │ + } else if (parts.length == 4) { │ │ │ │ │ + return 1 * parts[0] >= 2 * parts[1] ? "longdashdot" : "dashdot" │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - matrix = this.matrixIds[this.getIdentifier()] │ │ │ │ │ - } │ │ │ │ │ + return "solid" │ │ │ │ │ } │ │ │ │ │ - return matrix │ │ │ │ │ }, │ │ │ │ │ - getTileInfo: function(loc) { │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var fx = (loc.lon - this.tileOrigin.lon) / (res * this.tileSize.w); │ │ │ │ │ - var fy = (this.tileOrigin.lat - loc.lat) / (res * this.tileSize.h); │ │ │ │ │ - var col = Math.floor(fx); │ │ │ │ │ - var row = Math.floor(fy); │ │ │ │ │ - return { │ │ │ │ │ - col: col, │ │ │ │ │ - row: row, │ │ │ │ │ - i: Math.floor((fx - col) * this.tileSize.w), │ │ │ │ │ - j: Math.floor((fy - row) * this.tileSize.h) │ │ │ │ │ + createNode: function(type, id) { │ │ │ │ │ + var node = document.createElement(type); │ │ │ │ │ + if (id) { │ │ │ │ │ + node.id = id │ │ │ │ │ } │ │ │ │ │ + node.unselectable = "on"; │ │ │ │ │ + node.onselectstart = OpenLayers.Function.False; │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var url = ""; │ │ │ │ │ - if (!this.tileFullExtent || this.tileFullExtent.intersectsBounds(bounds)) { │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var info = this.getTileInfo(center); │ │ │ │ │ - var matrixId = this.matrix.identifier; │ │ │ │ │ - var dimensions = this.dimensions, │ │ │ │ │ - params; │ │ │ │ │ - if (OpenLayers.Util.isArray(this.url)) { │ │ │ │ │ - url = this.selectUrl([this.version, this.style, this.matrixSet, this.matrix.identifier, info.row, info.col].join(","), this.url) │ │ │ │ │ - } else { │ │ │ │ │ - url = this.url │ │ │ │ │ - } │ │ │ │ │ - if (this.requestEncoding.toUpperCase() === "REST") { │ │ │ │ │ - params = this.params; │ │ │ │ │ - if (url.indexOf("{") !== -1) { │ │ │ │ │ - var template = url.replace(/\{/g, "${"); │ │ │ │ │ - var context = { │ │ │ │ │ - style: this.style, │ │ │ │ │ - Style: this.style, │ │ │ │ │ - TileMatrixSet: this.matrixSet, │ │ │ │ │ - TileMatrix: this.matrix.identifier, │ │ │ │ │ - TileRow: info.row, │ │ │ │ │ - TileCol: info.col │ │ │ │ │ - }; │ │ │ │ │ - if (dimensions) { │ │ │ │ │ - var dimension, i; │ │ │ │ │ - for (i = dimensions.length - 1; i >= 0; --i) { │ │ │ │ │ - dimension = dimensions[i]; │ │ │ │ │ - context[dimension] = params[dimension.toUpperCase()] │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - url = OpenLayers.String.format(template, context) │ │ │ │ │ - } else { │ │ │ │ │ - var path = this.version + "/" + this.layer + "/" + this.style + "/"; │ │ │ │ │ - if (dimensions) { │ │ │ │ │ - for (var i = 0; i < dimensions.length; i++) { │ │ │ │ │ - if (params[dimensions[i]]) { │ │ │ │ │ - path = path + params[dimensions[i]] + "/" │ │ │ │ │ - } │ │ │ │ │ + nodeTypeCompare: function(node, type) { │ │ │ │ │ + var subType = type; │ │ │ │ │ + var splitIndex = subType.indexOf(":"); │ │ │ │ │ + if (splitIndex != -1) { │ │ │ │ │ + subType = subType.substr(splitIndex + 1) │ │ │ │ │ + } │ │ │ │ │ + var nodeName = node.nodeName; │ │ │ │ │ + splitIndex = nodeName.indexOf(":"); │ │ │ │ │ + if (splitIndex != -1) { │ │ │ │ │ + nodeName = nodeName.substr(splitIndex + 1) │ │ │ │ │ + } │ │ │ │ │ + return subType == nodeName │ │ │ │ │ + }, │ │ │ │ │ + createRenderRoot: function() { │ │ │ │ │ + return this.nodeFactory(this.container.id + "_vmlRoot", "div") │ │ │ │ │ + }, │ │ │ │ │ + createRoot: function(suffix) { │ │ │ │ │ + return this.nodeFactory(this.container.id + suffix, "olv:group") │ │ │ │ │ + }, │ │ │ │ │ + drawPoint: function(node, geometry) { │ │ │ │ │ + return this.drawCircle(node, geometry, 1) │ │ │ │ │ + }, │ │ │ │ │ + drawCircle: function(node, geometry, radius) { │ │ │ │ │ + if (!isNaN(geometry.x) && !isNaN(geometry.y)) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x | 0) - radius + "px"; │ │ │ │ │ + node.style.top = (geometry.y / resolution - this.offset.y | 0) - radius + "px"; │ │ │ │ │ + var diameter = radius * 2; │ │ │ │ │ + node.style.width = diameter + "px"; │ │ │ │ │ + node.style.height = diameter + "px"; │ │ │ │ │ + return node │ │ │ │ │ + } │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + drawLineString: function(node, geometry) { │ │ │ │ │ + return this.drawLine(node, geometry, false) │ │ │ │ │ + }, │ │ │ │ │ + drawLinearRing: function(node, geometry) { │ │ │ │ │ + return this.drawLine(node, geometry, true) │ │ │ │ │ + }, │ │ │ │ │ + drawLine: function(node, geometry, closeLine) { │ │ │ │ │ + this.setNodeDimension(node, geometry); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var numComponents = geometry.components.length; │ │ │ │ │ + var parts = new Array(numComponents); │ │ │ │ │ + var comp, x, y; │ │ │ │ │ + for (var i = 0; i < numComponents; i++) { │ │ │ │ │ + comp = geometry.components[i]; │ │ │ │ │ + x = (comp.x - this.featureDx) / resolution - this.offset.x | 0; │ │ │ │ │ + y = comp.y / resolution - this.offset.y | 0; │ │ │ │ │ + parts[i] = " " + x + "," + y + " l " │ │ │ │ │ + } │ │ │ │ │ + var end = closeLine ? " x e" : " e"; │ │ │ │ │ + node.path = "m" + parts.join("") + end; │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + drawPolygon: function(node, geometry) { │ │ │ │ │ + this.setNodeDimension(node, geometry); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var path = []; │ │ │ │ │ + var j, jj, points, area, first, second, i, ii, comp, pathComp, x, y; │ │ │ │ │ + for (j = 0, jj = geometry.components.length; j < jj; j++) { │ │ │ │ │ + path.push("m"); │ │ │ │ │ + points = geometry.components[j].components; │ │ │ │ │ + area = j === 0; │ │ │ │ │ + first = null; │ │ │ │ │ + second = null; │ │ │ │ │ + for (i = 0, ii = points.length; i < ii; i++) { │ │ │ │ │ + comp = points[i]; │ │ │ │ │ + x = (comp.x - this.featureDx) / resolution - this.offset.x | 0; │ │ │ │ │ + y = comp.y / resolution - this.offset.y | 0; │ │ │ │ │ + pathComp = " " + x + "," + y; │ │ │ │ │ + path.push(pathComp); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + path.push(" l") │ │ │ │ │ + } │ │ │ │ │ + if (!area) { │ │ │ │ │ + if (!first) { │ │ │ │ │ + first = pathComp │ │ │ │ │ + } else if (first != pathComp) { │ │ │ │ │ + if (!second) { │ │ │ │ │ + second = pathComp │ │ │ │ │ + } else if (second != pathComp) { │ │ │ │ │ + area = true │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - path = path + this.matrixSet + "/" + this.matrix.identifier + "/" + info.row + "/" + info.col + "." + this.formatSuffix; │ │ │ │ │ - if (!url.match(/\/$/)) { │ │ │ │ │ - url = url + "/" │ │ │ │ │ - } │ │ │ │ │ - url = url + path │ │ │ │ │ } │ │ │ │ │ - } else if (this.requestEncoding.toUpperCase() === "KVP") { │ │ │ │ │ - params = { │ │ │ │ │ - SERVICE: "WMTS", │ │ │ │ │ - REQUEST: "GetTile", │ │ │ │ │ - VERSION: this.version, │ │ │ │ │ - LAYER: this.layer, │ │ │ │ │ - STYLE: this.style, │ │ │ │ │ - TILEMATRIXSET: this.matrixSet, │ │ │ │ │ - TILEMATRIX: this.matrix.identifier, │ │ │ │ │ - TILEROW: info.row, │ │ │ │ │ - TILECOL: info.col, │ │ │ │ │ - FORMAT: this.format │ │ │ │ │ - }; │ │ │ │ │ - url = OpenLayers.Layer.Grid.prototype.getFullRequestString.apply(this, [params]) │ │ │ │ │ } │ │ │ │ │ + path.push(area ? " x " : " ") │ │ │ │ │ } │ │ │ │ │ - return url │ │ │ │ │ - }, │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - if (this.requestEncoding.toUpperCase() === "KVP") { │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, [OpenLayers.Util.upperCaseObject(newParams)]) │ │ │ │ │ - } │ │ │ │ │ + path.push("e"); │ │ │ │ │ + node.path = path.join(""); │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.WMTS" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.MapServer = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - mode: "map", │ │ │ │ │ - map_imagetype: "png" │ │ │ │ │ + drawRectangle: function(node, geometry) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x | 0) + "px"; │ │ │ │ │ + node.style.top = (geometry.y / resolution - this.offset.y | 0) + "px"; │ │ │ │ │ + node.style.width = (geometry.width / resolution | 0) + "px"; │ │ │ │ │ + node.style.height = (geometry.height / resolution | 0) + "px"; │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.params = OpenLayers.Util.applyDefaults(this.params, this.DEFAULT_PARAMS); │ │ │ │ │ - if (options == null || options.isBaseLayer == null) { │ │ │ │ │ - this.isBaseLayer = this.params.transparent != "true" && this.params.transparent != true │ │ │ │ │ + drawText: function(featureId, style, location) { │ │ │ │ │ + var label = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX, "olv:rect"); │ │ │ │ │ + var textbox = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX + "_textbox", "olv:textbox"); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + label.style.left = ((location.x - this.featureDx) / resolution - this.offset.x | 0) + "px"; │ │ │ │ │ + label.style.top = (location.y / resolution - this.offset.y | 0) + "px"; │ │ │ │ │ + label.style.flip = "y"; │ │ │ │ │ + textbox.innerText = style.label; │ │ │ │ │ + if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ + textbox.style.cursor = style.cursor │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.MapServer(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ + if (style.fontColor) { │ │ │ │ │ + textbox.style.color = style.fontColor │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ + if (style.fontOpacity) { │ │ │ │ │ + textbox.style.filter = "alpha(opacity=" + style.fontOpacity * 100 + ")" │ │ │ │ │ + } │ │ │ │ │ + if (style.fontFamily) { │ │ │ │ │ + textbox.style.fontFamily = style.fontFamily │ │ │ │ │ + } │ │ │ │ │ + if (style.fontSize) { │ │ │ │ │ + textbox.style.fontSize = style.fontSize │ │ │ │ │ + } │ │ │ │ │ + if (style.fontWeight) { │ │ │ │ │ + textbox.style.fontWeight = style.fontWeight │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStyle) { │ │ │ │ │ + textbox.style.fontStyle = style.fontStyle │ │ │ │ │ + } │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + label._featureId = featureId; │ │ │ │ │ + textbox._featureId = featureId; │ │ │ │ │ + textbox._geometry = location; │ │ │ │ │ + textbox._geometryClass = location.CLASS_NAME │ │ │ │ │ + } │ │ │ │ │ + textbox.style.whiteSpace = "nowrap"; │ │ │ │ │ + textbox.inset = "1px,0px,0px,0px"; │ │ │ │ │ + if (!label.parentNode) { │ │ │ │ │ + label.appendChild(textbox); │ │ │ │ │ + this.textRoot.appendChild(label) │ │ │ │ │ + } │ │ │ │ │ + var align = style.labelAlign || "cm"; │ │ │ │ │ + if (align.length == 1) { │ │ │ │ │ + align += "m" │ │ │ │ │ + } │ │ │ │ │ + var xshift = textbox.clientWidth * OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(0, 1)]; │ │ │ │ │ + var yshift = textbox.clientHeight * OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(1, 1)]; │ │ │ │ │ + label.style.left = parseInt(label.style.left) - xshift - 1 + "px"; │ │ │ │ │ + label.style.top = parseInt(label.style.top) + yshift + "px" │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var extent = [bounds.left, bounds.bottom, bounds.right, bounds.top]; │ │ │ │ │ - var imageSize = this.getImageSize(); │ │ │ │ │ - var url = this.getFullRequestString({ │ │ │ │ │ - mapext: extent, │ │ │ │ │ - imgext: extent, │ │ │ │ │ - map_size: [imageSize.w, imageSize.h], │ │ │ │ │ - imgx: imageSize.w / 2, │ │ │ │ │ - imgy: imageSize.h / 2, │ │ │ │ │ - imgxy: [imageSize.w, imageSize.h] │ │ │ │ │ - }); │ │ │ │ │ - return url │ │ │ │ │ + moveRoot: function(renderer) { │ │ │ │ │ + var layer = this.map.getLayer(renderer.container.id); │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Vector.RootContainer) { │ │ │ │ │ + layer = this.map.getLayer(this.container.id) │ │ │ │ │ + } │ │ │ │ │ + layer && layer.renderer.clear(); │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.moveRoot.apply(this, arguments); │ │ │ │ │ + layer && layer.redraw() │ │ │ │ │ }, │ │ │ │ │ - getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ - var url = altUrl == null ? this.url : altUrl; │ │ │ │ │ - var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ - allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ - var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(paramsString, url) │ │ │ │ │ + importSymbol: function(graphicName) { │ │ │ │ │ + var id = this.container.id + "-" + graphicName; │ │ │ │ │ + var cache = this.symbolCache[id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + return cache │ │ │ │ │ } │ │ │ │ │ - var urlParams = OpenLayers.Util.upperCaseObject(OpenLayers.Util.getParameters(url)); │ │ │ │ │ - for (var key in allParams) { │ │ │ │ │ - if (key.toUpperCase() in urlParams) { │ │ │ │ │ - delete allParams[key] │ │ │ │ │ - } │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(graphicName + " is not a valid symbol name") │ │ │ │ │ } │ │ │ │ │ - paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ - var requestString = url; │ │ │ │ │ - paramsString = paramsString.replace(/,/g, "+"); │ │ │ │ │ - if (paramsString != "") { │ │ │ │ │ - var lastServerChar = url.charAt(url.length - 1); │ │ │ │ │ - if (lastServerChar == "&" || lastServerChar == "?") { │ │ │ │ │ - requestString += paramsString │ │ │ │ │ - } else { │ │ │ │ │ - if (url.indexOf("?") == -1) { │ │ │ │ │ - requestString += "?" + paramsString │ │ │ │ │ - } else { │ │ │ │ │ - requestString += "&" + paramsString │ │ │ │ │ - } │ │ │ │ │ + var symbolExtent = new OpenLayers.Bounds(Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ + var pathitems = ["m"]; │ │ │ │ │ + for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + var x = symbol[i]; │ │ │ │ │ + var y = symbol[i + 1]; │ │ │ │ │ + symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ + symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ + symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ + symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ + pathitems.push(x); │ │ │ │ │ + pathitems.push(y); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + pathitems.push("l") │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return requestString │ │ │ │ │ + pathitems.push("x e"); │ │ │ │ │ + var path = pathitems.join(" "); │ │ │ │ │ + var diff = (symbolExtent.getWidth() - symbolExtent.getHeight()) / 2; │ │ │ │ │ + if (diff > 0) { │ │ │ │ │ + symbolExtent.bottom = symbolExtent.bottom - diff; │ │ │ │ │ + symbolExtent.top = symbolExtent.top + diff │ │ │ │ │ + } else { │ │ │ │ │ + symbolExtent.left = symbolExtent.left + diff; │ │ │ │ │ + symbolExtent.right = symbolExtent.right - diff │ │ │ │ │ + } │ │ │ │ │ + cache = { │ │ │ │ │ + path: path, │ │ │ │ │ + size: symbolExtent.getWidth(), │ │ │ │ │ + left: symbolExtent.left, │ │ │ │ │ + bottom: symbolExtent.bottom │ │ │ │ │ + }; │ │ │ │ │ + this.symbolCache[id] = cache; │ │ │ │ │ + return cache │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.MapServer" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.VML" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.ArcGIS93Rest = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - format: "png" │ │ │ │ │ +OpenLayers.Renderer.VML.LABEL_SHIFT = { │ │ │ │ │ + l: 0, │ │ │ │ │ + c: .5, │ │ │ │ │ + r: 1, │ │ │ │ │ + t: 0, │ │ │ │ │ + m: .5, │ │ │ │ │ + b: 1 │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.SVG = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ + xmlns: "http://www.w3.org/2000/svg", │ │ │ │ │ + xlinkns: "http://www.w3.org/1999/xlink", │ │ │ │ │ + MAX_PIXEL: 15e3, │ │ │ │ │ + translationParameters: null, │ │ │ │ │ + symbolMetrics: null, │ │ │ │ │ + initialize: function(containerID) { │ │ │ │ │ + if (!this.supported()) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.translationParameters = { │ │ │ │ │ + x: 0, │ │ │ │ │ + y: 0 │ │ │ │ │ + }; │ │ │ │ │ + this.symbolMetrics = {} │ │ │ │ │ }, │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ - newArguments.push(name, url, params, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - OpenLayers.Util.applyDefaults(this.params, OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS)); │ │ │ │ │ - if (this.params.TRANSPARENT && this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ - if (options == null || !options.isBaseLayer) { │ │ │ │ │ - this.isBaseLayer = false │ │ │ │ │ + supported: function() { │ │ │ │ │ + var svgFeature = "http://www.w3.org/TR/SVG11/feature#"; │ │ │ │ │ + return document.implementation && (document.implementation.hasFeature("org.w3c.svg", "1.0") || document.implementation.hasFeature(svgFeature + "SVG", "1.1") || document.implementation.hasFeature(svgFeature + "BasicStructure", "1.1")) │ │ │ │ │ + }, │ │ │ │ │ + inValidRange: function(x, y, xyOnly) { │ │ │ │ │ + var left = x + (xyOnly ? 0 : this.translationParameters.x); │ │ │ │ │ + var top = y + (xyOnly ? 0 : this.translationParameters.y); │ │ │ │ │ + return left >= -this.MAX_PIXEL && left <= this.MAX_PIXEL && top >= -this.MAX_PIXEL && top <= this.MAX_PIXEL │ │ │ │ │ + }, │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + var resolution = this.getResolution(), │ │ │ │ │ + left = -extent.left / resolution, │ │ │ │ │ + top = extent.top / resolution; │ │ │ │ │ + if (resolutionChanged) { │ │ │ │ │ + this.left = left; │ │ │ │ │ + this.top = top; │ │ │ │ │ + var extentString = "0 0 " + this.size.w + " " + this.size.h; │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "viewBox", extentString); │ │ │ │ │ + this.translate(this.xOffset, 0); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + var inRange = this.translate(left - this.left + this.xOffset, top - this.top); │ │ │ │ │ + if (!inRange) { │ │ │ │ │ + this.setExtent(extent, true) │ │ │ │ │ } │ │ │ │ │ - if (this.params.FORMAT == "jpg") { │ │ │ │ │ - this.params.FORMAT = OpenLayers.Util.alphaHack() ? "gif" : "png" │ │ │ │ │ + return coordSysUnchanged && inRange │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + translate: function(x, y) { │ │ │ │ │ + if (!this.inValidRange(x, y, true)) { │ │ │ │ │ + return false │ │ │ │ │ + } else { │ │ │ │ │ + var transformString = ""; │ │ │ │ │ + if (x || y) { │ │ │ │ │ + transformString = "translate(" + x + "," + y + ")" │ │ │ │ │ } │ │ │ │ │ + this.root.setAttributeNS(null, "transform", transformString); │ │ │ │ │ + this.translationParameters = { │ │ │ │ │ + x: x, │ │ │ │ │ + y: y │ │ │ │ │ + }; │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.ArcGIS93Rest(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "width", this.size.w); │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "height", this.size.h) │ │ │ │ │ + }, │ │ │ │ │ + getNodeType: function(geometry, style) { │ │ │ │ │ + var nodeType = null; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + nodeType = "image" │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + nodeType = "svg" │ │ │ │ │ + } else { │ │ │ │ │ + nodeType = "circle" │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + nodeType = "rect"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + nodeType = "polyline"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + nodeType = "polygon"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + case "OpenLayers.Geometry.Curve": │ │ │ │ │ + nodeType = "path"; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ + return nodeType │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var projWords = this.projection.getCode().split(":"); │ │ │ │ │ - var srid = projWords[projWords.length - 1]; │ │ │ │ │ - var imageSize = this.getImageSize(); │ │ │ │ │ - var newParams = { │ │ │ │ │ - BBOX: bounds.toBBOX(), │ │ │ │ │ - SIZE: imageSize.w + "," + imageSize.h, │ │ │ │ │ - F: "image", │ │ │ │ │ - BBOXSR: srid, │ │ │ │ │ - IMAGESR: srid │ │ │ │ │ - }; │ │ │ │ │ - if (this.layerDefs) { │ │ │ │ │ - var layerDefStrList = []; │ │ │ │ │ - var layerID; │ │ │ │ │ - for (layerID in this.layerDefs) { │ │ │ │ │ - if (this.layerDefs.hasOwnProperty(layerID)) { │ │ │ │ │ - if (this.layerDefs[layerID]) { │ │ │ │ │ - layerDefStrList.push(layerID); │ │ │ │ │ - layerDefStrList.push(":"); │ │ │ │ │ - layerDefStrList.push(this.layerDefs[layerID]); │ │ │ │ │ - layerDefStrList.push(";") │ │ │ │ │ - } │ │ │ │ │ + setStyle: function(node, style, options) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + options = options || node._options; │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + node.setAttributeNS(null, "title", title); │ │ │ │ │ + var titleNode = node.getElementsByTagName("title"); │ │ │ │ │ + if (titleNode.length > 0) { │ │ │ │ │ + titleNode[0].firstChild.textContent = title │ │ │ │ │ + } else { │ │ │ │ │ + var label = this.nodeFactory(null, "title"); │ │ │ │ │ + label.textContent = title; │ │ │ │ │ + node.appendChild(label) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var r = parseFloat(node.getAttributeNS(null, "r")); │ │ │ │ │ + var widthFactor = 1; │ │ │ │ │ + var pos; │ │ │ │ │ + if (node._geometryClass == "OpenLayers.Geometry.Point" && r) { │ │ │ │ │ + node.style.visibility = ""; │ │ │ │ │ + if (style.graphic === false) { │ │ │ │ │ + node.style.visibility = "hidden" │ │ │ │ │ + } else if (style.externalGraphic) { │ │ │ │ │ + pos = this.getPosition(node); │ │ │ │ │ + if (style.graphicWidth && style.graphicHeight) { │ │ │ │ │ + node.setAttributeNS(null, "preserveAspectRatio", "none") │ │ │ │ │ + } │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ + var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ + var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ + node.setAttributeNS(null, "x", (pos.x + xOffset).toFixed()); │ │ │ │ │ + node.setAttributeNS(null, "y", (pos.y + yOffset).toFixed()); │ │ │ │ │ + node.setAttributeNS(null, "width", width); │ │ │ │ │ + node.setAttributeNS(null, "height", height); │ │ │ │ │ + node.setAttributeNS(this.xlinkns, "xlink:href", style.externalGraphic); │ │ │ │ │ + node.setAttributeNS(null, "style", "opacity: " + opacity); │ │ │ │ │ + node.onclick = OpenLayers.Event.preventDefault │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + var offset = style.pointRadius * 3; │ │ │ │ │ + var size = offset * 2; │ │ │ │ │ + var src = this.importSymbol(style.graphicName); │ │ │ │ │ + pos = this.getPosition(node); │ │ │ │ │ + widthFactor = this.symbolMetrics[src.id][0] * 3 / size; │ │ │ │ │ + var parent = node.parentNode; │ │ │ │ │ + var nextSibling = node.nextSibling; │ │ │ │ │ + if (parent) { │ │ │ │ │ + parent.removeChild(node) │ │ │ │ │ + } │ │ │ │ │ + node.firstChild && node.removeChild(node.firstChild); │ │ │ │ │ + node.appendChild(src.firstChild.cloneNode(true)); │ │ │ │ │ + node.setAttributeNS(null, "viewBox", src.getAttributeNS(null, "viewBox")); │ │ │ │ │ + node.setAttributeNS(null, "width", size); │ │ │ │ │ + node.setAttributeNS(null, "height", size); │ │ │ │ │ + node.setAttributeNS(null, "x", pos.x - offset); │ │ │ │ │ + node.setAttributeNS(null, "y", pos.y - offset); │ │ │ │ │ + if (nextSibling) { │ │ │ │ │ + parent.insertBefore(node, nextSibling) │ │ │ │ │ + } else if (parent) { │ │ │ │ │ + parent.appendChild(node) │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "r", style.pointRadius) │ │ │ │ │ } │ │ │ │ │ - if (layerDefStrList.length > 0) { │ │ │ │ │ - newParams["LAYERDEFS"] = layerDefStrList.join("") │ │ │ │ │ + var rotation = style.rotation; │ │ │ │ │ + if ((rotation !== undefined || node._rotation !== undefined) && pos) { │ │ │ │ │ + node._rotation = rotation; │ │ │ │ │ + rotation |= 0; │ │ │ │ │ + if (node.nodeName !== "svg") { │ │ │ │ │ + node.setAttributeNS(null, "transform", "rotate(" + rotation + " " + pos.x + " " + pos.y + ")") │ │ │ │ │ + } else { │ │ │ │ │ + var metrics = this.symbolMetrics[src.id]; │ │ │ │ │ + node.firstChild.setAttributeNS(null, "transform", "rotate(" + rotation + " " + metrics[1] + " " + metrics[2] + ")") │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var requestString = this.getFullRequestString(newParams); │ │ │ │ │ - return requestString │ │ │ │ │ - }, │ │ │ │ │ - setLayerFilter: function(id, queryDef) { │ │ │ │ │ - if (!this.layerDefs) { │ │ │ │ │ - this.layerDefs = {} │ │ │ │ │ + if (options.isFilled) { │ │ │ │ │ + node.setAttributeNS(null, "fill", style.fillColor); │ │ │ │ │ + node.setAttributeNS(null, "fill-opacity", style.fillOpacity) │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "fill", "none") │ │ │ │ │ } │ │ │ │ │ - if (queryDef) { │ │ │ │ │ - this.layerDefs[id] = queryDef │ │ │ │ │ + if (options.isStroked) { │ │ │ │ │ + node.setAttributeNS(null, "stroke", style.strokeColor); │ │ │ │ │ + node.setAttributeNS(null, "stroke-opacity", style.strokeOpacity); │ │ │ │ │ + node.setAttributeNS(null, "stroke-width", style.strokeWidth * widthFactor); │ │ │ │ │ + node.setAttributeNS(null, "stroke-linecap", style.strokeLinecap || "round"); │ │ │ │ │ + node.setAttributeNS(null, "stroke-linejoin", "round"); │ │ │ │ │ + style.strokeDashstyle && node.setAttributeNS(null, "stroke-dasharray", this.dashStyle(style, widthFactor)) │ │ │ │ │ } else { │ │ │ │ │ - delete this.layerDefs[id] │ │ │ │ │ + node.setAttributeNS(null, "stroke", "none") │ │ │ │ │ + } │ │ │ │ │ + if (style.pointerEvents) { │ │ │ │ │ + node.setAttributeNS(null, "pointer-events", style.pointerEvents) │ │ │ │ │ + } │ │ │ │ │ + if (style.cursor != null) { │ │ │ │ │ + node.setAttributeNS(null, "cursor", style.cursor) │ │ │ │ │ } │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - clearLayerFilter: function(id) { │ │ │ │ │ + dashStyle: function(style, widthFactor) { │ │ │ │ │ + var w = style.strokeWidth * widthFactor; │ │ │ │ │ + var str = style.strokeDashstyle; │ │ │ │ │ + switch (str) { │ │ │ │ │ + case "solid": │ │ │ │ │ + return "none"; │ │ │ │ │ + case "dot": │ │ │ │ │ + return [1, 4 * w].join(); │ │ │ │ │ + case "dash": │ │ │ │ │ + return [4 * w, 4 * w].join(); │ │ │ │ │ + case "dashdot": │ │ │ │ │ + return [4 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ + case "longdash": │ │ │ │ │ + return [8 * w, 4 * w].join(); │ │ │ │ │ + case "longdashdot": │ │ │ │ │ + return [8 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ + default: │ │ │ │ │ + return OpenLayers.String.trim(str).replace(/\s+/g, ",") │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + createNode: function(type, id) { │ │ │ │ │ + var node = document.createElementNS(this.xmlns, type); │ │ │ │ │ if (id) { │ │ │ │ │ - delete this.layerDefs[id] │ │ │ │ │ - } else { │ │ │ │ │ - delete this.layerDefs │ │ │ │ │ + node.setAttributeNS(null, "id", id) │ │ │ │ │ } │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ - var newArguments = [upperParams]; │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, newArguments) │ │ │ │ │ + nodeTypeCompare: function(node, type) { │ │ │ │ │ + return type == node.nodeName │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.ArcGIS93Rest" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Zoomify = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - size: null, │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - standardTileSize: 256, │ │ │ │ │ - tileOriginCorner: "tl", │ │ │ │ │ - numberOfTiers: 0, │ │ │ │ │ - tileCountUpToTier: null, │ │ │ │ │ - tierSizeInTiles: null, │ │ │ │ │ - tierImageSize: null, │ │ │ │ │ - initialize: function(name, url, size, options) { │ │ │ │ │ - this.initializeZoomify(size); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name, url, size, {}, options]) │ │ │ │ │ + createRenderRoot: function() { │ │ │ │ │ + var svg = this.nodeFactory(this.container.id + "_svgRoot", "svg"); │ │ │ │ │ + svg.style.display = "block"; │ │ │ │ │ + return svg │ │ │ │ │ }, │ │ │ │ │ - initializeZoomify: function(size) { │ │ │ │ │ - var imageSize = size.clone(); │ │ │ │ │ - this.size = size.clone(); │ │ │ │ │ - var tiles = new OpenLayers.Size(Math.ceil(imageSize.w / this.standardTileSize), Math.ceil(imageSize.h / this.standardTileSize)); │ │ │ │ │ - this.tierSizeInTiles = [tiles]; │ │ │ │ │ - this.tierImageSize = [imageSize]; │ │ │ │ │ - while (imageSize.w > this.standardTileSize || imageSize.h > this.standardTileSize) { │ │ │ │ │ - imageSize = new OpenLayers.Size(Math.floor(imageSize.w / 2), Math.floor(imageSize.h / 2)); │ │ │ │ │ - tiles = new OpenLayers.Size(Math.ceil(imageSize.w / this.standardTileSize), Math.ceil(imageSize.h / this.standardTileSize)); │ │ │ │ │ - this.tierSizeInTiles.push(tiles); │ │ │ │ │ - this.tierImageSize.push(imageSize) │ │ │ │ │ - } │ │ │ │ │ - this.tierSizeInTiles.reverse(); │ │ │ │ │ - this.tierImageSize.reverse(); │ │ │ │ │ - this.numberOfTiers = this.tierSizeInTiles.length; │ │ │ │ │ - var resolutions = [1]; │ │ │ │ │ - this.tileCountUpToTier = [0]; │ │ │ │ │ - for (var i = 1; i < this.numberOfTiers; i++) { │ │ │ │ │ - resolutions.unshift(Math.pow(2, i)); │ │ │ │ │ - this.tileCountUpToTier.push(this.tierSizeInTiles[i - 1].w * this.tierSizeInTiles[i - 1].h + this.tileCountUpToTier[i - 1]) │ │ │ │ │ - } │ │ │ │ │ - if (!this.serverResolutions) { │ │ │ │ │ - this.serverResolutions = resolutions │ │ │ │ │ - } │ │ │ │ │ + createRoot: function(suffix) { │ │ │ │ │ + return this.nodeFactory(this.container.id + suffix, "g") │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.destroy.apply(this, arguments); │ │ │ │ │ - this.tileCountUpToTier.length = 0; │ │ │ │ │ - this.tierSizeInTiles.length = 0; │ │ │ │ │ - this.tierImageSize.length = 0 │ │ │ │ │ + createDefs: function() { │ │ │ │ │ + var defs = this.nodeFactory(this.container.id + "_defs", "defs"); │ │ │ │ │ + this.rendererRoot.appendChild(defs); │ │ │ │ │ + return defs │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Zoomify(this.name, this.url, this.size, this.options) │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ + drawPoint: function(node, geometry) { │ │ │ │ │ + return this.drawCircle(node, geometry, 1) │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((this.tileOrigin.lat - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getZoomForResolution(res); │ │ │ │ │ - var tileIndex = x + y * this.tierSizeInTiles[z].w + this.tileCountUpToTier[z]; │ │ │ │ │ - var path = "TileGroup" + Math.floor(tileIndex / 256) + "/" + z + "-" + x + "-" + y + ".jpg"; │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(path, url) │ │ │ │ │ + drawCircle: function(node, geometry, radius) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = (geometry.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y = this.top - geometry.y / resolution; │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + node.setAttributeNS(null, "cx", x); │ │ │ │ │ + node.setAttributeNS(null, "cy", y); │ │ │ │ │ + node.setAttributeNS(null, "r", radius); │ │ │ │ │ + return node │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return url + path │ │ │ │ │ }, │ │ │ │ │ - getImageSize: function() { │ │ │ │ │ - if (arguments.length > 0) { │ │ │ │ │ - var bounds = this.adjustBounds(arguments[0]); │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((this.tileOrigin.lat - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getZoomForResolution(res); │ │ │ │ │ - var w = this.standardTileSize; │ │ │ │ │ - var h = this.standardTileSize; │ │ │ │ │ - if (x == this.tierSizeInTiles[z].w - 1) { │ │ │ │ │ - var w = this.tierImageSize[z].w % this.standardTileSize │ │ │ │ │ - } │ │ │ │ │ - if (y == this.tierSizeInTiles[z].h - 1) { │ │ │ │ │ - var h = this.tierImageSize[z].h % this.standardTileSize │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Size(w, h) │ │ │ │ │ + drawLineString: function(node, geometry) { │ │ │ │ │ + var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ + if (componentsResult.path) { │ │ │ │ │ + node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ + return componentsResult.complete ? node : null │ │ │ │ │ } else { │ │ │ │ │ - return this.tileSize │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.map.maxExtent.left, this.map.maxExtent.top) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Zoomify" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Google.v3 = { │ │ │ │ │ - DEFAULTS: { │ │ │ │ │ - sphericalMercator: true, │ │ │ │ │ - projection: "EPSG:900913" │ │ │ │ │ + drawLinearRing: function(node, geometry) { │ │ │ │ │ + var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ + if (componentsResult.path) { │ │ │ │ │ + node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ + return componentsResult.complete ? node : null │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - animationEnabled: true, │ │ │ │ │ - loadMapObject: function() { │ │ │ │ │ - if (!this.type) { │ │ │ │ │ - this.type = google.maps.MapTypeId.ROADMAP │ │ │ │ │ + drawPolygon: function(node, geometry) { │ │ │ │ │ + var d = ""; │ │ │ │ │ + var draw = true; │ │ │ │ │ + var complete = true; │ │ │ │ │ + var linearRingResult, path; │ │ │ │ │ + for (var j = 0, len = geometry.components.length; j < len; j++) { │ │ │ │ │ + d += " M"; │ │ │ │ │ + linearRingResult = this.getComponentsString(geometry.components[j].components, " "); │ │ │ │ │ + path = linearRingResult.path; │ │ │ │ │ + if (path) { │ │ │ │ │ + d += " " + path; │ │ │ │ │ + complete = linearRingResult.complete && complete │ │ │ │ │ + } else { │ │ │ │ │ + draw = false │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var mapObject; │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - mapObject = cache.mapObject; │ │ │ │ │ - ++cache.count │ │ │ │ │ + d += " z"; │ │ │ │ │ + if (draw) { │ │ │ │ │ + node.setAttributeNS(null, "d", d); │ │ │ │ │ + node.setAttributeNS(null, "fill-rule", "evenodd"); │ │ │ │ │ + return complete ? node : null │ │ │ │ │ } else { │ │ │ │ │ - var center = this.map.getCenter(); │ │ │ │ │ - var container = document.createElement("div"); │ │ │ │ │ - container.className = "olForeignContainer"; │ │ │ │ │ - container.style.width = "100%"; │ │ │ │ │ - container.style.height = "100%"; │ │ │ │ │ - mapObject = new google.maps.Map(container, { │ │ │ │ │ - center: center ? new google.maps.LatLng(center.lat, center.lon) : new google.maps.LatLng(0, 0), │ │ │ │ │ - zoom: this.map.getZoom() || 0, │ │ │ │ │ - mapTypeId: this.type, │ │ │ │ │ - disableDefaultUI: true, │ │ │ │ │ - keyboardShortcuts: false, │ │ │ │ │ - draggable: false, │ │ │ │ │ - disableDoubleClickZoom: true, │ │ │ │ │ - scrollwheel: false, │ │ │ │ │ - streetViewControl: false │ │ │ │ │ - }); │ │ │ │ │ - var googleControl = document.createElement("div"); │ │ │ │ │ - googleControl.style.width = "100%"; │ │ │ │ │ - googleControl.style.height = "100%"; │ │ │ │ │ - mapObject.controls[google.maps.ControlPosition.TOP_LEFT].push(googleControl); │ │ │ │ │ - cache = { │ │ │ │ │ - googleControl: googleControl, │ │ │ │ │ - mapObject: mapObject, │ │ │ │ │ - count: 1 │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Layer.Google.cache[this.map.id] = cache │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - this.mapObject = mapObject; │ │ │ │ │ - this.setGMapVisibility(this.visibility) │ │ │ │ │ }, │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - if (this.visibility) { │ │ │ │ │ - google.maps.event.trigger(this.mapObject, "resize") │ │ │ │ │ + drawRectangle: function(node, geometry) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = (geometry.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y = this.top - geometry.y / resolution; │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + node.setAttributeNS(null, "x", x); │ │ │ │ │ + node.setAttributeNS(null, "y", y); │ │ │ │ │ + node.setAttributeNS(null, "width", geometry.width / resolution); │ │ │ │ │ + node.setAttributeNS(null, "height", geometry.height / resolution); │ │ │ │ │ + return node │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - setGMapVisibility: function(visible) { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - var map = this.map; │ │ │ │ │ - if (cache) { │ │ │ │ │ - var type = this.type; │ │ │ │ │ - var layers = map.layers; │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = layers[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Google && layer.visibility === true && layer.inRange === true) { │ │ │ │ │ - type = layer.type; │ │ │ │ │ - visible = true; │ │ │ │ │ - break │ │ │ │ │ + drawText: function(featureId, style, location) { │ │ │ │ │ + var drawOutline = !!style.labelOutlineWidth; │ │ │ │ │ + if (drawOutline) { │ │ │ │ │ + var outlineStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ + outlineStyle.fontColor = outlineStyle.labelOutlineColor; │ │ │ │ │ + outlineStyle.fontStrokeColor = outlineStyle.labelOutlineColor; │ │ │ │ │ + outlineStyle.fontStrokeWidth = style.labelOutlineWidth; │ │ │ │ │ + if (style.labelOutlineOpacity) { │ │ │ │ │ + outlineStyle.fontOpacity = style.labelOutlineOpacity │ │ │ │ │ + } │ │ │ │ │ + delete outlineStyle.labelOutlineWidth; │ │ │ │ │ + this.drawText(featureId, outlineStyle, location) │ │ │ │ │ + } │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = (location.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y = location.y / resolution - this.top; │ │ │ │ │ + var suffix = drawOutline ? this.LABEL_OUTLINE_SUFFIX : this.LABEL_ID_SUFFIX; │ │ │ │ │ + var label = this.nodeFactory(featureId + suffix, "text"); │ │ │ │ │ + label.setAttributeNS(null, "x", x); │ │ │ │ │ + label.setAttributeNS(null, "y", -y); │ │ │ │ │ + if (style.fontColor) { │ │ │ │ │ + label.setAttributeNS(null, "fill", style.fontColor) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStrokeColor) { │ │ │ │ │ + label.setAttributeNS(null, "stroke", style.fontStrokeColor) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStrokeWidth) { │ │ │ │ │ + label.setAttributeNS(null, "stroke-width", style.fontStrokeWidth) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontOpacity) { │ │ │ │ │ + label.setAttributeNS(null, "opacity", style.fontOpacity) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontFamily) { │ │ │ │ │ + label.setAttributeNS(null, "font-family", style.fontFamily) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontSize) { │ │ │ │ │ + label.setAttributeNS(null, "font-size", style.fontSize) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontWeight) { │ │ │ │ │ + label.setAttributeNS(null, "font-weight", style.fontWeight) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStyle) { │ │ │ │ │ + label.setAttributeNS(null, "font-style", style.fontStyle) │ │ │ │ │ + } │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + label.setAttributeNS(null, "pointer-events", "visible"); │ │ │ │ │ + label._featureId = featureId │ │ │ │ │ + } else { │ │ │ │ │ + label.setAttributeNS(null, "pointer-events", "none") │ │ │ │ │ + } │ │ │ │ │ + var align = style.labelAlign || OpenLayers.Renderer.defaultSymbolizer.labelAlign; │ │ │ │ │ + label.setAttributeNS(null, "text-anchor", OpenLayers.Renderer.SVG.LABEL_ALIGN[align[0]] || "middle"); │ │ │ │ │ + if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ + label.setAttributeNS(null, "dominant-baseline", OpenLayers.Renderer.SVG.LABEL_ALIGN[align[1]] || "central") │ │ │ │ │ + } │ │ │ │ │ + var labelRows = style.label.split("\n"); │ │ │ │ │ + var numRows = labelRows.length; │ │ │ │ │ + while (label.childNodes.length > numRows) { │ │ │ │ │ + label.removeChild(label.lastChild) │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + var tspan = this.nodeFactory(featureId + suffix + "_tspan_" + i, "tspan"); │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + tspan._featureId = featureId; │ │ │ │ │ + tspan._geometry = location; │ │ │ │ │ + tspan._geometryClass = location.CLASS_NAME │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.IS_GECKO === false) { │ │ │ │ │ + tspan.setAttributeNS(null, "baseline-shift", OpenLayers.Renderer.SVG.LABEL_VSHIFT[align[1]] || "-35%") │ │ │ │ │ + } │ │ │ │ │ + tspan.setAttribute("x", x); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + var vfactor = OpenLayers.Renderer.SVG.LABEL_VFACTOR[align[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5 │ │ │ │ │ } │ │ │ │ │ + tspan.setAttribute("dy", vfactor * (numRows - 1) + "em") │ │ │ │ │ + } else { │ │ │ │ │ + tspan.setAttribute("dy", "1em") │ │ │ │ │ } │ │ │ │ │ - var container = this.mapObject.getDiv(); │ │ │ │ │ - if (visible === true) { │ │ │ │ │ - if (container.parentNode !== map.div) { │ │ │ │ │ - if (!cache.rendered) { │ │ │ │ │ - var me = this; │ │ │ │ │ - google.maps.event.addListenerOnce(this.mapObject, "tilesloaded", function() { │ │ │ │ │ - cache.rendered = true; │ │ │ │ │ - me.setGMapVisibility(me.getVisibility()); │ │ │ │ │ - me.moveTo(me.map.getCenter()) │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - map.div.appendChild(container); │ │ │ │ │ - cache.googleControl.appendChild(map.viewPortDiv); │ │ │ │ │ - google.maps.event.trigger(this.mapObject, "resize") │ │ │ │ │ + tspan.textContent = labelRows[i] === "" ? " " : labelRows[i]; │ │ │ │ │ + if (!tspan.parentNode) { │ │ │ │ │ + label.appendChild(tspan) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!label.parentNode) { │ │ │ │ │ + this.textRoot.appendChild(label) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getComponentsString: function(components, separator) { │ │ │ │ │ + var renderCmp = []; │ │ │ │ │ + var complete = true; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + var strings = []; │ │ │ │ │ + var str, component; │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + component = components[i]; │ │ │ │ │ + renderCmp.push(component); │ │ │ │ │ + str = this.getShortString(component); │ │ │ │ │ + if (str) { │ │ │ │ │ + strings.push(str) │ │ │ │ │ + } else { │ │ │ │ │ + if (i > 0) { │ │ │ │ │ + if (this.getShortString(components[i - 1])) { │ │ │ │ │ + strings.push(this.clipLine(components[i], components[i - 1])) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.mapObject.setMapTypeId(type) │ │ │ │ │ - } else if (cache.googleControl.hasChildNodes()) { │ │ │ │ │ - map.div.appendChild(map.viewPortDiv); │ │ │ │ │ - map.div.removeChild(container) │ │ │ │ │ + if (i < len - 1) { │ │ │ │ │ + if (this.getShortString(components[i + 1])) { │ │ │ │ │ + strings.push(this.clipLine(components[i], components[i + 1])) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + complete = false │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - getMapContainer: function() { │ │ │ │ │ - return this.mapObject.getDiv() │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ - var moBounds = null; │ │ │ │ │ - if (olBounds != null) { │ │ │ │ │ - var sw = this.sphericalMercator ? this.inverseMercator(olBounds.bottom, olBounds.left) : new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ - var ne = this.sphericalMercator ? this.inverseMercator(olBounds.top, olBounds.right) : new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ - moBounds = new google.maps.LatLngBounds(new google.maps.LatLng(sw.lat, sw.lon), new google.maps.LatLng(ne.lat, ne.lon)) │ │ │ │ │ + return { │ │ │ │ │ + path: strings.join(separator || ","), │ │ │ │ │ + complete: complete │ │ │ │ │ } │ │ │ │ │ - return moBounds │ │ │ │ │ }, │ │ │ │ │ - getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - var size = this.map.getSize(); │ │ │ │ │ - var lon = this.getLongitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ - var lat = this.getLatitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ - var res = this.map.getResolution(); │ │ │ │ │ - var delta_x = moPixel.x - size.w / 2; │ │ │ │ │ - var delta_y = moPixel.y - size.h / 2; │ │ │ │ │ - var lonlat = new OpenLayers.LonLat(lon + delta_x * res, lat - delta_y * res); │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - lonlat = lonlat.wrapDateLine(this.maxExtent) │ │ │ │ │ + clipLine: function(badComponent, goodComponent) { │ │ │ │ │ + if (goodComponent.equals(badComponent)) { │ │ │ │ │ + return "" │ │ │ │ │ } │ │ │ │ │ - return this.getMapObjectLonLatFromLonLat(lonlat.lon, lonlat.lat) │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var maxX = this.MAX_PIXEL - this.translationParameters.x; │ │ │ │ │ + var maxY = this.MAX_PIXEL - this.translationParameters.y; │ │ │ │ │ + var x1 = (goodComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y1 = this.top - goodComponent.y / resolution; │ │ │ │ │ + var x2 = (badComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y2 = this.top - badComponent.y / resolution; │ │ │ │ │ + var k; │ │ │ │ │ + if (x2 < -maxX || x2 > maxX) { │ │ │ │ │ + k = (y2 - y1) / (x2 - x1); │ │ │ │ │ + x2 = x2 < 0 ? -maxX : maxX; │ │ │ │ │ + y2 = y1 + (x2 - x1) * k │ │ │ │ │ + } │ │ │ │ │ + if (y2 < -maxY || y2 > maxY) { │ │ │ │ │ + k = (x2 - x1) / (y2 - y1); │ │ │ │ │ + y2 = y2 < 0 ? -maxY : maxY; │ │ │ │ │ + x2 = x1 + (y2 - y1) * k │ │ │ │ │ + } │ │ │ │ │ + return x2 + "," + y2 │ │ │ │ │ }, │ │ │ │ │ - getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - var res = this.map.getResolution(); │ │ │ │ │ - var extent = this.map.getExtent(); │ │ │ │ │ - return this.getMapObjectPixelFromXY(1 / res * (lon - extent.left), 1 / res * (extent.top - lat)) │ │ │ │ │ + getShortString: function(point) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = (point.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y = this.top - point.y / resolution; │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + return x + "," + y │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - setMapObjectCenter: function(center, zoom) { │ │ │ │ │ - if (this.animationEnabled === false && zoom != this.mapObject.zoom) { │ │ │ │ │ - var mapContainer = this.getMapContainer(); │ │ │ │ │ - google.maps.event.addListenerOnce(this.mapObject, "idle", function() { │ │ │ │ │ - mapContainer.style.visibility = "" │ │ │ │ │ - }); │ │ │ │ │ - mapContainer.style.visibility = "hidden" │ │ │ │ │ + getPosition: function(node) { │ │ │ │ │ + return { │ │ │ │ │ + x: parseFloat(node.getAttributeNS(null, "cx")), │ │ │ │ │ + y: parseFloat(node.getAttributeNS(null, "cy")) │ │ │ │ │ } │ │ │ │ │ - this.mapObject.setOptions({ │ │ │ │ │ - center: center, │ │ │ │ │ - zoom: zoom │ │ │ │ │ - }) │ │ │ │ │ }, │ │ │ │ │ - getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - return this.mapObject.getBoundsZoomLevel(moBounds) │ │ │ │ │ + importSymbol: function(graphicName) { │ │ │ │ │ + if (!this.defs) { │ │ │ │ │ + this.defs = this.createDefs() │ │ │ │ │ + } │ │ │ │ │ + var id = this.container.id + "-" + graphicName; │ │ │ │ │ + var existing = document.getElementById(id); │ │ │ │ │ + if (existing != null) { │ │ │ │ │ + return existing │ │ │ │ │ + } │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(graphicName + " is not a valid symbol name") │ │ │ │ │ + } │ │ │ │ │ + var symbolNode = this.nodeFactory(id, "symbol"); │ │ │ │ │ + var node = this.nodeFactory(null, "polygon"); │ │ │ │ │ + symbolNode.appendChild(node); │ │ │ │ │ + var symbolExtent = new OpenLayers.Bounds(Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ + var points = []; │ │ │ │ │ + var x, y; │ │ │ │ │ + for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ + symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ + symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ + symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ + points.push(x, ",", y) │ │ │ │ │ + } │ │ │ │ │ + node.setAttributeNS(null, "points", points.join(" ")); │ │ │ │ │ + var width = symbolExtent.getWidth(); │ │ │ │ │ + var height = symbolExtent.getHeight(); │ │ │ │ │ + var viewBox = [symbolExtent.left - width, symbolExtent.bottom - height, width * 3, height * 3]; │ │ │ │ │ + symbolNode.setAttributeNS(null, "viewBox", viewBox.join(" ")); │ │ │ │ │ + this.symbolMetrics[id] = [Math.max(width, height), symbolExtent.getCenterLonLat().lon, symbolExtent.getCenterLonLat().lat]; │ │ │ │ │ + this.defs.appendChild(symbolNode); │ │ │ │ │ + return symbolNode │ │ │ │ │ }, │ │ │ │ │ - getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ - var gLatLng; │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ - gLatLng = new google.maps.LatLng(lonlat.lat, lonlat.lon) │ │ │ │ │ - } else { │ │ │ │ │ - gLatLng = new google.maps.LatLng(lat, lon) │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var featureId = OpenLayers.Renderer.Elements.prototype.getFeatureIdFromEvent.apply(this, arguments); │ │ │ │ │ + if (!featureId) { │ │ │ │ │ + var target = evt.target; │ │ │ │ │ + featureId = target.parentNode && target != this.rendererRoot ? target.parentNode._featureId : undefined │ │ │ │ │ } │ │ │ │ │ - return gLatLng │ │ │ │ │ + return featureId │ │ │ │ │ }, │ │ │ │ │ - getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ - return new google.maps.Point(x, y) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.SVG" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_ALIGN = { │ │ │ │ │ + l: "start", │ │ │ │ │ + r: "end", │ │ │ │ │ + b: "bottom", │ │ │ │ │ + t: "hanging" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_VSHIFT = { │ │ │ │ │ + t: "-70%", │ │ │ │ │ + b: "0" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_VFACTOR = { │ │ │ │ │ + t: 0, │ │ │ │ │ + b: -1 │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.SVG.preventDefault = function(e) { │ │ │ │ │ + OpenLayers.Event.preventDefault(e) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol.CSW = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, OpenLayers.Protocol.CSW.DEFAULTS); │ │ │ │ │ + var cls = OpenLayers.Protocol.CSW["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported CSW version: " + options.version │ │ │ │ │ } │ │ │ │ │ + return new cls(options) │ │ │ │ │ }; │ │ │ │ │ -OpenLayers.Layer.Vector.RootContainer = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - layers: null, │ │ │ │ │ - display: function() {}, │ │ │ │ │ - getFeatureFromEvent: function(evt) { │ │ │ │ │ - var layers = this.layers; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0; i < layers.length; i++) { │ │ │ │ │ - feature = layers[i].getFeatureFromEvent(evt); │ │ │ │ │ - if (feature) { │ │ │ │ │ - return feature │ │ │ │ │ +OpenLayers.Protocol.CSW.DEFAULTS = { │ │ │ │ │ + version: "2.0.2" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol.SOS = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, OpenLayers.Protocol.SOS.DEFAULTS); │ │ │ │ │ + var cls = OpenLayers.Protocol.SOS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported SOS version: " + options.version │ │ │ │ │ + } │ │ │ │ │ + return new cls(options) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol.SOS.DEFAULTS = { │ │ │ │ │ + version: "1.0.0" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol.WFS = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, OpenLayers.Protocol.WFS.DEFAULTS); │ │ │ │ │ + var cls = OpenLayers.Protocol.WFS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported WFS version: " + options.version │ │ │ │ │ + } │ │ │ │ │ + return new cls(options) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol.WFS.fromWMSLayer = function(layer, options) { │ │ │ │ │ + var typeName, featurePrefix; │ │ │ │ │ + var param = layer.params["LAYERS"]; │ │ │ │ │ + var parts = (OpenLayers.Util.isArray(param) ? param[0] : param).split(":"); │ │ │ │ │ + if (parts.length > 1) { │ │ │ │ │ + featurePrefix = parts[0] │ │ │ │ │ + } │ │ │ │ │ + typeName = parts.pop(); │ │ │ │ │ + var protocolOptions = { │ │ │ │ │ + url: layer.url, │ │ │ │ │ + featureType: typeName, │ │ │ │ │ + featurePrefix: featurePrefix, │ │ │ │ │ + srsName: layer.projection && layer.projection.getCode() || layer.map && layer.map.getProjectionObject().getCode(), │ │ │ │ │ + version: "1.1.0" │ │ │ │ │ + }; │ │ │ │ │ + return new OpenLayers.Protocol.WFS(OpenLayers.Util.applyDefaults(options, protocolOptions)) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol.WFS.DEFAULTS = { │ │ │ │ │ + version: "1.0.0" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol.HTTP = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + url: null, │ │ │ │ │ + headers: null, │ │ │ │ │ + params: null, │ │ │ │ │ + callback: null, │ │ │ │ │ + scope: null, │ │ │ │ │ + readWithPOST: false, │ │ │ │ │ + updateWithPOST: false, │ │ │ │ │ + deleteWithPOST: false, │ │ │ │ │ + wildcarded: false, │ │ │ │ │ + srsInBBOX: false, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + this.params = {}; │ │ │ │ │ + this.headers = {}; │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ + var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ + wildcarded: this.wildcarded, │ │ │ │ │ + srsInBBOX: this.srsInBBOX │ │ │ │ │ + }); │ │ │ │ │ + this.filterToParams = function(filter, params) { │ │ │ │ │ + return format.write(filter, params) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.collectRoots(); │ │ │ │ │ - map.events.register("changelayer", this, this.handleChangeLayer) │ │ │ │ │ - }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - map.events.unregister("changelayer", this, this.handleChangeLayer); │ │ │ │ │ - this.resetRoots(); │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.params = null; │ │ │ │ │ + this.headers = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ }, │ │ │ │ │ - collectRoots: function() { │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = 0; i < this.map.layers.length; ++i) { │ │ │ │ │ - layer = this.map.layers[i]; │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ - layer.renderer.moveRoot(this.renderer) │ │ │ │ │ - } │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.params = OpenLayers.Util.applyDefaults(options.params, this.options.params); │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + if (options.filter && this.filterToParams) { │ │ │ │ │ + options.params = this.filterToParams(options.filter, options.params) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - resetRoots: function() { │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = 0; i < this.layers.length; ++i) { │ │ │ │ │ - layer = this.layers[i]; │ │ │ │ │ - if (this.renderer && layer.renderer.getRenderLayerId() == this.id) { │ │ │ │ │ - this.renderer.moveRoot(layer.renderer) │ │ │ │ │ - } │ │ │ │ │ + var readWithPOST = options.readWithPOST !== undefined ? options.readWithPOST : this.readWithPOST; │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + if (readWithPOST) { │ │ │ │ │ + var headers = options.headers || {}; │ │ │ │ │ + headers["Content-Type"] = "application/x-www-form-urlencoded"; │ │ │ │ │ + resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ + data: OpenLayers.Util.getParameterString(options.params), │ │ │ │ │ + headers: headers │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + resp.priv = OpenLayers.Request.GET({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ + return resp │ │ │ │ │ }, │ │ │ │ │ - handleChangeLayer: function(evt) { │ │ │ │ │ - var layer = evt.layer; │ │ │ │ │ - if (evt.property == "order" && OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ - this.resetRoots(); │ │ │ │ │ - this.collectRoots() │ │ │ │ │ - } │ │ │ │ │ + handleRead: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Vector.RootContainer" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Popup.Anchored = OpenLayers.Class(OpenLayers.Popup, { │ │ │ │ │ - relativePosition: null, │ │ │ │ │ - keepInMap: true, │ │ │ │ │ - anchor: null, │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, closeBoxCallback) { │ │ │ │ │ - var newArguments = [id, lonlat, contentSize, contentHTML, closeBox, closeBoxCallback]; │ │ │ │ │ - OpenLayers.Popup.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - this.anchor = anchor != null ? anchor : { │ │ │ │ │ - size: new OpenLayers.Size(0, 0), │ │ │ │ │ - offset: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - } │ │ │ │ │ + create: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: features, │ │ │ │ │ + requestType: "create" │ │ │ │ │ + }); │ │ │ │ │ + resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleCreate, resp, options), │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(features) │ │ │ │ │ + }); │ │ │ │ │ + return resp │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.anchor = null; │ │ │ │ │ - this.relativePosition = null; │ │ │ │ │ - OpenLayers.Popup.prototype.destroy.apply(this, arguments) │ │ │ │ │ + handleCreate: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options) │ │ │ │ │ }, │ │ │ │ │ - show: function() { │ │ │ │ │ - this.updatePosition(); │ │ │ │ │ - OpenLayers.Popup.prototype.show.apply(this, arguments) │ │ │ │ │ + update: function(feature, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var url = options.url || feature.url || this.options.url + "/" + feature.fid; │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: feature, │ │ │ │ │ + requestType: "update" │ │ │ │ │ + }); │ │ │ │ │ + var method = this.updateWithPOST ? "POST" : "PUT"; │ │ │ │ │ + resp.priv = OpenLayers.Request[method]({ │ │ │ │ │ + url: url, │ │ │ │ │ + callback: this.createCallback(this.handleUpdate, resp, options), │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(feature) │ │ │ │ │ + }); │ │ │ │ │ + return resp │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - var oldRelativePosition = this.relativePosition; │ │ │ │ │ - this.relativePosition = this.calculateRelativePosition(px); │ │ │ │ │ - OpenLayers.Popup.prototype.moveTo.call(this, this.calculateNewPx(px)); │ │ │ │ │ - if (this.relativePosition != oldRelativePosition) { │ │ │ │ │ - this.updateRelativePosition() │ │ │ │ │ - } │ │ │ │ │ + handleUpdate: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options) │ │ │ │ │ }, │ │ │ │ │ - setSize: function(contentSize) { │ │ │ │ │ - OpenLayers.Popup.prototype.setSize.apply(this, arguments); │ │ │ │ │ - if (this.lonlat && this.map) { │ │ │ │ │ - var px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ - this.moveTo(px) │ │ │ │ │ + delete: function(feature, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var url = options.url || feature.url || this.options.url + "/" + feature.fid; │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: feature, │ │ │ │ │ + requestType: "delete" │ │ │ │ │ + }); │ │ │ │ │ + var method = this.deleteWithPOST ? "POST" : "DELETE"; │ │ │ │ │ + var requestOptions = { │ │ │ │ │ + url: url, │ │ │ │ │ + callback: this.createCallback(this.handleDelete, resp, options), │ │ │ │ │ + headers: options.headers │ │ │ │ │ + }; │ │ │ │ │ + if (this.deleteWithPOST) { │ │ │ │ │ + requestOptions.data = this.format.write(feature) │ │ │ │ │ } │ │ │ │ │ + resp.priv = OpenLayers.Request[method](requestOptions); │ │ │ │ │ + return resp │ │ │ │ │ }, │ │ │ │ │ - calculateRelativePosition: function(px) { │ │ │ │ │ - var lonlat = this.map.getLonLatFromLayerPx(px); │ │ │ │ │ - var extent = this.map.getExtent(); │ │ │ │ │ - var quadrant = extent.determineQuadrant(lonlat); │ │ │ │ │ - return OpenLayers.Bounds.oppositeQuadrant(quadrant) │ │ │ │ │ - }, │ │ │ │ │ - updateRelativePosition: function() {}, │ │ │ │ │ - calculateNewPx: function(px) { │ │ │ │ │ - var newPx = px.offset(this.anchor.offset); │ │ │ │ │ - var size = this.size || this.contentSize; │ │ │ │ │ - var top = this.relativePosition.charAt(0) == "t"; │ │ │ │ │ - newPx.y += top ? -size.h : this.anchor.size.h; │ │ │ │ │ - var left = this.relativePosition.charAt(1) == "l"; │ │ │ │ │ - newPx.x += left ? -size.w : this.anchor.size.w; │ │ │ │ │ - return newPx │ │ │ │ │ + handleDelete: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup.Anchored" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Popup.Framed = OpenLayers.Class(OpenLayers.Popup.Anchored, { │ │ │ │ │ - imageSrc: null, │ │ │ │ │ - imageSize: null, │ │ │ │ │ - isAlphaImage: false, │ │ │ │ │ - positionBlocks: null, │ │ │ │ │ - blocks: null, │ │ │ │ │ - fixedRelativePosition: false, │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, closeBoxCallback) { │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (this.fixedRelativePosition) { │ │ │ │ │ - this.updateRelativePosition(); │ │ │ │ │ - this.calculateRelativePosition = function(px) { │ │ │ │ │ - return this.relativePosition │ │ │ │ │ + handleResponse: function(resp, options) { │ │ │ │ │ + var request = resp.priv; │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + if (resp.requestType != "delete") { │ │ │ │ │ + resp.features = this.parseFeatures(request) │ │ │ │ │ + } │ │ │ │ │ + resp.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ + } else { │ │ │ │ │ + resp.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ } │ │ │ │ │ + options.callback.call(options.scope, resp) │ │ │ │ │ } │ │ │ │ │ - this.contentDiv.style.position = "absolute"; │ │ │ │ │ - this.contentDiv.style.zIndex = 1; │ │ │ │ │ - if (closeBox) { │ │ │ │ │ - this.closeDiv.style.zIndex = 1 │ │ │ │ │ + }, │ │ │ │ │ + parseFeatures: function(request) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ } │ │ │ │ │ - this.groupDiv.style.position = "absolute"; │ │ │ │ │ - this.groupDiv.style.top = "0px"; │ │ │ │ │ - this.groupDiv.style.left = "0px"; │ │ │ │ │ - this.groupDiv.style.height = "100%"; │ │ │ │ │ - this.groupDiv.style.width = "100%" │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ + return this.format.read(doc) │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.imageSrc = null; │ │ │ │ │ - this.imageSize = null; │ │ │ │ │ - this.isAlphaImage = null; │ │ │ │ │ - this.fixedRelativePosition = false; │ │ │ │ │ - this.positionBlocks = null; │ │ │ │ │ - for (var i = 0; i < this.blocks.length; i++) { │ │ │ │ │ - var block = this.blocks[i]; │ │ │ │ │ - if (block.image) { │ │ │ │ │ - block.div.removeChild(block.image) │ │ │ │ │ + commit: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = [], │ │ │ │ │ + nResponses = 0; │ │ │ │ │ + var types = {}; │ │ │ │ │ + types[OpenLayers.State.INSERT] = []; │ │ │ │ │ + types[OpenLayers.State.UPDATE] = []; │ │ │ │ │ + types[OpenLayers.State.DELETE] = []; │ │ │ │ │ + var feature, list, requestFeatures = []; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + list = types[feature.state]; │ │ │ │ │ + if (list) { │ │ │ │ │ + list.push(feature); │ │ │ │ │ + requestFeatures.push(feature) │ │ │ │ │ } │ │ │ │ │ - block.image = null; │ │ │ │ │ - if (block.div) { │ │ │ │ │ - this.groupDiv.removeChild(block.div) │ │ │ │ │ + } │ │ │ │ │ + var nRequests = (types[OpenLayers.State.INSERT].length > 0 ? 1 : 0) + types[OpenLayers.State.UPDATE].length + types[OpenLayers.State.DELETE].length; │ │ │ │ │ + var success = true; │ │ │ │ │ + var finalResponse = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: requestFeatures │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + function insertCallback(response) { │ │ │ │ │ + var len = response.features ? response.features.length : 0; │ │ │ │ │ + var fids = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + fids[i] = response.features[i].fid │ │ │ │ │ } │ │ │ │ │ - block.div = null │ │ │ │ │ + finalResponse.insertIds = fids; │ │ │ │ │ + callback.apply(this, [response]) │ │ │ │ │ } │ │ │ │ │ - this.blocks = null; │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - setBackgroundColor: function(color) {}, │ │ │ │ │ - setBorder: function() {}, │ │ │ │ │ - setOpacity: function(opacity) {}, │ │ │ │ │ - setSize: function(contentSize) { │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.setSize.apply(this, arguments); │ │ │ │ │ - this.updateBlocks() │ │ │ │ │ - }, │ │ │ │ │ - updateRelativePosition: function() { │ │ │ │ │ - this.padding = this.positionBlocks[this.relativePosition].padding; │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ - this.closeDiv.style.right = contentDivPadding.right + this.padding.right + "px"; │ │ │ │ │ - this.closeDiv.style.top = contentDivPadding.top + this.padding.top + "px" │ │ │ │ │ + │ │ │ │ │ + function callback(response) { │ │ │ │ │ + this.callUserCallback(response, options); │ │ │ │ │ + success = success && response.success(); │ │ │ │ │ + nResponses++; │ │ │ │ │ + if (nResponses >= nRequests) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + finalResponse.code = success ? OpenLayers.Protocol.Response.SUCCESS : OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + options.callback.apply(options.scope, [finalResponse]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.updateBlocks() │ │ │ │ │ - }, │ │ │ │ │ - calculateNewPx: function(px) { │ │ │ │ │ - var newPx = OpenLayers.Popup.Anchored.prototype.calculateNewPx.apply(this, arguments); │ │ │ │ │ - newPx = newPx.offset(this.positionBlocks[this.relativePosition].offset); │ │ │ │ │ - return newPx │ │ │ │ │ - }, │ │ │ │ │ - createBlocks: function() { │ │ │ │ │ - this.blocks = []; │ │ │ │ │ - var firstPosition = null; │ │ │ │ │ - for (var key in this.positionBlocks) { │ │ │ │ │ - firstPosition = key; │ │ │ │ │ - break │ │ │ │ │ + var queue = types[OpenLayers.State.INSERT]; │ │ │ │ │ + if (queue.length > 0) { │ │ │ │ │ + resp.push(this.create(queue, OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: insertCallback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options.create))) │ │ │ │ │ } │ │ │ │ │ - var position = this.positionBlocks[firstPosition]; │ │ │ │ │ - for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ - var block = {}; │ │ │ │ │ - this.blocks.push(block); │ │ │ │ │ - var divId = this.id + "_FrameDecorationDiv_" + i; │ │ │ │ │ - block.div = OpenLayers.Util.createDiv(divId, null, null, null, "absolute", null, "hidden", null); │ │ │ │ │ - var imgId = this.id + "_FrameDecorationImg_" + i; │ │ │ │ │ - var imageCreator = this.isAlphaImage ? OpenLayers.Util.createAlphaImageDiv : OpenLayers.Util.createImage; │ │ │ │ │ - block.image = imageCreator(imgId, null, this.imageSize, this.imageSrc, "absolute", null, null, null); │ │ │ │ │ - block.div.appendChild(block.image); │ │ │ │ │ - this.groupDiv.appendChild(block.div) │ │ │ │ │ + queue = types[OpenLayers.State.UPDATE]; │ │ │ │ │ + for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ + resp.push(this.update(queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: callback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options.update))) │ │ │ │ │ } │ │ │ │ │ + queue = types[OpenLayers.State.DELETE]; │ │ │ │ │ + for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ + resp.push(this["delete"](queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: callback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options["delete"]))) │ │ │ │ │ + } │ │ │ │ │ + return resp │ │ │ │ │ }, │ │ │ │ │ - updateBlocks: function() { │ │ │ │ │ - if (!this.blocks) { │ │ │ │ │ - this.createBlocks() │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + response.priv.abort() │ │ │ │ │ } │ │ │ │ │ - if (this.size && this.relativePosition) { │ │ │ │ │ - var position = this.positionBlocks[this.relativePosition]; │ │ │ │ │ - for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ - var positionBlock = position.blocks[i]; │ │ │ │ │ - var block = this.blocks[i]; │ │ │ │ │ - var l = positionBlock.anchor.left; │ │ │ │ │ - var b = positionBlock.anchor.bottom; │ │ │ │ │ - var r = positionBlock.anchor.right; │ │ │ │ │ - var t = positionBlock.anchor.top; │ │ │ │ │ - var w = isNaN(positionBlock.size.w) ? this.size.w - (r + l) : positionBlock.size.w; │ │ │ │ │ - var h = isNaN(positionBlock.size.h) ? this.size.h - (b + t) : positionBlock.size.h; │ │ │ │ │ - block.div.style.width = (w < 0 ? 0 : w) + "px"; │ │ │ │ │ - block.div.style.height = (h < 0 ? 0 : h) + "px"; │ │ │ │ │ - block.div.style.left = l != null ? l + "px" : ""; │ │ │ │ │ - block.div.style.bottom = b != null ? b + "px" : ""; │ │ │ │ │ - block.div.style.right = r != null ? r + "px" : ""; │ │ │ │ │ - block.div.style.top = t != null ? t + "px" : ""; │ │ │ │ │ - block.image.style.left = positionBlock.position.x + "px"; │ │ │ │ │ - block.image.style.top = positionBlock.position.y + "px" │ │ │ │ │ - } │ │ │ │ │ - this.contentDiv.style.left = this.padding.left + "px"; │ │ │ │ │ - this.contentDiv.style.top = this.padding.top + "px" │ │ │ │ │ + }, │ │ │ │ │ + callUserCallback: function(resp, options) { │ │ │ │ │ + var opt = options[resp.requestType]; │ │ │ │ │ + if (opt && opt.callback) { │ │ │ │ │ + opt.callback.call(opt.scope, resp) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup.Framed" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.HTTP" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Popup.FramedCloud = OpenLayers.Class(OpenLayers.Popup.Framed, { │ │ │ │ │ - contentDisplayClass: "olFramedCloudPopupContent", │ │ │ │ │ - autoSize: true, │ │ │ │ │ - panMapIfOutOfView: true, │ │ │ │ │ - imageSize: new OpenLayers.Size(1276, 736), │ │ │ │ │ - isAlphaImage: false, │ │ │ │ │ - fixedRelativePosition: false, │ │ │ │ │ - positionBlocks: { │ │ │ │ │ - tl: { │ │ │ │ │ - offset: new OpenLayers.Pixel(44, 0), │ │ │ │ │ - padding: new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ - blocks: [{ │ │ │ │ │ - size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size("auto", 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, 18), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -632) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(81, 35), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -688) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - tr: { │ │ │ │ │ - offset: new OpenLayers.Pixel(-45, 0), │ │ │ │ │ - padding: new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ - blocks: [{ │ │ │ │ │ - size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size("auto", 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -631) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(81, 35), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, null, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-215, -687) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - bl: { │ │ │ │ │ - offset: new OpenLayers.Pixel(45, 0), │ │ │ │ │ - padding: new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ - blocks: [{ │ │ │ │ │ - size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size("auto", 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(81, 33), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, null, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-101, -674) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - br: { │ │ │ │ │ - offset: new OpenLayers.Pixel(-44, 0), │ │ │ │ │ - padding: new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ - blocks: [{ │ │ │ │ │ - size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size("auto", 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(81, 33), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, null, null, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-311, -674) │ │ │ │ │ - }] │ │ │ │ │ +OpenLayers.Protocol.Script = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + url: null, │ │ │ │ │ + params: null, │ │ │ │ │ + callback: null, │ │ │ │ │ + callbackTemplate: "OpenLayers.Protocol.Script.registry.${id}", │ │ │ │ │ + callbackKey: "callback", │ │ │ │ │ + callbackPrefix: "", │ │ │ │ │ + scope: null, │ │ │ │ │ + format: null, │ │ │ │ │ + pendingRequests: null, │ │ │ │ │ + srsInBBOX: false, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + this.params = {}; │ │ │ │ │ + this.pendingRequests = {}; │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (!this.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.GeoJSON │ │ │ │ │ + } │ │ │ │ │ + if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ + var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ + srsInBBOX: this.srsInBBOX │ │ │ │ │ + }); │ │ │ │ │ + this.filterToParams = function(filter, params) { │ │ │ │ │ + return format.write(filter, params) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - minSize: new OpenLayers.Size(105, 10), │ │ │ │ │ - maxSize: new OpenLayers.Size(1200, 660), │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, closeBoxCallback) { │ │ │ │ │ - this.imageSrc = OpenLayers.Util.getImageLocation("cloud-popup-relative.png"); │ │ │ │ │ - OpenLayers.Popup.Framed.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.contentDiv.className = this.contentDisplayClass │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + options.params = OpenLayers.Util.applyDefaults(options.params, this.options.params); │ │ │ │ │ + if (options.filter && this.filterToParams) { │ │ │ │ │ + options.params = this.filterToParams(options.filter, options.params) │ │ │ │ │ + } │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + var request = this.createRequest(options.url, options.params, OpenLayers.Function.bind(function(data) { │ │ │ │ │ + response.data = data; │ │ │ │ │ + this.handleRead(response, options) │ │ │ │ │ + }, this)); │ │ │ │ │ + response.priv = request; │ │ │ │ │ + return response │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup.FramedCloud" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WFSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - defaultVersion: "1.1.0", │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSCapabilities" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WPSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - defaultVersion: "1.0.0", │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WPSCapabilities" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.Atom = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - namespaces: { │ │ │ │ │ - atom: "http://www.w3.org/2005/Atom", │ │ │ │ │ - georss: "http://www.georss.org/georss" │ │ │ │ │ + createRequest: function(url, params, callback) { │ │ │ │ │ + var id = OpenLayers.Protocol.Script.register(callback); │ │ │ │ │ + var name = OpenLayers.String.format(this.callbackTemplate, { │ │ │ │ │ + id: id │ │ │ │ │ + }); │ │ │ │ │ + params = OpenLayers.Util.extend({}, params); │ │ │ │ │ + params[this.callbackKey] = this.callbackPrefix + name; │ │ │ │ │ + url = OpenLayers.Util.urlAppend(url, OpenLayers.Util.getParameterString(params)); │ │ │ │ │ + var script = document.createElement("script"); │ │ │ │ │ + script.type = "text/javascript"; │ │ │ │ │ + script.src = url; │ │ │ │ │ + script.id = "OpenLayers_Protocol_Script_" + id; │ │ │ │ │ + this.pendingRequests[script.id] = script; │ │ │ │ │ + var head = document.getElementsByTagName("head")[0]; │ │ │ │ │ + head.appendChild(script); │ │ │ │ │ + return script │ │ │ │ │ }, │ │ │ │ │ - feedTitle: "untitled", │ │ │ │ │ - defaultEntryTitle: "untitled", │ │ │ │ │ - gmlParser: null, │ │ │ │ │ - xy: false, │ │ │ │ │ - read: function(doc) { │ │ │ │ │ - if (typeof doc == "string") { │ │ │ │ │ - doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]) │ │ │ │ │ + destroyRequest: function(script) { │ │ │ │ │ + OpenLayers.Protocol.Script.unregister(script.id.split("_").pop()); │ │ │ │ │ + delete this.pendingRequests[script.id]; │ │ │ │ │ + if (script.parentNode) { │ │ │ │ │ + script.parentNode.removeChild(script) │ │ │ │ │ } │ │ │ │ │ - return this.parseFeatures(doc) │ │ │ │ │ }, │ │ │ │ │ - write: function(features) { │ │ │ │ │ - var doc; │ │ │ │ │ - if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ - doc = this.createElementNSPlus("atom:feed"); │ │ │ │ │ - doc.appendChild(this.createElementNSPlus("atom:title", { │ │ │ │ │ - value: this.feedTitle │ │ │ │ │ - })); │ │ │ │ │ - for (var i = 0, ii = features.length; i < ii; i++) { │ │ │ │ │ - doc.appendChild(this.buildEntryNode(features[i])) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - doc = this.buildEntryNode(features) │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [doc]) │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + this.handleResponse(response, options) │ │ │ │ │ }, │ │ │ │ │ - buildContentNode: function(content) { │ │ │ │ │ - var node = this.createElementNSPlus("atom:content", { │ │ │ │ │ - attributes: { │ │ │ │ │ - type: content.type || null │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (content.src) { │ │ │ │ │ - node.setAttribute("src", content.src) │ │ │ │ │ - } else { │ │ │ │ │ - if (content.type == "text" || content.type == null) { │ │ │ │ │ - node.appendChild(this.createTextNode(content.value)) │ │ │ │ │ - } else if (content.type == "html") { │ │ │ │ │ - if (typeof content.value != "string") { │ │ │ │ │ - throw "HTML content must be in form of an escaped string" │ │ │ │ │ - } │ │ │ │ │ - node.appendChild(this.createTextNode(content.value)) │ │ │ │ │ - } else if (content.type == "xhtml") { │ │ │ │ │ - node.appendChild(content.value) │ │ │ │ │ - } else if (content.type == "xhtml" || content.type.match(/(\+|\/)xml$/)) { │ │ │ │ │ - node.appendChild(content.value) │ │ │ │ │ + handleResponse: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + if (response.data) { │ │ │ │ │ + response.features = this.parseFeatures(response.data); │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ } else { │ │ │ │ │ - node.appendChild(this.createTextNode(content.value)) │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ } │ │ │ │ │ + this.destroyRequest(response.priv); │ │ │ │ │ + options.callback.call(options.scope, response) │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ }, │ │ │ │ │ - buildEntryNode: function(feature) { │ │ │ │ │ - var attrib = feature.attributes; │ │ │ │ │ - var atomAttrib = attrib.atom || {}; │ │ │ │ │ - var entryNode = this.createElementNSPlus("atom:entry"); │ │ │ │ │ - if (atomAttrib.authors) { │ │ │ │ │ - var authors = OpenLayers.Util.isArray(atomAttrib.authors) ? atomAttrib.authors : [atomAttrib.authors]; │ │ │ │ │ - for (var i = 0, ii = authors.length; i < ii; i++) { │ │ │ │ │ - entryNode.appendChild(this.buildPersonConstructNode("author", authors[i])) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (atomAttrib.categories) { │ │ │ │ │ - var categories = OpenLayers.Util.isArray(atomAttrib.categories) ? atomAttrib.categories : [atomAttrib.categories]; │ │ │ │ │ - var category; │ │ │ │ │ - for (var i = 0, ii = categories.length; i < ii; i++) { │ │ │ │ │ - category = categories[i]; │ │ │ │ │ - entryNode.appendChild(this.createElementNSPlus("atom:category", { │ │ │ │ │ - attributes: { │ │ │ │ │ - term: category.term, │ │ │ │ │ - scheme: category.scheme || null, │ │ │ │ │ - label: category.label || null │ │ │ │ │ - } │ │ │ │ │ - })) │ │ │ │ │ + parseFeatures: function(data) { │ │ │ │ │ + return this.format.read(data) │ │ │ │ │ + }, │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + this.destroyRequest(response.priv) │ │ │ │ │ + } else { │ │ │ │ │ + for (var key in this.pendingRequests) { │ │ │ │ │ + this.destroyRequest(this.pendingRequests[key]) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (atomAttrib.content) { │ │ │ │ │ - entryNode.appendChild(this.buildContentNode(atomAttrib.content)) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.abort(); │ │ │ │ │ + delete this.params; │ │ │ │ │ + delete this.format; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.Script" │ │ │ │ │ +}); │ │ │ │ │ +(function() { │ │ │ │ │ + var o = OpenLayers.Protocol.Script; │ │ │ │ │ + var counter = 0; │ │ │ │ │ + o.registry = {}; │ │ │ │ │ + o.register = function(callback) { │ │ │ │ │ + var id = "c" + ++counter; │ │ │ │ │ + o.registry[id] = function() { │ │ │ │ │ + callback.apply(this, arguments) │ │ │ │ │ + }; │ │ │ │ │ + return id │ │ │ │ │ + }; │ │ │ │ │ + o.unregister = function(id) { │ │ │ │ │ + delete o.registry[id] │ │ │ │ │ + } │ │ │ │ │ +})(); │ │ │ │ │ +OpenLayers.Protocol.WFS.v1 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + version: null, │ │ │ │ │ + srsName: "EPSG:4326", │ │ │ │ │ + featureType: null, │ │ │ │ │ + featureNS: null, │ │ │ │ │ + geometryName: "the_geom", │ │ │ │ │ + schema: null, │ │ │ │ │ + featurePrefix: "feature", │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + readFormat: null, │ │ │ │ │ + readOptions: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!options.format) { │ │ │ │ │ + this.format = OpenLayers.Format.WFST(OpenLayers.Util.extend({ │ │ │ │ │ + version: this.version, │ │ │ │ │ + featureType: this.featureType, │ │ │ │ │ + featureNS: this.featureNS, │ │ │ │ │ + featurePrefix: this.featurePrefix, │ │ │ │ │ + geometryName: this.geometryName, │ │ │ │ │ + srsName: this.srsName, │ │ │ │ │ + schema: this.schema │ │ │ │ │ + }, this.formatOptions)) │ │ │ │ │ } │ │ │ │ │ - if (atomAttrib.contributors) { │ │ │ │ │ - var contributors = OpenLayers.Util.isArray(atomAttrib.contributors) ? atomAttrib.contributors : [atomAttrib.contributors]; │ │ │ │ │ - for (var i = 0, ii = contributors.length; i < ii; i++) { │ │ │ │ │ - entryNode.appendChild(this.buildPersonConstructNode("contributor", contributors[i])) │ │ │ │ │ - } │ │ │ │ │ + if (!options.geometryName && parseFloat(this.format.version) > 1) { │ │ │ │ │ + this.setGeometryName(null) │ │ │ │ │ } │ │ │ │ │ - if (feature.fid) { │ │ │ │ │ - entryNode.appendChild(this.createElementNSPlus("atom:id", { │ │ │ │ │ - value: feature.fid │ │ │ │ │ - })) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.options && !this.options.format) { │ │ │ │ │ + this.format.destroy() │ │ │ │ │ } │ │ │ │ │ - if (atomAttrib.links) { │ │ │ │ │ - var links = OpenLayers.Util.isArray(atomAttrib.links) ? atomAttrib.links : [atomAttrib.links]; │ │ │ │ │ - var link; │ │ │ │ │ - for (var i = 0, ii = links.length; i < ii; i++) { │ │ │ │ │ - link = links[i]; │ │ │ │ │ - entryNode.appendChild(this.createElementNSPlus("atom:link", { │ │ │ │ │ - attributes: { │ │ │ │ │ - href: link.href, │ │ │ │ │ - rel: link.rel || null, │ │ │ │ │ - type: link.type || null, │ │ │ │ │ - hreflang: link.hreflang || null, │ │ │ │ │ - title: link.title || null, │ │ │ │ │ - length: link.length || null │ │ │ │ │ + this.format = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ + }, │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + var data = OpenLayers.Format.XML.prototype.write.apply(this.format, [this.format.writeNode("wfs:GetFeature", options)]); │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: data │ │ │ │ │ + }); │ │ │ │ │ + return response │ │ │ │ │ + }, │ │ │ │ │ + setFeatureType: function(featureType) { │ │ │ │ │ + this.featureType = featureType; │ │ │ │ │ + this.format.featureType = featureType │ │ │ │ │ + }, │ │ │ │ │ + setGeometryName: function(geometryName) { │ │ │ │ │ + this.geometryName = geometryName; │ │ │ │ │ + this.format.geometryName = geometryName │ │ │ │ │ + }, │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + var result = this.parseResponse(request, options.readOptions); │ │ │ │ │ + if (result && result.success !== false) { │ │ │ │ │ + if (options.readOptions && options.readOptions.output == "object") { │ │ │ │ │ + OpenLayers.Util.extend(response, result) │ │ │ │ │ + } else { │ │ │ │ │ + response.features = result │ │ │ │ │ } │ │ │ │ │ - })) │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + response.error = result │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ } │ │ │ │ │ + options.callback.call(options.scope, response) │ │ │ │ │ } │ │ │ │ │ - if (atomAttrib.published) { │ │ │ │ │ - entryNode.appendChild(this.createElementNSPlus("atom:published", { │ │ │ │ │ - value: atomAttrib.published │ │ │ │ │ - })) │ │ │ │ │ - } │ │ │ │ │ - if (atomAttrib.rights) { │ │ │ │ │ - entryNode.appendChild(this.createElementNSPlus("atom:rights", { │ │ │ │ │ - value: atomAttrib.rights │ │ │ │ │ - })) │ │ │ │ │ - } │ │ │ │ │ - if (atomAttrib.summary || attrib.description) { │ │ │ │ │ - entryNode.appendChild(this.createElementNSPlus("atom:summary", { │ │ │ │ │ - value: atomAttrib.summary || attrib.description │ │ │ │ │ - })) │ │ │ │ │ + }, │ │ │ │ │ + parseResponse: function(request, options) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ } │ │ │ │ │ - entryNode.appendChild(this.createElementNSPlus("atom:title", { │ │ │ │ │ - value: atomAttrib.title || attrib.title || this.defaultEntryTitle │ │ │ │ │ - })); │ │ │ │ │ - if (atomAttrib.updated) { │ │ │ │ │ - entryNode.appendChild(this.createElementNSPlus("atom:updated", { │ │ │ │ │ - value: atomAttrib.updated │ │ │ │ │ - })) │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null │ │ │ │ │ } │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - var whereNode = this.createElementNSPlus("georss:where"); │ │ │ │ │ - whereNode.appendChild(this.buildGeometryNode(feature.geometry)); │ │ │ │ │ - entryNode.appendChild(whereNode) │ │ │ │ │ + var result = this.readFormat !== null ? this.readFormat.read(doc) : this.format.read(doc, options); │ │ │ │ │ + if (!this.featureNS) { │ │ │ │ │ + var format = this.readFormat || this.format; │ │ │ │ │ + this.featureNS = format.featureNS; │ │ │ │ │ + format.autoConfig = false; │ │ │ │ │ + if (!this.geometryName) { │ │ │ │ │ + this.setGeometryName(format.geometryName) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return entryNode │ │ │ │ │ + return result │ │ │ │ │ }, │ │ │ │ │ - initGmlParser: function() { │ │ │ │ │ - this.gmlParser = new OpenLayers.Format.GML.v3({ │ │ │ │ │ - xy: this.xy, │ │ │ │ │ - featureNS: "http://example.com#feature", │ │ │ │ │ - internalProjection: this.internalProjection, │ │ │ │ │ - externalProjection: this.externalProjection │ │ │ │ │ - }) │ │ │ │ │ + commit: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "commit", │ │ │ │ │ + reqFeatures: features │ │ │ │ │ + }); │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(features, options), │ │ │ │ │ + callback: this.createCallback(this.handleCommit, response, options) │ │ │ │ │ + }); │ │ │ │ │ + return response │ │ │ │ │ }, │ │ │ │ │ - buildGeometryNode: function(geometry) { │ │ │ │ │ - if (!this.gmlParser) { │ │ │ │ │ - this.initGmlParser() │ │ │ │ │ + handleCommit: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + var data = request.responseXML; │ │ │ │ │ + if (!data || !data.documentElement) { │ │ │ │ │ + data = request.responseText │ │ │ │ │ + } │ │ │ │ │ + var obj = this.format.read(data) || {}; │ │ │ │ │ + response.insertIds = obj.insertIds || []; │ │ │ │ │ + if (obj.success) { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + response.error = obj │ │ │ │ │ + } │ │ │ │ │ + options.callback.call(options.scope, response) │ │ │ │ │ } │ │ │ │ │ - var node = this.gmlParser.writeNode("feature:_geometry", geometry); │ │ │ │ │ - return node.firstChild │ │ │ │ │ }, │ │ │ │ │ - buildPersonConstructNode: function(name, value) { │ │ │ │ │ - var oNames = ["uri", "email"]; │ │ │ │ │ - var personNode = this.createElementNSPlus("atom:" + name); │ │ │ │ │ - personNode.appendChild(this.createElementNSPlus("atom:name", { │ │ │ │ │ - value: value.name │ │ │ │ │ - })); │ │ │ │ │ - for (var i = 0, ii = oNames.length; i < ii; i++) { │ │ │ │ │ - if (value[oNames[i]]) { │ │ │ │ │ - personNode.appendChild(this.createElementNSPlus("atom:" + oNames[i], { │ │ │ │ │ - value: value[oNames[i]] │ │ │ │ │ - })) │ │ │ │ │ + filterDelete: function(filter, options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "commit" │ │ │ │ │ + }); │ │ │ │ │ + var root = this.format.createElementNSPlus("wfs:Transaction", { │ │ │ │ │ + attributes: { │ │ │ │ │ + service: "WFS", │ │ │ │ │ + version: this.version │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + var deleteNode = this.format.createElementNSPlus("wfs:Delete", { │ │ │ │ │ + attributes: { │ │ │ │ │ + typeName: (options.featureNS ? this.featurePrefix + ":" : "") + options.featureType │ │ │ │ │ } │ │ │ │ │ + }); │ │ │ │ │ + if (options.featureNS) { │ │ │ │ │ + deleteNode.setAttribute("xmlns:" + this.featurePrefix, options.featureNS) │ │ │ │ │ } │ │ │ │ │ - return personNode │ │ │ │ │ + var filterNode = this.format.writeNode("ogc:Filter", filter); │ │ │ │ │ + deleteNode.appendChild(filterNode); │ │ │ │ │ + root.appendChild(deleteNode); │ │ │ │ │ + var data = OpenLayers.Format.XML.prototype.write.apply(this.format, [root]); │ │ │ │ │ + return OpenLayers.Request.POST({ │ │ │ │ │ + url: this.url, │ │ │ │ │ + callback: options.callback || function() {}, │ │ │ │ │ + data: data │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - getFirstChildValue: function(node, nsuri, name, def) { │ │ │ │ │ - var value; │ │ │ │ │ - var nodes = this.getElementsByTagNameNS(node, nsuri, name); │ │ │ │ │ - if (nodes && nodes.length > 0) { │ │ │ │ │ - value = this.getChildValue(nodes[0], def) │ │ │ │ │ - } else { │ │ │ │ │ - value = def │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + response.priv.abort() │ │ │ │ │ } │ │ │ │ │ - return value │ │ │ │ │ }, │ │ │ │ │ - parseFeature: function(node) { │ │ │ │ │ - var atomAttrib = {}; │ │ │ │ │ - var value = null; │ │ │ │ │ - var nodes = null; │ │ │ │ │ - var attval = null; │ │ │ │ │ - var atomns = this.namespaces.atom; │ │ │ │ │ - this.parsePersonConstructs(node, "author", atomAttrib); │ │ │ │ │ - nodes = this.getElementsByTagNameNS(node, atomns, "category"); │ │ │ │ │ - if (nodes.length > 0) { │ │ │ │ │ - atomAttrib.categories = [] │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ - value = {}; │ │ │ │ │ - value.term = nodes[i].getAttribute("term"); │ │ │ │ │ - attval = nodes[i].getAttribute("scheme"); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value.scheme = attval │ │ │ │ │ - } │ │ │ │ │ - attval = nodes[i].getAttribute("label"); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value.label = attval │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.WFS.v1" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Protocol.WFS.v1_1_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ + version: "1.1.0", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Protocol.WFS.v1.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (this.outputFormat && !this.readFormat) { │ │ │ │ │ + if (this.outputFormat.toLowerCase() == "gml2") { │ │ │ │ │ + this.readFormat = new OpenLayers.Format.GML.v2({ │ │ │ │ │ + featureType: this.featureType, │ │ │ │ │ + featureNS: this.featureNS, │ │ │ │ │ + geometryName: this.geometryName │ │ │ │ │ + }) │ │ │ │ │ + } else if (this.outputFormat.toLowerCase() == "json") { │ │ │ │ │ + this.readFormat = new OpenLayers.Format.GeoJSON │ │ │ │ │ } │ │ │ │ │ - atomAttrib.categories.push(value) │ │ │ │ │ } │ │ │ │ │ - nodes = this.getElementsByTagNameNS(node, atomns, "content"); │ │ │ │ │ - if (nodes.length > 0) { │ │ │ │ │ - value = {}; │ │ │ │ │ - attval = nodes[0].getAttribute("type"); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value.type = attval │ │ │ │ │ - } │ │ │ │ │ - attval = nodes[0].getAttribute("src"); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value.src = attval │ │ │ │ │ - } else { │ │ │ │ │ - if (value.type == "text" || value.type == "html" || value.type == null) { │ │ │ │ │ - value.value = this.getFirstChildValue(node, atomns, "content", null) │ │ │ │ │ - } else if (value.type == "xhtml" || value.type.match(/(\+|\/)xml$/)) { │ │ │ │ │ - value.value = this.getChildEl(nodes[0]) │ │ │ │ │ - } else { │ │ │ │ │ - value.value = this.getFirstChildValue(node, atomns, "content", null) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.WFS.v1_1_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.GML.v2 = OpenLayers.Class(OpenLayers.Format.GML.Base, { │ │ │ │ │ + schemaLocation: "http://www.opengis.net/gml http://schemas.opengis.net/gml/2.1.2/feature.xsd", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.GML.Base.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + gml: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + outerBoundaryIs: function(node, container) { │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + container.outer = obj.components[0] │ │ │ │ │ + }, │ │ │ │ │ + innerBoundaryIs: function(node, container) { │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + container.inner.push(obj.components[0]) │ │ │ │ │ + }, │ │ │ │ │ + Box: function(node, container) { │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + if (!container.components) { │ │ │ │ │ + container.components = [] │ │ │ │ │ } │ │ │ │ │ - atomAttrib.content = value │ │ │ │ │ + var min = obj.points[0]; │ │ │ │ │ + var max = obj.points[1]; │ │ │ │ │ + container.components.push(new OpenLayers.Bounds(min.x, min.y, max.x, max.y)) │ │ │ │ │ } │ │ │ │ │ + }, OpenLayers.Format.GML.Base.prototype.readers["gml"]), │ │ │ │ │ + feature: OpenLayers.Format.GML.Base.prototype.readers["feature"], │ │ │ │ │ + wfs: OpenLayers.Format.GML.Base.prototype.readers["wfs"] │ │ │ │ │ + }, │ │ │ │ │ + write: function(features) { │ │ │ │ │ + var name; │ │ │ │ │ + if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ + name = "wfs:FeatureCollection" │ │ │ │ │ + } else { │ │ │ │ │ + name = "gml:featureMember" │ │ │ │ │ } │ │ │ │ │ - this.parsePersonConstructs(node, "contributor", atomAttrib); │ │ │ │ │ - atomAttrib.id = this.getFirstChildValue(node, atomns, "id", null); │ │ │ │ │ - nodes = this.getElementsByTagNameNS(node, atomns, "link"); │ │ │ │ │ - if (nodes.length > 0) { │ │ │ │ │ - atomAttrib.links = new Array(nodes.length) │ │ │ │ │ - } │ │ │ │ │ - var oAtts = ["rel", "type", "hreflang", "title", "length"]; │ │ │ │ │ - for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ - value = {}; │ │ │ │ │ - value.href = nodes[i].getAttribute("href"); │ │ │ │ │ - for (var j = 0, jj = oAtts.length; j < jj; j++) { │ │ │ │ │ - attval = nodes[i].getAttribute(oAtts[j]); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value[oAtts[j]] = attval │ │ │ │ │ + var root = this.writeNode(name, features); │ │ │ │ │ + this.setAttributeNS(root, this.namespaces["xsi"], "xsi:schemaLocation", this.schemaLocation); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [root]) │ │ │ │ │ + }, │ │ │ │ │ + writers: { │ │ │ │ │ + gml: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Point: function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:Point"); │ │ │ │ │ + this.writeNode("coordinates", [geometry], node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + coordinates: function(points) { │ │ │ │ │ + var numPoints = points.length; │ │ │ │ │ + var parts = new Array(numPoints); │ │ │ │ │ + var point; │ │ │ │ │ + for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ + point = points[i]; │ │ │ │ │ + if (this.xy) { │ │ │ │ │ + parts[i] = point.x + "," + point.y │ │ │ │ │ + } else { │ │ │ │ │ + parts[i] = point.y + "," + point.x │ │ │ │ │ + } │ │ │ │ │ + if (point.z != undefined) { │ │ │ │ │ + parts[i] += "," + point.z │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return this.createElementNSPlus("gml:coordinates", { │ │ │ │ │ + attributes: { │ │ │ │ │ + decimal: ".", │ │ │ │ │ + cs: ",", │ │ │ │ │ + ts: " " │ │ │ │ │ + }, │ │ │ │ │ + value: numPoints == 1 ? parts[0] : parts.join(" ") │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + LineString: function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:LineString"); │ │ │ │ │ + this.writeNode("coordinates", geometry.components, node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + Polygon: function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:Polygon"); │ │ │ │ │ + this.writeNode("outerBoundaryIs", geometry.components[0], node); │ │ │ │ │ + for (var i = 1; i < geometry.components.length; ++i) { │ │ │ │ │ + this.writeNode("innerBoundaryIs", geometry.components[i], node) │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + outerBoundaryIs: function(ring) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:outerBoundaryIs"); │ │ │ │ │ + this.writeNode("LinearRing", ring, node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + innerBoundaryIs: function(ring) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:innerBoundaryIs"); │ │ │ │ │ + this.writeNode("LinearRing", ring, node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + LinearRing: function(ring) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:LinearRing"); │ │ │ │ │ + this.writeNode("coordinates", ring.components, node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + Box: function(bounds) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:Box"); │ │ │ │ │ + this.writeNode("coordinates", [{ │ │ │ │ │ + x: bounds.left, │ │ │ │ │ + y: bounds.bottom │ │ │ │ │ + }, { │ │ │ │ │ + x: bounds.right, │ │ │ │ │ + y: bounds.top │ │ │ │ │ + }], node); │ │ │ │ │ + if (this.srsName) { │ │ │ │ │ + node.setAttribute("srsName", this.srsName) │ │ │ │ │ } │ │ │ │ │ + return node │ │ │ │ │ } │ │ │ │ │ - atomAttrib.links[i] = value │ │ │ │ │ - } │ │ │ │ │ - value = this.getFirstChildValue(node, atomns, "published", null); │ │ │ │ │ - if (value) { │ │ │ │ │ - atomAttrib.published = value │ │ │ │ │ - } │ │ │ │ │ - value = this.getFirstChildValue(node, atomns, "rights", null); │ │ │ │ │ - if (value) { │ │ │ │ │ - atomAttrib.rights = value │ │ │ │ │ - } │ │ │ │ │ - value = this.getFirstChildValue(node, atomns, "summary", null); │ │ │ │ │ - if (value) { │ │ │ │ │ - atomAttrib.summary = value │ │ │ │ │ - } │ │ │ │ │ - atomAttrib.title = this.getFirstChildValue(node, atomns, "title", null); │ │ │ │ │ - atomAttrib.updated = this.getFirstChildValue(node, atomns, "updated", null); │ │ │ │ │ - var featureAttrib = { │ │ │ │ │ - title: atomAttrib.title, │ │ │ │ │ - description: atomAttrib.summary, │ │ │ │ │ - atom: atomAttrib │ │ │ │ │ - }; │ │ │ │ │ - var geometry = this.parseLocations(node)[0]; │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry, featureAttrib); │ │ │ │ │ - feature.fid = atomAttrib.id; │ │ │ │ │ - return feature │ │ │ │ │ + }, OpenLayers.Format.GML.Base.prototype.writers["gml"]), │ │ │ │ │ + feature: OpenLayers.Format.GML.Base.prototype.writers["feature"], │ │ │ │ │ + wfs: OpenLayers.Format.GML.Base.prototype.writers["wfs"] │ │ │ │ │ }, │ │ │ │ │ - parseFeatures: function(node) { │ │ │ │ │ - var features = []; │ │ │ │ │ - var entries = this.getElementsByTagNameNS(node, this.namespaces.atom, "entry"); │ │ │ │ │ - if (entries.length == 0) { │ │ │ │ │ - entries = [node] │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, ii = entries.length; i < ii; i++) { │ │ │ │ │ - features.push(this.parseFeature(entries[i])) │ │ │ │ │ - } │ │ │ │ │ - return features │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.GML.v2" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.Filter.v1_0_0 = OpenLayers.Class(OpenLayers.Format.GML.v2, OpenLayers.Format.Filter.v1, { │ │ │ │ │ + VERSION: "1.0.0", │ │ │ │ │ + schemaLocation: "http://www.opengis.net/ogc/filter/1.0.0/filter.xsd", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.GML.v2.prototype.initialize.apply(this, [options]) │ │ │ │ │ }, │ │ │ │ │ - parseLocations: function(node) { │ │ │ │ │ - var georssns = this.namespaces.georss; │ │ │ │ │ - var locations = { │ │ │ │ │ - components: [] │ │ │ │ │ - }; │ │ │ │ │ - var where = this.getElementsByTagNameNS(node, georssns, "where"); │ │ │ │ │ - if (where && where.length > 0) { │ │ │ │ │ - if (!this.gmlParser) { │ │ │ │ │ - this.initGmlParser() │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, ii = where.length; i < ii; i++) { │ │ │ │ │ - this.gmlParser.readChildNodes(where[i], locations) │ │ │ │ │ + readers: { │ │ │ │ │ + ogc: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + PropertyIsEqualTo: function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.EQUAL_TO │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + obj.filters.push(filter) │ │ │ │ │ + }, │ │ │ │ │ + PropertyIsNotEqualTo: function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.NOT_EQUAL_TO │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + obj.filters.push(filter) │ │ │ │ │ + }, │ │ │ │ │ + PropertyIsLike: function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.LIKE │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + var wildCard = node.getAttribute("wildCard"); │ │ │ │ │ + var singleChar = node.getAttribute("singleChar"); │ │ │ │ │ + var esc = node.getAttribute("escape"); │ │ │ │ │ + filter.value2regex(wildCard, singleChar, esc); │ │ │ │ │ + obj.filters.push(filter) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - var components = locations.components; │ │ │ │ │ - var point = this.getElementsByTagNameNS(node, georssns, "point"); │ │ │ │ │ - if (point && point.length > 0) { │ │ │ │ │ - for (var i = 0, ii = point.length; i < ii; i++) { │ │ │ │ │ - var xy = OpenLayers.String.trim(point[i].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ - if (xy.length != 2) { │ │ │ │ │ - xy = OpenLayers.String.trim(point[i].firstChild.nodeValue).split(/\s*,\s*/) │ │ │ │ │ + }, OpenLayers.Format.Filter.v1.prototype.readers["ogc"]), │ │ │ │ │ + gml: OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ + feature: OpenLayers.Format.GML.v2.prototype.readers["feature"] │ │ │ │ │ + }, │ │ │ │ │ + writers: { │ │ │ │ │ + ogc: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + PropertyIsEqualTo: function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsEqualTo"); │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + this.writeOgcExpression(filter.value, node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + PropertyIsNotEqualTo: function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsNotEqualTo"); │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + this.writeOgcExpression(filter.value, node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + PropertyIsLike: function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsLike", { │ │ │ │ │ + attributes: { │ │ │ │ │ + wildCard: "*", │ │ │ │ │ + singleChar: ".", │ │ │ │ │ + escape: "!" │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + this.writeNode("Literal", filter.regex2value(), node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + BBOX: function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:BBOX"); │ │ │ │ │ + filter.property && this.writeNode("PropertyName", filter, node); │ │ │ │ │ + var box = this.writeNode("gml:Box", filter.value, node); │ │ │ │ │ + if (filter.projection) { │ │ │ │ │ + box.setAttribute("srsName", filter.projection) │ │ │ │ │ } │ │ │ │ │ - components.push(new OpenLayers.Geometry.Point(xy[1], xy[0])) │ │ │ │ │ + return node │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - var line = this.getElementsByTagNameNS(node, georssns, "line"); │ │ │ │ │ - if (line && line.length > 0) { │ │ │ │ │ - var coords; │ │ │ │ │ - var p; │ │ │ │ │ - var points; │ │ │ │ │ - for (var i = 0, ii = line.length; i < ii; i++) { │ │ │ │ │ - coords = OpenLayers.String.trim(line[i].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ - points = []; │ │ │ │ │ - for (var j = 0, jj = coords.length; j < jj; j += 2) { │ │ │ │ │ - p = new OpenLayers.Geometry.Point(coords[j + 1], coords[j]); │ │ │ │ │ - points.push(p) │ │ │ │ │ - } │ │ │ │ │ - components.push(new OpenLayers.Geometry.LineString(points)) │ │ │ │ │ + }, OpenLayers.Format.Filter.v1.prototype.writers["ogc"]), │ │ │ │ │ + gml: OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ + feature: OpenLayers.Format.GML.v2.prototype.writers["feature"] │ │ │ │ │ + }, │ │ │ │ │ + writeSpatial: function(filter, name) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:" + name); │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + if (filter.value instanceof OpenLayers.Filter.Function) { │ │ │ │ │ + this.writeNode("Function", filter.value, node) │ │ │ │ │ + } else { │ │ │ │ │ + var child; │ │ │ │ │ + if (filter.value instanceof OpenLayers.Geometry) { │ │ │ │ │ + child = this.writeNode("feature:_geometry", filter.value).firstChild │ │ │ │ │ + } else { │ │ │ │ │ + child = this.writeNode("gml:Box", filter.value) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - var polygon = this.getElementsByTagNameNS(node, georssns, "polygon"); │ │ │ │ │ - if (polygon && polygon.length > 0) { │ │ │ │ │ - var coords; │ │ │ │ │ - var p; │ │ │ │ │ - var points; │ │ │ │ │ - for (var i = 0, ii = polygon.length; i < ii; i++) { │ │ │ │ │ - coords = OpenLayers.String.trim(polygon[i].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ - points = []; │ │ │ │ │ - for (var j = 0, jj = coords.length; j < jj; j += 2) { │ │ │ │ │ - p = new OpenLayers.Geometry.Point(coords[j + 1], coords[j]); │ │ │ │ │ - points.push(p) │ │ │ │ │ - } │ │ │ │ │ - components.push(new OpenLayers.Geometry.Polygon([new OpenLayers.Geometry.LinearRing(points)])) │ │ │ │ │ + if (filter.projection) { │ │ │ │ │ + child.setAttribute("srsName", filter.projection) │ │ │ │ │ } │ │ │ │ │ + node.appendChild(child) │ │ │ │ │ } │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - for (var i = 0, ii = components.length; i < ii; i++) { │ │ │ │ │ - if (components[i]) { │ │ │ │ │ - components[i].transform(this.externalProjection, this.internalProjection) │ │ │ │ │ - } │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.Filter.v1_0_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WFST.v1_0_0 = OpenLayers.Class(OpenLayers.Format.Filter.v1_0_0, OpenLayers.Format.WFST.v1, { │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + srsNameInQuery: false, │ │ │ │ │ + schemaLocations: { │ │ │ │ │ + wfs: "http://schemas.opengis.net/wfs/1.0.0/WFS-transaction.xsd" │ │ │ │ │ + }, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.Filter.v1_0_0.prototype.initialize.apply(this, [options]); │ │ │ │ │ + OpenLayers.Format.WFST.v1.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ + readNode: function(node, obj, first) { │ │ │ │ │ + return OpenLayers.Format.GML.v2.prototype.readNode.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + wfs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + WFS_TransactionResponse: function(node, obj) { │ │ │ │ │ + obj.insertIds = []; │ │ │ │ │ + obj.success = false; │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + InsertResult: function(node, container) { │ │ │ │ │ + var obj = { │ │ │ │ │ + fids: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + container.insertIds = container.insertIds.concat(obj.fids) │ │ │ │ │ + }, │ │ │ │ │ + TransactionResult: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + Status: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + SUCCESS: function(node, obj) { │ │ │ │ │ + obj.success = true │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return components │ │ │ │ │ + }, OpenLayers.Format.WFST.v1.prototype.readers["wfs"]), │ │ │ │ │ + gml: OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ + feature: OpenLayers.Format.GML.v2.prototype.readers["feature"], │ │ │ │ │ + ogc: OpenLayers.Format.Filter.v1_0_0.prototype.readers["ogc"] │ │ │ │ │ }, │ │ │ │ │ - parsePersonConstructs: function(node, name, data) { │ │ │ │ │ - var persons = []; │ │ │ │ │ - var atomns = this.namespaces.atom; │ │ │ │ │ - var nodes = this.getElementsByTagNameNS(node, atomns, name); │ │ │ │ │ - var oAtts = ["uri", "email"]; │ │ │ │ │ - for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ - var value = {}; │ │ │ │ │ - value.name = this.getFirstChildValue(nodes[i], atomns, "name", null); │ │ │ │ │ - for (var j = 0, jj = oAtts.length; j < jj; j++) { │ │ │ │ │ - var attval = this.getFirstChildValue(nodes[i], atomns, oAtts[j], null); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value[oAtts[j]] = attval │ │ │ │ │ + writers: { │ │ │ │ │ + wfs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Query: function(options) { │ │ │ │ │ + options = OpenLayers.Util.extend({ │ │ │ │ │ + featureNS: this.featureNS, │ │ │ │ │ + featurePrefix: this.featurePrefix, │ │ │ │ │ + featureType: this.featureType, │ │ │ │ │ + srsName: this.srsName, │ │ │ │ │ + srsNameInQuery: this.srsNameInQuery │ │ │ │ │ + }, options); │ │ │ │ │ + var prefix = options.featurePrefix; │ │ │ │ │ + var node = this.createElementNSPlus("wfs:Query", { │ │ │ │ │ + attributes: { │ │ │ │ │ + typeName: (prefix ? prefix + ":" : "") + options.featureType │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (options.srsNameInQuery && options.srsName) { │ │ │ │ │ + node.setAttribute("srsName", options.srsName) │ │ │ │ │ + } │ │ │ │ │ + if (options.featureNS) { │ │ │ │ │ + node.setAttribute("xmlns:" + prefix, options.featureNS) │ │ │ │ │ + } │ │ │ │ │ + if (options.propertyNames) { │ │ │ │ │ + for (var i = 0, len = options.propertyNames.length; i < len; i++) { │ │ │ │ │ + this.writeNode("ogc:PropertyName", { │ │ │ │ │ + property: options.propertyNames[i] │ │ │ │ │ + }, node) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (options.filter) { │ │ │ │ │ + this.setFilterProperty(options.filter); │ │ │ │ │ + this.writeNode("ogc:Filter", options.filter, node) │ │ │ │ │ } │ │ │ │ │ + return node │ │ │ │ │ } │ │ │ │ │ - persons.push(value) │ │ │ │ │ - } │ │ │ │ │ - if (persons.length > 0) { │ │ │ │ │ - data[name + "s"] = persons │ │ │ │ │ - } │ │ │ │ │ + }, OpenLayers.Format.WFST.v1.prototype.writers["wfs"]), │ │ │ │ │ + gml: OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ + feature: OpenLayers.Format.GML.v2.prototype.writers["feature"], │ │ │ │ │ + ogc: OpenLayers.Format.Filter.v1_0_0.prototype.writers["ogc"] │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.Atom" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFST.v1_0_0" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.SOSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - defaultVersion: "1.0.0", │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SOSCapabilities" │ │ │ │ │ +OpenLayers.Protocol.WFS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.WFS.v1_0_0" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Format.SOSGetFeatureOfInterest = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ VERSION: "1.0.0", │ │ │ │ │ namespaces: { │ │ │ │ │ sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ gml: "http://www.opengis.net/gml", │ │ │ │ │ sa: "http://www.opengis.net/sampling/1.0", │ │ │ │ │ @@ -18765,728 +18460,2328 @@ │ │ │ │ │ }); │ │ │ │ │ return node │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.SOSGetFeatureOfInterest" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.WMSDescribeLayer = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - defaultVersion: "1.1.1", │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSDescribeLayer" │ │ │ │ │ +OpenLayers.Protocol.SOS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + fois: null, │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!options.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.SOSGetFeatureOfInterest(this.formatOptions) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.options && !this.options.format) { │ │ │ │ │ + this.format.destroy() │ │ │ │ │ + } │ │ │ │ │ + this.format = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ + }, │ │ │ │ │ + read: function(options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + var format = this.format; │ │ │ │ │ + var data = OpenLayers.Format.XML.prototype.write.apply(format, [format.writeNode("sos:GetFeatureOfInterest", { │ │ │ │ │ + fois: this.fois │ │ │ │ │ + })]); │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ + data: data │ │ │ │ │ + }); │ │ │ │ │ + return response │ │ │ │ │ + }, │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + response.features = this.parseFeatures(request); │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ + } │ │ │ │ │ + options.callback.call(options.scope, response) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + parseFeatures: function(request) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ + } │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ + return this.format.read(doc) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.SOS.v1_0_0" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.SOSGetObservation = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Format.CSWGetRecords = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, OpenLayers.Format.CSWGetRecords.DEFAULTS); │ │ │ │ │ + var cls = OpenLayers.Format.CSWGetRecords["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported CSWGetRecords version: " + options.version │ │ │ │ │ + } │ │ │ │ │ + return new cls(options) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Format.CSWGetRecords.DEFAULTS = { │ │ │ │ │ + version: "2.0.2" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Format.CSWGetRecords.v2_0_2 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ namespaces: { │ │ │ │ │ - ows: "http://www.opengis.net/ows", │ │ │ │ │ - gml: "http://www.opengis.net/gml", │ │ │ │ │ - sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ + csw: "http://www.opengis.net/cat/csw/2.0.2", │ │ │ │ │ + dc: "http://purl.org/dc/elements/1.1/", │ │ │ │ │ + dct: "http://purl.org/dc/terms/", │ │ │ │ │ + gmd: "http://www.isotc211.org/2005/gmd", │ │ │ │ │ + geonet: "http://www.fao.org/geonetwork", │ │ │ │ │ ogc: "http://www.opengis.net/ogc", │ │ │ │ │ - om: "http://www.opengis.net/om/1.0", │ │ │ │ │ - sa: "http://www.opengis.net/sampling/1.0", │ │ │ │ │ + ows: "http://www.opengis.net/ows", │ │ │ │ │ xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ }, │ │ │ │ │ + defaultPrefix: "csw", │ │ │ │ │ + version: "2.0.2", │ │ │ │ │ + schemaLocation: "http://www.opengis.net/cat/csw/2.0.2 http://schemas.opengis.net/csw/2.0.2/CSW-discovery.xsd", │ │ │ │ │ + requestId: null, │ │ │ │ │ + resultType: null, │ │ │ │ │ + outputFormat: null, │ │ │ │ │ + outputSchema: null, │ │ │ │ │ + startPosition: null, │ │ │ │ │ + maxRecords: null, │ │ │ │ │ + DistributedSearch: null, │ │ │ │ │ + ResponseHandler: null, │ │ │ │ │ + Query: null, │ │ │ │ │ regExes: { │ │ │ │ │ trimSpace: /^\s*|\s*$/g, │ │ │ │ │ removeSpace: /\s*/g, │ │ │ │ │ splitSpace: /\s+/, │ │ │ │ │ trimComma: /\s*,\s*/g │ │ │ │ │ }, │ │ │ │ │ - VERSION: "1.0.0", │ │ │ │ │ - schemaLocation: "http://www.opengis.net/sos/1.0 http://schemas.opengis.net/sos/1.0.0/sosGetObservation.xsd", │ │ │ │ │ - defaultPrefix: "sos", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ read: function(data) { │ │ │ │ │ if (typeof data == "string") { │ │ │ │ │ data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ } │ │ │ │ │ if (data && data.nodeType == 9) { │ │ │ │ │ data = data.documentElement │ │ │ │ │ } │ │ │ │ │ - var info = { │ │ │ │ │ - measurements: [], │ │ │ │ │ - observations: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readNode(data, info); │ │ │ │ │ - return info │ │ │ │ │ - }, │ │ │ │ │ - write: function(options) { │ │ │ │ │ - var node = this.writeNode("sos:GetObservation", options); │ │ │ │ │ - node.setAttribute("xmlns:om", this.namespaces.om); │ │ │ │ │ - node.setAttribute("xmlns:ogc", this.namespaces.ogc); │ │ │ │ │ - this.setAttributeNS(node, this.namespaces.xsi, "xsi:schemaLocation", this.schemaLocation); │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [node]) │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readNode(data, obj); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ readers: { │ │ │ │ │ - om: { │ │ │ │ │ - ObservationCollection: function(node, obj) { │ │ │ │ │ - obj.id = this.getAttributeNS(node, this.namespaces.gml, "id"); │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ + csw: { │ │ │ │ │ + GetRecordsResponse: function(node, obj) { │ │ │ │ │ + obj.records = []; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + var version = this.getAttributeNS(node, "", "version"); │ │ │ │ │ + if (version != "") { │ │ │ │ │ + obj.version = version │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - member: function(node, observationCollection) { │ │ │ │ │ - this.readChildNodes(node, observationCollection) │ │ │ │ │ + RequestId: function(node, obj) { │ │ │ │ │ + obj.RequestId = this.getChildValue(node) │ │ │ │ │ }, │ │ │ │ │ - Measurement: function(node, observationCollection) { │ │ │ │ │ - var measurement = {}; │ │ │ │ │ - observationCollection.measurements.push(measurement); │ │ │ │ │ - this.readChildNodes(node, measurement) │ │ │ │ │ + SearchStatus: function(node, obj) { │ │ │ │ │ + obj.SearchStatus = {}; │ │ │ │ │ + var timestamp = this.getAttributeNS(node, "", "timestamp"); │ │ │ │ │ + if (timestamp != "") { │ │ │ │ │ + obj.SearchStatus.timestamp = timestamp │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - Observation: function(node, observationCollection) { │ │ │ │ │ - var observation = {}; │ │ │ │ │ - observationCollection.observations.push(observation); │ │ │ │ │ - this.readChildNodes(node, observation) │ │ │ │ │ + SearchResults: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + var attrs = node.attributes; │ │ │ │ │ + var SearchResults = {}; │ │ │ │ │ + for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ + if (attrs[i].name == "numberOfRecordsMatched" || attrs[i].name == "numberOfRecordsReturned" || attrs[i].name == "nextRecord") { │ │ │ │ │ + SearchResults[attrs[i].name] = parseInt(attrs[i].nodeValue) │ │ │ │ │ + } else { │ │ │ │ │ + SearchResults[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + obj.SearchResults = SearchResults │ │ │ │ │ }, │ │ │ │ │ - samplingTime: function(node, measurement) { │ │ │ │ │ - var samplingTime = {}; │ │ │ │ │ - measurement.samplingTime = samplingTime; │ │ │ │ │ - this.readChildNodes(node, samplingTime) │ │ │ │ │ + SummaryRecord: function(node, obj) { │ │ │ │ │ + var record = { │ │ │ │ │ + type: "SummaryRecord" │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, record); │ │ │ │ │ + obj.records.push(record) │ │ │ │ │ }, │ │ │ │ │ - observedProperty: function(node, measurement) { │ │ │ │ │ - measurement.observedProperty = this.getAttributeNS(node, this.namespaces.xlink, "href"); │ │ │ │ │ - this.readChildNodes(node, measurement) │ │ │ │ │ + BriefRecord: function(node, obj) { │ │ │ │ │ + var record = { │ │ │ │ │ + type: "BriefRecord" │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, record); │ │ │ │ │ + obj.records.push(record) │ │ │ │ │ }, │ │ │ │ │ - procedure: function(node, measurement) { │ │ │ │ │ - measurement.procedure = this.getAttributeNS(node, this.namespaces.xlink, "href"); │ │ │ │ │ - this.readChildNodes(node, measurement) │ │ │ │ │ + DCMIRecord: function(node, obj) { │ │ │ │ │ + var record = { │ │ │ │ │ + type: "DCMIRecord" │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, record); │ │ │ │ │ + obj.records.push(record) │ │ │ │ │ }, │ │ │ │ │ - featureOfInterest: function(node, observation) { │ │ │ │ │ - var foi = { │ │ │ │ │ - features: [] │ │ │ │ │ + Record: function(node, obj) { │ │ │ │ │ + var record = { │ │ │ │ │ + type: "Record" │ │ │ │ │ }; │ │ │ │ │ - observation.fois = []; │ │ │ │ │ - observation.fois.push(foi); │ │ │ │ │ - this.readChildNodes(node, foi); │ │ │ │ │ - var features = []; │ │ │ │ │ - for (var i = 0, len = foi.features.length; i < len; i++) { │ │ │ │ │ - var feature = foi.features[i]; │ │ │ │ │ - features.push(new OpenLayers.Feature.Vector(feature.components[0], feature.attributes)) │ │ │ │ │ - } │ │ │ │ │ - foi.features = features │ │ │ │ │ + this.readChildNodes(node, record); │ │ │ │ │ + obj.records.push(record) │ │ │ │ │ }, │ │ │ │ │ - result: function(node, measurement) { │ │ │ │ │ - var result = {}; │ │ │ │ │ - measurement.result = result; │ │ │ │ │ - if (this.getChildValue(node) !== "") { │ │ │ │ │ - result.value = this.getChildValue(node); │ │ │ │ │ - result.uom = node.getAttribute("uom") │ │ │ │ │ - } else { │ │ │ │ │ - this.readChildNodes(node, result) │ │ │ │ │ + "*": function(node, obj) { │ │ │ │ │ + var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ + obj[name] = this.getChildValue(node) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + geonet: { │ │ │ │ │ + info: function(node, obj) { │ │ │ │ │ + var gninfo = {}; │ │ │ │ │ + this.readChildNodes(node, gninfo); │ │ │ │ │ + obj.gninfo = gninfo │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + dc: { │ │ │ │ │ + "*": function(node, obj) { │ │ │ │ │ + var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ + if (!OpenLayers.Util.isArray(obj[name])) { │ │ │ │ │ + obj[name] = [] │ │ │ │ │ + } │ │ │ │ │ + var dc_element = {}; │ │ │ │ │ + var attrs = node.attributes; │ │ │ │ │ + for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ + dc_element[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ + } │ │ │ │ │ + dc_element.value = this.getChildValue(node); │ │ │ │ │ + if (dc_element.value != "") { │ │ │ │ │ + obj[name].push(dc_element) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - sa: OpenLayers.Format.SOSGetFeatureOfInterest.prototype.readers.sa, │ │ │ │ │ - gml: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - TimeInstant: function(node, samplingTime) { │ │ │ │ │ - var timeInstant = {}; │ │ │ │ │ - samplingTime.timeInstant = timeInstant; │ │ │ │ │ - this.readChildNodes(node, timeInstant) │ │ │ │ │ - }, │ │ │ │ │ - timePosition: function(node, timeInstant) { │ │ │ │ │ - timeInstant.timePosition = this.getChildValue(node) │ │ │ │ │ + dct: { │ │ │ │ │ + "*": function(node, obj) { │ │ │ │ │ + var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ + if (!OpenLayers.Util.isArray(obj[name])) { │ │ │ │ │ + obj[name] = [] │ │ │ │ │ + } │ │ │ │ │ + obj[name].push(this.getChildValue(node)) │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.SOSGetFeatureOfInterest.prototype.readers.gml) │ │ │ │ │ + }, │ │ │ │ │ + ows: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + BoundingBox: function(node, obj) { │ │ │ │ │ + if (obj.bounds) { │ │ │ │ │ + obj.BoundingBox = [{ │ │ │ │ │ + crs: obj.projection, │ │ │ │ │ + value: [obj.bounds.left, obj.bounds.bottom, obj.bounds.right, obj.bounds.top] │ │ │ │ │ + }]; │ │ │ │ │ + delete obj.projection; │ │ │ │ │ + delete obj.bounds │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Format.OWSCommon.v1_0_0.prototype.readers["ows"]["BoundingBox"].apply(this, arguments) │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.OWSCommon.v1_0_0.prototype.readers["ows"]) │ │ │ │ │ + }, │ │ │ │ │ + write: function(options) { │ │ │ │ │ + var node = this.writeNode("csw:GetRecords", options); │ │ │ │ │ + node.setAttribute("xmlns:gmd", this.namespaces.gmd); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [node]) │ │ │ │ │ }, │ │ │ │ │ writers: { │ │ │ │ │ - sos: { │ │ │ │ │ - GetObservation: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("GetObservation", { │ │ │ │ │ + csw: { │ │ │ │ │ + GetRecords: function(options) { │ │ │ │ │ + if (!options) { │ │ │ │ │ + options = {} │ │ │ │ │ + } │ │ │ │ │ + var node = this.createElementNSPlus("csw:GetRecords", { │ │ │ │ │ attributes: { │ │ │ │ │ - version: this.VERSION, │ │ │ │ │ - service: "SOS" │ │ │ │ │ + service: "CSW", │ │ │ │ │ + version: this.version, │ │ │ │ │ + requestId: options.requestId || this.requestId, │ │ │ │ │ + resultType: options.resultType || this.resultType, │ │ │ │ │ + outputFormat: options.outputFormat || this.outputFormat, │ │ │ │ │ + outputSchema: options.outputSchema || this.outputSchema, │ │ │ │ │ + startPosition: options.startPosition || this.startPosition, │ │ │ │ │ + maxRecords: options.maxRecords || this.maxRecords │ │ │ │ │ } │ │ │ │ │ }); │ │ │ │ │ - this.writeNode("offering", options, node); │ │ │ │ │ - if (options.eventTime) { │ │ │ │ │ - this.writeNode("eventTime", options, node) │ │ │ │ │ - } │ │ │ │ │ - for (var procedure in options.procedures) { │ │ │ │ │ - this.writeNode("procedure", options.procedures[procedure], node) │ │ │ │ │ - } │ │ │ │ │ - for (var observedProperty in options.observedProperties) { │ │ │ │ │ - this.writeNode("observedProperty", options.observedProperties[observedProperty], node) │ │ │ │ │ - } │ │ │ │ │ - if (options.foi) { │ │ │ │ │ - this.writeNode("featureOfInterest", options.foi, node) │ │ │ │ │ - } │ │ │ │ │ - this.writeNode("responseFormat", options, node); │ │ │ │ │ - if (options.resultModel) { │ │ │ │ │ - this.writeNode("resultModel", options, node) │ │ │ │ │ + if (options.DistributedSearch || this.DistributedSearch) { │ │ │ │ │ + this.writeNode("csw:DistributedSearch", options.DistributedSearch || this.DistributedSearch, node) │ │ │ │ │ } │ │ │ │ │ - if (options.responseMode) { │ │ │ │ │ - this.writeNode("responseMode", options, node) │ │ │ │ │ + var ResponseHandler = options.ResponseHandler || this.ResponseHandler; │ │ │ │ │ + if (OpenLayers.Util.isArray(ResponseHandler) && ResponseHandler.length > 0) { │ │ │ │ │ + for (var i = 0, len = ResponseHandler.length; i < len; i++) { │ │ │ │ │ + this.writeNode("csw:ResponseHandler", ResponseHandler[i], node) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + this.writeNode("Query", options.Query || this.Query, node); │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - featureOfInterest: function(foi) { │ │ │ │ │ - var node = this.createElementNSPlus("featureOfInterest"); │ │ │ │ │ - this.writeNode("ObjectID", foi.objectId, node); │ │ │ │ │ + DistributedSearch: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:DistributedSearch", { │ │ │ │ │ + attributes: { │ │ │ │ │ + hopCount: options.hopCount │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - ObjectID: function(options) { │ │ │ │ │ - return this.createElementNSPlus("ObjectID", { │ │ │ │ │ - value: options │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - responseFormat: function(options) { │ │ │ │ │ - return this.createElementNSPlus("responseFormat", { │ │ │ │ │ - value: options.responseFormat │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - procedure: function(procedure) { │ │ │ │ │ - return this.createElementNSPlus("procedure", { │ │ │ │ │ - value: procedure │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - offering: function(options) { │ │ │ │ │ - return this.createElementNSPlus("offering", { │ │ │ │ │ - value: options.offering │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - observedProperty: function(observedProperty) { │ │ │ │ │ - return this.createElementNSPlus("observedProperty", { │ │ │ │ │ - value: observedProperty │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - eventTime: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("eventTime"); │ │ │ │ │ - if (options.eventTime === "latest") { │ │ │ │ │ - this.writeNode("ogc:TM_Equals", options, node) │ │ │ │ │ - } │ │ │ │ │ + ResponseHandler: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:ResponseHandler", { │ │ │ │ │ + value: options.value │ │ │ │ │ + }); │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - resultModel: function(options) { │ │ │ │ │ - return this.createElementNSPlus("resultModel", { │ │ │ │ │ - value: options.resultModel │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - responseMode: function(options) { │ │ │ │ │ - return this.createElementNSPlus("responseMode", { │ │ │ │ │ - value: options.responseMode │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - ogc: { │ │ │ │ │ - TM_Equals: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:TM_Equals"); │ │ │ │ │ - this.writeNode("ogc:PropertyName", { │ │ │ │ │ - property: "urn:ogc:data:time:iso8601" │ │ │ │ │ - }, node); │ │ │ │ │ - if (options.eventTime === "latest") { │ │ │ │ │ - this.writeNode("gml:TimeInstant", { │ │ │ │ │ - value: "latest" │ │ │ │ │ + Query: function(options) { │ │ │ │ │ + if (!options) { │ │ │ │ │ + options = {} │ │ │ │ │ + } │ │ │ │ │ + var node = this.createElementNSPlus("csw:Query", { │ │ │ │ │ + attributes: { │ │ │ │ │ + typeNames: options.typeNames || "csw:Record" │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + var ElementName = options.ElementName; │ │ │ │ │ + if (OpenLayers.Util.isArray(ElementName) && ElementName.length > 0) { │ │ │ │ │ + for (var i = 0, len = ElementName.length; i < len; i++) { │ │ │ │ │ + this.writeNode("csw:ElementName", ElementName[i], node) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this.writeNode("csw:ElementSetName", options.ElementSetName || { │ │ │ │ │ + value: "summary" │ │ │ │ │ }, node) │ │ │ │ │ } │ │ │ │ │ + if (options.Constraint) { │ │ │ │ │ + this.writeNode("csw:Constraint", options.Constraint, node) │ │ │ │ │ + } │ │ │ │ │ + if (options.SortBy) { │ │ │ │ │ + this.writeNode("ogc:SortBy", options.SortBy, node) │ │ │ │ │ + } │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - PropertyName: function(options) { │ │ │ │ │ - return this.createElementNSPlus("ogc:PropertyName", { │ │ │ │ │ - value: options.property │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - gml: { │ │ │ │ │ - TimeInstant: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:TimeInstant"); │ │ │ │ │ - this.writeNode("gml:timePosition", options, node); │ │ │ │ │ + ElementName: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:ElementName", { │ │ │ │ │ + value: options.value │ │ │ │ │ + }); │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - timePosition: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:timePosition", { │ │ │ │ │ + ElementSetName: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:ElementSetName", { │ │ │ │ │ + attributes: { │ │ │ │ │ + typeNames: options.typeNames │ │ │ │ │ + }, │ │ │ │ │ value: options.value │ │ │ │ │ }); │ │ │ │ │ return node │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SOSGetObservation" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.Context = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - layerOptions: null, │ │ │ │ │ - layerParams: null, │ │ │ │ │ - read: function(data, options) { │ │ │ │ │ - var context = OpenLayers.Format.XML.VersionedOGC.prototype.read.apply(this, arguments); │ │ │ │ │ - var map; │ │ │ │ │ - if (options && options.map) { │ │ │ │ │ - this.context = context; │ │ │ │ │ - if (options.map instanceof OpenLayers.Map) { │ │ │ │ │ - map = this.mergeContextToMap(context, options.map) │ │ │ │ │ - } else { │ │ │ │ │ - var mapOptions = options.map; │ │ │ │ │ - if (OpenLayers.Util.isElement(mapOptions) || typeof mapOptions == "string") { │ │ │ │ │ - mapOptions = { │ │ │ │ │ - div: mapOptions │ │ │ │ │ + }, │ │ │ │ │ + Constraint: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:Constraint", { │ │ │ │ │ + attributes: { │ │ │ │ │ + version: options.version │ │ │ │ │ } │ │ │ │ │ + }); │ │ │ │ │ + if (options.Filter) { │ │ │ │ │ + var format = new OpenLayers.Format.Filter({ │ │ │ │ │ + version: options.version │ │ │ │ │ + }); │ │ │ │ │ + node.appendChild(format.write(options.Filter)) │ │ │ │ │ + } else if (options.CqlText) { │ │ │ │ │ + var child = this.createElementNSPlus("CqlText", { │ │ │ │ │ + value: options.CqlText.value │ │ │ │ │ + }); │ │ │ │ │ + node.appendChild(child) │ │ │ │ │ } │ │ │ │ │ - map = this.contextToMap(context, mapOptions) │ │ │ │ │ + return node │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - map = context │ │ │ │ │ - } │ │ │ │ │ - return map │ │ │ │ │ + }, │ │ │ │ │ + ogc: OpenLayers.Format.Filter.v1_1_0.prototype.writers["ogc"] │ │ │ │ │ }, │ │ │ │ │ - getLayerFromContext: function(layerContext) { │ │ │ │ │ - var i, len; │ │ │ │ │ - var options = { │ │ │ │ │ - queryable: layerContext.queryable, │ │ │ │ │ - visibility: layerContext.visibility, │ │ │ │ │ - maxExtent: layerContext.maxExtent, │ │ │ │ │ - metadata: OpenLayers.Util.applyDefaults(layerContext.metadata, { │ │ │ │ │ - styles: layerContext.styles, │ │ │ │ │ - formats: layerContext.formats, │ │ │ │ │ - abstract: layerContext["abstract"], │ │ │ │ │ - dataURL: layerContext.dataURL │ │ │ │ │ - }), │ │ │ │ │ - numZoomLevels: layerContext.numZoomLevels, │ │ │ │ │ - units: layerContext.units, │ │ │ │ │ - isBaseLayer: layerContext.isBaseLayer, │ │ │ │ │ - opacity: layerContext.opacity, │ │ │ │ │ - displayInLayerSwitcher: layerContext.displayInLayerSwitcher, │ │ │ │ │ - singleTile: layerContext.singleTile, │ │ │ │ │ - tileSize: layerContext.tileSize ? new OpenLayers.Size(layerContext.tileSize.width, layerContext.tileSize.height) : undefined, │ │ │ │ │ - minScale: layerContext.minScale || layerContext.maxScaleDenominator, │ │ │ │ │ - maxScale: layerContext.maxScale || layerContext.minScaleDenominator, │ │ │ │ │ - srs: layerContext.srs, │ │ │ │ │ - dimensions: layerContext.dimensions, │ │ │ │ │ - metadataURL: layerContext.metadataURL │ │ │ │ │ - }; │ │ │ │ │ - if (this.layerOptions) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.layerOptions) │ │ │ │ │ - } │ │ │ │ │ - var params = { │ │ │ │ │ - layers: layerContext.name, │ │ │ │ │ - transparent: layerContext.transparent, │ │ │ │ │ - version: layerContext.version │ │ │ │ │ - }; │ │ │ │ │ - if (layerContext.formats && layerContext.formats.length > 0) { │ │ │ │ │ - params.format = layerContext.formats[0].value; │ │ │ │ │ - for (i = 0, len = layerContext.formats.length; i < len; i++) { │ │ │ │ │ - var format = layerContext.formats[i]; │ │ │ │ │ - if (format.current == true) { │ │ │ │ │ - params.format = format.value; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (layerContext.styles && layerContext.styles.length > 0) { │ │ │ │ │ - for (i = 0, len = layerContext.styles.length; i < len; i++) { │ │ │ │ │ - var style = layerContext.styles[i]; │ │ │ │ │ - if (style.current == true) { │ │ │ │ │ - if (style.href) { │ │ │ │ │ - params.sld = style.href │ │ │ │ │ - } else if (style.body) { │ │ │ │ │ - params.sld_body = style.body │ │ │ │ │ - } else { │ │ │ │ │ - params.styles = style.name │ │ │ │ │ - } │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.layerParams) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(params, this.layerParams) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.CSWGetRecords.v2_0_2" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Protocol.CSW.v2_0_2 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!options.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.CSWGetRecords.v2_0_2(OpenLayers.Util.extend({}, this.formatOptions)) │ │ │ │ │ } │ │ │ │ │ - var layer = null; │ │ │ │ │ - var service = layerContext.service; │ │ │ │ │ - if (service == OpenLayers.Format.Context.serviceTypes.WFS) { │ │ │ │ │ - options.strategies = [new OpenLayers.Strategy.BBOX]; │ │ │ │ │ - options.protocol = new OpenLayers.Protocol.WFS({ │ │ │ │ │ - url: layerContext.url, │ │ │ │ │ - featurePrefix: layerContext.name.split(":")[0], │ │ │ │ │ - featureType: layerContext.name.split(":").pop() │ │ │ │ │ - }); │ │ │ │ │ - layer = new OpenLayers.Layer.Vector(layerContext.title || layerContext.name, options) │ │ │ │ │ - } else if (service == OpenLayers.Format.Context.serviceTypes.KML) { │ │ │ │ │ - options.strategies = [new OpenLayers.Strategy.Fixed]; │ │ │ │ │ - options.protocol = new OpenLayers.Protocol.HTTP({ │ │ │ │ │ - url: layerContext.url, │ │ │ │ │ - format: new OpenLayers.Format.KML │ │ │ │ │ - }); │ │ │ │ │ - layer = new OpenLayers.Layer.Vector(layerContext.title || layerContext.name, options) │ │ │ │ │ - } else if (service == OpenLayers.Format.Context.serviceTypes.GML) { │ │ │ │ │ - options.strategies = [new OpenLayers.Strategy.Fixed]; │ │ │ │ │ - options.protocol = new OpenLayers.Protocol.HTTP({ │ │ │ │ │ - url: layerContext.url, │ │ │ │ │ - format: new OpenLayers.Format.GML │ │ │ │ │ - }); │ │ │ │ │ - layer = new OpenLayers.Layer.Vector(layerContext.title || layerContext.name, options) │ │ │ │ │ - } else if (layerContext.features) { │ │ │ │ │ - layer = new OpenLayers.Layer.Vector(layerContext.title || layerContext.name, options); │ │ │ │ │ - layer.addFeatures(layerContext.features) │ │ │ │ │ - } else if (layerContext.categoryLayer !== true) { │ │ │ │ │ - layer = new OpenLayers.Layer.WMS(layerContext.title || layerContext.name, layerContext.url, params, options) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.options && !this.options.format) { │ │ │ │ │ + this.format.destroy() │ │ │ │ │ } │ │ │ │ │ - return layer │ │ │ │ │ + this.format = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ }, │ │ │ │ │ - getLayersFromContext: function(layersContext) { │ │ │ │ │ - var layers = []; │ │ │ │ │ - for (var i = 0, len = layersContext.length; i < len; i++) { │ │ │ │ │ - var layer = this.getLayerFromContext(layersContext[i]); │ │ │ │ │ - if (layer !== null) { │ │ │ │ │ - layers.push(layer) │ │ │ │ │ + read: function(options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + var data = this.format.write(options.params || options); │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: data │ │ │ │ │ + }); │ │ │ │ │ + return response │ │ │ │ │ + }, │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + response.data = this.parseData(request); │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ } │ │ │ │ │ + options.callback.call(options.scope, response) │ │ │ │ │ } │ │ │ │ │ - return layers │ │ │ │ │ }, │ │ │ │ │ - contextToMap: function(context, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - maxExtent: context.maxExtent, │ │ │ │ │ - projection: context.projection, │ │ │ │ │ - units: context.units │ │ │ │ │ - }, options); │ │ │ │ │ - if (options.maxExtent) { │ │ │ │ │ - options.maxResolution = options.maxExtent.getWidth() / OpenLayers.Map.TILE_WIDTH │ │ │ │ │ + parseData: function(request) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ } │ │ │ │ │ - var metadata = { │ │ │ │ │ - contactInformation: context.contactInformation, │ │ │ │ │ - abstract: context["abstract"], │ │ │ │ │ - keywords: context.keywords, │ │ │ │ │ - logo: context.logo, │ │ │ │ │ - descriptionURL: context.descriptionURL │ │ │ │ │ - }; │ │ │ │ │ - options.metadata = metadata; │ │ │ │ │ - var map = new OpenLayers.Map(options); │ │ │ │ │ - map.addLayers(this.getLayersFromContext(context.layersContext)); │ │ │ │ │ - map.setCenter(context.bounds.getCenterLonLat(), map.getZoomForExtent(context.bounds, true)); │ │ │ │ │ - return map │ │ │ │ │ - }, │ │ │ │ │ - mergeContextToMap: function(context, map) { │ │ │ │ │ - map.addLayers(this.getLayersFromContext(context.layersContext)); │ │ │ │ │ - return map │ │ │ │ │ - }, │ │ │ │ │ - write: function(obj, options) { │ │ │ │ │ - obj = this.toContext(obj); │ │ │ │ │ - return OpenLayers.Format.XML.VersionedOGC.prototype.write.apply(this, arguments) │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ + return this.format.read(doc) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.Context" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.CSW.v2_0_2" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.Context.serviceTypes = { │ │ │ │ │ - WMS: "urn:ogc:serviceType:WMS", │ │ │ │ │ - WFS: "urn:ogc:serviceType:WFS", │ │ │ │ │ - WCS: "urn:ogc:serviceType:WCS", │ │ │ │ │ - GML: "urn:ogc:serviceType:GML", │ │ │ │ │ - SLD: "urn:ogc:serviceType:SLD", │ │ │ │ │ - FES: "urn:ogc:serviceType:FES", │ │ │ │ │ - KML: "urn:ogc:serviceType:KML" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Format.WMC = OpenLayers.Class(OpenLayers.Format.Context, { │ │ │ │ │ +OpenLayers.Format.WCSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ defaultVersion: "1.1.0", │ │ │ │ │ - layerToContext: function(layer) { │ │ │ │ │ - var parser = this.getParser(); │ │ │ │ │ - var layerContext = { │ │ │ │ │ - queryable: layer.queryable, │ │ │ │ │ - visibility: layer.visibility, │ │ │ │ │ - name: layer.params["LAYERS"], │ │ │ │ │ - title: layer.name, │ │ │ │ │ - abstract: layer.metadata["abstract"], │ │ │ │ │ - dataURL: layer.metadata.dataURL, │ │ │ │ │ - metadataURL: layer.metadataURL, │ │ │ │ │ - server: { │ │ │ │ │ - version: layer.params["VERSION"], │ │ │ │ │ - url: layer.url │ │ │ │ │ - }, │ │ │ │ │ - maxExtent: layer.maxExtent, │ │ │ │ │ - transparent: layer.params["TRANSPARENT"], │ │ │ │ │ - numZoomLevels: layer.numZoomLevels, │ │ │ │ │ - units: layer.units, │ │ │ │ │ - isBaseLayer: layer.isBaseLayer, │ │ │ │ │ - opacity: layer.opacity == 1 ? undefined : layer.opacity, │ │ │ │ │ - displayInLayerSwitcher: layer.displayInLayerSwitcher, │ │ │ │ │ - singleTile: layer.singleTile, │ │ │ │ │ - tileSize: layer.singleTile || !layer.tileSize ? undefined : { │ │ │ │ │ - width: layer.tileSize.w, │ │ │ │ │ - height: layer.tileSize.h │ │ │ │ │ - }, │ │ │ │ │ - minScale: layer.options.resolutions || layer.options.scales || layer.options.maxResolution || layer.options.minScale ? layer.minScale : undefined, │ │ │ │ │ - maxScale: layer.options.resolutions || layer.options.scales || layer.options.minResolution || layer.options.maxScale ? layer.maxScale : undefined, │ │ │ │ │ - formats: [], │ │ │ │ │ - styles: [], │ │ │ │ │ - srs: layer.srs, │ │ │ │ │ - dimensions: layer.dimensions │ │ │ │ │ - }; │ │ │ │ │ - if (layer.metadata.servertitle) { │ │ │ │ │ - layerContext.server.title = layer.metadata.servertitle │ │ │ │ │ - } │ │ │ │ │ - if (layer.metadata.formats && layer.metadata.formats.length > 0) { │ │ │ │ │ - for (var i = 0, len = layer.metadata.formats.length; i < len; i++) { │ │ │ │ │ - var format = layer.metadata.formats[i]; │ │ │ │ │ - layerContext.formats.push({ │ │ │ │ │ - value: format.value, │ │ │ │ │ - current: format.value == layer.params["FORMAT"] │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - layerContext.formats.push({ │ │ │ │ │ - value: layer.params["FORMAT"], │ │ │ │ │ - current: true │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - if (layer.metadata.styles && layer.metadata.styles.length > 0) { │ │ │ │ │ - for (var i = 0, len = layer.metadata.styles.length; i < len; i++) { │ │ │ │ │ - var style = layer.metadata.styles[i]; │ │ │ │ │ - if (style.href == layer.params["SLD"] || style.body == layer.params["SLD_BODY"] || style.name == layer.params["STYLES"]) { │ │ │ │ │ - style.current = true │ │ │ │ │ - } else { │ │ │ │ │ - style.current = false │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WCSCapabilities" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + defaultVersion: "1.1.1", │ │ │ │ │ + profile: null, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.ArcXML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + fontStyleKeys: ["antialiasing", "blockout", "font", "fontcolor", "fontsize", "fontstyle", "glowing", "interval", "outline", "printmode", "shadow", "transparency"], │ │ │ │ │ + request: null, │ │ │ │ │ + response: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.request = new OpenLayers.Format.ArcXML.Request; │ │ │ │ │ + this.response = new OpenLayers.Format.ArcXML.Response; │ │ │ │ │ + if (options) { │ │ │ │ │ + if (options.requesttype == "feature") { │ │ │ │ │ + this.request.get_image = null; │ │ │ │ │ + var qry = this.request.get_feature.query; │ │ │ │ │ + this.addCoordSys(qry.featurecoordsys, options.featureCoordSys); │ │ │ │ │ + this.addCoordSys(qry.filtercoordsys, options.filterCoordSys); │ │ │ │ │ + if (options.polygon) { │ │ │ │ │ + qry.isspatial = true; │ │ │ │ │ + qry.spatialfilter.polygon = options.polygon │ │ │ │ │ + } else if (options.envelope) { │ │ │ │ │ + qry.isspatial = true; │ │ │ │ │ + qry.spatialfilter.envelope = { │ │ │ │ │ + minx: 0, │ │ │ │ │ + miny: 0, │ │ │ │ │ + maxx: 0, │ │ │ │ │ + maxy: 0 │ │ │ │ │ + }; │ │ │ │ │ + this.parseEnvelope(qry.spatialfilter.envelope, options.envelope) │ │ │ │ │ } │ │ │ │ │ - layerContext.styles.push(style) │ │ │ │ │ + } else if (options.requesttype == "image") { │ │ │ │ │ + this.request.get_feature = null; │ │ │ │ │ + var props = this.request.get_image.properties; │ │ │ │ │ + this.parseEnvelope(props.envelope, options.envelope); │ │ │ │ │ + this.addLayers(props.layerlist, options.layers); │ │ │ │ │ + this.addImageSize(props.imagesize, options.tileSize); │ │ │ │ │ + this.addCoordSys(props.featurecoordsys, options.featureCoordSys); │ │ │ │ │ + this.addCoordSys(props.filtercoordsys, options.filterCoordSys) │ │ │ │ │ + } else { │ │ │ │ │ + this.request = null │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - layerContext.styles.push({ │ │ │ │ │ - href: layer.params["SLD"], │ │ │ │ │ - body: layer.params["SLD_BODY"], │ │ │ │ │ - name: layer.params["STYLES"] || parser.defaultStyleName, │ │ │ │ │ - title: parser.defaultStyleTitle, │ │ │ │ │ - current: true │ │ │ │ │ - }) │ │ │ │ │ } │ │ │ │ │ - return layerContext │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ }, │ │ │ │ │ - toContext: function(obj) { │ │ │ │ │ - var context = {}; │ │ │ │ │ - var layers = obj.layers; │ │ │ │ │ - if (obj.CLASS_NAME == "OpenLayers.Map") { │ │ │ │ │ - var metadata = obj.metadata || {}; │ │ │ │ │ - context.size = obj.getSize(); │ │ │ │ │ - context.bounds = obj.getExtent(); │ │ │ │ │ - context.projection = obj.projection; │ │ │ │ │ - context.title = obj.title; │ │ │ │ │ - context.keywords = metadata.keywords; │ │ │ │ │ - context["abstract"] = metadata["abstract"]; │ │ │ │ │ - context.logo = metadata.logo; │ │ │ │ │ - context.descriptionURL = metadata.descriptionURL; │ │ │ │ │ - context.contactInformation = metadata.contactInformation; │ │ │ │ │ - context.maxExtent = obj.maxExtent │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Util.applyDefaults(context, obj); │ │ │ │ │ - if (context.layers != undefined) { │ │ │ │ │ - delete context.layers │ │ │ │ │ - } │ │ │ │ │ + parseEnvelope: function(env, arr) { │ │ │ │ │ + if (arr && arr.length == 4) { │ │ │ │ │ + env.minx = arr[0]; │ │ │ │ │ + env.miny = arr[1]; │ │ │ │ │ + env.maxx = arr[2]; │ │ │ │ │ + env.maxy = arr[3] │ │ │ │ │ } │ │ │ │ │ - if (context.layersContext == undefined) { │ │ │ │ │ - context.layersContext = [] │ │ │ │ │ + }, │ │ │ │ │ + addLayers: function(ll, lyrs) { │ │ │ │ │ + for (var lind = 0, len = lyrs.length; lind < len; lind++) { │ │ │ │ │ + ll.push(lyrs[lind]) │ │ │ │ │ } │ │ │ │ │ - if (layers != undefined && OpenLayers.Util.isArray(layers)) { │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - var layer = layers[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.WMS) { │ │ │ │ │ - context.layersContext.push(this.layerToContext(layer)) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ + addImageSize: function(imsize, olsize) { │ │ │ │ │ + if (olsize !== null) { │ │ │ │ │ + imsize.width = olsize.w; │ │ │ │ │ + imsize.height = olsize.h; │ │ │ │ │ + imsize.printwidth = olsize.w; │ │ │ │ │ + imsize.printheight = olsize.h │ │ │ │ │ } │ │ │ │ │ - return context │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMC" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.OSM = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - checkTags: false, │ │ │ │ │ - interestingTagsExclude: null, │ │ │ │ │ - areaTags: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - var layer_defaults = { │ │ │ │ │ - interestingTagsExclude: ["source", "source_ref", "source:ref", "history", "attribution", "created_by"], │ │ │ │ │ - areaTags: ["area", "building", "leisure", "tourism", "ruins", "historic", "landuse", "military", "natural", "sport"] │ │ │ │ │ - }; │ │ │ │ │ - layer_defaults = OpenLayers.Util.extend(layer_defaults, options); │ │ │ │ │ - var interesting = {}; │ │ │ │ │ - for (var i = 0; i < layer_defaults.interestingTagsExclude.length; i++) { │ │ │ │ │ - interesting[layer_defaults.interestingTagsExclude[i]] = true │ │ │ │ │ + addCoordSys: function(featOrFilt, fsys) { │ │ │ │ │ + if (typeof fsys == "string") { │ │ │ │ │ + featOrFilt.id = parseInt(fsys); │ │ │ │ │ + featOrFilt.string = fsys │ │ │ │ │ + } else if (typeof fsys == "object" && fsys.proj !== null) { │ │ │ │ │ + featOrFilt.id = fsys.proj.srsProjNumber; │ │ │ │ │ + featOrFilt.string = fsys.proj.srsCode │ │ │ │ │ + } else { │ │ │ │ │ + featOrFilt = fsys │ │ │ │ │ } │ │ │ │ │ - layer_defaults.interestingTagsExclude = interesting; │ │ │ │ │ - var area = {}; │ │ │ │ │ - for (var i = 0; i < layer_defaults.areaTags.length; i++) { │ │ │ │ │ - area[layer_defaults.areaTags[i]] = true │ │ │ │ │ + }, │ │ │ │ │ + iserror: function(data) { │ │ │ │ │ + var ret = null; │ │ │ │ │ + if (!data) { │ │ │ │ │ + ret = this.response.error !== "" │ │ │ │ │ + } else { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + var errorNodes = data.documentElement.getElementsByTagName("ERROR"); │ │ │ │ │ + ret = errorNodes !== null && errorNodes.length > 0 │ │ │ │ │ } │ │ │ │ │ - layer_defaults.areaTags = area; │ │ │ │ │ - this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [layer_defaults]) │ │ │ │ │ + return ret │ │ │ │ │ }, │ │ │ │ │ - read: function(doc) { │ │ │ │ │ - if (typeof doc == "string") { │ │ │ │ │ - doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]) │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ } │ │ │ │ │ - var nodes = this.getNodes(doc); │ │ │ │ │ - var ways = this.getWays(doc); │ │ │ │ │ - var feat_list = new Array(ways.length); │ │ │ │ │ - for (var i = 0; i < ways.length; i++) { │ │ │ │ │ - var point_list = new Array(ways[i].nodes.length); │ │ │ │ │ - var poly = this.isWayArea(ways[i]) ? 1 : 0; │ │ │ │ │ - for (var j = 0; j < ways[i].nodes.length; j++) { │ │ │ │ │ - var node = nodes[ways[i].nodes[j]]; │ │ │ │ │ - var point = new OpenLayers.Geometry.Point(node.lon, node.lat); │ │ │ │ │ - point.osm_id = parseInt(ways[i].nodes[j]); │ │ │ │ │ - point_list[j] = point; │ │ │ │ │ - node.used = true │ │ │ │ │ - } │ │ │ │ │ - var geometry = null; │ │ │ │ │ - if (poly) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.Polygon(new OpenLayers.Geometry.LinearRing(point_list)) │ │ │ │ │ + var arcNode = null; │ │ │ │ │ + if (data && data.documentElement) { │ │ │ │ │ + if (data.documentElement.nodeName == "ARCXML") { │ │ │ │ │ + arcNode = data.documentElement │ │ │ │ │ } else { │ │ │ │ │ - geometry = new OpenLayers.Geometry.LineString(point_list) │ │ │ │ │ - } │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ - } │ │ │ │ │ - var feat = new OpenLayers.Feature.Vector(geometry, ways[i].tags); │ │ │ │ │ - feat.osm_id = parseInt(ways[i].id); │ │ │ │ │ - feat.fid = "way." + feat.osm_id; │ │ │ │ │ - feat_list[i] = feat │ │ │ │ │ - } │ │ │ │ │ - for (var node_id in nodes) { │ │ │ │ │ - var node = nodes[node_id]; │ │ │ │ │ - if (!node.used || this.checkTags) { │ │ │ │ │ - var tags = null; │ │ │ │ │ - if (this.checkTags) { │ │ │ │ │ - var result = this.getTags(node.node, true); │ │ │ │ │ - if (node.used && !result[1]) { │ │ │ │ │ - continue │ │ │ │ │ - } │ │ │ │ │ - tags = result[0] │ │ │ │ │ - } else { │ │ │ │ │ - tags = this.getTags(node.node) │ │ │ │ │ - } │ │ │ │ │ - var feat = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Point(node["lon"], node["lat"]), tags); │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - feat.geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ - } │ │ │ │ │ - feat.osm_id = parseInt(node_id); │ │ │ │ │ - feat.fid = "node." + feat.osm_id; │ │ │ │ │ - feat_list.push(feat) │ │ │ │ │ + arcNode = data.documentElement.getElementsByTagName("ARCXML")[0] │ │ │ │ │ } │ │ │ │ │ - node.node = null │ │ │ │ │ } │ │ │ │ │ - return feat_list │ │ │ │ │ - }, │ │ │ │ │ - getNodes: function(doc) { │ │ │ │ │ - var node_list = doc.getElementsByTagName("node"); │ │ │ │ │ - var nodes = {}; │ │ │ │ │ - for (var i = 0; i < node_list.length; i++) { │ │ │ │ │ - var node = node_list[i]; │ │ │ │ │ - var id = node.getAttribute("id"); │ │ │ │ │ - nodes[id] = { │ │ │ │ │ - lat: node.getAttribute("lat"), │ │ │ │ │ - lon: node.getAttribute("lon"), │ │ │ │ │ - node: node │ │ │ │ │ + if (!arcNode || arcNode.firstChild.nodeName === "parsererror") { │ │ │ │ │ + var error, source; │ │ │ │ │ + try { │ │ │ │ │ + error = data.firstChild.nodeValue; │ │ │ │ │ + source = data.firstChild.childNodes[1].firstChild.nodeValue │ │ │ │ │ + } catch (err) {} │ │ │ │ │ + throw { │ │ │ │ │ + message: "Error parsing the ArcXML request", │ │ │ │ │ + error: error, │ │ │ │ │ + source: source │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return nodes │ │ │ │ │ + var response = this.parseResponse(arcNode); │ │ │ │ │ + return response │ │ │ │ │ }, │ │ │ │ │ - getWays: function(doc) { │ │ │ │ │ - var way_list = doc.getElementsByTagName("way"); │ │ │ │ │ - var return_ways = []; │ │ │ │ │ - for (var i = 0; i < way_list.length; i++) { │ │ │ │ │ - var way = way_list[i]; │ │ │ │ │ - var way_object = { │ │ │ │ │ - id: way.getAttribute("id") │ │ │ │ │ - }; │ │ │ │ │ - way_object.tags = this.getTags(way); │ │ │ │ │ - var node_list = way.getElementsByTagName("nd"); │ │ │ │ │ - way_object.nodes = new Array(node_list.length); │ │ │ │ │ - for (var j = 0; j < node_list.length; j++) { │ │ │ │ │ - way_object.nodes[j] = node_list[j].getAttribute("ref") │ │ │ │ │ - } │ │ │ │ │ - return_ways.push(way_object) │ │ │ │ │ + write: function(request) { │ │ │ │ │ + if (!request) { │ │ │ │ │ + request = this.request │ │ │ │ │ } │ │ │ │ │ - return return_ways │ │ │ │ │ - }, │ │ │ │ │ - getTags: function(dom_node, interesting_tags) { │ │ │ │ │ - var tag_list = dom_node.getElementsByTagName("tag"); │ │ │ │ │ - var tags = {}; │ │ │ │ │ - var interesting = false; │ │ │ │ │ - for (var j = 0; j < tag_list.length; j++) { │ │ │ │ │ - var key = tag_list[j].getAttribute("k"); │ │ │ │ │ - tags[key] = tag_list[j].getAttribute("v"); │ │ │ │ │ - if (interesting_tags) { │ │ │ │ │ - if (!this.interestingTagsExclude[key]) { │ │ │ │ │ - interesting = true │ │ │ │ │ + var root = this.createElementNS("", "ARCXML"); │ │ │ │ │ + root.setAttribute("version", "1.1"); │ │ │ │ │ + var reqElem = this.createElementNS("", "REQUEST"); │ │ │ │ │ + if (request.get_image != null) { │ │ │ │ │ + var getElem = this.createElementNS("", "GET_IMAGE"); │ │ │ │ │ + reqElem.appendChild(getElem); │ │ │ │ │ + var propElem = this.createElementNS("", "PROPERTIES"); │ │ │ │ │ + getElem.appendChild(propElem); │ │ │ │ │ + var props = request.get_image.properties; │ │ │ │ │ + if (props.featurecoordsys != null) { │ │ │ │ │ + var feat = this.createElementNS("", "FEATURECOORDSYS"); │ │ │ │ │ + propElem.appendChild(feat); │ │ │ │ │ + if (props.featurecoordsys.id === 0) { │ │ │ │ │ + feat.setAttribute("string", props.featurecoordsys["string"]) │ │ │ │ │ + } else { │ │ │ │ │ + feat.setAttribute("id", props.featurecoordsys.id) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return interesting_tags ? [tags, interesting] : tags │ │ │ │ │ - }, │ │ │ │ │ - isWayArea: function(way) { │ │ │ │ │ - var poly_shaped = false; │ │ │ │ │ - var poly_tags = false; │ │ │ │ │ - if (way.nodes[0] == way.nodes[way.nodes.length - 1]) { │ │ │ │ │ - poly_shaped = true │ │ │ │ │ - } │ │ │ │ │ - if (this.checkTags) { │ │ │ │ │ - for (var key in way.tags) { │ │ │ │ │ - if (this.areaTags[key]) { │ │ │ │ │ - poly_tags = true; │ │ │ │ │ - break │ │ │ │ │ + if (props.filtercoordsys != null) { │ │ │ │ │ + var filt = this.createElementNS("", "FILTERCOORDSYS"); │ │ │ │ │ + propElem.appendChild(filt); │ │ │ │ │ + if (props.filtercoordsys.id === 0) { │ │ │ │ │ + filt.setAttribute("string", props.filtercoordsys.string) │ │ │ │ │ + } else { │ │ │ │ │ + filt.setAttribute("id", props.filtercoordsys.id) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return poly_shaped && (this.checkTags ? poly_tags : true) │ │ │ │ │ - }, │ │ │ │ │ - write: function(features) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ - features = [features] │ │ │ │ │ - } │ │ │ │ │ - this.osm_id = 1; │ │ │ │ │ - this.created_nodes = {}; │ │ │ │ │ - var root_node = this.createElementNS(null, "osm"); │ │ │ │ │ - root_node.setAttribute("version", "0.5"); │ │ │ │ │ - root_node.setAttribute("generator", "OpenLayers " + OpenLayers.VERSION_NUMBER); │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - var nodes = this.createFeatureNodes(features[i]); │ │ │ │ │ - for (var j = 0; j < nodes.length; j++) { │ │ │ │ │ - root_node.appendChild(nodes[j]) │ │ │ │ │ + if (props.envelope != null) { │ │ │ │ │ + var env = this.createElementNS("", "ENVELOPE"); │ │ │ │ │ + propElem.appendChild(env); │ │ │ │ │ + env.setAttribute("minx", props.envelope.minx); │ │ │ │ │ + env.setAttribute("miny", props.envelope.miny); │ │ │ │ │ + env.setAttribute("maxx", props.envelope.maxx); │ │ │ │ │ + env.setAttribute("maxy", props.envelope.maxy) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [root_node]) │ │ │ │ │ - }, │ │ │ │ │ - createFeatureNodes: function(feature) { │ │ │ │ │ - var nodes = []; │ │ │ │ │ - var className = feature.geometry.CLASS_NAME; │ │ │ │ │ - var type = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ - type = type.toLowerCase(); │ │ │ │ │ - var builder = this.createXML[type]; │ │ │ │ │ - if (builder) { │ │ │ │ │ - nodes = builder.apply(this, [feature]) │ │ │ │ │ - } │ │ │ │ │ - return nodes │ │ │ │ │ - }, │ │ │ │ │ - createXML: { │ │ │ │ │ - point: function(point) { │ │ │ │ │ - var id = null; │ │ │ │ │ - var geometry = point.geometry ? point.geometry : point; │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry = geometry.clone(); │ │ │ │ │ - geometry.transform(this.internalProjection, this.externalProjection) │ │ │ │ │ + var imagesz = this.createElementNS("", "IMAGESIZE"); │ │ │ │ │ + propElem.appendChild(imagesz); │ │ │ │ │ + imagesz.setAttribute("height", props.imagesize.height); │ │ │ │ │ + imagesz.setAttribute("width", props.imagesize.width); │ │ │ │ │ + if (props.imagesize.height != props.imagesize.printheight || props.imagesize.width != props.imagesize.printwidth) { │ │ │ │ │ + imagesz.setAttribute("printheight", props.imagesize.printheight); │ │ │ │ │ + imagesz.setArrtibute("printwidth", props.imagesize.printwidth) │ │ │ │ │ } │ │ │ │ │ - var already_exists = false; │ │ │ │ │ - if (point.osm_id) { │ │ │ │ │ - id = point.osm_id; │ │ │ │ │ - if (this.created_nodes[id]) { │ │ │ │ │ - already_exists = true │ │ │ │ │ + if (props.background != null) { │ │ │ │ │ + var backgrnd = this.createElementNS("", "BACKGROUND"); │ │ │ │ │ + propElem.appendChild(backgrnd); │ │ │ │ │ + backgrnd.setAttribute("color", props.background.color.r + "," + props.background.color.g + "," + props.background.color.b); │ │ │ │ │ + if (props.background.transcolor !== null) { │ │ │ │ │ + backgrnd.setAttribute("transcolor", props.background.transcolor.r + "," + props.background.transcolor.g + "," + props.background.transcolor.b) │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - id = -this.osm_id; │ │ │ │ │ - this.osm_id++ │ │ │ │ │ } │ │ │ │ │ - if (already_exists) { │ │ │ │ │ - node = this.created_nodes[id] │ │ │ │ │ + if (props.layerlist != null && props.layerlist.length > 0) { │ │ │ │ │ + var layerlst = this.createElementNS("", "LAYERLIST"); │ │ │ │ │ + propElem.appendChild(layerlst); │ │ │ │ │ + for (var ld = 0; ld < props.layerlist.length; ld++) { │ │ │ │ │ + var ldef = this.createElementNS("", "LAYERDEF"); │ │ │ │ │ + layerlst.appendChild(ldef); │ │ │ │ │ + ldef.setAttribute("id", props.layerlist[ld].id); │ │ │ │ │ + ldef.setAttribute("visible", props.layerlist[ld].visible); │ │ │ │ │ + if (typeof props.layerlist[ld].query == "object") { │ │ │ │ │ + var query = props.layerlist[ld].query; │ │ │ │ │ + if (query.where.length < 0) { │ │ │ │ │ + continue │ │ │ │ │ + } │ │ │ │ │ + var queryElem = null; │ │ │ │ │ + if (typeof query.spatialfilter == "boolean" && query.spatialfilter) { │ │ │ │ │ + queryElem = this.createElementNS("", "SPATIALQUERY") │ │ │ │ │ + } else { │ │ │ │ │ + queryElem = this.createElementNS("", "QUERY") │ │ │ │ │ + } │ │ │ │ │ + queryElem.setAttribute("where", query.where); │ │ │ │ │ + if (typeof query.accuracy == "number" && query.accuracy > 0) { │ │ │ │ │ + queryElem.setAttribute("accuracy", query.accuracy) │ │ │ │ │ + } │ │ │ │ │ + if (typeof query.featurelimit == "number" && query.featurelimit < 2e3) { │ │ │ │ │ + queryElem.setAttribute("featurelimit", query.featurelimit) │ │ │ │ │ + } │ │ │ │ │ + if (typeof query.subfields == "string" && query.subfields != "#ALL#") { │ │ │ │ │ + queryElem.setAttribute("subfields", query.subfields) │ │ │ │ │ + } │ │ │ │ │ + if (typeof query.joinexpression == "string" && query.joinexpression.length > 0) { │ │ │ │ │ + queryElem.setAttribute("joinexpression", query.joinexpression) │ │ │ │ │ + } │ │ │ │ │ + if (typeof query.jointables == "string" && query.jointables.length > 0) { │ │ │ │ │ + queryElem.setAttribute("jointables", query.jointables) │ │ │ │ │ + } │ │ │ │ │ + ldef.appendChild(queryElem) │ │ │ │ │ + } │ │ │ │ │ + if (typeof props.layerlist[ld].renderer == "object") { │ │ │ │ │ + this.addRenderer(ldef, props.layerlist[ld].renderer) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else if (request.get_feature != null) { │ │ │ │ │ + var getElem = this.createElementNS("", "GET_FEATURES"); │ │ │ │ │ + getElem.setAttribute("outputmode", "newxml"); │ │ │ │ │ + getElem.setAttribute("checkesc", "true"); │ │ │ │ │ + if (request.get_feature.geometry) { │ │ │ │ │ + getElem.setAttribute("geometry", request.get_feature.geometry) │ │ │ │ │ } else { │ │ │ │ │ - var node = this.createElementNS(null, "node") │ │ │ │ │ + getElem.setAttribute("geometry", "false") │ │ │ │ │ } │ │ │ │ │ - this.created_nodes[id] = node; │ │ │ │ │ - node.setAttribute("id", id); │ │ │ │ │ - node.setAttribute("lon", geometry.x); │ │ │ │ │ - node.setAttribute("lat", geometry.y); │ │ │ │ │ - if (point.attributes) { │ │ │ │ │ - this.serializeTags(point, node) │ │ │ │ │ + if (request.get_feature.compact) { │ │ │ │ │ + getElem.setAttribute("compact", request.get_feature.compact) │ │ │ │ │ } │ │ │ │ │ - this.setState(point, node); │ │ │ │ │ - return already_exists ? [] : [node] │ │ │ │ │ - }, │ │ │ │ │ - linestring: function(feature) { │ │ │ │ │ - var id; │ │ │ │ │ - var nodes = []; │ │ │ │ │ - var geometry = feature.geometry; │ │ │ │ │ - if (feature.osm_id) { │ │ │ │ │ - id = feature.osm_id │ │ │ │ │ + if (request.get_feature.featurelimit == "number") { │ │ │ │ │ + getElem.setAttribute("featurelimit", request.get_feature.featurelimit) │ │ │ │ │ + } │ │ │ │ │ + getElem.setAttribute("globalenvelope", "true"); │ │ │ │ │ + reqElem.appendChild(getElem); │ │ │ │ │ + if (request.get_feature.layer != null && request.get_feature.layer.length > 0) { │ │ │ │ │ + var lyrElem = this.createElementNS("", "LAYER"); │ │ │ │ │ + lyrElem.setAttribute("id", request.get_feature.layer); │ │ │ │ │ + getElem.appendChild(lyrElem) │ │ │ │ │ + } │ │ │ │ │ + var fquery = request.get_feature.query; │ │ │ │ │ + if (fquery != null) { │ │ │ │ │ + var qElem = null; │ │ │ │ │ + if (fquery.isspatial) { │ │ │ │ │ + qElem = this.createElementNS("", "SPATIALQUERY") │ │ │ │ │ + } else { │ │ │ │ │ + qElem = this.createElementNS("", "QUERY") │ │ │ │ │ + } │ │ │ │ │ + getElem.appendChild(qElem); │ │ │ │ │ + if (typeof fquery.accuracy == "number") { │ │ │ │ │ + qElem.setAttribute("accuracy", fquery.accuracy) │ │ │ │ │ + } │ │ │ │ │ + if (fquery.featurecoordsys != null) { │ │ │ │ │ + var fcsElem1 = this.createElementNS("", "FEATURECOORDSYS"); │ │ │ │ │ + if (fquery.featurecoordsys.id == 0) { │ │ │ │ │ + fcsElem1.setAttribute("string", fquery.featurecoordsys.string) │ │ │ │ │ + } else { │ │ │ │ │ + fcsElem1.setAttribute("id", fquery.featurecoordsys.id) │ │ │ │ │ + } │ │ │ │ │ + qElem.appendChild(fcsElem1) │ │ │ │ │ + } │ │ │ │ │ + if (fquery.filtercoordsys != null) { │ │ │ │ │ + var fcsElem2 = this.createElementNS("", "FILTERCOORDSYS"); │ │ │ │ │ + if (fquery.filtercoordsys.id === 0) { │ │ │ │ │ + fcsElem2.setAttribute("string", fquery.filtercoordsys.string) │ │ │ │ │ + } else { │ │ │ │ │ + fcsElem2.setAttribute("id", fquery.filtercoordsys.id) │ │ │ │ │ + } │ │ │ │ │ + qElem.appendChild(fcsElem2) │ │ │ │ │ + } │ │ │ │ │ + if (fquery.buffer > 0) { │ │ │ │ │ + var bufElem = this.createElementNS("", "BUFFER"); │ │ │ │ │ + bufElem.setAttribute("distance", fquery.buffer); │ │ │ │ │ + qElem.appendChild(bufElem) │ │ │ │ │ + } │ │ │ │ │ + if (fquery.isspatial) { │ │ │ │ │ + var spfElem = this.createElementNS("", "SPATIALFILTER"); │ │ │ │ │ + spfElem.setAttribute("relation", fquery.spatialfilter.relation); │ │ │ │ │ + qElem.appendChild(spfElem); │ │ │ │ │ + if (fquery.spatialfilter.envelope) { │ │ │ │ │ + var envElem = this.createElementNS("", "ENVELOPE"); │ │ │ │ │ + envElem.setAttribute("minx", fquery.spatialfilter.envelope.minx); │ │ │ │ │ + envElem.setAttribute("miny", fquery.spatialfilter.envelope.miny); │ │ │ │ │ + envElem.setAttribute("maxx", fquery.spatialfilter.envelope.maxx); │ │ │ │ │ + envElem.setAttribute("maxy", fquery.spatialfilter.envelope.maxy); │ │ │ │ │ + spfElem.appendChild(envElem) │ │ │ │ │ + } else if (typeof fquery.spatialfilter.polygon == "object") { │ │ │ │ │ + spfElem.appendChild(this.writePolygonGeometry(fquery.spatialfilter.polygon)) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (fquery.where != null && fquery.where.length > 0) { │ │ │ │ │ + qElem.setAttribute("where", fquery.where) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + root.appendChild(reqElem); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [root]) │ │ │ │ │ + }, │ │ │ │ │ + addGroupRenderer: function(ldef, toprenderer) { │ │ │ │ │ + var topRelem = this.createElementNS("", "GROUPRENDERER"); │ │ │ │ │ + ldef.appendChild(topRelem); │ │ │ │ │ + for (var rind = 0; rind < toprenderer.length; rind++) { │ │ │ │ │ + var renderer = toprenderer[rind]; │ │ │ │ │ + this.addRenderer(topRelem, renderer) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addRenderer: function(topRelem, renderer) { │ │ │ │ │ + if (OpenLayers.Util.isArray(renderer)) { │ │ │ │ │ + this.addGroupRenderer(topRelem, renderer) │ │ │ │ │ + } else { │ │ │ │ │ + var renderElem = this.createElementNS("", renderer.type.toUpperCase() + "RENDERER"); │ │ │ │ │ + topRelem.appendChild(renderElem); │ │ │ │ │ + if (renderElem.tagName == "VALUEMAPRENDERER") { │ │ │ │ │ + this.addValueMapRenderer(renderElem, renderer) │ │ │ │ │ + } else if (renderElem.tagName == "VALUEMAPLABELRENDERER") { │ │ │ │ │ + this.addValueMapLabelRenderer(renderElem, renderer) │ │ │ │ │ + } else if (renderElem.tagName == "SIMPLELABELRENDERER") { │ │ │ │ │ + this.addSimpleLabelRenderer(renderElem, renderer) │ │ │ │ │ + } else if (renderElem.tagName == "SCALEDEPENDENTRENDERER") { │ │ │ │ │ + this.addScaleDependentRenderer(renderElem, renderer) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addScaleDependentRenderer: function(renderElem, renderer) { │ │ │ │ │ + if (typeof renderer.lower == "string" || typeof renderer.lower == "number") { │ │ │ │ │ + renderElem.setAttribute("lower", renderer.lower) │ │ │ │ │ + } │ │ │ │ │ + if (typeof renderer.upper == "string" || typeof renderer.upper == "number") { │ │ │ │ │ + renderElem.setAttribute("upper", renderer.upper) │ │ │ │ │ + } │ │ │ │ │ + this.addRenderer(renderElem, renderer.renderer) │ │ │ │ │ + }, │ │ │ │ │ + addValueMapLabelRenderer: function(renderElem, renderer) { │ │ │ │ │ + renderElem.setAttribute("lookupfield", renderer.lookupfield); │ │ │ │ │ + renderElem.setAttribute("labelfield", renderer.labelfield); │ │ │ │ │ + if (typeof renderer.exacts == "object") { │ │ │ │ │ + for (var ext = 0, extlen = renderer.exacts.length; ext < extlen; ext++) { │ │ │ │ │ + var exact = renderer.exacts[ext]; │ │ │ │ │ + var eelem = this.createElementNS("", "EXACT"); │ │ │ │ │ + if (typeof exact.value == "string") { │ │ │ │ │ + eelem.setAttribute("value", exact.value) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.label == "string") { │ │ │ │ │ + eelem.setAttribute("label", exact.label) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.method == "string") { │ │ │ │ │ + eelem.setAttribute("method", exact.method) │ │ │ │ │ + } │ │ │ │ │ + renderElem.appendChild(eelem); │ │ │ │ │ + if (typeof exact.symbol == "object") { │ │ │ │ │ + var selem = null; │ │ │ │ │ + if (exact.symbol.type == "text") { │ │ │ │ │ + selem = this.createElementNS("", "TEXTSYMBOL") │ │ │ │ │ + } │ │ │ │ │ + if (selem != null) { │ │ │ │ │ + var keys = this.fontStyleKeys; │ │ │ │ │ + for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ + var key = keys[i]; │ │ │ │ │ + if (exact.symbol[key]) { │ │ │ │ │ + selem.setAttribute(key, exact.symbol[key]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + eelem.appendChild(selem) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addValueMapRenderer: function(renderElem, renderer) { │ │ │ │ │ + renderElem.setAttribute("lookupfield", renderer.lookupfield); │ │ │ │ │ + if (typeof renderer.ranges == "object") { │ │ │ │ │ + for (var rng = 0, rnglen = renderer.ranges.length; rng < rnglen; rng++) { │ │ │ │ │ + var range = renderer.ranges[rng]; │ │ │ │ │ + var relem = this.createElementNS("", "RANGE"); │ │ │ │ │ + relem.setAttribute("lower", range.lower); │ │ │ │ │ + relem.setAttribute("upper", range.upper); │ │ │ │ │ + renderElem.appendChild(relem); │ │ │ │ │ + if (typeof range.symbol == "object") { │ │ │ │ │ + var selem = null; │ │ │ │ │ + if (range.symbol.type == "simplepolygon") { │ │ │ │ │ + selem = this.createElementNS("", "SIMPLEPOLYGONSYMBOL") │ │ │ │ │ + } │ │ │ │ │ + if (selem != null) { │ │ │ │ │ + if (typeof range.symbol.boundarycolor == "string") { │ │ │ │ │ + selem.setAttribute("boundarycolor", range.symbol.boundarycolor) │ │ │ │ │ + } │ │ │ │ │ + if (typeof range.symbol.fillcolor == "string") { │ │ │ │ │ + selem.setAttribute("fillcolor", range.symbol.fillcolor) │ │ │ │ │ + } │ │ │ │ │ + if (typeof range.symbol.filltransparency == "number") { │ │ │ │ │ + selem.setAttribute("filltransparency", range.symbol.filltransparency) │ │ │ │ │ + } │ │ │ │ │ + relem.appendChild(selem) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else if (typeof renderer.exacts == "object") { │ │ │ │ │ + for (var ext = 0, extlen = renderer.exacts.length; ext < extlen; ext++) { │ │ │ │ │ + var exact = renderer.exacts[ext]; │ │ │ │ │ + var eelem = this.createElementNS("", "EXACT"); │ │ │ │ │ + if (typeof exact.value == "string") { │ │ │ │ │ + eelem.setAttribute("value", exact.value) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.label == "string") { │ │ │ │ │ + eelem.setAttribute("label", exact.label) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.method == "string") { │ │ │ │ │ + eelem.setAttribute("method", exact.method) │ │ │ │ │ + } │ │ │ │ │ + renderElem.appendChild(eelem); │ │ │ │ │ + if (typeof exact.symbol == "object") { │ │ │ │ │ + var selem = null; │ │ │ │ │ + if (exact.symbol.type == "simplemarker") { │ │ │ │ │ + selem = this.createElementNS("", "SIMPLEMARKERSYMBOL") │ │ │ │ │ + } │ │ │ │ │ + if (selem != null) { │ │ │ │ │ + if (typeof exact.symbol.antialiasing == "string") { │ │ │ │ │ + selem.setAttribute("antialiasing", exact.symbol.antialiasing) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.color == "string") { │ │ │ │ │ + selem.setAttribute("color", exact.symbol.color) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.outline == "string") { │ │ │ │ │ + selem.setAttribute("outline", exact.symbol.outline) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.overlap == "string") { │ │ │ │ │ + selem.setAttribute("overlap", exact.symbol.overlap) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.shadow == "string") { │ │ │ │ │ + selem.setAttribute("shadow", exact.symbol.shadow) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.transparency == "number") { │ │ │ │ │ + selem.setAttribute("transparency", exact.symbol.transparency) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.usecentroid == "string") { │ │ │ │ │ + selem.setAttribute("usecentroid", exact.symbol.usecentroid) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.width == "number") { │ │ │ │ │ + selem.setAttribute("width", exact.symbol.width) │ │ │ │ │ + } │ │ │ │ │ + eelem.appendChild(selem) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addSimpleLabelRenderer: function(renderElem, renderer) { │ │ │ │ │ + renderElem.setAttribute("field", renderer.field); │ │ │ │ │ + var keys = ["featureweight", "howmanylabels", "labelbufferratio", "labelpriorities", "labelweight", "linelabelposition", "rotationalangles"]; │ │ │ │ │ + for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ + var key = keys[i]; │ │ │ │ │ + if (renderer[key]) { │ │ │ │ │ + renderElem.setAttribute(key, renderer[key]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (renderer.symbol.type == "text") { │ │ │ │ │ + var symbol = renderer.symbol; │ │ │ │ │ + var selem = this.createElementNS("", "TEXTSYMBOL"); │ │ │ │ │ + renderElem.appendChild(selem); │ │ │ │ │ + var keys = this.fontStyleKeys; │ │ │ │ │ + for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ + var key = keys[i]; │ │ │ │ │ + if (symbol[key]) { │ │ │ │ │ + selem.setAttribute(key, renderer[key]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + writePolygonGeometry: function(polygon) { │ │ │ │ │ + if (!(polygon instanceof OpenLayers.Geometry.Polygon)) { │ │ │ │ │ + throw { │ │ │ │ │ + message: "Cannot write polygon geometry to ArcXML with an " + polygon.CLASS_NAME + " object.", │ │ │ │ │ + geometry: polygon │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var polyElem = this.createElementNS("", "POLYGON"); │ │ │ │ │ + for (var ln = 0, lnlen = polygon.components.length; ln < lnlen; ln++) { │ │ │ │ │ + var ring = polygon.components[ln]; │ │ │ │ │ + var ringElem = this.createElementNS("", "RING"); │ │ │ │ │ + for (var rn = 0, rnlen = ring.components.length; rn < rnlen; rn++) { │ │ │ │ │ + var point = ring.components[rn]; │ │ │ │ │ + var pointElem = this.createElementNS("", "POINT"); │ │ │ │ │ + pointElem.setAttribute("x", point.x); │ │ │ │ │ + pointElem.setAttribute("y", point.y); │ │ │ │ │ + ringElem.appendChild(pointElem) │ │ │ │ │ + } │ │ │ │ │ + polyElem.appendChild(ringElem) │ │ │ │ │ + } │ │ │ │ │ + return polyElem │ │ │ │ │ + }, │ │ │ │ │ + parseResponse: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + var newData = new OpenLayers.Format.XML; │ │ │ │ │ + data = newData.read(data) │ │ │ │ │ + } │ │ │ │ │ + var response = new OpenLayers.Format.ArcXML.Response; │ │ │ │ │ + var errorNode = data.getElementsByTagName("ERROR"); │ │ │ │ │ + if (errorNode != null && errorNode.length > 0) { │ │ │ │ │ + response.error = this.getChildValue(errorNode, "Unknown error.") │ │ │ │ │ + } else { │ │ │ │ │ + var responseNode = data.getElementsByTagName("RESPONSE"); │ │ │ │ │ + if (responseNode == null || responseNode.length == 0) { │ │ │ │ │ + response.error = "No RESPONSE tag found in ArcXML response."; │ │ │ │ │ + return response │ │ │ │ │ + } │ │ │ │ │ + var rtype = responseNode[0].firstChild.nodeName; │ │ │ │ │ + if (rtype == "#text") { │ │ │ │ │ + rtype = responseNode[0].firstChild.nextSibling.nodeName │ │ │ │ │ + } │ │ │ │ │ + if (rtype == "IMAGE") { │ │ │ │ │ + var envelopeNode = data.getElementsByTagName("ENVELOPE"); │ │ │ │ │ + var outputNode = data.getElementsByTagName("OUTPUT"); │ │ │ │ │ + if (envelopeNode == null || envelopeNode.length == 0) { │ │ │ │ │ + response.error = "No ENVELOPE tag found in ArcXML response." │ │ │ │ │ + } else if (outputNode == null || outputNode.length == 0) { │ │ │ │ │ + response.error = "No OUTPUT tag found in ArcXML response." │ │ │ │ │ + } else { │ │ │ │ │ + var envAttr = this.parseAttributes(envelopeNode[0]); │ │ │ │ │ + var outputAttr = this.parseAttributes(outputNode[0]); │ │ │ │ │ + if (typeof outputAttr.type == "string") { │ │ │ │ │ + response.image = { │ │ │ │ │ + envelope: envAttr, │ │ │ │ │ + output: { │ │ │ │ │ + type: outputAttr.type, │ │ │ │ │ + data: this.getChildValue(outputNode[0]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + response.image = { │ │ │ │ │ + envelope: envAttr, │ │ │ │ │ + output: outputAttr │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else if (rtype == "FEATURES") { │ │ │ │ │ + var features = responseNode[0].getElementsByTagName("FEATURES"); │ │ │ │ │ + var featureCount = features[0].getElementsByTagName("FEATURECOUNT"); │ │ │ │ │ + response.features.featurecount = featureCount[0].getAttribute("count"); │ │ │ │ │ + if (response.features.featurecount > 0) { │ │ │ │ │ + var envelope = features[0].getElementsByTagName("ENVELOPE"); │ │ │ │ │ + response.features.envelope = this.parseAttributes(envelope[0], typeof 0); │ │ │ │ │ + var featureList = features[0].getElementsByTagName("FEATURE"); │ │ │ │ │ + for (var fn = 0; fn < featureList.length; fn++) { │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector; │ │ │ │ │ + var fields = featureList[fn].getElementsByTagName("FIELD"); │ │ │ │ │ + for (var fdn = 0; fdn < fields.length; fdn++) { │ │ │ │ │ + var fieldName = fields[fdn].getAttribute("name"); │ │ │ │ │ + var fieldValue = fields[fdn].getAttribute("value"); │ │ │ │ │ + feature.attributes[fieldName] = fieldValue │ │ │ │ │ + } │ │ │ │ │ + var geom = featureList[fn].getElementsByTagName("POLYGON"); │ │ │ │ │ + if (geom.length > 0) { │ │ │ │ │ + var ring = geom[0].getElementsByTagName("RING"); │ │ │ │ │ + var polys = []; │ │ │ │ │ + for (var rn = 0; rn < ring.length; rn++) { │ │ │ │ │ + var linearRings = []; │ │ │ │ │ + linearRings.push(this.parsePointGeometry(ring[rn])); │ │ │ │ │ + var holes = ring[rn].getElementsByTagName("HOLE"); │ │ │ │ │ + for (var hn = 0; hn < holes.length; hn++) { │ │ │ │ │ + linearRings.push(this.parsePointGeometry(holes[hn])) │ │ │ │ │ + } │ │ │ │ │ + holes = null; │ │ │ │ │ + polys.push(new OpenLayers.Geometry.Polygon(linearRings)); │ │ │ │ │ + linearRings = null │ │ │ │ │ + } │ │ │ │ │ + ring = null; │ │ │ │ │ + if (polys.length == 1) { │ │ │ │ │ + feature.geometry = polys[0] │ │ │ │ │ + } else { │ │ │ │ │ + feature.geometry = new OpenLayers.Geometry.MultiPolygon(polys) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + response.features.feature.push(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + response.error = "Unidentified response type." │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return response │ │ │ │ │ + }, │ │ │ │ │ + parseAttributes: function(node, type) { │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + for (var attr = 0; attr < node.attributes.length; attr++) { │ │ │ │ │ + if (type == "number") { │ │ │ │ │ + attributes[node.attributes[attr].nodeName] = parseFloat(node.attributes[attr].nodeValue) │ │ │ │ │ + } else { │ │ │ │ │ + attributes[node.attributes[attr].nodeName] = node.attributes[attr].nodeValue │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return attributes │ │ │ │ │ + }, │ │ │ │ │ + parsePointGeometry: function(node) { │ │ │ │ │ + var ringPoints = []; │ │ │ │ │ + var coords = node.getElementsByTagName("COORDS"); │ │ │ │ │ + if (coords.length > 0) { │ │ │ │ │ + var coordArr = this.getChildValue(coords[0]); │ │ │ │ │ + coordArr = coordArr.split(/;/); │ │ │ │ │ + for (var cn = 0; cn < coordArr.length; cn++) { │ │ │ │ │ + var coordItems = coordArr[cn].split(/ /); │ │ │ │ │ + ringPoints.push(new OpenLayers.Geometry.Point(coordItems[0], coordItems[1])) │ │ │ │ │ + } │ │ │ │ │ + coords = null │ │ │ │ │ + } else { │ │ │ │ │ + var point = node.getElementsByTagName("POINT"); │ │ │ │ │ + if (point.length > 0) { │ │ │ │ │ + for (var pn = 0; pn < point.length; pn++) { │ │ │ │ │ + ringPoints.push(new OpenLayers.Geometry.Point(parseFloat(point[pn].getAttribute("x")), parseFloat(point[pn].getAttribute("y")))) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + point = null │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.LinearRing(ringPoints) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.ArcXML" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.ArcXML.Request = OpenLayers.Class({ │ │ │ │ │ + initialize: function(params) { │ │ │ │ │ + var defaults = { │ │ │ │ │ + get_image: { │ │ │ │ │ + properties: { │ │ │ │ │ + background: null, │ │ │ │ │ + draw: true, │ │ │ │ │ + envelope: { │ │ │ │ │ + minx: 0, │ │ │ │ │ + miny: 0, │ │ │ │ │ + maxx: 0, │ │ │ │ │ + maxy: 0 │ │ │ │ │ + }, │ │ │ │ │ + featurecoordsys: { │ │ │ │ │ + id: 0, │ │ │ │ │ + string: "", │ │ │ │ │ + datumtransformid: 0, │ │ │ │ │ + datumtransformstring: "" │ │ │ │ │ + }, │ │ │ │ │ + filtercoordsys: { │ │ │ │ │ + id: 0, │ │ │ │ │ + string: "", │ │ │ │ │ + datumtransformid: 0, │ │ │ │ │ + datumtransformstring: "" │ │ │ │ │ + }, │ │ │ │ │ + imagesize: { │ │ │ │ │ + height: 0, │ │ │ │ │ + width: 0, │ │ │ │ │ + dpi: 96, │ │ │ │ │ + printheight: 0, │ │ │ │ │ + printwidth: 0, │ │ │ │ │ + scalesymbols: false │ │ │ │ │ + }, │ │ │ │ │ + layerlist: [], │ │ │ │ │ + output: { │ │ │ │ │ + baseurl: "", │ │ │ │ │ + legendbaseurl: "", │ │ │ │ │ + legendname: "", │ │ │ │ │ + legendpath: "", │ │ │ │ │ + legendurl: "", │ │ │ │ │ + name: "", │ │ │ │ │ + path: "", │ │ │ │ │ + type: "jpg", │ │ │ │ │ + url: "" │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + get_feature: { │ │ │ │ │ + layer: "", │ │ │ │ │ + query: { │ │ │ │ │ + isspatial: false, │ │ │ │ │ + featurecoordsys: { │ │ │ │ │ + id: 0, │ │ │ │ │ + string: "", │ │ │ │ │ + datumtransformid: 0, │ │ │ │ │ + datumtransformstring: "" │ │ │ │ │ + }, │ │ │ │ │ + filtercoordsys: { │ │ │ │ │ + id: 0, │ │ │ │ │ + string: "", │ │ │ │ │ + datumtransformid: 0, │ │ │ │ │ + datumtransformstring: "" │ │ │ │ │ + }, │ │ │ │ │ + buffer: 0, │ │ │ │ │ + where: "", │ │ │ │ │ + spatialfilter: { │ │ │ │ │ + relation: "envelope_intersection", │ │ │ │ │ + envelope: null │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + environment: { │ │ │ │ │ + separators: { │ │ │ │ │ + cs: " ", │ │ │ │ │ + ts: ";" │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + layer: [], │ │ │ │ │ + workspaces: [] │ │ │ │ │ + }; │ │ │ │ │ + return OpenLayers.Util.extend(this, defaults) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.ArcXML.Request" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.ArcXML.Response = OpenLayers.Class({ │ │ │ │ │ + initialize: function(params) { │ │ │ │ │ + var defaults = { │ │ │ │ │ + image: { │ │ │ │ │ + envelope: null, │ │ │ │ │ + output: "" │ │ │ │ │ + }, │ │ │ │ │ + features: { │ │ │ │ │ + featurecount: 0, │ │ │ │ │ + envelope: null, │ │ │ │ │ + feature: [] │ │ │ │ │ + }, │ │ │ │ │ + error: "" │ │ │ │ │ + }; │ │ │ │ │ + return OpenLayers.Util.extend(this, defaults) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.ArcXML.Response" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.Text = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ + defaultStyle: null, │ │ │ │ │ + extractStyles: true, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + if (options.extractStyles !== false) { │ │ │ │ │ + options.defaultStyle = { │ │ │ │ │ + externalGraphic: OpenLayers.Util.getImageLocation("marker.png"), │ │ │ │ │ + graphicWidth: 21, │ │ │ │ │ + graphicHeight: 25, │ │ │ │ │ + graphicXOffset: -10.5, │ │ │ │ │ + graphicYOffset: -12.5 │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Format.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ + read: function(text) { │ │ │ │ │ + var lines = text.split("\n"); │ │ │ │ │ + var columns; │ │ │ │ │ + var features = []; │ │ │ │ │ + for (var lcv = 0; lcv < lines.length - 1; lcv++) { │ │ │ │ │ + var currLine = lines[lcv].replace(/^\s*/, "").replace(/\s*$/, ""); │ │ │ │ │ + if (currLine.charAt(0) != "#") { │ │ │ │ │ + if (!columns) { │ │ │ │ │ + columns = currLine.split("\t") │ │ │ │ │ + } else { │ │ │ │ │ + var vals = currLine.split("\t"); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point(0, 0); │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + var style = this.defaultStyle ? OpenLayers.Util.applyDefaults({}, this.defaultStyle) : null; │ │ │ │ │ + var icon, iconSize, iconOffset, overflow; │ │ │ │ │ + var set = false; │ │ │ │ │ + for (var valIndex = 0; valIndex < vals.length; valIndex++) { │ │ │ │ │ + if (vals[valIndex]) { │ │ │ │ │ + if (columns[valIndex] == "point") { │ │ │ │ │ + var coords = vals[valIndex].split(","); │ │ │ │ │ + geometry.y = parseFloat(coords[0]); │ │ │ │ │ + geometry.x = parseFloat(coords[1]); │ │ │ │ │ + set = true │ │ │ │ │ + } else if (columns[valIndex] == "lat") { │ │ │ │ │ + geometry.y = parseFloat(vals[valIndex]); │ │ │ │ │ + set = true │ │ │ │ │ + } else if (columns[valIndex] == "lon") { │ │ │ │ │ + geometry.x = parseFloat(vals[valIndex]); │ │ │ │ │ + set = true │ │ │ │ │ + } else if (columns[valIndex] == "title") attributes["title"] = vals[valIndex]; │ │ │ │ │ + else if (columns[valIndex] == "image" || columns[valIndex] == "icon" && style) { │ │ │ │ │ + style["externalGraphic"] = vals[valIndex] │ │ │ │ │ + } else if (columns[valIndex] == "iconSize" && style) { │ │ │ │ │ + var size = vals[valIndex].split(","); │ │ │ │ │ + style["graphicWidth"] = parseFloat(size[0]); │ │ │ │ │ + style["graphicHeight"] = parseFloat(size[1]) │ │ │ │ │ + } else if (columns[valIndex] == "iconOffset" && style) { │ │ │ │ │ + var offset = vals[valIndex].split(","); │ │ │ │ │ + style["graphicXOffset"] = parseFloat(offset[0]); │ │ │ │ │ + style["graphicYOffset"] = parseFloat(offset[1]) │ │ │ │ │ + } else if (columns[valIndex] == "description") { │ │ │ │ │ + attributes["description"] = vals[valIndex] │ │ │ │ │ + } else if (columns[valIndex] == "overflow") { │ │ │ │ │ + attributes["overflow"] = vals[valIndex] │ │ │ │ │ + } else { │ │ │ │ │ + attributes[columns[valIndex]] = vals[valIndex] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (set) { │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ + } │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry, attributes, style); │ │ │ │ │ + features.push(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return features │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.Text" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.Context = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + layerOptions: null, │ │ │ │ │ + layerParams: null, │ │ │ │ │ + read: function(data, options) { │ │ │ │ │ + var context = OpenLayers.Format.XML.VersionedOGC.prototype.read.apply(this, arguments); │ │ │ │ │ + var map; │ │ │ │ │ + if (options && options.map) { │ │ │ │ │ + this.context = context; │ │ │ │ │ + if (options.map instanceof OpenLayers.Map) { │ │ │ │ │ + map = this.mergeContextToMap(context, options.map) │ │ │ │ │ + } else { │ │ │ │ │ + var mapOptions = options.map; │ │ │ │ │ + if (OpenLayers.Util.isElement(mapOptions) || typeof mapOptions == "string") { │ │ │ │ │ + mapOptions = { │ │ │ │ │ + div: mapOptions │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + map = this.contextToMap(context, mapOptions) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + map = context │ │ │ │ │ + } │ │ │ │ │ + return map │ │ │ │ │ + }, │ │ │ │ │ + getLayerFromContext: function(layerContext) { │ │ │ │ │ + var i, len; │ │ │ │ │ + var options = { │ │ │ │ │ + queryable: layerContext.queryable, │ │ │ │ │ + visibility: layerContext.visibility, │ │ │ │ │ + maxExtent: layerContext.maxExtent, │ │ │ │ │ + metadata: OpenLayers.Util.applyDefaults(layerContext.metadata, { │ │ │ │ │ + styles: layerContext.styles, │ │ │ │ │ + formats: layerContext.formats, │ │ │ │ │ + abstract: layerContext["abstract"], │ │ │ │ │ + dataURL: layerContext.dataURL │ │ │ │ │ + }), │ │ │ │ │ + numZoomLevels: layerContext.numZoomLevels, │ │ │ │ │ + units: layerContext.units, │ │ │ │ │ + isBaseLayer: layerContext.isBaseLayer, │ │ │ │ │ + opacity: layerContext.opacity, │ │ │ │ │ + displayInLayerSwitcher: layerContext.displayInLayerSwitcher, │ │ │ │ │ + singleTile: layerContext.singleTile, │ │ │ │ │ + tileSize: layerContext.tileSize ? new OpenLayers.Size(layerContext.tileSize.width, layerContext.tileSize.height) : undefined, │ │ │ │ │ + minScale: layerContext.minScale || layerContext.maxScaleDenominator, │ │ │ │ │ + maxScale: layerContext.maxScale || layerContext.minScaleDenominator, │ │ │ │ │ + srs: layerContext.srs, │ │ │ │ │ + dimensions: layerContext.dimensions, │ │ │ │ │ + metadataURL: layerContext.metadataURL │ │ │ │ │ + }; │ │ │ │ │ + if (this.layerOptions) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.layerOptions) │ │ │ │ │ + } │ │ │ │ │ + var params = { │ │ │ │ │ + layers: layerContext.name, │ │ │ │ │ + transparent: layerContext.transparent, │ │ │ │ │ + version: layerContext.version │ │ │ │ │ + }; │ │ │ │ │ + if (layerContext.formats && layerContext.formats.length > 0) { │ │ │ │ │ + params.format = layerContext.formats[0].value; │ │ │ │ │ + for (i = 0, len = layerContext.formats.length; i < len; i++) { │ │ │ │ │ + var format = layerContext.formats[i]; │ │ │ │ │ + if (format.current == true) { │ │ │ │ │ + params.format = format.value; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (layerContext.styles && layerContext.styles.length > 0) { │ │ │ │ │ + for (i = 0, len = layerContext.styles.length; i < len; i++) { │ │ │ │ │ + var style = layerContext.styles[i]; │ │ │ │ │ + if (style.current == true) { │ │ │ │ │ + if (style.href) { │ │ │ │ │ + params.sld = style.href │ │ │ │ │ + } else if (style.body) { │ │ │ │ │ + params.sld_body = style.body │ │ │ │ │ + } else { │ │ │ │ │ + params.styles = style.name │ │ │ │ │ + } │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.layerParams) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(params, this.layerParams) │ │ │ │ │ + } │ │ │ │ │ + var layer = null; │ │ │ │ │ + var service = layerContext.service; │ │ │ │ │ + if (service == OpenLayers.Format.Context.serviceTypes.WFS) { │ │ │ │ │ + options.strategies = [new OpenLayers.Strategy.BBOX]; │ │ │ │ │ + options.protocol = new OpenLayers.Protocol.WFS({ │ │ │ │ │ + url: layerContext.url, │ │ │ │ │ + featurePrefix: layerContext.name.split(":")[0], │ │ │ │ │ + featureType: layerContext.name.split(":").pop() │ │ │ │ │ + }); │ │ │ │ │ + layer = new OpenLayers.Layer.Vector(layerContext.title || layerContext.name, options) │ │ │ │ │ + } else if (service == OpenLayers.Format.Context.serviceTypes.KML) { │ │ │ │ │ + options.strategies = [new OpenLayers.Strategy.Fixed]; │ │ │ │ │ + options.protocol = new OpenLayers.Protocol.HTTP({ │ │ │ │ │ + url: layerContext.url, │ │ │ │ │ + format: new OpenLayers.Format.KML │ │ │ │ │ + }); │ │ │ │ │ + layer = new OpenLayers.Layer.Vector(layerContext.title || layerContext.name, options) │ │ │ │ │ + } else if (service == OpenLayers.Format.Context.serviceTypes.GML) { │ │ │ │ │ + options.strategies = [new OpenLayers.Strategy.Fixed]; │ │ │ │ │ + options.protocol = new OpenLayers.Protocol.HTTP({ │ │ │ │ │ + url: layerContext.url, │ │ │ │ │ + format: new OpenLayers.Format.GML │ │ │ │ │ + }); │ │ │ │ │ + layer = new OpenLayers.Layer.Vector(layerContext.title || layerContext.name, options) │ │ │ │ │ + } else if (layerContext.features) { │ │ │ │ │ + layer = new OpenLayers.Layer.Vector(layerContext.title || layerContext.name, options); │ │ │ │ │ + layer.addFeatures(layerContext.features) │ │ │ │ │ + } else if (layerContext.categoryLayer !== true) { │ │ │ │ │ + layer = new OpenLayers.Layer.WMS(layerContext.title || layerContext.name, layerContext.url, params, options) │ │ │ │ │ + } │ │ │ │ │ + return layer │ │ │ │ │ + }, │ │ │ │ │ + getLayersFromContext: function(layersContext) { │ │ │ │ │ + var layers = []; │ │ │ │ │ + for (var i = 0, len = layersContext.length; i < len; i++) { │ │ │ │ │ + var layer = this.getLayerFromContext(layersContext[i]); │ │ │ │ │ + if (layer !== null) { │ │ │ │ │ + layers.push(layer) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return layers │ │ │ │ │ + }, │ │ │ │ │ + contextToMap: function(context, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + maxExtent: context.maxExtent, │ │ │ │ │ + projection: context.projection, │ │ │ │ │ + units: context.units │ │ │ │ │ + }, options); │ │ │ │ │ + if (options.maxExtent) { │ │ │ │ │ + options.maxResolution = options.maxExtent.getWidth() / OpenLayers.Map.TILE_WIDTH │ │ │ │ │ + } │ │ │ │ │ + var metadata = { │ │ │ │ │ + contactInformation: context.contactInformation, │ │ │ │ │ + abstract: context["abstract"], │ │ │ │ │ + keywords: context.keywords, │ │ │ │ │ + logo: context.logo, │ │ │ │ │ + descriptionURL: context.descriptionURL │ │ │ │ │ + }; │ │ │ │ │ + options.metadata = metadata; │ │ │ │ │ + var map = new OpenLayers.Map(options); │ │ │ │ │ + map.addLayers(this.getLayersFromContext(context.layersContext)); │ │ │ │ │ + map.setCenter(context.bounds.getCenterLonLat(), map.getZoomForExtent(context.bounds, true)); │ │ │ │ │ + return map │ │ │ │ │ + }, │ │ │ │ │ + mergeContextToMap: function(context, map) { │ │ │ │ │ + map.addLayers(this.getLayersFromContext(context.layersContext)); │ │ │ │ │ + return map │ │ │ │ │ + }, │ │ │ │ │ + write: function(obj, options) { │ │ │ │ │ + obj = this.toContext(obj); │ │ │ │ │ + return OpenLayers.Format.XML.VersionedOGC.prototype.write.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.Context" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.Context.serviceTypes = { │ │ │ │ │ + WMS: "urn:ogc:serviceType:WMS", │ │ │ │ │ + WFS: "urn:ogc:serviceType:WFS", │ │ │ │ │ + WCS: "urn:ogc:serviceType:WCS", │ │ │ │ │ + GML: "urn:ogc:serviceType:GML", │ │ │ │ │ + SLD: "urn:ogc:serviceType:SLD", │ │ │ │ │ + FES: "urn:ogc:serviceType:FES", │ │ │ │ │ + KML: "urn:ogc:serviceType:KML" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Format.OWSContext = OpenLayers.Class(OpenLayers.Format.Context, { │ │ │ │ │ + defaultVersion: "0.3.1", │ │ │ │ │ + getVersion: function(root, options) { │ │ │ │ │ + var version = OpenLayers.Format.XML.VersionedOGC.prototype.getVersion.apply(this, arguments); │ │ │ │ │ + if (version === "0.3.0") { │ │ │ │ │ + version = this.defaultVersion │ │ │ │ │ + } │ │ │ │ │ + return version │ │ │ │ │ + }, │ │ │ │ │ + toContext: function(obj) { │ │ │ │ │ + var context = {}; │ │ │ │ │ + if (obj.CLASS_NAME == "OpenLayers.Map") { │ │ │ │ │ + context.bounds = obj.getExtent(); │ │ │ │ │ + context.maxExtent = obj.maxExtent; │ │ │ │ │ + context.projection = obj.projection; │ │ │ │ │ + context.size = obj.getSize(); │ │ │ │ │ + context.layers = obj.layers │ │ │ │ │ + } │ │ │ │ │ + return context │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.OWSContext" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.GPX = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + defaultDesc: "No description available", │ │ │ │ │ + extractWaypoints: true, │ │ │ │ │ + extractTracks: true, │ │ │ │ │ + extractRoutes: true, │ │ │ │ │ + extractAttributes: true, │ │ │ │ │ + namespaces: { │ │ │ │ │ + gpx: "http://www.topografix.com/GPX/1/1", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ + schemaLocation: "http://www.topografix.com/GPX/1/1 http://www.topografix.com/GPX/1/1/gpx.xsd", │ │ │ │ │ + creator: "OpenLayers", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ + read: function(doc) { │ │ │ │ │ + if (typeof doc == "string") { │ │ │ │ │ + doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]) │ │ │ │ │ + } │ │ │ │ │ + var features = []; │ │ │ │ │ + if (this.extractTracks) { │ │ │ │ │ + var tracks = doc.getElementsByTagName("trk"); │ │ │ │ │ + for (var i = 0, len = tracks.length; i < len; i++) { │ │ │ │ │ + var attrs = {}; │ │ │ │ │ + if (this.extractAttributes) { │ │ │ │ │ + attrs = this.parseAttributes(tracks[i]) │ │ │ │ │ + } │ │ │ │ │ + var segs = this.getElementsByTagNameNS(tracks[i], tracks[i].namespaceURI, "trkseg"); │ │ │ │ │ + for (var j = 0, seglen = segs.length; j < seglen; j++) { │ │ │ │ │ + var track = this.extractSegment(segs[j], "trkpt"); │ │ │ │ │ + features.push(new OpenLayers.Feature.Vector(track, attrs)) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.extractRoutes) { │ │ │ │ │ + var routes = doc.getElementsByTagName("rte"); │ │ │ │ │ + for (var k = 0, klen = routes.length; k < klen; k++) { │ │ │ │ │ + var attrs = {}; │ │ │ │ │ + if (this.extractAttributes) { │ │ │ │ │ + attrs = this.parseAttributes(routes[k]) │ │ │ │ │ + } │ │ │ │ │ + var route = this.extractSegment(routes[k], "rtept"); │ │ │ │ │ + features.push(new OpenLayers.Feature.Vector(route, attrs)) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.extractWaypoints) { │ │ │ │ │ + var waypoints = doc.getElementsByTagName("wpt"); │ │ │ │ │ + for (var l = 0, len = waypoints.length; l < len; l++) { │ │ │ │ │ + var attrs = {}; │ │ │ │ │ + if (this.extractAttributes) { │ │ │ │ │ + attrs = this.parseAttributes(waypoints[l]) │ │ │ │ │ + } │ │ │ │ │ + var wpt = new OpenLayers.Geometry.Point(waypoints[l].getAttribute("lon"), waypoints[l].getAttribute("lat")); │ │ │ │ │ + features.push(new OpenLayers.Feature.Vector(wpt, attrs)) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + for (var g = 0, featLength = features.length; g < featLength; g++) { │ │ │ │ │ + features[g].geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return features │ │ │ │ │ + }, │ │ │ │ │ + extractSegment: function(segment, segmentType) { │ │ │ │ │ + var points = this.getElementsByTagNameNS(segment, segment.namespaceURI, segmentType); │ │ │ │ │ + var point_features = []; │ │ │ │ │ + for (var i = 0, len = points.length; i < len; i++) { │ │ │ │ │ + point_features.push(new OpenLayers.Geometry.Point(points[i].getAttribute("lon"), points[i].getAttribute("lat"))) │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.LineString(point_features) │ │ │ │ │ + }, │ │ │ │ │ + parseAttributes: function(node) { │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + var attrNode = node.firstChild, │ │ │ │ │ + value, name; │ │ │ │ │ + while (attrNode) { │ │ │ │ │ + if (attrNode.nodeType == 1 && attrNode.firstChild) { │ │ │ │ │ + value = attrNode.firstChild; │ │ │ │ │ + if (value.nodeType == 3 || value.nodeType == 4) { │ │ │ │ │ + name = attrNode.prefix ? attrNode.nodeName.split(":")[1] : attrNode.nodeName; │ │ │ │ │ + if (name != "trkseg" && name != "rtept") { │ │ │ │ │ + attributes[name] = value.nodeValue │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + attrNode = attrNode.nextSibling │ │ │ │ │ + } │ │ │ │ │ + return attributes │ │ │ │ │ + }, │ │ │ │ │ + write: function(features, metadata) { │ │ │ │ │ + features = OpenLayers.Util.isArray(features) ? features : [features]; │ │ │ │ │ + var gpx = this.createElementNS(this.namespaces.gpx, "gpx"); │ │ │ │ │ + gpx.setAttribute("version", "1.1"); │ │ │ │ │ + gpx.setAttribute("creator", this.creator); │ │ │ │ │ + this.setAttributes(gpx, { │ │ │ │ │ + "xsi:schemaLocation": this.schemaLocation │ │ │ │ │ + }); │ │ │ │ │ + if (metadata && typeof metadata == "object") { │ │ │ │ │ + gpx.appendChild(this.buildMetadataNode(metadata)) │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + gpx.appendChild(this.buildFeatureNode(features[i])) │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [gpx]) │ │ │ │ │ + }, │ │ │ │ │ + buildMetadataNode: function(metadata) { │ │ │ │ │ + var types = ["name", "desc", "author"], │ │ │ │ │ + node = this.createElementNS(this.namespaces.gpx, "metadata"); │ │ │ │ │ + for (var i = 0; i < types.length; i++) { │ │ │ │ │ + var type = types[i]; │ │ │ │ │ + if (metadata[type]) { │ │ │ │ │ + var n = this.createElementNS(this.namespaces.gpx, type); │ │ │ │ │ + n.appendChild(this.createTextNode(metadata[type])); │ │ │ │ │ + node.appendChild(n) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + buildFeatureNode: function(feature) { │ │ │ │ │ + var geometry = feature.geometry; │ │ │ │ │ + geometry = geometry.clone(); │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry.transform(this.internalProjection, this.externalProjection) │ │ │ │ │ + } │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + var wpt = this.buildWptNode(geometry); │ │ │ │ │ + this.appendAttributesNode(wpt, feature); │ │ │ │ │ + return wpt │ │ │ │ │ + } else { │ │ │ │ │ + var trkNode = this.createElementNS(this.namespaces.gpx, "trk"); │ │ │ │ │ + this.appendAttributesNode(trkNode, feature); │ │ │ │ │ + var trkSegNodes = this.buildTrkSegNode(geometry); │ │ │ │ │ + trkSegNodes = OpenLayers.Util.isArray(trkSegNodes) ? trkSegNodes : [trkSegNodes]; │ │ │ │ │ + for (var i = 0, len = trkSegNodes.length; i < len; i++) { │ │ │ │ │ + trkNode.appendChild(trkSegNodes[i]) │ │ │ │ │ + } │ │ │ │ │ + return trkNode │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + buildTrkSegNode: function(geometry) { │ │ │ │ │ + var node, i, len, point, nodes; │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.LineString" || geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ + node = this.createElementNS(this.namespaces.gpx, "trkseg"); │ │ │ │ │ + for (i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + point = geometry.components[i]; │ │ │ │ │ + node.appendChild(this.buildTrkPtNode(point)) │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ + } else { │ │ │ │ │ + nodes = []; │ │ │ │ │ + for (i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + nodes.push(this.buildTrkSegNode(geometry.components[i])) │ │ │ │ │ + } │ │ │ │ │ + return nodes │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + buildTrkPtNode: function(point) { │ │ │ │ │ + var node = this.createElementNS(this.namespaces.gpx, "trkpt"); │ │ │ │ │ + node.setAttribute("lon", point.x); │ │ │ │ │ + node.setAttribute("lat", point.y); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + buildWptNode: function(geometry) { │ │ │ │ │ + var node = this.createElementNS(this.namespaces.gpx, "wpt"); │ │ │ │ │ + node.setAttribute("lon", geometry.x); │ │ │ │ │ + node.setAttribute("lat", geometry.y); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + appendAttributesNode: function(node, feature) { │ │ │ │ │ + var name = this.createElementNS(this.namespaces.gpx, "name"); │ │ │ │ │ + name.appendChild(this.createTextNode(feature.attributes.name || feature.id)); │ │ │ │ │ + node.appendChild(name); │ │ │ │ │ + var desc = this.createElementNS(this.namespaces.gpx, "desc"); │ │ │ │ │ + desc.appendChild(this.createTextNode(feature.attributes.description || this.defaultDesc)); │ │ │ │ │ + node.appendChild(desc) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.GPX" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMC = OpenLayers.Class(OpenLayers.Format.Context, { │ │ │ │ │ + defaultVersion: "1.1.0", │ │ │ │ │ + layerToContext: function(layer) { │ │ │ │ │ + var parser = this.getParser(); │ │ │ │ │ + var layerContext = { │ │ │ │ │ + queryable: layer.queryable, │ │ │ │ │ + visibility: layer.visibility, │ │ │ │ │ + name: layer.params["LAYERS"], │ │ │ │ │ + title: layer.name, │ │ │ │ │ + abstract: layer.metadata["abstract"], │ │ │ │ │ + dataURL: layer.metadata.dataURL, │ │ │ │ │ + metadataURL: layer.metadataURL, │ │ │ │ │ + server: { │ │ │ │ │ + version: layer.params["VERSION"], │ │ │ │ │ + url: layer.url │ │ │ │ │ + }, │ │ │ │ │ + maxExtent: layer.maxExtent, │ │ │ │ │ + transparent: layer.params["TRANSPARENT"], │ │ │ │ │ + numZoomLevels: layer.numZoomLevels, │ │ │ │ │ + units: layer.units, │ │ │ │ │ + isBaseLayer: layer.isBaseLayer, │ │ │ │ │ + opacity: layer.opacity == 1 ? undefined : layer.opacity, │ │ │ │ │ + displayInLayerSwitcher: layer.displayInLayerSwitcher, │ │ │ │ │ + singleTile: layer.singleTile, │ │ │ │ │ + tileSize: layer.singleTile || !layer.tileSize ? undefined : { │ │ │ │ │ + width: layer.tileSize.w, │ │ │ │ │ + height: layer.tileSize.h │ │ │ │ │ + }, │ │ │ │ │ + minScale: layer.options.resolutions || layer.options.scales || layer.options.maxResolution || layer.options.minScale ? layer.minScale : undefined, │ │ │ │ │ + maxScale: layer.options.resolutions || layer.options.scales || layer.options.minResolution || layer.options.maxScale ? layer.maxScale : undefined, │ │ │ │ │ + formats: [], │ │ │ │ │ + styles: [], │ │ │ │ │ + srs: layer.srs, │ │ │ │ │ + dimensions: layer.dimensions │ │ │ │ │ + }; │ │ │ │ │ + if (layer.metadata.servertitle) { │ │ │ │ │ + layerContext.server.title = layer.metadata.servertitle │ │ │ │ │ + } │ │ │ │ │ + if (layer.metadata.formats && layer.metadata.formats.length > 0) { │ │ │ │ │ + for (var i = 0, len = layer.metadata.formats.length; i < len; i++) { │ │ │ │ │ + var format = layer.metadata.formats[i]; │ │ │ │ │ + layerContext.formats.push({ │ │ │ │ │ + value: format.value, │ │ │ │ │ + current: format.value == layer.params["FORMAT"] │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + layerContext.formats.push({ │ │ │ │ │ + value: layer.params["FORMAT"], │ │ │ │ │ + current: true │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + if (layer.metadata.styles && layer.metadata.styles.length > 0) { │ │ │ │ │ + for (var i = 0, len = layer.metadata.styles.length; i < len; i++) { │ │ │ │ │ + var style = layer.metadata.styles[i]; │ │ │ │ │ + if (style.href == layer.params["SLD"] || style.body == layer.params["SLD_BODY"] || style.name == layer.params["STYLES"]) { │ │ │ │ │ + style.current = true │ │ │ │ │ + } else { │ │ │ │ │ + style.current = false │ │ │ │ │ + } │ │ │ │ │ + layerContext.styles.push(style) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + layerContext.styles.push({ │ │ │ │ │ + href: layer.params["SLD"], │ │ │ │ │ + body: layer.params["SLD_BODY"], │ │ │ │ │ + name: layer.params["STYLES"] || parser.defaultStyleName, │ │ │ │ │ + title: parser.defaultStyleTitle, │ │ │ │ │ + current: true │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + return layerContext │ │ │ │ │ + }, │ │ │ │ │ + toContext: function(obj) { │ │ │ │ │ + var context = {}; │ │ │ │ │ + var layers = obj.layers; │ │ │ │ │ + if (obj.CLASS_NAME == "OpenLayers.Map") { │ │ │ │ │ + var metadata = obj.metadata || {}; │ │ │ │ │ + context.size = obj.getSize(); │ │ │ │ │ + context.bounds = obj.getExtent(); │ │ │ │ │ + context.projection = obj.projection; │ │ │ │ │ + context.title = obj.title; │ │ │ │ │ + context.keywords = metadata.keywords; │ │ │ │ │ + context["abstract"] = metadata["abstract"]; │ │ │ │ │ + context.logo = metadata.logo; │ │ │ │ │ + context.descriptionURL = metadata.descriptionURL; │ │ │ │ │ + context.contactInformation = metadata.contactInformation; │ │ │ │ │ + context.maxExtent = obj.maxExtent │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Util.applyDefaults(context, obj); │ │ │ │ │ + if (context.layers != undefined) { │ │ │ │ │ + delete context.layers │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (context.layersContext == undefined) { │ │ │ │ │ + context.layersContext = [] │ │ │ │ │ + } │ │ │ │ │ + if (layers != undefined && OpenLayers.Util.isArray(layers)) { │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + var layer = layers[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.WMS) { │ │ │ │ │ + context.layersContext.push(this.layerToContext(layer)) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return context │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMC" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WPSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + defaultVersion: "1.0.0", │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WPSCapabilities" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSGetFeatureInfo = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + layerIdentifier: "_layer", │ │ │ │ │ + featureIdentifier: "_feature", │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: /^\s*|\s*$/g, │ │ │ │ │ + removeSpace: /\s*/g, │ │ │ │ │ + splitSpace: /\s+/, │ │ │ │ │ + trimComma: /\s*,\s*/g │ │ │ │ │ + }, │ │ │ │ │ + gmlFormat: null, │ │ │ │ │ + read: function(data) { │ │ │ │ │ + var result; │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + var root = data.documentElement; │ │ │ │ │ + if (root) { │ │ │ │ │ + var scope = this; │ │ │ │ │ + var read = this["read_" + root.nodeName]; │ │ │ │ │ + if (read) { │ │ │ │ │ + result = read.call(this, root) │ │ │ │ │ + } else { │ │ │ │ │ + result = new OpenLayers.Format.GML(this.options ? this.options : {}).read(data) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + result = data │ │ │ │ │ + } │ │ │ │ │ + return result │ │ │ │ │ + }, │ │ │ │ │ + read_msGMLOutput: function(data) { │ │ │ │ │ + var response = []; │ │ │ │ │ + var layerNodes = this.getSiblingNodesByTagCriteria(data, this.layerIdentifier); │ │ │ │ │ + if (layerNodes) { │ │ │ │ │ + for (var i = 0, len = layerNodes.length; i < len; ++i) { │ │ │ │ │ + var node = layerNodes[i]; │ │ │ │ │ + var layerName = node.nodeName; │ │ │ │ │ + if (node.prefix) { │ │ │ │ │ + layerName = layerName.split(":")[1] │ │ │ │ │ + } │ │ │ │ │ + var layerName = layerName.replace(this.layerIdentifier, ""); │ │ │ │ │ + var featureNodes = this.getSiblingNodesByTagCriteria(node, this.featureIdentifier); │ │ │ │ │ + if (featureNodes) { │ │ │ │ │ + for (var j = 0; j < featureNodes.length; j++) { │ │ │ │ │ + var featureNode = featureNodes[j]; │ │ │ │ │ + var geomInfo = this.parseGeometry(featureNode); │ │ │ │ │ + var attributes = this.parseAttributes(featureNode); │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geomInfo.geometry, attributes, null); │ │ │ │ │ + feature.bounds = geomInfo.bounds; │ │ │ │ │ + feature.type = layerName; │ │ │ │ │ + response.push(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return response │ │ │ │ │ + }, │ │ │ │ │ + read_FeatureInfoResponse: function(data) { │ │ │ │ │ + var response = []; │ │ │ │ │ + var featureNodes = this.getElementsByTagNameNS(data, "*", "FIELDS"); │ │ │ │ │ + for (var i = 0, len = featureNodes.length; i < len; i++) { │ │ │ │ │ + var featureNode = featureNodes[i]; │ │ │ │ │ + var geom = null; │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + var j; │ │ │ │ │ + var jlen = featureNode.attributes.length; │ │ │ │ │ + if (jlen > 0) { │ │ │ │ │ + for (j = 0; j < jlen; j++) { │ │ │ │ │ + var attribute = featureNode.attributes[j]; │ │ │ │ │ + attributes[attribute.nodeName] = attribute.nodeValue │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + var nodes = featureNode.childNodes; │ │ │ │ │ + for (j = 0, jlen = nodes.length; j < jlen; ++j) { │ │ │ │ │ + var node = nodes[j]; │ │ │ │ │ + if (node.nodeType != 3) { │ │ │ │ │ + attributes[node.getAttribute("name")] = node.getAttribute("value") │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + response.push(new OpenLayers.Feature.Vector(geom, attributes, null)) │ │ │ │ │ + } │ │ │ │ │ + return response │ │ │ │ │ + }, │ │ │ │ │ + getSiblingNodesByTagCriteria: function(node, criteria) { │ │ │ │ │ + var nodes = []; │ │ │ │ │ + var children, tagName, n, matchNodes, child; │ │ │ │ │ + if (node && node.hasChildNodes()) { │ │ │ │ │ + children = node.childNodes; │ │ │ │ │ + n = children.length; │ │ │ │ │ + for (var k = 0; k < n; k++) { │ │ │ │ │ + child = children[k]; │ │ │ │ │ + while (child && child.nodeType != 1) { │ │ │ │ │ + child = child.nextSibling; │ │ │ │ │ + k++ │ │ │ │ │ + } │ │ │ │ │ + tagName = child ? child.nodeName : ""; │ │ │ │ │ + if (tagName.length > 0 && tagName.indexOf(criteria) > -1) { │ │ │ │ │ + nodes.push(child) │ │ │ │ │ + } else { │ │ │ │ │ + matchNodes = this.getSiblingNodesByTagCriteria(child, criteria); │ │ │ │ │ + if (matchNodes.length > 0) { │ │ │ │ │ + nodes.length == 0 ? nodes = matchNodes : nodes.push(matchNodes) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return nodes │ │ │ │ │ + }, │ │ │ │ │ + parseAttributes: function(node) { │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + if (node.nodeType == 1) { │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + var n = children.length; │ │ │ │ │ + for (var i = 0; i < n; ++i) { │ │ │ │ │ + var child = children[i]; │ │ │ │ │ + if (child.nodeType == 1) { │ │ │ │ │ + var grandchildren = child.childNodes; │ │ │ │ │ + var name = child.prefix ? child.nodeName.split(":")[1] : child.nodeName; │ │ │ │ │ + if (grandchildren.length == 0) { │ │ │ │ │ + attributes[name] = null │ │ │ │ │ + } else if (grandchildren.length == 1) { │ │ │ │ │ + var grandchild = grandchildren[0]; │ │ │ │ │ + if (grandchild.nodeType == 3 || grandchild.nodeType == 4) { │ │ │ │ │ + var value = grandchild.nodeValue.replace(this.regExes.trimSpace, ""); │ │ │ │ │ + attributes[name] = value │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return attributes │ │ │ │ │ + }, │ │ │ │ │ + parseGeometry: function(node) { │ │ │ │ │ + if (!this.gmlFormat) { │ │ │ │ │ + this.gmlFormat = new OpenLayers.Format.GML │ │ │ │ │ + } │ │ │ │ │ + var feature = this.gmlFormat.parseFeature(node); │ │ │ │ │ + var geometry, bounds = null; │ │ │ │ │ + if (feature) { │ │ │ │ │ + geometry = feature.geometry && feature.geometry.clone(); │ │ │ │ │ + bounds = feature.bounds && feature.bounds.clone(); │ │ │ │ │ + feature.destroy() │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + geometry: geometry, │ │ │ │ │ + bounds: bounds │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSGetFeatureInfo" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WFSDescribeFeatureType = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: /^\s*|\s*$/g │ │ │ │ │ + }, │ │ │ │ │ + namespaces: { │ │ │ │ │ + xsd: "http://www.w3.org/2001/XMLSchema" │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + xsd: { │ │ │ │ │ + schema: function(node, obj) { │ │ │ │ │ + var complexTypes = []; │ │ │ │ │ + var customTypes = {}; │ │ │ │ │ + var schema = { │ │ │ │ │ + complexTypes: complexTypes, │ │ │ │ │ + customTypes: customTypes │ │ │ │ │ + }; │ │ │ │ │ + var i, len; │ │ │ │ │ + this.readChildNodes(node, schema); │ │ │ │ │ + var attributes = node.attributes; │ │ │ │ │ + var attr, name; │ │ │ │ │ + for (i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ + attr = attributes[i]; │ │ │ │ │ + name = attr.name; │ │ │ │ │ + if (name.indexOf("xmlns") === 0) { │ │ │ │ │ + this.setNamespace(name.split(":")[1] || "", attr.value) │ │ │ │ │ + } else { │ │ │ │ │ + obj[name] = attr.value │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + obj.featureTypes = complexTypes; │ │ │ │ │ + obj.targetPrefix = this.namespaceAlias[obj.targetNamespace]; │ │ │ │ │ + var complexType, customType; │ │ │ │ │ + for (i = 0, len = complexTypes.length; i < len; ++i) { │ │ │ │ │ + complexType = complexTypes[i]; │ │ │ │ │ + customType = customTypes[complexType.typeName]; │ │ │ │ │ + if (customTypes[complexType.typeName]) { │ │ │ │ │ + complexType.typeName = customType.name │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + complexType: function(node, obj) { │ │ │ │ │ + var complexType = { │ │ │ │ │ + typeName: node.getAttribute("name") │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, complexType); │ │ │ │ │ + obj.complexTypes.push(complexType) │ │ │ │ │ + }, │ │ │ │ │ + complexContent: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + extension: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + sequence: function(node, obj) { │ │ │ │ │ + var sequence = { │ │ │ │ │ + elements: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, sequence); │ │ │ │ │ + obj.properties = sequence.elements │ │ │ │ │ + }, │ │ │ │ │ + element: function(node, obj) { │ │ │ │ │ + var type; │ │ │ │ │ + if (obj.elements) { │ │ │ │ │ + var element = {}; │ │ │ │ │ + var attributes = node.attributes; │ │ │ │ │ + var attr; │ │ │ │ │ + for (var i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ + attr = attributes[i]; │ │ │ │ │ + element[attr.name] = attr.value │ │ │ │ │ + } │ │ │ │ │ + type = element.type; │ │ │ │ │ + if (!type) { │ │ │ │ │ + type = {}; │ │ │ │ │ + this.readChildNodes(node, type); │ │ │ │ │ + element.restriction = type; │ │ │ │ │ + element.type = type.base │ │ │ │ │ + } │ │ │ │ │ + var fullType = type.base || type; │ │ │ │ │ + element.localType = fullType.split(":").pop(); │ │ │ │ │ + obj.elements.push(element); │ │ │ │ │ + this.readChildNodes(node, element) │ │ │ │ │ + } │ │ │ │ │ + if (obj.complexTypes) { │ │ │ │ │ + type = node.getAttribute("type"); │ │ │ │ │ + var localType = type.split(":").pop(); │ │ │ │ │ + obj.customTypes[localType] = { │ │ │ │ │ + name: node.getAttribute("name"), │ │ │ │ │ + type: type │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + annotation: function(node, obj) { │ │ │ │ │ + obj.annotation = {}; │ │ │ │ │ + this.readChildNodes(node, obj.annotation) │ │ │ │ │ + }, │ │ │ │ │ + appinfo: function(node, obj) { │ │ │ │ │ + if (!obj.appinfo) { │ │ │ │ │ + obj.appinfo = [] │ │ │ │ │ + } │ │ │ │ │ + obj.appinfo.push(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + documentation: function(node, obj) { │ │ │ │ │ + if (!obj.documentation) { │ │ │ │ │ + obj.documentation = [] │ │ │ │ │ + } │ │ │ │ │ + var value = this.getChildValue(node); │ │ │ │ │ + obj.documentation.push({ │ │ │ │ │ + lang: node.getAttribute("xml:lang"), │ │ │ │ │ + textContent: value.replace(this.regExes.trimSpace, "") │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + simpleType: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + restriction: function(node, obj) { │ │ │ │ │ + obj.base = node.getAttribute("base"); │ │ │ │ │ + this.readRestriction(node, obj) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + readRestriction: function(node, obj) { │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + var child, nodeName, value; │ │ │ │ │ + for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ + child = children[i]; │ │ │ │ │ + if (child.nodeType == 1) { │ │ │ │ │ + nodeName = child.nodeName.split(":").pop(); │ │ │ │ │ + value = child.getAttribute("value"); │ │ │ │ │ + if (!obj[nodeName]) { │ │ │ │ │ + obj[nodeName] = value │ │ │ │ │ + } else { │ │ │ │ │ + if (typeof obj[nodeName] == "string") { │ │ │ │ │ + obj[nodeName] = [obj[nodeName]] │ │ │ │ │ + } │ │ │ │ │ + obj[nodeName].push(value) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement │ │ │ │ │ + } │ │ │ │ │ + var schema = {}; │ │ │ │ │ + if (data.nodeName.split(":").pop() === "ExceptionReport") { │ │ │ │ │ + var parser = new OpenLayers.Format.OGCExceptionReport; │ │ │ │ │ + schema.error = parser.read(data) │ │ │ │ │ + } else { │ │ │ │ │ + this.readNode(data, schema) │ │ │ │ │ + } │ │ │ │ │ + return schema │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSDescribeFeatureType" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.CSWGetDomain = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, OpenLayers.Format.CSWGetDomain.DEFAULTS); │ │ │ │ │ + var cls = OpenLayers.Format.CSWGetDomain["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported CSWGetDomain version: " + options.version │ │ │ │ │ + } │ │ │ │ │ + return new cls(options) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Format.CSWGetDomain.DEFAULTS = { │ │ │ │ │ + version: "2.0.2" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Format.WMTSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + defaultVersion: "1.0.0", │ │ │ │ │ + yx: { │ │ │ │ │ + "urn:ogc:def:crs:EPSG::4326": true │ │ │ │ │ + }, │ │ │ │ │ + createLayer: function(capabilities, config) { │ │ │ │ │ + var layer; │ │ │ │ │ + if (!("layer" in config)) { │ │ │ │ │ + throw new Error("Missing property 'layer' in configuration.") │ │ │ │ │ + } │ │ │ │ │ + var contents = capabilities.contents; │ │ │ │ │ + var layers = contents.layers; │ │ │ │ │ + var layerDef; │ │ │ │ │ + for (var i = 0, ii = contents.layers.length; i < ii; ++i) { │ │ │ │ │ + if (contents.layers[i].identifier === config.layer) { │ │ │ │ │ + layerDef = contents.layers[i]; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!layerDef) { │ │ │ │ │ + throw new Error("Layer not found") │ │ │ │ │ + } │ │ │ │ │ + var format = config.format; │ │ │ │ │ + if (!format && layerDef.formats && layerDef.formats.length) { │ │ │ │ │ + format = layerDef.formats[0] │ │ │ │ │ + } │ │ │ │ │ + var matrixSet; │ │ │ │ │ + if (config.matrixSet) { │ │ │ │ │ + matrixSet = contents.tileMatrixSets[config.matrixSet] │ │ │ │ │ + } else if (layerDef.tileMatrixSetLinks.length >= 1) { │ │ │ │ │ + matrixSet = contents.tileMatrixSets[layerDef.tileMatrixSetLinks[0].tileMatrixSet] │ │ │ │ │ + } │ │ │ │ │ + if (!matrixSet) { │ │ │ │ │ + throw new Error("matrixSet not found") │ │ │ │ │ + } │ │ │ │ │ + var style; │ │ │ │ │ + for (var i = 0, ii = layerDef.styles.length; i < ii; ++i) { │ │ │ │ │ + style = layerDef.styles[i]; │ │ │ │ │ + if (style.isDefault) { │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var requestEncoding = config.requestEncoding; │ │ │ │ │ + if (!requestEncoding) { │ │ │ │ │ + requestEncoding = "KVP"; │ │ │ │ │ + if (capabilities.operationsMetadata.GetTile.dcp.http) { │ │ │ │ │ + var http = capabilities.operationsMetadata.GetTile.dcp.http; │ │ │ │ │ + if (http.get[0].constraints) { │ │ │ │ │ + var constraints = http.get[0].constraints; │ │ │ │ │ + var allowedValues = constraints.GetEncoding.allowedValues; │ │ │ │ │ + if (!allowedValues.KVP && (allowedValues.REST || allowedValues.RESTful)) { │ │ │ │ │ + requestEncoding = "REST" │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var dimensions = []; │ │ │ │ │ + var params = config.params || {}; │ │ │ │ │ + delete config.params; │ │ │ │ │ + for (var id = 0, ld = layerDef.dimensions.length; id < ld; id++) { │ │ │ │ │ + var dimension = layerDef.dimensions[id]; │ │ │ │ │ + dimensions.push(dimension.identifier); │ │ │ │ │ + if (!params.hasOwnProperty(dimension.identifier)) { │ │ │ │ │ + params[dimension.identifier] = dimension["default"] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var projection = config.projection || matrixSet.supportedCRS.replace(/urn:ogc:def:crs:(\w+):(.*:)?(\w+)$/, "$1:$3"); │ │ │ │ │ + var units = config.units || (projection === "EPSG:4326" ? "degrees" : "m"); │ │ │ │ │ + var resolutions = []; │ │ │ │ │ + for (var mid in matrixSet.matrixIds) { │ │ │ │ │ + if (matrixSet.matrixIds.hasOwnProperty(mid)) { │ │ │ │ │ + resolutions.push(matrixSet.matrixIds[mid].scaleDenominator * 28e-5 / OpenLayers.METERS_PER_INCH / OpenLayers.INCHES_PER_UNIT[units]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var url; │ │ │ │ │ + if (requestEncoding === "REST" && layerDef.resourceUrls) { │ │ │ │ │ + url = []; │ │ │ │ │ + var resourceUrls = layerDef.resourceUrls, │ │ │ │ │ + resourceUrl; │ │ │ │ │ + for (var t = 0, tt = layerDef.resourceUrls.length; t < tt; ++t) { │ │ │ │ │ + resourceUrl = layerDef.resourceUrls[t]; │ │ │ │ │ + if (resourceUrl.format === format && resourceUrl.resourceType === "tile") { │ │ │ │ │ + url.push(resourceUrl.template) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + var httpGet = capabilities.operationsMetadata.GetTile.dcp.http.get; │ │ │ │ │ + url = []; │ │ │ │ │ + var constraint; │ │ │ │ │ + for (var i = 0, ii = httpGet.length; i < ii; i++) { │ │ │ │ │ + constraint = httpGet[i].constraints; │ │ │ │ │ + if (!constraint || constraint && constraint.GetEncoding.allowedValues[requestEncoding]) { │ │ │ │ │ + url.push(httpGet[i].url) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Layer.WMTS(OpenLayers.Util.applyDefaults(config, { │ │ │ │ │ + url: url, │ │ │ │ │ + requestEncoding: requestEncoding, │ │ │ │ │ + name: layerDef.title, │ │ │ │ │ + style: style.identifier, │ │ │ │ │ + format: format, │ │ │ │ │ + matrixIds: matrixSet.matrixIds, │ │ │ │ │ + matrixSet: matrixSet.identifier, │ │ │ │ │ + projection: projection, │ │ │ │ │ + units: units, │ │ │ │ │ + resolutions: config.isBaseLayer === false ? undefined : resolutions, │ │ │ │ │ + serverResolutions: resolutions, │ │ │ │ │ + tileFullExtent: matrixSet.bounds, │ │ │ │ │ + dimensions: dimensions, │ │ │ │ │ + params: params │ │ │ │ │ + })) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMTSCapabilities" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.OSM = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + checkTags: false, │ │ │ │ │ + interestingTagsExclude: null, │ │ │ │ │ + areaTags: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + var layer_defaults = { │ │ │ │ │ + interestingTagsExclude: ["source", "source_ref", "source:ref", "history", "attribution", "created_by"], │ │ │ │ │ + areaTags: ["area", "building", "leisure", "tourism", "ruins", "historic", "landuse", "military", "natural", "sport"] │ │ │ │ │ + }; │ │ │ │ │ + layer_defaults = OpenLayers.Util.extend(layer_defaults, options); │ │ │ │ │ + var interesting = {}; │ │ │ │ │ + for (var i = 0; i < layer_defaults.interestingTagsExclude.length; i++) { │ │ │ │ │ + interesting[layer_defaults.interestingTagsExclude[i]] = true │ │ │ │ │ + } │ │ │ │ │ + layer_defaults.interestingTagsExclude = interesting; │ │ │ │ │ + var area = {}; │ │ │ │ │ + for (var i = 0; i < layer_defaults.areaTags.length; i++) { │ │ │ │ │ + area[layer_defaults.areaTags[i]] = true │ │ │ │ │ + } │ │ │ │ │ + layer_defaults.areaTags = area; │ │ │ │ │ + this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [layer_defaults]) │ │ │ │ │ + }, │ │ │ │ │ + read: function(doc) { │ │ │ │ │ + if (typeof doc == "string") { │ │ │ │ │ + doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]) │ │ │ │ │ + } │ │ │ │ │ + var nodes = this.getNodes(doc); │ │ │ │ │ + var ways = this.getWays(doc); │ │ │ │ │ + var feat_list = new Array(ways.length); │ │ │ │ │ + for (var i = 0; i < ways.length; i++) { │ │ │ │ │ + var point_list = new Array(ways[i].nodes.length); │ │ │ │ │ + var poly = this.isWayArea(ways[i]) ? 1 : 0; │ │ │ │ │ + for (var j = 0; j < ways[i].nodes.length; j++) { │ │ │ │ │ + var node = nodes[ways[i].nodes[j]]; │ │ │ │ │ + var point = new OpenLayers.Geometry.Point(node.lon, node.lat); │ │ │ │ │ + point.osm_id = parseInt(ways[i].nodes[j]); │ │ │ │ │ + point_list[j] = point; │ │ │ │ │ + node.used = true │ │ │ │ │ + } │ │ │ │ │ + var geometry = null; │ │ │ │ │ + if (poly) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.Polygon(new OpenLayers.Geometry.LinearRing(point_list)) │ │ │ │ │ + } else { │ │ │ │ │ + geometry = new OpenLayers.Geometry.LineString(point_list) │ │ │ │ │ + } │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ + } │ │ │ │ │ + var feat = new OpenLayers.Feature.Vector(geometry, ways[i].tags); │ │ │ │ │ + feat.osm_id = parseInt(ways[i].id); │ │ │ │ │ + feat.fid = "way." + feat.osm_id; │ │ │ │ │ + feat_list[i] = feat │ │ │ │ │ + } │ │ │ │ │ + for (var node_id in nodes) { │ │ │ │ │ + var node = nodes[node_id]; │ │ │ │ │ + if (!node.used || this.checkTags) { │ │ │ │ │ + var tags = null; │ │ │ │ │ + if (this.checkTags) { │ │ │ │ │ + var result = this.getTags(node.node, true); │ │ │ │ │ + if (node.used && !result[1]) { │ │ │ │ │ + continue │ │ │ │ │ + } │ │ │ │ │ + tags = result[0] │ │ │ │ │ + } else { │ │ │ │ │ + tags = this.getTags(node.node) │ │ │ │ │ + } │ │ │ │ │ + var feat = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Point(node["lon"], node["lat"]), tags); │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + feat.geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ + } │ │ │ │ │ + feat.osm_id = parseInt(node_id); │ │ │ │ │ + feat.fid = "node." + feat.osm_id; │ │ │ │ │ + feat_list.push(feat) │ │ │ │ │ + } │ │ │ │ │ + node.node = null │ │ │ │ │ + } │ │ │ │ │ + return feat_list │ │ │ │ │ + }, │ │ │ │ │ + getNodes: function(doc) { │ │ │ │ │ + var node_list = doc.getElementsByTagName("node"); │ │ │ │ │ + var nodes = {}; │ │ │ │ │ + for (var i = 0; i < node_list.length; i++) { │ │ │ │ │ + var node = node_list[i]; │ │ │ │ │ + var id = node.getAttribute("id"); │ │ │ │ │ + nodes[id] = { │ │ │ │ │ + lat: node.getAttribute("lat"), │ │ │ │ │ + lon: node.getAttribute("lon"), │ │ │ │ │ + node: node │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return nodes │ │ │ │ │ + }, │ │ │ │ │ + getWays: function(doc) { │ │ │ │ │ + var way_list = doc.getElementsByTagName("way"); │ │ │ │ │ + var return_ways = []; │ │ │ │ │ + for (var i = 0; i < way_list.length; i++) { │ │ │ │ │ + var way = way_list[i]; │ │ │ │ │ + var way_object = { │ │ │ │ │ + id: way.getAttribute("id") │ │ │ │ │ + }; │ │ │ │ │ + way_object.tags = this.getTags(way); │ │ │ │ │ + var node_list = way.getElementsByTagName("nd"); │ │ │ │ │ + way_object.nodes = new Array(node_list.length); │ │ │ │ │ + for (var j = 0; j < node_list.length; j++) { │ │ │ │ │ + way_object.nodes[j] = node_list[j].getAttribute("ref") │ │ │ │ │ + } │ │ │ │ │ + return_ways.push(way_object) │ │ │ │ │ + } │ │ │ │ │ + return return_ways │ │ │ │ │ + }, │ │ │ │ │ + getTags: function(dom_node, interesting_tags) { │ │ │ │ │ + var tag_list = dom_node.getElementsByTagName("tag"); │ │ │ │ │ + var tags = {}; │ │ │ │ │ + var interesting = false; │ │ │ │ │ + for (var j = 0; j < tag_list.length; j++) { │ │ │ │ │ + var key = tag_list[j].getAttribute("k"); │ │ │ │ │ + tags[key] = tag_list[j].getAttribute("v"); │ │ │ │ │ + if (interesting_tags) { │ │ │ │ │ + if (!this.interestingTagsExclude[key]) { │ │ │ │ │ + interesting = true │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return interesting_tags ? [tags, interesting] : tags │ │ │ │ │ + }, │ │ │ │ │ + isWayArea: function(way) { │ │ │ │ │ + var poly_shaped = false; │ │ │ │ │ + var poly_tags = false; │ │ │ │ │ + if (way.nodes[0] == way.nodes[way.nodes.length - 1]) { │ │ │ │ │ + poly_shaped = true │ │ │ │ │ + } │ │ │ │ │ + if (this.checkTags) { │ │ │ │ │ + for (var key in way.tags) { │ │ │ │ │ + if (this.areaTags[key]) { │ │ │ │ │ + poly_tags = true; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return poly_shaped && (this.checkTags ? poly_tags : true) │ │ │ │ │ + }, │ │ │ │ │ + write: function(features) { │ │ │ │ │ + if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ + features = [features] │ │ │ │ │ + } │ │ │ │ │ + this.osm_id = 1; │ │ │ │ │ + this.created_nodes = {}; │ │ │ │ │ + var root_node = this.createElementNS(null, "osm"); │ │ │ │ │ + root_node.setAttribute("version", "0.5"); │ │ │ │ │ + root_node.setAttribute("generator", "OpenLayers " + OpenLayers.VERSION_NUMBER); │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + var nodes = this.createFeatureNodes(features[i]); │ │ │ │ │ + for (var j = 0; j < nodes.length; j++) { │ │ │ │ │ + root_node.appendChild(nodes[j]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [root_node]) │ │ │ │ │ + }, │ │ │ │ │ + createFeatureNodes: function(feature) { │ │ │ │ │ + var nodes = []; │ │ │ │ │ + var className = feature.geometry.CLASS_NAME; │ │ │ │ │ + var type = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ + type = type.toLowerCase(); │ │ │ │ │ + var builder = this.createXML[type]; │ │ │ │ │ + if (builder) { │ │ │ │ │ + nodes = builder.apply(this, [feature]) │ │ │ │ │ + } │ │ │ │ │ + return nodes │ │ │ │ │ + }, │ │ │ │ │ + createXML: { │ │ │ │ │ + point: function(point) { │ │ │ │ │ + var id = null; │ │ │ │ │ + var geometry = point.geometry ? point.geometry : point; │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry = geometry.clone(); │ │ │ │ │ + geometry.transform(this.internalProjection, this.externalProjection) │ │ │ │ │ + } │ │ │ │ │ + var already_exists = false; │ │ │ │ │ + if (point.osm_id) { │ │ │ │ │ + id = point.osm_id; │ │ │ │ │ + if (this.created_nodes[id]) { │ │ │ │ │ + already_exists = true │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + id = -this.osm_id; │ │ │ │ │ + this.osm_id++ │ │ │ │ │ + } │ │ │ │ │ + if (already_exists) { │ │ │ │ │ + node = this.created_nodes[id] │ │ │ │ │ + } else { │ │ │ │ │ + var node = this.createElementNS(null, "node") │ │ │ │ │ + } │ │ │ │ │ + this.created_nodes[id] = node; │ │ │ │ │ + node.setAttribute("id", id); │ │ │ │ │ + node.setAttribute("lon", geometry.x); │ │ │ │ │ + node.setAttribute("lat", geometry.y); │ │ │ │ │ + if (point.attributes) { │ │ │ │ │ + this.serializeTags(point, node) │ │ │ │ │ + } │ │ │ │ │ + this.setState(point, node); │ │ │ │ │ + return already_exists ? [] : [node] │ │ │ │ │ + }, │ │ │ │ │ + linestring: function(feature) { │ │ │ │ │ + var id; │ │ │ │ │ + var nodes = []; │ │ │ │ │ + var geometry = feature.geometry; │ │ │ │ │ + if (feature.osm_id) { │ │ │ │ │ + id = feature.osm_id │ │ │ │ │ } else { │ │ │ │ │ id = -this.osm_id; │ │ │ │ │ this.osm_id++ │ │ │ │ │ } │ │ │ │ │ var way = this.createElementNS(null, "way"); │ │ │ │ │ way.setAttribute("id", id); │ │ │ │ │ for (var i = 0; i < geometry.components.length; i++) { │ │ │ │ │ @@ -19537,35 +20832,129 @@ │ │ │ │ │ if (state) { │ │ │ │ │ node.setAttribute("action", state) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.OSM" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.OWSContext = OpenLayers.Class(OpenLayers.Format.Context, { │ │ │ │ │ - defaultVersion: "0.3.1", │ │ │ │ │ - getVersion: function(root, options) { │ │ │ │ │ - var version = OpenLayers.Format.XML.VersionedOGC.prototype.getVersion.apply(this, arguments); │ │ │ │ │ - if (version === "0.3.0") { │ │ │ │ │ - version = this.defaultVersion │ │ │ │ │ +OpenLayers.Format.WFS = OpenLayers.Class(OpenLayers.Format.GML, { │ │ │ │ │ + layer: null, │ │ │ │ │ + wfsns: "http://www.opengis.net/wfs", │ │ │ │ │ + ogcns: "http://www.opengis.net/ogc", │ │ │ │ │ + initialize: function(options, layer) { │ │ │ │ │ + OpenLayers.Format.GML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + if (this.layer.featureNS) { │ │ │ │ │ + this.featureNS = this.layer.featureNS │ │ │ │ │ + } │ │ │ │ │ + if (this.layer.options.geometry_column) { │ │ │ │ │ + this.geometryName = this.layer.options.geometry_column │ │ │ │ │ + } │ │ │ │ │ + if (this.layer.options.typename) { │ │ │ │ │ + this.featureName = this.layer.options.typename │ │ │ │ │ } │ │ │ │ │ - return version │ │ │ │ │ }, │ │ │ │ │ - toContext: function(obj) { │ │ │ │ │ - var context = {}; │ │ │ │ │ - if (obj.CLASS_NAME == "OpenLayers.Map") { │ │ │ │ │ - context.bounds = obj.getExtent(); │ │ │ │ │ - context.maxExtent = obj.maxExtent; │ │ │ │ │ - context.projection = obj.projection; │ │ │ │ │ - context.size = obj.getSize(); │ │ │ │ │ - context.layers = obj.layers │ │ │ │ │ + write: function(features) { │ │ │ │ │ + var transaction = this.createElementNS(this.wfsns, "wfs:Transaction"); │ │ │ │ │ + transaction.setAttribute("version", "1.0.0"); │ │ │ │ │ + transaction.setAttribute("service", "WFS"); │ │ │ │ │ + for (var i = 0; i < features.length; i++) { │ │ │ │ │ + switch (features[i].state) { │ │ │ │ │ + case OpenLayers.State.INSERT: │ │ │ │ │ + transaction.appendChild(this.insert(features[i])); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.State.UPDATE: │ │ │ │ │ + transaction.appendChild(this.update(features[i])); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.State.DELETE: │ │ │ │ │ + transaction.appendChild(this.remove(features[i])); │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return context │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [transaction]) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.OWSContext" │ │ │ │ │ + createFeatureXML: function(feature) { │ │ │ │ │ + var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ + var geomContainer = this.createElementNS(this.featureNS, "feature:" + this.geometryName); │ │ │ │ │ + geomContainer.appendChild(geometryNode); │ │ │ │ │ + var featureContainer = this.createElementNS(this.featureNS, "feature:" + this.featureName); │ │ │ │ │ + featureContainer.appendChild(geomContainer); │ │ │ │ │ + for (var attr in feature.attributes) { │ │ │ │ │ + var attrText = this.createTextNode(feature.attributes[attr]); │ │ │ │ │ + var nodename = attr; │ │ │ │ │ + if (attr.search(":") != -1) { │ │ │ │ │ + nodename = attr.split(":")[1] │ │ │ │ │ + } │ │ │ │ │ + var attrContainer = this.createElementNS(this.featureNS, "feature:" + nodename); │ │ │ │ │ + attrContainer.appendChild(attrText); │ │ │ │ │ + featureContainer.appendChild(attrContainer) │ │ │ │ │ + } │ │ │ │ │ + return featureContainer │ │ │ │ │ + }, │ │ │ │ │ + insert: function(feature) { │ │ │ │ │ + var insertNode = this.createElementNS(this.wfsns, "wfs:Insert"); │ │ │ │ │ + insertNode.appendChild(this.createFeatureXML(feature)); │ │ │ │ │ + return insertNode │ │ │ │ │ + }, │ │ │ │ │ + update: function(feature) { │ │ │ │ │ + if (!feature.fid) { │ │ │ │ │ + OpenLayers.Console.userError(OpenLayers.i18n("noFID")) │ │ │ │ │ + } │ │ │ │ │ + var updateNode = this.createElementNS(this.wfsns, "wfs:Update"); │ │ │ │ │ + updateNode.setAttribute("typeName", this.featurePrefix + ":" + this.featureName); │ │ │ │ │ + updateNode.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ + var propertyNode = this.createElementNS(this.wfsns, "wfs:Property"); │ │ │ │ │ + var nameNode = this.createElementNS(this.wfsns, "wfs:Name"); │ │ │ │ │ + var txtNode = this.createTextNode(this.geometryName); │ │ │ │ │ + nameNode.appendChild(txtNode); │ │ │ │ │ + propertyNode.appendChild(nameNode); │ │ │ │ │ + var valueNode = this.createElementNS(this.wfsns, "wfs:Value"); │ │ │ │ │ + var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ + if (feature.layer) { │ │ │ │ │ + geometryNode.setAttribute("srsName", feature.layer.projection.getCode()) │ │ │ │ │ + } │ │ │ │ │ + valueNode.appendChild(geometryNode); │ │ │ │ │ + propertyNode.appendChild(valueNode); │ │ │ │ │ + updateNode.appendChild(propertyNode); │ │ │ │ │ + for (var propName in feature.attributes) { │ │ │ │ │ + propertyNode = this.createElementNS(this.wfsns, "wfs:Property"); │ │ │ │ │ + nameNode = this.createElementNS(this.wfsns, "wfs:Name"); │ │ │ │ │ + nameNode.appendChild(this.createTextNode(propName)); │ │ │ │ │ + propertyNode.appendChild(nameNode); │ │ │ │ │ + valueNode = this.createElementNS(this.wfsns, "wfs:Value"); │ │ │ │ │ + valueNode.appendChild(this.createTextNode(feature.attributes[propName])); │ │ │ │ │ + propertyNode.appendChild(valueNode); │ │ │ │ │ + updateNode.appendChild(propertyNode) │ │ │ │ │ + } │ │ │ │ │ + var filterNode = this.createElementNS(this.ogcns, "ogc:Filter"); │ │ │ │ │ + var filterIdNode = this.createElementNS(this.ogcns, "ogc:FeatureId"); │ │ │ │ │ + filterIdNode.setAttribute("fid", feature.fid); │ │ │ │ │ + filterNode.appendChild(filterIdNode); │ │ │ │ │ + updateNode.appendChild(filterNode); │ │ │ │ │ + return updateNode │ │ │ │ │ + }, │ │ │ │ │ + remove: function(feature) { │ │ │ │ │ + if (!feature.fid) { │ │ │ │ │ + OpenLayers.Console.userError(OpenLayers.i18n("noFID")); │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + var deleteNode = this.createElementNS(this.wfsns, "wfs:Delete"); │ │ │ │ │ + deleteNode.setAttribute("typeName", this.featurePrefix + ":" + this.featureName); │ │ │ │ │ + deleteNode.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ + var filterNode = this.createElementNS(this.ogcns, "ogc:Filter"); │ │ │ │ │ + var filterIdNode = this.createElementNS(this.ogcns, "ogc:FeatureId"); │ │ │ │ │ + filterIdNode.setAttribute("fid", feature.fid); │ │ │ │ │ + filterNode.appendChild(filterIdNode); │ │ │ │ │ + deleteNode.appendChild(filterNode); │ │ │ │ │ + return deleteNode │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.layer = null │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFS" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Format.EncodedPolyline = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ geometryType: "linestring", │ │ │ │ │ initialize: function(options) { │ │ │ │ │ OpenLayers.Format.prototype.initialize.apply(this, [options]) │ │ │ │ │ }, │ │ │ │ │ read: function(encoded) { │ │ │ │ │ @@ -19771,263 +21160,377 @@ │ │ │ │ │ if (b < 32) break; │ │ │ │ │ shift += 5 │ │ │ │ │ } │ │ │ │ │ return result │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.EncodedPolyline" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.WMSGetFeatureInfo = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - layerIdentifier: "_layer", │ │ │ │ │ - featureIdentifier: "_feature", │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: /^\s*|\s*$/g, │ │ │ │ │ - removeSpace: /\s*/g, │ │ │ │ │ - splitSpace: /\s+/, │ │ │ │ │ - trimComma: /\s*,\s*/g │ │ │ │ │ - }, │ │ │ │ │ - gmlFormat: null, │ │ │ │ │ - read: function(data) { │ │ │ │ │ - var result; │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ +OpenLayers.Format.SOSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + defaultVersion: "1.0.0", │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SOSCapabilities" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.CQL = function() { │ │ │ │ │ + var tokens = ["PROPERTY", "COMPARISON", "VALUE", "LOGICAL"], │ │ │ │ │ + patterns = { │ │ │ │ │ + PROPERTY: /^[_a-zA-Z]\w*/, │ │ │ │ │ + COMPARISON: /^(=|<>|<=|<|>=|>|LIKE)/i, │ │ │ │ │ + IS_NULL: /^IS NULL/i, │ │ │ │ │ + COMMA: /^,/, │ │ │ │ │ + LOGICAL: /^(AND|OR)/i, │ │ │ │ │ + VALUE: /^('([^']|'')*'|\d+(\.\d*)?|\.\d+)/, │ │ │ │ │ + LPAREN: /^\(/, │ │ │ │ │ + RPAREN: /^\)/, │ │ │ │ │ + SPATIAL: /^(BBOX|INTERSECTS|DWITHIN|WITHIN|CONTAINS)/i, │ │ │ │ │ + NOT: /^NOT/i, │ │ │ │ │ + BETWEEN: /^BETWEEN/i, │ │ │ │ │ + GEOMETRY: function(text) { │ │ │ │ │ + var type = /^(POINT|LINESTRING|POLYGON|MULTIPOINT|MULTILINESTRING|MULTIPOLYGON|GEOMETRYCOLLECTION)/.exec(text); │ │ │ │ │ + if (type) { │ │ │ │ │ + var len = text.length; │ │ │ │ │ + var idx = text.indexOf("(", type[0].length); │ │ │ │ │ + if (idx > -1) { │ │ │ │ │ + var depth = 1; │ │ │ │ │ + while (idx < len && depth > 0) { │ │ │ │ │ + idx++; │ │ │ │ │ + switch (text.charAt(idx)) { │ │ │ │ │ + case "(": │ │ │ │ │ + depth++; │ │ │ │ │ + break; │ │ │ │ │ + case ")": │ │ │ │ │ + depth--; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return [text.substr(0, idx + 1)] │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + END: /^$/ │ │ │ │ │ + }, │ │ │ │ │ + follows = { │ │ │ │ │ + LPAREN: ["GEOMETRY", "SPATIAL", "PROPERTY", "VALUE", "LPAREN"], │ │ │ │ │ + RPAREN: ["NOT", "LOGICAL", "END", "RPAREN"], │ │ │ │ │ + PROPERTY: ["COMPARISON", "BETWEEN", "COMMA", "IS_NULL"], │ │ │ │ │ + BETWEEN: ["VALUE"], │ │ │ │ │ + IS_NULL: ["END"], │ │ │ │ │ + COMPARISON: ["VALUE"], │ │ │ │ │ + COMMA: ["GEOMETRY", "VALUE", "PROPERTY"], │ │ │ │ │ + VALUE: ["LOGICAL", "COMMA", "RPAREN", "END"], │ │ │ │ │ + SPATIAL: ["LPAREN"], │ │ │ │ │ + LOGICAL: ["NOT", "VALUE", "SPATIAL", "PROPERTY", "LPAREN"], │ │ │ │ │ + NOT: ["PROPERTY", "LPAREN"], │ │ │ │ │ + GEOMETRY: ["COMMA", "RPAREN"] │ │ │ │ │ + }, │ │ │ │ │ + operators = { │ │ │ │ │ + "=": OpenLayers.Filter.Comparison.EQUAL_TO, │ │ │ │ │ + "<>": OpenLayers.Filter.Comparison.NOT_EQUAL_TO, │ │ │ │ │ + "<": OpenLayers.Filter.Comparison.LESS_THAN, │ │ │ │ │ + "<=": OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO, │ │ │ │ │ + ">": OpenLayers.Filter.Comparison.GREATER_THAN, │ │ │ │ │ + ">=": OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO, │ │ │ │ │ + LIKE: OpenLayers.Filter.Comparison.LIKE, │ │ │ │ │ + BETWEEN: OpenLayers.Filter.Comparison.BETWEEN, │ │ │ │ │ + "IS NULL": OpenLayers.Filter.Comparison.IS_NULL │ │ │ │ │ + }, │ │ │ │ │ + operatorReverse = {}, │ │ │ │ │ + logicals = { │ │ │ │ │ + AND: OpenLayers.Filter.Logical.AND, │ │ │ │ │ + OR: OpenLayers.Filter.Logical.OR │ │ │ │ │ + }, │ │ │ │ │ + logicalReverse = {}, │ │ │ │ │ + precedence = { │ │ │ │ │ + RPAREN: 3, │ │ │ │ │ + LOGICAL: 2, │ │ │ │ │ + COMPARISON: 1 │ │ │ │ │ + }; │ │ │ │ │ + var i; │ │ │ │ │ + for (i in operators) { │ │ │ │ │ + if (operators.hasOwnProperty(i)) { │ │ │ │ │ + operatorReverse[operators[i]] = i │ │ │ │ │ } │ │ │ │ │ - var root = data.documentElement; │ │ │ │ │ - if (root) { │ │ │ │ │ - var scope = this; │ │ │ │ │ - var read = this["read_" + root.nodeName]; │ │ │ │ │ - if (read) { │ │ │ │ │ - result = read.call(this, root) │ │ │ │ │ - } else { │ │ │ │ │ - result = new OpenLayers.Format.GML(this.options ? this.options : {}).read(data) │ │ │ │ │ - } │ │ │ │ │ + } │ │ │ │ │ + for (i in logicals) { │ │ │ │ │ + if (logicals.hasOwnProperty(i)) { │ │ │ │ │ + logicalReverse[logicals[i]] = i │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function tryToken(text, pattern) { │ │ │ │ │ + if (pattern instanceof RegExp) { │ │ │ │ │ + return pattern.exec(text) │ │ │ │ │ } else { │ │ │ │ │ - result = data │ │ │ │ │ + return pattern(text) │ │ │ │ │ } │ │ │ │ │ - return result │ │ │ │ │ - }, │ │ │ │ │ - read_msGMLOutput: function(data) { │ │ │ │ │ - var response = []; │ │ │ │ │ - var layerNodes = this.getSiblingNodesByTagCriteria(data, this.layerIdentifier); │ │ │ │ │ - if (layerNodes) { │ │ │ │ │ - for (var i = 0, len = layerNodes.length; i < len; ++i) { │ │ │ │ │ - var node = layerNodes[i]; │ │ │ │ │ - var layerName = node.nodeName; │ │ │ │ │ - if (node.prefix) { │ │ │ │ │ - layerName = layerName.split(":")[1] │ │ │ │ │ - } │ │ │ │ │ - var layerName = layerName.replace(this.layerIdentifier, ""); │ │ │ │ │ - var featureNodes = this.getSiblingNodesByTagCriteria(node, this.featureIdentifier); │ │ │ │ │ - if (featureNodes) { │ │ │ │ │ - for (var j = 0; j < featureNodes.length; j++) { │ │ │ │ │ - var featureNode = featureNodes[j]; │ │ │ │ │ - var geomInfo = this.parseGeometry(featureNode); │ │ │ │ │ - var attributes = this.parseAttributes(featureNode); │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geomInfo.geometry, attributes, null); │ │ │ │ │ - feature.bounds = geomInfo.bounds; │ │ │ │ │ - feature.type = layerName; │ │ │ │ │ - response.push(feature) │ │ │ │ │ - } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function nextToken(text, tokens) { │ │ │ │ │ + var i, token, len = tokens.length; │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + token = tokens[i]; │ │ │ │ │ + var pat = patterns[token]; │ │ │ │ │ + var matches = tryToken(text, pat); │ │ │ │ │ + if (matches) { │ │ │ │ │ + var match = matches[0]; │ │ │ │ │ + var remainder = text.substr(match.length).replace(/^\s*/, ""); │ │ │ │ │ + return { │ │ │ │ │ + type: token, │ │ │ │ │ + text: match, │ │ │ │ │ + remainder: remainder │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return response │ │ │ │ │ - }, │ │ │ │ │ - read_FeatureInfoResponse: function(data) { │ │ │ │ │ - var response = []; │ │ │ │ │ - var featureNodes = this.getElementsByTagNameNS(data, "*", "FIELDS"); │ │ │ │ │ - for (var i = 0, len = featureNodes.length; i < len; i++) { │ │ │ │ │ - var featureNode = featureNodes[i]; │ │ │ │ │ - var geom = null; │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - var j; │ │ │ │ │ - var jlen = featureNode.attributes.length; │ │ │ │ │ - if (jlen > 0) { │ │ │ │ │ - for (j = 0; j < jlen; j++) { │ │ │ │ │ - var attribute = featureNode.attributes[j]; │ │ │ │ │ - attributes[attribute.nodeName] = attribute.nodeValue │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - var nodes = featureNode.childNodes; │ │ │ │ │ - for (j = 0, jlen = nodes.length; j < jlen; ++j) { │ │ │ │ │ - var node = nodes[j]; │ │ │ │ │ - if (node.nodeType != 3) { │ │ │ │ │ - attributes[node.getAttribute("name")] = node.getAttribute("value") │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - response.push(new OpenLayers.Feature.Vector(geom, attributes, null)) │ │ │ │ │ + var msg = "ERROR: In parsing: [" + text + "], expected one of: "; │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + token = tokens[i]; │ │ │ │ │ + msg += "\n " + token + ": " + patterns[token] │ │ │ │ │ } │ │ │ │ │ - return response │ │ │ │ │ - }, │ │ │ │ │ - getSiblingNodesByTagCriteria: function(node, criteria) { │ │ │ │ │ - var nodes = []; │ │ │ │ │ - var children, tagName, n, matchNodes, child; │ │ │ │ │ - if (node && node.hasChildNodes()) { │ │ │ │ │ - children = node.childNodes; │ │ │ │ │ - n = children.length; │ │ │ │ │ - for (var k = 0; k < n; k++) { │ │ │ │ │ - child = children[k]; │ │ │ │ │ - while (child && child.nodeType != 1) { │ │ │ │ │ - child = child.nextSibling; │ │ │ │ │ - k++ │ │ │ │ │ - } │ │ │ │ │ - tagName = child ? child.nodeName : ""; │ │ │ │ │ - if (tagName.length > 0 && tagName.indexOf(criteria) > -1) { │ │ │ │ │ - nodes.push(child) │ │ │ │ │ - } else { │ │ │ │ │ - matchNodes = this.getSiblingNodesByTagCriteria(child, criteria); │ │ │ │ │ - if (matchNodes.length > 0) { │ │ │ │ │ - nodes.length == 0 ? nodes = matchNodes : nodes.push(matchNodes) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + throw new Error(msg) │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function tokenize(text) { │ │ │ │ │ + var results = []; │ │ │ │ │ + var token, expect = ["NOT", "GEOMETRY", "SPATIAL", "PROPERTY", "LPAREN"]; │ │ │ │ │ + do { │ │ │ │ │ + token = nextToken(text, expect); │ │ │ │ │ + text = token.remainder; │ │ │ │ │ + expect = follows[token.type]; │ │ │ │ │ + if (token.type != "END" && !expect) { │ │ │ │ │ + throw new Error("No follows list for " + token.type) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return nodes │ │ │ │ │ - }, │ │ │ │ │ - parseAttributes: function(node) { │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - if (node.nodeType == 1) { │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - var n = children.length; │ │ │ │ │ - for (var i = 0; i < n; ++i) { │ │ │ │ │ - var child = children[i]; │ │ │ │ │ - if (child.nodeType == 1) { │ │ │ │ │ - var grandchildren = child.childNodes; │ │ │ │ │ - var name = child.prefix ? child.nodeName.split(":")[1] : child.nodeName; │ │ │ │ │ - if (grandchildren.length == 0) { │ │ │ │ │ - attributes[name] = null │ │ │ │ │ - } else if (grandchildren.length == 1) { │ │ │ │ │ - var grandchild = grandchildren[0]; │ │ │ │ │ - if (grandchild.nodeType == 3 || grandchild.nodeType == 4) { │ │ │ │ │ - var value = grandchild.nodeValue.replace(this.regExes.trimSpace, ""); │ │ │ │ │ - attributes[name] = value │ │ │ │ │ - } │ │ │ │ │ + results.push(token) │ │ │ │ │ + } while (token.type != "END"); │ │ │ │ │ + return results │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function buildAst(tokens) { │ │ │ │ │ + var operatorStack = [], │ │ │ │ │ + postfix = []; │ │ │ │ │ + while (tokens.length) { │ │ │ │ │ + var tok = tokens.shift(); │ │ │ │ │ + switch (tok.type) { │ │ │ │ │ + case "PROPERTY": │ │ │ │ │ + case "GEOMETRY": │ │ │ │ │ + case "VALUE": │ │ │ │ │ + postfix.push(tok); │ │ │ │ │ + break; │ │ │ │ │ + case "COMPARISON": │ │ │ │ │ + case "BETWEEN": │ │ │ │ │ + case "IS_NULL": │ │ │ │ │ + case "LOGICAL": │ │ │ │ │ + var p = precedence[tok.type]; │ │ │ │ │ + while (operatorStack.length > 0 && precedence[operatorStack[operatorStack.length - 1].type] <= p) { │ │ │ │ │ + postfix.push(operatorStack.pop()) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ + operatorStack.push(tok); │ │ │ │ │ + break; │ │ │ │ │ + case "SPATIAL": │ │ │ │ │ + case "NOT": │ │ │ │ │ + case "LPAREN": │ │ │ │ │ + operatorStack.push(tok); │ │ │ │ │ + break; │ │ │ │ │ + case "RPAREN": │ │ │ │ │ + while (operatorStack.length > 0 && operatorStack[operatorStack.length - 1].type != "LPAREN") { │ │ │ │ │ + postfix.push(operatorStack.pop()) │ │ │ │ │ + } │ │ │ │ │ + operatorStack.pop(); │ │ │ │ │ + if (operatorStack.length > 0 && operatorStack[operatorStack.length - 1].type == "SPATIAL") { │ │ │ │ │ + postfix.push(operatorStack.pop()) │ │ │ │ │ + } │ │ │ │ │ + case "COMMA": │ │ │ │ │ + case "END": │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + throw new Error("Unknown token type " + tok.type) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return attributes │ │ │ │ │ - }, │ │ │ │ │ - parseGeometry: function(node) { │ │ │ │ │ - if (!this.gmlFormat) { │ │ │ │ │ - this.gmlFormat = new OpenLayers.Format.GML │ │ │ │ │ + while (operatorStack.length > 0) { │ │ │ │ │ + postfix.push(operatorStack.pop()) │ │ │ │ │ } │ │ │ │ │ - var feature = this.gmlFormat.parseFeature(node); │ │ │ │ │ - var geometry, bounds = null; │ │ │ │ │ - if (feature) { │ │ │ │ │ - geometry = feature.geometry && feature.geometry.clone(); │ │ │ │ │ - bounds = feature.bounds && feature.bounds.clone(); │ │ │ │ │ - feature.destroy() │ │ │ │ │ + │ │ │ │ │ + function buildTree() { │ │ │ │ │ + var tok = postfix.pop(); │ │ │ │ │ + switch (tok.type) { │ │ │ │ │ + case "LOGICAL": │ │ │ │ │ + var rhs = buildTree(), │ │ │ │ │ + lhs = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Logical({ │ │ │ │ │ + filters: [lhs, rhs], │ │ │ │ │ + type: logicals[tok.text.toUpperCase()] │ │ │ │ │ + }); │ │ │ │ │ + case "NOT": │ │ │ │ │ + var operand = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Logical({ │ │ │ │ │ + filters: [operand], │ │ │ │ │ + type: OpenLayers.Filter.Logical.NOT │ │ │ │ │ + }); │ │ │ │ │ + case "BETWEEN": │ │ │ │ │ + var min, max, property; │ │ │ │ │ + postfix.pop(); │ │ │ │ │ + max = buildTree(); │ │ │ │ │ + min = buildTree(); │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Comparison({ │ │ │ │ │ + property: property, │ │ │ │ │ + lowerBoundary: min, │ │ │ │ │ + upperBoundary: max, │ │ │ │ │ + type: OpenLayers.Filter.Comparison.BETWEEN │ │ │ │ │ + }); │ │ │ │ │ + case "COMPARISON": │ │ │ │ │ + var value = buildTree(), │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Comparison({ │ │ │ │ │ + property: property, │ │ │ │ │ + value: value, │ │ │ │ │ + type: operators[tok.text.toUpperCase()] │ │ │ │ │ + }); │ │ │ │ │ + case "IS_NULL": │ │ │ │ │ + var property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Comparison({ │ │ │ │ │ + property: property, │ │ │ │ │ + type: operators[tok.text.toUpperCase()] │ │ │ │ │ + }); │ │ │ │ │ + case "VALUE": │ │ │ │ │ + var match = tok.text.match(/^'(.*)'$/); │ │ │ │ │ + if (match) { │ │ │ │ │ + return match[1].replace(/''/g, "'") │ │ │ │ │ + } else { │ │ │ │ │ + return Number(tok.text) │ │ │ │ │ + } │ │ │ │ │ + case "SPATIAL": │ │ │ │ │ + switch (tok.text.toUpperCase()) { │ │ │ │ │ + case "BBOX": │ │ │ │ │ + var maxy = buildTree(), │ │ │ │ │ + maxx = buildTree(), │ │ │ │ │ + miny = buildTree(), │ │ │ │ │ + minx = buildTree(), │ │ │ │ │ + prop = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ + property: prop, │ │ │ │ │ + value: OpenLayers.Bounds.fromArray([minx, miny, maxx, maxy]) │ │ │ │ │ + }); │ │ │ │ │ + case "INTERSECTS": │ │ │ │ │ + var value = buildTree(), │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: property, │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + case "WITHIN": │ │ │ │ │ + var value = buildTree(), │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.WITHIN, │ │ │ │ │ + property: property, │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + case "CONTAINS": │ │ │ │ │ + var value = buildTree(), │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.CONTAINS, │ │ │ │ │ + property: property, │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + case "DWITHIN": │ │ │ │ │ + var distance = buildTree(), │ │ │ │ │ + value = buildTree(), │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ + value: value, │ │ │ │ │ + property: property, │ │ │ │ │ + distance: Number(distance) │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + case "GEOMETRY": │ │ │ │ │ + return OpenLayers.Geometry.fromWKT(tok.text); │ │ │ │ │ + default: │ │ │ │ │ + return tok.text │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return { │ │ │ │ │ - geometry: geometry, │ │ │ │ │ - bounds: bounds │ │ │ │ │ + var result = buildTree(); │ │ │ │ │ + if (postfix.length > 0) { │ │ │ │ │ + var msg = "Remaining tokens after building AST: \n"; │ │ │ │ │ + for (var i = postfix.length - 1; i >= 0; i--) { │ │ │ │ │ + msg += postfix[i].type + ": " + postfix[i].text + "\n" │ │ │ │ │ + } │ │ │ │ │ + throw new Error(msg) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSGetFeatureInfo" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.CSWGetDomain = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, OpenLayers.Format.CSWGetDomain.DEFAULTS); │ │ │ │ │ - var cls = OpenLayers.Format.CSWGetDomain["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported CSWGetDomain version: " + options.version │ │ │ │ │ - } │ │ │ │ │ - return new cls(options) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Format.CSWGetDomain.DEFAULTS = { │ │ │ │ │ - version: "2.0.2" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Format.QueryStringFilter = function() { │ │ │ │ │ - var cmpToStr = {}; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.EQUAL_TO] = "eq"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.NOT_EQUAL_TO] = "ne"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.LESS_THAN] = "lt"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO] = "lte"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.GREATER_THAN] = "gt"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO] = "gte"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.LIKE] = "ilike"; │ │ │ │ │ - │ │ │ │ │ - function regex2value(value) { │ │ │ │ │ - value = value.replace(/%/g, "\\%"); │ │ │ │ │ - value = value.replace(/\\\\\.(\*)?/g, function($0, $1) { │ │ │ │ │ - return $1 ? $0 : "\\\\_" │ │ │ │ │ - }); │ │ │ │ │ - value = value.replace(/\\\\\.\*/g, "\\\\%"); │ │ │ │ │ - value = value.replace(/(\\)?\.(\*)?/g, function($0, $1, $2) { │ │ │ │ │ - return $1 || $2 ? $0 : "_" │ │ │ │ │ - }); │ │ │ │ │ - value = value.replace(/(\\)?\.\*/g, function($0, $1) { │ │ │ │ │ - return $1 ? $0 : "%" │ │ │ │ │ - }); │ │ │ │ │ - value = value.replace(/\\\./g, "."); │ │ │ │ │ - value = value.replace(/(\\)?\\\*/g, function($0, $1) { │ │ │ │ │ - return $1 ? $0 : "*" │ │ │ │ │ - }); │ │ │ │ │ - return value │ │ │ │ │ + return result │ │ │ │ │ } │ │ │ │ │ return OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ - wildcarded: false, │ │ │ │ │ - srsInBBOX: false, │ │ │ │ │ - write: function(filter, params) { │ │ │ │ │ - params = params || {}; │ │ │ │ │ - var className = filter.CLASS_NAME; │ │ │ │ │ - var filterType = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ - switch (filterType) { │ │ │ │ │ - case "Spatial": │ │ │ │ │ + read: function(text) { │ │ │ │ │ + var result = buildAst(tokenize(text)); │ │ │ │ │ + if (this.keepData) { │ │ │ │ │ + this.data = result │ │ │ │ │ + } │ │ │ │ │ + return result │ │ │ │ │ + }, │ │ │ │ │ + write: function(filter) { │ │ │ │ │ + if (filter instanceof OpenLayers.Geometry) { │ │ │ │ │ + return filter.toString() │ │ │ │ │ + } │ │ │ │ │ + switch (filter.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Filter.Spatial": │ │ │ │ │ switch (filter.type) { │ │ │ │ │ case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ - params.bbox = filter.value.toArray(); │ │ │ │ │ - if (this.srsInBBOX && filter.projection) { │ │ │ │ │ - params.bbox.push(filter.projection.getCode()) │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ + return "BBOX(" + filter.property + "," + filter.value.toBBOX() + ")"; │ │ │ │ │ case OpenLayers.Filter.Spatial.DWITHIN: │ │ │ │ │ - params.tolerance = filter.distance; │ │ │ │ │ + return "DWITHIN(" + filter.property + ", " + this.write(filter.value) + ", " + filter.distance + ")"; │ │ │ │ │ case OpenLayers.Filter.Spatial.WITHIN: │ │ │ │ │ - params.lon = filter.value.x; │ │ │ │ │ - params.lat = filter.value.y; │ │ │ │ │ - break; │ │ │ │ │ + return "WITHIN(" + filter.property + ", " + this.write(filter.value) + ")"; │ │ │ │ │ + case OpenLayers.Filter.Spatial.INTERSECTS: │ │ │ │ │ + return "INTERSECTS(" + filter.property + ", " + this.write(filter.value) + ")"; │ │ │ │ │ + case OpenLayers.Filter.Spatial.CONTAINS: │ │ │ │ │ + return "CONTAINS(" + filter.property + ", " + this.write(filter.value) + ")"; │ │ │ │ │ default: │ │ │ │ │ - OpenLayers.Console.warn("Unknown spatial filter type " + filter.type) │ │ │ │ │ + throw new Error("Unknown spatial filter type: " + filter.type) │ │ │ │ │ } │ │ │ │ │ - break; │ │ │ │ │ - case "Comparison": │ │ │ │ │ - var op = cmpToStr[filter.type]; │ │ │ │ │ - if (op !== undefined) { │ │ │ │ │ - var value = filter.value; │ │ │ │ │ - if (filter.type == OpenLayers.Filter.Comparison.LIKE) { │ │ │ │ │ - value = regex2value(value); │ │ │ │ │ - if (this.wildcarded) { │ │ │ │ │ - value = "%" + value + "%" │ │ │ │ │ + case "OpenLayers.Filter.Logical": │ │ │ │ │ + if (filter.type == OpenLayers.Filter.Logical.NOT) { │ │ │ │ │ + return "NOT (" + this.write(filter.filters[0]) + ")" │ │ │ │ │ + } else { │ │ │ │ │ + var res = "("; │ │ │ │ │ + var first = true; │ │ │ │ │ + for (var i = 0; i < filter.filters.length; i++) { │ │ │ │ │ + if (first) { │ │ │ │ │ + first = false │ │ │ │ │ + } else { │ │ │ │ │ + res += ") " + logicalReverse[filter.type] + " (" │ │ │ │ │ } │ │ │ │ │ + res += this.write(filter.filters[i]) │ │ │ │ │ } │ │ │ │ │ - params[filter.property + "__" + op] = value; │ │ │ │ │ - params.queryable = params.queryable || []; │ │ │ │ │ - params.queryable.push(filter.property) │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Console.warn("Unknown comparison filter type " + filter.type) │ │ │ │ │ + return res + ")" │ │ │ │ │ } │ │ │ │ │ - break; │ │ │ │ │ - case "Logical": │ │ │ │ │ - if (filter.type === OpenLayers.Filter.Logical.AND) { │ │ │ │ │ - for (var i = 0, len = filter.filters.length; i < len; i++) { │ │ │ │ │ - params = this.write(filter.filters[i], params) │ │ │ │ │ - } │ │ │ │ │ + case "OpenLayers.Filter.Comparison": │ │ │ │ │ + if (filter.type == OpenLayers.Filter.Comparison.BETWEEN) { │ │ │ │ │ + return filter.property + " BETWEEN " + this.write(filter.lowerBoundary) + " AND " + this.write(filter.upperBoundary) │ │ │ │ │ } else { │ │ │ │ │ - OpenLayers.Console.warn("Unsupported logical filter type " + filter.type) │ │ │ │ │ + return filter.value !== null ? filter.property + " " + operatorReverse[filter.type] + " " + this.write(filter.value) : filter.property + " " + operatorReverse[filter.type] │ │ │ │ │ + } │ │ │ │ │ + case undefined: │ │ │ │ │ + if (typeof filter === "string") { │ │ │ │ │ + return "'" + filter.replace(/'/g, "''") + "'" │ │ │ │ │ + } else if (typeof filter === "number") { │ │ │ │ │ + return String(filter) │ │ │ │ │ } │ │ │ │ │ - break; │ │ │ │ │ default: │ │ │ │ │ - OpenLayers.Console.warn("Unknown filter type " + filterType) │ │ │ │ │ + throw new Error("Can't encode: " + filter.CLASS_NAME + " " + filter) │ │ │ │ │ } │ │ │ │ │ - return params │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.QueryStringFilter" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.CQL" │ │ │ │ │ }) │ │ │ │ │ }(); │ │ │ │ │ +OpenLayers.Format.WMSDescribeLayer = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + defaultVersion: "1.1.1", │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSDescribeLayer" │ │ │ │ │ +}); │ │ │ │ │ OpenLayers.Format.KML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ namespaces: { │ │ │ │ │ kml: "http://www.opengis.net/kml/2.2", │ │ │ │ │ gx: "http://www.google.com/kml/ext/2.2" │ │ │ │ │ }, │ │ │ │ │ kmlns: "http://earth.google.com/kml/2.0", │ │ │ │ │ placemarksDesc: "No description available", │ │ │ │ │ @@ -20800,197 +22303,109 @@ │ │ │ │ │ return null │ │ │ │ │ } else { │ │ │ │ │ return extendedData │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.KML" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.WCSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - defaultVersion: "1.1.0", │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WCSCapabilities" │ │ │ │ │ +OpenLayers.Format.SLD = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + profile: null, │ │ │ │ │ + defaultVersion: "1.0.0", │ │ │ │ │ + stringifyOutput: true, │ │ │ │ │ + namedLayersAsArray: false, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SLD" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.GPX = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - defaultDesc: "No description available", │ │ │ │ │ - extractWaypoints: true, │ │ │ │ │ - extractTracks: true, │ │ │ │ │ - extractRoutes: true, │ │ │ │ │ - extractAttributes: true, │ │ │ │ │ - namespaces: { │ │ │ │ │ - gpx: "http://www.topografix.com/GPX/1/1", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ - }, │ │ │ │ │ - schemaLocation: "http://www.topografix.com/GPX/1/1 http://www.topografix.com/GPX/1/1/gpx.xsd", │ │ │ │ │ - creator: "OpenLayers", │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ - read: function(doc) { │ │ │ │ │ - if (typeof doc == "string") { │ │ │ │ │ - doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]) │ │ │ │ │ - } │ │ │ │ │ - var features = []; │ │ │ │ │ - if (this.extractTracks) { │ │ │ │ │ - var tracks = doc.getElementsByTagName("trk"); │ │ │ │ │ - for (var i = 0, len = tracks.length; i < len; i++) { │ │ │ │ │ - var attrs = {}; │ │ │ │ │ - if (this.extractAttributes) { │ │ │ │ │ - attrs = this.parseAttributes(tracks[i]) │ │ │ │ │ - } │ │ │ │ │ - var segs = this.getElementsByTagNameNS(tracks[i], tracks[i].namespaceURI, "trkseg"); │ │ │ │ │ - for (var j = 0, seglen = segs.length; j < seglen; j++) { │ │ │ │ │ - var track = this.extractSegment(segs[j], "trkpt"); │ │ │ │ │ - features.push(new OpenLayers.Feature.Vector(track, attrs)) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.extractRoutes) { │ │ │ │ │ - var routes = doc.getElementsByTagName("rte"); │ │ │ │ │ - for (var k = 0, klen = routes.length; k < klen; k++) { │ │ │ │ │ - var attrs = {}; │ │ │ │ │ - if (this.extractAttributes) { │ │ │ │ │ - attrs = this.parseAttributes(routes[k]) │ │ │ │ │ - } │ │ │ │ │ - var route = this.extractSegment(routes[k], "rtept"); │ │ │ │ │ - features.push(new OpenLayers.Feature.Vector(route, attrs)) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.extractWaypoints) { │ │ │ │ │ - var waypoints = doc.getElementsByTagName("wpt"); │ │ │ │ │ - for (var l = 0, len = waypoints.length; l < len; l++) { │ │ │ │ │ - var attrs = {}; │ │ │ │ │ - if (this.extractAttributes) { │ │ │ │ │ - attrs = this.parseAttributes(waypoints[l]) │ │ │ │ │ - } │ │ │ │ │ - var wpt = new OpenLayers.Geometry.Point(waypoints[l].getAttribute("lon"), waypoints[l].getAttribute("lat")); │ │ │ │ │ - features.push(new OpenLayers.Feature.Vector(wpt, attrs)) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - for (var g = 0, featLength = features.length; g < featLength; g++) { │ │ │ │ │ - features[g].geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return features │ │ │ │ │ - }, │ │ │ │ │ - extractSegment: function(segment, segmentType) { │ │ │ │ │ - var points = this.getElementsByTagNameNS(segment, segment.namespaceURI, segmentType); │ │ │ │ │ - var point_features = []; │ │ │ │ │ - for (var i = 0, len = points.length; i < len; i++) { │ │ │ │ │ - point_features.push(new OpenLayers.Geometry.Point(points[i].getAttribute("lon"), points[i].getAttribute("lat"))) │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Geometry.LineString(point_features) │ │ │ │ │ - }, │ │ │ │ │ - parseAttributes: function(node) { │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - var attrNode = node.firstChild, │ │ │ │ │ - value, name; │ │ │ │ │ - while (attrNode) { │ │ │ │ │ - if (attrNode.nodeType == 1 && attrNode.firstChild) { │ │ │ │ │ - value = attrNode.firstChild; │ │ │ │ │ - if (value.nodeType == 3 || value.nodeType == 4) { │ │ │ │ │ - name = attrNode.prefix ? attrNode.nodeName.split(":")[1] : attrNode.nodeName; │ │ │ │ │ - if (name != "trkseg" && name != "rtept") { │ │ │ │ │ - attributes[name] = value.nodeValue │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - attrNode = attrNode.nextSibling │ │ │ │ │ - } │ │ │ │ │ - return attributes │ │ │ │ │ - }, │ │ │ │ │ - write: function(features, metadata) { │ │ │ │ │ - features = OpenLayers.Util.isArray(features) ? features : [features]; │ │ │ │ │ - var gpx = this.createElementNS(this.namespaces.gpx, "gpx"); │ │ │ │ │ - gpx.setAttribute("version", "1.1"); │ │ │ │ │ - gpx.setAttribute("creator", this.creator); │ │ │ │ │ - this.setAttributes(gpx, { │ │ │ │ │ - "xsi:schemaLocation": this.schemaLocation │ │ │ │ │ +OpenLayers.Format.QueryStringFilter = function() { │ │ │ │ │ + var cmpToStr = {}; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.EQUAL_TO] = "eq"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.NOT_EQUAL_TO] = "ne"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.LESS_THAN] = "lt"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO] = "lte"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.GREATER_THAN] = "gt"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO] = "gte"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.LIKE] = "ilike"; │ │ │ │ │ + │ │ │ │ │ + function regex2value(value) { │ │ │ │ │ + value = value.replace(/%/g, "\\%"); │ │ │ │ │ + value = value.replace(/\\\\\.(\*)?/g, function($0, $1) { │ │ │ │ │ + return $1 ? $0 : "\\\\_" │ │ │ │ │ }); │ │ │ │ │ - if (metadata && typeof metadata == "object") { │ │ │ │ │ - gpx.appendChild(this.buildMetadataNode(metadata)) │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - gpx.appendChild(this.buildFeatureNode(features[i])) │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [gpx]) │ │ │ │ │ - }, │ │ │ │ │ - buildMetadataNode: function(metadata) { │ │ │ │ │ - var types = ["name", "desc", "author"], │ │ │ │ │ - node = this.createElementNS(this.namespaces.gpx, "metadata"); │ │ │ │ │ - for (var i = 0; i < types.length; i++) { │ │ │ │ │ - var type = types[i]; │ │ │ │ │ - if (metadata[type]) { │ │ │ │ │ - var n = this.createElementNS(this.namespaces.gpx, type); │ │ │ │ │ - n.appendChild(this.createTextNode(metadata[type])); │ │ │ │ │ - node.appendChild(n) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - buildFeatureNode: function(feature) { │ │ │ │ │ - var geometry = feature.geometry; │ │ │ │ │ - geometry = geometry.clone(); │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry.transform(this.internalProjection, this.externalProjection) │ │ │ │ │ - } │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - var wpt = this.buildWptNode(geometry); │ │ │ │ │ - this.appendAttributesNode(wpt, feature); │ │ │ │ │ - return wpt │ │ │ │ │ - } else { │ │ │ │ │ - var trkNode = this.createElementNS(this.namespaces.gpx, "trk"); │ │ │ │ │ - this.appendAttributesNode(trkNode, feature); │ │ │ │ │ - var trkSegNodes = this.buildTrkSegNode(geometry); │ │ │ │ │ - trkSegNodes = OpenLayers.Util.isArray(trkSegNodes) ? trkSegNodes : [trkSegNodes]; │ │ │ │ │ - for (var i = 0, len = trkSegNodes.length; i < len; i++) { │ │ │ │ │ - trkNode.appendChild(trkSegNodes[i]) │ │ │ │ │ - } │ │ │ │ │ - return trkNode │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - buildTrkSegNode: function(geometry) { │ │ │ │ │ - var node, i, len, point, nodes; │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.LineString" || geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ - node = this.createElementNS(this.namespaces.gpx, "trkseg"); │ │ │ │ │ - for (i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - point = geometry.components[i]; │ │ │ │ │ - node.appendChild(this.buildTrkPtNode(point)) │ │ │ │ │ - } │ │ │ │ │ - return node │ │ │ │ │ - } else { │ │ │ │ │ - nodes = []; │ │ │ │ │ - for (i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - nodes.push(this.buildTrkSegNode(geometry.components[i])) │ │ │ │ │ + value = value.replace(/\\\\\.\*/g, "\\\\%"); │ │ │ │ │ + value = value.replace(/(\\)?\.(\*)?/g, function($0, $1, $2) { │ │ │ │ │ + return $1 || $2 ? $0 : "_" │ │ │ │ │ + }); │ │ │ │ │ + value = value.replace(/(\\)?\.\*/g, function($0, $1) { │ │ │ │ │ + return $1 ? $0 : "%" │ │ │ │ │ + }); │ │ │ │ │ + value = value.replace(/\\\./g, "."); │ │ │ │ │ + value = value.replace(/(\\)?\\\*/g, function($0, $1) { │ │ │ │ │ + return $1 ? $0 : "*" │ │ │ │ │ + }); │ │ │ │ │ + return value │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ + wildcarded: false, │ │ │ │ │ + srsInBBOX: false, │ │ │ │ │ + write: function(filter, params) { │ │ │ │ │ + params = params || {}; │ │ │ │ │ + var className = filter.CLASS_NAME; │ │ │ │ │ + var filterType = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ + switch (filterType) { │ │ │ │ │ + case "Spatial": │ │ │ │ │ + switch (filter.type) { │ │ │ │ │ + case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ + params.bbox = filter.value.toArray(); │ │ │ │ │ + if (this.srsInBBOX && filter.projection) { │ │ │ │ │ + params.bbox.push(filter.projection.getCode()) │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Spatial.DWITHIN: │ │ │ │ │ + params.tolerance = filter.distance; │ │ │ │ │ + case OpenLayers.Filter.Spatial.WITHIN: │ │ │ │ │ + params.lon = filter.value.x; │ │ │ │ │ + params.lat = filter.value.y; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + OpenLayers.Console.warn("Unknown spatial filter type " + filter.type) │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "Comparison": │ │ │ │ │ + var op = cmpToStr[filter.type]; │ │ │ │ │ + if (op !== undefined) { │ │ │ │ │ + var value = filter.value; │ │ │ │ │ + if (filter.type == OpenLayers.Filter.Comparison.LIKE) { │ │ │ │ │ + value = regex2value(value); │ │ │ │ │ + if (this.wildcarded) { │ │ │ │ │ + value = "%" + value + "%" │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + params[filter.property + "__" + op] = value; │ │ │ │ │ + params.queryable = params.queryable || []; │ │ │ │ │ + params.queryable.push(filter.property) │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Console.warn("Unknown comparison filter type " + filter.type) │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "Logical": │ │ │ │ │ + if (filter.type === OpenLayers.Filter.Logical.AND) { │ │ │ │ │ + for (var i = 0, len = filter.filters.length; i < len; i++) { │ │ │ │ │ + params = this.write(filter.filters[i], params) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Console.warn("Unsupported logical filter type " + filter.type) │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + OpenLayers.Console.warn("Unknown filter type " + filterType) │ │ │ │ │ } │ │ │ │ │ - return nodes │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - buildTrkPtNode: function(point) { │ │ │ │ │ - var node = this.createElementNS(this.namespaces.gpx, "trkpt"); │ │ │ │ │ - node.setAttribute("lon", point.x); │ │ │ │ │ - node.setAttribute("lat", point.y); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - buildWptNode: function(geometry) { │ │ │ │ │ - var node = this.createElementNS(this.namespaces.gpx, "wpt"); │ │ │ │ │ - node.setAttribute("lon", geometry.x); │ │ │ │ │ - node.setAttribute("lat", geometry.y); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - appendAttributesNode: function(node, feature) { │ │ │ │ │ - var name = this.createElementNS(this.namespaces.gpx, "name"); │ │ │ │ │ - name.appendChild(this.createTextNode(feature.attributes.name || feature.id)); │ │ │ │ │ - node.appendChild(name); │ │ │ │ │ - var desc = this.createElementNS(this.namespaces.gpx, "desc"); │ │ │ │ │ - desc.appendChild(this.createTextNode(feature.attributes.description || this.defaultDesc)); │ │ │ │ │ - node.appendChild(desc) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.GPX" │ │ │ │ │ -}); │ │ │ │ │ + return params │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.QueryStringFilter" │ │ │ │ │ + }) │ │ │ │ │ +}(); │ │ │ │ │ OpenLayers.Format.GeoRSS = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ rssns: "http://backend.userland.com/rss2", │ │ │ │ │ featureNS: "http://mapserver.gis.umn.edu/mapserver", │ │ │ │ │ georssns: "http://www.georss.org/georss", │ │ │ │ │ geons: "http://www.w3.org/2003/01/geo/wgs84_pos#", │ │ │ │ │ featureTitle: "Untitled", │ │ │ │ │ featureDescription: "No Description", │ │ │ │ │ @@ -21190,1274 +22605,747 @@ │ │ │ │ │ } else { │ │ │ │ │ path = geometry.y + " " + geometry.x │ │ │ │ │ } │ │ │ │ │ return this.createTextNode(path) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.GeoRSS" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.WMSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - defaultVersion: "1.1.1", │ │ │ │ │ - profile: null, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities" │ │ │ │ │ +OpenLayers.Format.WFSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + defaultVersion: "1.1.0", │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSCapabilities" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.WMTSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - defaultVersion: "1.0.0", │ │ │ │ │ - yx: { │ │ │ │ │ - "urn:ogc:def:crs:EPSG::4326": true │ │ │ │ │ +OpenLayers.Format.Atom = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + namespaces: { │ │ │ │ │ + atom: "http://www.w3.org/2005/Atom", │ │ │ │ │ + georss: "http://www.georss.org/georss" │ │ │ │ │ }, │ │ │ │ │ - createLayer: function(capabilities, config) { │ │ │ │ │ - var layer; │ │ │ │ │ - if (!("layer" in config)) { │ │ │ │ │ - throw new Error("Missing property 'layer' in configuration.") │ │ │ │ │ + feedTitle: "untitled", │ │ │ │ │ + defaultEntryTitle: "untitled", │ │ │ │ │ + gmlParser: null, │ │ │ │ │ + xy: false, │ │ │ │ │ + read: function(doc) { │ │ │ │ │ + if (typeof doc == "string") { │ │ │ │ │ + doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]) │ │ │ │ │ } │ │ │ │ │ - var contents = capabilities.contents; │ │ │ │ │ - var layers = contents.layers; │ │ │ │ │ - var layerDef; │ │ │ │ │ - for (var i = 0, ii = contents.layers.length; i < ii; ++i) { │ │ │ │ │ - if (contents.layers[i].identifier === config.layer) { │ │ │ │ │ - layerDef = contents.layers[i]; │ │ │ │ │ - break │ │ │ │ │ + return this.parseFeatures(doc) │ │ │ │ │ + }, │ │ │ │ │ + write: function(features) { │ │ │ │ │ + var doc; │ │ │ │ │ + if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ + doc = this.createElementNSPlus("atom:feed"); │ │ │ │ │ + doc.appendChild(this.createElementNSPlus("atom:title", { │ │ │ │ │ + value: this.feedTitle │ │ │ │ │ + })); │ │ │ │ │ + for (var i = 0, ii = features.length; i < ii; i++) { │ │ │ │ │ + doc.appendChild(this.buildEntryNode(features[i])) │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + doc = this.buildEntryNode(features) │ │ │ │ │ } │ │ │ │ │ - if (!layerDef) { │ │ │ │ │ - throw new Error("Layer not found") │ │ │ │ │ - } │ │ │ │ │ - var format = config.format; │ │ │ │ │ - if (!format && layerDef.formats && layerDef.formats.length) { │ │ │ │ │ - format = layerDef.formats[0] │ │ │ │ │ - } │ │ │ │ │ - var matrixSet; │ │ │ │ │ - if (config.matrixSet) { │ │ │ │ │ - matrixSet = contents.tileMatrixSets[config.matrixSet] │ │ │ │ │ - } else if (layerDef.tileMatrixSetLinks.length >= 1) { │ │ │ │ │ - matrixSet = contents.tileMatrixSets[layerDef.tileMatrixSetLinks[0].tileMatrixSet] │ │ │ │ │ - } │ │ │ │ │ - if (!matrixSet) { │ │ │ │ │ - throw new Error("matrixSet not found") │ │ │ │ │ - } │ │ │ │ │ - var style; │ │ │ │ │ - for (var i = 0, ii = layerDef.styles.length; i < ii; ++i) { │ │ │ │ │ - style = layerDef.styles[i]; │ │ │ │ │ - if (style.isDefault) { │ │ │ │ │ - break │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [doc]) │ │ │ │ │ + }, │ │ │ │ │ + buildContentNode: function(content) { │ │ │ │ │ + var node = this.createElementNSPlus("atom:content", { │ │ │ │ │ + attributes: { │ │ │ │ │ + type: content.type || null │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - var requestEncoding = config.requestEncoding; │ │ │ │ │ - if (!requestEncoding) { │ │ │ │ │ - requestEncoding = "KVP"; │ │ │ │ │ - if (capabilities.operationsMetadata.GetTile.dcp.http) { │ │ │ │ │ - var http = capabilities.operationsMetadata.GetTile.dcp.http; │ │ │ │ │ - if (http.get[0].constraints) { │ │ │ │ │ - var constraints = http.get[0].constraints; │ │ │ │ │ - var allowedValues = constraints.GetEncoding.allowedValues; │ │ │ │ │ - if (!allowedValues.KVP && (allowedValues.REST || allowedValues.RESTful)) { │ │ │ │ │ - requestEncoding = "REST" │ │ │ │ │ - } │ │ │ │ │ + }); │ │ │ │ │ + if (content.src) { │ │ │ │ │ + node.setAttribute("src", content.src) │ │ │ │ │ + } else { │ │ │ │ │ + if (content.type == "text" || content.type == null) { │ │ │ │ │ + node.appendChild(this.createTextNode(content.value)) │ │ │ │ │ + } else if (content.type == "html") { │ │ │ │ │ + if (typeof content.value != "string") { │ │ │ │ │ + throw "HTML content must be in form of an escaped string" │ │ │ │ │ } │ │ │ │ │ + node.appendChild(this.createTextNode(content.value)) │ │ │ │ │ + } else if (content.type == "xhtml") { │ │ │ │ │ + node.appendChild(content.value) │ │ │ │ │ + } else if (content.type == "xhtml" || content.type.match(/(\+|\/)xml$/)) { │ │ │ │ │ + node.appendChild(content.value) │ │ │ │ │ + } else { │ │ │ │ │ + node.appendChild(this.createTextNode(content.value)) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var dimensions = []; │ │ │ │ │ - var params = config.params || {}; │ │ │ │ │ - delete config.params; │ │ │ │ │ - for (var id = 0, ld = layerDef.dimensions.length; id < ld; id++) { │ │ │ │ │ - var dimension = layerDef.dimensions[id]; │ │ │ │ │ - dimensions.push(dimension.identifier); │ │ │ │ │ - if (!params.hasOwnProperty(dimension.identifier)) { │ │ │ │ │ - params[dimension.identifier] = dimension["default"] │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + buildEntryNode: function(feature) { │ │ │ │ │ + var attrib = feature.attributes; │ │ │ │ │ + var atomAttrib = attrib.atom || {}; │ │ │ │ │ + var entryNode = this.createElementNSPlus("atom:entry"); │ │ │ │ │ + if (atomAttrib.authors) { │ │ │ │ │ + var authors = OpenLayers.Util.isArray(atomAttrib.authors) ? atomAttrib.authors : [atomAttrib.authors]; │ │ │ │ │ + for (var i = 0, ii = authors.length; i < ii; i++) { │ │ │ │ │ + entryNode.appendChild(this.buildPersonConstructNode("author", authors[i])) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var projection = config.projection || matrixSet.supportedCRS.replace(/urn:ogc:def:crs:(\w+):(.*:)?(\w+)$/, "$1:$3"); │ │ │ │ │ - var units = config.units || (projection === "EPSG:4326" ? "degrees" : "m"); │ │ │ │ │ - var resolutions = []; │ │ │ │ │ - for (var mid in matrixSet.matrixIds) { │ │ │ │ │ - if (matrixSet.matrixIds.hasOwnProperty(mid)) { │ │ │ │ │ - resolutions.push(matrixSet.matrixIds[mid].scaleDenominator * 28e-5 / OpenLayers.METERS_PER_INCH / OpenLayers.INCHES_PER_UNIT[units]) │ │ │ │ │ + if (atomAttrib.categories) { │ │ │ │ │ + var categories = OpenLayers.Util.isArray(atomAttrib.categories) ? atomAttrib.categories : [atomAttrib.categories]; │ │ │ │ │ + var category; │ │ │ │ │ + for (var i = 0, ii = categories.length; i < ii; i++) { │ │ │ │ │ + category = categories[i]; │ │ │ │ │ + entryNode.appendChild(this.createElementNSPlus("atom:category", { │ │ │ │ │ + attributes: { │ │ │ │ │ + term: category.term, │ │ │ │ │ + scheme: category.scheme || null, │ │ │ │ │ + label: category.label || null │ │ │ │ │ + } │ │ │ │ │ + })) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var url; │ │ │ │ │ - if (requestEncoding === "REST" && layerDef.resourceUrls) { │ │ │ │ │ - url = []; │ │ │ │ │ - var resourceUrls = layerDef.resourceUrls, │ │ │ │ │ - resourceUrl; │ │ │ │ │ - for (var t = 0, tt = layerDef.resourceUrls.length; t < tt; ++t) { │ │ │ │ │ - resourceUrl = layerDef.resourceUrls[t]; │ │ │ │ │ - if (resourceUrl.format === format && resourceUrl.resourceType === "tile") { │ │ │ │ │ - url.push(resourceUrl.template) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - var httpGet = capabilities.operationsMetadata.GetTile.dcp.http.get; │ │ │ │ │ - url = []; │ │ │ │ │ - var constraint; │ │ │ │ │ - for (var i = 0, ii = httpGet.length; i < ii; i++) { │ │ │ │ │ - constraint = httpGet[i].constraints; │ │ │ │ │ - if (!constraint || constraint && constraint.GetEncoding.allowedValues[requestEncoding]) { │ │ │ │ │ - url.push(httpGet[i].url) │ │ │ │ │ - } │ │ │ │ │ + if (atomAttrib.content) { │ │ │ │ │ + entryNode.appendChild(this.buildContentNode(atomAttrib.content)) │ │ │ │ │ + } │ │ │ │ │ + if (atomAttrib.contributors) { │ │ │ │ │ + var contributors = OpenLayers.Util.isArray(atomAttrib.contributors) ? atomAttrib.contributors : [atomAttrib.contributors]; │ │ │ │ │ + for (var i = 0, ii = contributors.length; i < ii; i++) { │ │ │ │ │ + entryNode.appendChild(this.buildPersonConstructNode("contributor", contributors[i])) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Layer.WMTS(OpenLayers.Util.applyDefaults(config, { │ │ │ │ │ - url: url, │ │ │ │ │ - requestEncoding: requestEncoding, │ │ │ │ │ - name: layerDef.title, │ │ │ │ │ - style: style.identifier, │ │ │ │ │ - format: format, │ │ │ │ │ - matrixIds: matrixSet.matrixIds, │ │ │ │ │ - matrixSet: matrixSet.identifier, │ │ │ │ │ - projection: projection, │ │ │ │ │ - units: units, │ │ │ │ │ - resolutions: config.isBaseLayer === false ? undefined : resolutions, │ │ │ │ │ - serverResolutions: resolutions, │ │ │ │ │ - tileFullExtent: matrixSet.bounds, │ │ │ │ │ - dimensions: dimensions, │ │ │ │ │ - params: params │ │ │ │ │ - })) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMTSCapabilities" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.CSWGetRecords = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, OpenLayers.Format.CSWGetRecords.DEFAULTS); │ │ │ │ │ - var cls = OpenLayers.Format.CSWGetRecords["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported CSWGetRecords version: " + options.version │ │ │ │ │ - } │ │ │ │ │ - return new cls(options) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Format.CSWGetRecords.DEFAULTS = { │ │ │ │ │ - version: "2.0.2" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Format.CQL = function() { │ │ │ │ │ - var tokens = ["PROPERTY", "COMPARISON", "VALUE", "LOGICAL"], │ │ │ │ │ - patterns = { │ │ │ │ │ - PROPERTY: /^[_a-zA-Z]\w*/, │ │ │ │ │ - COMPARISON: /^(=|<>|<=|<|>=|>|LIKE)/i, │ │ │ │ │ - IS_NULL: /^IS NULL/i, │ │ │ │ │ - COMMA: /^,/, │ │ │ │ │ - LOGICAL: /^(AND|OR)/i, │ │ │ │ │ - VALUE: /^('([^']|'')*'|\d+(\.\d*)?|\.\d+)/, │ │ │ │ │ - LPAREN: /^\(/, │ │ │ │ │ - RPAREN: /^\)/, │ │ │ │ │ - SPATIAL: /^(BBOX|INTERSECTS|DWITHIN|WITHIN|CONTAINS)/i, │ │ │ │ │ - NOT: /^NOT/i, │ │ │ │ │ - BETWEEN: /^BETWEEN/i, │ │ │ │ │ - GEOMETRY: function(text) { │ │ │ │ │ - var type = /^(POINT|LINESTRING|POLYGON|MULTIPOINT|MULTILINESTRING|MULTIPOLYGON|GEOMETRYCOLLECTION)/.exec(text); │ │ │ │ │ - if (type) { │ │ │ │ │ - var len = text.length; │ │ │ │ │ - var idx = text.indexOf("(", type[0].length); │ │ │ │ │ - if (idx > -1) { │ │ │ │ │ - var depth = 1; │ │ │ │ │ - while (idx < len && depth > 0) { │ │ │ │ │ - idx++; │ │ │ │ │ - switch (text.charAt(idx)) { │ │ │ │ │ - case "(": │ │ │ │ │ - depth++; │ │ │ │ │ - break; │ │ │ │ │ - case ")": │ │ │ │ │ - depth--; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + if (feature.fid) { │ │ │ │ │ + entryNode.appendChild(this.createElementNSPlus("atom:id", { │ │ │ │ │ + value: feature.fid │ │ │ │ │ + })) │ │ │ │ │ + } │ │ │ │ │ + if (atomAttrib.links) { │ │ │ │ │ + var links = OpenLayers.Util.isArray(atomAttrib.links) ? atomAttrib.links : [atomAttrib.links]; │ │ │ │ │ + var link; │ │ │ │ │ + for (var i = 0, ii = links.length; i < ii; i++) { │ │ │ │ │ + link = links[i]; │ │ │ │ │ + entryNode.appendChild(this.createElementNSPlus("atom:link", { │ │ │ │ │ + attributes: { │ │ │ │ │ + href: link.href, │ │ │ │ │ + rel: link.rel || null, │ │ │ │ │ + type: link.type || null, │ │ │ │ │ + hreflang: link.hreflang || null, │ │ │ │ │ + title: link.title || null, │ │ │ │ │ + length: link.length || null │ │ │ │ │ } │ │ │ │ │ - return [text.substr(0, idx + 1)] │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - END: /^$/ │ │ │ │ │ - }, │ │ │ │ │ - follows = { │ │ │ │ │ - LPAREN: ["GEOMETRY", "SPATIAL", "PROPERTY", "VALUE", "LPAREN"], │ │ │ │ │ - RPAREN: ["NOT", "LOGICAL", "END", "RPAREN"], │ │ │ │ │ - PROPERTY: ["COMPARISON", "BETWEEN", "COMMA", "IS_NULL"], │ │ │ │ │ - BETWEEN: ["VALUE"], │ │ │ │ │ - IS_NULL: ["END"], │ │ │ │ │ - COMPARISON: ["VALUE"], │ │ │ │ │ - COMMA: ["GEOMETRY", "VALUE", "PROPERTY"], │ │ │ │ │ - VALUE: ["LOGICAL", "COMMA", "RPAREN", "END"], │ │ │ │ │ - SPATIAL: ["LPAREN"], │ │ │ │ │ - LOGICAL: ["NOT", "VALUE", "SPATIAL", "PROPERTY", "LPAREN"], │ │ │ │ │ - NOT: ["PROPERTY", "LPAREN"], │ │ │ │ │ - GEOMETRY: ["COMMA", "RPAREN"] │ │ │ │ │ - }, │ │ │ │ │ - operators = { │ │ │ │ │ - "=": OpenLayers.Filter.Comparison.EQUAL_TO, │ │ │ │ │ - "<>": OpenLayers.Filter.Comparison.NOT_EQUAL_TO, │ │ │ │ │ - "<": OpenLayers.Filter.Comparison.LESS_THAN, │ │ │ │ │ - "<=": OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO, │ │ │ │ │ - ">": OpenLayers.Filter.Comparison.GREATER_THAN, │ │ │ │ │ - ">=": OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO, │ │ │ │ │ - LIKE: OpenLayers.Filter.Comparison.LIKE, │ │ │ │ │ - BETWEEN: OpenLayers.Filter.Comparison.BETWEEN, │ │ │ │ │ - "IS NULL": OpenLayers.Filter.Comparison.IS_NULL │ │ │ │ │ - }, │ │ │ │ │ - operatorReverse = {}, │ │ │ │ │ - logicals = { │ │ │ │ │ - AND: OpenLayers.Filter.Logical.AND, │ │ │ │ │ - OR: OpenLayers.Filter.Logical.OR │ │ │ │ │ - }, │ │ │ │ │ - logicalReverse = {}, │ │ │ │ │ - precedence = { │ │ │ │ │ - RPAREN: 3, │ │ │ │ │ - LOGICAL: 2, │ │ │ │ │ - COMPARISON: 1 │ │ │ │ │ - }; │ │ │ │ │ - var i; │ │ │ │ │ - for (i in operators) { │ │ │ │ │ - if (operators.hasOwnProperty(i)) { │ │ │ │ │ - operatorReverse[operators[i]] = i │ │ │ │ │ + })) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - for (i in logicals) { │ │ │ │ │ - if (logicals.hasOwnProperty(i)) { │ │ │ │ │ - logicalReverse[logicals[i]] = i │ │ │ │ │ + if (atomAttrib.published) { │ │ │ │ │ + entryNode.appendChild(this.createElementNSPlus("atom:published", { │ │ │ │ │ + value: atomAttrib.published │ │ │ │ │ + })) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function tryToken(text, pattern) { │ │ │ │ │ - if (pattern instanceof RegExp) { │ │ │ │ │ - return pattern.exec(text) │ │ │ │ │ - } else { │ │ │ │ │ - return pattern(text) │ │ │ │ │ + if (atomAttrib.rights) { │ │ │ │ │ + entryNode.appendChild(this.createElementNSPlus("atom:rights", { │ │ │ │ │ + value: atomAttrib.rights │ │ │ │ │ + })) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function nextToken(text, tokens) { │ │ │ │ │ - var i, token, len = tokens.length; │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - token = tokens[i]; │ │ │ │ │ - var pat = patterns[token]; │ │ │ │ │ - var matches = tryToken(text, pat); │ │ │ │ │ - if (matches) { │ │ │ │ │ - var match = matches[0]; │ │ │ │ │ - var remainder = text.substr(match.length).replace(/^\s*/, ""); │ │ │ │ │ - return { │ │ │ │ │ - type: token, │ │ │ │ │ - text: match, │ │ │ │ │ - remainder: remainder │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + if (atomAttrib.summary || attrib.description) { │ │ │ │ │ + entryNode.appendChild(this.createElementNSPlus("atom:summary", { │ │ │ │ │ + value: atomAttrib.summary || attrib.description │ │ │ │ │ + })) │ │ │ │ │ } │ │ │ │ │ - var msg = "ERROR: In parsing: [" + text + "], expected one of: "; │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - token = tokens[i]; │ │ │ │ │ - msg += "\n " + token + ": " + patterns[token] │ │ │ │ │ + entryNode.appendChild(this.createElementNSPlus("atom:title", { │ │ │ │ │ + value: atomAttrib.title || attrib.title || this.defaultEntryTitle │ │ │ │ │ + })); │ │ │ │ │ + if (atomAttrib.updated) { │ │ │ │ │ + entryNode.appendChild(this.createElementNSPlus("atom:updated", { │ │ │ │ │ + value: atomAttrib.updated │ │ │ │ │ + })) │ │ │ │ │ } │ │ │ │ │ - throw new Error(msg) │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function tokenize(text) { │ │ │ │ │ - var results = []; │ │ │ │ │ - var token, expect = ["NOT", "GEOMETRY", "SPATIAL", "PROPERTY", "LPAREN"]; │ │ │ │ │ - do { │ │ │ │ │ - token = nextToken(text, expect); │ │ │ │ │ - text = token.remainder; │ │ │ │ │ - expect = follows[token.type]; │ │ │ │ │ - if (token.type != "END" && !expect) { │ │ │ │ │ - throw new Error("No follows list for " + token.type) │ │ │ │ │ - } │ │ │ │ │ - results.push(token) │ │ │ │ │ - } while (token.type != "END"); │ │ │ │ │ - return results │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function buildAst(tokens) { │ │ │ │ │ - var operatorStack = [], │ │ │ │ │ - postfix = []; │ │ │ │ │ - while (tokens.length) { │ │ │ │ │ - var tok = tokens.shift(); │ │ │ │ │ - switch (tok.type) { │ │ │ │ │ - case "PROPERTY": │ │ │ │ │ - case "GEOMETRY": │ │ │ │ │ - case "VALUE": │ │ │ │ │ - postfix.push(tok); │ │ │ │ │ - break; │ │ │ │ │ - case "COMPARISON": │ │ │ │ │ - case "BETWEEN": │ │ │ │ │ - case "IS_NULL": │ │ │ │ │ - case "LOGICAL": │ │ │ │ │ - var p = precedence[tok.type]; │ │ │ │ │ - while (operatorStack.length > 0 && precedence[operatorStack[operatorStack.length - 1].type] <= p) { │ │ │ │ │ - postfix.push(operatorStack.pop()) │ │ │ │ │ - } │ │ │ │ │ - operatorStack.push(tok); │ │ │ │ │ - break; │ │ │ │ │ - case "SPATIAL": │ │ │ │ │ - case "NOT": │ │ │ │ │ - case "LPAREN": │ │ │ │ │ - operatorStack.push(tok); │ │ │ │ │ - break; │ │ │ │ │ - case "RPAREN": │ │ │ │ │ - while (operatorStack.length > 0 && operatorStack[operatorStack.length - 1].type != "LPAREN") { │ │ │ │ │ - postfix.push(operatorStack.pop()) │ │ │ │ │ - } │ │ │ │ │ - operatorStack.pop(); │ │ │ │ │ - if (operatorStack.length > 0 && operatorStack[operatorStack.length - 1].type == "SPATIAL") { │ │ │ │ │ - postfix.push(operatorStack.pop()) │ │ │ │ │ - } │ │ │ │ │ - case "COMMA": │ │ │ │ │ - case "END": │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - throw new Error("Unknown token type " + tok.type) │ │ │ │ │ - } │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + var whereNode = this.createElementNSPlus("georss:where"); │ │ │ │ │ + whereNode.appendChild(this.buildGeometryNode(feature.geometry)); │ │ │ │ │ + entryNode.appendChild(whereNode) │ │ │ │ │ } │ │ │ │ │ - while (operatorStack.length > 0) { │ │ │ │ │ - postfix.push(operatorStack.pop()) │ │ │ │ │ + return entryNode │ │ │ │ │ + }, │ │ │ │ │ + initGmlParser: function() { │ │ │ │ │ + this.gmlParser = new OpenLayers.Format.GML.v3({ │ │ │ │ │ + xy: this.xy, │ │ │ │ │ + featureNS: "http://example.com#feature", │ │ │ │ │ + internalProjection: this.internalProjection, │ │ │ │ │ + externalProjection: this.externalProjection │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + buildGeometryNode: function(geometry) { │ │ │ │ │ + if (!this.gmlParser) { │ │ │ │ │ + this.initGmlParser() │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - function buildTree() { │ │ │ │ │ - var tok = postfix.pop(); │ │ │ │ │ - switch (tok.type) { │ │ │ │ │ - case "LOGICAL": │ │ │ │ │ - var rhs = buildTree(), │ │ │ │ │ - lhs = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Logical({ │ │ │ │ │ - filters: [lhs, rhs], │ │ │ │ │ - type: logicals[tok.text.toUpperCase()] │ │ │ │ │ - }); │ │ │ │ │ - case "NOT": │ │ │ │ │ - var operand = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Logical({ │ │ │ │ │ - filters: [operand], │ │ │ │ │ - type: OpenLayers.Filter.Logical.NOT │ │ │ │ │ - }); │ │ │ │ │ - case "BETWEEN": │ │ │ │ │ - var min, max, property; │ │ │ │ │ - postfix.pop(); │ │ │ │ │ - max = buildTree(); │ │ │ │ │ - min = buildTree(); │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Comparison({ │ │ │ │ │ - property: property, │ │ │ │ │ - lowerBoundary: min, │ │ │ │ │ - upperBoundary: max, │ │ │ │ │ - type: OpenLayers.Filter.Comparison.BETWEEN │ │ │ │ │ - }); │ │ │ │ │ - case "COMPARISON": │ │ │ │ │ - var value = buildTree(), │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Comparison({ │ │ │ │ │ - property: property, │ │ │ │ │ - value: value, │ │ │ │ │ - type: operators[tok.text.toUpperCase()] │ │ │ │ │ - }); │ │ │ │ │ - case "IS_NULL": │ │ │ │ │ - var property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Comparison({ │ │ │ │ │ - property: property, │ │ │ │ │ - type: operators[tok.text.toUpperCase()] │ │ │ │ │ - }); │ │ │ │ │ - case "VALUE": │ │ │ │ │ - var match = tok.text.match(/^'(.*)'$/); │ │ │ │ │ - if (match) { │ │ │ │ │ - return match[1].replace(/''/g, "'") │ │ │ │ │ - } else { │ │ │ │ │ - return Number(tok.text) │ │ │ │ │ - } │ │ │ │ │ - case "SPATIAL": │ │ │ │ │ - switch (tok.text.toUpperCase()) { │ │ │ │ │ - case "BBOX": │ │ │ │ │ - var maxy = buildTree(), │ │ │ │ │ - maxx = buildTree(), │ │ │ │ │ - miny = buildTree(), │ │ │ │ │ - minx = buildTree(), │ │ │ │ │ - prop = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ - property: prop, │ │ │ │ │ - value: OpenLayers.Bounds.fromArray([minx, miny, maxx, maxy]) │ │ │ │ │ - }); │ │ │ │ │ - case "INTERSECTS": │ │ │ │ │ - var value = buildTree(), │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: property, │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - case "WITHIN": │ │ │ │ │ - var value = buildTree(), │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.WITHIN, │ │ │ │ │ - property: property, │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - case "CONTAINS": │ │ │ │ │ - var value = buildTree(), │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.CONTAINS, │ │ │ │ │ - property: property, │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - case "DWITHIN": │ │ │ │ │ - var distance = buildTree(), │ │ │ │ │ - value = buildTree(), │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ - value: value, │ │ │ │ │ - property: property, │ │ │ │ │ - distance: Number(distance) │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - case "GEOMETRY": │ │ │ │ │ - return OpenLayers.Geometry.fromWKT(tok.text); │ │ │ │ │ - default: │ │ │ │ │ - return tok.text │ │ │ │ │ + var node = this.gmlParser.writeNode("feature:_geometry", geometry); │ │ │ │ │ + return node.firstChild │ │ │ │ │ + }, │ │ │ │ │ + buildPersonConstructNode: function(name, value) { │ │ │ │ │ + var oNames = ["uri", "email"]; │ │ │ │ │ + var personNode = this.createElementNSPlus("atom:" + name); │ │ │ │ │ + personNode.appendChild(this.createElementNSPlus("atom:name", { │ │ │ │ │ + value: value.name │ │ │ │ │ + })); │ │ │ │ │ + for (var i = 0, ii = oNames.length; i < ii; i++) { │ │ │ │ │ + if (value[oNames[i]]) { │ │ │ │ │ + personNode.appendChild(this.createElementNSPlus("atom:" + oNames[i], { │ │ │ │ │ + value: value[oNames[i]] │ │ │ │ │ + })) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var result = buildTree(); │ │ │ │ │ - if (postfix.length > 0) { │ │ │ │ │ - var msg = "Remaining tokens after building AST: \n"; │ │ │ │ │ - for (var i = postfix.length - 1; i >= 0; i--) { │ │ │ │ │ - msg += postfix[i].type + ": " + postfix[i].text + "\n" │ │ │ │ │ - } │ │ │ │ │ - throw new Error(msg) │ │ │ │ │ + return personNode │ │ │ │ │ + }, │ │ │ │ │ + getFirstChildValue: function(node, nsuri, name, def) { │ │ │ │ │ + var value; │ │ │ │ │ + var nodes = this.getElementsByTagNameNS(node, nsuri, name); │ │ │ │ │ + if (nodes && nodes.length > 0) { │ │ │ │ │ + value = this.getChildValue(nodes[0], def) │ │ │ │ │ + } else { │ │ │ │ │ + value = def │ │ │ │ │ } │ │ │ │ │ - return result │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ - read: function(text) { │ │ │ │ │ - var result = buildAst(tokenize(text)); │ │ │ │ │ - if (this.keepData) { │ │ │ │ │ - this.data = result │ │ │ │ │ + return value │ │ │ │ │ + }, │ │ │ │ │ + parseFeature: function(node) { │ │ │ │ │ + var atomAttrib = {}; │ │ │ │ │ + var value = null; │ │ │ │ │ + var nodes = null; │ │ │ │ │ + var attval = null; │ │ │ │ │ + var atomns = this.namespaces.atom; │ │ │ │ │ + this.parsePersonConstructs(node, "author", atomAttrib); │ │ │ │ │ + nodes = this.getElementsByTagNameNS(node, atomns, "category"); │ │ │ │ │ + if (nodes.length > 0) { │ │ │ │ │ + atomAttrib.categories = [] │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ + value = {}; │ │ │ │ │ + value.term = nodes[i].getAttribute("term"); │ │ │ │ │ + attval = nodes[i].getAttribute("scheme"); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value.scheme = attval │ │ │ │ │ } │ │ │ │ │ - return result │ │ │ │ │ - }, │ │ │ │ │ - write: function(filter) { │ │ │ │ │ - if (filter instanceof OpenLayers.Geometry) { │ │ │ │ │ - return filter.toString() │ │ │ │ │ + attval = nodes[i].getAttribute("label"); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value.label = attval │ │ │ │ │ } │ │ │ │ │ - switch (filter.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Filter.Spatial": │ │ │ │ │ - switch (filter.type) { │ │ │ │ │ - case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ - return "BBOX(" + filter.property + "," + filter.value.toBBOX() + ")"; │ │ │ │ │ - case OpenLayers.Filter.Spatial.DWITHIN: │ │ │ │ │ - return "DWITHIN(" + filter.property + ", " + this.write(filter.value) + ", " + filter.distance + ")"; │ │ │ │ │ - case OpenLayers.Filter.Spatial.WITHIN: │ │ │ │ │ - return "WITHIN(" + filter.property + ", " + this.write(filter.value) + ")"; │ │ │ │ │ - case OpenLayers.Filter.Spatial.INTERSECTS: │ │ │ │ │ - return "INTERSECTS(" + filter.property + ", " + this.write(filter.value) + ")"; │ │ │ │ │ - case OpenLayers.Filter.Spatial.CONTAINS: │ │ │ │ │ - return "CONTAINS(" + filter.property + ", " + this.write(filter.value) + ")"; │ │ │ │ │ - default: │ │ │ │ │ - throw new Error("Unknown spatial filter type: " + filter.type) │ │ │ │ │ - } │ │ │ │ │ - case "OpenLayers.Filter.Logical": │ │ │ │ │ - if (filter.type == OpenLayers.Filter.Logical.NOT) { │ │ │ │ │ - return "NOT (" + this.write(filter.filters[0]) + ")" │ │ │ │ │ - } else { │ │ │ │ │ - var res = "("; │ │ │ │ │ - var first = true; │ │ │ │ │ - for (var i = 0; i < filter.filters.length; i++) { │ │ │ │ │ - if (first) { │ │ │ │ │ - first = false │ │ │ │ │ - } else { │ │ │ │ │ - res += ") " + logicalReverse[filter.type] + " (" │ │ │ │ │ - } │ │ │ │ │ - res += this.write(filter.filters[i]) │ │ │ │ │ - } │ │ │ │ │ - return res + ")" │ │ │ │ │ - } │ │ │ │ │ - case "OpenLayers.Filter.Comparison": │ │ │ │ │ - if (filter.type == OpenLayers.Filter.Comparison.BETWEEN) { │ │ │ │ │ - return filter.property + " BETWEEN " + this.write(filter.lowerBoundary) + " AND " + this.write(filter.upperBoundary) │ │ │ │ │ - } else { │ │ │ │ │ - return filter.value !== null ? filter.property + " " + operatorReverse[filter.type] + " " + this.write(filter.value) : filter.property + " " + operatorReverse[filter.type] │ │ │ │ │ - } │ │ │ │ │ - case undefined: │ │ │ │ │ - if (typeof filter === "string") { │ │ │ │ │ - return "'" + filter.replace(/'/g, "''") + "'" │ │ │ │ │ - } else if (typeof filter === "number") { │ │ │ │ │ - return String(filter) │ │ │ │ │ - } │ │ │ │ │ - default: │ │ │ │ │ - throw new Error("Can't encode: " + filter.CLASS_NAME + " " + filter) │ │ │ │ │ + atomAttrib.categories.push(value) │ │ │ │ │ + } │ │ │ │ │ + nodes = this.getElementsByTagNameNS(node, atomns, "content"); │ │ │ │ │ + if (nodes.length > 0) { │ │ │ │ │ + value = {}; │ │ │ │ │ + attval = nodes[0].getAttribute("type"); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value.type = attval │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.CQL" │ │ │ │ │ - }) │ │ │ │ │ -}(); │ │ │ │ │ -OpenLayers.Format.WFSDescribeFeatureType = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: /^\s*|\s*$/g │ │ │ │ │ - }, │ │ │ │ │ - namespaces: { │ │ │ │ │ - xsd: "http://www.w3.org/2001/XMLSchema" │ │ │ │ │ - }, │ │ │ │ │ - readers: { │ │ │ │ │ - xsd: { │ │ │ │ │ - schema: function(node, obj) { │ │ │ │ │ - var complexTypes = []; │ │ │ │ │ - var customTypes = {}; │ │ │ │ │ - var schema = { │ │ │ │ │ - complexTypes: complexTypes, │ │ │ │ │ - customTypes: customTypes │ │ │ │ │ - }; │ │ │ │ │ - var i, len; │ │ │ │ │ - this.readChildNodes(node, schema); │ │ │ │ │ - var attributes = node.attributes; │ │ │ │ │ - var attr, name; │ │ │ │ │ - for (i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ - attr = attributes[i]; │ │ │ │ │ - name = attr.name; │ │ │ │ │ - if (name.indexOf("xmlns") === 0) { │ │ │ │ │ - this.setNamespace(name.split(":")[1] || "", attr.value) │ │ │ │ │ - } else { │ │ │ │ │ - obj[name] = attr.value │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - obj.featureTypes = complexTypes; │ │ │ │ │ - obj.targetPrefix = this.namespaceAlias[obj.targetNamespace]; │ │ │ │ │ - var complexType, customType; │ │ │ │ │ - for (i = 0, len = complexTypes.length; i < len; ++i) { │ │ │ │ │ - complexType = complexTypes[i]; │ │ │ │ │ - customType = customTypes[complexType.typeName]; │ │ │ │ │ - if (customTypes[complexType.typeName]) { │ │ │ │ │ - complexType.typeName = customType.name │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - complexType: function(node, obj) { │ │ │ │ │ - var complexType = { │ │ │ │ │ - typeName: node.getAttribute("name") │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, complexType); │ │ │ │ │ - obj.complexTypes.push(complexType) │ │ │ │ │ - }, │ │ │ │ │ - complexContent: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - extension: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - sequence: function(node, obj) { │ │ │ │ │ - var sequence = { │ │ │ │ │ - elements: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, sequence); │ │ │ │ │ - obj.properties = sequence.elements │ │ │ │ │ - }, │ │ │ │ │ - element: function(node, obj) { │ │ │ │ │ - var type; │ │ │ │ │ - if (obj.elements) { │ │ │ │ │ - var element = {}; │ │ │ │ │ - var attributes = node.attributes; │ │ │ │ │ - var attr; │ │ │ │ │ - for (var i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ - attr = attributes[i]; │ │ │ │ │ - element[attr.name] = attr.value │ │ │ │ │ - } │ │ │ │ │ - type = element.type; │ │ │ │ │ - if (!type) { │ │ │ │ │ - type = {}; │ │ │ │ │ - this.readChildNodes(node, type); │ │ │ │ │ - element.restriction = type; │ │ │ │ │ - element.type = type.base │ │ │ │ │ - } │ │ │ │ │ - var fullType = type.base || type; │ │ │ │ │ - element.localType = fullType.split(":").pop(); │ │ │ │ │ - obj.elements.push(element); │ │ │ │ │ - this.readChildNodes(node, element) │ │ │ │ │ - } │ │ │ │ │ - if (obj.complexTypes) { │ │ │ │ │ - type = node.getAttribute("type"); │ │ │ │ │ - var localType = type.split(":").pop(); │ │ │ │ │ - obj.customTypes[localType] = { │ │ │ │ │ - name: node.getAttribute("name"), │ │ │ │ │ - type: type │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - annotation: function(node, obj) { │ │ │ │ │ - obj.annotation = {}; │ │ │ │ │ - this.readChildNodes(node, obj.annotation) │ │ │ │ │ - }, │ │ │ │ │ - appinfo: function(node, obj) { │ │ │ │ │ - if (!obj.appinfo) { │ │ │ │ │ - obj.appinfo = [] │ │ │ │ │ - } │ │ │ │ │ - obj.appinfo.push(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - documentation: function(node, obj) { │ │ │ │ │ - if (!obj.documentation) { │ │ │ │ │ - obj.documentation = [] │ │ │ │ │ + attval = nodes[0].getAttribute("src"); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value.src = attval │ │ │ │ │ + } else { │ │ │ │ │ + if (value.type == "text" || value.type == "html" || value.type == null) { │ │ │ │ │ + value.value = this.getFirstChildValue(node, atomns, "content", null) │ │ │ │ │ + } else if (value.type == "xhtml" || value.type.match(/(\+|\/)xml$/)) { │ │ │ │ │ + value.value = this.getChildEl(nodes[0]) │ │ │ │ │ + } else { │ │ │ │ │ + value.value = this.getFirstChildValue(node, atomns, "content", null) │ │ │ │ │ } │ │ │ │ │ - var value = this.getChildValue(node); │ │ │ │ │ - obj.documentation.push({ │ │ │ │ │ - lang: node.getAttribute("xml:lang"), │ │ │ │ │ - textContent: value.replace(this.regExes.trimSpace, "") │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - simpleType: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - restriction: function(node, obj) { │ │ │ │ │ - obj.base = node.getAttribute("base"); │ │ │ │ │ - this.readRestriction(node, obj) │ │ │ │ │ + atomAttrib.content = value │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - readRestriction: function(node, obj) { │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - var child, nodeName, value; │ │ │ │ │ - for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ - child = children[i]; │ │ │ │ │ - if (child.nodeType == 1) { │ │ │ │ │ - nodeName = child.nodeName.split(":").pop(); │ │ │ │ │ - value = child.getAttribute("value"); │ │ │ │ │ - if (!obj[nodeName]) { │ │ │ │ │ - obj[nodeName] = value │ │ │ │ │ - } else { │ │ │ │ │ - if (typeof obj[nodeName] == "string") { │ │ │ │ │ - obj[nodeName] = [obj[nodeName]] │ │ │ │ │ - } │ │ │ │ │ - obj[nodeName].push(value) │ │ │ │ │ + this.parsePersonConstructs(node, "contributor", atomAttrib); │ │ │ │ │ + atomAttrib.id = this.getFirstChildValue(node, atomns, "id", null); │ │ │ │ │ + nodes = this.getElementsByTagNameNS(node, atomns, "link"); │ │ │ │ │ + if (nodes.length > 0) { │ │ │ │ │ + atomAttrib.links = new Array(nodes.length) │ │ │ │ │ + } │ │ │ │ │ + var oAtts = ["rel", "type", "hreflang", "title", "length"]; │ │ │ │ │ + for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ + value = {}; │ │ │ │ │ + value.href = nodes[i].getAttribute("href"); │ │ │ │ │ + for (var j = 0, jj = oAtts.length; j < jj; j++) { │ │ │ │ │ + attval = nodes[i].getAttribute(oAtts[j]); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value[oAtts[j]] = attval │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + atomAttrib.links[i] = value │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + value = this.getFirstChildValue(node, atomns, "published", null); │ │ │ │ │ + if (value) { │ │ │ │ │ + atomAttrib.published = value │ │ │ │ │ } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement │ │ │ │ │ + value = this.getFirstChildValue(node, atomns, "rights", null); │ │ │ │ │ + if (value) { │ │ │ │ │ + atomAttrib.rights = value │ │ │ │ │ } │ │ │ │ │ - var schema = {}; │ │ │ │ │ - if (data.nodeName.split(":").pop() === "ExceptionReport") { │ │ │ │ │ - var parser = new OpenLayers.Format.OGCExceptionReport; │ │ │ │ │ - schema.error = parser.read(data) │ │ │ │ │ - } else { │ │ │ │ │ - this.readNode(data, schema) │ │ │ │ │ + value = this.getFirstChildValue(node, atomns, "summary", null); │ │ │ │ │ + if (value) { │ │ │ │ │ + atomAttrib.summary = value │ │ │ │ │ } │ │ │ │ │ - return schema │ │ │ │ │ + atomAttrib.title = this.getFirstChildValue(node, atomns, "title", null); │ │ │ │ │ + atomAttrib.updated = this.getFirstChildValue(node, atomns, "updated", null); │ │ │ │ │ + var featureAttrib = { │ │ │ │ │ + title: atomAttrib.title, │ │ │ │ │ + description: atomAttrib.summary, │ │ │ │ │ + atom: atomAttrib │ │ │ │ │ + }; │ │ │ │ │ + var geometry = this.parseLocations(node)[0]; │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry, featureAttrib); │ │ │ │ │ + feature.fid = atomAttrib.id; │ │ │ │ │ + return feature │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSDescribeFeatureType" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.SLD = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - profile: null, │ │ │ │ │ - defaultVersion: "1.0.0", │ │ │ │ │ - stringifyOutput: true, │ │ │ │ │ - namedLayersAsArray: false, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SLD" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WFS = OpenLayers.Class(OpenLayers.Format.GML, { │ │ │ │ │ - layer: null, │ │ │ │ │ - wfsns: "http://www.opengis.net/wfs", │ │ │ │ │ - ogcns: "http://www.opengis.net/ogc", │ │ │ │ │ - initialize: function(options, layer) { │ │ │ │ │ - OpenLayers.Format.GML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - if (this.layer.featureNS) { │ │ │ │ │ - this.featureNS = this.layer.featureNS │ │ │ │ │ - } │ │ │ │ │ - if (this.layer.options.geometry_column) { │ │ │ │ │ - this.geometryName = this.layer.options.geometry_column │ │ │ │ │ + parseFeatures: function(node) { │ │ │ │ │ + var features = []; │ │ │ │ │ + var entries = this.getElementsByTagNameNS(node, this.namespaces.atom, "entry"); │ │ │ │ │ + if (entries.length == 0) { │ │ │ │ │ + entries = [node] │ │ │ │ │ } │ │ │ │ │ - if (this.layer.options.typename) { │ │ │ │ │ - this.featureName = this.layer.options.typename │ │ │ │ │ + for (var i = 0, ii = entries.length; i < ii; i++) { │ │ │ │ │ + features.push(this.parseFeature(entries[i])) │ │ │ │ │ } │ │ │ │ │ + return features │ │ │ │ │ }, │ │ │ │ │ - write: function(features) { │ │ │ │ │ - var transaction = this.createElementNS(this.wfsns, "wfs:Transaction"); │ │ │ │ │ - transaction.setAttribute("version", "1.0.0"); │ │ │ │ │ - transaction.setAttribute("service", "WFS"); │ │ │ │ │ - for (var i = 0; i < features.length; i++) { │ │ │ │ │ - switch (features[i].state) { │ │ │ │ │ - case OpenLayers.State.INSERT: │ │ │ │ │ - transaction.appendChild(this.insert(features[i])); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.State.UPDATE: │ │ │ │ │ - transaction.appendChild(this.update(features[i])); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.State.DELETE: │ │ │ │ │ - transaction.appendChild(this.remove(features[i])); │ │ │ │ │ - break │ │ │ │ │ + parseLocations: function(node) { │ │ │ │ │ + var georssns = this.namespaces.georss; │ │ │ │ │ + var locations = { │ │ │ │ │ + components: [] │ │ │ │ │ + }; │ │ │ │ │ + var where = this.getElementsByTagNameNS(node, georssns, "where"); │ │ │ │ │ + if (where && where.length > 0) { │ │ │ │ │ + if (!this.gmlParser) { │ │ │ │ │ + this.initGmlParser() │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, ii = where.length; i < ii; i++) { │ │ │ │ │ + this.gmlParser.readChildNodes(where[i], locations) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [transaction]) │ │ │ │ │ - }, │ │ │ │ │ - createFeatureXML: function(feature) { │ │ │ │ │ - var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ - var geomContainer = this.createElementNS(this.featureNS, "feature:" + this.geometryName); │ │ │ │ │ - geomContainer.appendChild(geometryNode); │ │ │ │ │ - var featureContainer = this.createElementNS(this.featureNS, "feature:" + this.featureName); │ │ │ │ │ - featureContainer.appendChild(geomContainer); │ │ │ │ │ - for (var attr in feature.attributes) { │ │ │ │ │ - var attrText = this.createTextNode(feature.attributes[attr]); │ │ │ │ │ - var nodename = attr; │ │ │ │ │ - if (attr.search(":") != -1) { │ │ │ │ │ - nodename = attr.split(":")[1] │ │ │ │ │ + var components = locations.components; │ │ │ │ │ + var point = this.getElementsByTagNameNS(node, georssns, "point"); │ │ │ │ │ + if (point && point.length > 0) { │ │ │ │ │ + for (var i = 0, ii = point.length; i < ii; i++) { │ │ │ │ │ + var xy = OpenLayers.String.trim(point[i].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ + if (xy.length != 2) { │ │ │ │ │ + xy = OpenLayers.String.trim(point[i].firstChild.nodeValue).split(/\s*,\s*/) │ │ │ │ │ + } │ │ │ │ │ + components.push(new OpenLayers.Geometry.Point(xy[1], xy[0])) │ │ │ │ │ } │ │ │ │ │ - var attrContainer = this.createElementNS(this.featureNS, "feature:" + nodename); │ │ │ │ │ - attrContainer.appendChild(attrText); │ │ │ │ │ - featureContainer.appendChild(attrContainer) │ │ │ │ │ } │ │ │ │ │ - return featureContainer │ │ │ │ │ - }, │ │ │ │ │ - insert: function(feature) { │ │ │ │ │ - var insertNode = this.createElementNS(this.wfsns, "wfs:Insert"); │ │ │ │ │ - insertNode.appendChild(this.createFeatureXML(feature)); │ │ │ │ │ - return insertNode │ │ │ │ │ - }, │ │ │ │ │ - update: function(feature) { │ │ │ │ │ - if (!feature.fid) { │ │ │ │ │ - OpenLayers.Console.userError(OpenLayers.i18n("noFID")) │ │ │ │ │ + var line = this.getElementsByTagNameNS(node, georssns, "line"); │ │ │ │ │ + if (line && line.length > 0) { │ │ │ │ │ + var coords; │ │ │ │ │ + var p; │ │ │ │ │ + var points; │ │ │ │ │ + for (var i = 0, ii = line.length; i < ii; i++) { │ │ │ │ │ + coords = OpenLayers.String.trim(line[i].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ + points = []; │ │ │ │ │ + for (var j = 0, jj = coords.length; j < jj; j += 2) { │ │ │ │ │ + p = new OpenLayers.Geometry.Point(coords[j + 1], coords[j]); │ │ │ │ │ + points.push(p) │ │ │ │ │ + } │ │ │ │ │ + components.push(new OpenLayers.Geometry.LineString(points)) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var updateNode = this.createElementNS(this.wfsns, "wfs:Update"); │ │ │ │ │ - updateNode.setAttribute("typeName", this.featurePrefix + ":" + this.featureName); │ │ │ │ │ - updateNode.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ - var propertyNode = this.createElementNS(this.wfsns, "wfs:Property"); │ │ │ │ │ - var nameNode = this.createElementNS(this.wfsns, "wfs:Name"); │ │ │ │ │ - var txtNode = this.createTextNode(this.geometryName); │ │ │ │ │ - nameNode.appendChild(txtNode); │ │ │ │ │ - propertyNode.appendChild(nameNode); │ │ │ │ │ - var valueNode = this.createElementNS(this.wfsns, "wfs:Value"); │ │ │ │ │ - var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ - if (feature.layer) { │ │ │ │ │ - geometryNode.setAttribute("srsName", feature.layer.projection.getCode()) │ │ │ │ │ + var polygon = this.getElementsByTagNameNS(node, georssns, "polygon"); │ │ │ │ │ + if (polygon && polygon.length > 0) { │ │ │ │ │ + var coords; │ │ │ │ │ + var p; │ │ │ │ │ + var points; │ │ │ │ │ + for (var i = 0, ii = polygon.length; i < ii; i++) { │ │ │ │ │ + coords = OpenLayers.String.trim(polygon[i].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ + points = []; │ │ │ │ │ + for (var j = 0, jj = coords.length; j < jj; j += 2) { │ │ │ │ │ + p = new OpenLayers.Geometry.Point(coords[j + 1], coords[j]); │ │ │ │ │ + points.push(p) │ │ │ │ │ + } │ │ │ │ │ + components.push(new OpenLayers.Geometry.Polygon([new OpenLayers.Geometry.LinearRing(points)])) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - valueNode.appendChild(geometryNode); │ │ │ │ │ - propertyNode.appendChild(valueNode); │ │ │ │ │ - updateNode.appendChild(propertyNode); │ │ │ │ │ - for (var propName in feature.attributes) { │ │ │ │ │ - propertyNode = this.createElementNS(this.wfsns, "wfs:Property"); │ │ │ │ │ - nameNode = this.createElementNS(this.wfsns, "wfs:Name"); │ │ │ │ │ - nameNode.appendChild(this.createTextNode(propName)); │ │ │ │ │ - propertyNode.appendChild(nameNode); │ │ │ │ │ - valueNode = this.createElementNS(this.wfsns, "wfs:Value"); │ │ │ │ │ - valueNode.appendChild(this.createTextNode(feature.attributes[propName])); │ │ │ │ │ - propertyNode.appendChild(valueNode); │ │ │ │ │ - updateNode.appendChild(propertyNode) │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + for (var i = 0, ii = components.length; i < ii; i++) { │ │ │ │ │ + if (components[i]) { │ │ │ │ │ + components[i].transform(this.externalProjection, this.internalProjection) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var filterNode = this.createElementNS(this.ogcns, "ogc:Filter"); │ │ │ │ │ - var filterIdNode = this.createElementNS(this.ogcns, "ogc:FeatureId"); │ │ │ │ │ - filterIdNode.setAttribute("fid", feature.fid); │ │ │ │ │ - filterNode.appendChild(filterIdNode); │ │ │ │ │ - updateNode.appendChild(filterNode); │ │ │ │ │ - return updateNode │ │ │ │ │ + return components │ │ │ │ │ }, │ │ │ │ │ - remove: function(feature) { │ │ │ │ │ - if (!feature.fid) { │ │ │ │ │ - OpenLayers.Console.userError(OpenLayers.i18n("noFID")); │ │ │ │ │ - return false │ │ │ │ │ + parsePersonConstructs: function(node, name, data) { │ │ │ │ │ + var persons = []; │ │ │ │ │ + var atomns = this.namespaces.atom; │ │ │ │ │ + var nodes = this.getElementsByTagNameNS(node, atomns, name); │ │ │ │ │ + var oAtts = ["uri", "email"]; │ │ │ │ │ + for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ + var value = {}; │ │ │ │ │ + value.name = this.getFirstChildValue(nodes[i], atomns, "name", null); │ │ │ │ │ + for (var j = 0, jj = oAtts.length; j < jj; j++) { │ │ │ │ │ + var attval = this.getFirstChildValue(nodes[i], atomns, oAtts[j], null); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value[oAtts[j]] = attval │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + persons.push(value) │ │ │ │ │ + } │ │ │ │ │ + if (persons.length > 0) { │ │ │ │ │ + data[name + "s"] = persons │ │ │ │ │ } │ │ │ │ │ - var deleteNode = this.createElementNS(this.wfsns, "wfs:Delete"); │ │ │ │ │ - deleteNode.setAttribute("typeName", this.featurePrefix + ":" + this.featureName); │ │ │ │ │ - deleteNode.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ - var filterNode = this.createElementNS(this.ogcns, "ogc:Filter"); │ │ │ │ │ - var filterIdNode = this.createElementNS(this.ogcns, "ogc:FeatureId"); │ │ │ │ │ - filterIdNode.setAttribute("fid", feature.fid); │ │ │ │ │ - filterNode.appendChild(filterIdNode); │ │ │ │ │ - deleteNode.appendChild(filterNode); │ │ │ │ │ - return deleteNode │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.layer = null │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFS" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.XLS = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - defaultVersion: "1.1.0", │ │ │ │ │ - stringifyOutput: true, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.XLS" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.Atom" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.GML.v2 = OpenLayers.Class(OpenLayers.Format.GML.Base, { │ │ │ │ │ - schemaLocation: "http://www.opengis.net/gml http://schemas.opengis.net/gml/2.1.2/feature.xsd", │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.GML.Base.prototype.initialize.apply(this, [options]) │ │ │ │ │ +OpenLayers.Format.SOSGetObservation = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + namespaces: { │ │ │ │ │ + ows: "http://www.opengis.net/ows", │ │ │ │ │ + gml: "http://www.opengis.net/gml", │ │ │ │ │ + sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ + ogc: "http://www.opengis.net/ogc", │ │ │ │ │ + om: "http://www.opengis.net/om/1.0", │ │ │ │ │ + sa: "http://www.opengis.net/sampling/1.0", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: /^\s*|\s*$/g, │ │ │ │ │ + removeSpace: /\s*/g, │ │ │ │ │ + splitSpace: /\s+/, │ │ │ │ │ + trimComma: /\s*,\s*/g │ │ │ │ │ + }, │ │ │ │ │ + VERSION: "1.0.0", │ │ │ │ │ + schemaLocation: "http://www.opengis.net/sos/1.0 http://schemas.opengis.net/sos/1.0.0/sosGetObservation.xsd", │ │ │ │ │ + defaultPrefix: "sos", │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement │ │ │ │ │ + } │ │ │ │ │ + var info = { │ │ │ │ │ + measurements: [], │ │ │ │ │ + observations: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readNode(data, info); │ │ │ │ │ + return info │ │ │ │ │ + }, │ │ │ │ │ + write: function(options) { │ │ │ │ │ + var node = this.writeNode("sos:GetObservation", options); │ │ │ │ │ + node.setAttribute("xmlns:om", this.namespaces.om); │ │ │ │ │ + node.setAttribute("xmlns:ogc", this.namespaces.ogc); │ │ │ │ │ + this.setAttributeNS(node, this.namespaces.xsi, "xsi:schemaLocation", this.schemaLocation); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [node]) │ │ │ │ │ }, │ │ │ │ │ readers: { │ │ │ │ │ - gml: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - outerBoundaryIs: function(node, container) { │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - container.outer = obj.components[0] │ │ │ │ │ + om: { │ │ │ │ │ + ObservationCollection: function(node, obj) { │ │ │ │ │ + obj.id = this.getAttributeNS(node, this.namespaces.gml, "id"); │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ }, │ │ │ │ │ - innerBoundaryIs: function(node, container) { │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - container.inner.push(obj.components[0]) │ │ │ │ │ + member: function(node, observationCollection) { │ │ │ │ │ + this.readChildNodes(node, observationCollection) │ │ │ │ │ }, │ │ │ │ │ - Box: function(node, container) { │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - if (!container.components) { │ │ │ │ │ - container.components = [] │ │ │ │ │ + Measurement: function(node, observationCollection) { │ │ │ │ │ + var measurement = {}; │ │ │ │ │ + observationCollection.measurements.push(measurement); │ │ │ │ │ + this.readChildNodes(node, measurement) │ │ │ │ │ + }, │ │ │ │ │ + Observation: function(node, observationCollection) { │ │ │ │ │ + var observation = {}; │ │ │ │ │ + observationCollection.observations.push(observation); │ │ │ │ │ + this.readChildNodes(node, observation) │ │ │ │ │ + }, │ │ │ │ │ + samplingTime: function(node, measurement) { │ │ │ │ │ + var samplingTime = {}; │ │ │ │ │ + measurement.samplingTime = samplingTime; │ │ │ │ │ + this.readChildNodes(node, samplingTime) │ │ │ │ │ + }, │ │ │ │ │ + observedProperty: function(node, measurement) { │ │ │ │ │ + measurement.observedProperty = this.getAttributeNS(node, this.namespaces.xlink, "href"); │ │ │ │ │ + this.readChildNodes(node, measurement) │ │ │ │ │ + }, │ │ │ │ │ + procedure: function(node, measurement) { │ │ │ │ │ + measurement.procedure = this.getAttributeNS(node, this.namespaces.xlink, "href"); │ │ │ │ │ + this.readChildNodes(node, measurement) │ │ │ │ │ + }, │ │ │ │ │ + featureOfInterest: function(node, observation) { │ │ │ │ │ + var foi = { │ │ │ │ │ + features: [] │ │ │ │ │ + }; │ │ │ │ │ + observation.fois = []; │ │ │ │ │ + observation.fois.push(foi); │ │ │ │ │ + this.readChildNodes(node, foi); │ │ │ │ │ + var features = []; │ │ │ │ │ + for (var i = 0, len = foi.features.length; i < len; i++) { │ │ │ │ │ + var feature = foi.features[i]; │ │ │ │ │ + features.push(new OpenLayers.Feature.Vector(feature.components[0], feature.attributes)) │ │ │ │ │ + } │ │ │ │ │ + foi.features = features │ │ │ │ │ + }, │ │ │ │ │ + result: function(node, measurement) { │ │ │ │ │ + var result = {}; │ │ │ │ │ + measurement.result = result; │ │ │ │ │ + if (this.getChildValue(node) !== "") { │ │ │ │ │ + result.value = this.getChildValue(node); │ │ │ │ │ + result.uom = node.getAttribute("uom") │ │ │ │ │ + } else { │ │ │ │ │ + this.readChildNodes(node, result) │ │ │ │ │ } │ │ │ │ │ - var min = obj.points[0]; │ │ │ │ │ - var max = obj.points[1]; │ │ │ │ │ - container.components.push(new OpenLayers.Bounds(min.x, min.y, max.x, max.y)) │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.GML.Base.prototype.readers["gml"]), │ │ │ │ │ - feature: OpenLayers.Format.GML.Base.prototype.readers["feature"], │ │ │ │ │ - wfs: OpenLayers.Format.GML.Base.prototype.readers["wfs"] │ │ │ │ │ - }, │ │ │ │ │ - write: function(features) { │ │ │ │ │ - var name; │ │ │ │ │ - if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ - name = "wfs:FeatureCollection" │ │ │ │ │ - } else { │ │ │ │ │ - name = "gml:featureMember" │ │ │ │ │ - } │ │ │ │ │ - var root = this.writeNode(name, features); │ │ │ │ │ - this.setAttributeNS(root, this.namespaces["xsi"], "xsi:schemaLocation", this.schemaLocation); │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [root]) │ │ │ │ │ - }, │ │ │ │ │ - writers: { │ │ │ │ │ + }, │ │ │ │ │ + sa: OpenLayers.Format.SOSGetFeatureOfInterest.prototype.readers.sa, │ │ │ │ │ gml: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Point: function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:Point"); │ │ │ │ │ - this.writeNode("coordinates", [geometry], node); │ │ │ │ │ - return node │ │ │ │ │ + TimeInstant: function(node, samplingTime) { │ │ │ │ │ + var timeInstant = {}; │ │ │ │ │ + samplingTime.timeInstant = timeInstant; │ │ │ │ │ + this.readChildNodes(node, timeInstant) │ │ │ │ │ }, │ │ │ │ │ - coordinates: function(points) { │ │ │ │ │ - var numPoints = points.length; │ │ │ │ │ - var parts = new Array(numPoints); │ │ │ │ │ - var point; │ │ │ │ │ - for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ - point = points[i]; │ │ │ │ │ - if (this.xy) { │ │ │ │ │ - parts[i] = point.x + "," + point.y │ │ │ │ │ - } else { │ │ │ │ │ - parts[i] = point.y + "," + point.x │ │ │ │ │ - } │ │ │ │ │ - if (point.z != undefined) { │ │ │ │ │ - parts[i] += "," + point.z │ │ │ │ │ + timePosition: function(node, timeInstant) { │ │ │ │ │ + timeInstant.timePosition = this.getChildValue(node) │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.SOSGetFeatureOfInterest.prototype.readers.gml) │ │ │ │ │ + }, │ │ │ │ │ + writers: { │ │ │ │ │ + sos: { │ │ │ │ │ + GetObservation: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("GetObservation", { │ │ │ │ │ + attributes: { │ │ │ │ │ + version: this.VERSION, │ │ │ │ │ + service: "SOS" │ │ │ │ │ } │ │ │ │ │ + }); │ │ │ │ │ + this.writeNode("offering", options, node); │ │ │ │ │ + if (options.eventTime) { │ │ │ │ │ + this.writeNode("eventTime", options, node) │ │ │ │ │ } │ │ │ │ │ - return this.createElementNSPlus("gml:coordinates", { │ │ │ │ │ - attributes: { │ │ │ │ │ - decimal: ".", │ │ │ │ │ - cs: ",", │ │ │ │ │ - ts: " " │ │ │ │ │ - }, │ │ │ │ │ - value: numPoints == 1 ? parts[0] : parts.join(" ") │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - LineString: function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:LineString"); │ │ │ │ │ - this.writeNode("coordinates", geometry.components, node); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - Polygon: function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:Polygon"); │ │ │ │ │ - this.writeNode("outerBoundaryIs", geometry.components[0], node); │ │ │ │ │ - for (var i = 1; i < geometry.components.length; ++i) { │ │ │ │ │ - this.writeNode("innerBoundaryIs", geometry.components[i], node) │ │ │ │ │ + for (var procedure in options.procedures) { │ │ │ │ │ + this.writeNode("procedure", options.procedures[procedure], node) │ │ │ │ │ + } │ │ │ │ │ + for (var observedProperty in options.observedProperties) { │ │ │ │ │ + this.writeNode("observedProperty", options.observedProperties[observedProperty], node) │ │ │ │ │ + } │ │ │ │ │ + if (options.foi) { │ │ │ │ │ + this.writeNode("featureOfInterest", options.foi, node) │ │ │ │ │ + } │ │ │ │ │ + this.writeNode("responseFormat", options, node); │ │ │ │ │ + if (options.resultModel) { │ │ │ │ │ + this.writeNode("resultModel", options, node) │ │ │ │ │ + } │ │ │ │ │ + if (options.responseMode) { │ │ │ │ │ + this.writeNode("responseMode", options, node) │ │ │ │ │ } │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - outerBoundaryIs: function(ring) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:outerBoundaryIs"); │ │ │ │ │ - this.writeNode("LinearRing", ring, node); │ │ │ │ │ + featureOfInterest: function(foi) { │ │ │ │ │ + var node = this.createElementNSPlus("featureOfInterest"); │ │ │ │ │ + this.writeNode("ObjectID", foi.objectId, node); │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - innerBoundaryIs: function(ring) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:innerBoundaryIs"); │ │ │ │ │ - this.writeNode("LinearRing", ring, node); │ │ │ │ │ - return node │ │ │ │ │ + ObjectID: function(options) { │ │ │ │ │ + return this.createElementNSPlus("ObjectID", { │ │ │ │ │ + value: options │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - LinearRing: function(ring) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:LinearRing"); │ │ │ │ │ - this.writeNode("coordinates", ring.components, node); │ │ │ │ │ - return node │ │ │ │ │ + responseFormat: function(options) { │ │ │ │ │ + return this.createElementNSPlus("responseFormat", { │ │ │ │ │ + value: options.responseFormat │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - Box: function(bounds) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:Box"); │ │ │ │ │ - this.writeNode("coordinates", [{ │ │ │ │ │ - x: bounds.left, │ │ │ │ │ - y: bounds.bottom │ │ │ │ │ - }, { │ │ │ │ │ - x: bounds.right, │ │ │ │ │ - y: bounds.top │ │ │ │ │ - }], node); │ │ │ │ │ - if (this.srsName) { │ │ │ │ │ - node.setAttribute("srsName", this.srsName) │ │ │ │ │ + procedure: function(procedure) { │ │ │ │ │ + return this.createElementNSPlus("procedure", { │ │ │ │ │ + value: procedure │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + offering: function(options) { │ │ │ │ │ + return this.createElementNSPlus("offering", { │ │ │ │ │ + value: options.offering │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + observedProperty: function(observedProperty) { │ │ │ │ │ + return this.createElementNSPlus("observedProperty", { │ │ │ │ │ + value: observedProperty │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + eventTime: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("eventTime"); │ │ │ │ │ + if (options.eventTime === "latest") { │ │ │ │ │ + this.writeNode("ogc:TM_Equals", options, node) │ │ │ │ │ } │ │ │ │ │ return node │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.GML.Base.prototype.writers["gml"]), │ │ │ │ │ - feature: OpenLayers.Format.GML.Base.prototype.writers["feature"], │ │ │ │ │ - wfs: OpenLayers.Format.GML.Base.prototype.writers["wfs"] │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.GML.v2" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.Filter.v1_0_0 = OpenLayers.Class(OpenLayers.Format.GML.v2, OpenLayers.Format.Filter.v1, { │ │ │ │ │ - VERSION: "1.0.0", │ │ │ │ │ - schemaLocation: "http://www.opengis.net/ogc/filter/1.0.0/filter.xsd", │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.GML.v2.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ - readers: { │ │ │ │ │ - ogc: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - PropertyIsEqualTo: function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.EQUAL_TO │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - obj.filters.push(filter) │ │ │ │ │ }, │ │ │ │ │ - PropertyIsNotEqualTo: function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.NOT_EQUAL_TO │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - obj.filters.push(filter) │ │ │ │ │ + resultModel: function(options) { │ │ │ │ │ + return this.createElementNSPlus("resultModel", { │ │ │ │ │ + value: options.resultModel │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - PropertyIsLike: function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.LIKE │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - var wildCard = node.getAttribute("wildCard"); │ │ │ │ │ - var singleChar = node.getAttribute("singleChar"); │ │ │ │ │ - var esc = node.getAttribute("escape"); │ │ │ │ │ - filter.value2regex(wildCard, singleChar, esc); │ │ │ │ │ - obj.filters.push(filter) │ │ │ │ │ + responseMode: function(options) { │ │ │ │ │ + return this.createElementNSPlus("responseMode", { │ │ │ │ │ + value: options.responseMode │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.Filter.v1.prototype.readers["ogc"]), │ │ │ │ │ - gml: OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ - feature: OpenLayers.Format.GML.v2.prototype.readers["feature"] │ │ │ │ │ - }, │ │ │ │ │ - writers: { │ │ │ │ │ - ogc: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - PropertyIsEqualTo: function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsEqualTo"); │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - this.writeOgcExpression(filter.value, node); │ │ │ │ │ + }, │ │ │ │ │ + ogc: { │ │ │ │ │ + TM_Equals: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:TM_Equals"); │ │ │ │ │ + this.writeNode("ogc:PropertyName", { │ │ │ │ │ + property: "urn:ogc:data:time:iso8601" │ │ │ │ │ + }, node); │ │ │ │ │ + if (options.eventTime === "latest") { │ │ │ │ │ + this.writeNode("gml:TimeInstant", { │ │ │ │ │ + value: "latest" │ │ │ │ │ + }, node) │ │ │ │ │ + } │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - PropertyIsNotEqualTo: function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsNotEqualTo"); │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - this.writeOgcExpression(filter.value, node); │ │ │ │ │ + PropertyName: function(options) { │ │ │ │ │ + return this.createElementNSPlus("ogc:PropertyName", { │ │ │ │ │ + value: options.property │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + gml: { │ │ │ │ │ + TimeInstant: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:TimeInstant"); │ │ │ │ │ + this.writeNode("gml:timePosition", options, node); │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - PropertyIsLike: function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsLike", { │ │ │ │ │ - attributes: { │ │ │ │ │ - wildCard: "*", │ │ │ │ │ - singleChar: ".", │ │ │ │ │ - escape: "!" │ │ │ │ │ - } │ │ │ │ │ + timePosition: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:timePosition", { │ │ │ │ │ + value: options.value │ │ │ │ │ }); │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - this.writeNode("Literal", filter.regex2value(), node); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - BBOX: function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:BBOX"); │ │ │ │ │ - filter.property && this.writeNode("PropertyName", filter, node); │ │ │ │ │ - var box = this.writeNode("gml:Box", filter.value, node); │ │ │ │ │ - if (filter.projection) { │ │ │ │ │ - box.setAttribute("srsName", filter.projection) │ │ │ │ │ - } │ │ │ │ │ return node │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.Filter.v1.prototype.writers["ogc"]), │ │ │ │ │ - gml: OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ - feature: OpenLayers.Format.GML.v2.prototype.writers["feature"] │ │ │ │ │ - }, │ │ │ │ │ - writeSpatial: function(filter, name) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:" + name); │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - if (filter.value instanceof OpenLayers.Filter.Function) { │ │ │ │ │ - this.writeNode("Function", filter.value, node) │ │ │ │ │ - } else { │ │ │ │ │ - var child; │ │ │ │ │ - if (filter.value instanceof OpenLayers.Geometry) { │ │ │ │ │ - child = this.writeNode("feature:_geometry", filter.value).firstChild │ │ │ │ │ - } else { │ │ │ │ │ - child = this.writeNode("gml:Box", filter.value) │ │ │ │ │ - } │ │ │ │ │ - if (filter.projection) { │ │ │ │ │ - child.setAttribute("srsName", filter.projection) │ │ │ │ │ - } │ │ │ │ │ - node.appendChild(child) │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.Filter.v1_0_0" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SOSGetObservation" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.CSWGetRecords.v2_0_2 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Format.XLS = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + defaultVersion: "1.1.0", │ │ │ │ │ + stringifyOutput: true, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.XLS" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.CSWGetDomain.v2_0_2 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ namespaces: { │ │ │ │ │ - csw: "http://www.opengis.net/cat/csw/2.0.2", │ │ │ │ │ - dc: "http://purl.org/dc/elements/1.1/", │ │ │ │ │ - dct: "http://purl.org/dc/terms/", │ │ │ │ │ - gmd: "http://www.isotc211.org/2005/gmd", │ │ │ │ │ - geonet: "http://www.fao.org/geonetwork", │ │ │ │ │ - ogc: "http://www.opengis.net/ogc", │ │ │ │ │ - ows: "http://www.opengis.net/ows", │ │ │ │ │ xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + csw: "http://www.opengis.net/cat/csw/2.0.2" │ │ │ │ │ }, │ │ │ │ │ defaultPrefix: "csw", │ │ │ │ │ version: "2.0.2", │ │ │ │ │ schemaLocation: "http://www.opengis.net/cat/csw/2.0.2 http://schemas.opengis.net/csw/2.0.2/CSW-discovery.xsd", │ │ │ │ │ - requestId: null, │ │ │ │ │ - resultType: null, │ │ │ │ │ - outputFormat: null, │ │ │ │ │ - outputSchema: null, │ │ │ │ │ - startPosition: null, │ │ │ │ │ - maxRecords: null, │ │ │ │ │ - DistributedSearch: null, │ │ │ │ │ - ResponseHandler: null, │ │ │ │ │ - Query: null, │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: /^\s*|\s*$/g, │ │ │ │ │ - removeSpace: /\s*/g, │ │ │ │ │ - splitSpace: /\s+/, │ │ │ │ │ - trimComma: /\s*,\s*/g │ │ │ │ │ - }, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ + PropertyName: null, │ │ │ │ │ + ParameterName: null, │ │ │ │ │ read: function(data) { │ │ │ │ │ if (typeof data == "string") { │ │ │ │ │ data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ } │ │ │ │ │ if (data && data.nodeType == 9) { │ │ │ │ │ data = data.documentElement │ │ │ │ │ } │ │ │ │ │ var obj = {}; │ │ │ │ │ this.readNode(data, obj); │ │ │ │ │ return obj │ │ │ │ │ }, │ │ │ │ │ readers: { │ │ │ │ │ csw: { │ │ │ │ │ - GetRecordsResponse: function(node, obj) { │ │ │ │ │ - obj.records = []; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - var version = this.getAttributeNS(node, "", "version"); │ │ │ │ │ - if (version != "") { │ │ │ │ │ - obj.version = version │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - RequestId: function(node, obj) { │ │ │ │ │ - obj.RequestId = this.getChildValue(node) │ │ │ │ │ + GetDomainResponse: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ }, │ │ │ │ │ - SearchStatus: function(node, obj) { │ │ │ │ │ - obj.SearchStatus = {}; │ │ │ │ │ - var timestamp = this.getAttributeNS(node, "", "timestamp"); │ │ │ │ │ - if (timestamp != "") { │ │ │ │ │ - obj.SearchStatus.timestamp = timestamp │ │ │ │ │ + DomainValues: function(node, obj) { │ │ │ │ │ + if (!OpenLayers.Util.isArray(obj.DomainValues)) { │ │ │ │ │ + obj.DomainValues = [] │ │ │ │ │ + } │ │ │ │ │ + var attrs = node.attributes; │ │ │ │ │ + var domainValue = {}; │ │ │ │ │ + for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ + domainValue[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ } │ │ │ │ │ + this.readChildNodes(node, domainValue); │ │ │ │ │ + obj.DomainValues.push(domainValue) │ │ │ │ │ }, │ │ │ │ │ - SearchResults: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ + PropertyName: function(node, obj) { │ │ │ │ │ + obj.PropertyName = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + ParameterName: function(node, obj) { │ │ │ │ │ + obj.ParameterName = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + ListOfValues: function(node, obj) { │ │ │ │ │ + if (!OpenLayers.Util.isArray(obj.ListOfValues)) { │ │ │ │ │ + obj.ListOfValues = [] │ │ │ │ │ + } │ │ │ │ │ + this.readChildNodes(node, obj.ListOfValues) │ │ │ │ │ + }, │ │ │ │ │ + Value: function(node, obj) { │ │ │ │ │ var attrs = node.attributes; │ │ │ │ │ - var SearchResults = {}; │ │ │ │ │ + var value = {}; │ │ │ │ │ for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - if (attrs[i].name == "numberOfRecordsMatched" || attrs[i].name == "numberOfRecordsReturned" || attrs[i].name == "nextRecord") { │ │ │ │ │ - SearchResults[attrs[i].name] = parseInt(attrs[i].nodeValue) │ │ │ │ │ - } else { │ │ │ │ │ - SearchResults[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ - } │ │ │ │ │ + value[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ } │ │ │ │ │ - obj.SearchResults = SearchResults │ │ │ │ │ + value.value = this.getChildValue(node); │ │ │ │ │ + obj.push({ │ │ │ │ │ + Value: value │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - SummaryRecord: function(node, obj) { │ │ │ │ │ - var record = { │ │ │ │ │ - type: "SummaryRecord" │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, record); │ │ │ │ │ - obj.records.push(record) │ │ │ │ │ + ConceptualScheme: function(node, obj) { │ │ │ │ │ + obj.ConceptualScheme = {}; │ │ │ │ │ + this.readChildNodes(node, obj.ConceptualScheme) │ │ │ │ │ }, │ │ │ │ │ - BriefRecord: function(node, obj) { │ │ │ │ │ - var record = { │ │ │ │ │ - type: "BriefRecord" │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, record); │ │ │ │ │ - obj.records.push(record) │ │ │ │ │ + Name: function(node, obj) { │ │ │ │ │ + obj.Name = this.getChildValue(node) │ │ │ │ │ }, │ │ │ │ │ - DCMIRecord: function(node, obj) { │ │ │ │ │ - var record = { │ │ │ │ │ - type: "DCMIRecord" │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, record); │ │ │ │ │ - obj.records.push(record) │ │ │ │ │ + Document: function(node, obj) { │ │ │ │ │ + obj.Document = this.getChildValue(node) │ │ │ │ │ }, │ │ │ │ │ - Record: function(node, obj) { │ │ │ │ │ - var record = { │ │ │ │ │ - type: "Record" │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, record); │ │ │ │ │ - obj.records.push(record) │ │ │ │ │ + Authority: function(node, obj) { │ │ │ │ │ + obj.Authority = this.getChildValue(node) │ │ │ │ │ }, │ │ │ │ │ - "*": function(node, obj) { │ │ │ │ │ - var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ - obj[name] = this.getChildValue(node) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - geonet: { │ │ │ │ │ - info: function(node, obj) { │ │ │ │ │ - var gninfo = {}; │ │ │ │ │ - this.readChildNodes(node, gninfo); │ │ │ │ │ - obj.gninfo = gninfo │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - dc: { │ │ │ │ │ - "*": function(node, obj) { │ │ │ │ │ - var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ - if (!OpenLayers.Util.isArray(obj[name])) { │ │ │ │ │ - obj[name] = [] │ │ │ │ │ - } │ │ │ │ │ - var dc_element = {}; │ │ │ │ │ + RangeOfValues: function(node, obj) { │ │ │ │ │ + obj.RangeOfValues = {}; │ │ │ │ │ + this.readChildNodes(node, obj.RangeOfValues) │ │ │ │ │ + }, │ │ │ │ │ + MinValue: function(node, obj) { │ │ │ │ │ var attrs = node.attributes; │ │ │ │ │ + var value = {}; │ │ │ │ │ for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - dc_element[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ - } │ │ │ │ │ - dc_element.value = this.getChildValue(node); │ │ │ │ │ - if (dc_element.value != "") { │ │ │ │ │ - obj[name].push(dc_element) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - dct: { │ │ │ │ │ - "*": function(node, obj) { │ │ │ │ │ - var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ - if (!OpenLayers.Util.isArray(obj[name])) { │ │ │ │ │ - obj[name] = [] │ │ │ │ │ + value[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ } │ │ │ │ │ - obj[name].push(this.getChildValue(node)) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - ows: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - BoundingBox: function(node, obj) { │ │ │ │ │ - if (obj.bounds) { │ │ │ │ │ - obj.BoundingBox = [{ │ │ │ │ │ - crs: obj.projection, │ │ │ │ │ - value: [obj.bounds.left, obj.bounds.bottom, obj.bounds.right, obj.bounds.top] │ │ │ │ │ - }]; │ │ │ │ │ - delete obj.projection; │ │ │ │ │ - delete obj.bounds │ │ │ │ │ + value.value = this.getChildValue(node); │ │ │ │ │ + obj.MinValue = value │ │ │ │ │ + }, │ │ │ │ │ + MaxValue: function(node, obj) { │ │ │ │ │ + var attrs = node.attributes; │ │ │ │ │ + var value = {}; │ │ │ │ │ + for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ + value[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Format.OWSCommon.v1_0_0.prototype.readers["ows"]["BoundingBox"].apply(this, arguments) │ │ │ │ │ + value.value = this.getChildValue(node); │ │ │ │ │ + obj.MaxValue = value │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.OWSCommon.v1_0_0.prototype.readers["ows"]) │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ write: function(options) { │ │ │ │ │ - var node = this.writeNode("csw:GetRecords", options); │ │ │ │ │ - node.setAttribute("xmlns:gmd", this.namespaces.gmd); │ │ │ │ │ + var node = this.writeNode("csw:GetDomain", options); │ │ │ │ │ return OpenLayers.Format.XML.prototype.write.apply(this, [node]) │ │ │ │ │ }, │ │ │ │ │ writers: { │ │ │ │ │ csw: { │ │ │ │ │ - GetRecords: function(options) { │ │ │ │ │ - if (!options) { │ │ │ │ │ - options = {} │ │ │ │ │ - } │ │ │ │ │ - var node = this.createElementNSPlus("csw:GetRecords", { │ │ │ │ │ + GetDomain: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:GetDomain", { │ │ │ │ │ attributes: { │ │ │ │ │ service: "CSW", │ │ │ │ │ - version: this.version, │ │ │ │ │ - requestId: options.requestId || this.requestId, │ │ │ │ │ - resultType: options.resultType || this.resultType, │ │ │ │ │ - outputFormat: options.outputFormat || this.outputFormat, │ │ │ │ │ - outputSchema: options.outputSchema || this.outputSchema, │ │ │ │ │ - startPosition: options.startPosition || this.startPosition, │ │ │ │ │ - maxRecords: options.maxRecords || this.maxRecords │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (options.DistributedSearch || this.DistributedSearch) { │ │ │ │ │ - this.writeNode("csw:DistributedSearch", options.DistributedSearch || this.DistributedSearch, node) │ │ │ │ │ - } │ │ │ │ │ - var ResponseHandler = options.ResponseHandler || this.ResponseHandler; │ │ │ │ │ - if (OpenLayers.Util.isArray(ResponseHandler) && ResponseHandler.length > 0) { │ │ │ │ │ - for (var i = 0, len = ResponseHandler.length; i < len; i++) { │ │ │ │ │ - this.writeNode("csw:ResponseHandler", ResponseHandler[i], node) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.writeNode("Query", options.Query || this.Query, node); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - DistributedSearch: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:DistributedSearch", { │ │ │ │ │ - attributes: { │ │ │ │ │ - hopCount: options.hopCount │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - ResponseHandler: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:ResponseHandler", { │ │ │ │ │ - value: options.value │ │ │ │ │ - }); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - Query: function(options) { │ │ │ │ │ - if (!options) { │ │ │ │ │ - options = {} │ │ │ │ │ - } │ │ │ │ │ - var node = this.createElementNSPlus("csw:Query", { │ │ │ │ │ - attributes: { │ │ │ │ │ - typeNames: options.typeNames || "csw:Record" │ │ │ │ │ + version: this.version │ │ │ │ │ } │ │ │ │ │ }); │ │ │ │ │ - var ElementName = options.ElementName; │ │ │ │ │ - if (OpenLayers.Util.isArray(ElementName) && ElementName.length > 0) { │ │ │ │ │ - for (var i = 0, len = ElementName.length; i < len; i++) { │ │ │ │ │ - this.writeNode("csw:ElementName", ElementName[i], node) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.writeNode("csw:ElementSetName", options.ElementSetName || { │ │ │ │ │ - value: "summary" │ │ │ │ │ - }, node) │ │ │ │ │ - } │ │ │ │ │ - if (options.Constraint) { │ │ │ │ │ - this.writeNode("csw:Constraint", options.Constraint, node) │ │ │ │ │ - } │ │ │ │ │ - if (options.SortBy) { │ │ │ │ │ - this.writeNode("ogc:SortBy", options.SortBy, node) │ │ │ │ │ + if (options.PropertyName || this.PropertyName) { │ │ │ │ │ + this.writeNode("csw:PropertyName", options.PropertyName || this.PropertyName, node) │ │ │ │ │ + } else if (options.ParameterName || this.ParameterName) { │ │ │ │ │ + this.writeNode("csw:ParameterName", options.ParameterName || this.ParameterName, node) │ │ │ │ │ } │ │ │ │ │ + this.readChildNodes(node, options); │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - ElementName: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:ElementName", { │ │ │ │ │ - value: options.value │ │ │ │ │ - }); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - ElementSetName: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:ElementSetName", { │ │ │ │ │ - attributes: { │ │ │ │ │ - typeNames: options.typeNames │ │ │ │ │ - }, │ │ │ │ │ - value: options.value │ │ │ │ │ + PropertyName: function(value) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:PropertyName", { │ │ │ │ │ + value: value │ │ │ │ │ }); │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - Constraint: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:Constraint", { │ │ │ │ │ - attributes: { │ │ │ │ │ - version: options.version │ │ │ │ │ - } │ │ │ │ │ + ParameterName: function(value) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:ParameterName", { │ │ │ │ │ + value: value │ │ │ │ │ }); │ │ │ │ │ - if (options.Filter) { │ │ │ │ │ - var format = new OpenLayers.Format.Filter({ │ │ │ │ │ - version: options.version │ │ │ │ │ - }); │ │ │ │ │ - node.appendChild(format.write(options.Filter)) │ │ │ │ │ - } else if (options.CqlText) { │ │ │ │ │ - var child = this.createElementNSPlus("CqlText", { │ │ │ │ │ - value: options.CqlText.value │ │ │ │ │ - }); │ │ │ │ │ - node.appendChild(child) │ │ │ │ │ - } │ │ │ │ │ return node │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - ogc: OpenLayers.Format.Filter.v1_1_0.prototype.writers["ogc"] │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.CSWGetRecords.v2_0_2" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.CSWGetDomain.v2_0_2" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Format.SLD.v1 = OpenLayers.Class(OpenLayers.Format.Filter.v1_0_0, { │ │ │ │ │ namespaces: { │ │ │ │ │ sld: "http://www.opengis.net/sld", │ │ │ │ │ ogc: "http://www.opengis.net/ogc", │ │ │ │ │ gml: "http://www.opengis.net/gml", │ │ │ │ │ xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ @@ -23439,14 +24327,1023 @@ │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.SLD.v1" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Format.SLD.v1_0_0 = OpenLayers.Class(OpenLayers.Format.SLD.v1, { │ │ │ │ │ VERSION: "1.0.0", │ │ │ │ │ schemaLocation: "http://www.opengis.net/sld http://schemas.opengis.net/sld/1.0.0/StyledLayerDescriptor.xsd", │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.SLD.v1_0_0" │ │ │ │ │ }); │ │ │ │ │ +OpenLayers.Format.SLD.v1_0_0_GeoServer = OpenLayers.Class(OpenLayers.Format.SLD.v1_0_0, { │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + profile: "GeoServer", │ │ │ │ │ + readers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + sld: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Priority: function(node, obj) { │ │ │ │ │ + var value = this.readers.ogc._expression.call(this, node); │ │ │ │ │ + if (value) { │ │ │ │ │ + obj.priority = value │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + VendorOption: function(node, obj) { │ │ │ │ │ + if (!obj.vendorOptions) { │ │ │ │ │ + obj.vendorOptions = {} │ │ │ │ │ + } │ │ │ │ │ + obj.vendorOptions[node.getAttribute("name")] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + TextSymbolizer: function(node, rule) { │ │ │ │ │ + OpenLayers.Format.SLD.v1_0_0.prototype.readers.sld.TextSymbolizer.apply(this, arguments); │ │ │ │ │ + var symbolizer = this.multipleSymbolizers ? rule.symbolizers[rule.symbolizers.length - 1] : rule.symbolizer["Text"]; │ │ │ │ │ + if (symbolizer.graphic === undefined) { │ │ │ │ │ + symbolizer.graphic = false │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.SLD.v1_0_0.prototype.readers["sld"]) │ │ │ │ │ + }, OpenLayers.Format.SLD.v1_0_0.prototype.readers), │ │ │ │ │ + writers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + sld: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Priority: function(priority) { │ │ │ │ │ + return this.writers.sld._OGCExpression.call(this, "sld:Priority", priority) │ │ │ │ │ + }, │ │ │ │ │ + VendorOption: function(option) { │ │ │ │ │ + return this.createElementNSPlus("sld:VendorOption", { │ │ │ │ │ + attributes: { │ │ │ │ │ + name: option.name │ │ │ │ │ + }, │ │ │ │ │ + value: option.value │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + TextSymbolizer: function(symbolizer) { │ │ │ │ │ + var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ + var node = writers["sld"]["TextSymbolizer"].apply(this, arguments); │ │ │ │ │ + if (symbolizer.graphic !== false && (symbolizer.externalGraphic || symbolizer.graphicName)) { │ │ │ │ │ + this.writeNode("Graphic", symbolizer, node) │ │ │ │ │ + } │ │ │ │ │ + if ("priority" in symbolizer) { │ │ │ │ │ + this.writeNode("Priority", symbolizer.priority, node) │ │ │ │ │ + } │ │ │ │ │ + return this.addVendorOptions(node, symbolizer) │ │ │ │ │ + }, │ │ │ │ │ + PointSymbolizer: function(symbolizer) { │ │ │ │ │ + var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ + var node = writers["sld"]["PointSymbolizer"].apply(this, arguments); │ │ │ │ │ + return this.addVendorOptions(node, symbolizer) │ │ │ │ │ + }, │ │ │ │ │ + LineSymbolizer: function(symbolizer) { │ │ │ │ │ + var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ + var node = writers["sld"]["LineSymbolizer"].apply(this, arguments); │ │ │ │ │ + return this.addVendorOptions(node, symbolizer) │ │ │ │ │ + }, │ │ │ │ │ + PolygonSymbolizer: function(symbolizer) { │ │ │ │ │ + var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ + var node = writers["sld"]["PolygonSymbolizer"].apply(this, arguments); │ │ │ │ │ + return this.addVendorOptions(node, symbolizer) │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.SLD.v1_0_0.prototype.writers["sld"]) │ │ │ │ │ + }, OpenLayers.Format.SLD.v1_0_0.prototype.writers), │ │ │ │ │ + addVendorOptions: function(node, symbolizer) { │ │ │ │ │ + var options = symbolizer.vendorOptions; │ │ │ │ │ + if (options) { │ │ │ │ │ + for (var key in symbolizer.vendorOptions) { │ │ │ │ │ + this.writeNode("VendorOption", { │ │ │ │ │ + name: key, │ │ │ │ │ + value: symbolizer.vendorOptions[key] │ │ │ │ │ + }, node) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SLD.v1_0_0_GeoServer" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.ArcXML.Features = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + read: function(data) { │ │ │ │ │ + var axl = new OpenLayers.Format.ArcXML; │ │ │ │ │ + var parsed = axl.read(data); │ │ │ │ │ + return parsed.features.feature │ │ │ │ │ + } │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WFSCapabilities.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + namespaces: { │ │ │ │ │ + wfs: "http://www.opengis.net/wfs", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + ows: "http://www.opengis.net/ows" │ │ │ │ │ + }, │ │ │ │ │ + errorProperty: "featureTypeList", │ │ │ │ │ + defaultPrefix: "wfs", │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + var raw = data; │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + return capabilities │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + wfs: { │ │ │ │ │ + WFS_Capabilities: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + FeatureTypeList: function(node, request) { │ │ │ │ │ + request.featureTypeList = { │ │ │ │ │ + featureTypes: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, request.featureTypeList) │ │ │ │ │ + }, │ │ │ │ │ + FeatureType: function(node, featureTypeList) { │ │ │ │ │ + var featureType = {}; │ │ │ │ │ + this.readChildNodes(node, featureType); │ │ │ │ │ + featureTypeList.featureTypes.push(featureType) │ │ │ │ │ + }, │ │ │ │ │ + Name: function(node, obj) { │ │ │ │ │ + var name = this.getChildValue(node); │ │ │ │ │ + if (name) { │ │ │ │ │ + var parts = name.split(":"); │ │ │ │ │ + obj.name = parts.pop(); │ │ │ │ │ + if (parts.length > 0) { │ │ │ │ │ + obj.featureNS = this.lookupNamespaceURI(node, parts[0]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Title: function(node, obj) { │ │ │ │ │ + var title = this.getChildValue(node); │ │ │ │ │ + if (title) { │ │ │ │ │ + obj.title = title │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Abstract: function(node, obj) { │ │ │ │ │ + var abst = this.getChildValue(node); │ │ │ │ │ + if (abst) { │ │ │ │ │ + obj["abstract"] = abst │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WFSCapabilities.v1_1_0 = OpenLayers.Class(OpenLayers.Format.WFSCapabilities.v1, { │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: /^\s*|\s*$/g, │ │ │ │ │ + removeSpace: /\s*/g, │ │ │ │ │ + splitSpace: /\s+/, │ │ │ │ │ + trimComma: /\s*,\s*/g │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + wfs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + DefaultSRS: function(node, obj) { │ │ │ │ │ + var defaultSRS = this.getChildValue(node); │ │ │ │ │ + if (defaultSRS) { │ │ │ │ │ + obj.srs = defaultSRS │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WFSCapabilities.v1.prototype.readers["wfs"]), │ │ │ │ │ + ows: OpenLayers.Format.OWSCommon.v1.prototype.readers.ows │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1_1_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WFSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.WFSCapabilities.v1, { │ │ │ │ │ + readers: { │ │ │ │ │ + wfs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Service: function(node, capabilities) { │ │ │ │ │ + capabilities.service = {}; │ │ │ │ │ + this.readChildNodes(node, capabilities.service) │ │ │ │ │ + }, │ │ │ │ │ + Fees: function(node, service) { │ │ │ │ │ + var fees = this.getChildValue(node); │ │ │ │ │ + if (fees && fees.toLowerCase() != "none") { │ │ │ │ │ + service.fees = fees │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + AccessConstraints: function(node, service) { │ │ │ │ │ + var constraints = this.getChildValue(node); │ │ │ │ │ + if (constraints && constraints.toLowerCase() != "none") { │ │ │ │ │ + service.accessConstraints = constraints │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + OnlineResource: function(node, service) { │ │ │ │ │ + var onlineResource = this.getChildValue(node); │ │ │ │ │ + if (onlineResource && onlineResource.toLowerCase() != "none") { │ │ │ │ │ + service.onlineResource = onlineResource │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Keywords: function(node, service) { │ │ │ │ │ + var keywords = this.getChildValue(node); │ │ │ │ │ + if (keywords && keywords.toLowerCase() != "none") { │ │ │ │ │ + service.keywords = keywords.split(", ") │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Capability: function(node, capabilities) { │ │ │ │ │ + capabilities.capability = {}; │ │ │ │ │ + this.readChildNodes(node, capabilities.capability) │ │ │ │ │ + }, │ │ │ │ │ + Request: function(node, obj) { │ │ │ │ │ + obj.request = {}; │ │ │ │ │ + this.readChildNodes(node, obj.request) │ │ │ │ │ + }, │ │ │ │ │ + GetFeature: function(node, request) { │ │ │ │ │ + request.getfeature = { │ │ │ │ │ + href: {}, │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, request.getfeature) │ │ │ │ │ + }, │ │ │ │ │ + ResultFormat: function(node, obj) { │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + var childNode; │ │ │ │ │ + for (var i = 0; i < children.length; i++) { │ │ │ │ │ + childNode = children[i]; │ │ │ │ │ + if (childNode.nodeType == 1) { │ │ │ │ │ + obj.formats.push(childNode.nodeName) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + DCPType: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + HTTP: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj.href) │ │ │ │ │ + }, │ │ │ │ │ + Get: function(node, obj) { │ │ │ │ │ + obj.get = node.getAttribute("onlineResource") │ │ │ │ │ + }, │ │ │ │ │ + Post: function(node, obj) { │ │ │ │ │ + obj.post = node.getAttribute("onlineResource") │ │ │ │ │ + }, │ │ │ │ │ + SRS: function(node, obj) { │ │ │ │ │ + var srs = this.getChildValue(node); │ │ │ │ │ + if (srs) { │ │ │ │ │ + obj.srs = srs │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WFSCapabilities.v1.prototype.readers["wfs"]) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1_0_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + namespaces: { │ │ │ │ │ + wms: "http://www.opengis.net/wms", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ + defaultPrefix: "wms", │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + var raw = data; │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + if (capabilities.service === undefined) { │ │ │ │ │ + var parser = new OpenLayers.Format.OGCExceptionReport; │ │ │ │ │ + capabilities.error = parser.read(raw) │ │ │ │ │ + } │ │ │ │ │ + return capabilities │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + wms: { │ │ │ │ │ + Service: function(node, obj) { │ │ │ │ │ + obj.service = {}; │ │ │ │ │ + this.readChildNodes(node, obj.service) │ │ │ │ │ + }, │ │ │ │ │ + Name: function(node, obj) { │ │ │ │ │ + obj.name = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Title: function(node, obj) { │ │ │ │ │ + obj.title = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Abstract: function(node, obj) { │ │ │ │ │ + obj["abstract"] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + BoundingBox: function(node, obj) { │ │ │ │ │ + var bbox = {}; │ │ │ │ │ + bbox.bbox = [parseFloat(node.getAttribute("minx")), parseFloat(node.getAttribute("miny")), parseFloat(node.getAttribute("maxx")), parseFloat(node.getAttribute("maxy"))]; │ │ │ │ │ + var res = { │ │ │ │ │ + x: parseFloat(node.getAttribute("resx")), │ │ │ │ │ + y: parseFloat(node.getAttribute("resy")) │ │ │ │ │ + }; │ │ │ │ │ + if (!(isNaN(res.x) && isNaN(res.y))) { │ │ │ │ │ + bbox.res = res │ │ │ │ │ + } │ │ │ │ │ + return bbox │ │ │ │ │ + }, │ │ │ │ │ + OnlineResource: function(node, obj) { │ │ │ │ │ + obj.href = this.getAttributeNS(node, this.namespaces.xlink, "href") │ │ │ │ │ + }, │ │ │ │ │ + ContactInformation: function(node, obj) { │ │ │ │ │ + obj.contactInformation = {}; │ │ │ │ │ + this.readChildNodes(node, obj.contactInformation) │ │ │ │ │ + }, │ │ │ │ │ + ContactPersonPrimary: function(node, obj) { │ │ │ │ │ + obj.personPrimary = {}; │ │ │ │ │ + this.readChildNodes(node, obj.personPrimary) │ │ │ │ │ + }, │ │ │ │ │ + ContactPerson: function(node, obj) { │ │ │ │ │ + obj.person = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + ContactOrganization: function(node, obj) { │ │ │ │ │ + obj.organization = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + ContactPosition: function(node, obj) { │ │ │ │ │ + obj.position = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + ContactAddress: function(node, obj) { │ │ │ │ │ + obj.contactAddress = {}; │ │ │ │ │ + this.readChildNodes(node, obj.contactAddress) │ │ │ │ │ + }, │ │ │ │ │ + AddressType: function(node, obj) { │ │ │ │ │ + obj.type = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Address: function(node, obj) { │ │ │ │ │ + obj.address = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + City: function(node, obj) { │ │ │ │ │ + obj.city = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + StateOrProvince: function(node, obj) { │ │ │ │ │ + obj.stateOrProvince = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + PostCode: function(node, obj) { │ │ │ │ │ + obj.postcode = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Country: function(node, obj) { │ │ │ │ │ + obj.country = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + ContactVoiceTelephone: function(node, obj) { │ │ │ │ │ + obj.phone = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + ContactFacsimileTelephone: function(node, obj) { │ │ │ │ │ + obj.fax = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + ContactElectronicMailAddress: function(node, obj) { │ │ │ │ │ + obj.email = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Fees: function(node, obj) { │ │ │ │ │ + var fees = this.getChildValue(node); │ │ │ │ │ + if (fees && fees.toLowerCase() != "none") { │ │ │ │ │ + obj.fees = fees │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + AccessConstraints: function(node, obj) { │ │ │ │ │ + var constraints = this.getChildValue(node); │ │ │ │ │ + if (constraints && constraints.toLowerCase() != "none") { │ │ │ │ │ + obj.accessConstraints = constraints │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Capability: function(node, obj) { │ │ │ │ │ + obj.capability = { │ │ │ │ │ + nestedLayers: [], │ │ │ │ │ + layers: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.capability) │ │ │ │ │ + }, │ │ │ │ │ + Request: function(node, obj) { │ │ │ │ │ + obj.request = {}; │ │ │ │ │ + this.readChildNodes(node, obj.request) │ │ │ │ │ + }, │ │ │ │ │ + GetCapabilities: function(node, obj) { │ │ │ │ │ + obj.getcapabilities = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getcapabilities) │ │ │ │ │ + }, │ │ │ │ │ + Format: function(node, obj) { │ │ │ │ │ + if (OpenLayers.Util.isArray(obj.formats)) { │ │ │ │ │ + obj.formats.push(this.getChildValue(node)) │ │ │ │ │ + } else { │ │ │ │ │ + obj.format = this.getChildValue(node) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + DCPType: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + HTTP: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + Get: function(node, obj) { │ │ │ │ │ + obj.get = {}; │ │ │ │ │ + this.readChildNodes(node, obj.get); │ │ │ │ │ + if (!obj.href) { │ │ │ │ │ + obj.href = obj.get.href │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Post: function(node, obj) { │ │ │ │ │ + obj.post = {}; │ │ │ │ │ + this.readChildNodes(node, obj.post); │ │ │ │ │ + if (!obj.href) { │ │ │ │ │ + obj.href = obj.get.href │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + GetMap: function(node, obj) { │ │ │ │ │ + obj.getmap = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getmap) │ │ │ │ │ + }, │ │ │ │ │ + GetFeatureInfo: function(node, obj) { │ │ │ │ │ + obj.getfeatureinfo = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getfeatureinfo) │ │ │ │ │ + }, │ │ │ │ │ + Exception: function(node, obj) { │ │ │ │ │ + obj.exception = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.exception) │ │ │ │ │ + }, │ │ │ │ │ + Layer: function(node, obj) { │ │ │ │ │ + var parentLayer, capability; │ │ │ │ │ + if (obj.capability) { │ │ │ │ │ + capability = obj.capability; │ │ │ │ │ + parentLayer = obj │ │ │ │ │ + } else { │ │ │ │ │ + capability = obj │ │ │ │ │ + } │ │ │ │ │ + var attrNode = node.getAttributeNode("queryable"); │ │ │ │ │ + var queryable = attrNode && attrNode.specified ? node.getAttribute("queryable") : null; │ │ │ │ │ + attrNode = node.getAttributeNode("cascaded"); │ │ │ │ │ + var cascaded = attrNode && attrNode.specified ? node.getAttribute("cascaded") : null; │ │ │ │ │ + attrNode = node.getAttributeNode("opaque"); │ │ │ │ │ + var opaque = attrNode && attrNode.specified ? node.getAttribute("opaque") : null; │ │ │ │ │ + var noSubsets = node.getAttribute("noSubsets"); │ │ │ │ │ + var fixedWidth = node.getAttribute("fixedWidth"); │ │ │ │ │ + var fixedHeight = node.getAttribute("fixedHeight"); │ │ │ │ │ + var parent = parentLayer || {}, │ │ │ │ │ + extend = OpenLayers.Util.extend; │ │ │ │ │ + var layer = { │ │ │ │ │ + nestedLayers: [], │ │ │ │ │ + styles: parentLayer ? [].concat(parentLayer.styles) : [], │ │ │ │ │ + srs: parentLayer ? extend({}, parent.srs) : {}, │ │ │ │ │ + metadataURLs: [], │ │ │ │ │ + bbox: parentLayer ? extend({}, parent.bbox) : {}, │ │ │ │ │ + llbbox: parent.llbbox, │ │ │ │ │ + dimensions: parentLayer ? extend({}, parent.dimensions) : {}, │ │ │ │ │ + authorityURLs: parentLayer ? extend({}, parent.authorityURLs) : {}, │ │ │ │ │ + identifiers: {}, │ │ │ │ │ + keywords: [], │ │ │ │ │ + queryable: queryable && queryable !== "" ? queryable === "1" || queryable === "true" : parent.queryable || false, │ │ │ │ │ + cascaded: cascaded !== null ? parseInt(cascaded) : parent.cascaded || 0, │ │ │ │ │ + opaque: opaque ? opaque === "1" || opaque === "true" : parent.opaque || false, │ │ │ │ │ + noSubsets: noSubsets !== null ? noSubsets === "1" || noSubsets === "true" : parent.noSubsets || false, │ │ │ │ │ + fixedWidth: fixedWidth != null ? parseInt(fixedWidth) : parent.fixedWidth || 0, │ │ │ │ │ + fixedHeight: fixedHeight != null ? parseInt(fixedHeight) : parent.fixedHeight || 0, │ │ │ │ │ + minScale: parent.minScale, │ │ │ │ │ + maxScale: parent.maxScale, │ │ │ │ │ + attribution: parent.attribution │ │ │ │ │ + }; │ │ │ │ │ + obj.nestedLayers.push(layer); │ │ │ │ │ + layer.capability = capability; │ │ │ │ │ + this.readChildNodes(node, layer); │ │ │ │ │ + delete layer.capability; │ │ │ │ │ + if (layer.name) { │ │ │ │ │ + var parts = layer.name.split(":"), │ │ │ │ │ + request = capability.request, │ │ │ │ │ + gfi = request.getfeatureinfo; │ │ │ │ │ + if (parts.length > 0) { │ │ │ │ │ + layer.prefix = parts[0] │ │ │ │ │ + } │ │ │ │ │ + capability.layers.push(layer); │ │ │ │ │ + if (layer.formats === undefined) { │ │ │ │ │ + layer.formats = request.getmap.formats │ │ │ │ │ + } │ │ │ │ │ + if (layer.infoFormats === undefined && gfi) { │ │ │ │ │ + layer.infoFormats = gfi.formats │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Attribution: function(node, obj) { │ │ │ │ │ + obj.attribution = {}; │ │ │ │ │ + this.readChildNodes(node, obj.attribution) │ │ │ │ │ + }, │ │ │ │ │ + LogoURL: function(node, obj) { │ │ │ │ │ + obj.logo = { │ │ │ │ │ + width: node.getAttribute("width"), │ │ │ │ │ + height: node.getAttribute("height") │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.logo) │ │ │ │ │ + }, │ │ │ │ │ + Style: function(node, obj) { │ │ │ │ │ + var style = {}; │ │ │ │ │ + obj.styles.push(style); │ │ │ │ │ + this.readChildNodes(node, style) │ │ │ │ │ + }, │ │ │ │ │ + LegendURL: function(node, obj) { │ │ │ │ │ + var legend = { │ │ │ │ │ + width: node.getAttribute("width"), │ │ │ │ │ + height: node.getAttribute("height") │ │ │ │ │ + }; │ │ │ │ │ + obj.legend = legend; │ │ │ │ │ + this.readChildNodes(node, legend) │ │ │ │ │ + }, │ │ │ │ │ + MetadataURL: function(node, obj) { │ │ │ │ │ + var metadataURL = { │ │ │ │ │ + type: node.getAttribute("type") │ │ │ │ │ + }; │ │ │ │ │ + obj.metadataURLs.push(metadataURL); │ │ │ │ │ + this.readChildNodes(node, metadataURL) │ │ │ │ │ + }, │ │ │ │ │ + DataURL: function(node, obj) { │ │ │ │ │ + obj.dataURL = {}; │ │ │ │ │ + this.readChildNodes(node, obj.dataURL) │ │ │ │ │ + }, │ │ │ │ │ + FeatureListURL: function(node, obj) { │ │ │ │ │ + obj.featureListURL = {}; │ │ │ │ │ + this.readChildNodes(node, obj.featureListURL) │ │ │ │ │ + }, │ │ │ │ │ + AuthorityURL: function(node, obj) { │ │ │ │ │ + var name = node.getAttribute("name"); │ │ │ │ │ + var authority = {}; │ │ │ │ │ + this.readChildNodes(node, authority); │ │ │ │ │ + obj.authorityURLs[name] = authority.href │ │ │ │ │ + }, │ │ │ │ │ + Identifier: function(node, obj) { │ │ │ │ │ + var authority = node.getAttribute("authority"); │ │ │ │ │ + obj.identifiers[authority] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + KeywordList: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + SRS: function(node, obj) { │ │ │ │ │ + obj.srs[this.getChildValue(node)] = true │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_1 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1, { │ │ │ │ │ + readers: { │ │ │ │ │ + wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + WMT_MS_Capabilities: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + Keyword: function(node, obj) { │ │ │ │ │ + if (obj.keywords) { │ │ │ │ │ + obj.keywords.push(this.getChildValue(node)) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + DescribeLayer: function(node, obj) { │ │ │ │ │ + obj.describelayer = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.describelayer) │ │ │ │ │ + }, │ │ │ │ │ + GetLegendGraphic: function(node, obj) { │ │ │ │ │ + obj.getlegendgraphic = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getlegendgraphic) │ │ │ │ │ + }, │ │ │ │ │ + GetStyles: function(node, obj) { │ │ │ │ │ + obj.getstyles = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getstyles) │ │ │ │ │ + }, │ │ │ │ │ + PutStyles: function(node, obj) { │ │ │ │ │ + obj.putstyles = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.putstyles) │ │ │ │ │ + }, │ │ │ │ │ + UserDefinedSymbolization: function(node, obj) { │ │ │ │ │ + var userSymbols = { │ │ │ │ │ + supportSLD: parseInt(node.getAttribute("SupportSLD")) == 1, │ │ │ │ │ + userLayer: parseInt(node.getAttribute("UserLayer")) == 1, │ │ │ │ │ + userStyle: parseInt(node.getAttribute("UserStyle")) == 1, │ │ │ │ │ + remoteWFS: parseInt(node.getAttribute("RemoteWFS")) == 1 │ │ │ │ │ + }; │ │ │ │ │ + obj.userSymbols = userSymbols │ │ │ │ │ + }, │ │ │ │ │ + LatLonBoundingBox: function(node, obj) { │ │ │ │ │ + obj.llbbox = [parseFloat(node.getAttribute("minx")), parseFloat(node.getAttribute("miny")), parseFloat(node.getAttribute("maxx")), parseFloat(node.getAttribute("maxy"))] │ │ │ │ │ + }, │ │ │ │ │ + BoundingBox: function(node, obj) { │ │ │ │ │ + var bbox = OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"].BoundingBox.apply(this, [node, obj]); │ │ │ │ │ + bbox.srs = node.getAttribute("SRS"); │ │ │ │ │ + obj.bbox[bbox.srs] = bbox │ │ │ │ │ + }, │ │ │ │ │ + ScaleHint: function(node, obj) { │ │ │ │ │ + var min = node.getAttribute("min"); │ │ │ │ │ + var max = node.getAttribute("max"); │ │ │ │ │ + var rad2 = Math.pow(2, .5); │ │ │ │ │ + var ipm = OpenLayers.INCHES_PER_UNIT["m"]; │ │ │ │ │ + if (min != 0) { │ │ │ │ │ + obj.maxScale = parseFloat((min / rad2 * ipm * OpenLayers.DOTS_PER_INCH).toPrecision(13)) │ │ │ │ │ + } │ │ │ │ │ + if (max != Number.POSITIVE_INFINITY) { │ │ │ │ │ + obj.minScale = parseFloat((max / rad2 * ipm * OpenLayers.DOTS_PER_INCH).toPrecision(13)) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Dimension: function(node, obj) { │ │ │ │ │ + var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ + var dim = { │ │ │ │ │ + name: name, │ │ │ │ │ + units: node.getAttribute("units"), │ │ │ │ │ + unitsymbol: node.getAttribute("unitSymbol") │ │ │ │ │ + }; │ │ │ │ │ + obj.dimensions[dim.name] = dim │ │ │ │ │ + }, │ │ │ │ │ + Extent: function(node, obj) { │ │ │ │ │ + var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ + if (name in obj["dimensions"]) { │ │ │ │ │ + var extent = obj.dimensions[name]; │ │ │ │ │ + extent.nearestVal = node.getAttribute("nearestValue") === "1"; │ │ │ │ │ + extent.multipleVal = node.getAttribute("multipleValues") === "1"; │ │ │ │ │ + extent.current = node.getAttribute("current") === "1"; │ │ │ │ │ + extent["default"] = node.getAttribute("default") || ""; │ │ │ │ │ + var values = this.getChildValue(node); │ │ │ │ │ + extent.values = values.split(",") │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"]) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_1_0 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1_1, { │ │ │ │ │ + version: "1.1.0", │ │ │ │ │ + readers: { │ │ │ │ │ + wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + SRS: function(node, obj) { │ │ │ │ │ + var srs = this.getChildValue(node); │ │ │ │ │ + var values = srs.split(/ +/); │ │ │ │ │ + for (var i = 0, len = values.length; i < len; i++) { │ │ │ │ │ + obj.srs[values[i]] = true │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1_1.prototype.readers["wms"]) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_1_1 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1_1, { │ │ │ │ │ + version: "1.1.1", │ │ │ │ │ + readers: { │ │ │ │ │ + wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + SRS: function(node, obj) { │ │ │ │ │ + obj.srs[this.getChildValue(node)] = true │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1_1.prototype.readers["wms"]) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_1" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_1_1_WMSC = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1_1_1, { │ │ │ │ │ + version: "1.1.1", │ │ │ │ │ + profile: "WMSC", │ │ │ │ │ + readers: { │ │ │ │ │ + wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + VendorSpecificCapabilities: function(node, obj) { │ │ │ │ │ + obj.vendorSpecific = { │ │ │ │ │ + tileSets: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.vendorSpecific) │ │ │ │ │ + }, │ │ │ │ │ + TileSet: function(node, vendorSpecific) { │ │ │ │ │ + var tileset = { │ │ │ │ │ + srs: {}, │ │ │ │ │ + bbox: {}, │ │ │ │ │ + resolutions: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, tileset); │ │ │ │ │ + vendorSpecific.tileSets.push(tileset) │ │ │ │ │ + }, │ │ │ │ │ + Resolutions: function(node, tileset) { │ │ │ │ │ + var res = this.getChildValue(node).split(" "); │ │ │ │ │ + for (var i = 0, len = res.length; i < len; i++) { │ │ │ │ │ + if (res[i] != "") { │ │ │ │ │ + tileset.resolutions.push(parseFloat(res[i])) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Width: function(node, tileset) { │ │ │ │ │ + tileset.width = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + Height: function(node, tileset) { │ │ │ │ │ + tileset.height = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + Layers: function(node, tileset) { │ │ │ │ │ + tileset.layers = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Styles: function(node, tileset) { │ │ │ │ │ + tileset.styles = this.getChildValue(node) │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1_1_1.prototype.readers["wms"]) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_1_WMSC" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_3 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1, { │ │ │ │ │ + readers: { │ │ │ │ │ + wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + WMS_Capabilities: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + LayerLimit: function(node, obj) { │ │ │ │ │ + obj.layerLimit = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + MaxWidth: function(node, obj) { │ │ │ │ │ + obj.maxWidth = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + MaxHeight: function(node, obj) { │ │ │ │ │ + obj.maxHeight = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + BoundingBox: function(node, obj) { │ │ │ │ │ + var bbox = OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"].BoundingBox.apply(this, [node, obj]); │ │ │ │ │ + bbox.srs = node.getAttribute("CRS"); │ │ │ │ │ + obj.bbox[bbox.srs] = bbox │ │ │ │ │ + }, │ │ │ │ │ + CRS: function(node, obj) { │ │ │ │ │ + this.readers.wms.SRS.apply(this, [node, obj]) │ │ │ │ │ + }, │ │ │ │ │ + EX_GeographicBoundingBox: function(node, obj) { │ │ │ │ │ + obj.llbbox = []; │ │ │ │ │ + this.readChildNodes(node, obj.llbbox) │ │ │ │ │ + }, │ │ │ │ │ + westBoundLongitude: function(node, obj) { │ │ │ │ │ + obj[0] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + eastBoundLongitude: function(node, obj) { │ │ │ │ │ + obj[2] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + southBoundLatitude: function(node, obj) { │ │ │ │ │ + obj[1] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + northBoundLatitude: function(node, obj) { │ │ │ │ │ + obj[3] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + MinScaleDenominator: function(node, obj) { │ │ │ │ │ + obj.maxScale = parseFloat(this.getChildValue(node)).toPrecision(16) │ │ │ │ │ + }, │ │ │ │ │ + MaxScaleDenominator: function(node, obj) { │ │ │ │ │ + obj.minScale = parseFloat(this.getChildValue(node)).toPrecision(16) │ │ │ │ │ + }, │ │ │ │ │ + Dimension: function(node, obj) { │ │ │ │ │ + var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ + var dim = { │ │ │ │ │ + name: name, │ │ │ │ │ + units: node.getAttribute("units"), │ │ │ │ │ + unitsymbol: node.getAttribute("unitSymbol"), │ │ │ │ │ + nearestVal: node.getAttribute("nearestValue") === "1", │ │ │ │ │ + multipleVal: node.getAttribute("multipleValues") === "1", │ │ │ │ │ + default: node.getAttribute("default") || "", │ │ │ │ │ + current: node.getAttribute("current") === "1", │ │ │ │ │ + values: this.getChildValue(node).split(",") │ │ │ │ │ + }; │ │ │ │ │ + obj.dimensions[dim.name] = dim │ │ │ │ │ + }, │ │ │ │ │ + Keyword: function(node, obj) { │ │ │ │ │ + var keyword = { │ │ │ │ │ + value: this.getChildValue(node), │ │ │ │ │ + vocabulary: node.getAttribute("vocabulary") │ │ │ │ │ + }; │ │ │ │ │ + if (obj.keywords) { │ │ │ │ │ + obj.keywords.push(keyword) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"]), │ │ │ │ │ + sld: { │ │ │ │ │ + UserDefinedSymbolization: function(node, obj) { │ │ │ │ │ + this.readers.wms.UserDefinedSymbolization.apply(this, [node, obj]); │ │ │ │ │ + obj.userSymbols.inlineFeature = parseInt(node.getAttribute("InlineFeature")) == 1; │ │ │ │ │ + obj.userSymbols.remoteWCS = parseInt(node.getAttribute("RemoteWCS")) == 1 │ │ │ │ │ + }, │ │ │ │ │ + DescribeLayer: function(node, obj) { │ │ │ │ │ + this.readers.wms.DescribeLayer.apply(this, [node, obj]) │ │ │ │ │ + }, │ │ │ │ │ + GetLegendGraphic: function(node, obj) { │ │ │ │ │ + this.readers.wms.GetLegendGraphic.apply(this, [node, obj]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_3" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_3_0 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1_3, { │ │ │ │ │ + version: "1.3.0", │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_3_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMTSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.OWSCommon.v1_1_0, { │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + namespaces: { │ │ │ │ │ + ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ + wmts: "http://www.opengis.net/wmts/1.0", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ + }, │ │ │ │ │ + yx: null, │ │ │ │ │ + defaultPrefix: "wmts", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.options = options; │ │ │ │ │ + var yx = OpenLayers.Util.extend({}, OpenLayers.Format.WMTSCapabilities.prototype.yx); │ │ │ │ │ + this.yx = OpenLayers.Util.extend(yx, this.yx) │ │ │ │ │ + }, │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + capabilities.version = this.version; │ │ │ │ │ + return capabilities │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + wmts: { │ │ │ │ │ + Capabilities: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + Contents: function(node, obj) { │ │ │ │ │ + obj.contents = {}; │ │ │ │ │ + obj.contents.layers = []; │ │ │ │ │ + obj.contents.tileMatrixSets = {}; │ │ │ │ │ + this.readChildNodes(node, obj.contents) │ │ │ │ │ + }, │ │ │ │ │ + Layer: function(node, obj) { │ │ │ │ │ + var layer = { │ │ │ │ │ + styles: [], │ │ │ │ │ + formats: [], │ │ │ │ │ + dimensions: [], │ │ │ │ │ + tileMatrixSetLinks: [] │ │ │ │ │ + }; │ │ │ │ │ + layer.layers = []; │ │ │ │ │ + this.readChildNodes(node, layer); │ │ │ │ │ + obj.layers.push(layer) │ │ │ │ │ + }, │ │ │ │ │ + Style: function(node, obj) { │ │ │ │ │ + var style = {}; │ │ │ │ │ + style.isDefault = node.getAttribute("isDefault") === "true"; │ │ │ │ │ + this.readChildNodes(node, style); │ │ │ │ │ + obj.styles.push(style) │ │ │ │ │ + }, │ │ │ │ │ + Format: function(node, obj) { │ │ │ │ │ + obj.formats.push(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + TileMatrixSetLink: function(node, obj) { │ │ │ │ │ + var tileMatrixSetLink = {}; │ │ │ │ │ + this.readChildNodes(node, tileMatrixSetLink); │ │ │ │ │ + obj.tileMatrixSetLinks.push(tileMatrixSetLink) │ │ │ │ │ + }, │ │ │ │ │ + TileMatrixSet: function(node, obj) { │ │ │ │ │ + if (obj.layers) { │ │ │ │ │ + var tileMatrixSet = { │ │ │ │ │ + matrixIds: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, tileMatrixSet); │ │ │ │ │ + obj.tileMatrixSets[tileMatrixSet.identifier] = tileMatrixSet │ │ │ │ │ + } else { │ │ │ │ │ + obj.tileMatrixSet = this.getChildValue(node) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + TileMatrix: function(node, obj) { │ │ │ │ │ + var tileMatrix = { │ │ │ │ │ + supportedCRS: obj.supportedCRS │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, tileMatrix); │ │ │ │ │ + obj.matrixIds.push(tileMatrix) │ │ │ │ │ + }, │ │ │ │ │ + ScaleDenominator: function(node, obj) { │ │ │ │ │ + obj.scaleDenominator = parseFloat(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + TopLeftCorner: function(node, obj) { │ │ │ │ │ + var topLeftCorner = this.getChildValue(node); │ │ │ │ │ + var coords = topLeftCorner.split(" "); │ │ │ │ │ + var yx; │ │ │ │ │ + if (obj.supportedCRS) { │ │ │ │ │ + var crs = obj.supportedCRS.replace(/urn:ogc:def:crs:(\w+):.+:(\w+)$/, "urn:ogc:def:crs:$1::$2"); │ │ │ │ │ + yx = !!this.yx[crs] │ │ │ │ │ + } │ │ │ │ │ + if (yx) { │ │ │ │ │ + obj.topLeftCorner = new OpenLayers.LonLat(coords[1], coords[0]) │ │ │ │ │ + } else { │ │ │ │ │ + obj.topLeftCorner = new OpenLayers.LonLat(coords[0], coords[1]) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + TileWidth: function(node, obj) { │ │ │ │ │ + obj.tileWidth = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + TileHeight: function(node, obj) { │ │ │ │ │ + obj.tileHeight = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + MatrixWidth: function(node, obj) { │ │ │ │ │ + obj.matrixWidth = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + MatrixHeight: function(node, obj) { │ │ │ │ │ + obj.matrixHeight = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + ResourceURL: function(node, obj) { │ │ │ │ │ + obj.resourceUrl = obj.resourceUrl || {}; │ │ │ │ │ + var resourceType = node.getAttribute("resourceType"); │ │ │ │ │ + if (!obj.resourceUrls) { │ │ │ │ │ + obj.resourceUrls = [] │ │ │ │ │ + } │ │ │ │ │ + var resourceUrl = obj.resourceUrl[resourceType] = { │ │ │ │ │ + format: node.getAttribute("format"), │ │ │ │ │ + template: node.getAttribute("template"), │ │ │ │ │ + resourceType: resourceType │ │ │ │ │ + }; │ │ │ │ │ + obj.resourceUrls.push(resourceUrl) │ │ │ │ │ + }, │ │ │ │ │ + WSDL: function(node, obj) { │ │ │ │ │ + obj.wsdl = {}; │ │ │ │ │ + obj.wsdl.href = node.getAttribute("xlink:href") │ │ │ │ │ + }, │ │ │ │ │ + ServiceMetadataURL: function(node, obj) { │ │ │ │ │ + obj.serviceMetadataUrl = {}; │ │ │ │ │ + obj.serviceMetadataUrl.href = node.getAttribute("xlink:href") │ │ │ │ │ + }, │ │ │ │ │ + LegendURL: function(node, obj) { │ │ │ │ │ + obj.legend = {}; │ │ │ │ │ + obj.legend.href = node.getAttribute("xlink:href"); │ │ │ │ │ + obj.legend.format = node.getAttribute("format") │ │ │ │ │ + }, │ │ │ │ │ + Dimension: function(node, obj) { │ │ │ │ │ + var dimension = { │ │ │ │ │ + values: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, dimension); │ │ │ │ │ + obj.dimensions.push(dimension) │ │ │ │ │ + }, │ │ │ │ │ + Default: function(node, obj) { │ │ │ │ │ + obj["default"] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Value: function(node, obj) { │ │ │ │ │ + obj.values.push(this.getChildValue(node)) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + ows: OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMTSCapabilities.v1_0_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WPSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + namespaces: { │ │ │ │ │ + ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ + wps: "http://www.opengis.net/wps/1.0.0", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ + }, │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: /^\s*|\s*$/g, │ │ │ │ │ + removeSpace: /\s*/g, │ │ │ │ │ + splitSpace: /\s+/, │ │ │ │ │ + trimComma: /\s*,\s*/g │ │ │ │ │ + }, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + return capabilities │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + wps: { │ │ │ │ │ + Capabilities: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + ProcessOfferings: function(node, obj) { │ │ │ │ │ + obj.processOfferings = {}; │ │ │ │ │ + this.readChildNodes(node, obj.processOfferings) │ │ │ │ │ + }, │ │ │ │ │ + Process: function(node, processOfferings) { │ │ │ │ │ + var processVersion = this.getAttributeNS(node, this.namespaces.wps, "processVersion"); │ │ │ │ │ + var process = { │ │ │ │ │ + processVersion: processVersion │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, process); │ │ │ │ │ + processOfferings[process.identifier] = process │ │ │ │ │ + }, │ │ │ │ │ + Languages: function(node, obj) { │ │ │ │ │ + obj.languages = []; │ │ │ │ │ + this.readChildNodes(node, obj.languages) │ │ │ │ │ + }, │ │ │ │ │ + Default: function(node, languages) { │ │ │ │ │ + var language = { │ │ │ │ │ + isDefault: true │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, language); │ │ │ │ │ + languages.push(language) │ │ │ │ │ + }, │ │ │ │ │ + Supported: function(node, languages) { │ │ │ │ │ + var language = {}; │ │ │ │ │ + this.readChildNodes(node, language); │ │ │ │ │ + languages.push(language) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + ows: OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WPSCapabilities.v1_0_0" │ │ │ │ │ +}); │ │ │ │ │ OpenLayers.Format.OWSContext.v0_3_1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ namespaces: { │ │ │ │ │ owc: "http://www.opengis.net/ows-context", │ │ │ │ │ gml: "http://www.opengis.net/gml", │ │ │ │ │ kml: "http://www.opengis.net/kml/2.2", │ │ │ │ │ ogc: "http://www.opengis.net/ogc", │ │ │ │ │ ows: "http://www.opengis.net/ows", │ │ │ │ │ @@ -23849,14 +25746,264 @@ │ │ │ │ │ }, OpenLayers.Format.GML.v2.prototype.writers.gml), │ │ │ │ │ ows: OpenLayers.Format.OWSCommon.v1_0_0.prototype.writers.ows, │ │ │ │ │ sld: OpenLayers.Format.SLD.v1_0_0.prototype.writers.sld, │ │ │ │ │ feature: OpenLayers.Format.GML.v2.prototype.writers.feature │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.OWSContext.v0_3_1" │ │ │ │ │ }); │ │ │ │ │ +OpenLayers.Format.WMSDescribeLayer.v1_1_1 = OpenLayers.Class(OpenLayers.Format.WMSDescribeLayer, { │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.WMSDescribeLayer.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + var root = data.documentElement; │ │ │ │ │ + var children = root.childNodes; │ │ │ │ │ + var describelayer = { │ │ │ │ │ + layerDescriptions: [] │ │ │ │ │ + }; │ │ │ │ │ + var childNode, nodeName; │ │ │ │ │ + for (var i = 0; i < children.length; ++i) { │ │ │ │ │ + childNode = children[i]; │ │ │ │ │ + nodeName = childNode.nodeName; │ │ │ │ │ + if (nodeName == "LayerDescription") { │ │ │ │ │ + var layerName = childNode.getAttribute("name"); │ │ │ │ │ + var owsType = ""; │ │ │ │ │ + var owsURL = ""; │ │ │ │ │ + var typeName = ""; │ │ │ │ │ + if (childNode.getAttribute("owsType")) { │ │ │ │ │ + owsType = childNode.getAttribute("owsType"); │ │ │ │ │ + owsURL = childNode.getAttribute("owsURL") │ │ │ │ │ + } else { │ │ │ │ │ + if (childNode.getAttribute("wfs") != "") { │ │ │ │ │ + owsType = "WFS"; │ │ │ │ │ + owsURL = childNode.getAttribute("wfs") │ │ │ │ │ + } else if (childNode.getAttribute("wcs") != "") { │ │ │ │ │ + owsType = "WCS"; │ │ │ │ │ + owsURL = childNode.getAttribute("wcs") │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var query = childNode.getElementsByTagName("Query"); │ │ │ │ │ + if (query.length > 0) { │ │ │ │ │ + typeName = query[0].getAttribute("typeName"); │ │ │ │ │ + if (!typeName) { │ │ │ │ │ + typeName = query[0].getAttribute("typename") │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var layerDescription = { │ │ │ │ │ + layerName: layerName, │ │ │ │ │ + owsType: owsType, │ │ │ │ │ + owsURL: owsURL, │ │ │ │ │ + typeName: typeName │ │ │ │ │ + }; │ │ │ │ │ + describelayer.layerDescriptions.push(layerDescription); │ │ │ │ │ + describelayer.length = describelayer.layerDescriptions.length; │ │ │ │ │ + describelayer[describelayer.length - 1] = layerDescription │ │ │ │ │ + } else if (nodeName == "ServiceException") { │ │ │ │ │ + var parser = new OpenLayers.Format.OGCExceptionReport; │ │ │ │ │ + return { │ │ │ │ │ + error: parser.read(data) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return describelayer │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSDescribeLayer.v1_1_1" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSDescribeLayer.v1_1_0 = OpenLayers.Format.WMSDescribeLayer.v1_1_1; │ │ │ │ │ +OpenLayers.Format.WCSCapabilities.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: /^\s*|\s*$/g, │ │ │ │ │ + splitSpace: /\s+/ │ │ │ │ │ + }, │ │ │ │ │ + defaultPrefix: "wcs", │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + var raw = data; │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + return capabilities │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WCSCapabilities.v1_1_0 = OpenLayers.Class(OpenLayers.Format.WCSCapabilities.v1, { │ │ │ │ │ + namespaces: { │ │ │ │ │ + wcs: "http://www.opengis.net/wcs/1.1", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + ows: "http://www.opengis.net/ows/1.1" │ │ │ │ │ + }, │ │ │ │ │ + errorProperty: "operationsMetadata", │ │ │ │ │ + readers: { │ │ │ │ │ + wcs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Capabilities: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + Contents: function(node, request) { │ │ │ │ │ + request.contentMetadata = []; │ │ │ │ │ + this.readChildNodes(node, request.contentMetadata) │ │ │ │ │ + }, │ │ │ │ │ + CoverageSummary: function(node, contentMetadata) { │ │ │ │ │ + var coverageSummary = {}; │ │ │ │ │ + this.readChildNodes(node, coverageSummary); │ │ │ │ │ + contentMetadata.push(coverageSummary) │ │ │ │ │ + }, │ │ │ │ │ + Identifier: function(node, coverageSummary) { │ │ │ │ │ + coverageSummary.identifier = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Title: function(node, coverageSummary) { │ │ │ │ │ + coverageSummary.title = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Abstract: function(node, coverageSummary) { │ │ │ │ │ + coverageSummary["abstract"] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + SupportedCRS: function(node, coverageSummary) { │ │ │ │ │ + var crs = this.getChildValue(node); │ │ │ │ │ + if (crs) { │ │ │ │ │ + if (!coverageSummary.supportedCRS) { │ │ │ │ │ + coverageSummary.supportedCRS = [] │ │ │ │ │ + } │ │ │ │ │ + coverageSummary.supportedCRS.push(crs) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + SupportedFormat: function(node, coverageSummary) { │ │ │ │ │ + var format = this.getChildValue(node); │ │ │ │ │ + if (format) { │ │ │ │ │ + if (!coverageSummary.supportedFormat) { │ │ │ │ │ + coverageSummary.supportedFormat = [] │ │ │ │ │ + } │ │ │ │ │ + coverageSummary.supportedFormat.push(format) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WCSCapabilities.v1.prototype.readers["wcs"]), │ │ │ │ │ + ows: OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1_1_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WCSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.WCSCapabilities.v1, { │ │ │ │ │ + namespaces: { │ │ │ │ │ + wcs: "http://www.opengis.net/wcs", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + ows: "http://www.opengis.net/ows" │ │ │ │ │ + }, │ │ │ │ │ + errorProperty: "service", │ │ │ │ │ + readers: { │ │ │ │ │ + wcs: { │ │ │ │ │ + WCS_Capabilities: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + Service: function(node, obj) { │ │ │ │ │ + obj.service = {}; │ │ │ │ │ + this.readChildNodes(node, obj.service) │ │ │ │ │ + }, │ │ │ │ │ + name: function(node, service) { │ │ │ │ │ + service.name = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + label: function(node, service) { │ │ │ │ │ + service.label = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + keywords: function(node, service) { │ │ │ │ │ + service.keywords = []; │ │ │ │ │ + this.readChildNodes(node, service.keywords) │ │ │ │ │ + }, │ │ │ │ │ + keyword: function(node, keywords) { │ │ │ │ │ + keywords.push(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + responsibleParty: function(node, service) { │ │ │ │ │ + service.responsibleParty = {}; │ │ │ │ │ + this.readChildNodes(node, service.responsibleParty) │ │ │ │ │ + }, │ │ │ │ │ + individualName: function(node, responsibleParty) { │ │ │ │ │ + responsibleParty.individualName = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + organisationName: function(node, responsibleParty) { │ │ │ │ │ + responsibleParty.organisationName = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + positionName: function(node, responsibleParty) { │ │ │ │ │ + responsibleParty.positionName = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + contactInfo: function(node, responsibleParty) { │ │ │ │ │ + responsibleParty.contactInfo = {}; │ │ │ │ │ + this.readChildNodes(node, responsibleParty.contactInfo) │ │ │ │ │ + }, │ │ │ │ │ + phone: function(node, contactInfo) { │ │ │ │ │ + contactInfo.phone = {}; │ │ │ │ │ + this.readChildNodes(node, contactInfo.phone) │ │ │ │ │ + }, │ │ │ │ │ + voice: function(node, phone) { │ │ │ │ │ + phone.voice = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + facsimile: function(node, phone) { │ │ │ │ │ + phone.facsimile = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + address: function(node, contactInfo) { │ │ │ │ │ + contactInfo.address = {}; │ │ │ │ │ + this.readChildNodes(node, contactInfo.address) │ │ │ │ │ + }, │ │ │ │ │ + deliveryPoint: function(node, address) { │ │ │ │ │ + address.deliveryPoint = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + city: function(node, address) { │ │ │ │ │ + address.city = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + postalCode: function(node, address) { │ │ │ │ │ + address.postalCode = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + country: function(node, address) { │ │ │ │ │ + address.country = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + electronicMailAddress: function(node, address) { │ │ │ │ │ + address.electronicMailAddress = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + fees: function(node, service) { │ │ │ │ │ + service.fees = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + accessConstraints: function(node, service) { │ │ │ │ │ + service.accessConstraints = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + ContentMetadata: function(node, obj) { │ │ │ │ │ + obj.contentMetadata = []; │ │ │ │ │ + this.readChildNodes(node, obj.contentMetadata) │ │ │ │ │ + }, │ │ │ │ │ + CoverageOfferingBrief: function(node, contentMetadata) { │ │ │ │ │ + var coverageOfferingBrief = {}; │ │ │ │ │ + this.readChildNodes(node, coverageOfferingBrief); │ │ │ │ │ + contentMetadata.push(coverageOfferingBrief) │ │ │ │ │ + }, │ │ │ │ │ + name: function(node, coverageOfferingBrief) { │ │ │ │ │ + coverageOfferingBrief.name = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + label: function(node, coverageOfferingBrief) { │ │ │ │ │ + coverageOfferingBrief.label = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + lonLatEnvelope: function(node, coverageOfferingBrief) { │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, "http://www.opengis.net/gml", "pos"); │ │ │ │ │ + if (nodeList.length == 2) { │ │ │ │ │ + var min = {}; │ │ │ │ │ + var max = {}; │ │ │ │ │ + OpenLayers.Format.GML.v3.prototype.readers["gml"].pos.apply(this, [nodeList[0], min]); │ │ │ │ │ + OpenLayers.Format.GML.v3.prototype.readers["gml"].pos.apply(this, [nodeList[1], max]); │ │ │ │ │ + coverageOfferingBrief.lonLatEnvelope = {}; │ │ │ │ │ + coverageOfferingBrief.lonLatEnvelope.srsName = node.getAttribute("srsName"); │ │ │ │ │ + coverageOfferingBrief.lonLatEnvelope.min = min.points[0]; │ │ │ │ │ + coverageOfferingBrief.lonLatEnvelope.max = max.points[0] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1_0_0" │ │ │ │ │ +}); │ │ │ │ │ OpenLayers.Format.XLS.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ namespaces: { │ │ │ │ │ xls: "http://www.opengis.net/xls", │ │ │ │ │ gml: "http://www.opengis.net/gml", │ │ │ │ │ xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ }, │ │ │ │ │ regExes: { │ │ │ │ │ @@ -24063,789 +26210,14 @@ │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Format.XLS.v1_1_0 = OpenLayers.Class(OpenLayers.Format.XLS.v1, { │ │ │ │ │ VERSION: "1.1", │ │ │ │ │ schemaLocation: "http://www.opengis.net/xls http://schemas.opengis.net/ols/1.1.0/LocationUtilityService.xsd", │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.XLS.v1_1_0" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Format.XLS.v1_1 = OpenLayers.Format.XLS.v1_1_0; │ │ │ │ │ -OpenLayers.Format.WCSCapabilities.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: /^\s*|\s*$/g, │ │ │ │ │ - splitSpace: /\s+/ │ │ │ │ │ - }, │ │ │ │ │ - defaultPrefix: "wcs", │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ - } │ │ │ │ │ - var raw = data; │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement │ │ │ │ │ - } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - return capabilities │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WCSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.WCSCapabilities.v1, { │ │ │ │ │ - namespaces: { │ │ │ │ │ - wcs: "http://www.opengis.net/wcs", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - ows: "http://www.opengis.net/ows" │ │ │ │ │ - }, │ │ │ │ │ - errorProperty: "service", │ │ │ │ │ - readers: { │ │ │ │ │ - wcs: { │ │ │ │ │ - WCS_Capabilities: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - Service: function(node, obj) { │ │ │ │ │ - obj.service = {}; │ │ │ │ │ - this.readChildNodes(node, obj.service) │ │ │ │ │ - }, │ │ │ │ │ - name: function(node, service) { │ │ │ │ │ - service.name = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - label: function(node, service) { │ │ │ │ │ - service.label = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - keywords: function(node, service) { │ │ │ │ │ - service.keywords = []; │ │ │ │ │ - this.readChildNodes(node, service.keywords) │ │ │ │ │ - }, │ │ │ │ │ - keyword: function(node, keywords) { │ │ │ │ │ - keywords.push(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - responsibleParty: function(node, service) { │ │ │ │ │ - service.responsibleParty = {}; │ │ │ │ │ - this.readChildNodes(node, service.responsibleParty) │ │ │ │ │ - }, │ │ │ │ │ - individualName: function(node, responsibleParty) { │ │ │ │ │ - responsibleParty.individualName = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - organisationName: function(node, responsibleParty) { │ │ │ │ │ - responsibleParty.organisationName = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - positionName: function(node, responsibleParty) { │ │ │ │ │ - responsibleParty.positionName = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - contactInfo: function(node, responsibleParty) { │ │ │ │ │ - responsibleParty.contactInfo = {}; │ │ │ │ │ - this.readChildNodes(node, responsibleParty.contactInfo) │ │ │ │ │ - }, │ │ │ │ │ - phone: function(node, contactInfo) { │ │ │ │ │ - contactInfo.phone = {}; │ │ │ │ │ - this.readChildNodes(node, contactInfo.phone) │ │ │ │ │ - }, │ │ │ │ │ - voice: function(node, phone) { │ │ │ │ │ - phone.voice = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - facsimile: function(node, phone) { │ │ │ │ │ - phone.facsimile = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - address: function(node, contactInfo) { │ │ │ │ │ - contactInfo.address = {}; │ │ │ │ │ - this.readChildNodes(node, contactInfo.address) │ │ │ │ │ - }, │ │ │ │ │ - deliveryPoint: function(node, address) { │ │ │ │ │ - address.deliveryPoint = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - city: function(node, address) { │ │ │ │ │ - address.city = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - postalCode: function(node, address) { │ │ │ │ │ - address.postalCode = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - country: function(node, address) { │ │ │ │ │ - address.country = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - electronicMailAddress: function(node, address) { │ │ │ │ │ - address.electronicMailAddress = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - fees: function(node, service) { │ │ │ │ │ - service.fees = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - accessConstraints: function(node, service) { │ │ │ │ │ - service.accessConstraints = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ContentMetadata: function(node, obj) { │ │ │ │ │ - obj.contentMetadata = []; │ │ │ │ │ - this.readChildNodes(node, obj.contentMetadata) │ │ │ │ │ - }, │ │ │ │ │ - CoverageOfferingBrief: function(node, contentMetadata) { │ │ │ │ │ - var coverageOfferingBrief = {}; │ │ │ │ │ - this.readChildNodes(node, coverageOfferingBrief); │ │ │ │ │ - contentMetadata.push(coverageOfferingBrief) │ │ │ │ │ - }, │ │ │ │ │ - name: function(node, coverageOfferingBrief) { │ │ │ │ │ - coverageOfferingBrief.name = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - label: function(node, coverageOfferingBrief) { │ │ │ │ │ - coverageOfferingBrief.label = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - lonLatEnvelope: function(node, coverageOfferingBrief) { │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, "http://www.opengis.net/gml", "pos"); │ │ │ │ │ - if (nodeList.length == 2) { │ │ │ │ │ - var min = {}; │ │ │ │ │ - var max = {}; │ │ │ │ │ - OpenLayers.Format.GML.v3.prototype.readers["gml"].pos.apply(this, [nodeList[0], min]); │ │ │ │ │ - OpenLayers.Format.GML.v3.prototype.readers["gml"].pos.apply(this, [nodeList[1], max]); │ │ │ │ │ - coverageOfferingBrief.lonLatEnvelope = {}; │ │ │ │ │ - coverageOfferingBrief.lonLatEnvelope.srsName = node.getAttribute("srsName"); │ │ │ │ │ - coverageOfferingBrief.lonLatEnvelope.min = min.points[0]; │ │ │ │ │ - coverageOfferingBrief.lonLatEnvelope.max = max.points[0] │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1_0_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WCSCapabilities.v1_1_0 = OpenLayers.Class(OpenLayers.Format.WCSCapabilities.v1, { │ │ │ │ │ - namespaces: { │ │ │ │ │ - wcs: "http://www.opengis.net/wcs/1.1", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - ows: "http://www.opengis.net/ows/1.1" │ │ │ │ │ - }, │ │ │ │ │ - errorProperty: "operationsMetadata", │ │ │ │ │ - readers: { │ │ │ │ │ - wcs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Capabilities: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - Contents: function(node, request) { │ │ │ │ │ - request.contentMetadata = []; │ │ │ │ │ - this.readChildNodes(node, request.contentMetadata) │ │ │ │ │ - }, │ │ │ │ │ - CoverageSummary: function(node, contentMetadata) { │ │ │ │ │ - var coverageSummary = {}; │ │ │ │ │ - this.readChildNodes(node, coverageSummary); │ │ │ │ │ - contentMetadata.push(coverageSummary) │ │ │ │ │ - }, │ │ │ │ │ - Identifier: function(node, coverageSummary) { │ │ │ │ │ - coverageSummary.identifier = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Title: function(node, coverageSummary) { │ │ │ │ │ - coverageSummary.title = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Abstract: function(node, coverageSummary) { │ │ │ │ │ - coverageSummary["abstract"] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - SupportedCRS: function(node, coverageSummary) { │ │ │ │ │ - var crs = this.getChildValue(node); │ │ │ │ │ - if (crs) { │ │ │ │ │ - if (!coverageSummary.supportedCRS) { │ │ │ │ │ - coverageSummary.supportedCRS = [] │ │ │ │ │ - } │ │ │ │ │ - coverageSummary.supportedCRS.push(crs) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - SupportedFormat: function(node, coverageSummary) { │ │ │ │ │ - var format = this.getChildValue(node); │ │ │ │ │ - if (format) { │ │ │ │ │ - if (!coverageSummary.supportedFormat) { │ │ │ │ │ - coverageSummary.supportedFormat = [] │ │ │ │ │ - } │ │ │ │ │ - coverageSummary.supportedFormat.push(format) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WCSCapabilities.v1.prototype.readers["wcs"]), │ │ │ │ │ - ows: OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1_1_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.SLD.v1_0_0_GeoServer = OpenLayers.Class(OpenLayers.Format.SLD.v1_0_0, { │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - profile: "GeoServer", │ │ │ │ │ - readers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - sld: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Priority: function(node, obj) { │ │ │ │ │ - var value = this.readers.ogc._expression.call(this, node); │ │ │ │ │ - if (value) { │ │ │ │ │ - obj.priority = value │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - VendorOption: function(node, obj) { │ │ │ │ │ - if (!obj.vendorOptions) { │ │ │ │ │ - obj.vendorOptions = {} │ │ │ │ │ - } │ │ │ │ │ - obj.vendorOptions[node.getAttribute("name")] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - TextSymbolizer: function(node, rule) { │ │ │ │ │ - OpenLayers.Format.SLD.v1_0_0.prototype.readers.sld.TextSymbolizer.apply(this, arguments); │ │ │ │ │ - var symbolizer = this.multipleSymbolizers ? rule.symbolizers[rule.symbolizers.length - 1] : rule.symbolizer["Text"]; │ │ │ │ │ - if (symbolizer.graphic === undefined) { │ │ │ │ │ - symbolizer.graphic = false │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.SLD.v1_0_0.prototype.readers["sld"]) │ │ │ │ │ - }, OpenLayers.Format.SLD.v1_0_0.prototype.readers), │ │ │ │ │ - writers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - sld: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Priority: function(priority) { │ │ │ │ │ - return this.writers.sld._OGCExpression.call(this, "sld:Priority", priority) │ │ │ │ │ - }, │ │ │ │ │ - VendorOption: function(option) { │ │ │ │ │ - return this.createElementNSPlus("sld:VendorOption", { │ │ │ │ │ - attributes: { │ │ │ │ │ - name: option.name │ │ │ │ │ - }, │ │ │ │ │ - value: option.value │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - TextSymbolizer: function(symbolizer) { │ │ │ │ │ - var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ - var node = writers["sld"]["TextSymbolizer"].apply(this, arguments); │ │ │ │ │ - if (symbolizer.graphic !== false && (symbolizer.externalGraphic || symbolizer.graphicName)) { │ │ │ │ │ - this.writeNode("Graphic", symbolizer, node) │ │ │ │ │ - } │ │ │ │ │ - if ("priority" in symbolizer) { │ │ │ │ │ - this.writeNode("Priority", symbolizer.priority, node) │ │ │ │ │ - } │ │ │ │ │ - return this.addVendorOptions(node, symbolizer) │ │ │ │ │ - }, │ │ │ │ │ - PointSymbolizer: function(symbolizer) { │ │ │ │ │ - var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ - var node = writers["sld"]["PointSymbolizer"].apply(this, arguments); │ │ │ │ │ - return this.addVendorOptions(node, symbolizer) │ │ │ │ │ - }, │ │ │ │ │ - LineSymbolizer: function(symbolizer) { │ │ │ │ │ - var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ - var node = writers["sld"]["LineSymbolizer"].apply(this, arguments); │ │ │ │ │ - return this.addVendorOptions(node, symbolizer) │ │ │ │ │ - }, │ │ │ │ │ - PolygonSymbolizer: function(symbolizer) { │ │ │ │ │ - var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ - var node = writers["sld"]["PolygonSymbolizer"].apply(this, arguments); │ │ │ │ │ - return this.addVendorOptions(node, symbolizer) │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.SLD.v1_0_0.prototype.writers["sld"]) │ │ │ │ │ - }, OpenLayers.Format.SLD.v1_0_0.prototype.writers), │ │ │ │ │ - addVendorOptions: function(node, symbolizer) { │ │ │ │ │ - var options = symbolizer.vendorOptions; │ │ │ │ │ - if (options) { │ │ │ │ │ - for (var key in symbolizer.vendorOptions) { │ │ │ │ │ - this.writeNode("VendorOption", { │ │ │ │ │ - name: key, │ │ │ │ │ - value: symbolizer.vendorOptions[key] │ │ │ │ │ - }, node) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SLD.v1_0_0_GeoServer" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.CSWGetDomain.v2_0_2 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - namespaces: { │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - csw: "http://www.opengis.net/cat/csw/2.0.2" │ │ │ │ │ - }, │ │ │ │ │ - defaultPrefix: "csw", │ │ │ │ │ - version: "2.0.2", │ │ │ │ │ - schemaLocation: "http://www.opengis.net/cat/csw/2.0.2 http://schemas.opengis.net/csw/2.0.2/CSW-discovery.xsd", │ │ │ │ │ - PropertyName: null, │ │ │ │ │ - ParameterName: null, │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement │ │ │ │ │ - } │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readNode(data, obj); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - readers: { │ │ │ │ │ - csw: { │ │ │ │ │ - GetDomainResponse: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - DomainValues: function(node, obj) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(obj.DomainValues)) { │ │ │ │ │ - obj.DomainValues = [] │ │ │ │ │ - } │ │ │ │ │ - var attrs = node.attributes; │ │ │ │ │ - var domainValue = {}; │ │ │ │ │ - for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - domainValue[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ - } │ │ │ │ │ - this.readChildNodes(node, domainValue); │ │ │ │ │ - obj.DomainValues.push(domainValue) │ │ │ │ │ - }, │ │ │ │ │ - PropertyName: function(node, obj) { │ │ │ │ │ - obj.PropertyName = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ParameterName: function(node, obj) { │ │ │ │ │ - obj.ParameterName = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ListOfValues: function(node, obj) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(obj.ListOfValues)) { │ │ │ │ │ - obj.ListOfValues = [] │ │ │ │ │ - } │ │ │ │ │ - this.readChildNodes(node, obj.ListOfValues) │ │ │ │ │ - }, │ │ │ │ │ - Value: function(node, obj) { │ │ │ │ │ - var attrs = node.attributes; │ │ │ │ │ - var value = {}; │ │ │ │ │ - for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - value[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ - } │ │ │ │ │ - value.value = this.getChildValue(node); │ │ │ │ │ - obj.push({ │ │ │ │ │ - Value: value │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - ConceptualScheme: function(node, obj) { │ │ │ │ │ - obj.ConceptualScheme = {}; │ │ │ │ │ - this.readChildNodes(node, obj.ConceptualScheme) │ │ │ │ │ - }, │ │ │ │ │ - Name: function(node, obj) { │ │ │ │ │ - obj.Name = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Document: function(node, obj) { │ │ │ │ │ - obj.Document = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Authority: function(node, obj) { │ │ │ │ │ - obj.Authority = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - RangeOfValues: function(node, obj) { │ │ │ │ │ - obj.RangeOfValues = {}; │ │ │ │ │ - this.readChildNodes(node, obj.RangeOfValues) │ │ │ │ │ - }, │ │ │ │ │ - MinValue: function(node, obj) { │ │ │ │ │ - var attrs = node.attributes; │ │ │ │ │ - var value = {}; │ │ │ │ │ - for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - value[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ - } │ │ │ │ │ - value.value = this.getChildValue(node); │ │ │ │ │ - obj.MinValue = value │ │ │ │ │ - }, │ │ │ │ │ - MaxValue: function(node, obj) { │ │ │ │ │ - var attrs = node.attributes; │ │ │ │ │ - var value = {}; │ │ │ │ │ - for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - value[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ - } │ │ │ │ │ - value.value = this.getChildValue(node); │ │ │ │ │ - obj.MaxValue = value │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - write: function(options) { │ │ │ │ │ - var node = this.writeNode("csw:GetDomain", options); │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [node]) │ │ │ │ │ - }, │ │ │ │ │ - writers: { │ │ │ │ │ - csw: { │ │ │ │ │ - GetDomain: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:GetDomain", { │ │ │ │ │ - attributes: { │ │ │ │ │ - service: "CSW", │ │ │ │ │ - version: this.version │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (options.PropertyName || this.PropertyName) { │ │ │ │ │ - this.writeNode("csw:PropertyName", options.PropertyName || this.PropertyName, node) │ │ │ │ │ - } else if (options.ParameterName || this.ParameterName) { │ │ │ │ │ - this.writeNode("csw:ParameterName", options.ParameterName || this.ParameterName, node) │ │ │ │ │ - } │ │ │ │ │ - this.readChildNodes(node, options); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - PropertyName: function(value) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:PropertyName", { │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - ParameterName: function(value) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:ParameterName", { │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - return node │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.CSWGetDomain.v2_0_2" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WMTSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.OWSCommon.v1_1_0, { │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - namespaces: { │ │ │ │ │ - ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ - wmts: "http://www.opengis.net/wmts/1.0", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ - }, │ │ │ │ │ - yx: null, │ │ │ │ │ - defaultPrefix: "wmts", │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.options = options; │ │ │ │ │ - var yx = OpenLayers.Util.extend({}, OpenLayers.Format.WMTSCapabilities.prototype.yx); │ │ │ │ │ - this.yx = OpenLayers.Util.extend(yx, this.yx) │ │ │ │ │ - }, │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement │ │ │ │ │ - } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - capabilities.version = this.version; │ │ │ │ │ - return capabilities │ │ │ │ │ - }, │ │ │ │ │ - readers: { │ │ │ │ │ - wmts: { │ │ │ │ │ - Capabilities: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - Contents: function(node, obj) { │ │ │ │ │ - obj.contents = {}; │ │ │ │ │ - obj.contents.layers = []; │ │ │ │ │ - obj.contents.tileMatrixSets = {}; │ │ │ │ │ - this.readChildNodes(node, obj.contents) │ │ │ │ │ - }, │ │ │ │ │ - Layer: function(node, obj) { │ │ │ │ │ - var layer = { │ │ │ │ │ - styles: [], │ │ │ │ │ - formats: [], │ │ │ │ │ - dimensions: [], │ │ │ │ │ - tileMatrixSetLinks: [] │ │ │ │ │ - }; │ │ │ │ │ - layer.layers = []; │ │ │ │ │ - this.readChildNodes(node, layer); │ │ │ │ │ - obj.layers.push(layer) │ │ │ │ │ - }, │ │ │ │ │ - Style: function(node, obj) { │ │ │ │ │ - var style = {}; │ │ │ │ │ - style.isDefault = node.getAttribute("isDefault") === "true"; │ │ │ │ │ - this.readChildNodes(node, style); │ │ │ │ │ - obj.styles.push(style) │ │ │ │ │ - }, │ │ │ │ │ - Format: function(node, obj) { │ │ │ │ │ - obj.formats.push(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - TileMatrixSetLink: function(node, obj) { │ │ │ │ │ - var tileMatrixSetLink = {}; │ │ │ │ │ - this.readChildNodes(node, tileMatrixSetLink); │ │ │ │ │ - obj.tileMatrixSetLinks.push(tileMatrixSetLink) │ │ │ │ │ - }, │ │ │ │ │ - TileMatrixSet: function(node, obj) { │ │ │ │ │ - if (obj.layers) { │ │ │ │ │ - var tileMatrixSet = { │ │ │ │ │ - matrixIds: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, tileMatrixSet); │ │ │ │ │ - obj.tileMatrixSets[tileMatrixSet.identifier] = tileMatrixSet │ │ │ │ │ - } else { │ │ │ │ │ - obj.tileMatrixSet = this.getChildValue(node) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - TileMatrix: function(node, obj) { │ │ │ │ │ - var tileMatrix = { │ │ │ │ │ - supportedCRS: obj.supportedCRS │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, tileMatrix); │ │ │ │ │ - obj.matrixIds.push(tileMatrix) │ │ │ │ │ - }, │ │ │ │ │ - ScaleDenominator: function(node, obj) { │ │ │ │ │ - obj.scaleDenominator = parseFloat(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - TopLeftCorner: function(node, obj) { │ │ │ │ │ - var topLeftCorner = this.getChildValue(node); │ │ │ │ │ - var coords = topLeftCorner.split(" "); │ │ │ │ │ - var yx; │ │ │ │ │ - if (obj.supportedCRS) { │ │ │ │ │ - var crs = obj.supportedCRS.replace(/urn:ogc:def:crs:(\w+):.+:(\w+)$/, "urn:ogc:def:crs:$1::$2"); │ │ │ │ │ - yx = !!this.yx[crs] │ │ │ │ │ - } │ │ │ │ │ - if (yx) { │ │ │ │ │ - obj.topLeftCorner = new OpenLayers.LonLat(coords[1], coords[0]) │ │ │ │ │ - } else { │ │ │ │ │ - obj.topLeftCorner = new OpenLayers.LonLat(coords[0], coords[1]) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - TileWidth: function(node, obj) { │ │ │ │ │ - obj.tileWidth = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - TileHeight: function(node, obj) { │ │ │ │ │ - obj.tileHeight = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - MatrixWidth: function(node, obj) { │ │ │ │ │ - obj.matrixWidth = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - MatrixHeight: function(node, obj) { │ │ │ │ │ - obj.matrixHeight = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - ResourceURL: function(node, obj) { │ │ │ │ │ - obj.resourceUrl = obj.resourceUrl || {}; │ │ │ │ │ - var resourceType = node.getAttribute("resourceType"); │ │ │ │ │ - if (!obj.resourceUrls) { │ │ │ │ │ - obj.resourceUrls = [] │ │ │ │ │ - } │ │ │ │ │ - var resourceUrl = obj.resourceUrl[resourceType] = { │ │ │ │ │ - format: node.getAttribute("format"), │ │ │ │ │ - template: node.getAttribute("template"), │ │ │ │ │ - resourceType: resourceType │ │ │ │ │ - }; │ │ │ │ │ - obj.resourceUrls.push(resourceUrl) │ │ │ │ │ - }, │ │ │ │ │ - WSDL: function(node, obj) { │ │ │ │ │ - obj.wsdl = {}; │ │ │ │ │ - obj.wsdl.href = node.getAttribute("xlink:href") │ │ │ │ │ - }, │ │ │ │ │ - ServiceMetadataURL: function(node, obj) { │ │ │ │ │ - obj.serviceMetadataUrl = {}; │ │ │ │ │ - obj.serviceMetadataUrl.href = node.getAttribute("xlink:href") │ │ │ │ │ - }, │ │ │ │ │ - LegendURL: function(node, obj) { │ │ │ │ │ - obj.legend = {}; │ │ │ │ │ - obj.legend.href = node.getAttribute("xlink:href"); │ │ │ │ │ - obj.legend.format = node.getAttribute("format") │ │ │ │ │ - }, │ │ │ │ │ - Dimension: function(node, obj) { │ │ │ │ │ - var dimension = { │ │ │ │ │ - values: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, dimension); │ │ │ │ │ - obj.dimensions.push(dimension) │ │ │ │ │ - }, │ │ │ │ │ - Default: function(node, obj) { │ │ │ │ │ - obj["default"] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Value: function(node, obj) { │ │ │ │ │ - obj.values.push(this.getChildValue(node)) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - ows: OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMTSCapabilities.v1_0_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WPSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - namespaces: { │ │ │ │ │ - ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ - wps: "http://www.opengis.net/wps/1.0.0", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ - }, │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: /^\s*|\s*$/g, │ │ │ │ │ - removeSpace: /\s*/g, │ │ │ │ │ - splitSpace: /\s+/, │ │ │ │ │ - trimComma: /\s*,\s*/g │ │ │ │ │ - }, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement │ │ │ │ │ - } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - return capabilities │ │ │ │ │ - }, │ │ │ │ │ - readers: { │ │ │ │ │ - wps: { │ │ │ │ │ - Capabilities: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - ProcessOfferings: function(node, obj) { │ │ │ │ │ - obj.processOfferings = {}; │ │ │ │ │ - this.readChildNodes(node, obj.processOfferings) │ │ │ │ │ - }, │ │ │ │ │ - Process: function(node, processOfferings) { │ │ │ │ │ - var processVersion = this.getAttributeNS(node, this.namespaces.wps, "processVersion"); │ │ │ │ │ - var process = { │ │ │ │ │ - processVersion: processVersion │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, process); │ │ │ │ │ - processOfferings[process.identifier] = process │ │ │ │ │ - }, │ │ │ │ │ - Languages: function(node, obj) { │ │ │ │ │ - obj.languages = []; │ │ │ │ │ - this.readChildNodes(node, obj.languages) │ │ │ │ │ - }, │ │ │ │ │ - Default: function(node, languages) { │ │ │ │ │ - var language = { │ │ │ │ │ - isDefault: true │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, language); │ │ │ │ │ - languages.push(language) │ │ │ │ │ - }, │ │ │ │ │ - Supported: function(node, languages) { │ │ │ │ │ - var language = {}; │ │ │ │ │ - this.readChildNodes(node, language); │ │ │ │ │ - languages.push(language) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - ows: OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WPSCapabilities.v1_0_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WMSDescribeLayer.v1_1_1 = OpenLayers.Class(OpenLayers.Format.WMSDescribeLayer, { │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.WMSDescribeLayer.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ - } │ │ │ │ │ - var root = data.documentElement; │ │ │ │ │ - var children = root.childNodes; │ │ │ │ │ - var describelayer = { │ │ │ │ │ - layerDescriptions: [] │ │ │ │ │ - }; │ │ │ │ │ - var childNode, nodeName; │ │ │ │ │ - for (var i = 0; i < children.length; ++i) { │ │ │ │ │ - childNode = children[i]; │ │ │ │ │ - nodeName = childNode.nodeName; │ │ │ │ │ - if (nodeName == "LayerDescription") { │ │ │ │ │ - var layerName = childNode.getAttribute("name"); │ │ │ │ │ - var owsType = ""; │ │ │ │ │ - var owsURL = ""; │ │ │ │ │ - var typeName = ""; │ │ │ │ │ - if (childNode.getAttribute("owsType")) { │ │ │ │ │ - owsType = childNode.getAttribute("owsType"); │ │ │ │ │ - owsURL = childNode.getAttribute("owsURL") │ │ │ │ │ - } else { │ │ │ │ │ - if (childNode.getAttribute("wfs") != "") { │ │ │ │ │ - owsType = "WFS"; │ │ │ │ │ - owsURL = childNode.getAttribute("wfs") │ │ │ │ │ - } else if (childNode.getAttribute("wcs") != "") { │ │ │ │ │ - owsType = "WCS"; │ │ │ │ │ - owsURL = childNode.getAttribute("wcs") │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var query = childNode.getElementsByTagName("Query"); │ │ │ │ │ - if (query.length > 0) { │ │ │ │ │ - typeName = query[0].getAttribute("typeName"); │ │ │ │ │ - if (!typeName) { │ │ │ │ │ - typeName = query[0].getAttribute("typename") │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var layerDescription = { │ │ │ │ │ - layerName: layerName, │ │ │ │ │ - owsType: owsType, │ │ │ │ │ - owsURL: owsURL, │ │ │ │ │ - typeName: typeName │ │ │ │ │ - }; │ │ │ │ │ - describelayer.layerDescriptions.push(layerDescription); │ │ │ │ │ - describelayer.length = describelayer.layerDescriptions.length; │ │ │ │ │ - describelayer[describelayer.length - 1] = layerDescription │ │ │ │ │ - } else if (nodeName == "ServiceException") { │ │ │ │ │ - var parser = new OpenLayers.Format.OGCExceptionReport; │ │ │ │ │ - return { │ │ │ │ │ - error: parser.read(data) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return describelayer │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSDescribeLayer.v1_1_1" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WMSDescribeLayer.v1_1_0 = OpenLayers.Format.WMSDescribeLayer.v1_1_1; │ │ │ │ │ -OpenLayers.Format.ArcXML.Features = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - read: function(data) { │ │ │ │ │ - var axl = new OpenLayers.Format.ArcXML; │ │ │ │ │ - var parsed = axl.read(data); │ │ │ │ │ - return parsed.features.feature │ │ │ │ │ - } │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WFST.v1_0_0 = OpenLayers.Class(OpenLayers.Format.Filter.v1_0_0, OpenLayers.Format.WFST.v1, { │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - srsNameInQuery: false, │ │ │ │ │ - schemaLocations: { │ │ │ │ │ - wfs: "http://schemas.opengis.net/wfs/1.0.0/WFS-transaction.xsd" │ │ │ │ │ - }, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.Filter.v1_0_0.prototype.initialize.apply(this, [options]); │ │ │ │ │ - OpenLayers.Format.WFST.v1.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ - readNode: function(node, obj, first) { │ │ │ │ │ - return OpenLayers.Format.GML.v2.prototype.readNode.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - readers: { │ │ │ │ │ - wfs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - WFS_TransactionResponse: function(node, obj) { │ │ │ │ │ - obj.insertIds = []; │ │ │ │ │ - obj.success = false; │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - InsertResult: function(node, container) { │ │ │ │ │ - var obj = { │ │ │ │ │ - fids: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - container.insertIds = container.insertIds.concat(obj.fids) │ │ │ │ │ - }, │ │ │ │ │ - TransactionResult: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - Status: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - SUCCESS: function(node, obj) { │ │ │ │ │ - obj.success = true │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WFST.v1.prototype.readers["wfs"]), │ │ │ │ │ - gml: OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ - feature: OpenLayers.Format.GML.v2.prototype.readers["feature"], │ │ │ │ │ - ogc: OpenLayers.Format.Filter.v1_0_0.prototype.readers["ogc"] │ │ │ │ │ - }, │ │ │ │ │ - writers: { │ │ │ │ │ - wfs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Query: function(options) { │ │ │ │ │ - options = OpenLayers.Util.extend({ │ │ │ │ │ - featureNS: this.featureNS, │ │ │ │ │ - featurePrefix: this.featurePrefix, │ │ │ │ │ - featureType: this.featureType, │ │ │ │ │ - srsName: this.srsName, │ │ │ │ │ - srsNameInQuery: this.srsNameInQuery │ │ │ │ │ - }, options); │ │ │ │ │ - var prefix = options.featurePrefix; │ │ │ │ │ - var node = this.createElementNSPlus("wfs:Query", { │ │ │ │ │ - attributes: { │ │ │ │ │ - typeName: (prefix ? prefix + ":" : "") + options.featureType │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (options.srsNameInQuery && options.srsName) { │ │ │ │ │ - node.setAttribute("srsName", options.srsName) │ │ │ │ │ - } │ │ │ │ │ - if (options.featureNS) { │ │ │ │ │ - node.setAttribute("xmlns:" + prefix, options.featureNS) │ │ │ │ │ - } │ │ │ │ │ - if (options.propertyNames) { │ │ │ │ │ - for (var i = 0, len = options.propertyNames.length; i < len; i++) { │ │ │ │ │ - this.writeNode("ogc:PropertyName", { │ │ │ │ │ - property: options.propertyNames[i] │ │ │ │ │ - }, node) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (options.filter) { │ │ │ │ │ - this.setFilterProperty(options.filter); │ │ │ │ │ - this.writeNode("ogc:Filter", options.filter, node) │ │ │ │ │ - } │ │ │ │ │ - return node │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WFST.v1.prototype.writers["wfs"]), │ │ │ │ │ - gml: OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ - feature: OpenLayers.Format.GML.v2.prototype.writers["feature"], │ │ │ │ │ - ogc: OpenLayers.Format.Filter.v1_0_0.prototype.writers["ogc"] │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFST.v1_0_0" │ │ │ │ │ -}); │ │ │ │ │ OpenLayers.Format.SOSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.SOSCapabilities, { │ │ │ │ │ namespaces: { │ │ │ │ │ ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ gml: "http://www.opengis.net/gml", │ │ │ │ │ xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ }, │ │ │ │ │ @@ -25586,48 +26958,14 @@ │ │ │ │ │ if (links.length > 0) { │ │ │ │ │ this.read_wmc_OnlineResource(object, links[0]) │ │ │ │ │ } │ │ │ │ │ return object.href │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.WMC.v1" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.WMC.v1_0_0 = OpenLayers.Class(OpenLayers.Format.WMC.v1, { │ │ │ │ │ - VERSION: "1.0.0", │ │ │ │ │ - schemaLocation: "http://www.opengis.net/context http://schemas.opengis.net/context/1.0.0/context.xsd", │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.WMC.v1.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ - read_wmc_SRS: function(layerContext, node) { │ │ │ │ │ - var srs = this.getChildValue(node); │ │ │ │ │ - if (typeof layerContext.projections != "object") { │ │ │ │ │ - layerContext.projections = {} │ │ │ │ │ - } │ │ │ │ │ - var values = srs.split(/ +/); │ │ │ │ │ - for (var i = 0, len = values.length; i < len; i++) { │ │ │ │ │ - layerContext.projections[values[i]] = true │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - write_wmc_Layer: function(context) { │ │ │ │ │ - var node = OpenLayers.Format.WMC.v1.prototype.write_wmc_Layer.apply(this, [context]); │ │ │ │ │ - if (context.srs) { │ │ │ │ │ - var projections = []; │ │ │ │ │ - for (var name in context.srs) { │ │ │ │ │ - projections.push(name) │ │ │ │ │ - } │ │ │ │ │ - node.appendChild(this.createElementDefaultNS("SRS", projections.join(" "))) │ │ │ │ │ - } │ │ │ │ │ - node.appendChild(this.write_wmc_FormatList(context)); │ │ │ │ │ - node.appendChild(this.write_wmc_StyleList(context)); │ │ │ │ │ - if (context.dimensions) { │ │ │ │ │ - node.appendChild(this.write_wmc_DimensionList(context)) │ │ │ │ │ - } │ │ │ │ │ - node.appendChild(this.write_wmc_LayerExtension(context)) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMC.v1_0_0" │ │ │ │ │ -}); │ │ │ │ │ OpenLayers.Format.WMC.v1_1_0 = OpenLayers.Class(OpenLayers.Format.WMC.v1, { │ │ │ │ │ VERSION: "1.1.0", │ │ │ │ │ schemaLocation: "http://www.opengis.net/context http://schemas.opengis.net/context/1.1.0/context.xsd", │ │ │ │ │ initialize: function(options) { │ │ │ │ │ OpenLayers.Format.WMC.v1.prototype.initialize.apply(this, [options]) │ │ │ │ │ }, │ │ │ │ │ read_sld_MinScaleDenominator: function(layerContext, node) { │ │ │ │ │ @@ -25668,716 +27006,143 @@ │ │ │ │ │ node.appendChild(this.write_wmc_DimensionList(context)) │ │ │ │ │ } │ │ │ │ │ node.appendChild(this.write_wmc_LayerExtension(context)); │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.WMC.v1_1_0" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - namespaces: { │ │ │ │ │ - wms: "http://www.opengis.net/wms", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ +OpenLayers.Format.WMC.v1_0_0 = OpenLayers.Class(OpenLayers.Format.WMC.v1, { │ │ │ │ │ + VERSION: "1.0.0", │ │ │ │ │ + schemaLocation: "http://www.opengis.net/context http://schemas.opengis.net/context/1.0.0/context.xsd", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.WMC.v1.prototype.initialize.apply(this, [options]) │ │ │ │ │ }, │ │ │ │ │ - defaultPrefix: "wms", │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ - } │ │ │ │ │ - var raw = data; │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement │ │ │ │ │ + read_wmc_SRS: function(layerContext, node) { │ │ │ │ │ + var srs = this.getChildValue(node); │ │ │ │ │ + if (typeof layerContext.projections != "object") { │ │ │ │ │ + layerContext.projections = {} │ │ │ │ │ } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - if (capabilities.service === undefined) { │ │ │ │ │ - var parser = new OpenLayers.Format.OGCExceptionReport; │ │ │ │ │ - capabilities.error = parser.read(raw) │ │ │ │ │ + var values = srs.split(/ +/); │ │ │ │ │ + for (var i = 0, len = values.length; i < len; i++) { │ │ │ │ │ + layerContext.projections[values[i]] = true │ │ │ │ │ } │ │ │ │ │ - return capabilities │ │ │ │ │ }, │ │ │ │ │ - readers: { │ │ │ │ │ - wms: { │ │ │ │ │ - Service: function(node, obj) { │ │ │ │ │ - obj.service = {}; │ │ │ │ │ - this.readChildNodes(node, obj.service) │ │ │ │ │ - }, │ │ │ │ │ - Name: function(node, obj) { │ │ │ │ │ - obj.name = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Title: function(node, obj) { │ │ │ │ │ - obj.title = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Abstract: function(node, obj) { │ │ │ │ │ - obj["abstract"] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - BoundingBox: function(node, obj) { │ │ │ │ │ - var bbox = {}; │ │ │ │ │ - bbox.bbox = [parseFloat(node.getAttribute("minx")), parseFloat(node.getAttribute("miny")), parseFloat(node.getAttribute("maxx")), parseFloat(node.getAttribute("maxy"))]; │ │ │ │ │ - var res = { │ │ │ │ │ - x: parseFloat(node.getAttribute("resx")), │ │ │ │ │ - y: parseFloat(node.getAttribute("resy")) │ │ │ │ │ - }; │ │ │ │ │ - if (!(isNaN(res.x) && isNaN(res.y))) { │ │ │ │ │ - bbox.res = res │ │ │ │ │ - } │ │ │ │ │ - return bbox │ │ │ │ │ - }, │ │ │ │ │ - OnlineResource: function(node, obj) { │ │ │ │ │ - obj.href = this.getAttributeNS(node, this.namespaces.xlink, "href") │ │ │ │ │ - }, │ │ │ │ │ - ContactInformation: function(node, obj) { │ │ │ │ │ - obj.contactInformation = {}; │ │ │ │ │ - this.readChildNodes(node, obj.contactInformation) │ │ │ │ │ - }, │ │ │ │ │ - ContactPersonPrimary: function(node, obj) { │ │ │ │ │ - obj.personPrimary = {}; │ │ │ │ │ - this.readChildNodes(node, obj.personPrimary) │ │ │ │ │ - }, │ │ │ │ │ - ContactPerson: function(node, obj) { │ │ │ │ │ - obj.person = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ContactOrganization: function(node, obj) { │ │ │ │ │ - obj.organization = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ContactPosition: function(node, obj) { │ │ │ │ │ - obj.position = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ContactAddress: function(node, obj) { │ │ │ │ │ - obj.contactAddress = {}; │ │ │ │ │ - this.readChildNodes(node, obj.contactAddress) │ │ │ │ │ - }, │ │ │ │ │ - AddressType: function(node, obj) { │ │ │ │ │ - obj.type = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Address: function(node, obj) { │ │ │ │ │ - obj.address = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - City: function(node, obj) { │ │ │ │ │ - obj.city = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - StateOrProvince: function(node, obj) { │ │ │ │ │ - obj.stateOrProvince = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - PostCode: function(node, obj) { │ │ │ │ │ - obj.postcode = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Country: function(node, obj) { │ │ │ │ │ - obj.country = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ContactVoiceTelephone: function(node, obj) { │ │ │ │ │ - obj.phone = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ContactFacsimileTelephone: function(node, obj) { │ │ │ │ │ - obj.fax = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ContactElectronicMailAddress: function(node, obj) { │ │ │ │ │ - obj.email = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Fees: function(node, obj) { │ │ │ │ │ - var fees = this.getChildValue(node); │ │ │ │ │ - if (fees && fees.toLowerCase() != "none") { │ │ │ │ │ - obj.fees = fees │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - AccessConstraints: function(node, obj) { │ │ │ │ │ - var constraints = this.getChildValue(node); │ │ │ │ │ - if (constraints && constraints.toLowerCase() != "none") { │ │ │ │ │ - obj.accessConstraints = constraints │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - Capability: function(node, obj) { │ │ │ │ │ - obj.capability = { │ │ │ │ │ - nestedLayers: [], │ │ │ │ │ - layers: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.capability) │ │ │ │ │ - }, │ │ │ │ │ - Request: function(node, obj) { │ │ │ │ │ - obj.request = {}; │ │ │ │ │ - this.readChildNodes(node, obj.request) │ │ │ │ │ - }, │ │ │ │ │ - GetCapabilities: function(node, obj) { │ │ │ │ │ - obj.getcapabilities = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getcapabilities) │ │ │ │ │ - }, │ │ │ │ │ - Format: function(node, obj) { │ │ │ │ │ - if (OpenLayers.Util.isArray(obj.formats)) { │ │ │ │ │ - obj.formats.push(this.getChildValue(node)) │ │ │ │ │ - } else { │ │ │ │ │ - obj.format = this.getChildValue(node) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - DCPType: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - HTTP: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - Get: function(node, obj) { │ │ │ │ │ - obj.get = {}; │ │ │ │ │ - this.readChildNodes(node, obj.get); │ │ │ │ │ - if (!obj.href) { │ │ │ │ │ - obj.href = obj.get.href │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - Post: function(node, obj) { │ │ │ │ │ - obj.post = {}; │ │ │ │ │ - this.readChildNodes(node, obj.post); │ │ │ │ │ - if (!obj.href) { │ │ │ │ │ - obj.href = obj.get.href │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - GetMap: function(node, obj) { │ │ │ │ │ - obj.getmap = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getmap) │ │ │ │ │ - }, │ │ │ │ │ - GetFeatureInfo: function(node, obj) { │ │ │ │ │ - obj.getfeatureinfo = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getfeatureinfo) │ │ │ │ │ - }, │ │ │ │ │ - Exception: function(node, obj) { │ │ │ │ │ - obj.exception = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.exception) │ │ │ │ │ - }, │ │ │ │ │ - Layer: function(node, obj) { │ │ │ │ │ - var parentLayer, capability; │ │ │ │ │ - if (obj.capability) { │ │ │ │ │ - capability = obj.capability; │ │ │ │ │ - parentLayer = obj │ │ │ │ │ - } else { │ │ │ │ │ - capability = obj │ │ │ │ │ - } │ │ │ │ │ - var attrNode = node.getAttributeNode("queryable"); │ │ │ │ │ - var queryable = attrNode && attrNode.specified ? node.getAttribute("queryable") : null; │ │ │ │ │ - attrNode = node.getAttributeNode("cascaded"); │ │ │ │ │ - var cascaded = attrNode && attrNode.specified ? node.getAttribute("cascaded") : null; │ │ │ │ │ - attrNode = node.getAttributeNode("opaque"); │ │ │ │ │ - var opaque = attrNode && attrNode.specified ? node.getAttribute("opaque") : null; │ │ │ │ │ - var noSubsets = node.getAttribute("noSubsets"); │ │ │ │ │ - var fixedWidth = node.getAttribute("fixedWidth"); │ │ │ │ │ - var fixedHeight = node.getAttribute("fixedHeight"); │ │ │ │ │ - var parent = parentLayer || {}, │ │ │ │ │ - extend = OpenLayers.Util.extend; │ │ │ │ │ - var layer = { │ │ │ │ │ - nestedLayers: [], │ │ │ │ │ - styles: parentLayer ? [].concat(parentLayer.styles) : [], │ │ │ │ │ - srs: parentLayer ? extend({}, parent.srs) : {}, │ │ │ │ │ - metadataURLs: [], │ │ │ │ │ - bbox: parentLayer ? extend({}, parent.bbox) : {}, │ │ │ │ │ - llbbox: parent.llbbox, │ │ │ │ │ - dimensions: parentLayer ? extend({}, parent.dimensions) : {}, │ │ │ │ │ - authorityURLs: parentLayer ? extend({}, parent.authorityURLs) : {}, │ │ │ │ │ - identifiers: {}, │ │ │ │ │ - keywords: [], │ │ │ │ │ - queryable: queryable && queryable !== "" ? queryable === "1" || queryable === "true" : parent.queryable || false, │ │ │ │ │ - cascaded: cascaded !== null ? parseInt(cascaded) : parent.cascaded || 0, │ │ │ │ │ - opaque: opaque ? opaque === "1" || opaque === "true" : parent.opaque || false, │ │ │ │ │ - noSubsets: noSubsets !== null ? noSubsets === "1" || noSubsets === "true" : parent.noSubsets || false, │ │ │ │ │ - fixedWidth: fixedWidth != null ? parseInt(fixedWidth) : parent.fixedWidth || 0, │ │ │ │ │ - fixedHeight: fixedHeight != null ? parseInt(fixedHeight) : parent.fixedHeight || 0, │ │ │ │ │ - minScale: parent.minScale, │ │ │ │ │ - maxScale: parent.maxScale, │ │ │ │ │ - attribution: parent.attribution │ │ │ │ │ - }; │ │ │ │ │ - obj.nestedLayers.push(layer); │ │ │ │ │ - layer.capability = capability; │ │ │ │ │ - this.readChildNodes(node, layer); │ │ │ │ │ - delete layer.capability; │ │ │ │ │ - if (layer.name) { │ │ │ │ │ - var parts = layer.name.split(":"), │ │ │ │ │ - request = capability.request, │ │ │ │ │ - gfi = request.getfeatureinfo; │ │ │ │ │ - if (parts.length > 0) { │ │ │ │ │ - layer.prefix = parts[0] │ │ │ │ │ - } │ │ │ │ │ - capability.layers.push(layer); │ │ │ │ │ - if (layer.formats === undefined) { │ │ │ │ │ - layer.formats = request.getmap.formats │ │ │ │ │ - } │ │ │ │ │ - if (layer.infoFormats === undefined && gfi) { │ │ │ │ │ - layer.infoFormats = gfi.formats │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - Attribution: function(node, obj) { │ │ │ │ │ - obj.attribution = {}; │ │ │ │ │ - this.readChildNodes(node, obj.attribution) │ │ │ │ │ - }, │ │ │ │ │ - LogoURL: function(node, obj) { │ │ │ │ │ - obj.logo = { │ │ │ │ │ - width: node.getAttribute("width"), │ │ │ │ │ - height: node.getAttribute("height") │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.logo) │ │ │ │ │ - }, │ │ │ │ │ - Style: function(node, obj) { │ │ │ │ │ - var style = {}; │ │ │ │ │ - obj.styles.push(style); │ │ │ │ │ - this.readChildNodes(node, style) │ │ │ │ │ - }, │ │ │ │ │ - LegendURL: function(node, obj) { │ │ │ │ │ - var legend = { │ │ │ │ │ - width: node.getAttribute("width"), │ │ │ │ │ - height: node.getAttribute("height") │ │ │ │ │ - }; │ │ │ │ │ - obj.legend = legend; │ │ │ │ │ - this.readChildNodes(node, legend) │ │ │ │ │ - }, │ │ │ │ │ - MetadataURL: function(node, obj) { │ │ │ │ │ - var metadataURL = { │ │ │ │ │ - type: node.getAttribute("type") │ │ │ │ │ - }; │ │ │ │ │ - obj.metadataURLs.push(metadataURL); │ │ │ │ │ - this.readChildNodes(node, metadataURL) │ │ │ │ │ - }, │ │ │ │ │ - DataURL: function(node, obj) { │ │ │ │ │ - obj.dataURL = {}; │ │ │ │ │ - this.readChildNodes(node, obj.dataURL) │ │ │ │ │ - }, │ │ │ │ │ - FeatureListURL: function(node, obj) { │ │ │ │ │ - obj.featureListURL = {}; │ │ │ │ │ - this.readChildNodes(node, obj.featureListURL) │ │ │ │ │ - }, │ │ │ │ │ - AuthorityURL: function(node, obj) { │ │ │ │ │ - var name = node.getAttribute("name"); │ │ │ │ │ - var authority = {}; │ │ │ │ │ - this.readChildNodes(node, authority); │ │ │ │ │ - obj.authorityURLs[name] = authority.href │ │ │ │ │ - }, │ │ │ │ │ - Identifier: function(node, obj) { │ │ │ │ │ - var authority = node.getAttribute("authority"); │ │ │ │ │ - obj.identifiers[authority] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - KeywordList: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - SRS: function(node, obj) { │ │ │ │ │ - obj.srs[this.getChildValue(node)] = true │ │ │ │ │ + write_wmc_Layer: function(context) { │ │ │ │ │ + var node = OpenLayers.Format.WMC.v1.prototype.write_wmc_Layer.apply(this, [context]); │ │ │ │ │ + if (context.srs) { │ │ │ │ │ + var projections = []; │ │ │ │ │ + for (var name in context.srs) { │ │ │ │ │ + projections.push(name) │ │ │ │ │ } │ │ │ │ │ + node.appendChild(this.createElementDefaultNS("SRS", projections.join(" "))) │ │ │ │ │ } │ │ │ │ │ + node.appendChild(this.write_wmc_FormatList(context)); │ │ │ │ │ + node.appendChild(this.write_wmc_StyleList(context)); │ │ │ │ │ + if (context.dimensions) { │ │ │ │ │ + node.appendChild(this.write_wmc_DimensionList(context)) │ │ │ │ │ + } │ │ │ │ │ + node.appendChild(this.write_wmc_LayerExtension(context)) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_1 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1, { │ │ │ │ │ - readers: { │ │ │ │ │ - wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - WMT_MS_Capabilities: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - Keyword: function(node, obj) { │ │ │ │ │ - if (obj.keywords) { │ │ │ │ │ - obj.keywords.push(this.getChildValue(node)) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - DescribeLayer: function(node, obj) { │ │ │ │ │ - obj.describelayer = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.describelayer) │ │ │ │ │ - }, │ │ │ │ │ - GetLegendGraphic: function(node, obj) { │ │ │ │ │ - obj.getlegendgraphic = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getlegendgraphic) │ │ │ │ │ - }, │ │ │ │ │ - GetStyles: function(node, obj) { │ │ │ │ │ - obj.getstyles = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getstyles) │ │ │ │ │ - }, │ │ │ │ │ - PutStyles: function(node, obj) { │ │ │ │ │ - obj.putstyles = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.putstyles) │ │ │ │ │ - }, │ │ │ │ │ - UserDefinedSymbolization: function(node, obj) { │ │ │ │ │ - var userSymbols = { │ │ │ │ │ - supportSLD: parseInt(node.getAttribute("SupportSLD")) == 1, │ │ │ │ │ - userLayer: parseInt(node.getAttribute("UserLayer")) == 1, │ │ │ │ │ - userStyle: parseInt(node.getAttribute("UserStyle")) == 1, │ │ │ │ │ - remoteWFS: parseInt(node.getAttribute("RemoteWFS")) == 1 │ │ │ │ │ - }; │ │ │ │ │ - obj.userSymbols = userSymbols │ │ │ │ │ - }, │ │ │ │ │ - LatLonBoundingBox: function(node, obj) { │ │ │ │ │ - obj.llbbox = [parseFloat(node.getAttribute("minx")), parseFloat(node.getAttribute("miny")), parseFloat(node.getAttribute("maxx")), parseFloat(node.getAttribute("maxy"))] │ │ │ │ │ - }, │ │ │ │ │ - BoundingBox: function(node, obj) { │ │ │ │ │ - var bbox = OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"].BoundingBox.apply(this, [node, obj]); │ │ │ │ │ - bbox.srs = node.getAttribute("SRS"); │ │ │ │ │ - obj.bbox[bbox.srs] = bbox │ │ │ │ │ - }, │ │ │ │ │ - ScaleHint: function(node, obj) { │ │ │ │ │ - var min = node.getAttribute("min"); │ │ │ │ │ - var max = node.getAttribute("max"); │ │ │ │ │ - var rad2 = Math.pow(2, .5); │ │ │ │ │ - var ipm = OpenLayers.INCHES_PER_UNIT["m"]; │ │ │ │ │ - if (min != 0) { │ │ │ │ │ - obj.maxScale = parseFloat((min / rad2 * ipm * OpenLayers.DOTS_PER_INCH).toPrecision(13)) │ │ │ │ │ - } │ │ │ │ │ - if (max != Number.POSITIVE_INFINITY) { │ │ │ │ │ - obj.minScale = parseFloat((max / rad2 * ipm * OpenLayers.DOTS_PER_INCH).toPrecision(13)) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - Dimension: function(node, obj) { │ │ │ │ │ - var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ - var dim = { │ │ │ │ │ - name: name, │ │ │ │ │ - units: node.getAttribute("units"), │ │ │ │ │ - unitsymbol: node.getAttribute("unitSymbol") │ │ │ │ │ - }; │ │ │ │ │ - obj.dimensions[dim.name] = dim │ │ │ │ │ - }, │ │ │ │ │ - Extent: function(node, obj) { │ │ │ │ │ - var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ - if (name in obj["dimensions"]) { │ │ │ │ │ - var extent = obj.dimensions[name]; │ │ │ │ │ - extent.nearestVal = node.getAttribute("nearestValue") === "1"; │ │ │ │ │ - extent.multipleVal = node.getAttribute("multipleValues") === "1"; │ │ │ │ │ - extent.current = node.getAttribute("current") === "1"; │ │ │ │ │ - extent["default"] = node.getAttribute("default") || ""; │ │ │ │ │ - var values = this.getChildValue(node); │ │ │ │ │ - extent.values = values.split(",") │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"]) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMC.v1_0_0" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_1_1 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1_1, { │ │ │ │ │ - version: "1.1.1", │ │ │ │ │ - readers: { │ │ │ │ │ - wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - SRS: function(node, obj) { │ │ │ │ │ - obj.srs[this.getChildValue(node)] = true │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1_1.prototype.readers["wms"]) │ │ │ │ │ +OpenLayers.Events.buttonclick = OpenLayers.Class({ │ │ │ │ │ + target: null, │ │ │ │ │ + events: ["mousedown", "mouseup", "click", "dblclick", "touchstart", "touchmove", "touchend", "keydown"], │ │ │ │ │ + startRegEx: /^mousedown|touchstart$/, │ │ │ │ │ + cancelRegEx: /^touchmove$/, │ │ │ │ │ + completeRegEx: /^mouseup|touchend$/, │ │ │ │ │ + initialize: function(target) { │ │ │ │ │ + this.target = target; │ │ │ │ │ + for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ + this.target.register(this.events[i], this, this.buttonClick, { │ │ │ │ │ + extension: true │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_1" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_3 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1, { │ │ │ │ │ - readers: { │ │ │ │ │ - wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - WMS_Capabilities: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - LayerLimit: function(node, obj) { │ │ │ │ │ - obj.layerLimit = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - MaxWidth: function(node, obj) { │ │ │ │ │ - obj.maxWidth = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - MaxHeight: function(node, obj) { │ │ │ │ │ - obj.maxHeight = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - BoundingBox: function(node, obj) { │ │ │ │ │ - var bbox = OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"].BoundingBox.apply(this, [node, obj]); │ │ │ │ │ - bbox.srs = node.getAttribute("CRS"); │ │ │ │ │ - obj.bbox[bbox.srs] = bbox │ │ │ │ │ - }, │ │ │ │ │ - CRS: function(node, obj) { │ │ │ │ │ - this.readers.wms.SRS.apply(this, [node, obj]) │ │ │ │ │ - }, │ │ │ │ │ - EX_GeographicBoundingBox: function(node, obj) { │ │ │ │ │ - obj.llbbox = []; │ │ │ │ │ - this.readChildNodes(node, obj.llbbox) │ │ │ │ │ - }, │ │ │ │ │ - westBoundLongitude: function(node, obj) { │ │ │ │ │ - obj[0] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - eastBoundLongitude: function(node, obj) { │ │ │ │ │ - obj[2] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - southBoundLatitude: function(node, obj) { │ │ │ │ │ - obj[1] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - northBoundLatitude: function(node, obj) { │ │ │ │ │ - obj[3] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - MinScaleDenominator: function(node, obj) { │ │ │ │ │ - obj.maxScale = parseFloat(this.getChildValue(node)).toPrecision(16) │ │ │ │ │ - }, │ │ │ │ │ - MaxScaleDenominator: function(node, obj) { │ │ │ │ │ - obj.minScale = parseFloat(this.getChildValue(node)).toPrecision(16) │ │ │ │ │ - }, │ │ │ │ │ - Dimension: function(node, obj) { │ │ │ │ │ - var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ - var dim = { │ │ │ │ │ - name: name, │ │ │ │ │ - units: node.getAttribute("units"), │ │ │ │ │ - unitsymbol: node.getAttribute("unitSymbol"), │ │ │ │ │ - nearestVal: node.getAttribute("nearestValue") === "1", │ │ │ │ │ - multipleVal: node.getAttribute("multipleValues") === "1", │ │ │ │ │ - default: node.getAttribute("default") || "", │ │ │ │ │ - current: node.getAttribute("current") === "1", │ │ │ │ │ - values: this.getChildValue(node).split(",") │ │ │ │ │ - }; │ │ │ │ │ - obj.dimensions[dim.name] = dim │ │ │ │ │ - }, │ │ │ │ │ - Keyword: function(node, obj) { │ │ │ │ │ - var keyword = { │ │ │ │ │ - value: this.getChildValue(node), │ │ │ │ │ - vocabulary: node.getAttribute("vocabulary") │ │ │ │ │ - }; │ │ │ │ │ - if (obj.keywords) { │ │ │ │ │ - obj.keywords.push(keyword) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"]), │ │ │ │ │ - sld: { │ │ │ │ │ - UserDefinedSymbolization: function(node, obj) { │ │ │ │ │ - this.readers.wms.UserDefinedSymbolization.apply(this, [node, obj]); │ │ │ │ │ - obj.userSymbols.inlineFeature = parseInt(node.getAttribute("InlineFeature")) == 1; │ │ │ │ │ - obj.userSymbols.remoteWCS = parseInt(node.getAttribute("RemoteWCS")) == 1 │ │ │ │ │ - }, │ │ │ │ │ - DescribeLayer: function(node, obj) { │ │ │ │ │ - this.readers.wms.DescribeLayer.apply(this, [node, obj]) │ │ │ │ │ - }, │ │ │ │ │ - GetLegendGraphic: function(node, obj) { │ │ │ │ │ - this.readers.wms.GetLegendGraphic.apply(this, [node, obj]) │ │ │ │ │ - } │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ + this.target.unregister(this.events[i], this, this.buttonClick) │ │ │ │ │ } │ │ │ │ │ + delete this.target │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_3" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_3_0 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1_3, { │ │ │ │ │ - version: "1.3.0", │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_3_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_1_1_WMSC = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1_1_1, { │ │ │ │ │ - version: "1.1.1", │ │ │ │ │ - profile: "WMSC", │ │ │ │ │ - readers: { │ │ │ │ │ - wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - VendorSpecificCapabilities: function(node, obj) { │ │ │ │ │ - obj.vendorSpecific = { │ │ │ │ │ - tileSets: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.vendorSpecific) │ │ │ │ │ - }, │ │ │ │ │ - TileSet: function(node, vendorSpecific) { │ │ │ │ │ - var tileset = { │ │ │ │ │ - srs: {}, │ │ │ │ │ - bbox: {}, │ │ │ │ │ - resolutions: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, tileset); │ │ │ │ │ - vendorSpecific.tileSets.push(tileset) │ │ │ │ │ - }, │ │ │ │ │ - Resolutions: function(node, tileset) { │ │ │ │ │ - var res = this.getChildValue(node).split(" "); │ │ │ │ │ - for (var i = 0, len = res.length; i < len; i++) { │ │ │ │ │ - if (res[i] != "") { │ │ │ │ │ - tileset.resolutions.push(parseFloat(res[i])) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - Width: function(node, tileset) { │ │ │ │ │ - tileset.width = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - Height: function(node, tileset) { │ │ │ │ │ - tileset.height = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - Layers: function(node, tileset) { │ │ │ │ │ - tileset.layers = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Styles: function(node, tileset) { │ │ │ │ │ - tileset.styles = this.getChildValue(node) │ │ │ │ │ + getPressedButton: function(element) { │ │ │ │ │ + var depth = 3, │ │ │ │ │ + button; │ │ │ │ │ + do { │ │ │ │ │ + if (OpenLayers.Element.hasClass(element, "olButton")) { │ │ │ │ │ + button = element; │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1_1_1.prototype.readers["wms"]) │ │ │ │ │ + element = element.parentNode │ │ │ │ │ + } while (--depth > 0 && element); │ │ │ │ │ + return button │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_1_WMSC" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_1_0 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1_1, { │ │ │ │ │ - version: "1.1.0", │ │ │ │ │ - readers: { │ │ │ │ │ - wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - SRS: function(node, obj) { │ │ │ │ │ - var srs = this.getChildValue(node); │ │ │ │ │ - var values = srs.split(/ +/); │ │ │ │ │ - for (var i = 0, len = values.length; i < len; i++) { │ │ │ │ │ - obj.srs[values[i]] = true │ │ │ │ │ - } │ │ │ │ │ + ignore: function(element) { │ │ │ │ │ + var depth = 3, │ │ │ │ │ + ignore = false; │ │ │ │ │ + do { │ │ │ │ │ + if (element.nodeName.toLowerCase() === "a") { │ │ │ │ │ + ignore = true; │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1_1.prototype.readers["wms"]) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WFSCapabilities.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - namespaces: { │ │ │ │ │ - wfs: "http://www.opengis.net/wfs", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - ows: "http://www.opengis.net/ows" │ │ │ │ │ - }, │ │ │ │ │ - errorProperty: "featureTypeList", │ │ │ │ │ - defaultPrefix: "wfs", │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ - } │ │ │ │ │ - var raw = data; │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement │ │ │ │ │ - } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - return capabilities │ │ │ │ │ + element = element.parentNode │ │ │ │ │ + } while (--depth > 0 && element); │ │ │ │ │ + return ignore │ │ │ │ │ }, │ │ │ │ │ - readers: { │ │ │ │ │ - wfs: { │ │ │ │ │ - WFS_Capabilities: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - FeatureTypeList: function(node, request) { │ │ │ │ │ - request.featureTypeList = { │ │ │ │ │ - featureTypes: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, request.featureTypeList) │ │ │ │ │ - }, │ │ │ │ │ - FeatureType: function(node, featureTypeList) { │ │ │ │ │ - var featureType = {}; │ │ │ │ │ - this.readChildNodes(node, featureType); │ │ │ │ │ - featureTypeList.featureTypes.push(featureType) │ │ │ │ │ - }, │ │ │ │ │ - Name: function(node, obj) { │ │ │ │ │ - var name = this.getChildValue(node); │ │ │ │ │ - if (name) { │ │ │ │ │ - var parts = name.split(":"); │ │ │ │ │ - obj.name = parts.pop(); │ │ │ │ │ - if (parts.length > 0) { │ │ │ │ │ - obj.featureNS = this.lookupNamespaceURI(node, parts[0]) │ │ │ │ │ + buttonClick: function(evt) { │ │ │ │ │ + var propagate = true, │ │ │ │ │ + element = OpenLayers.Event.element(evt); │ │ │ │ │ + if (element && (OpenLayers.Event.isLeftClick(evt) || !~evt.type.indexOf("mouse"))) { │ │ │ │ │ + var button = this.getPressedButton(element); │ │ │ │ │ + if (button) { │ │ │ │ │ + if (evt.type === "keydown") { │ │ │ │ │ + switch (evt.keyCode) { │ │ │ │ │ + case OpenLayers.Event.KEY_RETURN: │ │ │ │ │ + case OpenLayers.Event.KEY_SPACE: │ │ │ │ │ + this.target.triggerEvent("buttonclick", { │ │ │ │ │ + buttonElement: button │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false; │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - Title: function(node, obj) { │ │ │ │ │ - var title = this.getChildValue(node); │ │ │ │ │ - if (title) { │ │ │ │ │ - obj.title = title │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - Abstract: function(node, obj) { │ │ │ │ │ - var abst = this.getChildValue(node); │ │ │ │ │ - if (abst) { │ │ │ │ │ - obj["abstract"] = abst │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WFSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.WFSCapabilities.v1, { │ │ │ │ │ - readers: { │ │ │ │ │ - wfs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Service: function(node, capabilities) { │ │ │ │ │ - capabilities.service = {}; │ │ │ │ │ - this.readChildNodes(node, capabilities.service) │ │ │ │ │ - }, │ │ │ │ │ - Fees: function(node, service) { │ │ │ │ │ - var fees = this.getChildValue(node); │ │ │ │ │ - if (fees && fees.toLowerCase() != "none") { │ │ │ │ │ - service.fees = fees │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - AccessConstraints: function(node, service) { │ │ │ │ │ - var constraints = this.getChildValue(node); │ │ │ │ │ - if (constraints && constraints.toLowerCase() != "none") { │ │ │ │ │ - service.accessConstraints = constraints │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - OnlineResource: function(node, service) { │ │ │ │ │ - var onlineResource = this.getChildValue(node); │ │ │ │ │ - if (onlineResource && onlineResource.toLowerCase() != "none") { │ │ │ │ │ - service.onlineResource = onlineResource │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - Keywords: function(node, service) { │ │ │ │ │ - var keywords = this.getChildValue(node); │ │ │ │ │ - if (keywords && keywords.toLowerCase() != "none") { │ │ │ │ │ - service.keywords = keywords.split(", ") │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - Capability: function(node, capabilities) { │ │ │ │ │ - capabilities.capability = {}; │ │ │ │ │ - this.readChildNodes(node, capabilities.capability) │ │ │ │ │ - }, │ │ │ │ │ - Request: function(node, obj) { │ │ │ │ │ - obj.request = {}; │ │ │ │ │ - this.readChildNodes(node, obj.request) │ │ │ │ │ - }, │ │ │ │ │ - GetFeature: function(node, request) { │ │ │ │ │ - request.getfeature = { │ │ │ │ │ - href: {}, │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, request.getfeature) │ │ │ │ │ - }, │ │ │ │ │ - ResultFormat: function(node, obj) { │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - var childNode; │ │ │ │ │ - for (var i = 0; i < children.length; i++) { │ │ │ │ │ - childNode = children[i]; │ │ │ │ │ - if (childNode.nodeType == 1) { │ │ │ │ │ - obj.formats.push(childNode.nodeName) │ │ │ │ │ + } else if (this.startEvt) { │ │ │ │ │ + if (this.completeRegEx.test(evt.type)) { │ │ │ │ │ + var pos = OpenLayers.Util.pagePosition(button); │ │ │ │ │ + var viewportElement = OpenLayers.Util.getViewportElement(); │ │ │ │ │ + var scrollTop = window.pageYOffset || viewportElement.scrollTop; │ │ │ │ │ + var scrollLeft = window.pageXOffset || viewportElement.scrollLeft; │ │ │ │ │ + pos[0] = pos[0] - scrollLeft; │ │ │ │ │ + pos[1] = pos[1] - scrollTop; │ │ │ │ │ + this.target.triggerEvent("buttonclick", { │ │ │ │ │ + buttonElement: button, │ │ │ │ │ + buttonXY: { │ │ │ │ │ + x: this.startEvt.clientX - pos[0], │ │ │ │ │ + y: this.startEvt.clientY - pos[1] │ │ │ │ │ + } │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ + if (this.cancelRegEx.test(evt.type)) { │ │ │ │ │ + delete this.startEvt │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - DCPType: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - HTTP: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj.href) │ │ │ │ │ - }, │ │ │ │ │ - Get: function(node, obj) { │ │ │ │ │ - obj.get = node.getAttribute("onlineResource") │ │ │ │ │ - }, │ │ │ │ │ - Post: function(node, obj) { │ │ │ │ │ - obj.post = node.getAttribute("onlineResource") │ │ │ │ │ - }, │ │ │ │ │ - SRS: function(node, obj) { │ │ │ │ │ - var srs = this.getChildValue(node); │ │ │ │ │ - if (srs) { │ │ │ │ │ - obj.srs = srs │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WFSCapabilities.v1.prototype.readers["wfs"]) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1_0_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WFSCapabilities.v1_1_0 = OpenLayers.Class(OpenLayers.Format.WFSCapabilities.v1, { │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: /^\s*|\s*$/g, │ │ │ │ │ - removeSpace: /\s*/g, │ │ │ │ │ - splitSpace: /\s+/, │ │ │ │ │ - trimComma: /\s*,\s*/g │ │ │ │ │ - }, │ │ │ │ │ - readers: { │ │ │ │ │ - wfs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - DefaultSRS: function(node, obj) { │ │ │ │ │ - var defaultSRS = this.getChildValue(node); │ │ │ │ │ - if (defaultSRS) { │ │ │ │ │ - obj.srs = defaultSRS │ │ │ │ │ + if (this.startRegEx.test(evt.type)) { │ │ │ │ │ + this.startEvt = evt; │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + propagate = !this.ignore(OpenLayers.Event.element(evt)); │ │ │ │ │ + delete this.startEvt │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.WFSCapabilities.v1.prototype.readers["wfs"]), │ │ │ │ │ - ows: OpenLayers.Format.OWSCommon.v1.prototype.readers.ows │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1_1_0" │ │ │ │ │ + } │ │ │ │ │ + return propagate │ │ │ │ │ + } │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Events.featureclick = OpenLayers.Class({ │ │ │ │ │ cache: null, │ │ │ │ │ map: null, │ │ │ │ │ provides: ["featureclick", "nofeatureclick", "featureover", "featureout"], │ │ │ │ │ initialize: function(target) { │ │ │ │ │ this.target = target; │ │ │ │ │ @@ -26567,3042 +27332,1344 @@ │ │ │ │ │ delete this.map; │ │ │ │ │ delete this.target │ │ │ │ │ } │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Events.nofeatureclick = OpenLayers.Events.featureclick; │ │ │ │ │ OpenLayers.Events.featureover = OpenLayers.Events.featureclick; │ │ │ │ │ OpenLayers.Events.featureout = OpenLayers.Events.featureclick; │ │ │ │ │ -OpenLayers.Events.buttonclick = OpenLayers.Class({ │ │ │ │ │ - target: null, │ │ │ │ │ - events: ["mousedown", "mouseup", "click", "dblclick", "touchstart", "touchmove", "touchend", "keydown"], │ │ │ │ │ - startRegEx: /^mousedown|touchstart$/, │ │ │ │ │ - cancelRegEx: /^touchmove$/, │ │ │ │ │ - completeRegEx: /^mouseup|touchend$/, │ │ │ │ │ - initialize: function(target) { │ │ │ │ │ - this.target = target; │ │ │ │ │ - for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ - this.target.register(this.events[i], this, this.buttonClick, { │ │ │ │ │ - extension: true │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ - this.target.unregister(this.events[i], this, this.buttonClick) │ │ │ │ │ - } │ │ │ │ │ - delete this.target │ │ │ │ │ +OpenLayers.Control.ZoomBox = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ + out: false, │ │ │ │ │ + keyMask: null, │ │ │ │ │ + alwaysZoom: false, │ │ │ │ │ + zoomOnClick: true, │ │ │ │ │ + draw: function() { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Box(this, { │ │ │ │ │ + done: this.zoomBox │ │ │ │ │ + }, { │ │ │ │ │ + keyMask: this.keyMask │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - getPressedButton: function(element) { │ │ │ │ │ - var depth = 3, │ │ │ │ │ - button; │ │ │ │ │ - do { │ │ │ │ │ - if (OpenLayers.Element.hasClass(element, "olButton")) { │ │ │ │ │ - button = element; │ │ │ │ │ - break │ │ │ │ │ + zoomBox: function(position) { │ │ │ │ │ + if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ + var bounds, targetCenterPx = position.getCenterPixel(); │ │ │ │ │ + if (!this.out) { │ │ │ │ │ + var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.left, │ │ │ │ │ + y: position.bottom │ │ │ │ │ + }); │ │ │ │ │ + var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.right, │ │ │ │ │ + y: position.top │ │ │ │ │ + }); │ │ │ │ │ + bounds = new OpenLayers.Bounds(minXY.lon, minXY.lat, maxXY.lon, maxXY.lat) │ │ │ │ │ + } else { │ │ │ │ │ + var pixWidth = position.right - position.left; │ │ │ │ │ + var pixHeight = position.bottom - position.top; │ │ │ │ │ + var zoomFactor = Math.min(this.map.size.h / pixHeight, this.map.size.w / pixWidth); │ │ │ │ │ + var extent = this.map.getExtent(); │ │ │ │ │ + var center = this.map.getLonLatFromPixel(targetCenterPx); │ │ │ │ │ + var xmin = center.lon - extent.getWidth() / 2 * zoomFactor; │ │ │ │ │ + var xmax = center.lon + extent.getWidth() / 2 * zoomFactor; │ │ │ │ │ + var ymin = center.lat - extent.getHeight() / 2 * zoomFactor; │ │ │ │ │ + var ymax = center.lat + extent.getHeight() / 2 * zoomFactor; │ │ │ │ │ + bounds = new OpenLayers.Bounds(xmin, ymin, xmax, ymax) │ │ │ │ │ } │ │ │ │ │ - element = element.parentNode │ │ │ │ │ - } while (--depth > 0 && element); │ │ │ │ │ - return button │ │ │ │ │ - }, │ │ │ │ │ - ignore: function(element) { │ │ │ │ │ - var depth = 3, │ │ │ │ │ - ignore = false; │ │ │ │ │ - do { │ │ │ │ │ - if (element.nodeName.toLowerCase() === "a") { │ │ │ │ │ - ignore = true; │ │ │ │ │ - break │ │ │ │ │ + var lastZoom = this.map.getZoom(), │ │ │ │ │ + size = this.map.getSize(), │ │ │ │ │ + centerPx = { │ │ │ │ │ + x: size.w / 2, │ │ │ │ │ + y: size.h / 2 │ │ │ │ │ + }, │ │ │ │ │ + zoom = this.map.getZoomForExtent(bounds), │ │ │ │ │ + oldRes = this.map.getResolution(), │ │ │ │ │ + newRes = this.map.getResolutionForZoom(zoom); │ │ │ │ │ + if (oldRes == newRes) { │ │ │ │ │ + this.map.setCenter(this.map.getLonLatFromPixel(targetCenterPx)) │ │ │ │ │ + } else { │ │ │ │ │ + var zoomOriginPx = { │ │ │ │ │ + x: (oldRes * targetCenterPx.x - newRes * centerPx.x) / (oldRes - newRes), │ │ │ │ │ + y: (oldRes * targetCenterPx.y - newRes * centerPx.y) / (oldRes - newRes) │ │ │ │ │ + }; │ │ │ │ │ + this.map.zoomTo(zoom, zoomOriginPx) │ │ │ │ │ } │ │ │ │ │ - element = element.parentNode │ │ │ │ │ - } while (--depth > 0 && element); │ │ │ │ │ - return ignore │ │ │ │ │ - }, │ │ │ │ │ - buttonClick: function(evt) { │ │ │ │ │ - var propagate = true, │ │ │ │ │ - element = OpenLayers.Event.element(evt); │ │ │ │ │ - if (element && (OpenLayers.Event.isLeftClick(evt) || !~evt.type.indexOf("mouse"))) { │ │ │ │ │ - var button = this.getPressedButton(element); │ │ │ │ │ - if (button) { │ │ │ │ │ - if (evt.type === "keydown") { │ │ │ │ │ - switch (evt.keyCode) { │ │ │ │ │ - case OpenLayers.Event.KEY_RETURN: │ │ │ │ │ - case OpenLayers.Event.KEY_SPACE: │ │ │ │ │ - this.target.triggerEvent("buttonclick", { │ │ │ │ │ - buttonElement: button │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } else if (this.startEvt) { │ │ │ │ │ - if (this.completeRegEx.test(evt.type)) { │ │ │ │ │ - var pos = OpenLayers.Util.pagePosition(button); │ │ │ │ │ - var viewportElement = OpenLayers.Util.getViewportElement(); │ │ │ │ │ - var scrollTop = window.pageYOffset || viewportElement.scrollTop; │ │ │ │ │ - var scrollLeft = window.pageXOffset || viewportElement.scrollLeft; │ │ │ │ │ - pos[0] = pos[0] - scrollLeft; │ │ │ │ │ - pos[1] = pos[1] - scrollTop; │ │ │ │ │ - this.target.triggerEvent("buttonclick", { │ │ │ │ │ - buttonElement: button, │ │ │ │ │ - buttonXY: { │ │ │ │ │ - x: this.startEvt.clientX - pos[0], │ │ │ │ │ - y: this.startEvt.clientY - pos[1] │ │ │ │ │ - } │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - if (this.cancelRegEx.test(evt.type)) { │ │ │ │ │ - delete this.startEvt │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false │ │ │ │ │ - } │ │ │ │ │ - if (this.startRegEx.test(evt.type)) { │ │ │ │ │ - this.startEvt = evt; │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false │ │ │ │ │ - } │ │ │ │ │ + if (lastZoom == this.map.getZoom() && this.alwaysZoom == true) { │ │ │ │ │ + this.map.zoomTo(lastZoom + (this.out ? -1 : 1)) │ │ │ │ │ + } │ │ │ │ │ + } else if (this.zoomOnClick) { │ │ │ │ │ + if (!this.out) { │ │ │ │ │ + this.map.zoomTo(this.map.getZoom() + 1, position) │ │ │ │ │ } else { │ │ │ │ │ - propagate = !this.ignore(OpenLayers.Event.element(evt)); │ │ │ │ │ - delete this.startEvt │ │ │ │ │ + this.map.zoomTo(this.map.getZoom() - 1, position) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return propagate │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomBox" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Handler.Point = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - point: null, │ │ │ │ │ - layer: null, │ │ │ │ │ - multi: false, │ │ │ │ │ - citeCompliant: false, │ │ │ │ │ - mouseDown: false, │ │ │ │ │ - stoppedDown: null, │ │ │ │ │ - lastDown: null, │ │ │ │ │ - lastUp: null, │ │ │ │ │ - persist: false, │ │ │ │ │ - stopDown: false, │ │ │ │ │ - stopUp: false, │ │ │ │ │ - layerOptions: null, │ │ │ │ │ - pixelTolerance: 5, │ │ │ │ │ - lastTouchPx: null, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ - this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style["default"], {}) │ │ │ │ │ +OpenLayers.Control.DragPan = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ + panned: false, │ │ │ │ │ + interval: 0, │ │ │ │ │ + documentDrag: false, │ │ │ │ │ + kinetic: null, │ │ │ │ │ + enableKinetic: true, │ │ │ │ │ + kineticInterval: 10, │ │ │ │ │ + draw: function() { │ │ │ │ │ + if (this.enableKinetic && OpenLayers.Kinetic) { │ │ │ │ │ + var config = { │ │ │ │ │ + interval: this.kineticInterval │ │ │ │ │ + }; │ │ │ │ │ + if (typeof this.enableKinetic === "object") { │ │ │ │ │ + config = OpenLayers.Util.extend(config, this.enableKinetic) │ │ │ │ │ + } │ │ │ │ │ + this.kinetic = new OpenLayers.Kinetic(config) │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments) │ │ │ │ │ + this.handler = new OpenLayers.Handler.Drag(this, { │ │ │ │ │ + move: this.panMap, │ │ │ │ │ + done: this.panMapDone, │ │ │ │ │ + down: this.panMapStart │ │ │ │ │ + }, { │ │ │ │ │ + interval: this.interval, │ │ │ │ │ + documentDrag: this.documentDrag │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (!OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - return false │ │ │ │ │ + panMapStart: function() { │ │ │ │ │ + if (this.kinetic) { │ │ │ │ │ + this.kinetic.begin() │ │ │ │ │ } │ │ │ │ │ - var options = OpenLayers.Util.extend({ │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - calculateInRange: OpenLayers.Function.True, │ │ │ │ │ - wrapDateLine: this.citeCompliant │ │ │ │ │ - }, this.layerOptions); │ │ │ │ │ - this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ - this.map.addLayer(this.layer); │ │ │ │ │ - return true │ │ │ │ │ }, │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.point]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.point], { │ │ │ │ │ - silent: true │ │ │ │ │ + panMap: function(xy) { │ │ │ │ │ + if (this.kinetic) { │ │ │ │ │ + this.kinetic.update(xy) │ │ │ │ │ + } │ │ │ │ │ + this.panned = true; │ │ │ │ │ + this.map.pan(this.handler.last.x - xy.x, this.handler.last.y - xy.y, { │ │ │ │ │ + dragging: true, │ │ │ │ │ + animate: false │ │ │ │ │ }) │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (!OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - this.cancel(); │ │ │ │ │ - if (this.layer.map != null) { │ │ │ │ │ - this.destroyFeature(true); │ │ │ │ │ - this.layer.destroy(false) │ │ │ │ │ + panMapDone: function(xy) { │ │ │ │ │ + if (this.panned) { │ │ │ │ │ + var res = null; │ │ │ │ │ + if (this.kinetic) { │ │ │ │ │ + res = this.kinetic.end(xy) │ │ │ │ │ + } │ │ │ │ │ + this.map.pan(this.handler.last.x - xy.x, this.handler.last.y - xy.y, { │ │ │ │ │ + dragging: !!res, │ │ │ │ │ + animate: false │ │ │ │ │ + }); │ │ │ │ │ + if (res) { │ │ │ │ │ + var self = this; │ │ │ │ │ + this.kinetic.move(res, function(x, y, end) { │ │ │ │ │ + self.map.pan(x, y, { │ │ │ │ │ + dragging: !end, │ │ │ │ │ + animate: false │ │ │ │ │ + }) │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + this.panned = false │ │ │ │ │ } │ │ │ │ │ - this.layer = null; │ │ │ │ │ - return true │ │ │ │ │ }, │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - if (this.layer && (force || !this.persist)) { │ │ │ │ │ - this.layer.destroyFeatures() │ │ │ │ │ - } │ │ │ │ │ - this.point = null │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.DragPan" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Scale = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + element: null, │ │ │ │ │ + geodesic: false, │ │ │ │ │ + initialize: function(element, options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.element = OpenLayers.Util.getElement(element) │ │ │ │ │ }, │ │ │ │ │ - destroyPersistedFeature: function() { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - if (layer && layer.features.length > 1) { │ │ │ │ │ - this.layer.features[0].destroy() │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (!this.element) { │ │ │ │ │ + this.element = document.createElement("div"); │ │ │ │ │ + this.div.appendChild(this.element) │ │ │ │ │ } │ │ │ │ │ + this.map.events.register("moveend", this, this.updateScale); │ │ │ │ │ + this.updateScale(); │ │ │ │ │ + return this.div │ │ │ │ │ }, │ │ │ │ │ - finalize: function(cancel) { │ │ │ │ │ - var key = cancel ? "cancel" : "done"; │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - this.lastDown = null; │ │ │ │ │ - this.lastUp = null; │ │ │ │ │ - this.lastTouchPx = null; │ │ │ │ │ - this.callback(key, [this.geometryClone()]); │ │ │ │ │ - this.destroyFeature(cancel) │ │ │ │ │ + updateScale: function() { │ │ │ │ │ + var scale; │ │ │ │ │ + if (this.geodesic === true) { │ │ │ │ │ + var units = this.map.getUnits(); │ │ │ │ │ + if (!units) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var inches = OpenLayers.INCHES_PER_UNIT; │ │ │ │ │ + scale = (this.map.getGeodesicPixelSize().w || 1e-6) * inches["km"] * OpenLayers.DOTS_PER_INCH │ │ │ │ │ + } else { │ │ │ │ │ + scale = this.map.getScale() │ │ │ │ │ + } │ │ │ │ │ + if (!scale) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (scale >= 9500 && scale <= 95e4) { │ │ │ │ │ + scale = Math.round(scale / 1e3) + "K" │ │ │ │ │ + } else if (scale >= 95e4) { │ │ │ │ │ + scale = Math.round(scale / 1e6) + "M" │ │ │ │ │ + } else { │ │ │ │ │ + scale = Math.round(scale) │ │ │ │ │ + } │ │ │ │ │ + this.element.innerHTML = OpenLayers.i18n("Scale = 1 : ${scaleDenom}", { │ │ │ │ │ + scaleDenom: scale │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - cancel: function() { │ │ │ │ │ - this.finalize(true) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Scale" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.PinchZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ + pinchOrigin: null, │ │ │ │ │ + currentCenter: null, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + preserveCenter: false, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.handler = new OpenLayers.Handler.Pinch(this, { │ │ │ │ │ + start: this.pinchStart, │ │ │ │ │ + move: this.pinchMove, │ │ │ │ │ + done: this.pinchDone │ │ │ │ │ + }, this.handlerOptions) │ │ │ │ │ }, │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - return false │ │ │ │ │ + pinchStart: function(evt, pinchData) { │ │ │ │ │ + var xy = this.preserveCenter ? this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ + this.pinchOrigin = xy; │ │ │ │ │ + this.currentCenter = xy │ │ │ │ │ }, │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - return false │ │ │ │ │ + pinchMove: function(evt, pinchData) { │ │ │ │ │ + var scale = pinchData.scale; │ │ │ │ │ + var containerOrigin = this.map.layerContainerOriginPx; │ │ │ │ │ + var pinchOrigin = this.pinchOrigin; │ │ │ │ │ + var current = this.preserveCenter ? this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ + var dx = Math.round(containerOrigin.x + current.x - pinchOrigin.x + (scale - 1) * (containerOrigin.x - pinchOrigin.x)); │ │ │ │ │ + var dy = Math.round(containerOrigin.y + current.y - pinchOrigin.y + (scale - 1) * (containerOrigin.y - pinchOrigin.y)); │ │ │ │ │ + this.map.applyTransform(dx, dy, scale); │ │ │ │ │ + this.currentCenter = current │ │ │ │ │ }, │ │ │ │ │ - modifyFeature: function(pixel) { │ │ │ │ │ - if (!this.point) { │ │ │ │ │ - this.createFeature(pixel) │ │ │ │ │ + pinchDone: function(evt, start, last) { │ │ │ │ │ + this.map.applyTransform(); │ │ │ │ │ + var zoom = this.map.getZoomForResolution(this.map.getResolution() / last.scale, true); │ │ │ │ │ + if (zoom !== this.map.getZoom() || !this.currentCenter.equals(this.pinchOrigin)) { │ │ │ │ │ + var resolution = this.map.getResolutionForZoom(zoom); │ │ │ │ │ + var location = this.map.getLonLatFromPixel(this.pinchOrigin); │ │ │ │ │ + var zoomPixel = this.currentCenter; │ │ │ │ │ + var size = this.map.getSize(); │ │ │ │ │ + location.lon += resolution * (size.w / 2 - zoomPixel.x); │ │ │ │ │ + location.lat -= resolution * (size.h / 2 - zoomPixel.y); │ │ │ │ │ + this.map.div.clientWidth = this.map.div.clientWidth; │ │ │ │ │ + this.map.setCenter(location, zoom) │ │ │ │ │ } │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point.geometry.x = lonlat.lon; │ │ │ │ │ - this.point.geometry.y = lonlat.lat; │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.point, false]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.drawFeature() │ │ │ │ │ }, │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.point, this.style) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.PinchZoom" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.TouchNavigation = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + dragPan: null, │ │ │ │ │ + dragPanOptions: null, │ │ │ │ │ + pinchZoom: null, │ │ │ │ │ + pinchZoomOptions: null, │ │ │ │ │ + clickHandlerOptions: null, │ │ │ │ │ + documentDrag: false, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.handlers = {}; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.point && this.point.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiPoint([geometry]) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + if (this.dragPan) { │ │ │ │ │ + this.dragPan.destroy() │ │ │ │ │ } │ │ │ │ │ - return geometry │ │ │ │ │ - }, │ │ │ │ │ - geometryClone: function() { │ │ │ │ │ - var geom = this.getGeometry(); │ │ │ │ │ - return geom && geom.clone() │ │ │ │ │ + this.dragPan = null; │ │ │ │ │ + if (this.pinchZoom) { │ │ │ │ │ + this.pinchZoom.destroy(); │ │ │ │ │ + delete this.pinchZoom │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - return this.down(evt) │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.dragPan.activate(); │ │ │ │ │ + this.handlers.click.activate(); │ │ │ │ │ + this.pinchZoom.activate(); │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - this.lastTouchPx = evt.xy; │ │ │ │ │ - return this.down(evt) │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.dragPan.deactivate(); │ │ │ │ │ + this.handlers.click.deactivate(); │ │ │ │ │ + this.pinchZoom.deactivate(); │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - return this.move(evt) │ │ │ │ │ + draw: function() { │ │ │ │ │ + var clickCallbacks = { │ │ │ │ │ + click: this.defaultClick, │ │ │ │ │ + dblclick: this.defaultDblClick │ │ │ │ │ + }; │ │ │ │ │ + var clickOptions = OpenLayers.Util.extend({ │ │ │ │ │ + double: true, │ │ │ │ │ + stopDouble: true, │ │ │ │ │ + pixelTolerance: 2 │ │ │ │ │ + }, this.clickHandlerOptions); │ │ │ │ │ + this.handlers.click = new OpenLayers.Handler.Click(this, clickCallbacks, clickOptions); │ │ │ │ │ + this.dragPan = new OpenLayers.Control.DragPan(OpenLayers.Util.extend({ │ │ │ │ │ + map: this.map, │ │ │ │ │ + documentDrag: this.documentDrag │ │ │ │ │ + }, this.dragPanOptions)); │ │ │ │ │ + this.dragPan.draw(); │ │ │ │ │ + this.pinchZoom = new OpenLayers.Control.PinchZoom(OpenLayers.Util.extend({ │ │ │ │ │ + map: this.map │ │ │ │ │ + }, this.pinchZoomOptions)) │ │ │ │ │ }, │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - this.lastTouchPx = evt.xy; │ │ │ │ │ - return this.move(evt) │ │ │ │ │ + defaultClick: function(evt) { │ │ │ │ │ + if (evt.lastTouches && evt.lastTouches.length == 2) { │ │ │ │ │ + this.map.zoomOut() │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - return this.up(evt) │ │ │ │ │ + defaultDblClick: function(evt) { │ │ │ │ │ + this.map.zoomTo(this.map.zoom + 1, evt.xy) │ │ │ │ │ }, │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - evt.xy = this.lastTouchPx; │ │ │ │ │ - return this.up(evt) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.TouchNavigation" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.KeyboardDefaults = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + slideFactor: 75, │ │ │ │ │ + observeElement: null, │ │ │ │ │ + draw: function() { │ │ │ │ │ + var observeElement = this.observeElement || document; │ │ │ │ │ + this.handler = new OpenLayers.Handler.Keyboard(this, { │ │ │ │ │ + keydown: this.defaultKeyPress │ │ │ │ │ + }, { │ │ │ │ │ + observeElement: observeElement │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - down: function(evt) { │ │ │ │ │ - this.mouseDown = true; │ │ │ │ │ - this.lastDown = evt.xy; │ │ │ │ │ - if (!this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy) │ │ │ │ │ + defaultKeyPress: function(evt) { │ │ │ │ │ + var size, handled = true; │ │ │ │ │ + var target = OpenLayers.Event.element(evt); │ │ │ │ │ + if (target && (target.tagName == "INPUT" || target.tagName == "TEXTAREA" || target.tagName == "SELECT")) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - return !this.stopDown │ │ │ │ │ - }, │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ - this.modifyFeature(evt.xy) │ │ │ │ │ + switch (evt.keyCode) { │ │ │ │ │ + case OpenLayers.Event.KEY_LEFT: │ │ │ │ │ + this.map.pan(-this.slideFactor, 0); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Event.KEY_RIGHT: │ │ │ │ │ + this.map.pan(this.slideFactor, 0); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Event.KEY_UP: │ │ │ │ │ + this.map.pan(0, -this.slideFactor); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Event.KEY_DOWN: │ │ │ │ │ + this.map.pan(0, this.slideFactor); │ │ │ │ │ + break; │ │ │ │ │ + case 33: │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + this.map.pan(0, -.75 * size.h); │ │ │ │ │ + break; │ │ │ │ │ + case 34: │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + this.map.pan(0, .75 * size.h); │ │ │ │ │ + break; │ │ │ │ │ + case 35: │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + this.map.pan(.75 * size.w, 0); │ │ │ │ │ + break; │ │ │ │ │ + case 36: │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + this.map.pan(-.75 * size.w, 0); │ │ │ │ │ + break; │ │ │ │ │ + case 43: │ │ │ │ │ + case 61: │ │ │ │ │ + case 187: │ │ │ │ │ + case 107: │ │ │ │ │ + this.map.zoomIn(); │ │ │ │ │ + break; │ │ │ │ │ + case 45: │ │ │ │ │ + case 109: │ │ │ │ │ + case 189: │ │ │ │ │ + case 95: │ │ │ │ │ + this.map.zoomOut(); │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + handled = false │ │ │ │ │ + } │ │ │ │ │ + if (handled) { │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ } │ │ │ │ │ - return true │ │ │ │ │ }, │ │ │ │ │ - up: function(evt) { │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - if (!this.checkModifiers(evt)) { │ │ │ │ │ - return true │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.KeyboardDefaults" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.ArgParser = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + center: null, │ │ │ │ │ + zoom: null, │ │ │ │ │ + layers: null, │ │ │ │ │ + displayProjection: null, │ │ │ │ │ + getParameters: function(url) { │ │ │ │ │ + url = url || window.location.href; │ │ │ │ │ + var parameters = OpenLayers.Util.getParameters(url); │ │ │ │ │ + var index = url.indexOf("#"); │ │ │ │ │ + if (index > 0) { │ │ │ │ │ + url = "?" + url.substring(index + 1, url.length); │ │ │ │ │ + OpenLayers.Util.extend(parameters, OpenLayers.Util.getParameters(url)) │ │ │ │ │ } │ │ │ │ │ - if (this.lastUp && this.lastUp.equals(evt.xy)) { │ │ │ │ │ - return true │ │ │ │ │ + return parameters │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + for (var i = 0, len = this.map.controls.length; i < len; i++) { │ │ │ │ │ + var control = this.map.controls[i]; │ │ │ │ │ + if (control != this && control.CLASS_NAME == "OpenLayers.Control.ArgParser") { │ │ │ │ │ + if (control.displayProjection != this.displayProjection) { │ │ │ │ │ + this.displayProjection = control.displayProjection │ │ │ │ │ + } │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (this.lastDown && this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance)) { │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy) │ │ │ │ │ + if (i == this.map.controls.length) { │ │ │ │ │ + var args = this.getParameters(); │ │ │ │ │ + if (args.layers) { │ │ │ │ │ + this.layers = args.layers; │ │ │ │ │ + this.map.events.register("addlayer", this, this.configureLayers); │ │ │ │ │ + this.configureLayers() │ │ │ │ │ } │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature() │ │ │ │ │ + if (args.lat && args.lon) { │ │ │ │ │ + this.center = new OpenLayers.LonLat(parseFloat(args.lon), parseFloat(args.lat)); │ │ │ │ │ + if (args.zoom) { │ │ │ │ │ + this.zoom = parseFloat(args.zoom) │ │ │ │ │ + } │ │ │ │ │ + this.map.events.register("changebaselayer", this, this.setCenter); │ │ │ │ │ + this.setCenter() │ │ │ │ │ } │ │ │ │ │ - this.lastUp = evt.xy; │ │ │ │ │ - this.finalize(); │ │ │ │ │ - return !this.stopUp │ │ │ │ │ - } else { │ │ │ │ │ - return true │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - mouseout: function(evt) { │ │ │ │ │ - if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - this.mouseDown = false │ │ │ │ │ + setCenter: function() { │ │ │ │ │ + if (this.map.baseLayer) { │ │ │ │ │ + this.map.events.unregister("changebaselayer", this, this.setCenter); │ │ │ │ │ + if (this.displayProjection) { │ │ │ │ │ + this.center.transform(this.displayProjection, this.map.getProjectionObject()) │ │ │ │ │ + } │ │ │ │ │ + this.map.setCenter(this.center, this.zoom) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - passesTolerance: function(pixel1, pixel2, tolerance) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - if (tolerance != null && pixel1 && pixel2) { │ │ │ │ │ - var dist = pixel1.distanceTo(pixel2); │ │ │ │ │ - if (dist > tolerance) { │ │ │ │ │ - passes = false │ │ │ │ │ + configureLayers: function() { │ │ │ │ │ + if (this.layers.length == this.map.layers.length) { │ │ │ │ │ + this.map.events.unregister("addlayer", this, this.configureLayers); │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + var c = this.layers.charAt(i); │ │ │ │ │ + if (c == "B") { │ │ │ │ │ + this.map.setBaseLayer(layer) │ │ │ │ │ + } else if (c == "T" || c == "F") { │ │ │ │ │ + layer.setVisibility(c == "T") │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return passes │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Point" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ArgParser" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Handler.Feature = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - EVENTMAP: { │ │ │ │ │ - click: { │ │ │ │ │ - in: "click", │ │ │ │ │ - out: "clickout" │ │ │ │ │ - }, │ │ │ │ │ - mousemove: { │ │ │ │ │ - in: "over", │ │ │ │ │ - out: "out" │ │ │ │ │ - }, │ │ │ │ │ - dblclick: { │ │ │ │ │ - in: "dblclick", │ │ │ │ │ - out: null │ │ │ │ │ - }, │ │ │ │ │ - mousedown: { │ │ │ │ │ - in: null, │ │ │ │ │ - out: null │ │ │ │ │ - }, │ │ │ │ │ - mouseup: { │ │ │ │ │ - in: null, │ │ │ │ │ - out: null │ │ │ │ │ - }, │ │ │ │ │ - touchstart: { │ │ │ │ │ - in: "click", │ │ │ │ │ - out: "clickout" │ │ │ │ │ +OpenLayers.Control.WMTSGetFeatureInfo = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + hover: false, │ │ │ │ │ + requestEncoding: "KVP", │ │ │ │ │ + drillDown: false, │ │ │ │ │ + maxFeatures: 10, │ │ │ │ │ + clickCallback: "click", │ │ │ │ │ + layers: null, │ │ │ │ │ + queryVisible: true, │ │ │ │ │ + infoFormat: "text/html", │ │ │ │ │ + vendorParams: {}, │ │ │ │ │ + format: null, │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + handler: null, │ │ │ │ │ + hoverRequest: null, │ │ │ │ │ + pending: 0, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!this.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.WMSGetFeatureInfo(options.formatOptions) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - feature: null, │ │ │ │ │ - lastFeature: null, │ │ │ │ │ - down: null, │ │ │ │ │ - up: null, │ │ │ │ │ - clickTolerance: 4, │ │ │ │ │ - geometryTypes: null, │ │ │ │ │ - stopClick: true, │ │ │ │ │ - stopDown: true, │ │ │ │ │ - stopUp: false, │ │ │ │ │ - initialize: function(control, layer, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ - this.layer = layer │ │ │ │ │ - }, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - return OpenLayers.Event.isMultiTouch(evt) ? true : this.mousedown(evt) │ │ │ │ │ - }, │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - OpenLayers.Event.preventDefault(evt) │ │ │ │ │ - }, │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - if (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ - this.down = evt.xy │ │ │ │ │ + if (this.drillDown === true) { │ │ │ │ │ + this.hover = false │ │ │ │ │ } │ │ │ │ │ - return this.handle(evt) ? !this.stopDown : true │ │ │ │ │ - }, │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - this.up = evt.xy; │ │ │ │ │ - return this.handle(evt) ? !this.stopUp : true │ │ │ │ │ - }, │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - return this.handle(evt) ? !this.stopClick : true │ │ │ │ │ - }, │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - if (!this.callbacks["over"] && !this.callbacks["out"]) { │ │ │ │ │ - return true │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ + move: this.cancelHover, │ │ │ │ │ + pause: this.getInfoForHover │ │ │ │ │ + }, OpenLayers.Util.extend(this.handlerOptions.hover || {}, { │ │ │ │ │ + delay: 250 │ │ │ │ │ + })) │ │ │ │ │ + } else { │ │ │ │ │ + var callbacks = {}; │ │ │ │ │ + callbacks[this.clickCallback] = this.getInfoForClick; │ │ │ │ │ + this.handler = new OpenLayers.Handler.Click(this, callbacks, this.handlerOptions.click || {}) │ │ │ │ │ } │ │ │ │ │ - this.handle(evt); │ │ │ │ │ - return true │ │ │ │ │ }, │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - return !this.handle(evt) │ │ │ │ │ + getInfoForClick: function(evt) { │ │ │ │ │ + this.request(evt.xy, {}) │ │ │ │ │ }, │ │ │ │ │ - geometryTypeMatches: function(feature) { │ │ │ │ │ - return this.geometryTypes == null || OpenLayers.Util.indexOf(this.geometryTypes, feature.geometry.CLASS_NAME) > -1 │ │ │ │ │ + getInfoForHover: function(evt) { │ │ │ │ │ + this.request(evt.xy, { │ │ │ │ │ + hover: true │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - handle: function(evt) { │ │ │ │ │ - if (this.feature && !this.feature.layer) { │ │ │ │ │ - this.feature = null │ │ │ │ │ - } │ │ │ │ │ - var type = evt.type; │ │ │ │ │ - var handled = false; │ │ │ │ │ - var previouslyIn = !!this.feature; │ │ │ │ │ - var click = type == "click" || type == "dblclick" || type == "touchstart"; │ │ │ │ │ - this.feature = this.layer.getFeatureFromEvent(evt); │ │ │ │ │ - if (this.feature && !this.feature.layer) { │ │ │ │ │ - this.feature = null │ │ │ │ │ - } │ │ │ │ │ - if (this.lastFeature && !this.lastFeature.layer) { │ │ │ │ │ - this.lastFeature = null │ │ │ │ │ - } │ │ │ │ │ - if (this.feature) { │ │ │ │ │ - if (type === "touchstart") { │ │ │ │ │ - OpenLayers.Event.preventDefault(evt) │ │ │ │ │ - } │ │ │ │ │ - var inNew = this.feature != this.lastFeature; │ │ │ │ │ - if (this.geometryTypeMatches(this.feature)) { │ │ │ │ │ - if (previouslyIn && inNew) { │ │ │ │ │ - if (this.lastFeature) { │ │ │ │ │ - this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ - } │ │ │ │ │ - this.triggerCallback(type, "in", [this.feature]) │ │ │ │ │ - } else if (!previouslyIn || click) { │ │ │ │ │ - this.triggerCallback(type, "in", [this.feature]) │ │ │ │ │ - } │ │ │ │ │ - this.lastFeature = this.feature; │ │ │ │ │ - handled = true │ │ │ │ │ - } else { │ │ │ │ │ - if (this.lastFeature && (previouslyIn && inNew || click)) { │ │ │ │ │ - this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ - } │ │ │ │ │ - this.feature = null │ │ │ │ │ + cancelHover: function() { │ │ │ │ │ + if (this.hoverRequest) { │ │ │ │ │ + --this.pending; │ │ │ │ │ + if (this.pending <= 0) { │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + this.pending = 0 │ │ │ │ │ } │ │ │ │ │ - } else if (this.lastFeature && (previouslyIn || click)) { │ │ │ │ │ - this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ + this.hoverRequest.abort(); │ │ │ │ │ + this.hoverRequest = null │ │ │ │ │ } │ │ │ │ │ - return handled │ │ │ │ │ }, │ │ │ │ │ - triggerCallback: function(type, mode, args) { │ │ │ │ │ - var key = this.EVENTMAP[type][mode]; │ │ │ │ │ - if (key) { │ │ │ │ │ - if (type == "click" && this.up && this.down) { │ │ │ │ │ - var dpx = Math.sqrt(Math.pow(this.up.x - this.down.x, 2) + Math.pow(this.up.y - this.down.y, 2)); │ │ │ │ │ - if (dpx <= this.clickTolerance) { │ │ │ │ │ - this.callback(key, args) │ │ │ │ │ + findLayers: function() { │ │ │ │ │ + var candidates = this.layers || this.map.layers; │ │ │ │ │ + var layers = []; │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = candidates[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.WMTS && layer.requestEncoding === this.requestEncoding && (!this.queryVisible || layer.getVisibility())) { │ │ │ │ │ + layers.push(layer); │ │ │ │ │ + if (!this.drillDown || this.hover) { │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - this.up = this.down = null │ │ │ │ │ - } else { │ │ │ │ │ - this.callback(key, args) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return layers │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerToTop(); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - removelayer: this.handleMapEvents, │ │ │ │ │ - changelayer: this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - activated = true │ │ │ │ │ + buildRequestOptions: function(layer, xy) { │ │ │ │ │ + var loc = this.map.getLonLatFromPixel(xy); │ │ │ │ │ + var getTileUrl = layer.getURL(new OpenLayers.Bounds(loc.lon, loc.lat, loc.lon, loc.lat)); │ │ │ │ │ + var params = OpenLayers.Util.getParameters(getTileUrl); │ │ │ │ │ + var tileInfo = layer.getTileInfo(loc); │ │ │ │ │ + OpenLayers.Util.extend(params, { │ │ │ │ │ + service: "WMTS", │ │ │ │ │ + version: layer.version, │ │ │ │ │ + request: "GetFeatureInfo", │ │ │ │ │ + infoFormat: this.infoFormat, │ │ │ │ │ + i: tileInfo.i, │ │ │ │ │ + j: tileInfo.j │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Util.applyDefaults(params, this.vendorParams); │ │ │ │ │ + return { │ │ │ │ │ + url: OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url, │ │ │ │ │ + params: OpenLayers.Util.upperCaseObject(params), │ │ │ │ │ + callback: function(request) { │ │ │ │ │ + this.handleResponse(xy, request, layer) │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ } │ │ │ │ │ - return activated │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerBack(); │ │ │ │ │ - this.feature = null; │ │ │ │ │ - this.lastFeature = null; │ │ │ │ │ - this.down = null; │ │ │ │ │ - this.up = null; │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - removelayer: this.handleMapEvents, │ │ │ │ │ - changelayer: this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - deactivated = true │ │ │ │ │ + request: function(xy, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var layers = this.findLayers(); │ │ │ │ │ + if (layers.length > 0) { │ │ │ │ │ + var issue, layer; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + layer = layers[i]; │ │ │ │ │ + issue = this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ + xy: xy, │ │ │ │ │ + layer: layer │ │ │ │ │ + }); │ │ │ │ │ + if (issue !== false) { │ │ │ │ │ + ++this.pending; │ │ │ │ │ + var requestOptions = this.buildRequestOptions(layer, xy); │ │ │ │ │ + var request = OpenLayers.Request.GET(requestOptions); │ │ │ │ │ + if (options.hover === true) { │ │ │ │ │ + this.hoverRequest = request │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.pending > 0) { │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait") │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return deactivated │ │ │ │ │ }, │ │ │ │ │ - handleMapEvents: function(evt) { │ │ │ │ │ - if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ - this.moveLayerToTop() │ │ │ │ │ + handleResponse: function(xy, request, layer) { │ │ │ │ │ + --this.pending; │ │ │ │ │ + if (this.pending <= 0) { │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + this.pending = 0 │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - moveLayerToTop: function() { │ │ │ │ │ - var index = Math.max(this.map.Z_INDEX_BASE["Feature"] - 1, this.layer.getZIndex()) + 1; │ │ │ │ │ - this.layer.setZIndex(index) │ │ │ │ │ - }, │ │ │ │ │ - moveLayerBack: function() { │ │ │ │ │ - var index = this.layer.getZIndex() - 1; │ │ │ │ │ - if (index >= this.map.Z_INDEX_BASE["Feature"]) { │ │ │ │ │ - this.layer.setZIndex(index) │ │ │ │ │ + if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ + this.events.triggerEvent("exception", { │ │ │ │ │ + xy: xy, │ │ │ │ │ + request: request, │ │ │ │ │ + layer: layer │ │ │ │ │ + }) │ │ │ │ │ } else { │ │ │ │ │ - this.map.setLayerZIndex(this.layer, this.map.getLayerIndex(this.layer)) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Feature" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.Drag = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - started: false, │ │ │ │ │ - stopDown: true, │ │ │ │ │ - dragging: false, │ │ │ │ │ - last: null, │ │ │ │ │ - start: null, │ │ │ │ │ - lastMoveEvt: null, │ │ │ │ │ - oldOnselectstart: null, │ │ │ │ │ - interval: 0, │ │ │ │ │ - timeoutId: null, │ │ │ │ │ - documentDrag: false, │ │ │ │ │ - documentEvents: null, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (this.documentDrag === true) { │ │ │ │ │ - var me = this; │ │ │ │ │ - this._docMove = function(evt) { │ │ │ │ │ - me.mousemove({ │ │ │ │ │ - xy: { │ │ │ │ │ - x: evt.clientX, │ │ │ │ │ - y: evt.clientY │ │ │ │ │ - }, │ │ │ │ │ - element: document │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ + } │ │ │ │ │ + var features, except; │ │ │ │ │ + try { │ │ │ │ │ + features = this.format.read(doc) │ │ │ │ │ + } catch (error) { │ │ │ │ │ + except = true; │ │ │ │ │ + this.events.triggerEvent("exception", { │ │ │ │ │ + xy: xy, │ │ │ │ │ + request: request, │ │ │ │ │ + error: error, │ │ │ │ │ + layer: layer │ │ │ │ │ }) │ │ │ │ │ - }; │ │ │ │ │ - this._docUp = function(evt) { │ │ │ │ │ - me.mouseup({ │ │ │ │ │ - xy: { │ │ │ │ │ - x: evt.clientX, │ │ │ │ │ - y: evt.clientY │ │ │ │ │ - } │ │ │ │ │ + } │ │ │ │ │ + if (!except) { │ │ │ │ │ + this.events.triggerEvent("getfeatureinfo", { │ │ │ │ │ + text: request.responseText, │ │ │ │ │ + features: features, │ │ │ │ │ + request: request, │ │ │ │ │ + xy: xy, │ │ │ │ │ + layer: layer │ │ │ │ │ }) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - dragstart: function(evt) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - if (this.checkModifiers(evt) && (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt))) { │ │ │ │ │ - this.started = true; │ │ │ │ │ - this.start = evt.xy; │ │ │ │ │ - this.last = evt.xy; │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ - this.down(evt); │ │ │ │ │ - this.callback("down", [evt.xy]); │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ - if (!this.oldOnselectstart) { │ │ │ │ │ - this.oldOnselectstart = document.onselectstart ? document.onselectstart : OpenLayers.Function.True │ │ │ │ │ - } │ │ │ │ │ - document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ - propagate = !this.stopDown │ │ │ │ │ - } else { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.WMTSGetFeatureInfo" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Button = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + type: OpenLayers.Control.TYPE_BUTTON, │ │ │ │ │ + trigger: function() {}, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Button" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.ZoomOut = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ + trigger: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.zoomOut() │ │ │ │ │ } │ │ │ │ │ - return propagate │ │ │ │ │ }, │ │ │ │ │ - dragmove: function(evt) { │ │ │ │ │ - this.lastMoveEvt = evt; │ │ │ │ │ - if (this.started && !this.timeoutId && (evt.xy.x != this.last.x || evt.xy.y != this.last.y)) { │ │ │ │ │ - if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ - if (evt.element === document) { │ │ │ │ │ - this.adjustXY(evt); │ │ │ │ │ - this.setEvent(evt) │ │ │ │ │ - } else { │ │ │ │ │ - this.removeDocumentEvents() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.interval > 0) { │ │ │ │ │ - this.timeoutId = setTimeout(OpenLayers.Function.bind(this.removeTimeout, this), this.interval) │ │ │ │ │ - } │ │ │ │ │ - this.dragging = true; │ │ │ │ │ - this.move(evt); │ │ │ │ │ - this.callback("move", [evt.xy]); │ │ │ │ │ - if (!this.oldOnselectstart) { │ │ │ │ │ - this.oldOnselectstart = document.onselectstart; │ │ │ │ │ - document.onselectstart = OpenLayers.Function.False │ │ │ │ │ - } │ │ │ │ │ - this.last = evt.xy │ │ │ │ │ - } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - dragend: function(evt) { │ │ │ │ │ - if (this.started) { │ │ │ │ │ - if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ - this.adjustXY(evt); │ │ │ │ │ - this.removeDocumentEvents() │ │ │ │ │ - } │ │ │ │ │ - var dragged = this.start != this.last; │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ - this.up(evt); │ │ │ │ │ - this.callback("up", [evt.xy]); │ │ │ │ │ - if (dragged) { │ │ │ │ │ - this.callback("done", [evt.xy]) │ │ │ │ │ - } │ │ │ │ │ - document.onselectstart = this.oldOnselectstart │ │ │ │ │ - } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - down: function(evt) {}, │ │ │ │ │ - move: function(evt) {}, │ │ │ │ │ - up: function(evt) {}, │ │ │ │ │ - out: function(evt) {}, │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - return this.dragstart(evt) │ │ │ │ │ - }, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - return this.dragstart(evt) │ │ │ │ │ - }, │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - return this.dragmove(evt) │ │ │ │ │ - }, │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - return this.dragmove(evt) │ │ │ │ │ - }, │ │ │ │ │ - removeTimeout: function() { │ │ │ │ │ - this.timeoutId = null; │ │ │ │ │ - if (this.dragging) { │ │ │ │ │ - this.mousemove(this.lastMoveEvt) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - return this.dragend(evt) │ │ │ │ │ - }, │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - evt.xy = this.last; │ │ │ │ │ - return this.dragend(evt) │ │ │ │ │ - }, │ │ │ │ │ - mouseout: function(evt) { │ │ │ │ │ - if (this.started && OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ - if (this.documentDrag === true) { │ │ │ │ │ - this.addDocumentEvents() │ │ │ │ │ - } else { │ │ │ │ │ - var dragged = this.start != this.last; │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ - this.out(evt); │ │ │ │ │ - this.callback("out", []); │ │ │ │ │ - if (dragged) { │ │ │ │ │ - this.callback("done", [evt.xy]) │ │ │ │ │ - } │ │ │ │ │ - if (document.onselectstart) { │ │ │ │ │ - document.onselectstart = this.oldOnselectstart │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - return this.start == this.last │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - activated = true │ │ │ │ │ - } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - deactivated = true; │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown") │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - adjustXY: function(evt) { │ │ │ │ │ - var pos = OpenLayers.Util.pagePosition(this.map.viewPortDiv); │ │ │ │ │ - evt.xy.x -= pos[0]; │ │ │ │ │ - evt.xy.y -= pos[1] │ │ │ │ │ - }, │ │ │ │ │ - addDocumentEvents: function() { │ │ │ │ │ - OpenLayers.Element.addClass(document.body, "olDragDown"); │ │ │ │ │ - this.documentEvents = true; │ │ │ │ │ - OpenLayers.Event.observe(document, "mousemove", this._docMove); │ │ │ │ │ - OpenLayers.Event.observe(document, "mouseup", this._docUp) │ │ │ │ │ - }, │ │ │ │ │ - removeDocumentEvents: function() { │ │ │ │ │ - OpenLayers.Element.removeClass(document.body, "olDragDown"); │ │ │ │ │ - this.documentEvents = false; │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mousemove", this._docMove); │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mouseup", this._docUp) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Drag" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.RegularPolygon = OpenLayers.Class(OpenLayers.Handler.Drag, { │ │ │ │ │ - sides: 4, │ │ │ │ │ - radius: null, │ │ │ │ │ - snapAngle: null, │ │ │ │ │ - snapToggle: "shiftKey", │ │ │ │ │ - layerOptions: null, │ │ │ │ │ - persist: false, │ │ │ │ │ - irregular: false, │ │ │ │ │ - citeCompliant: false, │ │ │ │ │ - angle: null, │ │ │ │ │ - fixedRadius: false, │ │ │ │ │ - feature: null, │ │ │ │ │ - layer: null, │ │ │ │ │ - origin: null, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ - this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style["default"], {}) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Handler.Drag.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ - this.options = options ? options : {} │ │ │ │ │ - }, │ │ │ │ │ - setOptions: function(newOptions) { │ │ │ │ │ - OpenLayers.Util.extend(this.options, newOptions); │ │ │ │ │ - OpenLayers.Util.extend(this, newOptions) │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.Drag.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - var options = OpenLayers.Util.extend({ │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - calculateInRange: OpenLayers.Function.True, │ │ │ │ │ - wrapDateLine: this.citeCompliant │ │ │ │ │ - }, this.layerOptions); │ │ │ │ │ - this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ - this.map.addLayer(this.layer); │ │ │ │ │ - activated = true │ │ │ │ │ - } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.Drag.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - if (this.dragging) { │ │ │ │ │ - this.cancel() │ │ │ │ │ - } │ │ │ │ │ - if (this.layer.map != null) { │ │ │ │ │ - this.layer.destroy(false); │ │ │ │ │ - if (this.feature) { │ │ │ │ │ - this.feature.destroy() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.layer = null; │ │ │ │ │ - this.feature = null; │ │ │ │ │ - deactivated = true │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - down: function(evt) { │ │ │ │ │ - this.fixedRadius = !!this.radius; │ │ │ │ │ - var maploc = this.layer.getLonLatFromViewPortPx(evt.xy); │ │ │ │ │ - this.origin = new OpenLayers.Geometry.Point(maploc.lon, maploc.lat); │ │ │ │ │ - if (!this.fixedRadius || this.irregular) { │ │ │ │ │ - this.radius = this.map.getResolution() │ │ │ │ │ - } │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.clear() │ │ │ │ │ - } │ │ │ │ │ - this.feature = new OpenLayers.Feature.Vector; │ │ │ │ │ - this.createGeometry(); │ │ │ │ │ - this.callback("create", [this.origin, this.feature]); │ │ │ │ │ - this.layer.addFeatures([this.feature], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.layer.drawFeature(this.feature, this.style) │ │ │ │ │ - }, │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - var maploc = this.layer.getLonLatFromViewPortPx(evt.xy); │ │ │ │ │ - var point = new OpenLayers.Geometry.Point(maploc.lon, maploc.lat); │ │ │ │ │ - if (this.irregular) { │ │ │ │ │ - var ry = Math.sqrt(2) * Math.abs(point.y - this.origin.y) / 2; │ │ │ │ │ - this.radius = Math.max(this.map.getResolution() / 2, ry) │ │ │ │ │ - } else if (this.fixedRadius) { │ │ │ │ │ - this.origin = point │ │ │ │ │ - } else { │ │ │ │ │ - this.calculateAngle(point, evt); │ │ │ │ │ - this.radius = Math.max(this.map.getResolution() / 2, point.distanceTo(this.origin)) │ │ │ │ │ - } │ │ │ │ │ - this.modifyGeometry(); │ │ │ │ │ - if (this.irregular) { │ │ │ │ │ - var dx = point.x - this.origin.x; │ │ │ │ │ - var dy = point.y - this.origin.y; │ │ │ │ │ - var ratio; │ │ │ │ │ - if (dy == 0) { │ │ │ │ │ - ratio = dx / (this.radius * Math.sqrt(2)) │ │ │ │ │ - } else { │ │ │ │ │ - ratio = dx / dy │ │ │ │ │ - } │ │ │ │ │ - this.feature.geometry.resize(1, this.origin, ratio); │ │ │ │ │ - this.feature.geometry.move(dx / 2, dy / 2) │ │ │ │ │ - } │ │ │ │ │ - this.layer.drawFeature(this.feature, this.style) │ │ │ │ │ - }, │ │ │ │ │ - up: function(evt) { │ │ │ │ │ - this.finalize(); │ │ │ │ │ - if (this.start == this.last) { │ │ │ │ │ - this.callback("done", [evt.xy]) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - out: function(evt) { │ │ │ │ │ - this.finalize() │ │ │ │ │ - }, │ │ │ │ │ - createGeometry: function() { │ │ │ │ │ - this.angle = Math.PI * (1 / this.sides - 1 / 2); │ │ │ │ │ - if (this.snapAngle) { │ │ │ │ │ - this.angle += this.snapAngle * (Math.PI / 180) │ │ │ │ │ - } │ │ │ │ │ - this.feature.geometry = OpenLayers.Geometry.Polygon.createRegularPolygon(this.origin, this.radius, this.sides, this.snapAngle) │ │ │ │ │ - }, │ │ │ │ │ - modifyGeometry: function() { │ │ │ │ │ - var angle, point; │ │ │ │ │ - var ring = this.feature.geometry.components[0]; │ │ │ │ │ - if (ring.components.length != this.sides + 1) { │ │ │ │ │ - this.createGeometry(); │ │ │ │ │ - ring = this.feature.geometry.components[0] │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0; i < this.sides; ++i) { │ │ │ │ │ - point = ring.components[i]; │ │ │ │ │ - angle = this.angle + i * 2 * Math.PI / this.sides; │ │ │ │ │ - point.x = this.origin.x + this.radius * Math.cos(angle); │ │ │ │ │ - point.y = this.origin.y + this.radius * Math.sin(angle); │ │ │ │ │ - point.clearBounds() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - calculateAngle: function(point, evt) { │ │ │ │ │ - var alpha = Math.atan2(point.y - this.origin.y, point.x - this.origin.x); │ │ │ │ │ - if (this.snapAngle && (this.snapToggle && !evt[this.snapToggle])) { │ │ │ │ │ - var snapAngleRad = Math.PI / 180 * this.snapAngle; │ │ │ │ │ - this.angle = Math.round(alpha / snapAngleRad) * snapAngleRad │ │ │ │ │ - } else { │ │ │ │ │ - this.angle = alpha │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - cancel: function() { │ │ │ │ │ - this.callback("cancel", null); │ │ │ │ │ - this.finalize() │ │ │ │ │ - }, │ │ │ │ │ - finalize: function() { │ │ │ │ │ - this.origin = null; │ │ │ │ │ - this.radius = this.options.radius │ │ │ │ │ - }, │ │ │ │ │ - clear: function() { │ │ │ │ │ - if (this.layer) { │ │ │ │ │ - this.layer.renderer.clear(); │ │ │ │ │ - this.layer.destroyFeatures() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - callback: function(name, args) { │ │ │ │ │ - if (this.callbacks[name]) { │ │ │ │ │ - this.callbacks[name].apply(this.control, [this.feature.geometry.clone()]) │ │ │ │ │ - } │ │ │ │ │ - if (!this.persist && (name == "done" || name == "cancel")) { │ │ │ │ │ - this.clear() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.RegularPolygon" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.Click = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - delay: 300, │ │ │ │ │ - single: true, │ │ │ │ │ - double: false, │ │ │ │ │ - pixelTolerance: 0, │ │ │ │ │ - dblclickTolerance: 13, │ │ │ │ │ - stopSingle: false, │ │ │ │ │ - stopDouble: false, │ │ │ │ │ - timerId: null, │ │ │ │ │ - down: null, │ │ │ │ │ - last: null, │ │ │ │ │ - first: null, │ │ │ │ │ - rightclickTimerId: null, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - this.down = this.getEventInfo(evt); │ │ │ │ │ - this.last = this.getEventInfo(evt); │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - this.last = this.getEventInfo(evt); │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - if (this.down) { │ │ │ │ │ - evt.xy = this.last.xy; │ │ │ │ │ - evt.lastTouches = this.last.touches; │ │ │ │ │ - this.handleSingle(evt); │ │ │ │ │ - this.down = null │ │ │ │ │ - } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - this.down = this.getEventInfo(evt); │ │ │ │ │ - this.last = this.getEventInfo(evt); │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - if (this.checkModifiers(evt) && this.control.handleRightClicks && OpenLayers.Event.isRightClick(evt)) { │ │ │ │ │ - propagate = this.rightclick(evt) │ │ │ │ │ - } │ │ │ │ │ - return propagate │ │ │ │ │ - }, │ │ │ │ │ - rightclick: function(evt) { │ │ │ │ │ - if (this.passesTolerance(evt)) { │ │ │ │ │ - if (this.rightclickTimerId != null) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.callback("dblrightclick", [evt]); │ │ │ │ │ - return !this.stopDouble │ │ │ │ │ - } else { │ │ │ │ │ - var clickEvent = this["double"] ? OpenLayers.Util.extend({}, evt) : this.callback("rightclick", [evt]); │ │ │ │ │ - var delayedRightCall = OpenLayers.Function.bind(this.delayedRightCall, this, clickEvent); │ │ │ │ │ - this.rightclickTimerId = window.setTimeout(delayedRightCall, this.delay) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return !this.stopSingle │ │ │ │ │ - }, │ │ │ │ │ - delayedRightCall: function(evt) { │ │ │ │ │ - this.rightclickTimerId = null; │ │ │ │ │ - if (evt) { │ │ │ │ │ - this.callback("rightclick", [evt]) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - if (!this.last) { │ │ │ │ │ - this.last = this.getEventInfo(evt) │ │ │ │ │ - } │ │ │ │ │ - this.handleSingle(evt); │ │ │ │ │ - return !this.stopSingle │ │ │ │ │ - }, │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - this.handleDouble(evt); │ │ │ │ │ - return !this.stopDouble │ │ │ │ │ - }, │ │ │ │ │ - handleDouble: function(evt) { │ │ │ │ │ - if (this.passesDblclickTolerance(evt)) { │ │ │ │ │ - if (this["double"]) { │ │ │ │ │ - this.callback("dblclick", [evt]) │ │ │ │ │ - } │ │ │ │ │ - this.clearTimer() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - handleSingle: function(evt) { │ │ │ │ │ - if (this.passesTolerance(evt)) { │ │ │ │ │ - if (this.timerId != null) { │ │ │ │ │ - if (this.last.touches && this.last.touches.length === 1) { │ │ │ │ │ - if (this["double"]) { │ │ │ │ │ - OpenLayers.Event.preventDefault(evt) │ │ │ │ │ - } │ │ │ │ │ - this.handleDouble(evt) │ │ │ │ │ - } │ │ │ │ │ - if (!this.last.touches || this.last.touches.length !== 2) { │ │ │ │ │ - this.clearTimer() │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.first = this.getEventInfo(evt); │ │ │ │ │ - var clickEvent = this.single ? OpenLayers.Util.extend({}, evt) : null; │ │ │ │ │ - this.queuePotentialClick(clickEvent) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - queuePotentialClick: function(evt) { │ │ │ │ │ - this.timerId = window.setTimeout(OpenLayers.Function.bind(this.delayedCall, this, evt), this.delay) │ │ │ │ │ - }, │ │ │ │ │ - passesTolerance: function(evt) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - if (this.pixelTolerance != null && this.down && this.down.xy) { │ │ │ │ │ - passes = this.pixelTolerance >= this.down.xy.distanceTo(evt.xy); │ │ │ │ │ - if (passes && this.touch && this.down.touches.length === this.last.touches.length) { │ │ │ │ │ - for (var i = 0, ii = this.down.touches.length; i < ii; ++i) { │ │ │ │ │ - if (this.getTouchDistance(this.down.touches[i], this.last.touches[i]) > this.pixelTolerance) { │ │ │ │ │ - passes = false; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return passes │ │ │ │ │ - }, │ │ │ │ │ - getTouchDistance: function(from, to) { │ │ │ │ │ - return Math.sqrt(Math.pow(from.clientX - to.clientX, 2) + Math.pow(from.clientY - to.clientY, 2)) │ │ │ │ │ - }, │ │ │ │ │ - passesDblclickTolerance: function(evt) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - if (this.down && this.first) { │ │ │ │ │ - passes = this.down.xy.distanceTo(this.first.xy) <= this.dblclickTolerance │ │ │ │ │ - } │ │ │ │ │ - return passes │ │ │ │ │ - }, │ │ │ │ │ - clearTimer: function() { │ │ │ │ │ - if (this.timerId != null) { │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null │ │ │ │ │ - } │ │ │ │ │ - if (this.rightclickTimerId != null) { │ │ │ │ │ - window.clearTimeout(this.rightclickTimerId); │ │ │ │ │ - this.rightclickTimerId = null │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - delayedCall: function(evt) { │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - if (evt) { │ │ │ │ │ - this.callback("click", [evt]) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getEventInfo: function(evt) { │ │ │ │ │ - var touches; │ │ │ │ │ - if (evt.touches) { │ │ │ │ │ - var len = evt.touches.length; │ │ │ │ │ - touches = new Array(len); │ │ │ │ │ - var touch; │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - touch = evt.touches[i]; │ │ │ │ │ - touches[i] = { │ │ │ │ │ - clientX: touch.olClientX, │ │ │ │ │ - clientY: touch.olClientY │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - xy: evt.xy, │ │ │ │ │ - touches: touches │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.down = null; │ │ │ │ │ - this.first = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - deactivated = true │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Click" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomOut" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Handler.Path = OpenLayers.Class(OpenLayers.Handler.Point, { │ │ │ │ │ - line: null, │ │ │ │ │ - maxVertices: null, │ │ │ │ │ - doubleTouchTolerance: 20, │ │ │ │ │ - freehand: false, │ │ │ │ │ - freehandToggle: "shiftKey", │ │ │ │ │ - timerId: null, │ │ │ │ │ - redoStack: null, │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.line = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.LineString([this.point.geometry])); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.line, this.point], { │ │ │ │ │ - silent: true │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - OpenLayers.Handler.Point.prototype.destroyFeature.call(this, force); │ │ │ │ │ - this.line = null │ │ │ │ │ - }, │ │ │ │ │ - destroyPersistedFeature: function() { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - if (layer && layer.features.length > 2) { │ │ │ │ │ - this.layer.features[0].destroy() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - removePoint: function() { │ │ │ │ │ - if (this.point) { │ │ │ │ │ - this.layer.removeFeatures([this.point]) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - addPoint: function(pixel) { │ │ │ │ │ - this.layer.removeFeatures([this.point]); │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat)); │ │ │ │ │ - this.line.geometry.addComponent(this.point.geometry, this.line.geometry.components.length); │ │ │ │ │ - this.layer.addFeatures([this.point]); │ │ │ │ │ - this.callback("point", [this.point.geometry, this.getGeometry()]); │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - delete this.redoStack │ │ │ │ │ - }, │ │ │ │ │ - insertXY: function(x, y) { │ │ │ │ │ - this.line.geometry.addComponent(new OpenLayers.Geometry.Point(x, y), this.getCurrentPointIndex()); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - delete this.redoStack │ │ │ │ │ - }, │ │ │ │ │ - insertDeltaXY: function(dx, dy) { │ │ │ │ │ - var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ - var p0 = this.line.geometry.components[previousIndex]; │ │ │ │ │ - if (p0 && !isNaN(p0.x) && !isNaN(p0.y)) { │ │ │ │ │ - this.insertXY(p0.x + dx, p0.y + dy) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - insertDirectionLength: function(direction, length) { │ │ │ │ │ - direction *= Math.PI / 180; │ │ │ │ │ - var dx = length * Math.cos(direction); │ │ │ │ │ - var dy = length * Math.sin(direction); │ │ │ │ │ - this.insertDeltaXY(dx, dy) │ │ │ │ │ +OpenLayers.Layer.WMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + service: "WMS", │ │ │ │ │ + version: "1.1.1", │ │ │ │ │ + request: "GetMap", │ │ │ │ │ + styles: "", │ │ │ │ │ + format: "image/jpeg" │ │ │ │ │ }, │ │ │ │ │ - insertDeflectionLength: function(deflection, length) { │ │ │ │ │ - var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ - if (previousIndex > 0) { │ │ │ │ │ - var p1 = this.line.geometry.components[previousIndex]; │ │ │ │ │ - var p0 = this.line.geometry.components[previousIndex - 1]; │ │ │ │ │ - var theta = Math.atan2(p1.y - p0.y, p1.x - p0.x); │ │ │ │ │ - this.insertDirectionLength(theta * 180 / Math.PI + deflection, length) │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + encodeBBOX: false, │ │ │ │ │ + noMagic: false, │ │ │ │ │ + yx: {}, │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ + if (parseFloat(params.VERSION) >= 1.3 && !params.EXCEPTIONS) { │ │ │ │ │ + params.EXCEPTIONS = "INIMAGE" │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - getCurrentPointIndex: function() { │ │ │ │ │ - return this.line.geometry.components.length - 1 │ │ │ │ │ - }, │ │ │ │ │ - undo: function() { │ │ │ │ │ - var geometry = this.line.geometry; │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var index = this.getCurrentPointIndex() - 1; │ │ │ │ │ - var target = components[index]; │ │ │ │ │ - var undone = geometry.removeComponent(target); │ │ │ │ │ - if (undone) { │ │ │ │ │ - if (this.touch && index > 0) { │ │ │ │ │ - components = geometry.components; │ │ │ │ │ - var lastpt = components[index - 1]; │ │ │ │ │ - var curptidx = this.getCurrentPointIndex(); │ │ │ │ │ - var curpt = components[curptidx]; │ │ │ │ │ - curpt.x = lastpt.x; │ │ │ │ │ - curpt.y = lastpt.y │ │ │ │ │ + newArguments.push(name, url, params, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + OpenLayers.Util.applyDefaults(this.params, OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS)); │ │ │ │ │ + if (!this.noMagic && this.params.TRANSPARENT && this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ + if (options == null || !options.isBaseLayer) { │ │ │ │ │ + this.isBaseLayer = false │ │ │ │ │ } │ │ │ │ │ - if (!this.redoStack) { │ │ │ │ │ - this.redoStack = [] │ │ │ │ │ + if (this.params.FORMAT == "image/jpeg") { │ │ │ │ │ + this.params.FORMAT = OpenLayers.Util.alphaHack() ? "image/gif" : "image/png" │ │ │ │ │ } │ │ │ │ │ - this.redoStack.push(target); │ │ │ │ │ - this.drawFeature() │ │ │ │ │ - } │ │ │ │ │ - return undone │ │ │ │ │ - }, │ │ │ │ │ - redo: function() { │ │ │ │ │ - var target = this.redoStack && this.redoStack.pop(); │ │ │ │ │ - if (target) { │ │ │ │ │ - this.line.geometry.addComponent(target, this.getCurrentPointIndex()); │ │ │ │ │ - this.drawFeature() │ │ │ │ │ } │ │ │ │ │ - return !!target │ │ │ │ │ - }, │ │ │ │ │ - freehandMode: function(evt) { │ │ │ │ │ - return this.freehandToggle && evt[this.freehandToggle] ? !this.freehand : this.freehand │ │ │ │ │ }, │ │ │ │ │ - modifyFeature: function(pixel, drawing) { │ │ │ │ │ - if (!this.line) { │ │ │ │ │ - this.createFeature(pixel) │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.WMS(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point.geometry.x = lonlat.lon; │ │ │ │ │ - this.point.geometry.y = lonlat.lat; │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.getSketch(), drawing]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.drawFeature() │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.line, this.style); │ │ │ │ │ - this.layer.drawFeature(this.point, this.style) │ │ │ │ │ + reverseAxisOrder: function() { │ │ │ │ │ + var projCode = this.projection.getCode(); │ │ │ │ │ + return parseFloat(this.params.VERSION) >= 1.3 && !!(this.yx[projCode] || OpenLayers.Projection.defaults[projCode] && OpenLayers.Projection.defaults[projCode].yx) │ │ │ │ │ }, │ │ │ │ │ - getSketch: function() { │ │ │ │ │ - return this.line │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var imageSize = this.getImageSize(); │ │ │ │ │ + var newParams = {}; │ │ │ │ │ + var reverseAxisOrder = this.reverseAxisOrder(); │ │ │ │ │ + newParams.BBOX = this.encodeBBOX ? bounds.toBBOX(null, reverseAxisOrder) : bounds.toArray(reverseAxisOrder); │ │ │ │ │ + newParams.WIDTH = imageSize.w; │ │ │ │ │ + newParams.HEIGHT = imageSize.h; │ │ │ │ │ + var requestString = this.getFullRequestString(newParams); │ │ │ │ │ + return requestString │ │ │ │ │ }, │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.line && this.line.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiLineString([geometry]) │ │ │ │ │ - } │ │ │ │ │ - return geometry │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ + var newArguments = [upperParams]; │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, newArguments) │ │ │ │ │ }, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - if (this.timerId && this.passesTolerance(this.lastTouchPx, evt.xy, this.doubleTouchTolerance)) { │ │ │ │ │ - this.finishGeometry(); │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - return false │ │ │ │ │ + getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ + var mapProjection = this.map.getProjectionObject(); │ │ │ │ │ + var projectionCode = this.projection && this.projection.equals(mapProjection) ? this.projection.getCode() : mapProjection.getCode(); │ │ │ │ │ + var value = projectionCode == "none" ? null : projectionCode; │ │ │ │ │ + if (parseFloat(this.params.VERSION) >= 1.3) { │ │ │ │ │ + this.params.CRS = value │ │ │ │ │ } else { │ │ │ │ │ - if (this.timerId) { │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null │ │ │ │ │ - } │ │ │ │ │ - this.timerId = window.setTimeout(OpenLayers.Function.bind(function() { │ │ │ │ │ - this.timerId = null │ │ │ │ │ - }, this), 300); │ │ │ │ │ - return OpenLayers.Handler.Point.prototype.touchstart.call(this, evt) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - down: function(evt) { │ │ │ │ │ - var stopDown = this.stopDown; │ │ │ │ │ - if (this.freehandMode(evt)) { │ │ │ │ │ - stopDown = true; │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!this.touch && (!this.lastDown || !this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance))) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp) │ │ │ │ │ - } │ │ │ │ │ - this.mouseDown = true; │ │ │ │ │ - this.lastDown = evt.xy; │ │ │ │ │ - this.stoppedDown = stopDown; │ │ │ │ │ - return !stopDown │ │ │ │ │ - }, │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature() │ │ │ │ │ - } │ │ │ │ │ - if (this.maxVertices && this.line && this.line.geometry.components.length === this.maxVertices) { │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize() │ │ │ │ │ - } else { │ │ │ │ │ - this.addPoint(evt.xy) │ │ │ │ │ - } │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp) │ │ │ │ │ - } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - up: function(evt) { │ │ │ │ │ - if (this.mouseDown && (!this.lastUp || !this.lastUp.equals(evt.xy))) { │ │ │ │ │ - if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature() │ │ │ │ │ - } │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize() │ │ │ │ │ - } else { │ │ │ │ │ - if (this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance)) { │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy) │ │ │ │ │ - } │ │ │ │ │ - if (this.lastUp == null && this.persist) { │ │ │ │ │ - this.destroyPersistedFeature() │ │ │ │ │ - } │ │ │ │ │ - this.addPoint(evt.xy); │ │ │ │ │ - this.lastUp = evt.xy; │ │ │ │ │ - if (this.line.geometry.components.length === this.maxVertices + 1) { │ │ │ │ │ - this.finishGeometry() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + this.params.SRS = value │ │ │ │ │ } │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - return !this.stopUp │ │ │ │ │ - }, │ │ │ │ │ - finishGeometry: function() { │ │ │ │ │ - var index = this.line.geometry.components.length - 1; │ │ │ │ │ - this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize() │ │ │ │ │ - }, │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - if (!this.freehandMode(evt)) { │ │ │ │ │ - this.finishGeometry() │ │ │ │ │ + if (typeof this.params.TRANSPARENT == "boolean") { │ │ │ │ │ + newParams.TRANSPARENT = this.params.TRANSPARENT ? "TRUE" : "FALSE" │ │ │ │ │ } │ │ │ │ │ - return false │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.getFullRequestString.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Path" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.WMS" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Handler.Polygon = OpenLayers.Class(OpenLayers.Handler.Path, { │ │ │ │ │ - holeModifier: null, │ │ │ │ │ - drawingHole: false, │ │ │ │ │ - polygon: null, │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.line = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.LinearRing([this.point.geometry])); │ │ │ │ │ - this.polygon = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Polygon([this.line.geometry])); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.polygon, this.point], { │ │ │ │ │ - silent: true │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - addPoint: function(pixel) { │ │ │ │ │ - if (!this.drawingHole && this.holeModifier && this.evt && this.evt[this.holeModifier]) { │ │ │ │ │ - var geometry = this.point.geometry; │ │ │ │ │ - var features = this.control.layer.features; │ │ │ │ │ - var candidate, polygon; │ │ │ │ │ - for (var i = features.length - 1; i >= 0; --i) { │ │ │ │ │ - candidate = features[i].geometry; │ │ │ │ │ - if ((candidate instanceof OpenLayers.Geometry.Polygon || candidate instanceof OpenLayers.Geometry.MultiPolygon) && candidate.intersects(geometry)) { │ │ │ │ │ - polygon = features[i]; │ │ │ │ │ - this.control.layer.removeFeatures([polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.control.layer.events.registerPriority("sketchcomplete", this, this.finalizeInteriorRing); │ │ │ │ │ - this.control.layer.events.registerPriority("sketchmodified", this, this.enforceTopology); │ │ │ │ │ - polygon.geometry.addComponent(this.line.geometry); │ │ │ │ │ - this.polygon = polygon; │ │ │ │ │ - this.drawingHole = true; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Handler.Path.prototype.addPoint.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - getCurrentPointIndex: function() { │ │ │ │ │ - return this.line.geometry.components.length - 2 │ │ │ │ │ - }, │ │ │ │ │ - enforceTopology: function(event) { │ │ │ │ │ - var point = event.vertex; │ │ │ │ │ - var components = this.line.geometry.components; │ │ │ │ │ - if (!this.polygon.geometry.intersects(point)) { │ │ │ │ │ - var last = components[components.length - 3]; │ │ │ │ │ - point.x = last.x; │ │ │ │ │ - point.y = last.y │ │ │ │ │ +OpenLayers.Control.SLDSelect = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + clearOnDeactivate: false, │ │ │ │ │ + layers: null, │ │ │ │ │ + callbacks: null, │ │ │ │ │ + selectionSymbolizer: { │ │ │ │ │ + Polygon: { │ │ │ │ │ + fillColor: "#FF0000", │ │ │ │ │ + stroke: false │ │ │ │ │ + }, │ │ │ │ │ + Line: { │ │ │ │ │ + strokeColor: "#FF0000", │ │ │ │ │ + strokeWidth: 2 │ │ │ │ │ + }, │ │ │ │ │ + Point: { │ │ │ │ │ + graphicName: "square", │ │ │ │ │ + fillColor: "#FF0000", │ │ │ │ │ + pointRadius: 5 │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - finishGeometry: function() { │ │ │ │ │ - var index = this.line.geometry.components.length - 2; │ │ │ │ │ - this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize() │ │ │ │ │ - }, │ │ │ │ │ - finalizeInteriorRing: function() { │ │ │ │ │ - var ring = this.line.geometry; │ │ │ │ │ - var modified = ring.getArea() !== 0; │ │ │ │ │ - if (modified) { │ │ │ │ │ - var rings = this.polygon.geometry.components; │ │ │ │ │ - for (var i = rings.length - 2; i >= 0; --i) { │ │ │ │ │ - if (ring.intersects(rings[i])) { │ │ │ │ │ - modified = false; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (modified) { │ │ │ │ │ - var target; │ │ │ │ │ - outer: for (var i = rings.length - 2; i > 0; --i) { │ │ │ │ │ - var points = rings[i].components; │ │ │ │ │ - for (var j = 0, jj = points.length; j < jj; ++j) { │ │ │ │ │ - if (ring.containsPoint(points[j])) { │ │ │ │ │ - modified = false; │ │ │ │ │ - break outer │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (modified) { │ │ │ │ │ - if (this.polygon.state !== OpenLayers.State.INSERT) { │ │ │ │ │ - this.polygon.state = OpenLayers.State.UPDATE │ │ │ │ │ + layerOptions: null, │ │ │ │ │ + sketchStyle: null, │ │ │ │ │ + wfsCache: {}, │ │ │ │ │ + layerCache: {}, │ │ │ │ │ + initialize: function(handler, options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.callbacks = OpenLayers.Util.extend({ │ │ │ │ │ + done: this.select, │ │ │ │ │ + click: this.select │ │ │ │ │ + }, this.callbacks); │ │ │ │ │ + this.handlerOptions = this.handlerOptions || {}; │ │ │ │ │ + this.layerOptions = OpenLayers.Util.applyDefaults(this.layerOptions, { │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + tileOptions: { │ │ │ │ │ + maxGetUrlLength: 2048 │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - this.polygon.geometry.removeComponent(ring) │ │ │ │ │ - } │ │ │ │ │ - this.restoreFeature(); │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - cancel: function() { │ │ │ │ │ - if (this.drawingHole) { │ │ │ │ │ - this.polygon.geometry.removeComponent(this.line.geometry); │ │ │ │ │ - this.restoreFeature(true) │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Handler.Path.prototype.cancel.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - restoreFeature: function(cancel) { │ │ │ │ │ - this.control.layer.events.unregister("sketchcomplete", this, this.finalizeInteriorRing); │ │ │ │ │ - this.control.layer.events.unregister("sketchmodified", this, this.enforceTopology); │ │ │ │ │ - this.layer.removeFeatures([this.polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.control.layer.addFeatures([this.polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ }); │ │ │ │ │ - this.drawingHole = false; │ │ │ │ │ - if (!cancel) { │ │ │ │ │ - this.control.layer.events.triggerEvent("sketchcomplete", { │ │ │ │ │ - feature: this.polygon │ │ │ │ │ + if (this.sketchStyle) { │ │ │ │ │ + this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults(this.handlerOptions.layerOptions, { │ │ │ │ │ + styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ + default: this.sketchStyle │ │ │ │ │ + }) │ │ │ │ │ }) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - OpenLayers.Handler.Path.prototype.destroyFeature.call(this, force); │ │ │ │ │ - this.polygon = null │ │ │ │ │ - }, │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.polygon, this.style); │ │ │ │ │ - this.layer.drawFeature(this.point, this.style) │ │ │ │ │ - }, │ │ │ │ │ - getSketch: function() { │ │ │ │ │ - return this.polygon │ │ │ │ │ - }, │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.polygon && this.polygon.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiPolygon([geometry]) │ │ │ │ │ - } │ │ │ │ │ - return geometry │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Polygon" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.MouseWheel = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - wheelListener: null, │ │ │ │ │ - interval: 0, │ │ │ │ │ - maxDelta: Number.POSITIVE_INFINITY, │ │ │ │ │ - delta: 0, │ │ │ │ │ - cumulative: true, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.wheelListener = OpenLayers.Function.bindAsEventListener(this.onWheelEvent, this) │ │ │ │ │ + this.handler = new handler(this, this.callbacks, this.handlerOptions) │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ - this.wheelListener = null │ │ │ │ │ - }, │ │ │ │ │ - onWheelEvent: function(e) { │ │ │ │ │ - if (!this.map || !this.checkModifiers(e)) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var overScrollableDiv = false; │ │ │ │ │ - var allowScroll = false; │ │ │ │ │ - var overMapDiv = false; │ │ │ │ │ - var elem = OpenLayers.Event.element(e); │ │ │ │ │ - while (elem != null && !overMapDiv && !overScrollableDiv) { │ │ │ │ │ - if (!overScrollableDiv) { │ │ │ │ │ - try { │ │ │ │ │ - var overflow; │ │ │ │ │ - if (elem.currentStyle) { │ │ │ │ │ - overflow = elem.currentStyle["overflow"] │ │ │ │ │ - } else { │ │ │ │ │ - var style = document.defaultView.getComputedStyle(elem, null); │ │ │ │ │ - overflow = style.getPropertyValue("overflow") │ │ │ │ │ - } │ │ │ │ │ - overScrollableDiv = overflow && overflow == "auto" || overflow == "scroll" │ │ │ │ │ - } catch (err) {} │ │ │ │ │ - } │ │ │ │ │ - if (!allowScroll) { │ │ │ │ │ - allowScroll = OpenLayers.Element.hasClass(elem, "olScrollable"); │ │ │ │ │ - if (!allowScroll) { │ │ │ │ │ - for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if (elem == layer.div || elem == layer.pane) { │ │ │ │ │ - allowScroll = true; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - overMapDiv = elem == this.map.div; │ │ │ │ │ - elem = elem.parentNode │ │ │ │ │ - } │ │ │ │ │ - if (!overScrollableDiv && overMapDiv) { │ │ │ │ │ - if (allowScroll) { │ │ │ │ │ - var delta = 0; │ │ │ │ │ - if (e.wheelDelta) { │ │ │ │ │ - delta = e.wheelDelta; │ │ │ │ │ - if (delta % 160 === 0) { │ │ │ │ │ - delta = delta * .75 │ │ │ │ │ - } │ │ │ │ │ - delta = delta / 120 │ │ │ │ │ - } else if (e.detail) { │ │ │ │ │ - delta = -(e.detail / Math.abs(e.detail)) │ │ │ │ │ - } │ │ │ │ │ - this.delta += delta; │ │ │ │ │ - window.clearTimeout(this._timeoutId); │ │ │ │ │ - if (this.interval && Math.abs(this.delta) < this.maxDelta) { │ │ │ │ │ - var evt = OpenLayers.Util.extend({}, e); │ │ │ │ │ - this._timeoutId = window.setTimeout(OpenLayers.Function.bind(function() { │ │ │ │ │ - this.wheelZoom(evt) │ │ │ │ │ - }, this), this.interval) │ │ │ │ │ - } else { │ │ │ │ │ - this.wheelZoom(e) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.stop(e) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - wheelZoom: function(e) { │ │ │ │ │ - var delta = this.delta; │ │ │ │ │ - this.delta = 0; │ │ │ │ │ - if (delta) { │ │ │ │ │ - e.xy = this.map.events.getMousePosition(e); │ │ │ │ │ - if (delta < 0) { │ │ │ │ │ - this.callback("down", [e, this.cumulative ? Math.max(-this.maxDelta, delta) : -1]) │ │ │ │ │ - } else { │ │ │ │ │ - this.callback("up", [e, this.cumulative ? Math.min(this.maxDelta, delta) : 1]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - activate: function(evt) { │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - var wheelListener = this.wheelListener; │ │ │ │ │ - OpenLayers.Event.observe(window, "DOMMouseScroll", wheelListener); │ │ │ │ │ - OpenLayers.Event.observe(window, "mousewheel", wheelListener); │ │ │ │ │ - OpenLayers.Event.observe(document, "mousewheel", wheelListener); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + for (var key in this.layerCache) { │ │ │ │ │ + delete this.layerCache[key] │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function(evt) { │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - var wheelListener = this.wheelListener; │ │ │ │ │ - OpenLayers.Event.stopObserving(window, "DOMMouseScroll", wheelListener); │ │ │ │ │ - OpenLayers.Event.stopObserving(window, "mousewheel", wheelListener); │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mousewheel", wheelListener); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + for (var key in this.wfsCache) { │ │ │ │ │ + delete this.wfsCache[key] │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.MouseWheel" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.Keyboard = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - KEY_EVENTS: ["keydown", "keyup"], │ │ │ │ │ - eventListener: null, │ │ │ │ │ - observeElement: null, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.eventListener = OpenLayers.Function.bindAsEventListener(this.handleKeyEvent, this) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.eventListener = null; │ │ │ │ │ - OpenLayers.Handler.prototype.destroy.apply(this, arguments) │ │ │ │ │ + coupleLayerVisiblity: function(evt) { │ │ │ │ │ + this.setVisibility(evt.object.getVisibility()) │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.observeElement = this.observeElement || document; │ │ │ │ │ - for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ - OpenLayers.Event.observe(this.observeElement, this.KEY_EVENTS[i], this.eventListener) │ │ │ │ │ + createSelectionLayer: function(source) { │ │ │ │ │ + var selectionLayer; │ │ │ │ │ + if (!this.layerCache[source.id]) { │ │ │ │ │ + selectionLayer = new OpenLayers.Layer.WMS(source.name, source.url, source.params, OpenLayers.Util.applyDefaults(this.layerOptions, source.getOptions())); │ │ │ │ │ + this.layerCache[source.id] = selectionLayer; │ │ │ │ │ + if (this.layerOptions.displayInLayerSwitcher === false) { │ │ │ │ │ + source.events.on({ │ │ │ │ │ + visibilitychanged: this.coupleLayerVisiblity, │ │ │ │ │ + scope: selectionLayer │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - return true │ │ │ │ │ + this.map.addLayer(selectionLayer) │ │ │ │ │ } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ - OpenLayers.Event.stopObserving(this.observeElement, this.KEY_EVENTS[i], this.eventListener) │ │ │ │ │ - } │ │ │ │ │ - deactivated = true │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - handleKeyEvent: function(evt) { │ │ │ │ │ - if (this.checkModifiers(evt)) { │ │ │ │ │ - this.callback(evt.type, [evt]) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Keyboard" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.Hover = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - delay: 500, │ │ │ │ │ - pixelTolerance: null, │ │ │ │ │ - stopMove: false, │ │ │ │ │ - px: null, │ │ │ │ │ - timerId: null, │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - if (this.passesTolerance(evt.xy)) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.callback("move", [evt]); │ │ │ │ │ - this.px = evt.xy; │ │ │ │ │ - evt = OpenLayers.Util.extend({}, evt); │ │ │ │ │ - this.timerId = window.setTimeout(OpenLayers.Function.bind(this.delayedCall, this, evt), this.delay) │ │ │ │ │ - } │ │ │ │ │ - return !this.stopMove │ │ │ │ │ - }, │ │ │ │ │ - mouseout: function(evt) { │ │ │ │ │ - if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.callback("move", [evt]) │ │ │ │ │ - } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - passesTolerance: function(px) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - if (this.pixelTolerance && this.px) { │ │ │ │ │ - var dpx = Math.sqrt(Math.pow(this.px.x - px.x, 2) + Math.pow(this.px.y - px.y, 2)); │ │ │ │ │ - if (dpx < this.pixelTolerance) { │ │ │ │ │ - passes = false │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return passes │ │ │ │ │ - }, │ │ │ │ │ - clearTimer: function() { │ │ │ │ │ - if (this.timerId != null) { │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - delayedCall: function(evt) { │ │ │ │ │ - this.callback("pause", [evt]) │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - deactivated = true │ │ │ │ │ + selectionLayer = this.layerCache[source.id] │ │ │ │ │ } │ │ │ │ │ - return deactivated │ │ │ │ │ + return selectionLayer │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Hover" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.Pinch = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - started: false, │ │ │ │ │ - stopDown: false, │ │ │ │ │ - pinching: false, │ │ │ │ │ - last: null, │ │ │ │ │ - start: null, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - if (OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ - this.started = true; │ │ │ │ │ - this.last = this.start = { │ │ │ │ │ - distance: this.getDistance(evt.touches), │ │ │ │ │ - delta: 0, │ │ │ │ │ - scale: 1 │ │ │ │ │ + createSLD: function(layer, filters, geometryAttributes) { │ │ │ │ │ + var sld = { │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + namedLayers: {} │ │ │ │ │ + }; │ │ │ │ │ + var layerNames = [layer.params.LAYERS].join(",").split(","); │ │ │ │ │ + for (var i = 0, len = layerNames.length; i < len; i++) { │ │ │ │ │ + var name = layerNames[i]; │ │ │ │ │ + sld.namedLayers[name] = { │ │ │ │ │ + name: name, │ │ │ │ │ + userStyles: [] │ │ │ │ │ }; │ │ │ │ │ - this.callback("start", [evt, this.start]); │ │ │ │ │ - propagate = !this.stopDown │ │ │ │ │ - } else if (this.started) { │ │ │ │ │ - return false │ │ │ │ │ - } else { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ - return propagate │ │ │ │ │ - }, │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - if (this.started && OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ - this.pinching = true; │ │ │ │ │ - var current = this.getPinchData(evt); │ │ │ │ │ - this.callback("move", [evt, current]); │ │ │ │ │ - this.last = current; │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ - } else if (this.started) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - if (this.started && !OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - this.callback("done", [evt, this.start, this.last]); │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - activated = true │ │ │ │ │ - } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - deactivated = true │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - getDistance: function(touches) { │ │ │ │ │ - var t0 = touches[0]; │ │ │ │ │ - var t1 = touches[1]; │ │ │ │ │ - return Math.sqrt(Math.pow(t0.olClientX - t1.olClientX, 2) + Math.pow(t0.olClientY - t1.olClientY, 2)) │ │ │ │ │ - }, │ │ │ │ │ - getPinchData: function(evt) { │ │ │ │ │ - var distance = this.getDistance(evt.touches); │ │ │ │ │ - var scale = distance / this.start.distance; │ │ │ │ │ - return { │ │ │ │ │ - distance: distance, │ │ │ │ │ - delta: this.last.distance - distance, │ │ │ │ │ - scale: scale │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Pinch" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.Box = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - dragHandler: null, │ │ │ │ │ - boxDivClassName: "olHandlerBoxZoomBox", │ │ │ │ │ - boxOffsets: null, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.dragHandler = new OpenLayers.Handler.Drag(this, { │ │ │ │ │ - down: this.startBox, │ │ │ │ │ - move: this.moveBox, │ │ │ │ │ - out: this.removeBox, │ │ │ │ │ - up: this.endBox │ │ │ │ │ - }, { │ │ │ │ │ - keyMask: this.keyMask │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ - if (this.dragHandler) { │ │ │ │ │ - this.dragHandler.destroy(); │ │ │ │ │ - this.dragHandler = null │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Handler.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (this.dragHandler) { │ │ │ │ │ - this.dragHandler.setMap(map) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - startBox: function(xy) { │ │ │ │ │ - this.callback("start", []); │ │ │ │ │ - this.zoomBox = OpenLayers.Util.createDiv("zoomBox", { │ │ │ │ │ - x: -9999, │ │ │ │ │ - y: -9999 │ │ │ │ │ - }); │ │ │ │ │ - this.zoomBox.className = this.boxDivClassName; │ │ │ │ │ - this.zoomBox.style.zIndex = this.map.Z_INDEX_BASE["Popup"] - 1; │ │ │ │ │ - this.map.viewPortDiv.appendChild(this.zoomBox); │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, "olDrawBox") │ │ │ │ │ - }, │ │ │ │ │ - moveBox: function(xy) { │ │ │ │ │ - var startX = this.dragHandler.start.x; │ │ │ │ │ - var startY = this.dragHandler.start.y; │ │ │ │ │ - var deltaX = Math.abs(startX - xy.x); │ │ │ │ │ - var deltaY = Math.abs(startY - xy.y); │ │ │ │ │ - var offset = this.getBoxOffsets(); │ │ │ │ │ - this.zoomBox.style.width = deltaX + offset.width + 1 + "px"; │ │ │ │ │ - this.zoomBox.style.height = deltaY + offset.height + 1 + "px"; │ │ │ │ │ - this.zoomBox.style.left = (xy.x < startX ? startX - deltaX - offset.left : startX - offset.left) + "px"; │ │ │ │ │ - this.zoomBox.style.top = (xy.y < startY ? startY - deltaY - offset.top : startY - offset.top) + "px" │ │ │ │ │ - }, │ │ │ │ │ - endBox: function(end) { │ │ │ │ │ - var result; │ │ │ │ │ - if (Math.abs(this.dragHandler.start.x - end.x) > 5 || Math.abs(this.dragHandler.start.y - end.y) > 5) { │ │ │ │ │ - var start = this.dragHandler.start; │ │ │ │ │ - var top = Math.min(start.y, end.y); │ │ │ │ │ - var bottom = Math.max(start.y, end.y); │ │ │ │ │ - var left = Math.min(start.x, end.x); │ │ │ │ │ - var right = Math.max(start.x, end.x); │ │ │ │ │ - result = new OpenLayers.Bounds(left, bottom, right, top) │ │ │ │ │ - } else { │ │ │ │ │ - result = this.dragHandler.start.clone() │ │ │ │ │ - } │ │ │ │ │ - this.removeBox(); │ │ │ │ │ - this.callback("done", [result]) │ │ │ │ │ - }, │ │ │ │ │ - removeBox: function() { │ │ │ │ │ - this.map.viewPortDiv.removeChild(this.zoomBox); │ │ │ │ │ - this.zoomBox = null; │ │ │ │ │ - this.boxOffsets = null; │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDrawBox") │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.dragHandler.activate(); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - if (this.dragHandler.deactivate()) { │ │ │ │ │ - if (this.zoomBox) { │ │ │ │ │ - this.removeBox() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getBoxOffsets: function() { │ │ │ │ │ - if (!this.boxOffsets) { │ │ │ │ │ - var testDiv = document.createElement("div"); │ │ │ │ │ - testDiv.style.position = "absolute"; │ │ │ │ │ - testDiv.style.border = "1px solid black"; │ │ │ │ │ - testDiv.style.width = "3px"; │ │ │ │ │ - document.body.appendChild(testDiv); │ │ │ │ │ - var w3cBoxModel = testDiv.clientWidth == 3; │ │ │ │ │ - document.body.removeChild(testDiv); │ │ │ │ │ - var left = parseInt(OpenLayers.Element.getStyle(this.zoomBox, "border-left-width")); │ │ │ │ │ - var right = parseInt(OpenLayers.Element.getStyle(this.zoomBox, "border-right-width")); │ │ │ │ │ - var top = parseInt(OpenLayers.Element.getStyle(this.zoomBox, "border-top-width")); │ │ │ │ │ - var bottom = parseInt(OpenLayers.Element.getStyle(this.zoomBox, "border-bottom-width")); │ │ │ │ │ - this.boxOffsets = { │ │ │ │ │ - left: left, │ │ │ │ │ - right: right, │ │ │ │ │ - top: top, │ │ │ │ │ - bottom: bottom, │ │ │ │ │ - width: w3cBoxModel === false ? left + right : 0, │ │ │ │ │ - height: w3cBoxModel === false ? top + bottom : 0 │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return this.boxOffsets │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Box" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Marker.Box = OpenLayers.Class(OpenLayers.Marker, { │ │ │ │ │ - bounds: null, │ │ │ │ │ - div: null, │ │ │ │ │ - initialize: function(bounds, borderColor, borderWidth) { │ │ │ │ │ - this.bounds = bounds; │ │ │ │ │ - this.div = OpenLayers.Util.createDiv(); │ │ │ │ │ - this.div.style.overflow = "hidden"; │ │ │ │ │ - this.events = new OpenLayers.Events(this, this.div); │ │ │ │ │ - this.setBorder(borderColor, borderWidth) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.bounds = null; │ │ │ │ │ - this.div = null; │ │ │ │ │ - OpenLayers.Marker.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - setBorder: function(color, width) { │ │ │ │ │ - if (!color) { │ │ │ │ │ - color = "red" │ │ │ │ │ - } │ │ │ │ │ - if (!width) { │ │ │ │ │ - width = 2 │ │ │ │ │ - } │ │ │ │ │ - this.div.style.border = width + "px solid " + color │ │ │ │ │ - }, │ │ │ │ │ - draw: function(px, sz) { │ │ │ │ │ - OpenLayers.Util.modifyDOMElement(this.div, null, px, sz); │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - onScreen: function() { │ │ │ │ │ - var onScreen = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - var screenBounds = this.map.getExtent(); │ │ │ │ │ - onScreen = screenBounds.containsBounds(this.bounds, true, true) │ │ │ │ │ - } │ │ │ │ │ - return onScreen │ │ │ │ │ - }, │ │ │ │ │ - display: function(display) { │ │ │ │ │ - this.div.style.display = display ? "" : "none" │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Marker.Box" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Strategy.Cluster = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - distance: 20, │ │ │ │ │ - threshold: null, │ │ │ │ │ - features: null, │ │ │ │ │ - clusters: null, │ │ │ │ │ - clustering: false, │ │ │ │ │ - resolution: null, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - beforefeaturesadded: this.cacheFeatures, │ │ │ │ │ - featuresremoved: this.clearCache, │ │ │ │ │ - moveend: this.cluster, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.clearCache(); │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - beforefeaturesadded: this.cacheFeatures, │ │ │ │ │ - featuresremoved: this.clearCache, │ │ │ │ │ - moveend: this.cluster, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - cacheFeatures: function(event) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - if (!this.clustering) { │ │ │ │ │ - this.clearCache(); │ │ │ │ │ - this.features = event.features; │ │ │ │ │ - this.cluster(); │ │ │ │ │ - propagate = false │ │ │ │ │ - } │ │ │ │ │ - return propagate │ │ │ │ │ - }, │ │ │ │ │ - clearCache: function() { │ │ │ │ │ - if (!this.clustering) { │ │ │ │ │ - this.features = null │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - cluster: function(event) { │ │ │ │ │ - if ((!event || event.zoomChanged) && this.features) { │ │ │ │ │ - var resolution = this.layer.map.getResolution(); │ │ │ │ │ - if (resolution != this.resolution || !this.clustersExist()) { │ │ │ │ │ - this.resolution = resolution; │ │ │ │ │ - var clusters = []; │ │ │ │ │ - var feature, clustered, cluster; │ │ │ │ │ - for (var i = 0; i < this.features.length; ++i) { │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - clustered = false; │ │ │ │ │ - for (var j = clusters.length - 1; j >= 0; --j) { │ │ │ │ │ - cluster = clusters[j]; │ │ │ │ │ - if (this.shouldCluster(cluster, feature)) { │ │ │ │ │ - this.addToCluster(cluster, feature); │ │ │ │ │ - clustered = true; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!clustered) { │ │ │ │ │ - clusters.push(this.createCluster(this.features[i])) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + var symbolizer = this.selectionSymbolizer; │ │ │ │ │ + var geometryAttribute = geometryAttributes[i]; │ │ │ │ │ + if (geometryAttribute.type.indexOf("Polygon") >= 0) { │ │ │ │ │ + symbolizer = { │ │ │ │ │ + Polygon: this.selectionSymbolizer["Polygon"] │ │ │ │ │ } │ │ │ │ │ - this.clustering = true; │ │ │ │ │ - this.layer.removeAllFeatures(); │ │ │ │ │ - this.clustering = false; │ │ │ │ │ - if (clusters.length > 0) { │ │ │ │ │ - if (this.threshold > 1) { │ │ │ │ │ - var clone = clusters.slice(); │ │ │ │ │ - clusters = []; │ │ │ │ │ - var candidate; │ │ │ │ │ - for (var i = 0, len = clone.length; i < len; ++i) { │ │ │ │ │ - candidate = clone[i]; │ │ │ │ │ - if (candidate.attributes.count < this.threshold) { │ │ │ │ │ - Array.prototype.push.apply(clusters, candidate.cluster) │ │ │ │ │ - } else { │ │ │ │ │ - clusters.push(candidate) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.clustering = true; │ │ │ │ │ - this.layer.addFeatures(clusters); │ │ │ │ │ - this.clustering = false │ │ │ │ │ + } else if (geometryAttribute.type.indexOf("LineString") >= 0) { │ │ │ │ │ + symbolizer = { │ │ │ │ │ + Line: this.selectionSymbolizer["Line"] │ │ │ │ │ } │ │ │ │ │ - this.clusters = clusters │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - clustersExist: function() { │ │ │ │ │ - var exist = false; │ │ │ │ │ - if (this.clusters && this.clusters.length > 0 && this.clusters.length == this.layer.features.length) { │ │ │ │ │ - exist = true; │ │ │ │ │ - for (var i = 0; i < this.clusters.length; ++i) { │ │ │ │ │ - if (this.clusters[i] != this.layer.features[i]) { │ │ │ │ │ - exist = false; │ │ │ │ │ - break │ │ │ │ │ + } else if (geometryAttribute.type.indexOf("Point") >= 0) { │ │ │ │ │ + symbolizer = { │ │ │ │ │ + Point: this.selectionSymbolizer["Point"] │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return exist │ │ │ │ │ - }, │ │ │ │ │ - shouldCluster: function(cluster, feature) { │ │ │ │ │ - var cc = cluster.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - var fc = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - var distance = Math.sqrt(Math.pow(cc.lon - fc.lon, 2) + Math.pow(cc.lat - fc.lat, 2)) / this.resolution; │ │ │ │ │ - return distance <= this.distance │ │ │ │ │ - }, │ │ │ │ │ - addToCluster: function(cluster, feature) { │ │ │ │ │ - cluster.cluster.push(feature); │ │ │ │ │ - cluster.attributes.count += 1 │ │ │ │ │ - }, │ │ │ │ │ - createCluster: function(feature) { │ │ │ │ │ - var center = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - var cluster = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Point(center.lon, center.lat), { │ │ │ │ │ - count: 1 │ │ │ │ │ - }); │ │ │ │ │ - cluster.cluster = [feature]; │ │ │ │ │ - return cluster │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Cluster" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Strategy.Paging = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - features: null, │ │ │ │ │ - length: 10, │ │ │ │ │ - num: null, │ │ │ │ │ - paging: false, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - beforefeaturesadded: this.cacheFeatures, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.clearCache(); │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - beforefeaturesadded: this.cacheFeatures, │ │ │ │ │ - scope: this │ │ │ │ │ + var filter = filters[i]; │ │ │ │ │ + sld.namedLayers[name].userStyles.push({ │ │ │ │ │ + name: "default", │ │ │ │ │ + rules: [new OpenLayers.Rule({ │ │ │ │ │ + symbolizer: symbolizer, │ │ │ │ │ + filter: filter, │ │ │ │ │ + maxScaleDenominator: layer.options.minScale │ │ │ │ │ + })] │ │ │ │ │ }) │ │ │ │ │ } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - cacheFeatures: function(event) { │ │ │ │ │ - if (!this.paging) { │ │ │ │ │ - this.clearCache(); │ │ │ │ │ - this.features = event.features; │ │ │ │ │ - this.pageNext(event) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - clearCache: function() { │ │ │ │ │ - if (this.features) { │ │ │ │ │ - for (var i = 0; i < this.features.length; ++i) { │ │ │ │ │ - this.features[i].destroy() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.features = null; │ │ │ │ │ - this.num = null │ │ │ │ │ - }, │ │ │ │ │ - pageCount: function() { │ │ │ │ │ - var numFeatures = this.features ? this.features.length : 0; │ │ │ │ │ - return Math.ceil(numFeatures / this.length) │ │ │ │ │ - }, │ │ │ │ │ - pageNum: function() { │ │ │ │ │ - return this.num │ │ │ │ │ - }, │ │ │ │ │ - pageLength: function(newLength) { │ │ │ │ │ - if (newLength && newLength > 0) { │ │ │ │ │ - this.length = newLength │ │ │ │ │ - } │ │ │ │ │ - return this.length │ │ │ │ │ + return new OpenLayers.Format.SLD({ │ │ │ │ │ + srsName: this.map.getProjection() │ │ │ │ │ + }).write(sld) │ │ │ │ │ }, │ │ │ │ │ - pageNext: function(event) { │ │ │ │ │ - var changed = false; │ │ │ │ │ - if (this.features) { │ │ │ │ │ - if (this.num === null) { │ │ │ │ │ - this.num = -1 │ │ │ │ │ - } │ │ │ │ │ - var start = (this.num + 1) * this.length; │ │ │ │ │ - changed = this.page(start, event) │ │ │ │ │ + parseDescribeLayer: function(request) { │ │ │ │ │ + var format = new OpenLayers.Format.WMSDescribeLayer; │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ } │ │ │ │ │ - return changed │ │ │ │ │ - }, │ │ │ │ │ - pagePrevious: function() { │ │ │ │ │ - var changed = false; │ │ │ │ │ - if (this.features) { │ │ │ │ │ - if (this.num === null) { │ │ │ │ │ - this.num = this.pageCount() │ │ │ │ │ + var describeLayer = format.read(doc); │ │ │ │ │ + var typeNames = []; │ │ │ │ │ + var url = null; │ │ │ │ │ + for (var i = 0, len = describeLayer.length; i < len; i++) { │ │ │ │ │ + if (describeLayer[i].owsType == "WFS") { │ │ │ │ │ + typeNames.push(describeLayer[i].typeName); │ │ │ │ │ + url = describeLayer[i].owsURL │ │ │ │ │ } │ │ │ │ │ - var start = (this.num - 1) * this.length; │ │ │ │ │ - changed = this.page(start) │ │ │ │ │ } │ │ │ │ │ - return changed │ │ │ │ │ - }, │ │ │ │ │ - page: function(start, event) { │ │ │ │ │ - var changed = false; │ │ │ │ │ - if (this.features) { │ │ │ │ │ - if (start >= 0 && start < this.features.length) { │ │ │ │ │ - var num = Math.floor(start / this.length); │ │ │ │ │ - if (num != this.num) { │ │ │ │ │ - this.paging = true; │ │ │ │ │ - var features = this.features.slice(start, start + this.length); │ │ │ │ │ - this.layer.removeFeatures(this.layer.features); │ │ │ │ │ - this.num = num; │ │ │ │ │ - if (event && event.features) { │ │ │ │ │ - event.features = features │ │ │ │ │ - } else { │ │ │ │ │ - this.layer.addFeatures(features) │ │ │ │ │ - } │ │ │ │ │ - this.paging = false; │ │ │ │ │ - changed = true │ │ │ │ │ + var options = { │ │ │ │ │ + url: url, │ │ │ │ │ + params: { │ │ │ │ │ + SERVICE: "WFS", │ │ │ │ │ + TYPENAME: typeNames.toString(), │ │ │ │ │ + REQUEST: "DescribeFeatureType", │ │ │ │ │ + VERSION: "1.0.0" │ │ │ │ │ + }, │ │ │ │ │ + callback: function(request) { │ │ │ │ │ + var format = new OpenLayers.Format.WFSDescribeFeatureType; │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return changed │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Paging" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Strategy.Filter = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - filter: null, │ │ │ │ │ - cache: null, │ │ │ │ │ - caching: false, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.cache = []; │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - beforefeaturesadded: this.handleAdd, │ │ │ │ │ - beforefeaturesremoved: this.handleRemove, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - this.cache = null; │ │ │ │ │ - if (this.layer && this.layer.events) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - beforefeaturesadded: this.handleAdd, │ │ │ │ │ - beforefeaturesremoved: this.handleRemove, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Strategy.prototype.deactivate.apply(this, arguments) │ │ │ │ │ + var describeFeatureType = format.read(doc); │ │ │ │ │ + this.control.wfsCache[this.layer.id] = describeFeatureType; │ │ │ │ │ + this.control._queue && this.control.applySelection() │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Request.GET(options) │ │ │ │ │ }, │ │ │ │ │ - handleAdd: function(event) { │ │ │ │ │ - if (!this.caching && this.filter) { │ │ │ │ │ - var features = event.features; │ │ │ │ │ - event.features = []; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0, ii = features.length; i < ii; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (this.filter.evaluate(feature)) { │ │ │ │ │ - event.features.push(feature) │ │ │ │ │ - } else { │ │ │ │ │ - this.cache.push(feature) │ │ │ │ │ + getGeometryAttributes: function(layer) { │ │ │ │ │ + var result = []; │ │ │ │ │ + var cache = this.wfsCache[layer.id]; │ │ │ │ │ + for (var i = 0, len = cache.featureTypes.length; i < len; i++) { │ │ │ │ │ + var typeName = cache.featureTypes[i]; │ │ │ │ │ + var properties = typeName.properties; │ │ │ │ │ + for (var j = 0, lenj = properties.length; j < lenj; j++) { │ │ │ │ │ + var property = properties[j]; │ │ │ │ │ + var type = property.type; │ │ │ │ │ + if (type.indexOf("LineString") >= 0 || type.indexOf("GeometryAssociationType") >= 0 || type.indexOf("GeometryPropertyType") >= 0 || type.indexOf("Point") >= 0 || type.indexOf("Polygon") >= 0) { │ │ │ │ │ + result.push(property) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - handleRemove: function(event) { │ │ │ │ │ - if (!this.caching) { │ │ │ │ │ - this.cache = [] │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setFilter: function(filter) { │ │ │ │ │ - this.filter = filter; │ │ │ │ │ - var previousCache = this.cache; │ │ │ │ │ - this.cache = []; │ │ │ │ │ - this.handleAdd({ │ │ │ │ │ - features: this.layer.features │ │ │ │ │ - }); │ │ │ │ │ - if (this.cache.length > 0) { │ │ │ │ │ - this.caching = true; │ │ │ │ │ - this.layer.removeFeatures(this.cache.slice()); │ │ │ │ │ - this.caching = false │ │ │ │ │ - } │ │ │ │ │ - if (previousCache.length > 0) { │ │ │ │ │ - var event = { │ │ │ │ │ - features: previousCache │ │ │ │ │ - }; │ │ │ │ │ - this.handleAdd(event); │ │ │ │ │ - if (event.features.length > 0) { │ │ │ │ │ - this.caching = true; │ │ │ │ │ - this.layer.addFeatures(event.features); │ │ │ │ │ - this.caching = false │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Filter" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Strategy.Refresh = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - force: false, │ │ │ │ │ - interval: 0, │ │ │ │ │ - timer: null, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - if (this.layer.visibility === true) { │ │ │ │ │ - this.start() │ │ │ │ │ - } │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - visibilitychanged: this.reset, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.stop(); │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - visibilitychanged: this.reset, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - reset: function() { │ │ │ │ │ - if (this.layer.visibility === true) { │ │ │ │ │ - this.start() │ │ │ │ │ - } else { │ │ │ │ │ - this.stop() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - start: function() { │ │ │ │ │ - if (this.interval && typeof this.interval === "number" && this.interval > 0) { │ │ │ │ │ - this.timer = window.setInterval(OpenLayers.Function.bind(this.refresh, this), this.interval) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - refresh: function() { │ │ │ │ │ - if (this.layer && this.layer.refresh && typeof this.layer.refresh == "function") { │ │ │ │ │ - this.layer.refresh({ │ │ │ │ │ - force: this.force │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - stop: function() { │ │ │ │ │ - if (this.timer !== null) { │ │ │ │ │ - window.clearInterval(this.timer); │ │ │ │ │ - this.timer = null │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Refresh" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Strategy.Save = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - events: null, │ │ │ │ │ - auto: false, │ │ │ │ │ - timer: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Strategy.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.events = new OpenLayers.Events(this) │ │ │ │ │ + return result │ │ │ │ │ }, │ │ │ │ │ activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + var activated = OpenLayers.Control.prototype.activate.call(this); │ │ │ │ │ if (activated) { │ │ │ │ │ - if (this.auto) { │ │ │ │ │ - if (typeof this.auto === "number") { │ │ │ │ │ - this.timer = window.setInterval(OpenLayers.Function.bind(this.save, this), this.auto * 1e3) │ │ │ │ │ - } else { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - featureadded: this.triggerSave, │ │ │ │ │ - afterfeaturemodified: this.triggerSave, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.layers[i]; │ │ │ │ │ + if (layer && !this.wfsCache[layer.id]) { │ │ │ │ │ + var options = { │ │ │ │ │ + url: layer.url, │ │ │ │ │ + params: { │ │ │ │ │ + SERVICE: "WMS", │ │ │ │ │ + VERSION: layer.params.VERSION, │ │ │ │ │ + LAYERS: layer.params.LAYERS, │ │ │ │ │ + REQUEST: "DescribeLayer" │ │ │ │ │ + }, │ │ │ │ │ + callback: this.parseDescribeLayer, │ │ │ │ │ + scope: { │ │ │ │ │ + layer: layer, │ │ │ │ │ + control: this │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Request.GET(options) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ return activated │ │ │ │ │ }, │ │ │ │ │ deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + var deactivated = OpenLayers.Control.prototype.deactivate.call(this); │ │ │ │ │ if (deactivated) { │ │ │ │ │ - if (this.auto) { │ │ │ │ │ - if (typeof this.auto === "number") { │ │ │ │ │ - window.clearInterval(this.timer) │ │ │ │ │ - } else { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - featureadded: this.triggerSave, │ │ │ │ │ - afterfeaturemodified: this.triggerSave, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.layers[i]; │ │ │ │ │ + if (layer && this.clearOnDeactivate === true) { │ │ │ │ │ + var layerCache = this.layerCache; │ │ │ │ │ + var selectionLayer = layerCache[layer.id]; │ │ │ │ │ + if (selectionLayer) { │ │ │ │ │ + layer.events.un({ │ │ │ │ │ + visibilitychanged: this.coupleLayerVisiblity, │ │ │ │ │ + scope: selectionLayer │ │ │ │ │ + }); │ │ │ │ │ + selectionLayer.destroy(); │ │ │ │ │ + delete layerCache[layer.id] │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ return deactivated │ │ │ │ │ }, │ │ │ │ │ - triggerSave: function(event) { │ │ │ │ │ - var feature = event.feature; │ │ │ │ │ - if (feature.state === OpenLayers.State.INSERT || feature.state === OpenLayers.State.UPDATE || feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ - this.save([event.feature]) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - save: function(features) { │ │ │ │ │ - if (!features) { │ │ │ │ │ - features = this.layer.features │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("start", { │ │ │ │ │ - features: features │ │ │ │ │ - }); │ │ │ │ │ - var remote = this.layer.projection; │ │ │ │ │ - var local = this.layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var len = features.length; │ │ │ │ │ - var clones = new Array(len); │ │ │ │ │ - var orig, clone; │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - orig = features[i]; │ │ │ │ │ - clone = orig.clone(); │ │ │ │ │ - clone.fid = orig.fid; │ │ │ │ │ - clone.state = orig.state; │ │ │ │ │ - if (orig.url) { │ │ │ │ │ - clone.url = orig.url │ │ │ │ │ - } │ │ │ │ │ - clone._original = orig; │ │ │ │ │ - clone.geometry.transform(local, remote); │ │ │ │ │ - clones[i] = clone │ │ │ │ │ - } │ │ │ │ │ - features = clones │ │ │ │ │ - } │ │ │ │ │ - this.layer.protocol.commit(features, { │ │ │ │ │ - callback: this.onCommit, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - onCommit: function(response) { │ │ │ │ │ - var evt = { │ │ │ │ │ - response: response │ │ │ │ │ - }; │ │ │ │ │ - if (response.success()) { │ │ │ │ │ - var features = response.reqFeatures; │ │ │ │ │ - var state, feature; │ │ │ │ │ - var destroys = []; │ │ │ │ │ - var insertIds = response.insertIds || []; │ │ │ │ │ - var j = 0; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - feature = feature._original || feature; │ │ │ │ │ - state = feature.state; │ │ │ │ │ - if (state) { │ │ │ │ │ - if (state == OpenLayers.State.DELETE) { │ │ │ │ │ - destroys.push(feature) │ │ │ │ │ - } else if (state == OpenLayers.State.INSERT) { │ │ │ │ │ - feature.fid = insertIds[j]; │ │ │ │ │ - ++j │ │ │ │ │ - } │ │ │ │ │ - feature.state = null │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (destroys.length > 0) { │ │ │ │ │ - this.layer.destroyFeatures(destroys) │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("success", evt) │ │ │ │ │ + setLayers: function(layers) { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.layers = layers; │ │ │ │ │ + this.activate() │ │ │ │ │ } else { │ │ │ │ │ - this.events.triggerEvent("fail", evt) │ │ │ │ │ + this.layers = layers │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Save" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Strategy.Fixed = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - preload: false, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - refresh: this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - if (this.layer.visibility == true || this.preload) { │ │ │ │ │ - this.load() │ │ │ │ │ + createFilter: function(geometryAttribute, geometry) { │ │ │ │ │ + var filter = null; │ │ │ │ │ + if (this.handler instanceof OpenLayers.Handler.RegularPolygon) { │ │ │ │ │ + if (this.handler.irregular === true) { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry.getBounds() │ │ │ │ │ + }) │ │ │ │ │ } else { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - visibilitychanged: this.load, │ │ │ │ │ - scope: this │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry │ │ │ │ │ }) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - refresh: this.load, │ │ │ │ │ - visibilitychanged: this.load, │ │ │ │ │ - scope: this │ │ │ │ │ + } else if (this.handler instanceof OpenLayers.Handler.Polygon) { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry │ │ │ │ │ }) │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - load: function(options) { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - layer.events.triggerEvent("loadstart", { │ │ │ │ │ - filter: layer.filter │ │ │ │ │ - }); │ │ │ │ │ - layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: this.merge, │ │ │ │ │ - filter: layer.filter, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options)); │ │ │ │ │ - layer.events.un({ │ │ │ │ │ - visibilitychanged: this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - merge: function(resp) { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - layer.destroyFeatures(); │ │ │ │ │ - var features = resp.features; │ │ │ │ │ - if (features && features.length > 0) { │ │ │ │ │ - var remote = layer.projection; │ │ │ │ │ - var local = layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var geom; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - geom = features[i].geometry; │ │ │ │ │ - if (geom) { │ │ │ │ │ - geom.transform(remote, local) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + } else if (this.handler instanceof OpenLayers.Handler.Path) { │ │ │ │ │ + if (geometryAttribute.type.indexOf("Point") >= 0) { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + distance: this.map.getExtent().getWidth() * .01, │ │ │ │ │ + distanceUnits: this.map.getUnits(), │ │ │ │ │ + value: geometry │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } else if (this.handler instanceof OpenLayers.Handler.Click) { │ │ │ │ │ + if (geometryAttribute.type.indexOf("Polygon") >= 0) { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + distance: this.map.getExtent().getWidth() * .01, │ │ │ │ │ + distanceUnits: this.map.getUnits(), │ │ │ │ │ + value: geometry │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - layer.addFeatures(features) │ │ │ │ │ - } │ │ │ │ │ - layer.events.triggerEvent("loadend", { │ │ │ │ │ - response: resp │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Fixed" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Strategy.BBOX = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - bounds: null, │ │ │ │ │ - resolution: null, │ │ │ │ │ - ratio: 2, │ │ │ │ │ - resFactor: null, │ │ │ │ │ - response: null, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - moveend: this.update, │ │ │ │ │ - refresh: this.update, │ │ │ │ │ - visibilitychanged: this.update, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.update() │ │ │ │ │ - } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - moveend: this.update, │ │ │ │ │ - refresh: this.update, │ │ │ │ │ - visibilitychanged: this.update, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - update: function(options) { │ │ │ │ │ - var mapBounds = this.getMapBounds(); │ │ │ │ │ - if (mapBounds !== null && (options && options.force || this.layer.visibility && this.layer.calculateInRange() && this.invalidBounds(mapBounds))) { │ │ │ │ │ - this.calculateBounds(mapBounds); │ │ │ │ │ - this.resolution = this.layer.map.getResolution(); │ │ │ │ │ - this.triggerRead(options) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getMapBounds: function() { │ │ │ │ │ - if (this.layer.map === null) { │ │ │ │ │ - return null │ │ │ │ │ - } │ │ │ │ │ - var bounds = this.layer.map.getExtent(); │ │ │ │ │ - if (bounds && !this.layer.projection.equals(this.layer.map.getProjectionObject())) { │ │ │ │ │ - bounds = bounds.clone().transform(this.layer.map.getProjectionObject(), this.layer.projection) │ │ │ │ │ - } │ │ │ │ │ - return bounds │ │ │ │ │ - }, │ │ │ │ │ - invalidBounds: function(mapBounds) { │ │ │ │ │ - if (!mapBounds) { │ │ │ │ │ - mapBounds = this.getMapBounds() │ │ │ │ │ - } │ │ │ │ │ - var invalid = !this.bounds || !this.bounds.containsBounds(mapBounds); │ │ │ │ │ - if (!invalid && this.resFactor) { │ │ │ │ │ - var ratio = this.resolution / this.layer.map.getResolution(); │ │ │ │ │ - invalid = ratio >= this.resFactor || ratio <= 1 / this.resFactor │ │ │ │ │ - } │ │ │ │ │ - return invalid │ │ │ │ │ - }, │ │ │ │ │ - calculateBounds: function(mapBounds) { │ │ │ │ │ - if (!mapBounds) { │ │ │ │ │ - mapBounds = this.getMapBounds() │ │ │ │ │ - } │ │ │ │ │ - var center = mapBounds.getCenterLonLat(); │ │ │ │ │ - var dataWidth = mapBounds.getWidth() * this.ratio; │ │ │ │ │ - var dataHeight = mapBounds.getHeight() * this.ratio; │ │ │ │ │ - this.bounds = new OpenLayers.Bounds(center.lon - dataWidth / 2, center.lat - dataHeight / 2, center.lon + dataWidth / 2, center.lat + dataHeight / 2) │ │ │ │ │ - }, │ │ │ │ │ - triggerRead: function(options) { │ │ │ │ │ - if (this.response && !(options && options.noAbort === true)) { │ │ │ │ │ - this.layer.protocol.abort(this.response); │ │ │ │ │ - this.layer.events.triggerEvent("loadend") │ │ │ │ │ - } │ │ │ │ │ - var evt = { │ │ │ │ │ - filter: this.createFilter() │ │ │ │ │ - }; │ │ │ │ │ - this.layer.events.triggerEvent("loadstart", evt); │ │ │ │ │ - this.response = this.layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ - filter: evt.filter, │ │ │ │ │ - callback: this.merge, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options)) │ │ │ │ │ - }, │ │ │ │ │ - createFilter: function() { │ │ │ │ │ - var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ - value: this.bounds, │ │ │ │ │ - projection: this.layer.projection │ │ │ │ │ - }); │ │ │ │ │ - if (this.layer.filter) { │ │ │ │ │ - filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ - type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ - filters: [this.layer.filter, filter] │ │ │ │ │ - }) │ │ │ │ │ } │ │ │ │ │ return filter │ │ │ │ │ }, │ │ │ │ │ - merge: function(resp) { │ │ │ │ │ - this.layer.destroyFeatures(); │ │ │ │ │ - if (resp.success()) { │ │ │ │ │ - var features = resp.features; │ │ │ │ │ - if (features && features.length > 0) { │ │ │ │ │ - var remote = this.layer.projection; │ │ │ │ │ - var local = this.layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var geom; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - geom = features[i].geometry; │ │ │ │ │ - if (geom) { │ │ │ │ │ - geom.transform(remote, local) │ │ │ │ │ + select: function(geometry) { │ │ │ │ │ + this._queue = function() { │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.layers[i]; │ │ │ │ │ + var geometryAttributes = this.getGeometryAttributes(layer); │ │ │ │ │ + var filters = []; │ │ │ │ │ + for (var j = 0, lenj = geometryAttributes.length; j < lenj; j++) { │ │ │ │ │ + var geometryAttribute = geometryAttributes[j]; │ │ │ │ │ + if (geometryAttribute !== null) { │ │ │ │ │ + if (!(geometry instanceof OpenLayers.Geometry)) { │ │ │ │ │ + var point = this.map.getLonLatFromPixel(geometry.xy); │ │ │ │ │ + geometry = new OpenLayers.Geometry.Point(point.lon, point.lat) │ │ │ │ │ + } │ │ │ │ │ + var filter = this.createFilter(geometryAttribute, geometry); │ │ │ │ │ + if (filter !== null) { │ │ │ │ │ + filters.push(filter) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.layer.addFeatures(features) │ │ │ │ │ + var selectionLayer = this.createSelectionLayer(layer); │ │ │ │ │ + this.events.triggerEvent("selected", { │ │ │ │ │ + layer: layer, │ │ │ │ │ + filters: filters │ │ │ │ │ + }); │ │ │ │ │ + var sld = this.createSLD(layer, filters, geometryAttributes); │ │ │ │ │ + selectionLayer.mergeNewParams({ │ │ │ │ │ + SLD_BODY: sld │ │ │ │ │ + }); │ │ │ │ │ + delete this._queue │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + this.applySelection() │ │ │ │ │ + }, │ │ │ │ │ + applySelection: function() { │ │ │ │ │ + var canApply = true; │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + if (!this.wfsCache[this.layers[i].id]) { │ │ │ │ │ + canApply = false; │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - this.bounds = null │ │ │ │ │ } │ │ │ │ │ - this.response = null; │ │ │ │ │ - this.layer.events.triggerEvent("loadend", { │ │ │ │ │ - response: resp │ │ │ │ │ - }) │ │ │ │ │ + canApply && this._queue.call(this) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.BBOX" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.SLDSelect" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.Panel = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - controls: null, │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - defaultControl: null, │ │ │ │ │ - saveState: false, │ │ │ │ │ - allowDepress: false, │ │ │ │ │ - activeState: null, │ │ │ │ │ +OpenLayers.Control.PanZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + slideFactor: 50, │ │ │ │ │ + slideRatio: null, │ │ │ │ │ + buttons: null, │ │ │ │ │ + position: null, │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.controls = []; │ │ │ │ │ - this.activeState = {} │ │ │ │ │ + this.position = new OpenLayers.Pixel(OpenLayers.Control.PanZoom.X, OpenLayers.Control.PanZoom.Y); │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ if (this.map) { │ │ │ │ │ this.map.events.unregister("buttonclick", this, this.onButtonClick) │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - for (var ctl, i = this.controls.length - 1; i >= 0; i--) { │ │ │ │ │ - ctl = this.controls[i]; │ │ │ │ │ - if (ctl.events) { │ │ │ │ │ - ctl.events.un({ │ │ │ │ │ - activate: this.iconOn, │ │ │ │ │ - deactivate: this.iconOff │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - ctl.panel_div = null │ │ │ │ │ - } │ │ │ │ │ - this.activeState = null │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - control = this.controls[i]; │ │ │ │ │ - if (control === this.defaultControl || this.saveState && this.activeState[control.id]) { │ │ │ │ │ - control.activate() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.saveState === true) { │ │ │ │ │ - this.defaultControl = null │ │ │ │ │ - } │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ + this.removeButtons(); │ │ │ │ │ + this.buttons = null; │ │ │ │ │ + this.position = null; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - control = this.controls[i]; │ │ │ │ │ - this.activeState[control.id] = control.deactivate() │ │ │ │ │ - } │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ }, │ │ │ │ │ - draw: function() { │ │ │ │ │ + draw: function(px) { │ │ │ │ │ OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (this.outsideViewport) { │ │ │ │ │ - this.events.attachToElement(this.div); │ │ │ │ │ - this.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ - } else { │ │ │ │ │ - this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ - } │ │ │ │ │ - this.addControlsToMap(this.controls); │ │ │ │ │ + px = this.position; │ │ │ │ │ + this.buttons = []; │ │ │ │ │ + var sz = { │ │ │ │ │ + w: 18, │ │ │ │ │ + h: 18 │ │ │ │ │ + }; │ │ │ │ │ + var centered = new OpenLayers.Pixel(px.x + sz.w / 2, px.y); │ │ │ │ │ + this._addButton("panup", "north-mini.png", centered, sz); │ │ │ │ │ + px.y = centered.y + sz.h; │ │ │ │ │ + this._addButton("panleft", "west-mini.png", px, sz); │ │ │ │ │ + this._addButton("panright", "east-mini.png", px.add(sz.w, 0), sz); │ │ │ │ │ + this._addButton("pandown", "south-mini.png", centered.add(0, sz.h * 2), sz); │ │ │ │ │ + this._addButton("zoomin", "zoom-plus-mini.png", centered.add(0, sz.h * 3 + 5), sz); │ │ │ │ │ + this._addButton("zoomworld", "zoom-world-mini.png", centered.add(0, sz.h * 4 + 5), sz); │ │ │ │ │ + this._addButton("zoomout", "zoom-minus-mini.png", centered.add(0, sz.h * 5 + 5), sz); │ │ │ │ │ return this.div │ │ │ │ │ }, │ │ │ │ │ - redraw: function() { │ │ │ │ │ - for (var l = this.div.childNodes.length, i = l - 1; i >= 0; i--) { │ │ │ │ │ - this.div.removeChild(this.div.childNodes[i]) │ │ │ │ │ - } │ │ │ │ │ - this.div.innerHTML = ""; │ │ │ │ │ - if (this.active) { │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - this.div.appendChild(this.controls[i].panel_div) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - activateControl: function(control) { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - if (control.type == OpenLayers.Control.TYPE_BUTTON) { │ │ │ │ │ - control.trigger(); │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (control.type == OpenLayers.Control.TYPE_TOGGLE) { │ │ │ │ │ - if (control.active) { │ │ │ │ │ - control.deactivate() │ │ │ │ │ - } else { │ │ │ │ │ - control.activate() │ │ │ │ │ - } │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (this.allowDepress && control.active) { │ │ │ │ │ - control.deactivate() │ │ │ │ │ - } else { │ │ │ │ │ - var c; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - c = this.controls[i]; │ │ │ │ │ - if (c != control && (c.type === OpenLayers.Control.TYPE_TOOL || c.type == null)) { │ │ │ │ │ - c.deactivate() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - control.activate() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - addControls: function(controls) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(controls)) { │ │ │ │ │ - controls = [controls] │ │ │ │ │ - } │ │ │ │ │ - this.controls = this.controls.concat(controls); │ │ │ │ │ - for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ - var control = controls[i], │ │ │ │ │ - element = this.createControlMarkup(control); │ │ │ │ │ - OpenLayers.Element.addClass(element, control.displayClass + "ItemInactive"); │ │ │ │ │ - OpenLayers.Element.addClass(element, "olButton"); │ │ │ │ │ - if (control.title != "" && !element.title) { │ │ │ │ │ - element.title = control.title │ │ │ │ │ - } │ │ │ │ │ - control.panel_div = element │ │ │ │ │ - } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.addControlsToMap(controls); │ │ │ │ │ - this.redraw() │ │ │ │ │ - } │ │ │ │ │ + _addButton: function(id, img, xy, sz) { │ │ │ │ │ + var imgLocation = OpenLayers.Util.getImageLocation(img); │ │ │ │ │ + var btn = OpenLayers.Util.createAlphaImageDiv(this.id + "_" + id, xy, sz, imgLocation, "absolute"); │ │ │ │ │ + btn.style.cursor = "pointer"; │ │ │ │ │ + this.div.appendChild(btn); │ │ │ │ │ + btn.action = id; │ │ │ │ │ + btn.className = "olButton"; │ │ │ │ │ + this.buttons.push(btn); │ │ │ │ │ + return btn │ │ │ │ │ }, │ │ │ │ │ - createControlMarkup: function(control) { │ │ │ │ │ - return document.createElement("div") │ │ │ │ │ + _removeButton: function(btn) { │ │ │ │ │ + this.div.removeChild(btn); │ │ │ │ │ + OpenLayers.Util.removeItem(this.buttons, btn) │ │ │ │ │ }, │ │ │ │ │ - addControlsToMap: function(controls) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ - control = controls[i]; │ │ │ │ │ - if (control.autoActivate === true) { │ │ │ │ │ - control.autoActivate = false; │ │ │ │ │ - this.map.addControl(control); │ │ │ │ │ - control.autoActivate = true │ │ │ │ │ - } else { │ │ │ │ │ - this.map.addControl(control); │ │ │ │ │ - control.deactivate() │ │ │ │ │ - } │ │ │ │ │ - control.events.on({ │ │ │ │ │ - activate: this.iconOn, │ │ │ │ │ - deactivate: this.iconOff │ │ │ │ │ - }) │ │ │ │ │ + removeButtons: function() { │ │ │ │ │ + for (var i = this.buttons.length - 1; i >= 0; --i) { │ │ │ │ │ + this._removeButton(this.buttons[i]) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - iconOn: function() { │ │ │ │ │ - var d = this.panel_div; │ │ │ │ │ - var re = new RegExp("\\b(" + this.displayClass + "Item)Inactive\\b"); │ │ │ │ │ - d.className = d.className.replace(re, "$1Active") │ │ │ │ │ - }, │ │ │ │ │ - iconOff: function() { │ │ │ │ │ - var d = this.panel_div; │ │ │ │ │ - var re = new RegExp("\\b(" + this.displayClass + "Item)Active\\b"); │ │ │ │ │ - d.className = d.className.replace(re, "$1Inactive") │ │ │ │ │ - }, │ │ │ │ │ onButtonClick: function(evt) { │ │ │ │ │ - var controls = this.controls, │ │ │ │ │ - button = evt.buttonElement; │ │ │ │ │ - for (var i = controls.length - 1; i >= 0; --i) { │ │ │ │ │ - if (controls[i].panel_div === button) { │ │ │ │ │ - this.activateControl(controls[i]); │ │ │ │ │ + var btn = evt.buttonElement; │ │ │ │ │ + switch (btn.action) { │ │ │ │ │ + case "panup": │ │ │ │ │ + this.map.pan(0, -this.getSlideFactor("h")); │ │ │ │ │ + break; │ │ │ │ │ + case "pandown": │ │ │ │ │ + this.map.pan(0, this.getSlideFactor("h")); │ │ │ │ │ + break; │ │ │ │ │ + case "panleft": │ │ │ │ │ + this.map.pan(-this.getSlideFactor("w"), 0); │ │ │ │ │ + break; │ │ │ │ │ + case "panright": │ │ │ │ │ + this.map.pan(this.getSlideFactor("w"), 0); │ │ │ │ │ + break; │ │ │ │ │ + case "zoomin": │ │ │ │ │ + this.map.zoomIn(); │ │ │ │ │ + break; │ │ │ │ │ + case "zoomout": │ │ │ │ │ + this.map.zoomOut(); │ │ │ │ │ + break; │ │ │ │ │ + case "zoomworld": │ │ │ │ │ + this.map.zoomToMaxExtent(); │ │ │ │ │ break │ │ │ │ │ - } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - getControlsBy: function(property, match) { │ │ │ │ │ - var test = typeof match.test == "function"; │ │ │ │ │ - var found = OpenLayers.Array.filter(this.controls, function(item) { │ │ │ │ │ - return item[property] == match || test && match.test(item[property]) │ │ │ │ │ - }); │ │ │ │ │ - return found │ │ │ │ │ - }, │ │ │ │ │ - getControlsByName: function(match) { │ │ │ │ │ - return this.getControlsBy("name", match) │ │ │ │ │ - }, │ │ │ │ │ - getControlsByClass: function(match) { │ │ │ │ │ - return this.getControlsBy("CLASS_NAME", match) │ │ │ │ │ + getSlideFactor: function(dim) { │ │ │ │ │ + return this.slideRatio ? this.map.getSize()[dim] * this.slideRatio : this.slideFactor │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Panel" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.PanZoom" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.ZoomBox = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ - out: false, │ │ │ │ │ - keyMask: null, │ │ │ │ │ - alwaysZoom: false, │ │ │ │ │ - zoomOnClick: true, │ │ │ │ │ - draw: function() { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Box(this, { │ │ │ │ │ - done: this.zoomBox │ │ │ │ │ - }, { │ │ │ │ │ - keyMask: this.keyMask │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - zoomBox: function(position) { │ │ │ │ │ - if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ - var bounds, targetCenterPx = position.getCenterPixel(); │ │ │ │ │ - if (!this.out) { │ │ │ │ │ - var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.left, │ │ │ │ │ - y: position.bottom │ │ │ │ │ - }); │ │ │ │ │ - var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.right, │ │ │ │ │ - y: position.top │ │ │ │ │ - }); │ │ │ │ │ - bounds = new OpenLayers.Bounds(minXY.lon, minXY.lat, maxXY.lon, maxXY.lat) │ │ │ │ │ - } else { │ │ │ │ │ - var pixWidth = position.right - position.left; │ │ │ │ │ - var pixHeight = position.bottom - position.top; │ │ │ │ │ - var zoomFactor = Math.min(this.map.size.h / pixHeight, this.map.size.w / pixWidth); │ │ │ │ │ - var extent = this.map.getExtent(); │ │ │ │ │ - var center = this.map.getLonLatFromPixel(targetCenterPx); │ │ │ │ │ - var xmin = center.lon - extent.getWidth() / 2 * zoomFactor; │ │ │ │ │ - var xmax = center.lon + extent.getWidth() / 2 * zoomFactor; │ │ │ │ │ - var ymin = center.lat - extent.getHeight() / 2 * zoomFactor; │ │ │ │ │ - var ymax = center.lat + extent.getHeight() / 2 * zoomFactor; │ │ │ │ │ - bounds = new OpenLayers.Bounds(xmin, ymin, xmax, ymax) │ │ │ │ │ - } │ │ │ │ │ - var lastZoom = this.map.getZoom(), │ │ │ │ │ - size = this.map.getSize(), │ │ │ │ │ - centerPx = { │ │ │ │ │ - x: size.w / 2, │ │ │ │ │ - y: size.h / 2 │ │ │ │ │ - }, │ │ │ │ │ - zoom = this.map.getZoomForExtent(bounds), │ │ │ │ │ - oldRes = this.map.getResolution(), │ │ │ │ │ - newRes = this.map.getResolutionForZoom(zoom); │ │ │ │ │ - if (oldRes == newRes) { │ │ │ │ │ - this.map.setCenter(this.map.getLonLatFromPixel(targetCenterPx)) │ │ │ │ │ - } else { │ │ │ │ │ - var zoomOriginPx = { │ │ │ │ │ - x: (oldRes * targetCenterPx.x - newRes * centerPx.x) / (oldRes - newRes), │ │ │ │ │ - y: (oldRes * targetCenterPx.y - newRes * centerPx.y) / (oldRes - newRes) │ │ │ │ │ - }; │ │ │ │ │ - this.map.zoomTo(zoom, zoomOriginPx) │ │ │ │ │ - } │ │ │ │ │ - if (lastZoom == this.map.getZoom() && this.alwaysZoom == true) { │ │ │ │ │ - this.map.zoomTo(lastZoom + (this.out ? -1 : 1)) │ │ │ │ │ - } │ │ │ │ │ - } else if (this.zoomOnClick) { │ │ │ │ │ - if (!this.out) { │ │ │ │ │ - this.map.zoomTo(this.map.getZoom() + 1, position) │ │ │ │ │ - } else { │ │ │ │ │ - this.map.zoomTo(this.map.getZoom() - 1, position) │ │ │ │ │ - } │ │ │ │ │ +OpenLayers.Control.PanZoom.X = 4; │ │ │ │ │ +OpenLayers.Control.PanZoom.Y = 4; │ │ │ │ │ +OpenLayers.Control.WMSGetFeatureInfo = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + hover: false, │ │ │ │ │ + drillDown: false, │ │ │ │ │ + maxFeatures: 10, │ │ │ │ │ + clickCallback: "click", │ │ │ │ │ + output: "features", │ │ │ │ │ + layers: null, │ │ │ │ │ + queryVisible: false, │ │ │ │ │ + url: null, │ │ │ │ │ + layerUrls: null, │ │ │ │ │ + infoFormat: "text/html", │ │ │ │ │ + vendorParams: {}, │ │ │ │ │ + format: null, │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + handler: null, │ │ │ │ │ + hoverRequest: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!this.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.WMSGetFeatureInfo(options.formatOptions) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomBox" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.DragPan = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ - panned: false, │ │ │ │ │ - interval: 0, │ │ │ │ │ - documentDrag: false, │ │ │ │ │ - kinetic: null, │ │ │ │ │ - enableKinetic: true, │ │ │ │ │ - kineticInterval: 10, │ │ │ │ │ - draw: function() { │ │ │ │ │ - if (this.enableKinetic && OpenLayers.Kinetic) { │ │ │ │ │ - var config = { │ │ │ │ │ - interval: this.kineticInterval │ │ │ │ │ - }; │ │ │ │ │ - if (typeof this.enableKinetic === "object") { │ │ │ │ │ - config = OpenLayers.Util.extend(config, this.enableKinetic) │ │ │ │ │ - } │ │ │ │ │ - this.kinetic = new OpenLayers.Kinetic(config) │ │ │ │ │ + if (this.drillDown === true) { │ │ │ │ │ + this.hover = false │ │ │ │ │ } │ │ │ │ │ - this.handler = new OpenLayers.Handler.Drag(this, { │ │ │ │ │ - move: this.panMap, │ │ │ │ │ - done: this.panMapDone, │ │ │ │ │ - down: this.panMapStart │ │ │ │ │ - }, { │ │ │ │ │ - interval: this.interval, │ │ │ │ │ - documentDrag: this.documentDrag │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - panMapStart: function() { │ │ │ │ │ - if (this.kinetic) { │ │ │ │ │ - this.kinetic.begin() │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ + move: this.cancelHover, │ │ │ │ │ + pause: this.getInfoForHover │ │ │ │ │ + }, OpenLayers.Util.extend(this.handlerOptions.hover || {}, { │ │ │ │ │ + delay: 250 │ │ │ │ │ + })) │ │ │ │ │ + } else { │ │ │ │ │ + var callbacks = {}; │ │ │ │ │ + callbacks[this.clickCallback] = this.getInfoForClick; │ │ │ │ │ + this.handler = new OpenLayers.Handler.Click(this, callbacks, this.handlerOptions.click || {}) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - panMap: function(xy) { │ │ │ │ │ - if (this.kinetic) { │ │ │ │ │ - this.kinetic.update(xy) │ │ │ │ │ - } │ │ │ │ │ - this.panned = true; │ │ │ │ │ - this.map.pan(this.handler.last.x - xy.x, this.handler.last.y - xy.y, { │ │ │ │ │ - dragging: true, │ │ │ │ │ - animate: false │ │ │ │ │ + getInfoForClick: function(evt) { │ │ │ │ │ + this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ + xy: evt.xy │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + this.request(evt.xy, {}) │ │ │ │ │ + }, │ │ │ │ │ + getInfoForHover: function(evt) { │ │ │ │ │ + this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ + xy: evt.xy │ │ │ │ │ + }); │ │ │ │ │ + this.request(evt.xy, { │ │ │ │ │ + hover: true │ │ │ │ │ }) │ │ │ │ │ }, │ │ │ │ │ - panMapDone: function(xy) { │ │ │ │ │ - if (this.panned) { │ │ │ │ │ - var res = null; │ │ │ │ │ - if (this.kinetic) { │ │ │ │ │ - res = this.kinetic.end(xy) │ │ │ │ │ - } │ │ │ │ │ - this.map.pan(this.handler.last.x - xy.x, this.handler.last.y - xy.y, { │ │ │ │ │ - dragging: !!res, │ │ │ │ │ - animate: false │ │ │ │ │ - }); │ │ │ │ │ - if (res) { │ │ │ │ │ - var self = this; │ │ │ │ │ - this.kinetic.move(res, function(x, y, end) { │ │ │ │ │ - self.map.pan(x, y, { │ │ │ │ │ - dragging: !end, │ │ │ │ │ - animate: false │ │ │ │ │ - }) │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.panned = false │ │ │ │ │ + cancelHover: function() { │ │ │ │ │ + if (this.hoverRequest) { │ │ │ │ │ + this.hoverRequest.abort(); │ │ │ │ │ + this.hoverRequest = null │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.DragPan" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.Navigation = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - dragPan: null, │ │ │ │ │ - dragPanOptions: null, │ │ │ │ │ - pinchZoom: null, │ │ │ │ │ - pinchZoomOptions: null, │ │ │ │ │ - documentDrag: false, │ │ │ │ │ - zoomBox: null, │ │ │ │ │ - zoomBoxEnabled: true, │ │ │ │ │ - zoomWheelEnabled: true, │ │ │ │ │ - mouseWheelOptions: null, │ │ │ │ │ - handleRightClicks: false, │ │ │ │ │ - zoomBoxKeyMask: OpenLayers.Handler.MOD_SHIFT, │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.handlers = {}; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - if (this.dragPan) { │ │ │ │ │ - this.dragPan.destroy() │ │ │ │ │ - } │ │ │ │ │ - this.dragPan = null; │ │ │ │ │ - if (this.zoomBox) { │ │ │ │ │ - this.zoomBox.destroy() │ │ │ │ │ - } │ │ │ │ │ - this.zoomBox = null; │ │ │ │ │ - if (this.pinchZoom) { │ │ │ │ │ - this.pinchZoom.destroy() │ │ │ │ │ + findLayers: function() { │ │ │ │ │ + var candidates = this.layers || this.map.layers; │ │ │ │ │ + var layers = []; │ │ │ │ │ + var layer, url; │ │ │ │ │ + for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = candidates[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.WMS && (!this.queryVisible || layer.getVisibility())) { │ │ │ │ │ + url = OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url; │ │ │ │ │ + if (this.drillDown === false && !this.url) { │ │ │ │ │ + this.url = url │ │ │ │ │ + } │ │ │ │ │ + if (this.drillDown === true || this.urlMatches(url)) { │ │ │ │ │ + layers.push(layer) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.pinchZoom = null; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ + return layers │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - this.dragPan.activate(); │ │ │ │ │ - if (this.zoomWheelEnabled) { │ │ │ │ │ - this.handlers.wheel.activate() │ │ │ │ │ - } │ │ │ │ │ - this.handlers.click.activate(); │ │ │ │ │ - if (this.zoomBoxEnabled) { │ │ │ │ │ - this.zoomBox.activate() │ │ │ │ │ - } │ │ │ │ │ - if (this.pinchZoom) { │ │ │ │ │ - this.pinchZoom.activate() │ │ │ │ │ + urlMatches: function(url) { │ │ │ │ │ + var matches = OpenLayers.Util.isEquivalentUrl(this.url, url); │ │ │ │ │ + if (!matches && this.layerUrls) { │ │ │ │ │ + for (var i = 0, len = this.layerUrls.length; i < len; ++i) { │ │ │ │ │ + if (OpenLayers.Util.isEquivalentUrl(this.layerUrls[i], url)) { │ │ │ │ │ + matches = true; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Control.prototype.activate.apply(this, arguments) │ │ │ │ │ + return matches │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.pinchZoom) { │ │ │ │ │ - this.pinchZoom.deactivate() │ │ │ │ │ + buildWMSOptions: function(url, layers, clickPosition, format) { │ │ │ │ │ + var layerNames = [], │ │ │ │ │ + styleNames = []; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + if (layers[i].params.LAYERS != null) { │ │ │ │ │ + layerNames = layerNames.concat(layers[i].params.LAYERS); │ │ │ │ │ + styleNames = styleNames.concat(this.getStyleNames(layers[i])) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.zoomBox.deactivate(); │ │ │ │ │ - this.dragPan.deactivate(); │ │ │ │ │ - this.handlers.click.deactivate(); │ │ │ │ │ - this.handlers.wheel.deactivate(); │ │ │ │ │ - return OpenLayers.Control.prototype.deactivate.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - if (this.handleRightClicks) { │ │ │ │ │ - this.map.viewPortDiv.oncontextmenu = OpenLayers.Function.False │ │ │ │ │ + var firstLayer = layers[0]; │ │ │ │ │ + var projection = this.map.getProjection(); │ │ │ │ │ + var layerProj = firstLayer.projection; │ │ │ │ │ + if (layerProj && layerProj.equals(this.map.getProjectionObject())) { │ │ │ │ │ + projection = layerProj.getCode() │ │ │ │ │ } │ │ │ │ │ - var clickCallbacks = { │ │ │ │ │ - click: this.defaultClick, │ │ │ │ │ - dblclick: this.defaultDblClick, │ │ │ │ │ - dblrightclick: this.defaultDblRightClick │ │ │ │ │ - }; │ │ │ │ │ - var clickOptions = { │ │ │ │ │ - double: true, │ │ │ │ │ - stopDouble: true │ │ │ │ │ - }; │ │ │ │ │ - this.handlers.click = new OpenLayers.Handler.Click(this, clickCallbacks, clickOptions); │ │ │ │ │ - this.dragPan = new OpenLayers.Control.DragPan(OpenLayers.Util.extend({ │ │ │ │ │ - map: this.map, │ │ │ │ │ - documentDrag: this.documentDrag │ │ │ │ │ - }, this.dragPanOptions)); │ │ │ │ │ - this.zoomBox = new OpenLayers.Control.ZoomBox({ │ │ │ │ │ - map: this.map, │ │ │ │ │ - keyMask: this.zoomBoxKeyMask │ │ │ │ │ + var params = OpenLayers.Util.extend({ │ │ │ │ │ + service: "WMS", │ │ │ │ │ + version: firstLayer.params.VERSION, │ │ │ │ │ + request: "GetFeatureInfo", │ │ │ │ │ + exceptions: firstLayer.params.EXCEPTIONS, │ │ │ │ │ + bbox: this.map.getExtent().toBBOX(null, firstLayer.reverseAxisOrder()), │ │ │ │ │ + feature_count: this.maxFeatures, │ │ │ │ │ + height: this.map.getSize().h, │ │ │ │ │ + width: this.map.getSize().w, │ │ │ │ │ + format: format, │ │ │ │ │ + info_format: firstLayer.params.INFO_FORMAT || this.infoFormat │ │ │ │ │ + }, parseFloat(firstLayer.params.VERSION) >= 1.3 ? { │ │ │ │ │ + crs: projection, │ │ │ │ │ + i: parseInt(clickPosition.x), │ │ │ │ │ + j: parseInt(clickPosition.y) │ │ │ │ │ + } : { │ │ │ │ │ + srs: projection, │ │ │ │ │ + x: parseInt(clickPosition.x), │ │ │ │ │ + y: parseInt(clickPosition.y) │ │ │ │ │ }); │ │ │ │ │ - this.dragPan.draw(); │ │ │ │ │ - this.zoomBox.draw(); │ │ │ │ │ - var wheelOptions = this.map.fractionalZoom ? {} : { │ │ │ │ │ - cumulative: false, │ │ │ │ │ - interval: 50, │ │ │ │ │ - maxDelta: 6 │ │ │ │ │ - }; │ │ │ │ │ - this.handlers.wheel = new OpenLayers.Handler.MouseWheel(this, { │ │ │ │ │ - up: this.wheelUp, │ │ │ │ │ - down: this.wheelDown │ │ │ │ │ - }, OpenLayers.Util.extend(wheelOptions, this.mouseWheelOptions)); │ │ │ │ │ - if (OpenLayers.Control.PinchZoom) { │ │ │ │ │ - this.pinchZoom = new OpenLayers.Control.PinchZoom(OpenLayers.Util.extend({ │ │ │ │ │ - map: this.map │ │ │ │ │ - }, this.pinchZoomOptions)) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - defaultClick: function(evt) { │ │ │ │ │ - if (evt.lastTouches && evt.lastTouches.length == 2) { │ │ │ │ │ - this.map.zoomOut() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - defaultDblClick: function(evt) { │ │ │ │ │ - this.map.zoomTo(this.map.zoom + 1, evt.xy) │ │ │ │ │ - }, │ │ │ │ │ - defaultDblRightClick: function(evt) { │ │ │ │ │ - this.map.zoomTo(this.map.zoom - 1, evt.xy) │ │ │ │ │ - }, │ │ │ │ │ - wheelChange: function(evt, deltaZ) { │ │ │ │ │ - if (!this.map.fractionalZoom) { │ │ │ │ │ - deltaZ = Math.round(deltaZ) │ │ │ │ │ - } │ │ │ │ │ - var currentZoom = this.map.getZoom(), │ │ │ │ │ - newZoom = currentZoom + deltaZ; │ │ │ │ │ - newZoom = Math.max(newZoom, 0); │ │ │ │ │ - newZoom = Math.min(newZoom, this.map.getNumZoomLevels()); │ │ │ │ │ - if (newZoom === currentZoom) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - this.map.zoomTo(newZoom, evt.xy) │ │ │ │ │ - }, │ │ │ │ │ - wheelUp: function(evt, delta) { │ │ │ │ │ - this.wheelChange(evt, delta || 1) │ │ │ │ │ - }, │ │ │ │ │ - wheelDown: function(evt, delta) { │ │ │ │ │ - this.wheelChange(evt, delta || -1) │ │ │ │ │ - }, │ │ │ │ │ - disableZoomBox: function() { │ │ │ │ │ - this.zoomBoxEnabled = false; │ │ │ │ │ - this.zoomBox.deactivate() │ │ │ │ │ - }, │ │ │ │ │ - enableZoomBox: function() { │ │ │ │ │ - this.zoomBoxEnabled = true; │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.zoomBox.activate() │ │ │ │ │ + if (layerNames.length != 0) { │ │ │ │ │ + params = OpenLayers.Util.extend({ │ │ │ │ │ + layers: layerNames, │ │ │ │ │ + query_layers: layerNames, │ │ │ │ │ + styles: styleNames │ │ │ │ │ + }, params) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - disableZoomWheel: function() { │ │ │ │ │ - this.zoomWheelEnabled = false; │ │ │ │ │ - this.handlers.wheel.deactivate() │ │ │ │ │ - }, │ │ │ │ │ - enableZoomWheel: function() { │ │ │ │ │ - this.zoomWheelEnabled = true; │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.handlers.wheel.activate() │ │ │ │ │ + OpenLayers.Util.applyDefaults(params, this.vendorParams); │ │ │ │ │ + return { │ │ │ │ │ + url: url, │ │ │ │ │ + params: OpenLayers.Util.upperCaseObject(params), │ │ │ │ │ + callback: function(request) { │ │ │ │ │ + this.handleResponse(clickPosition, request, url) │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Navigation" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.NavToolbar = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.addControls([new OpenLayers.Control.Navigation, new OpenLayers.Control.ZoomBox]) │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - var div = OpenLayers.Control.Panel.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (this.defaultControl === null) { │ │ │ │ │ - this.defaultControl = this.controls[0] │ │ │ │ │ + getStyleNames: function(layer) { │ │ │ │ │ + var styleNames; │ │ │ │ │ + if (layer.params.STYLES) { │ │ │ │ │ + styleNames = layer.params.STYLES │ │ │ │ │ + } else { │ │ │ │ │ + if (OpenLayers.Util.isArray(layer.params.LAYERS)) { │ │ │ │ │ + styleNames = new Array(layer.params.LAYERS.length) │ │ │ │ │ + } else { │ │ │ │ │ + styleNames = layer.params.LAYERS.replace(/[^,]/g, "") │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return div │ │ │ │ │ + return styleNames │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.NavToolbar" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.CacheRead = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - fetchEvent: "tileloadstart", │ │ │ │ │ - layers: null, │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - var i, layers = this.layers || map.layers; │ │ │ │ │ - for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - this.addLayer({ │ │ │ │ │ - layer: layers[i] │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - if (!this.layers) { │ │ │ │ │ - map.events.on({ │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - removeLayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ + request: function(clickPosition, options) { │ │ │ │ │ + var layers = this.findLayers(); │ │ │ │ │ + if (layers.length == 0) { │ │ │ │ │ + this.events.triggerEvent("nogetfeatureinfo"); │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - addLayer: function(evt) { │ │ │ │ │ - evt.layer.events.register(this.fetchEvent, this, this.fetch) │ │ │ │ │ - }, │ │ │ │ │ - removeLayer: function(evt) { │ │ │ │ │ - evt.layer.events.unregister(this.fetchEvent, this, this.fetch) │ │ │ │ │ - }, │ │ │ │ │ - fetch: function(evt) { │ │ │ │ │ - if (this.active && window.localStorage && evt.tile instanceof OpenLayers.Tile.Image) { │ │ │ │ │ - var tile = evt.tile, │ │ │ │ │ - url = tile.url; │ │ │ │ │ - if (!tile.layer.crossOriginKeyword && OpenLayers.ProxyHost && url.indexOf(OpenLayers.ProxyHost) === 0) { │ │ │ │ │ - url = OpenLayers.Control.CacheWrite.urlMap[url] │ │ │ │ │ + options = options || {}; │ │ │ │ │ + if (this.drillDown === false) { │ │ │ │ │ + var wmsOptions = this.buildWMSOptions(this.url, layers, clickPosition, layers[0].params.FORMAT); │ │ │ │ │ + var request = OpenLayers.Request.GET(wmsOptions); │ │ │ │ │ + if (options.hover === true) { │ │ │ │ │ + this.hoverRequest = request │ │ │ │ │ } │ │ │ │ │ - var dataURI = window.localStorage.getItem("olCache_" + url); │ │ │ │ │ - if (dataURI) { │ │ │ │ │ - tile.url = dataURI; │ │ │ │ │ - if (evt.type === "tileerror") { │ │ │ │ │ - tile.setImgSrc(dataURI) │ │ │ │ │ + } else { │ │ │ │ │ + this._requestCount = 0; │ │ │ │ │ + this._numRequests = 0; │ │ │ │ │ + this.features = []; │ │ │ │ │ + var services = {}, │ │ │ │ │ + url; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + var layer = layers[i]; │ │ │ │ │ + var service, found = false; │ │ │ │ │ + url = OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url; │ │ │ │ │ + if (url in services) { │ │ │ │ │ + services[url].push(layer) │ │ │ │ │ + } else { │ │ │ │ │ + this._numRequests++; │ │ │ │ │ + services[url] = [layer] │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + var layers; │ │ │ │ │ + for (var url in services) { │ │ │ │ │ + layers = services[url]; │ │ │ │ │ + var wmsOptions = this.buildWMSOptions(url, layers, clickPosition, layers[0].params.FORMAT); │ │ │ │ │ + OpenLayers.Request.GET(wmsOptions) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.layers || this.map) { │ │ │ │ │ - var i, layers = this.layers || this.map.layers; │ │ │ │ │ - for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - this.removeLayer({ │ │ │ │ │ - layer: layers[i] │ │ │ │ │ + triggerGetFeatureInfo: function(request, xy, features) { │ │ │ │ │ + this.events.triggerEvent("getfeatureinfo", { │ │ │ │ │ + text: request.responseText, │ │ │ │ │ + features: features, │ │ │ │ │ + request: request, │ │ │ │ │ + xy: xy │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait") │ │ │ │ │ + }, │ │ │ │ │ + handleResponse: function(xy, request, url) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ + } │ │ │ │ │ + var features = this.format.read(doc); │ │ │ │ │ + if (this.drillDown === false) { │ │ │ │ │ + this.triggerGetFeatureInfo(request, xy, features) │ │ │ │ │ + } else { │ │ │ │ │ + this._requestCount++; │ │ │ │ │ + if (this.output === "object") { │ │ │ │ │ + this._features = (this._features || []).concat({ │ │ │ │ │ + url: url, │ │ │ │ │ + features: features │ │ │ │ │ }) │ │ │ │ │ + } else { │ │ │ │ │ + this._features = (this._features || []).concat(features) │ │ │ │ │ + } │ │ │ │ │ + if (this._requestCount === this._numRequests) { │ │ │ │ │ + this.triggerGetFeatureInfo(request, xy, this._features.concat()); │ │ │ │ │ + delete this._features; │ │ │ │ │ + delete this._requestCount; │ │ │ │ │ + delete this._numRequests │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - removeLayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.CacheRead" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.WMSGetFeatureInfo" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Control.ModifyFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ documentDrag: false, │ │ │ │ │ geometryTypes: null, │ │ │ │ │ clickout: true, │ │ │ │ │ toggle: true, │ │ │ │ │ standalone: false, │ │ │ │ │ @@ -30054,14 +29121,84 @@ │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.ModifyFeature" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Control.ModifyFeature.RESHAPE = 1; │ │ │ │ │ OpenLayers.Control.ModifyFeature.RESIZE = 2; │ │ │ │ │ OpenLayers.Control.ModifyFeature.ROTATE = 4; │ │ │ │ │ OpenLayers.Control.ModifyFeature.DRAG = 8; │ │ │ │ │ +OpenLayers.Control.ZoomIn = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ + trigger: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.zoomIn() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomIn" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Zoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + zoomInText: "+", │ │ │ │ │ + zoomInId: "olZoomInLink", │ │ │ │ │ + zoomOutText: "−", │ │ │ │ │ + zoomOutId: "olZoomOutLink", │ │ │ │ │ + draw: function() { │ │ │ │ │ + var div = OpenLayers.Control.prototype.draw.apply(this), │ │ │ │ │ + links = this.getOrCreateLinks(div), │ │ │ │ │ + zoomIn = links.zoomIn, │ │ │ │ │ + zoomOut = links.zoomOut, │ │ │ │ │ + eventsInstance = this.map.events; │ │ │ │ │ + if (zoomOut.parentNode !== div) { │ │ │ │ │ + eventsInstance = this.events; │ │ │ │ │ + eventsInstance.attachToElement(zoomOut.parentNode) │ │ │ │ │ + } │ │ │ │ │ + eventsInstance.register("buttonclick", this, this.onZoomClick); │ │ │ │ │ + this.zoomInLink = zoomIn; │ │ │ │ │ + this.zoomOutLink = zoomOut; │ │ │ │ │ + return div │ │ │ │ │ + }, │ │ │ │ │ + getOrCreateLinks: function(el) { │ │ │ │ │ + var zoomIn = document.getElementById(this.zoomInId), │ │ │ │ │ + zoomOut = document.getElementById(this.zoomOutId); │ │ │ │ │ + if (!zoomIn) { │ │ │ │ │ + zoomIn = document.createElement("a"); │ │ │ │ │ + zoomIn.href = "#zoomIn"; │ │ │ │ │ + zoomIn.appendChild(document.createTextNode(this.zoomInText)); │ │ │ │ │ + zoomIn.className = "olControlZoomIn"; │ │ │ │ │ + el.appendChild(zoomIn) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.addClass(zoomIn, "olButton"); │ │ │ │ │ + if (!zoomOut) { │ │ │ │ │ + zoomOut = document.createElement("a"); │ │ │ │ │ + zoomOut.href = "#zoomOut"; │ │ │ │ │ + zoomOut.appendChild(document.createTextNode(this.zoomOutText)); │ │ │ │ │ + zoomOut.className = "olControlZoomOut"; │ │ │ │ │ + el.appendChild(zoomOut) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.addClass(zoomOut, "olButton"); │ │ │ │ │ + return { │ │ │ │ │ + zoomIn: zoomIn, │ │ │ │ │ + zoomOut: zoomOut │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + onZoomClick: function(evt) { │ │ │ │ │ + var button = evt.buttonElement; │ │ │ │ │ + if (button === this.zoomInLink) { │ │ │ │ │ + this.map.zoomIn() │ │ │ │ │ + } else if (button === this.zoomOutLink) { │ │ │ │ │ + this.map.zoomOut() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onZoomClick) │ │ │ │ │ + } │ │ │ │ │ + delete this.zoomInLink; │ │ │ │ │ + delete this.zoomOutLink; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Zoom" │ │ │ │ │ +}); │ │ │ │ │ OpenLayers.Control.DrawFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ layer: null, │ │ │ │ │ callbacks: null, │ │ │ │ │ multi: false, │ │ │ │ │ featureAdded: function() {}, │ │ │ │ │ initialize: function(layer, handler, options) { │ │ │ │ │ OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ @@ -30143,251 +29280,325 @@ │ │ │ │ │ this.handler.finishGeometry() │ │ │ │ │ }, │ │ │ │ │ cancel: function() { │ │ │ │ │ this.handler.cancel() │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.DrawFeature" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.ArgParser = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - center: null, │ │ │ │ │ - zoom: null, │ │ │ │ │ - layers: null, │ │ │ │ │ +OpenLayers.Control.Permalink = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + argParserClass: OpenLayers.Control.ArgParser, │ │ │ │ │ + element: null, │ │ │ │ │ + anchor: false, │ │ │ │ │ + base: "", │ │ │ │ │ displayProjection: null, │ │ │ │ │ - getParameters: function(url) { │ │ │ │ │ - url = url || window.location.href; │ │ │ │ │ - var parameters = OpenLayers.Util.getParameters(url); │ │ │ │ │ - var index = url.indexOf("#"); │ │ │ │ │ - if (index > 0) { │ │ │ │ │ - url = "?" + url.substring(index + 1, url.length); │ │ │ │ │ - OpenLayers.Util.extend(parameters, OpenLayers.Util.getParameters(url)) │ │ │ │ │ + initialize: function(element, base, options) { │ │ │ │ │ + if (element !== null && typeof element == "object" && !OpenLayers.Util.isElement(element)) { │ │ │ │ │ + options = element; │ │ │ │ │ + this.base = document.location.href; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (this.element != null) { │ │ │ │ │ + this.element = OpenLayers.Util.getElement(this.element) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.element = OpenLayers.Util.getElement(element); │ │ │ │ │ + this.base = base || document.location.href │ │ │ │ │ } │ │ │ │ │ - return parameters │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.element && this.element.parentNode == this.div) { │ │ │ │ │ + this.div.removeChild(this.element); │ │ │ │ │ + this.element = null │ │ │ │ │ + } │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("moveend", this, this.updateLink) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ setMap: function(map) { │ │ │ │ │ OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ for (var i = 0, len = this.map.controls.length; i < len; i++) { │ │ │ │ │ var control = this.map.controls[i]; │ │ │ │ │ - if (control != this && control.CLASS_NAME == "OpenLayers.Control.ArgParser") { │ │ │ │ │ + if (control.CLASS_NAME == this.argParserClass.CLASS_NAME) { │ │ │ │ │ if (control.displayProjection != this.displayProjection) { │ │ │ │ │ this.displayProjection = control.displayProjection │ │ │ │ │ } │ │ │ │ │ break │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ if (i == this.map.controls.length) { │ │ │ │ │ - var args = this.getParameters(); │ │ │ │ │ - if (args.layers) { │ │ │ │ │ - this.layers = args.layers; │ │ │ │ │ - this.map.events.register("addlayer", this, this.configureLayers); │ │ │ │ │ - this.configureLayers() │ │ │ │ │ - } │ │ │ │ │ - if (args.lat && args.lon) { │ │ │ │ │ - this.center = new OpenLayers.LonLat(parseFloat(args.lon), parseFloat(args.lat)); │ │ │ │ │ - if (args.zoom) { │ │ │ │ │ - this.zoom = parseFloat(args.zoom) │ │ │ │ │ - } │ │ │ │ │ - this.map.events.register("changebaselayer", this, this.setCenter); │ │ │ │ │ - this.setCenter() │ │ │ │ │ - } │ │ │ │ │ + this.map.addControl(new this.argParserClass({ │ │ │ │ │ + displayProjection: this.displayProjection │ │ │ │ │ + })) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - setCenter: function() { │ │ │ │ │ - if (this.map.baseLayer) { │ │ │ │ │ - this.map.events.unregister("changebaselayer", this, this.setCenter); │ │ │ │ │ - if (this.displayProjection) { │ │ │ │ │ - this.center.transform(this.displayProjection, this.map.getProjectionObject()) │ │ │ │ │ - } │ │ │ │ │ - this.map.setCenter(this.center, this.zoom) │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (!this.element && !this.anchor) { │ │ │ │ │ + this.element = document.createElement("a"); │ │ │ │ │ + this.element.innerHTML = OpenLayers.i18n("Permalink"); │ │ │ │ │ + this.element.href = ""; │ │ │ │ │ + this.div.appendChild(this.element) │ │ │ │ │ } │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + moveend: this.updateLink, │ │ │ │ │ + changelayer: this.updateLink, │ │ │ │ │ + changebaselayer: this.updateLink, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.updateLink(); │ │ │ │ │ + return this.div │ │ │ │ │ }, │ │ │ │ │ - configureLayers: function() { │ │ │ │ │ - if (this.layers.length == this.map.layers.length) { │ │ │ │ │ - this.map.events.unregister("addlayer", this, this.configureLayers); │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - var c = this.layers.charAt(i); │ │ │ │ │ - if (c == "B") { │ │ │ │ │ - this.map.setBaseLayer(layer) │ │ │ │ │ - } else if (c == "T" || c == "F") { │ │ │ │ │ - layer.setVisibility(c == "T") │ │ │ │ │ + updateLink: function() { │ │ │ │ │ + var separator = this.anchor ? "#" : "?"; │ │ │ │ │ + var href = this.base; │ │ │ │ │ + var anchor = null; │ │ │ │ │ + if (href.indexOf("#") != -1 && this.anchor == false) { │ │ │ │ │ + anchor = href.substring(href.indexOf("#"), href.length) │ │ │ │ │ + } │ │ │ │ │ + if (href.indexOf(separator) != -1) { │ │ │ │ │ + href = href.substring(0, href.indexOf(separator)) │ │ │ │ │ + } │ │ │ │ │ + var splits = href.split("#"); │ │ │ │ │ + href = splits[0] + separator + OpenLayers.Util.getParameterString(this.createParams()); │ │ │ │ │ + if (anchor) { │ │ │ │ │ + href += anchor │ │ │ │ │ + } │ │ │ │ │ + if (this.anchor && !this.element) { │ │ │ │ │ + window.location.href = href │ │ │ │ │ + } else { │ │ │ │ │ + this.element.href = href │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + createParams: function(center, zoom, layers) { │ │ │ │ │ + center = center || this.map.getCenter(); │ │ │ │ │ + var params = OpenLayers.Util.getParameters(this.base); │ │ │ │ │ + if (center) { │ │ │ │ │ + params.zoom = zoom || this.map.getZoom(); │ │ │ │ │ + var lat = center.lat; │ │ │ │ │ + var lon = center.lon; │ │ │ │ │ + if (this.displayProjection) { │ │ │ │ │ + var mapPosition = OpenLayers.Projection.transform({ │ │ │ │ │ + x: lon, │ │ │ │ │ + y: lat │ │ │ │ │ + }, this.map.getProjectionObject(), this.displayProjection); │ │ │ │ │ + lon = mapPosition.x; │ │ │ │ │ + lat = mapPosition.y │ │ │ │ │ + } │ │ │ │ │ + params.lat = Math.round(lat * 1e5) / 1e5; │ │ │ │ │ + params.lon = Math.round(lon * 1e5) / 1e5; │ │ │ │ │ + layers = layers || this.map.layers; │ │ │ │ │ + params.layers = ""; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + var layer = layers[i]; │ │ │ │ │ + if (layer.isBaseLayer) { │ │ │ │ │ + params.layers += layer == this.map.baseLayer ? "B" : "0" │ │ │ │ │ + } else { │ │ │ │ │ + params.layers += layer.getVisibility() ? "T" : "F" │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return params │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ArgParser" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Permalink" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.Geolocate = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - geolocation: null, │ │ │ │ │ - available: "geolocation" in navigator, │ │ │ │ │ - bind: true, │ │ │ │ │ - watch: false, │ │ │ │ │ - geolocationOptions: null, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ +OpenLayers.Control.ZoomToMaxExtent = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ + trigger: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.zoomToMaxExtent() │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (this.available && !this.geolocation) { │ │ │ │ │ - this.geolocation = navigator.geolocation │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomToMaxExtent" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Panel = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + controls: null, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + defaultControl: null, │ │ │ │ │ + saveState: false, │ │ │ │ │ + allowDepress: false, │ │ │ │ │ + activeState: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.controls = []; │ │ │ │ │ + this.activeState = {} │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onButtonClick) │ │ │ │ │ } │ │ │ │ │ - if (!this.geolocation) { │ │ │ │ │ - this.events.triggerEvent("locationuncapable"); │ │ │ │ │ - return false │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + for (var ctl, i = this.controls.length - 1; i >= 0; i--) { │ │ │ │ │ + ctl = this.controls[i]; │ │ │ │ │ + if (ctl.events) { │ │ │ │ │ + ctl.events.un({ │ │ │ │ │ + activate: this.iconOn, │ │ │ │ │ + deactivate: this.iconOff │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + ctl.panel_div = null │ │ │ │ │ } │ │ │ │ │ + this.activeState = null │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - if (this.watch) { │ │ │ │ │ - this.watchId = this.geolocation.watchPosition(OpenLayers.Function.bind(this.geolocate, this), OpenLayers.Function.bind(this.failure, this), this.geolocationOptions) │ │ │ │ │ - } else { │ │ │ │ │ - this.getCurrentLocation() │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + control = this.controls[i]; │ │ │ │ │ + if (control === this.defaultControl || this.saveState && this.activeState[control.id]) { │ │ │ │ │ + control.activate() │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + if (this.saveState === true) { │ │ │ │ │ + this.defaultControl = null │ │ │ │ │ + } │ │ │ │ │ + this.redraw(); │ │ │ │ │ return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return false │ │ │ │ │ }, │ │ │ │ │ deactivate: function() { │ │ │ │ │ - if (this.active && this.watchId !== null) { │ │ │ │ │ - this.geolocation.clearWatch(this.watchId) │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Control.prototype.deactivate.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - geolocate: function(position) { │ │ │ │ │ - var center = new OpenLayers.LonLat(position.coords.longitude, position.coords.latitude).transform(new OpenLayers.Projection("EPSG:4326"), this.map.getProjectionObject()); │ │ │ │ │ - if (this.bind) { │ │ │ │ │ - this.map.setCenter(center) │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("locationupdated", { │ │ │ │ │ - position: position, │ │ │ │ │ - point: new OpenLayers.Geometry.Point(center.lon, center.lat) │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - getCurrentLocation: function() { │ │ │ │ │ - if (!this.active || this.watch) { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + control = this.controls[i]; │ │ │ │ │ + this.activeState[control.id] = control.deactivate() │ │ │ │ │ + } │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ return false │ │ │ │ │ } │ │ │ │ │ - this.geolocation.getCurrentPosition(OpenLayers.Function.bind(this.geolocate, this), OpenLayers.Function.bind(this.failure, this), this.geolocationOptions); │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - failure: function(error) { │ │ │ │ │ - this.events.triggerEvent("locationfailed", { │ │ │ │ │ - error: error │ │ │ │ │ - }) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Geolocate" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.CacheWrite = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - layers: null, │ │ │ │ │ - imageFormat: "image/png", │ │ │ │ │ - quotaRegEx: /quota/i, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - var i, layers = this.layers || map.layers; │ │ │ │ │ - for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - this.addLayer({ │ │ │ │ │ - layer: layers[i] │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - if (!this.layers) { │ │ │ │ │ - map.events.on({ │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - removeLayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (this.outsideViewport) { │ │ │ │ │ + this.events.attachToElement(this.div); │ │ │ │ │ + this.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ + } else { │ │ │ │ │ + this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ } │ │ │ │ │ + this.addControlsToMap(this.controls); │ │ │ │ │ + return this.div │ │ │ │ │ }, │ │ │ │ │ - addLayer: function(evt) { │ │ │ │ │ - evt.layer.events.on({ │ │ │ │ │ - tileloadstart: this.makeSameOrigin, │ │ │ │ │ - tileloaded: this.onTileLoaded, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - removeLayer: function(evt) { │ │ │ │ │ - evt.layer.events.un({ │ │ │ │ │ - tileloadstart: this.makeSameOrigin, │ │ │ │ │ - tileloaded: this.onTileLoaded, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - makeSameOrigin: function(evt) { │ │ │ │ │ + redraw: function() { │ │ │ │ │ + for (var l = this.div.childNodes.length, i = l - 1; i >= 0; i--) { │ │ │ │ │ + this.div.removeChild(this.div.childNodes[i]) │ │ │ │ │ + } │ │ │ │ │ + this.div.innerHTML = ""; │ │ │ │ │ if (this.active) { │ │ │ │ │ - var tile = evt.tile; │ │ │ │ │ - if (tile instanceof OpenLayers.Tile.Image && !tile.crossOriginKeyword && tile.url.substr(0, 5) !== "data:") { │ │ │ │ │ - var sameOriginUrl = OpenLayers.Request.makeSameOrigin(tile.url, OpenLayers.ProxyHost); │ │ │ │ │ - OpenLayers.Control.CacheWrite.urlMap[sameOriginUrl] = tile.url; │ │ │ │ │ - tile.url = sameOriginUrl │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + this.div.appendChild(this.controls[i].panel_div) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - onTileLoaded: function(evt) { │ │ │ │ │ - if (this.active && !evt.aborted && evt.tile instanceof OpenLayers.Tile.Image && evt.tile.url.substr(0, 5) !== "data:") { │ │ │ │ │ - this.cache({ │ │ │ │ │ - tile: evt.tile │ │ │ │ │ - }); │ │ │ │ │ - delete OpenLayers.Control.CacheWrite.urlMap[evt.tile.url] │ │ │ │ │ + activateControl: function(control) { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - cache: function(obj) { │ │ │ │ │ - if (window.localStorage) { │ │ │ │ │ - var tile = obj.tile; │ │ │ │ │ - try { │ │ │ │ │ - var canvasContext = tile.getCanvasContext(); │ │ │ │ │ - if (canvasContext) { │ │ │ │ │ - var urlMap = OpenLayers.Control.CacheWrite.urlMap; │ │ │ │ │ - var url = urlMap[tile.url] || tile.url; │ │ │ │ │ - window.localStorage.setItem("olCache_" + url, canvasContext.canvas.toDataURL(this.imageFormat)) │ │ │ │ │ - } │ │ │ │ │ - } catch (e) { │ │ │ │ │ - var reason = e.name || e.message; │ │ │ │ │ - if (reason && this.quotaRegEx.test(reason)) { │ │ │ │ │ - this.events.triggerEvent("cachefull", { │ │ │ │ │ - tile: tile │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Console.error(e.toString()) │ │ │ │ │ + if (control.type == OpenLayers.Control.TYPE_BUTTON) { │ │ │ │ │ + control.trigger(); │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (control.type == OpenLayers.Control.TYPE_TOGGLE) { │ │ │ │ │ + if (control.active) { │ │ │ │ │ + control.deactivate() │ │ │ │ │ + } else { │ │ │ │ │ + control.activate() │ │ │ │ │ + } │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (this.allowDepress && control.active) { │ │ │ │ │ + control.deactivate() │ │ │ │ │ + } else { │ │ │ │ │ + var c; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + c = this.controls[i]; │ │ │ │ │ + if (c != control && (c.type === OpenLayers.Control.TYPE_TOOL || c.type == null)) { │ │ │ │ │ + c.deactivate() │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + control.activate() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.layers || this.map) { │ │ │ │ │ - var i, layers = this.layers || this.map.layers; │ │ │ │ │ - for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - this.removeLayer({ │ │ │ │ │ - layer: layers[i] │ │ │ │ │ - }) │ │ │ │ │ + addControls: function(controls) { │ │ │ │ │ + if (!OpenLayers.Util.isArray(controls)) { │ │ │ │ │ + controls = [controls] │ │ │ │ │ + } │ │ │ │ │ + this.controls = this.controls.concat(controls); │ │ │ │ │ + for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ + var control = controls[i], │ │ │ │ │ + element = this.createControlMarkup(control); │ │ │ │ │ + OpenLayers.Element.addClass(element, control.displayClass + "ItemInactive"); │ │ │ │ │ + OpenLayers.Element.addClass(element, "olButton"); │ │ │ │ │ + if (control.title != "" && !element.title) { │ │ │ │ │ + element.title = control.title │ │ │ │ │ } │ │ │ │ │ + control.panel_div = element │ │ │ │ │ } │ │ │ │ │ if (this.map) { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - removeLayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ + this.addControlsToMap(controls); │ │ │ │ │ + this.redraw() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + createControlMarkup: function(control) { │ │ │ │ │ + return document.createElement("div") │ │ │ │ │ + }, │ │ │ │ │ + addControlsToMap: function(controls) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ + control = controls[i]; │ │ │ │ │ + if (control.autoActivate === true) { │ │ │ │ │ + control.autoActivate = false; │ │ │ │ │ + this.map.addControl(control); │ │ │ │ │ + control.autoActivate = true │ │ │ │ │ + } else { │ │ │ │ │ + this.map.addControl(control); │ │ │ │ │ + control.deactivate() │ │ │ │ │ + } │ │ │ │ │ + control.events.on({ │ │ │ │ │ + activate: this.iconOn, │ │ │ │ │ + deactivate: this.iconOff │ │ │ │ │ }) │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.CacheWrite" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.CacheWrite.clearCache = function() { │ │ │ │ │ - if (!window.localStorage) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var i, key; │ │ │ │ │ - for (i = window.localStorage.length - 1; i >= 0; --i) { │ │ │ │ │ - key = window.localStorage.key(i); │ │ │ │ │ - if (key.substr(0, 8) === "olCache_") { │ │ │ │ │ - window.localStorage.removeItem(key) │ │ │ │ │ + iconOn: function() { │ │ │ │ │ + var d = this.panel_div; │ │ │ │ │ + var re = new RegExp("\\b(" + this.displayClass + "Item)Inactive\\b"); │ │ │ │ │ + d.className = d.className.replace(re, "$1Active") │ │ │ │ │ + }, │ │ │ │ │ + iconOff: function() { │ │ │ │ │ + var d = this.panel_div; │ │ │ │ │ + var re = new RegExp("\\b(" + this.displayClass + "Item)Active\\b"); │ │ │ │ │ + d.className = d.className.replace(re, "$1Inactive") │ │ │ │ │ + }, │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + var controls = this.controls, │ │ │ │ │ + button = evt.buttonElement; │ │ │ │ │ + for (var i = controls.length - 1; i >= 0; --i) { │ │ │ │ │ + if (controls[i].panel_div === button) { │ │ │ │ │ + this.activateControl(controls[i]); │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Control.CacheWrite.urlMap = {}; │ │ │ │ │ -OpenLayers.Control.Button = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - type: OpenLayers.Control.TYPE_BUTTON, │ │ │ │ │ - trigger: function() {}, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Button" │ │ │ │ │ + }, │ │ │ │ │ + getControlsBy: function(property, match) { │ │ │ │ │ + var test = typeof match.test == "function"; │ │ │ │ │ + var found = OpenLayers.Array.filter(this.controls, function(item) { │ │ │ │ │ + return item[property] == match || test && match.test(item[property]) │ │ │ │ │ + }); │ │ │ │ │ + return found │ │ │ │ │ + }, │ │ │ │ │ + getControlsByName: function(match) { │ │ │ │ │ + return this.getControlsBy("name", match) │ │ │ │ │ + }, │ │ │ │ │ + getControlsByClass: function(match) { │ │ │ │ │ + return this.getControlsBy("CLASS_NAME", match) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Panel" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Control.Pan = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ slideFactor: 50, │ │ │ │ │ slideRatio: null, │ │ │ │ │ direction: null, │ │ │ │ │ initialize: function(direction, options) { │ │ │ │ │ this.direction = direction; │ │ │ │ │ @@ -30417,418 +29628,554 @@ │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.Pan" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Control.Pan.NORTH = "North"; │ │ │ │ │ OpenLayers.Control.Pan.SOUTH = "South"; │ │ │ │ │ OpenLayers.Control.Pan.EAST = "East"; │ │ │ │ │ OpenLayers.Control.Pan.WEST = "West"; │ │ │ │ │ -OpenLayers.Control.PanPanel = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ - slideFactor: 50, │ │ │ │ │ - slideRatio: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ - var options = { │ │ │ │ │ - slideFactor: this.slideFactor, │ │ │ │ │ - slideRatio: this.slideRatio │ │ │ │ │ - }; │ │ │ │ │ - this.addControls([new OpenLayers.Control.Pan(OpenLayers.Control.Pan.NORTH, options), new OpenLayers.Control.Pan(OpenLayers.Control.Pan.SOUTH, options), new OpenLayers.Control.Pan(OpenLayers.Control.Pan.EAST, options), new OpenLayers.Control.Pan(OpenLayers.Control.Pan.WEST, options)]) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.PanPanel" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.ZoomIn = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ - trigger: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.zoomIn() │ │ │ │ │ +OpenLayers.Layer.Vector = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ + isBaseLayer: false, │ │ │ │ │ + isFixed: false, │ │ │ │ │ + features: null, │ │ │ │ │ + filter: null, │ │ │ │ │ + selectedFeatures: null, │ │ │ │ │ + unrenderedFeatures: null, │ │ │ │ │ + reportError: true, │ │ │ │ │ + style: null, │ │ │ │ │ + styleMap: null, │ │ │ │ │ + strategies: null, │ │ │ │ │ + protocol: null, │ │ │ │ │ + renderers: ["SVG", "VML", "Canvas"], │ │ │ │ │ + renderer: null, │ │ │ │ │ + rendererOptions: null, │ │ │ │ │ + geometryType: null, │ │ │ │ │ + drawn: false, │ │ │ │ │ + ratio: 1, │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ + this.assignRenderer() │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomIn" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.ZoomOut = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ - trigger: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.zoomOut() │ │ │ │ │ + if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ + this.renderer = null; │ │ │ │ │ + this.displayError() │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomOut" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.ZoomToMaxExtent = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ - trigger: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.zoomToMaxExtent() │ │ │ │ │ + if (!this.styleMap) { │ │ │ │ │ + this.styleMap = new OpenLayers.StyleMap │ │ │ │ │ + } │ │ │ │ │ + this.features = []; │ │ │ │ │ + this.selectedFeatures = []; │ │ │ │ │ + this.unrenderedFeatures = {}; │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + for (var i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + this.strategies[i].setLayer(this) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomToMaxExtent" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.ZoomPanel = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.addControls([new OpenLayers.Control.ZoomIn, new OpenLayers.Control.ZoomToMaxExtent, new OpenLayers.Control.ZoomOut]) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomPanel" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.GetFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - protocol: null, │ │ │ │ │ - multipleKey: null, │ │ │ │ │ - toggleKey: null, │ │ │ │ │ - modifiers: null, │ │ │ │ │ - multiple: false, │ │ │ │ │ - click: true, │ │ │ │ │ - single: true, │ │ │ │ │ - clickout: true, │ │ │ │ │ - toggle: false, │ │ │ │ │ - clickTolerance: 5, │ │ │ │ │ - hover: false, │ │ │ │ │ - box: false, │ │ │ │ │ - maxFeatures: 10, │ │ │ │ │ - features: null, │ │ │ │ │ - hoverFeature: null, │ │ │ │ │ - handlers: null, │ │ │ │ │ - hoverResponse: null, │ │ │ │ │ - filterType: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.features = {}; │ │ │ │ │ - this.handlers = {}; │ │ │ │ │ - if (this.click) { │ │ │ │ │ - this.handlers.click = new OpenLayers.Handler.Click(this, { │ │ │ │ │ - click: this.selectClick │ │ │ │ │ - }, this.handlerOptions.click || {}) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoDestroy) { │ │ │ │ │ + strategy.destroy() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.strategies = null │ │ │ │ │ } │ │ │ │ │ - if (this.box) { │ │ │ │ │ - this.handlers.box = new OpenLayers.Handler.Box(this, { │ │ │ │ │ - done: this.selectBox │ │ │ │ │ - }, OpenLayers.Util.extend(this.handlerOptions.box, { │ │ │ │ │ - boxDivClassName: "olHandlerBoxSelectFeature" │ │ │ │ │ - })) │ │ │ │ │ + if (this.protocol) { │ │ │ │ │ + if (this.protocol.autoDestroy) { │ │ │ │ │ + this.protocol.destroy() │ │ │ │ │ + } │ │ │ │ │ + this.protocol = null │ │ │ │ │ } │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - this.handlers.hover = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ - move: this.cancelHover, │ │ │ │ │ - pause: this.selectHover │ │ │ │ │ - }, OpenLayers.Util.extend(this.handlerOptions.hover, { │ │ │ │ │ - delay: 250, │ │ │ │ │ - pixelTolerance: 2 │ │ │ │ │ - })) │ │ │ │ │ + this.destroyFeatures(); │ │ │ │ │ + this.features = null; │ │ │ │ │ + this.selectedFeatures = null; │ │ │ │ │ + this.unrenderedFeatures = null; │ │ │ │ │ + if (this.renderer) { │ │ │ │ │ + this.renderer.destroy() │ │ │ │ │ + } │ │ │ │ │ + this.renderer = null; │ │ │ │ │ + this.geometryType = null; │ │ │ │ │ + this.drawn = null; │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Vector(this.name, this.getOptions()) │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ + var features = this.features; │ │ │ │ │ + var len = features.length; │ │ │ │ │ + var clonedFeatures = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + clonedFeatures[i] = features[i].clone() │ │ │ │ │ } │ │ │ │ │ + obj.features = clonedFeatures; │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - for (var i in this.handlers) { │ │ │ │ │ - this.handlers[i].activate() │ │ │ │ │ - } │ │ │ │ │ + refresh: function(obj) { │ │ │ │ │ + if (this.calculateInRange() && this.visibility) { │ │ │ │ │ + this.events.triggerEvent("refresh", obj) │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Control.prototype.activate.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - for (var i in this.handlers) { │ │ │ │ │ - this.handlers[i].deactivate() │ │ │ │ │ + assignRenderer: function() { │ │ │ │ │ + for (var i = 0, len = this.renderers.length; i < len; i++) { │ │ │ │ │ + var rendererClass = this.renderers[i]; │ │ │ │ │ + var renderer = typeof rendererClass == "function" ? rendererClass : OpenLayers.Renderer[rendererClass]; │ │ │ │ │ + if (renderer && renderer.prototype.supported()) { │ │ │ │ │ + this.renderer = new renderer(this.div, this.rendererOptions); │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Control.prototype.deactivate.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - selectClick: function(evt) { │ │ │ │ │ - var bounds = this.pixelToBounds(evt.xy); │ │ │ │ │ - this.setModifiers(evt); │ │ │ │ │ - this.request(bounds, { │ │ │ │ │ - single: this.single │ │ │ │ │ - }) │ │ │ │ │ + displayError: function() { │ │ │ │ │ + if (this.reportError) { │ │ │ │ │ + OpenLayers.Console.userError(OpenLayers.i18n("browserNotSupported", { │ │ │ │ │ + renderers: this.renderers.join("\n") │ │ │ │ │ + })) │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - selectBox: function(position) { │ │ │ │ │ - var bounds; │ │ │ │ │ - if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ - var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.left, │ │ │ │ │ - y: position.bottom │ │ │ │ │ - }); │ │ │ │ │ - var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.right, │ │ │ │ │ - y: position.top │ │ │ │ │ - }); │ │ │ │ │ - bounds = new OpenLayers.Bounds(minXY.lon, minXY.lat, maxXY.lon, maxXY.lat) │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (!this.renderer) { │ │ │ │ │ + this.map.removeLayer(this) │ │ │ │ │ } else { │ │ │ │ │ - if (this.click) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - bounds = this.pixelToBounds(position) │ │ │ │ │ + this.renderer.map = this.map; │ │ │ │ │ + var newSize = this.map.getSize(); │ │ │ │ │ + newSize.w = newSize.w * this.ratio; │ │ │ │ │ + newSize.h = newSize.h * this.ratio; │ │ │ │ │ + this.renderer.setSize(newSize) │ │ │ │ │ } │ │ │ │ │ - this.setModifiers(this.handlers.box.dragHandler.evt); │ │ │ │ │ - this.request(bounds) │ │ │ │ │ - }, │ │ │ │ │ - selectHover: function(evt) { │ │ │ │ │ - var bounds = this.pixelToBounds(evt.xy); │ │ │ │ │ - this.request(bounds, { │ │ │ │ │ - single: true, │ │ │ │ │ - hover: true │ │ │ │ │ - }) │ │ │ │ │ }, │ │ │ │ │ - cancelHover: function() { │ │ │ │ │ - if (this.hoverResponse) { │ │ │ │ │ - this.protocol.abort(this.hoverResponse); │ │ │ │ │ - this.hoverResponse = null; │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait") │ │ │ │ │ + afterAdd: function() { │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoActivate) { │ │ │ │ │ + strategy.activate() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - request: function(bounds, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: this.filterType, │ │ │ │ │ - value: bounds │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - var response = this.protocol.read({ │ │ │ │ │ - maxFeatures: options.single == true ? this.maxFeatures : undefined, │ │ │ │ │ - filter: filter, │ │ │ │ │ - callback: function(result) { │ │ │ │ │ - if (result.success()) { │ │ │ │ │ - if (result.features.length) { │ │ │ │ │ - if (options.single == true) { │ │ │ │ │ - this.selectBestFeature(result.features, bounds.getCenterLonLat(), options) │ │ │ │ │ - } else { │ │ │ │ │ - this.select(result.features) │ │ │ │ │ - } │ │ │ │ │ - } else if (options.hover) { │ │ │ │ │ - this.hoverSelect() │ │ │ │ │ - } else { │ │ │ │ │ - this.events.triggerEvent("clickout"); │ │ │ │ │ - if (this.clickout) { │ │ │ │ │ - this.unselectAll() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + this.drawn = false; │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoActivate) { │ │ │ │ │ + strategy.deactivate() │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait") │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - if (options.hover == true) { │ │ │ │ │ - this.hoverResponse = response │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - selectBestFeature: function(features, clickPosition, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - if (features.length) { │ │ │ │ │ - var point = new OpenLayers.Geometry.Point(clickPosition.lon, clickPosition.lat); │ │ │ │ │ - var feature, resultFeature, dist; │ │ │ │ │ - var minDist = Number.MAX_VALUE; │ │ │ │ │ - for (var i = 0; i < features.length; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - dist = point.distanceTo(feature.geometry, { │ │ │ │ │ - edge: false │ │ │ │ │ - }); │ │ │ │ │ - if (dist < minDist) { │ │ │ │ │ - minDist = dist; │ │ │ │ │ - resultFeature = feature; │ │ │ │ │ - if (minDist == 0) { │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + OpenLayers.Layer.prototype.onMapResize.apply(this, arguments); │ │ │ │ │ + var newSize = this.map.getSize(); │ │ │ │ │ + newSize.w = newSize.w * this.ratio; │ │ │ │ │ + newSize.h = newSize.h * this.ratio; │ │ │ │ │ + this.renderer.setSize(newSize) │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + var coordSysUnchanged = true; │ │ │ │ │ + if (!dragging) { │ │ │ │ │ + this.renderer.root.style.visibility = "hidden"; │ │ │ │ │ + var viewSize = this.map.getSize(), │ │ │ │ │ + viewWidth = viewSize.w, │ │ │ │ │ + viewHeight = viewSize.h, │ │ │ │ │ + offsetLeft = viewWidth / 2 * this.ratio - viewWidth / 2, │ │ │ │ │ + offsetTop = viewHeight / 2 * this.ratio - viewHeight / 2; │ │ │ │ │ + offsetLeft += this.map.layerContainerOriginPx.x; │ │ │ │ │ + offsetLeft = -Math.round(offsetLeft); │ │ │ │ │ + offsetTop += this.map.layerContainerOriginPx.y; │ │ │ │ │ + offsetTop = -Math.round(offsetTop); │ │ │ │ │ + this.div.style.left = offsetLeft + "px"; │ │ │ │ │ + this.div.style.top = offsetTop + "px"; │ │ │ │ │ + var extent = this.map.getExtent().scale(this.ratio); │ │ │ │ │ + coordSysUnchanged = this.renderer.setExtent(extent, zoomChanged); │ │ │ │ │ + this.renderer.root.style.visibility = "visible"; │ │ │ │ │ + if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ + this.div.scrollLeft = this.div.scrollLeft │ │ │ │ │ + } │ │ │ │ │ + if (!zoomChanged && coordSysUnchanged) { │ │ │ │ │ + for (var i in this.unrenderedFeatures) { │ │ │ │ │ + var feature = this.unrenderedFeatures[i]; │ │ │ │ │ + this.drawFeature(feature) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (options.hover == true) { │ │ │ │ │ - this.hoverSelect(resultFeature) │ │ │ │ │ - } else { │ │ │ │ │ - this.select(resultFeature || features) │ │ │ │ │ + } │ │ │ │ │ + if (!this.drawn || zoomChanged || !coordSysUnchanged) { │ │ │ │ │ + this.drawn = true; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0, len = this.features.length; i < len; i++) { │ │ │ │ │ + this.renderer.locked = i !== len - 1; │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + this.drawFeature(feature) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - setModifiers: function(evt) { │ │ │ │ │ - this.modifiers = { │ │ │ │ │ - multiple: this.multiple || this.multipleKey && evt[this.multipleKey], │ │ │ │ │ - toggle: this.toggle || this.toggleKey && evt[this.toggleKey] │ │ │ │ │ + display: function(display) { │ │ │ │ │ + OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ + var currentDisplay = this.div.style.display; │ │ │ │ │ + if (currentDisplay != this.renderer.root.style.display) { │ │ │ │ │ + this.renderer.root.style.display = currentDisplay │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - select: function(features) { │ │ │ │ │ - if (!this.modifiers.multiple && !this.modifiers.toggle) { │ │ │ │ │ - this.unselectAll() │ │ │ │ │ - } │ │ │ │ │ + addFeatures: function(features, options) { │ │ │ │ │ if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ features = [features] │ │ │ │ │ } │ │ │ │ │ - var cont = this.events.triggerEvent("beforefeaturesselected", { │ │ │ │ │ - features: features │ │ │ │ │ - }); │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - var selectedFeatures = []; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (this.features[feature.fid || feature.id]) { │ │ │ │ │ - if (this.modifiers.toggle) { │ │ │ │ │ - this.unselect(this.features[feature.fid || feature.id]) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - cont = this.events.triggerEvent("beforefeatureselected", { │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ + if (notify) { │ │ │ │ │ + var event = { │ │ │ │ │ + features: features │ │ │ │ │ + }; │ │ │ │ │ + var ret = this.events.triggerEvent("beforefeaturesadded", event); │ │ │ │ │ + if (ret === false) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + features = event.features │ │ │ │ │ + } │ │ │ │ │ + var featuresAdded = []; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + if (i != features.length - 1) { │ │ │ │ │ + this.renderer.locked = true │ │ │ │ │ + } else { │ │ │ │ │ + this.renderer.locked = false │ │ │ │ │ + } │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + if (this.geometryType && !(feature.geometry instanceof this.geometryType)) { │ │ │ │ │ + throw new TypeError("addFeatures: component should be an " + this.geometryType.prototype.CLASS_NAME) │ │ │ │ │ + } │ │ │ │ │ + feature.layer = this; │ │ │ │ │ + if (!feature.style && this.style) { │ │ │ │ │ + feature.style = OpenLayers.Util.extend({}, this.style) │ │ │ │ │ + } │ │ │ │ │ + if (notify) { │ │ │ │ │ + if (this.events.triggerEvent("beforefeatureadded", { │ │ │ │ │ feature: feature │ │ │ │ │ - }); │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - this.features[feature.fid || feature.id] = feature; │ │ │ │ │ - selectedFeatures.push(feature); │ │ │ │ │ - this.events.triggerEvent("featureselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ + }) === false) { │ │ │ │ │ + continue │ │ │ │ │ } │ │ │ │ │ + this.preFeatureInsert(feature) │ │ │ │ │ + } │ │ │ │ │ + featuresAdded.push(feature); │ │ │ │ │ + this.features.push(feature); │ │ │ │ │ + this.drawFeature(feature); │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureadded", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.onFeatureInsert(feature) │ │ │ │ │ } │ │ │ │ │ - this.events.triggerEvent("featuresselected", { │ │ │ │ │ - features: selectedFeatures │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - hoverSelect: function(feature) { │ │ │ │ │ - var fid = feature ? feature.fid || feature.id : null; │ │ │ │ │ - var hfid = this.hoverFeature ? this.hoverFeature.fid || this.hoverFeature.id : null; │ │ │ │ │ - if (hfid && hfid != fid) { │ │ │ │ │ - this.events.triggerEvent("outfeature", { │ │ │ │ │ - feature: this.hoverFeature │ │ │ │ │ - }); │ │ │ │ │ - this.hoverFeature = null │ │ │ │ │ } │ │ │ │ │ - if (fid && fid != hfid) { │ │ │ │ │ - this.events.triggerEvent("hoverfeature", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - this.hoverFeature = feature │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresadded", { │ │ │ │ │ + features: featuresAdded │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - unselect: function(feature) { │ │ │ │ │ - delete this.features[feature.fid || feature.id]; │ │ │ │ │ - this.events.triggerEvent("featureunselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - unselectAll: function() { │ │ │ │ │ - for (var fid in this.features) { │ │ │ │ │ - this.unselect(this.features[fid]) │ │ │ │ │ + removeFeatures: function(features, options) { │ │ │ │ │ + if (!features || features.length === 0) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - for (var i in this.handlers) { │ │ │ │ │ - this.handlers[i].setMap(map) │ │ │ │ │ + if (features === this.features) { │ │ │ │ │ + return this.removeAllFeatures(options) │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - pixelToBounds: function(pixel) { │ │ │ │ │ - var llPx = pixel.add(-this.clickTolerance / 2, this.clickTolerance / 2); │ │ │ │ │ - var urPx = pixel.add(this.clickTolerance / 2, -this.clickTolerance / 2); │ │ │ │ │ - var ll = this.map.getLonLatFromPixel(llPx); │ │ │ │ │ - var ur = this.map.getLonLatFromPixel(urPx); │ │ │ │ │ - return new OpenLayers.Bounds(ll.lon, ll.lat, ur.lon, ur.lat) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.GetFeature" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.Snapping = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - DEFAULTS: { │ │ │ │ │ - tolerance: 10, │ │ │ │ │ - node: true, │ │ │ │ │ - edge: true, │ │ │ │ │ - vertex: true │ │ │ │ │ - }, │ │ │ │ │ - greedy: true, │ │ │ │ │ - precedence: ["node", "vertex", "edge"], │ │ │ │ │ - resolution: null, │ │ │ │ │ - geoToleranceCache: null, │ │ │ │ │ - layer: null, │ │ │ │ │ - feature: null, │ │ │ │ │ - point: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.options = options || {}; │ │ │ │ │ - if (this.options.layer) { │ │ │ │ │ - this.setLayer(this.options.layer) │ │ │ │ │ + if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ + features = [features] │ │ │ │ │ } │ │ │ │ │ - var defaults = OpenLayers.Util.extend({}, this.options.defaults); │ │ │ │ │ - this.defaults = OpenLayers.Util.applyDefaults(defaults, this.DEFAULTS); │ │ │ │ │ - this.setTargets(this.options.targets); │ │ │ │ │ - if (this.targets.length === 0 && this.layer) { │ │ │ │ │ - this.addTargetLayer(this.layer) │ │ │ │ │ + if (features === this.selectedFeatures) { │ │ │ │ │ + features = features.slice() │ │ │ │ │ } │ │ │ │ │ - this.geoToleranceCache = {} │ │ │ │ │ - }, │ │ │ │ │ - setLayer: function(layer) { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - this.activate() │ │ │ │ │ - } else { │ │ │ │ │ - this.layer = layer │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeaturesremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - setTargets: function(targets) { │ │ │ │ │ - this.targets = []; │ │ │ │ │ - if (targets && targets.length) { │ │ │ │ │ - var target; │ │ │ │ │ - for (var i = 0, len = targets.length; i < len; ++i) { │ │ │ │ │ - target = targets[i]; │ │ │ │ │ - if (target instanceof OpenLayers.Layer.Vector) { │ │ │ │ │ - this.addTargetLayer(target) │ │ │ │ │ - } else { │ │ │ │ │ - this.addTarget(target) │ │ │ │ │ - } │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + if (i != 0 && features[i - 1].geometry) { │ │ │ │ │ + this.renderer.locked = true │ │ │ │ │ + } else { │ │ │ │ │ + this.renderer.locked = false │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - addTargetLayer: function(layer) { │ │ │ │ │ - this.addTarget({ │ │ │ │ │ - layer: layer │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - addTarget: function(target) { │ │ │ │ │ - target = OpenLayers.Util.applyDefaults(target, this.defaults); │ │ │ │ │ - target.nodeTolerance = target.nodeTolerance || target.tolerance; │ │ │ │ │ - target.vertexTolerance = target.vertexTolerance || target.tolerance; │ │ │ │ │ - target.edgeTolerance = target.edgeTolerance || target.tolerance; │ │ │ │ │ - this.targets.push(target) │ │ │ │ │ - }, │ │ │ │ │ - removeTargetLayer: function(layer) { │ │ │ │ │ - var target; │ │ │ │ │ - for (var i = this.targets.length - 1; i >= 0; --i) { │ │ │ │ │ - target = this.targets[i]; │ │ │ │ │ - if (target.layer === layer) { │ │ │ │ │ - this.removeTarget(target) │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - removeTarget: function(target) { │ │ │ │ │ - return OpenLayers.Util.removeItem(this.targets, target) │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Control.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - if (this.layer && this.layer.events) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - sketchstarted: this.onSketchModified, │ │ │ │ │ - sketchmodified: this.onSketchModified, │ │ │ │ │ - vertexmodified: this.onVertexModified, │ │ │ │ │ - scope: this │ │ │ │ │ + this.features = OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ + feature.layer = null; │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + this.renderer.eraseFeatures(feature) │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.selectedFeatures, feature) != -1) { │ │ │ │ │ + OpenLayers.Util.removeItem(this.selectedFeatures, feature) │ │ │ │ │ + } │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ }) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return activated │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Control.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - if (this.layer && this.layer.events) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - sketchstarted: this.onSketchModified, │ │ │ │ │ - sketchmodified: this.onSketchModified, │ │ │ │ │ - vertexmodified: this.onVertexModified, │ │ │ │ │ - scope: this │ │ │ │ │ + removeAllFeatures: function(options) { │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ + var features = this.features; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeaturesremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + feature.layer = null; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ }) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.feature = null; │ │ │ │ │ - this.point = null; │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - onSketchModified: function(event) { │ │ │ │ │ - this.feature = event.feature; │ │ │ │ │ - this.considerSnapping(event.vertex, event.vertex) │ │ │ │ │ + this.renderer.clear(); │ │ │ │ │ + this.features = []; │ │ │ │ │ + this.unrenderedFeatures = {}; │ │ │ │ │ + this.selectedFeatures = []; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroyFeatures: function(features, options) { │ │ │ │ │ + var all = features == undefined; │ │ │ │ │ + if (all) { │ │ │ │ │ + features = this.features │ │ │ │ │ + } │ │ │ │ │ + if (features) { │ │ │ │ │ + this.removeFeatures(features, options); │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + features[i].destroy() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + drawFeature: function(feature, style) { │ │ │ │ │ + if (!this.drawn) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (typeof style != "object") { │ │ │ │ │ + if (!style && feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ + style = "delete" │ │ │ │ │ + } │ │ │ │ │ + var renderIntent = style || feature.renderIntent; │ │ │ │ │ + style = feature.style || this.style; │ │ │ │ │ + if (!style) { │ │ │ │ │ + style = this.styleMap.createSymbolizer(feature, renderIntent) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var drawn = this.renderer.drawFeature(feature, style); │ │ │ │ │ + if (drawn === false || drawn === null) { │ │ │ │ │ + this.unrenderedFeatures[feature.id] = feature │ │ │ │ │ + } else { │ │ │ │ │ + delete this.unrenderedFeatures[feature.id] │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + eraseFeatures: function(features) { │ │ │ │ │ + this.renderer.eraseFeatures(features) │ │ │ │ │ + }, │ │ │ │ │ + getFeatureFromEvent: function(evt) { │ │ │ │ │ + if (!this.renderer) { │ │ │ │ │ + throw new Error("getFeatureFromEvent called on layer with no " + "renderer. This usually means you destroyed a " + "layer, but not some handler which is associated " + "with it.") │ │ │ │ │ + } │ │ │ │ │ + var feature = null; │ │ │ │ │ + var featureId = this.renderer.getFeatureIdFromEvent(evt); │ │ │ │ │ + if (featureId) { │ │ │ │ │ + if (typeof featureId === "string") { │ │ │ │ │ + feature = this.getFeatureById(featureId) │ │ │ │ │ + } else { │ │ │ │ │ + feature = featureId │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return feature │ │ │ │ │ + }, │ │ │ │ │ + getFeatureBy: function(property, value) { │ │ │ │ │ + var feature = null; │ │ │ │ │ + for (var i = 0, len = this.features.length; i < len; ++i) { │ │ │ │ │ + if (this.features[i][property] == value) { │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return feature │ │ │ │ │ + }, │ │ │ │ │ + getFeatureById: function(featureId) { │ │ │ │ │ + return this.getFeatureBy("id", featureId) │ │ │ │ │ + }, │ │ │ │ │ + getFeatureByFid: function(featureFid) { │ │ │ │ │ + return this.getFeatureBy("fid", featureFid) │ │ │ │ │ + }, │ │ │ │ │ + getFeaturesByAttribute: function(attrName, attrValue) { │ │ │ │ │ + var i, feature, len = this.features.length, │ │ │ │ │ + foundFeatures = []; │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + if (feature && feature.attributes) { │ │ │ │ │ + if (feature.attributes[attrName] === attrValue) { │ │ │ │ │ + foundFeatures.push(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return foundFeatures │ │ │ │ │ + }, │ │ │ │ │ + onFeatureInsert: function(feature) {}, │ │ │ │ │ + preFeatureInsert: function(feature) {}, │ │ │ │ │ + getDataExtent: function() { │ │ │ │ │ + var maxExtent = null; │ │ │ │ │ + var features = this.features; │ │ │ │ │ + if (features && features.length > 0) { │ │ │ │ │ + var geometry = null; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + geometry = features[i].geometry; │ │ │ │ │ + if (geometry) { │ │ │ │ │ + if (maxExtent === null) { │ │ │ │ │ + maxExtent = new OpenLayers.Bounds │ │ │ │ │ + } │ │ │ │ │ + maxExtent.extend(geometry.getBounds()) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return maxExtent │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Vector" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Snapping = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + DEFAULTS: { │ │ │ │ │ + tolerance: 10, │ │ │ │ │ + node: true, │ │ │ │ │ + edge: true, │ │ │ │ │ + vertex: true │ │ │ │ │ + }, │ │ │ │ │ + greedy: true, │ │ │ │ │ + precedence: ["node", "vertex", "edge"], │ │ │ │ │ + resolution: null, │ │ │ │ │ + geoToleranceCache: null, │ │ │ │ │ + layer: null, │ │ │ │ │ + feature: null, │ │ │ │ │ + point: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.options = options || {}; │ │ │ │ │ + if (this.options.layer) { │ │ │ │ │ + this.setLayer(this.options.layer) │ │ │ │ │ + } │ │ │ │ │ + var defaults = OpenLayers.Util.extend({}, this.options.defaults); │ │ │ │ │ + this.defaults = OpenLayers.Util.applyDefaults(defaults, this.DEFAULTS); │ │ │ │ │ + this.setTargets(this.options.targets); │ │ │ │ │ + if (this.targets.length === 0 && this.layer) { │ │ │ │ │ + this.addTargetLayer(this.layer) │ │ │ │ │ + } │ │ │ │ │ + this.geoToleranceCache = {} │ │ │ │ │ + }, │ │ │ │ │ + setLayer: function(layer) { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + this.activate() │ │ │ │ │ + } else { │ │ │ │ │ + this.layer = layer │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setTargets: function(targets) { │ │ │ │ │ + this.targets = []; │ │ │ │ │ + if (targets && targets.length) { │ │ │ │ │ + var target; │ │ │ │ │ + for (var i = 0, len = targets.length; i < len; ++i) { │ │ │ │ │ + target = targets[i]; │ │ │ │ │ + if (target instanceof OpenLayers.Layer.Vector) { │ │ │ │ │ + this.addTargetLayer(target) │ │ │ │ │ + } else { │ │ │ │ │ + this.addTarget(target) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addTargetLayer: function(layer) { │ │ │ │ │ + this.addTarget({ │ │ │ │ │ + layer: layer │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + addTarget: function(target) { │ │ │ │ │ + target = OpenLayers.Util.applyDefaults(target, this.defaults); │ │ │ │ │ + target.nodeTolerance = target.nodeTolerance || target.tolerance; │ │ │ │ │ + target.vertexTolerance = target.vertexTolerance || target.tolerance; │ │ │ │ │ + target.edgeTolerance = target.edgeTolerance || target.tolerance; │ │ │ │ │ + this.targets.push(target) │ │ │ │ │ + }, │ │ │ │ │ + removeTargetLayer: function(layer) { │ │ │ │ │ + var target; │ │ │ │ │ + for (var i = this.targets.length - 1; i >= 0; --i) { │ │ │ │ │ + target = this.targets[i]; │ │ │ │ │ + if (target.layer === layer) { │ │ │ │ │ + this.removeTarget(target) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + removeTarget: function(target) { │ │ │ │ │ + return OpenLayers.Util.removeItem(this.targets, target) │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Control.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + if (this.layer && this.layer.events) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + sketchstarted: this.onSketchModified, │ │ │ │ │ + sketchmodified: this.onSketchModified, │ │ │ │ │ + vertexmodified: this.onVertexModified, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return activated │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Control.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + if (this.layer && this.layer.events) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + sketchstarted: this.onSketchModified, │ │ │ │ │ + sketchmodified: this.onSketchModified, │ │ │ │ │ + vertexmodified: this.onVertexModified, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.feature = null; │ │ │ │ │ + this.point = null; │ │ │ │ │ + return deactivated │ │ │ │ │ + }, │ │ │ │ │ + onSketchModified: function(event) { │ │ │ │ │ + this.feature = event.feature; │ │ │ │ │ + this.considerSnapping(event.vertex, event.vertex) │ │ │ │ │ }, │ │ │ │ │ onVertexModified: function(event) { │ │ │ │ │ this.feature = event.feature; │ │ │ │ │ var loc = this.layer.map.getLonLatFromViewPortPx(event.pixel); │ │ │ │ │ this.considerSnapping(event.vertex, new OpenLayers.Geometry.Point(loc.lon, loc.lat)) │ │ │ │ │ }, │ │ │ │ │ considerSnapping: function(point, loc) { │ │ │ │ │ @@ -30985,220 +30332,662 @@ │ │ │ │ │ } │ │ │ │ │ delete this.layer; │ │ │ │ │ delete this.targets; │ │ │ │ │ OpenLayers.Control.prototype.destroy.call(this) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.Snapping" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.WMTSGetFeatureInfo = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - hover: false, │ │ │ │ │ - requestEncoding: "KVP", │ │ │ │ │ - drillDown: false, │ │ │ │ │ - maxFeatures: 10, │ │ │ │ │ - clickCallback: "click", │ │ │ │ │ +OpenLayers.Control.CacheWrite = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ layers: null, │ │ │ │ │ - queryVisible: true, │ │ │ │ │ - infoFormat: "text/html", │ │ │ │ │ - vendorParams: {}, │ │ │ │ │ - format: null, │ │ │ │ │ - formatOptions: null, │ │ │ │ │ - handler: null, │ │ │ │ │ - hoverRequest: null, │ │ │ │ │ - pending: 0, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!this.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.WMSGetFeatureInfo(options.formatOptions) │ │ │ │ │ - } │ │ │ │ │ - if (this.drillDown === true) { │ │ │ │ │ - this.hover = false │ │ │ │ │ + imageFormat: "image/png", │ │ │ │ │ + quotaRegEx: /quota/i, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + var i, layers = this.layers || map.layers; │ │ │ │ │ + for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + this.addLayer({ │ │ │ │ │ + layer: layers[i] │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ - move: this.cancelHover, │ │ │ │ │ - pause: this.getInfoForHover │ │ │ │ │ - }, OpenLayers.Util.extend(this.handlerOptions.hover || {}, { │ │ │ │ │ - delay: 250 │ │ │ │ │ - })) │ │ │ │ │ - } else { │ │ │ │ │ - var callbacks = {}; │ │ │ │ │ - callbacks[this.clickCallback] = this.getInfoForClick; │ │ │ │ │ - this.handler = new OpenLayers.Handler.Click(this, callbacks, this.handlerOptions.click || {}) │ │ │ │ │ + if (!this.layers) { │ │ │ │ │ + map.events.on({ │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + removeLayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - getInfoForClick: function(evt) { │ │ │ │ │ - this.request(evt.xy, {}) │ │ │ │ │ + addLayer: function(evt) { │ │ │ │ │ + evt.layer.events.on({ │ │ │ │ │ + tileloadstart: this.makeSameOrigin, │ │ │ │ │ + tileloaded: this.onTileLoaded, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - getInfoForHover: function(evt) { │ │ │ │ │ - this.request(evt.xy, { │ │ │ │ │ - hover: true │ │ │ │ │ + removeLayer: function(evt) { │ │ │ │ │ + evt.layer.events.un({ │ │ │ │ │ + tileloadstart: this.makeSameOrigin, │ │ │ │ │ + tileloaded: this.onTileLoaded, │ │ │ │ │ + scope: this │ │ │ │ │ }) │ │ │ │ │ }, │ │ │ │ │ - cancelHover: function() { │ │ │ │ │ - if (this.hoverRequest) { │ │ │ │ │ - --this.pending; │ │ │ │ │ - if (this.pending <= 0) { │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - this.pending = 0 │ │ │ │ │ + makeSameOrigin: function(evt) { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + var tile = evt.tile; │ │ │ │ │ + if (tile instanceof OpenLayers.Tile.Image && !tile.crossOriginKeyword && tile.url.substr(0, 5) !== "data:") { │ │ │ │ │ + var sameOriginUrl = OpenLayers.Request.makeSameOrigin(tile.url, OpenLayers.ProxyHost); │ │ │ │ │ + OpenLayers.Control.CacheWrite.urlMap[sameOriginUrl] = tile.url; │ │ │ │ │ + tile.url = sameOriginUrl │ │ │ │ │ } │ │ │ │ │ - this.hoverRequest.abort(); │ │ │ │ │ - this.hoverRequest = null │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - findLayers: function() { │ │ │ │ │ - var candidates = this.layers || this.map.layers; │ │ │ │ │ - var layers = []; │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = candidates[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.WMTS && layer.requestEncoding === this.requestEncoding && (!this.queryVisible || layer.getVisibility())) { │ │ │ │ │ - layers.push(layer); │ │ │ │ │ - if (!this.drillDown || this.hover) { │ │ │ │ │ - break │ │ │ │ │ + onTileLoaded: function(evt) { │ │ │ │ │ + if (this.active && !evt.aborted && evt.tile instanceof OpenLayers.Tile.Image && evt.tile.url.substr(0, 5) !== "data:") { │ │ │ │ │ + this.cache({ │ │ │ │ │ + tile: evt.tile │ │ │ │ │ + }); │ │ │ │ │ + delete OpenLayers.Control.CacheWrite.urlMap[evt.tile.url] │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + cache: function(obj) { │ │ │ │ │ + if (window.localStorage) { │ │ │ │ │ + var tile = obj.tile; │ │ │ │ │ + try { │ │ │ │ │ + var canvasContext = tile.getCanvasContext(); │ │ │ │ │ + if (canvasContext) { │ │ │ │ │ + var urlMap = OpenLayers.Control.CacheWrite.urlMap; │ │ │ │ │ + var url = urlMap[tile.url] || tile.url; │ │ │ │ │ + window.localStorage.setItem("olCache_" + url, canvasContext.canvas.toDataURL(this.imageFormat)) │ │ │ │ │ + } │ │ │ │ │ + } catch (e) { │ │ │ │ │ + var reason = e.name || e.message; │ │ │ │ │ + if (reason && this.quotaRegEx.test(reason)) { │ │ │ │ │ + this.events.triggerEvent("cachefull", { │ │ │ │ │ + tile: tile │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Console.error(e.toString()) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return layers │ │ │ │ │ }, │ │ │ │ │ - buildRequestOptions: function(layer, xy) { │ │ │ │ │ - var loc = this.map.getLonLatFromPixel(xy); │ │ │ │ │ - var getTileUrl = layer.getURL(new OpenLayers.Bounds(loc.lon, loc.lat, loc.lon, loc.lat)); │ │ │ │ │ - var params = OpenLayers.Util.getParameters(getTileUrl); │ │ │ │ │ - var tileInfo = layer.getTileInfo(loc); │ │ │ │ │ - OpenLayers.Util.extend(params, { │ │ │ │ │ - service: "WMTS", │ │ │ │ │ - version: layer.version, │ │ │ │ │ - request: "GetFeatureInfo", │ │ │ │ │ - infoFormat: this.infoFormat, │ │ │ │ │ - i: tileInfo.i, │ │ │ │ │ - j: tileInfo.j │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Util.applyDefaults(params, this.vendorParams); │ │ │ │ │ - return { │ │ │ │ │ - url: OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url, │ │ │ │ │ - params: OpenLayers.Util.upperCaseObject(params), │ │ │ │ │ - callback: function(request) { │ │ │ │ │ - this.handleResponse(xy, request, layer) │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.layers || this.map) { │ │ │ │ │ + var i, layers = this.layers || this.map.layers; │ │ │ │ │ + for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + this.removeLayer({ │ │ │ │ │ + layer: layers[i] │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + removeLayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - request: function(xy, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var layers = this.findLayers(); │ │ │ │ │ - if (layers.length > 0) { │ │ │ │ │ - var issue, layer; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - layer = layers[i]; │ │ │ │ │ - issue = this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ - xy: xy, │ │ │ │ │ - layer: layer │ │ │ │ │ - }); │ │ │ │ │ - if (issue !== false) { │ │ │ │ │ - ++this.pending; │ │ │ │ │ - var requestOptions = this.buildRequestOptions(layer, xy); │ │ │ │ │ - var request = OpenLayers.Request.GET(requestOptions); │ │ │ │ │ - if (options.hover === true) { │ │ │ │ │ - this.hoverRequest = request │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.CacheWrite" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.CacheWrite.clearCache = function() { │ │ │ │ │ + if (!window.localStorage) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var i, key; │ │ │ │ │ + for (i = window.localStorage.length - 1; i >= 0; --i) { │ │ │ │ │ + key = window.localStorage.key(i); │ │ │ │ │ + if (key.substr(0, 8) === "olCache_") { │ │ │ │ │ + window.localStorage.removeItem(key) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Control.CacheWrite.urlMap = {}; │ │ │ │ │ +OpenLayers.Control.PanPanel = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ + slideFactor: 50, │ │ │ │ │ + slideRatio: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ + var options = { │ │ │ │ │ + slideFactor: this.slideFactor, │ │ │ │ │ + slideRatio: this.slideRatio │ │ │ │ │ + }; │ │ │ │ │ + this.addControls([new OpenLayers.Control.Pan(OpenLayers.Control.Pan.NORTH, options), new OpenLayers.Control.Pan(OpenLayers.Control.Pan.SOUTH, options), new OpenLayers.Control.Pan(OpenLayers.Control.Pan.EAST, options), new OpenLayers.Control.Pan(OpenLayers.Control.Pan.WEST, options)]) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.PanPanel" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.MousePosition = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + element: null, │ │ │ │ │ + prefix: "", │ │ │ │ │ + separator: ", ", │ │ │ │ │ + suffix: "", │ │ │ │ │ + numDigits: 5, │ │ │ │ │ + granularity: 10, │ │ │ │ │ + emptyString: null, │ │ │ │ │ + lastXy: null, │ │ │ │ │ + displayProjection: null, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.map.events.register("mousemove", this, this.redraw); │ │ │ │ │ + this.map.events.register("mouseout", this, this.reset); │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.map.events.unregister("mousemove", this, this.redraw); │ │ │ │ │ + this.map.events.unregister("mouseout", this, this.reset); │ │ │ │ │ + this.element.innerHTML = ""; │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (!this.element) { │ │ │ │ │ + this.div.left = ""; │ │ │ │ │ + this.div.top = ""; │ │ │ │ │ + this.element = this.div │ │ │ │ │ + } │ │ │ │ │ + return this.div │ │ │ │ │ + }, │ │ │ │ │ + redraw: function(evt) { │ │ │ │ │ + var lonLat; │ │ │ │ │ + if (evt == null) { │ │ │ │ │ + this.reset(); │ │ │ │ │ + return │ │ │ │ │ + } else { │ │ │ │ │ + if (this.lastXy == null || Math.abs(evt.xy.x - this.lastXy.x) > this.granularity || Math.abs(evt.xy.y - this.lastXy.y) > this.granularity) { │ │ │ │ │ + this.lastXy = evt.xy; │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - if (this.pending > 0) { │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait") │ │ │ │ │ + lonLat = this.map.getLonLatFromPixel(evt.xy); │ │ │ │ │ + if (!lonLat) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (this.displayProjection) { │ │ │ │ │ + lonLat.transform(this.map.getProjectionObject(), this.displayProjection) │ │ │ │ │ } │ │ │ │ │ + this.lastXy = evt.xy │ │ │ │ │ + } │ │ │ │ │ + var newHtml = this.formatOutput(lonLat); │ │ │ │ │ + if (newHtml != this.element.innerHTML) { │ │ │ │ │ + this.element.innerHTML = newHtml │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - handleResponse: function(xy, request, layer) { │ │ │ │ │ - --this.pending; │ │ │ │ │ - if (this.pending <= 0) { │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - this.pending = 0 │ │ │ │ │ + reset: function(evt) { │ │ │ │ │ + if (this.emptyString != null) { │ │ │ │ │ + this.element.innerHTML = this.emptyString │ │ │ │ │ } │ │ │ │ │ - if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ - this.events.triggerEvent("exception", { │ │ │ │ │ - xy: xy, │ │ │ │ │ - request: request, │ │ │ │ │ - layer: layer │ │ │ │ │ + }, │ │ │ │ │ + formatOutput: function(lonLat) { │ │ │ │ │ + var digits = parseInt(this.numDigits); │ │ │ │ │ + var newHtml = this.prefix + lonLat.lon.toFixed(digits) + this.separator + lonLat.lat.toFixed(digits) + this.suffix; │ │ │ │ │ + return newHtml │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.MousePosition" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Graticule = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + intervals: [45, 30, 20, 10, 5, 2, 1, .5, .2, .1, .05, .01, .005, .002, .001], │ │ │ │ │ + displayInLayerSwitcher: true, │ │ │ │ │ + visible: true, │ │ │ │ │ + numPoints: 50, │ │ │ │ │ + targetSize: 200, │ │ │ │ │ + layerName: null, │ │ │ │ │ + labelled: true, │ │ │ │ │ + labelFormat: "dm", │ │ │ │ │ + lineSymbolizer: { │ │ │ │ │ + strokeColor: "#333", │ │ │ │ │ + strokeWidth: 1, │ │ │ │ │ + strokeOpacity: .5 │ │ │ │ │ + }, │ │ │ │ │ + labelSymbolizer: {}, │ │ │ │ │ + gratLayer: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.layerName = options.layerName || OpenLayers.i18n("Graticule"); │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.labelSymbolizer.stroke = false; │ │ │ │ │ + this.labelSymbolizer.fill = false; │ │ │ │ │ + this.labelSymbolizer.label = "${label}"; │ │ │ │ │ + this.labelSymbolizer.labelAlign = "${labelAlign}"; │ │ │ │ │ + this.labelSymbolizer.labelXOffset = "${xOffset}"; │ │ │ │ │ + this.labelSymbolizer.labelYOffset = "${yOffset}" │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + if (this.gratLayer) { │ │ │ │ │ + this.gratLayer.destroy(); │ │ │ │ │ + this.gratLayer = null │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (!this.gratLayer) { │ │ │ │ │ + var gratStyle = new OpenLayers.Style({}, { │ │ │ │ │ + rules: [new OpenLayers.Rule({ │ │ │ │ │ + symbolizer: { │ │ │ │ │ + Point: this.labelSymbolizer, │ │ │ │ │ + Line: this.lineSymbolizer │ │ │ │ │ + } │ │ │ │ │ + })] │ │ │ │ │ + }); │ │ │ │ │ + this.gratLayer = new OpenLayers.Layer.Vector(this.layerName, { │ │ │ │ │ + styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ + default: gratStyle │ │ │ │ │ + }), │ │ │ │ │ + visibility: this.visible, │ │ │ │ │ + displayInLayerSwitcher: this.displayInLayerSwitcher │ │ │ │ │ }) │ │ │ │ │ + } │ │ │ │ │ + return this.div │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.map.addLayer(this.gratLayer); │ │ │ │ │ + this.map.events.register("moveend", this, this.update); │ │ │ │ │ + this.update(); │ │ │ │ │ + return true │ │ │ │ │ } else { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.map.events.unregister("moveend", this, this.update); │ │ │ │ │ + this.map.removeLayer(this.gratLayer); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + update: function() { │ │ │ │ │ + var mapBounds = this.map.getExtent(); │ │ │ │ │ + if (!mapBounds) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + this.gratLayer.destroyFeatures(); │ │ │ │ │ + var llProj = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + var mapProj = this.map.getProjectionObject(); │ │ │ │ │ + var mapRes = this.map.getResolution(); │ │ │ │ │ + if (mapProj.proj && mapProj.proj.projName == "longlat") { │ │ │ │ │ + this.numPoints = 1 │ │ │ │ │ + } │ │ │ │ │ + var mapCenter = this.map.getCenter(); │ │ │ │ │ + var mapCenterLL = new OpenLayers.Pixel(mapCenter.lon, mapCenter.lat); │ │ │ │ │ + OpenLayers.Projection.transform(mapCenterLL, mapProj, llProj); │ │ │ │ │ + var testSq = this.targetSize * mapRes; │ │ │ │ │ + testSq *= testSq; │ │ │ │ │ + var llInterval; │ │ │ │ │ + for (var i = 0; i < this.intervals.length; ++i) { │ │ │ │ │ + llInterval = this.intervals[i]; │ │ │ │ │ + var delta = llInterval / 2; │ │ │ │ │ + var p1 = mapCenterLL.offset({ │ │ │ │ │ + x: -delta, │ │ │ │ │ + y: -delta │ │ │ │ │ + }); │ │ │ │ │ + var p2 = mapCenterLL.offset({ │ │ │ │ │ + x: delta, │ │ │ │ │ + y: delta │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Projection.transform(p1, llProj, mapProj); │ │ │ │ │ + OpenLayers.Projection.transform(p2, llProj, mapProj); │ │ │ │ │ + var distSq = (p1.x - p2.x) * (p1.x - p2.x) + (p1.y - p2.y) * (p1.y - p2.y); │ │ │ │ │ + if (distSq <= testSq) { │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - var features, except; │ │ │ │ │ - try { │ │ │ │ │ - features = this.format.read(doc) │ │ │ │ │ - } catch (error) { │ │ │ │ │ - except = true; │ │ │ │ │ - this.events.triggerEvent("exception", { │ │ │ │ │ - xy: xy, │ │ │ │ │ - request: request, │ │ │ │ │ - error: error, │ │ │ │ │ - layer: layer │ │ │ │ │ - }) │ │ │ │ │ + } │ │ │ │ │ + mapCenterLL.x = Math.floor(mapCenterLL.x / llInterval) * llInterval; │ │ │ │ │ + mapCenterLL.y = Math.floor(mapCenterLL.y / llInterval) * llInterval; │ │ │ │ │ + var iter = 0; │ │ │ │ │ + var centerLonPoints = [mapCenterLL.clone()]; │ │ │ │ │ + var newPoint = mapCenterLL.clone(); │ │ │ │ │ + var mapXY; │ │ │ │ │ + do { │ │ │ │ │ + newPoint = newPoint.offset({ │ │ │ │ │ + x: 0, │ │ │ │ │ + y: llInterval │ │ │ │ │ + }); │ │ │ │ │ + mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ + centerLonPoints.unshift(newPoint) │ │ │ │ │ + } while (mapBounds.containsPixel(mapXY) && ++iter < 1e3); │ │ │ │ │ + newPoint = mapCenterLL.clone(); │ │ │ │ │ + do { │ │ │ │ │ + newPoint = newPoint.offset({ │ │ │ │ │ + x: 0, │ │ │ │ │ + y: -llInterval │ │ │ │ │ + }); │ │ │ │ │ + mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ + centerLonPoints.push(newPoint) │ │ │ │ │ + } while (mapBounds.containsPixel(mapXY) && ++iter < 1e3); │ │ │ │ │ + iter = 0; │ │ │ │ │ + var centerLatPoints = [mapCenterLL.clone()]; │ │ │ │ │ + newPoint = mapCenterLL.clone(); │ │ │ │ │ + do { │ │ │ │ │ + newPoint = newPoint.offset({ │ │ │ │ │ + x: -llInterval, │ │ │ │ │ + y: 0 │ │ │ │ │ + }); │ │ │ │ │ + mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ + centerLatPoints.unshift(newPoint) │ │ │ │ │ + } while (mapBounds.containsPixel(mapXY) && ++iter < 1e3); │ │ │ │ │ + newPoint = mapCenterLL.clone(); │ │ │ │ │ + do { │ │ │ │ │ + newPoint = newPoint.offset({ │ │ │ │ │ + x: llInterval, │ │ │ │ │ + y: 0 │ │ │ │ │ + }); │ │ │ │ │ + mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ + centerLatPoints.push(newPoint) │ │ │ │ │ + } while (mapBounds.containsPixel(mapXY) && ++iter < 1e3); │ │ │ │ │ + var lines = []; │ │ │ │ │ + for (var i = 0; i < centerLatPoints.length; ++i) { │ │ │ │ │ + var lon = centerLatPoints[i].x; │ │ │ │ │ + var pointList = []; │ │ │ │ │ + var labelPoint = null; │ │ │ │ │ + var latEnd = Math.min(centerLonPoints[0].y, 90); │ │ │ │ │ + var latStart = Math.max(centerLonPoints[centerLonPoints.length - 1].y, -90); │ │ │ │ │ + var latDelta = (latEnd - latStart) / this.numPoints; │ │ │ │ │ + var lat = latStart; │ │ │ │ │ + for (var j = 0; j <= this.numPoints; ++j) { │ │ │ │ │ + var gridPoint = new OpenLayers.Geometry.Point(lon, lat); │ │ │ │ │ + gridPoint.transform(llProj, mapProj); │ │ │ │ │ + pointList.push(gridPoint); │ │ │ │ │ + lat += latDelta; │ │ │ │ │ + if (gridPoint.y >= mapBounds.bottom && !labelPoint) { │ │ │ │ │ + labelPoint = gridPoint │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (!except) { │ │ │ │ │ - this.events.triggerEvent("getfeatureinfo", { │ │ │ │ │ - text: request.responseText, │ │ │ │ │ - features: features, │ │ │ │ │ - request: request, │ │ │ │ │ - xy: xy, │ │ │ │ │ - layer: layer │ │ │ │ │ - }) │ │ │ │ │ + if (this.labelled) { │ │ │ │ │ + var labelPos = new OpenLayers.Geometry.Point(labelPoint.x, mapBounds.bottom); │ │ │ │ │ + var labelAttrs = { │ │ │ │ │ + value: lon, │ │ │ │ │ + label: this.labelled ? OpenLayers.Util.getFormattedLonLat(lon, "lon", this.labelFormat) : "", │ │ │ │ │ + labelAlign: "cb", │ │ │ │ │ + xOffset: 0, │ │ │ │ │ + yOffset: 2 │ │ │ │ │ + }; │ │ │ │ │ + this.gratLayer.addFeatures(new OpenLayers.Feature.Vector(labelPos, labelAttrs)) │ │ │ │ │ + } │ │ │ │ │ + var geom = new OpenLayers.Geometry.LineString(pointList); │ │ │ │ │ + lines.push(new OpenLayers.Feature.Vector(geom)) │ │ │ │ │ + } │ │ │ │ │ + for (var j = 0; j < centerLonPoints.length; ++j) { │ │ │ │ │ + lat = centerLonPoints[j].y; │ │ │ │ │ + if (lat < -90 || lat > 90) { │ │ │ │ │ + continue │ │ │ │ │ + } │ │ │ │ │ + var pointList = []; │ │ │ │ │ + var lonStart = centerLatPoints[0].x; │ │ │ │ │ + var lonEnd = centerLatPoints[centerLatPoints.length - 1].x; │ │ │ │ │ + var lonDelta = (lonEnd - lonStart) / this.numPoints; │ │ │ │ │ + var lon = lonStart; │ │ │ │ │ + var labelPoint = null; │ │ │ │ │ + for (var i = 0; i <= this.numPoints; ++i) { │ │ │ │ │ + var gridPoint = new OpenLayers.Geometry.Point(lon, lat); │ │ │ │ │ + gridPoint.transform(llProj, mapProj); │ │ │ │ │ + pointList.push(gridPoint); │ │ │ │ │ + lon += lonDelta; │ │ │ │ │ + if (gridPoint.x < mapBounds.right) { │ │ │ │ │ + labelPoint = gridPoint │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.labelled) { │ │ │ │ │ + var labelPos = new OpenLayers.Geometry.Point(mapBounds.right, labelPoint.y); │ │ │ │ │ + var labelAttrs = { │ │ │ │ │ + value: lat, │ │ │ │ │ + label: this.labelled ? OpenLayers.Util.getFormattedLonLat(lat, "lat", this.labelFormat) : "", │ │ │ │ │ + labelAlign: "rb", │ │ │ │ │ + xOffset: -2, │ │ │ │ │ + yOffset: 2 │ │ │ │ │ + }; │ │ │ │ │ + this.gratLayer.addFeatures(new OpenLayers.Feature.Vector(labelPos, labelAttrs)) │ │ │ │ │ } │ │ │ │ │ + var geom = new OpenLayers.Geometry.LineString(pointList); │ │ │ │ │ + lines.push(new OpenLayers.Feature.Vector(geom)) │ │ │ │ │ } │ │ │ │ │ + this.gratLayer.addFeatures(lines) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.WMTSGetFeatureInfo" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Graticule" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.PinchZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ - pinchOrigin: null, │ │ │ │ │ - currentCenter: null, │ │ │ │ │ +OpenLayers.Control.Navigation = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + dragPan: null, │ │ │ │ │ + dragPanOptions: null, │ │ │ │ │ + pinchZoom: null, │ │ │ │ │ + pinchZoomOptions: null, │ │ │ │ │ + documentDrag: false, │ │ │ │ │ + zoomBox: null, │ │ │ │ │ + zoomBoxEnabled: true, │ │ │ │ │ + zoomWheelEnabled: true, │ │ │ │ │ + mouseWheelOptions: null, │ │ │ │ │ + handleRightClicks: false, │ │ │ │ │ + zoomBoxKeyMask: OpenLayers.Handler.MOD_SHIFT, │ │ │ │ │ autoActivate: true, │ │ │ │ │ - preserveCenter: false, │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.handler = new OpenLayers.Handler.Pinch(this, { │ │ │ │ │ - start: this.pinchStart, │ │ │ │ │ - move: this.pinchMove, │ │ │ │ │ - done: this.pinchDone │ │ │ │ │ - }, this.handlerOptions) │ │ │ │ │ + this.handlers = {}; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - pinchStart: function(evt, pinchData) { │ │ │ │ │ - var xy = this.preserveCenter ? this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ - this.pinchOrigin = xy; │ │ │ │ │ - this.currentCenter = xy │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + if (this.dragPan) { │ │ │ │ │ + this.dragPan.destroy() │ │ │ │ │ + } │ │ │ │ │ + this.dragPan = null; │ │ │ │ │ + if (this.zoomBox) { │ │ │ │ │ + this.zoomBox.destroy() │ │ │ │ │ + } │ │ │ │ │ + this.zoomBox = null; │ │ │ │ │ + if (this.pinchZoom) { │ │ │ │ │ + this.pinchZoom.destroy() │ │ │ │ │ + } │ │ │ │ │ + this.pinchZoom = null; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - pinchMove: function(evt, pinchData) { │ │ │ │ │ - var scale = pinchData.scale; │ │ │ │ │ - var containerOrigin = this.map.layerContainerOriginPx; │ │ │ │ │ - var pinchOrigin = this.pinchOrigin; │ │ │ │ │ - var current = this.preserveCenter ? this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ - var dx = Math.round(containerOrigin.x + current.x - pinchOrigin.x + (scale - 1) * (containerOrigin.x - pinchOrigin.x)); │ │ │ │ │ - var dy = Math.round(containerOrigin.y + current.y - pinchOrigin.y + (scale - 1) * (containerOrigin.y - pinchOrigin.y)); │ │ │ │ │ - this.map.applyTransform(dx, dy, scale); │ │ │ │ │ - this.currentCenter = current │ │ │ │ │ + activate: function() { │ │ │ │ │ + this.dragPan.activate(); │ │ │ │ │ + if (this.zoomWheelEnabled) { │ │ │ │ │ + this.handlers.wheel.activate() │ │ │ │ │ + } │ │ │ │ │ + this.handlers.click.activate(); │ │ │ │ │ + if (this.zoomBoxEnabled) { │ │ │ │ │ + this.zoomBox.activate() │ │ │ │ │ + } │ │ │ │ │ + if (this.pinchZoom) { │ │ │ │ │ + this.pinchZoom.activate() │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Control.prototype.activate.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - pinchDone: function(evt, start, last) { │ │ │ │ │ - this.map.applyTransform(); │ │ │ │ │ - var zoom = this.map.getZoomForResolution(this.map.getResolution() / last.scale, true); │ │ │ │ │ - if (zoom !== this.map.getZoom() || !this.currentCenter.equals(this.pinchOrigin)) { │ │ │ │ │ - var resolution = this.map.getResolutionForZoom(zoom); │ │ │ │ │ - var location = this.map.getLonLatFromPixel(this.pinchOrigin); │ │ │ │ │ - var zoomPixel = this.currentCenter; │ │ │ │ │ - var size = this.map.getSize(); │ │ │ │ │ - location.lon += resolution * (size.w / 2 - zoomPixel.x); │ │ │ │ │ - location.lat -= resolution * (size.h / 2 - zoomPixel.y); │ │ │ │ │ - this.map.div.clientWidth = this.map.div.clientWidth; │ │ │ │ │ - this.map.setCenter(location, zoom) │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.pinchZoom) { │ │ │ │ │ + this.pinchZoom.deactivate() │ │ │ │ │ + } │ │ │ │ │ + this.zoomBox.deactivate(); │ │ │ │ │ + this.dragPan.deactivate(); │ │ │ │ │ + this.handlers.click.deactivate(); │ │ │ │ │ + this.handlers.wheel.deactivate(); │ │ │ │ │ + return OpenLayers.Control.prototype.deactivate.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + if (this.handleRightClicks) { │ │ │ │ │ + this.map.viewPortDiv.oncontextmenu = OpenLayers.Function.False │ │ │ │ │ + } │ │ │ │ │ + var clickCallbacks = { │ │ │ │ │ + click: this.defaultClick, │ │ │ │ │ + dblclick: this.defaultDblClick, │ │ │ │ │ + dblrightclick: this.defaultDblRightClick │ │ │ │ │ + }; │ │ │ │ │ + var clickOptions = { │ │ │ │ │ + double: true, │ │ │ │ │ + stopDouble: true │ │ │ │ │ + }; │ │ │ │ │ + this.handlers.click = new OpenLayers.Handler.Click(this, clickCallbacks, clickOptions); │ │ │ │ │ + this.dragPan = new OpenLayers.Control.DragPan(OpenLayers.Util.extend({ │ │ │ │ │ + map: this.map, │ │ │ │ │ + documentDrag: this.documentDrag │ │ │ │ │ + }, this.dragPanOptions)); │ │ │ │ │ + this.zoomBox = new OpenLayers.Control.ZoomBox({ │ │ │ │ │ + map: this.map, │ │ │ │ │ + keyMask: this.zoomBoxKeyMask │ │ │ │ │ + }); │ │ │ │ │ + this.dragPan.draw(); │ │ │ │ │ + this.zoomBox.draw(); │ │ │ │ │ + var wheelOptions = this.map.fractionalZoom ? {} : { │ │ │ │ │ + cumulative: false, │ │ │ │ │ + interval: 50, │ │ │ │ │ + maxDelta: 6 │ │ │ │ │ + }; │ │ │ │ │ + this.handlers.wheel = new OpenLayers.Handler.MouseWheel(this, { │ │ │ │ │ + up: this.wheelUp, │ │ │ │ │ + down: this.wheelDown │ │ │ │ │ + }, OpenLayers.Util.extend(wheelOptions, this.mouseWheelOptions)); │ │ │ │ │ + if (OpenLayers.Control.PinchZoom) { │ │ │ │ │ + this.pinchZoom = new OpenLayers.Control.PinchZoom(OpenLayers.Util.extend({ │ │ │ │ │ + map: this.map │ │ │ │ │ + }, this.pinchZoomOptions)) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.PinchZoom" │ │ │ │ │ + defaultClick: function(evt) { │ │ │ │ │ + if (evt.lastTouches && evt.lastTouches.length == 2) { │ │ │ │ │ + this.map.zoomOut() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + defaultDblClick: function(evt) { │ │ │ │ │ + this.map.zoomTo(this.map.zoom + 1, evt.xy) │ │ │ │ │ + }, │ │ │ │ │ + defaultDblRightClick: function(evt) { │ │ │ │ │ + this.map.zoomTo(this.map.zoom - 1, evt.xy) │ │ │ │ │ + }, │ │ │ │ │ + wheelChange: function(evt, deltaZ) { │ │ │ │ │ + if (!this.map.fractionalZoom) { │ │ │ │ │ + deltaZ = Math.round(deltaZ) │ │ │ │ │ + } │ │ │ │ │ + var currentZoom = this.map.getZoom(), │ │ │ │ │ + newZoom = currentZoom + deltaZ; │ │ │ │ │ + newZoom = Math.max(newZoom, 0); │ │ │ │ │ + newZoom = Math.min(newZoom, this.map.getNumZoomLevels()); │ │ │ │ │ + if (newZoom === currentZoom) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + this.map.zoomTo(newZoom, evt.xy) │ │ │ │ │ + }, │ │ │ │ │ + wheelUp: function(evt, delta) { │ │ │ │ │ + this.wheelChange(evt, delta || 1) │ │ │ │ │ + }, │ │ │ │ │ + wheelDown: function(evt, delta) { │ │ │ │ │ + this.wheelChange(evt, delta || -1) │ │ │ │ │ + }, │ │ │ │ │ + disableZoomBox: function() { │ │ │ │ │ + this.zoomBoxEnabled = false; │ │ │ │ │ + this.zoomBox.deactivate() │ │ │ │ │ + }, │ │ │ │ │ + enableZoomBox: function() { │ │ │ │ │ + this.zoomBoxEnabled = true; │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.zoomBox.activate() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + disableZoomWheel: function() { │ │ │ │ │ + this.zoomWheelEnabled = false; │ │ │ │ │ + this.handlers.wheel.deactivate() │ │ │ │ │ + }, │ │ │ │ │ + enableZoomWheel: function() { │ │ │ │ │ + this.zoomWheelEnabled = true; │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.handlers.wheel.activate() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Navigation" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.NavToolbar = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.addControls([new OpenLayers.Control.Navigation, new OpenLayers.Control.ZoomBox]) │ │ │ │ │ + }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + var div = OpenLayers.Control.Panel.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (this.defaultControl === null) { │ │ │ │ │ + this.defaultControl = this.controls[0] │ │ │ │ │ + } │ │ │ │ │ + return div │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.NavToolbar" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.CacheRead = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + fetchEvent: "tileloadstart", │ │ │ │ │ + layers: null, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + var i, layers = this.layers || map.layers; │ │ │ │ │ + for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + this.addLayer({ │ │ │ │ │ + layer: layers[i] │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + if (!this.layers) { │ │ │ │ │ + map.events.on({ │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + removeLayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addLayer: function(evt) { │ │ │ │ │ + evt.layer.events.register(this.fetchEvent, this, this.fetch) │ │ │ │ │ + }, │ │ │ │ │ + removeLayer: function(evt) { │ │ │ │ │ + evt.layer.events.unregister(this.fetchEvent, this, this.fetch) │ │ │ │ │ + }, │ │ │ │ │ + fetch: function(evt) { │ │ │ │ │ + if (this.active && window.localStorage && evt.tile instanceof OpenLayers.Tile.Image) { │ │ │ │ │ + var tile = evt.tile, │ │ │ │ │ + url = tile.url; │ │ │ │ │ + if (!tile.layer.crossOriginKeyword && OpenLayers.ProxyHost && url.indexOf(OpenLayers.ProxyHost) === 0) { │ │ │ │ │ + url = OpenLayers.Control.CacheWrite.urlMap[url] │ │ │ │ │ + } │ │ │ │ │ + var dataURI = window.localStorage.getItem("olCache_" + url); │ │ │ │ │ + if (dataURI) { │ │ │ │ │ + tile.url = dataURI; │ │ │ │ │ + if (evt.type === "tileerror") { │ │ │ │ │ + tile.setImgSrc(dataURI) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.layers || this.map) { │ │ │ │ │ + var i, layers = this.layers || this.map.layers; │ │ │ │ │ + for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + this.removeLayer({ │ │ │ │ │ + layer: layers[i] │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + removeLayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.CacheRead" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Control.OverviewMap = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ element: null, │ │ │ │ │ ovmap: null, │ │ │ │ │ size: { │ │ │ │ │ w: 180, │ │ │ │ │ h: 90 │ │ │ │ │ @@ -31559,529 +31348,251 @@ │ │ │ │ │ x: Math.round(1 / res * (lonlat.lon - extent.left)), │ │ │ │ │ y: Math.round(1 / res * (extent.top - lonlat.lat)) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.OverviewMap" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.WMSGetFeatureInfo = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - hover: false, │ │ │ │ │ - drillDown: false, │ │ │ │ │ - maxFeatures: 10, │ │ │ │ │ - clickCallback: "click", │ │ │ │ │ - output: "features", │ │ │ │ │ - layers: null, │ │ │ │ │ - queryVisible: false, │ │ │ │ │ - url: null, │ │ │ │ │ - layerUrls: null, │ │ │ │ │ - infoFormat: "text/html", │ │ │ │ │ - vendorParams: {}, │ │ │ │ │ - format: null, │ │ │ │ │ - formatOptions: null, │ │ │ │ │ - handler: null, │ │ │ │ │ - hoverRequest: null, │ │ │ │ │ +OpenLayers.Control.LayerSwitcher = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + layerStates: null, │ │ │ │ │ + layersDiv: null, │ │ │ │ │ + baseLayersDiv: null, │ │ │ │ │ + baseLayers: null, │ │ │ │ │ + dataLbl: null, │ │ │ │ │ + dataLayersDiv: null, │ │ │ │ │ + dataLayers: null, │ │ │ │ │ + minimizeDiv: null, │ │ │ │ │ + maximizeDiv: null, │ │ │ │ │ + ascending: true, │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!this.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.WMSGetFeatureInfo(options.formatOptions) │ │ │ │ │ - } │ │ │ │ │ - if (this.drillDown === true) { │ │ │ │ │ - this.hover = false │ │ │ │ │ - } │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ - move: this.cancelHover, │ │ │ │ │ - pause: this.getInfoForHover │ │ │ │ │ - }, OpenLayers.Util.extend(this.handlerOptions.hover || {}, { │ │ │ │ │ - delay: 250 │ │ │ │ │ - })) │ │ │ │ │ - } else { │ │ │ │ │ - var callbacks = {}; │ │ │ │ │ - callbacks[this.clickCallback] = this.getInfoForClick; │ │ │ │ │ - this.handler = new OpenLayers.Handler.Click(this, callbacks, this.handlerOptions.click || {}) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getInfoForClick: function(evt) { │ │ │ │ │ - this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ - xy: evt.xy │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - this.request(evt.xy, {}) │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.layerStates = [] │ │ │ │ │ }, │ │ │ │ │ - getInfoForHover: function(evt) { │ │ │ │ │ - this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ - xy: evt.xy │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.clearLayersArray("base"); │ │ │ │ │ + this.clearLayersArray("data"); │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + buttonclick: this.onButtonClick, │ │ │ │ │ + addlayer: this.redraw, │ │ │ │ │ + changelayer: this.redraw, │ │ │ │ │ + removelayer: this.redraw, │ │ │ │ │ + changebaselayer: this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ }); │ │ │ │ │ - this.request(evt.xy, { │ │ │ │ │ - hover: true │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - cancelHover: function() { │ │ │ │ │ - if (this.hoverRequest) { │ │ │ │ │ - this.hoverRequest.abort(); │ │ │ │ │ - this.hoverRequest = null │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - findLayers: function() { │ │ │ │ │ - var candidates = this.layers || this.map.layers; │ │ │ │ │ - var layers = []; │ │ │ │ │ - var layer, url; │ │ │ │ │ - for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = candidates[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.WMS && (!this.queryVisible || layer.getVisibility())) { │ │ │ │ │ - url = OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url; │ │ │ │ │ - if (this.drillDown === false && !this.url) { │ │ │ │ │ - this.url = url │ │ │ │ │ - } │ │ │ │ │ - if (this.drillDown === true || this.urlMatches(url)) { │ │ │ │ │ - layers.push(layer) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return layers │ │ │ │ │ - }, │ │ │ │ │ - urlMatches: function(url) { │ │ │ │ │ - var matches = OpenLayers.Util.isEquivalentUrl(this.url, url); │ │ │ │ │ - if (!matches && this.layerUrls) { │ │ │ │ │ - for (var i = 0, len = this.layerUrls.length; i < len; ++i) { │ │ │ │ │ - if (OpenLayers.Util.isEquivalentUrl(this.layerUrls[i], url)) { │ │ │ │ │ - matches = true; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return matches │ │ │ │ │ + this.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - buildWMSOptions: function(url, layers, clickPosition, format) { │ │ │ │ │ - var layerNames = [], │ │ │ │ │ - styleNames = []; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - if (layers[i].params.LAYERS != null) { │ │ │ │ │ - layerNames = layerNames.concat(layers[i].params.LAYERS); │ │ │ │ │ - styleNames = styleNames.concat(this.getStyleNames(layers[i])) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var firstLayer = layers[0]; │ │ │ │ │ - var projection = this.map.getProjection(); │ │ │ │ │ - var layerProj = firstLayer.projection; │ │ │ │ │ - if (layerProj && layerProj.equals(this.map.getProjectionObject())) { │ │ │ │ │ - projection = layerProj.getCode() │ │ │ │ │ - } │ │ │ │ │ - var params = OpenLayers.Util.extend({ │ │ │ │ │ - service: "WMS", │ │ │ │ │ - version: firstLayer.params.VERSION, │ │ │ │ │ - request: "GetFeatureInfo", │ │ │ │ │ - exceptions: firstLayer.params.EXCEPTIONS, │ │ │ │ │ - bbox: this.map.getExtent().toBBOX(null, firstLayer.reverseAxisOrder()), │ │ │ │ │ - feature_count: this.maxFeatures, │ │ │ │ │ - height: this.map.getSize().h, │ │ │ │ │ - width: this.map.getSize().w, │ │ │ │ │ - format: format, │ │ │ │ │ - info_format: firstLayer.params.INFO_FORMAT || this.infoFormat │ │ │ │ │ - }, parseFloat(firstLayer.params.VERSION) >= 1.3 ? { │ │ │ │ │ - crs: projection, │ │ │ │ │ - i: parseInt(clickPosition.x), │ │ │ │ │ - j: parseInt(clickPosition.y) │ │ │ │ │ - } : { │ │ │ │ │ - srs: projection, │ │ │ │ │ - x: parseInt(clickPosition.x), │ │ │ │ │ - y: parseInt(clickPosition.y) │ │ │ │ │ - }); │ │ │ │ │ - if (layerNames.length != 0) { │ │ │ │ │ - params = OpenLayers.Util.extend({ │ │ │ │ │ - layers: layerNames, │ │ │ │ │ - query_layers: layerNames, │ │ │ │ │ - styles: styleNames │ │ │ │ │ - }, params) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.applyDefaults(params, this.vendorParams); │ │ │ │ │ - return { │ │ │ │ │ - url: url, │ │ │ │ │ - params: OpenLayers.Util.upperCaseObject(params), │ │ │ │ │ - callback: function(request) { │ │ │ │ │ - this.handleResponse(clickPosition, request, url) │ │ │ │ │ - }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + addlayer: this.redraw, │ │ │ │ │ + changelayer: this.redraw, │ │ │ │ │ + removelayer: this.redraw, │ │ │ │ │ + changebaselayer: this.redraw, │ │ │ │ │ scope: this │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getStyleNames: function(layer) { │ │ │ │ │ - var styleNames; │ │ │ │ │ - if (layer.params.STYLES) { │ │ │ │ │ - styleNames = layer.params.STYLES │ │ │ │ │ + }); │ │ │ │ │ + if (this.outsideViewport) { │ │ │ │ │ + this.events.attachToElement(this.div); │ │ │ │ │ + this.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ } else { │ │ │ │ │ - if (OpenLayers.Util.isArray(layer.params.LAYERS)) { │ │ │ │ │ - styleNames = new Array(layer.params.LAYERS.length) │ │ │ │ │ - } else { │ │ │ │ │ - styleNames = layer.params.LAYERS.replace(/[^,]/g, "") │ │ │ │ │ - } │ │ │ │ │ + this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ } │ │ │ │ │ - return styleNames │ │ │ │ │ }, │ │ │ │ │ - request: function(clickPosition, options) { │ │ │ │ │ - var layers = this.findLayers(); │ │ │ │ │ - if (layers.length == 0) { │ │ │ │ │ - this.events.triggerEvent("nogetfeatureinfo"); │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - return │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this); │ │ │ │ │ + this.loadContents(); │ │ │ │ │ + if (!this.outsideViewport) { │ │ │ │ │ + this.minimizeControl() │ │ │ │ │ } │ │ │ │ │ - options = options || {}; │ │ │ │ │ - if (this.drillDown === false) { │ │ │ │ │ - var wmsOptions = this.buildWMSOptions(this.url, layers, clickPosition, layers[0].params.FORMAT); │ │ │ │ │ - var request = OpenLayers.Request.GET(wmsOptions); │ │ │ │ │ - if (options.hover === true) { │ │ │ │ │ - this.hoverRequest = request │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return this.div │ │ │ │ │ + }, │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + var button = evt.buttonElement; │ │ │ │ │ + if (button === this.minimizeDiv) { │ │ │ │ │ + this.minimizeControl() │ │ │ │ │ + } else if (button === this.maximizeDiv) { │ │ │ │ │ + this.maximizeControl() │ │ │ │ │ + } else if (button._layerSwitcher === this.id) { │ │ │ │ │ + if (button["for"]) { │ │ │ │ │ + button = document.getElementById(button["for"]) │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - this._requestCount = 0; │ │ │ │ │ - this._numRequests = 0; │ │ │ │ │ - this.features = []; │ │ │ │ │ - var services = {}, │ │ │ │ │ - url; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - var layer = layers[i]; │ │ │ │ │ - var service, found = false; │ │ │ │ │ - url = OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url; │ │ │ │ │ - if (url in services) { │ │ │ │ │ - services[url].push(layer) │ │ │ │ │ + if (!button.disabled) { │ │ │ │ │ + if (button.type == "radio") { │ │ │ │ │ + button.checked = true; │ │ │ │ │ + this.map.setBaseLayer(this.map.getLayer(button._layer)) │ │ │ │ │ } else { │ │ │ │ │ - this._numRequests++; │ │ │ │ │ - services[url] = [layer] │ │ │ │ │ + button.checked = !button.checked; │ │ │ │ │ + this.updateMap() │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var layers; │ │ │ │ │ - for (var url in services) { │ │ │ │ │ - layers = services[url]; │ │ │ │ │ - var wmsOptions = this.buildWMSOptions(url, layers, clickPosition, layers[0].params.FORMAT); │ │ │ │ │ - OpenLayers.Request.GET(wmsOptions) │ │ │ │ │ - } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - triggerGetFeatureInfo: function(request, xy, features) { │ │ │ │ │ - this.events.triggerEvent("getfeatureinfo", { │ │ │ │ │ - text: request.responseText, │ │ │ │ │ - features: features, │ │ │ │ │ - request: request, │ │ │ │ │ - xy: xy │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait") │ │ │ │ │ + clearLayersArray: function(layersType) { │ │ │ │ │ + this[layersType + "LayersDiv"].innerHTML = ""; │ │ │ │ │ + this[layersType + "Layers"] = [] │ │ │ │ │ }, │ │ │ │ │ - handleResponse: function(xy, request, url) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ + checkRedraw: function() { │ │ │ │ │ + if (!this.layerStates.length || this.map.layers.length != this.layerStates.length) { │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ - var features = this.format.read(doc); │ │ │ │ │ - if (this.drillDown === false) { │ │ │ │ │ - this.triggerGetFeatureInfo(request, xy, features) │ │ │ │ │ - } else { │ │ │ │ │ - this._requestCount++; │ │ │ │ │ - if (this.output === "object") { │ │ │ │ │ - this._features = (this._features || []).concat({ │ │ │ │ │ - url: url, │ │ │ │ │ - features: features │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - this._features = (this._features || []).concat(features) │ │ │ │ │ - } │ │ │ │ │ - if (this._requestCount === this._numRequests) { │ │ │ │ │ - this.triggerGetFeatureInfo(request, xy, this._features.concat()); │ │ │ │ │ - delete this._features; │ │ │ │ │ - delete this._requestCount; │ │ │ │ │ - delete this._numRequests │ │ │ │ │ + for (var i = 0, len = this.layerStates.length; i < len; i++) { │ │ │ │ │ + var layerState = this.layerStates[i]; │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if (layerState.name != layer.name || layerState.inRange != layer.inRange || layerState.id != layer.id || layerState.visibility != layer.visibility) { │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.WMSGetFeatureInfo" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.EditingToolbar = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ - citeCompliant: false, │ │ │ │ │ - initialize: function(layer, options) { │ │ │ │ │ - OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.addControls([new OpenLayers.Control.Navigation]); │ │ │ │ │ - var controls = [new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Point, { │ │ │ │ │ - displayClass: "olControlDrawFeaturePoint", │ │ │ │ │ - handlerOptions: { │ │ │ │ │ - citeCompliant: this.citeCompliant │ │ │ │ │ - } │ │ │ │ │ - }), new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Path, { │ │ │ │ │ - displayClass: "olControlDrawFeaturePath", │ │ │ │ │ - handlerOptions: { │ │ │ │ │ - citeCompliant: this.citeCompliant │ │ │ │ │ - } │ │ │ │ │ - }), new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Polygon, { │ │ │ │ │ - displayClass: "olControlDrawFeaturePolygon", │ │ │ │ │ - handlerOptions: { │ │ │ │ │ - citeCompliant: this.citeCompliant │ │ │ │ │ + redraw: function() { │ │ │ │ │ + if (!this.checkRedraw()) { │ │ │ │ │ + return this.div │ │ │ │ │ + } │ │ │ │ │ + this.clearLayersArray("base"); │ │ │ │ │ + this.clearLayersArray("data"); │ │ │ │ │ + var containsOverlays = false; │ │ │ │ │ + var containsBaseLayers = false; │ │ │ │ │ + var len = this.map.layers.length; │ │ │ │ │ + this.layerStates = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + this.layerStates[i] = { │ │ │ │ │ + name: layer.name, │ │ │ │ │ + visibility: layer.visibility, │ │ │ │ │ + inRange: layer.inRange, │ │ │ │ │ + id: layer.id │ │ │ │ │ } │ │ │ │ │ - })]; │ │ │ │ │ - this.addControls(controls) │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - var div = OpenLayers.Control.Panel.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (this.defaultControl === null) { │ │ │ │ │ - this.defaultControl = this.controls[0] │ │ │ │ │ } │ │ │ │ │ - return div │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.EditingToolbar" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.Attribution = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - separator: ", ", │ │ │ │ │ - template: "${layers}", │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - removelayer: this.updateAttribution, │ │ │ │ │ - addlayer: this.updateAttribution, │ │ │ │ │ - changelayer: this.updateAttribution, │ │ │ │ │ - changebaselayer: this.updateAttribution, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - changebaselayer: this.updateAttribution, │ │ │ │ │ - changelayer: this.updateAttribution, │ │ │ │ │ - addlayer: this.updateAttribution, │ │ │ │ │ - removelayer: this.updateAttribution, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.updateAttribution(); │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - updateAttribution: function() { │ │ │ │ │ - var attributions = []; │ │ │ │ │ - if (this.map && this.map.layers) { │ │ │ │ │ - for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if (layer.attribution && layer.getVisibility()) { │ │ │ │ │ - if (OpenLayers.Util.indexOf(attributions, layer.attribution) === -1) { │ │ │ │ │ - attributions.push(layer.attribution) │ │ │ │ │ - } │ │ │ │ │ + var layers = this.map.layers.slice(); │ │ │ │ │ + if (!this.ascending) { │ │ │ │ │ + layers.reverse() │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + var layer = layers[i]; │ │ │ │ │ + var baseLayer = layer.isBaseLayer; │ │ │ │ │ + if (layer.displayInLayerSwitcher) { │ │ │ │ │ + if (baseLayer) { │ │ │ │ │ + containsBaseLayers = true │ │ │ │ │ + } else { │ │ │ │ │ + containsOverlays = true │ │ │ │ │ + } │ │ │ │ │ + var checked = baseLayer ? layer == this.map.baseLayer : layer.getVisibility(); │ │ │ │ │ + var inputElem = document.createElement("input"), │ │ │ │ │ + inputId = OpenLayers.Util.createUniqueID(this.id + "_input_"); │ │ │ │ │ + inputElem.id = inputId; │ │ │ │ │ + inputElem.name = baseLayer ? this.id + "_baseLayers" : layer.name; │ │ │ │ │ + inputElem.type = baseLayer ? "radio" : "checkbox"; │ │ │ │ │ + inputElem.value = layer.name; │ │ │ │ │ + inputElem.checked = checked; │ │ │ │ │ + inputElem.defaultChecked = checked; │ │ │ │ │ + inputElem.className = "olButton"; │ │ │ │ │ + inputElem._layer = layer.id; │ │ │ │ │ + inputElem._layerSwitcher = this.id; │ │ │ │ │ + if (!baseLayer && !layer.inRange) { │ │ │ │ │ + inputElem.disabled = true │ │ │ │ │ } │ │ │ │ │ + var labelSpan = document.createElement("label"); │ │ │ │ │ + labelSpan["for"] = inputElem.id; │ │ │ │ │ + OpenLayers.Element.addClass(labelSpan, "labelSpan olButton"); │ │ │ │ │ + labelSpan._layer = layer.id; │ │ │ │ │ + labelSpan._layerSwitcher = this.id; │ │ │ │ │ + if (!baseLayer && !layer.inRange) { │ │ │ │ │ + labelSpan.style.color = "gray" │ │ │ │ │ + } │ │ │ │ │ + labelSpan.innerHTML = layer.name; │ │ │ │ │ + labelSpan.style.verticalAlign = baseLayer ? "bottom" : "baseline"; │ │ │ │ │ + var br = document.createElement("br"); │ │ │ │ │ + var groupArray = baseLayer ? this.baseLayers : this.dataLayers; │ │ │ │ │ + groupArray.push({ │ │ │ │ │ + layer: layer, │ │ │ │ │ + inputElem: inputElem, │ │ │ │ │ + labelSpan: labelSpan │ │ │ │ │ + }); │ │ │ │ │ + var groupDiv = baseLayer ? this.baseLayersDiv : this.dataLayersDiv; │ │ │ │ │ + groupDiv.appendChild(inputElem); │ │ │ │ │ + groupDiv.appendChild(labelSpan); │ │ │ │ │ + groupDiv.appendChild(br) │ │ │ │ │ } │ │ │ │ │ - this.div.innerHTML = OpenLayers.String.format(this.template, { │ │ │ │ │ - layers: attributions.join(this.separator) │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Attribution" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.PanZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - slideFactor: 50, │ │ │ │ │ - slideRatio: null, │ │ │ │ │ - buttons: null, │ │ │ │ │ - position: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.position = new OpenLayers.Pixel(OpenLayers.Control.PanZoom.X, OpenLayers.Control.PanZoom.Y); │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onButtonClick) │ │ │ │ │ } │ │ │ │ │ - this.removeButtons(); │ │ │ │ │ - this.buttons = null; │ │ │ │ │ - this.position = null; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ - }, │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - px = this.position; │ │ │ │ │ - this.buttons = []; │ │ │ │ │ - var sz = { │ │ │ │ │ - w: 18, │ │ │ │ │ - h: 18 │ │ │ │ │ - }; │ │ │ │ │ - var centered = new OpenLayers.Pixel(px.x + sz.w / 2, px.y); │ │ │ │ │ - this._addButton("panup", "north-mini.png", centered, sz); │ │ │ │ │ - px.y = centered.y + sz.h; │ │ │ │ │ - this._addButton("panleft", "west-mini.png", px, sz); │ │ │ │ │ - this._addButton("panright", "east-mini.png", px.add(sz.w, 0), sz); │ │ │ │ │ - this._addButton("pandown", "south-mini.png", centered.add(0, sz.h * 2), sz); │ │ │ │ │ - this._addButton("zoomin", "zoom-plus-mini.png", centered.add(0, sz.h * 3 + 5), sz); │ │ │ │ │ - this._addButton("zoomworld", "zoom-world-mini.png", centered.add(0, sz.h * 4 + 5), sz); │ │ │ │ │ - this._addButton("zoomout", "zoom-minus-mini.png", centered.add(0, sz.h * 5 + 5), sz); │ │ │ │ │ + this.dataLbl.style.display = containsOverlays ? "" : "none"; │ │ │ │ │ + this.baseLbl.style.display = containsBaseLayers ? "" : "none"; │ │ │ │ │ return this.div │ │ │ │ │ }, │ │ │ │ │ - _addButton: function(id, img, xy, sz) { │ │ │ │ │ - var imgLocation = OpenLayers.Util.getImageLocation(img); │ │ │ │ │ - var btn = OpenLayers.Util.createAlphaImageDiv(this.id + "_" + id, xy, sz, imgLocation, "absolute"); │ │ │ │ │ - btn.style.cursor = "pointer"; │ │ │ │ │ - this.div.appendChild(btn); │ │ │ │ │ - btn.action = id; │ │ │ │ │ - btn.className = "olButton"; │ │ │ │ │ - this.buttons.push(btn); │ │ │ │ │ - return btn │ │ │ │ │ - }, │ │ │ │ │ - _removeButton: function(btn) { │ │ │ │ │ - this.div.removeChild(btn); │ │ │ │ │ - OpenLayers.Util.removeItem(this.buttons, btn) │ │ │ │ │ - }, │ │ │ │ │ - removeButtons: function() { │ │ │ │ │ - for (var i = this.buttons.length - 1; i >= 0; --i) { │ │ │ │ │ - this._removeButton(this.buttons[i]) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - var btn = evt.buttonElement; │ │ │ │ │ - switch (btn.action) { │ │ │ │ │ - case "panup": │ │ │ │ │ - this.map.pan(0, -this.getSlideFactor("h")); │ │ │ │ │ - break; │ │ │ │ │ - case "pandown": │ │ │ │ │ - this.map.pan(0, this.getSlideFactor("h")); │ │ │ │ │ - break; │ │ │ │ │ - case "panleft": │ │ │ │ │ - this.map.pan(-this.getSlideFactor("w"), 0); │ │ │ │ │ - break; │ │ │ │ │ - case "panright": │ │ │ │ │ - this.map.pan(this.getSlideFactor("w"), 0); │ │ │ │ │ - break; │ │ │ │ │ - case "zoomin": │ │ │ │ │ - this.map.zoomIn(); │ │ │ │ │ - break; │ │ │ │ │ - case "zoomout": │ │ │ │ │ - this.map.zoomOut(); │ │ │ │ │ - break; │ │ │ │ │ - case "zoomworld": │ │ │ │ │ - this.map.zoomToMaxExtent(); │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getSlideFactor: function(dim) { │ │ │ │ │ - return this.slideRatio ? this.map.getSize()[dim] * this.slideRatio : this.slideFactor │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.PanZoom" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.PanZoom.X = 4; │ │ │ │ │ -OpenLayers.Control.PanZoom.Y = 4; │ │ │ │ │ -OpenLayers.Control.Permalink = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - argParserClass: OpenLayers.Control.ArgParser, │ │ │ │ │ - element: null, │ │ │ │ │ - anchor: false, │ │ │ │ │ - base: "", │ │ │ │ │ - displayProjection: null, │ │ │ │ │ - initialize: function(element, base, options) { │ │ │ │ │ - if (element !== null && typeof element == "object" && !OpenLayers.Util.isElement(element)) { │ │ │ │ │ - options = element; │ │ │ │ │ - this.base = document.location.href; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (this.element != null) { │ │ │ │ │ - this.element = OpenLayers.Util.getElement(this.element) │ │ │ │ │ + updateMap: function() { │ │ │ │ │ + for (var i = 0, len = this.baseLayers.length; i < len; i++) { │ │ │ │ │ + var layerEntry = this.baseLayers[i]; │ │ │ │ │ + if (layerEntry.inputElem.checked) { │ │ │ │ │ + this.map.setBaseLayer(layerEntry.layer, false) │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.element = OpenLayers.Util.getElement(element); │ │ │ │ │ - this.base = base || document.location.href │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.element && this.element.parentNode == this.div) { │ │ │ │ │ - this.div.removeChild(this.element); │ │ │ │ │ - this.element = null │ │ │ │ │ } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("moveend", this, this.updateLink) │ │ │ │ │ + for (var i = 0, len = this.dataLayers.length; i < len; i++) { │ │ │ │ │ + var layerEntry = this.dataLayers[i]; │ │ │ │ │ + layerEntry.layer.setVisibility(layerEntry.inputElem.checked) │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - for (var i = 0, len = this.map.controls.length; i < len; i++) { │ │ │ │ │ - var control = this.map.controls[i]; │ │ │ │ │ - if (control.CLASS_NAME == this.argParserClass.CLASS_NAME) { │ │ │ │ │ - if (control.displayProjection != this.displayProjection) { │ │ │ │ │ - this.displayProjection = control.displayProjection │ │ │ │ │ - } │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (i == this.map.controls.length) { │ │ │ │ │ - this.map.addControl(new this.argParserClass({ │ │ │ │ │ - displayProjection: this.displayProjection │ │ │ │ │ - })) │ │ │ │ │ + maximizeControl: function(e) { │ │ │ │ │ + this.div.style.width = ""; │ │ │ │ │ + this.div.style.height = ""; │ │ │ │ │ + this.showControls(false); │ │ │ │ │ + if (e != null) { │ │ │ │ │ + OpenLayers.Event.stop(e) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (!this.element && !this.anchor) { │ │ │ │ │ - this.element = document.createElement("a"); │ │ │ │ │ - this.element.innerHTML = OpenLayers.i18n("Permalink"); │ │ │ │ │ - this.element.href = ""; │ │ │ │ │ - this.div.appendChild(this.element) │ │ │ │ │ + minimizeControl: function(e) { │ │ │ │ │ + this.div.style.width = "0px"; │ │ │ │ │ + this.div.style.height = "0px"; │ │ │ │ │ + this.showControls(true); │ │ │ │ │ + if (e != null) { │ │ │ │ │ + OpenLayers.Event.stop(e) │ │ │ │ │ } │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - moveend: this.updateLink, │ │ │ │ │ - changelayer: this.updateLink, │ │ │ │ │ - changebaselayer: this.updateLink, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.updateLink(); │ │ │ │ │ - return this.div │ │ │ │ │ }, │ │ │ │ │ - updateLink: function() { │ │ │ │ │ - var separator = this.anchor ? "#" : "?"; │ │ │ │ │ - var href = this.base; │ │ │ │ │ - var anchor = null; │ │ │ │ │ - if (href.indexOf("#") != -1 && this.anchor == false) { │ │ │ │ │ - anchor = href.substring(href.indexOf("#"), href.length) │ │ │ │ │ - } │ │ │ │ │ - if (href.indexOf(separator) != -1) { │ │ │ │ │ - href = href.substring(0, href.indexOf(separator)) │ │ │ │ │ - } │ │ │ │ │ - var splits = href.split("#"); │ │ │ │ │ - href = splits[0] + separator + OpenLayers.Util.getParameterString(this.createParams()); │ │ │ │ │ - if (anchor) { │ │ │ │ │ - href += anchor │ │ │ │ │ - } │ │ │ │ │ - if (this.anchor && !this.element) { │ │ │ │ │ - window.location.href = href │ │ │ │ │ - } else { │ │ │ │ │ - this.element.href = href │ │ │ │ │ - } │ │ │ │ │ + showControls: function(minimize) { │ │ │ │ │ + this.maximizeDiv.style.display = minimize ? "" : "none"; │ │ │ │ │ + this.minimizeDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ + this.layersDiv.style.display = minimize ? "none" : "" │ │ │ │ │ }, │ │ │ │ │ - createParams: function(center, zoom, layers) { │ │ │ │ │ - center = center || this.map.getCenter(); │ │ │ │ │ - var params = OpenLayers.Util.getParameters(this.base); │ │ │ │ │ - if (center) { │ │ │ │ │ - params.zoom = zoom || this.map.getZoom(); │ │ │ │ │ - var lat = center.lat; │ │ │ │ │ - var lon = center.lon; │ │ │ │ │ - if (this.displayProjection) { │ │ │ │ │ - var mapPosition = OpenLayers.Projection.transform({ │ │ │ │ │ - x: lon, │ │ │ │ │ - y: lat │ │ │ │ │ - }, this.map.getProjectionObject(), this.displayProjection); │ │ │ │ │ - lon = mapPosition.x; │ │ │ │ │ - lat = mapPosition.y │ │ │ │ │ - } │ │ │ │ │ - params.lat = Math.round(lat * 1e5) / 1e5; │ │ │ │ │ - params.lon = Math.round(lon * 1e5) / 1e5; │ │ │ │ │ - layers = layers || this.map.layers; │ │ │ │ │ - params.layers = ""; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - var layer = layers[i]; │ │ │ │ │ - if (layer.isBaseLayer) { │ │ │ │ │ - params.layers += layer == this.map.baseLayer ? "B" : "0" │ │ │ │ │ - } else { │ │ │ │ │ - params.layers += layer.getVisibility() ? "T" : "F" │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + loadContents: function() { │ │ │ │ │ + this.layersDiv = document.createElement("div"); │ │ │ │ │ + this.layersDiv.id = this.id + "_layersDiv"; │ │ │ │ │ + OpenLayers.Element.addClass(this.layersDiv, "layersDiv"); │ │ │ │ │ + this.baseLbl = document.createElement("div"); │ │ │ │ │ + this.baseLbl.innerHTML = OpenLayers.i18n("Base Layer"); │ │ │ │ │ + OpenLayers.Element.addClass(this.baseLbl, "baseLbl"); │ │ │ │ │ + this.baseLayersDiv = document.createElement("div"); │ │ │ │ │ + OpenLayers.Element.addClass(this.baseLayersDiv, "baseLayersDiv"); │ │ │ │ │ + this.dataLbl = document.createElement("div"); │ │ │ │ │ + this.dataLbl.innerHTML = OpenLayers.i18n("Overlays"); │ │ │ │ │ + OpenLayers.Element.addClass(this.dataLbl, "dataLbl"); │ │ │ │ │ + this.dataLayersDiv = document.createElement("div"); │ │ │ │ │ + OpenLayers.Element.addClass(this.dataLayersDiv, "dataLayersDiv"); │ │ │ │ │ + if (this.ascending) { │ │ │ │ │ + this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLayersDiv) │ │ │ │ │ + } else { │ │ │ │ │ + this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLayersDiv) │ │ │ │ │ } │ │ │ │ │ - return params │ │ │ │ │ + this.div.appendChild(this.layersDiv); │ │ │ │ │ + var img = OpenLayers.Util.getImageLocation("layer-switcher-maximize.png"); │ │ │ │ │ + this.maximizeDiv = OpenLayers.Util.createAlphaImageDiv("OpenLayers_Control_MaximizeDiv", null, null, img, "absolute"); │ │ │ │ │ + OpenLayers.Element.addClass(this.maximizeDiv, "maximizeDiv olButton"); │ │ │ │ │ + this.maximizeDiv.style.display = "none"; │ │ │ │ │ + this.div.appendChild(this.maximizeDiv); │ │ │ │ │ + var img = OpenLayers.Util.getImageLocation("layer-switcher-minimize.png"); │ │ │ │ │ + this.minimizeDiv = OpenLayers.Util.createAlphaImageDiv("OpenLayers_Control_MinimizeDiv", null, null, img, "absolute"); │ │ │ │ │ + OpenLayers.Element.addClass(this.minimizeDiv, "minimizeDiv olButton"); │ │ │ │ │ + this.minimizeDiv.style.display = "none"; │ │ │ │ │ + this.div.appendChild(this.minimizeDiv) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Permalink" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.LayerSwitcher" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Control.Split = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ layer: null, │ │ │ │ │ source: null, │ │ │ │ │ sourceOptions: null, │ │ │ │ │ tolerance: null, │ │ │ │ │ edge: true, │ │ │ │ │ @@ -32300,230 +31811,14 @@ │ │ │ │ │ if (this.active) { │ │ │ │ │ this.deactivate() │ │ │ │ │ } │ │ │ │ │ OpenLayers.Control.prototype.destroy.call(this) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.Split" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.PanZoomBar = OpenLayers.Class(OpenLayers.Control.PanZoom, { │ │ │ │ │ - zoomStopWidth: 18, │ │ │ │ │ - zoomStopHeight: 11, │ │ │ │ │ - slider: null, │ │ │ │ │ - sliderEvents: null, │ │ │ │ │ - zoombarDiv: null, │ │ │ │ │ - zoomWorldIcon: false, │ │ │ │ │ - panIcons: true, │ │ │ │ │ - forceFixedZoomLevel: false, │ │ │ │ │ - mouseDragStart: null, │ │ │ │ │ - deltaY: null, │ │ │ │ │ - zoomStart: null, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this._removeZoomBar(); │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - changebaselayer: this.redraw, │ │ │ │ │ - updatesize: this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Control.PanZoom.prototype.destroy.apply(this, arguments); │ │ │ │ │ - delete this.mouseDragStart; │ │ │ │ │ - delete this.zoomStart │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.PanZoom.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - changebaselayer: this.redraw, │ │ │ │ │ - updatesize: this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - redraw: function() { │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.removeButtons(); │ │ │ │ │ - this._removeZoomBar() │ │ │ │ │ - } │ │ │ │ │ - this.draw() │ │ │ │ │ - }, │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - px = this.position.clone(); │ │ │ │ │ - this.buttons = []; │ │ │ │ │ - var sz = { │ │ │ │ │ - w: 18, │ │ │ │ │ - h: 18 │ │ │ │ │ - }; │ │ │ │ │ - if (this.panIcons) { │ │ │ │ │ - var centered = new OpenLayers.Pixel(px.x + sz.w / 2, px.y); │ │ │ │ │ - var wposition = sz.w; │ │ │ │ │ - if (this.zoomWorldIcon) { │ │ │ │ │ - centered = new OpenLayers.Pixel(px.x + sz.w, px.y) │ │ │ │ │ - } │ │ │ │ │ - this._addButton("panup", "north-mini.png", centered, sz); │ │ │ │ │ - px.y = centered.y + sz.h; │ │ │ │ │ - this._addButton("panleft", "west-mini.png", px, sz); │ │ │ │ │ - if (this.zoomWorldIcon) { │ │ │ │ │ - this._addButton("zoomworld", "zoom-world-mini.png", px.add(sz.w, 0), sz); │ │ │ │ │ - wposition *= 2 │ │ │ │ │ - } │ │ │ │ │ - this._addButton("panright", "east-mini.png", px.add(wposition, 0), sz); │ │ │ │ │ - this._addButton("pandown", "south-mini.png", centered.add(0, sz.h * 2), sz); │ │ │ │ │ - this._addButton("zoomin", "zoom-plus-mini.png", centered.add(0, sz.h * 3 + 5), sz); │ │ │ │ │ - centered = this._addZoomBar(centered.add(0, sz.h * 4 + 5)); │ │ │ │ │ - this._addButton("zoomout", "zoom-minus-mini.png", centered, sz) │ │ │ │ │ - } else { │ │ │ │ │ - this._addButton("zoomin", "zoom-plus-mini.png", px, sz); │ │ │ │ │ - centered = this._addZoomBar(px.add(0, sz.h)); │ │ │ │ │ - this._addButton("zoomout", "zoom-minus-mini.png", centered, sz); │ │ │ │ │ - if (this.zoomWorldIcon) { │ │ │ │ │ - centered = centered.add(0, sz.h + 3); │ │ │ │ │ - this._addButton("zoomworld", "zoom-world-mini.png", centered, sz) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - _addZoomBar: function(centered) { │ │ │ │ │ - var imgLocation = OpenLayers.Util.getImageLocation("slider.png"); │ │ │ │ │ - var id = this.id + "_" + this.map.id; │ │ │ │ │ - var minZoom = this.map.getMinZoom(); │ │ │ │ │ - var zoomsToEnd = this.map.getNumZoomLevels() - 1 - this.map.getZoom(); │ │ │ │ │ - var slider = OpenLayers.Util.createAlphaImageDiv(id, centered.add(-1, zoomsToEnd * this.zoomStopHeight), { │ │ │ │ │ - w: 20, │ │ │ │ │ - h: 9 │ │ │ │ │ - }, imgLocation, "absolute"); │ │ │ │ │ - slider.style.cursor = "move"; │ │ │ │ │ - this.slider = slider; │ │ │ │ │ - this.sliderEvents = new OpenLayers.Events(this, slider, null, true, { │ │ │ │ │ - includeXY: true │ │ │ │ │ - }); │ │ │ │ │ - this.sliderEvents.on({ │ │ │ │ │ - touchstart: this.zoomBarDown, │ │ │ │ │ - touchmove: this.zoomBarDrag, │ │ │ │ │ - touchend: this.zoomBarUp, │ │ │ │ │ - mousedown: this.zoomBarDown, │ │ │ │ │ - mousemove: this.zoomBarDrag, │ │ │ │ │ - mouseup: this.zoomBarUp │ │ │ │ │ - }); │ │ │ │ │ - var sz = { │ │ │ │ │ - w: this.zoomStopWidth, │ │ │ │ │ - h: this.zoomStopHeight * (this.map.getNumZoomLevels() - minZoom) │ │ │ │ │ - }; │ │ │ │ │ - var imgLocation = OpenLayers.Util.getImageLocation("zoombar.png"); │ │ │ │ │ - var div = null; │ │ │ │ │ - if (OpenLayers.Util.alphaHack()) { │ │ │ │ │ - var id = this.id + "_" + this.map.id; │ │ │ │ │ - div = OpenLayers.Util.createAlphaImageDiv(id, centered, { │ │ │ │ │ - w: sz.w, │ │ │ │ │ - h: this.zoomStopHeight │ │ │ │ │ - }, imgLocation, "absolute", null, "crop"); │ │ │ │ │ - div.style.height = sz.h + "px" │ │ │ │ │ - } else { │ │ │ │ │ - div = OpenLayers.Util.createDiv("OpenLayers_Control_PanZoomBar_Zoombar" + this.map.id, centered, sz, imgLocation) │ │ │ │ │ - } │ │ │ │ │ - div.style.cursor = "pointer"; │ │ │ │ │ - div.className = "olButton"; │ │ │ │ │ - this.zoombarDiv = div; │ │ │ │ │ - this.div.appendChild(div); │ │ │ │ │ - this.startTop = parseInt(div.style.top); │ │ │ │ │ - this.div.appendChild(slider); │ │ │ │ │ - this.map.events.register("zoomend", this, this.moveZoomBar); │ │ │ │ │ - centered = centered.add(0, this.zoomStopHeight * (this.map.getNumZoomLevels() - minZoom)); │ │ │ │ │ - return centered │ │ │ │ │ - }, │ │ │ │ │ - _removeZoomBar: function() { │ │ │ │ │ - this.sliderEvents.un({ │ │ │ │ │ - touchstart: this.zoomBarDown, │ │ │ │ │ - touchmove: this.zoomBarDrag, │ │ │ │ │ - touchend: this.zoomBarUp, │ │ │ │ │ - mousedown: this.zoomBarDown, │ │ │ │ │ - mousemove: this.zoomBarDrag, │ │ │ │ │ - mouseup: this.zoomBarUp │ │ │ │ │ - }); │ │ │ │ │ - this.sliderEvents.destroy(); │ │ │ │ │ - this.div.removeChild(this.zoombarDiv); │ │ │ │ │ - this.zoombarDiv = null; │ │ │ │ │ - this.div.removeChild(this.slider); │ │ │ │ │ - this.slider = null; │ │ │ │ │ - this.map.events.unregister("zoomend", this, this.moveZoomBar) │ │ │ │ │ - }, │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - OpenLayers.Control.PanZoom.prototype.onButtonClick.apply(this, arguments); │ │ │ │ │ - if (evt.buttonElement === this.zoombarDiv) { │ │ │ │ │ - var levels = evt.buttonXY.y / this.zoomStopHeight; │ │ │ │ │ - if (this.forceFixedZoomLevel || !this.map.fractionalZoom) { │ │ │ │ │ - levels = Math.floor(levels) │ │ │ │ │ - } │ │ │ │ │ - var zoom = this.map.getNumZoomLevels() - 1 - levels; │ │ │ │ │ - zoom = Math.min(Math.max(zoom, 0), this.map.getNumZoomLevels() - 1); │ │ │ │ │ - this.map.zoomTo(zoom) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - passEventToSlider: function(evt) { │ │ │ │ │ - this.sliderEvents.handleBrowserEvent(evt) │ │ │ │ │ - }, │ │ │ │ │ - zoomBarDown: function(evt) { │ │ │ │ │ - if (!OpenLayers.Event.isLeftClick(evt) && !OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - touchmove: this.passEventToSlider, │ │ │ │ │ - mousemove: this.passEventToSlider, │ │ │ │ │ - mouseup: this.passEventToSlider, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.mouseDragStart = evt.xy.clone(); │ │ │ │ │ - this.zoomStart = evt.xy.clone(); │ │ │ │ │ - this.div.style.cursor = "move"; │ │ │ │ │ - this.zoombarDiv.offsets = null; │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ - }, │ │ │ │ │ - zoomBarDrag: function(evt) { │ │ │ │ │ - if (this.mouseDragStart != null) { │ │ │ │ │ - var deltaY = this.mouseDragStart.y - evt.xy.y; │ │ │ │ │ - var offsets = OpenLayers.Util.pagePosition(this.zoombarDiv); │ │ │ │ │ - if (evt.clientY - offsets[1] > 0 && evt.clientY - offsets[1] < parseInt(this.zoombarDiv.style.height) - 2) { │ │ │ │ │ - var newTop = parseInt(this.slider.style.top) - deltaY; │ │ │ │ │ - this.slider.style.top = newTop + "px"; │ │ │ │ │ - this.mouseDragStart = evt.xy.clone() │ │ │ │ │ - } │ │ │ │ │ - this.deltaY = this.zoomStart.y - evt.xy.y; │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - zoomBarUp: function(evt) { │ │ │ │ │ - if (!OpenLayers.Event.isLeftClick(evt) && evt.type !== "touchend") { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (this.mouseDragStart) { │ │ │ │ │ - this.div.style.cursor = ""; │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - touchmove: this.passEventToSlider, │ │ │ │ │ - mouseup: this.passEventToSlider, │ │ │ │ │ - mousemove: this.passEventToSlider, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - var zoomLevel = this.map.zoom; │ │ │ │ │ - if (!this.forceFixedZoomLevel && this.map.fractionalZoom) { │ │ │ │ │ - zoomLevel += this.deltaY / this.zoomStopHeight; │ │ │ │ │ - zoomLevel = Math.min(Math.max(zoomLevel, 0), this.map.getNumZoomLevels() - 1) │ │ │ │ │ - } else { │ │ │ │ │ - zoomLevel += this.deltaY / this.zoomStopHeight; │ │ │ │ │ - zoomLevel = Math.max(Math.round(zoomLevel), 0) │ │ │ │ │ - } │ │ │ │ │ - this.map.zoomTo(zoomLevel); │ │ │ │ │ - this.mouseDragStart = null; │ │ │ │ │ - this.zoomStart = null; │ │ │ │ │ - this.deltaY = 0; │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - moveZoomBar: function() { │ │ │ │ │ - var newTop = (this.map.getNumZoomLevels() - 1 - this.map.getZoom()) * this.zoomStopHeight + this.startTop + 1; │ │ │ │ │ - this.slider.style.top = newTop + "px" │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.PanZoomBar" │ │ │ │ │ -}); │ │ │ │ │ OpenLayers.Control.DragFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ geometryTypes: null, │ │ │ │ │ onStart: function(feature, pixel) {}, │ │ │ │ │ onDrag: function(feature, pixel) {}, │ │ │ │ │ onComplete: function(feature, pixel) {}, │ │ │ │ │ onEnter: function(feature) {}, │ │ │ │ │ onLeave: function(feature) {}, │ │ │ │ │ @@ -32641,105 +31936,14 @@ │ │ │ │ │ setMap: function(map) { │ │ │ │ │ this.handlers.drag.setMap(map); │ │ │ │ │ this.handlers.feature.setMap(map); │ │ │ │ │ OpenLayers.Control.prototype.setMap.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.DragFeature" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.ScaleLine = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - maxWidth: 100, │ │ │ │ │ - topOutUnits: "km", │ │ │ │ │ - topInUnits: "m", │ │ │ │ │ - bottomOutUnits: "mi", │ │ │ │ │ - bottomInUnits: "ft", │ │ │ │ │ - eTop: null, │ │ │ │ │ - eBottom: null, │ │ │ │ │ - geodesic: false, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (!this.eTop) { │ │ │ │ │ - this.eTop = document.createElement("div"); │ │ │ │ │ - this.eTop.className = this.displayClass + "Top"; │ │ │ │ │ - var theLen = this.topInUnits.length; │ │ │ │ │ - this.div.appendChild(this.eTop); │ │ │ │ │ - if (this.topOutUnits == "" || this.topInUnits == "") { │ │ │ │ │ - this.eTop.style.visibility = "hidden" │ │ │ │ │ - } else { │ │ │ │ │ - this.eTop.style.visibility = "visible" │ │ │ │ │ - } │ │ │ │ │ - this.eBottom = document.createElement("div"); │ │ │ │ │ - this.eBottom.className = this.displayClass + "Bottom"; │ │ │ │ │ - this.div.appendChild(this.eBottom); │ │ │ │ │ - if (this.bottomOutUnits == "" || this.bottomInUnits == "") { │ │ │ │ │ - this.eBottom.style.visibility = "hidden" │ │ │ │ │ - } else { │ │ │ │ │ - this.eBottom.style.visibility = "visible" │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.map.events.register("moveend", this, this.update); │ │ │ │ │ - this.update(); │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - getBarLen: function(maxLen) { │ │ │ │ │ - var digits = parseInt(Math.log(maxLen) / Math.log(10)); │ │ │ │ │ - var pow10 = Math.pow(10, digits); │ │ │ │ │ - var firstChar = parseInt(maxLen / pow10); │ │ │ │ │ - var barLen; │ │ │ │ │ - if (firstChar > 5) { │ │ │ │ │ - barLen = 5 │ │ │ │ │ - } else if (firstChar > 2) { │ │ │ │ │ - barLen = 2 │ │ │ │ │ - } else { │ │ │ │ │ - barLen = 1 │ │ │ │ │ - } │ │ │ │ │ - return barLen * pow10 │ │ │ │ │ - }, │ │ │ │ │ - update: function() { │ │ │ │ │ - var res = this.map.getResolution(); │ │ │ │ │ - if (!res) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var curMapUnits = this.map.getUnits(); │ │ │ │ │ - var inches = OpenLayers.INCHES_PER_UNIT; │ │ │ │ │ - var maxSizeData = this.maxWidth * res * inches[curMapUnits]; │ │ │ │ │ - var geodesicRatio = 1; │ │ │ │ │ - if (this.geodesic === true) { │ │ │ │ │ - var maxSizeGeodesic = (this.map.getGeodesicPixelSize().w || 1e-6) * this.maxWidth; │ │ │ │ │ - var maxSizeKilometers = maxSizeData / inches["km"]; │ │ │ │ │ - geodesicRatio = maxSizeGeodesic / maxSizeKilometers; │ │ │ │ │ - maxSizeData *= geodesicRatio │ │ │ │ │ - } │ │ │ │ │ - var topUnits; │ │ │ │ │ - var bottomUnits; │ │ │ │ │ - if (maxSizeData > 1e5) { │ │ │ │ │ - topUnits = this.topOutUnits; │ │ │ │ │ - bottomUnits = this.bottomOutUnits │ │ │ │ │ - } else { │ │ │ │ │ - topUnits = this.topInUnits; │ │ │ │ │ - bottomUnits = this.bottomInUnits │ │ │ │ │ - } │ │ │ │ │ - var topMax = maxSizeData / inches[topUnits]; │ │ │ │ │ - var bottomMax = maxSizeData / inches[bottomUnits]; │ │ │ │ │ - var topRounded = this.getBarLen(topMax); │ │ │ │ │ - var bottomRounded = this.getBarLen(bottomMax); │ │ │ │ │ - topMax = topRounded / inches[curMapUnits] * inches[topUnits]; │ │ │ │ │ - bottomMax = bottomRounded / inches[curMapUnits] * inches[bottomUnits]; │ │ │ │ │ - var topPx = topMax / res / geodesicRatio; │ │ │ │ │ - var bottomPx = bottomMax / res / geodesicRatio; │ │ │ │ │ - if (this.eBottom.style.visibility == "visible") { │ │ │ │ │ - this.eBottom.style.width = Math.round(bottomPx) + "px"; │ │ │ │ │ - this.eBottom.innerHTML = bottomRounded + " " + bottomUnits │ │ │ │ │ - } │ │ │ │ │ - if (this.eTop.style.visibility == "visible") { │ │ │ │ │ - this.eTop.style.width = Math.round(topPx) + "px"; │ │ │ │ │ - this.eTop.innerHTML = topRounded + " " + topUnits │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ScaleLine" │ │ │ │ │ -}); │ │ │ │ │ OpenLayers.Control.TransformFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ geometryTypes: null, │ │ │ │ │ layer: null, │ │ │ │ │ preserveAspectRatio: false, │ │ │ │ │ rotate: true, │ │ │ │ │ feature: null, │ │ │ │ │ renderIntent: "temporary", │ │ │ │ │ @@ -33064,312 +32268,623 @@ │ │ │ │ │ this.layer = null; │ │ │ │ │ this.dragControl.destroy(); │ │ │ │ │ this.dragControl = null; │ │ │ │ │ OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.TransformFeature" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.KeyboardDefaults = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - slideFactor: 75, │ │ │ │ │ - observeElement: null, │ │ │ │ │ - draw: function() { │ │ │ │ │ - var observeElement = this.observeElement || document; │ │ │ │ │ - this.handler = new OpenLayers.Handler.Keyboard(this, { │ │ │ │ │ - keydown: this.defaultKeyPress │ │ │ │ │ - }, { │ │ │ │ │ - observeElement: observeElement │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - defaultKeyPress: function(evt) { │ │ │ │ │ - var size, handled = true; │ │ │ │ │ - var target = OpenLayers.Event.element(evt); │ │ │ │ │ - if (target && (target.tagName == "INPUT" || target.tagName == "TEXTAREA" || target.tagName == "SELECT")) { │ │ │ │ │ - return │ │ │ │ │ +OpenLayers.Control.GetFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + protocol: null, │ │ │ │ │ + multipleKey: null, │ │ │ │ │ + toggleKey: null, │ │ │ │ │ + modifiers: null, │ │ │ │ │ + multiple: false, │ │ │ │ │ + click: true, │ │ │ │ │ + single: true, │ │ │ │ │ + clickout: true, │ │ │ │ │ + toggle: false, │ │ │ │ │ + clickTolerance: 5, │ │ │ │ │ + hover: false, │ │ │ │ │ + box: false, │ │ │ │ │ + maxFeatures: 10, │ │ │ │ │ + features: null, │ │ │ │ │ + hoverFeature: null, │ │ │ │ │ + handlers: null, │ │ │ │ │ + hoverResponse: null, │ │ │ │ │ + filterType: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.features = {}; │ │ │ │ │ + this.handlers = {}; │ │ │ │ │ + if (this.click) { │ │ │ │ │ + this.handlers.click = new OpenLayers.Handler.Click(this, { │ │ │ │ │ + click: this.selectClick │ │ │ │ │ + }, this.handlerOptions.click || {}) │ │ │ │ │ } │ │ │ │ │ - switch (evt.keyCode) { │ │ │ │ │ - case OpenLayers.Event.KEY_LEFT: │ │ │ │ │ - this.map.pan(-this.slideFactor, 0); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Event.KEY_RIGHT: │ │ │ │ │ - this.map.pan(this.slideFactor, 0); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Event.KEY_UP: │ │ │ │ │ - this.map.pan(0, -this.slideFactor); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Event.KEY_DOWN: │ │ │ │ │ - this.map.pan(0, this.slideFactor); │ │ │ │ │ - break; │ │ │ │ │ - case 33: │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - this.map.pan(0, -.75 * size.h); │ │ │ │ │ - break; │ │ │ │ │ - case 34: │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - this.map.pan(0, .75 * size.h); │ │ │ │ │ - break; │ │ │ │ │ - case 35: │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - this.map.pan(.75 * size.w, 0); │ │ │ │ │ - break; │ │ │ │ │ - case 36: │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - this.map.pan(-.75 * size.w, 0); │ │ │ │ │ - break; │ │ │ │ │ - case 43: │ │ │ │ │ - case 61: │ │ │ │ │ - case 187: │ │ │ │ │ - case 107: │ │ │ │ │ - this.map.zoomIn(); │ │ │ │ │ - break; │ │ │ │ │ - case 45: │ │ │ │ │ - case 109: │ │ │ │ │ - case 189: │ │ │ │ │ - case 95: │ │ │ │ │ - this.map.zoomOut(); │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - handled = false │ │ │ │ │ + if (this.box) { │ │ │ │ │ + this.handlers.box = new OpenLayers.Handler.Box(this, { │ │ │ │ │ + done: this.selectBox │ │ │ │ │ + }, OpenLayers.Util.extend(this.handlerOptions.box, { │ │ │ │ │ + boxDivClassName: "olHandlerBoxSelectFeature" │ │ │ │ │ + })) │ │ │ │ │ } │ │ │ │ │ - if (handled) { │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + this.handlers.hover = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ + move: this.cancelHover, │ │ │ │ │ + pause: this.selectHover │ │ │ │ │ + }, OpenLayers.Util.extend(this.handlerOptions.hover, { │ │ │ │ │ + delay: 250, │ │ │ │ │ + pixelTolerance: 2 │ │ │ │ │ + })) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.KeyboardDefaults" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.UTFGrid = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - layers: null, │ │ │ │ │ - defaultHandlerOptions: { │ │ │ │ │ - delay: 300, │ │ │ │ │ - pixelTolerance: 4, │ │ │ │ │ - stopMove: false, │ │ │ │ │ - single: true, │ │ │ │ │ - double: false, │ │ │ │ │ - stopSingle: false, │ │ │ │ │ - stopDouble: false │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + for (var i in this.handlers) { │ │ │ │ │ + this.handlers[i].activate() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Control.prototype.activate.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - handlerMode: "click", │ │ │ │ │ - setHandler: function(hm) { │ │ │ │ │ - this.handlerMode = hm; │ │ │ │ │ - this.resetHandler() │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + for (var i in this.handlers) { │ │ │ │ │ + this.handlers[i].deactivate() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Control.prototype.deactivate.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - resetHandler: function() { │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.deactivate(); │ │ │ │ │ - this.handler.destroy(); │ │ │ │ │ - this.handler = null │ │ │ │ │ + selectClick: function(evt) { │ │ │ │ │ + var bounds = this.pixelToBounds(evt.xy); │ │ │ │ │ + this.setModifiers(evt); │ │ │ │ │ + this.request(bounds, { │ │ │ │ │ + single: this.single │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + selectBox: function(position) { │ │ │ │ │ + var bounds; │ │ │ │ │ + if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ + var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.left, │ │ │ │ │ + y: position.bottom │ │ │ │ │ + }); │ │ │ │ │ + var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.right, │ │ │ │ │ + y: position.top │ │ │ │ │ + }); │ │ │ │ │ + bounds = new OpenLayers.Bounds(minXY.lon, minXY.lat, maxXY.lon, maxXY.lat) │ │ │ │ │ + } else { │ │ │ │ │ + if (this.click) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + bounds = this.pixelToBounds(position) │ │ │ │ │ } │ │ │ │ │ - if (this.handlerMode == "hover") { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ - pause: this.handleEvent, │ │ │ │ │ - move: this.reset │ │ │ │ │ - }, this.handlerOptions) │ │ │ │ │ - } else if (this.handlerMode == "click") { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Click(this, { │ │ │ │ │ - click: this.handleEvent │ │ │ │ │ - }, this.handlerOptions) │ │ │ │ │ - } else if (this.handlerMode == "move") { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ - pause: this.handleEvent, │ │ │ │ │ - move: this.handleEvent │ │ │ │ │ - }, this.handlerOptions) │ │ │ │ │ + this.setModifiers(this.handlers.box.dragHandler.evt); │ │ │ │ │ + this.request(bounds) │ │ │ │ │ + }, │ │ │ │ │ + selectHover: function(evt) { │ │ │ │ │ + var bounds = this.pixelToBounds(evt.xy); │ │ │ │ │ + this.request(bounds, { │ │ │ │ │ + single: true, │ │ │ │ │ + hover: true │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + cancelHover: function() { │ │ │ │ │ + if (this.hoverResponse) { │ │ │ │ │ + this.protocol.abort(this.hoverResponse); │ │ │ │ │ + this.hoverResponse = null; │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait") │ │ │ │ │ } │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + }, │ │ │ │ │ + request: function(bounds, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: this.filterType, │ │ │ │ │ + value: bounds │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + var response = this.protocol.read({ │ │ │ │ │ + maxFeatures: options.single == true ? this.maxFeatures : undefined, │ │ │ │ │ + filter: filter, │ │ │ │ │ + callback: function(result) { │ │ │ │ │ + if (result.success()) { │ │ │ │ │ + if (result.features.length) { │ │ │ │ │ + if (options.single == true) { │ │ │ │ │ + this.selectBestFeature(result.features, bounds.getCenterLonLat(), options) │ │ │ │ │ + } else { │ │ │ │ │ + this.select(result.features) │ │ │ │ │ + } │ │ │ │ │ + } else if (options.hover) { │ │ │ │ │ + this.hoverSelect() │ │ │ │ │ + } else { │ │ │ │ │ + this.events.triggerEvent("clickout"); │ │ │ │ │ + if (this.clickout) { │ │ │ │ │ + this.unselectAll() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait") │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + if (options.hover == true) { │ │ │ │ │ + this.hoverResponse = response │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ + selectBestFeature: function(features, clickPosition, options) { │ │ │ │ │ options = options || {}; │ │ │ │ │ - options.handlerOptions = options.handlerOptions || this.defaultHandlerOptions; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.resetHandler() │ │ │ │ │ + if (features.length) { │ │ │ │ │ + var point = new OpenLayers.Geometry.Point(clickPosition.lon, clickPosition.lat); │ │ │ │ │ + var feature, resultFeature, dist; │ │ │ │ │ + var minDist = Number.MAX_VALUE; │ │ │ │ │ + for (var i = 0; i < features.length; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + dist = point.distanceTo(feature.geometry, { │ │ │ │ │ + edge: false │ │ │ │ │ + }); │ │ │ │ │ + if (dist < minDist) { │ │ │ │ │ + minDist = dist; │ │ │ │ │ + resultFeature = feature; │ │ │ │ │ + if (minDist == 0) { │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (options.hover == true) { │ │ │ │ │ + this.hoverSelect(resultFeature) │ │ │ │ │ + } else { │ │ │ │ │ + this.select(resultFeature || features) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - handleEvent: function(evt) { │ │ │ │ │ - if (evt == null) { │ │ │ │ │ - this.reset(); │ │ │ │ │ - return │ │ │ │ │ + setModifiers: function(evt) { │ │ │ │ │ + this.modifiers = { │ │ │ │ │ + multiple: this.multiple || this.multipleKey && evt[this.multipleKey], │ │ │ │ │ + toggle: this.toggle || this.toggleKey && evt[this.toggleKey] │ │ │ │ │ } │ │ │ │ │ - var lonLat = this.map.getLonLatFromPixel(evt.xy); │ │ │ │ │ - if (!lonLat) { │ │ │ │ │ - return │ │ │ │ │ + }, │ │ │ │ │ + select: function(features) { │ │ │ │ │ + if (!this.modifiers.multiple && !this.modifiers.toggle) { │ │ │ │ │ + this.unselectAll() │ │ │ │ │ } │ │ │ │ │ - var layers = this.findLayers(); │ │ │ │ │ - if (layers.length > 0) { │ │ │ │ │ - var infoLookup = {}; │ │ │ │ │ - var layer, idx; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - layer = layers[i]; │ │ │ │ │ - idx = OpenLayers.Util.indexOf(this.map.layers, layer); │ │ │ │ │ - infoLookup[idx] = layer.getFeatureInfo(lonLat) │ │ │ │ │ + if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ + features = [features] │ │ │ │ │ + } │ │ │ │ │ + var cont = this.events.triggerEvent("beforefeaturesselected", { │ │ │ │ │ + features: features │ │ │ │ │ + }); │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + var selectedFeatures = []; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (this.features[feature.fid || feature.id]) { │ │ │ │ │ + if (this.modifiers.toggle) { │ │ │ │ │ + this.unselect(this.features[feature.fid || feature.id]) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + cont = this.events.triggerEvent("beforefeatureselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + this.features[feature.fid || feature.id] = feature; │ │ │ │ │ + selectedFeatures.push(feature); │ │ │ │ │ + this.events.triggerEvent("featureselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.callback(infoLookup, lonLat, evt.xy) │ │ │ │ │ + this.events.triggerEvent("featuresselected", { │ │ │ │ │ + features: selectedFeatures │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - callback: function(infoLookup) {}, │ │ │ │ │ - reset: function(evt) { │ │ │ │ │ - this.callback(null) │ │ │ │ │ + hoverSelect: function(feature) { │ │ │ │ │ + var fid = feature ? feature.fid || feature.id : null; │ │ │ │ │ + var hfid = this.hoverFeature ? this.hoverFeature.fid || this.hoverFeature.id : null; │ │ │ │ │ + if (hfid && hfid != fid) { │ │ │ │ │ + this.events.triggerEvent("outfeature", { │ │ │ │ │ + feature: this.hoverFeature │ │ │ │ │ + }); │ │ │ │ │ + this.hoverFeature = null │ │ │ │ │ + } │ │ │ │ │ + if (fid && fid != hfid) { │ │ │ │ │ + this.events.triggerEvent("hoverfeature", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.hoverFeature = feature │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - findLayers: function() { │ │ │ │ │ - var candidates = this.layers || this.map.layers; │ │ │ │ │ - var layers = []; │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = candidates[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.UTFGrid) { │ │ │ │ │ - layers.push(layer) │ │ │ │ │ - } │ │ │ │ │ + unselect: function(feature) { │ │ │ │ │ + delete this.features[feature.fid || feature.id]; │ │ │ │ │ + this.events.triggerEvent("featureunselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + unselectAll: function() { │ │ │ │ │ + for (var fid in this.features) { │ │ │ │ │ + this.unselect(this.features[fid]) │ │ │ │ │ } │ │ │ │ │ - return layers │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.UTFGrid" │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + for (var i in this.handlers) { │ │ │ │ │ + this.handlers[i].setMap(map) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + pixelToBounds: function(pixel) { │ │ │ │ │ + var llPx = pixel.add(-this.clickTolerance / 2, this.clickTolerance / 2); │ │ │ │ │ + var urPx = pixel.add(this.clickTolerance / 2, -this.clickTolerance / 2); │ │ │ │ │ + var ll = this.map.getLonLatFromPixel(llPx); │ │ │ │ │ + var ur = this.map.getLonLatFromPixel(urPx); │ │ │ │ │ + return new OpenLayers.Bounds(ll.lon, ll.lat, ur.lon, ur.lat) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.GetFeature" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.MousePosition = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - element: null, │ │ │ │ │ - prefix: "", │ │ │ │ │ - separator: ", ", │ │ │ │ │ - suffix: "", │ │ │ │ │ - numDigits: 5, │ │ │ │ │ - granularity: 10, │ │ │ │ │ - emptyString: null, │ │ │ │ │ - lastXy: null, │ │ │ │ │ - displayProjection: null, │ │ │ │ │ +OpenLayers.Control.PanZoomBar = OpenLayers.Class(OpenLayers.Control.PanZoom, { │ │ │ │ │ + zoomStopWidth: 18, │ │ │ │ │ + zoomStopHeight: 11, │ │ │ │ │ + slider: null, │ │ │ │ │ + sliderEvents: null, │ │ │ │ │ + zoombarDiv: null, │ │ │ │ │ + zoomWorldIcon: false, │ │ │ │ │ + panIcons: true, │ │ │ │ │ + forceFixedZoomLevel: false, │ │ │ │ │ + mouseDragStart: null, │ │ │ │ │ + deltaY: null, │ │ │ │ │ + zoomStart: null, │ │ │ │ │ destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ + this._removeZoomBar(); │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + changebaselayer: this.redraw, │ │ │ │ │ + updatesize: this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Control.PanZoom.prototype.destroy.apply(this, arguments); │ │ │ │ │ + delete this.mouseDragStart; │ │ │ │ │ + delete this.zoomStart │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.map.events.register("mousemove", this, this.redraw); │ │ │ │ │ - this.map.events.register("mouseout", this, this.reset); │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.PanZoom.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + changebaselayer: this.redraw, │ │ │ │ │ + updatesize: this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.map.events.unregister("mousemove", this, this.redraw); │ │ │ │ │ - this.map.events.unregister("mouseout", this, this.reset); │ │ │ │ │ - this.element.innerHTML = ""; │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + redraw: function() { │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.removeButtons(); │ │ │ │ │ + this._removeZoomBar() │ │ │ │ │ } │ │ │ │ │ + this.draw() │ │ │ │ │ }, │ │ │ │ │ - draw: function() { │ │ │ │ │ + draw: function(px) { │ │ │ │ │ OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (!this.element) { │ │ │ │ │ - this.div.left = ""; │ │ │ │ │ - this.div.top = ""; │ │ │ │ │ - this.element = this.div │ │ │ │ │ + px = this.position.clone(); │ │ │ │ │ + this.buttons = []; │ │ │ │ │ + var sz = { │ │ │ │ │ + w: 18, │ │ │ │ │ + h: 18 │ │ │ │ │ + }; │ │ │ │ │ + if (this.panIcons) { │ │ │ │ │ + var centered = new OpenLayers.Pixel(px.x + sz.w / 2, px.y); │ │ │ │ │ + var wposition = sz.w; │ │ │ │ │ + if (this.zoomWorldIcon) { │ │ │ │ │ + centered = new OpenLayers.Pixel(px.x + sz.w, px.y) │ │ │ │ │ + } │ │ │ │ │ + this._addButton("panup", "north-mini.png", centered, sz); │ │ │ │ │ + px.y = centered.y + sz.h; │ │ │ │ │ + this._addButton("panleft", "west-mini.png", px, sz); │ │ │ │ │ + if (this.zoomWorldIcon) { │ │ │ │ │ + this._addButton("zoomworld", "zoom-world-mini.png", px.add(sz.w, 0), sz); │ │ │ │ │ + wposition *= 2 │ │ │ │ │ + } │ │ │ │ │ + this._addButton("panright", "east-mini.png", px.add(wposition, 0), sz); │ │ │ │ │ + this._addButton("pandown", "south-mini.png", centered.add(0, sz.h * 2), sz); │ │ │ │ │ + this._addButton("zoomin", "zoom-plus-mini.png", centered.add(0, sz.h * 3 + 5), sz); │ │ │ │ │ + centered = this._addZoomBar(centered.add(0, sz.h * 4 + 5)); │ │ │ │ │ + this._addButton("zoomout", "zoom-minus-mini.png", centered, sz) │ │ │ │ │ + } else { │ │ │ │ │ + this._addButton("zoomin", "zoom-plus-mini.png", px, sz); │ │ │ │ │ + centered = this._addZoomBar(px.add(0, sz.h)); │ │ │ │ │ + this._addButton("zoomout", "zoom-minus-mini.png", centered, sz); │ │ │ │ │ + if (this.zoomWorldIcon) { │ │ │ │ │ + centered = centered.add(0, sz.h + 3); │ │ │ │ │ + this._addButton("zoomworld", "zoom-world-mini.png", centered, sz) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ return this.div │ │ │ │ │ }, │ │ │ │ │ - redraw: function(evt) { │ │ │ │ │ - var lonLat; │ │ │ │ │ - if (evt == null) { │ │ │ │ │ - this.reset(); │ │ │ │ │ - return │ │ │ │ │ + _addZoomBar: function(centered) { │ │ │ │ │ + var imgLocation = OpenLayers.Util.getImageLocation("slider.png"); │ │ │ │ │ + var id = this.id + "_" + this.map.id; │ │ │ │ │ + var minZoom = this.map.getMinZoom(); │ │ │ │ │ + var zoomsToEnd = this.map.getNumZoomLevels() - 1 - this.map.getZoom(); │ │ │ │ │ + var slider = OpenLayers.Util.createAlphaImageDiv(id, centered.add(-1, zoomsToEnd * this.zoomStopHeight), { │ │ │ │ │ + w: 20, │ │ │ │ │ + h: 9 │ │ │ │ │ + }, imgLocation, "absolute"); │ │ │ │ │ + slider.style.cursor = "move"; │ │ │ │ │ + this.slider = slider; │ │ │ │ │ + this.sliderEvents = new OpenLayers.Events(this, slider, null, true, { │ │ │ │ │ + includeXY: true │ │ │ │ │ + }); │ │ │ │ │ + this.sliderEvents.on({ │ │ │ │ │ + touchstart: this.zoomBarDown, │ │ │ │ │ + touchmove: this.zoomBarDrag, │ │ │ │ │ + touchend: this.zoomBarUp, │ │ │ │ │ + mousedown: this.zoomBarDown, │ │ │ │ │ + mousemove: this.zoomBarDrag, │ │ │ │ │ + mouseup: this.zoomBarUp │ │ │ │ │ + }); │ │ │ │ │ + var sz = { │ │ │ │ │ + w: this.zoomStopWidth, │ │ │ │ │ + h: this.zoomStopHeight * (this.map.getNumZoomLevels() - minZoom) │ │ │ │ │ + }; │ │ │ │ │ + var imgLocation = OpenLayers.Util.getImageLocation("zoombar.png"); │ │ │ │ │ + var div = null; │ │ │ │ │ + if (OpenLayers.Util.alphaHack()) { │ │ │ │ │ + var id = this.id + "_" + this.map.id; │ │ │ │ │ + div = OpenLayers.Util.createAlphaImageDiv(id, centered, { │ │ │ │ │ + w: sz.w, │ │ │ │ │ + h: this.zoomStopHeight │ │ │ │ │ + }, imgLocation, "absolute", null, "crop"); │ │ │ │ │ + div.style.height = sz.h + "px" │ │ │ │ │ } else { │ │ │ │ │ - if (this.lastXy == null || Math.abs(evt.xy.x - this.lastXy.x) > this.granularity || Math.abs(evt.xy.y - this.lastXy.y) > this.granularity) { │ │ │ │ │ - this.lastXy = evt.xy; │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - lonLat = this.map.getLonLatFromPixel(evt.xy); │ │ │ │ │ - if (!lonLat) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (this.displayProjection) { │ │ │ │ │ - lonLat.transform(this.map.getProjectionObject(), this.displayProjection) │ │ │ │ │ + div = OpenLayers.Util.createDiv("OpenLayers_Control_PanZoomBar_Zoombar" + this.map.id, centered, sz, imgLocation) │ │ │ │ │ + } │ │ │ │ │ + div.style.cursor = "pointer"; │ │ │ │ │ + div.className = "olButton"; │ │ │ │ │ + this.zoombarDiv = div; │ │ │ │ │ + this.div.appendChild(div); │ │ │ │ │ + this.startTop = parseInt(div.style.top); │ │ │ │ │ + this.div.appendChild(slider); │ │ │ │ │ + this.map.events.register("zoomend", this, this.moveZoomBar); │ │ │ │ │ + centered = centered.add(0, this.zoomStopHeight * (this.map.getNumZoomLevels() - minZoom)); │ │ │ │ │ + return centered │ │ │ │ │ + }, │ │ │ │ │ + _removeZoomBar: function() { │ │ │ │ │ + this.sliderEvents.un({ │ │ │ │ │ + touchstart: this.zoomBarDown, │ │ │ │ │ + touchmove: this.zoomBarDrag, │ │ │ │ │ + touchend: this.zoomBarUp, │ │ │ │ │ + mousedown: this.zoomBarDown, │ │ │ │ │ + mousemove: this.zoomBarDrag, │ │ │ │ │ + mouseup: this.zoomBarUp │ │ │ │ │ + }); │ │ │ │ │ + this.sliderEvents.destroy(); │ │ │ │ │ + this.div.removeChild(this.zoombarDiv); │ │ │ │ │ + this.zoombarDiv = null; │ │ │ │ │ + this.div.removeChild(this.slider); │ │ │ │ │ + this.slider = null; │ │ │ │ │ + this.map.events.unregister("zoomend", this, this.moveZoomBar) │ │ │ │ │ + }, │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + OpenLayers.Control.PanZoom.prototype.onButtonClick.apply(this, arguments); │ │ │ │ │ + if (evt.buttonElement === this.zoombarDiv) { │ │ │ │ │ + var levels = evt.buttonXY.y / this.zoomStopHeight; │ │ │ │ │ + if (this.forceFixedZoomLevel || !this.map.fractionalZoom) { │ │ │ │ │ + levels = Math.floor(levels) │ │ │ │ │ } │ │ │ │ │ - this.lastXy = evt.xy │ │ │ │ │ + var zoom = this.map.getNumZoomLevels() - 1 - levels; │ │ │ │ │ + zoom = Math.min(Math.max(zoom, 0), this.map.getNumZoomLevels() - 1); │ │ │ │ │ + this.map.zoomTo(zoom) │ │ │ │ │ } │ │ │ │ │ - var newHtml = this.formatOutput(lonLat); │ │ │ │ │ - if (newHtml != this.element.innerHTML) { │ │ │ │ │ - this.element.innerHTML = newHtml │ │ │ │ │ + }, │ │ │ │ │ + passEventToSlider: function(evt) { │ │ │ │ │ + this.sliderEvents.handleBrowserEvent(evt) │ │ │ │ │ + }, │ │ │ │ │ + zoomBarDown: function(evt) { │ │ │ │ │ + if (!OpenLayers.Event.isLeftClick(evt) && !OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + touchmove: this.passEventToSlider, │ │ │ │ │ + mousemove: this.passEventToSlider, │ │ │ │ │ + mouseup: this.passEventToSlider, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.mouseDragStart = evt.xy.clone(); │ │ │ │ │ + this.zoomStart = evt.xy.clone(); │ │ │ │ │ + this.div.style.cursor = "move"; │ │ │ │ │ + this.zoombarDiv.offsets = null; │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ + }, │ │ │ │ │ + zoomBarDrag: function(evt) { │ │ │ │ │ + if (this.mouseDragStart != null) { │ │ │ │ │ + var deltaY = this.mouseDragStart.y - evt.xy.y; │ │ │ │ │ + var offsets = OpenLayers.Util.pagePosition(this.zoombarDiv); │ │ │ │ │ + if (evt.clientY - offsets[1] > 0 && evt.clientY - offsets[1] < parseInt(this.zoombarDiv.style.height) - 2) { │ │ │ │ │ + var newTop = parseInt(this.slider.style.top) - deltaY; │ │ │ │ │ + this.slider.style.top = newTop + "px"; │ │ │ │ │ + this.mouseDragStart = evt.xy.clone() │ │ │ │ │ + } │ │ │ │ │ + this.deltaY = this.zoomStart.y - evt.xy.y; │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - reset: function(evt) { │ │ │ │ │ - if (this.emptyString != null) { │ │ │ │ │ - this.element.innerHTML = this.emptyString │ │ │ │ │ + zoomBarUp: function(evt) { │ │ │ │ │ + if (!OpenLayers.Event.isLeftClick(evt) && evt.type !== "touchend") { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (this.mouseDragStart) { │ │ │ │ │ + this.div.style.cursor = ""; │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + touchmove: this.passEventToSlider, │ │ │ │ │ + mouseup: this.passEventToSlider, │ │ │ │ │ + mousemove: this.passEventToSlider, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + var zoomLevel = this.map.zoom; │ │ │ │ │ + if (!this.forceFixedZoomLevel && this.map.fractionalZoom) { │ │ │ │ │ + zoomLevel += this.deltaY / this.zoomStopHeight; │ │ │ │ │ + zoomLevel = Math.min(Math.max(zoomLevel, 0), this.map.getNumZoomLevels() - 1) │ │ │ │ │ + } else { │ │ │ │ │ + zoomLevel += this.deltaY / this.zoomStopHeight; │ │ │ │ │ + zoomLevel = Math.max(Math.round(zoomLevel), 0) │ │ │ │ │ + } │ │ │ │ │ + this.map.zoomTo(zoomLevel); │ │ │ │ │ + this.mouseDragStart = null; │ │ │ │ │ + this.zoomStart = null; │ │ │ │ │ + this.deltaY = 0; │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - formatOutput: function(lonLat) { │ │ │ │ │ - var digits = parseInt(this.numDigits); │ │ │ │ │ - var newHtml = this.prefix + lonLat.lon.toFixed(digits) + this.separator + lonLat.lat.toFixed(digits) + this.suffix; │ │ │ │ │ - return newHtml │ │ │ │ │ + moveZoomBar: function() { │ │ │ │ │ + var newTop = (this.map.getNumZoomLevels() - 1 - this.map.getZoom()) * this.zoomStopHeight + this.startTop + 1; │ │ │ │ │ + this.slider.style.top = newTop + "px" │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.MousePosition" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.PanZoomBar" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.Zoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - zoomInText: "+", │ │ │ │ │ - zoomInId: "olZoomInLink", │ │ │ │ │ - zoomOutText: "−", │ │ │ │ │ - zoomOutId: "olZoomOutLink", │ │ │ │ │ +OpenLayers.Control.ScaleLine = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + maxWidth: 100, │ │ │ │ │ + topOutUnits: "km", │ │ │ │ │ + topInUnits: "m", │ │ │ │ │ + bottomOutUnits: "mi", │ │ │ │ │ + bottomInUnits: "ft", │ │ │ │ │ + eTop: null, │ │ │ │ │ + eBottom: null, │ │ │ │ │ + geodesic: false, │ │ │ │ │ draw: function() { │ │ │ │ │ - var div = OpenLayers.Control.prototype.draw.apply(this), │ │ │ │ │ - links = this.getOrCreateLinks(div), │ │ │ │ │ - zoomIn = links.zoomIn, │ │ │ │ │ - zoomOut = links.zoomOut, │ │ │ │ │ - eventsInstance = this.map.events; │ │ │ │ │ - if (zoomOut.parentNode !== div) { │ │ │ │ │ - eventsInstance = this.events; │ │ │ │ │ - eventsInstance.attachToElement(zoomOut.parentNode) │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (!this.eTop) { │ │ │ │ │ + this.eTop = document.createElement("div"); │ │ │ │ │ + this.eTop.className = this.displayClass + "Top"; │ │ │ │ │ + var theLen = this.topInUnits.length; │ │ │ │ │ + this.div.appendChild(this.eTop); │ │ │ │ │ + if (this.topOutUnits == "" || this.topInUnits == "") { │ │ │ │ │ + this.eTop.style.visibility = "hidden" │ │ │ │ │ + } else { │ │ │ │ │ + this.eTop.style.visibility = "visible" │ │ │ │ │ + } │ │ │ │ │ + this.eBottom = document.createElement("div"); │ │ │ │ │ + this.eBottom.className = this.displayClass + "Bottom"; │ │ │ │ │ + this.div.appendChild(this.eBottom); │ │ │ │ │ + if (this.bottomOutUnits == "" || this.bottomInUnits == "") { │ │ │ │ │ + this.eBottom.style.visibility = "hidden" │ │ │ │ │ + } else { │ │ │ │ │ + this.eBottom.style.visibility = "visible" │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - eventsInstance.register("buttonclick", this, this.onZoomClick); │ │ │ │ │ - this.zoomInLink = zoomIn; │ │ │ │ │ - this.zoomOutLink = zoomOut; │ │ │ │ │ - return div │ │ │ │ │ + this.map.events.register("moveend", this, this.update); │ │ │ │ │ + this.update(); │ │ │ │ │ + return this.div │ │ │ │ │ }, │ │ │ │ │ - getOrCreateLinks: function(el) { │ │ │ │ │ - var zoomIn = document.getElementById(this.zoomInId), │ │ │ │ │ - zoomOut = document.getElementById(this.zoomOutId); │ │ │ │ │ - if (!zoomIn) { │ │ │ │ │ - zoomIn = document.createElement("a"); │ │ │ │ │ - zoomIn.href = "#zoomIn"; │ │ │ │ │ - zoomIn.appendChild(document.createTextNode(this.zoomInText)); │ │ │ │ │ - zoomIn.className = "olControlZoomIn"; │ │ │ │ │ - el.appendChild(zoomIn) │ │ │ │ │ + getBarLen: function(maxLen) { │ │ │ │ │ + var digits = parseInt(Math.log(maxLen) / Math.log(10)); │ │ │ │ │ + var pow10 = Math.pow(10, digits); │ │ │ │ │ + var firstChar = parseInt(maxLen / pow10); │ │ │ │ │ + var barLen; │ │ │ │ │ + if (firstChar > 5) { │ │ │ │ │ + barLen = 5 │ │ │ │ │ + } else if (firstChar > 2) { │ │ │ │ │ + barLen = 2 │ │ │ │ │ + } else { │ │ │ │ │ + barLen = 1 │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Element.addClass(zoomIn, "olButton"); │ │ │ │ │ - if (!zoomOut) { │ │ │ │ │ - zoomOut = document.createElement("a"); │ │ │ │ │ - zoomOut.href = "#zoomOut"; │ │ │ │ │ - zoomOut.appendChild(document.createTextNode(this.zoomOutText)); │ │ │ │ │ - zoomOut.className = "olControlZoomOut"; │ │ │ │ │ - el.appendChild(zoomOut) │ │ │ │ │ + return barLen * pow10 │ │ │ │ │ + }, │ │ │ │ │ + update: function() { │ │ │ │ │ + var res = this.map.getResolution(); │ │ │ │ │ + if (!res) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Element.addClass(zoomOut, "olButton"); │ │ │ │ │ - return { │ │ │ │ │ - zoomIn: zoomIn, │ │ │ │ │ - zoomOut: zoomOut │ │ │ │ │ + var curMapUnits = this.map.getUnits(); │ │ │ │ │ + var inches = OpenLayers.INCHES_PER_UNIT; │ │ │ │ │ + var maxSizeData = this.maxWidth * res * inches[curMapUnits]; │ │ │ │ │ + var geodesicRatio = 1; │ │ │ │ │ + if (this.geodesic === true) { │ │ │ │ │ + var maxSizeGeodesic = (this.map.getGeodesicPixelSize().w || 1e-6) * this.maxWidth; │ │ │ │ │ + var maxSizeKilometers = maxSizeData / inches["km"]; │ │ │ │ │ + geodesicRatio = maxSizeGeodesic / maxSizeKilometers; │ │ │ │ │ + maxSizeData *= geodesicRatio │ │ │ │ │ + } │ │ │ │ │ + var topUnits; │ │ │ │ │ + var bottomUnits; │ │ │ │ │ + if (maxSizeData > 1e5) { │ │ │ │ │ + topUnits = this.topOutUnits; │ │ │ │ │ + bottomUnits = this.bottomOutUnits │ │ │ │ │ + } else { │ │ │ │ │ + topUnits = this.topInUnits; │ │ │ │ │ + bottomUnits = this.bottomInUnits │ │ │ │ │ + } │ │ │ │ │ + var topMax = maxSizeData / inches[topUnits]; │ │ │ │ │ + var bottomMax = maxSizeData / inches[bottomUnits]; │ │ │ │ │ + var topRounded = this.getBarLen(topMax); │ │ │ │ │ + var bottomRounded = this.getBarLen(bottomMax); │ │ │ │ │ + topMax = topRounded / inches[curMapUnits] * inches[topUnits]; │ │ │ │ │ + bottomMax = bottomRounded / inches[curMapUnits] * inches[bottomUnits]; │ │ │ │ │ + var topPx = topMax / res / geodesicRatio; │ │ │ │ │ + var bottomPx = bottomMax / res / geodesicRatio; │ │ │ │ │ + if (this.eBottom.style.visibility == "visible") { │ │ │ │ │ + this.eBottom.style.width = Math.round(bottomPx) + "px"; │ │ │ │ │ + this.eBottom.innerHTML = bottomRounded + " " + bottomUnits │ │ │ │ │ + } │ │ │ │ │ + if (this.eTop.style.visibility == "visible") { │ │ │ │ │ + this.eTop.style.width = Math.round(topPx) + "px"; │ │ │ │ │ + this.eTop.innerHTML = topRounded + " " + topUnits │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - onZoomClick: function(evt) { │ │ │ │ │ - var button = evt.buttonElement; │ │ │ │ │ - if (button === this.zoomInLink) { │ │ │ │ │ - this.map.zoomIn() │ │ │ │ │ - } else if (button === this.zoomOutLink) { │ │ │ │ │ - this.map.zoomOut() │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ScaleLine" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Vector.RootContainer = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + layers: null, │ │ │ │ │ + display: function() {}, │ │ │ │ │ + getFeatureFromEvent: function(evt) { │ │ │ │ │ + var layers = this.layers; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0; i < layers.length; i++) { │ │ │ │ │ + feature = layers[i].getFeatureFromEvent(evt); │ │ │ │ │ + if (feature) { │ │ │ │ │ + return feature │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onZoomClick) │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.collectRoots(); │ │ │ │ │ + map.events.register("changelayer", this, this.handleChangeLayer) │ │ │ │ │ + }, │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + map.events.unregister("changelayer", this, this.handleChangeLayer); │ │ │ │ │ + this.resetRoots(); │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + collectRoots: function() { │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = 0; i < this.map.layers.length; ++i) { │ │ │ │ │ + layer = this.map.layers[i]; │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ + layer.renderer.moveRoot(this.renderer) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - delete this.zoomInLink; │ │ │ │ │ - delete this.zoomOutLink; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Zoom" │ │ │ │ │ + resetRoots: function() { │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = 0; i < this.layers.length; ++i) { │ │ │ │ │ + layer = this.layers[i]; │ │ │ │ │ + if (this.renderer && layer.renderer.getRenderLayerId() == this.id) { │ │ │ │ │ + this.renderer.moveRoot(layer.renderer) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + handleChangeLayer: function(evt) { │ │ │ │ │ + var layer = evt.layer; │ │ │ │ │ + if (evt.property == "order" && OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ + this.resetRoots(); │ │ │ │ │ + this.collectRoots() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Vector.RootContainer" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Control.SelectFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ multipleKey: null, │ │ │ │ │ toggleKey: null, │ │ │ │ │ multiple: false, │ │ │ │ │ clickout: true, │ │ │ │ │ toggle: false, │ │ │ │ │ @@ -33658,14 +33173,236 @@ │ │ │ │ │ this.handlers.feature.layer = this.layer; │ │ │ │ │ if (isActive) { │ │ │ │ │ this.activate() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.SelectFeature" │ │ │ │ │ }); │ │ │ │ │ +OpenLayers.Control.EditingToolbar = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ + citeCompliant: false, │ │ │ │ │ + initialize: function(layer, options) { │ │ │ │ │ + OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.addControls([new OpenLayers.Control.Navigation]); │ │ │ │ │ + var controls = [new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Point, { │ │ │ │ │ + displayClass: "olControlDrawFeaturePoint", │ │ │ │ │ + handlerOptions: { │ │ │ │ │ + citeCompliant: this.citeCompliant │ │ │ │ │ + } │ │ │ │ │ + }), new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Path, { │ │ │ │ │ + displayClass: "olControlDrawFeaturePath", │ │ │ │ │ + handlerOptions: { │ │ │ │ │ + citeCompliant: this.citeCompliant │ │ │ │ │ + } │ │ │ │ │ + }), new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Polygon, { │ │ │ │ │ + displayClass: "olControlDrawFeaturePolygon", │ │ │ │ │ + handlerOptions: { │ │ │ │ │ + citeCompliant: this.citeCompliant │ │ │ │ │ + } │ │ │ │ │ + })]; │ │ │ │ │ + this.addControls(controls) │ │ │ │ │ + }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + var div = OpenLayers.Control.Panel.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (this.defaultControl === null) { │ │ │ │ │ + this.defaultControl = this.controls[0] │ │ │ │ │ + } │ │ │ │ │ + return div │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.EditingToolbar" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Attribution = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + separator: ", ", │ │ │ │ │ + template: "${layers}", │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + removelayer: this.updateAttribution, │ │ │ │ │ + addlayer: this.updateAttribution, │ │ │ │ │ + changelayer: this.updateAttribution, │ │ │ │ │ + changebaselayer: this.updateAttribution, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + changebaselayer: this.updateAttribution, │ │ │ │ │ + changelayer: this.updateAttribution, │ │ │ │ │ + addlayer: this.updateAttribution, │ │ │ │ │ + removelayer: this.updateAttribution, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.updateAttribution(); │ │ │ │ │ + return this.div │ │ │ │ │ + }, │ │ │ │ │ + updateAttribution: function() { │ │ │ │ │ + var attributions = []; │ │ │ │ │ + if (this.map && this.map.layers) { │ │ │ │ │ + for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if (layer.attribution && layer.getVisibility()) { │ │ │ │ │ + if (OpenLayers.Util.indexOf(attributions, layer.attribution) === -1) { │ │ │ │ │ + attributions.push(layer.attribution) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.div.innerHTML = OpenLayers.String.format(this.template, { │ │ │ │ │ + layers: attributions.join(this.separator) │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Attribution" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Geolocate = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + geolocation: null, │ │ │ │ │ + available: "geolocation" in navigator, │ │ │ │ │ + bind: true, │ │ │ │ │ + watch: false, │ │ │ │ │ + geolocationOptions: null, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (this.available && !this.geolocation) { │ │ │ │ │ + this.geolocation = navigator.geolocation │ │ │ │ │ + } │ │ │ │ │ + if (!this.geolocation) { │ │ │ │ │ + this.events.triggerEvent("locationuncapable"); │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + if (this.watch) { │ │ │ │ │ + this.watchId = this.geolocation.watchPosition(OpenLayers.Function.bind(this.geolocate, this), OpenLayers.Function.bind(this.failure, this), this.geolocationOptions) │ │ │ │ │ + } else { │ │ │ │ │ + this.getCurrentLocation() │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active && this.watchId !== null) { │ │ │ │ │ + this.geolocation.clearWatch(this.watchId) │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Control.prototype.deactivate.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + geolocate: function(position) { │ │ │ │ │ + var center = new OpenLayers.LonLat(position.coords.longitude, position.coords.latitude).transform(new OpenLayers.Projection("EPSG:4326"), this.map.getProjectionObject()); │ │ │ │ │ + if (this.bind) { │ │ │ │ │ + this.map.setCenter(center) │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("locationupdated", { │ │ │ │ │ + position: position, │ │ │ │ │ + point: new OpenLayers.Geometry.Point(center.lon, center.lat) │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + getCurrentLocation: function() { │ │ │ │ │ + if (!this.active || this.watch) { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + this.geolocation.getCurrentPosition(OpenLayers.Function.bind(this.geolocate, this), OpenLayers.Function.bind(this.failure, this), this.geolocationOptions); │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + failure: function(error) { │ │ │ │ │ + this.events.triggerEvent("locationfailed", { │ │ │ │ │ + error: error │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Geolocate" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.UTFGrid = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + layers: null, │ │ │ │ │ + defaultHandlerOptions: { │ │ │ │ │ + delay: 300, │ │ │ │ │ + pixelTolerance: 4, │ │ │ │ │ + stopMove: false, │ │ │ │ │ + single: true, │ │ │ │ │ + double: false, │ │ │ │ │ + stopSingle: false, │ │ │ │ │ + stopDouble: false │ │ │ │ │ + }, │ │ │ │ │ + handlerMode: "click", │ │ │ │ │ + setHandler: function(hm) { │ │ │ │ │ + this.handlerMode = hm; │ │ │ │ │ + this.resetHandler() │ │ │ │ │ + }, │ │ │ │ │ + resetHandler: function() { │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.deactivate(); │ │ │ │ │ + this.handler.destroy(); │ │ │ │ │ + this.handler = null │ │ │ │ │ + } │ │ │ │ │ + if (this.handlerMode == "hover") { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ + pause: this.handleEvent, │ │ │ │ │ + move: this.reset │ │ │ │ │ + }, this.handlerOptions) │ │ │ │ │ + } else if (this.handlerMode == "click") { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Click(this, { │ │ │ │ │ + click: this.handleEvent │ │ │ │ │ + }, this.handlerOptions) │ │ │ │ │ + } else if (this.handlerMode == "move") { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ + pause: this.handleEvent, │ │ │ │ │ + move: this.handleEvent │ │ │ │ │ + }, this.handlerOptions) │ │ │ │ │ + } │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.handlerOptions = options.handlerOptions || this.defaultHandlerOptions; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.resetHandler() │ │ │ │ │ + }, │ │ │ │ │ + handleEvent: function(evt) { │ │ │ │ │ + if (evt == null) { │ │ │ │ │ + this.reset(); │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var lonLat = this.map.getLonLatFromPixel(evt.xy); │ │ │ │ │ + if (!lonLat) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var layers = this.findLayers(); │ │ │ │ │ + if (layers.length > 0) { │ │ │ │ │ + var infoLookup = {}; │ │ │ │ │ + var layer, idx; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + layer = layers[i]; │ │ │ │ │ + idx = OpenLayers.Util.indexOf(this.map.layers, layer); │ │ │ │ │ + infoLookup[idx] = layer.getFeatureInfo(lonLat) │ │ │ │ │ + } │ │ │ │ │ + this.callback(infoLookup, lonLat, evt.xy) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + callback: function(infoLookup) {}, │ │ │ │ │ + reset: function(evt) { │ │ │ │ │ + this.callback(null) │ │ │ │ │ + }, │ │ │ │ │ + findLayers: function() { │ │ │ │ │ + var candidates = this.layers || this.map.layers; │ │ │ │ │ + var layers = []; │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = candidates[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.UTFGrid) { │ │ │ │ │ + layers.push(layer) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return layers │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.UTFGrid" │ │ │ │ │ +}); │ │ │ │ │ OpenLayers.Control.NavigationHistory = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ type: OpenLayers.Control.TYPE_TOGGLE, │ │ │ │ │ previous: null, │ │ │ │ │ previousOptions: null, │ │ │ │ │ next: null, │ │ │ │ │ nextOptions: null, │ │ │ │ │ limit: 50, │ │ │ │ │ @@ -33843,549 +33580,14 @@ │ │ │ │ │ deactivated = true │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ return deactivated │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.NavigationHistory" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.TouchNavigation = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - dragPan: null, │ │ │ │ │ - dragPanOptions: null, │ │ │ │ │ - pinchZoom: null, │ │ │ │ │ - pinchZoomOptions: null, │ │ │ │ │ - clickHandlerOptions: null, │ │ │ │ │ - documentDrag: false, │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.handlers = {}; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - if (this.dragPan) { │ │ │ │ │ - this.dragPan.destroy() │ │ │ │ │ - } │ │ │ │ │ - this.dragPan = null; │ │ │ │ │ - if (this.pinchZoom) { │ │ │ │ │ - this.pinchZoom.destroy(); │ │ │ │ │ - delete this.pinchZoom │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.dragPan.activate(); │ │ │ │ │ - this.handlers.click.activate(); │ │ │ │ │ - this.pinchZoom.activate(); │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.dragPan.deactivate(); │ │ │ │ │ - this.handlers.click.deactivate(); │ │ │ │ │ - this.pinchZoom.deactivate(); │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - var clickCallbacks = { │ │ │ │ │ - click: this.defaultClick, │ │ │ │ │ - dblclick: this.defaultDblClick │ │ │ │ │ - }; │ │ │ │ │ - var clickOptions = OpenLayers.Util.extend({ │ │ │ │ │ - double: true, │ │ │ │ │ - stopDouble: true, │ │ │ │ │ - pixelTolerance: 2 │ │ │ │ │ - }, this.clickHandlerOptions); │ │ │ │ │ - this.handlers.click = new OpenLayers.Handler.Click(this, clickCallbacks, clickOptions); │ │ │ │ │ - this.dragPan = new OpenLayers.Control.DragPan(OpenLayers.Util.extend({ │ │ │ │ │ - map: this.map, │ │ │ │ │ - documentDrag: this.documentDrag │ │ │ │ │ - }, this.dragPanOptions)); │ │ │ │ │ - this.dragPan.draw(); │ │ │ │ │ - this.pinchZoom = new OpenLayers.Control.PinchZoom(OpenLayers.Util.extend({ │ │ │ │ │ - map: this.map │ │ │ │ │ - }, this.pinchZoomOptions)) │ │ │ │ │ - }, │ │ │ │ │ - defaultClick: function(evt) { │ │ │ │ │ - if (evt.lastTouches && evt.lastTouches.length == 2) { │ │ │ │ │ - this.map.zoomOut() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - defaultDblClick: function(evt) { │ │ │ │ │ - this.map.zoomTo(this.map.zoom + 1, evt.xy) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.TouchNavigation" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.LayerSwitcher = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - layerStates: null, │ │ │ │ │ - layersDiv: null, │ │ │ │ │ - baseLayersDiv: null, │ │ │ │ │ - baseLayers: null, │ │ │ │ │ - dataLbl: null, │ │ │ │ │ - dataLayersDiv: null, │ │ │ │ │ - dataLayers: null, │ │ │ │ │ - minimizeDiv: null, │ │ │ │ │ - maximizeDiv: null, │ │ │ │ │ - ascending: true, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.layerStates = [] │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.clearLayersArray("base"); │ │ │ │ │ - this.clearLayersArray("data"); │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - buttonclick: this.onButtonClick, │ │ │ │ │ - addlayer: this.redraw, │ │ │ │ │ - changelayer: this.redraw, │ │ │ │ │ - removelayer: this.redraw, │ │ │ │ │ - changebaselayer: this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - addlayer: this.redraw, │ │ │ │ │ - changelayer: this.redraw, │ │ │ │ │ - removelayer: this.redraw, │ │ │ │ │ - changebaselayer: this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - if (this.outsideViewport) { │ │ │ │ │ - this.events.attachToElement(this.div); │ │ │ │ │ - this.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ - } else { │ │ │ │ │ - this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this); │ │ │ │ │ - this.loadContents(); │ │ │ │ │ - if (!this.outsideViewport) { │ │ │ │ │ - this.minimizeControl() │ │ │ │ │ - } │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - var button = evt.buttonElement; │ │ │ │ │ - if (button === this.minimizeDiv) { │ │ │ │ │ - this.minimizeControl() │ │ │ │ │ - } else if (button === this.maximizeDiv) { │ │ │ │ │ - this.maximizeControl() │ │ │ │ │ - } else if (button._layerSwitcher === this.id) { │ │ │ │ │ - if (button["for"]) { │ │ │ │ │ - button = document.getElementById(button["for"]) │ │ │ │ │ - } │ │ │ │ │ - if (!button.disabled) { │ │ │ │ │ - if (button.type == "radio") { │ │ │ │ │ - button.checked = true; │ │ │ │ │ - this.map.setBaseLayer(this.map.getLayer(button._layer)) │ │ │ │ │ - } else { │ │ │ │ │ - button.checked = !button.checked; │ │ │ │ │ - this.updateMap() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - clearLayersArray: function(layersType) { │ │ │ │ │ - this[layersType + "LayersDiv"].innerHTML = ""; │ │ │ │ │ - this[layersType + "Layers"] = [] │ │ │ │ │ - }, │ │ │ │ │ - checkRedraw: function() { │ │ │ │ │ - if (!this.layerStates.length || this.map.layers.length != this.layerStates.length) { │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = this.layerStates.length; i < len; i++) { │ │ │ │ │ - var layerState = this.layerStates[i]; │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if (layerState.name != layer.name || layerState.inRange != layer.inRange || layerState.id != layer.id || layerState.visibility != layer.visibility) { │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - redraw: function() { │ │ │ │ │ - if (!this.checkRedraw()) { │ │ │ │ │ - return this.div │ │ │ │ │ - } │ │ │ │ │ - this.clearLayersArray("base"); │ │ │ │ │ - this.clearLayersArray("data"); │ │ │ │ │ - var containsOverlays = false; │ │ │ │ │ - var containsBaseLayers = false; │ │ │ │ │ - var len = this.map.layers.length; │ │ │ │ │ - this.layerStates = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - this.layerStates[i] = { │ │ │ │ │ - name: layer.name, │ │ │ │ │ - visibility: layer.visibility, │ │ │ │ │ - inRange: layer.inRange, │ │ │ │ │ - id: layer.id │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var layers = this.map.layers.slice(); │ │ │ │ │ - if (!this.ascending) { │ │ │ │ │ - layers.reverse() │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - var layer = layers[i]; │ │ │ │ │ - var baseLayer = layer.isBaseLayer; │ │ │ │ │ - if (layer.displayInLayerSwitcher) { │ │ │ │ │ - if (baseLayer) { │ │ │ │ │ - containsBaseLayers = true │ │ │ │ │ - } else { │ │ │ │ │ - containsOverlays = true │ │ │ │ │ - } │ │ │ │ │ - var checked = baseLayer ? layer == this.map.baseLayer : layer.getVisibility(); │ │ │ │ │ - var inputElem = document.createElement("input"), │ │ │ │ │ - inputId = OpenLayers.Util.createUniqueID(this.id + "_input_"); │ │ │ │ │ - inputElem.id = inputId; │ │ │ │ │ - inputElem.name = baseLayer ? this.id + "_baseLayers" : layer.name; │ │ │ │ │ - inputElem.type = baseLayer ? "radio" : "checkbox"; │ │ │ │ │ - inputElem.value = layer.name; │ │ │ │ │ - inputElem.checked = checked; │ │ │ │ │ - inputElem.defaultChecked = checked; │ │ │ │ │ - inputElem.className = "olButton"; │ │ │ │ │ - inputElem._layer = layer.id; │ │ │ │ │ - inputElem._layerSwitcher = this.id; │ │ │ │ │ - if (!baseLayer && !layer.inRange) { │ │ │ │ │ - inputElem.disabled = true │ │ │ │ │ - } │ │ │ │ │ - var labelSpan = document.createElement("label"); │ │ │ │ │ - labelSpan["for"] = inputElem.id; │ │ │ │ │ - OpenLayers.Element.addClass(labelSpan, "labelSpan olButton"); │ │ │ │ │ - labelSpan._layer = layer.id; │ │ │ │ │ - labelSpan._layerSwitcher = this.id; │ │ │ │ │ - if (!baseLayer && !layer.inRange) { │ │ │ │ │ - labelSpan.style.color = "gray" │ │ │ │ │ - } │ │ │ │ │ - labelSpan.innerHTML = layer.name; │ │ │ │ │ - labelSpan.style.verticalAlign = baseLayer ? "bottom" : "baseline"; │ │ │ │ │ - var br = document.createElement("br"); │ │ │ │ │ - var groupArray = baseLayer ? this.baseLayers : this.dataLayers; │ │ │ │ │ - groupArray.push({ │ │ │ │ │ - layer: layer, │ │ │ │ │ - inputElem: inputElem, │ │ │ │ │ - labelSpan: labelSpan │ │ │ │ │ - }); │ │ │ │ │ - var groupDiv = baseLayer ? this.baseLayersDiv : this.dataLayersDiv; │ │ │ │ │ - groupDiv.appendChild(inputElem); │ │ │ │ │ - groupDiv.appendChild(labelSpan); │ │ │ │ │ - groupDiv.appendChild(br) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.dataLbl.style.display = containsOverlays ? "" : "none"; │ │ │ │ │ - this.baseLbl.style.display = containsBaseLayers ? "" : "none"; │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - updateMap: function() { │ │ │ │ │ - for (var i = 0, len = this.baseLayers.length; i < len; i++) { │ │ │ │ │ - var layerEntry = this.baseLayers[i]; │ │ │ │ │ - if (layerEntry.inputElem.checked) { │ │ │ │ │ - this.map.setBaseLayer(layerEntry.layer, false) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = this.dataLayers.length; i < len; i++) { │ │ │ │ │ - var layerEntry = this.dataLayers[i]; │ │ │ │ │ - layerEntry.layer.setVisibility(layerEntry.inputElem.checked) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - maximizeControl: function(e) { │ │ │ │ │ - this.div.style.width = ""; │ │ │ │ │ - this.div.style.height = ""; │ │ │ │ │ - this.showControls(false); │ │ │ │ │ - if (e != null) { │ │ │ │ │ - OpenLayers.Event.stop(e) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - minimizeControl: function(e) { │ │ │ │ │ - this.div.style.width = "0px"; │ │ │ │ │ - this.div.style.height = "0px"; │ │ │ │ │ - this.showControls(true); │ │ │ │ │ - if (e != null) { │ │ │ │ │ - OpenLayers.Event.stop(e) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - showControls: function(minimize) { │ │ │ │ │ - this.maximizeDiv.style.display = minimize ? "" : "none"; │ │ │ │ │ - this.minimizeDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ - this.layersDiv.style.display = minimize ? "none" : "" │ │ │ │ │ - }, │ │ │ │ │ - loadContents: function() { │ │ │ │ │ - this.layersDiv = document.createElement("div"); │ │ │ │ │ - this.layersDiv.id = this.id + "_layersDiv"; │ │ │ │ │ - OpenLayers.Element.addClass(this.layersDiv, "layersDiv"); │ │ │ │ │ - this.baseLbl = document.createElement("div"); │ │ │ │ │ - this.baseLbl.innerHTML = OpenLayers.i18n("Base Layer"); │ │ │ │ │ - OpenLayers.Element.addClass(this.baseLbl, "baseLbl"); │ │ │ │ │ - this.baseLayersDiv = document.createElement("div"); │ │ │ │ │ - OpenLayers.Element.addClass(this.baseLayersDiv, "baseLayersDiv"); │ │ │ │ │ - this.dataLbl = document.createElement("div"); │ │ │ │ │ - this.dataLbl.innerHTML = OpenLayers.i18n("Overlays"); │ │ │ │ │ - OpenLayers.Element.addClass(this.dataLbl, "dataLbl"); │ │ │ │ │ - this.dataLayersDiv = document.createElement("div"); │ │ │ │ │ - OpenLayers.Element.addClass(this.dataLayersDiv, "dataLayersDiv"); │ │ │ │ │ - if (this.ascending) { │ │ │ │ │ - this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLayersDiv) │ │ │ │ │ - } else { │ │ │ │ │ - this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLayersDiv) │ │ │ │ │ - } │ │ │ │ │ - this.div.appendChild(this.layersDiv); │ │ │ │ │ - var img = OpenLayers.Util.getImageLocation("layer-switcher-maximize.png"); │ │ │ │ │ - this.maximizeDiv = OpenLayers.Util.createAlphaImageDiv("OpenLayers_Control_MaximizeDiv", null, null, img, "absolute"); │ │ │ │ │ - OpenLayers.Element.addClass(this.maximizeDiv, "maximizeDiv olButton"); │ │ │ │ │ - this.maximizeDiv.style.display = "none"; │ │ │ │ │ - this.div.appendChild(this.maximizeDiv); │ │ │ │ │ - var img = OpenLayers.Util.getImageLocation("layer-switcher-minimize.png"); │ │ │ │ │ - this.minimizeDiv = OpenLayers.Util.createAlphaImageDiv("OpenLayers_Control_MinimizeDiv", null, null, img, "absolute"); │ │ │ │ │ - OpenLayers.Element.addClass(this.minimizeDiv, "minimizeDiv olButton"); │ │ │ │ │ - this.minimizeDiv.style.display = "none"; │ │ │ │ │ - this.div.appendChild(this.minimizeDiv) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.LayerSwitcher" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.Graticule = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - intervals: [45, 30, 20, 10, 5, 2, 1, .5, .2, .1, .05, .01, .005, .002, .001], │ │ │ │ │ - displayInLayerSwitcher: true, │ │ │ │ │ - visible: true, │ │ │ │ │ - numPoints: 50, │ │ │ │ │ - targetSize: 200, │ │ │ │ │ - layerName: null, │ │ │ │ │ - labelled: true, │ │ │ │ │ - labelFormat: "dm", │ │ │ │ │ - lineSymbolizer: { │ │ │ │ │ - strokeColor: "#333", │ │ │ │ │ - strokeWidth: 1, │ │ │ │ │ - strokeOpacity: .5 │ │ │ │ │ - }, │ │ │ │ │ - labelSymbolizer: {}, │ │ │ │ │ - gratLayer: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.layerName = options.layerName || OpenLayers.i18n("Graticule"); │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.labelSymbolizer.stroke = false; │ │ │ │ │ - this.labelSymbolizer.fill = false; │ │ │ │ │ - this.labelSymbolizer.label = "${label}"; │ │ │ │ │ - this.labelSymbolizer.labelAlign = "${labelAlign}"; │ │ │ │ │ - this.labelSymbolizer.labelXOffset = "${xOffset}"; │ │ │ │ │ - this.labelSymbolizer.labelYOffset = "${yOffset}" │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - if (this.gratLayer) { │ │ │ │ │ - this.gratLayer.destroy(); │ │ │ │ │ - this.gratLayer = null │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (!this.gratLayer) { │ │ │ │ │ - var gratStyle = new OpenLayers.Style({}, { │ │ │ │ │ - rules: [new OpenLayers.Rule({ │ │ │ │ │ - symbolizer: { │ │ │ │ │ - Point: this.labelSymbolizer, │ │ │ │ │ - Line: this.lineSymbolizer │ │ │ │ │ - } │ │ │ │ │ - })] │ │ │ │ │ - }); │ │ │ │ │ - this.gratLayer = new OpenLayers.Layer.Vector(this.layerName, { │ │ │ │ │ - styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ - default: gratStyle │ │ │ │ │ - }), │ │ │ │ │ - visibility: this.visible, │ │ │ │ │ - displayInLayerSwitcher: this.displayInLayerSwitcher │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.map.addLayer(this.gratLayer); │ │ │ │ │ - this.map.events.register("moveend", this, this.update); │ │ │ │ │ - this.update(); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.map.events.unregister("moveend", this, this.update); │ │ │ │ │ - this.map.removeLayer(this.gratLayer); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - update: function() { │ │ │ │ │ - var mapBounds = this.map.getExtent(); │ │ │ │ │ - if (!mapBounds) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - this.gratLayer.destroyFeatures(); │ │ │ │ │ - var llProj = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - var mapProj = this.map.getProjectionObject(); │ │ │ │ │ - var mapRes = this.map.getResolution(); │ │ │ │ │ - if (mapProj.proj && mapProj.proj.projName == "longlat") { │ │ │ │ │ - this.numPoints = 1 │ │ │ │ │ - } │ │ │ │ │ - var mapCenter = this.map.getCenter(); │ │ │ │ │ - var mapCenterLL = new OpenLayers.Pixel(mapCenter.lon, mapCenter.lat); │ │ │ │ │ - OpenLayers.Projection.transform(mapCenterLL, mapProj, llProj); │ │ │ │ │ - var testSq = this.targetSize * mapRes; │ │ │ │ │ - testSq *= testSq; │ │ │ │ │ - var llInterval; │ │ │ │ │ - for (var i = 0; i < this.intervals.length; ++i) { │ │ │ │ │ - llInterval = this.intervals[i]; │ │ │ │ │ - var delta = llInterval / 2; │ │ │ │ │ - var p1 = mapCenterLL.offset({ │ │ │ │ │ - x: -delta, │ │ │ │ │ - y: -delta │ │ │ │ │ - }); │ │ │ │ │ - var p2 = mapCenterLL.offset({ │ │ │ │ │ - x: delta, │ │ │ │ │ - y: delta │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Projection.transform(p1, llProj, mapProj); │ │ │ │ │ - OpenLayers.Projection.transform(p2, llProj, mapProj); │ │ │ │ │ - var distSq = (p1.x - p2.x) * (p1.x - p2.x) + (p1.y - p2.y) * (p1.y - p2.y); │ │ │ │ │ - if (distSq <= testSq) { │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - mapCenterLL.x = Math.floor(mapCenterLL.x / llInterval) * llInterval; │ │ │ │ │ - mapCenterLL.y = Math.floor(mapCenterLL.y / llInterval) * llInterval; │ │ │ │ │ - var iter = 0; │ │ │ │ │ - var centerLonPoints = [mapCenterLL.clone()]; │ │ │ │ │ - var newPoint = mapCenterLL.clone(); │ │ │ │ │ - var mapXY; │ │ │ │ │ - do { │ │ │ │ │ - newPoint = newPoint.offset({ │ │ │ │ │ - x: 0, │ │ │ │ │ - y: llInterval │ │ │ │ │ - }); │ │ │ │ │ - mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ - centerLonPoints.unshift(newPoint) │ │ │ │ │ - } while (mapBounds.containsPixel(mapXY) && ++iter < 1e3); │ │ │ │ │ - newPoint = mapCenterLL.clone(); │ │ │ │ │ - do { │ │ │ │ │ - newPoint = newPoint.offset({ │ │ │ │ │ - x: 0, │ │ │ │ │ - y: -llInterval │ │ │ │ │ - }); │ │ │ │ │ - mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ - centerLonPoints.push(newPoint) │ │ │ │ │ - } while (mapBounds.containsPixel(mapXY) && ++iter < 1e3); │ │ │ │ │ - iter = 0; │ │ │ │ │ - var centerLatPoints = [mapCenterLL.clone()]; │ │ │ │ │ - newPoint = mapCenterLL.clone(); │ │ │ │ │ - do { │ │ │ │ │ - newPoint = newPoint.offset({ │ │ │ │ │ - x: -llInterval, │ │ │ │ │ - y: 0 │ │ │ │ │ - }); │ │ │ │ │ - mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ - centerLatPoints.unshift(newPoint) │ │ │ │ │ - } while (mapBounds.containsPixel(mapXY) && ++iter < 1e3); │ │ │ │ │ - newPoint = mapCenterLL.clone(); │ │ │ │ │ - do { │ │ │ │ │ - newPoint = newPoint.offset({ │ │ │ │ │ - x: llInterval, │ │ │ │ │ - y: 0 │ │ │ │ │ - }); │ │ │ │ │ - mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ - centerLatPoints.push(newPoint) │ │ │ │ │ - } while (mapBounds.containsPixel(mapXY) && ++iter < 1e3); │ │ │ │ │ - var lines = []; │ │ │ │ │ - for (var i = 0; i < centerLatPoints.length; ++i) { │ │ │ │ │ - var lon = centerLatPoints[i].x; │ │ │ │ │ - var pointList = []; │ │ │ │ │ - var labelPoint = null; │ │ │ │ │ - var latEnd = Math.min(centerLonPoints[0].y, 90); │ │ │ │ │ - var latStart = Math.max(centerLonPoints[centerLonPoints.length - 1].y, -90); │ │ │ │ │ - var latDelta = (latEnd - latStart) / this.numPoints; │ │ │ │ │ - var lat = latStart; │ │ │ │ │ - for (var j = 0; j <= this.numPoints; ++j) { │ │ │ │ │ - var gridPoint = new OpenLayers.Geometry.Point(lon, lat); │ │ │ │ │ - gridPoint.transform(llProj, mapProj); │ │ │ │ │ - pointList.push(gridPoint); │ │ │ │ │ - lat += latDelta; │ │ │ │ │ - if (gridPoint.y >= mapBounds.bottom && !labelPoint) { │ │ │ │ │ - labelPoint = gridPoint │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.labelled) { │ │ │ │ │ - var labelPos = new OpenLayers.Geometry.Point(labelPoint.x, mapBounds.bottom); │ │ │ │ │ - var labelAttrs = { │ │ │ │ │ - value: lon, │ │ │ │ │ - label: this.labelled ? OpenLayers.Util.getFormattedLonLat(lon, "lon", this.labelFormat) : "", │ │ │ │ │ - labelAlign: "cb", │ │ │ │ │ - xOffset: 0, │ │ │ │ │ - yOffset: 2 │ │ │ │ │ - }; │ │ │ │ │ - this.gratLayer.addFeatures(new OpenLayers.Feature.Vector(labelPos, labelAttrs)) │ │ │ │ │ - } │ │ │ │ │ - var geom = new OpenLayers.Geometry.LineString(pointList); │ │ │ │ │ - lines.push(new OpenLayers.Feature.Vector(geom)) │ │ │ │ │ - } │ │ │ │ │ - for (var j = 0; j < centerLonPoints.length; ++j) { │ │ │ │ │ - lat = centerLonPoints[j].y; │ │ │ │ │ - if (lat < -90 || lat > 90) { │ │ │ │ │ - continue │ │ │ │ │ - } │ │ │ │ │ - var pointList = []; │ │ │ │ │ - var lonStart = centerLatPoints[0].x; │ │ │ │ │ - var lonEnd = centerLatPoints[centerLatPoints.length - 1].x; │ │ │ │ │ - var lonDelta = (lonEnd - lonStart) / this.numPoints; │ │ │ │ │ - var lon = lonStart; │ │ │ │ │ - var labelPoint = null; │ │ │ │ │ - for (var i = 0; i <= this.numPoints; ++i) { │ │ │ │ │ - var gridPoint = new OpenLayers.Geometry.Point(lon, lat); │ │ │ │ │ - gridPoint.transform(llProj, mapProj); │ │ │ │ │ - pointList.push(gridPoint); │ │ │ │ │ - lon += lonDelta; │ │ │ │ │ - if (gridPoint.x < mapBounds.right) { │ │ │ │ │ - labelPoint = gridPoint │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.labelled) { │ │ │ │ │ - var labelPos = new OpenLayers.Geometry.Point(mapBounds.right, labelPoint.y); │ │ │ │ │ - var labelAttrs = { │ │ │ │ │ - value: lat, │ │ │ │ │ - label: this.labelled ? OpenLayers.Util.getFormattedLonLat(lat, "lat", this.labelFormat) : "", │ │ │ │ │ - labelAlign: "rb", │ │ │ │ │ - xOffset: -2, │ │ │ │ │ - yOffset: 2 │ │ │ │ │ - }; │ │ │ │ │ - this.gratLayer.addFeatures(new OpenLayers.Feature.Vector(labelPos, labelAttrs)) │ │ │ │ │ - } │ │ │ │ │ - var geom = new OpenLayers.Geometry.LineString(pointList); │ │ │ │ │ - lines.push(new OpenLayers.Feature.Vector(geom)) │ │ │ │ │ - } │ │ │ │ │ - this.gratLayer.addFeatures(lines) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Graticule" │ │ │ │ │ -}); │ │ │ │ │ OpenLayers.Control.Measure = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ callbacks: null, │ │ │ │ │ displaySystem: "metric", │ │ │ │ │ geodesic: false, │ │ │ │ │ displaySystemUnits: { │ │ │ │ │ geographic: ["dd"], │ │ │ │ │ english: ["mi", "ft", "in"], │ │ │ │ │ @@ -34534,3164 +33736,3962 @@ │ │ │ │ │ var inPerMapUnit = OpenLayers.INCHES_PER_UNIT[geomUnits]; │ │ │ │ │ length *= inPerMapUnit / inPerDisplayUnit │ │ │ │ │ } │ │ │ │ │ return length │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.Measure" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.Scale = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - element: null, │ │ │ │ │ - geodesic: false, │ │ │ │ │ - initialize: function(element, options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.element = OpenLayers.Util.getElement(element) │ │ │ │ │ +OpenLayers.Control.ZoomPanel = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.addControls([new OpenLayers.Control.ZoomIn, new OpenLayers.Control.ZoomToMaxExtent, new OpenLayers.Control.ZoomOut]) │ │ │ │ │ }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (!this.element) { │ │ │ │ │ - this.element = document.createElement("div"); │ │ │ │ │ - this.div.appendChild(this.element) │ │ │ │ │ - } │ │ │ │ │ - this.map.events.register("moveend", this, this.updateScale); │ │ │ │ │ - this.updateScale(); │ │ │ │ │ - return this.div │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomPanel" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Tile.UTFGrid = OpenLayers.Class(OpenLayers.Tile, { │ │ │ │ │ + url: null, │ │ │ │ │ + utfgridResolution: 2, │ │ │ │ │ + json: null, │ │ │ │ │ + format: null, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.clear(); │ │ │ │ │ + OpenLayers.Tile.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - updateScale: function() { │ │ │ │ │ - var scale; │ │ │ │ │ - if (this.geodesic === true) { │ │ │ │ │ - var units = this.map.getUnits(); │ │ │ │ │ - if (!units) { │ │ │ │ │ - return │ │ │ │ │ + draw: function() { │ │ │ │ │ + var drawn = OpenLayers.Tile.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (drawn) { │ │ │ │ │ + if (this.isLoading) { │ │ │ │ │ + this.abortLoading(); │ │ │ │ │ + this.events.triggerEvent("reload") │ │ │ │ │ + } else { │ │ │ │ │ + this.isLoading = true; │ │ │ │ │ + this.events.triggerEvent("loadstart") │ │ │ │ │ + } │ │ │ │ │ + this.url = this.layer.getURL(this.bounds); │ │ │ │ │ + if (this.layer.useJSONP) { │ │ │ │ │ + var ols = new OpenLayers.Protocol.Script({ │ │ │ │ │ + url: this.url, │ │ │ │ │ + callback: function(response) { │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + this.json = response.data │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + ols.read(); │ │ │ │ │ + this.request = ols │ │ │ │ │ + } else { │ │ │ │ │ + this.request = OpenLayers.Request.GET({ │ │ │ │ │ + url: this.url, │ │ │ │ │ + callback: function(response) { │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + if (response.status === 200) { │ │ │ │ │ + this.parseData(response.responseText) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - var inches = OpenLayers.INCHES_PER_UNIT; │ │ │ │ │ - scale = (this.map.getGeodesicPixelSize().w || 1e-6) * inches["km"] * OpenLayers.DOTS_PER_INCH │ │ │ │ │ } else { │ │ │ │ │ - scale = this.map.getScale() │ │ │ │ │ - } │ │ │ │ │ - if (!scale) { │ │ │ │ │ - return │ │ │ │ │ + this.unload() │ │ │ │ │ } │ │ │ │ │ - if (scale >= 9500 && scale <= 95e4) { │ │ │ │ │ - scale = Math.round(scale / 1e3) + "K" │ │ │ │ │ - } else if (scale >= 95e4) { │ │ │ │ │ - scale = Math.round(scale / 1e6) + "M" │ │ │ │ │ - } else { │ │ │ │ │ - scale = Math.round(scale) │ │ │ │ │ + return drawn │ │ │ │ │ + }, │ │ │ │ │ + abortLoading: function() { │ │ │ │ │ + if (this.request) { │ │ │ │ │ + this.request.abort(); │ │ │ │ │ + delete this.request │ │ │ │ │ } │ │ │ │ │ - this.element.innerHTML = OpenLayers.i18n("Scale = 1 : ${scaleDenom}", { │ │ │ │ │ - scaleDenom: scale │ │ │ │ │ - }) │ │ │ │ │ + this.isLoading = false │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Scale" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.SLDSelect = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - clearOnDeactivate: false, │ │ │ │ │ - layers: null, │ │ │ │ │ - callbacks: null, │ │ │ │ │ - selectionSymbolizer: { │ │ │ │ │ - Polygon: { │ │ │ │ │ - fillColor: "#FF0000", │ │ │ │ │ - stroke: false │ │ │ │ │ - }, │ │ │ │ │ - Line: { │ │ │ │ │ - strokeColor: "#FF0000", │ │ │ │ │ - strokeWidth: 2 │ │ │ │ │ - }, │ │ │ │ │ - Point: { │ │ │ │ │ - graphicName: "square", │ │ │ │ │ - fillColor: "#FF0000", │ │ │ │ │ - pointRadius: 5 │ │ │ │ │ + getFeatureInfo: function(i, j) { │ │ │ │ │ + var info = null; │ │ │ │ │ + if (this.json) { │ │ │ │ │ + var id = this.getFeatureId(i, j); │ │ │ │ │ + if (id !== null) { │ │ │ │ │ + info = { │ │ │ │ │ + id: id, │ │ │ │ │ + data: this.json.data[id] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return info │ │ │ │ │ }, │ │ │ │ │ - layerOptions: null, │ │ │ │ │ - sketchStyle: null, │ │ │ │ │ - wfsCache: {}, │ │ │ │ │ - layerCache: {}, │ │ │ │ │ - initialize: function(handler, options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.callbacks = OpenLayers.Util.extend({ │ │ │ │ │ - done: this.select, │ │ │ │ │ - click: this.select │ │ │ │ │ - }, this.callbacks); │ │ │ │ │ - this.handlerOptions = this.handlerOptions || {}; │ │ │ │ │ - this.layerOptions = OpenLayers.Util.applyDefaults(this.layerOptions, { │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - tileOptions: { │ │ │ │ │ - maxGetUrlLength: 2048 │ │ │ │ │ + getFeatureId: function(i, j) { │ │ │ │ │ + var id = null; │ │ │ │ │ + if (this.json) { │ │ │ │ │ + var resolution = this.utfgridResolution; │ │ │ │ │ + var row = Math.floor(j / resolution); │ │ │ │ │ + var col = Math.floor(i / resolution); │ │ │ │ │ + var charCode = this.json.grid[row].charCodeAt(col); │ │ │ │ │ + var index = this.indexFromCharCode(charCode); │ │ │ │ │ + var keys = this.json.keys; │ │ │ │ │ + if (!isNaN(index) && index in keys) { │ │ │ │ │ + id = keys[index] │ │ │ │ │ } │ │ │ │ │ - }); │ │ │ │ │ - if (this.sketchStyle) { │ │ │ │ │ - this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults(this.handlerOptions.layerOptions, { │ │ │ │ │ - styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ - default: this.sketchStyle │ │ │ │ │ - }) │ │ │ │ │ - }) │ │ │ │ │ } │ │ │ │ │ - this.handler = new handler(this, this.callbacks, this.handlerOptions) │ │ │ │ │ + return id │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var key in this.layerCache) { │ │ │ │ │ - delete this.layerCache[key] │ │ │ │ │ + indexFromCharCode: function(charCode) { │ │ │ │ │ + if (charCode >= 93) { │ │ │ │ │ + charCode-- │ │ │ │ │ } │ │ │ │ │ - for (var key in this.wfsCache) { │ │ │ │ │ - delete this.wfsCache[key] │ │ │ │ │ + if (charCode >= 35) { │ │ │ │ │ + charCode-- │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - coupleLayerVisiblity: function(evt) { │ │ │ │ │ - this.setVisibility(evt.object.getVisibility()) │ │ │ │ │ + return charCode - 32 │ │ │ │ │ }, │ │ │ │ │ - createSelectionLayer: function(source) { │ │ │ │ │ - var selectionLayer; │ │ │ │ │ - if (!this.layerCache[source.id]) { │ │ │ │ │ - selectionLayer = new OpenLayers.Layer.WMS(source.name, source.url, source.params, OpenLayers.Util.applyDefaults(this.layerOptions, source.getOptions())); │ │ │ │ │ - this.layerCache[source.id] = selectionLayer; │ │ │ │ │ - if (this.layerOptions.displayInLayerSwitcher === false) { │ │ │ │ │ - source.events.on({ │ │ │ │ │ - visibilitychanged: this.coupleLayerVisiblity, │ │ │ │ │ - scope: selectionLayer │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.map.addLayer(selectionLayer) │ │ │ │ │ - } else { │ │ │ │ │ - selectionLayer = this.layerCache[source.id] │ │ │ │ │ + parseData: function(str) { │ │ │ │ │ + if (!this.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.JSON │ │ │ │ │ } │ │ │ │ │ - return selectionLayer │ │ │ │ │ + this.json = this.format.read(str) │ │ │ │ │ }, │ │ │ │ │ - createSLD: function(layer, filters, geometryAttributes) { │ │ │ │ │ - var sld = { │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - namedLayers: {} │ │ │ │ │ - }; │ │ │ │ │ - var layerNames = [layer.params.LAYERS].join(",").split(","); │ │ │ │ │ - for (var i = 0, len = layerNames.length; i < len; i++) { │ │ │ │ │ - var name = layerNames[i]; │ │ │ │ │ - sld.namedLayers[name] = { │ │ │ │ │ - name: name, │ │ │ │ │ - userStyles: [] │ │ │ │ │ - }; │ │ │ │ │ - var symbolizer = this.selectionSymbolizer; │ │ │ │ │ - var geometryAttribute = geometryAttributes[i]; │ │ │ │ │ - if (geometryAttribute.type.indexOf("Polygon") >= 0) { │ │ │ │ │ - symbolizer = { │ │ │ │ │ - Polygon: this.selectionSymbolizer["Polygon"] │ │ │ │ │ - } │ │ │ │ │ - } else if (geometryAttribute.type.indexOf("LineString") >= 0) { │ │ │ │ │ - symbolizer = { │ │ │ │ │ - Line: this.selectionSymbolizer["Line"] │ │ │ │ │ + clear: function() { │ │ │ │ │ + this.json = null │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Tile.UTFGrid" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Tile.Image.IFrame = { │ │ │ │ │ + useIFrame: null, │ │ │ │ │ + blankImageUrl: "data:image/gif;base64,R0lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAQAIBRAA7", │ │ │ │ │ + draw: function() { │ │ │ │ │ + var draw = OpenLayers.Tile.Image.prototype.shouldDraw.call(this); │ │ │ │ │ + if (draw) { │ │ │ │ │ + var url = this.layer.getURL(this.bounds); │ │ │ │ │ + var usedIFrame = this.useIFrame; │ │ │ │ │ + this.useIFrame = this.maxGetUrlLength !== null && !this.layer.async && url.length > this.maxGetUrlLength; │ │ │ │ │ + var fromIFrame = usedIFrame && !this.useIFrame; │ │ │ │ │ + var toIFrame = !usedIFrame && this.useIFrame; │ │ │ │ │ + if (fromIFrame || toIFrame) { │ │ │ │ │ + if (this.imgDiv && this.imgDiv.parentNode === this.frame) { │ │ │ │ │ + this.frame.removeChild(this.imgDiv) │ │ │ │ │ } │ │ │ │ │ - } else if (geometryAttribute.type.indexOf("Point") >= 0) { │ │ │ │ │ - symbolizer = { │ │ │ │ │ - Point: this.selectionSymbolizer["Point"] │ │ │ │ │ + this.imgDiv = null; │ │ │ │ │ + if (fromIFrame) { │ │ │ │ │ + this.frame.removeChild(this.frame.firstChild) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var filter = filters[i]; │ │ │ │ │ - sld.namedLayers[name].userStyles.push({ │ │ │ │ │ - name: "default", │ │ │ │ │ - rules: [new OpenLayers.Rule({ │ │ │ │ │ - symbolizer: symbolizer, │ │ │ │ │ - filter: filter, │ │ │ │ │ - maxScaleDenominator: layer.options.minScale │ │ │ │ │ - })] │ │ │ │ │ - }) │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Format.SLD({ │ │ │ │ │ - srsName: this.map.getProjection() │ │ │ │ │ - }).write(sld) │ │ │ │ │ + return OpenLayers.Tile.Image.prototype.draw.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - parseDescribeLayer: function(request) { │ │ │ │ │ - var format = new OpenLayers.Format.WMSDescribeLayer; │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ - } │ │ │ │ │ - var describeLayer = format.read(doc); │ │ │ │ │ - var typeNames = []; │ │ │ │ │ - var url = null; │ │ │ │ │ - for (var i = 0, len = describeLayer.length; i < len; i++) { │ │ │ │ │ - if (describeLayer[i].owsType == "WFS") { │ │ │ │ │ - typeNames.push(describeLayer[i].typeName); │ │ │ │ │ - url = describeLayer[i].owsURL │ │ │ │ │ + getImage: function() { │ │ │ │ │ + if (this.useIFrame === true) { │ │ │ │ │ + if (!this.frame.childNodes.length) { │ │ │ │ │ + var eventPane = document.createElement("div"), │ │ │ │ │ + style = eventPane.style; │ │ │ │ │ + style.position = "absolute"; │ │ │ │ │ + style.width = "100%"; │ │ │ │ │ + style.height = "100%"; │ │ │ │ │ + style.zIndex = 1; │ │ │ │ │ + style.backgroundImage = "url(" + this.blankImageUrl + ")"; │ │ │ │ │ + this.frame.appendChild(eventPane) │ │ │ │ │ + } │ │ │ │ │ + var id = this.id + "_iFrame", │ │ │ │ │ + iframe; │ │ │ │ │ + if (parseFloat(navigator.appVersion.split("MSIE")[1]) < 9) { │ │ │ │ │ + iframe = document.createElement('<iframe name="' + id + '">'); │ │ │ │ │ + iframe.style.backgroundColor = "#FFFFFF"; │ │ │ │ │ + iframe.style.filter = "chroma(color=#FFFFFF)" │ │ │ │ │ + } else { │ │ │ │ │ + iframe = document.createElement("iframe"); │ │ │ │ │ + iframe.style.backgroundColor = "transparent"; │ │ │ │ │ + iframe.name = id │ │ │ │ │ + } │ │ │ │ │ + iframe.scrolling = "no"; │ │ │ │ │ + iframe.marginWidth = "0px"; │ │ │ │ │ + iframe.marginHeight = "0px"; │ │ │ │ │ + iframe.frameBorder = "0"; │ │ │ │ │ + iframe.style.position = "absolute"; │ │ │ │ │ + iframe.style.width = "100%"; │ │ │ │ │ + iframe.style.height = "100%"; │ │ │ │ │ + if (this.layer.opacity < 1) { │ │ │ │ │ + OpenLayers.Util.modifyDOMElement(iframe, null, null, null, null, null, null, this.layer.opacity) │ │ │ │ │ } │ │ │ │ │ + this.frame.appendChild(iframe); │ │ │ │ │ + this.imgDiv = iframe; │ │ │ │ │ + return iframe │ │ │ │ │ + } else { │ │ │ │ │ + return OpenLayers.Tile.Image.prototype.getImage.apply(this, arguments) │ │ │ │ │ } │ │ │ │ │ - var options = { │ │ │ │ │ - url: url, │ │ │ │ │ - params: { │ │ │ │ │ - SERVICE: "WFS", │ │ │ │ │ - TYPENAME: typeNames.toString(), │ │ │ │ │ - REQUEST: "DescribeFeatureType", │ │ │ │ │ - VERSION: "1.0.0" │ │ │ │ │ - }, │ │ │ │ │ - callback: function(request) { │ │ │ │ │ - var format = new OpenLayers.Format.WFSDescribeFeatureType; │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ - } │ │ │ │ │ - var describeFeatureType = format.read(doc); │ │ │ │ │ - this.control.wfsCache[this.layer.id] = describeFeatureType; │ │ │ │ │ - this.control._queue && this.control.applySelection() │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Request.GET(options) │ │ │ │ │ }, │ │ │ │ │ - getGeometryAttributes: function(layer) { │ │ │ │ │ - var result = []; │ │ │ │ │ - var cache = this.wfsCache[layer.id]; │ │ │ │ │ - for (var i = 0, len = cache.featureTypes.length; i < len; i++) { │ │ │ │ │ - var typeName = cache.featureTypes[i]; │ │ │ │ │ - var properties = typeName.properties; │ │ │ │ │ - for (var j = 0, lenj = properties.length; j < lenj; j++) { │ │ │ │ │ - var property = properties[j]; │ │ │ │ │ - var type = property.type; │ │ │ │ │ - if (type.indexOf("LineString") >= 0 || type.indexOf("GeometryAssociationType") >= 0 || type.indexOf("GeometryPropertyType") >= 0 || type.indexOf("Point") >= 0 || type.indexOf("Polygon") >= 0) { │ │ │ │ │ - result.push(property) │ │ │ │ │ - } │ │ │ │ │ + createRequestForm: function() { │ │ │ │ │ + var form = document.createElement("form"); │ │ │ │ │ + form.method = "POST"; │ │ │ │ │ + var cacheId = this.layer.params["_OLSALT"]; │ │ │ │ │ + cacheId = (cacheId ? cacheId + "_" : "") + this.bounds.toBBOX(); │ │ │ │ │ + form.action = OpenLayers.Util.urlAppend(this.layer.url, cacheId); │ │ │ │ │ + form.target = this.id + "_iFrame"; │ │ │ │ │ + var imageSize = this.layer.getImageSize(), │ │ │ │ │ + params = OpenLayers.Util.getParameters(this.url), │ │ │ │ │ + field; │ │ │ │ │ + for (var par in params) { │ │ │ │ │ + field = document.createElement("input"); │ │ │ │ │ + field.type = "hidden"; │ │ │ │ │ + field.name = par; │ │ │ │ │ + field.value = params[par]; │ │ │ │ │ + form.appendChild(field) │ │ │ │ │ + } │ │ │ │ │ + return form │ │ │ │ │ + }, │ │ │ │ │ + setImgSrc: function(url) { │ │ │ │ │ + if (this.useIFrame === true) { │ │ │ │ │ + if (url) { │ │ │ │ │ + var form = this.createRequestForm(); │ │ │ │ │ + this.frame.appendChild(form); │ │ │ │ │ + form.submit(); │ │ │ │ │ + this.frame.removeChild(form) │ │ │ │ │ + } else if (this.imgDiv.parentNode === this.frame) { │ │ │ │ │ + this.frame.removeChild(this.imgDiv); │ │ │ │ │ + this.imgDiv = null │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Tile.Image.prototype.setImgSrc.apply(this, arguments) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + onImageLoad: function() { │ │ │ │ │ + OpenLayers.Tile.Image.prototype.onImageLoad.apply(this, arguments); │ │ │ │ │ + if (this.useIFrame === true) { │ │ │ │ │ + this.imgDiv.style.opacity = 1; │ │ │ │ │ + this.frame.style.opacity = this.layer.opacity │ │ │ │ │ } │ │ │ │ │ - return result │ │ │ │ │ }, │ │ │ │ │ + createBackBuffer: function() { │ │ │ │ │ + var backBuffer; │ │ │ │ │ + if (this.useIFrame === false) { │ │ │ │ │ + backBuffer = OpenLayers.Tile.Image.prototype.createBackBuffer.call(this) │ │ │ │ │ + } │ │ │ │ │ + return backBuffer │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Strategy.BBOX = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + bounds: null, │ │ │ │ │ + resolution: null, │ │ │ │ │ + ratio: 2, │ │ │ │ │ + resFactor: null, │ │ │ │ │ + response: null, │ │ │ │ │ activate: function() { │ │ │ │ │ - var activated = OpenLayers.Control.prototype.activate.call(this); │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ if (activated) { │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.layers[i]; │ │ │ │ │ - if (layer && !this.wfsCache[layer.id]) { │ │ │ │ │ - var options = { │ │ │ │ │ - url: layer.url, │ │ │ │ │ - params: { │ │ │ │ │ - SERVICE: "WMS", │ │ │ │ │ - VERSION: layer.params.VERSION, │ │ │ │ │ - LAYERS: layer.params.LAYERS, │ │ │ │ │ - REQUEST: "DescribeLayer" │ │ │ │ │ - }, │ │ │ │ │ - callback: this.parseDescribeLayer, │ │ │ │ │ - scope: { │ │ │ │ │ - layer: layer, │ │ │ │ │ - control: this │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Request.GET(options) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + moveend: this.update, │ │ │ │ │ + refresh: this.update, │ │ │ │ │ + visibilitychanged: this.update, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.update() │ │ │ │ │ } │ │ │ │ │ return activated │ │ │ │ │ }, │ │ │ │ │ deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Control.prototype.deactivate.call(this); │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ if (deactivated) { │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.layers[i]; │ │ │ │ │ - if (layer && this.clearOnDeactivate === true) { │ │ │ │ │ - var layerCache = this.layerCache; │ │ │ │ │ - var selectionLayer = layerCache[layer.id]; │ │ │ │ │ - if (selectionLayer) { │ │ │ │ │ - layer.events.un({ │ │ │ │ │ - visibilitychanged: this.coupleLayerVisiblity, │ │ │ │ │ - scope: selectionLayer │ │ │ │ │ - }); │ │ │ │ │ - selectionLayer.destroy(); │ │ │ │ │ - delete layerCache[layer.id] │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + moveend: this.update, │ │ │ │ │ + refresh: this.update, │ │ │ │ │ + visibilitychanged: this.update, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ return deactivated │ │ │ │ │ }, │ │ │ │ │ - setLayers: function(layers) { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.layers = layers; │ │ │ │ │ - this.activate() │ │ │ │ │ - } else { │ │ │ │ │ - this.layers = layers │ │ │ │ │ + update: function(options) { │ │ │ │ │ + var mapBounds = this.getMapBounds(); │ │ │ │ │ + if (mapBounds !== null && (options && options.force || this.layer.visibility && this.layer.calculateInRange() && this.invalidBounds(mapBounds))) { │ │ │ │ │ + this.calculateBounds(mapBounds); │ │ │ │ │ + this.resolution = this.layer.map.getResolution(); │ │ │ │ │ + this.triggerRead(options) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - createFilter: function(geometryAttribute, geometry) { │ │ │ │ │ - var filter = null; │ │ │ │ │ - if (this.handler instanceof OpenLayers.Handler.RegularPolygon) { │ │ │ │ │ - if (this.handler.irregular === true) { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry.getBounds() │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } else if (this.handler instanceof OpenLayers.Handler.Polygon) { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry │ │ │ │ │ - }) │ │ │ │ │ - } else if (this.handler instanceof OpenLayers.Handler.Path) { │ │ │ │ │ - if (geometryAttribute.type.indexOf("Point") >= 0) { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - distance: this.map.getExtent().getWidth() * .01, │ │ │ │ │ - distanceUnits: this.map.getUnits(), │ │ │ │ │ - value: geometry │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } else if (this.handler instanceof OpenLayers.Handler.Click) { │ │ │ │ │ - if (geometryAttribute.type.indexOf("Polygon") >= 0) { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - distance: this.map.getExtent().getWidth() * .01, │ │ │ │ │ - distanceUnits: this.map.getUnits(), │ │ │ │ │ - value: geometry │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ + getMapBounds: function() { │ │ │ │ │ + if (this.layer.map === null) { │ │ │ │ │ + return null │ │ │ │ │ } │ │ │ │ │ - return filter │ │ │ │ │ - }, │ │ │ │ │ - select: function(geometry) { │ │ │ │ │ - this._queue = function() { │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.layers[i]; │ │ │ │ │ - var geometryAttributes = this.getGeometryAttributes(layer); │ │ │ │ │ - var filters = []; │ │ │ │ │ - for (var j = 0, lenj = geometryAttributes.length; j < lenj; j++) { │ │ │ │ │ - var geometryAttribute = geometryAttributes[j]; │ │ │ │ │ - if (geometryAttribute !== null) { │ │ │ │ │ - if (!(geometry instanceof OpenLayers.Geometry)) { │ │ │ │ │ - var point = this.map.getLonLatFromPixel(geometry.xy); │ │ │ │ │ - geometry = new OpenLayers.Geometry.Point(point.lon, point.lat) │ │ │ │ │ - } │ │ │ │ │ - var filter = this.createFilter(geometryAttribute, geometry); │ │ │ │ │ - if (filter !== null) { │ │ │ │ │ - filters.push(filter) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var selectionLayer = this.createSelectionLayer(layer); │ │ │ │ │ - this.events.triggerEvent("selected", { │ │ │ │ │ - layer: layer, │ │ │ │ │ - filters: filters │ │ │ │ │ - }); │ │ │ │ │ - var sld = this.createSLD(layer, filters, geometryAttributes); │ │ │ │ │ - selectionLayer.mergeNewParams({ │ │ │ │ │ - SLD_BODY: sld │ │ │ │ │ - }); │ │ │ │ │ - delete this._queue │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - this.applySelection() │ │ │ │ │ + var bounds = this.layer.map.getExtent(); │ │ │ │ │ + if (bounds && !this.layer.projection.equals(this.layer.map.getProjectionObject())) { │ │ │ │ │ + bounds = bounds.clone().transform(this.layer.map.getProjectionObject(), this.layer.projection) │ │ │ │ │ + } │ │ │ │ │ + return bounds │ │ │ │ │ }, │ │ │ │ │ - applySelection: function() { │ │ │ │ │ - var canApply = true; │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - if (!this.wfsCache[this.layers[i].id]) { │ │ │ │ │ - canApply = false; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ + invalidBounds: function(mapBounds) { │ │ │ │ │ + if (!mapBounds) { │ │ │ │ │ + mapBounds = this.getMapBounds() │ │ │ │ │ } │ │ │ │ │ - canApply && this._queue.call(this) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.SLDSelect" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Renderer.Canvas = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ - hitDetection: true, │ │ │ │ │ - hitOverflow: 0, │ │ │ │ │ - canvas: null, │ │ │ │ │ - features: null, │ │ │ │ │ - pendingRedraw: false, │ │ │ │ │ - cachedSymbolBounds: {}, │ │ │ │ │ - initialize: function(containerID, options) { │ │ │ │ │ - OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.root = document.createElement("canvas"); │ │ │ │ │ - this.container.appendChild(this.root); │ │ │ │ │ - this.canvas = this.root.getContext("2d"); │ │ │ │ │ - this.features = {}; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitCanvas = document.createElement("canvas"); │ │ │ │ │ - this.hitContext = this.hitCanvas.getContext("2d") │ │ │ │ │ + var invalid = !this.bounds || !this.bounds.containsBounds(mapBounds); │ │ │ │ │ + if (!invalid && this.resFactor) { │ │ │ │ │ + var ratio = this.resolution / this.layer.map.getResolution(); │ │ │ │ │ + invalid = ratio >= this.resFactor || ratio <= 1 / this.resFactor │ │ │ │ │ } │ │ │ │ │ + return invalid │ │ │ │ │ }, │ │ │ │ │ - setExtent: function() { │ │ │ │ │ - OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - eraseGeometry: function(geometry, featureId) { │ │ │ │ │ - this.eraseFeatures(this.features[featureId][0]) │ │ │ │ │ - }, │ │ │ │ │ - supported: function() { │ │ │ │ │ - return OpenLayers.CANVAS_SUPPORTED │ │ │ │ │ - }, │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - this.size = size.clone(); │ │ │ │ │ - var root = this.root; │ │ │ │ │ - root.style.width = size.w + "px"; │ │ │ │ │ - root.style.height = size.h + "px"; │ │ │ │ │ - root.width = size.w; │ │ │ │ │ - root.height = size.h; │ │ │ │ │ - this.resolution = null; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - var hitCanvas = this.hitCanvas; │ │ │ │ │ - hitCanvas.style.width = size.w + "px"; │ │ │ │ │ - hitCanvas.style.height = size.h + "px"; │ │ │ │ │ - hitCanvas.width = size.w; │ │ │ │ │ - hitCanvas.height = size.h │ │ │ │ │ + calculateBounds: function(mapBounds) { │ │ │ │ │ + if (!mapBounds) { │ │ │ │ │ + mapBounds = this.getMapBounds() │ │ │ │ │ } │ │ │ │ │ + var center = mapBounds.getCenterLonLat(); │ │ │ │ │ + var dataWidth = mapBounds.getWidth() * this.ratio; │ │ │ │ │ + var dataHeight = mapBounds.getHeight() * this.ratio; │ │ │ │ │ + this.bounds = new OpenLayers.Bounds(center.lon - dataWidth / 2, center.lat - dataHeight / 2, center.lon + dataWidth / 2, center.lat + dataHeight / 2) │ │ │ │ │ }, │ │ │ │ │ - drawFeature: function(feature, style) { │ │ │ │ │ - var rendered; │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - style = this.applyDefaultSymbolizer(style || feature.style); │ │ │ │ │ - var bounds = feature.geometry.getBounds(); │ │ │ │ │ - var worldBounds; │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - worldBounds = this.map.getMaxExtent() │ │ │ │ │ - } │ │ │ │ │ - var intersects = bounds && bounds.intersectsBounds(this.extent, { │ │ │ │ │ - worldBounds: worldBounds │ │ │ │ │ - }); │ │ │ │ │ - rendered = style.display !== "none" && !!bounds && intersects; │ │ │ │ │ - if (rendered) { │ │ │ │ │ - this.features[feature.id] = [feature, style] │ │ │ │ │ - } else { │ │ │ │ │ - delete this.features[feature.id] │ │ │ │ │ - } │ │ │ │ │ - this.pendingRedraw = true │ │ │ │ │ + triggerRead: function(options) { │ │ │ │ │ + if (this.response && !(options && options.noAbort === true)) { │ │ │ │ │ + this.layer.protocol.abort(this.response); │ │ │ │ │ + this.layer.events.triggerEvent("loadend") │ │ │ │ │ } │ │ │ │ │ - if (this.pendingRedraw && !this.locked) { │ │ │ │ │ - this.redraw(); │ │ │ │ │ - this.pendingRedraw = false │ │ │ │ │ + var evt = { │ │ │ │ │ + filter: this.createFilter() │ │ │ │ │ + }; │ │ │ │ │ + this.layer.events.triggerEvent("loadstart", evt); │ │ │ │ │ + this.response = this.layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ + filter: evt.filter, │ │ │ │ │ + callback: this.merge, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options)) │ │ │ │ │ + }, │ │ │ │ │ + createFilter: function() { │ │ │ │ │ + var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ + value: this.bounds, │ │ │ │ │ + projection: this.layer.projection │ │ │ │ │ + }); │ │ │ │ │ + if (this.layer.filter) { │ │ │ │ │ + filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ + type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ + filters: [this.layer.filter, filter] │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - return rendered │ │ │ │ │ + return filter │ │ │ │ │ }, │ │ │ │ │ - drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ - var className = geometry.CLASS_NAME; │ │ │ │ │ - if (className == "OpenLayers.Geometry.Collection" || className == "OpenLayers.Geometry.MultiPoint" || className == "OpenLayers.Geometry.MultiLineString" || className == "OpenLayers.Geometry.MultiPolygon") { │ │ │ │ │ - for (var i = 0; i < geometry.components.length; i++) { │ │ │ │ │ - this.drawGeometry(geometry.components[i], style, featureId) │ │ │ │ │ + merge: function(resp) { │ │ │ │ │ + this.layer.destroyFeatures(); │ │ │ │ │ + if (resp.success()) { │ │ │ │ │ + var features = resp.features; │ │ │ │ │ + if (features && features.length > 0) { │ │ │ │ │ + var remote = this.layer.projection; │ │ │ │ │ + var local = this.layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var geom; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + geom = features[i].geometry; │ │ │ │ │ + if (geom) { │ │ │ │ │ + geom.transform(remote, local) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.layer.addFeatures(features) │ │ │ │ │ } │ │ │ │ │ - return │ │ │ │ │ + } else { │ │ │ │ │ + this.bounds = null │ │ │ │ │ } │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - this.drawPoint(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - this.drawLineString(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - this.drawPolygon(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break │ │ │ │ │ + this.response = null; │ │ │ │ │ + this.layer.events.triggerEvent("loadend", { │ │ │ │ │ + response: resp │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.BBOX" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Strategy.Filter = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + filter: null, │ │ │ │ │ + cache: null, │ │ │ │ │ + caching: false, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.cache = []; │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + beforefeaturesadded: this.handleAdd, │ │ │ │ │ + beforefeaturesremoved: this.handleRemove, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - drawExternalGraphic: function(geometry, style, featureId) { │ │ │ │ │ - var img = new Image; │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - img.title = title │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + this.cache = null; │ │ │ │ │ + if (this.layer && this.layer.events) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + beforefeaturesadded: this.handleAdd, │ │ │ │ │ + beforefeaturesremoved: this.handleRemove, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ - var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ - var onLoad = function() { │ │ │ │ │ - if (!this.features[featureId]) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ - if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ - var x = p0 + xOffset | 0; │ │ │ │ │ - var y = p1 + yOffset | 0; │ │ │ │ │ - var canvas = this.canvas; │ │ │ │ │ - canvas.globalAlpha = opacity; │ │ │ │ │ - var factor = OpenLayers.Renderer.Canvas.drawImageScaleFactor || (OpenLayers.Renderer.Canvas.drawImageScaleFactor = /android 2.1/.test(navigator.userAgent.toLowerCase()) ? 320 / window.screen.width : 1); │ │ │ │ │ - canvas.drawImage(img, x * factor, y * factor, width * factor, height * factor); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId); │ │ │ │ │ - this.hitContext.fillRect(x, y, width, height) │ │ │ │ │ + return OpenLayers.Strategy.prototype.deactivate.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + handleAdd: function(event) { │ │ │ │ │ + if (!this.caching && this.filter) { │ │ │ │ │ + var features = event.features; │ │ │ │ │ + event.features = []; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0, ii = features.length; i < ii; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (this.filter.evaluate(feature)) { │ │ │ │ │ + event.features.push(feature) │ │ │ │ │ + } else { │ │ │ │ │ + this.cache.push(feature) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }; │ │ │ │ │ - img.onload = OpenLayers.Function.bind(onLoad, this); │ │ │ │ │ - img.src = style.externalGraphic │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - drawNamedSymbol: function(geometry, style, featureId) { │ │ │ │ │ - var x, y, cx, cy, i, symbolBounds, scaling, angle; │ │ │ │ │ - var unscaledStrokeWidth; │ │ │ │ │ - var deg2rad = Math.PI / 180; │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[style.graphicName]; │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(style.graphicName + " is not a valid symbol name") │ │ │ │ │ + handleRemove: function(event) { │ │ │ │ │ + if (!this.caching) { │ │ │ │ │ + this.cache = [] │ │ │ │ │ } │ │ │ │ │ - if (!symbol.length || symbol.length < 2) return; │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ - if (isNaN(p0) || isNaN(p1)) return; │ │ │ │ │ - this.canvas.lineCap = "round"; │ │ │ │ │ - this.canvas.lineJoin = "round"; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.lineCap = "round"; │ │ │ │ │ - this.hitContext.lineJoin = "round" │ │ │ │ │ + }, │ │ │ │ │ + setFilter: function(filter) { │ │ │ │ │ + this.filter = filter; │ │ │ │ │ + var previousCache = this.cache; │ │ │ │ │ + this.cache = []; │ │ │ │ │ + this.handleAdd({ │ │ │ │ │ + features: this.layer.features │ │ │ │ │ + }); │ │ │ │ │ + if (this.cache.length > 0) { │ │ │ │ │ + this.caching = true; │ │ │ │ │ + this.layer.removeFeatures(this.cache.slice()); │ │ │ │ │ + this.caching = false │ │ │ │ │ } │ │ │ │ │ - if (style.graphicName in this.cachedSymbolBounds) { │ │ │ │ │ - symbolBounds = this.cachedSymbolBounds[style.graphicName] │ │ │ │ │ - } else { │ │ │ │ │ - symbolBounds = new OpenLayers.Bounds; │ │ │ │ │ - for (i = 0; i < symbol.length; i += 2) { │ │ │ │ │ - symbolBounds.extend(new OpenLayers.LonLat(symbol[i], symbol[i + 1])) │ │ │ │ │ + if (previousCache.length > 0) { │ │ │ │ │ + var event = { │ │ │ │ │ + features: previousCache │ │ │ │ │ + }; │ │ │ │ │ + this.handleAdd(event); │ │ │ │ │ + if (event.features.length > 0) { │ │ │ │ │ + this.caching = true; │ │ │ │ │ + this.layer.addFeatures(event.features); │ │ │ │ │ + this.caching = false │ │ │ │ │ } │ │ │ │ │ - this.cachedSymbolBounds[style.graphicName] = symbolBounds │ │ │ │ │ - } │ │ │ │ │ - this.canvas.save(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.save() │ │ │ │ │ } │ │ │ │ │ - this.canvas.translate(p0, p1); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.translate(p0, p1) │ │ │ │ │ - } │ │ │ │ │ - angle = deg2rad * style.rotation; │ │ │ │ │ - if (!isNaN(angle)) { │ │ │ │ │ - this.canvas.rotate(angle); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.rotate(angle) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Filter" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Strategy.Refresh = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + force: false, │ │ │ │ │ + interval: 0, │ │ │ │ │ + timer: null, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + if (this.layer.visibility === true) { │ │ │ │ │ + this.start() │ │ │ │ │ } │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + visibilitychanged: this.reset, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - scaling = 2 * style.pointRadius / Math.max(symbolBounds.getWidth(), symbolBounds.getHeight()); │ │ │ │ │ - this.canvas.scale(scaling, scaling); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.scale(scaling, scaling) │ │ │ │ │ + return activated │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.stop(); │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + visibilitychanged: this.reset, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - cx = symbolBounds.getCenterLonLat().lon; │ │ │ │ │ - cy = symbolBounds.getCenterLonLat().lat; │ │ │ │ │ - this.canvas.translate(-cx, -cy); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.translate(-cx, -cy) │ │ │ │ │ + return deactivated │ │ │ │ │ + }, │ │ │ │ │ + reset: function() { │ │ │ │ │ + if (this.layer.visibility === true) { │ │ │ │ │ + this.start() │ │ │ │ │ + } else { │ │ │ │ │ + this.stop() │ │ │ │ │ } │ │ │ │ │ - unscaledStrokeWidth = style.strokeWidth; │ │ │ │ │ - style.strokeWidth = unscaledStrokeWidth / scaling; │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.canvas.lineTo(x, y) │ │ │ │ │ - } │ │ │ │ │ - this.canvas.closePath(); │ │ │ │ │ - this.canvas.fill(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.hitContext.lineTo(x, y) │ │ │ │ │ - } │ │ │ │ │ - this.hitContext.closePath(); │ │ │ │ │ - this.hitContext.fill() │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ + start: function() { │ │ │ │ │ + if (this.interval && typeof this.interval === "number" && this.interval > 0) { │ │ │ │ │ + this.timer = window.setInterval(OpenLayers.Function.bind(this.refresh, this), this.interval) │ │ │ │ │ } │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.canvas.lineTo(x, y) │ │ │ │ │ - } │ │ │ │ │ - this.canvas.closePath(); │ │ │ │ │ - this.canvas.stroke(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style, scaling); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.hitContext.moveTo(x, y); │ │ │ │ │ - this.hitContext.lineTo(x, y) │ │ │ │ │ - } │ │ │ │ │ - this.hitContext.closePath(); │ │ │ │ │ - this.hitContext.stroke() │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ + refresh: function() { │ │ │ │ │ + if (this.layer && this.layer.refresh && typeof this.layer.refresh == "function") { │ │ │ │ │ + this.layer.refresh({ │ │ │ │ │ + force: this.force │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - style.strokeWidth = unscaledStrokeWidth; │ │ │ │ │ - this.canvas.restore(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.restore() │ │ │ │ │ + }, │ │ │ │ │ + stop: function() { │ │ │ │ │ + if (this.timer !== null) { │ │ │ │ │ + window.clearInterval(this.timer); │ │ │ │ │ + this.timer = null │ │ │ │ │ } │ │ │ │ │ - this.setCanvasStyle("reset") │ │ │ │ │ }, │ │ │ │ │ - setCanvasStyle: function(type, style) { │ │ │ │ │ - if (type === "fill") { │ │ │ │ │ - this.canvas.globalAlpha = style["fillOpacity"]; │ │ │ │ │ - this.canvas.fillStyle = style["fillColor"] │ │ │ │ │ - } else if (type === "stroke") { │ │ │ │ │ - this.canvas.globalAlpha = style["strokeOpacity"]; │ │ │ │ │ - this.canvas.strokeStyle = style["strokeColor"]; │ │ │ │ │ - this.canvas.lineWidth = style["strokeWidth"] │ │ │ │ │ - } else { │ │ │ │ │ - this.canvas.globalAlpha = 0; │ │ │ │ │ - this.canvas.lineWidth = 1 │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Refresh" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Strategy.Cluster = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + distance: 20, │ │ │ │ │ + threshold: null, │ │ │ │ │ + features: null, │ │ │ │ │ + clusters: null, │ │ │ │ │ + clustering: false, │ │ │ │ │ + resolution: null, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + beforefeaturesadded: this.cacheFeatures, │ │ │ │ │ + featuresremoved: this.clearCache, │ │ │ │ │ + moveend: this.cluster, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - featureIdToHex: function(featureId) { │ │ │ │ │ - var id = Number(featureId.split("_").pop()) + 1; │ │ │ │ │ - if (id >= 16777216) { │ │ │ │ │ - this.hitOverflow = id - 16777215; │ │ │ │ │ - id = id % 16777216 + 1 │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.clearCache(); │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + beforefeaturesadded: this.cacheFeatures, │ │ │ │ │ + featuresremoved: this.clearCache, │ │ │ │ │ + moveend: this.cluster, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - var hex = "000000" + id.toString(16); │ │ │ │ │ - var len = hex.length; │ │ │ │ │ - hex = "#" + hex.substring(len - 6, len); │ │ │ │ │ - return hex │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - setHitContextStyle: function(type, featureId, symbolizer, strokeScaling) { │ │ │ │ │ - var hex = this.featureIdToHex(featureId); │ │ │ │ │ - if (type == "fill") { │ │ │ │ │ - this.hitContext.globalAlpha = 1; │ │ │ │ │ - this.hitContext.fillStyle = hex │ │ │ │ │ - } else if (type == "stroke") { │ │ │ │ │ - this.hitContext.globalAlpha = 1; │ │ │ │ │ - this.hitContext.strokeStyle = hex; │ │ │ │ │ - if (typeof strokeScaling === "undefined") { │ │ │ │ │ - this.hitContext.lineWidth = symbolizer.strokeWidth + 2 │ │ │ │ │ - } else { │ │ │ │ │ - if (!isNaN(strokeScaling)) { │ │ │ │ │ - this.hitContext.lineWidth = symbolizer.strokeWidth + 2 / strokeScaling │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.hitContext.globalAlpha = 0; │ │ │ │ │ - this.hitContext.lineWidth = 1 │ │ │ │ │ + cacheFeatures: function(event) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + if (!this.clustering) { │ │ │ │ │ + this.clearCache(); │ │ │ │ │ + this.features = event.features; │ │ │ │ │ + this.cluster(); │ │ │ │ │ + propagate = false │ │ │ │ │ } │ │ │ │ │ + return propagate │ │ │ │ │ }, │ │ │ │ │ - drawPoint: function(geometry, style, featureId) { │ │ │ │ │ - if (style.graphic !== false) { │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - this.drawExternalGraphic(geometry, style, featureId) │ │ │ │ │ - } else if (style.graphicName && style.graphicName != "circle") { │ │ │ │ │ - this.drawNamedSymbol(geometry, style, featureId) │ │ │ │ │ - } else { │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ - if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ - var twoPi = Math.PI * 2; │ │ │ │ │ - var radius = style.pointRadius; │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.canvas.fill(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.hitContext.fill() │ │ │ │ │ + clearCache: function() { │ │ │ │ │ + if (!this.clustering) { │ │ │ │ │ + this.features = null │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + cluster: function(event) { │ │ │ │ │ + if ((!event || event.zoomChanged) && this.features) { │ │ │ │ │ + var resolution = this.layer.map.getResolution(); │ │ │ │ │ + if (resolution != this.resolution || !this.clustersExist()) { │ │ │ │ │ + this.resolution = resolution; │ │ │ │ │ + var clusters = []; │ │ │ │ │ + var feature, clustered, cluster; │ │ │ │ │ + for (var i = 0; i < this.features.length; ++i) { │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + clustered = false; │ │ │ │ │ + for (var j = clusters.length - 1; j >= 0; --j) { │ │ │ │ │ + cluster = clusters[j]; │ │ │ │ │ + if (this.shouldCluster(cluster, feature)) { │ │ │ │ │ + this.addToCluster(cluster, feature); │ │ │ │ │ + clustered = true; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!clustered) { │ │ │ │ │ + clusters.push(this.createCluster(this.features[i])) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.canvas.stroke(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.hitContext.stroke() │ │ │ │ │ + } │ │ │ │ │ + this.clustering = true; │ │ │ │ │ + this.layer.removeAllFeatures(); │ │ │ │ │ + this.clustering = false; │ │ │ │ │ + if (clusters.length > 0) { │ │ │ │ │ + if (this.threshold > 1) { │ │ │ │ │ + var clone = clusters.slice(); │ │ │ │ │ + clusters = []; │ │ │ │ │ + var candidate; │ │ │ │ │ + for (var i = 0, len = clone.length; i < len; ++i) { │ │ │ │ │ + candidate = clone[i]; │ │ │ │ │ + if (candidate.attributes.count < this.threshold) { │ │ │ │ │ + Array.prototype.push.apply(clusters, candidate.cluster) │ │ │ │ │ + } else { │ │ │ │ │ + clusters.push(candidate) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.setCanvasStyle("reset") │ │ │ │ │ } │ │ │ │ │ + this.clustering = true; │ │ │ │ │ + this.layer.addFeatures(clusters); │ │ │ │ │ + this.clustering = false │ │ │ │ │ } │ │ │ │ │ + this.clusters = clusters │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - drawLineString: function(geometry, style, featureId) { │ │ │ │ │ - style = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - fill: false │ │ │ │ │ - }, style); │ │ │ │ │ - this.drawLinearRing(geometry, style, featureId) │ │ │ │ │ - }, │ │ │ │ │ - drawLinearRing: function(geometry, style, featureId) { │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.renderPath(this.canvas, geometry, style, featureId, "fill"); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.renderPath(this.hitContext, geometry, style, featureId, "fill") │ │ │ │ │ + clustersExist: function() { │ │ │ │ │ + var exist = false; │ │ │ │ │ + if (this.clusters && this.clusters.length > 0 && this.clusters.length == this.layer.features.length) { │ │ │ │ │ + exist = true; │ │ │ │ │ + for (var i = 0; i < this.clusters.length; ++i) { │ │ │ │ │ + if (this.clusters[i] != this.layer.features[i]) { │ │ │ │ │ + exist = false; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.renderPath(this.canvas, geometry, style, featureId, "stroke"); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ - this.renderPath(this.hitContext, geometry, style, featureId, "stroke") │ │ │ │ │ + return exist │ │ │ │ │ + }, │ │ │ │ │ + shouldCluster: function(cluster, feature) { │ │ │ │ │ + var cc = cluster.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + var fc = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + var distance = Math.sqrt(Math.pow(cc.lon - fc.lon, 2) + Math.pow(cc.lat - fc.lat, 2)) / this.resolution; │ │ │ │ │ + return distance <= this.distance │ │ │ │ │ + }, │ │ │ │ │ + addToCluster: function(cluster, feature) { │ │ │ │ │ + cluster.cluster.push(feature); │ │ │ │ │ + cluster.attributes.count += 1 │ │ │ │ │ + }, │ │ │ │ │ + createCluster: function(feature) { │ │ │ │ │ + var center = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + var cluster = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Point(center.lon, center.lat), { │ │ │ │ │ + count: 1 │ │ │ │ │ + }); │ │ │ │ │ + cluster.cluster = [feature]; │ │ │ │ │ + return cluster │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Cluster" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Strategy.Save = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + events: null, │ │ │ │ │ + auto: false, │ │ │ │ │ + timer: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Strategy.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.events = new OpenLayers.Events(this) │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + if (this.auto) { │ │ │ │ │ + if (typeof this.auto === "number") { │ │ │ │ │ + this.timer = window.setInterval(OpenLayers.Function.bind(this.save, this), this.auto * 1e3) │ │ │ │ │ + } else { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + featureadded: this.triggerSave, │ │ │ │ │ + afterfeaturemodified: this.triggerSave, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.setCanvasStyle("reset") │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - renderPath: function(context, geometry, style, featureId, type) { │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - context.beginPath(); │ │ │ │ │ - var start = this.getLocalXY(components[0]); │ │ │ │ │ - var x = start[0]; │ │ │ │ │ - var y = start[1]; │ │ │ │ │ - if (!isNaN(x) && !isNaN(y)) { │ │ │ │ │ - context.moveTo(start[0], start[1]); │ │ │ │ │ - for (var i = 1; i < len; ++i) { │ │ │ │ │ - var pt = this.getLocalXY(components[i]); │ │ │ │ │ - context.lineTo(pt[0], pt[1]) │ │ │ │ │ - } │ │ │ │ │ - if (type === "fill") { │ │ │ │ │ - context.fill() │ │ │ │ │ - } else { │ │ │ │ │ - context.stroke() │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + if (this.auto) { │ │ │ │ │ + if (typeof this.auto === "number") { │ │ │ │ │ + window.clearInterval(this.timer) │ │ │ │ │ + } else { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + featureadded: this.triggerSave, │ │ │ │ │ + afterfeaturemodified: this.triggerSave, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - drawPolygon: function(geometry, style, featureId) { │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - this.drawLinearRing(components[0], style, featureId); │ │ │ │ │ - for (var i = 1; i < len; ++i) { │ │ │ │ │ - this.canvas.globalCompositeOperation = "destination-out"; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.globalCompositeOperation = "destination-out" │ │ │ │ │ - } │ │ │ │ │ - this.drawLinearRing(components[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - stroke: false, │ │ │ │ │ - fillOpacity: 1 │ │ │ │ │ - }, style), featureId); │ │ │ │ │ - this.canvas.globalCompositeOperation = "source-over"; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.globalCompositeOperation = "source-over" │ │ │ │ │ - } │ │ │ │ │ - this.drawLinearRing(components[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - fill: false │ │ │ │ │ - }, style), featureId) │ │ │ │ │ + triggerSave: function(event) { │ │ │ │ │ + var feature = event.feature; │ │ │ │ │ + if (feature.state === OpenLayers.State.INSERT || feature.state === OpenLayers.State.UPDATE || feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ + this.save([event.feature]) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - drawText: function(location, style) { │ │ │ │ │ - var pt = this.getLocalXY(location); │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ - this.canvas.fillStyle = style.fontColor; │ │ │ │ │ - this.canvas.globalAlpha = style.fontOpacity || 1; │ │ │ │ │ - var fontStyle = [style.fontStyle ? style.fontStyle : "normal", "normal", style.fontWeight ? style.fontWeight : "normal", style.fontSize ? style.fontSize : "1em", style.fontFamily ? style.fontFamily : "sans-serif"].join(" "); │ │ │ │ │ - var labelRows = style.label.split("\n"); │ │ │ │ │ - var numRows = labelRows.length; │ │ │ │ │ - if (this.canvas.fillText) { │ │ │ │ │ - this.canvas.font = fontStyle; │ │ │ │ │ - this.canvas.textAlign = OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[0]] || "center"; │ │ │ │ │ - this.canvas.textBaseline = OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[1]] || "middle"; │ │ │ │ │ - var vfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5 │ │ │ │ │ - } │ │ │ │ │ - var lineHeight = this.canvas.measureText("Mg").height || this.canvas.measureText("xx").width; │ │ │ │ │ - pt[1] += lineHeight * vfactor * (numRows - 1); │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - if (style.labelOutlineWidth) { │ │ │ │ │ - this.canvas.save(); │ │ │ │ │ - this.canvas.globalAlpha = style.labelOutlineOpacity || style.fontOpacity || 1; │ │ │ │ │ - this.canvas.strokeStyle = style.labelOutlineColor; │ │ │ │ │ - this.canvas.lineWidth = style.labelOutlineWidth; │ │ │ │ │ - this.canvas.strokeText(labelRows[i], pt[0], pt[1] + lineHeight * i + 1); │ │ │ │ │ - this.canvas.restore() │ │ │ │ │ + save: function(features) { │ │ │ │ │ + if (!features) { │ │ │ │ │ + features = this.layer.features │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("start", { │ │ │ │ │ + features: features │ │ │ │ │ + }); │ │ │ │ │ + var remote = this.layer.projection; │ │ │ │ │ + var local = this.layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var len = features.length; │ │ │ │ │ + var clones = new Array(len); │ │ │ │ │ + var orig, clone; │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + orig = features[i]; │ │ │ │ │ + clone = orig.clone(); │ │ │ │ │ + clone.fid = orig.fid; │ │ │ │ │ + clone.state = orig.state; │ │ │ │ │ + if (orig.url) { │ │ │ │ │ + clone.url = orig.url │ │ │ │ │ } │ │ │ │ │ - this.canvas.fillText(labelRows[i], pt[0], pt[1] + lineHeight * i) │ │ │ │ │ - } │ │ │ │ │ - } else if (this.canvas.mozDrawText) { │ │ │ │ │ - this.canvas.mozTextStyle = fontStyle; │ │ │ │ │ - var hfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[0]]; │ │ │ │ │ - if (hfactor == null) { │ │ │ │ │ - hfactor = -.5 │ │ │ │ │ + clone._original = orig; │ │ │ │ │ + clone.geometry.transform(local, remote); │ │ │ │ │ + clones[i] = clone │ │ │ │ │ } │ │ │ │ │ - var vfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5 │ │ │ │ │ + features = clones │ │ │ │ │ + } │ │ │ │ │ + this.layer.protocol.commit(features, { │ │ │ │ │ + callback: this.onCommit, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + onCommit: function(response) { │ │ │ │ │ + var evt = { │ │ │ │ │ + response: response │ │ │ │ │ + }; │ │ │ │ │ + if (response.success()) { │ │ │ │ │ + var features = response.reqFeatures; │ │ │ │ │ + var state, feature; │ │ │ │ │ + var destroys = []; │ │ │ │ │ + var insertIds = response.insertIds || []; │ │ │ │ │ + var j = 0; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + feature = feature._original || feature; │ │ │ │ │ + state = feature.state; │ │ │ │ │ + if (state) { │ │ │ │ │ + if (state == OpenLayers.State.DELETE) { │ │ │ │ │ + destroys.push(feature) │ │ │ │ │ + } else if (state == OpenLayers.State.INSERT) { │ │ │ │ │ + feature.fid = insertIds[j]; │ │ │ │ │ + ++j │ │ │ │ │ + } │ │ │ │ │ + feature.state = null │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var lineHeight = this.canvas.mozMeasureText("xx"); │ │ │ │ │ - pt[1] += lineHeight * (1 + vfactor * numRows); │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - var x = pt[0] + hfactor * this.canvas.mozMeasureText(labelRows[i]); │ │ │ │ │ - var y = pt[1] + i * lineHeight; │ │ │ │ │ - this.canvas.translate(x, y); │ │ │ │ │ - this.canvas.mozDrawText(labelRows[i]); │ │ │ │ │ - this.canvas.translate(-x, -y) │ │ │ │ │ + if (destroys.length > 0) { │ │ │ │ │ + this.layer.destroyFeatures(destroys) │ │ │ │ │ } │ │ │ │ │ + this.events.triggerEvent("success", evt) │ │ │ │ │ + } else { │ │ │ │ │ + this.events.triggerEvent("fail", evt) │ │ │ │ │ } │ │ │ │ │ - this.setCanvasStyle("reset") │ │ │ │ │ }, │ │ │ │ │ - getLocalXY: function(point) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var extent = this.extent; │ │ │ │ │ - var x = (point.x - this.featureDx) / resolution + -extent.left / resolution; │ │ │ │ │ - var y = extent.top / resolution - point.y / resolution; │ │ │ │ │ - return [x, y] │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Save" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Strategy.Paging = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + features: null, │ │ │ │ │ + length: 10, │ │ │ │ │ + num: null, │ │ │ │ │ + paging: false, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + beforefeaturesadded: this.cacheFeatures, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - clear: function() { │ │ │ │ │ - var height = this.root.height; │ │ │ │ │ - var width = this.root.width; │ │ │ │ │ - this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ - this.features = {}; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.clearRect(0, 0, width, height) │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.clearCache(); │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + beforefeaturesadded: this.cacheFeatures, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var featureId, feature; │ │ │ │ │ - if (this.hitDetection && this.root.style.display !== "none") { │ │ │ │ │ - if (!this.map.dragging) { │ │ │ │ │ - var xy = evt.xy; │ │ │ │ │ - var x = xy.x | 0; │ │ │ │ │ - var y = xy.y | 0; │ │ │ │ │ - var data = this.hitContext.getImageData(x, y, 1, 1).data; │ │ │ │ │ - if (data[3] === 255) { │ │ │ │ │ - var id = data[2] + 256 * (data[1] + 256 * data[0]); │ │ │ │ │ - if (id) { │ │ │ │ │ - featureId = "OpenLayers_Feature_Vector_" + (id - 1 + this.hitOverflow); │ │ │ │ │ - try { │ │ │ │ │ - feature = this.features[featureId][0] │ │ │ │ │ - } catch (err) {} │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + cacheFeatures: function(event) { │ │ │ │ │ + if (!this.paging) { │ │ │ │ │ + this.clearCache(); │ │ │ │ │ + this.features = event.features; │ │ │ │ │ + this.pageNext(event) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + clearCache: function() { │ │ │ │ │ + if (this.features) { │ │ │ │ │ + for (var i = 0; i < this.features.length; ++i) { │ │ │ │ │ + this.features[i].destroy() │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return feature │ │ │ │ │ + this.features = null; │ │ │ │ │ + this.num = null │ │ │ │ │ }, │ │ │ │ │ - eraseFeatures: function(features) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ - features = [features] │ │ │ │ │ + pageCount: function() { │ │ │ │ │ + var numFeatures = this.features ? this.features.length : 0; │ │ │ │ │ + return Math.ceil(numFeatures / this.length) │ │ │ │ │ + }, │ │ │ │ │ + pageNum: function() { │ │ │ │ │ + return this.num │ │ │ │ │ + }, │ │ │ │ │ + pageLength: function(newLength) { │ │ │ │ │ + if (newLength && newLength > 0) { │ │ │ │ │ + this.length = newLength │ │ │ │ │ } │ │ │ │ │ - for (var i = 0; i < features.length; ++i) { │ │ │ │ │ - delete this.features[features[i].id] │ │ │ │ │ + return this.length │ │ │ │ │ + }, │ │ │ │ │ + pageNext: function(event) { │ │ │ │ │ + var changed = false; │ │ │ │ │ + if (this.features) { │ │ │ │ │ + if (this.num === null) { │ │ │ │ │ + this.num = -1 │ │ │ │ │ + } │ │ │ │ │ + var start = (this.num + 1) * this.length; │ │ │ │ │ + changed = this.page(start, event) │ │ │ │ │ } │ │ │ │ │ - this.redraw() │ │ │ │ │ + return changed │ │ │ │ │ }, │ │ │ │ │ - redraw: function() { │ │ │ │ │ - if (!this.locked) { │ │ │ │ │ - var height = this.root.height; │ │ │ │ │ - var width = this.root.width; │ │ │ │ │ - this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.clearRect(0, 0, width, height) │ │ │ │ │ + pagePrevious: function() { │ │ │ │ │ + var changed = false; │ │ │ │ │ + if (this.features) { │ │ │ │ │ + if (this.num === null) { │ │ │ │ │ + this.num = this.pageCount() │ │ │ │ │ } │ │ │ │ │ - var labelMap = []; │ │ │ │ │ - var feature, geometry, style; │ │ │ │ │ - var worldBounds = this.map.baseLayer && this.map.baseLayer.wrapDateLine && this.map.getMaxExtent(); │ │ │ │ │ - for (var id in this.features) { │ │ │ │ │ - if (!this.features.hasOwnProperty(id)) { │ │ │ │ │ - continue │ │ │ │ │ - } │ │ │ │ │ - feature = this.features[id][0]; │ │ │ │ │ - geometry = feature.geometry; │ │ │ │ │ - this.calculateFeatureDx(geometry.getBounds(), worldBounds); │ │ │ │ │ - style = this.features[id][1]; │ │ │ │ │ - this.drawGeometry(geometry, style, feature.id); │ │ │ │ │ - if (style.label) { │ │ │ │ │ - labelMap.push([feature, style]) │ │ │ │ │ + var start = (this.num - 1) * this.length; │ │ │ │ │ + changed = this.page(start) │ │ │ │ │ + } │ │ │ │ │ + return changed │ │ │ │ │ + }, │ │ │ │ │ + page: function(start, event) { │ │ │ │ │ + var changed = false; │ │ │ │ │ + if (this.features) { │ │ │ │ │ + if (start >= 0 && start < this.features.length) { │ │ │ │ │ + var num = Math.floor(start / this.length); │ │ │ │ │ + if (num != this.num) { │ │ │ │ │ + this.paging = true; │ │ │ │ │ + var features = this.features.slice(start, start + this.length); │ │ │ │ │ + this.layer.removeFeatures(this.layer.features); │ │ │ │ │ + this.num = num; │ │ │ │ │ + if (event && event.features) { │ │ │ │ │ + event.features = features │ │ │ │ │ + } else { │ │ │ │ │ + this.layer.addFeatures(features) │ │ │ │ │ + } │ │ │ │ │ + this.paging = false; │ │ │ │ │ + changed = true │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var item; │ │ │ │ │ - for (var i = 0, len = labelMap.length; i < len; ++i) { │ │ │ │ │ - item = labelMap[i]; │ │ │ │ │ - this.drawText(item[0].geometry.getCentroid(), item[1]) │ │ │ │ │ - } │ │ │ │ │ } │ │ │ │ │ + return changed │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.Canvas" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Paging" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Renderer.Canvas.LABEL_ALIGN = { │ │ │ │ │ - l: "left", │ │ │ │ │ - r: "right", │ │ │ │ │ - t: "top", │ │ │ │ │ - b: "bottom" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.Canvas.LABEL_FACTOR = { │ │ │ │ │ - l: 0, │ │ │ │ │ - r: -1, │ │ │ │ │ - t: 0, │ │ │ │ │ - b: -1 │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.Canvas.drawImageScaleFactor = null; │ │ │ │ │ -OpenLayers.ElementsIndexer = OpenLayers.Class({ │ │ │ │ │ - maxZIndex: null, │ │ │ │ │ - order: null, │ │ │ │ │ - indices: null, │ │ │ │ │ - compare: null, │ │ │ │ │ - initialize: function(yOrdering) { │ │ │ │ │ - this.compare = yOrdering ? OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_Y_ORDER : OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_DRAWING_ORDER; │ │ │ │ │ - this.clear() │ │ │ │ │ - }, │ │ │ │ │ - insert: function(newNode) { │ │ │ │ │ - if (this.exists(newNode)) { │ │ │ │ │ - this.remove(newNode) │ │ │ │ │ - } │ │ │ │ │ - var nodeId = newNode.id; │ │ │ │ │ - this.determineZIndex(newNode); │ │ │ │ │ - var leftIndex = -1; │ │ │ │ │ - var rightIndex = this.order.length; │ │ │ │ │ - var middle; │ │ │ │ │ - while (rightIndex - leftIndex > 1) { │ │ │ │ │ - middle = parseInt((leftIndex + rightIndex) / 2); │ │ │ │ │ - var placement = this.compare(this, newNode, OpenLayers.Util.getElement(this.order[middle])); │ │ │ │ │ - if (placement > 0) { │ │ │ │ │ - leftIndex = middle │ │ │ │ │ +OpenLayers.Strategy.Fixed = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + preload: false, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + refresh: this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + if (this.layer.visibility == true || this.preload) { │ │ │ │ │ + this.load() │ │ │ │ │ } else { │ │ │ │ │ - rightIndex = middle │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + visibilitychanged: this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.order.splice(rightIndex, 0, nodeId); │ │ │ │ │ - this.indices[nodeId] = this.getZIndex(newNode); │ │ │ │ │ - return this.getNextElement(rightIndex) │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - remove: function(node) { │ │ │ │ │ - var nodeId = node.id; │ │ │ │ │ - var arrayIndex = OpenLayers.Util.indexOf(this.order, nodeId); │ │ │ │ │ - if (arrayIndex >= 0) { │ │ │ │ │ - this.order.splice(arrayIndex, 1); │ │ │ │ │ - delete this.indices[nodeId]; │ │ │ │ │ - if (this.order.length > 0) { │ │ │ │ │ - var lastId = this.order[this.order.length - 1]; │ │ │ │ │ - this.maxZIndex = this.indices[lastId] │ │ │ │ │ - } else { │ │ │ │ │ - this.maxZIndex = 0 │ │ │ │ │ - } │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + refresh: this.load, │ │ │ │ │ + visibilitychanged: this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - clear: function() { │ │ │ │ │ - this.order = []; │ │ │ │ │ - this.indices = {}; │ │ │ │ │ - this.maxZIndex = 0 │ │ │ │ │ + load: function(options) { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + layer.events.triggerEvent("loadstart", { │ │ │ │ │ + filter: layer.filter │ │ │ │ │ + }); │ │ │ │ │ + layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: this.merge, │ │ │ │ │ + filter: layer.filter, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options)); │ │ │ │ │ + layer.events.un({ │ │ │ │ │ + visibilitychanged: this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - exists: function(node) { │ │ │ │ │ - return this.indices[node.id] != null │ │ │ │ │ + merge: function(resp) { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + layer.destroyFeatures(); │ │ │ │ │ + var features = resp.features; │ │ │ │ │ + if (features && features.length > 0) { │ │ │ │ │ + var remote = layer.projection; │ │ │ │ │ + var local = layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var geom; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + geom = features[i].geometry; │ │ │ │ │ + if (geom) { │ │ │ │ │ + geom.transform(remote, local) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + layer.addFeatures(features) │ │ │ │ │ + } │ │ │ │ │ + layer.events.triggerEvent("loadend", { │ │ │ │ │ + response: resp │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - getZIndex: function(node) { │ │ │ │ │ - return node._style.graphicZIndex │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Fixed" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Marker.Box = OpenLayers.Class(OpenLayers.Marker, { │ │ │ │ │ + bounds: null, │ │ │ │ │ + div: null, │ │ │ │ │ + initialize: function(bounds, borderColor, borderWidth) { │ │ │ │ │ + this.bounds = bounds; │ │ │ │ │ + this.div = OpenLayers.Util.createDiv(); │ │ │ │ │ + this.div.style.overflow = "hidden"; │ │ │ │ │ + this.events = new OpenLayers.Events(this, this.div); │ │ │ │ │ + this.setBorder(borderColor, borderWidth) │ │ │ │ │ }, │ │ │ │ │ - determineZIndex: function(node) { │ │ │ │ │ - var zIndex = node._style.graphicZIndex; │ │ │ │ │ - if (zIndex == null) { │ │ │ │ │ - zIndex = this.maxZIndex; │ │ │ │ │ - node._style.graphicZIndex = zIndex │ │ │ │ │ - } else if (zIndex > this.maxZIndex) { │ │ │ │ │ - this.maxZIndex = zIndex │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.bounds = null; │ │ │ │ │ + this.div = null; │ │ │ │ │ + OpenLayers.Marker.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + setBorder: function(color, width) { │ │ │ │ │ + if (!color) { │ │ │ │ │ + color = "red" │ │ │ │ │ + } │ │ │ │ │ + if (!width) { │ │ │ │ │ + width = 2 │ │ │ │ │ } │ │ │ │ │ + this.div.style.border = width + "px solid " + color │ │ │ │ │ }, │ │ │ │ │ - getNextElement: function(index) { │ │ │ │ │ - var nextIndex = index + 1; │ │ │ │ │ - if (nextIndex < this.order.length) { │ │ │ │ │ - var nextElement = OpenLayers.Util.getElement(this.order[nextIndex]); │ │ │ │ │ - if (nextElement == undefined) { │ │ │ │ │ - nextElement = this.getNextElement(nextIndex) │ │ │ │ │ - } │ │ │ │ │ - return nextElement │ │ │ │ │ - } else { │ │ │ │ │ - return null │ │ │ │ │ + draw: function(px, sz) { │ │ │ │ │ + OpenLayers.Util.modifyDOMElement(this.div, null, px, sz); │ │ │ │ │ + return this.div │ │ │ │ │ + }, │ │ │ │ │ + onScreen: function() { │ │ │ │ │ + var onScreen = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + var screenBounds = this.map.getExtent(); │ │ │ │ │ + onScreen = screenBounds.containsBounds(this.bounds, true, true) │ │ │ │ │ } │ │ │ │ │ + return onScreen │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.ElementsIndexer" │ │ │ │ │ + display: function(display) { │ │ │ │ │ + this.div.style.display = display ? "" : "none" │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Marker.Box" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.ElementsIndexer.IndexingMethods = { │ │ │ │ │ - Z_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var newZIndex = indexer.getZIndex(newNode); │ │ │ │ │ - var returnVal = 0; │ │ │ │ │ - if (nextNode) { │ │ │ │ │ - var nextZIndex = indexer.getZIndex(nextNode); │ │ │ │ │ - returnVal = newZIndex - nextZIndex │ │ │ │ │ +OpenLayers.Layer.XYZ = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + sphericalMercator: false, │ │ │ │ │ + zoomOffset: 0, │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + if (options && options.sphericalMercator || this.sphericalMercator) { │ │ │ │ │ + options = OpenLayers.Util.extend({ │ │ │ │ │ + projection: "EPSG:900913", │ │ │ │ │ + numZoomLevels: 19 │ │ │ │ │ + }, options) │ │ │ │ │ } │ │ │ │ │ - return returnVal │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name || this.name, url || this.url, {}, options]) │ │ │ │ │ }, │ │ │ │ │ - Z_ORDER_DRAWING_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER(indexer, newNode, nextNode); │ │ │ │ │ - if (nextNode && returnVal == 0) { │ │ │ │ │ - returnVal = 1 │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.XYZ(this.name, this.url, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - return returnVal │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - Z_ORDER_Y_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER(indexer, newNode, nextNode); │ │ │ │ │ - if (nextNode && returnVal === 0) { │ │ │ │ │ - var result = nextNode._boundsBottom - newNode._boundsBottom; │ │ │ │ │ - returnVal = result === 0 ? 1 : result │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var xyz = this.getXYZ(bounds); │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + var s = "" + xyz.x + xyz.y + xyz.z; │ │ │ │ │ + url = this.selectUrl(s, url) │ │ │ │ │ } │ │ │ │ │ - return returnVal │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.Elements = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ - rendererRoot: null, │ │ │ │ │ - root: null, │ │ │ │ │ - vectorRoot: null, │ │ │ │ │ - textRoot: null, │ │ │ │ │ - xmlns: null, │ │ │ │ │ - xOffset: 0, │ │ │ │ │ - indexer: null, │ │ │ │ │ - BACKGROUND_ID_SUFFIX: "_background", │ │ │ │ │ - LABEL_ID_SUFFIX: "_label", │ │ │ │ │ - LABEL_OUTLINE_SUFFIX: "_outline", │ │ │ │ │ - initialize: function(containerID, options) { │ │ │ │ │ - OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.rendererRoot = this.createRenderRoot(); │ │ │ │ │ - this.root = this.createRoot("_root"); │ │ │ │ │ - this.vectorRoot = this.createRoot("_vroot"); │ │ │ │ │ - this.textRoot = this.createRoot("_troot"); │ │ │ │ │ - this.root.appendChild(this.vectorRoot); │ │ │ │ │ - this.root.appendChild(this.textRoot); │ │ │ │ │ - this.rendererRoot.appendChild(this.root); │ │ │ │ │ - this.container.appendChild(this.rendererRoot); │ │ │ │ │ - if (options && (options.zIndexing || options.yOrdering)) { │ │ │ │ │ - this.indexer = new OpenLayers.ElementsIndexer(options.yOrdering) │ │ │ │ │ + return OpenLayers.String.format(url, xyz) │ │ │ │ │ + }, │ │ │ │ │ + getXYZ: function(bounds) { │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.maxExtent.left) / (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((this.maxExtent.top - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getServerZoom(); │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + var limit = Math.pow(2, z); │ │ │ │ │ + x = (x % limit + limit) % limit │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + x: x, │ │ │ │ │ + y: y, │ │ │ │ │ + z: z │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (!this.tileOrigin) { │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, this.maxExtent.bottom) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.XYZ" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Markers = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ + isBaseLayer: false, │ │ │ │ │ + markers: null, │ │ │ │ │ + drawn: false, │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.markers = [] │ │ │ │ │ + }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - this.clear(); │ │ │ │ │ - this.rendererRoot = null; │ │ │ │ │ - this.root = null; │ │ │ │ │ - this.xmlns = null; │ │ │ │ │ - OpenLayers.Renderer.prototype.destroy.apply(this, arguments) │ │ │ │ │ + this.clearMarkers(); │ │ │ │ │ + this.markers = null; │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - clear: function() { │ │ │ │ │ - var child; │ │ │ │ │ - var root = this.vectorRoot; │ │ │ │ │ - if (root) { │ │ │ │ │ - while (child = root.firstChild) { │ │ │ │ │ - root.removeChild(child) │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + if (opacity != this.opacity) { │ │ │ │ │ + this.opacity = opacity; │ │ │ │ │ + for (var i = 0, len = this.markers.length; i < len; i++) { │ │ │ │ │ + this.markers[i].setOpacity(this.opacity) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - root = this.textRoot; │ │ │ │ │ - if (root) { │ │ │ │ │ - while (child = root.firstChild) { │ │ │ │ │ - root.removeChild(child) │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + if (zoomChanged || !this.drawn) { │ │ │ │ │ + for (var i = 0, len = this.markers.length; i < len; i++) { │ │ │ │ │ + this.drawMarker(this.markers[i]) │ │ │ │ │ } │ │ │ │ │ + this.drawn = true │ │ │ │ │ } │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - this.indexer.clear() │ │ │ │ │ + }, │ │ │ │ │ + addMarker: function(marker) { │ │ │ │ │ + this.markers.push(marker); │ │ │ │ │ + if (this.opacity < 1) { │ │ │ │ │ + marker.setOpacity(this.opacity) │ │ │ │ │ + } │ │ │ │ │ + if (this.map && this.map.getExtent()) { │ │ │ │ │ + marker.map = this.map; │ │ │ │ │ + this.drawMarker(marker) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - var rightOfDateLine, ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ - extent = extent.scale(1 / ratio), │ │ │ │ │ - world = this.map.getMaxExtent(); │ │ │ │ │ - if (world.right > extent.left && world.right < extent.right) { │ │ │ │ │ - rightOfDateLine = true │ │ │ │ │ - } else if (world.left > extent.left && world.left < extent.right) { │ │ │ │ │ - rightOfDateLine = false │ │ │ │ │ + removeMarker: function(marker) { │ │ │ │ │ + if (this.markers && this.markers.length) { │ │ │ │ │ + OpenLayers.Util.removeItem(this.markers, marker); │ │ │ │ │ + marker.erase() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + clearMarkers: function() { │ │ │ │ │ + if (this.markers != null) { │ │ │ │ │ + while (this.markers.length > 0) { │ │ │ │ │ + this.removeMarker(this.markers[0]) │ │ │ │ │ } │ │ │ │ │ - if (rightOfDateLine !== this.rightOfDateLine || resolutionChanged) { │ │ │ │ │ - coordSysUnchanged = false; │ │ │ │ │ - this.xOffset = rightOfDateLine === true ? world.getWidth() / resolution : 0 │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + drawMarker: function(marker) { │ │ │ │ │ + var px = this.map.getLayerPxFromLonLat(marker.lonlat); │ │ │ │ │ + if (px == null) { │ │ │ │ │ + marker.display(false) │ │ │ │ │ + } else { │ │ │ │ │ + if (!marker.isDrawn()) { │ │ │ │ │ + var markerImg = marker.draw(px); │ │ │ │ │ + this.div.appendChild(markerImg) │ │ │ │ │ + } else if (marker.icon) { │ │ │ │ │ + marker.icon.moveTo(px) │ │ │ │ │ } │ │ │ │ │ - this.rightOfDateLine = rightOfDateLine │ │ │ │ │ } │ │ │ │ │ - return coordSysUnchanged │ │ │ │ │ }, │ │ │ │ │ - getNodeType: function(geometry, style) {}, │ │ │ │ │ - drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ - var className = geometry.CLASS_NAME; │ │ │ │ │ - var rendered = true; │ │ │ │ │ - if (className == "OpenLayers.Geometry.Collection" || className == "OpenLayers.Geometry.MultiPoint" || className == "OpenLayers.Geometry.MultiLineString" || className == "OpenLayers.Geometry.MultiPolygon") { │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - rendered = this.drawGeometry(geometry.components[i], style, featureId) && rendered │ │ │ │ │ + getDataExtent: function() { │ │ │ │ │ + var maxExtent = null; │ │ │ │ │ + if (this.markers && this.markers.length > 0) { │ │ │ │ │ + var maxExtent = new OpenLayers.Bounds; │ │ │ │ │ + for (var i = 0, len = this.markers.length; i < len; i++) { │ │ │ │ │ + var marker = this.markers[i]; │ │ │ │ │ + maxExtent.extend(marker.lonlat) │ │ │ │ │ } │ │ │ │ │ - return rendered │ │ │ │ │ } │ │ │ │ │ - rendered = false; │ │ │ │ │ - var removeBackground = false; │ │ │ │ │ - if (style.display != "none") { │ │ │ │ │ - if (style.backgroundGraphic) { │ │ │ │ │ - this.redrawBackgroundNode(geometry.id, geometry, style, featureId) │ │ │ │ │ - } else { │ │ │ │ │ - removeBackground = true │ │ │ │ │ + return maxExtent │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Markers" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Text = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ + location: null, │ │ │ │ │ + features: null, │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + selectedFeature: null, │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.features = [] │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.destroy.apply(this, arguments); │ │ │ │ │ + this.clearFeatures(); │ │ │ │ │ + this.features = null │ │ │ │ │ + }, │ │ │ │ │ + loadText: function() { │ │ │ │ │ + if (!this.loaded) { │ │ │ │ │ + if (this.location != null) { │ │ │ │ │ + var onFail = function(e) { │ │ │ │ │ + this.events.triggerEvent("loadend") │ │ │ │ │ + }; │ │ │ │ │ + this.events.triggerEvent("loadstart"); │ │ │ │ │ + OpenLayers.Request.GET({ │ │ │ │ │ + url: this.location, │ │ │ │ │ + success: this.parseData, │ │ │ │ │ + failure: onFail, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.loaded = true │ │ │ │ │ } │ │ │ │ │ - rendered = this.redrawNode(geometry.id, geometry, style, featureId) │ │ │ │ │ } │ │ │ │ │ - if (rendered == false) { │ │ │ │ │ - var node = document.getElementById(geometry.id); │ │ │ │ │ - if (node) { │ │ │ │ │ - if (node._style.backgroundGraphic) { │ │ │ │ │ - removeBackground = true │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(bounds, zoomChanged, minor) { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + if (this.visibility && !this.loaded) { │ │ │ │ │ + this.loadText() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + parseData: function(ajaxRequest) { │ │ │ │ │ + var text = ajaxRequest.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 parser = new OpenLayers.Format.Text(options); │ │ │ │ │ + var features = parser.read(text); │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + var data = {}; │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + var location; │ │ │ │ │ + var iconSize, iconOffset; │ │ │ │ │ + location = new OpenLayers.LonLat(feature.geometry.x, feature.geometry.y); │ │ │ │ │ + if (feature.style.graphicWidth && feature.style.graphicHeight) { │ │ │ │ │ + iconSize = new OpenLayers.Size(feature.style.graphicWidth, feature.style.graphicHeight) │ │ │ │ │ + } │ │ │ │ │ + if (feature.style.graphicXOffset !== undefined && feature.style.graphicYOffset !== undefined) { │ │ │ │ │ + iconOffset = new OpenLayers.Pixel(feature.style.graphicXOffset, feature.style.graphicYOffset) │ │ │ │ │ + } │ │ │ │ │ + if (feature.style.externalGraphic != null) { │ │ │ │ │ + data.icon = new OpenLayers.Icon(feature.style.externalGraphic, iconSize, iconOffset) │ │ │ │ │ + } else { │ │ │ │ │ + data.icon = OpenLayers.Marker.defaultIcon(); │ │ │ │ │ + if (iconSize != null) { │ │ │ │ │ + data.icon.setSize(iconSize) │ │ │ │ │ } │ │ │ │ │ - node.parentNode.removeChild(node) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - if (removeBackground) { │ │ │ │ │ - var node = document.getElementById(geometry.id + this.BACKGROUND_ID_SUFFIX); │ │ │ │ │ - if (node) { │ │ │ │ │ - node.parentNode.removeChild(node) │ │ │ │ │ + if (feature.attributes.title != null && feature.attributes.description != null) { │ │ │ │ │ + data["popupContentHTML"] = "<h2>" + feature.attributes.title + "</h2>" + "<p>" + feature.attributes.description + "</p>" │ │ │ │ │ } │ │ │ │ │ + data["overflow"] = feature.attributes.overflow || "auto"; │ │ │ │ │ + var markerFeature = new OpenLayers.Feature(this, location, data); │ │ │ │ │ + this.features.push(markerFeature); │ │ │ │ │ + var marker = markerFeature.createMarker(); │ │ │ │ │ + if (feature.attributes.title != null && feature.attributes.description != null) { │ │ │ │ │ + marker.events.register("click", markerFeature, this.markerClick) │ │ │ │ │ + } │ │ │ │ │ + this.addMarker(marker) │ │ │ │ │ } │ │ │ │ │ - return rendered │ │ │ │ │ + this.events.triggerEvent("loadend") │ │ │ │ │ }, │ │ │ │ │ - redrawNode: function(id, geometry, style, featureId) { │ │ │ │ │ - style = this.applyDefaultSymbolizer(style); │ │ │ │ │ - var node = this.nodeFactory(id, this.getNodeType(geometry, style)); │ │ │ │ │ - node._featureId = featureId; │ │ │ │ │ - node._boundsBottom = geometry.getBounds().bottom; │ │ │ │ │ - node._geometryClass = geometry.CLASS_NAME; │ │ │ │ │ - node._style = style; │ │ │ │ │ - var drawResult = this.drawGeometryNode(node, geometry, style); │ │ │ │ │ - if (drawResult === false) { │ │ │ │ │ - return false │ │ │ │ │ + markerClick: function(evt) { │ │ │ │ │ + var sameMarkerClicked = this == this.layer.selectedFeature; │ │ │ │ │ + this.layer.selectedFeature = !sameMarkerClicked ? this : null; │ │ │ │ │ + for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ + this.layer.map.removePopup(this.layer.map.popups[i]) │ │ │ │ │ } │ │ │ │ │ - node = drawResult.node; │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - var insert = this.indexer.insert(node); │ │ │ │ │ - if (insert) { │ │ │ │ │ - this.vectorRoot.insertBefore(node, insert) │ │ │ │ │ - } else { │ │ │ │ │ - this.vectorRoot.appendChild(node) │ │ │ │ │ + if (!sameMarkerClicked) { │ │ │ │ │ + this.layer.map.addPopup(this.createPopup()) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ + }, │ │ │ │ │ + clearFeatures: function() { │ │ │ │ │ + if (this.features != null) { │ │ │ │ │ + while (this.features.length > 0) { │ │ │ │ │ + var feature = this.features[0]; │ │ │ │ │ + OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ + feature.destroy() │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - if (node.parentNode !== this.vectorRoot) { │ │ │ │ │ - this.vectorRoot.appendChild(node) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Text" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.ArcGISCache = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ + url: null, │ │ │ │ │ + tileOrigin: null, │ │ │ │ │ + tileSize: new OpenLayers.Size(256, 256), │ │ │ │ │ + useArcGISServer: true, │ │ │ │ │ + type: "png", │ │ │ │ │ + useScales: false, │ │ │ │ │ + overrideDPI: false, │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (this.resolutions) { │ │ │ │ │ + this.serverResolutions = this.resolutions; │ │ │ │ │ + this.maxExtent = this.getMaxExtentForResolution(this.resolutions[0]) │ │ │ │ │ + } │ │ │ │ │ + if (this.layerInfo) { │ │ │ │ │ + var info = this.layerInfo; │ │ │ │ │ + var startingTileExtent = new OpenLayers.Bounds(info.fullExtent.xmin, info.fullExtent.ymin, info.fullExtent.xmax, info.fullExtent.ymax); │ │ │ │ │ + this.projection = "EPSG:" + info.spatialReference.wkid; │ │ │ │ │ + this.sphericalMercator = info.spatialReference.wkid == 102100; │ │ │ │ │ + this.units = info.units == "esriFeet" ? "ft" : "m"; │ │ │ │ │ + if (!!info.tileInfo) { │ │ │ │ │ + this.tileSize = new OpenLayers.Size(info.tileInfo.width || info.tileInfo.cols, info.tileInfo.height || info.tileInfo.rows); │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(info.tileInfo.origin.x, info.tileInfo.origin.y); │ │ │ │ │ + var upperLeft = new OpenLayers.Geometry.Point(startingTileExtent.left, startingTileExtent.top); │ │ │ │ │ + var bottomRight = new OpenLayers.Geometry.Point(startingTileExtent.right, startingTileExtent.bottom); │ │ │ │ │ + if (this.useScales) { │ │ │ │ │ + this.scales = [] │ │ │ │ │ + } else { │ │ │ │ │ + this.resolutions = [] │ │ │ │ │ + } │ │ │ │ │ + this.lods = []; │ │ │ │ │ + for (var key in info.tileInfo.lods) { │ │ │ │ │ + if (info.tileInfo.lods.hasOwnProperty(key)) { │ │ │ │ │ + var lod = info.tileInfo.lods[key]; │ │ │ │ │ + if (this.useScales) { │ │ │ │ │ + this.scales.push(lod.scale) │ │ │ │ │ + } else { │ │ │ │ │ + this.resolutions.push(lod.resolution) │ │ │ │ │ + } │ │ │ │ │ + var start = this.getContainingTileCoords(upperLeft, lod.resolution); │ │ │ │ │ + lod.startTileCol = start.x; │ │ │ │ │ + lod.startTileRow = start.y; │ │ │ │ │ + var end = this.getContainingTileCoords(bottomRight, lod.resolution); │ │ │ │ │ + lod.endTileCol = end.x; │ │ │ │ │ + lod.endTileRow = end.y; │ │ │ │ │ + this.lods.push(lod) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.maxExtent = this.calculateMaxExtentWithLOD(this.lods[0]); │ │ │ │ │ + this.serverResolutions = this.resolutions; │ │ │ │ │ + if (this.overrideDPI && info.tileInfo.dpi) { │ │ │ │ │ + OpenLayers.DOTS_PER_INCH = info.tileInfo.dpi │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.postDraw(node); │ │ │ │ │ - return drawResult.complete │ │ │ │ │ }, │ │ │ │ │ - redrawBackgroundNode: function(id, geometry, style, featureId) { │ │ │ │ │ - var backgroundStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ - backgroundStyle.externalGraphic = backgroundStyle.backgroundGraphic; │ │ │ │ │ - backgroundStyle.graphicXOffset = backgroundStyle.backgroundXOffset; │ │ │ │ │ - backgroundStyle.graphicYOffset = backgroundStyle.backgroundYOffset; │ │ │ │ │ - backgroundStyle.graphicZIndex = backgroundStyle.backgroundGraphicZIndex; │ │ │ │ │ - backgroundStyle.graphicWidth = backgroundStyle.backgroundWidth || backgroundStyle.graphicWidth; │ │ │ │ │ - backgroundStyle.graphicHeight = backgroundStyle.backgroundHeight || backgroundStyle.graphicHeight; │ │ │ │ │ - backgroundStyle.backgroundGraphic = null; │ │ │ │ │ - backgroundStyle.backgroundXOffset = null; │ │ │ │ │ - backgroundStyle.backgroundYOffset = null; │ │ │ │ │ - backgroundStyle.backgroundGraphicZIndex = null; │ │ │ │ │ - return this.redrawNode(id + this.BACKGROUND_ID_SUFFIX, geometry, backgroundStyle, null) │ │ │ │ │ + getContainingTileCoords: function(point, res) { │ │ │ │ │ + return new OpenLayers.Pixel(Math.max(Math.floor((point.x - this.tileOrigin.lon) / (this.tileSize.w * res)), 0), Math.max(Math.floor((this.tileOrigin.lat - point.y) / (this.tileSize.h * res)), 0)) │ │ │ │ │ }, │ │ │ │ │ - drawGeometryNode: function(node, geometry, style) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - var options = { │ │ │ │ │ - isFilled: style.fill === undefined ? true : style.fill, │ │ │ │ │ - isStroked: style.stroke === undefined ? !!style.strokeWidth : style.stroke │ │ │ │ │ + calculateMaxExtentWithLOD: function(lod) { │ │ │ │ │ + var numTileCols = lod.endTileCol - lod.startTileCol + 1; │ │ │ │ │ + var numTileRows = lod.endTileRow - lod.startTileRow + 1; │ │ │ │ │ + var minX = this.tileOrigin.lon + lod.startTileCol * this.tileSize.w * lod.resolution; │ │ │ │ │ + var maxX = minX + numTileCols * this.tileSize.w * lod.resolution; │ │ │ │ │ + var maxY = this.tileOrigin.lat - lod.startTileRow * this.tileSize.h * lod.resolution; │ │ │ │ │ + var minY = maxY - numTileRows * this.tileSize.h * lod.resolution; │ │ │ │ │ + return new OpenLayers.Bounds(minX, minY, maxX, maxY) │ │ │ │ │ + }, │ │ │ │ │ + calculateMaxExtentWithExtent: function(extent, res) { │ │ │ │ │ + var upperLeft = new OpenLayers.Geometry.Point(extent.left, extent.top); │ │ │ │ │ + var bottomRight = new OpenLayers.Geometry.Point(extent.right, extent.bottom); │ │ │ │ │ + var start = this.getContainingTileCoords(upperLeft, res); │ │ │ │ │ + var end = this.getContainingTileCoords(bottomRight, res); │ │ │ │ │ + var lod = { │ │ │ │ │ + resolution: res, │ │ │ │ │ + startTileCol: start.x, │ │ │ │ │ + startTileRow: start.y, │ │ │ │ │ + endTileCol: end.x, │ │ │ │ │ + endTileRow: end.y │ │ │ │ │ }; │ │ │ │ │ - var drawn; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.graphic === false) { │ │ │ │ │ - options.isFilled = false; │ │ │ │ │ - options.isStroked = false │ │ │ │ │ - } │ │ │ │ │ - drawn = this.drawPoint(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - options.isFilled = false; │ │ │ │ │ - drawn = this.drawLineString(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - drawn = this.drawLinearRing(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - drawn = this.drawPolygon(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - drawn = this.drawRectangle(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break │ │ │ │ │ + return this.calculateMaxExtentWithLOD(lod) │ │ │ │ │ + }, │ │ │ │ │ + getUpperLeftTileCoord: function(res) { │ │ │ │ │ + var upperLeft = new OpenLayers.Geometry.Point(this.maxExtent.left, this.maxExtent.top); │ │ │ │ │ + return this.getContainingTileCoords(upperLeft, res) │ │ │ │ │ + }, │ │ │ │ │ + getLowerRightTileCoord: function(res) { │ │ │ │ │ + var bottomRight = new OpenLayers.Geometry.Point(this.maxExtent.right, this.maxExtent.bottom); │ │ │ │ │ + return this.getContainingTileCoords(bottomRight, res) │ │ │ │ │ + }, │ │ │ │ │ + getMaxExtentForResolution: function(res) { │ │ │ │ │ + var start = this.getUpperLeftTileCoord(res); │ │ │ │ │ + var end = this.getLowerRightTileCoord(res); │ │ │ │ │ + var numTileCols = end.x - start.x + 1; │ │ │ │ │ + var numTileRows = end.y - start.y + 1; │ │ │ │ │ + var minX = this.tileOrigin.lon + start.x * this.tileSize.w * res; │ │ │ │ │ + var maxX = minX + numTileCols * this.tileSize.w * res; │ │ │ │ │ + var maxY = this.tileOrigin.lat - start.y * this.tileSize.h * res; │ │ │ │ │ + var minY = maxY - numTileRows * this.tileSize.h * res; │ │ │ │ │ + return new OpenLayers.Bounds(minX, minY, maxX, maxY) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.ArcGISCache(this.name, this.url, this.options) │ │ │ │ │ } │ │ │ │ │ - node._options = options; │ │ │ │ │ - if (drawn != false) { │ │ │ │ │ - return { │ │ │ │ │ - node: this.setStyle(node, style, options, geometry), │ │ │ │ │ - complete: drawn │ │ │ │ │ + return OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]) │ │ │ │ │ + }, │ │ │ │ │ + initGriddedTiles: function(bounds) { │ │ │ │ │ + delete this._tileOrigin; │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initGriddedTiles.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + getMaxExtent: function() { │ │ │ │ │ + var resolution = this.map.getResolution(); │ │ │ │ │ + return this.maxExtent = this.getMaxExtentForResolution(resolution) │ │ │ │ │ + }, │ │ │ │ │ + getTileOrigin: function() { │ │ │ │ │ + if (!this._tileOrigin) { │ │ │ │ │ + var extent = this.getMaxExtent(); │ │ │ │ │ + this._tileOrigin = new OpenLayers.LonLat(extent.left, extent.bottom) │ │ │ │ │ + } │ │ │ │ │ + return this._tileOrigin │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var res = this.getResolution(); │ │ │ │ │ + var originTileX = this.tileOrigin.lon + res * this.tileSize.w / 2; │ │ │ │ │ + var originTileY = this.tileOrigin.lat - res * this.tileSize.h / 2; │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var point = { │ │ │ │ │ + x: center.lon, │ │ │ │ │ + y: center.lat │ │ │ │ │ + }; │ │ │ │ │ + var x = Math.round(Math.abs((center.lon - originTileX) / (res * this.tileSize.w))); │ │ │ │ │ + var y = Math.round(Math.abs((originTileY - center.lat) / (res * this.tileSize.h))); │ │ │ │ │ + var z = this.map.getZoom(); │ │ │ │ │ + if (this.lods) { │ │ │ │ │ + var lod = this.lods[this.map.getZoom()]; │ │ │ │ │ + if (x < lod.startTileCol || x > lod.endTileCol || (y < lod.startTileRow || y > lod.endTileRow)) { │ │ │ │ │ + return null │ │ │ │ │ } │ │ │ │ │ } else { │ │ │ │ │ - return false │ │ │ │ │ + var start = this.getUpperLeftTileCoord(res); │ │ │ │ │ + var end = this.getLowerRightTileCoord(res); │ │ │ │ │ + if (x < start.x || x >= end.x || (y < start.y || y >= end.y)) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - postDraw: function(node) {}, │ │ │ │ │ - drawPoint: function(node, geometry) {}, │ │ │ │ │ - drawLineString: function(node, geometry) {}, │ │ │ │ │ - drawLinearRing: function(node, geometry) {}, │ │ │ │ │ - drawPolygon: function(node, geometry) {}, │ │ │ │ │ - drawRectangle: function(node, geometry) {}, │ │ │ │ │ - drawCircle: function(node, geometry) {}, │ │ │ │ │ - removeText: function(featureId) { │ │ │ │ │ - var label = document.getElementById(featureId + this.LABEL_ID_SUFFIX); │ │ │ │ │ - if (label) { │ │ │ │ │ - this.textRoot.removeChild(label) │ │ │ │ │ + var url = this.url; │ │ │ │ │ + var s = "" + x + y + z; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(s, url) │ │ │ │ │ } │ │ │ │ │ - var outline = document.getElementById(featureId + this.LABEL_OUTLINE_SUFFIX); │ │ │ │ │ - if (outline) { │ │ │ │ │ - this.textRoot.removeChild(outline) │ │ │ │ │ + if (this.useArcGISServer) { │ │ │ │ │ + url = url + "/tile/${z}/${y}/${x}" │ │ │ │ │ + } else { │ │ │ │ │ + x = "C" + OpenLayers.Number.zeroPad(x, 8, 16); │ │ │ │ │ + y = "R" + OpenLayers.Number.zeroPad(y, 8, 16); │ │ │ │ │ + z = "L" + OpenLayers.Number.zeroPad(z, 2, 10); │ │ │ │ │ + url = url + "/${z}/${y}/${x}." + this.type │ │ │ │ │ } │ │ │ │ │ + url = OpenLayers.String.format(url, { │ │ │ │ │ + x: x, │ │ │ │ │ + y: y, │ │ │ │ │ + z: z │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Util.urlAppend(url, OpenLayers.Util.getParameterString(this.params)) │ │ │ │ │ }, │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var target = evt.target; │ │ │ │ │ - var useElement = target && target.correspondingUseElement; │ │ │ │ │ - var node = useElement ? useElement : target || evt.srcElement; │ │ │ │ │ - return node._featureId │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.ArcGISCache" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.ArcIMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + ClientVersion: "9.2", │ │ │ │ │ + ServiceName: "" │ │ │ │ │ }, │ │ │ │ │ - eraseGeometry: function(geometry, featureId) { │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPoint" || geometry.CLASS_NAME == "OpenLayers.Geometry.MultiLineString" || geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPolygon" || geometry.CLASS_NAME == "OpenLayers.Geometry.Collection") { │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - this.eraseGeometry(geometry.components[i], featureId) │ │ │ │ │ + featureCoordSys: "4326", │ │ │ │ │ + filterCoordSys: "4326", │ │ │ │ │ + layers: null, │ │ │ │ │ + async: true, │ │ │ │ │ + name: "ArcIMS", │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + DEFAULT_OPTIONS: { │ │ │ │ │ + tileSize: new OpenLayers.Size(512, 512), │ │ │ │ │ + featureCoordSys: "4326", │ │ │ │ │ + filterCoordSys: "4326", │ │ │ │ │ + layers: null, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + async: true, │ │ │ │ │ + name: "ArcIMS" │ │ │ │ │ + }, │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + this.tileSize = new OpenLayers.Size(512, 512); │ │ │ │ │ + this.params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + ServiceName: options.serviceName │ │ │ │ │ + }, this.DEFAULT_PARAMS); │ │ │ │ │ + this.options = OpenLayers.Util.applyDefaults(options, this.DEFAULT_OPTIONS); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name, url, this.params, options]); │ │ │ │ │ + if (this.transparent) { │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + this.isBaseLayer = false │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - var element = OpenLayers.Util.getElement(geometry.id); │ │ │ │ │ - if (element && element.parentNode) { │ │ │ │ │ - if (element.geometry) { │ │ │ │ │ - element.geometry.destroy(); │ │ │ │ │ - element.geometry = null │ │ │ │ │ - } │ │ │ │ │ - element.parentNode.removeChild(element); │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - this.indexer.remove(element) │ │ │ │ │ - } │ │ │ │ │ - if (element._style.backgroundGraphic) { │ │ │ │ │ - var backgroundId = geometry.id + this.BACKGROUND_ID_SUFFIX; │ │ │ │ │ - var bElem = OpenLayers.Util.getElement(backgroundId); │ │ │ │ │ - if (bElem && bElem.parentNode) { │ │ │ │ │ - bElem.parentNode.removeChild(bElem) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + if (this.format == "image/jpeg") { │ │ │ │ │ + this.format = OpenLayers.Util.alphaHack() ? "image/gif" : "image/png" │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + if (this.options.layers === null) { │ │ │ │ │ + this.options.layers = [] │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - nodeFactory: function(id, type) { │ │ │ │ │ - var node = OpenLayers.Util.getElement(id); │ │ │ │ │ - if (node) { │ │ │ │ │ - if (!this.nodeTypeCompare(node, type)) { │ │ │ │ │ - node.parentNode.removeChild(node); │ │ │ │ │ - node = this.nodeFactory(id, type) │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var url = ""; │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var axlReq = new OpenLayers.Format.ArcXML(OpenLayers.Util.extend(this.options, { │ │ │ │ │ + requesttype: "image", │ │ │ │ │ + envelope: bounds.toArray(), │ │ │ │ │ + tileSize: this.tileSize │ │ │ │ │ + })); │ │ │ │ │ + var req = new OpenLayers.Request.POST({ │ │ │ │ │ + url: this.getFullRequestString(), │ │ │ │ │ + data: axlReq.write(), │ │ │ │ │ + async: false │ │ │ │ │ + }); │ │ │ │ │ + if (req != null) { │ │ │ │ │ + var doc = req.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = req.responseText │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - node = this.createNode(type, id) │ │ │ │ │ + var axlResp = new OpenLayers.Format.ArcXML; │ │ │ │ │ + var arcxml = axlResp.read(doc); │ │ │ │ │ + url = this.getUrlOrImage(arcxml.image.output) │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ + return url │ │ │ │ │ }, │ │ │ │ │ - nodeTypeCompare: function(node, type) {}, │ │ │ │ │ - createNode: function(type, id) {}, │ │ │ │ │ - moveRoot: function(renderer) { │ │ │ │ │ - var root = this.root; │ │ │ │ │ - if (renderer.root.parentNode == this.rendererRoot) { │ │ │ │ │ - root = renderer.root │ │ │ │ │ - } │ │ │ │ │ - root.parentNode.removeChild(root); │ │ │ │ │ - renderer.rendererRoot.appendChild(root) │ │ │ │ │ + getURLasync: function(bounds, callback, scope) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var axlReq = new OpenLayers.Format.ArcXML(OpenLayers.Util.extend(this.options, { │ │ │ │ │ + requesttype: "image", │ │ │ │ │ + envelope: bounds.toArray(), │ │ │ │ │ + tileSize: this.tileSize │ │ │ │ │ + })); │ │ │ │ │ + OpenLayers.Request.POST({ │ │ │ │ │ + url: this.getFullRequestString(), │ │ │ │ │ + async: true, │ │ │ │ │ + data: axlReq.write(), │ │ │ │ │ + callback: function(req) { │ │ │ │ │ + var doc = req.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = req.responseText │ │ │ │ │ + } │ │ │ │ │ + var axlResp = new OpenLayers.Format.ArcXML; │ │ │ │ │ + var arcxml = axlResp.read(doc); │ │ │ │ │ + callback.call(scope, this.getUrlOrImage(arcxml.image.output)) │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - getRenderLayerId: function() { │ │ │ │ │ - return this.root.parentNode.parentNode.id │ │ │ │ │ + getUrlOrImage: function(output) { │ │ │ │ │ + var ret = ""; │ │ │ │ │ + if (output.url) { │ │ │ │ │ + ret = output.url │ │ │ │ │ + } else if (output.data) { │ │ │ │ │ + ret = "data:image/" + output.type + ";base64," + output.data │ │ │ │ │ + } │ │ │ │ │ + return ret │ │ │ │ │ }, │ │ │ │ │ - isComplexSymbol: function(graphicName) { │ │ │ │ │ - return graphicName != "circle" && !!graphicName │ │ │ │ │ + setLayerQuery: function(id, querydef) { │ │ │ │ │ + for (var lyr = 0; lyr < this.options.layers.length; lyr++) { │ │ │ │ │ + if (id == this.options.layers[lyr].id) { │ │ │ │ │ + this.options.layers[lyr].query = querydef; │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.options.layers.push({ │ │ │ │ │ + id: id, │ │ │ │ │ + visible: true, │ │ │ │ │ + query: querydef │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.Elements" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Renderer.VML = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ - xmlns: "urn:schemas-microsoft-com:vml", │ │ │ │ │ - symbolCache: {}, │ │ │ │ │ - offset: null, │ │ │ │ │ - initialize: function(containerID) { │ │ │ │ │ - if (!this.supported()) { │ │ │ │ │ - return │ │ │ │ │ + getFeatureInfo: function(geometry, layer, options) { │ │ │ │ │ + var buffer = options.buffer || 1; │ │ │ │ │ + var callback = options.callback || function() {}; │ │ │ │ │ + var scope = options.scope || window; │ │ │ │ │ + var requestOptions = {}; │ │ │ │ │ + OpenLayers.Util.extend(requestOptions, this.options); │ │ │ │ │ + requestOptions.requesttype = "feature"; │ │ │ │ │ + if (geometry instanceof OpenLayers.LonLat) { │ │ │ │ │ + requestOptions.polygon = null; │ │ │ │ │ + requestOptions.envelope = [geometry.lon - buffer, geometry.lat - buffer, geometry.lon + buffer, geometry.lat + buffer] │ │ │ │ │ + } else if (geometry instanceof OpenLayers.Geometry.Polygon) { │ │ │ │ │ + requestOptions.envelope = null; │ │ │ │ │ + requestOptions.polygon = geometry │ │ │ │ │ } │ │ │ │ │ - if (!document.namespaces.olv) { │ │ │ │ │ - document.namespaces.add("olv", this.xmlns); │ │ │ │ │ - var style = document.createStyleSheet(); │ │ │ │ │ - var shapes = ["shape", "rect", "oval", "fill", "stroke", "imagedata", "group", "textbox"]; │ │ │ │ │ - for (var i = 0, len = shapes.length; i < len; i++) { │ │ │ │ │ - style.addRule("olv\\:" + shapes[i], "behavior: url(#default#VML); " + "position: absolute; display: inline-block;") │ │ │ │ │ + var arcxml = new OpenLayers.Format.ArcXML(requestOptions); │ │ │ │ │ + OpenLayers.Util.extend(arcxml.request.get_feature, options); │ │ │ │ │ + arcxml.request.get_feature.layer = layer.id; │ │ │ │ │ + if (typeof layer.query.accuracy == "number") { │ │ │ │ │ + arcxml.request.get_feature.query.accuracy = layer.query.accuracy │ │ │ │ │ + } else { │ │ │ │ │ + var mapCenter = this.map.getCenter(); │ │ │ │ │ + var viewPx = this.map.getViewPortPxFromLonLat(mapCenter); │ │ │ │ │ + viewPx.x++; │ │ │ │ │ + var mapOffCenter = this.map.getLonLatFromPixel(viewPx); │ │ │ │ │ + arcxml.request.get_feature.query.accuracy = mapOffCenter.lon - mapCenter.lon │ │ │ │ │ + } │ │ │ │ │ + arcxml.request.get_feature.query.where = layer.query.where; │ │ │ │ │ + arcxml.request.get_feature.query.spatialfilter.relation = "area_intersection"; │ │ │ │ │ + OpenLayers.Request.POST({ │ │ │ │ │ + url: this.getFullRequestString({ │ │ │ │ │ + CustomService: "Query" │ │ │ │ │ + }), │ │ │ │ │ + data: arcxml.write(), │ │ │ │ │ + callback: function(request) { │ │ │ │ │ + var response = arcxml.parseResponse(request.responseText); │ │ │ │ │ + if (!arcxml.iserror()) { │ │ │ │ │ + callback.call(scope, response.features) │ │ │ │ │ + } else { │ │ │ │ │ + callback.call(scope, null) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.ArcIMS(this.name, this.url, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.initialize.apply(this, arguments) │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - supported: function() { │ │ │ │ │ - return !!document.namespaces │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.ArcIMS" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.PointGrid = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ + dx: null, │ │ │ │ │ + dy: null, │ │ │ │ │ + ratio: 1.5, │ │ │ │ │ + maxFeatures: 250, │ │ │ │ │ + rotation: 0, │ │ │ │ │ + origin: null, │ │ │ │ │ + gridBounds: null, │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ + config = config || {}; │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.initialize.apply(this, [config.name, config]) │ │ │ │ │ }, │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var left = extent.left / resolution | 0; │ │ │ │ │ - var top = extent.top / resolution - this.size.h | 0; │ │ │ │ │ - if (resolutionChanged || !this.offset) { │ │ │ │ │ - this.offset = { │ │ │ │ │ - x: left, │ │ │ │ │ - y: top │ │ │ │ │ - }; │ │ │ │ │ - left = 0; │ │ │ │ │ - top = 0 │ │ │ │ │ - } else { │ │ │ │ │ - left = left - this.offset.x; │ │ │ │ │ - top = top - this.offset.y │ │ │ │ │ - } │ │ │ │ │ - var org = left - this.xOffset + " " + top; │ │ │ │ │ - this.root.coordorigin = org; │ │ │ │ │ - var roots = [this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ - var root; │ │ │ │ │ - for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ - root = roots[i]; │ │ │ │ │ - var size = this.size.w + " " + this.size.h; │ │ │ │ │ - root.coordsize = size │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ + map.events.register("moveend", this, this.onMoveEnd) │ │ │ │ │ + }, │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + map.events.unregister("moveend", this, this.onMoveEnd); │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + setRatio: function(ratio) { │ │ │ │ │ + this.ratio = ratio; │ │ │ │ │ + this.updateGrid(true) │ │ │ │ │ + }, │ │ │ │ │ + setMaxFeatures: function(maxFeatures) { │ │ │ │ │ + this.maxFeatures = maxFeatures; │ │ │ │ │ + this.updateGrid(true) │ │ │ │ │ + }, │ │ │ │ │ + setSpacing: function(dx, dy) { │ │ │ │ │ + this.dx = dx; │ │ │ │ │ + this.dy = dy || dx; │ │ │ │ │ + this.updateGrid(true) │ │ │ │ │ + }, │ │ │ │ │ + setOrigin: function(origin) { │ │ │ │ │ + this.origin = origin; │ │ │ │ │ + this.updateGrid(true) │ │ │ │ │ + }, │ │ │ │ │ + getOrigin: function() { │ │ │ │ │ + if (!this.origin) { │ │ │ │ │ + this.origin = this.map.getExtent().getCenterLonLat() │ │ │ │ │ } │ │ │ │ │ - this.root.style.flip = "y"; │ │ │ │ │ - return coordSysUnchanged │ │ │ │ │ + return this.origin │ │ │ │ │ }, │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ - var roots = [this.rendererRoot, this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ - var w = this.size.w + "px"; │ │ │ │ │ - var h = this.size.h + "px"; │ │ │ │ │ - var root; │ │ │ │ │ - for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ - root = roots[i]; │ │ │ │ │ - root.style.width = w; │ │ │ │ │ - root.style.height = h │ │ │ │ │ + setRotation: function(rotation) { │ │ │ │ │ + this.rotation = rotation; │ │ │ │ │ + this.updateGrid(true) │ │ │ │ │ + }, │ │ │ │ │ + onMoveEnd: function() { │ │ │ │ │ + this.updateGrid() │ │ │ │ │ + }, │ │ │ │ │ + getViewBounds: function() { │ │ │ │ │ + var bounds = this.map.getExtent(); │ │ │ │ │ + if (this.rotation) { │ │ │ │ │ + var origin = this.getOrigin(); │ │ │ │ │ + var rotationOrigin = new OpenLayers.Geometry.Point(origin.lon, origin.lat); │ │ │ │ │ + var rect = bounds.toGeometry(); │ │ │ │ │ + rect.rotate(-this.rotation, rotationOrigin); │ │ │ │ │ + bounds = rect.getBounds() │ │ │ │ │ } │ │ │ │ │ + return bounds │ │ │ │ │ }, │ │ │ │ │ - getNodeType: function(geometry, style) { │ │ │ │ │ - var nodeType = null; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - nodeType = "olv:rect" │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - nodeType = "olv:shape" │ │ │ │ │ - } else { │ │ │ │ │ - nodeType = "olv:oval" │ │ │ │ │ + updateGrid: function(force) { │ │ │ │ │ + if (force || this.invalidBounds()) { │ │ │ │ │ + var viewBounds = this.getViewBounds(); │ │ │ │ │ + var origin = this.getOrigin(); │ │ │ │ │ + var rotationOrigin = new OpenLayers.Geometry.Point(origin.lon, origin.lat); │ │ │ │ │ + var viewBoundsWidth = viewBounds.getWidth(); │ │ │ │ │ + var viewBoundsHeight = viewBounds.getHeight(); │ │ │ │ │ + var aspectRatio = viewBoundsWidth / viewBoundsHeight; │ │ │ │ │ + var maxHeight = Math.sqrt(this.dx * this.dy * this.maxFeatures / aspectRatio); │ │ │ │ │ + var maxWidth = maxHeight * aspectRatio; │ │ │ │ │ + var gridWidth = Math.min(viewBoundsWidth * this.ratio, maxWidth); │ │ │ │ │ + var gridHeight = Math.min(viewBoundsHeight * this.ratio, maxHeight); │ │ │ │ │ + var center = viewBounds.getCenterLonLat(); │ │ │ │ │ + this.gridBounds = new OpenLayers.Bounds(center.lon - gridWidth / 2, center.lat - gridHeight / 2, center.lon + gridWidth / 2, center.lat + gridHeight / 2); │ │ │ │ │ + var rows = Math.floor(gridHeight / this.dy); │ │ │ │ │ + var cols = Math.floor(gridWidth / this.dx); │ │ │ │ │ + var gridLeft = origin.lon + this.dx * Math.ceil((this.gridBounds.left - origin.lon) / this.dx); │ │ │ │ │ + var gridBottom = origin.lat + this.dy * Math.ceil((this.gridBounds.bottom - origin.lat) / this.dy); │ │ │ │ │ + var features = new Array(rows * cols); │ │ │ │ │ + var x, y, point; │ │ │ │ │ + for (var i = 0; i < cols; ++i) { │ │ │ │ │ + x = gridLeft + i * this.dx; │ │ │ │ │ + for (var j = 0; j < rows; ++j) { │ │ │ │ │ + y = gridBottom + j * this.dy; │ │ │ │ │ + point = new OpenLayers.Geometry.Point(x, y); │ │ │ │ │ + if (this.rotation) { │ │ │ │ │ + point.rotate(this.rotation, rotationOrigin) │ │ │ │ │ + } │ │ │ │ │ + features[i * rows + j] = new OpenLayers.Feature.Vector(point) │ │ │ │ │ } │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - nodeType = "olv:rect"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - case "OpenLayers.Geometry.Curve": │ │ │ │ │ - nodeType = "olv:shape"; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break │ │ │ │ │ + } │ │ │ │ │ + this.destroyFeatures(this.features, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.addFeatures(features, { │ │ │ │ │ + silent: true │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - return nodeType │ │ │ │ │ }, │ │ │ │ │ - setStyle: function(node, style, options, geometry) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - options = options || node._options; │ │ │ │ │ - var fillColor = style.fillColor; │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - node.title = title │ │ │ │ │ + invalidBounds: function() { │ │ │ │ │ + return !this.gridBounds || !this.gridBounds.containsBounds(this.getViewBounds()) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.PointGrid" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.TileCache = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + format: "image/png", │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + initialize: function(name, url, layername, options) { │ │ │ │ │ + this.layername = layername; │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name, url, {}, options]); │ │ │ │ │ + this.extension = this.format.split("/")[1].toLowerCase(); │ │ │ │ │ + this.extension = this.extension == "jpg" ? "jpeg" : this.extension │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.TileCache(this.name, this.url, this.layername, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - options.isFilled = true; │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ - var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ - node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x + xOffset | 0) + "px"; │ │ │ │ │ - node.style.top = (geometry.y / resolution - this.offset.y - (yOffset + height) | 0) + "px"; │ │ │ │ │ - node.style.width = width + "px"; │ │ │ │ │ - node.style.height = height + "px"; │ │ │ │ │ - node.style.flip = "y"; │ │ │ │ │ - fillColor = "none"; │ │ │ │ │ - options.isStroked = false │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - var cache = this.importSymbol(style.graphicName); │ │ │ │ │ - node.path = cache.path; │ │ │ │ │ - node.coordorigin = cache.left + "," + cache.bottom; │ │ │ │ │ - var size = cache.size; │ │ │ │ │ - node.coordsize = size + "," + size; │ │ │ │ │ - this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ - node.style.flip = "y" │ │ │ │ │ - } else { │ │ │ │ │ - this.drawCircle(node, geometry, style.pointRadius) │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var bbox = this.maxExtent; │ │ │ │ │ + var size = this.tileSize; │ │ │ │ │ + var tileX = Math.round((bounds.left - bbox.left) / (res * size.w)); │ │ │ │ │ + var tileY = Math.round((bounds.bottom - bbox.bottom) / (res * size.h)); │ │ │ │ │ + var tileZ = this.serverResolutions != null ? OpenLayers.Util.indexOf(this.serverResolutions, res) : this.map.getZoom(); │ │ │ │ │ + var components = [this.layername, OpenLayers.Number.zeroPad(tileZ, 2), OpenLayers.Number.zeroPad(parseInt(tileX / 1e6), 3), OpenLayers.Number.zeroPad(parseInt(tileX / 1e3) % 1e3, 3), OpenLayers.Number.zeroPad(parseInt(tileX) % 1e3, 3), OpenLayers.Number.zeroPad(parseInt(tileY / 1e6), 3), OpenLayers.Number.zeroPad(parseInt(tileY / 1e3) % 1e3, 3), OpenLayers.Number.zeroPad(parseInt(tileY) % 1e3, 3) + "." + this.extension]; │ │ │ │ │ + var path = components.join("/"); │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(path, url) │ │ │ │ │ + } │ │ │ │ │ + url = url.charAt(url.length - 1) == "/" ? url : url + "/"; │ │ │ │ │ + return url + path │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.TileCache" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.WMTS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + requestEncoding: "KVP", │ │ │ │ │ + url: null, │ │ │ │ │ + layer: null, │ │ │ │ │ + matrixSet: null, │ │ │ │ │ + style: null, │ │ │ │ │ + format: "image/jpeg", │ │ │ │ │ + tileOrigin: null, │ │ │ │ │ + tileFullExtent: null, │ │ │ │ │ + formatSuffix: null, │ │ │ │ │ + matrixIds: null, │ │ │ │ │ + dimensions: null, │ │ │ │ │ + params: null, │ │ │ │ │ + zoomOffset: 0, │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + formatSuffixMap: { │ │ │ │ │ + "image/png": "png", │ │ │ │ │ + "image/png8": "png", │ │ │ │ │ + "image/png24": "png", │ │ │ │ │ + "image/png32": "png", │ │ │ │ │ + png: "png", │ │ │ │ │ + "image/jpeg": "jpg", │ │ │ │ │ + "image/jpg": "jpg", │ │ │ │ │ + jpeg: "jpg", │ │ │ │ │ + jpg: "jpg" │ │ │ │ │ + }, │ │ │ │ │ + matrix: null, │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ + var required = { │ │ │ │ │ + url: true, │ │ │ │ │ + layer: true, │ │ │ │ │ + style: true, │ │ │ │ │ + matrixSet: true │ │ │ │ │ + }; │ │ │ │ │ + for (var prop in required) { │ │ │ │ │ + if (!(prop in config)) { │ │ │ │ │ + throw new Error("Missing property '" + prop + "' in layer configuration.") │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (options.isFilled) { │ │ │ │ │ - node.fillcolor = fillColor │ │ │ │ │ - } else { │ │ │ │ │ - node.filled = "false" │ │ │ │ │ + config.params = OpenLayers.Util.upperCaseObject(config.params); │ │ │ │ │ + var args = [config.name, config.url, config.params, config]; │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, args); │ │ │ │ │ + if (!this.formatSuffix) { │ │ │ │ │ + this.formatSuffix = this.formatSuffixMap[this.format] || this.format.split("/").pop() │ │ │ │ │ } │ │ │ │ │ - var fills = node.getElementsByTagName("fill"); │ │ │ │ │ - var fill = fills.length == 0 ? null : fills[0]; │ │ │ │ │ - if (!options.isFilled) { │ │ │ │ │ - if (fill) { │ │ │ │ │ - node.removeChild(fill) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - if (!fill) { │ │ │ │ │ - fill = this.createNode("olv:fill", node.id + "_fill") │ │ │ │ │ - } │ │ │ │ │ - fill.opacity = style.fillOpacity; │ │ │ │ │ - if (node._geometryClass === "OpenLayers.Geometry.Point" && style.externalGraphic) { │ │ │ │ │ - if (style.graphicOpacity) { │ │ │ │ │ - fill.opacity = style.graphicOpacity │ │ │ │ │ - } │ │ │ │ │ - fill.src = style.externalGraphic; │ │ │ │ │ - fill.type = "frame"; │ │ │ │ │ - if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ - fill.aspect = "atmost" │ │ │ │ │ + if (this.matrixIds) { │ │ │ │ │ + var len = this.matrixIds.length; │ │ │ │ │ + if (len && typeof this.matrixIds[0] === "string") { │ │ │ │ │ + var ids = this.matrixIds; │ │ │ │ │ + this.matrixIds = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + this.matrixIds[i] = { │ │ │ │ │ + identifier: ids[i] │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (fill.parentNode != node) { │ │ │ │ │ - node.appendChild(fill) │ │ │ │ │ - } │ │ │ │ │ } │ │ │ │ │ - var rotation = style.rotation; │ │ │ │ │ - if (rotation !== undefined || node._rotation !== undefined) { │ │ │ │ │ - node._rotation = rotation; │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ - fill.opacity = 0 │ │ │ │ │ - } else if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ - node.style.rotation = rotation || 0 │ │ │ │ │ + }, │ │ │ │ │ + setMap: function() { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + updateMatrixProperties: function() { │ │ │ │ │ + this.matrix = this.getMatrix(); │ │ │ │ │ + if (this.matrix) { │ │ │ │ │ + if (this.matrix.topLeftCorner) { │ │ │ │ │ + this.tileOrigin = this.matrix.topLeftCorner │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - var strokes = node.getElementsByTagName("stroke"); │ │ │ │ │ - var stroke = strokes.length == 0 ? null : strokes[0]; │ │ │ │ │ - if (!options.isStroked) { │ │ │ │ │ - node.stroked = false; │ │ │ │ │ - if (stroke) { │ │ │ │ │ - stroke.on = false │ │ │ │ │ + if (this.matrix.tileWidth && this.matrix.tileHeight) { │ │ │ │ │ + this.tileSize = new OpenLayers.Size(this.matrix.tileWidth, this.matrix.tileHeight) │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - if (!stroke) { │ │ │ │ │ - stroke = this.createNode("olv:stroke", node.id + "_stroke"); │ │ │ │ │ - node.appendChild(stroke) │ │ │ │ │ + if (!this.tileOrigin) { │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, this.maxExtent.top) │ │ │ │ │ } │ │ │ │ │ - stroke.on = true; │ │ │ │ │ - stroke.color = style.strokeColor; │ │ │ │ │ - stroke.weight = style.strokeWidth + "px"; │ │ │ │ │ - stroke.opacity = style.strokeOpacity; │ │ │ │ │ - stroke.endcap = style.strokeLinecap == "butt" ? "flat" : style.strokeLinecap || "round"; │ │ │ │ │ - if (style.strokeDashstyle) { │ │ │ │ │ - stroke.dashstyle = this.dashStyle(style) │ │ │ │ │ + if (!this.tileFullExtent) { │ │ │ │ │ + this.tileFullExtent = this.maxExtent │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ - node.style.cursor = style.cursor │ │ │ │ │ - } │ │ │ │ │ - return node │ │ │ │ │ }, │ │ │ │ │ - graphicRotate: function(node, xOffset, yOffset, style) { │ │ │ │ │ - var style = style || node._style; │ │ │ │ │ - var rotation = style.rotation || 0; │ │ │ │ │ - var aspectRatio, size; │ │ │ │ │ - if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ - var img = new Image; │ │ │ │ │ - img.onreadystatechange = OpenLayers.Function.bind(function() { │ │ │ │ │ - if (img.readyState == "complete" || img.readyState == "interactive") { │ │ │ │ │ - aspectRatio = img.width / img.height; │ │ │ │ │ - size = Math.max(style.pointRadius * 2, style.graphicWidth || 0, style.graphicHeight || 0); │ │ │ │ │ - xOffset = xOffset * aspectRatio; │ │ │ │ │ - style.graphicWidth = size * aspectRatio; │ │ │ │ │ - style.graphicHeight = size; │ │ │ │ │ - this.graphicRotate(node, xOffset, yOffset, style) │ │ │ │ │ - } │ │ │ │ │ - }, this); │ │ │ │ │ - img.src = style.externalGraphic; │ │ │ │ │ - return │ │ │ │ │ - } else { │ │ │ │ │ - size = Math.max(style.graphicWidth, style.graphicHeight); │ │ │ │ │ - aspectRatio = style.graphicWidth / style.graphicHeight │ │ │ │ │ - } │ │ │ │ │ - var width = Math.round(style.graphicWidth || size * aspectRatio); │ │ │ │ │ - var height = Math.round(style.graphicHeight || size); │ │ │ │ │ - node.style.width = width + "px"; │ │ │ │ │ - node.style.height = height + "px"; │ │ │ │ │ - var image = document.getElementById(node.id + "_image"); │ │ │ │ │ - if (!image) { │ │ │ │ │ - image = this.createNode("olv:imagedata", node.id + "_image"); │ │ │ │ │ - node.appendChild(image) │ │ │ │ │ - } │ │ │ │ │ - image.style.width = width + "px"; │ │ │ │ │ - image.style.height = height + "px"; │ │ │ │ │ - image.src = style.externalGraphic; │ │ │ │ │ - image.style.filter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(" + "src='', sizingMethod='scale')"; │ │ │ │ │ - var rot = rotation * Math.PI / 180; │ │ │ │ │ - var sintheta = Math.sin(rot); │ │ │ │ │ - var costheta = Math.cos(rot); │ │ │ │ │ - var filter = "progid:DXImageTransform.Microsoft.Matrix(M11=" + costheta + ",M12=" + -sintheta + ",M21=" + sintheta + ",M22=" + costheta + ",SizingMethod='auto expand')\n"; │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ - if (opacity && opacity != 1) { │ │ │ │ │ - filter += "progid:DXImageTransform.Microsoft.BasicImage(opacity=" + opacity + ")\n" │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + if (zoomChanged || !this.matrix) { │ │ │ │ │ + this.updateMatrixProperties() │ │ │ │ │ } │ │ │ │ │ - node.style.filter = filter; │ │ │ │ │ - var centerPoint = new OpenLayers.Geometry.Point(-xOffset, -yOffset); │ │ │ │ │ - var imgBox = new OpenLayers.Bounds(0, 0, width, height).toGeometry(); │ │ │ │ │ - imgBox.rotate(style.rotation, centerPoint); │ │ │ │ │ - var imgBounds = imgBox.getBounds(); │ │ │ │ │ - node.style.left = Math.round(parseInt(node.style.left) + imgBounds.left) + "px"; │ │ │ │ │ - node.style.top = Math.round(parseInt(node.style.top) - imgBounds.bottom) + "px" │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.moveTo.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - postDraw: function(node) { │ │ │ │ │ - node.style.visibility = "visible"; │ │ │ │ │ - var fillColor = node._style.fillColor; │ │ │ │ │ - var strokeColor = node._style.strokeColor; │ │ │ │ │ - if (fillColor == "none" && node.fillcolor != fillColor) { │ │ │ │ │ - node.fillcolor = fillColor │ │ │ │ │ - } │ │ │ │ │ - if (strokeColor == "none" && node.strokecolor != strokeColor) { │ │ │ │ │ - node.strokecolor = strokeColor │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.WMTS(this.options) │ │ │ │ │ } │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - setNodeDimension: function(node, geometry) { │ │ │ │ │ - var bbox = geometry.getBounds(); │ │ │ │ │ - if (bbox) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var scaledBox = new OpenLayers.Bounds((bbox.left - this.featureDx) / resolution - this.offset.x | 0, bbox.bottom / resolution - this.offset.y | 0, (bbox.right - this.featureDx) / resolution - this.offset.x | 0, bbox.top / resolution - this.offset.y | 0); │ │ │ │ │ - node.style.left = scaledBox.left + "px"; │ │ │ │ │ - node.style.top = scaledBox.top + "px"; │ │ │ │ │ - node.style.width = scaledBox.getWidth() + "px"; │ │ │ │ │ - node.style.height = scaledBox.getHeight() + "px"; │ │ │ │ │ - node.coordorigin = scaledBox.left + " " + scaledBox.top; │ │ │ │ │ - node.coordsize = scaledBox.getWidth() + " " + scaledBox.getHeight() │ │ │ │ │ - } │ │ │ │ │ + getIdentifier: function() { │ │ │ │ │ + return this.getServerZoom() │ │ │ │ │ }, │ │ │ │ │ - dashStyle: function(style) { │ │ │ │ │ - var dash = style.strokeDashstyle; │ │ │ │ │ - switch (dash) { │ │ │ │ │ - case "solid": │ │ │ │ │ - case "dot": │ │ │ │ │ - case "dash": │ │ │ │ │ - case "dashdot": │ │ │ │ │ - case "longdash": │ │ │ │ │ - case "longdashdot": │ │ │ │ │ - return dash; │ │ │ │ │ - default: │ │ │ │ │ - var parts = dash.split(/[ ,]/); │ │ │ │ │ - if (parts.length == 2) { │ │ │ │ │ - if (1 * parts[0] >= 2 * parts[1]) { │ │ │ │ │ - return "longdash" │ │ │ │ │ + getMatrix: function() { │ │ │ │ │ + var matrix; │ │ │ │ │ + if (!this.matrixIds || this.matrixIds.length === 0) { │ │ │ │ │ + matrix = { │ │ │ │ │ + identifier: this.getIdentifier() │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + if ("scaleDenominator" in this.matrixIds[0]) { │ │ │ │ │ + var denom = OpenLayers.METERS_PER_INCH * OpenLayers.INCHES_PER_UNIT[this.units] * this.getServerResolution() / 28e-5; │ │ │ │ │ + var diff = Number.POSITIVE_INFINITY; │ │ │ │ │ + var delta; │ │ │ │ │ + for (var i = 0, ii = this.matrixIds.length; i < ii; ++i) { │ │ │ │ │ + delta = Math.abs(1 - this.matrixIds[i].scaleDenominator / denom); │ │ │ │ │ + if (delta < diff) { │ │ │ │ │ + diff = delta; │ │ │ │ │ + matrix = this.matrixIds[i] │ │ │ │ │ } │ │ │ │ │ - return parts[0] == 1 || parts[1] == 1 ? "dot" : "dash" │ │ │ │ │ - } else if (parts.length == 4) { │ │ │ │ │ - return 1 * parts[0] >= 2 * parts[1] ? "longdashdot" : "dashdot" │ │ │ │ │ } │ │ │ │ │ - return "solid" │ │ │ │ │ + } else { │ │ │ │ │ + matrix = this.matrixIds[this.getIdentifier()] │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return matrix │ │ │ │ │ }, │ │ │ │ │ - createNode: function(type, id) { │ │ │ │ │ - var node = document.createElement(type); │ │ │ │ │ - if (id) { │ │ │ │ │ - node.id = id │ │ │ │ │ + getTileInfo: function(loc) { │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var fx = (loc.lon - this.tileOrigin.lon) / (res * this.tileSize.w); │ │ │ │ │ + var fy = (this.tileOrigin.lat - loc.lat) / (res * this.tileSize.h); │ │ │ │ │ + var col = Math.floor(fx); │ │ │ │ │ + var row = Math.floor(fy); │ │ │ │ │ + return { │ │ │ │ │ + col: col, │ │ │ │ │ + row: row, │ │ │ │ │ + i: Math.floor((fx - col) * this.tileSize.w), │ │ │ │ │ + j: Math.floor((fy - row) * this.tileSize.h) │ │ │ │ │ } │ │ │ │ │ - node.unselectable = "on"; │ │ │ │ │ - node.onselectstart = OpenLayers.Function.False; │ │ │ │ │ - return node │ │ │ │ │ }, │ │ │ │ │ - nodeTypeCompare: function(node, type) { │ │ │ │ │ - var subType = type; │ │ │ │ │ - var splitIndex = subType.indexOf(":"); │ │ │ │ │ - if (splitIndex != -1) { │ │ │ │ │ - subType = subType.substr(splitIndex + 1) │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var url = ""; │ │ │ │ │ + if (!this.tileFullExtent || this.tileFullExtent.intersectsBounds(bounds)) { │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var info = this.getTileInfo(center); │ │ │ │ │ + var matrixId = this.matrix.identifier; │ │ │ │ │ + var dimensions = this.dimensions, │ │ │ │ │ + params; │ │ │ │ │ + if (OpenLayers.Util.isArray(this.url)) { │ │ │ │ │ + url = this.selectUrl([this.version, this.style, this.matrixSet, this.matrix.identifier, info.row, info.col].join(","), this.url) │ │ │ │ │ + } else { │ │ │ │ │ + url = this.url │ │ │ │ │ + } │ │ │ │ │ + if (this.requestEncoding.toUpperCase() === "REST") { │ │ │ │ │ + params = this.params; │ │ │ │ │ + if (url.indexOf("{") !== -1) { │ │ │ │ │ + var template = url.replace(/\{/g, "${"); │ │ │ │ │ + var context = { │ │ │ │ │ + style: this.style, │ │ │ │ │ + Style: this.style, │ │ │ │ │ + TileMatrixSet: this.matrixSet, │ │ │ │ │ + TileMatrix: this.matrix.identifier, │ │ │ │ │ + TileRow: info.row, │ │ │ │ │ + TileCol: info.col │ │ │ │ │ + }; │ │ │ │ │ + if (dimensions) { │ │ │ │ │ + var dimension, i; │ │ │ │ │ + for (i = dimensions.length - 1; i >= 0; --i) { │ │ │ │ │ + dimension = dimensions[i]; │ │ │ │ │ + context[dimension] = params[dimension.toUpperCase()] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + url = OpenLayers.String.format(template, context) │ │ │ │ │ + } else { │ │ │ │ │ + var path = this.version + "/" + this.layer + "/" + this.style + "/"; │ │ │ │ │ + if (dimensions) { │ │ │ │ │ + for (var i = 0; i < dimensions.length; i++) { │ │ │ │ │ + if (params[dimensions[i]]) { │ │ │ │ │ + path = path + params[dimensions[i]] + "/" │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + path = path + this.matrixSet + "/" + this.matrix.identifier + "/" + info.row + "/" + info.col + "." + this.formatSuffix; │ │ │ │ │ + if (!url.match(/\/$/)) { │ │ │ │ │ + url = url + "/" │ │ │ │ │ + } │ │ │ │ │ + url = url + path │ │ │ │ │ + } │ │ │ │ │ + } else if (this.requestEncoding.toUpperCase() === "KVP") { │ │ │ │ │ + params = { │ │ │ │ │ + SERVICE: "WMTS", │ │ │ │ │ + REQUEST: "GetTile", │ │ │ │ │ + VERSION: this.version, │ │ │ │ │ + LAYER: this.layer, │ │ │ │ │ + STYLE: this.style, │ │ │ │ │ + TILEMATRIXSET: this.matrixSet, │ │ │ │ │ + TILEMATRIX: this.matrix.identifier, │ │ │ │ │ + TILEROW: info.row, │ │ │ │ │ + TILECOL: info.col, │ │ │ │ │ + FORMAT: this.format │ │ │ │ │ + }; │ │ │ │ │ + url = OpenLayers.Layer.Grid.prototype.getFullRequestString.apply(this, [params]) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var nodeName = node.nodeName; │ │ │ │ │ - splitIndex = nodeName.indexOf(":"); │ │ │ │ │ - if (splitIndex != -1) { │ │ │ │ │ - nodeName = nodeName.substr(splitIndex + 1) │ │ │ │ │ + return url │ │ │ │ │ + }, │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + if (this.requestEncoding.toUpperCase() === "KVP") { │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, [OpenLayers.Util.upperCaseObject(newParams)]) │ │ │ │ │ } │ │ │ │ │ - return subType == nodeName │ │ │ │ │ }, │ │ │ │ │ - createRenderRoot: function() { │ │ │ │ │ - return this.nodeFactory(this.container.id + "_vmlRoot", "div") │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.WMTS" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.EventPane = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ + smoothDragPan: true, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + isFixed: true, │ │ │ │ │ + pane: null, │ │ │ │ │ + mapObject: null, │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (this.pane == null) { │ │ │ │ │ + this.pane = OpenLayers.Util.createDiv(this.div.id + "_EventPane") │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - createRoot: function(suffix) { │ │ │ │ │ - return this.nodeFactory(this.container.id + suffix, "olv:group") │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.mapObject = null; │ │ │ │ │ + this.pane = null; │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - drawPoint: function(node, geometry) { │ │ │ │ │ - return this.drawCircle(node, geometry, 1) │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.pane.style.zIndex = parseInt(this.div.style.zIndex) + 1; │ │ │ │ │ + this.pane.style.display = this.div.style.display; │ │ │ │ │ + this.pane.style.width = "100%"; │ │ │ │ │ + this.pane.style.height = "100%"; │ │ │ │ │ + if (OpenLayers.BROWSER_NAME == "msie") { │ │ │ │ │ + this.pane.style.background = "url(" + OpenLayers.Util.getImageLocation("blank.gif") + ")" │ │ │ │ │ + } │ │ │ │ │ + if (this.isFixed) { │ │ │ │ │ + this.map.viewPortDiv.appendChild(this.pane) │ │ │ │ │ + } else { │ │ │ │ │ + this.map.layerContainerDiv.appendChild(this.pane) │ │ │ │ │ + } │ │ │ │ │ + this.loadMapObject(); │ │ │ │ │ + if (this.mapObject == null) { │ │ │ │ │ + this.loadWarningMessage() │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - drawCircle: function(node, geometry, radius) { │ │ │ │ │ - if (!isNaN(geometry.x) && !isNaN(geometry.y)) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x | 0) - radius + "px"; │ │ │ │ │ - node.style.top = (geometry.y / resolution - this.offset.y | 0) - radius + "px"; │ │ │ │ │ - var diameter = radius * 2; │ │ │ │ │ - node.style.width = diameter + "px"; │ │ │ │ │ - node.style.height = diameter + "px"; │ │ │ │ │ - return node │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + if (this.pane && this.pane.parentNode) { │ │ │ │ │ + this.pane.parentNode.removeChild(this.pane) │ │ │ │ │ } │ │ │ │ │ - return false │ │ │ │ │ + OpenLayers.Layer.prototype.removeMap.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - drawLineString: function(node, geometry) { │ │ │ │ │ - return this.drawLine(node, geometry, false) │ │ │ │ │ + loadWarningMessage: function() { │ │ │ │ │ + this.div.style.backgroundColor = "darkblue"; │ │ │ │ │ + var viewSize = this.map.getSize(); │ │ │ │ │ + var msgW = Math.min(viewSize.w, 300); │ │ │ │ │ + var msgH = Math.min(viewSize.h, 200); │ │ │ │ │ + var size = new OpenLayers.Size(msgW, msgH); │ │ │ │ │ + var centerPx = new OpenLayers.Pixel(viewSize.w / 2, viewSize.h / 2); │ │ │ │ │ + var topLeft = centerPx.add(-size.w / 2, -size.h / 2); │ │ │ │ │ + var div = OpenLayers.Util.createDiv(this.name + "_warning", topLeft, size, null, null, null, "auto"); │ │ │ │ │ + div.style.padding = "7px"; │ │ │ │ │ + div.style.backgroundColor = "yellow"; │ │ │ │ │ + div.innerHTML = this.getWarningHTML(); │ │ │ │ │ + this.div.appendChild(div) │ │ │ │ │ }, │ │ │ │ │ - drawLinearRing: function(node, geometry) { │ │ │ │ │ - return this.drawLine(node, geometry, true) │ │ │ │ │ + getWarningHTML: function() { │ │ │ │ │ + return "" │ │ │ │ │ }, │ │ │ │ │ - drawLine: function(node, geometry, closeLine) { │ │ │ │ │ - this.setNodeDimension(node, geometry); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var numComponents = geometry.components.length; │ │ │ │ │ - var parts = new Array(numComponents); │ │ │ │ │ - var comp, x, y; │ │ │ │ │ - for (var i = 0; i < numComponents; i++) { │ │ │ │ │ - comp = geometry.components[i]; │ │ │ │ │ - x = (comp.x - this.featureDx) / resolution - this.offset.x | 0; │ │ │ │ │ - y = comp.y / resolution - this.offset.y | 0; │ │ │ │ │ - parts[i] = " " + x + "," + y + " l " │ │ │ │ │ + display: function(display) { │ │ │ │ │ + OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ + this.pane.style.display = this.div.style.display │ │ │ │ │ + }, │ │ │ │ │ + setZIndex: function(zIndex) { │ │ │ │ │ + OpenLayers.Layer.prototype.setZIndex.apply(this, arguments); │ │ │ │ │ + this.pane.style.zIndex = parseInt(this.div.style.zIndex) + 1 │ │ │ │ │ + }, │ │ │ │ │ + moveByPx: function(dx, dy) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveByPx.apply(this, arguments); │ │ │ │ │ + if (this.dragPanMapObject) { │ │ │ │ │ + this.dragPanMapObject(dx, -dy) │ │ │ │ │ + } else { │ │ │ │ │ + this.moveTo(this.map.getCachedCenter()) │ │ │ │ │ } │ │ │ │ │ - var end = closeLine ? " x e" : " e"; │ │ │ │ │ - node.path = "m" + parts.join("") + end; │ │ │ │ │ - return node │ │ │ │ │ }, │ │ │ │ │ - drawPolygon: function(node, geometry) { │ │ │ │ │ - this.setNodeDimension(node, geometry); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var path = []; │ │ │ │ │ - var j, jj, points, area, first, second, i, ii, comp, pathComp, x, y; │ │ │ │ │ - for (j = 0, jj = geometry.components.length; j < jj; j++) { │ │ │ │ │ - path.push("m"); │ │ │ │ │ - points = geometry.components[j].components; │ │ │ │ │ - area = j === 0; │ │ │ │ │ - first = null; │ │ │ │ │ - second = null; │ │ │ │ │ - for (i = 0, ii = points.length; i < ii; i++) { │ │ │ │ │ - comp = points[i]; │ │ │ │ │ - x = (comp.x - this.featureDx) / resolution - this.offset.x | 0; │ │ │ │ │ - y = comp.y / resolution - this.offset.y | 0; │ │ │ │ │ - pathComp = " " + x + "," + y; │ │ │ │ │ - path.push(pathComp); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - path.push(" l") │ │ │ │ │ - } │ │ │ │ │ - if (!area) { │ │ │ │ │ - if (!first) { │ │ │ │ │ - first = pathComp │ │ │ │ │ - } else if (first != pathComp) { │ │ │ │ │ - if (!second) { │ │ │ │ │ - second = pathComp │ │ │ │ │ - } else if (second != pathComp) { │ │ │ │ │ - area = true │ │ │ │ │ - } │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + if (this.mapObject != null) { │ │ │ │ │ + var newCenter = this.map.getCenter(); │ │ │ │ │ + var newZoom = this.map.getZoom(); │ │ │ │ │ + if (newCenter != null) { │ │ │ │ │ + var moOldCenter = this.getMapObjectCenter(); │ │ │ │ │ + var oldCenter = this.getOLLonLatFromMapObjectLonLat(moOldCenter); │ │ │ │ │ + var moOldZoom = this.getMapObjectZoom(); │ │ │ │ │ + var oldZoom = this.getOLZoomFromMapObjectZoom(moOldZoom); │ │ │ │ │ + if (!newCenter.equals(oldCenter) || newZoom != oldZoom) { │ │ │ │ │ + if (!zoomChanged && oldCenter && this.dragPanMapObject && this.smoothDragPan) { │ │ │ │ │ + var oldPx = this.map.getViewPortPxFromLonLat(oldCenter); │ │ │ │ │ + var newPx = this.map.getViewPortPxFromLonLat(newCenter); │ │ │ │ │ + this.dragPanMapObject(newPx.x - oldPx.x, oldPx.y - newPx.y) │ │ │ │ │ + } else { │ │ │ │ │ + var center = this.getMapObjectLonLatFromOLLonLat(newCenter); │ │ │ │ │ + var zoom = this.getMapObjectZoomFromOLZoom(newZoom); │ │ │ │ │ + this.setMapObjectCenter(center, zoom, dragging) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - path.push(area ? " x " : " ") │ │ │ │ │ } │ │ │ │ │ - path.push("e"); │ │ │ │ │ - node.path = path.join(""); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - drawRectangle: function(node, geometry) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x | 0) + "px"; │ │ │ │ │ - node.style.top = (geometry.y / resolution - this.offset.y | 0) + "px"; │ │ │ │ │ - node.style.width = (geometry.width / resolution | 0) + "px"; │ │ │ │ │ - node.style.height = (geometry.height / resolution | 0) + "px"; │ │ │ │ │ - return node │ │ │ │ │ }, │ │ │ │ │ - drawText: function(featureId, style, location) { │ │ │ │ │ - var label = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX, "olv:rect"); │ │ │ │ │ - var textbox = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX + "_textbox", "olv:textbox"); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - label.style.left = ((location.x - this.featureDx) / resolution - this.offset.x | 0) + "px"; │ │ │ │ │ - label.style.top = (location.y / resolution - this.offset.y | 0) + "px"; │ │ │ │ │ - label.style.flip = "y"; │ │ │ │ │ - textbox.innerText = style.label; │ │ │ │ │ - if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ - textbox.style.cursor = style.cursor │ │ │ │ │ - } │ │ │ │ │ - if (style.fontColor) { │ │ │ │ │ - textbox.style.color = style.fontColor │ │ │ │ │ - } │ │ │ │ │ - if (style.fontOpacity) { │ │ │ │ │ - textbox.style.filter = "alpha(opacity=" + style.fontOpacity * 100 + ")" │ │ │ │ │ + getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ + var lonlat = null; │ │ │ │ │ + if (this.mapObject != null && this.getMapObjectCenter() != null) { │ │ │ │ │ + var moPixel = this.getMapObjectPixelFromOLPixel(viewPortPx); │ │ │ │ │ + var moLonLat = this.getMapObjectLonLatFromMapObjectPixel(moPixel); │ │ │ │ │ + lonlat = this.getOLLonLatFromMapObjectLonLat(moLonLat) │ │ │ │ │ } │ │ │ │ │ - if (style.fontFamily) { │ │ │ │ │ - textbox.style.fontFamily = style.fontFamily │ │ │ │ │ + return lonlat │ │ │ │ │ + }, │ │ │ │ │ + getViewPortPxFromLonLat: function(lonlat) { │ │ │ │ │ + var viewPortPx = null; │ │ │ │ │ + if (this.mapObject != null && this.getMapObjectCenter() != null) { │ │ │ │ │ + var moLonLat = this.getMapObjectLonLatFromOLLonLat(lonlat); │ │ │ │ │ + var moPixel = this.getMapObjectPixelFromMapObjectLonLat(moLonLat); │ │ │ │ │ + viewPortPx = this.getOLPixelFromMapObjectPixel(moPixel) │ │ │ │ │ } │ │ │ │ │ - if (style.fontSize) { │ │ │ │ │ - textbox.style.fontSize = style.fontSize │ │ │ │ │ + return viewPortPx │ │ │ │ │ + }, │ │ │ │ │ + getOLLonLatFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + var olLonLat = null; │ │ │ │ │ + if (moLonLat != null) { │ │ │ │ │ + var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + olLonLat = new OpenLayers.LonLat(lon, lat) │ │ │ │ │ } │ │ │ │ │ - if (style.fontWeight) { │ │ │ │ │ - textbox.style.fontWeight = style.fontWeight │ │ │ │ │ + return olLonLat │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectLonLatFromOLLonLat: function(olLonLat) { │ │ │ │ │ + var moLatLng = null; │ │ │ │ │ + if (olLonLat != null) { │ │ │ │ │ + moLatLng = this.getMapObjectLonLatFromLonLat(olLonLat.lon, olLonLat.lat) │ │ │ │ │ } │ │ │ │ │ - if (style.fontStyle) { │ │ │ │ │ - textbox.style.fontStyle = style.fontStyle │ │ │ │ │ + return moLatLng │ │ │ │ │ + }, │ │ │ │ │ + getOLPixelFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + var olPixel = null; │ │ │ │ │ + if (moPixel != null) { │ │ │ │ │ + var x = this.getXFromMapObjectPixel(moPixel); │ │ │ │ │ + var y = this.getYFromMapObjectPixel(moPixel); │ │ │ │ │ + olPixel = new OpenLayers.Pixel(x, y) │ │ │ │ │ } │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - label._featureId = featureId; │ │ │ │ │ - textbox._featureId = featureId; │ │ │ │ │ - textbox._geometry = location; │ │ │ │ │ - textbox._geometryClass = location.CLASS_NAME │ │ │ │ │ + return olPixel │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectPixelFromOLPixel: function(olPixel) { │ │ │ │ │ + var moPixel = null; │ │ │ │ │ + if (olPixel != null) { │ │ │ │ │ + moPixel = this.getMapObjectPixelFromXY(olPixel.x, olPixel.y) │ │ │ │ │ } │ │ │ │ │ - textbox.style.whiteSpace = "nowrap"; │ │ │ │ │ - textbox.inset = "1px,0px,0px,0px"; │ │ │ │ │ - if (!label.parentNode) { │ │ │ │ │ - label.appendChild(textbox); │ │ │ │ │ - this.textRoot.appendChild(label) │ │ │ │ │ + return moPixel │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.EventPane" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.PointTrack = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ + dataFrom: null, │ │ │ │ │ + styleFrom: null, │ │ │ │ │ + addNodes: function(pointFeatures, options) { │ │ │ │ │ + if (pointFeatures.length < 2) { │ │ │ │ │ + throw new Error("At least two point features have to be added to " + "create a line from") │ │ │ │ │ } │ │ │ │ │ - var align = style.labelAlign || "cm"; │ │ │ │ │ - if (align.length == 1) { │ │ │ │ │ - align += "m" │ │ │ │ │ + var lines = new Array(pointFeatures.length - 1); │ │ │ │ │ + var pointFeature, startPoint, endPoint; │ │ │ │ │ + for (var i = 0, len = pointFeatures.length; i < len; i++) { │ │ │ │ │ + pointFeature = pointFeatures[i]; │ │ │ │ │ + endPoint = pointFeature.geometry; │ │ │ │ │ + if (!endPoint) { │ │ │ │ │ + var lonlat = pointFeature.lonlat; │ │ │ │ │ + endPoint = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat) │ │ │ │ │ + } else if (endPoint.CLASS_NAME != "OpenLayers.Geometry.Point") { │ │ │ │ │ + throw new TypeError("Only features with point geometries are supported.") │ │ │ │ │ + } │ │ │ │ │ + if (i > 0) { │ │ │ │ │ + var attributes = this.dataFrom != null ? pointFeatures[i + this.dataFrom].data || pointFeatures[i + this.dataFrom].attributes : null; │ │ │ │ │ + var style = this.styleFrom != null ? pointFeatures[i + this.styleFrom].style : null; │ │ │ │ │ + var line = new OpenLayers.Geometry.LineString([startPoint, endPoint]); │ │ │ │ │ + lines[i - 1] = new OpenLayers.Feature.Vector(line, attributes, style) │ │ │ │ │ + } │ │ │ │ │ + startPoint = endPoint │ │ │ │ │ } │ │ │ │ │ - var xshift = textbox.clientWidth * OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(0, 1)]; │ │ │ │ │ - var yshift = textbox.clientHeight * OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(1, 1)]; │ │ │ │ │ - label.style.left = parseInt(label.style.left) - xshift - 1 + "px"; │ │ │ │ │ - label.style.top = parseInt(label.style.top) + yshift + "px" │ │ │ │ │ + this.addFeatures(lines, options) │ │ │ │ │ }, │ │ │ │ │ - moveRoot: function(renderer) { │ │ │ │ │ - var layer = this.map.getLayer(renderer.container.id); │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Vector.RootContainer) { │ │ │ │ │ - layer = this.map.getLayer(this.container.id) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.PointTrack" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.PointTrack.SOURCE_NODE = -1; │ │ │ │ │ +OpenLayers.Layer.PointTrack.TARGET_NODE = 0; │ │ │ │ │ +OpenLayers.Layer.PointTrack.dataFrom = { │ │ │ │ │ + SOURCE_NODE: -1, │ │ │ │ │ + TARGET_NODE: 0 │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Layer.FixedZoomLevels = OpenLayers.Class({ │ │ │ │ │ + initialize: function() {}, │ │ │ │ │ + initResolutions: function() { │ │ │ │ │ + var props = ["minZoomLevel", "maxZoomLevel", "numZoomLevels"]; │ │ │ │ │ + for (var i = 0, len = props.length; i < len; i++) { │ │ │ │ │ + var property = props[i]; │ │ │ │ │ + this[property] = this.options[property] != null ? this.options[property] : this.map[property] │ │ │ │ │ } │ │ │ │ │ - layer && layer.renderer.clear(); │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.moveRoot.apply(this, arguments); │ │ │ │ │ - layer && layer.redraw() │ │ │ │ │ - }, │ │ │ │ │ - importSymbol: function(graphicName) { │ │ │ │ │ - var id = this.container.id + "-" + graphicName; │ │ │ │ │ - var cache = this.symbolCache[id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - return cache │ │ │ │ │ + if (this.minZoomLevel == null || this.minZoomLevel < this.MIN_ZOOM_LEVEL) { │ │ │ │ │ + this.minZoomLevel = this.MIN_ZOOM_LEVEL │ │ │ │ │ } │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(graphicName + " is not a valid symbol name") │ │ │ │ │ + var desiredZoomLevels; │ │ │ │ │ + var limitZoomLevels = this.MAX_ZOOM_LEVEL - this.minZoomLevel + 1; │ │ │ │ │ + if (this.options.numZoomLevels == null && this.options.maxZoomLevel != null || this.numZoomLevels == null && this.maxZoomLevel != null) { │ │ │ │ │ + desiredZoomLevels = this.maxZoomLevel - this.minZoomLevel + 1 │ │ │ │ │ + } else { │ │ │ │ │ + desiredZoomLevels = this.numZoomLevels │ │ │ │ │ } │ │ │ │ │ - var symbolExtent = new OpenLayers.Bounds(Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ - var pathitems = ["m"]; │ │ │ │ │ - for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - var x = symbol[i]; │ │ │ │ │ - var y = symbol[i + 1]; │ │ │ │ │ - symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ - symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ - symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ - symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ - pathitems.push(x); │ │ │ │ │ - pathitems.push(y); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - pathitems.push("l") │ │ │ │ │ + if (desiredZoomLevels != null) { │ │ │ │ │ + this.numZoomLevels = Math.min(desiredZoomLevels, limitZoomLevels) │ │ │ │ │ + } else { │ │ │ │ │ + this.numZoomLevels = limitZoomLevels │ │ │ │ │ + } │ │ │ │ │ + this.maxZoomLevel = this.minZoomLevel + this.numZoomLevels - 1; │ │ │ │ │ + if (this.RESOLUTIONS != null) { │ │ │ │ │ + var resolutionsIndex = 0; │ │ │ │ │ + this.resolutions = []; │ │ │ │ │ + for (var i = this.minZoomLevel; i <= this.maxZoomLevel; i++) { │ │ │ │ │ + this.resolutions[resolutionsIndex++] = this.RESOLUTIONS[i] │ │ │ │ │ } │ │ │ │ │ + this.maxResolution = this.resolutions[0]; │ │ │ │ │ + this.minResolution = this.resolutions[this.resolutions.length - 1] │ │ │ │ │ } │ │ │ │ │ - pathitems.push("x e"); │ │ │ │ │ - var path = pathitems.join(" "); │ │ │ │ │ - var diff = (symbolExtent.getWidth() - symbolExtent.getHeight()) / 2; │ │ │ │ │ - if (diff > 0) { │ │ │ │ │ - symbolExtent.bottom = symbolExtent.bottom - diff; │ │ │ │ │ - symbolExtent.top = symbolExtent.top + diff │ │ │ │ │ + }, │ │ │ │ │ + getResolution: function() { │ │ │ │ │ + if (this.resolutions != null) { │ │ │ │ │ + return OpenLayers.Layer.prototype.getResolution.apply(this, arguments) │ │ │ │ │ } else { │ │ │ │ │ - symbolExtent.left = symbolExtent.left + diff; │ │ │ │ │ - symbolExtent.right = symbolExtent.right - diff │ │ │ │ │ + var resolution = null; │ │ │ │ │ + var viewSize = this.map.getSize(); │ │ │ │ │ + var extent = this.getExtent(); │ │ │ │ │ + if (viewSize != null && extent != null) { │ │ │ │ │ + resolution = Math.max(extent.getWidth() / viewSize.w, extent.getHeight() / viewSize.h) │ │ │ │ │ + } │ │ │ │ │ + return resolution │ │ │ │ │ } │ │ │ │ │ - cache = { │ │ │ │ │ - path: path, │ │ │ │ │ - size: symbolExtent.getWidth(), │ │ │ │ │ - left: symbolExtent.left, │ │ │ │ │ - bottom: symbolExtent.bottom │ │ │ │ │ - }; │ │ │ │ │ - this.symbolCache[id] = cache; │ │ │ │ │ - return cache │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.VML" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Renderer.VML.LABEL_SHIFT = { │ │ │ │ │ - l: 0, │ │ │ │ │ - c: .5, │ │ │ │ │ - r: 1, │ │ │ │ │ - t: 0, │ │ │ │ │ - m: .5, │ │ │ │ │ - b: 1 │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.SVG = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ - xmlns: "http://www.w3.org/2000/svg", │ │ │ │ │ - xlinkns: "http://www.w3.org/1999/xlink", │ │ │ │ │ - MAX_PIXEL: 15e3, │ │ │ │ │ - translationParameters: null, │ │ │ │ │ - symbolMetrics: null, │ │ │ │ │ - initialize: function(containerID) { │ │ │ │ │ - if (!this.supported()) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.translationParameters = { │ │ │ │ │ + getExtent: function() { │ │ │ │ │ + var size = this.map.getSize(); │ │ │ │ │ + var tl = this.getLonLatFromViewPortPx({ │ │ │ │ │ x: 0, │ │ │ │ │ y: 0 │ │ │ │ │ - }; │ │ │ │ │ - this.symbolMetrics = {} │ │ │ │ │ - }, │ │ │ │ │ - supported: function() { │ │ │ │ │ - var svgFeature = "http://www.w3.org/TR/SVG11/feature#"; │ │ │ │ │ - return document.implementation && (document.implementation.hasFeature("org.w3c.svg", "1.0") || document.implementation.hasFeature(svgFeature + "SVG", "1.1") || document.implementation.hasFeature(svgFeature + "BasicStructure", "1.1")) │ │ │ │ │ - }, │ │ │ │ │ - inValidRange: function(x, y, xyOnly) { │ │ │ │ │ - var left = x + (xyOnly ? 0 : this.translationParameters.x); │ │ │ │ │ - var top = y + (xyOnly ? 0 : this.translationParameters.y); │ │ │ │ │ - return left >= -this.MAX_PIXEL && left <= this.MAX_PIXEL && top >= -this.MAX_PIXEL && top <= this.MAX_PIXEL │ │ │ │ │ + }); │ │ │ │ │ + var br = this.getLonLatFromViewPortPx({ │ │ │ │ │ + x: size.w, │ │ │ │ │ + y: size.h │ │ │ │ │ + }); │ │ │ │ │ + if (tl != null && br != null) { │ │ │ │ │ + return new OpenLayers.Bounds(tl.lon, br.lat, br.lon, tl.lat) │ │ │ │ │ + } else { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - var resolution = this.getResolution(), │ │ │ │ │ - left = -extent.left / resolution, │ │ │ │ │ - top = extent.top / resolution; │ │ │ │ │ - if (resolutionChanged) { │ │ │ │ │ - this.left = left; │ │ │ │ │ - this.top = top; │ │ │ │ │ - var extentString = "0 0 " + this.size.w + " " + this.size.h; │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "viewBox", extentString); │ │ │ │ │ - this.translate(this.xOffset, 0); │ │ │ │ │ - return true │ │ │ │ │ + getZoomForResolution: function(resolution) { │ │ │ │ │ + if (this.resolutions != null) { │ │ │ │ │ + return OpenLayers.Layer.prototype.getZoomForResolution.apply(this, arguments) │ │ │ │ │ } else { │ │ │ │ │ - var inRange = this.translate(left - this.left + this.xOffset, top - this.top); │ │ │ │ │ - if (!inRange) { │ │ │ │ │ - this.setExtent(extent, true) │ │ │ │ │ + var extent = OpenLayers.Layer.prototype.getExtent.apply(this, []); │ │ │ │ │ + return this.getZoomForExtent(extent) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getOLZoomFromMapObjectZoom: function(moZoom) { │ │ │ │ │ + var zoom = null; │ │ │ │ │ + if (moZoom != null) { │ │ │ │ │ + zoom = moZoom - this.minZoomLevel; │ │ │ │ │ + if (this.map.baseLayer !== this) { │ │ │ │ │ + zoom = this.map.baseLayer.getZoomForResolution(this.getResolutionForZoom(zoom)) │ │ │ │ │ } │ │ │ │ │ - return coordSysUnchanged && inRange │ │ │ │ │ } │ │ │ │ │ + return zoom │ │ │ │ │ }, │ │ │ │ │ - translate: function(x, y) { │ │ │ │ │ - if (!this.inValidRange(x, y, true)) { │ │ │ │ │ - return false │ │ │ │ │ - } else { │ │ │ │ │ - var transformString = ""; │ │ │ │ │ - if (x || y) { │ │ │ │ │ - transformString = "translate(" + x + "," + y + ")" │ │ │ │ │ + getMapObjectZoomFromOLZoom: function(olZoom) { │ │ │ │ │ + var zoom = null; │ │ │ │ │ + if (olZoom != null) { │ │ │ │ │ + zoom = olZoom + this.minZoomLevel; │ │ │ │ │ + if (this.map.baseLayer !== this) { │ │ │ │ │ + zoom = this.getZoomForResolution(this.map.baseLayer.getResolutionForZoom(zoom)) │ │ │ │ │ } │ │ │ │ │ - this.root.setAttributeNS(null, "transform", transformString); │ │ │ │ │ - this.translationParameters = { │ │ │ │ │ - x: x, │ │ │ │ │ - y: y │ │ │ │ │ - }; │ │ │ │ │ - return true │ │ │ │ │ } │ │ │ │ │ + return zoom │ │ │ │ │ }, │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "width", this.size.w); │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "height", this.size.h) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.FixedZoomLevels" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Zoomify = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + size: null, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + standardTileSize: 256, │ │ │ │ │ + tileOriginCorner: "tl", │ │ │ │ │ + numberOfTiers: 0, │ │ │ │ │ + tileCountUpToTier: null, │ │ │ │ │ + tierSizeInTiles: null, │ │ │ │ │ + tierImageSize: null, │ │ │ │ │ + initialize: function(name, url, size, options) { │ │ │ │ │ + this.initializeZoomify(size); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name, url, size, {}, options]) │ │ │ │ │ }, │ │ │ │ │ - getNodeType: function(geometry, style) { │ │ │ │ │ - var nodeType = null; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - nodeType = "image" │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - nodeType = "svg" │ │ │ │ │ - } else { │ │ │ │ │ - nodeType = "circle" │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - nodeType = "rect"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - nodeType = "polyline"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - nodeType = "polygon"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - case "OpenLayers.Geometry.Curve": │ │ │ │ │ - nodeType = "path"; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break │ │ │ │ │ + initializeZoomify: function(size) { │ │ │ │ │ + var imageSize = size.clone(); │ │ │ │ │ + this.size = size.clone(); │ │ │ │ │ + var tiles = new OpenLayers.Size(Math.ceil(imageSize.w / this.standardTileSize), Math.ceil(imageSize.h / this.standardTileSize)); │ │ │ │ │ + this.tierSizeInTiles = [tiles]; │ │ │ │ │ + this.tierImageSize = [imageSize]; │ │ │ │ │ + while (imageSize.w > this.standardTileSize || imageSize.h > this.standardTileSize) { │ │ │ │ │ + imageSize = new OpenLayers.Size(Math.floor(imageSize.w / 2), Math.floor(imageSize.h / 2)); │ │ │ │ │ + tiles = new OpenLayers.Size(Math.ceil(imageSize.w / this.standardTileSize), Math.ceil(imageSize.h / this.standardTileSize)); │ │ │ │ │ + this.tierSizeInTiles.push(tiles); │ │ │ │ │ + this.tierImageSize.push(imageSize) │ │ │ │ │ + } │ │ │ │ │ + this.tierSizeInTiles.reverse(); │ │ │ │ │ + this.tierImageSize.reverse(); │ │ │ │ │ + this.numberOfTiers = this.tierSizeInTiles.length; │ │ │ │ │ + var resolutions = [1]; │ │ │ │ │ + this.tileCountUpToTier = [0]; │ │ │ │ │ + for (var i = 1; i < this.numberOfTiers; i++) { │ │ │ │ │ + resolutions.unshift(Math.pow(2, i)); │ │ │ │ │ + this.tileCountUpToTier.push(this.tierSizeInTiles[i - 1].w * this.tierSizeInTiles[i - 1].h + this.tileCountUpToTier[i - 1]) │ │ │ │ │ + } │ │ │ │ │ + if (!this.serverResolutions) { │ │ │ │ │ + this.serverResolutions = resolutions │ │ │ │ │ } │ │ │ │ │ - return nodeType │ │ │ │ │ }, │ │ │ │ │ - setStyle: function(node, style, options) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - options = options || node._options; │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - node.setAttributeNS(null, "title", title); │ │ │ │ │ - var titleNode = node.getElementsByTagName("title"); │ │ │ │ │ - if (titleNode.length > 0) { │ │ │ │ │ - titleNode[0].firstChild.textContent = title │ │ │ │ │ - } else { │ │ │ │ │ - var label = this.nodeFactory(null, "title"); │ │ │ │ │ - label.textContent = title; │ │ │ │ │ - node.appendChild(label) │ │ │ │ │ - } │ │ │ │ │ + destroy: function() { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.destroy.apply(this, arguments); │ │ │ │ │ + this.tileCountUpToTier.length = 0; │ │ │ │ │ + this.tierSizeInTiles.length = 0; │ │ │ │ │ + this.tierImageSize.length = 0 │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Zoomify(this.name, this.url, this.size, this.options) │ │ │ │ │ } │ │ │ │ │ - var r = parseFloat(node.getAttributeNS(null, "r")); │ │ │ │ │ - var widthFactor = 1; │ │ │ │ │ - var pos; │ │ │ │ │ - if (node._geometryClass == "OpenLayers.Geometry.Point" && r) { │ │ │ │ │ - node.style.visibility = ""; │ │ │ │ │ - if (style.graphic === false) { │ │ │ │ │ - node.style.visibility = "hidden" │ │ │ │ │ - } else if (style.externalGraphic) { │ │ │ │ │ - pos = this.getPosition(node); │ │ │ │ │ - if (style.graphicWidth && style.graphicHeight) { │ │ │ │ │ - node.setAttributeNS(null, "preserveAspectRatio", "none") │ │ │ │ │ - } │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ - var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ - node.setAttributeNS(null, "x", (pos.x + xOffset).toFixed()); │ │ │ │ │ - node.setAttributeNS(null, "y", (pos.y + yOffset).toFixed()); │ │ │ │ │ - node.setAttributeNS(null, "width", width); │ │ │ │ │ - node.setAttributeNS(null, "height", height); │ │ │ │ │ - node.setAttributeNS(this.xlinkns, "xlink:href", style.externalGraphic); │ │ │ │ │ - node.setAttributeNS(null, "style", "opacity: " + opacity); │ │ │ │ │ - node.onclick = OpenLayers.Event.preventDefault │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - var offset = style.pointRadius * 3; │ │ │ │ │ - var size = offset * 2; │ │ │ │ │ - var src = this.importSymbol(style.graphicName); │ │ │ │ │ - pos = this.getPosition(node); │ │ │ │ │ - widthFactor = this.symbolMetrics[src.id][0] * 3 / size; │ │ │ │ │ - var parent = node.parentNode; │ │ │ │ │ - var nextSibling = node.nextSibling; │ │ │ │ │ - if (parent) { │ │ │ │ │ - parent.removeChild(node) │ │ │ │ │ - } │ │ │ │ │ - node.firstChild && node.removeChild(node.firstChild); │ │ │ │ │ - node.appendChild(src.firstChild.cloneNode(true)); │ │ │ │ │ - node.setAttributeNS(null, "viewBox", src.getAttributeNS(null, "viewBox")); │ │ │ │ │ - node.setAttributeNS(null, "width", size); │ │ │ │ │ - node.setAttributeNS(null, "height", size); │ │ │ │ │ - node.setAttributeNS(null, "x", pos.x - offset); │ │ │ │ │ - node.setAttributeNS(null, "y", pos.y - offset); │ │ │ │ │ - if (nextSibling) { │ │ │ │ │ - parent.insertBefore(node, nextSibling) │ │ │ │ │ - } else if (parent) { │ │ │ │ │ - parent.appendChild(node) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "r", style.pointRadius) │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((this.tileOrigin.lat - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getZoomForResolution(res); │ │ │ │ │ + var tileIndex = x + y * this.tierSizeInTiles[z].w + this.tileCountUpToTier[z]; │ │ │ │ │ + var path = "TileGroup" + Math.floor(tileIndex / 256) + "/" + z + "-" + x + "-" + y + ".jpg"; │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(path, url) │ │ │ │ │ + } │ │ │ │ │ + return url + path │ │ │ │ │ + }, │ │ │ │ │ + getImageSize: function() { │ │ │ │ │ + if (arguments.length > 0) { │ │ │ │ │ + var bounds = this.adjustBounds(arguments[0]); │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((this.tileOrigin.lat - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getZoomForResolution(res); │ │ │ │ │ + var w = this.standardTileSize; │ │ │ │ │ + var h = this.standardTileSize; │ │ │ │ │ + if (x == this.tierSizeInTiles[z].w - 1) { │ │ │ │ │ + var w = this.tierImageSize[z].w % this.standardTileSize │ │ │ │ │ } │ │ │ │ │ - var rotation = style.rotation; │ │ │ │ │ - if ((rotation !== undefined || node._rotation !== undefined) && pos) { │ │ │ │ │ - node._rotation = rotation; │ │ │ │ │ - rotation |= 0; │ │ │ │ │ - if (node.nodeName !== "svg") { │ │ │ │ │ - node.setAttributeNS(null, "transform", "rotate(" + rotation + " " + pos.x + " " + pos.y + ")") │ │ │ │ │ - } else { │ │ │ │ │ - var metrics = this.symbolMetrics[src.id]; │ │ │ │ │ - node.firstChild.setAttributeNS(null, "transform", "rotate(" + rotation + " " + metrics[1] + " " + metrics[2] + ")") │ │ │ │ │ - } │ │ │ │ │ + if (y == this.tierSizeInTiles[z].h - 1) { │ │ │ │ │ + var h = this.tierImageSize[z].h % this.standardTileSize │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - if (options.isFilled) { │ │ │ │ │ - node.setAttributeNS(null, "fill", style.fillColor); │ │ │ │ │ - node.setAttributeNS(null, "fill-opacity", style.fillOpacity) │ │ │ │ │ + return new OpenLayers.Size(w, h) │ │ │ │ │ } else { │ │ │ │ │ - node.setAttributeNS(null, "fill", "none") │ │ │ │ │ + return this.tileSize │ │ │ │ │ } │ │ │ │ │ - if (options.isStroked) { │ │ │ │ │ - node.setAttributeNS(null, "stroke", style.strokeColor); │ │ │ │ │ - node.setAttributeNS(null, "stroke-opacity", style.strokeOpacity); │ │ │ │ │ - node.setAttributeNS(null, "stroke-width", style.strokeWidth * widthFactor); │ │ │ │ │ - node.setAttributeNS(null, "stroke-linecap", style.strokeLinecap || "round"); │ │ │ │ │ - node.setAttributeNS(null, "stroke-linejoin", "round"); │ │ │ │ │ - style.strokeDashstyle && node.setAttributeNS(null, "stroke-dasharray", this.dashStyle(style, widthFactor)) │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.map.maxExtent.left, this.map.maxExtent.top) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Zoomify" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Boxes = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ + drawMarker: function(marker) { │ │ │ │ │ + var topleft = this.map.getLayerPxFromLonLat({ │ │ │ │ │ + lon: marker.bounds.left, │ │ │ │ │ + lat: marker.bounds.top │ │ │ │ │ + }); │ │ │ │ │ + var botright = this.map.getLayerPxFromLonLat({ │ │ │ │ │ + lon: marker.bounds.right, │ │ │ │ │ + lat: marker.bounds.bottom │ │ │ │ │ + }); │ │ │ │ │ + if (botright == null || topleft == null) { │ │ │ │ │ + marker.display(false) │ │ │ │ │ } else { │ │ │ │ │ - node.setAttributeNS(null, "stroke", "none") │ │ │ │ │ + var markerDiv = marker.draw(topleft, { │ │ │ │ │ + w: Math.max(1, botright.x - topleft.x), │ │ │ │ │ + h: Math.max(1, botright.y - topleft.y) │ │ │ │ │ + }); │ │ │ │ │ + if (!marker.drawn) { │ │ │ │ │ + this.div.appendChild(markerDiv); │ │ │ │ │ + marker.drawn = true │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (style.pointerEvents) { │ │ │ │ │ - node.setAttributeNS(null, "pointer-events", style.pointerEvents) │ │ │ │ │ + }, │ │ │ │ │ + removeMarker: function(marker) { │ │ │ │ │ + OpenLayers.Util.removeItem(this.markers, marker); │ │ │ │ │ + if (marker.div != null && marker.div.parentNode == this.div) { │ │ │ │ │ + this.div.removeChild(marker.div) │ │ │ │ │ } │ │ │ │ │ - if (style.cursor != null) { │ │ │ │ │ - node.setAttributeNS(null, "cursor", style.cursor) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Boxes" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Bing = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ + key: null, │ │ │ │ │ + serverResolutions: [156543.03390625, 78271.516953125, 39135.7584765625, 19567.87923828125, 9783.939619140625, 4891.9698095703125, 2445.9849047851562, 1222.9924523925781, 611.4962261962891, 305.74811309814453, 152.87405654907226, 76.43702827453613, 38.218514137268066, 19.109257068634033, 9.554628534317017, 4.777314267158508, 2.388657133579254, 1.194328566789627, .5971642833948135, .29858214169740677, .14929107084870338, .07464553542435169], │ │ │ │ │ + attributionTemplate: '<span class="olBingAttribution ${type}">' + '<div><a target="_blank" href="http://www.bing.com/maps/">' + '<img src="${logo}" /></a></div>${copyrights}' + '<a style="white-space: nowrap" target="_blank" ' + 'href="http://www.microsoft.com/maps/product/terms.html">' + "Terms of Use</a></span>", │ │ │ │ │ + metadata: null, │ │ │ │ │ + protocolRegex: /^http:/i, │ │ │ │ │ + type: "Road", │ │ │ │ │ + culture: "en-US", │ │ │ │ │ + metadataParams: null, │ │ │ │ │ + tileOptions: null, │ │ │ │ │ + protocol: ~window.location.href.indexOf("http") ? "" : "http:", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + sphericalMercator: true │ │ │ │ │ + }, options); │ │ │ │ │ + var name = options.name || "Bing " + (options.type || this.type); │ │ │ │ │ + var newArgs = [name, null, options]; │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, newArgs); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + crossOriginKeyword: "anonymous" │ │ │ │ │ + }, this.options.tileOptions); │ │ │ │ │ + this.loadMetadata() │ │ │ │ │ + }, │ │ │ │ │ + loadMetadata: function() { │ │ │ │ │ + this._callbackId = "_callback_" + this.id.replace(/\./g, "_"); │ │ │ │ │ + window[this._callbackId] = OpenLayers.Function.bind(OpenLayers.Layer.Bing.processMetadata, this); │ │ │ │ │ + var params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + key: this.key, │ │ │ │ │ + jsonp: this._callbackId, │ │ │ │ │ + include: "ImageryProviders" │ │ │ │ │ + }, this.metadataParams); │ │ │ │ │ + var url = this.protocol + "//dev.virtualearth.net/REST/v1/Imagery/Metadata/" + this.type + "?" + OpenLayers.Util.getParameterString(params); │ │ │ │ │ + var script = document.createElement("script"); │ │ │ │ │ + script.type = "text/javascript"; │ │ │ │ │ + script.src = url; │ │ │ │ │ + script.id = this._callbackId; │ │ │ │ │ + document.getElementsByTagName("head")[0].appendChild(script) │ │ │ │ │ + }, │ │ │ │ │ + initLayer: function() { │ │ │ │ │ + var res = this.metadata.resourceSets[0].resources[0]; │ │ │ │ │ + var url = res.imageUrl.replace("{quadkey}", "${quadkey}"); │ │ │ │ │ + url = url.replace("{culture}", this.culture); │ │ │ │ │ + url = url.replace(this.protocolRegex, this.protocol); │ │ │ │ │ + this.url = []; │ │ │ │ │ + for (var i = 0; i < res.imageUrlSubdomains.length; ++i) { │ │ │ │ │ + this.url.push(url.replace("{subdomain}", res.imageUrlSubdomains[i])) │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ + this.addOptions({ │ │ │ │ │ + maxResolution: Math.min(this.serverResolutions[res.zoomMin], this.maxResolution || Number.POSITIVE_INFINITY), │ │ │ │ │ + numZoomLevels: Math.min(res.zoomMax + 1 - res.zoomMin, this.numZoomLevels) │ │ │ │ │ + }, true); │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + this.redraw() │ │ │ │ │ + } │ │ │ │ │ + this.updateAttribution() │ │ │ │ │ }, │ │ │ │ │ - dashStyle: function(style, widthFactor) { │ │ │ │ │ - var w = style.strokeWidth * widthFactor; │ │ │ │ │ - var str = style.strokeDashstyle; │ │ │ │ │ - switch (str) { │ │ │ │ │ - case "solid": │ │ │ │ │ - return "none"; │ │ │ │ │ - case "dot": │ │ │ │ │ - return [1, 4 * w].join(); │ │ │ │ │ - case "dash": │ │ │ │ │ - return [4 * w, 4 * w].join(); │ │ │ │ │ - case "dashdot": │ │ │ │ │ - return [4 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ - case "longdash": │ │ │ │ │ - return [8 * w, 4 * w].join(); │ │ │ │ │ - case "longdashdot": │ │ │ │ │ - return [8 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ - default: │ │ │ │ │ - return OpenLayers.String.trim(str).replace(/\s+/g, ",") │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + if (!this.url) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ + var xyz = this.getXYZ(bounds), │ │ │ │ │ + x = xyz.x, │ │ │ │ │ + y = xyz.y, │ │ │ │ │ + z = xyz.z; │ │ │ │ │ + var quadDigits = []; │ │ │ │ │ + for (var i = z; i > 0; --i) { │ │ │ │ │ + var digit = "0"; │ │ │ │ │ + var mask = 1 << i - 1; │ │ │ │ │ + if ((x & mask) != 0) { │ │ │ │ │ + digit++ │ │ │ │ │ + } │ │ │ │ │ + if ((y & mask) != 0) { │ │ │ │ │ + digit++; │ │ │ │ │ + digit++ │ │ │ │ │ + } │ │ │ │ │ + quadDigits.push(digit) │ │ │ │ │ + } │ │ │ │ │ + var quadKey = quadDigits.join(""); │ │ │ │ │ + var url = this.selectUrl("" + x + y + z, this.url); │ │ │ │ │ + return OpenLayers.String.format(url, { │ │ │ │ │ + quadkey: quadKey │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - createNode: function(type, id) { │ │ │ │ │ - var node = document.createElementNS(this.xmlns, type); │ │ │ │ │ - if (id) { │ │ │ │ │ - node.setAttributeNS(null, "id", id) │ │ │ │ │ + updateAttribution: function() { │ │ │ │ │ + var metadata = this.metadata; │ │ │ │ │ + if (!metadata.resourceSets || !this.map || !this.map.center) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ + var res = metadata.resourceSets[0].resources[0]; │ │ │ │ │ + var extent = this.map.getExtent().transform(this.map.getProjectionObject(), new OpenLayers.Projection("EPSG:4326")); │ │ │ │ │ + var providers = res.imageryProviders || [], │ │ │ │ │ + zoom = OpenLayers.Util.indexOf(this.serverResolutions, this.getServerResolution()), │ │ │ │ │ + copyrights = "", │ │ │ │ │ + provider, i, ii, j, jj, bbox, coverage; │ │ │ │ │ + for (i = 0, ii = providers.length; i < ii; ++i) { │ │ │ │ │ + provider = providers[i]; │ │ │ │ │ + for (j = 0, jj = provider.coverageAreas.length; j < jj; ++j) { │ │ │ │ │ + coverage = provider.coverageAreas[j]; │ │ │ │ │ + bbox = OpenLayers.Bounds.fromArray(coverage.bbox, true); │ │ │ │ │ + if (extent.intersectsBounds(bbox) && zoom <= coverage.zoomMax && zoom >= coverage.zoomMin) { │ │ │ │ │ + copyrights += provider.attribution + " " │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var logo = metadata.brandLogoUri.replace(this.protocolRegex, this.protocol); │ │ │ │ │ + this.attribution = OpenLayers.String.format(this.attributionTemplate, { │ │ │ │ │ + type: this.type.toLowerCase(), │ │ │ │ │ + logo: logo, │ │ │ │ │ + copyrights: copyrights │ │ │ │ │ + }); │ │ │ │ │ + this.map && this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "attribution" │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - nodeTypeCompare: function(node, type) { │ │ │ │ │ - return type == node.nodeName │ │ │ │ │ + setMap: function() { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.register("moveend", this, this.updateAttribution) │ │ │ │ │ }, │ │ │ │ │ - createRenderRoot: function() { │ │ │ │ │ - var svg = this.nodeFactory(this.container.id + "_svgRoot", "svg"); │ │ │ │ │ - svg.style.display = "block"; │ │ │ │ │ - return svg │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Bing(this.options) │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - createRoot: function(suffix) { │ │ │ │ │ - return this.nodeFactory(this.container.id + suffix, "g") │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.map && this.map.events.unregister("moveend", this, this.updateAttribution); │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - createDefs: function() { │ │ │ │ │ - var defs = this.nodeFactory(this.container.id + "_defs", "defs"); │ │ │ │ │ - this.rendererRoot.appendChild(defs); │ │ │ │ │ - return defs │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Bing" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Bing.processMetadata = function(metadata) { │ │ │ │ │ + this.metadata = metadata; │ │ │ │ │ + this.initLayer(); │ │ │ │ │ + var script = document.getElementById(this._callbackId); │ │ │ │ │ + script.parentNode.removeChild(script); │ │ │ │ │ + window[this._callbackId] = undefined; │ │ │ │ │ + delete this._callbackId │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Layer.MapGuide = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + useHttpTile: false, │ │ │ │ │ + singleTile: false, │ │ │ │ │ + useOverlay: false, │ │ │ │ │ + useAsyncOverlay: true, │ │ │ │ │ + TILE_PARAMS: { │ │ │ │ │ + operation: "GETTILEIMAGE", │ │ │ │ │ + version: "1.2.0" │ │ │ │ │ }, │ │ │ │ │ - drawPoint: function(node, geometry) { │ │ │ │ │ - return this.drawCircle(node, geometry, 1) │ │ │ │ │ + SINGLE_TILE_PARAMS: { │ │ │ │ │ + operation: "GETMAPIMAGE", │ │ │ │ │ + format: "PNG", │ │ │ │ │ + locale: "en", │ │ │ │ │ + clip: "1", │ │ │ │ │ + version: "1.0.0" │ │ │ │ │ }, │ │ │ │ │ - drawCircle: function(node, geometry, radius) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = (geometry.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y = this.top - geometry.y / resolution; │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - node.setAttributeNS(null, "cx", x); │ │ │ │ │ - node.setAttributeNS(null, "cy", y); │ │ │ │ │ - node.setAttributeNS(null, "r", radius); │ │ │ │ │ - return node │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ + OVERLAY_PARAMS: { │ │ │ │ │ + operation: "GETDYNAMICMAPOVERLAYIMAGE", │ │ │ │ │ + format: "PNG", │ │ │ │ │ + locale: "en", │ │ │ │ │ + clip: "1", │ │ │ │ │ + version: "2.0.0" │ │ │ │ │ }, │ │ │ │ │ - drawLineString: function(node, geometry) { │ │ │ │ │ - var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ - if (componentsResult.path) { │ │ │ │ │ - node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ - return componentsResult.complete ? node : null │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ + FOLDER_PARAMS: { │ │ │ │ │ + tileColumnsPerFolder: 30, │ │ │ │ │ + tileRowsPerFolder: 30, │ │ │ │ │ + format: "png", │ │ │ │ │ + querystring: null │ │ │ │ │ }, │ │ │ │ │ - drawLinearRing: function(node, geometry) { │ │ │ │ │ - var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ - if (componentsResult.path) { │ │ │ │ │ - node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ - return componentsResult.complete ? node : null │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + defaultSize: new OpenLayers.Size(300, 300), │ │ │ │ │ + tileOriginCorner: "tl", │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (options == null || options.isBaseLayer == null) { │ │ │ │ │ + this.isBaseLayer = this.transparent != "true" && this.transparent != true │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - drawPolygon: function(node, geometry) { │ │ │ │ │ - var d = ""; │ │ │ │ │ - var draw = true; │ │ │ │ │ - var complete = true; │ │ │ │ │ - var linearRingResult, path; │ │ │ │ │ - for (var j = 0, len = geometry.components.length; j < len; j++) { │ │ │ │ │ - d += " M"; │ │ │ │ │ - linearRingResult = this.getComponentsString(geometry.components[j].components, " "); │ │ │ │ │ - path = linearRingResult.path; │ │ │ │ │ - if (path) { │ │ │ │ │ - d += " " + path; │ │ │ │ │ - complete = linearRingResult.complete && complete │ │ │ │ │ + if (options && options.useOverlay != null) { │ │ │ │ │ + this.useOverlay = options.useOverlay │ │ │ │ │ + } │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + if (this.useOverlay) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(this.params, this.OVERLAY_PARAMS); │ │ │ │ │ + if (!this.useAsyncOverlay) { │ │ │ │ │ + this.params.version = "1.0.0" │ │ │ │ │ + } │ │ │ │ │ } else { │ │ │ │ │ - draw = false │ │ │ │ │ + OpenLayers.Util.applyDefaults(this.params, this.SINGLE_TILE_PARAMS) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - d += " z"; │ │ │ │ │ - if (draw) { │ │ │ │ │ - node.setAttributeNS(null, "d", d); │ │ │ │ │ - node.setAttributeNS(null, "fill-rule", "evenodd"); │ │ │ │ │ - return complete ? node : null │ │ │ │ │ } else { │ │ │ │ │ - return false │ │ │ │ │ + if (this.useHttpTile) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(this.params, this.FOLDER_PARAMS) │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Util.applyDefaults(this.params, this.TILE_PARAMS) │ │ │ │ │ + } │ │ │ │ │ + this.setTileSize(this.defaultSize) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - drawRectangle: function(node, geometry) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = (geometry.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y = this.top - geometry.y / resolution; │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - node.setAttributeNS(null, "x", x); │ │ │ │ │ - node.setAttributeNS(null, "y", y); │ │ │ │ │ - node.setAttributeNS(null, "width", geometry.width / resolution); │ │ │ │ │ - node.setAttributeNS(null, "height", geometry.height / resolution); │ │ │ │ │ - return node │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.MapGuide(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - drawText: function(featureId, style, location) { │ │ │ │ │ - var drawOutline = !!style.labelOutlineWidth; │ │ │ │ │ - if (drawOutline) { │ │ │ │ │ - var outlineStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ - outlineStyle.fontColor = outlineStyle.labelOutlineColor; │ │ │ │ │ - outlineStyle.fontStrokeColor = outlineStyle.labelOutlineColor; │ │ │ │ │ - outlineStyle.fontStrokeWidth = style.labelOutlineWidth; │ │ │ │ │ - if (style.labelOutlineOpacity) { │ │ │ │ │ - outlineStyle.fontOpacity = style.labelOutlineOpacity │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var url; │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var mapSize = this.map.getSize(); │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + var params = { │ │ │ │ │ + setdisplaydpi: OpenLayers.DOTS_PER_INCH, │ │ │ │ │ + setdisplayheight: mapSize.h * this.ratio, │ │ │ │ │ + setdisplaywidth: mapSize.w * this.ratio, │ │ │ │ │ + setviewcenterx: center.lon, │ │ │ │ │ + setviewcentery: center.lat, │ │ │ │ │ + setviewscale: this.map.getScale() │ │ │ │ │ + }; │ │ │ │ │ + if (this.useOverlay && !this.useAsyncOverlay) { │ │ │ │ │ + var getVisParams = {}; │ │ │ │ │ + getVisParams = OpenLayers.Util.extend(getVisParams, params); │ │ │ │ │ + getVisParams.operation = "GETVISIBLEMAPEXTENT"; │ │ │ │ │ + getVisParams.version = "1.0.0"; │ │ │ │ │ + getVisParams.session = this.params.session; │ │ │ │ │ + getVisParams.mapName = this.params.mapName; │ │ │ │ │ + getVisParams.format = "text/xml"; │ │ │ │ │ + url = this.getFullRequestString(getVisParams); │ │ │ │ │ + OpenLayers.Request.GET({ │ │ │ │ │ + url: url, │ │ │ │ │ + async: false │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + url = this.getFullRequestString(params) │ │ │ │ │ + } else { │ │ │ │ │ + var currentRes = this.map.getResolution(); │ │ │ │ │ + var colidx = Math.floor((bounds.left - this.maxExtent.left) / currentRes); │ │ │ │ │ + colidx = Math.round(colidx / this.tileSize.w); │ │ │ │ │ + var rowidx = Math.floor((this.maxExtent.top - bounds.top) / currentRes); │ │ │ │ │ + rowidx = Math.round(rowidx / this.tileSize.h); │ │ │ │ │ + if (this.useHttpTile) { │ │ │ │ │ + url = this.getImageFilePath({ │ │ │ │ │ + tilecol: colidx, │ │ │ │ │ + tilerow: rowidx, │ │ │ │ │ + scaleindex: this.resolutions.length - this.map.zoom - 1 │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + url = this.getFullRequestString({ │ │ │ │ │ + tilecol: colidx, │ │ │ │ │ + tilerow: rowidx, │ │ │ │ │ + scaleindex: this.resolutions.length - this.map.zoom - 1 │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - delete outlineStyle.labelOutlineWidth; │ │ │ │ │ - this.drawText(featureId, outlineStyle, location) │ │ │ │ │ - } │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = (location.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y = location.y / resolution - this.top; │ │ │ │ │ - var suffix = drawOutline ? this.LABEL_OUTLINE_SUFFIX : this.LABEL_ID_SUFFIX; │ │ │ │ │ - var label = this.nodeFactory(featureId + suffix, "text"); │ │ │ │ │ - label.setAttributeNS(null, "x", x); │ │ │ │ │ - label.setAttributeNS(null, "y", -y); │ │ │ │ │ - if (style.fontColor) { │ │ │ │ │ - label.setAttributeNS(null, "fill", style.fontColor) │ │ │ │ │ } │ │ │ │ │ - if (style.fontStrokeColor) { │ │ │ │ │ - label.setAttributeNS(null, "stroke", style.fontStrokeColor) │ │ │ │ │ + return url │ │ │ │ │ + }, │ │ │ │ │ + getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ + var url = altUrl == null ? this.url : altUrl; │ │ │ │ │ + if (typeof url == "object") { │ │ │ │ │ + url = url[Math.floor(Math.random() * url.length)] │ │ │ │ │ } │ │ │ │ │ - if (style.fontStrokeWidth) { │ │ │ │ │ - label.setAttributeNS(null, "stroke-width", style.fontStrokeWidth) │ │ │ │ │ + var requestString = url; │ │ │ │ │ + var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ + allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ + var urlParams = OpenLayers.Util.upperCaseObject(OpenLayers.Util.getParameters(url)); │ │ │ │ │ + for (var key in allParams) { │ │ │ │ │ + if (key.toUpperCase() in urlParams) { │ │ │ │ │ + delete allParams[key] │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (style.fontOpacity) { │ │ │ │ │ - label.setAttributeNS(null, "opacity", style.fontOpacity) │ │ │ │ │ + var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + paramsString = paramsString.replace(/,/g, "+"); │ │ │ │ │ + if (paramsString != "") { │ │ │ │ │ + var lastServerChar = url.charAt(url.length - 1); │ │ │ │ │ + if (lastServerChar == "&" || lastServerChar == "?") { │ │ │ │ │ + requestString += paramsString │ │ │ │ │ + } else { │ │ │ │ │ + if (url.indexOf("?") == -1) { │ │ │ │ │ + requestString += "?" + paramsString │ │ │ │ │ + } else { │ │ │ │ │ + requestString += "&" + paramsString │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (style.fontFamily) { │ │ │ │ │ - label.setAttributeNS(null, "font-family", style.fontFamily) │ │ │ │ │ + return requestString │ │ │ │ │ + }, │ │ │ │ │ + getImageFilePath: function(newParams, altUrl) { │ │ │ │ │ + var url = altUrl == null ? this.url : altUrl; │ │ │ │ │ + if (typeof url == "object") { │ │ │ │ │ + url = url[Math.floor(Math.random() * url.length)] │ │ │ │ │ } │ │ │ │ │ - if (style.fontSize) { │ │ │ │ │ - label.setAttributeNS(null, "font-size", style.fontSize) │ │ │ │ │ + var requestString = url; │ │ │ │ │ + var tileRowGroup = ""; │ │ │ │ │ + var tileColGroup = ""; │ │ │ │ │ + if (newParams.tilerow < 0) { │ │ │ │ │ + tileRowGroup = "-" │ │ │ │ │ } │ │ │ │ │ - if (style.fontWeight) { │ │ │ │ │ - label.setAttributeNS(null, "font-weight", style.fontWeight) │ │ │ │ │ + if (newParams.tilerow == 0) { │ │ │ │ │ + tileRowGroup += "0" │ │ │ │ │ + } else { │ │ │ │ │ + tileRowGroup += Math.floor(Math.abs(newParams.tilerow / this.params.tileRowsPerFolder)) * this.params.tileRowsPerFolder │ │ │ │ │ } │ │ │ │ │ - if (style.fontStyle) { │ │ │ │ │ - label.setAttributeNS(null, "font-style", style.fontStyle) │ │ │ │ │ + if (newParams.tilecol < 0) { │ │ │ │ │ + tileColGroup = "-" │ │ │ │ │ } │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - label.setAttributeNS(null, "pointer-events", "visible"); │ │ │ │ │ - label._featureId = featureId │ │ │ │ │ + if (newParams.tilecol == 0) { │ │ │ │ │ + tileColGroup += "0" │ │ │ │ │ } else { │ │ │ │ │ - label.setAttributeNS(null, "pointer-events", "none") │ │ │ │ │ + tileColGroup += Math.floor(Math.abs(newParams.tilecol / this.params.tileColumnsPerFolder)) * this.params.tileColumnsPerFolder │ │ │ │ │ } │ │ │ │ │ - var align = style.labelAlign || OpenLayers.Renderer.defaultSymbolizer.labelAlign; │ │ │ │ │ - label.setAttributeNS(null, "text-anchor", OpenLayers.Renderer.SVG.LABEL_ALIGN[align[0]] || "middle"); │ │ │ │ │ - if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ - label.setAttributeNS(null, "dominant-baseline", OpenLayers.Renderer.SVG.LABEL_ALIGN[align[1]] || "central") │ │ │ │ │ + var tilePath = "/S" + Math.floor(newParams.scaleindex) + "/" + this.params.basemaplayergroupname + "/R" + tileRowGroup + "/C" + tileColGroup + "/" + newParams.tilerow % this.params.tileRowsPerFolder + "_" + newParams.tilecol % this.params.tileColumnsPerFolder + "." + this.params.format; │ │ │ │ │ + if (this.params.querystring) { │ │ │ │ │ + tilePath += "?" + this.params.querystring │ │ │ │ │ } │ │ │ │ │ - var labelRows = style.label.split("\n"); │ │ │ │ │ - var numRows = labelRows.length; │ │ │ │ │ - while (label.childNodes.length > numRows) { │ │ │ │ │ - label.removeChild(label.lastChild) │ │ │ │ │ + requestString += tilePath; │ │ │ │ │ + return requestString │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.MapGuide" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.TMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + serviceVersion: "1.0.0", │ │ │ │ │ + layername: null, │ │ │ │ │ + type: null, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + tileOrigin: null, │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + zoomOffset: 0, │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + newArguments.push(name, url, {}, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.TMS(this.name, this.url, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - var tspan = this.nodeFactory(featureId + suffix + "_tspan_" + i, "tspan"); │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - tspan._featureId = featureId; │ │ │ │ │ - tspan._geometry = location; │ │ │ │ │ - tspan._geometryClass = location.CLASS_NAME │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.IS_GECKO === false) { │ │ │ │ │ - tspan.setAttributeNS(null, "baseline-shift", OpenLayers.Renderer.SVG.LABEL_VSHIFT[align[1]] || "-35%") │ │ │ │ │ - } │ │ │ │ │ - tspan.setAttribute("x", x); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - var vfactor = OpenLayers.Renderer.SVG.LABEL_VFACTOR[align[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5 │ │ │ │ │ - } │ │ │ │ │ - tspan.setAttribute("dy", vfactor * (numRows - 1) + "em") │ │ │ │ │ - } else { │ │ │ │ │ - tspan.setAttribute("dy", "1em") │ │ │ │ │ - } │ │ │ │ │ - tspan.textContent = labelRows[i] === "" ? " " : labelRows[i]; │ │ │ │ │ - if (!tspan.parentNode) { │ │ │ │ │ - label.appendChild(tspan) │ │ │ │ │ - } │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((bounds.bottom - this.tileOrigin.lat) / (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getServerZoom(); │ │ │ │ │ + var path = this.serviceVersion + "/" + this.layername + "/" + z + "/" + x + "/" + y + "." + this.type; │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(path, url) │ │ │ │ │ } │ │ │ │ │ - if (!label.parentNode) { │ │ │ │ │ - this.textRoot.appendChild(label) │ │ │ │ │ + return url + path │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (!this.tileOrigin) { │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.map.maxExtent.left, this.map.maxExtent.bottom) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - getComponentsString: function(components, separator) { │ │ │ │ │ - var renderCmp = []; │ │ │ │ │ - var complete = true; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - var strings = []; │ │ │ │ │ - var str, component; │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - component = components[i]; │ │ │ │ │ - renderCmp.push(component); │ │ │ │ │ - str = this.getShortString(component); │ │ │ │ │ - if (str) { │ │ │ │ │ - strings.push(str) │ │ │ │ │ - } else { │ │ │ │ │ - if (i > 0) { │ │ │ │ │ - if (this.getShortString(components[i - 1])) { │ │ │ │ │ - strings.push(this.clipLine(components[i], components[i - 1])) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (i < len - 1) { │ │ │ │ │ - if (this.getShortString(components[i + 1])) { │ │ │ │ │ - strings.push(this.clipLine(components[i], components[i + 1])) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - complete = false │ │ │ │ │ - } │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.TMS" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.OSM = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ + name: "OpenStreetMap", │ │ │ │ │ + url: ["http://a.tile.openstreetmap.org/${z}/${x}/${y}.png", "http://b.tile.openstreetmap.org/${z}/${x}/${y}.png", "http://c.tile.openstreetmap.org/${z}/${x}/${y}.png"], │ │ │ │ │ + attribution: "© <a href='http://www.openstreetmap.org/copyright'>OpenStreetMap</a> contributors", │ │ │ │ │ + sphericalMercator: true, │ │ │ │ │ + wrapDateLine: true, │ │ │ │ │ + tileOptions: null, │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + crossOriginKeyword: "anonymous" │ │ │ │ │ + }, this.options && this.options.tileOptions) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.OSM(this.name, this.url, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - return { │ │ │ │ │ - path: strings.join(separator || ","), │ │ │ │ │ - complete: complete │ │ │ │ │ + obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.OSM" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.SphericalMercator = { │ │ │ │ │ + getExtent: function() { │ │ │ │ │ + var extent = null; │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + extent = this.map.calculateBounds() │ │ │ │ │ + } else { │ │ │ │ │ + extent = OpenLayers.Layer.FixedZoomLevels.prototype.getExtent.apply(this) │ │ │ │ │ } │ │ │ │ │ + return extent │ │ │ │ │ }, │ │ │ │ │ - clipLine: function(badComponent, goodComponent) { │ │ │ │ │ - if (goodComponent.equals(badComponent)) { │ │ │ │ │ - return "" │ │ │ │ │ + getLonLatFromViewPortPx: function(viewPortPx) { │ │ │ │ │ + return OpenLayers.Layer.prototype.getLonLatFromViewPortPx.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + getViewPortPxFromLonLat: function(lonlat) { │ │ │ │ │ + return OpenLayers.Layer.prototype.getViewPortPxFromLonLat.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + initMercatorParameters: function() { │ │ │ │ │ + this.RESOLUTIONS = []; │ │ │ │ │ + var maxResolution = 156543.03390625; │ │ │ │ │ + for (var zoom = 0; zoom <= this.MAX_ZOOM_LEVEL; ++zoom) { │ │ │ │ │ + this.RESOLUTIONS[zoom] = maxResolution / Math.pow(2, zoom) │ │ │ │ │ } │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var maxX = this.MAX_PIXEL - this.translationParameters.x; │ │ │ │ │ - var maxY = this.MAX_PIXEL - this.translationParameters.y; │ │ │ │ │ - var x1 = (goodComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y1 = this.top - goodComponent.y / resolution; │ │ │ │ │ - var x2 = (badComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y2 = this.top - badComponent.y / resolution; │ │ │ │ │ - var k; │ │ │ │ │ - if (x2 < -maxX || x2 > maxX) { │ │ │ │ │ - k = (y2 - y1) / (x2 - x1); │ │ │ │ │ - x2 = x2 < 0 ? -maxX : maxX; │ │ │ │ │ - y2 = y1 + (x2 - x1) * k │ │ │ │ │ + this.units = "m"; │ │ │ │ │ + this.projection = this.projection || "EPSG:900913" │ │ │ │ │ + }, │ │ │ │ │ + forwardMercator: function() { │ │ │ │ │ + var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + var sm = new OpenLayers.Projection("EPSG:900913"); │ │ │ │ │ + return function(lon, lat) { │ │ │ │ │ + var point = OpenLayers.Projection.transform({ │ │ │ │ │ + x: lon, │ │ │ │ │ + y: lat │ │ │ │ │ + }, gg, sm); │ │ │ │ │ + return new OpenLayers.LonLat(point.x, point.y) │ │ │ │ │ } │ │ │ │ │ - if (y2 < -maxY || y2 > maxY) { │ │ │ │ │ - k = (x2 - x1) / (y2 - y1); │ │ │ │ │ - y2 = y2 < 0 ? -maxY : maxY; │ │ │ │ │ - x2 = x1 + (y2 - y1) * k │ │ │ │ │ + }(), │ │ │ │ │ + inverseMercator: function() { │ │ │ │ │ + var gg = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + var sm = new OpenLayers.Projection("EPSG:900913"); │ │ │ │ │ + return function(x, y) { │ │ │ │ │ + var point = OpenLayers.Projection.transform({ │ │ │ │ │ + x: x, │ │ │ │ │ + y: y │ │ │ │ │ + }, sm, gg); │ │ │ │ │ + return new OpenLayers.LonLat(point.x, point.y) │ │ │ │ │ } │ │ │ │ │ - return x2 + "," + y2 │ │ │ │ │ + }() │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Layer.WorldWind = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + DEFAULT_PARAMS: {}, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + lzd: null, │ │ │ │ │ + zoomLevels: null, │ │ │ │ │ + initialize: function(name, url, lzd, zoomLevels, params, options) { │ │ │ │ │ + this.lzd = lzd; │ │ │ │ │ + this.zoomLevels = zoomLevels; │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + newArguments.push(name, url, params, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + this.params = OpenLayers.Util.applyDefaults(this.params, this.DEFAULT_PARAMS) │ │ │ │ │ }, │ │ │ │ │ - getShortString: function(point) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = (point.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y = this.top - point.y / resolution; │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - return x + "," + y │ │ │ │ │ + getZoom: function() { │ │ │ │ │ + var zoom = this.map.getZoom(); │ │ │ │ │ + var extent = this.map.getMaxExtent(); │ │ │ │ │ + zoom = zoom - Math.log(this.maxResolution / (this.lzd / 512)) / Math.log(2); │ │ │ │ │ + return zoom │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var zoom = this.getZoom(); │ │ │ │ │ + var extent = this.map.getMaxExtent(); │ │ │ │ │ + var deg = this.lzd / Math.pow(2, this.getZoom()); │ │ │ │ │ + var x = Math.floor((bounds.left - extent.left) / deg); │ │ │ │ │ + var y = Math.floor((bounds.bottom - extent.bottom) / deg); │ │ │ │ │ + if (this.map.getResolution() <= this.lzd / 512 && this.getZoom() <= this.zoomLevels) { │ │ │ │ │ + return this.getFullRequestString({ │ │ │ │ │ + L: zoom, │ │ │ │ │ + X: x, │ │ │ │ │ + Y: y │ │ │ │ │ + }) │ │ │ │ │ } else { │ │ │ │ │ - return false │ │ │ │ │ + return OpenLayers.Util.getImageLocation("blank.gif") │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - getPosition: function(node) { │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.WorldWind" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.KaMap = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + i: "jpeg", │ │ │ │ │ + map: "" │ │ │ │ │ + }, │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.params = OpenLayers.Util.applyDefaults(this.params, this.DEFAULT_PARAMS) │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var mapRes = this.map.getResolution(); │ │ │ │ │ + var scale = Math.round(this.map.getScale() * 1e4) / 1e4; │ │ │ │ │ + var pX = Math.round(bounds.left / mapRes); │ │ │ │ │ + var pY = -Math.round(bounds.top / mapRes); │ │ │ │ │ + return this.getFullRequestString({ │ │ │ │ │ + t: pY, │ │ │ │ │ + l: pX, │ │ │ │ │ + s: scale │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + calculateGridLayout: function(bounds, origin, resolution) { │ │ │ │ │ + var tilelon = resolution * this.tileSize.w; │ │ │ │ │ + var tilelat = resolution * this.tileSize.h; │ │ │ │ │ + var offsetlon = bounds.left; │ │ │ │ │ + var tilecol = Math.floor(offsetlon / tilelon) - this.buffer; │ │ │ │ │ + var offsetlat = bounds.top; │ │ │ │ │ + var tilerow = Math.floor(offsetlat / tilelat) + this.buffer; │ │ │ │ │ return { │ │ │ │ │ - x: parseFloat(node.getAttributeNS(null, "cx")), │ │ │ │ │ - y: parseFloat(node.getAttributeNS(null, "cy")) │ │ │ │ │ + tilelon: tilelon, │ │ │ │ │ + tilelat: tilelat, │ │ │ │ │ + startcol: tilecol, │ │ │ │ │ + startrow: tilerow │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - importSymbol: function(graphicName) { │ │ │ │ │ - if (!this.defs) { │ │ │ │ │ - this.defs = this.createDefs() │ │ │ │ │ - } │ │ │ │ │ - var id = this.container.id + "-" + graphicName; │ │ │ │ │ - var existing = document.getElementById(id); │ │ │ │ │ - if (existing != null) { │ │ │ │ │ - return existing │ │ │ │ │ - } │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(graphicName + " is not a valid symbol name") │ │ │ │ │ + getTileBoundsForGridIndex: function(row, col) { │ │ │ │ │ + var origin = this.getTileOrigin(); │ │ │ │ │ + var tileLayout = this.gridLayout; │ │ │ │ │ + var tilelon = tileLayout.tilelon; │ │ │ │ │ + var tilelat = tileLayout.tilelat; │ │ │ │ │ + var minX = (tileLayout.startcol + col) * tilelon; │ │ │ │ │ + var minY = (tileLayout.startrow - row) * tilelat; │ │ │ │ │ + return new OpenLayers.Bounds(minX, minY, minX + tilelon, minY + tilelat) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.KaMap(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - var symbolNode = this.nodeFactory(id, "symbol"); │ │ │ │ │ - var node = this.nodeFactory(null, "polygon"); │ │ │ │ │ - symbolNode.appendChild(node); │ │ │ │ │ - var symbolExtent = new OpenLayers.Bounds(Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ - var points = []; │ │ │ │ │ - var x, y; │ │ │ │ │ - for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ - symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ - symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ - symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ - points.push(x, ",", y) │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + if (this.tileSize != null) { │ │ │ │ │ + obj.tileSize = this.tileSize.clone() │ │ │ │ │ } │ │ │ │ │ - node.setAttributeNS(null, "points", points.join(" ")); │ │ │ │ │ - var width = symbolExtent.getWidth(); │ │ │ │ │ - var height = symbolExtent.getHeight(); │ │ │ │ │ - var viewBox = [symbolExtent.left - width, symbolExtent.bottom - height, width * 3, height * 3]; │ │ │ │ │ - symbolNode.setAttributeNS(null, "viewBox", viewBox.join(" ")); │ │ │ │ │ - this.symbolMetrics[id] = [Math.max(width, height), symbolExtent.getCenterLonLat().lon, symbolExtent.getCenterLonLat().lat]; │ │ │ │ │ - this.defs.appendChild(symbolNode); │ │ │ │ │ - return symbolNode │ │ │ │ │ + obj.grid = []; │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var featureId = OpenLayers.Renderer.Elements.prototype.getFeatureIdFromEvent.apply(this, arguments); │ │ │ │ │ - if (!featureId) { │ │ │ │ │ - var target = evt.target; │ │ │ │ │ - featureId = target.parentNode && target != this.rendererRoot ? target.parentNode._featureId : undefined │ │ │ │ │ + getTileBounds: function(viewPortPx) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var tileMapWidth = resolution * this.tileSize.w; │ │ │ │ │ + var tileMapHeight = resolution * this.tileSize.h; │ │ │ │ │ + var mapPoint = this.getLonLatFromViewPortPx(viewPortPx); │ │ │ │ │ + var tileLeft = tileMapWidth * Math.floor(mapPoint.lon / tileMapWidth); │ │ │ │ │ + var tileBottom = tileMapHeight * Math.floor(mapPoint.lat / tileMapHeight); │ │ │ │ │ + return new OpenLayers.Bounds(tileLeft, tileBottom, tileLeft + tileMapWidth, tileBottom + tileMapHeight) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.KaMap" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.KaMapCache = OpenLayers.Class(OpenLayers.Layer.KaMap, { │ │ │ │ │ + IMAGE_EXTENSIONS: { │ │ │ │ │ + jpeg: "jpg", │ │ │ │ │ + gif: "gif", │ │ │ │ │ + png: "png", │ │ │ │ │ + png8: "png", │ │ │ │ │ + png24: "png", │ │ │ │ │ + dithered: "png" │ │ │ │ │ + }, │ │ │ │ │ + DEFAULT_FORMAT: "jpeg", │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.KaMap.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.extension = this.IMAGE_EXTENSIONS[this.params.i.toLowerCase() || this.DEFAULT_FORMAT] │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var mapRes = this.map.getResolution(); │ │ │ │ │ + var scale = Math.round(this.map.getScale() * 1e4) / 1e4; │ │ │ │ │ + var pX = Math.round(bounds.left / mapRes); │ │ │ │ │ + var pY = -Math.round(bounds.top / mapRes); │ │ │ │ │ + var metaX = Math.floor(pX / this.tileSize.w / this.params.metaTileSize.w) * this.tileSize.w * this.params.metaTileSize.w; │ │ │ │ │ + var metaY = Math.floor(pY / this.tileSize.h / this.params.metaTileSize.h) * this.tileSize.h * this.params.metaTileSize.h; │ │ │ │ │ + var components = ["/", this.params.map, "/", scale, "/", this.params.g.replace(/\s/g, "_"), "/def/t", metaY, "/l", metaX, "/t", pY, "l", pX, ".", this.extension]; │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(components.join(""), url) │ │ │ │ │ } │ │ │ │ │ - return featureId │ │ │ │ │ + return url + components.join("") │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.SVG" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.KaMapCache" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_ALIGN = { │ │ │ │ │ - l: "start", │ │ │ │ │ - r: "end", │ │ │ │ │ - b: "bottom", │ │ │ │ │ - t: "hanging" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_VSHIFT = { │ │ │ │ │ - t: "-70%", │ │ │ │ │ - b: "0" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_VFACTOR = { │ │ │ │ │ - t: 0, │ │ │ │ │ - b: -1 │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.SVG.preventDefault = function(e) { │ │ │ │ │ - OpenLayers.Event.preventDefault(e) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Tile.Image.IFrame = { │ │ │ │ │ - useIFrame: null, │ │ │ │ │ - blankImageUrl: "data:image/gif;base64,R0lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAQAIBRAA7", │ │ │ │ │ - draw: function() { │ │ │ │ │ - var draw = OpenLayers.Tile.Image.prototype.shouldDraw.call(this); │ │ │ │ │ - if (draw) { │ │ │ │ │ - var url = this.layer.getURL(this.bounds); │ │ │ │ │ - var usedIFrame = this.useIFrame; │ │ │ │ │ - this.useIFrame = this.maxGetUrlLength !== null && !this.layer.async && url.length > this.maxGetUrlLength; │ │ │ │ │ - var fromIFrame = usedIFrame && !this.useIFrame; │ │ │ │ │ - var toIFrame = !usedIFrame && this.useIFrame; │ │ │ │ │ - if (fromIFrame || toIFrame) { │ │ │ │ │ - if (this.imgDiv && this.imgDiv.parentNode === this.frame) { │ │ │ │ │ - this.frame.removeChild(this.imgDiv) │ │ │ │ │ - } │ │ │ │ │ - this.imgDiv = null; │ │ │ │ │ - if (fromIFrame) { │ │ │ │ │ - this.frame.removeChild(this.frame.firstChild) │ │ │ │ │ - } │ │ │ │ │ +OpenLayers.Layer.ArcGIS93Rest = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + format: "png" │ │ │ │ │ + }, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ + newArguments.push(name, url, params, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + OpenLayers.Util.applyDefaults(this.params, OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS)); │ │ │ │ │ + if (this.params.TRANSPARENT && this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ + if (options == null || !options.isBaseLayer) { │ │ │ │ │ + this.isBaseLayer = false │ │ │ │ │ + } │ │ │ │ │ + if (this.params.FORMAT == "jpg") { │ │ │ │ │ + this.params.FORMAT = OpenLayers.Util.alphaHack() ? "gif" : "png" │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Tile.Image.prototype.draw.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - getImage: function() { │ │ │ │ │ - if (this.useIFrame === true) { │ │ │ │ │ - if (!this.frame.childNodes.length) { │ │ │ │ │ - var eventPane = document.createElement("div"), │ │ │ │ │ - style = eventPane.style; │ │ │ │ │ - style.position = "absolute"; │ │ │ │ │ - style.width = "100%"; │ │ │ │ │ - style.height = "100%"; │ │ │ │ │ - style.zIndex = 1; │ │ │ │ │ - style.backgroundImage = "url(" + this.blankImageUrl + ")"; │ │ │ │ │ - this.frame.appendChild(eventPane) │ │ │ │ │ - } │ │ │ │ │ - var id = this.id + "_iFrame", │ │ │ │ │ - iframe; │ │ │ │ │ - if (parseFloat(navigator.appVersion.split("MSIE")[1]) < 9) { │ │ │ │ │ - iframe = document.createElement('<iframe name="' + id + '">'); │ │ │ │ │ - iframe.style.backgroundColor = "#FFFFFF"; │ │ │ │ │ - iframe.style.filter = "chroma(color=#FFFFFF)" │ │ │ │ │ - } else { │ │ │ │ │ - iframe = document.createElement("iframe"); │ │ │ │ │ - iframe.style.backgroundColor = "transparent"; │ │ │ │ │ - iframe.name = id │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.ArcGIS93Rest(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var projWords = this.projection.getCode().split(":"); │ │ │ │ │ + var srid = projWords[projWords.length - 1]; │ │ │ │ │ + var imageSize = this.getImageSize(); │ │ │ │ │ + var newParams = { │ │ │ │ │ + BBOX: bounds.toBBOX(), │ │ │ │ │ + SIZE: imageSize.w + "," + imageSize.h, │ │ │ │ │ + F: "image", │ │ │ │ │ + BBOXSR: srid, │ │ │ │ │ + IMAGESR: srid │ │ │ │ │ + }; │ │ │ │ │ + if (this.layerDefs) { │ │ │ │ │ + var layerDefStrList = []; │ │ │ │ │ + var layerID; │ │ │ │ │ + for (layerID in this.layerDefs) { │ │ │ │ │ + if (this.layerDefs.hasOwnProperty(layerID)) { │ │ │ │ │ + if (this.layerDefs[layerID]) { │ │ │ │ │ + layerDefStrList.push(layerID); │ │ │ │ │ + layerDefStrList.push(":"); │ │ │ │ │ + layerDefStrList.push(this.layerDefs[layerID]); │ │ │ │ │ + layerDefStrList.push(";") │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - iframe.scrolling = "no"; │ │ │ │ │ - iframe.marginWidth = "0px"; │ │ │ │ │ - iframe.marginHeight = "0px"; │ │ │ │ │ - iframe.frameBorder = "0"; │ │ │ │ │ - iframe.style.position = "absolute"; │ │ │ │ │ - iframe.style.width = "100%"; │ │ │ │ │ - iframe.style.height = "100%"; │ │ │ │ │ - if (this.layer.opacity < 1) { │ │ │ │ │ - OpenLayers.Util.modifyDOMElement(iframe, null, null, null, null, null, null, this.layer.opacity) │ │ │ │ │ + if (layerDefStrList.length > 0) { │ │ │ │ │ + newParams["LAYERDEFS"] = layerDefStrList.join("") │ │ │ │ │ } │ │ │ │ │ - this.frame.appendChild(iframe); │ │ │ │ │ - this.imgDiv = iframe; │ │ │ │ │ - return iframe │ │ │ │ │ - } else { │ │ │ │ │ - return OpenLayers.Tile.Image.prototype.getImage.apply(this, arguments) │ │ │ │ │ } │ │ │ │ │ + var requestString = this.getFullRequestString(newParams); │ │ │ │ │ + return requestString │ │ │ │ │ }, │ │ │ │ │ - createRequestForm: function() { │ │ │ │ │ - var form = document.createElement("form"); │ │ │ │ │ - form.method = "POST"; │ │ │ │ │ - var cacheId = this.layer.params["_OLSALT"]; │ │ │ │ │ - cacheId = (cacheId ? cacheId + "_" : "") + this.bounds.toBBOX(); │ │ │ │ │ - form.action = OpenLayers.Util.urlAppend(this.layer.url, cacheId); │ │ │ │ │ - form.target = this.id + "_iFrame"; │ │ │ │ │ - var imageSize = this.layer.getImageSize(), │ │ │ │ │ - params = OpenLayers.Util.getParameters(this.url), │ │ │ │ │ - field; │ │ │ │ │ - for (var par in params) { │ │ │ │ │ - field = document.createElement("input"); │ │ │ │ │ - field.type = "hidden"; │ │ │ │ │ - field.name = par; │ │ │ │ │ - field.value = params[par]; │ │ │ │ │ - form.appendChild(field) │ │ │ │ │ + setLayerFilter: function(id, queryDef) { │ │ │ │ │ + if (!this.layerDefs) { │ │ │ │ │ + this.layerDefs = {} │ │ │ │ │ + } │ │ │ │ │ + if (queryDef) { │ │ │ │ │ + this.layerDefs[id] = queryDef │ │ │ │ │ + } else { │ │ │ │ │ + delete this.layerDefs[id] │ │ │ │ │ } │ │ │ │ │ - return form │ │ │ │ │ }, │ │ │ │ │ - setImgSrc: function(url) { │ │ │ │ │ - if (this.useIFrame === true) { │ │ │ │ │ - if (url) { │ │ │ │ │ - var form = this.createRequestForm(); │ │ │ │ │ - this.frame.appendChild(form); │ │ │ │ │ - form.submit(); │ │ │ │ │ - this.frame.removeChild(form) │ │ │ │ │ - } else if (this.imgDiv.parentNode === this.frame) { │ │ │ │ │ - this.frame.removeChild(this.imgDiv); │ │ │ │ │ - this.imgDiv = null │ │ │ │ │ - } │ │ │ │ │ + clearLayerFilter: function(id) { │ │ │ │ │ + if (id) { │ │ │ │ │ + delete this.layerDefs[id] │ │ │ │ │ } else { │ │ │ │ │ - OpenLayers.Tile.Image.prototype.setImgSrc.apply(this, arguments) │ │ │ │ │ + delete this.layerDefs │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - onImageLoad: function() { │ │ │ │ │ - OpenLayers.Tile.Image.prototype.onImageLoad.apply(this, arguments); │ │ │ │ │ - if (this.useIFrame === true) { │ │ │ │ │ - this.imgDiv.style.opacity = 1; │ │ │ │ │ - this.frame.style.opacity = this.layer.opacity │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ + var newArguments = [upperParams]; │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, newArguments) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.ArcGIS93Rest" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Image = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + url: null, │ │ │ │ │ + extent: null, │ │ │ │ │ + size: null, │ │ │ │ │ + tile: null, │ │ │ │ │ + aspectRatio: null, │ │ │ │ │ + initialize: function(name, url, extent, size, options) { │ │ │ │ │ + this.url = url; │ │ │ │ │ + this.extent = extent; │ │ │ │ │ + this.maxExtent = extent; │ │ │ │ │ + this.size = size; │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ + this.aspectRatio = this.extent.getHeight() / this.size.h / (this.extent.getWidth() / this.size.w) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.tile) { │ │ │ │ │ + this.removeTileMonitoringHooks(this.tile); │ │ │ │ │ + this.tile.destroy(); │ │ │ │ │ + this.tile = null │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - createBackBuffer: function() { │ │ │ │ │ - var backBuffer; │ │ │ │ │ - if (this.useIFrame === false) { │ │ │ │ │ - backBuffer = OpenLayers.Tile.Image.prototype.createBackBuffer.call(this) │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Image(this.name, this.url, this.extent, this.size, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - return backBuffer │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.SOS = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, OpenLayers.Protocol.SOS.DEFAULTS); │ │ │ │ │ - var cls = OpenLayers.Protocol.SOS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported SOS version: " + options.version │ │ │ │ │ - } │ │ │ │ │ - return new cls(options) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.SOS.DEFAULTS = { │ │ │ │ │ - version: "1.0.0" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.CSW = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, OpenLayers.Protocol.CSW.DEFAULTS); │ │ │ │ │ - var cls = OpenLayers.Protocol.CSW["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported CSW version: " + options.version │ │ │ │ │ - } │ │ │ │ │ - return new cls(options) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.CSW.DEFAULTS = { │ │ │ │ │ - version: "2.0.2" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.WFS = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, OpenLayers.Protocol.WFS.DEFAULTS); │ │ │ │ │ - var cls = OpenLayers.Protocol.WFS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported WFS version: " + options.version │ │ │ │ │ - } │ │ │ │ │ - return new cls(options) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.WFS.fromWMSLayer = function(layer, options) { │ │ │ │ │ - var typeName, featurePrefix; │ │ │ │ │ - var param = layer.params["LAYERS"]; │ │ │ │ │ - var parts = (OpenLayers.Util.isArray(param) ? param[0] : param).split(":"); │ │ │ │ │ - if (parts.length > 1) { │ │ │ │ │ - featurePrefix = parts[0] │ │ │ │ │ - } │ │ │ │ │ - typeName = parts.pop(); │ │ │ │ │ - var protocolOptions = { │ │ │ │ │ - url: layer.url, │ │ │ │ │ - featureType: typeName, │ │ │ │ │ - featurePrefix: featurePrefix, │ │ │ │ │ - srsName: layer.projection && layer.projection.getCode() || layer.map && layer.map.getProjectionObject().getCode(), │ │ │ │ │ - version: "1.1.0" │ │ │ │ │ - }; │ │ │ │ │ - return new OpenLayers.Protocol.WFS(OpenLayers.Util.applyDefaults(options, protocolOptions)) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.WFS.DEFAULTS = { │ │ │ │ │ - version: "1.0.0" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.HTTP = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ - url: null, │ │ │ │ │ - headers: null, │ │ │ │ │ - params: null, │ │ │ │ │ - callback: null, │ │ │ │ │ - scope: null, │ │ │ │ │ - readWithPOST: false, │ │ │ │ │ - updateWithPOST: false, │ │ │ │ │ - deleteWithPOST: false, │ │ │ │ │ - wildcarded: false, │ │ │ │ │ - srsInBBOX: false, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - this.params = {}; │ │ │ │ │ - this.headers = {}; │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ - var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ - wildcarded: this.wildcarded, │ │ │ │ │ - srsInBBOX: this.srsInBBOX │ │ │ │ │ + obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + if (this.options.maxResolution == null) { │ │ │ │ │ + this.options.maxResolution = this.aspectRatio * this.extent.getWidth() / this.size.w │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.prototype.setMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + var firstRendering = this.tile == null; │ │ │ │ │ + if (zoomChanged || firstRendering) { │ │ │ │ │ + this.setTileSize(); │ │ │ │ │ + var ulPx = this.map.getLayerPxFromLonLat({ │ │ │ │ │ + lon: this.extent.left, │ │ │ │ │ + lat: this.extent.top │ │ │ │ │ }); │ │ │ │ │ - this.filterToParams = function(filter, params) { │ │ │ │ │ - return format.write(filter, params) │ │ │ │ │ + if (firstRendering) { │ │ │ │ │ + this.tile = new OpenLayers.Tile.Image(this, ulPx, this.extent, null, this.tileSize); │ │ │ │ │ + this.addTileMonitoringHooks(this.tile) │ │ │ │ │ + } else { │ │ │ │ │ + this.tile.size = this.tileSize.clone(); │ │ │ │ │ + this.tile.position = ulPx.clone() │ │ │ │ │ } │ │ │ │ │ + this.tile.draw() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.params = null; │ │ │ │ │ - this.headers = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ + setTileSize: function() { │ │ │ │ │ + var tileWidth = this.extent.getWidth() / this.map.getResolution(); │ │ │ │ │ + var tileHeight = this.extent.getHeight() / this.map.getResolution(); │ │ │ │ │ + this.tileSize = new OpenLayers.Size(tileWidth, tileHeight) │ │ │ │ │ }, │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + addTileMonitoringHooks: function(tile) { │ │ │ │ │ + tile.onLoadStart = function() { │ │ │ │ │ + this.events.triggerEvent("loadstart") │ │ │ │ │ + }; │ │ │ │ │ + tile.events.register("loadstart", this, tile.onLoadStart); │ │ │ │ │ + tile.onLoadEnd = function() { │ │ │ │ │ + this.events.triggerEvent("loadend") │ │ │ │ │ + }; │ │ │ │ │ + tile.events.register("loadend", this, tile.onLoadEnd); │ │ │ │ │ + tile.events.register("unload", this, tile.onLoadEnd) │ │ │ │ │ + }, │ │ │ │ │ + removeTileMonitoringHooks: function(tile) { │ │ │ │ │ + tile.unload(); │ │ │ │ │ + tile.events.un({ │ │ │ │ │ + loadstart: tile.onLoadStart, │ │ │ │ │ + loadend: tile.onLoadEnd, │ │ │ │ │ + unload: tile.onLoadEnd, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + setUrl: function(newUrl) { │ │ │ │ │ + this.url = newUrl; │ │ │ │ │ + this.tile.draw() │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + return this.url │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Image" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Google = OpenLayers.Class(OpenLayers.Layer.EventPane, OpenLayers.Layer.FixedZoomLevels, { │ │ │ │ │ + MIN_ZOOM_LEVEL: 0, │ │ │ │ │ + MAX_ZOOM_LEVEL: 21, │ │ │ │ │ + RESOLUTIONS: [1.40625, .703125, .3515625, .17578125, .087890625, .0439453125, .02197265625, .010986328125, .0054931640625, .00274658203125, .001373291015625, .0006866455078125, .00034332275390625, .000171661376953125, 858306884765625e-19, 4291534423828125e-20, 2145767211914062e-20, 1072883605957031e-20, 536441802978515e-20, 268220901489257e-20, 1341104507446289e-21, 6.705522537231445e-7], │ │ │ │ │ + type: null, │ │ │ │ │ + wrapDateLine: true, │ │ │ │ │ + sphericalMercator: false, │ │ │ │ │ + version: null, │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ options = options || {}; │ │ │ │ │ - options.params = OpenLayers.Util.applyDefaults(options.params, this.options.params); │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - if (options.filter && this.filterToParams) { │ │ │ │ │ - options.params = this.filterToParams(options.filter, options.params) │ │ │ │ │ + if (!options.version) { │ │ │ │ │ + options.version = typeof GMap2 === "function" ? "2" : "3" │ │ │ │ │ } │ │ │ │ │ - var readWithPOST = options.readWithPOST !== undefined ? options.readWithPOST : this.readWithPOST; │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - if (readWithPOST) { │ │ │ │ │ - var headers = options.headers || {}; │ │ │ │ │ - headers["Content-Type"] = "application/x-www-form-urlencoded"; │ │ │ │ │ - resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ - data: OpenLayers.Util.getParameterString(options.params), │ │ │ │ │ - headers: headers │ │ │ │ │ - }) │ │ │ │ │ + var mixin = OpenLayers.Layer.Google["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (mixin) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, mixin) │ │ │ │ │ } else { │ │ │ │ │ - resp.priv = OpenLayers.Request.GET({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers │ │ │ │ │ - }) │ │ │ │ │ + throw "Unsupported Google Maps API version: " + options.version │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, mixin.DEFAULTS); │ │ │ │ │ + if (options.maxExtent) { │ │ │ │ │ + options.maxExtent = options.maxExtent.clone() │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ + OpenLayers.Layer.FixedZoomLevels.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + OpenLayers.Util.extend(this, OpenLayers.Layer.SphericalMercator); │ │ │ │ │ + this.initMercatorParameters() │ │ │ │ │ } │ │ │ │ │ - return resp │ │ │ │ │ }, │ │ │ │ │ - handleRead: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options) │ │ │ │ │ + clone: function() { │ │ │ │ │ + return new OpenLayers.Layer.Google(this.name, this.getOptions()) │ │ │ │ │ }, │ │ │ │ │ - create: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: features, │ │ │ │ │ - requestType: "create" │ │ │ │ │ - }); │ │ │ │ │ - resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleCreate, resp, options), │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(features) │ │ │ │ │ - }); │ │ │ │ │ - return resp │ │ │ │ │ + setVisibility: function(visible) { │ │ │ │ │ + var opacity = this.opacity == null ? 1 : this.opacity; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.setVisibility.apply(this, arguments); │ │ │ │ │ + this.setOpacity(opacity) │ │ │ │ │ }, │ │ │ │ │ - handleCreate: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options) │ │ │ │ │ + display: function(visible) { │ │ │ │ │ + if (!this._dragging) { │ │ │ │ │ + this.setGMapVisibility(visible) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.display.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - update: function(feature, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var url = options.url || feature.url || this.options.url + "/" + feature.fid; │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: feature, │ │ │ │ │ - requestType: "update" │ │ │ │ │ - }); │ │ │ │ │ - var method = this.updateWithPOST ? "POST" : "PUT"; │ │ │ │ │ - resp.priv = OpenLayers.Request[method]({ │ │ │ │ │ - url: url, │ │ │ │ │ - callback: this.createCallback(this.handleUpdate, resp, options), │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(feature) │ │ │ │ │ - }); │ │ │ │ │ - return resp │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + this._dragging = dragging; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + delete this._dragging │ │ │ │ │ }, │ │ │ │ │ - handleUpdate: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options) │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + if (opacity !== this.opacity) { │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "opacity" │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + this.opacity = opacity │ │ │ │ │ + } │ │ │ │ │ + if (this.getVisibility()) { │ │ │ │ │ + var container = this.getMapContainer(); │ │ │ │ │ + OpenLayers.Util.modifyDOMElement(container, null, null, null, null, null, null, opacity) │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - delete: function(feature, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var url = options.url || feature.url || this.options.url + "/" + feature.fid; │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: feature, │ │ │ │ │ - requestType: "delete" │ │ │ │ │ - }); │ │ │ │ │ - var method = this.deleteWithPOST ? "POST" : "DELETE"; │ │ │ │ │ - var requestOptions = { │ │ │ │ │ - url: url, │ │ │ │ │ - callback: this.createCallback(this.handleDelete, resp, options), │ │ │ │ │ - headers: options.headers │ │ │ │ │ - }; │ │ │ │ │ - if (this.deleteWithPOST) { │ │ │ │ │ - requestOptions.data = this.format.write(feature) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.setGMapVisibility(false); │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache && cache.count <= 1) { │ │ │ │ │ + this.removeGMapElements() │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - resp.priv = OpenLayers.Request[method](requestOptions); │ │ │ │ │ - return resp │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - handleDelete: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options) │ │ │ │ │ + removeGMapElements: function() { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + var container = this.mapObject && this.getMapContainer(); │ │ │ │ │ + if (container && container.parentNode) { │ │ │ │ │ + container.parentNode.removeChild(container) │ │ │ │ │ + } │ │ │ │ │ + var termsOfUse = cache.termsOfUse; │ │ │ │ │ + if (termsOfUse && termsOfUse.parentNode) { │ │ │ │ │ + termsOfUse.parentNode.removeChild(termsOfUse) │ │ │ │ │ + } │ │ │ │ │ + var poweredBy = cache.poweredBy; │ │ │ │ │ + if (poweredBy && poweredBy.parentNode) { │ │ │ │ │ + poweredBy.parentNode.removeChild(poweredBy) │ │ │ │ │ + } │ │ │ │ │ + if (this.mapObject && window.google && google.maps && google.maps.event && google.maps.event.clearListeners) { │ │ │ │ │ + google.maps.event.clearListeners(this.mapObject, "tilesloaded") │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - handleResponse: function(resp, options) { │ │ │ │ │ - var request = resp.priv; │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - if (resp.requestType != "delete") { │ │ │ │ │ - resp.features = this.parseFeatures(request) │ │ │ │ │ - } │ │ │ │ │ - resp.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + if (this.visibility && this.mapObject) { │ │ │ │ │ + this.setGMapVisibility(false) │ │ │ │ │ + } │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + if (cache.count <= 1) { │ │ │ │ │ + this.removeGMapElements(); │ │ │ │ │ + delete OpenLayers.Layer.Google.cache[map.id] │ │ │ │ │ } else { │ │ │ │ │ - resp.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ + --cache.count │ │ │ │ │ } │ │ │ │ │ - options.callback.call(options.scope, resp) │ │ │ │ │ } │ │ │ │ │ + delete this.termsOfUse; │ │ │ │ │ + delete this.poweredBy; │ │ │ │ │ + delete this.mapObject; │ │ │ │ │ + delete this.dragObject; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.removeMap.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - parseFeatures: function(request) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null │ │ │ │ │ + getOLBoundsFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + var olBounds = null; │ │ │ │ │ + if (moBounds != null) { │ │ │ │ │ + var sw = moBounds.getSouthWest(); │ │ │ │ │ + var ne = moBounds.getNorthEast(); │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + sw = this.forwardMercator(sw.lng(), sw.lat()); │ │ │ │ │ + ne = this.forwardMercator(ne.lng(), ne.lat()) │ │ │ │ │ + } else { │ │ │ │ │ + sw = new OpenLayers.LonLat(sw.lng(), sw.lat()); │ │ │ │ │ + ne = new OpenLayers.LonLat(ne.lng(), ne.lat()) │ │ │ │ │ + } │ │ │ │ │ + olBounds = new OpenLayers.Bounds(sw.lon, sw.lat, ne.lon, ne.lat) │ │ │ │ │ } │ │ │ │ │ - return this.format.read(doc) │ │ │ │ │ + return olBounds │ │ │ │ │ }, │ │ │ │ │ - commit: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = [], │ │ │ │ │ - nResponses = 0; │ │ │ │ │ - var types = {}; │ │ │ │ │ - types[OpenLayers.State.INSERT] = []; │ │ │ │ │ - types[OpenLayers.State.UPDATE] = []; │ │ │ │ │ - types[OpenLayers.State.DELETE] = []; │ │ │ │ │ - var feature, list, requestFeatures = []; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - list = types[feature.state]; │ │ │ │ │ - if (list) { │ │ │ │ │ - list.push(feature); │ │ │ │ │ - requestFeatures.push(feature) │ │ │ │ │ + getWarningHTML: function() { │ │ │ │ │ + return OpenLayers.i18n("googleWarning") │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectCenter: function() { │ │ │ │ │ + return this.mapObject.getCenter() │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectZoom: function() { │ │ │ │ │ + return this.mapObject.getZoom() │ │ │ │ │ + }, │ │ │ │ │ + getLongitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + return this.sphericalMercator ? this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lon : moLonLat.lng() │ │ │ │ │ + }, │ │ │ │ │ + getLatitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + var lat = this.sphericalMercator ? this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lat : moLonLat.lat(); │ │ │ │ │ + return lat │ │ │ │ │ + }, │ │ │ │ │ + getXFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return moPixel.x │ │ │ │ │ + }, │ │ │ │ │ + getYFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return moPixel.y │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Google" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Google.cache = {}; │ │ │ │ │ +OpenLayers.Layer.Google.v2 = { │ │ │ │ │ + termsOfUse: null, │ │ │ │ │ + poweredBy: null, │ │ │ │ │ + dragObject: null, │ │ │ │ │ + loadMapObject: function() { │ │ │ │ │ + if (!this.type) { │ │ │ │ │ + this.type = G_NORMAL_MAP │ │ │ │ │ + } │ │ │ │ │ + var mapObject, termsOfUse, poweredBy; │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + mapObject = cache.mapObject; │ │ │ │ │ + termsOfUse = cache.termsOfUse; │ │ │ │ │ + poweredBy = cache.poweredBy; │ │ │ │ │ + ++cache.count │ │ │ │ │ + } else { │ │ │ │ │ + var container = this.map.viewPortDiv; │ │ │ │ │ + var div = document.createElement("div"); │ │ │ │ │ + div.id = this.map.id + "_GMap2Container"; │ │ │ │ │ + div.style.position = "absolute"; │ │ │ │ │ + div.style.width = "100%"; │ │ │ │ │ + div.style.height = "100%"; │ │ │ │ │ + container.appendChild(div); │ │ │ │ │ + try { │ │ │ │ │ + mapObject = new GMap2(div); │ │ │ │ │ + termsOfUse = div.lastChild; │ │ │ │ │ + container.appendChild(termsOfUse); │ │ │ │ │ + termsOfUse.style.zIndex = "1100"; │ │ │ │ │ + termsOfUse.style.right = ""; │ │ │ │ │ + termsOfUse.style.bottom = ""; │ │ │ │ │ + termsOfUse.className = "olLayerGoogleCopyright"; │ │ │ │ │ + poweredBy = div.lastChild; │ │ │ │ │ + container.appendChild(poweredBy); │ │ │ │ │ + poweredBy.style.zIndex = "1100"; │ │ │ │ │ + poweredBy.style.right = ""; │ │ │ │ │ + poweredBy.style.bottom = ""; │ │ │ │ │ + poweredBy.className = "olLayerGooglePoweredBy gmnoprint" │ │ │ │ │ + } catch (e) { │ │ │ │ │ + throw e │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.Google.cache[this.map.id] = { │ │ │ │ │ + mapObject: mapObject, │ │ │ │ │ + termsOfUse: termsOfUse, │ │ │ │ │ + poweredBy: poweredBy, │ │ │ │ │ + count: 1 │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var nRequests = (types[OpenLayers.State.INSERT].length > 0 ? 1 : 0) + types[OpenLayers.State.UPDATE].length + types[OpenLayers.State.DELETE].length; │ │ │ │ │ - var success = true; │ │ │ │ │ - var finalResponse = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: requestFeatures │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - function insertCallback(response) { │ │ │ │ │ - var len = response.features ? response.features.length : 0; │ │ │ │ │ - var fids = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - fids[i] = response.features[i].fid │ │ │ │ │ + this.mapObject = mapObject; │ │ │ │ │ + this.termsOfUse = termsOfUse; │ │ │ │ │ + this.poweredBy = poweredBy; │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.mapObject.getMapTypes(), this.type) === -1) { │ │ │ │ │ + this.mapObject.addMapType(this.type) │ │ │ │ │ + } │ │ │ │ │ + if (typeof mapObject.getDragObject == "function") { │ │ │ │ │ + this.dragObject = mapObject.getDragObject() │ │ │ │ │ + } else { │ │ │ │ │ + this.dragPanMapObject = null │ │ │ │ │ + } │ │ │ │ │ + if (this.isBaseLayer === false) { │ │ │ │ │ + this.setGMapVisibility(this.div.style.display !== "none") │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + if (this.visibility && this.mapObject.isLoaded()) { │ │ │ │ │ + this.mapObject.checkResize() │ │ │ │ │ + } else { │ │ │ │ │ + if (!this._resized) { │ │ │ │ │ + var layer = this; │ │ │ │ │ + var handle = GEvent.addListener(this.mapObject, "load", function() { │ │ │ │ │ + GEvent.removeListener(handle); │ │ │ │ │ + delete layer._resized; │ │ │ │ │ + layer.mapObject.checkResize(); │ │ │ │ │ + layer.moveTo(layer.map.getCenter(), layer.map.getZoom()) │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - finalResponse.insertIds = fids; │ │ │ │ │ - callback.apply(this, [response]) │ │ │ │ │ + this._resized = true │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - function callback(response) { │ │ │ │ │ - this.callUserCallback(response, options); │ │ │ │ │ - success = success && response.success(); │ │ │ │ │ - nResponses++; │ │ │ │ │ - if (nResponses >= nRequests) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - finalResponse.code = success ? OpenLayers.Protocol.Response.SUCCESS : OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - options.callback.apply(options.scope, [finalResponse]) │ │ │ │ │ + }, │ │ │ │ │ + setGMapVisibility: function(visible) { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + var container = this.mapObject.getContainer(); │ │ │ │ │ + if (visible === true) { │ │ │ │ │ + this.mapObject.setMapType(this.type); │ │ │ │ │ + container.style.display = ""; │ │ │ │ │ + this.termsOfUse.style.left = ""; │ │ │ │ │ + this.termsOfUse.style.display = ""; │ │ │ │ │ + this.poweredBy.style.display = ""; │ │ │ │ │ + cache.displayed = this.id │ │ │ │ │ + } else { │ │ │ │ │ + if (cache.displayed === this.id) { │ │ │ │ │ + delete cache.displayed │ │ │ │ │ + } │ │ │ │ │ + if (!cache.displayed) { │ │ │ │ │ + container.style.display = "none"; │ │ │ │ │ + this.termsOfUse.style.display = "none"; │ │ │ │ │ + this.termsOfUse.style.left = "-9999px"; │ │ │ │ │ + this.poweredBy.style.display = "none" │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var queue = types[OpenLayers.State.INSERT]; │ │ │ │ │ - if (queue.length > 0) { │ │ │ │ │ - resp.push(this.create(queue, OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: insertCallback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options.create))) │ │ │ │ │ + }, │ │ │ │ │ + getMapContainer: function() { │ │ │ │ │ + return this.mapObject.getContainer() │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ + var moBounds = null; │ │ │ │ │ + if (olBounds != null) { │ │ │ │ │ + var sw = this.sphericalMercator ? this.inverseMercator(olBounds.bottom, olBounds.left) : new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ + var ne = this.sphericalMercator ? this.inverseMercator(olBounds.top, olBounds.right) : new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ + moBounds = new GLatLngBounds(new GLatLng(sw.lat, sw.lon), new GLatLng(ne.lat, ne.lon)) │ │ │ │ │ } │ │ │ │ │ - queue = types[OpenLayers.State.UPDATE]; │ │ │ │ │ - for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ - resp.push(this.update(queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: callback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options.update))) │ │ │ │ │ + return moBounds │ │ │ │ │ + }, │ │ │ │ │ + setMapObjectCenter: function(center, zoom) { │ │ │ │ │ + this.mapObject.setCenter(center, zoom) │ │ │ │ │ + }, │ │ │ │ │ + dragPanMapObject: function(dX, dY) { │ │ │ │ │ + this.dragObject.moveBy(new GSize(-dX, dY)) │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return this.mapObject.fromContainerPixelToLatLng(moPixel) │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + return this.mapObject.fromLatLngToContainerPixel(moLonLat) │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + return this.mapObject.getBoundsZoomLevel(moBounds) │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ + var gLatLng; │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ + gLatLng = new GLatLng(lonlat.lat, lonlat.lon) │ │ │ │ │ + } else { │ │ │ │ │ + gLatLng = new GLatLng(lat, lon) │ │ │ │ │ } │ │ │ │ │ - queue = types[OpenLayers.State.DELETE]; │ │ │ │ │ - for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ - resp.push(this["delete"](queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: callback, │ │ │ │ │ + return gLatLng │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ + return new GPoint(x, y) │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Layer.GeoRSS = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ + location: null, │ │ │ │ │ + features: null, │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + selectedFeature: null, │ │ │ │ │ + icon: null, │ │ │ │ │ + popupSize: null, │ │ │ │ │ + useFeedTitle: true, │ │ │ │ │ + initialize: function(name, location, options) { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ + this.location = location; │ │ │ │ │ + this.features = [] │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.destroy.apply(this, arguments); │ │ │ │ │ + this.clearFeatures(); │ │ │ │ │ + this.features = null │ │ │ │ │ + }, │ │ │ │ │ + loadRSS: function() { │ │ │ │ │ + if (!this.loaded) { │ │ │ │ │ + this.events.triggerEvent("loadstart"); │ │ │ │ │ + OpenLayers.Request.GET({ │ │ │ │ │ + url: this.location, │ │ │ │ │ + success: this.parseData, │ │ │ │ │ scope: this │ │ │ │ │ - }, options["delete"]))) │ │ │ │ │ + }); │ │ │ │ │ + this.loaded = true │ │ │ │ │ } │ │ │ │ │ - return resp │ │ │ │ │ }, │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - response.priv.abort() │ │ │ │ │ + moveTo: function(bounds, zoomChanged, minor) { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + if (this.visibility && !this.loaded) { │ │ │ │ │ + this.loadRSS() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - callUserCallback: function(resp, options) { │ │ │ │ │ - var opt = options[resp.requestType]; │ │ │ │ │ - if (opt && opt.callback) { │ │ │ │ │ - opt.callback.call(opt.scope, resp) │ │ │ │ │ + parseData: function(ajaxRequest) { │ │ │ │ │ + var doc = ajaxRequest.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = OpenLayers.Format.XML.prototype.read(ajaxRequest.responseText) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.HTTP" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Protocol.Script = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ - url: null, │ │ │ │ │ - params: null, │ │ │ │ │ - callback: null, │ │ │ │ │ - callbackTemplate: "OpenLayers.Protocol.Script.registry.${id}", │ │ │ │ │ - callbackKey: "callback", │ │ │ │ │ - callbackPrefix: "", │ │ │ │ │ - scope: null, │ │ │ │ │ - format: null, │ │ │ │ │ - pendingRequests: null, │ │ │ │ │ - srsInBBOX: false, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - this.params = {}; │ │ │ │ │ - this.pendingRequests = {}; │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (!this.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.GeoJSON │ │ │ │ │ + if (this.useFeedTitle) { │ │ │ │ │ + var name = null; │ │ │ │ │ + try { │ │ │ │ │ + name = doc.getElementsByTagNameNS("*", "title")[0].firstChild.nodeValue │ │ │ │ │ + } catch (e) { │ │ │ │ │ + try { │ │ │ │ │ + name = doc.getElementsByTagName("title")[0].firstChild.nodeValue │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + } │ │ │ │ │ + if (name) { │ │ │ │ │ + this.setName(name) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ - var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ - srsInBBOX: this.srsInBBOX │ │ │ │ │ - }); │ │ │ │ │ - this.filterToParams = function(filter, params) { │ │ │ │ │ - return format.write(filter, params) │ │ │ │ │ + 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 format = new OpenLayers.Format.GeoRSS(options); │ │ │ │ │ + var features = format.read(doc); │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + var data = {}; │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + if (!feature.geometry) { │ │ │ │ │ + continue │ │ │ │ │ } │ │ │ │ │ + var title = feature.attributes.title ? feature.attributes.title : "Untitled"; │ │ │ │ │ + var description = feature.attributes.description ? feature.attributes.description : "No description."; │ │ │ │ │ + var link = feature.attributes.link ? feature.attributes.link : ""; │ │ │ │ │ + var location = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + data.icon = this.icon == null ? OpenLayers.Marker.defaultIcon() : this.icon.clone(); │ │ │ │ │ + data.popupSize = this.popupSize ? this.popupSize.clone() : new OpenLayers.Size(250, 120); │ │ │ │ │ + if (title || description) { │ │ │ │ │ + data.title = title; │ │ │ │ │ + data.description = description; │ │ │ │ │ + var contentHTML = '<div class="olLayerGeoRSSClose">[x]</div>'; │ │ │ │ │ + contentHTML += '<div class="olLayerGeoRSSTitle">'; │ │ │ │ │ + if (link) { │ │ │ │ │ + contentHTML += '<a class="link" href="' + link + '" target="_blank">' │ │ │ │ │ + } │ │ │ │ │ + contentHTML += title; │ │ │ │ │ + if (link) { │ │ │ │ │ + contentHTML += "</a>" │ │ │ │ │ + } │ │ │ │ │ + contentHTML += "</div>"; │ │ │ │ │ + contentHTML += '<div style="" class="olLayerGeoRSSDescription">'; │ │ │ │ │ + contentHTML += description; │ │ │ │ │ + contentHTML += "</div>"; │ │ │ │ │ + data["popupContentHTML"] = contentHTML │ │ │ │ │ + } │ │ │ │ │ + var feature = new OpenLayers.Feature(this, location, data); │ │ │ │ │ + this.features.push(feature); │ │ │ │ │ + var marker = feature.createMarker(); │ │ │ │ │ + marker.events.register("click", feature, this.markerClick); │ │ │ │ │ + this.addMarker(marker) │ │ │ │ │ } │ │ │ │ │ + this.events.triggerEvent("loadend") │ │ │ │ │ }, │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - options.params = OpenLayers.Util.applyDefaults(options.params, this.options.params); │ │ │ │ │ - if (options.filter && this.filterToParams) { │ │ │ │ │ - options.params = this.filterToParams(options.filter, options.params) │ │ │ │ │ + markerClick: function(evt) { │ │ │ │ │ + var sameMarkerClicked = this == this.layer.selectedFeature; │ │ │ │ │ + this.layer.selectedFeature = !sameMarkerClicked ? this : null; │ │ │ │ │ + for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ + this.layer.map.removePopup(this.layer.map.popups[i]) │ │ │ │ │ } │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - var request = this.createRequest(options.url, options.params, OpenLayers.Function.bind(function(data) { │ │ │ │ │ - response.data = data; │ │ │ │ │ - this.handleRead(response, options) │ │ │ │ │ - }, this)); │ │ │ │ │ - response.priv = request; │ │ │ │ │ - return response │ │ │ │ │ - }, │ │ │ │ │ - createRequest: function(url, params, callback) { │ │ │ │ │ - var id = OpenLayers.Protocol.Script.register(callback); │ │ │ │ │ - var name = OpenLayers.String.format(this.callbackTemplate, { │ │ │ │ │ - id: id │ │ │ │ │ - }); │ │ │ │ │ - params = OpenLayers.Util.extend({}, params); │ │ │ │ │ - params[this.callbackKey] = this.callbackPrefix + name; │ │ │ │ │ - url = OpenLayers.Util.urlAppend(url, OpenLayers.Util.getParameterString(params)); │ │ │ │ │ - var script = document.createElement("script"); │ │ │ │ │ - script.type = "text/javascript"; │ │ │ │ │ - script.src = url; │ │ │ │ │ - script.id = "OpenLayers_Protocol_Script_" + id; │ │ │ │ │ - this.pendingRequests[script.id] = script; │ │ │ │ │ - var head = document.getElementsByTagName("head")[0]; │ │ │ │ │ - head.appendChild(script); │ │ │ │ │ - return script │ │ │ │ │ + if (!sameMarkerClicked) { │ │ │ │ │ + var popup = this.createPopup(); │ │ │ │ │ + OpenLayers.Event.observe(popup.div, "click", OpenLayers.Function.bind(function() { │ │ │ │ │ + for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ + this.layer.map.removePopup(this.layer.map.popups[i]) │ │ │ │ │ + } │ │ │ │ │ + }, this)); │ │ │ │ │ + this.layer.map.addPopup(popup) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ }, │ │ │ │ │ - destroyRequest: function(script) { │ │ │ │ │ - OpenLayers.Protocol.Script.unregister(script.id.split("_").pop()); │ │ │ │ │ - delete this.pendingRequests[script.id]; │ │ │ │ │ - if (script.parentNode) { │ │ │ │ │ - script.parentNode.removeChild(script) │ │ │ │ │ + clearFeatures: function() { │ │ │ │ │ + if (this.features != null) { │ │ │ │ │ + while (this.features.length > 0) { │ │ │ │ │ + var feature = this.features[0]; │ │ │ │ │ + OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ + feature.destroy() │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - this.handleResponse(response, options) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.GeoRSS" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.MapServer = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + mode: "map", │ │ │ │ │ + map_imagetype: "png" │ │ │ │ │ }, │ │ │ │ │ - handleResponse: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - if (response.data) { │ │ │ │ │ - response.features = this.parseFeatures(response.data); │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ - } │ │ │ │ │ - this.destroyRequest(response.priv); │ │ │ │ │ - options.callback.call(options.scope, response) │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.params = OpenLayers.Util.applyDefaults(this.params, this.DEFAULT_PARAMS); │ │ │ │ │ + if (options == null || options.isBaseLayer == null) { │ │ │ │ │ + this.isBaseLayer = this.params.transparent != "true" && this.params.transparent != true │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - parseFeatures: function(data) { │ │ │ │ │ - return this.format.read(data) │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.MapServer(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - this.destroyRequest(response.priv) │ │ │ │ │ - } else { │ │ │ │ │ - for (var key in this.pendingRequests) { │ │ │ │ │ - this.destroyRequest(this.pendingRequests[key]) │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var extent = [bounds.left, bounds.bottom, bounds.right, bounds.top]; │ │ │ │ │ + var imageSize = this.getImageSize(); │ │ │ │ │ + var url = this.getFullRequestString({ │ │ │ │ │ + mapext: extent, │ │ │ │ │ + imgext: extent, │ │ │ │ │ + map_size: [imageSize.w, imageSize.h], │ │ │ │ │ + imgx: imageSize.w / 2, │ │ │ │ │ + imgy: imageSize.h / 2, │ │ │ │ │ + imgxy: [imageSize.w, imageSize.h] │ │ │ │ │ + }); │ │ │ │ │ + return url │ │ │ │ │ + }, │ │ │ │ │ + getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ + var url = altUrl == null ? this.url : altUrl; │ │ │ │ │ + var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ + allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ + var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(paramsString, url) │ │ │ │ │ + } │ │ │ │ │ + var urlParams = OpenLayers.Util.upperCaseObject(OpenLayers.Util.getParameters(url)); │ │ │ │ │ + for (var key in allParams) { │ │ │ │ │ + if (key.toUpperCase() in urlParams) { │ │ │ │ │ + delete allParams[key] │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + var requestString = url; │ │ │ │ │ + paramsString = paramsString.replace(/,/g, "+"); │ │ │ │ │ + if (paramsString != "") { │ │ │ │ │ + var lastServerChar = url.charAt(url.length - 1); │ │ │ │ │ + if (lastServerChar == "&" || lastServerChar == "?") { │ │ │ │ │ + requestString += paramsString │ │ │ │ │ + } else { │ │ │ │ │ + if (url.indexOf("?") == -1) { │ │ │ │ │ + requestString += "?" + paramsString │ │ │ │ │ + } else { │ │ │ │ │ + requestString += "&" + paramsString │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return requestString │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.abort(); │ │ │ │ │ - delete this.params; │ │ │ │ │ - delete this.format; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.Script" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.MapServer" │ │ │ │ │ }); │ │ │ │ │ -(function() { │ │ │ │ │ - var o = OpenLayers.Protocol.Script; │ │ │ │ │ - var counter = 0; │ │ │ │ │ - o.registry = {}; │ │ │ │ │ - o.register = function(callback) { │ │ │ │ │ - var id = "c" + ++counter; │ │ │ │ │ - o.registry[id] = function() { │ │ │ │ │ - callback.apply(this, arguments) │ │ │ │ │ - }; │ │ │ │ │ - return id │ │ │ │ │ - }; │ │ │ │ │ - o.unregister = function(id) { │ │ │ │ │ - delete o.registry[id] │ │ │ │ │ - } │ │ │ │ │ -})(); │ │ │ │ │ -OpenLayers.Protocol.WFS.v1 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ - version: null, │ │ │ │ │ - srsName: "EPSG:4326", │ │ │ │ │ - featureType: null, │ │ │ │ │ - featureNS: null, │ │ │ │ │ - geometryName: "the_geom", │ │ │ │ │ - schema: null, │ │ │ │ │ - featurePrefix: "feature", │ │ │ │ │ - formatOptions: null, │ │ │ │ │ - readFormat: null, │ │ │ │ │ - readOptions: null, │ │ │ │ │ +OpenLayers.Layer.UTFGrid = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ + isBaseLayer: false, │ │ │ │ │ + projection: new OpenLayers.Projection("EPSG:900913"), │ │ │ │ │ + useJSONP: false, │ │ │ │ │ + tileClass: OpenLayers.Tile.UTFGrid, │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!options.format) { │ │ │ │ │ - this.format = OpenLayers.Format.WFST(OpenLayers.Util.extend({ │ │ │ │ │ - version: this.version, │ │ │ │ │ - featureType: this.featureType, │ │ │ │ │ - featureNS: this.featureNS, │ │ │ │ │ - featurePrefix: this.featurePrefix, │ │ │ │ │ - geometryName: this.geometryName, │ │ │ │ │ - srsName: this.srsName, │ │ │ │ │ - schema: this.schema │ │ │ │ │ - }, this.formatOptions)) │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [options.name, options.url, {}, options]); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + utfgridResolution: this.utfgridResolution │ │ │ │ │ + }, this.tileOptions) │ │ │ │ │ + }, │ │ │ │ │ + createBackBuffer: function() {}, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.UTFGrid(this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - if (!options.geometryName && parseFloat(this.format.version) > 1) { │ │ │ │ │ - this.setGeometryName(null) │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + getFeatureInfo: function(location) { │ │ │ │ │ + var info = null; │ │ │ │ │ + var tileInfo = this.getTileData(location); │ │ │ │ │ + if (tileInfo && tileInfo.tile) { │ │ │ │ │ + info = tileInfo.tile.getFeatureInfo(tileInfo.i, tileInfo.j) │ │ │ │ │ } │ │ │ │ │ + return info │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.options && !this.options.format) { │ │ │ │ │ - this.format.destroy() │ │ │ │ │ + getFeatureId: function(location) { │ │ │ │ │ + var id = null; │ │ │ │ │ + var info = this.getTileData(location); │ │ │ │ │ + if (info.tile) { │ │ │ │ │ + id = info.tile.getFeatureId(info.i, info.j) │ │ │ │ │ } │ │ │ │ │ - this.format = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ + return id │ │ │ │ │ }, │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - var data = OpenLayers.Format.XML.prototype.write.apply(this.format, [this.format.writeNode("wfs:GetFeature", options)]); │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: data │ │ │ │ │ - }); │ │ │ │ │ - return response │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.UTFGrid" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Google.v3 = { │ │ │ │ │ + DEFAULTS: { │ │ │ │ │ + sphericalMercator: true, │ │ │ │ │ + projection: "EPSG:900913" │ │ │ │ │ }, │ │ │ │ │ - setFeatureType: function(featureType) { │ │ │ │ │ - this.featureType = featureType; │ │ │ │ │ - this.format.featureType = featureType │ │ │ │ │ + animationEnabled: true, │ │ │ │ │ + loadMapObject: function() { │ │ │ │ │ + if (!this.type) { │ │ │ │ │ + this.type = google.maps.MapTypeId.ROADMAP │ │ │ │ │ + } │ │ │ │ │ + var mapObject; │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + mapObject = cache.mapObject; │ │ │ │ │ + ++cache.count │ │ │ │ │ + } else { │ │ │ │ │ + var center = this.map.getCenter(); │ │ │ │ │ + var container = document.createElement("div"); │ │ │ │ │ + container.className = "olForeignContainer"; │ │ │ │ │ + container.style.width = "100%"; │ │ │ │ │ + container.style.height = "100%"; │ │ │ │ │ + mapObject = new google.maps.Map(container, { │ │ │ │ │ + center: center ? new google.maps.LatLng(center.lat, center.lon) : new google.maps.LatLng(0, 0), │ │ │ │ │ + zoom: this.map.getZoom() || 0, │ │ │ │ │ + mapTypeId: this.type, │ │ │ │ │ + disableDefaultUI: true, │ │ │ │ │ + keyboardShortcuts: false, │ │ │ │ │ + draggable: false, │ │ │ │ │ + disableDoubleClickZoom: true, │ │ │ │ │ + scrollwheel: false, │ │ │ │ │ + streetViewControl: false │ │ │ │ │ + }); │ │ │ │ │ + var googleControl = document.createElement("div"); │ │ │ │ │ + googleControl.style.width = "100%"; │ │ │ │ │ + googleControl.style.height = "100%"; │ │ │ │ │ + mapObject.controls[google.maps.ControlPosition.TOP_LEFT].push(googleControl); │ │ │ │ │ + cache = { │ │ │ │ │ + googleControl: googleControl, │ │ │ │ │ + mapObject: mapObject, │ │ │ │ │ + count: 1 │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Layer.Google.cache[this.map.id] = cache │ │ │ │ │ + } │ │ │ │ │ + this.mapObject = mapObject; │ │ │ │ │ + this.setGMapVisibility(this.visibility) │ │ │ │ │ }, │ │ │ │ │ - setGeometryName: function(geometryName) { │ │ │ │ │ - this.geometryName = geometryName; │ │ │ │ │ - this.format.geometryName = geometryName │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + if (this.visibility) { │ │ │ │ │ + google.maps.event.trigger(this.mapObject, "resize") │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - var result = this.parseResponse(request, options.readOptions); │ │ │ │ │ - if (result && result.success !== false) { │ │ │ │ │ - if (options.readOptions && options.readOptions.output == "object") { │ │ │ │ │ - OpenLayers.Util.extend(response, result) │ │ │ │ │ + setGMapVisibility: function(visible) { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + var map = this.map; │ │ │ │ │ + if (cache) { │ │ │ │ │ + var type = this.type; │ │ │ │ │ + var layers = map.layers; │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = layers[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Google && layer.visibility === true && layer.inRange === true) { │ │ │ │ │ + type = layer.type; │ │ │ │ │ + visible = true; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var container = this.mapObject.getDiv(); │ │ │ │ │ + if (visible === true) { │ │ │ │ │ + if (container.parentNode !== map.div) { │ │ │ │ │ + if (!cache.rendered) { │ │ │ │ │ + var me = this; │ │ │ │ │ + google.maps.event.addListenerOnce(this.mapObject, "tilesloaded", function() { │ │ │ │ │ + cache.rendered = true; │ │ │ │ │ + me.setGMapVisibility(me.getVisibility()); │ │ │ │ │ + me.moveTo(me.map.getCenter()) │ │ │ │ │ + }) │ │ │ │ │ } else { │ │ │ │ │ - response.features = result │ │ │ │ │ + map.div.appendChild(container); │ │ │ │ │ + cache.googleControl.appendChild(map.viewPortDiv); │ │ │ │ │ + google.maps.event.trigger(this.mapObject, "resize") │ │ │ │ │ } │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - response.error = result │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ + this.mapObject.setMapTypeId(type) │ │ │ │ │ + } else if (cache.googleControl.hasChildNodes()) { │ │ │ │ │ + map.div.appendChild(map.viewPortDiv); │ │ │ │ │ + map.div.removeChild(container) │ │ │ │ │ } │ │ │ │ │ - options.callback.call(options.scope, response) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - parseResponse: function(request, options) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null │ │ │ │ │ - } │ │ │ │ │ - var result = this.readFormat !== null ? this.readFormat.read(doc) : this.format.read(doc, options); │ │ │ │ │ - if (!this.featureNS) { │ │ │ │ │ - var format = this.readFormat || this.format; │ │ │ │ │ - this.featureNS = format.featureNS; │ │ │ │ │ - format.autoConfig = false; │ │ │ │ │ - if (!this.geometryName) { │ │ │ │ │ - this.setGeometryName(format.geometryName) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return result │ │ │ │ │ + getMapContainer: function() { │ │ │ │ │ + return this.mapObject.getDiv() │ │ │ │ │ }, │ │ │ │ │ - commit: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "commit", │ │ │ │ │ - reqFeatures: features │ │ │ │ │ - }); │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(features, options), │ │ │ │ │ - callback: this.createCallback(this.handleCommit, response, options) │ │ │ │ │ - }); │ │ │ │ │ - return response │ │ │ │ │ + getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ + var moBounds = null; │ │ │ │ │ + if (olBounds != null) { │ │ │ │ │ + var sw = this.sphericalMercator ? this.inverseMercator(olBounds.bottom, olBounds.left) : new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ + var ne = this.sphericalMercator ? this.inverseMercator(olBounds.top, olBounds.right) : new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ + moBounds = new google.maps.LatLngBounds(new google.maps.LatLng(sw.lat, sw.lon), new google.maps.LatLng(ne.lat, ne.lon)) │ │ │ │ │ + } │ │ │ │ │ + return moBounds │ │ │ │ │ }, │ │ │ │ │ - handleCommit: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - var data = request.responseXML; │ │ │ │ │ - if (!data || !data.documentElement) { │ │ │ │ │ - data = request.responseText │ │ │ │ │ - } │ │ │ │ │ - var obj = this.format.read(data) || {}; │ │ │ │ │ - response.insertIds = obj.insertIds || []; │ │ │ │ │ - if (obj.success) { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - response.error = obj │ │ │ │ │ - } │ │ │ │ │ - options.callback.call(options.scope, response) │ │ │ │ │ + getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + var size = this.map.getSize(); │ │ │ │ │ + var lon = this.getLongitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ + var lat = this.getLatitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ + var res = this.map.getResolution(); │ │ │ │ │ + var delta_x = moPixel.x - size.w / 2; │ │ │ │ │ + var delta_y = moPixel.y - size.h / 2; │ │ │ │ │ + var lonlat = new OpenLayers.LonLat(lon + delta_x * res, lat - delta_y * res); │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + lonlat = lonlat.wrapDateLine(this.maxExtent) │ │ │ │ │ } │ │ │ │ │ + return this.getMapObjectLonLatFromLonLat(lonlat.lon, lonlat.lat) │ │ │ │ │ }, │ │ │ │ │ - filterDelete: function(filter, options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "commit" │ │ │ │ │ - }); │ │ │ │ │ - var root = this.format.createElementNSPlus("wfs:Transaction", { │ │ │ │ │ - attributes: { │ │ │ │ │ - service: "WFS", │ │ │ │ │ - version: this.version │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - var deleteNode = this.format.createElementNSPlus("wfs:Delete", { │ │ │ │ │ - attributes: { │ │ │ │ │ - typeName: (options.featureNS ? this.featurePrefix + ":" : "") + options.featureType │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (options.featureNS) { │ │ │ │ │ - deleteNode.setAttribute("xmlns:" + this.featurePrefix, options.featureNS) │ │ │ │ │ + getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + var res = this.map.getResolution(); │ │ │ │ │ + var extent = this.map.getExtent(); │ │ │ │ │ + return this.getMapObjectPixelFromXY(1 / res * (lon - extent.left), 1 / res * (extent.top - lat)) │ │ │ │ │ + }, │ │ │ │ │ + setMapObjectCenter: function(center, zoom) { │ │ │ │ │ + if (this.animationEnabled === false && zoom != this.mapObject.zoom) { │ │ │ │ │ + var mapContainer = this.getMapContainer(); │ │ │ │ │ + google.maps.event.addListenerOnce(this.mapObject, "idle", function() { │ │ │ │ │ + mapContainer.style.visibility = "" │ │ │ │ │ + }); │ │ │ │ │ + mapContainer.style.visibility = "hidden" │ │ │ │ │ } │ │ │ │ │ - var filterNode = this.format.writeNode("ogc:Filter", filter); │ │ │ │ │ - deleteNode.appendChild(filterNode); │ │ │ │ │ - root.appendChild(deleteNode); │ │ │ │ │ - var data = OpenLayers.Format.XML.prototype.write.apply(this.format, [root]); │ │ │ │ │ - return OpenLayers.Request.POST({ │ │ │ │ │ - url: this.url, │ │ │ │ │ - callback: options.callback || function() {}, │ │ │ │ │ - data: data │ │ │ │ │ + this.mapObject.setOptions({ │ │ │ │ │ + center: center, │ │ │ │ │ + zoom: zoom │ │ │ │ │ }) │ │ │ │ │ }, │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - response.priv.abort() │ │ │ │ │ - } │ │ │ │ │ + getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + return this.mapObject.getBoundsZoomLevel(moBounds) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.WFS.v1" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Protocol.WFS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.WFS.v1_0_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Protocol.WFS.v1_1_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ - version: "1.1.0", │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.WFS.v1.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (this.outputFormat && !this.readFormat) { │ │ │ │ │ - if (this.outputFormat.toLowerCase() == "gml2") { │ │ │ │ │ - this.readFormat = new OpenLayers.Format.GML.v2({ │ │ │ │ │ - featureType: this.featureType, │ │ │ │ │ - featureNS: this.featureNS, │ │ │ │ │ - geometryName: this.geometryName │ │ │ │ │ - }) │ │ │ │ │ - } else if (this.outputFormat.toLowerCase() == "json") { │ │ │ │ │ - this.readFormat = new OpenLayers.Format.GeoJSON │ │ │ │ │ - } │ │ │ │ │ + getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ + var gLatLng; │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ + gLatLng = new google.maps.LatLng(lonlat.lat, lonlat.lon) │ │ │ │ │ + } else { │ │ │ │ │ + gLatLng = new google.maps.LatLng(lat, lon) │ │ │ │ │ } │ │ │ │ │ + return gLatLng │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.WFS.v1_1_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Protocol.CSW.v2_0_2 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ - formatOptions: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!options.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.CSWGetRecords.v2_0_2(OpenLayers.Util.extend({}, this.formatOptions)) │ │ │ │ │ + getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ + return new google.maps.Point(x, y) │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Popup.Anchored = OpenLayers.Class(OpenLayers.Popup, { │ │ │ │ │ + relativePosition: null, │ │ │ │ │ + keepInMap: true, │ │ │ │ │ + anchor: null, │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, closeBoxCallback) { │ │ │ │ │ + var newArguments = [id, lonlat, contentSize, contentHTML, closeBox, closeBoxCallback]; │ │ │ │ │ + OpenLayers.Popup.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + this.anchor = anchor != null ? anchor : { │ │ │ │ │ + size: new OpenLayers.Size(0, 0), │ │ │ │ │ + offset: new OpenLayers.Pixel(0, 0) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - if (this.options && !this.options.format) { │ │ │ │ │ - this.format.destroy() │ │ │ │ │ - } │ │ │ │ │ - this.format = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ + this.anchor = null; │ │ │ │ │ + this.relativePosition = null; │ │ │ │ │ + OpenLayers.Popup.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - read: function(options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - var data = this.format.write(options.params || options); │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: data │ │ │ │ │ - }); │ │ │ │ │ - return response │ │ │ │ │ + show: function() { │ │ │ │ │ + this.updatePosition(); │ │ │ │ │ + OpenLayers.Popup.prototype.show.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - response.data = this.parseData(request); │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ - } │ │ │ │ │ - options.callback.call(options.scope, response) │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + var oldRelativePosition = this.relativePosition; │ │ │ │ │ + this.relativePosition = this.calculateRelativePosition(px); │ │ │ │ │ + OpenLayers.Popup.prototype.moveTo.call(this, this.calculateNewPx(px)); │ │ │ │ │ + if (this.relativePosition != oldRelativePosition) { │ │ │ │ │ + this.updateRelativePosition() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - parseData: function(request) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null │ │ │ │ │ + setSize: function(contentSize) { │ │ │ │ │ + OpenLayers.Popup.prototype.setSize.apply(this, arguments); │ │ │ │ │ + if (this.lonlat && this.map) { │ │ │ │ │ + var px = this.map.getLayerPxFromLonLat(this.lonlat); │ │ │ │ │ + this.moveTo(px) │ │ │ │ │ } │ │ │ │ │ - return this.format.read(doc) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.CSW.v2_0_2" │ │ │ │ │ + calculateRelativePosition: function(px) { │ │ │ │ │ + var lonlat = this.map.getLonLatFromLayerPx(px); │ │ │ │ │ + var extent = this.map.getExtent(); │ │ │ │ │ + var quadrant = extent.determineQuadrant(lonlat); │ │ │ │ │ + return OpenLayers.Bounds.oppositeQuadrant(quadrant) │ │ │ │ │ + }, │ │ │ │ │ + updateRelativePosition: function() {}, │ │ │ │ │ + calculateNewPx: function(px) { │ │ │ │ │ + var newPx = px.offset(this.anchor.offset); │ │ │ │ │ + var size = this.size || this.contentSize; │ │ │ │ │ + var top = this.relativePosition.charAt(0) == "t"; │ │ │ │ │ + newPx.y += top ? -size.h : this.anchor.size.h; │ │ │ │ │ + var left = this.relativePosition.charAt(1) == "l"; │ │ │ │ │ + newPx.x += left ? -size.w : this.anchor.size.w; │ │ │ │ │ + return newPx │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup.Anchored" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Protocol.SOS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ - fois: null, │ │ │ │ │ - formatOptions: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!options.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.SOSGetFeatureOfInterest(this.formatOptions) │ │ │ │ │ +OpenLayers.Popup.Framed = OpenLayers.Class(OpenLayers.Popup.Anchored, { │ │ │ │ │ + imageSrc: null, │ │ │ │ │ + imageSize: null, │ │ │ │ │ + isAlphaImage: false, │ │ │ │ │ + positionBlocks: null, │ │ │ │ │ + blocks: null, │ │ │ │ │ + fixedRelativePosition: false, │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, closeBoxCallback) { │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (this.fixedRelativePosition) { │ │ │ │ │ + this.updateRelativePosition(); │ │ │ │ │ + this.calculateRelativePosition = function(px) { │ │ │ │ │ + return this.relativePosition │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.contentDiv.style.position = "absolute"; │ │ │ │ │ + this.contentDiv.style.zIndex = 1; │ │ │ │ │ + if (closeBox) { │ │ │ │ │ + this.closeDiv.style.zIndex = 1 │ │ │ │ │ } │ │ │ │ │ + this.groupDiv.style.position = "absolute"; │ │ │ │ │ + this.groupDiv.style.top = "0px"; │ │ │ │ │ + this.groupDiv.style.left = "0px"; │ │ │ │ │ + this.groupDiv.style.height = "100%"; │ │ │ │ │ + this.groupDiv.style.width = "100%" │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - if (this.options && !this.options.format) { │ │ │ │ │ - this.format.destroy() │ │ │ │ │ + this.imageSrc = null; │ │ │ │ │ + this.imageSize = null; │ │ │ │ │ + this.isAlphaImage = null; │ │ │ │ │ + this.fixedRelativePosition = false; │ │ │ │ │ + this.positionBlocks = null; │ │ │ │ │ + for (var i = 0; i < this.blocks.length; i++) { │ │ │ │ │ + var block = this.blocks[i]; │ │ │ │ │ + if (block.image) { │ │ │ │ │ + block.div.removeChild(block.image) │ │ │ │ │ + } │ │ │ │ │ + block.image = null; │ │ │ │ │ + if (block.div) { │ │ │ │ │ + this.groupDiv.removeChild(block.div) │ │ │ │ │ + } │ │ │ │ │ + block.div = null │ │ │ │ │ } │ │ │ │ │ - this.format = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ + this.blocks = null; │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - read: function(options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - var format = this.format; │ │ │ │ │ - var data = OpenLayers.Format.XML.prototype.write.apply(format, [format.writeNode("sos:GetFeatureOfInterest", { │ │ │ │ │ - fois: this.fois │ │ │ │ │ - })]); │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ - data: data │ │ │ │ │ - }); │ │ │ │ │ - return response │ │ │ │ │ + setBackgroundColor: function(color) {}, │ │ │ │ │ + setBorder: function() {}, │ │ │ │ │ + setOpacity: function(opacity) {}, │ │ │ │ │ + setSize: function(contentSize) { │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.setSize.apply(this, arguments); │ │ │ │ │ + this.updateBlocks() │ │ │ │ │ }, │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - response.features = this.parseFeatures(request); │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ - } │ │ │ │ │ - options.callback.call(options.scope, response) │ │ │ │ │ + updateRelativePosition: function() { │ │ │ │ │ + this.padding = this.positionBlocks[this.relativePosition].padding; │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ + this.closeDiv.style.right = contentDivPadding.right + this.padding.right + "px"; │ │ │ │ │ + this.closeDiv.style.top = contentDivPadding.top + this.padding.top + "px" │ │ │ │ │ } │ │ │ │ │ + this.updateBlocks() │ │ │ │ │ }, │ │ │ │ │ - parseFeatures: function(request) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ + calculateNewPx: function(px) { │ │ │ │ │ + var newPx = OpenLayers.Popup.Anchored.prototype.calculateNewPx.apply(this, arguments); │ │ │ │ │ + newPx = newPx.offset(this.positionBlocks[this.relativePosition].offset); │ │ │ │ │ + return newPx │ │ │ │ │ + }, │ │ │ │ │ + createBlocks: function() { │ │ │ │ │ + this.blocks = []; │ │ │ │ │ + var firstPosition = null; │ │ │ │ │ + for (var key in this.positionBlocks) { │ │ │ │ │ + firstPosition = key; │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null │ │ │ │ │ + var position = this.positionBlocks[firstPosition]; │ │ │ │ │ + for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ + var block = {}; │ │ │ │ │ + this.blocks.push(block); │ │ │ │ │ + var divId = this.id + "_FrameDecorationDiv_" + i; │ │ │ │ │ + block.div = OpenLayers.Util.createDiv(divId, null, null, null, "absolute", null, "hidden", null); │ │ │ │ │ + var imgId = this.id + "_FrameDecorationImg_" + i; │ │ │ │ │ + var imageCreator = this.isAlphaImage ? OpenLayers.Util.createAlphaImageDiv : OpenLayers.Util.createImage; │ │ │ │ │ + block.image = imageCreator(imgId, null, this.imageSize, this.imageSrc, "absolute", null, null, null); │ │ │ │ │ + block.div.appendChild(block.image); │ │ │ │ │ + this.groupDiv.appendChild(block.div) │ │ │ │ │ } │ │ │ │ │ - return this.format.read(doc) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.SOS.v1_0_0" │ │ │ │ │ + updateBlocks: function() { │ │ │ │ │ + if (!this.blocks) { │ │ │ │ │ + this.createBlocks() │ │ │ │ │ + } │ │ │ │ │ + if (this.size && this.relativePosition) { │ │ │ │ │ + var position = this.positionBlocks[this.relativePosition]; │ │ │ │ │ + for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ + var positionBlock = position.blocks[i]; │ │ │ │ │ + var block = this.blocks[i]; │ │ │ │ │ + var l = positionBlock.anchor.left; │ │ │ │ │ + var b = positionBlock.anchor.bottom; │ │ │ │ │ + var r = positionBlock.anchor.right; │ │ │ │ │ + var t = positionBlock.anchor.top; │ │ │ │ │ + var w = isNaN(positionBlock.size.w) ? this.size.w - (r + l) : positionBlock.size.w; │ │ │ │ │ + var h = isNaN(positionBlock.size.h) ? this.size.h - (b + t) : positionBlock.size.h; │ │ │ │ │ + block.div.style.width = (w < 0 ? 0 : w) + "px"; │ │ │ │ │ + block.div.style.height = (h < 0 ? 0 : h) + "px"; │ │ │ │ │ + block.div.style.left = l != null ? l + "px" : ""; │ │ │ │ │ + block.div.style.bottom = b != null ? b + "px" : ""; │ │ │ │ │ + block.div.style.right = r != null ? r + "px" : ""; │ │ │ │ │ + block.div.style.top = t != null ? t + "px" : ""; │ │ │ │ │ + block.image.style.left = positionBlock.position.x + "px"; │ │ │ │ │ + block.image.style.top = positionBlock.position.y + "px" │ │ │ │ │ + } │ │ │ │ │ + this.contentDiv.style.left = this.padding.left + "px"; │ │ │ │ │ + this.contentDiv.style.top = this.padding.top + "px" │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup.Framed" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Popup.FramedCloud = OpenLayers.Class(OpenLayers.Popup.Framed, { │ │ │ │ │ + contentDisplayClass: "olFramedCloudPopupContent", │ │ │ │ │ + autoSize: true, │ │ │ │ │ + panMapIfOutOfView: true, │ │ │ │ │ + imageSize: new OpenLayers.Size(1276, 736), │ │ │ │ │ + isAlphaImage: false, │ │ │ │ │ + fixedRelativePosition: false, │ │ │ │ │ + positionBlocks: { │ │ │ │ │ + tl: { │ │ │ │ │ + offset: new OpenLayers.Pixel(44, 0), │ │ │ │ │ + padding: new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ + blocks: [{ │ │ │ │ │ + size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size("auto", 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, 18), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -632) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(81, 35), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -688) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + tr: { │ │ │ │ │ + offset: new OpenLayers.Pixel(-45, 0), │ │ │ │ │ + padding: new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ + blocks: [{ │ │ │ │ │ + size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size("auto", 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -631) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(81, 35), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, null, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-215, -687) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + bl: { │ │ │ │ │ + offset: new OpenLayers.Pixel(45, 0), │ │ │ │ │ + padding: new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ + blocks: [{ │ │ │ │ │ + size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size("auto", 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(81, 33), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, null, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-101, -674) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + br: { │ │ │ │ │ + offset: new OpenLayers.Pixel(-44, 0), │ │ │ │ │ + padding: new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ + blocks: [{ │ │ │ │ │ + size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size("auto", 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(81, 33), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, null, null, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-311, -674) │ │ │ │ │ + }] │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + minSize: new OpenLayers.Size(105, 10), │ │ │ │ │ + maxSize: new OpenLayers.Size(1200, 660), │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, closeBoxCallback) { │ │ │ │ │ + this.imageSrc = OpenLayers.Util.getImageLocation("cloud-popup-relative.png"); │ │ │ │ │ + OpenLayers.Popup.Framed.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.contentDiv.className = this.contentDisplayClass │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup.FramedCloud" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Console.warn("OpenLayers.Rico is deprecated"); │ │ │ │ │ OpenLayers.Rico = OpenLayers.Rico || {}; │ │ │ │ │ OpenLayers.Rico.Color = OpenLayers.Class({ │ │ │ │ │ initialize: function(red, green, blue) { │ │ │ │ │ this.rgb = { │ │ │ │ │ r: red, │ │ │ ├── ./usr/share/javascript/openlayers/OpenLayers.mobile.js │ │ │ │ ├── js-beautify {} │ │ │ │ │ @@ -263,449 +263,14 @@ │ │ │ │ │ source.hasOwnProperty && source.hasOwnProperty("toString")) { │ │ │ │ │ destination.toString = source.toString; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ return destination; │ │ │ │ │ }; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Util/vendorPrefix.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/SingleFile.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Util = OpenLayers.Util || {}; │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Util.vendorPrefix │ │ │ │ │ - * A collection of utility functions to detect vendor prefixed features │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Util.vendorPrefix = (function() { │ │ │ │ │ - "use strict"; │ │ │ │ │ - │ │ │ │ │ - var VENDOR_PREFIXES = ["", "O", "ms", "Moz", "Webkit"], │ │ │ │ │ - divStyle = document.createElement("div").style, │ │ │ │ │ - cssCache = {}, │ │ │ │ │ - jsCache = {}; │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Function: domToCss │ │ │ │ │ - * Converts a upper camel case DOM style property name to a CSS property │ │ │ │ │ - * i.e. transformOrigin -> transform-origin │ │ │ │ │ - * or WebkitTransformOrigin -> -webkit-transform-origin │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * prefixedDom - {String} The property to convert │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The CSS property │ │ │ │ │ - */ │ │ │ │ │ - function domToCss(prefixedDom) { │ │ │ │ │ - if (!prefixedDom) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - return prefixedDom. │ │ │ │ │ - replace(/([A-Z])/g, function(c) { │ │ │ │ │ - return "-" + c.toLowerCase(); │ │ │ │ │ - }). │ │ │ │ │ - replace(/^ms-/, "-ms-"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: css │ │ │ │ │ - * Detect which property is used for a CSS property │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * property - {String} The standard (unprefixed) CSS property name │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The standard CSS property, prefixed property or null if not │ │ │ │ │ - * supported │ │ │ │ │ - */ │ │ │ │ │ - function css(property) { │ │ │ │ │ - if (cssCache[property] === undefined) { │ │ │ │ │ - var domProperty = property. │ │ │ │ │ - replace(/(-[\s\S])/g, function(c) { │ │ │ │ │ - return c.charAt(1).toUpperCase(); │ │ │ │ │ - }); │ │ │ │ │ - var prefixedDom = style(domProperty); │ │ │ │ │ - cssCache[property] = domToCss(prefixedDom); │ │ │ │ │ - } │ │ │ │ │ - return cssCache[property]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: js │ │ │ │ │ - * Detect which property is used for a JS property/method │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} The object to test on │ │ │ │ │ - * property - {String} The standard (unprefixed) JS property name │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The standard JS property, prefixed property or null if not │ │ │ │ │ - * supported │ │ │ │ │ - */ │ │ │ │ │ - function js(obj, property) { │ │ │ │ │ - if (jsCache[property] === undefined) { │ │ │ │ │ - var tmpProp, │ │ │ │ │ - i = 0, │ │ │ │ │ - l = VENDOR_PREFIXES.length, │ │ │ │ │ - prefix, │ │ │ │ │ - isStyleObj = (typeof obj.cssText !== "undefined"); │ │ │ │ │ - │ │ │ │ │ - jsCache[property] = null; │ │ │ │ │ - for (; i < l; i++) { │ │ │ │ │ - prefix = VENDOR_PREFIXES[i]; │ │ │ │ │ - if (prefix) { │ │ │ │ │ - if (!isStyleObj) { │ │ │ │ │ - // js prefix should be lower-case, while style │ │ │ │ │ - // properties have upper case on first character │ │ │ │ │ - prefix = prefix.toLowerCase(); │ │ │ │ │ - } │ │ │ │ │ - tmpProp = prefix + property.charAt(0).toUpperCase() + property.slice(1); │ │ │ │ │ - } else { │ │ │ │ │ - tmpProp = property; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (obj[tmpProp] !== undefined) { │ │ │ │ │ - jsCache[property] = tmpProp; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return jsCache[property]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: style │ │ │ │ │ - * Detect which property is used for a DOM style property │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * property - {String} The standard (unprefixed) style property name │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The standard style property, prefixed property or null if not │ │ │ │ │ - * supported │ │ │ │ │ - */ │ │ │ │ │ - function style(property) { │ │ │ │ │ - return js(divStyle, property); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return { │ │ │ │ │ - css: css, │ │ │ │ │ - js: js, │ │ │ │ │ - style: style, │ │ │ │ │ - │ │ │ │ │ - // used for testing │ │ │ │ │ - cssCache: cssCache, │ │ │ │ │ - jsCache: jsCache │ │ │ │ │ - }; │ │ │ │ │ -}()); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Animation.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/SingleFile.js │ │ │ │ │ - * @requires OpenLayers/Util/vendorPrefix.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Animation │ │ │ │ │ - * A collection of utility functions for executing methods that repaint a │ │ │ │ │ - * portion of the browser window. These methods take advantage of the │ │ │ │ │ - * browser's scheduled repaints where requestAnimationFrame is available. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Animation = (function(window) { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: isNative │ │ │ │ │ - * {Boolean} true if a native requestAnimationFrame function is available │ │ │ │ │ - */ │ │ │ │ │ - var requestAnimationFrame = OpenLayers.Util.vendorPrefix.js(window, "requestAnimationFrame"); │ │ │ │ │ - var isNative = !!(requestAnimationFrame); │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Function: requestFrame │ │ │ │ │ - * Schedule a function to be called at the next available animation frame. │ │ │ │ │ - * Uses the native method where available. Where requestAnimationFrame is │ │ │ │ │ - * not available, setTimeout will be called with a 16ms delay. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * callback - {Function} The function to be called at the next animation frame. │ │ │ │ │ - * element - {DOMElement} Optional element that visually bounds the animation. │ │ │ │ │ - */ │ │ │ │ │ - var requestFrame = (function() { │ │ │ │ │ - var request = window[requestAnimationFrame] || │ │ │ │ │ - function(callback, element) { │ │ │ │ │ - window.setTimeout(callback, 16); │ │ │ │ │ - }; │ │ │ │ │ - // bind to window to avoid illegal invocation of native function │ │ │ │ │ - return function(callback, element) { │ │ │ │ │ - request.apply(window, [callback, element]); │ │ │ │ │ - }; │ │ │ │ │ - })(); │ │ │ │ │ - │ │ │ │ │ - // private variables for animation loops │ │ │ │ │ - var counter = 0; │ │ │ │ │ - var loops = {}; │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Function: start │ │ │ │ │ - * Executes a method with <requestFrame> in series for some │ │ │ │ │ - * duration. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * callback - {Function} The function to be called at the next animation frame. │ │ │ │ │ - * duration - {Number} Optional duration for the loop. If not provided, the │ │ │ │ │ - * animation loop will execute indefinitely. │ │ │ │ │ - * element - {DOMElement} Optional element that visually bounds the animation. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number} Identifier for the animation loop. Used to stop animations with │ │ │ │ │ - * <stop>. │ │ │ │ │ - */ │ │ │ │ │ - function start(callback, duration, element) { │ │ │ │ │ - duration = duration > 0 ? duration : Number.POSITIVE_INFINITY; │ │ │ │ │ - var id = ++counter; │ │ │ │ │ - var start = +new Date; │ │ │ │ │ - loops[id] = function() { │ │ │ │ │ - if (loops[id] && +new Date - start <= duration) { │ │ │ │ │ - callback(); │ │ │ │ │ - if (loops[id]) { │ │ │ │ │ - requestFrame(loops[id], element); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - delete loops[id]; │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - requestFrame(loops[id], element); │ │ │ │ │ - return id; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Function: stop │ │ │ │ │ - * Terminates an animation loop started with <start>. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {Number} Identifier returned from <start>. │ │ │ │ │ - */ │ │ │ │ │ - function stop(id) { │ │ │ │ │ - delete loops[id]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return { │ │ │ │ │ - isNative: isNative, │ │ │ │ │ - requestFrame: requestFrame, │ │ │ │ │ - start: start, │ │ │ │ │ - stop: stop │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ -})(window); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Kinetic.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Animation.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Kinetic = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: threshold │ │ │ │ │ - * In most cases changing the threshold isn't needed. │ │ │ │ │ - * In px/ms, default to 0. │ │ │ │ │ - */ │ │ │ │ │ - threshold: 0, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: deceleration │ │ │ │ │ - * {Float} the deseleration in px/ms², default to 0.0035. │ │ │ │ │ - */ │ │ │ │ │ - deceleration: 0.0035, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: nbPoints │ │ │ │ │ - * {Integer} the number of points we use to calculate the kinetic │ │ │ │ │ - * initial values. │ │ │ │ │ - */ │ │ │ │ │ - nbPoints: 100, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: delay │ │ │ │ │ - * {Float} time to consider to calculate the kinetic initial values. │ │ │ │ │ - * In ms, default to 200. │ │ │ │ │ - */ │ │ │ │ │ - delay: 200, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: points │ │ │ │ │ - * List of points use to calculate the kinetic initial values. │ │ │ │ │ - */ │ │ │ │ │ - points: undefined, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: timerId │ │ │ │ │ - * ID of the timer. │ │ │ │ │ - */ │ │ │ │ │ - timerId: undefined, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Kinetic │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: begin │ │ │ │ │ - * Begins the dragging. │ │ │ │ │ - */ │ │ │ │ │ - begin: function() { │ │ │ │ │ - OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ - this.timerId = undefined; │ │ │ │ │ - this.points = []; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: update │ │ │ │ │ - * Updates during the dragging. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} The new position. │ │ │ │ │ - */ │ │ │ │ │ - update: function(xy) { │ │ │ │ │ - this.points.unshift({ │ │ │ │ │ - xy: xy, │ │ │ │ │ - tick: new Date().getTime() │ │ │ │ │ - }); │ │ │ │ │ - if (this.points.length > this.nbPoints) { │ │ │ │ │ - this.points.pop(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: end │ │ │ │ │ - * Ends the dragging, start the kinetic. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} The last position. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object with two properties: "speed", and "theta". The │ │ │ │ │ - * "speed" and "theta" values are to be passed to the move │ │ │ │ │ - * function when starting the animation. │ │ │ │ │ - */ │ │ │ │ │ - end: function(xy) { │ │ │ │ │ - var last, now = new Date().getTime(); │ │ │ │ │ - for (var i = 0, l = this.points.length, point; i < l; i++) { │ │ │ │ │ - point = this.points[i]; │ │ │ │ │ - if (now - point.tick > this.delay) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - last = point; │ │ │ │ │ - } │ │ │ │ │ - if (!last) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var time = new Date().getTime() - last.tick; │ │ │ │ │ - var dist = Math.sqrt(Math.pow(xy.x - last.xy.x, 2) + │ │ │ │ │ - Math.pow(xy.y - last.xy.y, 2)); │ │ │ │ │ - var speed = dist / time; │ │ │ │ │ - if (speed == 0 || speed < this.threshold) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var theta = Math.asin((xy.y - last.xy.y) / dist); │ │ │ │ │ - if (last.xy.x <= xy.x) { │ │ │ │ │ - theta = Math.PI - theta; │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - speed: speed, │ │ │ │ │ - theta: theta │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * Launch the kinetic move pan. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * info - {Object} An object with two properties, "speed", and "theta". │ │ │ │ │ - * These values are those returned from the "end" call. │ │ │ │ │ - * callback - {Function} Function called on every step of the animation, │ │ │ │ │ - * receives x, y (values to pan), end (is the last point). │ │ │ │ │ - */ │ │ │ │ │ - move: function(info, callback) { │ │ │ │ │ - var v0 = info.speed; │ │ │ │ │ - var fx = Math.cos(info.theta); │ │ │ │ │ - var fy = -Math.sin(info.theta); │ │ │ │ │ - │ │ │ │ │ - var initialTime = new Date().getTime(); │ │ │ │ │ - │ │ │ │ │ - var lastX = 0; │ │ │ │ │ - var lastY = 0; │ │ │ │ │ - │ │ │ │ │ - var timerCallback = function() { │ │ │ │ │ - if (this.timerId == null) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var t = new Date().getTime() - initialTime; │ │ │ │ │ - │ │ │ │ │ - var p = (-this.deceleration * Math.pow(t, 2)) / 2.0 + v0 * t; │ │ │ │ │ - var x = p * fx; │ │ │ │ │ - var y = p * fy; │ │ │ │ │ - │ │ │ │ │ - var args = {}; │ │ │ │ │ - args.end = false; │ │ │ │ │ - var v = -this.deceleration * t + v0; │ │ │ │ │ - │ │ │ │ │ - if (v <= 0) { │ │ │ │ │ - OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - args.end = true; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - args.x = x - lastX; │ │ │ │ │ - args.y = y - lastY; │ │ │ │ │ - lastX = x; │ │ │ │ │ - lastY = y; │ │ │ │ │ - callback(args.x, args.y, args.end); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - this.timerId = OpenLayers.Animation.start( │ │ │ │ │ - OpenLayers.Function.bind(timerCallback, this) │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Kinetic" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ OpenLayers/BaseTypes.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -4861,14 +4426,155 @@ │ │ │ │ │ } else { │ │ │ │ │ str += coordinate < 0 ? OpenLayers.i18n("S") : OpenLayers.i18n("N"); │ │ │ │ │ } │ │ │ │ │ return str; │ │ │ │ │ }; │ │ │ │ │ │ │ │ │ │ /* ====================================================================== │ │ │ │ │ + OpenLayers/Util/vendorPrefix.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/SingleFile.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Util = OpenLayers.Util || {}; │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Util.vendorPrefix │ │ │ │ │ + * A collection of utility functions to detect vendor prefixed features │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Util.vendorPrefix = (function() { │ │ │ │ │ + "use strict"; │ │ │ │ │ + │ │ │ │ │ + var VENDOR_PREFIXES = ["", "O", "ms", "Moz", "Webkit"], │ │ │ │ │ + divStyle = document.createElement("div").style, │ │ │ │ │ + cssCache = {}, │ │ │ │ │ + jsCache = {}; │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Function: domToCss │ │ │ │ │ + * Converts a upper camel case DOM style property name to a CSS property │ │ │ │ │ + * i.e. transformOrigin -> transform-origin │ │ │ │ │ + * or WebkitTransformOrigin -> -webkit-transform-origin │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * prefixedDom - {String} The property to convert │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The CSS property │ │ │ │ │ + */ │ │ │ │ │ + function domToCss(prefixedDom) { │ │ │ │ │ + if (!prefixedDom) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + return prefixedDom. │ │ │ │ │ + replace(/([A-Z])/g, function(c) { │ │ │ │ │ + return "-" + c.toLowerCase(); │ │ │ │ │ + }). │ │ │ │ │ + replace(/^ms-/, "-ms-"); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: css │ │ │ │ │ + * Detect which property is used for a CSS property │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * property - {String} The standard (unprefixed) CSS property name │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The standard CSS property, prefixed property or null if not │ │ │ │ │ + * supported │ │ │ │ │ + */ │ │ │ │ │ + function css(property) { │ │ │ │ │ + if (cssCache[property] === undefined) { │ │ │ │ │ + var domProperty = property. │ │ │ │ │ + replace(/(-[\s\S])/g, function(c) { │ │ │ │ │ + return c.charAt(1).toUpperCase(); │ │ │ │ │ + }); │ │ │ │ │ + var prefixedDom = style(domProperty); │ │ │ │ │ + cssCache[property] = domToCss(prefixedDom); │ │ │ │ │ + } │ │ │ │ │ + return cssCache[property]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: js │ │ │ │ │ + * Detect which property is used for a JS property/method │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} The object to test on │ │ │ │ │ + * property - {String} The standard (unprefixed) JS property name │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The standard JS property, prefixed property or null if not │ │ │ │ │ + * supported │ │ │ │ │ + */ │ │ │ │ │ + function js(obj, property) { │ │ │ │ │ + if (jsCache[property] === undefined) { │ │ │ │ │ + var tmpProp, │ │ │ │ │ + i = 0, │ │ │ │ │ + l = VENDOR_PREFIXES.length, │ │ │ │ │ + prefix, │ │ │ │ │ + isStyleObj = (typeof obj.cssText !== "undefined"); │ │ │ │ │ + │ │ │ │ │ + jsCache[property] = null; │ │ │ │ │ + for (; i < l; i++) { │ │ │ │ │ + prefix = VENDOR_PREFIXES[i]; │ │ │ │ │ + if (prefix) { │ │ │ │ │ + if (!isStyleObj) { │ │ │ │ │ + // js prefix should be lower-case, while style │ │ │ │ │ + // properties have upper case on first character │ │ │ │ │ + prefix = prefix.toLowerCase(); │ │ │ │ │ + } │ │ │ │ │ + tmpProp = prefix + property.charAt(0).toUpperCase() + property.slice(1); │ │ │ │ │ + } else { │ │ │ │ │ + tmpProp = property; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (obj[tmpProp] !== undefined) { │ │ │ │ │ + jsCache[property] = tmpProp; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return jsCache[property]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: style │ │ │ │ │ + * Detect which property is used for a DOM style property │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * property - {String} The standard (unprefixed) style property name │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The standard style property, prefixed property or null if not │ │ │ │ │ + * supported │ │ │ │ │ + */ │ │ │ │ │ + function style(property) { │ │ │ │ │ + return js(divStyle, property); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return { │ │ │ │ │ + css: css, │ │ │ │ │ + js: js, │ │ │ │ │ + style: style, │ │ │ │ │ + │ │ │ │ │ + // used for testing │ │ │ │ │ + cssCache: cssCache, │ │ │ │ │ + jsCache: jsCache │ │ │ │ │ + }; │ │ │ │ │ +}()); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ OpenLayers/Events.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -6037,14 +5743,120 @@ │ │ │ │ │ │ │ │ │ │ OpenLayers.Event.observe(element, 'MSPointerUp', cb); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Events" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ + OpenLayers/Animation.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/SingleFile.js │ │ │ │ │ + * @requires OpenLayers/Util/vendorPrefix.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Animation │ │ │ │ │ + * A collection of utility functions for executing methods that repaint a │ │ │ │ │ + * portion of the browser window. These methods take advantage of the │ │ │ │ │ + * browser's scheduled repaints where requestAnimationFrame is available. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Animation = (function(window) { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: isNative │ │ │ │ │ + * {Boolean} true if a native requestAnimationFrame function is available │ │ │ │ │ + */ │ │ │ │ │ + var requestAnimationFrame = OpenLayers.Util.vendorPrefix.js(window, "requestAnimationFrame"); │ │ │ │ │ + var isNative = !!(requestAnimationFrame); │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Function: requestFrame │ │ │ │ │ + * Schedule a function to be called at the next available animation frame. │ │ │ │ │ + * Uses the native method where available. Where requestAnimationFrame is │ │ │ │ │ + * not available, setTimeout will be called with a 16ms delay. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * callback - {Function} The function to be called at the next animation frame. │ │ │ │ │ + * element - {DOMElement} Optional element that visually bounds the animation. │ │ │ │ │ + */ │ │ │ │ │ + var requestFrame = (function() { │ │ │ │ │ + var request = window[requestAnimationFrame] || │ │ │ │ │ + function(callback, element) { │ │ │ │ │ + window.setTimeout(callback, 16); │ │ │ │ │ + }; │ │ │ │ │ + // bind to window to avoid illegal invocation of native function │ │ │ │ │ + return function(callback, element) { │ │ │ │ │ + request.apply(window, [callback, element]); │ │ │ │ │ + }; │ │ │ │ │ + })(); │ │ │ │ │ + │ │ │ │ │ + // private variables for animation loops │ │ │ │ │ + var counter = 0; │ │ │ │ │ + var loops = {}; │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Function: start │ │ │ │ │ + * Executes a method with <requestFrame> in series for some │ │ │ │ │ + * duration. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * callback - {Function} The function to be called at the next animation frame. │ │ │ │ │ + * duration - {Number} Optional duration for the loop. If not provided, the │ │ │ │ │ + * animation loop will execute indefinitely. │ │ │ │ │ + * element - {DOMElement} Optional element that visually bounds the animation. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number} Identifier for the animation loop. Used to stop animations with │ │ │ │ │ + * <stop>. │ │ │ │ │ + */ │ │ │ │ │ + function start(callback, duration, element) { │ │ │ │ │ + duration = duration > 0 ? duration : Number.POSITIVE_INFINITY; │ │ │ │ │ + var id = ++counter; │ │ │ │ │ + var start = +new Date; │ │ │ │ │ + loops[id] = function() { │ │ │ │ │ + if (loops[id] && +new Date - start <= duration) { │ │ │ │ │ + callback(); │ │ │ │ │ + if (loops[id]) { │ │ │ │ │ + requestFrame(loops[id], element); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + delete loops[id]; │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + requestFrame(loops[id], element); │ │ │ │ │ + return id; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Function: stop │ │ │ │ │ + * Terminates an animation loop started with <start>. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {Number} Identifier returned from <start>. │ │ │ │ │ + */ │ │ │ │ │ + function stop(id) { │ │ │ │ │ + delete loops[id]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return { │ │ │ │ │ + isNative: isNative, │ │ │ │ │ + requestFrame: requestFrame, │ │ │ │ │ + start: start, │ │ │ │ │ + stop: stop │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ +})(window); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ OpenLayers/Tween.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -9645,14 +9457,202 @@ │ │ │ │ │ OpenLayers.Map.TILE_WIDTH = 256; │ │ │ │ │ /** │ │ │ │ │ * Constant: TILE_HEIGHT │ │ │ │ │ * {Integer} 256 Default tile height (unless otherwise specified) │ │ │ │ │ */ │ │ │ │ │ OpenLayers.Map.TILE_HEIGHT = 256; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ + OpenLayers/Kinetic.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Animation.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Kinetic = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: threshold │ │ │ │ │ + * In most cases changing the threshold isn't needed. │ │ │ │ │ + * In px/ms, default to 0. │ │ │ │ │ + */ │ │ │ │ │ + threshold: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: deceleration │ │ │ │ │ + * {Float} the deseleration in px/ms², default to 0.0035. │ │ │ │ │ + */ │ │ │ │ │ + deceleration: 0.0035, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: nbPoints │ │ │ │ │ + * {Integer} the number of points we use to calculate the kinetic │ │ │ │ │ + * initial values. │ │ │ │ │ + */ │ │ │ │ │ + nbPoints: 100, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: delay │ │ │ │ │ + * {Float} time to consider to calculate the kinetic initial values. │ │ │ │ │ + * In ms, default to 200. │ │ │ │ │ + */ │ │ │ │ │ + delay: 200, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: points │ │ │ │ │ + * List of points use to calculate the kinetic initial values. │ │ │ │ │ + */ │ │ │ │ │ + points: undefined, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: timerId │ │ │ │ │ + * ID of the timer. │ │ │ │ │ + */ │ │ │ │ │ + timerId: undefined, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Kinetic │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: begin │ │ │ │ │ + * Begins the dragging. │ │ │ │ │ + */ │ │ │ │ │ + begin: function() { │ │ │ │ │ + OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ + this.timerId = undefined; │ │ │ │ │ + this.points = []; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: update │ │ │ │ │ + * Updates during the dragging. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} The new position. │ │ │ │ │ + */ │ │ │ │ │ + update: function(xy) { │ │ │ │ │ + this.points.unshift({ │ │ │ │ │ + xy: xy, │ │ │ │ │ + tick: new Date().getTime() │ │ │ │ │ + }); │ │ │ │ │ + if (this.points.length > this.nbPoints) { │ │ │ │ │ + this.points.pop(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: end │ │ │ │ │ + * Ends the dragging, start the kinetic. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} The last position. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object with two properties: "speed", and "theta". The │ │ │ │ │ + * "speed" and "theta" values are to be passed to the move │ │ │ │ │ + * function when starting the animation. │ │ │ │ │ + */ │ │ │ │ │ + end: function(xy) { │ │ │ │ │ + var last, now = new Date().getTime(); │ │ │ │ │ + for (var i = 0, l = this.points.length, point; i < l; i++) { │ │ │ │ │ + point = this.points[i]; │ │ │ │ │ + if (now - point.tick > this.delay) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + last = point; │ │ │ │ │ + } │ │ │ │ │ + if (!last) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var time = new Date().getTime() - last.tick; │ │ │ │ │ + var dist = Math.sqrt(Math.pow(xy.x - last.xy.x, 2) + │ │ │ │ │ + Math.pow(xy.y - last.xy.y, 2)); │ │ │ │ │ + var speed = dist / time; │ │ │ │ │ + if (speed == 0 || speed < this.threshold) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var theta = Math.asin((xy.y - last.xy.y) / dist); │ │ │ │ │ + if (last.xy.x <= xy.x) { │ │ │ │ │ + theta = Math.PI - theta; │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + speed: speed, │ │ │ │ │ + theta: theta │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: move │ │ │ │ │ + * Launch the kinetic move pan. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * info - {Object} An object with two properties, "speed", and "theta". │ │ │ │ │ + * These values are those returned from the "end" call. │ │ │ │ │ + * callback - {Function} Function called on every step of the animation, │ │ │ │ │ + * receives x, y (values to pan), end (is the last point). │ │ │ │ │ + */ │ │ │ │ │ + move: function(info, callback) { │ │ │ │ │ + var v0 = info.speed; │ │ │ │ │ + var fx = Math.cos(info.theta); │ │ │ │ │ + var fy = -Math.sin(info.theta); │ │ │ │ │ + │ │ │ │ │ + var initialTime = new Date().getTime(); │ │ │ │ │ + │ │ │ │ │ + var lastX = 0; │ │ │ │ │ + var lastY = 0; │ │ │ │ │ + │ │ │ │ │ + var timerCallback = function() { │ │ │ │ │ + if (this.timerId == null) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var t = new Date().getTime() - initialTime; │ │ │ │ │ + │ │ │ │ │ + var p = (-this.deceleration * Math.pow(t, 2)) / 2.0 + v0 * t; │ │ │ │ │ + var x = p * fx; │ │ │ │ │ + var y = p * fy; │ │ │ │ │ + │ │ │ │ │ + var args = {}; │ │ │ │ │ + args.end = false; │ │ │ │ │ + var v = -this.deceleration * t + v0; │ │ │ │ │ + │ │ │ │ │ + if (v <= 0) { │ │ │ │ │ + OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + args.end = true; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + args.x = x - lastX; │ │ │ │ │ + args.y = y - lastY; │ │ │ │ │ + lastX = x; │ │ │ │ │ + lastY = y; │ │ │ │ │ + callback(args.x, args.y, args.end); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + this.timerId = OpenLayers.Animation.start( │ │ │ │ │ + OpenLayers.Function.bind(timerCallback, this) │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Kinetic" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ OpenLayers/Layer.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -13915,5390 +13915,2007 @@ │ │ │ │ │ this.tileCache = null; │ │ │ │ │ this.tileCacheIndex = null; │ │ │ │ │ this._destroyed = true; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/XYZ.js │ │ │ │ │ + OpenLayers/Handler.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.XYZ │ │ │ │ │ - * The XYZ class is designed to make it easier for people who have tiles │ │ │ │ │ - * arranged by a standard XYZ grid. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Handler │ │ │ │ │ + * Base class to construct a higher-level handler for event sequences. All │ │ │ │ │ + * handlers have activate and deactivate methods. In addition, they have │ │ │ │ │ + * methods named like browser events. When a handler is activated, any │ │ │ │ │ + * additional methods named like a browser event is registered as a │ │ │ │ │ + * listener for the corresponding event. When a handler is deactivated, │ │ │ │ │ + * those same methods are unregistered as event listeners. │ │ │ │ │ + * │ │ │ │ │ + * Handlers also typically have a callbacks object with keys named like │ │ │ │ │ + * the abstracted events or event sequences that they are in charge of │ │ │ │ │ + * handling. The controls that wrap handlers define the methods that │ │ │ │ │ + * correspond to these abstract events - so instead of listening for │ │ │ │ │ + * individual browser events, they only listen for the abstract events │ │ │ │ │ + * defined by the handler. │ │ │ │ │ + * │ │ │ │ │ + * Handlers are created by controls, which ultimately have the responsibility │ │ │ │ │ + * of making changes to the the state of the application. Handlers │ │ │ │ │ + * themselves may make temporary changes, but in general are expected to │ │ │ │ │ + * return the application in the same state that they found it. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.XYZ = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ +OpenLayers.Handler = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * Default is true, as this is designed to be a base tile source. │ │ │ │ │ + * Property: id │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + id: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: sphericalMercator │ │ │ │ │ - * Whether the tile extents should be set to the defaults for │ │ │ │ │ - * spherical mercator. Useful for things like OpenStreetMap. │ │ │ │ │ - * Default is false, except for the OSM subclass. │ │ │ │ │ + * APIProperty: control │ │ │ │ │ + * {<OpenLayers.Control>}. The control that initialized this handler. The │ │ │ │ │ + * control is assumed to have a valid map property - that map is used │ │ │ │ │ + * in the handler's own setMap method. │ │ │ │ │ */ │ │ │ │ │ - sphericalMercator: false, │ │ │ │ │ + control: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomOffset │ │ │ │ │ - * {Number} If your cache has more zoom levels than you want to provide │ │ │ │ │ - * access to with this layer, supply a zoomOffset. This zoom offset │ │ │ │ │ - * is added to the current map zoom level to determine the level │ │ │ │ │ - * for a requested tile. For example, if you supply a zoomOffset │ │ │ │ │ - * of 3, when the map is at the zoom 0, tiles will be requested from │ │ │ │ │ - * level 3 of your cache. Default is 0 (assumes cache level and map │ │ │ │ │ - * zoom are equivalent). Using <zoomOffset> is an alternative to │ │ │ │ │ - * setting <serverResolutions> if you only want to expose a subset │ │ │ │ │ - * of the server resolutions. │ │ │ │ │ + * Property: map │ │ │ │ │ + * {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - zoomOffset: 0, │ │ │ │ │ + map: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: serverResolutions │ │ │ │ │ - * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ - * property if the map resolutions differ from the server. This │ │ │ │ │ - * property serves two purposes. (a) <serverResolutions> can include │ │ │ │ │ - * resolutions that the server supports and that you don't want to │ │ │ │ │ - * provide with this layer; you can also look at <zoomOffset>, which is │ │ │ │ │ - * an alternative to <serverResolutions> for that specific purpose. │ │ │ │ │ - * (b) The map can work with resolutions that aren't supported by │ │ │ │ │ - * the server, i.e. that aren't in <serverResolutions>. When the │ │ │ │ │ - * map is displayed in such a resolution data for the closest │ │ │ │ │ - * server-supported resolution is loaded and the layer div is │ │ │ │ │ - * stretched as necessary. │ │ │ │ │ + * APIProperty: keyMask │ │ │ │ │ + * {Integer} Use bitwise operators and one or more of the OpenLayers.Handler │ │ │ │ │ + * constants to construct a keyMask. The keyMask is used by │ │ │ │ │ + * <checkModifiers>. If the keyMask matches the combination of keys │ │ │ │ │ + * down on an event, checkModifiers returns true. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * // handler only responds if the Shift key is down │ │ │ │ │ + * handler.keyMask = OpenLayers.Handler.MOD_SHIFT; │ │ │ │ │ + * │ │ │ │ │ + * // handler only responds if Ctrl-Shift is down │ │ │ │ │ + * handler.keyMask = OpenLayers.Handler.MOD_SHIFT | │ │ │ │ │ + * OpenLayers.Handler.MOD_CTRL; │ │ │ │ │ + * (end) │ │ │ │ │ */ │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ + keyMask: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.XYZ │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * url - {String} │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + * Property: active │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - if (options && options.sphericalMercator || this.sphericalMercator) { │ │ │ │ │ - options = OpenLayers.Util.extend({ │ │ │ │ │ - projection: "EPSG:900913", │ │ │ │ │ - numZoomLevels: 19 │ │ │ │ │ - }, options); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [ │ │ │ │ │ - name || this.name, url || this.url, {}, │ │ │ │ │ - options │ │ │ │ │ - ]); │ │ │ │ │ - }, │ │ │ │ │ + active: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ + * Property: evt │ │ │ │ │ + * {Event} This property references the last event handled by the handler. │ │ │ │ │ + * Note that this property is not part of the stable API. Use of the │ │ │ │ │ + * evt property should be restricted to controls in the library │ │ │ │ │ + * or other applications that are willing to update with changes to │ │ │ │ │ + * the OpenLayers code. │ │ │ │ │ + */ │ │ │ │ │ + evt: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: touch │ │ │ │ │ + * {Boolean} Indicates the support of touch events. When touch events are │ │ │ │ │ + * started touch will be true and all mouse related listeners will do │ │ │ │ │ + * nothing. │ │ │ │ │ + */ │ │ │ │ │ + touch: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Handler │ │ │ │ │ + * Construct a handler. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {Object} Is this ever used? │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.XYZ>} An exact clone of this OpenLayers.Layer.XYZ │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that initialized this │ │ │ │ │ + * handler. The control is assumed to have a valid map property; that │ │ │ │ │ + * map is used in the handler's own setMap method. If a map property │ │ │ │ │ + * is present in the options argument it will be used instead. │ │ │ │ │ + * callbacks - {Object} An object whose properties correspond to abstracted │ │ │ │ │ + * events or sequences of browser events. The values for these │ │ │ │ │ + * properties are functions defined by the control that get called by │ │ │ │ │ + * the handler. │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * the handler. │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.control = control; │ │ │ │ │ + this.callbacks = callbacks; │ │ │ │ │ │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.XYZ(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ + var map = this.map || control.map; │ │ │ │ │ + if (map) { │ │ │ │ │ + this.setMap(map); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as │ │ │ │ │ - * parameters │ │ │ │ │ + * Method: setMap │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var xyz = this.getXYZ(bounds); │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - var s = '' + xyz.x + xyz.y + xyz.z; │ │ │ │ │ - url = this.selectUrl(s, url); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return OpenLayers.String.format(url, xyz); │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.map = map; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getXYZ │ │ │ │ │ - * Calculates x, y and z for the given bounds. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * Method: checkModifiers │ │ │ │ │ + * Check the keyMask on the handler. If no <keyMask> is set, this always │ │ │ │ │ + * returns true. If a <keyMask> is set and it matches the combination │ │ │ │ │ + * of keys down on an event, this returns true. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} - an object with x, y and z properties. │ │ │ │ │ + * {Boolean} The keyMask matches the keys down on an event. │ │ │ │ │ */ │ │ │ │ │ - getXYZ: function(bounds) { │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.maxExtent.left) / │ │ │ │ │ - (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((this.maxExtent.top - bounds.top) / │ │ │ │ │ - (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getServerZoom(); │ │ │ │ │ - │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - var limit = Math.pow(2, z); │ │ │ │ │ - x = ((x % limit) + limit) % limit; │ │ │ │ │ + checkModifiers: function(evt) { │ │ │ │ │ + if (this.keyMask == null) { │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ + /* calculate the keyboard modifier mask for this event */ │ │ │ │ │ + var keyModifiers = │ │ │ │ │ + (evt.shiftKey ? OpenLayers.Handler.MOD_SHIFT : 0) | │ │ │ │ │ + (evt.ctrlKey ? OpenLayers.Handler.MOD_CTRL : 0) | │ │ │ │ │ + (evt.altKey ? OpenLayers.Handler.MOD_ALT : 0) | │ │ │ │ │ + (evt.metaKey ? OpenLayers.Handler.MOD_META : 0); │ │ │ │ │ │ │ │ │ │ - return { │ │ │ │ │ - 'x': x, │ │ │ │ │ - 'y': y, │ │ │ │ │ - 'z': z │ │ │ │ │ - }; │ │ │ │ │ + /* if it differs from the handler object's key mask, │ │ │ │ │ + bail out of the event handler */ │ │ │ │ │ + return (keyModifiers == this.keyMask); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /* APIMethod: setMap │ │ │ │ │ - * When the layer is added to a map, then we can fetch our origin │ │ │ │ │ - * (if we don't have one.) │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Turn on the handler. Returns false if the handler was already active. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was activated. │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (!this.tileOrigin) { │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, │ │ │ │ │ - this.maxExtent.bottom); │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + // register for event handlers defined on this class. │ │ │ │ │ + var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.register(events[i], this[events[i]]); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + this.active = true; │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.XYZ" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/OSM.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.OSM │ │ │ │ │ - * This layer allows accessing OpenStreetMap tiles. By default the OpenStreetMap │ │ │ │ │ - * hosted tile.openstreetmap.org Mapnik tileset is used. If you wish to use │ │ │ │ │ - * a different layer instead, you need to provide a different │ │ │ │ │ - * URL to the constructor. Here's an example for using OpenCycleMap: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * new OpenLayers.Layer.OSM("OpenCycleMap", │ │ │ │ │ - * ["http://a.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ - * "http://b.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ - * "http://c.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png"]); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.XYZ> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.OSM = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: name │ │ │ │ │ - * {String} The layer name. Defaults to "OpenStreetMap" if the first │ │ │ │ │ - * argument to the constructor is null or undefined. │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Turn off the handler. Returns false if the handler was already inactive. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was deactivated. │ │ │ │ │ */ │ │ │ │ │ - name: "OpenStreetMap", │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + // unregister event handlers defined on this class. │ │ │ │ │ + var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.unregister(events[i], this[events[i]]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.touch = false; │ │ │ │ │ + this.active = false; │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String} The tileset URL scheme. Defaults to │ │ │ │ │ - * : http://[a|b|c].tile.openstreetmap.org/${z}/${x}/${y}.png │ │ │ │ │ - * (the official OSM tileset) if the second argument to the constructor │ │ │ │ │ - * is null or undefined. To use another tileset you can have something │ │ │ │ │ - * like this: │ │ │ │ │ - * (code) │ │ │ │ │ - * new OpenLayers.Layer.OSM("OpenCycleMap", │ │ │ │ │ - * ["http://a.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ - * "http://b.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ - * "http://c.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png"]); │ │ │ │ │ - * (end) │ │ │ │ │ + * Method: startTouch │ │ │ │ │ + * Start touch events, this method must be called by subclasses in │ │ │ │ │ + * "touchstart" method. When touch events are started <touch> will be │ │ │ │ │ + * true and all mouse related listeners will do nothing. │ │ │ │ │ */ │ │ │ │ │ - url: [ │ │ │ │ │ - 'http://a.tile.openstreetmap.org/${z}/${x}/${y}.png', │ │ │ │ │ - 'http://b.tile.openstreetmap.org/${z}/${x}/${y}.png', │ │ │ │ │ - 'http://c.tile.openstreetmap.org/${z}/${x}/${y}.png' │ │ │ │ │ - ], │ │ │ │ │ + startTouch: function() { │ │ │ │ │ + if (!this.touch) { │ │ │ │ │ + this.touch = true; │ │ │ │ │ + var events = [ │ │ │ │ │ + "mousedown", "mouseup", "mousemove", "click", "dblclick", │ │ │ │ │ + "mouseout" │ │ │ │ │ + ]; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.unregister(events[i], this[events[i]]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: attribution │ │ │ │ │ - * {String} The layer attribution. │ │ │ │ │ + * Method: callback │ │ │ │ │ + * Trigger the control's named callback with the given arguments │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} The key for the callback that is one of the properties │ │ │ │ │ + * of the handler's callbacks object. │ │ │ │ │ + * args - {Array(*)} An array of arguments (any type) with which to call │ │ │ │ │ + * the callback (defined by the control). │ │ │ │ │ */ │ │ │ │ │ - attribution: "© <a href='http://www.openstreetmap.org/copyright'>OpenStreetMap</a> contributors", │ │ │ │ │ + callback: function(name, args) { │ │ │ │ │ + if (name && this.callbacks[name]) { │ │ │ │ │ + this.callbacks[name].apply(this.control, args); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: sphericalMercator │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * Method: register │ │ │ │ │ + * register an event on the map │ │ │ │ │ */ │ │ │ │ │ - sphericalMercator: true, │ │ │ │ │ + register: function(name, method) { │ │ │ │ │ + // TODO: deal with registerPriority in 3.0 │ │ │ │ │ + this.map.events.registerPriority(name, this, method); │ │ │ │ │ + this.map.events.registerPriority(name, this, this.setEvent); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: wrapDateLine │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - wrapDateLine: true, │ │ │ │ │ - │ │ │ │ │ - /** APIProperty: tileOptions │ │ │ │ │ - * {Object} optional configuration options for <OpenLayers.Tile> instances │ │ │ │ │ - * created by this Layer. Default is │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * {crossOriginKeyword: 'anonymous'} │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * When using OSM tilesets other than the default ones, it may be │ │ │ │ │ - * necessary to set this to │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * {crossOriginKeyword: null} │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * if the server does not send Access-Control-Allow-Origin headers. │ │ │ │ │ + * Method: unregister │ │ │ │ │ + * unregister an event from the map │ │ │ │ │ */ │ │ │ │ │ - tileOptions: null, │ │ │ │ │ + unregister: function(name, method) { │ │ │ │ │ + this.map.events.unregister(name, this, method); │ │ │ │ │ + this.map.events.unregister(name, this, this.setEvent); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.OSM │ │ │ │ │ + * Method: setEvent │ │ │ │ │ + * With each registered browser event, the handler sets its own evt │ │ │ │ │ + * property. This property can be accessed by controls if needed │ │ │ │ │ + * to get more information about the event that the handler is │ │ │ │ │ + * processing. │ │ │ │ │ + * │ │ │ │ │ + * This allows modifier keys on the event to be checked (alt, shift, ctrl, │ │ │ │ │ + * and meta cannot be checked with the keyboard handler). For a │ │ │ │ │ + * control to determine which modifier keys are associated with the │ │ │ │ │ + * event that a handler is currently processing, it should access │ │ │ │ │ + * (code)handler.evt.altKey || handler.evt.shiftKey || │ │ │ │ │ + * handler.evt.ctrlKey || handler.evt.metaKey(end). │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} The layer name. │ │ │ │ │ - * url - {String} The tileset URL scheme. │ │ │ │ │ - * options - {Object} Configuration options for the layer. Any inherited │ │ │ │ │ - * layer option can be set in this object (e.g. │ │ │ │ │ - * <OpenLayers.Layer.Grid.buffer>). │ │ │ │ │ + * evt - {Event} The browser event. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - crossOriginKeyword: 'anonymous' │ │ │ │ │ - }, this.options && this.options.tileOptions); │ │ │ │ │ + setEvent: function(evt) { │ │ │ │ │ + this.evt = evt; │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clone │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Deconstruct the handler. │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.OSM( │ │ │ │ │ - this.name, this.url, this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj; │ │ │ │ │ + destroy: function() { │ │ │ │ │ + // unregister event listeners │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + // eliminate circular references │ │ │ │ │ + this.control = this.map = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.OSM" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_NONE │ │ │ │ │ + * If set as the <keyMask>, <checkModifiers> returns false if any key is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_NONE = 0; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_SHIFT │ │ │ │ │ + * If set as the <keyMask>, <checkModifiers> returns false if Shift is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_SHIFT = 1; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_CTRL │ │ │ │ │ + * If set as the <keyMask>, <checkModifiers> returns false if Ctrl is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_CTRL = 2; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_ALT │ │ │ │ │ + * If set as the <keyMask>, <checkModifiers> returns false if Alt is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_ALT = 4; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_META │ │ │ │ │ + * If set as the <keyMask>, <checkModifiers> returns false if Cmd is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_META = 8; │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Bing.js │ │ │ │ │ + OpenLayers/Geometry.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.Bing │ │ │ │ │ - * Bing layer using direct tile access as provided by Bing Maps REST Services. │ │ │ │ │ - * See http://msdn.microsoft.com/en-us/library/ff701713.aspx for more │ │ │ │ │ - * information. Note: Terms of Service compliant use requires the map to be │ │ │ │ │ - * configured with an <OpenLayers.Control.Attribution> control and the │ │ │ │ │ - * attribution placed on or near the map. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.XYZ> │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Geometry │ │ │ │ │ + * A Geometry is a description of a geographic object. Create an instance of │ │ │ │ │ + * this class with the <OpenLayers.Geometry> constructor. This is a base class, │ │ │ │ │ + * typical geometry types are described by subclasses of this class. │ │ │ │ │ + * │ │ │ │ │ + * Note that if you use the <OpenLayers.Geometry.fromWKT> method, you must │ │ │ │ │ + * explicitly include the OpenLayers.Format.WKT in your build. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Bing = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ +OpenLayers.Geometry = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: key │ │ │ │ │ - * {String} API key for Bing maps, get your own key │ │ │ │ │ - * at http://bingmapsportal.com/ . │ │ │ │ │ + * Property: id │ │ │ │ │ + * {String} A unique identifier for this geometry. │ │ │ │ │ */ │ │ │ │ │ - key: null, │ │ │ │ │ + id: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: serverResolutions │ │ │ │ │ - * {Array} the resolutions provided by the Bing servers. │ │ │ │ │ + * Property: parent │ │ │ │ │ + * {<OpenLayers.Geometry>}This is set when a Geometry is added as component │ │ │ │ │ + * of another geometry │ │ │ │ │ */ │ │ │ │ │ - serverResolutions: [ │ │ │ │ │ - 156543.03390625, 78271.516953125, 39135.7584765625, │ │ │ │ │ - 19567.87923828125, 9783.939619140625, 4891.9698095703125, │ │ │ │ │ - 2445.9849047851562, 1222.9924523925781, 611.4962261962891, │ │ │ │ │ - 305.74811309814453, 152.87405654907226, 76.43702827453613, │ │ │ │ │ - 38.218514137268066, 19.109257068634033, 9.554628534317017, │ │ │ │ │ - 4.777314267158508, 2.388657133579254, 1.194328566789627, │ │ │ │ │ - 0.5971642833948135, 0.29858214169740677, 0.14929107084870338, │ │ │ │ │ - 0.07464553542435169 │ │ │ │ │ - ], │ │ │ │ │ + parent: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: attributionTemplate │ │ │ │ │ - * {String} │ │ │ │ │ + * Property: bounds │ │ │ │ │ + * {<OpenLayers.Bounds>} The bounds of this geometry │ │ │ │ │ */ │ │ │ │ │ - attributionTemplate: '<span class="olBingAttribution ${type}">' + │ │ │ │ │ - '<div><a target="_blank" href="http://www.bing.com/maps/">' + │ │ │ │ │ - '<img src="${logo}" /></a></div>${copyrights}' + │ │ │ │ │ - '<a style="white-space: nowrap" target="_blank" ' + │ │ │ │ │ - 'href="http://www.microsoft.com/maps/product/terms.html">' + │ │ │ │ │ - 'Terms of Use</a></span>', │ │ │ │ │ + bounds: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: metadata │ │ │ │ │ - * {Object} Metadata for this layer, as returned by the callback script │ │ │ │ │ + * Constructor: OpenLayers.Geometry │ │ │ │ │ + * Creates a geometry object. │ │ │ │ │ */ │ │ │ │ │ - metadata: null, │ │ │ │ │ + initialize: function() { │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: protocolRegex │ │ │ │ │ - * {RegExp} Regular expression to match and replace http: in bing urls │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Destroy this geometry. │ │ │ │ │ */ │ │ │ │ │ - protocolRegex: /^http:/i, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.id = null; │ │ │ │ │ + this.bounds = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: type │ │ │ │ │ - * {String} The layer identifier. Any non-birdseye imageryType │ │ │ │ │ - * from http://msdn.microsoft.com/en-us/library/ff701716.aspx can be │ │ │ │ │ - * used. Default is "Road". │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Create a clone of this geometry. Does not set any non-standard │ │ │ │ │ + * properties of the cloned geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry>} An exact clone of this geometry. │ │ │ │ │ */ │ │ │ │ │ - type: "Road", │ │ │ │ │ + clone: function() { │ │ │ │ │ + return new OpenLayers.Geometry(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: culture │ │ │ │ │ - * {String} The culture identifier. See http://msdn.microsoft.com/en-us/library/ff701709.aspx │ │ │ │ │ - * for the definition and the possible values. Default is "en-US". │ │ │ │ │ + * Method: setBounds │ │ │ │ │ + * Set the bounds for this Geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ */ │ │ │ │ │ - culture: "en-US", │ │ │ │ │ + setBounds: function(bounds) { │ │ │ │ │ + if (bounds) { │ │ │ │ │ + this.bounds = bounds.clone(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: metadataParams │ │ │ │ │ - * {Object} Optional url parameters for the Get Imagery Metadata request │ │ │ │ │ - * as described here: http://msdn.microsoft.com/en-us/library/ff701716.aspx │ │ │ │ │ - */ │ │ │ │ │ - metadataParams: null, │ │ │ │ │ - │ │ │ │ │ - /** APIProperty: tileOptions │ │ │ │ │ - * {Object} optional configuration options for <OpenLayers.Tile> instances │ │ │ │ │ - * created by this Layer. Default is │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * {crossOriginKeyword: 'anonymous'} │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ - tileOptions: null, │ │ │ │ │ - │ │ │ │ │ - /** APIProperty: protocol │ │ │ │ │ - * {String} Protocol to use to fetch Imagery Metadata, tiles and bing logo │ │ │ │ │ - * Can be 'http:' 'https:' or '' │ │ │ │ │ - * │ │ │ │ │ - * Warning: tiles may not be available under both HTTP and HTTPS protocols. │ │ │ │ │ - * Microsoft approved use of both HTTP and HTTPS urls for tiles. However │ │ │ │ │ - * this is undocumented and the Imagery Metadata API always returns HTTP │ │ │ │ │ - * urls. │ │ │ │ │ - * │ │ │ │ │ - * Default is '', unless when executed from a file:/// uri, in which case │ │ │ │ │ - * it is 'http:'. │ │ │ │ │ + * Method: clearBounds │ │ │ │ │ + * Nullify this components bounds and that of its parent as well. │ │ │ │ │ */ │ │ │ │ │ - protocol: ~window.location.href.indexOf('http') ? '' : 'http:', │ │ │ │ │ + clearBounds: function() { │ │ │ │ │ + this.bounds = null; │ │ │ │ │ + if (this.parent) { │ │ │ │ │ + this.parent.clearBounds(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Bing │ │ │ │ │ - * Create a new Bing layer. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var road = new OpenLayers.Layer.Bing({ │ │ │ │ │ - * name: "My Bing Aerial Layer", │ │ │ │ │ - * type: "Aerial", │ │ │ │ │ - * key: "my-api-key-here", │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ + * Method: extendBounds │ │ │ │ │ + * Extend the existing bounds to include the new bounds. │ │ │ │ │ + * If geometry's bounds is not yet set, then set a new Bounds. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Configuration properties for the layer. │ │ │ │ │ - * │ │ │ │ │ - * Required configuration properties: │ │ │ │ │ - * key - {String} Bing Maps API key for your application. Get one at │ │ │ │ │ - * http://bingmapsportal.com/. │ │ │ │ │ - * type - {String} The layer identifier. Any non-birdseye imageryType │ │ │ │ │ - * from http://msdn.microsoft.com/en-us/library/ff701716.aspx can be │ │ │ │ │ - * used. │ │ │ │ │ - * │ │ │ │ │ - * Any other documented layer properties can be provided in the config object. │ │ │ │ │ + * newBounds - {<OpenLayers.Bounds>} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - sphericalMercator: true │ │ │ │ │ - }, options); │ │ │ │ │ - var name = options.name || "Bing " + (options.type || this.type); │ │ │ │ │ - │ │ │ │ │ - var newArgs = [name, null, options]; │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, newArgs); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - crossOriginKeyword: 'anonymous' │ │ │ │ │ - }, this.options.tileOptions); │ │ │ │ │ - this.loadMetadata(); │ │ │ │ │ + extendBounds: function(newBounds) { │ │ │ │ │ + var bounds = this.getBounds(); │ │ │ │ │ + if (!bounds) { │ │ │ │ │ + this.setBounds(newBounds); │ │ │ │ │ + } else { │ │ │ │ │ + this.bounds.extend(newBounds); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: loadMetadata │ │ │ │ │ + * APIMethod: getBounds │ │ │ │ │ + * Get the bounds for this Geometry. If bounds is not set, it │ │ │ │ │ + * is calculated again, this makes queries faster. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} │ │ │ │ │ */ │ │ │ │ │ - loadMetadata: function() { │ │ │ │ │ - this._callbackId = "_callback_" + this.id.replace(/\./g, "_"); │ │ │ │ │ - // link the processMetadata method to the global scope and bind it │ │ │ │ │ - // to this instance │ │ │ │ │ - window[this._callbackId] = OpenLayers.Function.bind( │ │ │ │ │ - OpenLayers.Layer.Bing.processMetadata, this │ │ │ │ │ - ); │ │ │ │ │ - var params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - key: this.key, │ │ │ │ │ - jsonp: this._callbackId, │ │ │ │ │ - include: "ImageryProviders" │ │ │ │ │ - }, this.metadataParams); │ │ │ │ │ - var url = this.protocol + "//dev.virtualearth.net/REST/v1/Imagery/Metadata/" + │ │ │ │ │ - this.type + "?" + OpenLayers.Util.getParameterString(params); │ │ │ │ │ - var script = document.createElement("script"); │ │ │ │ │ - script.type = "text/javascript"; │ │ │ │ │ - script.src = url; │ │ │ │ │ - script.id = this._callbackId; │ │ │ │ │ - document.getElementsByTagName("head")[0].appendChild(script); │ │ │ │ │ + getBounds: function() { │ │ │ │ │ + if (this.bounds == null) { │ │ │ │ │ + this.calculateBounds(); │ │ │ │ │ + } │ │ │ │ │ + return this.bounds; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: initLayer │ │ │ │ │ - * │ │ │ │ │ - * Sets layer properties according to the metadata provided by the API │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: calculateBounds │ │ │ │ │ + * Recalculate the bounds for the geometry. │ │ │ │ │ */ │ │ │ │ │ - initLayer: function() { │ │ │ │ │ - var res = this.metadata.resourceSets[0].resources[0]; │ │ │ │ │ - var url = res.imageUrl.replace("{quadkey}", "${quadkey}"); │ │ │ │ │ - url = url.replace("{culture}", this.culture); │ │ │ │ │ - url = url.replace(this.protocolRegex, this.protocol); │ │ │ │ │ - this.url = []; │ │ │ │ │ - for (var i = 0; i < res.imageUrlSubdomains.length; ++i) { │ │ │ │ │ - this.url.push(url.replace("{subdomain}", res.imageUrlSubdomains[i])); │ │ │ │ │ - } │ │ │ │ │ - this.addOptions({ │ │ │ │ │ - maxResolution: Math.min( │ │ │ │ │ - this.serverResolutions[res.zoomMin], │ │ │ │ │ - this.maxResolution || Number.POSITIVE_INFINITY │ │ │ │ │ - ), │ │ │ │ │ - numZoomLevels: Math.min( │ │ │ │ │ - res.zoomMax + 1 - res.zoomMin, this.numZoomLevels │ │ │ │ │ - ) │ │ │ │ │ - }, true); │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - this.redraw(); │ │ │ │ │ - } │ │ │ │ │ - this.updateAttribution(); │ │ │ │ │ + calculateBounds: function() { │ │ │ │ │ + // │ │ │ │ │ + // This should be overridden by subclasses. │ │ │ │ │ + // │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ + * APIMethod: distanceTo │ │ │ │ │ + * Calculate the closest distance between two geometries (on the x-y plane). │ │ │ │ │ * │ │ │ │ │ - * Paramters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ + * options - {Object} Optional properties for configuring the distance │ │ │ │ │ + * calculation. │ │ │ │ │ + * │ │ │ │ │ + * Valid options depend on the specific geometry type. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number | Object} The distance between this geometry and the target. │ │ │ │ │ + * If details is true, the return will be an object with distance, │ │ │ │ │ + * x0, y0, x1, and x2 properties. The x0 and y0 properties represent │ │ │ │ │ + * the coordinates of the closest point on this geometry. The x1 and y1 │ │ │ │ │ + * properties represent the coordinates of the closest point on the │ │ │ │ │ + * target geometry. │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - if (!this.url) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var xyz = this.getXYZ(bounds), │ │ │ │ │ - x = xyz.x, │ │ │ │ │ - y = xyz.y, │ │ │ │ │ - z = xyz.z; │ │ │ │ │ - var quadDigits = []; │ │ │ │ │ - for (var i = z; i > 0; --i) { │ │ │ │ │ - var digit = '0'; │ │ │ │ │ - var mask = 1 << (i - 1); │ │ │ │ │ - if ((x & mask) != 0) { │ │ │ │ │ - digit++; │ │ │ │ │ - } │ │ │ │ │ - if ((y & mask) != 0) { │ │ │ │ │ - digit++; │ │ │ │ │ - digit++; │ │ │ │ │ - } │ │ │ │ │ - quadDigits.push(digit); │ │ │ │ │ - } │ │ │ │ │ - var quadKey = quadDigits.join(""); │ │ │ │ │ - var url = this.selectUrl('' + x + y + z, this.url); │ │ │ │ │ + distanceTo: function(geometry, options) {}, │ │ │ │ │ │ │ │ │ │ - return OpenLayers.String.format(url, { │ │ │ │ │ - 'quadkey': quadKey │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getVertices │ │ │ │ │ + * Return a list of all points in this geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * nodes - {Boolean} For lines, only return vertices that are │ │ │ │ │ + * endpoints. If false, for lines, only vertices that are not │ │ │ │ │ + * endpoints will be returned. If not provided, all vertices will │ │ │ │ │ + * be returned. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} A list of all vertices in the geometry. │ │ │ │ │ + */ │ │ │ │ │ + getVertices: function(nodes) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: updateAttribution │ │ │ │ │ - * Updates the attribution according to the requirements outlined in │ │ │ │ │ - * http://gis.638310.n2.nabble.com/Bing-imagery-td5789168.html │ │ │ │ │ + * Method: atPoint │ │ │ │ │ + * Note - This is only an approximation based on the bounds of the │ │ │ │ │ + * geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>|Object} OpenLayers.LonLat or an │ │ │ │ │ + * object with a 'lon' and 'lat' properties. │ │ │ │ │ + * toleranceLon - {float} Optional tolerance in Geometric Coords │ │ │ │ │ + * toleranceLat - {float} Optional tolerance in Geographic Coords │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the geometry is at the specified location │ │ │ │ │ */ │ │ │ │ │ - updateAttribution: function() { │ │ │ │ │ - var metadata = this.metadata; │ │ │ │ │ - if (!metadata.resourceSets || !this.map || !this.map.center) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var res = metadata.resourceSets[0].resources[0]; │ │ │ │ │ - var extent = this.map.getExtent().transform( │ │ │ │ │ - this.map.getProjectionObject(), │ │ │ │ │ - new OpenLayers.Projection("EPSG:4326") │ │ │ │ │ - ); │ │ │ │ │ - var providers = res.imageryProviders || [], │ │ │ │ │ - zoom = OpenLayers.Util.indexOf(this.serverResolutions, │ │ │ │ │ - this.getServerResolution()), │ │ │ │ │ - copyrights = "", │ │ │ │ │ - provider, i, ii, j, jj, bbox, coverage; │ │ │ │ │ - for (i = 0, ii = providers.length; i < ii; ++i) { │ │ │ │ │ - provider = providers[i]; │ │ │ │ │ - for (j = 0, jj = provider.coverageAreas.length; j < jj; ++j) { │ │ │ │ │ - coverage = provider.coverageAreas[j]; │ │ │ │ │ - // axis order provided is Y,X │ │ │ │ │ - bbox = OpenLayers.Bounds.fromArray(coverage.bbox, true); │ │ │ │ │ - if (extent.intersectsBounds(bbox) && │ │ │ │ │ - zoom <= coverage.zoomMax && zoom >= coverage.zoomMin) { │ │ │ │ │ - copyrights += provider.attribution + " "; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + atPoint: function(lonlat, toleranceLon, toleranceLat) { │ │ │ │ │ + var atPoint = false; │ │ │ │ │ + var bounds = this.getBounds(); │ │ │ │ │ + if ((bounds != null) && (lonlat != null)) { │ │ │ │ │ + │ │ │ │ │ + var dX = (toleranceLon != null) ? toleranceLon : 0; │ │ │ │ │ + var dY = (toleranceLat != null) ? toleranceLat : 0; │ │ │ │ │ + │ │ │ │ │ + var toleranceBounds = │ │ │ │ │ + new OpenLayers.Bounds(this.bounds.left - dX, │ │ │ │ │ + this.bounds.bottom - dY, │ │ │ │ │ + this.bounds.right + dX, │ │ │ │ │ + this.bounds.top + dY); │ │ │ │ │ + │ │ │ │ │ + atPoint = toleranceBounds.containsLonLat(lonlat); │ │ │ │ │ } │ │ │ │ │ - var logo = metadata.brandLogoUri.replace(this.protocolRegex, this.protocol); │ │ │ │ │ - this.attribution = OpenLayers.String.format(this.attributionTemplate, { │ │ │ │ │ - type: this.type.toLowerCase(), │ │ │ │ │ - logo: logo, │ │ │ │ │ - copyrights: copyrights │ │ │ │ │ - }); │ │ │ │ │ - this.map && this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "attribution" │ │ │ │ │ - }); │ │ │ │ │ + return atPoint; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ + * Method: getLength │ │ │ │ │ + * Calculate the length of this geometry. This method is defined in │ │ │ │ │ + * subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The length of the collection by summing its parts │ │ │ │ │ */ │ │ │ │ │ - setMap: function() { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.register("moveend", this, this.updateAttribution); │ │ │ │ │ + getLength: function() { │ │ │ │ │ + //to be overridden by geometries that actually have a length │ │ │ │ │ + // │ │ │ │ │ + return 0.0; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} │ │ │ │ │ + * Method: getArea │ │ │ │ │ + * Calculate the area of this geometry. This method is defined in subclasses. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Bing>} An exact clone of this <OpenLayers.Layer.Bing> │ │ │ │ │ + * {Float} The area of the collection by summing its parts │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Bing(this.options); │ │ │ │ │ - } │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - return obj; │ │ │ │ │ + getArea: function() { │ │ │ │ │ + //to be overridden by geometries that actually have an area │ │ │ │ │ + // │ │ │ │ │ + return 0.0; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * APIMethod: getCentroid │ │ │ │ │ + * Calculate the centroid of this geometry. This method is defined in subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Point>} The centroid of the collection │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.map && │ │ │ │ │ - this.map.events.unregister("moveend", this, this.updateAttribution); │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.destroy.apply(this, arguments); │ │ │ │ │ + getCentroid: function() { │ │ │ │ │ + return null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Bing" │ │ │ │ │ + /** │ │ │ │ │ + * Method: toString │ │ │ │ │ + * Returns a text representation of the geometry. If the WKT format is │ │ │ │ │ + * included in a build, this will be the Well-Known Text │ │ │ │ │ + * representation. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} String representation of this geometry. │ │ │ │ │ + */ │ │ │ │ │ + toString: function() { │ │ │ │ │ + var string; │ │ │ │ │ + if (OpenLayers.Format && OpenLayers.Format.WKT) { │ │ │ │ │ + string = OpenLayers.Format.WKT.prototype.write( │ │ │ │ │ + new OpenLayers.Feature.Vector(this) │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + string = Object.prototype.toString.call(this); │ │ │ │ │ + } │ │ │ │ │ + return string; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Geometry" │ │ │ │ │ }); │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Function: OpenLayers.Layer.Bing.processMetadata │ │ │ │ │ - * This function will be bound to an instance, linked to the global scope with │ │ │ │ │ - * an id, and called by the JSONP script returned by the API. │ │ │ │ │ + * Function: OpenLayers.Geometry.fromWKT │ │ │ │ │ + * Generate a geometry given a Well-Known Text string. For this method to │ │ │ │ │ + * work, you must include the OpenLayers.Format.WKT in your build │ │ │ │ │ + * explicitly. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * metadata - {Object} metadata as returned by the API │ │ │ │ │ + * wkt - {String} A string representing the geometry in Well-Known Text. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry>} A geometry of the appropriate class. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Bing.processMetadata = function(metadata) { │ │ │ │ │ - this.metadata = metadata; │ │ │ │ │ - this.initLayer(); │ │ │ │ │ - var script = document.getElementById(this._callbackId); │ │ │ │ │ - script.parentNode.removeChild(script); │ │ │ │ │ - window[this._callbackId] = undefined; // cannot delete from window in IE │ │ │ │ │ - delete this._callbackId; │ │ │ │ │ +OpenLayers.Geometry.fromWKT = function(wkt) { │ │ │ │ │ + var geom; │ │ │ │ │ + if (OpenLayers.Format && OpenLayers.Format.WKT) { │ │ │ │ │ + var format = OpenLayers.Geometry.fromWKT.format; │ │ │ │ │ + if (!format) { │ │ │ │ │ + format = new OpenLayers.Format.WKT(); │ │ │ │ │ + OpenLayers.Geometry.fromWKT.format = format; │ │ │ │ │ + } │ │ │ │ │ + var result = format.read(wkt); │ │ │ │ │ + if (result instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ + geom = result.geometry; │ │ │ │ │ + } else if (OpenLayers.Util.isArray(result)) { │ │ │ │ │ + var len = result.length; │ │ │ │ │ + var components = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + components[i] = result[i].geometry; │ │ │ │ │ + } │ │ │ │ │ + geom = new OpenLayers.Geometry.Collection(components); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return geom; │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Method: OpenLayers.Geometry.segmentsIntersect │ │ │ │ │ + * Determine whether two line segments intersect. Optionally calculates │ │ │ │ │ + * and returns the intersection point. This function is optimized for │ │ │ │ │ + * cases where seg1.x2 >= seg2.x1 || seg2.x2 >= seg1.x1. In those │ │ │ │ │ + * obvious cases where there is no intersection, the function should │ │ │ │ │ + * not be called. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * seg1 - {Object} Object representing a segment with properties x1, y1, x2, │ │ │ │ │ + * and y2. The start point is represented by x1 and y1. The end point │ │ │ │ │ + * is represented by x2 and y2. Start and end are ordered so that x1 < x2. │ │ │ │ │ + * seg2 - {Object} Object representing a segment with properties x1, y1, x2, │ │ │ │ │ + * and y2. The start point is represented by x1 and y1. The end point │ │ │ │ │ + * is represented by x2 and y2. Start and end are ordered so that x1 < x2. │ │ │ │ │ + * options - {Object} Optional properties for calculating the intersection. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * point - {Boolean} Return the intersection point. If false, the actual │ │ │ │ │ + * intersection point will not be calculated. If true and the segments │ │ │ │ │ + * intersect, the intersection point will be returned. If true and │ │ │ │ │ + * the segments do not intersect, false will be returned. If true and │ │ │ │ │ + * the segments are coincident, true will be returned. │ │ │ │ │ + * tolerance - {Number} If a non-null value is provided, if the segments are │ │ │ │ │ + * within the tolerance distance, this will be considered an intersection. │ │ │ │ │ + * In addition, if the point option is true and the calculated intersection │ │ │ │ │ + * is within the tolerance distance of an end point, the endpoint will be │ │ │ │ │ + * returned instead of the calculated intersection. Further, if the │ │ │ │ │ + * intersection is within the tolerance of endpoints on both segments, or │ │ │ │ │ + * if two segment endpoints are within the tolerance distance of eachother │ │ │ │ │ + * (but no intersection is otherwise calculated), an endpoint on the │ │ │ │ │ + * first segment provided will be returned. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean | <OpenLayers.Geometry.Point>} The two segments intersect. │ │ │ │ │ + * If the point argument is true, the return will be the intersection │ │ │ │ │ + * point or false if none exists. If point is true and the segments │ │ │ │ │ + * are coincident, return will be true (and the instersection is equal │ │ │ │ │ + * to the shorter segment). │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Geometry.segmentsIntersect = function(seg1, seg2, options) { │ │ │ │ │ + var point = options && options.point; │ │ │ │ │ + var tolerance = options && options.tolerance; │ │ │ │ │ + var intersection = false; │ │ │ │ │ + var x11_21 = seg1.x1 - seg2.x1; │ │ │ │ │ + var y11_21 = seg1.y1 - seg2.y1; │ │ │ │ │ + var x12_11 = seg1.x2 - seg1.x1; │ │ │ │ │ + var y12_11 = seg1.y2 - seg1.y1; │ │ │ │ │ + var y22_21 = seg2.y2 - seg2.y1; │ │ │ │ │ + var x22_21 = seg2.x2 - seg2.x1; │ │ │ │ │ + var d = (y22_21 * x12_11) - (x22_21 * y12_11); │ │ │ │ │ + var n1 = (x22_21 * y11_21) - (y22_21 * x11_21); │ │ │ │ │ + var n2 = (x12_11 * y11_21) - (y12_11 * x11_21); │ │ │ │ │ + if (d == 0) { │ │ │ │ │ + // parallel │ │ │ │ │ + if (n1 == 0 && n2 == 0) { │ │ │ │ │ + // coincident │ │ │ │ │ + intersection = true; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + var along1 = n1 / d; │ │ │ │ │ + var along2 = n2 / d; │ │ │ │ │ + if (along1 >= 0 && along1 <= 1 && along2 >= 0 && along2 <= 1) { │ │ │ │ │ + // intersect │ │ │ │ │ + if (!point) { │ │ │ │ │ + intersection = true; │ │ │ │ │ + } else { │ │ │ │ │ + // calculate the intersection point │ │ │ │ │ + var x = seg1.x1 + (along1 * x12_11); │ │ │ │ │ + var y = seg1.y1 + (along1 * y12_11); │ │ │ │ │ + intersection = new OpenLayers.Geometry.Point(x, y); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (tolerance) { │ │ │ │ │ + var dist; │ │ │ │ │ + if (intersection) { │ │ │ │ │ + if (point) { │ │ │ │ │ + var segs = [seg1, seg2]; │ │ │ │ │ + var seg, x, y; │ │ │ │ │ + // check segment endpoints for proximity to intersection │ │ │ │ │ + // set intersection to first endpoint within the tolerance │ │ │ │ │ + outer: for (var i = 0; i < 2; ++i) { │ │ │ │ │ + seg = segs[i]; │ │ │ │ │ + for (var j = 1; j < 3; ++j) { │ │ │ │ │ + x = seg["x" + j]; │ │ │ │ │ + y = seg["y" + j]; │ │ │ │ │ + dist = Math.sqrt( │ │ │ │ │ + Math.pow(x - intersection.x, 2) + │ │ │ │ │ + Math.pow(y - intersection.y, 2) │ │ │ │ │ + ); │ │ │ │ │ + if (dist < tolerance) { │ │ │ │ │ + intersection.x = x; │ │ │ │ │ + intersection.y = y; │ │ │ │ │ + break outer; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // no calculated intersection, but segments could be within │ │ │ │ │ + // the tolerance of one another │ │ │ │ │ + var segs = [seg1, seg2]; │ │ │ │ │ + var source, target, x, y, p, result; │ │ │ │ │ + // check segment endpoints for proximity to intersection │ │ │ │ │ + // set intersection to first endpoint within the tolerance │ │ │ │ │ + outer: for (var i = 0; i < 2; ++i) { │ │ │ │ │ + source = segs[i]; │ │ │ │ │ + target = segs[(i + 1) % 2]; │ │ │ │ │ + for (var j = 1; j < 3; ++j) { │ │ │ │ │ + p = { │ │ │ │ │ + x: source["x" + j], │ │ │ │ │ + y: source["y" + j] │ │ │ │ │ + }; │ │ │ │ │ + result = OpenLayers.Geometry.distanceToSegment(p, target); │ │ │ │ │ + if (result.distance < tolerance) { │ │ │ │ │ + if (point) { │ │ │ │ │ + intersection = new OpenLayers.Geometry.Point(p.x, p.y); │ │ │ │ │ + } else { │ │ │ │ │ + intersection = true; │ │ │ │ │ + } │ │ │ │ │ + break outer; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return intersection; │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Function: OpenLayers.Geometry.distanceToSegment │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {Object} An object with x and y properties representing the │ │ │ │ │ + * point coordinates. │ │ │ │ │ + * segment - {Object} An object with x1, y1, x2, and y2 properties │ │ │ │ │ + * representing endpoint coordinates. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object with distance, along, x, and y properties. The distance │ │ │ │ │ + * will be the shortest distance between the input point and segment. │ │ │ │ │ + * The x and y properties represent the coordinates along the segment │ │ │ │ │ + * where the shortest distance meets the segment. The along attribute │ │ │ │ │ + * describes how far between the two segment points the given point is. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Geometry.distanceToSegment = function(point, segment) { │ │ │ │ │ + var result = OpenLayers.Geometry.distanceSquaredToSegment(point, segment); │ │ │ │ │ + result.distance = Math.sqrt(result.distance); │ │ │ │ │ + return result; │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Function: OpenLayers.Geometry.distanceSquaredToSegment │ │ │ │ │ + * │ │ │ │ │ + * Usually the distanceToSegment function should be used. This variant however │ │ │ │ │ + * can be used for comparisons where the exact distance is not important. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {Object} An object with x and y properties representing the │ │ │ │ │ + * point coordinates. │ │ │ │ │ + * segment - {Object} An object with x1, y1, x2, and y2 properties │ │ │ │ │ + * representing endpoint coordinates. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object with squared distance, along, x, and y properties. │ │ │ │ │ + * The distance will be the shortest distance between the input point and │ │ │ │ │ + * segment. The x and y properties represent the coordinates along the │ │ │ │ │ + * segment where the shortest distance meets the segment. The along │ │ │ │ │ + * attribute describes how far between the two segment points the given │ │ │ │ │ + * point is. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Geometry.distanceSquaredToSegment = function(point, segment) { │ │ │ │ │ + var x0 = point.x; │ │ │ │ │ + var y0 = point.y; │ │ │ │ │ + var x1 = segment.x1; │ │ │ │ │ + var y1 = segment.y1; │ │ │ │ │ + var x2 = segment.x2; │ │ │ │ │ + var y2 = segment.y2; │ │ │ │ │ + var dx = x2 - x1; │ │ │ │ │ + var dy = y2 - y1; │ │ │ │ │ + var along = ((dx * (x0 - x1)) + (dy * (y0 - y1))) / │ │ │ │ │ + (Math.pow(dx, 2) + Math.pow(dy, 2)); │ │ │ │ │ + var x, y; │ │ │ │ │ + if (along <= 0.0) { │ │ │ │ │ + x = x1; │ │ │ │ │ + y = y1; │ │ │ │ │ + } else if (along >= 1.0) { │ │ │ │ │ + x = x2; │ │ │ │ │ + y = y2; │ │ │ │ │ + } else { │ │ │ │ │ + x = x1 + along * dx; │ │ │ │ │ + y = y1 + along * dy; │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + distance: Math.pow(x - x0, 2) + Math.pow(y - y0, 2), │ │ │ │ │ + x: x, │ │ │ │ │ + y: y, │ │ │ │ │ + along: along │ │ │ │ │ + }; │ │ │ │ │ }; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Renderer.js │ │ │ │ │ + OpenLayers/Geometry/Point.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Geometry.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Renderer │ │ │ │ │ - * This is the base class for all renderers. │ │ │ │ │ - * │ │ │ │ │ - * This is based on a merger code written by Paul Spencer and Bertil Chapuis. │ │ │ │ │ - * It is largely composed of virtual functions that are to be implemented │ │ │ │ │ - * in technology-specific subclasses, but there is some generic code too. │ │ │ │ │ - * │ │ │ │ │ - * The functions that *are* implemented here merely deal with the maintenance │ │ │ │ │ - * of the size and extent variables, as well as the cached 'resolution' │ │ │ │ │ - * value. │ │ │ │ │ + * Class: OpenLayers.Geometry.Point │ │ │ │ │ + * Point geometry class. │ │ │ │ │ * │ │ │ │ │ - * A note to the user that all subclasses should use getResolution() instead │ │ │ │ │ - * of directly accessing this.resolution in order to correctly use the │ │ │ │ │ - * cacheing system. │ │ │ │ │ - * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Geometry> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Renderer = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: container │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - container: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: root │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - root: null, │ │ │ │ │ +OpenLayers.Geometry.Point = OpenLayers.Class(OpenLayers.Geometry, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: extent │ │ │ │ │ - * {<OpenLayers.Bounds>} │ │ │ │ │ - */ │ │ │ │ │ - extent: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: locked │ │ │ │ │ - * {Boolean} If the renderer is currently in a state where many things │ │ │ │ │ - * are changing, the 'locked' property is set to true. This means │ │ │ │ │ - * that renderers can expect at least one more drawFeature event to be │ │ │ │ │ - * called with the 'locked' property set to 'true': In some renderers, │ │ │ │ │ - * this might make sense to use as a 'only update local information' │ │ │ │ │ - * flag. │ │ │ │ │ + * APIProperty: x │ │ │ │ │ + * {float} │ │ │ │ │ */ │ │ │ │ │ - locked: false, │ │ │ │ │ + x: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: size │ │ │ │ │ - * {<OpenLayers.Size>} │ │ │ │ │ - */ │ │ │ │ │ - size: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: resolution │ │ │ │ │ - * {Float} cache of current map resolution │ │ │ │ │ - */ │ │ │ │ │ - resolution: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: map │ │ │ │ │ - * {<OpenLayers.Map>} Reference to the map -- this is set in Vector's setMap() │ │ │ │ │ - */ │ │ │ │ │ - map: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: featureDx │ │ │ │ │ - * {Number} Feature offset in x direction. Will be calculated for and │ │ │ │ │ - * applied to the current feature while rendering (see │ │ │ │ │ - * <calculateFeatureDx>). │ │ │ │ │ + * APIProperty: y │ │ │ │ │ + * {float} │ │ │ │ │ */ │ │ │ │ │ - featureDx: 0, │ │ │ │ │ + y: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Renderer │ │ │ │ │ + * Constructor: OpenLayers.Geometry.Point │ │ │ │ │ + * Construct a point geometry. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * containerID - {<String>} │ │ │ │ │ - * options - {Object} options for this renderer. See sublcasses for │ │ │ │ │ - * supported options. │ │ │ │ │ + * x - {float} │ │ │ │ │ + * y - {float} │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - initialize: function(containerID, options) { │ │ │ │ │ - this.container = OpenLayers.Util.getElement(containerID); │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - }, │ │ │ │ │ + initialize: function(x, y) { │ │ │ │ │ + OpenLayers.Geometry.prototype.initialize.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.container = null; │ │ │ │ │ - this.extent = null; │ │ │ │ │ - this.size = null; │ │ │ │ │ - this.resolution = null; │ │ │ │ │ - this.map = null; │ │ │ │ │ + this.x = parseFloat(x); │ │ │ │ │ + this.y = parseFloat(y); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: supported │ │ │ │ │ - * This should be overridden by specific subclasses │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Whether or not the browser supports the renderer class │ │ │ │ │ + * {<OpenLayers.Geometry.Point>} An exact clone of this OpenLayers.Geometry.Point │ │ │ │ │ */ │ │ │ │ │ - supported: function() { │ │ │ │ │ - return false; │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Geometry.Point(this.x, this.y); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // catch any randomly tagged-on properties │ │ │ │ │ + OpenLayers.Util.applyDefaults(obj, this); │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setExtent │ │ │ │ │ - * Set the visible part of the layer. │ │ │ │ │ - * │ │ │ │ │ - * Resolution has probably changed, so we nullify the resolution │ │ │ │ │ - * cache (this.resolution) -- this way it will be re-computed when │ │ │ │ │ - * next it is needed. │ │ │ │ │ - * We nullify the resolution cache (this.resolution) if resolutionChanged │ │ │ │ │ - * is set to true - this way it will be re-computed on the next │ │ │ │ │ - * getResolution() request. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} │ │ │ │ │ - * resolutionChanged - {Boolean} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ - * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ - * False otherwise. │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateBounds │ │ │ │ │ + * Create a new Bounds based on the lon/lat │ │ │ │ │ */ │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - this.extent = extent.clone(); │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - var ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ - extent = extent.scale(1 / ratio); │ │ │ │ │ - this.extent = extent.wrapDateLine(this.map.getMaxExtent()).scale(ratio); │ │ │ │ │ - } │ │ │ │ │ - if (resolutionChanged) { │ │ │ │ │ - this.resolution = null; │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ + calculateBounds: function() { │ │ │ │ │ + this.bounds = new OpenLayers.Bounds(this.x, this.y, │ │ │ │ │ + this.x, this.y); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setSize │ │ │ │ │ - * Sets the size of the drawing surface. │ │ │ │ │ - * │ │ │ │ │ - * Resolution has probably changed, so we nullify the resolution │ │ │ │ │ - * cache (this.resolution) -- this way it will be re-computed when │ │ │ │ │ - * next it is needed. │ │ │ │ │ + * APIMethod: distanceTo │ │ │ │ │ + * Calculate the closest distance between two geometries (on the x-y plane). │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ + * options - {Object} Optional properties for configuring the distance │ │ │ │ │ + * calculation. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * details - {Boolean} Return details from the distance calculation. │ │ │ │ │ + * Default is false. │ │ │ │ │ + * edge - {Boolean} Calculate the distance from this geometry to the │ │ │ │ │ + * nearest edge of the target geometry. Default is true. If true, │ │ │ │ │ + * calling distanceTo from a geometry that is wholly contained within │ │ │ │ │ + * the target will result in a non-zero distance. If false, whenever │ │ │ │ │ + * geometries intersect, calling distanceTo will return 0. If false, │ │ │ │ │ + * details cannot be returned. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number | Object} The distance between this geometry and the target. │ │ │ │ │ + * If details is true, the return will be an object with distance, │ │ │ │ │ + * x0, y0, x1, and x2 properties. The x0 and y0 properties represent │ │ │ │ │ + * the coordinates of the closest point on this geometry. The x1 and y1 │ │ │ │ │ + * properties represent the coordinates of the closest point on the │ │ │ │ │ + * target geometry. │ │ │ │ │ */ │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - this.size = size.clone(); │ │ │ │ │ - this.resolution = null; │ │ │ │ │ + distanceTo: function(geometry, options) { │ │ │ │ │ + var edge = !(options && options.edge === false); │ │ │ │ │ + var details = edge && options && options.details; │ │ │ │ │ + var distance, x0, y0, x1, y1, result; │ │ │ │ │ + if (geometry instanceof OpenLayers.Geometry.Point) { │ │ │ │ │ + x0 = this.x; │ │ │ │ │ + y0 = this.y; │ │ │ │ │ + x1 = geometry.x; │ │ │ │ │ + y1 = geometry.y; │ │ │ │ │ + distance = Math.sqrt(Math.pow(x0 - x1, 2) + Math.pow(y0 - y1, 2)); │ │ │ │ │ + result = !details ? │ │ │ │ │ + distance : { │ │ │ │ │ + x0: x0, │ │ │ │ │ + y0: y0, │ │ │ │ │ + x1: x1, │ │ │ │ │ + y1: y1, │ │ │ │ │ + distance: distance │ │ │ │ │ + }; │ │ │ │ │ + } else { │ │ │ │ │ + result = geometry.distanceTo(this, options); │ │ │ │ │ + if (details) { │ │ │ │ │ + // switch coord order since this geom is target │ │ │ │ │ + result = { │ │ │ │ │ + x0: result.x1, │ │ │ │ │ + y0: result.y1, │ │ │ │ │ + x1: result.x0, │ │ │ │ │ + y1: result.y0, │ │ │ │ │ + distance: result.distance │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return result; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getResolution │ │ │ │ │ - * Uses cached copy of resolution if available to minimize computing │ │ │ │ │ + * APIMethod: equals │ │ │ │ │ + * Determine whether another geometry is equivalent to this one. Geometries │ │ │ │ │ + * are considered equivalent if all components have the same coordinates. │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geom - {<OpenLayers.Geometry.Point>} The geometry to test. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Float} The current map's resolution │ │ │ │ │ + * {Boolean} The supplied geometry is equivalent to this geometry. │ │ │ │ │ */ │ │ │ │ │ - getResolution: function() { │ │ │ │ │ - this.resolution = this.resolution || this.map.getResolution(); │ │ │ │ │ - return this.resolution; │ │ │ │ │ + equals: function(geom) { │ │ │ │ │ + var equals = false; │ │ │ │ │ + if (geom != null) { │ │ │ │ │ + equals = ((this.x == geom.x && this.y == geom.y) || │ │ │ │ │ + (isNaN(this.x) && isNaN(this.y) && isNaN(geom.x) && isNaN(geom.y))); │ │ │ │ │ + } │ │ │ │ │ + return equals; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawFeature │ │ │ │ │ - * Draw the feature. The optional style argument can be used │ │ │ │ │ - * to override the feature's own style. This method should only │ │ │ │ │ - * be called from layer.drawFeature(). │ │ │ │ │ + * Method: toShortString │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * style - {<Object>} │ │ │ │ │ - * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} true if the feature has been drawn completely, false if not, │ │ │ │ │ - * undefined if the feature had no geometry │ │ │ │ │ + * {String} Shortened String representation of Point object. │ │ │ │ │ + * (ex. <i>"5, 42"</i>) │ │ │ │ │ */ │ │ │ │ │ - drawFeature: function(feature, style) { │ │ │ │ │ - if (style == null) { │ │ │ │ │ - style = feature.style; │ │ │ │ │ - } │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - var bounds = feature.geometry.getBounds(); │ │ │ │ │ - if (bounds) { │ │ │ │ │ - var worldBounds; │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - worldBounds = this.map.getMaxExtent(); │ │ │ │ │ - } │ │ │ │ │ - if (!bounds.intersectsBounds(this.extent, { │ │ │ │ │ - worldBounds: worldBounds │ │ │ │ │ - })) { │ │ │ │ │ - style = { │ │ │ │ │ - display: "none" │ │ │ │ │ - }; │ │ │ │ │ - } else { │ │ │ │ │ - this.calculateFeatureDx(bounds, worldBounds); │ │ │ │ │ - } │ │ │ │ │ - var rendered = this.drawGeometry(feature.geometry, style, feature.id); │ │ │ │ │ - if (style.display != "none" && style.label && rendered !== false) { │ │ │ │ │ - │ │ │ │ │ - var location = feature.geometry.getCentroid(); │ │ │ │ │ - if (style.labelXOffset || style.labelYOffset) { │ │ │ │ │ - var xOffset = isNaN(style.labelXOffset) ? 0 : style.labelXOffset; │ │ │ │ │ - var yOffset = isNaN(style.labelYOffset) ? 0 : style.labelYOffset; │ │ │ │ │ - var res = this.getResolution(); │ │ │ │ │ - location.move(xOffset * res, yOffset * res); │ │ │ │ │ - } │ │ │ │ │ - this.drawText(feature.id, style, location); │ │ │ │ │ - } else { │ │ │ │ │ - this.removeText(feature.id); │ │ │ │ │ - } │ │ │ │ │ - return rendered; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + toShortString: function() { │ │ │ │ │ + return (this.x + ", " + this.y); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: calculateFeatureDx │ │ │ │ │ - * {Number} Calculates the feature offset in x direction. Looking at the │ │ │ │ │ - * center of the feature bounds and the renderer extent, we calculate how │ │ │ │ │ - * many world widths the two are away from each other. This distance is │ │ │ │ │ - * used to shift the feature as close as possible to the center of the │ │ │ │ │ - * current enderer extent, which ensures that the feature is visible in the │ │ │ │ │ - * current viewport. │ │ │ │ │ + * APIMethod: move │ │ │ │ │ + * Moves a geometry by the given displacement along positive x and y axes. │ │ │ │ │ + * This modifies the position of the geometry and clears the cached │ │ │ │ │ + * bounds. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} Bounds of the feature │ │ │ │ │ - * worldBounds - {<OpenLayers.Bounds>} Bounds of the world │ │ │ │ │ + * x - {Float} Distance to move geometry in positive x direction. │ │ │ │ │ + * y - {Float} Distance to move geometry in positive y direction. │ │ │ │ │ */ │ │ │ │ │ - calculateFeatureDx: function(bounds, worldBounds) { │ │ │ │ │ - this.featureDx = 0; │ │ │ │ │ - if (worldBounds) { │ │ │ │ │ - var worldWidth = worldBounds.getWidth(), │ │ │ │ │ - rendererCenterX = (this.extent.left + this.extent.right) / 2, │ │ │ │ │ - featureCenterX = (bounds.left + bounds.right) / 2, │ │ │ │ │ - worldsAway = Math.round((featureCenterX - rendererCenterX) / worldWidth); │ │ │ │ │ - this.featureDx = worldsAway * worldWidth; │ │ │ │ │ - } │ │ │ │ │ + move: function(x, y) { │ │ │ │ │ + this.x = this.x + x; │ │ │ │ │ + this.y = this.y + y; │ │ │ │ │ + this.clearBounds(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawGeometry │ │ │ │ │ - * │ │ │ │ │ - * Draw a geometry. This should only be called from the renderer itself. │ │ │ │ │ - * Use layer.drawFeature() from outside the renderer. │ │ │ │ │ - * virtual function │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: rotate │ │ │ │ │ + * Rotate a point around another. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {<String>} │ │ │ │ │ + * angle - {Float} Rotation angle in degrees (measured counterclockwise │ │ │ │ │ + * from the positive x-axis) │ │ │ │ │ + * origin - {<OpenLayers.Geometry.Point>} Center point for the rotation │ │ │ │ │ */ │ │ │ │ │ - drawGeometry: function(geometry, style, featureId) {}, │ │ │ │ │ + rotate: function(angle, origin) { │ │ │ │ │ + angle *= Math.PI / 180; │ │ │ │ │ + var radius = this.distanceTo(origin); │ │ │ │ │ + var theta = angle + Math.atan2(this.y - origin.y, this.x - origin.x); │ │ │ │ │ + this.x = origin.x + (radius * Math.cos(theta)); │ │ │ │ │ + this.y = origin.y + (radius * Math.sin(theta)); │ │ │ │ │ + this.clearBounds(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawText │ │ │ │ │ - * Function for drawing text labels. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * style - │ │ │ │ │ - * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * APIMethod: getCentroid │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Point>} The centroid of the collection │ │ │ │ │ */ │ │ │ │ │ - drawText: function(featureId, style, location) {}, │ │ │ │ │ + getCentroid: function() { │ │ │ │ │ + return new OpenLayers.Geometry.Point(this.x, this.y); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removeText │ │ │ │ │ - * Function for removing text labels. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ + * APIMethod: resize │ │ │ │ │ + * Resize a point relative to some origin. For points, this has the effect │ │ │ │ │ + * of scaling a vector (from the origin to the point). This method is │ │ │ │ │ + * more useful on geometry collection subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * scale - {Float} Ratio of the new distance from the origin to the old │ │ │ │ │ + * distance from the origin. A scale of 2 doubles the │ │ │ │ │ + * distance between the point and origin. │ │ │ │ │ + * origin - {<OpenLayers.Geometry.Point>} Point of origin for resizing │ │ │ │ │ + * ratio - {Float} Optional x:y ratio for resizing. Default ratio is 1. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - */ │ │ │ │ │ - removeText: function(featureId) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clear │ │ │ │ │ - * Clear all vectors from the renderer. │ │ │ │ │ - * virtual function. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry>} - The current geometry. │ │ │ │ │ */ │ │ │ │ │ - clear: function() {}, │ │ │ │ │ + resize: function(scale, origin, ratio) { │ │ │ │ │ + ratio = (ratio == undefined) ? 1 : ratio; │ │ │ │ │ + this.x = origin.x + (scale * ratio * (this.x - origin.x)); │ │ │ │ │ + this.y = origin.y + (scale * (this.y - origin.y)); │ │ │ │ │ + this.clearBounds(); │ │ │ │ │ + return this; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureIdFromEvent │ │ │ │ │ - * Returns a feature id from an event on the renderer. │ │ │ │ │ - * How this happens is specific to the renderer. This should be │ │ │ │ │ - * called from layer.getFeatureFromEvent(). │ │ │ │ │ - * Virtual function. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: intersects │ │ │ │ │ + * Determine if the input geometry intersects this one. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} Any type of geometry. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A feature id or undefined. │ │ │ │ │ - */ │ │ │ │ │ - getFeatureIdFromEvent: function(evt) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: eraseFeatures │ │ │ │ │ - * This is called by the layer to erase features │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * {Boolean} The input geometry intersects this one. │ │ │ │ │ */ │ │ │ │ │ - eraseFeatures: function(features) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - this.eraseGeometry(feature.geometry, feature.id); │ │ │ │ │ - this.removeText(feature.id); │ │ │ │ │ + intersects: function(geometry) { │ │ │ │ │ + var intersect = false; │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + intersect = this.equals(geometry); │ │ │ │ │ + } else { │ │ │ │ │ + intersect = geometry.intersects(this); │ │ │ │ │ } │ │ │ │ │ + return intersect; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: eraseGeometry │ │ │ │ │ - * Remove a geometry from the renderer (by id). │ │ │ │ │ - * virtual function. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - */ │ │ │ │ │ - eraseGeometry: function(geometry, featureId) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveRoot │ │ │ │ │ - * moves this renderer's root to a (different) renderer. │ │ │ │ │ - * To be implemented by subclasses that require a common renderer root for │ │ │ │ │ - * feature selection. │ │ │ │ │ + * APIMethod: transform │ │ │ │ │ + * Translate the x,y properties of the point from source to dest. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * renderer - {<OpenLayers.Renderer>} target renderer for the moved root │ │ │ │ │ - */ │ │ │ │ │ - moveRoot: function(renderer) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getRenderLayerId │ │ │ │ │ - * Gets the layer that this renderer's output appears on. If moveRoot was │ │ │ │ │ - * used, this will be different from the id of the layer containing the │ │ │ │ │ - * features rendered by this renderer. │ │ │ │ │ + * source - {<OpenLayers.Projection>} │ │ │ │ │ + * dest - {<OpenLayers.Projection>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} the id of the output layer. │ │ │ │ │ + * {<OpenLayers.Geometry>} │ │ │ │ │ */ │ │ │ │ │ - getRenderLayerId: function() { │ │ │ │ │ - return this.container.id; │ │ │ │ │ + transform: function(source, dest) { │ │ │ │ │ + if ((source && dest)) { │ │ │ │ │ + OpenLayers.Projection.transform( │ │ │ │ │ + this, source, dest); │ │ │ │ │ + this.bounds = null; │ │ │ │ │ + } │ │ │ │ │ + return this; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: applyDefaultSymbolizer │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getVertices │ │ │ │ │ + * Return a list of all points in this geometry. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * symbolizer - {Object} │ │ │ │ │ - * │ │ │ │ │ + * nodes - {Boolean} For lines, only return vertices that are │ │ │ │ │ + * endpoints. If false, for lines, only vertices that are not │ │ │ │ │ + * endpoints will be returned. If not provided, all vertices will │ │ │ │ │ + * be returned. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} │ │ │ │ │ + * {Array} A list of all vertices in the geometry. │ │ │ │ │ */ │ │ │ │ │ - applyDefaultSymbolizer: function(symbolizer) { │ │ │ │ │ - var result = OpenLayers.Util.extend({}, │ │ │ │ │ - OpenLayers.Renderer.defaultSymbolizer); │ │ │ │ │ - if (symbolizer.stroke === false) { │ │ │ │ │ - delete result.strokeWidth; │ │ │ │ │ - delete result.strokeColor; │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.fill === false) { │ │ │ │ │ - delete result.fillColor; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.extend(result, symbolizer); │ │ │ │ │ - return result; │ │ │ │ │ + getVertices: function(nodes) { │ │ │ │ │ + return [this]; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Geometry.Point" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.defaultSymbolizer │ │ │ │ │ - * {Object} Properties from this symbolizer will be applied to symbolizers │ │ │ │ │ - * with missing properties. This can also be used to set a global │ │ │ │ │ - * symbolizer default in OpenLayers. To be SLD 1.x compliant, add the │ │ │ │ │ - * following code before rendering any vector features: │ │ │ │ │ - * (code) │ │ │ │ │ - * OpenLayers.Renderer.defaultSymbolizer = { │ │ │ │ │ - * fillColor: "#808080", │ │ │ │ │ - * fillOpacity: 1, │ │ │ │ │ - * strokeColor: "#000000", │ │ │ │ │ - * strokeOpacity: 1, │ │ │ │ │ - * strokeWidth: 1, │ │ │ │ │ - * pointRadius: 3, │ │ │ │ │ - * graphicName: "square" │ │ │ │ │ - * }; │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.defaultSymbolizer = { │ │ │ │ │ - fillColor: "#000000", │ │ │ │ │ - strokeColor: "#000000", │ │ │ │ │ - strokeWidth: 2, │ │ │ │ │ - fillOpacity: 1, │ │ │ │ │ - strokeOpacity: 1, │ │ │ │ │ - pointRadius: 0, │ │ │ │ │ - labelAlign: 'cm' │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.symbol │ │ │ │ │ - * Coordinate arrays for well known (named) symbols. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.symbol = { │ │ │ │ │ - "star": [350, 75, 379, 161, 469, 161, 397, 215, 423, 301, 350, 250, 277, 301, │ │ │ │ │ - 303, 215, 231, 161, 321, 161, 350, 75 │ │ │ │ │ - ], │ │ │ │ │ - "cross": [4, 0, 6, 0, 6, 4, 10, 4, 10, 6, 6, 6, 6, 10, 4, 10, 4, 6, 0, 6, 0, 4, 4, 4, │ │ │ │ │ - 4, 0 │ │ │ │ │ - ], │ │ │ │ │ - "x": [0, 0, 25, 0, 50, 35, 75, 0, 100, 0, 65, 50, 100, 100, 75, 100, 50, 65, 25, 100, 0, 100, 35, 50, 0, 0], │ │ │ │ │ - "square": [0, 0, 0, 1, 1, 1, 1, 0, 0, 0], │ │ │ │ │ - "triangle": [0, 10, 10, 10, 5, 0, 0, 10] │ │ │ │ │ -}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Feature.js │ │ │ │ │ + OpenLayers/Handler/Point.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Feature │ │ │ │ │ - * Features are combinations of geography and attributes. The OpenLayers.Feature │ │ │ │ │ - * class specifically combines a marker and a lonlat. │ │ │ │ │ + * Class: OpenLayers.Handler.Point │ │ │ │ │ + * Handler to draw a point on the map. Point is displayed on activation, │ │ │ │ │ + * moves on mouse move, and is finished on mouse up. The handler triggers │ │ │ │ │ + * callbacks for 'done', 'cancel', and 'modify'. The modify callback is │ │ │ │ │ + * called with each change in the sketch and will receive the latest point │ │ │ │ │ + * drawn. Create a new instance with the <OpenLayers.Handler.Point> │ │ │ │ │ + * constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Feature = OpenLayers.Class({ │ │ │ │ │ +OpenLayers.Handler.Point = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer>} │ │ │ │ │ + /** │ │ │ │ │ + * Property: point │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} The currently drawn point │ │ │ │ │ */ │ │ │ │ │ - layer: null, │ │ │ │ │ + point: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: id │ │ │ │ │ - * {String} │ │ │ │ │ + /** │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} The temporary drawing layer │ │ │ │ │ */ │ │ │ │ │ - id: null, │ │ │ │ │ + layer: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: lonlat │ │ │ │ │ - * {<OpenLayers.LonLat>} │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: multi │ │ │ │ │ + * {Boolean} Cast features to multi-part geometries before passing to the │ │ │ │ │ + * layer. Default is false. │ │ │ │ │ */ │ │ │ │ │ - lonlat: null, │ │ │ │ │ + multi: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: data │ │ │ │ │ - * {Object} │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: citeCompliant │ │ │ │ │ + * {Boolean} If set to true, coordinates of features drawn in a map extent │ │ │ │ │ + * crossing the date line won't exceed the world bounds. Default is false. │ │ │ │ │ */ │ │ │ │ │ - data: null, │ │ │ │ │ + citeCompliant: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: marker │ │ │ │ │ - * {<OpenLayers.Marker>} │ │ │ │ │ + /** │ │ │ │ │ + * Property: mouseDown │ │ │ │ │ + * {Boolean} The mouse is down │ │ │ │ │ */ │ │ │ │ │ - marker: null, │ │ │ │ │ + mouseDown: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: popupClass │ │ │ │ │ - * {<OpenLayers.Class>} The class which will be used to instantiate │ │ │ │ │ - * a new Popup. Default is <OpenLayers.Popup.Anchored>. │ │ │ │ │ + * Property: stoppedDown │ │ │ │ │ + * {Boolean} Indicate whether the last mousedown stopped the event │ │ │ │ │ + * propagation. │ │ │ │ │ */ │ │ │ │ │ - popupClass: null, │ │ │ │ │ + stoppedDown: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: popup │ │ │ │ │ - * {<OpenLayers.Popup>} │ │ │ │ │ + /** │ │ │ │ │ + * Property: lastDown │ │ │ │ │ + * {<OpenLayers.Pixel>} Location of the last mouse down │ │ │ │ │ */ │ │ │ │ │ - popup: null, │ │ │ │ │ + lastDown: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Feature │ │ │ │ │ - * Constructor for features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer>} │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ - * data - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature>} │ │ │ │ │ + /** │ │ │ │ │ + * Property: lastUp │ │ │ │ │ + * {<OpenLayers.Pixel>} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(layer, lonlat, data) { │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - this.lonlat = lonlat; │ │ │ │ │ - this.data = (data != null) ? data : {}; │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ - }, │ │ │ │ │ + lastUp: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * nullify references to prevent circular references and memory leaks │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: persist │ │ │ │ │ + * {Boolean} Leave the feature rendered until destroyFeature is called. │ │ │ │ │ + * Default is false. If set to true, the feature remains rendered until │ │ │ │ │ + * destroyFeature is called, typically by deactivating the handler or │ │ │ │ │ + * starting another drawing. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ + persist: false, │ │ │ │ │ │ │ │ │ │ - //remove the popup from the map │ │ │ │ │ - if ((this.layer != null) && (this.layer.map != null)) { │ │ │ │ │ - if (this.popup != null) { │ │ │ │ │ - this.layer.map.removePopup(this.popup); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // remove the marker from the layer │ │ │ │ │ - if (this.layer != null && this.marker != null) { │ │ │ │ │ - this.layer.removeMarker(this.marker); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: stopDown │ │ │ │ │ + * {Boolean} Stop event propagation on mousedown. Must be false to │ │ │ │ │ + * allow "pan while drawing". Defaults to false. │ │ │ │ │ + */ │ │ │ │ │ + stopDown: false, │ │ │ │ │ │ │ │ │ │ - this.layer = null; │ │ │ │ │ - this.id = null; │ │ │ │ │ - this.lonlat = null; │ │ │ │ │ - this.data = null; │ │ │ │ │ - if (this.marker != null) { │ │ │ │ │ - this.destroyMarker(this.marker); │ │ │ │ │ - this.marker = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.popup != null) { │ │ │ │ │ - this.destroyPopup(this.popup); │ │ │ │ │ - this.popup = null; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIPropery: stopUp │ │ │ │ │ + * {Boolean} Stop event propagation on mouse. Must be false to │ │ │ │ │ + * allow "pan while dragging". Defaults to fase. │ │ │ │ │ + */ │ │ │ │ │ + stopUp: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onScreen │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the feature is currently visible on screen │ │ │ │ │ - * (based on its 'lonlat' property) │ │ │ │ │ + * Property: layerOptions │ │ │ │ │ + * {Object} Any optional properties to be set on the sketch layer. │ │ │ │ │ */ │ │ │ │ │ - onScreen: function() { │ │ │ │ │ + layerOptions: null, │ │ │ │ │ │ │ │ │ │ - var onScreen = false; │ │ │ │ │ - if ((this.layer != null) && (this.layer.map != null)) { │ │ │ │ │ - var screenBounds = this.layer.map.getExtent(); │ │ │ │ │ - onScreen = screenBounds.containsLonLat(this.lonlat); │ │ │ │ │ - } │ │ │ │ │ - return onScreen; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: pixelTolerance │ │ │ │ │ + * {Number} Maximum number of pixels between down and up (mousedown │ │ │ │ │ + * and mouseup, or touchstart and touchend) for the handler to │ │ │ │ │ + * add a new point. If set to an integer value, if the │ │ │ │ │ + * displacement between down and up is great to this value │ │ │ │ │ + * no point will be added. Default value is 5. │ │ │ │ │ + */ │ │ │ │ │ + pixelTolerance: 5, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: lastTouchPx │ │ │ │ │ + * {<OpenLayers.Pixel>} The last pixel used to know the distance between │ │ │ │ │ + * two touches (for double touch). │ │ │ │ │ + */ │ │ │ │ │ + lastTouchPx: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createMarker │ │ │ │ │ - * Based on the data associated with the Feature, create and return a marker object. │ │ │ │ │ + * Constructor: OpenLayers.Handler.Point │ │ │ │ │ + * Create a new point handler. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Marker>} A Marker Object created from the 'lonlat' and 'icon' properties │ │ │ │ │ - * set in this.data. If no 'lonlat' is set, returns null. If no │ │ │ │ │ - * 'icon' is set, OpenLayers.Marker() will load the default image. │ │ │ │ │ - * │ │ │ │ │ - * Note - this.marker is set to return value │ │ │ │ │ - * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ + * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ + * functions. Various callbacks described below. │ │ │ │ │ + * options - {Object} An optional object with properties to be set on the │ │ │ │ │ + * handler │ │ │ │ │ + * │ │ │ │ │ + * Named callbacks: │ │ │ │ │ + * create - Called when a sketch is first created. Callback called with │ │ │ │ │ + * the creation point geometry and sketch feature. │ │ │ │ │ + * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ + * geometry and the sketch feature. │ │ │ │ │ + * done - Called when the point drawing is finished. The callback will │ │ │ │ │ + * recieve a single argument, the point geometry. │ │ │ │ │ + * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ + * cancel callback will receive a geometry. │ │ │ │ │ */ │ │ │ │ │ - createMarker: function() { │ │ │ │ │ - │ │ │ │ │ - if (this.lonlat != null) { │ │ │ │ │ - this.marker = new OpenLayers.Marker(this.lonlat, this.data.icon); │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ + this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style['default'], {}); │ │ │ │ │ } │ │ │ │ │ - return this.marker; │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroyMarker │ │ │ │ │ - * Destroys marker. │ │ │ │ │ - * If user overrides the createMarker() function, s/he should be able │ │ │ │ │ - * to also specify an alternative function for destroying it │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * turn on the handler │ │ │ │ │ */ │ │ │ │ │ - destroyMarker: function() { │ │ │ │ │ - this.marker.destroy(); │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (!OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + // create temporary vector layer for rendering geometry sketch │ │ │ │ │ + // TBD: this could be moved to initialize/destroy - setting visibility here │ │ │ │ │ + var options = OpenLayers.Util.extend({ │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + // indicate that the temp vector layer will never be out of range │ │ │ │ │ + // without this, resolution properties must be specified at the │ │ │ │ │ + // map-level for this temporary layer to init its resolutions │ │ │ │ │ + // correctly │ │ │ │ │ + calculateInRange: OpenLayers.Function.True, │ │ │ │ │ + wrapDateLine: this.citeCompliant │ │ │ │ │ + }, this.layerOptions); │ │ │ │ │ + this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ + this.map.addLayer(this.layer); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createPopup │ │ │ │ │ - * Creates a popup object created from the 'lonlat', 'popupSize', │ │ │ │ │ - * and 'popupContentHTML' properties set in this.data. It uses │ │ │ │ │ - * this.marker.icon as default anchor. │ │ │ │ │ - * │ │ │ │ │ - * If no 'lonlat' is set, returns null. │ │ │ │ │ - * If no this.marker has been created, no anchor is sent. │ │ │ │ │ + * Method: createFeature │ │ │ │ │ + * Add temporary features │ │ │ │ │ * │ │ │ │ │ - * Note - the returned popup object is 'owned' by the feature, so you │ │ │ │ │ - * cannot use the popup's destroy method to discard the popup. │ │ │ │ │ - * Instead, you must use the feature's destroyPopup │ │ │ │ │ - * │ │ │ │ │ - * Note - this.popup is set to return value │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * closeBox - {Boolean} create popup with closebox or not │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Popup>} Returns the created popup, which is also set │ │ │ │ │ - * as 'popup' property of this feature. Will be of whatever type │ │ │ │ │ - * specified by this feature's 'popupClass' property, but must be │ │ │ │ │ - * of type <OpenLayers.Popup>. │ │ │ │ │ - * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} A pixel location on the map. │ │ │ │ │ */ │ │ │ │ │ - createPopup: function(closeBox) { │ │ │ │ │ - │ │ │ │ │ - if (this.lonlat != null) { │ │ │ │ │ - if (!this.popup) { │ │ │ │ │ - var anchor = (this.marker) ? this.marker.icon : null; │ │ │ │ │ - var popupClass = this.popupClass ? │ │ │ │ │ - this.popupClass : OpenLayers.Popup.Anchored; │ │ │ │ │ - this.popup = new popupClass(this.id + "_popup", │ │ │ │ │ - this.lonlat, │ │ │ │ │ - this.data.popupSize, │ │ │ │ │ - this.data.popupContentHTML, │ │ │ │ │ - anchor, │ │ │ │ │ - closeBox); │ │ │ │ │ - } │ │ │ │ │ - if (this.data.overflow != null) { │ │ │ │ │ - this.popup.contentDiv.style.overflow = this.data.overflow; │ │ │ │ │ - } │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ + lonlat.lon, lonlat.lat │ │ │ │ │ + ); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.point]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.popup.feature = this; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * turn off the handler │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (!OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - return this.popup; │ │ │ │ │ + this.cancel(); │ │ │ │ │ + // If a layer's map property is set to null, it means that that layer │ │ │ │ │ + // isn't added to the map. Since we ourself added the layer to the map │ │ │ │ │ + // in activate(), we can assume that if this.layer.map is null it means │ │ │ │ │ + // that the layer has been destroyed (as a result of map.destroy() for │ │ │ │ │ + // example. │ │ │ │ │ + if (this.layer.map != null) { │ │ │ │ │ + this.destroyFeature(true); │ │ │ │ │ + this.layer.destroy(false); │ │ │ │ │ + } │ │ │ │ │ + this.layer = null; │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: destroyPopup │ │ │ │ │ - * Destroys the popup created via createPopup. │ │ │ │ │ + * Method: destroyFeature │ │ │ │ │ + * Destroy the temporary geometries │ │ │ │ │ * │ │ │ │ │ - * As with the marker, if user overrides the createPopup() function, s/he │ │ │ │ │ - * should also be able to override the destruction │ │ │ │ │ + * Parameters: │ │ │ │ │ + * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ */ │ │ │ │ │ - destroyPopup: function() { │ │ │ │ │ - if (this.popup) { │ │ │ │ │ - this.popup.feature = null; │ │ │ │ │ - this.popup.destroy(); │ │ │ │ │ - this.popup = null; │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + if (this.layer && (force || !this.persist)) { │ │ │ │ │ + this.layer.destroyFeatures(); │ │ │ │ │ } │ │ │ │ │ + this.point = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Feature" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Feature/Vector.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -// TRASH THIS │ │ │ │ │ -OpenLayers.State = { │ │ │ │ │ - /** states */ │ │ │ │ │ - UNKNOWN: 'Unknown', │ │ │ │ │ - INSERT: 'Insert', │ │ │ │ │ - UPDATE: 'Update', │ │ │ │ │ - DELETE: 'Delete' │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Feature.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Feature.Vector │ │ │ │ │ - * Vector features use the OpenLayers.Geometry classes as geometry description. │ │ │ │ │ - * They have an 'attributes' property, which is the data object, and a 'style' │ │ │ │ │ - * property, the default values of which are defined in the │ │ │ │ │ - * <OpenLayers.Feature.Vector.style> objects. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Feature> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Feature.Vector = OpenLayers.Class(OpenLayers.Feature, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: fid │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - fid: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geometry │ │ │ │ │ - * {<OpenLayers.Geometry>} │ │ │ │ │ - */ │ │ │ │ │ - geometry: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: attributes │ │ │ │ │ - * {Object} This object holds arbitrary, serializable properties that │ │ │ │ │ - * describe the feature. │ │ │ │ │ - */ │ │ │ │ │ - attributes: null, │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Property: bounds │ │ │ │ │ - * {<OpenLayers.Bounds>} The box bounding that feature's geometry, that │ │ │ │ │ - * property can be set by an <OpenLayers.Format> object when │ │ │ │ │ - * deserializing the feature, so in most cases it represents an │ │ │ │ │ - * information set by the server. │ │ │ │ │ - */ │ │ │ │ │ - bounds: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: state │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - state: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: style │ │ │ │ │ - * {Object} │ │ │ │ │ + * Method: destroyPersistedFeature │ │ │ │ │ + * Destroy the persisted feature. │ │ │ │ │ */ │ │ │ │ │ - style: null, │ │ │ │ │ + destroyPersistedFeature: function() { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + if (layer && layer.features.length > 1) { │ │ │ │ │ + this.layer.features[0].destroy(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String} If this property is set it will be taken into account by │ │ │ │ │ - * {<OpenLayers.HTTP>} when upadting or deleting the feature. │ │ │ │ │ + * Method: finalize │ │ │ │ │ + * Finish the geometry and call the "done" callback. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * cancel - {Boolean} Call cancel instead of done callback. Default │ │ │ │ │ + * is false. │ │ │ │ │ */ │ │ │ │ │ - url: null, │ │ │ │ │ + finalize: function(cancel) { │ │ │ │ │ + var key = cancel ? "cancel" : "done"; │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + this.lastDown = null; │ │ │ │ │ + this.lastUp = null; │ │ │ │ │ + this.lastTouchPx = null; │ │ │ │ │ + this.callback(key, [this.geometryClone()]); │ │ │ │ │ + this.destroyFeature(cancel); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: renderIntent │ │ │ │ │ - * {String} rendering intent currently being used │ │ │ │ │ + * APIMethod: cancel │ │ │ │ │ + * Finish the geometry and call the "cancel" callback. │ │ │ │ │ */ │ │ │ │ │ - renderIntent: "default", │ │ │ │ │ + cancel: function() { │ │ │ │ │ + this.finalize(true); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: modified │ │ │ │ │ - * {Object} An object with the originals of the geometry and attributes of │ │ │ │ │ - * the feature, if they were changed. Currently this property is only read │ │ │ │ │ - * by <OpenLayers.Format.WFST.v1>, and written by │ │ │ │ │ - * <OpenLayers.Control.ModifyFeature>, which sets the geometry property. │ │ │ │ │ - * Applications can set the originals of modified attributes in the │ │ │ │ │ - * attributes property. Note that applications have to check if this │ │ │ │ │ - * object and the attributes property is already created before using it. │ │ │ │ │ - * After a change made with ModifyFeature, this object could look like │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * { │ │ │ │ │ - * geometry: >Object │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * When an application has made changes to feature attributes, it could │ │ │ │ │ - * have set the attributes to something like this: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * { │ │ │ │ │ - * attributes: { │ │ │ │ │ - * myAttribute: "original" │ │ │ │ │ - * } │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ + * Method: click │ │ │ │ │ + * Handle clicks. Clicks are stopped from propagating to other listeners │ │ │ │ │ + * on map.events or other dom elements. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ * │ │ │ │ │ - * Note that <OpenLayers.Format.WFST.v1> only checks for truthy values in │ │ │ │ │ - * *modified.geometry* and the attribute names in *modified.attributes*, │ │ │ │ │ - * but it is recommended to set the original values (and not just true) as │ │ │ │ │ - * attribute value, so applications could use this information to undo │ │ │ │ │ - * changes. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - modified: null, │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Feature.Vector │ │ │ │ │ - * Create a vector feature. │ │ │ │ │ + /** │ │ │ │ │ + * Method: dblclick │ │ │ │ │ + * Handle double-clicks. Double-clicks are stopped from propagating to other │ │ │ │ │ + * listeners on map.events or other dom elements. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} The geometry that this feature │ │ │ │ │ - * represents. │ │ │ │ │ - * attributes - {Object} An optional object that will be mapped to the │ │ │ │ │ - * <attributes> property. │ │ │ │ │ - * style - {Object} An optional style object. │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - initialize: function(geometry, attributes, style) { │ │ │ │ │ - OpenLayers.Feature.prototype.initialize.apply(this, │ │ │ │ │ - [null, null, attributes]); │ │ │ │ │ - this.lonlat = null; │ │ │ │ │ - this.geometry = geometry ? geometry : null; │ │ │ │ │ - this.state = null; │ │ │ │ │ - this.attributes = {}; │ │ │ │ │ - if (attributes) { │ │ │ │ │ - this.attributes = OpenLayers.Util.extend(this.attributes, │ │ │ │ │ - attributes); │ │ │ │ │ - } │ │ │ │ │ - this.style = style ? style : null; │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * nullify references to prevent circular references and memory leaks │ │ │ │ │ + /** │ │ │ │ │ + * Method: modifyFeature │ │ │ │ │ + * Modify the existing geometry given a pixel location. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} A pixel location on the map. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.layer) { │ │ │ │ │ - this.layer.removeFeatures(this); │ │ │ │ │ - this.layer = null; │ │ │ │ │ + modifyFeature: function(pixel) { │ │ │ │ │ + if (!this.point) { │ │ │ │ │ + this.createFeature(pixel); │ │ │ │ │ } │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point.geometry.x = lonlat.lon; │ │ │ │ │ + this.point.geometry.y = lonlat.lat; │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.point, false]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.geometry = null; │ │ │ │ │ - this.modified = null; │ │ │ │ │ - OpenLayers.Feature.prototype.destroy.apply(this, arguments); │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawFeature │ │ │ │ │ + * Render features on the temporary layer. │ │ │ │ │ + */ │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.point, this.style); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this vector feature. Does not set any non-standard │ │ │ │ │ - * properties. │ │ │ │ │ + * Method: getGeometry │ │ │ │ │ + * Return the sketch geometry. If <multi> is true, this will return │ │ │ │ │ + * a multi-part geometry. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} An exact clone of this vector feature. │ │ │ │ │ + * {<OpenLayers.Geometry.Point>} │ │ │ │ │ */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - return new OpenLayers.Feature.Vector( │ │ │ │ │ - this.geometry ? this.geometry.clone() : null, │ │ │ │ │ - this.attributes, │ │ │ │ │ - this.style); │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.point && this.point.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiPoint([geometry]); │ │ │ │ │ + } │ │ │ │ │ + return geometry; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onScreen │ │ │ │ │ - * Determine whether the feature is within the map viewport. This method │ │ │ │ │ - * tests for an intersection between the geometry and the viewport │ │ │ │ │ - * bounds. If a more effecient but less precise geometry bounds │ │ │ │ │ - * intersection is desired, call the method with the boundsOnly │ │ │ │ │ - * parameter true. │ │ │ │ │ + * Method: geometryClone │ │ │ │ │ + * Return a clone of the relevant geometry. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * boundsOnly - {Boolean} Only test whether a feature's bounds intersects │ │ │ │ │ - * the viewport bounds. Default is false. If false, the feature's │ │ │ │ │ - * geometry must intersect the viewport for onScreen to return true. │ │ │ │ │ - * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The feature is currently visible on screen (optionally │ │ │ │ │ - * based on its bounds if boundsOnly is true). │ │ │ │ │ + * {<OpenLayers.Geometry>} │ │ │ │ │ */ │ │ │ │ │ - onScreen: function(boundsOnly) { │ │ │ │ │ - var onScreen = false; │ │ │ │ │ - if (this.layer && this.layer.map) { │ │ │ │ │ - var screenBounds = this.layer.map.getExtent(); │ │ │ │ │ - if (boundsOnly) { │ │ │ │ │ - var featureBounds = this.geometry.getBounds(); │ │ │ │ │ - onScreen = screenBounds.intersectsBounds(featureBounds); │ │ │ │ │ - } else { │ │ │ │ │ - var screenPoly = screenBounds.toGeometry(); │ │ │ │ │ - onScreen = screenPoly.intersects(this.geometry); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return onScreen; │ │ │ │ │ + geometryClone: function() { │ │ │ │ │ + var geom = this.getGeometry(); │ │ │ │ │ + return geom && geom.clone(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getVisibility │ │ │ │ │ - * Determine whether the feature is displayed or not. It may not displayed │ │ │ │ │ - * because: │ │ │ │ │ - * - its style display property is set to 'none', │ │ │ │ │ - * - it doesn't belong to any layer, │ │ │ │ │ - * - the styleMap creates a symbolizer with display property set to 'none' │ │ │ │ │ - * for it, │ │ │ │ │ - * - the layer which it belongs to is not visible. │ │ │ │ │ + * Method: mousedown │ │ │ │ │ + * Handle mousedown. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The feature is currently displayed. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - getVisibility: function() { │ │ │ │ │ - return !(this.style && this.style.display == 'none' || │ │ │ │ │ - !this.layer || │ │ │ │ │ - this.layer && this.layer.styleMap && │ │ │ │ │ - this.layer.styleMap.createSymbolizer(this, this.renderIntent).display == 'none' || │ │ │ │ │ - this.layer && !this.layer.getVisibility()); │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + return this.down(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createMarker │ │ │ │ │ - * HACK - we need to decide if all vector features should be able to │ │ │ │ │ - * create markers │ │ │ │ │ + * Method: touchstart │ │ │ │ │ + * Handle touchstart. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Marker>} For now just returns null │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - createMarker: function() { │ │ │ │ │ - return null; │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + this.lastTouchPx = evt.xy; │ │ │ │ │ + return this.down(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroyMarker │ │ │ │ │ - * HACK - we need to decide if all vector features should be able to │ │ │ │ │ - * delete markers │ │ │ │ │ + * Method: mousemove │ │ │ │ │ + * Handle mousemove. │ │ │ │ │ * │ │ │ │ │ - * If user overrides the createMarker() function, s/he should be able │ │ │ │ │ - * to also specify an alternative function for destroying it │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - destroyMarker: function() { │ │ │ │ │ - // pass │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + return this.move(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createPopup │ │ │ │ │ - * HACK - we need to decide if all vector features should be able to │ │ │ │ │ - * create popups │ │ │ │ │ + * Method: touchmove │ │ │ │ │ + * Handle touchmove. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Popup>} For now just returns null │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - createPopup: function() { │ │ │ │ │ - return null; │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + this.lastTouchPx = evt.xy; │ │ │ │ │ + return this.move(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: atPoint │ │ │ │ │ - * Determins whether the feature intersects with the specified location. │ │ │ │ │ + * Method: mouseup │ │ │ │ │ + * Handle mouseup. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>|Object} OpenLayers.LonLat or an │ │ │ │ │ - * object with a 'lon' and 'lat' properties. │ │ │ │ │ - * toleranceLon - {float} Optional tolerance in Geometric Coords │ │ │ │ │ - * toleranceLat - {float} Optional tolerance in Geographic Coords │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ + */ │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + return this.up(evt); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: touchend │ │ │ │ │ + * Handle touchend. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the feature is at the specified location │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - atPoint: function(lonlat, toleranceLon, toleranceLat) { │ │ │ │ │ - var atPoint = false; │ │ │ │ │ - if (this.geometry) { │ │ │ │ │ - atPoint = this.geometry.atPoint(lonlat, toleranceLon, │ │ │ │ │ - toleranceLat); │ │ │ │ │ - } │ │ │ │ │ - return atPoint; │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + evt.xy = this.lastTouchPx; │ │ │ │ │ + return this.up(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroyPopup │ │ │ │ │ - * HACK - we need to decide if all vector features should be able to │ │ │ │ │ - * delete popups │ │ │ │ │ + * Method: down │ │ │ │ │ + * Handle mousedown and touchstart. Adjust the geometry and redraw. │ │ │ │ │ + * Return determines whether to propagate the event on the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - destroyPopup: function() { │ │ │ │ │ - // pass │ │ │ │ │ + down: function(evt) { │ │ │ │ │ + this.mouseDown = true; │ │ │ │ │ + this.lastDown = evt.xy; │ │ │ │ │ + if (!this.touch) { // no point displayed until up on touch devices │ │ │ │ │ + this.modifyFeature(evt.xy); │ │ │ │ │ + } │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + return !this.stopDown; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Method: move │ │ │ │ │ - * Moves the feature and redraws it at its new location │ │ │ │ │ + * Handle mousemove and touchmove. Adjust the geometry and redraw. │ │ │ │ │ + * Return determines whether to propagate the event on the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ + */ │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + if (!this.touch // no point displayed until up on touch devices │ │ │ │ │ + && │ │ │ │ │ + (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ + this.modifyFeature(evt.xy); │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: up │ │ │ │ │ + * Handle mouseup and touchend. Send the latest point in the geometry to the control. │ │ │ │ │ + * Return determines whether to propagate the event on the map. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * location - {<OpenLayers.LonLat> or <OpenLayers.Pixel>} the │ │ │ │ │ - * location to which to move the feature. │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - move: function(location) { │ │ │ │ │ + up: function(evt) { │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ │ │ │ │ │ - if (!this.layer || !this.geometry.move) { │ │ │ │ │ - //do nothing if no layer or immoveable geometry │ │ │ │ │ - return undefined; │ │ │ │ │ + // check keyboard modifiers │ │ │ │ │ + if (!this.checkModifiers(evt)) { │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var pixel; │ │ │ │ │ - if (location.CLASS_NAME == "OpenLayers.LonLat") { │ │ │ │ │ - pixel = this.layer.getViewPortPxFromLonLat(location); │ │ │ │ │ + // ignore double-clicks │ │ │ │ │ + if (this.lastUp && this.lastUp.equals(evt.xy)) { │ │ │ │ │ + return true; │ │ │ │ │ + } │ │ │ │ │ + if (this.lastDown && this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ + this.pixelTolerance)) { │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy); │ │ │ │ │ + } │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature(); │ │ │ │ │ + } │ │ │ │ │ + this.lastUp = evt.xy; │ │ │ │ │ + this.finalize(); │ │ │ │ │ + return !this.stopUp; │ │ │ │ │ } else { │ │ │ │ │ - pixel = location; │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var lastPixel = this.layer.getViewPortPxFromLonLat(this.geometry.getBounds().getCenterLonLat()); │ │ │ │ │ - var res = this.layer.map.getResolution(); │ │ │ │ │ - this.geometry.move(res * (pixel.x - lastPixel.x), │ │ │ │ │ - res * (lastPixel.y - pixel.y)); │ │ │ │ │ - this.layer.drawFeature(this); │ │ │ │ │ - return lastPixel; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: toState │ │ │ │ │ - * Sets the new state │ │ │ │ │ + * Method: mouseout │ │ │ │ │ + * Handle mouse out. For better user experience reset mouseDown │ │ │ │ │ + * and stoppedDown when the mouse leaves the map viewport. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * state - {String} │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ */ │ │ │ │ │ - toState: function(state) { │ │ │ │ │ - if (state == OpenLayers.State.UPDATE) { │ │ │ │ │ - switch (this.state) { │ │ │ │ │ - case OpenLayers.State.UNKNOWN: │ │ │ │ │ - case OpenLayers.State.DELETE: │ │ │ │ │ - this.state = state; │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.State.UPDATE: │ │ │ │ │ - case OpenLayers.State.INSERT: │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } else if (state == OpenLayers.State.INSERT) { │ │ │ │ │ - switch (this.state) { │ │ │ │ │ - case OpenLayers.State.UNKNOWN: │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - this.state = state; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } else if (state == OpenLayers.State.DELETE) { │ │ │ │ │ - switch (this.state) { │ │ │ │ │ - case OpenLayers.State.INSERT: │ │ │ │ │ - // the feature should be destroyed │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.State.DELETE: │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.State.UNKNOWN: │ │ │ │ │ - case OpenLayers.State.UPDATE: │ │ │ │ │ - this.state = state; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } else if (state == OpenLayers.State.UNKNOWN) { │ │ │ │ │ - this.state = state; │ │ │ │ │ + mouseout: function(evt) { │ │ │ │ │ + if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Feature.Vector" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Feature.Vector.style │ │ │ │ │ - * OpenLayers features can have a number of style attributes. The 'default' │ │ │ │ │ - * style will typically be used if no other style is specified. These │ │ │ │ │ - * styles correspond for the most part, to the styling properties defined │ │ │ │ │ - * by the SVG standard. │ │ │ │ │ - * Information on fill properties: http://www.w3.org/TR/SVG/painting.html#FillProperties │ │ │ │ │ - * Information on stroke properties: http://www.w3.org/TR/SVG/painting.html#StrokeProperties │ │ │ │ │ - * │ │ │ │ │ - * Symbolizer properties: │ │ │ │ │ - * fill - {Boolean} Set to false if no fill is desired. │ │ │ │ │ - * fillColor - {String} Hex fill color. Default is "#ee9900". │ │ │ │ │ - * fillOpacity - {Number} Fill opacity (0-1). Default is 0.4 │ │ │ │ │ - * stroke - {Boolean} Set to false if no stroke is desired. │ │ │ │ │ - * strokeColor - {String} Hex stroke color. Default is "#ee9900". │ │ │ │ │ - * strokeOpacity - {Number} Stroke opacity (0-1). Default is 1. │ │ │ │ │ - * strokeWidth - {Number} Pixel stroke width. Default is 1. │ │ │ │ │ - * strokeLinecap - {String} Stroke cap type. Default is "round". [butt | round | square] │ │ │ │ │ - * strokeDashstyle - {String} Stroke dash style. Default is "solid". [dot | dash | dashdot | longdash | longdashdot | solid] │ │ │ │ │ - * graphic - {Boolean} Set to false if no graphic is desired. │ │ │ │ │ - * pointRadius - {Number} Pixel point radius. Default is 6. │ │ │ │ │ - * pointerEvents - {String} Default is "visiblePainted". │ │ │ │ │ - * cursor - {String} Default is "". │ │ │ │ │ - * externalGraphic - {String} Url to an external graphic that will be used for rendering points. │ │ │ │ │ - * graphicWidth - {Number} Pixel width for sizing an external graphic. │ │ │ │ │ - * graphicHeight - {Number} Pixel height for sizing an external graphic. │ │ │ │ │ - * graphicOpacity - {Number} Opacity (0-1) for an external graphic. │ │ │ │ │ - * graphicXOffset - {Number} Pixel offset along the positive x axis for displacing an external graphic. │ │ │ │ │ - * graphicYOffset - {Number} Pixel offset along the positive y axis for displacing an external graphic. │ │ │ │ │ - * rotation - {Number} For point symbolizers, this is the rotation of a graphic in the clockwise direction about its center point (or any point off center as specified by graphicXOffset and graphicYOffset). │ │ │ │ │ - * graphicZIndex - {Number} The integer z-index value to use in rendering. │ │ │ │ │ - * graphicName - {String} Named graphic to use when rendering points. Supported values include "circle" (default), │ │ │ │ │ - * "square", "star", "x", "cross", "triangle". │ │ │ │ │ - * graphicTitle - {String} Tooltip when hovering over a feature. *deprecated*, use title instead │ │ │ │ │ - * title - {String} Tooltip when hovering over a feature. Not supported by the canvas renderer. │ │ │ │ │ - * backgroundGraphic - {String} Url to a graphic to be used as the background under an externalGraphic. │ │ │ │ │ - * backgroundGraphicZIndex - {Number} The integer z-index value to use in rendering the background graphic. │ │ │ │ │ - * backgroundXOffset - {Number} The x offset (in pixels) for the background graphic. │ │ │ │ │ - * backgroundYOffset - {Number} The y offset (in pixels) for the background graphic. │ │ │ │ │ - * backgroundHeight - {Number} The height of the background graphic. If not provided, the graphicHeight will be used. │ │ │ │ │ - * backgroundWidth - {Number} The width of the background width. If not provided, the graphicWidth will be used. │ │ │ │ │ - * label - {String} The text for an optional label. For browsers that use the canvas renderer, this requires either │ │ │ │ │ - * fillText or mozDrawText to be available. │ │ │ │ │ - * labelAlign - {String} Label alignment. This specifies the insertion point relative to the text. It is a string │ │ │ │ │ - * composed of two characters. The first character is for the horizontal alignment, the second for the vertical │ │ │ │ │ - * alignment. Valid values for horizontal alignment: "l"=left, "c"=center, "r"=right. Valid values for vertical │ │ │ │ │ - * alignment: "t"=top, "m"=middle, "b"=bottom. Example values: "lt", "cm", "rb". Default is "cm". │ │ │ │ │ - * labelXOffset - {Number} Pixel offset along the positive x axis for displacing the label. Not supported by the canvas renderer. │ │ │ │ │ - * labelYOffset - {Number} Pixel offset along the positive y axis for displacing the label. Not supported by the canvas renderer. │ │ │ │ │ - * labelSelect - {Boolean} If set to true, labels will be selectable using SelectFeature or similar controls. │ │ │ │ │ - * Default is false. │ │ │ │ │ - * labelOutlineColor - {String} The color of the label outline. Default is 'white'. Only supported by the canvas & SVG renderers. │ │ │ │ │ - * labelOutlineWidth - {Number} The width of the label outline. Default is 3, set to 0 or null to disable. Only supported by the SVG renderers. │ │ │ │ │ - * labelOutlineOpacity - {Number} The opacity (0-1) of the label outline. Default is fontOpacity. Only supported by the canvas & SVG renderers. │ │ │ │ │ - * fontColor - {String} The font color for the label, to be provided like CSS. │ │ │ │ │ - * fontOpacity - {Number} Opacity (0-1) for the label │ │ │ │ │ - * fontFamily - {String} The font family for the label, to be provided like in CSS. │ │ │ │ │ - * fontSize - {String} The font size for the label, to be provided like in CSS. │ │ │ │ │ - * fontStyle - {String} The font style for the label, to be provided like in CSS. │ │ │ │ │ - * fontWeight - {String} The font weight for the label, to be provided like in CSS. │ │ │ │ │ - * display - {String} Symbolizers will have no effect if display is set to "none". All other values have no effect. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Feature.Vector.style = { │ │ │ │ │ - 'default': { │ │ │ │ │ - fillColor: "#ee9900", │ │ │ │ │ - fillOpacity: 0.4, │ │ │ │ │ - hoverFillColor: "white", │ │ │ │ │ - hoverFillOpacity: 0.8, │ │ │ │ │ - strokeColor: "#ee9900", │ │ │ │ │ - strokeOpacity: 1, │ │ │ │ │ - strokeWidth: 1, │ │ │ │ │ - strokeLinecap: "round", │ │ │ │ │ - strokeDashstyle: "solid", │ │ │ │ │ - hoverStrokeColor: "red", │ │ │ │ │ - hoverStrokeOpacity: 1, │ │ │ │ │ - hoverStrokeWidth: 0.2, │ │ │ │ │ - pointRadius: 6, │ │ │ │ │ - hoverPointRadius: 1, │ │ │ │ │ - hoverPointUnit: "%", │ │ │ │ │ - pointerEvents: "visiblePainted", │ │ │ │ │ - cursor: "inherit", │ │ │ │ │ - fontColor: "#000000", │ │ │ │ │ - labelAlign: "cm", │ │ │ │ │ - labelOutlineColor: "white", │ │ │ │ │ - labelOutlineWidth: 3 │ │ │ │ │ - }, │ │ │ │ │ - 'select': { │ │ │ │ │ - fillColor: "blue", │ │ │ │ │ - fillOpacity: 0.4, │ │ │ │ │ - hoverFillColor: "white", │ │ │ │ │ - hoverFillOpacity: 0.8, │ │ │ │ │ - strokeColor: "blue", │ │ │ │ │ - strokeOpacity: 1, │ │ │ │ │ - strokeWidth: 2, │ │ │ │ │ - strokeLinecap: "round", │ │ │ │ │ - strokeDashstyle: "solid", │ │ │ │ │ - hoverStrokeColor: "red", │ │ │ │ │ - hoverStrokeOpacity: 1, │ │ │ │ │ - hoverStrokeWidth: 0.2, │ │ │ │ │ - pointRadius: 6, │ │ │ │ │ - hoverPointRadius: 1, │ │ │ │ │ - hoverPointUnit: "%", │ │ │ │ │ - pointerEvents: "visiblePainted", │ │ │ │ │ - cursor: "pointer", │ │ │ │ │ - fontColor: "#000000", │ │ │ │ │ - labelAlign: "cm", │ │ │ │ │ - labelOutlineColor: "white", │ │ │ │ │ - labelOutlineWidth: 3 │ │ │ │ │ + /** │ │ │ │ │ + * Method: passesTolerance │ │ │ │ │ + * Determine whether the event is within the optional pixel tolerance. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The event is within the pixel tolerance (if specified). │ │ │ │ │ + */ │ │ │ │ │ + passesTolerance: function(pixel1, pixel2, tolerance) { │ │ │ │ │ + var passes = true; │ │ │ │ │ │ │ │ │ │ + if (tolerance != null && pixel1 && pixel2) { │ │ │ │ │ + var dist = pixel1.distanceTo(pixel2); │ │ │ │ │ + if (dist > tolerance) { │ │ │ │ │ + passes = false; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return passes; │ │ │ │ │ }, │ │ │ │ │ - 'temporary': { │ │ │ │ │ - fillColor: "#66cccc", │ │ │ │ │ - fillOpacity: 0.2, │ │ │ │ │ - hoverFillColor: "white", │ │ │ │ │ - hoverFillOpacity: 0.8, │ │ │ │ │ - strokeColor: "#66cccc", │ │ │ │ │ - strokeOpacity: 1, │ │ │ │ │ - strokeLinecap: "round", │ │ │ │ │ - strokeWidth: 2, │ │ │ │ │ - strokeDashstyle: "solid", │ │ │ │ │ - hoverStrokeColor: "red", │ │ │ │ │ - hoverStrokeOpacity: 1, │ │ │ │ │ - hoverStrokeWidth: 0.2, │ │ │ │ │ - pointRadius: 6, │ │ │ │ │ - hoverPointRadius: 1, │ │ │ │ │ - hoverPointUnit: "%", │ │ │ │ │ - pointerEvents: "visiblePainted", │ │ │ │ │ - cursor: "inherit", │ │ │ │ │ - fontColor: "#000000", │ │ │ │ │ - labelAlign: "cm", │ │ │ │ │ - labelOutlineColor: "white", │ │ │ │ │ - labelOutlineWidth: 3 │ │ │ │ │ │ │ │ │ │ - }, │ │ │ │ │ - 'delete': { │ │ │ │ │ - display: "none" │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Point" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Style.js │ │ │ │ │ + OpenLayers/Geometry/Collection.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Geometry.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Style │ │ │ │ │ - * This class represents a UserStyle obtained │ │ │ │ │ - * from a SLD, containing styling rules. │ │ │ │ │ + * Class: OpenLayers.Geometry.Collection │ │ │ │ │ + * A Collection is exactly what it sounds like: A collection of different │ │ │ │ │ + * Geometries. These are stored in the local parameter <components> (which │ │ │ │ │ + * can be passed as a parameter to the constructor). │ │ │ │ │ + * │ │ │ │ │ + * As new geometries are added to the collection, they are NOT cloned. │ │ │ │ │ + * When removing geometries, they need to be specified by reference (ie you │ │ │ │ │ + * have to pass in the *exact* geometry to be removed). │ │ │ │ │ + * │ │ │ │ │ + * The <getArea> and <getLength> functions here merely iterate through │ │ │ │ │ + * the components, summing their respective areas and lengths. │ │ │ │ │ + * │ │ │ │ │ + * Create a new instance with the <OpenLayers.Geometry.Collection> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Geometry> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Style = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: id │ │ │ │ │ - * {String} A unique id for this session. │ │ │ │ │ - */ │ │ │ │ │ - id: null, │ │ │ │ │ +OpenLayers.Geometry.Collection = OpenLayers.Class(OpenLayers.Geometry, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: name │ │ │ │ │ - * {String} │ │ │ │ │ + * APIProperty: components │ │ │ │ │ + * {Array(<OpenLayers.Geometry>)} The component parts of this geometry │ │ │ │ │ */ │ │ │ │ │ - name: null, │ │ │ │ │ + components: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: title │ │ │ │ │ - * {String} Title of this style (set if included in SLD) │ │ │ │ │ + * Property: componentTypes │ │ │ │ │ + * {Array(String)} An array of class names representing the types of │ │ │ │ │ + * components that the collection can include. A null value means the │ │ │ │ │ + * component types are not restricted. │ │ │ │ │ */ │ │ │ │ │ - title: null, │ │ │ │ │ + componentTypes: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: description │ │ │ │ │ - * {String} Description of this style (set if abstract is included in SLD) │ │ │ │ │ + * Constructor: OpenLayers.Geometry.Collection │ │ │ │ │ + * Creates a Geometry Collection -- a list of geoms. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * components - {Array(<OpenLayers.Geometry>)} Optional array of geometries │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - description: null, │ │ │ │ │ + initialize: function(components) { │ │ │ │ │ + OpenLayers.Geometry.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.components = []; │ │ │ │ │ + if (components != null) { │ │ │ │ │ + this.addComponents(components); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: layerName │ │ │ │ │ - * {<String>} name of the layer that this style belongs to, usually │ │ │ │ │ - * according to the NamedLayer attribute of an SLD document. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Destroy this geometry. │ │ │ │ │ */ │ │ │ │ │ - layerName: null, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.components.length = 0; │ │ │ │ │ + this.components = null; │ │ │ │ │ + OpenLayers.Geometry.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isDefault │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Clone this geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Collection>} An exact clone of this collection │ │ │ │ │ */ │ │ │ │ │ - isDefault: false, │ │ │ │ │ + clone: function() { │ │ │ │ │ + var geometry = eval("new " + this.CLASS_NAME + "()"); │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ + geometry.addComponent(this.components[i].clone()); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: rules │ │ │ │ │ - * {Array(<OpenLayers.Rule>)} │ │ │ │ │ - */ │ │ │ │ │ - rules: null, │ │ │ │ │ + // catch any randomly tagged-on properties │ │ │ │ │ + OpenLayers.Util.applyDefaults(geometry, this); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: context │ │ │ │ │ - * {Object} An optional object with properties that symbolizers' property │ │ │ │ │ - * values should be evaluated against. If no context is specified, │ │ │ │ │ - * feature.attributes will be used │ │ │ │ │ - */ │ │ │ │ │ - context: null, │ │ │ │ │ + return geometry; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: defaultStyle │ │ │ │ │ - * {Object} hash of style properties to use as default for merging │ │ │ │ │ - * rule-based style symbolizers onto. If no rules are defined, │ │ │ │ │ - * createSymbolizer will return this style. If <defaultsPerSymbolizer> is set to │ │ │ │ │ - * true, the defaultStyle will only be taken into account if there are │ │ │ │ │ - * rules defined. │ │ │ │ │ + * Method: getComponentsString │ │ │ │ │ + * Get a string representing the components for this collection │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string representation of the components of this geometry │ │ │ │ │ */ │ │ │ │ │ - defaultStyle: null, │ │ │ │ │ + getComponentsString: function() { │ │ │ │ │ + var strings = []; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ + strings.push(this.components[i].toShortString()); │ │ │ │ │ + } │ │ │ │ │ + return strings.join(","); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: defaultsPerSymbolizer │ │ │ │ │ - * {Boolean} If set to true, the <defaultStyle> will extend the symbolizer │ │ │ │ │ - * of every rule. Properties of the <defaultStyle> will also be used to set │ │ │ │ │ - * missing symbolizer properties if the symbolizer has stroke, fill or │ │ │ │ │ - * graphic set to true. Default is false. │ │ │ │ │ + * APIMethod: calculateBounds │ │ │ │ │ + * Recalculate the bounds by iterating through the components and │ │ │ │ │ + * calling calling extendBounds() on each item. │ │ │ │ │ */ │ │ │ │ │ - defaultsPerSymbolizer: false, │ │ │ │ │ + calculateBounds: function() { │ │ │ │ │ + this.bounds = null; │ │ │ │ │ + var bounds = new OpenLayers.Bounds(); │ │ │ │ │ + var components = this.components; │ │ │ │ │ + if (components) { │ │ │ │ │ + for (var i = 0, len = components.length; i < len; i++) { │ │ │ │ │ + bounds.extend(components[i].getBounds()); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // to preserve old behavior, we only set bounds if non-null │ │ │ │ │ + // in the future, we could add bounds.isEmpty() │ │ │ │ │ + if (bounds.left != null && bounds.bottom != null && │ │ │ │ │ + bounds.right != null && bounds.top != null) { │ │ │ │ │ + this.setBounds(bounds); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: propertyStyles │ │ │ │ │ - * {Hash of Boolean} cache of style properties that need to be parsed for │ │ │ │ │ - * propertyNames. Property names are keys, values won't be used. │ │ │ │ │ - */ │ │ │ │ │ - propertyStyles: null, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Style │ │ │ │ │ - * Creates a UserStyle. │ │ │ │ │ + * APIMethod: addComponents │ │ │ │ │ + * Add components to this geometry. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * style - {Object} Optional hash of style properties that will be │ │ │ │ │ - * used as default style for this style object. This style │ │ │ │ │ - * applies if no rules are specified. Symbolizers defined in │ │ │ │ │ - * rules will extend this default style. │ │ │ │ │ - * options - {Object} An optional object with properties to set on the │ │ │ │ │ - * style. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * rules - {Array(<OpenLayers.Rule>)} List of rules to be added to the │ │ │ │ │ - * style. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Style>} │ │ │ │ │ + * components - {Array(<OpenLayers.Geometry>)} An array of geometries to add │ │ │ │ │ */ │ │ │ │ │ - initialize: function(style, options) { │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.rules = []; │ │ │ │ │ - if (options && options.rules) { │ │ │ │ │ - this.addRules(options.rules); │ │ │ │ │ + addComponents: function(components) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(components))) { │ │ │ │ │ + components = [components]; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // use the default style from OpenLayers.Feature.Vector if no style │ │ │ │ │ - // was given in the constructor │ │ │ │ │ - this.setDefaultStyle(style || │ │ │ │ │ - OpenLayers.Feature.Vector.style["default"]); │ │ │ │ │ - │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * nullify references to prevent circular references and memory leaks │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i = 0, len = this.rules.length; i < len; i++) { │ │ │ │ │ - this.rules[i].destroy(); │ │ │ │ │ - this.rules[i] = null; │ │ │ │ │ + for (var i = 0, len = components.length; i < len; i++) { │ │ │ │ │ + this.addComponent(components[i]); │ │ │ │ │ } │ │ │ │ │ - this.rules = null; │ │ │ │ │ - this.defaultStyle = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createSymbolizer │ │ │ │ │ - * creates a style by applying all feature-dependent rules to the base │ │ │ │ │ - * style. │ │ │ │ │ + * Method: addComponent │ │ │ │ │ + * Add a new component (geometry) to the collection. If this.componentTypes │ │ │ │ │ + * is set, then the component class name must be in the componentTypes array. │ │ │ │ │ + * │ │ │ │ │ + * The bounds cache is reset. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature>} feature to evaluate rules for │ │ │ │ │ - * │ │ │ │ │ + * component - {<OpenLayers.Geometry>} A geometry to add │ │ │ │ │ + * index - {int} Optional index into the array to insert the component │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} symbolizer hash │ │ │ │ │ + * {Boolean} The component geometry was successfully added │ │ │ │ │ */ │ │ │ │ │ - createSymbolizer: function(feature) { │ │ │ │ │ - var style = this.defaultsPerSymbolizer ? {} : this.createLiterals( │ │ │ │ │ - OpenLayers.Util.extend({}, this.defaultStyle), feature); │ │ │ │ │ - │ │ │ │ │ - var rules = this.rules; │ │ │ │ │ - │ │ │ │ │ - var rule, context; │ │ │ │ │ - var elseRules = []; │ │ │ │ │ - var appliedRules = false; │ │ │ │ │ - for (var i = 0, len = rules.length; i < len; i++) { │ │ │ │ │ - rule = rules[i]; │ │ │ │ │ - // does the rule apply? │ │ │ │ │ - var applies = rule.evaluate(feature); │ │ │ │ │ + addComponent: function(component, index) { │ │ │ │ │ + var added = false; │ │ │ │ │ + if (component) { │ │ │ │ │ + if (this.componentTypes == null || │ │ │ │ │ + (OpenLayers.Util.indexOf(this.componentTypes, │ │ │ │ │ + component.CLASS_NAME) > -1)) { │ │ │ │ │ │ │ │ │ │ - if (applies) { │ │ │ │ │ - if (rule instanceof OpenLayers.Rule && rule.elseFilter) { │ │ │ │ │ - elseRules.push(rule); │ │ │ │ │ + if (index != null && (index < this.components.length)) { │ │ │ │ │ + var components1 = this.components.slice(0, index); │ │ │ │ │ + var components2 = this.components.slice(index, │ │ │ │ │ + this.components.length); │ │ │ │ │ + components1.push(component); │ │ │ │ │ + this.components = components1.concat(components2); │ │ │ │ │ } else { │ │ │ │ │ - appliedRules = true; │ │ │ │ │ - this.applySymbolizer(rule, style, feature); │ │ │ │ │ + this.components.push(component); │ │ │ │ │ } │ │ │ │ │ + component.parent = this; │ │ │ │ │ + this.clearBounds(); │ │ │ │ │ + added = true; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // if no other rules apply, apply the rules with else filters │ │ │ │ │ - if (appliedRules == false && elseRules.length > 0) { │ │ │ │ │ - appliedRules = true; │ │ │ │ │ - for (var i = 0, len = elseRules.length; i < len; i++) { │ │ │ │ │ - this.applySymbolizer(elseRules[i], style, feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // don't display if there were rules but none applied │ │ │ │ │ - if (rules.length > 0 && appliedRules == false) { │ │ │ │ │ - style.display = "none"; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (style.label != null && typeof style.label !== "string") { │ │ │ │ │ - style.label = String(style.label); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return style; │ │ │ │ │ + return added; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: applySymbolizer │ │ │ │ │ + * APIMethod: removeComponents │ │ │ │ │ + * Remove components from this geometry. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * rule - {<OpenLayers.Rule>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * feature - {<OpenLayer.Feature.Vector>} │ │ │ │ │ + * components - {Array(<OpenLayers.Geometry>)} The components to be removed │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} A style with new symbolizer applied. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} A component was removed. │ │ │ │ │ */ │ │ │ │ │ - applySymbolizer: function(rule, style, feature) { │ │ │ │ │ - var symbolizerPrefix = feature.geometry ? │ │ │ │ │ - this.getSymbolizerPrefix(feature.geometry) : │ │ │ │ │ - OpenLayers.Style.SYMBOLIZER_PREFIXES[0]; │ │ │ │ │ - │ │ │ │ │ - var symbolizer = rule.symbolizer[symbolizerPrefix] || rule.symbolizer; │ │ │ │ │ + removeComponents: function(components) { │ │ │ │ │ + var removed = false; │ │ │ │ │ │ │ │ │ │ - if (this.defaultsPerSymbolizer === true) { │ │ │ │ │ - var defaults = this.defaultStyle; │ │ │ │ │ - OpenLayers.Util.applyDefaults(symbolizer, { │ │ │ │ │ - pointRadius: defaults.pointRadius │ │ │ │ │ - }); │ │ │ │ │ - if (symbolizer.stroke === true || symbolizer.graphic === true) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(symbolizer, { │ │ │ │ │ - strokeWidth: defaults.strokeWidth, │ │ │ │ │ - strokeColor: defaults.strokeColor, │ │ │ │ │ - strokeOpacity: defaults.strokeOpacity, │ │ │ │ │ - strokeDashstyle: defaults.strokeDashstyle, │ │ │ │ │ - strokeLinecap: defaults.strokeLinecap │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.fill === true || symbolizer.graphic === true) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(symbolizer, { │ │ │ │ │ - fillColor: defaults.fillColor, │ │ │ │ │ - fillOpacity: defaults.fillOpacity │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.graphic === true) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(symbolizer, { │ │ │ │ │ - pointRadius: this.defaultStyle.pointRadius, │ │ │ │ │ - externalGraphic: this.defaultStyle.externalGraphic, │ │ │ │ │ - graphicName: this.defaultStyle.graphicName, │ │ │ │ │ - graphicOpacity: this.defaultStyle.graphicOpacity, │ │ │ │ │ - graphicWidth: this.defaultStyle.graphicWidth, │ │ │ │ │ - graphicHeight: this.defaultStyle.graphicHeight, │ │ │ │ │ - graphicXOffset: this.defaultStyle.graphicXOffset, │ │ │ │ │ - graphicYOffset: this.defaultStyle.graphicYOffset │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ + if (!(OpenLayers.Util.isArray(components))) { │ │ │ │ │ + components = [components]; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // merge the style with the current style │ │ │ │ │ - return this.createLiterals( │ │ │ │ │ - OpenLayers.Util.extend(style, symbolizer), feature); │ │ │ │ │ + for (var i = components.length - 1; i >= 0; --i) { │ │ │ │ │ + removed = this.removeComponent(components[i]) || removed; │ │ │ │ │ + } │ │ │ │ │ + return removed; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createLiterals │ │ │ │ │ - * creates literals for all style properties that have an entry in │ │ │ │ │ - * <this.propertyStyles>. │ │ │ │ │ - * │ │ │ │ │ + * Method: removeComponent │ │ │ │ │ + * Remove a component from this geometry. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * style - {Object} style to create literals for. Will be modified │ │ │ │ │ - * inline. │ │ │ │ │ - * feature - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} the modified style │ │ │ │ │ + * component - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The component was removed. │ │ │ │ │ */ │ │ │ │ │ - createLiterals: function(style, feature) { │ │ │ │ │ - var context = OpenLayers.Util.extend({}, feature.attributes || feature.data); │ │ │ │ │ - OpenLayers.Util.extend(context, this.context); │ │ │ │ │ + removeComponent: function(component) { │ │ │ │ │ │ │ │ │ │ - for (var i in this.propertyStyles) { │ │ │ │ │ - style[i] = OpenLayers.Style.createLiteral(style[i], context, feature, i); │ │ │ │ │ - } │ │ │ │ │ - return style; │ │ │ │ │ + OpenLayers.Util.removeItem(this.components, component); │ │ │ │ │ + │ │ │ │ │ + // clearBounds() so that it gets recalculated on the next call │ │ │ │ │ + // to this.getBounds(); │ │ │ │ │ + this.clearBounds(); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: findPropertyStyles │ │ │ │ │ - * Looks into all rules for this style and the defaultStyle to collect │ │ │ │ │ - * all the style hash property names containing ${...} strings that have │ │ │ │ │ - * to be replaced using the createLiteral method before returning them. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getLength │ │ │ │ │ + * Calculate the length of this geometry │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} hash of property names that need createLiteral parsing. The │ │ │ │ │ - * name of the property is the key, and the value is true; │ │ │ │ │ + * {Float} The length of the geometry │ │ │ │ │ */ │ │ │ │ │ - findPropertyStyles: function() { │ │ │ │ │ - var propertyStyles = {}; │ │ │ │ │ - │ │ │ │ │ - // check the default style │ │ │ │ │ - var style = this.defaultStyle; │ │ │ │ │ - this.addPropertyStyles(propertyStyles, style); │ │ │ │ │ - │ │ │ │ │ - // walk through all rules to check for properties in their symbolizer │ │ │ │ │ - var rules = this.rules; │ │ │ │ │ - var symbolizer, value; │ │ │ │ │ - for (var i = 0, len = rules.length; i < len; i++) { │ │ │ │ │ - symbolizer = rules[i].symbolizer; │ │ │ │ │ - for (var key in symbolizer) { │ │ │ │ │ - value = symbolizer[key]; │ │ │ │ │ - if (typeof value == "object") { │ │ │ │ │ - // symbolizer key is "Point", "Line" or "Polygon" │ │ │ │ │ - this.addPropertyStyles(propertyStyles, value); │ │ │ │ │ - } else { │ │ │ │ │ - // symbolizer is a hash of style properties │ │ │ │ │ - this.addPropertyStyles(propertyStyles, symbolizer); │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + getLength: function() { │ │ │ │ │ + var length = 0.0; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ + length += this.components[i].getLength(); │ │ │ │ │ } │ │ │ │ │ - return propertyStyles; │ │ │ │ │ + return length; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: addPropertyStyles │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * propertyStyles - {Object} hash to add new property styles to. Will be │ │ │ │ │ - * modified inline │ │ │ │ │ - * symbolizer - {Object} search this symbolizer for property styles │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getArea │ │ │ │ │ + * Calculate the area of this geometry. Note how this function is overridden │ │ │ │ │ + * in <OpenLayers.Geometry.Polygon>. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} propertyStyles hash │ │ │ │ │ + * {Float} The area of the collection by summing its parts │ │ │ │ │ */ │ │ │ │ │ - addPropertyStyles: function(propertyStyles, symbolizer) { │ │ │ │ │ - var property; │ │ │ │ │ - for (var key in symbolizer) { │ │ │ │ │ - property = symbolizer[key]; │ │ │ │ │ - if (typeof property == "string" && │ │ │ │ │ - property.match(/\$\{\w+\}/)) { │ │ │ │ │ - propertyStyles[key] = true; │ │ │ │ │ - } │ │ │ │ │ + getArea: function() { │ │ │ │ │ + var area = 0.0; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ + area += this.components[i].getArea(); │ │ │ │ │ } │ │ │ │ │ - return propertyStyles; │ │ │ │ │ + return area; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: addRules │ │ │ │ │ - * Adds rules to this style. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getGeodesicArea │ │ │ │ │ + * Calculate the approximate area of the polygon were it projected onto │ │ │ │ │ + * the earth. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * rules - {Array(<OpenLayers.Rule>)} │ │ │ │ │ - */ │ │ │ │ │ - addRules: function(rules) { │ │ │ │ │ - Array.prototype.push.apply(this.rules, rules); │ │ │ │ │ - this.propertyStyles = this.findPropertyStyles(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setDefaultStyle │ │ │ │ │ - * Sets the default style for this style object. │ │ │ │ │ + * projection - {<OpenLayers.Projection>} The spatial reference system │ │ │ │ │ + * for the geometry coordinates. If not provided, Geographic/WGS84 is │ │ │ │ │ + * assumed. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * style - {Object} Hash of style properties │ │ │ │ │ + * Reference: │ │ │ │ │ + * Robert. G. Chamberlain and William H. Duquette, "Some Algorithms for │ │ │ │ │ + * Polygons on a Sphere", JPL Publication 07-03, Jet Propulsion │ │ │ │ │ + * Laboratory, Pasadena, CA, June 2007 http://trs-new.jpl.nasa.gov/dspace/handle/2014/40409 │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {float} The approximate geodesic area of the geometry in square meters. │ │ │ │ │ */ │ │ │ │ │ - setDefaultStyle: function(style) { │ │ │ │ │ - this.defaultStyle = style; │ │ │ │ │ - this.propertyStyles = this.findPropertyStyles(); │ │ │ │ │ + getGeodesicArea: function(projection) { │ │ │ │ │ + var area = 0.0; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ + area += this.components[i].getGeodesicArea(projection); │ │ │ │ │ + } │ │ │ │ │ + return area; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getSymbolizerPrefix │ │ │ │ │ - * Returns the correct symbolizer prefix according to the │ │ │ │ │ - * geometry type of the passed geometry │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getCentroid │ │ │ │ │ + * │ │ │ │ │ + * Compute the centroid for this geometry collection. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * weighted - {Boolean} Perform the getCentroid computation recursively, │ │ │ │ │ + * returning an area weighted average of all geometries in this collection. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} key of the according symbolizer │ │ │ │ │ + * {<OpenLayers.Geometry.Point>} The centroid of the collection │ │ │ │ │ */ │ │ │ │ │ - getSymbolizerPrefix: function(geometry) { │ │ │ │ │ - var prefixes = OpenLayers.Style.SYMBOLIZER_PREFIXES; │ │ │ │ │ - for (var i = 0, len = prefixes.length; i < len; i++) { │ │ │ │ │ - if (geometry.CLASS_NAME.indexOf(prefixes[i]) != -1) { │ │ │ │ │ - return prefixes[i]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Clones this style. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Style>} Clone of this style. │ │ │ │ │ - */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - var options = OpenLayers.Util.extend({}, this); │ │ │ │ │ - // clone rules │ │ │ │ │ - if (this.rules) { │ │ │ │ │ - options.rules = []; │ │ │ │ │ - for (var i = 0, len = this.rules.length; i < len; ++i) { │ │ │ │ │ - options.rules.push(this.rules[i].clone()); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // clone context │ │ │ │ │ - options.context = this.context && OpenLayers.Util.extend({}, this.context); │ │ │ │ │ - //clone default style │ │ │ │ │ - var defaultStyle = OpenLayers.Util.extend({}, this.defaultStyle); │ │ │ │ │ - return new OpenLayers.Style(defaultStyle, options); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Style" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Function: createLiteral │ │ │ │ │ - * converts a style value holding a combination of PropertyName and Literal │ │ │ │ │ - * into a Literal, taking the property values from the passed features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * value - {String} value to parse. If this string contains a construct like │ │ │ │ │ - * "foo ${bar}", then "foo " will be taken as literal, and "${bar}" │ │ │ │ │ - * will be replaced by the value of the "bar" attribute of the passed │ │ │ │ │ - * feature. │ │ │ │ │ - * context - {Object} context to take attribute values from │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} optional feature to pass to │ │ │ │ │ - * <OpenLayers.String.format> for evaluating functions in the │ │ │ │ │ - * context. │ │ │ │ │ - * property - {String} optional, name of the property for which the literal is │ │ │ │ │ - * being created for evaluating functions in the context. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} the parsed value. In the example of the value parameter above, the │ │ │ │ │ - * result would be "foo valueOfBar", assuming that the passed feature has an │ │ │ │ │ - * attribute named "bar" with the value "valueOfBar". │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Style.createLiteral = function(value, context, feature, property) { │ │ │ │ │ - if (typeof value == "string" && value.indexOf("${") != -1) { │ │ │ │ │ - value = OpenLayers.String.format(value, context, [feature, property]); │ │ │ │ │ - value = (isNaN(value) || !value) ? value : parseFloat(value); │ │ │ │ │ - } │ │ │ │ │ - return value; │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Style.SYMBOLIZER_PREFIXES │ │ │ │ │ - * {Array} prefixes of the sld symbolizers. These are the │ │ │ │ │ - * same as the main geometry types │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Style.SYMBOLIZER_PREFIXES = ['Point', 'Line', 'Polygon', 'Text', │ │ │ │ │ - 'Raster' │ │ │ │ │ -]; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/StyleMap.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Style.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.StyleMap │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.StyleMap = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: styles │ │ │ │ │ - * {Object} Hash of {<OpenLayers.Style>}, keyed by names of well known │ │ │ │ │ - * rendering intents (e.g. "default", "temporary", "select", "delete"). │ │ │ │ │ - */ │ │ │ │ │ - styles: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: extendDefault │ │ │ │ │ - * {Boolean} if true, every render intent will extend the symbolizers │ │ │ │ │ - * specified for the "default" intent at rendering time. Otherwise, every │ │ │ │ │ - * rendering intent will be treated as a completely independent style. │ │ │ │ │ - */ │ │ │ │ │ - extendDefault: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.StyleMap │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * style - {Object} Optional. Either a style hash, or a style object, or │ │ │ │ │ - * a hash of style objects (style hashes) keyed by rendering │ │ │ │ │ - * intent. If just one style hash or style object is passed, │ │ │ │ │ - * this will be used for all known render intents (default, │ │ │ │ │ - * select, temporary) │ │ │ │ │ - * options - {Object} optional hash of additional options for this │ │ │ │ │ - * instance │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(style, options) { │ │ │ │ │ - this.styles = { │ │ │ │ │ - "default": new OpenLayers.Style( │ │ │ │ │ - OpenLayers.Feature.Vector.style["default"]), │ │ │ │ │ - "select": new OpenLayers.Style( │ │ │ │ │ - OpenLayers.Feature.Vector.style["select"]), │ │ │ │ │ - "temporary": new OpenLayers.Style( │ │ │ │ │ - OpenLayers.Feature.Vector.style["temporary"]), │ │ │ │ │ - "delete": new OpenLayers.Style( │ │ │ │ │ - OpenLayers.Feature.Vector.style["delete"]) │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // take whatever the user passed as style parameter and convert it │ │ │ │ │ - // into parts of stylemap. │ │ │ │ │ - if (style instanceof OpenLayers.Style) { │ │ │ │ │ - // user passed a style object │ │ │ │ │ - this.styles["default"] = style; │ │ │ │ │ - this.styles["select"] = style; │ │ │ │ │ - this.styles["temporary"] = style; │ │ │ │ │ - this.styles["delete"] = style; │ │ │ │ │ - } else if (typeof style == "object") { │ │ │ │ │ - for (var key in style) { │ │ │ │ │ - if (style[key] instanceof OpenLayers.Style) { │ │ │ │ │ - // user passed a hash of style objects │ │ │ │ │ - this.styles[key] = style[key]; │ │ │ │ │ - } else if (typeof style[key] == "object") { │ │ │ │ │ - // user passsed a hash of style hashes │ │ │ │ │ - this.styles[key] = new OpenLayers.Style(style[key]); │ │ │ │ │ - } else { │ │ │ │ │ - // user passed a style hash (i.e. symbolizer) │ │ │ │ │ - this.styles["default"] = new OpenLayers.Style(style); │ │ │ │ │ - this.styles["select"] = new OpenLayers.Style(style); │ │ │ │ │ - this.styles["temporary"] = new OpenLayers.Style(style); │ │ │ │ │ - this.styles["delete"] = new OpenLayers.Style(style); │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var key in this.styles) { │ │ │ │ │ - this.styles[key].destroy(); │ │ │ │ │ - } │ │ │ │ │ - this.styles = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createSymbolizer │ │ │ │ │ - * Creates the symbolizer for a feature for a render intent. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature>} The feature to evaluate the rules │ │ │ │ │ - * of the intended style against. │ │ │ │ │ - * intent - {String} The intent determines the symbolizer that will be │ │ │ │ │ - * used to draw the feature. Well known intents are "default" │ │ │ │ │ - * (for just drawing the features), "select" (for selected │ │ │ │ │ - * features) and "temporary" (for drawing features). │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} symbolizer hash │ │ │ │ │ - */ │ │ │ │ │ - createSymbolizer: function(feature, intent) { │ │ │ │ │ - if (!feature) { │ │ │ │ │ - feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ - } │ │ │ │ │ - if (!this.styles[intent]) { │ │ │ │ │ - intent = "default"; │ │ │ │ │ - } │ │ │ │ │ - feature.renderIntent = intent; │ │ │ │ │ - var defaultSymbolizer = {}; │ │ │ │ │ - if (this.extendDefault && intent != "default") { │ │ │ │ │ - defaultSymbolizer = this.styles["default"].createSymbolizer(feature); │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Util.extend(defaultSymbolizer, │ │ │ │ │ - this.styles[intent].createSymbolizer(feature)); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addUniqueValueRules │ │ │ │ │ - * Convenience method to create comparison rules for unique values of a │ │ │ │ │ - * property. The rules will be added to the style object for a specified │ │ │ │ │ - * rendering intent. This method is a shortcut for creating something like │ │ │ │ │ - * the "unique value legends" familiar from well known desktop GIS systems │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * renderIntent - {String} rendering intent to add the rules to │ │ │ │ │ - * property - {String} values of feature attributes to create the │ │ │ │ │ - * rules for │ │ │ │ │ - * symbolizers - {Object} Hash of symbolizers, keyed by the desired │ │ │ │ │ - * property values │ │ │ │ │ - * context - {Object} An optional object with properties that │ │ │ │ │ - * symbolizers' property values should be evaluated │ │ │ │ │ - * against. If no context is specified, feature.attributes │ │ │ │ │ - * will be used │ │ │ │ │ - */ │ │ │ │ │ - addUniqueValueRules: function(renderIntent, property, symbolizers, context) { │ │ │ │ │ - var rules = []; │ │ │ │ │ - for (var value in symbolizers) { │ │ │ │ │ - rules.push(new OpenLayers.Rule({ │ │ │ │ │ - symbolizer: symbolizers[value], │ │ │ │ │ - context: context, │ │ │ │ │ - filter: new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.EQUAL_TO, │ │ │ │ │ - property: property, │ │ │ │ │ - value: value │ │ │ │ │ - }) │ │ │ │ │ - })); │ │ │ │ │ - } │ │ │ │ │ - this.styles[renderIntent].addRules(rules); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.StyleMap" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Vector.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer.js │ │ │ │ │ - * @requires OpenLayers/Renderer.js │ │ │ │ │ - * @requires OpenLayers/StyleMap.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Console.js │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.Vector │ │ │ │ │ - * Instances of OpenLayers.Layer.Vector are used to render vector data from │ │ │ │ │ - * a variety of sources. Create a new vector layer with the │ │ │ │ │ - * <OpenLayers.Layer.Vector> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.Vector = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * layer.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Listeners will be called with a reference to an event object. The │ │ │ │ │ - * properties of this event depends on exactly what happened. │ │ │ │ │ - * │ │ │ │ │ - * All event objects have at least the following properties: │ │ │ │ │ - * object - {Object} A reference to layer.events.object. │ │ │ │ │ - * element - {DOMElement} A reference to layer.events.element. │ │ │ │ │ - * │ │ │ │ │ - * Supported map event types (in addition to those from <OpenLayers.Layer.events>): │ │ │ │ │ - * beforefeatureadded - Triggered before a feature is added. Listeners │ │ │ │ │ - * will receive an object with a *feature* property referencing the │ │ │ │ │ - * feature to be added. To stop the feature from being added, a │ │ │ │ │ - * listener should return false. │ │ │ │ │ - * beforefeaturesadded - Triggered before an array of features is added. │ │ │ │ │ - * Listeners will receive an object with a *features* property │ │ │ │ │ - * referencing the feature to be added. To stop the features from │ │ │ │ │ - * being added, a listener should return false. │ │ │ │ │ - * featureadded - Triggered after a feature is added. The event │ │ │ │ │ - * object passed to listeners will have a *feature* property with a │ │ │ │ │ - * reference to the added feature. │ │ │ │ │ - * featuresadded - Triggered after features are added. The event │ │ │ │ │ - * object passed to listeners will have a *features* property with a │ │ │ │ │ - * reference to an array of added features. │ │ │ │ │ - * beforefeatureremoved - Triggered before a feature is removed. Listeners │ │ │ │ │ - * will receive an object with a *feature* property referencing the │ │ │ │ │ - * feature to be removed. │ │ │ │ │ - * beforefeaturesremoved - Triggered before multiple features are removed. │ │ │ │ │ - * Listeners will receive an object with a *features* property │ │ │ │ │ - * referencing the features to be removed. │ │ │ │ │ - * featureremoved - Triggerd after a feature is removed. The event │ │ │ │ │ - * object passed to listeners will have a *feature* property with a │ │ │ │ │ - * reference to the removed feature. │ │ │ │ │ - * featuresremoved - Triggered after features are removed. The event │ │ │ │ │ - * object passed to listeners will have a *features* property with a │ │ │ │ │ - * reference to an array of removed features. │ │ │ │ │ - * beforefeatureselected - Triggered before a feature is selected. Listeners │ │ │ │ │ - * will receive an object with a *feature* property referencing the │ │ │ │ │ - * feature to be selected. To stop the feature from being selectd, a │ │ │ │ │ - * listener should return false. │ │ │ │ │ - * featureselected - Triggered after a feature is selected. Listeners │ │ │ │ │ - * will receive an object with a *feature* property referencing the │ │ │ │ │ - * selected feature. │ │ │ │ │ - * featureunselected - Triggered after a feature is unselected. │ │ │ │ │ - * Listeners will receive an object with a *feature* property │ │ │ │ │ - * referencing the unselected feature. │ │ │ │ │ - * beforefeaturemodified - Triggered when a feature is selected to │ │ │ │ │ - * be modified. Listeners will receive an object with a *feature* │ │ │ │ │ - * property referencing the selected feature. │ │ │ │ │ - * featuremodified - Triggered when a feature has been modified. │ │ │ │ │ - * Listeners will receive an object with a *feature* property referencing │ │ │ │ │ - * the modified feature. │ │ │ │ │ - * afterfeaturemodified - Triggered when a feature is finished being modified. │ │ │ │ │ - * Listeners will receive an object with a *feature* property referencing │ │ │ │ │ - * the modified feature. │ │ │ │ │ - * vertexmodified - Triggered when a vertex within any feature geometry │ │ │ │ │ - * has been modified. Listeners will receive an object with a │ │ │ │ │ - * *feature* property referencing the modified feature, a *vertex* │ │ │ │ │ - * property referencing the vertex modified (always a point geometry), │ │ │ │ │ - * and a *pixel* property referencing the pixel location of the │ │ │ │ │ - * modification. │ │ │ │ │ - * vertexremoved - Triggered when a vertex within any feature geometry │ │ │ │ │ - * has been deleted. Listeners will receive an object with a │ │ │ │ │ - * *feature* property referencing the modified feature, a *vertex* │ │ │ │ │ - * property referencing the vertex modified (always a point geometry), │ │ │ │ │ - * and a *pixel* property referencing the pixel location of the │ │ │ │ │ - * removal. │ │ │ │ │ - * sketchstarted - Triggered when a feature sketch bound for this layer │ │ │ │ │ - * is started. Listeners will receive an object with a *feature* │ │ │ │ │ - * property referencing the new sketch feature and a *vertex* property │ │ │ │ │ - * referencing the creation point. │ │ │ │ │ - * sketchmodified - Triggered when a feature sketch bound for this layer │ │ │ │ │ - * is modified. Listeners will receive an object with a *vertex* │ │ │ │ │ - * property referencing the modified vertex and a *feature* property │ │ │ │ │ - * referencing the sketch feature. │ │ │ │ │ - * sketchcomplete - Triggered when a feature sketch bound for this layer │ │ │ │ │ - * is complete. Listeners will receive an object with a *feature* │ │ │ │ │ - * property referencing the sketch feature. By returning false, a │ │ │ │ │ - * listener can stop the sketch feature from being added to the layer. │ │ │ │ │ - * refresh - Triggered when something wants a strategy to ask the protocol │ │ │ │ │ - * for a new set of features. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} The layer is a base layer. Default is false. Set this property │ │ │ │ │ - * in the layer options. │ │ │ │ │ - */ │ │ │ │ │ - isBaseLayer: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isFixed │ │ │ │ │ - * {Boolean} Whether the layer remains in one place while dragging the │ │ │ │ │ - * map. Note that setting this to true will move the layer to the bottom │ │ │ │ │ - * of the layer stack. │ │ │ │ │ - */ │ │ │ │ │ - isFixed: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: features │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ - */ │ │ │ │ │ - features: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: filter │ │ │ │ │ - * {<OpenLayers.Filter>} The filter set in this layer, │ │ │ │ │ - * a strategy launching read requests can combined │ │ │ │ │ - * this filter with its own filter. │ │ │ │ │ - */ │ │ │ │ │ - filter: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: selectedFeatures │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ - */ │ │ │ │ │ - selectedFeatures: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: unrenderedFeatures │ │ │ │ │ - * {Object} hash of features, keyed by feature.id, that the renderer │ │ │ │ │ - * failed to draw │ │ │ │ │ - */ │ │ │ │ │ - unrenderedFeatures: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: reportError │ │ │ │ │ - * {Boolean} report friendly error message when loading of renderer │ │ │ │ │ - * fails. │ │ │ │ │ - */ │ │ │ │ │ - reportError: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: style │ │ │ │ │ - * {Object} Default style for the layer │ │ │ │ │ - */ │ │ │ │ │ - style: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: styleMap │ │ │ │ │ - * {<OpenLayers.StyleMap>} │ │ │ │ │ - */ │ │ │ │ │ - styleMap: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: strategies │ │ │ │ │ - * {Array(<OpenLayers.Strategy>})} Optional list of strategies for the layer. │ │ │ │ │ - */ │ │ │ │ │ - strategies: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: protocol │ │ │ │ │ - * {<OpenLayers.Protocol>} Optional protocol for the layer. │ │ │ │ │ - */ │ │ │ │ │ - protocol: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: renderers │ │ │ │ │ - * {Array(String)} List of supported Renderer classes. Add to this list to │ │ │ │ │ - * add support for additional renderers. This list is ordered: │ │ │ │ │ - * the first renderer which returns true for the 'supported()' │ │ │ │ │ - * method will be used, if not defined in the 'renderer' option. │ │ │ │ │ - */ │ │ │ │ │ - renderers: ['SVG', 'VML', 'Canvas'], │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: renderer │ │ │ │ │ - * {<OpenLayers.Renderer>} │ │ │ │ │ - */ │ │ │ │ │ - renderer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: rendererOptions │ │ │ │ │ - * {Object} Options for the renderer. See {<OpenLayers.Renderer>} for │ │ │ │ │ - * supported options. │ │ │ │ │ - */ │ │ │ │ │ - rendererOptions: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geometryType │ │ │ │ │ - * {String} geometryType allows you to limit the types of geometries this │ │ │ │ │ - * layer supports. This should be set to something like │ │ │ │ │ - * "OpenLayers.Geometry.Point" to limit types. │ │ │ │ │ - */ │ │ │ │ │ - geometryType: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: drawn │ │ │ │ │ - * {Boolean} Whether the Vector Layer features have been drawn yet. │ │ │ │ │ - */ │ │ │ │ │ - drawn: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: ratio │ │ │ │ │ - * {Float} This specifies the ratio of the size of the visiblity of the Vector Layer features to the size of the map. │ │ │ │ │ - */ │ │ │ │ │ - ratio: 1, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Vector │ │ │ │ │ - * Create a new vector layer │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} A name for the layer │ │ │ │ │ - * options - {Object} Optional object with non-default properties to set on │ │ │ │ │ - * the layer. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} A new vector layer │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - // allow user-set renderer, otherwise assign one │ │ │ │ │ - if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ - this.assignRenderer(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // if no valid renderer found, display error │ │ │ │ │ - if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ - this.renderer = null; │ │ │ │ │ - this.displayError(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (!this.styleMap) { │ │ │ │ │ - this.styleMap = new OpenLayers.StyleMap(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.features = []; │ │ │ │ │ - this.selectedFeatures = []; │ │ │ │ │ - this.unrenderedFeatures = {}; │ │ │ │ │ - │ │ │ │ │ - // Allow for custom layer behavior │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - for (var i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - this.strategies[i].setLayer(this); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Destroy this layer │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoDestroy) { │ │ │ │ │ - strategy.destroy(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.strategies = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.protocol) { │ │ │ │ │ - if (this.protocol.autoDestroy) { │ │ │ │ │ - this.protocol.destroy(); │ │ │ │ │ - } │ │ │ │ │ - this.protocol = null; │ │ │ │ │ - } │ │ │ │ │ - this.destroyFeatures(); │ │ │ │ │ - this.features = null; │ │ │ │ │ - this.selectedFeatures = null; │ │ │ │ │ - this.unrenderedFeatures = null; │ │ │ │ │ - if (this.renderer) { │ │ │ │ │ - this.renderer.destroy(); │ │ │ │ │ - } │ │ │ │ │ - this.renderer = null; │ │ │ │ │ - this.geometryType = null; │ │ │ │ │ - this.drawn = null; │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer. │ │ │ │ │ - * │ │ │ │ │ - * Note: Features of the layer are also cloned. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} An exact clone of this layer │ │ │ │ │ - */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Vector(this.name, this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ - │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - var features = this.features; │ │ │ │ │ - var len = features.length; │ │ │ │ │ - var clonedFeatures = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - clonedFeatures[i] = features[i].clone(); │ │ │ │ │ - } │ │ │ │ │ - obj.features = clonedFeatures; │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: refresh │ │ │ │ │ - * Ask the layer to request features again and redraw them. Triggers │ │ │ │ │ - * the refresh event if the layer is in range and visible. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} Optional object with properties for any listener of │ │ │ │ │ - * the refresh event. │ │ │ │ │ - */ │ │ │ │ │ - refresh: function(obj) { │ │ │ │ │ - if (this.calculateInRange() && this.visibility) { │ │ │ │ │ - this.events.triggerEvent("refresh", obj); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: assignRenderer │ │ │ │ │ - * Iterates through the available renderer implementations and selects │ │ │ │ │ - * and assigns the first one whose "supported()" function returns true. │ │ │ │ │ - */ │ │ │ │ │ - assignRenderer: function() { │ │ │ │ │ - for (var i = 0, len = this.renderers.length; i < len; i++) { │ │ │ │ │ - var rendererClass = this.renderers[i]; │ │ │ │ │ - var renderer = (typeof rendererClass == "function") ? │ │ │ │ │ - rendererClass : │ │ │ │ │ - OpenLayers.Renderer[rendererClass]; │ │ │ │ │ - if (renderer && renderer.prototype.supported()) { │ │ │ │ │ - this.renderer = new renderer(this.div, this.rendererOptions); │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: displayError │ │ │ │ │ - * Let the user know their browser isn't supported. │ │ │ │ │ - */ │ │ │ │ │ - displayError: function() { │ │ │ │ │ - if (this.reportError) { │ │ │ │ │ - OpenLayers.Console.userError(OpenLayers.i18n("browserNotSupported", { │ │ │ │ │ - renderers: this.renderers.join('\n') │ │ │ │ │ - })); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * The layer has been added to the map. │ │ │ │ │ - * │ │ │ │ │ - * If there is no renderer set, the layer can't be used. Remove it. │ │ │ │ │ - * Otherwise, give the renderer a reference to the map and set its size. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - if (!this.renderer) { │ │ │ │ │ - this.map.removeLayer(this); │ │ │ │ │ - } else { │ │ │ │ │ - this.renderer.map = this.map; │ │ │ │ │ - │ │ │ │ │ - var newSize = this.map.getSize(); │ │ │ │ │ - newSize.w = newSize.w * this.ratio; │ │ │ │ │ - newSize.h = newSize.h * this.ratio; │ │ │ │ │ - this.renderer.setSize(newSize); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: afterAdd │ │ │ │ │ - * Called at the end of the map.addLayer sequence. At this point, the map │ │ │ │ │ - * will have a base layer. Any autoActivate strategies will be │ │ │ │ │ - * activated here. │ │ │ │ │ - */ │ │ │ │ │ - afterAdd: function() { │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoActivate) { │ │ │ │ │ - strategy.activate(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeMap │ │ │ │ │ - * The layer has been removed from the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - this.drawn = false; │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoActivate) { │ │ │ │ │ - strategy.deactivate(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onMapResize │ │ │ │ │ - * Notify the renderer of the change in size. │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - OpenLayers.Layer.prototype.onMapResize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - var newSize = this.map.getSize(); │ │ │ │ │ - newSize.w = newSize.w * this.ratio; │ │ │ │ │ - newSize.h = newSize.h * this.ratio; │ │ │ │ │ - this.renderer.setSize(newSize); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * Reset the vector layer's div so that it once again is lined up with │ │ │ │ │ - * the map. Notify the renderer of the change of extent, and in the │ │ │ │ │ - * case of a change of zoom level (resolution), have the │ │ │ │ │ - * renderer redraw features. │ │ │ │ │ - * │ │ │ │ │ - * If the layer has not yet been drawn, cycle through the layer's │ │ │ │ │ - * features and draw each one. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * zoomChanged - {Boolean} │ │ │ │ │ - * dragging - {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - var coordSysUnchanged = true; │ │ │ │ │ - if (!dragging) { │ │ │ │ │ - this.renderer.root.style.visibility = 'hidden'; │ │ │ │ │ - │ │ │ │ │ - var viewSize = this.map.getSize(), │ │ │ │ │ - viewWidth = viewSize.w, │ │ │ │ │ - viewHeight = viewSize.h, │ │ │ │ │ - offsetLeft = (viewWidth / 2 * this.ratio) - viewWidth / 2, │ │ │ │ │ - offsetTop = (viewHeight / 2 * this.ratio) - viewHeight / 2; │ │ │ │ │ - offsetLeft += this.map.layerContainerOriginPx.x; │ │ │ │ │ - offsetLeft = -Math.round(offsetLeft); │ │ │ │ │ - offsetTop += this.map.layerContainerOriginPx.y; │ │ │ │ │ - offsetTop = -Math.round(offsetTop); │ │ │ │ │ - │ │ │ │ │ - this.div.style.left = offsetLeft + 'px'; │ │ │ │ │ - this.div.style.top = offsetTop + 'px'; │ │ │ │ │ - │ │ │ │ │ - var extent = this.map.getExtent().scale(this.ratio); │ │ │ │ │ - coordSysUnchanged = this.renderer.setExtent(extent, zoomChanged); │ │ │ │ │ - │ │ │ │ │ - this.renderer.root.style.visibility = 'visible'; │ │ │ │ │ - │ │ │ │ │ - // Force a reflow on gecko based browsers to prevent jump/flicker. │ │ │ │ │ - // This seems to happen on only certain configurations; it was originally │ │ │ │ │ - // noticed in FF 2.0 and Linux. │ │ │ │ │ - if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ - this.div.scrollLeft = this.div.scrollLeft; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (!zoomChanged && coordSysUnchanged) { │ │ │ │ │ - for (var i in this.unrenderedFeatures) { │ │ │ │ │ - var feature = this.unrenderedFeatures[i]; │ │ │ │ │ - this.drawFeature(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!this.drawn || zoomChanged || !coordSysUnchanged) { │ │ │ │ │ - this.drawn = true; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0, len = this.features.length; i < len; i++) { │ │ │ │ │ - this.renderer.locked = (i !== (len - 1)); │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - this.drawFeature(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: display │ │ │ │ │ - * Hide or show the Layer │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * display - {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - display: function(display) { │ │ │ │ │ - OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ - // we need to set the display style of the root in case it is attached │ │ │ │ │ - // to a foreign layer │ │ │ │ │ - var currentDisplay = this.div.style.display; │ │ │ │ │ - if (currentDisplay != this.renderer.root.style.display) { │ │ │ │ │ - this.renderer.root.style.display = currentDisplay; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: addFeatures │ │ │ │ │ - * Add Features to the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - addFeatures: function(features, options) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ - if (notify) { │ │ │ │ │ - var event = { │ │ │ │ │ - features: features │ │ │ │ │ - }; │ │ │ │ │ - var ret = this.events.triggerEvent("beforefeaturesadded", event); │ │ │ │ │ - if (ret === false) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - features = event.features; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // Track successfully added features for featuresadded event, since │ │ │ │ │ - // beforefeatureadded can veto single features. │ │ │ │ │ - var featuresAdded = []; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - if (i != (features.length - 1)) { │ │ │ │ │ - this.renderer.locked = true; │ │ │ │ │ - } else { │ │ │ │ │ - this.renderer.locked = false; │ │ │ │ │ - } │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - │ │ │ │ │ - if (this.geometryType && │ │ │ │ │ - !(feature.geometry instanceof this.geometryType)) { │ │ │ │ │ - throw new TypeError('addFeatures: component should be an ' + │ │ │ │ │ - this.geometryType.prototype.CLASS_NAME); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //give feature reference to its layer │ │ │ │ │ - feature.layer = this; │ │ │ │ │ - │ │ │ │ │ - if (!feature.style && this.style) { │ │ │ │ │ - feature.style = OpenLayers.Util.extend({}, this.style); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (notify) { │ │ │ │ │ - if (this.events.triggerEvent("beforefeatureadded", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) === false) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - this.preFeatureInsert(feature); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - featuresAdded.push(feature); │ │ │ │ │ - this.features.push(feature); │ │ │ │ │ - this.drawFeature(feature); │ │ │ │ │ - │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureadded", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - this.onFeatureInsert(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresadded", { │ │ │ │ │ - features: featuresAdded │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: removeFeatures │ │ │ │ │ - * Remove features from the layer. This erases any drawn features and │ │ │ │ │ - * removes them from the layer's control. The beforefeatureremoved │ │ │ │ │ - * and featureremoved events will be triggered for each feature. The │ │ │ │ │ - * featuresremoved event will be triggered after all features have │ │ │ │ │ - * been removed. To supress event triggering, use the silent option. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} List of features to be │ │ │ │ │ - * removed. │ │ │ │ │ - * options - {Object} Optional properties for changing behavior of the │ │ │ │ │ - * removal. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * silent - {Boolean} Supress event triggering. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - removeFeatures: function(features, options) { │ │ │ │ │ - if (!features || features.length === 0) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (features === this.features) { │ │ │ │ │ - return this.removeAllFeatures(options); │ │ │ │ │ - } │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ - if (features === this.selectedFeatures) { │ │ │ │ │ - features = features.slice(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ - │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent( │ │ │ │ │ - "beforefeaturesremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - // We remain locked so long as we're not at 0 │ │ │ │ │ - // and the 'next' feature has a geometry. We do the geometry check │ │ │ │ │ - // because if all the features after the current one are 'null', we │ │ │ │ │ - // won't call eraseGeometry, so we break the 'renderer functions │ │ │ │ │ - // will always be called with locked=false *last*' rule. The end result │ │ │ │ │ - // is a possible gratiutious unlocking to save a loop through the rest │ │ │ │ │ - // of the list checking the remaining features every time. So long as │ │ │ │ │ - // null geoms are rare, this is probably okay. │ │ │ │ │ - if (i != 0 && features[i - 1].geometry) { │ │ │ │ │ - this.renderer.locked = true; │ │ │ │ │ - } else { │ │ │ │ │ - this.renderer.locked = false; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ - │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.features = OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ - // feature has no layer at this point │ │ │ │ │ - feature.layer = null; │ │ │ │ │ - │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - this.renderer.eraseFeatures(feature); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //in the case that this feature is one of the selected features, │ │ │ │ │ - // remove it from that array as well. │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.selectedFeatures, feature) != -1) { │ │ │ │ │ - OpenLayers.Util.removeItem(this.selectedFeatures, feature); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: removeAllFeatures │ │ │ │ │ - * Remove all features from the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional properties for changing behavior of the │ │ │ │ │ - * removal. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * silent - {Boolean} Supress event triggering. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - removeAllFeatures: function(options) { │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ - var features = this.features; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent( │ │ │ │ │ - "beforefeaturesremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - feature.layer = null; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.renderer.clear(); │ │ │ │ │ - this.features = []; │ │ │ │ │ - this.unrenderedFeatures = {}; │ │ │ │ │ - this.selectedFeatures = []; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroyFeatures │ │ │ │ │ - * Erase and destroy features on the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} An optional array of │ │ │ │ │ - * features to destroy. If not supplied, all features on the layer │ │ │ │ │ - * will be destroyed. │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - destroyFeatures: function(features, options) { │ │ │ │ │ - var all = (features == undefined); // evaluates to true if │ │ │ │ │ - // features is null │ │ │ │ │ - if (all) { │ │ │ │ │ - features = this.features; │ │ │ │ │ - } │ │ │ │ │ - if (features) { │ │ │ │ │ - this.removeFeatures(features, options); │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - features[i].destroy(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: drawFeature │ │ │ │ │ - * Draw (or redraw) a feature on the layer. If the optional style argument │ │ │ │ │ - * is included, this style will be used. If no style is included, the │ │ │ │ │ - * feature's style will be used. If the feature doesn't have a style, │ │ │ │ │ - * the layer's style will be used. │ │ │ │ │ - * │ │ │ │ │ - * This function is not designed to be used when adding features to │ │ │ │ │ - * the layer (use addFeatures instead). It is meant to be used when │ │ │ │ │ - * the style of a feature has changed, or in some other way needs to │ │ │ │ │ - * visually updated *after* it has already been added to a layer. You │ │ │ │ │ - * must add the feature to the layer for most layer-related events to │ │ │ │ │ - * happen. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * style - {String | Object} Named render intent or full symbolizer object. │ │ │ │ │ - */ │ │ │ │ │ - drawFeature: function(feature, style) { │ │ │ │ │ - // don't try to draw the feature with the renderer if the layer is not │ │ │ │ │ - // drawn itself │ │ │ │ │ - if (!this.drawn) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (typeof style != "object") { │ │ │ │ │ - if (!style && feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ - style = "delete"; │ │ │ │ │ - } │ │ │ │ │ - var renderIntent = style || feature.renderIntent; │ │ │ │ │ - style = feature.style || this.style; │ │ │ │ │ - if (!style) { │ │ │ │ │ - style = this.styleMap.createSymbolizer(feature, renderIntent); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var drawn = this.renderer.drawFeature(feature, style); │ │ │ │ │ - //TODO remove the check for null when we get rid of Renderer.SVG │ │ │ │ │ - if (drawn === false || drawn === null) { │ │ │ │ │ - this.unrenderedFeatures[feature.id] = feature; │ │ │ │ │ - } else { │ │ │ │ │ - delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: eraseFeatures │ │ │ │ │ - * Erase features from the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ - */ │ │ │ │ │ - eraseFeatures: function(features) { │ │ │ │ │ - this.renderer.eraseFeatures(features); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getFeatureFromEvent │ │ │ │ │ - * Given an event, return a feature if the event occurred over one. │ │ │ │ │ - * Otherwise, return null. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A feature if one was under the event. │ │ │ │ │ - */ │ │ │ │ │ - getFeatureFromEvent: function(evt) { │ │ │ │ │ - if (!this.renderer) { │ │ │ │ │ - throw new Error('getFeatureFromEvent called on layer with no ' + │ │ │ │ │ - 'renderer. This usually means you destroyed a ' + │ │ │ │ │ - 'layer, but not some handler which is associated ' + │ │ │ │ │ - 'with it.'); │ │ │ │ │ - } │ │ │ │ │ - var feature = null; │ │ │ │ │ - var featureId = this.renderer.getFeatureIdFromEvent(evt); │ │ │ │ │ - if (featureId) { │ │ │ │ │ - if (typeof featureId === "string") { │ │ │ │ │ - feature = this.getFeatureById(featureId); │ │ │ │ │ - } else { │ │ │ │ │ - feature = featureId; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return feature; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getFeatureBy │ │ │ │ │ - * Given a property value, return the feature if it exists in the features array │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * property - {String} │ │ │ │ │ - * value - {String} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A feature corresponding to the given │ │ │ │ │ - * property value or null if there is no such feature. │ │ │ │ │ - */ │ │ │ │ │ - getFeatureBy: function(property, value) { │ │ │ │ │ - //TBD - would it be more efficient to use a hash for this.features? │ │ │ │ │ - var feature = null; │ │ │ │ │ - for (var i = 0, len = this.features.length; i < len; ++i) { │ │ │ │ │ - if (this.features[i][property] == value) { │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return feature; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getFeatureById │ │ │ │ │ - * Given a feature id, return the feature if it exists in the features array │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A feature corresponding to the given │ │ │ │ │ - * featureId or null if there is no such feature. │ │ │ │ │ - */ │ │ │ │ │ - getFeatureById: function(featureId) { │ │ │ │ │ - return this.getFeatureBy('id', featureId); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getFeatureByFid │ │ │ │ │ - * Given a feature fid, return the feature if it exists in the features array │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureFid - {String} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A feature corresponding to the given │ │ │ │ │ - * featureFid or null if there is no such feature. │ │ │ │ │ - */ │ │ │ │ │ - getFeatureByFid: function(featureFid) { │ │ │ │ │ - return this.getFeatureBy('fid', featureFid); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getFeaturesByAttribute │ │ │ │ │ - * Returns an array of features that have the given attribute key set to the │ │ │ │ │ - * given value. Comparison of attribute values takes care of datatypes, e.g. │ │ │ │ │ - * the string '1234' is not equal to the number 1234. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * attrName - {String} │ │ │ │ │ - * attrValue - {Mixed} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * Array({<OpenLayers.Feature.Vector>}) An array of features that have the │ │ │ │ │ - * passed named attribute set to the given value. │ │ │ │ │ - */ │ │ │ │ │ - getFeaturesByAttribute: function(attrName, attrValue) { │ │ │ │ │ - var i, │ │ │ │ │ - feature, │ │ │ │ │ - len = this.features.length, │ │ │ │ │ - foundFeatures = []; │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - if (feature && feature.attributes) { │ │ │ │ │ - if (feature.attributes[attrName] === attrValue) { │ │ │ │ │ - foundFeatures.push(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return foundFeatures; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Unselect the selected features │ │ │ │ │ - * i.e. clears the featureSelection array │ │ │ │ │ - * change the style back │ │ │ │ │ - clearSelection: function() { │ │ │ │ │ - │ │ │ │ │ - var vectorLayer = this.map.vectorLayer; │ │ │ │ │ - for (var i = 0; i < this.map.featureSelection.length; i++) { │ │ │ │ │ - var featureSelection = this.map.featureSelection[i]; │ │ │ │ │ - vectorLayer.drawFeature(featureSelection, vectorLayer.style); │ │ │ │ │ - } │ │ │ │ │ - this.map.featureSelection = []; │ │ │ │ │ - }, │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: onFeatureInsert │ │ │ │ │ - * method called after a feature is inserted. │ │ │ │ │ - * Does nothing by default. Override this if you │ │ │ │ │ - * need to do something on feature updates. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - onFeatureInsert: function(feature) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: preFeatureInsert │ │ │ │ │ - * method called before a feature is inserted. │ │ │ │ │ - * Does nothing by default. Override this if you │ │ │ │ │ - * need to do something when features are first added to the │ │ │ │ │ - * layer, but before they are drawn, such as adjust the style. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - preFeatureInsert: function(feature) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getDataExtent │ │ │ │ │ - * Calculates the max extent which includes all of the features. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} or null if the layer has no features with │ │ │ │ │ - * geometries. │ │ │ │ │ - */ │ │ │ │ │ - getDataExtent: function() { │ │ │ │ │ - var maxExtent = null; │ │ │ │ │ - var features = this.features; │ │ │ │ │ - if (features && (features.length > 0)) { │ │ │ │ │ - var geometry = null; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - geometry = features[i].geometry; │ │ │ │ │ - if (geometry) { │ │ │ │ │ - if (maxExtent === null) { │ │ │ │ │ - maxExtent = new OpenLayers.Bounds(); │ │ │ │ │ - } │ │ │ │ │ - maxExtent.extend(geometry.getBounds()); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return maxExtent; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Vector" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/WMS.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.WMS │ │ │ │ │ - * Instances of OpenLayers.Layer.WMS are used to display data from OGC Web │ │ │ │ │ - * Mapping Services. Create a new WMS layer with the <OpenLayers.Layer.WMS> │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.WMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: DEFAULT_PARAMS │ │ │ │ │ - * {Object} Hashtable of default parameter key/value pairs │ │ │ │ │ - */ │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - service: "WMS", │ │ │ │ │ - version: "1.1.1", │ │ │ │ │ - request: "GetMap", │ │ │ │ │ - styles: "", │ │ │ │ │ - format: "image/jpeg" │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} Default is true for WMS layer │ │ │ │ │ - */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: encodeBBOX │ │ │ │ │ - * {Boolean} Should the BBOX commas be encoded? The WMS spec says 'no', │ │ │ │ │ - * but some services want it that way. Default false. │ │ │ │ │ - */ │ │ │ │ │ - encodeBBOX: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: noMagic │ │ │ │ │ - * {Boolean} If true, the image format will not be automagicaly switched │ │ │ │ │ - * from image/jpeg to image/png or image/gif when using │ │ │ │ │ - * TRANSPARENT=TRUE. Also isBaseLayer will not changed by the │ │ │ │ │ - * constructor. Default false. │ │ │ │ │ - */ │ │ │ │ │ - noMagic: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: yx │ │ │ │ │ - * {Object} Keys in this object are EPSG codes for which the axis order │ │ │ │ │ - * is to be reversed (yx instead of xy, LatLon instead of LonLat), with │ │ │ │ │ - * true as value. This is only relevant for WMS versions >= 1.3.0, and │ │ │ │ │ - * only if yx is not set in <OpenLayers.Projection.defaults> for the │ │ │ │ │ - * used projection. │ │ │ │ │ - */ │ │ │ │ │ - yx: {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.WMS │ │ │ │ │ - * Create a new WMS layer object │ │ │ │ │ - * │ │ │ │ │ - * Examples: │ │ │ │ │ - * │ │ │ │ │ - * The code below creates a simple WMS layer using the image/jpeg format. │ │ │ │ │ - * (code) │ │ │ │ │ - * var wms = new OpenLayers.Layer.WMS("NASA Global Mosaic", │ │ │ │ │ - * "http://wms.jpl.nasa.gov/wms.cgi", │ │ │ │ │ - * {layers: "modis,global_mosaic"}); │ │ │ │ │ - * (end) │ │ │ │ │ - * Note the 3rd argument (params). Properties added to this object will be │ │ │ │ │ - * added to the WMS GetMap requests used for this layer's tiles. The only │ │ │ │ │ - * mandatory parameter is "layers". Other common WMS params include │ │ │ │ │ - * "transparent", "styles" and "format". Note that the "srs" param will │ │ │ │ │ - * always be ignored. Instead, it will be derived from the baseLayer's or │ │ │ │ │ - * map's projection. │ │ │ │ │ - * │ │ │ │ │ - * The code below creates a transparent WMS layer with additional options. │ │ │ │ │ - * (code) │ │ │ │ │ - * var wms = new OpenLayers.Layer.WMS("NASA Global Mosaic", │ │ │ │ │ - * "http://wms.jpl.nasa.gov/wms.cgi", │ │ │ │ │ - * { │ │ │ │ │ - * layers: "modis,global_mosaic", │ │ │ │ │ - * transparent: true │ │ │ │ │ - * }, { │ │ │ │ │ - * opacity: 0.5, │ │ │ │ │ - * singleTile: true │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * Note that by default, a WMS layer is configured as baseLayer. Setting │ │ │ │ │ - * the "transparent" param to true will apply some magic (see <noMagic>). │ │ │ │ │ - * The default image format changes from image/jpeg to image/png, and the │ │ │ │ │ - * layer is not configured as baseLayer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} A name for the layer │ │ │ │ │ - * url - {String} Base url for the WMS │ │ │ │ │ - * (e.g. http://wms.jpl.nasa.gov/wms.cgi) │ │ │ │ │ - * params - {Object} An object with key/value pairs representing the │ │ │ │ │ - * GetMap query string parameters and parameter values. │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer. │ │ │ │ │ - * These options include all properties listed above, plus the ones │ │ │ │ │ - * inherited from superclasses. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - //uppercase params │ │ │ │ │ - params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ - if (parseFloat(params.VERSION) >= 1.3 && !params.EXCEPTIONS) { │ │ │ │ │ - params.EXCEPTIONS = "INIMAGE"; │ │ │ │ │ - } │ │ │ │ │ - newArguments.push(name, url, params, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, │ │ │ │ │ - OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS) │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - //layer is transparent │ │ │ │ │ - if (!this.noMagic && this.params.TRANSPARENT && │ │ │ │ │ - this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ - │ │ │ │ │ - // unless explicitly set in options, make layer an overlay │ │ │ │ │ - if ((options == null) || (!options.isBaseLayer)) { │ │ │ │ │ - this.isBaseLayer = false; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // jpegs can never be transparent, so intelligently switch the │ │ │ │ │ - // format, depending on the browser's capabilities │ │ │ │ │ - if (this.params.FORMAT == "image/jpeg") { │ │ │ │ │ - this.params.FORMAT = OpenLayers.Util.alphaHack() ? "image/gif" : │ │ │ │ │ - "image/png"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.WMS>} An exact clone of this layer │ │ │ │ │ - */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.WMS(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.params, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: reverseAxisOrder │ │ │ │ │ - * Returns true if the axis order is reversed for the WMS version and │ │ │ │ │ - * projection of the layer. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true if the axis order is reversed, false otherwise. │ │ │ │ │ - */ │ │ │ │ │ - reverseAxisOrder: function() { │ │ │ │ │ - var projCode = this.projection.getCode(); │ │ │ │ │ - return parseFloat(this.params.VERSION) >= 1.3 && │ │ │ │ │ - !!(this.yx[projCode] || (OpenLayers.Projection.defaults[projCode] && │ │ │ │ │ - OpenLayers.Projection.defaults[projCode].yx)); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * Return a GetMap query string for this layer │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox for the │ │ │ │ │ - * request. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as │ │ │ │ │ - * parameters. │ │ │ │ │ - */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - │ │ │ │ │ - var imageSize = this.getImageSize(); │ │ │ │ │ - var newParams = {}; │ │ │ │ │ - // WMS 1.3 introduced axis order │ │ │ │ │ - var reverseAxisOrder = this.reverseAxisOrder(); │ │ │ │ │ - newParams.BBOX = this.encodeBBOX ? │ │ │ │ │ - bounds.toBBOX(null, reverseAxisOrder) : │ │ │ │ │ - bounds.toArray(reverseAxisOrder); │ │ │ │ │ - newParams.WIDTH = imageSize.w; │ │ │ │ │ - newParams.HEIGHT = imageSize.h; │ │ │ │ │ - var requestString = this.getFullRequestString(newParams); │ │ │ │ │ - return requestString; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: mergeNewParams │ │ │ │ │ - * Catch changeParams and uppercase the new params to be merged in │ │ │ │ │ - * before calling changeParams on the super class. │ │ │ │ │ - * │ │ │ │ │ - * Once params have been changed, the tiles will be reloaded with │ │ │ │ │ - * the new parameters. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newParams - {Object} Hashtable of new params to use │ │ │ │ │ - */ │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ - var newArguments = [upperParams]; │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, │ │ │ │ │ - newArguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getFullRequestString │ │ │ │ │ - * Combine the layer's url with its params and these newParams. │ │ │ │ │ - * │ │ │ │ │ - * Add the SRS parameter from projection -- this is probably │ │ │ │ │ - * more eloquently done via a setProjection() method, but this │ │ │ │ │ - * works for now and always. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newParams - {Object} │ │ │ │ │ - * altUrl - {String} Use this as the url instead of the layer's url │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ - var mapProjection = this.map.getProjectionObject(); │ │ │ │ │ - var projectionCode = this.projection && this.projection.equals(mapProjection) ? │ │ │ │ │ - this.projection.getCode() : │ │ │ │ │ - mapProjection.getCode(); │ │ │ │ │ - var value = (projectionCode == "none") ? null : projectionCode; │ │ │ │ │ - if (parseFloat(this.params.VERSION) >= 1.3) { │ │ │ │ │ - this.params.CRS = value; │ │ │ │ │ - } else { │ │ │ │ │ - this.params.SRS = value; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (typeof this.params.TRANSPARENT == "boolean") { │ │ │ │ │ - newParams.TRANSPARENT = this.params.TRANSPARENT ? "TRUE" : "FALSE"; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.getFullRequestString.apply( │ │ │ │ │ - this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.WMS" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format │ │ │ │ │ - * Base class for format reading/writing a variety of formats. Subclasses │ │ │ │ │ - * of OpenLayers.Format are expected to have read and write methods. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: options │ │ │ │ │ - * {Object} A reference to options passed to the constructor. │ │ │ │ │ - */ │ │ │ │ │ - options: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: externalProjection │ │ │ │ │ - * {<OpenLayers.Projection>} When passed a externalProjection and │ │ │ │ │ - * internalProjection, the format will reproject the geometries it │ │ │ │ │ - * reads or writes. The externalProjection is the projection used by │ │ │ │ │ - * the content which is passed into read or which comes out of write. │ │ │ │ │ - * In order to reproject, a projection transformation function for the │ │ │ │ │ - * specified projections must be available. This support may be │ │ │ │ │ - * provided via proj4js or via a custom transformation function. See │ │ │ │ │ - * {<OpenLayers.Projection.addTransform>} for more information on │ │ │ │ │ - * custom transformations. │ │ │ │ │ - */ │ │ │ │ │ - externalProjection: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: internalProjection │ │ │ │ │ - * {<OpenLayers.Projection>} When passed a externalProjection and │ │ │ │ │ - * internalProjection, the format will reproject the geometries it │ │ │ │ │ - * reads or writes. The internalProjection is the projection used by │ │ │ │ │ - * the geometries which are returned by read or which are passed into │ │ │ │ │ - * write. In order to reproject, a projection transformation function │ │ │ │ │ - * for the specified projections must be available. This support may be │ │ │ │ │ - * provided via proj4js or via a custom transformation function. See │ │ │ │ │ - * {<OpenLayers.Projection.addTransform>} for more information on │ │ │ │ │ - * custom transformations. │ │ │ │ │ - */ │ │ │ │ │ - internalProjection: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: data │ │ │ │ │ - * {Object} When <keepData> is true, this is the parsed string sent to │ │ │ │ │ - * <read>. │ │ │ │ │ - */ │ │ │ │ │ - data: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: keepData │ │ │ │ │ - * {Object} Maintain a reference (<data>) to the most recently read data. │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - keepData: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format │ │ │ │ │ - * Instances of this class are not useful. See one of the subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object with properties to set on the │ │ │ │ │ - * format │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * keepData - {Boolean} If true, upon <read>, the data property will be │ │ │ │ │ - * set to the parsed object (e.g. the json or xml object). │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * An instance of OpenLayers.Format │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.options = options; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: read │ │ │ │ │ - * Read data from a string, and return an object whose type depends on the │ │ │ │ │ - * subclass. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {string} Data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * Depends on the subclass │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - throw new Error('Read not implemented.'); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: write │ │ │ │ │ - * Accept an object, and return a string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * object - {Object} Object to be serialized │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string representation of the object. │ │ │ │ │ - */ │ │ │ │ │ - write: function(object) { │ │ │ │ │ - throw new Error('Write not implemented.'); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/JSON.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Note: │ │ │ │ │ - * This work draws heavily from the public domain JSON serializer/deserializer │ │ │ │ │ - * at http://www.json.org/json.js. Rewritten so that it doesn't modify │ │ │ │ │ - * basic data prototypes. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.JSON │ │ │ │ │ - * A parser to read/write JSON safely. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Format.JSON> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.JSON = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: indent │ │ │ │ │ - * {String} For "pretty" printing, the indent string will be used once for │ │ │ │ │ - * each indentation level. │ │ │ │ │ - */ │ │ │ │ │ - indent: " ", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: space │ │ │ │ │ - * {String} For "pretty" printing, the space string will be used after │ │ │ │ │ - * the ":" separating a name/value pair. │ │ │ │ │ - */ │ │ │ │ │ - space: " ", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: newline │ │ │ │ │ - * {String} For "pretty" printing, the newline string will be used at the │ │ │ │ │ - * end of each name/value pair or array item. │ │ │ │ │ - */ │ │ │ │ │ - newline: "\n", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: level │ │ │ │ │ - * {Integer} For "pretty" printing, this is incremented/decremented during │ │ │ │ │ - * serialization. │ │ │ │ │ - */ │ │ │ │ │ - level: 0, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: pretty │ │ │ │ │ - * {Boolean} Serialize with extra whitespace for structure. This is set │ │ │ │ │ - * by the <write> method. │ │ │ │ │ - */ │ │ │ │ │ - pretty: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: nativeJSON │ │ │ │ │ - * {Boolean} Does the browser support native json? │ │ │ │ │ - */ │ │ │ │ │ - nativeJSON: (function() { │ │ │ │ │ - return !!(window.JSON && typeof JSON.parse == "function" && typeof JSON.stringify == "function"); │ │ │ │ │ - })(), │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.JSON │ │ │ │ │ - * Create a new parser for JSON. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Deserialize a json string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * json - {String} A JSON string │ │ │ │ │ - * filter - {Function} A function which will be called for every key and │ │ │ │ │ - * value at every level of the final result. Each value will be │ │ │ │ │ - * replaced by the result of the filter function. This can be used to │ │ │ │ │ - * reform generic objects into instances of classes, or to transform │ │ │ │ │ - * date strings into Date objects. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object, array, string, or number . │ │ │ │ │ - */ │ │ │ │ │ - read: function(json, filter) { │ │ │ │ │ - var object; │ │ │ │ │ - if (this.nativeJSON) { │ │ │ │ │ - object = JSON.parse(json, filter); │ │ │ │ │ - } else try { │ │ │ │ │ - /** │ │ │ │ │ - * Parsing happens in three stages. In the first stage, we run the │ │ │ │ │ - * text against a regular expression which looks for non-JSON │ │ │ │ │ - * characters. We are especially concerned with '()' and 'new' │ │ │ │ │ - * because they can cause invocation, and '=' because it can │ │ │ │ │ - * cause mutation. But just to be safe, we will reject all │ │ │ │ │ - * unexpected characters. │ │ │ │ │ - */ │ │ │ │ │ - if (/^[\],:{}\s]*$/.test(json.replace(/\\["\\\/bfnrtu]/g, '@').replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, ']').replace(/(?:^|:|,)(?:\s*\[)+/g, ''))) { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * In the second stage we use the eval function to compile the │ │ │ │ │ - * text into a JavaScript structure. The '{' operator is │ │ │ │ │ - * subject to a syntactic ambiguity in JavaScript - it can │ │ │ │ │ - * begin a block or an object literal. We wrap the text in │ │ │ │ │ - * parens to eliminate the ambiguity. │ │ │ │ │ - */ │ │ │ │ │ - object = eval('(' + json + ')'); │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * In the optional third stage, we recursively walk the new │ │ │ │ │ - * structure, passing each name/value pair to a filter │ │ │ │ │ - * function for possible transformation. │ │ │ │ │ - */ │ │ │ │ │ - if (typeof filter === 'function') { │ │ │ │ │ - function walk(k, v) { │ │ │ │ │ - if (v && typeof v === 'object') { │ │ │ │ │ - for (var i in v) { │ │ │ │ │ - if (v.hasOwnProperty(i)) { │ │ │ │ │ - v[i] = walk(i, v[i]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return filter(k, v); │ │ │ │ │ - } │ │ │ │ │ - object = walk('', object); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } catch (e) { │ │ │ │ │ - // Fall through if the regexp test fails. │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.keepData) { │ │ │ │ │ - this.data = object; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return object; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Serialize an object into a JSON string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * value - {String} The object, array, string, number, boolean or date │ │ │ │ │ - * to be serialized. │ │ │ │ │ - * pretty - {Boolean} Structure the output with newlines and indentation. │ │ │ │ │ - * Default is false. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The JSON string representation of the input value. │ │ │ │ │ - */ │ │ │ │ │ - write: function(value, pretty) { │ │ │ │ │ - this.pretty = !!pretty; │ │ │ │ │ - var json = null; │ │ │ │ │ - var type = typeof value; │ │ │ │ │ - if (this.serialize[type]) { │ │ │ │ │ - try { │ │ │ │ │ - json = (!this.pretty && this.nativeJSON) ? │ │ │ │ │ - JSON.stringify(value) : │ │ │ │ │ - this.serialize[type].apply(this, [value]); │ │ │ │ │ - } catch (err) { │ │ │ │ │ - OpenLayers.Console.error("Trouble serializing: " + err); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return json; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: writeIndent │ │ │ │ │ - * Output an indentation string depending on the indentation level. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} An appropriate indentation string. │ │ │ │ │ - */ │ │ │ │ │ - writeIndent: function() { │ │ │ │ │ - var pieces = []; │ │ │ │ │ - if (this.pretty) { │ │ │ │ │ - for (var i = 0; i < this.level; ++i) { │ │ │ │ │ - pieces.push(this.indent); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return pieces.join(''); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: writeNewline │ │ │ │ │ - * Output a string representing a newline if in pretty printing mode. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string representing a new line. │ │ │ │ │ - */ │ │ │ │ │ - writeNewline: function() { │ │ │ │ │ - return (this.pretty) ? this.newline : ''; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: writeSpace │ │ │ │ │ - * Output a string representing a space if in pretty printing mode. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A space. │ │ │ │ │ - */ │ │ │ │ │ - writeSpace: function() { │ │ │ │ │ - return (this.pretty) ? this.space : ''; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: serialize │ │ │ │ │ - * Object with properties corresponding to the serializable data types. │ │ │ │ │ - * Property values are functions that do the actual serializing. │ │ │ │ │ - */ │ │ │ │ │ - serialize: { │ │ │ │ │ - /** │ │ │ │ │ - * Method: serialize.object │ │ │ │ │ - * Transform an object into a JSON string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * object - {Object} The object to be serialized. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A JSON string representing the object. │ │ │ │ │ - */ │ │ │ │ │ - 'object': function(object) { │ │ │ │ │ - // three special objects that we want to treat differently │ │ │ │ │ - if (object == null) { │ │ │ │ │ - return "null"; │ │ │ │ │ - } │ │ │ │ │ - if (object.constructor == Date) { │ │ │ │ │ - return this.serialize.date.apply(this, [object]); │ │ │ │ │ - } │ │ │ │ │ - if (object.constructor == Array) { │ │ │ │ │ - return this.serialize.array.apply(this, [object]); │ │ │ │ │ - } │ │ │ │ │ - var pieces = ['{']; │ │ │ │ │ - this.level += 1; │ │ │ │ │ - var key, keyJSON, valueJSON; │ │ │ │ │ - │ │ │ │ │ - var addComma = false; │ │ │ │ │ - for (key in object) { │ │ │ │ │ - if (object.hasOwnProperty(key)) { │ │ │ │ │ - // recursive calls need to allow for sub-classing │ │ │ │ │ - keyJSON = OpenLayers.Format.JSON.prototype.write.apply(this, │ │ │ │ │ - [key, this.pretty]); │ │ │ │ │ - valueJSON = OpenLayers.Format.JSON.prototype.write.apply(this, │ │ │ │ │ - [object[key], this.pretty]); │ │ │ │ │ - if (keyJSON != null && valueJSON != null) { │ │ │ │ │ - if (addComma) { │ │ │ │ │ - pieces.push(','); │ │ │ │ │ - } │ │ │ │ │ - pieces.push(this.writeNewline(), this.writeIndent(), │ │ │ │ │ - keyJSON, ':', this.writeSpace(), valueJSON); │ │ │ │ │ - addComma = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.level -= 1; │ │ │ │ │ - pieces.push(this.writeNewline(), this.writeIndent(), '}'); │ │ │ │ │ - return pieces.join(''); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: serialize.array │ │ │ │ │ - * Transform an array into a JSON string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * array - {Array} The array to be serialized │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A JSON string representing the array. │ │ │ │ │ - */ │ │ │ │ │ - 'array': function(array) { │ │ │ │ │ - var json; │ │ │ │ │ - var pieces = ['[']; │ │ │ │ │ - this.level += 1; │ │ │ │ │ - │ │ │ │ │ - for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ - // recursive calls need to allow for sub-classing │ │ │ │ │ - json = OpenLayers.Format.JSON.prototype.write.apply(this, │ │ │ │ │ - [array[i], this.pretty]); │ │ │ │ │ - if (json != null) { │ │ │ │ │ - if (i > 0) { │ │ │ │ │ - pieces.push(','); │ │ │ │ │ - } │ │ │ │ │ - pieces.push(this.writeNewline(), this.writeIndent(), json); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.level -= 1; │ │ │ │ │ - pieces.push(this.writeNewline(), this.writeIndent(), ']'); │ │ │ │ │ - return pieces.join(''); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: serialize.string │ │ │ │ │ - * Transform a string into a JSON string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * string - {String} The string to be serialized │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A JSON string representing the string. │ │ │ │ │ - */ │ │ │ │ │ - 'string': function(string) { │ │ │ │ │ - // If the string contains no control characters, no quote characters, and no │ │ │ │ │ - // backslash characters, then we can simply slap some quotes around it. │ │ │ │ │ - // Otherwise we must also replace the offending characters with safe │ │ │ │ │ - // sequences. │ │ │ │ │ - var m = { │ │ │ │ │ - '\b': '\\b', │ │ │ │ │ - '\t': '\\t', │ │ │ │ │ - '\n': '\\n', │ │ │ │ │ - '\f': '\\f', │ │ │ │ │ - '\r': '\\r', │ │ │ │ │ - '"': '\\"', │ │ │ │ │ - '\\': '\\\\' │ │ │ │ │ - }; │ │ │ │ │ - if (/["\\\x00-\x1f]/.test(string)) { │ │ │ │ │ - return '"' + string.replace(/([\x00-\x1f\\"])/g, function(a, b) { │ │ │ │ │ - var c = m[b]; │ │ │ │ │ - if (c) { │ │ │ │ │ - return c; │ │ │ │ │ - } │ │ │ │ │ - c = b.charCodeAt(); │ │ │ │ │ - return '\\u00' + │ │ │ │ │ - Math.floor(c / 16).toString(16) + │ │ │ │ │ - (c % 16).toString(16); │ │ │ │ │ - }) + '"'; │ │ │ │ │ - } │ │ │ │ │ - return '"' + string + '"'; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: serialize.number │ │ │ │ │ - * Transform a number into a JSON string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * number - {Number} The number to be serialized. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A JSON string representing the number. │ │ │ │ │ - */ │ │ │ │ │ - 'number': function(number) { │ │ │ │ │ - return isFinite(number) ? String(number) : "null"; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: serialize.boolean │ │ │ │ │ - * Transform a boolean into a JSON string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bool - {Boolean} The boolean to be serialized. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A JSON string representing the boolean. │ │ │ │ │ - */ │ │ │ │ │ - 'boolean': function(bool) { │ │ │ │ │ - return String(bool); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: serialize.object │ │ │ │ │ - * Transform a date into a JSON string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * date - {Date} The date to be serialized. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A JSON string representing the date. │ │ │ │ │ - */ │ │ │ │ │ - 'date': function(date) { │ │ │ │ │ - function format(number) { │ │ │ │ │ - // Format integers to have at least two digits. │ │ │ │ │ - return (number < 10) ? '0' + number : number; │ │ │ │ │ - } │ │ │ │ │ - return '"' + date.getFullYear() + '-' + │ │ │ │ │ - format(date.getMonth() + 1) + '-' + │ │ │ │ │ - format(date.getDate()) + 'T' + │ │ │ │ │ - format(date.getHours()) + ':' + │ │ │ │ │ - format(date.getMinutes()) + ':' + │ │ │ │ │ - format(date.getSeconds()) + '"'; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.JSON" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Geometry.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Geometry │ │ │ │ │ - * A Geometry is a description of a geographic object. Create an instance of │ │ │ │ │ - * this class with the <OpenLayers.Geometry> constructor. This is a base class, │ │ │ │ │ - * typical geometry types are described by subclasses of this class. │ │ │ │ │ - * │ │ │ │ │ - * Note that if you use the <OpenLayers.Geometry.fromWKT> method, you must │ │ │ │ │ - * explicitly include the OpenLayers.Format.WKT in your build. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: id │ │ │ │ │ - * {String} A unique identifier for this geometry. │ │ │ │ │ - */ │ │ │ │ │ - id: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: parent │ │ │ │ │ - * {<OpenLayers.Geometry>}This is set when a Geometry is added as component │ │ │ │ │ - * of another geometry │ │ │ │ │ - */ │ │ │ │ │ - parent: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: bounds │ │ │ │ │ - * {<OpenLayers.Bounds>} The bounds of this geometry │ │ │ │ │ - */ │ │ │ │ │ - bounds: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Geometry │ │ │ │ │ - * Creates a geometry object. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function() { │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Destroy this geometry. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.id = null; │ │ │ │ │ - this.bounds = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Create a clone of this geometry. Does not set any non-standard │ │ │ │ │ - * properties of the cloned geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} An exact clone of this geometry. │ │ │ │ │ - */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - return new OpenLayers.Geometry(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setBounds │ │ │ │ │ - * Set the bounds for this Geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - */ │ │ │ │ │ - setBounds: function(bounds) { │ │ │ │ │ - if (bounds) { │ │ │ │ │ - this.bounds = bounds.clone(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clearBounds │ │ │ │ │ - * Nullify this components bounds and that of its parent as well. │ │ │ │ │ - */ │ │ │ │ │ - clearBounds: function() { │ │ │ │ │ - this.bounds = null; │ │ │ │ │ - if (this.parent) { │ │ │ │ │ - this.parent.clearBounds(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: extendBounds │ │ │ │ │ - * Extend the existing bounds to include the new bounds. │ │ │ │ │ - * If geometry's bounds is not yet set, then set a new Bounds. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newBounds - {<OpenLayers.Bounds>} │ │ │ │ │ - */ │ │ │ │ │ - extendBounds: function(newBounds) { │ │ │ │ │ - var bounds = this.getBounds(); │ │ │ │ │ - if (!bounds) { │ │ │ │ │ - this.setBounds(newBounds); │ │ │ │ │ - } else { │ │ │ │ │ - this.bounds.extend(newBounds); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getBounds │ │ │ │ │ - * Get the bounds for this Geometry. If bounds is not set, it │ │ │ │ │ - * is calculated again, this makes queries faster. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} │ │ │ │ │ - */ │ │ │ │ │ - getBounds: function() { │ │ │ │ │ - if (this.bounds == null) { │ │ │ │ │ - this.calculateBounds(); │ │ │ │ │ - } │ │ │ │ │ - return this.bounds; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: calculateBounds │ │ │ │ │ - * Recalculate the bounds for the geometry. │ │ │ │ │ - */ │ │ │ │ │ - calculateBounds: function() { │ │ │ │ │ - // │ │ │ │ │ - // This should be overridden by subclasses. │ │ │ │ │ - // │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: distanceTo │ │ │ │ │ - * Calculate the closest distance between two geometries (on the x-y plane). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ - * options - {Object} Optional properties for configuring the distance │ │ │ │ │ - * calculation. │ │ │ │ │ - * │ │ │ │ │ - * Valid options depend on the specific geometry type. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number | Object} The distance between this geometry and the target. │ │ │ │ │ - * If details is true, the return will be an object with distance, │ │ │ │ │ - * x0, y0, x1, and x2 properties. The x0 and y0 properties represent │ │ │ │ │ - * the coordinates of the closest point on this geometry. The x1 and y1 │ │ │ │ │ - * properties represent the coordinates of the closest point on the │ │ │ │ │ - * target geometry. │ │ │ │ │ - */ │ │ │ │ │ - distanceTo: function(geometry, options) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getVertices │ │ │ │ │ - * Return a list of all points in this geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * nodes - {Boolean} For lines, only return vertices that are │ │ │ │ │ - * endpoints. If false, for lines, only vertices that are not │ │ │ │ │ - * endpoints will be returned. If not provided, all vertices will │ │ │ │ │ - * be returned. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} A list of all vertices in the geometry. │ │ │ │ │ - */ │ │ │ │ │ - getVertices: function(nodes) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: atPoint │ │ │ │ │ - * Note - This is only an approximation based on the bounds of the │ │ │ │ │ - * geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>|Object} OpenLayers.LonLat or an │ │ │ │ │ - * object with a 'lon' and 'lat' properties. │ │ │ │ │ - * toleranceLon - {float} Optional tolerance in Geometric Coords │ │ │ │ │ - * toleranceLat - {float} Optional tolerance in Geographic Coords │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the geometry is at the specified location │ │ │ │ │ - */ │ │ │ │ │ - atPoint: function(lonlat, toleranceLon, toleranceLat) { │ │ │ │ │ - var atPoint = false; │ │ │ │ │ - var bounds = this.getBounds(); │ │ │ │ │ - if ((bounds != null) && (lonlat != null)) { │ │ │ │ │ - │ │ │ │ │ - var dX = (toleranceLon != null) ? toleranceLon : 0; │ │ │ │ │ - var dY = (toleranceLat != null) ? toleranceLat : 0; │ │ │ │ │ - │ │ │ │ │ - var toleranceBounds = │ │ │ │ │ - new OpenLayers.Bounds(this.bounds.left - dX, │ │ │ │ │ - this.bounds.bottom - dY, │ │ │ │ │ - this.bounds.right + dX, │ │ │ │ │ - this.bounds.top + dY); │ │ │ │ │ - │ │ │ │ │ - atPoint = toleranceBounds.containsLonLat(lonlat); │ │ │ │ │ - } │ │ │ │ │ - return atPoint; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getLength │ │ │ │ │ - * Calculate the length of this geometry. This method is defined in │ │ │ │ │ - * subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The length of the collection by summing its parts │ │ │ │ │ - */ │ │ │ │ │ - getLength: function() { │ │ │ │ │ - //to be overridden by geometries that actually have a length │ │ │ │ │ - // │ │ │ │ │ - return 0.0; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getArea │ │ │ │ │ - * Calculate the area of this geometry. This method is defined in subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The area of the collection by summing its parts │ │ │ │ │ - */ │ │ │ │ │ - getArea: function() { │ │ │ │ │ - //to be overridden by geometries that actually have an area │ │ │ │ │ - // │ │ │ │ │ - return 0.0; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getCentroid │ │ │ │ │ - * Calculate the centroid of this geometry. This method is defined in subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} The centroid of the collection │ │ │ │ │ - */ │ │ │ │ │ - getCentroid: function() { │ │ │ │ │ - return null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: toString │ │ │ │ │ - * Returns a text representation of the geometry. If the WKT format is │ │ │ │ │ - * included in a build, this will be the Well-Known Text │ │ │ │ │ - * representation. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} String representation of this geometry. │ │ │ │ │ - */ │ │ │ │ │ - toString: function() { │ │ │ │ │ - var string; │ │ │ │ │ - if (OpenLayers.Format && OpenLayers.Format.WKT) { │ │ │ │ │ - string = OpenLayers.Format.WKT.prototype.write( │ │ │ │ │ - new OpenLayers.Feature.Vector(this) │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - string = Object.prototype.toString.call(this); │ │ │ │ │ - } │ │ │ │ │ - return string; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Geometry" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Function: OpenLayers.Geometry.fromWKT │ │ │ │ │ - * Generate a geometry given a Well-Known Text string. For this method to │ │ │ │ │ - * work, you must include the OpenLayers.Format.WKT in your build │ │ │ │ │ - * explicitly. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * wkt - {String} A string representing the geometry in Well-Known Text. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} A geometry of the appropriate class. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.fromWKT = function(wkt) { │ │ │ │ │ - var geom; │ │ │ │ │ - if (OpenLayers.Format && OpenLayers.Format.WKT) { │ │ │ │ │ - var format = OpenLayers.Geometry.fromWKT.format; │ │ │ │ │ - if (!format) { │ │ │ │ │ - format = new OpenLayers.Format.WKT(); │ │ │ │ │ - OpenLayers.Geometry.fromWKT.format = format; │ │ │ │ │ - } │ │ │ │ │ - var result = format.read(wkt); │ │ │ │ │ - if (result instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ - geom = result.geometry; │ │ │ │ │ - } else if (OpenLayers.Util.isArray(result)) { │ │ │ │ │ - var len = result.length; │ │ │ │ │ - var components = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - components[i] = result[i].geometry; │ │ │ │ │ - } │ │ │ │ │ - geom = new OpenLayers.Geometry.Collection(components); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return geom; │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Method: OpenLayers.Geometry.segmentsIntersect │ │ │ │ │ - * Determine whether two line segments intersect. Optionally calculates │ │ │ │ │ - * and returns the intersection point. This function is optimized for │ │ │ │ │ - * cases where seg1.x2 >= seg2.x1 || seg2.x2 >= seg1.x1. In those │ │ │ │ │ - * obvious cases where there is no intersection, the function should │ │ │ │ │ - * not be called. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * seg1 - {Object} Object representing a segment with properties x1, y1, x2, │ │ │ │ │ - * and y2. The start point is represented by x1 and y1. The end point │ │ │ │ │ - * is represented by x2 and y2. Start and end are ordered so that x1 < x2. │ │ │ │ │ - * seg2 - {Object} Object representing a segment with properties x1, y1, x2, │ │ │ │ │ - * and y2. The start point is represented by x1 and y1. The end point │ │ │ │ │ - * is represented by x2 and y2. Start and end are ordered so that x1 < x2. │ │ │ │ │ - * options - {Object} Optional properties for calculating the intersection. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * point - {Boolean} Return the intersection point. If false, the actual │ │ │ │ │ - * intersection point will not be calculated. If true and the segments │ │ │ │ │ - * intersect, the intersection point will be returned. If true and │ │ │ │ │ - * the segments do not intersect, false will be returned. If true and │ │ │ │ │ - * the segments are coincident, true will be returned. │ │ │ │ │ - * tolerance - {Number} If a non-null value is provided, if the segments are │ │ │ │ │ - * within the tolerance distance, this will be considered an intersection. │ │ │ │ │ - * In addition, if the point option is true and the calculated intersection │ │ │ │ │ - * is within the tolerance distance of an end point, the endpoint will be │ │ │ │ │ - * returned instead of the calculated intersection. Further, if the │ │ │ │ │ - * intersection is within the tolerance of endpoints on both segments, or │ │ │ │ │ - * if two segment endpoints are within the tolerance distance of eachother │ │ │ │ │ - * (but no intersection is otherwise calculated), an endpoint on the │ │ │ │ │ - * first segment provided will be returned. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean | <OpenLayers.Geometry.Point>} The two segments intersect. │ │ │ │ │ - * If the point argument is true, the return will be the intersection │ │ │ │ │ - * point or false if none exists. If point is true and the segments │ │ │ │ │ - * are coincident, return will be true (and the instersection is equal │ │ │ │ │ - * to the shorter segment). │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.segmentsIntersect = function(seg1, seg2, options) { │ │ │ │ │ - var point = options && options.point; │ │ │ │ │ - var tolerance = options && options.tolerance; │ │ │ │ │ - var intersection = false; │ │ │ │ │ - var x11_21 = seg1.x1 - seg2.x1; │ │ │ │ │ - var y11_21 = seg1.y1 - seg2.y1; │ │ │ │ │ - var x12_11 = seg1.x2 - seg1.x1; │ │ │ │ │ - var y12_11 = seg1.y2 - seg1.y1; │ │ │ │ │ - var y22_21 = seg2.y2 - seg2.y1; │ │ │ │ │ - var x22_21 = seg2.x2 - seg2.x1; │ │ │ │ │ - var d = (y22_21 * x12_11) - (x22_21 * y12_11); │ │ │ │ │ - var n1 = (x22_21 * y11_21) - (y22_21 * x11_21); │ │ │ │ │ - var n2 = (x12_11 * y11_21) - (y12_11 * x11_21); │ │ │ │ │ - if (d == 0) { │ │ │ │ │ - // parallel │ │ │ │ │ - if (n1 == 0 && n2 == 0) { │ │ │ │ │ - // coincident │ │ │ │ │ - intersection = true; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - var along1 = n1 / d; │ │ │ │ │ - var along2 = n2 / d; │ │ │ │ │ - if (along1 >= 0 && along1 <= 1 && along2 >= 0 && along2 <= 1) { │ │ │ │ │ - // intersect │ │ │ │ │ - if (!point) { │ │ │ │ │ - intersection = true; │ │ │ │ │ - } else { │ │ │ │ │ - // calculate the intersection point │ │ │ │ │ - var x = seg1.x1 + (along1 * x12_11); │ │ │ │ │ - var y = seg1.y1 + (along1 * y12_11); │ │ │ │ │ - intersection = new OpenLayers.Geometry.Point(x, y); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (tolerance) { │ │ │ │ │ - var dist; │ │ │ │ │ - if (intersection) { │ │ │ │ │ - if (point) { │ │ │ │ │ - var segs = [seg1, seg2]; │ │ │ │ │ - var seg, x, y; │ │ │ │ │ - // check segment endpoints for proximity to intersection │ │ │ │ │ - // set intersection to first endpoint within the tolerance │ │ │ │ │ - outer: for (var i = 0; i < 2; ++i) { │ │ │ │ │ - seg = segs[i]; │ │ │ │ │ - for (var j = 1; j < 3; ++j) { │ │ │ │ │ - x = seg["x" + j]; │ │ │ │ │ - y = seg["y" + j]; │ │ │ │ │ - dist = Math.sqrt( │ │ │ │ │ - Math.pow(x - intersection.x, 2) + │ │ │ │ │ - Math.pow(y - intersection.y, 2) │ │ │ │ │ - ); │ │ │ │ │ - if (dist < tolerance) { │ │ │ │ │ - intersection.x = x; │ │ │ │ │ - intersection.y = y; │ │ │ │ │ - break outer; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - // no calculated intersection, but segments could be within │ │ │ │ │ - // the tolerance of one another │ │ │ │ │ - var segs = [seg1, seg2]; │ │ │ │ │ - var source, target, x, y, p, result; │ │ │ │ │ - // check segment endpoints for proximity to intersection │ │ │ │ │ - // set intersection to first endpoint within the tolerance │ │ │ │ │ - outer: for (var i = 0; i < 2; ++i) { │ │ │ │ │ - source = segs[i]; │ │ │ │ │ - target = segs[(i + 1) % 2]; │ │ │ │ │ - for (var j = 1; j < 3; ++j) { │ │ │ │ │ - p = { │ │ │ │ │ - x: source["x" + j], │ │ │ │ │ - y: source["y" + j] │ │ │ │ │ - }; │ │ │ │ │ - result = OpenLayers.Geometry.distanceToSegment(p, target); │ │ │ │ │ - if (result.distance < tolerance) { │ │ │ │ │ - if (point) { │ │ │ │ │ - intersection = new OpenLayers.Geometry.Point(p.x, p.y); │ │ │ │ │ - } else { │ │ │ │ │ - intersection = true; │ │ │ │ │ - } │ │ │ │ │ - break outer; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return intersection; │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Function: OpenLayers.Geometry.distanceToSegment │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {Object} An object with x and y properties representing the │ │ │ │ │ - * point coordinates. │ │ │ │ │ - * segment - {Object} An object with x1, y1, x2, and y2 properties │ │ │ │ │ - * representing endpoint coordinates. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object with distance, along, x, and y properties. The distance │ │ │ │ │ - * will be the shortest distance between the input point and segment. │ │ │ │ │ - * The x and y properties represent the coordinates along the segment │ │ │ │ │ - * where the shortest distance meets the segment. The along attribute │ │ │ │ │ - * describes how far between the two segment points the given point is. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.distanceToSegment = function(point, segment) { │ │ │ │ │ - var result = OpenLayers.Geometry.distanceSquaredToSegment(point, segment); │ │ │ │ │ - result.distance = Math.sqrt(result.distance); │ │ │ │ │ - return result; │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Function: OpenLayers.Geometry.distanceSquaredToSegment │ │ │ │ │ - * │ │ │ │ │ - * Usually the distanceToSegment function should be used. This variant however │ │ │ │ │ - * can be used for comparisons where the exact distance is not important. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {Object} An object with x and y properties representing the │ │ │ │ │ - * point coordinates. │ │ │ │ │ - * segment - {Object} An object with x1, y1, x2, and y2 properties │ │ │ │ │ - * representing endpoint coordinates. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object with squared distance, along, x, and y properties. │ │ │ │ │ - * The distance will be the shortest distance between the input point and │ │ │ │ │ - * segment. The x and y properties represent the coordinates along the │ │ │ │ │ - * segment where the shortest distance meets the segment. The along │ │ │ │ │ - * attribute describes how far between the two segment points the given │ │ │ │ │ - * point is. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.distanceSquaredToSegment = function(point, segment) { │ │ │ │ │ - var x0 = point.x; │ │ │ │ │ - var y0 = point.y; │ │ │ │ │ - var x1 = segment.x1; │ │ │ │ │ - var y1 = segment.y1; │ │ │ │ │ - var x2 = segment.x2; │ │ │ │ │ - var y2 = segment.y2; │ │ │ │ │ - var dx = x2 - x1; │ │ │ │ │ - var dy = y2 - y1; │ │ │ │ │ - var along = ((dx * (x0 - x1)) + (dy * (y0 - y1))) / │ │ │ │ │ - (Math.pow(dx, 2) + Math.pow(dy, 2)); │ │ │ │ │ - var x, y; │ │ │ │ │ - if (along <= 0.0) { │ │ │ │ │ - x = x1; │ │ │ │ │ - y = y1; │ │ │ │ │ - } else if (along >= 1.0) { │ │ │ │ │ - x = x2; │ │ │ │ │ - y = y2; │ │ │ │ │ - } else { │ │ │ │ │ - x = x1 + along * dx; │ │ │ │ │ - y = y1 + along * dy; │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - distance: Math.pow(x - x0, 2) + Math.pow(y - y0, 2), │ │ │ │ │ - x: x, │ │ │ │ │ - y: y, │ │ │ │ │ - along: along │ │ │ │ │ - }; │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Geometry/Point.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Geometry.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Geometry.Point │ │ │ │ │ - * Point geometry class. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Geometry> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.Point = OpenLayers.Class(OpenLayers.Geometry, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: x │ │ │ │ │ - * {float} │ │ │ │ │ - */ │ │ │ │ │ - x: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: y │ │ │ │ │ - * {float} │ │ │ │ │ - */ │ │ │ │ │ - y: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Geometry.Point │ │ │ │ │ - * Construct a point geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {float} │ │ │ │ │ - * y - {float} │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(x, y) { │ │ │ │ │ - OpenLayers.Geometry.prototype.initialize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - this.x = parseFloat(x); │ │ │ │ │ - this.y = parseFloat(y); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} An exact clone of this OpenLayers.Geometry.Point │ │ │ │ │ - */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Geometry.Point(this.x, this.y); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // catch any randomly tagged-on properties │ │ │ │ │ - OpenLayers.Util.applyDefaults(obj, this); │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateBounds │ │ │ │ │ - * Create a new Bounds based on the lon/lat │ │ │ │ │ - */ │ │ │ │ │ - calculateBounds: function() { │ │ │ │ │ - this.bounds = new OpenLayers.Bounds(this.x, this.y, │ │ │ │ │ - this.x, this.y); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: distanceTo │ │ │ │ │ - * Calculate the closest distance between two geometries (on the x-y plane). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ - * options - {Object} Optional properties for configuring the distance │ │ │ │ │ - * calculation. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * details - {Boolean} Return details from the distance calculation. │ │ │ │ │ - * Default is false. │ │ │ │ │ - * edge - {Boolean} Calculate the distance from this geometry to the │ │ │ │ │ - * nearest edge of the target geometry. Default is true. If true, │ │ │ │ │ - * calling distanceTo from a geometry that is wholly contained within │ │ │ │ │ - * the target will result in a non-zero distance. If false, whenever │ │ │ │ │ - * geometries intersect, calling distanceTo will return 0. If false, │ │ │ │ │ - * details cannot be returned. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number | Object} The distance between this geometry and the target. │ │ │ │ │ - * If details is true, the return will be an object with distance, │ │ │ │ │ - * x0, y0, x1, and x2 properties. The x0 and y0 properties represent │ │ │ │ │ - * the coordinates of the closest point on this geometry. The x1 and y1 │ │ │ │ │ - * properties represent the coordinates of the closest point on the │ │ │ │ │ - * target geometry. │ │ │ │ │ - */ │ │ │ │ │ - distanceTo: function(geometry, options) { │ │ │ │ │ - var edge = !(options && options.edge === false); │ │ │ │ │ - var details = edge && options && options.details; │ │ │ │ │ - var distance, x0, y0, x1, y1, result; │ │ │ │ │ - if (geometry instanceof OpenLayers.Geometry.Point) { │ │ │ │ │ - x0 = this.x; │ │ │ │ │ - y0 = this.y; │ │ │ │ │ - x1 = geometry.x; │ │ │ │ │ - y1 = geometry.y; │ │ │ │ │ - distance = Math.sqrt(Math.pow(x0 - x1, 2) + Math.pow(y0 - y1, 2)); │ │ │ │ │ - result = !details ? │ │ │ │ │ - distance : { │ │ │ │ │ - x0: x0, │ │ │ │ │ - y0: y0, │ │ │ │ │ - x1: x1, │ │ │ │ │ - y1: y1, │ │ │ │ │ - distance: distance │ │ │ │ │ - }; │ │ │ │ │ - } else { │ │ │ │ │ - result = geometry.distanceTo(this, options); │ │ │ │ │ - if (details) { │ │ │ │ │ - // switch coord order since this geom is target │ │ │ │ │ - result = { │ │ │ │ │ - x0: result.x1, │ │ │ │ │ - y0: result.y1, │ │ │ │ │ - x1: result.x0, │ │ │ │ │ - y1: result.y0, │ │ │ │ │ - distance: result.distance │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return result; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: equals │ │ │ │ │ - * Determine whether another geometry is equivalent to this one. Geometries │ │ │ │ │ - * are considered equivalent if all components have the same coordinates. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geom - {<OpenLayers.Geometry.Point>} The geometry to test. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The supplied geometry is equivalent to this geometry. │ │ │ │ │ - */ │ │ │ │ │ - equals: function(geom) { │ │ │ │ │ - var equals = false; │ │ │ │ │ - if (geom != null) { │ │ │ │ │ - equals = ((this.x == geom.x && this.y == geom.y) || │ │ │ │ │ - (isNaN(this.x) && isNaN(this.y) && isNaN(geom.x) && isNaN(geom.y))); │ │ │ │ │ - } │ │ │ │ │ - return equals; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: toShortString │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} Shortened String representation of Point object. │ │ │ │ │ - * (ex. <i>"5, 42"</i>) │ │ │ │ │ - */ │ │ │ │ │ - toShortString: function() { │ │ │ │ │ - return (this.x + ", " + this.y); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: move │ │ │ │ │ - * Moves a geometry by the given displacement along positive x and y axes. │ │ │ │ │ - * This modifies the position of the geometry and clears the cached │ │ │ │ │ - * bounds. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {Float} Distance to move geometry in positive x direction. │ │ │ │ │ - * y - {Float} Distance to move geometry in positive y direction. │ │ │ │ │ - */ │ │ │ │ │ - move: function(x, y) { │ │ │ │ │ - this.x = this.x + x; │ │ │ │ │ - this.y = this.y + y; │ │ │ │ │ - this.clearBounds(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: rotate │ │ │ │ │ - * Rotate a point around another. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * angle - {Float} Rotation angle in degrees (measured counterclockwise │ │ │ │ │ - * from the positive x-axis) │ │ │ │ │ - * origin - {<OpenLayers.Geometry.Point>} Center point for the rotation │ │ │ │ │ - */ │ │ │ │ │ - rotate: function(angle, origin) { │ │ │ │ │ - angle *= Math.PI / 180; │ │ │ │ │ - var radius = this.distanceTo(origin); │ │ │ │ │ - var theta = angle + Math.atan2(this.y - origin.y, this.x - origin.x); │ │ │ │ │ - this.x = origin.x + (radius * Math.cos(theta)); │ │ │ │ │ - this.y = origin.y + (radius * Math.sin(theta)); │ │ │ │ │ - this.clearBounds(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getCentroid │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} The centroid of the collection │ │ │ │ │ - */ │ │ │ │ │ - getCentroid: function() { │ │ │ │ │ - return new OpenLayers.Geometry.Point(this.x, this.y); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: resize │ │ │ │ │ - * Resize a point relative to some origin. For points, this has the effect │ │ │ │ │ - * of scaling a vector (from the origin to the point). This method is │ │ │ │ │ - * more useful on geometry collection subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * scale - {Float} Ratio of the new distance from the origin to the old │ │ │ │ │ - * distance from the origin. A scale of 2 doubles the │ │ │ │ │ - * distance between the point and origin. │ │ │ │ │ - * origin - {<OpenLayers.Geometry.Point>} Point of origin for resizing │ │ │ │ │ - * ratio - {Float} Optional x:y ratio for resizing. Default ratio is 1. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} - The current geometry. │ │ │ │ │ - */ │ │ │ │ │ - resize: function(scale, origin, ratio) { │ │ │ │ │ - ratio = (ratio == undefined) ? 1 : ratio; │ │ │ │ │ - this.x = origin.x + (scale * ratio * (this.x - origin.x)); │ │ │ │ │ - this.y = origin.y + (scale * (this.y - origin.y)); │ │ │ │ │ - this.clearBounds(); │ │ │ │ │ - return this; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: intersects │ │ │ │ │ - * Determine if the input geometry intersects this one. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} Any type of geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The input geometry intersects this one. │ │ │ │ │ - */ │ │ │ │ │ - intersects: function(geometry) { │ │ │ │ │ - var intersect = false; │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - intersect = this.equals(geometry); │ │ │ │ │ - } else { │ │ │ │ │ - intersect = geometry.intersects(this); │ │ │ │ │ - } │ │ │ │ │ - return intersect; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: transform │ │ │ │ │ - * Translate the x,y properties of the point from source to dest. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * source - {<OpenLayers.Projection>} │ │ │ │ │ - * dest - {<OpenLayers.Projection>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} │ │ │ │ │ - */ │ │ │ │ │ - transform: function(source, dest) { │ │ │ │ │ - if ((source && dest)) { │ │ │ │ │ - OpenLayers.Projection.transform( │ │ │ │ │ - this, source, dest); │ │ │ │ │ - this.bounds = null; │ │ │ │ │ - } │ │ │ │ │ - return this; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getVertices │ │ │ │ │ - * Return a list of all points in this geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * nodes - {Boolean} For lines, only return vertices that are │ │ │ │ │ - * endpoints. If false, for lines, only vertices that are not │ │ │ │ │ - * endpoints will be returned. If not provided, all vertices will │ │ │ │ │ - * be returned. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} A list of all vertices in the geometry. │ │ │ │ │ - */ │ │ │ │ │ - getVertices: function(nodes) { │ │ │ │ │ - return [this]; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Geometry.Point" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Geometry/Collection.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Geometry.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Geometry.Collection │ │ │ │ │ - * A Collection is exactly what it sounds like: A collection of different │ │ │ │ │ - * Geometries. These are stored in the local parameter <components> (which │ │ │ │ │ - * can be passed as a parameter to the constructor). │ │ │ │ │ - * │ │ │ │ │ - * As new geometries are added to the collection, they are NOT cloned. │ │ │ │ │ - * When removing geometries, they need to be specified by reference (ie you │ │ │ │ │ - * have to pass in the *exact* geometry to be removed). │ │ │ │ │ - * │ │ │ │ │ - * The <getArea> and <getLength> functions here merely iterate through │ │ │ │ │ - * the components, summing their respective areas and lengths. │ │ │ │ │ - * │ │ │ │ │ - * Create a new instance with the <OpenLayers.Geometry.Collection> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Geometry> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.Collection = OpenLayers.Class(OpenLayers.Geometry, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: components │ │ │ │ │ - * {Array(<OpenLayers.Geometry>)} The component parts of this geometry │ │ │ │ │ - */ │ │ │ │ │ - components: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: componentTypes │ │ │ │ │ - * {Array(String)} An array of class names representing the types of │ │ │ │ │ - * components that the collection can include. A null value means the │ │ │ │ │ - * component types are not restricted. │ │ │ │ │ - */ │ │ │ │ │ - componentTypes: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Geometry.Collection │ │ │ │ │ - * Creates a Geometry Collection -- a list of geoms. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * components - {Array(<OpenLayers.Geometry>)} Optional array of geometries │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(components) { │ │ │ │ │ - OpenLayers.Geometry.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.components = []; │ │ │ │ │ - if (components != null) { │ │ │ │ │ - this.addComponents(components); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Destroy this geometry. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.components.length = 0; │ │ │ │ │ - this.components = null; │ │ │ │ │ - OpenLayers.Geometry.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Clone this geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Collection>} An exact clone of this collection │ │ │ │ │ - */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - var geometry = eval("new " + this.CLASS_NAME + "()"); │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ - geometry.addComponent(this.components[i].clone()); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // catch any randomly tagged-on properties │ │ │ │ │ - OpenLayers.Util.applyDefaults(geometry, this); │ │ │ │ │ - │ │ │ │ │ - return geometry; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getComponentsString │ │ │ │ │ - * Get a string representing the components for this collection │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string representation of the components of this geometry │ │ │ │ │ - */ │ │ │ │ │ - getComponentsString: function() { │ │ │ │ │ - var strings = []; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ - strings.push(this.components[i].toShortString()); │ │ │ │ │ - } │ │ │ │ │ - return strings.join(","); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: calculateBounds │ │ │ │ │ - * Recalculate the bounds by iterating through the components and │ │ │ │ │ - * calling calling extendBounds() on each item. │ │ │ │ │ - */ │ │ │ │ │ - calculateBounds: function() { │ │ │ │ │ - this.bounds = null; │ │ │ │ │ - var bounds = new OpenLayers.Bounds(); │ │ │ │ │ - var components = this.components; │ │ │ │ │ - if (components) { │ │ │ │ │ - for (var i = 0, len = components.length; i < len; i++) { │ │ │ │ │ - bounds.extend(components[i].getBounds()); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // to preserve old behavior, we only set bounds if non-null │ │ │ │ │ - // in the future, we could add bounds.isEmpty() │ │ │ │ │ - if (bounds.left != null && bounds.bottom != null && │ │ │ │ │ - bounds.right != null && bounds.top != null) { │ │ │ │ │ - this.setBounds(bounds); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: addComponents │ │ │ │ │ - * Add components to this geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * components - {Array(<OpenLayers.Geometry>)} An array of geometries to add │ │ │ │ │ - */ │ │ │ │ │ - addComponents: function(components) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(components))) { │ │ │ │ │ - components = [components]; │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = components.length; i < len; i++) { │ │ │ │ │ - this.addComponent(components[i]); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addComponent │ │ │ │ │ - * Add a new component (geometry) to the collection. If this.componentTypes │ │ │ │ │ - * is set, then the component class name must be in the componentTypes array. │ │ │ │ │ - * │ │ │ │ │ - * The bounds cache is reset. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * component - {<OpenLayers.Geometry>} A geometry to add │ │ │ │ │ - * index - {int} Optional index into the array to insert the component │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The component geometry was successfully added │ │ │ │ │ - */ │ │ │ │ │ - addComponent: function(component, index) { │ │ │ │ │ - var added = false; │ │ │ │ │ - if (component) { │ │ │ │ │ - if (this.componentTypes == null || │ │ │ │ │ - (OpenLayers.Util.indexOf(this.componentTypes, │ │ │ │ │ - component.CLASS_NAME) > -1)) { │ │ │ │ │ - │ │ │ │ │ - if (index != null && (index < this.components.length)) { │ │ │ │ │ - var components1 = this.components.slice(0, index); │ │ │ │ │ - var components2 = this.components.slice(index, │ │ │ │ │ - this.components.length); │ │ │ │ │ - components1.push(component); │ │ │ │ │ - this.components = components1.concat(components2); │ │ │ │ │ - } else { │ │ │ │ │ - this.components.push(component); │ │ │ │ │ - } │ │ │ │ │ - component.parent = this; │ │ │ │ │ - this.clearBounds(); │ │ │ │ │ - added = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return added; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: removeComponents │ │ │ │ │ - * Remove components from this geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * components - {Array(<OpenLayers.Geometry>)} The components to be removed │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} A component was removed. │ │ │ │ │ - */ │ │ │ │ │ - removeComponents: function(components) { │ │ │ │ │ - var removed = false; │ │ │ │ │ - │ │ │ │ │ - if (!(OpenLayers.Util.isArray(components))) { │ │ │ │ │ - components = [components]; │ │ │ │ │ - } │ │ │ │ │ - for (var i = components.length - 1; i >= 0; --i) { │ │ │ │ │ - removed = this.removeComponent(components[i]) || removed; │ │ │ │ │ - } │ │ │ │ │ - return removed; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeComponent │ │ │ │ │ - * Remove a component from this geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * component - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The component was removed. │ │ │ │ │ - */ │ │ │ │ │ - removeComponent: function(component) { │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Util.removeItem(this.components, component); │ │ │ │ │ - │ │ │ │ │ - // clearBounds() so that it gets recalculated on the next call │ │ │ │ │ - // to this.getBounds(); │ │ │ │ │ - this.clearBounds(); │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getLength │ │ │ │ │ - * Calculate the length of this geometry │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The length of the geometry │ │ │ │ │ - */ │ │ │ │ │ - getLength: function() { │ │ │ │ │ - var length = 0.0; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ - length += this.components[i].getLength(); │ │ │ │ │ - } │ │ │ │ │ - return length; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getArea │ │ │ │ │ - * Calculate the area of this geometry. Note how this function is overridden │ │ │ │ │ - * in <OpenLayers.Geometry.Polygon>. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The area of the collection by summing its parts │ │ │ │ │ - */ │ │ │ │ │ - getArea: function() { │ │ │ │ │ - var area = 0.0; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ - area += this.components[i].getArea(); │ │ │ │ │ - } │ │ │ │ │ - return area; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getGeodesicArea │ │ │ │ │ - * Calculate the approximate area of the polygon were it projected onto │ │ │ │ │ - * the earth. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * projection - {<OpenLayers.Projection>} The spatial reference system │ │ │ │ │ - * for the geometry coordinates. If not provided, Geographic/WGS84 is │ │ │ │ │ - * assumed. │ │ │ │ │ - * │ │ │ │ │ - * Reference: │ │ │ │ │ - * Robert. G. Chamberlain and William H. Duquette, "Some Algorithms for │ │ │ │ │ - * Polygons on a Sphere", JPL Publication 07-03, Jet Propulsion │ │ │ │ │ - * Laboratory, Pasadena, CA, June 2007 http://trs-new.jpl.nasa.gov/dspace/handle/2014/40409 │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {float} The approximate geodesic area of the geometry in square meters. │ │ │ │ │ - */ │ │ │ │ │ - getGeodesicArea: function(projection) { │ │ │ │ │ - var area = 0.0; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; i++) { │ │ │ │ │ - area += this.components[i].getGeodesicArea(projection); │ │ │ │ │ - } │ │ │ │ │ - return area; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getCentroid │ │ │ │ │ - * │ │ │ │ │ - * Compute the centroid for this geometry collection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * weighted - {Boolean} Perform the getCentroid computation recursively, │ │ │ │ │ - * returning an area weighted average of all geometries in this collection. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} The centroid of the collection │ │ │ │ │ - */ │ │ │ │ │ - getCentroid: function(weighted) { │ │ │ │ │ - if (!weighted) { │ │ │ │ │ - return this.components.length && this.components[0].getCentroid(); │ │ │ │ │ - } │ │ │ │ │ - var len = this.components.length; │ │ │ │ │ - if (!len) { │ │ │ │ │ - return false; │ │ │ │ │ + getCentroid: function(weighted) { │ │ │ │ │ + if (!weighted) { │ │ │ │ │ + return this.components.length && this.components[0].getCentroid(); │ │ │ │ │ + } │ │ │ │ │ + var len = this.components.length; │ │ │ │ │ + if (!len) { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ var areas = []; │ │ │ │ │ var centroids = []; │ │ │ │ │ var areaSum = 0; │ │ │ │ │ var minArea = Number.MAX_VALUE; │ │ │ │ │ var component; │ │ │ │ │ @@ -20404,275 +17021,562 @@ │ │ │ │ │ return this; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Geometry.LineString" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Geometry/MultiLineString.js │ │ │ │ │ + OpenLayers/Handler/Path.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Geometry/Collection.js │ │ │ │ │ + * @requires OpenLayers/Handler/Point.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Geometry.MultiLineString │ │ │ │ │ - * A MultiLineString is a geometry with multiple <OpenLayers.Geometry.LineString> │ │ │ │ │ - * components. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Handler.Path │ │ │ │ │ + * Handler to draw a path on the map. Path is displayed on mouse down, │ │ │ │ │ + * moves on mouse move, and is finished on mouse up. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Geometry.Collection> │ │ │ │ │ - * - <OpenLayers.Geometry> │ │ │ │ │ + * - <OpenLayers.Handler.Point> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Geometry.MultiLineString = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Geometry.Collection, { │ │ │ │ │ +OpenLayers.Handler.Path = OpenLayers.Class(OpenLayers.Handler.Point, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: componentTypes │ │ │ │ │ - * {Array(String)} An array of class names representing the types of │ │ │ │ │ - * components that the collection can include. A null value means the │ │ │ │ │ - * component types are not restricted. │ │ │ │ │ - */ │ │ │ │ │ - componentTypes: ["OpenLayers.Geometry.LineString"], │ │ │ │ │ + /** │ │ │ │ │ + * Property: line │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + line: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Geometry.MultiLineString │ │ │ │ │ - * Constructor for a MultiLineString Geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * components - {Array(<OpenLayers.Geometry.LineString>)} │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxVertices │ │ │ │ │ + * {Number} The maximum number of vertices which can be drawn by this │ │ │ │ │ + * handler. When the number of vertices reaches maxVertices, the │ │ │ │ │ + * geometry is automatically finalized. Default is null. │ │ │ │ │ + */ │ │ │ │ │ + maxVertices: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: split │ │ │ │ │ - * Use this geometry (the source) to attempt to split a target geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ - * options - {Object} Properties of this object will be used to determine │ │ │ │ │ - * how the split is conducted. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * mutual - {Boolean} Split the source geometry in addition to the target │ │ │ │ │ - * geometry. Default is false. │ │ │ │ │ - * edge - {Boolean} Allow splitting when only edges intersect. Default is │ │ │ │ │ - * true. If false, a vertex on the source must be within the tolerance │ │ │ │ │ - * distance of the intersection to be considered a split. │ │ │ │ │ - * tolerance - {Number} If a non-null value is provided, intersections │ │ │ │ │ - * within the tolerance distance of an existing vertex on the source │ │ │ │ │ - * will be assumed to occur at the vertex. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} A list of geometries (of this same type as the target) that │ │ │ │ │ - * result from splitting the target with the source geometry. The │ │ │ │ │ - * source and target geometry will remain unmodified. If no split │ │ │ │ │ - * results, null will be returned. If mutual is true and a split │ │ │ │ │ - * results, return will be an array of two arrays - the first will be │ │ │ │ │ - * all geometries that result from splitting the source geometry and │ │ │ │ │ - * the second will be all geometries that result from splitting the │ │ │ │ │ - * target geometry. │ │ │ │ │ - */ │ │ │ │ │ - split: function(geometry, options) { │ │ │ │ │ - var results = null; │ │ │ │ │ - var mutual = options && options.mutual; │ │ │ │ │ - var splits, sourceLine, sourceLines, sourceSplit, targetSplit; │ │ │ │ │ - var sourceParts = []; │ │ │ │ │ - var targetParts = [geometry]; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ - sourceLine = this.components[i]; │ │ │ │ │ - sourceSplit = false; │ │ │ │ │ - for (var j = 0; j < targetParts.length; ++j) { │ │ │ │ │ - splits = sourceLine.split(targetParts[j], options); │ │ │ │ │ - if (splits) { │ │ │ │ │ - if (mutual) { │ │ │ │ │ - sourceLines = splits[0]; │ │ │ │ │ - for (var k = 0, klen = sourceLines.length; k < klen; ++k) { │ │ │ │ │ - if (k === 0 && sourceParts.length) { │ │ │ │ │ - sourceParts[sourceParts.length - 1].addComponent( │ │ │ │ │ - sourceLines[k] │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - sourceParts.push( │ │ │ │ │ - new OpenLayers.Geometry.MultiLineString([ │ │ │ │ │ - sourceLines[k] │ │ │ │ │ - ]) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - sourceSplit = true; │ │ │ │ │ - splits = splits[1]; │ │ │ │ │ - } │ │ │ │ │ - if (splits.length) { │ │ │ │ │ - // splice in new target parts │ │ │ │ │ - splits.unshift(j, 1); │ │ │ │ │ - Array.prototype.splice.apply(targetParts, splits); │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!sourceSplit) { │ │ │ │ │ - // source line was not hit │ │ │ │ │ - if (sourceParts.length) { │ │ │ │ │ - // add line to existing multi │ │ │ │ │ - sourceParts[sourceParts.length - 1].addComponent( │ │ │ │ │ - sourceLine.clone() │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - // create a fresh multi │ │ │ │ │ - sourceParts = [ │ │ │ │ │ - new OpenLayers.Geometry.MultiLineString( │ │ │ │ │ - sourceLine.clone() │ │ │ │ │ - ) │ │ │ │ │ - ]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (sourceParts && sourceParts.length > 1) { │ │ │ │ │ - sourceSplit = true; │ │ │ │ │ - } else { │ │ │ │ │ - sourceParts = []; │ │ │ │ │ + /** │ │ │ │ │ + * Property: doubleTouchTolerance │ │ │ │ │ + * {Number} Maximum number of pixels between two touches for │ │ │ │ │ + * the gesture to be considered a "finalize feature" action. │ │ │ │ │ + * Default is 20. │ │ │ │ │ + */ │ │ │ │ │ + doubleTouchTolerance: 20, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: freehand │ │ │ │ │ + * {Boolean} In freehand mode, the handler starts the path on mouse down, │ │ │ │ │ + * adds a point for every mouse move, and finishes the path on mouse up. │ │ │ │ │ + * Outside of freehand mode, a point is added to the path on every mouse │ │ │ │ │ + * click and double-click finishes the path. │ │ │ │ │ + */ │ │ │ │ │ + freehand: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: freehandToggle │ │ │ │ │ + * {String} If set, freehandToggle is checked on mouse events and will set │ │ │ │ │ + * the freehand mode to the opposite of this.freehand. To disallow │ │ │ │ │ + * toggling between freehand and non-freehand mode, set freehandToggle to │ │ │ │ │ + * null. Acceptable toggle values are 'shiftKey', 'ctrlKey', and 'altKey'. │ │ │ │ │ + */ │ │ │ │ │ + freehandToggle: 'shiftKey', │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: timerId │ │ │ │ │ + * {Integer} The timer used to test the double touch. │ │ │ │ │ + */ │ │ │ │ │ + timerId: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: redoStack │ │ │ │ │ + * {Array} Stack containing points removed with <undo>. │ │ │ │ │ + */ │ │ │ │ │ + redoStack: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Handler.Path │ │ │ │ │ + * Create a new path hander │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ + * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ + * functions. Various callbacks described below. │ │ │ │ │ + * options - {Object} An optional object with properties to be set on the │ │ │ │ │ + * handler │ │ │ │ │ + * │ │ │ │ │ + * Named callbacks: │ │ │ │ │ + * create - Called when a sketch is first created. Callback called with │ │ │ │ │ + * the creation point geometry and sketch feature. │ │ │ │ │ + * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ + * geometry and the sketch feature. │ │ │ │ │ + * point - Called as each point is added. Receives the new point geometry. │ │ │ │ │ + * done - Called when the point drawing is finished. The callback will │ │ │ │ │ + * recieve a single argument, the linestring geometry. │ │ │ │ │ + * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ + * cancel callback will receive a geometry. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createFeature │ │ │ │ │ + * Add temporary geometries │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} The initial pixel location for the new │ │ │ │ │ + * feature. │ │ │ │ │ + */ │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ + lonlat.lon, lonlat.lat │ │ │ │ │ + ); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.line = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.LineString([this.point.geometry]) │ │ │ │ │ + ); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.line, this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroyFeature │ │ │ │ │ + * Destroy temporary geometries │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ + */ │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + OpenLayers.Handler.Point.prototype.destroyFeature.call( │ │ │ │ │ + this, force); │ │ │ │ │ + this.line = null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroyPersistedFeature │ │ │ │ │ + * Destroy the persisted feature. │ │ │ │ │ + */ │ │ │ │ │ + destroyPersistedFeature: function() { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + if (layer && layer.features.length > 2) { │ │ │ │ │ + this.layer.features[0].destroy(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: removePoint │ │ │ │ │ + * Destroy the temporary point. │ │ │ │ │ + */ │ │ │ │ │ + removePoint: function() { │ │ │ │ │ + if (this.point) { │ │ │ │ │ + this.layer.removeFeatures([this.point]); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: addPoint │ │ │ │ │ + * Add point to geometry. Send the point index to override │ │ │ │ │ + * the behavior of LinearRing that disregards adding duplicate points. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} The pixel location for the new point. │ │ │ │ │ + */ │ │ │ │ │ + addPoint: function(pixel) { │ │ │ │ │ + this.layer.removeFeatures([this.point]); │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat) │ │ │ │ │ + ); │ │ │ │ │ + this.line.geometry.addComponent( │ │ │ │ │ + this.point.geometry, this.line.geometry.components.length │ │ │ │ │ + ); │ │ │ │ │ + this.layer.addFeatures([this.point]); │ │ │ │ │ + this.callback("point", [this.point.geometry, this.getGeometry()]); │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + delete this.redoStack; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: insertXY │ │ │ │ │ + * Insert a point in the current sketch given x & y coordinates. The new │ │ │ │ │ + * point is inserted immediately before the most recently drawn point. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * x - {Number} The x-coordinate of the point. │ │ │ │ │ + * y - {Number} The y-coordinate of the point. │ │ │ │ │ + */ │ │ │ │ │ + insertXY: function(x, y) { │ │ │ │ │ + this.line.geometry.addComponent( │ │ │ │ │ + new OpenLayers.Geometry.Point(x, y), │ │ │ │ │ + this.getCurrentPointIndex() │ │ │ │ │ + ); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + delete this.redoStack; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: insertDeltaXY │ │ │ │ │ + * Insert a point given offsets from the previously inserted point. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * dx - {Number} The x-coordinate offset of the point. │ │ │ │ │ + * dy - {Number} The y-coordinate offset of the point. │ │ │ │ │ + */ │ │ │ │ │ + insertDeltaXY: function(dx, dy) { │ │ │ │ │ + var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ + var p0 = this.line.geometry.components[previousIndex]; │ │ │ │ │ + if (p0 && !isNaN(p0.x) && !isNaN(p0.y)) { │ │ │ │ │ + this.insertXY(p0.x + dx, p0.y + dy); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: insertDirectionLength │ │ │ │ │ + * Insert a point in the current sketch given a direction and a length. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * direction - {Number} Degrees clockwise from the positive x-axis. │ │ │ │ │ + * length - {Number} Distance from the previously drawn point. │ │ │ │ │ + */ │ │ │ │ │ + insertDirectionLength: function(direction, length) { │ │ │ │ │ + direction *= Math.PI / 180; │ │ │ │ │ + var dx = length * Math.cos(direction); │ │ │ │ │ + var dy = length * Math.sin(direction); │ │ │ │ │ + this.insertDeltaXY(dx, dy); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: insertDeflectionLength │ │ │ │ │ + * Insert a point in the current sketch given a deflection and a length. │ │ │ │ │ + * The deflection should be degrees clockwise from the previously │ │ │ │ │ + * digitized segment. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * deflection - {Number} Degrees clockwise from the previous segment. │ │ │ │ │ + * length - {Number} Distance from the previously drawn point. │ │ │ │ │ + */ │ │ │ │ │ + insertDeflectionLength: function(deflection, length) { │ │ │ │ │ + var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ + if (previousIndex > 0) { │ │ │ │ │ + var p1 = this.line.geometry.components[previousIndex]; │ │ │ │ │ + var p0 = this.line.geometry.components[previousIndex - 1]; │ │ │ │ │ + var theta = Math.atan2(p1.y - p0.y, p1.x - p0.x); │ │ │ │ │ + this.insertDirectionLength( │ │ │ │ │ + (theta * 180 / Math.PI) + deflection, length │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getCurrentPointIndex │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number} The index of the most recently drawn point. │ │ │ │ │ + */ │ │ │ │ │ + getCurrentPointIndex: function() { │ │ │ │ │ + return this.line.geometry.components.length - 1; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: undo │ │ │ │ │ + * Remove the most recently added point in the sketch geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} A point was removed. │ │ │ │ │ + */ │ │ │ │ │ + undo: function() { │ │ │ │ │ + var geometry = this.line.geometry; │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var index = this.getCurrentPointIndex() - 1; │ │ │ │ │ + var target = components[index]; │ │ │ │ │ + var undone = geometry.removeComponent(target); │ │ │ │ │ + if (undone) { │ │ │ │ │ + // On touch devices, set the current ("mouse location") point to │ │ │ │ │ + // match the last digitized point. │ │ │ │ │ + if (this.touch && index > 0) { │ │ │ │ │ + components = geometry.components; // safety │ │ │ │ │ + var lastpt = components[index - 1]; │ │ │ │ │ + var curptidx = this.getCurrentPointIndex(); │ │ │ │ │ + var curpt = components[curptidx]; │ │ │ │ │ + curpt.x = lastpt.x; │ │ │ │ │ + curpt.y = lastpt.y; │ │ │ │ │ } │ │ │ │ │ - if (targetParts && targetParts.length > 1) { │ │ │ │ │ - targetSplit = true; │ │ │ │ │ - } else { │ │ │ │ │ - targetParts = []; │ │ │ │ │ + if (!this.redoStack) { │ │ │ │ │ + this.redoStack = []; │ │ │ │ │ } │ │ │ │ │ - if (sourceSplit || targetSplit) { │ │ │ │ │ - if (mutual) { │ │ │ │ │ - results = [sourceParts, targetParts]; │ │ │ │ │ - } else { │ │ │ │ │ - results = targetParts; │ │ │ │ │ - } │ │ │ │ │ + this.redoStack.push(target); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + } │ │ │ │ │ + return undone; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: redo │ │ │ │ │ + * Reinsert the most recently removed point resulting from an <undo> call. │ │ │ │ │ + * The undo stack is deleted whenever a point is added by other means. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} A point was added. │ │ │ │ │ + */ │ │ │ │ │ + redo: function() { │ │ │ │ │ + var target = this.redoStack && this.redoStack.pop(); │ │ │ │ │ + if (target) { │ │ │ │ │ + this.line.geometry.addComponent(target, this.getCurrentPointIndex()); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + } │ │ │ │ │ + return !!target; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: freehandMode │ │ │ │ │ + * Determine whether to behave in freehand mode or not. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + freehandMode: function(evt) { │ │ │ │ │ + return (this.freehandToggle && evt[this.freehandToggle]) ? │ │ │ │ │ + !this.freehand : this.freehand; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: modifyFeature │ │ │ │ │ + * Modify the existing geometry given the new point │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} The updated pixel location for the latest │ │ │ │ │ + * point. │ │ │ │ │ + * drawing - {Boolean} Indicate if we're currently drawing. │ │ │ │ │ + */ │ │ │ │ │ + modifyFeature: function(pixel, drawing) { │ │ │ │ │ + if (!this.line) { │ │ │ │ │ + this.createFeature(pixel); │ │ │ │ │ + } │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point.geometry.x = lonlat.lon; │ │ │ │ │ + this.point.geometry.y = lonlat.lat; │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.getSketch(), drawing]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawFeature │ │ │ │ │ + * Render geometries on the temporary layer. │ │ │ │ │ + */ │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.line, this.style); │ │ │ │ │ + this.layer.drawFeature(this.point, this.style); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getSketch │ │ │ │ │ + * Return the sketch feature. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + getSketch: function() { │ │ │ │ │ + return this.line; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getGeometry │ │ │ │ │ + * Return the sketch geometry. If <multi> is true, this will return │ │ │ │ │ + * a multi-part geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.LineString>} │ │ │ │ │ + */ │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.line && this.line.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiLineString([geometry]); │ │ │ │ │ + } │ │ │ │ │ + return geometry; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * method: touchstart │ │ │ │ │ + * handle touchstart. │ │ │ │ │ + * │ │ │ │ │ + * parameters: │ │ │ │ │ + * evt - {event} the browser event │ │ │ │ │ + * │ │ │ │ │ + * returns: │ │ │ │ │ + * {boolean} allow event propagation │ │ │ │ │ + */ │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + if (this.timerId && │ │ │ │ │ + this.passesTolerance(this.lastTouchPx, evt.xy, │ │ │ │ │ + this.doubleTouchTolerance)) { │ │ │ │ │ + // double-tap, finalize the geometry │ │ │ │ │ + this.finishGeometry(); │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + return false; │ │ │ │ │ + } else { │ │ │ │ │ + if (this.timerId) { │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ } │ │ │ │ │ - return results; │ │ │ │ │ - }, │ │ │ │ │ + this.timerId = window.setTimeout( │ │ │ │ │ + OpenLayers.Function.bind(function() { │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + }, this), 300); │ │ │ │ │ + return OpenLayers.Handler.Point.prototype.touchstart.call(this, evt); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: splitWith │ │ │ │ │ - * Split this geometry (the target) with the given geometry (the source). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} A geometry used to split this │ │ │ │ │ - * geometry (the source). │ │ │ │ │ - * options - {Object} Properties of this object will be used to determine │ │ │ │ │ - * how the split is conducted. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * mutual - {Boolean} Split the source geometry in addition to the target │ │ │ │ │ - * geometry. Default is false. │ │ │ │ │ - * edge - {Boolean} Allow splitting when only edges intersect. Default is │ │ │ │ │ - * true. If false, a vertex on the source must be within the tolerance │ │ │ │ │ - * distance of the intersection to be considered a split. │ │ │ │ │ - * tolerance - {Number} If a non-null value is provided, intersections │ │ │ │ │ - * within the tolerance distance of an existing vertex on the source │ │ │ │ │ - * will be assumed to occur at the vertex. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} A list of geometries (of this same type as the target) that │ │ │ │ │ - * result from splitting the target with the source geometry. The │ │ │ │ │ - * source and target geometry will remain unmodified. If no split │ │ │ │ │ - * results, null will be returned. If mutual is true and a split │ │ │ │ │ - * results, return will be an array of two arrays - the first will be │ │ │ │ │ - * all geometries that result from splitting the source geometry and │ │ │ │ │ - * the second will be all geometries that result from splitting the │ │ │ │ │ - * target geometry. │ │ │ │ │ - */ │ │ │ │ │ - splitWith: function(geometry, options) { │ │ │ │ │ - var results = null; │ │ │ │ │ - var mutual = options && options.mutual; │ │ │ │ │ - var splits, targetLine, sourceLines, sourceSplit, targetSplit, sourceParts, targetParts; │ │ │ │ │ - if (geometry instanceof OpenLayers.Geometry.LineString) { │ │ │ │ │ - targetParts = []; │ │ │ │ │ - sourceParts = [geometry]; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ - targetSplit = false; │ │ │ │ │ - targetLine = this.components[i]; │ │ │ │ │ - for (var j = 0; j < sourceParts.length; ++j) { │ │ │ │ │ - splits = sourceParts[j].split(targetLine, options); │ │ │ │ │ - if (splits) { │ │ │ │ │ - if (mutual) { │ │ │ │ │ - sourceLines = splits[0]; │ │ │ │ │ - if (sourceLines.length) { │ │ │ │ │ - // splice in new source parts │ │ │ │ │ - sourceLines.unshift(j, 1); │ │ │ │ │ - Array.prototype.splice.apply(sourceParts, sourceLines); │ │ │ │ │ - j += sourceLines.length - 2; │ │ │ │ │ - } │ │ │ │ │ - splits = splits[1]; │ │ │ │ │ - if (splits.length === 0) { │ │ │ │ │ - splits = [targetLine.clone()]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - for (var k = 0, klen = splits.length; k < klen; ++k) { │ │ │ │ │ - if (k === 0 && targetParts.length) { │ │ │ │ │ - targetParts[targetParts.length - 1].addComponent( │ │ │ │ │ - splits[k] │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - targetParts.push( │ │ │ │ │ - new OpenLayers.Geometry.MultiLineString([ │ │ │ │ │ - splits[k] │ │ │ │ │ - ]) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - targetSplit = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!targetSplit) { │ │ │ │ │ - // target component was not hit │ │ │ │ │ - if (targetParts.length) { │ │ │ │ │ - // add it to any existing multi-line │ │ │ │ │ - targetParts[targetParts.length - 1].addComponent( │ │ │ │ │ - targetLine.clone() │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - // or start with a fresh multi-line │ │ │ │ │ - targetParts = [ │ │ │ │ │ - new OpenLayers.Geometry.MultiLineString([ │ │ │ │ │ - targetLine.clone() │ │ │ │ │ - ]) │ │ │ │ │ - ]; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: down │ │ │ │ │ + * Handle mousedown and touchstart. Add a new point to the geometry and │ │ │ │ │ + * render it. Return determines whether to propagate the event on the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ + */ │ │ │ │ │ + down: function(evt) { │ │ │ │ │ + var stopDown = this.stopDown; │ │ │ │ │ + if (this.freehandMode(evt)) { │ │ │ │ │ + stopDown = true; │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!this.touch && (!this.lastDown || │ │ │ │ │ + !this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ + this.pixelTolerance))) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ + } │ │ │ │ │ + this.mouseDown = true; │ │ │ │ │ + this.lastDown = evt.xy; │ │ │ │ │ + this.stoppedDown = stopDown; │ │ │ │ │ + return !stopDown; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - results = geometry.split(this); │ │ │ │ │ + /** │ │ │ │ │ + * Method: move │ │ │ │ │ + * Handle mousemove and touchmove. Adjust the geometry and redraw. │ │ │ │ │ + * Return determines whether to propagate the event on the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ + */ │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature(); │ │ │ │ │ } │ │ │ │ │ - if (sourceParts && sourceParts.length > 1) { │ │ │ │ │ - sourceSplit = true; │ │ │ │ │ + if (this.maxVertices && this.line && │ │ │ │ │ + this.line.geometry.components.length === this.maxVertices) { │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize(); │ │ │ │ │ } else { │ │ │ │ │ - sourceParts = []; │ │ │ │ │ + this.addPoint(evt.xy); │ │ │ │ │ } │ │ │ │ │ - if (targetParts && targetParts.length > 1) { │ │ │ │ │ - targetSplit = true; │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: up │ │ │ │ │ + * Handle mouseup and touchend. Send the latest point in the geometry to │ │ │ │ │ + * the control. Return determines whether to propagate the event on the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ + */ │ │ │ │ │ + up: function(evt) { │ │ │ │ │ + if (this.mouseDown && (!this.lastUp || !this.lastUp.equals(evt.xy))) { │ │ │ │ │ + if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature(); │ │ │ │ │ + } │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize(); │ │ │ │ │ } else { │ │ │ │ │ - targetParts = []; │ │ │ │ │ - } │ │ │ │ │ - if (sourceSplit || targetSplit) { │ │ │ │ │ - if (mutual) { │ │ │ │ │ - results = [sourceParts, targetParts]; │ │ │ │ │ - } else { │ │ │ │ │ - results = targetParts; │ │ │ │ │ + if (this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ + this.pixelTolerance)) { │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy); │ │ │ │ │ + } │ │ │ │ │ + if (this.lastUp == null && this.persist) { │ │ │ │ │ + this.destroyPersistedFeature(); │ │ │ │ │ + } │ │ │ │ │ + this.addPoint(evt.xy); │ │ │ │ │ + this.lastUp = evt.xy; │ │ │ │ │ + if (this.line.geometry.components.length === this.maxVertices + 1) { │ │ │ │ │ + this.finishGeometry(); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return results; │ │ │ │ │ - }, │ │ │ │ │ + } │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + return !this.stopUp; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Geometry.MultiLineString" │ │ │ │ │ - }); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: finishGeometry │ │ │ │ │ + * Finish the geometry and send it back to the control. │ │ │ │ │ + */ │ │ │ │ │ + finishGeometry: function() { │ │ │ │ │ + var index = this.line.geometry.components.length - 1; │ │ │ │ │ + this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: dblclick │ │ │ │ │ + * Handle double-clicks. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ + */ │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + if (!this.freehandMode(evt)) { │ │ │ │ │ + this.finishGeometry(); │ │ │ │ │ + } │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Path" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Geometry/LinearRing.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ @@ -21363,140 +18267,13659 @@ │ │ │ │ │ y = origin.y + (radius * Math.sin(rotatedAngle)); │ │ │ │ │ points.push(new OpenLayers.Geometry.Point(x, y)); │ │ │ │ │ } │ │ │ │ │ var ring = new OpenLayers.Geometry.LinearRing(points); │ │ │ │ │ return new OpenLayers.Geometry.Polygon([ring]); │ │ │ │ │ }; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Geometry/MultiPolygon.js │ │ │ │ │ + OpenLayers/Handler/Polygon.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Geometry/Collection.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Geometry.MultiPolygon │ │ │ │ │ - * MultiPolygon is a geometry with multiple <OpenLayers.Geometry.Polygon> │ │ │ │ │ - * components. Create a new instance with the <OpenLayers.Geometry.MultiPolygon> │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Geometry.Collection> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Geometry.MultiPolygon = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Geometry.Collection, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: componentTypes │ │ │ │ │ - * {Array(String)} An array of class names representing the types of │ │ │ │ │ - * components that the collection can include. A null value means the │ │ │ │ │ - * component types are not restricted. │ │ │ │ │ - */ │ │ │ │ │ - componentTypes: ["OpenLayers.Geometry.Polygon"], │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Geometry.MultiPolygon │ │ │ │ │ - * Create a new MultiPolygon geometry │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * components - {Array(<OpenLayers.Geometry.Polygon>)} An array of polygons │ │ │ │ │ - * used to generate the MultiPolygon │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Geometry.MultiPolygon" │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/GeoJSON.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/JSON.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/MultiPoint.js │ │ │ │ │ - * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ - * @requires OpenLayers/Geometry/MultiLineString.js │ │ │ │ │ + * @requires OpenLayers/Handler/Path.js │ │ │ │ │ * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ - * @requires OpenLayers/Geometry/MultiPolygon.js │ │ │ │ │ - * @requires OpenLayers/Console.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.GeoJSON │ │ │ │ │ - * Read and write GeoJSON. Create a new parser with the │ │ │ │ │ - * <OpenLayers.Format.GeoJSON> constructor. │ │ │ │ │ + * Class: OpenLayers.Handler.Polygon │ │ │ │ │ + * Handler to draw a polygon on the map. Polygon is displayed on mouse down, │ │ │ │ │ + * moves on mouse move, and is finished on mouse up. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.JSON> │ │ │ │ │ + * - <OpenLayers.Handler.Path> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.GeoJSON = OpenLayers.Class(OpenLayers.Format.JSON, { │ │ │ │ │ +OpenLayers.Handler.Polygon = OpenLayers.Class(OpenLayers.Handler.Path, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: holeModifier │ │ │ │ │ + * {String} Key modifier to trigger hole digitizing. Acceptable values are │ │ │ │ │ + * "altKey", "shiftKey", or "ctrlKey". If not set, no hole digitizing │ │ │ │ │ + * will take place. Default is null. │ │ │ │ │ + */ │ │ │ │ │ + holeModifier: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: ignoreExtraDims │ │ │ │ │ - * {Boolean} Ignore dimensions higher than 2 when reading geometry │ │ │ │ │ - * coordinates. │ │ │ │ │ + * Property: drawingHole │ │ │ │ │ + * {Boolean} Currently drawing an interior ring. │ │ │ │ │ */ │ │ │ │ │ - ignoreExtraDims: false, │ │ │ │ │ + drawingHole: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.GeoJSON │ │ │ │ │ - * Create a new parser for GeoJSON. │ │ │ │ │ + * Property: polygon │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + polygon: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Handler.Polygon │ │ │ │ │ + * Create a Polygon Handler. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ + * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ + * functions. Various callbacks described below. │ │ │ │ │ + * options - {Object} An optional object with properties to be set on the │ │ │ │ │ + * handler │ │ │ │ │ + * │ │ │ │ │ + * Named callbacks: │ │ │ │ │ + * create - Called when a sketch is first created. Callback called with │ │ │ │ │ + * the creation point geometry and sketch feature. │ │ │ │ │ + * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ + * geometry and the sketch feature. │ │ │ │ │ + * point - Called as each point is added. Receives the new point geometry. │ │ │ │ │ + * done - Called when the point drawing is finished. The callback will │ │ │ │ │ + * recieve a single argument, the polygon geometry. │ │ │ │ │ + * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ + * cancel callback will receive a geometry. │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Deserialize a GeoJSON string. │ │ │ │ │ + * Method: createFeature │ │ │ │ │ + * Add temporary geometries │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * json - {String} A GeoJSON string │ │ │ │ │ - * type - {String} Optional string that determines the structure of │ │ │ │ │ - * the output. Supported values are "Geometry", "Feature", and │ │ │ │ │ - * "FeatureCollection". If absent or null, a default of │ │ │ │ │ - * "FeatureCollection" is assumed. │ │ │ │ │ - * filter - {Function} A function which will be called for every key and │ │ │ │ │ - * value at every level of the final result. Each value will be │ │ │ │ │ - * replaced by the result of the filter function. This can be used to │ │ │ │ │ - * reform generic objects into instances of classes, or to transform │ │ │ │ │ - * date strings into Date objects. │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} The initial pixel location for the new │ │ │ │ │ + * feature. │ │ │ │ │ + */ │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ + lonlat.lon, lonlat.lat │ │ │ │ │ + ); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.line = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.LinearRing([this.point.geometry]) │ │ │ │ │ + ); │ │ │ │ │ + this.polygon = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.Polygon([this.line.geometry]) │ │ │ │ │ + ); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.polygon, this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: addPoint │ │ │ │ │ + * Add point to geometry. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The return depends on the value of the type argument. If type │ │ │ │ │ - * is "FeatureCollection" (the default), the return will be an array │ │ │ │ │ - * of <OpenLayers.Feature.Vector>. If type is "Geometry", the input json │ │ │ │ │ - * must represent a single geometry, and the return will be an │ │ │ │ │ - * <OpenLayers.Geometry>. If type is "Feature", the input json must │ │ │ │ │ - * represent a single feature, and the return will be an │ │ │ │ │ - * <OpenLayers.Feature.Vector>. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} The pixel location for the new point. │ │ │ │ │ */ │ │ │ │ │ - read: function(json, type, filter) { │ │ │ │ │ - type = (type) ? type : "FeatureCollection"; │ │ │ │ │ - var results = null; │ │ │ │ │ - var obj = null; │ │ │ │ │ - if (typeof json == "string") { │ │ │ │ │ - obj = OpenLayers.Format.JSON.prototype.read.apply(this, │ │ │ │ │ - [json, filter]); │ │ │ │ │ - } else { │ │ │ │ │ - obj = json; │ │ │ │ │ + addPoint: function(pixel) { │ │ │ │ │ + if (!this.drawingHole && this.holeModifier && │ │ │ │ │ + this.evt && this.evt[this.holeModifier]) { │ │ │ │ │ + var geometry = this.point.geometry; │ │ │ │ │ + var features = this.control.layer.features; │ │ │ │ │ + var candidate, polygon; │ │ │ │ │ + // look for intersections, last drawn gets priority │ │ │ │ │ + for (var i = features.length - 1; i >= 0; --i) { │ │ │ │ │ + candidate = features[i].geometry; │ │ │ │ │ + if ((candidate instanceof OpenLayers.Geometry.Polygon || │ │ │ │ │ + candidate instanceof OpenLayers.Geometry.MultiPolygon) && │ │ │ │ │ + candidate.intersects(geometry)) { │ │ │ │ │ + polygon = features[i]; │ │ │ │ │ + this.control.layer.removeFeatures([polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.control.layer.events.registerPriority( │ │ │ │ │ + "sketchcomplete", this, this.finalizeInteriorRing │ │ │ │ │ + ); │ │ │ │ │ + this.control.layer.events.registerPriority( │ │ │ │ │ + "sketchmodified", this, this.enforceTopology │ │ │ │ │ + ); │ │ │ │ │ + polygon.geometry.addComponent(this.line.geometry); │ │ │ │ │ + this.polygon = polygon; │ │ │ │ │ + this.drawingHole = true; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Handler.Path.prototype.addPoint.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getCurrentPointIndex │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number} The index of the most recently drawn point. │ │ │ │ │ + */ │ │ │ │ │ + getCurrentPointIndex: function() { │ │ │ │ │ + return this.line.geometry.components.length - 2; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: enforceTopology │ │ │ │ │ + * Simple topology enforcement for drawing interior rings. Ensures vertices │ │ │ │ │ + * of interior rings are contained by exterior ring. Other topology │ │ │ │ │ + * rules are enforced in <finalizeInteriorRing> to allow drawing of │ │ │ │ │ + * rings that intersect only during the sketch (e.g. a "C" shaped ring │ │ │ │ │ + * that nearly encloses another ring). │ │ │ │ │ + */ │ │ │ │ │ + enforceTopology: function(event) { │ │ │ │ │ + var point = event.vertex; │ │ │ │ │ + var components = this.line.geometry.components; │ │ │ │ │ + // ensure that vertices of interior ring are contained by exterior ring │ │ │ │ │ + if (!this.polygon.geometry.intersects(point)) { │ │ │ │ │ + var last = components[components.length - 3]; │ │ │ │ │ + point.x = last.x; │ │ │ │ │ + point.y = last.y; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: finishGeometry │ │ │ │ │ + * Finish the geometry and send it back to the control. │ │ │ │ │ + */ │ │ │ │ │ + finishGeometry: function() { │ │ │ │ │ + var index = this.line.geometry.components.length - 2; │ │ │ │ │ + this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: finalizeInteriorRing │ │ │ │ │ + * Enforces that new ring has some area and doesn't contain vertices of any │ │ │ │ │ + * other rings. │ │ │ │ │ + */ │ │ │ │ │ + finalizeInteriorRing: function() { │ │ │ │ │ + var ring = this.line.geometry; │ │ │ │ │ + // ensure that ring has some area │ │ │ │ │ + var modified = (ring.getArea() !== 0); │ │ │ │ │ + if (modified) { │ │ │ │ │ + // ensure that new ring doesn't intersect any other rings │ │ │ │ │ + var rings = this.polygon.geometry.components; │ │ │ │ │ + for (var i = rings.length - 2; i >= 0; --i) { │ │ │ │ │ + if (ring.intersects(rings[i])) { │ │ │ │ │ + modified = false; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (modified) { │ │ │ │ │ + // ensure that new ring doesn't contain any other rings │ │ │ │ │ + var target; │ │ │ │ │ + outer: for (var i = rings.length - 2; i > 0; --i) { │ │ │ │ │ + var points = rings[i].components; │ │ │ │ │ + for (var j = 0, jj = points.length; j < jj; ++j) { │ │ │ │ │ + if (ring.containsPoint(points[j])) { │ │ │ │ │ + modified = false; │ │ │ │ │ + break outer; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (modified) { │ │ │ │ │ + if (this.polygon.state !== OpenLayers.State.INSERT) { │ │ │ │ │ + this.polygon.state = OpenLayers.State.UPDATE; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this.polygon.geometry.removeComponent(ring); │ │ │ │ │ + } │ │ │ │ │ + this.restoreFeature(); │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: cancel │ │ │ │ │ + * Finish the geometry and call the "cancel" callback. │ │ │ │ │ + */ │ │ │ │ │ + cancel: function() { │ │ │ │ │ + if (this.drawingHole) { │ │ │ │ │ + this.polygon.geometry.removeComponent(this.line.geometry); │ │ │ │ │ + this.restoreFeature(true); │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Handler.Path.prototype.cancel.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: restoreFeature │ │ │ │ │ + * Move the feature from the sketch layer to the target layer. │ │ │ │ │ + * │ │ │ │ │ + * Properties: │ │ │ │ │ + * cancel - {Boolean} Cancel drawing. If falsey, the "sketchcomplete" event │ │ │ │ │ + * will be fired. │ │ │ │ │ + */ │ │ │ │ │ + restoreFeature: function(cancel) { │ │ │ │ │ + this.control.layer.events.unregister( │ │ │ │ │ + "sketchcomplete", this, this.finalizeInteriorRing │ │ │ │ │ + ); │ │ │ │ │ + this.control.layer.events.unregister( │ │ │ │ │ + "sketchmodified", this, this.enforceTopology │ │ │ │ │ + ); │ │ │ │ │ + this.layer.removeFeatures([this.polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.control.layer.addFeatures([this.polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.drawingHole = false; │ │ │ │ │ + if (!cancel) { │ │ │ │ │ + // Re-trigger "sketchcomplete" so other listeners can do their │ │ │ │ │ + // business. While this is somewhat sloppy (if a listener is │ │ │ │ │ + // registered with registerPriority - not common - between the start │ │ │ │ │ + // and end of a single ring drawing - very uncommon - it will be │ │ │ │ │ + // called twice). │ │ │ │ │ + // TODO: In 3.0, collapse sketch handlers into geometry specific │ │ │ │ │ + // drawing controls. │ │ │ │ │ + this.control.layer.events.triggerEvent( │ │ │ │ │ + "sketchcomplete", { │ │ │ │ │ + feature: this.polygon │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroyFeature │ │ │ │ │ + * Destroy temporary geometries │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ + */ │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + OpenLayers.Handler.Path.prototype.destroyFeature.call( │ │ │ │ │ + this, force); │ │ │ │ │ + this.polygon = null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawFeature │ │ │ │ │ + * Render geometries on the temporary layer. │ │ │ │ │ + */ │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.polygon, this.style); │ │ │ │ │ + this.layer.drawFeature(this.point, this.style); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getSketch │ │ │ │ │ + * Return the sketch feature. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + getSketch: function() { │ │ │ │ │ + return this.polygon; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getGeometry │ │ │ │ │ + * Return the sketch geometry. If <multi> is true, this will return │ │ │ │ │ + * a multi-part geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Polygon>} │ │ │ │ │ + */ │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.polygon && this.polygon.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiPolygon([geometry]); │ │ │ │ │ + } │ │ │ │ │ + return geometry; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Polygon" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Renderer.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Renderer │ │ │ │ │ + * This is the base class for all renderers. │ │ │ │ │ + * │ │ │ │ │ + * This is based on a merger code written by Paul Spencer and Bertil Chapuis. │ │ │ │ │ + * It is largely composed of virtual functions that are to be implemented │ │ │ │ │ + * in technology-specific subclasses, but there is some generic code too. │ │ │ │ │ + * │ │ │ │ │ + * The functions that *are* implemented here merely deal with the maintenance │ │ │ │ │ + * of the size and extent variables, as well as the cached 'resolution' │ │ │ │ │ + * value. │ │ │ │ │ + * │ │ │ │ │ + * A note to the user that all subclasses should use getResolution() instead │ │ │ │ │ + * of directly accessing this.resolution in order to correctly use the │ │ │ │ │ + * cacheing system. │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: container │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + container: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: root │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + root: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: extent │ │ │ │ │ + * {<OpenLayers.Bounds>} │ │ │ │ │ + */ │ │ │ │ │ + extent: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: locked │ │ │ │ │ + * {Boolean} If the renderer is currently in a state where many things │ │ │ │ │ + * are changing, the 'locked' property is set to true. This means │ │ │ │ │ + * that renderers can expect at least one more drawFeature event to be │ │ │ │ │ + * called with the 'locked' property set to 'true': In some renderers, │ │ │ │ │ + * this might make sense to use as a 'only update local information' │ │ │ │ │ + * flag. │ │ │ │ │ + */ │ │ │ │ │ + locked: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: size │ │ │ │ │ + * {<OpenLayers.Size>} │ │ │ │ │ + */ │ │ │ │ │ + size: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: resolution │ │ │ │ │ + * {Float} cache of current map resolution │ │ │ │ │ + */ │ │ │ │ │ + resolution: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: map │ │ │ │ │ + * {<OpenLayers.Map>} Reference to the map -- this is set in Vector's setMap() │ │ │ │ │ + */ │ │ │ │ │ + map: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: featureDx │ │ │ │ │ + * {Number} Feature offset in x direction. Will be calculated for and │ │ │ │ │ + * applied to the current feature while rendering (see │ │ │ │ │ + * <calculateFeatureDx>). │ │ │ │ │ + */ │ │ │ │ │ + featureDx: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Renderer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * containerID - {<String>} │ │ │ │ │ + * options - {Object} options for this renderer. See sublcasses for │ │ │ │ │ + * supported options. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(containerID, options) { │ │ │ │ │ + this.container = OpenLayers.Util.getElement(containerID); │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.container = null; │ │ │ │ │ + this.extent = null; │ │ │ │ │ + this.size = null; │ │ │ │ │ + this.resolution = null; │ │ │ │ │ + this.map = null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: supported │ │ │ │ │ + * This should be overridden by specific subclasses │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the browser supports the renderer class │ │ │ │ │ + */ │ │ │ │ │ + supported: function() { │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setExtent │ │ │ │ │ + * Set the visible part of the layer. │ │ │ │ │ + * │ │ │ │ │ + * Resolution has probably changed, so we nullify the resolution │ │ │ │ │ + * cache (this.resolution) -- this way it will be re-computed when │ │ │ │ │ + * next it is needed. │ │ │ │ │ + * We nullify the resolution cache (this.resolution) if resolutionChanged │ │ │ │ │ + * is set to true - this way it will be re-computed on the next │ │ │ │ │ + * getResolution() request. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} │ │ │ │ │ + * resolutionChanged - {Boolean} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ + * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ + * False otherwise. │ │ │ │ │ + */ │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + this.extent = extent.clone(); │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + var ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ + extent = extent.scale(1 / ratio); │ │ │ │ │ + this.extent = extent.wrapDateLine(this.map.getMaxExtent()).scale(ratio); │ │ │ │ │ + } │ │ │ │ │ + if (resolutionChanged) { │ │ │ │ │ + this.resolution = null; │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setSize │ │ │ │ │ + * Sets the size of the drawing surface. │ │ │ │ │ + * │ │ │ │ │ + * Resolution has probably changed, so we nullify the resolution │ │ │ │ │ + * cache (this.resolution) -- this way it will be re-computed when │ │ │ │ │ + * next it is needed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * size - {<OpenLayers.Size>} │ │ │ │ │ + */ │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + this.size = size.clone(); │ │ │ │ │ + this.resolution = null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getResolution │ │ │ │ │ + * Uses cached copy of resolution if available to minimize computing │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The current map's resolution │ │ │ │ │ + */ │ │ │ │ │ + getResolution: function() { │ │ │ │ │ + this.resolution = this.resolution || this.map.getResolution(); │ │ │ │ │ + return this.resolution; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawFeature │ │ │ │ │ + * Draw the feature. The optional style argument can be used │ │ │ │ │ + * to override the feature's own style. This method should only │ │ │ │ │ + * be called from layer.drawFeature(). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * style - {<Object>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the feature has been drawn completely, false if not, │ │ │ │ │ + * undefined if the feature had no geometry │ │ │ │ │ + */ │ │ │ │ │ + drawFeature: function(feature, style) { │ │ │ │ │ + if (style == null) { │ │ │ │ │ + style = feature.style; │ │ │ │ │ + } │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + var bounds = feature.geometry.getBounds(); │ │ │ │ │ + if (bounds) { │ │ │ │ │ + var worldBounds; │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + worldBounds = this.map.getMaxExtent(); │ │ │ │ │ + } │ │ │ │ │ + if (!bounds.intersectsBounds(this.extent, { │ │ │ │ │ + worldBounds: worldBounds │ │ │ │ │ + })) { │ │ │ │ │ + style = { │ │ │ │ │ + display: "none" │ │ │ │ │ + }; │ │ │ │ │ + } else { │ │ │ │ │ + this.calculateFeatureDx(bounds, worldBounds); │ │ │ │ │ + } │ │ │ │ │ + var rendered = this.drawGeometry(feature.geometry, style, feature.id); │ │ │ │ │ + if (style.display != "none" && style.label && rendered !== false) { │ │ │ │ │ + │ │ │ │ │ + var location = feature.geometry.getCentroid(); │ │ │ │ │ + if (style.labelXOffset || style.labelYOffset) { │ │ │ │ │ + var xOffset = isNaN(style.labelXOffset) ? 0 : style.labelXOffset; │ │ │ │ │ + var yOffset = isNaN(style.labelYOffset) ? 0 : style.labelYOffset; │ │ │ │ │ + var res = this.getResolution(); │ │ │ │ │ + location.move(xOffset * res, yOffset * res); │ │ │ │ │ + } │ │ │ │ │ + this.drawText(feature.id, style, location); │ │ │ │ │ + } else { │ │ │ │ │ + this.removeText(feature.id); │ │ │ │ │ + } │ │ │ │ │ + return rendered; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateFeatureDx │ │ │ │ │ + * {Number} Calculates the feature offset in x direction. Looking at the │ │ │ │ │ + * center of the feature bounds and the renderer extent, we calculate how │ │ │ │ │ + * many world widths the two are away from each other. This distance is │ │ │ │ │ + * used to shift the feature as close as possible to the center of the │ │ │ │ │ + * current enderer extent, which ensures that the feature is visible in the │ │ │ │ │ + * current viewport. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} Bounds of the feature │ │ │ │ │ + * worldBounds - {<OpenLayers.Bounds>} Bounds of the world │ │ │ │ │ + */ │ │ │ │ │ + calculateFeatureDx: function(bounds, worldBounds) { │ │ │ │ │ + this.featureDx = 0; │ │ │ │ │ + if (worldBounds) { │ │ │ │ │ + var worldWidth = worldBounds.getWidth(), │ │ │ │ │ + rendererCenterX = (this.extent.left + this.extent.right) / 2, │ │ │ │ │ + featureCenterX = (bounds.left + bounds.right) / 2, │ │ │ │ │ + worldsAway = Math.round((featureCenterX - rendererCenterX) / worldWidth); │ │ │ │ │ + this.featureDx = worldsAway * worldWidth; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawGeometry │ │ │ │ │ + * │ │ │ │ │ + * Draw a geometry. This should only be called from the renderer itself. │ │ │ │ │ + * Use layer.drawFeature() from outside the renderer. │ │ │ │ │ + * virtual function │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {<String>} │ │ │ │ │ + */ │ │ │ │ │ + drawGeometry: function(geometry, style, featureId) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawText │ │ │ │ │ + * Function for drawing text labels. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * style - │ │ │ │ │ + * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + */ │ │ │ │ │ + drawText: function(featureId, style, location) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeText │ │ │ │ │ + * Function for removing text labels. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + removeText: function(featureId) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clear │ │ │ │ │ + * Clear all vectors from the renderer. │ │ │ │ │ + * virtual function. │ │ │ │ │ + */ │ │ │ │ │ + clear: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getFeatureIdFromEvent │ │ │ │ │ + * Returns a feature id from an event on the renderer. │ │ │ │ │ + * How this happens is specific to the renderer. This should be │ │ │ │ │ + * called from layer.getFeatureFromEvent(). │ │ │ │ │ + * Virtual function. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A feature id or undefined. │ │ │ │ │ + */ │ │ │ │ │ + getFeatureIdFromEvent: function(evt) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: eraseFeatures │ │ │ │ │ + * This is called by the layer to erase features │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + */ │ │ │ │ │ + eraseFeatures: function(features) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + this.eraseGeometry(feature.geometry, feature.id); │ │ │ │ │ + this.removeText(feature.id); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: eraseGeometry │ │ │ │ │ + * Remove a geometry from the renderer (by id). │ │ │ │ │ + * virtual function. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + eraseGeometry: function(geometry, featureId) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveRoot │ │ │ │ │ + * moves this renderer's root to a (different) renderer. │ │ │ │ │ + * To be implemented by subclasses that require a common renderer root for │ │ │ │ │ + * feature selection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * renderer - {<OpenLayers.Renderer>} target renderer for the moved root │ │ │ │ │ + */ │ │ │ │ │ + moveRoot: function(renderer) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getRenderLayerId │ │ │ │ │ + * Gets the layer that this renderer's output appears on. If moveRoot was │ │ │ │ │ + * used, this will be different from the id of the layer containing the │ │ │ │ │ + * features rendered by this renderer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} the id of the output layer. │ │ │ │ │ + */ │ │ │ │ │ + getRenderLayerId: function() { │ │ │ │ │ + return this.container.id; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: applyDefaultSymbolizer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * symbolizer - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ + applyDefaultSymbolizer: function(symbolizer) { │ │ │ │ │ + var result = OpenLayers.Util.extend({}, │ │ │ │ │ + OpenLayers.Renderer.defaultSymbolizer); │ │ │ │ │ + if (symbolizer.stroke === false) { │ │ │ │ │ + delete result.strokeWidth; │ │ │ │ │ + delete result.strokeColor; │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.fill === false) { │ │ │ │ │ + delete result.fillColor; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.extend(result, symbolizer); │ │ │ │ │ + return result; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.defaultSymbolizer │ │ │ │ │ + * {Object} Properties from this symbolizer will be applied to symbolizers │ │ │ │ │ + * with missing properties. This can also be used to set a global │ │ │ │ │ + * symbolizer default in OpenLayers. To be SLD 1.x compliant, add the │ │ │ │ │ + * following code before rendering any vector features: │ │ │ │ │ + * (code) │ │ │ │ │ + * OpenLayers.Renderer.defaultSymbolizer = { │ │ │ │ │ + * fillColor: "#808080", │ │ │ │ │ + * fillOpacity: 1, │ │ │ │ │ + * strokeColor: "#000000", │ │ │ │ │ + * strokeOpacity: 1, │ │ │ │ │ + * strokeWidth: 1, │ │ │ │ │ + * pointRadius: 3, │ │ │ │ │ + * graphicName: "square" │ │ │ │ │ + * }; │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.defaultSymbolizer = { │ │ │ │ │ + fillColor: "#000000", │ │ │ │ │ + strokeColor: "#000000", │ │ │ │ │ + strokeWidth: 2, │ │ │ │ │ + fillOpacity: 1, │ │ │ │ │ + strokeOpacity: 1, │ │ │ │ │ + pointRadius: 0, │ │ │ │ │ + labelAlign: 'cm' │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.symbol │ │ │ │ │ + * Coordinate arrays for well known (named) symbols. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.symbol = { │ │ │ │ │ + "star": [350, 75, 379, 161, 469, 161, 397, 215, 423, 301, 350, 250, 277, 301, │ │ │ │ │ + 303, 215, 231, 161, 321, 161, 350, 75 │ │ │ │ │ + ], │ │ │ │ │ + "cross": [4, 0, 6, 0, 6, 4, 10, 4, 10, 6, 6, 6, 6, 10, 4, 10, 4, 6, 0, 6, 0, 4, 4, 4, │ │ │ │ │ + 4, 0 │ │ │ │ │ + ], │ │ │ │ │ + "x": [0, 0, 25, 0, 50, 35, 75, 0, 100, 0, 65, 50, 100, 100, 75, 100, 50, 65, 25, 100, 0, 100, 35, 50, 0, 0], │ │ │ │ │ + "square": [0, 0, 0, 1, 1, 1, 1, 0, 0, 0], │ │ │ │ │ + "triangle": [0, 10, 10, 10, 5, 0, 0, 10] │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Renderer/Canvas.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Renderer.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Renderer.Canvas │ │ │ │ │ + * A renderer based on the 2D 'canvas' drawing element. │ │ │ │ │ + * │ │ │ │ │ + * Inherits: │ │ │ │ │ + * - <OpenLayers.Renderer> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.Canvas = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: hitDetection │ │ │ │ │ + * {Boolean} Allow for hit detection of features. Default is true. │ │ │ │ │ + */ │ │ │ │ │ + hitDetection: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: hitOverflow │ │ │ │ │ + * {Number} The method for converting feature identifiers to color values │ │ │ │ │ + * supports 16777215 sequential values. Two features cannot be │ │ │ │ │ + * predictably detected if their identifiers differ by more than this │ │ │ │ │ + * value. The hitOverflow allows for bigger numbers (but the │ │ │ │ │ + * difference in values is still limited). │ │ │ │ │ + */ │ │ │ │ │ + hitOverflow: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: canvas │ │ │ │ │ + * {Canvas} The canvas context object. │ │ │ │ │ + */ │ │ │ │ │ + canvas: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Object} Internal object of feature/style pairs for use in redrawing the layer. │ │ │ │ │ + */ │ │ │ │ │ + features: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: pendingRedraw │ │ │ │ │ + * {Boolean} The renderer needs a redraw call to render features added while │ │ │ │ │ + * the renderer was locked. │ │ │ │ │ + */ │ │ │ │ │ + pendingRedraw: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: cachedSymbolBounds │ │ │ │ │ + * {Object} Internal cache of calculated symbol extents. │ │ │ │ │ + */ │ │ │ │ │ + cachedSymbolBounds: {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Renderer.Canvas │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * containerID - {<String>} │ │ │ │ │ + * options - {Object} Optional properties to be set on the renderer. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(containerID, options) { │ │ │ │ │ + OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.root = document.createElement("canvas"); │ │ │ │ │ + this.container.appendChild(this.root); │ │ │ │ │ + this.canvas = this.root.getContext("2d"); │ │ │ │ │ + this.features = {}; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitCanvas = document.createElement("canvas"); │ │ │ │ │ + this.hitContext = this.hitCanvas.getContext("2d"); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setExtent │ │ │ │ │ + * Set the visible part of the layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} │ │ │ │ │ + * resolutionChanged - {Boolean} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ + * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ + * False otherwise. │ │ │ │ │ + */ │ │ │ │ │ + setExtent: function() { │ │ │ │ │ + OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + // always redraw features │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: eraseGeometry │ │ │ │ │ + * Erase a geometry from the renderer. Because the Canvas renderer has │ │ │ │ │ + * 'memory' of the features that it has drawn, we have to remove the │ │ │ │ │ + * feature so it doesn't redraw. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + eraseGeometry: function(geometry, featureId) { │ │ │ │ │ + this.eraseFeatures(this.features[featureId][0]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: supported │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the browser supports the renderer class │ │ │ │ │ + */ │ │ │ │ │ + supported: function() { │ │ │ │ │ + return OpenLayers.CANVAS_SUPPORTED; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setSize │ │ │ │ │ + * Sets the size of the drawing surface. │ │ │ │ │ + * │ │ │ │ │ + * Once the size is updated, redraw the canvas. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * size - {<OpenLayers.Size>} │ │ │ │ │ + */ │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + this.size = size.clone(); │ │ │ │ │ + var root = this.root; │ │ │ │ │ + root.style.width = size.w + "px"; │ │ │ │ │ + root.style.height = size.h + "px"; │ │ │ │ │ + root.width = size.w; │ │ │ │ │ + root.height = size.h; │ │ │ │ │ + this.resolution = null; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + var hitCanvas = this.hitCanvas; │ │ │ │ │ + hitCanvas.style.width = size.w + "px"; │ │ │ │ │ + hitCanvas.style.height = size.h + "px"; │ │ │ │ │ + hitCanvas.width = size.w; │ │ │ │ │ + hitCanvas.height = size.h; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawFeature │ │ │ │ │ + * Draw the feature. Stores the feature in the features list, │ │ │ │ │ + * then redraws the layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * style - {<Object>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The feature has been drawn completely. If the feature has no │ │ │ │ │ + * geometry, undefined will be returned. If the feature is not rendered │ │ │ │ │ + * for other reasons, false will be returned. │ │ │ │ │ + */ │ │ │ │ │ + drawFeature: function(feature, style) { │ │ │ │ │ + var rendered; │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + style = this.applyDefaultSymbolizer(style || feature.style); │ │ │ │ │ + // don't render if display none or feature outside extent │ │ │ │ │ + var bounds = feature.geometry.getBounds(); │ │ │ │ │ + │ │ │ │ │ + var worldBounds; │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + worldBounds = this.map.getMaxExtent(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var intersects = bounds && bounds.intersectsBounds(this.extent, { │ │ │ │ │ + worldBounds: worldBounds │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + rendered = (style.display !== "none") && !!bounds && intersects; │ │ │ │ │ + if (rendered) { │ │ │ │ │ + // keep track of what we have rendered for redraw │ │ │ │ │ + this.features[feature.id] = [feature, style]; │ │ │ │ │ + } else { │ │ │ │ │ + // remove from features tracked for redraw │ │ │ │ │ + delete(this.features[feature.id]); │ │ │ │ │ + } │ │ │ │ │ + this.pendingRedraw = true; │ │ │ │ │ + } │ │ │ │ │ + if (this.pendingRedraw && !this.locked) { │ │ │ │ │ + this.redraw(); │ │ │ │ │ + this.pendingRedraw = false; │ │ │ │ │ + } │ │ │ │ │ + return rendered; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawGeometry │ │ │ │ │ + * Used when looping (in redraw) over the features; draws │ │ │ │ │ + * the canvas. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + */ │ │ │ │ │ + drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ + var className = geometry.CLASS_NAME; │ │ │ │ │ + if ((className == "OpenLayers.Geometry.Collection") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiPolygon")) { │ │ │ │ │ + for (var i = 0; i < geometry.components.length; i++) { │ │ │ │ │ + this.drawGeometry(geometry.components[i], style, featureId); │ │ │ │ │ + } │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + this.drawPoint(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + this.drawLineString(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + this.drawPolygon(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawExternalGraphic │ │ │ │ │ + * Called to draw External graphics. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + drawExternalGraphic: function(geometry, style, featureId) { │ │ │ │ │ + var img = new Image(); │ │ │ │ │ + │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + img.title = title; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ + var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ + style.graphicXOffset : -(0.5 * width); │ │ │ │ │ + var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ + style.graphicYOffset : -(0.5 * height); │ │ │ │ │ + │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ + │ │ │ │ │ + var onLoad = function() { │ │ │ │ │ + if (!this.features[featureId]) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ + if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ + var x = (p0 + xOffset) | 0; │ │ │ │ │ + var y = (p1 + yOffset) | 0; │ │ │ │ │ + var canvas = this.canvas; │ │ │ │ │ + canvas.globalAlpha = opacity; │ │ │ │ │ + var factor = OpenLayers.Renderer.Canvas.drawImageScaleFactor || │ │ │ │ │ + (OpenLayers.Renderer.Canvas.drawImageScaleFactor = │ │ │ │ │ + /android 2.1/.test(navigator.userAgent.toLowerCase()) ? │ │ │ │ │ + // 320 is the screen width of the G1 phone, for │ │ │ │ │ + // which drawImage works out of the box. │ │ │ │ │ + 320 / window.screen.width : 1 │ │ │ │ │ + ); │ │ │ │ │ + canvas.drawImage( │ │ │ │ │ + img, x * factor, y * factor, width * factor, height * factor │ │ │ │ │ + ); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId); │ │ │ │ │ + this.hitContext.fillRect(x, y, width, height); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + img.onload = OpenLayers.Function.bind(onLoad, this); │ │ │ │ │ + img.src = style.externalGraphic; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawNamedSymbol │ │ │ │ │ + * Called to draw Well Known Graphic Symbol Name. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + drawNamedSymbol: function(geometry, style, featureId) { │ │ │ │ │ + var x, y, cx, cy, i, symbolBounds, scaling, angle; │ │ │ │ │ + var unscaledStrokeWidth; │ │ │ │ │ + var deg2rad = Math.PI / 180.0; │ │ │ │ │ + │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[style.graphicName]; │ │ │ │ │ + │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(style.graphicName + ' is not a valid symbol name'); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (!symbol.length || symbol.length < 2) return; │ │ │ │ │ + │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ + │ │ │ │ │ + if (isNaN(p0) || isNaN(p1)) return; │ │ │ │ │ + │ │ │ │ │ + // Use rounded line caps │ │ │ │ │ + this.canvas.lineCap = "round"; │ │ │ │ │ + this.canvas.lineJoin = "round"; │ │ │ │ │ + │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.lineCap = "round"; │ │ │ │ │ + this.hitContext.lineJoin = "round"; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Scale and rotate symbols, using precalculated bounds whenever possible. │ │ │ │ │ + if (style.graphicName in this.cachedSymbolBounds) { │ │ │ │ │ + symbolBounds = this.cachedSymbolBounds[style.graphicName]; │ │ │ │ │ + } else { │ │ │ │ │ + symbolBounds = new OpenLayers.Bounds(); │ │ │ │ │ + for (i = 0; i < symbol.length; i += 2) { │ │ │ │ │ + symbolBounds.extend(new OpenLayers.LonLat(symbol[i], symbol[i + 1])); │ │ │ │ │ + } │ │ │ │ │ + this.cachedSymbolBounds[style.graphicName] = symbolBounds; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Push symbol scaling, translation and rotation onto the transformation stack in reverse order. │ │ │ │ │ + // Don't forget to apply all canvas transformations to the hitContext canvas as well(!) │ │ │ │ │ + this.canvas.save(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.save(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Step 3: place symbol at the desired location │ │ │ │ │ + this.canvas.translate(p0, p1); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.translate(p0, p1); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Step 2a. rotate the symbol if necessary │ │ │ │ │ + angle = deg2rad * style.rotation; // will be NaN when style.rotation is undefined. │ │ │ │ │ + if (!isNaN(angle)) { │ │ │ │ │ + this.canvas.rotate(angle); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.rotate(angle); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // // Step 2: scale symbol such that pointRadius equals half the maximum symbol dimension. │ │ │ │ │ + scaling = 2.0 * style.pointRadius / Math.max(symbolBounds.getWidth(), symbolBounds.getHeight()); │ │ │ │ │ + this.canvas.scale(scaling, scaling); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.scale(scaling, scaling); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Step 1: center the symbol at the origin │ │ │ │ │ + cx = symbolBounds.getCenterLonLat().lon; │ │ │ │ │ + cy = symbolBounds.getCenterLonLat().lat; │ │ │ │ │ + this.canvas.translate(-cx, -cy); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.translate(-cx, -cy); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Don't forget to scale stroke widths, because they are affected by canvas scale transformations as well(!) │ │ │ │ │ + // Alternative: scale symbol coordinates manually, so stroke width scaling is not needed anymore. │ │ │ │ │ + unscaledStrokeWidth = style.strokeWidth; │ │ │ │ │ + style.strokeWidth = unscaledStrokeWidth / scaling; │ │ │ │ │ + │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.canvas.lineTo(x, y); │ │ │ │ │ + } │ │ │ │ │ + this.canvas.closePath(); │ │ │ │ │ + this.canvas.fill(); │ │ │ │ │ + │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.hitContext.lineTo(x, y); │ │ │ │ │ + } │ │ │ │ │ + this.hitContext.closePath(); │ │ │ │ │ + this.hitContext.fill(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.canvas.lineTo(x, y); │ │ │ │ │ + } │ │ │ │ │ + this.canvas.closePath(); │ │ │ │ │ + this.canvas.stroke(); │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style, scaling); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.hitContext.moveTo(x, y); │ │ │ │ │ + this.hitContext.lineTo(x, y); │ │ │ │ │ + } │ │ │ │ │ + this.hitContext.closePath(); │ │ │ │ │ + this.hitContext.stroke(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + style.strokeWidth = unscaledStrokeWidth; │ │ │ │ │ + this.canvas.restore(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.restore(); │ │ │ │ │ + } │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setCanvasStyle │ │ │ │ │ + * Prepare the canvas for drawing by setting various global settings. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * type - {String} one of 'stroke', 'fill', or 'reset' │ │ │ │ │ + * style - {Object} Symbolizer hash │ │ │ │ │ + */ │ │ │ │ │ + setCanvasStyle: function(type, style) { │ │ │ │ │ + if (type === "fill") { │ │ │ │ │ + this.canvas.globalAlpha = style['fillOpacity']; │ │ │ │ │ + this.canvas.fillStyle = style['fillColor']; │ │ │ │ │ + } else if (type === "stroke") { │ │ │ │ │ + this.canvas.globalAlpha = style['strokeOpacity']; │ │ │ │ │ + this.canvas.strokeStyle = style['strokeColor']; │ │ │ │ │ + this.canvas.lineWidth = style['strokeWidth']; │ │ │ │ │ + } else { │ │ │ │ │ + this.canvas.globalAlpha = 0; │ │ │ │ │ + this.canvas.lineWidth = 1; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: featureIdToHex │ │ │ │ │ + * Convert a feature ID string into an RGB hex string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureId - {String} Feature id │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} RGB hex string. │ │ │ │ │ + */ │ │ │ │ │ + featureIdToHex: function(featureId) { │ │ │ │ │ + var id = Number(featureId.split("_").pop()) + 1; // zero for no feature │ │ │ │ │ + if (id >= 16777216) { │ │ │ │ │ + this.hitOverflow = id - 16777215; │ │ │ │ │ + id = id % 16777216 + 1; │ │ │ │ │ + } │ │ │ │ │ + var hex = "000000" + id.toString(16); │ │ │ │ │ + var len = hex.length; │ │ │ │ │ + hex = "#" + hex.substring(len - 6, len); │ │ │ │ │ + return hex; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setHitContextStyle │ │ │ │ │ + * Prepare the hit canvas for drawing by setting various global settings. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * type - {String} one of 'stroke', 'fill', or 'reset' │ │ │ │ │ + * featureId - {String} The feature id. │ │ │ │ │ + * symbolizer - {<OpenLayers.Symbolizer>} The symbolizer. │ │ │ │ │ + */ │ │ │ │ │ + setHitContextStyle: function(type, featureId, symbolizer, strokeScaling) { │ │ │ │ │ + var hex = this.featureIdToHex(featureId); │ │ │ │ │ + if (type == "fill") { │ │ │ │ │ + this.hitContext.globalAlpha = 1.0; │ │ │ │ │ + this.hitContext.fillStyle = hex; │ │ │ │ │ + } else if (type == "stroke") { │ │ │ │ │ + this.hitContext.globalAlpha = 1.0; │ │ │ │ │ + this.hitContext.strokeStyle = hex; │ │ │ │ │ + // bump up stroke width to deal with antialiasing. If strokeScaling is defined, we're rendering a symbol │ │ │ │ │ + // on a transformed canvas, so the antialias width bump has to scale as well. │ │ │ │ │ + if (typeof strokeScaling === "undefined") { │ │ │ │ │ + this.hitContext.lineWidth = symbolizer.strokeWidth + 2; │ │ │ │ │ + } else { │ │ │ │ │ + if (!isNaN(strokeScaling)) { │ │ │ │ │ + this.hitContext.lineWidth = symbolizer.strokeWidth + 2.0 / strokeScaling; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this.hitContext.globalAlpha = 0; │ │ │ │ │ + this.hitContext.lineWidth = 1; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawPoint │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + drawPoint: function(geometry, style, featureId) { │ │ │ │ │ + if (style.graphic !== false) { │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + this.drawExternalGraphic(geometry, style, featureId); │ │ │ │ │ + } else if (style.graphicName && (style.graphicName != "circle")) { │ │ │ │ │ + this.drawNamedSymbol(geometry, style, featureId); │ │ │ │ │ + } else { │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ + if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ + var twoPi = Math.PI * 2; │ │ │ │ │ + var radius = style.pointRadius; │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.canvas.fill(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.hitContext.fill(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.canvas.stroke(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.hitContext.stroke(); │ │ │ │ │ + } │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawLineString │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + drawLineString: function(geometry, style, featureId) { │ │ │ │ │ + style = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + fill: false │ │ │ │ │ + }, style); │ │ │ │ │ + this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawLinearRing │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + drawLinearRing: function(geometry, style, featureId) { │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.renderPath(this.canvas, geometry, style, featureId, "fill"); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.renderPath(this.hitContext, geometry, style, featureId, "fill"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.renderPath(this.canvas, geometry, style, featureId, "stroke"); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ + this.renderPath(this.hitContext, geometry, style, featureId, "stroke"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: renderPath │ │ │ │ │ + * Render a path with stroke and optional fill. │ │ │ │ │ + */ │ │ │ │ │ + renderPath: function(context, geometry, style, featureId, type) { │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + context.beginPath(); │ │ │ │ │ + var start = this.getLocalXY(components[0]); │ │ │ │ │ + var x = start[0]; │ │ │ │ │ + var y = start[1]; │ │ │ │ │ + if (!isNaN(x) && !isNaN(y)) { │ │ │ │ │ + context.moveTo(start[0], start[1]); │ │ │ │ │ + for (var i = 1; i < len; ++i) { │ │ │ │ │ + var pt = this.getLocalXY(components[i]); │ │ │ │ │ + context.lineTo(pt[0], pt[1]); │ │ │ │ │ + } │ │ │ │ │ + if (type === "fill") { │ │ │ │ │ + context.fill(); │ │ │ │ │ + } else { │ │ │ │ │ + context.stroke(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawPolygon │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + drawPolygon: function(geometry, style, featureId) { │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + this.drawLinearRing(components[0], style, featureId); │ │ │ │ │ + // erase inner rings │ │ │ │ │ + for (var i = 1; i < len; ++i) { │ │ │ │ │ + /** │ │ │ │ │ + * Note that this is overly agressive. Here we punch holes through │ │ │ │ │ + * all previously rendered features on the same canvas. A better │ │ │ │ │ + * solution for polygons with interior rings would be to draw the │ │ │ │ │ + * polygon on a sketch canvas first. We could erase all holes │ │ │ │ │ + * there and then copy the drawing to the layer canvas. │ │ │ │ │ + * TODO: http://trac.osgeo.org/openlayers/ticket/3130 │ │ │ │ │ + */ │ │ │ │ │ + this.canvas.globalCompositeOperation = "destination-out"; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.globalCompositeOperation = "destination-out"; │ │ │ │ │ + } │ │ │ │ │ + this.drawLinearRing( │ │ │ │ │ + components[i], │ │ │ │ │ + OpenLayers.Util.applyDefaults({ │ │ │ │ │ + stroke: false, │ │ │ │ │ + fillOpacity: 1.0 │ │ │ │ │ + }, style), │ │ │ │ │ + featureId │ │ │ │ │ + ); │ │ │ │ │ + this.canvas.globalCompositeOperation = "source-over"; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.globalCompositeOperation = "source-over"; │ │ │ │ │ + } │ │ │ │ │ + this.drawLinearRing( │ │ │ │ │ + components[i], │ │ │ │ │ + OpenLayers.Util.applyDefaults({ │ │ │ │ │ + fill: false │ │ │ │ │ + }, style), │ │ │ │ │ + featureId │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawText │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * location - {<OpenLayers.Point>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + */ │ │ │ │ │ + drawText: function(location, style) { │ │ │ │ │ + var pt = this.getLocalXY(location); │ │ │ │ │ + │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ + this.canvas.fillStyle = style.fontColor; │ │ │ │ │ + this.canvas.globalAlpha = style.fontOpacity || 1.0; │ │ │ │ │ + var fontStyle = [style.fontStyle ? style.fontStyle : "normal", │ │ │ │ │ + "normal", // "font-variant" not supported │ │ │ │ │ + style.fontWeight ? style.fontWeight : "normal", │ │ │ │ │ + style.fontSize ? style.fontSize : "1em", │ │ │ │ │ + style.fontFamily ? style.fontFamily : "sans-serif" │ │ │ │ │ + ].join(" "); │ │ │ │ │ + var labelRows = style.label.split('\n'); │ │ │ │ │ + var numRows = labelRows.length; │ │ │ │ │ + if (this.canvas.fillText) { │ │ │ │ │ + // HTML5 │ │ │ │ │ + this.canvas.font = fontStyle; │ │ │ │ │ + this.canvas.textAlign = │ │ │ │ │ + OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[0]] || │ │ │ │ │ + "center"; │ │ │ │ │ + this.canvas.textBaseline = │ │ │ │ │ + OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[1]] || │ │ │ │ │ + "middle"; │ │ │ │ │ + var vfactor = │ │ │ │ │ + OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5; │ │ │ │ │ + } │ │ │ │ │ + var lineHeight = │ │ │ │ │ + this.canvas.measureText('Mg').height || │ │ │ │ │ + this.canvas.measureText('xx').width; │ │ │ │ │ + pt[1] += lineHeight * vfactor * (numRows - 1); │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + if (style.labelOutlineWidth) { │ │ │ │ │ + this.canvas.save(); │ │ │ │ │ + this.canvas.globalAlpha = style.labelOutlineOpacity || style.fontOpacity || 1.0; │ │ │ │ │ + this.canvas.strokeStyle = style.labelOutlineColor; │ │ │ │ │ + this.canvas.lineWidth = style.labelOutlineWidth; │ │ │ │ │ + this.canvas.strokeText(labelRows[i], pt[0], pt[1] + (lineHeight * i) + 1); │ │ │ │ │ + this.canvas.restore(); │ │ │ │ │ + } │ │ │ │ │ + this.canvas.fillText(labelRows[i], pt[0], pt[1] + (lineHeight * i)); │ │ │ │ │ + } │ │ │ │ │ + } else if (this.canvas.mozDrawText) { │ │ │ │ │ + // Mozilla pre-Gecko1.9.1 (<FF3.1) │ │ │ │ │ + this.canvas.mozTextStyle = fontStyle; │ │ │ │ │ + // No built-in text alignment, so we measure and adjust the position │ │ │ │ │ + var hfactor = │ │ │ │ │ + OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[0]]; │ │ │ │ │ + if (hfactor == null) { │ │ │ │ │ + hfactor = -.5; │ │ │ │ │ + } │ │ │ │ │ + var vfactor = │ │ │ │ │ + OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5; │ │ │ │ │ + } │ │ │ │ │ + var lineHeight = this.canvas.mozMeasureText('xx'); │ │ │ │ │ + pt[1] += lineHeight * (1 + (vfactor * numRows)); │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + var x = pt[0] + (hfactor * this.canvas.mozMeasureText(labelRows[i])); │ │ │ │ │ + var y = pt[1] + (i * lineHeight); │ │ │ │ │ + this.canvas.translate(x, y); │ │ │ │ │ + this.canvas.mozDrawText(labelRows[i]); │ │ │ │ │ + this.canvas.translate(-x, -y); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getLocalXY │ │ │ │ │ + * transform geographic xy into pixel xy │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + */ │ │ │ │ │ + getLocalXY: function(point) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var extent = this.extent; │ │ │ │ │ + var x = ((point.x - this.featureDx) / resolution + (-extent.left / resolution)); │ │ │ │ │ + var y = ((extent.top / resolution) - point.y / resolution); │ │ │ │ │ + return [x, y]; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clear │ │ │ │ │ + * Clear all vectors from the renderer. │ │ │ │ │ + */ │ │ │ │ │ + clear: function() { │ │ │ │ │ + var height = this.root.height; │ │ │ │ │ + var width = this.root.width; │ │ │ │ │ + this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ + this.features = {}; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.clearRect(0, 0, width, height); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getFeatureIdFromEvent │ │ │ │ │ + * Returns a feature id from an event on the renderer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector} A feature or undefined. This method returns a │ │ │ │ │ + * feature instead of a feature id to avoid an unnecessary lookup on the │ │ │ │ │ + * layer. │ │ │ │ │ + */ │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var featureId, feature; │ │ │ │ │ + │ │ │ │ │ + if (this.hitDetection && this.root.style.display !== "none") { │ │ │ │ │ + // this dragging check should go in the feature handler │ │ │ │ │ + if (!this.map.dragging) { │ │ │ │ │ + var xy = evt.xy; │ │ │ │ │ + var x = xy.x | 0; │ │ │ │ │ + var y = xy.y | 0; │ │ │ │ │ + var data = this.hitContext.getImageData(x, y, 1, 1).data; │ │ │ │ │ + if (data[3] === 255) { // antialiased │ │ │ │ │ + var id = data[2] + (256 * (data[1] + (256 * data[0]))); │ │ │ │ │ + if (id) { │ │ │ │ │ + featureId = "OpenLayers_Feature_Vector_" + (id - 1 + this.hitOverflow); │ │ │ │ │ + try { │ │ │ │ │ + feature = this.features[featureId][0]; │ │ │ │ │ + } catch (err) { │ │ │ │ │ + // Because of antialiasing on the canvas, when the hit location is at a point where the edge of │ │ │ │ │ + // one symbol intersects the interior of another symbol, a wrong hit color (and therefore id) results. │ │ │ │ │ + // todo: set Antialiasing = 'off' on the hitContext as soon as browsers allow it. │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return feature; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: eraseFeatures │ │ │ │ │ + * This is called by the layer to erase features; removes the feature from │ │ │ │ │ + * the list, then redraws the layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + */ │ │ │ │ │ + eraseFeatures: function(features) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0; i < features.length; ++i) { │ │ │ │ │ + delete this.features[features[i].id]; │ │ │ │ │ + } │ │ │ │ │ + this.redraw(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: redraw │ │ │ │ │ + * The real 'meat' of the function: any time things have changed, │ │ │ │ │ + * redraw() can be called to loop over all the data and (you guessed │ │ │ │ │ + * it) redraw it. Unlike Elements-based Renderers, we can't interact │ │ │ │ │ + * with things once they're drawn, to remove them, for example, so │ │ │ │ │ + * instead we have to just clear everything and draw from scratch. │ │ │ │ │ + */ │ │ │ │ │ + redraw: function() { │ │ │ │ │ + if (!this.locked) { │ │ │ │ │ + var height = this.root.height; │ │ │ │ │ + var width = this.root.width; │ │ │ │ │ + this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.clearRect(0, 0, width, height); │ │ │ │ │ + } │ │ │ │ │ + var labelMap = []; │ │ │ │ │ + var feature, geometry, style; │ │ │ │ │ + var worldBounds = (this.map.baseLayer && this.map.baseLayer.wrapDateLine) && this.map.getMaxExtent(); │ │ │ │ │ + for (var id in this.features) { │ │ │ │ │ + if (!this.features.hasOwnProperty(id)) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + feature = this.features[id][0]; │ │ │ │ │ + geometry = feature.geometry; │ │ │ │ │ + this.calculateFeatureDx(geometry.getBounds(), worldBounds); │ │ │ │ │ + style = this.features[id][1]; │ │ │ │ │ + this.drawGeometry(geometry, style, feature.id); │ │ │ │ │ + if (style.label) { │ │ │ │ │ + labelMap.push([feature, style]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var item; │ │ │ │ │ + for (var i = 0, len = labelMap.length; i < len; ++i) { │ │ │ │ │ + item = labelMap[i]; │ │ │ │ │ + this.drawText(item[0].geometry.getCentroid(), item[1]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.Canvas" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.Canvas.LABEL_ALIGN │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.Canvas.LABEL_ALIGN = { │ │ │ │ │ + "l": "left", │ │ │ │ │ + "r": "right", │ │ │ │ │ + "t": "top", │ │ │ │ │ + "b": "bottom" │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.Canvas.LABEL_FACTOR │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.Canvas.LABEL_FACTOR = { │ │ │ │ │ + "l": 0, │ │ │ │ │ + "r": -1, │ │ │ │ │ + "t": 0, │ │ │ │ │ + "b": -1 │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.Canvas.drawImageScaleFactor │ │ │ │ │ + * {Number} Scale factor to apply to the canvas drawImage arguments. This │ │ │ │ │ + * is always 1 except for Android 2.1 devices, to work around │ │ │ │ │ + * http://code.google.com/p/android/issues/detail?id=5141. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.Canvas.drawImageScaleFactor = null; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Renderer/Elements.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Renderer.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.ElementsIndexer │ │ │ │ │ + * This class takes care of figuring out which order elements should be │ │ │ │ │ + * placed in the DOM based on given indexing methods. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.ElementsIndexer = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: maxZIndex │ │ │ │ │ + * {Integer} This is the largest-most z-index value for a node │ │ │ │ │ + * contained within the indexer. │ │ │ │ │ + */ │ │ │ │ │ + maxZIndex: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: order │ │ │ │ │ + * {Array<String>} This is an array of node id's stored in the │ │ │ │ │ + * order that they should show up on screen. Id's higher up in the │ │ │ │ │ + * array (higher array index) represent nodes with higher z-indeces. │ │ │ │ │ + */ │ │ │ │ │ + order: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: indices │ │ │ │ │ + * {Object} This is a hash that maps node ids to their z-index value │ │ │ │ │ + * stored in the indexer. This is done to make finding a nodes z-index │ │ │ │ │ + * value O(1). │ │ │ │ │ + */ │ │ │ │ │ + indices: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: compare │ │ │ │ │ + * {Function} This is the function used to determine placement of │ │ │ │ │ + * of a new node within the indexer. If null, this defaults to to │ │ │ │ │ + * the Z_ORDER_DRAWING_ORDER comparison method. │ │ │ │ │ + */ │ │ │ │ │ + compare: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: initialize │ │ │ │ │ + * Create a new indexer with │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * yOrdering - {Boolean} Whether to use y-ordering. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(yOrdering) { │ │ │ │ │ + │ │ │ │ │ + this.compare = yOrdering ? │ │ │ │ │ + OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_Y_ORDER : │ │ │ │ │ + OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_DRAWING_ORDER; │ │ │ │ │ + │ │ │ │ │ + this.clear(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: insert │ │ │ │ │ + * Insert a new node into the indexer. In order to find the correct │ │ │ │ │ + * positioning for the node to be inserted, this method uses a binary │ │ │ │ │ + * search. This makes inserting O(log(n)). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newNode - {DOMElement} The new node to be inserted. │ │ │ │ │ + * │ │ │ │ │ + * Returns │ │ │ │ │ + * {DOMElement} the node before which we should insert our newNode, or │ │ │ │ │ + * null if newNode can just be appended. │ │ │ │ │ + */ │ │ │ │ │ + insert: function(newNode) { │ │ │ │ │ + // If the node is known to the indexer, remove it so we can │ │ │ │ │ + // recalculate where it should go. │ │ │ │ │ + if (this.exists(newNode)) { │ │ │ │ │ + this.remove(newNode); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var nodeId = newNode.id; │ │ │ │ │ + │ │ │ │ │ + this.determineZIndex(newNode); │ │ │ │ │ + │ │ │ │ │ + var leftIndex = -1; │ │ │ │ │ + var rightIndex = this.order.length; │ │ │ │ │ + var middle; │ │ │ │ │ + │ │ │ │ │ + while (rightIndex - leftIndex > 1) { │ │ │ │ │ + middle = parseInt((leftIndex + rightIndex) / 2); │ │ │ │ │ + │ │ │ │ │ + var placement = this.compare(this, newNode, │ │ │ │ │ + OpenLayers.Util.getElement(this.order[middle])); │ │ │ │ │ + │ │ │ │ │ + if (placement > 0) { │ │ │ │ │ + leftIndex = middle; │ │ │ │ │ + } else { │ │ │ │ │ + rightIndex = middle; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.order.splice(rightIndex, 0, nodeId); │ │ │ │ │ + this.indices[nodeId] = this.getZIndex(newNode); │ │ │ │ │ + │ │ │ │ │ + // If the new node should be before another in the index │ │ │ │ │ + // order, return the node before which we have to insert the new one; │ │ │ │ │ + // else, return null to indicate that the new node can be appended. │ │ │ │ │ + return this.getNextElement(rightIndex); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: remove │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The node to be removed. │ │ │ │ │ + */ │ │ │ │ │ + remove: function(node) { │ │ │ │ │ + var nodeId = node.id; │ │ │ │ │ + var arrayIndex = OpenLayers.Util.indexOf(this.order, nodeId); │ │ │ │ │ + if (arrayIndex >= 0) { │ │ │ │ │ + // Remove it from the order array, as well as deleting the node │ │ │ │ │ + // from the indeces hash. │ │ │ │ │ + this.order.splice(arrayIndex, 1); │ │ │ │ │ + delete this.indices[nodeId]; │ │ │ │ │ + │ │ │ │ │ + // Reset the maxium z-index based on the last item in the │ │ │ │ │ + // order array. │ │ │ │ │ + if (this.order.length > 0) { │ │ │ │ │ + var lastId = this.order[this.order.length - 1]; │ │ │ │ │ + this.maxZIndex = this.indices[lastId]; │ │ │ │ │ + } else { │ │ │ │ │ + this.maxZIndex = 0; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clear │ │ │ │ │ + */ │ │ │ │ │ + clear: function() { │ │ │ │ │ + this.order = []; │ │ │ │ │ + this.indices = {}; │ │ │ │ │ + this.maxZIndex = 0; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: exists │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The node to test for existence. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the node exists in the indexer? │ │ │ │ │ + */ │ │ │ │ │ + exists: function(node) { │ │ │ │ │ + return (this.indices[node.id] != null); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getZIndex │ │ │ │ │ + * Get the z-index value for the current node from the node data itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The node whose z-index to get. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} The z-index value for the specified node (from the node │ │ │ │ │ + * data itself). │ │ │ │ │ + */ │ │ │ │ │ + getZIndex: function(node) { │ │ │ │ │ + return node._style.graphicZIndex; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: determineZIndex │ │ │ │ │ + * Determine the z-index for the current node if there isn't one, │ │ │ │ │ + * and set the maximum value if we've found a new maximum. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + determineZIndex: function(node) { │ │ │ │ │ + var zIndex = node._style.graphicZIndex; │ │ │ │ │ + │ │ │ │ │ + // Everything must have a zIndex. If none is specified, │ │ │ │ │ + // this means the user *must* (hint: assumption) want this │ │ │ │ │ + // node to succomb to drawing order. To enforce drawing order │ │ │ │ │ + // over all indexing methods, we'll create a new z-index that's │ │ │ │ │ + // greater than any currently in the indexer. │ │ │ │ │ + if (zIndex == null) { │ │ │ │ │ + zIndex = this.maxZIndex; │ │ │ │ │ + node._style.graphicZIndex = zIndex; │ │ │ │ │ + } else if (zIndex > this.maxZIndex) { │ │ │ │ │ + this.maxZIndex = zIndex; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getNextElement │ │ │ │ │ + * Get the next element in the order stack. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * index - {Integer} The index of the current node in this.order. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} the node following the index passed in, or │ │ │ │ │ + * null. │ │ │ │ │ + */ │ │ │ │ │ + getNextElement: function(index) { │ │ │ │ │ + var nextIndex = index + 1; │ │ │ │ │ + if (nextIndex < this.order.length) { │ │ │ │ │ + var nextElement = OpenLayers.Util.getElement(this.order[nextIndex]); │ │ │ │ │ + if (nextElement == undefined) { │ │ │ │ │ + nextElement = this.getNextElement(nextIndex); │ │ │ │ │ + } │ │ │ │ │ + return nextElement; │ │ │ │ │ + } else { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.ElementsIndexer" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.ElementsIndexer.IndexingMethods │ │ │ │ │ + * These are the compare methods for figuring out where a new node should be │ │ │ │ │ + * placed within the indexer. These methods are very similar to general │ │ │ │ │ + * sorting methods in that they return -1, 0, and 1 to specify the │ │ │ │ │ + * direction in which new nodes fall in the ordering. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.ElementsIndexer.IndexingMethods = { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: Z_ORDER │ │ │ │ │ + * This compare method is used by other comparison methods. │ │ │ │ │ + * It can be used individually for ordering, but is not recommended, │ │ │ │ │ + * because it doesn't subscribe to drawing order. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ + * newNode - {DOMElement} │ │ │ │ │ + * nextNode - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} │ │ │ │ │ + */ │ │ │ │ │ + Z_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var newZIndex = indexer.getZIndex(newNode); │ │ │ │ │ + │ │ │ │ │ + var returnVal = 0; │ │ │ │ │ + if (nextNode) { │ │ │ │ │ + var nextZIndex = indexer.getZIndex(nextNode); │ │ │ │ │ + returnVal = newZIndex - nextZIndex; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return returnVal; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: Z_ORDER_DRAWING_ORDER │ │ │ │ │ + * This method orders nodes by their z-index, but does so in a way │ │ │ │ │ + * that, if there are other nodes with the same z-index, the newest │ │ │ │ │ + * drawn will be the front most within that z-index. This is the │ │ │ │ │ + * default indexing method. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ + * newNode - {DOMElement} │ │ │ │ │ + * nextNode - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} │ │ │ │ │ + */ │ │ │ │ │ + Z_ORDER_DRAWING_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER( │ │ │ │ │ + indexer, │ │ │ │ │ + newNode, │ │ │ │ │ + nextNode │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + // Make Z_ORDER subscribe to drawing order by pushing it above │ │ │ │ │ + // all of the other nodes with the same z-index. │ │ │ │ │ + if (nextNode && returnVal == 0) { │ │ │ │ │ + returnVal = 1; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return returnVal; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: Z_ORDER_Y_ORDER │ │ │ │ │ + * This one should really be called Z_ORDER_Y_ORDER_DRAWING_ORDER, as it │ │ │ │ │ + * best describes which ordering methods have precedence (though, the │ │ │ │ │ + * name would be too long). This method orders nodes by their z-index, │ │ │ │ │ + * but does so in a way that, if there are other nodes with the same │ │ │ │ │ + * z-index, the nodes with the lower y position will be "closer" than │ │ │ │ │ + * those with a higher y position. If two nodes have the exact same y │ │ │ │ │ + * position, however, then this method will revert to using drawing │ │ │ │ │ + * order to decide placement. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ + * newNode - {DOMElement} │ │ │ │ │ + * nextNode - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} │ │ │ │ │ + */ │ │ │ │ │ + Z_ORDER_Y_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER( │ │ │ │ │ + indexer, │ │ │ │ │ + newNode, │ │ │ │ │ + nextNode │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + if (nextNode && returnVal === 0) { │ │ │ │ │ + var result = nextNode._boundsBottom - newNode._boundsBottom; │ │ │ │ │ + returnVal = (result === 0) ? 1 : result; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return returnVal; │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Renderer.Elements │ │ │ │ │ + * This is another virtual class in that it should never be instantiated by │ │ │ │ │ + * itself as a Renderer. It exists because there is *tons* of shared │ │ │ │ │ + * functionality between different vector libraries which use nodes/elements │ │ │ │ │ + * as a base for rendering vectors. │ │ │ │ │ + * │ │ │ │ │ + * The highlevel bits of code that are implemented here are the adding and │ │ │ │ │ + * removing of geometries, which is essentially the same for any │ │ │ │ │ + * element-based renderer. The details of creating each node and drawing the │ │ │ │ │ + * paths are of course different, but the machinery is the same. │ │ │ │ │ + * │ │ │ │ │ + * Inherits: │ │ │ │ │ + * - <OpenLayers.Renderer> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.Elements = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: rendererRoot │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + rendererRoot: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: root │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + root: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: vectorRoot │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + vectorRoot: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: textRoot │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + textRoot: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: xmlns │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + xmlns: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: xOffset │ │ │ │ │ + * {Number} Offset to apply to the renderer viewport translation in x │ │ │ │ │ + * direction. If the renderer extent's center is on the right of the │ │ │ │ │ + * dateline (i.e. exceeds the world bounds), we shift the viewport to the │ │ │ │ │ + * left by one world width. This avoids that features disappear from the │ │ │ │ │ + * map viewport. Because our dateline handling logic in other places │ │ │ │ │ + * ensures that extents crossing the dateline always have a center │ │ │ │ │ + * exceeding the world bounds on the left, we need this offset to make sure │ │ │ │ │ + * that the same is true for the renderer extent in pixel space as well. │ │ │ │ │ + */ │ │ │ │ │ + xOffset: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: rightOfDateLine │ │ │ │ │ + * {Boolean} Keeps track of the location of the map extent relative to the │ │ │ │ │ + * date line. The <setExtent> method compares this value (which is the one │ │ │ │ │ + * from the previous <setExtent> call) with the current position of the map │ │ │ │ │ + * extent relative to the date line and updates the xOffset when the extent │ │ │ │ │ + * has moved from one side of the date line to the other. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: Indexer │ │ │ │ │ + * {<OpenLayers.ElementIndexer>} An instance of OpenLayers.ElementsIndexer │ │ │ │ │ + * created upon initialization if the zIndexing or yOrdering options │ │ │ │ │ + * passed to this renderer's constructor are set to true. │ │ │ │ │ + */ │ │ │ │ │ + indexer: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: BACKGROUND_ID_SUFFIX │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + BACKGROUND_ID_SUFFIX: "_background", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: LABEL_ID_SUFFIX │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + LABEL_ID_SUFFIX: "_label", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: LABEL_OUTLINE_SUFFIX │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + LABEL_OUTLINE_SUFFIX: "_outline", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Renderer.Elements │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * containerID - {String} │ │ │ │ │ + * options - {Object} options for this renderer. │ │ │ │ │ + * │ │ │ │ │ + * Supported options are: │ │ │ │ │ + * yOrdering - {Boolean} Whether to use y-ordering │ │ │ │ │ + * zIndexing - {Boolean} Whether to use z-indexing. Will be ignored │ │ │ │ │ + * if yOrdering is set to true. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(containerID, options) { │ │ │ │ │ + OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + this.rendererRoot = this.createRenderRoot(); │ │ │ │ │ + this.root = this.createRoot("_root"); │ │ │ │ │ + this.vectorRoot = this.createRoot("_vroot"); │ │ │ │ │ + this.textRoot = this.createRoot("_troot"); │ │ │ │ │ + │ │ │ │ │ + this.root.appendChild(this.vectorRoot); │ │ │ │ │ + this.root.appendChild(this.textRoot); │ │ │ │ │ + │ │ │ │ │ + this.rendererRoot.appendChild(this.root); │ │ │ │ │ + this.container.appendChild(this.rendererRoot); │ │ │ │ │ + │ │ │ │ │ + if (options && (options.zIndexing || options.yOrdering)) { │ │ │ │ │ + this.indexer = new OpenLayers.ElementsIndexer(options.yOrdering); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + │ │ │ │ │ + this.clear(); │ │ │ │ │ + │ │ │ │ │ + this.rendererRoot = null; │ │ │ │ │ + this.root = null; │ │ │ │ │ + this.xmlns = null; │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Renderer.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clear │ │ │ │ │ + * Remove all the elements from the root │ │ │ │ │ + */ │ │ │ │ │ + clear: function() { │ │ │ │ │ + var child; │ │ │ │ │ + var root = this.vectorRoot; │ │ │ │ │ + if (root) { │ │ │ │ │ + while (child = root.firstChild) { │ │ │ │ │ + root.removeChild(child); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + root = this.textRoot; │ │ │ │ │ + if (root) { │ │ │ │ │ + while (child = root.firstChild) { │ │ │ │ │ + root.removeChild(child); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + this.indexer.clear(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setExtent │ │ │ │ │ + * Set the visible part of the layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} │ │ │ │ │ + * resolutionChanged - {Boolean} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ + * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ + * False otherwise. │ │ │ │ │ + */ │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + var rightOfDateLine, │ │ │ │ │ + ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ + extent = extent.scale(1 / ratio), │ │ │ │ │ + world = this.map.getMaxExtent(); │ │ │ │ │ + if (world.right > extent.left && world.right < extent.right) { │ │ │ │ │ + rightOfDateLine = true; │ │ │ │ │ + } else if (world.left > extent.left && world.left < extent.right) { │ │ │ │ │ + rightOfDateLine = false; │ │ │ │ │ + } │ │ │ │ │ + if (rightOfDateLine !== this.rightOfDateLine || resolutionChanged) { │ │ │ │ │ + coordSysUnchanged = false; │ │ │ │ │ + this.xOffset = rightOfDateLine === true ? │ │ │ │ │ + world.getWidth() / resolution : 0; │ │ │ │ │ + } │ │ │ │ │ + this.rightOfDateLine = rightOfDateLine; │ │ │ │ │ + } │ │ │ │ │ + return coordSysUnchanged; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getNodeType │ │ │ │ │ + * This function is in charge of asking the specific renderer which type │ │ │ │ │ + * of node to create for the given geometry and style. All geometries │ │ │ │ │ + * in an Elements-based renderer consist of one node and some │ │ │ │ │ + * attributes. We have the nodeFactory() function which creates a node │ │ │ │ │ + * for us, but it takes a 'type' as input, and that is precisely what │ │ │ │ │ + * this function tells us. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The corresponding node type for the specified geometry │ │ │ │ │ + */ │ │ │ │ │ + getNodeType: function(geometry, style) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawGeometry │ │ │ │ │ + * Draw the geometry, creating new nodes, setting paths, setting style, │ │ │ │ │ + * setting featureId on the node. This method should only be called │ │ │ │ │ + * by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the geometry has been drawn completely; null if │ │ │ │ │ + * incomplete; false otherwise │ │ │ │ │ + */ │ │ │ │ │ + drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ + var className = geometry.CLASS_NAME; │ │ │ │ │ + var rendered = true; │ │ │ │ │ + if ((className == "OpenLayers.Geometry.Collection") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiPolygon")) { │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + rendered = this.drawGeometry( │ │ │ │ │ + geometry.components[i], style, featureId) && rendered; │ │ │ │ │ + } │ │ │ │ │ + return rendered; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + rendered = false; │ │ │ │ │ + var removeBackground = false; │ │ │ │ │ + if (style.display != "none") { │ │ │ │ │ + if (style.backgroundGraphic) { │ │ │ │ │ + this.redrawBackgroundNode(geometry.id, geometry, style, │ │ │ │ │ + featureId); │ │ │ │ │ + } else { │ │ │ │ │ + removeBackground = true; │ │ │ │ │ + } │ │ │ │ │ + rendered = this.redrawNode(geometry.id, geometry, style, │ │ │ │ │ + featureId); │ │ │ │ │ + } │ │ │ │ │ + if (rendered == false) { │ │ │ │ │ + var node = document.getElementById(geometry.id); │ │ │ │ │ + if (node) { │ │ │ │ │ + if (node._style.backgroundGraphic) { │ │ │ │ │ + removeBackground = true; │ │ │ │ │ + } │ │ │ │ │ + node.parentNode.removeChild(node); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (removeBackground) { │ │ │ │ │ + var node = document.getElementById( │ │ │ │ │ + geometry.id + this.BACKGROUND_ID_SUFFIX); │ │ │ │ │ + if (node) { │ │ │ │ │ + node.parentNode.removeChild(node); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return rendered; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: redrawNode │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the complete geometry could be drawn, null if parts of │ │ │ │ │ + * the geometry could not be drawn, false otherwise │ │ │ │ │ + */ │ │ │ │ │ + redrawNode: function(id, geometry, style, featureId) { │ │ │ │ │ + style = this.applyDefaultSymbolizer(style); │ │ │ │ │ + // Get the node if it's already on the map. │ │ │ │ │ + var node = this.nodeFactory(id, this.getNodeType(geometry, style)); │ │ │ │ │ + │ │ │ │ │ + // Set the data for the node, then draw it. │ │ │ │ │ + node._featureId = featureId; │ │ │ │ │ + node._boundsBottom = geometry.getBounds().bottom; │ │ │ │ │ + node._geometryClass = geometry.CLASS_NAME; │ │ │ │ │ + node._style = style; │ │ │ │ │ + │ │ │ │ │ + var drawResult = this.drawGeometryNode(node, geometry, style); │ │ │ │ │ + if (drawResult === false) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + node = drawResult.node; │ │ │ │ │ + │ │ │ │ │ + // Insert the node into the indexer so it can show us where to │ │ │ │ │ + // place it. Note that this operation is O(log(n)). If there's a │ │ │ │ │ + // performance problem (when dragging, for instance) this is │ │ │ │ │ + // likely where it would be. │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + var insert = this.indexer.insert(node); │ │ │ │ │ + if (insert) { │ │ │ │ │ + this.vectorRoot.insertBefore(node, insert); │ │ │ │ │ + } else { │ │ │ │ │ + this.vectorRoot.appendChild(node); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // if there's no indexer, simply append the node to root, │ │ │ │ │ + // but only if the node is a new one │ │ │ │ │ + if (node.parentNode !== this.vectorRoot) { │ │ │ │ │ + this.vectorRoot.appendChild(node); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.postDraw(node); │ │ │ │ │ + │ │ │ │ │ + return drawResult.complete; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: redrawBackgroundNode │ │ │ │ │ + * Redraws the node using special 'background' style properties. Basically │ │ │ │ │ + * just calls redrawNode(), but instead of directly using the │ │ │ │ │ + * 'externalGraphic', 'graphicXOffset', 'graphicYOffset', and │ │ │ │ │ + * 'graphicZIndex' properties directly from the specified 'style' │ │ │ │ │ + * parameter, we create a new style object and set those properties │ │ │ │ │ + * from the corresponding 'background'-prefixed properties from │ │ │ │ │ + * specified 'style' parameter. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the complete geometry could be drawn, null if parts of │ │ │ │ │ + * the geometry could not be drawn, false otherwise │ │ │ │ │ + */ │ │ │ │ │ + redrawBackgroundNode: function(id, geometry, style, featureId) { │ │ │ │ │ + var backgroundStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ + │ │ │ │ │ + // Set regular style attributes to apply to the background styles. │ │ │ │ │ + backgroundStyle.externalGraphic = backgroundStyle.backgroundGraphic; │ │ │ │ │ + backgroundStyle.graphicXOffset = backgroundStyle.backgroundXOffset; │ │ │ │ │ + backgroundStyle.graphicYOffset = backgroundStyle.backgroundYOffset; │ │ │ │ │ + backgroundStyle.graphicZIndex = backgroundStyle.backgroundGraphicZIndex; │ │ │ │ │ + backgroundStyle.graphicWidth = backgroundStyle.backgroundWidth || backgroundStyle.graphicWidth; │ │ │ │ │ + backgroundStyle.graphicHeight = backgroundStyle.backgroundHeight || backgroundStyle.graphicHeight; │ │ │ │ │ + │ │ │ │ │ + // Erase background styles. │ │ │ │ │ + backgroundStyle.backgroundGraphic = null; │ │ │ │ │ + backgroundStyle.backgroundXOffset = null; │ │ │ │ │ + backgroundStyle.backgroundYOffset = null; │ │ │ │ │ + backgroundStyle.backgroundGraphicZIndex = null; │ │ │ │ │ + │ │ │ │ │ + return this.redrawNode( │ │ │ │ │ + id + this.BACKGROUND_ID_SUFFIX, │ │ │ │ │ + geometry, │ │ │ │ │ + backgroundStyle, │ │ │ │ │ + null │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawGeometryNode │ │ │ │ │ + * Given a node, draw a geometry on the specified layer. │ │ │ │ │ + * node and geometry are required arguments, style is optional. │ │ │ │ │ + * This method is only called by the render itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} a hash with properties "node" (the drawn node) and "complete" │ │ │ │ │ + * (null if parts of the geometry could not be drawn, false if nothing │ │ │ │ │ + * could be drawn) │ │ │ │ │ + */ │ │ │ │ │ + drawGeometryNode: function(node, geometry, style) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + │ │ │ │ │ + var options = { │ │ │ │ │ + 'isFilled': style.fill === undefined ? │ │ │ │ │ + true : style.fill, │ │ │ │ │ + 'isStroked': style.stroke === undefined ? │ │ │ │ │ + !!style.strokeWidth : style.stroke │ │ │ │ │ + }; │ │ │ │ │ + var drawn; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.graphic === false) { │ │ │ │ │ + options.isFilled = false; │ │ │ │ │ + options.isStroked = false; │ │ │ │ │ + } │ │ │ │ │ + drawn = this.drawPoint(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + options.isFilled = false; │ │ │ │ │ + drawn = this.drawLineString(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + drawn = this.drawLinearRing(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + drawn = this.drawPolygon(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + drawn = this.drawRectangle(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + node._options = options; │ │ │ │ │ + │ │ │ │ │ + //set style │ │ │ │ │ + //TBD simplify this │ │ │ │ │ + if (drawn != false) { │ │ │ │ │ + return { │ │ │ │ │ + node: this.setStyle(node, style, options, geometry), │ │ │ │ │ + complete: drawn │ │ │ │ │ + }; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: postDraw │ │ │ │ │ + * Things that have do be done after the geometry node is appended │ │ │ │ │ + * to its parent node. To be overridden by subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + postDraw: function(node) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawPoint │ │ │ │ │ + * Virtual function for drawing Point Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the point │ │ │ │ │ + */ │ │ │ │ │ + drawPoint: function(node, geometry) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawLineString │ │ │ │ │ + * Virtual function for drawing LineString Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components of │ │ │ │ │ + * the linestring, or false if nothing could be drawn │ │ │ │ │ + */ │ │ │ │ │ + drawLineString: function(node, geometry) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawLinearRing │ │ │ │ │ + * Virtual function for drawing LinearRing Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ + * of the linear ring, or false if nothing could be drawn │ │ │ │ │ + */ │ │ │ │ │ + drawLinearRing: function(node, geometry) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawPolygon │ │ │ │ │ + * Virtual function for drawing Polygon Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ + * of the polygon, or false if nothing could be drawn │ │ │ │ │ + */ │ │ │ │ │ + drawPolygon: function(node, geometry) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawRectangle │ │ │ │ │ + * Virtual function for drawing Rectangle Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the rectangle │ │ │ │ │ + */ │ │ │ │ │ + drawRectangle: function(node, geometry) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawCircle │ │ │ │ │ + * Virtual function for drawing Circle Geometry. │ │ │ │ │ + * Should be implemented by subclasses. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the circle │ │ │ │ │ + */ │ │ │ │ │ + drawCircle: function(node, geometry) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeText │ │ │ │ │ + * Removes a label │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + removeText: function(featureId) { │ │ │ │ │ + var label = document.getElementById(featureId + this.LABEL_ID_SUFFIX); │ │ │ │ │ + if (label) { │ │ │ │ │ + this.textRoot.removeChild(label); │ │ │ │ │ + } │ │ │ │ │ + var outline = document.getElementById(featureId + this.LABEL_OUTLINE_SUFFIX); │ │ │ │ │ + if (outline) { │ │ │ │ │ + this.textRoot.removeChild(outline); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getFeatureIdFromEvent │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} An <OpenLayers.Event> object │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A feature id or undefined. │ │ │ │ │ + */ │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var target = evt.target; │ │ │ │ │ + var useElement = target && target.correspondingUseElement; │ │ │ │ │ + var node = useElement ? useElement : (target || evt.srcElement); │ │ │ │ │ + return node._featureId; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: eraseGeometry │ │ │ │ │ + * Erase a geometry from the renderer. In the case of a multi-geometry, │ │ │ │ │ + * we cycle through and recurse on ourselves. Otherwise, we look for a │ │ │ │ │ + * node with the geometry.id, destroy its geometry, and remove it from │ │ │ │ │ + * the DOM. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + eraseGeometry: function(geometry, featureId) { │ │ │ │ │ + if ((geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ + (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ + (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPolygon") || │ │ │ │ │ + (geometry.CLASS_NAME == "OpenLayers.Geometry.Collection")) { │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + this.eraseGeometry(geometry.components[i], featureId); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + var element = OpenLayers.Util.getElement(geometry.id); │ │ │ │ │ + if (element && element.parentNode) { │ │ │ │ │ + if (element.geometry) { │ │ │ │ │ + element.geometry.destroy(); │ │ │ │ │ + element.geometry = null; │ │ │ │ │ + } │ │ │ │ │ + element.parentNode.removeChild(element); │ │ │ │ │ + │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + this.indexer.remove(element); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (element._style.backgroundGraphic) { │ │ │ │ │ + var backgroundId = geometry.id + this.BACKGROUND_ID_SUFFIX; │ │ │ │ │ + var bElem = OpenLayers.Util.getElement(backgroundId); │ │ │ │ │ + if (bElem && bElem.parentNode) { │ │ │ │ │ + // No need to destroy the geometry since the element and the background │ │ │ │ │ + // node share the same geometry. │ │ │ │ │ + bElem.parentNode.removeChild(bElem); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: nodeFactory │ │ │ │ │ + * Create new node of the specified type, with the (optional) specified id. │ │ │ │ │ + * │ │ │ │ │ + * If node already exists with same ID and a different type, we remove it │ │ │ │ │ + * and then call ourselves again to recreate it. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} │ │ │ │ │ + * type - {String} type Kind of node to draw. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A new node of the given type and id. │ │ │ │ │ + */ │ │ │ │ │ + nodeFactory: function(id, type) { │ │ │ │ │ + var node = OpenLayers.Util.getElement(id); │ │ │ │ │ + if (node) { │ │ │ │ │ + if (!this.nodeTypeCompare(node, type)) { │ │ │ │ │ + node.parentNode.removeChild(node); │ │ │ │ │ + node = this.nodeFactory(id, type); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + node = this.createNode(type, id); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: nodeTypeCompare │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * type - {String} Kind of node │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ + * This function must be overridden by subclasses. │ │ │ │ │ + */ │ │ │ │ │ + nodeTypeCompare: function(node, type) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createNode │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * type - {String} Kind of node to draw. │ │ │ │ │ + * id - {String} Id for node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A new node of the given type and id. │ │ │ │ │ + * This function must be overridden by subclasses. │ │ │ │ │ + */ │ │ │ │ │ + createNode: function(type, id) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveRoot │ │ │ │ │ + * moves this renderer's root to a different renderer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * renderer - {<OpenLayers.Renderer>} target renderer for the moved root │ │ │ │ │ + */ │ │ │ │ │ + moveRoot: function(renderer) { │ │ │ │ │ + var root = this.root; │ │ │ │ │ + if (renderer.root.parentNode == this.rendererRoot) { │ │ │ │ │ + root = renderer.root; │ │ │ │ │ + } │ │ │ │ │ + root.parentNode.removeChild(root); │ │ │ │ │ + renderer.rendererRoot.appendChild(root); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getRenderLayerId │ │ │ │ │ + * Gets the layer that this renderer's output appears on. If moveRoot was │ │ │ │ │ + * used, this will be different from the id of the layer containing the │ │ │ │ │ + * features rendered by this renderer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} the id of the output layer. │ │ │ │ │ + */ │ │ │ │ │ + getRenderLayerId: function() { │ │ │ │ │ + return this.root.parentNode.parentNode.id; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: isComplexSymbol │ │ │ │ │ + * Determines if a symbol cannot be rendered using drawCircle │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * graphicName - {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns │ │ │ │ │ + * {Boolean} true if the symbol is complex, false if not │ │ │ │ │ + */ │ │ │ │ │ + isComplexSymbol: function(graphicName) { │ │ │ │ │ + return (graphicName != "circle") && !!graphicName; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.Elements" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Renderer/SVG.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Renderer/Elements.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Renderer.SVG │ │ │ │ │ + * │ │ │ │ │ + * Inherits: │ │ │ │ │ + * - <OpenLayers.Renderer.Elements> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.SVG = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: xmlns │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + xmlns: "http://www.w3.org/2000/svg", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: xlinkns │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + xlinkns: "http://www.w3.org/1999/xlink", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: MAX_PIXEL │ │ │ │ │ + * {Integer} Firefox has a limitation where values larger or smaller than │ │ │ │ │ + * about 15000 in an SVG document lock the browser up. This │ │ │ │ │ + * works around it. │ │ │ │ │ + */ │ │ │ │ │ + MAX_PIXEL: 15000, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: translationParameters │ │ │ │ │ + * {Object} Hash with "x" and "y" properties │ │ │ │ │ + */ │ │ │ │ │ + translationParameters: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: symbolMetrics │ │ │ │ │ + * {Object} Cache for symbol metrics according to their svg coordinate │ │ │ │ │ + * space. This is an object keyed by the symbol's id, and values are │ │ │ │ │ + * an array of [width, centerX, centerY]. │ │ │ │ │ + */ │ │ │ │ │ + symbolMetrics: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Renderer.SVG │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * containerID - {String} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(containerID) { │ │ │ │ │ + if (!this.supported()) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.initialize.apply(this, │ │ │ │ │ + arguments); │ │ │ │ │ + this.translationParameters = { │ │ │ │ │ + x: 0, │ │ │ │ │ + y: 0 │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + this.symbolMetrics = {}; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: supported │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the browser supports the SVG renderer │ │ │ │ │ + */ │ │ │ │ │ + supported: function() { │ │ │ │ │ + var svgFeature = "http://www.w3.org/TR/SVG11/feature#"; │ │ │ │ │ + return (document.implementation && │ │ │ │ │ + (document.implementation.hasFeature("org.w3c.svg", "1.0") || │ │ │ │ │ + document.implementation.hasFeature(svgFeature + "SVG", "1.1") || │ │ │ │ │ + document.implementation.hasFeature(svgFeature + "BasicStructure", "1.1"))); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: inValidRange │ │ │ │ │ + * See #669 for more information │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * x - {Integer} │ │ │ │ │ + * y - {Integer} │ │ │ │ │ + * xyOnly - {Boolean} whether or not to just check for x and y, which means │ │ │ │ │ + * to not take the current translation parameters into account if true. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the 'x' and 'y' coordinates are in the │ │ │ │ │ + * valid range. │ │ │ │ │ + */ │ │ │ │ │ + inValidRange: function(x, y, xyOnly) { │ │ │ │ │ + var left = x + (xyOnly ? 0 : this.translationParameters.x); │ │ │ │ │ + var top = y + (xyOnly ? 0 : this.translationParameters.y); │ │ │ │ │ + return (left >= -this.MAX_PIXEL && left <= this.MAX_PIXEL && │ │ │ │ │ + top >= -this.MAX_PIXEL && top <= this.MAX_PIXEL); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setExtent │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} │ │ │ │ │ + * resolutionChanged - {Boolean} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ + * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ + * False otherwise. │ │ │ │ │ + */ │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + var resolution = this.getResolution(), │ │ │ │ │ + left = -extent.left / resolution, │ │ │ │ │ + top = extent.top / resolution; │ │ │ │ │ + │ │ │ │ │ + // If the resolution has changed, start over changing the corner, because │ │ │ │ │ + // the features will redraw. │ │ │ │ │ + if (resolutionChanged) { │ │ │ │ │ + this.left = left; │ │ │ │ │ + this.top = top; │ │ │ │ │ + // Set the viewbox │ │ │ │ │ + var extentString = "0 0 " + this.size.w + " " + this.size.h; │ │ │ │ │ + │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "viewBox", extentString); │ │ │ │ │ + this.translate(this.xOffset, 0); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + var inRange = this.translate(left - this.left + this.xOffset, top - this.top); │ │ │ │ │ + if (!inRange) { │ │ │ │ │ + // recenter the coordinate system │ │ │ │ │ + this.setExtent(extent, true); │ │ │ │ │ + } │ │ │ │ │ + return coordSysUnchanged && inRange; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: translate │ │ │ │ │ + * Transforms the SVG coordinate system │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * x - {Float} │ │ │ │ │ + * y - {Float} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the translation parameters are in the valid coordinates │ │ │ │ │ + * range, false otherwise. │ │ │ │ │ + */ │ │ │ │ │ + translate: function(x, y) { │ │ │ │ │ + if (!this.inValidRange(x, y, true)) { │ │ │ │ │ + return false; │ │ │ │ │ + } else { │ │ │ │ │ + var transformString = ""; │ │ │ │ │ + if (x || y) { │ │ │ │ │ + transformString = "translate(" + x + "," + y + ")"; │ │ │ │ │ + } │ │ │ │ │ + this.root.setAttributeNS(null, "transform", transformString); │ │ │ │ │ + this.translationParameters = { │ │ │ │ │ + x: x, │ │ │ │ │ + y: y │ │ │ │ │ + }; │ │ │ │ │ + return true; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setSize │ │ │ │ │ + * Sets the size of the drawing surface. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * size - {<OpenLayers.Size>} The size of the drawing surface │ │ │ │ │ + */ │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "width", this.size.w); │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "height", this.size.h); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getNodeType │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The corresponding node type for the specified geometry │ │ │ │ │ + */ │ │ │ │ │ + getNodeType: function(geometry, style) { │ │ │ │ │ + var nodeType = null; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + nodeType = "image"; │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + nodeType = "svg"; │ │ │ │ │ + } else { │ │ │ │ │ + nodeType = "circle"; │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + nodeType = "rect"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + nodeType = "polyline"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + nodeType = "polygon"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + case "OpenLayers.Geometry.Curve": │ │ │ │ │ + nodeType = "path"; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + return nodeType; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setStyle │ │ │ │ │ + * Use to set all the style attributes to a SVG node. │ │ │ │ │ + * │ │ │ │ │ + * Takes care to adjust stroke width and point radius to be │ │ │ │ │ + * resolution-relative │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {SVGDomElement} An SVG element to decorate │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * options - {Object} Currently supported options include │ │ │ │ │ + * 'isFilled' {Boolean} and │ │ │ │ │ + * 'isStroked' {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + setStyle: function(node, style, options) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + options = options || node._options; │ │ │ │ │ + │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + node.setAttributeNS(null, "title", title); │ │ │ │ │ + //Standards-conformant SVG │ │ │ │ │ + // Prevent duplicate nodes. See issue https://github.com/openlayers/openlayers/issues/92 │ │ │ │ │ + var titleNode = node.getElementsByTagName("title"); │ │ │ │ │ + if (titleNode.length > 0) { │ │ │ │ │ + titleNode[0].firstChild.textContent = title; │ │ │ │ │ + } else { │ │ │ │ │ + var label = this.nodeFactory(null, "title"); │ │ │ │ │ + label.textContent = title; │ │ │ │ │ + node.appendChild(label); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var r = parseFloat(node.getAttributeNS(null, "r")); │ │ │ │ │ + var widthFactor = 1; │ │ │ │ │ + var pos; │ │ │ │ │ + if (node._geometryClass == "OpenLayers.Geometry.Point" && r) { │ │ │ │ │ + node.style.visibility = ""; │ │ │ │ │ + if (style.graphic === false) { │ │ │ │ │ + node.style.visibility = "hidden"; │ │ │ │ │ + } else if (style.externalGraphic) { │ │ │ │ │ + pos = this.getPosition(node); │ │ │ │ │ + if (style.graphicWidth && style.graphicHeight) { │ │ │ │ │ + node.setAttributeNS(null, "preserveAspectRatio", "none"); │ │ │ │ │ + } │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ + var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ + style.graphicXOffset : -(0.5 * width); │ │ │ │ │ + var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ + style.graphicYOffset : -(0.5 * height); │ │ │ │ │ + │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ + │ │ │ │ │ + node.setAttributeNS(null, "x", (pos.x + xOffset).toFixed()); │ │ │ │ │ + node.setAttributeNS(null, "y", (pos.y + yOffset).toFixed()); │ │ │ │ │ + node.setAttributeNS(null, "width", width); │ │ │ │ │ + node.setAttributeNS(null, "height", height); │ │ │ │ │ + node.setAttributeNS(this.xlinkns, "xlink:href", style.externalGraphic); │ │ │ │ │ + node.setAttributeNS(null, "style", "opacity: " + opacity); │ │ │ │ │ + node.onclick = OpenLayers.Event.preventDefault; │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + // the symbol viewBox is three times as large as the symbol │ │ │ │ │ + var offset = style.pointRadius * 3; │ │ │ │ │ + var size = offset * 2; │ │ │ │ │ + var src = this.importSymbol(style.graphicName); │ │ │ │ │ + pos = this.getPosition(node); │ │ │ │ │ + widthFactor = this.symbolMetrics[src.id][0] * 3 / size; │ │ │ │ │ + │ │ │ │ │ + // remove the node from the dom before we modify it. This │ │ │ │ │ + // prevents various rendering issues in Safari and FF │ │ │ │ │ + var parent = node.parentNode; │ │ │ │ │ + var nextSibling = node.nextSibling; │ │ │ │ │ + if (parent) { │ │ │ │ │ + parent.removeChild(node); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // The more appropriate way to implement this would be use/defs, │ │ │ │ │ + // but due to various issues in several browsers, it is safer to │ │ │ │ │ + // copy the symbols instead of referencing them. │ │ │ │ │ + // See e.g. ticket http://trac.osgeo.org/openlayers/ticket/2985 │ │ │ │ │ + // and this email thread │ │ │ │ │ + // http://osgeo-org.1803224.n2.nabble.com/Select-Control-Ctrl-click-on-Feature-with-a-graphicName-opens-new-browser-window-tc5846039.html │ │ │ │ │ + node.firstChild && node.removeChild(node.firstChild); │ │ │ │ │ + node.appendChild(src.firstChild.cloneNode(true)); │ │ │ │ │ + node.setAttributeNS(null, "viewBox", src.getAttributeNS(null, "viewBox")); │ │ │ │ │ + │ │ │ │ │ + node.setAttributeNS(null, "width", size); │ │ │ │ │ + node.setAttributeNS(null, "height", size); │ │ │ │ │ + node.setAttributeNS(null, "x", pos.x - offset); │ │ │ │ │ + node.setAttributeNS(null, "y", pos.y - offset); │ │ │ │ │ + │ │ │ │ │ + // now that the node has all its new properties, insert it │ │ │ │ │ + // back into the dom where it was │ │ │ │ │ + if (nextSibling) { │ │ │ │ │ + parent.insertBefore(node, nextSibling); │ │ │ │ │ + } else if (parent) { │ │ │ │ │ + parent.appendChild(node); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "r", style.pointRadius); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var rotation = style.rotation; │ │ │ │ │ + │ │ │ │ │ + if ((rotation !== undefined || node._rotation !== undefined) && pos) { │ │ │ │ │ + node._rotation = rotation; │ │ │ │ │ + rotation |= 0; │ │ │ │ │ + if (node.nodeName !== "svg") { │ │ │ │ │ + node.setAttributeNS(null, "transform", │ │ │ │ │ + "rotate(" + rotation + " " + pos.x + " " + │ │ │ │ │ + pos.y + ")"); │ │ │ │ │ + } else { │ │ │ │ │ + var metrics = this.symbolMetrics[src.id]; │ │ │ │ │ + node.firstChild.setAttributeNS(null, "transform", "rotate(" + │ │ │ │ │ + rotation + " " + │ │ │ │ │ + metrics[1] + " " + │ │ │ │ │ + metrics[2] + ")"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (options.isFilled) { │ │ │ │ │ + node.setAttributeNS(null, "fill", style.fillColor); │ │ │ │ │ + node.setAttributeNS(null, "fill-opacity", style.fillOpacity); │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "fill", "none"); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (options.isStroked) { │ │ │ │ │ + node.setAttributeNS(null, "stroke", style.strokeColor); │ │ │ │ │ + node.setAttributeNS(null, "stroke-opacity", style.strokeOpacity); │ │ │ │ │ + node.setAttributeNS(null, "stroke-width", style.strokeWidth * widthFactor); │ │ │ │ │ + node.setAttributeNS(null, "stroke-linecap", style.strokeLinecap || "round"); │ │ │ │ │ + // Hard-coded linejoin for now, to make it look the same as in VML. │ │ │ │ │ + // There is no strokeLinejoin property yet for symbolizers. │ │ │ │ │ + node.setAttributeNS(null, "stroke-linejoin", "round"); │ │ │ │ │ + style.strokeDashstyle && node.setAttributeNS(null, │ │ │ │ │ + "stroke-dasharray", this.dashStyle(style, widthFactor)); │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "stroke", "none"); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (style.pointerEvents) { │ │ │ │ │ + node.setAttributeNS(null, "pointer-events", style.pointerEvents); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (style.cursor != null) { │ │ │ │ │ + node.setAttributeNS(null, "cursor", style.cursor); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: dashStyle │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * widthFactor - {Number} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A SVG compliant 'stroke-dasharray' value │ │ │ │ │ + */ │ │ │ │ │ + dashStyle: function(style, widthFactor) { │ │ │ │ │ + var w = style.strokeWidth * widthFactor; │ │ │ │ │ + var str = style.strokeDashstyle; │ │ │ │ │ + switch (str) { │ │ │ │ │ + case 'solid': │ │ │ │ │ + return 'none'; │ │ │ │ │ + case 'dot': │ │ │ │ │ + return [1, 4 * w].join(); │ │ │ │ │ + case 'dash': │ │ │ │ │ + return [4 * w, 4 * w].join(); │ │ │ │ │ + case 'dashdot': │ │ │ │ │ + return [4 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ + case 'longdash': │ │ │ │ │ + return [8 * w, 4 * w].join(); │ │ │ │ │ + case 'longdashdot': │ │ │ │ │ + return [8 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ + default: │ │ │ │ │ + return OpenLayers.String.trim(str).replace(/\s+/g, ","); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createNode │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * type - {String} Kind of node to draw │ │ │ │ │ + * id - {String} Id for node │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A new node of the given type and id │ │ │ │ │ + */ │ │ │ │ │ + createNode: function(type, id) { │ │ │ │ │ + var node = document.createElementNS(this.xmlns, type); │ │ │ │ │ + if (id) { │ │ │ │ │ + node.setAttributeNS(null, "id", id); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: nodeTypeCompare │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {SVGDomElement} An SVG element │ │ │ │ │ + * type - {String} Kind of node │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ + */ │ │ │ │ │ + nodeTypeCompare: function(node, type) { │ │ │ │ │ + return (type == node.nodeName); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createRenderRoot │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The specific render engine's root element │ │ │ │ │ + */ │ │ │ │ │ + createRenderRoot: function() { │ │ │ │ │ + var svg = this.nodeFactory(this.container.id + "_svgRoot", "svg"); │ │ │ │ │ + svg.style.display = "block"; │ │ │ │ │ + return svg; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createRoot │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * suffix - {String} suffix to append to the id │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + createRoot: function(suffix) { │ │ │ │ │ + return this.nodeFactory(this.container.id + suffix, "g"); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createDefs │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The element to which we'll add the symbol definitions │ │ │ │ │ + */ │ │ │ │ │ + createDefs: function() { │ │ │ │ │ + var defs = this.nodeFactory(this.container.id + "_defs", "defs"); │ │ │ │ │ + this.rendererRoot.appendChild(defs); │ │ │ │ │ + return defs; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /************************************** │ │ │ │ │ + * * │ │ │ │ │ + * GEOMETRY DRAWING FUNCTIONS * │ │ │ │ │ + * * │ │ │ │ │ + **************************************/ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawPoint │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the point │ │ │ │ │ + */ │ │ │ │ │ + drawPoint: function(node, geometry) { │ │ │ │ │ + return this.drawCircle(node, geometry, 1); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawCircle │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * radius - {Float} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the circle │ │ │ │ │ + */ │ │ │ │ │ + drawCircle: function(node, geometry, radius) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = ((geometry.x - this.featureDx) / resolution + this.left); │ │ │ │ │ + var y = (this.top - geometry.y / resolution); │ │ │ │ │ + │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + node.setAttributeNS(null, "cx", x); │ │ │ │ │ + node.setAttributeNS(null, "cy", y); │ │ │ │ │ + node.setAttributeNS(null, "r", radius); │ │ │ │ │ + return node; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawLineString │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components of │ │ │ │ │ + * the linestring, or false if nothing could be drawn │ │ │ │ │ + */ │ │ │ │ │ + drawLineString: function(node, geometry) { │ │ │ │ │ + var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ + if (componentsResult.path) { │ │ │ │ │ + node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ + return (componentsResult.complete ? node : null); │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawLinearRing │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ + * of the linear ring, or false if nothing could be drawn │ │ │ │ │ + */ │ │ │ │ │ + drawLinearRing: function(node, geometry) { │ │ │ │ │ + var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ + if (componentsResult.path) { │ │ │ │ │ + node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ + return (componentsResult.complete ? node : null); │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawPolygon │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ + * of the polygon, or false if nothing could be drawn │ │ │ │ │ + */ │ │ │ │ │ + drawPolygon: function(node, geometry) { │ │ │ │ │ + var d = ""; │ │ │ │ │ + var draw = true; │ │ │ │ │ + var complete = true; │ │ │ │ │ + var linearRingResult, path; │ │ │ │ │ + for (var j = 0, len = geometry.components.length; j < len; j++) { │ │ │ │ │ + d += " M"; │ │ │ │ │ + linearRingResult = this.getComponentsString( │ │ │ │ │ + geometry.components[j].components, " "); │ │ │ │ │ + path = linearRingResult.path; │ │ │ │ │ + if (path) { │ │ │ │ │ + d += " " + path; │ │ │ │ │ + complete = linearRingResult.complete && complete; │ │ │ │ │ + } else { │ │ │ │ │ + draw = false; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + d += " z"; │ │ │ │ │ + if (draw) { │ │ │ │ │ + node.setAttributeNS(null, "d", d); │ │ │ │ │ + node.setAttributeNS(null, "fill-rule", "evenodd"); │ │ │ │ │ + return complete ? node : null; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawRectangle │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the rectangle │ │ │ │ │ + */ │ │ │ │ │ + drawRectangle: function(node, geometry) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = ((geometry.x - this.featureDx) / resolution + this.left); │ │ │ │ │ + var y = (this.top - geometry.y / resolution); │ │ │ │ │ + │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + node.setAttributeNS(null, "x", x); │ │ │ │ │ + node.setAttributeNS(null, "y", y); │ │ │ │ │ + node.setAttributeNS(null, "width", geometry.width / resolution); │ │ │ │ │ + node.setAttributeNS(null, "height", geometry.height / resolution); │ │ │ │ │ + return node; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawText │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * style - │ │ │ │ │ + * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + */ │ │ │ │ │ + drawText: function(featureId, style, location) { │ │ │ │ │ + var drawOutline = (!!style.labelOutlineWidth); │ │ │ │ │ + // First draw text in halo color and size and overlay the │ │ │ │ │ + // normal text afterwards │ │ │ │ │ + if (drawOutline) { │ │ │ │ │ + var outlineStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ + outlineStyle.fontColor = outlineStyle.labelOutlineColor; │ │ │ │ │ + outlineStyle.fontStrokeColor = outlineStyle.labelOutlineColor; │ │ │ │ │ + outlineStyle.fontStrokeWidth = style.labelOutlineWidth; │ │ │ │ │ + if (style.labelOutlineOpacity) { │ │ │ │ │ + outlineStyle.fontOpacity = style.labelOutlineOpacity; │ │ │ │ │ + } │ │ │ │ │ + delete outlineStyle.labelOutlineWidth; │ │ │ │ │ + this.drawText(featureId, outlineStyle, location); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + │ │ │ │ │ + var x = ((location.x - this.featureDx) / resolution + this.left); │ │ │ │ │ + var y = (location.y / resolution - this.top); │ │ │ │ │ + │ │ │ │ │ + var suffix = (drawOutline) ? this.LABEL_OUTLINE_SUFFIX : this.LABEL_ID_SUFFIX; │ │ │ │ │ + var label = this.nodeFactory(featureId + suffix, "text"); │ │ │ │ │ + │ │ │ │ │ + label.setAttributeNS(null, "x", x); │ │ │ │ │ + label.setAttributeNS(null, "y", -y); │ │ │ │ │ + │ │ │ │ │ + if (style.fontColor) { │ │ │ │ │ + label.setAttributeNS(null, "fill", style.fontColor); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStrokeColor) { │ │ │ │ │ + label.setAttributeNS(null, "stroke", style.fontStrokeColor); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStrokeWidth) { │ │ │ │ │ + label.setAttributeNS(null, "stroke-width", style.fontStrokeWidth); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontOpacity) { │ │ │ │ │ + label.setAttributeNS(null, "opacity", style.fontOpacity); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontFamily) { │ │ │ │ │ + label.setAttributeNS(null, "font-family", style.fontFamily); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontSize) { │ │ │ │ │ + label.setAttributeNS(null, "font-size", style.fontSize); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontWeight) { │ │ │ │ │ + label.setAttributeNS(null, "font-weight", style.fontWeight); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStyle) { │ │ │ │ │ + label.setAttributeNS(null, "font-style", style.fontStyle); │ │ │ │ │ + } │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + label.setAttributeNS(null, "pointer-events", "visible"); │ │ │ │ │ + label._featureId = featureId; │ │ │ │ │ + } else { │ │ │ │ │ + label.setAttributeNS(null, "pointer-events", "none"); │ │ │ │ │ + } │ │ │ │ │ + var align = style.labelAlign || OpenLayers.Renderer.defaultSymbolizer.labelAlign; │ │ │ │ │ + label.setAttributeNS(null, "text-anchor", │ │ │ │ │ + OpenLayers.Renderer.SVG.LABEL_ALIGN[align[0]] || "middle"); │ │ │ │ │ + │ │ │ │ │ + if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ + label.setAttributeNS(null, "dominant-baseline", │ │ │ │ │ + OpenLayers.Renderer.SVG.LABEL_ALIGN[align[1]] || "central"); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var labelRows = style.label.split('\n'); │ │ │ │ │ + var numRows = labelRows.length; │ │ │ │ │ + while (label.childNodes.length > numRows) { │ │ │ │ │ + label.removeChild(label.lastChild); │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + var tspan = this.nodeFactory(featureId + suffix + "_tspan_" + i, "tspan"); │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + tspan._featureId = featureId; │ │ │ │ │ + tspan._geometry = location; │ │ │ │ │ + tspan._geometryClass = location.CLASS_NAME; │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.IS_GECKO === false) { │ │ │ │ │ + tspan.setAttributeNS(null, "baseline-shift", │ │ │ │ │ + OpenLayers.Renderer.SVG.LABEL_VSHIFT[align[1]] || "-35%"); │ │ │ │ │ + } │ │ │ │ │ + tspan.setAttribute("x", x); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + var vfactor = OpenLayers.Renderer.SVG.LABEL_VFACTOR[align[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5; │ │ │ │ │ + } │ │ │ │ │ + tspan.setAttribute("dy", (vfactor * (numRows - 1)) + "em"); │ │ │ │ │ + } else { │ │ │ │ │ + tspan.setAttribute("dy", "1em"); │ │ │ │ │ + } │ │ │ │ │ + tspan.textContent = (labelRows[i] === '') ? ' ' : labelRows[i]; │ │ │ │ │ + if (!tspan.parentNode) { │ │ │ │ │ + label.appendChild(tspan); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (!label.parentNode) { │ │ │ │ │ + this.textRoot.appendChild(label); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getComponentString │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * components - {Array(<OpenLayers.Geometry.Point>)} Array of points │ │ │ │ │ + * separator - {String} character between coordinate pairs. Defaults to "," │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} hash with properties "path" (the string created from the │ │ │ │ │ + * components and "complete" (false if the renderer was unable to │ │ │ │ │ + * draw all components) │ │ │ │ │ + */ │ │ │ │ │ + getComponentsString: function(components, separator) { │ │ │ │ │ + var renderCmp = []; │ │ │ │ │ + var complete = true; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + var strings = []; │ │ │ │ │ + var str, component; │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + component = components[i]; │ │ │ │ │ + renderCmp.push(component); │ │ │ │ │ + str = this.getShortString(component); │ │ │ │ │ + if (str) { │ │ │ │ │ + strings.push(str); │ │ │ │ │ + } else { │ │ │ │ │ + // The current component is outside the valid range. Let's │ │ │ │ │ + // see if the previous or next component is inside the range. │ │ │ │ │ + // If so, add the coordinate of the intersection with the │ │ │ │ │ + // valid range bounds. │ │ │ │ │ + if (i > 0) { │ │ │ │ │ + if (this.getShortString(components[i - 1])) { │ │ │ │ │ + strings.push(this.clipLine(components[i], │ │ │ │ │ + components[i - 1])); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (i < len - 1) { │ │ │ │ │ + if (this.getShortString(components[i + 1])) { │ │ │ │ │ + strings.push(this.clipLine(components[i], │ │ │ │ │ + components[i + 1])); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + complete = false; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return { │ │ │ │ │ + path: strings.join(separator || ","), │ │ │ │ │ + complete: complete │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clipLine │ │ │ │ │ + * Given two points (one inside the valid range, and one outside), │ │ │ │ │ + * clips the line betweeen the two points so that the new points are both │ │ │ │ │ + * inside the valid range. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * badComponent - {<OpenLayers.Geometry.Point>} original geometry of the │ │ │ │ │ + * invalid point │ │ │ │ │ + * goodComponent - {<OpenLayers.Geometry.Point>} original geometry of the │ │ │ │ │ + * valid point │ │ │ │ │ + * Returns │ │ │ │ │ + * {String} the SVG coordinate pair of the clipped point (like │ │ │ │ │ + * getShortString), or an empty string if both passed componets are at │ │ │ │ │ + * the same point. │ │ │ │ │ + */ │ │ │ │ │ + clipLine: function(badComponent, goodComponent) { │ │ │ │ │ + if (goodComponent.equals(badComponent)) { │ │ │ │ │ + return ""; │ │ │ │ │ + } │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var maxX = this.MAX_PIXEL - this.translationParameters.x; │ │ │ │ │ + var maxY = this.MAX_PIXEL - this.translationParameters.y; │ │ │ │ │ + var x1 = (goodComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y1 = this.top - goodComponent.y / resolution; │ │ │ │ │ + var x2 = (badComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y2 = this.top - badComponent.y / resolution; │ │ │ │ │ + var k; │ │ │ │ │ + if (x2 < -maxX || x2 > maxX) { │ │ │ │ │ + k = (y2 - y1) / (x2 - x1); │ │ │ │ │ + x2 = x2 < 0 ? -maxX : maxX; │ │ │ │ │ + y2 = y1 + (x2 - x1) * k; │ │ │ │ │ + } │ │ │ │ │ + if (y2 < -maxY || y2 > maxY) { │ │ │ │ │ + k = (x2 - x1) / (y2 - y1); │ │ │ │ │ + y2 = y2 < 0 ? -maxY : maxY; │ │ │ │ │ + x2 = x1 + (y2 - y1) * k; │ │ │ │ │ + } │ │ │ │ │ + return x2 + "," + y2; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getShortString │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} or false if point is outside the valid range │ │ │ │ │ + */ │ │ │ │ │ + getShortString: function(point) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = ((point.x - this.featureDx) / resolution + this.left); │ │ │ │ │ + var y = (this.top - point.y / resolution); │ │ │ │ │ + │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + return x + "," + y; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getPosition │ │ │ │ │ + * Finds the position of an svg node. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} hash with x and y properties, representing the coordinates │ │ │ │ │ + * within the svg coordinate system │ │ │ │ │ + */ │ │ │ │ │ + getPosition: function(node) { │ │ │ │ │ + return ({ │ │ │ │ │ + x: parseFloat(node.getAttributeNS(null, "cx")), │ │ │ │ │ + y: parseFloat(node.getAttributeNS(null, "cy")) │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: importSymbol │ │ │ │ │ + * add a new symbol definition from the rendererer's symbol hash │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * graphicName - {String} name of the symbol to import │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} - the imported symbol │ │ │ │ │ + */ │ │ │ │ │ + importSymbol: function(graphicName) { │ │ │ │ │ + if (!this.defs) { │ │ │ │ │ + // create svg defs tag │ │ │ │ │ + this.defs = this.createDefs(); │ │ │ │ │ + } │ │ │ │ │ + var id = this.container.id + "-" + graphicName; │ │ │ │ │ + │ │ │ │ │ + // check if symbol already exists in the defs │ │ │ │ │ + var existing = document.getElementById(id); │ │ │ │ │ + if (existing != null) { │ │ │ │ │ + return existing; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(graphicName + ' is not a valid symbol name'); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var symbolNode = this.nodeFactory(id, "symbol"); │ │ │ │ │ + var node = this.nodeFactory(null, "polygon"); │ │ │ │ │ + symbolNode.appendChild(node); │ │ │ │ │ + var symbolExtent = new OpenLayers.Bounds( │ │ │ │ │ + Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ + │ │ │ │ │ + var points = []; │ │ │ │ │ + var x, y; │ │ │ │ │ + for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ + symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ + symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ + symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ + points.push(x, ",", y); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + node.setAttributeNS(null, "points", points.join(" ")); │ │ │ │ │ + │ │ │ │ │ + var width = symbolExtent.getWidth(); │ │ │ │ │ + var height = symbolExtent.getHeight(); │ │ │ │ │ + // create a viewBox three times as large as the symbol itself, │ │ │ │ │ + // to allow for strokeWidth being displayed correctly at the corners. │ │ │ │ │ + var viewBox = [symbolExtent.left - width, │ │ │ │ │ + symbolExtent.bottom - height, width * 3, height * 3 │ │ │ │ │ + ]; │ │ │ │ │ + symbolNode.setAttributeNS(null, "viewBox", viewBox.join(" ")); │ │ │ │ │ + this.symbolMetrics[id] = [ │ │ │ │ │ + Math.max(width, height), │ │ │ │ │ + symbolExtent.getCenterLonLat().lon, │ │ │ │ │ + symbolExtent.getCenterLonLat().lat │ │ │ │ │ + ]; │ │ │ │ │ + │ │ │ │ │ + this.defs.appendChild(symbolNode); │ │ │ │ │ + return symbolNode; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getFeatureIdFromEvent │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} An <OpenLayers.Event> object │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A feature id or undefined. │ │ │ │ │ + */ │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var featureId = OpenLayers.Renderer.Elements.prototype.getFeatureIdFromEvent.apply(this, arguments); │ │ │ │ │ + if (!featureId) { │ │ │ │ │ + var target = evt.target; │ │ │ │ │ + featureId = target.parentNode && target != this.rendererRoot ? │ │ │ │ │ + target.parentNode._featureId : undefined; │ │ │ │ │ + } │ │ │ │ │ + return featureId; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.SVG" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.SVG.LABEL_ALIGN │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_ALIGN = { │ │ │ │ │ + "l": "start", │ │ │ │ │ + "r": "end", │ │ │ │ │ + "b": "bottom", │ │ │ │ │ + "t": "hanging" │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.SVG.LABEL_VSHIFT │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_VSHIFT = { │ │ │ │ │ + // according to │ │ │ │ │ + // http://www.w3.org/Graphics/SVG/Test/20061213/htmlObjectHarness/full-text-align-02-b.html │ │ │ │ │ + // a baseline-shift of -70% shifts the text exactly from the │ │ │ │ │ + // bottom to the top of the baseline, so -35% moves the text to │ │ │ │ │ + // the center of the baseline. │ │ │ │ │ + "t": "-70%", │ │ │ │ │ + "b": "0" │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.SVG.LABEL_VFACTOR │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_VFACTOR = { │ │ │ │ │ + "t": 0, │ │ │ │ │ + "b": -1 │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Function: OpenLayers.Renderer.SVG.preventDefault │ │ │ │ │ + * *Deprecated*. Use <OpenLayers.Event.preventDefault> method instead. │ │ │ │ │ + * Used to prevent default events (especially opening images in a new tab on │ │ │ │ │ + * ctrl-click) from being executed for externalGraphic symbols │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.SVG.preventDefault = function(e) { │ │ │ │ │ + OpenLayers.Event.preventDefault(e); │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Protocol.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol │ │ │ │ │ + * Abstract vector layer protocol class. Not to be instantiated directly. Use │ │ │ │ │ + * one of the protocol subclasses instead. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: format │ │ │ │ │ + * {<OpenLayers.Format>} The format used by this protocol. │ │ │ │ │ + */ │ │ │ │ │ + format: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: options │ │ │ │ │ + * {Object} Any options sent to the constructor. │ │ │ │ │ + */ │ │ │ │ │ + options: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: autoDestroy │ │ │ │ │ + * {Boolean} The creator of the protocol can set autoDestroy to false │ │ │ │ │ + * to fully control when the protocol is destroyed. Defaults to │ │ │ │ │ + * true. │ │ │ │ │ + */ │ │ │ │ │ + autoDestroy: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultFilter │ │ │ │ │ + * {<OpenLayers.Filter>} Optional default filter to read requests │ │ │ │ │ + */ │ │ │ │ │ + defaultFilter: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Protocol │ │ │ │ │ + * Abstract class for vector protocols. Create instances of a subclass. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.options = options; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: mergeWithDefaultFilter │ │ │ │ │ + * Merge filter passed to the read method with the default one │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {<OpenLayers.Filter>} │ │ │ │ │ + */ │ │ │ │ │ + mergeWithDefaultFilter: function(filter) { │ │ │ │ │ + var merged; │ │ │ │ │ + if (filter && this.defaultFilter) { │ │ │ │ │ + merged = new OpenLayers.Filter.Logical({ │ │ │ │ │ + type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ + filters: [this.defaultFilter, filter] │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + merged = filter || this.defaultFilter || undefined; │ │ │ │ │ + } │ │ │ │ │ + return merged; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up the protocol. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.options = null; │ │ │ │ │ + this.format = null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Construct a request for reading new features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, the same object will be passed to the callback function passed │ │ │ │ │ + * if one exists in the options object. │ │ │ │ │ + */ │ │ │ │ │ + read: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.filter = this.mergeWithDefaultFilter(options.filter); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: create │ │ │ │ │ + * Construct a request for writing newly created features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, the same object will be passed to the callback function passed │ │ │ │ │ + * if one exists in the options object. │ │ │ │ │ + */ │ │ │ │ │ + create: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: update │ │ │ │ │ + * Construct a request updating modified features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, the same object will be passed to the callback function passed │ │ │ │ │ + * if one exists in the options object. │ │ │ │ │ + */ │ │ │ │ │ + update: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: delete │ │ │ │ │ + * Construct a request deleting a removed feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, the same object will be passed to the callback function passed │ │ │ │ │ + * if one exists in the options object. │ │ │ │ │ + */ │ │ │ │ │ + "delete": function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: commit │ │ │ │ │ + * Go over the features and for each take action │ │ │ │ │ + * based on the feature state. Possible actions are create, │ │ │ │ │ + * update and delete. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array({<OpenLayers.Feature.Vector>})} │ │ │ │ │ + * options - {Object} Object whose possible keys are "create", "update", │ │ │ │ │ + * "delete", "callback" and "scope", the values referenced by the │ │ │ │ │ + * first three are objects as passed to the "create", "update", and │ │ │ │ │ + * "delete" methods, the value referenced by the "callback" key is │ │ │ │ │ + * a function which is called when the commit operation is complete │ │ │ │ │ + * using the scope referenced by the "scope" key. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array({<OpenLayers.Protocol.Response>})} An array of │ │ │ │ │ + * <OpenLayers.Protocol.Response> objects. │ │ │ │ │ + */ │ │ │ │ │ + commit: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: abort │ │ │ │ │ + * Abort an ongoing request. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ + */ │ │ │ │ │ + abort: function(response) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createCallback │ │ │ │ │ + * Returns a function that applies the given public method with resp and │ │ │ │ │ + * options arguments. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * method - {Function} The method to be applied by the callback. │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The protocol response object. │ │ │ │ │ + * options - {Object} Options sent to the protocol method │ │ │ │ │ + */ │ │ │ │ │ + createCallback: function(method, response, options) { │ │ │ │ │ + return OpenLayers.Function.bind(function() { │ │ │ │ │ + method.apply(this, [response, options]); │ │ │ │ │ + }, this); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.Response │ │ │ │ │ + * Protocols return Response objects to their users. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.Response = OpenLayers.Class({ │ │ │ │ │ + /** │ │ │ │ │ + * Property: code │ │ │ │ │ + * {Number} - OpenLayers.Protocol.Response.SUCCESS or │ │ │ │ │ + * OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ + */ │ │ │ │ │ + code: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: requestType │ │ │ │ │ + * {String} The type of request this response corresponds to. Either │ │ │ │ │ + * "create", "read", "update" or "delete". │ │ │ │ │ + */ │ │ │ │ │ + requestType: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: last │ │ │ │ │ + * {Boolean} - true if this is the last response expected in a commit, │ │ │ │ │ + * false otherwise, defaults to true. │ │ │ │ │ + */ │ │ │ │ │ + last: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Array({<OpenLayers.Feature.Vector>})} or {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * The features returned in the response by the server. Depending on the │ │ │ │ │ + * protocol's read payload, either features or data will be populated. │ │ │ │ │ + */ │ │ │ │ │ + features: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: data │ │ │ │ │ + * {Object} │ │ │ │ │ + * The data returned in the response by the server. Depending on the │ │ │ │ │ + * protocol's read payload, either features or data will be populated. │ │ │ │ │ + */ │ │ │ │ │ + data: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: reqFeatures │ │ │ │ │ + * {Array({<OpenLayers.Feature.Vector>})} or {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * The features provided by the user and placed in the request by the │ │ │ │ │ + * protocol. │ │ │ │ │ + */ │ │ │ │ │ + reqFeatures: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: priv │ │ │ │ │ + */ │ │ │ │ │ + priv: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: error │ │ │ │ │ + * {Object} The error object in case a service exception was encountered. │ │ │ │ │ + */ │ │ │ │ │ + error: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Protocol.Response │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: success │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} - true on success, false otherwise │ │ │ │ │ + */ │ │ │ │ │ + success: function() { │ │ │ │ │ + return this.code > 0; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.Response" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Protocol.Response.SUCCESS = 1; │ │ │ │ │ +OpenLayers.Protocol.Response.FAILURE = 0; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Protocol/WFS.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Protocol.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.WFS │ │ │ │ │ + * Used to create a versioned WFS protocol. Default version is 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol>} A WFS protocol of the given version. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var protocol = new OpenLayers.Protocol.WFS({ │ │ │ │ │ + * version: "1.1.0", │ │ │ │ │ + * url: "http://demo.opengeo.org/geoserver/wfs", │ │ │ │ │ + * featureType: "tasmania_roads", │ │ │ │ │ + * featureNS: "http://www.openplans.org/topp", │ │ │ │ │ + * geometryName: "the_geom" │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * See the protocols for specific WFS versions for more detail. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.WFS = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options, OpenLayers.Protocol.WFS.DEFAULTS │ │ │ │ │ + ); │ │ │ │ │ + var cls = OpenLayers.Protocol.WFS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported WFS version: " + options.version; │ │ │ │ │ + } │ │ │ │ │ + return new cls(options); │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Function: fromWMSLayer │ │ │ │ │ + * Convenience function to create a WFS protocol from a WMS layer. This makes │ │ │ │ │ + * the assumption that a WFS requests can be issued at the same URL as │ │ │ │ │ + * WMS requests and that a WFS featureType exists with the same name as the │ │ │ │ │ + * WMS layer. │ │ │ │ │ + * │ │ │ │ │ + * This function is designed to auto-configure <url>, <featureType>, │ │ │ │ │ + * <featurePrefix> and <srsName> for WFS <version> 1.1.0. Note that │ │ │ │ │ + * srsName matching with the WMS layer will not work with WFS 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer.WMS>} WMS layer that has a matching WFS │ │ │ │ │ + * FeatureType at the same server url with the same typename. │ │ │ │ │ + * options - {Object} Default properties to be set on the protocol. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.WFS>} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.WFS.fromWMSLayer = function(layer, options) { │ │ │ │ │ + var typeName, featurePrefix; │ │ │ │ │ + var param = layer.params["LAYERS"]; │ │ │ │ │ + var parts = (OpenLayers.Util.isArray(param) ? param[0] : param).split(":"); │ │ │ │ │ + if (parts.length > 1) { │ │ │ │ │ + featurePrefix = parts[0]; │ │ │ │ │ + } │ │ │ │ │ + typeName = parts.pop(); │ │ │ │ │ + var protocolOptions = { │ │ │ │ │ + url: layer.url, │ │ │ │ │ + featureType: typeName, │ │ │ │ │ + featurePrefix: featurePrefix, │ │ │ │ │ + srsName: layer.projection && layer.projection.getCode() || │ │ │ │ │ + layer.map && layer.map.getProjectionObject().getCode(), │ │ │ │ │ + version: "1.1.0" │ │ │ │ │ + }; │ │ │ │ │ + return new OpenLayers.Protocol.WFS(OpenLayers.Util.applyDefaults( │ │ │ │ │ + options, protocolOptions │ │ │ │ │ + )); │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Protocol.WFS.DEFAULTS │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.WFS.DEFAULTS = { │ │ │ │ │ + "version": "1.0.0" │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Request.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ + * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * TODO: deprecate me │ │ │ │ │ + * Use OpenLayers.Request.proxy instead. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.ProxyHost = ""; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Request │ │ │ │ │ + * The OpenLayers.Request namespace contains convenience methods for working │ │ │ │ │ + * with XMLHttpRequests. These methods work with a cross-browser │ │ │ │ │ + * W3C compliant <OpenLayers.Request.XMLHttpRequest> class. │ │ │ │ │ + */ │ │ │ │ │ +if (!OpenLayers.Request) { │ │ │ │ │ + /** │ │ │ │ │ + * This allows for OpenLayers/Request/XMLHttpRequest.js to be included │ │ │ │ │ + * before or after this script. │ │ │ │ │ + */ │ │ │ │ │ + OpenLayers.Request = {}; │ │ │ │ │ +} │ │ │ │ │ +OpenLayers.Util.extend(OpenLayers.Request, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: DEFAULT_CONFIG │ │ │ │ │ + * {Object} Default configuration for all requests. │ │ │ │ │ + */ │ │ │ │ │ + DEFAULT_CONFIG: { │ │ │ │ │ + method: "GET", │ │ │ │ │ + url: window.location.href, │ │ │ │ │ + async: true, │ │ │ │ │ + user: undefined, │ │ │ │ │ + password: undefined, │ │ │ │ │ + params: null, │ │ │ │ │ + proxy: OpenLayers.ProxyHost, │ │ │ │ │ + headers: {}, │ │ │ │ │ + data: null, │ │ │ │ │ + callback: function() {}, │ │ │ │ │ + success: null, │ │ │ │ │ + failure: null, │ │ │ │ │ + scope: null │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: URL_SPLIT_REGEX │ │ │ │ │ + */ │ │ │ │ │ + URL_SPLIT_REGEX: /([^:]*:)\/\/([^:]*:?[^@]*@)?([^:\/\?]*):?([^\/\?]*)/, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} An events object that handles all │ │ │ │ │ + * events on the {<OpenLayers.Request>} object. │ │ │ │ │ + * │ │ │ │ │ + * All event listeners will receive an event object with three properties: │ │ │ │ │ + * request - {<OpenLayers.Request.XMLHttpRequest>} The request object. │ │ │ │ │ + * config - {Object} The config object sent to the specific request method. │ │ │ │ │ + * requestUrl - {String} The request url. │ │ │ │ │ + * │ │ │ │ │ + * Supported event types: │ │ │ │ │ + * complete - Triggered when we have a response from the request, if a │ │ │ │ │ + * listener returns false, no further response processing will take │ │ │ │ │ + * place. │ │ │ │ │ + * success - Triggered when the HTTP response has a success code (200-299). │ │ │ │ │ + * failure - Triggered when the HTTP response does not have a success code. │ │ │ │ │ + */ │ │ │ │ │ + events: new OpenLayers.Events(this), │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: makeSameOrigin │ │ │ │ │ + * Using the specified proxy, returns a same origin url of the provided url. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * url - {String} An arbitrary url │ │ │ │ │ + * proxy {String|Function} The proxy to use to make the provided url a │ │ │ │ │ + * same origin url. │ │ │ │ │ + * │ │ │ │ │ + * Returns │ │ │ │ │ + * {String} the same origin url. If no proxy is provided, the returned url │ │ │ │ │ + * will be the same as the provided url. │ │ │ │ │ + */ │ │ │ │ │ + makeSameOrigin: function(url, proxy) { │ │ │ │ │ + var sameOrigin = url.indexOf("http") !== 0; │ │ │ │ │ + var urlParts = !sameOrigin && url.match(this.URL_SPLIT_REGEX); │ │ │ │ │ + if (urlParts) { │ │ │ │ │ + var location = window.location; │ │ │ │ │ + sameOrigin = │ │ │ │ │ + urlParts[1] == location.protocol && │ │ │ │ │ + urlParts[3] == location.hostname; │ │ │ │ │ + var uPort = urlParts[4], │ │ │ │ │ + lPort = location.port; │ │ │ │ │ + if (uPort != 80 && uPort != "" || lPort != "80" && lPort != "") { │ │ │ │ │ + sameOrigin = sameOrigin && uPort == lPort; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!sameOrigin) { │ │ │ │ │ + if (proxy) { │ │ │ │ │ + if (typeof proxy == "function") { │ │ │ │ │ + url = proxy(url); │ │ │ │ │ + } else { │ │ │ │ │ + url = proxy + encodeURIComponent(url); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return url; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: issue │ │ │ │ │ + * Create a new XMLHttpRequest object, open it, set any headers, bind │ │ │ │ │ + * a callback to done state, and send any data. It is recommended that │ │ │ │ │ + * you use one <GET>, <POST>, <PUT>, <DELETE>, <OPTIONS>, or <HEAD>. │ │ │ │ │ + * This method is only documented to provide detail on the configuration │ │ │ │ │ + * options available to all request methods. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Object containing properties for configuring the │ │ │ │ │ + * request. Allowed configuration properties are described below. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Allowed config properties: │ │ │ │ │ + * method - {String} One of GET, POST, PUT, DELETE, HEAD, or │ │ │ │ │ + * OPTIONS. Default is GET. │ │ │ │ │ + * url - {String} URL for the request. │ │ │ │ │ + * async - {Boolean} Open an asynchronous request. Default is true. │ │ │ │ │ + * user - {String} User for relevant authentication scheme. Set │ │ │ │ │ + * to null to clear current user. │ │ │ │ │ + * password - {String} Password for relevant authentication scheme. │ │ │ │ │ + * Set to null to clear current password. │ │ │ │ │ + * proxy - {String} Optional proxy. Defaults to │ │ │ │ │ + * <OpenLayers.ProxyHost>. │ │ │ │ │ + * params - {Object} Any key:value pairs to be appended to the │ │ │ │ │ + * url as a query string. Assumes url doesn't already include a query │ │ │ │ │ + * string or hash. Typically, this is only appropriate for <GET> │ │ │ │ │ + * requests where the query string will be appended to the url. │ │ │ │ │ + * Parameter values that are arrays will be │ │ │ │ │ + * concatenated with a comma (note that this goes against form-encoding) │ │ │ │ │ + * as is done with <OpenLayers.Util.getParameterString>. │ │ │ │ │ + * headers - {Object} Object with header:value pairs to be set on │ │ │ │ │ + * the request. │ │ │ │ │ + * data - {String | Document} Optional data to send with the request. │ │ │ │ │ + * Typically, this is only used with <POST> and <PUT> requests. │ │ │ │ │ + * Make sure to provide the appropriate "Content-Type" header for your │ │ │ │ │ + * data. For <POST> and <PUT> requests, the content type defaults to │ │ │ │ │ + * "application-xml". If your data is a different content type, or │ │ │ │ │ + * if you are using a different HTTP method, set the "Content-Type" │ │ │ │ │ + * header to match your data type. │ │ │ │ │ + * callback - {Function} Function to call when request is done. │ │ │ │ │ + * To determine if the request failed, check request.status (200 │ │ │ │ │ + * indicates success). │ │ │ │ │ + * success - {Function} Optional function to call if request status is in │ │ │ │ │ + * the 200s. This will be called in addition to callback above and │ │ │ │ │ + * would typically only be used as an alternative. │ │ │ │ │ + * failure - {Function} Optional function to call if request status is not │ │ │ │ │ + * in the 200s. This will be called in addition to callback above and │ │ │ │ │ + * would typically only be used as an alternative. │ │ │ │ │ + * scope - {Object} If callback is a public method on some object, │ │ │ │ │ + * set the scope to that object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLHttpRequest} Request object. To abort the request before a response │ │ │ │ │ + * is received, call abort() on the request object. │ │ │ │ │ + */ │ │ │ │ │ + issue: function(config) { │ │ │ │ │ + // apply default config - proxy host may have changed │ │ │ │ │ + var defaultConfig = OpenLayers.Util.extend( │ │ │ │ │ + this.DEFAULT_CONFIG, { │ │ │ │ │ + proxy: OpenLayers.ProxyHost │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + config = config || {}; │ │ │ │ │ + config.headers = config.headers || {}; │ │ │ │ │ + config = OpenLayers.Util.applyDefaults(config, defaultConfig); │ │ │ │ │ + config.headers = OpenLayers.Util.applyDefaults(config.headers, defaultConfig.headers); │ │ │ │ │ + // Always set the "X-Requested-With" header to signal that this request │ │ │ │ │ + // was issued through the XHR-object. Since header keys are case │ │ │ │ │ + // insensitive and we want to allow overriding of the "X-Requested-With" │ │ │ │ │ + // header through the user we cannot use applyDefaults, but have to │ │ │ │ │ + // check manually whether we were called with a "X-Requested-With" │ │ │ │ │ + // header. │ │ │ │ │ + var customRequestedWithHeader = false, │ │ │ │ │ + headerKey; │ │ │ │ │ + for (headerKey in config.headers) { │ │ │ │ │ + if (config.headers.hasOwnProperty(headerKey)) { │ │ │ │ │ + if (headerKey.toLowerCase() === 'x-requested-with') { │ │ │ │ │ + customRequestedWithHeader = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (customRequestedWithHeader === false) { │ │ │ │ │ + // we did not have a custom "X-Requested-With" header │ │ │ │ │ + config.headers['X-Requested-With'] = 'XMLHttpRequest'; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // create request, open, and set headers │ │ │ │ │ + var request = new OpenLayers.Request.XMLHttpRequest(); │ │ │ │ │ + var url = OpenLayers.Util.urlAppend(config.url, │ │ │ │ │ + OpenLayers.Util.getParameterString(config.params || {})); │ │ │ │ │ + url = OpenLayers.Request.makeSameOrigin(url, config.proxy); │ │ │ │ │ + request.open( │ │ │ │ │ + config.method, url, config.async, config.user, config.password │ │ │ │ │ + ); │ │ │ │ │ + for (var header in config.headers) { │ │ │ │ │ + request.setRequestHeader(header, config.headers[header]); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var events = this.events; │ │ │ │ │ + │ │ │ │ │ + // we want to execute runCallbacks with "this" as the │ │ │ │ │ + // execution scope │ │ │ │ │ + var self = this; │ │ │ │ │ + │ │ │ │ │ + request.onreadystatechange = function() { │ │ │ │ │ + if (request.readyState == OpenLayers.Request.XMLHttpRequest.DONE) { │ │ │ │ │ + var proceed = events.triggerEvent( │ │ │ │ │ + "complete", { │ │ │ │ │ + request: request, │ │ │ │ │ + config: config, │ │ │ │ │ + requestUrl: url │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + if (proceed !== false) { │ │ │ │ │ + self.runCallbacks({ │ │ │ │ │ + request: request, │ │ │ │ │ + config: config, │ │ │ │ │ + requestUrl: url │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // send request (optionally with data) and return │ │ │ │ │ + // call in a timeout for asynchronous requests so the return is │ │ │ │ │ + // available before readyState == 4 for cached docs │ │ │ │ │ + if (config.async === false) { │ │ │ │ │ + request.send(config.data); │ │ │ │ │ + } else { │ │ │ │ │ + window.setTimeout(function() { │ │ │ │ │ + if (request.readyState !== 0) { // W3C: 0-UNSENT │ │ │ │ │ + request.send(config.data); │ │ │ │ │ + } │ │ │ │ │ + }, 0); │ │ │ │ │ + } │ │ │ │ │ + return request; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: runCallbacks │ │ │ │ │ + * Calls the complete, success and failure callbacks. Application │ │ │ │ │ + * can listen to the "complete" event, have the listener │ │ │ │ │ + * display a confirm window and always return false, and │ │ │ │ │ + * execute OpenLayers.Request.runCallbacks if the user │ │ │ │ │ + * hits "yes" in the confirm window. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Hash containing request, config and requestUrl keys │ │ │ │ │ + */ │ │ │ │ │ + runCallbacks: function(options) { │ │ │ │ │ + var request = options.request; │ │ │ │ │ + var config = options.config; │ │ │ │ │ + │ │ │ │ │ + // bind callbacks to readyState 4 (done) │ │ │ │ │ + var complete = (config.scope) ? │ │ │ │ │ + OpenLayers.Function.bind(config.callback, config.scope) : │ │ │ │ │ + config.callback; │ │ │ │ │ + │ │ │ │ │ + // optional success callback │ │ │ │ │ + var success; │ │ │ │ │ + if (config.success) { │ │ │ │ │ + success = (config.scope) ? │ │ │ │ │ + OpenLayers.Function.bind(config.success, config.scope) : │ │ │ │ │ + config.success; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // optional failure callback │ │ │ │ │ + var failure; │ │ │ │ │ + if (config.failure) { │ │ │ │ │ + failure = (config.scope) ? │ │ │ │ │ + OpenLayers.Function.bind(config.failure, config.scope) : │ │ │ │ │ + config.failure; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (OpenLayers.Util.createUrlObject(config.url).protocol == "file:" && │ │ │ │ │ + request.responseText) { │ │ │ │ │ + request.status = 200; │ │ │ │ │ + } │ │ │ │ │ + complete(request); │ │ │ │ │ + │ │ │ │ │ + if (!request.status || (request.status >= 200 && request.status < 300)) { │ │ │ │ │ + this.events.triggerEvent("success", options); │ │ │ │ │ + if (success) { │ │ │ │ │ + success(request); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ + this.events.triggerEvent("failure", options); │ │ │ │ │ + if (failure) { │ │ │ │ │ + failure(request); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: GET │ │ │ │ │ + * Send an HTTP GET request. Additional configuration properties are │ │ │ │ │ + * documented in the <issue> method, with the method property set │ │ │ │ │ + * to GET. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the <issue> method for documentation of allowed properties. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ + */ │ │ │ │ │ + GET: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "GET" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: POST │ │ │ │ │ + * Send a POST request. Additional configuration properties are │ │ │ │ │ + * documented in the <issue> method, with the method property set │ │ │ │ │ + * to POST and "Content-Type" header set to "application/xml". │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the <issue> method for documentation of allowed properties. The │ │ │ │ │ + * default "Content-Type" header will be set to "application-xml" if │ │ │ │ │ + * none is provided. This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ + */ │ │ │ │ │ + POST: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "POST" │ │ │ │ │ + }); │ │ │ │ │ + // set content type to application/xml if it isn't already set │ │ │ │ │ + config.headers = config.headers ? config.headers : {}; │ │ │ │ │ + if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ + config.headers["Content-Type"] = "application/xml"; │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: PUT │ │ │ │ │ + * Send an HTTP PUT request. Additional configuration properties are │ │ │ │ │ + * documented in the <issue> method, with the method property set │ │ │ │ │ + * to PUT and "Content-Type" header set to "application/xml". │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the <issue> method for documentation of allowed properties. The │ │ │ │ │ + * default "Content-Type" header will be set to "application-xml" if │ │ │ │ │ + * none is provided. This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ + */ │ │ │ │ │ + PUT: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "PUT" │ │ │ │ │ + }); │ │ │ │ │ + // set content type to application/xml if it isn't already set │ │ │ │ │ + config.headers = config.headers ? config.headers : {}; │ │ │ │ │ + if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ + config.headers["Content-Type"] = "application/xml"; │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: DELETE │ │ │ │ │ + * Send an HTTP DELETE request. Additional configuration properties are │ │ │ │ │ + * documented in the <issue> method, with the method property set │ │ │ │ │ + * to DELETE. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the <issue> method for documentation of allowed properties. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ + */ │ │ │ │ │ + DELETE: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "DELETE" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: HEAD │ │ │ │ │ + * Send an HTTP HEAD request. Additional configuration properties are │ │ │ │ │ + * documented in the <issue> method, with the method property set │ │ │ │ │ + * to HEAD. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the <issue> method for documentation of allowed properties. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ + */ │ │ │ │ │ + HEAD: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "HEAD" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: OPTIONS │ │ │ │ │ + * Send an HTTP OPTIONS request. Additional configuration properties are │ │ │ │ │ + * documented in the <issue> method, with the method property set │ │ │ │ │ + * to OPTIONS. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Object with properties for configuring the request. │ │ │ │ │ + * See the <issue> method for documentation of allowed properties. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLHttpRequest} Request object. │ │ │ │ │ + */ │ │ │ │ │ + OPTIONS: function(config) { │ │ │ │ │ + config = OpenLayers.Util.extend(config, { │ │ │ │ │ + method: "OPTIONS" │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Request.issue(config); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +// XMLHttpRequest.js Copyright (C) 2010 Sergey Ilinsky (http://www.ilinsky.com) │ │ │ │ │ +// │ │ │ │ │ +// Licensed under the Apache License, Version 2.0 (the "License"); │ │ │ │ │ +// you may not use this file except in compliance with the License. │ │ │ │ │ +// You may obtain a copy of the License at │ │ │ │ │ +// │ │ │ │ │ +// http://www.apache.org/licenses/LICENSE-2.0 │ │ │ │ │ +// │ │ │ │ │ +// Unless required by applicable law or agreed to in writing, software │ │ │ │ │ +// distributed under the License is distributed on an "AS IS" BASIS, │ │ │ │ │ +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. │ │ │ │ │ +// See the License for the specific language governing permissions and │ │ │ │ │ +// limitations under the License. │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Request.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +(function() { │ │ │ │ │ + │ │ │ │ │ + // Save reference to earlier defined object implementation (if any) │ │ │ │ │ + var oXMLHttpRequest = window.XMLHttpRequest; │ │ │ │ │ + │ │ │ │ │ + // Define on browser type │ │ │ │ │ + var bGecko = !!window.controllers, │ │ │ │ │ + bIE = window.document.all && !window.opera, │ │ │ │ │ + bIE7 = bIE && window.navigator.userAgent.match(/MSIE 7.0/); │ │ │ │ │ + │ │ │ │ │ + // Enables "XMLHttpRequest()" call next to "new XMLHttpReques()" │ │ │ │ │ + function fXMLHttpRequest() { │ │ │ │ │ + this._object = oXMLHttpRequest && !bIE7 ? new oXMLHttpRequest : new window.ActiveXObject("Microsoft.XMLHTTP"); │ │ │ │ │ + this._listeners = []; │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // Constructor │ │ │ │ │ + function cXMLHttpRequest() { │ │ │ │ │ + return new fXMLHttpRequest; │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype = fXMLHttpRequest.prototype; │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: Firefox with Firebug installed would break pages if not executed │ │ │ │ │ + if (bGecko && oXMLHttpRequest.wrapped) │ │ │ │ │ + cXMLHttpRequest.wrapped = oXMLHttpRequest.wrapped; │ │ │ │ │ + │ │ │ │ │ + // Constants │ │ │ │ │ + cXMLHttpRequest.UNSENT = 0; │ │ │ │ │ + cXMLHttpRequest.OPENED = 1; │ │ │ │ │ + cXMLHttpRequest.HEADERS_RECEIVED = 2; │ │ │ │ │ + cXMLHttpRequest.LOADING = 3; │ │ │ │ │ + cXMLHttpRequest.DONE = 4; │ │ │ │ │ + │ │ │ │ │ + // Public Properties │ │ │ │ │ + cXMLHttpRequest.prototype.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + cXMLHttpRequest.prototype.responseText = ''; │ │ │ │ │ + cXMLHttpRequest.prototype.responseXML = null; │ │ │ │ │ + cXMLHttpRequest.prototype.status = 0; │ │ │ │ │ + cXMLHttpRequest.prototype.statusText = ''; │ │ │ │ │ + │ │ │ │ │ + // Priority proposal │ │ │ │ │ + cXMLHttpRequest.prototype.priority = "NORMAL"; │ │ │ │ │ + │ │ │ │ │ + // Instance-level Events Handlers │ │ │ │ │ + cXMLHttpRequest.prototype.onreadystatechange = null; │ │ │ │ │ + │ │ │ │ │ + // Class-level Events Handlers │ │ │ │ │ + cXMLHttpRequest.onreadystatechange = null; │ │ │ │ │ + cXMLHttpRequest.onopen = null; │ │ │ │ │ + cXMLHttpRequest.onsend = null; │ │ │ │ │ + cXMLHttpRequest.onabort = null; │ │ │ │ │ + │ │ │ │ │ + // Public Methods │ │ │ │ │ + cXMLHttpRequest.prototype.open = function(sMethod, sUrl, bAsync, sUser, sPassword) { │ │ │ │ │ + // Delete headers, required when object is reused │ │ │ │ │ + delete this._headers; │ │ │ │ │ + │ │ │ │ │ + // When bAsync parameter value is omitted, use true as default │ │ │ │ │ + if (arguments.length < 3) │ │ │ │ │ + bAsync = true; │ │ │ │ │ + │ │ │ │ │ + // Save async parameter for fixing Gecko bug with missing readystatechange in synchronous requests │ │ │ │ │ + this._async = bAsync; │ │ │ │ │ + │ │ │ │ │ + // Set the onreadystatechange handler │ │ │ │ │ + var oRequest = this, │ │ │ │ │ + nState = this.readyState, │ │ │ │ │ + fOnUnload; │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: IE - memory leak on page unload (inter-page leak) │ │ │ │ │ + if (bIE && bAsync) { │ │ │ │ │ + fOnUnload = function() { │ │ │ │ │ + if (nState != cXMLHttpRequest.DONE) { │ │ │ │ │ + fCleanTransport(oRequest); │ │ │ │ │ + // Safe to abort here since onreadystatechange handler removed │ │ │ │ │ + oRequest.abort(); │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + window.attachEvent("onunload", fOnUnload); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // Add method sniffer │ │ │ │ │ + if (cXMLHttpRequest.onopen) │ │ │ │ │ + cXMLHttpRequest.onopen.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + if (arguments.length > 4) │ │ │ │ │ + this._object.open(sMethod, sUrl, bAsync, sUser, sPassword); │ │ │ │ │ + else │ │ │ │ │ + if (arguments.length > 3) │ │ │ │ │ + this._object.open(sMethod, sUrl, bAsync, sUser); │ │ │ │ │ + else │ │ │ │ │ + this._object.open(sMethod, sUrl, bAsync); │ │ │ │ │ + │ │ │ │ │ + this.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ + fReadyStateChange(this); │ │ │ │ │ + │ │ │ │ │ + this._object.onreadystatechange = function() { │ │ │ │ │ + if (bGecko && !bAsync) │ │ │ │ │ + return; │ │ │ │ │ + │ │ │ │ │ + // Synchronize state │ │ │ │ │ + oRequest.readyState = oRequest._object.readyState; │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + fSynchronizeValues(oRequest); │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: Firefox fires unnecessary DONE when aborting │ │ │ │ │ + if (oRequest._aborted) { │ │ │ │ │ + // Reset readyState to UNSENT │ │ │ │ │ + oRequest.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + │ │ │ │ │ + // Return now │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (oRequest.readyState == cXMLHttpRequest.DONE) { │ │ │ │ │ + // Free up queue │ │ │ │ │ + delete oRequest._data; │ │ │ │ │ + /* if (bAsync) │ │ │ │ │ + fQueue_remove(oRequest);*/ │ │ │ │ │ + // │ │ │ │ │ + fCleanTransport(oRequest); │ │ │ │ │ + // Uncomment this block if you need a fix for IE cache │ │ │ │ │ + /* │ │ │ │ │ + // BUGFIX: IE - cache issue │ │ │ │ │ + if (!oRequest._object.getResponseHeader("Date")) { │ │ │ │ │ + // Save object to cache │ │ │ │ │ + oRequest._cached = oRequest._object; │ │ │ │ │ + │ │ │ │ │ + // Instantiate a new transport object │ │ │ │ │ + cXMLHttpRequest.call(oRequest); │ │ │ │ │ + │ │ │ │ │ + // Re-send request │ │ │ │ │ + if (sUser) { │ │ │ │ │ + if (sPassword) │ │ │ │ │ + oRequest._object.open(sMethod, sUrl, bAsync, sUser, sPassword); │ │ │ │ │ + else │ │ │ │ │ + oRequest._object.open(sMethod, sUrl, bAsync, sUser); │ │ │ │ │ + } │ │ │ │ │ + else │ │ │ │ │ + oRequest._object.open(sMethod, sUrl, bAsync); │ │ │ │ │ + oRequest._object.setRequestHeader("If-Modified-Since", oRequest._cached.getResponseHeader("Last-Modified") || new window.Date(0)); │ │ │ │ │ + // Copy headers set │ │ │ │ │ + if (oRequest._headers) │ │ │ │ │ + for (var sHeader in oRequest._headers) │ │ │ │ │ + if (typeof oRequest._headers[sHeader] == "string") // Some frameworks prototype objects with functions │ │ │ │ │ + oRequest._object.setRequestHeader(sHeader, oRequest._headers[sHeader]); │ │ │ │ │ + │ │ │ │ │ + oRequest._object.onreadystatechange = function() { │ │ │ │ │ + // Synchronize state │ │ │ │ │ + oRequest.readyState = oRequest._object.readyState; │ │ │ │ │ + │ │ │ │ │ + if (oRequest._aborted) { │ │ │ │ │ + // │ │ │ │ │ + oRequest.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + │ │ │ │ │ + // Return │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (oRequest.readyState == cXMLHttpRequest.DONE) { │ │ │ │ │ + // Clean Object │ │ │ │ │ + fCleanTransport(oRequest); │ │ │ │ │ + │ │ │ │ │ + // get cached request │ │ │ │ │ + if (oRequest.status == 304) │ │ │ │ │ + oRequest._object = oRequest._cached; │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + delete oRequest._cached; │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + fSynchronizeValues(oRequest); │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + fReadyStateChange(oRequest); │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: IE - memory leak in interrupted │ │ │ │ │ + if (bIE && bAsync) │ │ │ │ │ + window.detachEvent("onunload", fOnUnload); │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + oRequest._object.send(null); │ │ │ │ │ + │ │ │ │ │ + // Return now - wait until re-sent request is finished │ │ │ │ │ + return; │ │ │ │ │ + }; │ │ │ │ │ + */ │ │ │ │ │ + // BUGFIX: IE - memory leak in interrupted │ │ │ │ │ + if (bIE && bAsync) │ │ │ │ │ + window.detachEvent("onunload", fOnUnload); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: Some browsers (Internet Explorer, Gecko) fire OPEN readystate twice │ │ │ │ │ + if (nState != oRequest.readyState) │ │ │ │ │ + fReadyStateChange(oRequest); │ │ │ │ │ + │ │ │ │ │ + nState = oRequest.readyState; │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fXMLHttpRequest_send(oRequest) { │ │ │ │ │ + oRequest._object.send(oRequest._data); │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: Gecko - missing readystatechange calls in synchronous requests │ │ │ │ │ + if (bGecko && !oRequest._async) { │ │ │ │ │ + oRequest.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ + │ │ │ │ │ + // Synchronize state │ │ │ │ │ + fSynchronizeValues(oRequest); │ │ │ │ │ + │ │ │ │ │ + // Simulate missing states │ │ │ │ │ + while (oRequest.readyState < cXMLHttpRequest.DONE) { │ │ │ │ │ + oRequest.readyState++; │ │ │ │ │ + fReadyStateChange(oRequest); │ │ │ │ │ + // Check if we are aborted │ │ │ │ │ + if (oRequest._aborted) │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.send = function(vData) { │ │ │ │ │ + // Add method sniffer │ │ │ │ │ + if (cXMLHttpRequest.onsend) │ │ │ │ │ + cXMLHttpRequest.onsend.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + if (!arguments.length) │ │ │ │ │ + vData = null; │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: Safari - fails sending documents created/modified dynamically, so an explicit serialization required │ │ │ │ │ + // BUGFIX: IE - rewrites any custom mime-type to "text/xml" in case an XMLNode is sent │ │ │ │ │ + // BUGFIX: Gecko - fails sending Element (this is up to the implementation either to standard) │ │ │ │ │ + if (vData && vData.nodeType) { │ │ │ │ │ + vData = window.XMLSerializer ? new window.XMLSerializer().serializeToString(vData) : vData.xml; │ │ │ │ │ + if (!this._headers["Content-Type"]) │ │ │ │ │ + this._object.setRequestHeader("Content-Type", "application/xml"); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this._data = vData; │ │ │ │ │ + /* │ │ │ │ │ + // Add to queue │ │ │ │ │ + if (this._async) │ │ │ │ │ + fQueue_add(this); │ │ │ │ │ + else*/ │ │ │ │ │ + fXMLHttpRequest_send(this); │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.abort = function() { │ │ │ │ │ + // Add method sniffer │ │ │ │ │ + if (cXMLHttpRequest.onabort) │ │ │ │ │ + cXMLHttpRequest.onabort.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: Gecko - unnecessary DONE when aborting │ │ │ │ │ + if (this.readyState > cXMLHttpRequest.UNSENT) │ │ │ │ │ + this._aborted = true; │ │ │ │ │ + │ │ │ │ │ + this._object.abort(); │ │ │ │ │ + │ │ │ │ │ + // BUGFIX: IE - memory leak │ │ │ │ │ + fCleanTransport(this); │ │ │ │ │ + │ │ │ │ │ + this.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ + │ │ │ │ │ + delete this._data; │ │ │ │ │ + /* if (this._async) │ │ │ │ │ + fQueue_remove(this);*/ │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.getAllResponseHeaders = function() { │ │ │ │ │ + return this._object.getAllResponseHeaders(); │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.getResponseHeader = function(sName) { │ │ │ │ │ + return this._object.getResponseHeader(sName); │ │ │ │ │ + }; │ │ │ │ │ + cXMLHttpRequest.prototype.setRequestHeader = function(sName, sValue) { │ │ │ │ │ + // BUGFIX: IE - cache issue │ │ │ │ │ + if (!this._headers) │ │ │ │ │ + this._headers = {}; │ │ │ │ │ + this._headers[sName] = sValue; │ │ │ │ │ + │ │ │ │ │ + return this._object.setRequestHeader(sName, sValue); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // EventTarget interface implementation │ │ │ │ │ + cXMLHttpRequest.prototype.addEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ + for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ + if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) │ │ │ │ │ + return; │ │ │ │ │ + // Add listener │ │ │ │ │ + this._listeners.push([sName, fHandler, bUseCapture]); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + cXMLHttpRequest.prototype.removeEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ + for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ + if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) │ │ │ │ │ + break; │ │ │ │ │ + // Remove listener │ │ │ │ │ + if (oListener) │ │ │ │ │ + this._listeners.splice(nIndex, 1); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + cXMLHttpRequest.prototype.dispatchEvent = function(oEvent) { │ │ │ │ │ + var oEventPseudo = { │ │ │ │ │ + 'type': oEvent.type, │ │ │ │ │ + 'target': this, │ │ │ │ │ + 'currentTarget': this, │ │ │ │ │ + 'eventPhase': 2, │ │ │ │ │ + 'bubbles': oEvent.bubbles, │ │ │ │ │ + 'cancelable': oEvent.cancelable, │ │ │ │ │ + 'timeStamp': oEvent.timeStamp, │ │ │ │ │ + 'stopPropagation': function() {}, // There is no flow │ │ │ │ │ + 'preventDefault': function() {}, // There is no default action │ │ │ │ │ + 'initEvent': function() {} // Original event object should be initialized │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // Execute onreadystatechange │ │ │ │ │ + if (oEventPseudo.type == "readystatechange" && this.onreadystatechange) │ │ │ │ │ + (this.onreadystatechange.handleEvent || this.onreadystatechange).apply(this, [oEventPseudo]); │ │ │ │ │ + │ │ │ │ │ + // Execute listeners │ │ │ │ │ + for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ + if (oListener[0] == oEventPseudo.type && !oListener[2]) │ │ │ │ │ + (oListener[1].handleEvent || oListener[1]).apply(this, [oEventPseudo]); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // │ │ │ │ │ + cXMLHttpRequest.prototype.toString = function() { │ │ │ │ │ + return '[' + "object" + ' ' + "XMLHttpRequest" + ']'; │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + cXMLHttpRequest.toString = function() { │ │ │ │ │ + return '[' + "XMLHttpRequest" + ']'; │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // Helper function │ │ │ │ │ + function fReadyStateChange(oRequest) { │ │ │ │ │ + // Sniffing code │ │ │ │ │ + if (cXMLHttpRequest.onreadystatechange) │ │ │ │ │ + cXMLHttpRequest.onreadystatechange.apply(oRequest); │ │ │ │ │ + │ │ │ │ │ + // Fake event │ │ │ │ │ + oRequest.dispatchEvent({ │ │ │ │ │ + 'type': "readystatechange", │ │ │ │ │ + 'bubbles': false, │ │ │ │ │ + 'cancelable': false, │ │ │ │ │ + 'timeStamp': new Date + 0 │ │ │ │ │ + }); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fGetDocument(oRequest) { │ │ │ │ │ + var oDocument = oRequest.responseXML, │ │ │ │ │ + sResponse = oRequest.responseText; │ │ │ │ │ + // Try parsing responseText │ │ │ │ │ + if (bIE && sResponse && oDocument && !oDocument.documentElement && oRequest.getResponseHeader("Content-Type").match(/[^\/]+\/[^\+]+\+xml/)) { │ │ │ │ │ + oDocument = new window.ActiveXObject("Microsoft.XMLDOM"); │ │ │ │ │ + oDocument.async = false; │ │ │ │ │ + oDocument.validateOnParse = false; │ │ │ │ │ + oDocument.loadXML(sResponse); │ │ │ │ │ + } │ │ │ │ │ + // Check if there is no error in document │ │ │ │ │ + if (oDocument) │ │ │ │ │ + if ((bIE && oDocument.parseError != 0) || !oDocument.documentElement || (oDocument.documentElement && oDocument.documentElement.tagName == "parsererror")) │ │ │ │ │ + return null; │ │ │ │ │ + return oDocument; │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fSynchronizeValues(oRequest) { │ │ │ │ │ + try { │ │ │ │ │ + oRequest.responseText = oRequest._object.responseText; │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + try { │ │ │ │ │ + oRequest.responseXML = fGetDocument(oRequest._object); │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + try { │ │ │ │ │ + oRequest.status = oRequest._object.status; │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + try { │ │ │ │ │ + oRequest.statusText = oRequest._object.statusText; │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fCleanTransport(oRequest) { │ │ │ │ │ + // BUGFIX: IE - memory leak (on-page leak) │ │ │ │ │ + oRequest._object.onreadystatechange = new window.Function; │ │ │ │ │ + }; │ │ │ │ │ + /* │ │ │ │ │ + // Queue manager │ │ │ │ │ + var oQueuePending = {"CRITICAL":[],"HIGH":[],"NORMAL":[],"LOW":[],"LOWEST":[]}, │ │ │ │ │ + aQueueRunning = []; │ │ │ │ │ + function fQueue_add(oRequest) { │ │ │ │ │ + oQueuePending[oRequest.priority in oQueuePending ? oRequest.priority : "NORMAL"].push(oRequest); │ │ │ │ │ + // │ │ │ │ │ + setTimeout(fQueue_process); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fQueue_remove(oRequest) { │ │ │ │ │ + for (var nIndex = 0, bFound = false; nIndex < aQueueRunning.length; nIndex++) │ │ │ │ │ + if (bFound) │ │ │ │ │ + aQueueRunning[nIndex - 1] = aQueueRunning[nIndex]; │ │ │ │ │ + else │ │ │ │ │ + if (aQueueRunning[nIndex] == oRequest) │ │ │ │ │ + bFound = true; │ │ │ │ │ + if (bFound) │ │ │ │ │ + aQueueRunning.length--; │ │ │ │ │ + // │ │ │ │ │ + setTimeout(fQueue_process); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + function fQueue_process() { │ │ │ │ │ + if (aQueueRunning.length < 6) { │ │ │ │ │ + for (var sPriority in oQueuePending) { │ │ │ │ │ + if (oQueuePending[sPriority].length) { │ │ │ │ │ + var oRequest = oQueuePending[sPriority][0]; │ │ │ │ │ + oQueuePending[sPriority] = oQueuePending[sPriority].slice(1); │ │ │ │ │ + // │ │ │ │ │ + aQueueRunning.push(oRequest); │ │ │ │ │ + // Send request │ │ │ │ │ + fXMLHttpRequest_send(oRequest); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + */ │ │ │ │ │ + // Internet Explorer 5.0 (missing apply) │ │ │ │ │ + if (!window.Function.prototype.apply) { │ │ │ │ │ + window.Function.prototype.apply = function(oRequest, oArguments) { │ │ │ │ │ + if (!oArguments) │ │ │ │ │ + oArguments = []; │ │ │ │ │ + oRequest.__func = this; │ │ │ │ │ + oRequest.__func(oArguments[0], oArguments[1], oArguments[2], oArguments[3], oArguments[4]); │ │ │ │ │ + delete oRequest.__func; │ │ │ │ │ + }; │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // Register new object with window │ │ │ │ │ + /** │ │ │ │ │ + * Class: OpenLayers.Request.XMLHttpRequest │ │ │ │ │ + * Standard-compliant (W3C) cross-browser implementation of the │ │ │ │ │ + * XMLHttpRequest object. From │ │ │ │ │ + * http://code.google.com/p/xmlhttprequest/. │ │ │ │ │ + */ │ │ │ │ │ + if (!OpenLayers.Request) { │ │ │ │ │ + /** │ │ │ │ │ + * This allows for OpenLayers/Request.js to be included │ │ │ │ │ + * before or after this script. │ │ │ │ │ + */ │ │ │ │ │ + OpenLayers.Request = {}; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Request.XMLHttpRequest = cXMLHttpRequest; │ │ │ │ │ +})(); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Protocol/HTTP.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Protocol.js │ │ │ │ │ + * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * if application uses the query string, for example, for BBOX parameters, │ │ │ │ │ + * OpenLayers/Format/QueryStringFilter.js should be included in the build config file │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.HTTP │ │ │ │ │ + * A basic HTTP protocol for vector layers. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Protocol.HTTP> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Protocol> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.HTTP = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: url │ │ │ │ │ + * {String} Service URL, read-only, set through the options │ │ │ │ │ + * passed to constructor. │ │ │ │ │ + */ │ │ │ │ │ + url: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: headers │ │ │ │ │ + * {Object} HTTP request headers, read-only, set through the options │ │ │ │ │ + * passed to the constructor, │ │ │ │ │ + * Example: {'Content-Type': 'plain/text'} │ │ │ │ │ + */ │ │ │ │ │ + headers: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: params │ │ │ │ │ + * {Object} Parameters of GET requests, read-only, set through the options │ │ │ │ │ + * passed to the constructor, │ │ │ │ │ + * Example: {'bbox': '5,5,5,5'} │ │ │ │ │ + */ │ │ │ │ │ + params: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: callback │ │ │ │ │ + * {Object} Function to be called when the <read>, <create>, │ │ │ │ │ + * <update>, <delete> or <commit> operation completes, read-only, │ │ │ │ │ + * set through the options passed to the constructor. │ │ │ │ │ + */ │ │ │ │ │ + callback: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: scope │ │ │ │ │ + * {Object} Callback execution scope, read-only, set through the │ │ │ │ │ + * options passed to the constructor. │ │ │ │ │ + */ │ │ │ │ │ + scope: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: readWithPOST │ │ │ │ │ + * {Boolean} true if read operations are done with POST requests │ │ │ │ │ + * instead of GET, defaults to false. │ │ │ │ │ + */ │ │ │ │ │ + readWithPOST: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: updateWithPOST │ │ │ │ │ + * {Boolean} true if update operations are done with POST requests │ │ │ │ │ + * defaults to false. │ │ │ │ │ + */ │ │ │ │ │ + updateWithPOST: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: deleteWithPOST │ │ │ │ │ + * {Boolean} true if delete operations are done with POST requests │ │ │ │ │ + * defaults to false. │ │ │ │ │ + * if true, POST data is set to output of format.write(). │ │ │ │ │ + */ │ │ │ │ │ + deleteWithPOST: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: wildcarded. │ │ │ │ │ + * {Boolean} If true percent signs are added around values │ │ │ │ │ + * read from LIKE filters, for example if the protocol │ │ │ │ │ + * read method is passed a LIKE filter whose property │ │ │ │ │ + * is "foo" and whose value is "bar" the string │ │ │ │ │ + * "foo__ilike=%bar%" will be sent in the query string; │ │ │ │ │ + * defaults to false. │ │ │ │ │ + */ │ │ │ │ │ + wildcarded: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: srsInBBOX │ │ │ │ │ + * {Boolean} Include the SRS identifier in BBOX query string parameter. │ │ │ │ │ + * Default is false. If true and the layer has a projection object set, │ │ │ │ │ + * any BBOX filter will be serialized with a fifth item identifying the │ │ │ │ │ + * projection. E.g. bbox=-1000,-1000,1000,1000,EPSG:900913 │ │ │ │ │ + */ │ │ │ │ │ + srsInBBOX: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Protocol.HTTP │ │ │ │ │ + * A class for giving layers generic HTTP protocol. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + * │ │ │ │ │ + * Valid options include: │ │ │ │ │ + * url - {String} │ │ │ │ │ + * headers - {Object} │ │ │ │ │ + * params - {Object} URL parameters for GET requests │ │ │ │ │ + * format - {<OpenLayers.Format>} │ │ │ │ │ + * callback - {Function} │ │ │ │ │ + * scope - {Object} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + this.params = {}; │ │ │ │ │ + this.headers = {}; │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ + var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ + wildcarded: this.wildcarded, │ │ │ │ │ + srsInBBOX: this.srsInBBOX │ │ │ │ │ + }); │ │ │ │ │ + this.filterToParams = function(filter, params) { │ │ │ │ │ + return format.write(filter, params); │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up the protocol. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.params = null; │ │ │ │ │ + this.headers = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: filterToParams │ │ │ │ │ + * Optional method to translate an <OpenLayers.Filter> object into an object │ │ │ │ │ + * that can be serialized as request query string provided. If a custom │ │ │ │ │ + * method is not provided, the filter will be serialized using the │ │ │ │ │ + * <OpenLayers.Format.QueryStringFilter> class. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {<OpenLayers.Filter>} filter to convert. │ │ │ │ │ + * params - {Object} The parameters object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The resulting parameters object. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Construct a request for reading new features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * url - {String} Url for the request. │ │ │ │ │ + * params - {Object} Parameters to get serialized as a query string. │ │ │ │ │ + * headers - {Object} Headers to be set on the request. │ │ │ │ │ + * filter - {<OpenLayers.Filter>} Filter to get serialized as a │ │ │ │ │ + * query string. │ │ │ │ │ + * readWithPOST - {Boolean} If the request should be done with POST. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} A response object, whose "priv" property │ │ │ │ │ + * references the HTTP request, this object is also passed to the │ │ │ │ │ + * callback function when the request completes, its "features" property │ │ │ │ │ + * is then populated with the features received from the server. │ │ │ │ │ + */ │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options.params, this.options.params); │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + if (options.filter && this.filterToParams) { │ │ │ │ │ + options.params = this.filterToParams( │ │ │ │ │ + options.filter, options.params │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + var readWithPOST = (options.readWithPOST !== undefined) ? │ │ │ │ │ + options.readWithPOST : this.readWithPOST; │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + if (readWithPOST) { │ │ │ │ │ + var headers = options.headers || {}; │ │ │ │ │ + headers["Content-Type"] = "application/x-www-form-urlencoded"; │ │ │ │ │ + resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ + data: OpenLayers.Util.getParameterString(options.params), │ │ │ │ │ + headers: headers │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + resp.priv = OpenLayers.Request.GET({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return resp; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleRead │ │ │ │ │ + * Individual callbacks are created for read, create and update, should │ │ │ │ │ + * a subclass need to override each one separately. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * the user callback. │ │ │ │ │ + * options - {Object} The user options passed to the read call. │ │ │ │ │ + */ │ │ │ │ │ + handleRead: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: create │ │ │ │ │ + * Construct a request for writing newly created features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, whose "priv" property references the HTTP request, this │ │ │ │ │ + * object is also passed to the callback function when the request │ │ │ │ │ + * completes, its "features" property is then populated with the │ │ │ │ │ + * the features received from the server. │ │ │ │ │ + */ │ │ │ │ │ + create: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: features, │ │ │ │ │ + requestType: "create" │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleCreate, resp, options), │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(features) │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + return resp; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleCreate │ │ │ │ │ + * Called the the request issued by <create> is complete. May be overridden │ │ │ │ │ + * by subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the create call. │ │ │ │ │ + */ │ │ │ │ │ + handleCreate: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: update │ │ │ │ │ + * Construct a request updating modified feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, whose "priv" property references the HTTP request, this │ │ │ │ │ + * object is also passed to the callback function when the request │ │ │ │ │ + * completes, its "features" property is then populated with the │ │ │ │ │ + * the feature received from the server. │ │ │ │ │ + */ │ │ │ │ │ + update: function(feature, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var url = options.url || │ │ │ │ │ + feature.url || │ │ │ │ │ + this.options.url + "/" + feature.fid; │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: feature, │ │ │ │ │ + requestType: "update" │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + var method = this.updateWithPOST ? "POST" : "PUT"; │ │ │ │ │ + resp.priv = OpenLayers.Request[method]({ │ │ │ │ │ + url: url, │ │ │ │ │ + callback: this.createCallback(this.handleUpdate, resp, options), │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(feature) │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + return resp; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleUpdate │ │ │ │ │ + * Called the the request issued by <update> is complete. May be overridden │ │ │ │ │ + * by subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the update call. │ │ │ │ │ + */ │ │ │ │ │ + handleUpdate: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: delete │ │ │ │ │ + * Construct a request deleting a removed feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, whose "priv" property references the HTTP request, this │ │ │ │ │ + * object is also passed to the callback function when the request │ │ │ │ │ + * completes. │ │ │ │ │ + */ │ │ │ │ │ + "delete": function(feature, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var url = options.url || │ │ │ │ │ + feature.url || │ │ │ │ │ + this.options.url + "/" + feature.fid; │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: feature, │ │ │ │ │ + requestType: "delete" │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + var method = this.deleteWithPOST ? "POST" : "DELETE"; │ │ │ │ │ + var requestOptions = { │ │ │ │ │ + url: url, │ │ │ │ │ + callback: this.createCallback(this.handleDelete, resp, options), │ │ │ │ │ + headers: options.headers │ │ │ │ │ + }; │ │ │ │ │ + if (this.deleteWithPOST) { │ │ │ │ │ + requestOptions.data = this.format.write(feature); │ │ │ │ │ + } │ │ │ │ │ + resp.priv = OpenLayers.Request[method](requestOptions); │ │ │ │ │ + │ │ │ │ │ + return resp; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleDelete │ │ │ │ │ + * Called the the request issued by <delete> is complete. May be overridden │ │ │ │ │ + * by subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the delete call. │ │ │ │ │ + */ │ │ │ │ │ + handleDelete: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleResponse │ │ │ │ │ + * Called by CRUD specific handlers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the create, read, update, │ │ │ │ │ + * or delete call. │ │ │ │ │ + */ │ │ │ │ │ + handleResponse: function(resp, options) { │ │ │ │ │ + var request = resp.priv; │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + // success │ │ │ │ │ + if (resp.requestType != "delete") { │ │ │ │ │ + resp.features = this.parseFeatures(request); │ │ │ │ │ + } │ │ │ │ │ + resp.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + } else { │ │ │ │ │ + // failure │ │ │ │ │ + resp.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + } │ │ │ │ │ + options.callback.call(options.scope, resp); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseFeatures │ │ │ │ │ + * Read HTTP response body and return features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * request - {XMLHttpRequest} The request object │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} Array of features or a single feature. │ │ │ │ │ + */ │ │ │ │ │ + parseFeatures: function(request) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ + } │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + return this.format.read(doc); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: commit │ │ │ │ │ + * Iterate over each feature and take action based on the feature state. │ │ │ │ │ + * Possible actions are create, update and delete. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array({<OpenLayers.Feature.Vector>})} │ │ │ │ │ + * options - {Object} Optional object for setting up intermediate commit │ │ │ │ │ + * callbacks. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * create - {Object} Optional object to be passed to the <create> method. │ │ │ │ │ + * update - {Object} Optional object to be passed to the <update> method. │ │ │ │ │ + * delete - {Object} Optional object to be passed to the <delete> method. │ │ │ │ │ + * callback - {Function} Optional function to be called when the commit │ │ │ │ │ + * is complete. │ │ │ │ │ + * scope - {Object} Optional object to be set as the scope of the callback. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Protocol.Response>)} An array of response objects, │ │ │ │ │ + * one per request made to the server, each object's "priv" property │ │ │ │ │ + * references the corresponding HTTP request. │ │ │ │ │ + */ │ │ │ │ │ + commit: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = [], │ │ │ │ │ + nResponses = 0; │ │ │ │ │ + │ │ │ │ │ + // Divide up features before issuing any requests. This properly │ │ │ │ │ + // counts requests in the event that any responses come in before │ │ │ │ │ + // all requests have been issued. │ │ │ │ │ + var types = {}; │ │ │ │ │ + types[OpenLayers.State.INSERT] = []; │ │ │ │ │ + types[OpenLayers.State.UPDATE] = []; │ │ │ │ │ + types[OpenLayers.State.DELETE] = []; │ │ │ │ │ + var feature, list, requestFeatures = []; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + list = types[feature.state]; │ │ │ │ │ + if (list) { │ │ │ │ │ + list.push(feature); │ │ │ │ │ + requestFeatures.push(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // tally up number of requests │ │ │ │ │ + var nRequests = (types[OpenLayers.State.INSERT].length > 0 ? 1 : 0) + │ │ │ │ │ + types[OpenLayers.State.UPDATE].length + │ │ │ │ │ + types[OpenLayers.State.DELETE].length; │ │ │ │ │ + │ │ │ │ │ + // This response will be sent to the final callback after all the others │ │ │ │ │ + // have been fired. │ │ │ │ │ + var success = true; │ │ │ │ │ + var finalResponse = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: requestFeatures │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + function insertCallback(response) { │ │ │ │ │ + var len = response.features ? response.features.length : 0; │ │ │ │ │ + var fids = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + fids[i] = response.features[i].fid; │ │ │ │ │ + } │ │ │ │ │ + finalResponse.insertIds = fids; │ │ │ │ │ + callback.apply(this, [response]); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function callback(response) { │ │ │ │ │ + this.callUserCallback(response, options); │ │ │ │ │ + success = success && response.success(); │ │ │ │ │ + nResponses++; │ │ │ │ │ + if (nResponses >= nRequests) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + finalResponse.code = success ? │ │ │ │ │ + OpenLayers.Protocol.Response.SUCCESS : │ │ │ │ │ + OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + options.callback.apply(options.scope, [finalResponse]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // start issuing requests │ │ │ │ │ + var queue = types[OpenLayers.State.INSERT]; │ │ │ │ │ + if (queue.length > 0) { │ │ │ │ │ + resp.push(this.create( │ │ │ │ │ + queue, OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: insertCallback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options.create) │ │ │ │ │ + )); │ │ │ │ │ + } │ │ │ │ │ + queue = types[OpenLayers.State.UPDATE]; │ │ │ │ │ + for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ + resp.push(this.update( │ │ │ │ │ + queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: callback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options.update))); │ │ │ │ │ + } │ │ │ │ │ + queue = types[OpenLayers.State.DELETE]; │ │ │ │ │ + for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ + resp.push(this["delete"]( │ │ │ │ │ + queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: callback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options["delete"]))); │ │ │ │ │ + } │ │ │ │ │ + return resp; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: abort │ │ │ │ │ + * Abort an ongoing request, the response object passed to │ │ │ │ │ + * this method must come from this HTTP protocol (as a result │ │ │ │ │ + * of a create, read, update, delete or commit operation). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ + */ │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + response.priv.abort(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: callUserCallback │ │ │ │ │ + * This method is used from within the commit method each time an │ │ │ │ │ + * an HTTP response is received from the server, it is responsible │ │ │ │ │ + * for calling the user-supplied callbacks. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} │ │ │ │ │ + * options - {Object} The map of options passed to the commit call. │ │ │ │ │ + */ │ │ │ │ │ + callUserCallback: function(resp, options) { │ │ │ │ │ + var opt = options[resp.requestType]; │ │ │ │ │ + if (opt && opt.callback) { │ │ │ │ │ + opt.callback.call(opt.scope, resp); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.HTTP" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Protocol/WFS/v1.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Protocol/WFS.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.WFS.v1 │ │ │ │ │ + * Abstract class for for v1.0.0 and v1.1.0 protocol. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Protocol> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.WFS.v1 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} WFS version number. │ │ │ │ │ + */ │ │ │ │ │ + version: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: srsName │ │ │ │ │ + * {String} Name of spatial reference system. Default is "EPSG:4326". │ │ │ │ │ + */ │ │ │ │ │ + srsName: "EPSG:4326", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: featureType │ │ │ │ │ + * {String} Local feature typeName. │ │ │ │ │ + */ │ │ │ │ │ + featureType: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: featureNS │ │ │ │ │ + * {String} Feature namespace. │ │ │ │ │ + */ │ │ │ │ │ + featureNS: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: geometryName │ │ │ │ │ + * {String} Name of the geometry attribute for features. Default is │ │ │ │ │ + * "the_geom" for WFS <version> 1.0, and null for higher versions. │ │ │ │ │ + */ │ │ │ │ │ + geometryName: "the_geom", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: maxFeatures │ │ │ │ │ + * {Integer} Optional maximum number of features to retrieve. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: schema │ │ │ │ │ + * {String} Optional schema location that will be included in the │ │ │ │ │ + * schemaLocation attribute value. Note that the feature type schema │ │ │ │ │ + * is required for a strict XML validator (on transactions with an │ │ │ │ │ + * insert for example), but is *not* required by the WFS specification │ │ │ │ │ + * (since the server is supposed to know about feature type schemas). │ │ │ │ │ + */ │ │ │ │ │ + schema: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: featurePrefix │ │ │ │ │ + * {String} Namespace alias for feature type. Default is "feature". │ │ │ │ │ + */ │ │ │ │ │ + featurePrefix: "feature", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: formatOptions │ │ │ │ │ + * {Object} Optional options for the format. If a format is not provided, │ │ │ │ │ + * this property can be used to extend the default format options. │ │ │ │ │ + */ │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readFormat │ │ │ │ │ + * {<OpenLayers.Format>} For WFS requests it is possible to get a │ │ │ │ │ + * different output format than GML. In that case, we cannot parse │ │ │ │ │ + * the response with the default format (WFST) and we need a different │ │ │ │ │ + * format for reading. │ │ │ │ │ + */ │ │ │ │ │ + readFormat: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readOptions │ │ │ │ │ + * {Object} Optional object to pass to format's read. │ │ │ │ │ + */ │ │ │ │ │ + readOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Protocol.WFS │ │ │ │ │ + * A class for giving layers WFS protocol. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + * │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * url - {String} URL to send requests to (required). │ │ │ │ │ + * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ + * featureNS - {String} Feature namespace (required, but can be autodetected │ │ │ │ │ + * during the first query if GML is used as readFormat and │ │ │ │ │ + * featurePrefix is provided and matches the prefix used by the server │ │ │ │ │ + * for this featureType). │ │ │ │ │ + * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ + * for writing if featureNS is provided). Default is 'feature'. │ │ │ │ │ + * geometryName - {String} Name of geometry attribute. The default is │ │ │ │ │ + * 'the_geom' for WFS <version> 1.0, and null for higher versions. If │ │ │ │ │ + * null, it will be set to the name of the first geometry found in the │ │ │ │ │ + * first read operation. │ │ │ │ │ + * multi - {Boolean} If set to true, geometries will be casted to Multi │ │ │ │ │ + * geometries before they are written in a transaction. No casting will │ │ │ │ │ + * be done when reading features. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!options.format) { │ │ │ │ │ + this.format = OpenLayers.Format.WFST(OpenLayers.Util.extend({ │ │ │ │ │ + version: this.version, │ │ │ │ │ + featureType: this.featureType, │ │ │ │ │ + featureNS: this.featureNS, │ │ │ │ │ + featurePrefix: this.featurePrefix, │ │ │ │ │ + geometryName: this.geometryName, │ │ │ │ │ + srsName: this.srsName, │ │ │ │ │ + schema: this.schema │ │ │ │ │ + }, this.formatOptions)); │ │ │ │ │ + } │ │ │ │ │ + if (!options.geometryName && parseFloat(this.format.version) > 1.0) { │ │ │ │ │ + this.setGeometryName(null); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up the protocol. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.options && !this.options.format) { │ │ │ │ │ + this.format.destroy(); │ │ │ │ │ + } │ │ │ │ │ + this.format = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Construct a request for reading new features. Since WFS splits the │ │ │ │ │ + * basic CRUD operations into GetFeature requests (for read) and │ │ │ │ │ + * Transactions (for all others), this method does not make use of the │ │ │ │ │ + * format's read method (that is only about reading transaction │ │ │ │ │ + * responses). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Options for the read operation, in addition to the │ │ │ │ │ + * options set on the instance (options set here will take precedence). │ │ │ │ │ + * │ │ │ │ │ + * To use a configured protocol to get e.g. a WFS hit count, applications │ │ │ │ │ + * could do the following: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * protocol.read({ │ │ │ │ │ + * readOptions: {output: "object"}, │ │ │ │ │ + * resultType: "hits", │ │ │ │ │ + * maxFeatures: null, │ │ │ │ │ + * callback: function(resp) { │ │ │ │ │ + * // process resp.numberOfFeatures here │ │ │ │ │ + * } │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * To use a configured protocol to use WFS paging (if supported by the │ │ │ │ │ + * server), applications could do the following: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * protocol.read({ │ │ │ │ │ + * startIndex: 0, │ │ │ │ │ + * count: 50 │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * To limit the attributes returned by the GetFeature request, applications │ │ │ │ │ + * can use the propertyNames option to specify the properties to include in │ │ │ │ │ + * the response: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * protocol.read({ │ │ │ │ │ + * propertyNames: ["DURATION", "INTENSITY"] │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + var data = OpenLayers.Format.XML.prototype.write.apply( │ │ │ │ │ + this.format, [this.format.writeNode("wfs:GetFeature", options)] │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: data │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + return response; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setFeatureType │ │ │ │ │ + * Change the feature type on the fly. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureType - {String} Local (without prefix) feature typeName. │ │ │ │ │ + */ │ │ │ │ │ + setFeatureType: function(featureType) { │ │ │ │ │ + this.featureType = featureType; │ │ │ │ │ + this.format.featureType = featureType; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setGeometryName │ │ │ │ │ + * Sets the geometryName option after instantiation. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometryName - {String} Name of geometry attribute. │ │ │ │ │ + */ │ │ │ │ │ + setGeometryName: function(geometryName) { │ │ │ │ │ + this.geometryName = geometryName; │ │ │ │ │ + this.format.geometryName = geometryName; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleRead │ │ │ │ │ + * Deal with response from the read request. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ + * to the user callback. │ │ │ │ │ + * options - {Object} The user options passed to the read call. │ │ │ │ │ + */ │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + // success │ │ │ │ │ + var result = this.parseResponse(request, options.readOptions); │ │ │ │ │ + if (result && result.success !== false) { │ │ │ │ │ + if (options.readOptions && options.readOptions.output == "object") { │ │ │ │ │ + OpenLayers.Util.extend(response, result); │ │ │ │ │ + } else { │ │ │ │ │ + response.features = result; │ │ │ │ │ + } │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + } else { │ │ │ │ │ + // failure (service exception) │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + response.error = result; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // failure │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + } │ │ │ │ │ + options.callback.call(options.scope, response); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseResponse │ │ │ │ │ + * Read HTTP response body and return features │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * request - {XMLHttpRequest} The request object │ │ │ │ │ + * options - {Object} Optional object to pass to format's read │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} or {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * An object with a features property, an array of features or a single │ │ │ │ │ + * feature. │ │ │ │ │ + */ │ │ │ │ │ + parseResponse: function(request, options) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ + } │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + var result = (this.readFormat !== null) ? this.readFormat.read(doc) : │ │ │ │ │ + this.format.read(doc, options); │ │ │ │ │ + if (!this.featureNS) { │ │ │ │ │ + var format = this.readFormat || this.format; │ │ │ │ │ + this.featureNS = format.featureNS; │ │ │ │ │ + // no need to auto-configure again on subsequent reads │ │ │ │ │ + format.autoConfig = false; │ │ │ │ │ + if (!this.geometryName) { │ │ │ │ │ + this.setGeometryName(format.geometryName); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return result; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: commit │ │ │ │ │ + * Given a list of feature, assemble a batch request for update, create, │ │ │ │ │ + * and delete transactions. A commit call on the prototype amounts │ │ │ │ │ + * to writing a WFS transaction - so the write method on the format │ │ │ │ │ + * is used. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * options - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * nativeElements - {Array({Object})} Array of objects with information for writing │ │ │ │ │ + * out <Native> elements, these objects have vendorId, safeToIgnore and │ │ │ │ │ + * value properties. The <Native> element is intended to allow access to │ │ │ │ │ + * vendor specific capabilities of any particular web feature server or │ │ │ │ │ + * datastore. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} A response object with a features │ │ │ │ │ + * property containing any insertIds and a priv property referencing │ │ │ │ │ + * the XMLHttpRequest object. │ │ │ │ │ + */ │ │ │ │ │ + commit: function(features, options) { │ │ │ │ │ + │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "commit", │ │ │ │ │ + reqFeatures: features │ │ │ │ │ + }); │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(features, options), │ │ │ │ │ + callback: this.createCallback(this.handleCommit, response, options) │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + return response; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleCommit │ │ │ │ │ + * Called when the commit request returns. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ + * to the user callback. │ │ │ │ │ + * options - {Object} The user options passed to the commit call. │ │ │ │ │ + */ │ │ │ │ │ + handleCommit: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + │ │ │ │ │ + // ensure that we have an xml doc │ │ │ │ │ + var data = request.responseXML; │ │ │ │ │ + if (!data || !data.documentElement) { │ │ │ │ │ + data = request.responseText; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var obj = this.format.read(data) || {}; │ │ │ │ │ + │ │ │ │ │ + response.insertIds = obj.insertIds || []; │ │ │ │ │ + if (obj.success) { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + response.error = obj; │ │ │ │ │ + } │ │ │ │ │ + options.callback.call(options.scope, response); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: filterDelete │ │ │ │ │ + * Send a request that deletes all features by their filter. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {<OpenLayers.Filter>} filter │ │ │ │ │ + */ │ │ │ │ │ + filterDelete: function(filter, options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "commit" │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + var root = this.format.createElementNSPlus("wfs:Transaction", { │ │ │ │ │ + attributes: { │ │ │ │ │ + service: "WFS", │ │ │ │ │ + version: this.version │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + var deleteNode = this.format.createElementNSPlus("wfs:Delete", { │ │ │ │ │ + attributes: { │ │ │ │ │ + typeName: (options.featureNS ? this.featurePrefix + ":" : "") + │ │ │ │ │ + options.featureType │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + if (options.featureNS) { │ │ │ │ │ + deleteNode.setAttribute("xmlns:" + this.featurePrefix, options.featureNS); │ │ │ │ │ + } │ │ │ │ │ + var filterNode = this.format.writeNode("ogc:Filter", filter); │ │ │ │ │ + │ │ │ │ │ + deleteNode.appendChild(filterNode); │ │ │ │ │ + │ │ │ │ │ + root.appendChild(deleteNode); │ │ │ │ │ + │ │ │ │ │ + var data = OpenLayers.Format.XML.prototype.write.apply( │ │ │ │ │ + this.format, [root] │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Request.POST({ │ │ │ │ │ + url: this.url, │ │ │ │ │ + callback: options.callback || function() {}, │ │ │ │ │ + data: data │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: abort │ │ │ │ │ + * Abort an ongoing request, the response object passed to │ │ │ │ │ + * this method must come from this protocol (as a result │ │ │ │ │ + * of a read, or commit operation). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ + */ │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + response.priv.abort(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.WFS.v1" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format │ │ │ │ │ + * Base class for format reading/writing a variety of formats. Subclasses │ │ │ │ │ + * of OpenLayers.Format are expected to have read and write methods. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: options │ │ │ │ │ + * {Object} A reference to options passed to the constructor. │ │ │ │ │ + */ │ │ │ │ │ + options: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: externalProjection │ │ │ │ │ + * {<OpenLayers.Projection>} When passed a externalProjection and │ │ │ │ │ + * internalProjection, the format will reproject the geometries it │ │ │ │ │ + * reads or writes. The externalProjection is the projection used by │ │ │ │ │ + * the content which is passed into read or which comes out of write. │ │ │ │ │ + * In order to reproject, a projection transformation function for the │ │ │ │ │ + * specified projections must be available. This support may be │ │ │ │ │ + * provided via proj4js or via a custom transformation function. See │ │ │ │ │ + * {<OpenLayers.Projection.addTransform>} for more information on │ │ │ │ │ + * custom transformations. │ │ │ │ │ + */ │ │ │ │ │ + externalProjection: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: internalProjection │ │ │ │ │ + * {<OpenLayers.Projection>} When passed a externalProjection and │ │ │ │ │ + * internalProjection, the format will reproject the geometries it │ │ │ │ │ + * reads or writes. The internalProjection is the projection used by │ │ │ │ │ + * the geometries which are returned by read or which are passed into │ │ │ │ │ + * write. In order to reproject, a projection transformation function │ │ │ │ │ + * for the specified projections must be available. This support may be │ │ │ │ │ + * provided via proj4js or via a custom transformation function. See │ │ │ │ │ + * {<OpenLayers.Projection.addTransform>} for more information on │ │ │ │ │ + * custom transformations. │ │ │ │ │ + */ │ │ │ │ │ + internalProjection: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: data │ │ │ │ │ + * {Object} When <keepData> is true, this is the parsed string sent to │ │ │ │ │ + * <read>. │ │ │ │ │ + */ │ │ │ │ │ + data: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: keepData │ │ │ │ │ + * {Object} Maintain a reference (<data>) to the most recently read data. │ │ │ │ │ + * Default is false. │ │ │ │ │ + */ │ │ │ │ │ + keepData: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format │ │ │ │ │ + * Instances of this class are not useful. See one of the subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object with properties to set on the │ │ │ │ │ + * format │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * keepData - {Boolean} If true, upon <read>, the data property will be │ │ │ │ │ + * set to the parsed object (e.g. the json or xml object). │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * An instance of OpenLayers.Format │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.options = options; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: read │ │ │ │ │ + * Read data from a string, and return an object whose type depends on the │ │ │ │ │ + * subclass. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {string} Data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * Depends on the subclass │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + throw new Error('Read not implemented.'); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: write │ │ │ │ │ + * Accept an object, and return a string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * object - {Object} Object to be serialized │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string representation of the object. │ │ │ │ │ + */ │ │ │ │ │ + write: function(object) { │ │ │ │ │ + throw new Error('Write not implemented.'); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/XML.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.XML │ │ │ │ │ + * Read and write XML. For cross-browser XML generation, use methods on an │ │ │ │ │ + * instance of the XML format class instead of on <code>document<end>. │ │ │ │ │ + * The DOM creation and traversing methods exposed here all mimic the │ │ │ │ │ + * W3C XML DOM methods. Create a new parser with the │ │ │ │ │ + * <OpenLayers.Format.XML> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.XML = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. Properties │ │ │ │ │ + * of this object should not be set individually. Read-only. All │ │ │ │ │ + * XML subclasses should have their own namespaces object. Use │ │ │ │ │ + * <setNamespace> to add or set a namespace alias after construction. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaceAlias │ │ │ │ │ + * {Object} Mapping of namespace URI to namespace alias. This object │ │ │ │ │ + * is read-only. Use <setNamespace> to add or set a namespace alias. │ │ │ │ │ + */ │ │ │ │ │ + namespaceAlias: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + * {String} The default namespace alias for creating element nodes. │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the <readers> property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ + */ │ │ │ │ │ + writers: {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: xmldom │ │ │ │ │ + * {XMLDom} If this browser uses ActiveX, this will be set to a XMLDOM │ │ │ │ │ + * object. It is not intended to be a browser sniffing property. │ │ │ │ │ + * Instead, the xmldom property is used instead of <code>document<end> │ │ │ │ │ + * where namespaced node creation methods are not supported. In all │ │ │ │ │ + * other browsers, this remains null. │ │ │ │ │ + */ │ │ │ │ │ + xmldom: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.XML │ │ │ │ │ + * Construct an XML parser. The parser is used to read and write XML. │ │ │ │ │ + * Reading XML from a string returns a DOM element. Writing XML from │ │ │ │ │ + * a DOM element returns a string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on │ │ │ │ │ + * the object. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + if (window.ActiveXObject) { │ │ │ │ │ + this.xmldom = new ActiveXObject("Microsoft.XMLDOM"); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Format.prototype.initialize.apply(this, [options]); │ │ │ │ │ + // clone the namespace object and set all namespace aliases │ │ │ │ │ + this.namespaces = OpenLayers.Util.extend({}, this.namespaces); │ │ │ │ │ + this.namespaceAlias = {}; │ │ │ │ │ + for (var alias in this.namespaces) { │ │ │ │ │ + this.namespaceAlias[this.namespaces[alias]] = alias; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.xmldom = null; │ │ │ │ │ + OpenLayers.Format.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setNamespace │ │ │ │ │ + * Set a namespace alias and URI for the format. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * alias - {String} The namespace alias (prefix). │ │ │ │ │ + * uri - {String} The namespace URI. │ │ │ │ │ + */ │ │ │ │ │ + setNamespace: function(alias, uri) { │ │ │ │ │ + this.namespaces[alias] = uri; │ │ │ │ │ + this.namespaceAlias[uri] = alias; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Deserialize a XML string and return a DOM node. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * text - {String} A XML string │ │ │ │ │ + │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A DOM node │ │ │ │ │ + */ │ │ │ │ │ + read: function(text) { │ │ │ │ │ + var index = text.indexOf('<'); │ │ │ │ │ + if (index > 0) { │ │ │ │ │ + text = text.substring(index); │ │ │ │ │ + } │ │ │ │ │ + var node = OpenLayers.Util.Try( │ │ │ │ │ + OpenLayers.Function.bind(( │ │ │ │ │ + function() { │ │ │ │ │ + var xmldom; │ │ │ │ │ + /** │ │ │ │ │ + * Since we want to be able to call this method on the prototype │ │ │ │ │ + * itself, this.xmldom may not exist even if in IE. │ │ │ │ │ + */ │ │ │ │ │ + if (window.ActiveXObject && !this.xmldom) { │ │ │ │ │ + xmldom = new ActiveXObject("Microsoft.XMLDOM"); │ │ │ │ │ + } else { │ │ │ │ │ + xmldom = this.xmldom; │ │ │ │ │ + │ │ │ │ │ + } │ │ │ │ │ + xmldom.loadXML(text); │ │ │ │ │ + return xmldom; │ │ │ │ │ + } │ │ │ │ │ + ), this), │ │ │ │ │ + function() { │ │ │ │ │ + return new DOMParser().parseFromString(text, 'text/xml'); │ │ │ │ │ + }, │ │ │ │ │ + function() { │ │ │ │ │ + var req = new XMLHttpRequest(); │ │ │ │ │ + req.open("GET", "data:" + "text/xml" + │ │ │ │ │ + ";charset=utf-8," + encodeURIComponent(text), false); │ │ │ │ │ + if (req.overrideMimeType) { │ │ │ │ │ + req.overrideMimeType("text/xml"); │ │ │ │ │ + } │ │ │ │ │ + req.send(null); │ │ │ │ │ + return req.responseXML; │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + if (this.keepData) { │ │ │ │ │ + this.data = node; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Serialize a DOM node into a XML string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A DOM node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The XML string representation of the input node. │ │ │ │ │ + */ │ │ │ │ │ + write: function(node) { │ │ │ │ │ + var data; │ │ │ │ │ + if (this.xmldom) { │ │ │ │ │ + data = node.xml; │ │ │ │ │ + } else { │ │ │ │ │ + var serializer = new XMLSerializer(); │ │ │ │ │ + if (node.nodeType == 1) { │ │ │ │ │ + // Add nodes to a document before serializing. Everything else │ │ │ │ │ + // is serialized as is. This may need more work. See #1218 . │ │ │ │ │ + var doc = document.implementation.createDocument("", "", null); │ │ │ │ │ + if (doc.importNode) { │ │ │ │ │ + node = doc.importNode(node, true); │ │ │ │ │ + } │ │ │ │ │ + doc.appendChild(node); │ │ │ │ │ + data = serializer.serializeToString(doc); │ │ │ │ │ + } else { │ │ │ │ │ + data = serializer.serializeToString(node); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return data; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: createElementNS │ │ │ │ │ + * Create a new element with namespace. This node can be appended to │ │ │ │ │ + * another node with the standard node.appendChild method. For │ │ │ │ │ + * cross-browser support, this method must be used instead of │ │ │ │ │ + * document.createElementNS. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * uri - {String} Namespace URI for the element. │ │ │ │ │ + * name - {String} The qualified name of the element (prefix:localname). │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Element} A DOM element with namespace. │ │ │ │ │ + */ │ │ │ │ │ + createElementNS: function(uri, name) { │ │ │ │ │ + var element; │ │ │ │ │ + if (this.xmldom) { │ │ │ │ │ + if (typeof uri == "string") { │ │ │ │ │ + element = this.xmldom.createNode(1, name, uri); │ │ │ │ │ + } else { │ │ │ │ │ + element = this.xmldom.createNode(1, name, ""); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + element = document.createElementNS(uri, name); │ │ │ │ │ + } │ │ │ │ │ + return element; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: createDocumentFragment │ │ │ │ │ + * Create a document fragment node that can be appended to another node │ │ │ │ │ + * created by createElementNS. This will call │ │ │ │ │ + * document.createDocumentFragment outside of IE. In IE, the ActiveX │ │ │ │ │ + * object's createDocumentFragment method is used. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Element} A document fragment. │ │ │ │ │ + */ │ │ │ │ │ + createDocumentFragment: function() { │ │ │ │ │ + var element; │ │ │ │ │ + if (this.xmldom) { │ │ │ │ │ + element = this.xmldom.createDocumentFragment(); │ │ │ │ │ + } else { │ │ │ │ │ + element = document.createDocumentFragment(); │ │ │ │ │ + } │ │ │ │ │ + return element; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: createTextNode │ │ │ │ │ + * Create a text node. This node can be appended to another node with │ │ │ │ │ + * the standard node.appendChild method. For cross-browser support, │ │ │ │ │ + * this method must be used instead of document.createTextNode. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * text - {String} The text of the node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A DOM text node. │ │ │ │ │ + */ │ │ │ │ │ + createTextNode: function(text) { │ │ │ │ │ + var node; │ │ │ │ │ + if (typeof text !== "string") { │ │ │ │ │ + text = String(text); │ │ │ │ │ + } │ │ │ │ │ + if (this.xmldom) { │ │ │ │ │ + node = this.xmldom.createTextNode(text); │ │ │ │ │ + } else { │ │ │ │ │ + node = document.createTextNode(text); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getElementsByTagNameNS │ │ │ │ │ + * Get a list of elements on a node given the namespace URI and local name. │ │ │ │ │ + * To return all nodes in a given namespace, use '*' for the name │ │ │ │ │ + * argument. To return all nodes of a given (local) name, regardless │ │ │ │ │ + * of namespace, use '*' for the uri argument. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {Element} Node on which to search for other nodes. │ │ │ │ │ + * uri - {String} Namespace URI. │ │ │ │ │ + * name - {String} Local name of the tag (without the prefix). │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {NodeList} A node list or array of elements. │ │ │ │ │ + */ │ │ │ │ │ + getElementsByTagNameNS: function(node, uri, name) { │ │ │ │ │ + var elements = []; │ │ │ │ │ + if (node.getElementsByTagNameNS) { │ │ │ │ │ + elements = node.getElementsByTagNameNS(uri, name); │ │ │ │ │ + } else { │ │ │ │ │ + // brute force method │ │ │ │ │ + var allNodes = node.getElementsByTagName("*"); │ │ │ │ │ + var potentialNode, fullName; │ │ │ │ │ + for (var i = 0, len = allNodes.length; i < len; ++i) { │ │ │ │ │ + potentialNode = allNodes[i]; │ │ │ │ │ + fullName = (potentialNode.prefix) ? │ │ │ │ │ + (potentialNode.prefix + ":" + name) : name; │ │ │ │ │ + if ((name == "*") || (fullName == potentialNode.nodeName)) { │ │ │ │ │ + if ((uri == "*") || (uri == potentialNode.namespaceURI)) { │ │ │ │ │ + elements.push(potentialNode); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return elements; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getAttributeNodeNS │ │ │ │ │ + * Get an attribute node given the namespace URI and local name. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {Element} Node on which to search for attribute nodes. │ │ │ │ │ + * uri - {String} Namespace URI. │ │ │ │ │ + * name - {String} Local name of the attribute (without the prefix). │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} An attribute node or null if none found. │ │ │ │ │ + */ │ │ │ │ │ + getAttributeNodeNS: function(node, uri, name) { │ │ │ │ │ + var attributeNode = null; │ │ │ │ │ + if (node.getAttributeNodeNS) { │ │ │ │ │ + attributeNode = node.getAttributeNodeNS(uri, name); │ │ │ │ │ + } else { │ │ │ │ │ + var attributes = node.attributes; │ │ │ │ │ + var potentialNode, fullName; │ │ │ │ │ + for (var i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ + potentialNode = attributes[i]; │ │ │ │ │ + if (potentialNode.namespaceURI == uri) { │ │ │ │ │ + fullName = (potentialNode.prefix) ? │ │ │ │ │ + (potentialNode.prefix + ":" + name) : name; │ │ │ │ │ + if (fullName == potentialNode.nodeName) { │ │ │ │ │ + attributeNode = potentialNode; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return attributeNode; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getAttributeNS │ │ │ │ │ + * Get an attribute value given the namespace URI and local name. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {Element} Node on which to search for an attribute. │ │ │ │ │ + * uri - {String} Namespace URI. │ │ │ │ │ + * name - {String} Local name of the attribute (without the prefix). │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} An attribute value or and empty string if none found. │ │ │ │ │ + */ │ │ │ │ │ + getAttributeNS: function(node, uri, name) { │ │ │ │ │ + var attributeValue = ""; │ │ │ │ │ + if (node.getAttributeNS) { │ │ │ │ │ + attributeValue = node.getAttributeNS(uri, name) || ""; │ │ │ │ │ + } else { │ │ │ │ │ + var attributeNode = this.getAttributeNodeNS(node, uri, name); │ │ │ │ │ + if (attributeNode) { │ │ │ │ │ + attributeValue = attributeNode.nodeValue; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return attributeValue; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getChildValue │ │ │ │ │ + * Get the textual value of the node if it exists, or return an │ │ │ │ │ + * optional default string. Returns an empty string if no first child │ │ │ │ │ + * exists and no default value is supplied. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The element used to look for a first child value. │ │ │ │ │ + * def - {String} Optional string to return in the event that no │ │ │ │ │ + * first child value exists. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The value of the first child of the given node. │ │ │ │ │ + */ │ │ │ │ │ + getChildValue: function(node, def) { │ │ │ │ │ + var value = def || ""; │ │ │ │ │ + if (node) { │ │ │ │ │ + for (var child = node.firstChild; child; child = child.nextSibling) { │ │ │ │ │ + switch (child.nodeType) { │ │ │ │ │ + case 3: // text node │ │ │ │ │ + case 4: // cdata section │ │ │ │ │ + value += child.nodeValue; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return value; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: isSimpleContent │ │ │ │ │ + * Test if the given node has only simple content (i.e. no child element │ │ │ │ │ + * nodes). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} An element node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The node has no child element nodes (nodes of type 1). │ │ │ │ │ + */ │ │ │ │ │ + isSimpleContent: function(node) { │ │ │ │ │ + var simple = true; │ │ │ │ │ + for (var child = node.firstChild; child; child = child.nextSibling) { │ │ │ │ │ + if (child.nodeType === 1) { │ │ │ │ │ + simple = false; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return simple; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: contentType │ │ │ │ │ + * Determine the content type for a given node. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} One of OpenLayers.Format.XML.CONTENT_TYPE.{EMPTY,SIMPLE,COMPLEX,MIXED} │ │ │ │ │ + * if the node has no, simple, complex, or mixed content. │ │ │ │ │ + */ │ │ │ │ │ + contentType: function(node) { │ │ │ │ │ + var simple = false, │ │ │ │ │ + complex = false; │ │ │ │ │ + │ │ │ │ │ + var type = OpenLayers.Format.XML.CONTENT_TYPE.EMPTY; │ │ │ │ │ + │ │ │ │ │ + for (var child = node.firstChild; child; child = child.nextSibling) { │ │ │ │ │ + switch (child.nodeType) { │ │ │ │ │ + case 1: // element │ │ │ │ │ + complex = true; │ │ │ │ │ + break; │ │ │ │ │ + case 8: // comment │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + simple = true; │ │ │ │ │ + } │ │ │ │ │ + if (complex && simple) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (complex && simple) { │ │ │ │ │ + type = OpenLayers.Format.XML.CONTENT_TYPE.MIXED; │ │ │ │ │ + } else if (complex) { │ │ │ │ │ + return OpenLayers.Format.XML.CONTENT_TYPE.COMPLEX; │ │ │ │ │ + } else if (simple) { │ │ │ │ │ + return OpenLayers.Format.XML.CONTENT_TYPE.SIMPLE; │ │ │ │ │ + } │ │ │ │ │ + return type; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: hasAttributeNS │ │ │ │ │ + * Determine whether a node has a particular attribute matching the given │ │ │ │ │ + * name and namespace. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {Element} Node on which to search for an attribute. │ │ │ │ │ + * uri - {String} Namespace URI. │ │ │ │ │ + * name - {String} Local name of the attribute (without the prefix). │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The node has an attribute matching the name and namespace. │ │ │ │ │ + */ │ │ │ │ │ + hasAttributeNS: function(node, uri, name) { │ │ │ │ │ + var found = false; │ │ │ │ │ + if (node.hasAttributeNS) { │ │ │ │ │ + found = node.hasAttributeNS(uri, name); │ │ │ │ │ + } else { │ │ │ │ │ + found = !!this.getAttributeNodeNS(node, uri, name); │ │ │ │ │ + } │ │ │ │ │ + return found; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setAttributeNS │ │ │ │ │ + * Adds a new attribute or changes the value of an attribute with the given │ │ │ │ │ + * namespace and name. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {Element} Element node on which to set the attribute. │ │ │ │ │ + * uri - {String} Namespace URI for the attribute. │ │ │ │ │ + * name - {String} Qualified name (prefix:localname) for the attribute. │ │ │ │ │ + * value - {String} Attribute value. │ │ │ │ │ + */ │ │ │ │ │ + setAttributeNS: function(node, uri, name, value) { │ │ │ │ │ + if (node.setAttributeNS) { │ │ │ │ │ + node.setAttributeNS(uri, name, value); │ │ │ │ │ + } else { │ │ │ │ │ + if (this.xmldom) { │ │ │ │ │ + if (uri) { │ │ │ │ │ + var attribute = node.ownerDocument.createNode( │ │ │ │ │ + 2, name, uri │ │ │ │ │ + ); │ │ │ │ │ + attribute.nodeValue = value; │ │ │ │ │ + node.setAttributeNode(attribute); │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttribute(name, value); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + throw "setAttributeNS not implemented"; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createElementNSPlus │ │ │ │ │ + * Shorthand for creating namespaced elements with optional attributes and │ │ │ │ │ + * child text nodes. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} The qualified node name. │ │ │ │ │ + * options - {Object} Optional object for node configuration. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * uri - {String} Optional namespace uri for the element - supply a prefix │ │ │ │ │ + * instead if the namespace uri is a property of the format's namespace │ │ │ │ │ + * object. │ │ │ │ │ + * attributes - {Object} Optional attributes to be set using the │ │ │ │ │ + * <setAttributes> method. │ │ │ │ │ + * value - {String} Optional text to be appended as a text node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Element} An element node. │ │ │ │ │ + */ │ │ │ │ │ + createElementNSPlus: function(name, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + // order of prefix preference │ │ │ │ │ + // 1. in the uri option │ │ │ │ │ + // 2. in the prefix option │ │ │ │ │ + // 3. in the qualified name │ │ │ │ │ + // 4. from the defaultPrefix │ │ │ │ │ + var uri = options.uri || this.namespaces[options.prefix]; │ │ │ │ │ + if (!uri) { │ │ │ │ │ + var loc = name.indexOf(":"); │ │ │ │ │ + uri = this.namespaces[name.substring(0, loc)]; │ │ │ │ │ + } │ │ │ │ │ + if (!uri) { │ │ │ │ │ + uri = this.namespaces[this.defaultPrefix]; │ │ │ │ │ + } │ │ │ │ │ + var node = this.createElementNS(uri, name); │ │ │ │ │ + if (options.attributes) { │ │ │ │ │ + this.setAttributes(node, options.attributes); │ │ │ │ │ + } │ │ │ │ │ + var value = options.value; │ │ │ │ │ + if (value != null) { │ │ │ │ │ + node.appendChild(this.createTextNode(value)); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setAttributes │ │ │ │ │ + * Set multiple attributes given key value pairs from an object. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {Element} An element node. │ │ │ │ │ + * obj - {Object || Array} An object whose properties represent attribute │ │ │ │ │ + * names and values represent attribute values. If an attribute name │ │ │ │ │ + * is a qualified name ("prefix:local"), the prefix will be looked up │ │ │ │ │ + * in the parsers {namespaces} object. If the prefix is found, │ │ │ │ │ + * setAttributeNS will be used instead of setAttribute. │ │ │ │ │ + */ │ │ │ │ │ + setAttributes: function(node, obj) { │ │ │ │ │ + var value, uri; │ │ │ │ │ + for (var name in obj) { │ │ │ │ │ + if (obj[name] != null && obj[name].toString) { │ │ │ │ │ + value = obj[name].toString(); │ │ │ │ │ + // check for qualified attribute name ("prefix:local") │ │ │ │ │ + uri = this.namespaces[name.substring(0, name.indexOf(":"))] || null; │ │ │ │ │ + this.setAttributeNS(node, uri, name, value); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: readNode │ │ │ │ │ + * Shorthand for applying one of the named readers given the node │ │ │ │ │ + * namespace and local name. Readers take two args (node, obj) and │ │ │ │ │ + * generally extend or modify the second. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The node to be read (required). │ │ │ │ │ + * obj - {Object} The object to be modified (optional). │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The input object, modified (or a new one if none was provided). │ │ │ │ │ + */ │ │ │ │ │ + readNode: function(node, obj) { │ │ │ │ │ + if (!obj) { │ │ │ │ │ + obj = {}; │ │ │ │ │ + } │ │ │ │ │ + var group = this.readers[node.namespaceURI ? this.namespaceAlias[node.namespaceURI] : this.defaultPrefix]; │ │ │ │ │ + if (group) { │ │ │ │ │ + var local = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ + var reader = group[local] || group["*"]; │ │ │ │ │ + if (reader) { │ │ │ │ │ + reader.apply(this, [node, obj]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: readChildNodes │ │ │ │ │ + * Shorthand for applying the named readers to all children of a node. │ │ │ │ │ + * For each child of type 1 (element), <readSelf> is called. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The node to be read (required). │ │ │ │ │ + * obj - {Object} The object to be modified (optional). │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The input object, modified. │ │ │ │ │ + */ │ │ │ │ │ + readChildNodes: function(node, obj) { │ │ │ │ │ + if (!obj) { │ │ │ │ │ + obj = {}; │ │ │ │ │ + } │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + var child; │ │ │ │ │ + for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ + child = children[i]; │ │ │ │ │ + if (child.nodeType == 1) { │ │ │ │ │ + this.readNode(child, obj); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: writeNode │ │ │ │ │ + * Shorthand for applying one of the named writers and appending the │ │ │ │ │ + * results to a node. If a qualified name is not provided for the │ │ │ │ │ + * second argument (and a local name is used instead), the namespace │ │ │ │ │ + * of the parent node will be assumed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} The name of a node to generate. If a qualified name │ │ │ │ │ + * (e.g. "pre:Name") is used, the namespace prefix is assumed to be │ │ │ │ │ + * in the <writers> group. If a local name is used (e.g. "Name") then │ │ │ │ │ + * the namespace of the parent is assumed. If a local name is used │ │ │ │ │ + * and no parent is supplied, then the default namespace is assumed. │ │ │ │ │ + * obj - {Object} Structure containing data for the writer. │ │ │ │ │ + * parent - {DOMElement} Result will be appended to this node. If no parent │ │ │ │ │ + * is supplied, the node will not be appended to anything. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The child node. │ │ │ │ │ + */ │ │ │ │ │ + writeNode: function(name, obj, parent) { │ │ │ │ │ + var prefix, local; │ │ │ │ │ + var split = name.indexOf(":"); │ │ │ │ │ + if (split > 0) { │ │ │ │ │ + prefix = name.substring(0, split); │ │ │ │ │ + local = name.substring(split + 1); │ │ │ │ │ + } else { │ │ │ │ │ + if (parent) { │ │ │ │ │ + prefix = this.namespaceAlias[parent.namespaceURI]; │ │ │ │ │ + } else { │ │ │ │ │ + prefix = this.defaultPrefix; │ │ │ │ │ + } │ │ │ │ │ + local = name; │ │ │ │ │ + } │ │ │ │ │ + var child = this.writers[prefix][local].apply(this, [obj]); │ │ │ │ │ + if (parent) { │ │ │ │ │ + parent.appendChild(child); │ │ │ │ │ + } │ │ │ │ │ + return child; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getChildEl │ │ │ │ │ + * Get the first child element. Optionally only return the first child │ │ │ │ │ + * if it matches the given name and namespace URI. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The parent node. │ │ │ │ │ + * name - {String} Optional node name (local) to search for. │ │ │ │ │ + * uri - {String} Optional namespace URI to search for. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The first child. Returns null if no element is found, if │ │ │ │ │ + * something significant besides an element is found, or if the element │ │ │ │ │ + * found does not match the optional name and uri. │ │ │ │ │ + */ │ │ │ │ │ + getChildEl: function(node, name, uri) { │ │ │ │ │ + return node && this.getThisOrNextEl(node.firstChild, name, uri); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getNextEl │ │ │ │ │ + * Get the next sibling element. Optionally get the first sibling only │ │ │ │ │ + * if it matches the given local name and namespace URI. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The node. │ │ │ │ │ + * name - {String} Optional local name of the sibling to search for. │ │ │ │ │ + * uri - {String} Optional namespace URI of the sibling to search for. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The next sibling element. Returns null if no element is │ │ │ │ │ + * found, something significant besides an element is found, or the │ │ │ │ │ + * found element does not match the optional name and uri. │ │ │ │ │ + */ │ │ │ │ │ + getNextEl: function(node, name, uri) { │ │ │ │ │ + return node && this.getThisOrNextEl(node.nextSibling, name, uri); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getThisOrNextEl │ │ │ │ │ + * Return this node or the next element node. Optionally get the first │ │ │ │ │ + * sibling with the given local name or namespace URI. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The node. │ │ │ │ │ + * name - {String} Optional local name of the sibling to search for. │ │ │ │ │ + * uri - {String} Optional namespace URI of the sibling to search for. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The next sibling element. Returns null if no element is │ │ │ │ │ + * found, something significant besides an element is found, or the │ │ │ │ │ + * found element does not match the query. │ │ │ │ │ + */ │ │ │ │ │ + getThisOrNextEl: function(node, name, uri) { │ │ │ │ │ + outer: for (var sibling = node; sibling; sibling = sibling.nextSibling) { │ │ │ │ │ + switch (sibling.nodeType) { │ │ │ │ │ + case 1: // Element │ │ │ │ │ + if ((!name || name === (sibling.localName || sibling.nodeName.split(":").pop())) && │ │ │ │ │ + (!uri || uri === sibling.namespaceURI)) { │ │ │ │ │ + // matches │ │ │ │ │ + break outer; │ │ │ │ │ + } │ │ │ │ │ + sibling = null; │ │ │ │ │ + break outer; │ │ │ │ │ + case 3: // Text │ │ │ │ │ + if (/^\s*$/.test(sibling.nodeValue)) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + case 4: // CDATA │ │ │ │ │ + case 6: // ENTITY_NODE │ │ │ │ │ + case 12: // NOTATION_NODE │ │ │ │ │ + case 10: // DOCUMENT_TYPE_NODE │ │ │ │ │ + case 11: // DOCUMENT_FRAGMENT_NODE │ │ │ │ │ + sibling = null; │ │ │ │ │ + break outer; │ │ │ │ │ + } // ignore comments and processing instructions │ │ │ │ │ + } │ │ │ │ │ + return sibling || null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: lookupNamespaceURI │ │ │ │ │ + * Takes a prefix and returns the namespace URI associated with it on the given │ │ │ │ │ + * node if found (and null if not). Supplying null for the prefix will │ │ │ │ │ + * return the default namespace. │ │ │ │ │ + * │ │ │ │ │ + * For browsers that support it, this calls the native lookupNamesapceURI │ │ │ │ │ + * function. In other browsers, this is an implementation of │ │ │ │ │ + * http://www.w3.org/TR/DOM-Level-3-Core/core.html#Node3-lookupNamespaceURI. │ │ │ │ │ + * │ │ │ │ │ + * For browsers that don't support the attribute.ownerElement property, this │ │ │ │ │ + * method cannot be called on attribute nodes. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The node from which to start looking. │ │ │ │ │ + * prefix - {String} The prefix to lookup or null to lookup the default namespace. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The namespace URI for the given prefix. Returns null if the prefix │ │ │ │ │ + * cannot be found or the node is the wrong type. │ │ │ │ │ + */ │ │ │ │ │ + lookupNamespaceURI: function(node, prefix) { │ │ │ │ │ + var uri = null; │ │ │ │ │ + if (node) { │ │ │ │ │ + if (node.lookupNamespaceURI) { │ │ │ │ │ + uri = node.lookupNamespaceURI(prefix); │ │ │ │ │ + } else { │ │ │ │ │ + outer: switch (node.nodeType) { │ │ │ │ │ + case 1: // ELEMENT_NODE │ │ │ │ │ + if (node.namespaceURI !== null && node.prefix === prefix) { │ │ │ │ │ + uri = node.namespaceURI; │ │ │ │ │ + break outer; │ │ │ │ │ + } │ │ │ │ │ + var len = node.attributes.length; │ │ │ │ │ + if (len) { │ │ │ │ │ + var attr; │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + attr = node.attributes[i]; │ │ │ │ │ + if (attr.prefix === "xmlns" && attr.name === "xmlns:" + prefix) { │ │ │ │ │ + uri = attr.value || null; │ │ │ │ │ + break outer; │ │ │ │ │ + } else if (attr.name === "xmlns" && prefix === null) { │ │ │ │ │ + uri = attr.value || null; │ │ │ │ │ + break outer; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + uri = this.lookupNamespaceURI(node.parentNode, prefix); │ │ │ │ │ + break outer; │ │ │ │ │ + case 2: // ATTRIBUTE_NODE │ │ │ │ │ + uri = this.lookupNamespaceURI(node.ownerElement, prefix); │ │ │ │ │ + break outer; │ │ │ │ │ + case 9: // DOCUMENT_NODE │ │ │ │ │ + uri = this.lookupNamespaceURI(node.documentElement, prefix); │ │ │ │ │ + break outer; │ │ │ │ │ + case 6: // ENTITY_NODE │ │ │ │ │ + case 12: // NOTATION_NODE │ │ │ │ │ + case 10: // DOCUMENT_TYPE_NODE │ │ │ │ │ + case 11: // DOCUMENT_FRAGMENT_NODE │ │ │ │ │ + break outer; │ │ │ │ │ + default: │ │ │ │ │ + // TEXT_NODE (3), CDATA_SECTION_NODE (4), ENTITY_REFERENCE_NODE (5), │ │ │ │ │ + // PROCESSING_INSTRUCTION_NODE (7), COMMENT_NODE (8) │ │ │ │ │ + uri = this.lookupNamespaceURI(node.parentNode, prefix); │ │ │ │ │ + break outer; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return uri; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getXMLDoc │ │ │ │ │ + * Get an XML document for nodes that are not supported in HTML (e.g. │ │ │ │ │ + * createCDATASection). On IE, this will either return an existing or │ │ │ │ │ + * create a new <xmldom> on the instance. On other browsers, this will │ │ │ │ │ + * either return an existing or create a new shared document (see │ │ │ │ │ + * <OpenLayers.Format.XML.document>). │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XMLDocument} │ │ │ │ │ + */ │ │ │ │ │ + getXMLDoc: function() { │ │ │ │ │ + if (!OpenLayers.Format.XML.document && !this.xmldom) { │ │ │ │ │ + if (document.implementation && document.implementation.createDocument) { │ │ │ │ │ + OpenLayers.Format.XML.document = │ │ │ │ │ + document.implementation.createDocument("", "", null); │ │ │ │ │ + } else if (!this.xmldom && window.ActiveXObject) { │ │ │ │ │ + this.xmldom = new ActiveXObject("Microsoft.XMLDOM"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Format.XML.document || this.xmldom; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.XML" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Format.XML.CONTENT_TYPE = { │ │ │ │ │ + EMPTY: 0, │ │ │ │ │ + SIMPLE: 1, │ │ │ │ │ + COMPLEX: 2, │ │ │ │ │ + MIXED: 3 │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * APIFunction: OpenLayers.Format.XML.lookupNamespaceURI │ │ │ │ │ + * Takes a prefix and returns the namespace URI associated with it on the given │ │ │ │ │ + * node if found (and null if not). Supplying null for the prefix will │ │ │ │ │ + * return the default namespace. │ │ │ │ │ + * │ │ │ │ │ + * For browsers that support it, this calls the native lookupNamesapceURI │ │ │ │ │ + * function. In other browsers, this is an implementation of │ │ │ │ │ + * http://www.w3.org/TR/DOM-Level-3-Core/core.html#Node3-lookupNamespaceURI. │ │ │ │ │ + * │ │ │ │ │ + * For browsers that don't support the attribute.ownerElement property, this │ │ │ │ │ + * method cannot be called on attribute nodes. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The node from which to start looking. │ │ │ │ │ + * prefix - {String} The prefix to lookup or null to lookup the default namespace. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The namespace URI for the given prefix. Returns null if the prefix │ │ │ │ │ + * cannot be found or the node is the wrong type. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.XML.lookupNamespaceURI = OpenLayers.Function.bind( │ │ │ │ │ + OpenLayers.Format.XML.prototype.lookupNamespaceURI, │ │ │ │ │ + OpenLayers.Format.XML.prototype │ │ │ │ │ +); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Property: OpenLayers.Format.XML.document │ │ │ │ │ + * {XMLDocument} XML document to reuse for creating non-HTML compliant nodes, │ │ │ │ │ + * like document.createCDATASection. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.XML.document = null; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WFST.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Function: OpenLayers.Format.WFST │ │ │ │ │ + * Used to create a versioned WFS protocol. Default version is 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Format>} A WFST format of the given version. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WFST = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options, OpenLayers.Format.WFST.DEFAULTS │ │ │ │ │ + ); │ │ │ │ │ + var cls = OpenLayers.Format.WFST["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported WFST version: " + options.version; │ │ │ │ │ + } │ │ │ │ │ + return new cls(options); │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Format.WFST.DEFAULTS │ │ │ │ │ + * {Object} Default properties for the WFST format. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WFST.DEFAULTS = { │ │ │ │ │ + "version": "1.0.0" │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Feature.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Feature │ │ │ │ │ + * Features are combinations of geography and attributes. The OpenLayers.Feature │ │ │ │ │ + * class specifically combines a marker and a lonlat. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Feature = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {<OpenLayers.Layer>} │ │ │ │ │ + */ │ │ │ │ │ + layer: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: id │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + id: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: lonlat │ │ │ │ │ + * {<OpenLayers.LonLat>} │ │ │ │ │ + */ │ │ │ │ │ + lonlat: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: data │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ + data: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: marker │ │ │ │ │ + * {<OpenLayers.Marker>} │ │ │ │ │ + */ │ │ │ │ │ + marker: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: popupClass │ │ │ │ │ + * {<OpenLayers.Class>} The class which will be used to instantiate │ │ │ │ │ + * a new Popup. Default is <OpenLayers.Popup.Anchored>. │ │ │ │ │ + */ │ │ │ │ │ + popupClass: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: popup │ │ │ │ │ + * {<OpenLayers.Popup>} │ │ │ │ │ + */ │ │ │ │ │ + popup: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Feature │ │ │ │ │ + * Constructor for features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer>} │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ + * data - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature>} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(layer, lonlat, data) { │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + this.lonlat = lonlat; │ │ │ │ │ + this.data = (data != null) ? data : {}; │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * nullify references to prevent circular references and memory leaks │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + │ │ │ │ │ + //remove the popup from the map │ │ │ │ │ + if ((this.layer != null) && (this.layer.map != null)) { │ │ │ │ │ + if (this.popup != null) { │ │ │ │ │ + this.layer.map.removePopup(this.popup); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // remove the marker from the layer │ │ │ │ │ + if (this.layer != null && this.marker != null) { │ │ │ │ │ + this.layer.removeMarker(this.marker); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.layer = null; │ │ │ │ │ + this.id = null; │ │ │ │ │ + this.lonlat = null; │ │ │ │ │ + this.data = null; │ │ │ │ │ + if (this.marker != null) { │ │ │ │ │ + this.destroyMarker(this.marker); │ │ │ │ │ + this.marker = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.popup != null) { │ │ │ │ │ + this.destroyPopup(this.popup); │ │ │ │ │ + this.popup = null; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: onScreen │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the feature is currently visible on screen │ │ │ │ │ + * (based on its 'lonlat' property) │ │ │ │ │ + */ │ │ │ │ │ + onScreen: function() { │ │ │ │ │ + │ │ │ │ │ + var onScreen = false; │ │ │ │ │ + if ((this.layer != null) && (this.layer.map != null)) { │ │ │ │ │ + var screenBounds = this.layer.map.getExtent(); │ │ │ │ │ + onScreen = screenBounds.containsLonLat(this.lonlat); │ │ │ │ │ + } │ │ │ │ │ + return onScreen; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createMarker │ │ │ │ │ + * Based on the data associated with the Feature, create and return a marker object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Marker>} A Marker Object created from the 'lonlat' and 'icon' properties │ │ │ │ │ + * set in this.data. If no 'lonlat' is set, returns null. If no │ │ │ │ │ + * 'icon' is set, OpenLayers.Marker() will load the default image. │ │ │ │ │ + * │ │ │ │ │ + * Note - this.marker is set to return value │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + createMarker: function() { │ │ │ │ │ + │ │ │ │ │ + if (this.lonlat != null) { │ │ │ │ │ + this.marker = new OpenLayers.Marker(this.lonlat, this.data.icon); │ │ │ │ │ + } │ │ │ │ │ + return this.marker; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroyMarker │ │ │ │ │ + * Destroys marker. │ │ │ │ │ + * If user overrides the createMarker() function, s/he should be able │ │ │ │ │ + * to also specify an alternative function for destroying it │ │ │ │ │ + */ │ │ │ │ │ + destroyMarker: function() { │ │ │ │ │ + this.marker.destroy(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createPopup │ │ │ │ │ + * Creates a popup object created from the 'lonlat', 'popupSize', │ │ │ │ │ + * and 'popupContentHTML' properties set in this.data. It uses │ │ │ │ │ + * this.marker.icon as default anchor. │ │ │ │ │ + * │ │ │ │ │ + * If no 'lonlat' is set, returns null. │ │ │ │ │ + * If no this.marker has been created, no anchor is sent. │ │ │ │ │ + * │ │ │ │ │ + * Note - the returned popup object is 'owned' by the feature, so you │ │ │ │ │ + * cannot use the popup's destroy method to discard the popup. │ │ │ │ │ + * Instead, you must use the feature's destroyPopup │ │ │ │ │ + * │ │ │ │ │ + * Note - this.popup is set to return value │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * closeBox - {Boolean} create popup with closebox or not │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Popup>} Returns the created popup, which is also set │ │ │ │ │ + * as 'popup' property of this feature. Will be of whatever type │ │ │ │ │ + * specified by this feature's 'popupClass' property, but must be │ │ │ │ │ + * of type <OpenLayers.Popup>. │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + createPopup: function(closeBox) { │ │ │ │ │ + │ │ │ │ │ + if (this.lonlat != null) { │ │ │ │ │ + if (!this.popup) { │ │ │ │ │ + var anchor = (this.marker) ? this.marker.icon : null; │ │ │ │ │ + var popupClass = this.popupClass ? │ │ │ │ │ + this.popupClass : OpenLayers.Popup.Anchored; │ │ │ │ │ + this.popup = new popupClass(this.id + "_popup", │ │ │ │ │ + this.lonlat, │ │ │ │ │ + this.data.popupSize, │ │ │ │ │ + this.data.popupContentHTML, │ │ │ │ │ + anchor, │ │ │ │ │ + closeBox); │ │ │ │ │ + } │ │ │ │ │ + if (this.data.overflow != null) { │ │ │ │ │ + this.popup.contentDiv.style.overflow = this.data.overflow; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.popup.feature = this; │ │ │ │ │ + } │ │ │ │ │ + return this.popup; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroyPopup │ │ │ │ │ + * Destroys the popup created via createPopup. │ │ │ │ │ + * │ │ │ │ │ + * As with the marker, if user overrides the createPopup() function, s/he │ │ │ │ │ + * should also be able to override the destruction │ │ │ │ │ + */ │ │ │ │ │ + destroyPopup: function() { │ │ │ │ │ + if (this.popup) { │ │ │ │ │ + this.popup.feature = null; │ │ │ │ │ + this.popup.destroy(); │ │ │ │ │ + this.popup = null; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Feature" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Feature/Vector.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +// TRASH THIS │ │ │ │ │ +OpenLayers.State = { │ │ │ │ │ + /** states */ │ │ │ │ │ + UNKNOWN: 'Unknown', │ │ │ │ │ + INSERT: 'Insert', │ │ │ │ │ + UPDATE: 'Update', │ │ │ │ │ + DELETE: 'Delete' │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Feature.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Feature.Vector │ │ │ │ │ + * Vector features use the OpenLayers.Geometry classes as geometry description. │ │ │ │ │ + * They have an 'attributes' property, which is the data object, and a 'style' │ │ │ │ │ + * property, the default values of which are defined in the │ │ │ │ │ + * <OpenLayers.Feature.Vector.style> objects. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Feature> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Feature.Vector = OpenLayers.Class(OpenLayers.Feature, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: fid │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + fid: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: geometry │ │ │ │ │ + * {<OpenLayers.Geometry>} │ │ │ │ │ + */ │ │ │ │ │ + geometry: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: attributes │ │ │ │ │ + * {Object} This object holds arbitrary, serializable properties that │ │ │ │ │ + * describe the feature. │ │ │ │ │ + */ │ │ │ │ │ + attributes: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: bounds │ │ │ │ │ + * {<OpenLayers.Bounds>} The box bounding that feature's geometry, that │ │ │ │ │ + * property can be set by an <OpenLayers.Format> object when │ │ │ │ │ + * deserializing the feature, so in most cases it represents an │ │ │ │ │ + * information set by the server. │ │ │ │ │ + */ │ │ │ │ │ + bounds: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: state │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + state: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: style │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ + style: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String} If this property is set it will be taken into account by │ │ │ │ │ + * {<OpenLayers.HTTP>} when upadting or deleting the feature. │ │ │ │ │ + */ │ │ │ │ │ + url: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: renderIntent │ │ │ │ │ + * {String} rendering intent currently being used │ │ │ │ │ + */ │ │ │ │ │ + renderIntent: "default", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: modified │ │ │ │ │ + * {Object} An object with the originals of the geometry and attributes of │ │ │ │ │ + * the feature, if they were changed. Currently this property is only read │ │ │ │ │ + * by <OpenLayers.Format.WFST.v1>, and written by │ │ │ │ │ + * <OpenLayers.Control.ModifyFeature>, which sets the geometry property. │ │ │ │ │ + * Applications can set the originals of modified attributes in the │ │ │ │ │ + * attributes property. Note that applications have to check if this │ │ │ │ │ + * object and the attributes property is already created before using it. │ │ │ │ │ + * After a change made with ModifyFeature, this object could look like │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * { │ │ │ │ │ + * geometry: >Object │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * When an application has made changes to feature attributes, it could │ │ │ │ │ + * have set the attributes to something like this: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * { │ │ │ │ │ + * attributes: { │ │ │ │ │ + * myAttribute: "original" │ │ │ │ │ + * } │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Note that <OpenLayers.Format.WFST.v1> only checks for truthy values in │ │ │ │ │ + * *modified.geometry* and the attribute names in *modified.attributes*, │ │ │ │ │ + * but it is recommended to set the original values (and not just true) as │ │ │ │ │ + * attribute value, so applications could use this information to undo │ │ │ │ │ + * changes. │ │ │ │ │ + */ │ │ │ │ │ + modified: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Feature.Vector │ │ │ │ │ + * Create a vector feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} The geometry that this feature │ │ │ │ │ + * represents. │ │ │ │ │ + * attributes - {Object} An optional object that will be mapped to the │ │ │ │ │ + * <attributes> property. │ │ │ │ │ + * style - {Object} An optional style object. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(geometry, attributes, style) { │ │ │ │ │ + OpenLayers.Feature.prototype.initialize.apply(this, │ │ │ │ │ + [null, null, attributes]); │ │ │ │ │ + this.lonlat = null; │ │ │ │ │ + this.geometry = geometry ? geometry : null; │ │ │ │ │ + this.state = null; │ │ │ │ │ + this.attributes = {}; │ │ │ │ │ + if (attributes) { │ │ │ │ │ + this.attributes = OpenLayers.Util.extend(this.attributes, │ │ │ │ │ + attributes); │ │ │ │ │ + } │ │ │ │ │ + this.style = style ? style : null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * nullify references to prevent circular references and memory leaks │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.layer) { │ │ │ │ │ + this.layer.removeFeatures(this); │ │ │ │ │ + this.layer = null; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.geometry = null; │ │ │ │ │ + this.modified = null; │ │ │ │ │ + OpenLayers.Feature.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this vector feature. Does not set any non-standard │ │ │ │ │ + * properties. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} An exact clone of this vector feature. │ │ │ │ │ + */ │ │ │ │ │ + clone: function() { │ │ │ │ │ + return new OpenLayers.Feature.Vector( │ │ │ │ │ + this.geometry ? this.geometry.clone() : null, │ │ │ │ │ + this.attributes, │ │ │ │ │ + this.style); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: onScreen │ │ │ │ │ + * Determine whether the feature is within the map viewport. This method │ │ │ │ │ + * tests for an intersection between the geometry and the viewport │ │ │ │ │ + * bounds. If a more effecient but less precise geometry bounds │ │ │ │ │ + * intersection is desired, call the method with the boundsOnly │ │ │ │ │ + * parameter true. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * boundsOnly - {Boolean} Only test whether a feature's bounds intersects │ │ │ │ │ + * the viewport bounds. Default is false. If false, the feature's │ │ │ │ │ + * geometry must intersect the viewport for onScreen to return true. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The feature is currently visible on screen (optionally │ │ │ │ │ + * based on its bounds if boundsOnly is true). │ │ │ │ │ + */ │ │ │ │ │ + onScreen: function(boundsOnly) { │ │ │ │ │ + var onScreen = false; │ │ │ │ │ + if (this.layer && this.layer.map) { │ │ │ │ │ + var screenBounds = this.layer.map.getExtent(); │ │ │ │ │ + if (boundsOnly) { │ │ │ │ │ + var featureBounds = this.geometry.getBounds(); │ │ │ │ │ + onScreen = screenBounds.intersectsBounds(featureBounds); │ │ │ │ │ + } else { │ │ │ │ │ + var screenPoly = screenBounds.toGeometry(); │ │ │ │ │ + onScreen = screenPoly.intersects(this.geometry); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return onScreen; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getVisibility │ │ │ │ │ + * Determine whether the feature is displayed or not. It may not displayed │ │ │ │ │ + * because: │ │ │ │ │ + * - its style display property is set to 'none', │ │ │ │ │ + * - it doesn't belong to any layer, │ │ │ │ │ + * - the styleMap creates a symbolizer with display property set to 'none' │ │ │ │ │ + * for it, │ │ │ │ │ + * - the layer which it belongs to is not visible. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The feature is currently displayed. │ │ │ │ │ + */ │ │ │ │ │ + getVisibility: function() { │ │ │ │ │ + return !(this.style && this.style.display == 'none' || │ │ │ │ │ + !this.layer || │ │ │ │ │ + this.layer && this.layer.styleMap && │ │ │ │ │ + this.layer.styleMap.createSymbolizer(this, this.renderIntent).display == 'none' || │ │ │ │ │ + this.layer && !this.layer.getVisibility()); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createMarker │ │ │ │ │ + * HACK - we need to decide if all vector features should be able to │ │ │ │ │ + * create markers │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Marker>} For now just returns null │ │ │ │ │ + */ │ │ │ │ │ + createMarker: function() { │ │ │ │ │ + return null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroyMarker │ │ │ │ │ + * HACK - we need to decide if all vector features should be able to │ │ │ │ │ + * delete markers │ │ │ │ │ + * │ │ │ │ │ + * If user overrides the createMarker() function, s/he should be able │ │ │ │ │ + * to also specify an alternative function for destroying it │ │ │ │ │ + */ │ │ │ │ │ + destroyMarker: function() { │ │ │ │ │ + // pass │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createPopup │ │ │ │ │ + * HACK - we need to decide if all vector features should be able to │ │ │ │ │ + * create popups │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Popup>} For now just returns null │ │ │ │ │ + */ │ │ │ │ │ + createPopup: function() { │ │ │ │ │ + return null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: atPoint │ │ │ │ │ + * Determins whether the feature intersects with the specified location. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>|Object} OpenLayers.LonLat or an │ │ │ │ │ + * object with a 'lon' and 'lat' properties. │ │ │ │ │ + * toleranceLon - {float} Optional tolerance in Geometric Coords │ │ │ │ │ + * toleranceLat - {float} Optional tolerance in Geographic Coords │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the feature is at the specified location │ │ │ │ │ + */ │ │ │ │ │ + atPoint: function(lonlat, toleranceLon, toleranceLat) { │ │ │ │ │ + var atPoint = false; │ │ │ │ │ + if (this.geometry) { │ │ │ │ │ + atPoint = this.geometry.atPoint(lonlat, toleranceLon, │ │ │ │ │ + toleranceLat); │ │ │ │ │ + } │ │ │ │ │ + return atPoint; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroyPopup │ │ │ │ │ + * HACK - we need to decide if all vector features should be able to │ │ │ │ │ + * delete popups │ │ │ │ │ + */ │ │ │ │ │ + destroyPopup: function() { │ │ │ │ │ + // pass │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: move │ │ │ │ │ + * Moves the feature and redraws it at its new location │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * location - {<OpenLayers.LonLat> or <OpenLayers.Pixel>} the │ │ │ │ │ + * location to which to move the feature. │ │ │ │ │ + */ │ │ │ │ │ + move: function(location) { │ │ │ │ │ + │ │ │ │ │ + if (!this.layer || !this.geometry.move) { │ │ │ │ │ + //do nothing if no layer or immoveable geometry │ │ │ │ │ + return undefined; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var pixel; │ │ │ │ │ + if (location.CLASS_NAME == "OpenLayers.LonLat") { │ │ │ │ │ + pixel = this.layer.getViewPortPxFromLonLat(location); │ │ │ │ │ + } else { │ │ │ │ │ + pixel = location; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var lastPixel = this.layer.getViewPortPxFromLonLat(this.geometry.getBounds().getCenterLonLat()); │ │ │ │ │ + var res = this.layer.map.getResolution(); │ │ │ │ │ + this.geometry.move(res * (pixel.x - lastPixel.x), │ │ │ │ │ + res * (lastPixel.y - pixel.y)); │ │ │ │ │ + this.layer.drawFeature(this); │ │ │ │ │ + return lastPixel; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: toState │ │ │ │ │ + * Sets the new state │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * state - {String} │ │ │ │ │ + */ │ │ │ │ │ + toState: function(state) { │ │ │ │ │ + if (state == OpenLayers.State.UPDATE) { │ │ │ │ │ + switch (this.state) { │ │ │ │ │ + case OpenLayers.State.UNKNOWN: │ │ │ │ │ + case OpenLayers.State.DELETE: │ │ │ │ │ + this.state = state; │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.State.UPDATE: │ │ │ │ │ + case OpenLayers.State.INSERT: │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } else if (state == OpenLayers.State.INSERT) { │ │ │ │ │ + switch (this.state) { │ │ │ │ │ + case OpenLayers.State.UNKNOWN: │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + this.state = state; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } else if (state == OpenLayers.State.DELETE) { │ │ │ │ │ + switch (this.state) { │ │ │ │ │ + case OpenLayers.State.INSERT: │ │ │ │ │ + // the feature should be destroyed │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.State.DELETE: │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.State.UNKNOWN: │ │ │ │ │ + case OpenLayers.State.UPDATE: │ │ │ │ │ + this.state = state; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } else if (state == OpenLayers.State.UNKNOWN) { │ │ │ │ │ + this.state = state; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Feature.Vector" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Feature.Vector.style │ │ │ │ │ + * OpenLayers features can have a number of style attributes. The 'default' │ │ │ │ │ + * style will typically be used if no other style is specified. These │ │ │ │ │ + * styles correspond for the most part, to the styling properties defined │ │ │ │ │ + * by the SVG standard. │ │ │ │ │ + * Information on fill properties: http://www.w3.org/TR/SVG/painting.html#FillProperties │ │ │ │ │ + * Information on stroke properties: http://www.w3.org/TR/SVG/painting.html#StrokeProperties │ │ │ │ │ + * │ │ │ │ │ + * Symbolizer properties: │ │ │ │ │ + * fill - {Boolean} Set to false if no fill is desired. │ │ │ │ │ + * fillColor - {String} Hex fill color. Default is "#ee9900". │ │ │ │ │ + * fillOpacity - {Number} Fill opacity (0-1). Default is 0.4 │ │ │ │ │ + * stroke - {Boolean} Set to false if no stroke is desired. │ │ │ │ │ + * strokeColor - {String} Hex stroke color. Default is "#ee9900". │ │ │ │ │ + * strokeOpacity - {Number} Stroke opacity (0-1). Default is 1. │ │ │ │ │ + * strokeWidth - {Number} Pixel stroke width. Default is 1. │ │ │ │ │ + * strokeLinecap - {String} Stroke cap type. Default is "round". [butt | round | square] │ │ │ │ │ + * strokeDashstyle - {String} Stroke dash style. Default is "solid". [dot | dash | dashdot | longdash | longdashdot | solid] │ │ │ │ │ + * graphic - {Boolean} Set to false if no graphic is desired. │ │ │ │ │ + * pointRadius - {Number} Pixel point radius. Default is 6. │ │ │ │ │ + * pointerEvents - {String} Default is "visiblePainted". │ │ │ │ │ + * cursor - {String} Default is "". │ │ │ │ │ + * externalGraphic - {String} Url to an external graphic that will be used for rendering points. │ │ │ │ │ + * graphicWidth - {Number} Pixel width for sizing an external graphic. │ │ │ │ │ + * graphicHeight - {Number} Pixel height for sizing an external graphic. │ │ │ │ │ + * graphicOpacity - {Number} Opacity (0-1) for an external graphic. │ │ │ │ │ + * graphicXOffset - {Number} Pixel offset along the positive x axis for displacing an external graphic. │ │ │ │ │ + * graphicYOffset - {Number} Pixel offset along the positive y axis for displacing an external graphic. │ │ │ │ │ + * rotation - {Number} For point symbolizers, this is the rotation of a graphic in the clockwise direction about its center point (or any point off center as specified by graphicXOffset and graphicYOffset). │ │ │ │ │ + * graphicZIndex - {Number} The integer z-index value to use in rendering. │ │ │ │ │ + * graphicName - {String} Named graphic to use when rendering points. Supported values include "circle" (default), │ │ │ │ │ + * "square", "star", "x", "cross", "triangle". │ │ │ │ │ + * graphicTitle - {String} Tooltip when hovering over a feature. *deprecated*, use title instead │ │ │ │ │ + * title - {String} Tooltip when hovering over a feature. Not supported by the canvas renderer. │ │ │ │ │ + * backgroundGraphic - {String} Url to a graphic to be used as the background under an externalGraphic. │ │ │ │ │ + * backgroundGraphicZIndex - {Number} The integer z-index value to use in rendering the background graphic. │ │ │ │ │ + * backgroundXOffset - {Number} The x offset (in pixels) for the background graphic. │ │ │ │ │ + * backgroundYOffset - {Number} The y offset (in pixels) for the background graphic. │ │ │ │ │ + * backgroundHeight - {Number} The height of the background graphic. If not provided, the graphicHeight will be used. │ │ │ │ │ + * backgroundWidth - {Number} The width of the background width. If not provided, the graphicWidth will be used. │ │ │ │ │ + * label - {String} The text for an optional label. For browsers that use the canvas renderer, this requires either │ │ │ │ │ + * fillText or mozDrawText to be available. │ │ │ │ │ + * labelAlign - {String} Label alignment. This specifies the insertion point relative to the text. It is a string │ │ │ │ │ + * composed of two characters. The first character is for the horizontal alignment, the second for the vertical │ │ │ │ │ + * alignment. Valid values for horizontal alignment: "l"=left, "c"=center, "r"=right. Valid values for vertical │ │ │ │ │ + * alignment: "t"=top, "m"=middle, "b"=bottom. Example values: "lt", "cm", "rb". Default is "cm". │ │ │ │ │ + * labelXOffset - {Number} Pixel offset along the positive x axis for displacing the label. Not supported by the canvas renderer. │ │ │ │ │ + * labelYOffset - {Number} Pixel offset along the positive y axis for displacing the label. Not supported by the canvas renderer. │ │ │ │ │ + * labelSelect - {Boolean} If set to true, labels will be selectable using SelectFeature or similar controls. │ │ │ │ │ + * Default is false. │ │ │ │ │ + * labelOutlineColor - {String} The color of the label outline. Default is 'white'. Only supported by the canvas & SVG renderers. │ │ │ │ │ + * labelOutlineWidth - {Number} The width of the label outline. Default is 3, set to 0 or null to disable. Only supported by the SVG renderers. │ │ │ │ │ + * labelOutlineOpacity - {Number} The opacity (0-1) of the label outline. Default is fontOpacity. Only supported by the canvas & SVG renderers. │ │ │ │ │ + * fontColor - {String} The font color for the label, to be provided like CSS. │ │ │ │ │ + * fontOpacity - {Number} Opacity (0-1) for the label │ │ │ │ │ + * fontFamily - {String} The font family for the label, to be provided like in CSS. │ │ │ │ │ + * fontSize - {String} The font size for the label, to be provided like in CSS. │ │ │ │ │ + * fontStyle - {String} The font style for the label, to be provided like in CSS. │ │ │ │ │ + * fontWeight - {String} The font weight for the label, to be provided like in CSS. │ │ │ │ │ + * display - {String} Symbolizers will have no effect if display is set to "none". All other values have no effect. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Feature.Vector.style = { │ │ │ │ │ + 'default': { │ │ │ │ │ + fillColor: "#ee9900", │ │ │ │ │ + fillOpacity: 0.4, │ │ │ │ │ + hoverFillColor: "white", │ │ │ │ │ + hoverFillOpacity: 0.8, │ │ │ │ │ + strokeColor: "#ee9900", │ │ │ │ │ + strokeOpacity: 1, │ │ │ │ │ + strokeWidth: 1, │ │ │ │ │ + strokeLinecap: "round", │ │ │ │ │ + strokeDashstyle: "solid", │ │ │ │ │ + hoverStrokeColor: "red", │ │ │ │ │ + hoverStrokeOpacity: 1, │ │ │ │ │ + hoverStrokeWidth: 0.2, │ │ │ │ │ + pointRadius: 6, │ │ │ │ │ + hoverPointRadius: 1, │ │ │ │ │ + hoverPointUnit: "%", │ │ │ │ │ + pointerEvents: "visiblePainted", │ │ │ │ │ + cursor: "inherit", │ │ │ │ │ + fontColor: "#000000", │ │ │ │ │ + labelAlign: "cm", │ │ │ │ │ + labelOutlineColor: "white", │ │ │ │ │ + labelOutlineWidth: 3 │ │ │ │ │ + }, │ │ │ │ │ + 'select': { │ │ │ │ │ + fillColor: "blue", │ │ │ │ │ + fillOpacity: 0.4, │ │ │ │ │ + hoverFillColor: "white", │ │ │ │ │ + hoverFillOpacity: 0.8, │ │ │ │ │ + strokeColor: "blue", │ │ │ │ │ + strokeOpacity: 1, │ │ │ │ │ + strokeWidth: 2, │ │ │ │ │ + strokeLinecap: "round", │ │ │ │ │ + strokeDashstyle: "solid", │ │ │ │ │ + hoverStrokeColor: "red", │ │ │ │ │ + hoverStrokeOpacity: 1, │ │ │ │ │ + hoverStrokeWidth: 0.2, │ │ │ │ │ + pointRadius: 6, │ │ │ │ │ + hoverPointRadius: 1, │ │ │ │ │ + hoverPointUnit: "%", │ │ │ │ │ + pointerEvents: "visiblePainted", │ │ │ │ │ + cursor: "pointer", │ │ │ │ │ + fontColor: "#000000", │ │ │ │ │ + labelAlign: "cm", │ │ │ │ │ + labelOutlineColor: "white", │ │ │ │ │ + labelOutlineWidth: 3 │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + 'temporary': { │ │ │ │ │ + fillColor: "#66cccc", │ │ │ │ │ + fillOpacity: 0.2, │ │ │ │ │ + hoverFillColor: "white", │ │ │ │ │ + hoverFillOpacity: 0.8, │ │ │ │ │ + strokeColor: "#66cccc", │ │ │ │ │ + strokeOpacity: 1, │ │ │ │ │ + strokeLinecap: "round", │ │ │ │ │ + strokeWidth: 2, │ │ │ │ │ + strokeDashstyle: "solid", │ │ │ │ │ + hoverStrokeColor: "red", │ │ │ │ │ + hoverStrokeOpacity: 1, │ │ │ │ │ + hoverStrokeWidth: 0.2, │ │ │ │ │ + pointRadius: 6, │ │ │ │ │ + hoverPointRadius: 1, │ │ │ │ │ + hoverPointUnit: "%", │ │ │ │ │ + pointerEvents: "visiblePainted", │ │ │ │ │ + cursor: "inherit", │ │ │ │ │ + fontColor: "#000000", │ │ │ │ │ + labelAlign: "cm", │ │ │ │ │ + labelOutlineColor: "white", │ │ │ │ │ + labelOutlineWidth: 3 │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + 'delete': { │ │ │ │ │ + display: "none" │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Style.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Style │ │ │ │ │ + * This class represents a UserStyle obtained │ │ │ │ │ + * from a SLD, containing styling rules. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Style = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: id │ │ │ │ │ + * {String} A unique id for this session. │ │ │ │ │ + */ │ │ │ │ │ + id: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: name │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + name: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: title │ │ │ │ │ + * {String} Title of this style (set if included in SLD) │ │ │ │ │ + */ │ │ │ │ │ + title: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: description │ │ │ │ │ + * {String} Description of this style (set if abstract is included in SLD) │ │ │ │ │ + */ │ │ │ │ │ + description: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layerName │ │ │ │ │ + * {<String>} name of the layer that this style belongs to, usually │ │ │ │ │ + * according to the NamedLayer attribute of an SLD document. │ │ │ │ │ + */ │ │ │ │ │ + layerName: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isDefault │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + isDefault: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: rules │ │ │ │ │ + * {Array(<OpenLayers.Rule>)} │ │ │ │ │ + */ │ │ │ │ │ + rules: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: context │ │ │ │ │ + * {Object} An optional object with properties that symbolizers' property │ │ │ │ │ + * values should be evaluated against. If no context is specified, │ │ │ │ │ + * feature.attributes will be used │ │ │ │ │ + */ │ │ │ │ │ + context: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultStyle │ │ │ │ │ + * {Object} hash of style properties to use as default for merging │ │ │ │ │ + * rule-based style symbolizers onto. If no rules are defined, │ │ │ │ │ + * createSymbolizer will return this style. If <defaultsPerSymbolizer> is set to │ │ │ │ │ + * true, the defaultStyle will only be taken into account if there are │ │ │ │ │ + * rules defined. │ │ │ │ │ + */ │ │ │ │ │ + defaultStyle: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultsPerSymbolizer │ │ │ │ │ + * {Boolean} If set to true, the <defaultStyle> will extend the symbolizer │ │ │ │ │ + * of every rule. Properties of the <defaultStyle> will also be used to set │ │ │ │ │ + * missing symbolizer properties if the symbolizer has stroke, fill or │ │ │ │ │ + * graphic set to true. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + defaultsPerSymbolizer: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: propertyStyles │ │ │ │ │ + * {Hash of Boolean} cache of style properties that need to be parsed for │ │ │ │ │ + * propertyNames. Property names are keys, values won't be used. │ │ │ │ │ + */ │ │ │ │ │ + propertyStyles: null, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Style │ │ │ │ │ + * Creates a UserStyle. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * style - {Object} Optional hash of style properties that will be │ │ │ │ │ + * used as default style for this style object. This style │ │ │ │ │ + * applies if no rules are specified. Symbolizers defined in │ │ │ │ │ + * rules will extend this default style. │ │ │ │ │ + * options - {Object} An optional object with properties to set on the │ │ │ │ │ + * style. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * rules - {Array(<OpenLayers.Rule>)} List of rules to be added to the │ │ │ │ │ + * style. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Style>} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(style, options) { │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.rules = []; │ │ │ │ │ + if (options && options.rules) { │ │ │ │ │ + this.addRules(options.rules); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // use the default style from OpenLayers.Feature.Vector if no style │ │ │ │ │ + // was given in the constructor │ │ │ │ │ + this.setDefaultStyle(style || │ │ │ │ │ + OpenLayers.Feature.Vector.style["default"]); │ │ │ │ │ + │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * nullify references to prevent circular references and memory leaks │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i = 0, len = this.rules.length; i < len; i++) { │ │ │ │ │ + this.rules[i].destroy(); │ │ │ │ │ + this.rules[i] = null; │ │ │ │ │ + } │ │ │ │ │ + this.rules = null; │ │ │ │ │ + this.defaultStyle = null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createSymbolizer │ │ │ │ │ + * creates a style by applying all feature-dependent rules to the base │ │ │ │ │ + * style. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature>} feature to evaluate rules for │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} symbolizer hash │ │ │ │ │ + */ │ │ │ │ │ + createSymbolizer: function(feature) { │ │ │ │ │ + var style = this.defaultsPerSymbolizer ? {} : this.createLiterals( │ │ │ │ │ + OpenLayers.Util.extend({}, this.defaultStyle), feature); │ │ │ │ │ + │ │ │ │ │ + var rules = this.rules; │ │ │ │ │ + │ │ │ │ │ + var rule, context; │ │ │ │ │ + var elseRules = []; │ │ │ │ │ + var appliedRules = false; │ │ │ │ │ + for (var i = 0, len = rules.length; i < len; i++) { │ │ │ │ │ + rule = rules[i]; │ │ │ │ │ + // does the rule apply? │ │ │ │ │ + var applies = rule.evaluate(feature); │ │ │ │ │ + │ │ │ │ │ + if (applies) { │ │ │ │ │ + if (rule instanceof OpenLayers.Rule && rule.elseFilter) { │ │ │ │ │ + elseRules.push(rule); │ │ │ │ │ + } else { │ │ │ │ │ + appliedRules = true; │ │ │ │ │ + this.applySymbolizer(rule, style, feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // if no other rules apply, apply the rules with else filters │ │ │ │ │ + if (appliedRules == false && elseRules.length > 0) { │ │ │ │ │ + appliedRules = true; │ │ │ │ │ + for (var i = 0, len = elseRules.length; i < len; i++) { │ │ │ │ │ + this.applySymbolizer(elseRules[i], style, feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // don't display if there were rules but none applied │ │ │ │ │ + if (rules.length > 0 && appliedRules == false) { │ │ │ │ │ + style.display = "none"; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (style.label != null && typeof style.label !== "string") { │ │ │ │ │ + style.label = String(style.label); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return style; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: applySymbolizer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * rule - {<OpenLayers.Rule>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * feature - {<OpenLayer.Feature.Vector>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} A style with new symbolizer applied. │ │ │ │ │ + */ │ │ │ │ │ + applySymbolizer: function(rule, style, feature) { │ │ │ │ │ + var symbolizerPrefix = feature.geometry ? │ │ │ │ │ + this.getSymbolizerPrefix(feature.geometry) : │ │ │ │ │ + OpenLayers.Style.SYMBOLIZER_PREFIXES[0]; │ │ │ │ │ + │ │ │ │ │ + var symbolizer = rule.symbolizer[symbolizerPrefix] || rule.symbolizer; │ │ │ │ │ + │ │ │ │ │ + if (this.defaultsPerSymbolizer === true) { │ │ │ │ │ + var defaults = this.defaultStyle; │ │ │ │ │ + OpenLayers.Util.applyDefaults(symbolizer, { │ │ │ │ │ + pointRadius: defaults.pointRadius │ │ │ │ │ + }); │ │ │ │ │ + if (symbolizer.stroke === true || symbolizer.graphic === true) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(symbolizer, { │ │ │ │ │ + strokeWidth: defaults.strokeWidth, │ │ │ │ │ + strokeColor: defaults.strokeColor, │ │ │ │ │ + strokeOpacity: defaults.strokeOpacity, │ │ │ │ │ + strokeDashstyle: defaults.strokeDashstyle, │ │ │ │ │ + strokeLinecap: defaults.strokeLinecap │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.fill === true || symbolizer.graphic === true) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(symbolizer, { │ │ │ │ │ + fillColor: defaults.fillColor, │ │ │ │ │ + fillOpacity: defaults.fillOpacity │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.graphic === true) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(symbolizer, { │ │ │ │ │ + pointRadius: this.defaultStyle.pointRadius, │ │ │ │ │ + externalGraphic: this.defaultStyle.externalGraphic, │ │ │ │ │ + graphicName: this.defaultStyle.graphicName, │ │ │ │ │ + graphicOpacity: this.defaultStyle.graphicOpacity, │ │ │ │ │ + graphicWidth: this.defaultStyle.graphicWidth, │ │ │ │ │ + graphicHeight: this.defaultStyle.graphicHeight, │ │ │ │ │ + graphicXOffset: this.defaultStyle.graphicXOffset, │ │ │ │ │ + graphicYOffset: this.defaultStyle.graphicYOffset │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // merge the style with the current style │ │ │ │ │ + return this.createLiterals( │ │ │ │ │ + OpenLayers.Util.extend(style, symbolizer), feature); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createLiterals │ │ │ │ │ + * creates literals for all style properties that have an entry in │ │ │ │ │ + * <this.propertyStyles>. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * style - {Object} style to create literals for. Will be modified │ │ │ │ │ + * inline. │ │ │ │ │ + * feature - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} the modified style │ │ │ │ │ + */ │ │ │ │ │ + createLiterals: function(style, feature) { │ │ │ │ │ + var context = OpenLayers.Util.extend({}, feature.attributes || feature.data); │ │ │ │ │ + OpenLayers.Util.extend(context, this.context); │ │ │ │ │ + │ │ │ │ │ + for (var i in this.propertyStyles) { │ │ │ │ │ + style[i] = OpenLayers.Style.createLiteral(style[i], context, feature, i); │ │ │ │ │ + } │ │ │ │ │ + return style; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: findPropertyStyles │ │ │ │ │ + * Looks into all rules for this style and the defaultStyle to collect │ │ │ │ │ + * all the style hash property names containing ${...} strings that have │ │ │ │ │ + * to be replaced using the createLiteral method before returning them. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} hash of property names that need createLiteral parsing. The │ │ │ │ │ + * name of the property is the key, and the value is true; │ │ │ │ │ + */ │ │ │ │ │ + findPropertyStyles: function() { │ │ │ │ │ + var propertyStyles = {}; │ │ │ │ │ + │ │ │ │ │ + // check the default style │ │ │ │ │ + var style = this.defaultStyle; │ │ │ │ │ + this.addPropertyStyles(propertyStyles, style); │ │ │ │ │ + │ │ │ │ │ + // walk through all rules to check for properties in their symbolizer │ │ │ │ │ + var rules = this.rules; │ │ │ │ │ + var symbolizer, value; │ │ │ │ │ + for (var i = 0, len = rules.length; i < len; i++) { │ │ │ │ │ + symbolizer = rules[i].symbolizer; │ │ │ │ │ + for (var key in symbolizer) { │ │ │ │ │ + value = symbolizer[key]; │ │ │ │ │ + if (typeof value == "object") { │ │ │ │ │ + // symbolizer key is "Point", "Line" or "Polygon" │ │ │ │ │ + this.addPropertyStyles(propertyStyles, value); │ │ │ │ │ + } else { │ │ │ │ │ + // symbolizer is a hash of style properties │ │ │ │ │ + this.addPropertyStyles(propertyStyles, symbolizer); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return propertyStyles; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: addPropertyStyles │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * propertyStyles - {Object} hash to add new property styles to. Will be │ │ │ │ │ + * modified inline │ │ │ │ │ + * symbolizer - {Object} search this symbolizer for property styles │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} propertyStyles hash │ │ │ │ │ + */ │ │ │ │ │ + addPropertyStyles: function(propertyStyles, symbolizer) { │ │ │ │ │ + var property; │ │ │ │ │ + for (var key in symbolizer) { │ │ │ │ │ + property = symbolizer[key]; │ │ │ │ │ + if (typeof property == "string" && │ │ │ │ │ + property.match(/\$\{\w+\}/)) { │ │ │ │ │ + propertyStyles[key] = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return propertyStyles; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: addRules │ │ │ │ │ + * Adds rules to this style. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * rules - {Array(<OpenLayers.Rule>)} │ │ │ │ │ + */ │ │ │ │ │ + addRules: function(rules) { │ │ │ │ │ + Array.prototype.push.apply(this.rules, rules); │ │ │ │ │ + this.propertyStyles = this.findPropertyStyles(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setDefaultStyle │ │ │ │ │ + * Sets the default style for this style object. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * style - {Object} Hash of style properties │ │ │ │ │ + */ │ │ │ │ │ + setDefaultStyle: function(style) { │ │ │ │ │ + this.defaultStyle = style; │ │ │ │ │ + this.propertyStyles = this.findPropertyStyles(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getSymbolizerPrefix │ │ │ │ │ + * Returns the correct symbolizer prefix according to the │ │ │ │ │ + * geometry type of the passed geometry │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} key of the according symbolizer │ │ │ │ │ + */ │ │ │ │ │ + getSymbolizerPrefix: function(geometry) { │ │ │ │ │ + var prefixes = OpenLayers.Style.SYMBOLIZER_PREFIXES; │ │ │ │ │ + for (var i = 0, len = prefixes.length; i < len; i++) { │ │ │ │ │ + if (geometry.CLASS_NAME.indexOf(prefixes[i]) != -1) { │ │ │ │ │ + return prefixes[i]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Clones this style. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Style>} Clone of this style. │ │ │ │ │ + */ │ │ │ │ │ + clone: function() { │ │ │ │ │ + var options = OpenLayers.Util.extend({}, this); │ │ │ │ │ + // clone rules │ │ │ │ │ + if (this.rules) { │ │ │ │ │ + options.rules = []; │ │ │ │ │ + for (var i = 0, len = this.rules.length; i < len; ++i) { │ │ │ │ │ + options.rules.push(this.rules[i].clone()); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // clone context │ │ │ │ │ + options.context = this.context && OpenLayers.Util.extend({}, this.context); │ │ │ │ │ + //clone default style │ │ │ │ │ + var defaultStyle = OpenLayers.Util.extend({}, this.defaultStyle); │ │ │ │ │ + return new OpenLayers.Style(defaultStyle, options); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Style" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Function: createLiteral │ │ │ │ │ + * converts a style value holding a combination of PropertyName and Literal │ │ │ │ │ + * into a Literal, taking the property values from the passed features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * value - {String} value to parse. If this string contains a construct like │ │ │ │ │ + * "foo ${bar}", then "foo " will be taken as literal, and "${bar}" │ │ │ │ │ + * will be replaced by the value of the "bar" attribute of the passed │ │ │ │ │ + * feature. │ │ │ │ │ + * context - {Object} context to take attribute values from │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} optional feature to pass to │ │ │ │ │ + * <OpenLayers.String.format> for evaluating functions in the │ │ │ │ │ + * context. │ │ │ │ │ + * property - {String} optional, name of the property for which the literal is │ │ │ │ │ + * being created for evaluating functions in the context. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} the parsed value. In the example of the value parameter above, the │ │ │ │ │ + * result would be "foo valueOfBar", assuming that the passed feature has an │ │ │ │ │ + * attribute named "bar" with the value "valueOfBar". │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Style.createLiteral = function(value, context, feature, property) { │ │ │ │ │ + if (typeof value == "string" && value.indexOf("${") != -1) { │ │ │ │ │ + value = OpenLayers.String.format(value, context, [feature, property]); │ │ │ │ │ + value = (isNaN(value) || !value) ? value : parseFloat(value); │ │ │ │ │ + } │ │ │ │ │ + return value; │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Style.SYMBOLIZER_PREFIXES │ │ │ │ │ + * {Array} prefixes of the sld symbolizers. These are the │ │ │ │ │ + * same as the main geometry types │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Style.SYMBOLIZER_PREFIXES = ['Point', 'Line', 'Polygon', 'Text', │ │ │ │ │ + 'Raster' │ │ │ │ │ +]; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Filter.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + * @requires OpenLayers/Style.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Filter │ │ │ │ │ + * This class represents an OGC Filter. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Filter = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Filter │ │ │ │ │ + * This class represents a generic filter. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter>} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Remove reference to anything added. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: evaluate │ │ │ │ │ + * Evaluates this filter in a specific context. Instances or subclasses │ │ │ │ │ + * are supposed to override this method. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * context - {Object} Context to use in evaluating the filter. If a vector │ │ │ │ │ + * feature is provided, the feature.attributes will be used as context. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The filter applies. │ │ │ │ │ + */ │ │ │ │ │ + evaluate: function(context) { │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Clones this filter. Should be implemented by subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter>} Clone of this filter. │ │ │ │ │ + */ │ │ │ │ │ + clone: function() { │ │ │ │ │ + return null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: toString │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} Include <OpenLayers.Format.CQL> in your build to get a CQL │ │ │ │ │ + * representation of the filter returned. Otherwise "[Object object]" │ │ │ │ │ + * will be returned. │ │ │ │ │ + */ │ │ │ │ │ + toString: function() { │ │ │ │ │ + var string; │ │ │ │ │ + if (OpenLayers.Format && OpenLayers.Format.CQL) { │ │ │ │ │ + string = OpenLayers.Format.CQL.prototype.write(this); │ │ │ │ │ + } else { │ │ │ │ │ + string = Object.prototype.toString.call(this); │ │ │ │ │ + } │ │ │ │ │ + return string; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Filter" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Filter/Spatial.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Filter.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Filter.Spatial │ │ │ │ │ + * This class represents a spatial filter. │ │ │ │ │ + * Currently implemented: BBOX, DWithin and Intersects │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Filter> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Filter.Spatial = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: type │ │ │ │ │ + * {String} Type of spatial filter. │ │ │ │ │ + * │ │ │ │ │ + * The type should be one of: │ │ │ │ │ + * - OpenLayers.Filter.Spatial.BBOX │ │ │ │ │ + * - OpenLayers.Filter.Spatial.INTERSECTS │ │ │ │ │ + * - OpenLayers.Filter.Spatial.DWITHIN │ │ │ │ │ + * - OpenLayers.Filter.Spatial.WITHIN │ │ │ │ │ + * - OpenLayers.Filter.Spatial.CONTAINS │ │ │ │ │ + */ │ │ │ │ │ + type: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: property │ │ │ │ │ + * {String} Name of the context property to compare. │ │ │ │ │ + */ │ │ │ │ │ + property: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: value │ │ │ │ │ + * {<OpenLayers.Bounds> || <OpenLayers.Geometry>} The bounds or geometry │ │ │ │ │ + * to be used by the filter. Use bounds for BBOX filters and geometry │ │ │ │ │ + * for INTERSECTS or DWITHIN filters. │ │ │ │ │ + */ │ │ │ │ │ + value: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: distance │ │ │ │ │ + * {Number} The distance to use in a DWithin spatial filter. │ │ │ │ │ + */ │ │ │ │ │ + distance: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: distanceUnits │ │ │ │ │ + * {String} The units to use for the distance, e.g. 'm'. │ │ │ │ │ + */ │ │ │ │ │ + distanceUnits: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Filter.Spatial │ │ │ │ │ + * Creates a spatial filter. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object with properties to set on the │ │ │ │ │ + * filter. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter.Spatial>} │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: evaluate │ │ │ │ │ + * Evaluates this filter for a specific feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} feature to apply the filter to. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The feature meets filter criteria. │ │ │ │ │ + */ │ │ │ │ │ + evaluate: function(feature) { │ │ │ │ │ + var intersect = false; │ │ │ │ │ + switch (this.type) { │ │ │ │ │ + case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ + case OpenLayers.Filter.Spatial.INTERSECTS: │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + var geom = this.value; │ │ │ │ │ + if (this.value.CLASS_NAME == "OpenLayers.Bounds") { │ │ │ │ │ + geom = this.value.toGeometry(); │ │ │ │ │ + } │ │ │ │ │ + if (feature.geometry.intersects(geom)) { │ │ │ │ │ + intersect = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + throw new Error('evaluate is not implemented for this filter type.'); │ │ │ │ │ + } │ │ │ │ │ + return intersect; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Clones this filter. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter.Spatial>} Clone of this filter. │ │ │ │ │ + */ │ │ │ │ │ + clone: function() { │ │ │ │ │ + var options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + value: this.value && this.value.clone && this.value.clone() │ │ │ │ │ + }, this); │ │ │ │ │ + return new OpenLayers.Filter.Spatial(options); │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Filter.Spatial" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Filter.Spatial.BBOX = "BBOX"; │ │ │ │ │ +OpenLayers.Filter.Spatial.INTERSECTS = "INTERSECTS"; │ │ │ │ │ +OpenLayers.Filter.Spatial.DWITHIN = "DWITHIN"; │ │ │ │ │ +OpenLayers.Filter.Spatial.WITHIN = "WITHIN"; │ │ │ │ │ +OpenLayers.Filter.Spatial.CONTAINS = "CONTAINS"; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Filter/FeatureId.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Filter.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Filter.FeatureId │ │ │ │ │ + * This class represents a ogc:FeatureId Filter, as being used for rule-based SLD │ │ │ │ │ + * styling │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Filter> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Filter.FeatureId = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: fids │ │ │ │ │ + * {Array(String)} Feature Ids to evaluate this rule against. │ │ │ │ │ + * To be passed inside the params object. │ │ │ │ │ + */ │ │ │ │ │ + fids: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: type │ │ │ │ │ + * {String} Type to identify this filter. │ │ │ │ │ + */ │ │ │ │ │ + type: "FID", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Filter.FeatureId │ │ │ │ │ + * Creates an ogc:FeatureId rule. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object with properties to set on the │ │ │ │ │ + * rule │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter.FeatureId>} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.fids = []; │ │ │ │ │ + OpenLayers.Filter.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: evaluate │ │ │ │ │ + * evaluates this rule for a specific feature │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature>} feature to apply the rule to. │ │ │ │ │ + * For vector features, the check is run against the fid, │ │ │ │ │ + * for plain features against the id. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the rule applies, false if it does not │ │ │ │ │ + */ │ │ │ │ │ + evaluate: function(feature) { │ │ │ │ │ + for (var i = 0, len = this.fids.length; i < len; i++) { │ │ │ │ │ + var fid = feature.fid || feature.id; │ │ │ │ │ + if (fid == this.fids[i]) { │ │ │ │ │ + return true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Clones this filter. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter.FeatureId>} Clone of this filter. │ │ │ │ │ + */ │ │ │ │ │ + clone: function() { │ │ │ │ │ + var filter = new OpenLayers.Filter.FeatureId(); │ │ │ │ │ + OpenLayers.Util.extend(filter, this); │ │ │ │ │ + filter.fids = this.fids.slice(); │ │ │ │ │ + return filter; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Filter.FeatureId" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WFST/v1.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Format/WFST.js │ │ │ │ │ + * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ + * @requires OpenLayers/Filter/FeatureId.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WFST.v1 │ │ │ │ │ + * Superclass for WFST parsers. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WFST.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + wfs: "http://www.opengis.net/wfs", │ │ │ │ │ + gml: "http://www.opengis.net/gml", │ │ │ │ │ + ogc: "http://www.opengis.net/ogc", │ │ │ │ │ + ows: "http://www.opengis.net/ows" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: "wfs", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} WFS version number. │ │ │ │ │ + */ │ │ │ │ │ + version: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} Schema location for a particular minor version. │ │ │ │ │ + */ │ │ │ │ │ + schemaLocations: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: srsName │ │ │ │ │ + * {String} URI for spatial reference system. │ │ │ │ │ + */ │ │ │ │ │ + srsName: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: extractAttributes │ │ │ │ │ + * {Boolean} Extract attributes from GML. Default is true. │ │ │ │ │ + */ │ │ │ │ │ + extractAttributes: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: xy │ │ │ │ │ + * {Boolean} Order of the GML coordinate true:(x,y) or false:(y,x) │ │ │ │ │ + * Changing is not recommended, a new Format should be instantiated. │ │ │ │ │ + */ │ │ │ │ │ + xy: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: stateName │ │ │ │ │ + * {Object} Maps feature states to node names. │ │ │ │ │ + */ │ │ │ │ │ + stateName: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WFST.v1 │ │ │ │ │ + * Instances of this class are not created directly. Use the │ │ │ │ │ + * <OpenLayers.Format.WFST.v1_0_0> or <OpenLayers.Format.WFST.v1_1_0> │ │ │ │ │ + * constructor instead. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + // set state name mapping │ │ │ │ │ + this.stateName = {}; │ │ │ │ │ + this.stateName[OpenLayers.State.INSERT] = "wfs:Insert"; │ │ │ │ │ + this.stateName[OpenLayers.State.UPDATE] = "wfs:Update"; │ │ │ │ │ + this.stateName[OpenLayers.State.DELETE] = "wfs:Delete"; │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getSrsName │ │ │ │ │ + */ │ │ │ │ │ + getSrsName: function(feature, options) { │ │ │ │ │ + var srsName = options && options.srsName; │ │ │ │ │ + if (!srsName) { │ │ │ │ │ + if (feature && feature.layer) { │ │ │ │ │ + srsName = feature.layer.projection.getCode(); │ │ │ │ │ + } else { │ │ │ │ │ + srsName = this.srsName; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return srsName; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Parse the response from a transaction. Because WFS is split into │ │ │ │ │ + * Transaction requests (create, update, and delete) and GetFeature │ │ │ │ │ + * requests (read), this method handles parsing of both types of │ │ │ │ │ + * responses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String | Document} The WFST document to read │ │ │ │ │ + * options - {Object} Options for the reader │ │ │ │ │ + * │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * output - {String} either "features" or "object". The default is │ │ │ │ │ + * "features", which means that the method will return an array of │ │ │ │ │ + * features. If set to "object", an object with a "features" property │ │ │ │ │ + * and other properties read by the parser will be returned. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array | Object} Output depending on the output option. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, { │ │ │ │ │ + output: "features" │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + var obj = {}; │ │ │ │ │ + if (data) { │ │ │ │ │ + this.readNode(data, obj, true); │ │ │ │ │ + } │ │ │ │ │ + if (obj.features && options.output === "features") { │ │ │ │ │ + obj = obj.features; │ │ │ │ │ + } │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wfs": { │ │ │ │ │ + "FeatureCollection": function(node, obj) { │ │ │ │ │ + obj.features = []; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: write │ │ │ │ │ + * Given an array of features, write a WFS transaction. This assumes │ │ │ │ │ + * the features have a state property that determines the operation │ │ │ │ │ + * type - insert, update, or delete. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} A list of features. See │ │ │ │ │ + * below for a more detailed description of the influence of the │ │ │ │ │ + * feature's *modified* property. │ │ │ │ │ + * options - {Object} │ │ │ │ │ + * │ │ │ │ │ + * feature.modified rules: │ │ │ │ │ + * If a feature has a modified property set, the following checks will be │ │ │ │ │ + * made before a feature's geometry or attribute is included in an Update │ │ │ │ │ + * transaction: │ │ │ │ │ + * - *modified* is not set at all: The geometry and all attributes will be │ │ │ │ │ + * included. │ │ │ │ │ + * - *modified.geometry* is set (null or a geometry): The geometry will be │ │ │ │ │ + * included. If *modified.attributes* is not set, all attributes will │ │ │ │ │ + * be included. │ │ │ │ │ + * - *modified.attributes* is set: Only the attributes set (i.e. to null or │ │ │ │ │ + * a value) in *modified.attributes* will be included. │ │ │ │ │ + * If *modified.geometry* is not set, the geometry will not be included. │ │ │ │ │ + * │ │ │ │ │ + * Valid options include: │ │ │ │ │ + * - *multi* {Boolean} If set to true, geometries will be casted to │ │ │ │ │ + * Multi geometries before writing. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A serialized WFS transaction. │ │ │ │ │ + */ │ │ │ │ │ + write: function(features, options) { │ │ │ │ │ + var node = this.writeNode("wfs:Transaction", { │ │ │ │ │ + features: features, │ │ │ │ │ + options: options │ │ │ │ │ + }); │ │ │ │ │ + var value = this.schemaLocationAttr(); │ │ │ │ │ + if (value) { │ │ │ │ │ + this.setAttributeNS( │ │ │ │ │ + node, this.namespaces["xsi"], "xsi:schemaLocation", value │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [node]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ + */ │ │ │ │ │ + writers: { │ │ │ │ │ + "wfs": { │ │ │ │ │ + "GetFeature": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("wfs:GetFeature", { │ │ │ │ │ + attributes: { │ │ │ │ │ + service: "WFS", │ │ │ │ │ + version: this.version, │ │ │ │ │ + handle: options && options.handle, │ │ │ │ │ + outputFormat: options && options.outputFormat, │ │ │ │ │ + maxFeatures: options && options.maxFeatures, │ │ │ │ │ + "xsi:schemaLocation": this.schemaLocationAttr(options) │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (typeof this.featureType == "string") { │ │ │ │ │ + this.writeNode("Query", options, node); │ │ │ │ │ + } else { │ │ │ │ │ + for (var i = 0, len = this.featureType.length; i < len; i++) { │ │ │ │ │ + options.featureType = this.featureType[i]; │ │ │ │ │ + this.writeNode("Query", options, node); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Transaction": function(obj) { │ │ │ │ │ + obj = obj || {}; │ │ │ │ │ + var options = obj.options || {}; │ │ │ │ │ + var node = this.createElementNSPlus("wfs:Transaction", { │ │ │ │ │ + attributes: { │ │ │ │ │ + service: "WFS", │ │ │ │ │ + version: this.version, │ │ │ │ │ + handle: options.handle │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + var i, len; │ │ │ │ │ + var features = obj.features; │ │ │ │ │ + if (features) { │ │ │ │ │ + // temporarily re-assigning geometry types │ │ │ │ │ + if (options.multi === true) { │ │ │ │ │ + OpenLayers.Util.extend(this.geometryTypes, { │ │ │ │ │ + "OpenLayers.Geometry.Point": "MultiPoint", │ │ │ │ │ + "OpenLayers.Geometry.LineString": (this.multiCurve === true) ? "MultiCurve" : "MultiLineString", │ │ │ │ │ + "OpenLayers.Geometry.Polygon": (this.multiSurface === true) ? "MultiSurface" : "MultiPolygon" │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + var name, feature; │ │ │ │ │ + for (i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + name = this.stateName[feature.state]; │ │ │ │ │ + if (name) { │ │ │ │ │ + this.writeNode(name, { │ │ │ │ │ + feature: feature, │ │ │ │ │ + options: options │ │ │ │ │ + }, node); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // switch back to original geometry types assignment │ │ │ │ │ + if (options.multi === true) { │ │ │ │ │ + this.setGeometryTypes(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (options.nativeElements) { │ │ │ │ │ + for (i = 0, len = options.nativeElements.length; i < len; ++i) { │ │ │ │ │ + this.writeNode("wfs:Native", │ │ │ │ │ + options.nativeElements[i], node); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Native": function(nativeElement) { │ │ │ │ │ + var node = this.createElementNSPlus("wfs:Native", { │ │ │ │ │ + attributes: { │ │ │ │ │ + vendorId: nativeElement.vendorId, │ │ │ │ │ + safeToIgnore: nativeElement.safeToIgnore │ │ │ │ │ + }, │ │ │ │ │ + value: nativeElement.value │ │ │ │ │ + }); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Insert": function(obj) { │ │ │ │ │ + var feature = obj.feature; │ │ │ │ │ + var options = obj.options; │ │ │ │ │ + var node = this.createElementNSPlus("wfs:Insert", { │ │ │ │ │ + attributes: { │ │ │ │ │ + handle: options && options.handle │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + this.srsName = this.getSrsName(feature); │ │ │ │ │ + this.writeNode("feature:_typeName", feature, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Update": function(obj) { │ │ │ │ │ + var feature = obj.feature; │ │ │ │ │ + var options = obj.options; │ │ │ │ │ + var node = this.createElementNSPlus("wfs:Update", { │ │ │ │ │ + attributes: { │ │ │ │ │ + handle: options && options.handle, │ │ │ │ │ + typeName: (this.featureNS ? this.featurePrefix + ":" : "") + │ │ │ │ │ + this.featureType │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (this.featureNS) { │ │ │ │ │ + node.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // add in geometry │ │ │ │ │ + var modified = feature.modified; │ │ │ │ │ + if (this.geometryName !== null && (!modified || modified.geometry !== undefined)) { │ │ │ │ │ + this.srsName = this.getSrsName(feature); │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "Property", { │ │ │ │ │ + name: this.geometryName, │ │ │ │ │ + value: feature.geometry │ │ │ │ │ + }, node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // add in attributes │ │ │ │ │ + for (var key in feature.attributes) { │ │ │ │ │ + if (feature.attributes[key] !== undefined && │ │ │ │ │ + (!modified || !modified.attributes || │ │ │ │ │ + (modified.attributes && modified.attributes[key] !== undefined))) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "Property", { │ │ │ │ │ + name: key, │ │ │ │ │ + value: feature.attributes[key] │ │ │ │ │ + }, node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // add feature id filter │ │ │ │ │ + this.writeNode("ogc:Filter", new OpenLayers.Filter.FeatureId({ │ │ │ │ │ + fids: [feature.fid] │ │ │ │ │ + }), node); │ │ │ │ │ + │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Property": function(obj) { │ │ │ │ │ + var node = this.createElementNSPlus("wfs:Property"); │ │ │ │ │ + this.writeNode("Name", obj.name, node); │ │ │ │ │ + if (obj.value !== null) { │ │ │ │ │ + this.writeNode("Value", obj.value, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Name": function(name) { │ │ │ │ │ + return this.createElementNSPlus("wfs:Name", { │ │ │ │ │ + value: name │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "Value": function(obj) { │ │ │ │ │ + var node; │ │ │ │ │ + if (obj instanceof OpenLayers.Geometry) { │ │ │ │ │ + node = this.createElementNSPlus("wfs:Value"); │ │ │ │ │ + var geom = this.writeNode("feature:_geometry", obj).firstChild; │ │ │ │ │ + node.appendChild(geom); │ │ │ │ │ + } else { │ │ │ │ │ + node = this.createElementNSPlus("wfs:Value", { │ │ │ │ │ + value: obj │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Delete": function(obj) { │ │ │ │ │ + var feature = obj.feature; │ │ │ │ │ + var options = obj.options; │ │ │ │ │ + var node = this.createElementNSPlus("wfs:Delete", { │ │ │ │ │ + attributes: { │ │ │ │ │ + handle: options && options.handle, │ │ │ │ │ + typeName: (this.featureNS ? this.featurePrefix + ":" : "") + │ │ │ │ │ + this.featureType │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (this.featureNS) { │ │ │ │ │ + node.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ + } │ │ │ │ │ + this.writeNode("ogc:Filter", new OpenLayers.Filter.FeatureId({ │ │ │ │ │ + fids: [feature.fid] │ │ │ │ │ + }), node); │ │ │ │ │ + return node; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: schemaLocationAttr │ │ │ │ │ + * Generate the xsi:schemaLocation attribute value. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The xsi:schemaLocation attribute or undefined if none. │ │ │ │ │ + */ │ │ │ │ │ + schemaLocationAttr: function(options) { │ │ │ │ │ + options = OpenLayers.Util.extend({ │ │ │ │ │ + featurePrefix: this.featurePrefix, │ │ │ │ │ + schema: this.schema │ │ │ │ │ + }, options); │ │ │ │ │ + var schemaLocations = OpenLayers.Util.extend({}, this.schemaLocations); │ │ │ │ │ + if (options.schema) { │ │ │ │ │ + schemaLocations[options.featurePrefix] = options.schema; │ │ │ │ │ + } │ │ │ │ │ + var parts = []; │ │ │ │ │ + var uri; │ │ │ │ │ + for (var key in schemaLocations) { │ │ │ │ │ + uri = this.namespaces[key]; │ │ │ │ │ + if (uri) { │ │ │ │ │ + parts.push(uri + " " + schemaLocations[key]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var value = parts.join(" ") || undefined; │ │ │ │ │ + return value; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setFilterProperty │ │ │ │ │ + * Set the property of each spatial filter. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {<OpenLayers.Filter>} │ │ │ │ │ + */ │ │ │ │ │ + setFilterProperty: function(filter) { │ │ │ │ │ + if (filter.filters) { │ │ │ │ │ + for (var i = 0, len = filter.filters.length; i < len; ++i) { │ │ │ │ │ + OpenLayers.Format.WFST.v1.prototype.setFilterProperty.call(this, filter.filters[i]); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + if (filter instanceof OpenLayers.Filter.Spatial && !filter.property) { │ │ │ │ │ + // got a spatial filter without property, so set it │ │ │ │ │ + filter.property = this.geometryName; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFST.v1" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Geometry/MultiLineString.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Geometry/Collection.js │ │ │ │ │ + * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Geometry.MultiLineString │ │ │ │ │ + * A MultiLineString is a geometry with multiple <OpenLayers.Geometry.LineString> │ │ │ │ │ + * components. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Geometry.Collection> │ │ │ │ │ + * - <OpenLayers.Geometry> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Geometry.MultiLineString = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Geometry.Collection, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: componentTypes │ │ │ │ │ + * {Array(String)} An array of class names representing the types of │ │ │ │ │ + * components that the collection can include. A null value means the │ │ │ │ │ + * component types are not restricted. │ │ │ │ │ + */ │ │ │ │ │ + componentTypes: ["OpenLayers.Geometry.LineString"], │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Geometry.MultiLineString │ │ │ │ │ + * Constructor for a MultiLineString Geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * components - {Array(<OpenLayers.Geometry.LineString>)} │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: split │ │ │ │ │ + * Use this geometry (the source) to attempt to split a target geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} The target geometry. │ │ │ │ │ + * options - {Object} Properties of this object will be used to determine │ │ │ │ │ + * how the split is conducted. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * mutual - {Boolean} Split the source geometry in addition to the target │ │ │ │ │ + * geometry. Default is false. │ │ │ │ │ + * edge - {Boolean} Allow splitting when only edges intersect. Default is │ │ │ │ │ + * true. If false, a vertex on the source must be within the tolerance │ │ │ │ │ + * distance of the intersection to be considered a split. │ │ │ │ │ + * tolerance - {Number} If a non-null value is provided, intersections │ │ │ │ │ + * within the tolerance distance of an existing vertex on the source │ │ │ │ │ + * will be assumed to occur at the vertex. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} A list of geometries (of this same type as the target) that │ │ │ │ │ + * result from splitting the target with the source geometry. The │ │ │ │ │ + * source and target geometry will remain unmodified. If no split │ │ │ │ │ + * results, null will be returned. If mutual is true and a split │ │ │ │ │ + * results, return will be an array of two arrays - the first will be │ │ │ │ │ + * all geometries that result from splitting the source geometry and │ │ │ │ │ + * the second will be all geometries that result from splitting the │ │ │ │ │ + * target geometry. │ │ │ │ │ + */ │ │ │ │ │ + split: function(geometry, options) { │ │ │ │ │ + var results = null; │ │ │ │ │ + var mutual = options && options.mutual; │ │ │ │ │ + var splits, sourceLine, sourceLines, sourceSplit, targetSplit; │ │ │ │ │ + var sourceParts = []; │ │ │ │ │ + var targetParts = [geometry]; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ + sourceLine = this.components[i]; │ │ │ │ │ + sourceSplit = false; │ │ │ │ │ + for (var j = 0; j < targetParts.length; ++j) { │ │ │ │ │ + splits = sourceLine.split(targetParts[j], options); │ │ │ │ │ + if (splits) { │ │ │ │ │ + if (mutual) { │ │ │ │ │ + sourceLines = splits[0]; │ │ │ │ │ + for (var k = 0, klen = sourceLines.length; k < klen; ++k) { │ │ │ │ │ + if (k === 0 && sourceParts.length) { │ │ │ │ │ + sourceParts[sourceParts.length - 1].addComponent( │ │ │ │ │ + sourceLines[k] │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + sourceParts.push( │ │ │ │ │ + new OpenLayers.Geometry.MultiLineString([ │ │ │ │ │ + sourceLines[k] │ │ │ │ │ + ]) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + sourceSplit = true; │ │ │ │ │ + splits = splits[1]; │ │ │ │ │ + } │ │ │ │ │ + if (splits.length) { │ │ │ │ │ + // splice in new target parts │ │ │ │ │ + splits.unshift(j, 1); │ │ │ │ │ + Array.prototype.splice.apply(targetParts, splits); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!sourceSplit) { │ │ │ │ │ + // source line was not hit │ │ │ │ │ + if (sourceParts.length) { │ │ │ │ │ + // add line to existing multi │ │ │ │ │ + sourceParts[sourceParts.length - 1].addComponent( │ │ │ │ │ + sourceLine.clone() │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + // create a fresh multi │ │ │ │ │ + sourceParts = [ │ │ │ │ │ + new OpenLayers.Geometry.MultiLineString( │ │ │ │ │ + sourceLine.clone() │ │ │ │ │ + ) │ │ │ │ │ + ]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (sourceParts && sourceParts.length > 1) { │ │ │ │ │ + sourceSplit = true; │ │ │ │ │ + } else { │ │ │ │ │ + sourceParts = []; │ │ │ │ │ + } │ │ │ │ │ + if (targetParts && targetParts.length > 1) { │ │ │ │ │ + targetSplit = true; │ │ │ │ │ + } else { │ │ │ │ │ + targetParts = []; │ │ │ │ │ + } │ │ │ │ │ + if (sourceSplit || targetSplit) { │ │ │ │ │ + if (mutual) { │ │ │ │ │ + results = [sourceParts, targetParts]; │ │ │ │ │ + } else { │ │ │ │ │ + results = targetParts; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return results; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: splitWith │ │ │ │ │ + * Split this geometry (the target) with the given geometry (the source). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} A geometry used to split this │ │ │ │ │ + * geometry (the source). │ │ │ │ │ + * options - {Object} Properties of this object will be used to determine │ │ │ │ │ + * how the split is conducted. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * mutual - {Boolean} Split the source geometry in addition to the target │ │ │ │ │ + * geometry. Default is false. │ │ │ │ │ + * edge - {Boolean} Allow splitting when only edges intersect. Default is │ │ │ │ │ + * true. If false, a vertex on the source must be within the tolerance │ │ │ │ │ + * distance of the intersection to be considered a split. │ │ │ │ │ + * tolerance - {Number} If a non-null value is provided, intersections │ │ │ │ │ + * within the tolerance distance of an existing vertex on the source │ │ │ │ │ + * will be assumed to occur at the vertex. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} A list of geometries (of this same type as the target) that │ │ │ │ │ + * result from splitting the target with the source geometry. The │ │ │ │ │ + * source and target geometry will remain unmodified. If no split │ │ │ │ │ + * results, null will be returned. If mutual is true and a split │ │ │ │ │ + * results, return will be an array of two arrays - the first will be │ │ │ │ │ + * all geometries that result from splitting the source geometry and │ │ │ │ │ + * the second will be all geometries that result from splitting the │ │ │ │ │ + * target geometry. │ │ │ │ │ + */ │ │ │ │ │ + splitWith: function(geometry, options) { │ │ │ │ │ + var results = null; │ │ │ │ │ + var mutual = options && options.mutual; │ │ │ │ │ + var splits, targetLine, sourceLines, sourceSplit, targetSplit, sourceParts, targetParts; │ │ │ │ │ + if (geometry instanceof OpenLayers.Geometry.LineString) { │ │ │ │ │ + targetParts = []; │ │ │ │ │ + sourceParts = [geometry]; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ + targetSplit = false; │ │ │ │ │ + targetLine = this.components[i]; │ │ │ │ │ + for (var j = 0; j < sourceParts.length; ++j) { │ │ │ │ │ + splits = sourceParts[j].split(targetLine, options); │ │ │ │ │ + if (splits) { │ │ │ │ │ + if (mutual) { │ │ │ │ │ + sourceLines = splits[0]; │ │ │ │ │ + if (sourceLines.length) { │ │ │ │ │ + // splice in new source parts │ │ │ │ │ + sourceLines.unshift(j, 1); │ │ │ │ │ + Array.prototype.splice.apply(sourceParts, sourceLines); │ │ │ │ │ + j += sourceLines.length - 2; │ │ │ │ │ + } │ │ │ │ │ + splits = splits[1]; │ │ │ │ │ + if (splits.length === 0) { │ │ │ │ │ + splits = [targetLine.clone()]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + for (var k = 0, klen = splits.length; k < klen; ++k) { │ │ │ │ │ + if (k === 0 && targetParts.length) { │ │ │ │ │ + targetParts[targetParts.length - 1].addComponent( │ │ │ │ │ + splits[k] │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + targetParts.push( │ │ │ │ │ + new OpenLayers.Geometry.MultiLineString([ │ │ │ │ │ + splits[k] │ │ │ │ │ + ]) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + targetSplit = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!targetSplit) { │ │ │ │ │ + // target component was not hit │ │ │ │ │ + if (targetParts.length) { │ │ │ │ │ + // add it to any existing multi-line │ │ │ │ │ + targetParts[targetParts.length - 1].addComponent( │ │ │ │ │ + targetLine.clone() │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + // or start with a fresh multi-line │ │ │ │ │ + targetParts = [ │ │ │ │ │ + new OpenLayers.Geometry.MultiLineString([ │ │ │ │ │ + targetLine.clone() │ │ │ │ │ + ]) │ │ │ │ │ + ]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + results = geometry.split(this); │ │ │ │ │ + } │ │ │ │ │ + if (sourceParts && sourceParts.length > 1) { │ │ │ │ │ + sourceSplit = true; │ │ │ │ │ + } else { │ │ │ │ │ + sourceParts = []; │ │ │ │ │ + } │ │ │ │ │ + if (targetParts && targetParts.length > 1) { │ │ │ │ │ + targetSplit = true; │ │ │ │ │ + } else { │ │ │ │ │ + targetParts = []; │ │ │ │ │ + } │ │ │ │ │ + if (sourceSplit || targetSplit) { │ │ │ │ │ + if (mutual) { │ │ │ │ │ + results = [sourceParts, targetParts]; │ │ │ │ │ + } else { │ │ │ │ │ + results = targetParts; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return results; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Geometry.MultiLineString" │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Geometry/MultiPolygon.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Geometry/Collection.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Geometry.MultiPolygon │ │ │ │ │ + * MultiPolygon is a geometry with multiple <OpenLayers.Geometry.Polygon> │ │ │ │ │ + * components. Create a new instance with the <OpenLayers.Geometry.MultiPolygon> │ │ │ │ │ + * constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Geometry.Collection> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Geometry.MultiPolygon = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Geometry.Collection, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: componentTypes │ │ │ │ │ + * {Array(String)} An array of class names representing the types of │ │ │ │ │ + * components that the collection can include. A null value means the │ │ │ │ │ + * component types are not restricted. │ │ │ │ │ + */ │ │ │ │ │ + componentTypes: ["OpenLayers.Geometry.Polygon"], │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Geometry.MultiPolygon │ │ │ │ │ + * Create a new MultiPolygon geometry │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * components - {Array(<OpenLayers.Geometry.Polygon>)} An array of polygons │ │ │ │ │ + * used to generate the MultiPolygon │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Geometry.MultiPolygon" │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/GML.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Geometry/MultiPoint.js │ │ │ │ │ + * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ + * @requires OpenLayers/Geometry/MultiLineString.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ + * @requires OpenLayers/Geometry/MultiPolygon.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.GML │ │ │ │ │ + * Read/Write GML. Create a new instance with the <OpenLayers.Format.GML> │ │ │ │ │ + * constructor. Supports the GML simple features profile. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.GML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: featureNS │ │ │ │ │ + * {String} Namespace used for feature attributes. Default is │ │ │ │ │ + * "http://mapserver.gis.umn.edu/mapserver". │ │ │ │ │ + */ │ │ │ │ │ + featureNS: "http://mapserver.gis.umn.edu/mapserver", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: featurePrefix │ │ │ │ │ + * {String} Namespace alias (or prefix) for feature nodes. Default is │ │ │ │ │ + * "feature". │ │ │ │ │ + */ │ │ │ │ │ + featurePrefix: "feature", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: featureName │ │ │ │ │ + * {String} Element name for features. Default is "featureMember". │ │ │ │ │ + */ │ │ │ │ │ + featureName: "featureMember", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layerName │ │ │ │ │ + * {String} Name of data layer. Default is "features". │ │ │ │ │ + */ │ │ │ │ │ + layerName: "features", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: geometryName │ │ │ │ │ + * {String} Name of geometry element. Defaults to "geometry". │ │ │ │ │ + */ │ │ │ │ │ + geometryName: "geometry", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: collectionName │ │ │ │ │ + * {String} Name of featureCollection element. │ │ │ │ │ + */ │ │ │ │ │ + collectionName: "FeatureCollection", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: gmlns │ │ │ │ │ + * {String} GML Namespace. │ │ │ │ │ + */ │ │ │ │ │ + gmlns: "http://www.opengis.net/gml", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: extractAttributes │ │ │ │ │ + * {Boolean} Extract attributes from GML. │ │ │ │ │ + */ │ │ │ │ │ + extractAttributes: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: xy │ │ │ │ │ + * {Boolean} Order of the GML coordinate true:(x,y) or false:(y,x) │ │ │ │ │ + * Changing is not recommended, a new Format should be instantiated. │ │ │ │ │ + */ │ │ │ │ │ + xy: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.GML │ │ │ │ │ + * Create a new parser for GML. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + // compile regular expressions once instead of every time they are used │ │ │ │ │ + this.regExes = { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read data from a string, and return a list of features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + var featureNodes = this.getElementsByTagNameNS(data.documentElement, │ │ │ │ │ + this.gmlns, │ │ │ │ │ + this.featureName); │ │ │ │ │ + var features = []; │ │ │ │ │ + for (var i = 0; i < featureNodes.length; i++) { │ │ │ │ │ + var feature = this.parseFeature(featureNodes[i]); │ │ │ │ │ + if (feature) { │ │ │ │ │ + features.push(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return features; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseFeature │ │ │ │ │ + * This function is the core of the GML parsing code in OpenLayers. │ │ │ │ │ + * It creates the geometries that are then attached to the returned │ │ │ │ │ + * feature, and calls parseAttributes() to get attribute data out. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A GML feature node. │ │ │ │ │ + */ │ │ │ │ │ + parseFeature: function(node) { │ │ │ │ │ + // only accept one geometry per feature - look for highest "order" │ │ │ │ │ + var order = ["MultiPolygon", "Polygon", │ │ │ │ │ + "MultiLineString", "LineString", │ │ │ │ │ + "MultiPoint", "Point", "Envelope" │ │ │ │ │ + ]; │ │ │ │ │ + // FIXME: In case we parse a feature with no geometry, but boundedBy an Envelope, │ │ │ │ │ + // this code creates a geometry derived from the Envelope. This is not correct. │ │ │ │ │ + var type, nodeList, geometry, parser; │ │ │ │ │ + for (var i = 0; i < order.length; ++i) { │ │ │ │ │ + type = order[i]; │ │ │ │ │ + nodeList = this.getElementsByTagNameNS(node, this.gmlns, type); │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + // only deal with first geometry of this type │ │ │ │ │ + parser = this.parseGeometry[type.toLowerCase()]; │ │ │ │ │ + if (parser) { │ │ │ │ │ + geometry = parser.apply(this, [nodeList[0]]); │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry.transform(this.externalProjection, │ │ │ │ │ + this.internalProjection); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + throw new TypeError("Unsupported geometry type: " + type); │ │ │ │ │ + } │ │ │ │ │ + // stop looking for different geometry types │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var bounds; │ │ │ │ │ + var boxNodes = this.getElementsByTagNameNS(node, this.gmlns, "Box"); │ │ │ │ │ + for (i = 0; i < boxNodes.length; ++i) { │ │ │ │ │ + var boxNode = boxNodes[i]; │ │ │ │ │ + var box = this.parseGeometry["box"].apply(this, [boxNode]); │ │ │ │ │ + var parentNode = boxNode.parentNode; │ │ │ │ │ + var parentName = parentNode.localName || │ │ │ │ │ + parentNode.nodeName.split(":").pop(); │ │ │ │ │ + if (parentName === "boundedBy") { │ │ │ │ │ + bounds = box; │ │ │ │ │ + } else { │ │ │ │ │ + geometry = box.toGeometry(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // construct feature (optionally with attributes) │ │ │ │ │ + var attributes; │ │ │ │ │ + if (this.extractAttributes) { │ │ │ │ │ + attributes = this.parseAttributes(node); │ │ │ │ │ + } │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry, attributes); │ │ │ │ │ + feature.bounds = bounds; │ │ │ │ │ + │ │ │ │ │ + feature.gml = { │ │ │ │ │ + featureType: node.firstChild.nodeName.split(":")[1], │ │ │ │ │ + featureNS: node.firstChild.namespaceURI, │ │ │ │ │ + featureNSPrefix: node.firstChild.prefix │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + // assign fid - this can come from a "fid" or "id" attribute │ │ │ │ │ + var childNode = node.firstChild; │ │ │ │ │ + var fid; │ │ │ │ │ + while (childNode) { │ │ │ │ │ + if (childNode.nodeType == 1) { │ │ │ │ │ + fid = childNode.getAttribute("fid") || │ │ │ │ │ + childNode.getAttribute("id"); │ │ │ │ │ + if (fid) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + childNode = childNode.nextSibling; │ │ │ │ │ + } │ │ │ │ │ + feature.fid = fid; │ │ │ │ │ + return feature; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: parseGeometry │ │ │ │ │ + * Properties of this object are the functions that parse geometries based │ │ │ │ │ + * on their type. │ │ │ │ │ + */ │ │ │ │ │ + parseGeometry: { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseGeometry.point │ │ │ │ │ + * Given a GML node representing a point geometry, create an OpenLayers │ │ │ │ │ + * point geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A GML node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Point>} A point geometry. │ │ │ │ │ + */ │ │ │ │ │ + point: function(node) { │ │ │ │ │ + /** │ │ │ │ │ + * Three coordinate variations to consider: │ │ │ │ │ + * 1) <gml:pos>x y z</gml:pos> │ │ │ │ │ + * 2) <gml:coordinates>x, y, z</gml:coordinates> │ │ │ │ │ + * 3) <gml:coord><gml:X>x</gml:X><gml:Y>y</gml:Y></gml:coord> │ │ │ │ │ + */ │ │ │ │ │ + var nodeList, coordString; │ │ │ │ │ + var coords = []; │ │ │ │ │ + │ │ │ │ │ + // look for <gml:pos> │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, this.gmlns, "pos"); │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + coordString = nodeList[0].firstChild.nodeValue; │ │ │ │ │ + coordString = coordString.replace(this.regExes.trimSpace, ""); │ │ │ │ │ + coords = coordString.split(this.regExes.splitSpace); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // look for <gml:coordinates> │ │ │ │ │ + if (coords.length == 0) { │ │ │ │ │ + nodeList = this.getElementsByTagNameNS(node, this.gmlns, │ │ │ │ │ + "coordinates"); │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + coordString = nodeList[0].firstChild.nodeValue; │ │ │ │ │ + coordString = coordString.replace(this.regExes.removeSpace, │ │ │ │ │ + ""); │ │ │ │ │ + coords = coordString.split(","); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // look for <gml:coord> │ │ │ │ │ + if (coords.length == 0) { │ │ │ │ │ + nodeList = this.getElementsByTagNameNS(node, this.gmlns, │ │ │ │ │ + "coord"); │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + var xList = this.getElementsByTagNameNS(nodeList[0], │ │ │ │ │ + this.gmlns, "X"); │ │ │ │ │ + var yList = this.getElementsByTagNameNS(nodeList[0], │ │ │ │ │ + this.gmlns, "Y"); │ │ │ │ │ + if (xList.length > 0 && yList.length > 0) { │ │ │ │ │ + coords = [xList[0].firstChild.nodeValue, │ │ │ │ │ + yList[0].firstChild.nodeValue │ │ │ │ │ + ]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // preserve third dimension │ │ │ │ │ + if (coords.length == 2) { │ │ │ │ │ + coords[2] = null; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.xy) { │ │ │ │ │ + return new OpenLayers.Geometry.Point(coords[0], coords[1], │ │ │ │ │ + coords[2]); │ │ │ │ │ + } else { │ │ │ │ │ + return new OpenLayers.Geometry.Point(coords[1], coords[0], │ │ │ │ │ + coords[2]); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseGeometry.multipoint │ │ │ │ │ + * Given a GML node representing a multipoint geometry, create an │ │ │ │ │ + * OpenLayers multipoint geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A GML node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.MultiPoint>} A multipoint geometry. │ │ │ │ │ + */ │ │ │ │ │ + multipoint: function(node) { │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, this.gmlns, │ │ │ │ │ + "Point"); │ │ │ │ │ + var components = []; │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + var point; │ │ │ │ │ + for (var i = 0; i < nodeList.length; ++i) { │ │ │ │ │ + point = this.parseGeometry.point.apply(this, [nodeList[i]]); │ │ │ │ │ + if (point) { │ │ │ │ │ + components.push(point); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.MultiPoint(components); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseGeometry.linestring │ │ │ │ │ + * Given a GML node representing a linestring geometry, create an │ │ │ │ │ + * OpenLayers linestring geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A GML node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.LineString>} A linestring geometry. │ │ │ │ │ + */ │ │ │ │ │ + linestring: function(node, ring) { │ │ │ │ │ + /** │ │ │ │ │ + * Two coordinate variations to consider: │ │ │ │ │ + * 1) <gml:posList dimension="d">x0 y0 z0 x1 y1 z1</gml:posList> │ │ │ │ │ + * 2) <gml:coordinates>x0, y0, z0 x1, y1, z1</gml:coordinates> │ │ │ │ │ + */ │ │ │ │ │ + var nodeList, coordString; │ │ │ │ │ + var coords = []; │ │ │ │ │ + var points = []; │ │ │ │ │ + │ │ │ │ │ + // look for <gml:posList> │ │ │ │ │ + nodeList = this.getElementsByTagNameNS(node, this.gmlns, "posList"); │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + coordString = this.getChildValue(nodeList[0]); │ │ │ │ │ + coordString = coordString.replace(this.regExes.trimSpace, ""); │ │ │ │ │ + coords = coordString.split(this.regExes.splitSpace); │ │ │ │ │ + var dim = parseInt(nodeList[0].getAttribute("dimension")); │ │ │ │ │ + var j, x, y, z; │ │ │ │ │ + for (var i = 0; i < coords.length / dim; ++i) { │ │ │ │ │ + j = i * dim; │ │ │ │ │ + x = coords[j]; │ │ │ │ │ + y = coords[j + 1]; │ │ │ │ │ + z = (dim == 2) ? null : coords[j + 2]; │ │ │ │ │ + if (this.xy) { │ │ │ │ │ + points.push(new OpenLayers.Geometry.Point(x, y, z)); │ │ │ │ │ + } else { │ │ │ │ │ + points.push(new OpenLayers.Geometry.Point(y, x, z)); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // look for <gml:coordinates> │ │ │ │ │ + if (coords.length == 0) { │ │ │ │ │ + nodeList = this.getElementsByTagNameNS(node, this.gmlns, │ │ │ │ │ + "coordinates"); │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + coordString = this.getChildValue(nodeList[0]); │ │ │ │ │ + coordString = coordString.replace(this.regExes.trimSpace, │ │ │ │ │ + ""); │ │ │ │ │ + coordString = coordString.replace(this.regExes.trimComma, │ │ │ │ │ + ","); │ │ │ │ │ + var pointList = coordString.split(this.regExes.splitSpace); │ │ │ │ │ + for (var i = 0; i < pointList.length; ++i) { │ │ │ │ │ + coords = pointList[i].split(","); │ │ │ │ │ + if (coords.length == 2) { │ │ │ │ │ + coords[2] = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.xy) { │ │ │ │ │ + points.push(new OpenLayers.Geometry.Point(coords[0], │ │ │ │ │ + coords[1], │ │ │ │ │ + coords[2])); │ │ │ │ │ + } else { │ │ │ │ │ + points.push(new OpenLayers.Geometry.Point(coords[1], │ │ │ │ │ + coords[0], │ │ │ │ │ + coords[2])); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var line = null; │ │ │ │ │ + if (points.length != 0) { │ │ │ │ │ + if (ring) { │ │ │ │ │ + line = new OpenLayers.Geometry.LinearRing(points); │ │ │ │ │ + } else { │ │ │ │ │ + line = new OpenLayers.Geometry.LineString(points); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return line; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseGeometry.multilinestring │ │ │ │ │ + * Given a GML node representing a multilinestring geometry, create an │ │ │ │ │ + * OpenLayers multilinestring geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A GML node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.MultiLineString>} A multilinestring geometry. │ │ │ │ │ + */ │ │ │ │ │ + multilinestring: function(node) { │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, this.gmlns, │ │ │ │ │ + "LineString"); │ │ │ │ │ + var components = []; │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + var line; │ │ │ │ │ + for (var i = 0; i < nodeList.length; ++i) { │ │ │ │ │ + line = this.parseGeometry.linestring.apply(this, │ │ │ │ │ + [nodeList[i]]); │ │ │ │ │ + if (line) { │ │ │ │ │ + components.push(line); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.MultiLineString(components); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseGeometry.polygon │ │ │ │ │ + * Given a GML node representing a polygon geometry, create an │ │ │ │ │ + * OpenLayers polygon geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A GML node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Polygon>} A polygon geometry. │ │ │ │ │ + */ │ │ │ │ │ + polygon: function(node) { │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, this.gmlns, │ │ │ │ │ + "LinearRing"); │ │ │ │ │ + var components = []; │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + // this assumes exterior ring first, inner rings after │ │ │ │ │ + var ring; │ │ │ │ │ + for (var i = 0; i < nodeList.length; ++i) { │ │ │ │ │ + ring = this.parseGeometry.linestring.apply(this, │ │ │ │ │ + [nodeList[i], true]); │ │ │ │ │ + if (ring) { │ │ │ │ │ + components.push(ring); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.Polygon(components); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseGeometry.multipolygon │ │ │ │ │ + * Given a GML node representing a multipolygon geometry, create an │ │ │ │ │ + * OpenLayers multipolygon geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A GML node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.MultiPolygon>} A multipolygon geometry. │ │ │ │ │ + */ │ │ │ │ │ + multipolygon: function(node) { │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, this.gmlns, │ │ │ │ │ + "Polygon"); │ │ │ │ │ + var components = []; │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + var polygon; │ │ │ │ │ + for (var i = 0; i < nodeList.length; ++i) { │ │ │ │ │ + polygon = this.parseGeometry.polygon.apply(this, │ │ │ │ │ + [nodeList[i]]); │ │ │ │ │ + if (polygon) { │ │ │ │ │ + components.push(polygon); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.MultiPolygon(components); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + envelope: function(node) { │ │ │ │ │ + var components = []; │ │ │ │ │ + var coordString; │ │ │ │ │ + var envelope; │ │ │ │ │ + │ │ │ │ │ + var lpoint = this.getElementsByTagNameNS(node, this.gmlns, "lowerCorner"); │ │ │ │ │ + if (lpoint.length > 0) { │ │ │ │ │ + var coords = []; │ │ │ │ │ + │ │ │ │ │ + if (lpoint.length > 0) { │ │ │ │ │ + coordString = lpoint[0].firstChild.nodeValue; │ │ │ │ │ + coordString = coordString.replace(this.regExes.trimSpace, ""); │ │ │ │ │ + coords = coordString.split(this.regExes.splitSpace); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (coords.length == 2) { │ │ │ │ │ + coords[2] = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.xy) { │ │ │ │ │ + var lowerPoint = new OpenLayers.Geometry.Point(coords[0], coords[1], coords[2]); │ │ │ │ │ + } else { │ │ │ │ │ + var lowerPoint = new OpenLayers.Geometry.Point(coords[1], coords[0], coords[2]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var upoint = this.getElementsByTagNameNS(node, this.gmlns, "upperCorner"); │ │ │ │ │ + if (upoint.length > 0) { │ │ │ │ │ + var coords = []; │ │ │ │ │ + │ │ │ │ │ + if (upoint.length > 0) { │ │ │ │ │ + coordString = upoint[0].firstChild.nodeValue; │ │ │ │ │ + coordString = coordString.replace(this.regExes.trimSpace, ""); │ │ │ │ │ + coords = coordString.split(this.regExes.splitSpace); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (coords.length == 2) { │ │ │ │ │ + coords[2] = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.xy) { │ │ │ │ │ + var upperPoint = new OpenLayers.Geometry.Point(coords[0], coords[1], coords[2]); │ │ │ │ │ + } else { │ │ │ │ │ + var upperPoint = new OpenLayers.Geometry.Point(coords[1], coords[0], coords[2]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (lowerPoint && upperPoint) { │ │ │ │ │ + components.push(new OpenLayers.Geometry.Point(lowerPoint.x, lowerPoint.y)); │ │ │ │ │ + components.push(new OpenLayers.Geometry.Point(upperPoint.x, lowerPoint.y)); │ │ │ │ │ + components.push(new OpenLayers.Geometry.Point(upperPoint.x, upperPoint.y)); │ │ │ │ │ + components.push(new OpenLayers.Geometry.Point(lowerPoint.x, upperPoint.y)); │ │ │ │ │ + components.push(new OpenLayers.Geometry.Point(lowerPoint.x, lowerPoint.y)); │ │ │ │ │ + │ │ │ │ │ + var ring = new OpenLayers.Geometry.LinearRing(components); │ │ │ │ │ + envelope = new OpenLayers.Geometry.Polygon([ring]); │ │ │ │ │ + } │ │ │ │ │ + return envelope; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseGeometry.box │ │ │ │ │ + * Given a GML node representing a box geometry, create an │ │ │ │ │ + * OpenLayers.Bounds. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A GML node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} A bounds representing the box. │ │ │ │ │ + */ │ │ │ │ │ + box: function(node) { │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, this.gmlns, │ │ │ │ │ + "coordinates"); │ │ │ │ │ + var coordString; │ │ │ │ │ + var coords, beginPoint = null, │ │ │ │ │ + endPoint = null; │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + coordString = nodeList[0].firstChild.nodeValue; │ │ │ │ │ + coords = coordString.split(" "); │ │ │ │ │ + if (coords.length == 2) { │ │ │ │ │ + beginPoint = coords[0].split(","); │ │ │ │ │ + endPoint = coords[1].split(","); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (beginPoint !== null && endPoint !== null) { │ │ │ │ │ + return new OpenLayers.Bounds(parseFloat(beginPoint[0]), │ │ │ │ │ + parseFloat(beginPoint[1]), │ │ │ │ │ + parseFloat(endPoint[0]), │ │ │ │ │ + parseFloat(endPoint[1])); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseAttributes │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An attributes object. │ │ │ │ │ + */ │ │ │ │ │ + parseAttributes: function(node) { │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + // assume attributes are children of the first type 1 child │ │ │ │ │ + var childNode = node.firstChild; │ │ │ │ │ + var children, i, child, grandchildren, grandchild, name, value; │ │ │ │ │ + while (childNode) { │ │ │ │ │ + if (childNode.nodeType == 1) { │ │ │ │ │ + // attributes are type 1 children with one type 3 child │ │ │ │ │ + children = childNode.childNodes; │ │ │ │ │ + for (i = 0; i < children.length; ++i) { │ │ │ │ │ + child = children[i]; │ │ │ │ │ + if (child.nodeType == 1) { │ │ │ │ │ + grandchildren = child.childNodes; │ │ │ │ │ + if (grandchildren.length == 1) { │ │ │ │ │ + grandchild = grandchildren[0]; │ │ │ │ │ + if (grandchild.nodeType == 3 || │ │ │ │ │ + grandchild.nodeType == 4) { │ │ │ │ │ + name = (child.prefix) ? │ │ │ │ │ + child.nodeName.split(":")[1] : │ │ │ │ │ + child.nodeName; │ │ │ │ │ + value = grandchild.nodeValue.replace( │ │ │ │ │ + this.regExes.trimSpace, ""); │ │ │ │ │ + attributes[name] = value; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // If child has no childNodes (grandchildren), │ │ │ │ │ + // set an attribute with null value. │ │ │ │ │ + // e.g. <prefix:fieldname/> becomes │ │ │ │ │ + // {fieldname: null} │ │ │ │ │ + attributes[child.nodeName.split(":").pop()] = null; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + childNode = childNode.nextSibling; │ │ │ │ │ + } │ │ │ │ │ + return attributes; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Generate a GML document string given a list of features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} List of features to │ │ │ │ │ + * serialize into a string. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string representing the GML document. │ │ │ │ │ + */ │ │ │ │ │ + write: function(features) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ + } │ │ │ │ │ + var gml = this.createElementNS("http://www.opengis.net/wfs", │ │ │ │ │ + "wfs:" + this.collectionName); │ │ │ │ │ + for (var i = 0; i < features.length; i++) { │ │ │ │ │ + gml.appendChild(this.createFeatureXML(features[i])); │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [gml]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createFeatureXML │ │ │ │ │ + * Accept an OpenLayers.Feature.Vector, and build a GML node for it. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} The feature to be built as GML. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A node reprensting the feature in GML. │ │ │ │ │ + */ │ │ │ │ │ + createFeatureXML: function(feature) { │ │ │ │ │ + var geometry = feature.geometry; │ │ │ │ │ + var geometryNode = this.buildGeometryNode(geometry); │ │ │ │ │ + var geomContainer = this.createElementNS(this.featureNS, │ │ │ │ │ + this.featurePrefix + ":" + │ │ │ │ │ + this.geometryName); │ │ │ │ │ + geomContainer.appendChild(geometryNode); │ │ │ │ │ + var featureNode = this.createElementNS(this.gmlns, │ │ │ │ │ + "gml:" + this.featureName); │ │ │ │ │ + var featureContainer = this.createElementNS(this.featureNS, │ │ │ │ │ + this.featurePrefix + ":" + │ │ │ │ │ + this.layerName); │ │ │ │ │ + var fid = feature.fid || feature.id; │ │ │ │ │ + featureContainer.setAttribute("fid", fid); │ │ │ │ │ + featureContainer.appendChild(geomContainer); │ │ │ │ │ + for (var attr in feature.attributes) { │ │ │ │ │ + var attrText = this.createTextNode(feature.attributes[attr]); │ │ │ │ │ + var nodename = attr.substring(attr.lastIndexOf(":") + 1); │ │ │ │ │ + var attrContainer = this.createElementNS(this.featureNS, │ │ │ │ │ + this.featurePrefix + ":" + │ │ │ │ │ + nodename); │ │ │ │ │ + attrContainer.appendChild(attrText); │ │ │ │ │ + featureContainer.appendChild(attrContainer); │ │ │ │ │ + } │ │ │ │ │ + featureNode.appendChild(featureContainer); │ │ │ │ │ + return featureNode; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: buildGeometryNode │ │ │ │ │ + */ │ │ │ │ │ + buildGeometryNode: function(geometry) { │ │ │ │ │ + if (this.externalProjection && this.internalProjection) { │ │ │ │ │ + geometry = geometry.clone(); │ │ │ │ │ + geometry.transform(this.internalProjection, │ │ │ │ │ + this.externalProjection); │ │ │ │ │ + } │ │ │ │ │ + var className = geometry.CLASS_NAME; │ │ │ │ │ + var type = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ + var builder = this.buildGeometry[type.toLowerCase()]; │ │ │ │ │ + return builder.apply(this, [geometry]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: buildGeometry │ │ │ │ │ + * Object containing methods to do the actual geometry node building │ │ │ │ │ + * based on geometry type. │ │ │ │ │ + */ │ │ │ │ │ + buildGeometry: { │ │ │ │ │ + // TBD retrieve the srs from layer │ │ │ │ │ + // srsName is non-standard, so not including it until it's right. │ │ │ │ │ + // gml.setAttribute("srsName", │ │ │ │ │ + // "http://www.opengis.net/gml/srs/epsg.xml#4326"); │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.point │ │ │ │ │ + * Given an OpenLayers point geometry, create a GML point. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Point>} A point geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A GML point node. │ │ │ │ │ + */ │ │ │ │ │ + point: function(geometry) { │ │ │ │ │ + var gml = this.createElementNS(this.gmlns, "gml:Point"); │ │ │ │ │ + gml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + return gml; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.multipoint │ │ │ │ │ + * Given an OpenLayers multipoint geometry, create a GML multipoint. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.MultiPoint>} A multipoint geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A GML multipoint node. │ │ │ │ │ + */ │ │ │ │ │ + multipoint: function(geometry) { │ │ │ │ │ + var gml = this.createElementNS(this.gmlns, "gml:MultiPoint"); │ │ │ │ │ + var points = geometry.components; │ │ │ │ │ + var pointMember, pointGeom; │ │ │ │ │ + for (var i = 0; i < points.length; i++) { │ │ │ │ │ + pointMember = this.createElementNS(this.gmlns, │ │ │ │ │ + "gml:pointMember"); │ │ │ │ │ + pointGeom = this.buildGeometry.point.apply(this, │ │ │ │ │ + [points[i]]); │ │ │ │ │ + pointMember.appendChild(pointGeom); │ │ │ │ │ + gml.appendChild(pointMember); │ │ │ │ │ + } │ │ │ │ │ + return gml; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.linestring │ │ │ │ │ + * Given an OpenLayers linestring geometry, create a GML linestring. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.LineString>} A linestring geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A GML linestring node. │ │ │ │ │ + */ │ │ │ │ │ + linestring: function(geometry) { │ │ │ │ │ + var gml = this.createElementNS(this.gmlns, "gml:LineString"); │ │ │ │ │ + gml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + return gml; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.multilinestring │ │ │ │ │ + * Given an OpenLayers multilinestring geometry, create a GML │ │ │ │ │ + * multilinestring. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.MultiLineString>} A multilinestring │ │ │ │ │ + * geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A GML multilinestring node. │ │ │ │ │ + */ │ │ │ │ │ + multilinestring: function(geometry) { │ │ │ │ │ + var gml = this.createElementNS(this.gmlns, "gml:MultiLineString"); │ │ │ │ │ + var lines = geometry.components; │ │ │ │ │ + var lineMember, lineGeom; │ │ │ │ │ + for (var i = 0; i < lines.length; ++i) { │ │ │ │ │ + lineMember = this.createElementNS(this.gmlns, │ │ │ │ │ + "gml:lineStringMember"); │ │ │ │ │ + lineGeom = this.buildGeometry.linestring.apply(this, │ │ │ │ │ + [lines[i]]); │ │ │ │ │ + lineMember.appendChild(lineGeom); │ │ │ │ │ + gml.appendChild(lineMember); │ │ │ │ │ + } │ │ │ │ │ + return gml; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.linearring │ │ │ │ │ + * Given an OpenLayers linearring geometry, create a GML linearring. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.LinearRing>} A linearring geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A GML linearring node. │ │ │ │ │ + */ │ │ │ │ │ + linearring: function(geometry) { │ │ │ │ │ + var gml = this.createElementNS(this.gmlns, "gml:LinearRing"); │ │ │ │ │ + gml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + return gml; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.polygon │ │ │ │ │ + * Given an OpenLayers polygon geometry, create a GML polygon. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Polygon>} A polygon geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A GML polygon node. │ │ │ │ │ + */ │ │ │ │ │ + polygon: function(geometry) { │ │ │ │ │ + var gml = this.createElementNS(this.gmlns, "gml:Polygon"); │ │ │ │ │ + var rings = geometry.components; │ │ │ │ │ + var ringMember, ringGeom, type; │ │ │ │ │ + for (var i = 0; i < rings.length; ++i) { │ │ │ │ │ + type = (i == 0) ? "outerBoundaryIs" : "innerBoundaryIs"; │ │ │ │ │ + ringMember = this.createElementNS(this.gmlns, │ │ │ │ │ + "gml:" + type); │ │ │ │ │ + ringGeom = this.buildGeometry.linearring.apply(this, │ │ │ │ │ + [rings[i]]); │ │ │ │ │ + ringMember.appendChild(ringGeom); │ │ │ │ │ + gml.appendChild(ringMember); │ │ │ │ │ + } │ │ │ │ │ + return gml; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.multipolygon │ │ │ │ │ + * Given an OpenLayers multipolygon geometry, create a GML multipolygon. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.MultiPolygon>} A multipolygon │ │ │ │ │ + * geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A GML multipolygon node. │ │ │ │ │ + */ │ │ │ │ │ + multipolygon: function(geometry) { │ │ │ │ │ + var gml = this.createElementNS(this.gmlns, "gml:MultiPolygon"); │ │ │ │ │ + var polys = geometry.components; │ │ │ │ │ + var polyMember, polyGeom; │ │ │ │ │ + for (var i = 0; i < polys.length; ++i) { │ │ │ │ │ + polyMember = this.createElementNS(this.gmlns, │ │ │ │ │ + "gml:polygonMember"); │ │ │ │ │ + polyGeom = this.buildGeometry.polygon.apply(this, │ │ │ │ │ + [polys[i]]); │ │ │ │ │ + polyMember.appendChild(polyGeom); │ │ │ │ │ + gml.appendChild(polyMember); │ │ │ │ │ + } │ │ │ │ │ + return gml; │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.bounds │ │ │ │ │ + * Given an OpenLayers bounds, create a GML box. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Geometry.Bounds>} A bounds object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A GML box node. │ │ │ │ │ + */ │ │ │ │ │ + bounds: function(bounds) { │ │ │ │ │ + var gml = this.createElementNS(this.gmlns, "gml:Box"); │ │ │ │ │ + gml.appendChild(this.buildCoordinatesNode(bounds)); │ │ │ │ │ + return gml; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildCoordinates │ │ │ │ │ + * builds the coordinates XmlNode │ │ │ │ │ + * (code) │ │ │ │ │ + * <gml:coordinates decimal="." cs="," ts=" ">...</gml:coordinates> │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {XmlNode} created xmlNode │ │ │ │ │ + */ │ │ │ │ │ + buildCoordinatesNode: function(geometry) { │ │ │ │ │ + var coordinatesNode = this.createElementNS(this.gmlns, │ │ │ │ │ + "gml:coordinates"); │ │ │ │ │ + coordinatesNode.setAttribute("decimal", "."); │ │ │ │ │ + coordinatesNode.setAttribute("cs", ","); │ │ │ │ │ + coordinatesNode.setAttribute("ts", " "); │ │ │ │ │ + │ │ │ │ │ + var parts = []; │ │ │ │ │ + │ │ │ │ │ + if (geometry instanceof OpenLayers.Bounds) { │ │ │ │ │ + parts.push(geometry.left + "," + geometry.bottom); │ │ │ │ │ + parts.push(geometry.right + "," + geometry.top); │ │ │ │ │ + } else { │ │ │ │ │ + var points = (geometry.components) ? geometry.components : [geometry]; │ │ │ │ │ + for (var i = 0; i < points.length; i++) { │ │ │ │ │ + parts.push(points[i].x + "," + points[i].y); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var txtNode = this.createTextNode(parts.join(" ")); │ │ │ │ │ + coordinatesNode.appendChild(txtNode); │ │ │ │ │ + │ │ │ │ │ + return coordinatesNode; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.GML" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/GML/Base.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Format/GML.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Though required in the full build, if the GML format is excluded, we set │ │ │ │ │ + * the namespace here. │ │ │ │ │ + */ │ │ │ │ │ +if (!OpenLayers.Format.GML) { │ │ │ │ │ + OpenLayers.Format.GML = {}; │ │ │ │ │ +} │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.GML.Base │ │ │ │ │ + * Superclass for GML parsers. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.GML.Base = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + gml: "http://www.opengis.net/gml", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + wfs: "http://www.opengis.net/wfs" // this is a convenience for reading wfs:FeatureCollection │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: "gml", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} Schema location for a particular minor version. │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: featureType │ │ │ │ │ + * {Array(String) or String} The local (without prefix) feature typeName(s). │ │ │ │ │ + */ │ │ │ │ │ + featureType: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: featureNS │ │ │ │ │ + * {String} The feature namespace. Must be set in the options at │ │ │ │ │ + * construction. │ │ │ │ │ + */ │ │ │ │ │ + featureNS: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: geometry │ │ │ │ │ + * {String} Name of geometry element. Defaults to "geometry". If null, it │ │ │ │ │ + * will be set on <read> when the first geometry is parsed. │ │ │ │ │ + */ │ │ │ │ │ + geometryName: "geometry", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: extractAttributes │ │ │ │ │ + * {Boolean} Extract attributes from GML. Default is true. │ │ │ │ │ + */ │ │ │ │ │ + extractAttributes: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: srsName │ │ │ │ │ + * {String} URI for spatial reference system. This is optional for │ │ │ │ │ + * single part geometries and mandatory for collections and multis. │ │ │ │ │ + * If set, the srsName attribute will be written for all geometries. │ │ │ │ │ + * Default is null. │ │ │ │ │ + */ │ │ │ │ │ + srsName: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: xy │ │ │ │ │ + * {Boolean} Order of the GML coordinate true:(x,y) or false:(y,x) │ │ │ │ │ + * Changing is not recommended, a new Format should be instantiated. │ │ │ │ │ + */ │ │ │ │ │ + xy: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: geometryTypes │ │ │ │ │ + * {Object} Maps OpenLayers geometry class names to GML element names. │ │ │ │ │ + * Use <setGeometryTypes> before accessing this property. │ │ │ │ │ + */ │ │ │ │ │ + geometryTypes: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: singleFeatureType │ │ │ │ │ + * {Boolean} True if there is only 1 featureType, and not an array │ │ │ │ │ + * of featuretypes. │ │ │ │ │ + */ │ │ │ │ │ + singleFeatureType: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: autoConfig │ │ │ │ │ + * {Boolean} Indicates if the format was configured without a <featureNS>, │ │ │ │ │ + * but auto-configured <featureNS> and <featureType> during read. │ │ │ │ │ + * Subclasses making use of <featureType> auto-configuration should make │ │ │ │ │ + * the first call to the <readNode> method (usually in the read method) │ │ │ │ │ + * with true as 3rd argument, so the auto-configured featureType can be │ │ │ │ │ + * reset and the format can be reused for subsequent reads with data from │ │ │ │ │ + * different featureTypes. Set to false after read if you want to keep the │ │ │ │ │ + * auto-configured values. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ + */ │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g), │ │ │ │ │ + featureMember: (/^(.*:)?featureMembers?$/) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.GML.Base │ │ │ │ │ + * Instances of this class are not created directly. Use the │ │ │ │ │ + * <OpenLayers.Format.GML.v2> or <OpenLayers.Format.GML.v3> constructor │ │ │ │ │ + * instead. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + * │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * featureType - {Array(String) or String} Local (without prefix) feature │ │ │ │ │ + * typeName(s) (required for write). │ │ │ │ │ + * featureNS - {String} Feature namespace (required for write). │ │ │ │ │ + * geometryName - {String} Geometry element name (required for write). │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.setGeometryTypes(); │ │ │ │ │ + if (options && options.featureNS) { │ │ │ │ │ + this.setNamespace("feature", options.featureNS); │ │ │ │ │ + } │ │ │ │ │ + this.singleFeatureType = !options || (typeof options.featureType === "string"); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: read │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {DOMElement} A gml:featureMember element, a gml:featureMembers │ │ │ │ │ + * element, or an element containing either of the above at any level. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + var features = []; │ │ │ │ │ + this.readNode(data, { │ │ │ │ │ + features: features │ │ │ │ │ + }, true); │ │ │ │ │ + if (features.length == 0) { │ │ │ │ │ + // look for gml:featureMember elements │ │ │ │ │ + var elements = this.getElementsByTagNameNS( │ │ │ │ │ + data, this.namespaces.gml, "featureMember" │ │ │ │ │ + ); │ │ │ │ │ + if (elements.length) { │ │ │ │ │ + for (var i = 0, len = elements.length; i < len; ++i) { │ │ │ │ │ + this.readNode(elements[i], { │ │ │ │ │ + features: features │ │ │ │ │ + }, true); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // look for gml:featureMembers elements (this is v3, but does no harm here) │ │ │ │ │ + var elements = this.getElementsByTagNameNS( │ │ │ │ │ + data, this.namespaces.gml, "featureMembers" │ │ │ │ │ + ); │ │ │ │ │ + if (elements.length) { │ │ │ │ │ + // there can be only one │ │ │ │ │ + this.readNode(elements[0], { │ │ │ │ │ + features: features │ │ │ │ │ + }, true); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return features; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: readNode │ │ │ │ │ + * Shorthand for applying one of the named readers given the node │ │ │ │ │ + * namespace and local name. Readers take two args (node, obj) and │ │ │ │ │ + * generally extend or modify the second. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The node to be read (required). │ │ │ │ │ + * obj - {Object} The object to be modified (optional). │ │ │ │ │ + * first - {Boolean} Should be set to true for the first node read. This │ │ │ │ │ + * is usually the readNode call in the read method. Without this being │ │ │ │ │ + * set, auto-configured properties will stick on subsequent reads. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The input object, modified (or a new one if none was provided). │ │ │ │ │ + */ │ │ │ │ │ + readNode: function(node, obj, first) { │ │ │ │ │ + // on subsequent calls of format.read(), we want to reset auto- │ │ │ │ │ + // configured properties and auto-configure again. │ │ │ │ │ + if (first === true && this.autoConfig === true) { │ │ │ │ │ + this.featureType = null; │ │ │ │ │ + delete this.namespaceAlias[this.featureNS]; │ │ │ │ │ + delete this.namespaces["feature"]; │ │ │ │ │ + this.featureNS = null; │ │ │ │ │ + } │ │ │ │ │ + // featureType auto-configuration │ │ │ │ │ + if (!this.featureNS && (!(node.prefix in this.namespaces) && │ │ │ │ │ + node.parentNode.namespaceURI == this.namespaces["gml"] && │ │ │ │ │ + this.regExes.featureMember.test(node.parentNode.nodeName))) { │ │ │ │ │ + this.featureType = node.nodeName.split(":").pop(); │ │ │ │ │ + this.setNamespace("feature", node.namespaceURI); │ │ │ │ │ + this.featureNS = node.namespaceURI; │ │ │ │ │ + this.autoConfig = true; │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Format.XML.prototype.readNode.apply(this, [node, obj]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "gml": { │ │ │ │ │ + "_inherit": function(node, obj, container) { │ │ │ │ │ + // To be implemented by version specific parsers │ │ │ │ │ + }, │ │ │ │ │ + "featureMember": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "featureMembers": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "name": function(node, obj) { │ │ │ │ │ + obj.name = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "boundedBy": function(node, obj) { │ │ │ │ │ + var container = {}; │ │ │ │ │ + this.readChildNodes(node, container); │ │ │ │ │ + if (container.components && container.components.length > 0) { │ │ │ │ │ + obj.bounds = container.components[0]; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Point": function(node, container) { │ │ │ │ │ + var obj = { │ │ │ │ │ + points: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + if (!container.components) { │ │ │ │ │ + container.components = []; │ │ │ │ │ + } │ │ │ │ │ + container.components.push(obj.points[0]); │ │ │ │ │ + }, │ │ │ │ │ + "coordinates": function(node, obj) { │ │ │ │ │ + var str = this.getChildValue(node).replace( │ │ │ │ │ + this.regExes.trimSpace, "" │ │ │ │ │ + ); │ │ │ │ │ + str = str.replace(this.regExes.trimComma, ","); │ │ │ │ │ + var pointList = str.split(this.regExes.splitSpace); │ │ │ │ │ + var coords; │ │ │ │ │ + var numPoints = pointList.length; │ │ │ │ │ + var points = new Array(numPoints); │ │ │ │ │ + for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ + coords = pointList[i].split(","); │ │ │ │ │ + if (this.xy) { │ │ │ │ │ + points[i] = new OpenLayers.Geometry.Point( │ │ │ │ │ + coords[0], coords[1], coords[2] │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + points[i] = new OpenLayers.Geometry.Point( │ │ │ │ │ + coords[1], coords[0], coords[2] │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + obj.points = points; │ │ │ │ │ + }, │ │ │ │ │ + "coord": function(node, obj) { │ │ │ │ │ + var coord = {}; │ │ │ │ │ + this.readChildNodes(node, coord); │ │ │ │ │ + if (!obj.points) { │ │ │ │ │ + obj.points = []; │ │ │ │ │ + } │ │ │ │ │ + obj.points.push(new OpenLayers.Geometry.Point( │ │ │ │ │ + coord.x, coord.y, coord.z │ │ │ │ │ + )); │ │ │ │ │ + }, │ │ │ │ │ + "X": function(node, coord) { │ │ │ │ │ + coord.x = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Y": function(node, coord) { │ │ │ │ │ + coord.y = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Z": function(node, coord) { │ │ │ │ │ + coord.z = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "MultiPoint": function(node, container) { │ │ │ │ │ + var obj = { │ │ │ │ │ + components: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readers.gml._inherit.apply(this, [node, obj, container]); │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + container.components = [ │ │ │ │ │ + new OpenLayers.Geometry.MultiPoint(obj.components) │ │ │ │ │ + ]; │ │ │ │ │ + }, │ │ │ │ │ + "pointMember": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "LineString": function(node, container) { │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readers.gml._inherit.apply(this, [node, obj, container]); │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + if (!container.components) { │ │ │ │ │ + container.components = []; │ │ │ │ │ + } │ │ │ │ │ + container.components.push( │ │ │ │ │ + new OpenLayers.Geometry.LineString(obj.points) │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + "MultiLineString": function(node, container) { │ │ │ │ │ + var obj = { │ │ │ │ │ + components: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readers.gml._inherit.apply(this, [node, obj, container]); │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + container.components = [ │ │ │ │ │ + new OpenLayers.Geometry.MultiLineString(obj.components) │ │ │ │ │ + ]; │ │ │ │ │ + }, │ │ │ │ │ + "lineStringMember": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "Polygon": function(node, container) { │ │ │ │ │ + var obj = { │ │ │ │ │ + outer: null, │ │ │ │ │ + inner: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readers.gml._inherit.apply(this, [node, obj, container]); │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + obj.inner.unshift(obj.outer); │ │ │ │ │ + if (!container.components) { │ │ │ │ │ + container.components = []; │ │ │ │ │ + } │ │ │ │ │ + container.components.push( │ │ │ │ │ + new OpenLayers.Geometry.Polygon(obj.inner) │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + "LinearRing": function(node, obj) { │ │ │ │ │ + var container = {}; │ │ │ │ │ + this.readers.gml._inherit.apply(this, [node, container]); │ │ │ │ │ + this.readChildNodes(node, container); │ │ │ │ │ + obj.components = [new OpenLayers.Geometry.LinearRing( │ │ │ │ │ + container.points │ │ │ │ │ + )]; │ │ │ │ │ + }, │ │ │ │ │ + "MultiPolygon": function(node, container) { │ │ │ │ │ + var obj = { │ │ │ │ │ + components: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readers.gml._inherit.apply(this, [node, obj, container]); │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + container.components = [ │ │ │ │ │ + new OpenLayers.Geometry.MultiPolygon(obj.components) │ │ │ │ │ + ]; │ │ │ │ │ + }, │ │ │ │ │ + "polygonMember": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "GeometryCollection": function(node, container) { │ │ │ │ │ + var obj = { │ │ │ │ │ + components: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readers.gml._inherit.apply(this, [node, obj, container]); │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + container.components = [ │ │ │ │ │ + new OpenLayers.Geometry.Collection(obj.components) │ │ │ │ │ + ]; │ │ │ │ │ + }, │ │ │ │ │ + "geometryMember": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "feature": { │ │ │ │ │ + "*": function(node, obj) { │ │ │ │ │ + // The node can either be named like the featureType, or it │ │ │ │ │ + // can be a child of the feature:featureType. Children can be │ │ │ │ │ + // geometry or attributes. │ │ │ │ │ + var name; │ │ │ │ │ + var local = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ + // Since an attribute can have the same name as the feature type │ │ │ │ │ + // we only want to read the node as a feature if the parent │ │ │ │ │ + // node can have feature nodes as children. In this case, the │ │ │ │ │ + // obj.features property is set. │ │ │ │ │ + if (obj.features) { │ │ │ │ │ + if (!this.singleFeatureType && │ │ │ │ │ + (OpenLayers.Util.indexOf(this.featureType, local) !== -1)) { │ │ │ │ │ + name = "_typeName"; │ │ │ │ │ + } else if (local === this.featureType) { │ │ │ │ │ + name = "_typeName"; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // Assume attribute elements have one child node and that the child │ │ │ │ │ + // is a text node. Otherwise assume it is a geometry node. │ │ │ │ │ + if (node.childNodes.length == 0 || │ │ │ │ │ + (node.childNodes.length == 1 && node.firstChild.nodeType == 3)) { │ │ │ │ │ + if (this.extractAttributes) { │ │ │ │ │ + name = "_attribute"; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + name = "_geometry"; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (name) { │ │ │ │ │ + this.readers.feature[name].apply(this, [node, obj]); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "_typeName": function(node, obj) { │ │ │ │ │ + var container = { │ │ │ │ │ + components: [], │ │ │ │ │ + attributes: {} │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, container); │ │ │ │ │ + // look for common gml namespaced elements │ │ │ │ │ + if (container.name) { │ │ │ │ │ + container.attributes.name = container.name; │ │ │ │ │ + } │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector( │ │ │ │ │ + container.components[0], container.attributes │ │ │ │ │ + ); │ │ │ │ │ + if (!this.singleFeatureType) { │ │ │ │ │ + feature.type = node.nodeName.split(":").pop(); │ │ │ │ │ + feature.namespace = node.namespaceURI; │ │ │ │ │ + } │ │ │ │ │ + var fid = node.getAttribute("fid") || │ │ │ │ │ + this.getAttributeNS(node, this.namespaces["gml"], "id"); │ │ │ │ │ + if (fid) { │ │ │ │ │ + feature.fid = fid; │ │ │ │ │ + } │ │ │ │ │ + if (this.internalProjection && this.externalProjection && │ │ │ │ │ + feature.geometry) { │ │ │ │ │ + feature.geometry.transform( │ │ │ │ │ + this.externalProjection, this.internalProjection │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (container.bounds) { │ │ │ │ │ + feature.bounds = container.bounds; │ │ │ │ │ + } │ │ │ │ │ + obj.features.push(feature); │ │ │ │ │ + }, │ │ │ │ │ + "_geometry": function(node, obj) { │ │ │ │ │ + if (!this.geometryName) { │ │ │ │ │ + this.geometryName = node.nodeName.split(":").pop(); │ │ │ │ │ + } │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "_attribute": function(node, obj) { │ │ │ │ │ + var local = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ + var value = this.getChildValue(node); │ │ │ │ │ + obj.attributes[local] = value; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "wfs": { │ │ │ │ │ + "FeatureCollection": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: write │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>) | OpenLayers.Feature.Vector} │ │ │ │ │ + * An array of features or a single feature. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} Given an array of features, a doc with a gml:featureMembers │ │ │ │ │ + * element will be returned. Given a single feature, a doc with a │ │ │ │ │ + * gml:featureMember element will be returned. │ │ │ │ │ + */ │ │ │ │ │ + write: function(features) { │ │ │ │ │ + var name; │ │ │ │ │ + if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ + name = "featureMembers"; │ │ │ │ │ + } else { │ │ │ │ │ + name = "featureMember"; │ │ │ │ │ + } │ │ │ │ │ + var root = this.writeNode("gml:" + name, features); │ │ │ │ │ + this.setAttributeNS( │ │ │ │ │ + root, this.namespaces["xsi"], │ │ │ │ │ + "xsi:schemaLocation", this.schemaLocation │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [root]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ + */ │ │ │ │ │ + writers: { │ │ │ │ │ + "gml": { │ │ │ │ │ + "featureMember": function(feature) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:featureMember"); │ │ │ │ │ + this.writeNode("feature:_typeName", feature, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "MultiPoint": function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:MultiPoint"); │ │ │ │ │ + var components = geometry.components || [geometry]; │ │ │ │ │ + for (var i = 0, ii = components.length; i < ii; ++i) { │ │ │ │ │ + this.writeNode("pointMember", components[i], node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "pointMember": function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:pointMember"); │ │ │ │ │ + this.writeNode("Point", geometry, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "MultiLineString": function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:MultiLineString"); │ │ │ │ │ + var components = geometry.components || [geometry]; │ │ │ │ │ + for (var i = 0, ii = components.length; i < ii; ++i) { │ │ │ │ │ + this.writeNode("lineStringMember", components[i], node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "lineStringMember": function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:lineStringMember"); │ │ │ │ │ + this.writeNode("LineString", geometry, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "MultiPolygon": function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:MultiPolygon"); │ │ │ │ │ + var components = geometry.components || [geometry]; │ │ │ │ │ + for (var i = 0, ii = components.length; i < ii; ++i) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "polygonMember", components[i], node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "polygonMember": function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:polygonMember"); │ │ │ │ │ + this.writeNode("Polygon", geometry, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "GeometryCollection": function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:GeometryCollection"); │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; ++i) { │ │ │ │ │ + this.writeNode("geometryMember", geometry.components[i], node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "geometryMember": function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:geometryMember"); │ │ │ │ │ + var child = this.writeNode("feature:_geometry", geometry); │ │ │ │ │ + node.appendChild(child.firstChild); │ │ │ │ │ + return node; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "feature": { │ │ │ │ │ + "_typeName": function(feature) { │ │ │ │ │ + var node = this.createElementNSPlus("feature:" + this.featureType, { │ │ │ │ │ + attributes: { │ │ │ │ │ + fid: feature.fid │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + this.writeNode("feature:_geometry", feature.geometry, node); │ │ │ │ │ + } │ │ │ │ │ + for (var name in feature.attributes) { │ │ │ │ │ + var value = feature.attributes[name]; │ │ │ │ │ + if (value != null) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "feature:_attribute", { │ │ │ │ │ + name: name, │ │ │ │ │ + value: value │ │ │ │ │ + }, node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "_geometry": function(geometry) { │ │ │ │ │ + if (this.externalProjection && this.internalProjection) { │ │ │ │ │ + geometry = geometry.clone().transform( │ │ │ │ │ + this.internalProjection, this.externalProjection │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + var node = this.createElementNSPlus( │ │ │ │ │ + "feature:" + this.geometryName │ │ │ │ │ + ); │ │ │ │ │ + var type = this.geometryTypes[geometry.CLASS_NAME]; │ │ │ │ │ + var child = this.writeNode("gml:" + type, geometry, node); │ │ │ │ │ + if (this.srsName) { │ │ │ │ │ + child.setAttribute("srsName", this.srsName); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "_attribute": function(obj) { │ │ │ │ │ + return this.createElementNSPlus("feature:" + obj.name, { │ │ │ │ │ + value: obj.value │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "wfs": { │ │ │ │ │ + "FeatureCollection": function(features) { │ │ │ │ │ + /** │ │ │ │ │ + * This is only here because GML2 only describes abstract │ │ │ │ │ + * feature collections. Typically, you would not be using │ │ │ │ │ + * the GML format to write wfs elements. This just provides │ │ │ │ │ + * some way to write out lists of features. GML3 defines the │ │ │ │ │ + * featureMembers element, so that is used by default instead. │ │ │ │ │ + */ │ │ │ │ │ + var node = this.createElementNSPlus("wfs:FeatureCollection"); │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + this.writeNode("gml:featureMember", features[i], node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setGeometryTypes │ │ │ │ │ + * Sets the <geometryTypes> mapping. │ │ │ │ │ + */ │ │ │ │ │ + setGeometryTypes: function() { │ │ │ │ │ + this.geometryTypes = { │ │ │ │ │ + "OpenLayers.Geometry.Point": "Point", │ │ │ │ │ + "OpenLayers.Geometry.MultiPoint": "MultiPoint", │ │ │ │ │ + "OpenLayers.Geometry.LineString": "LineString", │ │ │ │ │ + "OpenLayers.Geometry.MultiLineString": "MultiLineString", │ │ │ │ │ + "OpenLayers.Geometry.Polygon": "Polygon", │ │ │ │ │ + "OpenLayers.Geometry.MultiPolygon": "MultiPolygon", │ │ │ │ │ + "OpenLayers.Geometry.Collection": "GeometryCollection" │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.GML.Base" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/GML/v2.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/GML/Base.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.GML.v2 │ │ │ │ │ + * Parses GML version 2. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.GML.Base> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.GML.v2 = OpenLayers.Class(OpenLayers.Format.GML.Base, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} Schema location for a particular minor version. │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: "http://www.opengis.net/gml http://schemas.opengis.net/gml/2.1.2/feature.xsd", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.GML.v2 │ │ │ │ │ + * Create a parser for GML v2. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + * │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ + * featureNS - {String} Feature namespace (required). │ │ │ │ │ + * geometryName - {String} Geometry element name. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.GML.Base.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "outerBoundaryIs": function(node, container) { │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + container.outer = obj.components[0]; │ │ │ │ │ + }, │ │ │ │ │ + "innerBoundaryIs": function(node, container) { │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + container.inner.push(obj.components[0]); │ │ │ │ │ + }, │ │ │ │ │ + "Box": function(node, container) { │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + if (!container.components) { │ │ │ │ │ + container.components = []; │ │ │ │ │ + } │ │ │ │ │ + var min = obj.points[0]; │ │ │ │ │ + var max = obj.points[1]; │ │ │ │ │ + container.components.push( │ │ │ │ │ + new OpenLayers.Bounds(min.x, min.y, max.x, max.y) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.GML.Base.prototype.readers["gml"]), │ │ │ │ │ + "feature": OpenLayers.Format.GML.Base.prototype.readers["feature"], │ │ │ │ │ + "wfs": OpenLayers.Format.GML.Base.prototype.readers["wfs"] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: write │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>) | OpenLayers.Feature.Vector} │ │ │ │ │ + * An array of features or a single feature. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} Given an array of features, a doc with a gml:featureMembers │ │ │ │ │ + * element will be returned. Given a single feature, a doc with a │ │ │ │ │ + * gml:featureMember element will be returned. │ │ │ │ │ + */ │ │ │ │ │ + write: function(features) { │ │ │ │ │ + var name; │ │ │ │ │ + if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ + // GML2 only has abstract feature collections │ │ │ │ │ + // wfs provides a feature collection from a well-known schema │ │ │ │ │ + name = "wfs:FeatureCollection"; │ │ │ │ │ + } else { │ │ │ │ │ + name = "gml:featureMember"; │ │ │ │ │ + } │ │ │ │ │ + var root = this.writeNode(name, features); │ │ │ │ │ + this.setAttributeNS( │ │ │ │ │ + root, this.namespaces["xsi"], │ │ │ │ │ + "xsi:schemaLocation", this.schemaLocation │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [root]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ + */ │ │ │ │ │ + writers: { │ │ │ │ │ + "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "Point": function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:Point"); │ │ │ │ │ + this.writeNode("coordinates", [geometry], node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "coordinates": function(points) { │ │ │ │ │ + var numPoints = points.length; │ │ │ │ │ + var parts = new Array(numPoints); │ │ │ │ │ + var point; │ │ │ │ │ + for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ + point = points[i]; │ │ │ │ │ + if (this.xy) { │ │ │ │ │ + parts[i] = point.x + "," + point.y; │ │ │ │ │ + } else { │ │ │ │ │ + parts[i] = point.y + "," + point.x; │ │ │ │ │ + } │ │ │ │ │ + if (point.z != undefined) { // allow null or undefined │ │ │ │ │ + parts[i] += "," + point.z; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return this.createElementNSPlus("gml:coordinates", { │ │ │ │ │ + attributes: { │ │ │ │ │ + decimal: ".", │ │ │ │ │ + cs: ",", │ │ │ │ │ + ts: " " │ │ │ │ │ + }, │ │ │ │ │ + value: (numPoints == 1) ? parts[0] : parts.join(" ") │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "LineString": function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:LineString"); │ │ │ │ │ + this.writeNode("coordinates", geometry.components, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Polygon": function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:Polygon"); │ │ │ │ │ + this.writeNode("outerBoundaryIs", geometry.components[0], node); │ │ │ │ │ + for (var i = 1; i < geometry.components.length; ++i) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "innerBoundaryIs", geometry.components[i], node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "outerBoundaryIs": function(ring) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:outerBoundaryIs"); │ │ │ │ │ + this.writeNode("LinearRing", ring, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "innerBoundaryIs": function(ring) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:innerBoundaryIs"); │ │ │ │ │ + this.writeNode("LinearRing", ring, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "LinearRing": function(ring) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:LinearRing"); │ │ │ │ │ + this.writeNode("coordinates", ring.components, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Box": function(bounds) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:Box"); │ │ │ │ │ + this.writeNode("coordinates", [{ │ │ │ │ │ + x: bounds.left, │ │ │ │ │ + y: bounds.bottom │ │ │ │ │ + }, { │ │ │ │ │ + x: bounds.right, │ │ │ │ │ + y: bounds.top │ │ │ │ │ + }], node); │ │ │ │ │ + // srsName attribute is optional for gml:Box │ │ │ │ │ + if (this.srsName) { │ │ │ │ │ + node.setAttribute("srsName", this.srsName); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.GML.Base.prototype.writers["gml"]), │ │ │ │ │ + "feature": OpenLayers.Format.GML.Base.prototype.writers["feature"], │ │ │ │ │ + "wfs": OpenLayers.Format.GML.Base.prototype.writers["wfs"] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.GML.v2" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/OGCExceptionReport.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.OGCExceptionReport │ │ │ │ │ + * Class to read exception reports for various OGC services and versions. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.OGCExceptionReport = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + ogc: "http://www.opengis.net/ogc" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ + */ │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: "ogc", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.OGCExceptionReport │ │ │ │ │ + * Create a new parser for OGC exception reports. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read OGC exception report data from a string, and return an object with │ │ │ │ │ + * information about the exceptions. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Information about the exceptions that occurred. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + var result; │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + var root = data.documentElement; │ │ │ │ │ + var exceptionInfo = { │ │ │ │ │ + exceptionReport: null │ │ │ │ │ + }; │ │ │ │ │ + if (root) { │ │ │ │ │ + this.readChildNodes(data, exceptionInfo); │ │ │ │ │ + if (exceptionInfo.exceptionReport === null) { │ │ │ │ │ + // fall-back to OWSCommon since this is a common output format for exceptions │ │ │ │ │ + // we cannot easily use the ows readers directly since they differ for 1.0 and 1.1 │ │ │ │ │ + exceptionInfo = new OpenLayers.Format.OWSCommon().read(data); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return exceptionInfo; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "ogc": { │ │ │ │ │ + "ServiceExceptionReport": function(node, obj) { │ │ │ │ │ + obj.exceptionReport = { │ │ │ │ │ + exceptions: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.exceptionReport); │ │ │ │ │ + }, │ │ │ │ │ + "ServiceException": function(node, exceptionReport) { │ │ │ │ │ + var exception = { │ │ │ │ │ + code: node.getAttribute("code"), │ │ │ │ │ + locator: node.getAttribute("locator"), │ │ │ │ │ + text: this.getChildValue(node) │ │ │ │ │ + }; │ │ │ │ │ + exceptionReport.exceptions.push(exception); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.OGCExceptionReport" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Format/OGCExceptionReport.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.XML.VersionedOGC │ │ │ │ │ + * Base class for versioned formats, i.e. a format which supports multiple │ │ │ │ │ + * versions. │ │ │ │ │ + * │ │ │ │ │ + * To enable checking if parsing succeeded, you will need to define a property │ │ │ │ │ + * called errorProperty on the parser you want to check. The parser will then │ │ │ │ │ + * check the returned object to see if that property is present. If it is, it │ │ │ │ │ + * assumes the parsing was successful. If it is not present (or is null), it will │ │ │ │ │ + * pass the document through an OGCExceptionReport parser. │ │ │ │ │ + * │ │ │ │ │ + * If errorProperty is undefined for the parser, this error checking mechanism │ │ │ │ │ + * will be disabled. │ │ │ │ │ + * │ │ │ │ │ + * │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.XML.VersionedOGC = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. │ │ │ │ │ + */ │ │ │ │ │ + defaultVersion: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: version │ │ │ │ │ + * {String} Specify a version string if one is known. │ │ │ │ │ + */ │ │ │ │ │ + version: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: profile │ │ │ │ │ + * {String} If provided, use a custom profile. │ │ │ │ │ + */ │ │ │ │ │ + profile: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: allowFallback │ │ │ │ │ + * {Boolean} If a profiled parser cannot be found for the returned version, │ │ │ │ │ + * use a non-profiled parser as the fallback. Application code using this │ │ │ │ │ + * should take into account that the return object structure might be │ │ │ │ │ + * missing the specifics of the profile. Defaults to false. │ │ │ │ │ + */ │ │ │ │ │ + allowFallback: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: name │ │ │ │ │ + * {String} The name of this parser, this is the part of the CLASS_NAME │ │ │ │ │ + * except for "OpenLayers.Format." │ │ │ │ │ + */ │ │ │ │ │ + name: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: stringifyOutput │ │ │ │ │ + * {Boolean} If true, write will return a string otherwise a DOMElement. │ │ │ │ │ + * Default is false. │ │ │ │ │ + */ │ │ │ │ │ + stringifyOutput: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: parser │ │ │ │ │ + * {Object} Instance of the versioned parser. Cached for multiple read and │ │ │ │ │ + * write calls of the same version. │ │ │ │ │ + */ │ │ │ │ │ + parser: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.XML.VersionedOGC. │ │ │ │ │ + * Constructor. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on │ │ │ │ │ + * the object. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + var className = this.CLASS_NAME; │ │ │ │ │ + this.name = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getVersion │ │ │ │ │ + * Returns the version to use. Subclasses can override this function │ │ │ │ │ + * if a different version detection is needed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * root - {DOMElement} │ │ │ │ │ + * options - {Object} Optional configuration object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The version to use. │ │ │ │ │ + */ │ │ │ │ │ + getVersion: function(root, options) { │ │ │ │ │ + var version; │ │ │ │ │ + // read │ │ │ │ │ + if (root) { │ │ │ │ │ + version = this.version; │ │ │ │ │ + if (!version) { │ │ │ │ │ + version = root.getAttribute("version"); │ │ │ │ │ + if (!version) { │ │ │ │ │ + version = this.defaultVersion; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { // write │ │ │ │ │ + version = (options && options.version) || │ │ │ │ │ + this.version || this.defaultVersion; │ │ │ │ │ + } │ │ │ │ │ + return version; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getParser │ │ │ │ │ + * Get an instance of the cached parser if available, otherwise create one. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * version - {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Format>} │ │ │ │ │ + */ │ │ │ │ │ + getParser: function(version) { │ │ │ │ │ + version = version || this.defaultVersion; │ │ │ │ │ + var profile = this.profile ? "_" + this.profile : ""; │ │ │ │ │ + if (!this.parser || this.parser.VERSION != version) { │ │ │ │ │ + var format = OpenLayers.Format[this.name][ │ │ │ │ │ + "v" + version.replace(/\./g, "_") + profile │ │ │ │ │ + ]; │ │ │ │ │ + if (!format) { │ │ │ │ │ + if (profile !== "" && this.allowFallback) { │ │ │ │ │ + // fallback to the non-profiled version of the parser │ │ │ │ │ + profile = ""; │ │ │ │ │ + format = OpenLayers.Format[this.name][ │ │ │ │ │ + "v" + version.replace(/\./g, "_") │ │ │ │ │ + ]; │ │ │ │ │ + } │ │ │ │ │ + if (!format) { │ │ │ │ │ + throw "Can't find a " + this.name + " parser for version " + │ │ │ │ │ + version + profile; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.parser = new format(this.options); │ │ │ │ │ + } │ │ │ │ │ + return this.parser; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Write a document. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} An object representing the document. │ │ │ │ │ + * options - {Object} Optional configuration object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The document as a string │ │ │ │ │ + */ │ │ │ │ │ + write: function(obj, options) { │ │ │ │ │ + var version = this.getVersion(null, options); │ │ │ │ │ + this.parser = this.getParser(version); │ │ │ │ │ + var root = this.parser.write(obj, options); │ │ │ │ │ + if (this.stringifyOutput === false) { │ │ │ │ │ + return root; │ │ │ │ │ + } else { │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [root]); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read a doc and return an object representing the document. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String | DOMElement} Data to read. │ │ │ │ │ + * options - {Object} Options for the reader. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object representing the document. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data, options) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + var root = data.documentElement; │ │ │ │ │ + var version = this.getVersion(root); │ │ │ │ │ + this.parser = this.getParser(version); // Select the parser │ │ │ │ │ + var obj = this.parser.read(data, options); // Parse the data │ │ │ │ │ + │ │ │ │ │ + var errorProperty = this.parser.errorProperty || null; │ │ │ │ │ + if (errorProperty !== null && obj[errorProperty] === undefined) { │ │ │ │ │ + // an error must have happened, so parse it and report back │ │ │ │ │ + var format = new OpenLayers.Format.OGCExceptionReport(); │ │ │ │ │ + obj.error = format.read(data); │ │ │ │ │ + } │ │ │ │ │ + obj.version = version; │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.XML.VersionedOGC" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Filter/Logical.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Filter.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Filter.Logical │ │ │ │ │ + * This class represents ogc:And, ogc:Or and ogc:Not rules. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Filter> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Filter.Logical = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: filters │ │ │ │ │ + * {Array(<OpenLayers.Filter>)} Child filters for this filter. │ │ │ │ │ + */ │ │ │ │ │ + filters: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: type │ │ │ │ │ + * {String} type of logical operator. Available types are: │ │ │ │ │ + * - OpenLayers.Filter.Logical.AND = "&&"; │ │ │ │ │ + * - OpenLayers.Filter.Logical.OR = "||"; │ │ │ │ │ + * - OpenLayers.Filter.Logical.NOT = "!"; │ │ │ │ │ + */ │ │ │ │ │ + type: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Filter.Logical │ │ │ │ │ + * Creates a logical filter (And, Or, Not). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object with properties to set on the │ │ │ │ │ + * filter. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter.Logical>} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.filters = []; │ │ │ │ │ + OpenLayers.Filter.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Remove reference to child filters. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.filters = null; │ │ │ │ │ + OpenLayers.Filter.prototype.destroy.apply(this); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: evaluate │ │ │ │ │ + * Evaluates this filter in a specific context. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * context - {Object} Context to use in evaluating the filter. A vector │ │ │ │ │ + * feature may also be provided to evaluate feature attributes in │ │ │ │ │ + * comparison filters or geometries in spatial filters. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The filter applies. │ │ │ │ │ + */ │ │ │ │ │ + evaluate: function(context) { │ │ │ │ │ + var i, len; │ │ │ │ │ + switch (this.type) { │ │ │ │ │ + case OpenLayers.Filter.Logical.AND: │ │ │ │ │ + for (i = 0, len = this.filters.length; i < len; i++) { │ │ │ │ │ + if (this.filters[i].evaluate(context) == false) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ + │ │ │ │ │ + case OpenLayers.Filter.Logical.OR: │ │ │ │ │ + for (i = 0, len = this.filters.length; i < len; i++) { │ │ │ │ │ + if (this.filters[i].evaluate(context) == true) { │ │ │ │ │ + return true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return false; │ │ │ │ │ + │ │ │ │ │ + case OpenLayers.Filter.Logical.NOT: │ │ │ │ │ + return (!this.filters[0].evaluate(context)); │ │ │ │ │ + } │ │ │ │ │ + return undefined; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Clones this filter. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter.Logical>} Clone of this filter. │ │ │ │ │ + */ │ │ │ │ │ + clone: function() { │ │ │ │ │ + var filters = []; │ │ │ │ │ + for (var i = 0, len = this.filters.length; i < len; ++i) { │ │ │ │ │ + filters.push(this.filters[i].clone()); │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Filter.Logical({ │ │ │ │ │ + type: this.type, │ │ │ │ │ + filters: filters │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Filter.Logical" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Filter.Logical.AND = "&&"; │ │ │ │ │ +OpenLayers.Filter.Logical.OR = "||"; │ │ │ │ │ +OpenLayers.Filter.Logical.NOT = "!"; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Filter/Comparison.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Filter.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Filter.Comparison │ │ │ │ │ + * This class represents a comparison filter. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Filter> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Filter.Comparison = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: type │ │ │ │ │ + * {String} type: type of the comparison. This is one of │ │ │ │ │ + * - OpenLayers.Filter.Comparison.EQUAL_TO = "=="; │ │ │ │ │ + * - OpenLayers.Filter.Comparison.NOT_EQUAL_TO = "!="; │ │ │ │ │ + * - OpenLayers.Filter.Comparison.LESS_THAN = "<"; │ │ │ │ │ + * - OpenLayers.Filter.Comparison.GREATER_THAN = ">"; │ │ │ │ │ + * - OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO = "<="; │ │ │ │ │ + * - OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO = ">="; │ │ │ │ │ + * - OpenLayers.Filter.Comparison.BETWEEN = ".."; │ │ │ │ │ + * - OpenLayers.Filter.Comparison.LIKE = "~"; │ │ │ │ │ + * - OpenLayers.Filter.Comparison.IS_NULL = "NULL"; │ │ │ │ │ + */ │ │ │ │ │ + type: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: property │ │ │ │ │ + * {String} │ │ │ │ │ + * name of the context property to compare │ │ │ │ │ + */ │ │ │ │ │ + property: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: value │ │ │ │ │ + * {Number} or {String} │ │ │ │ │ + * comparison value for binary comparisons. In the case of a String, this │ │ │ │ │ + * can be a combination of text and propertyNames in the form │ │ │ │ │ + * "literal ${propertyName}" │ │ │ │ │ + */ │ │ │ │ │ + value: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: matchCase │ │ │ │ │ + * {Boolean} Force case sensitive searches for EQUAL_TO and NOT_EQUAL_TO │ │ │ │ │ + * comparisons. The Filter Encoding 1.1 specification added a matchCase │ │ │ │ │ + * attribute to ogc:PropertyIsEqualTo and ogc:PropertyIsNotEqualTo │ │ │ │ │ + * elements. This property will be serialized with those elements only │ │ │ │ │ + * if using the v1.1.0 filter format. However, when evaluating filters │ │ │ │ │ + * here, the matchCase property will always be respected (for EQUAL_TO │ │ │ │ │ + * and NOT_EQUAL_TO). Default is true. │ │ │ │ │ + */ │ │ │ │ │ + matchCase: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: lowerBoundary │ │ │ │ │ + * {Number} or {String} │ │ │ │ │ + * lower boundary for between comparisons. In the case of a String, this │ │ │ │ │ + * can be a combination of text and propertyNames in the form │ │ │ │ │ + * "literal ${propertyName}" │ │ │ │ │ + */ │ │ │ │ │ + lowerBoundary: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: upperBoundary │ │ │ │ │ + * {Number} or {String} │ │ │ │ │ + * upper boundary for between comparisons. In the case of a String, this │ │ │ │ │ + * can be a combination of text and propertyNames in the form │ │ │ │ │ + * "literal ${propertyName}" │ │ │ │ │ + */ │ │ │ │ │ + upperBoundary: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Filter.Comparison │ │ │ │ │ + * Creates a comparison rule. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object with properties to set on the │ │ │ │ │ + * rule │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter.Comparison>} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Filter.prototype.initialize.apply(this, [options]); │ │ │ │ │ + // since matchCase on PropertyIsLike is not schema compliant, we only │ │ │ │ │ + // want to use this if explicitly asked for │ │ │ │ │ + if (this.type === OpenLayers.Filter.Comparison.LIKE && │ │ │ │ │ + options.matchCase === undefined) { │ │ │ │ │ + this.matchCase = null; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: evaluate │ │ │ │ │ + * Evaluates this filter in a specific context. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * context - {Object} Context to use in evaluating the filter. If a vector │ │ │ │ │ + * feature is provided, the feature.attributes will be used as context. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The filter applies. │ │ │ │ │ + */ │ │ │ │ │ + evaluate: function(context) { │ │ │ │ │ + if (context instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ + context = context.attributes; │ │ │ │ │ + } │ │ │ │ │ + var result = false; │ │ │ │ │ + var got = context[this.property]; │ │ │ │ │ + var exp; │ │ │ │ │ + switch (this.type) { │ │ │ │ │ + case OpenLayers.Filter.Comparison.EQUAL_TO: │ │ │ │ │ + exp = this.value; │ │ │ │ │ + if (!this.matchCase && │ │ │ │ │ + typeof got == "string" && typeof exp == "string") { │ │ │ │ │ + result = (got.toUpperCase() == exp.toUpperCase()); │ │ │ │ │ + } else { │ │ │ │ │ + result = (got == exp); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Comparison.NOT_EQUAL_TO: │ │ │ │ │ + exp = this.value; │ │ │ │ │ + if (!this.matchCase && │ │ │ │ │ + typeof got == "string" && typeof exp == "string") { │ │ │ │ │ + result = (got.toUpperCase() != exp.toUpperCase()); │ │ │ │ │ + } else { │ │ │ │ │ + result = (got != exp); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Comparison.LESS_THAN: │ │ │ │ │ + result = got < this.value; │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Comparison.GREATER_THAN: │ │ │ │ │ + result = got > this.value; │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO: │ │ │ │ │ + result = got <= this.value; │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO: │ │ │ │ │ + result = got >= this.value; │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Comparison.BETWEEN: │ │ │ │ │ + result = (got >= this.lowerBoundary) && │ │ │ │ │ + (got <= this.upperBoundary); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Comparison.LIKE: │ │ │ │ │ + var regexp = new RegExp(this.value, "gi"); │ │ │ │ │ + result = regexp.test(got); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Comparison.IS_NULL: │ │ │ │ │ + result = (got === null); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + return result; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: value2regex │ │ │ │ │ + * Converts the value of this rule into a regular expression string, │ │ │ │ │ + * according to the wildcard characters specified. This method has to │ │ │ │ │ + * be called after instantiation of this class, if the value is not a │ │ │ │ │ + * regular expression already. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * wildCard - {Char} wildcard character in the above value, default │ │ │ │ │ + * is "*" │ │ │ │ │ + * singleChar - {Char} single-character wildcard in the above value │ │ │ │ │ + * default is "." │ │ │ │ │ + * escapeChar - {Char} escape character in the above value, default is │ │ │ │ │ + * "!" │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} regular expression string │ │ │ │ │ + */ │ │ │ │ │ + value2regex: function(wildCard, singleChar, escapeChar) { │ │ │ │ │ + if (wildCard == ".") { │ │ │ │ │ + throw new Error("'.' is an unsupported wildCard character for " + │ │ │ │ │ + "OpenLayers.Filter.Comparison"); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + // set UMN MapServer defaults for unspecified parameters │ │ │ │ │ + wildCard = wildCard ? wildCard : "*"; │ │ │ │ │ + singleChar = singleChar ? singleChar : "."; │ │ │ │ │ + escapeChar = escapeChar ? escapeChar : "!"; │ │ │ │ │ + │ │ │ │ │ + this.value = this.value.replace( │ │ │ │ │ + new RegExp("\\" + escapeChar + "(.|$)", "g"), "\\$1"); │ │ │ │ │ + this.value = this.value.replace( │ │ │ │ │ + new RegExp("\\" + singleChar, "g"), "."); │ │ │ │ │ + this.value = this.value.replace( │ │ │ │ │ + new RegExp("\\" + wildCard, "g"), ".*"); │ │ │ │ │ + this.value = this.value.replace( │ │ │ │ │ + new RegExp("\\\\.\\*", "g"), "\\" + wildCard); │ │ │ │ │ + this.value = this.value.replace( │ │ │ │ │ + new RegExp("\\\\\\.", "g"), "\\" + singleChar); │ │ │ │ │ + │ │ │ │ │ + return this.value; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: regex2value │ │ │ │ │ + * Convert the value of this rule from a regular expression string into an │ │ │ │ │ + * ogc literal string using a wildCard of *, a singleChar of ., and an │ │ │ │ │ + * escape of !. Leaves the <value> property unmodified. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string value. │ │ │ │ │ + */ │ │ │ │ │ + regex2value: function() { │ │ │ │ │ + │ │ │ │ │ + var value = this.value; │ │ │ │ │ + │ │ │ │ │ + // replace ! with !! │ │ │ │ │ + value = value.replace(/!/g, "!!"); │ │ │ │ │ + │ │ │ │ │ + // replace \. with !. (watching out for \\.) │ │ │ │ │ + value = value.replace(/(\\)?\\\./g, function($0, $1) { │ │ │ │ │ + return $1 ? $0 : "!."; │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + // replace \* with #* (watching out for \\*) │ │ │ │ │ + value = value.replace(/(\\)?\\\*/g, function($0, $1) { │ │ │ │ │ + return $1 ? $0 : "!*"; │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + // replace \\ with \ │ │ │ │ │ + value = value.replace(/\\\\/g, "\\"); │ │ │ │ │ + │ │ │ │ │ + // convert .* to * (the sequence #.* is not allowed) │ │ │ │ │ + value = value.replace(/\.\*/g, "*"); │ │ │ │ │ + │ │ │ │ │ + return value; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Clones this filter. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter.Comparison>} Clone of this filter. │ │ │ │ │ + */ │ │ │ │ │ + clone: function() { │ │ │ │ │ + return OpenLayers.Util.extend(new OpenLayers.Filter.Comparison(), this); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Filter.Comparison" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Filter.Comparison.EQUAL_TO = "=="; │ │ │ │ │ +OpenLayers.Filter.Comparison.NOT_EQUAL_TO = "!="; │ │ │ │ │ +OpenLayers.Filter.Comparison.LESS_THAN = "<"; │ │ │ │ │ +OpenLayers.Filter.Comparison.GREATER_THAN = ">"; │ │ │ │ │ +OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO = "<="; │ │ │ │ │ +OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO = ">="; │ │ │ │ │ +OpenLayers.Filter.Comparison.BETWEEN = ".."; │ │ │ │ │ +OpenLayers.Filter.Comparison.LIKE = "~"; │ │ │ │ │ +OpenLayers.Filter.Comparison.IS_NULL = "NULL"; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/Filter.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + * @requires OpenLayers/Filter/FeatureId.js │ │ │ │ │ + * @requires OpenLayers/Filter/Logical.js │ │ │ │ │ + * @requires OpenLayers/Filter/Comparison.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.Filter │ │ │ │ │ + * Read/Write ogc:Filter. Create a new instance with the <OpenLayers.Format.Filter> │ │ │ │ │ + * constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.Filter = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.0.0". │ │ │ │ │ + */ │ │ │ │ │ + defaultVersion: "1.0.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Write an ogc:Filter given a filter object. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {<OpenLayers.Filter>} An filter. │ │ │ │ │ + * options - {Object} Optional configuration object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Elment} An ogc:Filter element node. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read and Filter doc and return an object representing the Filter. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String | DOMElement} Data to read. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter>} A filter object. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.Filter" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Filter/Function.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Filter.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Filter.Function │ │ │ │ │ + * This class represents a filter function. │ │ │ │ │ + * We are using this class for creation of complex │ │ │ │ │ + * filters that can contain filter functions as values. │ │ │ │ │ + * Nesting function as other functions parameter is supported. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Filter> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Filter.Function = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: name │ │ │ │ │ + * {String} Name of the function. │ │ │ │ │ + */ │ │ │ │ │ + name: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: params │ │ │ │ │ + * {Array(<OpenLayers.Filter.Function> || String || Number)} Function parameters │ │ │ │ │ + * For now support only other Functions, String or Number │ │ │ │ │ + */ │ │ │ │ │ + params: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Filter.Function │ │ │ │ │ + * Creates a filter function. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object with properties to set on the │ │ │ │ │ + * function. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter.Function>} │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Filter.Function" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/BaseTypes/Date.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/SingleFile.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Date │ │ │ │ │ + * Contains implementations of Date.parse and date.toISOString that match the │ │ │ │ │ + * ECMAScript 5 specification for parsing RFC 3339 dates. │ │ │ │ │ + * http://tools.ietf.org/html/rfc3339 │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Date = { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: dateRegEx │ │ │ │ │ + * The regex to be used for validating dates. You can provide your own │ │ │ │ │ + * regex for instance for adding support for years before BC. Default │ │ │ │ │ + * value is: /^(?:(\d{4})(?:-(\d{2})(?:-(\d{2}))?)?)?(?:(?:T(\d{1,2}):(\d{2}):(\d{2}(?:\.\d+)?)(Z|(?:[+-]\d{1,2}(?::(\d{2}))?)))|Z)?$/ │ │ │ │ │ + */ │ │ │ │ │ + dateRegEx: /^(?:(\d{4})(?:-(\d{2})(?:-(\d{2}))?)?)?(?:(?:T(\d{1,2}):(\d{2}):(\d{2}(?:\.\d+)?)(Z|(?:[+-]\d{1,2}(?::(\d{2}))?)))|Z)?$/, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: toISOString │ │ │ │ │ + * Generates a string representing a date. The format of the string follows │ │ │ │ │ + * the profile of ISO 8601 for date and time on the Internet (see │ │ │ │ │ + * http://tools.ietf.org/html/rfc3339). If the toISOString method is │ │ │ │ │ + * available on the Date prototype, that is used. The toISOString │ │ │ │ │ + * method for Date instances is defined in ECMA-262. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * date - {Date} A date object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string representing the date (e.g. │ │ │ │ │ + * "2010-08-07T16:58:23.123Z"). If the date does not have a valid time │ │ │ │ │ + * (i.e. isNaN(date.getTime())) this method returns the string "Invalid │ │ │ │ │ + * Date". The ECMA standard says the toISOString method should throw │ │ │ │ │ + * RangeError in this case, but Firefox returns a string instead. For │ │ │ │ │ + * best results, use isNaN(date.getTime()) to determine date validity │ │ │ │ │ + * before generating date strings. │ │ │ │ │ + */ │ │ │ │ │ + toISOString: (function() { │ │ │ │ │ + if ("toISOString" in Date.prototype) { │ │ │ │ │ + return function(date) { │ │ │ │ │ + return date.toISOString(); │ │ │ │ │ + }; │ │ │ │ │ + } else { │ │ │ │ │ + return function(date) { │ │ │ │ │ + var str; │ │ │ │ │ + if (isNaN(date.getTime())) { │ │ │ │ │ + // ECMA-262 says throw RangeError, Firefox returns │ │ │ │ │ + // "Invalid Date" │ │ │ │ │ + str = "Invalid Date"; │ │ │ │ │ + } else { │ │ │ │ │ + str = │ │ │ │ │ + date.getUTCFullYear() + "-" + │ │ │ │ │ + OpenLayers.Number.zeroPad(date.getUTCMonth() + 1, 2) + "-" + │ │ │ │ │ + OpenLayers.Number.zeroPad(date.getUTCDate(), 2) + "T" + │ │ │ │ │ + OpenLayers.Number.zeroPad(date.getUTCHours(), 2) + ":" + │ │ │ │ │ + OpenLayers.Number.zeroPad(date.getUTCMinutes(), 2) + ":" + │ │ │ │ │ + OpenLayers.Number.zeroPad(date.getUTCSeconds(), 2) + "." + │ │ │ │ │ + OpenLayers.Number.zeroPad(date.getUTCMilliseconds(), 3) + "Z"; │ │ │ │ │ + } │ │ │ │ │ + return str; │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + })(), │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: parse │ │ │ │ │ + * Generate a date object from a string. The format for the string follows │ │ │ │ │ + * the profile of ISO 8601 for date and time on the Internet (see │ │ │ │ │ + * http://tools.ietf.org/html/rfc3339). We don't call the native │ │ │ │ │ + * Date.parse because of inconsistency between implmentations. In │ │ │ │ │ + * Chrome, calling Date.parse with a string that doesn't contain any │ │ │ │ │ + * indication of the timezone (e.g. "2011"), the date is interpreted │ │ │ │ │ + * in local time. On Firefox, the assumption is UTC. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * str - {String} A string representing the date (e.g. │ │ │ │ │ + * "2010", "2010-08", "2010-08-07", "2010-08-07T16:58:23.123Z", │ │ │ │ │ + * "2010-08-07T11:58:23.123-06"). │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Date} A date object. If the string could not be parsed, an invalid │ │ │ │ │ + * date is returned (i.e. isNaN(date.getTime())). │ │ │ │ │ + */ │ │ │ │ │ + parse: function(str) { │ │ │ │ │ + var date; │ │ │ │ │ + var match = str.match(this.dateRegEx); │ │ │ │ │ + if (match && (match[1] || match[7])) { // must have at least year or time │ │ │ │ │ + var year = parseInt(match[1], 10) || 0; │ │ │ │ │ + var month = (parseInt(match[2], 10) - 1) || 0; │ │ │ │ │ + var day = parseInt(match[3], 10) || 1; │ │ │ │ │ + date = new Date(Date.UTC(year, month, day)); │ │ │ │ │ + // optional time │ │ │ │ │ + var type = match[7]; │ │ │ │ │ + if (type) { │ │ │ │ │ + var hours = parseInt(match[4], 10); │ │ │ │ │ + var minutes = parseInt(match[5], 10); │ │ │ │ │ + var secFrac = parseFloat(match[6]); │ │ │ │ │ + var seconds = secFrac | 0; │ │ │ │ │ + var milliseconds = Math.round(1000 * (secFrac - seconds)); │ │ │ │ │ + date.setUTCHours(hours, minutes, seconds, milliseconds); │ │ │ │ │ + // check offset │ │ │ │ │ + if (type !== "Z") { │ │ │ │ │ + var hoursOffset = parseInt(type, 10); │ │ │ │ │ + var minutesOffset = parseInt(match[8], 10) || 0; │ │ │ │ │ + var offset = -1000 * (60 * (hoursOffset * 60) + minutesOffset * 60); │ │ │ │ │ + date = new Date(date.getTime() + offset); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + date = new Date("invalid"); │ │ │ │ │ + } │ │ │ │ │ + return date; │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/Filter/v1.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/Filter.js │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Filter/Function.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Date.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.Filter.v1 │ │ │ │ │ + * Superclass for Filter version 1 parsers. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.Filter.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + ogc: "http://www.opengis.net/ogc", │ │ │ │ │ + gml: "http://www.opengis.net/gml", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: "ogc", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} Schema location for a particular minor version. │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.Filter.v1 │ │ │ │ │ + * Instances of this class are not created directly. Use the │ │ │ │ │ + * <OpenLayers.Format.Filter> constructor instead. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: read │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {DOMElement} A Filter document element. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter>} A filter object. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readers.ogc["Filter"].apply(this, [data, obj]); │ │ │ │ │ + return obj.filter; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "ogc": { │ │ │ │ │ + "_expression": function(node) { │ │ │ │ │ + // only the simplest of ogc:expression handled │ │ │ │ │ + // "some text and an <PropertyName>attribute</PropertyName>"} │ │ │ │ │ + var obj, value = ""; │ │ │ │ │ + for (var child = node.firstChild; child; child = child.nextSibling) { │ │ │ │ │ + switch (child.nodeType) { │ │ │ │ │ + case 1: │ │ │ │ │ + obj = this.readNode(child); │ │ │ │ │ + if (obj.property) { │ │ │ │ │ + value += "${" + obj.property + "}"; │ │ │ │ │ + } else if (obj.value !== undefined) { │ │ │ │ │ + value += obj.value; │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case 3: // text node │ │ │ │ │ + case 4: // cdata section │ │ │ │ │ + value += child.nodeValue; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return value; │ │ │ │ │ + }, │ │ │ │ │ + "Filter": function(node, parent) { │ │ │ │ │ + // Filters correspond to subclasses of OpenLayers.Filter. │ │ │ │ │ + // Since they contain information we don't persist, we │ │ │ │ │ + // create a temporary object and then pass on the filter │ │ │ │ │ + // (ogc:Filter) to the parent obj. │ │ │ │ │ + var obj = { │ │ │ │ │ + fids: [], │ │ │ │ │ + filters: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + if (obj.fids.length > 0) { │ │ │ │ │ + parent.filter = new OpenLayers.Filter.FeatureId({ │ │ │ │ │ + fids: obj.fids │ │ │ │ │ + }); │ │ │ │ │ + } else if (obj.filters.length > 0) { │ │ │ │ │ + parent.filter = obj.filters[0]; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "FeatureId": function(node, obj) { │ │ │ │ │ + var fid = node.getAttribute("fid"); │ │ │ │ │ + if (fid) { │ │ │ │ │ + obj.fids.push(fid); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "And": function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ + type: OpenLayers.Filter.Logical.AND │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + obj.filters.push(filter); │ │ │ │ │ + }, │ │ │ │ │ + "Or": function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ + type: OpenLayers.Filter.Logical.OR │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + obj.filters.push(filter); │ │ │ │ │ + }, │ │ │ │ │ + "Not": function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ + type: OpenLayers.Filter.Logical.NOT │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + obj.filters.push(filter); │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsLessThan": function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.LESS_THAN │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + obj.filters.push(filter); │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsGreaterThan": function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.GREATER_THAN │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + obj.filters.push(filter); │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsLessThanOrEqualTo": function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + obj.filters.push(filter); │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsGreaterThanOrEqualTo": function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + obj.filters.push(filter); │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsBetween": function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.BETWEEN │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + obj.filters.push(filter); │ │ │ │ │ + }, │ │ │ │ │ + "Literal": function(node, obj) { │ │ │ │ │ + obj.value = OpenLayers.String.numericIf( │ │ │ │ │ + this.getChildValue(node), true); │ │ │ │ │ + }, │ │ │ │ │ + "PropertyName": function(node, filter) { │ │ │ │ │ + filter.property = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "LowerBoundary": function(node, filter) { │ │ │ │ │ + filter.lowerBoundary = OpenLayers.String.numericIf( │ │ │ │ │ + this.readers.ogc._expression.call(this, node), true); │ │ │ │ │ + }, │ │ │ │ │ + "UpperBoundary": function(node, filter) { │ │ │ │ │ + filter.upperBoundary = OpenLayers.String.numericIf( │ │ │ │ │ + this.readers.ogc._expression.call(this, node), true); │ │ │ │ │ + }, │ │ │ │ │ + "Intersects": function(node, obj) { │ │ │ │ │ + this.readSpatial(node, obj, OpenLayers.Filter.Spatial.INTERSECTS); │ │ │ │ │ + }, │ │ │ │ │ + "Within": function(node, obj) { │ │ │ │ │ + this.readSpatial(node, obj, OpenLayers.Filter.Spatial.WITHIN); │ │ │ │ │ + }, │ │ │ │ │ + "Contains": function(node, obj) { │ │ │ │ │ + this.readSpatial(node, obj, OpenLayers.Filter.Spatial.CONTAINS); │ │ │ │ │ + }, │ │ │ │ │ + "DWithin": function(node, obj) { │ │ │ │ │ + this.readSpatial(node, obj, OpenLayers.Filter.Spatial.DWITHIN); │ │ │ │ │ + }, │ │ │ │ │ + "Distance": function(node, obj) { │ │ │ │ │ + obj.distance = parseInt(this.getChildValue(node)); │ │ │ │ │ + obj.distanceUnits = node.getAttribute("units"); │ │ │ │ │ + }, │ │ │ │ │ + "Function": function(node, obj) { │ │ │ │ │ + //TODO write decoder for it │ │ │ │ │ + return; │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsNull": function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.IS_NULL │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + obj.filters.push(filter); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: readSpatial │ │ │ │ │ + * │ │ │ │ │ + * Read a {<OpenLayers.Filter.Spatial>} filter. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A DOM element that contains an ogc:expression. │ │ │ │ │ + * obj - {Object} The target object. │ │ │ │ │ + * type - {String} One of the OpenLayers.Filter.Spatial.* constants. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter.Spatial>} The created filter. │ │ │ │ │ + */ │ │ │ │ │ + readSpatial: function(node, obj, type) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: type │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + filter.value = filter.components[0]; │ │ │ │ │ + delete filter.components; │ │ │ │ │ + obj.filters.push(filter); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: encodeLiteral │ │ │ │ │ + * Generates the string representation of a value for use in <Literal> │ │ │ │ │ + * elements. The default encoder writes Date values as ISO 8601 │ │ │ │ │ + * strings. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * value - {Object} Literal value to encode │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} String representation of the provided value. │ │ │ │ │ + */ │ │ │ │ │ + encodeLiteral: function(value) { │ │ │ │ │ + if (value instanceof Date) { │ │ │ │ │ + value = OpenLayers.Date.toISOString(value); │ │ │ │ │ + } │ │ │ │ │ + return value; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: writeOgcExpression │ │ │ │ │ + * Limited support for writing OGC expressions. Currently it supports │ │ │ │ │ + * (<OpenLayers.Filter.Function> || String || Number) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * value - (<OpenLayers.Filter.Function> || String || Number) │ │ │ │ │ + * node - {DOMElement} A parent DOM element │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} Updated node element. │ │ │ │ │ + */ │ │ │ │ │ + writeOgcExpression: function(value, node) { │ │ │ │ │ + if (value instanceof OpenLayers.Filter.Function) { │ │ │ │ │ + this.writeNode("Function", value, node); │ │ │ │ │ + } else { │ │ │ │ │ + this.writeNode("Literal", value, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: write │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {<OpenLayers.Filter>} A filter object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} An ogc:Filter element. │ │ │ │ │ + */ │ │ │ │ │ + write: function(filter) { │ │ │ │ │ + return this.writers.ogc["Filter"].apply(this, [filter]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ + */ │ │ │ │ │ + writers: { │ │ │ │ │ + "ogc": { │ │ │ │ │ + "Filter": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:Filter"); │ │ │ │ │ + this.writeNode(this.getFilterType(filter), filter, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "_featureIds": function(filter) { │ │ │ │ │ + var node = this.createDocumentFragment(); │ │ │ │ │ + for (var i = 0, ii = filter.fids.length; i < ii; ++i) { │ │ │ │ │ + this.writeNode("ogc:FeatureId", filter.fids[i], node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "FeatureId": function(fid) { │ │ │ │ │ + return this.createElementNSPlus("ogc:FeatureId", { │ │ │ │ │ + attributes: { │ │ │ │ │ + fid: fid │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "And": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:And"); │ │ │ │ │ + var childFilter; │ │ │ │ │ + for (var i = 0, ii = filter.filters.length; i < ii; ++i) { │ │ │ │ │ + childFilter = filter.filters[i]; │ │ │ │ │ + this.writeNode( │ │ │ │ │ + this.getFilterType(childFilter), childFilter, node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Or": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:Or"); │ │ │ │ │ + var childFilter; │ │ │ │ │ + for (var i = 0, ii = filter.filters.length; i < ii; ++i) { │ │ │ │ │ + childFilter = filter.filters[i]; │ │ │ │ │ + this.writeNode( │ │ │ │ │ + this.getFilterType(childFilter), childFilter, node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Not": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:Not"); │ │ │ │ │ + var childFilter = filter.filters[0]; │ │ │ │ │ + this.writeNode( │ │ │ │ │ + this.getFilterType(childFilter), childFilter, node │ │ │ │ │ + ); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsLessThan": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsLessThan"); │ │ │ │ │ + // no ogc:expression handling for PropertyName for now │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + // handle Literals or Functions for now │ │ │ │ │ + this.writeOgcExpression(filter.value, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsGreaterThan": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsGreaterThan"); │ │ │ │ │ + // no ogc:expression handling for PropertyName for now │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + // handle Literals or Functions for now │ │ │ │ │ + this.writeOgcExpression(filter.value, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsLessThanOrEqualTo": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsLessThanOrEqualTo"); │ │ │ │ │ + // no ogc:expression handling for PropertyName for now │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + // handle Literals or Functions for now │ │ │ │ │ + this.writeOgcExpression(filter.value, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsGreaterThanOrEqualTo": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsGreaterThanOrEqualTo"); │ │ │ │ │ + // no ogc:expression handling for PropertyName for now │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + // handle Literals or Functions for now │ │ │ │ │ + this.writeOgcExpression(filter.value, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsBetween": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsBetween"); │ │ │ │ │ + // no ogc:expression handling for PropertyName for now │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + this.writeNode("LowerBoundary", filter, node); │ │ │ │ │ + this.writeNode("UpperBoundary", filter, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "PropertyName": function(filter) { │ │ │ │ │ + // no ogc:expression handling for now │ │ │ │ │ + return this.createElementNSPlus("ogc:PropertyName", { │ │ │ │ │ + value: filter.property │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "Literal": function(value) { │ │ │ │ │ + var encode = this.encodeLiteral || │ │ │ │ │ + OpenLayers.Format.Filter.v1.prototype.encodeLiteral; │ │ │ │ │ + return this.createElementNSPlus("ogc:Literal", { │ │ │ │ │ + value: encode(value) │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "LowerBoundary": function(filter) { │ │ │ │ │ + // handle Literals or Functions for now │ │ │ │ │ + var node = this.createElementNSPlus("ogc:LowerBoundary"); │ │ │ │ │ + this.writeOgcExpression(filter.lowerBoundary, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "UpperBoundary": function(filter) { │ │ │ │ │ + // handle Literals or Functions for now │ │ │ │ │ + var node = this.createElementNSPlus("ogc:UpperBoundary"); │ │ │ │ │ + this.writeNode("Literal", filter.upperBoundary, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "INTERSECTS": function(filter) { │ │ │ │ │ + return this.writeSpatial(filter, "Intersects"); │ │ │ │ │ + }, │ │ │ │ │ + "WITHIN": function(filter) { │ │ │ │ │ + return this.writeSpatial(filter, "Within"); │ │ │ │ │ + }, │ │ │ │ │ + "CONTAINS": function(filter) { │ │ │ │ │ + return this.writeSpatial(filter, "Contains"); │ │ │ │ │ + }, │ │ │ │ │ + "DWITHIN": function(filter) { │ │ │ │ │ + var node = this.writeSpatial(filter, "DWithin"); │ │ │ │ │ + this.writeNode("Distance", filter, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Distance": function(filter) { │ │ │ │ │ + return this.createElementNSPlus("ogc:Distance", { │ │ │ │ │ + attributes: { │ │ │ │ │ + units: filter.distanceUnits │ │ │ │ │ + }, │ │ │ │ │ + value: filter.distance │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "Function": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:Function", { │ │ │ │ │ + attributes: { │ │ │ │ │ + name: filter.name │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + var params = filter.params; │ │ │ │ │ + for (var i = 0, len = params.length; i < len; i++) { │ │ │ │ │ + this.writeOgcExpression(params[i], node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsNull": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsNull"); │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + return node; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getFilterType │ │ │ │ │ + */ │ │ │ │ │ + getFilterType: function(filter) { │ │ │ │ │ + var filterType = this.filterMap[filter.type]; │ │ │ │ │ + if (!filterType) { │ │ │ │ │ + throw "Filter writing not supported for rule type: " + filter.type; │ │ │ │ │ + } │ │ │ │ │ + return filterType; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: filterMap │ │ │ │ │ + * {Object} Contains a member for each filter type. Values are node names │ │ │ │ │ + * for corresponding OGC Filter child elements. │ │ │ │ │ + */ │ │ │ │ │ + filterMap: { │ │ │ │ │ + "&&": "And", │ │ │ │ │ + "||": "Or", │ │ │ │ │ + "!": "Not", │ │ │ │ │ + "==": "PropertyIsEqualTo", │ │ │ │ │ + "!=": "PropertyIsNotEqualTo", │ │ │ │ │ + "<": "PropertyIsLessThan", │ │ │ │ │ + ">": "PropertyIsGreaterThan", │ │ │ │ │ + "<=": "PropertyIsLessThanOrEqualTo", │ │ │ │ │ + ">=": "PropertyIsGreaterThanOrEqualTo", │ │ │ │ │ + "..": "PropertyIsBetween", │ │ │ │ │ + "~": "PropertyIsLike", │ │ │ │ │ + "NULL": "PropertyIsNull", │ │ │ │ │ + "BBOX": "BBOX", │ │ │ │ │ + "DWITHIN": "DWITHIN", │ │ │ │ │ + "WITHIN": "WITHIN", │ │ │ │ │ + "CONTAINS": "CONTAINS", │ │ │ │ │ + "INTERSECTS": "INTERSECTS", │ │ │ │ │ + "FID": "_featureIds" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.Filter.v1" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/Filter/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/GML/v2.js │ │ │ │ │ + * @requires OpenLayers/Format/Filter/v1.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.Filter.v1_0_0 │ │ │ │ │ + * Write ogc:Filter version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.GML.v2> │ │ │ │ │ + * - <OpenLayers.Format.Filter.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.Filter.v1_0_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.GML.v2, OpenLayers.Format.Filter.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: VERSION │ │ │ │ │ + * {String} 1.0.0 │ │ │ │ │ + */ │ │ │ │ │ + VERSION: "1.0.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} http://www.opengis.net/ogc/filter/1.0.0/filter.xsd │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: "http://www.opengis.net/ogc/filter/1.0.0/filter.xsd", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.Filter.v1_0_0 │ │ │ │ │ + * Instances of this class are not created directly. Use the │ │ │ │ │ + * <OpenLayers.Format.Filter> constructor instead. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.GML.v2.prototype.initialize.apply( │ │ │ │ │ + this, [options] │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "ogc": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "PropertyIsEqualTo": function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.EQUAL_TO │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + obj.filters.push(filter); │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsNotEqualTo": function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.NOT_EQUAL_TO │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + obj.filters.push(filter); │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsLike": function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.LIKE │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + var wildCard = node.getAttribute("wildCard"); │ │ │ │ │ + var singleChar = node.getAttribute("singleChar"); │ │ │ │ │ + var esc = node.getAttribute("escape"); │ │ │ │ │ + filter.value2regex(wildCard, singleChar, esc); │ │ │ │ │ + obj.filters.push(filter); │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.Filter.v1.prototype.readers["ogc"]), │ │ │ │ │ + "gml": OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ + "feature": OpenLayers.Format.GML.v2.prototype.readers["feature"] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ + */ │ │ │ │ │ + writers: { │ │ │ │ │ + "ogc": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "PropertyIsEqualTo": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsEqualTo"); │ │ │ │ │ + // no ogc:expression handling for PropertyName for now │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + // handle Literals or Functions for now │ │ │ │ │ + this.writeOgcExpression(filter.value, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsNotEqualTo": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsNotEqualTo"); │ │ │ │ │ + // no ogc:expression handling for PropertyName for now │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + // handle Literals or Functions for now │ │ │ │ │ + this.writeOgcExpression(filter.value, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "PropertyIsLike": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsLike", { │ │ │ │ │ + attributes: { │ │ │ │ │ + wildCard: "*", │ │ │ │ │ + singleChar: ".", │ │ │ │ │ + escape: "!" │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + // no ogc:expression handling for now │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + // convert regex string to ogc string │ │ │ │ │ + this.writeNode("Literal", filter.regex2value(), node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "BBOX": function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:BBOX"); │ │ │ │ │ + // PropertyName is mandatory in 1.0.0, but e.g. GeoServer also │ │ │ │ │ + // accepts filters without it. When this is used with │ │ │ │ │ + // OpenLayers.Protocol.WFS, OpenLayers.Format.WFST will set a │ │ │ │ │ + // missing filter.property to the geometryName that is │ │ │ │ │ + // configured with the protocol, which defaults to "the_geom". │ │ │ │ │ + // So the only way to omit this mandatory property is to not │ │ │ │ │ + // set the property on the filter and to set the geometryName │ │ │ │ │ + // on the WFS protocol to null. The latter also happens when │ │ │ │ │ + // the protocol is configured without a geometryName and a │ │ │ │ │ + // featureNS. │ │ │ │ │ + filter.property && this.writeNode("PropertyName", filter, node); │ │ │ │ │ + var box = this.writeNode("gml:Box", filter.value, node); │ │ │ │ │ + if (filter.projection) { │ │ │ │ │ + box.setAttribute("srsName", filter.projection); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.Filter.v1.prototype.writers["ogc"]), │ │ │ │ │ + "gml": OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ + "feature": OpenLayers.Format.GML.v2.prototype.writers["feature"] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: writeSpatial │ │ │ │ │ + * │ │ │ │ │ + * Read a {<OpenLayers.Filter.Spatial>} filter and converts it into XML. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {<OpenLayers.Filter.Spatial>} The filter. │ │ │ │ │ + * name - {String} Name of the generated XML element. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The created XML element. │ │ │ │ │ + */ │ │ │ │ │ + writeSpatial: function(filter, name) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:" + name); │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + if (filter.value instanceof OpenLayers.Filter.Function) { │ │ │ │ │ + this.writeNode("Function", filter.value, node); │ │ │ │ │ + } else { │ │ │ │ │ + var child; │ │ │ │ │ + if (filter.value instanceof OpenLayers.Geometry) { │ │ │ │ │ + child = this.writeNode("feature:_geometry", filter.value).firstChild; │ │ │ │ │ + } else { │ │ │ │ │ + child = this.writeNode("gml:Box", filter.value); │ │ │ │ │ + } │ │ │ │ │ + if (filter.projection) { │ │ │ │ │ + child.setAttribute("srsName", filter.projection); │ │ │ │ │ + } │ │ │ │ │ + node.appendChild(child); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.Filter.v1_0_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WFST/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WFST/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/Filter/v1_0_0.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WFST.v1_0_0 │ │ │ │ │ + * A format for creating WFS v1.0.0 transactions. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Format.WFST.v1_0_0> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.Filter.v1_0_0> │ │ │ │ │ + * - <OpenLayers.Format.WFST.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WFST.v1_0_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.Filter.v1_0_0, OpenLayers.Format.WFST.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} WFS version number. │ │ │ │ │ + */ │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: srsNameInQuery │ │ │ │ │ + * {Boolean} If true the reference system is passed in Query requests │ │ │ │ │ + * via the "srsName" attribute to the "wfs:Query" element, this │ │ │ │ │ + * property defaults to false as it isn't WFS 1.0.0 compliant. │ │ │ │ │ + */ │ │ │ │ │ + srsNameInQuery: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocations │ │ │ │ │ + * {Object} Properties are namespace aliases, values are schema locations. │ │ │ │ │ + */ │ │ │ │ │ + schemaLocations: { │ │ │ │ │ + "wfs": "http://schemas.opengis.net/wfs/1.0.0/WFS-transaction.xsd" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WFST.v1_0_0 │ │ │ │ │ + * A class for parsing and generating WFS v1.0.0 transactions. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + * │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ + * featureNS - {String} Feature namespace (optional). │ │ │ │ │ + * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ + * if featureNS is provided). Default is 'feature'. │ │ │ │ │ + * geometryName - {String} Name of geometry attribute. Default is 'the_geom'. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.Filter.v1_0_0.prototype.initialize.apply(this, [options]); │ │ │ │ │ + OpenLayers.Format.WFST.v1.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: readNode │ │ │ │ │ + * Shorthand for applying one of the named readers given the node │ │ │ │ │ + * namespace and local name. Readers take two args (node, obj) and │ │ │ │ │ + * generally extend or modify the second. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The node to be read (required). │ │ │ │ │ + * obj - {Object} The object to be modified (optional). │ │ │ │ │ + * first - {Boolean} Should be set to true for the first node read. This │ │ │ │ │ + * is usually the readNode call in the read method. Without this being │ │ │ │ │ + * set, auto-configured properties will stick on subsequent reads. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The input object, modified (or a new one if none was provided). │ │ │ │ │ + */ │ │ │ │ │ + readNode: function(node, obj, first) { │ │ │ │ │ + // Not the superclass, only the mixin classes inherit from │ │ │ │ │ + // Format.GML.v2. We need this because we don't want to get readNode │ │ │ │ │ + // from the superclass's superclass, which is OpenLayers.Format.XML. │ │ │ │ │ + return OpenLayers.Format.GML.v2.prototype.readNode.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "WFS_TransactionResponse": function(node, obj) { │ │ │ │ │ + obj.insertIds = []; │ │ │ │ │ + obj.success = false; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "InsertResult": function(node, container) { │ │ │ │ │ + var obj = { │ │ │ │ │ + fids: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + container.insertIds = container.insertIds.concat(obj.fids); │ │ │ │ │ + }, │ │ │ │ │ + "TransactionResult": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "Status": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "SUCCESS": function(node, obj) { │ │ │ │ │ + obj.success = true; │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WFST.v1.prototype.readers["wfs"]), │ │ │ │ │ + "gml": OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ + "feature": OpenLayers.Format.GML.v2.prototype.readers["feature"], │ │ │ │ │ + "ogc": OpenLayers.Format.Filter.v1_0_0.prototype.readers["ogc"] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ + */ │ │ │ │ │ + writers: { │ │ │ │ │ + "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "Query": function(options) { │ │ │ │ │ + options = OpenLayers.Util.extend({ │ │ │ │ │ + featureNS: this.featureNS, │ │ │ │ │ + featurePrefix: this.featurePrefix, │ │ │ │ │ + featureType: this.featureType, │ │ │ │ │ + srsName: this.srsName, │ │ │ │ │ + srsNameInQuery: this.srsNameInQuery │ │ │ │ │ + }, options); │ │ │ │ │ + var prefix = options.featurePrefix; │ │ │ │ │ + var node = this.createElementNSPlus("wfs:Query", { │ │ │ │ │ + attributes: { │ │ │ │ │ + typeName: (prefix ? prefix + ":" : "") + │ │ │ │ │ + options.featureType │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (options.srsNameInQuery && options.srsName) { │ │ │ │ │ + node.setAttribute("srsName", options.srsName); │ │ │ │ │ + } │ │ │ │ │ + if (options.featureNS) { │ │ │ │ │ + node.setAttribute("xmlns:" + prefix, options.featureNS); │ │ │ │ │ + } │ │ │ │ │ + if (options.propertyNames) { │ │ │ │ │ + for (var i = 0, len = options.propertyNames.length; i < len; i++) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "ogc:PropertyName", { │ │ │ │ │ + property: options.propertyNames[i] │ │ │ │ │ + }, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (options.filter) { │ │ │ │ │ + this.setFilterProperty(options.filter); │ │ │ │ │ + this.writeNode("ogc:Filter", options.filter, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WFST.v1.prototype.writers["wfs"]), │ │ │ │ │ + "gml": OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ + "feature": OpenLayers.Format.GML.v2.prototype.writers["feature"], │ │ │ │ │ + "ogc": OpenLayers.Format.Filter.v1_0_0.prototype.writers["ogc"] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFST.v1_0_0" │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Protocol/WFS/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Protocol/WFS/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/WFST/v1_0_0.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.WFS.v1_0_0 │ │ │ │ │ + * A WFS v1.0.0 protocol for vector layers. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Protocol.WFS.v1_0_0> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Protocol.WFS.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.WFS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} WFS version number. │ │ │ │ │ + */ │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Protocol.WFS.v1_0_0 │ │ │ │ │ + * A class for giving layers WFS v1.0.0 protocol. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + * │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ + * featureNS - {String} Feature namespace (optional). │ │ │ │ │ + * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ + * if featureNS is provided). Default is 'feature'. │ │ │ │ │ + * geometryName - {String} Name of geometry attribute. Default is 'the_geom'. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.WFS.v1_0_0" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/JSON.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Note: │ │ │ │ │ + * This work draws heavily from the public domain JSON serializer/deserializer │ │ │ │ │ + * at http://www.json.org/json.js. Rewritten so that it doesn't modify │ │ │ │ │ + * basic data prototypes. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.JSON │ │ │ │ │ + * A parser to read/write JSON safely. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Format.JSON> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.JSON = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: indent │ │ │ │ │ + * {String} For "pretty" printing, the indent string will be used once for │ │ │ │ │ + * each indentation level. │ │ │ │ │ + */ │ │ │ │ │ + indent: " ", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: space │ │ │ │ │ + * {String} For "pretty" printing, the space string will be used after │ │ │ │ │ + * the ":" separating a name/value pair. │ │ │ │ │ + */ │ │ │ │ │ + space: " ", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: newline │ │ │ │ │ + * {String} For "pretty" printing, the newline string will be used at the │ │ │ │ │ + * end of each name/value pair or array item. │ │ │ │ │ + */ │ │ │ │ │ + newline: "\n", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: level │ │ │ │ │ + * {Integer} For "pretty" printing, this is incremented/decremented during │ │ │ │ │ + * serialization. │ │ │ │ │ + */ │ │ │ │ │ + level: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: pretty │ │ │ │ │ + * {Boolean} Serialize with extra whitespace for structure. This is set │ │ │ │ │ + * by the <write> method. │ │ │ │ │ + */ │ │ │ │ │ + pretty: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: nativeJSON │ │ │ │ │ + * {Boolean} Does the browser support native json? │ │ │ │ │ + */ │ │ │ │ │ + nativeJSON: (function() { │ │ │ │ │ + return !!(window.JSON && typeof JSON.parse == "function" && typeof JSON.stringify == "function"); │ │ │ │ │ + })(), │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.JSON │ │ │ │ │ + * Create a new parser for JSON. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Deserialize a json string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * json - {String} A JSON string │ │ │ │ │ + * filter - {Function} A function which will be called for every key and │ │ │ │ │ + * value at every level of the final result. Each value will be │ │ │ │ │ + * replaced by the result of the filter function. This can be used to │ │ │ │ │ + * reform generic objects into instances of classes, or to transform │ │ │ │ │ + * date strings into Date objects. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object, array, string, or number . │ │ │ │ │ + */ │ │ │ │ │ + read: function(json, filter) { │ │ │ │ │ + var object; │ │ │ │ │ + if (this.nativeJSON) { │ │ │ │ │ + object = JSON.parse(json, filter); │ │ │ │ │ + } else try { │ │ │ │ │ + /** │ │ │ │ │ + * Parsing happens in three stages. In the first stage, we run the │ │ │ │ │ + * text against a regular expression which looks for non-JSON │ │ │ │ │ + * characters. We are especially concerned with '()' and 'new' │ │ │ │ │ + * because they can cause invocation, and '=' because it can │ │ │ │ │ + * cause mutation. But just to be safe, we will reject all │ │ │ │ │ + * unexpected characters. │ │ │ │ │ + */ │ │ │ │ │ + if (/^[\],:{}\s]*$/.test(json.replace(/\\["\\\/bfnrtu]/g, '@').replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, ']').replace(/(?:^|:|,)(?:\s*\[)+/g, ''))) { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * In the second stage we use the eval function to compile the │ │ │ │ │ + * text into a JavaScript structure. The '{' operator is │ │ │ │ │ + * subject to a syntactic ambiguity in JavaScript - it can │ │ │ │ │ + * begin a block or an object literal. We wrap the text in │ │ │ │ │ + * parens to eliminate the ambiguity. │ │ │ │ │ + */ │ │ │ │ │ + object = eval('(' + json + ')'); │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * In the optional third stage, we recursively walk the new │ │ │ │ │ + * structure, passing each name/value pair to a filter │ │ │ │ │ + * function for possible transformation. │ │ │ │ │ + */ │ │ │ │ │ + if (typeof filter === 'function') { │ │ │ │ │ + function walk(k, v) { │ │ │ │ │ + if (v && typeof v === 'object') { │ │ │ │ │ + for (var i in v) { │ │ │ │ │ + if (v.hasOwnProperty(i)) { │ │ │ │ │ + v[i] = walk(i, v[i]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return filter(k, v); │ │ │ │ │ + } │ │ │ │ │ + object = walk('', object); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } catch (e) { │ │ │ │ │ + // Fall through if the regexp test fails. │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.keepData) { │ │ │ │ │ + this.data = object; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return object; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Serialize an object into a JSON string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * value - {String} The object, array, string, number, boolean or date │ │ │ │ │ + * to be serialized. │ │ │ │ │ + * pretty - {Boolean} Structure the output with newlines and indentation. │ │ │ │ │ + * Default is false. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The JSON string representation of the input value. │ │ │ │ │ + */ │ │ │ │ │ + write: function(value, pretty) { │ │ │ │ │ + this.pretty = !!pretty; │ │ │ │ │ + var json = null; │ │ │ │ │ + var type = typeof value; │ │ │ │ │ + if (this.serialize[type]) { │ │ │ │ │ + try { │ │ │ │ │ + json = (!this.pretty && this.nativeJSON) ? │ │ │ │ │ + JSON.stringify(value) : │ │ │ │ │ + this.serialize[type].apply(this, [value]); │ │ │ │ │ + } catch (err) { │ │ │ │ │ + OpenLayers.Console.error("Trouble serializing: " + err); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return json; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: writeIndent │ │ │ │ │ + * Output an indentation string depending on the indentation level. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} An appropriate indentation string. │ │ │ │ │ + */ │ │ │ │ │ + writeIndent: function() { │ │ │ │ │ + var pieces = []; │ │ │ │ │ + if (this.pretty) { │ │ │ │ │ + for (var i = 0; i < this.level; ++i) { │ │ │ │ │ + pieces.push(this.indent); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return pieces.join(''); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: writeNewline │ │ │ │ │ + * Output a string representing a newline if in pretty printing mode. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string representing a new line. │ │ │ │ │ + */ │ │ │ │ │ + writeNewline: function() { │ │ │ │ │ + return (this.pretty) ? this.newline : ''; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: writeSpace │ │ │ │ │ + * Output a string representing a space if in pretty printing mode. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A space. │ │ │ │ │ + */ │ │ │ │ │ + writeSpace: function() { │ │ │ │ │ + return (this.pretty) ? this.space : ''; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: serialize │ │ │ │ │ + * Object with properties corresponding to the serializable data types. │ │ │ │ │ + * Property values are functions that do the actual serializing. │ │ │ │ │ + */ │ │ │ │ │ + serialize: { │ │ │ │ │ + /** │ │ │ │ │ + * Method: serialize.object │ │ │ │ │ + * Transform an object into a JSON string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * object - {Object} The object to be serialized. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A JSON string representing the object. │ │ │ │ │ + */ │ │ │ │ │ + 'object': function(object) { │ │ │ │ │ + // three special objects that we want to treat differently │ │ │ │ │ + if (object == null) { │ │ │ │ │ + return "null"; │ │ │ │ │ + } │ │ │ │ │ + if (object.constructor == Date) { │ │ │ │ │ + return this.serialize.date.apply(this, [object]); │ │ │ │ │ + } │ │ │ │ │ + if (object.constructor == Array) { │ │ │ │ │ + return this.serialize.array.apply(this, [object]); │ │ │ │ │ + } │ │ │ │ │ + var pieces = ['{']; │ │ │ │ │ + this.level += 1; │ │ │ │ │ + var key, keyJSON, valueJSON; │ │ │ │ │ + │ │ │ │ │ + var addComma = false; │ │ │ │ │ + for (key in object) { │ │ │ │ │ + if (object.hasOwnProperty(key)) { │ │ │ │ │ + // recursive calls need to allow for sub-classing │ │ │ │ │ + keyJSON = OpenLayers.Format.JSON.prototype.write.apply(this, │ │ │ │ │ + [key, this.pretty]); │ │ │ │ │ + valueJSON = OpenLayers.Format.JSON.prototype.write.apply(this, │ │ │ │ │ + [object[key], this.pretty]); │ │ │ │ │ + if (keyJSON != null && valueJSON != null) { │ │ │ │ │ + if (addComma) { │ │ │ │ │ + pieces.push(','); │ │ │ │ │ + } │ │ │ │ │ + pieces.push(this.writeNewline(), this.writeIndent(), │ │ │ │ │ + keyJSON, ':', this.writeSpace(), valueJSON); │ │ │ │ │ + addComma = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.level -= 1; │ │ │ │ │ + pieces.push(this.writeNewline(), this.writeIndent(), '}'); │ │ │ │ │ + return pieces.join(''); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: serialize.array │ │ │ │ │ + * Transform an array into a JSON string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * array - {Array} The array to be serialized │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A JSON string representing the array. │ │ │ │ │ + */ │ │ │ │ │ + 'array': function(array) { │ │ │ │ │ + var json; │ │ │ │ │ + var pieces = ['[']; │ │ │ │ │ + this.level += 1; │ │ │ │ │ + │ │ │ │ │ + for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ + // recursive calls need to allow for sub-classing │ │ │ │ │ + json = OpenLayers.Format.JSON.prototype.write.apply(this, │ │ │ │ │ + [array[i], this.pretty]); │ │ │ │ │ + if (json != null) { │ │ │ │ │ + if (i > 0) { │ │ │ │ │ + pieces.push(','); │ │ │ │ │ + } │ │ │ │ │ + pieces.push(this.writeNewline(), this.writeIndent(), json); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.level -= 1; │ │ │ │ │ + pieces.push(this.writeNewline(), this.writeIndent(), ']'); │ │ │ │ │ + return pieces.join(''); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: serialize.string │ │ │ │ │ + * Transform a string into a JSON string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * string - {String} The string to be serialized │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A JSON string representing the string. │ │ │ │ │ + */ │ │ │ │ │ + 'string': function(string) { │ │ │ │ │ + // If the string contains no control characters, no quote characters, and no │ │ │ │ │ + // backslash characters, then we can simply slap some quotes around it. │ │ │ │ │ + // Otherwise we must also replace the offending characters with safe │ │ │ │ │ + // sequences. │ │ │ │ │ + var m = { │ │ │ │ │ + '\b': '\\b', │ │ │ │ │ + '\t': '\\t', │ │ │ │ │ + '\n': '\\n', │ │ │ │ │ + '\f': '\\f', │ │ │ │ │ + '\r': '\\r', │ │ │ │ │ + '"': '\\"', │ │ │ │ │ + '\\': '\\\\' │ │ │ │ │ + }; │ │ │ │ │ + if (/["\\\x00-\x1f]/.test(string)) { │ │ │ │ │ + return '"' + string.replace(/([\x00-\x1f\\"])/g, function(a, b) { │ │ │ │ │ + var c = m[b]; │ │ │ │ │ + if (c) { │ │ │ │ │ + return c; │ │ │ │ │ + } │ │ │ │ │ + c = b.charCodeAt(); │ │ │ │ │ + return '\\u00' + │ │ │ │ │ + Math.floor(c / 16).toString(16) + │ │ │ │ │ + (c % 16).toString(16); │ │ │ │ │ + }) + '"'; │ │ │ │ │ + } │ │ │ │ │ + return '"' + string + '"'; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: serialize.number │ │ │ │ │ + * Transform a number into a JSON string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * number - {Number} The number to be serialized. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A JSON string representing the number. │ │ │ │ │ + */ │ │ │ │ │ + 'number': function(number) { │ │ │ │ │ + return isFinite(number) ? String(number) : "null"; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: serialize.boolean │ │ │ │ │ + * Transform a boolean into a JSON string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bool - {Boolean} The boolean to be serialized. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A JSON string representing the boolean. │ │ │ │ │ + */ │ │ │ │ │ + 'boolean': function(bool) { │ │ │ │ │ + return String(bool); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: serialize.object │ │ │ │ │ + * Transform a date into a JSON string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * date - {Date} The date to be serialized. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A JSON string representing the date. │ │ │ │ │ + */ │ │ │ │ │ + 'date': function(date) { │ │ │ │ │ + function format(number) { │ │ │ │ │ + // Format integers to have at least two digits. │ │ │ │ │ + return (number < 10) ? '0' + number : number; │ │ │ │ │ + } │ │ │ │ │ + return '"' + date.getFullYear() + '-' + │ │ │ │ │ + format(date.getMonth() + 1) + '-' + │ │ │ │ │ + format(date.getDate()) + 'T' + │ │ │ │ │ + format(date.getHours()) + ':' + │ │ │ │ │ + format(date.getMinutes()) + ':' + │ │ │ │ │ + format(date.getSeconds()) + '"'; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.JSON" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/GeoJSON.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/JSON.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Geometry/MultiPoint.js │ │ │ │ │ + * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ + * @requires OpenLayers/Geometry/MultiLineString.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ + * @requires OpenLayers/Geometry/MultiPolygon.js │ │ │ │ │ + * @requires OpenLayers/Console.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.GeoJSON │ │ │ │ │ + * Read and write GeoJSON. Create a new parser with the │ │ │ │ │ + * <OpenLayers.Format.GeoJSON> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.JSON> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.GeoJSON = OpenLayers.Class(OpenLayers.Format.JSON, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: ignoreExtraDims │ │ │ │ │ + * {Boolean} Ignore dimensions higher than 2 when reading geometry │ │ │ │ │ + * coordinates. │ │ │ │ │ + */ │ │ │ │ │ + ignoreExtraDims: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.GeoJSON │ │ │ │ │ + * Create a new parser for GeoJSON. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Deserialize a GeoJSON string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * json - {String} A GeoJSON string │ │ │ │ │ + * type - {String} Optional string that determines the structure of │ │ │ │ │ + * the output. Supported values are "Geometry", "Feature", and │ │ │ │ │ + * "FeatureCollection". If absent or null, a default of │ │ │ │ │ + * "FeatureCollection" is assumed. │ │ │ │ │ + * filter - {Function} A function which will be called for every key and │ │ │ │ │ + * value at every level of the final result. Each value will be │ │ │ │ │ + * replaced by the result of the filter function. This can be used to │ │ │ │ │ + * reform generic objects into instances of classes, or to transform │ │ │ │ │ + * date strings into Date objects. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The return depends on the value of the type argument. If type │ │ │ │ │ + * is "FeatureCollection" (the default), the return will be an array │ │ │ │ │ + * of <OpenLayers.Feature.Vector>. If type is "Geometry", the input json │ │ │ │ │ + * must represent a single geometry, and the return will be an │ │ │ │ │ + * <OpenLayers.Geometry>. If type is "Feature", the input json must │ │ │ │ │ + * represent a single feature, and the return will be an │ │ │ │ │ + * <OpenLayers.Feature.Vector>. │ │ │ │ │ + */ │ │ │ │ │ + read: function(json, type, filter) { │ │ │ │ │ + type = (type) ? type : "FeatureCollection"; │ │ │ │ │ + var results = null; │ │ │ │ │ + var obj = null; │ │ │ │ │ + if (typeof json == "string") { │ │ │ │ │ + obj = OpenLayers.Format.JSON.prototype.read.apply(this, │ │ │ │ │ + [json, filter]); │ │ │ │ │ + } else { │ │ │ │ │ + obj = json; │ │ │ │ │ } │ │ │ │ │ if (!obj) { │ │ │ │ │ OpenLayers.Console.error("Bad JSON: " + json); │ │ │ │ │ } else if (typeof(obj.type) != "string") { │ │ │ │ │ OpenLayers.Console.error("Bad GeoJSON - no type: " + json); │ │ │ │ │ } else if (this.isValidType(obj, type)) { │ │ │ │ │ switch (type) { │ │ │ │ │ @@ -22130,1970 +32553,14 @@ │ │ │ │ │ │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.GeoJSON" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/BaseTypes/Date.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/SingleFile.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Date │ │ │ │ │ - * Contains implementations of Date.parse and date.toISOString that match the │ │ │ │ │ - * ECMAScript 5 specification for parsing RFC 3339 dates. │ │ │ │ │ - * http://tools.ietf.org/html/rfc3339 │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Date = { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: dateRegEx │ │ │ │ │ - * The regex to be used for validating dates. You can provide your own │ │ │ │ │ - * regex for instance for adding support for years before BC. Default │ │ │ │ │ - * value is: /^(?:(\d{4})(?:-(\d{2})(?:-(\d{2}))?)?)?(?:(?:T(\d{1,2}):(\d{2}):(\d{2}(?:\.\d+)?)(Z|(?:[+-]\d{1,2}(?::(\d{2}))?)))|Z)?$/ │ │ │ │ │ - */ │ │ │ │ │ - dateRegEx: /^(?:(\d{4})(?:-(\d{2})(?:-(\d{2}))?)?)?(?:(?:T(\d{1,2}):(\d{2}):(\d{2}(?:\.\d+)?)(Z|(?:[+-]\d{1,2}(?::(\d{2}))?)))|Z)?$/, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: toISOString │ │ │ │ │ - * Generates a string representing a date. The format of the string follows │ │ │ │ │ - * the profile of ISO 8601 for date and time on the Internet (see │ │ │ │ │ - * http://tools.ietf.org/html/rfc3339). If the toISOString method is │ │ │ │ │ - * available on the Date prototype, that is used. The toISOString │ │ │ │ │ - * method for Date instances is defined in ECMA-262. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * date - {Date} A date object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string representing the date (e.g. │ │ │ │ │ - * "2010-08-07T16:58:23.123Z"). If the date does not have a valid time │ │ │ │ │ - * (i.e. isNaN(date.getTime())) this method returns the string "Invalid │ │ │ │ │ - * Date". The ECMA standard says the toISOString method should throw │ │ │ │ │ - * RangeError in this case, but Firefox returns a string instead. For │ │ │ │ │ - * best results, use isNaN(date.getTime()) to determine date validity │ │ │ │ │ - * before generating date strings. │ │ │ │ │ - */ │ │ │ │ │ - toISOString: (function() { │ │ │ │ │ - if ("toISOString" in Date.prototype) { │ │ │ │ │ - return function(date) { │ │ │ │ │ - return date.toISOString(); │ │ │ │ │ - }; │ │ │ │ │ - } else { │ │ │ │ │ - return function(date) { │ │ │ │ │ - var str; │ │ │ │ │ - if (isNaN(date.getTime())) { │ │ │ │ │ - // ECMA-262 says throw RangeError, Firefox returns │ │ │ │ │ - // "Invalid Date" │ │ │ │ │ - str = "Invalid Date"; │ │ │ │ │ - } else { │ │ │ │ │ - str = │ │ │ │ │ - date.getUTCFullYear() + "-" + │ │ │ │ │ - OpenLayers.Number.zeroPad(date.getUTCMonth() + 1, 2) + "-" + │ │ │ │ │ - OpenLayers.Number.zeroPad(date.getUTCDate(), 2) + "T" + │ │ │ │ │ - OpenLayers.Number.zeroPad(date.getUTCHours(), 2) + ":" + │ │ │ │ │ - OpenLayers.Number.zeroPad(date.getUTCMinutes(), 2) + ":" + │ │ │ │ │ - OpenLayers.Number.zeroPad(date.getUTCSeconds(), 2) + "." + │ │ │ │ │ - OpenLayers.Number.zeroPad(date.getUTCMilliseconds(), 3) + "Z"; │ │ │ │ │ - } │ │ │ │ │ - return str; │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - })(), │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: parse │ │ │ │ │ - * Generate a date object from a string. The format for the string follows │ │ │ │ │ - * the profile of ISO 8601 for date and time on the Internet (see │ │ │ │ │ - * http://tools.ietf.org/html/rfc3339). We don't call the native │ │ │ │ │ - * Date.parse because of inconsistency between implmentations. In │ │ │ │ │ - * Chrome, calling Date.parse with a string that doesn't contain any │ │ │ │ │ - * indication of the timezone (e.g. "2011"), the date is interpreted │ │ │ │ │ - * in local time. On Firefox, the assumption is UTC. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * str - {String} A string representing the date (e.g. │ │ │ │ │ - * "2010", "2010-08", "2010-08-07", "2010-08-07T16:58:23.123Z", │ │ │ │ │ - * "2010-08-07T11:58:23.123-06"). │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Date} A date object. If the string could not be parsed, an invalid │ │ │ │ │ - * date is returned (i.e. isNaN(date.getTime())). │ │ │ │ │ - */ │ │ │ │ │ - parse: function(str) { │ │ │ │ │ - var date; │ │ │ │ │ - var match = str.match(this.dateRegEx); │ │ │ │ │ - if (match && (match[1] || match[7])) { // must have at least year or time │ │ │ │ │ - var year = parseInt(match[1], 10) || 0; │ │ │ │ │ - var month = (parseInt(match[2], 10) - 1) || 0; │ │ │ │ │ - var day = parseInt(match[3], 10) || 1; │ │ │ │ │ - date = new Date(Date.UTC(year, month, day)); │ │ │ │ │ - // optional time │ │ │ │ │ - var type = match[7]; │ │ │ │ │ - if (type) { │ │ │ │ │ - var hours = parseInt(match[4], 10); │ │ │ │ │ - var minutes = parseInt(match[5], 10); │ │ │ │ │ - var secFrac = parseFloat(match[6]); │ │ │ │ │ - var seconds = secFrac | 0; │ │ │ │ │ - var milliseconds = Math.round(1000 * (secFrac - seconds)); │ │ │ │ │ - date.setUTCHours(hours, minutes, seconds, milliseconds); │ │ │ │ │ - // check offset │ │ │ │ │ - if (type !== "Z") { │ │ │ │ │ - var hoursOffset = parseInt(type, 10); │ │ │ │ │ - var minutesOffset = parseInt(match[8], 10) || 0; │ │ │ │ │ - var offset = -1000 * (60 * (hoursOffset * 60) + minutesOffset * 60); │ │ │ │ │ - date = new Date(date.getTime() + offset); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - date = new Date("invalid"); │ │ │ │ │ - } │ │ │ │ │ - return date; │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/XML.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.XML │ │ │ │ │ - * Read and write XML. For cross-browser XML generation, use methods on an │ │ │ │ │ - * instance of the XML format class instead of on <code>document<end>. │ │ │ │ │ - * The DOM creation and traversing methods exposed here all mimic the │ │ │ │ │ - * W3C XML DOM methods. Create a new parser with the │ │ │ │ │ - * <OpenLayers.Format.XML> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.XML = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. Properties │ │ │ │ │ - * of this object should not be set individually. Read-only. All │ │ │ │ │ - * XML subclasses should have their own namespaces object. Use │ │ │ │ │ - * <setNamespace> to add or set a namespace alias after construction. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaceAlias │ │ │ │ │ - * {Object} Mapping of namespace URI to namespace alias. This object │ │ │ │ │ - * is read-only. Use <setNamespace> to add or set a namespace alias. │ │ │ │ │ - */ │ │ │ │ │ - namespaceAlias: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - * {String} The default namespace alias for creating element nodes. │ │ │ │ │ - */ │ │ │ │ │ - defaultPrefix: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the <readers> property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ - */ │ │ │ │ │ - writers: {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: xmldom │ │ │ │ │ - * {XMLDom} If this browser uses ActiveX, this will be set to a XMLDOM │ │ │ │ │ - * object. It is not intended to be a browser sniffing property. │ │ │ │ │ - * Instead, the xmldom property is used instead of <code>document<end> │ │ │ │ │ - * where namespaced node creation methods are not supported. In all │ │ │ │ │ - * other browsers, this remains null. │ │ │ │ │ - */ │ │ │ │ │ - xmldom: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.XML │ │ │ │ │ - * Construct an XML parser. The parser is used to read and write XML. │ │ │ │ │ - * Reading XML from a string returns a DOM element. Writing XML from │ │ │ │ │ - * a DOM element returns a string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on │ │ │ │ │ - * the object. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - if (window.ActiveXObject) { │ │ │ │ │ - this.xmldom = new ActiveXObject("Microsoft.XMLDOM"); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Format.prototype.initialize.apply(this, [options]); │ │ │ │ │ - // clone the namespace object and set all namespace aliases │ │ │ │ │ - this.namespaces = OpenLayers.Util.extend({}, this.namespaces); │ │ │ │ │ - this.namespaceAlias = {}; │ │ │ │ │ - for (var alias in this.namespaces) { │ │ │ │ │ - this.namespaceAlias[this.namespaces[alias]] = alias; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.xmldom = null; │ │ │ │ │ - OpenLayers.Format.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setNamespace │ │ │ │ │ - * Set a namespace alias and URI for the format. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * alias - {String} The namespace alias (prefix). │ │ │ │ │ - * uri - {String} The namespace URI. │ │ │ │ │ - */ │ │ │ │ │ - setNamespace: function(alias, uri) { │ │ │ │ │ - this.namespaces[alias] = uri; │ │ │ │ │ - this.namespaceAlias[uri] = alias; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Deserialize a XML string and return a DOM node. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * text - {String} A XML string │ │ │ │ │ - │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A DOM node │ │ │ │ │ - */ │ │ │ │ │ - read: function(text) { │ │ │ │ │ - var index = text.indexOf('<'); │ │ │ │ │ - if (index > 0) { │ │ │ │ │ - text = text.substring(index); │ │ │ │ │ - } │ │ │ │ │ - var node = OpenLayers.Util.Try( │ │ │ │ │ - OpenLayers.Function.bind(( │ │ │ │ │ - function() { │ │ │ │ │ - var xmldom; │ │ │ │ │ - /** │ │ │ │ │ - * Since we want to be able to call this method on the prototype │ │ │ │ │ - * itself, this.xmldom may not exist even if in IE. │ │ │ │ │ - */ │ │ │ │ │ - if (window.ActiveXObject && !this.xmldom) { │ │ │ │ │ - xmldom = new ActiveXObject("Microsoft.XMLDOM"); │ │ │ │ │ - } else { │ │ │ │ │ - xmldom = this.xmldom; │ │ │ │ │ - │ │ │ │ │ - } │ │ │ │ │ - xmldom.loadXML(text); │ │ │ │ │ - return xmldom; │ │ │ │ │ - } │ │ │ │ │ - ), this), │ │ │ │ │ - function() { │ │ │ │ │ - return new DOMParser().parseFromString(text, 'text/xml'); │ │ │ │ │ - }, │ │ │ │ │ - function() { │ │ │ │ │ - var req = new XMLHttpRequest(); │ │ │ │ │ - req.open("GET", "data:" + "text/xml" + │ │ │ │ │ - ";charset=utf-8," + encodeURIComponent(text), false); │ │ │ │ │ - if (req.overrideMimeType) { │ │ │ │ │ - req.overrideMimeType("text/xml"); │ │ │ │ │ - } │ │ │ │ │ - req.send(null); │ │ │ │ │ - return req.responseXML; │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - if (this.keepData) { │ │ │ │ │ - this.data = node; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Serialize a DOM node into a XML string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A DOM node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The XML string representation of the input node. │ │ │ │ │ - */ │ │ │ │ │ - write: function(node) { │ │ │ │ │ - var data; │ │ │ │ │ - if (this.xmldom) { │ │ │ │ │ - data = node.xml; │ │ │ │ │ - } else { │ │ │ │ │ - var serializer = new XMLSerializer(); │ │ │ │ │ - if (node.nodeType == 1) { │ │ │ │ │ - // Add nodes to a document before serializing. Everything else │ │ │ │ │ - // is serialized as is. This may need more work. See #1218 . │ │ │ │ │ - var doc = document.implementation.createDocument("", "", null); │ │ │ │ │ - if (doc.importNode) { │ │ │ │ │ - node = doc.importNode(node, true); │ │ │ │ │ - } │ │ │ │ │ - doc.appendChild(node); │ │ │ │ │ - data = serializer.serializeToString(doc); │ │ │ │ │ - } else { │ │ │ │ │ - data = serializer.serializeToString(node); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return data; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: createElementNS │ │ │ │ │ - * Create a new element with namespace. This node can be appended to │ │ │ │ │ - * another node with the standard node.appendChild method. For │ │ │ │ │ - * cross-browser support, this method must be used instead of │ │ │ │ │ - * document.createElementNS. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * uri - {String} Namespace URI for the element. │ │ │ │ │ - * name - {String} The qualified name of the element (prefix:localname). │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Element} A DOM element with namespace. │ │ │ │ │ - */ │ │ │ │ │ - createElementNS: function(uri, name) { │ │ │ │ │ - var element; │ │ │ │ │ - if (this.xmldom) { │ │ │ │ │ - if (typeof uri == "string") { │ │ │ │ │ - element = this.xmldom.createNode(1, name, uri); │ │ │ │ │ - } else { │ │ │ │ │ - element = this.xmldom.createNode(1, name, ""); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - element = document.createElementNS(uri, name); │ │ │ │ │ - } │ │ │ │ │ - return element; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: createDocumentFragment │ │ │ │ │ - * Create a document fragment node that can be appended to another node │ │ │ │ │ - * created by createElementNS. This will call │ │ │ │ │ - * document.createDocumentFragment outside of IE. In IE, the ActiveX │ │ │ │ │ - * object's createDocumentFragment method is used. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Element} A document fragment. │ │ │ │ │ - */ │ │ │ │ │ - createDocumentFragment: function() { │ │ │ │ │ - var element; │ │ │ │ │ - if (this.xmldom) { │ │ │ │ │ - element = this.xmldom.createDocumentFragment(); │ │ │ │ │ - } else { │ │ │ │ │ - element = document.createDocumentFragment(); │ │ │ │ │ - } │ │ │ │ │ - return element; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: createTextNode │ │ │ │ │ - * Create a text node. This node can be appended to another node with │ │ │ │ │ - * the standard node.appendChild method. For cross-browser support, │ │ │ │ │ - * this method must be used instead of document.createTextNode. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * text - {String} The text of the node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A DOM text node. │ │ │ │ │ - */ │ │ │ │ │ - createTextNode: function(text) { │ │ │ │ │ - var node; │ │ │ │ │ - if (typeof text !== "string") { │ │ │ │ │ - text = String(text); │ │ │ │ │ - } │ │ │ │ │ - if (this.xmldom) { │ │ │ │ │ - node = this.xmldom.createTextNode(text); │ │ │ │ │ - } else { │ │ │ │ │ - node = document.createTextNode(text); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getElementsByTagNameNS │ │ │ │ │ - * Get a list of elements on a node given the namespace URI and local name. │ │ │ │ │ - * To return all nodes in a given namespace, use '*' for the name │ │ │ │ │ - * argument. To return all nodes of a given (local) name, regardless │ │ │ │ │ - * of namespace, use '*' for the uri argument. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {Element} Node on which to search for other nodes. │ │ │ │ │ - * uri - {String} Namespace URI. │ │ │ │ │ - * name - {String} Local name of the tag (without the prefix). │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {NodeList} A node list or array of elements. │ │ │ │ │ - */ │ │ │ │ │ - getElementsByTagNameNS: function(node, uri, name) { │ │ │ │ │ - var elements = []; │ │ │ │ │ - if (node.getElementsByTagNameNS) { │ │ │ │ │ - elements = node.getElementsByTagNameNS(uri, name); │ │ │ │ │ - } else { │ │ │ │ │ - // brute force method │ │ │ │ │ - var allNodes = node.getElementsByTagName("*"); │ │ │ │ │ - var potentialNode, fullName; │ │ │ │ │ - for (var i = 0, len = allNodes.length; i < len; ++i) { │ │ │ │ │ - potentialNode = allNodes[i]; │ │ │ │ │ - fullName = (potentialNode.prefix) ? │ │ │ │ │ - (potentialNode.prefix + ":" + name) : name; │ │ │ │ │ - if ((name == "*") || (fullName == potentialNode.nodeName)) { │ │ │ │ │ - if ((uri == "*") || (uri == potentialNode.namespaceURI)) { │ │ │ │ │ - elements.push(potentialNode); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return elements; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getAttributeNodeNS │ │ │ │ │ - * Get an attribute node given the namespace URI and local name. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {Element} Node on which to search for attribute nodes. │ │ │ │ │ - * uri - {String} Namespace URI. │ │ │ │ │ - * name - {String} Local name of the attribute (without the prefix). │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} An attribute node or null if none found. │ │ │ │ │ - */ │ │ │ │ │ - getAttributeNodeNS: function(node, uri, name) { │ │ │ │ │ - var attributeNode = null; │ │ │ │ │ - if (node.getAttributeNodeNS) { │ │ │ │ │ - attributeNode = node.getAttributeNodeNS(uri, name); │ │ │ │ │ - } else { │ │ │ │ │ - var attributes = node.attributes; │ │ │ │ │ - var potentialNode, fullName; │ │ │ │ │ - for (var i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ - potentialNode = attributes[i]; │ │ │ │ │ - if (potentialNode.namespaceURI == uri) { │ │ │ │ │ - fullName = (potentialNode.prefix) ? │ │ │ │ │ - (potentialNode.prefix + ":" + name) : name; │ │ │ │ │ - if (fullName == potentialNode.nodeName) { │ │ │ │ │ - attributeNode = potentialNode; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return attributeNode; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getAttributeNS │ │ │ │ │ - * Get an attribute value given the namespace URI and local name. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {Element} Node on which to search for an attribute. │ │ │ │ │ - * uri - {String} Namespace URI. │ │ │ │ │ - * name - {String} Local name of the attribute (without the prefix). │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} An attribute value or and empty string if none found. │ │ │ │ │ - */ │ │ │ │ │ - getAttributeNS: function(node, uri, name) { │ │ │ │ │ - var attributeValue = ""; │ │ │ │ │ - if (node.getAttributeNS) { │ │ │ │ │ - attributeValue = node.getAttributeNS(uri, name) || ""; │ │ │ │ │ - } else { │ │ │ │ │ - var attributeNode = this.getAttributeNodeNS(node, uri, name); │ │ │ │ │ - if (attributeNode) { │ │ │ │ │ - attributeValue = attributeNode.nodeValue; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return attributeValue; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getChildValue │ │ │ │ │ - * Get the textual value of the node if it exists, or return an │ │ │ │ │ - * optional default string. Returns an empty string if no first child │ │ │ │ │ - * exists and no default value is supplied. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} The element used to look for a first child value. │ │ │ │ │ - * def - {String} Optional string to return in the event that no │ │ │ │ │ - * first child value exists. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The value of the first child of the given node. │ │ │ │ │ - */ │ │ │ │ │ - getChildValue: function(node, def) { │ │ │ │ │ - var value = def || ""; │ │ │ │ │ - if (node) { │ │ │ │ │ - for (var child = node.firstChild; child; child = child.nextSibling) { │ │ │ │ │ - switch (child.nodeType) { │ │ │ │ │ - case 3: // text node │ │ │ │ │ - case 4: // cdata section │ │ │ │ │ - value += child.nodeValue; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return value; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: isSimpleContent │ │ │ │ │ - * Test if the given node has only simple content (i.e. no child element │ │ │ │ │ - * nodes). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} An element node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The node has no child element nodes (nodes of type 1). │ │ │ │ │ - */ │ │ │ │ │ - isSimpleContent: function(node) { │ │ │ │ │ - var simple = true; │ │ │ │ │ - for (var child = node.firstChild; child; child = child.nextSibling) { │ │ │ │ │ - if (child.nodeType === 1) { │ │ │ │ │ - simple = false; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return simple; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: contentType │ │ │ │ │ - * Determine the content type for a given node. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} One of OpenLayers.Format.XML.CONTENT_TYPE.{EMPTY,SIMPLE,COMPLEX,MIXED} │ │ │ │ │ - * if the node has no, simple, complex, or mixed content. │ │ │ │ │ - */ │ │ │ │ │ - contentType: function(node) { │ │ │ │ │ - var simple = false, │ │ │ │ │ - complex = false; │ │ │ │ │ - │ │ │ │ │ - var type = OpenLayers.Format.XML.CONTENT_TYPE.EMPTY; │ │ │ │ │ - │ │ │ │ │ - for (var child = node.firstChild; child; child = child.nextSibling) { │ │ │ │ │ - switch (child.nodeType) { │ │ │ │ │ - case 1: // element │ │ │ │ │ - complex = true; │ │ │ │ │ - break; │ │ │ │ │ - case 8: // comment │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - simple = true; │ │ │ │ │ - } │ │ │ │ │ - if (complex && simple) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (complex && simple) { │ │ │ │ │ - type = OpenLayers.Format.XML.CONTENT_TYPE.MIXED; │ │ │ │ │ - } else if (complex) { │ │ │ │ │ - return OpenLayers.Format.XML.CONTENT_TYPE.COMPLEX; │ │ │ │ │ - } else if (simple) { │ │ │ │ │ - return OpenLayers.Format.XML.CONTENT_TYPE.SIMPLE; │ │ │ │ │ - } │ │ │ │ │ - return type; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: hasAttributeNS │ │ │ │ │ - * Determine whether a node has a particular attribute matching the given │ │ │ │ │ - * name and namespace. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {Element} Node on which to search for an attribute. │ │ │ │ │ - * uri - {String} Namespace URI. │ │ │ │ │ - * name - {String} Local name of the attribute (without the prefix). │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The node has an attribute matching the name and namespace. │ │ │ │ │ - */ │ │ │ │ │ - hasAttributeNS: function(node, uri, name) { │ │ │ │ │ - var found = false; │ │ │ │ │ - if (node.hasAttributeNS) { │ │ │ │ │ - found = node.hasAttributeNS(uri, name); │ │ │ │ │ - } else { │ │ │ │ │ - found = !!this.getAttributeNodeNS(node, uri, name); │ │ │ │ │ - } │ │ │ │ │ - return found; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setAttributeNS │ │ │ │ │ - * Adds a new attribute or changes the value of an attribute with the given │ │ │ │ │ - * namespace and name. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {Element} Element node on which to set the attribute. │ │ │ │ │ - * uri - {String} Namespace URI for the attribute. │ │ │ │ │ - * name - {String} Qualified name (prefix:localname) for the attribute. │ │ │ │ │ - * value - {String} Attribute value. │ │ │ │ │ - */ │ │ │ │ │ - setAttributeNS: function(node, uri, name, value) { │ │ │ │ │ - if (node.setAttributeNS) { │ │ │ │ │ - node.setAttributeNS(uri, name, value); │ │ │ │ │ - } else { │ │ │ │ │ - if (this.xmldom) { │ │ │ │ │ - if (uri) { │ │ │ │ │ - var attribute = node.ownerDocument.createNode( │ │ │ │ │ - 2, name, uri │ │ │ │ │ - ); │ │ │ │ │ - attribute.nodeValue = value; │ │ │ │ │ - node.setAttributeNode(attribute); │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttribute(name, value); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - throw "setAttributeNS not implemented"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createElementNSPlus │ │ │ │ │ - * Shorthand for creating namespaced elements with optional attributes and │ │ │ │ │ - * child text nodes. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} The qualified node name. │ │ │ │ │ - * options - {Object} Optional object for node configuration. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * uri - {String} Optional namespace uri for the element - supply a prefix │ │ │ │ │ - * instead if the namespace uri is a property of the format's namespace │ │ │ │ │ - * object. │ │ │ │ │ - * attributes - {Object} Optional attributes to be set using the │ │ │ │ │ - * <setAttributes> method. │ │ │ │ │ - * value - {String} Optional text to be appended as a text node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Element} An element node. │ │ │ │ │ - */ │ │ │ │ │ - createElementNSPlus: function(name, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - // order of prefix preference │ │ │ │ │ - // 1. in the uri option │ │ │ │ │ - // 2. in the prefix option │ │ │ │ │ - // 3. in the qualified name │ │ │ │ │ - // 4. from the defaultPrefix │ │ │ │ │ - var uri = options.uri || this.namespaces[options.prefix]; │ │ │ │ │ - if (!uri) { │ │ │ │ │ - var loc = name.indexOf(":"); │ │ │ │ │ - uri = this.namespaces[name.substring(0, loc)]; │ │ │ │ │ - } │ │ │ │ │ - if (!uri) { │ │ │ │ │ - uri = this.namespaces[this.defaultPrefix]; │ │ │ │ │ - } │ │ │ │ │ - var node = this.createElementNS(uri, name); │ │ │ │ │ - if (options.attributes) { │ │ │ │ │ - this.setAttributes(node, options.attributes); │ │ │ │ │ - } │ │ │ │ │ - var value = options.value; │ │ │ │ │ - if (value != null) { │ │ │ │ │ - node.appendChild(this.createTextNode(value)); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setAttributes │ │ │ │ │ - * Set multiple attributes given key value pairs from an object. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {Element} An element node. │ │ │ │ │ - * obj - {Object || Array} An object whose properties represent attribute │ │ │ │ │ - * names and values represent attribute values. If an attribute name │ │ │ │ │ - * is a qualified name ("prefix:local"), the prefix will be looked up │ │ │ │ │ - * in the parsers {namespaces} object. If the prefix is found, │ │ │ │ │ - * setAttributeNS will be used instead of setAttribute. │ │ │ │ │ - */ │ │ │ │ │ - setAttributes: function(node, obj) { │ │ │ │ │ - var value, uri; │ │ │ │ │ - for (var name in obj) { │ │ │ │ │ - if (obj[name] != null && obj[name].toString) { │ │ │ │ │ - value = obj[name].toString(); │ │ │ │ │ - // check for qualified attribute name ("prefix:local") │ │ │ │ │ - uri = this.namespaces[name.substring(0, name.indexOf(":"))] || null; │ │ │ │ │ - this.setAttributeNS(node, uri, name, value); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: readNode │ │ │ │ │ - * Shorthand for applying one of the named readers given the node │ │ │ │ │ - * namespace and local name. Readers take two args (node, obj) and │ │ │ │ │ - * generally extend or modify the second. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} The node to be read (required). │ │ │ │ │ - * obj - {Object} The object to be modified (optional). │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The input object, modified (or a new one if none was provided). │ │ │ │ │ - */ │ │ │ │ │ - readNode: function(node, obj) { │ │ │ │ │ - if (!obj) { │ │ │ │ │ - obj = {}; │ │ │ │ │ - } │ │ │ │ │ - var group = this.readers[node.namespaceURI ? this.namespaceAlias[node.namespaceURI] : this.defaultPrefix]; │ │ │ │ │ - if (group) { │ │ │ │ │ - var local = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ - var reader = group[local] || group["*"]; │ │ │ │ │ - if (reader) { │ │ │ │ │ - reader.apply(this, [node, obj]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: readChildNodes │ │ │ │ │ - * Shorthand for applying the named readers to all children of a node. │ │ │ │ │ - * For each child of type 1 (element), <readSelf> is called. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} The node to be read (required). │ │ │ │ │ - * obj - {Object} The object to be modified (optional). │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The input object, modified. │ │ │ │ │ - */ │ │ │ │ │ - readChildNodes: function(node, obj) { │ │ │ │ │ - if (!obj) { │ │ │ │ │ - obj = {}; │ │ │ │ │ - } │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - var child; │ │ │ │ │ - for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ - child = children[i]; │ │ │ │ │ - if (child.nodeType == 1) { │ │ │ │ │ - this.readNode(child, obj); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: writeNode │ │ │ │ │ - * Shorthand for applying one of the named writers and appending the │ │ │ │ │ - * results to a node. If a qualified name is not provided for the │ │ │ │ │ - * second argument (and a local name is used instead), the namespace │ │ │ │ │ - * of the parent node will be assumed. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} The name of a node to generate. If a qualified name │ │ │ │ │ - * (e.g. "pre:Name") is used, the namespace prefix is assumed to be │ │ │ │ │ - * in the <writers> group. If a local name is used (e.g. "Name") then │ │ │ │ │ - * the namespace of the parent is assumed. If a local name is used │ │ │ │ │ - * and no parent is supplied, then the default namespace is assumed. │ │ │ │ │ - * obj - {Object} Structure containing data for the writer. │ │ │ │ │ - * parent - {DOMElement} Result will be appended to this node. If no parent │ │ │ │ │ - * is supplied, the node will not be appended to anything. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The child node. │ │ │ │ │ - */ │ │ │ │ │ - writeNode: function(name, obj, parent) { │ │ │ │ │ - var prefix, local; │ │ │ │ │ - var split = name.indexOf(":"); │ │ │ │ │ - if (split > 0) { │ │ │ │ │ - prefix = name.substring(0, split); │ │ │ │ │ - local = name.substring(split + 1); │ │ │ │ │ - } else { │ │ │ │ │ - if (parent) { │ │ │ │ │ - prefix = this.namespaceAlias[parent.namespaceURI]; │ │ │ │ │ - } else { │ │ │ │ │ - prefix = this.defaultPrefix; │ │ │ │ │ - } │ │ │ │ │ - local = name; │ │ │ │ │ - } │ │ │ │ │ - var child = this.writers[prefix][local].apply(this, [obj]); │ │ │ │ │ - if (parent) { │ │ │ │ │ - parent.appendChild(child); │ │ │ │ │ - } │ │ │ │ │ - return child; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getChildEl │ │ │ │ │ - * Get the first child element. Optionally only return the first child │ │ │ │ │ - * if it matches the given name and namespace URI. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} The parent node. │ │ │ │ │ - * name - {String} Optional node name (local) to search for. │ │ │ │ │ - * uri - {String} Optional namespace URI to search for. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The first child. Returns null if no element is found, if │ │ │ │ │ - * something significant besides an element is found, or if the element │ │ │ │ │ - * found does not match the optional name and uri. │ │ │ │ │ - */ │ │ │ │ │ - getChildEl: function(node, name, uri) { │ │ │ │ │ - return node && this.getThisOrNextEl(node.firstChild, name, uri); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getNextEl │ │ │ │ │ - * Get the next sibling element. Optionally get the first sibling only │ │ │ │ │ - * if it matches the given local name and namespace URI. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} The node. │ │ │ │ │ - * name - {String} Optional local name of the sibling to search for. │ │ │ │ │ - * uri - {String} Optional namespace URI of the sibling to search for. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The next sibling element. Returns null if no element is │ │ │ │ │ - * found, something significant besides an element is found, or the │ │ │ │ │ - * found element does not match the optional name and uri. │ │ │ │ │ - */ │ │ │ │ │ - getNextEl: function(node, name, uri) { │ │ │ │ │ - return node && this.getThisOrNextEl(node.nextSibling, name, uri); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getThisOrNextEl │ │ │ │ │ - * Return this node or the next element node. Optionally get the first │ │ │ │ │ - * sibling with the given local name or namespace URI. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} The node. │ │ │ │ │ - * name - {String} Optional local name of the sibling to search for. │ │ │ │ │ - * uri - {String} Optional namespace URI of the sibling to search for. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The next sibling element. Returns null if no element is │ │ │ │ │ - * found, something significant besides an element is found, or the │ │ │ │ │ - * found element does not match the query. │ │ │ │ │ - */ │ │ │ │ │ - getThisOrNextEl: function(node, name, uri) { │ │ │ │ │ - outer: for (var sibling = node; sibling; sibling = sibling.nextSibling) { │ │ │ │ │ - switch (sibling.nodeType) { │ │ │ │ │ - case 1: // Element │ │ │ │ │ - if ((!name || name === (sibling.localName || sibling.nodeName.split(":").pop())) && │ │ │ │ │ - (!uri || uri === sibling.namespaceURI)) { │ │ │ │ │ - // matches │ │ │ │ │ - break outer; │ │ │ │ │ - } │ │ │ │ │ - sibling = null; │ │ │ │ │ - break outer; │ │ │ │ │ - case 3: // Text │ │ │ │ │ - if (/^\s*$/.test(sibling.nodeValue)) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - case 4: // CDATA │ │ │ │ │ - case 6: // ENTITY_NODE │ │ │ │ │ - case 12: // NOTATION_NODE │ │ │ │ │ - case 10: // DOCUMENT_TYPE_NODE │ │ │ │ │ - case 11: // DOCUMENT_FRAGMENT_NODE │ │ │ │ │ - sibling = null; │ │ │ │ │ - break outer; │ │ │ │ │ - } // ignore comments and processing instructions │ │ │ │ │ - } │ │ │ │ │ - return sibling || null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: lookupNamespaceURI │ │ │ │ │ - * Takes a prefix and returns the namespace URI associated with it on the given │ │ │ │ │ - * node if found (and null if not). Supplying null for the prefix will │ │ │ │ │ - * return the default namespace. │ │ │ │ │ - * │ │ │ │ │ - * For browsers that support it, this calls the native lookupNamesapceURI │ │ │ │ │ - * function. In other browsers, this is an implementation of │ │ │ │ │ - * http://www.w3.org/TR/DOM-Level-3-Core/core.html#Node3-lookupNamespaceURI. │ │ │ │ │ - * │ │ │ │ │ - * For browsers that don't support the attribute.ownerElement property, this │ │ │ │ │ - * method cannot be called on attribute nodes. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} The node from which to start looking. │ │ │ │ │ - * prefix - {String} The prefix to lookup or null to lookup the default namespace. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The namespace URI for the given prefix. Returns null if the prefix │ │ │ │ │ - * cannot be found or the node is the wrong type. │ │ │ │ │ - */ │ │ │ │ │ - lookupNamespaceURI: function(node, prefix) { │ │ │ │ │ - var uri = null; │ │ │ │ │ - if (node) { │ │ │ │ │ - if (node.lookupNamespaceURI) { │ │ │ │ │ - uri = node.lookupNamespaceURI(prefix); │ │ │ │ │ - } else { │ │ │ │ │ - outer: switch (node.nodeType) { │ │ │ │ │ - case 1: // ELEMENT_NODE │ │ │ │ │ - if (node.namespaceURI !== null && node.prefix === prefix) { │ │ │ │ │ - uri = node.namespaceURI; │ │ │ │ │ - break outer; │ │ │ │ │ - } │ │ │ │ │ - var len = node.attributes.length; │ │ │ │ │ - if (len) { │ │ │ │ │ - var attr; │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - attr = node.attributes[i]; │ │ │ │ │ - if (attr.prefix === "xmlns" && attr.name === "xmlns:" + prefix) { │ │ │ │ │ - uri = attr.value || null; │ │ │ │ │ - break outer; │ │ │ │ │ - } else if (attr.name === "xmlns" && prefix === null) { │ │ │ │ │ - uri = attr.value || null; │ │ │ │ │ - break outer; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - uri = this.lookupNamespaceURI(node.parentNode, prefix); │ │ │ │ │ - break outer; │ │ │ │ │ - case 2: // ATTRIBUTE_NODE │ │ │ │ │ - uri = this.lookupNamespaceURI(node.ownerElement, prefix); │ │ │ │ │ - break outer; │ │ │ │ │ - case 9: // DOCUMENT_NODE │ │ │ │ │ - uri = this.lookupNamespaceURI(node.documentElement, prefix); │ │ │ │ │ - break outer; │ │ │ │ │ - case 6: // ENTITY_NODE │ │ │ │ │ - case 12: // NOTATION_NODE │ │ │ │ │ - case 10: // DOCUMENT_TYPE_NODE │ │ │ │ │ - case 11: // DOCUMENT_FRAGMENT_NODE │ │ │ │ │ - break outer; │ │ │ │ │ - default: │ │ │ │ │ - // TEXT_NODE (3), CDATA_SECTION_NODE (4), ENTITY_REFERENCE_NODE (5), │ │ │ │ │ - // PROCESSING_INSTRUCTION_NODE (7), COMMENT_NODE (8) │ │ │ │ │ - uri = this.lookupNamespaceURI(node.parentNode, prefix); │ │ │ │ │ - break outer; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return uri; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getXMLDoc │ │ │ │ │ - * Get an XML document for nodes that are not supported in HTML (e.g. │ │ │ │ │ - * createCDATASection). On IE, this will either return an existing or │ │ │ │ │ - * create a new <xmldom> on the instance. On other browsers, this will │ │ │ │ │ - * either return an existing or create a new shared document (see │ │ │ │ │ - * <OpenLayers.Format.XML.document>). │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLDocument} │ │ │ │ │ - */ │ │ │ │ │ - getXMLDoc: function() { │ │ │ │ │ - if (!OpenLayers.Format.XML.document && !this.xmldom) { │ │ │ │ │ - if (document.implementation && document.implementation.createDocument) { │ │ │ │ │ - OpenLayers.Format.XML.document = │ │ │ │ │ - document.implementation.createDocument("", "", null); │ │ │ │ │ - } else if (!this.xmldom && window.ActiveXObject) { │ │ │ │ │ - this.xmldom = new ActiveXObject("Microsoft.XMLDOM"); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Format.XML.document || this.xmldom; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.XML" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Format.XML.CONTENT_TYPE = { │ │ │ │ │ - EMPTY: 0, │ │ │ │ │ - SIMPLE: 1, │ │ │ │ │ - COMPLEX: 2, │ │ │ │ │ - MIXED: 3 │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * APIFunction: OpenLayers.Format.XML.lookupNamespaceURI │ │ │ │ │ - * Takes a prefix and returns the namespace URI associated with it on the given │ │ │ │ │ - * node if found (and null if not). Supplying null for the prefix will │ │ │ │ │ - * return the default namespace. │ │ │ │ │ - * │ │ │ │ │ - * For browsers that support it, this calls the native lookupNamesapceURI │ │ │ │ │ - * function. In other browsers, this is an implementation of │ │ │ │ │ - * http://www.w3.org/TR/DOM-Level-3-Core/core.html#Node3-lookupNamespaceURI. │ │ │ │ │ - * │ │ │ │ │ - * For browsers that don't support the attribute.ownerElement property, this │ │ │ │ │ - * method cannot be called on attribute nodes. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} The node from which to start looking. │ │ │ │ │ - * prefix - {String} The prefix to lookup or null to lookup the default namespace. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The namespace URI for the given prefix. Returns null if the prefix │ │ │ │ │ - * cannot be found or the node is the wrong type. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.XML.lookupNamespaceURI = OpenLayers.Function.bind( │ │ │ │ │ - OpenLayers.Format.XML.prototype.lookupNamespaceURI, │ │ │ │ │ - OpenLayers.Format.XML.prototype │ │ │ │ │ -); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Property: OpenLayers.Format.XML.document │ │ │ │ │ - * {XMLDocument} XML document to reuse for creating non-HTML compliant nodes, │ │ │ │ │ - * like document.createCDATASection. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.XML.document = null; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Request.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ - * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * TODO: deprecate me │ │ │ │ │ - * Use OpenLayers.Request.proxy instead. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.ProxyHost = ""; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Request │ │ │ │ │ - * The OpenLayers.Request namespace contains convenience methods for working │ │ │ │ │ - * with XMLHttpRequests. These methods work with a cross-browser │ │ │ │ │ - * W3C compliant <OpenLayers.Request.XMLHttpRequest> class. │ │ │ │ │ - */ │ │ │ │ │ -if (!OpenLayers.Request) { │ │ │ │ │ - /** │ │ │ │ │ - * This allows for OpenLayers/Request/XMLHttpRequest.js to be included │ │ │ │ │ - * before or after this script. │ │ │ │ │ - */ │ │ │ │ │ - OpenLayers.Request = {}; │ │ │ │ │ -} │ │ │ │ │ -OpenLayers.Util.extend(OpenLayers.Request, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: DEFAULT_CONFIG │ │ │ │ │ - * {Object} Default configuration for all requests. │ │ │ │ │ - */ │ │ │ │ │ - DEFAULT_CONFIG: { │ │ │ │ │ - method: "GET", │ │ │ │ │ - url: window.location.href, │ │ │ │ │ - async: true, │ │ │ │ │ - user: undefined, │ │ │ │ │ - password: undefined, │ │ │ │ │ - params: null, │ │ │ │ │ - proxy: OpenLayers.ProxyHost, │ │ │ │ │ - headers: {}, │ │ │ │ │ - data: null, │ │ │ │ │ - callback: function() {}, │ │ │ │ │ - success: null, │ │ │ │ │ - failure: null, │ │ │ │ │ - scope: null │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: URL_SPLIT_REGEX │ │ │ │ │ - */ │ │ │ │ │ - URL_SPLIT_REGEX: /([^:]*:)\/\/([^:]*:?[^@]*@)?([^:\/\?]*):?([^\/\?]*)/, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} An events object that handles all │ │ │ │ │ - * events on the {<OpenLayers.Request>} object. │ │ │ │ │ - * │ │ │ │ │ - * All event listeners will receive an event object with three properties: │ │ │ │ │ - * request - {<OpenLayers.Request.XMLHttpRequest>} The request object. │ │ │ │ │ - * config - {Object} The config object sent to the specific request method. │ │ │ │ │ - * requestUrl - {String} The request url. │ │ │ │ │ - * │ │ │ │ │ - * Supported event types: │ │ │ │ │ - * complete - Triggered when we have a response from the request, if a │ │ │ │ │ - * listener returns false, no further response processing will take │ │ │ │ │ - * place. │ │ │ │ │ - * success - Triggered when the HTTP response has a success code (200-299). │ │ │ │ │ - * failure - Triggered when the HTTP response does not have a success code. │ │ │ │ │ - */ │ │ │ │ │ - events: new OpenLayers.Events(this), │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: makeSameOrigin │ │ │ │ │ - * Using the specified proxy, returns a same origin url of the provided url. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * url - {String} An arbitrary url │ │ │ │ │ - * proxy {String|Function} The proxy to use to make the provided url a │ │ │ │ │ - * same origin url. │ │ │ │ │ - * │ │ │ │ │ - * Returns │ │ │ │ │ - * {String} the same origin url. If no proxy is provided, the returned url │ │ │ │ │ - * will be the same as the provided url. │ │ │ │ │ - */ │ │ │ │ │ - makeSameOrigin: function(url, proxy) { │ │ │ │ │ - var sameOrigin = url.indexOf("http") !== 0; │ │ │ │ │ - var urlParts = !sameOrigin && url.match(this.URL_SPLIT_REGEX); │ │ │ │ │ - if (urlParts) { │ │ │ │ │ - var location = window.location; │ │ │ │ │ - sameOrigin = │ │ │ │ │ - urlParts[1] == location.protocol && │ │ │ │ │ - urlParts[3] == location.hostname; │ │ │ │ │ - var uPort = urlParts[4], │ │ │ │ │ - lPort = location.port; │ │ │ │ │ - if (uPort != 80 && uPort != "" || lPort != "80" && lPort != "") { │ │ │ │ │ - sameOrigin = sameOrigin && uPort == lPort; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!sameOrigin) { │ │ │ │ │ - if (proxy) { │ │ │ │ │ - if (typeof proxy == "function") { │ │ │ │ │ - url = proxy(url); │ │ │ │ │ - } else { │ │ │ │ │ - url = proxy + encodeURIComponent(url); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return url; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: issue │ │ │ │ │ - * Create a new XMLHttpRequest object, open it, set any headers, bind │ │ │ │ │ - * a callback to done state, and send any data. It is recommended that │ │ │ │ │ - * you use one <GET>, <POST>, <PUT>, <DELETE>, <OPTIONS>, or <HEAD>. │ │ │ │ │ - * This method is only documented to provide detail on the configuration │ │ │ │ │ - * options available to all request methods. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Object containing properties for configuring the │ │ │ │ │ - * request. Allowed configuration properties are described below. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Allowed config properties: │ │ │ │ │ - * method - {String} One of GET, POST, PUT, DELETE, HEAD, or │ │ │ │ │ - * OPTIONS. Default is GET. │ │ │ │ │ - * url - {String} URL for the request. │ │ │ │ │ - * async - {Boolean} Open an asynchronous request. Default is true. │ │ │ │ │ - * user - {String} User for relevant authentication scheme. Set │ │ │ │ │ - * to null to clear current user. │ │ │ │ │ - * password - {String} Password for relevant authentication scheme. │ │ │ │ │ - * Set to null to clear current password. │ │ │ │ │ - * proxy - {String} Optional proxy. Defaults to │ │ │ │ │ - * <OpenLayers.ProxyHost>. │ │ │ │ │ - * params - {Object} Any key:value pairs to be appended to the │ │ │ │ │ - * url as a query string. Assumes url doesn't already include a query │ │ │ │ │ - * string or hash. Typically, this is only appropriate for <GET> │ │ │ │ │ - * requests where the query string will be appended to the url. │ │ │ │ │ - * Parameter values that are arrays will be │ │ │ │ │ - * concatenated with a comma (note that this goes against form-encoding) │ │ │ │ │ - * as is done with <OpenLayers.Util.getParameterString>. │ │ │ │ │ - * headers - {Object} Object with header:value pairs to be set on │ │ │ │ │ - * the request. │ │ │ │ │ - * data - {String | Document} Optional data to send with the request. │ │ │ │ │ - * Typically, this is only used with <POST> and <PUT> requests. │ │ │ │ │ - * Make sure to provide the appropriate "Content-Type" header for your │ │ │ │ │ - * data. For <POST> and <PUT> requests, the content type defaults to │ │ │ │ │ - * "application-xml". If your data is a different content type, or │ │ │ │ │ - * if you are using a different HTTP method, set the "Content-Type" │ │ │ │ │ - * header to match your data type. │ │ │ │ │ - * callback - {Function} Function to call when request is done. │ │ │ │ │ - * To determine if the request failed, check request.status (200 │ │ │ │ │ - * indicates success). │ │ │ │ │ - * success - {Function} Optional function to call if request status is in │ │ │ │ │ - * the 200s. This will be called in addition to callback above and │ │ │ │ │ - * would typically only be used as an alternative. │ │ │ │ │ - * failure - {Function} Optional function to call if request status is not │ │ │ │ │ - * in the 200s. This will be called in addition to callback above and │ │ │ │ │ - * would typically only be used as an alternative. │ │ │ │ │ - * scope - {Object} If callback is a public method on some object, │ │ │ │ │ - * set the scope to that object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. To abort the request before a response │ │ │ │ │ - * is received, call abort() on the request object. │ │ │ │ │ - */ │ │ │ │ │ - issue: function(config) { │ │ │ │ │ - // apply default config - proxy host may have changed │ │ │ │ │ - var defaultConfig = OpenLayers.Util.extend( │ │ │ │ │ - this.DEFAULT_CONFIG, { │ │ │ │ │ - proxy: OpenLayers.ProxyHost │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - config = config || {}; │ │ │ │ │ - config.headers = config.headers || {}; │ │ │ │ │ - config = OpenLayers.Util.applyDefaults(config, defaultConfig); │ │ │ │ │ - config.headers = OpenLayers.Util.applyDefaults(config.headers, defaultConfig.headers); │ │ │ │ │ - // Always set the "X-Requested-With" header to signal that this request │ │ │ │ │ - // was issued through the XHR-object. Since header keys are case │ │ │ │ │ - // insensitive and we want to allow overriding of the "X-Requested-With" │ │ │ │ │ - // header through the user we cannot use applyDefaults, but have to │ │ │ │ │ - // check manually whether we were called with a "X-Requested-With" │ │ │ │ │ - // header. │ │ │ │ │ - var customRequestedWithHeader = false, │ │ │ │ │ - headerKey; │ │ │ │ │ - for (headerKey in config.headers) { │ │ │ │ │ - if (config.headers.hasOwnProperty(headerKey)) { │ │ │ │ │ - if (headerKey.toLowerCase() === 'x-requested-with') { │ │ │ │ │ - customRequestedWithHeader = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (customRequestedWithHeader === false) { │ │ │ │ │ - // we did not have a custom "X-Requested-With" header │ │ │ │ │ - config.headers['X-Requested-With'] = 'XMLHttpRequest'; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // create request, open, and set headers │ │ │ │ │ - var request = new OpenLayers.Request.XMLHttpRequest(); │ │ │ │ │ - var url = OpenLayers.Util.urlAppend(config.url, │ │ │ │ │ - OpenLayers.Util.getParameterString(config.params || {})); │ │ │ │ │ - url = OpenLayers.Request.makeSameOrigin(url, config.proxy); │ │ │ │ │ - request.open( │ │ │ │ │ - config.method, url, config.async, config.user, config.password │ │ │ │ │ - ); │ │ │ │ │ - for (var header in config.headers) { │ │ │ │ │ - request.setRequestHeader(header, config.headers[header]); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var events = this.events; │ │ │ │ │ - │ │ │ │ │ - // we want to execute runCallbacks with "this" as the │ │ │ │ │ - // execution scope │ │ │ │ │ - var self = this; │ │ │ │ │ - │ │ │ │ │ - request.onreadystatechange = function() { │ │ │ │ │ - if (request.readyState == OpenLayers.Request.XMLHttpRequest.DONE) { │ │ │ │ │ - var proceed = events.triggerEvent( │ │ │ │ │ - "complete", { │ │ │ │ │ - request: request, │ │ │ │ │ - config: config, │ │ │ │ │ - requestUrl: url │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - if (proceed !== false) { │ │ │ │ │ - self.runCallbacks({ │ │ │ │ │ - request: request, │ │ │ │ │ - config: config, │ │ │ │ │ - requestUrl: url │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // send request (optionally with data) and return │ │ │ │ │ - // call in a timeout for asynchronous requests so the return is │ │ │ │ │ - // available before readyState == 4 for cached docs │ │ │ │ │ - if (config.async === false) { │ │ │ │ │ - request.send(config.data); │ │ │ │ │ - } else { │ │ │ │ │ - window.setTimeout(function() { │ │ │ │ │ - if (request.readyState !== 0) { // W3C: 0-UNSENT │ │ │ │ │ - request.send(config.data); │ │ │ │ │ - } │ │ │ │ │ - }, 0); │ │ │ │ │ - } │ │ │ │ │ - return request; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: runCallbacks │ │ │ │ │ - * Calls the complete, success and failure callbacks. Application │ │ │ │ │ - * can listen to the "complete" event, have the listener │ │ │ │ │ - * display a confirm window and always return false, and │ │ │ │ │ - * execute OpenLayers.Request.runCallbacks if the user │ │ │ │ │ - * hits "yes" in the confirm window. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Hash containing request, config and requestUrl keys │ │ │ │ │ - */ │ │ │ │ │ - runCallbacks: function(options) { │ │ │ │ │ - var request = options.request; │ │ │ │ │ - var config = options.config; │ │ │ │ │ - │ │ │ │ │ - // bind callbacks to readyState 4 (done) │ │ │ │ │ - var complete = (config.scope) ? │ │ │ │ │ - OpenLayers.Function.bind(config.callback, config.scope) : │ │ │ │ │ - config.callback; │ │ │ │ │ - │ │ │ │ │ - // optional success callback │ │ │ │ │ - var success; │ │ │ │ │ - if (config.success) { │ │ │ │ │ - success = (config.scope) ? │ │ │ │ │ - OpenLayers.Function.bind(config.success, config.scope) : │ │ │ │ │ - config.success; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // optional failure callback │ │ │ │ │ - var failure; │ │ │ │ │ - if (config.failure) { │ │ │ │ │ - failure = (config.scope) ? │ │ │ │ │ - OpenLayers.Function.bind(config.failure, config.scope) : │ │ │ │ │ - config.failure; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (OpenLayers.Util.createUrlObject(config.url).protocol == "file:" && │ │ │ │ │ - request.responseText) { │ │ │ │ │ - request.status = 200; │ │ │ │ │ - } │ │ │ │ │ - complete(request); │ │ │ │ │ - │ │ │ │ │ - if (!request.status || (request.status >= 200 && request.status < 300)) { │ │ │ │ │ - this.events.triggerEvent("success", options); │ │ │ │ │ - if (success) { │ │ │ │ │ - success(request); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ - this.events.triggerEvent("failure", options); │ │ │ │ │ - if (failure) { │ │ │ │ │ - failure(request); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: GET │ │ │ │ │ - * Send an HTTP GET request. Additional configuration properties are │ │ │ │ │ - * documented in the <issue> method, with the method property set │ │ │ │ │ - * to GET. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the <issue> method for documentation of allowed properties. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ - */ │ │ │ │ │ - GET: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "GET" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: POST │ │ │ │ │ - * Send a POST request. Additional configuration properties are │ │ │ │ │ - * documented in the <issue> method, with the method property set │ │ │ │ │ - * to POST and "Content-Type" header set to "application/xml". │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the <issue> method for documentation of allowed properties. The │ │ │ │ │ - * default "Content-Type" header will be set to "application-xml" if │ │ │ │ │ - * none is provided. This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ - */ │ │ │ │ │ - POST: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "POST" │ │ │ │ │ - }); │ │ │ │ │ - // set content type to application/xml if it isn't already set │ │ │ │ │ - config.headers = config.headers ? config.headers : {}; │ │ │ │ │ - if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ - config.headers["Content-Type"] = "application/xml"; │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: PUT │ │ │ │ │ - * Send an HTTP PUT request. Additional configuration properties are │ │ │ │ │ - * documented in the <issue> method, with the method property set │ │ │ │ │ - * to PUT and "Content-Type" header set to "application/xml". │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the <issue> method for documentation of allowed properties. The │ │ │ │ │ - * default "Content-Type" header will be set to "application-xml" if │ │ │ │ │ - * none is provided. This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ - */ │ │ │ │ │ - PUT: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "PUT" │ │ │ │ │ - }); │ │ │ │ │ - // set content type to application/xml if it isn't already set │ │ │ │ │ - config.headers = config.headers ? config.headers : {}; │ │ │ │ │ - if (!("CONTENT-TYPE" in OpenLayers.Util.upperCaseObject(config.headers))) { │ │ │ │ │ - config.headers["Content-Type"] = "application/xml"; │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: DELETE │ │ │ │ │ - * Send an HTTP DELETE request. Additional configuration properties are │ │ │ │ │ - * documented in the <issue> method, with the method property set │ │ │ │ │ - * to DELETE. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the <issue> method for documentation of allowed properties. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ - */ │ │ │ │ │ - DELETE: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "DELETE" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: HEAD │ │ │ │ │ - * Send an HTTP HEAD request. Additional configuration properties are │ │ │ │ │ - * documented in the <issue> method, with the method property set │ │ │ │ │ - * to HEAD. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the <issue> method for documentation of allowed properties. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ - */ │ │ │ │ │ - HEAD: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "HEAD" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: OPTIONS │ │ │ │ │ - * Send an HTTP OPTIONS request. Additional configuration properties are │ │ │ │ │ - * documented in the <issue> method, with the method property set │ │ │ │ │ - * to OPTIONS. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Object with properties for configuring the request. │ │ │ │ │ - * See the <issue> method for documentation of allowed properties. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XMLHttpRequest} Request object. │ │ │ │ │ - */ │ │ │ │ │ - OPTIONS: function(config) { │ │ │ │ │ - config = OpenLayers.Util.extend(config, { │ │ │ │ │ - method: "OPTIONS" │ │ │ │ │ - }); │ │ │ │ │ - return OpenLayers.Request.issue(config); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -// XMLHttpRequest.js Copyright (C) 2010 Sergey Ilinsky (http://www.ilinsky.com) │ │ │ │ │ -// │ │ │ │ │ -// Licensed under the Apache License, Version 2.0 (the "License"); │ │ │ │ │ -// you may not use this file except in compliance with the License. │ │ │ │ │ -// You may obtain a copy of the License at │ │ │ │ │ -// │ │ │ │ │ -// http://www.apache.org/licenses/LICENSE-2.0 │ │ │ │ │ -// │ │ │ │ │ -// Unless required by applicable law or agreed to in writing, software │ │ │ │ │ -// distributed under the License is distributed on an "AS IS" BASIS, │ │ │ │ │ -// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. │ │ │ │ │ -// See the License for the specific language governing permissions and │ │ │ │ │ -// limitations under the License. │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Request.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -(function() { │ │ │ │ │ - │ │ │ │ │ - // Save reference to earlier defined object implementation (if any) │ │ │ │ │ - var oXMLHttpRequest = window.XMLHttpRequest; │ │ │ │ │ - │ │ │ │ │ - // Define on browser type │ │ │ │ │ - var bGecko = !!window.controllers, │ │ │ │ │ - bIE = window.document.all && !window.opera, │ │ │ │ │ - bIE7 = bIE && window.navigator.userAgent.match(/MSIE 7.0/); │ │ │ │ │ - │ │ │ │ │ - // Enables "XMLHttpRequest()" call next to "new XMLHttpReques()" │ │ │ │ │ - function fXMLHttpRequest() { │ │ │ │ │ - this._object = oXMLHttpRequest && !bIE7 ? new oXMLHttpRequest : new window.ActiveXObject("Microsoft.XMLHTTP"); │ │ │ │ │ - this._listeners = []; │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // Constructor │ │ │ │ │ - function cXMLHttpRequest() { │ │ │ │ │ - return new fXMLHttpRequest; │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype = fXMLHttpRequest.prototype; │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: Firefox with Firebug installed would break pages if not executed │ │ │ │ │ - if (bGecko && oXMLHttpRequest.wrapped) │ │ │ │ │ - cXMLHttpRequest.wrapped = oXMLHttpRequest.wrapped; │ │ │ │ │ - │ │ │ │ │ - // Constants │ │ │ │ │ - cXMLHttpRequest.UNSENT = 0; │ │ │ │ │ - cXMLHttpRequest.OPENED = 1; │ │ │ │ │ - cXMLHttpRequest.HEADERS_RECEIVED = 2; │ │ │ │ │ - cXMLHttpRequest.LOADING = 3; │ │ │ │ │ - cXMLHttpRequest.DONE = 4; │ │ │ │ │ - │ │ │ │ │ - // Public Properties │ │ │ │ │ - cXMLHttpRequest.prototype.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ - cXMLHttpRequest.prototype.responseText = ''; │ │ │ │ │ - cXMLHttpRequest.prototype.responseXML = null; │ │ │ │ │ - cXMLHttpRequest.prototype.status = 0; │ │ │ │ │ - cXMLHttpRequest.prototype.statusText = ''; │ │ │ │ │ - │ │ │ │ │ - // Priority proposal │ │ │ │ │ - cXMLHttpRequest.prototype.priority = "NORMAL"; │ │ │ │ │ - │ │ │ │ │ - // Instance-level Events Handlers │ │ │ │ │ - cXMLHttpRequest.prototype.onreadystatechange = null; │ │ │ │ │ - │ │ │ │ │ - // Class-level Events Handlers │ │ │ │ │ - cXMLHttpRequest.onreadystatechange = null; │ │ │ │ │ - cXMLHttpRequest.onopen = null; │ │ │ │ │ - cXMLHttpRequest.onsend = null; │ │ │ │ │ - cXMLHttpRequest.onabort = null; │ │ │ │ │ - │ │ │ │ │ - // Public Methods │ │ │ │ │ - cXMLHttpRequest.prototype.open = function(sMethod, sUrl, bAsync, sUser, sPassword) { │ │ │ │ │ - // Delete headers, required when object is reused │ │ │ │ │ - delete this._headers; │ │ │ │ │ - │ │ │ │ │ - // When bAsync parameter value is omitted, use true as default │ │ │ │ │ - if (arguments.length < 3) │ │ │ │ │ - bAsync = true; │ │ │ │ │ - │ │ │ │ │ - // Save async parameter for fixing Gecko bug with missing readystatechange in synchronous requests │ │ │ │ │ - this._async = bAsync; │ │ │ │ │ - │ │ │ │ │ - // Set the onreadystatechange handler │ │ │ │ │ - var oRequest = this, │ │ │ │ │ - nState = this.readyState, │ │ │ │ │ - fOnUnload; │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: IE - memory leak on page unload (inter-page leak) │ │ │ │ │ - if (bIE && bAsync) { │ │ │ │ │ - fOnUnload = function() { │ │ │ │ │ - if (nState != cXMLHttpRequest.DONE) { │ │ │ │ │ - fCleanTransport(oRequest); │ │ │ │ │ - // Safe to abort here since onreadystatechange handler removed │ │ │ │ │ - oRequest.abort(); │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - window.attachEvent("onunload", fOnUnload); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // Add method sniffer │ │ │ │ │ - if (cXMLHttpRequest.onopen) │ │ │ │ │ - cXMLHttpRequest.onopen.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - if (arguments.length > 4) │ │ │ │ │ - this._object.open(sMethod, sUrl, bAsync, sUser, sPassword); │ │ │ │ │ - else │ │ │ │ │ - if (arguments.length > 3) │ │ │ │ │ - this._object.open(sMethod, sUrl, bAsync, sUser); │ │ │ │ │ - else │ │ │ │ │ - this._object.open(sMethod, sUrl, bAsync); │ │ │ │ │ - │ │ │ │ │ - this.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ - fReadyStateChange(this); │ │ │ │ │ - │ │ │ │ │ - this._object.onreadystatechange = function() { │ │ │ │ │ - if (bGecko && !bAsync) │ │ │ │ │ - return; │ │ │ │ │ - │ │ │ │ │ - // Synchronize state │ │ │ │ │ - oRequest.readyState = oRequest._object.readyState; │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - fSynchronizeValues(oRequest); │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: Firefox fires unnecessary DONE when aborting │ │ │ │ │ - if (oRequest._aborted) { │ │ │ │ │ - // Reset readyState to UNSENT │ │ │ │ │ - oRequest.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ - │ │ │ │ │ - // Return now │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (oRequest.readyState == cXMLHttpRequest.DONE) { │ │ │ │ │ - // Free up queue │ │ │ │ │ - delete oRequest._data; │ │ │ │ │ - /* if (bAsync) │ │ │ │ │ - fQueue_remove(oRequest);*/ │ │ │ │ │ - // │ │ │ │ │ - fCleanTransport(oRequest); │ │ │ │ │ - // Uncomment this block if you need a fix for IE cache │ │ │ │ │ - /* │ │ │ │ │ - // BUGFIX: IE - cache issue │ │ │ │ │ - if (!oRequest._object.getResponseHeader("Date")) { │ │ │ │ │ - // Save object to cache │ │ │ │ │ - oRequest._cached = oRequest._object; │ │ │ │ │ - │ │ │ │ │ - // Instantiate a new transport object │ │ │ │ │ - cXMLHttpRequest.call(oRequest); │ │ │ │ │ - │ │ │ │ │ - // Re-send request │ │ │ │ │ - if (sUser) { │ │ │ │ │ - if (sPassword) │ │ │ │ │ - oRequest._object.open(sMethod, sUrl, bAsync, sUser, sPassword); │ │ │ │ │ - else │ │ │ │ │ - oRequest._object.open(sMethod, sUrl, bAsync, sUser); │ │ │ │ │ - } │ │ │ │ │ - else │ │ │ │ │ - oRequest._object.open(sMethod, sUrl, bAsync); │ │ │ │ │ - oRequest._object.setRequestHeader("If-Modified-Since", oRequest._cached.getResponseHeader("Last-Modified") || new window.Date(0)); │ │ │ │ │ - // Copy headers set │ │ │ │ │ - if (oRequest._headers) │ │ │ │ │ - for (var sHeader in oRequest._headers) │ │ │ │ │ - if (typeof oRequest._headers[sHeader] == "string") // Some frameworks prototype objects with functions │ │ │ │ │ - oRequest._object.setRequestHeader(sHeader, oRequest._headers[sHeader]); │ │ │ │ │ - │ │ │ │ │ - oRequest._object.onreadystatechange = function() { │ │ │ │ │ - // Synchronize state │ │ │ │ │ - oRequest.readyState = oRequest._object.readyState; │ │ │ │ │ - │ │ │ │ │ - if (oRequest._aborted) { │ │ │ │ │ - // │ │ │ │ │ - oRequest.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ - │ │ │ │ │ - // Return │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (oRequest.readyState == cXMLHttpRequest.DONE) { │ │ │ │ │ - // Clean Object │ │ │ │ │ - fCleanTransport(oRequest); │ │ │ │ │ - │ │ │ │ │ - // get cached request │ │ │ │ │ - if (oRequest.status == 304) │ │ │ │ │ - oRequest._object = oRequest._cached; │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - delete oRequest._cached; │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - fSynchronizeValues(oRequest); │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - fReadyStateChange(oRequest); │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: IE - memory leak in interrupted │ │ │ │ │ - if (bIE && bAsync) │ │ │ │ │ - window.detachEvent("onunload", fOnUnload); │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - oRequest._object.send(null); │ │ │ │ │ - │ │ │ │ │ - // Return now - wait until re-sent request is finished │ │ │ │ │ - return; │ │ │ │ │ - }; │ │ │ │ │ - */ │ │ │ │ │ - // BUGFIX: IE - memory leak in interrupted │ │ │ │ │ - if (bIE && bAsync) │ │ │ │ │ - window.detachEvent("onunload", fOnUnload); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: Some browsers (Internet Explorer, Gecko) fire OPEN readystate twice │ │ │ │ │ - if (nState != oRequest.readyState) │ │ │ │ │ - fReadyStateChange(oRequest); │ │ │ │ │ - │ │ │ │ │ - nState = oRequest.readyState; │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - function fXMLHttpRequest_send(oRequest) { │ │ │ │ │ - oRequest._object.send(oRequest._data); │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: Gecko - missing readystatechange calls in synchronous requests │ │ │ │ │ - if (bGecko && !oRequest._async) { │ │ │ │ │ - oRequest.readyState = cXMLHttpRequest.OPENED; │ │ │ │ │ - │ │ │ │ │ - // Synchronize state │ │ │ │ │ - fSynchronizeValues(oRequest); │ │ │ │ │ - │ │ │ │ │ - // Simulate missing states │ │ │ │ │ - while (oRequest.readyState < cXMLHttpRequest.DONE) { │ │ │ │ │ - oRequest.readyState++; │ │ │ │ │ - fReadyStateChange(oRequest); │ │ │ │ │ - // Check if we are aborted │ │ │ │ │ - if (oRequest._aborted) │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.send = function(vData) { │ │ │ │ │ - // Add method sniffer │ │ │ │ │ - if (cXMLHttpRequest.onsend) │ │ │ │ │ - cXMLHttpRequest.onsend.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - if (!arguments.length) │ │ │ │ │ - vData = null; │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: Safari - fails sending documents created/modified dynamically, so an explicit serialization required │ │ │ │ │ - // BUGFIX: IE - rewrites any custom mime-type to "text/xml" in case an XMLNode is sent │ │ │ │ │ - // BUGFIX: Gecko - fails sending Element (this is up to the implementation either to standard) │ │ │ │ │ - if (vData && vData.nodeType) { │ │ │ │ │ - vData = window.XMLSerializer ? new window.XMLSerializer().serializeToString(vData) : vData.xml; │ │ │ │ │ - if (!this._headers["Content-Type"]) │ │ │ │ │ - this._object.setRequestHeader("Content-Type", "application/xml"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this._data = vData; │ │ │ │ │ - /* │ │ │ │ │ - // Add to queue │ │ │ │ │ - if (this._async) │ │ │ │ │ - fQueue_add(this); │ │ │ │ │ - else*/ │ │ │ │ │ - fXMLHttpRequest_send(this); │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.abort = function() { │ │ │ │ │ - // Add method sniffer │ │ │ │ │ - if (cXMLHttpRequest.onabort) │ │ │ │ │ - cXMLHttpRequest.onabort.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: Gecko - unnecessary DONE when aborting │ │ │ │ │ - if (this.readyState > cXMLHttpRequest.UNSENT) │ │ │ │ │ - this._aborted = true; │ │ │ │ │ - │ │ │ │ │ - this._object.abort(); │ │ │ │ │ - │ │ │ │ │ - // BUGFIX: IE - memory leak │ │ │ │ │ - fCleanTransport(this); │ │ │ │ │ - │ │ │ │ │ - this.readyState = cXMLHttpRequest.UNSENT; │ │ │ │ │ - │ │ │ │ │ - delete this._data; │ │ │ │ │ - /* if (this._async) │ │ │ │ │ - fQueue_remove(this);*/ │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.getAllResponseHeaders = function() { │ │ │ │ │ - return this._object.getAllResponseHeaders(); │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.getResponseHeader = function(sName) { │ │ │ │ │ - return this._object.getResponseHeader(sName); │ │ │ │ │ - }; │ │ │ │ │ - cXMLHttpRequest.prototype.setRequestHeader = function(sName, sValue) { │ │ │ │ │ - // BUGFIX: IE - cache issue │ │ │ │ │ - if (!this._headers) │ │ │ │ │ - this._headers = {}; │ │ │ │ │ - this._headers[sName] = sValue; │ │ │ │ │ - │ │ │ │ │ - return this._object.setRequestHeader(sName, sValue); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // EventTarget interface implementation │ │ │ │ │ - cXMLHttpRequest.prototype.addEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ - for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ - if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) │ │ │ │ │ - return; │ │ │ │ │ - // Add listener │ │ │ │ │ - this._listeners.push([sName, fHandler, bUseCapture]); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - cXMLHttpRequest.prototype.removeEventListener = function(sName, fHandler, bUseCapture) { │ │ │ │ │ - for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ - if (oListener[0] == sName && oListener[1] == fHandler && oListener[2] == bUseCapture) │ │ │ │ │ - break; │ │ │ │ │ - // Remove listener │ │ │ │ │ - if (oListener) │ │ │ │ │ - this._listeners.splice(nIndex, 1); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - cXMLHttpRequest.prototype.dispatchEvent = function(oEvent) { │ │ │ │ │ - var oEventPseudo = { │ │ │ │ │ - 'type': oEvent.type, │ │ │ │ │ - 'target': this, │ │ │ │ │ - 'currentTarget': this, │ │ │ │ │ - 'eventPhase': 2, │ │ │ │ │ - 'bubbles': oEvent.bubbles, │ │ │ │ │ - 'cancelable': oEvent.cancelable, │ │ │ │ │ - 'timeStamp': oEvent.timeStamp, │ │ │ │ │ - 'stopPropagation': function() {}, // There is no flow │ │ │ │ │ - 'preventDefault': function() {}, // There is no default action │ │ │ │ │ - 'initEvent': function() {} // Original event object should be initialized │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // Execute onreadystatechange │ │ │ │ │ - if (oEventPseudo.type == "readystatechange" && this.onreadystatechange) │ │ │ │ │ - (this.onreadystatechange.handleEvent || this.onreadystatechange).apply(this, [oEventPseudo]); │ │ │ │ │ - │ │ │ │ │ - // Execute listeners │ │ │ │ │ - for (var nIndex = 0, oListener; oListener = this._listeners[nIndex]; nIndex++) │ │ │ │ │ - if (oListener[0] == oEventPseudo.type && !oListener[2]) │ │ │ │ │ - (oListener[1].handleEvent || oListener[1]).apply(this, [oEventPseudo]); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // │ │ │ │ │ - cXMLHttpRequest.prototype.toString = function() { │ │ │ │ │ - return '[' + "object" + ' ' + "XMLHttpRequest" + ']'; │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - cXMLHttpRequest.toString = function() { │ │ │ │ │ - return '[' + "XMLHttpRequest" + ']'; │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // Helper function │ │ │ │ │ - function fReadyStateChange(oRequest) { │ │ │ │ │ - // Sniffing code │ │ │ │ │ - if (cXMLHttpRequest.onreadystatechange) │ │ │ │ │ - cXMLHttpRequest.onreadystatechange.apply(oRequest); │ │ │ │ │ - │ │ │ │ │ - // Fake event │ │ │ │ │ - oRequest.dispatchEvent({ │ │ │ │ │ - 'type': "readystatechange", │ │ │ │ │ - 'bubbles': false, │ │ │ │ │ - 'cancelable': false, │ │ │ │ │ - 'timeStamp': new Date + 0 │ │ │ │ │ - }); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - function fGetDocument(oRequest) { │ │ │ │ │ - var oDocument = oRequest.responseXML, │ │ │ │ │ - sResponse = oRequest.responseText; │ │ │ │ │ - // Try parsing responseText │ │ │ │ │ - if (bIE && sResponse && oDocument && !oDocument.documentElement && oRequest.getResponseHeader("Content-Type").match(/[^\/]+\/[^\+]+\+xml/)) { │ │ │ │ │ - oDocument = new window.ActiveXObject("Microsoft.XMLDOM"); │ │ │ │ │ - oDocument.async = false; │ │ │ │ │ - oDocument.validateOnParse = false; │ │ │ │ │ - oDocument.loadXML(sResponse); │ │ │ │ │ - } │ │ │ │ │ - // Check if there is no error in document │ │ │ │ │ - if (oDocument) │ │ │ │ │ - if ((bIE && oDocument.parseError != 0) || !oDocument.documentElement || (oDocument.documentElement && oDocument.documentElement.tagName == "parsererror")) │ │ │ │ │ - return null; │ │ │ │ │ - return oDocument; │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - function fSynchronizeValues(oRequest) { │ │ │ │ │ - try { │ │ │ │ │ - oRequest.responseText = oRequest._object.responseText; │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - try { │ │ │ │ │ - oRequest.responseXML = fGetDocument(oRequest._object); │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - try { │ │ │ │ │ - oRequest.status = oRequest._object.status; │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - try { │ │ │ │ │ - oRequest.statusText = oRequest._object.statusText; │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - function fCleanTransport(oRequest) { │ │ │ │ │ - // BUGFIX: IE - memory leak (on-page leak) │ │ │ │ │ - oRequest._object.onreadystatechange = new window.Function; │ │ │ │ │ - }; │ │ │ │ │ - /* │ │ │ │ │ - // Queue manager │ │ │ │ │ - var oQueuePending = {"CRITICAL":[],"HIGH":[],"NORMAL":[],"LOW":[],"LOWEST":[]}, │ │ │ │ │ - aQueueRunning = []; │ │ │ │ │ - function fQueue_add(oRequest) { │ │ │ │ │ - oQueuePending[oRequest.priority in oQueuePending ? oRequest.priority : "NORMAL"].push(oRequest); │ │ │ │ │ - // │ │ │ │ │ - setTimeout(fQueue_process); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - function fQueue_remove(oRequest) { │ │ │ │ │ - for (var nIndex = 0, bFound = false; nIndex < aQueueRunning.length; nIndex++) │ │ │ │ │ - if (bFound) │ │ │ │ │ - aQueueRunning[nIndex - 1] = aQueueRunning[nIndex]; │ │ │ │ │ - else │ │ │ │ │ - if (aQueueRunning[nIndex] == oRequest) │ │ │ │ │ - bFound = true; │ │ │ │ │ - if (bFound) │ │ │ │ │ - aQueueRunning.length--; │ │ │ │ │ - // │ │ │ │ │ - setTimeout(fQueue_process); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - function fQueue_process() { │ │ │ │ │ - if (aQueueRunning.length < 6) { │ │ │ │ │ - for (var sPriority in oQueuePending) { │ │ │ │ │ - if (oQueuePending[sPriority].length) { │ │ │ │ │ - var oRequest = oQueuePending[sPriority][0]; │ │ │ │ │ - oQueuePending[sPriority] = oQueuePending[sPriority].slice(1); │ │ │ │ │ - // │ │ │ │ │ - aQueueRunning.push(oRequest); │ │ │ │ │ - // Send request │ │ │ │ │ - fXMLHttpRequest_send(oRequest); │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - */ │ │ │ │ │ - // Internet Explorer 5.0 (missing apply) │ │ │ │ │ - if (!window.Function.prototype.apply) { │ │ │ │ │ - window.Function.prototype.apply = function(oRequest, oArguments) { │ │ │ │ │ - if (!oArguments) │ │ │ │ │ - oArguments = []; │ │ │ │ │ - oRequest.__func = this; │ │ │ │ │ - oRequest.__func(oArguments[0], oArguments[1], oArguments[2], oArguments[3], oArguments[4]); │ │ │ │ │ - delete oRequest.__func; │ │ │ │ │ - }; │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // Register new object with window │ │ │ │ │ - /** │ │ │ │ │ - * Class: OpenLayers.Request.XMLHttpRequest │ │ │ │ │ - * Standard-compliant (W3C) cross-browser implementation of the │ │ │ │ │ - * XMLHttpRequest object. From │ │ │ │ │ - * http://code.google.com/p/xmlhttprequest/. │ │ │ │ │ - */ │ │ │ │ │ - if (!OpenLayers.Request) { │ │ │ │ │ - /** │ │ │ │ │ - * This allows for OpenLayers/Request.js to be included │ │ │ │ │ - * before or after this script. │ │ │ │ │ - */ │ │ │ │ │ - OpenLayers.Request = {}; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Request.XMLHttpRequest = cXMLHttpRequest; │ │ │ │ │ -})(); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ OpenLayers/Format/KML.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -25206,2441 +33673,413 @@ │ │ │ │ │ key = data.getAttribute("name"); │ │ │ │ │ ed['value'] = this.getChildValue(data); │ │ │ │ │ if (this.kvpAttributes) { │ │ │ │ │ attributes[key] = ed['value']; │ │ │ │ │ } else { │ │ │ │ │ ed['displayName'] = key; │ │ │ │ │ attributes[key] = ed; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return attributes; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseProperty │ │ │ │ │ - * Convenience method to find a node and return its value │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xmlNode - {<DOMElement>} │ │ │ │ │ - * namespace - {String} namespace of the node to find │ │ │ │ │ - * tagName - {String} name of the property to parse │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The value for the requested property (defaults to null) │ │ │ │ │ - */ │ │ │ │ │ - parseProperty: function(xmlNode, namespace, tagName) { │ │ │ │ │ - var value; │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(xmlNode, namespace, tagName); │ │ │ │ │ - try { │ │ │ │ │ - value = OpenLayers.Util.getXmlNodeValue(nodeList[0]); │ │ │ │ │ - } catch (e) { │ │ │ │ │ - value = null; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return value; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Accept Feature Collection, and return a string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A KML string. │ │ │ │ │ - */ │ │ │ │ │ - write: function(features) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "kml"); │ │ │ │ │ - var folder = this.createFolderXML(); │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - folder.appendChild(this.createPlacemarkXML(features[i])); │ │ │ │ │ - } │ │ │ │ │ - kml.appendChild(folder); │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [kml]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createFolderXML │ │ │ │ │ - * Creates and returns a KML folder node │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - createFolderXML: function() { │ │ │ │ │ - // Folder │ │ │ │ │ - var folder = this.createElementNS(this.kmlns, "Folder"); │ │ │ │ │ - │ │ │ │ │ - // Folder name │ │ │ │ │ - if (this.foldersName) { │ │ │ │ │ - var folderName = this.createElementNS(this.kmlns, "name"); │ │ │ │ │ - var folderNameText = this.createTextNode(this.foldersName); │ │ │ │ │ - folderName.appendChild(folderNameText); │ │ │ │ │ - folder.appendChild(folderName); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // Folder description │ │ │ │ │ - if (this.foldersDesc) { │ │ │ │ │ - var folderDesc = this.createElementNS(this.kmlns, "description"); │ │ │ │ │ - var folderDescText = this.createTextNode(this.foldersDesc); │ │ │ │ │ - folderDesc.appendChild(folderDescText); │ │ │ │ │ - folder.appendChild(folderDesc); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return folder; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createPlacemarkXML │ │ │ │ │ - * Creates and returns a KML placemark node representing the given feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - createPlacemarkXML: function(feature) { │ │ │ │ │ - // Placemark name │ │ │ │ │ - var placemarkName = this.createElementNS(this.kmlns, "name"); │ │ │ │ │ - var label = (feature.style && feature.style.label) ? feature.style.label : feature.id; │ │ │ │ │ - var name = feature.attributes.name || label; │ │ │ │ │ - placemarkName.appendChild(this.createTextNode(name)); │ │ │ │ │ - │ │ │ │ │ - // Placemark description │ │ │ │ │ - var placemarkDesc = this.createElementNS(this.kmlns, "description"); │ │ │ │ │ - var desc = feature.attributes.description || this.placemarksDesc; │ │ │ │ │ - placemarkDesc.appendChild(this.createTextNode(desc)); │ │ │ │ │ - │ │ │ │ │ - // Placemark │ │ │ │ │ - var placemarkNode = this.createElementNS(this.kmlns, "Placemark"); │ │ │ │ │ - if (feature.fid != null) { │ │ │ │ │ - placemarkNode.setAttribute("id", feature.fid); │ │ │ │ │ - } │ │ │ │ │ - placemarkNode.appendChild(placemarkName); │ │ │ │ │ - placemarkNode.appendChild(placemarkDesc); │ │ │ │ │ - │ │ │ │ │ - // Geometry node (Point, LineString, etc. nodes) │ │ │ │ │ - var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ - placemarkNode.appendChild(geometryNode); │ │ │ │ │ - │ │ │ │ │ - // output attributes as extendedData │ │ │ │ │ - if (feature.attributes) { │ │ │ │ │ - var edNode = this.buildExtendedData(feature.attributes); │ │ │ │ │ - if (edNode) { │ │ │ │ │ - placemarkNode.appendChild(edNode); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return placemarkNode; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometryNode │ │ │ │ │ - * Builds and returns a KML geometry node with the given geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - buildGeometryNode: function(geometry) { │ │ │ │ │ - var className = geometry.CLASS_NAME; │ │ │ │ │ - var type = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ - var builder = this.buildGeometry[type.toLowerCase()]; │ │ │ │ │ - var node = null; │ │ │ │ │ - if (builder) { │ │ │ │ │ - node = builder.apply(this, [geometry]); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: buildGeometry │ │ │ │ │ - * Object containing methods to do the actual geometry node building │ │ │ │ │ - * based on geometry type. │ │ │ │ │ - */ │ │ │ │ │ - buildGeometry: { │ │ │ │ │ - // TBD: Anybody care about namespace aliases here (these nodes have │ │ │ │ │ - // no prefixes)? │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.point │ │ │ │ │ - * Given an OpenLayers point geometry, create a KML point. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Point>} A point geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML point node. │ │ │ │ │ - */ │ │ │ │ │ - point: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "Point"); │ │ │ │ │ - kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - return kml; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.multipoint │ │ │ │ │ - * Given an OpenLayers multipoint geometry, create a KML │ │ │ │ │ - * GeometryCollection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Point>} A multipoint geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ - */ │ │ │ │ │ - multipoint: function(geometry) { │ │ │ │ │ - return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.linestring │ │ │ │ │ - * Given an OpenLayers linestring geometry, create a KML linestring. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.LineString>} A linestring geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML linestring node. │ │ │ │ │ - */ │ │ │ │ │ - linestring: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "LineString"); │ │ │ │ │ - kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - return kml; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.multilinestring │ │ │ │ │ - * Given an OpenLayers multilinestring geometry, create a KML │ │ │ │ │ - * GeometryCollection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Point>} A multilinestring geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ - */ │ │ │ │ │ - multilinestring: function(geometry) { │ │ │ │ │ - return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.linearring │ │ │ │ │ - * Given an OpenLayers linearring geometry, create a KML linearring. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.LinearRing>} A linearring geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML linearring node. │ │ │ │ │ - */ │ │ │ │ │ - linearring: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "LinearRing"); │ │ │ │ │ - kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - return kml; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.polygon │ │ │ │ │ - * Given an OpenLayers polygon geometry, create a KML polygon. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Polygon>} A polygon geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML polygon node. │ │ │ │ │ - */ │ │ │ │ │ - polygon: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "Polygon"); │ │ │ │ │ - var rings = geometry.components; │ │ │ │ │ - var ringMember, ringGeom, type; │ │ │ │ │ - for (var i = 0, len = rings.length; i < len; ++i) { │ │ │ │ │ - type = (i == 0) ? "outerBoundaryIs" : "innerBoundaryIs"; │ │ │ │ │ - ringMember = this.createElementNS(this.kmlns, type); │ │ │ │ │ - ringGeom = this.buildGeometry.linearring.apply(this, │ │ │ │ │ - [rings[i]]); │ │ │ │ │ - ringMember.appendChild(ringGeom); │ │ │ │ │ - kml.appendChild(ringMember); │ │ │ │ │ - } │ │ │ │ │ - return kml; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.multipolygon │ │ │ │ │ - * Given an OpenLayers multipolygon geometry, create a KML │ │ │ │ │ - * GeometryCollection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Point>} A multipolygon geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ - */ │ │ │ │ │ - multipolygon: function(geometry) { │ │ │ │ │ - return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.collection │ │ │ │ │ - * Given an OpenLayers geometry collection, create a KML MultiGeometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Collection>} A geometry collection. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML MultiGeometry node. │ │ │ │ │ - */ │ │ │ │ │ - collection: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "MultiGeometry"); │ │ │ │ │ - var child; │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; ++i) { │ │ │ │ │ - child = this.buildGeometryNode.apply(this, │ │ │ │ │ - [geometry.components[i]]); │ │ │ │ │ - if (child) { │ │ │ │ │ - kml.appendChild(child); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return kml; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildCoordinatesNode │ │ │ │ │ - * Builds and returns the KML coordinates node with the given geometry │ │ │ │ │ - * <coordinates>...</coordinates> │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - buildCoordinatesNode: function(geometry) { │ │ │ │ │ - var coordinatesNode = this.createElementNS(this.kmlns, "coordinates"); │ │ │ │ │ - │ │ │ │ │ - var path; │ │ │ │ │ - var points = geometry.components; │ │ │ │ │ - if (points) { │ │ │ │ │ - // LineString or LinearRing │ │ │ │ │ - var point; │ │ │ │ │ - var numPoints = points.length; │ │ │ │ │ - var parts = new Array(numPoints); │ │ │ │ │ - for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ - point = points[i]; │ │ │ │ │ - parts[i] = this.buildCoordinates(point); │ │ │ │ │ - } │ │ │ │ │ - path = parts.join(" "); │ │ │ │ │ - } else { │ │ │ │ │ - // Point │ │ │ │ │ - path = this.buildCoordinates(geometry); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var txtNode = this.createTextNode(path); │ │ │ │ │ - coordinatesNode.appendChild(txtNode); │ │ │ │ │ - │ │ │ │ │ - return coordinatesNode; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildCoordinates │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ - * │ │ │ │ │ - * Returns │ │ │ │ │ - * {String} a coordinate pair │ │ │ │ │ - */ │ │ │ │ │ - buildCoordinates: function(point) { │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - point = point.clone(); │ │ │ │ │ - point.transform(this.internalProjection, │ │ │ │ │ - this.externalProjection); │ │ │ │ │ - } │ │ │ │ │ - return point.x + "," + point.y; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildExtendedData │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * attributes - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns │ │ │ │ │ - * {DOMElement} A KML ExtendedData node or {null} if no attributes. │ │ │ │ │ - */ │ │ │ │ │ - buildExtendedData: function(attributes) { │ │ │ │ │ - var extendedData = this.createElementNS(this.kmlns, "ExtendedData"); │ │ │ │ │ - for (var attributeName in attributes) { │ │ │ │ │ - // empty, name, description, styleUrl attributes ignored │ │ │ │ │ - if (attributes[attributeName] && attributeName != "name" && attributeName != "description" && attributeName != "styleUrl") { │ │ │ │ │ - var data = this.createElementNS(this.kmlns, "Data"); │ │ │ │ │ - data.setAttribute("name", attributeName); │ │ │ │ │ - var value = this.createElementNS(this.kmlns, "value"); │ │ │ │ │ - if (typeof attributes[attributeName] == "object") { │ │ │ │ │ - // cater for object attributes with 'value' properties │ │ │ │ │ - // other object properties will output an empty node │ │ │ │ │ - if (attributes[attributeName].value) { │ │ │ │ │ - value.appendChild(this.createTextNode(attributes[attributeName].value)); │ │ │ │ │ - } │ │ │ │ │ - if (attributes[attributeName].displayName) { │ │ │ │ │ - var displayName = this.createElementNS(this.kmlns, "displayName"); │ │ │ │ │ - // displayName always written as CDATA │ │ │ │ │ - displayName.appendChild(this.getXMLDoc().createCDATASection(attributes[attributeName].displayName)); │ │ │ │ │ - data.appendChild(displayName); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - value.appendChild(this.createTextNode(attributes[attributeName])); │ │ │ │ │ - } │ │ │ │ │ - data.appendChild(value); │ │ │ │ │ - extendedData.appendChild(data); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.isSimpleContent(extendedData)) { │ │ │ │ │ - return null; │ │ │ │ │ - } else { │ │ │ │ │ - return extendedData; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.KML" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Handler │ │ │ │ │ - * Base class to construct a higher-level handler for event sequences. All │ │ │ │ │ - * handlers have activate and deactivate methods. In addition, they have │ │ │ │ │ - * methods named like browser events. When a handler is activated, any │ │ │ │ │ - * additional methods named like a browser event is registered as a │ │ │ │ │ - * listener for the corresponding event. When a handler is deactivated, │ │ │ │ │ - * those same methods are unregistered as event listeners. │ │ │ │ │ - * │ │ │ │ │ - * Handlers also typically have a callbacks object with keys named like │ │ │ │ │ - * the abstracted events or event sequences that they are in charge of │ │ │ │ │ - * handling. The controls that wrap handlers define the methods that │ │ │ │ │ - * correspond to these abstract events - so instead of listening for │ │ │ │ │ - * individual browser events, they only listen for the abstract events │ │ │ │ │ - * defined by the handler. │ │ │ │ │ - * │ │ │ │ │ - * Handlers are created by controls, which ultimately have the responsibility │ │ │ │ │ - * of making changes to the the state of the application. Handlers │ │ │ │ │ - * themselves may make temporary changes, but in general are expected to │ │ │ │ │ - * return the application in the same state that they found it. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: id │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - id: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: control │ │ │ │ │ - * {<OpenLayers.Control>}. The control that initialized this handler. The │ │ │ │ │ - * control is assumed to have a valid map property - that map is used │ │ │ │ │ - * in the handler's own setMap method. │ │ │ │ │ - */ │ │ │ │ │ - control: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: map │ │ │ │ │ - * {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - map: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: keyMask │ │ │ │ │ - * {Integer} Use bitwise operators and one or more of the OpenLayers.Handler │ │ │ │ │ - * constants to construct a keyMask. The keyMask is used by │ │ │ │ │ - * <checkModifiers>. If the keyMask matches the combination of keys │ │ │ │ │ - * down on an event, checkModifiers returns true. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * // handler only responds if the Shift key is down │ │ │ │ │ - * handler.keyMask = OpenLayers.Handler.MOD_SHIFT; │ │ │ │ │ - * │ │ │ │ │ - * // handler only responds if Ctrl-Shift is down │ │ │ │ │ - * handler.keyMask = OpenLayers.Handler.MOD_SHIFT | │ │ │ │ │ - * OpenLayers.Handler.MOD_CTRL; │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ - keyMask: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: active │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - active: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: evt │ │ │ │ │ - * {Event} This property references the last event handled by the handler. │ │ │ │ │ - * Note that this property is not part of the stable API. Use of the │ │ │ │ │ - * evt property should be restricted to controls in the library │ │ │ │ │ - * or other applications that are willing to update with changes to │ │ │ │ │ - * the OpenLayers code. │ │ │ │ │ - */ │ │ │ │ │ - evt: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: touch │ │ │ │ │ - * {Boolean} Indicates the support of touch events. When touch events are │ │ │ │ │ - * started touch will be true and all mouse related listeners will do │ │ │ │ │ - * nothing. │ │ │ │ │ - */ │ │ │ │ │ - touch: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Handler │ │ │ │ │ - * Construct a handler. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that initialized this │ │ │ │ │ - * handler. The control is assumed to have a valid map property; that │ │ │ │ │ - * map is used in the handler's own setMap method. If a map property │ │ │ │ │ - * is present in the options argument it will be used instead. │ │ │ │ │ - * callbacks - {Object} An object whose properties correspond to abstracted │ │ │ │ │ - * events or sequences of browser events. The values for these │ │ │ │ │ - * properties are functions defined by the control that get called by │ │ │ │ │ - * the handler. │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * the handler. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.control = control; │ │ │ │ │ - this.callbacks = callbacks; │ │ │ │ │ - │ │ │ │ │ - var map = this.map || control.map; │ │ │ │ │ - if (map) { │ │ │ │ │ - this.setMap(map); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.map = map; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: checkModifiers │ │ │ │ │ - * Check the keyMask on the handler. If no <keyMask> is set, this always │ │ │ │ │ - * returns true. If a <keyMask> is set and it matches the combination │ │ │ │ │ - * of keys down on an event, this returns true. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The keyMask matches the keys down on an event. │ │ │ │ │ - */ │ │ │ │ │ - checkModifiers: function(evt) { │ │ │ │ │ - if (this.keyMask == null) { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - /* calculate the keyboard modifier mask for this event */ │ │ │ │ │ - var keyModifiers = │ │ │ │ │ - (evt.shiftKey ? OpenLayers.Handler.MOD_SHIFT : 0) | │ │ │ │ │ - (evt.ctrlKey ? OpenLayers.Handler.MOD_CTRL : 0) | │ │ │ │ │ - (evt.altKey ? OpenLayers.Handler.MOD_ALT : 0) | │ │ │ │ │ - (evt.metaKey ? OpenLayers.Handler.MOD_META : 0); │ │ │ │ │ - │ │ │ │ │ - /* if it differs from the handler object's key mask, │ │ │ │ │ - bail out of the event handler */ │ │ │ │ │ - return (keyModifiers == this.keyMask); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Turn on the handler. Returns false if the handler was already active. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - // register for event handlers defined on this class. │ │ │ │ │ - var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.register(events[i], this[events[i]]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.active = true; │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Turn off the handler. Returns false if the handler was already inactive. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - // unregister event handlers defined on this class. │ │ │ │ │ - var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.unregister(events[i], this[events[i]]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.touch = false; │ │ │ │ │ - this.active = false; │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: startTouch │ │ │ │ │ - * Start touch events, this method must be called by subclasses in │ │ │ │ │ - * "touchstart" method. When touch events are started <touch> will be │ │ │ │ │ - * true and all mouse related listeners will do nothing. │ │ │ │ │ - */ │ │ │ │ │ - startTouch: function() { │ │ │ │ │ - if (!this.touch) { │ │ │ │ │ - this.touch = true; │ │ │ │ │ - var events = [ │ │ │ │ │ - "mousedown", "mouseup", "mousemove", "click", "dblclick", │ │ │ │ │ - "mouseout" │ │ │ │ │ - ]; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.unregister(events[i], this[events[i]]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: callback │ │ │ │ │ - * Trigger the control's named callback with the given arguments │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} The key for the callback that is one of the properties │ │ │ │ │ - * of the handler's callbacks object. │ │ │ │ │ - * args - {Array(*)} An array of arguments (any type) with which to call │ │ │ │ │ - * the callback (defined by the control). │ │ │ │ │ - */ │ │ │ │ │ - callback: function(name, args) { │ │ │ │ │ - if (name && this.callbacks[name]) { │ │ │ │ │ - this.callbacks[name].apply(this.control, args); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: register │ │ │ │ │ - * register an event on the map │ │ │ │ │ - */ │ │ │ │ │ - register: function(name, method) { │ │ │ │ │ - // TODO: deal with registerPriority in 3.0 │ │ │ │ │ - this.map.events.registerPriority(name, this, method); │ │ │ │ │ - this.map.events.registerPriority(name, this, this.setEvent); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: unregister │ │ │ │ │ - * unregister an event from the map │ │ │ │ │ - */ │ │ │ │ │ - unregister: function(name, method) { │ │ │ │ │ - this.map.events.unregister(name, this, method); │ │ │ │ │ - this.map.events.unregister(name, this, this.setEvent); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setEvent │ │ │ │ │ - * With each registered browser event, the handler sets its own evt │ │ │ │ │ - * property. This property can be accessed by controls if needed │ │ │ │ │ - * to get more information about the event that the handler is │ │ │ │ │ - * processing. │ │ │ │ │ - * │ │ │ │ │ - * This allows modifier keys on the event to be checked (alt, shift, ctrl, │ │ │ │ │ - * and meta cannot be checked with the keyboard handler). For a │ │ │ │ │ - * control to determine which modifier keys are associated with the │ │ │ │ │ - * event that a handler is currently processing, it should access │ │ │ │ │ - * (code)handler.evt.altKey || handler.evt.shiftKey || │ │ │ │ │ - * handler.evt.ctrlKey || handler.evt.metaKey(end). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event. │ │ │ │ │ - */ │ │ │ │ │ - setEvent: function(evt) { │ │ │ │ │ - this.evt = evt; │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Deconstruct the handler. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - // unregister event listeners │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - // eliminate circular references │ │ │ │ │ - this.control = this.map = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_NONE │ │ │ │ │ - * If set as the <keyMask>, <checkModifiers> returns false if any key is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_NONE = 0; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_SHIFT │ │ │ │ │ - * If set as the <keyMask>, <checkModifiers> returns false if Shift is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_SHIFT = 1; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_CTRL │ │ │ │ │ - * If set as the <keyMask>, <checkModifiers> returns false if Ctrl is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_CTRL = 2; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_ALT │ │ │ │ │ - * If set as the <keyMask>, <checkModifiers> returns false if Alt is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_ALT = 4; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_META │ │ │ │ │ - * If set as the <keyMask>, <checkModifiers> returns false if Cmd is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_META = 8; │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Point.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Handler.Point │ │ │ │ │ - * Handler to draw a point on the map. Point is displayed on activation, │ │ │ │ │ - * moves on mouse move, and is finished on mouse up. The handler triggers │ │ │ │ │ - * callbacks for 'done', 'cancel', and 'modify'. The modify callback is │ │ │ │ │ - * called with each change in the sketch and will receive the latest point │ │ │ │ │ - * drawn. Create a new instance with the <OpenLayers.Handler.Point> │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.Point = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: point │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} The currently drawn point │ │ │ │ │ - */ │ │ │ │ │ - point: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} The temporary drawing layer │ │ │ │ │ - */ │ │ │ │ │ - layer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: multi │ │ │ │ │ - * {Boolean} Cast features to multi-part geometries before passing to the │ │ │ │ │ - * layer. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - multi: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: citeCompliant │ │ │ │ │ - * {Boolean} If set to true, coordinates of features drawn in a map extent │ │ │ │ │ - * crossing the date line won't exceed the world bounds. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - citeCompliant: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: mouseDown │ │ │ │ │ - * {Boolean} The mouse is down │ │ │ │ │ - */ │ │ │ │ │ - mouseDown: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: stoppedDown │ │ │ │ │ - * {Boolean} Indicate whether the last mousedown stopped the event │ │ │ │ │ - * propagation. │ │ │ │ │ - */ │ │ │ │ │ - stoppedDown: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: lastDown │ │ │ │ │ - * {<OpenLayers.Pixel>} Location of the last mouse down │ │ │ │ │ - */ │ │ │ │ │ - lastDown: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: lastUp │ │ │ │ │ - * {<OpenLayers.Pixel>} │ │ │ │ │ - */ │ │ │ │ │ - lastUp: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: persist │ │ │ │ │ - * {Boolean} Leave the feature rendered until destroyFeature is called. │ │ │ │ │ - * Default is false. If set to true, the feature remains rendered until │ │ │ │ │ - * destroyFeature is called, typically by deactivating the handler or │ │ │ │ │ - * starting another drawing. │ │ │ │ │ - */ │ │ │ │ │ - persist: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: stopDown │ │ │ │ │ - * {Boolean} Stop event propagation on mousedown. Must be false to │ │ │ │ │ - * allow "pan while drawing". Defaults to false. │ │ │ │ │ - */ │ │ │ │ │ - stopDown: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIPropery: stopUp │ │ │ │ │ - * {Boolean} Stop event propagation on mouse. Must be false to │ │ │ │ │ - * allow "pan while dragging". Defaults to fase. │ │ │ │ │ - */ │ │ │ │ │ - stopUp: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layerOptions │ │ │ │ │ - * {Object} Any optional properties to be set on the sketch layer. │ │ │ │ │ - */ │ │ │ │ │ - layerOptions: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: pixelTolerance │ │ │ │ │ - * {Number} Maximum number of pixels between down and up (mousedown │ │ │ │ │ - * and mouseup, or touchstart and touchend) for the handler to │ │ │ │ │ - * add a new point. If set to an integer value, if the │ │ │ │ │ - * displacement between down and up is great to this value │ │ │ │ │ - * no point will be added. Default value is 5. │ │ │ │ │ - */ │ │ │ │ │ - pixelTolerance: 5, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: lastTouchPx │ │ │ │ │ - * {<OpenLayers.Pixel>} The last pixel used to know the distance between │ │ │ │ │ - * two touches (for double touch). │ │ │ │ │ - */ │ │ │ │ │ - lastTouchPx: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Point │ │ │ │ │ - * Create a new point handler. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ - * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ - * functions. Various callbacks described below. │ │ │ │ │ - * options - {Object} An optional object with properties to be set on the │ │ │ │ │ - * handler │ │ │ │ │ - * │ │ │ │ │ - * Named callbacks: │ │ │ │ │ - * create - Called when a sketch is first created. Callback called with │ │ │ │ │ - * the creation point geometry and sketch feature. │ │ │ │ │ - * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ - * geometry and the sketch feature. │ │ │ │ │ - * done - Called when the point drawing is finished. The callback will │ │ │ │ │ - * recieve a single argument, the point geometry. │ │ │ │ │ - * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ - * cancel callback will receive a geometry. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ - this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style['default'], {}); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * turn on the handler │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (!OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - // create temporary vector layer for rendering geometry sketch │ │ │ │ │ - // TBD: this could be moved to initialize/destroy - setting visibility here │ │ │ │ │ - var options = OpenLayers.Util.extend({ │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - // indicate that the temp vector layer will never be out of range │ │ │ │ │ - // without this, resolution properties must be specified at the │ │ │ │ │ - // map-level for this temporary layer to init its resolutions │ │ │ │ │ - // correctly │ │ │ │ │ - calculateInRange: OpenLayers.Function.True, │ │ │ │ │ - wrapDateLine: this.citeCompliant │ │ │ │ │ - }, this.layerOptions); │ │ │ │ │ - this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ - this.map.addLayer(this.layer); │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createFeature │ │ │ │ │ - * Add temporary features │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} A pixel location on the map. │ │ │ │ │ - */ │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ - lonlat.lon, lonlat.lat │ │ │ │ │ - ); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.point]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.point], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * turn off the handler │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (!OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - this.cancel(); │ │ │ │ │ - // If a layer's map property is set to null, it means that that layer │ │ │ │ │ - // isn't added to the map. Since we ourself added the layer to the map │ │ │ │ │ - // in activate(), we can assume that if this.layer.map is null it means │ │ │ │ │ - // that the layer has been destroyed (as a result of map.destroy() for │ │ │ │ │ - // example. │ │ │ │ │ - if (this.layer.map != null) { │ │ │ │ │ - this.destroyFeature(true); │ │ │ │ │ - this.layer.destroy(false); │ │ │ │ │ - } │ │ │ │ │ - this.layer = null; │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroyFeature │ │ │ │ │ - * Destroy the temporary geometries │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ - */ │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - if (this.layer && (force || !this.persist)) { │ │ │ │ │ - this.layer.destroyFeatures(); │ │ │ │ │ - } │ │ │ │ │ - this.point = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroyPersistedFeature │ │ │ │ │ - * Destroy the persisted feature. │ │ │ │ │ - */ │ │ │ │ │ - destroyPersistedFeature: function() { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - if (layer && layer.features.length > 1) { │ │ │ │ │ - this.layer.features[0].destroy(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: finalize │ │ │ │ │ - * Finish the geometry and call the "done" callback. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * cancel - {Boolean} Call cancel instead of done callback. Default │ │ │ │ │ - * is false. │ │ │ │ │ - */ │ │ │ │ │ - finalize: function(cancel) { │ │ │ │ │ - var key = cancel ? "cancel" : "done"; │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - this.lastDown = null; │ │ │ │ │ - this.lastUp = null; │ │ │ │ │ - this.lastTouchPx = null; │ │ │ │ │ - this.callback(key, [this.geometryClone()]); │ │ │ │ │ - this.destroyFeature(cancel); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: cancel │ │ │ │ │ - * Finish the geometry and call the "cancel" callback. │ │ │ │ │ - */ │ │ │ │ │ - cancel: function() { │ │ │ │ │ - this.finalize(true); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: click │ │ │ │ │ - * Handle clicks. Clicks are stopped from propagating to other listeners │ │ │ │ │ - * on map.events or other dom elements. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: dblclick │ │ │ │ │ - * Handle double-clicks. Double-clicks are stopped from propagating to other │ │ │ │ │ - * listeners on map.events or other dom elements. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: modifyFeature │ │ │ │ │ - * Modify the existing geometry given a pixel location. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} A pixel location on the map. │ │ │ │ │ - */ │ │ │ │ │ - modifyFeature: function(pixel) { │ │ │ │ │ - if (!this.point) { │ │ │ │ │ - this.createFeature(pixel); │ │ │ │ │ - } │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point.geometry.x = lonlat.lon; │ │ │ │ │ - this.point.geometry.y = lonlat.lat; │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.point, false]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawFeature │ │ │ │ │ - * Render features on the temporary layer. │ │ │ │ │ - */ │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.point, this.style); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getGeometry │ │ │ │ │ - * Return the sketch geometry. If <multi> is true, this will return │ │ │ │ │ - * a multi-part geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} │ │ │ │ │ - */ │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.point && this.point.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiPoint([geometry]); │ │ │ │ │ - } │ │ │ │ │ - return geometry; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: geometryClone │ │ │ │ │ - * Return a clone of the relevant geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} │ │ │ │ │ - */ │ │ │ │ │ - geometryClone: function() { │ │ │ │ │ - var geom = this.getGeometry(); │ │ │ │ │ - return geom && geom.clone(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mousedown │ │ │ │ │ - * Handle mousedown. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - return this.down(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchstart │ │ │ │ │ - * Handle touchstart. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - this.lastTouchPx = evt.xy; │ │ │ │ │ - return this.down(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mousemove │ │ │ │ │ - * Handle mousemove. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - return this.move(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchmove │ │ │ │ │ - * Handle touchmove. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - this.lastTouchPx = evt.xy; │ │ │ │ │ - return this.move(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mouseup │ │ │ │ │ - * Handle mouseup. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - return this.up(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchend │ │ │ │ │ - * Handle touchend. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - evt.xy = this.lastTouchPx; │ │ │ │ │ - return this.up(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: down │ │ │ │ │ - * Handle mousedown and touchstart. Adjust the geometry and redraw. │ │ │ │ │ - * Return determines whether to propagate the event on the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - down: function(evt) { │ │ │ │ │ - this.mouseDown = true; │ │ │ │ │ - this.lastDown = evt.xy; │ │ │ │ │ - if (!this.touch) { // no point displayed until up on touch devices │ │ │ │ │ - this.modifyFeature(evt.xy); │ │ │ │ │ - } │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - return !this.stopDown; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * Handle mousemove and touchmove. Adjust the geometry and redraw. │ │ │ │ │ - * Return determines whether to propagate the event on the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - if (!this.touch // no point displayed until up on touch devices │ │ │ │ │ - && │ │ │ │ │ - (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ - this.modifyFeature(evt.xy); │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: up │ │ │ │ │ - * Handle mouseup and touchend. Send the latest point in the geometry to the control. │ │ │ │ │ - * Return determines whether to propagate the event on the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - up: function(evt) { │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - │ │ │ │ │ - // check keyboard modifiers │ │ │ │ │ - if (!this.checkModifiers(evt)) { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - // ignore double-clicks │ │ │ │ │ - if (this.lastUp && this.lastUp.equals(evt.xy)) { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - if (this.lastDown && this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ - this.pixelTolerance)) { │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy); │ │ │ │ │ - } │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature(); │ │ │ │ │ - } │ │ │ │ │ - this.lastUp = evt.xy; │ │ │ │ │ - this.finalize(); │ │ │ │ │ - return !this.stopUp; │ │ │ │ │ - } else { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mouseout │ │ │ │ │ - * Handle mouse out. For better user experience reset mouseDown │ │ │ │ │ - * and stoppedDown when the mouse leaves the map viewport. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - */ │ │ │ │ │ - mouseout: function(evt) { │ │ │ │ │ - if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: passesTolerance │ │ │ │ │ - * Determine whether the event is within the optional pixel tolerance. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The event is within the pixel tolerance (if specified). │ │ │ │ │ - */ │ │ │ │ │ - passesTolerance: function(pixel1, pixel2, tolerance) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - │ │ │ │ │ - if (tolerance != null && pixel1 && pixel2) { │ │ │ │ │ - var dist = pixel1.distanceTo(pixel2); │ │ │ │ │ - if (dist > tolerance) { │ │ │ │ │ - passes = false; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return passes; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Point" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Path.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Handler/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Handler.Path │ │ │ │ │ - * Handler to draw a path on the map. Path is displayed on mouse down, │ │ │ │ │ - * moves on mouse move, and is finished on mouse up. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler.Point> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.Path = OpenLayers.Class(OpenLayers.Handler.Point, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: line │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - line: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: maxVertices │ │ │ │ │ - * {Number} The maximum number of vertices which can be drawn by this │ │ │ │ │ - * handler. When the number of vertices reaches maxVertices, the │ │ │ │ │ - * geometry is automatically finalized. Default is null. │ │ │ │ │ - */ │ │ │ │ │ - maxVertices: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: doubleTouchTolerance │ │ │ │ │ - * {Number} Maximum number of pixels between two touches for │ │ │ │ │ - * the gesture to be considered a "finalize feature" action. │ │ │ │ │ - * Default is 20. │ │ │ │ │ - */ │ │ │ │ │ - doubleTouchTolerance: 20, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: freehand │ │ │ │ │ - * {Boolean} In freehand mode, the handler starts the path on mouse down, │ │ │ │ │ - * adds a point for every mouse move, and finishes the path on mouse up. │ │ │ │ │ - * Outside of freehand mode, a point is added to the path on every mouse │ │ │ │ │ - * click and double-click finishes the path. │ │ │ │ │ - */ │ │ │ │ │ - freehand: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: freehandToggle │ │ │ │ │ - * {String} If set, freehandToggle is checked on mouse events and will set │ │ │ │ │ - * the freehand mode to the opposite of this.freehand. To disallow │ │ │ │ │ - * toggling between freehand and non-freehand mode, set freehandToggle to │ │ │ │ │ - * null. Acceptable toggle values are 'shiftKey', 'ctrlKey', and 'altKey'. │ │ │ │ │ - */ │ │ │ │ │ - freehandToggle: 'shiftKey', │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: timerId │ │ │ │ │ - * {Integer} The timer used to test the double touch. │ │ │ │ │ - */ │ │ │ │ │ - timerId: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: redoStack │ │ │ │ │ - * {Array} Stack containing points removed with <undo>. │ │ │ │ │ - */ │ │ │ │ │ - redoStack: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Path │ │ │ │ │ - * Create a new path hander │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ - * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ - * functions. Various callbacks described below. │ │ │ │ │ - * options - {Object} An optional object with properties to be set on the │ │ │ │ │ - * handler │ │ │ │ │ - * │ │ │ │ │ - * Named callbacks: │ │ │ │ │ - * create - Called when a sketch is first created. Callback called with │ │ │ │ │ - * the creation point geometry and sketch feature. │ │ │ │ │ - * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ - * geometry and the sketch feature. │ │ │ │ │ - * point - Called as each point is added. Receives the new point geometry. │ │ │ │ │ - * done - Called when the point drawing is finished. The callback will │ │ │ │ │ - * recieve a single argument, the linestring geometry. │ │ │ │ │ - * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ - * cancel callback will receive a geometry. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createFeature │ │ │ │ │ - * Add temporary geometries │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} The initial pixel location for the new │ │ │ │ │ - * feature. │ │ │ │ │ - */ │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ - lonlat.lon, lonlat.lat │ │ │ │ │ - ); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.line = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.LineString([this.point.geometry]) │ │ │ │ │ - ); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.line, this.point], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroyFeature │ │ │ │ │ - * Destroy temporary geometries │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ - */ │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - OpenLayers.Handler.Point.prototype.destroyFeature.call( │ │ │ │ │ - this, force); │ │ │ │ │ - this.line = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroyPersistedFeature │ │ │ │ │ - * Destroy the persisted feature. │ │ │ │ │ - */ │ │ │ │ │ - destroyPersistedFeature: function() { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - if (layer && layer.features.length > 2) { │ │ │ │ │ - this.layer.features[0].destroy(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: removePoint │ │ │ │ │ - * Destroy the temporary point. │ │ │ │ │ - */ │ │ │ │ │ - removePoint: function() { │ │ │ │ │ - if (this.point) { │ │ │ │ │ - this.layer.removeFeatures([this.point]); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addPoint │ │ │ │ │ - * Add point to geometry. Send the point index to override │ │ │ │ │ - * the behavior of LinearRing that disregards adding duplicate points. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} The pixel location for the new point. │ │ │ │ │ - */ │ │ │ │ │ - addPoint: function(pixel) { │ │ │ │ │ - this.layer.removeFeatures([this.point]); │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat) │ │ │ │ │ - ); │ │ │ │ │ - this.line.geometry.addComponent( │ │ │ │ │ - this.point.geometry, this.line.geometry.components.length │ │ │ │ │ - ); │ │ │ │ │ - this.layer.addFeatures([this.point]); │ │ │ │ │ - this.callback("point", [this.point.geometry, this.getGeometry()]); │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - delete this.redoStack; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: insertXY │ │ │ │ │ - * Insert a point in the current sketch given x & y coordinates. The new │ │ │ │ │ - * point is inserted immediately before the most recently drawn point. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {Number} The x-coordinate of the point. │ │ │ │ │ - * y - {Number} The y-coordinate of the point. │ │ │ │ │ - */ │ │ │ │ │ - insertXY: function(x, y) { │ │ │ │ │ - this.line.geometry.addComponent( │ │ │ │ │ - new OpenLayers.Geometry.Point(x, y), │ │ │ │ │ - this.getCurrentPointIndex() │ │ │ │ │ - ); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - delete this.redoStack; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: insertDeltaXY │ │ │ │ │ - * Insert a point given offsets from the previously inserted point. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * dx - {Number} The x-coordinate offset of the point. │ │ │ │ │ - * dy - {Number} The y-coordinate offset of the point. │ │ │ │ │ - */ │ │ │ │ │ - insertDeltaXY: function(dx, dy) { │ │ │ │ │ - var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ - var p0 = this.line.geometry.components[previousIndex]; │ │ │ │ │ - if (p0 && !isNaN(p0.x) && !isNaN(p0.y)) { │ │ │ │ │ - this.insertXY(p0.x + dx, p0.y + dy); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: insertDirectionLength │ │ │ │ │ - * Insert a point in the current sketch given a direction and a length. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * direction - {Number} Degrees clockwise from the positive x-axis. │ │ │ │ │ - * length - {Number} Distance from the previously drawn point. │ │ │ │ │ - */ │ │ │ │ │ - insertDirectionLength: function(direction, length) { │ │ │ │ │ - direction *= Math.PI / 180; │ │ │ │ │ - var dx = length * Math.cos(direction); │ │ │ │ │ - var dy = length * Math.sin(direction); │ │ │ │ │ - this.insertDeltaXY(dx, dy); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: insertDeflectionLength │ │ │ │ │ - * Insert a point in the current sketch given a deflection and a length. │ │ │ │ │ - * The deflection should be degrees clockwise from the previously │ │ │ │ │ - * digitized segment. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * deflection - {Number} Degrees clockwise from the previous segment. │ │ │ │ │ - * length - {Number} Distance from the previously drawn point. │ │ │ │ │ - */ │ │ │ │ │ - insertDeflectionLength: function(deflection, length) { │ │ │ │ │ - var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ - if (previousIndex > 0) { │ │ │ │ │ - var p1 = this.line.geometry.components[previousIndex]; │ │ │ │ │ - var p0 = this.line.geometry.components[previousIndex - 1]; │ │ │ │ │ - var theta = Math.atan2(p1.y - p0.y, p1.x - p0.x); │ │ │ │ │ - this.insertDirectionLength( │ │ │ │ │ - (theta * 180 / Math.PI) + deflection, length │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getCurrentPointIndex │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number} The index of the most recently drawn point. │ │ │ │ │ - */ │ │ │ │ │ - getCurrentPointIndex: function() { │ │ │ │ │ - return this.line.geometry.components.length - 1; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: undo │ │ │ │ │ - * Remove the most recently added point in the sketch geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} A point was removed. │ │ │ │ │ - */ │ │ │ │ │ - undo: function() { │ │ │ │ │ - var geometry = this.line.geometry; │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var index = this.getCurrentPointIndex() - 1; │ │ │ │ │ - var target = components[index]; │ │ │ │ │ - var undone = geometry.removeComponent(target); │ │ │ │ │ - if (undone) { │ │ │ │ │ - // On touch devices, set the current ("mouse location") point to │ │ │ │ │ - // match the last digitized point. │ │ │ │ │ - if (this.touch && index > 0) { │ │ │ │ │ - components = geometry.components; // safety │ │ │ │ │ - var lastpt = components[index - 1]; │ │ │ │ │ - var curptidx = this.getCurrentPointIndex(); │ │ │ │ │ - var curpt = components[curptidx]; │ │ │ │ │ - curpt.x = lastpt.x; │ │ │ │ │ - curpt.y = lastpt.y; │ │ │ │ │ - } │ │ │ │ │ - if (!this.redoStack) { │ │ │ │ │ - this.redoStack = []; │ │ │ │ │ - } │ │ │ │ │ - this.redoStack.push(target); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - } │ │ │ │ │ - return undone; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: redo │ │ │ │ │ - * Reinsert the most recently removed point resulting from an <undo> call. │ │ │ │ │ - * The undo stack is deleted whenever a point is added by other means. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} A point was added. │ │ │ │ │ - */ │ │ │ │ │ - redo: function() { │ │ │ │ │ - var target = this.redoStack && this.redoStack.pop(); │ │ │ │ │ - if (target) { │ │ │ │ │ - this.line.geometry.addComponent(target, this.getCurrentPointIndex()); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - } │ │ │ │ │ - return !!target; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: freehandMode │ │ │ │ │ - * Determine whether to behave in freehand mode or not. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - freehandMode: function(evt) { │ │ │ │ │ - return (this.freehandToggle && evt[this.freehandToggle]) ? │ │ │ │ │ - !this.freehand : this.freehand; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: modifyFeature │ │ │ │ │ - * Modify the existing geometry given the new point │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} The updated pixel location for the latest │ │ │ │ │ - * point. │ │ │ │ │ - * drawing - {Boolean} Indicate if we're currently drawing. │ │ │ │ │ - */ │ │ │ │ │ - modifyFeature: function(pixel, drawing) { │ │ │ │ │ - if (!this.line) { │ │ │ │ │ - this.createFeature(pixel); │ │ │ │ │ - } │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point.geometry.x = lonlat.lon; │ │ │ │ │ - this.point.geometry.y = lonlat.lat; │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.getSketch(), drawing]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawFeature │ │ │ │ │ - * Render geometries on the temporary layer. │ │ │ │ │ - */ │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.line, this.style); │ │ │ │ │ - this.layer.drawFeature(this.point, this.style); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getSketch │ │ │ │ │ - * Return the sketch feature. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - getSketch: function() { │ │ │ │ │ - return this.line; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getGeometry │ │ │ │ │ - * Return the sketch geometry. If <multi> is true, this will return │ │ │ │ │ - * a multi-part geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.LineString>} │ │ │ │ │ - */ │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.line && this.line.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiLineString([geometry]); │ │ │ │ │ - } │ │ │ │ │ - return geometry; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * method: touchstart │ │ │ │ │ - * handle touchstart. │ │ │ │ │ - * │ │ │ │ │ - * parameters: │ │ │ │ │ - * evt - {event} the browser event │ │ │ │ │ - * │ │ │ │ │ - * returns: │ │ │ │ │ - * {boolean} allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - if (this.timerId && │ │ │ │ │ - this.passesTolerance(this.lastTouchPx, evt.xy, │ │ │ │ │ - this.doubleTouchTolerance)) { │ │ │ │ │ - // double-tap, finalize the geometry │ │ │ │ │ - this.finishGeometry(); │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - return false; │ │ │ │ │ - } else { │ │ │ │ │ - if (this.timerId) { │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - } │ │ │ │ │ - this.timerId = window.setTimeout( │ │ │ │ │ - OpenLayers.Function.bind(function() { │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - }, this), 300); │ │ │ │ │ - return OpenLayers.Handler.Point.prototype.touchstart.call(this, evt); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: down │ │ │ │ │ - * Handle mousedown and touchstart. Add a new point to the geometry and │ │ │ │ │ - * render it. Return determines whether to propagate the event on the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - down: function(evt) { │ │ │ │ │ - var stopDown = this.stopDown; │ │ │ │ │ - if (this.freehandMode(evt)) { │ │ │ │ │ - stopDown = true; │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!this.touch && (!this.lastDown || │ │ │ │ │ - !this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ - this.pixelTolerance))) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ - } │ │ │ │ │ - this.mouseDown = true; │ │ │ │ │ - this.lastDown = evt.xy; │ │ │ │ │ - this.stoppedDown = stopDown; │ │ │ │ │ - return !stopDown; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * Handle mousemove and touchmove. Adjust the geometry and redraw. │ │ │ │ │ - * Return determines whether to propagate the event on the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature(); │ │ │ │ │ - } │ │ │ │ │ - if (this.maxVertices && this.line && │ │ │ │ │ - this.line.geometry.components.length === this.maxVertices) { │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize(); │ │ │ │ │ - } else { │ │ │ │ │ - this.addPoint(evt.xy); │ │ │ │ │ - } │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: up │ │ │ │ │ - * Handle mouseup and touchend. Send the latest point in the geometry to │ │ │ │ │ - * the control. Return determines whether to propagate the event on the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - up: function(evt) { │ │ │ │ │ - if (this.mouseDown && (!this.lastUp || !this.lastUp.equals(evt.xy))) { │ │ │ │ │ - if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature(); │ │ │ │ │ - } │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize(); │ │ │ │ │ - } else { │ │ │ │ │ - if (this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ - this.pixelTolerance)) { │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy); │ │ │ │ │ - } │ │ │ │ │ - if (this.lastUp == null && this.persist) { │ │ │ │ │ - this.destroyPersistedFeature(); │ │ │ │ │ - } │ │ │ │ │ - this.addPoint(evt.xy); │ │ │ │ │ - this.lastUp = evt.xy; │ │ │ │ │ - if (this.line.geometry.components.length === this.maxVertices + 1) { │ │ │ │ │ - this.finishGeometry(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - return !this.stopUp; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: finishGeometry │ │ │ │ │ - * Finish the geometry and send it back to the control. │ │ │ │ │ - */ │ │ │ │ │ - finishGeometry: function() { │ │ │ │ │ - var index = this.line.geometry.components.length - 1; │ │ │ │ │ - this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: dblclick │ │ │ │ │ - * Handle double-clicks. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - if (!this.freehandMode(evt)) { │ │ │ │ │ - this.finishGeometry(); │ │ │ │ │ - } │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Path" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Polygon.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Handler/Path.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Handler.Polygon │ │ │ │ │ - * Handler to draw a polygon on the map. Polygon is displayed on mouse down, │ │ │ │ │ - * moves on mouse move, and is finished on mouse up. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler.Path> │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.Polygon = OpenLayers.Class(OpenLayers.Handler.Path, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: holeModifier │ │ │ │ │ - * {String} Key modifier to trigger hole digitizing. Acceptable values are │ │ │ │ │ - * "altKey", "shiftKey", or "ctrlKey". If not set, no hole digitizing │ │ │ │ │ - * will take place. Default is null. │ │ │ │ │ - */ │ │ │ │ │ - holeModifier: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: drawingHole │ │ │ │ │ - * {Boolean} Currently drawing an interior ring. │ │ │ │ │ - */ │ │ │ │ │ - drawingHole: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: polygon │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - polygon: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Polygon │ │ │ │ │ - * Create a Polygon Handler. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ - * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ - * functions. Various callbacks described below. │ │ │ │ │ - * options - {Object} An optional object with properties to be set on the │ │ │ │ │ - * handler │ │ │ │ │ - * │ │ │ │ │ - * Named callbacks: │ │ │ │ │ - * create - Called when a sketch is first created. Callback called with │ │ │ │ │ - * the creation point geometry and sketch feature. │ │ │ │ │ - * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ - * geometry and the sketch feature. │ │ │ │ │ - * point - Called as each point is added. Receives the new point geometry. │ │ │ │ │ - * done - Called when the point drawing is finished. The callback will │ │ │ │ │ - * recieve a single argument, the polygon geometry. │ │ │ │ │ - * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ - * cancel callback will receive a geometry. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createFeature │ │ │ │ │ - * Add temporary geometries │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} The initial pixel location for the new │ │ │ │ │ - * feature. │ │ │ │ │ - */ │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ - lonlat.lon, lonlat.lat │ │ │ │ │ - ); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.line = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.LinearRing([this.point.geometry]) │ │ │ │ │ - ); │ │ │ │ │ - this.polygon = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.Polygon([this.line.geometry]) │ │ │ │ │ - ); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.polygon, this.point], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addPoint │ │ │ │ │ - * Add point to geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} The pixel location for the new point. │ │ │ │ │ - */ │ │ │ │ │ - addPoint: function(pixel) { │ │ │ │ │ - if (!this.drawingHole && this.holeModifier && │ │ │ │ │ - this.evt && this.evt[this.holeModifier]) { │ │ │ │ │ - var geometry = this.point.geometry; │ │ │ │ │ - var features = this.control.layer.features; │ │ │ │ │ - var candidate, polygon; │ │ │ │ │ - // look for intersections, last drawn gets priority │ │ │ │ │ - for (var i = features.length - 1; i >= 0; --i) { │ │ │ │ │ - candidate = features[i].geometry; │ │ │ │ │ - if ((candidate instanceof OpenLayers.Geometry.Polygon || │ │ │ │ │ - candidate instanceof OpenLayers.Geometry.MultiPolygon) && │ │ │ │ │ - candidate.intersects(geometry)) { │ │ │ │ │ - polygon = features[i]; │ │ │ │ │ - this.control.layer.removeFeatures([polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.control.layer.events.registerPriority( │ │ │ │ │ - "sketchcomplete", this, this.finalizeInteriorRing │ │ │ │ │ - ); │ │ │ │ │ - this.control.layer.events.registerPriority( │ │ │ │ │ - "sketchmodified", this, this.enforceTopology │ │ │ │ │ - ); │ │ │ │ │ - polygon.geometry.addComponent(this.line.geometry); │ │ │ │ │ - this.polygon = polygon; │ │ │ │ │ - this.drawingHole = true; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Handler.Path.prototype.addPoint.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getCurrentPointIndex │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number} The index of the most recently drawn point. │ │ │ │ │ - */ │ │ │ │ │ - getCurrentPointIndex: function() { │ │ │ │ │ - return this.line.geometry.components.length - 2; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: enforceTopology │ │ │ │ │ - * Simple topology enforcement for drawing interior rings. Ensures vertices │ │ │ │ │ - * of interior rings are contained by exterior ring. Other topology │ │ │ │ │ - * rules are enforced in <finalizeInteriorRing> to allow drawing of │ │ │ │ │ - * rings that intersect only during the sketch (e.g. a "C" shaped ring │ │ │ │ │ - * that nearly encloses another ring). │ │ │ │ │ - */ │ │ │ │ │ - enforceTopology: function(event) { │ │ │ │ │ - var point = event.vertex; │ │ │ │ │ - var components = this.line.geometry.components; │ │ │ │ │ - // ensure that vertices of interior ring are contained by exterior ring │ │ │ │ │ - if (!this.polygon.geometry.intersects(point)) { │ │ │ │ │ - var last = components[components.length - 3]; │ │ │ │ │ - point.x = last.x; │ │ │ │ │ - point.y = last.y; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: finishGeometry │ │ │ │ │ - * Finish the geometry and send it back to the control. │ │ │ │ │ - */ │ │ │ │ │ - finishGeometry: function() { │ │ │ │ │ - var index = this.line.geometry.components.length - 2; │ │ │ │ │ - this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: finalizeInteriorRing │ │ │ │ │ - * Enforces that new ring has some area and doesn't contain vertices of any │ │ │ │ │ - * other rings. │ │ │ │ │ - */ │ │ │ │ │ - finalizeInteriorRing: function() { │ │ │ │ │ - var ring = this.line.geometry; │ │ │ │ │ - // ensure that ring has some area │ │ │ │ │ - var modified = (ring.getArea() !== 0); │ │ │ │ │ - if (modified) { │ │ │ │ │ - // ensure that new ring doesn't intersect any other rings │ │ │ │ │ - var rings = this.polygon.geometry.components; │ │ │ │ │ - for (var i = rings.length - 2; i >= 0; --i) { │ │ │ │ │ - if (ring.intersects(rings[i])) { │ │ │ │ │ - modified = false; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (modified) { │ │ │ │ │ - // ensure that new ring doesn't contain any other rings │ │ │ │ │ - var target; │ │ │ │ │ - outer: for (var i = rings.length - 2; i > 0; --i) { │ │ │ │ │ - var points = rings[i].components; │ │ │ │ │ - for (var j = 0, jj = points.length; j < jj; ++j) { │ │ │ │ │ - if (ring.containsPoint(points[j])) { │ │ │ │ │ - modified = false; │ │ │ │ │ - break outer; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (modified) { │ │ │ │ │ - if (this.polygon.state !== OpenLayers.State.INSERT) { │ │ │ │ │ - this.polygon.state = OpenLayers.State.UPDATE; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.polygon.geometry.removeComponent(ring); │ │ │ │ │ - } │ │ │ │ │ - this.restoreFeature(); │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: cancel │ │ │ │ │ - * Finish the geometry and call the "cancel" callback. │ │ │ │ │ - */ │ │ │ │ │ - cancel: function() { │ │ │ │ │ - if (this.drawingHole) { │ │ │ │ │ - this.polygon.geometry.removeComponent(this.line.geometry); │ │ │ │ │ - this.restoreFeature(true); │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Handler.Path.prototype.cancel.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: restoreFeature │ │ │ │ │ - * Move the feature from the sketch layer to the target layer. │ │ │ │ │ - * │ │ │ │ │ - * Properties: │ │ │ │ │ - * cancel - {Boolean} Cancel drawing. If falsey, the "sketchcomplete" event │ │ │ │ │ - * will be fired. │ │ │ │ │ - */ │ │ │ │ │ - restoreFeature: function(cancel) { │ │ │ │ │ - this.control.layer.events.unregister( │ │ │ │ │ - "sketchcomplete", this, this.finalizeInteriorRing │ │ │ │ │ - ); │ │ │ │ │ - this.control.layer.events.unregister( │ │ │ │ │ - "sketchmodified", this, this.enforceTopology │ │ │ │ │ - ); │ │ │ │ │ - this.layer.removeFeatures([this.polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.control.layer.addFeatures([this.polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.drawingHole = false; │ │ │ │ │ - if (!cancel) { │ │ │ │ │ - // Re-trigger "sketchcomplete" so other listeners can do their │ │ │ │ │ - // business. While this is somewhat sloppy (if a listener is │ │ │ │ │ - // registered with registerPriority - not common - between the start │ │ │ │ │ - // and end of a single ring drawing - very uncommon - it will be │ │ │ │ │ - // called twice). │ │ │ │ │ - // TODO: In 3.0, collapse sketch handlers into geometry specific │ │ │ │ │ - // drawing controls. │ │ │ │ │ - this.control.layer.events.triggerEvent( │ │ │ │ │ - "sketchcomplete", { │ │ │ │ │ - feature: this.polygon │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroyFeature │ │ │ │ │ - * Destroy temporary geometries │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ - */ │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - OpenLayers.Handler.Path.prototype.destroyFeature.call( │ │ │ │ │ - this, force); │ │ │ │ │ - this.polygon = null; │ │ │ │ │ - }, │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawFeature │ │ │ │ │ - * Render geometries on the temporary layer. │ │ │ │ │ - */ │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.polygon, this.style); │ │ │ │ │ - this.layer.drawFeature(this.point, this.style); │ │ │ │ │ + return attributes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getSketch │ │ │ │ │ - * Return the sketch feature. │ │ │ │ │ + * Method: parseProperty │ │ │ │ │ + * Convenience method to find a node and return its value │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * xmlNode - {<DOMElement>} │ │ │ │ │ + * namespace - {String} namespace of the node to find │ │ │ │ │ + * tagName - {String} name of the property to parse │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * {String} The value for the requested property (defaults to null) │ │ │ │ │ */ │ │ │ │ │ - getSketch: function() { │ │ │ │ │ - return this.polygon; │ │ │ │ │ + parseProperty: function(xmlNode, namespace, tagName) { │ │ │ │ │ + var value; │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(xmlNode, namespace, tagName); │ │ │ │ │ + try { │ │ │ │ │ + value = OpenLayers.Util.getXmlNodeValue(nodeList[0]); │ │ │ │ │ + } catch (e) { │ │ │ │ │ + value = null; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return value; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getGeometry │ │ │ │ │ - * Return the sketch geometry. If <multi> is true, this will return │ │ │ │ │ - * a multi-part geometry. │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Accept Feature Collection, and return a string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Polygon>} │ │ │ │ │ + * {String} A KML string. │ │ │ │ │ */ │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.polygon && this.polygon.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiPolygon([geometry]); │ │ │ │ │ + write: function(features) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ } │ │ │ │ │ - return geometry; │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "kml"); │ │ │ │ │ + var folder = this.createFolderXML(); │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + folder.appendChild(this.createPlacemarkXML(features[i])); │ │ │ │ │ + } │ │ │ │ │ + kml.appendChild(folder); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [kml]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Polygon" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: createFolderXML │ │ │ │ │ + * Creates and returns a KML folder node │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + createFolderXML: function() { │ │ │ │ │ + // Folder │ │ │ │ │ + var folder = this.createElementNS(this.kmlns, "Folder"); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - */ │ │ │ │ │ + // Folder name │ │ │ │ │ + if (this.foldersName) { │ │ │ │ │ + var folderName = this.createElementNS(this.kmlns, "name"); │ │ │ │ │ + var folderNameText = this.createTextNode(this.foldersName); │ │ │ │ │ + folderName.appendChild(folderNameText); │ │ │ │ │ + folder.appendChild(folderName); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Strategy │ │ │ │ │ - * Abstract vector layer strategy class. Not to be instantiated directly. Use │ │ │ │ │ - * one of the strategy subclasses instead. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Strategy = OpenLayers.Class({ │ │ │ │ │ + // Folder description │ │ │ │ │ + if (this.foldersDesc) { │ │ │ │ │ + var folderDesc = this.createElementNS(this.kmlns, "description"); │ │ │ │ │ + var folderDescText = this.createTextNode(this.foldersDesc); │ │ │ │ │ + folderDesc.appendChild(folderDescText); │ │ │ │ │ + folder.appendChild(folderDesc); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} The layer this strategy belongs to. │ │ │ │ │ - */ │ │ │ │ │ - layer: null, │ │ │ │ │ + return folder; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: options │ │ │ │ │ - * {Object} Any options sent to the constructor. │ │ │ │ │ + * Method: createPlacemarkXML │ │ │ │ │ + * Creates and returns a KML placemark node representing the given feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - options: null, │ │ │ │ │ + createPlacemarkXML: function(feature) { │ │ │ │ │ + // Placemark name │ │ │ │ │ + var placemarkName = this.createElementNS(this.kmlns, "name"); │ │ │ │ │ + var label = (feature.style && feature.style.label) ? feature.style.label : feature.id; │ │ │ │ │ + var name = feature.attributes.name || label; │ │ │ │ │ + placemarkName.appendChild(this.createTextNode(name)); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: active │ │ │ │ │ - * {Boolean} The control is active. │ │ │ │ │ - */ │ │ │ │ │ - active: null, │ │ │ │ │ + // Placemark description │ │ │ │ │ + var placemarkDesc = this.createElementNS(this.kmlns, "description"); │ │ │ │ │ + var desc = feature.attributes.description || this.placemarksDesc; │ │ │ │ │ + placemarkDesc.appendChild(this.createTextNode(desc)); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: autoActivate │ │ │ │ │ - * {Boolean} The creator of the strategy can set autoActivate to false │ │ │ │ │ - * to fully control when the protocol is activated and deactivated. │ │ │ │ │ - * Defaults to true. │ │ │ │ │ - */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ + // Placemark │ │ │ │ │ + var placemarkNode = this.createElementNS(this.kmlns, "Placemark"); │ │ │ │ │ + if (feature.fid != null) { │ │ │ │ │ + placemarkNode.setAttribute("id", feature.fid); │ │ │ │ │ + } │ │ │ │ │ + placemarkNode.appendChild(placemarkName); │ │ │ │ │ + placemarkNode.appendChild(placemarkDesc); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: autoDestroy │ │ │ │ │ - * {Boolean} The creator of the strategy can set autoDestroy to false │ │ │ │ │ - * to fully control when the strategy is destroyed. Defaults to │ │ │ │ │ - * true. │ │ │ │ │ - */ │ │ │ │ │ - autoDestroy: true, │ │ │ │ │ + // Geometry node (Point, LineString, etc. nodes) │ │ │ │ │ + var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ + placemarkNode.appendChild(geometryNode); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy │ │ │ │ │ - * Abstract class for vector strategies. Create instances of a subclass. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.options = options; │ │ │ │ │ - // set the active property here, so that user cannot override it │ │ │ │ │ - this.active = false; │ │ │ │ │ - }, │ │ │ │ │ + // output attributes as extendedData │ │ │ │ │ + if (feature.attributes) { │ │ │ │ │ + var edNode = this.buildExtendedData(feature.attributes); │ │ │ │ │ + if (edNode) { │ │ │ │ │ + placemarkNode.appendChild(edNode); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up the strategy. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.layer = null; │ │ │ │ │ - this.options = null; │ │ │ │ │ + return placemarkNode; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setLayer │ │ │ │ │ - * Called to set the <layer> property. │ │ │ │ │ - * │ │ │ │ │ + * Method: buildGeometryNode │ │ │ │ │ + * Builds and returns a KML geometry node with the given geometry. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - setLayer: function(layer) { │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ - * │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} True if the strategy was successfully activated or false if │ │ │ │ │ - * the strategy was already active. │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - this.active = true; │ │ │ │ │ - return true; │ │ │ │ │ + buildGeometryNode: function(geometry) { │ │ │ │ │ + var className = geometry.CLASS_NAME; │ │ │ │ │ + var type = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ + var builder = this.buildGeometry[type.toLowerCase()]; │ │ │ │ │ + var node = null; │ │ │ │ │ + if (builder) { │ │ │ │ │ + node = builder.apply(this, [geometry]); │ │ │ │ │ } │ │ │ │ │ - return false; │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ - * tear-down. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} True if the strategy was successfully deactivated or false if │ │ │ │ │ - * the strategy was already inactive. │ │ │ │ │ + * Property: buildGeometry │ │ │ │ │ + * Object containing methods to do the actual geometry node building │ │ │ │ │ + * based on geometry type. │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.active = false; │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ + buildGeometry: { │ │ │ │ │ + // TBD: Anybody care about namespace aliases here (these nodes have │ │ │ │ │ + // no prefixes)? │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy/Fixed.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.point │ │ │ │ │ + * Given an OpenLayers point geometry, create a KML point. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Point>} A point geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML point node. │ │ │ │ │ + */ │ │ │ │ │ + point: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "Point"); │ │ │ │ │ + kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + return kml; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.multipoint │ │ │ │ │ + * Given an OpenLayers multipoint geometry, create a KML │ │ │ │ │ + * GeometryCollection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Point>} A multipoint geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ + */ │ │ │ │ │ + multipoint: function(geometry) { │ │ │ │ │ + return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Strategy.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.linestring │ │ │ │ │ + * Given an OpenLayers linestring geometry, create a KML linestring. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.LineString>} A linestring geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML linestring node. │ │ │ │ │ + */ │ │ │ │ │ + linestring: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "LineString"); │ │ │ │ │ + kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + return kml; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Strategy.Fixed │ │ │ │ │ - * A simple strategy that requests features once and never requests new data. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Strategy> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Strategy.Fixed = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.multilinestring │ │ │ │ │ + * Given an OpenLayers multilinestring geometry, create a KML │ │ │ │ │ + * GeometryCollection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Point>} A multilinestring geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ + */ │ │ │ │ │ + multilinestring: function(geometry) { │ │ │ │ │ + return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: preload │ │ │ │ │ - * {Boolean} Load data before layer made visible. Enabling this may result │ │ │ │ │ - * in considerable overhead if your application loads many data layers │ │ │ │ │ - * that are not visible by default. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - preload: false, │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.linearring │ │ │ │ │ + * Given an OpenLayers linearring geometry, create a KML linearring. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.LinearRing>} A linearring geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML linearring node. │ │ │ │ │ + */ │ │ │ │ │ + linearring: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "LinearRing"); │ │ │ │ │ + kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + return kml; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy.Fixed │ │ │ │ │ - * Create a new Fixed strategy. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.polygon │ │ │ │ │ + * Given an OpenLayers polygon geometry, create a KML polygon. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Polygon>} A polygon geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML polygon node. │ │ │ │ │ + */ │ │ │ │ │ + polygon: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "Polygon"); │ │ │ │ │ + var rings = geometry.components; │ │ │ │ │ + var ringMember, ringGeom, type; │ │ │ │ │ + for (var i = 0, len = rings.length; i < len; ++i) { │ │ │ │ │ + type = (i == 0) ? "outerBoundaryIs" : "innerBoundaryIs"; │ │ │ │ │ + ringMember = this.createElementNS(this.kmlns, type); │ │ │ │ │ + ringGeom = this.buildGeometry.linearring.apply(this, │ │ │ │ │ + [rings[i]]); │ │ │ │ │ + ringMember.appendChild(ringGeom); │ │ │ │ │ + kml.appendChild(ringMember); │ │ │ │ │ + } │ │ │ │ │ + return kml; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Activate the strategy: load data or add listener to load when visible │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} True if the strategy was successfully activated or false if │ │ │ │ │ - * the strategy was already active. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "refresh": this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - if (this.layer.visibility == true || this.preload) { │ │ │ │ │ - this.load(); │ │ │ │ │ - } else { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "visibilitychanged": this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.multipolygon │ │ │ │ │ + * Given an OpenLayers multipolygon geometry, create a KML │ │ │ │ │ + * GeometryCollection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Point>} A multipolygon geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ + */ │ │ │ │ │ + multipolygon: function(geometry) { │ │ │ │ │ + return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.collection │ │ │ │ │ + * Given an OpenLayers geometry collection, create a KML MultiGeometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Collection>} A geometry collection. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML MultiGeometry node. │ │ │ │ │ + */ │ │ │ │ │ + collection: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "MultiGeometry"); │ │ │ │ │ + var child; │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; ++i) { │ │ │ │ │ + child = this.buildGeometryNode.apply(this, │ │ │ │ │ + [geometry.components[i]]); │ │ │ │ │ + if (child) { │ │ │ │ │ + kml.appendChild(child); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return kml; │ │ │ │ │ } │ │ │ │ │ - return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Deactivate the strategy. Undo what is done in <activate>. │ │ │ │ │ + * Method: buildCoordinatesNode │ │ │ │ │ + * Builds and returns the KML coordinates node with the given geometry │ │ │ │ │ + * <coordinates>...</coordinates> │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - "refresh": this.load, │ │ │ │ │ - "visibilitychanged": this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + buildCoordinatesNode: function(geometry) { │ │ │ │ │ + var coordinatesNode = this.createElementNS(this.kmlns, "coordinates"); │ │ │ │ │ + │ │ │ │ │ + var path; │ │ │ │ │ + var points = geometry.components; │ │ │ │ │ + if (points) { │ │ │ │ │ + // LineString or LinearRing │ │ │ │ │ + var point; │ │ │ │ │ + var numPoints = points.length; │ │ │ │ │ + var parts = new Array(numPoints); │ │ │ │ │ + for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ + point = points[i]; │ │ │ │ │ + parts[i] = this.buildCoordinates(point); │ │ │ │ │ + } │ │ │ │ │ + path = parts.join(" "); │ │ │ │ │ + } else { │ │ │ │ │ + // Point │ │ │ │ │ + path = this.buildCoordinates(geometry); │ │ │ │ │ } │ │ │ │ │ - return deactivated; │ │ │ │ │ + │ │ │ │ │ + var txtNode = this.createTextNode(path); │ │ │ │ │ + coordinatesNode.appendChild(txtNode); │ │ │ │ │ + │ │ │ │ │ + return coordinatesNode; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: load │ │ │ │ │ - * Tells protocol to load data and unhooks the visibilitychanged event │ │ │ │ │ + * Method: buildCoordinates │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} options to pass to protocol read. │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * │ │ │ │ │ + * Returns │ │ │ │ │ + * {String} a coordinate pair │ │ │ │ │ */ │ │ │ │ │ - load: function(options) { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - layer.events.triggerEvent("loadstart", { │ │ │ │ │ - filter: layer.filter │ │ │ │ │ - }); │ │ │ │ │ - layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: this.merge, │ │ │ │ │ - filter: layer.filter, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options)); │ │ │ │ │ - layer.events.un({ │ │ │ │ │ - "visibilitychanged": this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + buildCoordinates: function(point) { │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + point = point.clone(); │ │ │ │ │ + point.transform(this.internalProjection, │ │ │ │ │ + this.externalProjection); │ │ │ │ │ + } │ │ │ │ │ + return point.x + "," + point.y; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: merge │ │ │ │ │ - * Add all features to the layer. │ │ │ │ │ - * If the layer projection differs from the map projection, features │ │ │ │ │ - * will be transformed from the layer projection to the map projection. │ │ │ │ │ + * Method: buildExtendedData │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object passed │ │ │ │ │ - * by the protocol. │ │ │ │ │ + * attributes - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns │ │ │ │ │ + * {DOMElement} A KML ExtendedData node or {null} if no attributes. │ │ │ │ │ */ │ │ │ │ │ - merge: function(resp) { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - layer.destroyFeatures(); │ │ │ │ │ - var features = resp.features; │ │ │ │ │ - if (features && features.length > 0) { │ │ │ │ │ - var remote = layer.projection; │ │ │ │ │ - var local = layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var geom; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - geom = features[i].geometry; │ │ │ │ │ - if (geom) { │ │ │ │ │ - geom.transform(remote, local); │ │ │ │ │ + buildExtendedData: function(attributes) { │ │ │ │ │ + var extendedData = this.createElementNS(this.kmlns, "ExtendedData"); │ │ │ │ │ + for (var attributeName in attributes) { │ │ │ │ │ + // empty, name, description, styleUrl attributes ignored │ │ │ │ │ + if (attributes[attributeName] && attributeName != "name" && attributeName != "description" && attributeName != "styleUrl") { │ │ │ │ │ + var data = this.createElementNS(this.kmlns, "Data"); │ │ │ │ │ + data.setAttribute("name", attributeName); │ │ │ │ │ + var value = this.createElementNS(this.kmlns, "value"); │ │ │ │ │ + if (typeof attributes[attributeName] == "object") { │ │ │ │ │ + // cater for object attributes with 'value' properties │ │ │ │ │ + // other object properties will output an empty node │ │ │ │ │ + if (attributes[attributeName].value) { │ │ │ │ │ + value.appendChild(this.createTextNode(attributes[attributeName].value)); │ │ │ │ │ + } │ │ │ │ │ + if (attributes[attributeName].displayName) { │ │ │ │ │ + var displayName = this.createElementNS(this.kmlns, "displayName"); │ │ │ │ │ + // displayName always written as CDATA │ │ │ │ │ + displayName.appendChild(this.getXMLDoc().createCDATASection(attributes[attributeName].displayName)); │ │ │ │ │ + data.appendChild(displayName); │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + value.appendChild(this.createTextNode(attributes[attributeName])); │ │ │ │ │ } │ │ │ │ │ + data.appendChild(value); │ │ │ │ │ + extendedData.appendChild(data); │ │ │ │ │ } │ │ │ │ │ - layer.addFeatures(features); │ │ │ │ │ } │ │ │ │ │ - layer.events.triggerEvent("loadend", { │ │ │ │ │ - response: resp │ │ │ │ │ - }); │ │ │ │ │ + if (this.isSimpleContent(extendedData)) { │ │ │ │ │ + return null; │ │ │ │ │ + } else { │ │ │ │ │ + return extendedData; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Fixed" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.KML" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Control.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ @@ -28347,3852 +34786,230 @@ │ │ │ │ │ * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ mousedown: function(evt) { │ │ │ │ │ return this.dragstart(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchstart │ │ │ │ │ - * Handle touchstart events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - return this.dragstart(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mousemove │ │ │ │ │ - * Handle mousemove events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - return this.dragmove(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchmove │ │ │ │ │ - * Handle touchmove events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - return this.dragmove(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeTimeout │ │ │ │ │ - * Private. Called by mousemove() to remove the drag timeout. │ │ │ │ │ - */ │ │ │ │ │ - removeTimeout: function() { │ │ │ │ │ - this.timeoutId = null; │ │ │ │ │ - // if timeout expires while we're still dragging (mouseup │ │ │ │ │ - // hasn't occurred) then call mousemove to move to the │ │ │ │ │ - // correct position │ │ │ │ │ - if (this.dragging) { │ │ │ │ │ - this.mousemove(this.lastMoveEvt); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mouseup │ │ │ │ │ - * Handle mouseup events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - return this.dragend(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchend │ │ │ │ │ - * Handle touchend events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - // override evt.xy with last position since touchend does not have │ │ │ │ │ - // any touch position │ │ │ │ │ - evt.xy = this.last; │ │ │ │ │ - return this.dragend(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mouseout │ │ │ │ │ - * Handle mouseout events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - mouseout: function(evt) { │ │ │ │ │ - if (this.started && OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ - if (this.documentDrag === true) { │ │ │ │ │ - this.addDocumentEvents(); │ │ │ │ │ - } else { │ │ │ │ │ - var dragged = (this.start != this.last); │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - OpenLayers.Element.removeClass( │ │ │ │ │ - this.map.viewPortDiv, "olDragDown" │ │ │ │ │ - ); │ │ │ │ │ - this.out(evt); │ │ │ │ │ - this.callback("out", []); │ │ │ │ │ - if (dragged) { │ │ │ │ │ - this.callback("done", [evt.xy]); │ │ │ │ │ - } │ │ │ │ │ - if (document.onselectstart) { │ │ │ │ │ - document.onselectstart = this.oldOnselectstart; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: click │ │ │ │ │ - * The drag handler captures the click event. If something else registers │ │ │ │ │ - * for clicks on the same element, its listener will not be called │ │ │ │ │ - * after a drag. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - // let the click event propagate only if the mouse moved │ │ │ │ │ - return (this.start == this.last); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Activate the handler. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - activated = true; │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Deactivate the handler. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - deactivated = true; │ │ │ │ │ - OpenLayers.Element.removeClass( │ │ │ │ │ - this.map.viewPortDiv, "olDragDown" │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: adjustXY │ │ │ │ │ - * Converts event coordinates that are relative to the document body to │ │ │ │ │ - * ones that are relative to the map viewport. The latter is the default in │ │ │ │ │ - * OpenLayers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ - */ │ │ │ │ │ - adjustXY: function(evt) { │ │ │ │ │ - var pos = OpenLayers.Util.pagePosition(this.map.viewPortDiv); │ │ │ │ │ - evt.xy.x -= pos[0]; │ │ │ │ │ - evt.xy.y -= pos[1]; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addDocumentEvents │ │ │ │ │ - * Start observing document events when documentDrag is true and the mouse │ │ │ │ │ - * cursor leaves the map viewport while dragging. │ │ │ │ │ - */ │ │ │ │ │ - addDocumentEvents: function() { │ │ │ │ │ - OpenLayers.Element.addClass(document.body, "olDragDown"); │ │ │ │ │ - this.documentEvents = true; │ │ │ │ │ - OpenLayers.Event.observe(document, "mousemove", this._docMove); │ │ │ │ │ - OpenLayers.Event.observe(document, "mouseup", this._docUp); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeDocumentEvents │ │ │ │ │ - * Stops observing document events when documentDrag is true and the mouse │ │ │ │ │ - * cursor re-enters the map viewport while dragging. │ │ │ │ │ - */ │ │ │ │ │ - removeDocumentEvents: function() { │ │ │ │ │ - OpenLayers.Element.removeClass(document.body, "olDragDown"); │ │ │ │ │ - this.documentEvents = false; │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mousemove", this._docMove); │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mouseup", this._docUp); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Drag" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Keyboard.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.handler.Keyboard │ │ │ │ │ - * A handler for keyboard events. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Handler.Keyboard> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.Keyboard = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - │ │ │ │ │ - /* http://www.quirksmode.org/js/keys.html explains key x-browser │ │ │ │ │ - key handling quirks in pretty nice detail */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: KEY_EVENTS │ │ │ │ │ - * keydown, keypress, keyup │ │ │ │ │ - */ │ │ │ │ │ - KEY_EVENTS: ["keydown", "keyup"], │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: eventListener │ │ │ │ │ - * {Function} │ │ │ │ │ - */ │ │ │ │ │ - eventListener: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: observeElement │ │ │ │ │ - * {DOMElement|String} The DOM element on which we listen for │ │ │ │ │ - * key events. Default to the document. │ │ │ │ │ - */ │ │ │ │ │ - observeElement: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Keyboard │ │ │ │ │ - * Returns a new keyboard handler. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ - * this handler. If a handler is being used without a control, the │ │ │ │ │ - * handlers setMap method must be overridden to deal properly with │ │ │ │ │ - * the map. │ │ │ │ │ - * callbacks - {Object} An object containing a single function to be │ │ │ │ │ - * called when the drag operation is finished. The callback should │ │ │ │ │ - * expect to recieve a single argument, the pixel location of the event. │ │ │ │ │ - * Callbacks for 'keydown', 'keypress', and 'keyup' are supported. │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * handler. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - // cache the bound event listener method so it can be unobserved later │ │ │ │ │ - this.eventListener = OpenLayers.Function.bindAsEventListener( │ │ │ │ │ - this.handleKeyEvent, this │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.eventListener = null; │ │ │ │ │ - OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.observeElement = this.observeElement || document; │ │ │ │ │ - for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ - OpenLayers.Event.observe( │ │ │ │ │ - this.observeElement, this.KEY_EVENTS[i], this.eventListener); │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ - OpenLayers.Event.stopObserving( │ │ │ │ │ - this.observeElement, this.KEY_EVENTS[i], this.eventListener); │ │ │ │ │ - } │ │ │ │ │ - deactivated = true; │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleKeyEvent │ │ │ │ │ - */ │ │ │ │ │ - handleKeyEvent: function(evt) { │ │ │ │ │ - if (this.checkModifiers(evt)) { │ │ │ │ │ - this.callback(evt.type, [evt]); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Keyboard" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/ModifyFeature.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Handler/Drag.js │ │ │ │ │ - * @requires OpenLayers/Handler/Keyboard.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.ModifyFeature │ │ │ │ │ - * Control to modify features. When activated, a click renders the vertices │ │ │ │ │ - * of a feature - these vertices can then be dragged. By default, the │ │ │ │ │ - * delete key will delete the vertex under the mouse. New features are │ │ │ │ │ - * added by dragging "virtual vertices" between vertices. Create a new │ │ │ │ │ - * control with the <OpenLayers.Control.ModifyFeature> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits From: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ModifyFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: documentDrag │ │ │ │ │ - * {Boolean} If set to true, dragging vertices will continue even if the │ │ │ │ │ - * mouse cursor leaves the map viewport. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - documentDrag: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geometryTypes │ │ │ │ │ - * {Array(String)} To restrict modification to a limited set of geometry │ │ │ │ │ - * types, send a list of strings corresponding to the geometry class │ │ │ │ │ - * names. │ │ │ │ │ - */ │ │ │ │ │ - geometryTypes: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: clickout │ │ │ │ │ - * {Boolean} Unselect features when clicking outside any feature. │ │ │ │ │ - * Default is true. │ │ │ │ │ - */ │ │ │ │ │ - clickout: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: toggle │ │ │ │ │ - * {Boolean} Unselect a selected feature on click. │ │ │ │ │ - * Default is true. │ │ │ │ │ - */ │ │ │ │ │ - toggle: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: standalone │ │ │ │ │ - * {Boolean} Set to true to create a control without SelectFeature │ │ │ │ │ - * capabilities. Default is false. If standalone is true, to modify │ │ │ │ │ - * a feature, call the <selectFeature> method with the target feature. │ │ │ │ │ - * Note that you must call the <unselectFeature> method to finish │ │ │ │ │ - * feature modification in standalone mode (before starting to modify │ │ │ │ │ - * another feature). │ │ │ │ │ - */ │ │ │ │ │ - standalone: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - layer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: feature │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} Feature currently available for modification. │ │ │ │ │ - */ │ │ │ │ │ - feature: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: vertex │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} Vertex currently being modified. │ │ │ │ │ - */ │ │ │ │ │ - vertex: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: vertices │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} Verticies currently available │ │ │ │ │ - * for dragging. │ │ │ │ │ - */ │ │ │ │ │ - vertices: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: virtualVertices │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} Virtual vertices in the middle │ │ │ │ │ - * of each edge. │ │ │ │ │ - */ │ │ │ │ │ - virtualVertices: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: handlers │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ - handlers: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: deleteCodes │ │ │ │ │ - * {Array(Integer)} Keycodes for deleting verticies. Set to null to disable │ │ │ │ │ - * vertex deltion by keypress. If non-null, keypresses with codes │ │ │ │ │ - * in this array will delete vertices under the mouse. Default │ │ │ │ │ - * is 46 and 68, the 'delete' and lowercase 'd' keys. │ │ │ │ │ - */ │ │ │ │ │ - deleteCodes: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: virtualStyle │ │ │ │ │ - * {Object} A symbolizer to be used for virtual vertices. │ │ │ │ │ - */ │ │ │ │ │ - virtualStyle: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: vertexRenderIntent │ │ │ │ │ - * {String} The renderIntent to use for vertices. If no <virtualStyle> is │ │ │ │ │ - * provided, this renderIntent will also be used for virtual vertices, with │ │ │ │ │ - * a fillOpacity and strokeOpacity of 0.3. Default is null, which means │ │ │ │ │ - * that the layer's default style will be used for vertices. │ │ │ │ │ - */ │ │ │ │ │ - vertexRenderIntent: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: mode │ │ │ │ │ - * {Integer} Bitfields specifying the modification mode. Defaults to │ │ │ │ │ - * OpenLayers.Control.ModifyFeature.RESHAPE. To set the mode to a │ │ │ │ │ - * combination of options, use the | operator. For example, to allow │ │ │ │ │ - * the control to both resize and rotate features, use the following │ │ │ │ │ - * syntax │ │ │ │ │ - * (code) │ │ │ │ │ - * control.mode = OpenLayers.Control.ModifyFeature.RESIZE | │ │ │ │ │ - * OpenLayers.Control.ModifyFeature.ROTATE; │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ - mode: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: createVertices │ │ │ │ │ - * {Boolean} Create new vertices by dragging the virtual vertices │ │ │ │ │ - * in the middle of each edge. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - createVertices: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: modified │ │ │ │ │ - * {Boolean} The currently selected feature has been modified. │ │ │ │ │ - */ │ │ │ │ │ - modified: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: radiusHandle │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A handle for rotating/resizing a feature. │ │ │ │ │ - */ │ │ │ │ │ - radiusHandle: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: dragHandle │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A handle for dragging a feature. │ │ │ │ │ - */ │ │ │ │ │ - dragHandle: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: onModificationStart │ │ │ │ │ - * {Function} *Deprecated*. Register for "beforefeaturemodified" instead. │ │ │ │ │ - * The "beforefeaturemodified" event is triggered on the layer before │ │ │ │ │ - * any modification begins. │ │ │ │ │ - * │ │ │ │ │ - * Optional function to be called when a feature is selected │ │ │ │ │ - * to be modified. The function should expect to be called with a │ │ │ │ │ - * feature. This could be used for example to allow to lock the │ │ │ │ │ - * feature on server-side. │ │ │ │ │ - */ │ │ │ │ │ - onModificationStart: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: onModification │ │ │ │ │ - * {Function} *Deprecated*. Register for "featuremodified" instead. │ │ │ │ │ - * The "featuremodified" event is triggered on the layer with each │ │ │ │ │ - * feature modification. │ │ │ │ │ - * │ │ │ │ │ - * Optional function to be called when a feature has been │ │ │ │ │ - * modified. The function should expect to be called with a feature. │ │ │ │ │ - */ │ │ │ │ │ - onModification: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: onModificationEnd │ │ │ │ │ - * {Function} *Deprecated*. Register for "afterfeaturemodified" instead. │ │ │ │ │ - * The "afterfeaturemodified" event is triggered on the layer after │ │ │ │ │ - * a feature has been modified. │ │ │ │ │ - * │ │ │ │ │ - * Optional function to be called when a feature is finished │ │ │ │ │ - * being modified. The function should expect to be called with a │ │ │ │ │ - * feature. │ │ │ │ │ - */ │ │ │ │ │ - onModificationEnd: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.ModifyFeature │ │ │ │ │ - * Create a new modify feature control. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.Vector>} Layer that contains features that │ │ │ │ │ - * will be modified. │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * control. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(layer, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - this.virtualVertices = []; │ │ │ │ │ - this.virtualStyle = OpenLayers.Util.extend({}, │ │ │ │ │ - this.layer.style || │ │ │ │ │ - this.layer.styleMap.createSymbolizer(null, options.vertexRenderIntent) │ │ │ │ │ - ); │ │ │ │ │ - this.virtualStyle.fillOpacity = 0.3; │ │ │ │ │ - this.virtualStyle.strokeOpacity = 0.3; │ │ │ │ │ - this.deleteCodes = [46, 68]; │ │ │ │ │ - this.mode = OpenLayers.Control.ModifyFeature.RESHAPE; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!(OpenLayers.Util.isArray(this.deleteCodes))) { │ │ │ │ │ - this.deleteCodes = [this.deleteCodes]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // configure the drag handler │ │ │ │ │ - var dragCallbacks = { │ │ │ │ │ - down: function(pixel) { │ │ │ │ │ - this.vertex = null; │ │ │ │ │ - var feature = this.layer.getFeatureFromEvent( │ │ │ │ │ - this.handlers.drag.evt); │ │ │ │ │ - if (feature) { │ │ │ │ │ - this.dragStart(feature); │ │ │ │ │ - } else if (this.clickout) { │ │ │ │ │ - this._unselect = this.feature; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - move: function(pixel) { │ │ │ │ │ - delete this._unselect; │ │ │ │ │ - if (this.vertex) { │ │ │ │ │ - this.dragVertex(this.vertex, pixel); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - up: function() { │ │ │ │ │ - this.handlers.drag.stopDown = false; │ │ │ │ │ - if (this._unselect) { │ │ │ │ │ - this.unselectFeature(this._unselect); │ │ │ │ │ - delete this._unselect; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - done: function(pixel) { │ │ │ │ │ - if (this.vertex) { │ │ │ │ │ - this.dragComplete(this.vertex); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - var dragOptions = { │ │ │ │ │ - documentDrag: this.documentDrag, │ │ │ │ │ - stopDown: false │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - // configure the keyboard handler │ │ │ │ │ - var keyboardOptions = { │ │ │ │ │ - keydown: this.handleKeypress │ │ │ │ │ - }; │ │ │ │ │ - this.handlers = { │ │ │ │ │ - keyboard: new OpenLayers.Handler.Keyboard(this, keyboardOptions), │ │ │ │ │ - drag: new OpenLayers.Handler.Drag(this, dragCallbacks, dragOptions) │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Take care of things that are not handled in superclass. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - "removelayer": this.handleMapEvents, │ │ │ │ │ - "changelayer": this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - this.layer = null; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, []); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Activate the control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Successfully activated the control. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - this.moveLayerToTop(); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - "removelayer": this.handleMapEvents, │ │ │ │ │ - "changelayer": this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - return (this.handlers.keyboard.activate() && │ │ │ │ │ - this.handlers.drag.activate() && │ │ │ │ │ - OpenLayers.Control.prototype.activate.apply(this, arguments)); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Successfully deactivated the control. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - // the return from the controls is unimportant in this case │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerBack(); │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - "removelayer": this.handleMapEvents, │ │ │ │ │ - "changelayer": this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.layer.removeFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.layer.removeFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - this.handlers.drag.deactivate(); │ │ │ │ │ - this.handlers.keyboard.deactivate(); │ │ │ │ │ - var feature = this.feature; │ │ │ │ │ - if (feature && feature.geometry && feature.layer) { │ │ │ │ │ - this.unselectFeature(feature); │ │ │ │ │ - } │ │ │ │ │ - deactivated = true; │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: beforeSelectFeature │ │ │ │ │ - * Called before a feature is selected. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} The feature about to be selected. │ │ │ │ │ - */ │ │ │ │ │ - beforeSelectFeature: function(feature) { │ │ │ │ │ - return this.layer.events.triggerEvent( │ │ │ │ │ - "beforefeaturemodified", { │ │ │ │ │ - feature: feature │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: selectFeature │ │ │ │ │ - * Select a feature for modification in standalone mode. In non-standalone │ │ │ │ │ - * mode, this method is called when a feature is selected by clicking. │ │ │ │ │ - * Register a listener to the beforefeaturemodified event and return false │ │ │ │ │ - * to prevent feature modification. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} the selected feature. │ │ │ │ │ - */ │ │ │ │ │ - selectFeature: function(feature) { │ │ │ │ │ - if (this.feature === feature || │ │ │ │ │ - (this.geometryTypes && OpenLayers.Util.indexOf(this.geometryTypes, │ │ │ │ │ - feature.geometry.CLASS_NAME) == -1)) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (this.beforeSelectFeature(feature) !== false) { │ │ │ │ │ - if (this.feature) { │ │ │ │ │ - this.unselectFeature(this.feature); │ │ │ │ │ - } │ │ │ │ │ - this.feature = feature; │ │ │ │ │ - this.layer.selectedFeatures.push(feature); │ │ │ │ │ - this.layer.drawFeature(feature, 'select'); │ │ │ │ │ - this.modified = false; │ │ │ │ │ - this.resetVertices(); │ │ │ │ │ - this.onModificationStart(this.feature); │ │ │ │ │ - } │ │ │ │ │ - // keep track of geometry modifications │ │ │ │ │ - var modified = feature.modified; │ │ │ │ │ - if (feature.geometry && !(modified && modified.geometry)) { │ │ │ │ │ - this._originalGeometry = feature.geometry.clone(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: unselectFeature │ │ │ │ │ - * Called when the select feature control unselects a feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} The unselected feature. │ │ │ │ │ - */ │ │ │ │ │ - unselectFeature: function(feature) { │ │ │ │ │ - this.layer.removeFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - this.layer.destroyFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.virtualVertices = []; │ │ │ │ │ - if (this.dragHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.dragHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - delete this.dragHandle; │ │ │ │ │ - } │ │ │ │ │ - if (this.radiusHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - delete this.radiusHandle; │ │ │ │ │ - } │ │ │ │ │ - this.layer.drawFeature(this.feature, 'default'); │ │ │ │ │ - this.feature = null; │ │ │ │ │ - OpenLayers.Util.removeItem(this.layer.selectedFeatures, feature); │ │ │ │ │ - this.onModificationEnd(feature); │ │ │ │ │ - this.layer.events.triggerEvent("afterfeaturemodified", { │ │ │ │ │ - feature: feature, │ │ │ │ │ - modified: this.modified │ │ │ │ │ - }); │ │ │ │ │ - this.modified = false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: dragStart │ │ │ │ │ - * Called by the drag handler before a feature is dragged. This method is │ │ │ │ │ - * used to differentiate between points and vertices │ │ │ │ │ - * of higher order geometries. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} The point or vertex about to be │ │ │ │ │ - * dragged. │ │ │ │ │ - */ │ │ │ │ │ - dragStart: function(feature) { │ │ │ │ │ - var isPoint = feature.geometry.CLASS_NAME == │ │ │ │ │ - 'OpenLayers.Geometry.Point'; │ │ │ │ │ - if (!this.standalone && │ │ │ │ │ - ((!feature._sketch && isPoint) || !feature._sketch)) { │ │ │ │ │ - if (this.toggle && this.feature === feature) { │ │ │ │ │ - // mark feature for unselection │ │ │ │ │ - this._unselect = feature; │ │ │ │ │ - } │ │ │ │ │ - this.selectFeature(feature); │ │ │ │ │ - } │ │ │ │ │ - if (feature._sketch || isPoint) { │ │ │ │ │ - // feature is a drag or virtual handle or point │ │ │ │ │ - this.vertex = feature; │ │ │ │ │ - this.handlers.drag.stopDown = true; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: dragVertex │ │ │ │ │ - * Called by the drag handler with each drag move of a vertex. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * vertex - {<OpenLayers.Feature.Vector>} The vertex being dragged. │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} Pixel location of the mouse event. │ │ │ │ │ - */ │ │ │ │ │ - dragVertex: function(vertex, pixel) { │ │ │ │ │ - var pos = this.map.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geom = vertex.geometry; │ │ │ │ │ - geom.move(pos.lon - geom.x, pos.lat - geom.y); │ │ │ │ │ - this.modified = true; │ │ │ │ │ - /** │ │ │ │ │ - * Five cases: │ │ │ │ │ - * 1) dragging a simple point │ │ │ │ │ - * 2) dragging a virtual vertex │ │ │ │ │ - * 3) dragging a drag handle │ │ │ │ │ - * 4) dragging a real vertex │ │ │ │ │ - * 5) dragging a radius handle │ │ │ │ │ - */ │ │ │ │ │ - if (this.feature.geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - // dragging a simple point │ │ │ │ │ - this.layer.events.triggerEvent("vertexmodified", { │ │ │ │ │ - vertex: vertex.geometry, │ │ │ │ │ - feature: this.feature, │ │ │ │ │ - pixel: pixel │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - if (vertex._index) { │ │ │ │ │ - // dragging a virtual vertex │ │ │ │ │ - vertex.geometry.parent.addComponent(vertex.geometry, │ │ │ │ │ - vertex._index); │ │ │ │ │ - // move from virtual to real vertex │ │ │ │ │ - delete vertex._index; │ │ │ │ │ - OpenLayers.Util.removeItem(this.virtualVertices, vertex); │ │ │ │ │ - this.vertices.push(vertex); │ │ │ │ │ - } else if (vertex == this.dragHandle) { │ │ │ │ │ - // dragging a drag handle │ │ │ │ │ - this.layer.removeFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - if (this.radiusHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.radiusHandle = null; │ │ │ │ │ - } │ │ │ │ │ - } else if (vertex !== this.radiusHandle) { │ │ │ │ │ - // dragging a real vertex │ │ │ │ │ - this.layer.events.triggerEvent("vertexmodified", { │ │ │ │ │ - vertex: vertex.geometry, │ │ │ │ │ - feature: this.feature, │ │ │ │ │ - pixel: pixel │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - // dragging a radius handle - no special treatment │ │ │ │ │ - if (this.virtualVertices.length > 0) { │ │ │ │ │ - this.layer.destroyFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.virtualVertices = []; │ │ │ │ │ - } │ │ │ │ │ - this.layer.drawFeature(this.feature, this.standalone ? undefined : │ │ │ │ │ - 'select'); │ │ │ │ │ - } │ │ │ │ │ - // keep the vertex on top so it gets the mouseout after dragging │ │ │ │ │ - // this should be removed in favor of an option to draw under or │ │ │ │ │ - // maintain node z-index │ │ │ │ │ - this.layer.drawFeature(vertex); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: dragComplete │ │ │ │ │ - * Called by the drag handler when the feature dragging is complete. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * vertex - {<OpenLayers.Feature.Vector>} The vertex being dragged. │ │ │ │ │ - */ │ │ │ │ │ - dragComplete: function(vertex) { │ │ │ │ │ - this.resetVertices(); │ │ │ │ │ - this.setFeatureState(); │ │ │ │ │ - this.onModification(this.feature); │ │ │ │ │ - this.layer.events.triggerEvent("featuremodified", { │ │ │ │ │ - feature: this.feature │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setFeatureState │ │ │ │ │ - * Called when the feature is modified. If the current state is not │ │ │ │ │ - * INSERT or DELETE, the state is set to UPDATE. │ │ │ │ │ - */ │ │ │ │ │ - setFeatureState: function() { │ │ │ │ │ - if (this.feature.state != OpenLayers.State.INSERT && │ │ │ │ │ - this.feature.state != OpenLayers.State.DELETE) { │ │ │ │ │ - this.feature.state = OpenLayers.State.UPDATE; │ │ │ │ │ - if (this.modified && this._originalGeometry) { │ │ │ │ │ - var feature = this.feature; │ │ │ │ │ - feature.modified = OpenLayers.Util.extend(feature.modified, { │ │ │ │ │ - geometry: this._originalGeometry │ │ │ │ │ - }); │ │ │ │ │ - delete this._originalGeometry; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: resetVertices │ │ │ │ │ - */ │ │ │ │ │ - resetVertices: function() { │ │ │ │ │ - if (this.vertices.length > 0) { │ │ │ │ │ - this.layer.removeFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - } │ │ │ │ │ - if (this.virtualVertices.length > 0) { │ │ │ │ │ - this.layer.removeFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.virtualVertices = []; │ │ │ │ │ - } │ │ │ │ │ - if (this.dragHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.dragHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.dragHandle = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.radiusHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.radiusHandle = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.feature && │ │ │ │ │ - this.feature.geometry.CLASS_NAME != "OpenLayers.Geometry.Point") { │ │ │ │ │ - if ((this.mode & OpenLayers.Control.ModifyFeature.DRAG)) { │ │ │ │ │ - this.collectDragHandle(); │ │ │ │ │ - } │ │ │ │ │ - if ((this.mode & (OpenLayers.Control.ModifyFeature.ROTATE | │ │ │ │ │ - OpenLayers.Control.ModifyFeature.RESIZE))) { │ │ │ │ │ - this.collectRadiusHandle(); │ │ │ │ │ - } │ │ │ │ │ - if (this.mode & OpenLayers.Control.ModifyFeature.RESHAPE) { │ │ │ │ │ - // Don't collect vertices when we're resizing │ │ │ │ │ - if (!(this.mode & OpenLayers.Control.ModifyFeature.RESIZE)) { │ │ │ │ │ - this.collectVertices(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleKeypress │ │ │ │ │ - * Called by the feature handler on keypress. This is used to delete │ │ │ │ │ - * vertices. If the <deleteCode> property is set, vertices will │ │ │ │ │ - * be deleted when a feature is selected for modification and │ │ │ │ │ - * the mouse is over a vertex. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} Keypress event. │ │ │ │ │ - */ │ │ │ │ │ - handleKeypress: function(evt) { │ │ │ │ │ - var code = evt.keyCode; │ │ │ │ │ - │ │ │ │ │ - // check for delete key │ │ │ │ │ - if (this.feature && │ │ │ │ │ - OpenLayers.Util.indexOf(this.deleteCodes, code) != -1) { │ │ │ │ │ - var vertex = this.layer.getFeatureFromEvent(this.handlers.drag.evt); │ │ │ │ │ - if (vertex && │ │ │ │ │ - OpenLayers.Util.indexOf(this.vertices, vertex) != -1 && │ │ │ │ │ - !this.handlers.drag.dragging && vertex.geometry.parent) { │ │ │ │ │ - // remove the vertex │ │ │ │ │ - vertex.geometry.parent.removeComponent(vertex.geometry); │ │ │ │ │ - this.layer.events.triggerEvent("vertexremoved", { │ │ │ │ │ - vertex: vertex.geometry, │ │ │ │ │ - feature: this.feature, │ │ │ │ │ - pixel: evt.xy │ │ │ │ │ - }); │ │ │ │ │ - this.layer.drawFeature(this.feature, this.standalone ? │ │ │ │ │ - undefined : 'select'); │ │ │ │ │ - this.modified = true; │ │ │ │ │ - this.resetVertices(); │ │ │ │ │ - this.setFeatureState(); │ │ │ │ │ - this.onModification(this.feature); │ │ │ │ │ - this.layer.events.triggerEvent("featuremodified", { │ │ │ │ │ - feature: this.feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: collectVertices │ │ │ │ │ - * Collect the vertices from the modifiable feature's geometry and push │ │ │ │ │ - * them on to the control's vertices array. │ │ │ │ │ - */ │ │ │ │ │ - collectVertices: function() { │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - this.virtualVertices = []; │ │ │ │ │ - var control = this; │ │ │ │ │ - │ │ │ │ │ - function collectComponentVertices(geometry) { │ │ │ │ │ - var i, vertex, component, len; │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - vertex = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - vertex._sketch = true; │ │ │ │ │ - vertex.renderIntent = control.vertexRenderIntent; │ │ │ │ │ - control.vertices.push(vertex); │ │ │ │ │ - } else { │ │ │ │ │ - var numVert = geometry.components.length; │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ - numVert -= 1; │ │ │ │ │ - } │ │ │ │ │ - for (i = 0; i < numVert; ++i) { │ │ │ │ │ - component = geometry.components[i]; │ │ │ │ │ - if (component.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - vertex = new OpenLayers.Feature.Vector(component); │ │ │ │ │ - vertex._sketch = true; │ │ │ │ │ - vertex.renderIntent = control.vertexRenderIntent; │ │ │ │ │ - control.vertices.push(vertex); │ │ │ │ │ - } else { │ │ │ │ │ - collectComponentVertices(component); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // add virtual vertices in the middle of each edge │ │ │ │ │ - if (control.createVertices && geometry.CLASS_NAME != "OpenLayers.Geometry.MultiPoint") { │ │ │ │ │ - for (i = 0, len = geometry.components.length; i < len - 1; ++i) { │ │ │ │ │ - var prevVertex = geometry.components[i]; │ │ │ │ │ - var nextVertex = geometry.components[i + 1]; │ │ │ │ │ - if (prevVertex.CLASS_NAME == "OpenLayers.Geometry.Point" && │ │ │ │ │ - nextVertex.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - var x = (prevVertex.x + nextVertex.x) / 2; │ │ │ │ │ - var y = (prevVertex.y + nextVertex.y) / 2; │ │ │ │ │ - var point = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.Point(x, y), │ │ │ │ │ - null, control.virtualStyle │ │ │ │ │ - ); │ │ │ │ │ - // set the virtual parent and intended index │ │ │ │ │ - point.geometry.parent = geometry; │ │ │ │ │ - point._index = i + 1; │ │ │ │ │ - point._sketch = true; │ │ │ │ │ - control.virtualVertices.push(point); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - collectComponentVertices.call(this, this.feature.geometry); │ │ │ │ │ - this.layer.addFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.layer.addFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: collectDragHandle │ │ │ │ │ - * Collect the drag handle for the selected geometry. │ │ │ │ │ - */ │ │ │ │ │ - collectDragHandle: function() { │ │ │ │ │ - var geometry = this.feature.geometry; │ │ │ │ │ - var center = geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - var originGeometry = new OpenLayers.Geometry.Point( │ │ │ │ │ - center.lon, center.lat │ │ │ │ │ - ); │ │ │ │ │ - var origin = new OpenLayers.Feature.Vector(originGeometry); │ │ │ │ │ - originGeometry.move = function(x, y) { │ │ │ │ │ - OpenLayers.Geometry.Point.prototype.move.call(this, x, y); │ │ │ │ │ - geometry.move(x, y); │ │ │ │ │ - }; │ │ │ │ │ - origin._sketch = true; │ │ │ │ │ - this.dragHandle = origin; │ │ │ │ │ - this.dragHandle.renderIntent = this.vertexRenderIntent; │ │ │ │ │ - this.layer.addFeatures([this.dragHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: collectRadiusHandle │ │ │ │ │ - * Collect the radius handle for the selected geometry. │ │ │ │ │ - */ │ │ │ │ │ - collectRadiusHandle: function() { │ │ │ │ │ - var geometry = this.feature.geometry; │ │ │ │ │ - var bounds = geometry.getBounds(); │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var originGeometry = new OpenLayers.Geometry.Point( │ │ │ │ │ - center.lon, center.lat │ │ │ │ │ - ); │ │ │ │ │ - var radiusGeometry = new OpenLayers.Geometry.Point( │ │ │ │ │ - bounds.right, bounds.bottom │ │ │ │ │ - ); │ │ │ │ │ - var radius = new OpenLayers.Feature.Vector(radiusGeometry); │ │ │ │ │ - var resize = (this.mode & OpenLayers.Control.ModifyFeature.RESIZE); │ │ │ │ │ - var reshape = (this.mode & OpenLayers.Control.ModifyFeature.RESHAPE); │ │ │ │ │ - var rotate = (this.mode & OpenLayers.Control.ModifyFeature.ROTATE); │ │ │ │ │ - │ │ │ │ │ - radiusGeometry.move = function(x, y) { │ │ │ │ │ - OpenLayers.Geometry.Point.prototype.move.call(this, x, y); │ │ │ │ │ - var dx1 = this.x - originGeometry.x; │ │ │ │ │ - var dy1 = this.y - originGeometry.y; │ │ │ │ │ - var dx0 = dx1 - x; │ │ │ │ │ - var dy0 = dy1 - y; │ │ │ │ │ - if (rotate) { │ │ │ │ │ - var a0 = Math.atan2(dy0, dx0); │ │ │ │ │ - var a1 = Math.atan2(dy1, dx1); │ │ │ │ │ - var angle = a1 - a0; │ │ │ │ │ - angle *= 180 / Math.PI; │ │ │ │ │ - geometry.rotate(angle, originGeometry); │ │ │ │ │ - } │ │ │ │ │ - if (resize) { │ │ │ │ │ - var scale, ratio; │ │ │ │ │ - // 'resize' together with 'reshape' implies that the aspect │ │ │ │ │ - // ratio of the geometry will not be preserved whilst resizing │ │ │ │ │ - if (reshape) { │ │ │ │ │ - scale = dy1 / dy0; │ │ │ │ │ - ratio = (dx1 / dx0) / scale; │ │ │ │ │ - } else { │ │ │ │ │ - var l0 = Math.sqrt((dx0 * dx0) + (dy0 * dy0)); │ │ │ │ │ - var l1 = Math.sqrt((dx1 * dx1) + (dy1 * dy1)); │ │ │ │ │ - scale = l1 / l0; │ │ │ │ │ - } │ │ │ │ │ - geometry.resize(scale, originGeometry, ratio); │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - radius._sketch = true; │ │ │ │ │ - this.radiusHandle = radius; │ │ │ │ │ - this.radiusHandle.renderIntent = this.vertexRenderIntent; │ │ │ │ │ - this.layer.addFeatures([this.radiusHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the control and all handlers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} The control's map. │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.handlers.drag.setMap(map); │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleMapEvents │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ - */ │ │ │ │ │ - handleMapEvents: function(evt) { │ │ │ │ │ - if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ - this.moveLayerToTop(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveLayerToTop │ │ │ │ │ - * Moves the layer for this handler to the top, so mouse events can reach │ │ │ │ │ - * it. │ │ │ │ │ - */ │ │ │ │ │ - moveLayerToTop: function() { │ │ │ │ │ - var index = Math.max(this.map.Z_INDEX_BASE['Feature'] - 1, │ │ │ │ │ - this.layer.getZIndex()) + 1; │ │ │ │ │ - this.layer.setZIndex(index); │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveLayerBack │ │ │ │ │ - * Moves the layer back to the position determined by the map's layers │ │ │ │ │ - * array. │ │ │ │ │ - */ │ │ │ │ │ - moveLayerBack: function() { │ │ │ │ │ - var index = this.layer.getZIndex() - 1; │ │ │ │ │ - if (index >= this.map.Z_INDEX_BASE['Feature']) { │ │ │ │ │ - this.layer.setZIndex(index); │ │ │ │ │ - } else { │ │ │ │ │ - this.map.setLayerZIndex(this.layer, │ │ │ │ │ - this.map.getLayerIndex(this.layer)); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ModifyFeature" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: RESHAPE │ │ │ │ │ - * {Integer} Constant used to make the control work in reshape mode │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ModifyFeature.RESHAPE = 1; │ │ │ │ │ -/** │ │ │ │ │ - * Constant: RESIZE │ │ │ │ │ - * {Integer} Constant used to make the control work in resize mode │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ModifyFeature.RESIZE = 2; │ │ │ │ │ -/** │ │ │ │ │ - * Constant: ROTATE │ │ │ │ │ - * {Integer} Constant used to make the control work in rotate mode │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ModifyFeature.ROTATE = 4; │ │ │ │ │ -/** │ │ │ │ │ - * Constant: DRAG │ │ │ │ │ - * {Integer} Constant used to make the control work in drag mode │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ModifyFeature.DRAG = 8; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/DrawFeature.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.DrawFeature │ │ │ │ │ - * The DrawFeature control draws point, line or polygon features on a vector │ │ │ │ │ - * layer when active. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.DrawFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - layer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: callbacks │ │ │ │ │ - * {Object} The functions that are sent to the handler for callback │ │ │ │ │ - */ │ │ │ │ │ - callbacks: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * featureadded - Triggered when a feature is added │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: multi │ │ │ │ │ - * {Boolean} Cast features to multi-part geometries before passing to the │ │ │ │ │ - * layer. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - multi: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: featureAdded │ │ │ │ │ - * {Function} Called after each feature is added │ │ │ │ │ - */ │ │ │ │ │ - featureAdded: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: handlerOptions │ │ │ │ │ - * {Object} Used to set non-default properties on the control's handler │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.DrawFeature │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ - * handler - {<OpenLayers.Handler>} │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(layer, handler, options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.callbacks = OpenLayers.Util.extend({ │ │ │ │ │ - done: this.drawFeature, │ │ │ │ │ - modify: function(vertex, feature) { │ │ │ │ │ - this.layer.events.triggerEvent( │ │ │ │ │ - "sketchmodified", { │ │ │ │ │ - vertex: vertex, │ │ │ │ │ - feature: feature │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - create: function(vertex, feature) { │ │ │ │ │ - this.layer.events.triggerEvent( │ │ │ │ │ - "sketchstarted", { │ │ │ │ │ - vertex: vertex, │ │ │ │ │ - feature: feature │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - this.callbacks │ │ │ │ │ - ); │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - this.handlerOptions = this.handlerOptions || {}; │ │ │ │ │ - this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.handlerOptions.layerOptions, { │ │ │ │ │ - renderers: layer.renderers, │ │ │ │ │ - rendererOptions: layer.rendererOptions │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - if (!("multi" in this.handlerOptions)) { │ │ │ │ │ - this.handlerOptions.multi = this.multi; │ │ │ │ │ - } │ │ │ │ │ - var sketchStyle = this.layer.styleMap && this.layer.styleMap.styles.temporary; │ │ │ │ │ - if (sketchStyle) { │ │ │ │ │ - this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.handlerOptions.layerOptions, { │ │ │ │ │ - styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ - "default": sketchStyle │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - this.handler = new handler(this, this.callbacks, this.handlerOptions); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawFeature │ │ │ │ │ - */ │ │ │ │ │ - drawFeature: function(geometry) { │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - var proceed = this.layer.events.triggerEvent( │ │ │ │ │ - "sketchcomplete", { │ │ │ │ │ - feature: feature │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - if (proceed !== false) { │ │ │ │ │ - feature.state = OpenLayers.State.INSERT; │ │ │ │ │ - this.layer.addFeatures([feature]); │ │ │ │ │ - this.featureAdded(feature); │ │ │ │ │ - this.events.triggerEvent("featureadded", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: insertXY │ │ │ │ │ - * Insert a point in the current sketch given x & y coordinates. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {Number} The x-coordinate of the point. │ │ │ │ │ - * y - {Number} The y-coordinate of the point. │ │ │ │ │ - */ │ │ │ │ │ - insertXY: function(x, y) { │ │ │ │ │ - if (this.handler && this.handler.line) { │ │ │ │ │ - this.handler.insertXY(x, y); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: insertDeltaXY │ │ │ │ │ - * Insert a point given offsets from the previously inserted point. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * dx - {Number} The x-coordinate offset of the point. │ │ │ │ │ - * dy - {Number} The y-coordinate offset of the point. │ │ │ │ │ - */ │ │ │ │ │ - insertDeltaXY: function(dx, dy) { │ │ │ │ │ - if (this.handler && this.handler.line) { │ │ │ │ │ - this.handler.insertDeltaXY(dx, dy); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: insertDirectionLength │ │ │ │ │ - * Insert a point in the current sketch given a direction and a length. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * direction - {Number} Degrees clockwise from the positive x-axis. │ │ │ │ │ - * length - {Number} Distance from the previously drawn point. │ │ │ │ │ - */ │ │ │ │ │ - insertDirectionLength: function(direction, length) { │ │ │ │ │ - if (this.handler && this.handler.line) { │ │ │ │ │ - this.handler.insertDirectionLength(direction, length); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: insertDeflectionLength │ │ │ │ │ - * Insert a point in the current sketch given a deflection and a length. │ │ │ │ │ - * The deflection should be degrees clockwise from the previously │ │ │ │ │ - * digitized segment. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * deflection - {Number} Degrees clockwise from the previous segment. │ │ │ │ │ - * length - {Number} Distance from the previously drawn point. │ │ │ │ │ - */ │ │ │ │ │ - insertDeflectionLength: function(deflection, length) { │ │ │ │ │ - if (this.handler && this.handler.line) { │ │ │ │ │ - this.handler.insertDeflectionLength(deflection, length); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: undo │ │ │ │ │ - * Remove the most recently added point in the current sketch geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} An edit was undone. │ │ │ │ │ - */ │ │ │ │ │ - undo: function() { │ │ │ │ │ - return this.handler.undo && this.handler.undo(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: redo │ │ │ │ │ - * Reinsert the most recently removed point resulting from an <undo> call. │ │ │ │ │ - * The undo stack is deleted whenever a point is added by other means. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} An edit was redone. │ │ │ │ │ - */ │ │ │ │ │ - redo: function() { │ │ │ │ │ - return this.handler.redo && this.handler.redo(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: finishSketch │ │ │ │ │ - * Finishes the sketch without including the currently drawn point. │ │ │ │ │ - * This method can be called to terminate drawing programmatically │ │ │ │ │ - * instead of waiting for the user to end the sketch. │ │ │ │ │ - */ │ │ │ │ │ - finishSketch: function() { │ │ │ │ │ - this.handler.finishGeometry(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: cancel │ │ │ │ │ - * Cancel the current sketch. This removes the current sketch and keeps │ │ │ │ │ - * the drawing control active. │ │ │ │ │ - */ │ │ │ │ │ - cancel: function() { │ │ │ │ │ - this.handler.cancel(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.DrawFeature" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Geolocate.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Projection.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Geolocate │ │ │ │ │ - * The Geolocate control wraps w3c geolocation API into control that can be │ │ │ │ │ - * bound to a map, and generate events on location update │ │ │ │ │ - * │ │ │ │ │ - * To use this control requires to load the proj4js library if the projection │ │ │ │ │ - * of the map is not EPSG:4326 or EPSG:900913. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Geolocate = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * locationupdated - Triggered when browser return a new position. Listeners will │ │ │ │ │ - * receive an object with a 'position' property which is the browser.geolocation.position │ │ │ │ │ - * native object, as well as a 'point' property which is the location transformed in the │ │ │ │ │ - * current map projection. │ │ │ │ │ - * locationfailed - Triggered when geolocation has failed │ │ │ │ │ - * locationuncapable - Triggered when control is activated on a browser │ │ │ │ │ - * which doesn't support geolocation │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: geolocation │ │ │ │ │ - * {Object} The geolocation engine, as a property to be possibly mocked. │ │ │ │ │ - * This is set lazily to avoid a memory leak in IE9. │ │ │ │ │ - */ │ │ │ │ │ - geolocation: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: available │ │ │ │ │ - * {Boolean} The navigator.geolocation object is available. │ │ │ │ │ - */ │ │ │ │ │ - available: ('geolocation' in navigator), │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: bind │ │ │ │ │ - * {Boolean} If true, map center will be set on location update. │ │ │ │ │ - */ │ │ │ │ │ - bind: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: watch │ │ │ │ │ - * {Boolean} If true, position will be update regularly. │ │ │ │ │ - */ │ │ │ │ │ - watch: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geolocationOptions │ │ │ │ │ - * {Object} Options to pass to the navigator's geolocation API. See │ │ │ │ │ - * <http://dev.w3.org/geo/api/spec-source.html>. No specific │ │ │ │ │ - * option is passed to the geolocation API by default. │ │ │ │ │ - */ │ │ │ │ │ - geolocationOptions: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Geolocate │ │ │ │ │ - * Create a new control to deal with browser geolocation API │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Activates the control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The control was effectively activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (this.available && !this.geolocation) { │ │ │ │ │ - // set lazily to avoid IE9 memory leak │ │ │ │ │ - this.geolocation = navigator.geolocation; │ │ │ │ │ - } │ │ │ │ │ - if (!this.geolocation) { │ │ │ │ │ - this.events.triggerEvent("locationuncapable"); │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - if (this.watch) { │ │ │ │ │ - this.watchId = this.geolocation.watchPosition( │ │ │ │ │ - OpenLayers.Function.bind(this.geolocate, this), │ │ │ │ │ - OpenLayers.Function.bind(this.failure, this), │ │ │ │ │ - this.geolocationOptions │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - this.getCurrentLocation(); │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Deactivates the control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The control was effectively deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active && this.watchId !== null) { │ │ │ │ │ - this.geolocation.clearWatch(this.watchId); │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Control.prototype.deactivate.apply( │ │ │ │ │ - this, arguments │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: geolocate │ │ │ │ │ - * Activates the control. │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - geolocate: function(position) { │ │ │ │ │ - var center = new OpenLayers.LonLat( │ │ │ │ │ - position.coords.longitude, │ │ │ │ │ - position.coords.latitude │ │ │ │ │ - ).transform( │ │ │ │ │ - new OpenLayers.Projection("EPSG:4326"), │ │ │ │ │ - this.map.getProjectionObject() │ │ │ │ │ - ); │ │ │ │ │ - if (this.bind) { │ │ │ │ │ - this.map.setCenter(center); │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("locationupdated", { │ │ │ │ │ - position: position, │ │ │ │ │ - point: new OpenLayers.Geometry.Point( │ │ │ │ │ - center.lon, center.lat │ │ │ │ │ - ) │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getCurrentLocation │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Returns true if a event will be fired (successfull │ │ │ │ │ - * registration) │ │ │ │ │ - */ │ │ │ │ │ - getCurrentLocation: function() { │ │ │ │ │ - if (!this.active || this.watch) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - this.geolocation.getCurrentPosition( │ │ │ │ │ - OpenLayers.Function.bind(this.geolocate, this), │ │ │ │ │ - OpenLayers.Function.bind(this.failure, this), │ │ │ │ │ - this.geolocationOptions │ │ │ │ │ - ); │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: failure │ │ │ │ │ - * method called on browser's geolocation failure │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - failure: function(error) { │ │ │ │ │ - this.events.triggerEvent("locationfailed", { │ │ │ │ │ - error: error │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Geolocate" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Events/buttonclick.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Events.buttonclick │ │ │ │ │ - * Extension event type for handling buttons on top of a dom element. This │ │ │ │ │ - * event type fires "buttonclick" on its <target> when a button was │ │ │ │ │ - * clicked. Buttons are detected by the "olButton" class. │ │ │ │ │ - * │ │ │ │ │ - * This event type makes sure that button clicks do not interfere with other │ │ │ │ │ - * events that are registered on the same <element>. │ │ │ │ │ - * │ │ │ │ │ - * Event types provided by this extension: │ │ │ │ │ - * - *buttonclick* Triggered when a button is clicked. Listeners receive an │ │ │ │ │ - * object with a *buttonElement* property referencing the dom element of │ │ │ │ │ - * the clicked button, and an *buttonXY* property with the click position │ │ │ │ │ - * relative to the button. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Events.buttonclick = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: target │ │ │ │ │ - * {<OpenLayers.Events>} The events instance that the buttonclick event will │ │ │ │ │ - * be triggered on. │ │ │ │ │ - */ │ │ │ │ │ - target: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: events │ │ │ │ │ - * {Array} Events to observe and conditionally stop from propagating when │ │ │ │ │ - * an element with the olButton class (or its olAlphaImg child) is │ │ │ │ │ - * clicked. │ │ │ │ │ - */ │ │ │ │ │ - events: [ │ │ │ │ │ - 'mousedown', 'mouseup', 'click', 'dblclick', │ │ │ │ │ - 'touchstart', 'touchmove', 'touchend', 'keydown' │ │ │ │ │ - ], │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: startRegEx │ │ │ │ │ - * {RegExp} Regular expression to test Event.type for events that start │ │ │ │ │ - * a buttonclick sequence. │ │ │ │ │ - */ │ │ │ │ │ - startRegEx: /^mousedown|touchstart$/, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: cancelRegEx │ │ │ │ │ - * {RegExp} Regular expression to test Event.type for events that cancel │ │ │ │ │ - * a buttonclick sequence. │ │ │ │ │ - */ │ │ │ │ │ - cancelRegEx: /^touchmove$/, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: completeRegEx │ │ │ │ │ - * {RegExp} Regular expression to test Event.type for events that complete │ │ │ │ │ - * a buttonclick sequence. │ │ │ │ │ - */ │ │ │ │ │ - completeRegEx: /^mouseup|touchend$/, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: startEvt │ │ │ │ │ - * {Event} The event that started the click sequence │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Events.buttonclick │ │ │ │ │ - * Construct a buttonclick event type. Applications are not supposed to │ │ │ │ │ - * create instances of this class - they are created on demand by │ │ │ │ │ - * <OpenLayers.Events> instances. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * target - {<OpenLayers.Events>} The events instance that the buttonclick │ │ │ │ │ - * event will be triggered on. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(target) { │ │ │ │ │ - this.target = target; │ │ │ │ │ - for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ - this.target.register(this.events[i], this, this.buttonClick, { │ │ │ │ │ - extension: true │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ - this.target.unregister(this.events[i], this, this.buttonClick); │ │ │ │ │ - } │ │ │ │ │ - delete this.target; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getPressedButton │ │ │ │ │ - * Get the pressed button, if any. Returns undefined if no button │ │ │ │ │ - * was pressed. │ │ │ │ │ - * │ │ │ │ │ - * Arguments: │ │ │ │ │ - * element - {DOMElement} The event target. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The button element, or undefined. │ │ │ │ │ - */ │ │ │ │ │ - getPressedButton: function(element) { │ │ │ │ │ - var depth = 3, // limit the search depth │ │ │ │ │ - button; │ │ │ │ │ - do { │ │ │ │ │ - if (OpenLayers.Element.hasClass(element, "olButton")) { │ │ │ │ │ - // hit! │ │ │ │ │ - button = element; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - element = element.parentNode; │ │ │ │ │ - } while (--depth > 0 && element); │ │ │ │ │ - return button; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: ignore │ │ │ │ │ - * Check for event target elements that should be ignored by OpenLayers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * element - {DOMElement} The event target. │ │ │ │ │ - */ │ │ │ │ │ - ignore: function(element) { │ │ │ │ │ - var depth = 3, │ │ │ │ │ - ignore = false; │ │ │ │ │ - do { │ │ │ │ │ - if (element.nodeName.toLowerCase() === 'a') { │ │ │ │ │ - ignore = true; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - element = element.parentNode; │ │ │ │ │ - } while (--depth > 0 && element); │ │ │ │ │ - return ignore; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buttonClick │ │ │ │ │ - * Check if a button was clicked, and fire the buttonclick event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - buttonClick: function(evt) { │ │ │ │ │ - var propagate = true, │ │ │ │ │ - element = OpenLayers.Event.element(evt); │ │ │ │ │ - if (element && (OpenLayers.Event.isLeftClick(evt) || !~evt.type.indexOf("mouse"))) { │ │ │ │ │ - // was a button pressed? │ │ │ │ │ - var button = this.getPressedButton(element); │ │ │ │ │ - if (button) { │ │ │ │ │ - if (evt.type === "keydown") { │ │ │ │ │ - switch (evt.keyCode) { │ │ │ │ │ - case OpenLayers.Event.KEY_RETURN: │ │ │ │ │ - case OpenLayers.Event.KEY_SPACE: │ │ │ │ │ - this.target.triggerEvent("buttonclick", { │ │ │ │ │ - buttonElement: button │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } else if (this.startEvt) { │ │ │ │ │ - if (this.completeRegEx.test(evt.type)) { │ │ │ │ │ - var pos = OpenLayers.Util.pagePosition(button); │ │ │ │ │ - var viewportElement = OpenLayers.Util.getViewportElement(); │ │ │ │ │ - var scrollTop = window.pageYOffset || viewportElement.scrollTop; │ │ │ │ │ - var scrollLeft = window.pageXOffset || viewportElement.scrollLeft; │ │ │ │ │ - pos[0] = pos[0] - scrollLeft; │ │ │ │ │ - pos[1] = pos[1] - scrollTop; │ │ │ │ │ - │ │ │ │ │ - this.target.triggerEvent("buttonclick", { │ │ │ │ │ - buttonElement: button, │ │ │ │ │ - buttonXY: { │ │ │ │ │ - x: this.startEvt.clientX - pos[0], │ │ │ │ │ - y: this.startEvt.clientY - pos[1] │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - if (this.cancelRegEx.test(evt.type)) { │ │ │ │ │ - delete this.startEvt; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false; │ │ │ │ │ - } │ │ │ │ │ - if (this.startRegEx.test(evt.type)) { │ │ │ │ │ - this.startEvt = evt; │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - propagate = !this.ignore(OpenLayers.Event.element(evt)); │ │ │ │ │ - delete this.startEvt; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return propagate; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Panel.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Panel │ │ │ │ │ - * The Panel control is a container for other controls. With it toolbars │ │ │ │ │ - * may be composed. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Panel = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - /** │ │ │ │ │ - * Property: controls │ │ │ │ │ - * {Array(<OpenLayers.Control>)} │ │ │ │ │ - */ │ │ │ │ │ - controls: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ - */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: defaultControl │ │ │ │ │ - * {<OpenLayers.Control>} The control which is activated when the control is │ │ │ │ │ - * activated (turned on), which also happens at instantiation. │ │ │ │ │ - * If <saveState> is true, <defaultControl> will be nullified after the │ │ │ │ │ - * first activation of the panel. │ │ │ │ │ - */ │ │ │ │ │ - defaultControl: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: saveState │ │ │ │ │ - * {Boolean} If set to true, the active state of this panel's controls will │ │ │ │ │ - * be stored on panel deactivation, and restored on reactivation. Default │ │ │ │ │ - * is false. │ │ │ │ │ - */ │ │ │ │ │ - saveState: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: allowDepress │ │ │ │ │ - * {Boolean} If is true the <OpenLayers.Control.TYPE_TOOL> controls can │ │ │ │ │ - * be deactivated by clicking the icon that represents them. Default │ │ │ │ │ - * is false. │ │ │ │ │ - */ │ │ │ │ │ - allowDepress: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: activeState │ │ │ │ │ - * {Object} stores the active state of this panel's controls. │ │ │ │ │ - */ │ │ │ │ │ - activeState: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Panel │ │ │ │ │ - * Create a new control panel. │ │ │ │ │ - * │ │ │ │ │ - * Each control in the panel is represented by an icon. When clicking │ │ │ │ │ - * on an icon, the <activateControl> method is called. │ │ │ │ │ - * │ │ │ │ │ - * Specific properties for controls on a panel: │ │ │ │ │ - * type - {Number} One of <OpenLayers.Control.TYPE_TOOL>, │ │ │ │ │ - * <OpenLayers.Control.TYPE_TOGGLE>, <OpenLayers.Control.TYPE_BUTTON>. │ │ │ │ │ - * If not provided, <OpenLayers.Control.TYPE_TOOL> is assumed. │ │ │ │ │ - * title - {string} Text displayed when mouse is over the icon that │ │ │ │ │ - * represents the control. │ │ │ │ │ - * │ │ │ │ │ - * The <OpenLayers.Control.type> of a control determines the behavior when │ │ │ │ │ - * clicking its icon: │ │ │ │ │ - * <OpenLayers.Control.TYPE_TOOL> - The control is activated and other │ │ │ │ │ - * controls of this type in the same panel are deactivated. This is │ │ │ │ │ - * the default type. │ │ │ │ │ - * <OpenLayers.Control.TYPE_TOGGLE> - The active state of the control is │ │ │ │ │ - * toggled. │ │ │ │ │ - * <OpenLayers.Control.TYPE_BUTTON> - The │ │ │ │ │ - * <OpenLayers.Control.Button.trigger> method of the control is called, │ │ │ │ │ - * but its active state is not changed. │ │ │ │ │ - * │ │ │ │ │ - * If a control is <OpenLayers.Control.active>, it will be drawn with the │ │ │ │ │ - * olControl[Name]ItemActive class, otherwise with the │ │ │ │ │ - * olControl[Name]ItemInactive class. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be used │ │ │ │ │ - * to extend the control. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.controls = []; │ │ │ │ │ - this.activeState = {}; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - for (var ctl, i = this.controls.length - 1; i >= 0; i--) { │ │ │ │ │ - ctl = this.controls[i]; │ │ │ │ │ - if (ctl.events) { │ │ │ │ │ - ctl.events.un({ │ │ │ │ │ - activate: this.iconOn, │ │ │ │ │ - deactivate: this.iconOff │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - ctl.panel_div = null; │ │ │ │ │ - } │ │ │ │ │ - this.activeState = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - control = this.controls[i]; │ │ │ │ │ - if (control === this.defaultControl || │ │ │ │ │ - (this.saveState && this.activeState[control.id])) { │ │ │ │ │ - control.activate(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.saveState === true) { │ │ │ │ │ - this.defaultControl = null; │ │ │ │ │ - } │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - control = this.controls[i]; │ │ │ │ │ - this.activeState[control.id] = control.deactivate(); │ │ │ │ │ - } │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (this.outsideViewport) { │ │ │ │ │ - this.events.attachToElement(this.div); │ │ │ │ │ - this.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ - } else { │ │ │ │ │ - this.map.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ - } │ │ │ │ │ - this.addControlsToMap(this.controls); │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: redraw │ │ │ │ │ - */ │ │ │ │ │ - redraw: function() { │ │ │ │ │ - for (var l = this.div.childNodes.length, i = l - 1; i >= 0; i--) { │ │ │ │ │ - this.div.removeChild(this.div.childNodes[i]); │ │ │ │ │ - } │ │ │ │ │ - this.div.innerHTML = ""; │ │ │ │ │ - if (this.active) { │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - this.div.appendChild(this.controls[i].panel_div); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activateControl │ │ │ │ │ - * This method is called when the user click on the icon representing a │ │ │ │ │ - * control in the panel. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} │ │ │ │ │ - */ │ │ │ │ │ - activateControl: function(control) { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - if (control.type == OpenLayers.Control.TYPE_BUTTON) { │ │ │ │ │ - control.trigger(); │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (control.type == OpenLayers.Control.TYPE_TOGGLE) { │ │ │ │ │ - if (control.active) { │ │ │ │ │ - control.deactivate(); │ │ │ │ │ - } else { │ │ │ │ │ - control.activate(); │ │ │ │ │ - } │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (this.allowDepress && control.active) { │ │ │ │ │ - control.deactivate(); │ │ │ │ │ - } else { │ │ │ │ │ - var c; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - c = this.controls[i]; │ │ │ │ │ - if (c != control && │ │ │ │ │ - (c.type === OpenLayers.Control.TYPE_TOOL || c.type == null)) { │ │ │ │ │ - c.deactivate(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - control.activate(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: addControls │ │ │ │ │ - * To build a toolbar, you add a set of controls to it. addControls │ │ │ │ │ - * lets you add a single control or a list of controls to the │ │ │ │ │ - * Control Panel. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * controls - {<OpenLayers.Control>} Controls to add in the panel. │ │ │ │ │ - */ │ │ │ │ │ - addControls: function(controls) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(controls))) { │ │ │ │ │ - controls = [controls]; │ │ │ │ │ - } │ │ │ │ │ - this.controls = this.controls.concat(controls); │ │ │ │ │ - │ │ │ │ │ - for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ - var control = controls[i], │ │ │ │ │ - element = this.createControlMarkup(control); │ │ │ │ │ - OpenLayers.Element.addClass(element, │ │ │ │ │ - control.displayClass + "ItemInactive"); │ │ │ │ │ - OpenLayers.Element.addClass(element, "olButton"); │ │ │ │ │ - if (control.title != "" && !element.title) { │ │ │ │ │ - element.title = control.title; │ │ │ │ │ - } │ │ │ │ │ - control.panel_div = element; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.map) { // map.addControl() has already been called on the panel │ │ │ │ │ - this.addControlsToMap(controls); │ │ │ │ │ - this.redraw(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: createControlMarkup │ │ │ │ │ - * This function just creates a div for the control. If specific HTML │ │ │ │ │ - * markup is needed this function can be overridden in specific classes, │ │ │ │ │ - * or at panel instantiation time: │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var panel = new OpenLayers.Control.Panel({ │ │ │ │ │ - * defaultControl: control, │ │ │ │ │ - * // ovverride createControlMarkup to create actual buttons │ │ │ │ │ - * // including texts wrapped into span elements. │ │ │ │ │ - * createControlMarkup: function(control) { │ │ │ │ │ - * var button = document.createElement('button'), │ │ │ │ │ - * span = document.createElement('span'); │ │ │ │ │ - * if (control.text) { │ │ │ │ │ - * span.innerHTML = control.text; │ │ │ │ │ - * } │ │ │ │ │ - * return button; │ │ │ │ │ - * } │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control to create the HTML │ │ │ │ │ - * markup for. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The markup. │ │ │ │ │ - */ │ │ │ │ │ - createControlMarkup: function(control) { │ │ │ │ │ - return document.createElement("div"); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addControlsToMap │ │ │ │ │ - * Only for internal use in draw() and addControls() methods. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * controls - {Array(<OpenLayers.Control>)} Controls to add into map. │ │ │ │ │ - */ │ │ │ │ │ - addControlsToMap: function(controls) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ - control = controls[i]; │ │ │ │ │ - if (control.autoActivate === true) { │ │ │ │ │ - control.autoActivate = false; │ │ │ │ │ - this.map.addControl(control); │ │ │ │ │ - control.autoActivate = true; │ │ │ │ │ - } else { │ │ │ │ │ - this.map.addControl(control); │ │ │ │ │ - control.deactivate(); │ │ │ │ │ - } │ │ │ │ │ - control.events.on({ │ │ │ │ │ - activate: this.iconOn, │ │ │ │ │ - deactivate: this.iconOff │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: iconOn │ │ │ │ │ - * Internal use, for use only with "controls[i].events.on/un". │ │ │ │ │ - */ │ │ │ │ │ - iconOn: function() { │ │ │ │ │ - var d = this.panel_div; // "this" refers to a control on panel! │ │ │ │ │ - var re = new RegExp("\\b(" + this.displayClass + "Item)Inactive\\b"); │ │ │ │ │ - d.className = d.className.replace(re, "$1Active"); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: iconOff │ │ │ │ │ - * Internal use, for use only with "controls[i].events.on/un". │ │ │ │ │ - */ │ │ │ │ │ - iconOff: function() { │ │ │ │ │ - var d = this.panel_div; // "this" refers to a control on panel! │ │ │ │ │ - var re = new RegExp("\\b(" + this.displayClass + "Item)Active\\b"); │ │ │ │ │ - d.className = d.className.replace(re, "$1Inactive"); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onButtonClick │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - var controls = this.controls, │ │ │ │ │ - button = evt.buttonElement; │ │ │ │ │ - for (var i = controls.length - 1; i >= 0; --i) { │ │ │ │ │ - if (controls[i].panel_div === button) { │ │ │ │ │ - this.activateControl(controls[i]); │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getControlsBy │ │ │ │ │ - * Get a list of controls with properties matching the given criteria. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * property - {String} A control property to be matched. │ │ │ │ │ - * match - {String | Object} A string to match. Can also be a regular │ │ │ │ │ - * expression literal or object. In addition, it can be any object │ │ │ │ │ - * with a method named test. For reqular expressions or other, if │ │ │ │ │ - * match.test(control[property]) evaluates to true, the control will be │ │ │ │ │ - * included in the array returned. If no controls are found, an empty │ │ │ │ │ - * array is returned. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Control>)} A list of controls matching the given criteria. │ │ │ │ │ - * An empty array is returned if no matches are found. │ │ │ │ │ - */ │ │ │ │ │ - getControlsBy: function(property, match) { │ │ │ │ │ - var test = (typeof match.test == "function"); │ │ │ │ │ - var found = OpenLayers.Array.filter(this.controls, function(item) { │ │ │ │ │ - return item[property] == match || (test && match.test(item[property])); │ │ │ │ │ - }); │ │ │ │ │ - return found; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getControlsByName │ │ │ │ │ - * Get a list of contorls with names matching the given name. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * match - {String | Object} A control name. The name can also be a regular │ │ │ │ │ - * expression literal or object. In addition, it can be any object │ │ │ │ │ - * with a method named test. For reqular expressions or other, if │ │ │ │ │ - * name.test(control.name) evaluates to true, the control will be included │ │ │ │ │ - * in the list of controls returned. If no controls are found, an empty │ │ │ │ │ - * array is returned. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Control>)} A list of controls matching the given name. │ │ │ │ │ - * An empty array is returned if no matches are found. │ │ │ │ │ - */ │ │ │ │ │ - getControlsByName: function(match) { │ │ │ │ │ - return this.getControlsBy("name", match); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getControlsByClass │ │ │ │ │ - * Get a list of controls of a given type (CLASS_NAME). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * match - {String | Object} A control class name. The type can also be a │ │ │ │ │ - * regular expression literal or object. In addition, it can be any │ │ │ │ │ - * object with a method named test. For reqular expressions or other, │ │ │ │ │ - * if type.test(control.CLASS_NAME) evaluates to true, the control will │ │ │ │ │ - * be included in the list of controls returned. If no controls are │ │ │ │ │ - * found, an empty array is returned. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Control>)} A list of controls matching the given type. │ │ │ │ │ - * An empty array is returned if no matches are found. │ │ │ │ │ - */ │ │ │ │ │ - getControlsByClass: function(match) { │ │ │ │ │ - return this.getControlsBy("CLASS_NAME", match); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Panel" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Attribution.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Attribution │ │ │ │ │ - * The attribution control adds attribution from layers to the map display. │ │ │ │ │ - * It uses 'attribution' property of each layer. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Attribution = │ │ │ │ │ - OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: separator │ │ │ │ │ - * {String} String used to separate layers. │ │ │ │ │ - */ │ │ │ │ │ - separator: ", ", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: template │ │ │ │ │ - * {String} Template for the attribution. This has to include the substring │ │ │ │ │ - * "${layers}", which will be replaced by the layer specific │ │ │ │ │ - * attributions, separated by <separator>. The default is "${layers}". │ │ │ │ │ - */ │ │ │ │ │ - template: "${layers}", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Attribution │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Options for control. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Destroy control. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - "removelayer": this.updateAttribution, │ │ │ │ │ - "addlayer": this.updateAttribution, │ │ │ │ │ - "changelayer": this.updateAttribution, │ │ │ │ │ - "changebaselayer": this.updateAttribution, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Initialize control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - 'changebaselayer': this.updateAttribution, │ │ │ │ │ - 'changelayer': this.updateAttribution, │ │ │ │ │ - 'addlayer': this.updateAttribution, │ │ │ │ │ - 'removelayer': this.updateAttribution, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.updateAttribution(); │ │ │ │ │ - │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateAttribution │ │ │ │ │ - * Update attribution string. │ │ │ │ │ - */ │ │ │ │ │ - updateAttribution: function() { │ │ │ │ │ - var attributions = []; │ │ │ │ │ - if (this.map && this.map.layers) { │ │ │ │ │ - for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if (layer.attribution && layer.getVisibility()) { │ │ │ │ │ - // add attribution only if attribution text is unique │ │ │ │ │ - if (OpenLayers.Util.indexOf( │ │ │ │ │ - attributions, layer.attribution) === -1) { │ │ │ │ │ - attributions.push(layer.attribution); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.div.innerHTML = OpenLayers.String.format(this.template, { │ │ │ │ │ - layers: attributions.join(this.separator) │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Attribution" │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Zoom.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Zoom │ │ │ │ │ - * The Zoom control is a pair of +/- links for zooming in and out. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Zoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomInText │ │ │ │ │ - * {String} │ │ │ │ │ - * Text for zoom-in link. Default is "+". │ │ │ │ │ - */ │ │ │ │ │ - zoomInText: "+", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomInId │ │ │ │ │ - * {String} │ │ │ │ │ - * Instead of having the control create a zoom in link, you can provide │ │ │ │ │ - * the identifier for an anchor element already added to the document. │ │ │ │ │ - * By default, an element with id "olZoomInLink" will be searched for │ │ │ │ │ - * and used if it exists. │ │ │ │ │ - */ │ │ │ │ │ - zoomInId: "olZoomInLink", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomOutText │ │ │ │ │ - * {String} │ │ │ │ │ - * Text for zoom-out link. Default is "\u2212". │ │ │ │ │ - */ │ │ │ │ │ - zoomOutText: "\u2212", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomOutId │ │ │ │ │ - * {String} │ │ │ │ │ - * Instead of having the control create a zoom out link, you can provide │ │ │ │ │ - * the identifier for an anchor element already added to the document. │ │ │ │ │ - * By default, an element with id "olZoomOutLink" will be searched for │ │ │ │ │ - * and used if it exists. │ │ │ │ │ - */ │ │ │ │ │ - zoomOutId: "olZoomOutLink", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A reference to the DOMElement containing the zoom links. │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - var div = OpenLayers.Control.prototype.draw.apply(this), │ │ │ │ │ - links = this.getOrCreateLinks(div), │ │ │ │ │ - zoomIn = links.zoomIn, │ │ │ │ │ - zoomOut = links.zoomOut, │ │ │ │ │ - eventsInstance = this.map.events; │ │ │ │ │ - │ │ │ │ │ - if (zoomOut.parentNode !== div) { │ │ │ │ │ - eventsInstance = this.events; │ │ │ │ │ - eventsInstance.attachToElement(zoomOut.parentNode); │ │ │ │ │ - } │ │ │ │ │ - eventsInstance.register("buttonclick", this, this.onZoomClick); │ │ │ │ │ - │ │ │ │ │ - this.zoomInLink = zoomIn; │ │ │ │ │ - this.zoomOutLink = zoomOut; │ │ │ │ │ - return div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getOrCreateLinks │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * el - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ - * Return: │ │ │ │ │ - * {Object} Object with zoomIn and zoomOut properties referencing links. │ │ │ │ │ - */ │ │ │ │ │ - getOrCreateLinks: function(el) { │ │ │ │ │ - var zoomIn = document.getElementById(this.zoomInId), │ │ │ │ │ - zoomOut = document.getElementById(this.zoomOutId); │ │ │ │ │ - if (!zoomIn) { │ │ │ │ │ - zoomIn = document.createElement("a"); │ │ │ │ │ - zoomIn.href = "#zoomIn"; │ │ │ │ │ - zoomIn.appendChild(document.createTextNode(this.zoomInText)); │ │ │ │ │ - zoomIn.className = "olControlZoomIn"; │ │ │ │ │ - el.appendChild(zoomIn); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Element.addClass(zoomIn, "olButton"); │ │ │ │ │ - if (!zoomOut) { │ │ │ │ │ - zoomOut = document.createElement("a"); │ │ │ │ │ - zoomOut.href = "#zoomOut"; │ │ │ │ │ - zoomOut.appendChild(document.createTextNode(this.zoomOutText)); │ │ │ │ │ - zoomOut.className = "olControlZoomOut"; │ │ │ │ │ - el.appendChild(zoomOut); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Element.addClass(zoomOut, "olButton"); │ │ │ │ │ - return { │ │ │ │ │ - zoomIn: zoomIn, │ │ │ │ │ - zoomOut: zoomOut │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onZoomClick │ │ │ │ │ - * Called when zoomin/out link is clicked. │ │ │ │ │ - */ │ │ │ │ │ - onZoomClick: function(evt) { │ │ │ │ │ - var button = evt.buttonElement; │ │ │ │ │ - if (button === this.zoomInLink) { │ │ │ │ │ - this.map.zoomIn(); │ │ │ │ │ - } else if (button === this.zoomOutLink) { │ │ │ │ │ - this.map.zoomOut(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Clean up. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onZoomClick); │ │ │ │ │ - } │ │ │ │ │ - delete this.zoomInLink; │ │ │ │ │ - delete this.zoomOutLink; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Zoom" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Feature.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Handler.Feature │ │ │ │ │ - * Handler to respond to mouse events related to a drawn feature. Callbacks │ │ │ │ │ - * with the following keys will be notified of the following events │ │ │ │ │ - * associated with features: click, clickout, over, out, and dblclick. │ │ │ │ │ - * │ │ │ │ │ - * This handler stops event propagation for mousedown and mouseup if those │ │ │ │ │ - * browser events target features that can be selected. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.Feature = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: EVENTMAP │ │ │ │ │ - * {Object} A object mapping the browser events to objects with callback │ │ │ │ │ - * keys for in and out. │ │ │ │ │ - */ │ │ │ │ │ - EVENTMAP: { │ │ │ │ │ - 'click': { │ │ │ │ │ - 'in': 'click', │ │ │ │ │ - 'out': 'clickout' │ │ │ │ │ - }, │ │ │ │ │ - 'mousemove': { │ │ │ │ │ - 'in': 'over', │ │ │ │ │ - 'out': 'out' │ │ │ │ │ - }, │ │ │ │ │ - 'dblclick': { │ │ │ │ │ - 'in': 'dblclick', │ │ │ │ │ - 'out': null │ │ │ │ │ - }, │ │ │ │ │ - 'mousedown': { │ │ │ │ │ - 'in': null, │ │ │ │ │ - 'out': null │ │ │ │ │ - }, │ │ │ │ │ - 'mouseup': { │ │ │ │ │ - 'in': null, │ │ │ │ │ - 'out': null │ │ │ │ │ - }, │ │ │ │ │ - 'touchstart': { │ │ │ │ │ - 'in': 'click', │ │ │ │ │ - 'out': 'clickout' │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: feature │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} The last feature that was hovered. │ │ │ │ │ - */ │ │ │ │ │ - feature: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: lastFeature │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} The last feature that was handled. │ │ │ │ │ - */ │ │ │ │ │ - lastFeature: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: down │ │ │ │ │ - * {<OpenLayers.Pixel>} The location of the last mousedown. │ │ │ │ │ - */ │ │ │ │ │ - down: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: up │ │ │ │ │ - * {<OpenLayers.Pixel>} The location of the last mouseup. │ │ │ │ │ - */ │ │ │ │ │ - up: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: clickTolerance │ │ │ │ │ - * {Number} The number of pixels the mouse can move between mousedown │ │ │ │ │ - * and mouseup for the event to still be considered a click. │ │ │ │ │ - * Dragging the map should not trigger the click and clickout callbacks │ │ │ │ │ - * unless the map is moved by less than this tolerance. Defaults to 4. │ │ │ │ │ - */ │ │ │ │ │ - clickTolerance: 4, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: geometryTypes │ │ │ │ │ - * To restrict dragging to a limited set of geometry types, send a list │ │ │ │ │ - * of strings corresponding to the geometry class names. │ │ │ │ │ - * │ │ │ │ │ - * @type Array(String) │ │ │ │ │ - */ │ │ │ │ │ - geometryTypes: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: stopClick │ │ │ │ │ - * {Boolean} If stopClick is set to true, handled clicks do not │ │ │ │ │ - * propagate to other click listeners. Otherwise, handled clicks │ │ │ │ │ - * do propagate. Unhandled clicks always propagate, whatever the │ │ │ │ │ - * value of stopClick. Defaults to true. │ │ │ │ │ - */ │ │ │ │ │ - stopClick: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: stopDown │ │ │ │ │ - * {Boolean} If stopDown is set to true, handled mousedowns do not │ │ │ │ │ - * propagate to other mousedown listeners. Otherwise, handled │ │ │ │ │ - * mousedowns do propagate. Unhandled mousedowns always propagate, │ │ │ │ │ - * whatever the value of stopDown. Defaults to true. │ │ │ │ │ - */ │ │ │ │ │ - stopDown: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: stopUp │ │ │ │ │ - * {Boolean} If stopUp is set to true, handled mouseups do not │ │ │ │ │ - * propagate to other mouseup listeners. Otherwise, handled mouseups │ │ │ │ │ - * do propagate. Unhandled mouseups always propagate, whatever the │ │ │ │ │ - * value of stopUp. Defaults to false. │ │ │ │ │ - */ │ │ │ │ │ - stopUp: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Feature │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} │ │ │ │ │ - * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ - * callbacks - {Object} An object with a 'over' property whos value is │ │ │ │ │ - * a function to be called when the mouse is over a feature. The │ │ │ │ │ - * callback should expect to recieve a single argument, the feature. │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(control, layer, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchstart │ │ │ │ │ - * Handle touchstart events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ - */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - return OpenLayers.Event.isMultiTouch(evt) ? │ │ │ │ │ - true : this.mousedown(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchmove │ │ │ │ │ - * Handle touchmove events. We just prevent the browser default behavior, │ │ │ │ │ - * for Android Webkit not to select text when moving the finger after │ │ │ │ │ - * selecting a feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mousedown │ │ │ │ │ - * Handle mouse down. Stop propagation if a feature is targeted by this │ │ │ │ │ - * event (stops map dragging during feature selection). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - // Feature selection is only done with a left click. Other handlers may stop the │ │ │ │ │ - // propagation of left-click mousedown events but not right-click mousedown events. │ │ │ │ │ - // This mismatch causes problems when comparing the location of the down and up │ │ │ │ │ - // events in the click function so it is important ignore right-clicks. │ │ │ │ │ - if (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ - this.down = evt.xy; │ │ │ │ │ - } │ │ │ │ │ - return this.handle(evt) ? !this.stopDown : true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mouseup │ │ │ │ │ - * Handle mouse up. Stop propagation if a feature is targeted by this │ │ │ │ │ - * event. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - this.up = evt.xy; │ │ │ │ │ - return this.handle(evt) ? !this.stopUp : true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: click │ │ │ │ │ - * Handle click. Call the "click" callback if click on a feature, │ │ │ │ │ - * or the "clickout" callback if click outside any feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - return this.handle(evt) ? !this.stopClick : true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mousemove │ │ │ │ │ - * Handle mouse moves. Call the "over" callback if moving in to a feature, │ │ │ │ │ - * or the "out" callback if moving out of a feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - if (!this.callbacks['over'] && !this.callbacks['out']) { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - this.handle(evt); │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: dblclick │ │ │ │ │ - * Handle dblclick. Call the "dblclick" callback if dblclick on a feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - return !this.handle(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: geometryTypeMatches │ │ │ │ │ - * Return true if the geometry type of the passed feature matches │ │ │ │ │ - * one of the geometry types in the geometryTypes array. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Vector.Feature>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - geometryTypeMatches: function(feature) { │ │ │ │ │ - return this.geometryTypes == null || │ │ │ │ │ - OpenLayers.Util.indexOf(this.geometryTypes, │ │ │ │ │ - feature.geometry.CLASS_NAME) > -1; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: handle │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The event occurred over a relevant feature. │ │ │ │ │ - */ │ │ │ │ │ - handle: function(evt) { │ │ │ │ │ - if (this.feature && !this.feature.layer) { │ │ │ │ │ - // feature has been destroyed │ │ │ │ │ - this.feature = null; │ │ │ │ │ - } │ │ │ │ │ - var type = evt.type; │ │ │ │ │ - var handled = false; │ │ │ │ │ - var previouslyIn = !!(this.feature); // previously in a feature │ │ │ │ │ - var click = (type == "click" || type == "dblclick" || type == "touchstart"); │ │ │ │ │ - this.feature = this.layer.getFeatureFromEvent(evt); │ │ │ │ │ - if (this.feature && !this.feature.layer) { │ │ │ │ │ - // feature has been destroyed │ │ │ │ │ - this.feature = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.lastFeature && !this.lastFeature.layer) { │ │ │ │ │ - // last feature has been destroyed │ │ │ │ │ - this.lastFeature = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.feature) { │ │ │ │ │ - if (type === "touchstart") { │ │ │ │ │ - // stop the event to prevent Android Webkit from │ │ │ │ │ - // "flashing" the map div │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ - } │ │ │ │ │ - var inNew = (this.feature != this.lastFeature); │ │ │ │ │ - if (this.geometryTypeMatches(this.feature)) { │ │ │ │ │ - // in to a feature │ │ │ │ │ - if (previouslyIn && inNew) { │ │ │ │ │ - // out of last feature and in to another │ │ │ │ │ - if (this.lastFeature) { │ │ │ │ │ - this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ - } │ │ │ │ │ - this.triggerCallback(type, 'in', [this.feature]); │ │ │ │ │ - } else if (!previouslyIn || click) { │ │ │ │ │ - // in feature for the first time │ │ │ │ │ - this.triggerCallback(type, 'in', [this.feature]); │ │ │ │ │ - } │ │ │ │ │ - this.lastFeature = this.feature; │ │ │ │ │ - handled = true; │ │ │ │ │ - } else { │ │ │ │ │ - // not in to a feature │ │ │ │ │ - if (this.lastFeature && (previouslyIn && inNew || click)) { │ │ │ │ │ - // out of last feature for the first time │ │ │ │ │ - this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ - } │ │ │ │ │ - // next time the mouse goes in a feature whose geometry type │ │ │ │ │ - // doesn't match we don't want to call the 'out' callback │ │ │ │ │ - // again, so let's set this.feature to null so that │ │ │ │ │ - // previouslyIn will evaluate to false the next time │ │ │ │ │ - // we enter handle. Yes, a bit hackish... │ │ │ │ │ - this.feature = null; │ │ │ │ │ - } │ │ │ │ │ - } else if (this.lastFeature && (previouslyIn || click)) { │ │ │ │ │ - this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ - } │ │ │ │ │ - return handled; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: triggerCallback │ │ │ │ │ - * Call the callback keyed in the event map with the supplied arguments. │ │ │ │ │ - * For click and clickout, the <clickTolerance> is checked first. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * type - {String} │ │ │ │ │ - */ │ │ │ │ │ - triggerCallback: function(type, mode, args) { │ │ │ │ │ - var key = this.EVENTMAP[type][mode]; │ │ │ │ │ - if (key) { │ │ │ │ │ - if (type == 'click' && this.up && this.down) { │ │ │ │ │ - // for click/clickout, only trigger callback if tolerance is met │ │ │ │ │ - var dpx = Math.sqrt( │ │ │ │ │ - Math.pow(this.up.x - this.down.x, 2) + │ │ │ │ │ - Math.pow(this.up.y - this.down.y, 2) │ │ │ │ │ - ); │ │ │ │ │ - if (dpx <= this.clickTolerance) { │ │ │ │ │ - this.callback(key, args); │ │ │ │ │ - } │ │ │ │ │ - // we're done with this set of events now: clear the cached │ │ │ │ │ - // positions so we can't trip over them later (this can occur │ │ │ │ │ - // if one of the up/down events gets eaten before it gets to us │ │ │ │ │ - // but we still get the click) │ │ │ │ │ - this.up = this.down = null; │ │ │ │ │ - } else { │ │ │ │ │ - this.callback(key, args); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Turn on the handler. Returns false if the handler was already active. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerToTop(); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - "removelayer": this.handleMapEvents, │ │ │ │ │ - "changelayer": this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - activated = true; │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Turn off the handler. Returns false if the handler was already active. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerBack(); │ │ │ │ │ - this.feature = null; │ │ │ │ │ - this.lastFeature = null; │ │ │ │ │ - this.down = null; │ │ │ │ │ - this.up = null; │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - "removelayer": this.handleMapEvents, │ │ │ │ │ - "changelayer": this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - deactivated = true; │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleMapEvents │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ - */ │ │ │ │ │ - handleMapEvents: function(evt) { │ │ │ │ │ - if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ - this.moveLayerToTop(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveLayerToTop │ │ │ │ │ - * Moves the layer for this handler to the top, so mouse events can reach │ │ │ │ │ - * it. │ │ │ │ │ - */ │ │ │ │ │ - moveLayerToTop: function() { │ │ │ │ │ - var index = Math.max(this.map.Z_INDEX_BASE['Feature'] - 1, │ │ │ │ │ - this.layer.getZIndex()) + 1; │ │ │ │ │ - this.layer.setZIndex(index); │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveLayerBack │ │ │ │ │ - * Moves the layer back to the position determined by the map's layers │ │ │ │ │ - * array. │ │ │ │ │ - */ │ │ │ │ │ - moveLayerBack: function() { │ │ │ │ │ - var index = this.layer.getZIndex() - 1; │ │ │ │ │ - if (index >= this.map.Z_INDEX_BASE['Feature']) { │ │ │ │ │ - this.layer.setZIndex(index); │ │ │ │ │ - } else { │ │ │ │ │ - this.map.setLayerZIndex(this.layer, │ │ │ │ │ - this.map.getLayerIndex(this.layer)); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Feature" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Vector/RootContainer.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.Vector.RootContainer │ │ │ │ │ - * A special layer type to combine multiple vector layers inside a single │ │ │ │ │ - * renderer root container. This class is not supposed to be instantiated │ │ │ │ │ - * from user space, it is a helper class for controls that require event │ │ │ │ │ - * processing for multiple vector layers. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Vector> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.Vector.RootContainer = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: displayInLayerSwitcher │ │ │ │ │ - * Set to false for this layer type │ │ │ │ │ - */ │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: layers │ │ │ │ │ - * Layers that are attached to this container. Required config option. │ │ │ │ │ - */ │ │ │ │ │ - layers: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Vector.RootContainer │ │ │ │ │ - * Create a new root container for multiple vector layer. This constructor │ │ │ │ │ - * is not supposed to be used from user space, it is only to be used by │ │ │ │ │ - * controls that need feature selection across multiple vector layers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} A name for the layer │ │ │ │ │ - * options - {Object} Optional object with non-default properties to set on │ │ │ │ │ - * the layer. │ │ │ │ │ - * │ │ │ │ │ - * Required options properties: │ │ │ │ │ - * layers - {Array(<OpenLayers.Layer.Vector>)} The layers managed by this │ │ │ │ │ - * container │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Vector.RootContainer>} A new vector layer root │ │ │ │ │ - * container │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: display │ │ │ │ │ - */ │ │ │ │ │ - display: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getFeatureFromEvent │ │ │ │ │ - * walk through the layers to find the feature returned by the event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} event object with a feature property │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - getFeatureFromEvent: function(evt) { │ │ │ │ │ - var layers = this.layers; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0; i < layers.length; i++) { │ │ │ │ │ - feature = layers[i].getFeatureFromEvent(evt); │ │ │ │ │ - if (feature) { │ │ │ │ │ - return feature; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.collectRoots(); │ │ │ │ │ - map.events.register("changelayer", this, this.handleChangeLayer); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeMap │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - map.events.unregister("changelayer", this, this.handleChangeLayer); │ │ │ │ │ - this.resetRoots(); │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: collectRoots │ │ │ │ │ - * Collects the root nodes of all layers this control is configured with │ │ │ │ │ - * and moveswien the nodes to this control's layer │ │ │ │ │ - */ │ │ │ │ │ - collectRoots: function() { │ │ │ │ │ - var layer; │ │ │ │ │ - // walk through all map layers, because we want to keep the order │ │ │ │ │ - for (var i = 0; i < this.map.layers.length; ++i) { │ │ │ │ │ - layer = this.map.layers[i]; │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ - layer.renderer.moveRoot(this.renderer); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: resetRoots │ │ │ │ │ - * Resets the root nodes back into the layers they belong to. │ │ │ │ │ - */ │ │ │ │ │ - resetRoots: function() { │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = 0; i < this.layers.length; ++i) { │ │ │ │ │ - layer = this.layers[i]; │ │ │ │ │ - if (this.renderer && layer.renderer.getRenderLayerId() == this.id) { │ │ │ │ │ - this.renderer.moveRoot(layer.renderer); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleChangeLayer │ │ │ │ │ - * Event handler for the map's changelayer event. We need to rebuild │ │ │ │ │ - * this container's layer dom if order of one of its layers changes. │ │ │ │ │ - * This handler is added with the setMap method, and removed with the │ │ │ │ │ - * removeMap method. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ - */ │ │ │ │ │ - handleChangeLayer: function(evt) { │ │ │ │ │ - var layer = evt.layer; │ │ │ │ │ - if (evt.property == "order" && │ │ │ │ │ - OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ - this.resetRoots(); │ │ │ │ │ - this.collectRoots(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Vector.RootContainer" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/SelectFeature.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Handler/Feature.js │ │ │ │ │ - * @requires OpenLayers/Layer/Vector/RootContainer.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.SelectFeature │ │ │ │ │ - * The SelectFeature control selects vector features from a given layer on │ │ │ │ │ - * click or hover. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.SelectFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * beforefeaturehighlighted - Triggered before a feature is highlighted │ │ │ │ │ - * featurehighlighted - Triggered when a feature is highlighted │ │ │ │ │ - * featureunhighlighted - Triggered when a feature is unhighlighted │ │ │ │ │ - * boxselectionstart - Triggered before box selection starts │ │ │ │ │ - * boxselectionend - Triggered after box selection ends │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: multipleKey │ │ │ │ │ - * {String} An event modifier ('altKey' or 'shiftKey') that temporarily sets │ │ │ │ │ - * the <multiple> property to true. Default is null. │ │ │ │ │ - */ │ │ │ │ │ - multipleKey: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: toggleKey │ │ │ │ │ - * {String} An event modifier ('altKey' or 'shiftKey') that temporarily sets │ │ │ │ │ - * the <toggle> property to true. Default is null. │ │ │ │ │ - */ │ │ │ │ │ - toggleKey: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: multiple │ │ │ │ │ - * {Boolean} Allow selection of multiple geometries. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - multiple: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: clickout │ │ │ │ │ - * {Boolean} Unselect features when clicking outside any feature. │ │ │ │ │ - * Default is true. │ │ │ │ │ - */ │ │ │ │ │ - clickout: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: toggle │ │ │ │ │ - * {Boolean} Unselect a selected feature on click. Default is false. Only │ │ │ │ │ - * has meaning if hover is false. │ │ │ │ │ - */ │ │ │ │ │ - toggle: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: hover │ │ │ │ │ - * {Boolean} Select on mouse over and deselect on mouse out. If true, this │ │ │ │ │ - * ignores clicks and only listens to mouse moves. │ │ │ │ │ - */ │ │ │ │ │ - hover: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: highlightOnly │ │ │ │ │ - * {Boolean} If true do not actually select features (that is place them in │ │ │ │ │ - * the layer's selected features array), just highlight them. This property │ │ │ │ │ - * has no effect if hover is false. Defaults to false. │ │ │ │ │ - */ │ │ │ │ │ - highlightOnly: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: box │ │ │ │ │ - * {Boolean} Allow feature selection by drawing a box. │ │ │ │ │ - */ │ │ │ │ │ - box: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: onBeforeSelect │ │ │ │ │ - * {Function} Optional function to be called before a feature is selected. │ │ │ │ │ - * The function should expect to be called with a feature. │ │ │ │ │ - */ │ │ │ │ │ - onBeforeSelect: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: onSelect │ │ │ │ │ - * {Function} Optional function to be called when a feature is selected. │ │ │ │ │ - * The function should expect to be called with a feature. │ │ │ │ │ - */ │ │ │ │ │ - onSelect: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: onUnselect │ │ │ │ │ - * {Function} Optional function to be called when a feature is unselected. │ │ │ │ │ - * The function should expect to be called with a feature. │ │ │ │ │ - */ │ │ │ │ │ - onUnselect: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: scope │ │ │ │ │ - * {Object} The scope to use with the onBeforeSelect, onSelect, onUnselect │ │ │ │ │ - * callbacks. If null the scope will be this control. │ │ │ │ │ - */ │ │ │ │ │ - scope: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geometryTypes │ │ │ │ │ - * {Array(String)} To restrict selecting to a limited set of geometry types, │ │ │ │ │ - * send a list of strings corresponding to the geometry class names. │ │ │ │ │ - */ │ │ │ │ │ - geometryTypes: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} The vector layer with a common renderer │ │ │ │ │ - * root for all layers this control is configured with (if an array of │ │ │ │ │ - * layers was passed to the constructor), or the vector layer the control │ │ │ │ │ - * was configured with (if a single layer was passed to the constructor). │ │ │ │ │ - */ │ │ │ │ │ - layer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layers │ │ │ │ │ - * {Array(<OpenLayers.Layer.Vector>)} The layers this control will work on, │ │ │ │ │ - * or null if the control was configured with a single layer │ │ │ │ │ - */ │ │ │ │ │ - layers: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: callbacks │ │ │ │ │ - * {Object} The functions that are sent to the handlers.feature for callback │ │ │ │ │ - */ │ │ │ │ │ - callbacks: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: selectStyle │ │ │ │ │ - * {Object} Hash of styles │ │ │ │ │ - */ │ │ │ │ │ - selectStyle: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: renderIntent │ │ │ │ │ - * {String} key used to retrieve the select style from the layer's │ │ │ │ │ - * style map. │ │ │ │ │ - */ │ │ │ │ │ - renderIntent: "select", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: handlers │ │ │ │ │ - * {Object} Object with references to multiple <OpenLayers.Handler> │ │ │ │ │ - * instances. │ │ │ │ │ - */ │ │ │ │ │ - handlers: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.SelectFeature │ │ │ │ │ - * Create a new control for selecting features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layers - {<OpenLayers.Layer.Vector>}, or an array of vector layers. The │ │ │ │ │ - * layer(s) this control will select features from. │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(layers, options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - │ │ │ │ │ - if (this.scope === null) { │ │ │ │ │ - this.scope = this; │ │ │ │ │ - } │ │ │ │ │ - this.initLayer(layers); │ │ │ │ │ - var callbacks = { │ │ │ │ │ - click: this.clickFeature, │ │ │ │ │ - clickout: this.clickoutFeature │ │ │ │ │ - }; │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - callbacks.over = this.overFeature; │ │ │ │ │ - callbacks.out = this.outFeature; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.callbacks = OpenLayers.Util.extend(callbacks, this.callbacks); │ │ │ │ │ - this.handlers = { │ │ │ │ │ - feature: new OpenLayers.Handler.Feature( │ │ │ │ │ - this, this.layer, this.callbacks, { │ │ │ │ │ - geometryTypes: this.geometryTypes │ │ │ │ │ - } │ │ │ │ │ - ) │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - if (this.box) { │ │ │ │ │ - this.handlers.box = new OpenLayers.Handler.Box( │ │ │ │ │ - this, { │ │ │ │ │ - done: this.selectBox │ │ │ │ │ - }, { │ │ │ │ │ - boxDivClassName: "olHandlerBoxSelectFeature" │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: initLayer │ │ │ │ │ - * Assign the layer property. If layers is an array, we need to use │ │ │ │ │ - * a RootContainer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layers - {<OpenLayers.Layer.Vector>}, or an array of vector layers. │ │ │ │ │ - */ │ │ │ │ │ - initLayer: function(layers) { │ │ │ │ │ - if (OpenLayers.Util.isArray(layers)) { │ │ │ │ │ - this.layers = layers; │ │ │ │ │ - this.layer = new OpenLayers.Layer.Vector.RootContainer( │ │ │ │ │ - this.id + "_container", { │ │ │ │ │ - layers: layers │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - this.layer = layers; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.active && this.layers) { │ │ │ │ │ - this.map.removeLayer(this.layer); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - if (this.layers) { │ │ │ │ │ - this.layer.destroy(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Activates the control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The control was effectively activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - if (this.layers) { │ │ │ │ │ - this.map.addLayer(this.layer); │ │ │ │ │ - } │ │ │ │ │ - this.handlers.feature.activate(); │ │ │ │ │ - if (this.box && this.handlers.box) { │ │ │ │ │ - this.handlers.box.activate(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Control.prototype.activate.apply( │ │ │ │ │ - this, arguments │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Deactivates the control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The control was effectively deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.handlers.feature.deactivate(); │ │ │ │ │ - if (this.handlers.box) { │ │ │ │ │ - this.handlers.box.deactivate(); │ │ │ │ │ - } │ │ │ │ │ - if (this.layers) { │ │ │ │ │ - this.map.removeLayer(this.layer); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Control.prototype.deactivate.apply( │ │ │ │ │ - this, arguments │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: unselectAll │ │ │ │ │ - * Unselect all selected features. To unselect all except for a single │ │ │ │ │ - * feature, set the options.except property to the feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional configuration object. │ │ │ │ │ - */ │ │ │ │ │ - unselectAll: function(options) { │ │ │ │ │ - // we'll want an option to supress notification here │ │ │ │ │ - var layers = this.layers || [this.layer], │ │ │ │ │ - layer, feature, l, numExcept; │ │ │ │ │ - for (l = 0; l < layers.length; ++l) { │ │ │ │ │ - layer = layers[l]; │ │ │ │ │ - numExcept = 0; │ │ │ │ │ - //layer.selectedFeatures is null when layer is destroyed and │ │ │ │ │ - //one of it's preremovelayer listener calls setLayer │ │ │ │ │ - //with another layer on this control │ │ │ │ │ - if (layer.selectedFeatures != null) { │ │ │ │ │ - while (layer.selectedFeatures.length > numExcept) { │ │ │ │ │ - feature = layer.selectedFeatures[numExcept]; │ │ │ │ │ - if (!options || options.except != feature) { │ │ │ │ │ - this.unselect(feature); │ │ │ │ │ - } else { │ │ │ │ │ - ++numExcept; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clickFeature │ │ │ │ │ - * Called on click in a feature │ │ │ │ │ - * Only responds if this.hover is false. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - clickFeature: function(feature) { │ │ │ │ │ - if (!this.hover) { │ │ │ │ │ - var selected = (OpenLayers.Util.indexOf( │ │ │ │ │ - feature.layer.selectedFeatures, feature) > -1); │ │ │ │ │ - if (selected) { │ │ │ │ │ - if (this.toggleSelect()) { │ │ │ │ │ - this.unselect(feature); │ │ │ │ │ - } else if (!this.multipleSelect()) { │ │ │ │ │ - this.unselectAll({ │ │ │ │ │ - except: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - if (!this.multipleSelect()) { │ │ │ │ │ - this.unselectAll({ │ │ │ │ │ - except: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - this.select(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: multipleSelect │ │ │ │ │ - * Allow for multiple selected features based on <multiple> property and │ │ │ │ │ - * <multipleKey> event modifier. │ │ │ │ │ + * Method: touchstart │ │ │ │ │ + * Handle touchstart events │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Allow for multiple selected features. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - multipleSelect: function() { │ │ │ │ │ - return this.multiple || (this.handlers.feature.evt && │ │ │ │ │ - this.handlers.feature.evt[this.multipleKey]); │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + return this.dragstart(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: toggleSelect │ │ │ │ │ - * Event should toggle the selected state of a feature based on <toggle> │ │ │ │ │ - * property and <toggleKey> event modifier. │ │ │ │ │ + * Method: mousemove │ │ │ │ │ + * Handle mousemove events │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Toggle the selected state of a feature. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - toggleSelect: function() { │ │ │ │ │ - return this.toggle || (this.handlers.feature.evt && │ │ │ │ │ - this.handlers.feature.evt[this.toggleKey]); │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + return this.dragmove(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clickoutFeature │ │ │ │ │ - * Called on click outside a previously clicked (selected) feature. │ │ │ │ │ - * Only responds if this.hover is false. │ │ │ │ │ + * Method: touchmove │ │ │ │ │ + * Handle touchmove events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Vector.Feature>} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - clickoutFeature: function(feature) { │ │ │ │ │ - if (!this.hover && this.clickout) { │ │ │ │ │ - this.unselectAll(); │ │ │ │ │ - } │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + return this.dragmove(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: overFeature │ │ │ │ │ - * Called on over a feature. │ │ │ │ │ - * Only responds if this.hover is true. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * Method: removeTimeout │ │ │ │ │ + * Private. Called by mousemove() to remove the drag timeout. │ │ │ │ │ */ │ │ │ │ │ - overFeature: function(feature) { │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - if (this.highlightOnly) { │ │ │ │ │ - this.highlight(feature); │ │ │ │ │ - } else if (OpenLayers.Util.indexOf( │ │ │ │ │ - layer.selectedFeatures, feature) == -1) { │ │ │ │ │ - this.select(feature); │ │ │ │ │ - } │ │ │ │ │ + removeTimeout: function() { │ │ │ │ │ + this.timeoutId = null; │ │ │ │ │ + // if timeout expires while we're still dragging (mouseup │ │ │ │ │ + // hasn't occurred) then call mousemove to move to the │ │ │ │ │ + // correct position │ │ │ │ │ + if (this.dragging) { │ │ │ │ │ + this.mousemove(this.lastMoveEvt); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: outFeature │ │ │ │ │ - * Called on out of a selected feature. │ │ │ │ │ - * Only responds if this.hover is true. │ │ │ │ │ + * Method: mouseup │ │ │ │ │ + * Handle mouseup events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - outFeature: function(feature) { │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - if (this.highlightOnly) { │ │ │ │ │ - // we do nothing if we're not the last highlighter of the │ │ │ │ │ - // feature │ │ │ │ │ - if (feature._lastHighlighter == this.id) { │ │ │ │ │ - // if another select control had highlighted the feature before │ │ │ │ │ - // we did it ourself then we use that control to highlight the │ │ │ │ │ - // feature as it was before we highlighted it, else we just │ │ │ │ │ - // unhighlight it │ │ │ │ │ - if (feature._prevHighlighter && │ │ │ │ │ - feature._prevHighlighter != this.id) { │ │ │ │ │ - delete feature._lastHighlighter; │ │ │ │ │ - var control = this.map.getControl( │ │ │ │ │ - feature._prevHighlighter); │ │ │ │ │ - if (control) { │ │ │ │ │ - control.highlight(feature); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.unhighlight(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.unselect(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + return this.dragend(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: highlight │ │ │ │ │ - * Redraw feature with the select style. │ │ │ │ │ + * Method: touchend │ │ │ │ │ + * Handle touchend events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - highlight: function(feature) { │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - var cont = this.events.triggerEvent("beforefeaturehighlighted", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - feature._prevHighlighter = feature._lastHighlighter; │ │ │ │ │ - feature._lastHighlighter = this.id; │ │ │ │ │ - var style = this.selectStyle || this.renderIntent; │ │ │ │ │ - layer.drawFeature(feature, style); │ │ │ │ │ - this.events.triggerEvent("featurehighlighted", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + // override evt.xy with last position since touchend does not have │ │ │ │ │ + // any touch position │ │ │ │ │ + evt.xy = this.last; │ │ │ │ │ + return this.dragend(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: unhighlight │ │ │ │ │ - * Redraw feature with the "default" style │ │ │ │ │ + * Method: mouseout │ │ │ │ │ + * Handle mouseout events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - unhighlight: function(feature) { │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - // three cases: │ │ │ │ │ - // 1. there's no other highlighter, in that case _prev is undefined, │ │ │ │ │ - // and we just need to undef _last │ │ │ │ │ - // 2. another control highlighted the feature after we did it, in │ │ │ │ │ - // that case _last references this other control, and we just │ │ │ │ │ - // need to undef _prev │ │ │ │ │ - // 3. another control highlighted the feature before we did it, in │ │ │ │ │ - // that case _prev references this other control, and we need to │ │ │ │ │ - // set _last to _prev and undef _prev │ │ │ │ │ - if (feature._prevHighlighter == undefined) { │ │ │ │ │ - delete feature._lastHighlighter; │ │ │ │ │ - } else if (feature._prevHighlighter == this.id) { │ │ │ │ │ - delete feature._prevHighlighter; │ │ │ │ │ - } else { │ │ │ │ │ - feature._lastHighlighter = feature._prevHighlighter; │ │ │ │ │ - delete feature._prevHighlighter; │ │ │ │ │ + mouseout: function(evt) { │ │ │ │ │ + if (this.started && OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ + if (this.documentDrag === true) { │ │ │ │ │ + this.addDocumentEvents(); │ │ │ │ │ + } else { │ │ │ │ │ + var dragged = (this.start != this.last); │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + OpenLayers.Element.removeClass( │ │ │ │ │ + this.map.viewPortDiv, "olDragDown" │ │ │ │ │ + ); │ │ │ │ │ + this.out(evt); │ │ │ │ │ + this.callback("out", []); │ │ │ │ │ + if (dragged) { │ │ │ │ │ + this.callback("done", [evt.xy]); │ │ │ │ │ + } │ │ │ │ │ + if (document.onselectstart) { │ │ │ │ │ + document.onselectstart = this.oldOnselectstart; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - layer.drawFeature(feature, feature.style || feature.layer.style || │ │ │ │ │ - "default"); │ │ │ │ │ - this.events.triggerEvent("featureunhighlighted", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: select │ │ │ │ │ - * Add feature to the layer's selectedFeature array, render the feature as │ │ │ │ │ - * selected, and call the onSelect function. │ │ │ │ │ + * Method: click │ │ │ │ │ + * The drag handler captures the click event. If something else registers │ │ │ │ │ + * for clicks on the same element, its listener will not be called │ │ │ │ │ + * after a drag. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - select: function(feature) { │ │ │ │ │ - var cont = this.onBeforeSelect.call(this.scope, feature); │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - cont = layer.events.triggerEvent("beforefeatureselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - layer.selectedFeatures.push(feature); │ │ │ │ │ - this.highlight(feature); │ │ │ │ │ - // if the feature handler isn't involved in the feature │ │ │ │ │ - // selection (because the box handler is used or the │ │ │ │ │ - // feature is selected programatically) we fake the │ │ │ │ │ - // feature handler to allow unselecting on click │ │ │ │ │ - if (!this.handlers.feature.lastFeature) { │ │ │ │ │ - this.handlers.feature.lastFeature = layer.selectedFeatures[0]; │ │ │ │ │ - } │ │ │ │ │ - layer.events.triggerEvent("featureselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - this.onSelect.call(this.scope, feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + // let the click event propagate only if the mouse moved │ │ │ │ │ + return (this.start == this.last); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: unselect │ │ │ │ │ - * Remove feature from the layer's selectedFeature array, render the feature as │ │ │ │ │ - * normal, and call the onUnselect function. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activate the handler. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was successfully activated. │ │ │ │ │ */ │ │ │ │ │ - unselect: function(feature) { │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - // Store feature style for restoration later │ │ │ │ │ - this.unhighlight(feature); │ │ │ │ │ - OpenLayers.Util.removeItem(layer.selectedFeatures, feature); │ │ │ │ │ - layer.events.triggerEvent("featureunselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - this.onUnselect.call(this.scope, feature); │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + activated = true; │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: selectBox │ │ │ │ │ - * Callback from the handlers.box set up when <box> selection is true │ │ │ │ │ - * on. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * position - {<OpenLayers.Bounds> || <OpenLayers.Pixel> } │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivate the handler. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ - selectBox: function(position) { │ │ │ │ │ - if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ - var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.left, │ │ │ │ │ - y: position.bottom │ │ │ │ │ - }); │ │ │ │ │ - var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.right, │ │ │ │ │ - y: position.top │ │ │ │ │ - }); │ │ │ │ │ - var bounds = new OpenLayers.Bounds( │ │ │ │ │ - minXY.lon, minXY.lat, maxXY.lon, maxXY.lat │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + deactivated = true; │ │ │ │ │ + OpenLayers.Element.removeClass( │ │ │ │ │ + this.map.viewPortDiv, "olDragDown" │ │ │ │ │ ); │ │ │ │ │ - │ │ │ │ │ - // if multiple is false, first deselect currently selected features │ │ │ │ │ - if (!this.multipleSelect()) { │ │ │ │ │ - this.unselectAll(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // because we're using a box, we consider we want multiple selection │ │ │ │ │ - var prevMultiple = this.multiple; │ │ │ │ │ - this.multiple = true; │ │ │ │ │ - var layers = this.layers || [this.layer]; │ │ │ │ │ - this.events.triggerEvent("boxselectionstart", { │ │ │ │ │ - layers: layers │ │ │ │ │ - }); │ │ │ │ │ - var layer; │ │ │ │ │ - for (var l = 0; l < layers.length; ++l) { │ │ │ │ │ - layer = layers[l]; │ │ │ │ │ - for (var i = 0, len = layer.features.length; i < len; ++i) { │ │ │ │ │ - var feature = layer.features[i]; │ │ │ │ │ - // check if the feature is displayed │ │ │ │ │ - if (!feature.getVisibility()) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.geometryTypes == null || OpenLayers.Util.indexOf( │ │ │ │ │ - this.geometryTypes, feature.geometry.CLASS_NAME) > -1) { │ │ │ │ │ - if (bounds.toGeometry().intersects(feature.geometry)) { │ │ │ │ │ - if (OpenLayers.Util.indexOf(layer.selectedFeatures, feature) == -1) { │ │ │ │ │ - this.select(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.multiple = prevMultiple; │ │ │ │ │ - this.events.triggerEvent("boxselectionend", { │ │ │ │ │ - layers: layers │ │ │ │ │ - }); │ │ │ │ │ } │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the control. │ │ │ │ │ + /** │ │ │ │ │ + * Method: adjustXY │ │ │ │ │ + * Converts event coordinates that are relative to the document body to │ │ │ │ │ + * ones that are relative to the map viewport. The latter is the default in │ │ │ │ │ + * OpenLayers. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * evt - {Object} │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.handlers.feature.setMap(map); │ │ │ │ │ - if (this.box) { │ │ │ │ │ - this.handlers.box.setMap(map); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + adjustXY: function(evt) { │ │ │ │ │ + var pos = OpenLayers.Util.pagePosition(this.map.viewPortDiv); │ │ │ │ │ + evt.xy.x -= pos[0]; │ │ │ │ │ + evt.xy.y -= pos[1]; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setLayer │ │ │ │ │ - * Attach a new layer to the control, overriding any existing layers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layers - Array of {<OpenLayers.Layer.Vector>} or a single │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} │ │ │ │ │ + * Method: addDocumentEvents │ │ │ │ │ + * Start observing document events when documentDrag is true and the mouse │ │ │ │ │ + * cursor leaves the map viewport while dragging. │ │ │ │ │ */ │ │ │ │ │ - setLayer: function(layers) { │ │ │ │ │ - var isActive = this.active; │ │ │ │ │ - this.unselectAll(); │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - if (this.layers) { │ │ │ │ │ - this.layer.destroy(); │ │ │ │ │ - this.layers = null; │ │ │ │ │ - } │ │ │ │ │ - this.initLayer(layers); │ │ │ │ │ - this.handlers.feature.layer = this.layer; │ │ │ │ │ - if (isActive) { │ │ │ │ │ - this.activate(); │ │ │ │ │ - } │ │ │ │ │ + addDocumentEvents: function() { │ │ │ │ │ + OpenLayers.Element.addClass(document.body, "olDragDown"); │ │ │ │ │ + this.documentEvents = true; │ │ │ │ │ + OpenLayers.Event.observe(document, "mousemove", this._docMove); │ │ │ │ │ + OpenLayers.Event.observe(document, "mouseup", this._docUp); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.SelectFeature" │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeDocumentEvents │ │ │ │ │ + * Stops observing document events when documentDrag is true and the mouse │ │ │ │ │ + * cursor re-enters the map viewport while dragging. │ │ │ │ │ + */ │ │ │ │ │ + removeDocumentEvents: function() { │ │ │ │ │ + OpenLayers.Element.removeClass(document.body, "olDragDown"); │ │ │ │ │ + this.documentEvents = false; │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mousemove", this._docMove); │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mouseup", this._docUp); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Drag" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Control/DragPan.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ @@ -33456,8824 +36273,6007 @@ │ │ │ │ │ defaultDblClick: function(evt) { │ │ │ │ │ this.map.zoomTo(this.map.zoom + 1, evt.xy); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.TouchNavigation" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Renderer/Canvas.js │ │ │ │ │ + OpenLayers/Handler/Keyboard.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Renderer.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Renderer.Canvas │ │ │ │ │ - * A renderer based on the 2D 'canvas' drawing element. │ │ │ │ │ + * Class: OpenLayers.handler.Keyboard │ │ │ │ │ + * A handler for keyboard events. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Handler.Keyboard> constructor. │ │ │ │ │ * │ │ │ │ │ - * Inherits: │ │ │ │ │ - * - <OpenLayers.Renderer> │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Renderer.Canvas = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: hitDetection │ │ │ │ │ - * {Boolean} Allow for hit detection of features. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - hitDetection: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: hitOverflow │ │ │ │ │ - * {Number} The method for converting feature identifiers to color values │ │ │ │ │ - * supports 16777215 sequential values. Two features cannot be │ │ │ │ │ - * predictably detected if their identifiers differ by more than this │ │ │ │ │ - * value. The hitOverflow allows for bigger numbers (but the │ │ │ │ │ - * difference in values is still limited). │ │ │ │ │ - */ │ │ │ │ │ - hitOverflow: 0, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: canvas │ │ │ │ │ - * {Canvas} The canvas context object. │ │ │ │ │ - */ │ │ │ │ │ - canvas: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Object} Internal object of feature/style pairs for use in redrawing the layer. │ │ │ │ │ - */ │ │ │ │ │ - features: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: pendingRedraw │ │ │ │ │ - * {Boolean} The renderer needs a redraw call to render features added while │ │ │ │ │ - * the renderer was locked. │ │ │ │ │ - */ │ │ │ │ │ - pendingRedraw: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: cachedSymbolBounds │ │ │ │ │ - * {Object} Internal cache of calculated symbol extents. │ │ │ │ │ - */ │ │ │ │ │ - cachedSymbolBounds: {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Renderer.Canvas │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * containerID - {<String>} │ │ │ │ │ - * options - {Object} Optional properties to be set on the renderer. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(containerID, options) { │ │ │ │ │ - OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.root = document.createElement("canvas"); │ │ │ │ │ - this.container.appendChild(this.root); │ │ │ │ │ - this.canvas = this.root.getContext("2d"); │ │ │ │ │ - this.features = {}; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitCanvas = document.createElement("canvas"); │ │ │ │ │ - this.hitContext = this.hitCanvas.getContext("2d"); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +OpenLayers.Handler.Keyboard = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setExtent │ │ │ │ │ - * Set the visible part of the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} │ │ │ │ │ - * resolutionChanged - {Boolean} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ - * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ - * False otherwise. │ │ │ │ │ - */ │ │ │ │ │ - setExtent: function() { │ │ │ │ │ - OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - // always redraw features │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ + /* http://www.quirksmode.org/js/keys.html explains key x-browser │ │ │ │ │ + key handling quirks in pretty nice detail */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: eraseGeometry │ │ │ │ │ - * Erase a geometry from the renderer. Because the Canvas renderer has │ │ │ │ │ - * 'memory' of the features that it has drawn, we have to remove the │ │ │ │ │ - * feature so it doesn't redraw. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - */ │ │ │ │ │ - eraseGeometry: function(geometry, featureId) { │ │ │ │ │ - this.eraseFeatures(this.features[featureId][0]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: supported │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the browser supports the renderer class │ │ │ │ │ - */ │ │ │ │ │ - supported: function() { │ │ │ │ │ - return OpenLayers.CANVAS_SUPPORTED; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setSize │ │ │ │ │ - * Sets the size of the drawing surface. │ │ │ │ │ - * │ │ │ │ │ - * Once the size is updated, redraw the canvas. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>} │ │ │ │ │ - */ │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - this.size = size.clone(); │ │ │ │ │ - var root = this.root; │ │ │ │ │ - root.style.width = size.w + "px"; │ │ │ │ │ - root.style.height = size.h + "px"; │ │ │ │ │ - root.width = size.w; │ │ │ │ │ - root.height = size.h; │ │ │ │ │ - this.resolution = null; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - var hitCanvas = this.hitCanvas; │ │ │ │ │ - hitCanvas.style.width = size.w + "px"; │ │ │ │ │ - hitCanvas.style.height = size.h + "px"; │ │ │ │ │ - hitCanvas.width = size.w; │ │ │ │ │ - hitCanvas.height = size.h; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawFeature │ │ │ │ │ - * Draw the feature. Stores the feature in the features list, │ │ │ │ │ - * then redraws the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * style - {<Object>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The feature has been drawn completely. If the feature has no │ │ │ │ │ - * geometry, undefined will be returned. If the feature is not rendered │ │ │ │ │ - * for other reasons, false will be returned. │ │ │ │ │ + * Constant: KEY_EVENTS │ │ │ │ │ + * keydown, keypress, keyup │ │ │ │ │ */ │ │ │ │ │ - drawFeature: function(feature, style) { │ │ │ │ │ - var rendered; │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - style = this.applyDefaultSymbolizer(style || feature.style); │ │ │ │ │ - // don't render if display none or feature outside extent │ │ │ │ │ - var bounds = feature.geometry.getBounds(); │ │ │ │ │ - │ │ │ │ │ - var worldBounds; │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - worldBounds = this.map.getMaxExtent(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var intersects = bounds && bounds.intersectsBounds(this.extent, { │ │ │ │ │ - worldBounds: worldBounds │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - rendered = (style.display !== "none") && !!bounds && intersects; │ │ │ │ │ - if (rendered) { │ │ │ │ │ - // keep track of what we have rendered for redraw │ │ │ │ │ - this.features[feature.id] = [feature, style]; │ │ │ │ │ - } else { │ │ │ │ │ - // remove from features tracked for redraw │ │ │ │ │ - delete(this.features[feature.id]); │ │ │ │ │ - } │ │ │ │ │ - this.pendingRedraw = true; │ │ │ │ │ - } │ │ │ │ │ - if (this.pendingRedraw && !this.locked) { │ │ │ │ │ - this.redraw(); │ │ │ │ │ - this.pendingRedraw = false; │ │ │ │ │ - } │ │ │ │ │ - return rendered; │ │ │ │ │ - }, │ │ │ │ │ + KEY_EVENTS: ["keydown", "keyup"], │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawGeometry │ │ │ │ │ - * Used when looping (in redraw) over the features; draws │ │ │ │ │ - * the canvas. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ + * Property: eventListener │ │ │ │ │ + * {Function} │ │ │ │ │ */ │ │ │ │ │ - drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ - var className = geometry.CLASS_NAME; │ │ │ │ │ - if ((className == "OpenLayers.Geometry.Collection") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiPolygon")) { │ │ │ │ │ - for (var i = 0; i < geometry.components.length; i++) { │ │ │ │ │ - this.drawGeometry(geometry.components[i], style, featureId); │ │ │ │ │ - } │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - this.drawPoint(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - this.drawLineString(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - this.drawPolygon(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + eventListener: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawExternalGraphic │ │ │ │ │ - * Called to draw External graphics. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * Property: observeElement │ │ │ │ │ + * {DOMElement|String} The DOM element on which we listen for │ │ │ │ │ + * key events. Default to the document. │ │ │ │ │ */ │ │ │ │ │ - drawExternalGraphic: function(geometry, style, featureId) { │ │ │ │ │ - var img = new Image(); │ │ │ │ │ - │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - img.title = title; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ - style.graphicXOffset : -(0.5 * width); │ │ │ │ │ - var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ - style.graphicYOffset : -(0.5 * height); │ │ │ │ │ - │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ - │ │ │ │ │ - var onLoad = function() { │ │ │ │ │ - if (!this.features[featureId]) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ - if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ - var x = (p0 + xOffset) | 0; │ │ │ │ │ - var y = (p1 + yOffset) | 0; │ │ │ │ │ - var canvas = this.canvas; │ │ │ │ │ - canvas.globalAlpha = opacity; │ │ │ │ │ - var factor = OpenLayers.Renderer.Canvas.drawImageScaleFactor || │ │ │ │ │ - (OpenLayers.Renderer.Canvas.drawImageScaleFactor = │ │ │ │ │ - /android 2.1/.test(navigator.userAgent.toLowerCase()) ? │ │ │ │ │ - // 320 is the screen width of the G1 phone, for │ │ │ │ │ - // which drawImage works out of the box. │ │ │ │ │ - 320 / window.screen.width : 1 │ │ │ │ │ - ); │ │ │ │ │ - canvas.drawImage( │ │ │ │ │ - img, x * factor, y * factor, width * factor, height * factor │ │ │ │ │ - ); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId); │ │ │ │ │ - this.hitContext.fillRect(x, y, width, height); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - img.onload = OpenLayers.Function.bind(onLoad, this); │ │ │ │ │ - img.src = style.externalGraphic; │ │ │ │ │ - }, │ │ │ │ │ + observeElement: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawNamedSymbol │ │ │ │ │ - * Called to draw Well Known Graphic Symbol Name. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ + * Constructor: OpenLayers.Handler.Keyboard │ │ │ │ │ + * Returns a new keyboard handler. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - */ │ │ │ │ │ - drawNamedSymbol: function(geometry, style, featureId) { │ │ │ │ │ - var x, y, cx, cy, i, symbolBounds, scaling, angle; │ │ │ │ │ - var unscaledStrokeWidth; │ │ │ │ │ - var deg2rad = Math.PI / 180.0; │ │ │ │ │ - │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[style.graphicName]; │ │ │ │ │ - │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(style.graphicName + ' is not a valid symbol name'); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (!symbol.length || symbol.length < 2) return; │ │ │ │ │ - │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ - │ │ │ │ │ - if (isNaN(p0) || isNaN(p1)) return; │ │ │ │ │ - │ │ │ │ │ - // Use rounded line caps │ │ │ │ │ - this.canvas.lineCap = "round"; │ │ │ │ │ - this.canvas.lineJoin = "round"; │ │ │ │ │ - │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.lineCap = "round"; │ │ │ │ │ - this.hitContext.lineJoin = "round"; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // Scale and rotate symbols, using precalculated bounds whenever possible. │ │ │ │ │ - if (style.graphicName in this.cachedSymbolBounds) { │ │ │ │ │ - symbolBounds = this.cachedSymbolBounds[style.graphicName]; │ │ │ │ │ - } else { │ │ │ │ │ - symbolBounds = new OpenLayers.Bounds(); │ │ │ │ │ - for (i = 0; i < symbol.length; i += 2) { │ │ │ │ │ - symbolBounds.extend(new OpenLayers.LonLat(symbol[i], symbol[i + 1])); │ │ │ │ │ - } │ │ │ │ │ - this.cachedSymbolBounds[style.graphicName] = symbolBounds; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // Push symbol scaling, translation and rotation onto the transformation stack in reverse order. │ │ │ │ │ - // Don't forget to apply all canvas transformations to the hitContext canvas as well(!) │ │ │ │ │ - this.canvas.save(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.save(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // Step 3: place symbol at the desired location │ │ │ │ │ - this.canvas.translate(p0, p1); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.translate(p0, p1); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // Step 2a. rotate the symbol if necessary │ │ │ │ │ - angle = deg2rad * style.rotation; // will be NaN when style.rotation is undefined. │ │ │ │ │ - if (!isNaN(angle)) { │ │ │ │ │ - this.canvas.rotate(angle); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.rotate(angle); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // // Step 2: scale symbol such that pointRadius equals half the maximum symbol dimension. │ │ │ │ │ - scaling = 2.0 * style.pointRadius / Math.max(symbolBounds.getWidth(), symbolBounds.getHeight()); │ │ │ │ │ - this.canvas.scale(scaling, scaling); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.scale(scaling, scaling); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // Step 1: center the symbol at the origin │ │ │ │ │ - cx = symbolBounds.getCenterLonLat().lon; │ │ │ │ │ - cy = symbolBounds.getCenterLonLat().lat; │ │ │ │ │ - this.canvas.translate(-cx, -cy); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.translate(-cx, -cy); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // Don't forget to scale stroke widths, because they are affected by canvas scale transformations as well(!) │ │ │ │ │ - // Alternative: scale symbol coordinates manually, so stroke width scaling is not needed anymore. │ │ │ │ │ - unscaledStrokeWidth = style.strokeWidth; │ │ │ │ │ - style.strokeWidth = unscaledStrokeWidth / scaling; │ │ │ │ │ - │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.canvas.lineTo(x, y); │ │ │ │ │ - } │ │ │ │ │ - this.canvas.closePath(); │ │ │ │ │ - this.canvas.fill(); │ │ │ │ │ - │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.hitContext.lineTo(x, y); │ │ │ │ │ - } │ │ │ │ │ - this.hitContext.closePath(); │ │ │ │ │ - this.hitContext.fill(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.canvas.lineTo(x, y); │ │ │ │ │ - } │ │ │ │ │ - this.canvas.closePath(); │ │ │ │ │ - this.canvas.stroke(); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style, scaling); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.hitContext.moveTo(x, y); │ │ │ │ │ - this.hitContext.lineTo(x, y); │ │ │ │ │ - } │ │ │ │ │ - this.hitContext.closePath(); │ │ │ │ │ - this.hitContext.stroke(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - style.strokeWidth = unscaledStrokeWidth; │ │ │ │ │ - this.canvas.restore(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.restore(); │ │ │ │ │ - } │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setCanvasStyle │ │ │ │ │ - * Prepare the canvas for drawing by setting various global settings. │ │ │ │ │ - * │ │ │ │ │ * Parameters: │ │ │ │ │ - * type - {String} one of 'stroke', 'fill', or 'reset' │ │ │ │ │ - * style - {Object} Symbolizer hash │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ + * this handler. If a handler is being used without a control, the │ │ │ │ │ + * handlers setMap method must be overridden to deal properly with │ │ │ │ │ + * the map. │ │ │ │ │ + * callbacks - {Object} An object containing a single function to be │ │ │ │ │ + * called when the drag operation is finished. The callback should │ │ │ │ │ + * expect to recieve a single argument, the pixel location of the event. │ │ │ │ │ + * Callbacks for 'keydown', 'keypress', and 'keyup' are supported. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * handler. │ │ │ │ │ */ │ │ │ │ │ - setCanvasStyle: function(type, style) { │ │ │ │ │ - if (type === "fill") { │ │ │ │ │ - this.canvas.globalAlpha = style['fillOpacity']; │ │ │ │ │ - this.canvas.fillStyle = style['fillColor']; │ │ │ │ │ - } else if (type === "stroke") { │ │ │ │ │ - this.canvas.globalAlpha = style['strokeOpacity']; │ │ │ │ │ - this.canvas.strokeStyle = style['strokeColor']; │ │ │ │ │ - this.canvas.lineWidth = style['strokeWidth']; │ │ │ │ │ - } else { │ │ │ │ │ - this.canvas.globalAlpha = 0; │ │ │ │ │ - this.canvas.lineWidth = 1; │ │ │ │ │ - } │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + // cache the bound event listener method so it can be unobserved later │ │ │ │ │ + this.eventListener = OpenLayers.Function.bindAsEventListener( │ │ │ │ │ + this.handleKeyEvent, this │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: featureIdToHex │ │ │ │ │ - * Convert a feature ID string into an RGB hex string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureId - {String} Feature id │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} RGB hex string. │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ - featureIdToHex: function(featureId) { │ │ │ │ │ - var id = Number(featureId.split("_").pop()) + 1; // zero for no feature │ │ │ │ │ - if (id >= 16777216) { │ │ │ │ │ - this.hitOverflow = id - 16777215; │ │ │ │ │ - id = id % 16777216 + 1; │ │ │ │ │ - } │ │ │ │ │ - var hex = "000000" + id.toString(16); │ │ │ │ │ - var len = hex.length; │ │ │ │ │ - hex = "#" + hex.substring(len - 6, len); │ │ │ │ │ - return hex; │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.eventListener = null; │ │ │ │ │ + OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setHitContextStyle │ │ │ │ │ - * Prepare the hit canvas for drawing by setting various global settings. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * type - {String} one of 'stroke', 'fill', or 'reset' │ │ │ │ │ - * featureId - {String} The feature id. │ │ │ │ │ - * symbolizer - {<OpenLayers.Symbolizer>} The symbolizer. │ │ │ │ │ + * Method: activate │ │ │ │ │ */ │ │ │ │ │ - setHitContextStyle: function(type, featureId, symbolizer, strokeScaling) { │ │ │ │ │ - var hex = this.featureIdToHex(featureId); │ │ │ │ │ - if (type == "fill") { │ │ │ │ │ - this.hitContext.globalAlpha = 1.0; │ │ │ │ │ - this.hitContext.fillStyle = hex; │ │ │ │ │ - } else if (type == "stroke") { │ │ │ │ │ - this.hitContext.globalAlpha = 1.0; │ │ │ │ │ - this.hitContext.strokeStyle = hex; │ │ │ │ │ - // bump up stroke width to deal with antialiasing. If strokeScaling is defined, we're rendering a symbol │ │ │ │ │ - // on a transformed canvas, so the antialias width bump has to scale as well. │ │ │ │ │ - if (typeof strokeScaling === "undefined") { │ │ │ │ │ - this.hitContext.lineWidth = symbolizer.strokeWidth + 2; │ │ │ │ │ - } else { │ │ │ │ │ - if (!isNaN(strokeScaling)) { │ │ │ │ │ - this.hitContext.lineWidth = symbolizer.strokeWidth + 2.0 / strokeScaling; │ │ │ │ │ - } │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.observeElement = this.observeElement || document; │ │ │ │ │ + for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ + OpenLayers.Event.observe( │ │ │ │ │ + this.observeElement, this.KEY_EVENTS[i], this.eventListener); │ │ │ │ │ } │ │ │ │ │ + return true; │ │ │ │ │ } else { │ │ │ │ │ - this.hitContext.globalAlpha = 0; │ │ │ │ │ - this.hitContext.lineWidth = 1; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawPoint │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - */ │ │ │ │ │ - drawPoint: function(geometry, style, featureId) { │ │ │ │ │ - if (style.graphic !== false) { │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - this.drawExternalGraphic(geometry, style, featureId); │ │ │ │ │ - } else if (style.graphicName && (style.graphicName != "circle")) { │ │ │ │ │ - this.drawNamedSymbol(geometry, style, featureId); │ │ │ │ │ - } else { │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ - if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ - var twoPi = Math.PI * 2; │ │ │ │ │ - var radius = style.pointRadius; │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.canvas.fill(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.hitContext.fill(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.canvas.stroke(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.hitContext.stroke(); │ │ │ │ │ - } │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawLineString │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - */ │ │ │ │ │ - drawLineString: function(geometry, style, featureId) { │ │ │ │ │ - style = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - fill: false │ │ │ │ │ - }, style); │ │ │ │ │ - this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawLinearRing │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - */ │ │ │ │ │ - drawLinearRing: function(geometry, style, featureId) { │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.renderPath(this.canvas, geometry, style, featureId, "fill"); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.renderPath(this.hitContext, geometry, style, featureId, "fill"); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.renderPath(this.canvas, geometry, style, featureId, "stroke"); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ - this.renderPath(this.hitContext, geometry, style, featureId, "stroke"); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: renderPath │ │ │ │ │ - * Render a path with stroke and optional fill. │ │ │ │ │ - */ │ │ │ │ │ - renderPath: function(context, geometry, style, featureId, type) { │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - context.beginPath(); │ │ │ │ │ - var start = this.getLocalXY(components[0]); │ │ │ │ │ - var x = start[0]; │ │ │ │ │ - var y = start[1]; │ │ │ │ │ - if (!isNaN(x) && !isNaN(y)) { │ │ │ │ │ - context.moveTo(start[0], start[1]); │ │ │ │ │ - for (var i = 1; i < len; ++i) { │ │ │ │ │ - var pt = this.getLocalXY(components[i]); │ │ │ │ │ - context.lineTo(pt[0], pt[1]); │ │ │ │ │ - } │ │ │ │ │ - if (type === "fill") { │ │ │ │ │ - context.fill(); │ │ │ │ │ - } else { │ │ │ │ │ - context.stroke(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawPolygon │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - */ │ │ │ │ │ - drawPolygon: function(geometry, style, featureId) { │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - this.drawLinearRing(components[0], style, featureId); │ │ │ │ │ - // erase inner rings │ │ │ │ │ - for (var i = 1; i < len; ++i) { │ │ │ │ │ - /** │ │ │ │ │ - * Note that this is overly agressive. Here we punch holes through │ │ │ │ │ - * all previously rendered features on the same canvas. A better │ │ │ │ │ - * solution for polygons with interior rings would be to draw the │ │ │ │ │ - * polygon on a sketch canvas first. We could erase all holes │ │ │ │ │ - * there and then copy the drawing to the layer canvas. │ │ │ │ │ - * TODO: http://trac.osgeo.org/openlayers/ticket/3130 │ │ │ │ │ - */ │ │ │ │ │ - this.canvas.globalCompositeOperation = "destination-out"; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.globalCompositeOperation = "destination-out"; │ │ │ │ │ - } │ │ │ │ │ - this.drawLinearRing( │ │ │ │ │ - components[i], │ │ │ │ │ - OpenLayers.Util.applyDefaults({ │ │ │ │ │ - stroke: false, │ │ │ │ │ - fillOpacity: 1.0 │ │ │ │ │ - }, style), │ │ │ │ │ - featureId │ │ │ │ │ - ); │ │ │ │ │ - this.canvas.globalCompositeOperation = "source-over"; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.globalCompositeOperation = "source-over"; │ │ │ │ │ - } │ │ │ │ │ - this.drawLinearRing( │ │ │ │ │ - components[i], │ │ │ │ │ - OpenLayers.Util.applyDefaults({ │ │ │ │ │ - fill: false │ │ │ │ │ - }, style), │ │ │ │ │ - featureId │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawText │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * location - {<OpenLayers.Point>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - */ │ │ │ │ │ - drawText: function(location, style) { │ │ │ │ │ - var pt = this.getLocalXY(location); │ │ │ │ │ - │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ - this.canvas.fillStyle = style.fontColor; │ │ │ │ │ - this.canvas.globalAlpha = style.fontOpacity || 1.0; │ │ │ │ │ - var fontStyle = [style.fontStyle ? style.fontStyle : "normal", │ │ │ │ │ - "normal", // "font-variant" not supported │ │ │ │ │ - style.fontWeight ? style.fontWeight : "normal", │ │ │ │ │ - style.fontSize ? style.fontSize : "1em", │ │ │ │ │ - style.fontFamily ? style.fontFamily : "sans-serif" │ │ │ │ │ - ].join(" "); │ │ │ │ │ - var labelRows = style.label.split('\n'); │ │ │ │ │ - var numRows = labelRows.length; │ │ │ │ │ - if (this.canvas.fillText) { │ │ │ │ │ - // HTML5 │ │ │ │ │ - this.canvas.font = fontStyle; │ │ │ │ │ - this.canvas.textAlign = │ │ │ │ │ - OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[0]] || │ │ │ │ │ - "center"; │ │ │ │ │ - this.canvas.textBaseline = │ │ │ │ │ - OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[1]] || │ │ │ │ │ - "middle"; │ │ │ │ │ - var vfactor = │ │ │ │ │ - OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5; │ │ │ │ │ - } │ │ │ │ │ - var lineHeight = │ │ │ │ │ - this.canvas.measureText('Mg').height || │ │ │ │ │ - this.canvas.measureText('xx').width; │ │ │ │ │ - pt[1] += lineHeight * vfactor * (numRows - 1); │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - if (style.labelOutlineWidth) { │ │ │ │ │ - this.canvas.save(); │ │ │ │ │ - this.canvas.globalAlpha = style.labelOutlineOpacity || style.fontOpacity || 1.0; │ │ │ │ │ - this.canvas.strokeStyle = style.labelOutlineColor; │ │ │ │ │ - this.canvas.lineWidth = style.labelOutlineWidth; │ │ │ │ │ - this.canvas.strokeText(labelRows[i], pt[0], pt[1] + (lineHeight * i) + 1); │ │ │ │ │ - this.canvas.restore(); │ │ │ │ │ - } │ │ │ │ │ - this.canvas.fillText(labelRows[i], pt[0], pt[1] + (lineHeight * i)); │ │ │ │ │ - } │ │ │ │ │ - } else if (this.canvas.mozDrawText) { │ │ │ │ │ - // Mozilla pre-Gecko1.9.1 (<FF3.1) │ │ │ │ │ - this.canvas.mozTextStyle = fontStyle; │ │ │ │ │ - // No built-in text alignment, so we measure and adjust the position │ │ │ │ │ - var hfactor = │ │ │ │ │ - OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[0]]; │ │ │ │ │ - if (hfactor == null) { │ │ │ │ │ - hfactor = -.5; │ │ │ │ │ - } │ │ │ │ │ - var vfactor = │ │ │ │ │ - OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5; │ │ │ │ │ - } │ │ │ │ │ - var lineHeight = this.canvas.mozMeasureText('xx'); │ │ │ │ │ - pt[1] += lineHeight * (1 + (vfactor * numRows)); │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - var x = pt[0] + (hfactor * this.canvas.mozMeasureText(labelRows[i])); │ │ │ │ │ - var y = pt[1] + (i * lineHeight); │ │ │ │ │ - this.canvas.translate(x, y); │ │ │ │ │ - this.canvas.mozDrawText(labelRows[i]); │ │ │ │ │ - this.canvas.translate(-x, -y); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getLocalXY │ │ │ │ │ - * transform geographic xy into pixel xy │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ - */ │ │ │ │ │ - getLocalXY: function(point) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var extent = this.extent; │ │ │ │ │ - var x = ((point.x - this.featureDx) / resolution + (-extent.left / resolution)); │ │ │ │ │ - var y = ((extent.top / resolution) - point.y / resolution); │ │ │ │ │ - return [x, y]; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clear │ │ │ │ │ - * Clear all vectors from the renderer. │ │ │ │ │ - */ │ │ │ │ │ - clear: function() { │ │ │ │ │ - var height = this.root.height; │ │ │ │ │ - var width = this.root.width; │ │ │ │ │ - this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ - this.features = {}; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.clearRect(0, 0, width, height); │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureIdFromEvent │ │ │ │ │ - * Returns a feature id from an event on the renderer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector} A feature or undefined. This method returns a │ │ │ │ │ - * feature instead of a feature id to avoid an unnecessary lookup on the │ │ │ │ │ - * layer. │ │ │ │ │ + * Method: deactivate │ │ │ │ │ */ │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var featureId, feature; │ │ │ │ │ - │ │ │ │ │ - if (this.hitDetection && this.root.style.display !== "none") { │ │ │ │ │ - // this dragging check should go in the feature handler │ │ │ │ │ - if (!this.map.dragging) { │ │ │ │ │ - var xy = evt.xy; │ │ │ │ │ - var x = xy.x | 0; │ │ │ │ │ - var y = xy.y | 0; │ │ │ │ │ - var data = this.hitContext.getImageData(x, y, 1, 1).data; │ │ │ │ │ - if (data[3] === 255) { // antialiased │ │ │ │ │ - var id = data[2] + (256 * (data[1] + (256 * data[0]))); │ │ │ │ │ - if (id) { │ │ │ │ │ - featureId = "OpenLayers_Feature_Vector_" + (id - 1 + this.hitOverflow); │ │ │ │ │ - try { │ │ │ │ │ - feature = this.features[featureId][0]; │ │ │ │ │ - } catch (err) { │ │ │ │ │ - // Because of antialiasing on the canvas, when the hit location is at a point where the edge of │ │ │ │ │ - // one symbol intersects the interior of another symbol, a wrong hit color (and therefore id) results. │ │ │ │ │ - // todo: set Antialiasing = 'off' on the hitContext as soon as browsers allow it. │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ + OpenLayers.Event.stopObserving( │ │ │ │ │ + this.observeElement, this.KEY_EVENTS[i], this.eventListener); │ │ │ │ │ } │ │ │ │ │ + deactivated = true; │ │ │ │ │ } │ │ │ │ │ - return feature; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: eraseFeatures │ │ │ │ │ - * This is called by the layer to erase features; removes the feature from │ │ │ │ │ - * the list, then redraws the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ - */ │ │ │ │ │ - eraseFeatures: function(features) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0; i < features.length; ++i) { │ │ │ │ │ - delete this.features[features[i].id]; │ │ │ │ │ - } │ │ │ │ │ - this.redraw(); │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: redraw │ │ │ │ │ - * The real 'meat' of the function: any time things have changed, │ │ │ │ │ - * redraw() can be called to loop over all the data and (you guessed │ │ │ │ │ - * it) redraw it. Unlike Elements-based Renderers, we can't interact │ │ │ │ │ - * with things once they're drawn, to remove them, for example, so │ │ │ │ │ - * instead we have to just clear everything and draw from scratch. │ │ │ │ │ + * Method: handleKeyEvent │ │ │ │ │ */ │ │ │ │ │ - redraw: function() { │ │ │ │ │ - if (!this.locked) { │ │ │ │ │ - var height = this.root.height; │ │ │ │ │ - var width = this.root.width; │ │ │ │ │ - this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.clearRect(0, 0, width, height); │ │ │ │ │ - } │ │ │ │ │ - var labelMap = []; │ │ │ │ │ - var feature, geometry, style; │ │ │ │ │ - var worldBounds = (this.map.baseLayer && this.map.baseLayer.wrapDateLine) && this.map.getMaxExtent(); │ │ │ │ │ - for (var id in this.features) { │ │ │ │ │ - if (!this.features.hasOwnProperty(id)) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - feature = this.features[id][0]; │ │ │ │ │ - geometry = feature.geometry; │ │ │ │ │ - this.calculateFeatureDx(geometry.getBounds(), worldBounds); │ │ │ │ │ - style = this.features[id][1]; │ │ │ │ │ - this.drawGeometry(geometry, style, feature.id); │ │ │ │ │ - if (style.label) { │ │ │ │ │ - labelMap.push([feature, style]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var item; │ │ │ │ │ - for (var i = 0, len = labelMap.length; i < len; ++i) { │ │ │ │ │ - item = labelMap[i]; │ │ │ │ │ - this.drawText(item[0].geometry.getCentroid(), item[1]); │ │ │ │ │ - } │ │ │ │ │ + handleKeyEvent: function(evt) { │ │ │ │ │ + if (this.checkModifiers(evt)) { │ │ │ │ │ + this.callback(evt.type, [evt]); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.Canvas" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Keyboard" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.Canvas.LABEL_ALIGN │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.Canvas.LABEL_ALIGN = { │ │ │ │ │ - "l": "left", │ │ │ │ │ - "r": "right", │ │ │ │ │ - "t": "top", │ │ │ │ │ - "b": "bottom" │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.Canvas.LABEL_FACTOR │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.Canvas.LABEL_FACTOR = { │ │ │ │ │ - "l": 0, │ │ │ │ │ - "r": -1, │ │ │ │ │ - "t": 0, │ │ │ │ │ - "b": -1 │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.Canvas.drawImageScaleFactor │ │ │ │ │ - * {Number} Scale factor to apply to the canvas drawImage arguments. This │ │ │ │ │ - * is always 1 except for Android 2.1 devices, to work around │ │ │ │ │ - * http://code.google.com/p/android/issues/detail?id=5141. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.Canvas.drawImageScaleFactor = null; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Renderer/Elements.js │ │ │ │ │ + OpenLayers/Control/ModifyFeature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Renderer.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Handler/Drag.js │ │ │ │ │ + * @requires OpenLayers/Handler/Keyboard.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.ElementsIndexer │ │ │ │ │ - * This class takes care of figuring out which order elements should be │ │ │ │ │ - * placed in the DOM based on given indexing methods. │ │ │ │ │ + * Class: OpenLayers.Control.ModifyFeature │ │ │ │ │ + * Control to modify features. When activated, a click renders the vertices │ │ │ │ │ + * of a feature - these vertices can then be dragged. By default, the │ │ │ │ │ + * delete key will delete the vertex under the mouse. New features are │ │ │ │ │ + * added by dragging "virtual vertices" between vertices. Create a new │ │ │ │ │ + * control with the <OpenLayers.Control.ModifyFeature> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits From: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.ElementsIndexer = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: maxZIndex │ │ │ │ │ - * {Integer} This is the largest-most z-index value for a node │ │ │ │ │ - * contained within the indexer. │ │ │ │ │ - */ │ │ │ │ │ - maxZIndex: null, │ │ │ │ │ +OpenLayers.Control.ModifyFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: order │ │ │ │ │ - * {Array<String>} This is an array of node id's stored in the │ │ │ │ │ - * order that they should show up on screen. Id's higher up in the │ │ │ │ │ - * array (higher array index) represent nodes with higher z-indeces. │ │ │ │ │ + * APIProperty: documentDrag │ │ │ │ │ + * {Boolean} If set to true, dragging vertices will continue even if the │ │ │ │ │ + * mouse cursor leaves the map viewport. Default is false. │ │ │ │ │ */ │ │ │ │ │ - order: null, │ │ │ │ │ + documentDrag: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: indices │ │ │ │ │ - * {Object} This is a hash that maps node ids to their z-index value │ │ │ │ │ - * stored in the indexer. This is done to make finding a nodes z-index │ │ │ │ │ - * value O(1). │ │ │ │ │ + * APIProperty: geometryTypes │ │ │ │ │ + * {Array(String)} To restrict modification to a limited set of geometry │ │ │ │ │ + * types, send a list of strings corresponding to the geometry class │ │ │ │ │ + * names. │ │ │ │ │ */ │ │ │ │ │ - indices: null, │ │ │ │ │ + geometryTypes: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: compare │ │ │ │ │ - * {Function} This is the function used to determine placement of │ │ │ │ │ - * of a new node within the indexer. If null, this defaults to to │ │ │ │ │ - * the Z_ORDER_DRAWING_ORDER comparison method. │ │ │ │ │ + * APIProperty: clickout │ │ │ │ │ + * {Boolean} Unselect features when clicking outside any feature. │ │ │ │ │ + * Default is true. │ │ │ │ │ */ │ │ │ │ │ - compare: null, │ │ │ │ │ + clickout: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: initialize │ │ │ │ │ - * Create a new indexer with │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * yOrdering - {Boolean} Whether to use y-ordering. │ │ │ │ │ + * APIProperty: toggle │ │ │ │ │ + * {Boolean} Unselect a selected feature on click. │ │ │ │ │ + * Default is true. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(yOrdering) { │ │ │ │ │ - │ │ │ │ │ - this.compare = yOrdering ? │ │ │ │ │ - OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_Y_ORDER : │ │ │ │ │ - OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_DRAWING_ORDER; │ │ │ │ │ - │ │ │ │ │ - this.clear(); │ │ │ │ │ - }, │ │ │ │ │ + toggle: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: insert │ │ │ │ │ - * Insert a new node into the indexer. In order to find the correct │ │ │ │ │ - * positioning for the node to be inserted, this method uses a binary │ │ │ │ │ - * search. This makes inserting O(log(n)). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newNode - {DOMElement} The new node to be inserted. │ │ │ │ │ - * │ │ │ │ │ - * Returns │ │ │ │ │ - * {DOMElement} the node before which we should insert our newNode, or │ │ │ │ │ - * null if newNode can just be appended. │ │ │ │ │ + * APIProperty: standalone │ │ │ │ │ + * {Boolean} Set to true to create a control without SelectFeature │ │ │ │ │ + * capabilities. Default is false. If standalone is true, to modify │ │ │ │ │ + * a feature, call the <selectFeature> method with the target feature. │ │ │ │ │ + * Note that you must call the <unselectFeature> method to finish │ │ │ │ │ + * feature modification in standalone mode (before starting to modify │ │ │ │ │ + * another feature). │ │ │ │ │ */ │ │ │ │ │ - insert: function(newNode) { │ │ │ │ │ - // If the node is known to the indexer, remove it so we can │ │ │ │ │ - // recalculate where it should go. │ │ │ │ │ - if (this.exists(newNode)) { │ │ │ │ │ - this.remove(newNode); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var nodeId = newNode.id; │ │ │ │ │ - │ │ │ │ │ - this.determineZIndex(newNode); │ │ │ │ │ - │ │ │ │ │ - var leftIndex = -1; │ │ │ │ │ - var rightIndex = this.order.length; │ │ │ │ │ - var middle; │ │ │ │ │ - │ │ │ │ │ - while (rightIndex - leftIndex > 1) { │ │ │ │ │ - middle = parseInt((leftIndex + rightIndex) / 2); │ │ │ │ │ - │ │ │ │ │ - var placement = this.compare(this, newNode, │ │ │ │ │ - OpenLayers.Util.getElement(this.order[middle])); │ │ │ │ │ - │ │ │ │ │ - if (placement > 0) { │ │ │ │ │ - leftIndex = middle; │ │ │ │ │ - } else { │ │ │ │ │ - rightIndex = middle; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.order.splice(rightIndex, 0, nodeId); │ │ │ │ │ - this.indices[nodeId] = this.getZIndex(newNode); │ │ │ │ │ - │ │ │ │ │ - // If the new node should be before another in the index │ │ │ │ │ - // order, return the node before which we have to insert the new one; │ │ │ │ │ - // else, return null to indicate that the new node can be appended. │ │ │ │ │ - return this.getNextElement(rightIndex); │ │ │ │ │ - }, │ │ │ │ │ + standalone: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: remove │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} The node to be removed. │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} │ │ │ │ │ */ │ │ │ │ │ - remove: function(node) { │ │ │ │ │ - var nodeId = node.id; │ │ │ │ │ - var arrayIndex = OpenLayers.Util.indexOf(this.order, nodeId); │ │ │ │ │ - if (arrayIndex >= 0) { │ │ │ │ │ - // Remove it from the order array, as well as deleting the node │ │ │ │ │ - // from the indeces hash. │ │ │ │ │ - this.order.splice(arrayIndex, 1); │ │ │ │ │ - delete this.indices[nodeId]; │ │ │ │ │ - │ │ │ │ │ - // Reset the maxium z-index based on the last item in the │ │ │ │ │ - // order array. │ │ │ │ │ - if (this.order.length > 0) { │ │ │ │ │ - var lastId = this.order[this.order.length - 1]; │ │ │ │ │ - this.maxZIndex = this.indices[lastId]; │ │ │ │ │ - } else { │ │ │ │ │ - this.maxZIndex = 0; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + layer: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clear │ │ │ │ │ + * Property: feature │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} Feature currently available for modification. │ │ │ │ │ */ │ │ │ │ │ - clear: function() { │ │ │ │ │ - this.order = []; │ │ │ │ │ - this.indices = {}; │ │ │ │ │ - this.maxZIndex = 0; │ │ │ │ │ - }, │ │ │ │ │ + feature: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: exists │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} The node to test for existence. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the node exists in the indexer? │ │ │ │ │ + * Property: vertex │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} Vertex currently being modified. │ │ │ │ │ */ │ │ │ │ │ - exists: function(node) { │ │ │ │ │ - return (this.indices[node.id] != null); │ │ │ │ │ - }, │ │ │ │ │ + vertex: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getZIndex │ │ │ │ │ - * Get the z-index value for the current node from the node data itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} The node whose z-index to get. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} The z-index value for the specified node (from the node │ │ │ │ │ - * data itself). │ │ │ │ │ + * Property: vertices │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} Verticies currently available │ │ │ │ │ + * for dragging. │ │ │ │ │ */ │ │ │ │ │ - getZIndex: function(node) { │ │ │ │ │ - return node._style.graphicZIndex; │ │ │ │ │ - }, │ │ │ │ │ + vertices: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: determineZIndex │ │ │ │ │ - * Determine the z-index for the current node if there isn't one, │ │ │ │ │ - * and set the maximum value if we've found a new maximum. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ + * Property: virtualVertices │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} Virtual vertices in the middle │ │ │ │ │ + * of each edge. │ │ │ │ │ */ │ │ │ │ │ - determineZIndex: function(node) { │ │ │ │ │ - var zIndex = node._style.graphicZIndex; │ │ │ │ │ - │ │ │ │ │ - // Everything must have a zIndex. If none is specified, │ │ │ │ │ - // this means the user *must* (hint: assumption) want this │ │ │ │ │ - // node to succomb to drawing order. To enforce drawing order │ │ │ │ │ - // over all indexing methods, we'll create a new z-index that's │ │ │ │ │ - // greater than any currently in the indexer. │ │ │ │ │ - if (zIndex == null) { │ │ │ │ │ - zIndex = this.maxZIndex; │ │ │ │ │ - node._style.graphicZIndex = zIndex; │ │ │ │ │ - } else if (zIndex > this.maxZIndex) { │ │ │ │ │ - this.maxZIndex = zIndex; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + virtualVertices: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getNextElement │ │ │ │ │ - * Get the next element in the order stack. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * index - {Integer} The index of the current node in this.order. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} the node following the index passed in, or │ │ │ │ │ - * null. │ │ │ │ │ + * Property: handlers │ │ │ │ │ + * {Object} │ │ │ │ │ */ │ │ │ │ │ - getNextElement: function(index) { │ │ │ │ │ - var nextIndex = index + 1; │ │ │ │ │ - if (nextIndex < this.order.length) { │ │ │ │ │ - var nextElement = OpenLayers.Util.getElement(this.order[nextIndex]); │ │ │ │ │ - if (nextElement == undefined) { │ │ │ │ │ - nextElement = this.getNextElement(nextIndex); │ │ │ │ │ - } │ │ │ │ │ - return nextElement; │ │ │ │ │ - } else { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.ElementsIndexer" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.ElementsIndexer.IndexingMethods │ │ │ │ │ - * These are the compare methods for figuring out where a new node should be │ │ │ │ │ - * placed within the indexer. These methods are very similar to general │ │ │ │ │ - * sorting methods in that they return -1, 0, and 1 to specify the │ │ │ │ │ - * direction in which new nodes fall in the ordering. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.ElementsIndexer.IndexingMethods = { │ │ │ │ │ + handlers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: Z_ORDER │ │ │ │ │ - * This compare method is used by other comparison methods. │ │ │ │ │ - * It can be used individually for ordering, but is not recommended, │ │ │ │ │ - * because it doesn't subscribe to drawing order. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ - * newNode - {DOMElement} │ │ │ │ │ - * nextNode - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} │ │ │ │ │ + * APIProperty: deleteCodes │ │ │ │ │ + * {Array(Integer)} Keycodes for deleting verticies. Set to null to disable │ │ │ │ │ + * vertex deltion by keypress. If non-null, keypresses with codes │ │ │ │ │ + * in this array will delete vertices under the mouse. Default │ │ │ │ │ + * is 46 and 68, the 'delete' and lowercase 'd' keys. │ │ │ │ │ */ │ │ │ │ │ - Z_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var newZIndex = indexer.getZIndex(newNode); │ │ │ │ │ - │ │ │ │ │ - var returnVal = 0; │ │ │ │ │ - if (nextNode) { │ │ │ │ │ - var nextZIndex = indexer.getZIndex(nextNode); │ │ │ │ │ - returnVal = newZIndex - nextZIndex; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return returnVal; │ │ │ │ │ - }, │ │ │ │ │ + deleteCodes: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: Z_ORDER_DRAWING_ORDER │ │ │ │ │ - * This method orders nodes by their z-index, but does so in a way │ │ │ │ │ - * that, if there are other nodes with the same z-index, the newest │ │ │ │ │ - * drawn will be the front most within that z-index. This is the │ │ │ │ │ - * default indexing method. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ - * newNode - {DOMElement} │ │ │ │ │ - * nextNode - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} │ │ │ │ │ + * APIProperty: virtualStyle │ │ │ │ │ + * {Object} A symbolizer to be used for virtual vertices. │ │ │ │ │ */ │ │ │ │ │ - Z_ORDER_DRAWING_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER( │ │ │ │ │ - indexer, │ │ │ │ │ - newNode, │ │ │ │ │ - nextNode │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - // Make Z_ORDER subscribe to drawing order by pushing it above │ │ │ │ │ - // all of the other nodes with the same z-index. │ │ │ │ │ - if (nextNode && returnVal == 0) { │ │ │ │ │ - returnVal = 1; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return returnVal; │ │ │ │ │ - }, │ │ │ │ │ + virtualStyle: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: Z_ORDER_Y_ORDER │ │ │ │ │ - * This one should really be called Z_ORDER_Y_ORDER_DRAWING_ORDER, as it │ │ │ │ │ - * best describes which ordering methods have precedence (though, the │ │ │ │ │ - * name would be too long). This method orders nodes by their z-index, │ │ │ │ │ - * but does so in a way that, if there are other nodes with the same │ │ │ │ │ - * z-index, the nodes with the lower y position will be "closer" than │ │ │ │ │ - * those with a higher y position. If two nodes have the exact same y │ │ │ │ │ - * position, however, then this method will revert to using drawing │ │ │ │ │ - * order to decide placement. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * indexer - {<OpenLayers.ElementsIndexer>} │ │ │ │ │ - * newNode - {DOMElement} │ │ │ │ │ - * nextNode - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} │ │ │ │ │ + * APIProperty: vertexRenderIntent │ │ │ │ │ + * {String} The renderIntent to use for vertices. If no <virtualStyle> is │ │ │ │ │ + * provided, this renderIntent will also be used for virtual vertices, with │ │ │ │ │ + * a fillOpacity and strokeOpacity of 0.3. Default is null, which means │ │ │ │ │ + * that the layer's default style will be used for vertices. │ │ │ │ │ */ │ │ │ │ │ - Z_ORDER_Y_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER( │ │ │ │ │ - indexer, │ │ │ │ │ - newNode, │ │ │ │ │ - nextNode │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - if (nextNode && returnVal === 0) { │ │ │ │ │ - var result = nextNode._boundsBottom - newNode._boundsBottom; │ │ │ │ │ - returnVal = (result === 0) ? 1 : result; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return returnVal; │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Renderer.Elements │ │ │ │ │ - * This is another virtual class in that it should never be instantiated by │ │ │ │ │ - * itself as a Renderer. It exists because there is *tons* of shared │ │ │ │ │ - * functionality between different vector libraries which use nodes/elements │ │ │ │ │ - * as a base for rendering vectors. │ │ │ │ │ - * │ │ │ │ │ - * The highlevel bits of code that are implemented here are the adding and │ │ │ │ │ - * removing of geometries, which is essentially the same for any │ │ │ │ │ - * element-based renderer. The details of creating each node and drawing the │ │ │ │ │ - * paths are of course different, but the machinery is the same. │ │ │ │ │ - * │ │ │ │ │ - * Inherits: │ │ │ │ │ - * - <OpenLayers.Renderer> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.Elements = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ + vertexRenderIntent: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: rendererRoot │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * APIProperty: mode │ │ │ │ │ + * {Integer} Bitfields specifying the modification mode. Defaults to │ │ │ │ │ + * OpenLayers.Control.ModifyFeature.RESHAPE. To set the mode to a │ │ │ │ │ + * combination of options, use the | operator. For example, to allow │ │ │ │ │ + * the control to both resize and rotate features, use the following │ │ │ │ │ + * syntax │ │ │ │ │ + * (code) │ │ │ │ │ + * control.mode = OpenLayers.Control.ModifyFeature.RESIZE | │ │ │ │ │ + * OpenLayers.Control.ModifyFeature.ROTATE; │ │ │ │ │ + * (end) │ │ │ │ │ */ │ │ │ │ │ - rendererRoot: null, │ │ │ │ │ + mode: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: root │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * APIProperty: createVertices │ │ │ │ │ + * {Boolean} Create new vertices by dragging the virtual vertices │ │ │ │ │ + * in the middle of each edge. Default is true. │ │ │ │ │ */ │ │ │ │ │ - root: null, │ │ │ │ │ + createVertices: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: vectorRoot │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Property: modified │ │ │ │ │ + * {Boolean} The currently selected feature has been modified. │ │ │ │ │ */ │ │ │ │ │ - vectorRoot: null, │ │ │ │ │ + modified: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: textRoot │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Property: radiusHandle │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A handle for rotating/resizing a feature. │ │ │ │ │ */ │ │ │ │ │ - textRoot: null, │ │ │ │ │ + radiusHandle: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: xmlns │ │ │ │ │ - * {String} │ │ │ │ │ + * Property: dragHandle │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A handle for dragging a feature. │ │ │ │ │ */ │ │ │ │ │ - xmlns: null, │ │ │ │ │ + dragHandle: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: xOffset │ │ │ │ │ - * {Number} Offset to apply to the renderer viewport translation in x │ │ │ │ │ - * direction. If the renderer extent's center is on the right of the │ │ │ │ │ - * dateline (i.e. exceeds the world bounds), we shift the viewport to the │ │ │ │ │ - * left by one world width. This avoids that features disappear from the │ │ │ │ │ - * map viewport. Because our dateline handling logic in other places │ │ │ │ │ - * ensures that extents crossing the dateline always have a center │ │ │ │ │ - * exceeding the world bounds on the left, we need this offset to make sure │ │ │ │ │ - * that the same is true for the renderer extent in pixel space as well. │ │ │ │ │ + * APIProperty: onModificationStart │ │ │ │ │ + * {Function} *Deprecated*. Register for "beforefeaturemodified" instead. │ │ │ │ │ + * The "beforefeaturemodified" event is triggered on the layer before │ │ │ │ │ + * any modification begins. │ │ │ │ │ + * │ │ │ │ │ + * Optional function to be called when a feature is selected │ │ │ │ │ + * to be modified. The function should expect to be called with a │ │ │ │ │ + * feature. This could be used for example to allow to lock the │ │ │ │ │ + * feature on server-side. │ │ │ │ │ */ │ │ │ │ │ - xOffset: 0, │ │ │ │ │ + onModificationStart: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: rightOfDateLine │ │ │ │ │ - * {Boolean} Keeps track of the location of the map extent relative to the │ │ │ │ │ - * date line. The <setExtent> method compares this value (which is the one │ │ │ │ │ - * from the previous <setExtent> call) with the current position of the map │ │ │ │ │ - * extent relative to the date line and updates the xOffset when the extent │ │ │ │ │ - * has moved from one side of the date line to the other. │ │ │ │ │ + * APIProperty: onModification │ │ │ │ │ + * {Function} *Deprecated*. Register for "featuremodified" instead. │ │ │ │ │ + * The "featuremodified" event is triggered on the layer with each │ │ │ │ │ + * feature modification. │ │ │ │ │ + * │ │ │ │ │ + * Optional function to be called when a feature has been │ │ │ │ │ + * modified. The function should expect to be called with a feature. │ │ │ │ │ */ │ │ │ │ │ + onModification: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: Indexer │ │ │ │ │ - * {<OpenLayers.ElementIndexer>} An instance of OpenLayers.ElementsIndexer │ │ │ │ │ - * created upon initialization if the zIndexing or yOrdering options │ │ │ │ │ - * passed to this renderer's constructor are set to true. │ │ │ │ │ + * APIProperty: onModificationEnd │ │ │ │ │ + * {Function} *Deprecated*. Register for "afterfeaturemodified" instead. │ │ │ │ │ + * The "afterfeaturemodified" event is triggered on the layer after │ │ │ │ │ + * a feature has been modified. │ │ │ │ │ + * │ │ │ │ │ + * Optional function to be called when a feature is finished │ │ │ │ │ + * being modified. The function should expect to be called with a │ │ │ │ │ + * feature. │ │ │ │ │ */ │ │ │ │ │ - indexer: null, │ │ │ │ │ + onModificationEnd: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: BACKGROUND_ID_SUFFIX │ │ │ │ │ - * {String} │ │ │ │ │ + * Constructor: OpenLayers.Control.ModifyFeature │ │ │ │ │ + * Create a new modify feature control. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer.Vector>} Layer that contains features that │ │ │ │ │ + * will be modified. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * control. │ │ │ │ │ */ │ │ │ │ │ - BACKGROUND_ID_SUFFIX: "_background", │ │ │ │ │ + initialize: function(layer, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + this.virtualVertices = []; │ │ │ │ │ + this.virtualStyle = OpenLayers.Util.extend({}, │ │ │ │ │ + this.layer.style || │ │ │ │ │ + this.layer.styleMap.createSymbolizer(null, options.vertexRenderIntent) │ │ │ │ │ + ); │ │ │ │ │ + this.virtualStyle.fillOpacity = 0.3; │ │ │ │ │ + this.virtualStyle.strokeOpacity = 0.3; │ │ │ │ │ + this.deleteCodes = [46, 68]; │ │ │ │ │ + this.mode = OpenLayers.Control.ModifyFeature.RESHAPE; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!(OpenLayers.Util.isArray(this.deleteCodes))) { │ │ │ │ │ + this.deleteCodes = [this.deleteCodes]; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: LABEL_ID_SUFFIX │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - LABEL_ID_SUFFIX: "_label", │ │ │ │ │ + // configure the drag handler │ │ │ │ │ + var dragCallbacks = { │ │ │ │ │ + down: function(pixel) { │ │ │ │ │ + this.vertex = null; │ │ │ │ │ + var feature = this.layer.getFeatureFromEvent( │ │ │ │ │ + this.handlers.drag.evt); │ │ │ │ │ + if (feature) { │ │ │ │ │ + this.dragStart(feature); │ │ │ │ │ + } else if (this.clickout) { │ │ │ │ │ + this._unselect = this.feature; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + move: function(pixel) { │ │ │ │ │ + delete this._unselect; │ │ │ │ │ + if (this.vertex) { │ │ │ │ │ + this.dragVertex(this.vertex, pixel); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + up: function() { │ │ │ │ │ + this.handlers.drag.stopDown = false; │ │ │ │ │ + if (this._unselect) { │ │ │ │ │ + this.unselectFeature(this._unselect); │ │ │ │ │ + delete this._unselect; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + done: function(pixel) { │ │ │ │ │ + if (this.vertex) { │ │ │ │ │ + this.dragComplete(this.vertex); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + var dragOptions = { │ │ │ │ │ + documentDrag: this.documentDrag, │ │ │ │ │ + stopDown: false │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: LABEL_OUTLINE_SUFFIX │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - LABEL_OUTLINE_SUFFIX: "_outline", │ │ │ │ │ + // configure the keyboard handler │ │ │ │ │ + var keyboardOptions = { │ │ │ │ │ + keydown: this.handleKeypress │ │ │ │ │ + }; │ │ │ │ │ + this.handlers = { │ │ │ │ │ + keyboard: new OpenLayers.Handler.Keyboard(this, keyboardOptions), │ │ │ │ │ + drag: new OpenLayers.Handler.Drag(this, dragCallbacks, dragOptions) │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Renderer.Elements │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * containerID - {String} │ │ │ │ │ - * options - {Object} options for this renderer. │ │ │ │ │ - * │ │ │ │ │ - * Supported options are: │ │ │ │ │ - * yOrdering - {Boolean} Whether to use y-ordering │ │ │ │ │ - * zIndexing - {Boolean} Whether to use z-indexing. Will be ignored │ │ │ │ │ - * if yOrdering is set to true. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Take care of things that are not handled in superclass. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(containerID, options) { │ │ │ │ │ - OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - this.rendererRoot = this.createRenderRoot(); │ │ │ │ │ - this.root = this.createRoot("_root"); │ │ │ │ │ - this.vectorRoot = this.createRoot("_vroot"); │ │ │ │ │ - this.textRoot = this.createRoot("_troot"); │ │ │ │ │ - │ │ │ │ │ - this.root.appendChild(this.vectorRoot); │ │ │ │ │ - this.root.appendChild(this.textRoot); │ │ │ │ │ - │ │ │ │ │ - this.rendererRoot.appendChild(this.root); │ │ │ │ │ - this.container.appendChild(this.rendererRoot); │ │ │ │ │ - │ │ │ │ │ - if (options && (options.zIndexing || options.yOrdering)) { │ │ │ │ │ - this.indexer = new OpenLayers.ElementsIndexer(options.yOrdering); │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + "removelayer": this.handleMapEvents, │ │ │ │ │ + "changelayer": this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ + this.layer = null; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, []); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Activate the control. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Successfully activated the control. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - │ │ │ │ │ - this.clear(); │ │ │ │ │ - │ │ │ │ │ - this.rendererRoot = null; │ │ │ │ │ - this.root = null; │ │ │ │ │ - this.xmlns = null; │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Renderer.prototype.destroy.apply(this, arguments); │ │ │ │ │ + activate: function() { │ │ │ │ │ + this.moveLayerToTop(); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + "removelayer": this.handleMapEvents, │ │ │ │ │ + "changelayer": this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + return (this.handlers.keyboard.activate() && │ │ │ │ │ + this.handlers.drag.activate() && │ │ │ │ │ + OpenLayers.Control.prototype.activate.apply(this, arguments)); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clear │ │ │ │ │ - * Remove all the elements from the root │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the control. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Successfully deactivated the control. │ │ │ │ │ */ │ │ │ │ │ - clear: function() { │ │ │ │ │ - var child; │ │ │ │ │ - var root = this.vectorRoot; │ │ │ │ │ - if (root) { │ │ │ │ │ - while (child = root.firstChild) { │ │ │ │ │ - root.removeChild(child); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - root = this.textRoot; │ │ │ │ │ - if (root) { │ │ │ │ │ - while (child = root.firstChild) { │ │ │ │ │ - root.removeChild(child); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + // the return from the controls is unimportant in this case │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerBack(); │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + "removelayer": this.handleMapEvents, │ │ │ │ │ + "changelayer": this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.layer.removeFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.layer.removeFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + this.handlers.drag.deactivate(); │ │ │ │ │ + this.handlers.keyboard.deactivate(); │ │ │ │ │ + var feature = this.feature; │ │ │ │ │ + if (feature && feature.geometry && feature.layer) { │ │ │ │ │ + this.unselectFeature(feature); │ │ │ │ │ } │ │ │ │ │ + deactivated = true; │ │ │ │ │ } │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - this.indexer.clear(); │ │ │ │ │ - } │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setExtent │ │ │ │ │ - * Set the visible part of the layer. │ │ │ │ │ + * Method: beforeSelectFeature │ │ │ │ │ + * Called before a feature is selected. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} │ │ │ │ │ - * resolutionChanged - {Boolean} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ - * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ - * False otherwise. │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} The feature about to be selected. │ │ │ │ │ */ │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - var rightOfDateLine, │ │ │ │ │ - ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ - extent = extent.scale(1 / ratio), │ │ │ │ │ - world = this.map.getMaxExtent(); │ │ │ │ │ - if (world.right > extent.left && world.right < extent.right) { │ │ │ │ │ - rightOfDateLine = true; │ │ │ │ │ - } else if (world.left > extent.left && world.left < extent.right) { │ │ │ │ │ - rightOfDateLine = false; │ │ │ │ │ - } │ │ │ │ │ - if (rightOfDateLine !== this.rightOfDateLine || resolutionChanged) { │ │ │ │ │ - coordSysUnchanged = false; │ │ │ │ │ - this.xOffset = rightOfDateLine === true ? │ │ │ │ │ - world.getWidth() / resolution : 0; │ │ │ │ │ + beforeSelectFeature: function(feature) { │ │ │ │ │ + return this.layer.events.triggerEvent( │ │ │ │ │ + "beforefeaturemodified", { │ │ │ │ │ + feature: feature │ │ │ │ │ } │ │ │ │ │ - this.rightOfDateLine = rightOfDateLine; │ │ │ │ │ - } │ │ │ │ │ - return coordSysUnchanged; │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getNodeType │ │ │ │ │ - * This function is in charge of asking the specific renderer which type │ │ │ │ │ - * of node to create for the given geometry and style. All geometries │ │ │ │ │ - * in an Elements-based renderer consist of one node and some │ │ │ │ │ - * attributes. We have the nodeFactory() function which creates a node │ │ │ │ │ - * for us, but it takes a 'type' as input, and that is precisely what │ │ │ │ │ - * this function tells us. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The corresponding node type for the specified geometry │ │ │ │ │ - */ │ │ │ │ │ - getNodeType: function(geometry, style) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawGeometry │ │ │ │ │ - * Draw the geometry, creating new nodes, setting paths, setting style, │ │ │ │ │ - * setting featureId on the node. This method should only be called │ │ │ │ │ - * by the renderer itself. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: selectFeature │ │ │ │ │ + * Select a feature for modification in standalone mode. In non-standalone │ │ │ │ │ + * mode, this method is called when a feature is selected by clicking. │ │ │ │ │ + * Register a listener to the beforefeaturemodified event and return false │ │ │ │ │ + * to prevent feature modification. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true if the geometry has been drawn completely; null if │ │ │ │ │ - * incomplete; false otherwise │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} the selected feature. │ │ │ │ │ */ │ │ │ │ │ - drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ - var className = geometry.CLASS_NAME; │ │ │ │ │ - var rendered = true; │ │ │ │ │ - if ((className == "OpenLayers.Geometry.Collection") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiPolygon")) { │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - rendered = this.drawGeometry( │ │ │ │ │ - geometry.components[i], style, featureId) && rendered; │ │ │ │ │ - } │ │ │ │ │ - return rendered; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - rendered = false; │ │ │ │ │ - var removeBackground = false; │ │ │ │ │ - if (style.display != "none") { │ │ │ │ │ - if (style.backgroundGraphic) { │ │ │ │ │ - this.redrawBackgroundNode(geometry.id, geometry, style, │ │ │ │ │ - featureId); │ │ │ │ │ - } else { │ │ │ │ │ - removeBackground = true; │ │ │ │ │ - } │ │ │ │ │ - rendered = this.redrawNode(geometry.id, geometry, style, │ │ │ │ │ - featureId); │ │ │ │ │ + selectFeature: function(feature) { │ │ │ │ │ + if (this.feature === feature || │ │ │ │ │ + (this.geometryTypes && OpenLayers.Util.indexOf(this.geometryTypes, │ │ │ │ │ + feature.geometry.CLASS_NAME) == -1)) { │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ - if (rendered == false) { │ │ │ │ │ - var node = document.getElementById(geometry.id); │ │ │ │ │ - if (node) { │ │ │ │ │ - if (node._style.backgroundGraphic) { │ │ │ │ │ - removeBackground = true; │ │ │ │ │ - } │ │ │ │ │ - node.parentNode.removeChild(node); │ │ │ │ │ + if (this.beforeSelectFeature(feature) !== false) { │ │ │ │ │ + if (this.feature) { │ │ │ │ │ + this.unselectFeature(this.feature); │ │ │ │ │ } │ │ │ │ │ + this.feature = feature; │ │ │ │ │ + this.layer.selectedFeatures.push(feature); │ │ │ │ │ + this.layer.drawFeature(feature, 'select'); │ │ │ │ │ + this.modified = false; │ │ │ │ │ + this.resetVertices(); │ │ │ │ │ + this.onModificationStart(this.feature); │ │ │ │ │ } │ │ │ │ │ - if (removeBackground) { │ │ │ │ │ - var node = document.getElementById( │ │ │ │ │ - geometry.id + this.BACKGROUND_ID_SUFFIX); │ │ │ │ │ - if (node) { │ │ │ │ │ - node.parentNode.removeChild(node); │ │ │ │ │ - } │ │ │ │ │ + // keep track of geometry modifications │ │ │ │ │ + var modified = feature.modified; │ │ │ │ │ + if (feature.geometry && !(modified && modified.geometry)) { │ │ │ │ │ + this._originalGeometry = feature.geometry.clone(); │ │ │ │ │ } │ │ │ │ │ - return rendered; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: redrawNode │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: unselectFeature │ │ │ │ │ + * Called when the select feature control unselects a feature. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true if the complete geometry could be drawn, null if parts of │ │ │ │ │ - * the geometry could not be drawn, false otherwise │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} The unselected feature. │ │ │ │ │ */ │ │ │ │ │ - redrawNode: function(id, geometry, style, featureId) { │ │ │ │ │ - style = this.applyDefaultSymbolizer(style); │ │ │ │ │ - // Get the node if it's already on the map. │ │ │ │ │ - var node = this.nodeFactory(id, this.getNodeType(geometry, style)); │ │ │ │ │ - │ │ │ │ │ - // Set the data for the node, then draw it. │ │ │ │ │ - node._featureId = featureId; │ │ │ │ │ - node._boundsBottom = geometry.getBounds().bottom; │ │ │ │ │ - node._geometryClass = geometry.CLASS_NAME; │ │ │ │ │ - node._style = style; │ │ │ │ │ - │ │ │ │ │ - var drawResult = this.drawGeometryNode(node, geometry, style); │ │ │ │ │ - if (drawResult === false) { │ │ │ │ │ - return false; │ │ │ │ │ + unselectFeature: function(feature) { │ │ │ │ │ + this.layer.removeFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + this.layer.destroyFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.virtualVertices = []; │ │ │ │ │ + if (this.dragHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.dragHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + delete this.dragHandle; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - node = drawResult.node; │ │ │ │ │ - │ │ │ │ │ - // Insert the node into the indexer so it can show us where to │ │ │ │ │ - // place it. Note that this operation is O(log(n)). If there's a │ │ │ │ │ - // performance problem (when dragging, for instance) this is │ │ │ │ │ - // likely where it would be. │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - var insert = this.indexer.insert(node); │ │ │ │ │ - if (insert) { │ │ │ │ │ - this.vectorRoot.insertBefore(node, insert); │ │ │ │ │ - } else { │ │ │ │ │ - this.vectorRoot.appendChild(node); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - // if there's no indexer, simply append the node to root, │ │ │ │ │ - // but only if the node is a new one │ │ │ │ │ - if (node.parentNode !== this.vectorRoot) { │ │ │ │ │ - this.vectorRoot.appendChild(node); │ │ │ │ │ - } │ │ │ │ │ + if (this.radiusHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + delete this.radiusHandle; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - this.postDraw(node); │ │ │ │ │ - │ │ │ │ │ - return drawResult.complete; │ │ │ │ │ + this.layer.drawFeature(this.feature, 'default'); │ │ │ │ │ + this.feature = null; │ │ │ │ │ + OpenLayers.Util.removeItem(this.layer.selectedFeatures, feature); │ │ │ │ │ + this.onModificationEnd(feature); │ │ │ │ │ + this.layer.events.triggerEvent("afterfeaturemodified", { │ │ │ │ │ + feature: feature, │ │ │ │ │ + modified: this.modified │ │ │ │ │ + }); │ │ │ │ │ + this.modified = false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: redrawBackgroundNode │ │ │ │ │ - * Redraws the node using special 'background' style properties. Basically │ │ │ │ │ - * just calls redrawNode(), but instead of directly using the │ │ │ │ │ - * 'externalGraphic', 'graphicXOffset', 'graphicYOffset', and │ │ │ │ │ - * 'graphicZIndex' properties directly from the specified 'style' │ │ │ │ │ - * parameter, we create a new style object and set those properties │ │ │ │ │ - * from the corresponding 'background'-prefixed properties from │ │ │ │ │ - * specified 'style' parameter. │ │ │ │ │ - * │ │ │ │ │ + * Method: dragStart │ │ │ │ │ + * Called by the drag handler before a feature is dragged. This method is │ │ │ │ │ + * used to differentiate between points and vertices │ │ │ │ │ + * of higher order geometries. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true if the complete geometry could be drawn, null if parts of │ │ │ │ │ - * the geometry could not be drawn, false otherwise │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} The point or vertex about to be │ │ │ │ │ + * dragged. │ │ │ │ │ */ │ │ │ │ │ - redrawBackgroundNode: function(id, geometry, style, featureId) { │ │ │ │ │ - var backgroundStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ - │ │ │ │ │ - // Set regular style attributes to apply to the background styles. │ │ │ │ │ - backgroundStyle.externalGraphic = backgroundStyle.backgroundGraphic; │ │ │ │ │ - backgroundStyle.graphicXOffset = backgroundStyle.backgroundXOffset; │ │ │ │ │ - backgroundStyle.graphicYOffset = backgroundStyle.backgroundYOffset; │ │ │ │ │ - backgroundStyle.graphicZIndex = backgroundStyle.backgroundGraphicZIndex; │ │ │ │ │ - backgroundStyle.graphicWidth = backgroundStyle.backgroundWidth || backgroundStyle.graphicWidth; │ │ │ │ │ - backgroundStyle.graphicHeight = backgroundStyle.backgroundHeight || backgroundStyle.graphicHeight; │ │ │ │ │ - │ │ │ │ │ - // Erase background styles. │ │ │ │ │ - backgroundStyle.backgroundGraphic = null; │ │ │ │ │ - backgroundStyle.backgroundXOffset = null; │ │ │ │ │ - backgroundStyle.backgroundYOffset = null; │ │ │ │ │ - backgroundStyle.backgroundGraphicZIndex = null; │ │ │ │ │ - │ │ │ │ │ - return this.redrawNode( │ │ │ │ │ - id + this.BACKGROUND_ID_SUFFIX, │ │ │ │ │ - geometry, │ │ │ │ │ - backgroundStyle, │ │ │ │ │ - null │ │ │ │ │ - ); │ │ │ │ │ + dragStart: function(feature) { │ │ │ │ │ + var isPoint = feature.geometry.CLASS_NAME == │ │ │ │ │ + 'OpenLayers.Geometry.Point'; │ │ │ │ │ + if (!this.standalone && │ │ │ │ │ + ((!feature._sketch && isPoint) || !feature._sketch)) { │ │ │ │ │ + if (this.toggle && this.feature === feature) { │ │ │ │ │ + // mark feature for unselection │ │ │ │ │ + this._unselect = feature; │ │ │ │ │ + } │ │ │ │ │ + this.selectFeature(feature); │ │ │ │ │ + } │ │ │ │ │ + if (feature._sketch || isPoint) { │ │ │ │ │ + // feature is a drag or virtual handle or point │ │ │ │ │ + this.vertex = feature; │ │ │ │ │ + this.handlers.drag.stopDown = true; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawGeometryNode │ │ │ │ │ - * Given a node, draw a geometry on the specified layer. │ │ │ │ │ - * node and geometry are required arguments, style is optional. │ │ │ │ │ - * This method is only called by the render itself. │ │ │ │ │ + * Method: dragVertex │ │ │ │ │ + * Called by the drag handler with each drag move of a vertex. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} a hash with properties "node" (the drawn node) and "complete" │ │ │ │ │ - * (null if parts of the geometry could not be drawn, false if nothing │ │ │ │ │ - * could be drawn) │ │ │ │ │ + * vertex - {<OpenLayers.Feature.Vector>} The vertex being dragged. │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} Pixel location of the mouse event. │ │ │ │ │ */ │ │ │ │ │ - drawGeometryNode: function(node, geometry, style) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - │ │ │ │ │ - var options = { │ │ │ │ │ - 'isFilled': style.fill === undefined ? │ │ │ │ │ - true : style.fill, │ │ │ │ │ - 'isStroked': style.stroke === undefined ? │ │ │ │ │ - !!style.strokeWidth : style.stroke │ │ │ │ │ - }; │ │ │ │ │ - var drawn; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.graphic === false) { │ │ │ │ │ - options.isFilled = false; │ │ │ │ │ - options.isStroked = false; │ │ │ │ │ - } │ │ │ │ │ - drawn = this.drawPoint(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - options.isFilled = false; │ │ │ │ │ - drawn = this.drawLineString(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - drawn = this.drawLinearRing(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - drawn = this.drawPolygon(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - drawn = this.drawRectangle(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - node._options = options; │ │ │ │ │ - │ │ │ │ │ - //set style │ │ │ │ │ - //TBD simplify this │ │ │ │ │ - if (drawn != false) { │ │ │ │ │ - return { │ │ │ │ │ - node: this.setStyle(node, style, options, geometry), │ │ │ │ │ - complete: drawn │ │ │ │ │ - }; │ │ │ │ │ + dragVertex: function(vertex, pixel) { │ │ │ │ │ + var pos = this.map.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geom = vertex.geometry; │ │ │ │ │ + geom.move(pos.lon - geom.x, pos.lat - geom.y); │ │ │ │ │ + this.modified = true; │ │ │ │ │ + /** │ │ │ │ │ + * Five cases: │ │ │ │ │ + * 1) dragging a simple point │ │ │ │ │ + * 2) dragging a virtual vertex │ │ │ │ │ + * 3) dragging a drag handle │ │ │ │ │ + * 4) dragging a real vertex │ │ │ │ │ + * 5) dragging a radius handle │ │ │ │ │ + */ │ │ │ │ │ + if (this.feature.geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + // dragging a simple point │ │ │ │ │ + this.layer.events.triggerEvent("vertexmodified", { │ │ │ │ │ + vertex: vertex.geometry, │ │ │ │ │ + feature: this.feature, │ │ │ │ │ + pixel: pixel │ │ │ │ │ + }); │ │ │ │ │ } else { │ │ │ │ │ - return false; │ │ │ │ │ + if (vertex._index) { │ │ │ │ │ + // dragging a virtual vertex │ │ │ │ │ + vertex.geometry.parent.addComponent(vertex.geometry, │ │ │ │ │ + vertex._index); │ │ │ │ │ + // move from virtual to real vertex │ │ │ │ │ + delete vertex._index; │ │ │ │ │ + OpenLayers.Util.removeItem(this.virtualVertices, vertex); │ │ │ │ │ + this.vertices.push(vertex); │ │ │ │ │ + } else if (vertex == this.dragHandle) { │ │ │ │ │ + // dragging a drag handle │ │ │ │ │ + this.layer.removeFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + if (this.radiusHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.radiusHandle = null; │ │ │ │ │ + } │ │ │ │ │ + } else if (vertex !== this.radiusHandle) { │ │ │ │ │ + // dragging a real vertex │ │ │ │ │ + this.layer.events.triggerEvent("vertexmodified", { │ │ │ │ │ + vertex: vertex.geometry, │ │ │ │ │ + feature: this.feature, │ │ │ │ │ + pixel: pixel │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + // dragging a radius handle - no special treatment │ │ │ │ │ + if (this.virtualVertices.length > 0) { │ │ │ │ │ + this.layer.destroyFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.virtualVertices = []; │ │ │ │ │ + } │ │ │ │ │ + this.layer.drawFeature(this.feature, this.standalone ? undefined : │ │ │ │ │ + 'select'); │ │ │ │ │ } │ │ │ │ │ + // keep the vertex on top so it gets the mouseout after dragging │ │ │ │ │ + // this should be removed in favor of an option to draw under or │ │ │ │ │ + // maintain node z-index │ │ │ │ │ + this.layer.drawFeature(vertex); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: postDraw │ │ │ │ │ - * Things that have do be done after the geometry node is appended │ │ │ │ │ - * to its parent node. To be overridden by subclasses. │ │ │ │ │ - * │ │ │ │ │ + * Method: dragComplete │ │ │ │ │ + * Called by the drag handler when the feature dragging is complete. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - postDraw: function(node) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawPoint │ │ │ │ │ - * Virtual function for drawing Point Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the point │ │ │ │ │ - */ │ │ │ │ │ - drawPoint: function(node, geometry) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawLineString │ │ │ │ │ - * Virtual function for drawing LineString Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components of │ │ │ │ │ - * the linestring, or false if nothing could be drawn │ │ │ │ │ - */ │ │ │ │ │ - drawLineString: function(node, geometry) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawLinearRing │ │ │ │ │ - * Virtual function for drawing LinearRing Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ - * of the linear ring, or false if nothing could be drawn │ │ │ │ │ - */ │ │ │ │ │ - drawLinearRing: function(node, geometry) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawPolygon │ │ │ │ │ - * Virtual function for drawing Polygon Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ - * of the polygon, or false if nothing could be drawn │ │ │ │ │ - */ │ │ │ │ │ - drawPolygon: function(node, geometry) {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawRectangle │ │ │ │ │ - * Virtual function for drawing Rectangle Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the rectangle │ │ │ │ │ + * vertex - {<OpenLayers.Feature.Vector>} The vertex being dragged. │ │ │ │ │ */ │ │ │ │ │ - drawRectangle: function(node, geometry) {}, │ │ │ │ │ + dragComplete: function(vertex) { │ │ │ │ │ + this.resetVertices(); │ │ │ │ │ + this.setFeatureState(); │ │ │ │ │ + this.onModification(this.feature); │ │ │ │ │ + this.layer.events.triggerEvent("featuremodified", { │ │ │ │ │ + feature: this.feature │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawCircle │ │ │ │ │ - * Virtual function for drawing Circle Geometry. │ │ │ │ │ - * Should be implemented by subclasses. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the circle │ │ │ │ │ + * Method: setFeatureState │ │ │ │ │ + * Called when the feature is modified. If the current state is not │ │ │ │ │ + * INSERT or DELETE, the state is set to UPDATE. │ │ │ │ │ */ │ │ │ │ │ - drawCircle: function(node, geometry) {}, │ │ │ │ │ + setFeatureState: function() { │ │ │ │ │ + if (this.feature.state != OpenLayers.State.INSERT && │ │ │ │ │ + this.feature.state != OpenLayers.State.DELETE) { │ │ │ │ │ + this.feature.state = OpenLayers.State.UPDATE; │ │ │ │ │ + if (this.modified && this._originalGeometry) { │ │ │ │ │ + var feature = this.feature; │ │ │ │ │ + feature.modified = OpenLayers.Util.extend(feature.modified, { │ │ │ │ │ + geometry: this._originalGeometry │ │ │ │ │ + }); │ │ │ │ │ + delete this._originalGeometry; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removeText │ │ │ │ │ - * Removes a label │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * Method: resetVertices │ │ │ │ │ */ │ │ │ │ │ - removeText: function(featureId) { │ │ │ │ │ - var label = document.getElementById(featureId + this.LABEL_ID_SUFFIX); │ │ │ │ │ - if (label) { │ │ │ │ │ - this.textRoot.removeChild(label); │ │ │ │ │ + resetVertices: function() { │ │ │ │ │ + if (this.vertices.length > 0) { │ │ │ │ │ + this.layer.removeFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.vertices = []; │ │ │ │ │ } │ │ │ │ │ - var outline = document.getElementById(featureId + this.LABEL_OUTLINE_SUFFIX); │ │ │ │ │ - if (outline) { │ │ │ │ │ - this.textRoot.removeChild(outline); │ │ │ │ │ + if (this.virtualVertices.length > 0) { │ │ │ │ │ + this.layer.removeFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.virtualVertices = []; │ │ │ │ │ + } │ │ │ │ │ + if (this.dragHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.dragHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.dragHandle = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.radiusHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.radiusHandle = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.feature && │ │ │ │ │ + this.feature.geometry.CLASS_NAME != "OpenLayers.Geometry.Point") { │ │ │ │ │ + if ((this.mode & OpenLayers.Control.ModifyFeature.DRAG)) { │ │ │ │ │ + this.collectDragHandle(); │ │ │ │ │ + } │ │ │ │ │ + if ((this.mode & (OpenLayers.Control.ModifyFeature.ROTATE | │ │ │ │ │ + OpenLayers.Control.ModifyFeature.RESIZE))) { │ │ │ │ │ + this.collectRadiusHandle(); │ │ │ │ │ + } │ │ │ │ │ + if (this.mode & OpenLayers.Control.ModifyFeature.RESHAPE) { │ │ │ │ │ + // Don't collect vertices when we're resizing │ │ │ │ │ + if (!(this.mode & OpenLayers.Control.ModifyFeature.RESIZE)) { │ │ │ │ │ + this.collectVertices(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureIdFromEvent │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} An <OpenLayers.Event> object │ │ │ │ │ + * Method: handleKeypress │ │ │ │ │ + * Called by the feature handler on keypress. This is used to delete │ │ │ │ │ + * vertices. If the <deleteCode> property is set, vertices will │ │ │ │ │ + * be deleted when a feature is selected for modification and │ │ │ │ │ + * the mouse is over a vertex. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A feature id or undefined. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} Keypress event. │ │ │ │ │ */ │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var target = evt.target; │ │ │ │ │ - var useElement = target && target.correspondingUseElement; │ │ │ │ │ - var node = useElement ? useElement : (target || evt.srcElement); │ │ │ │ │ - return node._featureId; │ │ │ │ │ + handleKeypress: function(evt) { │ │ │ │ │ + var code = evt.keyCode; │ │ │ │ │ + │ │ │ │ │ + // check for delete key │ │ │ │ │ + if (this.feature && │ │ │ │ │ + OpenLayers.Util.indexOf(this.deleteCodes, code) != -1) { │ │ │ │ │ + var vertex = this.layer.getFeatureFromEvent(this.handlers.drag.evt); │ │ │ │ │ + if (vertex && │ │ │ │ │ + OpenLayers.Util.indexOf(this.vertices, vertex) != -1 && │ │ │ │ │ + !this.handlers.drag.dragging && vertex.geometry.parent) { │ │ │ │ │ + // remove the vertex │ │ │ │ │ + vertex.geometry.parent.removeComponent(vertex.geometry); │ │ │ │ │ + this.layer.events.triggerEvent("vertexremoved", { │ │ │ │ │ + vertex: vertex.geometry, │ │ │ │ │ + feature: this.feature, │ │ │ │ │ + pixel: evt.xy │ │ │ │ │ + }); │ │ │ │ │ + this.layer.drawFeature(this.feature, this.standalone ? │ │ │ │ │ + undefined : 'select'); │ │ │ │ │ + this.modified = true; │ │ │ │ │ + this.resetVertices(); │ │ │ │ │ + this.setFeatureState(); │ │ │ │ │ + this.onModification(this.feature); │ │ │ │ │ + this.layer.events.triggerEvent("featuremodified", { │ │ │ │ │ + feature: this.feature │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: eraseGeometry │ │ │ │ │ - * Erase a geometry from the renderer. In the case of a multi-geometry, │ │ │ │ │ - * we cycle through and recurse on ourselves. Otherwise, we look for a │ │ │ │ │ - * node with the geometry.id, destroy its geometry, and remove it from │ │ │ │ │ - * the DOM. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + /** │ │ │ │ │ + * Method: collectVertices │ │ │ │ │ + * Collect the vertices from the modifiable feature's geometry and push │ │ │ │ │ + * them on to the control's vertices array. │ │ │ │ │ */ │ │ │ │ │ - eraseGeometry: function(geometry, featureId) { │ │ │ │ │ - if ((geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ - (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ - (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPolygon") || │ │ │ │ │ - (geometry.CLASS_NAME == "OpenLayers.Geometry.Collection")) { │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - this.eraseGeometry(geometry.components[i], featureId); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - var element = OpenLayers.Util.getElement(geometry.id); │ │ │ │ │ - if (element && element.parentNode) { │ │ │ │ │ - if (element.geometry) { │ │ │ │ │ - element.geometry.destroy(); │ │ │ │ │ - element.geometry = null; │ │ │ │ │ - } │ │ │ │ │ - element.parentNode.removeChild(element); │ │ │ │ │ + collectVertices: function() { │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + this.virtualVertices = []; │ │ │ │ │ + var control = this; │ │ │ │ │ │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - this.indexer.remove(element); │ │ │ │ │ + function collectComponentVertices(geometry) { │ │ │ │ │ + var i, vertex, component, len; │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + vertex = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + vertex._sketch = true; │ │ │ │ │ + vertex.renderIntent = control.vertexRenderIntent; │ │ │ │ │ + control.vertices.push(vertex); │ │ │ │ │ + } else { │ │ │ │ │ + var numVert = geometry.components.length; │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ + numVert -= 1; │ │ │ │ │ + } │ │ │ │ │ + for (i = 0; i < numVert; ++i) { │ │ │ │ │ + component = geometry.components[i]; │ │ │ │ │ + if (component.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + vertex = new OpenLayers.Feature.Vector(component); │ │ │ │ │ + vertex._sketch = true; │ │ │ │ │ + vertex.renderIntent = control.vertexRenderIntent; │ │ │ │ │ + control.vertices.push(vertex); │ │ │ │ │ + } else { │ │ │ │ │ + collectComponentVertices(component); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - if (element._style.backgroundGraphic) { │ │ │ │ │ - var backgroundId = geometry.id + this.BACKGROUND_ID_SUFFIX; │ │ │ │ │ - var bElem = OpenLayers.Util.getElement(backgroundId); │ │ │ │ │ - if (bElem && bElem.parentNode) { │ │ │ │ │ - // No need to destroy the geometry since the element and the background │ │ │ │ │ - // node share the same geometry. │ │ │ │ │ - bElem.parentNode.removeChild(bElem); │ │ │ │ │ + // add virtual vertices in the middle of each edge │ │ │ │ │ + if (control.createVertices && geometry.CLASS_NAME != "OpenLayers.Geometry.MultiPoint") { │ │ │ │ │ + for (i = 0, len = geometry.components.length; i < len - 1; ++i) { │ │ │ │ │ + var prevVertex = geometry.components[i]; │ │ │ │ │ + var nextVertex = geometry.components[i + 1]; │ │ │ │ │ + if (prevVertex.CLASS_NAME == "OpenLayers.Geometry.Point" && │ │ │ │ │ + nextVertex.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + var x = (prevVertex.x + nextVertex.x) / 2; │ │ │ │ │ + var y = (prevVertex.y + nextVertex.y) / 2; │ │ │ │ │ + var point = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.Point(x, y), │ │ │ │ │ + null, control.virtualStyle │ │ │ │ │ + ); │ │ │ │ │ + // set the virtual parent and intended index │ │ │ │ │ + point.geometry.parent = geometry; │ │ │ │ │ + point._index = i + 1; │ │ │ │ │ + point._sketch = true; │ │ │ │ │ + control.virtualVertices.push(point); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + collectComponentVertices.call(this, this.feature.geometry); │ │ │ │ │ + this.layer.addFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.layer.addFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: nodeFactory │ │ │ │ │ - * Create new node of the specified type, with the (optional) specified id. │ │ │ │ │ - * │ │ │ │ │ - * If node already exists with same ID and a different type, we remove it │ │ │ │ │ - * and then call ourselves again to recreate it. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * type - {String} type Kind of node to draw. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A new node of the given type and id. │ │ │ │ │ + /** │ │ │ │ │ + * Method: collectDragHandle │ │ │ │ │ + * Collect the drag handle for the selected geometry. │ │ │ │ │ */ │ │ │ │ │ - nodeFactory: function(id, type) { │ │ │ │ │ - var node = OpenLayers.Util.getElement(id); │ │ │ │ │ - if (node) { │ │ │ │ │ - if (!this.nodeTypeCompare(node, type)) { │ │ │ │ │ - node.parentNode.removeChild(node); │ │ │ │ │ - node = this.nodeFactory(id, type); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - node = this.createNode(type, id); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ + collectDragHandle: function() { │ │ │ │ │ + var geometry = this.feature.geometry; │ │ │ │ │ + var center = geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + var originGeometry = new OpenLayers.Geometry.Point( │ │ │ │ │ + center.lon, center.lat │ │ │ │ │ + ); │ │ │ │ │ + var origin = new OpenLayers.Feature.Vector(originGeometry); │ │ │ │ │ + originGeometry.move = function(x, y) { │ │ │ │ │ + OpenLayers.Geometry.Point.prototype.move.call(this, x, y); │ │ │ │ │ + geometry.move(x, y); │ │ │ │ │ + }; │ │ │ │ │ + origin._sketch = true; │ │ │ │ │ + this.dragHandle = origin; │ │ │ │ │ + this.dragHandle.renderIntent = this.vertexRenderIntent; │ │ │ │ │ + this.layer.addFeatures([this.dragHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: nodeTypeCompare │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * type - {String} Kind of node │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ - * This function must be overridden by subclasses. │ │ │ │ │ + /** │ │ │ │ │ + * Method: collectRadiusHandle │ │ │ │ │ + * Collect the radius handle for the selected geometry. │ │ │ │ │ */ │ │ │ │ │ - nodeTypeCompare: function(node, type) {}, │ │ │ │ │ + collectRadiusHandle: function() { │ │ │ │ │ + var geometry = this.feature.geometry; │ │ │ │ │ + var bounds = geometry.getBounds(); │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var originGeometry = new OpenLayers.Geometry.Point( │ │ │ │ │ + center.lon, center.lat │ │ │ │ │ + ); │ │ │ │ │ + var radiusGeometry = new OpenLayers.Geometry.Point( │ │ │ │ │ + bounds.right, bounds.bottom │ │ │ │ │ + ); │ │ │ │ │ + var radius = new OpenLayers.Feature.Vector(radiusGeometry); │ │ │ │ │ + var resize = (this.mode & OpenLayers.Control.ModifyFeature.RESIZE); │ │ │ │ │ + var reshape = (this.mode & OpenLayers.Control.ModifyFeature.RESHAPE); │ │ │ │ │ + var rotate = (this.mode & OpenLayers.Control.ModifyFeature.ROTATE); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createNode │ │ │ │ │ - * │ │ │ │ │ + radiusGeometry.move = function(x, y) { │ │ │ │ │ + OpenLayers.Geometry.Point.prototype.move.call(this, x, y); │ │ │ │ │ + var dx1 = this.x - originGeometry.x; │ │ │ │ │ + var dy1 = this.y - originGeometry.y; │ │ │ │ │ + var dx0 = dx1 - x; │ │ │ │ │ + var dy0 = dy1 - y; │ │ │ │ │ + if (rotate) { │ │ │ │ │ + var a0 = Math.atan2(dy0, dx0); │ │ │ │ │ + var a1 = Math.atan2(dy1, dx1); │ │ │ │ │ + var angle = a1 - a0; │ │ │ │ │ + angle *= 180 / Math.PI; │ │ │ │ │ + geometry.rotate(angle, originGeometry); │ │ │ │ │ + } │ │ │ │ │ + if (resize) { │ │ │ │ │ + var scale, ratio; │ │ │ │ │ + // 'resize' together with 'reshape' implies that the aspect │ │ │ │ │ + // ratio of the geometry will not be preserved whilst resizing │ │ │ │ │ + if (reshape) { │ │ │ │ │ + scale = dy1 / dy0; │ │ │ │ │ + ratio = (dx1 / dx0) / scale; │ │ │ │ │ + } else { │ │ │ │ │ + var l0 = Math.sqrt((dx0 * dx0) + (dy0 * dy0)); │ │ │ │ │ + var l1 = Math.sqrt((dx1 * dx1) + (dy1 * dy1)); │ │ │ │ │ + scale = l1 / l0; │ │ │ │ │ + } │ │ │ │ │ + geometry.resize(scale, originGeometry, ratio); │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + radius._sketch = true; │ │ │ │ │ + this.radiusHandle = radius; │ │ │ │ │ + this.radiusHandle.renderIntent = this.vertexRenderIntent; │ │ │ │ │ + this.layer.addFeatures([this.radiusHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the control and all handlers. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * type - {String} Kind of node to draw. │ │ │ │ │ - * id - {String} Id for node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A new node of the given type and id. │ │ │ │ │ - * This function must be overridden by subclasses. │ │ │ │ │ + * map - {<OpenLayers.Map>} The control's map. │ │ │ │ │ */ │ │ │ │ │ - createNode: function(type, id) {}, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.handlers.drag.setMap(map); │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveRoot │ │ │ │ │ - * moves this renderer's root to a different renderer. │ │ │ │ │ + * Method: handleMapEvents │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * renderer - {<OpenLayers.Renderer>} target renderer for the moved root │ │ │ │ │ + * evt - {Object} │ │ │ │ │ */ │ │ │ │ │ - moveRoot: function(renderer) { │ │ │ │ │ - var root = this.root; │ │ │ │ │ - if (renderer.root.parentNode == this.rendererRoot) { │ │ │ │ │ - root = renderer.root; │ │ │ │ │ + handleMapEvents: function(evt) { │ │ │ │ │ + if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ + this.moveLayerToTop(); │ │ │ │ │ } │ │ │ │ │ - root.parentNode.removeChild(root); │ │ │ │ │ - renderer.rendererRoot.appendChild(root); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getRenderLayerId │ │ │ │ │ - * Gets the layer that this renderer's output appears on. If moveRoot was │ │ │ │ │ - * used, this will be different from the id of the layer containing the │ │ │ │ │ - * features rendered by this renderer. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} the id of the output layer. │ │ │ │ │ + * Method: moveLayerToTop │ │ │ │ │ + * Moves the layer for this handler to the top, so mouse events can reach │ │ │ │ │ + * it. │ │ │ │ │ */ │ │ │ │ │ - getRenderLayerId: function() { │ │ │ │ │ - return this.root.parentNode.parentNode.id; │ │ │ │ │ + moveLayerToTop: function() { │ │ │ │ │ + var index = Math.max(this.map.Z_INDEX_BASE['Feature'] - 1, │ │ │ │ │ + this.layer.getZIndex()) + 1; │ │ │ │ │ + this.layer.setZIndex(index); │ │ │ │ │ + │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: isComplexSymbol │ │ │ │ │ - * Determines if a symbol cannot be rendered using drawCircle │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * graphicName - {String} │ │ │ │ │ - * │ │ │ │ │ - * Returns │ │ │ │ │ - * {Boolean} true if the symbol is complex, false if not │ │ │ │ │ + * Method: moveLayerBack │ │ │ │ │ + * Moves the layer back to the position determined by the map's layers │ │ │ │ │ + * array. │ │ │ │ │ */ │ │ │ │ │ - isComplexSymbol: function(graphicName) { │ │ │ │ │ - return (graphicName != "circle") && !!graphicName; │ │ │ │ │ + moveLayerBack: function() { │ │ │ │ │ + var index = this.layer.getZIndex() - 1; │ │ │ │ │ + if (index >= this.map.Z_INDEX_BASE['Feature']) { │ │ │ │ │ + this.layer.setZIndex(index); │ │ │ │ │ + } else { │ │ │ │ │ + this.map.setLayerZIndex(this.layer, │ │ │ │ │ + this.map.getLayerIndex(this.layer)); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.Elements" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ModifyFeature" │ │ │ │ │ }); │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * Constant: RESHAPE │ │ │ │ │ + * {Integer} Constant used to make the control work in reshape mode │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.ModifyFeature.RESHAPE = 1; │ │ │ │ │ +/** │ │ │ │ │ + * Constant: RESIZE │ │ │ │ │ + * {Integer} Constant used to make the control work in resize mode │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.ModifyFeature.RESIZE = 2; │ │ │ │ │ +/** │ │ │ │ │ + * Constant: ROTATE │ │ │ │ │ + * {Integer} Constant used to make the control work in rotate mode │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.ModifyFeature.ROTATE = 4; │ │ │ │ │ +/** │ │ │ │ │ + * Constant: DRAG │ │ │ │ │ + * {Integer} Constant used to make the control work in drag mode │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.ModifyFeature.DRAG = 8; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Renderer/SVG.js │ │ │ │ │ + OpenLayers/Events/buttonclick.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Renderer/Elements.js │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Renderer.SVG │ │ │ │ │ - * │ │ │ │ │ - * Inherits: │ │ │ │ │ - * - <OpenLayers.Renderer.Elements> │ │ │ │ │ + * Class: OpenLayers.Events.buttonclick │ │ │ │ │ + * Extension event type for handling buttons on top of a dom element. This │ │ │ │ │ + * event type fires "buttonclick" on its <target> when a button was │ │ │ │ │ + * clicked. Buttons are detected by the "olButton" class. │ │ │ │ │ + * │ │ │ │ │ + * This event type makes sure that button clicks do not interfere with other │ │ │ │ │ + * events that are registered on the same <element>. │ │ │ │ │ + * │ │ │ │ │ + * Event types provided by this extension: │ │ │ │ │ + * - *buttonclick* Triggered when a button is clicked. Listeners receive an │ │ │ │ │ + * object with a *buttonElement* property referencing the dom element of │ │ │ │ │ + * the clicked button, and an *buttonXY* property with the click position │ │ │ │ │ + * relative to the button. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Renderer.SVG = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: xmlns │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - xmlns: "http://www.w3.org/2000/svg", │ │ │ │ │ +OpenLayers.Events.buttonclick = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: xlinkns │ │ │ │ │ - * {String} │ │ │ │ │ + * Property: target │ │ │ │ │ + * {<OpenLayers.Events>} The events instance that the buttonclick event will │ │ │ │ │ + * be triggered on. │ │ │ │ │ */ │ │ │ │ │ - xlinkns: "http://www.w3.org/1999/xlink", │ │ │ │ │ + target: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: MAX_PIXEL │ │ │ │ │ - * {Integer} Firefox has a limitation where values larger or smaller than │ │ │ │ │ - * about 15000 in an SVG document lock the browser up. This │ │ │ │ │ - * works around it. │ │ │ │ │ + * Property: events │ │ │ │ │ + * {Array} Events to observe and conditionally stop from propagating when │ │ │ │ │ + * an element with the olButton class (or its olAlphaImg child) is │ │ │ │ │ + * clicked. │ │ │ │ │ */ │ │ │ │ │ - MAX_PIXEL: 15000, │ │ │ │ │ + events: [ │ │ │ │ │ + 'mousedown', 'mouseup', 'click', 'dblclick', │ │ │ │ │ + 'touchstart', 'touchmove', 'touchend', 'keydown' │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: translationParameters │ │ │ │ │ - * {Object} Hash with "x" and "y" properties │ │ │ │ │ + * Property: startRegEx │ │ │ │ │ + * {RegExp} Regular expression to test Event.type for events that start │ │ │ │ │ + * a buttonclick sequence. │ │ │ │ │ */ │ │ │ │ │ - translationParameters: null, │ │ │ │ │ + startRegEx: /^mousedown|touchstart$/, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: symbolMetrics │ │ │ │ │ - * {Object} Cache for symbol metrics according to their svg coordinate │ │ │ │ │ - * space. This is an object keyed by the symbol's id, and values are │ │ │ │ │ - * an array of [width, centerX, centerY]. │ │ │ │ │ + * Property: cancelRegEx │ │ │ │ │ + * {RegExp} Regular expression to test Event.type for events that cancel │ │ │ │ │ + * a buttonclick sequence. │ │ │ │ │ */ │ │ │ │ │ - symbolMetrics: null, │ │ │ │ │ + cancelRegEx: /^touchmove$/, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Renderer.SVG │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * containerID - {String} │ │ │ │ │ + * Property: completeRegEx │ │ │ │ │ + * {RegExp} Regular expression to test Event.type for events that complete │ │ │ │ │ + * a buttonclick sequence. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(containerID) { │ │ │ │ │ - if (!this.supported()) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.initialize.apply(this, │ │ │ │ │ - arguments); │ │ │ │ │ - this.translationParameters = { │ │ │ │ │ - x: 0, │ │ │ │ │ - y: 0 │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - this.symbolMetrics = {}; │ │ │ │ │ - }, │ │ │ │ │ + completeRegEx: /^mouseup|touchend$/, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: supported │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the browser supports the SVG renderer │ │ │ │ │ + * Property: startEvt │ │ │ │ │ + * {Event} The event that started the click sequence │ │ │ │ │ */ │ │ │ │ │ - supported: function() { │ │ │ │ │ - var svgFeature = "http://www.w3.org/TR/SVG11/feature#"; │ │ │ │ │ - return (document.implementation && │ │ │ │ │ - (document.implementation.hasFeature("org.w3c.svg", "1.0") || │ │ │ │ │ - document.implementation.hasFeature(svgFeature + "SVG", "1.1") || │ │ │ │ │ - document.implementation.hasFeature(svgFeature + "BasicStructure", "1.1"))); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: inValidRange │ │ │ │ │ - * See #669 for more information │ │ │ │ │ + * Constructor: OpenLayers.Events.buttonclick │ │ │ │ │ + * Construct a buttonclick event type. Applications are not supposed to │ │ │ │ │ + * create instances of this class - they are created on demand by │ │ │ │ │ + * <OpenLayers.Events> instances. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * x - {Integer} │ │ │ │ │ - * y - {Integer} │ │ │ │ │ - * xyOnly - {Boolean} whether or not to just check for x and y, which means │ │ │ │ │ - * to not take the current translation parameters into account if true. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the 'x' and 'y' coordinates are in the │ │ │ │ │ - * valid range. │ │ │ │ │ + * target - {<OpenLayers.Events>} The events instance that the buttonclick │ │ │ │ │ + * event will be triggered on. │ │ │ │ │ */ │ │ │ │ │ - inValidRange: function(x, y, xyOnly) { │ │ │ │ │ - var left = x + (xyOnly ? 0 : this.translationParameters.x); │ │ │ │ │ - var top = y + (xyOnly ? 0 : this.translationParameters.y); │ │ │ │ │ - return (left >= -this.MAX_PIXEL && left <= this.MAX_PIXEL && │ │ │ │ │ - top >= -this.MAX_PIXEL && top <= this.MAX_PIXEL); │ │ │ │ │ + initialize: function(target) { │ │ │ │ │ + this.target = target; │ │ │ │ │ + for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ + this.target.register(this.events[i], this, this.buttonClick, { │ │ │ │ │ + extension: true │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setExtent │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} │ │ │ │ │ - * resolutionChanged - {Boolean} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ - * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ - * False otherwise. │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - var resolution = this.getResolution(), │ │ │ │ │ - left = -extent.left / resolution, │ │ │ │ │ - top = extent.top / resolution; │ │ │ │ │ - │ │ │ │ │ - // If the resolution has changed, start over changing the corner, because │ │ │ │ │ - // the features will redraw. │ │ │ │ │ - if (resolutionChanged) { │ │ │ │ │ - this.left = left; │ │ │ │ │ - this.top = top; │ │ │ │ │ - // Set the viewbox │ │ │ │ │ - var extentString = "0 0 " + this.size.w + " " + this.size.h; │ │ │ │ │ - │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "viewBox", extentString); │ │ │ │ │ - this.translate(this.xOffset, 0); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - var inRange = this.translate(left - this.left + this.xOffset, top - this.top); │ │ │ │ │ - if (!inRange) { │ │ │ │ │ - // recenter the coordinate system │ │ │ │ │ - this.setExtent(extent, true); │ │ │ │ │ - } │ │ │ │ │ - return coordSysUnchanged && inRange; │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ + this.target.unregister(this.events[i], this, this.buttonClick); │ │ │ │ │ } │ │ │ │ │ + delete this.target; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: translate │ │ │ │ │ - * Transforms the SVG coordinate system │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {Float} │ │ │ │ │ - * y - {Float} │ │ │ │ │ - * │ │ │ │ │ + * Method: getPressedButton │ │ │ │ │ + * Get the pressed button, if any. Returns undefined if no button │ │ │ │ │ + * was pressed. │ │ │ │ │ + * │ │ │ │ │ + * Arguments: │ │ │ │ │ + * element - {DOMElement} The event target. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} true if the translation parameters are in the valid coordinates │ │ │ │ │ - * range, false otherwise. │ │ │ │ │ + * {DOMElement} The button element, or undefined. │ │ │ │ │ */ │ │ │ │ │ - translate: function(x, y) { │ │ │ │ │ - if (!this.inValidRange(x, y, true)) { │ │ │ │ │ - return false; │ │ │ │ │ - } else { │ │ │ │ │ - var transformString = ""; │ │ │ │ │ - if (x || y) { │ │ │ │ │ - transformString = "translate(" + x + "," + y + ")"; │ │ │ │ │ + getPressedButton: function(element) { │ │ │ │ │ + var depth = 3, // limit the search depth │ │ │ │ │ + button; │ │ │ │ │ + do { │ │ │ │ │ + if (OpenLayers.Element.hasClass(element, "olButton")) { │ │ │ │ │ + // hit! │ │ │ │ │ + button = element; │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ - this.root.setAttributeNS(null, "transform", transformString); │ │ │ │ │ - this.translationParameters = { │ │ │ │ │ - x: x, │ │ │ │ │ - y: y │ │ │ │ │ - }; │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ + element = element.parentNode; │ │ │ │ │ + } while (--depth > 0 && element); │ │ │ │ │ + return button; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setSize │ │ │ │ │ - * Sets the size of the drawing surface. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>} The size of the drawing surface │ │ │ │ │ - */ │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "width", this.size.w); │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "height", this.size.h); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getNodeType │ │ │ │ │ - * │ │ │ │ │ + * Method: ignore │ │ │ │ │ + * Check for event target elements that should be ignored by OpenLayers. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The corresponding node type for the specified geometry │ │ │ │ │ + * element - {DOMElement} The event target. │ │ │ │ │ */ │ │ │ │ │ - getNodeType: function(geometry, style) { │ │ │ │ │ - var nodeType = null; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - nodeType = "image"; │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - nodeType = "svg"; │ │ │ │ │ - } else { │ │ │ │ │ - nodeType = "circle"; │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - nodeType = "rect"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - nodeType = "polyline"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - nodeType = "polygon"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - case "OpenLayers.Geometry.Curve": │ │ │ │ │ - nodeType = "path"; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ + ignore: function(element) { │ │ │ │ │ + var depth = 3, │ │ │ │ │ + ignore = false; │ │ │ │ │ + do { │ │ │ │ │ + if (element.nodeName.toLowerCase() === 'a') { │ │ │ │ │ + ignore = true; │ │ │ │ │ break; │ │ │ │ │ - } │ │ │ │ │ - return nodeType; │ │ │ │ │ + } │ │ │ │ │ + element = element.parentNode; │ │ │ │ │ + } while (--depth > 0 && element); │ │ │ │ │ + return ignore; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setStyle │ │ │ │ │ - * Use to set all the style attributes to a SVG node. │ │ │ │ │ - * │ │ │ │ │ - * Takes care to adjust stroke width and point radius to be │ │ │ │ │ - * resolution-relative │ │ │ │ │ + /** │ │ │ │ │ + * Method: buttonClick │ │ │ │ │ + * Check if a button was clicked, and fire the buttonclick event │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {SVGDomElement} An SVG element to decorate │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * options - {Object} Currently supported options include │ │ │ │ │ - * 'isFilled' {Boolean} and │ │ │ │ │ - * 'isStroked' {Boolean} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - setStyle: function(node, style, options) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - options = options || node._options; │ │ │ │ │ - │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - node.setAttributeNS(null, "title", title); │ │ │ │ │ - //Standards-conformant SVG │ │ │ │ │ - // Prevent duplicate nodes. See issue https://github.com/openlayers/openlayers/issues/92 │ │ │ │ │ - var titleNode = node.getElementsByTagName("title"); │ │ │ │ │ - if (titleNode.length > 0) { │ │ │ │ │ - titleNode[0].firstChild.textContent = title; │ │ │ │ │ - } else { │ │ │ │ │ - var label = this.nodeFactory(null, "title"); │ │ │ │ │ - label.textContent = title; │ │ │ │ │ - node.appendChild(label); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var r = parseFloat(node.getAttributeNS(null, "r")); │ │ │ │ │ - var widthFactor = 1; │ │ │ │ │ - var pos; │ │ │ │ │ - if (node._geometryClass == "OpenLayers.Geometry.Point" && r) { │ │ │ │ │ - node.style.visibility = ""; │ │ │ │ │ - if (style.graphic === false) { │ │ │ │ │ - node.style.visibility = "hidden"; │ │ │ │ │ - } else if (style.externalGraphic) { │ │ │ │ │ - pos = this.getPosition(node); │ │ │ │ │ - if (style.graphicWidth && style.graphicHeight) { │ │ │ │ │ - node.setAttributeNS(null, "preserveAspectRatio", "none"); │ │ │ │ │ - } │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ - style.graphicXOffset : -(0.5 * width); │ │ │ │ │ - var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ - style.graphicYOffset : -(0.5 * height); │ │ │ │ │ - │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ - │ │ │ │ │ - node.setAttributeNS(null, "x", (pos.x + xOffset).toFixed()); │ │ │ │ │ - node.setAttributeNS(null, "y", (pos.y + yOffset).toFixed()); │ │ │ │ │ - node.setAttributeNS(null, "width", width); │ │ │ │ │ - node.setAttributeNS(null, "height", height); │ │ │ │ │ - node.setAttributeNS(this.xlinkns, "xlink:href", style.externalGraphic); │ │ │ │ │ - node.setAttributeNS(null, "style", "opacity: " + opacity); │ │ │ │ │ - node.onclick = OpenLayers.Event.preventDefault; │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - // the symbol viewBox is three times as large as the symbol │ │ │ │ │ - var offset = style.pointRadius * 3; │ │ │ │ │ - var size = offset * 2; │ │ │ │ │ - var src = this.importSymbol(style.graphicName); │ │ │ │ │ - pos = this.getPosition(node); │ │ │ │ │ - widthFactor = this.symbolMetrics[src.id][0] * 3 / size; │ │ │ │ │ + buttonClick: function(evt) { │ │ │ │ │ + var propagate = true, │ │ │ │ │ + element = OpenLayers.Event.element(evt); │ │ │ │ │ + if (element && (OpenLayers.Event.isLeftClick(evt) || !~evt.type.indexOf("mouse"))) { │ │ │ │ │ + // was a button pressed? │ │ │ │ │ + var button = this.getPressedButton(element); │ │ │ │ │ + if (button) { │ │ │ │ │ + if (evt.type === "keydown") { │ │ │ │ │ + switch (evt.keyCode) { │ │ │ │ │ + case OpenLayers.Event.KEY_RETURN: │ │ │ │ │ + case OpenLayers.Event.KEY_SPACE: │ │ │ │ │ + this.target.triggerEvent("buttonclick", { │ │ │ │ │ + buttonElement: button │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } else if (this.startEvt) { │ │ │ │ │ + if (this.completeRegEx.test(evt.type)) { │ │ │ │ │ + var pos = OpenLayers.Util.pagePosition(button); │ │ │ │ │ + var viewportElement = OpenLayers.Util.getViewportElement(); │ │ │ │ │ + var scrollTop = window.pageYOffset || viewportElement.scrollTop; │ │ │ │ │ + var scrollLeft = window.pageXOffset || viewportElement.scrollLeft; │ │ │ │ │ + pos[0] = pos[0] - scrollLeft; │ │ │ │ │ + pos[1] = pos[1] - scrollTop; │ │ │ │ │ │ │ │ │ │ - // remove the node from the dom before we modify it. This │ │ │ │ │ - // prevents various rendering issues in Safari and FF │ │ │ │ │ - var parent = node.parentNode; │ │ │ │ │ - var nextSibling = node.nextSibling; │ │ │ │ │ - if (parent) { │ │ │ │ │ - parent.removeChild(node); │ │ │ │ │ + this.target.triggerEvent("buttonclick", { │ │ │ │ │ + buttonElement: button, │ │ │ │ │ + buttonXY: { │ │ │ │ │ + x: this.startEvt.clientX - pos[0], │ │ │ │ │ + y: this.startEvt.clientY - pos[1] │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + if (this.cancelRegEx.test(evt.type)) { │ │ │ │ │ + delete this.startEvt; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // The more appropriate way to implement this would be use/defs, │ │ │ │ │ - // but due to various issues in several browsers, it is safer to │ │ │ │ │ - // copy the symbols instead of referencing them. │ │ │ │ │ - // See e.g. ticket http://trac.osgeo.org/openlayers/ticket/2985 │ │ │ │ │ - // and this email thread │ │ │ │ │ - // http://osgeo-org.1803224.n2.nabble.com/Select-Control-Ctrl-click-on-Feature-with-a-graphicName-opens-new-browser-window-tc5846039.html │ │ │ │ │ - node.firstChild && node.removeChild(node.firstChild); │ │ │ │ │ - node.appendChild(src.firstChild.cloneNode(true)); │ │ │ │ │ - node.setAttributeNS(null, "viewBox", src.getAttributeNS(null, "viewBox")); │ │ │ │ │ - │ │ │ │ │ - node.setAttributeNS(null, "width", size); │ │ │ │ │ - node.setAttributeNS(null, "height", size); │ │ │ │ │ - node.setAttributeNS(null, "x", pos.x - offset); │ │ │ │ │ - node.setAttributeNS(null, "y", pos.y - offset); │ │ │ │ │ - │ │ │ │ │ - // now that the node has all its new properties, insert it │ │ │ │ │ - // back into the dom where it was │ │ │ │ │ - if (nextSibling) { │ │ │ │ │ - parent.insertBefore(node, nextSibling); │ │ │ │ │ - } else if (parent) { │ │ │ │ │ - parent.appendChild(node); │ │ │ │ │ + if (this.startRegEx.test(evt.type)) { │ │ │ │ │ + this.startEvt = evt; │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false; │ │ │ │ │ } │ │ │ │ │ } else { │ │ │ │ │ - node.setAttributeNS(null, "r", style.pointRadius); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var rotation = style.rotation; │ │ │ │ │ - │ │ │ │ │ - if ((rotation !== undefined || node._rotation !== undefined) && pos) { │ │ │ │ │ - node._rotation = rotation; │ │ │ │ │ - rotation |= 0; │ │ │ │ │ - if (node.nodeName !== "svg") { │ │ │ │ │ - node.setAttributeNS(null, "transform", │ │ │ │ │ - "rotate(" + rotation + " " + pos.x + " " + │ │ │ │ │ - pos.y + ")"); │ │ │ │ │ - } else { │ │ │ │ │ - var metrics = this.symbolMetrics[src.id]; │ │ │ │ │ - node.firstChild.setAttributeNS(null, "transform", "rotate(" + │ │ │ │ │ - rotation + " " + │ │ │ │ │ - metrics[1] + " " + │ │ │ │ │ - metrics[2] + ")"); │ │ │ │ │ - } │ │ │ │ │ + propagate = !this.ignore(OpenLayers.Event.element(evt)); │ │ │ │ │ + delete this.startEvt; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return propagate; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (options.isFilled) { │ │ │ │ │ - node.setAttributeNS(null, "fill", style.fillColor); │ │ │ │ │ - node.setAttributeNS(null, "fill-opacity", style.fillOpacity); │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "fill", "none"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (options.isStroked) { │ │ │ │ │ - node.setAttributeNS(null, "stroke", style.strokeColor); │ │ │ │ │ - node.setAttributeNS(null, "stroke-opacity", style.strokeOpacity); │ │ │ │ │ - node.setAttributeNS(null, "stroke-width", style.strokeWidth * widthFactor); │ │ │ │ │ - node.setAttributeNS(null, "stroke-linecap", style.strokeLinecap || "round"); │ │ │ │ │ - // Hard-coded linejoin for now, to make it look the same as in VML. │ │ │ │ │ - // There is no strokeLinejoin property yet for symbolizers. │ │ │ │ │ - node.setAttributeNS(null, "stroke-linejoin", "round"); │ │ │ │ │ - style.strokeDashstyle && node.setAttributeNS(null, │ │ │ │ │ - "stroke-dasharray", this.dashStyle(style, widthFactor)); │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "stroke", "none"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (style.pointerEvents) { │ │ │ │ │ - node.setAttributeNS(null, "pointer-events", style.pointerEvents); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (style.cursor != null) { │ │ │ │ │ - node.setAttributeNS(null, "cursor", style.cursor); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: dashStyle │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * widthFactor - {Number} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A SVG compliant 'stroke-dasharray' value │ │ │ │ │ - */ │ │ │ │ │ - dashStyle: function(style, widthFactor) { │ │ │ │ │ - var w = style.strokeWidth * widthFactor; │ │ │ │ │ - var str = style.strokeDashstyle; │ │ │ │ │ - switch (str) { │ │ │ │ │ - case 'solid': │ │ │ │ │ - return 'none'; │ │ │ │ │ - case 'dot': │ │ │ │ │ - return [1, 4 * w].join(); │ │ │ │ │ - case 'dash': │ │ │ │ │ - return [4 * w, 4 * w].join(); │ │ │ │ │ - case 'dashdot': │ │ │ │ │ - return [4 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ - case 'longdash': │ │ │ │ │ - return [8 * w, 4 * w].join(); │ │ │ │ │ - case 'longdashdot': │ │ │ │ │ - return [8 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ - default: │ │ │ │ │ - return OpenLayers.String.trim(str).replace(/\s+/g, ","); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/Zoom.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createNode │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * type - {String} Kind of node to draw │ │ │ │ │ - * id - {String} Id for node │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A new node of the given type and id │ │ │ │ │ - */ │ │ │ │ │ - createNode: function(type, id) { │ │ │ │ │ - var node = document.createElementNS(this.xmlns, type); │ │ │ │ │ - if (id) { │ │ │ │ │ - node.setAttributeNS(null, "id", id); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: nodeTypeCompare │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {SVGDomElement} An SVG element │ │ │ │ │ - * type - {String} Kind of node │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ - */ │ │ │ │ │ - nodeTypeCompare: function(node, type) { │ │ │ │ │ - return (type == node.nodeName); │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createRenderRoot │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The specific render engine's root element │ │ │ │ │ - */ │ │ │ │ │ - createRenderRoot: function() { │ │ │ │ │ - var svg = this.nodeFactory(this.container.id + "_svgRoot", "svg"); │ │ │ │ │ - svg.style.display = "block"; │ │ │ │ │ - return svg; │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.Zoom │ │ │ │ │ + * The Zoom control is a pair of +/- links for zooming in and out. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.Zoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createRoot │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * suffix - {String} suffix to append to the id │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * APIProperty: zoomInText │ │ │ │ │ + * {String} │ │ │ │ │ + * Text for zoom-in link. Default is "+". │ │ │ │ │ */ │ │ │ │ │ - createRoot: function(suffix) { │ │ │ │ │ - return this.nodeFactory(this.container.id + suffix, "g"); │ │ │ │ │ - }, │ │ │ │ │ + zoomInText: "+", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createDefs │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The element to which we'll add the symbol definitions │ │ │ │ │ + * APIProperty: zoomInId │ │ │ │ │ + * {String} │ │ │ │ │ + * Instead of having the control create a zoom in link, you can provide │ │ │ │ │ + * the identifier for an anchor element already added to the document. │ │ │ │ │ + * By default, an element with id "olZoomInLink" will be searched for │ │ │ │ │ + * and used if it exists. │ │ │ │ │ */ │ │ │ │ │ - createDefs: function() { │ │ │ │ │ - var defs = this.nodeFactory(this.container.id + "_defs", "defs"); │ │ │ │ │ - this.rendererRoot.appendChild(defs); │ │ │ │ │ - return defs; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /************************************** │ │ │ │ │ - * * │ │ │ │ │ - * GEOMETRY DRAWING FUNCTIONS * │ │ │ │ │ - * * │ │ │ │ │ - **************************************/ │ │ │ │ │ + zoomInId: "olZoomInLink", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPoint │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the point │ │ │ │ │ + * APIProperty: zoomOutText │ │ │ │ │ + * {String} │ │ │ │ │ + * Text for zoom-out link. Default is "\u2212". │ │ │ │ │ */ │ │ │ │ │ - drawPoint: function(node, geometry) { │ │ │ │ │ - return this.drawCircle(node, geometry, 1); │ │ │ │ │ - }, │ │ │ │ │ + zoomOutText: "\u2212", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawCircle │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * radius - {Float} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the circle │ │ │ │ │ + * APIProperty: zoomOutId │ │ │ │ │ + * {String} │ │ │ │ │ + * Instead of having the control create a zoom out link, you can provide │ │ │ │ │ + * the identifier for an anchor element already added to the document. │ │ │ │ │ + * By default, an element with id "olZoomOutLink" will be searched for │ │ │ │ │ + * and used if it exists. │ │ │ │ │ */ │ │ │ │ │ - drawCircle: function(node, geometry, radius) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = ((geometry.x - this.featureDx) / resolution + this.left); │ │ │ │ │ - var y = (this.top - geometry.y / resolution); │ │ │ │ │ - │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - node.setAttributeNS(null, "cx", x); │ │ │ │ │ - node.setAttributeNS(null, "cy", y); │ │ │ │ │ - node.setAttributeNS(null, "r", radius); │ │ │ │ │ - return node; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ + zoomOutId: "olZoomOutLink", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLineString │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * Method: draw │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components of │ │ │ │ │ - * the linestring, or false if nothing could be drawn │ │ │ │ │ + * {DOMElement} A reference to the DOMElement containing the zoom links. │ │ │ │ │ */ │ │ │ │ │ - drawLineString: function(node, geometry) { │ │ │ │ │ - var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ - if (componentsResult.path) { │ │ │ │ │ - node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ - return (componentsResult.complete ? node : null); │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + var div = OpenLayers.Control.prototype.draw.apply(this), │ │ │ │ │ + links = this.getOrCreateLinks(div), │ │ │ │ │ + zoomIn = links.zoomIn, │ │ │ │ │ + zoomOut = links.zoomOut, │ │ │ │ │ + eventsInstance = this.map.events; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawLinearRing │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ - * of the linear ring, or false if nothing could be drawn │ │ │ │ │ - */ │ │ │ │ │ - drawLinearRing: function(node, geometry) { │ │ │ │ │ - var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ - if (componentsResult.path) { │ │ │ │ │ - node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ - return (componentsResult.complete ? node : null); │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ + if (zoomOut.parentNode !== div) { │ │ │ │ │ + eventsInstance = this.events; │ │ │ │ │ + eventsInstance.attachToElement(zoomOut.parentNode); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ + eventsInstance.register("buttonclick", this, this.onZoomClick); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawPolygon │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ - * of the polygon, or false if nothing could be drawn │ │ │ │ │ - */ │ │ │ │ │ - drawPolygon: function(node, geometry) { │ │ │ │ │ - var d = ""; │ │ │ │ │ - var draw = true; │ │ │ │ │ - var complete = true; │ │ │ │ │ - var linearRingResult, path; │ │ │ │ │ - for (var j = 0, len = geometry.components.length; j < len; j++) { │ │ │ │ │ - d += " M"; │ │ │ │ │ - linearRingResult = this.getComponentsString( │ │ │ │ │ - geometry.components[j].components, " "); │ │ │ │ │ - path = linearRingResult.path; │ │ │ │ │ - if (path) { │ │ │ │ │ - d += " " + path; │ │ │ │ │ - complete = linearRingResult.complete && complete; │ │ │ │ │ - } else { │ │ │ │ │ - draw = false; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - d += " z"; │ │ │ │ │ - if (draw) { │ │ │ │ │ - node.setAttributeNS(null, "d", d); │ │ │ │ │ - node.setAttributeNS(null, "fill-rule", "evenodd"); │ │ │ │ │ - return complete ? node : null; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ + this.zoomInLink = zoomIn; │ │ │ │ │ + this.zoomOutLink = zoomOut; │ │ │ │ │ + return div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawRectangle │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * Method: getOrCreateLinks │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the rectangle │ │ │ │ │ - */ │ │ │ │ │ - drawRectangle: function(node, geometry) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = ((geometry.x - this.featureDx) / resolution + this.left); │ │ │ │ │ - var y = (this.top - geometry.y / resolution); │ │ │ │ │ - │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - node.setAttributeNS(null, "x", x); │ │ │ │ │ - node.setAttributeNS(null, "y", y); │ │ │ │ │ - node.setAttributeNS(null, "width", geometry.width / resolution); │ │ │ │ │ - node.setAttributeNS(null, "height", geometry.height / resolution); │ │ │ │ │ - return node; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawText │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * style - │ │ │ │ │ - * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * el - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Return: │ │ │ │ │ + * {Object} Object with zoomIn and zoomOut properties referencing links. │ │ │ │ │ */ │ │ │ │ │ - drawText: function(featureId, style, location) { │ │ │ │ │ - var drawOutline = (!!style.labelOutlineWidth); │ │ │ │ │ - // First draw text in halo color and size and overlay the │ │ │ │ │ - // normal text afterwards │ │ │ │ │ - if (drawOutline) { │ │ │ │ │ - var outlineStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ - outlineStyle.fontColor = outlineStyle.labelOutlineColor; │ │ │ │ │ - outlineStyle.fontStrokeColor = outlineStyle.labelOutlineColor; │ │ │ │ │ - outlineStyle.fontStrokeWidth = style.labelOutlineWidth; │ │ │ │ │ - if (style.labelOutlineOpacity) { │ │ │ │ │ - outlineStyle.fontOpacity = style.labelOutlineOpacity; │ │ │ │ │ - } │ │ │ │ │ - delete outlineStyle.labelOutlineWidth; │ │ │ │ │ - this.drawText(featureId, outlineStyle, location); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - │ │ │ │ │ - var x = ((location.x - this.featureDx) / resolution + this.left); │ │ │ │ │ - var y = (location.y / resolution - this.top); │ │ │ │ │ - │ │ │ │ │ - var suffix = (drawOutline) ? this.LABEL_OUTLINE_SUFFIX : this.LABEL_ID_SUFFIX; │ │ │ │ │ - var label = this.nodeFactory(featureId + suffix, "text"); │ │ │ │ │ - │ │ │ │ │ - label.setAttributeNS(null, "x", x); │ │ │ │ │ - label.setAttributeNS(null, "y", -y); │ │ │ │ │ - │ │ │ │ │ - if (style.fontColor) { │ │ │ │ │ - label.setAttributeNS(null, "fill", style.fontColor); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStrokeColor) { │ │ │ │ │ - label.setAttributeNS(null, "stroke", style.fontStrokeColor); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStrokeWidth) { │ │ │ │ │ - label.setAttributeNS(null, "stroke-width", style.fontStrokeWidth); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontOpacity) { │ │ │ │ │ - label.setAttributeNS(null, "opacity", style.fontOpacity); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontFamily) { │ │ │ │ │ - label.setAttributeNS(null, "font-family", style.fontFamily); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontSize) { │ │ │ │ │ - label.setAttributeNS(null, "font-size", style.fontSize); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontWeight) { │ │ │ │ │ - label.setAttributeNS(null, "font-weight", style.fontWeight); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStyle) { │ │ │ │ │ - label.setAttributeNS(null, "font-style", style.fontStyle); │ │ │ │ │ - } │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - label.setAttributeNS(null, "pointer-events", "visible"); │ │ │ │ │ - label._featureId = featureId; │ │ │ │ │ - } else { │ │ │ │ │ - label.setAttributeNS(null, "pointer-events", "none"); │ │ │ │ │ - } │ │ │ │ │ - var align = style.labelAlign || OpenLayers.Renderer.defaultSymbolizer.labelAlign; │ │ │ │ │ - label.setAttributeNS(null, "text-anchor", │ │ │ │ │ - OpenLayers.Renderer.SVG.LABEL_ALIGN[align[0]] || "middle"); │ │ │ │ │ - │ │ │ │ │ - if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ - label.setAttributeNS(null, "dominant-baseline", │ │ │ │ │ - OpenLayers.Renderer.SVG.LABEL_ALIGN[align[1]] || "central"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var labelRows = style.label.split('\n'); │ │ │ │ │ - var numRows = labelRows.length; │ │ │ │ │ - while (label.childNodes.length > numRows) { │ │ │ │ │ - label.removeChild(label.lastChild); │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - var tspan = this.nodeFactory(featureId + suffix + "_tspan_" + i, "tspan"); │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - tspan._featureId = featureId; │ │ │ │ │ - tspan._geometry = location; │ │ │ │ │ - tspan._geometryClass = location.CLASS_NAME; │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.IS_GECKO === false) { │ │ │ │ │ - tspan.setAttributeNS(null, "baseline-shift", │ │ │ │ │ - OpenLayers.Renderer.SVG.LABEL_VSHIFT[align[1]] || "-35%"); │ │ │ │ │ - } │ │ │ │ │ - tspan.setAttribute("x", x); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - var vfactor = OpenLayers.Renderer.SVG.LABEL_VFACTOR[align[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5; │ │ │ │ │ - } │ │ │ │ │ - tspan.setAttribute("dy", (vfactor * (numRows - 1)) + "em"); │ │ │ │ │ - } else { │ │ │ │ │ - tspan.setAttribute("dy", "1em"); │ │ │ │ │ - } │ │ │ │ │ - tspan.textContent = (labelRows[i] === '') ? ' ' : labelRows[i]; │ │ │ │ │ - if (!tspan.parentNode) { │ │ │ │ │ - label.appendChild(tspan); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (!label.parentNode) { │ │ │ │ │ - this.textRoot.appendChild(label); │ │ │ │ │ + getOrCreateLinks: function(el) { │ │ │ │ │ + var zoomIn = document.getElementById(this.zoomInId), │ │ │ │ │ + zoomOut = document.getElementById(this.zoomOutId); │ │ │ │ │ + if (!zoomIn) { │ │ │ │ │ + zoomIn = document.createElement("a"); │ │ │ │ │ + zoomIn.href = "#zoomIn"; │ │ │ │ │ + zoomIn.appendChild(document.createTextNode(this.zoomInText)); │ │ │ │ │ + zoomIn.className = "olControlZoomIn"; │ │ │ │ │ + el.appendChild(zoomIn); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getComponentString │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * components - {Array(<OpenLayers.Geometry.Point>)} Array of points │ │ │ │ │ - * separator - {String} character between coordinate pairs. Defaults to "," │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} hash with properties "path" (the string created from the │ │ │ │ │ - * components and "complete" (false if the renderer was unable to │ │ │ │ │ - * draw all components) │ │ │ │ │ - */ │ │ │ │ │ - getComponentsString: function(components, separator) { │ │ │ │ │ - var renderCmp = []; │ │ │ │ │ - var complete = true; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - var strings = []; │ │ │ │ │ - var str, component; │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - component = components[i]; │ │ │ │ │ - renderCmp.push(component); │ │ │ │ │ - str = this.getShortString(component); │ │ │ │ │ - if (str) { │ │ │ │ │ - strings.push(str); │ │ │ │ │ - } else { │ │ │ │ │ - // The current component is outside the valid range. Let's │ │ │ │ │ - // see if the previous or next component is inside the range. │ │ │ │ │ - // If so, add the coordinate of the intersection with the │ │ │ │ │ - // valid range bounds. │ │ │ │ │ - if (i > 0) { │ │ │ │ │ - if (this.getShortString(components[i - 1])) { │ │ │ │ │ - strings.push(this.clipLine(components[i], │ │ │ │ │ - components[i - 1])); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (i < len - 1) { │ │ │ │ │ - if (this.getShortString(components[i + 1])) { │ │ │ │ │ - strings.push(this.clipLine(components[i], │ │ │ │ │ - components[i + 1])); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - complete = false; │ │ │ │ │ - } │ │ │ │ │ + OpenLayers.Element.addClass(zoomIn, "olButton"); │ │ │ │ │ + if (!zoomOut) { │ │ │ │ │ + zoomOut = document.createElement("a"); │ │ │ │ │ + zoomOut.href = "#zoomOut"; │ │ │ │ │ + zoomOut.appendChild(document.createTextNode(this.zoomOutText)); │ │ │ │ │ + zoomOut.className = "olControlZoomOut"; │ │ │ │ │ + el.appendChild(zoomOut); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ + OpenLayers.Element.addClass(zoomOut, "olButton"); │ │ │ │ │ return { │ │ │ │ │ - path: strings.join(separator || ","), │ │ │ │ │ - complete: complete │ │ │ │ │ + zoomIn: zoomIn, │ │ │ │ │ + zoomOut: zoomOut │ │ │ │ │ }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clipLine │ │ │ │ │ - * Given two points (one inside the valid range, and one outside), │ │ │ │ │ - * clips the line betweeen the two points so that the new points are both │ │ │ │ │ - * inside the valid range. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * badComponent - {<OpenLayers.Geometry.Point>} original geometry of the │ │ │ │ │ - * invalid point │ │ │ │ │ - * goodComponent - {<OpenLayers.Geometry.Point>} original geometry of the │ │ │ │ │ - * valid point │ │ │ │ │ - * Returns │ │ │ │ │ - * {String} the SVG coordinate pair of the clipped point (like │ │ │ │ │ - * getShortString), or an empty string if both passed componets are at │ │ │ │ │ - * the same point. │ │ │ │ │ + * Method: onZoomClick │ │ │ │ │ + * Called when zoomin/out link is clicked. │ │ │ │ │ */ │ │ │ │ │ - clipLine: function(badComponent, goodComponent) { │ │ │ │ │ - if (goodComponent.equals(badComponent)) { │ │ │ │ │ - return ""; │ │ │ │ │ - } │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var maxX = this.MAX_PIXEL - this.translationParameters.x; │ │ │ │ │ - var maxY = this.MAX_PIXEL - this.translationParameters.y; │ │ │ │ │ - var x1 = (goodComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y1 = this.top - goodComponent.y / resolution; │ │ │ │ │ - var x2 = (badComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y2 = this.top - badComponent.y / resolution; │ │ │ │ │ - var k; │ │ │ │ │ - if (x2 < -maxX || x2 > maxX) { │ │ │ │ │ - k = (y2 - y1) / (x2 - x1); │ │ │ │ │ - x2 = x2 < 0 ? -maxX : maxX; │ │ │ │ │ - y2 = y1 + (x2 - x1) * k; │ │ │ │ │ - } │ │ │ │ │ - if (y2 < -maxY || y2 > maxY) { │ │ │ │ │ - k = (x2 - x1) / (y2 - y1); │ │ │ │ │ - y2 = y2 < 0 ? -maxY : maxY; │ │ │ │ │ - x2 = x1 + (y2 - y1) * k; │ │ │ │ │ + onZoomClick: function(evt) { │ │ │ │ │ + var button = evt.buttonElement; │ │ │ │ │ + if (button === this.zoomInLink) { │ │ │ │ │ + this.map.zoomIn(); │ │ │ │ │ + } else if (button === this.zoomOutLink) { │ │ │ │ │ + this.map.zoomOut(); │ │ │ │ │ } │ │ │ │ │ - return x2 + "," + y2; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getShortString │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} or false if point is outside the valid range │ │ │ │ │ - */ │ │ │ │ │ - getShortString: function(point) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = ((point.x - this.featureDx) / resolution + this.left); │ │ │ │ │ - var y = (this.top - point.y / resolution); │ │ │ │ │ - │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - return x + "," + y; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getPosition │ │ │ │ │ - * Finds the position of an svg node. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} hash with x and y properties, representing the coordinates │ │ │ │ │ - * within the svg coordinate system │ │ │ │ │ - */ │ │ │ │ │ - getPosition: function(node) { │ │ │ │ │ - return ({ │ │ │ │ │ - x: parseFloat(node.getAttributeNS(null, "cx")), │ │ │ │ │ - y: parseFloat(node.getAttributeNS(null, "cy")) │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: importSymbol │ │ │ │ │ - * add a new symbol definition from the rendererer's symbol hash │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * graphicName - {String} name of the symbol to import │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} - the imported symbol │ │ │ │ │ - */ │ │ │ │ │ - importSymbol: function(graphicName) { │ │ │ │ │ - if (!this.defs) { │ │ │ │ │ - // create svg defs tag │ │ │ │ │ - this.defs = this.createDefs(); │ │ │ │ │ - } │ │ │ │ │ - var id = this.container.id + "-" + graphicName; │ │ │ │ │ - │ │ │ │ │ - // check if symbol already exists in the defs │ │ │ │ │ - var existing = document.getElementById(id); │ │ │ │ │ - if (existing != null) { │ │ │ │ │ - return existing; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(graphicName + ' is not a valid symbol name'); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var symbolNode = this.nodeFactory(id, "symbol"); │ │ │ │ │ - var node = this.nodeFactory(null, "polygon"); │ │ │ │ │ - symbolNode.appendChild(node); │ │ │ │ │ - var symbolExtent = new OpenLayers.Bounds( │ │ │ │ │ - Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ - │ │ │ │ │ - var points = []; │ │ │ │ │ - var x, y; │ │ │ │ │ - for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ - symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ - symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ - symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ - points.push(x, ",", y); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - node.setAttributeNS(null, "points", points.join(" ")); │ │ │ │ │ - │ │ │ │ │ - var width = symbolExtent.getWidth(); │ │ │ │ │ - var height = symbolExtent.getHeight(); │ │ │ │ │ - // create a viewBox three times as large as the symbol itself, │ │ │ │ │ - // to allow for strokeWidth being displayed correctly at the corners. │ │ │ │ │ - var viewBox = [symbolExtent.left - width, │ │ │ │ │ - symbolExtent.bottom - height, width * 3, height * 3 │ │ │ │ │ - ]; │ │ │ │ │ - symbolNode.setAttributeNS(null, "viewBox", viewBox.join(" ")); │ │ │ │ │ - this.symbolMetrics[id] = [ │ │ │ │ │ - Math.max(width, height), │ │ │ │ │ - symbolExtent.getCenterLonLat().lon, │ │ │ │ │ - symbolExtent.getCenterLonLat().lat │ │ │ │ │ - ]; │ │ │ │ │ - │ │ │ │ │ - this.defs.appendChild(symbolNode); │ │ │ │ │ - return symbolNode; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getFeatureIdFromEvent │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} An <OpenLayers.Event> object │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A feature id or undefined. │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Clean up. │ │ │ │ │ */ │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var featureId = OpenLayers.Renderer.Elements.prototype.getFeatureIdFromEvent.apply(this, arguments); │ │ │ │ │ - if (!featureId) { │ │ │ │ │ - var target = evt.target; │ │ │ │ │ - featureId = target.parentNode && target != this.rendererRoot ? │ │ │ │ │ - target.parentNode._featureId : undefined; │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onZoomClick); │ │ │ │ │ } │ │ │ │ │ - return featureId; │ │ │ │ │ + delete this.zoomInLink; │ │ │ │ │ + delete this.zoomOutLink; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.SVG" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Zoom" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.SVG.LABEL_ALIGN │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_ALIGN = { │ │ │ │ │ - "l": "start", │ │ │ │ │ - "r": "end", │ │ │ │ │ - "b": "bottom", │ │ │ │ │ - "t": "hanging" │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.SVG.LABEL_VSHIFT │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_VSHIFT = { │ │ │ │ │ - // according to │ │ │ │ │ - // http://www.w3.org/Graphics/SVG/Test/20061213/htmlObjectHarness/full-text-align-02-b.html │ │ │ │ │ - // a baseline-shift of -70% shifts the text exactly from the │ │ │ │ │ - // bottom to the top of the baseline, so -35% moves the text to │ │ │ │ │ - // the center of the baseline. │ │ │ │ │ - "t": "-70%", │ │ │ │ │ - "b": "0" │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.SVG.LABEL_VFACTOR │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_VFACTOR = { │ │ │ │ │ - "t": 0, │ │ │ │ │ - "b": -1 │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Function: OpenLayers.Renderer.SVG.preventDefault │ │ │ │ │ - * *Deprecated*. Use <OpenLayers.Event.preventDefault> method instead. │ │ │ │ │ - * Used to prevent default events (especially opening images in a new tab on │ │ │ │ │ - * ctrl-click) from being executed for externalGraphic symbols │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.SVG.preventDefault = function(e) { │ │ │ │ │ - OpenLayers.Event.preventDefault(e); │ │ │ │ │ -}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol.js │ │ │ │ │ + OpenLayers/Control/DrawFeature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Protocol │ │ │ │ │ - * Abstract vector layer protocol class. Not to be instantiated directly. Use │ │ │ │ │ - * one of the protocol subclasses instead. │ │ │ │ │ + * Class: OpenLayers.Control.DrawFeature │ │ │ │ │ + * The DrawFeature control draws point, line or polygon features on a vector │ │ │ │ │ + * layer when active. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol = OpenLayers.Class({ │ │ │ │ │ +OpenLayers.Control.DrawFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: format │ │ │ │ │ - * {<OpenLayers.Format>} The format used by this protocol. │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} │ │ │ │ │ */ │ │ │ │ │ - format: null, │ │ │ │ │ + layer: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: options │ │ │ │ │ - * {Object} Any options sent to the constructor. │ │ │ │ │ + * Property: callbacks │ │ │ │ │ + * {Object} The functions that are sent to the handler for callback │ │ │ │ │ + */ │ │ │ │ │ + callbacks: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ + * featureadded - Triggered when a feature is added │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: multi │ │ │ │ │ + * {Boolean} Cast features to multi-part geometries before passing to the │ │ │ │ │ + * layer. Default is false. │ │ │ │ │ */ │ │ │ │ │ - options: null, │ │ │ │ │ + multi: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: autoDestroy │ │ │ │ │ - * {Boolean} The creator of the protocol can set autoDestroy to false │ │ │ │ │ - * to fully control when the protocol is destroyed. Defaults to │ │ │ │ │ - * true. │ │ │ │ │ + * APIProperty: featureAdded │ │ │ │ │ + * {Function} Called after each feature is added │ │ │ │ │ */ │ │ │ │ │ - autoDestroy: true, │ │ │ │ │ + featureAdded: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: defaultFilter │ │ │ │ │ - * {<OpenLayers.Filter>} Optional default filter to read requests │ │ │ │ │ + * APIProperty: handlerOptions │ │ │ │ │ + * {Object} Used to set non-default properties on the control's handler │ │ │ │ │ */ │ │ │ │ │ - defaultFilter: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol │ │ │ │ │ - * Abstract class for vector protocols. Create instances of a subclass. │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Control.DrawFeature │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ + * handler - {<OpenLayers.Handler>} │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.options = options; │ │ │ │ │ + initialize: function(layer, handler, options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.callbacks = OpenLayers.Util.extend({ │ │ │ │ │ + done: this.drawFeature, │ │ │ │ │ + modify: function(vertex, feature) { │ │ │ │ │ + this.layer.events.triggerEvent( │ │ │ │ │ + "sketchmodified", { │ │ │ │ │ + vertex: vertex, │ │ │ │ │ + feature: feature │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + create: function(vertex, feature) { │ │ │ │ │ + this.layer.events.triggerEvent( │ │ │ │ │ + "sketchstarted", { │ │ │ │ │ + vertex: vertex, │ │ │ │ │ + feature: feature │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + this.callbacks │ │ │ │ │ + ); │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + this.handlerOptions = this.handlerOptions || {}; │ │ │ │ │ + this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.handlerOptions.layerOptions, { │ │ │ │ │ + renderers: layer.renderers, │ │ │ │ │ + rendererOptions: layer.rendererOptions │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + if (!("multi" in this.handlerOptions)) { │ │ │ │ │ + this.handlerOptions.multi = this.multi; │ │ │ │ │ + } │ │ │ │ │ + var sketchStyle = this.layer.styleMap && this.layer.styleMap.styles.temporary; │ │ │ │ │ + if (sketchStyle) { │ │ │ │ │ + this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.handlerOptions.layerOptions, { │ │ │ │ │ + styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ + "default": sketchStyle │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + this.handler = new handler(this, this.callbacks, this.handlerOptions); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mergeWithDefaultFilter │ │ │ │ │ - * Merge filter passed to the read method with the default one │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} │ │ │ │ │ + * Method: drawFeature │ │ │ │ │ */ │ │ │ │ │ - mergeWithDefaultFilter: function(filter) { │ │ │ │ │ - var merged; │ │ │ │ │ - if (filter && this.defaultFilter) { │ │ │ │ │ - merged = new OpenLayers.Filter.Logical({ │ │ │ │ │ - type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ - filters: [this.defaultFilter, filter] │ │ │ │ │ + drawFeature: function(geometry) { │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + var proceed = this.layer.events.triggerEvent( │ │ │ │ │ + "sketchcomplete", { │ │ │ │ │ + feature: feature │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + if (proceed !== false) { │ │ │ │ │ + feature.state = OpenLayers.State.INSERT; │ │ │ │ │ + this.layer.addFeatures([feature]); │ │ │ │ │ + this.featureAdded(feature); │ │ │ │ │ + this.events.triggerEvent("featureadded", { │ │ │ │ │ + feature: feature │ │ │ │ │ }); │ │ │ │ │ - } else { │ │ │ │ │ - merged = filter || this.defaultFilter || undefined; │ │ │ │ │ } │ │ │ │ │ - return merged; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up the protocol. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.options = null; │ │ │ │ │ - this.format = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Construct a request for reading new features. │ │ │ │ │ + * APIMethod: insertXY │ │ │ │ │ + * Insert a point in the current sketch given x & y coordinates. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, the same object will be passed to the callback function passed │ │ │ │ │ - * if one exists in the options object. │ │ │ │ │ + * x - {Number} The x-coordinate of the point. │ │ │ │ │ + * y - {Number} The y-coordinate of the point. │ │ │ │ │ */ │ │ │ │ │ - read: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.filter = this.mergeWithDefaultFilter(options.filter); │ │ │ │ │ + insertXY: function(x, y) { │ │ │ │ │ + if (this.handler && this.handler.line) { │ │ │ │ │ + this.handler.insertXY(x, y); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: create │ │ │ │ │ - * Construct a request for writing newly created features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, the same object will be passed to the callback function passed │ │ │ │ │ - * if one exists in the options object. │ │ │ │ │ - */ │ │ │ │ │ - create: function() {}, │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: update │ │ │ │ │ - * Construct a request updating modified features. │ │ │ │ │ + * APIMethod: insertDeltaXY │ │ │ │ │ + * Insert a point given offsets from the previously inserted point. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, the same object will be passed to the callback function passed │ │ │ │ │ - * if one exists in the options object. │ │ │ │ │ + * dx - {Number} The x-coordinate offset of the point. │ │ │ │ │ + * dy - {Number} The y-coordinate offset of the point. │ │ │ │ │ */ │ │ │ │ │ - update: function() {}, │ │ │ │ │ + insertDeltaXY: function(dx, dy) { │ │ │ │ │ + if (this.handler && this.handler.line) { │ │ │ │ │ + this.handler.insertDeltaXY(dx, dy); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: delete │ │ │ │ │ - * Construct a request deleting a removed feature. │ │ │ │ │ + * APIMethod: insertDirectionLength │ │ │ │ │ + * Insert a point in the current sketch given a direction and a length. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, the same object will be passed to the callback function passed │ │ │ │ │ - * if one exists in the options object. │ │ │ │ │ + * direction - {Number} Degrees clockwise from the positive x-axis. │ │ │ │ │ + * length - {Number} Distance from the previously drawn point. │ │ │ │ │ */ │ │ │ │ │ - "delete": function() {}, │ │ │ │ │ + insertDirectionLength: function(direction, length) { │ │ │ │ │ + if (this.handler && this.handler.line) { │ │ │ │ │ + this.handler.insertDirectionLength(direction, length); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: commit │ │ │ │ │ - * Go over the features and for each take action │ │ │ │ │ - * based on the feature state. Possible actions are create, │ │ │ │ │ - * update and delete. │ │ │ │ │ + * APIMethod: insertDeflectionLength │ │ │ │ │ + * Insert a point in the current sketch given a deflection and a length. │ │ │ │ │ + * The deflection should be degrees clockwise from the previously │ │ │ │ │ + * digitized segment. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array({<OpenLayers.Feature.Vector>})} │ │ │ │ │ - * options - {Object} Object whose possible keys are "create", "update", │ │ │ │ │ - * "delete", "callback" and "scope", the values referenced by the │ │ │ │ │ - * first three are objects as passed to the "create", "update", and │ │ │ │ │ - * "delete" methods, the value referenced by the "callback" key is │ │ │ │ │ - * a function which is called when the commit operation is complete │ │ │ │ │ - * using the scope referenced by the "scope" key. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array({<OpenLayers.Protocol.Response>})} An array of │ │ │ │ │ - * <OpenLayers.Protocol.Response> objects. │ │ │ │ │ + * deflection - {Number} Degrees clockwise from the previous segment. │ │ │ │ │ + * length - {Number} Distance from the previously drawn point. │ │ │ │ │ */ │ │ │ │ │ - commit: function() {}, │ │ │ │ │ + insertDeflectionLength: function(deflection, length) { │ │ │ │ │ + if (this.handler && this.handler.line) { │ │ │ │ │ + this.handler.insertDeflectionLength(deflection, length); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: abort │ │ │ │ │ - * Abort an ongoing request. │ │ │ │ │ + * APIMethod: undo │ │ │ │ │ + * Remove the most recently added point in the current sketch geometry. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} An edit was undone. │ │ │ │ │ */ │ │ │ │ │ - abort: function(response) {}, │ │ │ │ │ + undo: function() { │ │ │ │ │ + return this.handler.undo && this.handler.undo(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createCallback │ │ │ │ │ - * Returns a function that applies the given public method with resp and │ │ │ │ │ - * options arguments. │ │ │ │ │ + * APIMethod: redo │ │ │ │ │ + * Reinsert the most recently removed point resulting from an <undo> call. │ │ │ │ │ + * The undo stack is deleted whenever a point is added by other means. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * method - {Function} The method to be applied by the callback. │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The protocol response object. │ │ │ │ │ - * options - {Object} Options sent to the protocol method │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} An edit was redone. │ │ │ │ │ */ │ │ │ │ │ - createCallback: function(method, response, options) { │ │ │ │ │ - return OpenLayers.Function.bind(function() { │ │ │ │ │ - method.apply(this, [response, options]); │ │ │ │ │ - }, this); │ │ │ │ │ + redo: function() { │ │ │ │ │ + return this.handler.redo && this.handler.redo(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Protocol.Response │ │ │ │ │ - * Protocols return Response objects to their users. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol.Response = OpenLayers.Class({ │ │ │ │ │ - /** │ │ │ │ │ - * Property: code │ │ │ │ │ - * {Number} - OpenLayers.Protocol.Response.SUCCESS or │ │ │ │ │ - * OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ - */ │ │ │ │ │ - code: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: requestType │ │ │ │ │ - * {String} The type of request this response corresponds to. Either │ │ │ │ │ - * "create", "read", "update" or "delete". │ │ │ │ │ - */ │ │ │ │ │ - requestType: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: last │ │ │ │ │ - * {Boolean} - true if this is the last response expected in a commit, │ │ │ │ │ - * false otherwise, defaults to true. │ │ │ │ │ - */ │ │ │ │ │ - last: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Array({<OpenLayers.Feature.Vector>})} or {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * The features returned in the response by the server. Depending on the │ │ │ │ │ - * protocol's read payload, either features or data will be populated. │ │ │ │ │ - */ │ │ │ │ │ - features: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: data │ │ │ │ │ - * {Object} │ │ │ │ │ - * The data returned in the response by the server. Depending on the │ │ │ │ │ - * protocol's read payload, either features or data will be populated. │ │ │ │ │ - */ │ │ │ │ │ - data: null, │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Property: reqFeatures │ │ │ │ │ - * {Array({<OpenLayers.Feature.Vector>})} or {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * The features provided by the user and placed in the request by the │ │ │ │ │ - * protocol. │ │ │ │ │ - */ │ │ │ │ │ - reqFeatures: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: priv │ │ │ │ │ - */ │ │ │ │ │ - priv: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: error │ │ │ │ │ - * {Object} The error object in case a service exception was encountered. │ │ │ │ │ - */ │ │ │ │ │ - error: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.Response │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * APIMethod: finishSketch │ │ │ │ │ + * Finishes the sketch without including the currently drawn point. │ │ │ │ │ + * This method can be called to terminate drawing programmatically │ │ │ │ │ + * instead of waiting for the user to end the sketch. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ + finishSketch: function() { │ │ │ │ │ + this.handler.finishGeometry(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: success │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} - true on success, false otherwise │ │ │ │ │ + * APIMethod: cancel │ │ │ │ │ + * Cancel the current sketch. This removes the current sketch and keeps │ │ │ │ │ + * the drawing control active. │ │ │ │ │ */ │ │ │ │ │ - success: function() { │ │ │ │ │ - return this.code > 0; │ │ │ │ │ + cancel: function() { │ │ │ │ │ + this.handler.cancel(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.Response" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.DrawFeature" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Protocol.Response.SUCCESS = 1; │ │ │ │ │ -OpenLayers.Protocol.Response.FAILURE = 0; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/WFS.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Protocol.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Protocol.WFS │ │ │ │ │ - * Used to create a versioned WFS protocol. Default version is 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol>} A WFS protocol of the given version. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var protocol = new OpenLayers.Protocol.WFS({ │ │ │ │ │ - * version: "1.1.0", │ │ │ │ │ - * url: "http://demo.opengeo.org/geoserver/wfs", │ │ │ │ │ - * featureType: "tasmania_roads", │ │ │ │ │ - * featureNS: "http://www.openplans.org/topp", │ │ │ │ │ - * geometryName: "the_geom" │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * See the protocols for specific WFS versions for more detail. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol.WFS = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options, OpenLayers.Protocol.WFS.DEFAULTS │ │ │ │ │ - ); │ │ │ │ │ - var cls = OpenLayers.Protocol.WFS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported WFS version: " + options.version; │ │ │ │ │ - } │ │ │ │ │ - return new cls(options); │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Function: fromWMSLayer │ │ │ │ │ - * Convenience function to create a WFS protocol from a WMS layer. This makes │ │ │ │ │ - * the assumption that a WFS requests can be issued at the same URL as │ │ │ │ │ - * WMS requests and that a WFS featureType exists with the same name as the │ │ │ │ │ - * WMS layer. │ │ │ │ │ - * │ │ │ │ │ - * This function is designed to auto-configure <url>, <featureType>, │ │ │ │ │ - * <featurePrefix> and <srsName> for WFS <version> 1.1.0. Note that │ │ │ │ │ - * srsName matching with the WMS layer will not work with WFS 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.WMS>} WMS layer that has a matching WFS │ │ │ │ │ - * FeatureType at the same server url with the same typename. │ │ │ │ │ - * options - {Object} Default properties to be set on the protocol. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.WFS>} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol.WFS.fromWMSLayer = function(layer, options) { │ │ │ │ │ - var typeName, featurePrefix; │ │ │ │ │ - var param = layer.params["LAYERS"]; │ │ │ │ │ - var parts = (OpenLayers.Util.isArray(param) ? param[0] : param).split(":"); │ │ │ │ │ - if (parts.length > 1) { │ │ │ │ │ - featurePrefix = parts[0]; │ │ │ │ │ - } │ │ │ │ │ - typeName = parts.pop(); │ │ │ │ │ - var protocolOptions = { │ │ │ │ │ - url: layer.url, │ │ │ │ │ - featureType: typeName, │ │ │ │ │ - featurePrefix: featurePrefix, │ │ │ │ │ - srsName: layer.projection && layer.projection.getCode() || │ │ │ │ │ - layer.map && layer.map.getProjectionObject().getCode(), │ │ │ │ │ - version: "1.1.0" │ │ │ │ │ - }; │ │ │ │ │ - return new OpenLayers.Protocol.WFS(OpenLayers.Util.applyDefaults( │ │ │ │ │ - options, protocolOptions │ │ │ │ │ - )); │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Protocol.WFS.DEFAULTS │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol.WFS.DEFAULTS = { │ │ │ │ │ - "version": "1.0.0" │ │ │ │ │ -}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/HTTP.js │ │ │ │ │ + OpenLayers/Control/Panel.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Protocol.js │ │ │ │ │ - * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * if application uses the query string, for example, for BBOX parameters, │ │ │ │ │ - * OpenLayers/Format/QueryStringFilter.js should be included in the build config file │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Protocol.HTTP │ │ │ │ │ - * A basic HTTP protocol for vector layers. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Protocol.HTTP> constructor. │ │ │ │ │ + * Class: OpenLayers.Control.Panel │ │ │ │ │ + * The Panel control is a container for other controls. With it toolbars │ │ │ │ │ + * may be composed. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.HTTP = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: url │ │ │ │ │ - * {String} Service URL, read-only, set through the options │ │ │ │ │ - * passed to constructor. │ │ │ │ │ - */ │ │ │ │ │ - url: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: headers │ │ │ │ │ - * {Object} HTTP request headers, read-only, set through the options │ │ │ │ │ - * passed to the constructor, │ │ │ │ │ - * Example: {'Content-Type': 'plain/text'} │ │ │ │ │ - */ │ │ │ │ │ - headers: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: params │ │ │ │ │ - * {Object} Parameters of GET requests, read-only, set through the options │ │ │ │ │ - * passed to the constructor, │ │ │ │ │ - * Example: {'bbox': '5,5,5,5'} │ │ │ │ │ - */ │ │ │ │ │ - params: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: callback │ │ │ │ │ - * {Object} Function to be called when the <read>, <create>, │ │ │ │ │ - * <update>, <delete> or <commit> operation completes, read-only, │ │ │ │ │ - * set through the options passed to the constructor. │ │ │ │ │ - */ │ │ │ │ │ - callback: null, │ │ │ │ │ - │ │ │ │ │ +OpenLayers.Control.Panel = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ /** │ │ │ │ │ - * Property: scope │ │ │ │ │ - * {Object} Callback execution scope, read-only, set through the │ │ │ │ │ - * options passed to the constructor. │ │ │ │ │ + * Property: controls │ │ │ │ │ + * {Array(<OpenLayers.Control>)} │ │ │ │ │ */ │ │ │ │ │ - scope: null, │ │ │ │ │ + controls: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: readWithPOST │ │ │ │ │ - * {Boolean} true if read operations are done with POST requests │ │ │ │ │ - * instead of GET, defaults to false. │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ - readWithPOST: false, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: updateWithPOST │ │ │ │ │ - * {Boolean} true if update operations are done with POST requests │ │ │ │ │ - * defaults to false. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultControl │ │ │ │ │ + * {<OpenLayers.Control>} The control which is activated when the control is │ │ │ │ │ + * activated (turned on), which also happens at instantiation. │ │ │ │ │ + * If <saveState> is true, <defaultControl> will be nullified after the │ │ │ │ │ + * first activation of the panel. │ │ │ │ │ */ │ │ │ │ │ - updateWithPOST: false, │ │ │ │ │ + defaultControl: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: deleteWithPOST │ │ │ │ │ - * {Boolean} true if delete operations are done with POST requests │ │ │ │ │ - * defaults to false. │ │ │ │ │ - * if true, POST data is set to output of format.write(). │ │ │ │ │ + * APIProperty: saveState │ │ │ │ │ + * {Boolean} If set to true, the active state of this panel's controls will │ │ │ │ │ + * be stored on panel deactivation, and restored on reactivation. Default │ │ │ │ │ + * is false. │ │ │ │ │ */ │ │ │ │ │ - deleteWithPOST: false, │ │ │ │ │ + saveState: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: wildcarded. │ │ │ │ │ - * {Boolean} If true percent signs are added around values │ │ │ │ │ - * read from LIKE filters, for example if the protocol │ │ │ │ │ - * read method is passed a LIKE filter whose property │ │ │ │ │ - * is "foo" and whose value is "bar" the string │ │ │ │ │ - * "foo__ilike=%bar%" will be sent in the query string; │ │ │ │ │ - * defaults to false. │ │ │ │ │ + * APIProperty: allowDepress │ │ │ │ │ + * {Boolean} If is true the <OpenLayers.Control.TYPE_TOOL> controls can │ │ │ │ │ + * be deactivated by clicking the icon that represents them. Default │ │ │ │ │ + * is false. │ │ │ │ │ */ │ │ │ │ │ - wildcarded: false, │ │ │ │ │ + allowDepress: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: srsInBBOX │ │ │ │ │ - * {Boolean} Include the SRS identifier in BBOX query string parameter. │ │ │ │ │ - * Default is false. If true and the layer has a projection object set, │ │ │ │ │ - * any BBOX filter will be serialized with a fifth item identifying the │ │ │ │ │ - * projection. E.g. bbox=-1000,-1000,1000,1000,EPSG:900913 │ │ │ │ │ + * Property: activeState │ │ │ │ │ + * {Object} stores the active state of this panel's controls. │ │ │ │ │ */ │ │ │ │ │ - srsInBBOX: false, │ │ │ │ │ + activeState: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.HTTP │ │ │ │ │ - * A class for giving layers generic HTTP protocol. │ │ │ │ │ + * Constructor: OpenLayers.Control.Panel │ │ │ │ │ + * Create a new control panel. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * Each control in the panel is represented by an icon. When clicking │ │ │ │ │ + * on an icon, the <activateControl> method is called. │ │ │ │ │ * │ │ │ │ │ - * Valid options include: │ │ │ │ │ - * url - {String} │ │ │ │ │ - * headers - {Object} │ │ │ │ │ - * params - {Object} URL parameters for GET requests │ │ │ │ │ - * format - {<OpenLayers.Format>} │ │ │ │ │ - * callback - {Function} │ │ │ │ │ - * scope - {Object} │ │ │ │ │ + * Specific properties for controls on a panel: │ │ │ │ │ + * type - {Number} One of <OpenLayers.Control.TYPE_TOOL>, │ │ │ │ │ + * <OpenLayers.Control.TYPE_TOGGLE>, <OpenLayers.Control.TYPE_BUTTON>. │ │ │ │ │ + * If not provided, <OpenLayers.Control.TYPE_TOOL> is assumed. │ │ │ │ │ + * title - {string} Text displayed when mouse is over the icon that │ │ │ │ │ + * represents the control. │ │ │ │ │ + * │ │ │ │ │ + * The <OpenLayers.Control.type> of a control determines the behavior when │ │ │ │ │ + * clicking its icon: │ │ │ │ │ + * <OpenLayers.Control.TYPE_TOOL> - The control is activated and other │ │ │ │ │ + * controls of this type in the same panel are deactivated. This is │ │ │ │ │ + * the default type. │ │ │ │ │ + * <OpenLayers.Control.TYPE_TOGGLE> - The active state of the control is │ │ │ │ │ + * toggled. │ │ │ │ │ + * <OpenLayers.Control.TYPE_BUTTON> - The │ │ │ │ │ + * <OpenLayers.Control.Button.trigger> method of the control is called, │ │ │ │ │ + * but its active state is not changed. │ │ │ │ │ + * │ │ │ │ │ + * If a control is <OpenLayers.Control.active>, it will be drawn with the │ │ │ │ │ + * olControl[Name]ItemActive class, otherwise with the │ │ │ │ │ + * olControl[Name]ItemInactive class. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be used │ │ │ │ │ + * to extend the control. │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - this.params = {}; │ │ │ │ │ - this.headers = {}; │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ - var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ - wildcarded: this.wildcarded, │ │ │ │ │ - srsInBBOX: this.srsInBBOX │ │ │ │ │ - }); │ │ │ │ │ - this.filterToParams = function(filter, params) { │ │ │ │ │ - return format.write(filter, params); │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.controls = []; │ │ │ │ │ + this.activeState = {}; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIMethod: destroy │ │ │ │ │ - * Clean up the protocol. │ │ │ │ │ */ │ │ │ │ │ destroy: function() { │ │ │ │ │ - this.params = null; │ │ │ │ │ - this.headers = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + for (var ctl, i = this.controls.length - 1; i >= 0; i--) { │ │ │ │ │ + ctl = this.controls[i]; │ │ │ │ │ + if (ctl.events) { │ │ │ │ │ + ctl.events.un({ │ │ │ │ │ + activate: this.iconOn, │ │ │ │ │ + deactivate: this.iconOff │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + ctl.panel_div = null; │ │ │ │ │ + } │ │ │ │ │ + this.activeState = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: filterToParams │ │ │ │ │ - * Optional method to translate an <OpenLayers.Filter> object into an object │ │ │ │ │ - * that can be serialized as request query string provided. If a custom │ │ │ │ │ - * method is not provided, the filter will be serialized using the │ │ │ │ │ - * <OpenLayers.Format.QueryStringFilter> class. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} filter to convert. │ │ │ │ │ - * params - {Object} The parameters object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The resulting parameters object. │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + control = this.controls[i]; │ │ │ │ │ + if (control === this.defaultControl || │ │ │ │ │ + (this.saveState && this.activeState[control.id])) { │ │ │ │ │ + control.activate(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.saveState === true) { │ │ │ │ │ + this.defaultControl = null; │ │ │ │ │ + } │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Construct a request for reading new features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * url - {String} Url for the request. │ │ │ │ │ - * params - {Object} Parameters to get serialized as a query string. │ │ │ │ │ - * headers - {Object} Headers to be set on the request. │ │ │ │ │ - * filter - {<OpenLayers.Filter>} Filter to get serialized as a │ │ │ │ │ - * query string. │ │ │ │ │ - * readWithPOST - {Boolean} If the request should be done with POST. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} A response object, whose "priv" property │ │ │ │ │ - * references the HTTP request, this object is also passed to the │ │ │ │ │ - * callback function when the request completes, its "features" property │ │ │ │ │ - * is then populated with the features received from the server. │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ */ │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options.params, this.options.params); │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - if (options.filter && this.filterToParams) { │ │ │ │ │ - options.params = this.filterToParams( │ │ │ │ │ - options.filter, options.params │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - var readWithPOST = (options.readWithPOST !== undefined) ? │ │ │ │ │ - options.readWithPOST : this.readWithPOST; │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - if (readWithPOST) { │ │ │ │ │ - var headers = options.headers || {}; │ │ │ │ │ - headers["Content-Type"] = "application/x-www-form-urlencoded"; │ │ │ │ │ - resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ - data: OpenLayers.Util.getParameterString(options.params), │ │ │ │ │ - headers: headers │ │ │ │ │ - }); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + control = this.controls[i]; │ │ │ │ │ + this.activeState[control.id] = control.deactivate(); │ │ │ │ │ + } │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return true; │ │ │ │ │ } else { │ │ │ │ │ - resp.priv = OpenLayers.Request.GET({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers │ │ │ │ │ - }); │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleRead │ │ │ │ │ - * Individual callbacks are created for read, create and update, should │ │ │ │ │ - * a subclass need to override each one separately. │ │ │ │ │ + * Method: draw │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * the user callback. │ │ │ │ │ - * options - {Object} The user options passed to the read call. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - handleRead: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options); │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (this.outsideViewport) { │ │ │ │ │ + this.events.attachToElement(this.div); │ │ │ │ │ + this.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ + } else { │ │ │ │ │ + this.map.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ + } │ │ │ │ │ + this.addControlsToMap(this.controls); │ │ │ │ │ + return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: create │ │ │ │ │ - * Construct a request for writing newly created features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, whose "priv" property references the HTTP request, this │ │ │ │ │ - * object is also passed to the callback function when the request │ │ │ │ │ - * completes, its "features" property is then populated with the │ │ │ │ │ - * the features received from the server. │ │ │ │ │ + * Method: redraw │ │ │ │ │ */ │ │ │ │ │ - create: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: features, │ │ │ │ │ - requestType: "create" │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleCreate, resp, options), │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(features) │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - return resp; │ │ │ │ │ + redraw: function() { │ │ │ │ │ + for (var l = this.div.childNodes.length, i = l - 1; i >= 0; i--) { │ │ │ │ │ + this.div.removeChild(this.div.childNodes[i]); │ │ │ │ │ + } │ │ │ │ │ + this.div.innerHTML = ""; │ │ │ │ │ + if (this.active) { │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + this.div.appendChild(this.controls[i].panel_div); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleCreate │ │ │ │ │ - * Called the the request issued by <create> is complete. May be overridden │ │ │ │ │ - * by subclasses. │ │ │ │ │ + * APIMethod: activateControl │ │ │ │ │ + * This method is called when the user click on the icon representing a │ │ │ │ │ + * control in the panel. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the create call. │ │ │ │ │ + * control - {<OpenLayers.Control>} │ │ │ │ │ */ │ │ │ │ │ - handleCreate: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options); │ │ │ │ │ + activateControl: function(control) { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + if (control.type == OpenLayers.Control.TYPE_BUTTON) { │ │ │ │ │ + control.trigger(); │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (control.type == OpenLayers.Control.TYPE_TOGGLE) { │ │ │ │ │ + if (control.active) { │ │ │ │ │ + control.deactivate(); │ │ │ │ │ + } else { │ │ │ │ │ + control.activate(); │ │ │ │ │ + } │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (this.allowDepress && control.active) { │ │ │ │ │ + control.deactivate(); │ │ │ │ │ + } else { │ │ │ │ │ + var c; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + c = this.controls[i]; │ │ │ │ │ + if (c != control && │ │ │ │ │ + (c.type === OpenLayers.Control.TYPE_TOOL || c.type == null)) { │ │ │ │ │ + c.deactivate(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + control.activate(); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: update │ │ │ │ │ - * Construct a request updating modified feature. │ │ │ │ │ + * APIMethod: addControls │ │ │ │ │ + * To build a toolbar, you add a set of controls to it. addControls │ │ │ │ │ + * lets you add a single control or a list of controls to the │ │ │ │ │ + * Control Panel. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, whose "priv" property references the HTTP request, this │ │ │ │ │ - * object is also passed to the callback function when the request │ │ │ │ │ - * completes, its "features" property is then populated with the │ │ │ │ │ - * the feature received from the server. │ │ │ │ │ + * controls - {<OpenLayers.Control>} Controls to add in the panel. │ │ │ │ │ */ │ │ │ │ │ - update: function(feature, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var url = options.url || │ │ │ │ │ - feature.url || │ │ │ │ │ - this.options.url + "/" + feature.fid; │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: feature, │ │ │ │ │ - requestType: "update" │ │ │ │ │ - }); │ │ │ │ │ + addControls: function(controls) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(controls))) { │ │ │ │ │ + controls = [controls]; │ │ │ │ │ + } │ │ │ │ │ + this.controls = this.controls.concat(controls); │ │ │ │ │ │ │ │ │ │ - var method = this.updateWithPOST ? "POST" : "PUT"; │ │ │ │ │ - resp.priv = OpenLayers.Request[method]({ │ │ │ │ │ - url: url, │ │ │ │ │ - callback: this.createCallback(this.handleUpdate, resp, options), │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(feature) │ │ │ │ │ - }); │ │ │ │ │ + for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ + var control = controls[i], │ │ │ │ │ + element = this.createControlMarkup(control); │ │ │ │ │ + OpenLayers.Element.addClass(element, │ │ │ │ │ + control.displayClass + "ItemInactive"); │ │ │ │ │ + OpenLayers.Element.addClass(element, "olButton"); │ │ │ │ │ + if (control.title != "" && !element.title) { │ │ │ │ │ + element.title = control.title; │ │ │ │ │ + } │ │ │ │ │ + control.panel_div = element; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - return resp; │ │ │ │ │ + if (this.map) { // map.addControl() has already been called on the panel │ │ │ │ │ + this.addControlsToMap(controls); │ │ │ │ │ + this.redraw(); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleUpdate │ │ │ │ │ - * Called the the request issued by <update> is complete. May be overridden │ │ │ │ │ - * by subclasses. │ │ │ │ │ + * APIMethod: createControlMarkup │ │ │ │ │ + * This function just creates a div for the control. If specific HTML │ │ │ │ │ + * markup is needed this function can be overridden in specific classes, │ │ │ │ │ + * or at panel instantiation time: │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var panel = new OpenLayers.Control.Panel({ │ │ │ │ │ + * defaultControl: control, │ │ │ │ │ + * // ovverride createControlMarkup to create actual buttons │ │ │ │ │ + * // including texts wrapped into span elements. │ │ │ │ │ + * createControlMarkup: function(control) { │ │ │ │ │ + * var button = document.createElement('button'), │ │ │ │ │ + * span = document.createElement('span'); │ │ │ │ │ + * if (control.text) { │ │ │ │ │ + * span.innerHTML = control.text; │ │ │ │ │ + * } │ │ │ │ │ + * return button; │ │ │ │ │ + * } │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the update call. │ │ │ │ │ + * control - {<OpenLayers.Control>} The control to create the HTML │ │ │ │ │ + * markup for. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The markup. │ │ │ │ │ */ │ │ │ │ │ - handleUpdate: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options); │ │ │ │ │ + createControlMarkup: function(control) { │ │ │ │ │ + return document.createElement("div"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: delete │ │ │ │ │ - * Construct a request deleting a removed feature. │ │ │ │ │ + * Method: addControlsToMap │ │ │ │ │ + * Only for internal use in draw() and addControls() methods. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, whose "priv" property references the HTTP request, this │ │ │ │ │ - * object is also passed to the callback function when the request │ │ │ │ │ - * completes. │ │ │ │ │ + * controls - {Array(<OpenLayers.Control>)} Controls to add into map. │ │ │ │ │ */ │ │ │ │ │ - "delete": function(feature, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var url = options.url || │ │ │ │ │ - feature.url || │ │ │ │ │ - this.options.url + "/" + feature.fid; │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: feature, │ │ │ │ │ - requestType: "delete" │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - var method = this.deleteWithPOST ? "POST" : "DELETE"; │ │ │ │ │ - var requestOptions = { │ │ │ │ │ - url: url, │ │ │ │ │ - callback: this.createCallback(this.handleDelete, resp, options), │ │ │ │ │ - headers: options.headers │ │ │ │ │ - }; │ │ │ │ │ - if (this.deleteWithPOST) { │ │ │ │ │ - requestOptions.data = this.format.write(feature); │ │ │ │ │ + addControlsToMap: function(controls) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ + control = controls[i]; │ │ │ │ │ + if (control.autoActivate === true) { │ │ │ │ │ + control.autoActivate = false; │ │ │ │ │ + this.map.addControl(control); │ │ │ │ │ + control.autoActivate = true; │ │ │ │ │ + } else { │ │ │ │ │ + this.map.addControl(control); │ │ │ │ │ + control.deactivate(); │ │ │ │ │ + } │ │ │ │ │ + control.events.on({ │ │ │ │ │ + activate: this.iconOn, │ │ │ │ │ + deactivate: this.iconOff │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - resp.priv = OpenLayers.Request[method](requestOptions); │ │ │ │ │ - │ │ │ │ │ - return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleDelete │ │ │ │ │ - * Called the the request issued by <delete> is complete. May be overridden │ │ │ │ │ - * by subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the delete call. │ │ │ │ │ + * Method: iconOn │ │ │ │ │ + * Internal use, for use only with "controls[i].events.on/un". │ │ │ │ │ */ │ │ │ │ │ - handleDelete: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options); │ │ │ │ │ + iconOn: function() { │ │ │ │ │ + var d = this.panel_div; // "this" refers to a control on panel! │ │ │ │ │ + var re = new RegExp("\\b(" + this.displayClass + "Item)Inactive\\b"); │ │ │ │ │ + d.className = d.className.replace(re, "$1Active"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleResponse │ │ │ │ │ - * Called by CRUD specific handlers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the create, read, update, │ │ │ │ │ - * or delete call. │ │ │ │ │ + * Method: iconOff │ │ │ │ │ + * Internal use, for use only with "controls[i].events.on/un". │ │ │ │ │ */ │ │ │ │ │ - handleResponse: function(resp, options) { │ │ │ │ │ - var request = resp.priv; │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - // success │ │ │ │ │ - if (resp.requestType != "delete") { │ │ │ │ │ - resp.features = this.parseFeatures(request); │ │ │ │ │ - } │ │ │ │ │ - resp.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ - } else { │ │ │ │ │ - // failure │ │ │ │ │ - resp.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - } │ │ │ │ │ - options.callback.call(options.scope, resp); │ │ │ │ │ - } │ │ │ │ │ + iconOff: function() { │ │ │ │ │ + var d = this.panel_div; // "this" refers to a control on panel! │ │ │ │ │ + var re = new RegExp("\\b(" + this.displayClass + "Item)Active\\b"); │ │ │ │ │ + d.className = d.className.replace(re, "$1Inactive"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseFeatures │ │ │ │ │ - * Read HTTP response body and return features. │ │ │ │ │ + * Method: onButtonClick │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * request - {XMLHttpRequest} The request object │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} Array of features or a single feature. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - parseFeatures: function(request) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null; │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + var controls = this.controls, │ │ │ │ │ + button = evt.buttonElement; │ │ │ │ │ + for (var i = controls.length - 1; i >= 0; --i) { │ │ │ │ │ + if (controls[i].panel_div === button) { │ │ │ │ │ + this.activateControl(controls[i]); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return this.format.read(doc); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: commit │ │ │ │ │ - * Iterate over each feature and take action based on the feature state. │ │ │ │ │ - * Possible actions are create, update and delete. │ │ │ │ │ + * APIMethod: getControlsBy │ │ │ │ │ + * Get a list of controls with properties matching the given criteria. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array({<OpenLayers.Feature.Vector>})} │ │ │ │ │ - * options - {Object} Optional object for setting up intermediate commit │ │ │ │ │ - * callbacks. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * create - {Object} Optional object to be passed to the <create> method. │ │ │ │ │ - * update - {Object} Optional object to be passed to the <update> method. │ │ │ │ │ - * delete - {Object} Optional object to be passed to the <delete> method. │ │ │ │ │ - * callback - {Function} Optional function to be called when the commit │ │ │ │ │ - * is complete. │ │ │ │ │ - * scope - {Object} Optional object to be set as the scope of the callback. │ │ │ │ │ + * property - {String} A control property to be matched. │ │ │ │ │ + * match - {String | Object} A string to match. Can also be a regular │ │ │ │ │ + * expression literal or object. In addition, it can be any object │ │ │ │ │ + * with a method named test. For reqular expressions or other, if │ │ │ │ │ + * match.test(control[property]) evaluates to true, the control will be │ │ │ │ │ + * included in the array returned. If no controls are found, an empty │ │ │ │ │ + * array is returned. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Protocol.Response>)} An array of response objects, │ │ │ │ │ - * one per request made to the server, each object's "priv" property │ │ │ │ │ - * references the corresponding HTTP request. │ │ │ │ │ + * {Array(<OpenLayers.Control>)} A list of controls matching the given criteria. │ │ │ │ │ + * An empty array is returned if no matches are found. │ │ │ │ │ */ │ │ │ │ │ - commit: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = [], │ │ │ │ │ - nResponses = 0; │ │ │ │ │ - │ │ │ │ │ - // Divide up features before issuing any requests. This properly │ │ │ │ │ - // counts requests in the event that any responses come in before │ │ │ │ │ - // all requests have been issued. │ │ │ │ │ - var types = {}; │ │ │ │ │ - types[OpenLayers.State.INSERT] = []; │ │ │ │ │ - types[OpenLayers.State.UPDATE] = []; │ │ │ │ │ - types[OpenLayers.State.DELETE] = []; │ │ │ │ │ - var feature, list, requestFeatures = []; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - list = types[feature.state]; │ │ │ │ │ - if (list) { │ │ │ │ │ - list.push(feature); │ │ │ │ │ - requestFeatures.push(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // tally up number of requests │ │ │ │ │ - var nRequests = (types[OpenLayers.State.INSERT].length > 0 ? 1 : 0) + │ │ │ │ │ - types[OpenLayers.State.UPDATE].length + │ │ │ │ │ - types[OpenLayers.State.DELETE].length; │ │ │ │ │ - │ │ │ │ │ - // This response will be sent to the final callback after all the others │ │ │ │ │ - // have been fired. │ │ │ │ │ - var success = true; │ │ │ │ │ - var finalResponse = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: requestFeatures │ │ │ │ │ + getControlsBy: function(property, match) { │ │ │ │ │ + var test = (typeof match.test == "function"); │ │ │ │ │ + var found = OpenLayers.Array.filter(this.controls, function(item) { │ │ │ │ │ + return item[property] == match || (test && match.test(item[property])); │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ - function insertCallback(response) { │ │ │ │ │ - var len = response.features ? response.features.length : 0; │ │ │ │ │ - var fids = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - fids[i] = response.features[i].fid; │ │ │ │ │ - } │ │ │ │ │ - finalResponse.insertIds = fids; │ │ │ │ │ - callback.apply(this, [response]); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function callback(response) { │ │ │ │ │ - this.callUserCallback(response, options); │ │ │ │ │ - success = success && response.success(); │ │ │ │ │ - nResponses++; │ │ │ │ │ - if (nResponses >= nRequests) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - finalResponse.code = success ? │ │ │ │ │ - OpenLayers.Protocol.Response.SUCCESS : │ │ │ │ │ - OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - options.callback.apply(options.scope, [finalResponse]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // start issuing requests │ │ │ │ │ - var queue = types[OpenLayers.State.INSERT]; │ │ │ │ │ - if (queue.length > 0) { │ │ │ │ │ - resp.push(this.create( │ │ │ │ │ - queue, OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: insertCallback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options.create) │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - queue = types[OpenLayers.State.UPDATE]; │ │ │ │ │ - for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ - resp.push(this.update( │ │ │ │ │ - queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: callback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options.update))); │ │ │ │ │ - } │ │ │ │ │ - queue = types[OpenLayers.State.DELETE]; │ │ │ │ │ - for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ - resp.push(this["delete"]( │ │ │ │ │ - queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: callback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options["delete"]))); │ │ │ │ │ - } │ │ │ │ │ - return resp; │ │ │ │ │ + return found; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: abort │ │ │ │ │ - * Abort an ongoing request, the response object passed to │ │ │ │ │ - * this method must come from this HTTP protocol (as a result │ │ │ │ │ - * of a create, read, update, delete or commit operation). │ │ │ │ │ + * APIMethod: getControlsByName │ │ │ │ │ + * Get a list of contorls with names matching the given name. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ + * match - {String | Object} A control name. The name can also be a regular │ │ │ │ │ + * expression literal or object. In addition, it can be any object │ │ │ │ │ + * with a method named test. For reqular expressions or other, if │ │ │ │ │ + * name.test(control.name) evaluates to true, the control will be included │ │ │ │ │ + * in the list of controls returned. If no controls are found, an empty │ │ │ │ │ + * array is returned. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Control>)} A list of controls matching the given name. │ │ │ │ │ + * An empty array is returned if no matches are found. │ │ │ │ │ */ │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - response.priv.abort(); │ │ │ │ │ - } │ │ │ │ │ + getControlsByName: function(match) { │ │ │ │ │ + return this.getControlsBy("name", match); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: callUserCallback │ │ │ │ │ - * This method is used from within the commit method each time an │ │ │ │ │ - * an HTTP response is received from the server, it is responsible │ │ │ │ │ - * for calling the user-supplied callbacks. │ │ │ │ │ + * APIMethod: getControlsByClass │ │ │ │ │ + * Get a list of controls of a given type (CLASS_NAME). │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} │ │ │ │ │ - * options - {Object} The map of options passed to the commit call. │ │ │ │ │ + * match - {String | Object} A control class name. The type can also be a │ │ │ │ │ + * regular expression literal or object. In addition, it can be any │ │ │ │ │ + * object with a method named test. For reqular expressions or other, │ │ │ │ │ + * if type.test(control.CLASS_NAME) evaluates to true, the control will │ │ │ │ │ + * be included in the list of controls returned. If no controls are │ │ │ │ │ + * found, an empty array is returned. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Control>)} A list of controls matching the given type. │ │ │ │ │ + * An empty array is returned if no matches are found. │ │ │ │ │ */ │ │ │ │ │ - callUserCallback: function(resp, options) { │ │ │ │ │ - var opt = options[resp.requestType]; │ │ │ │ │ - if (opt && opt.callback) { │ │ │ │ │ - opt.callback.call(opt.scope, resp); │ │ │ │ │ - } │ │ │ │ │ + getControlsByClass: function(match) { │ │ │ │ │ + return this.getControlsBy("CLASS_NAME", match); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.HTTP" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Panel" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/WFS/v1.js │ │ │ │ │ + OpenLayers/Handler/Feature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Protocol/WFS.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Protocol.WFS.v1 │ │ │ │ │ - * Abstract class for for v1.0.0 and v1.1.0 protocol. │ │ │ │ │ + * Class: OpenLayers.Handler.Feature │ │ │ │ │ + * Handler to respond to mouse events related to a drawn feature. Callbacks │ │ │ │ │ + * with the following keys will be notified of the following events │ │ │ │ │ + * associated with features: click, clickout, over, out, and dblclick. │ │ │ │ │ + * │ │ │ │ │ + * This handler stops event propagation for mousedown and mouseup if those │ │ │ │ │ + * browser events target features that can be selected. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.WFS.v1 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ +OpenLayers.Handler.Feature = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} WFS version number. │ │ │ │ │ + * Property: EVENTMAP │ │ │ │ │ + * {Object} A object mapping the browser events to objects with callback │ │ │ │ │ + * keys for in and out. │ │ │ │ │ */ │ │ │ │ │ - version: null, │ │ │ │ │ + EVENTMAP: { │ │ │ │ │ + 'click': { │ │ │ │ │ + 'in': 'click', │ │ │ │ │ + 'out': 'clickout' │ │ │ │ │ + }, │ │ │ │ │ + 'mousemove': { │ │ │ │ │ + 'in': 'over', │ │ │ │ │ + 'out': 'out' │ │ │ │ │ + }, │ │ │ │ │ + 'dblclick': { │ │ │ │ │ + 'in': 'dblclick', │ │ │ │ │ + 'out': null │ │ │ │ │ + }, │ │ │ │ │ + 'mousedown': { │ │ │ │ │ + 'in': null, │ │ │ │ │ + 'out': null │ │ │ │ │ + }, │ │ │ │ │ + 'mouseup': { │ │ │ │ │ + 'in': null, │ │ │ │ │ + 'out': null │ │ │ │ │ + }, │ │ │ │ │ + 'touchstart': { │ │ │ │ │ + 'in': 'click', │ │ │ │ │ + 'out': 'clickout' │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: srsName │ │ │ │ │ - * {String} Name of spatial reference system. Default is "EPSG:4326". │ │ │ │ │ + * Property: feature │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} The last feature that was hovered. │ │ │ │ │ */ │ │ │ │ │ - srsName: "EPSG:4326", │ │ │ │ │ + feature: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: featureType │ │ │ │ │ - * {String} Local feature typeName. │ │ │ │ │ + * Property: lastFeature │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} The last feature that was handled. │ │ │ │ │ */ │ │ │ │ │ - featureType: null, │ │ │ │ │ + lastFeature: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: featureNS │ │ │ │ │ - * {String} Feature namespace. │ │ │ │ │ + * Property: down │ │ │ │ │ + * {<OpenLayers.Pixel>} The location of the last mousedown. │ │ │ │ │ */ │ │ │ │ │ - featureNS: null, │ │ │ │ │ + down: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: geometryName │ │ │ │ │ - * {String} Name of the geometry attribute for features. Default is │ │ │ │ │ - * "the_geom" for WFS <version> 1.0, and null for higher versions. │ │ │ │ │ + * Property: up │ │ │ │ │ + * {<OpenLayers.Pixel>} The location of the last mouseup. │ │ │ │ │ */ │ │ │ │ │ - geometryName: "the_geom", │ │ │ │ │ + up: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: maxFeatures │ │ │ │ │ - * {Integer} Optional maximum number of features to retrieve. │ │ │ │ │ + * Property: clickTolerance │ │ │ │ │ + * {Number} The number of pixels the mouse can move between mousedown │ │ │ │ │ + * and mouseup for the event to still be considered a click. │ │ │ │ │ + * Dragging the map should not trigger the click and clickout callbacks │ │ │ │ │ + * unless the map is moved by less than this tolerance. Defaults to 4. │ │ │ │ │ */ │ │ │ │ │ + clickTolerance: 4, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: schema │ │ │ │ │ - * {String} Optional schema location that will be included in the │ │ │ │ │ - * schemaLocation attribute value. Note that the feature type schema │ │ │ │ │ - * is required for a strict XML validator (on transactions with an │ │ │ │ │ - * insert for example), but is *not* required by the WFS specification │ │ │ │ │ - * (since the server is supposed to know about feature type schemas). │ │ │ │ │ + * Property: geometryTypes │ │ │ │ │ + * To restrict dragging to a limited set of geometry types, send a list │ │ │ │ │ + * of strings corresponding to the geometry class names. │ │ │ │ │ + * │ │ │ │ │ + * @type Array(String) │ │ │ │ │ */ │ │ │ │ │ - schema: null, │ │ │ │ │ + geometryTypes: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: featurePrefix │ │ │ │ │ - * {String} Namespace alias for feature type. Default is "feature". │ │ │ │ │ + * Property: stopClick │ │ │ │ │ + * {Boolean} If stopClick is set to true, handled clicks do not │ │ │ │ │ + * propagate to other click listeners. Otherwise, handled clicks │ │ │ │ │ + * do propagate. Unhandled clicks always propagate, whatever the │ │ │ │ │ + * value of stopClick. Defaults to true. │ │ │ │ │ */ │ │ │ │ │ - featurePrefix: "feature", │ │ │ │ │ + stopClick: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: formatOptions │ │ │ │ │ - * {Object} Optional options for the format. If a format is not provided, │ │ │ │ │ - * this property can be used to extend the default format options. │ │ │ │ │ + * Property: stopDown │ │ │ │ │ + * {Boolean} If stopDown is set to true, handled mousedowns do not │ │ │ │ │ + * propagate to other mousedown listeners. Otherwise, handled │ │ │ │ │ + * mousedowns do propagate. Unhandled mousedowns always propagate, │ │ │ │ │ + * whatever the value of stopDown. Defaults to true. │ │ │ │ │ */ │ │ │ │ │ - formatOptions: null, │ │ │ │ │ + stopDown: true, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readFormat │ │ │ │ │ - * {<OpenLayers.Format>} For WFS requests it is possible to get a │ │ │ │ │ - * different output format than GML. In that case, we cannot parse │ │ │ │ │ - * the response with the default format (WFST) and we need a different │ │ │ │ │ - * format for reading. │ │ │ │ │ + /** │ │ │ │ │ + * Property: stopUp │ │ │ │ │ + * {Boolean} If stopUp is set to true, handled mouseups do not │ │ │ │ │ + * propagate to other mouseup listeners. Otherwise, handled mouseups │ │ │ │ │ + * do propagate. Unhandled mouseups always propagate, whatever the │ │ │ │ │ + * value of stopUp. Defaults to false. │ │ │ │ │ */ │ │ │ │ │ - readFormat: null, │ │ │ │ │ + stopUp: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: readOptions │ │ │ │ │ - * {Object} Optional object to pass to format's read. │ │ │ │ │ + * Constructor: OpenLayers.Handler.Feature │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * control - {<OpenLayers.Control>} │ │ │ │ │ + * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ + * callbacks - {Object} An object with a 'over' property whos value is │ │ │ │ │ + * a function to be called when the mouse is over a feature. The │ │ │ │ │ + * callback should expect to recieve a single argument, the feature. │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - readOptions: null, │ │ │ │ │ + initialize: function(control, layer, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.WFS │ │ │ │ │ - * A class for giving layers WFS protocol. │ │ │ │ │ + * Method: touchstart │ │ │ │ │ + * Handle touchstart events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * url - {String} URL to send requests to (required). │ │ │ │ │ - * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ - * featureNS - {String} Feature namespace (required, but can be autodetected │ │ │ │ │ - * during the first query if GML is used as readFormat and │ │ │ │ │ - * featurePrefix is provided and matches the prefix used by the server │ │ │ │ │ - * for this featureType). │ │ │ │ │ - * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ - * for writing if featureNS is provided). Default is 'feature'. │ │ │ │ │ - * geometryName - {String} Name of geometry attribute. The default is │ │ │ │ │ - * 'the_geom' for WFS <version> 1.0, and null for higher versions. If │ │ │ │ │ - * null, it will be set to the name of the first geometry found in the │ │ │ │ │ - * first read operation. │ │ │ │ │ - * multi - {Boolean} If set to true, geometries will be casted to Multi │ │ │ │ │ - * geometries before they are written in a transaction. No casting will │ │ │ │ │ - * be done when reading features. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!options.format) { │ │ │ │ │ - this.format = OpenLayers.Format.WFST(OpenLayers.Util.extend({ │ │ │ │ │ - version: this.version, │ │ │ │ │ - featureType: this.featureType, │ │ │ │ │ - featureNS: this.featureNS, │ │ │ │ │ - featurePrefix: this.featurePrefix, │ │ │ │ │ - geometryName: this.geometryName, │ │ │ │ │ - srsName: this.srsName, │ │ │ │ │ - schema: this.schema │ │ │ │ │ - }, this.formatOptions)); │ │ │ │ │ - } │ │ │ │ │ - if (!options.geometryName && parseFloat(this.format.version) > 1.0) { │ │ │ │ │ - this.setGeometryName(null); │ │ │ │ │ - } │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + return OpenLayers.Event.isMultiTouch(evt) ? │ │ │ │ │ + true : this.mousedown(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up the protocol. │ │ │ │ │ + * Method: touchmove │ │ │ │ │ + * Handle touchmove events. We just prevent the browser default behavior, │ │ │ │ │ + * for Android Webkit not to select text when moving the finger after │ │ │ │ │ + * selecting a feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.options && !this.options.format) { │ │ │ │ │ - this.format.destroy(); │ │ │ │ │ - } │ │ │ │ │ - this.format = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Construct a request for reading new features. Since WFS splits the │ │ │ │ │ - * basic CRUD operations into GetFeature requests (for read) and │ │ │ │ │ - * Transactions (for all others), this method does not make use of the │ │ │ │ │ - * format's read method (that is only about reading transaction │ │ │ │ │ - * responses). │ │ │ │ │ - * │ │ │ │ │ + * Method: mousedown │ │ │ │ │ + * Handle mouse down. Stop propagation if a feature is targeted by this │ │ │ │ │ + * event (stops map dragging during feature selection). │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Options for the read operation, in addition to the │ │ │ │ │ - * options set on the instance (options set here will take precedence). │ │ │ │ │ - * │ │ │ │ │ - * To use a configured protocol to get e.g. a WFS hit count, applications │ │ │ │ │ - * could do the following: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * protocol.read({ │ │ │ │ │ - * readOptions: {output: "object"}, │ │ │ │ │ - * resultType: "hits", │ │ │ │ │ - * maxFeatures: null, │ │ │ │ │ - * callback: function(resp) { │ │ │ │ │ - * // process resp.numberOfFeatures here │ │ │ │ │ - * } │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * To use a configured protocol to use WFS paging (if supported by the │ │ │ │ │ - * server), applications could do the following: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * protocol.read({ │ │ │ │ │ - * startIndex: 0, │ │ │ │ │ - * count: 50 │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * To limit the attributes returned by the GetFeature request, applications │ │ │ │ │ - * can use the propertyNames option to specify the properties to include in │ │ │ │ │ - * the response: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * protocol.read({ │ │ │ │ │ - * propertyNames: ["DURATION", "INTENSITY"] │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - var data = OpenLayers.Format.XML.prototype.write.apply( │ │ │ │ │ - this.format, [this.format.writeNode("wfs:GetFeature", options)] │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: data │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - return response; │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + // Feature selection is only done with a left click. Other handlers may stop the │ │ │ │ │ + // propagation of left-click mousedown events but not right-click mousedown events. │ │ │ │ │ + // This mismatch causes problems when comparing the location of the down and up │ │ │ │ │ + // events in the click function so it is important ignore right-clicks. │ │ │ │ │ + if (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ + this.down = evt.xy; │ │ │ │ │ + } │ │ │ │ │ + return this.handle(evt) ? !this.stopDown : true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setFeatureType │ │ │ │ │ - * Change the feature type on the fly. │ │ │ │ │ - * │ │ │ │ │ + * Method: mouseup │ │ │ │ │ + * Handle mouse up. Stop propagation if a feature is targeted by this │ │ │ │ │ + * event. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * featureType - {String} Local (without prefix) feature typeName. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - setFeatureType: function(featureType) { │ │ │ │ │ - this.featureType = featureType; │ │ │ │ │ - this.format.featureType = featureType; │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + this.up = evt.xy; │ │ │ │ │ + return this.handle(evt) ? !this.stopUp : true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setGeometryName │ │ │ │ │ - * Sets the geometryName option after instantiation. │ │ │ │ │ - * │ │ │ │ │ + * Method: click │ │ │ │ │ + * Handle click. Call the "click" callback if click on a feature, │ │ │ │ │ + * or the "clickout" callback if click outside any feature. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometryName - {String} Name of geometry attribute. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - setGeometryName: function(geometryName) { │ │ │ │ │ - this.geometryName = geometryName; │ │ │ │ │ - this.format.geometryName = geometryName; │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + return this.handle(evt) ? !this.stopClick : true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleRead │ │ │ │ │ - * Deal with response from the read request. │ │ │ │ │ - * │ │ │ │ │ + * Method: mousemove │ │ │ │ │ + * Handle mouse moves. Call the "over" callback if moving in to a feature, │ │ │ │ │ + * or the "out" callback if moving out of a feature. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ - * to the user callback. │ │ │ │ │ - * options - {Object} The user options passed to the read call. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - // success │ │ │ │ │ - var result = this.parseResponse(request, options.readOptions); │ │ │ │ │ - if (result && result.success !== false) { │ │ │ │ │ - if (options.readOptions && options.readOptions.output == "object") { │ │ │ │ │ - OpenLayers.Util.extend(response, result); │ │ │ │ │ - } else { │ │ │ │ │ - response.features = result; │ │ │ │ │ - } │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ - } else { │ │ │ │ │ - // failure (service exception) │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - response.error = result; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - // failure │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - } │ │ │ │ │ - options.callback.call(options.scope, response); │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + if (!this.callbacks['over'] && !this.callbacks['out']) { │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ + this.handle(evt); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseResponse │ │ │ │ │ - * Read HTTP response body and return features │ │ │ │ │ + * Method: dblclick │ │ │ │ │ + * Handle dblclick. Call the "dblclick" callback if dblclick on a feature. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * request - {XMLHttpRequest} The request object │ │ │ │ │ - * options - {Object} Optional object to pass to format's read │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} or {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * An object with a features property, an array of features or a single │ │ │ │ │ - * feature. │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - parseResponse: function(request, options) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - var result = (this.readFormat !== null) ? this.readFormat.read(doc) : │ │ │ │ │ - this.format.read(doc, options); │ │ │ │ │ - if (!this.featureNS) { │ │ │ │ │ - var format = this.readFormat || this.format; │ │ │ │ │ - this.featureNS = format.featureNS; │ │ │ │ │ - // no need to auto-configure again on subsequent reads │ │ │ │ │ - format.autoConfig = false; │ │ │ │ │ - if (!this.geometryName) { │ │ │ │ │ - this.setGeometryName(format.geometryName); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return result; │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + return !this.handle(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: commit │ │ │ │ │ - * Given a list of feature, assemble a batch request for update, create, │ │ │ │ │ - * and delete transactions. A commit call on the prototype amounts │ │ │ │ │ - * to writing a WFS transaction - so the write method on the format │ │ │ │ │ - * is used. │ │ │ │ │ + * Method: geometryTypeMatches │ │ │ │ │ + * Return true if the geometry type of the passed feature matches │ │ │ │ │ + * one of the geometry types in the geometryTypes array. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ - * options - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * nativeElements - {Array({Object})} Array of objects with information for writing │ │ │ │ │ - * out <Native> elements, these objects have vendorId, safeToIgnore and │ │ │ │ │ - * value properties. The <Native> element is intended to allow access to │ │ │ │ │ - * vendor specific capabilities of any particular web feature server or │ │ │ │ │ - * datastore. │ │ │ │ │ + * feature - {<OpenLayers.Vector.Feature>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} A response object with a features │ │ │ │ │ - * property containing any insertIds and a priv property referencing │ │ │ │ │ - * the XMLHttpRequest object. │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - commit: function(features, options) { │ │ │ │ │ - │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "commit", │ │ │ │ │ - reqFeatures: features │ │ │ │ │ - }); │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(features, options), │ │ │ │ │ - callback: this.createCallback(this.handleCommit, response, options) │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - return response; │ │ │ │ │ + geometryTypeMatches: function(feature) { │ │ │ │ │ + return this.geometryTypes == null || │ │ │ │ │ + OpenLayers.Util.indexOf(this.geometryTypes, │ │ │ │ │ + feature.geometry.CLASS_NAME) > -1; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleCommit │ │ │ │ │ - * Called when the commit request returns. │ │ │ │ │ - * │ │ │ │ │ + * Method: handle │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ - * to the user callback. │ │ │ │ │ - * options - {Object} The user options passed to the commit call. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The event occurred over a relevant feature. │ │ │ │ │ */ │ │ │ │ │ - handleCommit: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - │ │ │ │ │ - // ensure that we have an xml doc │ │ │ │ │ - var data = request.responseXML; │ │ │ │ │ - if (!data || !data.documentElement) { │ │ │ │ │ - data = request.responseText; │ │ │ │ │ + handle: function(evt) { │ │ │ │ │ + if (this.feature && !this.feature.layer) { │ │ │ │ │ + // feature has been destroyed │ │ │ │ │ + this.feature = null; │ │ │ │ │ + } │ │ │ │ │ + var type = evt.type; │ │ │ │ │ + var handled = false; │ │ │ │ │ + var previouslyIn = !!(this.feature); // previously in a feature │ │ │ │ │ + var click = (type == "click" || type == "dblclick" || type == "touchstart"); │ │ │ │ │ + this.feature = this.layer.getFeatureFromEvent(evt); │ │ │ │ │ + if (this.feature && !this.feature.layer) { │ │ │ │ │ + // feature has been destroyed │ │ │ │ │ + this.feature = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.lastFeature && !this.lastFeature.layer) { │ │ │ │ │ + // last feature has been destroyed │ │ │ │ │ + this.lastFeature = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.feature) { │ │ │ │ │ + if (type === "touchstart") { │ │ │ │ │ + // stop the event to prevent Android Webkit from │ │ │ │ │ + // "flashing" the map div │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var obj = this.format.read(data) || {}; │ │ │ │ │ - │ │ │ │ │ - response.insertIds = obj.insertIds || []; │ │ │ │ │ - if (obj.success) { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + var inNew = (this.feature != this.lastFeature); │ │ │ │ │ + if (this.geometryTypeMatches(this.feature)) { │ │ │ │ │ + // in to a feature │ │ │ │ │ + if (previouslyIn && inNew) { │ │ │ │ │ + // out of last feature and in to another │ │ │ │ │ + if (this.lastFeature) { │ │ │ │ │ + this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ + } │ │ │ │ │ + this.triggerCallback(type, 'in', [this.feature]); │ │ │ │ │ + } else if (!previouslyIn || click) { │ │ │ │ │ + // in feature for the first time │ │ │ │ │ + this.triggerCallback(type, 'in', [this.feature]); │ │ │ │ │ + } │ │ │ │ │ + this.lastFeature = this.feature; │ │ │ │ │ + handled = true; │ │ │ │ │ } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - response.error = obj; │ │ │ │ │ + // not in to a feature │ │ │ │ │ + if (this.lastFeature && (previouslyIn && inNew || click)) { │ │ │ │ │ + // out of last feature for the first time │ │ │ │ │ + this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ + } │ │ │ │ │ + // next time the mouse goes in a feature whose geometry type │ │ │ │ │ + // doesn't match we don't want to call the 'out' callback │ │ │ │ │ + // again, so let's set this.feature to null so that │ │ │ │ │ + // previouslyIn will evaluate to false the next time │ │ │ │ │ + // we enter handle. Yes, a bit hackish... │ │ │ │ │ + this.feature = null; │ │ │ │ │ } │ │ │ │ │ - options.callback.call(options.scope, response); │ │ │ │ │ + } else if (this.lastFeature && (previouslyIn || click)) { │ │ │ │ │ + this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ } │ │ │ │ │ + return handled; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: filterDelete │ │ │ │ │ - * Send a request that deletes all features by their filter. │ │ │ │ │ - * │ │ │ │ │ + * Method: triggerCallback │ │ │ │ │ + * Call the callback keyed in the event map with the supplied arguments. │ │ │ │ │ + * For click and clickout, the <clickTolerance> is checked first. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} filter │ │ │ │ │ + * type - {String} │ │ │ │ │ */ │ │ │ │ │ - filterDelete: function(filter, options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "commit" │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - var root = this.format.createElementNSPlus("wfs:Transaction", { │ │ │ │ │ - attributes: { │ │ │ │ │ - service: "WFS", │ │ │ │ │ - version: this.version │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - var deleteNode = this.format.createElementNSPlus("wfs:Delete", { │ │ │ │ │ - attributes: { │ │ │ │ │ - typeName: (options.featureNS ? this.featurePrefix + ":" : "") + │ │ │ │ │ - options.featureType │ │ │ │ │ + triggerCallback: function(type, mode, args) { │ │ │ │ │ + var key = this.EVENTMAP[type][mode]; │ │ │ │ │ + if (key) { │ │ │ │ │ + if (type == 'click' && this.up && this.down) { │ │ │ │ │ + // for click/clickout, only trigger callback if tolerance is met │ │ │ │ │ + var dpx = Math.sqrt( │ │ │ │ │ + Math.pow(this.up.x - this.down.x, 2) + │ │ │ │ │ + Math.pow(this.up.y - this.down.y, 2) │ │ │ │ │ + ); │ │ │ │ │ + if (dpx <= this.clickTolerance) { │ │ │ │ │ + this.callback(key, args); │ │ │ │ │ + } │ │ │ │ │ + // we're done with this set of events now: clear the cached │ │ │ │ │ + // positions so we can't trip over them later (this can occur │ │ │ │ │ + // if one of the up/down events gets eaten before it gets to us │ │ │ │ │ + // but we still get the click) │ │ │ │ │ + this.up = this.down = null; │ │ │ │ │ + } else { │ │ │ │ │ + this.callback(key, args); │ │ │ │ │ } │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - if (options.featureNS) { │ │ │ │ │ - deleteNode.setAttribute("xmlns:" + this.featurePrefix, options.featureNS); │ │ │ │ │ } │ │ │ │ │ - var filterNode = this.format.writeNode("ogc:Filter", filter); │ │ │ │ │ - │ │ │ │ │ - deleteNode.appendChild(filterNode); │ │ │ │ │ - │ │ │ │ │ - root.appendChild(deleteNode); │ │ │ │ │ - │ │ │ │ │ - var data = OpenLayers.Format.XML.prototype.write.apply( │ │ │ │ │ - this.format, [root] │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - return OpenLayers.Request.POST({ │ │ │ │ │ - url: this.url, │ │ │ │ │ - callback: options.callback || function() {}, │ │ │ │ │ - data: data │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: abort │ │ │ │ │ - * Abort an ongoing request, the response object passed to │ │ │ │ │ - * this method must come from this protocol (as a result │ │ │ │ │ - * of a read, or commit operation). │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Turn on the handler. Returns false if the handler was already active. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - response.priv.abort(); │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerToTop(); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + "removelayer": this.handleMapEvents, │ │ │ │ │ + "changelayer": this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + activated = true; │ │ │ │ │ } │ │ │ │ │ + return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.WFS.v1" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WFST.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Function: OpenLayers.Format.WFST │ │ │ │ │ - * Used to create a versioned WFS protocol. Default version is 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Format>} A WFST format of the given version. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WFST = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options, OpenLayers.Format.WFST.DEFAULTS │ │ │ │ │ - ); │ │ │ │ │ - var cls = OpenLayers.Format.WFST["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported WFST version: " + options.version; │ │ │ │ │ - } │ │ │ │ │ - return new cls(options); │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Format.WFST.DEFAULTS │ │ │ │ │ - * {Object} Default properties for the WFST format. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WFST.DEFAULTS = { │ │ │ │ │ - "version": "1.0.0" │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Filter.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - * @requires OpenLayers/Style.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Filter │ │ │ │ │ - * This class represents an OGC Filter. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Filter = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Filter │ │ │ │ │ - * This class represents a generic filter. │ │ │ │ │ + /** │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Turn off the handler. Returns false if the handler was already active. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Filter>} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerBack(); │ │ │ │ │ + this.feature = null; │ │ │ │ │ + this.lastFeature = null; │ │ │ │ │ + this.down = null; │ │ │ │ │ + this.up = null; │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + "removelayer": this.handleMapEvents, │ │ │ │ │ + "changelayer": this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + deactivated = true; │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Remove reference to anything added. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() {}, │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: evaluate │ │ │ │ │ - * Evaluates this filter in a specific context. Instances or subclasses │ │ │ │ │ - * are supposed to override this method. │ │ │ │ │ + * Method: handleMapEvents │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * context - {Object} Context to use in evaluating the filter. If a vector │ │ │ │ │ - * feature is provided, the feature.attributes will be used as context. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The filter applies. │ │ │ │ │ + * evt - {Object} │ │ │ │ │ */ │ │ │ │ │ - evaluate: function(context) { │ │ │ │ │ - return true; │ │ │ │ │ + handleMapEvents: function(evt) { │ │ │ │ │ + if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ + this.moveLayerToTop(); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Clones this filter. Should be implemented by subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Filter>} Clone of this filter. │ │ │ │ │ + * Method: moveLayerToTop │ │ │ │ │ + * Moves the layer for this handler to the top, so mouse events can reach │ │ │ │ │ + * it. │ │ │ │ │ */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - return null; │ │ │ │ │ + moveLayerToTop: function() { │ │ │ │ │ + var index = Math.max(this.map.Z_INDEX_BASE['Feature'] - 1, │ │ │ │ │ + this.layer.getZIndex()) + 1; │ │ │ │ │ + this.layer.setZIndex(index); │ │ │ │ │ + │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: toString │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} Include <OpenLayers.Format.CQL> in your build to get a CQL │ │ │ │ │ - * representation of the filter returned. Otherwise "[Object object]" │ │ │ │ │ - * will be returned. │ │ │ │ │ + * Method: moveLayerBack │ │ │ │ │ + * Moves the layer back to the position determined by the map's layers │ │ │ │ │ + * array. │ │ │ │ │ */ │ │ │ │ │ - toString: function() { │ │ │ │ │ - var string; │ │ │ │ │ - if (OpenLayers.Format && OpenLayers.Format.CQL) { │ │ │ │ │ - string = OpenLayers.Format.CQL.prototype.write(this); │ │ │ │ │ + moveLayerBack: function() { │ │ │ │ │ + var index = this.layer.getZIndex() - 1; │ │ │ │ │ + if (index >= this.map.Z_INDEX_BASE['Feature']) { │ │ │ │ │ + this.layer.setZIndex(index); │ │ │ │ │ } else { │ │ │ │ │ - string = Object.prototype.toString.call(this); │ │ │ │ │ + this.map.setLayerZIndex(this.layer, │ │ │ │ │ + this.map.getLayerIndex(this.layer)); │ │ │ │ │ } │ │ │ │ │ - return string; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Filter" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Feature" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Filter/Spatial.js │ │ │ │ │ + OpenLayers/StyleMap.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Filter.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Style.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Filter.Spatial │ │ │ │ │ - * This class represents a spatial filter. │ │ │ │ │ - * Currently implemented: BBOX, DWithin and Intersects │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Filter> │ │ │ │ │ + * Class: OpenLayers.StyleMap │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Filter.Spatial = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ +OpenLayers.StyleMap = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: type │ │ │ │ │ - * {String} Type of spatial filter. │ │ │ │ │ - * │ │ │ │ │ - * The type should be one of: │ │ │ │ │ - * - OpenLayers.Filter.Spatial.BBOX │ │ │ │ │ - * - OpenLayers.Filter.Spatial.INTERSECTS │ │ │ │ │ - * - OpenLayers.Filter.Spatial.DWITHIN │ │ │ │ │ - * - OpenLayers.Filter.Spatial.WITHIN │ │ │ │ │ - * - OpenLayers.Filter.Spatial.CONTAINS │ │ │ │ │ + * Property: styles │ │ │ │ │ + * {Object} Hash of {<OpenLayers.Style>}, keyed by names of well known │ │ │ │ │ + * rendering intents (e.g. "default", "temporary", "select", "delete"). │ │ │ │ │ */ │ │ │ │ │ - type: null, │ │ │ │ │ + styles: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: property │ │ │ │ │ - * {String} Name of the context property to compare. │ │ │ │ │ + * Property: extendDefault │ │ │ │ │ + * {Boolean} if true, every render intent will extend the symbolizers │ │ │ │ │ + * specified for the "default" intent at rendering time. Otherwise, every │ │ │ │ │ + * rendering intent will be treated as a completely independent style. │ │ │ │ │ */ │ │ │ │ │ - property: null, │ │ │ │ │ + extendDefault: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: value │ │ │ │ │ - * {<OpenLayers.Bounds> || <OpenLayers.Geometry>} The bounds or geometry │ │ │ │ │ - * to be used by the filter. Use bounds for BBOX filters and geometry │ │ │ │ │ - * for INTERSECTS or DWITHIN filters. │ │ │ │ │ + * Constructor: OpenLayers.StyleMap │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * style - {Object} Optional. Either a style hash, or a style object, or │ │ │ │ │ + * a hash of style objects (style hashes) keyed by rendering │ │ │ │ │ + * intent. If just one style hash or style object is passed, │ │ │ │ │ + * this will be used for all known render intents (default, │ │ │ │ │ + * select, temporary) │ │ │ │ │ + * options - {Object} optional hash of additional options for this │ │ │ │ │ + * instance │ │ │ │ │ */ │ │ │ │ │ - value: null, │ │ │ │ │ + initialize: function(style, options) { │ │ │ │ │ + this.styles = { │ │ │ │ │ + "default": new OpenLayers.Style( │ │ │ │ │ + OpenLayers.Feature.Vector.style["default"]), │ │ │ │ │ + "select": new OpenLayers.Style( │ │ │ │ │ + OpenLayers.Feature.Vector.style["select"]), │ │ │ │ │ + "temporary": new OpenLayers.Style( │ │ │ │ │ + OpenLayers.Feature.Vector.style["temporary"]), │ │ │ │ │ + "delete": new OpenLayers.Style( │ │ │ │ │ + OpenLayers.Feature.Vector.style["delete"]) │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: distance │ │ │ │ │ - * {Number} The distance to use in a DWithin spatial filter. │ │ │ │ │ - */ │ │ │ │ │ - distance: null, │ │ │ │ │ + // take whatever the user passed as style parameter and convert it │ │ │ │ │ + // into parts of stylemap. │ │ │ │ │ + if (style instanceof OpenLayers.Style) { │ │ │ │ │ + // user passed a style object │ │ │ │ │ + this.styles["default"] = style; │ │ │ │ │ + this.styles["select"] = style; │ │ │ │ │ + this.styles["temporary"] = style; │ │ │ │ │ + this.styles["delete"] = style; │ │ │ │ │ + } else if (typeof style == "object") { │ │ │ │ │ + for (var key in style) { │ │ │ │ │ + if (style[key] instanceof OpenLayers.Style) { │ │ │ │ │ + // user passed a hash of style objects │ │ │ │ │ + this.styles[key] = style[key]; │ │ │ │ │ + } else if (typeof style[key] == "object") { │ │ │ │ │ + // user passsed a hash of style hashes │ │ │ │ │ + this.styles[key] = new OpenLayers.Style(style[key]); │ │ │ │ │ + } else { │ │ │ │ │ + // user passed a style hash (i.e. symbolizer) │ │ │ │ │ + this.styles["default"] = new OpenLayers.Style(style); │ │ │ │ │ + this.styles["select"] = new OpenLayers.Style(style); │ │ │ │ │ + this.styles["temporary"] = new OpenLayers.Style(style); │ │ │ │ │ + this.styles["delete"] = new OpenLayers.Style(style); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: distanceUnits │ │ │ │ │ - * {String} The units to use for the distance, e.g. 'm'. │ │ │ │ │ - */ │ │ │ │ │ - distanceUnits: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Filter.Spatial │ │ │ │ │ - * Creates a spatial filter. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object with properties to set on the │ │ │ │ │ - * filter. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Filter.Spatial>} │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var key in this.styles) { │ │ │ │ │ + this.styles[key].destroy(); │ │ │ │ │ + } │ │ │ │ │ + this.styles = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: evaluate │ │ │ │ │ - * Evaluates this filter for a specific feature. │ │ │ │ │ + * Method: createSymbolizer │ │ │ │ │ + * Creates the symbolizer for a feature for a render intent. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} feature to apply the filter to. │ │ │ │ │ + * feature - {<OpenLayers.Feature>} The feature to evaluate the rules │ │ │ │ │ + * of the intended style against. │ │ │ │ │ + * intent - {String} The intent determines the symbolizer that will be │ │ │ │ │ + * used to draw the feature. Well known intents are "default" │ │ │ │ │ + * (for just drawing the features), "select" (for selected │ │ │ │ │ + * features) and "temporary" (for drawing features). │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The feature meets filter criteria. │ │ │ │ │ + * {Object} symbolizer hash │ │ │ │ │ */ │ │ │ │ │ - evaluate: function(feature) { │ │ │ │ │ - var intersect = false; │ │ │ │ │ - switch (this.type) { │ │ │ │ │ - case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ - case OpenLayers.Filter.Spatial.INTERSECTS: │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - var geom = this.value; │ │ │ │ │ - if (this.value.CLASS_NAME == "OpenLayers.Bounds") { │ │ │ │ │ - geom = this.value.toGeometry(); │ │ │ │ │ - } │ │ │ │ │ - if (feature.geometry.intersects(geom)) { │ │ │ │ │ - intersect = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - throw new Error('evaluate is not implemented for this filter type.'); │ │ │ │ │ + createSymbolizer: function(feature, intent) { │ │ │ │ │ + if (!feature) { │ │ │ │ │ + feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ } │ │ │ │ │ - return intersect; │ │ │ │ │ + if (!this.styles[intent]) { │ │ │ │ │ + intent = "default"; │ │ │ │ │ + } │ │ │ │ │ + feature.renderIntent = intent; │ │ │ │ │ + var defaultSymbolizer = {}; │ │ │ │ │ + if (this.extendDefault && intent != "default") { │ │ │ │ │ + defaultSymbolizer = this.styles["default"].createSymbolizer(feature); │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Util.extend(defaultSymbolizer, │ │ │ │ │ + this.styles[intent].createSymbolizer(feature)); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Clones this filter. │ │ │ │ │ + * Method: addUniqueValueRules │ │ │ │ │ + * Convenience method to create comparison rules for unique values of a │ │ │ │ │ + * property. The rules will be added to the style object for a specified │ │ │ │ │ + * rendering intent. This method is a shortcut for creating something like │ │ │ │ │ + * the "unique value legends" familiar from well known desktop GIS systems │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Filter.Spatial>} Clone of this filter. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * renderIntent - {String} rendering intent to add the rules to │ │ │ │ │ + * property - {String} values of feature attributes to create the │ │ │ │ │ + * rules for │ │ │ │ │ + * symbolizers - {Object} Hash of symbolizers, keyed by the desired │ │ │ │ │ + * property values │ │ │ │ │ + * context - {Object} An optional object with properties that │ │ │ │ │ + * symbolizers' property values should be evaluated │ │ │ │ │ + * against. If no context is specified, feature.attributes │ │ │ │ │ + * will be used │ │ │ │ │ */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - var options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - value: this.value && this.value.clone && this.value.clone() │ │ │ │ │ - }, this); │ │ │ │ │ - return new OpenLayers.Filter.Spatial(options); │ │ │ │ │ + addUniqueValueRules: function(renderIntent, property, symbolizers, context) { │ │ │ │ │ + var rules = []; │ │ │ │ │ + for (var value in symbolizers) { │ │ │ │ │ + rules.push(new OpenLayers.Rule({ │ │ │ │ │ + symbolizer: symbolizers[value], │ │ │ │ │ + context: context, │ │ │ │ │ + filter: new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.EQUAL_TO, │ │ │ │ │ + property: property, │ │ │ │ │ + value: value │ │ │ │ │ + }) │ │ │ │ │ + })); │ │ │ │ │ + } │ │ │ │ │ + this.styles[renderIntent].addRules(rules); │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Filter.Spatial" │ │ │ │ │ -}); │ │ │ │ │ │ │ │ │ │ -OpenLayers.Filter.Spatial.BBOX = "BBOX"; │ │ │ │ │ -OpenLayers.Filter.Spatial.INTERSECTS = "INTERSECTS"; │ │ │ │ │ -OpenLayers.Filter.Spatial.DWITHIN = "DWITHIN"; │ │ │ │ │ -OpenLayers.Filter.Spatial.WITHIN = "WITHIN"; │ │ │ │ │ -OpenLayers.Filter.Spatial.CONTAINS = "CONTAINS"; │ │ │ │ │ + CLASS_NAME: "OpenLayers.StyleMap" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Filter/FeatureId.js │ │ │ │ │ + OpenLayers/Layer/Vector.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Filter.js │ │ │ │ │ + * @requires OpenLayers/Layer.js │ │ │ │ │ + * @requires OpenLayers/Renderer.js │ │ │ │ │ + * @requires OpenLayers/StyleMap.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Console.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Filter.FeatureId │ │ │ │ │ - * This class represents a ogc:FeatureId Filter, as being used for rule-based SLD │ │ │ │ │ - * styling │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Layer.Vector │ │ │ │ │ + * Instances of OpenLayers.Layer.Vector are used to render vector data from │ │ │ │ │ + * a variety of sources. Create a new vector layer with the │ │ │ │ │ + * <OpenLayers.Layer.Vector> constructor. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Filter> │ │ │ │ │ + * - <OpenLayers.Layer> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Filter.FeatureId = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ +OpenLayers.Layer.Vector = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * layer.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Listeners will be called with a reference to an event object. The │ │ │ │ │ + * properties of this event depends on exactly what happened. │ │ │ │ │ + * │ │ │ │ │ + * All event objects have at least the following properties: │ │ │ │ │ + * object - {Object} A reference to layer.events.object. │ │ │ │ │ + * element - {DOMElement} A reference to layer.events.element. │ │ │ │ │ + * │ │ │ │ │ + * Supported map event types (in addition to those from <OpenLayers.Layer.events>): │ │ │ │ │ + * beforefeatureadded - Triggered before a feature is added. Listeners │ │ │ │ │ + * will receive an object with a *feature* property referencing the │ │ │ │ │ + * feature to be added. To stop the feature from being added, a │ │ │ │ │ + * listener should return false. │ │ │ │ │ + * beforefeaturesadded - Triggered before an array of features is added. │ │ │ │ │ + * Listeners will receive an object with a *features* property │ │ │ │ │ + * referencing the feature to be added. To stop the features from │ │ │ │ │ + * being added, a listener should return false. │ │ │ │ │ + * featureadded - Triggered after a feature is added. The event │ │ │ │ │ + * object passed to listeners will have a *feature* property with a │ │ │ │ │ + * reference to the added feature. │ │ │ │ │ + * featuresadded - Triggered after features are added. The event │ │ │ │ │ + * object passed to listeners will have a *features* property with a │ │ │ │ │ + * reference to an array of added features. │ │ │ │ │ + * beforefeatureremoved - Triggered before a feature is removed. Listeners │ │ │ │ │ + * will receive an object with a *feature* property referencing the │ │ │ │ │ + * feature to be removed. │ │ │ │ │ + * beforefeaturesremoved - Triggered before multiple features are removed. │ │ │ │ │ + * Listeners will receive an object with a *features* property │ │ │ │ │ + * referencing the features to be removed. │ │ │ │ │ + * featureremoved - Triggerd after a feature is removed. The event │ │ │ │ │ + * object passed to listeners will have a *feature* property with a │ │ │ │ │ + * reference to the removed feature. │ │ │ │ │ + * featuresremoved - Triggered after features are removed. The event │ │ │ │ │ + * object passed to listeners will have a *features* property with a │ │ │ │ │ + * reference to an array of removed features. │ │ │ │ │ + * beforefeatureselected - Triggered before a feature is selected. Listeners │ │ │ │ │ + * will receive an object with a *feature* property referencing the │ │ │ │ │ + * feature to be selected. To stop the feature from being selectd, a │ │ │ │ │ + * listener should return false. │ │ │ │ │ + * featureselected - Triggered after a feature is selected. Listeners │ │ │ │ │ + * will receive an object with a *feature* property referencing the │ │ │ │ │ + * selected feature. │ │ │ │ │ + * featureunselected - Triggered after a feature is unselected. │ │ │ │ │ + * Listeners will receive an object with a *feature* property │ │ │ │ │ + * referencing the unselected feature. │ │ │ │ │ + * beforefeaturemodified - Triggered when a feature is selected to │ │ │ │ │ + * be modified. Listeners will receive an object with a *feature* │ │ │ │ │ + * property referencing the selected feature. │ │ │ │ │ + * featuremodified - Triggered when a feature has been modified. │ │ │ │ │ + * Listeners will receive an object with a *feature* property referencing │ │ │ │ │ + * the modified feature. │ │ │ │ │ + * afterfeaturemodified - Triggered when a feature is finished being modified. │ │ │ │ │ + * Listeners will receive an object with a *feature* property referencing │ │ │ │ │ + * the modified feature. │ │ │ │ │ + * vertexmodified - Triggered when a vertex within any feature geometry │ │ │ │ │ + * has been modified. Listeners will receive an object with a │ │ │ │ │ + * *feature* property referencing the modified feature, a *vertex* │ │ │ │ │ + * property referencing the vertex modified (always a point geometry), │ │ │ │ │ + * and a *pixel* property referencing the pixel location of the │ │ │ │ │ + * modification. │ │ │ │ │ + * vertexremoved - Triggered when a vertex within any feature geometry │ │ │ │ │ + * has been deleted. Listeners will receive an object with a │ │ │ │ │ + * *feature* property referencing the modified feature, a *vertex* │ │ │ │ │ + * property referencing the vertex modified (always a point geometry), │ │ │ │ │ + * and a *pixel* property referencing the pixel location of the │ │ │ │ │ + * removal. │ │ │ │ │ + * sketchstarted - Triggered when a feature sketch bound for this layer │ │ │ │ │ + * is started. Listeners will receive an object with a *feature* │ │ │ │ │ + * property referencing the new sketch feature and a *vertex* property │ │ │ │ │ + * referencing the creation point. │ │ │ │ │ + * sketchmodified - Triggered when a feature sketch bound for this layer │ │ │ │ │ + * is modified. Listeners will receive an object with a *vertex* │ │ │ │ │ + * property referencing the modified vertex and a *feature* property │ │ │ │ │ + * referencing the sketch feature. │ │ │ │ │ + * sketchcomplete - Triggered when a feature sketch bound for this layer │ │ │ │ │ + * is complete. Listeners will receive an object with a *feature* │ │ │ │ │ + * property referencing the sketch feature. By returning false, a │ │ │ │ │ + * listener can stop the sketch feature from being added to the layer. │ │ │ │ │ + * refresh - Triggered when something wants a strategy to ask the protocol │ │ │ │ │ + * for a new set of features. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} The layer is a base layer. Default is false. Set this property │ │ │ │ │ + * in the layer options. │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: fids │ │ │ │ │ - * {Array(String)} Feature Ids to evaluate this rule against. │ │ │ │ │ - * To be passed inside the params object. │ │ │ │ │ + * APIProperty: isFixed │ │ │ │ │ + * {Boolean} Whether the layer remains in one place while dragging the │ │ │ │ │ + * map. Note that setting this to true will move the layer to the bottom │ │ │ │ │ + * of the layer stack. │ │ │ │ │ */ │ │ │ │ │ - fids: null, │ │ │ │ │ + isFixed: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: type │ │ │ │ │ - * {String} Type to identify this filter. │ │ │ │ │ + * APIProperty: features │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ */ │ │ │ │ │ - type: "FID", │ │ │ │ │ + features: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Filter.FeatureId │ │ │ │ │ - * Creates an ogc:FeatureId rule. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object with properties to set on the │ │ │ │ │ - * rule │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Filter.FeatureId>} │ │ │ │ │ + * Property: filter │ │ │ │ │ + * {<OpenLayers.Filter>} The filter set in this layer, │ │ │ │ │ + * a strategy launching read requests can combined │ │ │ │ │ + * this filter with its own filter. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.fids = []; │ │ │ │ │ - OpenLayers.Filter.prototype.initialize.apply(this, [options]); │ │ │ │ │ - }, │ │ │ │ │ + filter: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: evaluate │ │ │ │ │ - * evaluates this rule for a specific feature │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature>} feature to apply the rule to. │ │ │ │ │ - * For vector features, the check is run against the fid, │ │ │ │ │ - * for plain features against the id. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true if the rule applies, false if it does not │ │ │ │ │ + /** │ │ │ │ │ + * Property: selectedFeatures │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ */ │ │ │ │ │ - evaluate: function(feature) { │ │ │ │ │ - for (var i = 0, len = this.fids.length; i < len; i++) { │ │ │ │ │ - var fid = feature.fid || feature.id; │ │ │ │ │ - if (fid == this.fids[i]) { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ + selectedFeatures: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Clones this filter. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Filter.FeatureId>} Clone of this filter. │ │ │ │ │ + * Property: unrenderedFeatures │ │ │ │ │ + * {Object} hash of features, keyed by feature.id, that the renderer │ │ │ │ │ + * failed to draw │ │ │ │ │ */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - var filter = new OpenLayers.Filter.FeatureId(); │ │ │ │ │ - OpenLayers.Util.extend(filter, this); │ │ │ │ │ - filter.fids = this.fids.slice(); │ │ │ │ │ - return filter; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Filter.FeatureId" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WFST/v1.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/WFST.js │ │ │ │ │ - * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ - * @requires OpenLayers/Filter/FeatureId.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WFST.v1 │ │ │ │ │ - * Superclass for WFST parsers. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WFST.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + unrenderedFeatures: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + * APIProperty: reportError │ │ │ │ │ + * {Boolean} report friendly error message when loading of renderer │ │ │ │ │ + * fails. │ │ │ │ │ */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - wfs: "http://www.opengis.net/wfs", │ │ │ │ │ - gml: "http://www.opengis.net/gml", │ │ │ │ │ - ogc: "http://www.opengis.net/ogc", │ │ │ │ │ - ows: "http://www.opengis.net/ows" │ │ │ │ │ - }, │ │ │ │ │ + reportError: true, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: style │ │ │ │ │ + * {Object} Default style for the layer │ │ │ │ │ */ │ │ │ │ │ - defaultPrefix: "wfs", │ │ │ │ │ + style: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} WFS version number. │ │ │ │ │ + * Property: styleMap │ │ │ │ │ + * {<OpenLayers.StyleMap>} │ │ │ │ │ */ │ │ │ │ │ - version: null, │ │ │ │ │ + styleMap: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} Schema location for a particular minor version. │ │ │ │ │ + * Property: strategies │ │ │ │ │ + * {Array(<OpenLayers.Strategy>})} Optional list of strategies for the layer. │ │ │ │ │ */ │ │ │ │ │ - schemaLocations: null, │ │ │ │ │ + strategies: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: srsName │ │ │ │ │ - * {String} URI for spatial reference system. │ │ │ │ │ + * Property: protocol │ │ │ │ │ + * {<OpenLayers.Protocol>} Optional protocol for the layer. │ │ │ │ │ */ │ │ │ │ │ - srsName: null, │ │ │ │ │ + protocol: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: extractAttributes │ │ │ │ │ - * {Boolean} Extract attributes from GML. Default is true. │ │ │ │ │ + * Property: renderers │ │ │ │ │ + * {Array(String)} List of supported Renderer classes. Add to this list to │ │ │ │ │ + * add support for additional renderers. This list is ordered: │ │ │ │ │ + * the first renderer which returns true for the 'supported()' │ │ │ │ │ + * method will be used, if not defined in the 'renderer' option. │ │ │ │ │ */ │ │ │ │ │ - extractAttributes: true, │ │ │ │ │ + renderers: ['SVG', 'VML', 'Canvas'], │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: xy │ │ │ │ │ - * {Boolean} Order of the GML coordinate true:(x,y) or false:(y,x) │ │ │ │ │ - * Changing is not recommended, a new Format should be instantiated. │ │ │ │ │ + /** │ │ │ │ │ + * Property: renderer │ │ │ │ │ + * {<OpenLayers.Renderer>} │ │ │ │ │ */ │ │ │ │ │ - xy: true, │ │ │ │ │ + renderer: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: stateName │ │ │ │ │ - * {Object} Maps feature states to node names. │ │ │ │ │ + * APIProperty: rendererOptions │ │ │ │ │ + * {Object} Options for the renderer. See {<OpenLayers.Renderer>} for │ │ │ │ │ + * supported options. │ │ │ │ │ */ │ │ │ │ │ - stateName: null, │ │ │ │ │ + rendererOptions: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WFST.v1 │ │ │ │ │ - * Instances of this class are not created directly. Use the │ │ │ │ │ - * <OpenLayers.Format.WFST.v1_0_0> or <OpenLayers.Format.WFST.v1_1_0> │ │ │ │ │ - * constructor instead. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: geometryType │ │ │ │ │ + * {String} geometryType allows you to limit the types of geometries this │ │ │ │ │ + * layer supports. This should be set to something like │ │ │ │ │ + * "OpenLayers.Geometry.Point" to limit types. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - // set state name mapping │ │ │ │ │ - this.stateName = {}; │ │ │ │ │ - this.stateName[OpenLayers.State.INSERT] = "wfs:Insert"; │ │ │ │ │ - this.stateName[OpenLayers.State.UPDATE] = "wfs:Update"; │ │ │ │ │ - this.stateName[OpenLayers.State.DELETE] = "wfs:Delete"; │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - }, │ │ │ │ │ + geometryType: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getSrsName │ │ │ │ │ + /** │ │ │ │ │ + * Property: drawn │ │ │ │ │ + * {Boolean} Whether the Vector Layer features have been drawn yet. │ │ │ │ │ */ │ │ │ │ │ - getSrsName: function(feature, options) { │ │ │ │ │ - var srsName = options && options.srsName; │ │ │ │ │ - if (!srsName) { │ │ │ │ │ - if (feature && feature.layer) { │ │ │ │ │ - srsName = feature.layer.projection.getCode(); │ │ │ │ │ - } else { │ │ │ │ │ - srsName = this.srsName; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return srsName; │ │ │ │ │ - }, │ │ │ │ │ + drawn: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: ratio │ │ │ │ │ + * {Float} This specifies the ratio of the size of the visiblity of the Vector Layer features to the size of the map. │ │ │ │ │ + */ │ │ │ │ │ + ratio: 1, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Parse the response from a transaction. Because WFS is split into │ │ │ │ │ - * Transaction requests (create, update, and delete) and GetFeature │ │ │ │ │ - * requests (read), this method handles parsing of both types of │ │ │ │ │ - * responses. │ │ │ │ │ + * Constructor: OpenLayers.Layer.Vector │ │ │ │ │ + * Create a new vector layer │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {String | Document} The WFST document to read │ │ │ │ │ - * options - {Object} Options for the reader │ │ │ │ │ - * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * output - {String} either "features" or "object". The default is │ │ │ │ │ - * "features", which means that the method will return an array of │ │ │ │ │ - * features. If set to "object", an object with a "features" property │ │ │ │ │ - * and other properties read by the parser will be returned. │ │ │ │ │ + * name - {String} A name for the layer │ │ │ │ │ + * options - {Object} Optional object with non-default properties to set on │ │ │ │ │ + * the layer. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array | Object} Output depending on the output option. │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} A new vector layer │ │ │ │ │ */ │ │ │ │ │ - read: function(data, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, { │ │ │ │ │ - output: "features" │ │ │ │ │ - }); │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + // allow user-set renderer, otherwise assign one │ │ │ │ │ + if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ + this.assignRenderer(); │ │ │ │ │ } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ + │ │ │ │ │ + // if no valid renderer found, display error │ │ │ │ │ + if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ + this.renderer = null; │ │ │ │ │ + this.displayError(); │ │ │ │ │ } │ │ │ │ │ - var obj = {}; │ │ │ │ │ - if (data) { │ │ │ │ │ - this.readNode(data, obj, true); │ │ │ │ │ + │ │ │ │ │ + if (!this.styleMap) { │ │ │ │ │ + this.styleMap = new OpenLayers.StyleMap(); │ │ │ │ │ } │ │ │ │ │ - if (obj.features && options.output === "features") { │ │ │ │ │ - obj = obj.features; │ │ │ │ │ + │ │ │ │ │ + this.features = []; │ │ │ │ │ + this.selectedFeatures = []; │ │ │ │ │ + this.unrenderedFeatures = {}; │ │ │ │ │ + │ │ │ │ │ + // Allow for custom layer behavior │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + for (var i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + this.strategies[i].setLayer(this); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return obj; │ │ │ │ │ + │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Destroy this layer │ │ │ │ │ */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wfs": { │ │ │ │ │ - "FeatureCollection": function(node, obj) { │ │ │ │ │ - obj.features = []; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoDestroy) { │ │ │ │ │ + strategy.destroy(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.strategies = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.protocol) { │ │ │ │ │ + if (this.protocol.autoDestroy) { │ │ │ │ │ + this.protocol.destroy(); │ │ │ │ │ } │ │ │ │ │ + this.protocol = null; │ │ │ │ │ + } │ │ │ │ │ + this.destroyFeatures(); │ │ │ │ │ + this.features = null; │ │ │ │ │ + this.selectedFeatures = null; │ │ │ │ │ + this.unrenderedFeatures = null; │ │ │ │ │ + if (this.renderer) { │ │ │ │ │ + this.renderer.destroy(); │ │ │ │ │ } │ │ │ │ │ + this.renderer = null; │ │ │ │ │ + this.geometryType = null; │ │ │ │ │ + this.drawn = null; │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: write │ │ │ │ │ - * Given an array of features, write a WFS transaction. This assumes │ │ │ │ │ - * the features have a state property that determines the operation │ │ │ │ │ - * type - insert, update, or delete. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} A list of features. See │ │ │ │ │ - * below for a more detailed description of the influence of the │ │ │ │ │ - * feature's *modified* property. │ │ │ │ │ - * options - {Object} │ │ │ │ │ - * │ │ │ │ │ - * feature.modified rules: │ │ │ │ │ - * If a feature has a modified property set, the following checks will be │ │ │ │ │ - * made before a feature's geometry or attribute is included in an Update │ │ │ │ │ - * transaction: │ │ │ │ │ - * - *modified* is not set at all: The geometry and all attributes will be │ │ │ │ │ - * included. │ │ │ │ │ - * - *modified.geometry* is set (null or a geometry): The geometry will be │ │ │ │ │ - * included. If *modified.attributes* is not set, all attributes will │ │ │ │ │ - * be included. │ │ │ │ │ - * - *modified.attributes* is set: Only the attributes set (i.e. to null or │ │ │ │ │ - * a value) in *modified.attributes* will be included. │ │ │ │ │ - * If *modified.geometry* is not set, the geometry will not be included. │ │ │ │ │ - * │ │ │ │ │ - * Valid options include: │ │ │ │ │ - * - *multi* {Boolean} If set to true, geometries will be casted to │ │ │ │ │ - * Multi geometries before writing. │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer. │ │ │ │ │ + * │ │ │ │ │ + * Note: Features of the layer are also cloned. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A serialized WFS transaction. │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} An exact clone of this layer │ │ │ │ │ */ │ │ │ │ │ - write: function(features, options) { │ │ │ │ │ - var node = this.writeNode("wfs:Transaction", { │ │ │ │ │ - features: features, │ │ │ │ │ - options: options │ │ │ │ │ - }); │ │ │ │ │ - var value = this.schemaLocationAttr(); │ │ │ │ │ - if (value) { │ │ │ │ │ - this.setAttributeNS( │ │ │ │ │ - node, this.namespaces["xsi"], "xsi:schemaLocation", value │ │ │ │ │ - ); │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Vector(this.name, this.getOptions()); │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [node]); │ │ │ │ │ + │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ + │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + var features = this.features; │ │ │ │ │ + var len = features.length; │ │ │ │ │ + var clonedFeatures = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + clonedFeatures[i] = features[i].clone(); │ │ │ │ │ + } │ │ │ │ │ + obj.features = clonedFeatures; │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ + * Method: refresh │ │ │ │ │ + * Ask the layer to request features again and redraw them. Triggers │ │ │ │ │ + * the refresh event if the layer is in range and visible. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} Optional object with properties for any listener of │ │ │ │ │ + * the refresh event. │ │ │ │ │ */ │ │ │ │ │ - writers: { │ │ │ │ │ - "wfs": { │ │ │ │ │ - "GetFeature": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("wfs:GetFeature", { │ │ │ │ │ - attributes: { │ │ │ │ │ - service: "WFS", │ │ │ │ │ - version: this.version, │ │ │ │ │ - handle: options && options.handle, │ │ │ │ │ - outputFormat: options && options.outputFormat, │ │ │ │ │ - maxFeatures: options && options.maxFeatures, │ │ │ │ │ - "xsi:schemaLocation": this.schemaLocationAttr(options) │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (typeof this.featureType == "string") { │ │ │ │ │ - this.writeNode("Query", options, node); │ │ │ │ │ - } else { │ │ │ │ │ - for (var i = 0, len = this.featureType.length; i < len; i++) { │ │ │ │ │ - options.featureType = this.featureType[i]; │ │ │ │ │ - this.writeNode("Query", options, node); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Transaction": function(obj) { │ │ │ │ │ - obj = obj || {}; │ │ │ │ │ - var options = obj.options || {}; │ │ │ │ │ - var node = this.createElementNSPlus("wfs:Transaction", { │ │ │ │ │ - attributes: { │ │ │ │ │ - service: "WFS", │ │ │ │ │ - version: this.version, │ │ │ │ │ - handle: options.handle │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - var i, len; │ │ │ │ │ - var features = obj.features; │ │ │ │ │ - if (features) { │ │ │ │ │ - // temporarily re-assigning geometry types │ │ │ │ │ - if (options.multi === true) { │ │ │ │ │ - OpenLayers.Util.extend(this.geometryTypes, { │ │ │ │ │ - "OpenLayers.Geometry.Point": "MultiPoint", │ │ │ │ │ - "OpenLayers.Geometry.LineString": (this.multiCurve === true) ? "MultiCurve" : "MultiLineString", │ │ │ │ │ - "OpenLayers.Geometry.Polygon": (this.multiSurface === true) ? "MultiSurface" : "MultiPolygon" │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - var name, feature; │ │ │ │ │ - for (i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - name = this.stateName[feature.state]; │ │ │ │ │ - if (name) { │ │ │ │ │ - this.writeNode(name, { │ │ │ │ │ - feature: feature, │ │ │ │ │ - options: options │ │ │ │ │ - }, node); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // switch back to original geometry types assignment │ │ │ │ │ - if (options.multi === true) { │ │ │ │ │ - this.setGeometryTypes(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (options.nativeElements) { │ │ │ │ │ - for (i = 0, len = options.nativeElements.length; i < len; ++i) { │ │ │ │ │ - this.writeNode("wfs:Native", │ │ │ │ │ - options.nativeElements[i], node); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Native": function(nativeElement) { │ │ │ │ │ - var node = this.createElementNSPlus("wfs:Native", { │ │ │ │ │ - attributes: { │ │ │ │ │ - vendorId: nativeElement.vendorId, │ │ │ │ │ - safeToIgnore: nativeElement.safeToIgnore │ │ │ │ │ - }, │ │ │ │ │ - value: nativeElement.value │ │ │ │ │ - }); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Insert": function(obj) { │ │ │ │ │ - var feature = obj.feature; │ │ │ │ │ - var options = obj.options; │ │ │ │ │ - var node = this.createElementNSPlus("wfs:Insert", { │ │ │ │ │ - attributes: { │ │ │ │ │ - handle: options && options.handle │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - this.srsName = this.getSrsName(feature); │ │ │ │ │ - this.writeNode("feature:_typeName", feature, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Update": function(obj) { │ │ │ │ │ - var feature = obj.feature; │ │ │ │ │ - var options = obj.options; │ │ │ │ │ - var node = this.createElementNSPlus("wfs:Update", { │ │ │ │ │ - attributes: { │ │ │ │ │ - handle: options && options.handle, │ │ │ │ │ - typeName: (this.featureNS ? this.featurePrefix + ":" : "") + │ │ │ │ │ - this.featureType │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (this.featureNS) { │ │ │ │ │ - node.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ - } │ │ │ │ │ + refresh: function(obj) { │ │ │ │ │ + if (this.calculateInRange() && this.visibility) { │ │ │ │ │ + this.events.triggerEvent("refresh", obj); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // add in geometry │ │ │ │ │ - var modified = feature.modified; │ │ │ │ │ - if (this.geometryName !== null && (!modified || modified.geometry !== undefined)) { │ │ │ │ │ - this.srsName = this.getSrsName(feature); │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "Property", { │ │ │ │ │ - name: this.geometryName, │ │ │ │ │ - value: feature.geometry │ │ │ │ │ - }, node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: assignRenderer │ │ │ │ │ + * Iterates through the available renderer implementations and selects │ │ │ │ │ + * and assigns the first one whose "supported()" function returns true. │ │ │ │ │ + */ │ │ │ │ │ + assignRenderer: function() { │ │ │ │ │ + for (var i = 0, len = this.renderers.length; i < len; i++) { │ │ │ │ │ + var rendererClass = this.renderers[i]; │ │ │ │ │ + var renderer = (typeof rendererClass == "function") ? │ │ │ │ │ + rendererClass : │ │ │ │ │ + OpenLayers.Renderer[rendererClass]; │ │ │ │ │ + if (renderer && renderer.prototype.supported()) { │ │ │ │ │ + this.renderer = new renderer(this.div, this.rendererOptions); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // add in attributes │ │ │ │ │ - for (var key in feature.attributes) { │ │ │ │ │ - if (feature.attributes[key] !== undefined && │ │ │ │ │ - (!modified || !modified.attributes || │ │ │ │ │ - (modified.attributes && modified.attributes[key] !== undefined))) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "Property", { │ │ │ │ │ - name: key, │ │ │ │ │ - value: feature.attributes[key] │ │ │ │ │ - }, node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: displayError │ │ │ │ │ + * Let the user know their browser isn't supported. │ │ │ │ │ + */ │ │ │ │ │ + displayError: function() { │ │ │ │ │ + if (this.reportError) { │ │ │ │ │ + OpenLayers.Console.userError(OpenLayers.i18n("browserNotSupported", { │ │ │ │ │ + renderers: this.renderers.join('\n') │ │ │ │ │ + })); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // add feature id filter │ │ │ │ │ - this.writeNode("ogc:Filter", new OpenLayers.Filter.FeatureId({ │ │ │ │ │ - fids: [feature.fid] │ │ │ │ │ - }), node); │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * The layer has been added to the map. │ │ │ │ │ + * │ │ │ │ │ + * If there is no renderer set, the layer can't be used. Remove it. │ │ │ │ │ + * Otherwise, give the renderer a reference to the map and set its size. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Property": function(obj) { │ │ │ │ │ - var node = this.createElementNSPlus("wfs:Property"); │ │ │ │ │ - this.writeNode("Name", obj.name, node); │ │ │ │ │ - if (obj.value !== null) { │ │ │ │ │ - this.writeNode("Value", obj.value, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Name": function(name) { │ │ │ │ │ - return this.createElementNSPlus("wfs:Name", { │ │ │ │ │ - value: name │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "Value": function(obj) { │ │ │ │ │ - var node; │ │ │ │ │ - if (obj instanceof OpenLayers.Geometry) { │ │ │ │ │ - node = this.createElementNSPlus("wfs:Value"); │ │ │ │ │ - var geom = this.writeNode("feature:_geometry", obj).firstChild; │ │ │ │ │ - node.appendChild(geom); │ │ │ │ │ - } else { │ │ │ │ │ - node = this.createElementNSPlus("wfs:Value", { │ │ │ │ │ - value: obj │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Delete": function(obj) { │ │ │ │ │ - var feature = obj.feature; │ │ │ │ │ - var options = obj.options; │ │ │ │ │ - var node = this.createElementNSPlus("wfs:Delete", { │ │ │ │ │ - attributes: { │ │ │ │ │ - handle: options && options.handle, │ │ │ │ │ - typeName: (this.featureNS ? this.featurePrefix + ":" : "") + │ │ │ │ │ - this.featureType │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (this.featureNS) { │ │ │ │ │ - node.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ - } │ │ │ │ │ - this.writeNode("ogc:Filter", new OpenLayers.Filter.FeatureId({ │ │ │ │ │ - fids: [feature.fid] │ │ │ │ │ - }), node); │ │ │ │ │ - return node; │ │ │ │ │ - } │ │ │ │ │ + if (!this.renderer) { │ │ │ │ │ + this.map.removeLayer(this); │ │ │ │ │ + } else { │ │ │ │ │ + this.renderer.map = this.map; │ │ │ │ │ + │ │ │ │ │ + var newSize = this.map.getSize(); │ │ │ │ │ + newSize.w = newSize.w * this.ratio; │ │ │ │ │ + newSize.h = newSize.h * this.ratio; │ │ │ │ │ + this.renderer.setSize(newSize); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: schemaLocationAttr │ │ │ │ │ - * Generate the xsi:schemaLocation attribute value. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The xsi:schemaLocation attribute or undefined if none. │ │ │ │ │ + * Method: afterAdd │ │ │ │ │ + * Called at the end of the map.addLayer sequence. At this point, the map │ │ │ │ │ + * will have a base layer. Any autoActivate strategies will be │ │ │ │ │ + * activated here. │ │ │ │ │ */ │ │ │ │ │ - schemaLocationAttr: function(options) { │ │ │ │ │ - options = OpenLayers.Util.extend({ │ │ │ │ │ - featurePrefix: this.featurePrefix, │ │ │ │ │ - schema: this.schema │ │ │ │ │ - }, options); │ │ │ │ │ - var schemaLocations = OpenLayers.Util.extend({}, this.schemaLocations); │ │ │ │ │ - if (options.schema) { │ │ │ │ │ - schemaLocations[options.featurePrefix] = options.schema; │ │ │ │ │ - } │ │ │ │ │ - var parts = []; │ │ │ │ │ - var uri; │ │ │ │ │ - for (var key in schemaLocations) { │ │ │ │ │ - uri = this.namespaces[key]; │ │ │ │ │ - if (uri) { │ │ │ │ │ - parts.push(uri + " " + schemaLocations[key]); │ │ │ │ │ + afterAdd: function() { │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoActivate) { │ │ │ │ │ + strategy.activate(); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var value = parts.join(" ") || undefined; │ │ │ │ │ - return value; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setFilterProperty │ │ │ │ │ - * Set the property of each spatial filter. │ │ │ │ │ + * Method: removeMap │ │ │ │ │ + * The layer has been removed from the map. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - setFilterProperty: function(filter) { │ │ │ │ │ - if (filter.filters) { │ │ │ │ │ - for (var i = 0, len = filter.filters.length; i < len; ++i) { │ │ │ │ │ - OpenLayers.Format.WFST.v1.prototype.setFilterProperty.call(this, filter.filters[i]); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - if (filter instanceof OpenLayers.Filter.Spatial && !filter.property) { │ │ │ │ │ - // got a spatial filter without property, so set it │ │ │ │ │ - filter.property = this.geometryName; │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + this.drawn = false; │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoActivate) { │ │ │ │ │ + strategy.deactivate(); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFST.v1" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/GML.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/MultiPoint.js │ │ │ │ │ - * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ - * @requires OpenLayers/Geometry/MultiLineString.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ - * @requires OpenLayers/Geometry/MultiPolygon.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.GML │ │ │ │ │ - * Read/Write GML. Create a new instance with the <OpenLayers.Format.GML> │ │ │ │ │ - * constructor. Supports the GML simple features profile. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.GML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: featureNS │ │ │ │ │ - * {String} Namespace used for feature attributes. Default is │ │ │ │ │ - * "http://mapserver.gis.umn.edu/mapserver". │ │ │ │ │ + * Method: onMapResize │ │ │ │ │ + * Notify the renderer of the change in size. │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - featureNS: "http://mapserver.gis.umn.edu/mapserver", │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + OpenLayers.Layer.prototype.onMapResize.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: featurePrefix │ │ │ │ │ - * {String} Namespace alias (or prefix) for feature nodes. Default is │ │ │ │ │ - * "feature". │ │ │ │ │ - */ │ │ │ │ │ - featurePrefix: "feature", │ │ │ │ │ + var newSize = this.map.getSize(); │ │ │ │ │ + newSize.w = newSize.w * this.ratio; │ │ │ │ │ + newSize.h = newSize.h * this.ratio; │ │ │ │ │ + this.renderer.setSize(newSize); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: featureName │ │ │ │ │ - * {String} Element name for features. Default is "featureMember". │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * Reset the vector layer's div so that it once again is lined up with │ │ │ │ │ + * the map. Notify the renderer of the change of extent, and in the │ │ │ │ │ + * case of a change of zoom level (resolution), have the │ │ │ │ │ + * renderer redraw features. │ │ │ │ │ + * │ │ │ │ │ + * If the layer has not yet been drawn, cycle through the layer's │ │ │ │ │ + * features and draw each one. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * zoomChanged - {Boolean} │ │ │ │ │ + * dragging - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - featureName: "featureMember", │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: layerName │ │ │ │ │ - * {String} Name of data layer. Default is "features". │ │ │ │ │ - */ │ │ │ │ │ - layerName: "features", │ │ │ │ │ + var coordSysUnchanged = true; │ │ │ │ │ + if (!dragging) { │ │ │ │ │ + this.renderer.root.style.visibility = 'hidden'; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geometryName │ │ │ │ │ - * {String} Name of geometry element. Defaults to "geometry". │ │ │ │ │ - */ │ │ │ │ │ - geometryName: "geometry", │ │ │ │ │ + var viewSize = this.map.getSize(), │ │ │ │ │ + viewWidth = viewSize.w, │ │ │ │ │ + viewHeight = viewSize.h, │ │ │ │ │ + offsetLeft = (viewWidth / 2 * this.ratio) - viewWidth / 2, │ │ │ │ │ + offsetTop = (viewHeight / 2 * this.ratio) - viewHeight / 2; │ │ │ │ │ + offsetLeft += this.map.layerContainerOriginPx.x; │ │ │ │ │ + offsetLeft = -Math.round(offsetLeft); │ │ │ │ │ + offsetTop += this.map.layerContainerOriginPx.y; │ │ │ │ │ + offsetTop = -Math.round(offsetTop); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: collectionName │ │ │ │ │ - * {String} Name of featureCollection element. │ │ │ │ │ - */ │ │ │ │ │ - collectionName: "FeatureCollection", │ │ │ │ │ + this.div.style.left = offsetLeft + 'px'; │ │ │ │ │ + this.div.style.top = offsetTop + 'px'; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: gmlns │ │ │ │ │ - * {String} GML Namespace. │ │ │ │ │ - */ │ │ │ │ │ - gmlns: "http://www.opengis.net/gml", │ │ │ │ │ + var extent = this.map.getExtent().scale(this.ratio); │ │ │ │ │ + coordSysUnchanged = this.renderer.setExtent(extent, zoomChanged); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: extractAttributes │ │ │ │ │ - * {Boolean} Extract attributes from GML. │ │ │ │ │ - */ │ │ │ │ │ - extractAttributes: true, │ │ │ │ │ + this.renderer.root.style.visibility = 'visible'; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: xy │ │ │ │ │ - * {Boolean} Order of the GML coordinate true:(x,y) or false:(y,x) │ │ │ │ │ - * Changing is not recommended, a new Format should be instantiated. │ │ │ │ │ - */ │ │ │ │ │ - xy: true, │ │ │ │ │ + // Force a reflow on gecko based browsers to prevent jump/flicker. │ │ │ │ │ + // This seems to happen on only certain configurations; it was originally │ │ │ │ │ + // noticed in FF 2.0 and Linux. │ │ │ │ │ + if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ + this.div.scrollLeft = this.div.scrollLeft; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.GML │ │ │ │ │ - * Create a new parser for GML. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - // compile regular expressions once instead of every time they are used │ │ │ │ │ - this.regExes = { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!zoomChanged && coordSysUnchanged) { │ │ │ │ │ + for (var i in this.unrenderedFeatures) { │ │ │ │ │ + var feature = this.unrenderedFeatures[i]; │ │ │ │ │ + this.drawFeature(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!this.drawn || zoomChanged || !coordSysUnchanged) { │ │ │ │ │ + this.drawn = true; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0, len = this.features.length; i < len; i++) { │ │ │ │ │ + this.renderer.locked = (i !== (len - 1)); │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + this.drawFeature(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read data from a string, and return a list of features. │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: display │ │ │ │ │ + * Hide or show the Layer │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ + * display - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - var featureNodes = this.getElementsByTagNameNS(data.documentElement, │ │ │ │ │ - this.gmlns, │ │ │ │ │ - this.featureName); │ │ │ │ │ - var features = []; │ │ │ │ │ - for (var i = 0; i < featureNodes.length; i++) { │ │ │ │ │ - var feature = this.parseFeature(featureNodes[i]); │ │ │ │ │ - if (feature) { │ │ │ │ │ - features.push(feature); │ │ │ │ │ - } │ │ │ │ │ + display: function(display) { │ │ │ │ │ + OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ + // we need to set the display style of the root in case it is attached │ │ │ │ │ + // to a foreign layer │ │ │ │ │ + var currentDisplay = this.div.style.display; │ │ │ │ │ + if (currentDisplay != this.renderer.root.style.display) { │ │ │ │ │ + this.renderer.root.style.display = currentDisplay; │ │ │ │ │ } │ │ │ │ │ - return features; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseFeature │ │ │ │ │ - * This function is the core of the GML parsing code in OpenLayers. │ │ │ │ │ - * It creates the geometries that are then attached to the returned │ │ │ │ │ - * feature, and calls parseAttributes() to get attribute data out. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: addFeatures │ │ │ │ │ + * Add Features to the layer. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} A GML feature node. │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - parseFeature: function(node) { │ │ │ │ │ - // only accept one geometry per feature - look for highest "order" │ │ │ │ │ - var order = ["MultiPolygon", "Polygon", │ │ │ │ │ - "MultiLineString", "LineString", │ │ │ │ │ - "MultiPoint", "Point", "Envelope" │ │ │ │ │ - ]; │ │ │ │ │ - // FIXME: In case we parse a feature with no geometry, but boundedBy an Envelope, │ │ │ │ │ - // this code creates a geometry derived from the Envelope. This is not correct. │ │ │ │ │ - var type, nodeList, geometry, parser; │ │ │ │ │ - for (var i = 0; i < order.length; ++i) { │ │ │ │ │ - type = order[i]; │ │ │ │ │ - nodeList = this.getElementsByTagNameNS(node, this.gmlns, type); │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - // only deal with first geometry of this type │ │ │ │ │ - parser = this.parseGeometry[type.toLowerCase()]; │ │ │ │ │ - if (parser) { │ │ │ │ │ - geometry = parser.apply(this, [nodeList[0]]); │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry.transform(this.externalProjection, │ │ │ │ │ - this.internalProjection); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - throw new TypeError("Unsupported geometry type: " + type); │ │ │ │ │ - } │ │ │ │ │ - // stop looking for different geometry types │ │ │ │ │ - break; │ │ │ │ │ + addFeatures: function(features, options) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ + if (notify) { │ │ │ │ │ + var event = { │ │ │ │ │ + features: features │ │ │ │ │ + }; │ │ │ │ │ + var ret = this.events.triggerEvent("beforefeaturesadded", event); │ │ │ │ │ + if (ret === false) { │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ + features = event.features; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - var bounds; │ │ │ │ │ - var boxNodes = this.getElementsByTagNameNS(node, this.gmlns, "Box"); │ │ │ │ │ - for (i = 0; i < boxNodes.length; ++i) { │ │ │ │ │ - var boxNode = boxNodes[i]; │ │ │ │ │ - var box = this.parseGeometry["box"].apply(this, [boxNode]); │ │ │ │ │ - var parentNode = boxNode.parentNode; │ │ │ │ │ - var parentName = parentNode.localName || │ │ │ │ │ - parentNode.nodeName.split(":").pop(); │ │ │ │ │ - if (parentName === "boundedBy") { │ │ │ │ │ - bounds = box; │ │ │ │ │ + // Track successfully added features for featuresadded event, since │ │ │ │ │ + // beforefeatureadded can veto single features. │ │ │ │ │ + var featuresAdded = []; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + if (i != (features.length - 1)) { │ │ │ │ │ + this.renderer.locked = true; │ │ │ │ │ } else { │ │ │ │ │ - geometry = box.toGeometry(); │ │ │ │ │ + this.renderer.locked = false; │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ │ │ │ │ │ - // construct feature (optionally with attributes) │ │ │ │ │ - var attributes; │ │ │ │ │ - if (this.extractAttributes) { │ │ │ │ │ - attributes = this.parseAttributes(node); │ │ │ │ │ - } │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry, attributes); │ │ │ │ │ - feature.bounds = bounds; │ │ │ │ │ + if (this.geometryType && │ │ │ │ │ + !(feature.geometry instanceof this.geometryType)) { │ │ │ │ │ + throw new TypeError('addFeatures: component should be an ' + │ │ │ │ │ + this.geometryType.prototype.CLASS_NAME); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - feature.gml = { │ │ │ │ │ - featureType: node.firstChild.nodeName.split(":")[1], │ │ │ │ │ - featureNS: node.firstChild.namespaceURI, │ │ │ │ │ - featureNSPrefix: node.firstChild.prefix │ │ │ │ │ - }; │ │ │ │ │ + //give feature reference to its layer │ │ │ │ │ + feature.layer = this; │ │ │ │ │ │ │ │ │ │ - // assign fid - this can come from a "fid" or "id" attribute │ │ │ │ │ - var childNode = node.firstChild; │ │ │ │ │ - var fid; │ │ │ │ │ - while (childNode) { │ │ │ │ │ - if (childNode.nodeType == 1) { │ │ │ │ │ - fid = childNode.getAttribute("fid") || │ │ │ │ │ - childNode.getAttribute("id"); │ │ │ │ │ - if (fid) { │ │ │ │ │ - break; │ │ │ │ │ + if (!feature.style && this.style) { │ │ │ │ │ + feature.style = OpenLayers.Util.extend({}, this.style); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (notify) { │ │ │ │ │ + if (this.events.triggerEvent("beforefeatureadded", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) === false) { │ │ │ │ │ + continue; │ │ │ │ │ } │ │ │ │ │ + this.preFeatureInsert(feature); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + featuresAdded.push(feature); │ │ │ │ │ + this.features.push(feature); │ │ │ │ │ + this.drawFeature(feature); │ │ │ │ │ + │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureadded", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.onFeatureInsert(feature); │ │ │ │ │ } │ │ │ │ │ - childNode = childNode.nextSibling; │ │ │ │ │ } │ │ │ │ │ - feature.fid = fid; │ │ │ │ │ - return feature; │ │ │ │ │ + │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresadded", { │ │ │ │ │ + features: featuresAdded │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Property: parseGeometry │ │ │ │ │ - * Properties of this object are the functions that parse geometries based │ │ │ │ │ - * on their type. │ │ │ │ │ + * APIMethod: removeFeatures │ │ │ │ │ + * Remove features from the layer. This erases any drawn features and │ │ │ │ │ + * removes them from the layer's control. The beforefeatureremoved │ │ │ │ │ + * and featureremoved events will be triggered for each feature. The │ │ │ │ │ + * featuresremoved event will be triggered after all features have │ │ │ │ │ + * been removed. To supress event triggering, use the silent option. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} List of features to be │ │ │ │ │ + * removed. │ │ │ │ │ + * options - {Object} Optional properties for changing behavior of the │ │ │ │ │ + * removal. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * silent - {Boolean} Supress event triggering. Default is false. │ │ │ │ │ */ │ │ │ │ │ - parseGeometry: { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseGeometry.point │ │ │ │ │ - * Given a GML node representing a point geometry, create an OpenLayers │ │ │ │ │ - * point geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A GML node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} A point geometry. │ │ │ │ │ - */ │ │ │ │ │ - point: function(node) { │ │ │ │ │ - /** │ │ │ │ │ - * Three coordinate variations to consider: │ │ │ │ │ - * 1) <gml:pos>x y z</gml:pos> │ │ │ │ │ - * 2) <gml:coordinates>x, y, z</gml:coordinates> │ │ │ │ │ - * 3) <gml:coord><gml:X>x</gml:X><gml:Y>y</gml:Y></gml:coord> │ │ │ │ │ - */ │ │ │ │ │ - var nodeList, coordString; │ │ │ │ │ - var coords = []; │ │ │ │ │ - │ │ │ │ │ - // look for <gml:pos> │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, this.gmlns, "pos"); │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - coordString = nodeList[0].firstChild.nodeValue; │ │ │ │ │ - coordString = coordString.replace(this.regExes.trimSpace, ""); │ │ │ │ │ - coords = coordString.split(this.regExes.splitSpace); │ │ │ │ │ - } │ │ │ │ │ + removeFeatures: function(features, options) { │ │ │ │ │ + if (!features || features.length === 0) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (features === this.features) { │ │ │ │ │ + return this.removeAllFeatures(options); │ │ │ │ │ + } │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ + } │ │ │ │ │ + if (features === this.selectedFeatures) { │ │ │ │ │ + features = features.slice(); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // look for <gml:coordinates> │ │ │ │ │ - if (coords.length == 0) { │ │ │ │ │ - nodeList = this.getElementsByTagNameNS(node, this.gmlns, │ │ │ │ │ - "coordinates"); │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - coordString = nodeList[0].firstChild.nodeValue; │ │ │ │ │ - coordString = coordString.replace(this.regExes.removeSpace, │ │ │ │ │ - ""); │ │ │ │ │ - coords = coordString.split(","); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ │ │ │ │ │ - // look for <gml:coord> │ │ │ │ │ - if (coords.length == 0) { │ │ │ │ │ - nodeList = this.getElementsByTagNameNS(node, this.gmlns, │ │ │ │ │ - "coord"); │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - var xList = this.getElementsByTagNameNS(nodeList[0], │ │ │ │ │ - this.gmlns, "X"); │ │ │ │ │ - var yList = this.getElementsByTagNameNS(nodeList[0], │ │ │ │ │ - this.gmlns, "Y"); │ │ │ │ │ - if (xList.length > 0 && yList.length > 0) { │ │ │ │ │ - coords = [xList[0].firstChild.nodeValue, │ │ │ │ │ - yList[0].firstChild.nodeValue │ │ │ │ │ - ]; │ │ │ │ │ - } │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent( │ │ │ │ │ + "beforefeaturesremoved", { │ │ │ │ │ + features: features │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // preserve third dimension │ │ │ │ │ - if (coords.length == 2) { │ │ │ │ │ - coords[2] = null; │ │ │ │ │ - } │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (this.xy) { │ │ │ │ │ - return new OpenLayers.Geometry.Point(coords[0], coords[1], │ │ │ │ │ - coords[2]); │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + // We remain locked so long as we're not at 0 │ │ │ │ │ + // and the 'next' feature has a geometry. We do the geometry check │ │ │ │ │ + // because if all the features after the current one are 'null', we │ │ │ │ │ + // won't call eraseGeometry, so we break the 'renderer functions │ │ │ │ │ + // will always be called with locked=false *last*' rule. The end result │ │ │ │ │ + // is a possible gratiutious unlocking to save a loop through the rest │ │ │ │ │ + // of the list checking the remaining features every time. So long as │ │ │ │ │ + // null geoms are rare, this is probably okay. │ │ │ │ │ + if (i != 0 && features[i - 1].geometry) { │ │ │ │ │ + this.renderer.locked = true; │ │ │ │ │ } else { │ │ │ │ │ - return new OpenLayers.Geometry.Point(coords[1], coords[0], │ │ │ │ │ - coords[2]); │ │ │ │ │ + this.renderer.locked = false; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseGeometry.multipoint │ │ │ │ │ - * Given a GML node representing a multipoint geometry, create an │ │ │ │ │ - * OpenLayers multipoint geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A GML node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.MultiPoint>} A multipoint geometry. │ │ │ │ │ - */ │ │ │ │ │ - multipoint: function(node) { │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, this.gmlns, │ │ │ │ │ - "Point"); │ │ │ │ │ - var components = []; │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - var point; │ │ │ │ │ - for (var i = 0; i < nodeList.length; ++i) { │ │ │ │ │ - point = this.parseGeometry.point.apply(this, [nodeList[i]]); │ │ │ │ │ - if (point) { │ │ │ │ │ - components.push(point); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ + │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Geometry.MultiPoint(components); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseGeometry.linestring │ │ │ │ │ - * Given a GML node representing a linestring geometry, create an │ │ │ │ │ - * OpenLayers linestring geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A GML node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.LineString>} A linestring geometry. │ │ │ │ │ - */ │ │ │ │ │ - linestring: function(node, ring) { │ │ │ │ │ - /** │ │ │ │ │ - * Two coordinate variations to consider: │ │ │ │ │ - * 1) <gml:posList dimension="d">x0 y0 z0 x1 y1 z1</gml:posList> │ │ │ │ │ - * 2) <gml:coordinates>x0, y0, z0 x1, y1, z1</gml:coordinates> │ │ │ │ │ - */ │ │ │ │ │ - var nodeList, coordString; │ │ │ │ │ - var coords = []; │ │ │ │ │ - var points = []; │ │ │ │ │ + this.features = OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ + // feature has no layer at this point │ │ │ │ │ + feature.layer = null; │ │ │ │ │ │ │ │ │ │ - // look for <gml:posList> │ │ │ │ │ - nodeList = this.getElementsByTagNameNS(node, this.gmlns, "posList"); │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - coordString = this.getChildValue(nodeList[0]); │ │ │ │ │ - coordString = coordString.replace(this.regExes.trimSpace, ""); │ │ │ │ │ - coords = coordString.split(this.regExes.splitSpace); │ │ │ │ │ - var dim = parseInt(nodeList[0].getAttribute("dimension")); │ │ │ │ │ - var j, x, y, z; │ │ │ │ │ - for (var i = 0; i < coords.length / dim; ++i) { │ │ │ │ │ - j = i * dim; │ │ │ │ │ - x = coords[j]; │ │ │ │ │ - y = coords[j + 1]; │ │ │ │ │ - z = (dim == 2) ? null : coords[j + 2]; │ │ │ │ │ - if (this.xy) { │ │ │ │ │ - points.push(new OpenLayers.Geometry.Point(x, y, z)); │ │ │ │ │ - } else { │ │ │ │ │ - points.push(new OpenLayers.Geometry.Point(y, x, z)); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + this.renderer.eraseFeatures(feature); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // look for <gml:coordinates> │ │ │ │ │ - if (coords.length == 0) { │ │ │ │ │ - nodeList = this.getElementsByTagNameNS(node, this.gmlns, │ │ │ │ │ - "coordinates"); │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - coordString = this.getChildValue(nodeList[0]); │ │ │ │ │ - coordString = coordString.replace(this.regExes.trimSpace, │ │ │ │ │ - ""); │ │ │ │ │ - coordString = coordString.replace(this.regExes.trimComma, │ │ │ │ │ - ","); │ │ │ │ │ - var pointList = coordString.split(this.regExes.splitSpace); │ │ │ │ │ - for (var i = 0; i < pointList.length; ++i) { │ │ │ │ │ - coords = pointList[i].split(","); │ │ │ │ │ - if (coords.length == 2) { │ │ │ │ │ - coords[2] = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.xy) { │ │ │ │ │ - points.push(new OpenLayers.Geometry.Point(coords[0], │ │ │ │ │ - coords[1], │ │ │ │ │ - coords[2])); │ │ │ │ │ - } else { │ │ │ │ │ - points.push(new OpenLayers.Geometry.Point(coords[1], │ │ │ │ │ - coords[0], │ │ │ │ │ - coords[2])); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + //in the case that this feature is one of the selected features, │ │ │ │ │ + // remove it from that array as well. │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.selectedFeatures, feature) != -1) { │ │ │ │ │ + OpenLayers.Util.removeItem(this.selectedFeatures, feature); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - var line = null; │ │ │ │ │ - if (points.length != 0) { │ │ │ │ │ - if (ring) { │ │ │ │ │ - line = new OpenLayers.Geometry.LinearRing(points); │ │ │ │ │ - } else { │ │ │ │ │ - line = new OpenLayers.Geometry.LineString(points); │ │ │ │ │ - } │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - return line; │ │ │ │ │ - }, │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseGeometry.multilinestring │ │ │ │ │ - * Given a GML node representing a multilinestring geometry, create an │ │ │ │ │ - * OpenLayers multilinestring geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A GML node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.MultiLineString>} A multilinestring geometry. │ │ │ │ │ - */ │ │ │ │ │ - multilinestring: function(node) { │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, this.gmlns, │ │ │ │ │ - "LineString"); │ │ │ │ │ - var components = []; │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - var line; │ │ │ │ │ - for (var i = 0; i < nodeList.length; ++i) { │ │ │ │ │ - line = this.parseGeometry.linestring.apply(this, │ │ │ │ │ - [nodeList[i]]); │ │ │ │ │ - if (line) { │ │ │ │ │ - components.push(line); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Geometry.MultiLineString(components); │ │ │ │ │ - }, │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseGeometry.polygon │ │ │ │ │ - * Given a GML node representing a polygon geometry, create an │ │ │ │ │ - * OpenLayers polygon geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A GML node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Polygon>} A polygon geometry. │ │ │ │ │ - */ │ │ │ │ │ - polygon: function(node) { │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, this.gmlns, │ │ │ │ │ - "LinearRing"); │ │ │ │ │ - var components = []; │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - // this assumes exterior ring first, inner rings after │ │ │ │ │ - var ring; │ │ │ │ │ - for (var i = 0; i < nodeList.length; ++i) { │ │ │ │ │ - ring = this.parseGeometry.linestring.apply(this, │ │ │ │ │ - [nodeList[i], true]); │ │ │ │ │ - if (ring) { │ │ │ │ │ - components.push(ring); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: removeAllFeatures │ │ │ │ │ + * Remove all features from the layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional properties for changing behavior of the │ │ │ │ │ + * removal. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * silent - {Boolean} Supress event triggering. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + removeAllFeatures: function(options) { │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ + var features = this.features; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent( │ │ │ │ │ + "beforefeaturesremoved", { │ │ │ │ │ + features: features │ │ │ │ │ } │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Geometry.Polygon(components); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseGeometry.multipolygon │ │ │ │ │ - * Given a GML node representing a multipolygon geometry, create an │ │ │ │ │ - * OpenLayers multipolygon geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A GML node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.MultiPolygon>} A multipolygon geometry. │ │ │ │ │ - */ │ │ │ │ │ - multipolygon: function(node) { │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, this.gmlns, │ │ │ │ │ - "Polygon"); │ │ │ │ │ - var components = []; │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - var polygon; │ │ │ │ │ - for (var i = 0; i < nodeList.length; ++i) { │ │ │ │ │ - polygon = this.parseGeometry.polygon.apply(this, │ │ │ │ │ - [nodeList[i]]); │ │ │ │ │ - if (polygon) { │ │ │ │ │ - components.push(polygon); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + feature.layer = null; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Geometry.MultiPolygon(components); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - envelope: function(node) { │ │ │ │ │ - var components = []; │ │ │ │ │ - var coordString; │ │ │ │ │ - var envelope; │ │ │ │ │ - │ │ │ │ │ - var lpoint = this.getElementsByTagNameNS(node, this.gmlns, "lowerCorner"); │ │ │ │ │ - if (lpoint.length > 0) { │ │ │ │ │ - var coords = []; │ │ │ │ │ - │ │ │ │ │ - if (lpoint.length > 0) { │ │ │ │ │ - coordString = lpoint[0].firstChild.nodeValue; │ │ │ │ │ - coordString = coordString.replace(this.regExes.trimSpace, ""); │ │ │ │ │ - coords = coordString.split(this.regExes.splitSpace); │ │ │ │ │ - } │ │ │ │ │ + } │ │ │ │ │ + this.renderer.clear(); │ │ │ │ │ + this.features = []; │ │ │ │ │ + this.unrenderedFeatures = {}; │ │ │ │ │ + this.selectedFeatures = []; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (coords.length == 2) { │ │ │ │ │ - coords[2] = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.xy) { │ │ │ │ │ - var lowerPoint = new OpenLayers.Geometry.Point(coords[0], coords[1], coords[2]); │ │ │ │ │ - } else { │ │ │ │ │ - var lowerPoint = new OpenLayers.Geometry.Point(coords[1], coords[0], coords[2]); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroyFeatures │ │ │ │ │ + * Erase and destroy features on the layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} An optional array of │ │ │ │ │ + * features to destroy. If not supplied, all features on the layer │ │ │ │ │ + * will be destroyed. │ │ │ │ │ + * options - {Object} │ │ │ │ │ + */ │ │ │ │ │ + destroyFeatures: function(features, options) { │ │ │ │ │ + var all = (features == undefined); // evaluates to true if │ │ │ │ │ + // features is null │ │ │ │ │ + if (all) { │ │ │ │ │ + features = this.features; │ │ │ │ │ + } │ │ │ │ │ + if (features) { │ │ │ │ │ + this.removeFeatures(features, options); │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + features[i].destroy(); │ │ │ │ │ } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var upoint = this.getElementsByTagNameNS(node, this.gmlns, "upperCorner"); │ │ │ │ │ - if (upoint.length > 0) { │ │ │ │ │ - var coords = []; │ │ │ │ │ - │ │ │ │ │ - if (upoint.length > 0) { │ │ │ │ │ - coordString = upoint[0].firstChild.nodeValue; │ │ │ │ │ - coordString = coordString.replace(this.regExes.trimSpace, ""); │ │ │ │ │ - coords = coordString.split(this.regExes.splitSpace); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (coords.length == 2) { │ │ │ │ │ - coords[2] = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.xy) { │ │ │ │ │ - var upperPoint = new OpenLayers.Geometry.Point(coords[0], coords[1], coords[2]); │ │ │ │ │ - } else { │ │ │ │ │ - var upperPoint = new OpenLayers.Geometry.Point(coords[1], coords[0], coords[2]); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: drawFeature │ │ │ │ │ + * Draw (or redraw) a feature on the layer. If the optional style argument │ │ │ │ │ + * is included, this style will be used. If no style is included, the │ │ │ │ │ + * feature's style will be used. If the feature doesn't have a style, │ │ │ │ │ + * the layer's style will be used. │ │ │ │ │ + * │ │ │ │ │ + * This function is not designed to be used when adding features to │ │ │ │ │ + * the layer (use addFeatures instead). It is meant to be used when │ │ │ │ │ + * the style of a feature has changed, or in some other way needs to │ │ │ │ │ + * visually updated *after* it has already been added to a layer. You │ │ │ │ │ + * must add the feature to the layer for most layer-related events to │ │ │ │ │ + * happen. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * style - {String | Object} Named render intent or full symbolizer object. │ │ │ │ │ + */ │ │ │ │ │ + drawFeature: function(feature, style) { │ │ │ │ │ + // don't try to draw the feature with the renderer if the layer is not │ │ │ │ │ + // drawn itself │ │ │ │ │ + if (!this.drawn) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (typeof style != "object") { │ │ │ │ │ + if (!style && feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ + style = "delete"; │ │ │ │ │ + } │ │ │ │ │ + var renderIntent = style || feature.renderIntent; │ │ │ │ │ + style = feature.style || this.style; │ │ │ │ │ + if (!style) { │ │ │ │ │ + style = this.styleMap.createSymbolizer(feature, renderIntent); │ │ │ │ │ } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (lowerPoint && upperPoint) { │ │ │ │ │ - components.push(new OpenLayers.Geometry.Point(lowerPoint.x, lowerPoint.y)); │ │ │ │ │ - components.push(new OpenLayers.Geometry.Point(upperPoint.x, lowerPoint.y)); │ │ │ │ │ - components.push(new OpenLayers.Geometry.Point(upperPoint.x, upperPoint.y)); │ │ │ │ │ - components.push(new OpenLayers.Geometry.Point(lowerPoint.x, upperPoint.y)); │ │ │ │ │ - components.push(new OpenLayers.Geometry.Point(lowerPoint.x, lowerPoint.y)); │ │ │ │ │ + var drawn = this.renderer.drawFeature(feature, style); │ │ │ │ │ + //TODO remove the check for null when we get rid of Renderer.SVG │ │ │ │ │ + if (drawn === false || drawn === null) { │ │ │ │ │ + this.unrenderedFeatures[feature.id] = feature; │ │ │ │ │ + } else { │ │ │ │ │ + delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var ring = new OpenLayers.Geometry.LinearRing(components); │ │ │ │ │ - envelope = new OpenLayers.Geometry.Polygon([ring]); │ │ │ │ │ - } │ │ │ │ │ - return envelope; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: eraseFeatures │ │ │ │ │ + * Erase features from the layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + */ │ │ │ │ │ + eraseFeatures: function(features) { │ │ │ │ │ + this.renderer.eraseFeatures(features); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseGeometry.box │ │ │ │ │ - * Given a GML node representing a box geometry, create an │ │ │ │ │ - * OpenLayers.Bounds. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A GML node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} A bounds representing the box. │ │ │ │ │ - */ │ │ │ │ │ - box: function(node) { │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, this.gmlns, │ │ │ │ │ - "coordinates"); │ │ │ │ │ - var coordString; │ │ │ │ │ - var coords, beginPoint = null, │ │ │ │ │ - endPoint = null; │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - coordString = nodeList[0].firstChild.nodeValue; │ │ │ │ │ - coords = coordString.split(" "); │ │ │ │ │ - if (coords.length == 2) { │ │ │ │ │ - beginPoint = coords[0].split(","); │ │ │ │ │ - endPoint = coords[1].split(","); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (beginPoint !== null && endPoint !== null) { │ │ │ │ │ - return new OpenLayers.Bounds(parseFloat(beginPoint[0]), │ │ │ │ │ - parseFloat(beginPoint[1]), │ │ │ │ │ - parseFloat(endPoint[0]), │ │ │ │ │ - parseFloat(endPoint[1])); │ │ │ │ │ + /** │ │ │ │ │ + * Method: getFeatureFromEvent │ │ │ │ │ + * Given an event, return a feature if the event occurred over one. │ │ │ │ │ + * Otherwise, return null. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A feature if one was under the event. │ │ │ │ │ + */ │ │ │ │ │ + getFeatureFromEvent: function(evt) { │ │ │ │ │ + if (!this.renderer) { │ │ │ │ │ + throw new Error('getFeatureFromEvent called on layer with no ' + │ │ │ │ │ + 'renderer. This usually means you destroyed a ' + │ │ │ │ │ + 'layer, but not some handler which is associated ' + │ │ │ │ │ + 'with it.'); │ │ │ │ │ + } │ │ │ │ │ + var feature = null; │ │ │ │ │ + var featureId = this.renderer.getFeatureIdFromEvent(evt); │ │ │ │ │ + if (featureId) { │ │ │ │ │ + if (typeof featureId === "string") { │ │ │ │ │ + feature = this.getFeatureById(featureId); │ │ │ │ │ + } else { │ │ │ │ │ + feature = featureId; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ + return feature; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseAttributes │ │ │ │ │ + * APIMethod: getFeatureBy │ │ │ │ │ + * Given a property value, return the feature if it exists in the features array │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ + * property - {String} │ │ │ │ │ + * value - {String} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} An attributes object. │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A feature corresponding to the given │ │ │ │ │ + * property value or null if there is no such feature. │ │ │ │ │ */ │ │ │ │ │ - parseAttributes: function(node) { │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - // assume attributes are children of the first type 1 child │ │ │ │ │ - var childNode = node.firstChild; │ │ │ │ │ - var children, i, child, grandchildren, grandchild, name, value; │ │ │ │ │ - while (childNode) { │ │ │ │ │ - if (childNode.nodeType == 1) { │ │ │ │ │ - // attributes are type 1 children with one type 3 child │ │ │ │ │ - children = childNode.childNodes; │ │ │ │ │ - for (i = 0; i < children.length; ++i) { │ │ │ │ │ - child = children[i]; │ │ │ │ │ - if (child.nodeType == 1) { │ │ │ │ │ - grandchildren = child.childNodes; │ │ │ │ │ - if (grandchildren.length == 1) { │ │ │ │ │ - grandchild = grandchildren[0]; │ │ │ │ │ - if (grandchild.nodeType == 3 || │ │ │ │ │ - grandchild.nodeType == 4) { │ │ │ │ │ - name = (child.prefix) ? │ │ │ │ │ - child.nodeName.split(":")[1] : │ │ │ │ │ - child.nodeName; │ │ │ │ │ - value = grandchild.nodeValue.replace( │ │ │ │ │ - this.regExes.trimSpace, ""); │ │ │ │ │ - attributes[name] = value; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - // If child has no childNodes (grandchildren), │ │ │ │ │ - // set an attribute with null value. │ │ │ │ │ - // e.g. <prefix:fieldname/> becomes │ │ │ │ │ - // {fieldname: null} │ │ │ │ │ - attributes[child.nodeName.split(":").pop()] = null; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + getFeatureBy: function(property, value) { │ │ │ │ │ + //TBD - would it be more efficient to use a hash for this.features? │ │ │ │ │ + var feature = null; │ │ │ │ │ + for (var i = 0, len = this.features.length; i < len; ++i) { │ │ │ │ │ + if (this.features[i][property] == value) { │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ break; │ │ │ │ │ } │ │ │ │ │ - childNode = childNode.nextSibling; │ │ │ │ │ } │ │ │ │ │ - return attributes; │ │ │ │ │ + return feature; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Generate a GML document string given a list of features. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getFeatureById │ │ │ │ │ + * Given a feature id, return the feature if it exists in the features array │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} List of features to │ │ │ │ │ - * serialize into a string. │ │ │ │ │ + * featureId - {String} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A string representing the GML document. │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A feature corresponding to the given │ │ │ │ │ + * featureId or null if there is no such feature. │ │ │ │ │ */ │ │ │ │ │ - write: function(features) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ + getFeatureById: function(featureId) { │ │ │ │ │ + return this.getFeatureBy('id', featureId); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getFeatureByFid │ │ │ │ │ + * Given a feature fid, return the feature if it exists in the features array │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureFid - {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A feature corresponding to the given │ │ │ │ │ + * featureFid or null if there is no such feature. │ │ │ │ │ + */ │ │ │ │ │ + getFeatureByFid: function(featureFid) { │ │ │ │ │ + return this.getFeatureBy('fid', featureFid); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getFeaturesByAttribute │ │ │ │ │ + * Returns an array of features that have the given attribute key set to the │ │ │ │ │ + * given value. Comparison of attribute values takes care of datatypes, e.g. │ │ │ │ │ + * the string '1234' is not equal to the number 1234. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * attrName - {String} │ │ │ │ │ + * attrValue - {Mixed} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * Array({<OpenLayers.Feature.Vector>}) An array of features that have the │ │ │ │ │ + * passed named attribute set to the given value. │ │ │ │ │ + */ │ │ │ │ │ + getFeaturesByAttribute: function(attrName, attrValue) { │ │ │ │ │ + var i, │ │ │ │ │ + feature, │ │ │ │ │ + len = this.features.length, │ │ │ │ │ + foundFeatures = []; │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + if (feature && feature.attributes) { │ │ │ │ │ + if (feature.attributes[attrName] === attrValue) { │ │ │ │ │ + foundFeatures.push(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var gml = this.createElementNS("http://www.opengis.net/wfs", │ │ │ │ │ - "wfs:" + this.collectionName); │ │ │ │ │ - for (var i = 0; i < features.length; i++) { │ │ │ │ │ - gml.appendChild(this.createFeatureXML(features[i])); │ │ │ │ │ + return foundFeatures; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Unselect the selected features │ │ │ │ │ + * i.e. clears the featureSelection array │ │ │ │ │ + * change the style back │ │ │ │ │ + clearSelection: function() { │ │ │ │ │ + │ │ │ │ │ + var vectorLayer = this.map.vectorLayer; │ │ │ │ │ + for (var i = 0; i < this.map.featureSelection.length; i++) { │ │ │ │ │ + var featureSelection = this.map.featureSelection[i]; │ │ │ │ │ + vectorLayer.drawFeature(featureSelection, vectorLayer.style); │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [gml]); │ │ │ │ │ + this.map.featureSelection = []; │ │ │ │ │ }, │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createFeatureXML │ │ │ │ │ - * Accept an OpenLayers.Feature.Vector, and build a GML node for it. │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: onFeatureInsert │ │ │ │ │ + * method called after a feature is inserted. │ │ │ │ │ + * Does nothing by default. Override this if you │ │ │ │ │ + * need to do something on feature updates. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} The feature to be built as GML. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + onFeatureInsert: function(feature) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: preFeatureInsert │ │ │ │ │ + * method called before a feature is inserted. │ │ │ │ │ + * Does nothing by default. Override this if you │ │ │ │ │ + * need to do something when features are first added to the │ │ │ │ │ + * layer, but before they are drawn, such as adjust the style. │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + preFeatureInsert: function(feature) {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getDataExtent │ │ │ │ │ + * Calculates the max extent which includes all of the features. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} A node reprensting the feature in GML. │ │ │ │ │ + * {<OpenLayers.Bounds>} or null if the layer has no features with │ │ │ │ │ + * geometries. │ │ │ │ │ */ │ │ │ │ │ - createFeatureXML: function(feature) { │ │ │ │ │ - var geometry = feature.geometry; │ │ │ │ │ - var geometryNode = this.buildGeometryNode(geometry); │ │ │ │ │ - var geomContainer = this.createElementNS(this.featureNS, │ │ │ │ │ - this.featurePrefix + ":" + │ │ │ │ │ - this.geometryName); │ │ │ │ │ - geomContainer.appendChild(geometryNode); │ │ │ │ │ - var featureNode = this.createElementNS(this.gmlns, │ │ │ │ │ - "gml:" + this.featureName); │ │ │ │ │ - var featureContainer = this.createElementNS(this.featureNS, │ │ │ │ │ - this.featurePrefix + ":" + │ │ │ │ │ - this.layerName); │ │ │ │ │ - var fid = feature.fid || feature.id; │ │ │ │ │ - featureContainer.setAttribute("fid", fid); │ │ │ │ │ - featureContainer.appendChild(geomContainer); │ │ │ │ │ - for (var attr in feature.attributes) { │ │ │ │ │ - var attrText = this.createTextNode(feature.attributes[attr]); │ │ │ │ │ - var nodename = attr.substring(attr.lastIndexOf(":") + 1); │ │ │ │ │ - var attrContainer = this.createElementNS(this.featureNS, │ │ │ │ │ - this.featurePrefix + ":" + │ │ │ │ │ - nodename); │ │ │ │ │ - attrContainer.appendChild(attrText); │ │ │ │ │ - featureContainer.appendChild(attrContainer); │ │ │ │ │ + getDataExtent: function() { │ │ │ │ │ + var maxExtent = null; │ │ │ │ │ + var features = this.features; │ │ │ │ │ + if (features && (features.length > 0)) { │ │ │ │ │ + var geometry = null; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + geometry = features[i].geometry; │ │ │ │ │ + if (geometry) { │ │ │ │ │ + if (maxExtent === null) { │ │ │ │ │ + maxExtent = new OpenLayers.Bounds(); │ │ │ │ │ + } │ │ │ │ │ + maxExtent.extend(geometry.getBounds()); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - featureNode.appendChild(featureContainer); │ │ │ │ │ - return featureNode; │ │ │ │ │ + return maxExtent; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Vector" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/Vector/RootContainer.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.Vector.RootContainer │ │ │ │ │ + * A special layer type to combine multiple vector layers inside a single │ │ │ │ │ + * renderer root container. This class is not supposed to be instantiated │ │ │ │ │ + * from user space, it is a helper class for controls that require event │ │ │ │ │ + * processing for multiple vector layers. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Vector> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Vector.RootContainer = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: displayInLayerSwitcher │ │ │ │ │ + * Set to false for this layer type │ │ │ │ │ + */ │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layers │ │ │ │ │ + * Layers that are attached to this container. Required config option. │ │ │ │ │ + */ │ │ │ │ │ + layers: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.Vector.RootContainer │ │ │ │ │ + * Create a new root container for multiple vector layer. This constructor │ │ │ │ │ + * is not supposed to be used from user space, it is only to be used by │ │ │ │ │ + * controls that need feature selection across multiple vector layers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} A name for the layer │ │ │ │ │ + * options - {Object} Optional object with non-default properties to set on │ │ │ │ │ + * the layer. │ │ │ │ │ + * │ │ │ │ │ + * Required options properties: │ │ │ │ │ + * layers - {Array(<OpenLayers.Layer.Vector>)} The layers managed by this │ │ │ │ │ + * container │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Vector.RootContainer>} A new vector layer root │ │ │ │ │ + * container │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: display │ │ │ │ │ + */ │ │ │ │ │ + display: function() {}, │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: buildGeometryNode │ │ │ │ │ + * Method: getFeatureFromEvent │ │ │ │ │ + * walk through the layers to find the feature returned by the event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} event object with a feature property │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - buildGeometryNode: function(geometry) { │ │ │ │ │ - if (this.externalProjection && this.internalProjection) { │ │ │ │ │ - geometry = geometry.clone(); │ │ │ │ │ - geometry.transform(this.internalProjection, │ │ │ │ │ - this.externalProjection); │ │ │ │ │ + getFeatureFromEvent: function(evt) { │ │ │ │ │ + var layers = this.layers; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0; i < layers.length; i++) { │ │ │ │ │ + feature = layers[i].getFeatureFromEvent(evt); │ │ │ │ │ + if (feature) { │ │ │ │ │ + return feature; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var className = geometry.CLASS_NAME; │ │ │ │ │ - var type = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ - var builder = this.buildGeometry[type.toLowerCase()]; │ │ │ │ │ - return builder.apply(this, [geometry]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: buildGeometry │ │ │ │ │ - * Object containing methods to do the actual geometry node building │ │ │ │ │ - * based on geometry type. │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - buildGeometry: { │ │ │ │ │ - // TBD retrieve the srs from layer │ │ │ │ │ - // srsName is non-standard, so not including it until it's right. │ │ │ │ │ - // gml.setAttribute("srsName", │ │ │ │ │ - // "http://www.opengis.net/gml/srs/epsg.xml#4326"); │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.point │ │ │ │ │ - * Given an OpenLayers point geometry, create a GML point. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Point>} A point geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A GML point node. │ │ │ │ │ - */ │ │ │ │ │ - point: function(geometry) { │ │ │ │ │ - var gml = this.createElementNS(this.gmlns, "gml:Point"); │ │ │ │ │ - gml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - return gml; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.multipoint │ │ │ │ │ - * Given an OpenLayers multipoint geometry, create a GML multipoint. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.MultiPoint>} A multipoint geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A GML multipoint node. │ │ │ │ │ - */ │ │ │ │ │ - multipoint: function(geometry) { │ │ │ │ │ - var gml = this.createElementNS(this.gmlns, "gml:MultiPoint"); │ │ │ │ │ - var points = geometry.components; │ │ │ │ │ - var pointMember, pointGeom; │ │ │ │ │ - for (var i = 0; i < points.length; i++) { │ │ │ │ │ - pointMember = this.createElementNS(this.gmlns, │ │ │ │ │ - "gml:pointMember"); │ │ │ │ │ - pointGeom = this.buildGeometry.point.apply(this, │ │ │ │ │ - [points[i]]); │ │ │ │ │ - pointMember.appendChild(pointGeom); │ │ │ │ │ - gml.appendChild(pointMember); │ │ │ │ │ - } │ │ │ │ │ - return gml; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.linestring │ │ │ │ │ - * Given an OpenLayers linestring geometry, create a GML linestring. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.LineString>} A linestring geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A GML linestring node. │ │ │ │ │ - */ │ │ │ │ │ - linestring: function(geometry) { │ │ │ │ │ - var gml = this.createElementNS(this.gmlns, "gml:LineString"); │ │ │ │ │ - gml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - return gml; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.multilinestring │ │ │ │ │ - * Given an OpenLayers multilinestring geometry, create a GML │ │ │ │ │ - * multilinestring. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.MultiLineString>} A multilinestring │ │ │ │ │ - * geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A GML multilinestring node. │ │ │ │ │ - */ │ │ │ │ │ - multilinestring: function(geometry) { │ │ │ │ │ - var gml = this.createElementNS(this.gmlns, "gml:MultiLineString"); │ │ │ │ │ - var lines = geometry.components; │ │ │ │ │ - var lineMember, lineGeom; │ │ │ │ │ - for (var i = 0; i < lines.length; ++i) { │ │ │ │ │ - lineMember = this.createElementNS(this.gmlns, │ │ │ │ │ - "gml:lineStringMember"); │ │ │ │ │ - lineGeom = this.buildGeometry.linestring.apply(this, │ │ │ │ │ - [lines[i]]); │ │ │ │ │ - lineMember.appendChild(lineGeom); │ │ │ │ │ - gml.appendChild(lineMember); │ │ │ │ │ - } │ │ │ │ │ - return gml; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.linearring │ │ │ │ │ - * Given an OpenLayers linearring geometry, create a GML linearring. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.LinearRing>} A linearring geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A GML linearring node. │ │ │ │ │ - */ │ │ │ │ │ - linearring: function(geometry) { │ │ │ │ │ - var gml = this.createElementNS(this.gmlns, "gml:LinearRing"); │ │ │ │ │ - gml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - return gml; │ │ │ │ │ - }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.collectRoots(); │ │ │ │ │ + map.events.register("changelayer", this, this.handleChangeLayer); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.polygon │ │ │ │ │ - * Given an OpenLayers polygon geometry, create a GML polygon. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Polygon>} A polygon geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A GML polygon node. │ │ │ │ │ - */ │ │ │ │ │ - polygon: function(geometry) { │ │ │ │ │ - var gml = this.createElementNS(this.gmlns, "gml:Polygon"); │ │ │ │ │ - var rings = geometry.components; │ │ │ │ │ - var ringMember, ringGeom, type; │ │ │ │ │ - for (var i = 0; i < rings.length; ++i) { │ │ │ │ │ - type = (i == 0) ? "outerBoundaryIs" : "innerBoundaryIs"; │ │ │ │ │ - ringMember = this.createElementNS(this.gmlns, │ │ │ │ │ - "gml:" + type); │ │ │ │ │ - ringGeom = this.buildGeometry.linearring.apply(this, │ │ │ │ │ - [rings[i]]); │ │ │ │ │ - ringMember.appendChild(ringGeom); │ │ │ │ │ - gml.appendChild(ringMember); │ │ │ │ │ - } │ │ │ │ │ - return gml; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeMap │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + map.events.unregister("changelayer", this, this.handleChangeLayer); │ │ │ │ │ + this.resetRoots(); │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.multipolygon │ │ │ │ │ - * Given an OpenLayers multipolygon geometry, create a GML multipolygon. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.MultiPolygon>} A multipolygon │ │ │ │ │ - * geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A GML multipolygon node. │ │ │ │ │ - */ │ │ │ │ │ - multipolygon: function(geometry) { │ │ │ │ │ - var gml = this.createElementNS(this.gmlns, "gml:MultiPolygon"); │ │ │ │ │ - var polys = geometry.components; │ │ │ │ │ - var polyMember, polyGeom; │ │ │ │ │ - for (var i = 0; i < polys.length; ++i) { │ │ │ │ │ - polyMember = this.createElementNS(this.gmlns, │ │ │ │ │ - "gml:polygonMember"); │ │ │ │ │ - polyGeom = this.buildGeometry.polygon.apply(this, │ │ │ │ │ - [polys[i]]); │ │ │ │ │ - polyMember.appendChild(polyGeom); │ │ │ │ │ - gml.appendChild(polyMember); │ │ │ │ │ + /** │ │ │ │ │ + * Method: collectRoots │ │ │ │ │ + * Collects the root nodes of all layers this control is configured with │ │ │ │ │ + * and moveswien the nodes to this control's layer │ │ │ │ │ + */ │ │ │ │ │ + collectRoots: function() { │ │ │ │ │ + var layer; │ │ │ │ │ + // walk through all map layers, because we want to keep the order │ │ │ │ │ + for (var i = 0; i < this.map.layers.length; ++i) { │ │ │ │ │ + layer = this.map.layers[i]; │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ + layer.renderer.moveRoot(this.renderer); │ │ │ │ │ } │ │ │ │ │ - return gml; │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.bounds │ │ │ │ │ - * Given an OpenLayers bounds, create a GML box. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Geometry.Bounds>} A bounds object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A GML box node. │ │ │ │ │ - */ │ │ │ │ │ - bounds: function(bounds) { │ │ │ │ │ - var gml = this.createElementNS(this.gmlns, "gml:Box"); │ │ │ │ │ - gml.appendChild(this.buildCoordinatesNode(bounds)); │ │ │ │ │ - return gml; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildCoordinates │ │ │ │ │ - * builds the coordinates XmlNode │ │ │ │ │ - * (code) │ │ │ │ │ - * <gml:coordinates decimal="." cs="," ts=" ">...</gml:coordinates> │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {XmlNode} created xmlNode │ │ │ │ │ + * Method: resetRoots │ │ │ │ │ + * Resets the root nodes back into the layers they belong to. │ │ │ │ │ */ │ │ │ │ │ - buildCoordinatesNode: function(geometry) { │ │ │ │ │ - var coordinatesNode = this.createElementNS(this.gmlns, │ │ │ │ │ - "gml:coordinates"); │ │ │ │ │ - coordinatesNode.setAttribute("decimal", "."); │ │ │ │ │ - coordinatesNode.setAttribute("cs", ","); │ │ │ │ │ - coordinatesNode.setAttribute("ts", " "); │ │ │ │ │ - │ │ │ │ │ - var parts = []; │ │ │ │ │ - │ │ │ │ │ - if (geometry instanceof OpenLayers.Bounds) { │ │ │ │ │ - parts.push(geometry.left + "," + geometry.bottom); │ │ │ │ │ - parts.push(geometry.right + "," + geometry.top); │ │ │ │ │ - } else { │ │ │ │ │ - var points = (geometry.components) ? geometry.components : [geometry]; │ │ │ │ │ - for (var i = 0; i < points.length; i++) { │ │ │ │ │ - parts.push(points[i].x + "," + points[i].y); │ │ │ │ │ + resetRoots: function() { │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = 0; i < this.layers.length; ++i) { │ │ │ │ │ + layer = this.layers[i]; │ │ │ │ │ + if (this.renderer && layer.renderer.getRenderLayerId() == this.id) { │ │ │ │ │ + this.renderer.moveRoot(layer.renderer); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var txtNode = this.createTextNode(parts.join(" ")); │ │ │ │ │ - coordinatesNode.appendChild(txtNode); │ │ │ │ │ - │ │ │ │ │ - return coordinatesNode; │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleChangeLayer │ │ │ │ │ + * Event handler for the map's changelayer event. We need to rebuild │ │ │ │ │ + * this container's layer dom if order of one of its layers changes. │ │ │ │ │ + * This handler is added with the setMap method, and removed with the │ │ │ │ │ + * removeMap method. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} │ │ │ │ │ + */ │ │ │ │ │ + handleChangeLayer: function(evt) { │ │ │ │ │ + var layer = evt.layer; │ │ │ │ │ + if (evt.property == "order" && │ │ │ │ │ + OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ + this.resetRoots(); │ │ │ │ │ + this.collectRoots(); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.GML" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Vector.RootContainer" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/GML/Base.js │ │ │ │ │ + OpenLayers/Control/SelectFeature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/GML.js │ │ │ │ │ - */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Though required in the full build, if the GML format is excluded, we set │ │ │ │ │ - * the namespace here. │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Handler/Feature.js │ │ │ │ │ + * @requires OpenLayers/Layer/Vector/RootContainer.js │ │ │ │ │ */ │ │ │ │ │ -if (!OpenLayers.Format.GML) { │ │ │ │ │ - OpenLayers.Format.GML = {}; │ │ │ │ │ -} │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.GML.Base │ │ │ │ │ - * Superclass for GML parsers. │ │ │ │ │ + * Class: OpenLayers.Control.SelectFeature │ │ │ │ │ + * The SelectFeature control selects vector features from a given layer on │ │ │ │ │ + * click or hover. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.GML.Base = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Control.SelectFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ + * beforefeaturehighlighted - Triggered before a feature is highlighted │ │ │ │ │ + * featurehighlighted - Triggered when a feature is highlighted │ │ │ │ │ + * featureunhighlighted - Triggered when a feature is unhighlighted │ │ │ │ │ + * boxselectionstart - Triggered before box selection starts │ │ │ │ │ + * boxselectionend - Triggered after box selection ends │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + * Property: multipleKey │ │ │ │ │ + * {String} An event modifier ('altKey' or 'shiftKey') that temporarily sets │ │ │ │ │ + * the <multiple> property to true. Default is null. │ │ │ │ │ */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - gml: "http://www.opengis.net/gml", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - wfs: "http://www.opengis.net/wfs" // this is a convenience for reading wfs:FeatureCollection │ │ │ │ │ - }, │ │ │ │ │ + multipleKey: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ + * Property: toggleKey │ │ │ │ │ + * {String} An event modifier ('altKey' or 'shiftKey') that temporarily sets │ │ │ │ │ + * the <toggle> property to true. Default is null. │ │ │ │ │ */ │ │ │ │ │ - defaultPrefix: "gml", │ │ │ │ │ + toggleKey: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} Schema location for a particular minor version. │ │ │ │ │ + * APIProperty: multiple │ │ │ │ │ + * {Boolean} Allow selection of multiple geometries. Default is false. │ │ │ │ │ */ │ │ │ │ │ - schemaLocation: null, │ │ │ │ │ + multiple: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: featureType │ │ │ │ │ - * {Array(String) or String} The local (without prefix) feature typeName(s). │ │ │ │ │ + * APIProperty: clickout │ │ │ │ │ + * {Boolean} Unselect features when clicking outside any feature. │ │ │ │ │ + * Default is true. │ │ │ │ │ */ │ │ │ │ │ - featureType: null, │ │ │ │ │ + clickout: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: featureNS │ │ │ │ │ - * {String} The feature namespace. Must be set in the options at │ │ │ │ │ - * construction. │ │ │ │ │ + * APIProperty: toggle │ │ │ │ │ + * {Boolean} Unselect a selected feature on click. Default is false. Only │ │ │ │ │ + * has meaning if hover is false. │ │ │ │ │ */ │ │ │ │ │ - featureNS: null, │ │ │ │ │ + toggle: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: geometry │ │ │ │ │ - * {String} Name of geometry element. Defaults to "geometry". If null, it │ │ │ │ │ - * will be set on <read> when the first geometry is parsed. │ │ │ │ │ + * APIProperty: hover │ │ │ │ │ + * {Boolean} Select on mouse over and deselect on mouse out. If true, this │ │ │ │ │ + * ignores clicks and only listens to mouse moves. │ │ │ │ │ */ │ │ │ │ │ - geometryName: "geometry", │ │ │ │ │ + hover: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: extractAttributes │ │ │ │ │ - * {Boolean} Extract attributes from GML. Default is true. │ │ │ │ │ + * APIProperty: highlightOnly │ │ │ │ │ + * {Boolean} If true do not actually select features (that is place them in │ │ │ │ │ + * the layer's selected features array), just highlight them. This property │ │ │ │ │ + * has no effect if hover is false. Defaults to false. │ │ │ │ │ */ │ │ │ │ │ - extractAttributes: true, │ │ │ │ │ + highlightOnly: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: srsName │ │ │ │ │ - * {String} URI for spatial reference system. This is optional for │ │ │ │ │ - * single part geometries and mandatory for collections and multis. │ │ │ │ │ - * If set, the srsName attribute will be written for all geometries. │ │ │ │ │ - * Default is null. │ │ │ │ │ + * APIProperty: box │ │ │ │ │ + * {Boolean} Allow feature selection by drawing a box. │ │ │ │ │ */ │ │ │ │ │ - srsName: null, │ │ │ │ │ + box: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: xy │ │ │ │ │ - * {Boolean} Order of the GML coordinate true:(x,y) or false:(y,x) │ │ │ │ │ - * Changing is not recommended, a new Format should be instantiated. │ │ │ │ │ + * Property: onBeforeSelect │ │ │ │ │ + * {Function} Optional function to be called before a feature is selected. │ │ │ │ │ + * The function should expect to be called with a feature. │ │ │ │ │ */ │ │ │ │ │ - xy: true, │ │ │ │ │ + onBeforeSelect: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: geometryTypes │ │ │ │ │ - * {Object} Maps OpenLayers geometry class names to GML element names. │ │ │ │ │ - * Use <setGeometryTypes> before accessing this property. │ │ │ │ │ + * APIProperty: onSelect │ │ │ │ │ + * {Function} Optional function to be called when a feature is selected. │ │ │ │ │ + * The function should expect to be called with a feature. │ │ │ │ │ + */ │ │ │ │ │ + onSelect: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: onUnselect │ │ │ │ │ + * {Function} Optional function to be called when a feature is unselected. │ │ │ │ │ + * The function should expect to be called with a feature. │ │ │ │ │ + */ │ │ │ │ │ + onUnselect: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: scope │ │ │ │ │ + * {Object} The scope to use with the onBeforeSelect, onSelect, onUnselect │ │ │ │ │ + * callbacks. If null the scope will be this control. │ │ │ │ │ + */ │ │ │ │ │ + scope: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: geometryTypes │ │ │ │ │ + * {Array(String)} To restrict selecting to a limited set of geometry types, │ │ │ │ │ + * send a list of strings corresponding to the geometry class names. │ │ │ │ │ */ │ │ │ │ │ geometryTypes: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: singleFeatureType │ │ │ │ │ - * {Boolean} True if there is only 1 featureType, and not an array │ │ │ │ │ - * of featuretypes. │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} The vector layer with a common renderer │ │ │ │ │ + * root for all layers this control is configured with (if an array of │ │ │ │ │ + * layers was passed to the constructor), or the vector layer the control │ │ │ │ │ + * was configured with (if a single layer was passed to the constructor). │ │ │ │ │ */ │ │ │ │ │ - singleFeatureType: null, │ │ │ │ │ + layer: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: autoConfig │ │ │ │ │ - * {Boolean} Indicates if the format was configured without a <featureNS>, │ │ │ │ │ - * but auto-configured <featureNS> and <featureType> during read. │ │ │ │ │ - * Subclasses making use of <featureType> auto-configuration should make │ │ │ │ │ - * the first call to the <readNode> method (usually in the read method) │ │ │ │ │ - * with true as 3rd argument, so the auto-configured featureType can be │ │ │ │ │ - * reset and the format can be reused for subsequent reads with data from │ │ │ │ │ - * different featureTypes. Set to false after read if you want to keep the │ │ │ │ │ - * auto-configured values. │ │ │ │ │ + * Property: layers │ │ │ │ │ + * {Array(<OpenLayers.Layer.Vector>)} The layers this control will work on, │ │ │ │ │ + * or null if the control was configured with a single layer │ │ │ │ │ */ │ │ │ │ │ + layers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ + * APIProperty: callbacks │ │ │ │ │ + * {Object} The functions that are sent to the handlers.feature for callback │ │ │ │ │ */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g), │ │ │ │ │ - featureMember: (/^(.*:)?featureMembers?$/) │ │ │ │ │ - }, │ │ │ │ │ + callbacks: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.GML.Base │ │ │ │ │ - * Instances of this class are not created directly. Use the │ │ │ │ │ - * <OpenLayers.Format.GML.v2> or <OpenLayers.Format.GML.v3> constructor │ │ │ │ │ - * instead. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * featureType - {Array(String) or String} Local (without prefix) feature │ │ │ │ │ - * typeName(s) (required for write). │ │ │ │ │ - * featureNS - {String} Feature namespace (required for write). │ │ │ │ │ - * geometryName - {String} Geometry element name (required for write). │ │ │ │ │ + * APIProperty: selectStyle │ │ │ │ │ + * {Object} Hash of styles │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.setGeometryTypes(); │ │ │ │ │ - if (options && options.featureNS) { │ │ │ │ │ - this.setNamespace("feature", options.featureNS); │ │ │ │ │ - } │ │ │ │ │ - this.singleFeatureType = !options || (typeof options.featureType === "string"); │ │ │ │ │ - }, │ │ │ │ │ + selectStyle: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read │ │ │ │ │ + * Property: renderIntent │ │ │ │ │ + * {String} key used to retrieve the select style from the layer's │ │ │ │ │ + * style map. │ │ │ │ │ + */ │ │ │ │ │ + renderIntent: "select", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: handlers │ │ │ │ │ + * {Object} Object with references to multiple <OpenLayers.Handler> │ │ │ │ │ + * instances. │ │ │ │ │ + */ │ │ │ │ │ + handlers: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.SelectFeature │ │ │ │ │ + * Create a new control for selecting features. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {DOMElement} A gml:featureMember element, a gml:featureMembers │ │ │ │ │ - * element, or an element containing either of the above at any level. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ + * layers - {<OpenLayers.Layer.Vector>}, or an array of vector layers. The │ │ │ │ │ + * layer(s) this control will select features from. │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + initialize: function(layers, options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + │ │ │ │ │ + if (this.scope === null) { │ │ │ │ │ + this.scope = this; │ │ │ │ │ } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ + this.initLayer(layers); │ │ │ │ │ + var callbacks = { │ │ │ │ │ + click: this.clickFeature, │ │ │ │ │ + clickout: this.clickoutFeature │ │ │ │ │ + }; │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + callbacks.over = this.overFeature; │ │ │ │ │ + callbacks.out = this.outFeature; │ │ │ │ │ } │ │ │ │ │ - var features = []; │ │ │ │ │ - this.readNode(data, { │ │ │ │ │ - features: features │ │ │ │ │ - }, true); │ │ │ │ │ - if (features.length == 0) { │ │ │ │ │ - // look for gml:featureMember elements │ │ │ │ │ - var elements = this.getElementsByTagNameNS( │ │ │ │ │ - data, this.namespaces.gml, "featureMember" │ │ │ │ │ - ); │ │ │ │ │ - if (elements.length) { │ │ │ │ │ - for (var i = 0, len = elements.length; i < len; ++i) { │ │ │ │ │ - this.readNode(elements[i], { │ │ │ │ │ - features: features │ │ │ │ │ - }, true); │ │ │ │ │ + │ │ │ │ │ + this.callbacks = OpenLayers.Util.extend(callbacks, this.callbacks); │ │ │ │ │ + this.handlers = { │ │ │ │ │ + feature: new OpenLayers.Handler.Feature( │ │ │ │ │ + this, this.layer, this.callbacks, { │ │ │ │ │ + geometryTypes: this.geometryTypes │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - // look for gml:featureMembers elements (this is v3, but does no harm here) │ │ │ │ │ - var elements = this.getElementsByTagNameNS( │ │ │ │ │ - data, this.namespaces.gml, "featureMembers" │ │ │ │ │ - ); │ │ │ │ │ - if (elements.length) { │ │ │ │ │ - // there can be only one │ │ │ │ │ - this.readNode(elements[0], { │ │ │ │ │ - features: features │ │ │ │ │ - }, true); │ │ │ │ │ + ) │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + if (this.box) { │ │ │ │ │ + this.handlers.box = new OpenLayers.Handler.Box( │ │ │ │ │ + this, { │ │ │ │ │ + done: this.selectBox │ │ │ │ │ + }, { │ │ │ │ │ + boxDivClassName: "olHandlerBoxSelectFeature" │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - return features; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: readNode │ │ │ │ │ - * Shorthand for applying one of the named readers given the node │ │ │ │ │ - * namespace and local name. Readers take two args (node, obj) and │ │ │ │ │ - * generally extend or modify the second. │ │ │ │ │ + * Method: initLayer │ │ │ │ │ + * Assign the layer property. If layers is an array, we need to use │ │ │ │ │ + * a RootContainer. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} The node to be read (required). │ │ │ │ │ - * obj - {Object} The object to be modified (optional). │ │ │ │ │ - * first - {Boolean} Should be set to true for the first node read. This │ │ │ │ │ - * is usually the readNode call in the read method. Without this being │ │ │ │ │ - * set, auto-configured properties will stick on subsequent reads. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The input object, modified (or a new one if none was provided). │ │ │ │ │ + * layers - {<OpenLayers.Layer.Vector>}, or an array of vector layers. │ │ │ │ │ */ │ │ │ │ │ - readNode: function(node, obj, first) { │ │ │ │ │ - // on subsequent calls of format.read(), we want to reset auto- │ │ │ │ │ - // configured properties and auto-configure again. │ │ │ │ │ - if (first === true && this.autoConfig === true) { │ │ │ │ │ - this.featureType = null; │ │ │ │ │ - delete this.namespaceAlias[this.featureNS]; │ │ │ │ │ - delete this.namespaces["feature"]; │ │ │ │ │ - this.featureNS = null; │ │ │ │ │ + initLayer: function(layers) { │ │ │ │ │ + if (OpenLayers.Util.isArray(layers)) { │ │ │ │ │ + this.layers = layers; │ │ │ │ │ + this.layer = new OpenLayers.Layer.Vector.RootContainer( │ │ │ │ │ + this.id + "_container", { │ │ │ │ │ + layers: layers │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + this.layer = layers; │ │ │ │ │ } │ │ │ │ │ - // featureType auto-configuration │ │ │ │ │ - if (!this.featureNS && (!(node.prefix in this.namespaces) && │ │ │ │ │ - node.parentNode.namespaceURI == this.namespaces["gml"] && │ │ │ │ │ - this.regExes.featureMember.test(node.parentNode.nodeName))) { │ │ │ │ │ - this.featureType = node.nodeName.split(":").pop(); │ │ │ │ │ - this.setNamespace("feature", node.namespaceURI); │ │ │ │ │ - this.featureNS = node.namespaceURI; │ │ │ │ │ - this.autoConfig = true; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.active && this.layers) { │ │ │ │ │ + this.map.removeLayer(this.layer); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + if (this.layers) { │ │ │ │ │ + this.layer.destroy(); │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.readNode.apply(this, [node, obj]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activates the control. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The control was effectively activated. │ │ │ │ │ */ │ │ │ │ │ - readers: { │ │ │ │ │ - "gml": { │ │ │ │ │ - "_inherit": function(node, obj, container) { │ │ │ │ │ - // To be implemented by version specific parsers │ │ │ │ │ - }, │ │ │ │ │ - "featureMember": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "featureMembers": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "name": function(node, obj) { │ │ │ │ │ - obj.name = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "boundedBy": function(node, obj) { │ │ │ │ │ - var container = {}; │ │ │ │ │ - this.readChildNodes(node, container); │ │ │ │ │ - if (container.components && container.components.length > 0) { │ │ │ │ │ - obj.bounds = container.components[0]; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Point": function(node, container) { │ │ │ │ │ - var obj = { │ │ │ │ │ - points: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - if (!container.components) { │ │ │ │ │ - container.components = []; │ │ │ │ │ - } │ │ │ │ │ - container.components.push(obj.points[0]); │ │ │ │ │ - }, │ │ │ │ │ - "coordinates": function(node, obj) { │ │ │ │ │ - var str = this.getChildValue(node).replace( │ │ │ │ │ - this.regExes.trimSpace, "" │ │ │ │ │ - ); │ │ │ │ │ - str = str.replace(this.regExes.trimComma, ","); │ │ │ │ │ - var pointList = str.split(this.regExes.splitSpace); │ │ │ │ │ - var coords; │ │ │ │ │ - var numPoints = pointList.length; │ │ │ │ │ - var points = new Array(numPoints); │ │ │ │ │ - for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ - coords = pointList[i].split(","); │ │ │ │ │ - if (this.xy) { │ │ │ │ │ - points[i] = new OpenLayers.Geometry.Point( │ │ │ │ │ - coords[0], coords[1], coords[2] │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - points[i] = new OpenLayers.Geometry.Point( │ │ │ │ │ - coords[1], coords[0], coords[2] │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - obj.points = points; │ │ │ │ │ - }, │ │ │ │ │ - "coord": function(node, obj) { │ │ │ │ │ - var coord = {}; │ │ │ │ │ - this.readChildNodes(node, coord); │ │ │ │ │ - if (!obj.points) { │ │ │ │ │ - obj.points = []; │ │ │ │ │ - } │ │ │ │ │ - obj.points.push(new OpenLayers.Geometry.Point( │ │ │ │ │ - coord.x, coord.y, coord.z │ │ │ │ │ - )); │ │ │ │ │ - }, │ │ │ │ │ - "X": function(node, coord) { │ │ │ │ │ - coord.x = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Y": function(node, coord) { │ │ │ │ │ - coord.y = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Z": function(node, coord) { │ │ │ │ │ - coord.z = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "MultiPoint": function(node, container) { │ │ │ │ │ - var obj = { │ │ │ │ │ - components: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readers.gml._inherit.apply(this, [node, obj, container]); │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - container.components = [ │ │ │ │ │ - new OpenLayers.Geometry.MultiPoint(obj.components) │ │ │ │ │ - ]; │ │ │ │ │ - }, │ │ │ │ │ - "pointMember": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "LineString": function(node, container) { │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readers.gml._inherit.apply(this, [node, obj, container]); │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - if (!container.components) { │ │ │ │ │ - container.components = []; │ │ │ │ │ - } │ │ │ │ │ - container.components.push( │ │ │ │ │ - new OpenLayers.Geometry.LineString(obj.points) │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - "MultiLineString": function(node, container) { │ │ │ │ │ - var obj = { │ │ │ │ │ - components: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readers.gml._inherit.apply(this, [node, obj, container]); │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - container.components = [ │ │ │ │ │ - new OpenLayers.Geometry.MultiLineString(obj.components) │ │ │ │ │ - ]; │ │ │ │ │ - }, │ │ │ │ │ - "lineStringMember": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "Polygon": function(node, container) { │ │ │ │ │ - var obj = { │ │ │ │ │ - outer: null, │ │ │ │ │ - inner: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readers.gml._inherit.apply(this, [node, obj, container]); │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - obj.inner.unshift(obj.outer); │ │ │ │ │ - if (!container.components) { │ │ │ │ │ - container.components = []; │ │ │ │ │ - } │ │ │ │ │ - container.components.push( │ │ │ │ │ - new OpenLayers.Geometry.Polygon(obj.inner) │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - "LinearRing": function(node, obj) { │ │ │ │ │ - var container = {}; │ │ │ │ │ - this.readers.gml._inherit.apply(this, [node, container]); │ │ │ │ │ - this.readChildNodes(node, container); │ │ │ │ │ - obj.components = [new OpenLayers.Geometry.LinearRing( │ │ │ │ │ - container.points │ │ │ │ │ - )]; │ │ │ │ │ - }, │ │ │ │ │ - "MultiPolygon": function(node, container) { │ │ │ │ │ - var obj = { │ │ │ │ │ - components: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readers.gml._inherit.apply(this, [node, obj, container]); │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - container.components = [ │ │ │ │ │ - new OpenLayers.Geometry.MultiPolygon(obj.components) │ │ │ │ │ - ]; │ │ │ │ │ - }, │ │ │ │ │ - "polygonMember": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "GeometryCollection": function(node, container) { │ │ │ │ │ - var obj = { │ │ │ │ │ - components: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readers.gml._inherit.apply(this, [node, obj, container]); │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - container.components = [ │ │ │ │ │ - new OpenLayers.Geometry.Collection(obj.components) │ │ │ │ │ - ]; │ │ │ │ │ - }, │ │ │ │ │ - "geometryMember": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "feature": { │ │ │ │ │ - "*": function(node, obj) { │ │ │ │ │ - // The node can either be named like the featureType, or it │ │ │ │ │ - // can be a child of the feature:featureType. Children can be │ │ │ │ │ - // geometry or attributes. │ │ │ │ │ - var name; │ │ │ │ │ - var local = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ - // Since an attribute can have the same name as the feature type │ │ │ │ │ - // we only want to read the node as a feature if the parent │ │ │ │ │ - // node can have feature nodes as children. In this case, the │ │ │ │ │ - // obj.features property is set. │ │ │ │ │ - if (obj.features) { │ │ │ │ │ - if (!this.singleFeatureType && │ │ │ │ │ - (OpenLayers.Util.indexOf(this.featureType, local) !== -1)) { │ │ │ │ │ - name = "_typeName"; │ │ │ │ │ - } else if (local === this.featureType) { │ │ │ │ │ - name = "_typeName"; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - // Assume attribute elements have one child node and that the child │ │ │ │ │ - // is a text node. Otherwise assume it is a geometry node. │ │ │ │ │ - if (node.childNodes.length == 0 || │ │ │ │ │ - (node.childNodes.length == 1 && node.firstChild.nodeType == 3)) { │ │ │ │ │ - if (this.extractAttributes) { │ │ │ │ │ - name = "_attribute"; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - name = "_geometry"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (name) { │ │ │ │ │ - this.readers.feature[name].apply(this, [node, obj]); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "_typeName": function(node, obj) { │ │ │ │ │ - var container = { │ │ │ │ │ - components: [], │ │ │ │ │ - attributes: {} │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, container); │ │ │ │ │ - // look for common gml namespaced elements │ │ │ │ │ - if (container.name) { │ │ │ │ │ - container.attributes.name = container.name; │ │ │ │ │ - } │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector( │ │ │ │ │ - container.components[0], container.attributes │ │ │ │ │ - ); │ │ │ │ │ - if (!this.singleFeatureType) { │ │ │ │ │ - feature.type = node.nodeName.split(":").pop(); │ │ │ │ │ - feature.namespace = node.namespaceURI; │ │ │ │ │ - } │ │ │ │ │ - var fid = node.getAttribute("fid") || │ │ │ │ │ - this.getAttributeNS(node, this.namespaces["gml"], "id"); │ │ │ │ │ - if (fid) { │ │ │ │ │ - feature.fid = fid; │ │ │ │ │ - } │ │ │ │ │ - if (this.internalProjection && this.externalProjection && │ │ │ │ │ - feature.geometry) { │ │ │ │ │ - feature.geometry.transform( │ │ │ │ │ - this.externalProjection, this.internalProjection │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (container.bounds) { │ │ │ │ │ - feature.bounds = container.bounds; │ │ │ │ │ - } │ │ │ │ │ - obj.features.push(feature); │ │ │ │ │ - }, │ │ │ │ │ - "_geometry": function(node, obj) { │ │ │ │ │ - if (!this.geometryName) { │ │ │ │ │ - this.geometryName = node.nodeName.split(":").pop(); │ │ │ │ │ - } │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "_attribute": function(node, obj) { │ │ │ │ │ - var local = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ - var value = this.getChildValue(node); │ │ │ │ │ - obj.attributes[local] = value; │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + if (this.layers) { │ │ │ │ │ + this.map.addLayer(this.layer); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "wfs": { │ │ │ │ │ - "FeatureCollection": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ + this.handlers.feature.activate(); │ │ │ │ │ + if (this.box && this.handlers.box) { │ │ │ │ │ + this.handlers.box.activate(); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return OpenLayers.Control.prototype.activate.apply( │ │ │ │ │ + this, arguments │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: write │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>) | OpenLayers.Feature.Vector} │ │ │ │ │ - * An array of features or a single feature. │ │ │ │ │ - * │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivates the control. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} Given an array of features, a doc with a gml:featureMembers │ │ │ │ │ - * element will be returned. Given a single feature, a doc with a │ │ │ │ │ - * gml:featureMember element will be returned. │ │ │ │ │ + * {Boolean} The control was effectively deactivated. │ │ │ │ │ */ │ │ │ │ │ - write: function(features) { │ │ │ │ │ - var name; │ │ │ │ │ - if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ - name = "featureMembers"; │ │ │ │ │ - } else { │ │ │ │ │ - name = "featureMember"; │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.handlers.feature.deactivate(); │ │ │ │ │ + if (this.handlers.box) { │ │ │ │ │ + this.handlers.box.deactivate(); │ │ │ │ │ + } │ │ │ │ │ + if (this.layers) { │ │ │ │ │ + this.map.removeLayer(this.layer); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var root = this.writeNode("gml:" + name, features); │ │ │ │ │ - this.setAttributeNS( │ │ │ │ │ - root, this.namespaces["xsi"], │ │ │ │ │ - "xsi:schemaLocation", this.schemaLocation │ │ │ │ │ + return OpenLayers.Control.prototype.deactivate.apply( │ │ │ │ │ + this, arguments │ │ │ │ │ ); │ │ │ │ │ - │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [root]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ + * Method: unselectAll │ │ │ │ │ + * Unselect all selected features. To unselect all except for a single │ │ │ │ │ + * feature, set the options.except property to the feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional configuration object. │ │ │ │ │ */ │ │ │ │ │ - writers: { │ │ │ │ │ - "gml": { │ │ │ │ │ - "featureMember": function(feature) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:featureMember"); │ │ │ │ │ - this.writeNode("feature:_typeName", feature, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "MultiPoint": function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:MultiPoint"); │ │ │ │ │ - var components = geometry.components || [geometry]; │ │ │ │ │ - for (var i = 0, ii = components.length; i < ii; ++i) { │ │ │ │ │ - this.writeNode("pointMember", components[i], node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "pointMember": function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:pointMember"); │ │ │ │ │ - this.writeNode("Point", geometry, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "MultiLineString": function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:MultiLineString"); │ │ │ │ │ - var components = geometry.components || [geometry]; │ │ │ │ │ - for (var i = 0, ii = components.length; i < ii; ++i) { │ │ │ │ │ - this.writeNode("lineStringMember", components[i], node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "lineStringMember": function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:lineStringMember"); │ │ │ │ │ - this.writeNode("LineString", geometry, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "MultiPolygon": function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:MultiPolygon"); │ │ │ │ │ - var components = geometry.components || [geometry]; │ │ │ │ │ - for (var i = 0, ii = components.length; i < ii; ++i) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "polygonMember", components[i], node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "polygonMember": function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:polygonMember"); │ │ │ │ │ - this.writeNode("Polygon", geometry, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "GeometryCollection": function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:GeometryCollection"); │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; ++i) { │ │ │ │ │ - this.writeNode("geometryMember", geometry.components[i], node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "geometryMember": function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:geometryMember"); │ │ │ │ │ - var child = this.writeNode("feature:_geometry", geometry); │ │ │ │ │ - node.appendChild(child.firstChild); │ │ │ │ │ - return node; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "feature": { │ │ │ │ │ - "_typeName": function(feature) { │ │ │ │ │ - var node = this.createElementNSPlus("feature:" + this.featureType, { │ │ │ │ │ - attributes: { │ │ │ │ │ - fid: feature.fid │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - this.writeNode("feature:_geometry", feature.geometry, node); │ │ │ │ │ - } │ │ │ │ │ - for (var name in feature.attributes) { │ │ │ │ │ - var value = feature.attributes[name]; │ │ │ │ │ - if (value != null) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "feature:_attribute", { │ │ │ │ │ - name: name, │ │ │ │ │ - value: value │ │ │ │ │ - }, node │ │ │ │ │ - ); │ │ │ │ │ + unselectAll: function(options) { │ │ │ │ │ + // we'll want an option to supress notification here │ │ │ │ │ + var layers = this.layers || [this.layer], │ │ │ │ │ + layer, feature, l, numExcept; │ │ │ │ │ + for (l = 0; l < layers.length; ++l) { │ │ │ │ │ + layer = layers[l]; │ │ │ │ │ + numExcept = 0; │ │ │ │ │ + //layer.selectedFeatures is null when layer is destroyed and │ │ │ │ │ + //one of it's preremovelayer listener calls setLayer │ │ │ │ │ + //with another layer on this control │ │ │ │ │ + if (layer.selectedFeatures != null) { │ │ │ │ │ + while (layer.selectedFeatures.length > numExcept) { │ │ │ │ │ + feature = layer.selectedFeatures[numExcept]; │ │ │ │ │ + if (!options || options.except != feature) { │ │ │ │ │ + this.unselect(feature); │ │ │ │ │ + } else { │ │ │ │ │ + ++numExcept; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "_geometry": function(geometry) { │ │ │ │ │ - if (this.externalProjection && this.internalProjection) { │ │ │ │ │ - geometry = geometry.clone().transform( │ │ │ │ │ - this.internalProjection, this.externalProjection │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - var node = this.createElementNSPlus( │ │ │ │ │ - "feature:" + this.geometryName │ │ │ │ │ - ); │ │ │ │ │ - var type = this.geometryTypes[geometry.CLASS_NAME]; │ │ │ │ │ - var child = this.writeNode("gml:" + type, geometry, node); │ │ │ │ │ - if (this.srsName) { │ │ │ │ │ - child.setAttribute("srsName", this.srsName); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "_attribute": function(obj) { │ │ │ │ │ - return this.createElementNSPlus("feature:" + obj.name, { │ │ │ │ │ - value: obj.value │ │ │ │ │ - }); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "wfs": { │ │ │ │ │ - "FeatureCollection": function(features) { │ │ │ │ │ - /** │ │ │ │ │ - * This is only here because GML2 only describes abstract │ │ │ │ │ - * feature collections. Typically, you would not be using │ │ │ │ │ - * the GML format to write wfs elements. This just provides │ │ │ │ │ - * some way to write out lists of features. GML3 defines the │ │ │ │ │ - * featureMembers element, so that is used by default instead. │ │ │ │ │ - */ │ │ │ │ │ - var node = this.createElementNSPlus("wfs:FeatureCollection"); │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - this.writeNode("gml:featureMember", features[i], node); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clickFeature │ │ │ │ │ + * Called on click in a feature │ │ │ │ │ + * Only responds if this.hover is false. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + clickFeature: function(feature) { │ │ │ │ │ + if (!this.hover) { │ │ │ │ │ + var selected = (OpenLayers.Util.indexOf( │ │ │ │ │ + feature.layer.selectedFeatures, feature) > -1); │ │ │ │ │ + if (selected) { │ │ │ │ │ + if (this.toggleSelect()) { │ │ │ │ │ + this.unselect(feature); │ │ │ │ │ + } else if (!this.multipleSelect()) { │ │ │ │ │ + this.unselectAll({ │ │ │ │ │ + except: feature │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ + } else { │ │ │ │ │ + if (!this.multipleSelect()) { │ │ │ │ │ + this.unselectAll({ │ │ │ │ │ + except: feature │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + this.select(feature); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setGeometryTypes │ │ │ │ │ - * Sets the <geometryTypes> mapping. │ │ │ │ │ + * Method: multipleSelect │ │ │ │ │ + * Allow for multiple selected features based on <multiple> property and │ │ │ │ │ + * <multipleKey> event modifier. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow for multiple selected features. │ │ │ │ │ */ │ │ │ │ │ - setGeometryTypes: function() { │ │ │ │ │ - this.geometryTypes = { │ │ │ │ │ - "OpenLayers.Geometry.Point": "Point", │ │ │ │ │ - "OpenLayers.Geometry.MultiPoint": "MultiPoint", │ │ │ │ │ - "OpenLayers.Geometry.LineString": "LineString", │ │ │ │ │ - "OpenLayers.Geometry.MultiLineString": "MultiLineString", │ │ │ │ │ - "OpenLayers.Geometry.Polygon": "Polygon", │ │ │ │ │ - "OpenLayers.Geometry.MultiPolygon": "MultiPolygon", │ │ │ │ │ - "OpenLayers.Geometry.Collection": "GeometryCollection" │ │ │ │ │ - }; │ │ │ │ │ + multipleSelect: function() { │ │ │ │ │ + return this.multiple || (this.handlers.feature.evt && │ │ │ │ │ + this.handlers.feature.evt[this.multipleKey]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.GML.Base" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/GML/v2.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/GML/Base.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.GML.v2 │ │ │ │ │ - * Parses GML version 2. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.GML.Base> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.GML.v2 = OpenLayers.Class(OpenLayers.Format.GML.Base, { │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} Schema location for a particular minor version. │ │ │ │ │ + * Method: toggleSelect │ │ │ │ │ + * Event should toggle the selected state of a feature based on <toggle> │ │ │ │ │ + * property and <toggleKey> event modifier. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Toggle the selected state of a feature. │ │ │ │ │ */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/gml http://schemas.opengis.net/gml/2.1.2/feature.xsd", │ │ │ │ │ + toggleSelect: function() { │ │ │ │ │ + return this.toggle || (this.handlers.feature.evt && │ │ │ │ │ + this.handlers.feature.evt[this.toggleKey]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.GML.v2 │ │ │ │ │ - * Create a parser for GML v2. │ │ │ │ │ + * Method: clickoutFeature │ │ │ │ │ + * Called on click outside a previously clicked (selected) feature. │ │ │ │ │ + * Only responds if this.hover is false. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * feature - {<OpenLayers.Vector.Feature>} │ │ │ │ │ + */ │ │ │ │ │ + clickoutFeature: function(feature) { │ │ │ │ │ + if (!this.hover && this.clickout) { │ │ │ │ │ + this.unselectAll(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: overFeature │ │ │ │ │ + * Called on over a feature. │ │ │ │ │ + * Only responds if this.hover is true. │ │ │ │ │ * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ - * featureNS - {String} Feature namespace (required). │ │ │ │ │ - * geometryName - {String} Geometry element name. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.GML.Base.prototype.initialize.apply(this, [options]); │ │ │ │ │ + overFeature: function(feature) { │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + if (this.highlightOnly) { │ │ │ │ │ + this.highlight(feature); │ │ │ │ │ + } else if (OpenLayers.Util.indexOf( │ │ │ │ │ + layer.selectedFeatures, feature) == -1) { │ │ │ │ │ + this.select(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ + * Method: outFeature │ │ │ │ │ + * Called on out of a selected feature. │ │ │ │ │ + * Only responds if this.hover is true. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - readers: { │ │ │ │ │ - "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "outerBoundaryIs": function(node, container) { │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - container.outer = obj.components[0]; │ │ │ │ │ - }, │ │ │ │ │ - "innerBoundaryIs": function(node, container) { │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - container.inner.push(obj.components[0]); │ │ │ │ │ - }, │ │ │ │ │ - "Box": function(node, container) { │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - if (!container.components) { │ │ │ │ │ - container.components = []; │ │ │ │ │ + outFeature: function(feature) { │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + if (this.highlightOnly) { │ │ │ │ │ + // we do nothing if we're not the last highlighter of the │ │ │ │ │ + // feature │ │ │ │ │ + if (feature._lastHighlighter == this.id) { │ │ │ │ │ + // if another select control had highlighted the feature before │ │ │ │ │ + // we did it ourself then we use that control to highlight the │ │ │ │ │ + // feature as it was before we highlighted it, else we just │ │ │ │ │ + // unhighlight it │ │ │ │ │ + if (feature._prevHighlighter && │ │ │ │ │ + feature._prevHighlighter != this.id) { │ │ │ │ │ + delete feature._lastHighlighter; │ │ │ │ │ + var control = this.map.getControl( │ │ │ │ │ + feature._prevHighlighter); │ │ │ │ │ + if (control) { │ │ │ │ │ + control.highlight(feature); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this.unhighlight(feature); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var min = obj.points[0]; │ │ │ │ │ - var max = obj.points[1]; │ │ │ │ │ - container.components.push( │ │ │ │ │ - new OpenLayers.Bounds(min.x, min.y, max.x, max.y) │ │ │ │ │ - ); │ │ │ │ │ + } else { │ │ │ │ │ + this.unselect(feature); │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.GML.Base.prototype.readers["gml"]), │ │ │ │ │ - "feature": OpenLayers.Format.GML.Base.prototype.readers["feature"], │ │ │ │ │ - "wfs": OpenLayers.Format.GML.Base.prototype.readers["wfs"] │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: write │ │ │ │ │ + * Method: highlight │ │ │ │ │ + * Redraw feature with the select style. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>) | OpenLayers.Feature.Vector} │ │ │ │ │ - * An array of features or a single feature. │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + highlight: function(feature) { │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + var cont = this.events.triggerEvent("beforefeaturehighlighted", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + feature._prevHighlighter = feature._lastHighlighter; │ │ │ │ │ + feature._lastHighlighter = this.id; │ │ │ │ │ + var style = this.selectStyle || this.renderIntent; │ │ │ │ │ + layer.drawFeature(feature, style); │ │ │ │ │ + this.events.triggerEvent("featurehighlighted", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: unhighlight │ │ │ │ │ + * Redraw feature with the "default" style │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} Given an array of features, a doc with a gml:featureMembers │ │ │ │ │ - * element will be returned. Given a single feature, a doc with a │ │ │ │ │ - * gml:featureMember element will be returned. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - write: function(features) { │ │ │ │ │ - var name; │ │ │ │ │ - if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ - // GML2 only has abstract feature collections │ │ │ │ │ - // wfs provides a feature collection from a well-known schema │ │ │ │ │ - name = "wfs:FeatureCollection"; │ │ │ │ │ + unhighlight: function(feature) { │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + // three cases: │ │ │ │ │ + // 1. there's no other highlighter, in that case _prev is undefined, │ │ │ │ │ + // and we just need to undef _last │ │ │ │ │ + // 2. another control highlighted the feature after we did it, in │ │ │ │ │ + // that case _last references this other control, and we just │ │ │ │ │ + // need to undef _prev │ │ │ │ │ + // 3. another control highlighted the feature before we did it, in │ │ │ │ │ + // that case _prev references this other control, and we need to │ │ │ │ │ + // set _last to _prev and undef _prev │ │ │ │ │ + if (feature._prevHighlighter == undefined) { │ │ │ │ │ + delete feature._lastHighlighter; │ │ │ │ │ + } else if (feature._prevHighlighter == this.id) { │ │ │ │ │ + delete feature._prevHighlighter; │ │ │ │ │ } else { │ │ │ │ │ - name = "gml:featureMember"; │ │ │ │ │ + feature._lastHighlighter = feature._prevHighlighter; │ │ │ │ │ + delete feature._prevHighlighter; │ │ │ │ │ } │ │ │ │ │ - var root = this.writeNode(name, features); │ │ │ │ │ - this.setAttributeNS( │ │ │ │ │ - root, this.namespaces["xsi"], │ │ │ │ │ - "xsi:schemaLocation", this.schemaLocation │ │ │ │ │ - ); │ │ │ │ │ + layer.drawFeature(feature, feature.style || feature.layer.style || │ │ │ │ │ + "default"); │ │ │ │ │ + this.events.triggerEvent("featureunhighlighted", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [root]); │ │ │ │ │ + /** │ │ │ │ │ + * Method: select │ │ │ │ │ + * Add feature to the layer's selectedFeature array, render the feature as │ │ │ │ │ + * selected, and call the onSelect function. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + select: function(feature) { │ │ │ │ │ + var cont = this.onBeforeSelect.call(this.scope, feature); │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + cont = layer.events.triggerEvent("beforefeatureselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + layer.selectedFeatures.push(feature); │ │ │ │ │ + this.highlight(feature); │ │ │ │ │ + // if the feature handler isn't involved in the feature │ │ │ │ │ + // selection (because the box handler is used or the │ │ │ │ │ + // feature is selected programatically) we fake the │ │ │ │ │ + // feature handler to allow unselecting on click │ │ │ │ │ + if (!this.handlers.feature.lastFeature) { │ │ │ │ │ + this.handlers.feature.lastFeature = layer.selectedFeatures[0]; │ │ │ │ │ + } │ │ │ │ │ + layer.events.triggerEvent("featureselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.onSelect.call(this.scope, feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ + * Method: unselect │ │ │ │ │ + * Remove feature from the layer's selectedFeature array, render the feature as │ │ │ │ │ + * normal, and call the onUnselect function. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - writers: { │ │ │ │ │ - "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "Point": function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:Point"); │ │ │ │ │ - this.writeNode("coordinates", [geometry], node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "coordinates": function(points) { │ │ │ │ │ - var numPoints = points.length; │ │ │ │ │ - var parts = new Array(numPoints); │ │ │ │ │ - var point; │ │ │ │ │ - for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ - point = points[i]; │ │ │ │ │ - if (this.xy) { │ │ │ │ │ - parts[i] = point.x + "," + point.y; │ │ │ │ │ - } else { │ │ │ │ │ - parts[i] = point.y + "," + point.x; │ │ │ │ │ + unselect: function(feature) { │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + // Store feature style for restoration later │ │ │ │ │ + this.unhighlight(feature); │ │ │ │ │ + OpenLayers.Util.removeItem(layer.selectedFeatures, feature); │ │ │ │ │ + layer.events.triggerEvent("featureunselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.onUnselect.call(this.scope, feature); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: selectBox │ │ │ │ │ + * Callback from the handlers.box set up when <box> selection is true │ │ │ │ │ + * on. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * position - {<OpenLayers.Bounds> || <OpenLayers.Pixel> } │ │ │ │ │ + */ │ │ │ │ │ + selectBox: function(position) { │ │ │ │ │ + if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ + var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.left, │ │ │ │ │ + y: position.bottom │ │ │ │ │ + }); │ │ │ │ │ + var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.right, │ │ │ │ │ + y: position.top │ │ │ │ │ + }); │ │ │ │ │ + var bounds = new OpenLayers.Bounds( │ │ │ │ │ + minXY.lon, minXY.lat, maxXY.lon, maxXY.lat │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + // if multiple is false, first deselect currently selected features │ │ │ │ │ + if (!this.multipleSelect()) { │ │ │ │ │ + this.unselectAll(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // because we're using a box, we consider we want multiple selection │ │ │ │ │ + var prevMultiple = this.multiple; │ │ │ │ │ + this.multiple = true; │ │ │ │ │ + var layers = this.layers || [this.layer]; │ │ │ │ │ + this.events.triggerEvent("boxselectionstart", { │ │ │ │ │ + layers: layers │ │ │ │ │ + }); │ │ │ │ │ + var layer; │ │ │ │ │ + for (var l = 0; l < layers.length; ++l) { │ │ │ │ │ + layer = layers[l]; │ │ │ │ │ + for (var i = 0, len = layer.features.length; i < len; ++i) { │ │ │ │ │ + var feature = layer.features[i]; │ │ │ │ │ + // check if the feature is displayed │ │ │ │ │ + if (!feature.getVisibility()) { │ │ │ │ │ + continue; │ │ │ │ │ } │ │ │ │ │ - if (point.z != undefined) { // allow null or undefined │ │ │ │ │ - parts[i] += "," + point.z; │ │ │ │ │ + │ │ │ │ │ + if (this.geometryTypes == null || OpenLayers.Util.indexOf( │ │ │ │ │ + this.geometryTypes, feature.geometry.CLASS_NAME) > -1) { │ │ │ │ │ + if (bounds.toGeometry().intersects(feature.geometry)) { │ │ │ │ │ + if (OpenLayers.Util.indexOf(layer.selectedFeatures, feature) == -1) { │ │ │ │ │ + this.select(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return this.createElementNSPlus("gml:coordinates", { │ │ │ │ │ - attributes: { │ │ │ │ │ - decimal: ".", │ │ │ │ │ - cs: ",", │ │ │ │ │ - ts: " " │ │ │ │ │ - }, │ │ │ │ │ - value: (numPoints == 1) ? parts[0] : parts.join(" ") │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "LineString": function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:LineString"); │ │ │ │ │ - this.writeNode("coordinates", geometry.components, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Polygon": function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:Polygon"); │ │ │ │ │ - this.writeNode("outerBoundaryIs", geometry.components[0], node); │ │ │ │ │ - for (var i = 1; i < geometry.components.length; ++i) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "innerBoundaryIs", geometry.components[i], node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "outerBoundaryIs": function(ring) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:outerBoundaryIs"); │ │ │ │ │ - this.writeNode("LinearRing", ring, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "innerBoundaryIs": function(ring) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:innerBoundaryIs"); │ │ │ │ │ - this.writeNode("LinearRing", ring, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "LinearRing": function(ring) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:LinearRing"); │ │ │ │ │ - this.writeNode("coordinates", ring.components, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Box": function(bounds) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:Box"); │ │ │ │ │ - this.writeNode("coordinates", [{ │ │ │ │ │ - x: bounds.left, │ │ │ │ │ - y: bounds.bottom │ │ │ │ │ - }, { │ │ │ │ │ - x: bounds.right, │ │ │ │ │ - y: bounds.top │ │ │ │ │ - }], node); │ │ │ │ │ - // srsName attribute is optional for gml:Box │ │ │ │ │ - if (this.srsName) { │ │ │ │ │ - node.setAttribute("srsName", this.srsName); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.GML.Base.prototype.writers["gml"]), │ │ │ │ │ - "feature": OpenLayers.Format.GML.Base.prototype.writers["feature"], │ │ │ │ │ - "wfs": OpenLayers.Format.GML.Base.prototype.writers["wfs"] │ │ │ │ │ + this.multiple = prevMultiple; │ │ │ │ │ + this.events.triggerEvent("boxselectionend", { │ │ │ │ │ + layers: layers │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.GML.v2" │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the control. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.handlers.feature.setMap(map); │ │ │ │ │ + if (this.box) { │ │ │ │ │ + this.handlers.box.setMap(map); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setLayer │ │ │ │ │ + * Attach a new layer to the control, overriding any existing layers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layers - Array of {<OpenLayers.Layer.Vector>} or a single │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + setLayer: function(layers) { │ │ │ │ │ + var isActive = this.active; │ │ │ │ │ + this.unselectAll(); │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + if (this.layers) { │ │ │ │ │ + this.layer.destroy(); │ │ │ │ │ + this.layers = null; │ │ │ │ │ + } │ │ │ │ │ + this.initLayer(layers); │ │ │ │ │ + this.handlers.feature.layer = this.layer; │ │ │ │ │ + if (isActive) { │ │ │ │ │ + this.activate(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.SelectFeature" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/OGCExceptionReport.js │ │ │ │ │ + OpenLayers/Control/Attribution.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.OGCExceptionReport │ │ │ │ │ - * Class to read exception reports for various OGC services and versions. │ │ │ │ │ + * Class: OpenLayers.Control.Attribution │ │ │ │ │ + * The attribution control adds attribution from layers to the map display. │ │ │ │ │ + * It uses 'attribution' property of each layer. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.OGCExceptionReport = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Control.Attribution = │ │ │ │ │ + OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - ogc: "http://www.opengis.net/ogc" │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: separator │ │ │ │ │ + * {String} String used to separate layers. │ │ │ │ │ + */ │ │ │ │ │ + separator: ", ", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ - */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: template │ │ │ │ │ + * {String} Template for the attribution. This has to include the substring │ │ │ │ │ + * "${layers}", which will be replaced by the layer specific │ │ │ │ │ + * attributions, separated by <separator>. The default is "${layers}". │ │ │ │ │ + */ │ │ │ │ │ + template: "${layers}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - */ │ │ │ │ │ - defaultPrefix: "ogc", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.Attribution │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Options for control. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.OGCExceptionReport │ │ │ │ │ - * Create a new parser for OGC exception reports. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Destroy control. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + "removelayer": this.updateAttribution, │ │ │ │ │ + "addlayer": this.updateAttribution, │ │ │ │ │ + "changelayer": this.updateAttribution, │ │ │ │ │ + "changebaselayer": this.updateAttribution, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read OGC exception report data from a string, and return an object with │ │ │ │ │ - * information about the exceptions. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Information about the exceptions that occurred. │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - var result; │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - var root = data.documentElement; │ │ │ │ │ - var exceptionInfo = { │ │ │ │ │ - exceptionReport: null │ │ │ │ │ - }; │ │ │ │ │ - if (root) { │ │ │ │ │ - this.readChildNodes(data, exceptionInfo); │ │ │ │ │ - if (exceptionInfo.exceptionReport === null) { │ │ │ │ │ - // fall-back to OWSCommon since this is a common output format for exceptions │ │ │ │ │ - // we cannot easily use the ows readers directly since they differ for 1.0 and 1.1 │ │ │ │ │ - exceptionInfo = new OpenLayers.Format.OWSCommon().read(data); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return exceptionInfo; │ │ │ │ │ - }, │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "ogc": { │ │ │ │ │ - "ServiceExceptionReport": function(node, obj) { │ │ │ │ │ - obj.exceptionReport = { │ │ │ │ │ - exceptions: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.exceptionReport); │ │ │ │ │ - }, │ │ │ │ │ - "ServiceException": function(node, exceptionReport) { │ │ │ │ │ - var exception = { │ │ │ │ │ - code: node.getAttribute("code"), │ │ │ │ │ - locator: node.getAttribute("locator"), │ │ │ │ │ - text: this.getChildValue(node) │ │ │ │ │ - }; │ │ │ │ │ - exceptionReport.exceptions.push(exception); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Initialize control. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ + */ │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.OGCExceptionReport" │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + 'changebaselayer': this.updateAttribution, │ │ │ │ │ + 'changelayer': this.updateAttribution, │ │ │ │ │ + 'addlayer': this.updateAttribution, │ │ │ │ │ + 'removelayer': this.updateAttribution, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.updateAttribution(); │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ + return this.div; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateAttribution │ │ │ │ │ + * Update attribution string. │ │ │ │ │ + */ │ │ │ │ │ + updateAttribution: function() { │ │ │ │ │ + var attributions = []; │ │ │ │ │ + if (this.map && this.map.layers) { │ │ │ │ │ + for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if (layer.attribution && layer.getVisibility()) { │ │ │ │ │ + // add attribution only if attribution text is unique │ │ │ │ │ + if (OpenLayers.Util.indexOf( │ │ │ │ │ + attributions, layer.attribution) === -1) { │ │ │ │ │ + attributions.push(layer.attribution); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.div.innerHTML = OpenLayers.String.format(this.template, { │ │ │ │ │ + layers: attributions.join(this.separator) │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Attribution" │ │ │ │ │ + }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + OpenLayers/Control/Geolocate.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/OGCExceptionReport.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Projection.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.XML.VersionedOGC │ │ │ │ │ - * Base class for versioned formats, i.e. a format which supports multiple │ │ │ │ │ - * versions. │ │ │ │ │ - * │ │ │ │ │ - * To enable checking if parsing succeeded, you will need to define a property │ │ │ │ │ - * called errorProperty on the parser you want to check. The parser will then │ │ │ │ │ - * check the returned object to see if that property is present. If it is, it │ │ │ │ │ - * assumes the parsing was successful. If it is not present (or is null), it will │ │ │ │ │ - * pass the document through an OGCExceptionReport parser. │ │ │ │ │ - * │ │ │ │ │ - * If errorProperty is undefined for the parser, this error checking mechanism │ │ │ │ │ - * will be disabled. │ │ │ │ │ + * Class: OpenLayers.Control.Geolocate │ │ │ │ │ + * The Geolocate control wraps w3c geolocation API into control that can be │ │ │ │ │ + * bound to a map, and generate events on location update │ │ │ │ │ * │ │ │ │ │ + * To use this control requires to load the proj4js library if the projection │ │ │ │ │ + * of the map is not EPSG:4326 or EPSG:900913. │ │ │ │ │ * │ │ │ │ │ - * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.XML.VersionedOGC = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Control.Geolocate = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ + * locationupdated - Triggered when browser return a new position. Listeners will │ │ │ │ │ + * receive an object with a 'position' property which is the browser.geolocation.position │ │ │ │ │ + * native object, as well as a 'point' property which is the location transformed in the │ │ │ │ │ + * current map projection. │ │ │ │ │ + * locationfailed - Triggered when geolocation has failed │ │ │ │ │ + * locationuncapable - Triggered when control is activated on a browser │ │ │ │ │ + * which doesn't support geolocation │ │ │ │ │ */ │ │ │ │ │ - defaultVersion: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: version │ │ │ │ │ - * {String} Specify a version string if one is known. │ │ │ │ │ + * Property: geolocation │ │ │ │ │ + * {Object} The geolocation engine, as a property to be possibly mocked. │ │ │ │ │ + * This is set lazily to avoid a memory leak in IE9. │ │ │ │ │ */ │ │ │ │ │ - version: null, │ │ │ │ │ + geolocation: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: profile │ │ │ │ │ - * {String} If provided, use a custom profile. │ │ │ │ │ + * Property: available │ │ │ │ │ + * {Boolean} The navigator.geolocation object is available. │ │ │ │ │ */ │ │ │ │ │ - profile: null, │ │ │ │ │ + available: ('geolocation' in navigator), │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: allowFallback │ │ │ │ │ - * {Boolean} If a profiled parser cannot be found for the returned version, │ │ │ │ │ - * use a non-profiled parser as the fallback. Application code using this │ │ │ │ │ - * should take into account that the return object structure might be │ │ │ │ │ - * missing the specifics of the profile. Defaults to false. │ │ │ │ │ + * APIProperty: bind │ │ │ │ │ + * {Boolean} If true, map center will be set on location update. │ │ │ │ │ */ │ │ │ │ │ - allowFallback: false, │ │ │ │ │ + bind: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: name │ │ │ │ │ - * {String} The name of this parser, this is the part of the CLASS_NAME │ │ │ │ │ - * except for "OpenLayers.Format." │ │ │ │ │ + * APIProperty: watch │ │ │ │ │ + * {Boolean} If true, position will be update regularly. │ │ │ │ │ */ │ │ │ │ │ - name: null, │ │ │ │ │ + watch: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: stringifyOutput │ │ │ │ │ - * {Boolean} If true, write will return a string otherwise a DOMElement. │ │ │ │ │ - * Default is false. │ │ │ │ │ + * APIProperty: geolocationOptions │ │ │ │ │ + * {Object} Options to pass to the navigator's geolocation API. See │ │ │ │ │ + * <http://dev.w3.org/geo/api/spec-source.html>. No specific │ │ │ │ │ + * option is passed to the geolocation API by default. │ │ │ │ │ */ │ │ │ │ │ - stringifyOutput: false, │ │ │ │ │ + geolocationOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: parser │ │ │ │ │ - * {Object} Instance of the versioned parser. Cached for multiple read and │ │ │ │ │ - * write calls of the same version. │ │ │ │ │ + * Constructor: OpenLayers.Control.Geolocate │ │ │ │ │ + * Create a new control to deal with browser geolocation API │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - parser: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.XML.VersionedOGC. │ │ │ │ │ - * Constructor. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on │ │ │ │ │ - * the object. │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - var className = this.CLASS_NAME; │ │ │ │ │ - this.name = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getVersion │ │ │ │ │ - * Returns the version to use. Subclasses can override this function │ │ │ │ │ - * if a different version detection is needed. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * root - {DOMElement} │ │ │ │ │ - * options - {Object} Optional configuration object. │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activates the control. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} The version to use. │ │ │ │ │ + * {Boolean} The control was effectively activated. │ │ │ │ │ */ │ │ │ │ │ - getVersion: function(root, options) { │ │ │ │ │ - var version; │ │ │ │ │ - // read │ │ │ │ │ - if (root) { │ │ │ │ │ - version = this.version; │ │ │ │ │ - if (!version) { │ │ │ │ │ - version = root.getAttribute("version"); │ │ │ │ │ - if (!version) { │ │ │ │ │ - version = this.defaultVersion; │ │ │ │ │ - } │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (this.available && !this.geolocation) { │ │ │ │ │ + // set lazily to avoid IE9 memory leak │ │ │ │ │ + this.geolocation = navigator.geolocation; │ │ │ │ │ + } │ │ │ │ │ + if (!this.geolocation) { │ │ │ │ │ + this.events.triggerEvent("locationuncapable"); │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + if (this.watch) { │ │ │ │ │ + this.watchId = this.geolocation.watchPosition( │ │ │ │ │ + OpenLayers.Function.bind(this.geolocate, this), │ │ │ │ │ + OpenLayers.Function.bind(this.failure, this), │ │ │ │ │ + this.geolocationOptions │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + this.getCurrentLocation(); │ │ │ │ │ } │ │ │ │ │ - } else { // write │ │ │ │ │ - version = (options && options.version) || │ │ │ │ │ - this.version || this.defaultVersion; │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ - return version; │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getParser │ │ │ │ │ - * Get an instance of the cached parser if available, otherwise create one. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * version - {String} │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivates the control. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Format>} │ │ │ │ │ + * {Boolean} The control was effectively deactivated. │ │ │ │ │ */ │ │ │ │ │ - getParser: function(version) { │ │ │ │ │ - version = version || this.defaultVersion; │ │ │ │ │ - var profile = this.profile ? "_" + this.profile : ""; │ │ │ │ │ - if (!this.parser || this.parser.VERSION != version) { │ │ │ │ │ - var format = OpenLayers.Format[this.name][ │ │ │ │ │ - "v" + version.replace(/\./g, "_") + profile │ │ │ │ │ - ]; │ │ │ │ │ - if (!format) { │ │ │ │ │ - if (profile !== "" && this.allowFallback) { │ │ │ │ │ - // fallback to the non-profiled version of the parser │ │ │ │ │ - profile = ""; │ │ │ │ │ - format = OpenLayers.Format[this.name][ │ │ │ │ │ - "v" + version.replace(/\./g, "_") │ │ │ │ │ - ]; │ │ │ │ │ - } │ │ │ │ │ - if (!format) { │ │ │ │ │ - throw "Can't find a " + this.name + " parser for version " + │ │ │ │ │ - version + profile; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.parser = new format(this.options); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active && this.watchId !== null) { │ │ │ │ │ + this.geolocation.clearWatch(this.watchId); │ │ │ │ │ } │ │ │ │ │ - return this.parser; │ │ │ │ │ + return OpenLayers.Control.prototype.deactivate.apply( │ │ │ │ │ + this, arguments │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Write a document. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} An object representing the document. │ │ │ │ │ - * options - {Object} Optional configuration object. │ │ │ │ │ + * Method: geolocate │ │ │ │ │ + * Activates the control. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The document as a string │ │ │ │ │ */ │ │ │ │ │ - write: function(obj, options) { │ │ │ │ │ - var version = this.getVersion(null, options); │ │ │ │ │ - this.parser = this.getParser(version); │ │ │ │ │ - var root = this.parser.write(obj, options); │ │ │ │ │ - if (this.stringifyOutput === false) { │ │ │ │ │ - return root; │ │ │ │ │ - } else { │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [root]); │ │ │ │ │ + geolocate: function(position) { │ │ │ │ │ + var center = new OpenLayers.LonLat( │ │ │ │ │ + position.coords.longitude, │ │ │ │ │ + position.coords.latitude │ │ │ │ │ + ).transform( │ │ │ │ │ + new OpenLayers.Projection("EPSG:4326"), │ │ │ │ │ + this.map.getProjectionObject() │ │ │ │ │ + ); │ │ │ │ │ + if (this.bind) { │ │ │ │ │ + this.map.setCenter(center); │ │ │ │ │ } │ │ │ │ │ + this.events.triggerEvent("locationupdated", { │ │ │ │ │ + position: position, │ │ │ │ │ + point: new OpenLayers.Geometry.Point( │ │ │ │ │ + center.lon, center.lat │ │ │ │ │ + ) │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read a doc and return an object representing the document. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String | DOMElement} Data to read. │ │ │ │ │ - * options - {Object} Options for the reader. │ │ │ │ │ + * APIMethod: getCurrentLocation │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} An object representing the document. │ │ │ │ │ + * {Boolean} Returns true if a event will be fired (successfull │ │ │ │ │ + * registration) │ │ │ │ │ */ │ │ │ │ │ - read: function(data, options) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + getCurrentLocation: function() { │ │ │ │ │ + if (!this.active || this.watch) { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - var root = data.documentElement; │ │ │ │ │ - var version = this.getVersion(root); │ │ │ │ │ - this.parser = this.getParser(version); // Select the parser │ │ │ │ │ - var obj = this.parser.read(data, options); // Parse the data │ │ │ │ │ + this.geolocation.getCurrentPosition( │ │ │ │ │ + OpenLayers.Function.bind(this.geolocate, this), │ │ │ │ │ + OpenLayers.Function.bind(this.failure, this), │ │ │ │ │ + this.geolocationOptions │ │ │ │ │ + ); │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var errorProperty = this.parser.errorProperty || null; │ │ │ │ │ - if (errorProperty !== null && obj[errorProperty] === undefined) { │ │ │ │ │ - // an error must have happened, so parse it and report back │ │ │ │ │ - var format = new OpenLayers.Format.OGCExceptionReport(); │ │ │ │ │ - obj.error = format.read(data); │ │ │ │ │ - } │ │ │ │ │ - obj.version = version; │ │ │ │ │ - return obj; │ │ │ │ │ + /** │ │ │ │ │ + * Method: failure │ │ │ │ │ + * method called on browser's geolocation failure │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + failure: function(error) { │ │ │ │ │ + this.events.triggerEvent("locationfailed", { │ │ │ │ │ + error: error │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.XML.VersionedOGC" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Geolocate" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Filter/Logical.js │ │ │ │ │ + OpenLayers/Strategy.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Filter.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Filter.Logical │ │ │ │ │ - * This class represents ogc:And, ogc:Or and ogc:Not rules. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Filter> │ │ │ │ │ + * Class: OpenLayers.Strategy │ │ │ │ │ + * Abstract vector layer strategy class. Not to be instantiated directly. Use │ │ │ │ │ + * one of the strategy subclasses instead. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Filter.Logical = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ +OpenLayers.Strategy = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: filters │ │ │ │ │ - * {Array(<OpenLayers.Filter>)} Child filters for this filter. │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} The layer this strategy belongs to. │ │ │ │ │ */ │ │ │ │ │ - filters: null, │ │ │ │ │ + layer: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: type │ │ │ │ │ - * {String} type of logical operator. Available types are: │ │ │ │ │ - * - OpenLayers.Filter.Logical.AND = "&&"; │ │ │ │ │ - * - OpenLayers.Filter.Logical.OR = "||"; │ │ │ │ │ - * - OpenLayers.Filter.Logical.NOT = "!"; │ │ │ │ │ + * Property: options │ │ │ │ │ + * {Object} Any options sent to the constructor. │ │ │ │ │ */ │ │ │ │ │ - type: null, │ │ │ │ │ + options: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Filter.Logical │ │ │ │ │ - * Creates a logical filter (And, Or, Not). │ │ │ │ │ + * Property: active │ │ │ │ │ + * {Boolean} The control is active. │ │ │ │ │ + */ │ │ │ │ │ + active: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: autoActivate │ │ │ │ │ + * {Boolean} The creator of the strategy can set autoActivate to false │ │ │ │ │ + * to fully control when the protocol is activated and deactivated. │ │ │ │ │ + * Defaults to true. │ │ │ │ │ + */ │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: autoDestroy │ │ │ │ │ + * {Boolean} The creator of the strategy can set autoDestroy to false │ │ │ │ │ + * to fully control when the strategy is destroyed. Defaults to │ │ │ │ │ + * true. │ │ │ │ │ + */ │ │ │ │ │ + autoDestroy: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Strategy │ │ │ │ │ + * Abstract class for vector strategies. Create instances of a subclass. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object with properties to set on the │ │ │ │ │ - * filter. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Filter.Logical>} │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - this.filters = []; │ │ │ │ │ - OpenLayers.Filter.prototype.initialize.apply(this, [options]); │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.options = options; │ │ │ │ │ + // set the active property here, so that user cannot override it │ │ │ │ │ + this.active = false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ + /** │ │ │ │ │ * APIMethod: destroy │ │ │ │ │ - * Remove reference to child filters. │ │ │ │ │ + * Clean up the strategy. │ │ │ │ │ */ │ │ │ │ │ destroy: function() { │ │ │ │ │ - this.filters = null; │ │ │ │ │ - OpenLayers.Filter.prototype.destroy.apply(this); │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.layer = null; │ │ │ │ │ + this.options = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: evaluate │ │ │ │ │ - * Evaluates this filter in a specific context. │ │ │ │ │ - * │ │ │ │ │ + * Method: setLayer │ │ │ │ │ + * Called to set the <layer> property. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * context - {Object} Context to use in evaluating the filter. A vector │ │ │ │ │ - * feature may also be provided to evaluate feature attributes in │ │ │ │ │ - * comparison filters or geometries in spatial filters. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The filter applies. │ │ │ │ │ + * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ */ │ │ │ │ │ - evaluate: function(context) { │ │ │ │ │ - var i, len; │ │ │ │ │ - switch (this.type) { │ │ │ │ │ - case OpenLayers.Filter.Logical.AND: │ │ │ │ │ - for (i = 0, len = this.filters.length; i < len; i++) { │ │ │ │ │ - if (this.filters[i].evaluate(context) == false) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - │ │ │ │ │ - case OpenLayers.Filter.Logical.OR: │ │ │ │ │ - for (i = 0, len = this.filters.length; i < len; i++) { │ │ │ │ │ - if (this.filters[i].evaluate(context) == true) { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return false; │ │ │ │ │ + setLayer: function(layer) { │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - case OpenLayers.Filter.Logical.NOT: │ │ │ │ │ - return (!this.filters[0].evaluate(context)); │ │ │ │ │ + /** │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} True if the strategy was successfully activated or false if │ │ │ │ │ + * the strategy was already active. │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + this.active = true; │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ - return undefined; │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Clones this filter. │ │ │ │ │ - * │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ + * tear-down. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Filter.Logical>} Clone of this filter. │ │ │ │ │ + * {Boolean} True if the strategy was successfully deactivated or false if │ │ │ │ │ + * the strategy was already inactive. │ │ │ │ │ */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - var filters = []; │ │ │ │ │ - for (var i = 0, len = this.filters.length; i < len; ++i) { │ │ │ │ │ - filters.push(this.filters[i].clone()); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.active = false; │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Filter.Logical({ │ │ │ │ │ - type: this.type, │ │ │ │ │ - filters: filters │ │ │ │ │ - }); │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Filter.Logical" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Filter.Logical.AND = "&&"; │ │ │ │ │ -OpenLayers.Filter.Logical.OR = "||"; │ │ │ │ │ -OpenLayers.Filter.Logical.NOT = "!"; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Filter/Comparison.js │ │ │ │ │ + OpenLayers/Strategy/Fixed.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Filter.js │ │ │ │ │ + * @requires OpenLayers/Strategy.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Filter.Comparison │ │ │ │ │ - * This class represents a comparison filter. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Strategy.Fixed │ │ │ │ │ + * A simple strategy that requests features once and never requests new data. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Filter> │ │ │ │ │ + * - <OpenLayers.Strategy> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Filter.Comparison = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: type │ │ │ │ │ - * {String} type: type of the comparison. This is one of │ │ │ │ │ - * - OpenLayers.Filter.Comparison.EQUAL_TO = "=="; │ │ │ │ │ - * - OpenLayers.Filter.Comparison.NOT_EQUAL_TO = "!="; │ │ │ │ │ - * - OpenLayers.Filter.Comparison.LESS_THAN = "<"; │ │ │ │ │ - * - OpenLayers.Filter.Comparison.GREATER_THAN = ">"; │ │ │ │ │ - * - OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO = "<="; │ │ │ │ │ - * - OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO = ">="; │ │ │ │ │ - * - OpenLayers.Filter.Comparison.BETWEEN = ".."; │ │ │ │ │ - * - OpenLayers.Filter.Comparison.LIKE = "~"; │ │ │ │ │ - * - OpenLayers.Filter.Comparison.IS_NULL = "NULL"; │ │ │ │ │ - */ │ │ │ │ │ - type: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: property │ │ │ │ │ - * {String} │ │ │ │ │ - * name of the context property to compare │ │ │ │ │ - */ │ │ │ │ │ - property: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: value │ │ │ │ │ - * {Number} or {String} │ │ │ │ │ - * comparison value for binary comparisons. In the case of a String, this │ │ │ │ │ - * can be a combination of text and propertyNames in the form │ │ │ │ │ - * "literal ${propertyName}" │ │ │ │ │ - */ │ │ │ │ │ - value: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: matchCase │ │ │ │ │ - * {Boolean} Force case sensitive searches for EQUAL_TO and NOT_EQUAL_TO │ │ │ │ │ - * comparisons. The Filter Encoding 1.1 specification added a matchCase │ │ │ │ │ - * attribute to ogc:PropertyIsEqualTo and ogc:PropertyIsNotEqualTo │ │ │ │ │ - * elements. This property will be serialized with those elements only │ │ │ │ │ - * if using the v1.1.0 filter format. However, when evaluating filters │ │ │ │ │ - * here, the matchCase property will always be respected (for EQUAL_TO │ │ │ │ │ - * and NOT_EQUAL_TO). Default is true. │ │ │ │ │ - */ │ │ │ │ │ - matchCase: true, │ │ │ │ │ +OpenLayers.Strategy.Fixed = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: lowerBoundary │ │ │ │ │ - * {Number} or {String} │ │ │ │ │ - * lower boundary for between comparisons. In the case of a String, this │ │ │ │ │ - * can be a combination of text and propertyNames in the form │ │ │ │ │ - * "literal ${propertyName}" │ │ │ │ │ + * APIProperty: preload │ │ │ │ │ + * {Boolean} Load data before layer made visible. Enabling this may result │ │ │ │ │ + * in considerable overhead if your application loads many data layers │ │ │ │ │ + * that are not visible by default. Default is false. │ │ │ │ │ */ │ │ │ │ │ - lowerBoundary: null, │ │ │ │ │ + preload: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: upperBoundary │ │ │ │ │ - * {Number} or {String} │ │ │ │ │ - * upper boundary for between comparisons. In the case of a String, this │ │ │ │ │ - * can be a combination of text and propertyNames in the form │ │ │ │ │ - * "literal ${propertyName}" │ │ │ │ │ - */ │ │ │ │ │ - upperBoundary: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Filter.Comparison │ │ │ │ │ - * Creates a comparison rule. │ │ │ │ │ + * Constructor: OpenLayers.Strategy.Fixed │ │ │ │ │ + * Create a new Fixed strategy. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object with properties to set on the │ │ │ │ │ - * rule │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Filter.Comparison>} │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Filter.prototype.initialize.apply(this, [options]); │ │ │ │ │ - // since matchCase on PropertyIsLike is not schema compliant, we only │ │ │ │ │ - // want to use this if explicitly asked for │ │ │ │ │ - if (this.type === OpenLayers.Filter.Comparison.LIKE && │ │ │ │ │ - options.matchCase === undefined) { │ │ │ │ │ - this.matchCase = null; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: evaluate │ │ │ │ │ - * Evaluates this filter in a specific context. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * context - {Object} Context to use in evaluating the filter. If a vector │ │ │ │ │ - * feature is provided, the feature.attributes will be used as context. │ │ │ │ │ - * │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activate the strategy: load data or add listener to load when visible │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The filter applies. │ │ │ │ │ + * {Boolean} True if the strategy was successfully activated or false if │ │ │ │ │ + * the strategy was already active. │ │ │ │ │ */ │ │ │ │ │ - evaluate: function(context) { │ │ │ │ │ - if (context instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ - context = context.attributes; │ │ │ │ │ - } │ │ │ │ │ - var result = false; │ │ │ │ │ - var got = context[this.property]; │ │ │ │ │ - var exp; │ │ │ │ │ - switch (this.type) { │ │ │ │ │ - case OpenLayers.Filter.Comparison.EQUAL_TO: │ │ │ │ │ - exp = this.value; │ │ │ │ │ - if (!this.matchCase && │ │ │ │ │ - typeof got == "string" && typeof exp == "string") { │ │ │ │ │ - result = (got.toUpperCase() == exp.toUpperCase()); │ │ │ │ │ - } else { │ │ │ │ │ - result = (got == exp); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Comparison.NOT_EQUAL_TO: │ │ │ │ │ - exp = this.value; │ │ │ │ │ - if (!this.matchCase && │ │ │ │ │ - typeof got == "string" && typeof exp == "string") { │ │ │ │ │ - result = (got.toUpperCase() != exp.toUpperCase()); │ │ │ │ │ - } else { │ │ │ │ │ - result = (got != exp); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Comparison.LESS_THAN: │ │ │ │ │ - result = got < this.value; │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Comparison.GREATER_THAN: │ │ │ │ │ - result = got > this.value; │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO: │ │ │ │ │ - result = got <= this.value; │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO: │ │ │ │ │ - result = got >= this.value; │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Comparison.BETWEEN: │ │ │ │ │ - result = (got >= this.lowerBoundary) && │ │ │ │ │ - (got <= this.upperBoundary); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Comparison.LIKE: │ │ │ │ │ - var regexp = new RegExp(this.value, "gi"); │ │ │ │ │ - result = regexp.test(got); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Comparison.IS_NULL: │ │ │ │ │ - result = (got === null); │ │ │ │ │ - break; │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "refresh": this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + if (this.layer.visibility == true || this.preload) { │ │ │ │ │ + this.load(); │ │ │ │ │ + } else { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "visibilitychanged": this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return result; │ │ │ │ │ + return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: value2regex │ │ │ │ │ - * Converts the value of this rule into a regular expression string, │ │ │ │ │ - * according to the wildcard characters specified. This method has to │ │ │ │ │ - * be called after instantiation of this class, if the value is not a │ │ │ │ │ - * regular expression already. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * wildCard - {Char} wildcard character in the above value, default │ │ │ │ │ - * is "*" │ │ │ │ │ - * singleChar - {Char} single-character wildcard in the above value │ │ │ │ │ - * default is "." │ │ │ │ │ - * escapeChar - {Char} escape character in the above value, default is │ │ │ │ │ - * "!" │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivate the strategy. Undo what is done in <activate>. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} regular expression string │ │ │ │ │ + * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ - value2regex: function(wildCard, singleChar, escapeChar) { │ │ │ │ │ - if (wildCard == ".") { │ │ │ │ │ - throw new Error("'.' is an unsupported wildCard character for " + │ │ │ │ │ - "OpenLayers.Filter.Comparison"); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + "refresh": this.load, │ │ │ │ │ + "visibilitychanged": this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - // set UMN MapServer defaults for unspecified parameters │ │ │ │ │ - wildCard = wildCard ? wildCard : "*"; │ │ │ │ │ - singleChar = singleChar ? singleChar : "."; │ │ │ │ │ - escapeChar = escapeChar ? escapeChar : "!"; │ │ │ │ │ - │ │ │ │ │ - this.value = this.value.replace( │ │ │ │ │ - new RegExp("\\" + escapeChar + "(.|$)", "g"), "\\$1"); │ │ │ │ │ - this.value = this.value.replace( │ │ │ │ │ - new RegExp("\\" + singleChar, "g"), "."); │ │ │ │ │ - this.value = this.value.replace( │ │ │ │ │ - new RegExp("\\" + wildCard, "g"), ".*"); │ │ │ │ │ - this.value = this.value.replace( │ │ │ │ │ - new RegExp("\\\\.\\*", "g"), "\\" + wildCard); │ │ │ │ │ - this.value = this.value.replace( │ │ │ │ │ - new RegExp("\\\\\\.", "g"), "\\" + singleChar); │ │ │ │ │ - │ │ │ │ │ - return this.value; │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: regex2value │ │ │ │ │ - * Convert the value of this rule from a regular expression string into an │ │ │ │ │ - * ogc literal string using a wildCard of *, a singleChar of ., and an │ │ │ │ │ - * escape of !. Leaves the <value> property unmodified. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string value. │ │ │ │ │ + * Method: load │ │ │ │ │ + * Tells protocol to load data and unhooks the visibilitychanged event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} options to pass to protocol read. │ │ │ │ │ */ │ │ │ │ │ - regex2value: function() { │ │ │ │ │ - │ │ │ │ │ - var value = this.value; │ │ │ │ │ - │ │ │ │ │ - // replace ! with !! │ │ │ │ │ - value = value.replace(/!/g, "!!"); │ │ │ │ │ - │ │ │ │ │ - // replace \. with !. (watching out for \\.) │ │ │ │ │ - value = value.replace(/(\\)?\\\./g, function($0, $1) { │ │ │ │ │ - return $1 ? $0 : "!."; │ │ │ │ │ + load: function(options) { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + layer.events.triggerEvent("loadstart", { │ │ │ │ │ + filter: layer.filter │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ - // replace \* with #* (watching out for \\*) │ │ │ │ │ - value = value.replace(/(\\)?\\\*/g, function($0, $1) { │ │ │ │ │ - return $1 ? $0 : "!*"; │ │ │ │ │ + layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: this.merge, │ │ │ │ │ + filter: layer.filter, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options)); │ │ │ │ │ + layer.events.un({ │ │ │ │ │ + "visibilitychanged": this.load, │ │ │ │ │ + scope: this │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ - // replace \\ with \ │ │ │ │ │ - value = value.replace(/\\\\/g, "\\"); │ │ │ │ │ - │ │ │ │ │ - // convert .* to * (the sequence #.* is not allowed) │ │ │ │ │ - value = value.replace(/\.\*/g, "*"); │ │ │ │ │ - │ │ │ │ │ - return value; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Clones this filter. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Filter.Comparison>} Clone of this filter. │ │ │ │ │ + * Method: merge │ │ │ │ │ + * Add all features to the layer. │ │ │ │ │ + * If the layer projection differs from the map projection, features │ │ │ │ │ + * will be transformed from the layer projection to the map projection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object passed │ │ │ │ │ + * by the protocol. │ │ │ │ │ */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - return OpenLayers.Util.extend(new OpenLayers.Filter.Comparison(), this); │ │ │ │ │ + merge: function(resp) { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + layer.destroyFeatures(); │ │ │ │ │ + var features = resp.features; │ │ │ │ │ + if (features && features.length > 0) { │ │ │ │ │ + var remote = layer.projection; │ │ │ │ │ + var local = layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var geom; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + geom = features[i].geometry; │ │ │ │ │ + if (geom) { │ │ │ │ │ + geom.transform(remote, local); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + layer.addFeatures(features); │ │ │ │ │ + } │ │ │ │ │ + layer.events.triggerEvent("loadend", { │ │ │ │ │ + response: resp │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Filter.Comparison" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Fixed" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Filter.Comparison.EQUAL_TO = "=="; │ │ │ │ │ -OpenLayers.Filter.Comparison.NOT_EQUAL_TO = "!="; │ │ │ │ │ -OpenLayers.Filter.Comparison.LESS_THAN = "<"; │ │ │ │ │ -OpenLayers.Filter.Comparison.GREATER_THAN = ">"; │ │ │ │ │ -OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO = "<="; │ │ │ │ │ -OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO = ">="; │ │ │ │ │ -OpenLayers.Filter.Comparison.BETWEEN = ".."; │ │ │ │ │ -OpenLayers.Filter.Comparison.LIKE = "~"; │ │ │ │ │ -OpenLayers.Filter.Comparison.IS_NULL = "NULL"; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/Filter.js │ │ │ │ │ + OpenLayers/Layer/WMS.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ - * @requires OpenLayers/Filter/FeatureId.js │ │ │ │ │ - * @requires OpenLayers/Filter/Logical.js │ │ │ │ │ - * @requires OpenLayers/Filter/Comparison.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.Filter │ │ │ │ │ - * Read/Write ogc:Filter. Create a new instance with the <OpenLayers.Format.Filter> │ │ │ │ │ + * Class: OpenLayers.Layer.WMS │ │ │ │ │ + * Instances of OpenLayers.Layer.WMS are used to display data from OGC Web │ │ │ │ │ + * Mapping Services. Create a new WMS layer with the <OpenLayers.Layer.WMS> │ │ │ │ │ * constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.Filter = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ +OpenLayers.Layer.WMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.0.0". │ │ │ │ │ + * Constant: DEFAULT_PARAMS │ │ │ │ │ + * {Object} Hashtable of default parameter key/value pairs │ │ │ │ │ */ │ │ │ │ │ - defaultVersion: "1.0.0", │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + service: "WMS", │ │ │ │ │ + version: "1.1.1", │ │ │ │ │ + request: "GetMap", │ │ │ │ │ + styles: "", │ │ │ │ │ + format: "image/jpeg" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Write an ogc:Filter given a filter object. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} An filter. │ │ │ │ │ - * options - {Object} Optional configuration object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Elment} An ogc:Filter element node. │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} Default is true for WMS layer │ │ │ │ │ */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read and Filter doc and return an object representing the Filter. │ │ │ │ │ + * APIProperty: encodeBBOX │ │ │ │ │ + * {Boolean} Should the BBOX commas be encoded? The WMS spec says 'no', │ │ │ │ │ + * but some services want it that way. Default false. │ │ │ │ │ + */ │ │ │ │ │ + encodeBBOX: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: noMagic │ │ │ │ │ + * {Boolean} If true, the image format will not be automagicaly switched │ │ │ │ │ + * from image/jpeg to image/png or image/gif when using │ │ │ │ │ + * TRANSPARENT=TRUE. Also isBaseLayer will not changed by the │ │ │ │ │ + * constructor. Default false. │ │ │ │ │ + */ │ │ │ │ │ + noMagic: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: yx │ │ │ │ │ + * {Object} Keys in this object are EPSG codes for which the axis order │ │ │ │ │ + * is to be reversed (yx instead of xy, LatLon instead of LonLat), with │ │ │ │ │ + * true as value. This is only relevant for WMS versions >= 1.3.0, and │ │ │ │ │ + * only if yx is not set in <OpenLayers.Projection.defaults> for the │ │ │ │ │ + * used projection. │ │ │ │ │ + */ │ │ │ │ │ + yx: {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.WMS │ │ │ │ │ + * Create a new WMS layer object │ │ │ │ │ + * │ │ │ │ │ + * Examples: │ │ │ │ │ + * │ │ │ │ │ + * The code below creates a simple WMS layer using the image/jpeg format. │ │ │ │ │ + * (code) │ │ │ │ │ + * var wms = new OpenLayers.Layer.WMS("NASA Global Mosaic", │ │ │ │ │ + * "http://wms.jpl.nasa.gov/wms.cgi", │ │ │ │ │ + * {layers: "modis,global_mosaic"}); │ │ │ │ │ + * (end) │ │ │ │ │ + * Note the 3rd argument (params). Properties added to this object will be │ │ │ │ │ + * added to the WMS GetMap requests used for this layer's tiles. The only │ │ │ │ │ + * mandatory parameter is "layers". Other common WMS params include │ │ │ │ │ + * "transparent", "styles" and "format". Note that the "srs" param will │ │ │ │ │ + * always be ignored. Instead, it will be derived from the baseLayer's or │ │ │ │ │ + * map's projection. │ │ │ │ │ + * │ │ │ │ │ + * The code below creates a transparent WMS layer with additional options. │ │ │ │ │ + * (code) │ │ │ │ │ + * var wms = new OpenLayers.Layer.WMS("NASA Global Mosaic", │ │ │ │ │ + * "http://wms.jpl.nasa.gov/wms.cgi", │ │ │ │ │ + * { │ │ │ │ │ + * layers: "modis,global_mosaic", │ │ │ │ │ + * transparent: true │ │ │ │ │ + * }, { │ │ │ │ │ + * opacity: 0.5, │ │ │ │ │ + * singleTile: true │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * Note that by default, a WMS layer is configured as baseLayer. Setting │ │ │ │ │ + * the "transparent" param to true will apply some magic (see <noMagic>). │ │ │ │ │ + * The default image format changes from image/jpeg to image/png, and the │ │ │ │ │ + * layer is not configured as baseLayer. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {String | DOMElement} Data to read. │ │ │ │ │ + * name - {String} A name for the layer │ │ │ │ │ + * url - {String} Base url for the WMS │ │ │ │ │ + * (e.g. http://wms.jpl.nasa.gov/wms.cgi) │ │ │ │ │ + * params - {Object} An object with key/value pairs representing the │ │ │ │ │ + * GetMap query string parameters and parameter values. │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer. │ │ │ │ │ + * These options include all properties listed above, plus the ones │ │ │ │ │ + * inherited from superclasses. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + //uppercase params │ │ │ │ │ + params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ + if (parseFloat(params.VERSION) >= 1.3 && !params.EXCEPTIONS) { │ │ │ │ │ + params.EXCEPTIONS = "INIMAGE"; │ │ │ │ │ + } │ │ │ │ │ + newArguments.push(name, url, params, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, │ │ │ │ │ + OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS) │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + //layer is transparent │ │ │ │ │ + if (!this.noMagic && this.params.TRANSPARENT && │ │ │ │ │ + this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ + │ │ │ │ │ + // unless explicitly set in options, make layer an overlay │ │ │ │ │ + if ((options == null) || (!options.isBaseLayer)) { │ │ │ │ │ + this.isBaseLayer = false; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // jpegs can never be transparent, so intelligently switch the │ │ │ │ │ + // format, depending on the browser's capabilities │ │ │ │ │ + if (this.params.FORMAT == "image/jpeg") { │ │ │ │ │ + this.params.FORMAT = OpenLayers.Util.alphaHack() ? "image/gif" : │ │ │ │ │ + "image/png"; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Filter>} A filter object. │ │ │ │ │ + * {<OpenLayers.Layer.WMS>} An exact clone of this layer │ │ │ │ │ */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.Filter" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Filter/Function.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.WMS(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.params, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Filter.js │ │ │ │ │ - */ │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Filter.Function │ │ │ │ │ - * This class represents a filter function. │ │ │ │ │ - * We are using this class for creation of complex │ │ │ │ │ - * filters that can contain filter functions as values. │ │ │ │ │ - * Nesting function as other functions parameter is supported. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Filter> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Filter.Function = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: name │ │ │ │ │ - * {String} Name of the function. │ │ │ │ │ + * APIMethod: reverseAxisOrder │ │ │ │ │ + * Returns true if the axis order is reversed for the WMS version and │ │ │ │ │ + * projection of the layer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the axis order is reversed, false otherwise. │ │ │ │ │ */ │ │ │ │ │ - name: null, │ │ │ │ │ + reverseAxisOrder: function() { │ │ │ │ │ + var projCode = this.projection.getCode(); │ │ │ │ │ + return parseFloat(this.params.VERSION) >= 1.3 && │ │ │ │ │ + !!(this.yx[projCode] || (OpenLayers.Projection.defaults[projCode] && │ │ │ │ │ + OpenLayers.Projection.defaults[projCode].yx)); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: params │ │ │ │ │ - * {Array(<OpenLayers.Filter.Function> || String || Number)} Function parameters │ │ │ │ │ - * For now support only other Functions, String or Number │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * Return a GetMap query string for this layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox for the │ │ │ │ │ + * request. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as │ │ │ │ │ + * parameters. │ │ │ │ │ */ │ │ │ │ │ - params: null, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + │ │ │ │ │ + var imageSize = this.getImageSize(); │ │ │ │ │ + var newParams = {}; │ │ │ │ │ + // WMS 1.3 introduced axis order │ │ │ │ │ + var reverseAxisOrder = this.reverseAxisOrder(); │ │ │ │ │ + newParams.BBOX = this.encodeBBOX ? │ │ │ │ │ + bounds.toBBOX(null, reverseAxisOrder) : │ │ │ │ │ + bounds.toArray(reverseAxisOrder); │ │ │ │ │ + newParams.WIDTH = imageSize.w; │ │ │ │ │ + newParams.HEIGHT = imageSize.h; │ │ │ │ │ + var requestString = this.getFullRequestString(newParams); │ │ │ │ │ + return requestString; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: mergeNewParams │ │ │ │ │ + * Catch changeParams and uppercase the new params to be merged in │ │ │ │ │ + * before calling changeParams on the super class. │ │ │ │ │ + * │ │ │ │ │ + * Once params have been changed, the tiles will be reloaded with │ │ │ │ │ + * the new parameters. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newParams - {Object} Hashtable of new params to use │ │ │ │ │ + */ │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ + var newArguments = [upperParams]; │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, │ │ │ │ │ + newArguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Filter.Function │ │ │ │ │ - * Creates a filter function. │ │ │ │ │ + * APIMethod: getFullRequestString │ │ │ │ │ + * Combine the layer's url with its params and these newParams. │ │ │ │ │ + * │ │ │ │ │ + * Add the SRS parameter from projection -- this is probably │ │ │ │ │ + * more eloquently done via a setProjection() method, but this │ │ │ │ │ + * works for now and always. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object with properties to set on the │ │ │ │ │ - * function. │ │ │ │ │ + * newParams - {Object} │ │ │ │ │ + * altUrl - {String} Use this as the url instead of the layer's url │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Filter.Function>} │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ + getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ + var mapProjection = this.map.getProjectionObject(); │ │ │ │ │ + var projectionCode = this.projection && this.projection.equals(mapProjection) ? │ │ │ │ │ + this.projection.getCode() : │ │ │ │ │ + mapProjection.getCode(); │ │ │ │ │ + var value = (projectionCode == "none") ? null : projectionCode; │ │ │ │ │ + if (parseFloat(this.params.VERSION) >= 1.3) { │ │ │ │ │ + this.params.CRS = value; │ │ │ │ │ + } else { │ │ │ │ │ + this.params.SRS = value; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Filter.Function" │ │ │ │ │ -}); │ │ │ │ │ + if (typeof this.params.TRANSPARENT == "boolean") { │ │ │ │ │ + newParams.TRANSPARENT = this.params.TRANSPARENT ? "TRUE" : "FALSE"; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.getFullRequestString.apply( │ │ │ │ │ + this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.WMS" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/Filter/v1.js │ │ │ │ │ + OpenLayers/Layer/XYZ.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/Filter.js │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Filter/Function.js │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Date.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.Filter.v1 │ │ │ │ │ - * Superclass for Filter version 1 parsers. │ │ │ │ │ - * │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.XYZ │ │ │ │ │ + * The XYZ class is designed to make it easier for people who have tiles │ │ │ │ │ + * arranged by a standard XYZ grid. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.Filter.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Layer.XYZ = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * Default is true, as this is designed to be a base tile source. │ │ │ │ │ */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - ogc: "http://www.opengis.net/ogc", │ │ │ │ │ - gml: "http://www.opengis.net/gml", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ - }, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ + * APIProperty: sphericalMercator │ │ │ │ │ + * Whether the tile extents should be set to the defaults for │ │ │ │ │ + * spherical mercator. Useful for things like OpenStreetMap. │ │ │ │ │ + * Default is false, except for the OSM subclass. │ │ │ │ │ */ │ │ │ │ │ - defaultPrefix: "ogc", │ │ │ │ │ + sphericalMercator: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} Schema location for a particular minor version. │ │ │ │ │ + * APIProperty: zoomOffset │ │ │ │ │ + * {Number} If your cache has more zoom levels than you want to provide │ │ │ │ │ + * access to with this layer, supply a zoomOffset. This zoom offset │ │ │ │ │ + * is added to the current map zoom level to determine the level │ │ │ │ │ + * for a requested tile. For example, if you supply a zoomOffset │ │ │ │ │ + * of 3, when the map is at the zoom 0, tiles will be requested from │ │ │ │ │ + * level 3 of your cache. Default is 0 (assumes cache level and map │ │ │ │ │ + * zoom are equivalent). Using <zoomOffset> is an alternative to │ │ │ │ │ + * setting <serverResolutions> if you only want to expose a subset │ │ │ │ │ + * of the server resolutions. │ │ │ │ │ */ │ │ │ │ │ - schemaLocation: null, │ │ │ │ │ + zoomOffset: 0, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.Filter.v1 │ │ │ │ │ - * Instances of this class are not created directly. Use the │ │ │ │ │ - * <OpenLayers.Format.Filter> constructor instead. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * APIProperty: serverResolutions │ │ │ │ │ + * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ + * property if the map resolutions differ from the server. This │ │ │ │ │ + * property serves two purposes. (a) <serverResolutions> can include │ │ │ │ │ + * resolutions that the server supports and that you don't want to │ │ │ │ │ + * provide with this layer; you can also look at <zoomOffset>, which is │ │ │ │ │ + * an alternative to <serverResolutions> for that specific purpose. │ │ │ │ │ + * (b) The map can work with resolutions that aren't supported by │ │ │ │ │ + * the server, i.e. that aren't in <serverResolutions>. When the │ │ │ │ │ + * map is displayed in such a resolution data for the closest │ │ │ │ │ + * server-supported resolution is loaded and the layer div is │ │ │ │ │ + * stretched as necessary. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - }, │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read │ │ │ │ │ + * Constructor: OpenLayers.Layer.XYZ │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {DOMElement} A Filter document element. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Filter>} A filter object. │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readers.ogc["Filter"].apply(this, [data, obj]); │ │ │ │ │ - return obj.filter; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ + * name - {String} │ │ │ │ │ + * url - {String} │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ */ │ │ │ │ │ - readers: { │ │ │ │ │ - "ogc": { │ │ │ │ │ - "_expression": function(node) { │ │ │ │ │ - // only the simplest of ogc:expression handled │ │ │ │ │ - // "some text and an <PropertyName>attribute</PropertyName>"} │ │ │ │ │ - var obj, value = ""; │ │ │ │ │ - for (var child = node.firstChild; child; child = child.nextSibling) { │ │ │ │ │ - switch (child.nodeType) { │ │ │ │ │ - case 1: │ │ │ │ │ - obj = this.readNode(child); │ │ │ │ │ - if (obj.property) { │ │ │ │ │ - value += "${" + obj.property + "}"; │ │ │ │ │ - } else if (obj.value !== undefined) { │ │ │ │ │ - value += obj.value; │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case 3: // text node │ │ │ │ │ - case 4: // cdata section │ │ │ │ │ - value += child.nodeValue; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return value; │ │ │ │ │ - }, │ │ │ │ │ - "Filter": function(node, parent) { │ │ │ │ │ - // Filters correspond to subclasses of OpenLayers.Filter. │ │ │ │ │ - // Since they contain information we don't persist, we │ │ │ │ │ - // create a temporary object and then pass on the filter │ │ │ │ │ - // (ogc:Filter) to the parent obj. │ │ │ │ │ - var obj = { │ │ │ │ │ - fids: [], │ │ │ │ │ - filters: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - if (obj.fids.length > 0) { │ │ │ │ │ - parent.filter = new OpenLayers.Filter.FeatureId({ │ │ │ │ │ - fids: obj.fids │ │ │ │ │ - }); │ │ │ │ │ - } else if (obj.filters.length > 0) { │ │ │ │ │ - parent.filter = obj.filters[0]; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "FeatureId": function(node, obj) { │ │ │ │ │ - var fid = node.getAttribute("fid"); │ │ │ │ │ - if (fid) { │ │ │ │ │ - obj.fids.push(fid); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "And": function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ - type: OpenLayers.Filter.Logical.AND │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - obj.filters.push(filter); │ │ │ │ │ - }, │ │ │ │ │ - "Or": function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ - type: OpenLayers.Filter.Logical.OR │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - obj.filters.push(filter); │ │ │ │ │ - }, │ │ │ │ │ - "Not": function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ - type: OpenLayers.Filter.Logical.NOT │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - obj.filters.push(filter); │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsLessThan": function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.LESS_THAN │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - obj.filters.push(filter); │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsGreaterThan": function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.GREATER_THAN │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - obj.filters.push(filter); │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsLessThanOrEqualTo": function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - obj.filters.push(filter); │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsGreaterThanOrEqualTo": function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - obj.filters.push(filter); │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsBetween": function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.BETWEEN │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - obj.filters.push(filter); │ │ │ │ │ - }, │ │ │ │ │ - "Literal": function(node, obj) { │ │ │ │ │ - obj.value = OpenLayers.String.numericIf( │ │ │ │ │ - this.getChildValue(node), true); │ │ │ │ │ - }, │ │ │ │ │ - "PropertyName": function(node, filter) { │ │ │ │ │ - filter.property = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "LowerBoundary": function(node, filter) { │ │ │ │ │ - filter.lowerBoundary = OpenLayers.String.numericIf( │ │ │ │ │ - this.readers.ogc._expression.call(this, node), true); │ │ │ │ │ - }, │ │ │ │ │ - "UpperBoundary": function(node, filter) { │ │ │ │ │ - filter.upperBoundary = OpenLayers.String.numericIf( │ │ │ │ │ - this.readers.ogc._expression.call(this, node), true); │ │ │ │ │ - }, │ │ │ │ │ - "Intersects": function(node, obj) { │ │ │ │ │ - this.readSpatial(node, obj, OpenLayers.Filter.Spatial.INTERSECTS); │ │ │ │ │ - }, │ │ │ │ │ - "Within": function(node, obj) { │ │ │ │ │ - this.readSpatial(node, obj, OpenLayers.Filter.Spatial.WITHIN); │ │ │ │ │ - }, │ │ │ │ │ - "Contains": function(node, obj) { │ │ │ │ │ - this.readSpatial(node, obj, OpenLayers.Filter.Spatial.CONTAINS); │ │ │ │ │ - }, │ │ │ │ │ - "DWithin": function(node, obj) { │ │ │ │ │ - this.readSpatial(node, obj, OpenLayers.Filter.Spatial.DWITHIN); │ │ │ │ │ - }, │ │ │ │ │ - "Distance": function(node, obj) { │ │ │ │ │ - obj.distance = parseInt(this.getChildValue(node)); │ │ │ │ │ - obj.distanceUnits = node.getAttribute("units"); │ │ │ │ │ - }, │ │ │ │ │ - "Function": function(node, obj) { │ │ │ │ │ - //TODO write decoder for it │ │ │ │ │ - return; │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsNull": function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.IS_NULL │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - obj.filters.push(filter); │ │ │ │ │ - } │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + if (options && options.sphericalMercator || this.sphericalMercator) { │ │ │ │ │ + options = OpenLayers.Util.extend({ │ │ │ │ │ + projection: "EPSG:900913", │ │ │ │ │ + numZoomLevels: 19 │ │ │ │ │ + }, options); │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [ │ │ │ │ │ + name || this.name, url || this.url, {}, │ │ │ │ │ + options │ │ │ │ │ + ]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: readSpatial │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ * │ │ │ │ │ - * Read a {<OpenLayers.Filter.Spatial>} filter. │ │ │ │ │ - * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} A DOM element that contains an ogc:expression. │ │ │ │ │ - * obj - {Object} The target object. │ │ │ │ │ - * type - {String} One of the OpenLayers.Filter.Spatial.* constants. │ │ │ │ │ - * │ │ │ │ │ + * obj - {Object} Is this ever used? │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Filter.Spatial>} The created filter. │ │ │ │ │ + * {<OpenLayers.Layer.XYZ>} An exact clone of this OpenLayers.Layer.XYZ │ │ │ │ │ */ │ │ │ │ │ - readSpatial: function(node, obj, type) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: type │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - filter.value = filter.components[0]; │ │ │ │ │ - delete filter.components; │ │ │ │ │ - obj.filters.push(filter); │ │ │ │ │ - }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: encodeLiteral │ │ │ │ │ - * Generates the string representation of a value for use in <Literal> │ │ │ │ │ - * elements. The default encoder writes Date values as ISO 8601 │ │ │ │ │ - * strings. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * value - {Object} Literal value to encode │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} String representation of the provided value. │ │ │ │ │ - */ │ │ │ │ │ - encodeLiteral: function(value) { │ │ │ │ │ - if (value instanceof Date) { │ │ │ │ │ - value = OpenLayers.Date.toISOString(value); │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.XYZ(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ } │ │ │ │ │ - return value; │ │ │ │ │ + │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: writeOgcExpression │ │ │ │ │ - * Limited support for writing OGC expressions. Currently it supports │ │ │ │ │ - * (<OpenLayers.Filter.Function> || String || Number) │ │ │ │ │ + * Method: getURL │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * value - (<OpenLayers.Filter.Function> || String || Number) │ │ │ │ │ - * node - {DOMElement} A parent DOM element │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} Updated node element. │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as │ │ │ │ │ + * parameters │ │ │ │ │ */ │ │ │ │ │ - writeOgcExpression: function(value, node) { │ │ │ │ │ - if (value instanceof OpenLayers.Filter.Function) { │ │ │ │ │ - this.writeNode("Function", value, node); │ │ │ │ │ - } else { │ │ │ │ │ - this.writeNode("Literal", value, node); │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var xyz = this.getXYZ(bounds); │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + var s = '' + xyz.x + xyz.y + xyz.z; │ │ │ │ │ + url = this.selectUrl(s, url); │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.String.format(url, xyz); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: write │ │ │ │ │ + * Method: getXYZ │ │ │ │ │ + * Calculates x, y and z for the given bounds. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} A filter object. │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} An ogc:Filter element. │ │ │ │ │ + * {Object} - an object with x, y and z properties. │ │ │ │ │ */ │ │ │ │ │ - write: function(filter) { │ │ │ │ │ - return this.writers.ogc["Filter"].apply(this, [filter]); │ │ │ │ │ - }, │ │ │ │ │ + getXYZ: function(bounds) { │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.maxExtent.left) / │ │ │ │ │ + (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((this.maxExtent.top - bounds.top) / │ │ │ │ │ + (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getServerZoom(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ - */ │ │ │ │ │ - writers: { │ │ │ │ │ - "ogc": { │ │ │ │ │ - "Filter": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:Filter"); │ │ │ │ │ - this.writeNode(this.getFilterType(filter), filter, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "_featureIds": function(filter) { │ │ │ │ │ - var node = this.createDocumentFragment(); │ │ │ │ │ - for (var i = 0, ii = filter.fids.length; i < ii; ++i) { │ │ │ │ │ - this.writeNode("ogc:FeatureId", filter.fids[i], node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "FeatureId": function(fid) { │ │ │ │ │ - return this.createElementNSPlus("ogc:FeatureId", { │ │ │ │ │ - attributes: { │ │ │ │ │ - fid: fid │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "And": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:And"); │ │ │ │ │ - var childFilter; │ │ │ │ │ - for (var i = 0, ii = filter.filters.length; i < ii; ++i) { │ │ │ │ │ - childFilter = filter.filters[i]; │ │ │ │ │ - this.writeNode( │ │ │ │ │ - this.getFilterType(childFilter), childFilter, node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Or": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:Or"); │ │ │ │ │ - var childFilter; │ │ │ │ │ - for (var i = 0, ii = filter.filters.length; i < ii; ++i) { │ │ │ │ │ - childFilter = filter.filters[i]; │ │ │ │ │ - this.writeNode( │ │ │ │ │ - this.getFilterType(childFilter), childFilter, node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Not": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:Not"); │ │ │ │ │ - var childFilter = filter.filters[0]; │ │ │ │ │ - this.writeNode( │ │ │ │ │ - this.getFilterType(childFilter), childFilter, node │ │ │ │ │ - ); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsLessThan": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsLessThan"); │ │ │ │ │ - // no ogc:expression handling for PropertyName for now │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - // handle Literals or Functions for now │ │ │ │ │ - this.writeOgcExpression(filter.value, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsGreaterThan": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsGreaterThan"); │ │ │ │ │ - // no ogc:expression handling for PropertyName for now │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - // handle Literals or Functions for now │ │ │ │ │ - this.writeOgcExpression(filter.value, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsLessThanOrEqualTo": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsLessThanOrEqualTo"); │ │ │ │ │ - // no ogc:expression handling for PropertyName for now │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - // handle Literals or Functions for now │ │ │ │ │ - this.writeOgcExpression(filter.value, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsGreaterThanOrEqualTo": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsGreaterThanOrEqualTo"); │ │ │ │ │ - // no ogc:expression handling for PropertyName for now │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - // handle Literals or Functions for now │ │ │ │ │ - this.writeOgcExpression(filter.value, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsBetween": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsBetween"); │ │ │ │ │ - // no ogc:expression handling for PropertyName for now │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - this.writeNode("LowerBoundary", filter, node); │ │ │ │ │ - this.writeNode("UpperBoundary", filter, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "PropertyName": function(filter) { │ │ │ │ │ - // no ogc:expression handling for now │ │ │ │ │ - return this.createElementNSPlus("ogc:PropertyName", { │ │ │ │ │ - value: filter.property │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "Literal": function(value) { │ │ │ │ │ - var encode = this.encodeLiteral || │ │ │ │ │ - OpenLayers.Format.Filter.v1.prototype.encodeLiteral; │ │ │ │ │ - return this.createElementNSPlus("ogc:Literal", { │ │ │ │ │ - value: encode(value) │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "LowerBoundary": function(filter) { │ │ │ │ │ - // handle Literals or Functions for now │ │ │ │ │ - var node = this.createElementNSPlus("ogc:LowerBoundary"); │ │ │ │ │ - this.writeOgcExpression(filter.lowerBoundary, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "UpperBoundary": function(filter) { │ │ │ │ │ - // handle Literals or Functions for now │ │ │ │ │ - var node = this.createElementNSPlus("ogc:UpperBoundary"); │ │ │ │ │ - this.writeNode("Literal", filter.upperBoundary, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "INTERSECTS": function(filter) { │ │ │ │ │ - return this.writeSpatial(filter, "Intersects"); │ │ │ │ │ - }, │ │ │ │ │ - "WITHIN": function(filter) { │ │ │ │ │ - return this.writeSpatial(filter, "Within"); │ │ │ │ │ - }, │ │ │ │ │ - "CONTAINS": function(filter) { │ │ │ │ │ - return this.writeSpatial(filter, "Contains"); │ │ │ │ │ - }, │ │ │ │ │ - "DWITHIN": function(filter) { │ │ │ │ │ - var node = this.writeSpatial(filter, "DWithin"); │ │ │ │ │ - this.writeNode("Distance", filter, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Distance": function(filter) { │ │ │ │ │ - return this.createElementNSPlus("ogc:Distance", { │ │ │ │ │ - attributes: { │ │ │ │ │ - units: filter.distanceUnits │ │ │ │ │ - }, │ │ │ │ │ - value: filter.distance │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "Function": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:Function", { │ │ │ │ │ - attributes: { │ │ │ │ │ - name: filter.name │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - var params = filter.params; │ │ │ │ │ - for (var i = 0, len = params.length; i < len; i++) { │ │ │ │ │ - this.writeOgcExpression(params[i], node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsNull": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsNull"); │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - return node; │ │ │ │ │ - } │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + var limit = Math.pow(2, z); │ │ │ │ │ + x = ((x % limit) + limit) % limit; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getFilterType │ │ │ │ │ - */ │ │ │ │ │ - getFilterType: function(filter) { │ │ │ │ │ - var filterType = this.filterMap[filter.type]; │ │ │ │ │ - if (!filterType) { │ │ │ │ │ - throw "Filter writing not supported for rule type: " + filter.type; │ │ │ │ │ - } │ │ │ │ │ - return filterType; │ │ │ │ │ + return { │ │ │ │ │ + 'x': x, │ │ │ │ │ + 'y': y, │ │ │ │ │ + 'z': z │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: filterMap │ │ │ │ │ - * {Object} Contains a member for each filter type. Values are node names │ │ │ │ │ - * for corresponding OGC Filter child elements. │ │ │ │ │ + /* APIMethod: setMap │ │ │ │ │ + * When the layer is added to a map, then we can fetch our origin │ │ │ │ │ + * (if we don't have one.) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - filterMap: { │ │ │ │ │ - "&&": "And", │ │ │ │ │ - "||": "Or", │ │ │ │ │ - "!": "Not", │ │ │ │ │ - "==": "PropertyIsEqualTo", │ │ │ │ │ - "!=": "PropertyIsNotEqualTo", │ │ │ │ │ - "<": "PropertyIsLessThan", │ │ │ │ │ - ">": "PropertyIsGreaterThan", │ │ │ │ │ - "<=": "PropertyIsLessThanOrEqualTo", │ │ │ │ │ - ">=": "PropertyIsGreaterThanOrEqualTo", │ │ │ │ │ - "..": "PropertyIsBetween", │ │ │ │ │ - "~": "PropertyIsLike", │ │ │ │ │ - "NULL": "PropertyIsNull", │ │ │ │ │ - "BBOX": "BBOX", │ │ │ │ │ - "DWITHIN": "DWITHIN", │ │ │ │ │ - "WITHIN": "WITHIN", │ │ │ │ │ - "CONTAINS": "CONTAINS", │ │ │ │ │ - "INTERSECTS": "INTERSECTS", │ │ │ │ │ - "FID": "_featureIds" │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (!this.tileOrigin) { │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, │ │ │ │ │ + this.maxExtent.bottom); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.Filter.v1" │ │ │ │ │ - │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.XYZ" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/Filter/v1_0_0.js │ │ │ │ │ + OpenLayers/Layer/Bing.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/GML/v2.js │ │ │ │ │ - * @requires OpenLayers/Format/Filter/v1.js │ │ │ │ │ + * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.Filter.v1_0_0 │ │ │ │ │ - * Write ogc:Filter version 1.0.0. │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.Bing │ │ │ │ │ + * Bing layer using direct tile access as provided by Bing Maps REST Services. │ │ │ │ │ + * See http://msdn.microsoft.com/en-us/library/ff701713.aspx for more │ │ │ │ │ + * information. Note: Terms of Service compliant use requires the map to be │ │ │ │ │ + * configured with an <OpenLayers.Control.Attribution> control and the │ │ │ │ │ + * attribution placed on or near the map. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.GML.v2> │ │ │ │ │ - * - <OpenLayers.Format.Filter.v1> │ │ │ │ │ + * - <OpenLayers.Layer.XYZ> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.Filter.v1_0_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.GML.v2, OpenLayers.Format.Filter.v1, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: VERSION │ │ │ │ │ - * {String} 1.0.0 │ │ │ │ │ - */ │ │ │ │ │ - VERSION: "1.0.0", │ │ │ │ │ +OpenLayers.Layer.Bing = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} http://www.opengis.net/ogc/filter/1.0.0/filter.xsd │ │ │ │ │ - */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/ogc/filter/1.0.0/filter.xsd", │ │ │ │ │ + /** │ │ │ │ │ + * Property: key │ │ │ │ │ + * {String} API key for Bing maps, get your own key │ │ │ │ │ + * at http://bingmapsportal.com/ . │ │ │ │ │ + */ │ │ │ │ │ + key: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.Filter.v1_0_0 │ │ │ │ │ - * Instances of this class are not created directly. Use the │ │ │ │ │ - * <OpenLayers.Format.Filter> constructor instead. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.GML.v2.prototype.initialize.apply( │ │ │ │ │ - this, [options] │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: serverResolutions │ │ │ │ │ + * {Array} the resolutions provided by the Bing servers. │ │ │ │ │ + */ │ │ │ │ │ + serverResolutions: [ │ │ │ │ │ + 156543.03390625, 78271.516953125, 39135.7584765625, │ │ │ │ │ + 19567.87923828125, 9783.939619140625, 4891.9698095703125, │ │ │ │ │ + 2445.9849047851562, 1222.9924523925781, 611.4962261962891, │ │ │ │ │ + 305.74811309814453, 152.87405654907226, 76.43702827453613, │ │ │ │ │ + 38.218514137268066, 19.109257068634033, 9.554628534317017, │ │ │ │ │ + 4.777314267158508, 2.388657133579254, 1.194328566789627, │ │ │ │ │ + 0.5971642833948135, 0.29858214169740677, 0.14929107084870338, │ │ │ │ │ + 0.07464553542435169 │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "ogc": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "PropertyIsEqualTo": function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.EQUAL_TO │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - obj.filters.push(filter); │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsNotEqualTo": function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.NOT_EQUAL_TO │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - obj.filters.push(filter); │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsLike": function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.LIKE │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - var wildCard = node.getAttribute("wildCard"); │ │ │ │ │ - var singleChar = node.getAttribute("singleChar"); │ │ │ │ │ - var esc = node.getAttribute("escape"); │ │ │ │ │ - filter.value2regex(wildCard, singleChar, esc); │ │ │ │ │ - obj.filters.push(filter); │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.Filter.v1.prototype.readers["ogc"]), │ │ │ │ │ - "gml": OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ - "feature": OpenLayers.Format.GML.v2.prototype.readers["feature"] │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: attributionTemplate │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + attributionTemplate: '<span class="olBingAttribution ${type}">' + │ │ │ │ │ + '<div><a target="_blank" href="http://www.bing.com/maps/">' + │ │ │ │ │ + '<img src="${logo}" /></a></div>${copyrights}' + │ │ │ │ │ + '<a style="white-space: nowrap" target="_blank" ' + │ │ │ │ │ + 'href="http://www.microsoft.com/maps/product/terms.html">' + │ │ │ │ │ + 'Terms of Use</a></span>', │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ - */ │ │ │ │ │ - writers: { │ │ │ │ │ - "ogc": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "PropertyIsEqualTo": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsEqualTo"); │ │ │ │ │ - // no ogc:expression handling for PropertyName for now │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - // handle Literals or Functions for now │ │ │ │ │ - this.writeOgcExpression(filter.value, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsNotEqualTo": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsNotEqualTo"); │ │ │ │ │ - // no ogc:expression handling for PropertyName for now │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - // handle Literals or Functions for now │ │ │ │ │ - this.writeOgcExpression(filter.value, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "PropertyIsLike": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsLike", { │ │ │ │ │ - attributes: { │ │ │ │ │ - wildCard: "*", │ │ │ │ │ - singleChar: ".", │ │ │ │ │ - escape: "!" │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - // no ogc:expression handling for now │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - // convert regex string to ogc string │ │ │ │ │ - this.writeNode("Literal", filter.regex2value(), node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "BBOX": function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:BBOX"); │ │ │ │ │ - // PropertyName is mandatory in 1.0.0, but e.g. GeoServer also │ │ │ │ │ - // accepts filters without it. When this is used with │ │ │ │ │ - // OpenLayers.Protocol.WFS, OpenLayers.Format.WFST will set a │ │ │ │ │ - // missing filter.property to the geometryName that is │ │ │ │ │ - // configured with the protocol, which defaults to "the_geom". │ │ │ │ │ - // So the only way to omit this mandatory property is to not │ │ │ │ │ - // set the property on the filter and to set the geometryName │ │ │ │ │ - // on the WFS protocol to null. The latter also happens when │ │ │ │ │ - // the protocol is configured without a geometryName and a │ │ │ │ │ - // featureNS. │ │ │ │ │ - filter.property && this.writeNode("PropertyName", filter, node); │ │ │ │ │ - var box = this.writeNode("gml:Box", filter.value, node); │ │ │ │ │ - if (filter.projection) { │ │ │ │ │ - box.setAttribute("srsName", filter.projection); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.Filter.v1.prototype.writers["ogc"]), │ │ │ │ │ - "gml": OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ - "feature": OpenLayers.Format.GML.v2.prototype.writers["feature"] │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: metadata │ │ │ │ │ + * {Object} Metadata for this layer, as returned by the callback script │ │ │ │ │ + */ │ │ │ │ │ + metadata: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: writeSpatial │ │ │ │ │ - * │ │ │ │ │ - * Read a {<OpenLayers.Filter.Spatial>} filter and converts it into XML. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter.Spatial>} The filter. │ │ │ │ │ - * name - {String} Name of the generated XML element. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The created XML element. │ │ │ │ │ - */ │ │ │ │ │ - writeSpatial: function(filter, name) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:" + name); │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - if (filter.value instanceof OpenLayers.Filter.Function) { │ │ │ │ │ - this.writeNode("Function", filter.value, node); │ │ │ │ │ - } else { │ │ │ │ │ - var child; │ │ │ │ │ - if (filter.value instanceof OpenLayers.Geometry) { │ │ │ │ │ - child = this.writeNode("feature:_geometry", filter.value).firstChild; │ │ │ │ │ - } else { │ │ │ │ │ - child = this.writeNode("gml:Box", filter.value); │ │ │ │ │ - } │ │ │ │ │ - if (filter.projection) { │ │ │ │ │ - child.setAttribute("srsName", filter.projection); │ │ │ │ │ - } │ │ │ │ │ - node.appendChild(child); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: protocolRegex │ │ │ │ │ + * {RegExp} Regular expression to match and replace http: in bing urls │ │ │ │ │ + */ │ │ │ │ │ + protocolRegex: /^http:/i, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: type │ │ │ │ │ + * {String} The layer identifier. Any non-birdseye imageryType │ │ │ │ │ + * from http://msdn.microsoft.com/en-us/library/ff701716.aspx can be │ │ │ │ │ + * used. Default is "Road". │ │ │ │ │ + */ │ │ │ │ │ + type: "Road", │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.Filter.v1_0_0" │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: culture │ │ │ │ │ + * {String} The culture identifier. See http://msdn.microsoft.com/en-us/library/ff701709.aspx │ │ │ │ │ + * for the definition and the possible values. Default is "en-US". │ │ │ │ │ + */ │ │ │ │ │ + culture: "en-US", │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WFST/v1_0_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: metadataParams │ │ │ │ │ + * {Object} Optional url parameters for the Get Imagery Metadata request │ │ │ │ │ + * as described here: http://msdn.microsoft.com/en-us/library/ff701716.aspx │ │ │ │ │ + */ │ │ │ │ │ + metadataParams: null, │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** APIProperty: tileOptions │ │ │ │ │ + * {Object} optional configuration options for <OpenLayers.Tile> instances │ │ │ │ │ + * created by this Layer. Default is │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * {crossOriginKeyword: 'anonymous'} │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ + tileOptions: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WFST/v1.js │ │ │ │ │ - * @requires OpenLayers/Format/Filter/v1_0_0.js │ │ │ │ │ - */ │ │ │ │ │ + /** APIProperty: protocol │ │ │ │ │ + * {String} Protocol to use to fetch Imagery Metadata, tiles and bing logo │ │ │ │ │ + * Can be 'http:' 'https:' or '' │ │ │ │ │ + * │ │ │ │ │ + * Warning: tiles may not be available under both HTTP and HTTPS protocols. │ │ │ │ │ + * Microsoft approved use of both HTTP and HTTPS urls for tiles. However │ │ │ │ │ + * this is undocumented and the Imagery Metadata API always returns HTTP │ │ │ │ │ + * urls. │ │ │ │ │ + * │ │ │ │ │ + * Default is '', unless when executed from a file:/// uri, in which case │ │ │ │ │ + * it is 'http:'. │ │ │ │ │ + */ │ │ │ │ │ + protocol: ~window.location.href.indexOf('http') ? '' : 'http:', │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WFST.v1_0_0 │ │ │ │ │ - * A format for creating WFS v1.0.0 transactions. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Format.WFST.v1_0_0> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.Filter.v1_0_0> │ │ │ │ │ - * - <OpenLayers.Format.WFST.v1> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WFST.v1_0_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.Filter.v1_0_0, OpenLayers.Format.WFST.v1, { │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.Bing │ │ │ │ │ + * Create a new Bing layer. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var road = new OpenLayers.Layer.Bing({ │ │ │ │ │ + * name: "My Bing Aerial Layer", │ │ │ │ │ + * type: "Aerial", │ │ │ │ │ + * key: "my-api-key-here", │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Configuration properties for the layer. │ │ │ │ │ + * │ │ │ │ │ + * Required configuration properties: │ │ │ │ │ + * key - {String} Bing Maps API key for your application. Get one at │ │ │ │ │ + * http://bingmapsportal.com/. │ │ │ │ │ + * type - {String} The layer identifier. Any non-birdseye imageryType │ │ │ │ │ + * from http://msdn.microsoft.com/en-us/library/ff701716.aspx can be │ │ │ │ │ + * used. │ │ │ │ │ + * │ │ │ │ │ + * Any other documented layer properties can be provided in the config object. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + sphericalMercator: true │ │ │ │ │ + }, options); │ │ │ │ │ + var name = options.name || "Bing " + (options.type || this.type); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} WFS version number. │ │ │ │ │ - */ │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ + var newArgs = [name, null, options]; │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, newArgs); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + crossOriginKeyword: 'anonymous' │ │ │ │ │ + }, this.options.tileOptions); │ │ │ │ │ + this.loadMetadata(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: srsNameInQuery │ │ │ │ │ - * {Boolean} If true the reference system is passed in Query requests │ │ │ │ │ - * via the "srsName" attribute to the "wfs:Query" element, this │ │ │ │ │ - * property defaults to false as it isn't WFS 1.0.0 compliant. │ │ │ │ │ - */ │ │ │ │ │ - srsNameInQuery: false, │ │ │ │ │ + /** │ │ │ │ │ + * Method: loadMetadata │ │ │ │ │ + */ │ │ │ │ │ + loadMetadata: function() { │ │ │ │ │ + this._callbackId = "_callback_" + this.id.replace(/\./g, "_"); │ │ │ │ │ + // link the processMetadata method to the global scope and bind it │ │ │ │ │ + // to this instance │ │ │ │ │ + window[this._callbackId] = OpenLayers.Function.bind( │ │ │ │ │ + OpenLayers.Layer.Bing.processMetadata, this │ │ │ │ │ + ); │ │ │ │ │ + var params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + key: this.key, │ │ │ │ │ + jsonp: this._callbackId, │ │ │ │ │ + include: "ImageryProviders" │ │ │ │ │ + }, this.metadataParams); │ │ │ │ │ + var url = this.protocol + "//dev.virtualearth.net/REST/v1/Imagery/Metadata/" + │ │ │ │ │ + this.type + "?" + OpenLayers.Util.getParameterString(params); │ │ │ │ │ + var script = document.createElement("script"); │ │ │ │ │ + script.type = "text/javascript"; │ │ │ │ │ + script.src = url; │ │ │ │ │ + script.id = this._callbackId; │ │ │ │ │ + document.getElementsByTagName("head")[0].appendChild(script); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: schemaLocations │ │ │ │ │ - * {Object} Properties are namespace aliases, values are schema locations. │ │ │ │ │ - */ │ │ │ │ │ - schemaLocations: { │ │ │ │ │ - "wfs": "http://schemas.opengis.net/wfs/1.0.0/WFS-transaction.xsd" │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: initLayer │ │ │ │ │ + * │ │ │ │ │ + * Sets layer properties according to the metadata provided by the API │ │ │ │ │ + */ │ │ │ │ │ + initLayer: function() { │ │ │ │ │ + var res = this.metadata.resourceSets[0].resources[0]; │ │ │ │ │ + var url = res.imageUrl.replace("{quadkey}", "${quadkey}"); │ │ │ │ │ + url = url.replace("{culture}", this.culture); │ │ │ │ │ + url = url.replace(this.protocolRegex, this.protocol); │ │ │ │ │ + this.url = []; │ │ │ │ │ + for (var i = 0; i < res.imageUrlSubdomains.length; ++i) { │ │ │ │ │ + this.url.push(url.replace("{subdomain}", res.imageUrlSubdomains[i])); │ │ │ │ │ + } │ │ │ │ │ + this.addOptions({ │ │ │ │ │ + maxResolution: Math.min( │ │ │ │ │ + this.serverResolutions[res.zoomMin], │ │ │ │ │ + this.maxResolution || Number.POSITIVE_INFINITY │ │ │ │ │ + ), │ │ │ │ │ + numZoomLevels: Math.min( │ │ │ │ │ + res.zoomMax + 1 - res.zoomMin, this.numZoomLevels │ │ │ │ │ + ) │ │ │ │ │ + }, true); │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + this.redraw(); │ │ │ │ │ + } │ │ │ │ │ + this.updateAttribution(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WFST.v1_0_0 │ │ │ │ │ - * A class for parsing and generating WFS v1.0.0 transactions. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ - * featureNS - {String} Feature namespace (optional). │ │ │ │ │ - * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ - * if featureNS is provided). Default is 'feature'. │ │ │ │ │ - * geometryName - {String} Name of geometry attribute. Default is 'the_geom'. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.Filter.v1_0_0.prototype.initialize.apply(this, [options]); │ │ │ │ │ - OpenLayers.Format.WFST.v1.prototype.initialize.apply(this, [options]); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * │ │ │ │ │ + * Paramters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + if (!this.url) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var xyz = this.getXYZ(bounds), │ │ │ │ │ + x = xyz.x, │ │ │ │ │ + y = xyz.y, │ │ │ │ │ + z = xyz.z; │ │ │ │ │ + var quadDigits = []; │ │ │ │ │ + for (var i = z; i > 0; --i) { │ │ │ │ │ + var digit = '0'; │ │ │ │ │ + var mask = 1 << (i - 1); │ │ │ │ │ + if ((x & mask) != 0) { │ │ │ │ │ + digit++; │ │ │ │ │ + } │ │ │ │ │ + if ((y & mask) != 0) { │ │ │ │ │ + digit++; │ │ │ │ │ + digit++; │ │ │ │ │ + } │ │ │ │ │ + quadDigits.push(digit); │ │ │ │ │ + } │ │ │ │ │ + var quadKey = quadDigits.join(""); │ │ │ │ │ + var url = this.selectUrl('' + x + y + z, this.url); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: readNode │ │ │ │ │ - * Shorthand for applying one of the named readers given the node │ │ │ │ │ - * namespace and local name. Readers take two args (node, obj) and │ │ │ │ │ - * generally extend or modify the second. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} The node to be read (required). │ │ │ │ │ - * obj - {Object} The object to be modified (optional). │ │ │ │ │ - * first - {Boolean} Should be set to true for the first node read. This │ │ │ │ │ - * is usually the readNode call in the read method. Without this being │ │ │ │ │ - * set, auto-configured properties will stick on subsequent reads. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The input object, modified (or a new one if none was provided). │ │ │ │ │ - */ │ │ │ │ │ - readNode: function(node, obj, first) { │ │ │ │ │ - // Not the superclass, only the mixin classes inherit from │ │ │ │ │ - // Format.GML.v2. We need this because we don't want to get readNode │ │ │ │ │ - // from the superclass's superclass, which is OpenLayers.Format.XML. │ │ │ │ │ - return OpenLayers.Format.GML.v2.prototype.readNode.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + return OpenLayers.String.format(url, { │ │ │ │ │ + 'quadkey': quadKey │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "WFS_TransactionResponse": function(node, obj) { │ │ │ │ │ - obj.insertIds = []; │ │ │ │ │ - obj.success = false; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "InsertResult": function(node, container) { │ │ │ │ │ - var obj = { │ │ │ │ │ - fids: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - container.insertIds = container.insertIds.concat(obj.fids); │ │ │ │ │ - }, │ │ │ │ │ - "TransactionResult": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "Status": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "SUCCESS": function(node, obj) { │ │ │ │ │ - obj.success = true; │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateAttribution │ │ │ │ │ + * Updates the attribution according to the requirements outlined in │ │ │ │ │ + * http://gis.638310.n2.nabble.com/Bing-imagery-td5789168.html │ │ │ │ │ + */ │ │ │ │ │ + updateAttribution: function() { │ │ │ │ │ + var metadata = this.metadata; │ │ │ │ │ + if (!metadata.resourceSets || !this.map || !this.map.center) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var res = metadata.resourceSets[0].resources[0]; │ │ │ │ │ + var extent = this.map.getExtent().transform( │ │ │ │ │ + this.map.getProjectionObject(), │ │ │ │ │ + new OpenLayers.Projection("EPSG:4326") │ │ │ │ │ + ); │ │ │ │ │ + var providers = res.imageryProviders || [], │ │ │ │ │ + zoom = OpenLayers.Util.indexOf(this.serverResolutions, │ │ │ │ │ + this.getServerResolution()), │ │ │ │ │ + copyrights = "", │ │ │ │ │ + provider, i, ii, j, jj, bbox, coverage; │ │ │ │ │ + for (i = 0, ii = providers.length; i < ii; ++i) { │ │ │ │ │ + provider = providers[i]; │ │ │ │ │ + for (j = 0, jj = provider.coverageAreas.length; j < jj; ++j) { │ │ │ │ │ + coverage = provider.coverageAreas[j]; │ │ │ │ │ + // axis order provided is Y,X │ │ │ │ │ + bbox = OpenLayers.Bounds.fromArray(coverage.bbox, true); │ │ │ │ │ + if (extent.intersectsBounds(bbox) && │ │ │ │ │ + zoom <= coverage.zoomMax && zoom >= coverage.zoomMin) { │ │ │ │ │ + copyrights += provider.attribution + " "; │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.WFST.v1.prototype.readers["wfs"]), │ │ │ │ │ - "gml": OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ - "feature": OpenLayers.Format.GML.v2.prototype.readers["feature"], │ │ │ │ │ - "ogc": OpenLayers.Format.Filter.v1_0_0.prototype.readers["ogc"] │ │ │ │ │ - }, │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var logo = metadata.brandLogoUri.replace(this.protocolRegex, this.protocol); │ │ │ │ │ + this.attribution = OpenLayers.String.format(this.attributionTemplate, { │ │ │ │ │ + type: this.type.toLowerCase(), │ │ │ │ │ + logo: logo, │ │ │ │ │ + copyrights: copyrights │ │ │ │ │ + }); │ │ │ │ │ + this.map && this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "attribution" │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ - */ │ │ │ │ │ - writers: { │ │ │ │ │ - "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "Query": function(options) { │ │ │ │ │ - options = OpenLayers.Util.extend({ │ │ │ │ │ - featureNS: this.featureNS, │ │ │ │ │ - featurePrefix: this.featurePrefix, │ │ │ │ │ - featureType: this.featureType, │ │ │ │ │ - srsName: this.srsName, │ │ │ │ │ - srsNameInQuery: this.srsNameInQuery │ │ │ │ │ - }, options); │ │ │ │ │ - var prefix = options.featurePrefix; │ │ │ │ │ - var node = this.createElementNSPlus("wfs:Query", { │ │ │ │ │ - attributes: { │ │ │ │ │ - typeName: (prefix ? prefix + ":" : "") + │ │ │ │ │ - options.featureType │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (options.srsNameInQuery && options.srsName) { │ │ │ │ │ - node.setAttribute("srsName", options.srsName); │ │ │ │ │ - } │ │ │ │ │ - if (options.featureNS) { │ │ │ │ │ - node.setAttribute("xmlns:" + prefix, options.featureNS); │ │ │ │ │ - } │ │ │ │ │ - if (options.propertyNames) { │ │ │ │ │ - for (var i = 0, len = options.propertyNames.length; i < len; i++) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "ogc:PropertyName", { │ │ │ │ │ - property: options.propertyNames[i] │ │ │ │ │ - }, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (options.filter) { │ │ │ │ │ - this.setFilterProperty(options.filter); │ │ │ │ │ - this.writeNode("ogc:Filter", options.filter, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WFST.v1.prototype.writers["wfs"]), │ │ │ │ │ - "gml": OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ - "feature": OpenLayers.Format.GML.v2.prototype.writers["feature"], │ │ │ │ │ - "ogc": OpenLayers.Format.Filter.v1_0_0.prototype.writers["ogc"] │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + */ │ │ │ │ │ + setMap: function() { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.register("moveend", this, this.updateAttribution); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFST.v1_0_0" │ │ │ │ │ - }); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Bing>} An exact clone of this <OpenLayers.Layer.Bing> │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Bing(this.options); │ │ │ │ │ + } │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.map && │ │ │ │ │ + this.map.events.unregister("moveend", this, this.updateAttribution); │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Bing" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Function: OpenLayers.Layer.Bing.processMetadata │ │ │ │ │ + * This function will be bound to an instance, linked to the global scope with │ │ │ │ │ + * an id, and called by the JSONP script returned by the API. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * metadata - {Object} metadata as returned by the API │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Bing.processMetadata = function(metadata) { │ │ │ │ │ + this.metadata = metadata; │ │ │ │ │ + this.initLayer(); │ │ │ │ │ + var script = document.getElementById(this._callbackId); │ │ │ │ │ + script.parentNode.removeChild(script); │ │ │ │ │ + window[this._callbackId] = undefined; // cannot delete from window in IE │ │ │ │ │ + delete this._callbackId; │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/WFS/v1_0_0.js │ │ │ │ │ + OpenLayers/Layer/OSM.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Protocol/WFS/v1.js │ │ │ │ │ - * @requires OpenLayers/Format/WFST/v1_0_0.js │ │ │ │ │ + * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Protocol.WFS.v1_0_0 │ │ │ │ │ - * A WFS v1.0.0 protocol for vector layers. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Protocol.WFS.v1_0_0> constructor. │ │ │ │ │ + * Class: OpenLayers.Layer.OSM │ │ │ │ │ + * This layer allows accessing OpenStreetMap tiles. By default the OpenStreetMap │ │ │ │ │ + * hosted tile.openstreetmap.org Mapnik tileset is used. If you wish to use │ │ │ │ │ + * a different layer instead, you need to provide a different │ │ │ │ │ + * URL to the constructor. Here's an example for using OpenCycleMap: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * new OpenLayers.Layer.OSM("OpenCycleMap", │ │ │ │ │ + * ["http://a.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ + * "http://b.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ + * "http://c.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png"]); │ │ │ │ │ + * (end) │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol.WFS.v1> │ │ │ │ │ + * - <OpenLayers.Layer.XYZ> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.WFS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ +OpenLayers.Layer.OSM = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} WFS version number. │ │ │ │ │ + * APIProperty: name │ │ │ │ │ + * {String} The layer name. Defaults to "OpenStreetMap" if the first │ │ │ │ │ + * argument to the constructor is null or undefined. │ │ │ │ │ */ │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ + name: "OpenStreetMap", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.WFS.v1_0_0 │ │ │ │ │ - * A class for giving layers WFS v1.0.0 protocol. │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String} The tileset URL scheme. Defaults to │ │ │ │ │ + * : http://[a|b|c].tile.openstreetmap.org/${z}/${x}/${y}.png │ │ │ │ │ + * (the official OSM tileset) if the second argument to the constructor │ │ │ │ │ + * is null or undefined. To use another tileset you can have something │ │ │ │ │ + * like this: │ │ │ │ │ + * (code) │ │ │ │ │ + * new OpenLayers.Layer.OSM("OpenCycleMap", │ │ │ │ │ + * ["http://a.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ + * "http://b.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ + * "http://c.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png"]); │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ + url: [ │ │ │ │ │ + 'http://a.tile.openstreetmap.org/${z}/${x}/${y}.png', │ │ │ │ │ + 'http://b.tile.openstreetmap.org/${z}/${x}/${y}.png', │ │ │ │ │ + 'http://c.tile.openstreetmap.org/${z}/${x}/${y}.png' │ │ │ │ │ + ], │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: attribution │ │ │ │ │ + * {String} The layer attribution. │ │ │ │ │ + */ │ │ │ │ │ + attribution: "© <a href='http://www.openstreetmap.org/copyright'>OpenStreetMap</a> contributors", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: sphericalMercator │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + sphericalMercator: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: wrapDateLine │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + wrapDateLine: true, │ │ │ │ │ + │ │ │ │ │ + /** APIProperty: tileOptions │ │ │ │ │ + * {Object} optional configuration options for <OpenLayers.Tile> instances │ │ │ │ │ + * created by this Layer. Default is │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * (code) │ │ │ │ │ + * {crossOriginKeyword: 'anonymous'} │ │ │ │ │ + * (end) │ │ │ │ │ * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ - * featureNS - {String} Feature namespace (optional). │ │ │ │ │ - * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ - * if featureNS is provided). Default is 'feature'. │ │ │ │ │ - * geometryName - {String} Name of geometry attribute. Default is 'the_geom'. │ │ │ │ │ + * When using OSM tilesets other than the default ones, it may be │ │ │ │ │ + * necessary to set this to │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * {crossOriginKeyword: null} │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * if the server does not send Access-Control-Allow-Origin headers. │ │ │ │ │ */ │ │ │ │ │ + tileOptions: null, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.WFS.v1_0_0" │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.OSM │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} The layer name. │ │ │ │ │ + * url - {String} The tileset URL scheme. │ │ │ │ │ + * options - {Object} Configuration options for the layer. Any inherited │ │ │ │ │ + * layer option can be set in this object (e.g. │ │ │ │ │ + * <OpenLayers.Layer.Grid.buffer>). │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + crossOriginKeyword: 'anonymous' │ │ │ │ │ + }, this.options && this.options.tileOptions); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clone │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.OSM( │ │ │ │ │ + this.name, this.url, this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.OSM" │ │ │ │ │ }); │ │ │ ├── ./usr/share/javascript/openlayers/OpenLayers.mobile.min.js │ │ │ │ ├── js-beautify {} │ │ │ │ │ @@ -62,204 +62,14 @@ │ │ │ │ │ var sourceIsEvt = typeof window.Event == "function" && source instanceof window.Event; │ │ │ │ │ if (!sourceIsEvt && source.hasOwnProperty && source.hasOwnProperty("toString")) { │ │ │ │ │ destination.toString = source.toString │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ return destination │ │ │ │ │ }; │ │ │ │ │ -OpenLayers.Util = OpenLayers.Util || {}; │ │ │ │ │ -OpenLayers.Util.vendorPrefix = function() { │ │ │ │ │ - "use strict"; │ │ │ │ │ - var VENDOR_PREFIXES = ["", "O", "ms", "Moz", "Webkit"], │ │ │ │ │ - divStyle = document.createElement("div").style, │ │ │ │ │ - cssCache = {}, │ │ │ │ │ - jsCache = {}; │ │ │ │ │ - │ │ │ │ │ - function domToCss(prefixedDom) { │ │ │ │ │ - if (!prefixedDom) { │ │ │ │ │ - return null │ │ │ │ │ - } │ │ │ │ │ - return prefixedDom.replace(/([A-Z])/g, function(c) { │ │ │ │ │ - return "-" + c.toLowerCase() │ │ │ │ │ - }).replace(/^ms-/, "-ms-") │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function css(property) { │ │ │ │ │ - if (cssCache[property] === undefined) { │ │ │ │ │ - var domProperty = property.replace(/(-[\s\S])/g, function(c) { │ │ │ │ │ - return c.charAt(1).toUpperCase() │ │ │ │ │ - }); │ │ │ │ │ - var prefixedDom = style(domProperty); │ │ │ │ │ - cssCache[property] = domToCss(prefixedDom) │ │ │ │ │ - } │ │ │ │ │ - return cssCache[property] │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function js(obj, property) { │ │ │ │ │ - if (jsCache[property] === undefined) { │ │ │ │ │ - var tmpProp, i = 0, │ │ │ │ │ - l = VENDOR_PREFIXES.length, │ │ │ │ │ - prefix, isStyleObj = typeof obj.cssText !== "undefined"; │ │ │ │ │ - jsCache[property] = null; │ │ │ │ │ - for (; i < l; i++) { │ │ │ │ │ - prefix = VENDOR_PREFIXES[i]; │ │ │ │ │ - if (prefix) { │ │ │ │ │ - if (!isStyleObj) { │ │ │ │ │ - prefix = prefix.toLowerCase() │ │ │ │ │ - } │ │ │ │ │ - tmpProp = prefix + property.charAt(0).toUpperCase() + property.slice(1) │ │ │ │ │ - } else { │ │ │ │ │ - tmpProp = property │ │ │ │ │ - } │ │ │ │ │ - if (obj[tmpProp] !== undefined) { │ │ │ │ │ - jsCache[property] = tmpProp; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return jsCache[property] │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function style(property) { │ │ │ │ │ - return js(divStyle, property) │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - css: css, │ │ │ │ │ - js: js, │ │ │ │ │ - style: style, │ │ │ │ │ - cssCache: cssCache, │ │ │ │ │ - jsCache: jsCache │ │ │ │ │ - } │ │ │ │ │ -}(); │ │ │ │ │ -OpenLayers.Animation = function(window) { │ │ │ │ │ - var requestAnimationFrame = OpenLayers.Util.vendorPrefix.js(window, "requestAnimationFrame"); │ │ │ │ │ - var isNative = !!requestAnimationFrame; │ │ │ │ │ - var requestFrame = function() { │ │ │ │ │ - var request = window[requestAnimationFrame] || function(callback, element) { │ │ │ │ │ - window.setTimeout(callback, 16) │ │ │ │ │ - }; │ │ │ │ │ - return function(callback, element) { │ │ │ │ │ - request.apply(window, [callback, element]) │ │ │ │ │ - } │ │ │ │ │ - }(); │ │ │ │ │ - var counter = 0; │ │ │ │ │ - var loops = {}; │ │ │ │ │ - │ │ │ │ │ - function start(callback, duration, element) { │ │ │ │ │ - duration = duration > 0 ? duration : Number.POSITIVE_INFINITY; │ │ │ │ │ - var id = ++counter; │ │ │ │ │ - var start = +new Date; │ │ │ │ │ - loops[id] = function() { │ │ │ │ │ - if (loops[id] && +new Date - start <= duration) { │ │ │ │ │ - callback(); │ │ │ │ │ - if (loops[id]) { │ │ │ │ │ - requestFrame(loops[id], element) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - delete loops[id] │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - requestFrame(loops[id], element); │ │ │ │ │ - return id │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function stop(id) { │ │ │ │ │ - delete loops[id] │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - isNative: isNative, │ │ │ │ │ - requestFrame: requestFrame, │ │ │ │ │ - start: start, │ │ │ │ │ - stop: stop │ │ │ │ │ - } │ │ │ │ │ -}(window); │ │ │ │ │ -OpenLayers.Kinetic = OpenLayers.Class({ │ │ │ │ │ - threshold: 0, │ │ │ │ │ - deceleration: .0035, │ │ │ │ │ - nbPoints: 100, │ │ │ │ │ - delay: 200, │ │ │ │ │ - points: undefined, │ │ │ │ │ - timerId: undefined, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options) │ │ │ │ │ - }, │ │ │ │ │ - begin: function() { │ │ │ │ │ - OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ - this.timerId = undefined; │ │ │ │ │ - this.points = [] │ │ │ │ │ - }, │ │ │ │ │ - update: function(xy) { │ │ │ │ │ - this.points.unshift({ │ │ │ │ │ - xy: xy, │ │ │ │ │ - tick: (new Date).getTime() │ │ │ │ │ - }); │ │ │ │ │ - if (this.points.length > this.nbPoints) { │ │ │ │ │ - this.points.pop() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - end: function(xy) { │ │ │ │ │ - var last, now = (new Date).getTime(); │ │ │ │ │ - for (var i = 0, l = this.points.length, point; i < l; i++) { │ │ │ │ │ - point = this.points[i]; │ │ │ │ │ - if (now - point.tick > this.delay) { │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - last = point │ │ │ │ │ - } │ │ │ │ │ - if (!last) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var time = (new Date).getTime() - last.tick; │ │ │ │ │ - var dist = Math.sqrt(Math.pow(xy.x - last.xy.x, 2) + Math.pow(xy.y - last.xy.y, 2)); │ │ │ │ │ - var speed = dist / time; │ │ │ │ │ - if (speed == 0 || speed < this.threshold) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var theta = Math.asin((xy.y - last.xy.y) / dist); │ │ │ │ │ - if (last.xy.x <= xy.x) { │ │ │ │ │ - theta = Math.PI - theta │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - speed: speed, │ │ │ │ │ - theta: theta │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - move: function(info, callback) { │ │ │ │ │ - var v0 = info.speed; │ │ │ │ │ - var fx = Math.cos(info.theta); │ │ │ │ │ - var fy = -Math.sin(info.theta); │ │ │ │ │ - var initialTime = (new Date).getTime(); │ │ │ │ │ - var lastX = 0; │ │ │ │ │ - var lastY = 0; │ │ │ │ │ - var timerCallback = function() { │ │ │ │ │ - if (this.timerId == null) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var t = (new Date).getTime() - initialTime; │ │ │ │ │ - var p = -this.deceleration * Math.pow(t, 2) / 2 + v0 * t; │ │ │ │ │ - var x = p * fx; │ │ │ │ │ - var y = p * fy; │ │ │ │ │ - var args = {}; │ │ │ │ │ - args.end = false; │ │ │ │ │ - var v = -this.deceleration * t + v0; │ │ │ │ │ - if (v <= 0) { │ │ │ │ │ - OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - args.end = true │ │ │ │ │ - } │ │ │ │ │ - args.x = x - lastX; │ │ │ │ │ - args.y = y - lastY; │ │ │ │ │ - lastX = x; │ │ │ │ │ - lastY = y; │ │ │ │ │ - callback(args.x, args.y, args.end) │ │ │ │ │ - }; │ │ │ │ │ - this.timerId = OpenLayers.Animation.start(OpenLayers.Function.bind(timerCallback, this)) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Kinetic" │ │ │ │ │ -}); │ │ │ │ │ OpenLayers.String = { │ │ │ │ │ startsWith: function(str, sub) { │ │ │ │ │ return str.indexOf(sub) == 0 │ │ │ │ │ }, │ │ │ │ │ contains: function(str, sub) { │ │ │ │ │ return str.indexOf(sub) != -1 │ │ │ │ │ }, │ │ │ │ │ @@ -1795,14 +1605,78 @@ │ │ │ │ │ if (axis == "lon") { │ │ │ │ │ str += coordinate < 0 ? OpenLayers.i18n("W") : OpenLayers.i18n("E") │ │ │ │ │ } else { │ │ │ │ │ str += coordinate < 0 ? OpenLayers.i18n("S") : OpenLayers.i18n("N") │ │ │ │ │ } │ │ │ │ │ return str │ │ │ │ │ }; │ │ │ │ │ +OpenLayers.Util = OpenLayers.Util || {}; │ │ │ │ │ +OpenLayers.Util.vendorPrefix = function() { │ │ │ │ │ + "use strict"; │ │ │ │ │ + var VENDOR_PREFIXES = ["", "O", "ms", "Moz", "Webkit"], │ │ │ │ │ + divStyle = document.createElement("div").style, │ │ │ │ │ + cssCache = {}, │ │ │ │ │ + jsCache = {}; │ │ │ │ │ + │ │ │ │ │ + function domToCss(prefixedDom) { │ │ │ │ │ + if (!prefixedDom) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ + return prefixedDom.replace(/([A-Z])/g, function(c) { │ │ │ │ │ + return "-" + c.toLowerCase() │ │ │ │ │ + }).replace(/^ms-/, "-ms-") │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function css(property) { │ │ │ │ │ + if (cssCache[property] === undefined) { │ │ │ │ │ + var domProperty = property.replace(/(-[\s\S])/g, function(c) { │ │ │ │ │ + return c.charAt(1).toUpperCase() │ │ │ │ │ + }); │ │ │ │ │ + var prefixedDom = style(domProperty); │ │ │ │ │ + cssCache[property] = domToCss(prefixedDom) │ │ │ │ │ + } │ │ │ │ │ + return cssCache[property] │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function js(obj, property) { │ │ │ │ │ + if (jsCache[property] === undefined) { │ │ │ │ │ + var tmpProp, i = 0, │ │ │ │ │ + l = VENDOR_PREFIXES.length, │ │ │ │ │ + prefix, isStyleObj = typeof obj.cssText !== "undefined"; │ │ │ │ │ + jsCache[property] = null; │ │ │ │ │ + for (; i < l; i++) { │ │ │ │ │ + prefix = VENDOR_PREFIXES[i]; │ │ │ │ │ + if (prefix) { │ │ │ │ │ + if (!isStyleObj) { │ │ │ │ │ + prefix = prefix.toLowerCase() │ │ │ │ │ + } │ │ │ │ │ + tmpProp = prefix + property.charAt(0).toUpperCase() + property.slice(1) │ │ │ │ │ + } else { │ │ │ │ │ + tmpProp = property │ │ │ │ │ + } │ │ │ │ │ + if (obj[tmpProp] !== undefined) { │ │ │ │ │ + jsCache[property] = tmpProp; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return jsCache[property] │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function style(property) { │ │ │ │ │ + return js(divStyle, property) │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + css: css, │ │ │ │ │ + js: js, │ │ │ │ │ + style: style, │ │ │ │ │ + cssCache: cssCache, │ │ │ │ │ + jsCache: jsCache │ │ │ │ │ + } │ │ │ │ │ +}(); │ │ │ │ │ OpenLayers.Event = { │ │ │ │ │ observers: false, │ │ │ │ │ KEY_SPACE: 32, │ │ │ │ │ KEY_BACKSPACE: 8, │ │ │ │ │ KEY_TAB: 9, │ │ │ │ │ KEY_RETURN: 13, │ │ │ │ │ KEY_ESC: 27, │ │ │ │ │ @@ -2253,14 +2127,56 @@ │ │ │ │ │ e.touches = touches.slice(); │ │ │ │ │ handler(e) │ │ │ │ │ }; │ │ │ │ │ OpenLayers.Event.observe(element, "MSPointerUp", cb) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Events" │ │ │ │ │ }); │ │ │ │ │ +OpenLayers.Animation = function(window) { │ │ │ │ │ + var requestAnimationFrame = OpenLayers.Util.vendorPrefix.js(window, "requestAnimationFrame"); │ │ │ │ │ + var isNative = !!requestAnimationFrame; │ │ │ │ │ + var requestFrame = function() { │ │ │ │ │ + var request = window[requestAnimationFrame] || function(callback, element) { │ │ │ │ │ + window.setTimeout(callback, 16) │ │ │ │ │ + }; │ │ │ │ │ + return function(callback, element) { │ │ │ │ │ + request.apply(window, [callback, element]) │ │ │ │ │ + } │ │ │ │ │ + }(); │ │ │ │ │ + var counter = 0; │ │ │ │ │ + var loops = {}; │ │ │ │ │ + │ │ │ │ │ + function start(callback, duration, element) { │ │ │ │ │ + duration = duration > 0 ? duration : Number.POSITIVE_INFINITY; │ │ │ │ │ + var id = ++counter; │ │ │ │ │ + var start = +new Date; │ │ │ │ │ + loops[id] = function() { │ │ │ │ │ + if (loops[id] && +new Date - start <= duration) { │ │ │ │ │ + callback(); │ │ │ │ │ + if (loops[id]) { │ │ │ │ │ + requestFrame(loops[id], element) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + delete loops[id] │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + requestFrame(loops[id], element); │ │ │ │ │ + return id │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function stop(id) { │ │ │ │ │ + delete loops[id] │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + isNative: isNative, │ │ │ │ │ + requestFrame: requestFrame, │ │ │ │ │ + start: start, │ │ │ │ │ + stop: stop │ │ │ │ │ + } │ │ │ │ │ +}(window); │ │ │ │ │ OpenLayers.Tween = OpenLayers.Class({ │ │ │ │ │ easing: null, │ │ │ │ │ begin: null, │ │ │ │ │ finish: null, │ │ │ │ │ duration: null, │ │ │ │ │ callbacks: null, │ │ │ │ │ time: null, │ │ │ │ │ @@ -3681,14 +3597,98 @@ │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Map" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Map.TILE_WIDTH = 256; │ │ │ │ │ OpenLayers.Map.TILE_HEIGHT = 256; │ │ │ │ │ +OpenLayers.Kinetic = OpenLayers.Class({ │ │ │ │ │ + threshold: 0, │ │ │ │ │ + deceleration: .0035, │ │ │ │ │ + nbPoints: 100, │ │ │ │ │ + delay: 200, │ │ │ │ │ + points: undefined, │ │ │ │ │ + timerId: undefined, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options) │ │ │ │ │ + }, │ │ │ │ │ + begin: function() { │ │ │ │ │ + OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ + this.timerId = undefined; │ │ │ │ │ + this.points = [] │ │ │ │ │ + }, │ │ │ │ │ + update: function(xy) { │ │ │ │ │ + this.points.unshift({ │ │ │ │ │ + xy: xy, │ │ │ │ │ + tick: (new Date).getTime() │ │ │ │ │ + }); │ │ │ │ │ + if (this.points.length > this.nbPoints) { │ │ │ │ │ + this.points.pop() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + end: function(xy) { │ │ │ │ │ + var last, now = (new Date).getTime(); │ │ │ │ │ + for (var i = 0, l = this.points.length, point; i < l; i++) { │ │ │ │ │ + point = this.points[i]; │ │ │ │ │ + if (now - point.tick > this.delay) { │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + last = point │ │ │ │ │ + } │ │ │ │ │ + if (!last) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var time = (new Date).getTime() - last.tick; │ │ │ │ │ + var dist = Math.sqrt(Math.pow(xy.x - last.xy.x, 2) + Math.pow(xy.y - last.xy.y, 2)); │ │ │ │ │ + var speed = dist / time; │ │ │ │ │ + if (speed == 0 || speed < this.threshold) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var theta = Math.asin((xy.y - last.xy.y) / dist); │ │ │ │ │ + if (last.xy.x <= xy.x) { │ │ │ │ │ + theta = Math.PI - theta │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + speed: speed, │ │ │ │ │ + theta: theta │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + move: function(info, callback) { │ │ │ │ │ + var v0 = info.speed; │ │ │ │ │ + var fx = Math.cos(info.theta); │ │ │ │ │ + var fy = -Math.sin(info.theta); │ │ │ │ │ + var initialTime = (new Date).getTime(); │ │ │ │ │ + var lastX = 0; │ │ │ │ │ + var lastY = 0; │ │ │ │ │ + var timerCallback = function() { │ │ │ │ │ + if (this.timerId == null) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var t = (new Date).getTime() - initialTime; │ │ │ │ │ + var p = -this.deceleration * Math.pow(t, 2) / 2 + v0 * t; │ │ │ │ │ + var x = p * fx; │ │ │ │ │ + var y = p * fy; │ │ │ │ │ + var args = {}; │ │ │ │ │ + args.end = false; │ │ │ │ │ + var v = -this.deceleration * t + v0; │ │ │ │ │ + if (v <= 0) { │ │ │ │ │ + OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + args.end = true │ │ │ │ │ + } │ │ │ │ │ + args.x = x - lastX; │ │ │ │ │ + args.y = y - lastY; │ │ │ │ │ + lastX = x; │ │ │ │ │ + lastY = y; │ │ │ │ │ + callback(args.x, args.y, args.end) │ │ │ │ │ + }; │ │ │ │ │ + this.timerId = OpenLayers.Animation.start(OpenLayers.Function.bind(timerCallback, this)) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Kinetic" │ │ │ │ │ +}); │ │ │ │ │ OpenLayers.Layer = OpenLayers.Class({ │ │ │ │ │ id: null, │ │ │ │ │ name: null, │ │ │ │ │ div: null, │ │ │ │ │ opacity: 1, │ │ │ │ │ alwaysInRange: null, │ │ │ │ │ RESOLUTION_PROPERTIES: ["scales", "resolutions", "maxScale", "minScale", "maxResolution", "minResolution", "numZoomLevels", "maxZoomLevel"], │ │ │ │ │ @@ -5452,1560 +5452,108 @@ │ │ │ │ │ this.tileQueue = null; │ │ │ │ │ this.tileQueueId = null; │ │ │ │ │ this.tileCache = null; │ │ │ │ │ this.tileCacheIndex = null; │ │ │ │ │ this._destroyed = true │ │ │ │ │ } │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.XYZ = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - sphericalMercator: false, │ │ │ │ │ - zoomOffset: 0, │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - if (options && options.sphericalMercator || this.sphericalMercator) { │ │ │ │ │ - options = OpenLayers.Util.extend({ │ │ │ │ │ - projection: "EPSG:900913", │ │ │ │ │ - numZoomLevels: 19 │ │ │ │ │ - }, options) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name || this.name, url || this.url, {}, options]) │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.XYZ(this.name, this.url, this.getOptions()) │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var xyz = this.getXYZ(bounds); │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - var s = "" + xyz.x + xyz.y + xyz.z; │ │ │ │ │ - url = this.selectUrl(s, url) │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.String.format(url, xyz) │ │ │ │ │ - }, │ │ │ │ │ - getXYZ: function(bounds) { │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.maxExtent.left) / (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((this.maxExtent.top - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getServerZoom(); │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - var limit = Math.pow(2, z); │ │ │ │ │ - x = (x % limit + limit) % limit │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - x: x, │ │ │ │ │ - y: y, │ │ │ │ │ - z: z │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (!this.tileOrigin) { │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, this.maxExtent.bottom) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.XYZ" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.OSM = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ - name: "OpenStreetMap", │ │ │ │ │ - url: ["http://a.tile.openstreetmap.org/${z}/${x}/${y}.png", "http://b.tile.openstreetmap.org/${z}/${x}/${y}.png", "http://c.tile.openstreetmap.org/${z}/${x}/${y}.png"], │ │ │ │ │ - attribution: "© <a href='http://www.openstreetmap.org/copyright'>OpenStreetMap</a> contributors", │ │ │ │ │ - sphericalMercator: true, │ │ │ │ │ - wrapDateLine: true, │ │ │ │ │ - tileOptions: null, │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - crossOriginKeyword: "anonymous" │ │ │ │ │ - }, this.options && this.options.tileOptions) │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.OSM(this.name, this.url, this.getOptions()) │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.OSM" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Bing = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ - key: null, │ │ │ │ │ - serverResolutions: [156543.03390625, 78271.516953125, 39135.7584765625, 19567.87923828125, 9783.939619140625, 4891.9698095703125, 2445.9849047851562, 1222.9924523925781, 611.4962261962891, 305.74811309814453, 152.87405654907226, 76.43702827453613, 38.218514137268066, 19.109257068634033, 9.554628534317017, 4.777314267158508, 2.388657133579254, 1.194328566789627, .5971642833948135, .29858214169740677, .14929107084870338, .07464553542435169], │ │ │ │ │ - attributionTemplate: '<span class="olBingAttribution ${type}">' + '<div><a target="_blank" href="http://www.bing.com/maps/">' + '<img src="${logo}" /></a></div>${copyrights}' + '<a style="white-space: nowrap" target="_blank" ' + 'href="http://www.microsoft.com/maps/product/terms.html">' + "Terms of Use</a></span>", │ │ │ │ │ - metadata: null, │ │ │ │ │ - protocolRegex: /^http:/i, │ │ │ │ │ - type: "Road", │ │ │ │ │ - culture: "en-US", │ │ │ │ │ - metadataParams: null, │ │ │ │ │ - tileOptions: null, │ │ │ │ │ - protocol: ~window.location.href.indexOf("http") ? "" : "http:", │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - sphericalMercator: true │ │ │ │ │ - }, options); │ │ │ │ │ - var name = options.name || "Bing " + (options.type || this.type); │ │ │ │ │ - var newArgs = [name, null, options]; │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, newArgs); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - crossOriginKeyword: "anonymous" │ │ │ │ │ - }, this.options.tileOptions); │ │ │ │ │ - this.loadMetadata() │ │ │ │ │ - }, │ │ │ │ │ - loadMetadata: function() { │ │ │ │ │ - this._callbackId = "_callback_" + this.id.replace(/\./g, "_"); │ │ │ │ │ - window[this._callbackId] = OpenLayers.Function.bind(OpenLayers.Layer.Bing.processMetadata, this); │ │ │ │ │ - var params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - key: this.key, │ │ │ │ │ - jsonp: this._callbackId, │ │ │ │ │ - include: "ImageryProviders" │ │ │ │ │ - }, this.metadataParams); │ │ │ │ │ - var url = this.protocol + "//dev.virtualearth.net/REST/v1/Imagery/Metadata/" + this.type + "?" + OpenLayers.Util.getParameterString(params); │ │ │ │ │ - var script = document.createElement("script"); │ │ │ │ │ - script.type = "text/javascript"; │ │ │ │ │ - script.src = url; │ │ │ │ │ - script.id = this._callbackId; │ │ │ │ │ - document.getElementsByTagName("head")[0].appendChild(script) │ │ │ │ │ - }, │ │ │ │ │ - initLayer: function() { │ │ │ │ │ - var res = this.metadata.resourceSets[0].resources[0]; │ │ │ │ │ - var url = res.imageUrl.replace("{quadkey}", "${quadkey}"); │ │ │ │ │ - url = url.replace("{culture}", this.culture); │ │ │ │ │ - url = url.replace(this.protocolRegex, this.protocol); │ │ │ │ │ - this.url = []; │ │ │ │ │ - for (var i = 0; i < res.imageUrlSubdomains.length; ++i) { │ │ │ │ │ - this.url.push(url.replace("{subdomain}", res.imageUrlSubdomains[i])) │ │ │ │ │ - } │ │ │ │ │ - this.addOptions({ │ │ │ │ │ - maxResolution: Math.min(this.serverResolutions[res.zoomMin], this.maxResolution || Number.POSITIVE_INFINITY), │ │ │ │ │ - numZoomLevels: Math.min(res.zoomMax + 1 - res.zoomMin, this.numZoomLevels) │ │ │ │ │ - }, true); │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - this.redraw() │ │ │ │ │ - } │ │ │ │ │ - this.updateAttribution() │ │ │ │ │ - }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - if (!this.url) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var xyz = this.getXYZ(bounds), │ │ │ │ │ - x = xyz.x, │ │ │ │ │ - y = xyz.y, │ │ │ │ │ - z = xyz.z; │ │ │ │ │ - var quadDigits = []; │ │ │ │ │ - for (var i = z; i > 0; --i) { │ │ │ │ │ - var digit = "0"; │ │ │ │ │ - var mask = 1 << i - 1; │ │ │ │ │ - if ((x & mask) != 0) { │ │ │ │ │ - digit++ │ │ │ │ │ - } │ │ │ │ │ - if ((y & mask) != 0) { │ │ │ │ │ - digit++; │ │ │ │ │ - digit++ │ │ │ │ │ - } │ │ │ │ │ - quadDigits.push(digit) │ │ │ │ │ - } │ │ │ │ │ - var quadKey = quadDigits.join(""); │ │ │ │ │ - var url = this.selectUrl("" + x + y + z, this.url); │ │ │ │ │ - return OpenLayers.String.format(url, { │ │ │ │ │ - quadkey: quadKey │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - updateAttribution: function() { │ │ │ │ │ - var metadata = this.metadata; │ │ │ │ │ - if (!metadata.resourceSets || !this.map || !this.map.center) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var res = metadata.resourceSets[0].resources[0]; │ │ │ │ │ - var extent = this.map.getExtent().transform(this.map.getProjectionObject(), new OpenLayers.Projection("EPSG:4326")); │ │ │ │ │ - var providers = res.imageryProviders || [], │ │ │ │ │ - zoom = OpenLayers.Util.indexOf(this.serverResolutions, this.getServerResolution()), │ │ │ │ │ - copyrights = "", │ │ │ │ │ - provider, i, ii, j, jj, bbox, coverage; │ │ │ │ │ - for (i = 0, ii = providers.length; i < ii; ++i) { │ │ │ │ │ - provider = providers[i]; │ │ │ │ │ - for (j = 0, jj = provider.coverageAreas.length; j < jj; ++j) { │ │ │ │ │ - coverage = provider.coverageAreas[j]; │ │ │ │ │ - bbox = OpenLayers.Bounds.fromArray(coverage.bbox, true); │ │ │ │ │ - if (extent.intersectsBounds(bbox) && zoom <= coverage.zoomMax && zoom >= coverage.zoomMin) { │ │ │ │ │ - copyrights += provider.attribution + " " │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var logo = metadata.brandLogoUri.replace(this.protocolRegex, this.protocol); │ │ │ │ │ - this.attribution = OpenLayers.String.format(this.attributionTemplate, { │ │ │ │ │ - type: this.type.toLowerCase(), │ │ │ │ │ - logo: logo, │ │ │ │ │ - copyrights: copyrights │ │ │ │ │ - }); │ │ │ │ │ - this.map && this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "attribution" │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - setMap: function() { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.register("moveend", this, this.updateAttribution) │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Bing(this.options) │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.map && this.map.events.unregister("moveend", this, this.updateAttribution); │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Bing" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Bing.processMetadata = function(metadata) { │ │ │ │ │ - this.metadata = metadata; │ │ │ │ │ - this.initLayer(); │ │ │ │ │ - var script = document.getElementById(this._callbackId); │ │ │ │ │ - script.parentNode.removeChild(script); │ │ │ │ │ - window[this._callbackId] = undefined; │ │ │ │ │ - delete this._callbackId │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer = OpenLayers.Class({ │ │ │ │ │ - container: null, │ │ │ │ │ - root: null, │ │ │ │ │ - extent: null, │ │ │ │ │ - locked: false, │ │ │ │ │ - size: null, │ │ │ │ │ - resolution: null, │ │ │ │ │ - map: null, │ │ │ │ │ - featureDx: 0, │ │ │ │ │ - initialize: function(containerID, options) { │ │ │ │ │ - this.container = OpenLayers.Util.getElement(containerID); │ │ │ │ │ - OpenLayers.Util.extend(this, options) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.container = null; │ │ │ │ │ - this.extent = null; │ │ │ │ │ - this.size = null; │ │ │ │ │ - this.resolution = null; │ │ │ │ │ - this.map = null │ │ │ │ │ - }, │ │ │ │ │ - supported: function() { │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - this.extent = extent.clone(); │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - var ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ - extent = extent.scale(1 / ratio); │ │ │ │ │ - this.extent = extent.wrapDateLine(this.map.getMaxExtent()).scale(ratio) │ │ │ │ │ - } │ │ │ │ │ - if (resolutionChanged) { │ │ │ │ │ - this.resolution = null │ │ │ │ │ - } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - this.size = size.clone(); │ │ │ │ │ - this.resolution = null │ │ │ │ │ - }, │ │ │ │ │ - getResolution: function() { │ │ │ │ │ - this.resolution = this.resolution || this.map.getResolution(); │ │ │ │ │ - return this.resolution │ │ │ │ │ - }, │ │ │ │ │ - drawFeature: function(feature, style) { │ │ │ │ │ - if (style == null) { │ │ │ │ │ - style = feature.style │ │ │ │ │ - } │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - var bounds = feature.geometry.getBounds(); │ │ │ │ │ - if (bounds) { │ │ │ │ │ - var worldBounds; │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - worldBounds = this.map.getMaxExtent() │ │ │ │ │ - } │ │ │ │ │ - if (!bounds.intersectsBounds(this.extent, { │ │ │ │ │ - worldBounds: worldBounds │ │ │ │ │ - })) { │ │ │ │ │ - style = { │ │ │ │ │ - display: "none" │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.calculateFeatureDx(bounds, worldBounds) │ │ │ │ │ - } │ │ │ │ │ - var rendered = this.drawGeometry(feature.geometry, style, feature.id); │ │ │ │ │ - if (style.display != "none" && style.label && rendered !== false) { │ │ │ │ │ - var location = feature.geometry.getCentroid(); │ │ │ │ │ - if (style.labelXOffset || style.labelYOffset) { │ │ │ │ │ - var xOffset = isNaN(style.labelXOffset) ? 0 : style.labelXOffset; │ │ │ │ │ - var yOffset = isNaN(style.labelYOffset) ? 0 : style.labelYOffset; │ │ │ │ │ - var res = this.getResolution(); │ │ │ │ │ - location.move(xOffset * res, yOffset * res) │ │ │ │ │ - } │ │ │ │ │ - this.drawText(feature.id, style, location) │ │ │ │ │ - } else { │ │ │ │ │ - this.removeText(feature.id) │ │ │ │ │ - } │ │ │ │ │ - return rendered │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - calculateFeatureDx: function(bounds, worldBounds) { │ │ │ │ │ - this.featureDx = 0; │ │ │ │ │ - if (worldBounds) { │ │ │ │ │ - var worldWidth = worldBounds.getWidth(), │ │ │ │ │ - rendererCenterX = (this.extent.left + this.extent.right) / 2, │ │ │ │ │ - featureCenterX = (bounds.left + bounds.right) / 2, │ │ │ │ │ - worldsAway = Math.round((featureCenterX - rendererCenterX) / worldWidth); │ │ │ │ │ - this.featureDx = worldsAway * worldWidth │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - drawGeometry: function(geometry, style, featureId) {}, │ │ │ │ │ - drawText: function(featureId, style, location) {}, │ │ │ │ │ - removeText: function(featureId) {}, │ │ │ │ │ - clear: function() {}, │ │ │ │ │ - getFeatureIdFromEvent: function(evt) {}, │ │ │ │ │ - eraseFeatures: function(features) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ - features = [features] │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - this.eraseGeometry(feature.geometry, feature.id); │ │ │ │ │ - this.removeText(feature.id) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - eraseGeometry: function(geometry, featureId) {}, │ │ │ │ │ - moveRoot: function(renderer) {}, │ │ │ │ │ - getRenderLayerId: function() { │ │ │ │ │ - return this.container.id │ │ │ │ │ - }, │ │ │ │ │ - applyDefaultSymbolizer: function(symbolizer) { │ │ │ │ │ - var result = OpenLayers.Util.extend({}, OpenLayers.Renderer.defaultSymbolizer); │ │ │ │ │ - if (symbolizer.stroke === false) { │ │ │ │ │ - delete result.strokeWidth; │ │ │ │ │ - delete result.strokeColor │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.fill === false) { │ │ │ │ │ - delete result.fillColor │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.extend(result, symbolizer); │ │ │ │ │ - return result │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Renderer.defaultSymbolizer = { │ │ │ │ │ - fillColor: "#000000", │ │ │ │ │ - strokeColor: "#000000", │ │ │ │ │ - strokeWidth: 2, │ │ │ │ │ - fillOpacity: 1, │ │ │ │ │ - strokeOpacity: 1, │ │ │ │ │ - pointRadius: 0, │ │ │ │ │ - labelAlign: "cm" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.symbol = { │ │ │ │ │ - star: [350, 75, 379, 161, 469, 161, 397, 215, 423, 301, 350, 250, 277, 301, 303, 215, 231, 161, 321, 161, 350, 75], │ │ │ │ │ - cross: [4, 0, 6, 0, 6, 4, 10, 4, 10, 6, 6, 6, 6, 10, 4, 10, 4, 6, 0, 6, 0, 4, 4, 4, 4, 0], │ │ │ │ │ - x: [0, 0, 25, 0, 50, 35, 75, 0, 100, 0, 65, 50, 100, 100, 75, 100, 50, 65, 25, 100, 0, 100, 35, 50, 0, 0], │ │ │ │ │ - square: [0, 0, 0, 1, 1, 1, 1, 0, 0, 0], │ │ │ │ │ - triangle: [0, 10, 10, 10, 5, 0, 0, 10] │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Feature = OpenLayers.Class({ │ │ │ │ │ - layer: null, │ │ │ │ │ - id: null, │ │ │ │ │ - lonlat: null, │ │ │ │ │ - data: null, │ │ │ │ │ - marker: null, │ │ │ │ │ - popupClass: null, │ │ │ │ │ - popup: null, │ │ │ │ │ - initialize: function(layer, lonlat, data) { │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - this.lonlat = lonlat; │ │ │ │ │ - this.data = data != null ? data : {}; │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.layer != null && this.layer.map != null) { │ │ │ │ │ - if (this.popup != null) { │ │ │ │ │ - this.layer.map.removePopup(this.popup) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.layer != null && this.marker != null) { │ │ │ │ │ - this.layer.removeMarker(this.marker) │ │ │ │ │ - } │ │ │ │ │ - this.layer = null; │ │ │ │ │ - this.id = null; │ │ │ │ │ - this.lonlat = null; │ │ │ │ │ - this.data = null; │ │ │ │ │ - if (this.marker != null) { │ │ │ │ │ - this.destroyMarker(this.marker); │ │ │ │ │ - this.marker = null │ │ │ │ │ - } │ │ │ │ │ - if (this.popup != null) { │ │ │ │ │ - this.destroyPopup(this.popup); │ │ │ │ │ - this.popup = null │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - onScreen: function() { │ │ │ │ │ - var onScreen = false; │ │ │ │ │ - if (this.layer != null && this.layer.map != null) { │ │ │ │ │ - var screenBounds = this.layer.map.getExtent(); │ │ │ │ │ - onScreen = screenBounds.containsLonLat(this.lonlat) │ │ │ │ │ - } │ │ │ │ │ - return onScreen │ │ │ │ │ - }, │ │ │ │ │ - createMarker: function() { │ │ │ │ │ - if (this.lonlat != null) { │ │ │ │ │ - this.marker = new OpenLayers.Marker(this.lonlat, this.data.icon) │ │ │ │ │ - } │ │ │ │ │ - return this.marker │ │ │ │ │ - }, │ │ │ │ │ - destroyMarker: function() { │ │ │ │ │ - this.marker.destroy() │ │ │ │ │ - }, │ │ │ │ │ - createPopup: function(closeBox) { │ │ │ │ │ - if (this.lonlat != null) { │ │ │ │ │ - if (!this.popup) { │ │ │ │ │ - var anchor = this.marker ? this.marker.icon : null; │ │ │ │ │ - var popupClass = this.popupClass ? this.popupClass : OpenLayers.Popup.Anchored; │ │ │ │ │ - this.popup = new popupClass(this.id + "_popup", this.lonlat, this.data.popupSize, this.data.popupContentHTML, anchor, closeBox) │ │ │ │ │ - } │ │ │ │ │ - if (this.data.overflow != null) { │ │ │ │ │ - this.popup.contentDiv.style.overflow = this.data.overflow │ │ │ │ │ - } │ │ │ │ │ - this.popup.feature = this │ │ │ │ │ - } │ │ │ │ │ - return this.popup │ │ │ │ │ - }, │ │ │ │ │ - destroyPopup: function() { │ │ │ │ │ - if (this.popup) { │ │ │ │ │ - this.popup.feature = null; │ │ │ │ │ - this.popup.destroy(); │ │ │ │ │ - this.popup = null │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Feature" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.State = { │ │ │ │ │ - UNKNOWN: "Unknown", │ │ │ │ │ - INSERT: "Insert", │ │ │ │ │ - UPDATE: "Update", │ │ │ │ │ - DELETE: "Delete" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Feature.Vector = OpenLayers.Class(OpenLayers.Feature, { │ │ │ │ │ - fid: null, │ │ │ │ │ - geometry: null, │ │ │ │ │ - attributes: null, │ │ │ │ │ - bounds: null, │ │ │ │ │ - state: null, │ │ │ │ │ - style: null, │ │ │ │ │ - url: null, │ │ │ │ │ - renderIntent: "default", │ │ │ │ │ - modified: null, │ │ │ │ │ - initialize: function(geometry, attributes, style) { │ │ │ │ │ - OpenLayers.Feature.prototype.initialize.apply(this, [null, null, attributes]); │ │ │ │ │ - this.lonlat = null; │ │ │ │ │ - this.geometry = geometry ? geometry : null; │ │ │ │ │ - this.state = null; │ │ │ │ │ - this.attributes = {}; │ │ │ │ │ - if (attributes) { │ │ │ │ │ - this.attributes = OpenLayers.Util.extend(this.attributes, attributes) │ │ │ │ │ - } │ │ │ │ │ - this.style = style ? style : null │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.layer) { │ │ │ │ │ - this.layer.removeFeatures(this); │ │ │ │ │ - this.layer = null │ │ │ │ │ - } │ │ │ │ │ - this.geometry = null; │ │ │ │ │ - this.modified = null; │ │ │ │ │ - OpenLayers.Feature.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - return new OpenLayers.Feature.Vector(this.geometry ? this.geometry.clone() : null, this.attributes, this.style) │ │ │ │ │ - }, │ │ │ │ │ - onScreen: function(boundsOnly) { │ │ │ │ │ - var onScreen = false; │ │ │ │ │ - if (this.layer && this.layer.map) { │ │ │ │ │ - var screenBounds = this.layer.map.getExtent(); │ │ │ │ │ - if (boundsOnly) { │ │ │ │ │ - var featureBounds = this.geometry.getBounds(); │ │ │ │ │ - onScreen = screenBounds.intersectsBounds(featureBounds) │ │ │ │ │ - } else { │ │ │ │ │ - var screenPoly = screenBounds.toGeometry(); │ │ │ │ │ - onScreen = screenPoly.intersects(this.geometry) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return onScreen │ │ │ │ │ - }, │ │ │ │ │ - getVisibility: function() { │ │ │ │ │ - return !(this.style && this.style.display == "none" || !this.layer || this.layer && this.layer.styleMap && this.layer.styleMap.createSymbolizer(this, this.renderIntent).display == "none" || this.layer && !this.layer.getVisibility()) │ │ │ │ │ - }, │ │ │ │ │ - createMarker: function() { │ │ │ │ │ - return null │ │ │ │ │ - }, │ │ │ │ │ - destroyMarker: function() {}, │ │ │ │ │ - createPopup: function() { │ │ │ │ │ - return null │ │ │ │ │ - }, │ │ │ │ │ - atPoint: function(lonlat, toleranceLon, toleranceLat) { │ │ │ │ │ - var atPoint = false; │ │ │ │ │ - if (this.geometry) { │ │ │ │ │ - atPoint = this.geometry.atPoint(lonlat, toleranceLon, toleranceLat) │ │ │ │ │ - } │ │ │ │ │ - return atPoint │ │ │ │ │ - }, │ │ │ │ │ - destroyPopup: function() {}, │ │ │ │ │ - move: function(location) { │ │ │ │ │ - if (!this.layer || !this.geometry.move) { │ │ │ │ │ - return undefined │ │ │ │ │ - } │ │ │ │ │ - var pixel; │ │ │ │ │ - if (location.CLASS_NAME == "OpenLayers.LonLat") { │ │ │ │ │ - pixel = this.layer.getViewPortPxFromLonLat(location) │ │ │ │ │ - } else { │ │ │ │ │ - pixel = location │ │ │ │ │ - } │ │ │ │ │ - var lastPixel = this.layer.getViewPortPxFromLonLat(this.geometry.getBounds().getCenterLonLat()); │ │ │ │ │ - var res = this.layer.map.getResolution(); │ │ │ │ │ - this.geometry.move(res * (pixel.x - lastPixel.x), res * (lastPixel.y - pixel.y)); │ │ │ │ │ - this.layer.drawFeature(this); │ │ │ │ │ - return lastPixel │ │ │ │ │ - }, │ │ │ │ │ - toState: function(state) { │ │ │ │ │ - if (state == OpenLayers.State.UPDATE) { │ │ │ │ │ - switch (this.state) { │ │ │ │ │ - case OpenLayers.State.UNKNOWN: │ │ │ │ │ - case OpenLayers.State.DELETE: │ │ │ │ │ - this.state = state; │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.State.UPDATE: │ │ │ │ │ - case OpenLayers.State.INSERT: │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } else if (state == OpenLayers.State.INSERT) { │ │ │ │ │ - switch (this.state) { │ │ │ │ │ - case OpenLayers.State.UNKNOWN: │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - this.state = state; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } else if (state == OpenLayers.State.DELETE) { │ │ │ │ │ - switch (this.state) { │ │ │ │ │ - case OpenLayers.State.INSERT: │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.State.DELETE: │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.State.UNKNOWN: │ │ │ │ │ - case OpenLayers.State.UPDATE: │ │ │ │ │ - this.state = state; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } else if (state == OpenLayers.State.UNKNOWN) { │ │ │ │ │ - this.state = state │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Feature.Vector" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Feature.Vector.style = { │ │ │ │ │ - default: { │ │ │ │ │ - fillColor: "#ee9900", │ │ │ │ │ - fillOpacity: .4, │ │ │ │ │ - hoverFillColor: "white", │ │ │ │ │ - hoverFillOpacity: .8, │ │ │ │ │ - strokeColor: "#ee9900", │ │ │ │ │ - strokeOpacity: 1, │ │ │ │ │ - strokeWidth: 1, │ │ │ │ │ - strokeLinecap: "round", │ │ │ │ │ - strokeDashstyle: "solid", │ │ │ │ │ - hoverStrokeColor: "red", │ │ │ │ │ - hoverStrokeOpacity: 1, │ │ │ │ │ - hoverStrokeWidth: .2, │ │ │ │ │ - pointRadius: 6, │ │ │ │ │ - hoverPointRadius: 1, │ │ │ │ │ - hoverPointUnit: "%", │ │ │ │ │ - pointerEvents: "visiblePainted", │ │ │ │ │ - cursor: "inherit", │ │ │ │ │ - fontColor: "#000000", │ │ │ │ │ - labelAlign: "cm", │ │ │ │ │ - labelOutlineColor: "white", │ │ │ │ │ - labelOutlineWidth: 3 │ │ │ │ │ - }, │ │ │ │ │ - select: { │ │ │ │ │ - fillColor: "blue", │ │ │ │ │ - fillOpacity: .4, │ │ │ │ │ - hoverFillColor: "white", │ │ │ │ │ - hoverFillOpacity: .8, │ │ │ │ │ - strokeColor: "blue", │ │ │ │ │ - strokeOpacity: 1, │ │ │ │ │ - strokeWidth: 2, │ │ │ │ │ - strokeLinecap: "round", │ │ │ │ │ - strokeDashstyle: "solid", │ │ │ │ │ - hoverStrokeColor: "red", │ │ │ │ │ - hoverStrokeOpacity: 1, │ │ │ │ │ - hoverStrokeWidth: .2, │ │ │ │ │ - pointRadius: 6, │ │ │ │ │ - hoverPointRadius: 1, │ │ │ │ │ - hoverPointUnit: "%", │ │ │ │ │ - pointerEvents: "visiblePainted", │ │ │ │ │ - cursor: "pointer", │ │ │ │ │ - fontColor: "#000000", │ │ │ │ │ - labelAlign: "cm", │ │ │ │ │ - labelOutlineColor: "white", │ │ │ │ │ - labelOutlineWidth: 3 │ │ │ │ │ - }, │ │ │ │ │ - temporary: { │ │ │ │ │ - fillColor: "#66cccc", │ │ │ │ │ - fillOpacity: .2, │ │ │ │ │ - hoverFillColor: "white", │ │ │ │ │ - hoverFillOpacity: .8, │ │ │ │ │ - strokeColor: "#66cccc", │ │ │ │ │ - strokeOpacity: 1, │ │ │ │ │ - strokeLinecap: "round", │ │ │ │ │ - strokeWidth: 2, │ │ │ │ │ - strokeDashstyle: "solid", │ │ │ │ │ - hoverStrokeColor: "red", │ │ │ │ │ - hoverStrokeOpacity: 1, │ │ │ │ │ - hoverStrokeWidth: .2, │ │ │ │ │ - pointRadius: 6, │ │ │ │ │ - hoverPointRadius: 1, │ │ │ │ │ - hoverPointUnit: "%", │ │ │ │ │ - pointerEvents: "visiblePainted", │ │ │ │ │ - cursor: "inherit", │ │ │ │ │ - fontColor: "#000000", │ │ │ │ │ - labelAlign: "cm", │ │ │ │ │ - labelOutlineColor: "white", │ │ │ │ │ - labelOutlineWidth: 3 │ │ │ │ │ - }, │ │ │ │ │ - delete: { │ │ │ │ │ - display: "none" │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Style = OpenLayers.Class({ │ │ │ │ │ +OpenLayers.Handler = OpenLayers.Class({ │ │ │ │ │ id: null, │ │ │ │ │ - name: null, │ │ │ │ │ - title: null, │ │ │ │ │ - description: null, │ │ │ │ │ - layerName: null, │ │ │ │ │ - isDefault: false, │ │ │ │ │ - rules: null, │ │ │ │ │ - context: null, │ │ │ │ │ - defaultStyle: null, │ │ │ │ │ - defaultsPerSymbolizer: false, │ │ │ │ │ - propertyStyles: null, │ │ │ │ │ - initialize: function(style, options) { │ │ │ │ │ + control: null, │ │ │ │ │ + map: null, │ │ │ │ │ + keyMask: null, │ │ │ │ │ + active: false, │ │ │ │ │ + evt: null, │ │ │ │ │ + touch: false, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.rules = []; │ │ │ │ │ - if (options && options.rules) { │ │ │ │ │ - this.addRules(options.rules) │ │ │ │ │ + this.control = control; │ │ │ │ │ + this.callbacks = callbacks; │ │ │ │ │ + var map = this.map || control.map; │ │ │ │ │ + if (map) { │ │ │ │ │ + this.setMap(map) │ │ │ │ │ } │ │ │ │ │ - this.setDefaultStyle(style || OpenLayers.Feature.Vector.style["default"]); │ │ │ │ │ this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i = 0, len = this.rules.length; i < len; i++) { │ │ │ │ │ - this.rules[i].destroy(); │ │ │ │ │ - this.rules[i] = null │ │ │ │ │ - } │ │ │ │ │ - this.rules = null; │ │ │ │ │ - this.defaultStyle = null │ │ │ │ │ - }, │ │ │ │ │ - createSymbolizer: function(feature) { │ │ │ │ │ - var style = this.defaultsPerSymbolizer ? {} : this.createLiterals(OpenLayers.Util.extend({}, this.defaultStyle), feature); │ │ │ │ │ - var rules = this.rules; │ │ │ │ │ - var rule, context; │ │ │ │ │ - var elseRules = []; │ │ │ │ │ - var appliedRules = false; │ │ │ │ │ - for (var i = 0, len = rules.length; i < len; i++) { │ │ │ │ │ - rule = rules[i]; │ │ │ │ │ - var applies = rule.evaluate(feature); │ │ │ │ │ - if (applies) { │ │ │ │ │ - if (rule instanceof OpenLayers.Rule && rule.elseFilter) { │ │ │ │ │ - elseRules.push(rule) │ │ │ │ │ - } else { │ │ │ │ │ - appliedRules = true; │ │ │ │ │ - this.applySymbolizer(rule, style, feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (appliedRules == false && elseRules.length > 0) { │ │ │ │ │ - appliedRules = true; │ │ │ │ │ - for (var i = 0, len = elseRules.length; i < len; i++) { │ │ │ │ │ - this.applySymbolizer(elseRules[i], style, feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (rules.length > 0 && appliedRules == false) { │ │ │ │ │ - style.display = "none" │ │ │ │ │ - } │ │ │ │ │ - if (style.label != null && typeof style.label !== "string") { │ │ │ │ │ - style.label = String(style.label) │ │ │ │ │ - } │ │ │ │ │ - return style │ │ │ │ │ - }, │ │ │ │ │ - applySymbolizer: function(rule, style, feature) { │ │ │ │ │ - var symbolizerPrefix = feature.geometry ? this.getSymbolizerPrefix(feature.geometry) : OpenLayers.Style.SYMBOLIZER_PREFIXES[0]; │ │ │ │ │ - var symbolizer = rule.symbolizer[symbolizerPrefix] || rule.symbolizer; │ │ │ │ │ - if (this.defaultsPerSymbolizer === true) { │ │ │ │ │ - var defaults = this.defaultStyle; │ │ │ │ │ - OpenLayers.Util.applyDefaults(symbolizer, { │ │ │ │ │ - pointRadius: defaults.pointRadius │ │ │ │ │ - }); │ │ │ │ │ - if (symbolizer.stroke === true || symbolizer.graphic === true) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(symbolizer, { │ │ │ │ │ - strokeWidth: defaults.strokeWidth, │ │ │ │ │ - strokeColor: defaults.strokeColor, │ │ │ │ │ - strokeOpacity: defaults.strokeOpacity, │ │ │ │ │ - strokeDashstyle: defaults.strokeDashstyle, │ │ │ │ │ - strokeLinecap: defaults.strokeLinecap │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.fill === true || symbolizer.graphic === true) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(symbolizer, { │ │ │ │ │ - fillColor: defaults.fillColor, │ │ │ │ │ - fillOpacity: defaults.fillOpacity │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.graphic === true) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(symbolizer, { │ │ │ │ │ - pointRadius: this.defaultStyle.pointRadius, │ │ │ │ │ - externalGraphic: this.defaultStyle.externalGraphic, │ │ │ │ │ - graphicName: this.defaultStyle.graphicName, │ │ │ │ │ - graphicOpacity: this.defaultStyle.graphicOpacity, │ │ │ │ │ - graphicWidth: this.defaultStyle.graphicWidth, │ │ │ │ │ - graphicHeight: this.defaultStyle.graphicHeight, │ │ │ │ │ - graphicXOffset: this.defaultStyle.graphicXOffset, │ │ │ │ │ - graphicYOffset: this.defaultStyle.graphicYOffset │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return this.createLiterals(OpenLayers.Util.extend(style, symbolizer), feature) │ │ │ │ │ - }, │ │ │ │ │ - createLiterals: function(style, feature) { │ │ │ │ │ - var context = OpenLayers.Util.extend({}, feature.attributes || feature.data); │ │ │ │ │ - OpenLayers.Util.extend(context, this.context); │ │ │ │ │ - for (var i in this.propertyStyles) { │ │ │ │ │ - style[i] = OpenLayers.Style.createLiteral(style[i], context, feature, i) │ │ │ │ │ - } │ │ │ │ │ - return style │ │ │ │ │ - }, │ │ │ │ │ - findPropertyStyles: function() { │ │ │ │ │ - var propertyStyles = {}; │ │ │ │ │ - var style = this.defaultStyle; │ │ │ │ │ - this.addPropertyStyles(propertyStyles, style); │ │ │ │ │ - var rules = this.rules; │ │ │ │ │ - var symbolizer, value; │ │ │ │ │ - for (var i = 0, len = rules.length; i < len; i++) { │ │ │ │ │ - symbolizer = rules[i].symbolizer; │ │ │ │ │ - for (var key in symbolizer) { │ │ │ │ │ - value = symbolizer[key]; │ │ │ │ │ - if (typeof value == "object") { │ │ │ │ │ - this.addPropertyStyles(propertyStyles, value) │ │ │ │ │ - } else { │ │ │ │ │ - this.addPropertyStyles(propertyStyles, symbolizer); │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return propertyStyles │ │ │ │ │ - }, │ │ │ │ │ - addPropertyStyles: function(propertyStyles, symbolizer) { │ │ │ │ │ - var property; │ │ │ │ │ - for (var key in symbolizer) { │ │ │ │ │ - property = symbolizer[key]; │ │ │ │ │ - if (typeof property == "string" && property.match(/\$\{\w+\}/)) { │ │ │ │ │ - propertyStyles[key] = true │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return propertyStyles │ │ │ │ │ - }, │ │ │ │ │ - addRules: function(rules) { │ │ │ │ │ - Array.prototype.push.apply(this.rules, rules); │ │ │ │ │ - this.propertyStyles = this.findPropertyStyles() │ │ │ │ │ - }, │ │ │ │ │ - setDefaultStyle: function(style) { │ │ │ │ │ - this.defaultStyle = style; │ │ │ │ │ - this.propertyStyles = this.findPropertyStyles() │ │ │ │ │ - }, │ │ │ │ │ - getSymbolizerPrefix: function(geometry) { │ │ │ │ │ - var prefixes = OpenLayers.Style.SYMBOLIZER_PREFIXES; │ │ │ │ │ - for (var i = 0, len = prefixes.length; i < len; i++) { │ │ │ │ │ - if (geometry.CLASS_NAME.indexOf(prefixes[i]) != -1) { │ │ │ │ │ - return prefixes[i] │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - var options = OpenLayers.Util.extend({}, this); │ │ │ │ │ - if (this.rules) { │ │ │ │ │ - options.rules = []; │ │ │ │ │ - for (var i = 0, len = this.rules.length; i < len; ++i) { │ │ │ │ │ - options.rules.push(this.rules[i].clone()) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - options.context = this.context && OpenLayers.Util.extend({}, this.context); │ │ │ │ │ - var defaultStyle = OpenLayers.Util.extend({}, this.defaultStyle); │ │ │ │ │ - return new OpenLayers.Style(defaultStyle, options) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Style" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Style.createLiteral = function(value, context, feature, property) { │ │ │ │ │ - if (typeof value == "string" && value.indexOf("${") != -1) { │ │ │ │ │ - value = OpenLayers.String.format(value, context, [feature, property]); │ │ │ │ │ - value = isNaN(value) || !value ? value : parseFloat(value) │ │ │ │ │ - } │ │ │ │ │ - return value │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Style.SYMBOLIZER_PREFIXES = ["Point", "Line", "Polygon", "Text", "Raster"]; │ │ │ │ │ -OpenLayers.StyleMap = OpenLayers.Class({ │ │ │ │ │ - styles: null, │ │ │ │ │ - extendDefault: true, │ │ │ │ │ - initialize: function(style, options) { │ │ │ │ │ - this.styles = { │ │ │ │ │ - default: new OpenLayers.Style(OpenLayers.Feature.Vector.style["default"]), │ │ │ │ │ - select: new OpenLayers.Style(OpenLayers.Feature.Vector.style["select"]), │ │ │ │ │ - temporary: new OpenLayers.Style(OpenLayers.Feature.Vector.style["temporary"]), │ │ │ │ │ - delete: new OpenLayers.Style(OpenLayers.Feature.Vector.style["delete"]) │ │ │ │ │ - }; │ │ │ │ │ - if (style instanceof OpenLayers.Style) { │ │ │ │ │ - this.styles["default"] = style; │ │ │ │ │ - this.styles["select"] = style; │ │ │ │ │ - this.styles["temporary"] = style; │ │ │ │ │ - this.styles["delete"] = style │ │ │ │ │ - } else if (typeof style == "object") { │ │ │ │ │ - for (var key in style) { │ │ │ │ │ - if (style[key] instanceof OpenLayers.Style) { │ │ │ │ │ - this.styles[key] = style[key] │ │ │ │ │ - } else if (typeof style[key] == "object") { │ │ │ │ │ - this.styles[key] = new OpenLayers.Style(style[key]) │ │ │ │ │ - } else { │ │ │ │ │ - this.styles["default"] = new OpenLayers.Style(style); │ │ │ │ │ - this.styles["select"] = new OpenLayers.Style(style); │ │ │ │ │ - this.styles["temporary"] = new OpenLayers.Style(style); │ │ │ │ │ - this.styles["delete"] = new OpenLayers.Style(style); │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.extend(this, options) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var key in this.styles) { │ │ │ │ │ - this.styles[key].destroy() │ │ │ │ │ - } │ │ │ │ │ - this.styles = null │ │ │ │ │ - }, │ │ │ │ │ - createSymbolizer: function(feature, intent) { │ │ │ │ │ - if (!feature) { │ │ │ │ │ - feature = new OpenLayers.Feature.Vector │ │ │ │ │ - } │ │ │ │ │ - if (!this.styles[intent]) { │ │ │ │ │ - intent = "default" │ │ │ │ │ - } │ │ │ │ │ - feature.renderIntent = intent; │ │ │ │ │ - var defaultSymbolizer = {}; │ │ │ │ │ - if (this.extendDefault && intent != "default") { │ │ │ │ │ - defaultSymbolizer = this.styles["default"].createSymbolizer(feature) │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Util.extend(defaultSymbolizer, this.styles[intent].createSymbolizer(feature)) │ │ │ │ │ - }, │ │ │ │ │ - addUniqueValueRules: function(renderIntent, property, symbolizers, context) { │ │ │ │ │ - var rules = []; │ │ │ │ │ - for (var value in symbolizers) { │ │ │ │ │ - rules.push(new OpenLayers.Rule({ │ │ │ │ │ - symbolizer: symbolizers[value], │ │ │ │ │ - context: context, │ │ │ │ │ - filter: new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.EQUAL_TO, │ │ │ │ │ - property: property, │ │ │ │ │ - value: value │ │ │ │ │ - }) │ │ │ │ │ - })) │ │ │ │ │ - } │ │ │ │ │ - this.styles[renderIntent].addRules(rules) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.StyleMap" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Vector = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ - isBaseLayer: false, │ │ │ │ │ - isFixed: false, │ │ │ │ │ - features: null, │ │ │ │ │ - filter: null, │ │ │ │ │ - selectedFeatures: null, │ │ │ │ │ - unrenderedFeatures: null, │ │ │ │ │ - reportError: true, │ │ │ │ │ - style: null, │ │ │ │ │ - styleMap: null, │ │ │ │ │ - strategies: null, │ │ │ │ │ - protocol: null, │ │ │ │ │ - renderers: ["SVG", "VML", "Canvas"], │ │ │ │ │ - renderer: null, │ │ │ │ │ - rendererOptions: null, │ │ │ │ │ - geometryType: null, │ │ │ │ │ - drawn: false, │ │ │ │ │ - ratio: 1, │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ - this.assignRenderer() │ │ │ │ │ - } │ │ │ │ │ - if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ - this.renderer = null; │ │ │ │ │ - this.displayError() │ │ │ │ │ - } │ │ │ │ │ - if (!this.styleMap) { │ │ │ │ │ - this.styleMap = new OpenLayers.StyleMap │ │ │ │ │ - } │ │ │ │ │ - this.features = []; │ │ │ │ │ - this.selectedFeatures = []; │ │ │ │ │ - this.unrenderedFeatures = {}; │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - for (var i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - this.strategies[i].setLayer(this) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoDestroy) { │ │ │ │ │ - strategy.destroy() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.strategies = null │ │ │ │ │ - } │ │ │ │ │ - if (this.protocol) { │ │ │ │ │ - if (this.protocol.autoDestroy) { │ │ │ │ │ - this.protocol.destroy() │ │ │ │ │ - } │ │ │ │ │ - this.protocol = null │ │ │ │ │ - } │ │ │ │ │ - this.destroyFeatures(); │ │ │ │ │ - this.features = null; │ │ │ │ │ - this.selectedFeatures = null; │ │ │ │ │ - this.unrenderedFeatures = null; │ │ │ │ │ - if (this.renderer) { │ │ │ │ │ - this.renderer.destroy() │ │ │ │ │ - } │ │ │ │ │ - this.renderer = null; │ │ │ │ │ - this.geometryType = null; │ │ │ │ │ - this.drawn = null; │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Vector(this.name, this.getOptions()) │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ - var features = this.features; │ │ │ │ │ - var len = features.length; │ │ │ │ │ - var clonedFeatures = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - clonedFeatures[i] = features[i].clone() │ │ │ │ │ - } │ │ │ │ │ - obj.features = clonedFeatures; │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - refresh: function(obj) { │ │ │ │ │ - if (this.calculateInRange() && this.visibility) { │ │ │ │ │ - this.events.triggerEvent("refresh", obj) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - assignRenderer: function() { │ │ │ │ │ - for (var i = 0, len = this.renderers.length; i < len; i++) { │ │ │ │ │ - var rendererClass = this.renderers[i]; │ │ │ │ │ - var renderer = typeof rendererClass == "function" ? rendererClass : OpenLayers.Renderer[rendererClass]; │ │ │ │ │ - if (renderer && renderer.prototype.supported()) { │ │ │ │ │ - this.renderer = new renderer(this.div, this.rendererOptions); │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - displayError: function() { │ │ │ │ │ - if (this.reportError) { │ │ │ │ │ - OpenLayers.Console.userError(OpenLayers.i18n("browserNotSupported", { │ │ │ │ │ - renderers: this.renderers.join("\n") │ │ │ │ │ - })) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (!this.renderer) { │ │ │ │ │ - this.map.removeLayer(this) │ │ │ │ │ - } else { │ │ │ │ │ - this.renderer.map = this.map; │ │ │ │ │ - var newSize = this.map.getSize(); │ │ │ │ │ - newSize.w = newSize.w * this.ratio; │ │ │ │ │ - newSize.h = newSize.h * this.ratio; │ │ │ │ │ - this.renderer.setSize(newSize) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - afterAdd: function() { │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoActivate) { │ │ │ │ │ - strategy.activate() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - this.drawn = false; │ │ │ │ │ - if (this.strategies) { │ │ │ │ │ - var strategy, i, len; │ │ │ │ │ - for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ - strategy = this.strategies[i]; │ │ │ │ │ - if (strategy.autoActivate) { │ │ │ │ │ - strategy.deactivate() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - OpenLayers.Layer.prototype.onMapResize.apply(this, arguments); │ │ │ │ │ - var newSize = this.map.getSize(); │ │ │ │ │ - newSize.w = newSize.w * this.ratio; │ │ │ │ │ - newSize.h = newSize.h * this.ratio; │ │ │ │ │ - this.renderer.setSize(newSize) │ │ │ │ │ - }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - var coordSysUnchanged = true; │ │ │ │ │ - if (!dragging) { │ │ │ │ │ - this.renderer.root.style.visibility = "hidden"; │ │ │ │ │ - var viewSize = this.map.getSize(), │ │ │ │ │ - viewWidth = viewSize.w, │ │ │ │ │ - viewHeight = viewSize.h, │ │ │ │ │ - offsetLeft = viewWidth / 2 * this.ratio - viewWidth / 2, │ │ │ │ │ - offsetTop = viewHeight / 2 * this.ratio - viewHeight / 2; │ │ │ │ │ - offsetLeft += this.map.layerContainerOriginPx.x; │ │ │ │ │ - offsetLeft = -Math.round(offsetLeft); │ │ │ │ │ - offsetTop += this.map.layerContainerOriginPx.y; │ │ │ │ │ - offsetTop = -Math.round(offsetTop); │ │ │ │ │ - this.div.style.left = offsetLeft + "px"; │ │ │ │ │ - this.div.style.top = offsetTop + "px"; │ │ │ │ │ - var extent = this.map.getExtent().scale(this.ratio); │ │ │ │ │ - coordSysUnchanged = this.renderer.setExtent(extent, zoomChanged); │ │ │ │ │ - this.renderer.root.style.visibility = "visible"; │ │ │ │ │ - if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ - this.div.scrollLeft = this.div.scrollLeft │ │ │ │ │ - } │ │ │ │ │ - if (!zoomChanged && coordSysUnchanged) { │ │ │ │ │ - for (var i in this.unrenderedFeatures) { │ │ │ │ │ - var feature = this.unrenderedFeatures[i]; │ │ │ │ │ - this.drawFeature(feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!this.drawn || zoomChanged || !coordSysUnchanged) { │ │ │ │ │ - this.drawn = true; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0, len = this.features.length; i < len; i++) { │ │ │ │ │ - this.renderer.locked = i !== len - 1; │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - this.drawFeature(feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - display: function(display) { │ │ │ │ │ - OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ - var currentDisplay = this.div.style.display; │ │ │ │ │ - if (currentDisplay != this.renderer.root.style.display) { │ │ │ │ │ - this.renderer.root.style.display = currentDisplay │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - addFeatures: function(features, options) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ - features = [features] │ │ │ │ │ - } │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ - if (notify) { │ │ │ │ │ - var event = { │ │ │ │ │ - features: features │ │ │ │ │ - }; │ │ │ │ │ - var ret = this.events.triggerEvent("beforefeaturesadded", event); │ │ │ │ │ - if (ret === false) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - features = event.features │ │ │ │ │ - } │ │ │ │ │ - var featuresAdded = []; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - if (i != features.length - 1) { │ │ │ │ │ - this.renderer.locked = true │ │ │ │ │ - } else { │ │ │ │ │ - this.renderer.locked = false │ │ │ │ │ - } │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - if (this.geometryType && !(feature.geometry instanceof this.geometryType)) { │ │ │ │ │ - throw new TypeError("addFeatures: component should be an " + this.geometryType.prototype.CLASS_NAME) │ │ │ │ │ - } │ │ │ │ │ - feature.layer = this; │ │ │ │ │ - if (!feature.style && this.style) { │ │ │ │ │ - feature.style = OpenLayers.Util.extend({}, this.style) │ │ │ │ │ - } │ │ │ │ │ - if (notify) { │ │ │ │ │ - if (this.events.triggerEvent("beforefeatureadded", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) === false) { │ │ │ │ │ - continue │ │ │ │ │ - } │ │ │ │ │ - this.preFeatureInsert(feature) │ │ │ │ │ - } │ │ │ │ │ - featuresAdded.push(feature); │ │ │ │ │ - this.features.push(feature); │ │ │ │ │ - this.drawFeature(feature); │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureadded", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - this.onFeatureInsert(feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresadded", { │ │ │ │ │ - features: featuresAdded │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - removeFeatures: function(features, options) { │ │ │ │ │ - if (!features || features.length === 0) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (features === this.features) { │ │ │ │ │ - return this.removeAllFeatures(options) │ │ │ │ │ - } │ │ │ │ │ - if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ - features = [features] │ │ │ │ │ - } │ │ │ │ │ - if (features === this.selectedFeatures) { │ │ │ │ │ - features = features.slice() │ │ │ │ │ - } │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeaturesremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - if (i != 0 && features[i - 1].geometry) { │ │ │ │ │ - this.renderer.locked = true │ │ │ │ │ - } else { │ │ │ │ │ - this.renderer.locked = false │ │ │ │ │ - } │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.features = OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ - feature.layer = null; │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - this.renderer.eraseFeatures(feature) │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.selectedFeatures, feature) != -1) { │ │ │ │ │ - OpenLayers.Util.removeItem(this.selectedFeatures, feature) │ │ │ │ │ - } │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - removeAllFeatures: function(options) { │ │ │ │ │ - var notify = !options || !options.silent; │ │ │ │ │ - var features = this.features; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeaturesremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - feature.layer = null; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featureremoved", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.renderer.clear(); │ │ │ │ │ - this.features = []; │ │ │ │ │ - this.unrenderedFeatures = {}; │ │ │ │ │ - this.selectedFeatures = []; │ │ │ │ │ - if (notify) { │ │ │ │ │ - this.events.triggerEvent("featuresremoved", { │ │ │ │ │ - features: features │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroyFeatures: function(features, options) { │ │ │ │ │ - var all = features == undefined; │ │ │ │ │ - if (all) { │ │ │ │ │ - features = this.features │ │ │ │ │ - } │ │ │ │ │ - if (features) { │ │ │ │ │ - this.removeFeatures(features, options); │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - features[i].destroy() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - drawFeature: function(feature, style) { │ │ │ │ │ - if (!this.drawn) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (typeof style != "object") { │ │ │ │ │ - if (!style && feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ - style = "delete" │ │ │ │ │ - } │ │ │ │ │ - var renderIntent = style || feature.renderIntent; │ │ │ │ │ - style = feature.style || this.style; │ │ │ │ │ - if (!style) { │ │ │ │ │ - style = this.styleMap.createSymbolizer(feature, renderIntent) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var drawn = this.renderer.drawFeature(feature, style); │ │ │ │ │ - if (drawn === false || drawn === null) { │ │ │ │ │ - this.unrenderedFeatures[feature.id] = feature │ │ │ │ │ - } else { │ │ │ │ │ - delete this.unrenderedFeatures[feature.id] │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - eraseFeatures: function(features) { │ │ │ │ │ - this.renderer.eraseFeatures(features) │ │ │ │ │ - }, │ │ │ │ │ - getFeatureFromEvent: function(evt) { │ │ │ │ │ - if (!this.renderer) { │ │ │ │ │ - throw new Error("getFeatureFromEvent called on layer with no " + "renderer. This usually means you destroyed a " + "layer, but not some handler which is associated " + "with it.") │ │ │ │ │ - } │ │ │ │ │ - var feature = null; │ │ │ │ │ - var featureId = this.renderer.getFeatureIdFromEvent(evt); │ │ │ │ │ - if (featureId) { │ │ │ │ │ - if (typeof featureId === "string") { │ │ │ │ │ - feature = this.getFeatureById(featureId) │ │ │ │ │ - } else { │ │ │ │ │ - feature = featureId │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return feature │ │ │ │ │ + this.map = map │ │ │ │ │ }, │ │ │ │ │ - getFeatureBy: function(property, value) { │ │ │ │ │ - var feature = null; │ │ │ │ │ - for (var i = 0, len = this.features.length; i < len; ++i) { │ │ │ │ │ - if (this.features[i][property] == value) { │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ + checkModifiers: function(evt) { │ │ │ │ │ + if (this.keyMask == null) { │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ - return feature │ │ │ │ │ - }, │ │ │ │ │ - getFeatureById: function(featureId) { │ │ │ │ │ - return this.getFeatureBy("id", featureId) │ │ │ │ │ - }, │ │ │ │ │ - getFeatureByFid: function(featureFid) { │ │ │ │ │ - return this.getFeatureBy("fid", featureFid) │ │ │ │ │ + var keyModifiers = (evt.shiftKey ? OpenLayers.Handler.MOD_SHIFT : 0) | (evt.ctrlKey ? OpenLayers.Handler.MOD_CTRL : 0) | (evt.altKey ? OpenLayers.Handler.MOD_ALT : 0) | (evt.metaKey ? OpenLayers.Handler.MOD_META : 0); │ │ │ │ │ + return keyModifiers == this.keyMask │ │ │ │ │ }, │ │ │ │ │ - getFeaturesByAttribute: function(attrName, attrValue) { │ │ │ │ │ - var i, feature, len = this.features.length, │ │ │ │ │ - foundFeatures = []; │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - if (feature && feature.attributes) { │ │ │ │ │ - if (feature.attributes[attrName] === attrValue) { │ │ │ │ │ - foundFeatures.push(feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return foundFeatures │ │ │ │ │ - }, │ │ │ │ │ - onFeatureInsert: function(feature) {}, │ │ │ │ │ - preFeatureInsert: function(feature) {}, │ │ │ │ │ - getDataExtent: function() { │ │ │ │ │ - var maxExtent = null; │ │ │ │ │ - var features = this.features; │ │ │ │ │ - if (features && features.length > 0) { │ │ │ │ │ - var geometry = null; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - geometry = features[i].geometry; │ │ │ │ │ - if (geometry) { │ │ │ │ │ - if (maxExtent === null) { │ │ │ │ │ - maxExtent = new OpenLayers.Bounds │ │ │ │ │ - } │ │ │ │ │ - maxExtent.extend(geometry.getBounds()) │ │ │ │ │ - } │ │ │ │ │ + var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.register(events[i], this[events[i]]) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return maxExtent │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Vector" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.WMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - service: "WMS", │ │ │ │ │ - version: "1.1.1", │ │ │ │ │ - request: "GetMap", │ │ │ │ │ - styles: "", │ │ │ │ │ - format: "image/jpeg" │ │ │ │ │ + this.active = true; │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - encodeBBOX: false, │ │ │ │ │ - noMagic: false, │ │ │ │ │ - yx: {}, │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ - if (parseFloat(params.VERSION) >= 1.3 && !params.EXCEPTIONS) { │ │ │ │ │ - params.EXCEPTIONS = "INIMAGE" │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - newArguments.push(name, url, params, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - OpenLayers.Util.applyDefaults(this.params, OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS)); │ │ │ │ │ - if (!this.noMagic && this.params.TRANSPARENT && this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ - if (options == null || !options.isBaseLayer) { │ │ │ │ │ - this.isBaseLayer = false │ │ │ │ │ - } │ │ │ │ │ - if (this.params.FORMAT == "image/jpeg") { │ │ │ │ │ - this.params.FORMAT = OpenLayers.Util.alphaHack() ? "image/gif" : "image/png" │ │ │ │ │ + var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.unregister(events[i], this[events[i]]) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + this.touch = false; │ │ │ │ │ + this.active = false; │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.WMS(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - reverseAxisOrder: function() { │ │ │ │ │ - var projCode = this.projection.getCode(); │ │ │ │ │ - return parseFloat(this.params.VERSION) >= 1.3 && !!(this.yx[projCode] || OpenLayers.Projection.defaults[projCode] && OpenLayers.Projection.defaults[projCode].yx) │ │ │ │ │ - }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var imageSize = this.getImageSize(); │ │ │ │ │ - var newParams = {}; │ │ │ │ │ - var reverseAxisOrder = this.reverseAxisOrder(); │ │ │ │ │ - newParams.BBOX = this.encodeBBOX ? bounds.toBBOX(null, reverseAxisOrder) : bounds.toArray(reverseAxisOrder); │ │ │ │ │ - newParams.WIDTH = imageSize.w; │ │ │ │ │ - newParams.HEIGHT = imageSize.h; │ │ │ │ │ - var requestString = this.getFullRequestString(newParams); │ │ │ │ │ - return requestString │ │ │ │ │ - }, │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ - var newArguments = [upperParams]; │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, newArguments) │ │ │ │ │ - }, │ │ │ │ │ - getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ - var mapProjection = this.map.getProjectionObject(); │ │ │ │ │ - var projectionCode = this.projection && this.projection.equals(mapProjection) ? this.projection.getCode() : mapProjection.getCode(); │ │ │ │ │ - var value = projectionCode == "none" ? null : projectionCode; │ │ │ │ │ - if (parseFloat(this.params.VERSION) >= 1.3) { │ │ │ │ │ - this.params.CRS = value │ │ │ │ │ - } else { │ │ │ │ │ - this.params.SRS = value │ │ │ │ │ - } │ │ │ │ │ - if (typeof this.params.TRANSPARENT == "boolean") { │ │ │ │ │ - newParams.TRANSPARENT = this.params.TRANSPARENT ? "TRUE" : "FALSE" │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.getFullRequestString.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.WMS" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format = OpenLayers.Class({ │ │ │ │ │ - options: null, │ │ │ │ │ - externalProjection: null, │ │ │ │ │ - internalProjection: null, │ │ │ │ │ - data: null, │ │ │ │ │ - keepData: false, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.options = options │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() {}, │ │ │ │ │ - read: function(data) { │ │ │ │ │ - throw new Error("Read not implemented.") │ │ │ │ │ - }, │ │ │ │ │ - write: function(object) { │ │ │ │ │ - throw new Error("Write not implemented.") │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.JSON = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ - indent: " ", │ │ │ │ │ - space: " ", │ │ │ │ │ - newline: "\n", │ │ │ │ │ - level: 0, │ │ │ │ │ - pretty: false, │ │ │ │ │ - nativeJSON: function() { │ │ │ │ │ - return !!(window.JSON && typeof JSON.parse == "function" && typeof JSON.stringify == "function") │ │ │ │ │ - }(), │ │ │ │ │ - read: function(json, filter) { │ │ │ │ │ - var object; │ │ │ │ │ - if (this.nativeJSON) { │ │ │ │ │ - object = JSON.parse(json, filter) │ │ │ │ │ - } else try { │ │ │ │ │ - if (/^[\],:{}\s]*$/.test(json.replace(/\\["\\\/bfnrtu]/g, "@").replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, "]").replace(/(?:^|:|,)(?:\s*\[)+/g, ""))) { │ │ │ │ │ - object = eval("(" + json + ")"); │ │ │ │ │ - if (typeof filter === "function") { │ │ │ │ │ - function walk(k, v) { │ │ │ │ │ - if (v && typeof v === "object") { │ │ │ │ │ - for (var i in v) { │ │ │ │ │ - if (v.hasOwnProperty(i)) { │ │ │ │ │ - v[i] = walk(i, v[i]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return filter(k, v) │ │ │ │ │ - } │ │ │ │ │ - object = walk("", object) │ │ │ │ │ + startTouch: function() { │ │ │ │ │ + if (!this.touch) { │ │ │ │ │ + this.touch = true; │ │ │ │ │ + var events = ["mousedown", "mouseup", "mousemove", "click", "dblclick", "mouseout"]; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.unregister(events[i], this[events[i]]) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - if (this.keepData) { │ │ │ │ │ - this.data = object │ │ │ │ │ } │ │ │ │ │ - return object │ │ │ │ │ }, │ │ │ │ │ - write: function(value, pretty) { │ │ │ │ │ - this.pretty = !!pretty; │ │ │ │ │ - var json = null; │ │ │ │ │ - var type = typeof value; │ │ │ │ │ - if (this.serialize[type]) { │ │ │ │ │ - try { │ │ │ │ │ - json = !this.pretty && this.nativeJSON ? JSON.stringify(value) : this.serialize[type].apply(this, [value]) │ │ │ │ │ - } catch (err) { │ │ │ │ │ - OpenLayers.Console.error("Trouble serializing: " + err) │ │ │ │ │ - } │ │ │ │ │ + callback: function(name, args) { │ │ │ │ │ + if (name && this.callbacks[name]) { │ │ │ │ │ + this.callbacks[name].apply(this.control, args) │ │ │ │ │ } │ │ │ │ │ - return json │ │ │ │ │ }, │ │ │ │ │ - writeIndent: function() { │ │ │ │ │ - var pieces = []; │ │ │ │ │ - if (this.pretty) { │ │ │ │ │ - for (var i = 0; i < this.level; ++i) { │ │ │ │ │ - pieces.push(this.indent) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return pieces.join("") │ │ │ │ │ + register: function(name, method) { │ │ │ │ │ + this.map.events.registerPriority(name, this, method); │ │ │ │ │ + this.map.events.registerPriority(name, this, this.setEvent) │ │ │ │ │ }, │ │ │ │ │ - writeNewline: function() { │ │ │ │ │ - return this.pretty ? this.newline : "" │ │ │ │ │ + unregister: function(name, method) { │ │ │ │ │ + this.map.events.unregister(name, this, method); │ │ │ │ │ + this.map.events.unregister(name, this, this.setEvent) │ │ │ │ │ }, │ │ │ │ │ - writeSpace: function() { │ │ │ │ │ - return this.pretty ? this.space : "" │ │ │ │ │ + setEvent: function(evt) { │ │ │ │ │ + this.evt = evt; │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - serialize: { │ │ │ │ │ - object: function(object) { │ │ │ │ │ - if (object == null) { │ │ │ │ │ - return "null" │ │ │ │ │ - } │ │ │ │ │ - if (object.constructor == Date) { │ │ │ │ │ - return this.serialize.date.apply(this, [object]) │ │ │ │ │ - } │ │ │ │ │ - if (object.constructor == Array) { │ │ │ │ │ - return this.serialize.array.apply(this, [object]) │ │ │ │ │ - } │ │ │ │ │ - var pieces = ["{"]; │ │ │ │ │ - this.level += 1; │ │ │ │ │ - var key, keyJSON, valueJSON; │ │ │ │ │ - var addComma = false; │ │ │ │ │ - for (key in object) { │ │ │ │ │ - if (object.hasOwnProperty(key)) { │ │ │ │ │ - keyJSON = OpenLayers.Format.JSON.prototype.write.apply(this, [key, this.pretty]); │ │ │ │ │ - valueJSON = OpenLayers.Format.JSON.prototype.write.apply(this, [object[key], this.pretty]); │ │ │ │ │ - if (keyJSON != null && valueJSON != null) { │ │ │ │ │ - if (addComma) { │ │ │ │ │ - pieces.push(",") │ │ │ │ │ - } │ │ │ │ │ - pieces.push(this.writeNewline(), this.writeIndent(), keyJSON, ":", this.writeSpace(), valueJSON); │ │ │ │ │ - addComma = true │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.level -= 1; │ │ │ │ │ - pieces.push(this.writeNewline(), this.writeIndent(), "}"); │ │ │ │ │ - return pieces.join("") │ │ │ │ │ - }, │ │ │ │ │ - array: function(array) { │ │ │ │ │ - var json; │ │ │ │ │ - var pieces = ["["]; │ │ │ │ │ - this.level += 1; │ │ │ │ │ - for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ - json = OpenLayers.Format.JSON.prototype.write.apply(this, [array[i], this.pretty]); │ │ │ │ │ - if (json != null) { │ │ │ │ │ - if (i > 0) { │ │ │ │ │ - pieces.push(",") │ │ │ │ │ - } │ │ │ │ │ - pieces.push(this.writeNewline(), this.writeIndent(), json) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.level -= 1; │ │ │ │ │ - pieces.push(this.writeNewline(), this.writeIndent(), "]"); │ │ │ │ │ - return pieces.join("") │ │ │ │ │ - }, │ │ │ │ │ - string: function(string) { │ │ │ │ │ - var m = { │ │ │ │ │ - "\b": "\\b", │ │ │ │ │ - "\t": "\\t", │ │ │ │ │ - "\n": "\\n", │ │ │ │ │ - "\f": "\\f", │ │ │ │ │ - "\r": "\\r", │ │ │ │ │ - '"': '\\"', │ │ │ │ │ - "\\": "\\\\" │ │ │ │ │ - }; │ │ │ │ │ - if (/["\\\x00-\x1f]/.test(string)) { │ │ │ │ │ - return '"' + string.replace(/([\x00-\x1f\\"])/g, function(a, b) { │ │ │ │ │ - var c = m[b]; │ │ │ │ │ - if (c) { │ │ │ │ │ - return c │ │ │ │ │ - } │ │ │ │ │ - c = b.charCodeAt(); │ │ │ │ │ - return "\\u00" + Math.floor(c / 16).toString(16) + (c % 16).toString(16) │ │ │ │ │ - }) + '"' │ │ │ │ │ - } │ │ │ │ │ - return '"' + string + '"' │ │ │ │ │ - }, │ │ │ │ │ - number: function(number) { │ │ │ │ │ - return isFinite(number) ? String(number) : "null" │ │ │ │ │ - }, │ │ │ │ │ - boolean: function(bool) { │ │ │ │ │ - return String(bool) │ │ │ │ │ - }, │ │ │ │ │ - date: function(date) { │ │ │ │ │ - function format(number) { │ │ │ │ │ - return number < 10 ? "0" + number : number │ │ │ │ │ - } │ │ │ │ │ - return '"' + date.getFullYear() + "-" + format(date.getMonth() + 1) + "-" + format(date.getDate()) + "T" + format(date.getHours()) + ":" + format(date.getMinutes()) + ":" + format(date.getSeconds()) + '"' │ │ │ │ │ - } │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.control = this.map = null │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.JSON" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler" │ │ │ │ │ }); │ │ │ │ │ +OpenLayers.Handler.MOD_NONE = 0; │ │ │ │ │ +OpenLayers.Handler.MOD_SHIFT = 1; │ │ │ │ │ +OpenLayers.Handler.MOD_CTRL = 2; │ │ │ │ │ +OpenLayers.Handler.MOD_ALT = 4; │ │ │ │ │ +OpenLayers.Handler.MOD_META = 8; │ │ │ │ │ OpenLayers.Geometry = OpenLayers.Class({ │ │ │ │ │ id: null, │ │ │ │ │ parent: null, │ │ │ │ │ bounds: null, │ │ │ │ │ initialize: function() { │ │ │ │ │ this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ }, │ │ │ │ │ @@ -7305,14 +5853,205 @@ │ │ │ │ │ return this │ │ │ │ │ }, │ │ │ │ │ getVertices: function(nodes) { │ │ │ │ │ return [this] │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Geometry.Point" │ │ │ │ │ }); │ │ │ │ │ +OpenLayers.Handler.Point = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + point: null, │ │ │ │ │ + layer: null, │ │ │ │ │ + multi: false, │ │ │ │ │ + citeCompliant: false, │ │ │ │ │ + mouseDown: false, │ │ │ │ │ + stoppedDown: null, │ │ │ │ │ + lastDown: null, │ │ │ │ │ + lastUp: null, │ │ │ │ │ + persist: false, │ │ │ │ │ + stopDown: false, │ │ │ │ │ + stopUp: false, │ │ │ │ │ + layerOptions: null, │ │ │ │ │ + pixelTolerance: 5, │ │ │ │ │ + lastTouchPx: null, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ + this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style["default"], {}) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (!OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + var options = OpenLayers.Util.extend({ │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + calculateInRange: OpenLayers.Function.True, │ │ │ │ │ + wrapDateLine: this.citeCompliant │ │ │ │ │ + }, this.layerOptions); │ │ │ │ │ + this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ + this.map.addLayer(this.layer); │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.point]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (!OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + this.cancel(); │ │ │ │ │ + if (this.layer.map != null) { │ │ │ │ │ + this.destroyFeature(true); │ │ │ │ │ + this.layer.destroy(false) │ │ │ │ │ + } │ │ │ │ │ + this.layer = null; │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + if (this.layer && (force || !this.persist)) { │ │ │ │ │ + this.layer.destroyFeatures() │ │ │ │ │ + } │ │ │ │ │ + this.point = null │ │ │ │ │ + }, │ │ │ │ │ + destroyPersistedFeature: function() { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + if (layer && layer.features.length > 1) { │ │ │ │ │ + this.layer.features[0].destroy() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + finalize: function(cancel) { │ │ │ │ │ + var key = cancel ? "cancel" : "done"; │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + this.lastDown = null; │ │ │ │ │ + this.lastUp = null; │ │ │ │ │ + this.lastTouchPx = null; │ │ │ │ │ + this.callback(key, [this.geometryClone()]); │ │ │ │ │ + this.destroyFeature(cancel) │ │ │ │ │ + }, │ │ │ │ │ + cancel: function() { │ │ │ │ │ + this.finalize(true) │ │ │ │ │ + }, │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + modifyFeature: function(pixel) { │ │ │ │ │ + if (!this.point) { │ │ │ │ │ + this.createFeature(pixel) │ │ │ │ │ + } │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point.geometry.x = lonlat.lon; │ │ │ │ │ + this.point.geometry.y = lonlat.lat; │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.point, false]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.drawFeature() │ │ │ │ │ + }, │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.point, this.style) │ │ │ │ │ + }, │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.point && this.point.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiPoint([geometry]) │ │ │ │ │ + } │ │ │ │ │ + return geometry │ │ │ │ │ + }, │ │ │ │ │ + geometryClone: function() { │ │ │ │ │ + var geom = this.getGeometry(); │ │ │ │ │ + return geom && geom.clone() │ │ │ │ │ + }, │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + return this.down(evt) │ │ │ │ │ + }, │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + this.lastTouchPx = evt.xy; │ │ │ │ │ + return this.down(evt) │ │ │ │ │ + }, │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + return this.move(evt) │ │ │ │ │ + }, │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + this.lastTouchPx = evt.xy; │ │ │ │ │ + return this.move(evt) │ │ │ │ │ + }, │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + return this.up(evt) │ │ │ │ │ + }, │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + evt.xy = this.lastTouchPx; │ │ │ │ │ + return this.up(evt) │ │ │ │ │ + }, │ │ │ │ │ + down: function(evt) { │ │ │ │ │ + this.mouseDown = true; │ │ │ │ │ + this.lastDown = evt.xy; │ │ │ │ │ + if (!this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy) │ │ │ │ │ + } │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + return !this.stopDown │ │ │ │ │ + }, │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ + this.modifyFeature(evt.xy) │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + up: function(evt) { │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + if (!this.checkModifiers(evt)) { │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + if (this.lastUp && this.lastUp.equals(evt.xy)) { │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + if (this.lastDown && this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance)) { │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy) │ │ │ │ │ + } │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature() │ │ │ │ │ + } │ │ │ │ │ + this.lastUp = evt.xy; │ │ │ │ │ + this.finalize(); │ │ │ │ │ + return !this.stopUp │ │ │ │ │ + } else { │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + mouseout: function(evt) { │ │ │ │ │ + if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + this.mouseDown = false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + passesTolerance: function(pixel1, pixel2, tolerance) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + if (tolerance != null && pixel1 && pixel2) { │ │ │ │ │ + var dist = pixel1.distanceTo(pixel2); │ │ │ │ │ + if (dist > tolerance) { │ │ │ │ │ + passes = false │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return passes │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Point" │ │ │ │ │ +}); │ │ │ │ │ OpenLayers.Geometry.Collection = OpenLayers.Class(OpenLayers.Geometry, { │ │ │ │ │ components: null, │ │ │ │ │ componentTypes: null, │ │ │ │ │ initialize: function(components) { │ │ │ │ │ OpenLayers.Geometry.prototype.initialize.apply(this, arguments); │ │ │ │ │ this.components = []; │ │ │ │ │ if (components != null) { │ │ │ │ │ @@ -8005,140 +6744,241 @@ │ │ │ │ │ return new OpenLayers.Geometry.LineString(returnPoints) │ │ │ │ │ } else { │ │ │ │ │ return this │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Geometry.LineString" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Geometry.MultiLineString = OpenLayers.Class(OpenLayers.Geometry.Collection, { │ │ │ │ │ - componentTypes: ["OpenLayers.Geometry.LineString"], │ │ │ │ │ - split: function(geometry, options) { │ │ │ │ │ - var results = null; │ │ │ │ │ - var mutual = options && options.mutual; │ │ │ │ │ - var splits, sourceLine, sourceLines, sourceSplit, targetSplit; │ │ │ │ │ - var sourceParts = []; │ │ │ │ │ - var targetParts = [geometry]; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ - sourceLine = this.components[i]; │ │ │ │ │ - sourceSplit = false; │ │ │ │ │ - for (var j = 0; j < targetParts.length; ++j) { │ │ │ │ │ - splits = sourceLine.split(targetParts[j], options); │ │ │ │ │ - if (splits) { │ │ │ │ │ - if (mutual) { │ │ │ │ │ - sourceLines = splits[0]; │ │ │ │ │ - for (var k = 0, klen = sourceLines.length; k < klen; ++k) { │ │ │ │ │ - if (k === 0 && sourceParts.length) { │ │ │ │ │ - sourceParts[sourceParts.length - 1].addComponent(sourceLines[k]) │ │ │ │ │ - } else { │ │ │ │ │ - sourceParts.push(new OpenLayers.Geometry.MultiLineString([sourceLines[k]])) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - sourceSplit = true; │ │ │ │ │ - splits = splits[1] │ │ │ │ │ - } │ │ │ │ │ - if (splits.length) { │ │ │ │ │ - splits.unshift(j, 1); │ │ │ │ │ - Array.prototype.splice.apply(targetParts, splits); │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ +OpenLayers.Handler.Path = OpenLayers.Class(OpenLayers.Handler.Point, { │ │ │ │ │ + line: null, │ │ │ │ │ + maxVertices: null, │ │ │ │ │ + doubleTouchTolerance: 20, │ │ │ │ │ + freehand: false, │ │ │ │ │ + freehandToggle: "shiftKey", │ │ │ │ │ + timerId: null, │ │ │ │ │ + redoStack: null, │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.line = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.LineString([this.point.geometry])); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.line, this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + OpenLayers.Handler.Point.prototype.destroyFeature.call(this, force); │ │ │ │ │ + this.line = null │ │ │ │ │ + }, │ │ │ │ │ + destroyPersistedFeature: function() { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + if (layer && layer.features.length > 2) { │ │ │ │ │ + this.layer.features[0].destroy() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + removePoint: function() { │ │ │ │ │ + if (this.point) { │ │ │ │ │ + this.layer.removeFeatures([this.point]) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addPoint: function(pixel) { │ │ │ │ │ + this.layer.removeFeatures([this.point]); │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat)); │ │ │ │ │ + this.line.geometry.addComponent(this.point.geometry, this.line.geometry.components.length); │ │ │ │ │ + this.layer.addFeatures([this.point]); │ │ │ │ │ + this.callback("point", [this.point.geometry, this.getGeometry()]); │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + delete this.redoStack │ │ │ │ │ + }, │ │ │ │ │ + insertXY: function(x, y) { │ │ │ │ │ + this.line.geometry.addComponent(new OpenLayers.Geometry.Point(x, y), this.getCurrentPointIndex()); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + delete this.redoStack │ │ │ │ │ + }, │ │ │ │ │ + insertDeltaXY: function(dx, dy) { │ │ │ │ │ + var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ + var p0 = this.line.geometry.components[previousIndex]; │ │ │ │ │ + if (p0 && !isNaN(p0.x) && !isNaN(p0.y)) { │ │ │ │ │ + this.insertXY(p0.x + dx, p0.y + dy) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + insertDirectionLength: function(direction, length) { │ │ │ │ │ + direction *= Math.PI / 180; │ │ │ │ │ + var dx = length * Math.cos(direction); │ │ │ │ │ + var dy = length * Math.sin(direction); │ │ │ │ │ + this.insertDeltaXY(dx, dy) │ │ │ │ │ + }, │ │ │ │ │ + insertDeflectionLength: function(deflection, length) { │ │ │ │ │ + var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ + if (previousIndex > 0) { │ │ │ │ │ + var p1 = this.line.geometry.components[previousIndex]; │ │ │ │ │ + var p0 = this.line.geometry.components[previousIndex - 1]; │ │ │ │ │ + var theta = Math.atan2(p1.y - p0.y, p1.x - p0.x); │ │ │ │ │ + this.insertDirectionLength(theta * 180 / Math.PI + deflection, length) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getCurrentPointIndex: function() { │ │ │ │ │ + return this.line.geometry.components.length - 1 │ │ │ │ │ + }, │ │ │ │ │ + undo: function() { │ │ │ │ │ + var geometry = this.line.geometry; │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var index = this.getCurrentPointIndex() - 1; │ │ │ │ │ + var target = components[index]; │ │ │ │ │ + var undone = geometry.removeComponent(target); │ │ │ │ │ + if (undone) { │ │ │ │ │ + if (this.touch && index > 0) { │ │ │ │ │ + components = geometry.components; │ │ │ │ │ + var lastpt = components[index - 1]; │ │ │ │ │ + var curptidx = this.getCurrentPointIndex(); │ │ │ │ │ + var curpt = components[curptidx]; │ │ │ │ │ + curpt.x = lastpt.x; │ │ │ │ │ + curpt.y = lastpt.y │ │ │ │ │ } │ │ │ │ │ - if (!sourceSplit) { │ │ │ │ │ - if (sourceParts.length) { │ │ │ │ │ - sourceParts[sourceParts.length - 1].addComponent(sourceLine.clone()) │ │ │ │ │ - } else { │ │ │ │ │ - sourceParts = [new OpenLayers.Geometry.MultiLineString(sourceLine.clone())] │ │ │ │ │ - } │ │ │ │ │ + if (!this.redoStack) { │ │ │ │ │ + this.redoStack = [] │ │ │ │ │ } │ │ │ │ │ + this.redoStack.push(target); │ │ │ │ │ + this.drawFeature() │ │ │ │ │ } │ │ │ │ │ - if (sourceParts && sourceParts.length > 1) { │ │ │ │ │ - sourceSplit = true │ │ │ │ │ - } else { │ │ │ │ │ - sourceParts = [] │ │ │ │ │ + return undone │ │ │ │ │ + }, │ │ │ │ │ + redo: function() { │ │ │ │ │ + var target = this.redoStack && this.redoStack.pop(); │ │ │ │ │ + if (target) { │ │ │ │ │ + this.line.geometry.addComponent(target, this.getCurrentPointIndex()); │ │ │ │ │ + this.drawFeature() │ │ │ │ │ } │ │ │ │ │ - if (targetParts && targetParts.length > 1) { │ │ │ │ │ - targetSplit = true │ │ │ │ │ + return !!target │ │ │ │ │ + }, │ │ │ │ │ + freehandMode: function(evt) { │ │ │ │ │ + return this.freehandToggle && evt[this.freehandToggle] ? !this.freehand : this.freehand │ │ │ │ │ + }, │ │ │ │ │ + modifyFeature: function(pixel, drawing) { │ │ │ │ │ + if (!this.line) { │ │ │ │ │ + this.createFeature(pixel) │ │ │ │ │ + } │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point.geometry.x = lonlat.lon; │ │ │ │ │ + this.point.geometry.y = lonlat.lat; │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.getSketch(), drawing]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.drawFeature() │ │ │ │ │ + }, │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.line, this.style); │ │ │ │ │ + this.layer.drawFeature(this.point, this.style) │ │ │ │ │ + }, │ │ │ │ │ + getSketch: function() { │ │ │ │ │ + return this.line │ │ │ │ │ + }, │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.line && this.line.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiLineString([geometry]) │ │ │ │ │ + } │ │ │ │ │ + return geometry │ │ │ │ │ + }, │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + if (this.timerId && this.passesTolerance(this.lastTouchPx, evt.xy, this.doubleTouchTolerance)) { │ │ │ │ │ + this.finishGeometry(); │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + return false │ │ │ │ │ } else { │ │ │ │ │ - targetParts = [] │ │ │ │ │ + if (this.timerId) { │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null │ │ │ │ │ + } │ │ │ │ │ + this.timerId = window.setTimeout(OpenLayers.Function.bind(function() { │ │ │ │ │ + this.timerId = null │ │ │ │ │ + }, this), 300); │ │ │ │ │ + return OpenLayers.Handler.Point.prototype.touchstart.call(this, evt) │ │ │ │ │ } │ │ │ │ │ - if (sourceSplit || targetSplit) { │ │ │ │ │ - if (mutual) { │ │ │ │ │ - results = [sourceParts, targetParts] │ │ │ │ │ + }, │ │ │ │ │ + down: function(evt) { │ │ │ │ │ + var stopDown = this.stopDown; │ │ │ │ │ + if (this.freehandMode(evt)) { │ │ │ │ │ + stopDown = true; │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!this.touch && (!this.lastDown || !this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance))) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp) │ │ │ │ │ + } │ │ │ │ │ + this.mouseDown = true; │ │ │ │ │ + this.lastDown = evt.xy; │ │ │ │ │ + this.stoppedDown = stopDown; │ │ │ │ │ + return !stopDown │ │ │ │ │ + }, │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature() │ │ │ │ │ + } │ │ │ │ │ + if (this.maxVertices && this.line && this.line.geometry.components.length === this.maxVertices) { │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize() │ │ │ │ │ } else { │ │ │ │ │ - results = targetParts │ │ │ │ │ + this.addPoint(evt.xy) │ │ │ │ │ } │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return results │ │ │ │ │ + if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp) │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - splitWith: function(geometry, options) { │ │ │ │ │ - var results = null; │ │ │ │ │ - var mutual = options && options.mutual; │ │ │ │ │ - var splits, targetLine, sourceLines, sourceSplit, targetSplit, sourceParts, targetParts; │ │ │ │ │ - if (geometry instanceof OpenLayers.Geometry.LineString) { │ │ │ │ │ - targetParts = []; │ │ │ │ │ - sourceParts = [geometry]; │ │ │ │ │ - for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ - targetSplit = false; │ │ │ │ │ - targetLine = this.components[i]; │ │ │ │ │ - for (var j = 0; j < sourceParts.length; ++j) { │ │ │ │ │ - splits = sourceParts[j].split(targetLine, options); │ │ │ │ │ - if (splits) { │ │ │ │ │ - if (mutual) { │ │ │ │ │ - sourceLines = splits[0]; │ │ │ │ │ - if (sourceLines.length) { │ │ │ │ │ - sourceLines.unshift(j, 1); │ │ │ │ │ - Array.prototype.splice.apply(sourceParts, sourceLines); │ │ │ │ │ - j += sourceLines.length - 2 │ │ │ │ │ - } │ │ │ │ │ - splits = splits[1]; │ │ │ │ │ - if (splits.length === 0) { │ │ │ │ │ - splits = [targetLine.clone()] │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - for (var k = 0, klen = splits.length; k < klen; ++k) { │ │ │ │ │ - if (k === 0 && targetParts.length) { │ │ │ │ │ - targetParts[targetParts.length - 1].addComponent(splits[k]) │ │ │ │ │ - } else { │ │ │ │ │ - targetParts.push(new OpenLayers.Geometry.MultiLineString([splits[k]])) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - targetSplit = true │ │ │ │ │ - } │ │ │ │ │ + up: function(evt) { │ │ │ │ │ + if (this.mouseDown && (!this.lastUp || !this.lastUp.equals(evt.xy))) { │ │ │ │ │ + if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature() │ │ │ │ │ } │ │ │ │ │ - if (!targetSplit) { │ │ │ │ │ - if (targetParts.length) { │ │ │ │ │ - targetParts[targetParts.length - 1].addComponent(targetLine.clone()) │ │ │ │ │ - } else { │ │ │ │ │ - targetParts = [new OpenLayers.Geometry.MultiLineString([targetLine.clone()])] │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize() │ │ │ │ │ + } else { │ │ │ │ │ + if (this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance)) { │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy) │ │ │ │ │ + } │ │ │ │ │ + if (this.lastUp == null && this.persist) { │ │ │ │ │ + this.destroyPersistedFeature() │ │ │ │ │ + } │ │ │ │ │ + this.addPoint(evt.xy); │ │ │ │ │ + this.lastUp = evt.xy; │ │ │ │ │ + if (this.line.geometry.components.length === this.maxVertices + 1) { │ │ │ │ │ + this.finishGeometry() │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - results = geometry.split(this) │ │ │ │ │ - } │ │ │ │ │ - if (sourceParts && sourceParts.length > 1) { │ │ │ │ │ - sourceSplit = true │ │ │ │ │ - } else { │ │ │ │ │ - sourceParts = [] │ │ │ │ │ - } │ │ │ │ │ - if (targetParts && targetParts.length > 1) { │ │ │ │ │ - targetSplit = true │ │ │ │ │ - } else { │ │ │ │ │ - targetParts = [] │ │ │ │ │ } │ │ │ │ │ - if (sourceSplit || targetSplit) { │ │ │ │ │ - if (mutual) { │ │ │ │ │ - results = [sourceParts, targetParts] │ │ │ │ │ - } else { │ │ │ │ │ - results = targetParts │ │ │ │ │ - } │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + return !this.stopUp │ │ │ │ │ + }, │ │ │ │ │ + finishGeometry: function() { │ │ │ │ │ + var index = this.line.geometry.components.length - 1; │ │ │ │ │ + this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize() │ │ │ │ │ + }, │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + if (!this.freehandMode(evt)) { │ │ │ │ │ + this.finishGeometry() │ │ │ │ │ } │ │ │ │ │ - return results │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Geometry.MultiLineString" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Path" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Geometry.LinearRing = OpenLayers.Class(OpenLayers.Geometry.LineString, { │ │ │ │ │ componentTypes: ["OpenLayers.Geometry.Point"], │ │ │ │ │ addComponent: function(point, index) { │ │ │ │ │ var added = false; │ │ │ │ │ var lastPoint = this.components.pop(); │ │ │ │ │ if (index != null || !point.equals(lastPoint)) { │ │ │ │ │ @@ -8432,833 +7272,1828 @@ │ │ │ │ │ x = origin.x + radius * Math.cos(rotatedAngle); │ │ │ │ │ y = origin.y + radius * Math.sin(rotatedAngle); │ │ │ │ │ points.push(new OpenLayers.Geometry.Point(x, y)) │ │ │ │ │ } │ │ │ │ │ var ring = new OpenLayers.Geometry.LinearRing(points); │ │ │ │ │ return new OpenLayers.Geometry.Polygon([ring]) │ │ │ │ │ }; │ │ │ │ │ -OpenLayers.Geometry.MultiPolygon = OpenLayers.Class(OpenLayers.Geometry.Collection, { │ │ │ │ │ - componentTypes: ["OpenLayers.Geometry.Polygon"], │ │ │ │ │ - CLASS_NAME: "OpenLayers.Geometry.MultiPolygon" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.GeoJSON = OpenLayers.Class(OpenLayers.Format.JSON, { │ │ │ │ │ - ignoreExtraDims: false, │ │ │ │ │ - read: function(json, type, filter) { │ │ │ │ │ - type = type ? type : "FeatureCollection"; │ │ │ │ │ - var results = null; │ │ │ │ │ - var obj = null; │ │ │ │ │ - if (typeof json == "string") { │ │ │ │ │ - obj = OpenLayers.Format.JSON.prototype.read.apply(this, [json, filter]) │ │ │ │ │ - } else { │ │ │ │ │ - obj = json │ │ │ │ │ +OpenLayers.Handler.Polygon = OpenLayers.Class(OpenLayers.Handler.Path, { │ │ │ │ │ + holeModifier: null, │ │ │ │ │ + drawingHole: false, │ │ │ │ │ + polygon: null, │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.line = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.LinearRing([this.point.geometry])); │ │ │ │ │ + this.polygon = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Polygon([this.line.geometry])); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.polygon, this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + addPoint: function(pixel) { │ │ │ │ │ + if (!this.drawingHole && this.holeModifier && this.evt && this.evt[this.holeModifier]) { │ │ │ │ │ + var geometry = this.point.geometry; │ │ │ │ │ + var features = this.control.layer.features; │ │ │ │ │ + var candidate, polygon; │ │ │ │ │ + for (var i = features.length - 1; i >= 0; --i) { │ │ │ │ │ + candidate = features[i].geometry; │ │ │ │ │ + if ((candidate instanceof OpenLayers.Geometry.Polygon || candidate instanceof OpenLayers.Geometry.MultiPolygon) && candidate.intersects(geometry)) { │ │ │ │ │ + polygon = features[i]; │ │ │ │ │ + this.control.layer.removeFeatures([polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.control.layer.events.registerPriority("sketchcomplete", this, this.finalizeInteriorRing); │ │ │ │ │ + this.control.layer.events.registerPriority("sketchmodified", this, this.enforceTopology); │ │ │ │ │ + polygon.geometry.addComponent(this.line.geometry); │ │ │ │ │ + this.polygon = polygon; │ │ │ │ │ + this.drawingHole = true; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (!obj) { │ │ │ │ │ - OpenLayers.Console.error("Bad JSON: " + json) │ │ │ │ │ - } else if (typeof obj.type != "string") { │ │ │ │ │ - OpenLayers.Console.error("Bad GeoJSON - no type: " + json) │ │ │ │ │ - } else if (this.isValidType(obj, type)) { │ │ │ │ │ - switch (type) { │ │ │ │ │ - case "Geometry": │ │ │ │ │ - try { │ │ │ │ │ - results = this.parseGeometry(obj) │ │ │ │ │ - } catch (err) { │ │ │ │ │ - OpenLayers.Console.error(err) │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "Feature": │ │ │ │ │ - try { │ │ │ │ │ - results = this.parseFeature(obj); │ │ │ │ │ - results.type = "Feature" │ │ │ │ │ - } catch (err) { │ │ │ │ │ - OpenLayers.Console.error(err) │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "FeatureCollection": │ │ │ │ │ - results = []; │ │ │ │ │ - switch (obj.type) { │ │ │ │ │ - case "Feature": │ │ │ │ │ - try { │ │ │ │ │ - results.push(this.parseFeature(obj)) │ │ │ │ │ - } catch (err) { │ │ │ │ │ - results = null; │ │ │ │ │ - OpenLayers.Console.error(err) │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "FeatureCollection": │ │ │ │ │ - for (var i = 0, len = obj.features.length; i < len; ++i) { │ │ │ │ │ - try { │ │ │ │ │ - results.push(this.parseFeature(obj.features[i])) │ │ │ │ │ - } catch (err) { │ │ │ │ │ - results = null; │ │ │ │ │ - OpenLayers.Console.error(err) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - try { │ │ │ │ │ - var geom = this.parseGeometry(obj); │ │ │ │ │ - results.push(new OpenLayers.Feature.Vector(geom)) │ │ │ │ │ - } catch (err) { │ │ │ │ │ - results = null; │ │ │ │ │ - OpenLayers.Console.error(err) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + OpenLayers.Handler.Path.prototype.addPoint.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + getCurrentPointIndex: function() { │ │ │ │ │ + return this.line.geometry.components.length - 2 │ │ │ │ │ + }, │ │ │ │ │ + enforceTopology: function(event) { │ │ │ │ │ + var point = event.vertex; │ │ │ │ │ + var components = this.line.geometry.components; │ │ │ │ │ + if (!this.polygon.geometry.intersects(point)) { │ │ │ │ │ + var last = components[components.length - 3]; │ │ │ │ │ + point.x = last.x; │ │ │ │ │ + point.y = last.y │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + finishGeometry: function() { │ │ │ │ │ + var index = this.line.geometry.components.length - 2; │ │ │ │ │ + this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize() │ │ │ │ │ + }, │ │ │ │ │ + finalizeInteriorRing: function() { │ │ │ │ │ + var ring = this.line.geometry; │ │ │ │ │ + var modified = ring.getArea() !== 0; │ │ │ │ │ + if (modified) { │ │ │ │ │ + var rings = this.polygon.geometry.components; │ │ │ │ │ + for (var i = rings.length - 2; i >= 0; --i) { │ │ │ │ │ + if (ring.intersects(rings[i])) { │ │ │ │ │ + modified = false; │ │ │ │ │ break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (modified) { │ │ │ │ │ + var target; │ │ │ │ │ + outer: for (var i = rings.length - 2; i > 0; --i) { │ │ │ │ │ + var points = rings[i].components; │ │ │ │ │ + for (var j = 0, jj = points.length; j < jj; ++j) { │ │ │ │ │ + if (ring.containsPoint(points[j])) { │ │ │ │ │ + modified = false; │ │ │ │ │ + break outer │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return results │ │ │ │ │ + if (modified) { │ │ │ │ │ + if (this.polygon.state !== OpenLayers.State.INSERT) { │ │ │ │ │ + this.polygon.state = OpenLayers.State.UPDATE │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this.polygon.geometry.removeComponent(ring) │ │ │ │ │ + } │ │ │ │ │ + this.restoreFeature(); │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - isValidType: function(obj, type) { │ │ │ │ │ - var valid = false; │ │ │ │ │ - switch (type) { │ │ │ │ │ - case "Geometry": │ │ │ │ │ - if (OpenLayers.Util.indexOf(["Point", "MultiPoint", "LineString", "MultiLineString", "Polygon", "MultiPolygon", "Box", "GeometryCollection"], obj.type) == -1) { │ │ │ │ │ - OpenLayers.Console.error("Unsupported geometry type: " + obj.type) │ │ │ │ │ + cancel: function() { │ │ │ │ │ + if (this.drawingHole) { │ │ │ │ │ + this.polygon.geometry.removeComponent(this.line.geometry); │ │ │ │ │ + this.restoreFeature(true) │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Handler.Path.prototype.cancel.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + restoreFeature: function(cancel) { │ │ │ │ │ + this.control.layer.events.unregister("sketchcomplete", this, this.finalizeInteriorRing); │ │ │ │ │ + this.control.layer.events.unregister("sketchmodified", this, this.enforceTopology); │ │ │ │ │ + this.layer.removeFeatures([this.polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.control.layer.addFeatures([this.polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.drawingHole = false; │ │ │ │ │ + if (!cancel) { │ │ │ │ │ + this.control.layer.events.triggerEvent("sketchcomplete", { │ │ │ │ │ + feature: this.polygon │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + OpenLayers.Handler.Path.prototype.destroyFeature.call(this, force); │ │ │ │ │ + this.polygon = null │ │ │ │ │ + }, │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.polygon, this.style); │ │ │ │ │ + this.layer.drawFeature(this.point, this.style) │ │ │ │ │ + }, │ │ │ │ │ + getSketch: function() { │ │ │ │ │ + return this.polygon │ │ │ │ │ + }, │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.polygon && this.polygon.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiPolygon([geometry]) │ │ │ │ │ + } │ │ │ │ │ + return geometry │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Polygon" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Renderer = OpenLayers.Class({ │ │ │ │ │ + container: null, │ │ │ │ │ + root: null, │ │ │ │ │ + extent: null, │ │ │ │ │ + locked: false, │ │ │ │ │ + size: null, │ │ │ │ │ + resolution: null, │ │ │ │ │ + map: null, │ │ │ │ │ + featureDx: 0, │ │ │ │ │ + initialize: function(containerID, options) { │ │ │ │ │ + this.container = OpenLayers.Util.getElement(containerID); │ │ │ │ │ + OpenLayers.Util.extend(this, options) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.container = null; │ │ │ │ │ + this.extent = null; │ │ │ │ │ + this.size = null; │ │ │ │ │ + this.resolution = null; │ │ │ │ │ + this.map = null │ │ │ │ │ + }, │ │ │ │ │ + supported: function() { │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + this.extent = extent.clone(); │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + var ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ + extent = extent.scale(1 / ratio); │ │ │ │ │ + this.extent = extent.wrapDateLine(this.map.getMaxExtent()).scale(ratio) │ │ │ │ │ + } │ │ │ │ │ + if (resolutionChanged) { │ │ │ │ │ + this.resolution = null │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + this.size = size.clone(); │ │ │ │ │ + this.resolution = null │ │ │ │ │ + }, │ │ │ │ │ + getResolution: function() { │ │ │ │ │ + this.resolution = this.resolution || this.map.getResolution(); │ │ │ │ │ + return this.resolution │ │ │ │ │ + }, │ │ │ │ │ + drawFeature: function(feature, style) { │ │ │ │ │ + if (style == null) { │ │ │ │ │ + style = feature.style │ │ │ │ │ + } │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + var bounds = feature.geometry.getBounds(); │ │ │ │ │ + if (bounds) { │ │ │ │ │ + var worldBounds; │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + worldBounds = this.map.getMaxExtent() │ │ │ │ │ + } │ │ │ │ │ + if (!bounds.intersectsBounds(this.extent, { │ │ │ │ │ + worldBounds: worldBounds │ │ │ │ │ + })) { │ │ │ │ │ + style = { │ │ │ │ │ + display: "none" │ │ │ │ │ + } │ │ │ │ │ } else { │ │ │ │ │ - valid = true │ │ │ │ │ + this.calculateFeatureDx(bounds, worldBounds) │ │ │ │ │ } │ │ │ │ │ - break; │ │ │ │ │ - case "FeatureCollection": │ │ │ │ │ - valid = true; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - if (obj.type == type) { │ │ │ │ │ - valid = true │ │ │ │ │ + var rendered = this.drawGeometry(feature.geometry, style, feature.id); │ │ │ │ │ + if (style.display != "none" && style.label && rendered !== false) { │ │ │ │ │ + var location = feature.geometry.getCentroid(); │ │ │ │ │ + if (style.labelXOffset || style.labelYOffset) { │ │ │ │ │ + var xOffset = isNaN(style.labelXOffset) ? 0 : style.labelXOffset; │ │ │ │ │ + var yOffset = isNaN(style.labelYOffset) ? 0 : style.labelYOffset; │ │ │ │ │ + var res = this.getResolution(); │ │ │ │ │ + location.move(xOffset * res, yOffset * res) │ │ │ │ │ + } │ │ │ │ │ + this.drawText(feature.id, style, location) │ │ │ │ │ } else { │ │ │ │ │ - OpenLayers.Console.error("Cannot convert types from " + obj.type + " to " + type) │ │ │ │ │ + this.removeText(feature.id) │ │ │ │ │ } │ │ │ │ │ + return rendered │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return valid │ │ │ │ │ }, │ │ │ │ │ - parseFeature: function(obj) { │ │ │ │ │ - var feature, geometry, attributes, bbox; │ │ │ │ │ - attributes = obj.properties ? obj.properties : {}; │ │ │ │ │ - bbox = obj.geometry && obj.geometry.bbox || obj.bbox; │ │ │ │ │ - try { │ │ │ │ │ - geometry = this.parseGeometry(obj.geometry) │ │ │ │ │ - } catch (err) { │ │ │ │ │ - throw err │ │ │ │ │ + calculateFeatureDx: function(bounds, worldBounds) { │ │ │ │ │ + this.featureDx = 0; │ │ │ │ │ + if (worldBounds) { │ │ │ │ │ + var worldWidth = worldBounds.getWidth(), │ │ │ │ │ + rendererCenterX = (this.extent.left + this.extent.right) / 2, │ │ │ │ │ + featureCenterX = (bounds.left + bounds.right) / 2, │ │ │ │ │ + worldsAway = Math.round((featureCenterX - rendererCenterX) / worldWidth); │ │ │ │ │ + this.featureDx = worldsAway * worldWidth │ │ │ │ │ } │ │ │ │ │ - feature = new OpenLayers.Feature.Vector(geometry, attributes); │ │ │ │ │ - if (bbox) { │ │ │ │ │ - feature.bounds = OpenLayers.Bounds.fromArray(bbox) │ │ │ │ │ + }, │ │ │ │ │ + drawGeometry: function(geometry, style, featureId) {}, │ │ │ │ │ + drawText: function(featureId, style, location) {}, │ │ │ │ │ + removeText: function(featureId) {}, │ │ │ │ │ + clear: function() {}, │ │ │ │ │ + getFeatureIdFromEvent: function(evt) {}, │ │ │ │ │ + eraseFeatures: function(features) { │ │ │ │ │ + if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ + features = [features] │ │ │ │ │ } │ │ │ │ │ - if (obj.id) { │ │ │ │ │ - feature.fid = obj.id │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + this.eraseGeometry(feature.geometry, feature.id); │ │ │ │ │ + this.removeText(feature.id) │ │ │ │ │ } │ │ │ │ │ - return feature │ │ │ │ │ }, │ │ │ │ │ - parseGeometry: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - return null │ │ │ │ │ + eraseGeometry: function(geometry, featureId) {}, │ │ │ │ │ + moveRoot: function(renderer) {}, │ │ │ │ │ + getRenderLayerId: function() { │ │ │ │ │ + return this.container.id │ │ │ │ │ + }, │ │ │ │ │ + applyDefaultSymbolizer: function(symbolizer) { │ │ │ │ │ + var result = OpenLayers.Util.extend({}, OpenLayers.Renderer.defaultSymbolizer); │ │ │ │ │ + if (symbolizer.stroke === false) { │ │ │ │ │ + delete result.strokeWidth; │ │ │ │ │ + delete result.strokeColor │ │ │ │ │ } │ │ │ │ │ - var geometry, collection = false; │ │ │ │ │ - if (obj.type == "GeometryCollection") { │ │ │ │ │ - if (!OpenLayers.Util.isArray(obj.geometries)) { │ │ │ │ │ - throw "GeometryCollection must have geometries array: " + obj │ │ │ │ │ - } │ │ │ │ │ - var numGeom = obj.geometries.length; │ │ │ │ │ - var components = new Array(numGeom); │ │ │ │ │ - for (var i = 0; i < numGeom; ++i) { │ │ │ │ │ - components[i] = this.parseGeometry.apply(this, [obj.geometries[i]]) │ │ │ │ │ - } │ │ │ │ │ - geometry = new OpenLayers.Geometry.Collection(components); │ │ │ │ │ - collection = true │ │ │ │ │ - } else { │ │ │ │ │ - if (!OpenLayers.Util.isArray(obj.coordinates)) { │ │ │ │ │ - throw "Geometry must have coordinates array: " + obj │ │ │ │ │ + if (symbolizer.fill === false) { │ │ │ │ │ + delete result.fillColor │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.extend(result, symbolizer); │ │ │ │ │ + return result │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Renderer.defaultSymbolizer = { │ │ │ │ │ + fillColor: "#000000", │ │ │ │ │ + strokeColor: "#000000", │ │ │ │ │ + strokeWidth: 2, │ │ │ │ │ + fillOpacity: 1, │ │ │ │ │ + strokeOpacity: 1, │ │ │ │ │ + pointRadius: 0, │ │ │ │ │ + labelAlign: "cm" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.symbol = { │ │ │ │ │ + star: [350, 75, 379, 161, 469, 161, 397, 215, 423, 301, 350, 250, 277, 301, 303, 215, 231, 161, 321, 161, 350, 75], │ │ │ │ │ + cross: [4, 0, 6, 0, 6, 4, 10, 4, 10, 6, 6, 6, 6, 10, 4, 10, 4, 6, 0, 6, 0, 4, 4, 4, 4, 0], │ │ │ │ │ + x: [0, 0, 25, 0, 50, 35, 75, 0, 100, 0, 65, 50, 100, 100, 75, 100, 50, 65, 25, 100, 0, 100, 35, 50, 0, 0], │ │ │ │ │ + square: [0, 0, 0, 1, 1, 1, 1, 0, 0, 0], │ │ │ │ │ + triangle: [0, 10, 10, 10, 5, 0, 0, 10] │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.Canvas = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ + hitDetection: true, │ │ │ │ │ + hitOverflow: 0, │ │ │ │ │ + canvas: null, │ │ │ │ │ + features: null, │ │ │ │ │ + pendingRedraw: false, │ │ │ │ │ + cachedSymbolBounds: {}, │ │ │ │ │ + initialize: function(containerID, options) { │ │ │ │ │ + OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.root = document.createElement("canvas"); │ │ │ │ │ + this.container.appendChild(this.root); │ │ │ │ │ + this.canvas = this.root.getContext("2d"); │ │ │ │ │ + this.features = {}; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitCanvas = document.createElement("canvas"); │ │ │ │ │ + this.hitContext = this.hitCanvas.getContext("2d") │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setExtent: function() { │ │ │ │ │ + OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + eraseGeometry: function(geometry, featureId) { │ │ │ │ │ + this.eraseFeatures(this.features[featureId][0]) │ │ │ │ │ + }, │ │ │ │ │ + supported: function() { │ │ │ │ │ + return OpenLayers.CANVAS_SUPPORTED │ │ │ │ │ + }, │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + this.size = size.clone(); │ │ │ │ │ + var root = this.root; │ │ │ │ │ + root.style.width = size.w + "px"; │ │ │ │ │ + root.style.height = size.h + "px"; │ │ │ │ │ + root.width = size.w; │ │ │ │ │ + root.height = size.h; │ │ │ │ │ + this.resolution = null; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + var hitCanvas = this.hitCanvas; │ │ │ │ │ + hitCanvas.style.width = size.w + "px"; │ │ │ │ │ + hitCanvas.style.height = size.h + "px"; │ │ │ │ │ + hitCanvas.width = size.w; │ │ │ │ │ + hitCanvas.height = size.h │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + drawFeature: function(feature, style) { │ │ │ │ │ + var rendered; │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + style = this.applyDefaultSymbolizer(style || feature.style); │ │ │ │ │ + var bounds = feature.geometry.getBounds(); │ │ │ │ │ + var worldBounds; │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + worldBounds = this.map.getMaxExtent() │ │ │ │ │ } │ │ │ │ │ - if (!this.parseCoords[obj.type.toLowerCase()]) { │ │ │ │ │ - throw "Unsupported geometry type: " + obj.type │ │ │ │ │ + var intersects = bounds && bounds.intersectsBounds(this.extent, { │ │ │ │ │ + worldBounds: worldBounds │ │ │ │ │ + }); │ │ │ │ │ + rendered = style.display !== "none" && !!bounds && intersects; │ │ │ │ │ + if (rendered) { │ │ │ │ │ + this.features[feature.id] = [feature, style] │ │ │ │ │ + } else { │ │ │ │ │ + delete this.features[feature.id] │ │ │ │ │ } │ │ │ │ │ - try { │ │ │ │ │ - geometry = this.parseCoords[obj.type.toLowerCase()].apply(this, [obj.coordinates]) │ │ │ │ │ - } catch (err) { │ │ │ │ │ - throw err │ │ │ │ │ + this.pendingRedraw = true │ │ │ │ │ + } │ │ │ │ │ + if (this.pendingRedraw && !this.locked) { │ │ │ │ │ + this.redraw(); │ │ │ │ │ + this.pendingRedraw = false │ │ │ │ │ + } │ │ │ │ │ + return rendered │ │ │ │ │ + }, │ │ │ │ │ + drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ + var className = geometry.CLASS_NAME; │ │ │ │ │ + if (className == "OpenLayers.Geometry.Collection" || className == "OpenLayers.Geometry.MultiPoint" || className == "OpenLayers.Geometry.MultiLineString" || className == "OpenLayers.Geometry.MultiPolygon") { │ │ │ │ │ + for (var i = 0; i < geometry.components.length; i++) { │ │ │ │ │ + this.drawGeometry(geometry.components[i], style, featureId) │ │ │ │ │ } │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - if (this.internalProjection && this.externalProjection && !collection) { │ │ │ │ │ - geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + this.drawPoint(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + this.drawLineString(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + this.drawPolygon(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - return geometry │ │ │ │ │ }, │ │ │ │ │ - parseCoords: { │ │ │ │ │ - point: function(array) { │ │ │ │ │ - if (this.ignoreExtraDims == false && array.length != 2) { │ │ │ │ │ - throw "Only 2D points are supported: " + array │ │ │ │ │ + drawExternalGraphic: function(geometry, style, featureId) { │ │ │ │ │ + var img = new Image; │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + img.title = title │ │ │ │ │ + } │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ + var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ + var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ + var onLoad = function() { │ │ │ │ │ + if (!this.features[featureId]) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Geometry.Point(array[0], array[1]) │ │ │ │ │ - }, │ │ │ │ │ - multipoint: function(array) { │ │ │ │ │ - var points = []; │ │ │ │ │ - var p = null; │ │ │ │ │ - for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ - try { │ │ │ │ │ - p = this.parseCoords["point"].apply(this, [array[i]]) │ │ │ │ │ - } catch (err) { │ │ │ │ │ - throw err │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ + if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ + var x = p0 + xOffset | 0; │ │ │ │ │ + var y = p1 + yOffset | 0; │ │ │ │ │ + var canvas = this.canvas; │ │ │ │ │ + canvas.globalAlpha = opacity; │ │ │ │ │ + var factor = OpenLayers.Renderer.Canvas.drawImageScaleFactor || (OpenLayers.Renderer.Canvas.drawImageScaleFactor = /android 2.1/.test(navigator.userAgent.toLowerCase()) ? 320 / window.screen.width : 1); │ │ │ │ │ + canvas.drawImage(img, x * factor, y * factor, width * factor, height * factor); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId); │ │ │ │ │ + this.hitContext.fillRect(x, y, width, height) │ │ │ │ │ } │ │ │ │ │ - points.push(p) │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Geometry.MultiPoint(points) │ │ │ │ │ - }, │ │ │ │ │ - linestring: function(array) { │ │ │ │ │ - var points = []; │ │ │ │ │ - var p = null; │ │ │ │ │ - for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ - try { │ │ │ │ │ - p = this.parseCoords["point"].apply(this, [array[i]]) │ │ │ │ │ - } catch (err) { │ │ │ │ │ - throw err │ │ │ │ │ - } │ │ │ │ │ - points.push(p) │ │ │ │ │ + }; │ │ │ │ │ + img.onload = OpenLayers.Function.bind(onLoad, this); │ │ │ │ │ + img.src = style.externalGraphic │ │ │ │ │ + }, │ │ │ │ │ + drawNamedSymbol: function(geometry, style, featureId) { │ │ │ │ │ + var x, y, cx, cy, i, symbolBounds, scaling, angle; │ │ │ │ │ + var unscaledStrokeWidth; │ │ │ │ │ + var deg2rad = Math.PI / 180; │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[style.graphicName]; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(style.graphicName + " is not a valid symbol name") │ │ │ │ │ + } │ │ │ │ │ + if (!symbol.length || symbol.length < 2) return; │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ + if (isNaN(p0) || isNaN(p1)) return; │ │ │ │ │ + this.canvas.lineCap = "round"; │ │ │ │ │ + this.canvas.lineJoin = "round"; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.lineCap = "round"; │ │ │ │ │ + this.hitContext.lineJoin = "round" │ │ │ │ │ + } │ │ │ │ │ + if (style.graphicName in this.cachedSymbolBounds) { │ │ │ │ │ + symbolBounds = this.cachedSymbolBounds[style.graphicName] │ │ │ │ │ + } else { │ │ │ │ │ + symbolBounds = new OpenLayers.Bounds; │ │ │ │ │ + for (i = 0; i < symbol.length; i += 2) { │ │ │ │ │ + symbolBounds.extend(new OpenLayers.LonLat(symbol[i], symbol[i + 1])) │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Geometry.LineString(points) │ │ │ │ │ - }, │ │ │ │ │ - multilinestring: function(array) { │ │ │ │ │ - var lines = []; │ │ │ │ │ - var l = null; │ │ │ │ │ - for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ - try { │ │ │ │ │ - l = this.parseCoords["linestring"].apply(this, [array[i]]) │ │ │ │ │ - } catch (err) { │ │ │ │ │ - throw err │ │ │ │ │ - } │ │ │ │ │ - lines.push(l) │ │ │ │ │ + this.cachedSymbolBounds[style.graphicName] = symbolBounds │ │ │ │ │ + } │ │ │ │ │ + this.canvas.save(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.save() │ │ │ │ │ + } │ │ │ │ │ + this.canvas.translate(p0, p1); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.translate(p0, p1) │ │ │ │ │ + } │ │ │ │ │ + angle = deg2rad * style.rotation; │ │ │ │ │ + if (!isNaN(angle)) { │ │ │ │ │ + this.canvas.rotate(angle); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.rotate(angle) │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Geometry.MultiLineString(lines) │ │ │ │ │ - }, │ │ │ │ │ - polygon: function(array) { │ │ │ │ │ - var rings = []; │ │ │ │ │ - var r, l; │ │ │ │ │ - for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ - try { │ │ │ │ │ - l = this.parseCoords["linestring"].apply(this, [array[i]]) │ │ │ │ │ - } catch (err) { │ │ │ │ │ - throw err │ │ │ │ │ - } │ │ │ │ │ - r = new OpenLayers.Geometry.LinearRing(l.components); │ │ │ │ │ - rings.push(r) │ │ │ │ │ + } │ │ │ │ │ + scaling = 2 * style.pointRadius / Math.max(symbolBounds.getWidth(), symbolBounds.getHeight()); │ │ │ │ │ + this.canvas.scale(scaling, scaling); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.scale(scaling, scaling) │ │ │ │ │ + } │ │ │ │ │ + cx = symbolBounds.getCenterLonLat().lon; │ │ │ │ │ + cy = symbolBounds.getCenterLonLat().lat; │ │ │ │ │ + this.canvas.translate(-cx, -cy); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.translate(-cx, -cy) │ │ │ │ │ + } │ │ │ │ │ + unscaledStrokeWidth = style.strokeWidth; │ │ │ │ │ + style.strokeWidth = unscaledStrokeWidth / scaling; │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.canvas.lineTo(x, y) │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Geometry.Polygon(rings) │ │ │ │ │ - }, │ │ │ │ │ - multipolygon: function(array) { │ │ │ │ │ - var polys = []; │ │ │ │ │ - var p = null; │ │ │ │ │ - for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ - try { │ │ │ │ │ - p = this.parseCoords["polygon"].apply(this, [array[i]]) │ │ │ │ │ - } catch (err) { │ │ │ │ │ - throw err │ │ │ │ │ + this.canvas.closePath(); │ │ │ │ │ + this.canvas.fill(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.hitContext.lineTo(x, y) │ │ │ │ │ } │ │ │ │ │ - polys.push(p) │ │ │ │ │ + this.hitContext.closePath(); │ │ │ │ │ + this.hitContext.fill() │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Geometry.MultiPolygon(polys) │ │ │ │ │ - }, │ │ │ │ │ - box: function(array) { │ │ │ │ │ - if (array.length != 2) { │ │ │ │ │ - throw "GeoJSON box coordinates must have 2 elements" │ │ │ │ │ + } │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.canvas.lineTo(x, y) │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Geometry.Polygon([new OpenLayers.Geometry.LinearRing([new OpenLayers.Geometry.Point(array[0][0], array[0][1]), new OpenLayers.Geometry.Point(array[1][0], array[0][1]), new OpenLayers.Geometry.Point(array[1][0], array[1][1]), new OpenLayers.Geometry.Point(array[0][0], array[1][1]), new OpenLayers.Geometry.Point(array[0][0], array[0][1])])]) │ │ │ │ │ + this.canvas.closePath(); │ │ │ │ │ + this.canvas.stroke(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style, scaling); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.hitContext.moveTo(x, y); │ │ │ │ │ + this.hitContext.lineTo(x, y) │ │ │ │ │ + } │ │ │ │ │ + this.hitContext.closePath(); │ │ │ │ │ + this.hitContext.stroke() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + style.strokeWidth = unscaledStrokeWidth; │ │ │ │ │ + this.canvas.restore(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.restore() │ │ │ │ │ } │ │ │ │ │ + this.setCanvasStyle("reset") │ │ │ │ │ }, │ │ │ │ │ - write: function(obj, pretty) { │ │ │ │ │ - var geojson = { │ │ │ │ │ - type: null │ │ │ │ │ - }; │ │ │ │ │ - if (OpenLayers.Util.isArray(obj)) { │ │ │ │ │ - geojson.type = "FeatureCollection"; │ │ │ │ │ - var numFeatures = obj.length; │ │ │ │ │ - geojson.features = new Array(numFeatures); │ │ │ │ │ - for (var i = 0; i < numFeatures; ++i) { │ │ │ │ │ - var element = obj[i]; │ │ │ │ │ - if (!element instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ - var msg = "FeatureCollection only supports collections " + "of features: " + element; │ │ │ │ │ - throw msg │ │ │ │ │ + setCanvasStyle: function(type, style) { │ │ │ │ │ + if (type === "fill") { │ │ │ │ │ + this.canvas.globalAlpha = style["fillOpacity"]; │ │ │ │ │ + this.canvas.fillStyle = style["fillColor"] │ │ │ │ │ + } else if (type === "stroke") { │ │ │ │ │ + this.canvas.globalAlpha = style["strokeOpacity"]; │ │ │ │ │ + this.canvas.strokeStyle = style["strokeColor"]; │ │ │ │ │ + this.canvas.lineWidth = style["strokeWidth"] │ │ │ │ │ + } else { │ │ │ │ │ + this.canvas.globalAlpha = 0; │ │ │ │ │ + this.canvas.lineWidth = 1 │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + featureIdToHex: function(featureId) { │ │ │ │ │ + var id = Number(featureId.split("_").pop()) + 1; │ │ │ │ │ + if (id >= 16777216) { │ │ │ │ │ + this.hitOverflow = id - 16777215; │ │ │ │ │ + id = id % 16777216 + 1 │ │ │ │ │ + } │ │ │ │ │ + var hex = "000000" + id.toString(16); │ │ │ │ │ + var len = hex.length; │ │ │ │ │ + hex = "#" + hex.substring(len - 6, len); │ │ │ │ │ + return hex │ │ │ │ │ + }, │ │ │ │ │ + setHitContextStyle: function(type, featureId, symbolizer, strokeScaling) { │ │ │ │ │ + var hex = this.featureIdToHex(featureId); │ │ │ │ │ + if (type == "fill") { │ │ │ │ │ + this.hitContext.globalAlpha = 1; │ │ │ │ │ + this.hitContext.fillStyle = hex │ │ │ │ │ + } else if (type == "stroke") { │ │ │ │ │ + this.hitContext.globalAlpha = 1; │ │ │ │ │ + this.hitContext.strokeStyle = hex; │ │ │ │ │ + if (typeof strokeScaling === "undefined") { │ │ │ │ │ + this.hitContext.lineWidth = symbolizer.strokeWidth + 2 │ │ │ │ │ + } else { │ │ │ │ │ + if (!isNaN(strokeScaling)) { │ │ │ │ │ + this.hitContext.lineWidth = symbolizer.strokeWidth + 2 / strokeScaling │ │ │ │ │ } │ │ │ │ │ - geojson.features[i] = this.extract.feature.apply(this, [element]) │ │ │ │ │ - } │ │ │ │ │ - } else if (obj.CLASS_NAME.indexOf("OpenLayers.Geometry") == 0) { │ │ │ │ │ - geojson = this.extract.geometry.apply(this, [obj]) │ │ │ │ │ - } else if (obj instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ - geojson = this.extract.feature.apply(this, [obj]); │ │ │ │ │ - if (obj.layer && obj.layer.projection) { │ │ │ │ │ - geojson.crs = this.createCRSObject(obj) │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + this.hitContext.globalAlpha = 0; │ │ │ │ │ + this.hitContext.lineWidth = 1 │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Format.JSON.prototype.write.apply(this, [geojson, pretty]) │ │ │ │ │ }, │ │ │ │ │ - createCRSObject: function(object) { │ │ │ │ │ - var proj = object.layer.projection.toString(); │ │ │ │ │ - var crs = {}; │ │ │ │ │ - if (proj.match(/epsg:/i)) { │ │ │ │ │ - var code = parseInt(proj.substring(proj.indexOf(":") + 1)); │ │ │ │ │ - if (code == 4326) { │ │ │ │ │ - crs = { │ │ │ │ │ - type: "name", │ │ │ │ │ - properties: { │ │ │ │ │ - name: "urn:ogc:def:crs:OGC:1.3:CRS84" │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + drawPoint: function(geometry, style, featureId) { │ │ │ │ │ + if (style.graphic !== false) { │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + this.drawExternalGraphic(geometry, style, featureId) │ │ │ │ │ + } else if (style.graphicName && style.graphicName != "circle") { │ │ │ │ │ + this.drawNamedSymbol(geometry, style, featureId) │ │ │ │ │ } else { │ │ │ │ │ - crs = { │ │ │ │ │ - type: "name", │ │ │ │ │ - properties: { │ │ │ │ │ - name: "EPSG:" + code │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ + if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ + var twoPi = Math.PI * 2; │ │ │ │ │ + var radius = style.pointRadius; │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.canvas.fill(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.hitContext.fill() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.canvas.stroke(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.hitContext.stroke() │ │ │ │ │ + } │ │ │ │ │ + this.setCanvasStyle("reset") │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return crs │ │ │ │ │ }, │ │ │ │ │ - extract: { │ │ │ │ │ - feature: function(feature) { │ │ │ │ │ - var geom = this.extract.geometry.apply(this, [feature.geometry]); │ │ │ │ │ - var json = { │ │ │ │ │ - type: "Feature", │ │ │ │ │ - properties: feature.attributes, │ │ │ │ │ - geometry: geom │ │ │ │ │ - }; │ │ │ │ │ - if (feature.fid != null) { │ │ │ │ │ - json.id = feature.fid │ │ │ │ │ + drawLineString: function(geometry, style, featureId) { │ │ │ │ │ + style = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + fill: false │ │ │ │ │ + }, style); │ │ │ │ │ + this.drawLinearRing(geometry, style, featureId) │ │ │ │ │ + }, │ │ │ │ │ + drawLinearRing: function(geometry, style, featureId) { │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.renderPath(this.canvas, geometry, style, featureId, "fill"); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.renderPath(this.hitContext, geometry, style, featureId, "fill") │ │ │ │ │ } │ │ │ │ │ - return json │ │ │ │ │ - }, │ │ │ │ │ - geometry: function(geometry) { │ │ │ │ │ - if (geometry == null) { │ │ │ │ │ - return null │ │ │ │ │ + } │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.renderPath(this.canvas, geometry, style, featureId, "stroke"); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ + this.renderPath(this.hitContext, geometry, style, featureId, "stroke") │ │ │ │ │ } │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry = geometry.clone(); │ │ │ │ │ - geometry.transform(this.internalProjection, this.externalProjection) │ │ │ │ │ + } │ │ │ │ │ + this.setCanvasStyle("reset") │ │ │ │ │ + }, │ │ │ │ │ + renderPath: function(context, geometry, style, featureId, type) { │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + context.beginPath(); │ │ │ │ │ + var start = this.getLocalXY(components[0]); │ │ │ │ │ + var x = start[0]; │ │ │ │ │ + var y = start[1]; │ │ │ │ │ + if (!isNaN(x) && !isNaN(y)) { │ │ │ │ │ + context.moveTo(start[0], start[1]); │ │ │ │ │ + for (var i = 1; i < len; ++i) { │ │ │ │ │ + var pt = this.getLocalXY(components[i]); │ │ │ │ │ + context.lineTo(pt[0], pt[1]) │ │ │ │ │ } │ │ │ │ │ - var geometryType = geometry.CLASS_NAME.split(".")[2]; │ │ │ │ │ - var data = this.extract[geometryType.toLowerCase()].apply(this, [geometry]); │ │ │ │ │ - var json; │ │ │ │ │ - if (geometryType == "Collection") { │ │ │ │ │ - json = { │ │ │ │ │ - type: "GeometryCollection", │ │ │ │ │ - geometries: data │ │ │ │ │ - } │ │ │ │ │ + if (type === "fill") { │ │ │ │ │ + context.fill() │ │ │ │ │ } else { │ │ │ │ │ - json = { │ │ │ │ │ - type: geometryType, │ │ │ │ │ - coordinates: data │ │ │ │ │ - } │ │ │ │ │ + context.stroke() │ │ │ │ │ } │ │ │ │ │ - return json │ │ │ │ │ - }, │ │ │ │ │ - point: function(point) { │ │ │ │ │ - return [point.x, point.y] │ │ │ │ │ - }, │ │ │ │ │ - multipoint: function(multipoint) { │ │ │ │ │ - var array = []; │ │ │ │ │ - for (var i = 0, len = multipoint.components.length; i < len; ++i) { │ │ │ │ │ - array.push(this.extract.point.apply(this, [multipoint.components[i]])) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + drawPolygon: function(geometry, style, featureId) { │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + this.drawLinearRing(components[0], style, featureId); │ │ │ │ │ + for (var i = 1; i < len; ++i) { │ │ │ │ │ + this.canvas.globalCompositeOperation = "destination-out"; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.globalCompositeOperation = "destination-out" │ │ │ │ │ } │ │ │ │ │ - return array │ │ │ │ │ - }, │ │ │ │ │ - linestring: function(linestring) { │ │ │ │ │ - var array = []; │ │ │ │ │ - for (var i = 0, len = linestring.components.length; i < len; ++i) { │ │ │ │ │ - array.push(this.extract.point.apply(this, [linestring.components[i]])) │ │ │ │ │ + this.drawLinearRing(components[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + stroke: false, │ │ │ │ │ + fillOpacity: 1 │ │ │ │ │ + }, style), featureId); │ │ │ │ │ + this.canvas.globalCompositeOperation = "source-over"; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.globalCompositeOperation = "source-over" │ │ │ │ │ } │ │ │ │ │ - return array │ │ │ │ │ - }, │ │ │ │ │ - multilinestring: function(multilinestring) { │ │ │ │ │ - var array = []; │ │ │ │ │ - for (var i = 0, len = multilinestring.components.length; i < len; ++i) { │ │ │ │ │ - array.push(this.extract.linestring.apply(this, [multilinestring.components[i]])) │ │ │ │ │ + this.drawLinearRing(components[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + fill: false │ │ │ │ │ + }, style), featureId) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + drawText: function(location, style) { │ │ │ │ │ + var pt = this.getLocalXY(location); │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ + this.canvas.fillStyle = style.fontColor; │ │ │ │ │ + this.canvas.globalAlpha = style.fontOpacity || 1; │ │ │ │ │ + var fontStyle = [style.fontStyle ? style.fontStyle : "normal", "normal", style.fontWeight ? style.fontWeight : "normal", style.fontSize ? style.fontSize : "1em", style.fontFamily ? style.fontFamily : "sans-serif"].join(" "); │ │ │ │ │ + var labelRows = style.label.split("\n"); │ │ │ │ │ + var numRows = labelRows.length; │ │ │ │ │ + if (this.canvas.fillText) { │ │ │ │ │ + this.canvas.font = fontStyle; │ │ │ │ │ + this.canvas.textAlign = OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[0]] || "center"; │ │ │ │ │ + this.canvas.textBaseline = OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[1]] || "middle"; │ │ │ │ │ + var vfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5 │ │ │ │ │ } │ │ │ │ │ - return array │ │ │ │ │ - }, │ │ │ │ │ - polygon: function(polygon) { │ │ │ │ │ - var array = []; │ │ │ │ │ - for (var i = 0, len = polygon.components.length; i < len; ++i) { │ │ │ │ │ - array.push(this.extract.linestring.apply(this, [polygon.components[i]])) │ │ │ │ │ + var lineHeight = this.canvas.measureText("Mg").height || this.canvas.measureText("xx").width; │ │ │ │ │ + pt[1] += lineHeight * vfactor * (numRows - 1); │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + if (style.labelOutlineWidth) { │ │ │ │ │ + this.canvas.save(); │ │ │ │ │ + this.canvas.globalAlpha = style.labelOutlineOpacity || style.fontOpacity || 1; │ │ │ │ │ + this.canvas.strokeStyle = style.labelOutlineColor; │ │ │ │ │ + this.canvas.lineWidth = style.labelOutlineWidth; │ │ │ │ │ + this.canvas.strokeText(labelRows[i], pt[0], pt[1] + lineHeight * i + 1); │ │ │ │ │ + this.canvas.restore() │ │ │ │ │ + } │ │ │ │ │ + this.canvas.fillText(labelRows[i], pt[0], pt[1] + lineHeight * i) │ │ │ │ │ } │ │ │ │ │ - return array │ │ │ │ │ - }, │ │ │ │ │ - multipolygon: function(multipolygon) { │ │ │ │ │ - var array = []; │ │ │ │ │ - for (var i = 0, len = multipolygon.components.length; i < len; ++i) { │ │ │ │ │ - array.push(this.extract.polygon.apply(this, [multipolygon.components[i]])) │ │ │ │ │ + } else if (this.canvas.mozDrawText) { │ │ │ │ │ + this.canvas.mozTextStyle = fontStyle; │ │ │ │ │ + var hfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[0]]; │ │ │ │ │ + if (hfactor == null) { │ │ │ │ │ + hfactor = -.5 │ │ │ │ │ } │ │ │ │ │ - return array │ │ │ │ │ - }, │ │ │ │ │ - collection: function(collection) { │ │ │ │ │ - var len = collection.components.length; │ │ │ │ │ - var array = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - array[i] = this.extract.geometry.apply(this, [collection.components[i]]) │ │ │ │ │ + var vfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5 │ │ │ │ │ + } │ │ │ │ │ + var lineHeight = this.canvas.mozMeasureText("xx"); │ │ │ │ │ + pt[1] += lineHeight * (1 + vfactor * numRows); │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + var x = pt[0] + hfactor * this.canvas.mozMeasureText(labelRows[i]); │ │ │ │ │ + var y = pt[1] + i * lineHeight; │ │ │ │ │ + this.canvas.translate(x, y); │ │ │ │ │ + this.canvas.mozDrawText(labelRows[i]); │ │ │ │ │ + this.canvas.translate(-x, -y) │ │ │ │ │ } │ │ │ │ │ - return array │ │ │ │ │ } │ │ │ │ │ + this.setCanvasStyle("reset") │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.GeoJSON" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Date = { │ │ │ │ │ - dateRegEx: /^(?:(\d{4})(?:-(\d{2})(?:-(\d{2}))?)?)?(?:(?:T(\d{1,2}):(\d{2}):(\d{2}(?:\.\d+)?)(Z|(?:[+-]\d{1,2}(?::(\d{2}))?)))|Z)?$/, │ │ │ │ │ - toISOString: function() { │ │ │ │ │ - if ("toISOString" in Date.prototype) { │ │ │ │ │ - return function(date) { │ │ │ │ │ - return date.toISOString() │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - return function(date) { │ │ │ │ │ - var str; │ │ │ │ │ - if (isNaN(date.getTime())) { │ │ │ │ │ - str = "Invalid Date" │ │ │ │ │ - } else { │ │ │ │ │ - str = date.getUTCFullYear() + "-" + OpenLayers.Number.zeroPad(date.getUTCMonth() + 1, 2) + "-" + OpenLayers.Number.zeroPad(date.getUTCDate(), 2) + "T" + OpenLayers.Number.zeroPad(date.getUTCHours(), 2) + ":" + OpenLayers.Number.zeroPad(date.getUTCMinutes(), 2) + ":" + OpenLayers.Number.zeroPad(date.getUTCSeconds(), 2) + "." + OpenLayers.Number.zeroPad(date.getUTCMilliseconds(), 3) + "Z" │ │ │ │ │ - } │ │ │ │ │ - return str │ │ │ │ │ - } │ │ │ │ │ + getLocalXY: function(point) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var extent = this.extent; │ │ │ │ │ + var x = (point.x - this.featureDx) / resolution + -extent.left / resolution; │ │ │ │ │ + var y = extent.top / resolution - point.y / resolution; │ │ │ │ │ + return [x, y] │ │ │ │ │ + }, │ │ │ │ │ + clear: function() { │ │ │ │ │ + var height = this.root.height; │ │ │ │ │ + var width = this.root.width; │ │ │ │ │ + this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ + this.features = {}; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.clearRect(0, 0, width, height) │ │ │ │ │ } │ │ │ │ │ - }(), │ │ │ │ │ - parse: function(str) { │ │ │ │ │ - var date; │ │ │ │ │ - var match = str.match(this.dateRegEx); │ │ │ │ │ - if (match && (match[1] || match[7])) { │ │ │ │ │ - var year = parseInt(match[1], 10) || 0; │ │ │ │ │ - var month = parseInt(match[2], 10) - 1 || 0; │ │ │ │ │ - var day = parseInt(match[3], 10) || 1; │ │ │ │ │ - date = new Date(Date.UTC(year, month, day)); │ │ │ │ │ - var type = match[7]; │ │ │ │ │ - if (type) { │ │ │ │ │ - var hours = parseInt(match[4], 10); │ │ │ │ │ - var minutes = parseInt(match[5], 10); │ │ │ │ │ - var secFrac = parseFloat(match[6]); │ │ │ │ │ - var seconds = secFrac | 0; │ │ │ │ │ - var milliseconds = Math.round(1e3 * (secFrac - seconds)); │ │ │ │ │ - date.setUTCHours(hours, minutes, seconds, milliseconds); │ │ │ │ │ - if (type !== "Z") { │ │ │ │ │ - var hoursOffset = parseInt(type, 10); │ │ │ │ │ - var minutesOffset = parseInt(match[8], 10) || 0; │ │ │ │ │ - var offset = -1e3 * (60 * (hoursOffset * 60) + minutesOffset * 60); │ │ │ │ │ - date = new Date(date.getTime() + offset) │ │ │ │ │ + }, │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var featureId, feature; │ │ │ │ │ + if (this.hitDetection && this.root.style.display !== "none") { │ │ │ │ │ + if (!this.map.dragging) { │ │ │ │ │ + var xy = evt.xy; │ │ │ │ │ + var x = xy.x | 0; │ │ │ │ │ + var y = xy.y | 0; │ │ │ │ │ + var data = this.hitContext.getImageData(x, y, 1, 1).data; │ │ │ │ │ + if (data[3] === 255) { │ │ │ │ │ + var id = data[2] + 256 * (data[1] + 256 * data[0]); │ │ │ │ │ + if (id) { │ │ │ │ │ + featureId = "OpenLayers_Feature_Vector_" + (id - 1 + this.hitOverflow); │ │ │ │ │ + try { │ │ │ │ │ + feature = this.features[featureId][0] │ │ │ │ │ + } catch (err) {} │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - date = new Date("invalid") │ │ │ │ │ } │ │ │ │ │ - return date │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Format.XML = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ - namespaces: null, │ │ │ │ │ - namespaceAlias: null, │ │ │ │ │ - defaultPrefix: null, │ │ │ │ │ - readers: {}, │ │ │ │ │ - writers: {}, │ │ │ │ │ - xmldom: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - if (window.ActiveXObject) { │ │ │ │ │ - this.xmldom = new ActiveXObject("Microsoft.XMLDOM") │ │ │ │ │ + return feature │ │ │ │ │ + }, │ │ │ │ │ + eraseFeatures: function(features) { │ │ │ │ │ + if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ + features = [features] │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Format.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.namespaces = OpenLayers.Util.extend({}, this.namespaces); │ │ │ │ │ - this.namespaceAlias = {}; │ │ │ │ │ - for (var alias in this.namespaces) { │ │ │ │ │ - this.namespaceAlias[this.namespaces[alias]] = alias │ │ │ │ │ + for (var i = 0; i < features.length; ++i) { │ │ │ │ │ + delete this.features[features[i].id] │ │ │ │ │ } │ │ │ │ │ + this.redraw() │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.xmldom = null; │ │ │ │ │ - OpenLayers.Format.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - setNamespace: function(alias, uri) { │ │ │ │ │ - this.namespaces[alias] = uri; │ │ │ │ │ - this.namespaceAlias[uri] = alias │ │ │ │ │ - }, │ │ │ │ │ - read: function(text) { │ │ │ │ │ - var index = text.indexOf("<"); │ │ │ │ │ - if (index > 0) { │ │ │ │ │ - text = text.substring(index) │ │ │ │ │ - } │ │ │ │ │ - var node = OpenLayers.Util.Try(OpenLayers.Function.bind(function() { │ │ │ │ │ - var xmldom; │ │ │ │ │ - if (window.ActiveXObject && !this.xmldom) { │ │ │ │ │ - xmldom = new ActiveXObject("Microsoft.XMLDOM") │ │ │ │ │ - } else { │ │ │ │ │ - xmldom = this.xmldom │ │ │ │ │ + redraw: function() { │ │ │ │ │ + if (!this.locked) { │ │ │ │ │ + var height = this.root.height; │ │ │ │ │ + var width = this.root.width; │ │ │ │ │ + this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.clearRect(0, 0, width, height) │ │ │ │ │ } │ │ │ │ │ - xmldom.loadXML(text); │ │ │ │ │ - return xmldom │ │ │ │ │ - }, this), function() { │ │ │ │ │ - return (new DOMParser).parseFromString(text, "text/xml") │ │ │ │ │ - }, function() { │ │ │ │ │ - var req = new XMLHttpRequest; │ │ │ │ │ - req.open("GET", "data:" + "text/xml" + ";charset=utf-8," + encodeURIComponent(text), false); │ │ │ │ │ - if (req.overrideMimeType) { │ │ │ │ │ - req.overrideMimeType("text/xml") │ │ │ │ │ + var labelMap = []; │ │ │ │ │ + var feature, geometry, style; │ │ │ │ │ + var worldBounds = this.map.baseLayer && this.map.baseLayer.wrapDateLine && this.map.getMaxExtent(); │ │ │ │ │ + for (var id in this.features) { │ │ │ │ │ + if (!this.features.hasOwnProperty(id)) { │ │ │ │ │ + continue │ │ │ │ │ + } │ │ │ │ │ + feature = this.features[id][0]; │ │ │ │ │ + geometry = feature.geometry; │ │ │ │ │ + this.calculateFeatureDx(geometry.getBounds(), worldBounds); │ │ │ │ │ + style = this.features[id][1]; │ │ │ │ │ + this.drawGeometry(geometry, style, feature.id); │ │ │ │ │ + if (style.label) { │ │ │ │ │ + labelMap.push([feature, style]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var item; │ │ │ │ │ + for (var i = 0, len = labelMap.length; i < len; ++i) { │ │ │ │ │ + item = labelMap[i]; │ │ │ │ │ + this.drawText(item[0].geometry.getCentroid(), item[1]) │ │ │ │ │ } │ │ │ │ │ - req.send(null); │ │ │ │ │ - return req.responseXML │ │ │ │ │ - }); │ │ │ │ │ - if (this.keepData) { │ │ │ │ │ - this.data = node │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ }, │ │ │ │ │ - write: function(node) { │ │ │ │ │ - var data; │ │ │ │ │ - if (this.xmldom) { │ │ │ │ │ - data = node.xml │ │ │ │ │ - } else { │ │ │ │ │ - var serializer = new XMLSerializer; │ │ │ │ │ - if (node.nodeType == 1) { │ │ │ │ │ - var doc = document.implementation.createDocument("", "", null); │ │ │ │ │ - if (doc.importNode) { │ │ │ │ │ - node = doc.importNode(node, true) │ │ │ │ │ - } │ │ │ │ │ - doc.appendChild(node); │ │ │ │ │ - data = serializer.serializeToString(doc) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.Canvas" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Renderer.Canvas.LABEL_ALIGN = { │ │ │ │ │ + l: "left", │ │ │ │ │ + r: "right", │ │ │ │ │ + t: "top", │ │ │ │ │ + b: "bottom" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.Canvas.LABEL_FACTOR = { │ │ │ │ │ + l: 0, │ │ │ │ │ + r: -1, │ │ │ │ │ + t: 0, │ │ │ │ │ + b: -1 │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.Canvas.drawImageScaleFactor = null; │ │ │ │ │ +OpenLayers.ElementsIndexer = OpenLayers.Class({ │ │ │ │ │ + maxZIndex: null, │ │ │ │ │ + order: null, │ │ │ │ │ + indices: null, │ │ │ │ │ + compare: null, │ │ │ │ │ + initialize: function(yOrdering) { │ │ │ │ │ + this.compare = yOrdering ? OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_Y_ORDER : OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_DRAWING_ORDER; │ │ │ │ │ + this.clear() │ │ │ │ │ + }, │ │ │ │ │ + insert: function(newNode) { │ │ │ │ │ + if (this.exists(newNode)) { │ │ │ │ │ + this.remove(newNode) │ │ │ │ │ + } │ │ │ │ │ + var nodeId = newNode.id; │ │ │ │ │ + this.determineZIndex(newNode); │ │ │ │ │ + var leftIndex = -1; │ │ │ │ │ + var rightIndex = this.order.length; │ │ │ │ │ + var middle; │ │ │ │ │ + while (rightIndex - leftIndex > 1) { │ │ │ │ │ + middle = parseInt((leftIndex + rightIndex) / 2); │ │ │ │ │ + var placement = this.compare(this, newNode, OpenLayers.Util.getElement(this.order[middle])); │ │ │ │ │ + if (placement > 0) { │ │ │ │ │ + leftIndex = middle │ │ │ │ │ } else { │ │ │ │ │ - data = serializer.serializeToString(node) │ │ │ │ │ + rightIndex = middle │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return data │ │ │ │ │ + this.order.splice(rightIndex, 0, nodeId); │ │ │ │ │ + this.indices[nodeId] = this.getZIndex(newNode); │ │ │ │ │ + return this.getNextElement(rightIndex) │ │ │ │ │ }, │ │ │ │ │ - createElementNS: function(uri, name) { │ │ │ │ │ - var element; │ │ │ │ │ - if (this.xmldom) { │ │ │ │ │ - if (typeof uri == "string") { │ │ │ │ │ - element = this.xmldom.createNode(1, name, uri) │ │ │ │ │ + remove: function(node) { │ │ │ │ │ + var nodeId = node.id; │ │ │ │ │ + var arrayIndex = OpenLayers.Util.indexOf(this.order, nodeId); │ │ │ │ │ + if (arrayIndex >= 0) { │ │ │ │ │ + this.order.splice(arrayIndex, 1); │ │ │ │ │ + delete this.indices[nodeId]; │ │ │ │ │ + if (this.order.length > 0) { │ │ │ │ │ + var lastId = this.order[this.order.length - 1]; │ │ │ │ │ + this.maxZIndex = this.indices[lastId] │ │ │ │ │ } else { │ │ │ │ │ - element = this.xmldom.createNode(1, name, "") │ │ │ │ │ + this.maxZIndex = 0 │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - element = document.createElementNS(uri, name) │ │ │ │ │ } │ │ │ │ │ - return element │ │ │ │ │ }, │ │ │ │ │ - createDocumentFragment: function() { │ │ │ │ │ - var element; │ │ │ │ │ - if (this.xmldom) { │ │ │ │ │ - element = this.xmldom.createDocumentFragment() │ │ │ │ │ - } else { │ │ │ │ │ - element = document.createDocumentFragment() │ │ │ │ │ - } │ │ │ │ │ - return element │ │ │ │ │ + clear: function() { │ │ │ │ │ + this.order = []; │ │ │ │ │ + this.indices = {}; │ │ │ │ │ + this.maxZIndex = 0 │ │ │ │ │ }, │ │ │ │ │ - createTextNode: function(text) { │ │ │ │ │ - var node; │ │ │ │ │ - if (typeof text !== "string") { │ │ │ │ │ - text = String(text) │ │ │ │ │ + exists: function(node) { │ │ │ │ │ + return this.indices[node.id] != null │ │ │ │ │ + }, │ │ │ │ │ + getZIndex: function(node) { │ │ │ │ │ + return node._style.graphicZIndex │ │ │ │ │ + }, │ │ │ │ │ + determineZIndex: function(node) { │ │ │ │ │ + var zIndex = node._style.graphicZIndex; │ │ │ │ │ + if (zIndex == null) { │ │ │ │ │ + zIndex = this.maxZIndex; │ │ │ │ │ + node._style.graphicZIndex = zIndex │ │ │ │ │ + } else if (zIndex > this.maxZIndex) { │ │ │ │ │ + this.maxZIndex = zIndex │ │ │ │ │ } │ │ │ │ │ - if (this.xmldom) { │ │ │ │ │ - node = this.xmldom.createTextNode(text) │ │ │ │ │ + }, │ │ │ │ │ + getNextElement: function(index) { │ │ │ │ │ + var nextIndex = index + 1; │ │ │ │ │ + if (nextIndex < this.order.length) { │ │ │ │ │ + var nextElement = OpenLayers.Util.getElement(this.order[nextIndex]); │ │ │ │ │ + if (nextElement == undefined) { │ │ │ │ │ + nextElement = this.getNextElement(nextIndex) │ │ │ │ │ + } │ │ │ │ │ + return nextElement │ │ │ │ │ } else { │ │ │ │ │ - node = document.createTextNode(text) │ │ │ │ │ + return null │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ }, │ │ │ │ │ - getElementsByTagNameNS: function(node, uri, name) { │ │ │ │ │ - var elements = []; │ │ │ │ │ - if (node.getElementsByTagNameNS) { │ │ │ │ │ - elements = node.getElementsByTagNameNS(uri, name) │ │ │ │ │ - } else { │ │ │ │ │ - var allNodes = node.getElementsByTagName("*"); │ │ │ │ │ - var potentialNode, fullName; │ │ │ │ │ - for (var i = 0, len = allNodes.length; i < len; ++i) { │ │ │ │ │ - potentialNode = allNodes[i]; │ │ │ │ │ - fullName = potentialNode.prefix ? potentialNode.prefix + ":" + name : name; │ │ │ │ │ - if (name == "*" || fullName == potentialNode.nodeName) { │ │ │ │ │ - if (uri == "*" || uri == potentialNode.namespaceURI) { │ │ │ │ │ - elements.push(potentialNode) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + CLASS_NAME: "OpenLayers.ElementsIndexer" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.ElementsIndexer.IndexingMethods = { │ │ │ │ │ + Z_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var newZIndex = indexer.getZIndex(newNode); │ │ │ │ │ + var returnVal = 0; │ │ │ │ │ + if (nextNode) { │ │ │ │ │ + var nextZIndex = indexer.getZIndex(nextNode); │ │ │ │ │ + returnVal = newZIndex - nextZIndex │ │ │ │ │ } │ │ │ │ │ - return elements │ │ │ │ │ + return returnVal │ │ │ │ │ }, │ │ │ │ │ - getAttributeNodeNS: function(node, uri, name) { │ │ │ │ │ - var attributeNode = null; │ │ │ │ │ - if (node.getAttributeNodeNS) { │ │ │ │ │ - attributeNode = node.getAttributeNodeNS(uri, name) │ │ │ │ │ - } else { │ │ │ │ │ - var attributes = node.attributes; │ │ │ │ │ - var potentialNode, fullName; │ │ │ │ │ - for (var i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ - potentialNode = attributes[i]; │ │ │ │ │ - if (potentialNode.namespaceURI == uri) { │ │ │ │ │ - fullName = potentialNode.prefix ? potentialNode.prefix + ":" + name : name; │ │ │ │ │ - if (fullName == potentialNode.nodeName) { │ │ │ │ │ - attributeNode = potentialNode; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + Z_ORDER_DRAWING_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER(indexer, newNode, nextNode); │ │ │ │ │ + if (nextNode && returnVal == 0) { │ │ │ │ │ + returnVal = 1 │ │ │ │ │ } │ │ │ │ │ - return attributeNode │ │ │ │ │ + return returnVal │ │ │ │ │ }, │ │ │ │ │ - getAttributeNS: function(node, uri, name) { │ │ │ │ │ - var attributeValue = ""; │ │ │ │ │ - if (node.getAttributeNS) { │ │ │ │ │ - attributeValue = node.getAttributeNS(uri, name) || "" │ │ │ │ │ - } else { │ │ │ │ │ - var attributeNode = this.getAttributeNodeNS(node, uri, name); │ │ │ │ │ - if (attributeNode) { │ │ │ │ │ - attributeValue = attributeNode.nodeValue │ │ │ │ │ - } │ │ │ │ │ + Z_ORDER_Y_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ + var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER(indexer, newNode, nextNode); │ │ │ │ │ + if (nextNode && returnVal === 0) { │ │ │ │ │ + var result = nextNode._boundsBottom - newNode._boundsBottom; │ │ │ │ │ + returnVal = result === 0 ? 1 : result │ │ │ │ │ + } │ │ │ │ │ + return returnVal │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.Elements = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ + rendererRoot: null, │ │ │ │ │ + root: null, │ │ │ │ │ + vectorRoot: null, │ │ │ │ │ + textRoot: null, │ │ │ │ │ + xmlns: null, │ │ │ │ │ + xOffset: 0, │ │ │ │ │ + indexer: null, │ │ │ │ │ + BACKGROUND_ID_SUFFIX: "_background", │ │ │ │ │ + LABEL_ID_SUFFIX: "_label", │ │ │ │ │ + LABEL_OUTLINE_SUFFIX: "_outline", │ │ │ │ │ + initialize: function(containerID, options) { │ │ │ │ │ + OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.rendererRoot = this.createRenderRoot(); │ │ │ │ │ + this.root = this.createRoot("_root"); │ │ │ │ │ + this.vectorRoot = this.createRoot("_vroot"); │ │ │ │ │ + this.textRoot = this.createRoot("_troot"); │ │ │ │ │ + this.root.appendChild(this.vectorRoot); │ │ │ │ │ + this.root.appendChild(this.textRoot); │ │ │ │ │ + this.rendererRoot.appendChild(this.root); │ │ │ │ │ + this.container.appendChild(this.rendererRoot); │ │ │ │ │ + if (options && (options.zIndexing || options.yOrdering)) { │ │ │ │ │ + this.indexer = new OpenLayers.ElementsIndexer(options.yOrdering) │ │ │ │ │ } │ │ │ │ │ - return attributeValue │ │ │ │ │ }, │ │ │ │ │ - getChildValue: function(node, def) { │ │ │ │ │ - var value = def || ""; │ │ │ │ │ - if (node) { │ │ │ │ │ - for (var child = node.firstChild; child; child = child.nextSibling) { │ │ │ │ │ - switch (child.nodeType) { │ │ │ │ │ - case 3: │ │ │ │ │ - case 4: │ │ │ │ │ - value += child.nodeValue │ │ │ │ │ - } │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.clear(); │ │ │ │ │ + this.rendererRoot = null; │ │ │ │ │ + this.root = null; │ │ │ │ │ + this.xmlns = null; │ │ │ │ │ + OpenLayers.Renderer.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + clear: function() { │ │ │ │ │ + var child; │ │ │ │ │ + var root = this.vectorRoot; │ │ │ │ │ + if (root) { │ │ │ │ │ + while (child = root.firstChild) { │ │ │ │ │ + root.removeChild(child) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return value │ │ │ │ │ - }, │ │ │ │ │ - isSimpleContent: function(node) { │ │ │ │ │ - var simple = true; │ │ │ │ │ - for (var child = node.firstChild; child; child = child.nextSibling) { │ │ │ │ │ - if (child.nodeType === 1) { │ │ │ │ │ - simple = false; │ │ │ │ │ - break │ │ │ │ │ + root = this.textRoot; │ │ │ │ │ + if (root) { │ │ │ │ │ + while (child = root.firstChild) { │ │ │ │ │ + root.removeChild(child) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return simple │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + this.indexer.clear() │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - contentType: function(node) { │ │ │ │ │ - var simple = false, │ │ │ │ │ - complex = false; │ │ │ │ │ - var type = OpenLayers.Format.XML.CONTENT_TYPE.EMPTY; │ │ │ │ │ - for (var child = node.firstChild; child; child = child.nextSibling) { │ │ │ │ │ - switch (child.nodeType) { │ │ │ │ │ - case 1: │ │ │ │ │ - complex = true; │ │ │ │ │ - break; │ │ │ │ │ - case 8: │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - simple = true │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + var rightOfDateLine, ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ + extent = extent.scale(1 / ratio), │ │ │ │ │ + world = this.map.getMaxExtent(); │ │ │ │ │ + if (world.right > extent.left && world.right < extent.right) { │ │ │ │ │ + rightOfDateLine = true │ │ │ │ │ + } else if (world.left > extent.left && world.left < extent.right) { │ │ │ │ │ + rightOfDateLine = false │ │ │ │ │ } │ │ │ │ │ - if (complex && simple) { │ │ │ │ │ - break │ │ │ │ │ + if (rightOfDateLine !== this.rightOfDateLine || resolutionChanged) { │ │ │ │ │ + coordSysUnchanged = false; │ │ │ │ │ + this.xOffset = rightOfDateLine === true ? world.getWidth() / resolution : 0 │ │ │ │ │ } │ │ │ │ │ + this.rightOfDateLine = rightOfDateLine │ │ │ │ │ } │ │ │ │ │ - if (complex && simple) { │ │ │ │ │ - type = OpenLayers.Format.XML.CONTENT_TYPE.MIXED │ │ │ │ │ - } else if (complex) { │ │ │ │ │ - return OpenLayers.Format.XML.CONTENT_TYPE.COMPLEX │ │ │ │ │ - } else if (simple) { │ │ │ │ │ - return OpenLayers.Format.XML.CONTENT_TYPE.SIMPLE │ │ │ │ │ - } │ │ │ │ │ - return type │ │ │ │ │ + return coordSysUnchanged │ │ │ │ │ }, │ │ │ │ │ - hasAttributeNS: function(node, uri, name) { │ │ │ │ │ - var found = false; │ │ │ │ │ - if (node.hasAttributeNS) { │ │ │ │ │ - found = node.hasAttributeNS(uri, name) │ │ │ │ │ - } else { │ │ │ │ │ - found = !!this.getAttributeNodeNS(node, uri, name) │ │ │ │ │ + getNodeType: function(geometry, style) {}, │ │ │ │ │ + drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ + var className = geometry.CLASS_NAME; │ │ │ │ │ + var rendered = true; │ │ │ │ │ + if (className == "OpenLayers.Geometry.Collection" || className == "OpenLayers.Geometry.MultiPoint" || className == "OpenLayers.Geometry.MultiLineString" || className == "OpenLayers.Geometry.MultiPolygon") { │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + rendered = this.drawGeometry(geometry.components[i], style, featureId) && rendered │ │ │ │ │ + } │ │ │ │ │ + return rendered │ │ │ │ │ } │ │ │ │ │ - return found │ │ │ │ │ - }, │ │ │ │ │ - setAttributeNS: function(node, uri, name, value) { │ │ │ │ │ - if (node.setAttributeNS) { │ │ │ │ │ - node.setAttributeNS(uri, name, value) │ │ │ │ │ - } else { │ │ │ │ │ - if (this.xmldom) { │ │ │ │ │ - if (uri) { │ │ │ │ │ - var attribute = node.ownerDocument.createNode(2, name, uri); │ │ │ │ │ - attribute.nodeValue = value; │ │ │ │ │ - node.setAttributeNode(attribute) │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttribute(name, value) │ │ │ │ │ - } │ │ │ │ │ + rendered = false; │ │ │ │ │ + var removeBackground = false; │ │ │ │ │ + if (style.display != "none") { │ │ │ │ │ + if (style.backgroundGraphic) { │ │ │ │ │ + this.redrawBackgroundNode(geometry.id, geometry, style, featureId) │ │ │ │ │ } else { │ │ │ │ │ - throw "setAttributeNS not implemented" │ │ │ │ │ + removeBackground = true │ │ │ │ │ } │ │ │ │ │ + rendered = this.redrawNode(geometry.id, geometry, style, featureId) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - createElementNSPlus: function(name, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var uri = options.uri || this.namespaces[options.prefix]; │ │ │ │ │ - if (!uri) { │ │ │ │ │ - var loc = name.indexOf(":"); │ │ │ │ │ - uri = this.namespaces[name.substring(0, loc)] │ │ │ │ │ - } │ │ │ │ │ - if (!uri) { │ │ │ │ │ - uri = this.namespaces[this.defaultPrefix] │ │ │ │ │ - } │ │ │ │ │ - var node = this.createElementNS(uri, name); │ │ │ │ │ - if (options.attributes) { │ │ │ │ │ - this.setAttributes(node, options.attributes) │ │ │ │ │ - } │ │ │ │ │ - var value = options.value; │ │ │ │ │ - if (value != null) { │ │ │ │ │ - node.appendChild(this.createTextNode(value)) │ │ │ │ │ + if (rendered == false) { │ │ │ │ │ + var node = document.getElementById(geometry.id); │ │ │ │ │ + if (node) { │ │ │ │ │ + if (node._style.backgroundGraphic) { │ │ │ │ │ + removeBackground = true │ │ │ │ │ + } │ │ │ │ │ + node.parentNode.removeChild(node) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - setAttributes: function(node, obj) { │ │ │ │ │ - var value, uri; │ │ │ │ │ - for (var name in obj) { │ │ │ │ │ - if (obj[name] != null && obj[name].toString) { │ │ │ │ │ - value = obj[name].toString(); │ │ │ │ │ - uri = this.namespaces[name.substring(0, name.indexOf(":"))] || null; │ │ │ │ │ - this.setAttributeNS(node, uri, name, value) │ │ │ │ │ + if (removeBackground) { │ │ │ │ │ + var node = document.getElementById(geometry.id + this.BACKGROUND_ID_SUFFIX); │ │ │ │ │ + if (node) { │ │ │ │ │ + node.parentNode.removeChild(node) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return rendered │ │ │ │ │ }, │ │ │ │ │ - readNode: function(node, obj) { │ │ │ │ │ - if (!obj) { │ │ │ │ │ - obj = {} │ │ │ │ │ + redrawNode: function(id, geometry, style, featureId) { │ │ │ │ │ + style = this.applyDefaultSymbolizer(style); │ │ │ │ │ + var node = this.nodeFactory(id, this.getNodeType(geometry, style)); │ │ │ │ │ + node._featureId = featureId; │ │ │ │ │ + node._boundsBottom = geometry.getBounds().bottom; │ │ │ │ │ + node._geometryClass = geometry.CLASS_NAME; │ │ │ │ │ + node._style = style; │ │ │ │ │ + var drawResult = this.drawGeometryNode(node, geometry, style); │ │ │ │ │ + if (drawResult === false) { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - var group = this.readers[node.namespaceURI ? this.namespaceAlias[node.namespaceURI] : this.defaultPrefix]; │ │ │ │ │ - if (group) { │ │ │ │ │ - var local = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ - var reader = group[local] || group["*"]; │ │ │ │ │ - if (reader) { │ │ │ │ │ - reader.apply(this, [node, obj]) │ │ │ │ │ + node = drawResult.node; │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + var insert = this.indexer.insert(node); │ │ │ │ │ + if (insert) { │ │ │ │ │ + this.vectorRoot.insertBefore(node, insert) │ │ │ │ │ + } else { │ │ │ │ │ + this.vectorRoot.appendChild(node) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + if (node.parentNode !== this.vectorRoot) { │ │ │ │ │ + this.vectorRoot.appendChild(node) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return obj │ │ │ │ │ + this.postDraw(node); │ │ │ │ │ + return drawResult.complete │ │ │ │ │ }, │ │ │ │ │ - readChildNodes: function(node, obj) { │ │ │ │ │ - if (!obj) { │ │ │ │ │ - obj = {} │ │ │ │ │ + redrawBackgroundNode: function(id, geometry, style, featureId) { │ │ │ │ │ + var backgroundStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ + backgroundStyle.externalGraphic = backgroundStyle.backgroundGraphic; │ │ │ │ │ + backgroundStyle.graphicXOffset = backgroundStyle.backgroundXOffset; │ │ │ │ │ + backgroundStyle.graphicYOffset = backgroundStyle.backgroundYOffset; │ │ │ │ │ + backgroundStyle.graphicZIndex = backgroundStyle.backgroundGraphicZIndex; │ │ │ │ │ + backgroundStyle.graphicWidth = backgroundStyle.backgroundWidth || backgroundStyle.graphicWidth; │ │ │ │ │ + backgroundStyle.graphicHeight = backgroundStyle.backgroundHeight || backgroundStyle.graphicHeight; │ │ │ │ │ + backgroundStyle.backgroundGraphic = null; │ │ │ │ │ + backgroundStyle.backgroundXOffset = null; │ │ │ │ │ + backgroundStyle.backgroundYOffset = null; │ │ │ │ │ + backgroundStyle.backgroundGraphicZIndex = null; │ │ │ │ │ + return this.redrawNode(id + this.BACKGROUND_ID_SUFFIX, geometry, backgroundStyle, null) │ │ │ │ │ + }, │ │ │ │ │ + drawGeometryNode: function(node, geometry, style) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + var options = { │ │ │ │ │ + isFilled: style.fill === undefined ? true : style.fill, │ │ │ │ │ + isStroked: style.stroke === undefined ? !!style.strokeWidth : style.stroke │ │ │ │ │ + }; │ │ │ │ │ + var drawn; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.graphic === false) { │ │ │ │ │ + options.isFilled = false; │ │ │ │ │ + options.isStroked = false │ │ │ │ │ + } │ │ │ │ │ + drawn = this.drawPoint(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + options.isFilled = false; │ │ │ │ │ + drawn = this.drawLineString(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + drawn = this.drawLinearRing(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + drawn = this.drawPolygon(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + drawn = this.drawRectangle(node, geometry); │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - var child; │ │ │ │ │ - for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ - child = children[i]; │ │ │ │ │ - if (child.nodeType == 1) { │ │ │ │ │ - this.readNode(child, obj) │ │ │ │ │ + node._options = options; │ │ │ │ │ + if (drawn != false) { │ │ │ │ │ + return { │ │ │ │ │ + node: this.setStyle(node, style, options, geometry), │ │ │ │ │ + complete: drawn │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return obj │ │ │ │ │ }, │ │ │ │ │ - writeNode: function(name, obj, parent) { │ │ │ │ │ - var prefix, local; │ │ │ │ │ - var split = name.indexOf(":"); │ │ │ │ │ - if (split > 0) { │ │ │ │ │ - prefix = name.substring(0, split); │ │ │ │ │ - local = name.substring(split + 1) │ │ │ │ │ - } else { │ │ │ │ │ - if (parent) { │ │ │ │ │ - prefix = this.namespaceAlias[parent.namespaceURI] │ │ │ │ │ - } else { │ │ │ │ │ - prefix = this.defaultPrefix │ │ │ │ │ - } │ │ │ │ │ - local = name │ │ │ │ │ + postDraw: function(node) {}, │ │ │ │ │ + drawPoint: function(node, geometry) {}, │ │ │ │ │ + drawLineString: function(node, geometry) {}, │ │ │ │ │ + drawLinearRing: function(node, geometry) {}, │ │ │ │ │ + drawPolygon: function(node, geometry) {}, │ │ │ │ │ + drawRectangle: function(node, geometry) {}, │ │ │ │ │ + drawCircle: function(node, geometry) {}, │ │ │ │ │ + removeText: function(featureId) { │ │ │ │ │ + var label = document.getElementById(featureId + this.LABEL_ID_SUFFIX); │ │ │ │ │ + if (label) { │ │ │ │ │ + this.textRoot.removeChild(label) │ │ │ │ │ } │ │ │ │ │ - var child = this.writers[prefix][local].apply(this, [obj]); │ │ │ │ │ - if (parent) { │ │ │ │ │ - parent.appendChild(child) │ │ │ │ │ + var outline = document.getElementById(featureId + this.LABEL_OUTLINE_SUFFIX); │ │ │ │ │ + if (outline) { │ │ │ │ │ + this.textRoot.removeChild(outline) │ │ │ │ │ } │ │ │ │ │ - return child │ │ │ │ │ - }, │ │ │ │ │ - getChildEl: function(node, name, uri) { │ │ │ │ │ - return node && this.getThisOrNextEl(node.firstChild, name, uri) │ │ │ │ │ }, │ │ │ │ │ - getNextEl: function(node, name, uri) { │ │ │ │ │ - return node && this.getThisOrNextEl(node.nextSibling, name, uri) │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var target = evt.target; │ │ │ │ │ + var useElement = target && target.correspondingUseElement; │ │ │ │ │ + var node = useElement ? useElement : target || evt.srcElement; │ │ │ │ │ + return node._featureId │ │ │ │ │ }, │ │ │ │ │ - getThisOrNextEl: function(node, name, uri) { │ │ │ │ │ - outer: for (var sibling = node; sibling; sibling = sibling.nextSibling) { │ │ │ │ │ - switch (sibling.nodeType) { │ │ │ │ │ - case 1: │ │ │ │ │ - if ((!name || name === (sibling.localName || sibling.nodeName.split(":").pop())) && (!uri || uri === sibling.namespaceURI)) { │ │ │ │ │ - break outer │ │ │ │ │ - } │ │ │ │ │ - sibling = null; │ │ │ │ │ - break outer; │ │ │ │ │ - case 3: │ │ │ │ │ - if (/^\s*$/.test(sibling.nodeValue)) { │ │ │ │ │ - break │ │ │ │ │ + eraseGeometry: function(geometry, featureId) { │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPoint" || geometry.CLASS_NAME == "OpenLayers.Geometry.MultiLineString" || geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPolygon" || geometry.CLASS_NAME == "OpenLayers.Geometry.Collection") { │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + this.eraseGeometry(geometry.components[i], featureId) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + var element = OpenLayers.Util.getElement(geometry.id); │ │ │ │ │ + if (element && element.parentNode) { │ │ │ │ │ + if (element.geometry) { │ │ │ │ │ + element.geometry.destroy(); │ │ │ │ │ + element.geometry = null │ │ │ │ │ + } │ │ │ │ │ + element.parentNode.removeChild(element); │ │ │ │ │ + if (this.indexer) { │ │ │ │ │ + this.indexer.remove(element) │ │ │ │ │ + } │ │ │ │ │ + if (element._style.backgroundGraphic) { │ │ │ │ │ + var backgroundId = geometry.id + this.BACKGROUND_ID_SUFFIX; │ │ │ │ │ + var bElem = OpenLayers.Util.getElement(backgroundId); │ │ │ │ │ + if (bElem && bElem.parentNode) { │ │ │ │ │ + bElem.parentNode.removeChild(bElem) │ │ │ │ │ } │ │ │ │ │ - case 4: │ │ │ │ │ - case 6: │ │ │ │ │ - case 12: │ │ │ │ │ - case 10: │ │ │ │ │ - case 11: │ │ │ │ │ - sibling = null; │ │ │ │ │ - break outer │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return sibling || null │ │ │ │ │ }, │ │ │ │ │ - lookupNamespaceURI: function(node, prefix) { │ │ │ │ │ - var uri = null; │ │ │ │ │ + nodeFactory: function(id, type) { │ │ │ │ │ + var node = OpenLayers.Util.getElement(id); │ │ │ │ │ if (node) { │ │ │ │ │ - if (node.lookupNamespaceURI) { │ │ │ │ │ - uri = node.lookupNamespaceURI(prefix) │ │ │ │ │ + if (!this.nodeTypeCompare(node, type)) { │ │ │ │ │ + node.parentNode.removeChild(node); │ │ │ │ │ + node = this.nodeFactory(id, type) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + node = this.createNode(type, id) │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + nodeTypeCompare: function(node, type) {}, │ │ │ │ │ + createNode: function(type, id) {}, │ │ │ │ │ + moveRoot: function(renderer) { │ │ │ │ │ + var root = this.root; │ │ │ │ │ + if (renderer.root.parentNode == this.rendererRoot) { │ │ │ │ │ + root = renderer.root │ │ │ │ │ + } │ │ │ │ │ + root.parentNode.removeChild(root); │ │ │ │ │ + renderer.rendererRoot.appendChild(root) │ │ │ │ │ + }, │ │ │ │ │ + getRenderLayerId: function() { │ │ │ │ │ + return this.root.parentNode.parentNode.id │ │ │ │ │ + }, │ │ │ │ │ + isComplexSymbol: function(graphicName) { │ │ │ │ │ + return graphicName != "circle" && !!graphicName │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.Elements" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Renderer.SVG = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ + xmlns: "http://www.w3.org/2000/svg", │ │ │ │ │ + xlinkns: "http://www.w3.org/1999/xlink", │ │ │ │ │ + MAX_PIXEL: 15e3, │ │ │ │ │ + translationParameters: null, │ │ │ │ │ + symbolMetrics: null, │ │ │ │ │ + initialize: function(containerID) { │ │ │ │ │ + if (!this.supported()) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.translationParameters = { │ │ │ │ │ + x: 0, │ │ │ │ │ + y: 0 │ │ │ │ │ + }; │ │ │ │ │ + this.symbolMetrics = {} │ │ │ │ │ + }, │ │ │ │ │ + supported: function() { │ │ │ │ │ + var svgFeature = "http://www.w3.org/TR/SVG11/feature#"; │ │ │ │ │ + return document.implementation && (document.implementation.hasFeature("org.w3c.svg", "1.0") || document.implementation.hasFeature(svgFeature + "SVG", "1.1") || document.implementation.hasFeature(svgFeature + "BasicStructure", "1.1")) │ │ │ │ │ + }, │ │ │ │ │ + inValidRange: function(x, y, xyOnly) { │ │ │ │ │ + var left = x + (xyOnly ? 0 : this.translationParameters.x); │ │ │ │ │ + var top = y + (xyOnly ? 0 : this.translationParameters.y); │ │ │ │ │ + return left >= -this.MAX_PIXEL && left <= this.MAX_PIXEL && top >= -this.MAX_PIXEL && top <= this.MAX_PIXEL │ │ │ │ │ + }, │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + var resolution = this.getResolution(), │ │ │ │ │ + left = -extent.left / resolution, │ │ │ │ │ + top = extent.top / resolution; │ │ │ │ │ + if (resolutionChanged) { │ │ │ │ │ + this.left = left; │ │ │ │ │ + this.top = top; │ │ │ │ │ + var extentString = "0 0 " + this.size.w + " " + this.size.h; │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "viewBox", extentString); │ │ │ │ │ + this.translate(this.xOffset, 0); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + var inRange = this.translate(left - this.left + this.xOffset, top - this.top); │ │ │ │ │ + if (!inRange) { │ │ │ │ │ + this.setExtent(extent, true) │ │ │ │ │ + } │ │ │ │ │ + return coordSysUnchanged && inRange │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + translate: function(x, y) { │ │ │ │ │ + if (!this.inValidRange(x, y, true)) { │ │ │ │ │ + return false │ │ │ │ │ + } else { │ │ │ │ │ + var transformString = ""; │ │ │ │ │ + if (x || y) { │ │ │ │ │ + transformString = "translate(" + x + "," + y + ")" │ │ │ │ │ + } │ │ │ │ │ + this.root.setAttributeNS(null, "transform", transformString); │ │ │ │ │ + this.translationParameters = { │ │ │ │ │ + x: x, │ │ │ │ │ + y: y │ │ │ │ │ + }; │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "width", this.size.w); │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "height", this.size.h) │ │ │ │ │ + }, │ │ │ │ │ + getNodeType: function(geometry, style) { │ │ │ │ │ + var nodeType = null; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + nodeType = "image" │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + nodeType = "svg" │ │ │ │ │ + } else { │ │ │ │ │ + nodeType = "circle" │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + nodeType = "rect"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + nodeType = "polyline"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + nodeType = "polygon"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + case "OpenLayers.Geometry.Curve": │ │ │ │ │ + nodeType = "path"; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + return nodeType │ │ │ │ │ + }, │ │ │ │ │ + setStyle: function(node, style, options) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + options = options || node._options; │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + node.setAttributeNS(null, "title", title); │ │ │ │ │ + var titleNode = node.getElementsByTagName("title"); │ │ │ │ │ + if (titleNode.length > 0) { │ │ │ │ │ + titleNode[0].firstChild.textContent = title │ │ │ │ │ } else { │ │ │ │ │ - outer: switch (node.nodeType) { │ │ │ │ │ - case 1: │ │ │ │ │ - if (node.namespaceURI !== null && node.prefix === prefix) { │ │ │ │ │ - uri = node.namespaceURI; │ │ │ │ │ - break outer │ │ │ │ │ - } │ │ │ │ │ - var len = node.attributes.length; │ │ │ │ │ - if (len) { │ │ │ │ │ - var attr; │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - attr = node.attributes[i]; │ │ │ │ │ - if (attr.prefix === "xmlns" && attr.name === "xmlns:" + prefix) { │ │ │ │ │ - uri = attr.value || null; │ │ │ │ │ - break outer │ │ │ │ │ - } else if (attr.name === "xmlns" && prefix === null) { │ │ │ │ │ - uri = attr.value || null; │ │ │ │ │ - break outer │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - uri = this.lookupNamespaceURI(node.parentNode, prefix); │ │ │ │ │ - break outer; │ │ │ │ │ - case 2: │ │ │ │ │ - uri = this.lookupNamespaceURI(node.ownerElement, prefix); │ │ │ │ │ - break outer; │ │ │ │ │ - case 9: │ │ │ │ │ - uri = this.lookupNamespaceURI(node.documentElement, prefix); │ │ │ │ │ - break outer; │ │ │ │ │ - case 6: │ │ │ │ │ - case 12: │ │ │ │ │ - case 10: │ │ │ │ │ - case 11: │ │ │ │ │ - break outer; │ │ │ │ │ - default: │ │ │ │ │ - uri = this.lookupNamespaceURI(node.parentNode, prefix); │ │ │ │ │ - break outer │ │ │ │ │ + var label = this.nodeFactory(null, "title"); │ │ │ │ │ + label.textContent = title; │ │ │ │ │ + node.appendChild(label) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var r = parseFloat(node.getAttributeNS(null, "r")); │ │ │ │ │ + var widthFactor = 1; │ │ │ │ │ + var pos; │ │ │ │ │ + if (node._geometryClass == "OpenLayers.Geometry.Point" && r) { │ │ │ │ │ + node.style.visibility = ""; │ │ │ │ │ + if (style.graphic === false) { │ │ │ │ │ + node.style.visibility = "hidden" │ │ │ │ │ + } else if (style.externalGraphic) { │ │ │ │ │ + pos = this.getPosition(node); │ │ │ │ │ + if (style.graphicWidth && style.graphicHeight) { │ │ │ │ │ + node.setAttributeNS(null, "preserveAspectRatio", "none") │ │ │ │ │ + } │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ + var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ + var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ + node.setAttributeNS(null, "x", (pos.x + xOffset).toFixed()); │ │ │ │ │ + node.setAttributeNS(null, "y", (pos.y + yOffset).toFixed()); │ │ │ │ │ + node.setAttributeNS(null, "width", width); │ │ │ │ │ + node.setAttributeNS(null, "height", height); │ │ │ │ │ + node.setAttributeNS(this.xlinkns, "xlink:href", style.externalGraphic); │ │ │ │ │ + node.setAttributeNS(null, "style", "opacity: " + opacity); │ │ │ │ │ + node.onclick = OpenLayers.Event.preventDefault │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + var offset = style.pointRadius * 3; │ │ │ │ │ + var size = offset * 2; │ │ │ │ │ + var src = this.importSymbol(style.graphicName); │ │ │ │ │ + pos = this.getPosition(node); │ │ │ │ │ + widthFactor = this.symbolMetrics[src.id][0] * 3 / size; │ │ │ │ │ + var parent = node.parentNode; │ │ │ │ │ + var nextSibling = node.nextSibling; │ │ │ │ │ + if (parent) { │ │ │ │ │ + parent.removeChild(node) │ │ │ │ │ + } │ │ │ │ │ + node.firstChild && node.removeChild(node.firstChild); │ │ │ │ │ + node.appendChild(src.firstChild.cloneNode(true)); │ │ │ │ │ + node.setAttributeNS(null, "viewBox", src.getAttributeNS(null, "viewBox")); │ │ │ │ │ + node.setAttributeNS(null, "width", size); │ │ │ │ │ + node.setAttributeNS(null, "height", size); │ │ │ │ │ + node.setAttributeNS(null, "x", pos.x - offset); │ │ │ │ │ + node.setAttributeNS(null, "y", pos.y - offset); │ │ │ │ │ + if (nextSibling) { │ │ │ │ │ + parent.insertBefore(node, nextSibling) │ │ │ │ │ + } else if (parent) { │ │ │ │ │ + parent.appendChild(node) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "r", style.pointRadius) │ │ │ │ │ + } │ │ │ │ │ + var rotation = style.rotation; │ │ │ │ │ + if ((rotation !== undefined || node._rotation !== undefined) && pos) { │ │ │ │ │ + node._rotation = rotation; │ │ │ │ │ + rotation |= 0; │ │ │ │ │ + if (node.nodeName !== "svg") { │ │ │ │ │ + node.setAttributeNS(null, "transform", "rotate(" + rotation + " " + pos.x + " " + pos.y + ")") │ │ │ │ │ + } else { │ │ │ │ │ + var metrics = this.symbolMetrics[src.id]; │ │ │ │ │ + node.firstChild.setAttributeNS(null, "transform", "rotate(" + rotation + " " + metrics[1] + " " + metrics[2] + ")") │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return uri │ │ │ │ │ + if (options.isFilled) { │ │ │ │ │ + node.setAttributeNS(null, "fill", style.fillColor); │ │ │ │ │ + node.setAttributeNS(null, "fill-opacity", style.fillOpacity) │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "fill", "none") │ │ │ │ │ + } │ │ │ │ │ + if (options.isStroked) { │ │ │ │ │ + node.setAttributeNS(null, "stroke", style.strokeColor); │ │ │ │ │ + node.setAttributeNS(null, "stroke-opacity", style.strokeOpacity); │ │ │ │ │ + node.setAttributeNS(null, "stroke-width", style.strokeWidth * widthFactor); │ │ │ │ │ + node.setAttributeNS(null, "stroke-linecap", style.strokeLinecap || "round"); │ │ │ │ │ + node.setAttributeNS(null, "stroke-linejoin", "round"); │ │ │ │ │ + style.strokeDashstyle && node.setAttributeNS(null, "stroke-dasharray", this.dashStyle(style, widthFactor)) │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "stroke", "none") │ │ │ │ │ + } │ │ │ │ │ + if (style.pointerEvents) { │ │ │ │ │ + node.setAttributeNS(null, "pointer-events", style.pointerEvents) │ │ │ │ │ + } │ │ │ │ │ + if (style.cursor != null) { │ │ │ │ │ + node.setAttributeNS(null, "cursor", style.cursor) │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - getXMLDoc: function() { │ │ │ │ │ - if (!OpenLayers.Format.XML.document && !this.xmldom) { │ │ │ │ │ - if (document.implementation && document.implementation.createDocument) { │ │ │ │ │ - OpenLayers.Format.XML.document = document.implementation.createDocument("", "", null) │ │ │ │ │ - } else if (!this.xmldom && window.ActiveXObject) { │ │ │ │ │ - this.xmldom = new ActiveXObject("Microsoft.XMLDOM") │ │ │ │ │ + dashStyle: function(style, widthFactor) { │ │ │ │ │ + var w = style.strokeWidth * widthFactor; │ │ │ │ │ + var str = style.strokeDashstyle; │ │ │ │ │ + switch (str) { │ │ │ │ │ + case "solid": │ │ │ │ │ + return "none"; │ │ │ │ │ + case "dot": │ │ │ │ │ + return [1, 4 * w].join(); │ │ │ │ │ + case "dash": │ │ │ │ │ + return [4 * w, 4 * w].join(); │ │ │ │ │ + case "dashdot": │ │ │ │ │ + return [4 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ + case "longdash": │ │ │ │ │ + return [8 * w, 4 * w].join(); │ │ │ │ │ + case "longdashdot": │ │ │ │ │ + return [8 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ + default: │ │ │ │ │ + return OpenLayers.String.trim(str).replace(/\s+/g, ",") │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + createNode: function(type, id) { │ │ │ │ │ + var node = document.createElementNS(this.xmlns, type); │ │ │ │ │ + if (id) { │ │ │ │ │ + node.setAttributeNS(null, "id", id) │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + nodeTypeCompare: function(node, type) { │ │ │ │ │ + return type == node.nodeName │ │ │ │ │ + }, │ │ │ │ │ + createRenderRoot: function() { │ │ │ │ │ + var svg = this.nodeFactory(this.container.id + "_svgRoot", "svg"); │ │ │ │ │ + svg.style.display = "block"; │ │ │ │ │ + return svg │ │ │ │ │ + }, │ │ │ │ │ + createRoot: function(suffix) { │ │ │ │ │ + return this.nodeFactory(this.container.id + suffix, "g") │ │ │ │ │ + }, │ │ │ │ │ + createDefs: function() { │ │ │ │ │ + var defs = this.nodeFactory(this.container.id + "_defs", "defs"); │ │ │ │ │ + this.rendererRoot.appendChild(defs); │ │ │ │ │ + return defs │ │ │ │ │ + }, │ │ │ │ │ + drawPoint: function(node, geometry) { │ │ │ │ │ + return this.drawCircle(node, geometry, 1) │ │ │ │ │ + }, │ │ │ │ │ + drawCircle: function(node, geometry, radius) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = (geometry.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y = this.top - geometry.y / resolution; │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + node.setAttributeNS(null, "cx", x); │ │ │ │ │ + node.setAttributeNS(null, "cy", y); │ │ │ │ │ + node.setAttributeNS(null, "r", radius); │ │ │ │ │ + return node │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + drawLineString: function(node, geometry) { │ │ │ │ │ + var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ + if (componentsResult.path) { │ │ │ │ │ + node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ + return componentsResult.complete ? node : null │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + drawLinearRing: function(node, geometry) { │ │ │ │ │ + var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ + if (componentsResult.path) { │ │ │ │ │ + node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ + return componentsResult.complete ? node : null │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + drawPolygon: function(node, geometry) { │ │ │ │ │ + var d = ""; │ │ │ │ │ + var draw = true; │ │ │ │ │ + var complete = true; │ │ │ │ │ + var linearRingResult, path; │ │ │ │ │ + for (var j = 0, len = geometry.components.length; j < len; j++) { │ │ │ │ │ + d += " M"; │ │ │ │ │ + linearRingResult = this.getComponentsString(geometry.components[j].components, " "); │ │ │ │ │ + path = linearRingResult.path; │ │ │ │ │ + if (path) { │ │ │ │ │ + d += " " + path; │ │ │ │ │ + complete = linearRingResult.complete && complete │ │ │ │ │ + } else { │ │ │ │ │ + draw = false │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Format.XML.document || this.xmldom │ │ │ │ │ + d += " z"; │ │ │ │ │ + if (draw) { │ │ │ │ │ + node.setAttributeNS(null, "d", d); │ │ │ │ │ + node.setAttributeNS(null, "fill-rule", "evenodd"); │ │ │ │ │ + return complete ? node : null │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.XML" │ │ │ │ │ + drawRectangle: function(node, geometry) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = (geometry.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y = this.top - geometry.y / resolution; │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + node.setAttributeNS(null, "x", x); │ │ │ │ │ + node.setAttributeNS(null, "y", y); │ │ │ │ │ + node.setAttributeNS(null, "width", geometry.width / resolution); │ │ │ │ │ + node.setAttributeNS(null, "height", geometry.height / resolution); │ │ │ │ │ + return node │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + drawText: function(featureId, style, location) { │ │ │ │ │ + var drawOutline = !!style.labelOutlineWidth; │ │ │ │ │ + if (drawOutline) { │ │ │ │ │ + var outlineStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ + outlineStyle.fontColor = outlineStyle.labelOutlineColor; │ │ │ │ │ + outlineStyle.fontStrokeColor = outlineStyle.labelOutlineColor; │ │ │ │ │ + outlineStyle.fontStrokeWidth = style.labelOutlineWidth; │ │ │ │ │ + if (style.labelOutlineOpacity) { │ │ │ │ │ + outlineStyle.fontOpacity = style.labelOutlineOpacity │ │ │ │ │ + } │ │ │ │ │ + delete outlineStyle.labelOutlineWidth; │ │ │ │ │ + this.drawText(featureId, outlineStyle, location) │ │ │ │ │ + } │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = (location.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y = location.y / resolution - this.top; │ │ │ │ │ + var suffix = drawOutline ? this.LABEL_OUTLINE_SUFFIX : this.LABEL_ID_SUFFIX; │ │ │ │ │ + var label = this.nodeFactory(featureId + suffix, "text"); │ │ │ │ │ + label.setAttributeNS(null, "x", x); │ │ │ │ │ + label.setAttributeNS(null, "y", -y); │ │ │ │ │ + if (style.fontColor) { │ │ │ │ │ + label.setAttributeNS(null, "fill", style.fontColor) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStrokeColor) { │ │ │ │ │ + label.setAttributeNS(null, "stroke", style.fontStrokeColor) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStrokeWidth) { │ │ │ │ │ + label.setAttributeNS(null, "stroke-width", style.fontStrokeWidth) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontOpacity) { │ │ │ │ │ + label.setAttributeNS(null, "opacity", style.fontOpacity) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontFamily) { │ │ │ │ │ + label.setAttributeNS(null, "font-family", style.fontFamily) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontSize) { │ │ │ │ │ + label.setAttributeNS(null, "font-size", style.fontSize) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontWeight) { │ │ │ │ │ + label.setAttributeNS(null, "font-weight", style.fontWeight) │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStyle) { │ │ │ │ │ + label.setAttributeNS(null, "font-style", style.fontStyle) │ │ │ │ │ + } │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + label.setAttributeNS(null, "pointer-events", "visible"); │ │ │ │ │ + label._featureId = featureId │ │ │ │ │ + } else { │ │ │ │ │ + label.setAttributeNS(null, "pointer-events", "none") │ │ │ │ │ + } │ │ │ │ │ + var align = style.labelAlign || OpenLayers.Renderer.defaultSymbolizer.labelAlign; │ │ │ │ │ + label.setAttributeNS(null, "text-anchor", OpenLayers.Renderer.SVG.LABEL_ALIGN[align[0]] || "middle"); │ │ │ │ │ + if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ + label.setAttributeNS(null, "dominant-baseline", OpenLayers.Renderer.SVG.LABEL_ALIGN[align[1]] || "central") │ │ │ │ │ + } │ │ │ │ │ + var labelRows = style.label.split("\n"); │ │ │ │ │ + var numRows = labelRows.length; │ │ │ │ │ + while (label.childNodes.length > numRows) { │ │ │ │ │ + label.removeChild(label.lastChild) │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + var tspan = this.nodeFactory(featureId + suffix + "_tspan_" + i, "tspan"); │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + tspan._featureId = featureId; │ │ │ │ │ + tspan._geometry = location; │ │ │ │ │ + tspan._geometryClass = location.CLASS_NAME │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.IS_GECKO === false) { │ │ │ │ │ + tspan.setAttributeNS(null, "baseline-shift", OpenLayers.Renderer.SVG.LABEL_VSHIFT[align[1]] || "-35%") │ │ │ │ │ + } │ │ │ │ │ + tspan.setAttribute("x", x); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + var vfactor = OpenLayers.Renderer.SVG.LABEL_VFACTOR[align[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5 │ │ │ │ │ + } │ │ │ │ │ + tspan.setAttribute("dy", vfactor * (numRows - 1) + "em") │ │ │ │ │ + } else { │ │ │ │ │ + tspan.setAttribute("dy", "1em") │ │ │ │ │ + } │ │ │ │ │ + tspan.textContent = labelRows[i] === "" ? " " : labelRows[i]; │ │ │ │ │ + if (!tspan.parentNode) { │ │ │ │ │ + label.appendChild(tspan) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!label.parentNode) { │ │ │ │ │ + this.textRoot.appendChild(label) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getComponentsString: function(components, separator) { │ │ │ │ │ + var renderCmp = []; │ │ │ │ │ + var complete = true; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + var strings = []; │ │ │ │ │ + var str, component; │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + component = components[i]; │ │ │ │ │ + renderCmp.push(component); │ │ │ │ │ + str = this.getShortString(component); │ │ │ │ │ + if (str) { │ │ │ │ │ + strings.push(str) │ │ │ │ │ + } else { │ │ │ │ │ + if (i > 0) { │ │ │ │ │ + if (this.getShortString(components[i - 1])) { │ │ │ │ │ + strings.push(this.clipLine(components[i], components[i - 1])) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (i < len - 1) { │ │ │ │ │ + if (this.getShortString(components[i + 1])) { │ │ │ │ │ + strings.push(this.clipLine(components[i], components[i + 1])) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + complete = false │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + path: strings.join(separator || ","), │ │ │ │ │ + complete: complete │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + clipLine: function(badComponent, goodComponent) { │ │ │ │ │ + if (goodComponent.equals(badComponent)) { │ │ │ │ │ + return "" │ │ │ │ │ + } │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var maxX = this.MAX_PIXEL - this.translationParameters.x; │ │ │ │ │ + var maxY = this.MAX_PIXEL - this.translationParameters.y; │ │ │ │ │ + var x1 = (goodComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y1 = this.top - goodComponent.y / resolution; │ │ │ │ │ + var x2 = (badComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y2 = this.top - badComponent.y / resolution; │ │ │ │ │ + var k; │ │ │ │ │ + if (x2 < -maxX || x2 > maxX) { │ │ │ │ │ + k = (y2 - y1) / (x2 - x1); │ │ │ │ │ + x2 = x2 < 0 ? -maxX : maxX; │ │ │ │ │ + y2 = y1 + (x2 - x1) * k │ │ │ │ │ + } │ │ │ │ │ + if (y2 < -maxY || y2 > maxY) { │ │ │ │ │ + k = (x2 - x1) / (y2 - y1); │ │ │ │ │ + y2 = y2 < 0 ? -maxY : maxY; │ │ │ │ │ + x2 = x1 + (y2 - y1) * k │ │ │ │ │ + } │ │ │ │ │ + return x2 + "," + y2 │ │ │ │ │ + }, │ │ │ │ │ + getShortString: function(point) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = (point.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y = this.top - point.y / resolution; │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + return x + "," + y │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getPosition: function(node) { │ │ │ │ │ + return { │ │ │ │ │ + x: parseFloat(node.getAttributeNS(null, "cx")), │ │ │ │ │ + y: parseFloat(node.getAttributeNS(null, "cy")) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + importSymbol: function(graphicName) { │ │ │ │ │ + if (!this.defs) { │ │ │ │ │ + this.defs = this.createDefs() │ │ │ │ │ + } │ │ │ │ │ + var id = this.container.id + "-" + graphicName; │ │ │ │ │ + var existing = document.getElementById(id); │ │ │ │ │ + if (existing != null) { │ │ │ │ │ + return existing │ │ │ │ │ + } │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(graphicName + " is not a valid symbol name") │ │ │ │ │ + } │ │ │ │ │ + var symbolNode = this.nodeFactory(id, "symbol"); │ │ │ │ │ + var node = this.nodeFactory(null, "polygon"); │ │ │ │ │ + symbolNode.appendChild(node); │ │ │ │ │ + var symbolExtent = new OpenLayers.Bounds(Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ + var points = []; │ │ │ │ │ + var x, y; │ │ │ │ │ + for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ + symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ + symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ + symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ + points.push(x, ",", y) │ │ │ │ │ + } │ │ │ │ │ + node.setAttributeNS(null, "points", points.join(" ")); │ │ │ │ │ + var width = symbolExtent.getWidth(); │ │ │ │ │ + var height = symbolExtent.getHeight(); │ │ │ │ │ + var viewBox = [symbolExtent.left - width, symbolExtent.bottom - height, width * 3, height * 3]; │ │ │ │ │ + symbolNode.setAttributeNS(null, "viewBox", viewBox.join(" ")); │ │ │ │ │ + this.symbolMetrics[id] = [Math.max(width, height), symbolExtent.getCenterLonLat().lon, symbolExtent.getCenterLonLat().lat]; │ │ │ │ │ + this.defs.appendChild(symbolNode); │ │ │ │ │ + return symbolNode │ │ │ │ │ + }, │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var featureId = OpenLayers.Renderer.Elements.prototype.getFeatureIdFromEvent.apply(this, arguments); │ │ │ │ │ + if (!featureId) { │ │ │ │ │ + var target = evt.target; │ │ │ │ │ + featureId = target.parentNode && target != this.rendererRoot ? target.parentNode._featureId : undefined │ │ │ │ │ + } │ │ │ │ │ + return featureId │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.SVG" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.XML.CONTENT_TYPE = { │ │ │ │ │ - EMPTY: 0, │ │ │ │ │ - SIMPLE: 1, │ │ │ │ │ - COMPLEX: 2, │ │ │ │ │ - MIXED: 3 │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_ALIGN = { │ │ │ │ │ + l: "start", │ │ │ │ │ + r: "end", │ │ │ │ │ + b: "bottom", │ │ │ │ │ + t: "hanging" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_VSHIFT = { │ │ │ │ │ + t: "-70%", │ │ │ │ │ + b: "0" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_VFACTOR = { │ │ │ │ │ + t: 0, │ │ │ │ │ + b: -1 │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.SVG.preventDefault = function(e) { │ │ │ │ │ + OpenLayers.Event.preventDefault(e) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol = OpenLayers.Class({ │ │ │ │ │ + format: null, │ │ │ │ │ + options: null, │ │ │ │ │ + autoDestroy: true, │ │ │ │ │ + defaultFilter: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.options = options │ │ │ │ │ + }, │ │ │ │ │ + mergeWithDefaultFilter: function(filter) { │ │ │ │ │ + var merged; │ │ │ │ │ + if (filter && this.defaultFilter) { │ │ │ │ │ + merged = new OpenLayers.Filter.Logical({ │ │ │ │ │ + type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ + filters: [this.defaultFilter, filter] │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + merged = filter || this.defaultFilter || undefined │ │ │ │ │ + } │ │ │ │ │ + return merged │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.options = null; │ │ │ │ │ + this.format = null │ │ │ │ │ + }, │ │ │ │ │ + read: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.filter = this.mergeWithDefaultFilter(options.filter) │ │ │ │ │ + }, │ │ │ │ │ + create: function() {}, │ │ │ │ │ + update: function() {}, │ │ │ │ │ + delete: function() {}, │ │ │ │ │ + commit: function() {}, │ │ │ │ │ + abort: function(response) {}, │ │ │ │ │ + createCallback: function(method, response, options) { │ │ │ │ │ + return OpenLayers.Function.bind(function() { │ │ │ │ │ + method.apply(this, [response, options]) │ │ │ │ │ + }, this) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Protocol.Response = OpenLayers.Class({ │ │ │ │ │ + code: null, │ │ │ │ │ + requestType: null, │ │ │ │ │ + last: true, │ │ │ │ │ + features: null, │ │ │ │ │ + data: null, │ │ │ │ │ + reqFeatures: null, │ │ │ │ │ + priv: null, │ │ │ │ │ + error: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options) │ │ │ │ │ + }, │ │ │ │ │ + success: function() { │ │ │ │ │ + return this.code > 0 │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.Response" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Protocol.Response.SUCCESS = 1; │ │ │ │ │ +OpenLayers.Protocol.Response.FAILURE = 0; │ │ │ │ │ +OpenLayers.Protocol.WFS = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, OpenLayers.Protocol.WFS.DEFAULTS); │ │ │ │ │ + var cls = OpenLayers.Protocol.WFS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported WFS version: " + options.version │ │ │ │ │ + } │ │ │ │ │ + return new cls(options) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol.WFS.fromWMSLayer = function(layer, options) { │ │ │ │ │ + var typeName, featurePrefix; │ │ │ │ │ + var param = layer.params["LAYERS"]; │ │ │ │ │ + var parts = (OpenLayers.Util.isArray(param) ? param[0] : param).split(":"); │ │ │ │ │ + if (parts.length > 1) { │ │ │ │ │ + featurePrefix = parts[0] │ │ │ │ │ + } │ │ │ │ │ + typeName = parts.pop(); │ │ │ │ │ + var protocolOptions = { │ │ │ │ │ + url: layer.url, │ │ │ │ │ + featureType: typeName, │ │ │ │ │ + featurePrefix: featurePrefix, │ │ │ │ │ + srsName: layer.projection && layer.projection.getCode() || layer.map && layer.map.getProjectionObject().getCode(), │ │ │ │ │ + version: "1.1.0" │ │ │ │ │ + }; │ │ │ │ │ + return new OpenLayers.Protocol.WFS(OpenLayers.Util.applyDefaults(options, protocolOptions)) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol.WFS.DEFAULTS = { │ │ │ │ │ + version: "1.0.0" │ │ │ │ │ }; │ │ │ │ │ -OpenLayers.Format.XML.lookupNamespaceURI = OpenLayers.Function.bind(OpenLayers.Format.XML.prototype.lookupNamespaceURI, OpenLayers.Format.XML.prototype); │ │ │ │ │ -OpenLayers.Format.XML.document = null; │ │ │ │ │ OpenLayers.ProxyHost = ""; │ │ │ │ │ if (!OpenLayers.Request) { │ │ │ │ │ OpenLayers.Request = {} │ │ │ │ │ } │ │ │ │ │ OpenLayers.Util.extend(OpenLayers.Request, { │ │ │ │ │ DEFAULT_CONFIG: { │ │ │ │ │ method: "GET", │ │ │ │ │ @@ -9628,6170 +9463,1791 @@ │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ if (!OpenLayers.Request) { │ │ │ │ │ OpenLayers.Request = {} │ │ │ │ │ } │ │ │ │ │ OpenLayers.Request.XMLHttpRequest = cXMLHttpRequest │ │ │ │ │ })(); │ │ │ │ │ -OpenLayers.Format.KML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - namespaces: { │ │ │ │ │ - kml: "http://www.opengis.net/kml/2.2", │ │ │ │ │ - gx: "http://www.google.com/kml/ext/2.2" │ │ │ │ │ - }, │ │ │ │ │ - kmlns: "http://earth.google.com/kml/2.0", │ │ │ │ │ - placemarksDesc: "No description available", │ │ │ │ │ - foldersName: "OpenLayers export", │ │ │ │ │ - foldersDesc: "Exported on " + new Date, │ │ │ │ │ - extractAttributes: true, │ │ │ │ │ - kvpAttributes: false, │ │ │ │ │ - extractStyles: false, │ │ │ │ │ - extractTracks: false, │ │ │ │ │ - trackAttributes: null, │ │ │ │ │ - internalns: null, │ │ │ │ │ - features: null, │ │ │ │ │ - styles: null, │ │ │ │ │ - styleBaseUrl: "", │ │ │ │ │ - fetched: null, │ │ │ │ │ - maxDepth: 0, │ │ │ │ │ +OpenLayers.Protocol.HTTP = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + url: null, │ │ │ │ │ + headers: null, │ │ │ │ │ + params: null, │ │ │ │ │ + callback: null, │ │ │ │ │ + scope: null, │ │ │ │ │ + readWithPOST: false, │ │ │ │ │ + updateWithPOST: false, │ │ │ │ │ + deleteWithPOST: false, │ │ │ │ │ + wildcarded: false, │ │ │ │ │ + srsInBBOX: false, │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - this.regExes = { │ │ │ │ │ - trimSpace: /^\s*|\s*$/g, │ │ │ │ │ - removeSpace: /\s*/g, │ │ │ │ │ - splitSpace: /\s+/, │ │ │ │ │ - trimComma: /\s*,\s*/g, │ │ │ │ │ - kmlColor: /(\w{2})(\w{2})(\w{2})(\w{2})/, │ │ │ │ │ - kmlIconPalette: /root:\/\/icons\/palette-(\d+)(\.\w+)/, │ │ │ │ │ - straightBracket: /\$\[(.*?)\]/g │ │ │ │ │ - }; │ │ │ │ │ - this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ - read: function(data) { │ │ │ │ │ - this.features = []; │ │ │ │ │ - this.styles = {}; │ │ │ │ │ - this.fetched = {}; │ │ │ │ │ - var options = { │ │ │ │ │ - depth: 0, │ │ │ │ │ - styleBaseUrl: this.styleBaseUrl │ │ │ │ │ - }; │ │ │ │ │ - return this.parseData(data, options) │ │ │ │ │ - }, │ │ │ │ │ - parseData: function(data, options) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ - } │ │ │ │ │ - var types = ["Link", "NetworkLink", "Style", "StyleMap", "Placemark"]; │ │ │ │ │ - for (var i = 0, len = types.length; i < len; ++i) { │ │ │ │ │ - var type = types[i]; │ │ │ │ │ - var nodes = this.getElementsByTagNameNS(data, "*", type); │ │ │ │ │ - if (nodes.length == 0) { │ │ │ │ │ - continue │ │ │ │ │ - } │ │ │ │ │ - switch (type.toLowerCase()) { │ │ │ │ │ - case "link": │ │ │ │ │ - case "networklink": │ │ │ │ │ - this.parseLinks(nodes, options); │ │ │ │ │ - break; │ │ │ │ │ - case "style": │ │ │ │ │ - if (this.extractStyles) { │ │ │ │ │ - this.parseStyles(nodes, options) │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "stylemap": │ │ │ │ │ - if (this.extractStyles) { │ │ │ │ │ - this.parseStyleMaps(nodes, options) │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "placemark": │ │ │ │ │ - this.parseFeatures(nodes, options); │ │ │ │ │ - break │ │ │ │ │ + options = options || {}; │ │ │ │ │ + this.params = {}; │ │ │ │ │ + this.headers = {}; │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ + var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ + wildcarded: this.wildcarded, │ │ │ │ │ + srsInBBOX: this.srsInBBOX │ │ │ │ │ + }); │ │ │ │ │ + this.filterToParams = function(filter, params) { │ │ │ │ │ + return format.write(filter, params) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return this.features │ │ │ │ │ }, │ │ │ │ │ - parseLinks: function(nodes, options) { │ │ │ │ │ - if (options.depth >= this.maxDepth) { │ │ │ │ │ - return false │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.params = null; │ │ │ │ │ + this.headers = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ + }, │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.params = OpenLayers.Util.applyDefaults(options.params, this.options.params); │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + if (options.filter && this.filterToParams) { │ │ │ │ │ + options.params = this.filterToParams(options.filter, options.params) │ │ │ │ │ } │ │ │ │ │ - var newOptions = OpenLayers.Util.extend({}, options); │ │ │ │ │ - newOptions.depth++; │ │ │ │ │ - for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ - var href = this.parseProperty(nodes[i], "*", "href"); │ │ │ │ │ - if (href && !this.fetched[href]) { │ │ │ │ │ - this.fetched[href] = true; │ │ │ │ │ - var data = this.fetchLink(href); │ │ │ │ │ - if (data) { │ │ │ │ │ - this.parseData(data, newOptions) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + var readWithPOST = options.readWithPOST !== undefined ? options.readWithPOST : this.readWithPOST; │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + if (readWithPOST) { │ │ │ │ │ + var headers = options.headers || {}; │ │ │ │ │ + headers["Content-Type"] = "application/x-www-form-urlencoded"; │ │ │ │ │ + resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ + data: OpenLayers.Util.getParameterString(options.params), │ │ │ │ │ + headers: headers │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + resp.priv = OpenLayers.Request.GET({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ + return resp │ │ │ │ │ }, │ │ │ │ │ - fetchLink: function(href) { │ │ │ │ │ - var request = OpenLayers.Request.GET({ │ │ │ │ │ - url: href, │ │ │ │ │ - async: false │ │ │ │ │ + handleRead: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options) │ │ │ │ │ + }, │ │ │ │ │ + create: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: features, │ │ │ │ │ + requestType: "create" │ │ │ │ │ }); │ │ │ │ │ - if (request) { │ │ │ │ │ - return request.responseText │ │ │ │ │ - } │ │ │ │ │ + resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleCreate, resp, options), │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(features) │ │ │ │ │ + }); │ │ │ │ │ + return resp │ │ │ │ │ }, │ │ │ │ │ - parseStyles: function(nodes, options) { │ │ │ │ │ - for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ - var style = this.parseStyle(nodes[i]); │ │ │ │ │ - if (style) { │ │ │ │ │ - var styleName = (options.styleBaseUrl || "") + "#" + style.id; │ │ │ │ │ - this.styles[styleName] = style │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + handleCreate: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options) │ │ │ │ │ }, │ │ │ │ │ - parseKmlColor: function(kmlColor) { │ │ │ │ │ - var color = null; │ │ │ │ │ - if (kmlColor) { │ │ │ │ │ - var matches = kmlColor.match(this.regExes.kmlColor); │ │ │ │ │ - if (matches) { │ │ │ │ │ - color = { │ │ │ │ │ - color: "#" + matches[4] + matches[3] + matches[2], │ │ │ │ │ - opacity: parseInt(matches[1], 16) / 255 │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return color │ │ │ │ │ + update: function(feature, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var url = options.url || feature.url || this.options.url + "/" + feature.fid; │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: feature, │ │ │ │ │ + requestType: "update" │ │ │ │ │ + }); │ │ │ │ │ + var method = this.updateWithPOST ? "POST" : "PUT"; │ │ │ │ │ + resp.priv = OpenLayers.Request[method]({ │ │ │ │ │ + url: url, │ │ │ │ │ + callback: this.createCallback(this.handleUpdate, resp, options), │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(feature) │ │ │ │ │ + }); │ │ │ │ │ + return resp │ │ │ │ │ }, │ │ │ │ │ - parseStyle: function(node) { │ │ │ │ │ - var style = {}; │ │ │ │ │ - var types = ["LineStyle", "PolyStyle", "IconStyle", "BalloonStyle", "LabelStyle"]; │ │ │ │ │ - var type, styleTypeNode, nodeList, geometry, parser; │ │ │ │ │ - for (var i = 0, len = types.length; i < len; ++i) { │ │ │ │ │ - type = types[i]; │ │ │ │ │ - styleTypeNode = this.getElementsByTagNameNS(node, "*", type)[0]; │ │ │ │ │ - if (!styleTypeNode) { │ │ │ │ │ - continue │ │ │ │ │ - } │ │ │ │ │ - switch (type.toLowerCase()) { │ │ │ │ │ - case "linestyle": │ │ │ │ │ - var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ - var color = this.parseKmlColor(kmlColor); │ │ │ │ │ - if (color) { │ │ │ │ │ - style["strokeColor"] = color.color; │ │ │ │ │ - style["strokeOpacity"] = color.opacity │ │ │ │ │ - } │ │ │ │ │ - var width = this.parseProperty(styleTypeNode, "*", "width"); │ │ │ │ │ - if (width) { │ │ │ │ │ - style["strokeWidth"] = width │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "polystyle": │ │ │ │ │ - var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ - var color = this.parseKmlColor(kmlColor); │ │ │ │ │ - if (color) { │ │ │ │ │ - style["fillOpacity"] = color.opacity; │ │ │ │ │ - style["fillColor"] = color.color │ │ │ │ │ - } │ │ │ │ │ - var fill = this.parseProperty(styleTypeNode, "*", "fill"); │ │ │ │ │ - if (fill == "0") { │ │ │ │ │ - style["fillColor"] = "none" │ │ │ │ │ - } │ │ │ │ │ - var outline = this.parseProperty(styleTypeNode, "*", "outline"); │ │ │ │ │ - if (outline == "0") { │ │ │ │ │ - style["strokeWidth"] = "0" │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "iconstyle": │ │ │ │ │ - var scale = parseFloat(this.parseProperty(styleTypeNode, "*", "scale") || 1); │ │ │ │ │ - var width = 32 * scale; │ │ │ │ │ - var height = 32 * scale; │ │ │ │ │ - var iconNode = this.getElementsByTagNameNS(styleTypeNode, "*", "Icon")[0]; │ │ │ │ │ - if (iconNode) { │ │ │ │ │ - var href = this.parseProperty(iconNode, "*", "href"); │ │ │ │ │ - if (href) { │ │ │ │ │ - var w = this.parseProperty(iconNode, "*", "w"); │ │ │ │ │ - var h = this.parseProperty(iconNode, "*", "h"); │ │ │ │ │ - var google = "http://maps.google.com/mapfiles/kml"; │ │ │ │ │ - if (OpenLayers.String.startsWith(href, google) && !w && !h) { │ │ │ │ │ - w = 64; │ │ │ │ │ - h = 64; │ │ │ │ │ - scale = scale / 2 │ │ │ │ │ - } │ │ │ │ │ - w = w || h; │ │ │ │ │ - h = h || w; │ │ │ │ │ - if (w) { │ │ │ │ │ - width = parseInt(w) * scale │ │ │ │ │ - } │ │ │ │ │ - if (h) { │ │ │ │ │ - height = parseInt(h) * scale │ │ │ │ │ - } │ │ │ │ │ - var matches = href.match(this.regExes.kmlIconPalette); │ │ │ │ │ - if (matches) { │ │ │ │ │ - var palette = matches[1]; │ │ │ │ │ - var file_extension = matches[2]; │ │ │ │ │ - var x = this.parseProperty(iconNode, "*", "x"); │ │ │ │ │ - var y = this.parseProperty(iconNode, "*", "y"); │ │ │ │ │ - var posX = x ? x / 32 : 0; │ │ │ │ │ - var posY = y ? 7 - y / 32 : 7; │ │ │ │ │ - var pos = posY * 8 + posX; │ │ │ │ │ - href = "http://maps.google.com/mapfiles/kml/pal" + palette + "/icon" + pos + file_extension │ │ │ │ │ - } │ │ │ │ │ - style["graphicOpacity"] = 1; │ │ │ │ │ - style["externalGraphic"] = href │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var hotSpotNode = this.getElementsByTagNameNS(styleTypeNode, "*", "hotSpot")[0]; │ │ │ │ │ - if (hotSpotNode) { │ │ │ │ │ - var x = parseFloat(hotSpotNode.getAttribute("x")); │ │ │ │ │ - var y = parseFloat(hotSpotNode.getAttribute("y")); │ │ │ │ │ - var xUnits = hotSpotNode.getAttribute("xunits"); │ │ │ │ │ - if (xUnits == "pixels") { │ │ │ │ │ - style["graphicXOffset"] = -x * scale │ │ │ │ │ - } else if (xUnits == "insetPixels") { │ │ │ │ │ - style["graphicXOffset"] = -width + x * scale │ │ │ │ │ - } else if (xUnits == "fraction") { │ │ │ │ │ - style["graphicXOffset"] = -width * x │ │ │ │ │ - } │ │ │ │ │ - var yUnits = hotSpotNode.getAttribute("yunits"); │ │ │ │ │ - if (yUnits == "pixels") { │ │ │ │ │ - style["graphicYOffset"] = -height + y * scale + 1 │ │ │ │ │ - } else if (yUnits == "insetPixels") { │ │ │ │ │ - style["graphicYOffset"] = -(y * scale) + 1 │ │ │ │ │ - } else if (yUnits == "fraction") { │ │ │ │ │ - style["graphicYOffset"] = -height * (1 - y) + 1 │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - style["graphicWidth"] = width; │ │ │ │ │ - style["graphicHeight"] = height; │ │ │ │ │ - break; │ │ │ │ │ - case "balloonstyle": │ │ │ │ │ - var balloonStyle = OpenLayers.Util.getXmlNodeValue(styleTypeNode); │ │ │ │ │ - if (balloonStyle) { │ │ │ │ │ - style["balloonStyle"] = balloonStyle.replace(this.regExes.straightBracket, "${$1}") │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "labelstyle": │ │ │ │ │ - var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ - var color = this.parseKmlColor(kmlColor); │ │ │ │ │ - if (color) { │ │ │ │ │ - style["fontColor"] = color.color; │ │ │ │ │ - style["fontOpacity"] = color.opacity │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!style["strokeColor"] && style["fillColor"]) { │ │ │ │ │ - style["strokeColor"] = style["fillColor"] │ │ │ │ │ - } │ │ │ │ │ - var id = node.getAttribute("id"); │ │ │ │ │ - if (id && style) { │ │ │ │ │ - style.id = id │ │ │ │ │ - } │ │ │ │ │ - return style │ │ │ │ │ + handleUpdate: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options) │ │ │ │ │ }, │ │ │ │ │ - parseStyleMaps: function(nodes, options) { │ │ │ │ │ - for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ - var node = nodes[i]; │ │ │ │ │ - var pairs = this.getElementsByTagNameNS(node, "*", "Pair"); │ │ │ │ │ - var id = node.getAttribute("id"); │ │ │ │ │ - for (var j = 0, jlen = pairs.length; j < jlen; j++) { │ │ │ │ │ - var pair = pairs[j]; │ │ │ │ │ - var key = this.parseProperty(pair, "*", "key"); │ │ │ │ │ - var styleUrl = this.parseProperty(pair, "*", "styleUrl"); │ │ │ │ │ - if (styleUrl && key == "normal") { │ │ │ │ │ - this.styles[(options.styleBaseUrl || "") + "#" + id] = this.styles[(options.styleBaseUrl || "") + styleUrl] │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + delete: function(feature, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var url = options.url || feature.url || this.options.url + "/" + feature.fid; │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: feature, │ │ │ │ │ + requestType: "delete" │ │ │ │ │ + }); │ │ │ │ │ + var method = this.deleteWithPOST ? "POST" : "DELETE"; │ │ │ │ │ + var requestOptions = { │ │ │ │ │ + url: url, │ │ │ │ │ + callback: this.createCallback(this.handleDelete, resp, options), │ │ │ │ │ + headers: options.headers │ │ │ │ │ + }; │ │ │ │ │ + if (this.deleteWithPOST) { │ │ │ │ │ + requestOptions.data = this.format.write(feature) │ │ │ │ │ } │ │ │ │ │ + resp.priv = OpenLayers.Request[method](requestOptions); │ │ │ │ │ + return resp │ │ │ │ │ }, │ │ │ │ │ - parseFeatures: function(nodes, options) { │ │ │ │ │ - var features = []; │ │ │ │ │ - for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ - var featureNode = nodes[i]; │ │ │ │ │ - var feature = this.parseFeature.apply(this, [featureNode]); │ │ │ │ │ - if (feature) { │ │ │ │ │ - if (this.extractStyles && feature.attributes && feature.attributes.styleUrl) { │ │ │ │ │ - feature.style = this.getStyle(feature.attributes.styleUrl, options) │ │ │ │ │ - } │ │ │ │ │ - if (this.extractStyles) { │ │ │ │ │ - var inlineStyleNode = this.getElementsByTagNameNS(featureNode, "*", "Style")[0]; │ │ │ │ │ - if (inlineStyleNode) { │ │ │ │ │ - var inlineStyle = this.parseStyle(inlineStyleNode); │ │ │ │ │ - if (inlineStyle) { │ │ │ │ │ - feature.style = OpenLayers.Util.extend(feature.style, inlineStyle) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.extractTracks) { │ │ │ │ │ - var tracks = this.getElementsByTagNameNS(featureNode, this.namespaces.gx, "Track"); │ │ │ │ │ - if (tracks && tracks.length > 0) { │ │ │ │ │ - var track = tracks[0]; │ │ │ │ │ - var container = { │ │ │ │ │ - features: [], │ │ │ │ │ - feature: feature │ │ │ │ │ - }; │ │ │ │ │ - this.readNode(track, container); │ │ │ │ │ - if (container.features.length > 0) { │ │ │ │ │ - features.push.apply(features, container.features) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - features.push(feature) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - throw "Bad Placemark: " + i │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.features = this.features.concat(features) │ │ │ │ │ + handleDelete: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options) │ │ │ │ │ }, │ │ │ │ │ - readers: { │ │ │ │ │ - kml: { │ │ │ │ │ - when: function(node, container) { │ │ │ │ │ - container.whens.push(OpenLayers.Date.parse(this.getChildValue(node))) │ │ │ │ │ - }, │ │ │ │ │ - _trackPointAttribute: function(node, container) { │ │ │ │ │ - var name = node.nodeName.split(":").pop(); │ │ │ │ │ - container.attributes[name].push(this.getChildValue(node)) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - gx: { │ │ │ │ │ - Track: function(node, container) { │ │ │ │ │ - var obj = { │ │ │ │ │ - whens: [], │ │ │ │ │ - points: [], │ │ │ │ │ - angles: [] │ │ │ │ │ - }; │ │ │ │ │ - if (this.trackAttributes) { │ │ │ │ │ - var name; │ │ │ │ │ - obj.attributes = {}; │ │ │ │ │ - for (var i = 0, ii = this.trackAttributes.length; i < ii; ++i) { │ │ │ │ │ - name = this.trackAttributes[i]; │ │ │ │ │ - obj.attributes[name] = []; │ │ │ │ │ - if (!(name in this.readers.kml)) { │ │ │ │ │ - this.readers.kml[name] = this.readers.kml._trackPointAttribute │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - if (obj.whens.length !== obj.points.length) { │ │ │ │ │ - throw new Error("gx:Track with unequal number of when (" + obj.whens.length + ") and gx:coord (" + obj.points.length + ") elements.") │ │ │ │ │ - } │ │ │ │ │ - var hasAngles = obj.angles.length > 0; │ │ │ │ │ - if (hasAngles && obj.whens.length !== obj.angles.length) { │ │ │ │ │ - throw new Error("gx:Track with unequal number of when (" + obj.whens.length + ") and gx:angles (" + obj.angles.length + ") elements.") │ │ │ │ │ - } │ │ │ │ │ - var feature, point, angles; │ │ │ │ │ - for (var i = 0, ii = obj.whens.length; i < ii; ++i) { │ │ │ │ │ - feature = container.feature.clone(); │ │ │ │ │ - feature.fid = container.feature.fid || container.feature.id; │ │ │ │ │ - point = obj.points[i]; │ │ │ │ │ - feature.geometry = point; │ │ │ │ │ - if ("z" in point) { │ │ │ │ │ - feature.attributes.altitude = point.z │ │ │ │ │ - } │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - feature.geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ - } │ │ │ │ │ - if (this.trackAttributes) { │ │ │ │ │ - for (var j = 0, jj = this.trackAttributes.length; j < jj; ++j) { │ │ │ │ │ - var name = this.trackAttributes[j]; │ │ │ │ │ - feature.attributes[name] = obj.attributes[name][i] │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - feature.attributes.when = obj.whens[i]; │ │ │ │ │ - feature.attributes.trackId = container.feature.id; │ │ │ │ │ - if (hasAngles) { │ │ │ │ │ - angles = obj.angles[i]; │ │ │ │ │ - feature.attributes.heading = parseFloat(angles[0]); │ │ │ │ │ - feature.attributes.tilt = parseFloat(angles[1]); │ │ │ │ │ - feature.attributes.roll = parseFloat(angles[2]) │ │ │ │ │ - } │ │ │ │ │ - container.features.push(feature) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - coord: function(node, container) { │ │ │ │ │ - var str = this.getChildValue(node); │ │ │ │ │ - var coords = str.replace(this.regExes.trimSpace, "").split(/\s+/); │ │ │ │ │ - var point = new OpenLayers.Geometry.Point(coords[0], coords[1]); │ │ │ │ │ - if (coords.length > 2) { │ │ │ │ │ - point.z = parseFloat(coords[2]) │ │ │ │ │ + handleResponse: function(resp, options) { │ │ │ │ │ + var request = resp.priv; │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + if (resp.requestType != "delete") { │ │ │ │ │ + resp.features = this.parseFeatures(request) │ │ │ │ │ } │ │ │ │ │ - container.points.push(point) │ │ │ │ │ - }, │ │ │ │ │ - angles: function(node, container) { │ │ │ │ │ - var str = this.getChildValue(node); │ │ │ │ │ - var parts = str.replace(this.regExes.trimSpace, "").split(/\s+/); │ │ │ │ │ - container.angles.push(parts) │ │ │ │ │ + resp.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ + } else { │ │ │ │ │ + resp.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ } │ │ │ │ │ + options.callback.call(options.scope, resp) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - parseFeature: function(node) { │ │ │ │ │ - var order = ["MultiGeometry", "Polygon", "LineString", "Point"]; │ │ │ │ │ - var type, nodeList, geometry, parser; │ │ │ │ │ - for (var i = 0, len = order.length; i < len; ++i) { │ │ │ │ │ - type = order[i]; │ │ │ │ │ - this.internalns = node.namespaceURI ? node.namespaceURI : this.kmlns; │ │ │ │ │ - nodeList = this.getElementsByTagNameNS(node, this.internalns, type); │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - var parser = this.parseGeometry[type.toLowerCase()]; │ │ │ │ │ - if (parser) { │ │ │ │ │ - geometry = parser.apply(this, [nodeList[0]]); │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - throw new TypeError("Unsupported geometry type: " + type) │ │ │ │ │ - } │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var attributes; │ │ │ │ │ - if (this.extractAttributes) { │ │ │ │ │ - attributes = this.parseAttributes(node) │ │ │ │ │ + parseFeatures: function(request) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ } │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry, attributes); │ │ │ │ │ - var fid = node.getAttribute("id") || node.getAttribute("name"); │ │ │ │ │ - if (fid != null) { │ │ │ │ │ - feature.fid = fid │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null │ │ │ │ │ } │ │ │ │ │ - return feature │ │ │ │ │ + return this.format.read(doc) │ │ │ │ │ }, │ │ │ │ │ - getStyle: function(styleUrl, options) { │ │ │ │ │ - var styleBaseUrl = OpenLayers.Util.removeTail(styleUrl); │ │ │ │ │ - var newOptions = OpenLayers.Util.extend({}, options); │ │ │ │ │ - newOptions.depth++; │ │ │ │ │ - newOptions.styleBaseUrl = styleBaseUrl; │ │ │ │ │ - if (!this.styles[styleUrl] && !OpenLayers.String.startsWith(styleUrl, "#") && newOptions.depth <= this.maxDepth && !this.fetched[styleBaseUrl]) { │ │ │ │ │ - var data = this.fetchLink(styleBaseUrl); │ │ │ │ │ - if (data) { │ │ │ │ │ - this.parseData(data, newOptions) │ │ │ │ │ + commit: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = [], │ │ │ │ │ + nResponses = 0; │ │ │ │ │ + var types = {}; │ │ │ │ │ + types[OpenLayers.State.INSERT] = []; │ │ │ │ │ + types[OpenLayers.State.UPDATE] = []; │ │ │ │ │ + types[OpenLayers.State.DELETE] = []; │ │ │ │ │ + var feature, list, requestFeatures = []; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + list = types[feature.state]; │ │ │ │ │ + if (list) { │ │ │ │ │ + list.push(feature); │ │ │ │ │ + requestFeatures.push(feature) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var style = OpenLayers.Util.extend({}, this.styles[styleUrl]); │ │ │ │ │ - return style │ │ │ │ │ - }, │ │ │ │ │ - parseGeometry: { │ │ │ │ │ - point: function(node) { │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, this.internalns, "coordinates"); │ │ │ │ │ - var coords = []; │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - var coordString = nodeList[0].firstChild.nodeValue; │ │ │ │ │ - coordString = coordString.replace(this.regExes.removeSpace, ""); │ │ │ │ │ - coords = coordString.split(",") │ │ │ │ │ - } │ │ │ │ │ - var point = null; │ │ │ │ │ - if (coords.length > 1) { │ │ │ │ │ - if (coords.length == 2) { │ │ │ │ │ - coords[2] = null │ │ │ │ │ - } │ │ │ │ │ - point = new OpenLayers.Geometry.Point(coords[0], coords[1], coords[2]) │ │ │ │ │ - } else { │ │ │ │ │ - throw "Bad coordinate string: " + coordString │ │ │ │ │ - } │ │ │ │ │ - return point │ │ │ │ │ - }, │ │ │ │ │ - linestring: function(node, ring) { │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, this.internalns, "coordinates"); │ │ │ │ │ - var line = null; │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - var coordString = this.getChildValue(nodeList[0]); │ │ │ │ │ - coordString = coordString.replace(this.regExes.trimSpace, ""); │ │ │ │ │ - coordString = coordString.replace(this.regExes.trimComma, ","); │ │ │ │ │ - var pointList = coordString.split(this.regExes.splitSpace); │ │ │ │ │ - var numPoints = pointList.length; │ │ │ │ │ - var points = new Array(numPoints); │ │ │ │ │ - var coords, numCoords; │ │ │ │ │ - for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ - coords = pointList[i].split(","); │ │ │ │ │ - numCoords = coords.length; │ │ │ │ │ - if (numCoords > 1) { │ │ │ │ │ - if (coords.length == 2) { │ │ │ │ │ - coords[2] = null │ │ │ │ │ - } │ │ │ │ │ - points[i] = new OpenLayers.Geometry.Point(coords[0], coords[1], coords[2]) │ │ │ │ │ - } else { │ │ │ │ │ - throw "Bad LineString point coordinates: " + pointList[i] │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (numPoints) { │ │ │ │ │ - if (ring) { │ │ │ │ │ - line = new OpenLayers.Geometry.LinearRing(points) │ │ │ │ │ - } else { │ │ │ │ │ - line = new OpenLayers.Geometry.LineString(points) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - throw "Bad LineString coordinates: " + coordString │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return line │ │ │ │ │ - }, │ │ │ │ │ - polygon: function(node) { │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, this.internalns, "LinearRing"); │ │ │ │ │ - var numRings = nodeList.length; │ │ │ │ │ - var components = new Array(numRings); │ │ │ │ │ - if (numRings > 0) { │ │ │ │ │ - var ring; │ │ │ │ │ - for (var i = 0, len = nodeList.length; i < len; ++i) { │ │ │ │ │ - ring = this.parseGeometry.linestring.apply(this, [nodeList[i], true]); │ │ │ │ │ - if (ring) { │ │ │ │ │ - components[i] = ring │ │ │ │ │ - } else { │ │ │ │ │ - throw "Bad LinearRing geometry: " + i │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Geometry.Polygon(components) │ │ │ │ │ - }, │ │ │ │ │ - multigeometry: function(node) { │ │ │ │ │ - var child, parser; │ │ │ │ │ - var parts = []; │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ - child = children[i]; │ │ │ │ │ - if (child.nodeType == 1) { │ │ │ │ │ - var type = child.prefix ? child.nodeName.split(":")[1] : child.nodeName; │ │ │ │ │ - var parser = this.parseGeometry[type.toLowerCase()]; │ │ │ │ │ - if (parser) { │ │ │ │ │ - parts.push(parser.apply(this, [child])) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + var nRequests = (types[OpenLayers.State.INSERT].length > 0 ? 1 : 0) + types[OpenLayers.State.UPDATE].length + types[OpenLayers.State.DELETE].length; │ │ │ │ │ + var success = true; │ │ │ │ │ + var finalResponse = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: requestFeatures │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + function insertCallback(response) { │ │ │ │ │ + var len = response.features ? response.features.length : 0; │ │ │ │ │ + var fids = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + fids[i] = response.features[i].fid │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Geometry.Collection(parts) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - parseAttributes: function(node) { │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - var edNodes = node.getElementsByTagName("ExtendedData"); │ │ │ │ │ - if (edNodes.length) { │ │ │ │ │ - attributes = this.parseExtendedData(edNodes[0]) │ │ │ │ │ + finalResponse.insertIds = fids; │ │ │ │ │ + callback.apply(this, [response]) │ │ │ │ │ } │ │ │ │ │ - var child, grandchildren, grandchild; │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ - child = children[i]; │ │ │ │ │ - if (child.nodeType == 1) { │ │ │ │ │ - grandchildren = child.childNodes; │ │ │ │ │ - if (grandchildren.length >= 1 && grandchildren.length <= 3) { │ │ │ │ │ - var grandchild; │ │ │ │ │ - switch (grandchildren.length) { │ │ │ │ │ - case 1: │ │ │ │ │ - grandchild = grandchildren[0]; │ │ │ │ │ - break; │ │ │ │ │ - case 2: │ │ │ │ │ - var c1 = grandchildren[0]; │ │ │ │ │ - var c2 = grandchildren[1]; │ │ │ │ │ - grandchild = c1.nodeType == 3 || c1.nodeType == 4 ? c1 : c2; │ │ │ │ │ - break; │ │ │ │ │ - case 3: │ │ │ │ │ - default: │ │ │ │ │ - grandchild = grandchildren[1]; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - if (grandchild.nodeType == 3 || grandchild.nodeType == 4) { │ │ │ │ │ - var name = child.prefix ? child.nodeName.split(":")[1] : child.nodeName; │ │ │ │ │ - var value = OpenLayers.Util.getXmlNodeValue(grandchild); │ │ │ │ │ - if (value) { │ │ │ │ │ - value = value.replace(this.regExes.trimSpace, ""); │ │ │ │ │ - attributes[name] = value │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + │ │ │ │ │ + function callback(response) { │ │ │ │ │ + this.callUserCallback(response, options); │ │ │ │ │ + success = success && response.success(); │ │ │ │ │ + nResponses++; │ │ │ │ │ + if (nResponses >= nRequests) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + finalResponse.code = success ? OpenLayers.Protocol.Response.SUCCESS : OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + options.callback.apply(options.scope, [finalResponse]) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return attributes │ │ │ │ │ - }, │ │ │ │ │ - parseExtendedData: function(node) { │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - var i, len, data, key; │ │ │ │ │ - var dataNodes = node.getElementsByTagName("Data"); │ │ │ │ │ - for (i = 0, len = dataNodes.length; i < len; i++) { │ │ │ │ │ - data = dataNodes[i]; │ │ │ │ │ - key = data.getAttribute("name"); │ │ │ │ │ - var ed = {}; │ │ │ │ │ - var valueNode = data.getElementsByTagName("value"); │ │ │ │ │ - if (valueNode.length) { │ │ │ │ │ - ed["value"] = this.getChildValue(valueNode[0]) │ │ │ │ │ - } │ │ │ │ │ - if (this.kvpAttributes) { │ │ │ │ │ - attributes[key] = ed["value"] │ │ │ │ │ - } else { │ │ │ │ │ - var nameNode = data.getElementsByTagName("displayName"); │ │ │ │ │ - if (nameNode.length) { │ │ │ │ │ - ed["displayName"] = this.getChildValue(nameNode[0]) │ │ │ │ │ - } │ │ │ │ │ - attributes[key] = ed │ │ │ │ │ - } │ │ │ │ │ + var queue = types[OpenLayers.State.INSERT]; │ │ │ │ │ + if (queue.length > 0) { │ │ │ │ │ + resp.push(this.create(queue, OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: insertCallback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options.create))) │ │ │ │ │ } │ │ │ │ │ - var simpleDataNodes = node.getElementsByTagName("SimpleData"); │ │ │ │ │ - for (i = 0, len = simpleDataNodes.length; i < len; i++) { │ │ │ │ │ - var ed = {}; │ │ │ │ │ - data = simpleDataNodes[i]; │ │ │ │ │ - key = data.getAttribute("name"); │ │ │ │ │ - ed["value"] = this.getChildValue(data); │ │ │ │ │ - if (this.kvpAttributes) { │ │ │ │ │ - attributes[key] = ed["value"] │ │ │ │ │ - } else { │ │ │ │ │ - ed["displayName"] = key; │ │ │ │ │ - attributes[key] = ed │ │ │ │ │ - } │ │ │ │ │ + queue = types[OpenLayers.State.UPDATE]; │ │ │ │ │ + for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ + resp.push(this.update(queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: callback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options.update))) │ │ │ │ │ } │ │ │ │ │ - return attributes │ │ │ │ │ - }, │ │ │ │ │ - parseProperty: function(xmlNode, namespace, tagName) { │ │ │ │ │ - var value; │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(xmlNode, namespace, tagName); │ │ │ │ │ - try { │ │ │ │ │ - value = OpenLayers.Util.getXmlNodeValue(nodeList[0]) │ │ │ │ │ - } catch (e) { │ │ │ │ │ - value = null │ │ │ │ │ + queue = types[OpenLayers.State.DELETE]; │ │ │ │ │ + for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ + resp.push(this["delete"](queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: callback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options["delete"]))) │ │ │ │ │ } │ │ │ │ │ - return value │ │ │ │ │ + return resp │ │ │ │ │ }, │ │ │ │ │ - write: function(features) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ - features = [features] │ │ │ │ │ - } │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "kml"); │ │ │ │ │ - var folder = this.createFolderXML(); │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - folder.appendChild(this.createPlacemarkXML(features[i])) │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + response.priv.abort() │ │ │ │ │ } │ │ │ │ │ - kml.appendChild(folder); │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [kml]) │ │ │ │ │ }, │ │ │ │ │ - createFolderXML: function() { │ │ │ │ │ - var folder = this.createElementNS(this.kmlns, "Folder"); │ │ │ │ │ - if (this.foldersName) { │ │ │ │ │ - var folderName = this.createElementNS(this.kmlns, "name"); │ │ │ │ │ - var folderNameText = this.createTextNode(this.foldersName); │ │ │ │ │ - folderName.appendChild(folderNameText); │ │ │ │ │ - folder.appendChild(folderName) │ │ │ │ │ - } │ │ │ │ │ - if (this.foldersDesc) { │ │ │ │ │ - var folderDesc = this.createElementNS(this.kmlns, "description"); │ │ │ │ │ - var folderDescText = this.createTextNode(this.foldersDesc); │ │ │ │ │ - folderDesc.appendChild(folderDescText); │ │ │ │ │ - folder.appendChild(folderDesc) │ │ │ │ │ + callUserCallback: function(resp, options) { │ │ │ │ │ + var opt = options[resp.requestType]; │ │ │ │ │ + if (opt && opt.callback) { │ │ │ │ │ + opt.callback.call(opt.scope, resp) │ │ │ │ │ } │ │ │ │ │ - return folder │ │ │ │ │ }, │ │ │ │ │ - createPlacemarkXML: function(feature) { │ │ │ │ │ - var placemarkName = this.createElementNS(this.kmlns, "name"); │ │ │ │ │ - var label = feature.style && feature.style.label ? feature.style.label : feature.id; │ │ │ │ │ - var name = feature.attributes.name || label; │ │ │ │ │ - placemarkName.appendChild(this.createTextNode(name)); │ │ │ │ │ - var placemarkDesc = this.createElementNS(this.kmlns, "description"); │ │ │ │ │ - var desc = feature.attributes.description || this.placemarksDesc; │ │ │ │ │ - placemarkDesc.appendChild(this.createTextNode(desc)); │ │ │ │ │ - var placemarkNode = this.createElementNS(this.kmlns, "Placemark"); │ │ │ │ │ - if (feature.fid != null) { │ │ │ │ │ - placemarkNode.setAttribute("id", feature.fid) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.HTTP" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Protocol.WFS.v1 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + version: null, │ │ │ │ │ + srsName: "EPSG:4326", │ │ │ │ │ + featureType: null, │ │ │ │ │ + featureNS: null, │ │ │ │ │ + geometryName: "the_geom", │ │ │ │ │ + schema: null, │ │ │ │ │ + featurePrefix: "feature", │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + readFormat: null, │ │ │ │ │ + readOptions: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!options.format) { │ │ │ │ │ + this.format = OpenLayers.Format.WFST(OpenLayers.Util.extend({ │ │ │ │ │ + version: this.version, │ │ │ │ │ + featureType: this.featureType, │ │ │ │ │ + featureNS: this.featureNS, │ │ │ │ │ + featurePrefix: this.featurePrefix, │ │ │ │ │ + geometryName: this.geometryName, │ │ │ │ │ + srsName: this.srsName, │ │ │ │ │ + schema: this.schema │ │ │ │ │ + }, this.formatOptions)) │ │ │ │ │ } │ │ │ │ │ - placemarkNode.appendChild(placemarkName); │ │ │ │ │ - placemarkNode.appendChild(placemarkDesc); │ │ │ │ │ - var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ - placemarkNode.appendChild(geometryNode); │ │ │ │ │ - if (feature.attributes) { │ │ │ │ │ - var edNode = this.buildExtendedData(feature.attributes); │ │ │ │ │ - if (edNode) { │ │ │ │ │ - placemarkNode.appendChild(edNode) │ │ │ │ │ - } │ │ │ │ │ + if (!options.geometryName && parseFloat(this.format.version) > 1) { │ │ │ │ │ + this.setGeometryName(null) │ │ │ │ │ } │ │ │ │ │ - return placemarkNode │ │ │ │ │ }, │ │ │ │ │ - buildGeometryNode: function(geometry) { │ │ │ │ │ - var className = geometry.CLASS_NAME; │ │ │ │ │ - var type = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ - var builder = this.buildGeometry[type.toLowerCase()]; │ │ │ │ │ - var node = null; │ │ │ │ │ - if (builder) { │ │ │ │ │ - node = builder.apply(this, [geometry]) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.options && !this.options.format) { │ │ │ │ │ + this.format.destroy() │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ + this.format = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ }, │ │ │ │ │ - buildGeometry: { │ │ │ │ │ - point: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "Point"); │ │ │ │ │ - kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - return kml │ │ │ │ │ - }, │ │ │ │ │ - multipoint: function(geometry) { │ │ │ │ │ - return this.buildGeometry.collection.apply(this, [geometry]) │ │ │ │ │ - }, │ │ │ │ │ - linestring: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "LineString"); │ │ │ │ │ - kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - return kml │ │ │ │ │ - }, │ │ │ │ │ - multilinestring: function(geometry) { │ │ │ │ │ - return this.buildGeometry.collection.apply(this, [geometry]) │ │ │ │ │ - }, │ │ │ │ │ - linearring: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "LinearRing"); │ │ │ │ │ - kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - return kml │ │ │ │ │ - }, │ │ │ │ │ - polygon: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "Polygon"); │ │ │ │ │ - var rings = geometry.components; │ │ │ │ │ - var ringMember, ringGeom, type; │ │ │ │ │ - for (var i = 0, len = rings.length; i < len; ++i) { │ │ │ │ │ - type = i == 0 ? "outerBoundaryIs" : "innerBoundaryIs"; │ │ │ │ │ - ringMember = this.createElementNS(this.kmlns, type); │ │ │ │ │ - ringGeom = this.buildGeometry.linearring.apply(this, [rings[i]]); │ │ │ │ │ - ringMember.appendChild(ringGeom); │ │ │ │ │ - kml.appendChild(ringMember) │ │ │ │ │ - } │ │ │ │ │ - return kml │ │ │ │ │ - }, │ │ │ │ │ - multipolygon: function(geometry) { │ │ │ │ │ - return this.buildGeometry.collection.apply(this, [geometry]) │ │ │ │ │ - }, │ │ │ │ │ - collection: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "MultiGeometry"); │ │ │ │ │ - var child; │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; ++i) { │ │ │ │ │ - child = this.buildGeometryNode.apply(this, [geometry.components[i]]); │ │ │ │ │ - if (child) { │ │ │ │ │ - kml.appendChild(child) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return kml │ │ │ │ │ - } │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + var data = OpenLayers.Format.XML.prototype.write.apply(this.format, [this.format.writeNode("wfs:GetFeature", options)]); │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: data │ │ │ │ │ + }); │ │ │ │ │ + return response │ │ │ │ │ }, │ │ │ │ │ - buildCoordinatesNode: function(geometry) { │ │ │ │ │ - var coordinatesNode = this.createElementNS(this.kmlns, "coordinates"); │ │ │ │ │ - var path; │ │ │ │ │ - var points = geometry.components; │ │ │ │ │ - if (points) { │ │ │ │ │ - var point; │ │ │ │ │ - var numPoints = points.length; │ │ │ │ │ - var parts = new Array(numPoints); │ │ │ │ │ - for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ - point = points[i]; │ │ │ │ │ - parts[i] = this.buildCoordinates(point) │ │ │ │ │ - } │ │ │ │ │ - path = parts.join(" ") │ │ │ │ │ - } else { │ │ │ │ │ - path = this.buildCoordinates(geometry) │ │ │ │ │ - } │ │ │ │ │ - var txtNode = this.createTextNode(path); │ │ │ │ │ - coordinatesNode.appendChild(txtNode); │ │ │ │ │ - return coordinatesNode │ │ │ │ │ + setFeatureType: function(featureType) { │ │ │ │ │ + this.featureType = featureType; │ │ │ │ │ + this.format.featureType = featureType │ │ │ │ │ }, │ │ │ │ │ - buildCoordinates: function(point) { │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - point = point.clone(); │ │ │ │ │ - point.transform(this.internalProjection, this.externalProjection) │ │ │ │ │ - } │ │ │ │ │ - return point.x + "," + point.y │ │ │ │ │ + setGeometryName: function(geometryName) { │ │ │ │ │ + this.geometryName = geometryName; │ │ │ │ │ + this.format.geometryName = geometryName │ │ │ │ │ }, │ │ │ │ │ - buildExtendedData: function(attributes) { │ │ │ │ │ - var extendedData = this.createElementNS(this.kmlns, "ExtendedData"); │ │ │ │ │ - for (var attributeName in attributes) { │ │ │ │ │ - if (attributes[attributeName] && attributeName != "name" && attributeName != "description" && attributeName != "styleUrl") { │ │ │ │ │ - var data = this.createElementNS(this.kmlns, "Data"); │ │ │ │ │ - data.setAttribute("name", attributeName); │ │ │ │ │ - var value = this.createElementNS(this.kmlns, "value"); │ │ │ │ │ - if (typeof attributes[attributeName] == "object") { │ │ │ │ │ - if (attributes[attributeName].value) { │ │ │ │ │ - value.appendChild(this.createTextNode(attributes[attributeName].value)) │ │ │ │ │ - } │ │ │ │ │ - if (attributes[attributeName].displayName) { │ │ │ │ │ - var displayName = this.createElementNS(this.kmlns, "displayName"); │ │ │ │ │ - displayName.appendChild(this.getXMLDoc().createCDATASection(attributes[attributeName].displayName)); │ │ │ │ │ - data.appendChild(displayName) │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + var result = this.parseResponse(request, options.readOptions); │ │ │ │ │ + if (result && result.success !== false) { │ │ │ │ │ + if (options.readOptions && options.readOptions.output == "object") { │ │ │ │ │ + OpenLayers.Util.extend(response, result) │ │ │ │ │ + } else { │ │ │ │ │ + response.features = result │ │ │ │ │ } │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ } else { │ │ │ │ │ - value.appendChild(this.createTextNode(attributes[attributeName])) │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + response.error = result │ │ │ │ │ } │ │ │ │ │ - data.appendChild(value); │ │ │ │ │ - extendedData.appendChild(data) │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ } │ │ │ │ │ + options.callback.call(options.scope, response) │ │ │ │ │ } │ │ │ │ │ - if (this.isSimpleContent(extendedData)) { │ │ │ │ │ - return null │ │ │ │ │ - } else { │ │ │ │ │ - return extendedData │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.KML" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler = OpenLayers.Class({ │ │ │ │ │ - id: null, │ │ │ │ │ - control: null, │ │ │ │ │ - map: null, │ │ │ │ │ - keyMask: null, │ │ │ │ │ - active: false, │ │ │ │ │ - evt: null, │ │ │ │ │ - touch: false, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.control = control; │ │ │ │ │ - this.callbacks = callbacks; │ │ │ │ │ - var map = this.map || control.map; │ │ │ │ │ - if (map) { │ │ │ │ │ - this.setMap(map) │ │ │ │ │ - } │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.map = map │ │ │ │ │ - }, │ │ │ │ │ - checkModifiers: function(evt) { │ │ │ │ │ - if (this.keyMask == null) { │ │ │ │ │ - return true │ │ │ │ │ + parseResponse: function(request, options) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ } │ │ │ │ │ - var keyModifiers = (evt.shiftKey ? OpenLayers.Handler.MOD_SHIFT : 0) | (evt.ctrlKey ? OpenLayers.Handler.MOD_CTRL : 0) | (evt.altKey ? OpenLayers.Handler.MOD_ALT : 0) | (evt.metaKey ? OpenLayers.Handler.MOD_META : 0); │ │ │ │ │ - return keyModifiers == this.keyMask │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - return false │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null │ │ │ │ │ } │ │ │ │ │ - var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.register(events[i], this[events[i]]) │ │ │ │ │ + var result = this.readFormat !== null ? this.readFormat.read(doc) : this.format.read(doc, options); │ │ │ │ │ + if (!this.featureNS) { │ │ │ │ │ + var format = this.readFormat || this.format; │ │ │ │ │ + this.featureNS = format.featureNS; │ │ │ │ │ + format.autoConfig = false; │ │ │ │ │ + if (!this.geometryName) { │ │ │ │ │ + this.setGeometryName(format.geometryName) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.active = true; │ │ │ │ │ - return true │ │ │ │ │ + return result │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.unregister(events[i], this[events[i]]) │ │ │ │ │ + commit: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "commit", │ │ │ │ │ + reqFeatures: features │ │ │ │ │ + }); │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(features, options), │ │ │ │ │ + callback: this.createCallback(this.handleCommit, response, options) │ │ │ │ │ + }); │ │ │ │ │ + return response │ │ │ │ │ + }, │ │ │ │ │ + handleCommit: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + var data = request.responseXML; │ │ │ │ │ + if (!data || !data.documentElement) { │ │ │ │ │ + data = request.responseText │ │ │ │ │ + } │ │ │ │ │ + var obj = this.format.read(data) || {}; │ │ │ │ │ + response.insertIds = obj.insertIds || []; │ │ │ │ │ + if (obj.success) { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + response.error = obj │ │ │ │ │ } │ │ │ │ │ + options.callback.call(options.scope, response) │ │ │ │ │ } │ │ │ │ │ - this.touch = false; │ │ │ │ │ - this.active = false; │ │ │ │ │ - return true │ │ │ │ │ }, │ │ │ │ │ - startTouch: function() { │ │ │ │ │ - if (!this.touch) { │ │ │ │ │ - this.touch = true; │ │ │ │ │ - var events = ["mousedown", "mouseup", "mousemove", "click", "dblclick", "mouseout"]; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.unregister(events[i], this[events[i]]) │ │ │ │ │ - } │ │ │ │ │ + filterDelete: function(filter, options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "commit" │ │ │ │ │ + }); │ │ │ │ │ + var root = this.format.createElementNSPlus("wfs:Transaction", { │ │ │ │ │ + attributes: { │ │ │ │ │ + service: "WFS", │ │ │ │ │ + version: this.version │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + var deleteNode = this.format.createElementNSPlus("wfs:Delete", { │ │ │ │ │ + attributes: { │ │ │ │ │ + typeName: (options.featureNS ? this.featurePrefix + ":" : "") + options.featureType │ │ │ │ │ } │ │ │ │ │ + }); │ │ │ │ │ + if (options.featureNS) { │ │ │ │ │ + deleteNode.setAttribute("xmlns:" + this.featurePrefix, options.featureNS) │ │ │ │ │ } │ │ │ │ │ + var filterNode = this.format.writeNode("ogc:Filter", filter); │ │ │ │ │ + deleteNode.appendChild(filterNode); │ │ │ │ │ + root.appendChild(deleteNode); │ │ │ │ │ + var data = OpenLayers.Format.XML.prototype.write.apply(this.format, [root]); │ │ │ │ │ + return OpenLayers.Request.POST({ │ │ │ │ │ + url: this.url, │ │ │ │ │ + callback: options.callback || function() {}, │ │ │ │ │ + data: data │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - callback: function(name, args) { │ │ │ │ │ - if (name && this.callbacks[name]) { │ │ │ │ │ - this.callbacks[name].apply(this.control, args) │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + response.priv.abort() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - register: function(name, method) { │ │ │ │ │ - this.map.events.registerPriority(name, this, method); │ │ │ │ │ - this.map.events.registerPriority(name, this, this.setEvent) │ │ │ │ │ - }, │ │ │ │ │ - unregister: function(name, method) { │ │ │ │ │ - this.map.events.unregister(name, this, method); │ │ │ │ │ - this.map.events.unregister(name, this, this.setEvent) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.WFS.v1" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format = OpenLayers.Class({ │ │ │ │ │ + options: null, │ │ │ │ │ + externalProjection: null, │ │ │ │ │ + internalProjection: null, │ │ │ │ │ + data: null, │ │ │ │ │ + keepData: false, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.options = options │ │ │ │ │ }, │ │ │ │ │ - setEvent: function(evt) { │ │ │ │ │ - this.evt = evt; │ │ │ │ │ - return true │ │ │ │ │ + destroy: function() {}, │ │ │ │ │ + read: function(data) { │ │ │ │ │ + throw new Error("Read not implemented.") │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.control = this.map = null │ │ │ │ │ + write: function(object) { │ │ │ │ │ + throw new Error("Write not implemented.") │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Handler.MOD_NONE = 0; │ │ │ │ │ -OpenLayers.Handler.MOD_SHIFT = 1; │ │ │ │ │ -OpenLayers.Handler.MOD_CTRL = 2; │ │ │ │ │ -OpenLayers.Handler.MOD_ALT = 4; │ │ │ │ │ -OpenLayers.Handler.MOD_META = 8; │ │ │ │ │ -OpenLayers.Handler.Point = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - point: null, │ │ │ │ │ - layer: null, │ │ │ │ │ - multi: false, │ │ │ │ │ - citeCompliant: false, │ │ │ │ │ - mouseDown: false, │ │ │ │ │ - stoppedDown: null, │ │ │ │ │ - lastDown: null, │ │ │ │ │ - lastUp: null, │ │ │ │ │ - persist: false, │ │ │ │ │ - stopDown: false, │ │ │ │ │ - stopUp: false, │ │ │ │ │ - layerOptions: null, │ │ │ │ │ - pixelTolerance: 5, │ │ │ │ │ - lastTouchPx: null, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ - this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style["default"], {}) │ │ │ │ │ +OpenLayers.Format.XML = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ + namespaces: null, │ │ │ │ │ + namespaceAlias: null, │ │ │ │ │ + defaultPrefix: null, │ │ │ │ │ + readers: {}, │ │ │ │ │ + writers: {}, │ │ │ │ │ + xmldom: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + if (window.ActiveXObject) { │ │ │ │ │ + this.xmldom = new ActiveXObject("Microsoft.XMLDOM") │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (!OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - return false │ │ │ │ │ + OpenLayers.Format.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.namespaces = OpenLayers.Util.extend({}, this.namespaces); │ │ │ │ │ + this.namespaceAlias = {}; │ │ │ │ │ + for (var alias in this.namespaces) { │ │ │ │ │ + this.namespaceAlias[this.namespaces[alias]] = alias │ │ │ │ │ } │ │ │ │ │ - var options = OpenLayers.Util.extend({ │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - calculateInRange: OpenLayers.Function.True, │ │ │ │ │ - wrapDateLine: this.citeCompliant │ │ │ │ │ - }, this.layerOptions); │ │ │ │ │ - this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ - this.map.addLayer(this.layer); │ │ │ │ │ - return true │ │ │ │ │ }, │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.point]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.point], { │ │ │ │ │ - silent: true │ │ │ │ │ - }) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.xmldom = null; │ │ │ │ │ + OpenLayers.Format.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (!OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - return false │ │ │ │ │ + setNamespace: function(alias, uri) { │ │ │ │ │ + this.namespaces[alias] = uri; │ │ │ │ │ + this.namespaceAlias[uri] = alias │ │ │ │ │ + }, │ │ │ │ │ + read: function(text) { │ │ │ │ │ + var index = text.indexOf("<"); │ │ │ │ │ + if (index > 0) { │ │ │ │ │ + text = text.substring(index) │ │ │ │ │ } │ │ │ │ │ - this.cancel(); │ │ │ │ │ - if (this.layer.map != null) { │ │ │ │ │ - this.destroyFeature(true); │ │ │ │ │ - this.layer.destroy(false) │ │ │ │ │ + var node = OpenLayers.Util.Try(OpenLayers.Function.bind(function() { │ │ │ │ │ + var xmldom; │ │ │ │ │ + if (window.ActiveXObject && !this.xmldom) { │ │ │ │ │ + xmldom = new ActiveXObject("Microsoft.XMLDOM") │ │ │ │ │ + } else { │ │ │ │ │ + xmldom = this.xmldom │ │ │ │ │ + } │ │ │ │ │ + xmldom.loadXML(text); │ │ │ │ │ + return xmldom │ │ │ │ │ + }, this), function() { │ │ │ │ │ + return (new DOMParser).parseFromString(text, "text/xml") │ │ │ │ │ + }, function() { │ │ │ │ │ + var req = new XMLHttpRequest; │ │ │ │ │ + req.open("GET", "data:" + "text/xml" + ";charset=utf-8," + encodeURIComponent(text), false); │ │ │ │ │ + if (req.overrideMimeType) { │ │ │ │ │ + req.overrideMimeType("text/xml") │ │ │ │ │ + } │ │ │ │ │ + req.send(null); │ │ │ │ │ + return req.responseXML │ │ │ │ │ + }); │ │ │ │ │ + if (this.keepData) { │ │ │ │ │ + this.data = node │ │ │ │ │ } │ │ │ │ │ - this.layer = null; │ │ │ │ │ - return true │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - if (this.layer && (force || !this.persist)) { │ │ │ │ │ - this.layer.destroyFeatures() │ │ │ │ │ + write: function(node) { │ │ │ │ │ + var data; │ │ │ │ │ + if (this.xmldom) { │ │ │ │ │ + data = node.xml │ │ │ │ │ + } else { │ │ │ │ │ + var serializer = new XMLSerializer; │ │ │ │ │ + if (node.nodeType == 1) { │ │ │ │ │ + var doc = document.implementation.createDocument("", "", null); │ │ │ │ │ + if (doc.importNode) { │ │ │ │ │ + node = doc.importNode(node, true) │ │ │ │ │ + } │ │ │ │ │ + doc.appendChild(node); │ │ │ │ │ + data = serializer.serializeToString(doc) │ │ │ │ │ + } else { │ │ │ │ │ + data = serializer.serializeToString(node) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.point = null │ │ │ │ │ + return data │ │ │ │ │ }, │ │ │ │ │ - destroyPersistedFeature: function() { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - if (layer && layer.features.length > 1) { │ │ │ │ │ - this.layer.features[0].destroy() │ │ │ │ │ + createElementNS: function(uri, name) { │ │ │ │ │ + var element; │ │ │ │ │ + if (this.xmldom) { │ │ │ │ │ + if (typeof uri == "string") { │ │ │ │ │ + element = this.xmldom.createNode(1, name, uri) │ │ │ │ │ + } else { │ │ │ │ │ + element = this.xmldom.createNode(1, name, "") │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + element = document.createElementNS(uri, name) │ │ │ │ │ } │ │ │ │ │ + return element │ │ │ │ │ }, │ │ │ │ │ - finalize: function(cancel) { │ │ │ │ │ - var key = cancel ? "cancel" : "done"; │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - this.lastDown = null; │ │ │ │ │ - this.lastUp = null; │ │ │ │ │ - this.lastTouchPx = null; │ │ │ │ │ - this.callback(key, [this.geometryClone()]); │ │ │ │ │ - this.destroyFeature(cancel) │ │ │ │ │ - }, │ │ │ │ │ - cancel: function() { │ │ │ │ │ - this.finalize(true) │ │ │ │ │ - }, │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - modifyFeature: function(pixel) { │ │ │ │ │ - if (!this.point) { │ │ │ │ │ - this.createFeature(pixel) │ │ │ │ │ + createDocumentFragment: function() { │ │ │ │ │ + var element; │ │ │ │ │ + if (this.xmldom) { │ │ │ │ │ + element = this.xmldom.createDocumentFragment() │ │ │ │ │ + } else { │ │ │ │ │ + element = document.createDocumentFragment() │ │ │ │ │ } │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point.geometry.x = lonlat.lon; │ │ │ │ │ - this.point.geometry.y = lonlat.lat; │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.point, false]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.drawFeature() │ │ │ │ │ - }, │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.point, this.style) │ │ │ │ │ + return element │ │ │ │ │ }, │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.point && this.point.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiPoint([geometry]) │ │ │ │ │ + createTextNode: function(text) { │ │ │ │ │ + var node; │ │ │ │ │ + if (typeof text !== "string") { │ │ │ │ │ + text = String(text) │ │ │ │ │ } │ │ │ │ │ - return geometry │ │ │ │ │ - }, │ │ │ │ │ - geometryClone: function() { │ │ │ │ │ - var geom = this.getGeometry(); │ │ │ │ │ - return geom && geom.clone() │ │ │ │ │ - }, │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - return this.down(evt) │ │ │ │ │ - }, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - this.lastTouchPx = evt.xy; │ │ │ │ │ - return this.down(evt) │ │ │ │ │ - }, │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - return this.move(evt) │ │ │ │ │ - }, │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - this.lastTouchPx = evt.xy; │ │ │ │ │ - return this.move(evt) │ │ │ │ │ - }, │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - return this.up(evt) │ │ │ │ │ - }, │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - evt.xy = this.lastTouchPx; │ │ │ │ │ - return this.up(evt) │ │ │ │ │ - }, │ │ │ │ │ - down: function(evt) { │ │ │ │ │ - this.mouseDown = true; │ │ │ │ │ - this.lastDown = evt.xy; │ │ │ │ │ - if (!this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy) │ │ │ │ │ + if (this.xmldom) { │ │ │ │ │ + node = this.xmldom.createTextNode(text) │ │ │ │ │ + } else { │ │ │ │ │ + node = document.createTextNode(text) │ │ │ │ │ } │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - return !this.stopDown │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ - this.modifyFeature(evt.xy) │ │ │ │ │ + getElementsByTagNameNS: function(node, uri, name) { │ │ │ │ │ + var elements = []; │ │ │ │ │ + if (node.getElementsByTagNameNS) { │ │ │ │ │ + elements = node.getElementsByTagNameNS(uri, name) │ │ │ │ │ + } else { │ │ │ │ │ + var allNodes = node.getElementsByTagName("*"); │ │ │ │ │ + var potentialNode, fullName; │ │ │ │ │ + for (var i = 0, len = allNodes.length; i < len; ++i) { │ │ │ │ │ + potentialNode = allNodes[i]; │ │ │ │ │ + fullName = potentialNode.prefix ? potentialNode.prefix + ":" + name : name; │ │ │ │ │ + if (name == "*" || fullName == potentialNode.nodeName) { │ │ │ │ │ + if (uri == "*" || uri == potentialNode.namespaceURI) { │ │ │ │ │ + elements.push(potentialNode) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return true │ │ │ │ │ + return elements │ │ │ │ │ }, │ │ │ │ │ - up: function(evt) { │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - if (!this.checkModifiers(evt)) { │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - if (this.lastUp && this.lastUp.equals(evt.xy)) { │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - if (this.lastDown && this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance)) { │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy) │ │ │ │ │ - } │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature() │ │ │ │ │ - } │ │ │ │ │ - this.lastUp = evt.xy; │ │ │ │ │ - this.finalize(); │ │ │ │ │ - return !this.stopUp │ │ │ │ │ + getAttributeNodeNS: function(node, uri, name) { │ │ │ │ │ + var attributeNode = null; │ │ │ │ │ + if (node.getAttributeNodeNS) { │ │ │ │ │ + attributeNode = node.getAttributeNodeNS(uri, name) │ │ │ │ │ } else { │ │ │ │ │ - return true │ │ │ │ │ + var attributes = node.attributes; │ │ │ │ │ + var potentialNode, fullName; │ │ │ │ │ + for (var i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ + potentialNode = attributes[i]; │ │ │ │ │ + if (potentialNode.namespaceURI == uri) { │ │ │ │ │ + fullName = potentialNode.prefix ? potentialNode.prefix + ":" + name : name; │ │ │ │ │ + if (fullName == potentialNode.nodeName) { │ │ │ │ │ + attributeNode = potentialNode; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return attributeNode │ │ │ │ │ }, │ │ │ │ │ - mouseout: function(evt) { │ │ │ │ │ - if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - this.mouseDown = false │ │ │ │ │ + getAttributeNS: function(node, uri, name) { │ │ │ │ │ + var attributeValue = ""; │ │ │ │ │ + if (node.getAttributeNS) { │ │ │ │ │ + attributeValue = node.getAttributeNS(uri, name) || "" │ │ │ │ │ + } else { │ │ │ │ │ + var attributeNode = this.getAttributeNodeNS(node, uri, name); │ │ │ │ │ + if (attributeNode) { │ │ │ │ │ + attributeValue = attributeNode.nodeValue │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return attributeValue │ │ │ │ │ }, │ │ │ │ │ - passesTolerance: function(pixel1, pixel2, tolerance) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - if (tolerance != null && pixel1 && pixel2) { │ │ │ │ │ - var dist = pixel1.distanceTo(pixel2); │ │ │ │ │ - if (dist > tolerance) { │ │ │ │ │ - passes = false │ │ │ │ │ + getChildValue: function(node, def) { │ │ │ │ │ + var value = def || ""; │ │ │ │ │ + if (node) { │ │ │ │ │ + for (var child = node.firstChild; child; child = child.nextSibling) { │ │ │ │ │ + switch (child.nodeType) { │ │ │ │ │ + case 3: │ │ │ │ │ + case 4: │ │ │ │ │ + value += child.nodeValue │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return passes │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Point" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.Path = OpenLayers.Class(OpenLayers.Handler.Point, { │ │ │ │ │ - line: null, │ │ │ │ │ - maxVertices: null, │ │ │ │ │ - doubleTouchTolerance: 20, │ │ │ │ │ - freehand: false, │ │ │ │ │ - freehandToggle: "shiftKey", │ │ │ │ │ - timerId: null, │ │ │ │ │ - redoStack: null, │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.line = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.LineString([this.point.geometry])); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.line, this.point], { │ │ │ │ │ - silent: true │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - OpenLayers.Handler.Point.prototype.destroyFeature.call(this, force); │ │ │ │ │ - this.line = null │ │ │ │ │ + return value │ │ │ │ │ }, │ │ │ │ │ - destroyPersistedFeature: function() { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - if (layer && layer.features.length > 2) { │ │ │ │ │ - this.layer.features[0].destroy() │ │ │ │ │ + isSimpleContent: function(node) { │ │ │ │ │ + var simple = true; │ │ │ │ │ + for (var child = node.firstChild; child; child = child.nextSibling) { │ │ │ │ │ + if (child.nodeType === 1) { │ │ │ │ │ + simple = false; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return simple │ │ │ │ │ }, │ │ │ │ │ - removePoint: function() { │ │ │ │ │ - if (this.point) { │ │ │ │ │ - this.layer.removeFeatures([this.point]) │ │ │ │ │ + contentType: function(node) { │ │ │ │ │ + var simple = false, │ │ │ │ │ + complex = false; │ │ │ │ │ + var type = OpenLayers.Format.XML.CONTENT_TYPE.EMPTY; │ │ │ │ │ + for (var child = node.firstChild; child; child = child.nextSibling) { │ │ │ │ │ + switch (child.nodeType) { │ │ │ │ │ + case 1: │ │ │ │ │ + complex = true; │ │ │ │ │ + break; │ │ │ │ │ + case 8: │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + simple = true │ │ │ │ │ + } │ │ │ │ │ + if (complex && simple) { │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - addPoint: function(pixel) { │ │ │ │ │ - this.layer.removeFeatures([this.point]); │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat)); │ │ │ │ │ - this.line.geometry.addComponent(this.point.geometry, this.line.geometry.components.length); │ │ │ │ │ - this.layer.addFeatures([this.point]); │ │ │ │ │ - this.callback("point", [this.point.geometry, this.getGeometry()]); │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - delete this.redoStack │ │ │ │ │ - }, │ │ │ │ │ - insertXY: function(x, y) { │ │ │ │ │ - this.line.geometry.addComponent(new OpenLayers.Geometry.Point(x, y), this.getCurrentPointIndex()); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - delete this.redoStack │ │ │ │ │ - }, │ │ │ │ │ - insertDeltaXY: function(dx, dy) { │ │ │ │ │ - var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ - var p0 = this.line.geometry.components[previousIndex]; │ │ │ │ │ - if (p0 && !isNaN(p0.x) && !isNaN(p0.y)) { │ │ │ │ │ - this.insertXY(p0.x + dx, p0.y + dy) │ │ │ │ │ + if (complex && simple) { │ │ │ │ │ + type = OpenLayers.Format.XML.CONTENT_TYPE.MIXED │ │ │ │ │ + } else if (complex) { │ │ │ │ │ + return OpenLayers.Format.XML.CONTENT_TYPE.COMPLEX │ │ │ │ │ + } else if (simple) { │ │ │ │ │ + return OpenLayers.Format.XML.CONTENT_TYPE.SIMPLE │ │ │ │ │ } │ │ │ │ │ + return type │ │ │ │ │ }, │ │ │ │ │ - insertDirectionLength: function(direction, length) { │ │ │ │ │ - direction *= Math.PI / 180; │ │ │ │ │ - var dx = length * Math.cos(direction); │ │ │ │ │ - var dy = length * Math.sin(direction); │ │ │ │ │ - this.insertDeltaXY(dx, dy) │ │ │ │ │ - }, │ │ │ │ │ - insertDeflectionLength: function(deflection, length) { │ │ │ │ │ - var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ - if (previousIndex > 0) { │ │ │ │ │ - var p1 = this.line.geometry.components[previousIndex]; │ │ │ │ │ - var p0 = this.line.geometry.components[previousIndex - 1]; │ │ │ │ │ - var theta = Math.atan2(p1.y - p0.y, p1.x - p0.x); │ │ │ │ │ - this.insertDirectionLength(theta * 180 / Math.PI + deflection, length) │ │ │ │ │ + hasAttributeNS: function(node, uri, name) { │ │ │ │ │ + var found = false; │ │ │ │ │ + if (node.hasAttributeNS) { │ │ │ │ │ + found = node.hasAttributeNS(uri, name) │ │ │ │ │ + } else { │ │ │ │ │ + found = !!this.getAttributeNodeNS(node, uri, name) │ │ │ │ │ } │ │ │ │ │ + return found │ │ │ │ │ }, │ │ │ │ │ - getCurrentPointIndex: function() { │ │ │ │ │ - return this.line.geometry.components.length - 1 │ │ │ │ │ - }, │ │ │ │ │ - undo: function() { │ │ │ │ │ - var geometry = this.line.geometry; │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var index = this.getCurrentPointIndex() - 1; │ │ │ │ │ - var target = components[index]; │ │ │ │ │ - var undone = geometry.removeComponent(target); │ │ │ │ │ - if (undone) { │ │ │ │ │ - if (this.touch && index > 0) { │ │ │ │ │ - components = geometry.components; │ │ │ │ │ - var lastpt = components[index - 1]; │ │ │ │ │ - var curptidx = this.getCurrentPointIndex(); │ │ │ │ │ - var curpt = components[curptidx]; │ │ │ │ │ - curpt.x = lastpt.x; │ │ │ │ │ - curpt.y = lastpt.y │ │ │ │ │ - } │ │ │ │ │ - if (!this.redoStack) { │ │ │ │ │ - this.redoStack = [] │ │ │ │ │ + setAttributeNS: function(node, uri, name, value) { │ │ │ │ │ + if (node.setAttributeNS) { │ │ │ │ │ + node.setAttributeNS(uri, name, value) │ │ │ │ │ + } else { │ │ │ │ │ + if (this.xmldom) { │ │ │ │ │ + if (uri) { │ │ │ │ │ + var attribute = node.ownerDocument.createNode(2, name, uri); │ │ │ │ │ + attribute.nodeValue = value; │ │ │ │ │ + node.setAttributeNode(attribute) │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttribute(name, value) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + throw "setAttributeNS not implemented" │ │ │ │ │ } │ │ │ │ │ - this.redoStack.push(target); │ │ │ │ │ - this.drawFeature() │ │ │ │ │ } │ │ │ │ │ - return undone │ │ │ │ │ }, │ │ │ │ │ - redo: function() { │ │ │ │ │ - var target = this.redoStack && this.redoStack.pop(); │ │ │ │ │ - if (target) { │ │ │ │ │ - this.line.geometry.addComponent(target, this.getCurrentPointIndex()); │ │ │ │ │ - this.drawFeature() │ │ │ │ │ + createElementNSPlus: function(name, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var uri = options.uri || this.namespaces[options.prefix]; │ │ │ │ │ + if (!uri) { │ │ │ │ │ + var loc = name.indexOf(":"); │ │ │ │ │ + uri = this.namespaces[name.substring(0, loc)] │ │ │ │ │ } │ │ │ │ │ - return !!target │ │ │ │ │ - }, │ │ │ │ │ - freehandMode: function(evt) { │ │ │ │ │ - return this.freehandToggle && evt[this.freehandToggle] ? !this.freehand : this.freehand │ │ │ │ │ - }, │ │ │ │ │ - modifyFeature: function(pixel, drawing) { │ │ │ │ │ - if (!this.line) { │ │ │ │ │ - this.createFeature(pixel) │ │ │ │ │ + if (!uri) { │ │ │ │ │ + uri = this.namespaces[this.defaultPrefix] │ │ │ │ │ } │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point.geometry.x = lonlat.lon; │ │ │ │ │ - this.point.geometry.y = lonlat.lat; │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.getSketch(), drawing]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.drawFeature() │ │ │ │ │ - }, │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.line, this.style); │ │ │ │ │ - this.layer.drawFeature(this.point, this.style) │ │ │ │ │ - }, │ │ │ │ │ - getSketch: function() { │ │ │ │ │ - return this.line │ │ │ │ │ - }, │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.line && this.line.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiLineString([geometry]) │ │ │ │ │ + var node = this.createElementNS(uri, name); │ │ │ │ │ + if (options.attributes) { │ │ │ │ │ + this.setAttributes(node, options.attributes) │ │ │ │ │ } │ │ │ │ │ - return geometry │ │ │ │ │ + var value = options.value; │ │ │ │ │ + if (value != null) { │ │ │ │ │ + node.appendChild(this.createTextNode(value)) │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - if (this.timerId && this.passesTolerance(this.lastTouchPx, evt.xy, this.doubleTouchTolerance)) { │ │ │ │ │ - this.finishGeometry(); │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - return false │ │ │ │ │ - } else { │ │ │ │ │ - if (this.timerId) { │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null │ │ │ │ │ + setAttributes: function(node, obj) { │ │ │ │ │ + var value, uri; │ │ │ │ │ + for (var name in obj) { │ │ │ │ │ + if (obj[name] != null && obj[name].toString) { │ │ │ │ │ + value = obj[name].toString(); │ │ │ │ │ + uri = this.namespaces[name.substring(0, name.indexOf(":"))] || null; │ │ │ │ │ + this.setAttributeNS(node, uri, name, value) │ │ │ │ │ } │ │ │ │ │ - this.timerId = window.setTimeout(OpenLayers.Function.bind(function() { │ │ │ │ │ - this.timerId = null │ │ │ │ │ - }, this), 300); │ │ │ │ │ - return OpenLayers.Handler.Point.prototype.touchstart.call(this, evt) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - down: function(evt) { │ │ │ │ │ - var stopDown = this.stopDown; │ │ │ │ │ - if (this.freehandMode(evt)) { │ │ │ │ │ - stopDown = true; │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ - } │ │ │ │ │ + readNode: function(node, obj) { │ │ │ │ │ + if (!obj) { │ │ │ │ │ + obj = {} │ │ │ │ │ } │ │ │ │ │ - if (!this.touch && (!this.lastDown || !this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance))) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp) │ │ │ │ │ + var group = this.readers[node.namespaceURI ? this.namespaceAlias[node.namespaceURI] : this.defaultPrefix]; │ │ │ │ │ + if (group) { │ │ │ │ │ + var local = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ + var reader = group[local] || group["*"]; │ │ │ │ │ + if (reader) { │ │ │ │ │ + reader.apply(this, [node, obj]) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.mouseDown = true; │ │ │ │ │ - this.lastDown = evt.xy; │ │ │ │ │ - this.stoppedDown = stopDown; │ │ │ │ │ - return !stopDown │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature() │ │ │ │ │ + readChildNodes: function(node, obj) { │ │ │ │ │ + if (!obj) { │ │ │ │ │ + obj = {} │ │ │ │ │ + } │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + var child; │ │ │ │ │ + for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ + child = children[i]; │ │ │ │ │ + if (child.nodeType == 1) { │ │ │ │ │ + this.readNode(child, obj) │ │ │ │ │ } │ │ │ │ │ - if (this.maxVertices && this.line && this.line.geometry.components.length === this.maxVertices) { │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize() │ │ │ │ │ + } │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + writeNode: function(name, obj, parent) { │ │ │ │ │ + var prefix, local; │ │ │ │ │ + var split = name.indexOf(":"); │ │ │ │ │ + if (split > 0) { │ │ │ │ │ + prefix = name.substring(0, split); │ │ │ │ │ + local = name.substring(split + 1) │ │ │ │ │ + } else { │ │ │ │ │ + if (parent) { │ │ │ │ │ + prefix = this.namespaceAlias[parent.namespaceURI] │ │ │ │ │ } else { │ │ │ │ │ - this.addPoint(evt.xy) │ │ │ │ │ + prefix = this.defaultPrefix │ │ │ │ │ } │ │ │ │ │ - return false │ │ │ │ │ + local = name │ │ │ │ │ } │ │ │ │ │ - if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp) │ │ │ │ │ + var child = this.writers[prefix][local].apply(this, [obj]); │ │ │ │ │ + if (parent) { │ │ │ │ │ + parent.appendChild(child) │ │ │ │ │ } │ │ │ │ │ - return true │ │ │ │ │ + return child │ │ │ │ │ }, │ │ │ │ │ - up: function(evt) { │ │ │ │ │ - if (this.mouseDown && (!this.lastUp || !this.lastUp.equals(evt.xy))) { │ │ │ │ │ - if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature() │ │ │ │ │ - } │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize() │ │ │ │ │ - } else { │ │ │ │ │ - if (this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance)) { │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy) │ │ │ │ │ - } │ │ │ │ │ - if (this.lastUp == null && this.persist) { │ │ │ │ │ - this.destroyPersistedFeature() │ │ │ │ │ + getChildEl: function(node, name, uri) { │ │ │ │ │ + return node && this.getThisOrNextEl(node.firstChild, name, uri) │ │ │ │ │ + }, │ │ │ │ │ + getNextEl: function(node, name, uri) { │ │ │ │ │ + return node && this.getThisOrNextEl(node.nextSibling, name, uri) │ │ │ │ │ + }, │ │ │ │ │ + getThisOrNextEl: function(node, name, uri) { │ │ │ │ │ + outer: for (var sibling = node; sibling; sibling = sibling.nextSibling) { │ │ │ │ │ + switch (sibling.nodeType) { │ │ │ │ │ + case 1: │ │ │ │ │ + if ((!name || name === (sibling.localName || sibling.nodeName.split(":").pop())) && (!uri || uri === sibling.namespaceURI)) { │ │ │ │ │ + break outer │ │ │ │ │ } │ │ │ │ │ - this.addPoint(evt.xy); │ │ │ │ │ - this.lastUp = evt.xy; │ │ │ │ │ - if (this.line.geometry.components.length === this.maxVertices + 1) { │ │ │ │ │ - this.finishGeometry() │ │ │ │ │ + sibling = null; │ │ │ │ │ + break outer; │ │ │ │ │ + case 3: │ │ │ │ │ + if (/^\s*$/.test(sibling.nodeValue)) { │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ + case 4: │ │ │ │ │ + case 6: │ │ │ │ │ + case 12: │ │ │ │ │ + case 10: │ │ │ │ │ + case 11: │ │ │ │ │ + sibling = null; │ │ │ │ │ + break outer │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - return !this.stopUp │ │ │ │ │ + return sibling || null │ │ │ │ │ }, │ │ │ │ │ - finishGeometry: function() { │ │ │ │ │ - var index = this.line.geometry.components.length - 1; │ │ │ │ │ - this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize() │ │ │ │ │ + lookupNamespaceURI: function(node, prefix) { │ │ │ │ │ + var uri = null; │ │ │ │ │ + if (node) { │ │ │ │ │ + if (node.lookupNamespaceURI) { │ │ │ │ │ + uri = node.lookupNamespaceURI(prefix) │ │ │ │ │ + } else { │ │ │ │ │ + outer: switch (node.nodeType) { │ │ │ │ │ + case 1: │ │ │ │ │ + if (node.namespaceURI !== null && node.prefix === prefix) { │ │ │ │ │ + uri = node.namespaceURI; │ │ │ │ │ + break outer │ │ │ │ │ + } │ │ │ │ │ + var len = node.attributes.length; │ │ │ │ │ + if (len) { │ │ │ │ │ + var attr; │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + attr = node.attributes[i]; │ │ │ │ │ + if (attr.prefix === "xmlns" && attr.name === "xmlns:" + prefix) { │ │ │ │ │ + uri = attr.value || null; │ │ │ │ │ + break outer │ │ │ │ │ + } else if (attr.name === "xmlns" && prefix === null) { │ │ │ │ │ + uri = attr.value || null; │ │ │ │ │ + break outer │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + uri = this.lookupNamespaceURI(node.parentNode, prefix); │ │ │ │ │ + break outer; │ │ │ │ │ + case 2: │ │ │ │ │ + uri = this.lookupNamespaceURI(node.ownerElement, prefix); │ │ │ │ │ + break outer; │ │ │ │ │ + case 9: │ │ │ │ │ + uri = this.lookupNamespaceURI(node.documentElement, prefix); │ │ │ │ │ + break outer; │ │ │ │ │ + case 6: │ │ │ │ │ + case 12: │ │ │ │ │ + case 10: │ │ │ │ │ + case 11: │ │ │ │ │ + break outer; │ │ │ │ │ + default: │ │ │ │ │ + uri = this.lookupNamespaceURI(node.parentNode, prefix); │ │ │ │ │ + break outer │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return uri │ │ │ │ │ }, │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - if (!this.freehandMode(evt)) { │ │ │ │ │ - this.finishGeometry() │ │ │ │ │ + getXMLDoc: function() { │ │ │ │ │ + if (!OpenLayers.Format.XML.document && !this.xmldom) { │ │ │ │ │ + if (document.implementation && document.implementation.createDocument) { │ │ │ │ │ + OpenLayers.Format.XML.document = document.implementation.createDocument("", "", null) │ │ │ │ │ + } else if (!this.xmldom && window.ActiveXObject) { │ │ │ │ │ + this.xmldom = new ActiveXObject("Microsoft.XMLDOM") │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return false │ │ │ │ │ + return OpenLayers.Format.XML.document || this.xmldom │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Path" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.XML" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Handler.Polygon = OpenLayers.Class(OpenLayers.Handler.Path, { │ │ │ │ │ - holeModifier: null, │ │ │ │ │ - drawingHole: false, │ │ │ │ │ - polygon: null, │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.line = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.LinearRing([this.point.geometry])); │ │ │ │ │ - this.polygon = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Polygon([this.line.geometry])); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.polygon, this.point], { │ │ │ │ │ - silent: true │ │ │ │ │ - }) │ │ │ │ │ +OpenLayers.Format.XML.CONTENT_TYPE = { │ │ │ │ │ + EMPTY: 0, │ │ │ │ │ + SIMPLE: 1, │ │ │ │ │ + COMPLEX: 2, │ │ │ │ │ + MIXED: 3 │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Format.XML.lookupNamespaceURI = OpenLayers.Function.bind(OpenLayers.Format.XML.prototype.lookupNamespaceURI, OpenLayers.Format.XML.prototype); │ │ │ │ │ +OpenLayers.Format.XML.document = null; │ │ │ │ │ +OpenLayers.Format.WFST = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, OpenLayers.Format.WFST.DEFAULTS); │ │ │ │ │ + var cls = OpenLayers.Format.WFST["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported WFST version: " + options.version │ │ │ │ │ + } │ │ │ │ │ + return new cls(options) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Format.WFST.DEFAULTS = { │ │ │ │ │ + version: "1.0.0" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Feature = OpenLayers.Class({ │ │ │ │ │ + layer: null, │ │ │ │ │ + id: null, │ │ │ │ │ + lonlat: null, │ │ │ │ │ + data: null, │ │ │ │ │ + marker: null, │ │ │ │ │ + popupClass: null, │ │ │ │ │ + popup: null, │ │ │ │ │ + initialize: function(layer, lonlat, data) { │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + this.lonlat = lonlat; │ │ │ │ │ + this.data = data != null ? data : {}; │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ }, │ │ │ │ │ - addPoint: function(pixel) { │ │ │ │ │ - if (!this.drawingHole && this.holeModifier && this.evt && this.evt[this.holeModifier]) { │ │ │ │ │ - var geometry = this.point.geometry; │ │ │ │ │ - var features = this.control.layer.features; │ │ │ │ │ - var candidate, polygon; │ │ │ │ │ - for (var i = features.length - 1; i >= 0; --i) { │ │ │ │ │ - candidate = features[i].geometry; │ │ │ │ │ - if ((candidate instanceof OpenLayers.Geometry.Polygon || candidate instanceof OpenLayers.Geometry.MultiPolygon) && candidate.intersects(geometry)) { │ │ │ │ │ - polygon = features[i]; │ │ │ │ │ - this.control.layer.removeFeatures([polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.control.layer.events.registerPriority("sketchcomplete", this, this.finalizeInteriorRing); │ │ │ │ │ - this.control.layer.events.registerPriority("sketchmodified", this, this.enforceTopology); │ │ │ │ │ - polygon.geometry.addComponent(this.line.geometry); │ │ │ │ │ - this.polygon = polygon; │ │ │ │ │ - this.drawingHole = true; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.layer != null && this.layer.map != null) { │ │ │ │ │ + if (this.popup != null) { │ │ │ │ │ + this.layer.map.removePopup(this.popup) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Handler.Path.prototype.addPoint.apply(this, arguments) │ │ │ │ │ + if (this.layer != null && this.marker != null) { │ │ │ │ │ + this.layer.removeMarker(this.marker) │ │ │ │ │ + } │ │ │ │ │ + this.layer = null; │ │ │ │ │ + this.id = null; │ │ │ │ │ + this.lonlat = null; │ │ │ │ │ + this.data = null; │ │ │ │ │ + if (this.marker != null) { │ │ │ │ │ + this.destroyMarker(this.marker); │ │ │ │ │ + this.marker = null │ │ │ │ │ + } │ │ │ │ │ + if (this.popup != null) { │ │ │ │ │ + this.destroyPopup(this.popup); │ │ │ │ │ + this.popup = null │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - getCurrentPointIndex: function() { │ │ │ │ │ - return this.line.geometry.components.length - 2 │ │ │ │ │ + onScreen: function() { │ │ │ │ │ + var onScreen = false; │ │ │ │ │ + if (this.layer != null && this.layer.map != null) { │ │ │ │ │ + var screenBounds = this.layer.map.getExtent(); │ │ │ │ │ + onScreen = screenBounds.containsLonLat(this.lonlat) │ │ │ │ │ + } │ │ │ │ │ + return onScreen │ │ │ │ │ }, │ │ │ │ │ - enforceTopology: function(event) { │ │ │ │ │ - var point = event.vertex; │ │ │ │ │ - var components = this.line.geometry.components; │ │ │ │ │ - if (!this.polygon.geometry.intersects(point)) { │ │ │ │ │ - var last = components[components.length - 3]; │ │ │ │ │ - point.x = last.x; │ │ │ │ │ - point.y = last.y │ │ │ │ │ + createMarker: function() { │ │ │ │ │ + if (this.lonlat != null) { │ │ │ │ │ + this.marker = new OpenLayers.Marker(this.lonlat, this.data.icon) │ │ │ │ │ } │ │ │ │ │ + return this.marker │ │ │ │ │ }, │ │ │ │ │ - finishGeometry: function() { │ │ │ │ │ - var index = this.line.geometry.components.length - 2; │ │ │ │ │ - this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize() │ │ │ │ │ + destroyMarker: function() { │ │ │ │ │ + this.marker.destroy() │ │ │ │ │ }, │ │ │ │ │ - finalizeInteriorRing: function() { │ │ │ │ │ - var ring = this.line.geometry; │ │ │ │ │ - var modified = ring.getArea() !== 0; │ │ │ │ │ - if (modified) { │ │ │ │ │ - var rings = this.polygon.geometry.components; │ │ │ │ │ - for (var i = rings.length - 2; i >= 0; --i) { │ │ │ │ │ - if (ring.intersects(rings[i])) { │ │ │ │ │ - modified = false; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (modified) { │ │ │ │ │ - var target; │ │ │ │ │ - outer: for (var i = rings.length - 2; i > 0; --i) { │ │ │ │ │ - var points = rings[i].components; │ │ │ │ │ - for (var j = 0, jj = points.length; j < jj; ++j) { │ │ │ │ │ - if (ring.containsPoint(points[j])) { │ │ │ │ │ - modified = false; │ │ │ │ │ - break outer │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + createPopup: function(closeBox) { │ │ │ │ │ + if (this.lonlat != null) { │ │ │ │ │ + if (!this.popup) { │ │ │ │ │ + var anchor = this.marker ? this.marker.icon : null; │ │ │ │ │ + var popupClass = this.popupClass ? this.popupClass : OpenLayers.Popup.Anchored; │ │ │ │ │ + this.popup = new popupClass(this.id + "_popup", this.lonlat, this.data.popupSize, this.data.popupContentHTML, anchor, closeBox) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - if (modified) { │ │ │ │ │ - if (this.polygon.state !== OpenLayers.State.INSERT) { │ │ │ │ │ - this.polygon.state = OpenLayers.State.UPDATE │ │ │ │ │ + if (this.data.overflow != null) { │ │ │ │ │ + this.popup.contentDiv.style.overflow = this.data.overflow │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - this.polygon.geometry.removeComponent(ring) │ │ │ │ │ + this.popup.feature = this │ │ │ │ │ } │ │ │ │ │ - this.restoreFeature(); │ │ │ │ │ - return false │ │ │ │ │ + return this.popup │ │ │ │ │ }, │ │ │ │ │ - cancel: function() { │ │ │ │ │ - if (this.drawingHole) { │ │ │ │ │ - this.polygon.geometry.removeComponent(this.line.geometry); │ │ │ │ │ - this.restoreFeature(true) │ │ │ │ │ + destroyPopup: function() { │ │ │ │ │ + if (this.popup) { │ │ │ │ │ + this.popup.feature = null; │ │ │ │ │ + this.popup.destroy(); │ │ │ │ │ + this.popup = null │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Handler.Path.prototype.cancel.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - restoreFeature: function(cancel) { │ │ │ │ │ - this.control.layer.events.unregister("sketchcomplete", this, this.finalizeInteriorRing); │ │ │ │ │ - this.control.layer.events.unregister("sketchmodified", this, this.enforceTopology); │ │ │ │ │ - this.layer.removeFeatures([this.polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.control.layer.addFeatures([this.polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.drawingHole = false; │ │ │ │ │ - if (!cancel) { │ │ │ │ │ - this.control.layer.events.triggerEvent("sketchcomplete", { │ │ │ │ │ - feature: this.polygon │ │ │ │ │ - }) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Feature" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.State = { │ │ │ │ │ + UNKNOWN: "Unknown", │ │ │ │ │ + INSERT: "Insert", │ │ │ │ │ + UPDATE: "Update", │ │ │ │ │ + DELETE: "Delete" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Feature.Vector = OpenLayers.Class(OpenLayers.Feature, { │ │ │ │ │ + fid: null, │ │ │ │ │ + geometry: null, │ │ │ │ │ + attributes: null, │ │ │ │ │ + bounds: null, │ │ │ │ │ + state: null, │ │ │ │ │ + style: null, │ │ │ │ │ + url: null, │ │ │ │ │ + renderIntent: "default", │ │ │ │ │ + modified: null, │ │ │ │ │ + initialize: function(geometry, attributes, style) { │ │ │ │ │ + OpenLayers.Feature.prototype.initialize.apply(this, [null, null, attributes]); │ │ │ │ │ + this.lonlat = null; │ │ │ │ │ + this.geometry = geometry ? geometry : null; │ │ │ │ │ + this.state = null; │ │ │ │ │ + this.attributes = {}; │ │ │ │ │ + if (attributes) { │ │ │ │ │ + this.attributes = OpenLayers.Util.extend(this.attributes, attributes) │ │ │ │ │ } │ │ │ │ │ + this.style = style ? style : null │ │ │ │ │ }, │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - OpenLayers.Handler.Path.prototype.destroyFeature.call(this, force); │ │ │ │ │ - this.polygon = null │ │ │ │ │ - }, │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.polygon, this.style); │ │ │ │ │ - this.layer.drawFeature(this.point, this.style) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.layer) { │ │ │ │ │ + this.layer.removeFeatures(this); │ │ │ │ │ + this.layer = null │ │ │ │ │ + } │ │ │ │ │ + this.geometry = null; │ │ │ │ │ + this.modified = null; │ │ │ │ │ + OpenLayers.Feature.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - getSketch: function() { │ │ │ │ │ - return this.polygon │ │ │ │ │ + clone: function() { │ │ │ │ │ + return new OpenLayers.Feature.Vector(this.geometry ? this.geometry.clone() : null, this.attributes, this.style) │ │ │ │ │ }, │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.polygon && this.polygon.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiPolygon([geometry]) │ │ │ │ │ + onScreen: function(boundsOnly) { │ │ │ │ │ + var onScreen = false; │ │ │ │ │ + if (this.layer && this.layer.map) { │ │ │ │ │ + var screenBounds = this.layer.map.getExtent(); │ │ │ │ │ + if (boundsOnly) { │ │ │ │ │ + var featureBounds = this.geometry.getBounds(); │ │ │ │ │ + onScreen = screenBounds.intersectsBounds(featureBounds) │ │ │ │ │ + } else { │ │ │ │ │ + var screenPoly = screenBounds.toGeometry(); │ │ │ │ │ + onScreen = screenPoly.intersects(this.geometry) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return geometry │ │ │ │ │ + return onScreen │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Polygon" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Strategy = OpenLayers.Class({ │ │ │ │ │ - layer: null, │ │ │ │ │ - options: null, │ │ │ │ │ - active: null, │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - autoDestroy: true, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.options = options; │ │ │ │ │ - this.active = false │ │ │ │ │ + getVisibility: function() { │ │ │ │ │ + return !(this.style && this.style.display == "none" || !this.layer || this.layer && this.layer.styleMap && this.layer.styleMap.createSymbolizer(this, this.renderIntent).display == "none" || this.layer && !this.layer.getVisibility()) │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.layer = null; │ │ │ │ │ - this.options = null │ │ │ │ │ + createMarker: function() { │ │ │ │ │ + return null │ │ │ │ │ }, │ │ │ │ │ - setLayer: function(layer) { │ │ │ │ │ - this.layer = layer │ │ │ │ │ + destroyMarker: function() {}, │ │ │ │ │ + createPopup: function() { │ │ │ │ │ + return null │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - this.active = true; │ │ │ │ │ - return true │ │ │ │ │ + atPoint: function(lonlat, toleranceLon, toleranceLat) { │ │ │ │ │ + var atPoint = false; │ │ │ │ │ + if (this.geometry) { │ │ │ │ │ + atPoint = this.geometry.atPoint(lonlat, toleranceLon, toleranceLat) │ │ │ │ │ } │ │ │ │ │ - return false │ │ │ │ │ + return atPoint │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.active = false; │ │ │ │ │ - return true │ │ │ │ │ + destroyPopup: function() {}, │ │ │ │ │ + move: function(location) { │ │ │ │ │ + if (!this.layer || !this.geometry.move) { │ │ │ │ │ + return undefined │ │ │ │ │ } │ │ │ │ │ - return false │ │ │ │ │ + var pixel; │ │ │ │ │ + if (location.CLASS_NAME == "OpenLayers.LonLat") { │ │ │ │ │ + pixel = this.layer.getViewPortPxFromLonLat(location) │ │ │ │ │ + } else { │ │ │ │ │ + pixel = location │ │ │ │ │ + } │ │ │ │ │ + var lastPixel = this.layer.getViewPortPxFromLonLat(this.geometry.getBounds().getCenterLonLat()); │ │ │ │ │ + var res = this.layer.map.getResolution(); │ │ │ │ │ + this.geometry.move(res * (pixel.x - lastPixel.x), res * (lastPixel.y - pixel.y)); │ │ │ │ │ + this.layer.drawFeature(this); │ │ │ │ │ + return lastPixel │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Strategy.Fixed = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - preload: false, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - refresh: this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - if (this.layer.visibility == true || this.preload) { │ │ │ │ │ - this.load() │ │ │ │ │ - } else { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - visibilitychanged: this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ + toState: function(state) { │ │ │ │ │ + if (state == OpenLayers.State.UPDATE) { │ │ │ │ │ + switch (this.state) { │ │ │ │ │ + case OpenLayers.State.UNKNOWN: │ │ │ │ │ + case OpenLayers.State.DELETE: │ │ │ │ │ + this.state = state; │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.State.UPDATE: │ │ │ │ │ + case OpenLayers.State.INSERT: │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } else if (state == OpenLayers.State.INSERT) { │ │ │ │ │ + switch (this.state) { │ │ │ │ │ + case OpenLayers.State.UNKNOWN: │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + this.state = state; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } else if (state == OpenLayers.State.DELETE) { │ │ │ │ │ + switch (this.state) { │ │ │ │ │ + case OpenLayers.State.INSERT: │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.State.DELETE: │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.State.UNKNOWN: │ │ │ │ │ + case OpenLayers.State.UPDATE: │ │ │ │ │ + this.state = state; │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ + } else if (state == OpenLayers.State.UNKNOWN) { │ │ │ │ │ + this.state = state │ │ │ │ │ } │ │ │ │ │ - return activated │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - refresh: this.load, │ │ │ │ │ - visibilitychanged: this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ + CLASS_NAME: "OpenLayers.Feature.Vector" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Feature.Vector.style = { │ │ │ │ │ + default: { │ │ │ │ │ + fillColor: "#ee9900", │ │ │ │ │ + fillOpacity: .4, │ │ │ │ │ + hoverFillColor: "white", │ │ │ │ │ + hoverFillOpacity: .8, │ │ │ │ │ + strokeColor: "#ee9900", │ │ │ │ │ + strokeOpacity: 1, │ │ │ │ │ + strokeWidth: 1, │ │ │ │ │ + strokeLinecap: "round", │ │ │ │ │ + strokeDashstyle: "solid", │ │ │ │ │ + hoverStrokeColor: "red", │ │ │ │ │ + hoverStrokeOpacity: 1, │ │ │ │ │ + hoverStrokeWidth: .2, │ │ │ │ │ + pointRadius: 6, │ │ │ │ │ + hoverPointRadius: 1, │ │ │ │ │ + hoverPointUnit: "%", │ │ │ │ │ + pointerEvents: "visiblePainted", │ │ │ │ │ + cursor: "inherit", │ │ │ │ │ + fontColor: "#000000", │ │ │ │ │ + labelAlign: "cm", │ │ │ │ │ + labelOutlineColor: "white", │ │ │ │ │ + labelOutlineWidth: 3 │ │ │ │ │ }, │ │ │ │ │ - load: function(options) { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - layer.events.triggerEvent("loadstart", { │ │ │ │ │ - filter: layer.filter │ │ │ │ │ - }); │ │ │ │ │ - layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: this.merge, │ │ │ │ │ - filter: layer.filter, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options)); │ │ │ │ │ - layer.events.un({ │ │ │ │ │ - visibilitychanged: this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ + select: { │ │ │ │ │ + fillColor: "blue", │ │ │ │ │ + fillOpacity: .4, │ │ │ │ │ + hoverFillColor: "white", │ │ │ │ │ + hoverFillOpacity: .8, │ │ │ │ │ + strokeColor: "blue", │ │ │ │ │ + strokeOpacity: 1, │ │ │ │ │ + strokeWidth: 2, │ │ │ │ │ + strokeLinecap: "round", │ │ │ │ │ + strokeDashstyle: "solid", │ │ │ │ │ + hoverStrokeColor: "red", │ │ │ │ │ + hoverStrokeOpacity: 1, │ │ │ │ │ + hoverStrokeWidth: .2, │ │ │ │ │ + pointRadius: 6, │ │ │ │ │ + hoverPointRadius: 1, │ │ │ │ │ + hoverPointUnit: "%", │ │ │ │ │ + pointerEvents: "visiblePainted", │ │ │ │ │ + cursor: "pointer", │ │ │ │ │ + fontColor: "#000000", │ │ │ │ │ + labelAlign: "cm", │ │ │ │ │ + labelOutlineColor: "white", │ │ │ │ │ + labelOutlineWidth: 3 │ │ │ │ │ }, │ │ │ │ │ - merge: function(resp) { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - layer.destroyFeatures(); │ │ │ │ │ - var features = resp.features; │ │ │ │ │ - if (features && features.length > 0) { │ │ │ │ │ - var remote = layer.projection; │ │ │ │ │ - var local = layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var geom; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - geom = features[i].geometry; │ │ │ │ │ - if (geom) { │ │ │ │ │ - geom.transform(remote, local) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - layer.addFeatures(features) │ │ │ │ │ - } │ │ │ │ │ - layer.events.triggerEvent("loadend", { │ │ │ │ │ - response: resp │ │ │ │ │ - }) │ │ │ │ │ + temporary: { │ │ │ │ │ + fillColor: "#66cccc", │ │ │ │ │ + fillOpacity: .2, │ │ │ │ │ + hoverFillColor: "white", │ │ │ │ │ + hoverFillOpacity: .8, │ │ │ │ │ + strokeColor: "#66cccc", │ │ │ │ │ + strokeOpacity: 1, │ │ │ │ │ + strokeLinecap: "round", │ │ │ │ │ + strokeWidth: 2, │ │ │ │ │ + strokeDashstyle: "solid", │ │ │ │ │ + hoverStrokeColor: "red", │ │ │ │ │ + hoverStrokeOpacity: 1, │ │ │ │ │ + hoverStrokeWidth: .2, │ │ │ │ │ + pointRadius: 6, │ │ │ │ │ + hoverPointRadius: 1, │ │ │ │ │ + hoverPointUnit: "%", │ │ │ │ │ + pointerEvents: "visiblePainted", │ │ │ │ │ + cursor: "inherit", │ │ │ │ │ + fontColor: "#000000", │ │ │ │ │ + labelAlign: "cm", │ │ │ │ │ + labelOutlineColor: "white", │ │ │ │ │ + labelOutlineWidth: 3 │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Fixed" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control = OpenLayers.Class({ │ │ │ │ │ + delete: { │ │ │ │ │ + display: "none" │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Style = OpenLayers.Class({ │ │ │ │ │ id: null, │ │ │ │ │ - map: null, │ │ │ │ │ - div: null, │ │ │ │ │ - type: null, │ │ │ │ │ - allowSelection: false, │ │ │ │ │ - displayClass: "", │ │ │ │ │ - title: "", │ │ │ │ │ - autoActivate: false, │ │ │ │ │ - active: null, │ │ │ │ │ - handlerOptions: null, │ │ │ │ │ - handler: null, │ │ │ │ │ - eventListeners: null, │ │ │ │ │ - events: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.displayClass = this.CLASS_NAME.replace("OpenLayers.", "ol").replace(/\./g, ""); │ │ │ │ │ + name: null, │ │ │ │ │ + title: null, │ │ │ │ │ + description: null, │ │ │ │ │ + layerName: null, │ │ │ │ │ + isDefault: false, │ │ │ │ │ + rules: null, │ │ │ │ │ + context: null, │ │ │ │ │ + defaultStyle: null, │ │ │ │ │ + defaultsPerSymbolizer: false, │ │ │ │ │ + propertyStyles: null, │ │ │ │ │ + initialize: function(style, options) { │ │ │ │ │ OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.events = new OpenLayers.Events(this); │ │ │ │ │ - if (this.eventListeners instanceof Object) { │ │ │ │ │ - this.events.on(this.eventListeners) │ │ │ │ │ - } │ │ │ │ │ - if (this.id == null) { │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ + this.rules = []; │ │ │ │ │ + if (options && options.rules) { │ │ │ │ │ + this.addRules(options.rules) │ │ │ │ │ } │ │ │ │ │ + this.setDefaultStyle(style || OpenLayers.Feature.Vector.style["default"]); │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - if (this.events) { │ │ │ │ │ - if (this.eventListeners) { │ │ │ │ │ - this.events.un(this.eventListeners) │ │ │ │ │ - } │ │ │ │ │ - this.events.destroy(); │ │ │ │ │ - this.events = null │ │ │ │ │ - } │ │ │ │ │ - this.eventListeners = null; │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.destroy(); │ │ │ │ │ - this.handler = null │ │ │ │ │ + for (var i = 0, len = this.rules.length; i < len; i++) { │ │ │ │ │ + this.rules[i].destroy(); │ │ │ │ │ + this.rules[i] = null │ │ │ │ │ } │ │ │ │ │ - if (this.handlers) { │ │ │ │ │ - for (var key in this.handlers) { │ │ │ │ │ - if (this.handlers.hasOwnProperty(key) && typeof this.handlers[key].destroy == "function") { │ │ │ │ │ - this.handlers[key].destroy() │ │ │ │ │ + this.rules = null; │ │ │ │ │ + this.defaultStyle = null │ │ │ │ │ + }, │ │ │ │ │ + createSymbolizer: function(feature) { │ │ │ │ │ + var style = this.defaultsPerSymbolizer ? {} : this.createLiterals(OpenLayers.Util.extend({}, this.defaultStyle), feature); │ │ │ │ │ + var rules = this.rules; │ │ │ │ │ + var rule, context; │ │ │ │ │ + var elseRules = []; │ │ │ │ │ + var appliedRules = false; │ │ │ │ │ + for (var i = 0, len = rules.length; i < len; i++) { │ │ │ │ │ + rule = rules[i]; │ │ │ │ │ + var applies = rule.evaluate(feature); │ │ │ │ │ + if (applies) { │ │ │ │ │ + if (rule instanceof OpenLayers.Rule && rule.elseFilter) { │ │ │ │ │ + elseRules.push(rule) │ │ │ │ │ + } else { │ │ │ │ │ + appliedRules = true; │ │ │ │ │ + this.applySymbolizer(rule, style, feature) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.handlers = null │ │ │ │ │ - } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.removeControl(this); │ │ │ │ │ - this.map = null │ │ │ │ │ - } │ │ │ │ │ - this.div = null │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.map = map; │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.setMap(map) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - if (this.div == null) { │ │ │ │ │ - this.div = OpenLayers.Util.createDiv(this.id); │ │ │ │ │ - this.div.className = this.displayClass; │ │ │ │ │ - if (!this.allowSelection) { │ │ │ │ │ - this.div.className += " olControlNoSelect"; │ │ │ │ │ - this.div.setAttribute("unselectable", "on", 0); │ │ │ │ │ - this.div.onselectstart = OpenLayers.Function.False │ │ │ │ │ - } │ │ │ │ │ - if (this.title != "") { │ │ │ │ │ - this.div.title = this.title │ │ │ │ │ + if (appliedRules == false && elseRules.length > 0) { │ │ │ │ │ + appliedRules = true; │ │ │ │ │ + for (var i = 0, len = elseRules.length; i < len; i++) { │ │ │ │ │ + this.applySymbolizer(elseRules[i], style, feature) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (px != null) { │ │ │ │ │ - this.position = px.clone() │ │ │ │ │ - } │ │ │ │ │ - this.moveTo(this.position); │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - if (px != null && this.div != null) { │ │ │ │ │ - this.div.style.left = px.x + "px"; │ │ │ │ │ - this.div.style.top = px.y + "px" │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.activate() │ │ │ │ │ + if (rules.length > 0 && appliedRules == false) { │ │ │ │ │ + style.display = "none" │ │ │ │ │ } │ │ │ │ │ - this.active = true; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, this.displayClass.replace(/ /g, "") + "Active") │ │ │ │ │ + if (style.label != null && typeof style.label !== "string") { │ │ │ │ │ + style.label = String(style.label) │ │ │ │ │ } │ │ │ │ │ - this.events.triggerEvent("activate"); │ │ │ │ │ - return true │ │ │ │ │ + return style │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.deactivate() │ │ │ │ │ - } │ │ │ │ │ - this.active = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, this.displayClass.replace(/ /g, "") + "Active") │ │ │ │ │ + applySymbolizer: function(rule, style, feature) { │ │ │ │ │ + var symbolizerPrefix = feature.geometry ? this.getSymbolizerPrefix(feature.geometry) : OpenLayers.Style.SYMBOLIZER_PREFIXES[0]; │ │ │ │ │ + var symbolizer = rule.symbolizer[symbolizerPrefix] || rule.symbolizer; │ │ │ │ │ + if (this.defaultsPerSymbolizer === true) { │ │ │ │ │ + var defaults = this.defaultStyle; │ │ │ │ │ + OpenLayers.Util.applyDefaults(symbolizer, { │ │ │ │ │ + pointRadius: defaults.pointRadius │ │ │ │ │ + }); │ │ │ │ │ + if (symbolizer.stroke === true || symbolizer.graphic === true) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(symbolizer, { │ │ │ │ │ + strokeWidth: defaults.strokeWidth, │ │ │ │ │ + strokeColor: defaults.strokeColor, │ │ │ │ │ + strokeOpacity: defaults.strokeOpacity, │ │ │ │ │ + strokeDashstyle: defaults.strokeDashstyle, │ │ │ │ │ + strokeLinecap: defaults.strokeLinecap │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - this.events.triggerEvent("deactivate"); │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.TYPE_BUTTON = 1; │ │ │ │ │ -OpenLayers.Control.TYPE_TOGGLE = 2; │ │ │ │ │ -OpenLayers.Control.TYPE_TOOL = 3; │ │ │ │ │ -OpenLayers.Handler.Drag = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - started: false, │ │ │ │ │ - stopDown: true, │ │ │ │ │ - dragging: false, │ │ │ │ │ - last: null, │ │ │ │ │ - start: null, │ │ │ │ │ - lastMoveEvt: null, │ │ │ │ │ - oldOnselectstart: null, │ │ │ │ │ - interval: 0, │ │ │ │ │ - timeoutId: null, │ │ │ │ │ - documentDrag: false, │ │ │ │ │ - documentEvents: null, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (this.documentDrag === true) { │ │ │ │ │ - var me = this; │ │ │ │ │ - this._docMove = function(evt) { │ │ │ │ │ - me.mousemove({ │ │ │ │ │ - xy: { │ │ │ │ │ - x: evt.clientX, │ │ │ │ │ - y: evt.clientY │ │ │ │ │ - }, │ │ │ │ │ - element: document │ │ │ │ │ + if (symbolizer.fill === true || symbolizer.graphic === true) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(symbolizer, { │ │ │ │ │ + fillColor: defaults.fillColor, │ │ │ │ │ + fillOpacity: defaults.fillOpacity │ │ │ │ │ }) │ │ │ │ │ - }; │ │ │ │ │ - this._docUp = function(evt) { │ │ │ │ │ - me.mouseup({ │ │ │ │ │ - xy: { │ │ │ │ │ - x: evt.clientX, │ │ │ │ │ - y: evt.clientY │ │ │ │ │ - } │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.graphic === true) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(symbolizer, { │ │ │ │ │ + pointRadius: this.defaultStyle.pointRadius, │ │ │ │ │ + externalGraphic: this.defaultStyle.externalGraphic, │ │ │ │ │ + graphicName: this.defaultStyle.graphicName, │ │ │ │ │ + graphicOpacity: this.defaultStyle.graphicOpacity, │ │ │ │ │ + graphicWidth: this.defaultStyle.graphicWidth, │ │ │ │ │ + graphicHeight: this.defaultStyle.graphicHeight, │ │ │ │ │ + graphicXOffset: this.defaultStyle.graphicXOffset, │ │ │ │ │ + graphicYOffset: this.defaultStyle.graphicYOffset │ │ │ │ │ }) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return this.createLiterals(OpenLayers.Util.extend(style, symbolizer), feature) │ │ │ │ │ }, │ │ │ │ │ - dragstart: function(evt) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - if (this.checkModifiers(evt) && (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt))) { │ │ │ │ │ - this.started = true; │ │ │ │ │ - this.start = evt.xy; │ │ │ │ │ - this.last = evt.xy; │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ - this.down(evt); │ │ │ │ │ - this.callback("down", [evt.xy]); │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ - if (!this.oldOnselectstart) { │ │ │ │ │ - this.oldOnselectstart = document.onselectstart ? document.onselectstart : OpenLayers.Function.True │ │ │ │ │ - } │ │ │ │ │ - document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ - propagate = !this.stopDown │ │ │ │ │ - } else { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null │ │ │ │ │ + createLiterals: function(style, feature) { │ │ │ │ │ + var context = OpenLayers.Util.extend({}, feature.attributes || feature.data); │ │ │ │ │ + OpenLayers.Util.extend(context, this.context); │ │ │ │ │ + for (var i in this.propertyStyles) { │ │ │ │ │ + style[i] = OpenLayers.Style.createLiteral(style[i], context, feature, i) │ │ │ │ │ } │ │ │ │ │ - return propagate │ │ │ │ │ + return style │ │ │ │ │ }, │ │ │ │ │ - dragmove: function(evt) { │ │ │ │ │ - this.lastMoveEvt = evt; │ │ │ │ │ - if (this.started && !this.timeoutId && (evt.xy.x != this.last.x || evt.xy.y != this.last.y)) { │ │ │ │ │ - if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ - if (evt.element === document) { │ │ │ │ │ - this.adjustXY(evt); │ │ │ │ │ - this.setEvent(evt) │ │ │ │ │ + findPropertyStyles: function() { │ │ │ │ │ + var propertyStyles = {}; │ │ │ │ │ + var style = this.defaultStyle; │ │ │ │ │ + this.addPropertyStyles(propertyStyles, style); │ │ │ │ │ + var rules = this.rules; │ │ │ │ │ + var symbolizer, value; │ │ │ │ │ + for (var i = 0, len = rules.length; i < len; i++) { │ │ │ │ │ + symbolizer = rules[i].symbolizer; │ │ │ │ │ + for (var key in symbolizer) { │ │ │ │ │ + value = symbolizer[key]; │ │ │ │ │ + if (typeof value == "object") { │ │ │ │ │ + this.addPropertyStyles(propertyStyles, value) │ │ │ │ │ } else { │ │ │ │ │ - this.removeDocumentEvents() │ │ │ │ │ + this.addPropertyStyles(propertyStyles, symbolizer); │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (this.interval > 0) { │ │ │ │ │ - this.timeoutId = setTimeout(OpenLayers.Function.bind(this.removeTimeout, this), this.interval) │ │ │ │ │ - } │ │ │ │ │ - this.dragging = true; │ │ │ │ │ - this.move(evt); │ │ │ │ │ - this.callback("move", [evt.xy]); │ │ │ │ │ - if (!this.oldOnselectstart) { │ │ │ │ │ - this.oldOnselectstart = document.onselectstart; │ │ │ │ │ - document.onselectstart = OpenLayers.Function.False │ │ │ │ │ - } │ │ │ │ │ - this.last = evt.xy │ │ │ │ │ } │ │ │ │ │ - return true │ │ │ │ │ + return propertyStyles │ │ │ │ │ }, │ │ │ │ │ - dragend: function(evt) { │ │ │ │ │ - if (this.started) { │ │ │ │ │ - if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ - this.adjustXY(evt); │ │ │ │ │ - this.removeDocumentEvents() │ │ │ │ │ - } │ │ │ │ │ - var dragged = this.start != this.last; │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ - this.up(evt); │ │ │ │ │ - this.callback("up", [evt.xy]); │ │ │ │ │ - if (dragged) { │ │ │ │ │ - this.callback("done", [evt.xy]) │ │ │ │ │ + addPropertyStyles: function(propertyStyles, symbolizer) { │ │ │ │ │ + var property; │ │ │ │ │ + for (var key in symbolizer) { │ │ │ │ │ + property = symbolizer[key]; │ │ │ │ │ + if (typeof property == "string" && property.match(/\$\{\w+\}/)) { │ │ │ │ │ + propertyStyles[key] = true │ │ │ │ │ } │ │ │ │ │ - document.onselectstart = this.oldOnselectstart │ │ │ │ │ } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - down: function(evt) {}, │ │ │ │ │ - move: function(evt) {}, │ │ │ │ │ - up: function(evt) {}, │ │ │ │ │ - out: function(evt) {}, │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - return this.dragstart(evt) │ │ │ │ │ - }, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - return this.dragstart(evt) │ │ │ │ │ + return propertyStyles │ │ │ │ │ }, │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - return this.dragmove(evt) │ │ │ │ │ + addRules: function(rules) { │ │ │ │ │ + Array.prototype.push.apply(this.rules, rules); │ │ │ │ │ + this.propertyStyles = this.findPropertyStyles() │ │ │ │ │ }, │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - return this.dragmove(evt) │ │ │ │ │ + setDefaultStyle: function(style) { │ │ │ │ │ + this.defaultStyle = style; │ │ │ │ │ + this.propertyStyles = this.findPropertyStyles() │ │ │ │ │ }, │ │ │ │ │ - removeTimeout: function() { │ │ │ │ │ - this.timeoutId = null; │ │ │ │ │ - if (this.dragging) { │ │ │ │ │ - this.mousemove(this.lastMoveEvt) │ │ │ │ │ + getSymbolizerPrefix: function(geometry) { │ │ │ │ │ + var prefixes = OpenLayers.Style.SYMBOLIZER_PREFIXES; │ │ │ │ │ + for (var i = 0, len = prefixes.length; i < len; i++) { │ │ │ │ │ + if (geometry.CLASS_NAME.indexOf(prefixes[i]) != -1) { │ │ │ │ │ + return prefixes[i] │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - return this.dragend(evt) │ │ │ │ │ - }, │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - evt.xy = this.last; │ │ │ │ │ - return this.dragend(evt) │ │ │ │ │ - }, │ │ │ │ │ - mouseout: function(evt) { │ │ │ │ │ - if (this.started && OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ - if (this.documentDrag === true) { │ │ │ │ │ - this.addDocumentEvents() │ │ │ │ │ - } else { │ │ │ │ │ - var dragged = this.start != this.last; │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ - this.out(evt); │ │ │ │ │ - this.callback("out", []); │ │ │ │ │ - if (dragged) { │ │ │ │ │ - this.callback("done", [evt.xy]) │ │ │ │ │ - } │ │ │ │ │ - if (document.onselectstart) { │ │ │ │ │ - document.onselectstart = this.oldOnselectstart │ │ │ │ │ - } │ │ │ │ │ + clone: function() { │ │ │ │ │ + var options = OpenLayers.Util.extend({}, this); │ │ │ │ │ + if (this.rules) { │ │ │ │ │ + options.rules = []; │ │ │ │ │ + for (var i = 0, len = this.rules.length; i < len; ++i) { │ │ │ │ │ + options.rules.push(this.rules[i].clone()) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + options.context = this.context && OpenLayers.Util.extend({}, this.context); │ │ │ │ │ + var defaultStyle = OpenLayers.Util.extend({}, this.defaultStyle); │ │ │ │ │ + return new OpenLayers.Style(defaultStyle, options) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Style" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Style.createLiteral = function(value, context, feature, property) { │ │ │ │ │ + if (typeof value == "string" && value.indexOf("${") != -1) { │ │ │ │ │ + value = OpenLayers.String.format(value, context, [feature, property]); │ │ │ │ │ + value = isNaN(value) || !value ? value : parseFloat(value) │ │ │ │ │ + } │ │ │ │ │ + return value │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Style.SYMBOLIZER_PREFIXES = ["Point", "Line", "Polygon", "Text", "Raster"]; │ │ │ │ │ +OpenLayers.Filter = OpenLayers.Class({ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() {}, │ │ │ │ │ + evaluate: function(context) { │ │ │ │ │ return true │ │ │ │ │ }, │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - return this.start == this.last │ │ │ │ │ + clone: function() { │ │ │ │ │ + return null │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - activated = true │ │ │ │ │ + toString: function() { │ │ │ │ │ + var string; │ │ │ │ │ + if (OpenLayers.Format && OpenLayers.Format.CQL) { │ │ │ │ │ + string = OpenLayers.Format.CQL.prototype.write(this) │ │ │ │ │ + } else { │ │ │ │ │ + string = Object.prototype.toString.call(this) │ │ │ │ │ } │ │ │ │ │ - return activated │ │ │ │ │ + return string │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - deactivated = true; │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown") │ │ │ │ │ + CLASS_NAME: "OpenLayers.Filter" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Filter.Spatial = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ + type: null, │ │ │ │ │ + property: null, │ │ │ │ │ + value: null, │ │ │ │ │ + distance: null, │ │ │ │ │ + distanceUnits: null, │ │ │ │ │ + evaluate: function(feature) { │ │ │ │ │ + var intersect = false; │ │ │ │ │ + switch (this.type) { │ │ │ │ │ + case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ + case OpenLayers.Filter.Spatial.INTERSECTS: │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + var geom = this.value; │ │ │ │ │ + if (this.value.CLASS_NAME == "OpenLayers.Bounds") { │ │ │ │ │ + geom = this.value.toGeometry() │ │ │ │ │ + } │ │ │ │ │ + if (feature.geometry.intersects(geom)) { │ │ │ │ │ + intersect = true │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + throw new Error("evaluate is not implemented for this filter type.") │ │ │ │ │ } │ │ │ │ │ - return deactivated │ │ │ │ │ + return intersect │ │ │ │ │ }, │ │ │ │ │ - adjustXY: function(evt) { │ │ │ │ │ - var pos = OpenLayers.Util.pagePosition(this.map.viewPortDiv); │ │ │ │ │ - evt.xy.x -= pos[0]; │ │ │ │ │ - evt.xy.y -= pos[1] │ │ │ │ │ + clone: function() { │ │ │ │ │ + var options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + value: this.value && this.value.clone && this.value.clone() │ │ │ │ │ + }, this); │ │ │ │ │ + return new OpenLayers.Filter.Spatial(options) │ │ │ │ │ }, │ │ │ │ │ - addDocumentEvents: function() { │ │ │ │ │ - OpenLayers.Element.addClass(document.body, "olDragDown"); │ │ │ │ │ - this.documentEvents = true; │ │ │ │ │ - OpenLayers.Event.observe(document, "mousemove", this._docMove); │ │ │ │ │ - OpenLayers.Event.observe(document, "mouseup", this._docUp) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Filter.Spatial" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Filter.Spatial.BBOX = "BBOX"; │ │ │ │ │ +OpenLayers.Filter.Spatial.INTERSECTS = "INTERSECTS"; │ │ │ │ │ +OpenLayers.Filter.Spatial.DWITHIN = "DWITHIN"; │ │ │ │ │ +OpenLayers.Filter.Spatial.WITHIN = "WITHIN"; │ │ │ │ │ +OpenLayers.Filter.Spatial.CONTAINS = "CONTAINS"; │ │ │ │ │ +OpenLayers.Filter.FeatureId = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ + fids: null, │ │ │ │ │ + type: "FID", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.fids = []; │ │ │ │ │ + OpenLayers.Filter.prototype.initialize.apply(this, [options]) │ │ │ │ │ }, │ │ │ │ │ - removeDocumentEvents: function() { │ │ │ │ │ - OpenLayers.Element.removeClass(document.body, "olDragDown"); │ │ │ │ │ - this.documentEvents = false; │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mousemove", this._docMove); │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mouseup", this._docUp) │ │ │ │ │ + evaluate: function(feature) { │ │ │ │ │ + for (var i = 0, len = this.fids.length; i < len; i++) { │ │ │ │ │ + var fid = feature.fid || feature.id; │ │ │ │ │ + if (fid == this.fids[i]) { │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Drag" │ │ │ │ │ + clone: function() { │ │ │ │ │ + var filter = new OpenLayers.Filter.FeatureId; │ │ │ │ │ + OpenLayers.Util.extend(filter, this); │ │ │ │ │ + filter.fids = this.fids.slice(); │ │ │ │ │ + return filter │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Filter.FeatureId" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Handler.Keyboard = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - KEY_EVENTS: ["keydown", "keyup"], │ │ │ │ │ - eventListener: null, │ │ │ │ │ - observeElement: null, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.eventListener = OpenLayers.Function.bindAsEventListener(this.handleKeyEvent, this) │ │ │ │ │ +OpenLayers.Format.WFST.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + namespaces: { │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + wfs: "http://www.opengis.net/wfs", │ │ │ │ │ + gml: "http://www.opengis.net/gml", │ │ │ │ │ + ogc: "http://www.opengis.net/ogc", │ │ │ │ │ + ows: "http://www.opengis.net/ows" │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.eventListener = null; │ │ │ │ │ - OpenLayers.Handler.prototype.destroy.apply(this, arguments) │ │ │ │ │ + defaultPrefix: "wfs", │ │ │ │ │ + version: null, │ │ │ │ │ + schemaLocations: null, │ │ │ │ │ + srsName: null, │ │ │ │ │ + extractAttributes: true, │ │ │ │ │ + xy: true, │ │ │ │ │ + stateName: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.stateName = {}; │ │ │ │ │ + this.stateName[OpenLayers.State.INSERT] = "wfs:Insert"; │ │ │ │ │ + this.stateName[OpenLayers.State.UPDATE] = "wfs:Update"; │ │ │ │ │ + this.stateName[OpenLayers.State.DELETE] = "wfs:Delete"; │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.observeElement = this.observeElement || document; │ │ │ │ │ - for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ - OpenLayers.Event.observe(this.observeElement, this.KEY_EVENTS[i], this.eventListener) │ │ │ │ │ + getSrsName: function(feature, options) { │ │ │ │ │ + var srsName = options && options.srsName; │ │ │ │ │ + if (!srsName) { │ │ │ │ │ + if (feature && feature.layer) { │ │ │ │ │ + srsName = feature.layer.projection.getCode() │ │ │ │ │ + } else { │ │ │ │ │ + srsName = this.srsName │ │ │ │ │ } │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ } │ │ │ │ │ + return srsName │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ - OpenLayers.Event.stopObserving(this.observeElement, this.KEY_EVENTS[i], this.eventListener) │ │ │ │ │ - } │ │ │ │ │ - deactivated = true │ │ │ │ │ + read: function(data, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, { │ │ │ │ │ + output: "features" │ │ │ │ │ + }); │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ } │ │ │ │ │ - return deactivated │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement │ │ │ │ │ + } │ │ │ │ │ + var obj = {}; │ │ │ │ │ + if (data) { │ │ │ │ │ + this.readNode(data, obj, true) │ │ │ │ │ + } │ │ │ │ │ + if (obj.features && options.output === "features") { │ │ │ │ │ + obj = obj.features │ │ │ │ │ + } │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - handleKeyEvent: function(evt) { │ │ │ │ │ - if (this.checkModifiers(evt)) { │ │ │ │ │ - this.callback(evt.type, [evt]) │ │ │ │ │ + readers: { │ │ │ │ │ + wfs: { │ │ │ │ │ + FeatureCollection: function(node, obj) { │ │ │ │ │ + obj.features = []; │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Keyboard" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.ModifyFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - documentDrag: false, │ │ │ │ │ - geometryTypes: null, │ │ │ │ │ - clickout: true, │ │ │ │ │ - toggle: true, │ │ │ │ │ - standalone: false, │ │ │ │ │ - layer: null, │ │ │ │ │ - feature: null, │ │ │ │ │ - vertex: null, │ │ │ │ │ - vertices: null, │ │ │ │ │ - virtualVertices: null, │ │ │ │ │ - handlers: null, │ │ │ │ │ - deleteCodes: null, │ │ │ │ │ - virtualStyle: null, │ │ │ │ │ - vertexRenderIntent: null, │ │ │ │ │ - mode: null, │ │ │ │ │ - createVertices: true, │ │ │ │ │ - modified: false, │ │ │ │ │ - radiusHandle: null, │ │ │ │ │ - dragHandle: null, │ │ │ │ │ - onModificationStart: function() {}, │ │ │ │ │ - onModification: function() {}, │ │ │ │ │ - onModificationEnd: function() {}, │ │ │ │ │ - initialize: function(layer, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - this.virtualVertices = []; │ │ │ │ │ - this.virtualStyle = OpenLayers.Util.extend({}, this.layer.style || this.layer.styleMap.createSymbolizer(null, options.vertexRenderIntent)); │ │ │ │ │ - this.virtualStyle.fillOpacity = .3; │ │ │ │ │ - this.virtualStyle.strokeOpacity = .3; │ │ │ │ │ - this.deleteCodes = [46, 68]; │ │ │ │ │ - this.mode = OpenLayers.Control.ModifyFeature.RESHAPE; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!OpenLayers.Util.isArray(this.deleteCodes)) { │ │ │ │ │ - this.deleteCodes = [this.deleteCodes] │ │ │ │ │ + write: function(features, options) { │ │ │ │ │ + var node = this.writeNode("wfs:Transaction", { │ │ │ │ │ + features: features, │ │ │ │ │ + options: options │ │ │ │ │ + }); │ │ │ │ │ + var value = this.schemaLocationAttr(); │ │ │ │ │ + if (value) { │ │ │ │ │ + this.setAttributeNS(node, this.namespaces["xsi"], "xsi:schemaLocation", value) │ │ │ │ │ } │ │ │ │ │ - var dragCallbacks = { │ │ │ │ │ - down: function(pixel) { │ │ │ │ │ - this.vertex = null; │ │ │ │ │ - var feature = this.layer.getFeatureFromEvent(this.handlers.drag.evt); │ │ │ │ │ - if (feature) { │ │ │ │ │ - this.dragStart(feature) │ │ │ │ │ - } else if (this.clickout) { │ │ │ │ │ - this._unselect = this.feature │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [node]) │ │ │ │ │ + }, │ │ │ │ │ + writers: { │ │ │ │ │ + wfs: { │ │ │ │ │ + GetFeature: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("wfs:GetFeature", { │ │ │ │ │ + attributes: { │ │ │ │ │ + service: "WFS", │ │ │ │ │ + version: this.version, │ │ │ │ │ + handle: options && options.handle, │ │ │ │ │ + outputFormat: options && options.outputFormat, │ │ │ │ │ + maxFeatures: options && options.maxFeatures, │ │ │ │ │ + "xsi:schemaLocation": this.schemaLocationAttr(options) │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (typeof this.featureType == "string") { │ │ │ │ │ + this.writeNode("Query", options, node) │ │ │ │ │ + } else { │ │ │ │ │ + for (var i = 0, len = this.featureType.length; i < len; i++) { │ │ │ │ │ + options.featureType = this.featureType[i]; │ │ │ │ │ + this.writeNode("Query", options, node) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + Transaction: function(obj) { │ │ │ │ │ + obj = obj || {}; │ │ │ │ │ + var options = obj.options || {}; │ │ │ │ │ + var node = this.createElementNSPlus("wfs:Transaction", { │ │ │ │ │ + attributes: { │ │ │ │ │ + service: "WFS", │ │ │ │ │ + version: this.version, │ │ │ │ │ + handle: options.handle │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + var i, len; │ │ │ │ │ + var features = obj.features; │ │ │ │ │ + if (features) { │ │ │ │ │ + if (options.multi === true) { │ │ │ │ │ + OpenLayers.Util.extend(this.geometryTypes, { │ │ │ │ │ + "OpenLayers.Geometry.Point": "MultiPoint", │ │ │ │ │ + "OpenLayers.Geometry.LineString": this.multiCurve === true ? "MultiCurve" : "MultiLineString", │ │ │ │ │ + "OpenLayers.Geometry.Polygon": this.multiSurface === true ? "MultiSurface" : "MultiPolygon" │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + var name, feature; │ │ │ │ │ + for (i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + name = this.stateName[feature.state]; │ │ │ │ │ + if (name) { │ │ │ │ │ + this.writeNode(name, { │ │ │ │ │ + feature: feature, │ │ │ │ │ + options: options │ │ │ │ │ + }, node) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (options.multi === true) { │ │ │ │ │ + this.setGeometryTypes() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (options.nativeElements) { │ │ │ │ │ + for (i = 0, len = options.nativeElements.length; i < len; ++i) { │ │ │ │ │ + this.writeNode("wfs:Native", options.nativeElements[i], node) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - move: function(pixel) { │ │ │ │ │ - delete this._unselect; │ │ │ │ │ - if (this.vertex) { │ │ │ │ │ - this.dragVertex(this.vertex, pixel) │ │ │ │ │ + Native: function(nativeElement) { │ │ │ │ │ + var node = this.createElementNSPlus("wfs:Native", { │ │ │ │ │ + attributes: { │ │ │ │ │ + vendorId: nativeElement.vendorId, │ │ │ │ │ + safeToIgnore: nativeElement.safeToIgnore │ │ │ │ │ + }, │ │ │ │ │ + value: nativeElement.value │ │ │ │ │ + }); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + Insert: function(obj) { │ │ │ │ │ + var feature = obj.feature; │ │ │ │ │ + var options = obj.options; │ │ │ │ │ + var node = this.createElementNSPlus("wfs:Insert", { │ │ │ │ │ + attributes: { │ │ │ │ │ + handle: options && options.handle │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + this.srsName = this.getSrsName(feature); │ │ │ │ │ + this.writeNode("feature:_typeName", feature, node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + Update: function(obj) { │ │ │ │ │ + var feature = obj.feature; │ │ │ │ │ + var options = obj.options; │ │ │ │ │ + var node = this.createElementNSPlus("wfs:Update", { │ │ │ │ │ + attributes: { │ │ │ │ │ + handle: options && options.handle, │ │ │ │ │ + typeName: (this.featureNS ? this.featurePrefix + ":" : "") + this.featureType │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (this.featureNS) { │ │ │ │ │ + node.setAttribute("xmlns:" + this.featurePrefix, this.featureNS) │ │ │ │ │ + } │ │ │ │ │ + var modified = feature.modified; │ │ │ │ │ + if (this.geometryName !== null && (!modified || modified.geometry !== undefined)) { │ │ │ │ │ + this.srsName = this.getSrsName(feature); │ │ │ │ │ + this.writeNode("Property", { │ │ │ │ │ + name: this.geometryName, │ │ │ │ │ + value: feature.geometry │ │ │ │ │ + }, node) │ │ │ │ │ + } │ │ │ │ │ + for (var key in feature.attributes) { │ │ │ │ │ + if (feature.attributes[key] !== undefined && (!modified || !modified.attributes || modified.attributes && modified.attributes[key] !== undefined)) { │ │ │ │ │ + this.writeNode("Property", { │ │ │ │ │ + name: key, │ │ │ │ │ + value: feature.attributes[key] │ │ │ │ │ + }, node) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + this.writeNode("ogc:Filter", new OpenLayers.Filter.FeatureId({ │ │ │ │ │ + fids: [feature.fid] │ │ │ │ │ + }), node); │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - up: function() { │ │ │ │ │ - this.handlers.drag.stopDown = false; │ │ │ │ │ - if (this._unselect) { │ │ │ │ │ - this.unselectFeature(this._unselect); │ │ │ │ │ - delete this._unselect │ │ │ │ │ + Property: function(obj) { │ │ │ │ │ + var node = this.createElementNSPlus("wfs:Property"); │ │ │ │ │ + this.writeNode("Name", obj.name, node); │ │ │ │ │ + if (obj.value !== null) { │ │ │ │ │ + this.writeNode("Value", obj.value, node) │ │ │ │ │ } │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - done: function(pixel) { │ │ │ │ │ - if (this.vertex) { │ │ │ │ │ - this.dragComplete(this.vertex) │ │ │ │ │ + Name: function(name) { │ │ │ │ │ + return this.createElementNSPlus("wfs:Name", { │ │ │ │ │ + value: name │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + Value: function(obj) { │ │ │ │ │ + var node; │ │ │ │ │ + if (obj instanceof OpenLayers.Geometry) { │ │ │ │ │ + node = this.createElementNSPlus("wfs:Value"); │ │ │ │ │ + var geom = this.writeNode("feature:_geometry", obj).firstChild; │ │ │ │ │ + node.appendChild(geom) │ │ │ │ │ + } else { │ │ │ │ │ + node = this.createElementNSPlus("wfs:Value", { │ │ │ │ │ + value: obj │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + Delete: function(obj) { │ │ │ │ │ + var feature = obj.feature; │ │ │ │ │ + var options = obj.options; │ │ │ │ │ + var node = this.createElementNSPlus("wfs:Delete", { │ │ │ │ │ + attributes: { │ │ │ │ │ + handle: options && options.handle, │ │ │ │ │ + typeName: (this.featureNS ? this.featurePrefix + ":" : "") + this.featureType │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (this.featureNS) { │ │ │ │ │ + node.setAttribute("xmlns:" + this.featurePrefix, this.featureNS) │ │ │ │ │ + } │ │ │ │ │ + this.writeNode("ogc:Filter", new OpenLayers.Filter.FeatureId({ │ │ │ │ │ + fids: [feature.fid] │ │ │ │ │ + }), node); │ │ │ │ │ + return node │ │ │ │ │ } │ │ │ │ │ - }; │ │ │ │ │ - var dragOptions = { │ │ │ │ │ - documentDrag: this.documentDrag, │ │ │ │ │ - stopDown: false │ │ │ │ │ - }; │ │ │ │ │ - var keyboardOptions = { │ │ │ │ │ - keydown: this.handleKeypress │ │ │ │ │ - }; │ │ │ │ │ - this.handlers = { │ │ │ │ │ - keyboard: new OpenLayers.Handler.Keyboard(this, keyboardOptions), │ │ │ │ │ - drag: new OpenLayers.Handler.Drag(this, dragCallbacks, dragOptions) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - removelayer: this.handleMapEvents, │ │ │ │ │ - changelayer: this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ + schemaLocationAttr: function(options) { │ │ │ │ │ + options = OpenLayers.Util.extend({ │ │ │ │ │ + featurePrefix: this.featurePrefix, │ │ │ │ │ + schema: this.schema │ │ │ │ │ + }, options); │ │ │ │ │ + var schemaLocations = OpenLayers.Util.extend({}, this.schemaLocations); │ │ │ │ │ + if (options.schema) { │ │ │ │ │ + schemaLocations[options.featurePrefix] = options.schema │ │ │ │ │ } │ │ │ │ │ - this.layer = null; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, []) │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - this.moveLayerToTop(); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - removelayer: this.handleMapEvents, │ │ │ │ │ - changelayer: this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - return this.handlers.keyboard.activate() && this.handlers.drag.activate() && OpenLayers.Control.prototype.activate.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerBack(); │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - removelayer: this.handleMapEvents, │ │ │ │ │ - changelayer: this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.layer.removeFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.layer.removeFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - this.handlers.drag.deactivate(); │ │ │ │ │ - this.handlers.keyboard.deactivate(); │ │ │ │ │ - var feature = this.feature; │ │ │ │ │ - if (feature && feature.geometry && feature.layer) { │ │ │ │ │ - this.unselectFeature(feature) │ │ │ │ │ + var parts = []; │ │ │ │ │ + var uri; │ │ │ │ │ + for (var key in schemaLocations) { │ │ │ │ │ + uri = this.namespaces[key]; │ │ │ │ │ + if (uri) { │ │ │ │ │ + parts.push(uri + " " + schemaLocations[key]) │ │ │ │ │ } │ │ │ │ │ - deactivated = true │ │ │ │ │ } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - beforeSelectFeature: function(feature) { │ │ │ │ │ - return this.layer.events.triggerEvent("beforefeaturemodified", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ + var value = parts.join(" ") || undefined; │ │ │ │ │ + return value │ │ │ │ │ }, │ │ │ │ │ - selectFeature: function(feature) { │ │ │ │ │ - if (this.feature === feature || this.geometryTypes && OpenLayers.Util.indexOf(this.geometryTypes, feature.geometry.CLASS_NAME) == -1) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (this.beforeSelectFeature(feature) !== false) { │ │ │ │ │ - if (this.feature) { │ │ │ │ │ - this.unselectFeature(this.feature) │ │ │ │ │ + setFilterProperty: function(filter) { │ │ │ │ │ + if (filter.filters) { │ │ │ │ │ + for (var i = 0, len = filter.filters.length; i < len; ++i) { │ │ │ │ │ + OpenLayers.Format.WFST.v1.prototype.setFilterProperty.call(this, filter.filters[i]) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + if (filter instanceof OpenLayers.Filter.Spatial && !filter.property) { │ │ │ │ │ + filter.property = this.geometryName │ │ │ │ │ } │ │ │ │ │ - this.feature = feature; │ │ │ │ │ - this.layer.selectedFeatures.push(feature); │ │ │ │ │ - this.layer.drawFeature(feature, "select"); │ │ │ │ │ - this.modified = false; │ │ │ │ │ - this.resetVertices(); │ │ │ │ │ - this.onModificationStart(this.feature) │ │ │ │ │ - } │ │ │ │ │ - var modified = feature.modified; │ │ │ │ │ - if (feature.geometry && !(modified && modified.geometry)) { │ │ │ │ │ - this._originalGeometry = feature.geometry.clone() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - unselectFeature: function(feature) { │ │ │ │ │ - this.layer.removeFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - this.layer.destroyFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.virtualVertices = []; │ │ │ │ │ - if (this.dragHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.dragHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - delete this.dragHandle │ │ │ │ │ - } │ │ │ │ │ - if (this.radiusHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - delete this.radiusHandle │ │ │ │ │ } │ │ │ │ │ - this.layer.drawFeature(this.feature, "default"); │ │ │ │ │ - this.feature = null; │ │ │ │ │ - OpenLayers.Util.removeItem(this.layer.selectedFeatures, feature); │ │ │ │ │ - this.onModificationEnd(feature); │ │ │ │ │ - this.layer.events.triggerEvent("afterfeaturemodified", { │ │ │ │ │ - feature: feature, │ │ │ │ │ - modified: this.modified │ │ │ │ │ - }); │ │ │ │ │ - this.modified = false │ │ │ │ │ }, │ │ │ │ │ - dragStart: function(feature) { │ │ │ │ │ - var isPoint = feature.geometry.CLASS_NAME == "OpenLayers.Geometry.Point"; │ │ │ │ │ - if (!this.standalone && (!feature._sketch && isPoint || !feature._sketch)) { │ │ │ │ │ - if (this.toggle && this.feature === feature) { │ │ │ │ │ - this._unselect = feature │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFST.v1" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Geometry.MultiLineString = OpenLayers.Class(OpenLayers.Geometry.Collection, { │ │ │ │ │ + componentTypes: ["OpenLayers.Geometry.LineString"], │ │ │ │ │ + split: function(geometry, options) { │ │ │ │ │ + var results = null; │ │ │ │ │ + var mutual = options && options.mutual; │ │ │ │ │ + var splits, sourceLine, sourceLines, sourceSplit, targetSplit; │ │ │ │ │ + var sourceParts = []; │ │ │ │ │ + var targetParts = [geometry]; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ + sourceLine = this.components[i]; │ │ │ │ │ + sourceSplit = false; │ │ │ │ │ + for (var j = 0; j < targetParts.length; ++j) { │ │ │ │ │ + splits = sourceLine.split(targetParts[j], options); │ │ │ │ │ + if (splits) { │ │ │ │ │ + if (mutual) { │ │ │ │ │ + sourceLines = splits[0]; │ │ │ │ │ + for (var k = 0, klen = sourceLines.length; k < klen; ++k) { │ │ │ │ │ + if (k === 0 && sourceParts.length) { │ │ │ │ │ + sourceParts[sourceParts.length - 1].addComponent(sourceLines[k]) │ │ │ │ │ + } else { │ │ │ │ │ + sourceParts.push(new OpenLayers.Geometry.MultiLineString([sourceLines[k]])) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + sourceSplit = true; │ │ │ │ │ + splits = splits[1] │ │ │ │ │ + } │ │ │ │ │ + if (splits.length) { │ │ │ │ │ + splits.unshift(j, 1); │ │ │ │ │ + Array.prototype.splice.apply(targetParts, splits); │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!sourceSplit) { │ │ │ │ │ + if (sourceParts.length) { │ │ │ │ │ + sourceParts[sourceParts.length - 1].addComponent(sourceLine.clone()) │ │ │ │ │ + } else { │ │ │ │ │ + sourceParts = [new OpenLayers.Geometry.MultiLineString(sourceLine.clone())] │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.selectFeature(feature) │ │ │ │ │ } │ │ │ │ │ - if (feature._sketch || isPoint) { │ │ │ │ │ - this.vertex = feature; │ │ │ │ │ - this.handlers.drag.stopDown = true │ │ │ │ │ + if (sourceParts && sourceParts.length > 1) { │ │ │ │ │ + sourceSplit = true │ │ │ │ │ + } else { │ │ │ │ │ + sourceParts = [] │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - dragVertex: function(vertex, pixel) { │ │ │ │ │ - var pos = this.map.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geom = vertex.geometry; │ │ │ │ │ - geom.move(pos.lon - geom.x, pos.lat - geom.y); │ │ │ │ │ - this.modified = true; │ │ │ │ │ - if (this.feature.geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - this.layer.events.triggerEvent("vertexmodified", { │ │ │ │ │ - vertex: vertex.geometry, │ │ │ │ │ - feature: this.feature, │ │ │ │ │ - pixel: pixel │ │ │ │ │ - }) │ │ │ │ │ + if (targetParts && targetParts.length > 1) { │ │ │ │ │ + targetSplit = true │ │ │ │ │ } else { │ │ │ │ │ - if (vertex._index) { │ │ │ │ │ - vertex.geometry.parent.addComponent(vertex.geometry, vertex._index); │ │ │ │ │ - delete vertex._index; │ │ │ │ │ - OpenLayers.Util.removeItem(this.virtualVertices, vertex); │ │ │ │ │ - this.vertices.push(vertex) │ │ │ │ │ - } else if (vertex == this.dragHandle) { │ │ │ │ │ - this.layer.removeFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - if (this.radiusHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.radiusHandle = null │ │ │ │ │ - } │ │ │ │ │ - } else if (vertex !== this.radiusHandle) { │ │ │ │ │ - this.layer.events.triggerEvent("vertexmodified", { │ │ │ │ │ - vertex: vertex.geometry, │ │ │ │ │ - feature: this.feature, │ │ │ │ │ - pixel: pixel │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - if (this.virtualVertices.length > 0) { │ │ │ │ │ - this.layer.destroyFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.virtualVertices = [] │ │ │ │ │ + targetParts = [] │ │ │ │ │ + } │ │ │ │ │ + if (sourceSplit || targetSplit) { │ │ │ │ │ + if (mutual) { │ │ │ │ │ + results = [sourceParts, targetParts] │ │ │ │ │ + } else { │ │ │ │ │ + results = targetParts │ │ │ │ │ } │ │ │ │ │ - this.layer.drawFeature(this.feature, this.standalone ? undefined : "select") │ │ │ │ │ } │ │ │ │ │ - this.layer.drawFeature(vertex) │ │ │ │ │ + return results │ │ │ │ │ }, │ │ │ │ │ - dragComplete: function(vertex) { │ │ │ │ │ - this.resetVertices(); │ │ │ │ │ - this.setFeatureState(); │ │ │ │ │ - this.onModification(this.feature); │ │ │ │ │ - this.layer.events.triggerEvent("featuremodified", { │ │ │ │ │ - feature: this.feature │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - setFeatureState: function() { │ │ │ │ │ - if (this.feature.state != OpenLayers.State.INSERT && this.feature.state != OpenLayers.State.DELETE) { │ │ │ │ │ - this.feature.state = OpenLayers.State.UPDATE; │ │ │ │ │ - if (this.modified && this._originalGeometry) { │ │ │ │ │ - var feature = this.feature; │ │ │ │ │ - feature.modified = OpenLayers.Util.extend(feature.modified, { │ │ │ │ │ - geometry: this._originalGeometry │ │ │ │ │ - }); │ │ │ │ │ - delete this._originalGeometry │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - resetVertices: function() { │ │ │ │ │ - if (this.vertices.length > 0) { │ │ │ │ │ - this.layer.removeFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.vertices = [] │ │ │ │ │ - } │ │ │ │ │ - if (this.virtualVertices.length > 0) { │ │ │ │ │ - this.layer.removeFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.virtualVertices = [] │ │ │ │ │ - } │ │ │ │ │ - if (this.dragHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.dragHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.dragHandle = null │ │ │ │ │ - } │ │ │ │ │ - if (this.radiusHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.radiusHandle = null │ │ │ │ │ - } │ │ │ │ │ - if (this.feature && this.feature.geometry.CLASS_NAME != "OpenLayers.Geometry.Point") { │ │ │ │ │ - if (this.mode & OpenLayers.Control.ModifyFeature.DRAG) { │ │ │ │ │ - this.collectDragHandle() │ │ │ │ │ - } │ │ │ │ │ - if (this.mode & (OpenLayers.Control.ModifyFeature.ROTATE | OpenLayers.Control.ModifyFeature.RESIZE)) { │ │ │ │ │ - this.collectRadiusHandle() │ │ │ │ │ - } │ │ │ │ │ - if (this.mode & OpenLayers.Control.ModifyFeature.RESHAPE) { │ │ │ │ │ - if (!(this.mode & OpenLayers.Control.ModifyFeature.RESIZE)) { │ │ │ │ │ - this.collectVertices() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - handleKeypress: function(evt) { │ │ │ │ │ - var code = evt.keyCode; │ │ │ │ │ - if (this.feature && OpenLayers.Util.indexOf(this.deleteCodes, code) != -1) { │ │ │ │ │ - var vertex = this.layer.getFeatureFromEvent(this.handlers.drag.evt); │ │ │ │ │ - if (vertex && OpenLayers.Util.indexOf(this.vertices, vertex) != -1 && !this.handlers.drag.dragging && vertex.geometry.parent) { │ │ │ │ │ - vertex.geometry.parent.removeComponent(vertex.geometry); │ │ │ │ │ - this.layer.events.triggerEvent("vertexremoved", { │ │ │ │ │ - vertex: vertex.geometry, │ │ │ │ │ - feature: this.feature, │ │ │ │ │ - pixel: evt.xy │ │ │ │ │ - }); │ │ │ │ │ - this.layer.drawFeature(this.feature, this.standalone ? undefined : "select"); │ │ │ │ │ - this.modified = true; │ │ │ │ │ - this.resetVertices(); │ │ │ │ │ - this.setFeatureState(); │ │ │ │ │ - this.onModification(this.feature); │ │ │ │ │ - this.layer.events.triggerEvent("featuremodified", { │ │ │ │ │ - feature: this.feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - collectVertices: function() { │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - this.virtualVertices = []; │ │ │ │ │ - var control = this; │ │ │ │ │ - │ │ │ │ │ - function collectComponentVertices(geometry) { │ │ │ │ │ - var i, vertex, component, len; │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - vertex = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - vertex._sketch = true; │ │ │ │ │ - vertex.renderIntent = control.vertexRenderIntent; │ │ │ │ │ - control.vertices.push(vertex) │ │ │ │ │ - } else { │ │ │ │ │ - var numVert = geometry.components.length; │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ - numVert -= 1 │ │ │ │ │ - } │ │ │ │ │ - for (i = 0; i < numVert; ++i) { │ │ │ │ │ - component = geometry.components[i]; │ │ │ │ │ - if (component.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - vertex = new OpenLayers.Feature.Vector(component); │ │ │ │ │ - vertex._sketch = true; │ │ │ │ │ - vertex.renderIntent = control.vertexRenderIntent; │ │ │ │ │ - control.vertices.push(vertex) │ │ │ │ │ - } else { │ │ │ │ │ - collectComponentVertices(component) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (control.createVertices && geometry.CLASS_NAME != "OpenLayers.Geometry.MultiPoint") { │ │ │ │ │ - for (i = 0, len = geometry.components.length; i < len - 1; ++i) { │ │ │ │ │ - var prevVertex = geometry.components[i]; │ │ │ │ │ - var nextVertex = geometry.components[i + 1]; │ │ │ │ │ - if (prevVertex.CLASS_NAME == "OpenLayers.Geometry.Point" && nextVertex.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - var x = (prevVertex.x + nextVertex.x) / 2; │ │ │ │ │ - var y = (prevVertex.y + nextVertex.y) / 2; │ │ │ │ │ - var point = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Point(x, y), null, control.virtualStyle); │ │ │ │ │ - point.geometry.parent = geometry; │ │ │ │ │ - point._index = i + 1; │ │ │ │ │ - point._sketch = true; │ │ │ │ │ - control.virtualVertices.push(point) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - collectComponentVertices.call(this, this.feature.geometry); │ │ │ │ │ - this.layer.addFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.layer.addFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - collectDragHandle: function() { │ │ │ │ │ - var geometry = this.feature.geometry; │ │ │ │ │ - var center = geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - var originGeometry = new OpenLayers.Geometry.Point(center.lon, center.lat); │ │ │ │ │ - var origin = new OpenLayers.Feature.Vector(originGeometry); │ │ │ │ │ - originGeometry.move = function(x, y) { │ │ │ │ │ - OpenLayers.Geometry.Point.prototype.move.call(this, x, y); │ │ │ │ │ - geometry.move(x, y) │ │ │ │ │ - }; │ │ │ │ │ - origin._sketch = true; │ │ │ │ │ - this.dragHandle = origin; │ │ │ │ │ - this.dragHandle.renderIntent = this.vertexRenderIntent; │ │ │ │ │ - this.layer.addFeatures([this.dragHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - collectRadiusHandle: function() { │ │ │ │ │ - var geometry = this.feature.geometry; │ │ │ │ │ - var bounds = geometry.getBounds(); │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var originGeometry = new OpenLayers.Geometry.Point(center.lon, center.lat); │ │ │ │ │ - var radiusGeometry = new OpenLayers.Geometry.Point(bounds.right, bounds.bottom); │ │ │ │ │ - var radius = new OpenLayers.Feature.Vector(radiusGeometry); │ │ │ │ │ - var resize = this.mode & OpenLayers.Control.ModifyFeature.RESIZE; │ │ │ │ │ - var reshape = this.mode & OpenLayers.Control.ModifyFeature.RESHAPE; │ │ │ │ │ - var rotate = this.mode & OpenLayers.Control.ModifyFeature.ROTATE; │ │ │ │ │ - radiusGeometry.move = function(x, y) { │ │ │ │ │ - OpenLayers.Geometry.Point.prototype.move.call(this, x, y); │ │ │ │ │ - var dx1 = this.x - originGeometry.x; │ │ │ │ │ - var dy1 = this.y - originGeometry.y; │ │ │ │ │ - var dx0 = dx1 - x; │ │ │ │ │ - var dy0 = dy1 - y; │ │ │ │ │ - if (rotate) { │ │ │ │ │ - var a0 = Math.atan2(dy0, dx0); │ │ │ │ │ - var a1 = Math.atan2(dy1, dx1); │ │ │ │ │ - var angle = a1 - a0; │ │ │ │ │ - angle *= 180 / Math.PI; │ │ │ │ │ - geometry.rotate(angle, originGeometry) │ │ │ │ │ - } │ │ │ │ │ - if (resize) { │ │ │ │ │ - var scale, ratio; │ │ │ │ │ - if (reshape) { │ │ │ │ │ - scale = dy1 / dy0; │ │ │ │ │ - ratio = dx1 / dx0 / scale │ │ │ │ │ - } else { │ │ │ │ │ - var l0 = Math.sqrt(dx0 * dx0 + dy0 * dy0); │ │ │ │ │ - var l1 = Math.sqrt(dx1 * dx1 + dy1 * dy1); │ │ │ │ │ - scale = l1 / l0 │ │ │ │ │ - } │ │ │ │ │ - geometry.resize(scale, originGeometry, ratio) │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - radius._sketch = true; │ │ │ │ │ - this.radiusHandle = radius; │ │ │ │ │ - this.radiusHandle.renderIntent = this.vertexRenderIntent; │ │ │ │ │ - this.layer.addFeatures([this.radiusHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.handlers.drag.setMap(map); │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - handleMapEvents: function(evt) { │ │ │ │ │ - if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ - this.moveLayerToTop() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - moveLayerToTop: function() { │ │ │ │ │ - var index = Math.max(this.map.Z_INDEX_BASE["Feature"] - 1, this.layer.getZIndex()) + 1; │ │ │ │ │ - this.layer.setZIndex(index) │ │ │ │ │ - }, │ │ │ │ │ - moveLayerBack: function() { │ │ │ │ │ - var index = this.layer.getZIndex() - 1; │ │ │ │ │ - if (index >= this.map.Z_INDEX_BASE["Feature"]) { │ │ │ │ │ - this.layer.setZIndex(index) │ │ │ │ │ - } else { │ │ │ │ │ - this.map.setLayerZIndex(this.layer, this.map.getLayerIndex(this.layer)) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ModifyFeature" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.ModifyFeature.RESHAPE = 1; │ │ │ │ │ -OpenLayers.Control.ModifyFeature.RESIZE = 2; │ │ │ │ │ -OpenLayers.Control.ModifyFeature.ROTATE = 4; │ │ │ │ │ -OpenLayers.Control.ModifyFeature.DRAG = 8; │ │ │ │ │ -OpenLayers.Control.DrawFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - layer: null, │ │ │ │ │ - callbacks: null, │ │ │ │ │ - multi: false, │ │ │ │ │ - featureAdded: function() {}, │ │ │ │ │ - initialize: function(layer, handler, options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.callbacks = OpenLayers.Util.extend({ │ │ │ │ │ - done: this.drawFeature, │ │ │ │ │ - modify: function(vertex, feature) { │ │ │ │ │ - this.layer.events.triggerEvent("sketchmodified", { │ │ │ │ │ - vertex: vertex, │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - create: function(vertex, feature) { │ │ │ │ │ - this.layer.events.triggerEvent("sketchstarted", { │ │ │ │ │ - vertex: vertex, │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, this.callbacks); │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - this.handlerOptions = this.handlerOptions || {}; │ │ │ │ │ - this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults(this.handlerOptions.layerOptions, { │ │ │ │ │ - renderers: layer.renderers, │ │ │ │ │ - rendererOptions: layer.rendererOptions │ │ │ │ │ - }); │ │ │ │ │ - if (!("multi" in this.handlerOptions)) { │ │ │ │ │ - this.handlerOptions.multi = this.multi │ │ │ │ │ - } │ │ │ │ │ - var sketchStyle = this.layer.styleMap && this.layer.styleMap.styles.temporary; │ │ │ │ │ - if (sketchStyle) { │ │ │ │ │ - this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults(this.handlerOptions.layerOptions, { │ │ │ │ │ - styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ - default: sketchStyle │ │ │ │ │ - }) │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.handler = new handler(this, this.callbacks, this.handlerOptions) │ │ │ │ │ - }, │ │ │ │ │ - drawFeature: function(geometry) { │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - var proceed = this.layer.events.triggerEvent("sketchcomplete", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - if (proceed !== false) { │ │ │ │ │ - feature.state = OpenLayers.State.INSERT; │ │ │ │ │ - this.layer.addFeatures([feature]); │ │ │ │ │ - this.featureAdded(feature); │ │ │ │ │ - this.events.triggerEvent("featureadded", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - insertXY: function(x, y) { │ │ │ │ │ - if (this.handler && this.handler.line) { │ │ │ │ │ - this.handler.insertXY(x, y) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - insertDeltaXY: function(dx, dy) { │ │ │ │ │ - if (this.handler && this.handler.line) { │ │ │ │ │ - this.handler.insertDeltaXY(dx, dy) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - insertDirectionLength: function(direction, length) { │ │ │ │ │ - if (this.handler && this.handler.line) { │ │ │ │ │ - this.handler.insertDirectionLength(direction, length) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - insertDeflectionLength: function(deflection, length) { │ │ │ │ │ - if (this.handler && this.handler.line) { │ │ │ │ │ - this.handler.insertDeflectionLength(deflection, length) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - undo: function() { │ │ │ │ │ - return this.handler.undo && this.handler.undo() │ │ │ │ │ - }, │ │ │ │ │ - redo: function() { │ │ │ │ │ - return this.handler.redo && this.handler.redo() │ │ │ │ │ - }, │ │ │ │ │ - finishSketch: function() { │ │ │ │ │ - this.handler.finishGeometry() │ │ │ │ │ - }, │ │ │ │ │ - cancel: function() { │ │ │ │ │ - this.handler.cancel() │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.DrawFeature" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.Geolocate = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - geolocation: null, │ │ │ │ │ - available: "geolocation" in navigator, │ │ │ │ │ - bind: true, │ │ │ │ │ - watch: false, │ │ │ │ │ - geolocationOptions: null, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (this.available && !this.geolocation) { │ │ │ │ │ - this.geolocation = navigator.geolocation │ │ │ │ │ - } │ │ │ │ │ - if (!this.geolocation) { │ │ │ │ │ - this.events.triggerEvent("locationuncapable"); │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - if (this.watch) { │ │ │ │ │ - this.watchId = this.geolocation.watchPosition(OpenLayers.Function.bind(this.geolocate, this), OpenLayers.Function.bind(this.failure, this), this.geolocationOptions) │ │ │ │ │ - } else { │ │ │ │ │ - this.getCurrentLocation() │ │ │ │ │ - } │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active && this.watchId !== null) { │ │ │ │ │ - this.geolocation.clearWatch(this.watchId) │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Control.prototype.deactivate.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - geolocate: function(position) { │ │ │ │ │ - var center = new OpenLayers.LonLat(position.coords.longitude, position.coords.latitude).transform(new OpenLayers.Projection("EPSG:4326"), this.map.getProjectionObject()); │ │ │ │ │ - if (this.bind) { │ │ │ │ │ - this.map.setCenter(center) │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("locationupdated", { │ │ │ │ │ - position: position, │ │ │ │ │ - point: new OpenLayers.Geometry.Point(center.lon, center.lat) │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - getCurrentLocation: function() { │ │ │ │ │ - if (!this.active || this.watch) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - this.geolocation.getCurrentPosition(OpenLayers.Function.bind(this.geolocate, this), OpenLayers.Function.bind(this.failure, this), this.geolocationOptions); │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - failure: function(error) { │ │ │ │ │ - this.events.triggerEvent("locationfailed", { │ │ │ │ │ - error: error │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Geolocate" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Events.buttonclick = OpenLayers.Class({ │ │ │ │ │ - target: null, │ │ │ │ │ - events: ["mousedown", "mouseup", "click", "dblclick", "touchstart", "touchmove", "touchend", "keydown"], │ │ │ │ │ - startRegEx: /^mousedown|touchstart$/, │ │ │ │ │ - cancelRegEx: /^touchmove$/, │ │ │ │ │ - completeRegEx: /^mouseup|touchend$/, │ │ │ │ │ - initialize: function(target) { │ │ │ │ │ - this.target = target; │ │ │ │ │ - for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ - this.target.register(this.events[i], this, this.buttonClick, { │ │ │ │ │ - extension: true │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ - this.target.unregister(this.events[i], this, this.buttonClick) │ │ │ │ │ - } │ │ │ │ │ - delete this.target │ │ │ │ │ - }, │ │ │ │ │ - getPressedButton: function(element) { │ │ │ │ │ - var depth = 3, │ │ │ │ │ - button; │ │ │ │ │ - do { │ │ │ │ │ - if (OpenLayers.Element.hasClass(element, "olButton")) { │ │ │ │ │ - button = element; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - element = element.parentNode │ │ │ │ │ - } while (--depth > 0 && element); │ │ │ │ │ - return button │ │ │ │ │ - }, │ │ │ │ │ - ignore: function(element) { │ │ │ │ │ - var depth = 3, │ │ │ │ │ - ignore = false; │ │ │ │ │ - do { │ │ │ │ │ - if (element.nodeName.toLowerCase() === "a") { │ │ │ │ │ - ignore = true; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - element = element.parentNode │ │ │ │ │ - } while (--depth > 0 && element); │ │ │ │ │ - return ignore │ │ │ │ │ - }, │ │ │ │ │ - buttonClick: function(evt) { │ │ │ │ │ - var propagate = true, │ │ │ │ │ - element = OpenLayers.Event.element(evt); │ │ │ │ │ - if (element && (OpenLayers.Event.isLeftClick(evt) || !~evt.type.indexOf("mouse"))) { │ │ │ │ │ - var button = this.getPressedButton(element); │ │ │ │ │ - if (button) { │ │ │ │ │ - if (evt.type === "keydown") { │ │ │ │ │ - switch (evt.keyCode) { │ │ │ │ │ - case OpenLayers.Event.KEY_RETURN: │ │ │ │ │ - case OpenLayers.Event.KEY_SPACE: │ │ │ │ │ - this.target.triggerEvent("buttonclick", { │ │ │ │ │ - buttonElement: button │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } else if (this.startEvt) { │ │ │ │ │ - if (this.completeRegEx.test(evt.type)) { │ │ │ │ │ - var pos = OpenLayers.Util.pagePosition(button); │ │ │ │ │ - var viewportElement = OpenLayers.Util.getViewportElement(); │ │ │ │ │ - var scrollTop = window.pageYOffset || viewportElement.scrollTop; │ │ │ │ │ - var scrollLeft = window.pageXOffset || viewportElement.scrollLeft; │ │ │ │ │ - pos[0] = pos[0] - scrollLeft; │ │ │ │ │ - pos[1] = pos[1] - scrollTop; │ │ │ │ │ - this.target.triggerEvent("buttonclick", { │ │ │ │ │ - buttonElement: button, │ │ │ │ │ - buttonXY: { │ │ │ │ │ - x: this.startEvt.clientX - pos[0], │ │ │ │ │ - y: this.startEvt.clientY - pos[1] │ │ │ │ │ + splitWith: function(geometry, options) { │ │ │ │ │ + var results = null; │ │ │ │ │ + var mutual = options && options.mutual; │ │ │ │ │ + var splits, targetLine, sourceLines, sourceSplit, targetSplit, sourceParts, targetParts; │ │ │ │ │ + if (geometry instanceof OpenLayers.Geometry.LineString) { │ │ │ │ │ + targetParts = []; │ │ │ │ │ + sourceParts = [geometry]; │ │ │ │ │ + for (var i = 0, len = this.components.length; i < len; ++i) { │ │ │ │ │ + targetSplit = false; │ │ │ │ │ + targetLine = this.components[i]; │ │ │ │ │ + for (var j = 0; j < sourceParts.length; ++j) { │ │ │ │ │ + splits = sourceParts[j].split(targetLine, options); │ │ │ │ │ + if (splits) { │ │ │ │ │ + if (mutual) { │ │ │ │ │ + sourceLines = splits[0]; │ │ │ │ │ + if (sourceLines.length) { │ │ │ │ │ + sourceLines.unshift(j, 1); │ │ │ │ │ + Array.prototype.splice.apply(sourceParts, sourceLines); │ │ │ │ │ + j += sourceLines.length - 2 │ │ │ │ │ } │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - if (this.cancelRegEx.test(evt.type)) { │ │ │ │ │ - delete this.startEvt │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false │ │ │ │ │ - } │ │ │ │ │ - if (this.startRegEx.test(evt.type)) { │ │ │ │ │ - this.startEvt = evt; │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - propagate = !this.ignore(OpenLayers.Event.element(evt)); │ │ │ │ │ - delete this.startEvt │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return propagate │ │ │ │ │ - } │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.Panel = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - controls: null, │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - defaultControl: null, │ │ │ │ │ - saveState: false, │ │ │ │ │ - allowDepress: false, │ │ │ │ │ - activeState: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.controls = []; │ │ │ │ │ - this.activeState = {} │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onButtonClick) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - for (var ctl, i = this.controls.length - 1; i >= 0; i--) { │ │ │ │ │ - ctl = this.controls[i]; │ │ │ │ │ - if (ctl.events) { │ │ │ │ │ - ctl.events.un({ │ │ │ │ │ - activate: this.iconOn, │ │ │ │ │ - deactivate: this.iconOff │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - ctl.panel_div = null │ │ │ │ │ - } │ │ │ │ │ - this.activeState = null │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - control = this.controls[i]; │ │ │ │ │ - if (control === this.defaultControl || this.saveState && this.activeState[control.id]) { │ │ │ │ │ - control.activate() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.saveState === true) { │ │ │ │ │ - this.defaultControl = null │ │ │ │ │ - } │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - control = this.controls[i]; │ │ │ │ │ - this.activeState[control.id] = control.deactivate() │ │ │ │ │ - } │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (this.outsideViewport) { │ │ │ │ │ - this.events.attachToElement(this.div); │ │ │ │ │ - this.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ - } else { │ │ │ │ │ - this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ - } │ │ │ │ │ - this.addControlsToMap(this.controls); │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - redraw: function() { │ │ │ │ │ - for (var l = this.div.childNodes.length, i = l - 1; i >= 0; i--) { │ │ │ │ │ - this.div.removeChild(this.div.childNodes[i]) │ │ │ │ │ - } │ │ │ │ │ - this.div.innerHTML = ""; │ │ │ │ │ - if (this.active) { │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - this.div.appendChild(this.controls[i].panel_div) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - activateControl: function(control) { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - if (control.type == OpenLayers.Control.TYPE_BUTTON) { │ │ │ │ │ - control.trigger(); │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (control.type == OpenLayers.Control.TYPE_TOGGLE) { │ │ │ │ │ - if (control.active) { │ │ │ │ │ - control.deactivate() │ │ │ │ │ - } else { │ │ │ │ │ - control.activate() │ │ │ │ │ - } │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (this.allowDepress && control.active) { │ │ │ │ │ - control.deactivate() │ │ │ │ │ - } else { │ │ │ │ │ - var c; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - c = this.controls[i]; │ │ │ │ │ - if (c != control && (c.type === OpenLayers.Control.TYPE_TOOL || c.type == null)) { │ │ │ │ │ - c.deactivate() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - control.activate() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - addControls: function(controls) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(controls)) { │ │ │ │ │ - controls = [controls] │ │ │ │ │ - } │ │ │ │ │ - this.controls = this.controls.concat(controls); │ │ │ │ │ - for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ - var control = controls[i], │ │ │ │ │ - element = this.createControlMarkup(control); │ │ │ │ │ - OpenLayers.Element.addClass(element, control.displayClass + "ItemInactive"); │ │ │ │ │ - OpenLayers.Element.addClass(element, "olButton"); │ │ │ │ │ - if (control.title != "" && !element.title) { │ │ │ │ │ - element.title = control.title │ │ │ │ │ - } │ │ │ │ │ - control.panel_div = element │ │ │ │ │ - } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.addControlsToMap(controls); │ │ │ │ │ - this.redraw() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - createControlMarkup: function(control) { │ │ │ │ │ - return document.createElement("div") │ │ │ │ │ - }, │ │ │ │ │ - addControlsToMap: function(controls) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ - control = controls[i]; │ │ │ │ │ - if (control.autoActivate === true) { │ │ │ │ │ - control.autoActivate = false; │ │ │ │ │ - this.map.addControl(control); │ │ │ │ │ - control.autoActivate = true │ │ │ │ │ - } else { │ │ │ │ │ - this.map.addControl(control); │ │ │ │ │ - control.deactivate() │ │ │ │ │ - } │ │ │ │ │ - control.events.on({ │ │ │ │ │ - activate: this.iconOn, │ │ │ │ │ - deactivate: this.iconOff │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - iconOn: function() { │ │ │ │ │ - var d = this.panel_div; │ │ │ │ │ - var re = new RegExp("\\b(" + this.displayClass + "Item)Inactive\\b"); │ │ │ │ │ - d.className = d.className.replace(re, "$1Active") │ │ │ │ │ - }, │ │ │ │ │ - iconOff: function() { │ │ │ │ │ - var d = this.panel_div; │ │ │ │ │ - var re = new RegExp("\\b(" + this.displayClass + "Item)Active\\b"); │ │ │ │ │ - d.className = d.className.replace(re, "$1Inactive") │ │ │ │ │ - }, │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - var controls = this.controls, │ │ │ │ │ - button = evt.buttonElement; │ │ │ │ │ - for (var i = controls.length - 1; i >= 0; --i) { │ │ │ │ │ - if (controls[i].panel_div === button) { │ │ │ │ │ - this.activateControl(controls[i]); │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getControlsBy: function(property, match) { │ │ │ │ │ - var test = typeof match.test == "function"; │ │ │ │ │ - var found = OpenLayers.Array.filter(this.controls, function(item) { │ │ │ │ │ - return item[property] == match || test && match.test(item[property]) │ │ │ │ │ - }); │ │ │ │ │ - return found │ │ │ │ │ - }, │ │ │ │ │ - getControlsByName: function(match) { │ │ │ │ │ - return this.getControlsBy("name", match) │ │ │ │ │ - }, │ │ │ │ │ - getControlsByClass: function(match) { │ │ │ │ │ - return this.getControlsBy("CLASS_NAME", match) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Panel" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.Attribution = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - separator: ", ", │ │ │ │ │ - template: "${layers}", │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - removelayer: this.updateAttribution, │ │ │ │ │ - addlayer: this.updateAttribution, │ │ │ │ │ - changelayer: this.updateAttribution, │ │ │ │ │ - changebaselayer: this.updateAttribution, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - changebaselayer: this.updateAttribution, │ │ │ │ │ - changelayer: this.updateAttribution, │ │ │ │ │ - addlayer: this.updateAttribution, │ │ │ │ │ - removelayer: this.updateAttribution, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.updateAttribution(); │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - updateAttribution: function() { │ │ │ │ │ - var attributions = []; │ │ │ │ │ - if (this.map && this.map.layers) { │ │ │ │ │ - for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if (layer.attribution && layer.getVisibility()) { │ │ │ │ │ - if (OpenLayers.Util.indexOf(attributions, layer.attribution) === -1) { │ │ │ │ │ - attributions.push(layer.attribution) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.div.innerHTML = OpenLayers.String.format(this.template, { │ │ │ │ │ - layers: attributions.join(this.separator) │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Attribution" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.Zoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - zoomInText: "+", │ │ │ │ │ - zoomInId: "olZoomInLink", │ │ │ │ │ - zoomOutText: "−", │ │ │ │ │ - zoomOutId: "olZoomOutLink", │ │ │ │ │ - draw: function() { │ │ │ │ │ - var div = OpenLayers.Control.prototype.draw.apply(this), │ │ │ │ │ - links = this.getOrCreateLinks(div), │ │ │ │ │ - zoomIn = links.zoomIn, │ │ │ │ │ - zoomOut = links.zoomOut, │ │ │ │ │ - eventsInstance = this.map.events; │ │ │ │ │ - if (zoomOut.parentNode !== div) { │ │ │ │ │ - eventsInstance = this.events; │ │ │ │ │ - eventsInstance.attachToElement(zoomOut.parentNode) │ │ │ │ │ - } │ │ │ │ │ - eventsInstance.register("buttonclick", this, this.onZoomClick); │ │ │ │ │ - this.zoomInLink = zoomIn; │ │ │ │ │ - this.zoomOutLink = zoomOut; │ │ │ │ │ - return div │ │ │ │ │ - }, │ │ │ │ │ - getOrCreateLinks: function(el) { │ │ │ │ │ - var zoomIn = document.getElementById(this.zoomInId), │ │ │ │ │ - zoomOut = document.getElementById(this.zoomOutId); │ │ │ │ │ - if (!zoomIn) { │ │ │ │ │ - zoomIn = document.createElement("a"); │ │ │ │ │ - zoomIn.href = "#zoomIn"; │ │ │ │ │ - zoomIn.appendChild(document.createTextNode(this.zoomInText)); │ │ │ │ │ - zoomIn.className = "olControlZoomIn"; │ │ │ │ │ - el.appendChild(zoomIn) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Element.addClass(zoomIn, "olButton"); │ │ │ │ │ - if (!zoomOut) { │ │ │ │ │ - zoomOut = document.createElement("a"); │ │ │ │ │ - zoomOut.href = "#zoomOut"; │ │ │ │ │ - zoomOut.appendChild(document.createTextNode(this.zoomOutText)); │ │ │ │ │ - zoomOut.className = "olControlZoomOut"; │ │ │ │ │ - el.appendChild(zoomOut) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Element.addClass(zoomOut, "olButton"); │ │ │ │ │ - return { │ │ │ │ │ - zoomIn: zoomIn, │ │ │ │ │ - zoomOut: zoomOut │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - onZoomClick: function(evt) { │ │ │ │ │ - var button = evt.buttonElement; │ │ │ │ │ - if (button === this.zoomInLink) { │ │ │ │ │ - this.map.zoomIn() │ │ │ │ │ - } else if (button === this.zoomOutLink) { │ │ │ │ │ - this.map.zoomOut() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onZoomClick) │ │ │ │ │ - } │ │ │ │ │ - delete this.zoomInLink; │ │ │ │ │ - delete this.zoomOutLink; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Zoom" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.Feature = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - EVENTMAP: { │ │ │ │ │ - click: { │ │ │ │ │ - in: "click", │ │ │ │ │ - out: "clickout" │ │ │ │ │ - }, │ │ │ │ │ - mousemove: { │ │ │ │ │ - in: "over", │ │ │ │ │ - out: "out" │ │ │ │ │ - }, │ │ │ │ │ - dblclick: { │ │ │ │ │ - in: "dblclick", │ │ │ │ │ - out: null │ │ │ │ │ - }, │ │ │ │ │ - mousedown: { │ │ │ │ │ - in: null, │ │ │ │ │ - out: null │ │ │ │ │ - }, │ │ │ │ │ - mouseup: { │ │ │ │ │ - in: null, │ │ │ │ │ - out: null │ │ │ │ │ - }, │ │ │ │ │ - touchstart: { │ │ │ │ │ - in: "click", │ │ │ │ │ - out: "clickout" │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - feature: null, │ │ │ │ │ - lastFeature: null, │ │ │ │ │ - down: null, │ │ │ │ │ - up: null, │ │ │ │ │ - clickTolerance: 4, │ │ │ │ │ - geometryTypes: null, │ │ │ │ │ - stopClick: true, │ │ │ │ │ - stopDown: true, │ │ │ │ │ - stopUp: false, │ │ │ │ │ - initialize: function(control, layer, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ - this.layer = layer │ │ │ │ │ - }, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - return OpenLayers.Event.isMultiTouch(evt) ? true : this.mousedown(evt) │ │ │ │ │ - }, │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - OpenLayers.Event.preventDefault(evt) │ │ │ │ │ - }, │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - if (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ - this.down = evt.xy │ │ │ │ │ - } │ │ │ │ │ - return this.handle(evt) ? !this.stopDown : true │ │ │ │ │ - }, │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - this.up = evt.xy; │ │ │ │ │ - return this.handle(evt) ? !this.stopUp : true │ │ │ │ │ - }, │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - return this.handle(evt) ? !this.stopClick : true │ │ │ │ │ - }, │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - if (!this.callbacks["over"] && !this.callbacks["out"]) { │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - this.handle(evt); │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - return !this.handle(evt) │ │ │ │ │ - }, │ │ │ │ │ - geometryTypeMatches: function(feature) { │ │ │ │ │ - return this.geometryTypes == null || OpenLayers.Util.indexOf(this.geometryTypes, feature.geometry.CLASS_NAME) > -1 │ │ │ │ │ - }, │ │ │ │ │ - handle: function(evt) { │ │ │ │ │ - if (this.feature && !this.feature.layer) { │ │ │ │ │ - this.feature = null │ │ │ │ │ - } │ │ │ │ │ - var type = evt.type; │ │ │ │ │ - var handled = false; │ │ │ │ │ - var previouslyIn = !!this.feature; │ │ │ │ │ - var click = type == "click" || type == "dblclick" || type == "touchstart"; │ │ │ │ │ - this.feature = this.layer.getFeatureFromEvent(evt); │ │ │ │ │ - if (this.feature && !this.feature.layer) { │ │ │ │ │ - this.feature = null │ │ │ │ │ - } │ │ │ │ │ - if (this.lastFeature && !this.lastFeature.layer) { │ │ │ │ │ - this.lastFeature = null │ │ │ │ │ - } │ │ │ │ │ - if (this.feature) { │ │ │ │ │ - if (type === "touchstart") { │ │ │ │ │ - OpenLayers.Event.preventDefault(evt) │ │ │ │ │ - } │ │ │ │ │ - var inNew = this.feature != this.lastFeature; │ │ │ │ │ - if (this.geometryTypeMatches(this.feature)) { │ │ │ │ │ - if (previouslyIn && inNew) { │ │ │ │ │ - if (this.lastFeature) { │ │ │ │ │ - this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ - } │ │ │ │ │ - this.triggerCallback(type, "in", [this.feature]) │ │ │ │ │ - } else if (!previouslyIn || click) { │ │ │ │ │ - this.triggerCallback(type, "in", [this.feature]) │ │ │ │ │ - } │ │ │ │ │ - this.lastFeature = this.feature; │ │ │ │ │ - handled = true │ │ │ │ │ - } else { │ │ │ │ │ - if (this.lastFeature && (previouslyIn && inNew || click)) { │ │ │ │ │ - this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ - } │ │ │ │ │ - this.feature = null │ │ │ │ │ - } │ │ │ │ │ - } else if (this.lastFeature && (previouslyIn || click)) { │ │ │ │ │ - this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ - } │ │ │ │ │ - return handled │ │ │ │ │ - }, │ │ │ │ │ - triggerCallback: function(type, mode, args) { │ │ │ │ │ - var key = this.EVENTMAP[type][mode]; │ │ │ │ │ - if (key) { │ │ │ │ │ - if (type == "click" && this.up && this.down) { │ │ │ │ │ - var dpx = Math.sqrt(Math.pow(this.up.x - this.down.x, 2) + Math.pow(this.up.y - this.down.y, 2)); │ │ │ │ │ - if (dpx <= this.clickTolerance) { │ │ │ │ │ - this.callback(key, args) │ │ │ │ │ - } │ │ │ │ │ - this.up = this.down = null │ │ │ │ │ - } else { │ │ │ │ │ - this.callback(key, args) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerToTop(); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - removelayer: this.handleMapEvents, │ │ │ │ │ - changelayer: this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - activated = true │ │ │ │ │ - } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerBack(); │ │ │ │ │ - this.feature = null; │ │ │ │ │ - this.lastFeature = null; │ │ │ │ │ - this.down = null; │ │ │ │ │ - this.up = null; │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - removelayer: this.handleMapEvents, │ │ │ │ │ - changelayer: this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - deactivated = true │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - handleMapEvents: function(evt) { │ │ │ │ │ - if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ - this.moveLayerToTop() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - moveLayerToTop: function() { │ │ │ │ │ - var index = Math.max(this.map.Z_INDEX_BASE["Feature"] - 1, this.layer.getZIndex()) + 1; │ │ │ │ │ - this.layer.setZIndex(index) │ │ │ │ │ - }, │ │ │ │ │ - moveLayerBack: function() { │ │ │ │ │ - var index = this.layer.getZIndex() - 1; │ │ │ │ │ - if (index >= this.map.Z_INDEX_BASE["Feature"]) { │ │ │ │ │ - this.layer.setZIndex(index) │ │ │ │ │ - } else { │ │ │ │ │ - this.map.setLayerZIndex(this.layer, this.map.getLayerIndex(this.layer)) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Feature" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Vector.RootContainer = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - layers: null, │ │ │ │ │ - display: function() {}, │ │ │ │ │ - getFeatureFromEvent: function(evt) { │ │ │ │ │ - var layers = this.layers; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0; i < layers.length; i++) { │ │ │ │ │ - feature = layers[i].getFeatureFromEvent(evt); │ │ │ │ │ - if (feature) { │ │ │ │ │ - return feature │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.collectRoots(); │ │ │ │ │ - map.events.register("changelayer", this, this.handleChangeLayer) │ │ │ │ │ - }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - map.events.unregister("changelayer", this, this.handleChangeLayer); │ │ │ │ │ - this.resetRoots(); │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - collectRoots: function() { │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = 0; i < this.map.layers.length; ++i) { │ │ │ │ │ - layer = this.map.layers[i]; │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ - layer.renderer.moveRoot(this.renderer) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - resetRoots: function() { │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = 0; i < this.layers.length; ++i) { │ │ │ │ │ - layer = this.layers[i]; │ │ │ │ │ - if (this.renderer && layer.renderer.getRenderLayerId() == this.id) { │ │ │ │ │ - this.renderer.moveRoot(layer.renderer) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - handleChangeLayer: function(evt) { │ │ │ │ │ - var layer = evt.layer; │ │ │ │ │ - if (evt.property == "order" && OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ - this.resetRoots(); │ │ │ │ │ - this.collectRoots() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Vector.RootContainer" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.SelectFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - multipleKey: null, │ │ │ │ │ - toggleKey: null, │ │ │ │ │ - multiple: false, │ │ │ │ │ - clickout: true, │ │ │ │ │ - toggle: false, │ │ │ │ │ - hover: false, │ │ │ │ │ - highlightOnly: false, │ │ │ │ │ - box: false, │ │ │ │ │ - onBeforeSelect: function() {}, │ │ │ │ │ - onSelect: function() {}, │ │ │ │ │ - onUnselect: function() {}, │ │ │ │ │ - scope: null, │ │ │ │ │ - geometryTypes: null, │ │ │ │ │ - layer: null, │ │ │ │ │ - layers: null, │ │ │ │ │ - callbacks: null, │ │ │ │ │ - selectStyle: null, │ │ │ │ │ - renderIntent: "select", │ │ │ │ │ - handlers: null, │ │ │ │ │ - initialize: function(layers, options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (this.scope === null) { │ │ │ │ │ - this.scope = this │ │ │ │ │ - } │ │ │ │ │ - this.initLayer(layers); │ │ │ │ │ - var callbacks = { │ │ │ │ │ - click: this.clickFeature, │ │ │ │ │ - clickout: this.clickoutFeature │ │ │ │ │ - }; │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - callbacks.over = this.overFeature; │ │ │ │ │ - callbacks.out = this.outFeature │ │ │ │ │ - } │ │ │ │ │ - this.callbacks = OpenLayers.Util.extend(callbacks, this.callbacks); │ │ │ │ │ - this.handlers = { │ │ │ │ │ - feature: new OpenLayers.Handler.Feature(this, this.layer, this.callbacks, { │ │ │ │ │ - geometryTypes: this.geometryTypes │ │ │ │ │ - }) │ │ │ │ │ - }; │ │ │ │ │ - if (this.box) { │ │ │ │ │ - this.handlers.box = new OpenLayers.Handler.Box(this, { │ │ │ │ │ - done: this.selectBox │ │ │ │ │ - }, { │ │ │ │ │ - boxDivClassName: "olHandlerBoxSelectFeature" │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - initLayer: function(layers) { │ │ │ │ │ - if (OpenLayers.Util.isArray(layers)) { │ │ │ │ │ - this.layers = layers; │ │ │ │ │ - this.layer = new OpenLayers.Layer.Vector.RootContainer(this.id + "_container", { │ │ │ │ │ - layers: layers │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - this.layer = layers │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.active && this.layers) { │ │ │ │ │ - this.map.removeLayer(this.layer) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - if (this.layers) { │ │ │ │ │ - this.layer.destroy() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - if (this.layers) { │ │ │ │ │ - this.map.addLayer(this.layer) │ │ │ │ │ - } │ │ │ │ │ - this.handlers.feature.activate(); │ │ │ │ │ - if (this.box && this.handlers.box) { │ │ │ │ │ - this.handlers.box.activate() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Control.prototype.activate.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.handlers.feature.deactivate(); │ │ │ │ │ - if (this.handlers.box) { │ │ │ │ │ - this.handlers.box.deactivate() │ │ │ │ │ - } │ │ │ │ │ - if (this.layers) { │ │ │ │ │ - this.map.removeLayer(this.layer) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Control.prototype.deactivate.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - unselectAll: function(options) { │ │ │ │ │ - var layers = this.layers || [this.layer], │ │ │ │ │ - layer, feature, l, numExcept; │ │ │ │ │ - for (l = 0; l < layers.length; ++l) { │ │ │ │ │ - layer = layers[l]; │ │ │ │ │ - numExcept = 0; │ │ │ │ │ - if (layer.selectedFeatures != null) { │ │ │ │ │ - while (layer.selectedFeatures.length > numExcept) { │ │ │ │ │ - feature = layer.selectedFeatures[numExcept]; │ │ │ │ │ - if (!options || options.except != feature) { │ │ │ │ │ - this.unselect(feature) │ │ │ │ │ - } else { │ │ │ │ │ - ++numExcept │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - clickFeature: function(feature) { │ │ │ │ │ - if (!this.hover) { │ │ │ │ │ - var selected = OpenLayers.Util.indexOf(feature.layer.selectedFeatures, feature) > -1; │ │ │ │ │ - if (selected) { │ │ │ │ │ - if (this.toggleSelect()) { │ │ │ │ │ - this.unselect(feature) │ │ │ │ │ - } else if (!this.multipleSelect()) { │ │ │ │ │ - this.unselectAll({ │ │ │ │ │ - except: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - if (!this.multipleSelect()) { │ │ │ │ │ - this.unselectAll({ │ │ │ │ │ - except: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.select(feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - multipleSelect: function() { │ │ │ │ │ - return this.multiple || this.handlers.feature.evt && this.handlers.feature.evt[this.multipleKey] │ │ │ │ │ - }, │ │ │ │ │ - toggleSelect: function() { │ │ │ │ │ - return this.toggle || this.handlers.feature.evt && this.handlers.feature.evt[this.toggleKey] │ │ │ │ │ - }, │ │ │ │ │ - clickoutFeature: function(feature) { │ │ │ │ │ - if (!this.hover && this.clickout) { │ │ │ │ │ - this.unselectAll() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - overFeature: function(feature) { │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - if (this.highlightOnly) { │ │ │ │ │ - this.highlight(feature) │ │ │ │ │ - } else if (OpenLayers.Util.indexOf(layer.selectedFeatures, feature) == -1) { │ │ │ │ │ - this.select(feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - outFeature: function(feature) { │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - if (this.highlightOnly) { │ │ │ │ │ - if (feature._lastHighlighter == this.id) { │ │ │ │ │ - if (feature._prevHighlighter && feature._prevHighlighter != this.id) { │ │ │ │ │ - delete feature._lastHighlighter; │ │ │ │ │ - var control = this.map.getControl(feature._prevHighlighter); │ │ │ │ │ - if (control) { │ │ │ │ │ - control.highlight(feature) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.unhighlight(feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.unselect(feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - highlight: function(feature) { │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - var cont = this.events.triggerEvent("beforefeaturehighlighted", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - feature._prevHighlighter = feature._lastHighlighter; │ │ │ │ │ - feature._lastHighlighter = this.id; │ │ │ │ │ - var style = this.selectStyle || this.renderIntent; │ │ │ │ │ - layer.drawFeature(feature, style); │ │ │ │ │ - this.events.triggerEvent("featurehighlighted", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - unhighlight: function(feature) { │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - if (feature._prevHighlighter == undefined) { │ │ │ │ │ - delete feature._lastHighlighter │ │ │ │ │ - } else if (feature._prevHighlighter == this.id) { │ │ │ │ │ - delete feature._prevHighlighter │ │ │ │ │ - } else { │ │ │ │ │ - feature._lastHighlighter = feature._prevHighlighter; │ │ │ │ │ - delete feature._prevHighlighter │ │ │ │ │ - } │ │ │ │ │ - layer.drawFeature(feature, feature.style || feature.layer.style || "default"); │ │ │ │ │ - this.events.triggerEvent("featureunhighlighted", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - select: function(feature) { │ │ │ │ │ - var cont = this.onBeforeSelect.call(this.scope, feature); │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - cont = layer.events.triggerEvent("beforefeatureselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - layer.selectedFeatures.push(feature); │ │ │ │ │ - this.highlight(feature); │ │ │ │ │ - if (!this.handlers.feature.lastFeature) { │ │ │ │ │ - this.handlers.feature.lastFeature = layer.selectedFeatures[0] │ │ │ │ │ - } │ │ │ │ │ - layer.events.triggerEvent("featureselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - this.onSelect.call(this.scope, feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - unselect: function(feature) { │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - this.unhighlight(feature); │ │ │ │ │ - OpenLayers.Util.removeItem(layer.selectedFeatures, feature); │ │ │ │ │ - layer.events.triggerEvent("featureunselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - this.onUnselect.call(this.scope, feature) │ │ │ │ │ - }, │ │ │ │ │ - selectBox: function(position) { │ │ │ │ │ - if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ - var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.left, │ │ │ │ │ - y: position.bottom │ │ │ │ │ - }); │ │ │ │ │ - var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.right, │ │ │ │ │ - y: position.top │ │ │ │ │ - }); │ │ │ │ │ - var bounds = new OpenLayers.Bounds(minXY.lon, minXY.lat, maxXY.lon, maxXY.lat); │ │ │ │ │ - if (!this.multipleSelect()) { │ │ │ │ │ - this.unselectAll() │ │ │ │ │ - } │ │ │ │ │ - var prevMultiple = this.multiple; │ │ │ │ │ - this.multiple = true; │ │ │ │ │ - var layers = this.layers || [this.layer]; │ │ │ │ │ - this.events.triggerEvent("boxselectionstart", { │ │ │ │ │ - layers: layers │ │ │ │ │ - }); │ │ │ │ │ - var layer; │ │ │ │ │ - for (var l = 0; l < layers.length; ++l) { │ │ │ │ │ - layer = layers[l]; │ │ │ │ │ - for (var i = 0, len = layer.features.length; i < len; ++i) { │ │ │ │ │ - var feature = layer.features[i]; │ │ │ │ │ - if (!feature.getVisibility()) { │ │ │ │ │ - continue │ │ │ │ │ - } │ │ │ │ │ - if (this.geometryTypes == null || OpenLayers.Util.indexOf(this.geometryTypes, feature.geometry.CLASS_NAME) > -1) { │ │ │ │ │ - if (bounds.toGeometry().intersects(feature.geometry)) { │ │ │ │ │ - if (OpenLayers.Util.indexOf(layer.selectedFeatures, feature) == -1) { │ │ │ │ │ - this.select(feature) │ │ │ │ │ + splits = splits[1]; │ │ │ │ │ + if (splits.length === 0) { │ │ │ │ │ + splits = [targetLine.clone()] │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.multiple = prevMultiple; │ │ │ │ │ - this.events.triggerEvent("boxselectionend", { │ │ │ │ │ - layers: layers │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.handlers.feature.setMap(map); │ │ │ │ │ - if (this.box) { │ │ │ │ │ - this.handlers.box.setMap(map) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - setLayer: function(layers) { │ │ │ │ │ - var isActive = this.active; │ │ │ │ │ - this.unselectAll(); │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - if (this.layers) { │ │ │ │ │ - this.layer.destroy(); │ │ │ │ │ - this.layers = null │ │ │ │ │ - } │ │ │ │ │ - this.initLayer(layers); │ │ │ │ │ - this.handlers.feature.layer = this.layer; │ │ │ │ │ - if (isActive) { │ │ │ │ │ - this.activate() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.SelectFeature" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.DragPan = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ - panned: false, │ │ │ │ │ - interval: 0, │ │ │ │ │ - documentDrag: false, │ │ │ │ │ - kinetic: null, │ │ │ │ │ - enableKinetic: true, │ │ │ │ │ - kineticInterval: 10, │ │ │ │ │ - draw: function() { │ │ │ │ │ - if (this.enableKinetic && OpenLayers.Kinetic) { │ │ │ │ │ - var config = { │ │ │ │ │ - interval: this.kineticInterval │ │ │ │ │ - }; │ │ │ │ │ - if (typeof this.enableKinetic === "object") { │ │ │ │ │ - config = OpenLayers.Util.extend(config, this.enableKinetic) │ │ │ │ │ - } │ │ │ │ │ - this.kinetic = new OpenLayers.Kinetic(config) │ │ │ │ │ - } │ │ │ │ │ - this.handler = new OpenLayers.Handler.Drag(this, { │ │ │ │ │ - move: this.panMap, │ │ │ │ │ - done: this.panMapDone, │ │ │ │ │ - down: this.panMapStart │ │ │ │ │ - }, { │ │ │ │ │ - interval: this.interval, │ │ │ │ │ - documentDrag: this.documentDrag │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - panMapStart: function() { │ │ │ │ │ - if (this.kinetic) { │ │ │ │ │ - this.kinetic.begin() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - panMap: function(xy) { │ │ │ │ │ - if (this.kinetic) { │ │ │ │ │ - this.kinetic.update(xy) │ │ │ │ │ - } │ │ │ │ │ - this.panned = true; │ │ │ │ │ - this.map.pan(this.handler.last.x - xy.x, this.handler.last.y - xy.y, { │ │ │ │ │ - dragging: true, │ │ │ │ │ - animate: false │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - panMapDone: function(xy) { │ │ │ │ │ - if (this.panned) { │ │ │ │ │ - var res = null; │ │ │ │ │ - if (this.kinetic) { │ │ │ │ │ - res = this.kinetic.end(xy) │ │ │ │ │ - } │ │ │ │ │ - this.map.pan(this.handler.last.x - xy.x, this.handler.last.y - xy.y, { │ │ │ │ │ - dragging: !!res, │ │ │ │ │ - animate: false │ │ │ │ │ - }); │ │ │ │ │ - if (res) { │ │ │ │ │ - var self = this; │ │ │ │ │ - this.kinetic.move(res, function(x, y, end) { │ │ │ │ │ - self.map.pan(x, y, { │ │ │ │ │ - dragging: !end, │ │ │ │ │ - animate: false │ │ │ │ │ - }) │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.panned = false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.DragPan" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.Pinch = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - started: false, │ │ │ │ │ - stopDown: false, │ │ │ │ │ - pinching: false, │ │ │ │ │ - last: null, │ │ │ │ │ - start: null, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - if (OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ - this.started = true; │ │ │ │ │ - this.last = this.start = { │ │ │ │ │ - distance: this.getDistance(evt.touches), │ │ │ │ │ - delta: 0, │ │ │ │ │ - scale: 1 │ │ │ │ │ - }; │ │ │ │ │ - this.callback("start", [evt, this.start]); │ │ │ │ │ - propagate = !this.stopDown │ │ │ │ │ - } else if (this.started) { │ │ │ │ │ - return false │ │ │ │ │ - } else { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ - return propagate │ │ │ │ │ - }, │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - if (this.started && OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ - this.pinching = true; │ │ │ │ │ - var current = this.getPinchData(evt); │ │ │ │ │ - this.callback("move", [evt, current]); │ │ │ │ │ - this.last = current; │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ - } else if (this.started) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - if (this.started && !OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - this.callback("done", [evt, this.start, this.last]); │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - activated = true │ │ │ │ │ - } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - deactivated = true │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - getDistance: function(touches) { │ │ │ │ │ - var t0 = touches[0]; │ │ │ │ │ - var t1 = touches[1]; │ │ │ │ │ - return Math.sqrt(Math.pow(t0.olClientX - t1.olClientX, 2) + Math.pow(t0.olClientY - t1.olClientY, 2)) │ │ │ │ │ - }, │ │ │ │ │ - getPinchData: function(evt) { │ │ │ │ │ - var distance = this.getDistance(evt.touches); │ │ │ │ │ - var scale = distance / this.start.distance; │ │ │ │ │ - return { │ │ │ │ │ - distance: distance, │ │ │ │ │ - delta: this.last.distance - distance, │ │ │ │ │ - scale: scale │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Pinch" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.PinchZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ - pinchOrigin: null, │ │ │ │ │ - currentCenter: null, │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - preserveCenter: false, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.handler = new OpenLayers.Handler.Pinch(this, { │ │ │ │ │ - start: this.pinchStart, │ │ │ │ │ - move: this.pinchMove, │ │ │ │ │ - done: this.pinchDone │ │ │ │ │ - }, this.handlerOptions) │ │ │ │ │ - }, │ │ │ │ │ - pinchStart: function(evt, pinchData) { │ │ │ │ │ - var xy = this.preserveCenter ? this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ - this.pinchOrigin = xy; │ │ │ │ │ - this.currentCenter = xy │ │ │ │ │ - }, │ │ │ │ │ - pinchMove: function(evt, pinchData) { │ │ │ │ │ - var scale = pinchData.scale; │ │ │ │ │ - var containerOrigin = this.map.layerContainerOriginPx; │ │ │ │ │ - var pinchOrigin = this.pinchOrigin; │ │ │ │ │ - var current = this.preserveCenter ? this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ - var dx = Math.round(containerOrigin.x + current.x - pinchOrigin.x + (scale - 1) * (containerOrigin.x - pinchOrigin.x)); │ │ │ │ │ - var dy = Math.round(containerOrigin.y + current.y - pinchOrigin.y + (scale - 1) * (containerOrigin.y - pinchOrigin.y)); │ │ │ │ │ - this.map.applyTransform(dx, dy, scale); │ │ │ │ │ - this.currentCenter = current │ │ │ │ │ - }, │ │ │ │ │ - pinchDone: function(evt, start, last) { │ │ │ │ │ - this.map.applyTransform(); │ │ │ │ │ - var zoom = this.map.getZoomForResolution(this.map.getResolution() / last.scale, true); │ │ │ │ │ - if (zoom !== this.map.getZoom() || !this.currentCenter.equals(this.pinchOrigin)) { │ │ │ │ │ - var resolution = this.map.getResolutionForZoom(zoom); │ │ │ │ │ - var location = this.map.getLonLatFromPixel(this.pinchOrigin); │ │ │ │ │ - var zoomPixel = this.currentCenter; │ │ │ │ │ - var size = this.map.getSize(); │ │ │ │ │ - location.lon += resolution * (size.w / 2 - zoomPixel.x); │ │ │ │ │ - location.lat -= resolution * (size.h / 2 - zoomPixel.y); │ │ │ │ │ - this.map.div.clientWidth = this.map.div.clientWidth; │ │ │ │ │ - this.map.setCenter(location, zoom) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.PinchZoom" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.Click = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - delay: 300, │ │ │ │ │ - single: true, │ │ │ │ │ - double: false, │ │ │ │ │ - pixelTolerance: 0, │ │ │ │ │ - dblclickTolerance: 13, │ │ │ │ │ - stopSingle: false, │ │ │ │ │ - stopDouble: false, │ │ │ │ │ - timerId: null, │ │ │ │ │ - down: null, │ │ │ │ │ - last: null, │ │ │ │ │ - first: null, │ │ │ │ │ - rightclickTimerId: null, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - this.down = this.getEventInfo(evt); │ │ │ │ │ - this.last = this.getEventInfo(evt); │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - this.last = this.getEventInfo(evt); │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - if (this.down) { │ │ │ │ │ - evt.xy = this.last.xy; │ │ │ │ │ - evt.lastTouches = this.last.touches; │ │ │ │ │ - this.handleSingle(evt); │ │ │ │ │ - this.down = null │ │ │ │ │ - } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - this.down = this.getEventInfo(evt); │ │ │ │ │ - this.last = this.getEventInfo(evt); │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - if (this.checkModifiers(evt) && this.control.handleRightClicks && OpenLayers.Event.isRightClick(evt)) { │ │ │ │ │ - propagate = this.rightclick(evt) │ │ │ │ │ - } │ │ │ │ │ - return propagate │ │ │ │ │ - }, │ │ │ │ │ - rightclick: function(evt) { │ │ │ │ │ - if (this.passesTolerance(evt)) { │ │ │ │ │ - if (this.rightclickTimerId != null) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.callback("dblrightclick", [evt]); │ │ │ │ │ - return !this.stopDouble │ │ │ │ │ - } else { │ │ │ │ │ - var clickEvent = this["double"] ? OpenLayers.Util.extend({}, evt) : this.callback("rightclick", [evt]); │ │ │ │ │ - var delayedRightCall = OpenLayers.Function.bind(this.delayedRightCall, this, clickEvent); │ │ │ │ │ - this.rightclickTimerId = window.setTimeout(delayedRightCall, this.delay) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return !this.stopSingle │ │ │ │ │ - }, │ │ │ │ │ - delayedRightCall: function(evt) { │ │ │ │ │ - this.rightclickTimerId = null; │ │ │ │ │ - if (evt) { │ │ │ │ │ - this.callback("rightclick", [evt]) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - if (!this.last) { │ │ │ │ │ - this.last = this.getEventInfo(evt) │ │ │ │ │ - } │ │ │ │ │ - this.handleSingle(evt); │ │ │ │ │ - return !this.stopSingle │ │ │ │ │ - }, │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - this.handleDouble(evt); │ │ │ │ │ - return !this.stopDouble │ │ │ │ │ - }, │ │ │ │ │ - handleDouble: function(evt) { │ │ │ │ │ - if (this.passesDblclickTolerance(evt)) { │ │ │ │ │ - if (this["double"]) { │ │ │ │ │ - this.callback("dblclick", [evt]) │ │ │ │ │ - } │ │ │ │ │ - this.clearTimer() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - handleSingle: function(evt) { │ │ │ │ │ - if (this.passesTolerance(evt)) { │ │ │ │ │ - if (this.timerId != null) { │ │ │ │ │ - if (this.last.touches && this.last.touches.length === 1) { │ │ │ │ │ - if (this["double"]) { │ │ │ │ │ - OpenLayers.Event.preventDefault(evt) │ │ │ │ │ - } │ │ │ │ │ - this.handleDouble(evt) │ │ │ │ │ - } │ │ │ │ │ - if (!this.last.touches || this.last.touches.length !== 2) { │ │ │ │ │ - this.clearTimer() │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.first = this.getEventInfo(evt); │ │ │ │ │ - var clickEvent = this.single ? OpenLayers.Util.extend({}, evt) : null; │ │ │ │ │ - this.queuePotentialClick(clickEvent) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - queuePotentialClick: function(evt) { │ │ │ │ │ - this.timerId = window.setTimeout(OpenLayers.Function.bind(this.delayedCall, this, evt), this.delay) │ │ │ │ │ - }, │ │ │ │ │ - passesTolerance: function(evt) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - if (this.pixelTolerance != null && this.down && this.down.xy) { │ │ │ │ │ - passes = this.pixelTolerance >= this.down.xy.distanceTo(evt.xy); │ │ │ │ │ - if (passes && this.touch && this.down.touches.length === this.last.touches.length) { │ │ │ │ │ - for (var i = 0, ii = this.down.touches.length; i < ii; ++i) { │ │ │ │ │ - if (this.getTouchDistance(this.down.touches[i], this.last.touches[i]) > this.pixelTolerance) { │ │ │ │ │ - passes = false; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return passes │ │ │ │ │ - }, │ │ │ │ │ - getTouchDistance: function(from, to) { │ │ │ │ │ - return Math.sqrt(Math.pow(from.clientX - to.clientX, 2) + Math.pow(from.clientY - to.clientY, 2)) │ │ │ │ │ - }, │ │ │ │ │ - passesDblclickTolerance: function(evt) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - if (this.down && this.first) { │ │ │ │ │ - passes = this.down.xy.distanceTo(this.first.xy) <= this.dblclickTolerance │ │ │ │ │ - } │ │ │ │ │ - return passes │ │ │ │ │ - }, │ │ │ │ │ - clearTimer: function() { │ │ │ │ │ - if (this.timerId != null) { │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null │ │ │ │ │ - } │ │ │ │ │ - if (this.rightclickTimerId != null) { │ │ │ │ │ - window.clearTimeout(this.rightclickTimerId); │ │ │ │ │ - this.rightclickTimerId = null │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - delayedCall: function(evt) { │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - if (evt) { │ │ │ │ │ - this.callback("click", [evt]) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getEventInfo: function(evt) { │ │ │ │ │ - var touches; │ │ │ │ │ - if (evt.touches) { │ │ │ │ │ - var len = evt.touches.length; │ │ │ │ │ - touches = new Array(len); │ │ │ │ │ - var touch; │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - touch = evt.touches[i]; │ │ │ │ │ - touches[i] = { │ │ │ │ │ - clientX: touch.olClientX, │ │ │ │ │ - clientY: touch.olClientY │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - xy: evt.xy, │ │ │ │ │ - touches: touches │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.down = null; │ │ │ │ │ - this.first = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - deactivated = true │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Click" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.TouchNavigation = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - dragPan: null, │ │ │ │ │ - dragPanOptions: null, │ │ │ │ │ - pinchZoom: null, │ │ │ │ │ - pinchZoomOptions: null, │ │ │ │ │ - clickHandlerOptions: null, │ │ │ │ │ - documentDrag: false, │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.handlers = {}; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - if (this.dragPan) { │ │ │ │ │ - this.dragPan.destroy() │ │ │ │ │ - } │ │ │ │ │ - this.dragPan = null; │ │ │ │ │ - if (this.pinchZoom) { │ │ │ │ │ - this.pinchZoom.destroy(); │ │ │ │ │ - delete this.pinchZoom │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.dragPan.activate(); │ │ │ │ │ - this.handlers.click.activate(); │ │ │ │ │ - this.pinchZoom.activate(); │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.dragPan.deactivate(); │ │ │ │ │ - this.handlers.click.deactivate(); │ │ │ │ │ - this.pinchZoom.deactivate(); │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - var clickCallbacks = { │ │ │ │ │ - click: this.defaultClick, │ │ │ │ │ - dblclick: this.defaultDblClick │ │ │ │ │ - }; │ │ │ │ │ - var clickOptions = OpenLayers.Util.extend({ │ │ │ │ │ - double: true, │ │ │ │ │ - stopDouble: true, │ │ │ │ │ - pixelTolerance: 2 │ │ │ │ │ - }, this.clickHandlerOptions); │ │ │ │ │ - this.handlers.click = new OpenLayers.Handler.Click(this, clickCallbacks, clickOptions); │ │ │ │ │ - this.dragPan = new OpenLayers.Control.DragPan(OpenLayers.Util.extend({ │ │ │ │ │ - map: this.map, │ │ │ │ │ - documentDrag: this.documentDrag │ │ │ │ │ - }, this.dragPanOptions)); │ │ │ │ │ - this.dragPan.draw(); │ │ │ │ │ - this.pinchZoom = new OpenLayers.Control.PinchZoom(OpenLayers.Util.extend({ │ │ │ │ │ - map: this.map │ │ │ │ │ - }, this.pinchZoomOptions)) │ │ │ │ │ - }, │ │ │ │ │ - defaultClick: function(evt) { │ │ │ │ │ - if (evt.lastTouches && evt.lastTouches.length == 2) { │ │ │ │ │ - this.map.zoomOut() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - defaultDblClick: function(evt) { │ │ │ │ │ - this.map.zoomTo(this.map.zoom + 1, evt.xy) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.TouchNavigation" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Renderer.Canvas = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ - hitDetection: true, │ │ │ │ │ - hitOverflow: 0, │ │ │ │ │ - canvas: null, │ │ │ │ │ - features: null, │ │ │ │ │ - pendingRedraw: false, │ │ │ │ │ - cachedSymbolBounds: {}, │ │ │ │ │ - initialize: function(containerID, options) { │ │ │ │ │ - OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.root = document.createElement("canvas"); │ │ │ │ │ - this.container.appendChild(this.root); │ │ │ │ │ - this.canvas = this.root.getContext("2d"); │ │ │ │ │ - this.features = {}; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitCanvas = document.createElement("canvas"); │ │ │ │ │ - this.hitContext = this.hitCanvas.getContext("2d") │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setExtent: function() { │ │ │ │ │ - OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - eraseGeometry: function(geometry, featureId) { │ │ │ │ │ - this.eraseFeatures(this.features[featureId][0]) │ │ │ │ │ - }, │ │ │ │ │ - supported: function() { │ │ │ │ │ - return OpenLayers.CANVAS_SUPPORTED │ │ │ │ │ - }, │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - this.size = size.clone(); │ │ │ │ │ - var root = this.root; │ │ │ │ │ - root.style.width = size.w + "px"; │ │ │ │ │ - root.style.height = size.h + "px"; │ │ │ │ │ - root.width = size.w; │ │ │ │ │ - root.height = size.h; │ │ │ │ │ - this.resolution = null; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - var hitCanvas = this.hitCanvas; │ │ │ │ │ - hitCanvas.style.width = size.w + "px"; │ │ │ │ │ - hitCanvas.style.height = size.h + "px"; │ │ │ │ │ - hitCanvas.width = size.w; │ │ │ │ │ - hitCanvas.height = size.h │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - drawFeature: function(feature, style) { │ │ │ │ │ - var rendered; │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - style = this.applyDefaultSymbolizer(style || feature.style); │ │ │ │ │ - var bounds = feature.geometry.getBounds(); │ │ │ │ │ - var worldBounds; │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - worldBounds = this.map.getMaxExtent() │ │ │ │ │ - } │ │ │ │ │ - var intersects = bounds && bounds.intersectsBounds(this.extent, { │ │ │ │ │ - worldBounds: worldBounds │ │ │ │ │ - }); │ │ │ │ │ - rendered = style.display !== "none" && !!bounds && intersects; │ │ │ │ │ - if (rendered) { │ │ │ │ │ - this.features[feature.id] = [feature, style] │ │ │ │ │ - } else { │ │ │ │ │ - delete this.features[feature.id] │ │ │ │ │ - } │ │ │ │ │ - this.pendingRedraw = true │ │ │ │ │ - } │ │ │ │ │ - if (this.pendingRedraw && !this.locked) { │ │ │ │ │ - this.redraw(); │ │ │ │ │ - this.pendingRedraw = false │ │ │ │ │ - } │ │ │ │ │ - return rendered │ │ │ │ │ - }, │ │ │ │ │ - drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ - var className = geometry.CLASS_NAME; │ │ │ │ │ - if (className == "OpenLayers.Geometry.Collection" || className == "OpenLayers.Geometry.MultiPoint" || className == "OpenLayers.Geometry.MultiLineString" || className == "OpenLayers.Geometry.MultiPolygon") { │ │ │ │ │ - for (var i = 0; i < geometry.components.length; i++) { │ │ │ │ │ - this.drawGeometry(geometry.components[i], style, featureId) │ │ │ │ │ - } │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - this.drawPoint(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - this.drawLineString(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - this.drawPolygon(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - drawExternalGraphic: function(geometry, style, featureId) { │ │ │ │ │ - var img = new Image; │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - img.title = title │ │ │ │ │ - } │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ - var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ - var onLoad = function() { │ │ │ │ │ - if (!this.features[featureId]) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ - if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ - var x = p0 + xOffset | 0; │ │ │ │ │ - var y = p1 + yOffset | 0; │ │ │ │ │ - var canvas = this.canvas; │ │ │ │ │ - canvas.globalAlpha = opacity; │ │ │ │ │ - var factor = OpenLayers.Renderer.Canvas.drawImageScaleFactor || (OpenLayers.Renderer.Canvas.drawImageScaleFactor = /android 2.1/.test(navigator.userAgent.toLowerCase()) ? 320 / window.screen.width : 1); │ │ │ │ │ - canvas.drawImage(img, x * factor, y * factor, width * factor, height * factor); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId); │ │ │ │ │ - this.hitContext.fillRect(x, y, width, height) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - img.onload = OpenLayers.Function.bind(onLoad, this); │ │ │ │ │ - img.src = style.externalGraphic │ │ │ │ │ - }, │ │ │ │ │ - drawNamedSymbol: function(geometry, style, featureId) { │ │ │ │ │ - var x, y, cx, cy, i, symbolBounds, scaling, angle; │ │ │ │ │ - var unscaledStrokeWidth; │ │ │ │ │ - var deg2rad = Math.PI / 180; │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[style.graphicName]; │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(style.graphicName + " is not a valid symbol name") │ │ │ │ │ - } │ │ │ │ │ - if (!symbol.length || symbol.length < 2) return; │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ - if (isNaN(p0) || isNaN(p1)) return; │ │ │ │ │ - this.canvas.lineCap = "round"; │ │ │ │ │ - this.canvas.lineJoin = "round"; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.lineCap = "round"; │ │ │ │ │ - this.hitContext.lineJoin = "round" │ │ │ │ │ - } │ │ │ │ │ - if (style.graphicName in this.cachedSymbolBounds) { │ │ │ │ │ - symbolBounds = this.cachedSymbolBounds[style.graphicName] │ │ │ │ │ - } else { │ │ │ │ │ - symbolBounds = new OpenLayers.Bounds; │ │ │ │ │ - for (i = 0; i < symbol.length; i += 2) { │ │ │ │ │ - symbolBounds.extend(new OpenLayers.LonLat(symbol[i], symbol[i + 1])) │ │ │ │ │ - } │ │ │ │ │ - this.cachedSymbolBounds[style.graphicName] = symbolBounds │ │ │ │ │ - } │ │ │ │ │ - this.canvas.save(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.save() │ │ │ │ │ - } │ │ │ │ │ - this.canvas.translate(p0, p1); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.translate(p0, p1) │ │ │ │ │ - } │ │ │ │ │ - angle = deg2rad * style.rotation; │ │ │ │ │ - if (!isNaN(angle)) { │ │ │ │ │ - this.canvas.rotate(angle); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.rotate(angle) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - scaling = 2 * style.pointRadius / Math.max(symbolBounds.getWidth(), symbolBounds.getHeight()); │ │ │ │ │ - this.canvas.scale(scaling, scaling); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.scale(scaling, scaling) │ │ │ │ │ - } │ │ │ │ │ - cx = symbolBounds.getCenterLonLat().lon; │ │ │ │ │ - cy = symbolBounds.getCenterLonLat().lat; │ │ │ │ │ - this.canvas.translate(-cx, -cy); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.translate(-cx, -cy) │ │ │ │ │ - } │ │ │ │ │ - unscaledStrokeWidth = style.strokeWidth; │ │ │ │ │ - style.strokeWidth = unscaledStrokeWidth / scaling; │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.canvas.lineTo(x, y) │ │ │ │ │ - } │ │ │ │ │ - this.canvas.closePath(); │ │ │ │ │ - this.canvas.fill(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.hitContext.lineTo(x, y) │ │ │ │ │ - } │ │ │ │ │ - this.hitContext.closePath(); │ │ │ │ │ - this.hitContext.fill() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.canvas.lineTo(x, y) │ │ │ │ │ - } │ │ │ │ │ - this.canvas.closePath(); │ │ │ │ │ - this.canvas.stroke(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style, scaling); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.hitContext.moveTo(x, y); │ │ │ │ │ - this.hitContext.lineTo(x, y) │ │ │ │ │ - } │ │ │ │ │ - this.hitContext.closePath(); │ │ │ │ │ - this.hitContext.stroke() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - style.strokeWidth = unscaledStrokeWidth; │ │ │ │ │ - this.canvas.restore(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.restore() │ │ │ │ │ - } │ │ │ │ │ - this.setCanvasStyle("reset") │ │ │ │ │ - }, │ │ │ │ │ - setCanvasStyle: function(type, style) { │ │ │ │ │ - if (type === "fill") { │ │ │ │ │ - this.canvas.globalAlpha = style["fillOpacity"]; │ │ │ │ │ - this.canvas.fillStyle = style["fillColor"] │ │ │ │ │ - } else if (type === "stroke") { │ │ │ │ │ - this.canvas.globalAlpha = style["strokeOpacity"]; │ │ │ │ │ - this.canvas.strokeStyle = style["strokeColor"]; │ │ │ │ │ - this.canvas.lineWidth = style["strokeWidth"] │ │ │ │ │ - } else { │ │ │ │ │ - this.canvas.globalAlpha = 0; │ │ │ │ │ - this.canvas.lineWidth = 1 │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - featureIdToHex: function(featureId) { │ │ │ │ │ - var id = Number(featureId.split("_").pop()) + 1; │ │ │ │ │ - if (id >= 16777216) { │ │ │ │ │ - this.hitOverflow = id - 16777215; │ │ │ │ │ - id = id % 16777216 + 1 │ │ │ │ │ - } │ │ │ │ │ - var hex = "000000" + id.toString(16); │ │ │ │ │ - var len = hex.length; │ │ │ │ │ - hex = "#" + hex.substring(len - 6, len); │ │ │ │ │ - return hex │ │ │ │ │ - }, │ │ │ │ │ - setHitContextStyle: function(type, featureId, symbolizer, strokeScaling) { │ │ │ │ │ - var hex = this.featureIdToHex(featureId); │ │ │ │ │ - if (type == "fill") { │ │ │ │ │ - this.hitContext.globalAlpha = 1; │ │ │ │ │ - this.hitContext.fillStyle = hex │ │ │ │ │ - } else if (type == "stroke") { │ │ │ │ │ - this.hitContext.globalAlpha = 1; │ │ │ │ │ - this.hitContext.strokeStyle = hex; │ │ │ │ │ - if (typeof strokeScaling === "undefined") { │ │ │ │ │ - this.hitContext.lineWidth = symbolizer.strokeWidth + 2 │ │ │ │ │ - } else { │ │ │ │ │ - if (!isNaN(strokeScaling)) { │ │ │ │ │ - this.hitContext.lineWidth = symbolizer.strokeWidth + 2 / strokeScaling │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.hitContext.globalAlpha = 0; │ │ │ │ │ - this.hitContext.lineWidth = 1 │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - drawPoint: function(geometry, style, featureId) { │ │ │ │ │ - if (style.graphic !== false) { │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - this.drawExternalGraphic(geometry, style, featureId) │ │ │ │ │ - } else if (style.graphicName && style.graphicName != "circle") { │ │ │ │ │ - this.drawNamedSymbol(geometry, style, featureId) │ │ │ │ │ - } else { │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ - if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ - var twoPi = Math.PI * 2; │ │ │ │ │ - var radius = style.pointRadius; │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.canvas.fill(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.hitContext.fill() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.canvas.stroke(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.hitContext.stroke() │ │ │ │ │ + for (var k = 0, klen = splits.length; k < klen; ++k) { │ │ │ │ │ + if (k === 0 && targetParts.length) { │ │ │ │ │ + targetParts[targetParts.length - 1].addComponent(splits[k]) │ │ │ │ │ + } else { │ │ │ │ │ + targetParts.push(new OpenLayers.Geometry.MultiLineString([splits[k]])) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.setCanvasStyle("reset") │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - drawLineString: function(geometry, style, featureId) { │ │ │ │ │ - style = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - fill: false │ │ │ │ │ - }, style); │ │ │ │ │ - this.drawLinearRing(geometry, style, featureId) │ │ │ │ │ - }, │ │ │ │ │ - drawLinearRing: function(geometry, style, featureId) { │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.renderPath(this.canvas, geometry, style, featureId, "fill"); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.renderPath(this.hitContext, geometry, style, featureId, "fill") │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.renderPath(this.canvas, geometry, style, featureId, "stroke"); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ - this.renderPath(this.hitContext, geometry, style, featureId, "stroke") │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.setCanvasStyle("reset") │ │ │ │ │ - }, │ │ │ │ │ - renderPath: function(context, geometry, style, featureId, type) { │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - context.beginPath(); │ │ │ │ │ - var start = this.getLocalXY(components[0]); │ │ │ │ │ - var x = start[0]; │ │ │ │ │ - var y = start[1]; │ │ │ │ │ - if (!isNaN(x) && !isNaN(y)) { │ │ │ │ │ - context.moveTo(start[0], start[1]); │ │ │ │ │ - for (var i = 1; i < len; ++i) { │ │ │ │ │ - var pt = this.getLocalXY(components[i]); │ │ │ │ │ - context.lineTo(pt[0], pt[1]) │ │ │ │ │ - } │ │ │ │ │ - if (type === "fill") { │ │ │ │ │ - context.fill() │ │ │ │ │ - } else { │ │ │ │ │ - context.stroke() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - drawPolygon: function(geometry, style, featureId) { │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - this.drawLinearRing(components[0], style, featureId); │ │ │ │ │ - for (var i = 1; i < len; ++i) { │ │ │ │ │ - this.canvas.globalCompositeOperation = "destination-out"; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.globalCompositeOperation = "destination-out" │ │ │ │ │ - } │ │ │ │ │ - this.drawLinearRing(components[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - stroke: false, │ │ │ │ │ - fillOpacity: 1 │ │ │ │ │ - }, style), featureId); │ │ │ │ │ - this.canvas.globalCompositeOperation = "source-over"; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.globalCompositeOperation = "source-over" │ │ │ │ │ - } │ │ │ │ │ - this.drawLinearRing(components[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - fill: false │ │ │ │ │ - }, style), featureId) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - drawText: function(location, style) { │ │ │ │ │ - var pt = this.getLocalXY(location); │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ - this.canvas.fillStyle = style.fontColor; │ │ │ │ │ - this.canvas.globalAlpha = style.fontOpacity || 1; │ │ │ │ │ - var fontStyle = [style.fontStyle ? style.fontStyle : "normal", "normal", style.fontWeight ? style.fontWeight : "normal", style.fontSize ? style.fontSize : "1em", style.fontFamily ? style.fontFamily : "sans-serif"].join(" "); │ │ │ │ │ - var labelRows = style.label.split("\n"); │ │ │ │ │ - var numRows = labelRows.length; │ │ │ │ │ - if (this.canvas.fillText) { │ │ │ │ │ - this.canvas.font = fontStyle; │ │ │ │ │ - this.canvas.textAlign = OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[0]] || "center"; │ │ │ │ │ - this.canvas.textBaseline = OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[1]] || "middle"; │ │ │ │ │ - var vfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5 │ │ │ │ │ - } │ │ │ │ │ - var lineHeight = this.canvas.measureText("Mg").height || this.canvas.measureText("xx").width; │ │ │ │ │ - pt[1] += lineHeight * vfactor * (numRows - 1); │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - if (style.labelOutlineWidth) { │ │ │ │ │ - this.canvas.save(); │ │ │ │ │ - this.canvas.globalAlpha = style.labelOutlineOpacity || style.fontOpacity || 1; │ │ │ │ │ - this.canvas.strokeStyle = style.labelOutlineColor; │ │ │ │ │ - this.canvas.lineWidth = style.labelOutlineWidth; │ │ │ │ │ - this.canvas.strokeText(labelRows[i], pt[0], pt[1] + lineHeight * i + 1); │ │ │ │ │ - this.canvas.restore() │ │ │ │ │ - } │ │ │ │ │ - this.canvas.fillText(labelRows[i], pt[0], pt[1] + lineHeight * i) │ │ │ │ │ - } │ │ │ │ │ - } else if (this.canvas.mozDrawText) { │ │ │ │ │ - this.canvas.mozTextStyle = fontStyle; │ │ │ │ │ - var hfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[0]]; │ │ │ │ │ - if (hfactor == null) { │ │ │ │ │ - hfactor = -.5 │ │ │ │ │ - } │ │ │ │ │ - var vfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5 │ │ │ │ │ - } │ │ │ │ │ - var lineHeight = this.canvas.mozMeasureText("xx"); │ │ │ │ │ - pt[1] += lineHeight * (1 + vfactor * numRows); │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - var x = pt[0] + hfactor * this.canvas.mozMeasureText(labelRows[i]); │ │ │ │ │ - var y = pt[1] + i * lineHeight; │ │ │ │ │ - this.canvas.translate(x, y); │ │ │ │ │ - this.canvas.mozDrawText(labelRows[i]); │ │ │ │ │ - this.canvas.translate(-x, -y) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.setCanvasStyle("reset") │ │ │ │ │ - }, │ │ │ │ │ - getLocalXY: function(point) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var extent = this.extent; │ │ │ │ │ - var x = (point.x - this.featureDx) / resolution + -extent.left / resolution; │ │ │ │ │ - var y = extent.top / resolution - point.y / resolution; │ │ │ │ │ - return [x, y] │ │ │ │ │ - }, │ │ │ │ │ - clear: function() { │ │ │ │ │ - var height = this.root.height; │ │ │ │ │ - var width = this.root.width; │ │ │ │ │ - this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ - this.features = {}; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.clearRect(0, 0, width, height) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var featureId, feature; │ │ │ │ │ - if (this.hitDetection && this.root.style.display !== "none") { │ │ │ │ │ - if (!this.map.dragging) { │ │ │ │ │ - var xy = evt.xy; │ │ │ │ │ - var x = xy.x | 0; │ │ │ │ │ - var y = xy.y | 0; │ │ │ │ │ - var data = this.hitContext.getImageData(x, y, 1, 1).data; │ │ │ │ │ - if (data[3] === 255) { │ │ │ │ │ - var id = data[2] + 256 * (data[1] + 256 * data[0]); │ │ │ │ │ - if (id) { │ │ │ │ │ - featureId = "OpenLayers_Feature_Vector_" + (id - 1 + this.hitOverflow); │ │ │ │ │ - try { │ │ │ │ │ - feature = this.features[featureId][0] │ │ │ │ │ - } catch (err) {} │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return feature │ │ │ │ │ - }, │ │ │ │ │ - eraseFeatures: function(features) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ - features = [features] │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0; i < features.length; ++i) { │ │ │ │ │ - delete this.features[features[i].id] │ │ │ │ │ - } │ │ │ │ │ - this.redraw() │ │ │ │ │ - }, │ │ │ │ │ - redraw: function() { │ │ │ │ │ - if (!this.locked) { │ │ │ │ │ - var height = this.root.height; │ │ │ │ │ - var width = this.root.width; │ │ │ │ │ - this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.clearRect(0, 0, width, height) │ │ │ │ │ - } │ │ │ │ │ - var labelMap = []; │ │ │ │ │ - var feature, geometry, style; │ │ │ │ │ - var worldBounds = this.map.baseLayer && this.map.baseLayer.wrapDateLine && this.map.getMaxExtent(); │ │ │ │ │ - for (var id in this.features) { │ │ │ │ │ - if (!this.features.hasOwnProperty(id)) { │ │ │ │ │ - continue │ │ │ │ │ - } │ │ │ │ │ - feature = this.features[id][0]; │ │ │ │ │ - geometry = feature.geometry; │ │ │ │ │ - this.calculateFeatureDx(geometry.getBounds(), worldBounds); │ │ │ │ │ - style = this.features[id][1]; │ │ │ │ │ - this.drawGeometry(geometry, style, feature.id); │ │ │ │ │ - if (style.label) { │ │ │ │ │ - labelMap.push([feature, style]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var item; │ │ │ │ │ - for (var i = 0, len = labelMap.length; i < len; ++i) { │ │ │ │ │ - item = labelMap[i]; │ │ │ │ │ - this.drawText(item[0].geometry.getCentroid(), item[1]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.Canvas" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Renderer.Canvas.LABEL_ALIGN = { │ │ │ │ │ - l: "left", │ │ │ │ │ - r: "right", │ │ │ │ │ - t: "top", │ │ │ │ │ - b: "bottom" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.Canvas.LABEL_FACTOR = { │ │ │ │ │ - l: 0, │ │ │ │ │ - r: -1, │ │ │ │ │ - t: 0, │ │ │ │ │ - b: -1 │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.Canvas.drawImageScaleFactor = null; │ │ │ │ │ -OpenLayers.ElementsIndexer = OpenLayers.Class({ │ │ │ │ │ - maxZIndex: null, │ │ │ │ │ - order: null, │ │ │ │ │ - indices: null, │ │ │ │ │ - compare: null, │ │ │ │ │ - initialize: function(yOrdering) { │ │ │ │ │ - this.compare = yOrdering ? OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_Y_ORDER : OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER_DRAWING_ORDER; │ │ │ │ │ - this.clear() │ │ │ │ │ - }, │ │ │ │ │ - insert: function(newNode) { │ │ │ │ │ - if (this.exists(newNode)) { │ │ │ │ │ - this.remove(newNode) │ │ │ │ │ - } │ │ │ │ │ - var nodeId = newNode.id; │ │ │ │ │ - this.determineZIndex(newNode); │ │ │ │ │ - var leftIndex = -1; │ │ │ │ │ - var rightIndex = this.order.length; │ │ │ │ │ - var middle; │ │ │ │ │ - while (rightIndex - leftIndex > 1) { │ │ │ │ │ - middle = parseInt((leftIndex + rightIndex) / 2); │ │ │ │ │ - var placement = this.compare(this, newNode, OpenLayers.Util.getElement(this.order[middle])); │ │ │ │ │ - if (placement > 0) { │ │ │ │ │ - leftIndex = middle │ │ │ │ │ - } else { │ │ │ │ │ - rightIndex = middle │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.order.splice(rightIndex, 0, nodeId); │ │ │ │ │ - this.indices[nodeId] = this.getZIndex(newNode); │ │ │ │ │ - return this.getNextElement(rightIndex) │ │ │ │ │ - }, │ │ │ │ │ - remove: function(node) { │ │ │ │ │ - var nodeId = node.id; │ │ │ │ │ - var arrayIndex = OpenLayers.Util.indexOf(this.order, nodeId); │ │ │ │ │ - if (arrayIndex >= 0) { │ │ │ │ │ - this.order.splice(arrayIndex, 1); │ │ │ │ │ - delete this.indices[nodeId]; │ │ │ │ │ - if (this.order.length > 0) { │ │ │ │ │ - var lastId = this.order[this.order.length - 1]; │ │ │ │ │ - this.maxZIndex = this.indices[lastId] │ │ │ │ │ - } else { │ │ │ │ │ - this.maxZIndex = 0 │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - clear: function() { │ │ │ │ │ - this.order = []; │ │ │ │ │ - this.indices = {}; │ │ │ │ │ - this.maxZIndex = 0 │ │ │ │ │ - }, │ │ │ │ │ - exists: function(node) { │ │ │ │ │ - return this.indices[node.id] != null │ │ │ │ │ - }, │ │ │ │ │ - getZIndex: function(node) { │ │ │ │ │ - return node._style.graphicZIndex │ │ │ │ │ - }, │ │ │ │ │ - determineZIndex: function(node) { │ │ │ │ │ - var zIndex = node._style.graphicZIndex; │ │ │ │ │ - if (zIndex == null) { │ │ │ │ │ - zIndex = this.maxZIndex; │ │ │ │ │ - node._style.graphicZIndex = zIndex │ │ │ │ │ - } else if (zIndex > this.maxZIndex) { │ │ │ │ │ - this.maxZIndex = zIndex │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getNextElement: function(index) { │ │ │ │ │ - var nextIndex = index + 1; │ │ │ │ │ - if (nextIndex < this.order.length) { │ │ │ │ │ - var nextElement = OpenLayers.Util.getElement(this.order[nextIndex]); │ │ │ │ │ - if (nextElement == undefined) { │ │ │ │ │ - nextElement = this.getNextElement(nextIndex) │ │ │ │ │ - } │ │ │ │ │ - return nextElement │ │ │ │ │ - } else { │ │ │ │ │ - return null │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.ElementsIndexer" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.ElementsIndexer.IndexingMethods = { │ │ │ │ │ - Z_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var newZIndex = indexer.getZIndex(newNode); │ │ │ │ │ - var returnVal = 0; │ │ │ │ │ - if (nextNode) { │ │ │ │ │ - var nextZIndex = indexer.getZIndex(nextNode); │ │ │ │ │ - returnVal = newZIndex - nextZIndex │ │ │ │ │ - } │ │ │ │ │ - return returnVal │ │ │ │ │ - }, │ │ │ │ │ - Z_ORDER_DRAWING_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER(indexer, newNode, nextNode); │ │ │ │ │ - if (nextNode && returnVal == 0) { │ │ │ │ │ - returnVal = 1 │ │ │ │ │ - } │ │ │ │ │ - return returnVal │ │ │ │ │ - }, │ │ │ │ │ - Z_ORDER_Y_ORDER: function(indexer, newNode, nextNode) { │ │ │ │ │ - var returnVal = OpenLayers.ElementsIndexer.IndexingMethods.Z_ORDER(indexer, newNode, nextNode); │ │ │ │ │ - if (nextNode && returnVal === 0) { │ │ │ │ │ - var result = nextNode._boundsBottom - newNode._boundsBottom; │ │ │ │ │ - returnVal = result === 0 ? 1 : result │ │ │ │ │ - } │ │ │ │ │ - return returnVal │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.Elements = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ - rendererRoot: null, │ │ │ │ │ - root: null, │ │ │ │ │ - vectorRoot: null, │ │ │ │ │ - textRoot: null, │ │ │ │ │ - xmlns: null, │ │ │ │ │ - xOffset: 0, │ │ │ │ │ - indexer: null, │ │ │ │ │ - BACKGROUND_ID_SUFFIX: "_background", │ │ │ │ │ - LABEL_ID_SUFFIX: "_label", │ │ │ │ │ - LABEL_OUTLINE_SUFFIX: "_outline", │ │ │ │ │ - initialize: function(containerID, options) { │ │ │ │ │ - OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.rendererRoot = this.createRenderRoot(); │ │ │ │ │ - this.root = this.createRoot("_root"); │ │ │ │ │ - this.vectorRoot = this.createRoot("_vroot"); │ │ │ │ │ - this.textRoot = this.createRoot("_troot"); │ │ │ │ │ - this.root.appendChild(this.vectorRoot); │ │ │ │ │ - this.root.appendChild(this.textRoot); │ │ │ │ │ - this.rendererRoot.appendChild(this.root); │ │ │ │ │ - this.container.appendChild(this.rendererRoot); │ │ │ │ │ - if (options && (options.zIndexing || options.yOrdering)) { │ │ │ │ │ - this.indexer = new OpenLayers.ElementsIndexer(options.yOrdering) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.clear(); │ │ │ │ │ - this.rendererRoot = null; │ │ │ │ │ - this.root = null; │ │ │ │ │ - this.xmlns = null; │ │ │ │ │ - OpenLayers.Renderer.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - clear: function() { │ │ │ │ │ - var child; │ │ │ │ │ - var root = this.vectorRoot; │ │ │ │ │ - if (root) { │ │ │ │ │ - while (child = root.firstChild) { │ │ │ │ │ - root.removeChild(child) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - root = this.textRoot; │ │ │ │ │ - if (root) { │ │ │ │ │ - while (child = root.firstChild) { │ │ │ │ │ - root.removeChild(child) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - this.indexer.clear() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - var rightOfDateLine, ratio = extent.getWidth() / this.map.getExtent().getWidth(), │ │ │ │ │ - extent = extent.scale(1 / ratio), │ │ │ │ │ - world = this.map.getMaxExtent(); │ │ │ │ │ - if (world.right > extent.left && world.right < extent.right) { │ │ │ │ │ - rightOfDateLine = true │ │ │ │ │ - } else if (world.left > extent.left && world.left < extent.right) { │ │ │ │ │ - rightOfDateLine = false │ │ │ │ │ - } │ │ │ │ │ - if (rightOfDateLine !== this.rightOfDateLine || resolutionChanged) { │ │ │ │ │ - coordSysUnchanged = false; │ │ │ │ │ - this.xOffset = rightOfDateLine === true ? world.getWidth() / resolution : 0 │ │ │ │ │ - } │ │ │ │ │ - this.rightOfDateLine = rightOfDateLine │ │ │ │ │ - } │ │ │ │ │ - return coordSysUnchanged │ │ │ │ │ - }, │ │ │ │ │ - getNodeType: function(geometry, style) {}, │ │ │ │ │ - drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ - var className = geometry.CLASS_NAME; │ │ │ │ │ - var rendered = true; │ │ │ │ │ - if (className == "OpenLayers.Geometry.Collection" || className == "OpenLayers.Geometry.MultiPoint" || className == "OpenLayers.Geometry.MultiLineString" || className == "OpenLayers.Geometry.MultiPolygon") { │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - rendered = this.drawGeometry(geometry.components[i], style, featureId) && rendered │ │ │ │ │ - } │ │ │ │ │ - return rendered │ │ │ │ │ - } │ │ │ │ │ - rendered = false; │ │ │ │ │ - var removeBackground = false; │ │ │ │ │ - if (style.display != "none") { │ │ │ │ │ - if (style.backgroundGraphic) { │ │ │ │ │ - this.redrawBackgroundNode(geometry.id, geometry, style, featureId) │ │ │ │ │ - } else { │ │ │ │ │ - removeBackground = true │ │ │ │ │ - } │ │ │ │ │ - rendered = this.redrawNode(geometry.id, geometry, style, featureId) │ │ │ │ │ - } │ │ │ │ │ - if (rendered == false) { │ │ │ │ │ - var node = document.getElementById(geometry.id); │ │ │ │ │ - if (node) { │ │ │ │ │ - if (node._style.backgroundGraphic) { │ │ │ │ │ - removeBackground = true │ │ │ │ │ - } │ │ │ │ │ - node.parentNode.removeChild(node) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (removeBackground) { │ │ │ │ │ - var node = document.getElementById(geometry.id + this.BACKGROUND_ID_SUFFIX); │ │ │ │ │ - if (node) { │ │ │ │ │ - node.parentNode.removeChild(node) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return rendered │ │ │ │ │ - }, │ │ │ │ │ - redrawNode: function(id, geometry, style, featureId) { │ │ │ │ │ - style = this.applyDefaultSymbolizer(style); │ │ │ │ │ - var node = this.nodeFactory(id, this.getNodeType(geometry, style)); │ │ │ │ │ - node._featureId = featureId; │ │ │ │ │ - node._boundsBottom = geometry.getBounds().bottom; │ │ │ │ │ - node._geometryClass = geometry.CLASS_NAME; │ │ │ │ │ - node._style = style; │ │ │ │ │ - var drawResult = this.drawGeometryNode(node, geometry, style); │ │ │ │ │ - if (drawResult === false) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - node = drawResult.node; │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - var insert = this.indexer.insert(node); │ │ │ │ │ - if (insert) { │ │ │ │ │ - this.vectorRoot.insertBefore(node, insert) │ │ │ │ │ - } else { │ │ │ │ │ - this.vectorRoot.appendChild(node) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - if (node.parentNode !== this.vectorRoot) { │ │ │ │ │ - this.vectorRoot.appendChild(node) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.postDraw(node); │ │ │ │ │ - return drawResult.complete │ │ │ │ │ - }, │ │ │ │ │ - redrawBackgroundNode: function(id, geometry, style, featureId) { │ │ │ │ │ - var backgroundStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ - backgroundStyle.externalGraphic = backgroundStyle.backgroundGraphic; │ │ │ │ │ - backgroundStyle.graphicXOffset = backgroundStyle.backgroundXOffset; │ │ │ │ │ - backgroundStyle.graphicYOffset = backgroundStyle.backgroundYOffset; │ │ │ │ │ - backgroundStyle.graphicZIndex = backgroundStyle.backgroundGraphicZIndex; │ │ │ │ │ - backgroundStyle.graphicWidth = backgroundStyle.backgroundWidth || backgroundStyle.graphicWidth; │ │ │ │ │ - backgroundStyle.graphicHeight = backgroundStyle.backgroundHeight || backgroundStyle.graphicHeight; │ │ │ │ │ - backgroundStyle.backgroundGraphic = null; │ │ │ │ │ - backgroundStyle.backgroundXOffset = null; │ │ │ │ │ - backgroundStyle.backgroundYOffset = null; │ │ │ │ │ - backgroundStyle.backgroundGraphicZIndex = null; │ │ │ │ │ - return this.redrawNode(id + this.BACKGROUND_ID_SUFFIX, geometry, backgroundStyle, null) │ │ │ │ │ - }, │ │ │ │ │ - drawGeometryNode: function(node, geometry, style) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - var options = { │ │ │ │ │ - isFilled: style.fill === undefined ? true : style.fill, │ │ │ │ │ - isStroked: style.stroke === undefined ? !!style.strokeWidth : style.stroke │ │ │ │ │ - }; │ │ │ │ │ - var drawn; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.graphic === false) { │ │ │ │ │ - options.isFilled = false; │ │ │ │ │ - options.isStroked = false │ │ │ │ │ - } │ │ │ │ │ - drawn = this.drawPoint(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - options.isFilled = false; │ │ │ │ │ - drawn = this.drawLineString(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - drawn = this.drawLinearRing(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - drawn = this.drawPolygon(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - drawn = this.drawRectangle(node, geometry); │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - node._options = options; │ │ │ │ │ - if (drawn != false) { │ │ │ │ │ - return { │ │ │ │ │ - node: this.setStyle(node, style, options, geometry), │ │ │ │ │ - complete: drawn │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - postDraw: function(node) {}, │ │ │ │ │ - drawPoint: function(node, geometry) {}, │ │ │ │ │ - drawLineString: function(node, geometry) {}, │ │ │ │ │ - drawLinearRing: function(node, geometry) {}, │ │ │ │ │ - drawPolygon: function(node, geometry) {}, │ │ │ │ │ - drawRectangle: function(node, geometry) {}, │ │ │ │ │ - drawCircle: function(node, geometry) {}, │ │ │ │ │ - removeText: function(featureId) { │ │ │ │ │ - var label = document.getElementById(featureId + this.LABEL_ID_SUFFIX); │ │ │ │ │ - if (label) { │ │ │ │ │ - this.textRoot.removeChild(label) │ │ │ │ │ - } │ │ │ │ │ - var outline = document.getElementById(featureId + this.LABEL_OUTLINE_SUFFIX); │ │ │ │ │ - if (outline) { │ │ │ │ │ - this.textRoot.removeChild(outline) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var target = evt.target; │ │ │ │ │ - var useElement = target && target.correspondingUseElement; │ │ │ │ │ - var node = useElement ? useElement : target || evt.srcElement; │ │ │ │ │ - return node._featureId │ │ │ │ │ - }, │ │ │ │ │ - eraseGeometry: function(geometry, featureId) { │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPoint" || geometry.CLASS_NAME == "OpenLayers.Geometry.MultiLineString" || geometry.CLASS_NAME == "OpenLayers.Geometry.MultiPolygon" || geometry.CLASS_NAME == "OpenLayers.Geometry.Collection") { │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - this.eraseGeometry(geometry.components[i], featureId) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - var element = OpenLayers.Util.getElement(geometry.id); │ │ │ │ │ - if (element && element.parentNode) { │ │ │ │ │ - if (element.geometry) { │ │ │ │ │ - element.geometry.destroy(); │ │ │ │ │ - element.geometry = null │ │ │ │ │ - } │ │ │ │ │ - element.parentNode.removeChild(element); │ │ │ │ │ - if (this.indexer) { │ │ │ │ │ - this.indexer.remove(element) │ │ │ │ │ - } │ │ │ │ │ - if (element._style.backgroundGraphic) { │ │ │ │ │ - var backgroundId = geometry.id + this.BACKGROUND_ID_SUFFIX; │ │ │ │ │ - var bElem = OpenLayers.Util.getElement(backgroundId); │ │ │ │ │ - if (bElem && bElem.parentNode) { │ │ │ │ │ - bElem.parentNode.removeChild(bElem) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - nodeFactory: function(id, type) { │ │ │ │ │ - var node = OpenLayers.Util.getElement(id); │ │ │ │ │ - if (node) { │ │ │ │ │ - if (!this.nodeTypeCompare(node, type)) { │ │ │ │ │ - node.parentNode.removeChild(node); │ │ │ │ │ - node = this.nodeFactory(id, type) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - node = this.createNode(type, id) │ │ │ │ │ - } │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - nodeTypeCompare: function(node, type) {}, │ │ │ │ │ - createNode: function(type, id) {}, │ │ │ │ │ - moveRoot: function(renderer) { │ │ │ │ │ - var root = this.root; │ │ │ │ │ - if (renderer.root.parentNode == this.rendererRoot) { │ │ │ │ │ - root = renderer.root │ │ │ │ │ - } │ │ │ │ │ - root.parentNode.removeChild(root); │ │ │ │ │ - renderer.rendererRoot.appendChild(root) │ │ │ │ │ - }, │ │ │ │ │ - getRenderLayerId: function() { │ │ │ │ │ - return this.root.parentNode.parentNode.id │ │ │ │ │ - }, │ │ │ │ │ - isComplexSymbol: function(graphicName) { │ │ │ │ │ - return graphicName != "circle" && !!graphicName │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.Elements" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Renderer.SVG = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ - xmlns: "http://www.w3.org/2000/svg", │ │ │ │ │ - xlinkns: "http://www.w3.org/1999/xlink", │ │ │ │ │ - MAX_PIXEL: 15e3, │ │ │ │ │ - translationParameters: null, │ │ │ │ │ - symbolMetrics: null, │ │ │ │ │ - initialize: function(containerID) { │ │ │ │ │ - if (!this.supported()) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.translationParameters = { │ │ │ │ │ - x: 0, │ │ │ │ │ - y: 0 │ │ │ │ │ - }; │ │ │ │ │ - this.symbolMetrics = {} │ │ │ │ │ - }, │ │ │ │ │ - supported: function() { │ │ │ │ │ - var svgFeature = "http://www.w3.org/TR/SVG11/feature#"; │ │ │ │ │ - return document.implementation && (document.implementation.hasFeature("org.w3c.svg", "1.0") || document.implementation.hasFeature(svgFeature + "SVG", "1.1") || document.implementation.hasFeature(svgFeature + "BasicStructure", "1.1")) │ │ │ │ │ - }, │ │ │ │ │ - inValidRange: function(x, y, xyOnly) { │ │ │ │ │ - var left = x + (xyOnly ? 0 : this.translationParameters.x); │ │ │ │ │ - var top = y + (xyOnly ? 0 : this.translationParameters.y); │ │ │ │ │ - return left >= -this.MAX_PIXEL && left <= this.MAX_PIXEL && top >= -this.MAX_PIXEL && top <= this.MAX_PIXEL │ │ │ │ │ - }, │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - var resolution = this.getResolution(), │ │ │ │ │ - left = -extent.left / resolution, │ │ │ │ │ - top = extent.top / resolution; │ │ │ │ │ - if (resolutionChanged) { │ │ │ │ │ - this.left = left; │ │ │ │ │ - this.top = top; │ │ │ │ │ - var extentString = "0 0 " + this.size.w + " " + this.size.h; │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "viewBox", extentString); │ │ │ │ │ - this.translate(this.xOffset, 0); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - var inRange = this.translate(left - this.left + this.xOffset, top - this.top); │ │ │ │ │ - if (!inRange) { │ │ │ │ │ - this.setExtent(extent, true) │ │ │ │ │ - } │ │ │ │ │ - return coordSysUnchanged && inRange │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - translate: function(x, y) { │ │ │ │ │ - if (!this.inValidRange(x, y, true)) { │ │ │ │ │ - return false │ │ │ │ │ - } else { │ │ │ │ │ - var transformString = ""; │ │ │ │ │ - if (x || y) { │ │ │ │ │ - transformString = "translate(" + x + "," + y + ")" │ │ │ │ │ - } │ │ │ │ │ - this.root.setAttributeNS(null, "transform", transformString); │ │ │ │ │ - this.translationParameters = { │ │ │ │ │ - x: x, │ │ │ │ │ - y: y │ │ │ │ │ - }; │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "width", this.size.w); │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "height", this.size.h) │ │ │ │ │ - }, │ │ │ │ │ - getNodeType: function(geometry, style) { │ │ │ │ │ - var nodeType = null; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - nodeType = "image" │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - nodeType = "svg" │ │ │ │ │ - } else { │ │ │ │ │ - nodeType = "circle" │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - nodeType = "rect"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - nodeType = "polyline"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - nodeType = "polygon"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - case "OpenLayers.Geometry.Curve": │ │ │ │ │ - nodeType = "path"; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - return nodeType │ │ │ │ │ - }, │ │ │ │ │ - setStyle: function(node, style, options) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - options = options || node._options; │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - node.setAttributeNS(null, "title", title); │ │ │ │ │ - var titleNode = node.getElementsByTagName("title"); │ │ │ │ │ - if (titleNode.length > 0) { │ │ │ │ │ - titleNode[0].firstChild.textContent = title │ │ │ │ │ - } else { │ │ │ │ │ - var label = this.nodeFactory(null, "title"); │ │ │ │ │ - label.textContent = title; │ │ │ │ │ - node.appendChild(label) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var r = parseFloat(node.getAttributeNS(null, "r")); │ │ │ │ │ - var widthFactor = 1; │ │ │ │ │ - var pos; │ │ │ │ │ - if (node._geometryClass == "OpenLayers.Geometry.Point" && r) { │ │ │ │ │ - node.style.visibility = ""; │ │ │ │ │ - if (style.graphic === false) { │ │ │ │ │ - node.style.visibility = "hidden" │ │ │ │ │ - } else if (style.externalGraphic) { │ │ │ │ │ - pos = this.getPosition(node); │ │ │ │ │ - if (style.graphicWidth && style.graphicHeight) { │ │ │ │ │ - node.setAttributeNS(null, "preserveAspectRatio", "none") │ │ │ │ │ - } │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ - var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ - node.setAttributeNS(null, "x", (pos.x + xOffset).toFixed()); │ │ │ │ │ - node.setAttributeNS(null, "y", (pos.y + yOffset).toFixed()); │ │ │ │ │ - node.setAttributeNS(null, "width", width); │ │ │ │ │ - node.setAttributeNS(null, "height", height); │ │ │ │ │ - node.setAttributeNS(this.xlinkns, "xlink:href", style.externalGraphic); │ │ │ │ │ - node.setAttributeNS(null, "style", "opacity: " + opacity); │ │ │ │ │ - node.onclick = OpenLayers.Event.preventDefault │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - var offset = style.pointRadius * 3; │ │ │ │ │ - var size = offset * 2; │ │ │ │ │ - var src = this.importSymbol(style.graphicName); │ │ │ │ │ - pos = this.getPosition(node); │ │ │ │ │ - widthFactor = this.symbolMetrics[src.id][0] * 3 / size; │ │ │ │ │ - var parent = node.parentNode; │ │ │ │ │ - var nextSibling = node.nextSibling; │ │ │ │ │ - if (parent) { │ │ │ │ │ - parent.removeChild(node) │ │ │ │ │ - } │ │ │ │ │ - node.firstChild && node.removeChild(node.firstChild); │ │ │ │ │ - node.appendChild(src.firstChild.cloneNode(true)); │ │ │ │ │ - node.setAttributeNS(null, "viewBox", src.getAttributeNS(null, "viewBox")); │ │ │ │ │ - node.setAttributeNS(null, "width", size); │ │ │ │ │ - node.setAttributeNS(null, "height", size); │ │ │ │ │ - node.setAttributeNS(null, "x", pos.x - offset); │ │ │ │ │ - node.setAttributeNS(null, "y", pos.y - offset); │ │ │ │ │ - if (nextSibling) { │ │ │ │ │ - parent.insertBefore(node, nextSibling) │ │ │ │ │ - } else if (parent) { │ │ │ │ │ - parent.appendChild(node) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "r", style.pointRadius) │ │ │ │ │ - } │ │ │ │ │ - var rotation = style.rotation; │ │ │ │ │ - if ((rotation !== undefined || node._rotation !== undefined) && pos) { │ │ │ │ │ - node._rotation = rotation; │ │ │ │ │ - rotation |= 0; │ │ │ │ │ - if (node.nodeName !== "svg") { │ │ │ │ │ - node.setAttributeNS(null, "transform", "rotate(" + rotation + " " + pos.x + " " + pos.y + ")") │ │ │ │ │ - } else { │ │ │ │ │ - var metrics = this.symbolMetrics[src.id]; │ │ │ │ │ - node.firstChild.setAttributeNS(null, "transform", "rotate(" + rotation + " " + metrics[1] + " " + metrics[2] + ")") │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (options.isFilled) { │ │ │ │ │ - node.setAttributeNS(null, "fill", style.fillColor); │ │ │ │ │ - node.setAttributeNS(null, "fill-opacity", style.fillOpacity) │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "fill", "none") │ │ │ │ │ - } │ │ │ │ │ - if (options.isStroked) { │ │ │ │ │ - node.setAttributeNS(null, "stroke", style.strokeColor); │ │ │ │ │ - node.setAttributeNS(null, "stroke-opacity", style.strokeOpacity); │ │ │ │ │ - node.setAttributeNS(null, "stroke-width", style.strokeWidth * widthFactor); │ │ │ │ │ - node.setAttributeNS(null, "stroke-linecap", style.strokeLinecap || "round"); │ │ │ │ │ - node.setAttributeNS(null, "stroke-linejoin", "round"); │ │ │ │ │ - style.strokeDashstyle && node.setAttributeNS(null, "stroke-dasharray", this.dashStyle(style, widthFactor)) │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "stroke", "none") │ │ │ │ │ - } │ │ │ │ │ - if (style.pointerEvents) { │ │ │ │ │ - node.setAttributeNS(null, "pointer-events", style.pointerEvents) │ │ │ │ │ - } │ │ │ │ │ - if (style.cursor != null) { │ │ │ │ │ - node.setAttributeNS(null, "cursor", style.cursor) │ │ │ │ │ - } │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - dashStyle: function(style, widthFactor) { │ │ │ │ │ - var w = style.strokeWidth * widthFactor; │ │ │ │ │ - var str = style.strokeDashstyle; │ │ │ │ │ - switch (str) { │ │ │ │ │ - case "solid": │ │ │ │ │ - return "none"; │ │ │ │ │ - case "dot": │ │ │ │ │ - return [1, 4 * w].join(); │ │ │ │ │ - case "dash": │ │ │ │ │ - return [4 * w, 4 * w].join(); │ │ │ │ │ - case "dashdot": │ │ │ │ │ - return [4 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ - case "longdash": │ │ │ │ │ - return [8 * w, 4 * w].join(); │ │ │ │ │ - case "longdashdot": │ │ │ │ │ - return [8 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ - default: │ │ │ │ │ - return OpenLayers.String.trim(str).replace(/\s+/g, ",") │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - createNode: function(type, id) { │ │ │ │ │ - var node = document.createElementNS(this.xmlns, type); │ │ │ │ │ - if (id) { │ │ │ │ │ - node.setAttributeNS(null, "id", id) │ │ │ │ │ - } │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - nodeTypeCompare: function(node, type) { │ │ │ │ │ - return type == node.nodeName │ │ │ │ │ - }, │ │ │ │ │ - createRenderRoot: function() { │ │ │ │ │ - var svg = this.nodeFactory(this.container.id + "_svgRoot", "svg"); │ │ │ │ │ - svg.style.display = "block"; │ │ │ │ │ - return svg │ │ │ │ │ - }, │ │ │ │ │ - createRoot: function(suffix) { │ │ │ │ │ - return this.nodeFactory(this.container.id + suffix, "g") │ │ │ │ │ - }, │ │ │ │ │ - createDefs: function() { │ │ │ │ │ - var defs = this.nodeFactory(this.container.id + "_defs", "defs"); │ │ │ │ │ - this.rendererRoot.appendChild(defs); │ │ │ │ │ - return defs │ │ │ │ │ - }, │ │ │ │ │ - drawPoint: function(node, geometry) { │ │ │ │ │ - return this.drawCircle(node, geometry, 1) │ │ │ │ │ - }, │ │ │ │ │ - drawCircle: function(node, geometry, radius) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = (geometry.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y = this.top - geometry.y / resolution; │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - node.setAttributeNS(null, "cx", x); │ │ │ │ │ - node.setAttributeNS(null, "cy", y); │ │ │ │ │ - node.setAttributeNS(null, "r", radius); │ │ │ │ │ - return node │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - drawLineString: function(node, geometry) { │ │ │ │ │ - var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ - if (componentsResult.path) { │ │ │ │ │ - node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ - return componentsResult.complete ? node : null │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - drawLinearRing: function(node, geometry) { │ │ │ │ │ - var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ - if (componentsResult.path) { │ │ │ │ │ - node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ - return componentsResult.complete ? node : null │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - drawPolygon: function(node, geometry) { │ │ │ │ │ - var d = ""; │ │ │ │ │ - var draw = true; │ │ │ │ │ - var complete = true; │ │ │ │ │ - var linearRingResult, path; │ │ │ │ │ - for (var j = 0, len = geometry.components.length; j < len; j++) { │ │ │ │ │ - d += " M"; │ │ │ │ │ - linearRingResult = this.getComponentsString(geometry.components[j].components, " "); │ │ │ │ │ - path = linearRingResult.path; │ │ │ │ │ - if (path) { │ │ │ │ │ - d += " " + path; │ │ │ │ │ - complete = linearRingResult.complete && complete │ │ │ │ │ - } else { │ │ │ │ │ - draw = false │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - d += " z"; │ │ │ │ │ - if (draw) { │ │ │ │ │ - node.setAttributeNS(null, "d", d); │ │ │ │ │ - node.setAttributeNS(null, "fill-rule", "evenodd"); │ │ │ │ │ - return complete ? node : null │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - drawRectangle: function(node, geometry) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = (geometry.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y = this.top - geometry.y / resolution; │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - node.setAttributeNS(null, "x", x); │ │ │ │ │ - node.setAttributeNS(null, "y", y); │ │ │ │ │ - node.setAttributeNS(null, "width", geometry.width / resolution); │ │ │ │ │ - node.setAttributeNS(null, "height", geometry.height / resolution); │ │ │ │ │ - return node │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - drawText: function(featureId, style, location) { │ │ │ │ │ - var drawOutline = !!style.labelOutlineWidth; │ │ │ │ │ - if (drawOutline) { │ │ │ │ │ - var outlineStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ - outlineStyle.fontColor = outlineStyle.labelOutlineColor; │ │ │ │ │ - outlineStyle.fontStrokeColor = outlineStyle.labelOutlineColor; │ │ │ │ │ - outlineStyle.fontStrokeWidth = style.labelOutlineWidth; │ │ │ │ │ - if (style.labelOutlineOpacity) { │ │ │ │ │ - outlineStyle.fontOpacity = style.labelOutlineOpacity │ │ │ │ │ - } │ │ │ │ │ - delete outlineStyle.labelOutlineWidth; │ │ │ │ │ - this.drawText(featureId, outlineStyle, location) │ │ │ │ │ - } │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = (location.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y = location.y / resolution - this.top; │ │ │ │ │ - var suffix = drawOutline ? this.LABEL_OUTLINE_SUFFIX : this.LABEL_ID_SUFFIX; │ │ │ │ │ - var label = this.nodeFactory(featureId + suffix, "text"); │ │ │ │ │ - label.setAttributeNS(null, "x", x); │ │ │ │ │ - label.setAttributeNS(null, "y", -y); │ │ │ │ │ - if (style.fontColor) { │ │ │ │ │ - label.setAttributeNS(null, "fill", style.fontColor) │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStrokeColor) { │ │ │ │ │ - label.setAttributeNS(null, "stroke", style.fontStrokeColor) │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStrokeWidth) { │ │ │ │ │ - label.setAttributeNS(null, "stroke-width", style.fontStrokeWidth) │ │ │ │ │ - } │ │ │ │ │ - if (style.fontOpacity) { │ │ │ │ │ - label.setAttributeNS(null, "opacity", style.fontOpacity) │ │ │ │ │ - } │ │ │ │ │ - if (style.fontFamily) { │ │ │ │ │ - label.setAttributeNS(null, "font-family", style.fontFamily) │ │ │ │ │ - } │ │ │ │ │ - if (style.fontSize) { │ │ │ │ │ - label.setAttributeNS(null, "font-size", style.fontSize) │ │ │ │ │ - } │ │ │ │ │ - if (style.fontWeight) { │ │ │ │ │ - label.setAttributeNS(null, "font-weight", style.fontWeight) │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStyle) { │ │ │ │ │ - label.setAttributeNS(null, "font-style", style.fontStyle) │ │ │ │ │ - } │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - label.setAttributeNS(null, "pointer-events", "visible"); │ │ │ │ │ - label._featureId = featureId │ │ │ │ │ - } else { │ │ │ │ │ - label.setAttributeNS(null, "pointer-events", "none") │ │ │ │ │ - } │ │ │ │ │ - var align = style.labelAlign || OpenLayers.Renderer.defaultSymbolizer.labelAlign; │ │ │ │ │ - label.setAttributeNS(null, "text-anchor", OpenLayers.Renderer.SVG.LABEL_ALIGN[align[0]] || "middle"); │ │ │ │ │ - if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ - label.setAttributeNS(null, "dominant-baseline", OpenLayers.Renderer.SVG.LABEL_ALIGN[align[1]] || "central") │ │ │ │ │ - } │ │ │ │ │ - var labelRows = style.label.split("\n"); │ │ │ │ │ - var numRows = labelRows.length; │ │ │ │ │ - while (label.childNodes.length > numRows) { │ │ │ │ │ - label.removeChild(label.lastChild) │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - var tspan = this.nodeFactory(featureId + suffix + "_tspan_" + i, "tspan"); │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - tspan._featureId = featureId; │ │ │ │ │ - tspan._geometry = location; │ │ │ │ │ - tspan._geometryClass = location.CLASS_NAME │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.IS_GECKO === false) { │ │ │ │ │ - tspan.setAttributeNS(null, "baseline-shift", OpenLayers.Renderer.SVG.LABEL_VSHIFT[align[1]] || "-35%") │ │ │ │ │ - } │ │ │ │ │ - tspan.setAttribute("x", x); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - var vfactor = OpenLayers.Renderer.SVG.LABEL_VFACTOR[align[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5 │ │ │ │ │ - } │ │ │ │ │ - tspan.setAttribute("dy", vfactor * (numRows - 1) + "em") │ │ │ │ │ - } else { │ │ │ │ │ - tspan.setAttribute("dy", "1em") │ │ │ │ │ - } │ │ │ │ │ - tspan.textContent = labelRows[i] === "" ? " " : labelRows[i]; │ │ │ │ │ - if (!tspan.parentNode) { │ │ │ │ │ - label.appendChild(tspan) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!label.parentNode) { │ │ │ │ │ - this.textRoot.appendChild(label) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getComponentsString: function(components, separator) { │ │ │ │ │ - var renderCmp = []; │ │ │ │ │ - var complete = true; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - var strings = []; │ │ │ │ │ - var str, component; │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - component = components[i]; │ │ │ │ │ - renderCmp.push(component); │ │ │ │ │ - str = this.getShortString(component); │ │ │ │ │ - if (str) { │ │ │ │ │ - strings.push(str) │ │ │ │ │ - } else { │ │ │ │ │ - if (i > 0) { │ │ │ │ │ - if (this.getShortString(components[i - 1])) { │ │ │ │ │ - strings.push(this.clipLine(components[i], components[i - 1])) │ │ │ │ │ + targetSplit = true │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (i < len - 1) { │ │ │ │ │ - if (this.getShortString(components[i + 1])) { │ │ │ │ │ - strings.push(this.clipLine(components[i], components[i + 1])) │ │ │ │ │ + if (!targetSplit) { │ │ │ │ │ + if (targetParts.length) { │ │ │ │ │ + targetParts[targetParts.length - 1].addComponent(targetLine.clone()) │ │ │ │ │ + } else { │ │ │ │ │ + targetParts = [new OpenLayers.Geometry.MultiLineString([targetLine.clone()])] │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - complete = false │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - path: strings.join(separator || ","), │ │ │ │ │ - complete: complete │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - clipLine: function(badComponent, goodComponent) { │ │ │ │ │ - if (goodComponent.equals(badComponent)) { │ │ │ │ │ - return "" │ │ │ │ │ - } │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var maxX = this.MAX_PIXEL - this.translationParameters.x; │ │ │ │ │ - var maxY = this.MAX_PIXEL - this.translationParameters.y; │ │ │ │ │ - var x1 = (goodComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y1 = this.top - goodComponent.y / resolution; │ │ │ │ │ - var x2 = (badComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y2 = this.top - badComponent.y / resolution; │ │ │ │ │ - var k; │ │ │ │ │ - if (x2 < -maxX || x2 > maxX) { │ │ │ │ │ - k = (y2 - y1) / (x2 - x1); │ │ │ │ │ - x2 = x2 < 0 ? -maxX : maxX; │ │ │ │ │ - y2 = y1 + (x2 - x1) * k │ │ │ │ │ - } │ │ │ │ │ - if (y2 < -maxY || y2 > maxY) { │ │ │ │ │ - k = (x2 - x1) / (y2 - y1); │ │ │ │ │ - y2 = y2 < 0 ? -maxY : maxY; │ │ │ │ │ - x2 = x1 + (y2 - y1) * k │ │ │ │ │ - } │ │ │ │ │ - return x2 + "," + y2 │ │ │ │ │ - }, │ │ │ │ │ - getShortString: function(point) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = (point.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y = this.top - point.y / resolution; │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - return x + "," + y │ │ │ │ │ } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getPosition: function(node) { │ │ │ │ │ - return { │ │ │ │ │ - x: parseFloat(node.getAttributeNS(null, "cx")), │ │ │ │ │ - y: parseFloat(node.getAttributeNS(null, "cy")) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - importSymbol: function(graphicName) { │ │ │ │ │ - if (!this.defs) { │ │ │ │ │ - this.defs = this.createDefs() │ │ │ │ │ - } │ │ │ │ │ - var id = this.container.id + "-" + graphicName; │ │ │ │ │ - var existing = document.getElementById(id); │ │ │ │ │ - if (existing != null) { │ │ │ │ │ - return existing │ │ │ │ │ - } │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(graphicName + " is not a valid symbol name") │ │ │ │ │ - } │ │ │ │ │ - var symbolNode = this.nodeFactory(id, "symbol"); │ │ │ │ │ - var node = this.nodeFactory(null, "polygon"); │ │ │ │ │ - symbolNode.appendChild(node); │ │ │ │ │ - var symbolExtent = new OpenLayers.Bounds(Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ - var points = []; │ │ │ │ │ - var x, y; │ │ │ │ │ - for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ - symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ - symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ - symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ - points.push(x, ",", y) │ │ │ │ │ - } │ │ │ │ │ - node.setAttributeNS(null, "points", points.join(" ")); │ │ │ │ │ - var width = symbolExtent.getWidth(); │ │ │ │ │ - var height = symbolExtent.getHeight(); │ │ │ │ │ - var viewBox = [symbolExtent.left - width, symbolExtent.bottom - height, width * 3, height * 3]; │ │ │ │ │ - symbolNode.setAttributeNS(null, "viewBox", viewBox.join(" ")); │ │ │ │ │ - this.symbolMetrics[id] = [Math.max(width, height), symbolExtent.getCenterLonLat().lon, symbolExtent.getCenterLonLat().lat]; │ │ │ │ │ - this.defs.appendChild(symbolNode); │ │ │ │ │ - return symbolNode │ │ │ │ │ - }, │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var featureId = OpenLayers.Renderer.Elements.prototype.getFeatureIdFromEvent.apply(this, arguments); │ │ │ │ │ - if (!featureId) { │ │ │ │ │ - var target = evt.target; │ │ │ │ │ - featureId = target.parentNode && target != this.rendererRoot ? target.parentNode._featureId : undefined │ │ │ │ │ + results = geometry.split(this) │ │ │ │ │ } │ │ │ │ │ - return featureId │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.SVG" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_ALIGN = { │ │ │ │ │ - l: "start", │ │ │ │ │ - r: "end", │ │ │ │ │ - b: "bottom", │ │ │ │ │ - t: "hanging" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_VSHIFT = { │ │ │ │ │ - t: "-70%", │ │ │ │ │ - b: "0" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_VFACTOR = { │ │ │ │ │ - t: 0, │ │ │ │ │ - b: -1 │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.SVG.preventDefault = function(e) { │ │ │ │ │ - OpenLayers.Event.preventDefault(e) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol = OpenLayers.Class({ │ │ │ │ │ - format: null, │ │ │ │ │ - options: null, │ │ │ │ │ - autoDestroy: true, │ │ │ │ │ - defaultFilter: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.options = options │ │ │ │ │ - }, │ │ │ │ │ - mergeWithDefaultFilter: function(filter) { │ │ │ │ │ - var merged; │ │ │ │ │ - if (filter && this.defaultFilter) { │ │ │ │ │ - merged = new OpenLayers.Filter.Logical({ │ │ │ │ │ - type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ - filters: [this.defaultFilter, filter] │ │ │ │ │ - }) │ │ │ │ │ + if (sourceParts && sourceParts.length > 1) { │ │ │ │ │ + sourceSplit = true │ │ │ │ │ } else { │ │ │ │ │ - merged = filter || this.defaultFilter || undefined │ │ │ │ │ - } │ │ │ │ │ - return merged │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.options = null; │ │ │ │ │ - this.format = null │ │ │ │ │ - }, │ │ │ │ │ - read: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.filter = this.mergeWithDefaultFilter(options.filter) │ │ │ │ │ - }, │ │ │ │ │ - create: function() {}, │ │ │ │ │ - update: function() {}, │ │ │ │ │ - delete: function() {}, │ │ │ │ │ - commit: function() {}, │ │ │ │ │ - abort: function(response) {}, │ │ │ │ │ - createCallback: function(method, response, options) { │ │ │ │ │ - return OpenLayers.Function.bind(function() { │ │ │ │ │ - method.apply(this, [response, options]) │ │ │ │ │ - }, this) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Protocol.Response = OpenLayers.Class({ │ │ │ │ │ - code: null, │ │ │ │ │ - requestType: null, │ │ │ │ │ - last: true, │ │ │ │ │ - features: null, │ │ │ │ │ - data: null, │ │ │ │ │ - reqFeatures: null, │ │ │ │ │ - priv: null, │ │ │ │ │ - error: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options) │ │ │ │ │ - }, │ │ │ │ │ - success: function() { │ │ │ │ │ - return this.code > 0 │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.Response" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Protocol.Response.SUCCESS = 1; │ │ │ │ │ -OpenLayers.Protocol.Response.FAILURE = 0; │ │ │ │ │ -OpenLayers.Protocol.WFS = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, OpenLayers.Protocol.WFS.DEFAULTS); │ │ │ │ │ - var cls = OpenLayers.Protocol.WFS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported WFS version: " + options.version │ │ │ │ │ - } │ │ │ │ │ - return new cls(options) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.WFS.fromWMSLayer = function(layer, options) { │ │ │ │ │ - var typeName, featurePrefix; │ │ │ │ │ - var param = layer.params["LAYERS"]; │ │ │ │ │ - var parts = (OpenLayers.Util.isArray(param) ? param[0] : param).split(":"); │ │ │ │ │ - if (parts.length > 1) { │ │ │ │ │ - featurePrefix = parts[0] │ │ │ │ │ - } │ │ │ │ │ - typeName = parts.pop(); │ │ │ │ │ - var protocolOptions = { │ │ │ │ │ - url: layer.url, │ │ │ │ │ - featureType: typeName, │ │ │ │ │ - featurePrefix: featurePrefix, │ │ │ │ │ - srsName: layer.projection && layer.projection.getCode() || layer.map && layer.map.getProjectionObject().getCode(), │ │ │ │ │ - version: "1.1.0" │ │ │ │ │ - }; │ │ │ │ │ - return new OpenLayers.Protocol.WFS(OpenLayers.Util.applyDefaults(options, protocolOptions)) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.WFS.DEFAULTS = { │ │ │ │ │ - version: "1.0.0" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.HTTP = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ - url: null, │ │ │ │ │ - headers: null, │ │ │ │ │ - params: null, │ │ │ │ │ - callback: null, │ │ │ │ │ - scope: null, │ │ │ │ │ - readWithPOST: false, │ │ │ │ │ - updateWithPOST: false, │ │ │ │ │ - deleteWithPOST: false, │ │ │ │ │ - wildcarded: false, │ │ │ │ │ - srsInBBOX: false, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - this.params = {}; │ │ │ │ │ - this.headers = {}; │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ - var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ - wildcarded: this.wildcarded, │ │ │ │ │ - srsInBBOX: this.srsInBBOX │ │ │ │ │ - }); │ │ │ │ │ - this.filterToParams = function(filter, params) { │ │ │ │ │ - return format.write(filter, params) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.params = null; │ │ │ │ │ - this.headers = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ - }, │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.params = OpenLayers.Util.applyDefaults(options.params, this.options.params); │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - if (options.filter && this.filterToParams) { │ │ │ │ │ - options.params = this.filterToParams(options.filter, options.params) │ │ │ │ │ + sourceParts = [] │ │ │ │ │ } │ │ │ │ │ - var readWithPOST = options.readWithPOST !== undefined ? options.readWithPOST : this.readWithPOST; │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - if (readWithPOST) { │ │ │ │ │ - var headers = options.headers || {}; │ │ │ │ │ - headers["Content-Type"] = "application/x-www-form-urlencoded"; │ │ │ │ │ - resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ - data: OpenLayers.Util.getParameterString(options.params), │ │ │ │ │ - headers: headers │ │ │ │ │ - }) │ │ │ │ │ + if (targetParts && targetParts.length > 1) { │ │ │ │ │ + targetSplit = true │ │ │ │ │ } else { │ │ │ │ │ - resp.priv = OpenLayers.Request.GET({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return resp │ │ │ │ │ - }, │ │ │ │ │ - handleRead: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options) │ │ │ │ │ - }, │ │ │ │ │ - create: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: features, │ │ │ │ │ - requestType: "create" │ │ │ │ │ - }); │ │ │ │ │ - resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleCreate, resp, options), │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(features) │ │ │ │ │ - }); │ │ │ │ │ - return resp │ │ │ │ │ - }, │ │ │ │ │ - handleCreate: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options) │ │ │ │ │ - }, │ │ │ │ │ - update: function(feature, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var url = options.url || feature.url || this.options.url + "/" + feature.fid; │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: feature, │ │ │ │ │ - requestType: "update" │ │ │ │ │ - }); │ │ │ │ │ - var method = this.updateWithPOST ? "POST" : "PUT"; │ │ │ │ │ - resp.priv = OpenLayers.Request[method]({ │ │ │ │ │ - url: url, │ │ │ │ │ - callback: this.createCallback(this.handleUpdate, resp, options), │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(feature) │ │ │ │ │ - }); │ │ │ │ │ - return resp │ │ │ │ │ - }, │ │ │ │ │ - handleUpdate: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options) │ │ │ │ │ - }, │ │ │ │ │ - delete: function(feature, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var url = options.url || feature.url || this.options.url + "/" + feature.fid; │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: feature, │ │ │ │ │ - requestType: "delete" │ │ │ │ │ - }); │ │ │ │ │ - var method = this.deleteWithPOST ? "POST" : "DELETE"; │ │ │ │ │ - var requestOptions = { │ │ │ │ │ - url: url, │ │ │ │ │ - callback: this.createCallback(this.handleDelete, resp, options), │ │ │ │ │ - headers: options.headers │ │ │ │ │ - }; │ │ │ │ │ - if (this.deleteWithPOST) { │ │ │ │ │ - requestOptions.data = this.format.write(feature) │ │ │ │ │ - } │ │ │ │ │ - resp.priv = OpenLayers.Request[method](requestOptions); │ │ │ │ │ - return resp │ │ │ │ │ - }, │ │ │ │ │ - handleDelete: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options) │ │ │ │ │ - }, │ │ │ │ │ - handleResponse: function(resp, options) { │ │ │ │ │ - var request = resp.priv; │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - if (resp.requestType != "delete") { │ │ │ │ │ - resp.features = this.parseFeatures(request) │ │ │ │ │ - } │ │ │ │ │ - resp.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ - } else { │ │ │ │ │ - resp.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ - } │ │ │ │ │ - options.callback.call(options.scope, resp) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - parseFeatures: function(request) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null │ │ │ │ │ - } │ │ │ │ │ - return this.format.read(doc) │ │ │ │ │ - }, │ │ │ │ │ - commit: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = [], │ │ │ │ │ - nResponses = 0; │ │ │ │ │ - var types = {}; │ │ │ │ │ - types[OpenLayers.State.INSERT] = []; │ │ │ │ │ - types[OpenLayers.State.UPDATE] = []; │ │ │ │ │ - types[OpenLayers.State.DELETE] = []; │ │ │ │ │ - var feature, list, requestFeatures = []; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - list = types[feature.state]; │ │ │ │ │ - if (list) { │ │ │ │ │ - list.push(feature); │ │ │ │ │ - requestFeatures.push(feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var nRequests = (types[OpenLayers.State.INSERT].length > 0 ? 1 : 0) + types[OpenLayers.State.UPDATE].length + types[OpenLayers.State.DELETE].length; │ │ │ │ │ - var success = true; │ │ │ │ │ - var finalResponse = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: requestFeatures │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - function insertCallback(response) { │ │ │ │ │ - var len = response.features ? response.features.length : 0; │ │ │ │ │ - var fids = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - fids[i] = response.features[i].fid │ │ │ │ │ - } │ │ │ │ │ - finalResponse.insertIds = fids; │ │ │ │ │ - callback.apply(this, [response]) │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function callback(response) { │ │ │ │ │ - this.callUserCallback(response, options); │ │ │ │ │ - success = success && response.success(); │ │ │ │ │ - nResponses++; │ │ │ │ │ - if (nResponses >= nRequests) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - finalResponse.code = success ? OpenLayers.Protocol.Response.SUCCESS : OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - options.callback.apply(options.scope, [finalResponse]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var queue = types[OpenLayers.State.INSERT]; │ │ │ │ │ - if (queue.length > 0) { │ │ │ │ │ - resp.push(this.create(queue, OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: insertCallback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options.create))) │ │ │ │ │ - } │ │ │ │ │ - queue = types[OpenLayers.State.UPDATE]; │ │ │ │ │ - for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ - resp.push(this.update(queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: callback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options.update))) │ │ │ │ │ - } │ │ │ │ │ - queue = types[OpenLayers.State.DELETE]; │ │ │ │ │ - for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ - resp.push(this["delete"](queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: callback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options["delete"]))) │ │ │ │ │ - } │ │ │ │ │ - return resp │ │ │ │ │ - }, │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - response.priv.abort() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - callUserCallback: function(resp, options) { │ │ │ │ │ - var opt = options[resp.requestType]; │ │ │ │ │ - if (opt && opt.callback) { │ │ │ │ │ - opt.callback.call(opt.scope, resp) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.HTTP" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Protocol.WFS.v1 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ - version: null, │ │ │ │ │ - srsName: "EPSG:4326", │ │ │ │ │ - featureType: null, │ │ │ │ │ - featureNS: null, │ │ │ │ │ - geometryName: "the_geom", │ │ │ │ │ - schema: null, │ │ │ │ │ - featurePrefix: "feature", │ │ │ │ │ - formatOptions: null, │ │ │ │ │ - readFormat: null, │ │ │ │ │ - readOptions: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!options.format) { │ │ │ │ │ - this.format = OpenLayers.Format.WFST(OpenLayers.Util.extend({ │ │ │ │ │ - version: this.version, │ │ │ │ │ - featureType: this.featureType, │ │ │ │ │ - featureNS: this.featureNS, │ │ │ │ │ - featurePrefix: this.featurePrefix, │ │ │ │ │ - geometryName: this.geometryName, │ │ │ │ │ - srsName: this.srsName, │ │ │ │ │ - schema: this.schema │ │ │ │ │ - }, this.formatOptions)) │ │ │ │ │ - } │ │ │ │ │ - if (!options.geometryName && parseFloat(this.format.version) > 1) { │ │ │ │ │ - this.setGeometryName(null) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.options && !this.options.format) { │ │ │ │ │ - this.format.destroy() │ │ │ │ │ - } │ │ │ │ │ - this.format = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ - }, │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - var data = OpenLayers.Format.XML.prototype.write.apply(this.format, [this.format.writeNode("wfs:GetFeature", options)]); │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: data │ │ │ │ │ - }); │ │ │ │ │ - return response │ │ │ │ │ - }, │ │ │ │ │ - setFeatureType: function(featureType) { │ │ │ │ │ - this.featureType = featureType; │ │ │ │ │ - this.format.featureType = featureType │ │ │ │ │ - }, │ │ │ │ │ - setGeometryName: function(geometryName) { │ │ │ │ │ - this.geometryName = geometryName; │ │ │ │ │ - this.format.geometryName = geometryName │ │ │ │ │ - }, │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - var result = this.parseResponse(request, options.readOptions); │ │ │ │ │ - if (result && result.success !== false) { │ │ │ │ │ - if (options.readOptions && options.readOptions.output == "object") { │ │ │ │ │ - OpenLayers.Util.extend(response, result) │ │ │ │ │ - } else { │ │ │ │ │ - response.features = result │ │ │ │ │ - } │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - response.error = result │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ - } │ │ │ │ │ - options.callback.call(options.scope, response) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - parseResponse: function(request, options) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null │ │ │ │ │ - } │ │ │ │ │ - var result = this.readFormat !== null ? this.readFormat.read(doc) : this.format.read(doc, options); │ │ │ │ │ - if (!this.featureNS) { │ │ │ │ │ - var format = this.readFormat || this.format; │ │ │ │ │ - this.featureNS = format.featureNS; │ │ │ │ │ - format.autoConfig = false; │ │ │ │ │ - if (!this.geometryName) { │ │ │ │ │ - this.setGeometryName(format.geometryName) │ │ │ │ │ - } │ │ │ │ │ + targetParts = [] │ │ │ │ │ } │ │ │ │ │ - return result │ │ │ │ │ - }, │ │ │ │ │ - commit: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "commit", │ │ │ │ │ - reqFeatures: features │ │ │ │ │ - }); │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(features, options), │ │ │ │ │ - callback: this.createCallback(this.handleCommit, response, options) │ │ │ │ │ - }); │ │ │ │ │ - return response │ │ │ │ │ - }, │ │ │ │ │ - handleCommit: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - var data = request.responseXML; │ │ │ │ │ - if (!data || !data.documentElement) { │ │ │ │ │ - data = request.responseText │ │ │ │ │ - } │ │ │ │ │ - var obj = this.format.read(data) || {}; │ │ │ │ │ - response.insertIds = obj.insertIds || []; │ │ │ │ │ - if (obj.success) { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ + if (sourceSplit || targetSplit) { │ │ │ │ │ + if (mutual) { │ │ │ │ │ + results = [sourceParts, targetParts] │ │ │ │ │ } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - response.error = obj │ │ │ │ │ - } │ │ │ │ │ - options.callback.call(options.scope, response) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - filterDelete: function(filter, options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "commit" │ │ │ │ │ - }); │ │ │ │ │ - var root = this.format.createElementNSPlus("wfs:Transaction", { │ │ │ │ │ - attributes: { │ │ │ │ │ - service: "WFS", │ │ │ │ │ - version: this.version │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - var deleteNode = this.format.createElementNSPlus("wfs:Delete", { │ │ │ │ │ - attributes: { │ │ │ │ │ - typeName: (options.featureNS ? this.featurePrefix + ":" : "") + options.featureType │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (options.featureNS) { │ │ │ │ │ - deleteNode.setAttribute("xmlns:" + this.featurePrefix, options.featureNS) │ │ │ │ │ - } │ │ │ │ │ - var filterNode = this.format.writeNode("ogc:Filter", filter); │ │ │ │ │ - deleteNode.appendChild(filterNode); │ │ │ │ │ - root.appendChild(deleteNode); │ │ │ │ │ - var data = OpenLayers.Format.XML.prototype.write.apply(this.format, [root]); │ │ │ │ │ - return OpenLayers.Request.POST({ │ │ │ │ │ - url: this.url, │ │ │ │ │ - callback: options.callback || function() {}, │ │ │ │ │ - data: data │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - response.priv.abort() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.WFS.v1" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WFST = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, OpenLayers.Format.WFST.DEFAULTS); │ │ │ │ │ - var cls = OpenLayers.Format.WFST["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported WFST version: " + options.version │ │ │ │ │ - } │ │ │ │ │ - return new cls(options) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Format.WFST.DEFAULTS = { │ │ │ │ │ - version: "1.0.0" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Filter = OpenLayers.Class({ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() {}, │ │ │ │ │ - evaluate: function(context) { │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - return null │ │ │ │ │ - }, │ │ │ │ │ - toString: function() { │ │ │ │ │ - var string; │ │ │ │ │ - if (OpenLayers.Format && OpenLayers.Format.CQL) { │ │ │ │ │ - string = OpenLayers.Format.CQL.prototype.write(this) │ │ │ │ │ - } else { │ │ │ │ │ - string = Object.prototype.toString.call(this) │ │ │ │ │ - } │ │ │ │ │ - return string │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Filter" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Filter.Spatial = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ - type: null, │ │ │ │ │ - property: null, │ │ │ │ │ - value: null, │ │ │ │ │ - distance: null, │ │ │ │ │ - distanceUnits: null, │ │ │ │ │ - evaluate: function(feature) { │ │ │ │ │ - var intersect = false; │ │ │ │ │ - switch (this.type) { │ │ │ │ │ - case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ - case OpenLayers.Filter.Spatial.INTERSECTS: │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - var geom = this.value; │ │ │ │ │ - if (this.value.CLASS_NAME == "OpenLayers.Bounds") { │ │ │ │ │ - geom = this.value.toGeometry() │ │ │ │ │ - } │ │ │ │ │ - if (feature.geometry.intersects(geom)) { │ │ │ │ │ - intersect = true │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - throw new Error("evaluate is not implemented for this filter type.") │ │ │ │ │ - } │ │ │ │ │ - return intersect │ │ │ │ │ - }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - var options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - value: this.value && this.value.clone && this.value.clone() │ │ │ │ │ - }, this); │ │ │ │ │ - return new OpenLayers.Filter.Spatial(options) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Filter.Spatial" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Filter.Spatial.BBOX = "BBOX"; │ │ │ │ │ -OpenLayers.Filter.Spatial.INTERSECTS = "INTERSECTS"; │ │ │ │ │ -OpenLayers.Filter.Spatial.DWITHIN = "DWITHIN"; │ │ │ │ │ -OpenLayers.Filter.Spatial.WITHIN = "WITHIN"; │ │ │ │ │ -OpenLayers.Filter.Spatial.CONTAINS = "CONTAINS"; │ │ │ │ │ -OpenLayers.Filter.FeatureId = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ - fids: null, │ │ │ │ │ - type: "FID", │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.fids = []; │ │ │ │ │ - OpenLayers.Filter.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ - evaluate: function(feature) { │ │ │ │ │ - for (var i = 0, len = this.fids.length; i < len; i++) { │ │ │ │ │ - var fid = feature.fid || feature.id; │ │ │ │ │ - if (fid == this.fids[i]) { │ │ │ │ │ - return true │ │ │ │ │ + results = targetParts │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - var filter = new OpenLayers.Filter.FeatureId; │ │ │ │ │ - OpenLayers.Util.extend(filter, this); │ │ │ │ │ - filter.fids = this.fids.slice(); │ │ │ │ │ - return filter │ │ │ │ │ + return results │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Filter.FeatureId" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Geometry.MultiLineString" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.WFST.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - namespaces: { │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - wfs: "http://www.opengis.net/wfs", │ │ │ │ │ - gml: "http://www.opengis.net/gml", │ │ │ │ │ - ogc: "http://www.opengis.net/ogc", │ │ │ │ │ - ows: "http://www.opengis.net/ows" │ │ │ │ │ - }, │ │ │ │ │ - defaultPrefix: "wfs", │ │ │ │ │ - version: null, │ │ │ │ │ - schemaLocations: null, │ │ │ │ │ - srsName: null, │ │ │ │ │ - extractAttributes: true, │ │ │ │ │ - xy: true, │ │ │ │ │ - stateName: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.stateName = {}; │ │ │ │ │ - this.stateName[OpenLayers.State.INSERT] = "wfs:Insert"; │ │ │ │ │ - this.stateName[OpenLayers.State.UPDATE] = "wfs:Update"; │ │ │ │ │ - this.stateName[OpenLayers.State.DELETE] = "wfs:Delete"; │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ - getSrsName: function(feature, options) { │ │ │ │ │ - var srsName = options && options.srsName; │ │ │ │ │ - if (!srsName) { │ │ │ │ │ - if (feature && feature.layer) { │ │ │ │ │ - srsName = feature.layer.projection.getCode() │ │ │ │ │ - } else { │ │ │ │ │ - srsName = this.srsName │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return srsName │ │ │ │ │ - }, │ │ │ │ │ - read: function(data, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, { │ │ │ │ │ - output: "features" │ │ │ │ │ - }); │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement │ │ │ │ │ - } │ │ │ │ │ - var obj = {}; │ │ │ │ │ - if (data) { │ │ │ │ │ - this.readNode(data, obj, true) │ │ │ │ │ - } │ │ │ │ │ - if (obj.features && options.output === "features") { │ │ │ │ │ - obj = obj.features │ │ │ │ │ - } │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - readers: { │ │ │ │ │ - wfs: { │ │ │ │ │ - FeatureCollection: function(node, obj) { │ │ │ │ │ - obj.features = []; │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - write: function(features, options) { │ │ │ │ │ - var node = this.writeNode("wfs:Transaction", { │ │ │ │ │ - features: features, │ │ │ │ │ - options: options │ │ │ │ │ - }); │ │ │ │ │ - var value = this.schemaLocationAttr(); │ │ │ │ │ - if (value) { │ │ │ │ │ - this.setAttributeNS(node, this.namespaces["xsi"], "xsi:schemaLocation", value) │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [node]) │ │ │ │ │ - }, │ │ │ │ │ - writers: { │ │ │ │ │ - wfs: { │ │ │ │ │ - GetFeature: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("wfs:GetFeature", { │ │ │ │ │ - attributes: { │ │ │ │ │ - service: "WFS", │ │ │ │ │ - version: this.version, │ │ │ │ │ - handle: options && options.handle, │ │ │ │ │ - outputFormat: options && options.outputFormat, │ │ │ │ │ - maxFeatures: options && options.maxFeatures, │ │ │ │ │ - "xsi:schemaLocation": this.schemaLocationAttr(options) │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (typeof this.featureType == "string") { │ │ │ │ │ - this.writeNode("Query", options, node) │ │ │ │ │ - } else { │ │ │ │ │ - for (var i = 0, len = this.featureType.length; i < len; i++) { │ │ │ │ │ - options.featureType = this.featureType[i]; │ │ │ │ │ - this.writeNode("Query", options, node) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - Transaction: function(obj) { │ │ │ │ │ - obj = obj || {}; │ │ │ │ │ - var options = obj.options || {}; │ │ │ │ │ - var node = this.createElementNSPlus("wfs:Transaction", { │ │ │ │ │ - attributes: { │ │ │ │ │ - service: "WFS", │ │ │ │ │ - version: this.version, │ │ │ │ │ - handle: options.handle │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - var i, len; │ │ │ │ │ - var features = obj.features; │ │ │ │ │ - if (features) { │ │ │ │ │ - if (options.multi === true) { │ │ │ │ │ - OpenLayers.Util.extend(this.geometryTypes, { │ │ │ │ │ - "OpenLayers.Geometry.Point": "MultiPoint", │ │ │ │ │ - "OpenLayers.Geometry.LineString": this.multiCurve === true ? "MultiCurve" : "MultiLineString", │ │ │ │ │ - "OpenLayers.Geometry.Polygon": this.multiSurface === true ? "MultiSurface" : "MultiPolygon" │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - var name, feature; │ │ │ │ │ - for (i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - name = this.stateName[feature.state]; │ │ │ │ │ - if (name) { │ │ │ │ │ - this.writeNode(name, { │ │ │ │ │ - feature: feature, │ │ │ │ │ - options: options │ │ │ │ │ - }, node) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (options.multi === true) { │ │ │ │ │ - this.setGeometryTypes() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (options.nativeElements) { │ │ │ │ │ - for (i = 0, len = options.nativeElements.length; i < len; ++i) { │ │ │ │ │ - this.writeNode("wfs:Native", options.nativeElements[i], node) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - Native: function(nativeElement) { │ │ │ │ │ - var node = this.createElementNSPlus("wfs:Native", { │ │ │ │ │ - attributes: { │ │ │ │ │ - vendorId: nativeElement.vendorId, │ │ │ │ │ - safeToIgnore: nativeElement.safeToIgnore │ │ │ │ │ - }, │ │ │ │ │ - value: nativeElement.value │ │ │ │ │ - }); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - Insert: function(obj) { │ │ │ │ │ - var feature = obj.feature; │ │ │ │ │ - var options = obj.options; │ │ │ │ │ - var node = this.createElementNSPlus("wfs:Insert", { │ │ │ │ │ - attributes: { │ │ │ │ │ - handle: options && options.handle │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - this.srsName = this.getSrsName(feature); │ │ │ │ │ - this.writeNode("feature:_typeName", feature, node); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - Update: function(obj) { │ │ │ │ │ - var feature = obj.feature; │ │ │ │ │ - var options = obj.options; │ │ │ │ │ - var node = this.createElementNSPlus("wfs:Update", { │ │ │ │ │ - attributes: { │ │ │ │ │ - handle: options && options.handle, │ │ │ │ │ - typeName: (this.featureNS ? this.featurePrefix + ":" : "") + this.featureType │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (this.featureNS) { │ │ │ │ │ - node.setAttribute("xmlns:" + this.featurePrefix, this.featureNS) │ │ │ │ │ - } │ │ │ │ │ - var modified = feature.modified; │ │ │ │ │ - if (this.geometryName !== null && (!modified || modified.geometry !== undefined)) { │ │ │ │ │ - this.srsName = this.getSrsName(feature); │ │ │ │ │ - this.writeNode("Property", { │ │ │ │ │ - name: this.geometryName, │ │ │ │ │ - value: feature.geometry │ │ │ │ │ - }, node) │ │ │ │ │ - } │ │ │ │ │ - for (var key in feature.attributes) { │ │ │ │ │ - if (feature.attributes[key] !== undefined && (!modified || !modified.attributes || modified.attributes && modified.attributes[key] !== undefined)) { │ │ │ │ │ - this.writeNode("Property", { │ │ │ │ │ - name: key, │ │ │ │ │ - value: feature.attributes[key] │ │ │ │ │ - }, node) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.writeNode("ogc:Filter", new OpenLayers.Filter.FeatureId({ │ │ │ │ │ - fids: [feature.fid] │ │ │ │ │ - }), node); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - Property: function(obj) { │ │ │ │ │ - var node = this.createElementNSPlus("wfs:Property"); │ │ │ │ │ - this.writeNode("Name", obj.name, node); │ │ │ │ │ - if (obj.value !== null) { │ │ │ │ │ - this.writeNode("Value", obj.value, node) │ │ │ │ │ - } │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - Name: function(name) { │ │ │ │ │ - return this.createElementNSPlus("wfs:Name", { │ │ │ │ │ - value: name │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - Value: function(obj) { │ │ │ │ │ - var node; │ │ │ │ │ - if (obj instanceof OpenLayers.Geometry) { │ │ │ │ │ - node = this.createElementNSPlus("wfs:Value"); │ │ │ │ │ - var geom = this.writeNode("feature:_geometry", obj).firstChild; │ │ │ │ │ - node.appendChild(geom) │ │ │ │ │ - } else { │ │ │ │ │ - node = this.createElementNSPlus("wfs:Value", { │ │ │ │ │ - value: obj │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - Delete: function(obj) { │ │ │ │ │ - var feature = obj.feature; │ │ │ │ │ - var options = obj.options; │ │ │ │ │ - var node = this.createElementNSPlus("wfs:Delete", { │ │ │ │ │ - attributes: { │ │ │ │ │ - handle: options && options.handle, │ │ │ │ │ - typeName: (this.featureNS ? this.featurePrefix + ":" : "") + this.featureType │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (this.featureNS) { │ │ │ │ │ - node.setAttribute("xmlns:" + this.featurePrefix, this.featureNS) │ │ │ │ │ - } │ │ │ │ │ - this.writeNode("ogc:Filter", new OpenLayers.Filter.FeatureId({ │ │ │ │ │ - fids: [feature.fid] │ │ │ │ │ - }), node); │ │ │ │ │ - return node │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - schemaLocationAttr: function(options) { │ │ │ │ │ - options = OpenLayers.Util.extend({ │ │ │ │ │ - featurePrefix: this.featurePrefix, │ │ │ │ │ - schema: this.schema │ │ │ │ │ - }, options); │ │ │ │ │ - var schemaLocations = OpenLayers.Util.extend({}, this.schemaLocations); │ │ │ │ │ - if (options.schema) { │ │ │ │ │ - schemaLocations[options.featurePrefix] = options.schema │ │ │ │ │ - } │ │ │ │ │ - var parts = []; │ │ │ │ │ - var uri; │ │ │ │ │ - for (var key in schemaLocations) { │ │ │ │ │ - uri = this.namespaces[key]; │ │ │ │ │ - if (uri) { │ │ │ │ │ - parts.push(uri + " " + schemaLocations[key]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var value = parts.join(" ") || undefined; │ │ │ │ │ - return value │ │ │ │ │ - }, │ │ │ │ │ - setFilterProperty: function(filter) { │ │ │ │ │ - if (filter.filters) { │ │ │ │ │ - for (var i = 0, len = filter.filters.length; i < len; ++i) { │ │ │ │ │ - OpenLayers.Format.WFST.v1.prototype.setFilterProperty.call(this, filter.filters[i]) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - if (filter instanceof OpenLayers.Filter.Spatial && !filter.property) { │ │ │ │ │ - filter.property = this.geometryName │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFST.v1" │ │ │ │ │ +OpenLayers.Geometry.MultiPolygon = OpenLayers.Class(OpenLayers.Geometry.Collection, { │ │ │ │ │ + componentTypes: ["OpenLayers.Geometry.Polygon"], │ │ │ │ │ + CLASS_NAME: "OpenLayers.Geometry.MultiPolygon" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Format.GML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ featureNS: "http://mapserver.gis.umn.edu/mapserver", │ │ │ │ │ featurePrefix: "feature", │ │ │ │ │ featureName: "featureMember", │ │ │ │ │ layerName: "features", │ │ │ │ │ geometryName: "geometry", │ │ │ │ │ @@ -17068,48 +12524,96 @@ │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.Filter" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Filter.Function = OpenLayers.Class(OpenLayers.Filter, { │ │ │ │ │ name: null, │ │ │ │ │ params: null, │ │ │ │ │ CLASS_NAME: "OpenLayers.Filter.Function" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.Filter.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - namespaces: { │ │ │ │ │ - ogc: "http://www.opengis.net/ogc", │ │ │ │ │ - gml: "http://www.opengis.net/gml", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ - }, │ │ │ │ │ - defaultPrefix: "ogc", │ │ │ │ │ - schemaLocation: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ - read: function(data) { │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readers.ogc["Filter"].apply(this, [data, obj]); │ │ │ │ │ - return obj.filter │ │ │ │ │ - }, │ │ │ │ │ - readers: { │ │ │ │ │ - ogc: { │ │ │ │ │ - _expression: function(node) { │ │ │ │ │ - var obj, value = ""; │ │ │ │ │ - for (var child = node.firstChild; child; child = child.nextSibling) { │ │ │ │ │ - switch (child.nodeType) { │ │ │ │ │ - case 1: │ │ │ │ │ - obj = this.readNode(child); │ │ │ │ │ - if (obj.property) { │ │ │ │ │ - value += "${" + obj.property + "}" │ │ │ │ │ - } else if (obj.value !== undefined) { │ │ │ │ │ - value += obj.value │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case 3: │ │ │ │ │ - case 4: │ │ │ │ │ - value += child.nodeValue │ │ │ │ │ +OpenLayers.Date = { │ │ │ │ │ + dateRegEx: /^(?:(\d{4})(?:-(\d{2})(?:-(\d{2}))?)?)?(?:(?:T(\d{1,2}):(\d{2}):(\d{2}(?:\.\d+)?)(Z|(?:[+-]\d{1,2}(?::(\d{2}))?)))|Z)?$/, │ │ │ │ │ + toISOString: function() { │ │ │ │ │ + if ("toISOString" in Date.prototype) { │ │ │ │ │ + return function(date) { │ │ │ │ │ + return date.toISOString() │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + return function(date) { │ │ │ │ │ + var str; │ │ │ │ │ + if (isNaN(date.getTime())) { │ │ │ │ │ + str = "Invalid Date" │ │ │ │ │ + } else { │ │ │ │ │ + str = date.getUTCFullYear() + "-" + OpenLayers.Number.zeroPad(date.getUTCMonth() + 1, 2) + "-" + OpenLayers.Number.zeroPad(date.getUTCDate(), 2) + "T" + OpenLayers.Number.zeroPad(date.getUTCHours(), 2) + ":" + OpenLayers.Number.zeroPad(date.getUTCMinutes(), 2) + ":" + OpenLayers.Number.zeroPad(date.getUTCSeconds(), 2) + "." + OpenLayers.Number.zeroPad(date.getUTCMilliseconds(), 3) + "Z" │ │ │ │ │ + } │ │ │ │ │ + return str │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }(), │ │ │ │ │ + parse: function(str) { │ │ │ │ │ + var date; │ │ │ │ │ + var match = str.match(this.dateRegEx); │ │ │ │ │ + if (match && (match[1] || match[7])) { │ │ │ │ │ + var year = parseInt(match[1], 10) || 0; │ │ │ │ │ + var month = parseInt(match[2], 10) - 1 || 0; │ │ │ │ │ + var day = parseInt(match[3], 10) || 1; │ │ │ │ │ + date = new Date(Date.UTC(year, month, day)); │ │ │ │ │ + var type = match[7]; │ │ │ │ │ + if (type) { │ │ │ │ │ + var hours = parseInt(match[4], 10); │ │ │ │ │ + var minutes = parseInt(match[5], 10); │ │ │ │ │ + var secFrac = parseFloat(match[6]); │ │ │ │ │ + var seconds = secFrac | 0; │ │ │ │ │ + var milliseconds = Math.round(1e3 * (secFrac - seconds)); │ │ │ │ │ + date.setUTCHours(hours, minutes, seconds, milliseconds); │ │ │ │ │ + if (type !== "Z") { │ │ │ │ │ + var hoursOffset = parseInt(type, 10); │ │ │ │ │ + var minutesOffset = parseInt(match[8], 10) || 0; │ │ │ │ │ + var offset = -1e3 * (60 * (hoursOffset * 60) + minutesOffset * 60); │ │ │ │ │ + date = new Date(date.getTime() + offset) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + date = new Date("invalid") │ │ │ │ │ + } │ │ │ │ │ + return date │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Format.Filter.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + namespaces: { │ │ │ │ │ + ogc: "http://www.opengis.net/ogc", │ │ │ │ │ + gml: "http://www.opengis.net/gml", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ + defaultPrefix: "ogc", │ │ │ │ │ + schemaLocation: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ + read: function(data) { │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readers.ogc["Filter"].apply(this, [data, obj]); │ │ │ │ │ + return obj.filter │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + ogc: { │ │ │ │ │ + _expression: function(node) { │ │ │ │ │ + var obj, value = ""; │ │ │ │ │ + for (var child = node.firstChild; child; child = child.nextSibling) { │ │ │ │ │ + switch (child.nodeType) { │ │ │ │ │ + case 1: │ │ │ │ │ + obj = this.readNode(child); │ │ │ │ │ + if (obj.property) { │ │ │ │ │ + value += "${" + obj.property + "}" │ │ │ │ │ + } else if (obj.value !== undefined) { │ │ │ │ │ + value += obj.value │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case 3: │ │ │ │ │ + case 4: │ │ │ │ │ + value += child.nodeValue │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ return value │ │ │ │ │ }, │ │ │ │ │ Filter: function(node, parent) { │ │ │ │ │ var obj = { │ │ │ │ │ fids: [], │ │ │ │ │ @@ -17598,8 +13102,4504 @@ │ │ │ │ │ ogc: OpenLayers.Format.Filter.v1_0_0.prototype.writers["ogc"] │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.WFST.v1_0_0" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Protocol.WFS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ version: "1.0.0", │ │ │ │ │ CLASS_NAME: "OpenLayers.Protocol.WFS.v1_0_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.JSON = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ + indent: " ", │ │ │ │ │ + space: " ", │ │ │ │ │ + newline: "\n", │ │ │ │ │ + level: 0, │ │ │ │ │ + pretty: false, │ │ │ │ │ + nativeJSON: function() { │ │ │ │ │ + return !!(window.JSON && typeof JSON.parse == "function" && typeof JSON.stringify == "function") │ │ │ │ │ + }(), │ │ │ │ │ + read: function(json, filter) { │ │ │ │ │ + var object; │ │ │ │ │ + if (this.nativeJSON) { │ │ │ │ │ + object = JSON.parse(json, filter) │ │ │ │ │ + } else try { │ │ │ │ │ + if (/^[\],:{}\s]*$/.test(json.replace(/\\["\\\/bfnrtu]/g, "@").replace(/"[^"\\\n\r]*"|true|false|null|-?\d+(?:\.\d*)?(?:[eE][+\-]?\d+)?/g, "]").replace(/(?:^|:|,)(?:\s*\[)+/g, ""))) { │ │ │ │ │ + object = eval("(" + json + ")"); │ │ │ │ │ + if (typeof filter === "function") { │ │ │ │ │ + function walk(k, v) { │ │ │ │ │ + if (v && typeof v === "object") { │ │ │ │ │ + for (var i in v) { │ │ │ │ │ + if (v.hasOwnProperty(i)) { │ │ │ │ │ + v[i] = walk(i, v[i]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return filter(k, v) │ │ │ │ │ + } │ │ │ │ │ + object = walk("", object) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + if (this.keepData) { │ │ │ │ │ + this.data = object │ │ │ │ │ + } │ │ │ │ │ + return object │ │ │ │ │ + }, │ │ │ │ │ + write: function(value, pretty) { │ │ │ │ │ + this.pretty = !!pretty; │ │ │ │ │ + var json = null; │ │ │ │ │ + var type = typeof value; │ │ │ │ │ + if (this.serialize[type]) { │ │ │ │ │ + try { │ │ │ │ │ + json = !this.pretty && this.nativeJSON ? JSON.stringify(value) : this.serialize[type].apply(this, [value]) │ │ │ │ │ + } catch (err) { │ │ │ │ │ + OpenLayers.Console.error("Trouble serializing: " + err) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return json │ │ │ │ │ + }, │ │ │ │ │ + writeIndent: function() { │ │ │ │ │ + var pieces = []; │ │ │ │ │ + if (this.pretty) { │ │ │ │ │ + for (var i = 0; i < this.level; ++i) { │ │ │ │ │ + pieces.push(this.indent) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return pieces.join("") │ │ │ │ │ + }, │ │ │ │ │ + writeNewline: function() { │ │ │ │ │ + return this.pretty ? this.newline : "" │ │ │ │ │ + }, │ │ │ │ │ + writeSpace: function() { │ │ │ │ │ + return this.pretty ? this.space : "" │ │ │ │ │ + }, │ │ │ │ │ + serialize: { │ │ │ │ │ + object: function(object) { │ │ │ │ │ + if (object == null) { │ │ │ │ │ + return "null" │ │ │ │ │ + } │ │ │ │ │ + if (object.constructor == Date) { │ │ │ │ │ + return this.serialize.date.apply(this, [object]) │ │ │ │ │ + } │ │ │ │ │ + if (object.constructor == Array) { │ │ │ │ │ + return this.serialize.array.apply(this, [object]) │ │ │ │ │ + } │ │ │ │ │ + var pieces = ["{"]; │ │ │ │ │ + this.level += 1; │ │ │ │ │ + var key, keyJSON, valueJSON; │ │ │ │ │ + var addComma = false; │ │ │ │ │ + for (key in object) { │ │ │ │ │ + if (object.hasOwnProperty(key)) { │ │ │ │ │ + keyJSON = OpenLayers.Format.JSON.prototype.write.apply(this, [key, this.pretty]); │ │ │ │ │ + valueJSON = OpenLayers.Format.JSON.prototype.write.apply(this, [object[key], this.pretty]); │ │ │ │ │ + if (keyJSON != null && valueJSON != null) { │ │ │ │ │ + if (addComma) { │ │ │ │ │ + pieces.push(",") │ │ │ │ │ + } │ │ │ │ │ + pieces.push(this.writeNewline(), this.writeIndent(), keyJSON, ":", this.writeSpace(), valueJSON); │ │ │ │ │ + addComma = true │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.level -= 1; │ │ │ │ │ + pieces.push(this.writeNewline(), this.writeIndent(), "}"); │ │ │ │ │ + return pieces.join("") │ │ │ │ │ + }, │ │ │ │ │ + array: function(array) { │ │ │ │ │ + var json; │ │ │ │ │ + var pieces = ["["]; │ │ │ │ │ + this.level += 1; │ │ │ │ │ + for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ + json = OpenLayers.Format.JSON.prototype.write.apply(this, [array[i], this.pretty]); │ │ │ │ │ + if (json != null) { │ │ │ │ │ + if (i > 0) { │ │ │ │ │ + pieces.push(",") │ │ │ │ │ + } │ │ │ │ │ + pieces.push(this.writeNewline(), this.writeIndent(), json) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.level -= 1; │ │ │ │ │ + pieces.push(this.writeNewline(), this.writeIndent(), "]"); │ │ │ │ │ + return pieces.join("") │ │ │ │ │ + }, │ │ │ │ │ + string: function(string) { │ │ │ │ │ + var m = { │ │ │ │ │ + "\b": "\\b", │ │ │ │ │ + "\t": "\\t", │ │ │ │ │ + "\n": "\\n", │ │ │ │ │ + "\f": "\\f", │ │ │ │ │ + "\r": "\\r", │ │ │ │ │ + '"': '\\"', │ │ │ │ │ + "\\": "\\\\" │ │ │ │ │ + }; │ │ │ │ │ + if (/["\\\x00-\x1f]/.test(string)) { │ │ │ │ │ + return '"' + string.replace(/([\x00-\x1f\\"])/g, function(a, b) { │ │ │ │ │ + var c = m[b]; │ │ │ │ │ + if (c) { │ │ │ │ │ + return c │ │ │ │ │ + } │ │ │ │ │ + c = b.charCodeAt(); │ │ │ │ │ + return "\\u00" + Math.floor(c / 16).toString(16) + (c % 16).toString(16) │ │ │ │ │ + }) + '"' │ │ │ │ │ + } │ │ │ │ │ + return '"' + string + '"' │ │ │ │ │ + }, │ │ │ │ │ + number: function(number) { │ │ │ │ │ + return isFinite(number) ? String(number) : "null" │ │ │ │ │ + }, │ │ │ │ │ + boolean: function(bool) { │ │ │ │ │ + return String(bool) │ │ │ │ │ + }, │ │ │ │ │ + date: function(date) { │ │ │ │ │ + function format(number) { │ │ │ │ │ + return number < 10 ? "0" + number : number │ │ │ │ │ + } │ │ │ │ │ + return '"' + date.getFullYear() + "-" + format(date.getMonth() + 1) + "-" + format(date.getDate()) + "T" + format(date.getHours()) + ":" + format(date.getMinutes()) + ":" + format(date.getSeconds()) + '"' │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.JSON" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.GeoJSON = OpenLayers.Class(OpenLayers.Format.JSON, { │ │ │ │ │ + ignoreExtraDims: false, │ │ │ │ │ + read: function(json, type, filter) { │ │ │ │ │ + type = type ? type : "FeatureCollection"; │ │ │ │ │ + var results = null; │ │ │ │ │ + var obj = null; │ │ │ │ │ + if (typeof json == "string") { │ │ │ │ │ + obj = OpenLayers.Format.JSON.prototype.read.apply(this, [json, filter]) │ │ │ │ │ + } else { │ │ │ │ │ + obj = json │ │ │ │ │ + } │ │ │ │ │ + if (!obj) { │ │ │ │ │ + OpenLayers.Console.error("Bad JSON: " + json) │ │ │ │ │ + } else if (typeof obj.type != "string") { │ │ │ │ │ + OpenLayers.Console.error("Bad GeoJSON - no type: " + json) │ │ │ │ │ + } else if (this.isValidType(obj, type)) { │ │ │ │ │ + switch (type) { │ │ │ │ │ + case "Geometry": │ │ │ │ │ + try { │ │ │ │ │ + results = this.parseGeometry(obj) │ │ │ │ │ + } catch (err) { │ │ │ │ │ + OpenLayers.Console.error(err) │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "Feature": │ │ │ │ │ + try { │ │ │ │ │ + results = this.parseFeature(obj); │ │ │ │ │ + results.type = "Feature" │ │ │ │ │ + } catch (err) { │ │ │ │ │ + OpenLayers.Console.error(err) │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "FeatureCollection": │ │ │ │ │ + results = []; │ │ │ │ │ + switch (obj.type) { │ │ │ │ │ + case "Feature": │ │ │ │ │ + try { │ │ │ │ │ + results.push(this.parseFeature(obj)) │ │ │ │ │ + } catch (err) { │ │ │ │ │ + results = null; │ │ │ │ │ + OpenLayers.Console.error(err) │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "FeatureCollection": │ │ │ │ │ + for (var i = 0, len = obj.features.length; i < len; ++i) { │ │ │ │ │ + try { │ │ │ │ │ + results.push(this.parseFeature(obj.features[i])) │ │ │ │ │ + } catch (err) { │ │ │ │ │ + results = null; │ │ │ │ │ + OpenLayers.Console.error(err) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + try { │ │ │ │ │ + var geom = this.parseGeometry(obj); │ │ │ │ │ + results.push(new OpenLayers.Feature.Vector(geom)) │ │ │ │ │ + } catch (err) { │ │ │ │ │ + results = null; │ │ │ │ │ + OpenLayers.Console.error(err) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return results │ │ │ │ │ + }, │ │ │ │ │ + isValidType: function(obj, type) { │ │ │ │ │ + var valid = false; │ │ │ │ │ + switch (type) { │ │ │ │ │ + case "Geometry": │ │ │ │ │ + if (OpenLayers.Util.indexOf(["Point", "MultiPoint", "LineString", "MultiLineString", "Polygon", "MultiPolygon", "Box", "GeometryCollection"], obj.type) == -1) { │ │ │ │ │ + OpenLayers.Console.error("Unsupported geometry type: " + obj.type) │ │ │ │ │ + } else { │ │ │ │ │ + valid = true │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "FeatureCollection": │ │ │ │ │ + valid = true; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + if (obj.type == type) { │ │ │ │ │ + valid = true │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Console.error("Cannot convert types from " + obj.type + " to " + type) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return valid │ │ │ │ │ + }, │ │ │ │ │ + parseFeature: function(obj) { │ │ │ │ │ + var feature, geometry, attributes, bbox; │ │ │ │ │ + attributes = obj.properties ? obj.properties : {}; │ │ │ │ │ + bbox = obj.geometry && obj.geometry.bbox || obj.bbox; │ │ │ │ │ + try { │ │ │ │ │ + geometry = this.parseGeometry(obj.geometry) │ │ │ │ │ + } catch (err) { │ │ │ │ │ + throw err │ │ │ │ │ + } │ │ │ │ │ + feature = new OpenLayers.Feature.Vector(geometry, attributes); │ │ │ │ │ + if (bbox) { │ │ │ │ │ + feature.bounds = OpenLayers.Bounds.fromArray(bbox) │ │ │ │ │ + } │ │ │ │ │ + if (obj.id) { │ │ │ │ │ + feature.fid = obj.id │ │ │ │ │ + } │ │ │ │ │ + return feature │ │ │ │ │ + }, │ │ │ │ │ + parseGeometry: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ + var geometry, collection = false; │ │ │ │ │ + if (obj.type == "GeometryCollection") { │ │ │ │ │ + if (!OpenLayers.Util.isArray(obj.geometries)) { │ │ │ │ │ + throw "GeometryCollection must have geometries array: " + obj │ │ │ │ │ + } │ │ │ │ │ + var numGeom = obj.geometries.length; │ │ │ │ │ + var components = new Array(numGeom); │ │ │ │ │ + for (var i = 0; i < numGeom; ++i) { │ │ │ │ │ + components[i] = this.parseGeometry.apply(this, [obj.geometries[i]]) │ │ │ │ │ + } │ │ │ │ │ + geometry = new OpenLayers.Geometry.Collection(components); │ │ │ │ │ + collection = true │ │ │ │ │ + } else { │ │ │ │ │ + if (!OpenLayers.Util.isArray(obj.coordinates)) { │ │ │ │ │ + throw "Geometry must have coordinates array: " + obj │ │ │ │ │ + } │ │ │ │ │ + if (!this.parseCoords[obj.type.toLowerCase()]) { │ │ │ │ │ + throw "Unsupported geometry type: " + obj.type │ │ │ │ │ + } │ │ │ │ │ + try { │ │ │ │ │ + geometry = this.parseCoords[obj.type.toLowerCase()].apply(this, [obj.coordinates]) │ │ │ │ │ + } catch (err) { │ │ │ │ │ + throw err │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.internalProjection && this.externalProjection && !collection) { │ │ │ │ │ + geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ + } │ │ │ │ │ + return geometry │ │ │ │ │ + }, │ │ │ │ │ + parseCoords: { │ │ │ │ │ + point: function(array) { │ │ │ │ │ + if (this.ignoreExtraDims == false && array.length != 2) { │ │ │ │ │ + throw "Only 2D points are supported: " + array │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.Point(array[0], array[1]) │ │ │ │ │ + }, │ │ │ │ │ + multipoint: function(array) { │ │ │ │ │ + var points = []; │ │ │ │ │ + var p = null; │ │ │ │ │ + for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ + try { │ │ │ │ │ + p = this.parseCoords["point"].apply(this, [array[i]]) │ │ │ │ │ + } catch (err) { │ │ │ │ │ + throw err │ │ │ │ │ + } │ │ │ │ │ + points.push(p) │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.MultiPoint(points) │ │ │ │ │ + }, │ │ │ │ │ + linestring: function(array) { │ │ │ │ │ + var points = []; │ │ │ │ │ + var p = null; │ │ │ │ │ + for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ + try { │ │ │ │ │ + p = this.parseCoords["point"].apply(this, [array[i]]) │ │ │ │ │ + } catch (err) { │ │ │ │ │ + throw err │ │ │ │ │ + } │ │ │ │ │ + points.push(p) │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.LineString(points) │ │ │ │ │ + }, │ │ │ │ │ + multilinestring: function(array) { │ │ │ │ │ + var lines = []; │ │ │ │ │ + var l = null; │ │ │ │ │ + for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ + try { │ │ │ │ │ + l = this.parseCoords["linestring"].apply(this, [array[i]]) │ │ │ │ │ + } catch (err) { │ │ │ │ │ + throw err │ │ │ │ │ + } │ │ │ │ │ + lines.push(l) │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.MultiLineString(lines) │ │ │ │ │ + }, │ │ │ │ │ + polygon: function(array) { │ │ │ │ │ + var rings = []; │ │ │ │ │ + var r, l; │ │ │ │ │ + for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ + try { │ │ │ │ │ + l = this.parseCoords["linestring"].apply(this, [array[i]]) │ │ │ │ │ + } catch (err) { │ │ │ │ │ + throw err │ │ │ │ │ + } │ │ │ │ │ + r = new OpenLayers.Geometry.LinearRing(l.components); │ │ │ │ │ + rings.push(r) │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.Polygon(rings) │ │ │ │ │ + }, │ │ │ │ │ + multipolygon: function(array) { │ │ │ │ │ + var polys = []; │ │ │ │ │ + var p = null; │ │ │ │ │ + for (var i = 0, len = array.length; i < len; ++i) { │ │ │ │ │ + try { │ │ │ │ │ + p = this.parseCoords["polygon"].apply(this, [array[i]]) │ │ │ │ │ + } catch (err) { │ │ │ │ │ + throw err │ │ │ │ │ + } │ │ │ │ │ + polys.push(p) │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.MultiPolygon(polys) │ │ │ │ │ + }, │ │ │ │ │ + box: function(array) { │ │ │ │ │ + if (array.length != 2) { │ │ │ │ │ + throw "GeoJSON box coordinates must have 2 elements" │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.Polygon([new OpenLayers.Geometry.LinearRing([new OpenLayers.Geometry.Point(array[0][0], array[0][1]), new OpenLayers.Geometry.Point(array[1][0], array[0][1]), new OpenLayers.Geometry.Point(array[1][0], array[1][1]), new OpenLayers.Geometry.Point(array[0][0], array[1][1]), new OpenLayers.Geometry.Point(array[0][0], array[0][1])])]) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + write: function(obj, pretty) { │ │ │ │ │ + var geojson = { │ │ │ │ │ + type: null │ │ │ │ │ + }; │ │ │ │ │ + if (OpenLayers.Util.isArray(obj)) { │ │ │ │ │ + geojson.type = "FeatureCollection"; │ │ │ │ │ + var numFeatures = obj.length; │ │ │ │ │ + geojson.features = new Array(numFeatures); │ │ │ │ │ + for (var i = 0; i < numFeatures; ++i) { │ │ │ │ │ + var element = obj[i]; │ │ │ │ │ + if (!element instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ + var msg = "FeatureCollection only supports collections " + "of features: " + element; │ │ │ │ │ + throw msg │ │ │ │ │ + } │ │ │ │ │ + geojson.features[i] = this.extract.feature.apply(this, [element]) │ │ │ │ │ + } │ │ │ │ │ + } else if (obj.CLASS_NAME.indexOf("OpenLayers.Geometry") == 0) { │ │ │ │ │ + geojson = this.extract.geometry.apply(this, [obj]) │ │ │ │ │ + } else if (obj instanceof OpenLayers.Feature.Vector) { │ │ │ │ │ + geojson = this.extract.feature.apply(this, [obj]); │ │ │ │ │ + if (obj.layer && obj.layer.projection) { │ │ │ │ │ + geojson.crs = this.createCRSObject(obj) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Format.JSON.prototype.write.apply(this, [geojson, pretty]) │ │ │ │ │ + }, │ │ │ │ │ + createCRSObject: function(object) { │ │ │ │ │ + var proj = object.layer.projection.toString(); │ │ │ │ │ + var crs = {}; │ │ │ │ │ + if (proj.match(/epsg:/i)) { │ │ │ │ │ + var code = parseInt(proj.substring(proj.indexOf(":") + 1)); │ │ │ │ │ + if (code == 4326) { │ │ │ │ │ + crs = { │ │ │ │ │ + type: "name", │ │ │ │ │ + properties: { │ │ │ │ │ + name: "urn:ogc:def:crs:OGC:1.3:CRS84" │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + crs = { │ │ │ │ │ + type: "name", │ │ │ │ │ + properties: { │ │ │ │ │ + name: "EPSG:" + code │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return crs │ │ │ │ │ + }, │ │ │ │ │ + extract: { │ │ │ │ │ + feature: function(feature) { │ │ │ │ │ + var geom = this.extract.geometry.apply(this, [feature.geometry]); │ │ │ │ │ + var json = { │ │ │ │ │ + type: "Feature", │ │ │ │ │ + properties: feature.attributes, │ │ │ │ │ + geometry: geom │ │ │ │ │ + }; │ │ │ │ │ + if (feature.fid != null) { │ │ │ │ │ + json.id = feature.fid │ │ │ │ │ + } │ │ │ │ │ + return json │ │ │ │ │ + }, │ │ │ │ │ + geometry: function(geometry) { │ │ │ │ │ + if (geometry == null) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry = geometry.clone(); │ │ │ │ │ + geometry.transform(this.internalProjection, this.externalProjection) │ │ │ │ │ + } │ │ │ │ │ + var geometryType = geometry.CLASS_NAME.split(".")[2]; │ │ │ │ │ + var data = this.extract[geometryType.toLowerCase()].apply(this, [geometry]); │ │ │ │ │ + var json; │ │ │ │ │ + if (geometryType == "Collection") { │ │ │ │ │ + json = { │ │ │ │ │ + type: "GeometryCollection", │ │ │ │ │ + geometries: data │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + json = { │ │ │ │ │ + type: geometryType, │ │ │ │ │ + coordinates: data │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return json │ │ │ │ │ + }, │ │ │ │ │ + point: function(point) { │ │ │ │ │ + return [point.x, point.y] │ │ │ │ │ + }, │ │ │ │ │ + multipoint: function(multipoint) { │ │ │ │ │ + var array = []; │ │ │ │ │ + for (var i = 0, len = multipoint.components.length; i < len; ++i) { │ │ │ │ │ + array.push(this.extract.point.apply(this, [multipoint.components[i]])) │ │ │ │ │ + } │ │ │ │ │ + return array │ │ │ │ │ + }, │ │ │ │ │ + linestring: function(linestring) { │ │ │ │ │ + var array = []; │ │ │ │ │ + for (var i = 0, len = linestring.components.length; i < len; ++i) { │ │ │ │ │ + array.push(this.extract.point.apply(this, [linestring.components[i]])) │ │ │ │ │ + } │ │ │ │ │ + return array │ │ │ │ │ + }, │ │ │ │ │ + multilinestring: function(multilinestring) { │ │ │ │ │ + var array = []; │ │ │ │ │ + for (var i = 0, len = multilinestring.components.length; i < len; ++i) { │ │ │ │ │ + array.push(this.extract.linestring.apply(this, [multilinestring.components[i]])) │ │ │ │ │ + } │ │ │ │ │ + return array │ │ │ │ │ + }, │ │ │ │ │ + polygon: function(polygon) { │ │ │ │ │ + var array = []; │ │ │ │ │ + for (var i = 0, len = polygon.components.length; i < len; ++i) { │ │ │ │ │ + array.push(this.extract.linestring.apply(this, [polygon.components[i]])) │ │ │ │ │ + } │ │ │ │ │ + return array │ │ │ │ │ + }, │ │ │ │ │ + multipolygon: function(multipolygon) { │ │ │ │ │ + var array = []; │ │ │ │ │ + for (var i = 0, len = multipolygon.components.length; i < len; ++i) { │ │ │ │ │ + array.push(this.extract.polygon.apply(this, [multipolygon.components[i]])) │ │ │ │ │ + } │ │ │ │ │ + return array │ │ │ │ │ + }, │ │ │ │ │ + collection: function(collection) { │ │ │ │ │ + var len = collection.components.length; │ │ │ │ │ + var array = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + array[i] = this.extract.geometry.apply(this, [collection.components[i]]) │ │ │ │ │ + } │ │ │ │ │ + return array │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.GeoJSON" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.KML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + namespaces: { │ │ │ │ │ + kml: "http://www.opengis.net/kml/2.2", │ │ │ │ │ + gx: "http://www.google.com/kml/ext/2.2" │ │ │ │ │ + }, │ │ │ │ │ + kmlns: "http://earth.google.com/kml/2.0", │ │ │ │ │ + placemarksDesc: "No description available", │ │ │ │ │ + foldersName: "OpenLayers export", │ │ │ │ │ + foldersDesc: "Exported on " + new Date, │ │ │ │ │ + extractAttributes: true, │ │ │ │ │ + kvpAttributes: false, │ │ │ │ │ + extractStyles: false, │ │ │ │ │ + extractTracks: false, │ │ │ │ │ + trackAttributes: null, │ │ │ │ │ + internalns: null, │ │ │ │ │ + features: null, │ │ │ │ │ + styles: null, │ │ │ │ │ + styleBaseUrl: "", │ │ │ │ │ + fetched: null, │ │ │ │ │ + maxDepth: 0, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.regExes = { │ │ │ │ │ + trimSpace: /^\s*|\s*$/g, │ │ │ │ │ + removeSpace: /\s*/g, │ │ │ │ │ + splitSpace: /\s+/, │ │ │ │ │ + trimComma: /\s*,\s*/g, │ │ │ │ │ + kmlColor: /(\w{2})(\w{2})(\w{2})(\w{2})/, │ │ │ │ │ + kmlIconPalette: /root:\/\/icons\/palette-(\d+)(\.\w+)/, │ │ │ │ │ + straightBracket: /\$\[(.*?)\]/g │ │ │ │ │ + }; │ │ │ │ │ + this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ + read: function(data) { │ │ │ │ │ + this.features = []; │ │ │ │ │ + this.styles = {}; │ │ │ │ │ + this.fetched = {}; │ │ │ │ │ + var options = { │ │ │ │ │ + depth: 0, │ │ │ │ │ + styleBaseUrl: this.styleBaseUrl │ │ │ │ │ + }; │ │ │ │ │ + return this.parseData(data, options) │ │ │ │ │ + }, │ │ │ │ │ + parseData: function(data, options) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + var types = ["Link", "NetworkLink", "Style", "StyleMap", "Placemark"]; │ │ │ │ │ + for (var i = 0, len = types.length; i < len; ++i) { │ │ │ │ │ + var type = types[i]; │ │ │ │ │ + var nodes = this.getElementsByTagNameNS(data, "*", type); │ │ │ │ │ + if (nodes.length == 0) { │ │ │ │ │ + continue │ │ │ │ │ + } │ │ │ │ │ + switch (type.toLowerCase()) { │ │ │ │ │ + case "link": │ │ │ │ │ + case "networklink": │ │ │ │ │ + this.parseLinks(nodes, options); │ │ │ │ │ + break; │ │ │ │ │ + case "style": │ │ │ │ │ + if (this.extractStyles) { │ │ │ │ │ + this.parseStyles(nodes, options) │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "stylemap": │ │ │ │ │ + if (this.extractStyles) { │ │ │ │ │ + this.parseStyleMaps(nodes, options) │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "placemark": │ │ │ │ │ + this.parseFeatures(nodes, options); │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return this.features │ │ │ │ │ + }, │ │ │ │ │ + parseLinks: function(nodes, options) { │ │ │ │ │ + if (options.depth >= this.maxDepth) { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + var newOptions = OpenLayers.Util.extend({}, options); │ │ │ │ │ + newOptions.depth++; │ │ │ │ │ + for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ + var href = this.parseProperty(nodes[i], "*", "href"); │ │ │ │ │ + if (href && !this.fetched[href]) { │ │ │ │ │ + this.fetched[href] = true; │ │ │ │ │ + var data = this.fetchLink(href); │ │ │ │ │ + if (data) { │ │ │ │ │ + this.parseData(data, newOptions) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + fetchLink: function(href) { │ │ │ │ │ + var request = OpenLayers.Request.GET({ │ │ │ │ │ + url: href, │ │ │ │ │ + async: false │ │ │ │ │ + }); │ │ │ │ │ + if (request) { │ │ │ │ │ + return request.responseText │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + parseStyles: function(nodes, options) { │ │ │ │ │ + for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ + var style = this.parseStyle(nodes[i]); │ │ │ │ │ + if (style) { │ │ │ │ │ + var styleName = (options.styleBaseUrl || "") + "#" + style.id; │ │ │ │ │ + this.styles[styleName] = style │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + parseKmlColor: function(kmlColor) { │ │ │ │ │ + var color = null; │ │ │ │ │ + if (kmlColor) { │ │ │ │ │ + var matches = kmlColor.match(this.regExes.kmlColor); │ │ │ │ │ + if (matches) { │ │ │ │ │ + color = { │ │ │ │ │ + color: "#" + matches[4] + matches[3] + matches[2], │ │ │ │ │ + opacity: parseInt(matches[1], 16) / 255 │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return color │ │ │ │ │ + }, │ │ │ │ │ + parseStyle: function(node) { │ │ │ │ │ + var style = {}; │ │ │ │ │ + var types = ["LineStyle", "PolyStyle", "IconStyle", "BalloonStyle", "LabelStyle"]; │ │ │ │ │ + var type, styleTypeNode, nodeList, geometry, parser; │ │ │ │ │ + for (var i = 0, len = types.length; i < len; ++i) { │ │ │ │ │ + type = types[i]; │ │ │ │ │ + styleTypeNode = this.getElementsByTagNameNS(node, "*", type)[0]; │ │ │ │ │ + if (!styleTypeNode) { │ │ │ │ │ + continue │ │ │ │ │ + } │ │ │ │ │ + switch (type.toLowerCase()) { │ │ │ │ │ + case "linestyle": │ │ │ │ │ + var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ + var color = this.parseKmlColor(kmlColor); │ │ │ │ │ + if (color) { │ │ │ │ │ + style["strokeColor"] = color.color; │ │ │ │ │ + style["strokeOpacity"] = color.opacity │ │ │ │ │ + } │ │ │ │ │ + var width = this.parseProperty(styleTypeNode, "*", "width"); │ │ │ │ │ + if (width) { │ │ │ │ │ + style["strokeWidth"] = width │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "polystyle": │ │ │ │ │ + var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ + var color = this.parseKmlColor(kmlColor); │ │ │ │ │ + if (color) { │ │ │ │ │ + style["fillOpacity"] = color.opacity; │ │ │ │ │ + style["fillColor"] = color.color │ │ │ │ │ + } │ │ │ │ │ + var fill = this.parseProperty(styleTypeNode, "*", "fill"); │ │ │ │ │ + if (fill == "0") { │ │ │ │ │ + style["fillColor"] = "none" │ │ │ │ │ + } │ │ │ │ │ + var outline = this.parseProperty(styleTypeNode, "*", "outline"); │ │ │ │ │ + if (outline == "0") { │ │ │ │ │ + style["strokeWidth"] = "0" │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "iconstyle": │ │ │ │ │ + var scale = parseFloat(this.parseProperty(styleTypeNode, "*", "scale") || 1); │ │ │ │ │ + var width = 32 * scale; │ │ │ │ │ + var height = 32 * scale; │ │ │ │ │ + var iconNode = this.getElementsByTagNameNS(styleTypeNode, "*", "Icon")[0]; │ │ │ │ │ + if (iconNode) { │ │ │ │ │ + var href = this.parseProperty(iconNode, "*", "href"); │ │ │ │ │ + if (href) { │ │ │ │ │ + var w = this.parseProperty(iconNode, "*", "w"); │ │ │ │ │ + var h = this.parseProperty(iconNode, "*", "h"); │ │ │ │ │ + var google = "http://maps.google.com/mapfiles/kml"; │ │ │ │ │ + if (OpenLayers.String.startsWith(href, google) && !w && !h) { │ │ │ │ │ + w = 64; │ │ │ │ │ + h = 64; │ │ │ │ │ + scale = scale / 2 │ │ │ │ │ + } │ │ │ │ │ + w = w || h; │ │ │ │ │ + h = h || w; │ │ │ │ │ + if (w) { │ │ │ │ │ + width = parseInt(w) * scale │ │ │ │ │ + } │ │ │ │ │ + if (h) { │ │ │ │ │ + height = parseInt(h) * scale │ │ │ │ │ + } │ │ │ │ │ + var matches = href.match(this.regExes.kmlIconPalette); │ │ │ │ │ + if (matches) { │ │ │ │ │ + var palette = matches[1]; │ │ │ │ │ + var file_extension = matches[2]; │ │ │ │ │ + var x = this.parseProperty(iconNode, "*", "x"); │ │ │ │ │ + var y = this.parseProperty(iconNode, "*", "y"); │ │ │ │ │ + var posX = x ? x / 32 : 0; │ │ │ │ │ + var posY = y ? 7 - y / 32 : 7; │ │ │ │ │ + var pos = posY * 8 + posX; │ │ │ │ │ + href = "http://maps.google.com/mapfiles/kml/pal" + palette + "/icon" + pos + file_extension │ │ │ │ │ + } │ │ │ │ │ + style["graphicOpacity"] = 1; │ │ │ │ │ + style["externalGraphic"] = href │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var hotSpotNode = this.getElementsByTagNameNS(styleTypeNode, "*", "hotSpot")[0]; │ │ │ │ │ + if (hotSpotNode) { │ │ │ │ │ + var x = parseFloat(hotSpotNode.getAttribute("x")); │ │ │ │ │ + var y = parseFloat(hotSpotNode.getAttribute("y")); │ │ │ │ │ + var xUnits = hotSpotNode.getAttribute("xunits"); │ │ │ │ │ + if (xUnits == "pixels") { │ │ │ │ │ + style["graphicXOffset"] = -x * scale │ │ │ │ │ + } else if (xUnits == "insetPixels") { │ │ │ │ │ + style["graphicXOffset"] = -width + x * scale │ │ │ │ │ + } else if (xUnits == "fraction") { │ │ │ │ │ + style["graphicXOffset"] = -width * x │ │ │ │ │ + } │ │ │ │ │ + var yUnits = hotSpotNode.getAttribute("yunits"); │ │ │ │ │ + if (yUnits == "pixels") { │ │ │ │ │ + style["graphicYOffset"] = -height + y * scale + 1 │ │ │ │ │ + } else if (yUnits == "insetPixels") { │ │ │ │ │ + style["graphicYOffset"] = -(y * scale) + 1 │ │ │ │ │ + } else if (yUnits == "fraction") { │ │ │ │ │ + style["graphicYOffset"] = -height * (1 - y) + 1 │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + style["graphicWidth"] = width; │ │ │ │ │ + style["graphicHeight"] = height; │ │ │ │ │ + break; │ │ │ │ │ + case "balloonstyle": │ │ │ │ │ + var balloonStyle = OpenLayers.Util.getXmlNodeValue(styleTypeNode); │ │ │ │ │ + if (balloonStyle) { │ │ │ │ │ + style["balloonStyle"] = balloonStyle.replace(this.regExes.straightBracket, "${$1}") │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "labelstyle": │ │ │ │ │ + var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ + var color = this.parseKmlColor(kmlColor); │ │ │ │ │ + if (color) { │ │ │ │ │ + style["fontColor"] = color.color; │ │ │ │ │ + style["fontOpacity"] = color.opacity │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!style["strokeColor"] && style["fillColor"]) { │ │ │ │ │ + style["strokeColor"] = style["fillColor"] │ │ │ │ │ + } │ │ │ │ │ + var id = node.getAttribute("id"); │ │ │ │ │ + if (id && style) { │ │ │ │ │ + style.id = id │ │ │ │ │ + } │ │ │ │ │ + return style │ │ │ │ │ + }, │ │ │ │ │ + parseStyleMaps: function(nodes, options) { │ │ │ │ │ + for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ + var node = nodes[i]; │ │ │ │ │ + var pairs = this.getElementsByTagNameNS(node, "*", "Pair"); │ │ │ │ │ + var id = node.getAttribute("id"); │ │ │ │ │ + for (var j = 0, jlen = pairs.length; j < jlen; j++) { │ │ │ │ │ + var pair = pairs[j]; │ │ │ │ │ + var key = this.parseProperty(pair, "*", "key"); │ │ │ │ │ + var styleUrl = this.parseProperty(pair, "*", "styleUrl"); │ │ │ │ │ + if (styleUrl && key == "normal") { │ │ │ │ │ + this.styles[(options.styleBaseUrl || "") + "#" + id] = this.styles[(options.styleBaseUrl || "") + styleUrl] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + parseFeatures: function(nodes, options) { │ │ │ │ │ + var features = []; │ │ │ │ │ + for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ + var featureNode = nodes[i]; │ │ │ │ │ + var feature = this.parseFeature.apply(this, [featureNode]); │ │ │ │ │ + if (feature) { │ │ │ │ │ + if (this.extractStyles && feature.attributes && feature.attributes.styleUrl) { │ │ │ │ │ + feature.style = this.getStyle(feature.attributes.styleUrl, options) │ │ │ │ │ + } │ │ │ │ │ + if (this.extractStyles) { │ │ │ │ │ + var inlineStyleNode = this.getElementsByTagNameNS(featureNode, "*", "Style")[0]; │ │ │ │ │ + if (inlineStyleNode) { │ │ │ │ │ + var inlineStyle = this.parseStyle(inlineStyleNode); │ │ │ │ │ + if (inlineStyle) { │ │ │ │ │ + feature.style = OpenLayers.Util.extend(feature.style, inlineStyle) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.extractTracks) { │ │ │ │ │ + var tracks = this.getElementsByTagNameNS(featureNode, this.namespaces.gx, "Track"); │ │ │ │ │ + if (tracks && tracks.length > 0) { │ │ │ │ │ + var track = tracks[0]; │ │ │ │ │ + var container = { │ │ │ │ │ + features: [], │ │ │ │ │ + feature: feature │ │ │ │ │ + }; │ │ │ │ │ + this.readNode(track, container); │ │ │ │ │ + if (container.features.length > 0) { │ │ │ │ │ + features.push.apply(features, container.features) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + features.push(feature) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + throw "Bad Placemark: " + i │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.features = this.features.concat(features) │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + kml: { │ │ │ │ │ + when: function(node, container) { │ │ │ │ │ + container.whens.push(OpenLayers.Date.parse(this.getChildValue(node))) │ │ │ │ │ + }, │ │ │ │ │ + _trackPointAttribute: function(node, container) { │ │ │ │ │ + var name = node.nodeName.split(":").pop(); │ │ │ │ │ + container.attributes[name].push(this.getChildValue(node)) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + gx: { │ │ │ │ │ + Track: function(node, container) { │ │ │ │ │ + var obj = { │ │ │ │ │ + whens: [], │ │ │ │ │ + points: [], │ │ │ │ │ + angles: [] │ │ │ │ │ + }; │ │ │ │ │ + if (this.trackAttributes) { │ │ │ │ │ + var name; │ │ │ │ │ + obj.attributes = {}; │ │ │ │ │ + for (var i = 0, ii = this.trackAttributes.length; i < ii; ++i) { │ │ │ │ │ + name = this.trackAttributes[i]; │ │ │ │ │ + obj.attributes[name] = []; │ │ │ │ │ + if (!(name in this.readers.kml)) { │ │ │ │ │ + this.readers.kml[name] = this.readers.kml._trackPointAttribute │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + if (obj.whens.length !== obj.points.length) { │ │ │ │ │ + throw new Error("gx:Track with unequal number of when (" + obj.whens.length + ") and gx:coord (" + obj.points.length + ") elements.") │ │ │ │ │ + } │ │ │ │ │ + var hasAngles = obj.angles.length > 0; │ │ │ │ │ + if (hasAngles && obj.whens.length !== obj.angles.length) { │ │ │ │ │ + throw new Error("gx:Track with unequal number of when (" + obj.whens.length + ") and gx:angles (" + obj.angles.length + ") elements.") │ │ │ │ │ + } │ │ │ │ │ + var feature, point, angles; │ │ │ │ │ + for (var i = 0, ii = obj.whens.length; i < ii; ++i) { │ │ │ │ │ + feature = container.feature.clone(); │ │ │ │ │ + feature.fid = container.feature.fid || container.feature.id; │ │ │ │ │ + point = obj.points[i]; │ │ │ │ │ + feature.geometry = point; │ │ │ │ │ + if ("z" in point) { │ │ │ │ │ + feature.attributes.altitude = point.z │ │ │ │ │ + } │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + feature.geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ + } │ │ │ │ │ + if (this.trackAttributes) { │ │ │ │ │ + for (var j = 0, jj = this.trackAttributes.length; j < jj; ++j) { │ │ │ │ │ + var name = this.trackAttributes[j]; │ │ │ │ │ + feature.attributes[name] = obj.attributes[name][i] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + feature.attributes.when = obj.whens[i]; │ │ │ │ │ + feature.attributes.trackId = container.feature.id; │ │ │ │ │ + if (hasAngles) { │ │ │ │ │ + angles = obj.angles[i]; │ │ │ │ │ + feature.attributes.heading = parseFloat(angles[0]); │ │ │ │ │ + feature.attributes.tilt = parseFloat(angles[1]); │ │ │ │ │ + feature.attributes.roll = parseFloat(angles[2]) │ │ │ │ │ + } │ │ │ │ │ + container.features.push(feature) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + coord: function(node, container) { │ │ │ │ │ + var str = this.getChildValue(node); │ │ │ │ │ + var coords = str.replace(this.regExes.trimSpace, "").split(/\s+/); │ │ │ │ │ + var point = new OpenLayers.Geometry.Point(coords[0], coords[1]); │ │ │ │ │ + if (coords.length > 2) { │ │ │ │ │ + point.z = parseFloat(coords[2]) │ │ │ │ │ + } │ │ │ │ │ + container.points.push(point) │ │ │ │ │ + }, │ │ │ │ │ + angles: function(node, container) { │ │ │ │ │ + var str = this.getChildValue(node); │ │ │ │ │ + var parts = str.replace(this.regExes.trimSpace, "").split(/\s+/); │ │ │ │ │ + container.angles.push(parts) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + parseFeature: function(node) { │ │ │ │ │ + var order = ["MultiGeometry", "Polygon", "LineString", "Point"]; │ │ │ │ │ + var type, nodeList, geometry, parser; │ │ │ │ │ + for (var i = 0, len = order.length; i < len; ++i) { │ │ │ │ │ + type = order[i]; │ │ │ │ │ + this.internalns = node.namespaceURI ? node.namespaceURI : this.kmlns; │ │ │ │ │ + nodeList = this.getElementsByTagNameNS(node, this.internalns, type); │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + var parser = this.parseGeometry[type.toLowerCase()]; │ │ │ │ │ + if (parser) { │ │ │ │ │ + geometry = parser.apply(this, [nodeList[0]]); │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + throw new TypeError("Unsupported geometry type: " + type) │ │ │ │ │ + } │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var attributes; │ │ │ │ │ + if (this.extractAttributes) { │ │ │ │ │ + attributes = this.parseAttributes(node) │ │ │ │ │ + } │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry, attributes); │ │ │ │ │ + var fid = node.getAttribute("id") || node.getAttribute("name"); │ │ │ │ │ + if (fid != null) { │ │ │ │ │ + feature.fid = fid │ │ │ │ │ + } │ │ │ │ │ + return feature │ │ │ │ │ + }, │ │ │ │ │ + getStyle: function(styleUrl, options) { │ │ │ │ │ + var styleBaseUrl = OpenLayers.Util.removeTail(styleUrl); │ │ │ │ │ + var newOptions = OpenLayers.Util.extend({}, options); │ │ │ │ │ + newOptions.depth++; │ │ │ │ │ + newOptions.styleBaseUrl = styleBaseUrl; │ │ │ │ │ + if (!this.styles[styleUrl] && !OpenLayers.String.startsWith(styleUrl, "#") && newOptions.depth <= this.maxDepth && !this.fetched[styleBaseUrl]) { │ │ │ │ │ + var data = this.fetchLink(styleBaseUrl); │ │ │ │ │ + if (data) { │ │ │ │ │ + this.parseData(data, newOptions) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var style = OpenLayers.Util.extend({}, this.styles[styleUrl]); │ │ │ │ │ + return style │ │ │ │ │ + }, │ │ │ │ │ + parseGeometry: { │ │ │ │ │ + point: function(node) { │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, this.internalns, "coordinates"); │ │ │ │ │ + var coords = []; │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + var coordString = nodeList[0].firstChild.nodeValue; │ │ │ │ │ + coordString = coordString.replace(this.regExes.removeSpace, ""); │ │ │ │ │ + coords = coordString.split(",") │ │ │ │ │ + } │ │ │ │ │ + var point = null; │ │ │ │ │ + if (coords.length > 1) { │ │ │ │ │ + if (coords.length == 2) { │ │ │ │ │ + coords[2] = null │ │ │ │ │ + } │ │ │ │ │ + point = new OpenLayers.Geometry.Point(coords[0], coords[1], coords[2]) │ │ │ │ │ + } else { │ │ │ │ │ + throw "Bad coordinate string: " + coordString │ │ │ │ │ + } │ │ │ │ │ + return point │ │ │ │ │ + }, │ │ │ │ │ + linestring: function(node, ring) { │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, this.internalns, "coordinates"); │ │ │ │ │ + var line = null; │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + var coordString = this.getChildValue(nodeList[0]); │ │ │ │ │ + coordString = coordString.replace(this.regExes.trimSpace, ""); │ │ │ │ │ + coordString = coordString.replace(this.regExes.trimComma, ","); │ │ │ │ │ + var pointList = coordString.split(this.regExes.splitSpace); │ │ │ │ │ + var numPoints = pointList.length; │ │ │ │ │ + var points = new Array(numPoints); │ │ │ │ │ + var coords, numCoords; │ │ │ │ │ + for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ + coords = pointList[i].split(","); │ │ │ │ │ + numCoords = coords.length; │ │ │ │ │ + if (numCoords > 1) { │ │ │ │ │ + if (coords.length == 2) { │ │ │ │ │ + coords[2] = null │ │ │ │ │ + } │ │ │ │ │ + points[i] = new OpenLayers.Geometry.Point(coords[0], coords[1], coords[2]) │ │ │ │ │ + } else { │ │ │ │ │ + throw "Bad LineString point coordinates: " + pointList[i] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (numPoints) { │ │ │ │ │ + if (ring) { │ │ │ │ │ + line = new OpenLayers.Geometry.LinearRing(points) │ │ │ │ │ + } else { │ │ │ │ │ + line = new OpenLayers.Geometry.LineString(points) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + throw "Bad LineString coordinates: " + coordString │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return line │ │ │ │ │ + }, │ │ │ │ │ + polygon: function(node) { │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, this.internalns, "LinearRing"); │ │ │ │ │ + var numRings = nodeList.length; │ │ │ │ │ + var components = new Array(numRings); │ │ │ │ │ + if (numRings > 0) { │ │ │ │ │ + var ring; │ │ │ │ │ + for (var i = 0, len = nodeList.length; i < len; ++i) { │ │ │ │ │ + ring = this.parseGeometry.linestring.apply(this, [nodeList[i], true]); │ │ │ │ │ + if (ring) { │ │ │ │ │ + components[i] = ring │ │ │ │ │ + } else { │ │ │ │ │ + throw "Bad LinearRing geometry: " + i │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.Polygon(components) │ │ │ │ │ + }, │ │ │ │ │ + multigeometry: function(node) { │ │ │ │ │ + var child, parser; │ │ │ │ │ + var parts = []; │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ + child = children[i]; │ │ │ │ │ + if (child.nodeType == 1) { │ │ │ │ │ + var type = child.prefix ? child.nodeName.split(":")[1] : child.nodeName; │ │ │ │ │ + var parser = this.parseGeometry[type.toLowerCase()]; │ │ │ │ │ + if (parser) { │ │ │ │ │ + parts.push(parser.apply(this, [child])) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.Collection(parts) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + parseAttributes: function(node) { │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + var edNodes = node.getElementsByTagName("ExtendedData"); │ │ │ │ │ + if (edNodes.length) { │ │ │ │ │ + attributes = this.parseExtendedData(edNodes[0]) │ │ │ │ │ + } │ │ │ │ │ + var child, grandchildren, grandchild; │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ + child = children[i]; │ │ │ │ │ + if (child.nodeType == 1) { │ │ │ │ │ + grandchildren = child.childNodes; │ │ │ │ │ + if (grandchildren.length >= 1 && grandchildren.length <= 3) { │ │ │ │ │ + var grandchild; │ │ │ │ │ + switch (grandchildren.length) { │ │ │ │ │ + case 1: │ │ │ │ │ + grandchild = grandchildren[0]; │ │ │ │ │ + break; │ │ │ │ │ + case 2: │ │ │ │ │ + var c1 = grandchildren[0]; │ │ │ │ │ + var c2 = grandchildren[1]; │ │ │ │ │ + grandchild = c1.nodeType == 3 || c1.nodeType == 4 ? c1 : c2; │ │ │ │ │ + break; │ │ │ │ │ + case 3: │ │ │ │ │ + default: │ │ │ │ │ + grandchild = grandchildren[1]; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + if (grandchild.nodeType == 3 || grandchild.nodeType == 4) { │ │ │ │ │ + var name = child.prefix ? child.nodeName.split(":")[1] : child.nodeName; │ │ │ │ │ + var value = OpenLayers.Util.getXmlNodeValue(grandchild); │ │ │ │ │ + if (value) { │ │ │ │ │ + value = value.replace(this.regExes.trimSpace, ""); │ │ │ │ │ + attributes[name] = value │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return attributes │ │ │ │ │ + }, │ │ │ │ │ + parseExtendedData: function(node) { │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + var i, len, data, key; │ │ │ │ │ + var dataNodes = node.getElementsByTagName("Data"); │ │ │ │ │ + for (i = 0, len = dataNodes.length; i < len; i++) { │ │ │ │ │ + data = dataNodes[i]; │ │ │ │ │ + key = data.getAttribute("name"); │ │ │ │ │ + var ed = {}; │ │ │ │ │ + var valueNode = data.getElementsByTagName("value"); │ │ │ │ │ + if (valueNode.length) { │ │ │ │ │ + ed["value"] = this.getChildValue(valueNode[0]) │ │ │ │ │ + } │ │ │ │ │ + if (this.kvpAttributes) { │ │ │ │ │ + attributes[key] = ed["value"] │ │ │ │ │ + } else { │ │ │ │ │ + var nameNode = data.getElementsByTagName("displayName"); │ │ │ │ │ + if (nameNode.length) { │ │ │ │ │ + ed["displayName"] = this.getChildValue(nameNode[0]) │ │ │ │ │ + } │ │ │ │ │ + attributes[key] = ed │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var simpleDataNodes = node.getElementsByTagName("SimpleData"); │ │ │ │ │ + for (i = 0, len = simpleDataNodes.length; i < len; i++) { │ │ │ │ │ + var ed = {}; │ │ │ │ │ + data = simpleDataNodes[i]; │ │ │ │ │ + key = data.getAttribute("name"); │ │ │ │ │ + ed["value"] = this.getChildValue(data); │ │ │ │ │ + if (this.kvpAttributes) { │ │ │ │ │ + attributes[key] = ed["value"] │ │ │ │ │ + } else { │ │ │ │ │ + ed["displayName"] = key; │ │ │ │ │ + attributes[key] = ed │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return attributes │ │ │ │ │ + }, │ │ │ │ │ + parseProperty: function(xmlNode, namespace, tagName) { │ │ │ │ │ + var value; │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(xmlNode, namespace, tagName); │ │ │ │ │ + try { │ │ │ │ │ + value = OpenLayers.Util.getXmlNodeValue(nodeList[0]) │ │ │ │ │ + } catch (e) { │ │ │ │ │ + value = null │ │ │ │ │ + } │ │ │ │ │ + return value │ │ │ │ │ + }, │ │ │ │ │ + write: function(features) { │ │ │ │ │ + if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ + features = [features] │ │ │ │ │ + } │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "kml"); │ │ │ │ │ + var folder = this.createFolderXML(); │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + folder.appendChild(this.createPlacemarkXML(features[i])) │ │ │ │ │ + } │ │ │ │ │ + kml.appendChild(folder); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [kml]) │ │ │ │ │ + }, │ │ │ │ │ + createFolderXML: function() { │ │ │ │ │ + var folder = this.createElementNS(this.kmlns, "Folder"); │ │ │ │ │ + if (this.foldersName) { │ │ │ │ │ + var folderName = this.createElementNS(this.kmlns, "name"); │ │ │ │ │ + var folderNameText = this.createTextNode(this.foldersName); │ │ │ │ │ + folderName.appendChild(folderNameText); │ │ │ │ │ + folder.appendChild(folderName) │ │ │ │ │ + } │ │ │ │ │ + if (this.foldersDesc) { │ │ │ │ │ + var folderDesc = this.createElementNS(this.kmlns, "description"); │ │ │ │ │ + var folderDescText = this.createTextNode(this.foldersDesc); │ │ │ │ │ + folderDesc.appendChild(folderDescText); │ │ │ │ │ + folder.appendChild(folderDesc) │ │ │ │ │ + } │ │ │ │ │ + return folder │ │ │ │ │ + }, │ │ │ │ │ + createPlacemarkXML: function(feature) { │ │ │ │ │ + var placemarkName = this.createElementNS(this.kmlns, "name"); │ │ │ │ │ + var label = feature.style && feature.style.label ? feature.style.label : feature.id; │ │ │ │ │ + var name = feature.attributes.name || label; │ │ │ │ │ + placemarkName.appendChild(this.createTextNode(name)); │ │ │ │ │ + var placemarkDesc = this.createElementNS(this.kmlns, "description"); │ │ │ │ │ + var desc = feature.attributes.description || this.placemarksDesc; │ │ │ │ │ + placemarkDesc.appendChild(this.createTextNode(desc)); │ │ │ │ │ + var placemarkNode = this.createElementNS(this.kmlns, "Placemark"); │ │ │ │ │ + if (feature.fid != null) { │ │ │ │ │ + placemarkNode.setAttribute("id", feature.fid) │ │ │ │ │ + } │ │ │ │ │ + placemarkNode.appendChild(placemarkName); │ │ │ │ │ + placemarkNode.appendChild(placemarkDesc); │ │ │ │ │ + var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ + placemarkNode.appendChild(geometryNode); │ │ │ │ │ + if (feature.attributes) { │ │ │ │ │ + var edNode = this.buildExtendedData(feature.attributes); │ │ │ │ │ + if (edNode) { │ │ │ │ │ + placemarkNode.appendChild(edNode) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return placemarkNode │ │ │ │ │ + }, │ │ │ │ │ + buildGeometryNode: function(geometry) { │ │ │ │ │ + var className = geometry.CLASS_NAME; │ │ │ │ │ + var type = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ + var builder = this.buildGeometry[type.toLowerCase()]; │ │ │ │ │ + var node = null; │ │ │ │ │ + if (builder) { │ │ │ │ │ + node = builder.apply(this, [geometry]) │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + buildGeometry: { │ │ │ │ │ + point: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "Point"); │ │ │ │ │ + kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + return kml │ │ │ │ │ + }, │ │ │ │ │ + multipoint: function(geometry) { │ │ │ │ │ + return this.buildGeometry.collection.apply(this, [geometry]) │ │ │ │ │ + }, │ │ │ │ │ + linestring: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "LineString"); │ │ │ │ │ + kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + return kml │ │ │ │ │ + }, │ │ │ │ │ + multilinestring: function(geometry) { │ │ │ │ │ + return this.buildGeometry.collection.apply(this, [geometry]) │ │ │ │ │ + }, │ │ │ │ │ + linearring: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "LinearRing"); │ │ │ │ │ + kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + return kml │ │ │ │ │ + }, │ │ │ │ │ + polygon: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "Polygon"); │ │ │ │ │ + var rings = geometry.components; │ │ │ │ │ + var ringMember, ringGeom, type; │ │ │ │ │ + for (var i = 0, len = rings.length; i < len; ++i) { │ │ │ │ │ + type = i == 0 ? "outerBoundaryIs" : "innerBoundaryIs"; │ │ │ │ │ + ringMember = this.createElementNS(this.kmlns, type); │ │ │ │ │ + ringGeom = this.buildGeometry.linearring.apply(this, [rings[i]]); │ │ │ │ │ + ringMember.appendChild(ringGeom); │ │ │ │ │ + kml.appendChild(ringMember) │ │ │ │ │ + } │ │ │ │ │ + return kml │ │ │ │ │ + }, │ │ │ │ │ + multipolygon: function(geometry) { │ │ │ │ │ + return this.buildGeometry.collection.apply(this, [geometry]) │ │ │ │ │ + }, │ │ │ │ │ + collection: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "MultiGeometry"); │ │ │ │ │ + var child; │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; ++i) { │ │ │ │ │ + child = this.buildGeometryNode.apply(this, [geometry.components[i]]); │ │ │ │ │ + if (child) { │ │ │ │ │ + kml.appendChild(child) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return kml │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + buildCoordinatesNode: function(geometry) { │ │ │ │ │ + var coordinatesNode = this.createElementNS(this.kmlns, "coordinates"); │ │ │ │ │ + var path; │ │ │ │ │ + var points = geometry.components; │ │ │ │ │ + if (points) { │ │ │ │ │ + var point; │ │ │ │ │ + var numPoints = points.length; │ │ │ │ │ + var parts = new Array(numPoints); │ │ │ │ │ + for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ + point = points[i]; │ │ │ │ │ + parts[i] = this.buildCoordinates(point) │ │ │ │ │ + } │ │ │ │ │ + path = parts.join(" ") │ │ │ │ │ + } else { │ │ │ │ │ + path = this.buildCoordinates(geometry) │ │ │ │ │ + } │ │ │ │ │ + var txtNode = this.createTextNode(path); │ │ │ │ │ + coordinatesNode.appendChild(txtNode); │ │ │ │ │ + return coordinatesNode │ │ │ │ │ + }, │ │ │ │ │ + buildCoordinates: function(point) { │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + point = point.clone(); │ │ │ │ │ + point.transform(this.internalProjection, this.externalProjection) │ │ │ │ │ + } │ │ │ │ │ + return point.x + "," + point.y │ │ │ │ │ + }, │ │ │ │ │ + buildExtendedData: function(attributes) { │ │ │ │ │ + var extendedData = this.createElementNS(this.kmlns, "ExtendedData"); │ │ │ │ │ + for (var attributeName in attributes) { │ │ │ │ │ + if (attributes[attributeName] && attributeName != "name" && attributeName != "description" && attributeName != "styleUrl") { │ │ │ │ │ + var data = this.createElementNS(this.kmlns, "Data"); │ │ │ │ │ + data.setAttribute("name", attributeName); │ │ │ │ │ + var value = this.createElementNS(this.kmlns, "value"); │ │ │ │ │ + if (typeof attributes[attributeName] == "object") { │ │ │ │ │ + if (attributes[attributeName].value) { │ │ │ │ │ + value.appendChild(this.createTextNode(attributes[attributeName].value)) │ │ │ │ │ + } │ │ │ │ │ + if (attributes[attributeName].displayName) { │ │ │ │ │ + var displayName = this.createElementNS(this.kmlns, "displayName"); │ │ │ │ │ + displayName.appendChild(this.getXMLDoc().createCDATASection(attributes[attributeName].displayName)); │ │ │ │ │ + data.appendChild(displayName) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + value.appendChild(this.createTextNode(attributes[attributeName])) │ │ │ │ │ + } │ │ │ │ │ + data.appendChild(value); │ │ │ │ │ + extendedData.appendChild(data) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.isSimpleContent(extendedData)) { │ │ │ │ │ + return null │ │ │ │ │ + } else { │ │ │ │ │ + return extendedData │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.KML" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control = OpenLayers.Class({ │ │ │ │ │ + id: null, │ │ │ │ │ + map: null, │ │ │ │ │ + div: null, │ │ │ │ │ + type: null, │ │ │ │ │ + allowSelection: false, │ │ │ │ │ + displayClass: "", │ │ │ │ │ + title: "", │ │ │ │ │ + autoActivate: false, │ │ │ │ │ + active: null, │ │ │ │ │ + handlerOptions: null, │ │ │ │ │ + handler: null, │ │ │ │ │ + eventListeners: null, │ │ │ │ │ + events: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.displayClass = this.CLASS_NAME.replace("OpenLayers.", "ol").replace(/\./g, ""); │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.events = new OpenLayers.Events(this); │ │ │ │ │ + if (this.eventListeners instanceof Object) { │ │ │ │ │ + this.events.on(this.eventListeners) │ │ │ │ │ + } │ │ │ │ │ + if (this.id == null) { │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.events) { │ │ │ │ │ + if (this.eventListeners) { │ │ │ │ │ + this.events.un(this.eventListeners) │ │ │ │ │ + } │ │ │ │ │ + this.events.destroy(); │ │ │ │ │ + this.events = null │ │ │ │ │ + } │ │ │ │ │ + this.eventListeners = null; │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.destroy(); │ │ │ │ │ + this.handler = null │ │ │ │ │ + } │ │ │ │ │ + if (this.handlers) { │ │ │ │ │ + for (var key in this.handlers) { │ │ │ │ │ + if (this.handlers.hasOwnProperty(key) && typeof this.handlers[key].destroy == "function") { │ │ │ │ │ + this.handlers[key].destroy() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.handlers = null │ │ │ │ │ + } │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.removeControl(this); │ │ │ │ │ + this.map = null │ │ │ │ │ + } │ │ │ │ │ + this.div = null │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.map = map; │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.setMap(map) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + if (this.div == null) { │ │ │ │ │ + this.div = OpenLayers.Util.createDiv(this.id); │ │ │ │ │ + this.div.className = this.displayClass; │ │ │ │ │ + if (!this.allowSelection) { │ │ │ │ │ + this.div.className += " olControlNoSelect"; │ │ │ │ │ + this.div.setAttribute("unselectable", "on", 0); │ │ │ │ │ + this.div.onselectstart = OpenLayers.Function.False │ │ │ │ │ + } │ │ │ │ │ + if (this.title != "") { │ │ │ │ │ + this.div.title = this.title │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (px != null) { │ │ │ │ │ + this.position = px.clone() │ │ │ │ │ + } │ │ │ │ │ + this.moveTo(this.position); │ │ │ │ │ + return this.div │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + if (px != null && this.div != null) { │ │ │ │ │ + this.div.style.left = px.x + "px"; │ │ │ │ │ + this.div.style.top = px.y + "px" │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.activate() │ │ │ │ │ + } │ │ │ │ │ + this.active = true; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, this.displayClass.replace(/ /g, "") + "Active") │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("activate"); │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.deactivate() │ │ │ │ │ + } │ │ │ │ │ + this.active = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, this.displayClass.replace(/ /g, "") + "Active") │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("deactivate"); │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.TYPE_BUTTON = 1; │ │ │ │ │ +OpenLayers.Control.TYPE_TOGGLE = 2; │ │ │ │ │ +OpenLayers.Control.TYPE_TOOL = 3; │ │ │ │ │ +OpenLayers.Handler.Drag = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + started: false, │ │ │ │ │ + stopDown: true, │ │ │ │ │ + dragging: false, │ │ │ │ │ + last: null, │ │ │ │ │ + start: null, │ │ │ │ │ + lastMoveEvt: null, │ │ │ │ │ + oldOnselectstart: null, │ │ │ │ │ + interval: 0, │ │ │ │ │ + timeoutId: null, │ │ │ │ │ + documentDrag: false, │ │ │ │ │ + documentEvents: null, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (this.documentDrag === true) { │ │ │ │ │ + var me = this; │ │ │ │ │ + this._docMove = function(evt) { │ │ │ │ │ + me.mousemove({ │ │ │ │ │ + xy: { │ │ │ │ │ + x: evt.clientX, │ │ │ │ │ + y: evt.clientY │ │ │ │ │ + }, │ │ │ │ │ + element: document │ │ │ │ │ + }) │ │ │ │ │ + }; │ │ │ │ │ + this._docUp = function(evt) { │ │ │ │ │ + me.mouseup({ │ │ │ │ │ + xy: { │ │ │ │ │ + x: evt.clientX, │ │ │ │ │ + y: evt.clientY │ │ │ │ │ + } │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + dragstart: function(evt) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + if (this.checkModifiers(evt) && (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt))) { │ │ │ │ │ + this.started = true; │ │ │ │ │ + this.start = evt.xy; │ │ │ │ │ + this.last = evt.xy; │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ + this.down(evt); │ │ │ │ │ + this.callback("down", [evt.xy]); │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + if (!this.oldOnselectstart) { │ │ │ │ │ + this.oldOnselectstart = document.onselectstart ? document.onselectstart : OpenLayers.Function.True │ │ │ │ │ + } │ │ │ │ │ + document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ + propagate = !this.stopDown │ │ │ │ │ + } else { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null │ │ │ │ │ + } │ │ │ │ │ + return propagate │ │ │ │ │ + }, │ │ │ │ │ + dragmove: function(evt) { │ │ │ │ │ + this.lastMoveEvt = evt; │ │ │ │ │ + if (this.started && !this.timeoutId && (evt.xy.x != this.last.x || evt.xy.y != this.last.y)) { │ │ │ │ │ + if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ + if (evt.element === document) { │ │ │ │ │ + this.adjustXY(evt); │ │ │ │ │ + this.setEvent(evt) │ │ │ │ │ + } else { │ │ │ │ │ + this.removeDocumentEvents() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.interval > 0) { │ │ │ │ │ + this.timeoutId = setTimeout(OpenLayers.Function.bind(this.removeTimeout, this), this.interval) │ │ │ │ │ + } │ │ │ │ │ + this.dragging = true; │ │ │ │ │ + this.move(evt); │ │ │ │ │ + this.callback("move", [evt.xy]); │ │ │ │ │ + if (!this.oldOnselectstart) { │ │ │ │ │ + this.oldOnselectstart = document.onselectstart; │ │ │ │ │ + document.onselectstart = OpenLayers.Function.False │ │ │ │ │ + } │ │ │ │ │ + this.last = evt.xy │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + dragend: function(evt) { │ │ │ │ │ + if (this.started) { │ │ │ │ │ + if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ + this.adjustXY(evt); │ │ │ │ │ + this.removeDocumentEvents() │ │ │ │ │ + } │ │ │ │ │ + var dragged = this.start != this.last; │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ + this.up(evt); │ │ │ │ │ + this.callback("up", [evt.xy]); │ │ │ │ │ + if (dragged) { │ │ │ │ │ + this.callback("done", [evt.xy]) │ │ │ │ │ + } │ │ │ │ │ + document.onselectstart = this.oldOnselectstart │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + down: function(evt) {}, │ │ │ │ │ + move: function(evt) {}, │ │ │ │ │ + up: function(evt) {}, │ │ │ │ │ + out: function(evt) {}, │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + return this.dragstart(evt) │ │ │ │ │ + }, │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + return this.dragstart(evt) │ │ │ │ │ + }, │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + return this.dragmove(evt) │ │ │ │ │ + }, │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + return this.dragmove(evt) │ │ │ │ │ + }, │ │ │ │ │ + removeTimeout: function() { │ │ │ │ │ + this.timeoutId = null; │ │ │ │ │ + if (this.dragging) { │ │ │ │ │ + this.mousemove(this.lastMoveEvt) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + return this.dragend(evt) │ │ │ │ │ + }, │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + evt.xy = this.last; │ │ │ │ │ + return this.dragend(evt) │ │ │ │ │ + }, │ │ │ │ │ + mouseout: function(evt) { │ │ │ │ │ + if (this.started && OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ + if (this.documentDrag === true) { │ │ │ │ │ + this.addDocumentEvents() │ │ │ │ │ + } else { │ │ │ │ │ + var dragged = this.start != this.last; │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ + this.out(evt); │ │ │ │ │ + this.callback("out", []); │ │ │ │ │ + if (dragged) { │ │ │ │ │ + this.callback("done", [evt.xy]) │ │ │ │ │ + } │ │ │ │ │ + if (document.onselectstart) { │ │ │ │ │ + document.onselectstart = this.oldOnselectstart │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + return this.start == this.last │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + activated = true │ │ │ │ │ + } │ │ │ │ │ + return activated │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + deactivated = true; │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown") │ │ │ │ │ + } │ │ │ │ │ + return deactivated │ │ │ │ │ + }, │ │ │ │ │ + adjustXY: function(evt) { │ │ │ │ │ + var pos = OpenLayers.Util.pagePosition(this.map.viewPortDiv); │ │ │ │ │ + evt.xy.x -= pos[0]; │ │ │ │ │ + evt.xy.y -= pos[1] │ │ │ │ │ + }, │ │ │ │ │ + addDocumentEvents: function() { │ │ │ │ │ + OpenLayers.Element.addClass(document.body, "olDragDown"); │ │ │ │ │ + this.documentEvents = true; │ │ │ │ │ + OpenLayers.Event.observe(document, "mousemove", this._docMove); │ │ │ │ │ + OpenLayers.Event.observe(document, "mouseup", this._docUp) │ │ │ │ │ + }, │ │ │ │ │ + removeDocumentEvents: function() { │ │ │ │ │ + OpenLayers.Element.removeClass(document.body, "olDragDown"); │ │ │ │ │ + this.documentEvents = false; │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mousemove", this._docMove); │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mouseup", this._docUp) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Drag" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.DragPan = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ + panned: false, │ │ │ │ │ + interval: 0, │ │ │ │ │ + documentDrag: false, │ │ │ │ │ + kinetic: null, │ │ │ │ │ + enableKinetic: true, │ │ │ │ │ + kineticInterval: 10, │ │ │ │ │ + draw: function() { │ │ │ │ │ + if (this.enableKinetic && OpenLayers.Kinetic) { │ │ │ │ │ + var config = { │ │ │ │ │ + interval: this.kineticInterval │ │ │ │ │ + }; │ │ │ │ │ + if (typeof this.enableKinetic === "object") { │ │ │ │ │ + config = OpenLayers.Util.extend(config, this.enableKinetic) │ │ │ │ │ + } │ │ │ │ │ + this.kinetic = new OpenLayers.Kinetic(config) │ │ │ │ │ + } │ │ │ │ │ + this.handler = new OpenLayers.Handler.Drag(this, { │ │ │ │ │ + move: this.panMap, │ │ │ │ │ + done: this.panMapDone, │ │ │ │ │ + down: this.panMapStart │ │ │ │ │ + }, { │ │ │ │ │ + interval: this.interval, │ │ │ │ │ + documentDrag: this.documentDrag │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + panMapStart: function() { │ │ │ │ │ + if (this.kinetic) { │ │ │ │ │ + this.kinetic.begin() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + panMap: function(xy) { │ │ │ │ │ + if (this.kinetic) { │ │ │ │ │ + this.kinetic.update(xy) │ │ │ │ │ + } │ │ │ │ │ + this.panned = true; │ │ │ │ │ + this.map.pan(this.handler.last.x - xy.x, this.handler.last.y - xy.y, { │ │ │ │ │ + dragging: true, │ │ │ │ │ + animate: false │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + panMapDone: function(xy) { │ │ │ │ │ + if (this.panned) { │ │ │ │ │ + var res = null; │ │ │ │ │ + if (this.kinetic) { │ │ │ │ │ + res = this.kinetic.end(xy) │ │ │ │ │ + } │ │ │ │ │ + this.map.pan(this.handler.last.x - xy.x, this.handler.last.y - xy.y, { │ │ │ │ │ + dragging: !!res, │ │ │ │ │ + animate: false │ │ │ │ │ + }); │ │ │ │ │ + if (res) { │ │ │ │ │ + var self = this; │ │ │ │ │ + this.kinetic.move(res, function(x, y, end) { │ │ │ │ │ + self.map.pan(x, y, { │ │ │ │ │ + dragging: !end, │ │ │ │ │ + animate: false │ │ │ │ │ + }) │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + this.panned = false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.DragPan" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Handler.Pinch = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + started: false, │ │ │ │ │ + stopDown: false, │ │ │ │ │ + pinching: false, │ │ │ │ │ + last: null, │ │ │ │ │ + start: null, │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + if (OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ + this.started = true; │ │ │ │ │ + this.last = this.start = { │ │ │ │ │ + distance: this.getDistance(evt.touches), │ │ │ │ │ + delta: 0, │ │ │ │ │ + scale: 1 │ │ │ │ │ + }; │ │ │ │ │ + this.callback("start", [evt, this.start]); │ │ │ │ │ + propagate = !this.stopDown │ │ │ │ │ + } else if (this.started) { │ │ │ │ │ + return false │ │ │ │ │ + } else { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + return propagate │ │ │ │ │ + }, │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + if (this.started && OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ + this.pinching = true; │ │ │ │ │ + var current = this.getPinchData(evt); │ │ │ │ │ + this.callback("move", [evt, current]); │ │ │ │ │ + this.last = current; │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ + } else if (this.started) { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + if (this.started && !OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + this.callback("done", [evt, this.start, this.last]); │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + activated = true │ │ │ │ │ + } │ │ │ │ │ + return activated │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + deactivated = true │ │ │ │ │ + } │ │ │ │ │ + return deactivated │ │ │ │ │ + }, │ │ │ │ │ + getDistance: function(touches) { │ │ │ │ │ + var t0 = touches[0]; │ │ │ │ │ + var t1 = touches[1]; │ │ │ │ │ + return Math.sqrt(Math.pow(t0.olClientX - t1.olClientX, 2) + Math.pow(t0.olClientY - t1.olClientY, 2)) │ │ │ │ │ + }, │ │ │ │ │ + getPinchData: function(evt) { │ │ │ │ │ + var distance = this.getDistance(evt.touches); │ │ │ │ │ + var scale = distance / this.start.distance; │ │ │ │ │ + return { │ │ │ │ │ + distance: distance, │ │ │ │ │ + delta: this.last.distance - distance, │ │ │ │ │ + scale: scale │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Pinch" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.PinchZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ + pinchOrigin: null, │ │ │ │ │ + currentCenter: null, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + preserveCenter: false, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.handler = new OpenLayers.Handler.Pinch(this, { │ │ │ │ │ + start: this.pinchStart, │ │ │ │ │ + move: this.pinchMove, │ │ │ │ │ + done: this.pinchDone │ │ │ │ │ + }, this.handlerOptions) │ │ │ │ │ + }, │ │ │ │ │ + pinchStart: function(evt, pinchData) { │ │ │ │ │ + var xy = this.preserveCenter ? this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ + this.pinchOrigin = xy; │ │ │ │ │ + this.currentCenter = xy │ │ │ │ │ + }, │ │ │ │ │ + pinchMove: function(evt, pinchData) { │ │ │ │ │ + var scale = pinchData.scale; │ │ │ │ │ + var containerOrigin = this.map.layerContainerOriginPx; │ │ │ │ │ + var pinchOrigin = this.pinchOrigin; │ │ │ │ │ + var current = this.preserveCenter ? this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ + var dx = Math.round(containerOrigin.x + current.x - pinchOrigin.x + (scale - 1) * (containerOrigin.x - pinchOrigin.x)); │ │ │ │ │ + var dy = Math.round(containerOrigin.y + current.y - pinchOrigin.y + (scale - 1) * (containerOrigin.y - pinchOrigin.y)); │ │ │ │ │ + this.map.applyTransform(dx, dy, scale); │ │ │ │ │ + this.currentCenter = current │ │ │ │ │ + }, │ │ │ │ │ + pinchDone: function(evt, start, last) { │ │ │ │ │ + this.map.applyTransform(); │ │ │ │ │ + var zoom = this.map.getZoomForResolution(this.map.getResolution() / last.scale, true); │ │ │ │ │ + if (zoom !== this.map.getZoom() || !this.currentCenter.equals(this.pinchOrigin)) { │ │ │ │ │ + var resolution = this.map.getResolutionForZoom(zoom); │ │ │ │ │ + var location = this.map.getLonLatFromPixel(this.pinchOrigin); │ │ │ │ │ + var zoomPixel = this.currentCenter; │ │ │ │ │ + var size = this.map.getSize(); │ │ │ │ │ + location.lon += resolution * (size.w / 2 - zoomPixel.x); │ │ │ │ │ + location.lat -= resolution * (size.h / 2 - zoomPixel.y); │ │ │ │ │ + this.map.div.clientWidth = this.map.div.clientWidth; │ │ │ │ │ + this.map.setCenter(location, zoom) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.PinchZoom" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Handler.Click = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + delay: 300, │ │ │ │ │ + single: true, │ │ │ │ │ + double: false, │ │ │ │ │ + pixelTolerance: 0, │ │ │ │ │ + dblclickTolerance: 13, │ │ │ │ │ + stopSingle: false, │ │ │ │ │ + stopDouble: false, │ │ │ │ │ + timerId: null, │ │ │ │ │ + down: null, │ │ │ │ │ + last: null, │ │ │ │ │ + first: null, │ │ │ │ │ + rightclickTimerId: null, │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + this.down = this.getEventInfo(evt); │ │ │ │ │ + this.last = this.getEventInfo(evt); │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + this.last = this.getEventInfo(evt); │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + if (this.down) { │ │ │ │ │ + evt.xy = this.last.xy; │ │ │ │ │ + evt.lastTouches = this.last.touches; │ │ │ │ │ + this.handleSingle(evt); │ │ │ │ │ + this.down = null │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + this.down = this.getEventInfo(evt); │ │ │ │ │ + this.last = this.getEventInfo(evt); │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + if (this.checkModifiers(evt) && this.control.handleRightClicks && OpenLayers.Event.isRightClick(evt)) { │ │ │ │ │ + propagate = this.rightclick(evt) │ │ │ │ │ + } │ │ │ │ │ + return propagate │ │ │ │ │ + }, │ │ │ │ │ + rightclick: function(evt) { │ │ │ │ │ + if (this.passesTolerance(evt)) { │ │ │ │ │ + if (this.rightclickTimerId != null) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.callback("dblrightclick", [evt]); │ │ │ │ │ + return !this.stopDouble │ │ │ │ │ + } else { │ │ │ │ │ + var clickEvent = this["double"] ? OpenLayers.Util.extend({}, evt) : this.callback("rightclick", [evt]); │ │ │ │ │ + var delayedRightCall = OpenLayers.Function.bind(this.delayedRightCall, this, clickEvent); │ │ │ │ │ + this.rightclickTimerId = window.setTimeout(delayedRightCall, this.delay) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return !this.stopSingle │ │ │ │ │ + }, │ │ │ │ │ + delayedRightCall: function(evt) { │ │ │ │ │ + this.rightclickTimerId = null; │ │ │ │ │ + if (evt) { │ │ │ │ │ + this.callback("rightclick", [evt]) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + if (!this.last) { │ │ │ │ │ + this.last = this.getEventInfo(evt) │ │ │ │ │ + } │ │ │ │ │ + this.handleSingle(evt); │ │ │ │ │ + return !this.stopSingle │ │ │ │ │ + }, │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + this.handleDouble(evt); │ │ │ │ │ + return !this.stopDouble │ │ │ │ │ + }, │ │ │ │ │ + handleDouble: function(evt) { │ │ │ │ │ + if (this.passesDblclickTolerance(evt)) { │ │ │ │ │ + if (this["double"]) { │ │ │ │ │ + this.callback("dblclick", [evt]) │ │ │ │ │ + } │ │ │ │ │ + this.clearTimer() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + handleSingle: function(evt) { │ │ │ │ │ + if (this.passesTolerance(evt)) { │ │ │ │ │ + if (this.timerId != null) { │ │ │ │ │ + if (this.last.touches && this.last.touches.length === 1) { │ │ │ │ │ + if (this["double"]) { │ │ │ │ │ + OpenLayers.Event.preventDefault(evt) │ │ │ │ │ + } │ │ │ │ │ + this.handleDouble(evt) │ │ │ │ │ + } │ │ │ │ │ + if (!this.last.touches || this.last.touches.length !== 2) { │ │ │ │ │ + this.clearTimer() │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this.first = this.getEventInfo(evt); │ │ │ │ │ + var clickEvent = this.single ? OpenLayers.Util.extend({}, evt) : null; │ │ │ │ │ + this.queuePotentialClick(clickEvent) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + queuePotentialClick: function(evt) { │ │ │ │ │ + this.timerId = window.setTimeout(OpenLayers.Function.bind(this.delayedCall, this, evt), this.delay) │ │ │ │ │ + }, │ │ │ │ │ + passesTolerance: function(evt) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + if (this.pixelTolerance != null && this.down && this.down.xy) { │ │ │ │ │ + passes = this.pixelTolerance >= this.down.xy.distanceTo(evt.xy); │ │ │ │ │ + if (passes && this.touch && this.down.touches.length === this.last.touches.length) { │ │ │ │ │ + for (var i = 0, ii = this.down.touches.length; i < ii; ++i) { │ │ │ │ │ + if (this.getTouchDistance(this.down.touches[i], this.last.touches[i]) > this.pixelTolerance) { │ │ │ │ │ + passes = false; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return passes │ │ │ │ │ + }, │ │ │ │ │ + getTouchDistance: function(from, to) { │ │ │ │ │ + return Math.sqrt(Math.pow(from.clientX - to.clientX, 2) + Math.pow(from.clientY - to.clientY, 2)) │ │ │ │ │ + }, │ │ │ │ │ + passesDblclickTolerance: function(evt) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + if (this.down && this.first) { │ │ │ │ │ + passes = this.down.xy.distanceTo(this.first.xy) <= this.dblclickTolerance │ │ │ │ │ + } │ │ │ │ │ + return passes │ │ │ │ │ + }, │ │ │ │ │ + clearTimer: function() { │ │ │ │ │ + if (this.timerId != null) { │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null │ │ │ │ │ + } │ │ │ │ │ + if (this.rightclickTimerId != null) { │ │ │ │ │ + window.clearTimeout(this.rightclickTimerId); │ │ │ │ │ + this.rightclickTimerId = null │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + delayedCall: function(evt) { │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + if (evt) { │ │ │ │ │ + this.callback("click", [evt]) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getEventInfo: function(evt) { │ │ │ │ │ + var touches; │ │ │ │ │ + if (evt.touches) { │ │ │ │ │ + var len = evt.touches.length; │ │ │ │ │ + touches = new Array(len); │ │ │ │ │ + var touch; │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + touch = evt.touches[i]; │ │ │ │ │ + touches[i] = { │ │ │ │ │ + clientX: touch.olClientX, │ │ │ │ │ + clientY: touch.olClientY │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + xy: evt.xy, │ │ │ │ │ + touches: touches │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.down = null; │ │ │ │ │ + this.first = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + deactivated = true │ │ │ │ │ + } │ │ │ │ │ + return deactivated │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Click" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.TouchNavigation = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + dragPan: null, │ │ │ │ │ + dragPanOptions: null, │ │ │ │ │ + pinchZoom: null, │ │ │ │ │ + pinchZoomOptions: null, │ │ │ │ │ + clickHandlerOptions: null, │ │ │ │ │ + documentDrag: false, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.handlers = {}; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + if (this.dragPan) { │ │ │ │ │ + this.dragPan.destroy() │ │ │ │ │ + } │ │ │ │ │ + this.dragPan = null; │ │ │ │ │ + if (this.pinchZoom) { │ │ │ │ │ + this.pinchZoom.destroy(); │ │ │ │ │ + delete this.pinchZoom │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.dragPan.activate(); │ │ │ │ │ + this.handlers.click.activate(); │ │ │ │ │ + this.pinchZoom.activate(); │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.dragPan.deactivate(); │ │ │ │ │ + this.handlers.click.deactivate(); │ │ │ │ │ + this.pinchZoom.deactivate(); │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + var clickCallbacks = { │ │ │ │ │ + click: this.defaultClick, │ │ │ │ │ + dblclick: this.defaultDblClick │ │ │ │ │ + }; │ │ │ │ │ + var clickOptions = OpenLayers.Util.extend({ │ │ │ │ │ + double: true, │ │ │ │ │ + stopDouble: true, │ │ │ │ │ + pixelTolerance: 2 │ │ │ │ │ + }, this.clickHandlerOptions); │ │ │ │ │ + this.handlers.click = new OpenLayers.Handler.Click(this, clickCallbacks, clickOptions); │ │ │ │ │ + this.dragPan = new OpenLayers.Control.DragPan(OpenLayers.Util.extend({ │ │ │ │ │ + map: this.map, │ │ │ │ │ + documentDrag: this.documentDrag │ │ │ │ │ + }, this.dragPanOptions)); │ │ │ │ │ + this.dragPan.draw(); │ │ │ │ │ + this.pinchZoom = new OpenLayers.Control.PinchZoom(OpenLayers.Util.extend({ │ │ │ │ │ + map: this.map │ │ │ │ │ + }, this.pinchZoomOptions)) │ │ │ │ │ + }, │ │ │ │ │ + defaultClick: function(evt) { │ │ │ │ │ + if (evt.lastTouches && evt.lastTouches.length == 2) { │ │ │ │ │ + this.map.zoomOut() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + defaultDblClick: function(evt) { │ │ │ │ │ + this.map.zoomTo(this.map.zoom + 1, evt.xy) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.TouchNavigation" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Handler.Keyboard = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + KEY_EVENTS: ["keydown", "keyup"], │ │ │ │ │ + eventListener: null, │ │ │ │ │ + observeElement: null, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.eventListener = OpenLayers.Function.bindAsEventListener(this.handleKeyEvent, this) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.eventListener = null; │ │ │ │ │ + OpenLayers.Handler.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.observeElement = this.observeElement || document; │ │ │ │ │ + for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ + OpenLayers.Event.observe(this.observeElement, this.KEY_EVENTS[i], this.eventListener) │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ + OpenLayers.Event.stopObserving(this.observeElement, this.KEY_EVENTS[i], this.eventListener) │ │ │ │ │ + } │ │ │ │ │ + deactivated = true │ │ │ │ │ + } │ │ │ │ │ + return deactivated │ │ │ │ │ + }, │ │ │ │ │ + handleKeyEvent: function(evt) { │ │ │ │ │ + if (this.checkModifiers(evt)) { │ │ │ │ │ + this.callback(evt.type, [evt]) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Keyboard" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.ModifyFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + documentDrag: false, │ │ │ │ │ + geometryTypes: null, │ │ │ │ │ + clickout: true, │ │ │ │ │ + toggle: true, │ │ │ │ │ + standalone: false, │ │ │ │ │ + layer: null, │ │ │ │ │ + feature: null, │ │ │ │ │ + vertex: null, │ │ │ │ │ + vertices: null, │ │ │ │ │ + virtualVertices: null, │ │ │ │ │ + handlers: null, │ │ │ │ │ + deleteCodes: null, │ │ │ │ │ + virtualStyle: null, │ │ │ │ │ + vertexRenderIntent: null, │ │ │ │ │ + mode: null, │ │ │ │ │ + createVertices: true, │ │ │ │ │ + modified: false, │ │ │ │ │ + radiusHandle: null, │ │ │ │ │ + dragHandle: null, │ │ │ │ │ + onModificationStart: function() {}, │ │ │ │ │ + onModification: function() {}, │ │ │ │ │ + onModificationEnd: function() {}, │ │ │ │ │ + initialize: function(layer, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + this.virtualVertices = []; │ │ │ │ │ + this.virtualStyle = OpenLayers.Util.extend({}, this.layer.style || this.layer.styleMap.createSymbolizer(null, options.vertexRenderIntent)); │ │ │ │ │ + this.virtualStyle.fillOpacity = .3; │ │ │ │ │ + this.virtualStyle.strokeOpacity = .3; │ │ │ │ │ + this.deleteCodes = [46, 68]; │ │ │ │ │ + this.mode = OpenLayers.Control.ModifyFeature.RESHAPE; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!OpenLayers.Util.isArray(this.deleteCodes)) { │ │ │ │ │ + this.deleteCodes = [this.deleteCodes] │ │ │ │ │ + } │ │ │ │ │ + var dragCallbacks = { │ │ │ │ │ + down: function(pixel) { │ │ │ │ │ + this.vertex = null; │ │ │ │ │ + var feature = this.layer.getFeatureFromEvent(this.handlers.drag.evt); │ │ │ │ │ + if (feature) { │ │ │ │ │ + this.dragStart(feature) │ │ │ │ │ + } else if (this.clickout) { │ │ │ │ │ + this._unselect = this.feature │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + move: function(pixel) { │ │ │ │ │ + delete this._unselect; │ │ │ │ │ + if (this.vertex) { │ │ │ │ │ + this.dragVertex(this.vertex, pixel) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + up: function() { │ │ │ │ │ + this.handlers.drag.stopDown = false; │ │ │ │ │ + if (this._unselect) { │ │ │ │ │ + this.unselectFeature(this._unselect); │ │ │ │ │ + delete this._unselect │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + done: function(pixel) { │ │ │ │ │ + if (this.vertex) { │ │ │ │ │ + this.dragComplete(this.vertex) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + var dragOptions = { │ │ │ │ │ + documentDrag: this.documentDrag, │ │ │ │ │ + stopDown: false │ │ │ │ │ + }; │ │ │ │ │ + var keyboardOptions = { │ │ │ │ │ + keydown: this.handleKeypress │ │ │ │ │ + }; │ │ │ │ │ + this.handlers = { │ │ │ │ │ + keyboard: new OpenLayers.Handler.Keyboard(this, keyboardOptions), │ │ │ │ │ + drag: new OpenLayers.Handler.Drag(this, dragCallbacks, dragOptions) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + removelayer: this.handleMapEvents, │ │ │ │ │ + changelayer: this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + this.layer = null; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, []) │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + this.moveLayerToTop(); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + removelayer: this.handleMapEvents, │ │ │ │ │ + changelayer: this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + return this.handlers.keyboard.activate() && this.handlers.drag.activate() && OpenLayers.Control.prototype.activate.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerBack(); │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + removelayer: this.handleMapEvents, │ │ │ │ │ + changelayer: this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.layer.removeFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.layer.removeFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + this.handlers.drag.deactivate(); │ │ │ │ │ + this.handlers.keyboard.deactivate(); │ │ │ │ │ + var feature = this.feature; │ │ │ │ │ + if (feature && feature.geometry && feature.layer) { │ │ │ │ │ + this.unselectFeature(feature) │ │ │ │ │ + } │ │ │ │ │ + deactivated = true │ │ │ │ │ + } │ │ │ │ │ + return deactivated │ │ │ │ │ + }, │ │ │ │ │ + beforeSelectFeature: function(feature) { │ │ │ │ │ + return this.layer.events.triggerEvent("beforefeaturemodified", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + selectFeature: function(feature) { │ │ │ │ │ + if (this.feature === feature || this.geometryTypes && OpenLayers.Util.indexOf(this.geometryTypes, feature.geometry.CLASS_NAME) == -1) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (this.beforeSelectFeature(feature) !== false) { │ │ │ │ │ + if (this.feature) { │ │ │ │ │ + this.unselectFeature(this.feature) │ │ │ │ │ + } │ │ │ │ │ + this.feature = feature; │ │ │ │ │ + this.layer.selectedFeatures.push(feature); │ │ │ │ │ + this.layer.drawFeature(feature, "select"); │ │ │ │ │ + this.modified = false; │ │ │ │ │ + this.resetVertices(); │ │ │ │ │ + this.onModificationStart(this.feature) │ │ │ │ │ + } │ │ │ │ │ + var modified = feature.modified; │ │ │ │ │ + if (feature.geometry && !(modified && modified.geometry)) { │ │ │ │ │ + this._originalGeometry = feature.geometry.clone() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + unselectFeature: function(feature) { │ │ │ │ │ + this.layer.removeFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + this.layer.destroyFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.virtualVertices = []; │ │ │ │ │ + if (this.dragHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.dragHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + delete this.dragHandle │ │ │ │ │ + } │ │ │ │ │ + if (this.radiusHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + delete this.radiusHandle │ │ │ │ │ + } │ │ │ │ │ + this.layer.drawFeature(this.feature, "default"); │ │ │ │ │ + this.feature = null; │ │ │ │ │ + OpenLayers.Util.removeItem(this.layer.selectedFeatures, feature); │ │ │ │ │ + this.onModificationEnd(feature); │ │ │ │ │ + this.layer.events.triggerEvent("afterfeaturemodified", { │ │ │ │ │ + feature: feature, │ │ │ │ │ + modified: this.modified │ │ │ │ │ + }); │ │ │ │ │ + this.modified = false │ │ │ │ │ + }, │ │ │ │ │ + dragStart: function(feature) { │ │ │ │ │ + var isPoint = feature.geometry.CLASS_NAME == "OpenLayers.Geometry.Point"; │ │ │ │ │ + if (!this.standalone && (!feature._sketch && isPoint || !feature._sketch)) { │ │ │ │ │ + if (this.toggle && this.feature === feature) { │ │ │ │ │ + this._unselect = feature │ │ │ │ │ + } │ │ │ │ │ + this.selectFeature(feature) │ │ │ │ │ + } │ │ │ │ │ + if (feature._sketch || isPoint) { │ │ │ │ │ + this.vertex = feature; │ │ │ │ │ + this.handlers.drag.stopDown = true │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + dragVertex: function(vertex, pixel) { │ │ │ │ │ + var pos = this.map.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geom = vertex.geometry; │ │ │ │ │ + geom.move(pos.lon - geom.x, pos.lat - geom.y); │ │ │ │ │ + this.modified = true; │ │ │ │ │ + if (this.feature.geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + this.layer.events.triggerEvent("vertexmodified", { │ │ │ │ │ + vertex: vertex.geometry, │ │ │ │ │ + feature: this.feature, │ │ │ │ │ + pixel: pixel │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + if (vertex._index) { │ │ │ │ │ + vertex.geometry.parent.addComponent(vertex.geometry, vertex._index); │ │ │ │ │ + delete vertex._index; │ │ │ │ │ + OpenLayers.Util.removeItem(this.virtualVertices, vertex); │ │ │ │ │ + this.vertices.push(vertex) │ │ │ │ │ + } else if (vertex == this.dragHandle) { │ │ │ │ │ + this.layer.removeFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + if (this.radiusHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.radiusHandle = null │ │ │ │ │ + } │ │ │ │ │ + } else if (vertex !== this.radiusHandle) { │ │ │ │ │ + this.layer.events.triggerEvent("vertexmodified", { │ │ │ │ │ + vertex: vertex.geometry, │ │ │ │ │ + feature: this.feature, │ │ │ │ │ + pixel: pixel │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + if (this.virtualVertices.length > 0) { │ │ │ │ │ + this.layer.destroyFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.virtualVertices = [] │ │ │ │ │ + } │ │ │ │ │ + this.layer.drawFeature(this.feature, this.standalone ? undefined : "select") │ │ │ │ │ + } │ │ │ │ │ + this.layer.drawFeature(vertex) │ │ │ │ │ + }, │ │ │ │ │ + dragComplete: function(vertex) { │ │ │ │ │ + this.resetVertices(); │ │ │ │ │ + this.setFeatureState(); │ │ │ │ │ + this.onModification(this.feature); │ │ │ │ │ + this.layer.events.triggerEvent("featuremodified", { │ │ │ │ │ + feature: this.feature │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + setFeatureState: function() { │ │ │ │ │ + if (this.feature.state != OpenLayers.State.INSERT && this.feature.state != OpenLayers.State.DELETE) { │ │ │ │ │ + this.feature.state = OpenLayers.State.UPDATE; │ │ │ │ │ + if (this.modified && this._originalGeometry) { │ │ │ │ │ + var feature = this.feature; │ │ │ │ │ + feature.modified = OpenLayers.Util.extend(feature.modified, { │ │ │ │ │ + geometry: this._originalGeometry │ │ │ │ │ + }); │ │ │ │ │ + delete this._originalGeometry │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + resetVertices: function() { │ │ │ │ │ + if (this.vertices.length > 0) { │ │ │ │ │ + this.layer.removeFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.vertices = [] │ │ │ │ │ + } │ │ │ │ │ + if (this.virtualVertices.length > 0) { │ │ │ │ │ + this.layer.removeFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.virtualVertices = [] │ │ │ │ │ + } │ │ │ │ │ + if (this.dragHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.dragHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.dragHandle = null │ │ │ │ │ + } │ │ │ │ │ + if (this.radiusHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.radiusHandle = null │ │ │ │ │ + } │ │ │ │ │ + if (this.feature && this.feature.geometry.CLASS_NAME != "OpenLayers.Geometry.Point") { │ │ │ │ │ + if (this.mode & OpenLayers.Control.ModifyFeature.DRAG) { │ │ │ │ │ + this.collectDragHandle() │ │ │ │ │ + } │ │ │ │ │ + if (this.mode & (OpenLayers.Control.ModifyFeature.ROTATE | OpenLayers.Control.ModifyFeature.RESIZE)) { │ │ │ │ │ + this.collectRadiusHandle() │ │ │ │ │ + } │ │ │ │ │ + if (this.mode & OpenLayers.Control.ModifyFeature.RESHAPE) { │ │ │ │ │ + if (!(this.mode & OpenLayers.Control.ModifyFeature.RESIZE)) { │ │ │ │ │ + this.collectVertices() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + handleKeypress: function(evt) { │ │ │ │ │ + var code = evt.keyCode; │ │ │ │ │ + if (this.feature && OpenLayers.Util.indexOf(this.deleteCodes, code) != -1) { │ │ │ │ │ + var vertex = this.layer.getFeatureFromEvent(this.handlers.drag.evt); │ │ │ │ │ + if (vertex && OpenLayers.Util.indexOf(this.vertices, vertex) != -1 && !this.handlers.drag.dragging && vertex.geometry.parent) { │ │ │ │ │ + vertex.geometry.parent.removeComponent(vertex.geometry); │ │ │ │ │ + this.layer.events.triggerEvent("vertexremoved", { │ │ │ │ │ + vertex: vertex.geometry, │ │ │ │ │ + feature: this.feature, │ │ │ │ │ + pixel: evt.xy │ │ │ │ │ + }); │ │ │ │ │ + this.layer.drawFeature(this.feature, this.standalone ? undefined : "select"); │ │ │ │ │ + this.modified = true; │ │ │ │ │ + this.resetVertices(); │ │ │ │ │ + this.setFeatureState(); │ │ │ │ │ + this.onModification(this.feature); │ │ │ │ │ + this.layer.events.triggerEvent("featuremodified", { │ │ │ │ │ + feature: this.feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + collectVertices: function() { │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + this.virtualVertices = []; │ │ │ │ │ + var control = this; │ │ │ │ │ + │ │ │ │ │ + function collectComponentVertices(geometry) { │ │ │ │ │ + var i, vertex, component, len; │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + vertex = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + vertex._sketch = true; │ │ │ │ │ + vertex.renderIntent = control.vertexRenderIntent; │ │ │ │ │ + control.vertices.push(vertex) │ │ │ │ │ + } else { │ │ │ │ │ + var numVert = geometry.components.length; │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ + numVert -= 1 │ │ │ │ │ + } │ │ │ │ │ + for (i = 0; i < numVert; ++i) { │ │ │ │ │ + component = geometry.components[i]; │ │ │ │ │ + if (component.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + vertex = new OpenLayers.Feature.Vector(component); │ │ │ │ │ + vertex._sketch = true; │ │ │ │ │ + vertex.renderIntent = control.vertexRenderIntent; │ │ │ │ │ + control.vertices.push(vertex) │ │ │ │ │ + } else { │ │ │ │ │ + collectComponentVertices(component) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (control.createVertices && geometry.CLASS_NAME != "OpenLayers.Geometry.MultiPoint") { │ │ │ │ │ + for (i = 0, len = geometry.components.length; i < len - 1; ++i) { │ │ │ │ │ + var prevVertex = geometry.components[i]; │ │ │ │ │ + var nextVertex = geometry.components[i + 1]; │ │ │ │ │ + if (prevVertex.CLASS_NAME == "OpenLayers.Geometry.Point" && nextVertex.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + var x = (prevVertex.x + nextVertex.x) / 2; │ │ │ │ │ + var y = (prevVertex.y + nextVertex.y) / 2; │ │ │ │ │ + var point = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Point(x, y), null, control.virtualStyle); │ │ │ │ │ + point.geometry.parent = geometry; │ │ │ │ │ + point._index = i + 1; │ │ │ │ │ + point._sketch = true; │ │ │ │ │ + control.virtualVertices.push(point) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + collectComponentVertices.call(this, this.feature.geometry); │ │ │ │ │ + this.layer.addFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.layer.addFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + collectDragHandle: function() { │ │ │ │ │ + var geometry = this.feature.geometry; │ │ │ │ │ + var center = geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + var originGeometry = new OpenLayers.Geometry.Point(center.lon, center.lat); │ │ │ │ │ + var origin = new OpenLayers.Feature.Vector(originGeometry); │ │ │ │ │ + originGeometry.move = function(x, y) { │ │ │ │ │ + OpenLayers.Geometry.Point.prototype.move.call(this, x, y); │ │ │ │ │ + geometry.move(x, y) │ │ │ │ │ + }; │ │ │ │ │ + origin._sketch = true; │ │ │ │ │ + this.dragHandle = origin; │ │ │ │ │ + this.dragHandle.renderIntent = this.vertexRenderIntent; │ │ │ │ │ + this.layer.addFeatures([this.dragHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + collectRadiusHandle: function() { │ │ │ │ │ + var geometry = this.feature.geometry; │ │ │ │ │ + var bounds = geometry.getBounds(); │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var originGeometry = new OpenLayers.Geometry.Point(center.lon, center.lat); │ │ │ │ │ + var radiusGeometry = new OpenLayers.Geometry.Point(bounds.right, bounds.bottom); │ │ │ │ │ + var radius = new OpenLayers.Feature.Vector(radiusGeometry); │ │ │ │ │ + var resize = this.mode & OpenLayers.Control.ModifyFeature.RESIZE; │ │ │ │ │ + var reshape = this.mode & OpenLayers.Control.ModifyFeature.RESHAPE; │ │ │ │ │ + var rotate = this.mode & OpenLayers.Control.ModifyFeature.ROTATE; │ │ │ │ │ + radiusGeometry.move = function(x, y) { │ │ │ │ │ + OpenLayers.Geometry.Point.prototype.move.call(this, x, y); │ │ │ │ │ + var dx1 = this.x - originGeometry.x; │ │ │ │ │ + var dy1 = this.y - originGeometry.y; │ │ │ │ │ + var dx0 = dx1 - x; │ │ │ │ │ + var dy0 = dy1 - y; │ │ │ │ │ + if (rotate) { │ │ │ │ │ + var a0 = Math.atan2(dy0, dx0); │ │ │ │ │ + var a1 = Math.atan2(dy1, dx1); │ │ │ │ │ + var angle = a1 - a0; │ │ │ │ │ + angle *= 180 / Math.PI; │ │ │ │ │ + geometry.rotate(angle, originGeometry) │ │ │ │ │ + } │ │ │ │ │ + if (resize) { │ │ │ │ │ + var scale, ratio; │ │ │ │ │ + if (reshape) { │ │ │ │ │ + scale = dy1 / dy0; │ │ │ │ │ + ratio = dx1 / dx0 / scale │ │ │ │ │ + } else { │ │ │ │ │ + var l0 = Math.sqrt(dx0 * dx0 + dy0 * dy0); │ │ │ │ │ + var l1 = Math.sqrt(dx1 * dx1 + dy1 * dy1); │ │ │ │ │ + scale = l1 / l0 │ │ │ │ │ + } │ │ │ │ │ + geometry.resize(scale, originGeometry, ratio) │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + radius._sketch = true; │ │ │ │ │ + this.radiusHandle = radius; │ │ │ │ │ + this.radiusHandle.renderIntent = this.vertexRenderIntent; │ │ │ │ │ + this.layer.addFeatures([this.radiusHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.handlers.drag.setMap(map); │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + handleMapEvents: function(evt) { │ │ │ │ │ + if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ + this.moveLayerToTop() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + moveLayerToTop: function() { │ │ │ │ │ + var index = Math.max(this.map.Z_INDEX_BASE["Feature"] - 1, this.layer.getZIndex()) + 1; │ │ │ │ │ + this.layer.setZIndex(index) │ │ │ │ │ + }, │ │ │ │ │ + moveLayerBack: function() { │ │ │ │ │ + var index = this.layer.getZIndex() - 1; │ │ │ │ │ + if (index >= this.map.Z_INDEX_BASE["Feature"]) { │ │ │ │ │ + this.layer.setZIndex(index) │ │ │ │ │ + } else { │ │ │ │ │ + this.map.setLayerZIndex(this.layer, this.map.getLayerIndex(this.layer)) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ModifyFeature" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.ModifyFeature.RESHAPE = 1; │ │ │ │ │ +OpenLayers.Control.ModifyFeature.RESIZE = 2; │ │ │ │ │ +OpenLayers.Control.ModifyFeature.ROTATE = 4; │ │ │ │ │ +OpenLayers.Control.ModifyFeature.DRAG = 8; │ │ │ │ │ +OpenLayers.Events.buttonclick = OpenLayers.Class({ │ │ │ │ │ + target: null, │ │ │ │ │ + events: ["mousedown", "mouseup", "click", "dblclick", "touchstart", "touchmove", "touchend", "keydown"], │ │ │ │ │ + startRegEx: /^mousedown|touchstart$/, │ │ │ │ │ + cancelRegEx: /^touchmove$/, │ │ │ │ │ + completeRegEx: /^mouseup|touchend$/, │ │ │ │ │ + initialize: function(target) { │ │ │ │ │ + this.target = target; │ │ │ │ │ + for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ + this.target.register(this.events[i], this, this.buttonClick, { │ │ │ │ │ + extension: true │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ + this.target.unregister(this.events[i], this, this.buttonClick) │ │ │ │ │ + } │ │ │ │ │ + delete this.target │ │ │ │ │ + }, │ │ │ │ │ + getPressedButton: function(element) { │ │ │ │ │ + var depth = 3, │ │ │ │ │ + button; │ │ │ │ │ + do { │ │ │ │ │ + if (OpenLayers.Element.hasClass(element, "olButton")) { │ │ │ │ │ + button = element; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + element = element.parentNode │ │ │ │ │ + } while (--depth > 0 && element); │ │ │ │ │ + return button │ │ │ │ │ + }, │ │ │ │ │ + ignore: function(element) { │ │ │ │ │ + var depth = 3, │ │ │ │ │ + ignore = false; │ │ │ │ │ + do { │ │ │ │ │ + if (element.nodeName.toLowerCase() === "a") { │ │ │ │ │ + ignore = true; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + element = element.parentNode │ │ │ │ │ + } while (--depth > 0 && element); │ │ │ │ │ + return ignore │ │ │ │ │ + }, │ │ │ │ │ + buttonClick: function(evt) { │ │ │ │ │ + var propagate = true, │ │ │ │ │ + element = OpenLayers.Event.element(evt); │ │ │ │ │ + if (element && (OpenLayers.Event.isLeftClick(evt) || !~evt.type.indexOf("mouse"))) { │ │ │ │ │ + var button = this.getPressedButton(element); │ │ │ │ │ + if (button) { │ │ │ │ │ + if (evt.type === "keydown") { │ │ │ │ │ + switch (evt.keyCode) { │ │ │ │ │ + case OpenLayers.Event.KEY_RETURN: │ │ │ │ │ + case OpenLayers.Event.KEY_SPACE: │ │ │ │ │ + this.target.triggerEvent("buttonclick", { │ │ │ │ │ + buttonElement: button │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } else if (this.startEvt) { │ │ │ │ │ + if (this.completeRegEx.test(evt.type)) { │ │ │ │ │ + var pos = OpenLayers.Util.pagePosition(button); │ │ │ │ │ + var viewportElement = OpenLayers.Util.getViewportElement(); │ │ │ │ │ + var scrollTop = window.pageYOffset || viewportElement.scrollTop; │ │ │ │ │ + var scrollLeft = window.pageXOffset || viewportElement.scrollLeft; │ │ │ │ │ + pos[0] = pos[0] - scrollLeft; │ │ │ │ │ + pos[1] = pos[1] - scrollTop; │ │ │ │ │ + this.target.triggerEvent("buttonclick", { │ │ │ │ │ + buttonElement: button, │ │ │ │ │ + buttonXY: { │ │ │ │ │ + x: this.startEvt.clientX - pos[0], │ │ │ │ │ + y: this.startEvt.clientY - pos[1] │ │ │ │ │ + } │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + if (this.cancelRegEx.test(evt.type)) { │ │ │ │ │ + delete this.startEvt │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false │ │ │ │ │ + } │ │ │ │ │ + if (this.startRegEx.test(evt.type)) { │ │ │ │ │ + this.startEvt = evt; │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + propagate = !this.ignore(OpenLayers.Event.element(evt)); │ │ │ │ │ + delete this.startEvt │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return propagate │ │ │ │ │ + } │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Zoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + zoomInText: "+", │ │ │ │ │ + zoomInId: "olZoomInLink", │ │ │ │ │ + zoomOutText: "−", │ │ │ │ │ + zoomOutId: "olZoomOutLink", │ │ │ │ │ + draw: function() { │ │ │ │ │ + var div = OpenLayers.Control.prototype.draw.apply(this), │ │ │ │ │ + links = this.getOrCreateLinks(div), │ │ │ │ │ + zoomIn = links.zoomIn, │ │ │ │ │ + zoomOut = links.zoomOut, │ │ │ │ │ + eventsInstance = this.map.events; │ │ │ │ │ + if (zoomOut.parentNode !== div) { │ │ │ │ │ + eventsInstance = this.events; │ │ │ │ │ + eventsInstance.attachToElement(zoomOut.parentNode) │ │ │ │ │ + } │ │ │ │ │ + eventsInstance.register("buttonclick", this, this.onZoomClick); │ │ │ │ │ + this.zoomInLink = zoomIn; │ │ │ │ │ + this.zoomOutLink = zoomOut; │ │ │ │ │ + return div │ │ │ │ │ + }, │ │ │ │ │ + getOrCreateLinks: function(el) { │ │ │ │ │ + var zoomIn = document.getElementById(this.zoomInId), │ │ │ │ │ + zoomOut = document.getElementById(this.zoomOutId); │ │ │ │ │ + if (!zoomIn) { │ │ │ │ │ + zoomIn = document.createElement("a"); │ │ │ │ │ + zoomIn.href = "#zoomIn"; │ │ │ │ │ + zoomIn.appendChild(document.createTextNode(this.zoomInText)); │ │ │ │ │ + zoomIn.className = "olControlZoomIn"; │ │ │ │ │ + el.appendChild(zoomIn) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.addClass(zoomIn, "olButton"); │ │ │ │ │ + if (!zoomOut) { │ │ │ │ │ + zoomOut = document.createElement("a"); │ │ │ │ │ + zoomOut.href = "#zoomOut"; │ │ │ │ │ + zoomOut.appendChild(document.createTextNode(this.zoomOutText)); │ │ │ │ │ + zoomOut.className = "olControlZoomOut"; │ │ │ │ │ + el.appendChild(zoomOut) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.addClass(zoomOut, "olButton"); │ │ │ │ │ + return { │ │ │ │ │ + zoomIn: zoomIn, │ │ │ │ │ + zoomOut: zoomOut │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + onZoomClick: function(evt) { │ │ │ │ │ + var button = evt.buttonElement; │ │ │ │ │ + if (button === this.zoomInLink) { │ │ │ │ │ + this.map.zoomIn() │ │ │ │ │ + } else if (button === this.zoomOutLink) { │ │ │ │ │ + this.map.zoomOut() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onZoomClick) │ │ │ │ │ + } │ │ │ │ │ + delete this.zoomInLink; │ │ │ │ │ + delete this.zoomOutLink; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Zoom" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.DrawFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + layer: null, │ │ │ │ │ + callbacks: null, │ │ │ │ │ + multi: false, │ │ │ │ │ + featureAdded: function() {}, │ │ │ │ │ + initialize: function(layer, handler, options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.callbacks = OpenLayers.Util.extend({ │ │ │ │ │ + done: this.drawFeature, │ │ │ │ │ + modify: function(vertex, feature) { │ │ │ │ │ + this.layer.events.triggerEvent("sketchmodified", { │ │ │ │ │ + vertex: vertex, │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + create: function(vertex, feature) { │ │ │ │ │ + this.layer.events.triggerEvent("sketchstarted", { │ │ │ │ │ + vertex: vertex, │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, this.callbacks); │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + this.handlerOptions = this.handlerOptions || {}; │ │ │ │ │ + this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults(this.handlerOptions.layerOptions, { │ │ │ │ │ + renderers: layer.renderers, │ │ │ │ │ + rendererOptions: layer.rendererOptions │ │ │ │ │ + }); │ │ │ │ │ + if (!("multi" in this.handlerOptions)) { │ │ │ │ │ + this.handlerOptions.multi = this.multi │ │ │ │ │ + } │ │ │ │ │ + var sketchStyle = this.layer.styleMap && this.layer.styleMap.styles.temporary; │ │ │ │ │ + if (sketchStyle) { │ │ │ │ │ + this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults(this.handlerOptions.layerOptions, { │ │ │ │ │ + styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ + default: sketchStyle │ │ │ │ │ + }) │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + this.handler = new handler(this, this.callbacks, this.handlerOptions) │ │ │ │ │ + }, │ │ │ │ │ + drawFeature: function(geometry) { │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + var proceed = this.layer.events.triggerEvent("sketchcomplete", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + if (proceed !== false) { │ │ │ │ │ + feature.state = OpenLayers.State.INSERT; │ │ │ │ │ + this.layer.addFeatures([feature]); │ │ │ │ │ + this.featureAdded(feature); │ │ │ │ │ + this.events.triggerEvent("featureadded", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + insertXY: function(x, y) { │ │ │ │ │ + if (this.handler && this.handler.line) { │ │ │ │ │ + this.handler.insertXY(x, y) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + insertDeltaXY: function(dx, dy) { │ │ │ │ │ + if (this.handler && this.handler.line) { │ │ │ │ │ + this.handler.insertDeltaXY(dx, dy) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + insertDirectionLength: function(direction, length) { │ │ │ │ │ + if (this.handler && this.handler.line) { │ │ │ │ │ + this.handler.insertDirectionLength(direction, length) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + insertDeflectionLength: function(deflection, length) { │ │ │ │ │ + if (this.handler && this.handler.line) { │ │ │ │ │ + this.handler.insertDeflectionLength(deflection, length) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + undo: function() { │ │ │ │ │ + return this.handler.undo && this.handler.undo() │ │ │ │ │ + }, │ │ │ │ │ + redo: function() { │ │ │ │ │ + return this.handler.redo && this.handler.redo() │ │ │ │ │ + }, │ │ │ │ │ + finishSketch: function() { │ │ │ │ │ + this.handler.finishGeometry() │ │ │ │ │ + }, │ │ │ │ │ + cancel: function() { │ │ │ │ │ + this.handler.cancel() │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.DrawFeature" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Panel = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + controls: null, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + defaultControl: null, │ │ │ │ │ + saveState: false, │ │ │ │ │ + allowDepress: false, │ │ │ │ │ + activeState: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.controls = []; │ │ │ │ │ + this.activeState = {} │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onButtonClick) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + for (var ctl, i = this.controls.length - 1; i >= 0; i--) { │ │ │ │ │ + ctl = this.controls[i]; │ │ │ │ │ + if (ctl.events) { │ │ │ │ │ + ctl.events.un({ │ │ │ │ │ + activate: this.iconOn, │ │ │ │ │ + deactivate: this.iconOff │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + ctl.panel_div = null │ │ │ │ │ + } │ │ │ │ │ + this.activeState = null │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + control = this.controls[i]; │ │ │ │ │ + if (control === this.defaultControl || this.saveState && this.activeState[control.id]) { │ │ │ │ │ + control.activate() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.saveState === true) { │ │ │ │ │ + this.defaultControl = null │ │ │ │ │ + } │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + control = this.controls[i]; │ │ │ │ │ + this.activeState[control.id] = control.deactivate() │ │ │ │ │ + } │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (this.outsideViewport) { │ │ │ │ │ + this.events.attachToElement(this.div); │ │ │ │ │ + this.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ + } else { │ │ │ │ │ + this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ + } │ │ │ │ │ + this.addControlsToMap(this.controls); │ │ │ │ │ + return this.div │ │ │ │ │ + }, │ │ │ │ │ + redraw: function() { │ │ │ │ │ + for (var l = this.div.childNodes.length, i = l - 1; i >= 0; i--) { │ │ │ │ │ + this.div.removeChild(this.div.childNodes[i]) │ │ │ │ │ + } │ │ │ │ │ + this.div.innerHTML = ""; │ │ │ │ │ + if (this.active) { │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + this.div.appendChild(this.controls[i].panel_div) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + activateControl: function(control) { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + if (control.type == OpenLayers.Control.TYPE_BUTTON) { │ │ │ │ │ + control.trigger(); │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (control.type == OpenLayers.Control.TYPE_TOGGLE) { │ │ │ │ │ + if (control.active) { │ │ │ │ │ + control.deactivate() │ │ │ │ │ + } else { │ │ │ │ │ + control.activate() │ │ │ │ │ + } │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (this.allowDepress && control.active) { │ │ │ │ │ + control.deactivate() │ │ │ │ │ + } else { │ │ │ │ │ + var c; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + c = this.controls[i]; │ │ │ │ │ + if (c != control && (c.type === OpenLayers.Control.TYPE_TOOL || c.type == null)) { │ │ │ │ │ + c.deactivate() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + control.activate() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addControls: function(controls) { │ │ │ │ │ + if (!OpenLayers.Util.isArray(controls)) { │ │ │ │ │ + controls = [controls] │ │ │ │ │ + } │ │ │ │ │ + this.controls = this.controls.concat(controls); │ │ │ │ │ + for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ + var control = controls[i], │ │ │ │ │ + element = this.createControlMarkup(control); │ │ │ │ │ + OpenLayers.Element.addClass(element, control.displayClass + "ItemInactive"); │ │ │ │ │ + OpenLayers.Element.addClass(element, "olButton"); │ │ │ │ │ + if (control.title != "" && !element.title) { │ │ │ │ │ + element.title = control.title │ │ │ │ │ + } │ │ │ │ │ + control.panel_div = element │ │ │ │ │ + } │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.addControlsToMap(controls); │ │ │ │ │ + this.redraw() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + createControlMarkup: function(control) { │ │ │ │ │ + return document.createElement("div") │ │ │ │ │ + }, │ │ │ │ │ + addControlsToMap: function(controls) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ + control = controls[i]; │ │ │ │ │ + if (control.autoActivate === true) { │ │ │ │ │ + control.autoActivate = false; │ │ │ │ │ + this.map.addControl(control); │ │ │ │ │ + control.autoActivate = true │ │ │ │ │ + } else { │ │ │ │ │ + this.map.addControl(control); │ │ │ │ │ + control.deactivate() │ │ │ │ │ + } │ │ │ │ │ + control.events.on({ │ │ │ │ │ + activate: this.iconOn, │ │ │ │ │ + deactivate: this.iconOff │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + iconOn: function() { │ │ │ │ │ + var d = this.panel_div; │ │ │ │ │ + var re = new RegExp("\\b(" + this.displayClass + "Item)Inactive\\b"); │ │ │ │ │ + d.className = d.className.replace(re, "$1Active") │ │ │ │ │ + }, │ │ │ │ │ + iconOff: function() { │ │ │ │ │ + var d = this.panel_div; │ │ │ │ │ + var re = new RegExp("\\b(" + this.displayClass + "Item)Active\\b"); │ │ │ │ │ + d.className = d.className.replace(re, "$1Inactive") │ │ │ │ │ + }, │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + var controls = this.controls, │ │ │ │ │ + button = evt.buttonElement; │ │ │ │ │ + for (var i = controls.length - 1; i >= 0; --i) { │ │ │ │ │ + if (controls[i].panel_div === button) { │ │ │ │ │ + this.activateControl(controls[i]); │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + getControlsBy: function(property, match) { │ │ │ │ │ + var test = typeof match.test == "function"; │ │ │ │ │ + var found = OpenLayers.Array.filter(this.controls, function(item) { │ │ │ │ │ + return item[property] == match || test && match.test(item[property]) │ │ │ │ │ + }); │ │ │ │ │ + return found │ │ │ │ │ + }, │ │ │ │ │ + getControlsByName: function(match) { │ │ │ │ │ + return this.getControlsBy("name", match) │ │ │ │ │ + }, │ │ │ │ │ + getControlsByClass: function(match) { │ │ │ │ │ + return this.getControlsBy("CLASS_NAME", match) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Panel" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Handler.Feature = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + EVENTMAP: { │ │ │ │ │ + click: { │ │ │ │ │ + in: "click", │ │ │ │ │ + out: "clickout" │ │ │ │ │ + }, │ │ │ │ │ + mousemove: { │ │ │ │ │ + in: "over", │ │ │ │ │ + out: "out" │ │ │ │ │ + }, │ │ │ │ │ + dblclick: { │ │ │ │ │ + in: "dblclick", │ │ │ │ │ + out: null │ │ │ │ │ + }, │ │ │ │ │ + mousedown: { │ │ │ │ │ + in: null, │ │ │ │ │ + out: null │ │ │ │ │ + }, │ │ │ │ │ + mouseup: { │ │ │ │ │ + in: null, │ │ │ │ │ + out: null │ │ │ │ │ + }, │ │ │ │ │ + touchstart: { │ │ │ │ │ + in: "click", │ │ │ │ │ + out: "clickout" │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + feature: null, │ │ │ │ │ + lastFeature: null, │ │ │ │ │ + down: null, │ │ │ │ │ + up: null, │ │ │ │ │ + clickTolerance: 4, │ │ │ │ │ + geometryTypes: null, │ │ │ │ │ + stopClick: true, │ │ │ │ │ + stopDown: true, │ │ │ │ │ + stopUp: false, │ │ │ │ │ + initialize: function(control, layer, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ + this.layer = layer │ │ │ │ │ + }, │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + return OpenLayers.Event.isMultiTouch(evt) ? true : this.mousedown(evt) │ │ │ │ │ + }, │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + OpenLayers.Event.preventDefault(evt) │ │ │ │ │ + }, │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + if (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ + this.down = evt.xy │ │ │ │ │ + } │ │ │ │ │ + return this.handle(evt) ? !this.stopDown : true │ │ │ │ │ + }, │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + this.up = evt.xy; │ │ │ │ │ + return this.handle(evt) ? !this.stopUp : true │ │ │ │ │ + }, │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + return this.handle(evt) ? !this.stopClick : true │ │ │ │ │ + }, │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + if (!this.callbacks["over"] && !this.callbacks["out"]) { │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + this.handle(evt); │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + return !this.handle(evt) │ │ │ │ │ + }, │ │ │ │ │ + geometryTypeMatches: function(feature) { │ │ │ │ │ + return this.geometryTypes == null || OpenLayers.Util.indexOf(this.geometryTypes, feature.geometry.CLASS_NAME) > -1 │ │ │ │ │ + }, │ │ │ │ │ + handle: function(evt) { │ │ │ │ │ + if (this.feature && !this.feature.layer) { │ │ │ │ │ + this.feature = null │ │ │ │ │ + } │ │ │ │ │ + var type = evt.type; │ │ │ │ │ + var handled = false; │ │ │ │ │ + var previouslyIn = !!this.feature; │ │ │ │ │ + var click = type == "click" || type == "dblclick" || type == "touchstart"; │ │ │ │ │ + this.feature = this.layer.getFeatureFromEvent(evt); │ │ │ │ │ + if (this.feature && !this.feature.layer) { │ │ │ │ │ + this.feature = null │ │ │ │ │ + } │ │ │ │ │ + if (this.lastFeature && !this.lastFeature.layer) { │ │ │ │ │ + this.lastFeature = null │ │ │ │ │ + } │ │ │ │ │ + if (this.feature) { │ │ │ │ │ + if (type === "touchstart") { │ │ │ │ │ + OpenLayers.Event.preventDefault(evt) │ │ │ │ │ + } │ │ │ │ │ + var inNew = this.feature != this.lastFeature; │ │ │ │ │ + if (this.geometryTypeMatches(this.feature)) { │ │ │ │ │ + if (previouslyIn && inNew) { │ │ │ │ │ + if (this.lastFeature) { │ │ │ │ │ + this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ + } │ │ │ │ │ + this.triggerCallback(type, "in", [this.feature]) │ │ │ │ │ + } else if (!previouslyIn || click) { │ │ │ │ │ + this.triggerCallback(type, "in", [this.feature]) │ │ │ │ │ + } │ │ │ │ │ + this.lastFeature = this.feature; │ │ │ │ │ + handled = true │ │ │ │ │ + } else { │ │ │ │ │ + if (this.lastFeature && (previouslyIn && inNew || click)) { │ │ │ │ │ + this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ + } │ │ │ │ │ + this.feature = null │ │ │ │ │ + } │ │ │ │ │ + } else if (this.lastFeature && (previouslyIn || click)) { │ │ │ │ │ + this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ + } │ │ │ │ │ + return handled │ │ │ │ │ + }, │ │ │ │ │ + triggerCallback: function(type, mode, args) { │ │ │ │ │ + var key = this.EVENTMAP[type][mode]; │ │ │ │ │ + if (key) { │ │ │ │ │ + if (type == "click" && this.up && this.down) { │ │ │ │ │ + var dpx = Math.sqrt(Math.pow(this.up.x - this.down.x, 2) + Math.pow(this.up.y - this.down.y, 2)); │ │ │ │ │ + if (dpx <= this.clickTolerance) { │ │ │ │ │ + this.callback(key, args) │ │ │ │ │ + } │ │ │ │ │ + this.up = this.down = null │ │ │ │ │ + } else { │ │ │ │ │ + this.callback(key, args) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerToTop(); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + removelayer: this.handleMapEvents, │ │ │ │ │ + changelayer: this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + activated = true │ │ │ │ │ + } │ │ │ │ │ + return activated │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerBack(); │ │ │ │ │ + this.feature = null; │ │ │ │ │ + this.lastFeature = null; │ │ │ │ │ + this.down = null; │ │ │ │ │ + this.up = null; │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + removelayer: this.handleMapEvents, │ │ │ │ │ + changelayer: this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + deactivated = true │ │ │ │ │ + } │ │ │ │ │ + return deactivated │ │ │ │ │ + }, │ │ │ │ │ + handleMapEvents: function(evt) { │ │ │ │ │ + if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ + this.moveLayerToTop() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + moveLayerToTop: function() { │ │ │ │ │ + var index = Math.max(this.map.Z_INDEX_BASE["Feature"] - 1, this.layer.getZIndex()) + 1; │ │ │ │ │ + this.layer.setZIndex(index) │ │ │ │ │ + }, │ │ │ │ │ + moveLayerBack: function() { │ │ │ │ │ + var index = this.layer.getZIndex() - 1; │ │ │ │ │ + if (index >= this.map.Z_INDEX_BASE["Feature"]) { │ │ │ │ │ + this.layer.setZIndex(index) │ │ │ │ │ + } else { │ │ │ │ │ + this.map.setLayerZIndex(this.layer, this.map.getLayerIndex(this.layer)) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Feature" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.StyleMap = OpenLayers.Class({ │ │ │ │ │ + styles: null, │ │ │ │ │ + extendDefault: true, │ │ │ │ │ + initialize: function(style, options) { │ │ │ │ │ + this.styles = { │ │ │ │ │ + default: new OpenLayers.Style(OpenLayers.Feature.Vector.style["default"]), │ │ │ │ │ + select: new OpenLayers.Style(OpenLayers.Feature.Vector.style["select"]), │ │ │ │ │ + temporary: new OpenLayers.Style(OpenLayers.Feature.Vector.style["temporary"]), │ │ │ │ │ + delete: new OpenLayers.Style(OpenLayers.Feature.Vector.style["delete"]) │ │ │ │ │ + }; │ │ │ │ │ + if (style instanceof OpenLayers.Style) { │ │ │ │ │ + this.styles["default"] = style; │ │ │ │ │ + this.styles["select"] = style; │ │ │ │ │ + this.styles["temporary"] = style; │ │ │ │ │ + this.styles["delete"] = style │ │ │ │ │ + } else if (typeof style == "object") { │ │ │ │ │ + for (var key in style) { │ │ │ │ │ + if (style[key] instanceof OpenLayers.Style) { │ │ │ │ │ + this.styles[key] = style[key] │ │ │ │ │ + } else if (typeof style[key] == "object") { │ │ │ │ │ + this.styles[key] = new OpenLayers.Style(style[key]) │ │ │ │ │ + } else { │ │ │ │ │ + this.styles["default"] = new OpenLayers.Style(style); │ │ │ │ │ + this.styles["select"] = new OpenLayers.Style(style); │ │ │ │ │ + this.styles["temporary"] = new OpenLayers.Style(style); │ │ │ │ │ + this.styles["delete"] = new OpenLayers.Style(style); │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.extend(this, options) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var key in this.styles) { │ │ │ │ │ + this.styles[key].destroy() │ │ │ │ │ + } │ │ │ │ │ + this.styles = null │ │ │ │ │ + }, │ │ │ │ │ + createSymbolizer: function(feature, intent) { │ │ │ │ │ + if (!feature) { │ │ │ │ │ + feature = new OpenLayers.Feature.Vector │ │ │ │ │ + } │ │ │ │ │ + if (!this.styles[intent]) { │ │ │ │ │ + intent = "default" │ │ │ │ │ + } │ │ │ │ │ + feature.renderIntent = intent; │ │ │ │ │ + var defaultSymbolizer = {}; │ │ │ │ │ + if (this.extendDefault && intent != "default") { │ │ │ │ │ + defaultSymbolizer = this.styles["default"].createSymbolizer(feature) │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Util.extend(defaultSymbolizer, this.styles[intent].createSymbolizer(feature)) │ │ │ │ │ + }, │ │ │ │ │ + addUniqueValueRules: function(renderIntent, property, symbolizers, context) { │ │ │ │ │ + var rules = []; │ │ │ │ │ + for (var value in symbolizers) { │ │ │ │ │ + rules.push(new OpenLayers.Rule({ │ │ │ │ │ + symbolizer: symbolizers[value], │ │ │ │ │ + context: context, │ │ │ │ │ + filter: new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.EQUAL_TO, │ │ │ │ │ + property: property, │ │ │ │ │ + value: value │ │ │ │ │ + }) │ │ │ │ │ + })) │ │ │ │ │ + } │ │ │ │ │ + this.styles[renderIntent].addRules(rules) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.StyleMap" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Vector = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ + isBaseLayer: false, │ │ │ │ │ + isFixed: false, │ │ │ │ │ + features: null, │ │ │ │ │ + filter: null, │ │ │ │ │ + selectedFeatures: null, │ │ │ │ │ + unrenderedFeatures: null, │ │ │ │ │ + reportError: true, │ │ │ │ │ + style: null, │ │ │ │ │ + styleMap: null, │ │ │ │ │ + strategies: null, │ │ │ │ │ + protocol: null, │ │ │ │ │ + renderers: ["SVG", "VML", "Canvas"], │ │ │ │ │ + renderer: null, │ │ │ │ │ + rendererOptions: null, │ │ │ │ │ + geometryType: null, │ │ │ │ │ + drawn: false, │ │ │ │ │ + ratio: 1, │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ + this.assignRenderer() │ │ │ │ │ + } │ │ │ │ │ + if (!this.renderer || !this.renderer.supported()) { │ │ │ │ │ + this.renderer = null; │ │ │ │ │ + this.displayError() │ │ │ │ │ + } │ │ │ │ │ + if (!this.styleMap) { │ │ │ │ │ + this.styleMap = new OpenLayers.StyleMap │ │ │ │ │ + } │ │ │ │ │ + this.features = []; │ │ │ │ │ + this.selectedFeatures = []; │ │ │ │ │ + this.unrenderedFeatures = {}; │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + for (var i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + this.strategies[i].setLayer(this) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoDestroy) { │ │ │ │ │ + strategy.destroy() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.strategies = null │ │ │ │ │ + } │ │ │ │ │ + if (this.protocol) { │ │ │ │ │ + if (this.protocol.autoDestroy) { │ │ │ │ │ + this.protocol.destroy() │ │ │ │ │ + } │ │ │ │ │ + this.protocol = null │ │ │ │ │ + } │ │ │ │ │ + this.destroyFeatures(); │ │ │ │ │ + this.features = null; │ │ │ │ │ + this.selectedFeatures = null; │ │ │ │ │ + this.unrenderedFeatures = null; │ │ │ │ │ + if (this.renderer) { │ │ │ │ │ + this.renderer.destroy() │ │ │ │ │ + } │ │ │ │ │ + this.renderer = null; │ │ │ │ │ + this.geometryType = null; │ │ │ │ │ + this.drawn = null; │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Vector(this.name, this.getOptions()) │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ + var features = this.features; │ │ │ │ │ + var len = features.length; │ │ │ │ │ + var clonedFeatures = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + clonedFeatures[i] = features[i].clone() │ │ │ │ │ + } │ │ │ │ │ + obj.features = clonedFeatures; │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + refresh: function(obj) { │ │ │ │ │ + if (this.calculateInRange() && this.visibility) { │ │ │ │ │ + this.events.triggerEvent("refresh", obj) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + assignRenderer: function() { │ │ │ │ │ + for (var i = 0, len = this.renderers.length; i < len; i++) { │ │ │ │ │ + var rendererClass = this.renderers[i]; │ │ │ │ │ + var renderer = typeof rendererClass == "function" ? rendererClass : OpenLayers.Renderer[rendererClass]; │ │ │ │ │ + if (renderer && renderer.prototype.supported()) { │ │ │ │ │ + this.renderer = new renderer(this.div, this.rendererOptions); │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + displayError: function() { │ │ │ │ │ + if (this.reportError) { │ │ │ │ │ + OpenLayers.Console.userError(OpenLayers.i18n("browserNotSupported", { │ │ │ │ │ + renderers: this.renderers.join("\n") │ │ │ │ │ + })) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (!this.renderer) { │ │ │ │ │ + this.map.removeLayer(this) │ │ │ │ │ + } else { │ │ │ │ │ + this.renderer.map = this.map; │ │ │ │ │ + var newSize = this.map.getSize(); │ │ │ │ │ + newSize.w = newSize.w * this.ratio; │ │ │ │ │ + newSize.h = newSize.h * this.ratio; │ │ │ │ │ + this.renderer.setSize(newSize) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + afterAdd: function() { │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoActivate) { │ │ │ │ │ + strategy.activate() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + this.drawn = false; │ │ │ │ │ + if (this.strategies) { │ │ │ │ │ + var strategy, i, len; │ │ │ │ │ + for (i = 0, len = this.strategies.length; i < len; i++) { │ │ │ │ │ + strategy = this.strategies[i]; │ │ │ │ │ + if (strategy.autoActivate) { │ │ │ │ │ + strategy.deactivate() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + OpenLayers.Layer.prototype.onMapResize.apply(this, arguments); │ │ │ │ │ + var newSize = this.map.getSize(); │ │ │ │ │ + newSize.w = newSize.w * this.ratio; │ │ │ │ │ + newSize.h = newSize.h * this.ratio; │ │ │ │ │ + this.renderer.setSize(newSize) │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + var coordSysUnchanged = true; │ │ │ │ │ + if (!dragging) { │ │ │ │ │ + this.renderer.root.style.visibility = "hidden"; │ │ │ │ │ + var viewSize = this.map.getSize(), │ │ │ │ │ + viewWidth = viewSize.w, │ │ │ │ │ + viewHeight = viewSize.h, │ │ │ │ │ + offsetLeft = viewWidth / 2 * this.ratio - viewWidth / 2, │ │ │ │ │ + offsetTop = viewHeight / 2 * this.ratio - viewHeight / 2; │ │ │ │ │ + offsetLeft += this.map.layerContainerOriginPx.x; │ │ │ │ │ + offsetLeft = -Math.round(offsetLeft); │ │ │ │ │ + offsetTop += this.map.layerContainerOriginPx.y; │ │ │ │ │ + offsetTop = -Math.round(offsetTop); │ │ │ │ │ + this.div.style.left = offsetLeft + "px"; │ │ │ │ │ + this.div.style.top = offsetTop + "px"; │ │ │ │ │ + var extent = this.map.getExtent().scale(this.ratio); │ │ │ │ │ + coordSysUnchanged = this.renderer.setExtent(extent, zoomChanged); │ │ │ │ │ + this.renderer.root.style.visibility = "visible"; │ │ │ │ │ + if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ + this.div.scrollLeft = this.div.scrollLeft │ │ │ │ │ + } │ │ │ │ │ + if (!zoomChanged && coordSysUnchanged) { │ │ │ │ │ + for (var i in this.unrenderedFeatures) { │ │ │ │ │ + var feature = this.unrenderedFeatures[i]; │ │ │ │ │ + this.drawFeature(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!this.drawn || zoomChanged || !coordSysUnchanged) { │ │ │ │ │ + this.drawn = true; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0, len = this.features.length; i < len; i++) { │ │ │ │ │ + this.renderer.locked = i !== len - 1; │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + this.drawFeature(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + display: function(display) { │ │ │ │ │ + OpenLayers.Layer.prototype.display.apply(this, arguments); │ │ │ │ │ + var currentDisplay = this.div.style.display; │ │ │ │ │ + if (currentDisplay != this.renderer.root.style.display) { │ │ │ │ │ + this.renderer.root.style.display = currentDisplay │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addFeatures: function(features, options) { │ │ │ │ │ + if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ + features = [features] │ │ │ │ │ + } │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ + if (notify) { │ │ │ │ │ + var event = { │ │ │ │ │ + features: features │ │ │ │ │ + }; │ │ │ │ │ + var ret = this.events.triggerEvent("beforefeaturesadded", event); │ │ │ │ │ + if (ret === false) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + features = event.features │ │ │ │ │ + } │ │ │ │ │ + var featuresAdded = []; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + if (i != features.length - 1) { │ │ │ │ │ + this.renderer.locked = true │ │ │ │ │ + } else { │ │ │ │ │ + this.renderer.locked = false │ │ │ │ │ + } │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + if (this.geometryType && !(feature.geometry instanceof this.geometryType)) { │ │ │ │ │ + throw new TypeError("addFeatures: component should be an " + this.geometryType.prototype.CLASS_NAME) │ │ │ │ │ + } │ │ │ │ │ + feature.layer = this; │ │ │ │ │ + if (!feature.style && this.style) { │ │ │ │ │ + feature.style = OpenLayers.Util.extend({}, this.style) │ │ │ │ │ + } │ │ │ │ │ + if (notify) { │ │ │ │ │ + if (this.events.triggerEvent("beforefeatureadded", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) === false) { │ │ │ │ │ + continue │ │ │ │ │ + } │ │ │ │ │ + this.preFeatureInsert(feature) │ │ │ │ │ + } │ │ │ │ │ + featuresAdded.push(feature); │ │ │ │ │ + this.features.push(feature); │ │ │ │ │ + this.drawFeature(feature); │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureadded", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.onFeatureInsert(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresadded", { │ │ │ │ │ + features: featuresAdded │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + removeFeatures: function(features, options) { │ │ │ │ │ + if (!features || features.length === 0) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (features === this.features) { │ │ │ │ │ + return this.removeAllFeatures(options) │ │ │ │ │ + } │ │ │ │ │ + if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ + features = [features] │ │ │ │ │ + } │ │ │ │ │ + if (features === this.selectedFeatures) { │ │ │ │ │ + features = features.slice() │ │ │ │ │ + } │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeaturesremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + if (i != 0 && features[i - 1].geometry) { │ │ │ │ │ + this.renderer.locked = true │ │ │ │ │ + } else { │ │ │ │ │ + this.renderer.locked = false │ │ │ │ │ + } │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + delete this.unrenderedFeatures[feature.id]; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + this.features = OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ + feature.layer = null; │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + this.renderer.eraseFeatures(feature) │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.selectedFeatures, feature) != -1) { │ │ │ │ │ + OpenLayers.Util.removeItem(this.selectedFeatures, feature) │ │ │ │ │ + } │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + removeAllFeatures: function(options) { │ │ │ │ │ + var notify = !options || !options.silent; │ │ │ │ │ + var features = this.features; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeaturesremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("beforefeatureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + feature.layer = null; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featureremoved", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.renderer.clear(); │ │ │ │ │ + this.features = []; │ │ │ │ │ + this.unrenderedFeatures = {}; │ │ │ │ │ + this.selectedFeatures = []; │ │ │ │ │ + if (notify) { │ │ │ │ │ + this.events.triggerEvent("featuresremoved", { │ │ │ │ │ + features: features │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroyFeatures: function(features, options) { │ │ │ │ │ + var all = features == undefined; │ │ │ │ │ + if (all) { │ │ │ │ │ + features = this.features │ │ │ │ │ + } │ │ │ │ │ + if (features) { │ │ │ │ │ + this.removeFeatures(features, options); │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + features[i].destroy() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + drawFeature: function(feature, style) { │ │ │ │ │ + if (!this.drawn) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (typeof style != "object") { │ │ │ │ │ + if (!style && feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ + style = "delete" │ │ │ │ │ + } │ │ │ │ │ + var renderIntent = style || feature.renderIntent; │ │ │ │ │ + style = feature.style || this.style; │ │ │ │ │ + if (!style) { │ │ │ │ │ + style = this.styleMap.createSymbolizer(feature, renderIntent) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var drawn = this.renderer.drawFeature(feature, style); │ │ │ │ │ + if (drawn === false || drawn === null) { │ │ │ │ │ + this.unrenderedFeatures[feature.id] = feature │ │ │ │ │ + } else { │ │ │ │ │ + delete this.unrenderedFeatures[feature.id] │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + eraseFeatures: function(features) { │ │ │ │ │ + this.renderer.eraseFeatures(features) │ │ │ │ │ + }, │ │ │ │ │ + getFeatureFromEvent: function(evt) { │ │ │ │ │ + if (!this.renderer) { │ │ │ │ │ + throw new Error("getFeatureFromEvent called on layer with no " + "renderer. This usually means you destroyed a " + "layer, but not some handler which is associated " + "with it.") │ │ │ │ │ + } │ │ │ │ │ + var feature = null; │ │ │ │ │ + var featureId = this.renderer.getFeatureIdFromEvent(evt); │ │ │ │ │ + if (featureId) { │ │ │ │ │ + if (typeof featureId === "string") { │ │ │ │ │ + feature = this.getFeatureById(featureId) │ │ │ │ │ + } else { │ │ │ │ │ + feature = featureId │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return feature │ │ │ │ │ + }, │ │ │ │ │ + getFeatureBy: function(property, value) { │ │ │ │ │ + var feature = null; │ │ │ │ │ + for (var i = 0, len = this.features.length; i < len; ++i) { │ │ │ │ │ + if (this.features[i][property] == value) { │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return feature │ │ │ │ │ + }, │ │ │ │ │ + getFeatureById: function(featureId) { │ │ │ │ │ + return this.getFeatureBy("id", featureId) │ │ │ │ │ + }, │ │ │ │ │ + getFeatureByFid: function(featureFid) { │ │ │ │ │ + return this.getFeatureBy("fid", featureFid) │ │ │ │ │ + }, │ │ │ │ │ + getFeaturesByAttribute: function(attrName, attrValue) { │ │ │ │ │ + var i, feature, len = this.features.length, │ │ │ │ │ + foundFeatures = []; │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + if (feature && feature.attributes) { │ │ │ │ │ + if (feature.attributes[attrName] === attrValue) { │ │ │ │ │ + foundFeatures.push(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return foundFeatures │ │ │ │ │ + }, │ │ │ │ │ + onFeatureInsert: function(feature) {}, │ │ │ │ │ + preFeatureInsert: function(feature) {}, │ │ │ │ │ + getDataExtent: function() { │ │ │ │ │ + var maxExtent = null; │ │ │ │ │ + var features = this.features; │ │ │ │ │ + if (features && features.length > 0) { │ │ │ │ │ + var geometry = null; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + geometry = features[i].geometry; │ │ │ │ │ + if (geometry) { │ │ │ │ │ + if (maxExtent === null) { │ │ │ │ │ + maxExtent = new OpenLayers.Bounds │ │ │ │ │ + } │ │ │ │ │ + maxExtent.extend(geometry.getBounds()) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return maxExtent │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Vector" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Vector.RootContainer = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + layers: null, │ │ │ │ │ + display: function() {}, │ │ │ │ │ + getFeatureFromEvent: function(evt) { │ │ │ │ │ + var layers = this.layers; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0; i < layers.length; i++) { │ │ │ │ │ + feature = layers[i].getFeatureFromEvent(evt); │ │ │ │ │ + if (feature) { │ │ │ │ │ + return feature │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.collectRoots(); │ │ │ │ │ + map.events.register("changelayer", this, this.handleChangeLayer) │ │ │ │ │ + }, │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + map.events.unregister("changelayer", this, this.handleChangeLayer); │ │ │ │ │ + this.resetRoots(); │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + collectRoots: function() { │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = 0; i < this.map.layers.length; ++i) { │ │ │ │ │ + layer = this.map.layers[i]; │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ + layer.renderer.moveRoot(this.renderer) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + resetRoots: function() { │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = 0; i < this.layers.length; ++i) { │ │ │ │ │ + layer = this.layers[i]; │ │ │ │ │ + if (this.renderer && layer.renderer.getRenderLayerId() == this.id) { │ │ │ │ │ + this.renderer.moveRoot(layer.renderer) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + handleChangeLayer: function(evt) { │ │ │ │ │ + var layer = evt.layer; │ │ │ │ │ + if (evt.property == "order" && OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ + this.resetRoots(); │ │ │ │ │ + this.collectRoots() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Vector.RootContainer" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.SelectFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + multipleKey: null, │ │ │ │ │ + toggleKey: null, │ │ │ │ │ + multiple: false, │ │ │ │ │ + clickout: true, │ │ │ │ │ + toggle: false, │ │ │ │ │ + hover: false, │ │ │ │ │ + highlightOnly: false, │ │ │ │ │ + box: false, │ │ │ │ │ + onBeforeSelect: function() {}, │ │ │ │ │ + onSelect: function() {}, │ │ │ │ │ + onUnselect: function() {}, │ │ │ │ │ + scope: null, │ │ │ │ │ + geometryTypes: null, │ │ │ │ │ + layer: null, │ │ │ │ │ + layers: null, │ │ │ │ │ + callbacks: null, │ │ │ │ │ + selectStyle: null, │ │ │ │ │ + renderIntent: "select", │ │ │ │ │ + handlers: null, │ │ │ │ │ + initialize: function(layers, options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (this.scope === null) { │ │ │ │ │ + this.scope = this │ │ │ │ │ + } │ │ │ │ │ + this.initLayer(layers); │ │ │ │ │ + var callbacks = { │ │ │ │ │ + click: this.clickFeature, │ │ │ │ │ + clickout: this.clickoutFeature │ │ │ │ │ + }; │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + callbacks.over = this.overFeature; │ │ │ │ │ + callbacks.out = this.outFeature │ │ │ │ │ + } │ │ │ │ │ + this.callbacks = OpenLayers.Util.extend(callbacks, this.callbacks); │ │ │ │ │ + this.handlers = { │ │ │ │ │ + feature: new OpenLayers.Handler.Feature(this, this.layer, this.callbacks, { │ │ │ │ │ + geometryTypes: this.geometryTypes │ │ │ │ │ + }) │ │ │ │ │ + }; │ │ │ │ │ + if (this.box) { │ │ │ │ │ + this.handlers.box = new OpenLayers.Handler.Box(this, { │ │ │ │ │ + done: this.selectBox │ │ │ │ │ + }, { │ │ │ │ │ + boxDivClassName: "olHandlerBoxSelectFeature" │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + initLayer: function(layers) { │ │ │ │ │ + if (OpenLayers.Util.isArray(layers)) { │ │ │ │ │ + this.layers = layers; │ │ │ │ │ + this.layer = new OpenLayers.Layer.Vector.RootContainer(this.id + "_container", { │ │ │ │ │ + layers: layers │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + this.layer = layers │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.active && this.layers) { │ │ │ │ │ + this.map.removeLayer(this.layer) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + if (this.layers) { │ │ │ │ │ + this.layer.destroy() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + if (this.layers) { │ │ │ │ │ + this.map.addLayer(this.layer) │ │ │ │ │ + } │ │ │ │ │ + this.handlers.feature.activate(); │ │ │ │ │ + if (this.box && this.handlers.box) { │ │ │ │ │ + this.handlers.box.activate() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Control.prototype.activate.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.handlers.feature.deactivate(); │ │ │ │ │ + if (this.handlers.box) { │ │ │ │ │ + this.handlers.box.deactivate() │ │ │ │ │ + } │ │ │ │ │ + if (this.layers) { │ │ │ │ │ + this.map.removeLayer(this.layer) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Control.prototype.deactivate.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + unselectAll: function(options) { │ │ │ │ │ + var layers = this.layers || [this.layer], │ │ │ │ │ + layer, feature, l, numExcept; │ │ │ │ │ + for (l = 0; l < layers.length; ++l) { │ │ │ │ │ + layer = layers[l]; │ │ │ │ │ + numExcept = 0; │ │ │ │ │ + if (layer.selectedFeatures != null) { │ │ │ │ │ + while (layer.selectedFeatures.length > numExcept) { │ │ │ │ │ + feature = layer.selectedFeatures[numExcept]; │ │ │ │ │ + if (!options || options.except != feature) { │ │ │ │ │ + this.unselect(feature) │ │ │ │ │ + } else { │ │ │ │ │ + ++numExcept │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + clickFeature: function(feature) { │ │ │ │ │ + if (!this.hover) { │ │ │ │ │ + var selected = OpenLayers.Util.indexOf(feature.layer.selectedFeatures, feature) > -1; │ │ │ │ │ + if (selected) { │ │ │ │ │ + if (this.toggleSelect()) { │ │ │ │ │ + this.unselect(feature) │ │ │ │ │ + } else if (!this.multipleSelect()) { │ │ │ │ │ + this.unselectAll({ │ │ │ │ │ + except: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + if (!this.multipleSelect()) { │ │ │ │ │ + this.unselectAll({ │ │ │ │ │ + except: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + this.select(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + multipleSelect: function() { │ │ │ │ │ + return this.multiple || this.handlers.feature.evt && this.handlers.feature.evt[this.multipleKey] │ │ │ │ │ + }, │ │ │ │ │ + toggleSelect: function() { │ │ │ │ │ + return this.toggle || this.handlers.feature.evt && this.handlers.feature.evt[this.toggleKey] │ │ │ │ │ + }, │ │ │ │ │ + clickoutFeature: function(feature) { │ │ │ │ │ + if (!this.hover && this.clickout) { │ │ │ │ │ + this.unselectAll() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + overFeature: function(feature) { │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + if (this.highlightOnly) { │ │ │ │ │ + this.highlight(feature) │ │ │ │ │ + } else if (OpenLayers.Util.indexOf(layer.selectedFeatures, feature) == -1) { │ │ │ │ │ + this.select(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + outFeature: function(feature) { │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + if (this.highlightOnly) { │ │ │ │ │ + if (feature._lastHighlighter == this.id) { │ │ │ │ │ + if (feature._prevHighlighter && feature._prevHighlighter != this.id) { │ │ │ │ │ + delete feature._lastHighlighter; │ │ │ │ │ + var control = this.map.getControl(feature._prevHighlighter); │ │ │ │ │ + if (control) { │ │ │ │ │ + control.highlight(feature) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this.unhighlight(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this.unselect(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + highlight: function(feature) { │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + var cont = this.events.triggerEvent("beforefeaturehighlighted", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + feature._prevHighlighter = feature._lastHighlighter; │ │ │ │ │ + feature._lastHighlighter = this.id; │ │ │ │ │ + var style = this.selectStyle || this.renderIntent; │ │ │ │ │ + layer.drawFeature(feature, style); │ │ │ │ │ + this.events.triggerEvent("featurehighlighted", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + unhighlight: function(feature) { │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + if (feature._prevHighlighter == undefined) { │ │ │ │ │ + delete feature._lastHighlighter │ │ │ │ │ + } else if (feature._prevHighlighter == this.id) { │ │ │ │ │ + delete feature._prevHighlighter │ │ │ │ │ + } else { │ │ │ │ │ + feature._lastHighlighter = feature._prevHighlighter; │ │ │ │ │ + delete feature._prevHighlighter │ │ │ │ │ + } │ │ │ │ │ + layer.drawFeature(feature, feature.style || feature.layer.style || "default"); │ │ │ │ │ + this.events.triggerEvent("featureunhighlighted", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + select: function(feature) { │ │ │ │ │ + var cont = this.onBeforeSelect.call(this.scope, feature); │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + cont = layer.events.triggerEvent("beforefeatureselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + layer.selectedFeatures.push(feature); │ │ │ │ │ + this.highlight(feature); │ │ │ │ │ + if (!this.handlers.feature.lastFeature) { │ │ │ │ │ + this.handlers.feature.lastFeature = layer.selectedFeatures[0] │ │ │ │ │ + } │ │ │ │ │ + layer.events.triggerEvent("featureselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.onSelect.call(this.scope, feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + unselect: function(feature) { │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + this.unhighlight(feature); │ │ │ │ │ + OpenLayers.Util.removeItem(layer.selectedFeatures, feature); │ │ │ │ │ + layer.events.triggerEvent("featureunselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.onUnselect.call(this.scope, feature) │ │ │ │ │ + }, │ │ │ │ │ + selectBox: function(position) { │ │ │ │ │ + if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ + var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.left, │ │ │ │ │ + y: position.bottom │ │ │ │ │ + }); │ │ │ │ │ + var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.right, │ │ │ │ │ + y: position.top │ │ │ │ │ + }); │ │ │ │ │ + var bounds = new OpenLayers.Bounds(minXY.lon, minXY.lat, maxXY.lon, maxXY.lat); │ │ │ │ │ + if (!this.multipleSelect()) { │ │ │ │ │ + this.unselectAll() │ │ │ │ │ + } │ │ │ │ │ + var prevMultiple = this.multiple; │ │ │ │ │ + this.multiple = true; │ │ │ │ │ + var layers = this.layers || [this.layer]; │ │ │ │ │ + this.events.triggerEvent("boxselectionstart", { │ │ │ │ │ + layers: layers │ │ │ │ │ + }); │ │ │ │ │ + var layer; │ │ │ │ │ + for (var l = 0; l < layers.length; ++l) { │ │ │ │ │ + layer = layers[l]; │ │ │ │ │ + for (var i = 0, len = layer.features.length; i < len; ++i) { │ │ │ │ │ + var feature = layer.features[i]; │ │ │ │ │ + if (!feature.getVisibility()) { │ │ │ │ │ + continue │ │ │ │ │ + } │ │ │ │ │ + if (this.geometryTypes == null || OpenLayers.Util.indexOf(this.geometryTypes, feature.geometry.CLASS_NAME) > -1) { │ │ │ │ │ + if (bounds.toGeometry().intersects(feature.geometry)) { │ │ │ │ │ + if (OpenLayers.Util.indexOf(layer.selectedFeatures, feature) == -1) { │ │ │ │ │ + this.select(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.multiple = prevMultiple; │ │ │ │ │ + this.events.triggerEvent("boxselectionend", { │ │ │ │ │ + layers: layers │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.handlers.feature.setMap(map); │ │ │ │ │ + if (this.box) { │ │ │ │ │ + this.handlers.box.setMap(map) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + setLayer: function(layers) { │ │ │ │ │ + var isActive = this.active; │ │ │ │ │ + this.unselectAll(); │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + if (this.layers) { │ │ │ │ │ + this.layer.destroy(); │ │ │ │ │ + this.layers = null │ │ │ │ │ + } │ │ │ │ │ + this.initLayer(layers); │ │ │ │ │ + this.handlers.feature.layer = this.layer; │ │ │ │ │ + if (isActive) { │ │ │ │ │ + this.activate() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.SelectFeature" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Attribution = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + separator: ", ", │ │ │ │ │ + template: "${layers}", │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + removelayer: this.updateAttribution, │ │ │ │ │ + addlayer: this.updateAttribution, │ │ │ │ │ + changelayer: this.updateAttribution, │ │ │ │ │ + changebaselayer: this.updateAttribution, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + changebaselayer: this.updateAttribution, │ │ │ │ │ + changelayer: this.updateAttribution, │ │ │ │ │ + addlayer: this.updateAttribution, │ │ │ │ │ + removelayer: this.updateAttribution, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.updateAttribution(); │ │ │ │ │ + return this.div │ │ │ │ │ + }, │ │ │ │ │ + updateAttribution: function() { │ │ │ │ │ + var attributions = []; │ │ │ │ │ + if (this.map && this.map.layers) { │ │ │ │ │ + for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if (layer.attribution && layer.getVisibility()) { │ │ │ │ │ + if (OpenLayers.Util.indexOf(attributions, layer.attribution) === -1) { │ │ │ │ │ + attributions.push(layer.attribution) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.div.innerHTML = OpenLayers.String.format(this.template, { │ │ │ │ │ + layers: attributions.join(this.separator) │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Attribution" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Geolocate = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + geolocation: null, │ │ │ │ │ + available: "geolocation" in navigator, │ │ │ │ │ + bind: true, │ │ │ │ │ + watch: false, │ │ │ │ │ + geolocationOptions: null, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (this.available && !this.geolocation) { │ │ │ │ │ + this.geolocation = navigator.geolocation │ │ │ │ │ + } │ │ │ │ │ + if (!this.geolocation) { │ │ │ │ │ + this.events.triggerEvent("locationuncapable"); │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + if (this.watch) { │ │ │ │ │ + this.watchId = this.geolocation.watchPosition(OpenLayers.Function.bind(this.geolocate, this), OpenLayers.Function.bind(this.failure, this), this.geolocationOptions) │ │ │ │ │ + } else { │ │ │ │ │ + this.getCurrentLocation() │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active && this.watchId !== null) { │ │ │ │ │ + this.geolocation.clearWatch(this.watchId) │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Control.prototype.deactivate.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + geolocate: function(position) { │ │ │ │ │ + var center = new OpenLayers.LonLat(position.coords.longitude, position.coords.latitude).transform(new OpenLayers.Projection("EPSG:4326"), this.map.getProjectionObject()); │ │ │ │ │ + if (this.bind) { │ │ │ │ │ + this.map.setCenter(center) │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("locationupdated", { │ │ │ │ │ + position: position, │ │ │ │ │ + point: new OpenLayers.Geometry.Point(center.lon, center.lat) │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + getCurrentLocation: function() { │ │ │ │ │ + if (!this.active || this.watch) { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + this.geolocation.getCurrentPosition(OpenLayers.Function.bind(this.geolocate, this), OpenLayers.Function.bind(this.failure, this), this.geolocationOptions); │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + failure: function(error) { │ │ │ │ │ + this.events.triggerEvent("locationfailed", { │ │ │ │ │ + error: error │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Geolocate" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Strategy = OpenLayers.Class({ │ │ │ │ │ + layer: null, │ │ │ │ │ + options: null, │ │ │ │ │ + active: null, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + autoDestroy: true, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.options = options; │ │ │ │ │ + this.active = false │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.layer = null; │ │ │ │ │ + this.options = null │ │ │ │ │ + }, │ │ │ │ │ + setLayer: function(layer) { │ │ │ │ │ + this.layer = layer │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + this.active = true; │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.active = false; │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Strategy.Fixed = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + preload: false, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + refresh: this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + if (this.layer.visibility == true || this.preload) { │ │ │ │ │ + this.load() │ │ │ │ │ + } else { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + visibilitychanged: this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return activated │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + refresh: this.load, │ │ │ │ │ + visibilitychanged: this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + return deactivated │ │ │ │ │ + }, │ │ │ │ │ + load: function(options) { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + layer.events.triggerEvent("loadstart", { │ │ │ │ │ + filter: layer.filter │ │ │ │ │ + }); │ │ │ │ │ + layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: this.merge, │ │ │ │ │ + filter: layer.filter, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options)); │ │ │ │ │ + layer.events.un({ │ │ │ │ │ + visibilitychanged: this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + merge: function(resp) { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + layer.destroyFeatures(); │ │ │ │ │ + var features = resp.features; │ │ │ │ │ + if (features && features.length > 0) { │ │ │ │ │ + var remote = layer.projection; │ │ │ │ │ + var local = layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var geom; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + geom = features[i].geometry; │ │ │ │ │ + if (geom) { │ │ │ │ │ + geom.transform(remote, local) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + layer.addFeatures(features) │ │ │ │ │ + } │ │ │ │ │ + layer.events.triggerEvent("loadend", { │ │ │ │ │ + response: resp │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Fixed" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.WMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + service: "WMS", │ │ │ │ │ + version: "1.1.1", │ │ │ │ │ + request: "GetMap", │ │ │ │ │ + styles: "", │ │ │ │ │ + format: "image/jpeg" │ │ │ │ │ + }, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + encodeBBOX: false, │ │ │ │ │ + noMagic: false, │ │ │ │ │ + yx: {}, │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ + if (parseFloat(params.VERSION) >= 1.3 && !params.EXCEPTIONS) { │ │ │ │ │ + params.EXCEPTIONS = "INIMAGE" │ │ │ │ │ + } │ │ │ │ │ + newArguments.push(name, url, params, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + OpenLayers.Util.applyDefaults(this.params, OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS)); │ │ │ │ │ + if (!this.noMagic && this.params.TRANSPARENT && this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ + if (options == null || !options.isBaseLayer) { │ │ │ │ │ + this.isBaseLayer = false │ │ │ │ │ + } │ │ │ │ │ + if (this.params.FORMAT == "image/jpeg") { │ │ │ │ │ + this.params.FORMAT = OpenLayers.Util.alphaHack() ? "image/gif" : "image/png" │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.WMS(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + reverseAxisOrder: function() { │ │ │ │ │ + var projCode = this.projection.getCode(); │ │ │ │ │ + return parseFloat(this.params.VERSION) >= 1.3 && !!(this.yx[projCode] || OpenLayers.Projection.defaults[projCode] && OpenLayers.Projection.defaults[projCode].yx) │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var imageSize = this.getImageSize(); │ │ │ │ │ + var newParams = {}; │ │ │ │ │ + var reverseAxisOrder = this.reverseAxisOrder(); │ │ │ │ │ + newParams.BBOX = this.encodeBBOX ? bounds.toBBOX(null, reverseAxisOrder) : bounds.toArray(reverseAxisOrder); │ │ │ │ │ + newParams.WIDTH = imageSize.w; │ │ │ │ │ + newParams.HEIGHT = imageSize.h; │ │ │ │ │ + var requestString = this.getFullRequestString(newParams); │ │ │ │ │ + return requestString │ │ │ │ │ + }, │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ + var newArguments = [upperParams]; │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, newArguments) │ │ │ │ │ + }, │ │ │ │ │ + getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ + var mapProjection = this.map.getProjectionObject(); │ │ │ │ │ + var projectionCode = this.projection && this.projection.equals(mapProjection) ? this.projection.getCode() : mapProjection.getCode(); │ │ │ │ │ + var value = projectionCode == "none" ? null : projectionCode; │ │ │ │ │ + if (parseFloat(this.params.VERSION) >= 1.3) { │ │ │ │ │ + this.params.CRS = value │ │ │ │ │ + } else { │ │ │ │ │ + this.params.SRS = value │ │ │ │ │ + } │ │ │ │ │ + if (typeof this.params.TRANSPARENT == "boolean") { │ │ │ │ │ + newParams.TRANSPARENT = this.params.TRANSPARENT ? "TRUE" : "FALSE" │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.getFullRequestString.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.WMS" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.XYZ = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + sphericalMercator: false, │ │ │ │ │ + zoomOffset: 0, │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + if (options && options.sphericalMercator || this.sphericalMercator) { │ │ │ │ │ + options = OpenLayers.Util.extend({ │ │ │ │ │ + projection: "EPSG:900913", │ │ │ │ │ + numZoomLevels: 19 │ │ │ │ │ + }, options) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name || this.name, url || this.url, {}, options]) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.XYZ(this.name, this.url, this.getOptions()) │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var xyz = this.getXYZ(bounds); │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + var s = "" + xyz.x + xyz.y + xyz.z; │ │ │ │ │ + url = this.selectUrl(s, url) │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.String.format(url, xyz) │ │ │ │ │ + }, │ │ │ │ │ + getXYZ: function(bounds) { │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.maxExtent.left) / (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((this.maxExtent.top - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getServerZoom(); │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + var limit = Math.pow(2, z); │ │ │ │ │ + x = (x % limit + limit) % limit │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + x: x, │ │ │ │ │ + y: y, │ │ │ │ │ + z: z │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (!this.tileOrigin) { │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, this.maxExtent.bottom) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.XYZ" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Bing = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ + key: null, │ │ │ │ │ + serverResolutions: [156543.03390625, 78271.516953125, 39135.7584765625, 19567.87923828125, 9783.939619140625, 4891.9698095703125, 2445.9849047851562, 1222.9924523925781, 611.4962261962891, 305.74811309814453, 152.87405654907226, 76.43702827453613, 38.218514137268066, 19.109257068634033, 9.554628534317017, 4.777314267158508, 2.388657133579254, 1.194328566789627, .5971642833948135, .29858214169740677, .14929107084870338, .07464553542435169], │ │ │ │ │ + attributionTemplate: '<span class="olBingAttribution ${type}">' + '<div><a target="_blank" href="http://www.bing.com/maps/">' + '<img src="${logo}" /></a></div>${copyrights}' + '<a style="white-space: nowrap" target="_blank" ' + 'href="http://www.microsoft.com/maps/product/terms.html">' + "Terms of Use</a></span>", │ │ │ │ │ + metadata: null, │ │ │ │ │ + protocolRegex: /^http:/i, │ │ │ │ │ + type: "Road", │ │ │ │ │ + culture: "en-US", │ │ │ │ │ + metadataParams: null, │ │ │ │ │ + tileOptions: null, │ │ │ │ │ + protocol: ~window.location.href.indexOf("http") ? "" : "http:", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + sphericalMercator: true │ │ │ │ │ + }, options); │ │ │ │ │ + var name = options.name || "Bing " + (options.type || this.type); │ │ │ │ │ + var newArgs = [name, null, options]; │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, newArgs); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + crossOriginKeyword: "anonymous" │ │ │ │ │ + }, this.options.tileOptions); │ │ │ │ │ + this.loadMetadata() │ │ │ │ │ + }, │ │ │ │ │ + loadMetadata: function() { │ │ │ │ │ + this._callbackId = "_callback_" + this.id.replace(/\./g, "_"); │ │ │ │ │ + window[this._callbackId] = OpenLayers.Function.bind(OpenLayers.Layer.Bing.processMetadata, this); │ │ │ │ │ + var params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + key: this.key, │ │ │ │ │ + jsonp: this._callbackId, │ │ │ │ │ + include: "ImageryProviders" │ │ │ │ │ + }, this.metadataParams); │ │ │ │ │ + var url = this.protocol + "//dev.virtualearth.net/REST/v1/Imagery/Metadata/" + this.type + "?" + OpenLayers.Util.getParameterString(params); │ │ │ │ │ + var script = document.createElement("script"); │ │ │ │ │ + script.type = "text/javascript"; │ │ │ │ │ + script.src = url; │ │ │ │ │ + script.id = this._callbackId; │ │ │ │ │ + document.getElementsByTagName("head")[0].appendChild(script) │ │ │ │ │ + }, │ │ │ │ │ + initLayer: function() { │ │ │ │ │ + var res = this.metadata.resourceSets[0].resources[0]; │ │ │ │ │ + var url = res.imageUrl.replace("{quadkey}", "${quadkey}"); │ │ │ │ │ + url = url.replace("{culture}", this.culture); │ │ │ │ │ + url = url.replace(this.protocolRegex, this.protocol); │ │ │ │ │ + this.url = []; │ │ │ │ │ + for (var i = 0; i < res.imageUrlSubdomains.length; ++i) { │ │ │ │ │ + this.url.push(url.replace("{subdomain}", res.imageUrlSubdomains[i])) │ │ │ │ │ + } │ │ │ │ │ + this.addOptions({ │ │ │ │ │ + maxResolution: Math.min(this.serverResolutions[res.zoomMin], this.maxResolution || Number.POSITIVE_INFINITY), │ │ │ │ │ + numZoomLevels: Math.min(res.zoomMax + 1 - res.zoomMin, this.numZoomLevels) │ │ │ │ │ + }, true); │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + this.redraw() │ │ │ │ │ + } │ │ │ │ │ + this.updateAttribution() │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + if (!this.url) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var xyz = this.getXYZ(bounds), │ │ │ │ │ + x = xyz.x, │ │ │ │ │ + y = xyz.y, │ │ │ │ │ + z = xyz.z; │ │ │ │ │ + var quadDigits = []; │ │ │ │ │ + for (var i = z; i > 0; --i) { │ │ │ │ │ + var digit = "0"; │ │ │ │ │ + var mask = 1 << i - 1; │ │ │ │ │ + if ((x & mask) != 0) { │ │ │ │ │ + digit++ │ │ │ │ │ + } │ │ │ │ │ + if ((y & mask) != 0) { │ │ │ │ │ + digit++; │ │ │ │ │ + digit++ │ │ │ │ │ + } │ │ │ │ │ + quadDigits.push(digit) │ │ │ │ │ + } │ │ │ │ │ + var quadKey = quadDigits.join(""); │ │ │ │ │ + var url = this.selectUrl("" + x + y + z, this.url); │ │ │ │ │ + return OpenLayers.String.format(url, { │ │ │ │ │ + quadkey: quadKey │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + updateAttribution: function() { │ │ │ │ │ + var metadata = this.metadata; │ │ │ │ │ + if (!metadata.resourceSets || !this.map || !this.map.center) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var res = metadata.resourceSets[0].resources[0]; │ │ │ │ │ + var extent = this.map.getExtent().transform(this.map.getProjectionObject(), new OpenLayers.Projection("EPSG:4326")); │ │ │ │ │ + var providers = res.imageryProviders || [], │ │ │ │ │ + zoom = OpenLayers.Util.indexOf(this.serverResolutions, this.getServerResolution()), │ │ │ │ │ + copyrights = "", │ │ │ │ │ + provider, i, ii, j, jj, bbox, coverage; │ │ │ │ │ + for (i = 0, ii = providers.length; i < ii; ++i) { │ │ │ │ │ + provider = providers[i]; │ │ │ │ │ + for (j = 0, jj = provider.coverageAreas.length; j < jj; ++j) { │ │ │ │ │ + coverage = provider.coverageAreas[j]; │ │ │ │ │ + bbox = OpenLayers.Bounds.fromArray(coverage.bbox, true); │ │ │ │ │ + if (extent.intersectsBounds(bbox) && zoom <= coverage.zoomMax && zoom >= coverage.zoomMin) { │ │ │ │ │ + copyrights += provider.attribution + " " │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var logo = metadata.brandLogoUri.replace(this.protocolRegex, this.protocol); │ │ │ │ │ + this.attribution = OpenLayers.String.format(this.attributionTemplate, { │ │ │ │ │ + type: this.type.toLowerCase(), │ │ │ │ │ + logo: logo, │ │ │ │ │ + copyrights: copyrights │ │ │ │ │ + }); │ │ │ │ │ + this.map && this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "attribution" │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + setMap: function() { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.register("moveend", this, this.updateAttribution) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Bing(this.options) │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.map && this.map.events.unregister("moveend", this, this.updateAttribution); │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Bing" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Bing.processMetadata = function(metadata) { │ │ │ │ │ + this.metadata = metadata; │ │ │ │ │ + this.initLayer(); │ │ │ │ │ + var script = document.getElementById(this._callbackId); │ │ │ │ │ + script.parentNode.removeChild(script); │ │ │ │ │ + window[this._callbackId] = undefined; │ │ │ │ │ + delete this._callbackId │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Layer.OSM = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ + name: "OpenStreetMap", │ │ │ │ │ + url: ["http://a.tile.openstreetmap.org/${z}/${x}/${y}.png", "http://b.tile.openstreetmap.org/${z}/${x}/${y}.png", "http://c.tile.openstreetmap.org/${z}/${x}/${y}.png"], │ │ │ │ │ + attribution: "© <a href='http://www.openstreetmap.org/copyright'>OpenStreetMap</a> contributors", │ │ │ │ │ + sphericalMercator: true, │ │ │ │ │ + wrapDateLine: true, │ │ │ │ │ + tileOptions: null, │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + crossOriginKeyword: "anonymous" │ │ │ │ │ + }, this.options && this.options.tileOptions) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.OSM(this.name, this.url, this.getOptions()) │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.OSM" │ │ │ │ │ }); │ │ │ ├── ./usr/share/javascript/openlayers/OpenLayers.tests.js │ │ │ │ ├── js-beautify {} │ │ │ │ │ @@ -33176,1034 +33176,14 @@ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Constant: CORNER_SIZE │ │ │ │ │ * {Integer} 5. Border space for the RICO corners. │ │ │ │ │ */ │ │ │ │ │ OpenLayers.Popup.AnchoredBubble.CORNER_SIZE = 5; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Kinetic.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Animation.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Kinetic = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: threshold │ │ │ │ │ - * In most cases changing the threshold isn't needed. │ │ │ │ │ - * In px/ms, default to 0. │ │ │ │ │ - */ │ │ │ │ │ - threshold: 0, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: deceleration │ │ │ │ │ - * {Float} the deseleration in px/ms², default to 0.0035. │ │ │ │ │ - */ │ │ │ │ │ - deceleration: 0.0035, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: nbPoints │ │ │ │ │ - * {Integer} the number of points we use to calculate the kinetic │ │ │ │ │ - * initial values. │ │ │ │ │ - */ │ │ │ │ │ - nbPoints: 100, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: delay │ │ │ │ │ - * {Float} time to consider to calculate the kinetic initial values. │ │ │ │ │ - * In ms, default to 200. │ │ │ │ │ - */ │ │ │ │ │ - delay: 200, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: points │ │ │ │ │ - * List of points use to calculate the kinetic initial values. │ │ │ │ │ - */ │ │ │ │ │ - points: undefined, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: timerId │ │ │ │ │ - * ID of the timer. │ │ │ │ │ - */ │ │ │ │ │ - timerId: undefined, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Kinetic │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: begin │ │ │ │ │ - * Begins the dragging. │ │ │ │ │ - */ │ │ │ │ │ - begin: function() { │ │ │ │ │ - OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ - this.timerId = undefined; │ │ │ │ │ - this.points = []; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: update │ │ │ │ │ - * Updates during the dragging. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} The new position. │ │ │ │ │ - */ │ │ │ │ │ - update: function(xy) { │ │ │ │ │ - this.points.unshift({ │ │ │ │ │ - xy: xy, │ │ │ │ │ - tick: new Date().getTime() │ │ │ │ │ - }); │ │ │ │ │ - if (this.points.length > this.nbPoints) { │ │ │ │ │ - this.points.pop(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: end │ │ │ │ │ - * Ends the dragging, start the kinetic. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} The last position. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object with two properties: "speed", and "theta". The │ │ │ │ │ - * "speed" and "theta" values are to be passed to the move │ │ │ │ │ - * function when starting the animation. │ │ │ │ │ - */ │ │ │ │ │ - end: function(xy) { │ │ │ │ │ - var last, now = new Date().getTime(); │ │ │ │ │ - for (var i = 0, l = this.points.length, point; i < l; i++) { │ │ │ │ │ - point = this.points[i]; │ │ │ │ │ - if (now - point.tick > this.delay) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - last = point; │ │ │ │ │ - } │ │ │ │ │ - if (!last) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var time = new Date().getTime() - last.tick; │ │ │ │ │ - var dist = Math.sqrt(Math.pow(xy.x - last.xy.x, 2) + │ │ │ │ │ - Math.pow(xy.y - last.xy.y, 2)); │ │ │ │ │ - var speed = dist / time; │ │ │ │ │ - if (speed == 0 || speed < this.threshold) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var theta = Math.asin((xy.y - last.xy.y) / dist); │ │ │ │ │ - if (last.xy.x <= xy.x) { │ │ │ │ │ - theta = Math.PI - theta; │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - speed: speed, │ │ │ │ │ - theta: theta │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * Launch the kinetic move pan. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * info - {Object} An object with two properties, "speed", and "theta". │ │ │ │ │ - * These values are those returned from the "end" call. │ │ │ │ │ - * callback - {Function} Function called on every step of the animation, │ │ │ │ │ - * receives x, y (values to pan), end (is the last point). │ │ │ │ │ - */ │ │ │ │ │ - move: function(info, callback) { │ │ │ │ │ - var v0 = info.speed; │ │ │ │ │ - var fx = Math.cos(info.theta); │ │ │ │ │ - var fy = -Math.sin(info.theta); │ │ │ │ │ - │ │ │ │ │ - var initialTime = new Date().getTime(); │ │ │ │ │ - │ │ │ │ │ - var lastX = 0; │ │ │ │ │ - var lastY = 0; │ │ │ │ │ - │ │ │ │ │ - var timerCallback = function() { │ │ │ │ │ - if (this.timerId == null) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var t = new Date().getTime() - initialTime; │ │ │ │ │ - │ │ │ │ │ - var p = (-this.deceleration * Math.pow(t, 2)) / 2.0 + v0 * t; │ │ │ │ │ - var x = p * fx; │ │ │ │ │ - var y = p * fy; │ │ │ │ │ - │ │ │ │ │ - var args = {}; │ │ │ │ │ - args.end = false; │ │ │ │ │ - var v = -this.deceleration * t + v0; │ │ │ │ │ - │ │ │ │ │ - if (v <= 0) { │ │ │ │ │ - OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - args.end = true; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - args.x = x - lastX; │ │ │ │ │ - args.y = y - lastY; │ │ │ │ │ - lastX = x; │ │ │ │ │ - lastY = y; │ │ │ │ │ - callback(args.x, args.y, args.end); │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - this.timerId = OpenLayers.Animation.start( │ │ │ │ │ - OpenLayers.Function.bind(timerCallback, this) │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Kinetic" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Icon.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Icon │ │ │ │ │ - * │ │ │ │ │ - * The icon represents a graphical icon on the screen. Typically used in │ │ │ │ │ - * conjunction with a <OpenLayers.Marker> to represent markers on a screen. │ │ │ │ │ - * │ │ │ │ │ - * An icon has a url, size and position. It also contains an offset which │ │ │ │ │ - * allows the center point to be represented correctly. This can be │ │ │ │ │ - * provided either as a fixed offset or a function provided to calculate │ │ │ │ │ - * the desired offset. │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Icon = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: url │ │ │ │ │ - * {String} image url │ │ │ │ │ - */ │ │ │ │ │ - url: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: size │ │ │ │ │ - * {<OpenLayers.Size>|Object} An OpenLayers.Size or │ │ │ │ │ - * an object with a 'w' and 'h' properties. │ │ │ │ │ - */ │ │ │ │ │ - size: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: offset │ │ │ │ │ - * {<OpenLayers.Pixel>|Object} distance in pixels to offset the │ │ │ │ │ - * image when being rendered. An OpenLayers.Pixel or an object │ │ │ │ │ - * with a 'x' and 'y' properties. │ │ │ │ │ - */ │ │ │ │ │ - offset: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: calculateOffset │ │ │ │ │ - * {Function} Function to calculate the offset (based on the size) │ │ │ │ │ - */ │ │ │ │ │ - calculateOffset: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: imageDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - imageDiv: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: px │ │ │ │ │ - * {<OpenLayers.Pixel>|Object} An OpenLayers.Pixel or an object │ │ │ │ │ - * with a 'x' and 'y' properties. │ │ │ │ │ - */ │ │ │ │ │ - px: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Icon │ │ │ │ │ - * Creates an icon, which is an image tag in a div. │ │ │ │ │ - * │ │ │ │ │ - * url - {String} │ │ │ │ │ - * size - {<OpenLayers.Size>|Object} An OpenLayers.Size or an │ │ │ │ │ - * object with a 'w' and 'h' │ │ │ │ │ - * properties. │ │ │ │ │ - * offset - {<OpenLayers.Pixel>|Object} An OpenLayers.Pixel or an │ │ │ │ │ - * object with a 'x' and 'y' │ │ │ │ │ - * properties. │ │ │ │ │ - * calculateOffset - {Function} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(url, size, offset, calculateOffset) { │ │ │ │ │ - this.url = url; │ │ │ │ │ - this.size = size || { │ │ │ │ │ - w: 20, │ │ │ │ │ - h: 20 │ │ │ │ │ - }; │ │ │ │ │ - this.offset = offset || { │ │ │ │ │ - x: -(this.size.w / 2), │ │ │ │ │ - y: -(this.size.h / 2) │ │ │ │ │ - }; │ │ │ │ │ - this.calculateOffset = calculateOffset; │ │ │ │ │ - │ │ │ │ │ - var id = OpenLayers.Util.createUniqueID("OL_Icon_"); │ │ │ │ │ - this.imageDiv = OpenLayers.Util.createAlphaImageDiv(id); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Nullify references and remove event listeners to prevent circular │ │ │ │ │ - * references and memory leaks │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - // erase any drawn elements │ │ │ │ │ - this.erase(); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Event.stopObservingElement(this.imageDiv.firstChild); │ │ │ │ │ - this.imageDiv.innerHTML = ""; │ │ │ │ │ - this.imageDiv = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Icon>} A fresh copy of the icon. │ │ │ │ │ - */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - return new OpenLayers.Icon(this.url, │ │ │ │ │ - this.size, │ │ │ │ │ - this.offset, │ │ │ │ │ - this.calculateOffset); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setSize │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>|Object} An OpenLayers.Size or │ │ │ │ │ - * an object with a 'w' and 'h' properties. │ │ │ │ │ - */ │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - if (size != null) { │ │ │ │ │ - this.size = size; │ │ │ │ │ - } │ │ │ │ │ - this.draw(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setUrl │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * url - {String} │ │ │ │ │ - */ │ │ │ │ │ - setUrl: function(url) { │ │ │ │ │ - if (url != null) { │ │ │ │ │ - this.url = url; │ │ │ │ │ - } │ │ │ │ │ - this.draw(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Move the div to the given pixel. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>|Object} An OpenLayers.Pixel or an │ │ │ │ │ - * object with a 'x' and 'y' properties. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A new DOM Image of this icon set at the location passed-in │ │ │ │ │ - */ │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, │ │ │ │ │ - null, │ │ │ │ │ - null, │ │ │ │ │ - this.size, │ │ │ │ │ - this.url, │ │ │ │ │ - "absolute"); │ │ │ │ │ - this.moveTo(px); │ │ │ │ │ - return this.imageDiv; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: erase │ │ │ │ │ - * Erase the underlying image element. │ │ │ │ │ - */ │ │ │ │ │ - erase: function() { │ │ │ │ │ - if (this.imageDiv != null && this.imageDiv.parentNode != null) { │ │ │ │ │ - OpenLayers.Element.remove(this.imageDiv); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setOpacity │ │ │ │ │ - * Change the icon's opacity │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * opacity - {float} │ │ │ │ │ - */ │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, null, │ │ │ │ │ - null, null, null, null, opacity); │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * move icon to passed in px. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>|Object} the pixel position to move to. │ │ │ │ │ - * An OpenLayers.Pixel or an object with a 'x' and 'y' properties. │ │ │ │ │ - */ │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - //if no px passed in, use stored location │ │ │ │ │ - if (px != null) { │ │ │ │ │ - this.px = px; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.imageDiv != null) { │ │ │ │ │ - if (this.px == null) { │ │ │ │ │ - this.display(false); │ │ │ │ │ - } else { │ │ │ │ │ - if (this.calculateOffset) { │ │ │ │ │ - this.offset = this.calculateOffset(this.size); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, { │ │ │ │ │ - x: this.px.x + this.offset.x, │ │ │ │ │ - y: this.px.y + this.offset.y │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: display │ │ │ │ │ - * Hide or show the icon │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * display - {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - display: function(display) { │ │ │ │ │ - this.imageDiv.style.display = (display) ? "" : "none"; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: isDrawn │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the icon is drawn. │ │ │ │ │ - */ │ │ │ │ │ - isDrawn: function() { │ │ │ │ │ - // nodeType 11 for ie, whose nodes *always* have a parentNode │ │ │ │ │ - // (of type document fragment) │ │ │ │ │ - var isDrawn = (this.imageDiv && this.imageDiv.parentNode && │ │ │ │ │ - (this.imageDiv.parentNode.nodeType != 11)); │ │ │ │ │ - │ │ │ │ │ - return isDrawn; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Icon" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Marker.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ - * @requires OpenLayers/Icon.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Marker │ │ │ │ │ - * Instances of OpenLayers.Marker are a combination of a │ │ │ │ │ - * <OpenLayers.LonLat> and an <OpenLayers.Icon>. │ │ │ │ │ - * │ │ │ │ │ - * Markers are generally added to a special layer called │ │ │ │ │ - * <OpenLayers.Layer.Markers>. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var markers = new OpenLayers.Layer.Markers( "Markers" ); │ │ │ │ │ - * map.addLayer(markers); │ │ │ │ │ - * │ │ │ │ │ - * var size = new OpenLayers.Size(21,25); │ │ │ │ │ - * var offset = new OpenLayers.Pixel(-(size.w/2), -size.h); │ │ │ │ │ - * var icon = new OpenLayers.Icon('http://www.openlayers.org/dev/img/marker.png', size, offset); │ │ │ │ │ - * markers.addMarker(new OpenLayers.Marker(new OpenLayers.LonLat(0,0),icon)); │ │ │ │ │ - * markers.addMarker(new OpenLayers.Marker(new OpenLayers.LonLat(0,0),icon.clone())); │ │ │ │ │ - * │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Note that if you pass an icon into the Marker constructor, it will take │ │ │ │ │ - * that icon and use it. This means that you should not share icons between │ │ │ │ │ - * markers -- you use them once, but you should clone() for any additional │ │ │ │ │ - * markers using that same icon. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Marker = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: icon │ │ │ │ │ - * {<OpenLayers.Icon>} The icon used by this marker. │ │ │ │ │ - */ │ │ │ │ │ - icon: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: lonlat │ │ │ │ │ - * {<OpenLayers.LonLat>} location of object │ │ │ │ │ - */ │ │ │ │ │ - lonlat: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: events │ │ │ │ │ - * {<OpenLayers.Events>} the event handler. │ │ │ │ │ - */ │ │ │ │ │ - events: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: map │ │ │ │ │ - * {<OpenLayers.Map>} the map this marker is attached to │ │ │ │ │ - */ │ │ │ │ │ - map: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Marker │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>} the position of this marker │ │ │ │ │ - * icon - {<OpenLayers.Icon>} the icon for this marker │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(lonlat, icon) { │ │ │ │ │ - this.lonlat = lonlat; │ │ │ │ │ - │ │ │ │ │ - var newIcon = (icon) ? icon : OpenLayers.Marker.defaultIcon(); │ │ │ │ │ - if (this.icon == null) { │ │ │ │ │ - this.icon = newIcon; │ │ │ │ │ - } else { │ │ │ │ │ - this.icon.url = newIcon.url; │ │ │ │ │ - this.icon.size = newIcon.size; │ │ │ │ │ - this.icon.offset = newIcon.offset; │ │ │ │ │ - this.icon.calculateOffset = newIcon.calculateOffset; │ │ │ │ │ - } │ │ │ │ │ - this.events = new OpenLayers.Events(this, this.icon.imageDiv); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Destroy the marker. You must first remove the marker from any │ │ │ │ │ - * layer which it has been added to, or you will get buggy behavior. │ │ │ │ │ - * (This can not be done within the marker since the marker does not │ │ │ │ │ - * know which layer it is attached to.) │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - // erase any drawn features │ │ │ │ │ - this.erase(); │ │ │ │ │ - │ │ │ │ │ - this.map = null; │ │ │ │ │ - │ │ │ │ │ - this.events.destroy(); │ │ │ │ │ - this.events = null; │ │ │ │ │ - │ │ │ │ │ - if (this.icon != null) { │ │ │ │ │ - this.icon.destroy(); │ │ │ │ │ - this.icon = null; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Calls draw on the icon, and returns that output. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A new DOM Image with this marker's icon set at the │ │ │ │ │ - * location passed-in │ │ │ │ │ - */ │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - return this.icon.draw(px); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: erase │ │ │ │ │ - * Erases any drawn elements for this marker. │ │ │ │ │ - */ │ │ │ │ │ - erase: function() { │ │ │ │ │ - if (this.icon != null) { │ │ │ │ │ - this.icon.erase(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * Move the marker to the new location. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>|Object} the pixel position to move to. │ │ │ │ │ - * An OpenLayers.Pixel or an object with a 'x' and 'y' properties. │ │ │ │ │ - */ │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - if ((px != null) && (this.icon != null)) { │ │ │ │ │ - this.icon.moveTo(px); │ │ │ │ │ - } │ │ │ │ │ - this.lonlat = this.map.getLonLatFromLayerPx(px); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: isDrawn │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the marker is drawn. │ │ │ │ │ - */ │ │ │ │ │ - isDrawn: function() { │ │ │ │ │ - var isDrawn = (this.icon && this.icon.isDrawn()); │ │ │ │ │ - return isDrawn; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onScreen │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the marker is currently visible on screen. │ │ │ │ │ - */ │ │ │ │ │ - onScreen: function() { │ │ │ │ │ - │ │ │ │ │ - var onScreen = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - var screenBounds = this.map.getExtent(); │ │ │ │ │ - onScreen = screenBounds.containsLonLat(this.lonlat); │ │ │ │ │ - } │ │ │ │ │ - return onScreen; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: inflate │ │ │ │ │ - * Englarges the markers icon by the specified ratio. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * inflate - {float} the ratio to enlarge the marker by (passing 2 │ │ │ │ │ - * will double the size). │ │ │ │ │ - */ │ │ │ │ │ - inflate: function(inflate) { │ │ │ │ │ - if (this.icon) { │ │ │ │ │ - this.icon.setSize({ │ │ │ │ │ - w: this.icon.size.w * inflate, │ │ │ │ │ - h: this.icon.size.h * inflate │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setOpacity │ │ │ │ │ - * Change the opacity of the marker by changin the opacity of │ │ │ │ │ - * its icon │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * opacity - {float} Specified as fraction (0.4, etc) │ │ │ │ │ - */ │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - this.icon.setOpacity(opacity); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setUrl │ │ │ │ │ - * Change URL of the Icon Image. │ │ │ │ │ - * │ │ │ │ │ - * url - {String} │ │ │ │ │ - */ │ │ │ │ │ - setUrl: function(url) { │ │ │ │ │ - this.icon.setUrl(url); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: display │ │ │ │ │ - * Hide or show the icon │ │ │ │ │ - * │ │ │ │ │ - * display - {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - display: function(display) { │ │ │ │ │ - this.icon.display(display); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Marker" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Function: defaultIcon │ │ │ │ │ - * Creates a default <OpenLayers.Icon>. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Icon>} A default OpenLayers.Icon to use for a marker │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Marker.defaultIcon = function() { │ │ │ │ │ - return new OpenLayers.Icon(OpenLayers.Util.getImageLocation("marker.png"), { │ │ │ │ │ - w: 21, │ │ │ │ │ - h: 25 │ │ │ │ │ - }, { │ │ │ │ │ - x: -10.5, │ │ │ │ │ - y: -25 │ │ │ │ │ - }); │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Handler │ │ │ │ │ - * Base class to construct a higher-level handler for event sequences. All │ │ │ │ │ - * handlers have activate and deactivate methods. In addition, they have │ │ │ │ │ - * methods named like browser events. When a handler is activated, any │ │ │ │ │ - * additional methods named like a browser event is registered as a │ │ │ │ │ - * listener for the corresponding event. When a handler is deactivated, │ │ │ │ │ - * those same methods are unregistered as event listeners. │ │ │ │ │ - * │ │ │ │ │ - * Handlers also typically have a callbacks object with keys named like │ │ │ │ │ - * the abstracted events or event sequences that they are in charge of │ │ │ │ │ - * handling. The controls that wrap handlers define the methods that │ │ │ │ │ - * correspond to these abstract events - so instead of listening for │ │ │ │ │ - * individual browser events, they only listen for the abstract events │ │ │ │ │ - * defined by the handler. │ │ │ │ │ - * │ │ │ │ │ - * Handlers are created by controls, which ultimately have the responsibility │ │ │ │ │ - * of making changes to the the state of the application. Handlers │ │ │ │ │ - * themselves may make temporary changes, but in general are expected to │ │ │ │ │ - * return the application in the same state that they found it. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: id │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - id: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: control │ │ │ │ │ - * {<OpenLayers.Control>}. The control that initialized this handler. The │ │ │ │ │ - * control is assumed to have a valid map property - that map is used │ │ │ │ │ - * in the handler's own setMap method. │ │ │ │ │ - */ │ │ │ │ │ - control: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: map │ │ │ │ │ - * {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - map: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: keyMask │ │ │ │ │ - * {Integer} Use bitwise operators and one or more of the OpenLayers.Handler │ │ │ │ │ - * constants to construct a keyMask. The keyMask is used by │ │ │ │ │ - * <checkModifiers>. If the keyMask matches the combination of keys │ │ │ │ │ - * down on an event, checkModifiers returns true. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * // handler only responds if the Shift key is down │ │ │ │ │ - * handler.keyMask = OpenLayers.Handler.MOD_SHIFT; │ │ │ │ │ - * │ │ │ │ │ - * // handler only responds if Ctrl-Shift is down │ │ │ │ │ - * handler.keyMask = OpenLayers.Handler.MOD_SHIFT | │ │ │ │ │ - * OpenLayers.Handler.MOD_CTRL; │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ - keyMask: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: active │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - active: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: evt │ │ │ │ │ - * {Event} This property references the last event handled by the handler. │ │ │ │ │ - * Note that this property is not part of the stable API. Use of the │ │ │ │ │ - * evt property should be restricted to controls in the library │ │ │ │ │ - * or other applications that are willing to update with changes to │ │ │ │ │ - * the OpenLayers code. │ │ │ │ │ - */ │ │ │ │ │ - evt: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: touch │ │ │ │ │ - * {Boolean} Indicates the support of touch events. When touch events are │ │ │ │ │ - * started touch will be true and all mouse related listeners will do │ │ │ │ │ - * nothing. │ │ │ │ │ - */ │ │ │ │ │ - touch: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Handler │ │ │ │ │ - * Construct a handler. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that initialized this │ │ │ │ │ - * handler. The control is assumed to have a valid map property; that │ │ │ │ │ - * map is used in the handler's own setMap method. If a map property │ │ │ │ │ - * is present in the options argument it will be used instead. │ │ │ │ │ - * callbacks - {Object} An object whose properties correspond to abstracted │ │ │ │ │ - * events or sequences of browser events. The values for these │ │ │ │ │ - * properties are functions defined by the control that get called by │ │ │ │ │ - * the handler. │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * the handler. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.control = control; │ │ │ │ │ - this.callbacks = callbacks; │ │ │ │ │ - │ │ │ │ │ - var map = this.map || control.map; │ │ │ │ │ - if (map) { │ │ │ │ │ - this.setMap(map); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.map = map; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: checkModifiers │ │ │ │ │ - * Check the keyMask on the handler. If no <keyMask> is set, this always │ │ │ │ │ - * returns true. If a <keyMask> is set and it matches the combination │ │ │ │ │ - * of keys down on an event, this returns true. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The keyMask matches the keys down on an event. │ │ │ │ │ - */ │ │ │ │ │ - checkModifiers: function(evt) { │ │ │ │ │ - if (this.keyMask == null) { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - /* calculate the keyboard modifier mask for this event */ │ │ │ │ │ - var keyModifiers = │ │ │ │ │ - (evt.shiftKey ? OpenLayers.Handler.MOD_SHIFT : 0) | │ │ │ │ │ - (evt.ctrlKey ? OpenLayers.Handler.MOD_CTRL : 0) | │ │ │ │ │ - (evt.altKey ? OpenLayers.Handler.MOD_ALT : 0) | │ │ │ │ │ - (evt.metaKey ? OpenLayers.Handler.MOD_META : 0); │ │ │ │ │ - │ │ │ │ │ - /* if it differs from the handler object's key mask, │ │ │ │ │ - bail out of the event handler */ │ │ │ │ │ - return (keyModifiers == this.keyMask); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Turn on the handler. Returns false if the handler was already active. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - // register for event handlers defined on this class. │ │ │ │ │ - var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.register(events[i], this[events[i]]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.active = true; │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Turn off the handler. Returns false if the handler was already inactive. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - // unregister event handlers defined on this class. │ │ │ │ │ - var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.unregister(events[i], this[events[i]]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.touch = false; │ │ │ │ │ - this.active = false; │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: startTouch │ │ │ │ │ - * Start touch events, this method must be called by subclasses in │ │ │ │ │ - * "touchstart" method. When touch events are started <touch> will be │ │ │ │ │ - * true and all mouse related listeners will do nothing. │ │ │ │ │ - */ │ │ │ │ │ - startTouch: function() { │ │ │ │ │ - if (!this.touch) { │ │ │ │ │ - this.touch = true; │ │ │ │ │ - var events = [ │ │ │ │ │ - "mousedown", "mouseup", "mousemove", "click", "dblclick", │ │ │ │ │ - "mouseout" │ │ │ │ │ - ]; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.unregister(events[i], this[events[i]]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: callback │ │ │ │ │ - * Trigger the control's named callback with the given arguments │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} The key for the callback that is one of the properties │ │ │ │ │ - * of the handler's callbacks object. │ │ │ │ │ - * args - {Array(*)} An array of arguments (any type) with which to call │ │ │ │ │ - * the callback (defined by the control). │ │ │ │ │ - */ │ │ │ │ │ - callback: function(name, args) { │ │ │ │ │ - if (name && this.callbacks[name]) { │ │ │ │ │ - this.callbacks[name].apply(this.control, args); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: register │ │ │ │ │ - * register an event on the map │ │ │ │ │ - */ │ │ │ │ │ - register: function(name, method) { │ │ │ │ │ - // TODO: deal with registerPriority in 3.0 │ │ │ │ │ - this.map.events.registerPriority(name, this, method); │ │ │ │ │ - this.map.events.registerPriority(name, this, this.setEvent); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: unregister │ │ │ │ │ - * unregister an event from the map │ │ │ │ │ - */ │ │ │ │ │ - unregister: function(name, method) { │ │ │ │ │ - this.map.events.unregister(name, this, method); │ │ │ │ │ - this.map.events.unregister(name, this, this.setEvent); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setEvent │ │ │ │ │ - * With each registered browser event, the handler sets its own evt │ │ │ │ │ - * property. This property can be accessed by controls if needed │ │ │ │ │ - * to get more information about the event that the handler is │ │ │ │ │ - * processing. │ │ │ │ │ - * │ │ │ │ │ - * This allows modifier keys on the event to be checked (alt, shift, ctrl, │ │ │ │ │ - * and meta cannot be checked with the keyboard handler). For a │ │ │ │ │ - * control to determine which modifier keys are associated with the │ │ │ │ │ - * event that a handler is currently processing, it should access │ │ │ │ │ - * (code)handler.evt.altKey || handler.evt.shiftKey || │ │ │ │ │ - * handler.evt.ctrlKey || handler.evt.metaKey(end). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event. │ │ │ │ │ - */ │ │ │ │ │ - setEvent: function(evt) { │ │ │ │ │ - this.evt = evt; │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Deconstruct the handler. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - // unregister event listeners │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - // eliminate circular references │ │ │ │ │ - this.control = this.map = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_NONE │ │ │ │ │ - * If set as the <keyMask>, <checkModifiers> returns false if any key is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_NONE = 0; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_SHIFT │ │ │ │ │ - * If set as the <keyMask>, <checkModifiers> returns false if Shift is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_SHIFT = 1; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_CTRL │ │ │ │ │ - * If set as the <keyMask>, <checkModifiers> returns false if Ctrl is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_CTRL = 2; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_ALT │ │ │ │ │ - * If set as the <keyMask>, <checkModifiers> returns false if Alt is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_ALT = 4; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Handler.MOD_META │ │ │ │ │ - * If set as the <keyMask>, <checkModifiers> returns false if Cmd is down. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.MOD_META = 8; │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ OpenLayers/Filter.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -34286,793 +33266,14 @@ │ │ │ │ │ } │ │ │ │ │ return string; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Filter" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/TileManager.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - * @requires OpenLayers/BaseTypes.js │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Element.js │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ - * @requires OpenLayers/Tile/Image.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.TileManager │ │ │ │ │ - * Provides queueing of image requests and caching of image elements. │ │ │ │ │ - * │ │ │ │ │ - * Queueing avoids unnecessary image requests while changing zoom levels │ │ │ │ │ - * quickly, and helps improve dragging performance on mobile devices that show │ │ │ │ │ - * a lag in dragging when loading of new images starts. <zoomDelay> and │ │ │ │ │ - * <moveDelay> are the configuration options to control this behavior. │ │ │ │ │ - * │ │ │ │ │ - * Caching avoids setting the src on image elements for images that have already │ │ │ │ │ - * been used. Several maps can share a TileManager instance, in which case each │ │ │ │ │ - * map gets its own tile queue, but all maps share the same tile cache. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.TileManager = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: cacheSize │ │ │ │ │ - * {Number} Number of image elements to keep referenced in this instance's │ │ │ │ │ - * cache for fast reuse. Default is 256. │ │ │ │ │ - */ │ │ │ │ │ - cacheSize: 256, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: tilesPerFrame │ │ │ │ │ - * {Number} Number of queued tiles to load per frame (see <frameDelay>). │ │ │ │ │ - * Default is 2. │ │ │ │ │ - */ │ │ │ │ │ - tilesPerFrame: 2, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: frameDelay │ │ │ │ │ - * {Number} Delay between tile loading frames (see <tilesPerFrame>) in │ │ │ │ │ - * milliseconds. Default is 16. │ │ │ │ │ - */ │ │ │ │ │ - frameDelay: 16, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: moveDelay │ │ │ │ │ - * {Number} Delay in milliseconds after a map's move event before loading │ │ │ │ │ - * tiles. Default is 100. │ │ │ │ │ - */ │ │ │ │ │ - moveDelay: 100, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomDelay │ │ │ │ │ - * {Number} Delay in milliseconds after a map's zoomend event before loading │ │ │ │ │ - * tiles. Default is 200. │ │ │ │ │ - */ │ │ │ │ │ - zoomDelay: 200, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: maps │ │ │ │ │ - * {Array(<OpenLayers.Map>)} The maps to manage tiles on. │ │ │ │ │ - */ │ │ │ │ │ - maps: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: tileQueueId │ │ │ │ │ - * {Object} The ids of the <drawTilesFromQueue> loop, keyed by map id. │ │ │ │ │ - */ │ │ │ │ │ - tileQueueId: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: tileQueue │ │ │ │ │ - * {Object(Array(<OpenLayers.Tile>))} Tiles queued for drawing, keyed by │ │ │ │ │ - * map id. │ │ │ │ │ - */ │ │ │ │ │ - tileQueue: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: tileCache │ │ │ │ │ - * {Object} Cached image elements, keyed by URL. │ │ │ │ │ - */ │ │ │ │ │ - tileCache: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: tileCacheIndex │ │ │ │ │ - * {Array(String)} URLs of cached tiles. First entry is the least recently │ │ │ │ │ - * used. │ │ │ │ │ - */ │ │ │ │ │ - tileCacheIndex: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.TileManager │ │ │ │ │ - * Constructor for a new <OpenLayers.TileManager> instance. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Configuration for this instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.maps = []; │ │ │ │ │ - this.tileQueueId = {}; │ │ │ │ │ - this.tileQueue = {}; │ │ │ │ │ - this.tileCache = {}; │ │ │ │ │ - this.tileCacheIndex = []; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addMap │ │ │ │ │ - * Binds this instance to a map │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - addMap: function(map) { │ │ │ │ │ - if (this._destroyed || !OpenLayers.Layer.Grid) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - this.maps.push(map); │ │ │ │ │ - this.tileQueue[map.id] = []; │ │ │ │ │ - for (var i = 0, ii = map.layers.length; i < ii; ++i) { │ │ │ │ │ - this.addLayer({ │ │ │ │ │ - layer: map.layers[i] │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - map.events.on({ │ │ │ │ │ - move: this.move, │ │ │ │ │ - zoomend: this.zoomEnd, │ │ │ │ │ - changelayer: this.changeLayer, │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - preremovelayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeMap │ │ │ │ │ - * Unbinds this instance from a map │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - if (this._destroyed || !OpenLayers.Layer.Grid) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - window.clearTimeout(this.tileQueueId[map.id]); │ │ │ │ │ - if (map.layers) { │ │ │ │ │ - for (var i = 0, ii = map.layers.length; i < ii; ++i) { │ │ │ │ │ - this.removeLayer({ │ │ │ │ │ - layer: map.layers[i] │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (map.events) { │ │ │ │ │ - map.events.un({ │ │ │ │ │ - move: this.move, │ │ │ │ │ - zoomend: this.zoomEnd, │ │ │ │ │ - changelayer: this.changeLayer, │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - preremovelayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - delete this.tileQueue[map.id]; │ │ │ │ │ - delete this.tileQueueId[map.id]; │ │ │ │ │ - OpenLayers.Util.removeItem(this.maps, map); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * Handles the map's move event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} Listener argument │ │ │ │ │ - */ │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - this.updateTimeout(evt.object, this.moveDelay, true); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: zoomEnd │ │ │ │ │ - * Handles the map's zoomEnd event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} Listener argument │ │ │ │ │ - */ │ │ │ │ │ - zoomEnd: function(evt) { │ │ │ │ │ - this.updateTimeout(evt.object, this.zoomDelay); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: changeLayer │ │ │ │ │ - * Handles the map's changeLayer event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} Listener argument │ │ │ │ │ - */ │ │ │ │ │ - changeLayer: function(evt) { │ │ │ │ │ - if (evt.property === 'visibility' || evt.property === 'params') { │ │ │ │ │ - this.updateTimeout(evt.object, 0); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addLayer │ │ │ │ │ - * Handles the map's addlayer event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} The listener argument │ │ │ │ │ - */ │ │ │ │ │ - addLayer: function(evt) { │ │ │ │ │ - var layer = evt.layer; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ - layer.events.on({ │ │ │ │ │ - addtile: this.addTile, │ │ │ │ │ - retile: this.clearTileQueue, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - var i, j, tile; │ │ │ │ │ - for (i = layer.grid.length - 1; i >= 0; --i) { │ │ │ │ │ - for (j = layer.grid[i].length - 1; j >= 0; --j) { │ │ │ │ │ - tile = layer.grid[i][j]; │ │ │ │ │ - this.addTile({ │ │ │ │ │ - tile: tile │ │ │ │ │ - }); │ │ │ │ │ - if (tile.url && !tile.imgDiv) { │ │ │ │ │ - this.manageTileCache({ │ │ │ │ │ - object: tile │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeLayer │ │ │ │ │ - * Handles the map's preremovelayer event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} The listener argument │ │ │ │ │ - */ │ │ │ │ │ - removeLayer: function(evt) { │ │ │ │ │ - var layer = evt.layer; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ - this.clearTileQueue({ │ │ │ │ │ - object: layer │ │ │ │ │ - }); │ │ │ │ │ - if (layer.events) { │ │ │ │ │ - layer.events.un({ │ │ │ │ │ - addtile: this.addTile, │ │ │ │ │ - retile: this.clearTileQueue, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - if (layer.grid) { │ │ │ │ │ - var i, j, tile; │ │ │ │ │ - for (i = layer.grid.length - 1; i >= 0; --i) { │ │ │ │ │ - for (j = layer.grid[i].length - 1; j >= 0; --j) { │ │ │ │ │ - tile = layer.grid[i][j]; │ │ │ │ │ - this.unloadTile({ │ │ │ │ │ - object: tile │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateTimeout │ │ │ │ │ - * Applies the <moveDelay> or <zoomDelay> to the <drawTilesFromQueue> loop, │ │ │ │ │ - * and schedules more queue processing after <frameDelay> if there are still │ │ │ │ │ - * tiles in the queue. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} The map to update the timeout for │ │ │ │ │ - * delay - {Number} The delay to apply │ │ │ │ │ - * nice - {Boolean} If true, the timeout function will only be created if │ │ │ │ │ - * the tilequeue is not empty. This is used by the move handler to │ │ │ │ │ - * avoid impacts on dragging performance. For other events, the tile │ │ │ │ │ - * queue may not be populated yet, so we need to set the timer │ │ │ │ │ - * regardless of the queue size. │ │ │ │ │ - */ │ │ │ │ │ - updateTimeout: function(map, delay, nice) { │ │ │ │ │ - window.clearTimeout(this.tileQueueId[map.id]); │ │ │ │ │ - var tileQueue = this.tileQueue[map.id]; │ │ │ │ │ - if (!nice || tileQueue.length) { │ │ │ │ │ - this.tileQueueId[map.id] = window.setTimeout( │ │ │ │ │ - OpenLayers.Function.bind(function() { │ │ │ │ │ - this.drawTilesFromQueue(map); │ │ │ │ │ - if (tileQueue.length) { │ │ │ │ │ - this.updateTimeout(map, this.frameDelay); │ │ │ │ │ - } │ │ │ │ │ - }, this), delay │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addTile │ │ │ │ │ - * Listener for the layer's addtile event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} The listener argument │ │ │ │ │ - */ │ │ │ │ │ - addTile: function(evt) { │ │ │ │ │ - if (evt.tile instanceof OpenLayers.Tile.Image) { │ │ │ │ │ - evt.tile.events.on({ │ │ │ │ │ - beforedraw: this.queueTileDraw, │ │ │ │ │ - beforeload: this.manageTileCache, │ │ │ │ │ - loadend: this.addToCache, │ │ │ │ │ - unload: this.unloadTile, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - // Layer has the wrong tile type, so don't handle it any longer │ │ │ │ │ - this.removeLayer({ │ │ │ │ │ - layer: evt.tile.layer │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: unloadTile │ │ │ │ │ - * Listener for the tile's unload event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} The listener argument │ │ │ │ │ - */ │ │ │ │ │ - unloadTile: function(evt) { │ │ │ │ │ - var tile = evt.object; │ │ │ │ │ - tile.events.un({ │ │ │ │ │ - beforedraw: this.queueTileDraw, │ │ │ │ │ - beforeload: this.manageTileCache, │ │ │ │ │ - loadend: this.addToCache, │ │ │ │ │ - unload: this.unloadTile, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Util.removeItem(this.tileQueue[tile.layer.map.id], tile); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: queueTileDraw │ │ │ │ │ - * Adds a tile to the queue that will draw it. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} Listener argument of the tile's beforedraw event │ │ │ │ │ - */ │ │ │ │ │ - queueTileDraw: function(evt) { │ │ │ │ │ - var tile = evt.object; │ │ │ │ │ - var queued = false; │ │ │ │ │ - var layer = tile.layer; │ │ │ │ │ - var url = layer.getURL(tile.bounds); │ │ │ │ │ - var img = this.tileCache[url]; │ │ │ │ │ - if (img && img.className !== 'olTileImage') { │ │ │ │ │ - // cached image no longer valid, e.g. because we're olTileReplacing │ │ │ │ │ - delete this.tileCache[url]; │ │ │ │ │ - OpenLayers.Util.removeItem(this.tileCacheIndex, url); │ │ │ │ │ - img = null; │ │ │ │ │ - } │ │ │ │ │ - // queue only if image with same url not cached already │ │ │ │ │ - if (layer.url && (layer.async || !img)) { │ │ │ │ │ - // add to queue only if not in queue already │ │ │ │ │ - var tileQueue = this.tileQueue[layer.map.id]; │ │ │ │ │ - if (!~OpenLayers.Util.indexOf(tileQueue, tile)) { │ │ │ │ │ - tileQueue.push(tile); │ │ │ │ │ - } │ │ │ │ │ - queued = true; │ │ │ │ │ - } │ │ │ │ │ - return !queued; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawTilesFromQueue │ │ │ │ │ - * Draws tiles from the tileQueue, and unqueues the tiles │ │ │ │ │ - */ │ │ │ │ │ - drawTilesFromQueue: function(map) { │ │ │ │ │ - var tileQueue = this.tileQueue[map.id]; │ │ │ │ │ - var limit = this.tilesPerFrame; │ │ │ │ │ - var animating = map.zoomTween && map.zoomTween.playing; │ │ │ │ │ - while (!animating && tileQueue.length && limit) { │ │ │ │ │ - tileQueue.shift().draw(true); │ │ │ │ │ - --limit; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: manageTileCache │ │ │ │ │ - * Adds, updates, removes and fetches cache entries. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} Listener argument of the tile's beforeload event │ │ │ │ │ - */ │ │ │ │ │ - manageTileCache: function(evt) { │ │ │ │ │ - var tile = evt.object; │ │ │ │ │ - var img = this.tileCache[tile.url]; │ │ │ │ │ - if (img) { │ │ │ │ │ - // if image is on its layer's backbuffer, remove it from backbuffer │ │ │ │ │ - if (img.parentNode && │ │ │ │ │ - OpenLayers.Element.hasClass(img.parentNode, 'olBackBuffer')) { │ │ │ │ │ - img.parentNode.removeChild(img); │ │ │ │ │ - img.id = null; │ │ │ │ │ - } │ │ │ │ │ - // only use image from cache if it is not on a layer already │ │ │ │ │ - if (!img.parentNode) { │ │ │ │ │ - img.style.visibility = 'hidden'; │ │ │ │ │ - img.style.opacity = 0; │ │ │ │ │ - tile.setImage(img); │ │ │ │ │ - // LRU - move tile to the end of the array to mark it as the most │ │ │ │ │ - // recently used │ │ │ │ │ - OpenLayers.Util.removeItem(this.tileCacheIndex, tile.url); │ │ │ │ │ - this.tileCacheIndex.push(tile.url); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addToCache │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} Listener argument for the tile's loadend event │ │ │ │ │ - */ │ │ │ │ │ - addToCache: function(evt) { │ │ │ │ │ - var tile = evt.object; │ │ │ │ │ - if (!this.tileCache[tile.url]) { │ │ │ │ │ - if (!OpenLayers.Element.hasClass(tile.imgDiv, 'olImageLoadError')) { │ │ │ │ │ - if (this.tileCacheIndex.length >= this.cacheSize) { │ │ │ │ │ - delete this.tileCache[this.tileCacheIndex[0]]; │ │ │ │ │ - this.tileCacheIndex.shift(); │ │ │ │ │ - } │ │ │ │ │ - this.tileCache[tile.url] = tile.imgDiv; │ │ │ │ │ - this.tileCacheIndex.push(tile.url); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clearTileQueue │ │ │ │ │ - * Clears the tile queue from tiles of a specific layer │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} Listener argument of the layer's retile event │ │ │ │ │ - */ │ │ │ │ │ - clearTileQueue: function(evt) { │ │ │ │ │ - var layer = evt.object; │ │ │ │ │ - var tileQueue = this.tileQueue[layer.map.id]; │ │ │ │ │ - for (var i = tileQueue.length - 1; i >= 0; --i) { │ │ │ │ │ - if (tileQueue[i].layer === layer) { │ │ │ │ │ - tileQueue.splice(i, 1); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i = this.maps.length - 1; i >= 0; --i) { │ │ │ │ │ - this.removeMap(this.maps[i]); │ │ │ │ │ - } │ │ │ │ │ - this.maps = null; │ │ │ │ │ - this.tileQueue = null; │ │ │ │ │ - this.tileQueueId = null; │ │ │ │ │ - this.tileCache = null; │ │ │ │ │ - this.tileCacheIndex = null; │ │ │ │ │ - this._destroyed = true; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Symbolizer.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Symbolizer │ │ │ │ │ - * Base class representing a symbolizer used for feature rendering. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Symbolizer = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zIndex │ │ │ │ │ - * {Number} The zIndex determines the rendering order for a symbolizer. │ │ │ │ │ - * Symbolizers with larger zIndex values are rendered over symbolizers │ │ │ │ │ - * with smaller zIndex values. Default is 0. │ │ │ │ │ - */ │ │ │ │ │ - zIndex: 0, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Symbolizer │ │ │ │ │ - * Instances of this class are not useful. See one of the subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} An object containing properties to be set on the │ │ │ │ │ - * symbolizer. Any documented symbolizer property can be set at │ │ │ │ │ - * construction. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * A new symbolizer. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ - OpenLayers.Util.extend(this, config); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Create a copy of this symbolizer. │ │ │ │ │ - * │ │ │ │ │ - * Returns a symbolizer of the same type with the same properties. │ │ │ │ │ - */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - var Type = eval(this.CLASS_NAME); │ │ │ │ │ - return new Type(OpenLayers.Util.extend({}, this)); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Symbolizer" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Rule.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - * @requires OpenLayers/Style.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Rule │ │ │ │ │ - * This class represents an SLD Rule, as being used for rule-based SLD styling. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Rule = OpenLayers.Class({ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: id │ │ │ │ │ - * {String} A unique id for this session. │ │ │ │ │ - */ │ │ │ │ │ - id: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: name │ │ │ │ │ - * {String} name of this rule │ │ │ │ │ - */ │ │ │ │ │ - name: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: title │ │ │ │ │ - * {String} Title of this rule (set if included in SLD) │ │ │ │ │ - */ │ │ │ │ │ - title: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: description │ │ │ │ │ - * {String} Description of this rule (set if abstract is included in SLD) │ │ │ │ │ - */ │ │ │ │ │ - description: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: context │ │ │ │ │ - * {Object} An optional object with properties that the rule should be │ │ │ │ │ - * evaluated against. If no context is specified, feature.attributes will │ │ │ │ │ - * be used. │ │ │ │ │ - */ │ │ │ │ │ - context: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: filter │ │ │ │ │ - * {<OpenLayers.Filter>} Optional filter for the rule. │ │ │ │ │ - */ │ │ │ │ │ - filter: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: elseFilter │ │ │ │ │ - * {Boolean} Determines whether this rule is only to be applied only if │ │ │ │ │ - * no other rules match (ElseFilter according to the SLD specification). │ │ │ │ │ - * Default is false. For instances of OpenLayers.Rule, if elseFilter is │ │ │ │ │ - * false, the rule will always apply. For subclasses, the else property is │ │ │ │ │ - * ignored. │ │ │ │ │ - */ │ │ │ │ │ - elseFilter: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: symbolizer │ │ │ │ │ - * {Object} Symbolizer or hash of symbolizers for this rule. If hash of │ │ │ │ │ - * symbolizers, keys are one or more of ["Point", "Line", "Polygon"]. The │ │ │ │ │ - * latter if useful if it is required to style e.g. vertices of a line │ │ │ │ │ - * with a point symbolizer. Note, however, that this is not implemented │ │ │ │ │ - * yet in OpenLayers, but it is the way how symbolizers are defined in │ │ │ │ │ - * SLD. │ │ │ │ │ - */ │ │ │ │ │ - symbolizer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: symbolizers │ │ │ │ │ - * {Array} Collection of symbolizers associated with this rule. If │ │ │ │ │ - * provided at construction, the symbolizers array has precedence │ │ │ │ │ - * over the deprecated symbolizer property. Note that multiple │ │ │ │ │ - * symbolizers are not currently supported by the vector renderers. │ │ │ │ │ - * Rules with multiple symbolizers are currently only useful for │ │ │ │ │ - * maintaining elements in an SLD document. │ │ │ │ │ - */ │ │ │ │ │ - symbolizers: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: minScaleDenominator │ │ │ │ │ - * {Number} or {String} minimum scale at which to draw the feature. │ │ │ │ │ - * In the case of a String, this can be a combination of text and │ │ │ │ │ - * propertyNames in the form "literal ${propertyName}" │ │ │ │ │ - */ │ │ │ │ │ - minScaleDenominator: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: maxScaleDenominator │ │ │ │ │ - * {Number} or {String} maximum scale at which to draw the feature. │ │ │ │ │ - * In the case of a String, this can be a combination of text and │ │ │ │ │ - * propertyNames in the form "literal ${propertyName}" │ │ │ │ │ - */ │ │ │ │ │ - maxScaleDenominator: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Rule │ │ │ │ │ - * Creates a Rule. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object with properties to set on the │ │ │ │ │ - * rule │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Rule>} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.symbolizer = {}; │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - if (this.symbolizers) { │ │ │ │ │ - delete this.symbolizer; │ │ │ │ │ - } │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * nullify references to prevent circular references and memory leaks │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i in this.symbolizer) { │ │ │ │ │ - this.symbolizer[i] = null; │ │ │ │ │ - } │ │ │ │ │ - this.symbolizer = null; │ │ │ │ │ - delete this.symbolizers; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: evaluate │ │ │ │ │ - * evaluates this rule for a specific feature │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature>} feature to apply the rule to. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true if the rule applies, false if it does not. │ │ │ │ │ - * This rule is the default rule and always returns true. │ │ │ │ │ - */ │ │ │ │ │ - evaluate: function(feature) { │ │ │ │ │ - var context = this.getContext(feature); │ │ │ │ │ - var applies = true; │ │ │ │ │ - │ │ │ │ │ - if (this.minScaleDenominator || this.maxScaleDenominator) { │ │ │ │ │ - var scale = feature.layer.map.getScale(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // check if within minScale/maxScale bounds │ │ │ │ │ - if (this.minScaleDenominator) { │ │ │ │ │ - applies = scale >= OpenLayers.Style.createLiteral( │ │ │ │ │ - this.minScaleDenominator, context); │ │ │ │ │ - } │ │ │ │ │ - if (applies && this.maxScaleDenominator) { │ │ │ │ │ - applies = scale < OpenLayers.Style.createLiteral( │ │ │ │ │ - this.maxScaleDenominator, context); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // check if optional filter applies │ │ │ │ │ - if (applies && this.filter) { │ │ │ │ │ - // feature id filters get the feature, others get the context │ │ │ │ │ - if (this.filter.CLASS_NAME == "OpenLayers.Filter.FeatureId") { │ │ │ │ │ - applies = this.filter.evaluate(feature); │ │ │ │ │ - } else { │ │ │ │ │ - applies = this.filter.evaluate(context); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return applies; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getContext │ │ │ │ │ - * Gets the context for evaluating this rule │ │ │ │ │ - * │ │ │ │ │ - * Paramters: │ │ │ │ │ - * feature - {<OpenLayers.Feature>} feature to take the context from if │ │ │ │ │ - * none is specified. │ │ │ │ │ - */ │ │ │ │ │ - getContext: function(feature) { │ │ │ │ │ - var context = this.context; │ │ │ │ │ - if (!context) { │ │ │ │ │ - context = feature.attributes || feature.data; │ │ │ │ │ - } │ │ │ │ │ - if (typeof this.context == "function") { │ │ │ │ │ - context = this.context(feature); │ │ │ │ │ - } │ │ │ │ │ - return context; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Clones this rule. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Rule>} Clone of this rule. │ │ │ │ │ - */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - var options = OpenLayers.Util.extend({}, this); │ │ │ │ │ - if (this.symbolizers) { │ │ │ │ │ - // clone symbolizers │ │ │ │ │ - var len = this.symbolizers.length; │ │ │ │ │ - options.symbolizers = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - options.symbolizers[i] = this.symbolizers[i].clone(); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - // clone symbolizer │ │ │ │ │ - options.symbolizer = {}; │ │ │ │ │ - var value, type; │ │ │ │ │ - for (var key in this.symbolizer) { │ │ │ │ │ - value = this.symbolizer[key]; │ │ │ │ │ - type = typeof value; │ │ │ │ │ - if (type === "object") { │ │ │ │ │ - options.symbolizer[key] = OpenLayers.Util.extend({}, value); │ │ │ │ │ - } else if (type === "string") { │ │ │ │ │ - options.symbolizer[key] = value; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // clone filter │ │ │ │ │ - options.filter = this.filter && this.filter.clone(); │ │ │ │ │ - // clone context │ │ │ │ │ - options.context = this.context && OpenLayers.Util.extend({}, this.context); │ │ │ │ │ - return new OpenLayers.Rule(options); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Rule" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ OpenLayers/Format/GeoJSON.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -42205,14 +40406,517 @@ │ │ │ │ │ OpenLayers.Util.extend(this, options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.WPSProcess.ChainLink" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ + OpenLayers/Icon.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Icon │ │ │ │ │ + * │ │ │ │ │ + * The icon represents a graphical icon on the screen. Typically used in │ │ │ │ │ + * conjunction with a <OpenLayers.Marker> to represent markers on a screen. │ │ │ │ │ + * │ │ │ │ │ + * An icon has a url, size and position. It also contains an offset which │ │ │ │ │ + * allows the center point to be represented correctly. This can be │ │ │ │ │ + * provided either as a fixed offset or a function provided to calculate │ │ │ │ │ + * the desired offset. │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Icon = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: url │ │ │ │ │ + * {String} image url │ │ │ │ │ + */ │ │ │ │ │ + url: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: size │ │ │ │ │ + * {<OpenLayers.Size>|Object} An OpenLayers.Size or │ │ │ │ │ + * an object with a 'w' and 'h' properties. │ │ │ │ │ + */ │ │ │ │ │ + size: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: offset │ │ │ │ │ + * {<OpenLayers.Pixel>|Object} distance in pixels to offset the │ │ │ │ │ + * image when being rendered. An OpenLayers.Pixel or an object │ │ │ │ │ + * with a 'x' and 'y' properties. │ │ │ │ │ + */ │ │ │ │ │ + offset: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: calculateOffset │ │ │ │ │ + * {Function} Function to calculate the offset (based on the size) │ │ │ │ │ + */ │ │ │ │ │ + calculateOffset: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: imageDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + imageDiv: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: px │ │ │ │ │ + * {<OpenLayers.Pixel>|Object} An OpenLayers.Pixel or an object │ │ │ │ │ + * with a 'x' and 'y' properties. │ │ │ │ │ + */ │ │ │ │ │ + px: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Icon │ │ │ │ │ + * Creates an icon, which is an image tag in a div. │ │ │ │ │ + * │ │ │ │ │ + * url - {String} │ │ │ │ │ + * size - {<OpenLayers.Size>|Object} An OpenLayers.Size or an │ │ │ │ │ + * object with a 'w' and 'h' │ │ │ │ │ + * properties. │ │ │ │ │ + * offset - {<OpenLayers.Pixel>|Object} An OpenLayers.Pixel or an │ │ │ │ │ + * object with a 'x' and 'y' │ │ │ │ │ + * properties. │ │ │ │ │ + * calculateOffset - {Function} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(url, size, offset, calculateOffset) { │ │ │ │ │ + this.url = url; │ │ │ │ │ + this.size = size || { │ │ │ │ │ + w: 20, │ │ │ │ │ + h: 20 │ │ │ │ │ + }; │ │ │ │ │ + this.offset = offset || { │ │ │ │ │ + x: -(this.size.w / 2), │ │ │ │ │ + y: -(this.size.h / 2) │ │ │ │ │ + }; │ │ │ │ │ + this.calculateOffset = calculateOffset; │ │ │ │ │ + │ │ │ │ │ + var id = OpenLayers.Util.createUniqueID("OL_Icon_"); │ │ │ │ │ + this.imageDiv = OpenLayers.Util.createAlphaImageDiv(id); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Nullify references and remove event listeners to prevent circular │ │ │ │ │ + * references and memory leaks │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + // erase any drawn elements │ │ │ │ │ + this.erase(); │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Event.stopObservingElement(this.imageDiv.firstChild); │ │ │ │ │ + this.imageDiv.innerHTML = ""; │ │ │ │ │ + this.imageDiv = null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clone │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Icon>} A fresh copy of the icon. │ │ │ │ │ + */ │ │ │ │ │ + clone: function() { │ │ │ │ │ + return new OpenLayers.Icon(this.url, │ │ │ │ │ + this.size, │ │ │ │ │ + this.offset, │ │ │ │ │ + this.calculateOffset); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setSize │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * size - {<OpenLayers.Size>|Object} An OpenLayers.Size or │ │ │ │ │ + * an object with a 'w' and 'h' properties. │ │ │ │ │ + */ │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + if (size != null) { │ │ │ │ │ + this.size = size; │ │ │ │ │ + } │ │ │ │ │ + this.draw(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setUrl │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * url - {String} │ │ │ │ │ + */ │ │ │ │ │ + setUrl: function(url) { │ │ │ │ │ + if (url != null) { │ │ │ │ │ + this.url = url; │ │ │ │ │ + } │ │ │ │ │ + this.draw(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Move the div to the given pixel. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>|Object} An OpenLayers.Pixel or an │ │ │ │ │ + * object with a 'x' and 'y' properties. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A new DOM Image of this icon set at the location passed-in │ │ │ │ │ + */ │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, │ │ │ │ │ + null, │ │ │ │ │ + null, │ │ │ │ │ + this.size, │ │ │ │ │ + this.url, │ │ │ │ │ + "absolute"); │ │ │ │ │ + this.moveTo(px); │ │ │ │ │ + return this.imageDiv; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: erase │ │ │ │ │ + * Erase the underlying image element. │ │ │ │ │ + */ │ │ │ │ │ + erase: function() { │ │ │ │ │ + if (this.imageDiv != null && this.imageDiv.parentNode != null) { │ │ │ │ │ + OpenLayers.Element.remove(this.imageDiv); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setOpacity │ │ │ │ │ + * Change the icon's opacity │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * opacity - {float} │ │ │ │ │ + */ │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, null, │ │ │ │ │ + null, null, null, null, opacity); │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * move icon to passed in px. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>|Object} the pixel position to move to. │ │ │ │ │ + * An OpenLayers.Pixel or an object with a 'x' and 'y' properties. │ │ │ │ │ + */ │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + //if no px passed in, use stored location │ │ │ │ │ + if (px != null) { │ │ │ │ │ + this.px = px; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.imageDiv != null) { │ │ │ │ │ + if (this.px == null) { │ │ │ │ │ + this.display(false); │ │ │ │ │ + } else { │ │ │ │ │ + if (this.calculateOffset) { │ │ │ │ │ + this.offset = this.calculateOffset(this.size); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, { │ │ │ │ │ + x: this.px.x + this.offset.x, │ │ │ │ │ + y: this.px.y + this.offset.y │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: display │ │ │ │ │ + * Hide or show the icon │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * display - {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + display: function(display) { │ │ │ │ │ + this.imageDiv.style.display = (display) ? "" : "none"; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: isDrawn │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the icon is drawn. │ │ │ │ │ + */ │ │ │ │ │ + isDrawn: function() { │ │ │ │ │ + // nodeType 11 for ie, whose nodes *always* have a parentNode │ │ │ │ │ + // (of type document fragment) │ │ │ │ │ + var isDrawn = (this.imageDiv && this.imageDiv.parentNode && │ │ │ │ │ + (this.imageDiv.parentNode.nodeType != 11)); │ │ │ │ │ + │ │ │ │ │ + return isDrawn; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Icon" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Marker.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ + * @requires OpenLayers/Icon.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Marker │ │ │ │ │ + * Instances of OpenLayers.Marker are a combination of a │ │ │ │ │ + * <OpenLayers.LonLat> and an <OpenLayers.Icon>. │ │ │ │ │ + * │ │ │ │ │ + * Markers are generally added to a special layer called │ │ │ │ │ + * <OpenLayers.Layer.Markers>. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var markers = new OpenLayers.Layer.Markers( "Markers" ); │ │ │ │ │ + * map.addLayer(markers); │ │ │ │ │ + * │ │ │ │ │ + * var size = new OpenLayers.Size(21,25); │ │ │ │ │ + * var offset = new OpenLayers.Pixel(-(size.w/2), -size.h); │ │ │ │ │ + * var icon = new OpenLayers.Icon('http://www.openlayers.org/dev/img/marker.png', size, offset); │ │ │ │ │ + * markers.addMarker(new OpenLayers.Marker(new OpenLayers.LonLat(0,0),icon)); │ │ │ │ │ + * markers.addMarker(new OpenLayers.Marker(new OpenLayers.LonLat(0,0),icon.clone())); │ │ │ │ │ + * │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Note that if you pass an icon into the Marker constructor, it will take │ │ │ │ │ + * that icon and use it. This means that you should not share icons between │ │ │ │ │ + * markers -- you use them once, but you should clone() for any additional │ │ │ │ │ + * markers using that same icon. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Marker = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: icon │ │ │ │ │ + * {<OpenLayers.Icon>} The icon used by this marker. │ │ │ │ │ + */ │ │ │ │ │ + icon: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: lonlat │ │ │ │ │ + * {<OpenLayers.LonLat>} location of object │ │ │ │ │ + */ │ │ │ │ │ + lonlat: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: events │ │ │ │ │ + * {<OpenLayers.Events>} the event handler. │ │ │ │ │ + */ │ │ │ │ │ + events: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: map │ │ │ │ │ + * {<OpenLayers.Map>} the map this marker is attached to │ │ │ │ │ + */ │ │ │ │ │ + map: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Marker │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>} the position of this marker │ │ │ │ │ + * icon - {<OpenLayers.Icon>} the icon for this marker │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(lonlat, icon) { │ │ │ │ │ + this.lonlat = lonlat; │ │ │ │ │ + │ │ │ │ │ + var newIcon = (icon) ? icon : OpenLayers.Marker.defaultIcon(); │ │ │ │ │ + if (this.icon == null) { │ │ │ │ │ + this.icon = newIcon; │ │ │ │ │ + } else { │ │ │ │ │ + this.icon.url = newIcon.url; │ │ │ │ │ + this.icon.size = newIcon.size; │ │ │ │ │ + this.icon.offset = newIcon.offset; │ │ │ │ │ + this.icon.calculateOffset = newIcon.calculateOffset; │ │ │ │ │ + } │ │ │ │ │ + this.events = new OpenLayers.Events(this, this.icon.imageDiv); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Destroy the marker. You must first remove the marker from any │ │ │ │ │ + * layer which it has been added to, or you will get buggy behavior. │ │ │ │ │ + * (This can not be done within the marker since the marker does not │ │ │ │ │ + * know which layer it is attached to.) │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + // erase any drawn features │ │ │ │ │ + this.erase(); │ │ │ │ │ + │ │ │ │ │ + this.map = null; │ │ │ │ │ + │ │ │ │ │ + this.events.destroy(); │ │ │ │ │ + this.events = null; │ │ │ │ │ + │ │ │ │ │ + if (this.icon != null) { │ │ │ │ │ + this.icon.destroy(); │ │ │ │ │ + this.icon = null; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Calls draw on the icon, and returns that output. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A new DOM Image with this marker's icon set at the │ │ │ │ │ + * location passed-in │ │ │ │ │ + */ │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + return this.icon.draw(px); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: erase │ │ │ │ │ + * Erases any drawn elements for this marker. │ │ │ │ │ + */ │ │ │ │ │ + erase: function() { │ │ │ │ │ + if (this.icon != null) { │ │ │ │ │ + this.icon.erase(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * Move the marker to the new location. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>|Object} the pixel position to move to. │ │ │ │ │ + * An OpenLayers.Pixel or an object with a 'x' and 'y' properties. │ │ │ │ │ + */ │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + if ((px != null) && (this.icon != null)) { │ │ │ │ │ + this.icon.moveTo(px); │ │ │ │ │ + } │ │ │ │ │ + this.lonlat = this.map.getLonLatFromLayerPx(px); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: isDrawn │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the marker is drawn. │ │ │ │ │ + */ │ │ │ │ │ + isDrawn: function() { │ │ │ │ │ + var isDrawn = (this.icon && this.icon.isDrawn()); │ │ │ │ │ + return isDrawn; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: onScreen │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the marker is currently visible on screen. │ │ │ │ │ + */ │ │ │ │ │ + onScreen: function() { │ │ │ │ │ + │ │ │ │ │ + var onScreen = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + var screenBounds = this.map.getExtent(); │ │ │ │ │ + onScreen = screenBounds.containsLonLat(this.lonlat); │ │ │ │ │ + } │ │ │ │ │ + return onScreen; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: inflate │ │ │ │ │ + * Englarges the markers icon by the specified ratio. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * inflate - {float} the ratio to enlarge the marker by (passing 2 │ │ │ │ │ + * will double the size). │ │ │ │ │ + */ │ │ │ │ │ + inflate: function(inflate) { │ │ │ │ │ + if (this.icon) { │ │ │ │ │ + this.icon.setSize({ │ │ │ │ │ + w: this.icon.size.w * inflate, │ │ │ │ │ + h: this.icon.size.h * inflate │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setOpacity │ │ │ │ │ + * Change the opacity of the marker by changin the opacity of │ │ │ │ │ + * its icon │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * opacity - {float} Specified as fraction (0.4, etc) │ │ │ │ │ + */ │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + this.icon.setOpacity(opacity); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setUrl │ │ │ │ │ + * Change URL of the Icon Image. │ │ │ │ │ + * │ │ │ │ │ + * url - {String} │ │ │ │ │ + */ │ │ │ │ │ + setUrl: function(url) { │ │ │ │ │ + this.icon.setUrl(url); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: display │ │ │ │ │ + * Hide or show the icon │ │ │ │ │ + * │ │ │ │ │ + * display - {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + display: function(display) { │ │ │ │ │ + this.icon.display(display); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Marker" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Function: defaultIcon │ │ │ │ │ + * Creates a default <OpenLayers.Icon>. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Icon>} A default OpenLayers.Icon to use for a marker │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Marker.defaultIcon = function() { │ │ │ │ │ + return new OpenLayers.Icon(OpenLayers.Util.getImageLocation("marker.png"), { │ │ │ │ │ + w: 21, │ │ │ │ │ + h: 25 │ │ │ │ │ + }, { │ │ │ │ │ + x: -10.5, │ │ │ │ │ + y: -25 │ │ │ │ │ + }); │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ OpenLayers/Strategy.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -42330,14 +41034,922 @@ │ │ │ │ │ } │ │ │ │ │ return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Strategy" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ + OpenLayers/Kinetic.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Animation.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Kinetic = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: threshold │ │ │ │ │ + * In most cases changing the threshold isn't needed. │ │ │ │ │ + * In px/ms, default to 0. │ │ │ │ │ + */ │ │ │ │ │ + threshold: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: deceleration │ │ │ │ │ + * {Float} the deseleration in px/ms², default to 0.0035. │ │ │ │ │ + */ │ │ │ │ │ + deceleration: 0.0035, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: nbPoints │ │ │ │ │ + * {Integer} the number of points we use to calculate the kinetic │ │ │ │ │ + * initial values. │ │ │ │ │ + */ │ │ │ │ │ + nbPoints: 100, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: delay │ │ │ │ │ + * {Float} time to consider to calculate the kinetic initial values. │ │ │ │ │ + * In ms, default to 200. │ │ │ │ │ + */ │ │ │ │ │ + delay: 200, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: points │ │ │ │ │ + * List of points use to calculate the kinetic initial values. │ │ │ │ │ + */ │ │ │ │ │ + points: undefined, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: timerId │ │ │ │ │ + * ID of the timer. │ │ │ │ │ + */ │ │ │ │ │ + timerId: undefined, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Kinetic │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: begin │ │ │ │ │ + * Begins the dragging. │ │ │ │ │ + */ │ │ │ │ │ + begin: function() { │ │ │ │ │ + OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ + this.timerId = undefined; │ │ │ │ │ + this.points = []; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: update │ │ │ │ │ + * Updates during the dragging. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} The new position. │ │ │ │ │ + */ │ │ │ │ │ + update: function(xy) { │ │ │ │ │ + this.points.unshift({ │ │ │ │ │ + xy: xy, │ │ │ │ │ + tick: new Date().getTime() │ │ │ │ │ + }); │ │ │ │ │ + if (this.points.length > this.nbPoints) { │ │ │ │ │ + this.points.pop(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: end │ │ │ │ │ + * Ends the dragging, start the kinetic. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} The last position. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object with two properties: "speed", and "theta". The │ │ │ │ │ + * "speed" and "theta" values are to be passed to the move │ │ │ │ │ + * function when starting the animation. │ │ │ │ │ + */ │ │ │ │ │ + end: function(xy) { │ │ │ │ │ + var last, now = new Date().getTime(); │ │ │ │ │ + for (var i = 0, l = this.points.length, point; i < l; i++) { │ │ │ │ │ + point = this.points[i]; │ │ │ │ │ + if (now - point.tick > this.delay) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + last = point; │ │ │ │ │ + } │ │ │ │ │ + if (!last) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var time = new Date().getTime() - last.tick; │ │ │ │ │ + var dist = Math.sqrt(Math.pow(xy.x - last.xy.x, 2) + │ │ │ │ │ + Math.pow(xy.y - last.xy.y, 2)); │ │ │ │ │ + var speed = dist / time; │ │ │ │ │ + if (speed == 0 || speed < this.threshold) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var theta = Math.asin((xy.y - last.xy.y) / dist); │ │ │ │ │ + if (last.xy.x <= xy.x) { │ │ │ │ │ + theta = Math.PI - theta; │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + speed: speed, │ │ │ │ │ + theta: theta │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: move │ │ │ │ │ + * Launch the kinetic move pan. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * info - {Object} An object with two properties, "speed", and "theta". │ │ │ │ │ + * These values are those returned from the "end" call. │ │ │ │ │ + * callback - {Function} Function called on every step of the animation, │ │ │ │ │ + * receives x, y (values to pan), end (is the last point). │ │ │ │ │ + */ │ │ │ │ │ + move: function(info, callback) { │ │ │ │ │ + var v0 = info.speed; │ │ │ │ │ + var fx = Math.cos(info.theta); │ │ │ │ │ + var fy = -Math.sin(info.theta); │ │ │ │ │ + │ │ │ │ │ + var initialTime = new Date().getTime(); │ │ │ │ │ + │ │ │ │ │ + var lastX = 0; │ │ │ │ │ + var lastY = 0; │ │ │ │ │ + │ │ │ │ │ + var timerCallback = function() { │ │ │ │ │ + if (this.timerId == null) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var t = new Date().getTime() - initialTime; │ │ │ │ │ + │ │ │ │ │ + var p = (-this.deceleration * Math.pow(t, 2)) / 2.0 + v0 * t; │ │ │ │ │ + var x = p * fx; │ │ │ │ │ + var y = p * fy; │ │ │ │ │ + │ │ │ │ │ + var args = {}; │ │ │ │ │ + args.end = false; │ │ │ │ │ + var v = -this.deceleration * t + v0; │ │ │ │ │ + │ │ │ │ │ + if (v <= 0) { │ │ │ │ │ + OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + args.end = true; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + args.x = x - lastX; │ │ │ │ │ + args.y = y - lastY; │ │ │ │ │ + lastX = x; │ │ │ │ │ + lastY = y; │ │ │ │ │ + callback(args.x, args.y, args.end); │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + this.timerId = OpenLayers.Animation.start( │ │ │ │ │ + OpenLayers.Function.bind(timerCallback, this) │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Kinetic" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/TileManager.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Element.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/Tile/Image.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.TileManager │ │ │ │ │ + * Provides queueing of image requests and caching of image elements. │ │ │ │ │ + * │ │ │ │ │ + * Queueing avoids unnecessary image requests while changing zoom levels │ │ │ │ │ + * quickly, and helps improve dragging performance on mobile devices that show │ │ │ │ │ + * a lag in dragging when loading of new images starts. <zoomDelay> and │ │ │ │ │ + * <moveDelay> are the configuration options to control this behavior. │ │ │ │ │ + * │ │ │ │ │ + * Caching avoids setting the src on image elements for images that have already │ │ │ │ │ + * been used. Several maps can share a TileManager instance, in which case each │ │ │ │ │ + * map gets its own tile queue, but all maps share the same tile cache. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.TileManager = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: cacheSize │ │ │ │ │ + * {Number} Number of image elements to keep referenced in this instance's │ │ │ │ │ + * cache for fast reuse. Default is 256. │ │ │ │ │ + */ │ │ │ │ │ + cacheSize: 256, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: tilesPerFrame │ │ │ │ │ + * {Number} Number of queued tiles to load per frame (see <frameDelay>). │ │ │ │ │ + * Default is 2. │ │ │ │ │ + */ │ │ │ │ │ + tilesPerFrame: 2, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: frameDelay │ │ │ │ │ + * {Number} Delay between tile loading frames (see <tilesPerFrame>) in │ │ │ │ │ + * milliseconds. Default is 16. │ │ │ │ │ + */ │ │ │ │ │ + frameDelay: 16, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: moveDelay │ │ │ │ │ + * {Number} Delay in milliseconds after a map's move event before loading │ │ │ │ │ + * tiles. Default is 100. │ │ │ │ │ + */ │ │ │ │ │ + moveDelay: 100, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomDelay │ │ │ │ │ + * {Number} Delay in milliseconds after a map's zoomend event before loading │ │ │ │ │ + * tiles. Default is 200. │ │ │ │ │ + */ │ │ │ │ │ + zoomDelay: 200, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: maps │ │ │ │ │ + * {Array(<OpenLayers.Map>)} The maps to manage tiles on. │ │ │ │ │ + */ │ │ │ │ │ + maps: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: tileQueueId │ │ │ │ │ + * {Object} The ids of the <drawTilesFromQueue> loop, keyed by map id. │ │ │ │ │ + */ │ │ │ │ │ + tileQueueId: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: tileQueue │ │ │ │ │ + * {Object(Array(<OpenLayers.Tile>))} Tiles queued for drawing, keyed by │ │ │ │ │ + * map id. │ │ │ │ │ + */ │ │ │ │ │ + tileQueue: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: tileCache │ │ │ │ │ + * {Object} Cached image elements, keyed by URL. │ │ │ │ │ + */ │ │ │ │ │ + tileCache: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: tileCacheIndex │ │ │ │ │ + * {Array(String)} URLs of cached tiles. First entry is the least recently │ │ │ │ │ + * used. │ │ │ │ │ + */ │ │ │ │ │ + tileCacheIndex: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.TileManager │ │ │ │ │ + * Constructor for a new <OpenLayers.TileManager> instance. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Configuration for this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.maps = []; │ │ │ │ │ + this.tileQueueId = {}; │ │ │ │ │ + this.tileQueue = {}; │ │ │ │ │ + this.tileCache = {}; │ │ │ │ │ + this.tileCacheIndex = []; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: addMap │ │ │ │ │ + * Binds this instance to a map │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + addMap: function(map) { │ │ │ │ │ + if (this._destroyed || !OpenLayers.Layer.Grid) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + this.maps.push(map); │ │ │ │ │ + this.tileQueue[map.id] = []; │ │ │ │ │ + for (var i = 0, ii = map.layers.length; i < ii; ++i) { │ │ │ │ │ + this.addLayer({ │ │ │ │ │ + layer: map.layers[i] │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + map.events.on({ │ │ │ │ │ + move: this.move, │ │ │ │ │ + zoomend: this.zoomEnd, │ │ │ │ │ + changelayer: this.changeLayer, │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + preremovelayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeMap │ │ │ │ │ + * Unbinds this instance from a map │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + if (this._destroyed || !OpenLayers.Layer.Grid) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + window.clearTimeout(this.tileQueueId[map.id]); │ │ │ │ │ + if (map.layers) { │ │ │ │ │ + for (var i = 0, ii = map.layers.length; i < ii; ++i) { │ │ │ │ │ + this.removeLayer({ │ │ │ │ │ + layer: map.layers[i] │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (map.events) { │ │ │ │ │ + map.events.un({ │ │ │ │ │ + move: this.move, │ │ │ │ │ + zoomend: this.zoomEnd, │ │ │ │ │ + changelayer: this.changeLayer, │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + preremovelayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + delete this.tileQueue[map.id]; │ │ │ │ │ + delete this.tileQueueId[map.id]; │ │ │ │ │ + OpenLayers.Util.removeItem(this.maps, map); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: move │ │ │ │ │ + * Handles the map's move event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} Listener argument │ │ │ │ │ + */ │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + this.updateTimeout(evt.object, this.moveDelay, true); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: zoomEnd │ │ │ │ │ + * Handles the map's zoomEnd event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} Listener argument │ │ │ │ │ + */ │ │ │ │ │ + zoomEnd: function(evt) { │ │ │ │ │ + this.updateTimeout(evt.object, this.zoomDelay); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: changeLayer │ │ │ │ │ + * Handles the map's changeLayer event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} Listener argument │ │ │ │ │ + */ │ │ │ │ │ + changeLayer: function(evt) { │ │ │ │ │ + if (evt.property === 'visibility' || evt.property === 'params') { │ │ │ │ │ + this.updateTimeout(evt.object, 0); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: addLayer │ │ │ │ │ + * Handles the map's addlayer event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} The listener argument │ │ │ │ │ + */ │ │ │ │ │ + addLayer: function(evt) { │ │ │ │ │ + var layer = evt.layer; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ + layer.events.on({ │ │ │ │ │ + addtile: this.addTile, │ │ │ │ │ + retile: this.clearTileQueue, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + var i, j, tile; │ │ │ │ │ + for (i = layer.grid.length - 1; i >= 0; --i) { │ │ │ │ │ + for (j = layer.grid[i].length - 1; j >= 0; --j) { │ │ │ │ │ + tile = layer.grid[i][j]; │ │ │ │ │ + this.addTile({ │ │ │ │ │ + tile: tile │ │ │ │ │ + }); │ │ │ │ │ + if (tile.url && !tile.imgDiv) { │ │ │ │ │ + this.manageTileCache({ │ │ │ │ │ + object: tile │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeLayer │ │ │ │ │ + * Handles the map's preremovelayer event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} The listener argument │ │ │ │ │ + */ │ │ │ │ │ + removeLayer: function(evt) { │ │ │ │ │ + var layer = evt.layer; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ + this.clearTileQueue({ │ │ │ │ │ + object: layer │ │ │ │ │ + }); │ │ │ │ │ + if (layer.events) { │ │ │ │ │ + layer.events.un({ │ │ │ │ │ + addtile: this.addTile, │ │ │ │ │ + retile: this.clearTileQueue, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + if (layer.grid) { │ │ │ │ │ + var i, j, tile; │ │ │ │ │ + for (i = layer.grid.length - 1; i >= 0; --i) { │ │ │ │ │ + for (j = layer.grid[i].length - 1; j >= 0; --j) { │ │ │ │ │ + tile = layer.grid[i][j]; │ │ │ │ │ + this.unloadTile({ │ │ │ │ │ + object: tile │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateTimeout │ │ │ │ │ + * Applies the <moveDelay> or <zoomDelay> to the <drawTilesFromQueue> loop, │ │ │ │ │ + * and schedules more queue processing after <frameDelay> if there are still │ │ │ │ │ + * tiles in the queue. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} The map to update the timeout for │ │ │ │ │ + * delay - {Number} The delay to apply │ │ │ │ │ + * nice - {Boolean} If true, the timeout function will only be created if │ │ │ │ │ + * the tilequeue is not empty. This is used by the move handler to │ │ │ │ │ + * avoid impacts on dragging performance. For other events, the tile │ │ │ │ │ + * queue may not be populated yet, so we need to set the timer │ │ │ │ │ + * regardless of the queue size. │ │ │ │ │ + */ │ │ │ │ │ + updateTimeout: function(map, delay, nice) { │ │ │ │ │ + window.clearTimeout(this.tileQueueId[map.id]); │ │ │ │ │ + var tileQueue = this.tileQueue[map.id]; │ │ │ │ │ + if (!nice || tileQueue.length) { │ │ │ │ │ + this.tileQueueId[map.id] = window.setTimeout( │ │ │ │ │ + OpenLayers.Function.bind(function() { │ │ │ │ │ + this.drawTilesFromQueue(map); │ │ │ │ │ + if (tileQueue.length) { │ │ │ │ │ + this.updateTimeout(map, this.frameDelay); │ │ │ │ │ + } │ │ │ │ │ + }, this), delay │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: addTile │ │ │ │ │ + * Listener for the layer's addtile event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} The listener argument │ │ │ │ │ + */ │ │ │ │ │ + addTile: function(evt) { │ │ │ │ │ + if (evt.tile instanceof OpenLayers.Tile.Image) { │ │ │ │ │ + evt.tile.events.on({ │ │ │ │ │ + beforedraw: this.queueTileDraw, │ │ │ │ │ + beforeload: this.manageTileCache, │ │ │ │ │ + loadend: this.addToCache, │ │ │ │ │ + unload: this.unloadTile, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + // Layer has the wrong tile type, so don't handle it any longer │ │ │ │ │ + this.removeLayer({ │ │ │ │ │ + layer: evt.tile.layer │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: unloadTile │ │ │ │ │ + * Listener for the tile's unload event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} The listener argument │ │ │ │ │ + */ │ │ │ │ │ + unloadTile: function(evt) { │ │ │ │ │ + var tile = evt.object; │ │ │ │ │ + tile.events.un({ │ │ │ │ │ + beforedraw: this.queueTileDraw, │ │ │ │ │ + beforeload: this.manageTileCache, │ │ │ │ │ + loadend: this.addToCache, │ │ │ │ │ + unload: this.unloadTile, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Util.removeItem(this.tileQueue[tile.layer.map.id], tile); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: queueTileDraw │ │ │ │ │ + * Adds a tile to the queue that will draw it. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} Listener argument of the tile's beforedraw event │ │ │ │ │ + */ │ │ │ │ │ + queueTileDraw: function(evt) { │ │ │ │ │ + var tile = evt.object; │ │ │ │ │ + var queued = false; │ │ │ │ │ + var layer = tile.layer; │ │ │ │ │ + var url = layer.getURL(tile.bounds); │ │ │ │ │ + var img = this.tileCache[url]; │ │ │ │ │ + if (img && img.className !== 'olTileImage') { │ │ │ │ │ + // cached image no longer valid, e.g. because we're olTileReplacing │ │ │ │ │ + delete this.tileCache[url]; │ │ │ │ │ + OpenLayers.Util.removeItem(this.tileCacheIndex, url); │ │ │ │ │ + img = null; │ │ │ │ │ + } │ │ │ │ │ + // queue only if image with same url not cached already │ │ │ │ │ + if (layer.url && (layer.async || !img)) { │ │ │ │ │ + // add to queue only if not in queue already │ │ │ │ │ + var tileQueue = this.tileQueue[layer.map.id]; │ │ │ │ │ + if (!~OpenLayers.Util.indexOf(tileQueue, tile)) { │ │ │ │ │ + tileQueue.push(tile); │ │ │ │ │ + } │ │ │ │ │ + queued = true; │ │ │ │ │ + } │ │ │ │ │ + return !queued; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawTilesFromQueue │ │ │ │ │ + * Draws tiles from the tileQueue, and unqueues the tiles │ │ │ │ │ + */ │ │ │ │ │ + drawTilesFromQueue: function(map) { │ │ │ │ │ + var tileQueue = this.tileQueue[map.id]; │ │ │ │ │ + var limit = this.tilesPerFrame; │ │ │ │ │ + var animating = map.zoomTween && map.zoomTween.playing; │ │ │ │ │ + while (!animating && tileQueue.length && limit) { │ │ │ │ │ + tileQueue.shift().draw(true); │ │ │ │ │ + --limit; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: manageTileCache │ │ │ │ │ + * Adds, updates, removes and fetches cache entries. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} Listener argument of the tile's beforeload event │ │ │ │ │ + */ │ │ │ │ │ + manageTileCache: function(evt) { │ │ │ │ │ + var tile = evt.object; │ │ │ │ │ + var img = this.tileCache[tile.url]; │ │ │ │ │ + if (img) { │ │ │ │ │ + // if image is on its layer's backbuffer, remove it from backbuffer │ │ │ │ │ + if (img.parentNode && │ │ │ │ │ + OpenLayers.Element.hasClass(img.parentNode, 'olBackBuffer')) { │ │ │ │ │ + img.parentNode.removeChild(img); │ │ │ │ │ + img.id = null; │ │ │ │ │ + } │ │ │ │ │ + // only use image from cache if it is not on a layer already │ │ │ │ │ + if (!img.parentNode) { │ │ │ │ │ + img.style.visibility = 'hidden'; │ │ │ │ │ + img.style.opacity = 0; │ │ │ │ │ + tile.setImage(img); │ │ │ │ │ + // LRU - move tile to the end of the array to mark it as the most │ │ │ │ │ + // recently used │ │ │ │ │ + OpenLayers.Util.removeItem(this.tileCacheIndex, tile.url); │ │ │ │ │ + this.tileCacheIndex.push(tile.url); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: addToCache │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} Listener argument for the tile's loadend event │ │ │ │ │ + */ │ │ │ │ │ + addToCache: function(evt) { │ │ │ │ │ + var tile = evt.object; │ │ │ │ │ + if (!this.tileCache[tile.url]) { │ │ │ │ │ + if (!OpenLayers.Element.hasClass(tile.imgDiv, 'olImageLoadError')) { │ │ │ │ │ + if (this.tileCacheIndex.length >= this.cacheSize) { │ │ │ │ │ + delete this.tileCache[this.tileCacheIndex[0]]; │ │ │ │ │ + this.tileCacheIndex.shift(); │ │ │ │ │ + } │ │ │ │ │ + this.tileCache[tile.url] = tile.imgDiv; │ │ │ │ │ + this.tileCacheIndex.push(tile.url); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clearTileQueue │ │ │ │ │ + * Clears the tile queue from tiles of a specific layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} Listener argument of the layer's retile event │ │ │ │ │ + */ │ │ │ │ │ + clearTileQueue: function(evt) { │ │ │ │ │ + var layer = evt.object; │ │ │ │ │ + var tileQueue = this.tileQueue[layer.map.id]; │ │ │ │ │ + for (var i = tileQueue.length - 1; i >= 0; --i) { │ │ │ │ │ + if (tileQueue[i].layer === layer) { │ │ │ │ │ + tileQueue.splice(i, 1); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i = this.maps.length - 1; i >= 0; --i) { │ │ │ │ │ + this.removeMap(this.maps[i]); │ │ │ │ │ + } │ │ │ │ │ + this.maps = null; │ │ │ │ │ + this.tileQueue = null; │ │ │ │ │ + this.tileQueueId = null; │ │ │ │ │ + this.tileCache = null; │ │ │ │ │ + this.tileCacheIndex = null; │ │ │ │ │ + this._destroyed = true; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Rule.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + * @requires OpenLayers/Style.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Rule │ │ │ │ │ + * This class represents an SLD Rule, as being used for rule-based SLD styling. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Rule = OpenLayers.Class({ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: id │ │ │ │ │ + * {String} A unique id for this session. │ │ │ │ │ + */ │ │ │ │ │ + id: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: name │ │ │ │ │ + * {String} name of this rule │ │ │ │ │ + */ │ │ │ │ │ + name: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: title │ │ │ │ │ + * {String} Title of this rule (set if included in SLD) │ │ │ │ │ + */ │ │ │ │ │ + title: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: description │ │ │ │ │ + * {String} Description of this rule (set if abstract is included in SLD) │ │ │ │ │ + */ │ │ │ │ │ + description: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: context │ │ │ │ │ + * {Object} An optional object with properties that the rule should be │ │ │ │ │ + * evaluated against. If no context is specified, feature.attributes will │ │ │ │ │ + * be used. │ │ │ │ │ + */ │ │ │ │ │ + context: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: filter │ │ │ │ │ + * {<OpenLayers.Filter>} Optional filter for the rule. │ │ │ │ │ + */ │ │ │ │ │ + filter: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: elseFilter │ │ │ │ │ + * {Boolean} Determines whether this rule is only to be applied only if │ │ │ │ │ + * no other rules match (ElseFilter according to the SLD specification). │ │ │ │ │ + * Default is false. For instances of OpenLayers.Rule, if elseFilter is │ │ │ │ │ + * false, the rule will always apply. For subclasses, the else property is │ │ │ │ │ + * ignored. │ │ │ │ │ + */ │ │ │ │ │ + elseFilter: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: symbolizer │ │ │ │ │ + * {Object} Symbolizer or hash of symbolizers for this rule. If hash of │ │ │ │ │ + * symbolizers, keys are one or more of ["Point", "Line", "Polygon"]. The │ │ │ │ │ + * latter if useful if it is required to style e.g. vertices of a line │ │ │ │ │ + * with a point symbolizer. Note, however, that this is not implemented │ │ │ │ │ + * yet in OpenLayers, but it is the way how symbolizers are defined in │ │ │ │ │ + * SLD. │ │ │ │ │ + */ │ │ │ │ │ + symbolizer: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: symbolizers │ │ │ │ │ + * {Array} Collection of symbolizers associated with this rule. If │ │ │ │ │ + * provided at construction, the symbolizers array has precedence │ │ │ │ │ + * over the deprecated symbolizer property. Note that multiple │ │ │ │ │ + * symbolizers are not currently supported by the vector renderers. │ │ │ │ │ + * Rules with multiple symbolizers are currently only useful for │ │ │ │ │ + * maintaining elements in an SLD document. │ │ │ │ │ + */ │ │ │ │ │ + symbolizers: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: minScaleDenominator │ │ │ │ │ + * {Number} or {String} minimum scale at which to draw the feature. │ │ │ │ │ + * In the case of a String, this can be a combination of text and │ │ │ │ │ + * propertyNames in the form "literal ${propertyName}" │ │ │ │ │ + */ │ │ │ │ │ + minScaleDenominator: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxScaleDenominator │ │ │ │ │ + * {Number} or {String} maximum scale at which to draw the feature. │ │ │ │ │ + * In the case of a String, this can be a combination of text and │ │ │ │ │ + * propertyNames in the form "literal ${propertyName}" │ │ │ │ │ + */ │ │ │ │ │ + maxScaleDenominator: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Rule │ │ │ │ │ + * Creates a Rule. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object with properties to set on the │ │ │ │ │ + * rule │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Rule>} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.symbolizer = {}; │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + if (this.symbolizers) { │ │ │ │ │ + delete this.symbolizer; │ │ │ │ │ + } │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * nullify references to prevent circular references and memory leaks │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i in this.symbolizer) { │ │ │ │ │ + this.symbolizer[i] = null; │ │ │ │ │ + } │ │ │ │ │ + this.symbolizer = null; │ │ │ │ │ + delete this.symbolizers; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: evaluate │ │ │ │ │ + * evaluates this rule for a specific feature │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature>} feature to apply the rule to. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the rule applies, false if it does not. │ │ │ │ │ + * This rule is the default rule and always returns true. │ │ │ │ │ + */ │ │ │ │ │ + evaluate: function(feature) { │ │ │ │ │ + var context = this.getContext(feature); │ │ │ │ │ + var applies = true; │ │ │ │ │ + │ │ │ │ │ + if (this.minScaleDenominator || this.maxScaleDenominator) { │ │ │ │ │ + var scale = feature.layer.map.getScale(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // check if within minScale/maxScale bounds │ │ │ │ │ + if (this.minScaleDenominator) { │ │ │ │ │ + applies = scale >= OpenLayers.Style.createLiteral( │ │ │ │ │ + this.minScaleDenominator, context); │ │ │ │ │ + } │ │ │ │ │ + if (applies && this.maxScaleDenominator) { │ │ │ │ │ + applies = scale < OpenLayers.Style.createLiteral( │ │ │ │ │ + this.maxScaleDenominator, context); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // check if optional filter applies │ │ │ │ │ + if (applies && this.filter) { │ │ │ │ │ + // feature id filters get the feature, others get the context │ │ │ │ │ + if (this.filter.CLASS_NAME == "OpenLayers.Filter.FeatureId") { │ │ │ │ │ + applies = this.filter.evaluate(feature); │ │ │ │ │ + } else { │ │ │ │ │ + applies = this.filter.evaluate(context); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return applies; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getContext │ │ │ │ │ + * Gets the context for evaluating this rule │ │ │ │ │ + * │ │ │ │ │ + * Paramters: │ │ │ │ │ + * feature - {<OpenLayers.Feature>} feature to take the context from if │ │ │ │ │ + * none is specified. │ │ │ │ │ + */ │ │ │ │ │ + getContext: function(feature) { │ │ │ │ │ + var context = this.context; │ │ │ │ │ + if (!context) { │ │ │ │ │ + context = feature.attributes || feature.data; │ │ │ │ │ + } │ │ │ │ │ + if (typeof this.context == "function") { │ │ │ │ │ + context = this.context(feature); │ │ │ │ │ + } │ │ │ │ │ + return context; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Clones this rule. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Rule>} Clone of this rule. │ │ │ │ │ + */ │ │ │ │ │ + clone: function() { │ │ │ │ │ + var options = OpenLayers.Util.extend({}, this); │ │ │ │ │ + if (this.symbolizers) { │ │ │ │ │ + // clone symbolizers │ │ │ │ │ + var len = this.symbolizers.length; │ │ │ │ │ + options.symbolizers = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + options.symbolizers[i] = this.symbolizers[i].clone(); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // clone symbolizer │ │ │ │ │ + options.symbolizer = {}; │ │ │ │ │ + var value, type; │ │ │ │ │ + for (var key in this.symbolizer) { │ │ │ │ │ + value = this.symbolizer[key]; │ │ │ │ │ + type = typeof value; │ │ │ │ │ + if (type === "object") { │ │ │ │ │ + options.symbolizer[key] = OpenLayers.Util.extend({}, value); │ │ │ │ │ + } else if (type === "string") { │ │ │ │ │ + options.symbolizer[key] = value; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // clone filter │ │ │ │ │ + options.filter = this.filter && this.filter.clone(); │ │ │ │ │ + // clone context │ │ │ │ │ + options.context = this.context && OpenLayers.Util.extend({}, this.context); │ │ │ │ │ + return new OpenLayers.Rule(options); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Rule" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ OpenLayers/Format/WPSDescribeProcess.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -42746,84 +42358,72 @@ │ │ │ │ │ this.servers = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: 'OpenLayers.WPSClient' │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Spherical.js │ │ │ │ │ + OpenLayers/Symbolizer.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/SingleFile.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: Spherical │ │ │ │ │ - * The OpenLayers.Spherical namespace includes utility functions for │ │ │ │ │ - * calculations on the basis of a spherical earth (ignoring ellipsoidal │ │ │ │ │ - * effects), which is accurate enough for most purposes. │ │ │ │ │ - * │ │ │ │ │ - * Relevant links: │ │ │ │ │ - * * http://www.movable-type.co.uk/scripts/latlong.html │ │ │ │ │ - * * http://code.google.com/apis/maps/documentation/javascript/reference.html#spherical │ │ │ │ │ + * Class: OpenLayers.Symbolizer │ │ │ │ │ + * Base class representing a symbolizer used for feature rendering. │ │ │ │ │ */ │ │ │ │ │ +OpenLayers.Symbolizer = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ -OpenLayers.Spherical = OpenLayers.Spherical || {}; │ │ │ │ │ │ │ │ │ │ -OpenLayers.Spherical.DEFAULT_RADIUS = 6378137; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zIndex │ │ │ │ │ + * {Number} The zIndex determines the rendering order for a symbolizer. │ │ │ │ │ + * Symbolizers with larger zIndex values are rendered over symbolizers │ │ │ │ │ + * with smaller zIndex values. Default is 0. │ │ │ │ │ + */ │ │ │ │ │ + zIndex: 0, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * APIFunction: computeDistanceBetween │ │ │ │ │ - * Computes the distance between two LonLats. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * from - {<OpenLayers.LonLat>} or {Object} Starting point. A LonLat or │ │ │ │ │ - * a JavaScript literal with lon lat properties. │ │ │ │ │ - * to - {<OpenLayers.LonLat>} or {Object} Ending point. A LonLat or a │ │ │ │ │ - * JavaScript literal with lon lat properties. │ │ │ │ │ - * radius - {Float} The radius. Optional. Defaults to 6378137 meters. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The distance in meters. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Spherical.computeDistanceBetween = function(from, to, radius) { │ │ │ │ │ - var R = radius || OpenLayers.Spherical.DEFAULT_RADIUS; │ │ │ │ │ - var sinHalfDeltaLon = Math.sin(Math.PI * (to.lon - from.lon) / 360); │ │ │ │ │ - var sinHalfDeltaLat = Math.sin(Math.PI * (to.lat - from.lat) / 360); │ │ │ │ │ - var a = sinHalfDeltaLat * sinHalfDeltaLat + │ │ │ │ │ - sinHalfDeltaLon * sinHalfDeltaLon * Math.cos(Math.PI * from.lat / 180) * Math.cos(Math.PI * to.lat / 180); │ │ │ │ │ - return 2 * R * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a)); │ │ │ │ │ -}; │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Symbolizer │ │ │ │ │ + * Instances of this class are not useful. See one of the subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} An object containing properties to be set on the │ │ │ │ │ + * symbolizer. Any documented symbolizer property can be set at │ │ │ │ │ + * construction. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * A new symbolizer. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ + OpenLayers.Util.extend(this, config); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Create a copy of this symbolizer. │ │ │ │ │ + * │ │ │ │ │ + * Returns a symbolizer of the same type with the same properties. │ │ │ │ │ + */ │ │ │ │ │ + clone: function() { │ │ │ │ │ + var Type = eval(this.CLASS_NAME); │ │ │ │ │ + return new Type(OpenLayers.Util.extend({}, this)); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Symbolizer" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * APIFunction: computeHeading │ │ │ │ │ - * Computes the heading from one LonLat to another LonLat. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * from - {<OpenLayers.LonLat>} or {Object} Starting point. A LonLat or │ │ │ │ │ - * a JavaScript literal with lon lat properties. │ │ │ │ │ - * to - {<OpenLayers.LonLat>} or {Object} Ending point. A LonLat or a │ │ │ │ │ - * JavaScript literal with lon lat properties. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The heading in degrees. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Spherical.computeHeading = function(from, to) { │ │ │ │ │ - var y = Math.sin(Math.PI * (from.lon - to.lon) / 180) * Math.cos(Math.PI * to.lat / 180); │ │ │ │ │ - var x = Math.cos(Math.PI * from.lat / 180) * Math.sin(Math.PI * to.lat / 180) - │ │ │ │ │ - Math.sin(Math.PI * from.lat / 180) * Math.cos(Math.PI * to.lat / 180) * Math.cos(Math.PI * (from.lon - to.lon) / 180); │ │ │ │ │ - return 180 * Math.atan2(y, x) / Math.PI; │ │ │ │ │ -}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Symbolizer/Point.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ @@ -43376,16171 +42976,11970 @@ │ │ │ │ │ } │ │ │ │ │ return new OpenLayers.Style2(config); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Style2" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/TileCache.js │ │ │ │ │ + OpenLayers/Handler.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Class.js │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.TileCache │ │ │ │ │ - * A read only TileCache layer. Used to requests tiles cached by TileCache in │ │ │ │ │ - * a web accessible cache. This means that you have to pre-populate your │ │ │ │ │ - * cache before this layer can be used. It is meant only to read tiles │ │ │ │ │ - * created by TileCache, and not to make calls to TileCache for tile │ │ │ │ │ - * creation. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Layer.TileCache> constructor. │ │ │ │ │ + * Class: OpenLayers.Handler │ │ │ │ │ + * Base class to construct a higher-level handler for event sequences. All │ │ │ │ │ + * handlers have activate and deactivate methods. In addition, they have │ │ │ │ │ + * methods named like browser events. When a handler is activated, any │ │ │ │ │ + * additional methods named like a browser event is registered as a │ │ │ │ │ + * listener for the corresponding event. When a handler is deactivated, │ │ │ │ │ + * those same methods are unregistered as event listeners. │ │ │ │ │ * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ + * Handlers also typically have a callbacks object with keys named like │ │ │ │ │ + * the abstracted events or event sequences that they are in charge of │ │ │ │ │ + * handling. The controls that wrap handlers define the methods that │ │ │ │ │ + * correspond to these abstract events - so instead of listening for │ │ │ │ │ + * individual browser events, they only listen for the abstract events │ │ │ │ │ + * defined by the handler. │ │ │ │ │ + * │ │ │ │ │ + * Handlers are created by controls, which ultimately have the responsibility │ │ │ │ │ + * of making changes to the the state of the application. Handlers │ │ │ │ │ + * themselves may make temporary changes, but in general are expected to │ │ │ │ │ + * return the application in the same state that they found it. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.TileCache = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ +OpenLayers.Handler = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} Treat this layer as a base layer. Default is true. │ │ │ │ │ + /** │ │ │ │ │ + * Property: id │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + id: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: format │ │ │ │ │ - * {String} Mime type of the images returned. Default is image/png. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: control │ │ │ │ │ + * {<OpenLayers.Control>}. The control that initialized this handler. The │ │ │ │ │ + * control is assumed to have a valid map property - that map is used │ │ │ │ │ + * in the handler's own setMap method. │ │ │ │ │ */ │ │ │ │ │ - format: 'image/png', │ │ │ │ │ + control: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: serverResolutions │ │ │ │ │ - * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ - * property if the map resolutions differ from the server. This │ │ │ │ │ - * property serves two purposes. (a) <serverResolutions> can include │ │ │ │ │ - * resolutions that the server supports and that you don't want to │ │ │ │ │ - * provide with this layer. (b) The map can work with resolutions │ │ │ │ │ - * that aren't supported by the server, i.e. that aren't in │ │ │ │ │ - * <serverResolutions>. When the map is displayed in such a resolution │ │ │ │ │ - * data for the closest server-supported resolution is loaded and the │ │ │ │ │ - * layer div is stretched as necessary. │ │ │ │ │ + * Property: map │ │ │ │ │ + * {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ + map: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.TileCache │ │ │ │ │ - * Create a new read only TileCache layer. │ │ │ │ │ + * APIProperty: keyMask │ │ │ │ │ + * {Integer} Use bitwise operators and one or more of the OpenLayers.Handler │ │ │ │ │ + * constants to construct a keyMask. The keyMask is used by │ │ │ │ │ + * <checkModifiers>. If the keyMask matches the combination of keys │ │ │ │ │ + * down on an event, checkModifiers returns true. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} Name of the layer displayed in the interface │ │ │ │ │ - * url - {String} Location of the web accessible cache (not the location of │ │ │ │ │ - * your tilecache script!) │ │ │ │ │ - * layername - {String} Layer name as defined in the TileCache │ │ │ │ │ - * configuration │ │ │ │ │ - * options - {Object} Optional object with properties to be set on the │ │ │ │ │ - * layer. Note that you should speficy your resolutions to match │ │ │ │ │ - * your TileCache configuration. This can be done by setting │ │ │ │ │ - * the resolutions array directly (here or on the map), by setting │ │ │ │ │ - * maxResolution and numZoomLevels, or by using scale based properties. │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * // handler only responds if the Shift key is down │ │ │ │ │ + * handler.keyMask = OpenLayers.Handler.MOD_SHIFT; │ │ │ │ │ + * │ │ │ │ │ + * // handler only responds if Ctrl-Shift is down │ │ │ │ │ + * handler.keyMask = OpenLayers.Handler.MOD_SHIFT | │ │ │ │ │ + * OpenLayers.Handler.MOD_CTRL; │ │ │ │ │ + * (end) │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, layername, options) { │ │ │ │ │ - this.layername = layername; │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, │ │ │ │ │ - [name, url, {}, options]); │ │ │ │ │ - this.extension = this.format.split('/')[1].toLowerCase(); │ │ │ │ │ - this.extension = (this.extension == 'jpg') ? 'jpeg' : this.extension; │ │ │ │ │ - }, │ │ │ │ │ + keyMask: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * obj - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.TileCache>} An exact clone of this │ │ │ │ │ - * <OpenLayers.Layer.TileCache> │ │ │ │ │ + * Property: active │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.TileCache(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.layername, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + active: false, │ │ │ │ │ │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ + /** │ │ │ │ │ + * Property: evt │ │ │ │ │ + * {Event} This property references the last event handled by the handler. │ │ │ │ │ + * Note that this property is not part of the stable API. Use of the │ │ │ │ │ + * evt property should be restricted to controls in the library │ │ │ │ │ + * or other applications that are willing to update with changes to │ │ │ │ │ + * the OpenLayers code. │ │ │ │ │ + */ │ │ │ │ │ + evt: null, │ │ │ │ │ │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: touch │ │ │ │ │ + * {Boolean} Indicates the support of touch events. When touch events are │ │ │ │ │ + * started touch will be true and all mouse related listeners will do │ │ │ │ │ + * nothing. │ │ │ │ │ + */ │ │ │ │ │ + touch: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ + * Constructor: OpenLayers.Handler │ │ │ │ │ + * Construct a handler. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as parameters. │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that initialized this │ │ │ │ │ + * handler. The control is assumed to have a valid map property; that │ │ │ │ │ + * map is used in the handler's own setMap method. If a map property │ │ │ │ │ + * is present in the options argument it will be used instead. │ │ │ │ │ + * callbacks - {Object} An object whose properties correspond to abstracted │ │ │ │ │ + * events or sequences of browser events. The values for these │ │ │ │ │ + * properties are functions defined by the control that get called by │ │ │ │ │ + * the handler. │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * the handler. │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var bbox = this.maxExtent; │ │ │ │ │ - var size = this.tileSize; │ │ │ │ │ - var tileX = Math.round((bounds.left - bbox.left) / (res * size.w)); │ │ │ │ │ - var tileY = Math.round((bounds.bottom - bbox.bottom) / (res * size.h)); │ │ │ │ │ - var tileZ = this.serverResolutions != null ? │ │ │ │ │ - OpenLayers.Util.indexOf(this.serverResolutions, res) : │ │ │ │ │ - this.map.getZoom(); │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.control = control; │ │ │ │ │ + this.callbacks = callbacks; │ │ │ │ │ │ │ │ │ │ - var components = [ │ │ │ │ │ - this.layername, │ │ │ │ │ - OpenLayers.Number.zeroPad(tileZ, 2), │ │ │ │ │ - OpenLayers.Number.zeroPad(parseInt(tileX / 1000000), 3), │ │ │ │ │ - OpenLayers.Number.zeroPad((parseInt(tileX / 1000) % 1000), 3), │ │ │ │ │ - OpenLayers.Number.zeroPad((parseInt(tileX) % 1000), 3), │ │ │ │ │ - OpenLayers.Number.zeroPad(parseInt(tileY / 1000000), 3), │ │ │ │ │ - OpenLayers.Number.zeroPad((parseInt(tileY / 1000) % 1000), 3), │ │ │ │ │ - OpenLayers.Number.zeroPad((parseInt(tileY) % 1000), 3) + '.' + this.extension │ │ │ │ │ - ]; │ │ │ │ │ - var path = components.join('/'); │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(path, url); │ │ │ │ │ + var map = this.map || control.map; │ │ │ │ │ + if (map) { │ │ │ │ │ + this.setMap(map); │ │ │ │ │ } │ │ │ │ │ - url = (url.charAt(url.length - 1) == '/') ? url : url + '/'; │ │ │ │ │ - return url + path; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.TileCache" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/WorldWind.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_"); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.map = map; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.WorldWind │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.WorldWind = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + /** │ │ │ │ │ + * Method: checkModifiers │ │ │ │ │ + * Check the keyMask on the handler. If no <keyMask> is set, this always │ │ │ │ │ + * returns true. If a <keyMask> is set and it matches the combination │ │ │ │ │ + * of keys down on an event, this returns true. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The keyMask matches the keys down on an event. │ │ │ │ │ + */ │ │ │ │ │ + checkModifiers: function(evt) { │ │ │ │ │ + if (this.keyMask == null) { │ │ │ │ │ + return true; │ │ │ │ │ + } │ │ │ │ │ + /* calculate the keyboard modifier mask for this event */ │ │ │ │ │ + var keyModifiers = │ │ │ │ │ + (evt.shiftKey ? OpenLayers.Handler.MOD_SHIFT : 0) | │ │ │ │ │ + (evt.ctrlKey ? OpenLayers.Handler.MOD_CTRL : 0) | │ │ │ │ │ + (evt.altKey ? OpenLayers.Handler.MOD_ALT : 0) | │ │ │ │ │ + (evt.metaKey ? OpenLayers.Handler.MOD_META : 0); │ │ │ │ │ │ │ │ │ │ - DEFAULT_PARAMS: {}, │ │ │ │ │ + /* if it differs from the handler object's key mask, │ │ │ │ │ + bail out of the event handler */ │ │ │ │ │ + return (keyModifiers == this.keyMask); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} WorldWind layer is a base layer by default. │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Turn on the handler. Returns false if the handler was already active. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was activated. │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + // register for event handlers defined on this class. │ │ │ │ │ + var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.register(events[i], this[events[i]]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.active = true; │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: lzd │ │ │ │ │ - * {Float} LevelZeroTileSizeDegrees │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Turn off the handler. Returns false if the handler was already inactive. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was deactivated. │ │ │ │ │ */ │ │ │ │ │ - lzd: null, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + // unregister event handlers defined on this class. │ │ │ │ │ + var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.unregister(events[i], this[events[i]]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.touch = false; │ │ │ │ │ + this.active = false; │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomLevels │ │ │ │ │ - * {Integer} Number of zoom levels. │ │ │ │ │ + * Method: startTouch │ │ │ │ │ + * Start touch events, this method must be called by subclasses in │ │ │ │ │ + * "touchstart" method. When touch events are started <touch> will be │ │ │ │ │ + * true and all mouse related listeners will do nothing. │ │ │ │ │ */ │ │ │ │ │ - zoomLevels: null, │ │ │ │ │ + startTouch: function() { │ │ │ │ │ + if (!this.touch) { │ │ │ │ │ + this.touch = true; │ │ │ │ │ + var events = [ │ │ │ │ │ + "mousedown", "mouseup", "mousemove", "click", "dblclick", │ │ │ │ │ + "mouseout" │ │ │ │ │ + ]; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.unregister(events[i], this[events[i]]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.WorldWind │ │ │ │ │ - * │ │ │ │ │ + * Method: callback │ │ │ │ │ + * Trigger the control's named callback with the given arguments │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} Name of Layer │ │ │ │ │ - * url - {String} Base URL │ │ │ │ │ - * lzd - {Float} Level zero tile size degrees │ │ │ │ │ - * zoomLevels - {Integer} number of zoom levels │ │ │ │ │ - * params - {Object} additional parameters │ │ │ │ │ - * options - {Object} additional options │ │ │ │ │ + * name - {String} The key for the callback that is one of the properties │ │ │ │ │ + * of the handler's callbacks object. │ │ │ │ │ + * args - {Array(*)} An array of arguments (any type) with which to call │ │ │ │ │ + * the callback (defined by the control). │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, lzd, zoomLevels, params, options) { │ │ │ │ │ - this.lzd = lzd; │ │ │ │ │ - this.zoomLevels = zoomLevels; │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - newArguments.push(name, url, params, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - this.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, this.DEFAULT_PARAMS │ │ │ │ │ - ); │ │ │ │ │ + callback: function(name, args) { │ │ │ │ │ + if (name && this.callbacks[name]) { │ │ │ │ │ + this.callbacks[name].apply(this.control, args); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getZoom │ │ │ │ │ - * Convert map zoom to WW zoom. │ │ │ │ │ + * Method: register │ │ │ │ │ + * register an event on the map │ │ │ │ │ */ │ │ │ │ │ - getZoom: function() { │ │ │ │ │ - var zoom = this.map.getZoom(); │ │ │ │ │ - var extent = this.map.getMaxExtent(); │ │ │ │ │ - zoom = zoom - Math.log(this.maxResolution / (this.lzd / 512)) / Math.log(2); │ │ │ │ │ - return zoom; │ │ │ │ │ + register: function(name, method) { │ │ │ │ │ + // TODO: deal with registerPriority in 3.0 │ │ │ │ │ + this.map.events.registerPriority(name, this, method); │ │ │ │ │ + this.map.events.registerPriority(name, this, this.setEvent); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ + * Method: unregister │ │ │ │ │ + * unregister an event from the map │ │ │ │ │ + */ │ │ │ │ │ + unregister: function(name, method) { │ │ │ │ │ + this.map.events.unregister(name, this, method); │ │ │ │ │ + this.map.events.unregister(name, this, this.setEvent); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setEvent │ │ │ │ │ + * With each registered browser event, the handler sets its own evt │ │ │ │ │ + * property. This property can be accessed by controls if needed │ │ │ │ │ + * to get more information about the event that the handler is │ │ │ │ │ + * processing. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * This allows modifier keys on the event to be checked (alt, shift, ctrl, │ │ │ │ │ + * and meta cannot be checked with the keyboard handler). For a │ │ │ │ │ + * control to determine which modifier keys are associated with the │ │ │ │ │ + * event that a handler is currently processing, it should access │ │ │ │ │ + * (code)handler.evt.altKey || handler.evt.shiftKey || │ │ │ │ │ + * handler.evt.ctrlKey || handler.evt.metaKey(end). │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as │ │ │ │ │ - * parameters │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event. │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var zoom = this.getZoom(); │ │ │ │ │ - var extent = this.map.getMaxExtent(); │ │ │ │ │ - var deg = this.lzd / Math.pow(2, this.getZoom()); │ │ │ │ │ - var x = Math.floor((bounds.left - extent.left) / deg); │ │ │ │ │ - var y = Math.floor((bounds.bottom - extent.bottom) / deg); │ │ │ │ │ - if (this.map.getResolution() <= (this.lzd / 512) && │ │ │ │ │ - this.getZoom() <= this.zoomLevels) { │ │ │ │ │ - return this.getFullRequestString({ │ │ │ │ │ - L: zoom, │ │ │ │ │ - X: x, │ │ │ │ │ - Y: y │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - return OpenLayers.Util.getImageLocation("blank.gif"); │ │ │ │ │ - } │ │ │ │ │ + setEvent: function(evt) { │ │ │ │ │ + this.evt = evt; │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Deconstruct the handler. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + // unregister event listeners │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + // eliminate circular references │ │ │ │ │ + this.control = this.map = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.WorldWind" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_NONE │ │ │ │ │ + * If set as the <keyMask>, <checkModifiers> returns false if any key is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_NONE = 0; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_SHIFT │ │ │ │ │ + * If set as the <keyMask>, <checkModifiers> returns false if Shift is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_SHIFT = 1; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_CTRL │ │ │ │ │ + * If set as the <keyMask>, <checkModifiers> returns false if Ctrl is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_CTRL = 2; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_ALT │ │ │ │ │ + * If set as the <keyMask>, <checkModifiers> returns false if Alt is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_ALT = 4; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Handler.MOD_META │ │ │ │ │ + * If set as the <keyMask>, <checkModifiers> returns false if Cmd is down. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.MOD_META = 8; │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/TMS.js │ │ │ │ │ + OpenLayers/Spherical.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/SingleFile.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * Namespace: Spherical │ │ │ │ │ + * The OpenLayers.Spherical namespace includes utility functions for │ │ │ │ │ + * calculations on the basis of a spherical earth (ignoring ellipsoidal │ │ │ │ │ + * effects), which is accurate enough for most purposes. │ │ │ │ │ + * │ │ │ │ │ + * Relevant links: │ │ │ │ │ + * * http://www.movable-type.co.uk/scripts/latlong.html │ │ │ │ │ + * * http://code.google.com/apis/maps/documentation/javascript/reference.html#spherical │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ +OpenLayers.Spherical = OpenLayers.Spherical || {}; │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Spherical.DEFAULT_RADIUS = 6378137; │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.TMS │ │ │ │ │ - * Create a layer for accessing tiles from services that conform with the │ │ │ │ │ - * Tile Map Service Specification │ │ │ │ │ - * (http://wiki.osgeo.org/wiki/Tile_Map_Service_Specification). │ │ │ │ │ + * APIFunction: computeDistanceBetween │ │ │ │ │ + * Computes the distance between two LonLats. │ │ │ │ │ * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var layer = new OpenLayers.Layer.TMS( │ │ │ │ │ - * "My Layer", // name for display in LayerSwitcher │ │ │ │ │ - * "http://tilecache.osgeo.org/wms-c/Basic.py/", // service endpoint │ │ │ │ │ - * {layername: "basic", type: "png"} // required properties │ │ │ │ │ - * ); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ + * Parameters: │ │ │ │ │ + * from - {<OpenLayers.LonLat>} or {Object} Starting point. A LonLat or │ │ │ │ │ + * a JavaScript literal with lon lat properties. │ │ │ │ │ + * to - {<OpenLayers.LonLat>} or {Object} Ending point. A LonLat or a │ │ │ │ │ + * JavaScript literal with lon lat properties. │ │ │ │ │ + * radius - {Float} The radius. Optional. Defaults to 6378137 meters. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The distance in meters. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.TMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ +OpenLayers.Spherical.computeDistanceBetween = function(from, to, radius) { │ │ │ │ │ + var R = radius || OpenLayers.Spherical.DEFAULT_RADIUS; │ │ │ │ │ + var sinHalfDeltaLon = Math.sin(Math.PI * (to.lon - from.lon) / 360); │ │ │ │ │ + var sinHalfDeltaLat = Math.sin(Math.PI * (to.lat - from.lat) / 360); │ │ │ │ │ + var a = sinHalfDeltaLat * sinHalfDeltaLat + │ │ │ │ │ + sinHalfDeltaLon * sinHalfDeltaLon * Math.cos(Math.PI * from.lat / 180) * Math.cos(Math.PI * to.lat / 180); │ │ │ │ │ + return 2 * R * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a)); │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: serviceVersion │ │ │ │ │ - * {String} Service version for tile requests. Default is "1.0.0". │ │ │ │ │ - */ │ │ │ │ │ - serviceVersion: "1.0.0", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: layername │ │ │ │ │ - * {String} The identifier for the <TileMap> as advertised by the service. │ │ │ │ │ - * For example, if the service advertises a <TileMap> with │ │ │ │ │ - * 'href="http://tms.osgeo.org/1.0.0/vmap0"', the <layername> property │ │ │ │ │ - * would be set to "vmap0". │ │ │ │ │ - */ │ │ │ │ │ - layername: null, │ │ │ │ │ +/** │ │ │ │ │ + * APIFunction: computeHeading │ │ │ │ │ + * Computes the heading from one LonLat to another LonLat. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * from - {<OpenLayers.LonLat>} or {Object} Starting point. A LonLat or │ │ │ │ │ + * a JavaScript literal with lon lat properties. │ │ │ │ │ + * to - {<OpenLayers.LonLat>} or {Object} Ending point. A LonLat or a │ │ │ │ │ + * JavaScript literal with lon lat properties. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} The heading in degrees. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Spherical.computeHeading = function(from, to) { │ │ │ │ │ + var y = Math.sin(Math.PI * (from.lon - to.lon) / 180) * Math.cos(Math.PI * to.lat / 180); │ │ │ │ │ + var x = Math.cos(Math.PI * from.lat / 180) * Math.sin(Math.PI * to.lat / 180) - │ │ │ │ │ + Math.sin(Math.PI * from.lat / 180) * Math.cos(Math.PI * to.lat / 180) * Math.cos(Math.PI * (from.lon - to.lon) / 180); │ │ │ │ │ + return 180 * Math.atan2(y, x) / Math.PI; │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/hsb.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: type │ │ │ │ │ - * {String} The format extension corresponding to the requested tile image │ │ │ │ │ - * type. This is advertised in a <TileFormat> element as the │ │ │ │ │ - * "extension" attribute. For example, if the service advertises a │ │ │ │ │ - * <TileMap> with <TileFormat width="256" height="256" mime-type="image/jpeg" extension="jpg" />, │ │ │ │ │ - * the <type> property would be set to "jpg". │ │ │ │ │ - */ │ │ │ │ │ - type: null, │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Michawiki │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} Make this layer a base layer. Default is true. Set false to │ │ │ │ │ - * use the layer as an overlay. │ │ │ │ │ - */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: tileOrigin │ │ │ │ │ - * {<OpenLayers.LonLat>} Optional origin for aligning the grid of tiles. │ │ │ │ │ - * If provided, requests for tiles at all resolutions will be aligned │ │ │ │ │ - * with this location (no tiles shall overlap this location). If │ │ │ │ │ - * not provided, the grid of tiles will be aligned with the bottom-left │ │ │ │ │ - * corner of the map's <maxExtent>. Default is ``null``. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var layer = new OpenLayers.Layer.TMS( │ │ │ │ │ - * "My Layer", │ │ │ │ │ - * "http://tilecache.osgeo.org/wms-c/Basic.py/", │ │ │ │ │ - * { │ │ │ │ │ - * layername: "basic", │ │ │ │ │ - * type: "png", │ │ │ │ │ - * // set if different than the bottom left of map.maxExtent │ │ │ │ │ - * tileOrigin: new OpenLayers.LonLat(-180, -90) │ │ │ │ │ - * } │ │ │ │ │ - * ); │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ - tileOrigin: null, │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["hsb"] │ │ │ │ │ + * Dictionary for Hornjoserbsce. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["hsb"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: serverResolutions │ │ │ │ │ - * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ - * property if the map resolutions differ from the server. This │ │ │ │ │ - * property serves two purposes. (a) <serverResolutions> can include │ │ │ │ │ - * resolutions that the server supports and that you don't want to │ │ │ │ │ - * provide with this layer; you can also look at <zoomOffset>, which is │ │ │ │ │ - * an alternative to <serverResolutions> for that specific purpose. │ │ │ │ │ - * (b) The map can work with resolutions that aren't supported by │ │ │ │ │ - * the server, i.e. that aren't in <serverResolutions>. When the │ │ │ │ │ - * map is displayed in such a resolution data for the closest │ │ │ │ │ - * server-supported resolution is loaded and the layer div is │ │ │ │ │ - * stretched as necessary. │ │ │ │ │ - */ │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ + 'unhandledRequest': "Wotmołwa njewobdźěłaneho naprašowanja ${statusText}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomOffset │ │ │ │ │ - * {Number} If your cache has more zoom levels than you want to provide │ │ │ │ │ - * access to with this layer, supply a zoomOffset. This zoom offset │ │ │ │ │ - * is added to the current map zoom level to determine the level │ │ │ │ │ - * for a requested tile. For example, if you supply a zoomOffset │ │ │ │ │ - * of 3, when the map is at the zoom 0, tiles will be requested from │ │ │ │ │ - * level 3 of your cache. Default is 0 (assumes cache level and map │ │ │ │ │ - * zoom are equivalent). Using <zoomOffset> is an alternative to │ │ │ │ │ - * setting <serverResolutions> if you only want to expose a subset │ │ │ │ │ - * of the server resolutions. │ │ │ │ │ - */ │ │ │ │ │ - zoomOffset: 0, │ │ │ │ │ + 'Permalink': "Trajny wotkaz", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.TMS │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} Title to be displayed in a <OpenLayers.Control.LayerSwitcher> │ │ │ │ │ - * url - {String} Service endpoint (without the version number). E.g. │ │ │ │ │ - * "http://tms.osgeo.org/". │ │ │ │ │ - * options - {Object} Additional properties to be set on the layer. The │ │ │ │ │ - * <layername> and <type> properties must be set here. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - newArguments.push(name, url, {}, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - }, │ │ │ │ │ + 'Overlays': "Naworštowanja", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Create a complete copy of this layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} Should only be provided by subclasses that call this │ │ │ │ │ - * method. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.TMS>} An exact clone of this <OpenLayers.Layer.TMS> │ │ │ │ │ - */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ + 'Base Layer': "Zakładna runina", │ │ │ │ │ │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.TMS(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ + 'noFID': "Funkcija, za kotruž FID njeje, njeda so aktualizować.", │ │ │ │ │ │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + 'browserNotSupported': "Twój wobhladowak wektorowe rysowanje njepodpěruje. Tuchwilu podpěrowane rysowaki su:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ + 'minZoomLevelError': "Kajkosć minZoomLevel je jenož za wužiwanje z worštami myslena, kotrež wot FixedZoomLevels pochadźeja. Zo tuta woršta wfs za minZoomLevel přepruwuje, je relikt zańdźenosće. Njemóžemy wšak ju wotstronić, bjeztoho zo aplikacije, kotrež na OpenLayers bazěruja a snano tutu kajkosć wužiwaja, hižo njefunguja. Tohodla smy ju jako zestarjenu woznamjenili -- přepruwowanje za minZoomLevel budu so we wersiji 3.0 wotstronjeć. Prošu wužij město toho nastajenje min/max, kaž je tu wopisane: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ + 'commitSuccess': "WFS-Transakcija: WUSPĚŠNA ${response}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as │ │ │ │ │ - * parameters │ │ │ │ │ - */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((bounds.bottom - this.tileOrigin.lat) / (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getServerZoom(); │ │ │ │ │ - var path = this.serviceVersion + "/" + this.layername + "/" + z + "/" + x + "/" + y + "." + this.type; │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(path, url); │ │ │ │ │ - } │ │ │ │ │ - return url + path; │ │ │ │ │ - }, │ │ │ │ │ + 'commitFailed': "WFS-Transakcija: NJEPORADŹENA ${response}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * When the layer is added to a map, then we can fetch our origin │ │ │ │ │ - * (if we don't have one.) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (!this.tileOrigin) { │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.map.maxExtent.left, │ │ │ │ │ - this.map.maxExtent.bottom); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'googleWarning': "Woršta Google njemóžeše so korektnje začitać.\x3cbr\x3e\x3cbr\x3eZo by tutu zdźělenku wotbył, wubjer nowy BaseLayer z wuběra worštow horjeka naprawo.\x3cbr\x3e\x3cbr\x3eNajskerje so to stawa, dokelž skript biblioteki Google Maps pak njebu zapřijaty pak njewobsahuje korektny kluč API za twoje sydło.\x3cbr\x3e\x3cbr\x3eWuwiwarjo: Za pomoc ke korektnemu fungowanju worštow\n\x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3etu kliknyć\x3c/a\x3e", │ │ │ │ │ + │ │ │ │ │ + 'getLayerWarning': "Woršta ${layerType} njemóžeše so korektnje začitać.\x3cbr\x3e\x3cbr\x3eZo by tutu zdźělenku wotbył, wubjer nowy BaseLayer z wuběra worštow horjeka naprawo.\x3cbr\x3e\x3cbr\x3eNajskerje so to stawa, dokelž skript biblioteki ${layerLib} njebu korektnje zapřijaty.\x3cbr\x3e\x3cbr\x3eWuwiwarjo: Za pomoc ke korektnemu fungowanju worštow\n\x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3etu kliknyć\x3c/a\x3e", │ │ │ │ │ + │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Měritko = 1 : ${scaleDenom}", │ │ │ │ │ + │ │ │ │ │ + 'W': "Z", │ │ │ │ │ + │ │ │ │ │ + 'E': "W", │ │ │ │ │ + │ │ │ │ │ + 'N': "S", │ │ │ │ │ + │ │ │ │ │ + 'S': "J", │ │ │ │ │ + │ │ │ │ │ + 'reprojectDeprecated': "Wužiwaš opciju \"reproject\" wořšty ${layerName}. Tuta opcija je zestarjena: jeje wužiwanje bě myslene, zo by zwobraznjenje datow nad komercielnymi bazowymi kartami podpěrało, ale funkcionalnosć měła so nětko z pomocu Sperical Mercator docpěć. Dalše informacije steja na http://trac.openlayers.org/wiki/SphericalMercator k dispoziciji.", │ │ │ │ │ + │ │ │ │ │ + 'methodDeprecated': "Tuta metoda je so njeschwaliła a budźe so w 3.0 wotstronjeć. Prošu wužij ${newMethod} město toho." │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.TMS" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/XYZ.js │ │ │ │ │ + OpenLayers/Lang/nn.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Harald Khan │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.XYZ │ │ │ │ │ - * The XYZ class is designed to make it easier for people who have tiles │ │ │ │ │ - * arranged by a standard XYZ grid. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["nn"] │ │ │ │ │ + * Dictionary for ‪Norsk (nynorsk)‬. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.XYZ = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * Default is true, as this is designed to be a base tile source. │ │ │ │ │ - */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ +OpenLayers.Lang["nn"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: sphericalMercator │ │ │ │ │ - * Whether the tile extents should be set to the defaults for │ │ │ │ │ - * spherical mercator. Useful for things like OpenStreetMap. │ │ │ │ │ - * Default is false, except for the OSM subclass. │ │ │ │ │ - */ │ │ │ │ │ - sphericalMercator: false, │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Skala = 1 : ${scaleDenom}" │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomOffset │ │ │ │ │ - * {Number} If your cache has more zoom levels than you want to provide │ │ │ │ │ - * access to with this layer, supply a zoomOffset. This zoom offset │ │ │ │ │ - * is added to the current map zoom level to determine the level │ │ │ │ │ - * for a requested tile. For example, if you supply a zoomOffset │ │ │ │ │ - * of 3, when the map is at the zoom 0, tiles will be requested from │ │ │ │ │ - * level 3 of your cache. Default is 0 (assumes cache level and map │ │ │ │ │ - * zoom are equivalent). Using <zoomOffset> is an alternative to │ │ │ │ │ - * setting <serverResolutions> if you only want to expose a subset │ │ │ │ │ - * of the server resolutions. │ │ │ │ │ - */ │ │ │ │ │ - zoomOffset: 0, │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/ca.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: serverResolutions │ │ │ │ │ - * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ - * property if the map resolutions differ from the server. This │ │ │ │ │ - * property serves two purposes. (a) <serverResolutions> can include │ │ │ │ │ - * resolutions that the server supports and that you don't want to │ │ │ │ │ - * provide with this layer; you can also look at <zoomOffset>, which is │ │ │ │ │ - * an alternative to <serverResolutions> for that specific purpose. │ │ │ │ │ - * (b) The map can work with resolutions that aren't supported by │ │ │ │ │ - * the server, i.e. that aren't in <serverResolutions>. When the │ │ │ │ │ - * map is displayed in such a resolution data for the closest │ │ │ │ │ - * server-supported resolution is loaded and the layer div is │ │ │ │ │ - * stretched as necessary. │ │ │ │ │ - */ │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.XYZ │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * url - {String} │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - if (options && options.sphericalMercator || this.sphericalMercator) { │ │ │ │ │ - options = OpenLayers.Util.extend({ │ │ │ │ │ - projection: "EPSG:900913", │ │ │ │ │ - numZoomLevels: 19 │ │ │ │ │ - }, options); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [ │ │ │ │ │ - name || this.name, url || this.url, {}, │ │ │ │ │ - options │ │ │ │ │ - ]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} Is this ever used? │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.XYZ>} An exact clone of this OpenLayers.Layer.XYZ │ │ │ │ │ - */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["ca"] │ │ │ │ │ + * Dictionary for Catalan, UTF8 encoding. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang.ca = { │ │ │ │ │ │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.XYZ(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ + 'unhandledRequest': "Resposta a petició no gestionada ${statusText}", │ │ │ │ │ │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + 'Permalink': "Enllaç permanent", │ │ │ │ │ │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ + 'Overlays': "Capes addicionals", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as │ │ │ │ │ - * parameters │ │ │ │ │ - */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var xyz = this.getXYZ(bounds); │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - var s = '' + xyz.x + xyz.y + xyz.z; │ │ │ │ │ - url = this.selectUrl(s, url); │ │ │ │ │ - } │ │ │ │ │ + 'Base Layer': "Capa Base", │ │ │ │ │ │ │ │ │ │ - return OpenLayers.String.format(url, xyz); │ │ │ │ │ - }, │ │ │ │ │ + 'noFID': "No es pot actualitzar un element per al que no existeix FID.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getXYZ │ │ │ │ │ - * Calculates x, y and z for the given bounds. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} - an object with x, y and z properties. │ │ │ │ │ - */ │ │ │ │ │ - getXYZ: function(bounds) { │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.maxExtent.left) / │ │ │ │ │ - (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((this.maxExtent.top - bounds.top) / │ │ │ │ │ - (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getServerZoom(); │ │ │ │ │ + 'browserNotSupported': "El seu navegador no suporta renderització vectorial. Els renderitzadors suportats actualment són:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - var limit = Math.pow(2, z); │ │ │ │ │ - x = ((x % limit) + limit) % limit; │ │ │ │ │ - } │ │ │ │ │ + // console message │ │ │ │ │ + 'minZoomLevelError': "La propietat minZoomLevel s'ha d'utilitzar només " + │ │ │ │ │ + "amb les capes que tenen FixedZoomLevels. El fet que " + │ │ │ │ │ + "una capa wfs comprovi minZoomLevel és una relíquia del " + │ │ │ │ │ + "passat. No podem, però, eliminar-la sense trencar " + │ │ │ │ │ + "les aplicacions d'OpenLayers que en puguin dependre. " + │ │ │ │ │ + "Així doncs estem fent-la obsoleta -- la comprovació " + │ │ │ │ │ + "minZoomLevel s'eliminarà a la versió 3.0. Feu servir " + │ │ │ │ │ + "els paràmetres min/max resolution en substitució, tal com es descriu aquí: " + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - return { │ │ │ │ │ - 'x': x, │ │ │ │ │ - 'y': y, │ │ │ │ │ - 'z': z │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ + 'commitSuccess': "Transacció WFS: CORRECTA ${response}", │ │ │ │ │ │ │ │ │ │ - /* APIMethod: setMap │ │ │ │ │ - * When the layer is added to a map, then we can fetch our origin │ │ │ │ │ - * (if we don't have one.) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (!this.tileOrigin) { │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, │ │ │ │ │ - this.maxExtent.bottom); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'commitFailed': "Transacció WFS: HA FALLAT ${response}", │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.XYZ" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/ArcGISCache.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + 'googleWarning': "La capa Google no s'ha pogut carregar correctament.<br><br>" + │ │ │ │ │ + "Per evitar aquest missatge, seleccioneu una nova Capa Base " + │ │ │ │ │ + "al gestor de capes de la cantonada superior dreta.<br><br>" + │ │ │ │ │ + "Probablement això és degut a que l'script de la biblioteca de " + │ │ │ │ │ + "Google Maps no ha estat inclòs a la vostra pàgina, o no " + │ │ │ │ │ + "conté la clau de l'API correcta per a la vostra adreça.<br><br>" + │ │ │ │ │ + "Desenvolupadors: Per obtenir consells sobre com fer anar això, " + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ + "target='_blank'>féu clic aquí</a>", │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + 'getLayerWarning': "Per evitar aquest missatge, seleccioneu una nova Capa Base " + │ │ │ │ │ + "al gestor de capes de la cantonada superior dreta.<br><br>" + │ │ │ │ │ + "Probablement això és degut a que l'script de la biblioteca " + │ │ │ │ │ + "${layerLib} " + │ │ │ │ │ + "no ha estat inclòs a la vostra pàgina.<br><br>" + │ │ │ │ │ + "Desenvolupadors: Per obtenir consells sobre com fer anar això, " + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ + "target='_blank'>féu clic aquí</a>", │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ - */ │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Escala = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.ArcGISCache │ │ │ │ │ - * Layer for accessing cached map tiles from an ArcGIS Server style mapcache. │ │ │ │ │ - * Tile must already be cached for this layer to access it. This does not require │ │ │ │ │ - * ArcGIS Server itself. │ │ │ │ │ - * │ │ │ │ │ - * A few attempts have been made at this kind of layer before. See │ │ │ │ │ - * http://trac.osgeo.org/openlayers/ticket/1967 │ │ │ │ │ - * and │ │ │ │ │ - * http://trac.osgeo.org/openlayers/browser/sandbox/tschaub/arcgiscache/lib/OpenLayers/Layer/ArcGISCache.js │ │ │ │ │ - * │ │ │ │ │ - * Typically the problem encountered is that the tiles seem to "jump around". │ │ │ │ │ - * This is due to the fact that the actual max extent for the tiles on AGS layers │ │ │ │ │ - * changes at each zoom level due to the way these caches are constructed. │ │ │ │ │ - * We have attempted to use the resolutions, tile size, and tile origin │ │ │ │ │ - * from the cache meta data to make the appropriate changes to the max extent │ │ │ │ │ - * of the tile to compensate for this behavior. This must be done as zoom levels change │ │ │ │ │ - * and before tiles are requested, which is why methods from base classes are overridden. │ │ │ │ │ - * │ │ │ │ │ - * For reference, you can access mapcache meta data in two ways. For accessing a │ │ │ │ │ - * mapcache through ArcGIS Server, you can simply go to the landing page for the │ │ │ │ │ - * layer. (ie. http://services.arcgisonline.com/ArcGIS/rest/services/World_Street_Map/MapServer) │ │ │ │ │ - * For accessing it directly through HTTP, there should always be a conf.xml file │ │ │ │ │ - * in the root directory. │ │ │ │ │ - * (ie. http://serverx.esri.com/arcgiscache/DG_County_roads_yesA_backgroundDark/Layers/conf.xml) │ │ │ │ │ - * │ │ │ │ │ - *Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.XYZ> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.ArcGISCache = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ + //labels for the graticule control │ │ │ │ │ + 'W': 'O', │ │ │ │ │ + 'E': 'E', │ │ │ │ │ + 'N': 'N', │ │ │ │ │ + 'S': 'S', │ │ │ │ │ + 'Graticule': 'Retícula', │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String | Array} The base URL for the layer cache. You can also │ │ │ │ │ - * provide a list of URL strings for the layer if your cache is │ │ │ │ │ - * available from multiple origins. This must be set before the layer │ │ │ │ │ - * is drawn. │ │ │ │ │ - */ │ │ │ │ │ - url: null, │ │ │ │ │ + // console message │ │ │ │ │ + 'reprojectDeprecated': "Esteu fent servir l'opció 'reproject' a la capa " + │ │ │ │ │ + "${layerName}. Aquesta opció és obsoleta: el seu ús fou concebut " + │ │ │ │ │ + "per suportar la visualització de dades sobre mapes base comercials, " + │ │ │ │ │ + "però ara aquesta funcionalitat s'hauria d'assolir mitjançant el suport " + │ │ │ │ │ + "de la projecció Spherical Mercator. Més informació disponible a " + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: tileOrigin │ │ │ │ │ - * {<OpenLayers.LonLat>} The location of the tile origin for the cache. │ │ │ │ │ - * An ArcGIS cache has it's origin at the upper-left (lowest x value │ │ │ │ │ - * and highest y value of the coordinate system). The units for the │ │ │ │ │ - * tile origin should be the same as the units for the cached data. │ │ │ │ │ - */ │ │ │ │ │ - tileOrigin: null, │ │ │ │ │ + // console message │ │ │ │ │ + 'methodDeprecated': "Aquest mètode és obsolet i s'eliminarà a la versió 3.0. " + │ │ │ │ │ + "Si us plau feu servir em mètode alternatiu ${newMethod}.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: tileSize │ │ │ │ │ - * {<OpenLayers.Size>} This size of each tile. Defaults to 256 by 256 pixels. │ │ │ │ │ - */ │ │ │ │ │ - tileSize: new OpenLayers.Size(256, 256), │ │ │ │ │ + // **** end **** │ │ │ │ │ + 'end': '' │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: useAGS │ │ │ │ │ - * {Boolean} Indicates if we are going to be accessing the ArcGIS Server (AGS) │ │ │ │ │ - * cache via an AGS MapServer or directly through HTTP. When accessing via │ │ │ │ │ - * AGS the path structure uses a standard z/y/x structure. But AGS actually │ │ │ │ │ - * stores the tile images on disk using a hex based folder structure that looks │ │ │ │ │ - * like "http://example.com/mylayer/L00/R00000000/C00000000.png". Learn more │ │ │ │ │ - * about this here: │ │ │ │ │ - * http://blogs.esri.com/Support/blogs/mappingcenter/archive/2010/08/20/Checking-Your-Local-Cache-Folders.aspx │ │ │ │ │ - * Defaults to true; │ │ │ │ │ - */ │ │ │ │ │ - useArcGISServer: true, │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/oc.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: type │ │ │ │ │ - * {String} Image type for the layer. This becomes the filename extension │ │ │ │ │ - * in tile requests. Default is "png" (generating a url like │ │ │ │ │ - * "http://example.com/mylayer/L00/R00000000/C00000000.png"). │ │ │ │ │ - */ │ │ │ │ │ - type: 'png', │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Cedric31 │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: useScales │ │ │ │ │ - * {Boolean} Optional override to indicate that the layer should use 'scale' information │ │ │ │ │ - * returned from the server capabilities object instead of 'resolution' information. │ │ │ │ │ - * This can be important if your tile server uses an unusual DPI for the tiles. │ │ │ │ │ - */ │ │ │ │ │ - useScales: false, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: overrideDPI │ │ │ │ │ - * {Boolean} Optional override to change the OpenLayers.DOTS_PER_INCH setting based │ │ │ │ │ - * on the tile information in the server capabilities object. This can be useful │ │ │ │ │ - * if your server has a non-standard DPI setting on its tiles, and you're only using │ │ │ │ │ - * tiles with that DPI. This value is used while OpenLayers is calculating resolution │ │ │ │ │ - * using scales, and is not necessary if you have resolution information. (This is │ │ │ │ │ - * typically the case) Regardless, this setting can be useful, but is dangerous │ │ │ │ │ - * because it will impact other layers while calculating resolution. Only use this │ │ │ │ │ - * if you know what you are doing. (See OpenLayers.Util.getResolutionFromScale) │ │ │ │ │ - */ │ │ │ │ │ - overrideDPI: false, │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["oc"] │ │ │ │ │ + * Dictionary for Occitan. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["oc"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.ArcGISCache │ │ │ │ │ - * Creates a new instance of this class │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * url - {String} │ │ │ │ │ - * options - {Object} extra layer options │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ + 'unhandledRequest': "Requèsta pas gerida, retorna ${statusText}", │ │ │ │ │ │ │ │ │ │ - if (this.resolutions) { │ │ │ │ │ - this.serverResolutions = this.resolutions; │ │ │ │ │ - this.maxExtent = this.getMaxExtentForResolution(this.resolutions[0]); │ │ │ │ │ - } │ │ │ │ │ + 'Permalink': "Permaligam", │ │ │ │ │ │ │ │ │ │ - // this block steps through translating the values from the server layer JSON │ │ │ │ │ - // capabilities object into values that we can use. This is also a helpful │ │ │ │ │ - // reference when configuring this layer directly. │ │ │ │ │ - if (this.layerInfo) { │ │ │ │ │ - // alias the object │ │ │ │ │ - var info = this.layerInfo; │ │ │ │ │ + 'Overlays': "Calques", │ │ │ │ │ │ │ │ │ │ - // build our extents │ │ │ │ │ - var startingTileExtent = new OpenLayers.Bounds( │ │ │ │ │ - info.fullExtent.xmin, │ │ │ │ │ - info.fullExtent.ymin, │ │ │ │ │ - info.fullExtent.xmax, │ │ │ │ │ - info.fullExtent.ymax │ │ │ │ │ - ); │ │ │ │ │ + 'Base Layer': "Calc de basa", │ │ │ │ │ │ │ │ │ │ - // set our projection based on the given spatial reference. │ │ │ │ │ - // esri uses slightly different IDs, so this may not be comprehensive │ │ │ │ │ - this.projection = 'EPSG:' + info.spatialReference.wkid; │ │ │ │ │ - this.sphericalMercator = (info.spatialReference.wkid == 102100); │ │ │ │ │ + 'noFID': "Impossible de metre a jorn un objècte sens identificant (fid).", │ │ │ │ │ │ │ │ │ │ - // convert esri units into openlayers units (basic feet or meters only) │ │ │ │ │ - this.units = (info.units == "esriFeet") ? 'ft' : 'm'; │ │ │ │ │ + 'browserNotSupported': "Vòstre navegidor supòrta pas lo rendut vectorial. Los renderers actualament suportats son : \n${renderers}", │ │ │ │ │ │ │ │ │ │ - // optional extended section based on whether or not the server returned │ │ │ │ │ - // specific tile information │ │ │ │ │ - if (!!info.tileInfo) { │ │ │ │ │ - // either set the tiles based on rows/columns, or specific width/height │ │ │ │ │ - this.tileSize = new OpenLayers.Size( │ │ │ │ │ - info.tileInfo.width || info.tileInfo.cols, │ │ │ │ │ - info.tileInfo.height || info.tileInfo.rows │ │ │ │ │ - ); │ │ │ │ │ + 'minZoomLevelError': "La proprietat minZoomLevel deu èsser utilizada solament per de jaces FixedZoomLevels-descendent. Lo fach qu\'aqueste jaç WFS verifique la preséncia de minZoomLevel es una relica del passat. Çaquelà, la podèm suprimir sens copar d\'aplicacions que ne poirián dependre. Es per aquò que la depreciam -- la verificacion del minZoomLevel serà suprimida en version 3.0. A la plaça, mercés d\'utilizar los paramètres de resolucions min/max tal coma descrich sus : http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - // this must be set when manually configuring this layer │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat( │ │ │ │ │ - info.tileInfo.origin.x, │ │ │ │ │ - info.tileInfo.origin.y │ │ │ │ │ - ); │ │ │ │ │ + 'commitSuccess': "Transaccion WFS : SUCCES ${response}", │ │ │ │ │ │ │ │ │ │ - var upperLeft = new OpenLayers.Geometry.Point( │ │ │ │ │ - startingTileExtent.left, │ │ │ │ │ - startingTileExtent.top │ │ │ │ │ - ); │ │ │ │ │ + 'commitFailed': "Transaccion WFS : FRACAS ${response}", │ │ │ │ │ │ │ │ │ │ - var bottomRight = new OpenLayers.Geometry.Point( │ │ │ │ │ - startingTileExtent.right, │ │ │ │ │ - startingTileExtent.bottom │ │ │ │ │ - ); │ │ │ │ │ + 'googleWarning': "Lo jaç Google es pas estat en mesura de se cargar corrèctament.\x3cbr\x3e\x3cbr\x3ePer suprimir aqueste messatge, causissètz una BaseLayer novèla dins lo selector de jaç en naut a drecha.\x3cbr\x3e\x3cbr\x3eAquò es possiblament causat par la non-inclusion de la librariá Google Maps, o alara perque que la clau de l\'API correspond pas a vòstre site.\x3cbr\x3e\x3cbr\x3eDesvolopaires : per saber cossí corregir aquò, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eclicatz aicí\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - if (this.useScales) { │ │ │ │ │ - this.scales = []; │ │ │ │ │ - } else { │ │ │ │ │ - this.resolutions = []; │ │ │ │ │ - } │ │ │ │ │ + 'getLayerWarning': "Lo jaç ${layerType} es pas en mesura de se cargar corrèctament.\x3cbr\x3e\x3cbr\x3ePer suprimir aqueste messatge, causissètz una BaseLayer novèla dins lo selector de jaç en naut a drecha.\x3cbr\x3e\x3cbr\x3eAquò es possiblament causat per la non-inclusion de la librariá ${layerLib}.\x3cbr\x3e\x3cbr\x3eDesvolopaires : per saber cossí corregir aquí, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eclicatz aicí\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - this.lods = []; │ │ │ │ │ - for (var key in info.tileInfo.lods) { │ │ │ │ │ - if (info.tileInfo.lods.hasOwnProperty(key)) { │ │ │ │ │ - var lod = info.tileInfo.lods[key]; │ │ │ │ │ - if (this.useScales) { │ │ │ │ │ - this.scales.push(lod.scale); │ │ │ │ │ - } else { │ │ │ │ │ - this.resolutions.push(lod.resolution); │ │ │ │ │ - } │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Escala ~ 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - var start = this.getContainingTileCoords(upperLeft, lod.resolution); │ │ │ │ │ - lod.startTileCol = start.x; │ │ │ │ │ - lod.startTileRow = start.y; │ │ │ │ │ + 'W': "O", │ │ │ │ │ │ │ │ │ │ - var end = this.getContainingTileCoords(bottomRight, lod.resolution); │ │ │ │ │ - lod.endTileCol = end.x; │ │ │ │ │ - lod.endTileRow = end.y; │ │ │ │ │ - this.lods.push(lod); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + 'E': "È", │ │ │ │ │ │ │ │ │ │ - this.maxExtent = this.calculateMaxExtentWithLOD(this.lods[0]); │ │ │ │ │ - this.serverResolutions = this.resolutions; │ │ │ │ │ - if (this.overrideDPI && info.tileInfo.dpi) { │ │ │ │ │ - // see comment above for 'overrideDPI' │ │ │ │ │ - OpenLayers.DOTS_PER_INCH = info.tileInfo.dpi; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'N': "N", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getContainingTileCoords │ │ │ │ │ - * Calculates the x/y pixel corresponding to the position of the tile │ │ │ │ │ - * that contains the given point and for the for the given resolution. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ - * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Pixel>} The x/y pixel corresponding to the position │ │ │ │ │ - * of the upper left tile for the given resolution. │ │ │ │ │ - */ │ │ │ │ │ - getContainingTileCoords: function(point, res) { │ │ │ │ │ - return new OpenLayers.Pixel( │ │ │ │ │ - Math.max(Math.floor((point.x - this.tileOrigin.lon) / (this.tileSize.w * res)), 0), │ │ │ │ │ - Math.max(Math.floor((this.tileOrigin.lat - point.y) / (this.tileSize.h * res)), 0) │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ + 'S': "S", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateMaxExtentWithLOD │ │ │ │ │ - * Given a Level of Detail object from the server, this function │ │ │ │ │ - * calculates the actual max extent │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * lod - {Object} a Level of Detail Object from the server capabilities object │ │ │ │ │ - representing a particular zoom level │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} The actual extent of the tiles for the given zoom level │ │ │ │ │ - */ │ │ │ │ │ - calculateMaxExtentWithLOD: function(lod) { │ │ │ │ │ - // the max extent we're provided with just overlaps some tiles │ │ │ │ │ - // our real extent is the bounds of all the tiles we touch │ │ │ │ │ + 'reprojectDeprecated': "Utilizatz l\'opcion \'reproject\' sul jaç ${layerName}. Aquesta opcion es despreciada : Son usatge permetiá d\'afichar de donadas al dessús de jaces raster comercials. Aquesta foncionalitat ara es suportada en utilizant lo supòrt de la projeccion Mercator Esferica. Mai d\'informacion es disponibla sus http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ │ │ │ │ │ - var numTileCols = (lod.endTileCol - lod.startTileCol) + 1; │ │ │ │ │ - var numTileRows = (lod.endTileRow - lod.startTileRow) + 1; │ │ │ │ │ + 'methodDeprecated': "Aqueste metòde es despreciada, e serà suprimida a la version 3.0. Mercés d\'utilizar ${newMethod} a la plaça." │ │ │ │ │ │ │ │ │ │ - var minX = this.tileOrigin.lon + (lod.startTileCol * this.tileSize.w * lod.resolution); │ │ │ │ │ - var maxX = minX + (numTileCols * this.tileSize.w * lod.resolution); │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/nl.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - var maxY = this.tileOrigin.lat - (lod.startTileRow * this.tileSize.h * lod.resolution); │ │ │ │ │ - var minY = maxY - (numTileRows * this.tileSize.h * lod.resolution); │ │ │ │ │ - return new OpenLayers.Bounds(minX, minY, maxX, maxY); │ │ │ │ │ - }, │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Siebrand │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateMaxExtentWithExtent │ │ │ │ │ - * Given a 'suggested' max extent from the server, this function uses │ │ │ │ │ - * information about the actual tile sizes to determine the actual │ │ │ │ │ - * extent of the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} The 'suggested' extent for the layer │ │ │ │ │ - * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} The actual extent of the tiles for the given zoom level │ │ │ │ │ - */ │ │ │ │ │ - calculateMaxExtentWithExtent: function(extent, res) { │ │ │ │ │ - var upperLeft = new OpenLayers.Geometry.Point(extent.left, extent.top); │ │ │ │ │ - var bottomRight = new OpenLayers.Geometry.Point(extent.right, extent.bottom); │ │ │ │ │ - var start = this.getContainingTileCoords(upperLeft, res); │ │ │ │ │ - var end = this.getContainingTileCoords(bottomRight, res); │ │ │ │ │ - var lod = { │ │ │ │ │ - resolution: res, │ │ │ │ │ - startTileCol: start.x, │ │ │ │ │ - startTileRow: start.y, │ │ │ │ │ - endTileCol: end.x, │ │ │ │ │ - endTileRow: end.y │ │ │ │ │ - }; │ │ │ │ │ - return this.calculateMaxExtentWithLOD(lod); │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getUpperLeftTileCoord │ │ │ │ │ - * Calculates the x/y pixel corresponding to the position │ │ │ │ │ - * of the upper left tile for the given resolution. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Pixel>} The x/y pixel corresponding to the position │ │ │ │ │ - * of the upper left tile for the given resolution. │ │ │ │ │ - */ │ │ │ │ │ - getUpperLeftTileCoord: function(res) { │ │ │ │ │ - var upperLeft = new OpenLayers.Geometry.Point( │ │ │ │ │ - this.maxExtent.left, │ │ │ │ │ - this.maxExtent.top); │ │ │ │ │ - return this.getContainingTileCoords(upperLeft, res); │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["nl"] │ │ │ │ │ + * Dictionary for Nederlands. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["nl"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getLowerRightTileCoord │ │ │ │ │ - * Calculates the x/y pixel corresponding to the position │ │ │ │ │ - * of the lower right tile for the given resolution. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Pixel>} The x/y pixel corresponding to the position │ │ │ │ │ - * of the lower right tile for the given resolution. │ │ │ │ │ - */ │ │ │ │ │ - getLowerRightTileCoord: function(res) { │ │ │ │ │ - var bottomRight = new OpenLayers.Geometry.Point( │ │ │ │ │ - this.maxExtent.right, │ │ │ │ │ - this.maxExtent.bottom); │ │ │ │ │ - return this.getContainingTileCoords(bottomRight, res); │ │ │ │ │ - }, │ │ │ │ │ + 'unhandledRequest': "Het verzoek is niet afgehandeld met de volgende melding: ${statusText}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getMaxExtentForResolution │ │ │ │ │ - * Since the max extent of a set of tiles can change from zoom level │ │ │ │ │ - * to zoom level, we need to be able to calculate that max extent │ │ │ │ │ - * for a given resolution. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} The extent for this resolution │ │ │ │ │ - */ │ │ │ │ │ - getMaxExtentForResolution: function(res) { │ │ │ │ │ - var start = this.getUpperLeftTileCoord(res); │ │ │ │ │ - var end = this.getLowerRightTileCoord(res); │ │ │ │ │ + 'Permalink': "Permanente verwijzing", │ │ │ │ │ │ │ │ │ │ - var numTileCols = (end.x - start.x) + 1; │ │ │ │ │ - var numTileRows = (end.y - start.y) + 1; │ │ │ │ │ + 'Overlays': "Overlays", │ │ │ │ │ │ │ │ │ │ - var minX = this.tileOrigin.lon + (start.x * this.tileSize.w * res); │ │ │ │ │ - var maxX = minX + (numTileCols * this.tileSize.w * res); │ │ │ │ │ + 'Base Layer': "Achtergrondkaart", │ │ │ │ │ │ │ │ │ │ - var maxY = this.tileOrigin.lat - (start.y * this.tileSize.h * res); │ │ │ │ │ - var minY = maxY - (numTileRows * this.tileSize.h * res); │ │ │ │ │ - return new OpenLayers.Bounds(minX, minY, maxX, maxY); │ │ │ │ │ - }, │ │ │ │ │ + 'noFID': "Een optie die geen FID heeft kan niet bijgewerkt worden.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Returns an exact clone of this OpenLayers.Layer.ArcGISCache │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * [obj] - {Object} optional object to assign the cloned instance to. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.ArcGISCache>} clone of this instance │ │ │ │ │ - */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.ArcGISCache(this.name, this.url, this.options); │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ - }, │ │ │ │ │ + 'browserNotSupported': "Uw browser ondersteunt het weergeven van vectoren niet.\nMomenteel ondersteunde weergavemogelijkheden:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: initGriddedTiles │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - */ │ │ │ │ │ - initGriddedTiles: function(bounds) { │ │ │ │ │ - delete this._tileOrigin; │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initGriddedTiles.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + 'minZoomLevelError': "De eigenschap minZoomLevel is alleen bedoeld voor gebruik lagen met die afstammen van FixedZoomLevels-lagen.\nDat deze WFS-laag minZoomLevel controleert, is een overblijfsel uit het verleden.\nWe kunnen deze controle echter niet verwijderen zonder op OL gebaseerde applicaties die hervan afhankelijk zijn stuk te maken.\nDaarom heeft deze functionaliteit de eigenschap \'deprecated\' gekregen - de minZoomLevel wordt verwijderd in versie 3.0.\nGebruik in plaats van deze functie de mogelijkheid om min/max voor resolutie in te stellen zoals op de volgende pagina wordt beschreven:\nhttp://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getMaxExtent │ │ │ │ │ - * Get this layer's maximum extent. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} │ │ │ │ │ - */ │ │ │ │ │ - getMaxExtent: function() { │ │ │ │ │ - var resolution = this.map.getResolution(); │ │ │ │ │ - return this.maxExtent = this.getMaxExtentForResolution(resolution); │ │ │ │ │ - }, │ │ │ │ │ + 'commitSuccess': "WFS-transactie: succesvol ${response}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getTileOrigin │ │ │ │ │ - * Determine the origin for aligning the grid of tiles. │ │ │ │ │ - * The origin will be derived from the layer's <maxExtent> property. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.LonLat>} The tile origin. │ │ │ │ │ - */ │ │ │ │ │ - getTileOrigin: function() { │ │ │ │ │ - if (!this._tileOrigin) { │ │ │ │ │ - var extent = this.getMaxExtent(); │ │ │ │ │ - this._tileOrigin = new OpenLayers.LonLat(extent.left, extent.bottom); │ │ │ │ │ - } │ │ │ │ │ - return this._tileOrigin; │ │ │ │ │ - }, │ │ │ │ │ + 'commitFailed': "WFS-transactie: mislukt ${response}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * Determine the URL for a tile given the tile bounds. This is should support │ │ │ │ │ - * urls that access tiles through an ArcGIS Server MapServer or directly through │ │ │ │ │ - * the hex folder structure using HTTP. Just be sure to set the useArcGISServer │ │ │ │ │ - * property appropriately! This is basically the same as │ │ │ │ │ - * 'OpenLayers.Layer.TMS.getURL', but with the addition of hex addressing, │ │ │ │ │ - * and tile rounding. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The URL for a tile based on given bounds. │ │ │ │ │ - */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var res = this.getResolution(); │ │ │ │ │ + 'googleWarning': "De Google-Layer kon niet correct geladen worden.\x3cbr /\x3e\x3cbr /\x3e\nOm deze melding niet meer te krijgen, moet u een andere achtergrondkaart kiezen in de laagwisselaar in de rechterbovenhoek.\x3cbr /\x3e\x3cbr /\x3e\nDit komt waarschijnlijk doordat de bibliotheek ${layerLib} niet correct ingevoegd is.\x3cbr /\x3e\x3cbr /\x3e\nOntwikkelaars: \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eklik hier\x3c/a\x3e om dit werkend te krijgen.", │ │ │ │ │ │ │ │ │ │ - // tile center │ │ │ │ │ - var originTileX = (this.tileOrigin.lon + (res * this.tileSize.w / 2)); │ │ │ │ │ - var originTileY = (this.tileOrigin.lat - (res * this.tileSize.h / 2)); │ │ │ │ │ + 'getLayerWarning': "De laag ${layerType} kon niet goed geladen worden.\x3cbr /\x3e\x3cbr /\x3e\nOm deze melding niet meer te krijgen, moet u een andere achtergrondkaart kiezen in de laagwisselaar in de rechterbovenhoek.\x3cbr /\x3e\x3cbr /\x3e\nDit komt waarschijnlijk doordat de bibliotheek ${layerLib} niet correct is ingevoegd.\x3cbr /\x3e\x3cbr /\x3e\nOntwikkelaars: \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eklik hier\x3c/a\x3e om dit werkend te krijgen.", │ │ │ │ │ │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var point = { │ │ │ │ │ - x: center.lon, │ │ │ │ │ - y: center.lat │ │ │ │ │ - }; │ │ │ │ │ - var x = (Math.round(Math.abs((center.lon - originTileX) / (res * this.tileSize.w)))); │ │ │ │ │ - var y = (Math.round(Math.abs((originTileY - center.lat) / (res * this.tileSize.h)))); │ │ │ │ │ - var z = this.map.getZoom(); │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Schaal = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - // this prevents us from getting pink tiles (non-existant tiles) │ │ │ │ │ - if (this.lods) { │ │ │ │ │ - var lod = this.lods[this.map.getZoom()]; │ │ │ │ │ - if ((x < lod.startTileCol || x > lod.endTileCol) || │ │ │ │ │ - (y < lod.startTileRow || y > lod.endTileRow)) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - var start = this.getUpperLeftTileCoord(res); │ │ │ │ │ - var end = this.getLowerRightTileCoord(res); │ │ │ │ │ - if ((x < start.x || x >= end.x) || │ │ │ │ │ - (y < start.y || y >= end.y)) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + 'W': "W", │ │ │ │ │ │ │ │ │ │ - // Construct the url string │ │ │ │ │ - var url = this.url; │ │ │ │ │ - var s = '' + x + y + z; │ │ │ │ │ + 'E': "O", │ │ │ │ │ │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(s, url); │ │ │ │ │ - } │ │ │ │ │ + 'N': "N", │ │ │ │ │ │ │ │ │ │ - // Accessing tiles through ArcGIS Server uses a different path │ │ │ │ │ - // structure than direct access via the folder structure. │ │ │ │ │ - if (this.useArcGISServer) { │ │ │ │ │ - // AGS MapServers have pretty url access to tiles │ │ │ │ │ - url = url + '/tile/${z}/${y}/${x}'; │ │ │ │ │ - } else { │ │ │ │ │ - // The tile images are stored using hex values on disk. │ │ │ │ │ - x = 'C' + OpenLayers.Number.zeroPad(x, 8, 16); │ │ │ │ │ - y = 'R' + OpenLayers.Number.zeroPad(y, 8, 16); │ │ │ │ │ - z = 'L' + OpenLayers.Number.zeroPad(z, 2, 10); │ │ │ │ │ - url = url + '/${z}/${y}/${x}.' + this.type; │ │ │ │ │ - } │ │ │ │ │ + 'S': "Z", │ │ │ │ │ │ │ │ │ │ - // Write the values into our formatted url │ │ │ │ │ - url = OpenLayers.String.format(url, { │ │ │ │ │ - 'x': x, │ │ │ │ │ - 'y': y, │ │ │ │ │ - 'z': z │ │ │ │ │ - }); │ │ │ │ │ + 'reprojectDeprecated': "U gebruikt de optie \'reproject\' op de laag ${layerName}.\nDeze optie is vervallen: deze optie was ontwikkeld om gegevens over commerciële basiskaarten weer te geven, maar deze functionaliteit wordt nu bereikt door ondersteuning van Spherical Mercator.\nMeer informatie is beschikbaar op http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ │ │ │ │ │ - return OpenLayers.Util.urlAppend( │ │ │ │ │ - url, OpenLayers.Util.getParameterString(this.params) │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ + 'methodDeprecated': "Deze methode is verouderd en wordt verwijderd in versie 3.0.\nGebruik ${newMethod}." │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: 'OpenLayers.Layer.ArcGISCache' │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/ArcXML.js │ │ │ │ │ + OpenLayers/Lang/zh-TW.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/MultiPolygon.js │ │ │ │ │ - * @requires OpenLayers/Geometry/LinearRing.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.ArcXML │ │ │ │ │ - * Read/Write ArcXML. Create a new instance with the <OpenLayers.Format.ArcXML> │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * Namespace: OpenLayers.Lang["zh-TW"] │ │ │ │ │ + * Dictionary for Traditional Chinese. (Used Mainly in Taiwan) │ │ │ │ │ + * Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.ArcXML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Lang["zh-TW"] = { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: fontStyleKeys │ │ │ │ │ - * {Array} List of keys used in font styling. │ │ │ │ │ - */ │ │ │ │ │ - fontStyleKeys: [ │ │ │ │ │ - 'antialiasing', 'blockout', 'font', 'fontcolor', 'fontsize', 'fontstyle', │ │ │ │ │ - 'glowing', 'interval', 'outline', 'printmode', 'shadow', 'transparency' │ │ │ │ │ - ], │ │ │ │ │ + 'unhandledRequest': "未處理的請求,傳回值為 ${statusText}。", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: request │ │ │ │ │ - * A get_image request destined for an ArcIMS server. │ │ │ │ │ - */ │ │ │ │ │ - request: null, │ │ │ │ │ + 'Permalink': "永久連結", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: response │ │ │ │ │ - * A parsed response from an ArcIMS server. │ │ │ │ │ - */ │ │ │ │ │ - response: null, │ │ │ │ │ + 'Overlays': "額外圖層", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.ArcXML │ │ │ │ │ - * Create a new parser/writer for ArcXML. Create an instance of this class │ │ │ │ │ - * to begin authoring a request to an ArcIMS service. This is used │ │ │ │ │ - * primarily by the ArcIMS layer, but could be used to do other wild │ │ │ │ │ - * stuff, like geocoding. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.request = new OpenLayers.Format.ArcXML.Request(); │ │ │ │ │ - this.response = new OpenLayers.Format.ArcXML.Response(); │ │ │ │ │ + 'Base Layer': "基礎圖層", │ │ │ │ │ │ │ │ │ │ - if (options) { │ │ │ │ │ - if (options.requesttype == "feature") { │ │ │ │ │ - this.request.get_image = null; │ │ │ │ │ + 'noFID': "因為沒有 FID 所以無法更新 feature。", │ │ │ │ │ │ │ │ │ │ - var qry = this.request.get_feature.query; │ │ │ │ │ - this.addCoordSys(qry.featurecoordsys, options.featureCoordSys); │ │ │ │ │ - this.addCoordSys(qry.filtercoordsys, options.filterCoordSys); │ │ │ │ │ + 'browserNotSupported': "您的瀏覽器未支援向量渲染. 目前支援的渲染方式是:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - if (options.polygon) { │ │ │ │ │ - qry.isspatial = true; │ │ │ │ │ - qry.spatialfilter.polygon = options.polygon; │ │ │ │ │ - } else if (options.envelope) { │ │ │ │ │ - qry.isspatial = true; │ │ │ │ │ - qry.spatialfilter.envelope = { │ │ │ │ │ - minx: 0, │ │ │ │ │ - miny: 0, │ │ │ │ │ - maxx: 0, │ │ │ │ │ - maxy: 0 │ │ │ │ │ - }; │ │ │ │ │ - this.parseEnvelope(qry.spatialfilter.envelope, options.envelope); │ │ │ │ │ - } │ │ │ │ │ - } else if (options.requesttype == "image") { │ │ │ │ │ - this.request.get_feature = null; │ │ │ │ │ + // console message │ │ │ │ │ + 'minZoomLevelError': "minZoomLevel 屬性僅適合用在 " + │ │ │ │ │ + "FixedZoomLevels-descendent 類型的圖層. 這個" + │ │ │ │ │ + "wfs layer 的 minZoomLevel 是過去所遺留下來的," + │ │ │ │ │ + "然而我們不能移除它而不讓它將" + │ │ │ │ │ + "過去的程式相容性給破壞掉。" + │ │ │ │ │ + "因此我們將會迴避使用它 -- minZoomLevel " + │ │ │ │ │ + "會在3.0被移除,請改" + │ │ │ │ │ + "用在這邊描述的 min/max resolution 設定: " + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - var props = this.request.get_image.properties; │ │ │ │ │ - this.parseEnvelope(props.envelope, options.envelope); │ │ │ │ │ + 'commitSuccess': "WFS Transaction: 成功 ${response}", │ │ │ │ │ │ │ │ │ │ - this.addLayers(props.layerlist, options.layers); │ │ │ │ │ - this.addImageSize(props.imagesize, options.tileSize); │ │ │ │ │ - this.addCoordSys(props.featurecoordsys, options.featureCoordSys); │ │ │ │ │ - this.addCoordSys(props.filtercoordsys, options.filterCoordSys); │ │ │ │ │ - } else { │ │ │ │ │ - // if an arcxml object is being created with no request type, it is │ │ │ │ │ - // probably going to consume a response, so do not throw an error if │ │ │ │ │ - // the requesttype is not defined │ │ │ │ │ - this.request = null; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + 'commitFailed': "WFS Transaction: 失敗 ${response}", │ │ │ │ │ │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - }, │ │ │ │ │ + 'googleWarning': "The Google Layer 圖層無法被正確的載入。<br><br>" + │ │ │ │ │ + "要迴避這個訊息, 請在右上角的圖層改變器裡," + │ │ │ │ │ + "選一個新的基礎圖層。<br><br>" + │ │ │ │ │ + "很有可能是因為 Google Maps 的函式庫" + │ │ │ │ │ + "腳本沒有被正確的置入,或沒有包含 " + │ │ │ │ │ + "您網站上正確的 API key <br><br>" + │ │ │ │ │ + "開發者: 要幫助這個行為正確完成," + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ + "target='_blank'>請按這裡</a>", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseEnvelope │ │ │ │ │ - * Parse an array of coordinates into an ArcXML envelope structure. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * env - {Object} An envelope object that will contain the parsed coordinates. │ │ │ │ │ - * arr - {Array(double)} An array of coordinates in the order: [ minx, miny, maxx, maxy ] │ │ │ │ │ - */ │ │ │ │ │ - parseEnvelope: function(env, arr) { │ │ │ │ │ - if (arr && arr.length == 4) { │ │ │ │ │ - env.minx = arr[0]; │ │ │ │ │ - env.miny = arr[1]; │ │ │ │ │ - env.maxx = arr[2]; │ │ │ │ │ - env.maxy = arr[3]; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'getLayerWarning': "${layerType} 圖層無法被正確的載入。<br><br>" + │ │ │ │ │ + "要迴避這個訊息, 請在右上角的圖層改變器裡," + │ │ │ │ │ + "選一個新的基礎圖層。<br><br>" + │ │ │ │ │ + "很有可能是因為 ${layerLib} 的函式庫" + │ │ │ │ │ + "腳本沒有被正確的置入。<br><br>" + │ │ │ │ │ + "開發者: 要幫助這個行為正確完成," + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ + "target='_blank'>請按這裡</a>", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: addLayers │ │ │ │ │ - * Add a collection of layers to another collection of layers. Each layer in the list is tuple of │ │ │ │ │ - * { id, visible }. These layer collections represent the │ │ │ │ │ - * /ARCXML/REQUEST/get_image/PROPERTIES/LAYERLIST/LAYERDEF items in ArcXML │ │ │ │ │ - * │ │ │ │ │ - * TODO: Add support for dynamic layer rendering. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * ll - {Array({id,visible})} A list of layer definitions. │ │ │ │ │ - * lyrs - {Array({id,visible})} A list of layer definitions. │ │ │ │ │ - */ │ │ │ │ │ - addLayers: function(ll, lyrs) { │ │ │ │ │ - for (var lind = 0, len = lyrs.length; lind < len; lind++) { │ │ │ │ │ - ll.push(lyrs[lind]); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Scale = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: addImageSize │ │ │ │ │ - * Set the size of the requested image. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * imsize - {Object} An ArcXML imagesize object. │ │ │ │ │ - * olsize - {<OpenLayers.Size>} The image size to set. │ │ │ │ │ - */ │ │ │ │ │ - addImageSize: function(imsize, olsize) { │ │ │ │ │ - if (olsize !== null) { │ │ │ │ │ - imsize.width = olsize.w; │ │ │ │ │ - imsize.height = olsize.h; │ │ │ │ │ - imsize.printwidth = olsize.w; │ │ │ │ │ - imsize.printheight = olsize.h; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + // console message │ │ │ │ │ + 'reprojectDeprecated': "你正使用 'reproject' 這個選項 " + │ │ │ │ │ + "在 ${layerName} 層。這個選項已經不再使用:" + │ │ │ │ │ + "它的使用原本是設計用來支援在商業地圖上秀出資料," + │ │ │ │ │ + "但這個功能已經被" + │ │ │ │ │ + "Spherical Mercator所取代。更多的資訊可以在 " + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SphericalMercator 找到。", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: addCoordSys │ │ │ │ │ - * Add the coordinate system information to an object. The object may be │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featOrFilt - {Object} A featurecoordsys or filtercoordsys ArcXML structure. │ │ │ │ │ - * fsys - {String} or {<OpenLayers.Projection>} or {filtercoordsys} or │ │ │ │ │ - * {featurecoordsys} A projection representation. If it's a {String}, │ │ │ │ │ - * the value is assumed to be the SRID. If it's a {OpenLayers.Projection} │ │ │ │ │ - * AND Proj4js is available, the projection number and name are extracted │ │ │ │ │ - * from there. If it's a filter or feature ArcXML structure, it is copied. │ │ │ │ │ - */ │ │ │ │ │ - addCoordSys: function(featOrFilt, fsys) { │ │ │ │ │ - if (typeof fsys == "string") { │ │ │ │ │ - featOrFilt.id = parseInt(fsys); │ │ │ │ │ - featOrFilt.string = fsys; │ │ │ │ │ - } │ │ │ │ │ - // is this a proj4js instance? │ │ │ │ │ - else if (typeof fsys == "object" && fsys.proj !== null) { │ │ │ │ │ - featOrFilt.id = fsys.proj.srsProjNumber; │ │ │ │ │ - featOrFilt.string = fsys.proj.srsCode; │ │ │ │ │ - } else { │ │ │ │ │ - featOrFilt = fsys; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + // console message │ │ │ │ │ + 'methodDeprecated': "這個方法已經不再使用且在3.0將會被移除," + │ │ │ │ │ + "請使用 ${newMethod} 來代替。", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: iserror │ │ │ │ │ - * Check to see if the response from the server was an error. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. If nothing is supplied, │ │ │ │ │ - * the current response is examined. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true if the response was an error. │ │ │ │ │ - */ │ │ │ │ │ - iserror: function(data) { │ │ │ │ │ - var ret = null; │ │ │ │ │ + 'end': '' │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/hu.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - if (!data) { │ │ │ │ │ - ret = (this.response.error !== ''); │ │ │ │ │ - } else { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - var errorNodes = data.documentElement.getElementsByTagName("ERROR"); │ │ │ │ │ - ret = (errorNodes !== null && errorNodes.length > 0); │ │ │ │ │ - } │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - City-busz │ │ │ │ │ + * - Glanthor Reviol │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - return ret; │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read data from a string, and return an response. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Format.ArcXML.Response>} An ArcXML response. Note that this response │ │ │ │ │ - * data may change in the future. │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["hu"] │ │ │ │ │ + * Dictionary for Magyar. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["hu"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - var arcNode = null; │ │ │ │ │ - if (data && data.documentElement) { │ │ │ │ │ - if (data.documentElement.nodeName == "ARCXML") { │ │ │ │ │ - arcNode = data.documentElement; │ │ │ │ │ - } else { │ │ │ │ │ - arcNode = data.documentElement.getElementsByTagName("ARCXML")[0]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + 'unhandledRequest': "Nem kezelt kérés visszatérése ${statusText}", │ │ │ │ │ │ │ │ │ │ - // in Safari, arcNode will be there but will have a child named │ │ │ │ │ - // parsererror │ │ │ │ │ - if (!arcNode || arcNode.firstChild.nodeName === 'parsererror') { │ │ │ │ │ - var error, source; │ │ │ │ │ - try { │ │ │ │ │ - error = data.firstChild.nodeValue; │ │ │ │ │ - source = data.firstChild.childNodes[1].firstChild.nodeValue; │ │ │ │ │ - } catch (err) { │ │ │ │ │ - // pass │ │ │ │ │ - } │ │ │ │ │ - throw { │ │ │ │ │ - message: "Error parsing the ArcXML request", │ │ │ │ │ - error: error, │ │ │ │ │ - source: source │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ + 'Permalink': "Permalink", │ │ │ │ │ │ │ │ │ │ - var response = this.parseResponse(arcNode); │ │ │ │ │ - return response; │ │ │ │ │ - }, │ │ │ │ │ + 'Overlays': "Rávetítések", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Generate an ArcXml document string for sending to an ArcIMS server. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string representing the ArcXML document request. │ │ │ │ │ - */ │ │ │ │ │ - write: function(request) { │ │ │ │ │ - if (!request) { │ │ │ │ │ - request = this.request; │ │ │ │ │ - } │ │ │ │ │ - var root = this.createElementNS("", "ARCXML"); │ │ │ │ │ - root.setAttribute("version", "1.1"); │ │ │ │ │ + 'Base Layer': "Alapréteg", │ │ │ │ │ │ │ │ │ │ - var reqElem = this.createElementNS("", "REQUEST"); │ │ │ │ │ + 'noFID': "Nem frissíthető olyan jellemző, amely nem rendelkezik FID-del.", │ │ │ │ │ │ │ │ │ │ - if (request.get_image != null) { │ │ │ │ │ - var getElem = this.createElementNS("", "GET_IMAGE"); │ │ │ │ │ - reqElem.appendChild(getElem); │ │ │ │ │ + 'browserNotSupported': "A böngészője nem támogatja a vektoros renderelést. A jelenleg támogatott renderelők:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - var propElem = this.createElementNS("", "PROPERTIES"); │ │ │ │ │ - getElem.appendChild(propElem); │ │ │ │ │ + 'minZoomLevelError': "A minZoomLevel tulajdonságot csak a következővel való használatra szánták: FixedZoomLevels-leszármazott fóliák. Ez azt jelenti, hogy a minZoomLevel wfs fólia jelölőnégyzetei már a múlté. Mi azonban nem távolíthatjuk el annak a veszélye nélkül, hogy az esetlegesen ettől függő OL alapú alkalmazásokat tönkretennénk. Ezért ezt érvénytelenítjük -- a minZoomLevel az alul levő jelölőnégyzet a 3.0-s verzióból el lesz távolítva. Kérjük, helyette használja a min/max felbontás beállítást, amelyről az alábbi helyen talál leírást: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - var props = request.get_image.properties; │ │ │ │ │ - if (props.featurecoordsys != null) { │ │ │ │ │ - var feat = this.createElementNS("", "FEATURECOORDSYS"); │ │ │ │ │ - propElem.appendChild(feat); │ │ │ │ │ + 'commitSuccess': "WFS tranzakció: SIKERES ${response}", │ │ │ │ │ │ │ │ │ │ - if (props.featurecoordsys.id === 0) { │ │ │ │ │ - feat.setAttribute("string", props.featurecoordsys['string']); │ │ │ │ │ - } else { │ │ │ │ │ - feat.setAttribute("id", props.featurecoordsys.id); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + 'commitFailed': "WFS tranzakció: SIKERTELEN ${response}", │ │ │ │ │ │ │ │ │ │ - if (props.filtercoordsys != null) { │ │ │ │ │ - var filt = this.createElementNS("", "FILTERCOORDSYS"); │ │ │ │ │ - propElem.appendChild(filt); │ │ │ │ │ + 'googleWarning': "A Google fólia betöltése sikertelen.\x3cbr\x3e\x3cbr\x3eAhhoz, hogy ez az üzenet eltűnjön, válasszon egy új BaseLayer fóliát a jobb felső sarokban található fóliakapcsoló segítségével.\x3cbr\x3e\x3cbr\x3eNagy valószínűséggel ez azért van, mert a Google Maps könyvtár parancsfájlja nem található, vagy nem tartalmazza az Ön oldalához tartozó megfelelő API-kulcsot.\x3cbr\x3e\x3cbr\x3eFejlesztőknek: A helyes működtetésre vonatkozó segítség az alábbi helyen érhető el, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3ekattintson ide\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - if (props.filtercoordsys.id === 0) { │ │ │ │ │ - filt.setAttribute("string", props.filtercoordsys.string); │ │ │ │ │ - } else { │ │ │ │ │ - filt.setAttribute("id", props.filtercoordsys.id); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + 'getLayerWarning': "A(z) ${layerType} fólia nem töltődött be helyesen.\x3cbr\x3e\x3cbr\x3eAhhoz, hogy ez az üzenet eltűnjön, válasszon egy új BaseLayer fóliát a jobb felső sarokban található fóliakapcsoló segítségével.\x3cbr\x3e\x3cbr\x3eNagy valószínűséggel ez azért van, mert a(z) ${layerLib} könyvtár parancsfájlja helytelen.\x3cbr\x3e\x3cbr\x3eFejlesztőknek: A helyes működtetésre vonatkozó segítség az alábbi helyen érhető el, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3ekattintson ide\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - if (props.envelope != null) { │ │ │ │ │ - var env = this.createElementNS("", "ENVELOPE"); │ │ │ │ │ - propElem.appendChild(env); │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Lépték = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - env.setAttribute("minx", props.envelope.minx); │ │ │ │ │ - env.setAttribute("miny", props.envelope.miny); │ │ │ │ │ - env.setAttribute("maxx", props.envelope.maxx); │ │ │ │ │ - env.setAttribute("maxy", props.envelope.maxy); │ │ │ │ │ - } │ │ │ │ │ + 'W': "Ny", │ │ │ │ │ │ │ │ │ │ - var imagesz = this.createElementNS("", "IMAGESIZE"); │ │ │ │ │ - propElem.appendChild(imagesz); │ │ │ │ │ + 'E': "K", │ │ │ │ │ │ │ │ │ │ - imagesz.setAttribute("height", props.imagesize.height); │ │ │ │ │ - imagesz.setAttribute("width", props.imagesize.width); │ │ │ │ │ + 'N': "É", │ │ │ │ │ │ │ │ │ │ - if (props.imagesize.height != props.imagesize.printheight || │ │ │ │ │ - props.imagesize.width != props.imagesize.printwidth) { │ │ │ │ │ - imagesz.setAttribute("printheight", props.imagesize.printheight); │ │ │ │ │ - imagesz.setArrtibute("printwidth", props.imagesize.printwidth); │ │ │ │ │ - } │ │ │ │ │ + 'S': "D", │ │ │ │ │ │ │ │ │ │ - if (props.background != null) { │ │ │ │ │ - var backgrnd = this.createElementNS("", "BACKGROUND"); │ │ │ │ │ - propElem.appendChild(backgrnd); │ │ │ │ │ + 'reprojectDeprecated': "Ön a \'reproject\' beállítást használja a(z) ${layerName} fólián. Ez a beállítás érvénytelen: használata az üzleti alaptérképek fölötti adatok megjelenítésének támogatására szolgált, de ezt a funkció ezentúl a Gömbi Mercator használatával érhető el. További információ az alábbi helyen érhető el: http://trac.openlayers.org/wiki/SphericalMercator", │ │ │ │ │ │ │ │ │ │ - backgrnd.setAttribute("color", │ │ │ │ │ - props.background.color.r + "," + │ │ │ │ │ - props.background.color.g + "," + │ │ │ │ │ - props.background.color.b); │ │ │ │ │ + 'methodDeprecated': "Ez a módszer érvénytelenítve lett és a 3.0-s verzióból el lesz távolítva. Használja a(z) ${newMethod} módszert helyette." │ │ │ │ │ │ │ │ │ │ - if (props.background.transcolor !== null) { │ │ │ │ │ - backgrnd.setAttribute("transcolor", │ │ │ │ │ - props.background.transcolor.r + "," + │ │ │ │ │ - props.background.transcolor.g + "," + │ │ │ │ │ - props.background.transcolor.b); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/pt.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - if (props.layerlist != null && props.layerlist.length > 0) { │ │ │ │ │ - var layerlst = this.createElementNS("", "LAYERLIST"); │ │ │ │ │ - propElem.appendChild(layerlst); │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Hamilton Abreu │ │ │ │ │ + * - Malafaya │ │ │ │ │ + * - Waldir │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - for (var ld = 0; ld < props.layerlist.length; ld++) { │ │ │ │ │ - var ldef = this.createElementNS("", "LAYERDEF"); │ │ │ │ │ - layerlst.appendChild(ldef); │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - ldef.setAttribute("id", props.layerlist[ld].id); │ │ │ │ │ - ldef.setAttribute("visible", props.layerlist[ld].visible); │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["pt"] │ │ │ │ │ + * Dictionary for Português. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["pt"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - if (typeof props.layerlist[ld].query == "object") { │ │ │ │ │ - var query = props.layerlist[ld].query; │ │ │ │ │ + 'unhandledRequest': "Servidor devolveu erro não contemplado ${statusText}", │ │ │ │ │ │ │ │ │ │ - if (query.where.length < 0) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ + 'Permalink': "Ligação permanente", │ │ │ │ │ │ │ │ │ │ - var queryElem = null; │ │ │ │ │ - if (typeof query.spatialfilter == "boolean" && query.spatialfilter) { │ │ │ │ │ - // handle spatial filter madness │ │ │ │ │ - queryElem = this.createElementNS("", "SPATIALQUERY"); │ │ │ │ │ - } else { │ │ │ │ │ - queryElem = this.createElementNS("", "QUERY"); │ │ │ │ │ - } │ │ │ │ │ + 'Overlays': "Sobreposições", │ │ │ │ │ │ │ │ │ │ - queryElem.setAttribute("where", query.where); │ │ │ │ │ + 'Base Layer': "Camada Base", │ │ │ │ │ │ │ │ │ │ - if (typeof query.accuracy == "number" && query.accuracy > 0) { │ │ │ │ │ - queryElem.setAttribute("accuracy", query.accuracy); │ │ │ │ │ - } │ │ │ │ │ - if (typeof query.featurelimit == "number" && query.featurelimit < 2000) { │ │ │ │ │ - queryElem.setAttribute("featurelimit", query.featurelimit); │ │ │ │ │ - } │ │ │ │ │ - if (typeof query.subfields == "string" && query.subfields != "#ALL#") { │ │ │ │ │ - queryElem.setAttribute("subfields", query.subfields); │ │ │ │ │ - } │ │ │ │ │ - if (typeof query.joinexpression == "string" && query.joinexpression.length > 0) { │ │ │ │ │ - queryElem.setAttribute("joinexpression", query.joinexpression); │ │ │ │ │ - } │ │ │ │ │ - if (typeof query.jointables == "string" && query.jointables.length > 0) { │ │ │ │ │ - queryElem.setAttribute("jointables", query.jointables); │ │ │ │ │ - } │ │ │ │ │ + 'noFID': "Não é possível atualizar um elemento para a qual não há FID.", │ │ │ │ │ │ │ │ │ │ - ldef.appendChild(queryElem); │ │ │ │ │ - } │ │ │ │ │ + 'browserNotSupported': "O seu navegador não suporta renderização vetorial. Actualmente os renderizadores suportados são:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - if (typeof props.layerlist[ld].renderer == "object") { │ │ │ │ │ - this.addRenderer(ldef, props.layerlist[ld].renderer); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else if (request.get_feature != null) { │ │ │ │ │ - var getElem = this.createElementNS("", "GET_FEATURES"); │ │ │ │ │ - getElem.setAttribute("outputmode", "newxml"); │ │ │ │ │ - getElem.setAttribute("checkesc", "true"); │ │ │ │ │ + 'minZoomLevelError': "A propriedade minZoomLevel só deve ser usada com as camadas descendentes da FixedZoomLevels. A verificação da propriedade por esta camada wfs é uma relíquia do passado. No entanto, não podemos removê-la sem correr o risco de afectar aplicações OL que dependam dela. Portanto, estamos a torná-la obsoleta -- a verificação minZoomLevel será removida na versão 3.0. Em vez dela, por favor, use as opções de resolução min/max descritas aqui: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - if (request.get_feature.geometry) { │ │ │ │ │ - getElem.setAttribute("geometry", request.get_feature.geometry); │ │ │ │ │ - } else { │ │ │ │ │ - getElem.setAttribute("geometry", "false"); │ │ │ │ │ - } │ │ │ │ │ + 'commitSuccess': "Transacção WFS: SUCESSO ${response}", │ │ │ │ │ │ │ │ │ │ - if (request.get_feature.compact) { │ │ │ │ │ - getElem.setAttribute("compact", request.get_feature.compact); │ │ │ │ │ - } │ │ │ │ │ + 'commitFailed': "Transacção WFS: FALHOU ${response}", │ │ │ │ │ │ │ │ │ │ - if (request.get_feature.featurelimit == "number") { │ │ │ │ │ - getElem.setAttribute("featurelimit", request.get_feature.featurelimit); │ │ │ │ │ - } │ │ │ │ │ + 'googleWarning': "A Camada Google não foi correctamente carregada.\x3cbr\x3e\x3cbr\x3ePara deixar de receber esta mensagem, seleccione uma nova Camada-Base no \'\'switcher\'\' de camadas no canto superior direito.\x3cbr\x3e\x3cbr\x3eProvavelmente, isto acontece porque o \'\'script\'\' da biblioteca do Google Maps não foi incluído ou não contém a chave API correcta para o seu sítio.\x3cbr\x3e\x3cbr\x3eProgramadores: Para ajuda sobre como solucionar o problema \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eclique aqui\x3c/a\x3e .", │ │ │ │ │ │ │ │ │ │ - getElem.setAttribute("globalenvelope", "true"); │ │ │ │ │ - reqElem.appendChild(getElem); │ │ │ │ │ + 'getLayerWarning': "A camada ${layerType} não foi correctamente carregada.\x3cbr\x3e\x3cbr\x3ePara desactivar esta mensagem, seleccione uma nova Camada-Base no \'\'switcher\'\' de camadas no canto superior direito.\x3cbr\x3e\x3cbr\x3eProvavelmente, isto acontece porque o \'\'script\'\' da biblioteca ${layerLib} não foi incluído correctamente.\x3cbr\x3e\x3cbr\x3eProgramadores: Para ajuda sobre como solucionar o problema \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eclique aqui\x3c/a\x3e .", │ │ │ │ │ │ │ │ │ │ - if (request.get_feature.layer != null && request.get_feature.layer.length > 0) { │ │ │ │ │ - var lyrElem = this.createElementNS("", "LAYER"); │ │ │ │ │ - lyrElem.setAttribute("id", request.get_feature.layer); │ │ │ │ │ - getElem.appendChild(lyrElem); │ │ │ │ │ - } │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Escala = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - var fquery = request.get_feature.query; │ │ │ │ │ - if (fquery != null) { │ │ │ │ │ - var qElem = null; │ │ │ │ │ - if (fquery.isspatial) { │ │ │ │ │ - qElem = this.createElementNS("", "SPATIALQUERY"); │ │ │ │ │ - } else { │ │ │ │ │ - qElem = this.createElementNS("", "QUERY"); │ │ │ │ │ - } │ │ │ │ │ - getElem.appendChild(qElem); │ │ │ │ │ + 'W': "O", │ │ │ │ │ │ │ │ │ │ - if (typeof fquery.accuracy == "number") { │ │ │ │ │ - qElem.setAttribute("accuracy", fquery.accuracy); │ │ │ │ │ - } │ │ │ │ │ - //qElem.setAttribute("featurelimit", "5"); │ │ │ │ │ + 'E': "E", │ │ │ │ │ │ │ │ │ │ - if (fquery.featurecoordsys != null) { │ │ │ │ │ - var fcsElem1 = this.createElementNS("", "FEATURECOORDSYS"); │ │ │ │ │ + 'N': "N", │ │ │ │ │ │ │ │ │ │ - if (fquery.featurecoordsys.id == 0) { │ │ │ │ │ - fcsElem1.setAttribute("string", fquery.featurecoordsys.string); │ │ │ │ │ - } else { │ │ │ │ │ - fcsElem1.setAttribute("id", fquery.featurecoordsys.id); │ │ │ │ │ - } │ │ │ │ │ - qElem.appendChild(fcsElem1); │ │ │ │ │ - } │ │ │ │ │ + 'S': "S", │ │ │ │ │ │ │ │ │ │ - if (fquery.filtercoordsys != null) { │ │ │ │ │ - var fcsElem2 = this.createElementNS("", "FILTERCOORDSYS"); │ │ │ │ │ + 'reprojectDeprecated': "Está usando a opção \'reproject\' na camada ${layerName}. Esta opção é obsoleta: foi concebida para permitir a apresentação de dados sobre mapas-base comerciais, mas esta funcionalidade é agora suportada pelo Mercator Esférico. Mais informação está disponível em http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ │ │ │ │ │ - if (fquery.filtercoordsys.id === 0) { │ │ │ │ │ - fcsElem2.setAttribute("string", fquery.filtercoordsys.string); │ │ │ │ │ - } else { │ │ │ │ │ - fcsElem2.setAttribute("id", fquery.filtercoordsys.id); │ │ │ │ │ - } │ │ │ │ │ - qElem.appendChild(fcsElem2); │ │ │ │ │ - } │ │ │ │ │ + 'methodDeprecated': "Este método foi declarado obsoleto e será removido na versão 3.0. Por favor, use ${newMethod} em vez disso." │ │ │ │ │ │ │ │ │ │ - if (fquery.buffer > 0) { │ │ │ │ │ - var bufElem = this.createElementNS("", "BUFFER"); │ │ │ │ │ - bufElem.setAttribute("distance", fquery.buffer); │ │ │ │ │ - qElem.appendChild(bufElem); │ │ │ │ │ - } │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/hr.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - if (fquery.isspatial) { │ │ │ │ │ - var spfElem = this.createElementNS("", "SPATIALFILTER"); │ │ │ │ │ - spfElem.setAttribute("relation", fquery.spatialfilter.relation); │ │ │ │ │ - qElem.appendChild(spfElem); │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Mvrban │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (fquery.spatialfilter.envelope) { │ │ │ │ │ - var envElem = this.createElementNS("", "ENVELOPE"); │ │ │ │ │ - envElem.setAttribute("minx", fquery.spatialfilter.envelope.minx); │ │ │ │ │ - envElem.setAttribute("miny", fquery.spatialfilter.envelope.miny); │ │ │ │ │ - envElem.setAttribute("maxx", fquery.spatialfilter.envelope.maxx); │ │ │ │ │ - envElem.setAttribute("maxy", fquery.spatialfilter.envelope.maxy); │ │ │ │ │ - spfElem.appendChild(envElem); │ │ │ │ │ - } else if (typeof fquery.spatialfilter.polygon == "object") { │ │ │ │ │ - spfElem.appendChild(this.writePolygonGeometry(fquery.spatialfilter.polygon)); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (fquery.where != null && fquery.where.length > 0) { │ │ │ │ │ - qElem.setAttribute("where", fquery.where); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["hr"] │ │ │ │ │ + * Dictionary for Hrvatski. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["hr"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - root.appendChild(reqElem); │ │ │ │ │ + 'unhandledRequest': "Nepodržani zahtjev ${statusText}", │ │ │ │ │ │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [root]); │ │ │ │ │ - }, │ │ │ │ │ + 'Permalink': "Permalink", │ │ │ │ │ │ │ │ │ │ + 'Overlays': "Overlays", │ │ │ │ │ │ │ │ │ │ - addGroupRenderer: function(ldef, toprenderer) { │ │ │ │ │ - var topRelem = this.createElementNS("", "GROUPRENDERER"); │ │ │ │ │ - ldef.appendChild(topRelem); │ │ │ │ │ + 'Base Layer': "Osnovna karta", │ │ │ │ │ │ │ │ │ │ - for (var rind = 0; rind < toprenderer.length; rind++) { │ │ │ │ │ - var renderer = toprenderer[rind]; │ │ │ │ │ - this.addRenderer(topRelem, renderer); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'noFID': "Ne mogu ažurirati značajku za koju ne postoji FID.", │ │ │ │ │ │ │ │ │ │ + 'browserNotSupported': "Vaš preglednik ne podržava vektorsko renderiranje. Trenutno podržani rendereri su: ${renderers}", │ │ │ │ │ │ │ │ │ │ - addRenderer: function(topRelem, renderer) { │ │ │ │ │ - if (OpenLayers.Util.isArray(renderer)) { │ │ │ │ │ - this.addGroupRenderer(topRelem, renderer); │ │ │ │ │ - } else { │ │ │ │ │ - var renderElem = this.createElementNS("", renderer.type.toUpperCase() + "RENDERER"); │ │ │ │ │ - topRelem.appendChild(renderElem); │ │ │ │ │ + 'commitSuccess': "WFS Transakcija: USPJEŠNA ${response}", │ │ │ │ │ │ │ │ │ │ - if (renderElem.tagName == "VALUEMAPRENDERER") { │ │ │ │ │ - this.addValueMapRenderer(renderElem, renderer); │ │ │ │ │ - } else if (renderElem.tagName == "VALUEMAPLABELRENDERER") { │ │ │ │ │ - this.addValueMapLabelRenderer(renderElem, renderer); │ │ │ │ │ - } else if (renderElem.tagName == "SIMPLELABELRENDERER") { │ │ │ │ │ - this.addSimpleLabelRenderer(renderElem, renderer); │ │ │ │ │ - } else if (renderElem.tagName == "SCALEDEPENDENTRENDERER") { │ │ │ │ │ - this.addScaleDependentRenderer(renderElem, renderer); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'commitFailed': "WFS Transakcija: NEUSPJEŠNA ${response}", │ │ │ │ │ │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Mjerilo = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - addScaleDependentRenderer: function(renderElem, renderer) { │ │ │ │ │ - if (typeof renderer.lower == "string" || typeof renderer.lower == "number") { │ │ │ │ │ - renderElem.setAttribute("lower", renderer.lower); │ │ │ │ │ - } │ │ │ │ │ - if (typeof renderer.upper == "string" || typeof renderer.upper == "number") { │ │ │ │ │ - renderElem.setAttribute("upper", renderer.upper); │ │ │ │ │ - } │ │ │ │ │ + 'methodDeprecated': "Ova metoda nije odobrena i biti će maknuta u 3.0. Koristite ${newMethod}." │ │ │ │ │ │ │ │ │ │ - this.addRenderer(renderElem, renderer.renderer); │ │ │ │ │ - }, │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/ro.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["ro"] │ │ │ │ │ + * Dictionary for Romanian. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang.ro = { │ │ │ │ │ + 'unhandledRequest': "Cerere nesoluționată return ${statusText}", │ │ │ │ │ + 'Permalink': "Legatură permanentă", │ │ │ │ │ + 'Overlays': "Straturi vector", │ │ │ │ │ + 'Base Layer': "Straturi de bază", │ │ │ │ │ + 'noFID': "Nu pot actualiza un feature pentru care nu există FID.", │ │ │ │ │ + 'browserNotSupported': "Browserul tău nu suportă afișarea vectorilor. Supoetul curent pentru randare:\n${renderers}", │ │ │ │ │ + // console message │ │ │ │ │ + 'minZoomLevelError': "Proprietatea minZoomLevel este doar pentru a fi folosită " + │ │ │ │ │ + "cu straturile FixedZoomLevels-descendent. De aceea acest " + │ │ │ │ │ + "strat wfs verifică dacă minZoomLevel este o relicvă" + │ │ │ │ │ + ". Nu îl putem , oricum, înlătura fără " + │ │ │ │ │ + "a afecta aplicațiile Openlayers care depind de ea." + │ │ │ │ │ + " De aceea considerăm depreciat -- minZoomLevel " + │ │ │ │ │ + "și îl vom înlătura în 3.0. Folosește " + │ │ │ │ │ + "min/max resolution cum este descrisă aici: " + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + 'commitSuccess': "Tranzacție WFS: SUCCES ${response}", │ │ │ │ │ + 'commitFailed': "Tranzacție WFS : EȘEC ${response}", │ │ │ │ │ + 'googleWarning': "Stratul Google nu a putut fi încărcat corect.<br><br>" + │ │ │ │ │ + "Pentru a elimina acest mesaj, selectează un nou strat de bază " + │ │ │ │ │ + "în Layerswitcher din colțul dreata-sus.<br><br>" + │ │ │ │ │ + "Asta datorită, faptului că Google Maps library " + │ │ │ │ │ + "script nu este inclus, sau nu conține " + │ │ │ │ │ + "cheia API corectă pentru situl tău.<br><br>" + │ │ │ │ │ + "Developeri: Pentru ajutor, " + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ + "target='_blank'>apăsați aici</a>", │ │ │ │ │ + 'getLayerWarning': "Stratul ${layerType} nu a putut fi încărcat corect.<br><br>" + │ │ │ │ │ + "pentru a înlătura acest mesaj, selectează un nou strat de bază " + │ │ │ │ │ + "Acesta eroare apare de obicei când ${layerLib} library " + │ │ │ │ │ + "script nu a fost încărcat corect.<br><br>" + │ │ │ │ │ + "Developeri: Pentru ajutor privind utilizarea corectă, " + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ + "target='_blank'>apasă aici</a>", │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Scara = 1 : ${scaleDenom}", │ │ │ │ │ + //labels for the graticule control │ │ │ │ │ + 'W': 'V', │ │ │ │ │ + 'E': 'E', │ │ │ │ │ + 'N': 'N', │ │ │ │ │ + 'S': 'S', │ │ │ │ │ + 'Graticule': 'Graticule', │ │ │ │ │ + // console message │ │ │ │ │ + 'reprojectDeprecated': "folosești opțiunea 'reproject' " + │ │ │ │ │ + "pentru stratul ${layerName} . Această opțiune este depreciată: " + │ │ │ │ │ + "a fost utilizată pentru afișarea straturilor de bază comerciale " + │ │ │ │ │ + "Mai multe informații despre proiecția Mercator sunt disponibile aici " + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + // console message │ │ │ │ │ + 'methodDeprecated': "Această metodă este depreciată și va fi înlăturată in versiunea 3.0. " + │ │ │ │ │ + "folosește metoda ${newMethod}.", │ │ │ │ │ + // **** end **** │ │ │ │ │ + 'end': '' │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/ar.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - addValueMapLabelRenderer: function(renderElem, renderer) { │ │ │ │ │ - renderElem.setAttribute("lookupfield", renderer.lookupfield); │ │ │ │ │ - renderElem.setAttribute("labelfield", renderer.labelfield); │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Meno25 │ │ │ │ │ + * - Mutarjem horr │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (typeof renderer.exacts == "object") { │ │ │ │ │ - for (var ext = 0, extlen = renderer.exacts.length; ext < extlen; ext++) { │ │ │ │ │ - var exact = renderer.exacts[ext]; │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - var eelem = this.createElementNS("", "EXACT"); │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["ar"] │ │ │ │ │ + * Dictionary for العربية. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["ar"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - if (typeof exact.value == "string") { │ │ │ │ │ - eelem.setAttribute("value", exact.value); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.label == "string") { │ │ │ │ │ - eelem.setAttribute("label", exact.label); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.method == "string") { │ │ │ │ │ - eelem.setAttribute("method", exact.method); │ │ │ │ │ - } │ │ │ │ │ + 'Permalink': "وصلة دائمة", │ │ │ │ │ │ │ │ │ │ - renderElem.appendChild(eelem); │ │ │ │ │ + 'Base Layer': "الطبقة الاساسية", │ │ │ │ │ │ │ │ │ │ - if (typeof exact.symbol == "object") { │ │ │ │ │ - var selem = null; │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "النسبة = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - if (exact.symbol.type == "text") { │ │ │ │ │ - selem = this.createElementNS("", "TEXTSYMBOL"); │ │ │ │ │ - } │ │ │ │ │ + 'W': "غ", │ │ │ │ │ │ │ │ │ │ - if (selem != null) { │ │ │ │ │ - var keys = this.fontStyleKeys; │ │ │ │ │ - for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ - var key = keys[i]; │ │ │ │ │ - if (exact.symbol[key]) { │ │ │ │ │ - selem.setAttribute(key, exact.symbol[key]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - eelem.appendChild(selem); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } // for each exact │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'E': "شر", │ │ │ │ │ │ │ │ │ │ - addValueMapRenderer: function(renderElem, renderer) { │ │ │ │ │ - renderElem.setAttribute("lookupfield", renderer.lookupfield); │ │ │ │ │ + 'N': "شم", │ │ │ │ │ │ │ │ │ │ - if (typeof renderer.ranges == "object") { │ │ │ │ │ - for (var rng = 0, rnglen = renderer.ranges.length; rng < rnglen; rng++) { │ │ │ │ │ - var range = renderer.ranges[rng]; │ │ │ │ │ + 'S': "ج" │ │ │ │ │ │ │ │ │ │ - var relem = this.createElementNS("", "RANGE"); │ │ │ │ │ - relem.setAttribute("lower", range.lower); │ │ │ │ │ - relem.setAttribute("upper", range.upper); │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/ru.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - renderElem.appendChild(relem); │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Ferrer │ │ │ │ │ + * - Komzpa │ │ │ │ │ + * - Lockal │ │ │ │ │ + * - Александр Сигачёв │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (typeof range.symbol == "object") { │ │ │ │ │ - var selem = null; │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (range.symbol.type == "simplepolygon") { │ │ │ │ │ - selem = this.createElementNS("", "SIMPLEPOLYGONSYMBOL"); │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["ru"] │ │ │ │ │ + * Dictionary for Русский. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["ru"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - if (selem != null) { │ │ │ │ │ - if (typeof range.symbol.boundarycolor == "string") { │ │ │ │ │ - selem.setAttribute("boundarycolor", range.symbol.boundarycolor); │ │ │ │ │ - } │ │ │ │ │ - if (typeof range.symbol.fillcolor == "string") { │ │ │ │ │ - selem.setAttribute("fillcolor", range.symbol.fillcolor); │ │ │ │ │ - } │ │ │ │ │ - if (typeof range.symbol.filltransparency == "number") { │ │ │ │ │ - selem.setAttribute("filltransparency", range.symbol.filltransparency); │ │ │ │ │ - } │ │ │ │ │ - relem.appendChild(selem); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } // for each range │ │ │ │ │ - } else if (typeof renderer.exacts == "object") { │ │ │ │ │ - for (var ext = 0, extlen = renderer.exacts.length; ext < extlen; ext++) { │ │ │ │ │ - var exact = renderer.exacts[ext]; │ │ │ │ │ + 'unhandledRequest': "Необработанный запрос вернул ${statusText}", │ │ │ │ │ │ │ │ │ │ - var eelem = this.createElementNS("", "EXACT"); │ │ │ │ │ - if (typeof exact.value == "string") { │ │ │ │ │ - eelem.setAttribute("value", exact.value); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.label == "string") { │ │ │ │ │ - eelem.setAttribute("label", exact.label); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.method == "string") { │ │ │ │ │ - eelem.setAttribute("method", exact.method); │ │ │ │ │ - } │ │ │ │ │ + 'Permalink': "Постоянная ссылка", │ │ │ │ │ │ │ │ │ │ - renderElem.appendChild(eelem); │ │ │ │ │ + 'Overlays': "Слои", │ │ │ │ │ │ │ │ │ │ - if (typeof exact.symbol == "object") { │ │ │ │ │ - var selem = null; │ │ │ │ │ + 'Base Layer': "Основной слой", │ │ │ │ │ │ │ │ │ │ - if (exact.symbol.type == "simplemarker") { │ │ │ │ │ - selem = this.createElementNS("", "SIMPLEMARKERSYMBOL"); │ │ │ │ │ - } │ │ │ │ │ + 'noFID': "Невозможно обновить объект, для которого нет FID.", │ │ │ │ │ │ │ │ │ │ - if (selem != null) { │ │ │ │ │ - if (typeof exact.symbol.antialiasing == "string") { │ │ │ │ │ - selem.setAttribute("antialiasing", exact.symbol.antialiasing); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.color == "string") { │ │ │ │ │ - selem.setAttribute("color", exact.symbol.color); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.outline == "string") { │ │ │ │ │ - selem.setAttribute("outline", exact.symbol.outline); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.overlap == "string") { │ │ │ │ │ - selem.setAttribute("overlap", exact.symbol.overlap); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.shadow == "string") { │ │ │ │ │ - selem.setAttribute("shadow", exact.symbol.shadow); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.transparency == "number") { │ │ │ │ │ - selem.setAttribute("transparency", exact.symbol.transparency); │ │ │ │ │ - } │ │ │ │ │ - //if (typeof exact.symbol.type == "string") │ │ │ │ │ - // selem.setAttribute("type", exact.symbol.type); │ │ │ │ │ - if (typeof exact.symbol.usecentroid == "string") { │ │ │ │ │ - selem.setAttribute("usecentroid", exact.symbol.usecentroid); │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.width == "number") { │ │ │ │ │ - selem.setAttribute("width", exact.symbol.width); │ │ │ │ │ - } │ │ │ │ │ + 'browserNotSupported': "Ваш браузер не поддерживает векторную графику. На данный момент поддерживаются:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - eelem.appendChild(selem); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } // for each exact │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'minZoomLevelError': "Свойство minZoomLevel предназначено только для использования со слоями, являющимися потомками FixedZoomLevels. То, что этот WFS-слой проверяется на minZoomLevel — реликт прошлого. Однако мы не можем удалить эту функцию, так как, возможно, от неё зависят некоторые основанные на OpenLayers приложения. Функция объявлена устаревшей — проверка minZoomLevel будет удалена в 3.0. Пожалуйста, используйте вместо неё настройку мин/макс разрешения, описанную здесь: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ + 'commitSuccess': "Транзакция WFS: УСПЕШНО ${response}", │ │ │ │ │ │ │ │ │ │ - addSimpleLabelRenderer: function(renderElem, renderer) { │ │ │ │ │ - renderElem.setAttribute("field", renderer.field); │ │ │ │ │ - var keys = ['featureweight', 'howmanylabels', 'labelbufferratio', │ │ │ │ │ - 'labelpriorities', 'labelweight', 'linelabelposition', │ │ │ │ │ - 'rotationalangles' │ │ │ │ │ - ]; │ │ │ │ │ - for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ - var key = keys[i]; │ │ │ │ │ - if (renderer[key]) { │ │ │ │ │ - renderElem.setAttribute(key, renderer[key]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + 'commitFailed': "Транзакция WFS: ОШИБКА ${response}", │ │ │ │ │ │ │ │ │ │ - if (renderer.symbol.type == "text") { │ │ │ │ │ - var symbol = renderer.symbol; │ │ │ │ │ - var selem = this.createElementNS("", "TEXTSYMBOL"); │ │ │ │ │ - renderElem.appendChild(selem); │ │ │ │ │ + 'googleWarning': "Слой Google не удалось нормально загрузить.\x3cbr\x3e\x3cbr\x3eЧтобы избавиться от этого сообщения, выбите другой основной слой в переключателе в правом верхнем углу.\x3cbr\x3e\x3cbr\x3eСкорее всего, причина в том, что библиотека Google Maps не была включена или не содержит корректного API-ключа для вашего сайта.\x3cbr\x3e\x3cbr\x3eРазработчикам: чтобы узнать, как сделать, чтобы всё заработало, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eщёлкните тут\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - var keys = this.fontStyleKeys; │ │ │ │ │ - for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ - var key = keys[i]; │ │ │ │ │ - if (symbol[key]) { │ │ │ │ │ - selem.setAttribute(key, renderer[key]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'getLayerWarning': "Слой ${layerType} не удалось нормально загрузить. \x3cbr\x3e\x3cbr\x3eЧтобы избавиться от этого сообщения, выбите другой основной слой в переключателе в правом верхнем углу.\x3cbr\x3e\x3cbr\x3eСкорее всего, причина в том, что библиотека ${layerLib} не была включена или была включена некорректно.\x3cbr\x3e\x3cbr\x3eРазработчикам: чтобы узнать, как сделать, чтобы всё заработало, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eщёлкните тут\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - writePolygonGeometry: function(polygon) { │ │ │ │ │ - if (!(polygon instanceof OpenLayers.Geometry.Polygon)) { │ │ │ │ │ - throw { │ │ │ │ │ - message: 'Cannot write polygon geometry to ArcXML with an ' + │ │ │ │ │ - polygon.CLASS_NAME + ' object.', │ │ │ │ │ - geometry: polygon │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Масштаб = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - var polyElem = this.createElementNS("", "POLYGON"); │ │ │ │ │ + 'W': "З", │ │ │ │ │ │ │ │ │ │ - for (var ln = 0, lnlen = polygon.components.length; ln < lnlen; ln++) { │ │ │ │ │ - var ring = polygon.components[ln]; │ │ │ │ │ - var ringElem = this.createElementNS("", "RING"); │ │ │ │ │ + 'E': "В", │ │ │ │ │ │ │ │ │ │ - for (var rn = 0, rnlen = ring.components.length; rn < rnlen; rn++) { │ │ │ │ │ - var point = ring.components[rn]; │ │ │ │ │ - var pointElem = this.createElementNS("", "POINT"); │ │ │ │ │ + 'N': "С", │ │ │ │ │ │ │ │ │ │ - pointElem.setAttribute("x", point.x); │ │ │ │ │ - pointElem.setAttribute("y", point.y); │ │ │ │ │ + 'S': "Ю", │ │ │ │ │ │ │ │ │ │ - ringElem.appendChild(pointElem); │ │ │ │ │ - } │ │ │ │ │ + 'reprojectDeprecated': "Вы используете опцию \'reproject\' для слоя ${layerName}. Эта опция является устаревшей: ее использование предполагалось для поддержки показа данных поверх коммерческих базовых карт, но теперь этот функционал несёт встроенная поддержка сферической проекции Меркатора. Больше сведений доступно на http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ │ │ │ │ │ - polyElem.appendChild(ringElem); │ │ │ │ │ - } │ │ │ │ │ + 'methodDeprecated': "Этот метод считается устаревшим и будет удалён в версии 3.0. Пожалуйста, пользуйтесь ${newMethod}." │ │ │ │ │ │ │ │ │ │ - return polyElem; │ │ │ │ │ - }, │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/lt.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseResponse │ │ │ │ │ - * Take an ArcXML response, and parse in into this object's internal properties. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} The ArcXML response, as either a string or the │ │ │ │ │ - * top level DOMElement of the response. │ │ │ │ │ - */ │ │ │ │ │ - parseResponse: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - var newData = new OpenLayers.Format.XML(); │ │ │ │ │ - data = newData.read(data); │ │ │ │ │ - } │ │ │ │ │ - var response = new OpenLayers.Format.ArcXML.Response(); │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - var errorNode = data.getElementsByTagName("ERROR"); │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["lt"] │ │ │ │ │ + * Dictionary for Lithuanian. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang['lt'] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - if (errorNode != null && errorNode.length > 0) { │ │ │ │ │ - response.error = this.getChildValue(errorNode, "Unknown error."); │ │ │ │ │ - } else { │ │ │ │ │ - var responseNode = data.getElementsByTagName("RESPONSE"); │ │ │ │ │ + 'unhandledRequest': "Neapdorota užklausa gražino ${statusText}", │ │ │ │ │ │ │ │ │ │ - if (responseNode == null || responseNode.length == 0) { │ │ │ │ │ - response.error = "No RESPONSE tag found in ArcXML response."; │ │ │ │ │ - return response; │ │ │ │ │ - } │ │ │ │ │ + 'Permalink': "Pastovi nuoroda", │ │ │ │ │ │ │ │ │ │ - var rtype = responseNode[0].firstChild.nodeName; │ │ │ │ │ - if (rtype == "#text") { │ │ │ │ │ - rtype = responseNode[0].firstChild.nextSibling.nodeName; │ │ │ │ │ - } │ │ │ │ │ + 'Overlays': "Papildomi sluoksniai", │ │ │ │ │ │ │ │ │ │ - if (rtype == "IMAGE") { │ │ │ │ │ - var envelopeNode = data.getElementsByTagName("ENVELOPE"); │ │ │ │ │ - var outputNode = data.getElementsByTagName("OUTPUT"); │ │ │ │ │ + 'Base Layer': "Pagrindinis sluoksnis", │ │ │ │ │ │ │ │ │ │ - if (envelopeNode == null || envelopeNode.length == 0) { │ │ │ │ │ - response.error = "No ENVELOPE tag found in ArcXML response."; │ │ │ │ │ - } else if (outputNode == null || outputNode.length == 0) { │ │ │ │ │ - response.error = "No OUTPUT tag found in ArcXML response."; │ │ │ │ │ - } else { │ │ │ │ │ - var envAttr = this.parseAttributes(envelopeNode[0]); │ │ │ │ │ - var outputAttr = this.parseAttributes(outputNode[0]); │ │ │ │ │ + 'noFID': "Negaliu atnaujinti objekto, kuris neturi FID.", │ │ │ │ │ │ │ │ │ │ - if (typeof outputAttr.type == "string") { │ │ │ │ │ - response.image = { │ │ │ │ │ - envelope: envAttr, │ │ │ │ │ - output: { │ │ │ │ │ - type: outputAttr.type, │ │ │ │ │ - data: this.getChildValue(outputNode[0]) │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - } else { │ │ │ │ │ - response.image = { │ │ │ │ │ - envelope: envAttr, │ │ │ │ │ - output: outputAttr │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else if (rtype == "FEATURES") { │ │ │ │ │ - var features = responseNode[0].getElementsByTagName("FEATURES"); │ │ │ │ │ + 'browserNotSupported': "Jūsų naršyklė nemoka parodyti vektorių. Šiuo metu galima naudotis tokiais rodymo varikliais:\n{renderers}", │ │ │ │ │ │ │ │ │ │ - // get the feature count │ │ │ │ │ - var featureCount = features[0].getElementsByTagName("FEATURECOUNT"); │ │ │ │ │ - response.features.featurecount = featureCount[0].getAttribute("count"); │ │ │ │ │ + 'commitSuccess': "WFS Tranzakcija: PAVYKO ${response}", │ │ │ │ │ │ │ │ │ │ - if (response.features.featurecount > 0) { │ │ │ │ │ - // get the feature envelope │ │ │ │ │ - var envelope = features[0].getElementsByTagName("ENVELOPE"); │ │ │ │ │ - response.features.envelope = this.parseAttributes(envelope[0], typeof(0)); │ │ │ │ │ + 'commitFailed': "WFS Tranzakcija: ŽLUGO ${response}", │ │ │ │ │ │ │ │ │ │ - // get the field values per feature │ │ │ │ │ - var featureList = features[0].getElementsByTagName("FEATURE"); │ │ │ │ │ - for (var fn = 0; fn < featureList.length; fn++) { │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ - var fields = featureList[fn].getElementsByTagName("FIELD"); │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Mastelis = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - for (var fdn = 0; fdn < fields.length; fdn++) { │ │ │ │ │ - var fieldName = fields[fdn].getAttribute("name"); │ │ │ │ │ - var fieldValue = fields[fdn].getAttribute("value"); │ │ │ │ │ - feature.attributes[fieldName] = fieldValue; │ │ │ │ │ - } │ │ │ │ │ + //labels for the graticule control │ │ │ │ │ + 'W': 'V', │ │ │ │ │ + 'E': 'R', │ │ │ │ │ + 'N': 'Š', │ │ │ │ │ + 'S': 'P', │ │ │ │ │ + 'Graticule': 'Tinklelis', │ │ │ │ │ │ │ │ │ │ - var geom = featureList[fn].getElementsByTagName("POLYGON"); │ │ │ │ │ + // console message │ │ │ │ │ + 'methodDeprecated': "Šis metodas yra pasenęs ir 3.0 versijoje bus pašalintas. " + │ │ │ │ │ + "Prašome naudoti ${newMethod}.", │ │ │ │ │ │ │ │ │ │ - if (geom.length > 0) { │ │ │ │ │ - // if there is a polygon, create an openlayers polygon, and assign │ │ │ │ │ - // it to the .geometry property of the feature │ │ │ │ │ - var ring = geom[0].getElementsByTagName("RING"); │ │ │ │ │ + // **** end **** │ │ │ │ │ + 'end': '' │ │ │ │ │ │ │ │ │ │ - var polys = []; │ │ │ │ │ - for (var rn = 0; rn < ring.length; rn++) { │ │ │ │ │ - var linearRings = []; │ │ │ │ │ - linearRings.push(this.parsePointGeometry(ring[rn])); │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/nds.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - var holes = ring[rn].getElementsByTagName("HOLE"); │ │ │ │ │ - for (var hn = 0; hn < holes.length; hn++) { │ │ │ │ │ - linearRings.push(this.parsePointGeometry(holes[hn])); │ │ │ │ │ - } │ │ │ │ │ - holes = null; │ │ │ │ │ - polys.push(new OpenLayers.Geometry.Polygon(linearRings)); │ │ │ │ │ - linearRings = null; │ │ │ │ │ - } │ │ │ │ │ - ring = null; │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Slomox │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (polys.length == 1) { │ │ │ │ │ - feature.geometry = polys[0]; │ │ │ │ │ - } else { │ │ │ │ │ - feature.geometry = new OpenLayers.Geometry.MultiPolygon(polys); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - response.features.feature.push(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - response.error = "Unidentified response type."; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return response; │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["nds"] │ │ │ │ │ + * Dictionary for Plattdüütsch. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["nds"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ + 'unhandledRequest': "Unbehannelt Trüchmellels för de Anfraag ${statusText}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseAttributes │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {<DOMElement>} An element to parse attributes from. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An attributes object, with properties set to attribute values. │ │ │ │ │ - */ │ │ │ │ │ - parseAttributes: function(node, type) { │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - for (var attr = 0; attr < node.attributes.length; attr++) { │ │ │ │ │ - if (type == "number") { │ │ │ │ │ - attributes[node.attributes[attr].nodeName] = parseFloat(node.attributes[attr].nodeValue); │ │ │ │ │ - } else { │ │ │ │ │ - attributes[node.attributes[attr].nodeName] = node.attributes[attr].nodeValue; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return attributes; │ │ │ │ │ - }, │ │ │ │ │ + 'Permalink': "Permalink", │ │ │ │ │ │ │ │ │ │ + 'Overlays': "Overlays", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parsePointGeometry │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {<DOMElement>} An element to parse <COORDS> or <POINT> arcxml data from. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.LinearRing>} A linear ring represented by the node's points. │ │ │ │ │ - */ │ │ │ │ │ - parsePointGeometry: function(node) { │ │ │ │ │ - var ringPoints = []; │ │ │ │ │ - var coords = node.getElementsByTagName("COORDS"); │ │ │ │ │ + 'Base Layer': "Achtergrundkoort", │ │ │ │ │ │ │ │ │ │ - if (coords.length > 0) { │ │ │ │ │ - // if coords is present, it's the only coords item │ │ │ │ │ - var coordArr = this.getChildValue(coords[0]); │ │ │ │ │ - coordArr = coordArr.split(/;/); │ │ │ │ │ - for (var cn = 0; cn < coordArr.length; cn++) { │ │ │ │ │ - var coordItems = coordArr[cn].split(/ /); │ │ │ │ │ - ringPoints.push(new OpenLayers.Geometry.Point(coordItems[0], coordItems[1])); │ │ │ │ │ - } │ │ │ │ │ - coords = null; │ │ │ │ │ - } else { │ │ │ │ │ - var point = node.getElementsByTagName("POINT"); │ │ │ │ │ - if (point.length > 0) { │ │ │ │ │ - for (var pn = 0; pn < point.length; pn++) { │ │ │ │ │ - ringPoints.push( │ │ │ │ │ - new OpenLayers.Geometry.Point( │ │ │ │ │ - parseFloat(point[pn].getAttribute("x")), │ │ │ │ │ - parseFloat(point[pn].getAttribute("y")) │ │ │ │ │ - ) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - point = null; │ │ │ │ │ - } │ │ │ │ │ + 'noFID': "En Feature, dat keen FID hett, kann nich aktuell maakt warrn.", │ │ │ │ │ │ │ │ │ │ - return new OpenLayers.Geometry.LinearRing(ringPoints); │ │ │ │ │ - }, │ │ │ │ │ + 'browserNotSupported': "Dien Browser ünnerstütt keen Vektorbiller. Ünnerstütt Renderers:\n${renderers}", │ │ │ │ │ + │ │ │ │ │ + 'commitSuccess': "WFS-Transakschoon: hett klappt ${response}", │ │ │ │ │ + │ │ │ │ │ + 'commitFailed': "WFS-Transakschoon: hett nich klappt ${response}", │ │ │ │ │ + │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Skaal = 1 : ${scaleDenom}", │ │ │ │ │ + │ │ │ │ │ + 'methodDeprecated': "Disse Methood is oold un schall dat in 3.0 nich mehr geven. Bruuk dor man beter ${newMethod} för." │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.ArcXML" │ │ │ │ │ }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/br.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -OpenLayers.Format.ArcXML.Request = OpenLayers.Class({ │ │ │ │ │ - initialize: function(params) { │ │ │ │ │ - var defaults = { │ │ │ │ │ - get_image: { │ │ │ │ │ - properties: { │ │ │ │ │ - background: null, │ │ │ │ │ - /*{ │ │ │ │ │ - color: { r:255, g:255, b:255 }, │ │ │ │ │ - transcolor: null │ │ │ │ │ - },*/ │ │ │ │ │ - draw: true, │ │ │ │ │ - envelope: { │ │ │ │ │ - minx: 0, │ │ │ │ │ - miny: 0, │ │ │ │ │ - maxx: 0, │ │ │ │ │ - maxy: 0 │ │ │ │ │ - }, │ │ │ │ │ - featurecoordsys: { │ │ │ │ │ - id: 0, │ │ │ │ │ - string: "", │ │ │ │ │ - datumtransformid: 0, │ │ │ │ │ - datumtransformstring: "" │ │ │ │ │ - }, │ │ │ │ │ - filtercoordsys: { │ │ │ │ │ - id: 0, │ │ │ │ │ - string: "", │ │ │ │ │ - datumtransformid: 0, │ │ │ │ │ - datumtransformstring: "" │ │ │ │ │ - }, │ │ │ │ │ - imagesize: { │ │ │ │ │ - height: 0, │ │ │ │ │ - width: 0, │ │ │ │ │ - dpi: 96, │ │ │ │ │ - printheight: 0, │ │ │ │ │ - printwidth: 0, │ │ │ │ │ - scalesymbols: false │ │ │ │ │ - }, │ │ │ │ │ - layerlist: [], │ │ │ │ │ - /* no support for legends */ │ │ │ │ │ - output: { │ │ │ │ │ - baseurl: "", │ │ │ │ │ - legendbaseurl: "", │ │ │ │ │ - legendname: "", │ │ │ │ │ - legendpath: "", │ │ │ │ │ - legendurl: "", │ │ │ │ │ - name: "", │ │ │ │ │ - path: "", │ │ │ │ │ - type: "jpg", │ │ │ │ │ - url: "" │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Fulup │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - get_feature: { │ │ │ │ │ - layer: "", │ │ │ │ │ - query: { │ │ │ │ │ - isspatial: false, │ │ │ │ │ - featurecoordsys: { │ │ │ │ │ - id: 0, │ │ │ │ │ - string: "", │ │ │ │ │ - datumtransformid: 0, │ │ │ │ │ - datumtransformstring: "" │ │ │ │ │ - }, │ │ │ │ │ - filtercoordsys: { │ │ │ │ │ - id: 0, │ │ │ │ │ - string: "", │ │ │ │ │ - datumtransformid: 0, │ │ │ │ │ - datumtransformstring: "" │ │ │ │ │ - }, │ │ │ │ │ - buffer: 0, │ │ │ │ │ - where: "", │ │ │ │ │ - spatialfilter: { │ │ │ │ │ - relation: "envelope_intersection", │ │ │ │ │ - envelope: null │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - environment: { │ │ │ │ │ - separators: { │ │ │ │ │ - cs: " ", │ │ │ │ │ - ts: ";" │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["br"] │ │ │ │ │ + * Dictionary for Brezhoneg. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["br"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - layer: [], │ │ │ │ │ - workspaces: [] │ │ │ │ │ - }; │ │ │ │ │ + 'unhandledRequest': "Distro evel reked anveret ${statusText}", │ │ │ │ │ │ │ │ │ │ - return OpenLayers.Util.extend(this, defaults); │ │ │ │ │ - }, │ │ │ │ │ + 'Permalink': "Peurliamm", │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.ArcXML.Request" │ │ │ │ │ -}); │ │ │ │ │ + 'Overlays': "Gwiskadoù", │ │ │ │ │ │ │ │ │ │ -OpenLayers.Format.ArcXML.Response = OpenLayers.Class({ │ │ │ │ │ - initialize: function(params) { │ │ │ │ │ - var defaults = { │ │ │ │ │ - image: { │ │ │ │ │ - envelope: null, │ │ │ │ │ - output: '' │ │ │ │ │ - }, │ │ │ │ │ + 'Base Layer': "Gwiskad diazez", │ │ │ │ │ │ │ │ │ │ - features: { │ │ │ │ │ - featurecount: 0, │ │ │ │ │ - envelope: null, │ │ │ │ │ - feature: [] │ │ │ │ │ - }, │ │ │ │ │ + 'noFID': "N\'haller ket hizivaat un elfenn ma n\'eus ket a niverenn-anaout (FID) eviti.", │ │ │ │ │ │ │ │ │ │ - error: '' │ │ │ │ │ - }; │ │ │ │ │ + 'browserNotSupported': "N\'eo ket skoret an daskor vektorel gant ho merdeer. Setu aze an daskorerioù skoret evit ar poent :\n${renderers}", │ │ │ │ │ │ │ │ │ │ - return OpenLayers.Util.extend(this, defaults); │ │ │ │ │ - }, │ │ │ │ │ + 'minZoomLevelError': "Ne zleer implijout ar perzh minZoomLevel nemet evit gwiskadoù FixedZoomLevels-descendent. Ar fed ma wiria ar gwiskad WHS-se hag-eñ ez eus eus minZoomLevel zo un aspadenn gozh. Koulskoude n\'omp ket evit e ziverkañ kuit da derriñ arloadoù diazezet war OL a c\'hallfe bezañ stag outañ. Setu perak eo dispredet -- Lamet kuit e vo ar gwiriañ minZoomLevel a-is er stumm 3.0. Ober gant an arventennoù bihanañ/brasañ evel deskrivet amañ e plas : http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.ArcXML.Response" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/ArcIMS.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + 'commitSuccess': "Treuzgread WFS : MAT EO ${response}", │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + 'commitFailed': "Treuzgread WFS Transaction: C\'HWITET ${response}", │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ - * @requires OpenLayers/Format/ArcXML.js │ │ │ │ │ - * @requires OpenLayers/Request.js │ │ │ │ │ - */ │ │ │ │ │ + 'googleWarning': "N\'eus ket bet gallet kargañ ar gwiskad Google ent reizh.\x3cbr\x3e\x3cbr\x3eEvit en em zizober eus ar c\'hemenn-mañ, dibabit ur BaseLayer nevez en diuzer gwiskadoù er c\'horn dehoù el laez.\x3cbr\x3e\x3cbr\x3eSur a-walc\'h eo peogwir n\'eo ket bet ensoc\'het levraoueg Google Maps pe neuze ne glot ket an alc\'hwez API gant ho lec\'hienn.\x3cbr\x3e\x3cbr\x3eDiorroerien : Evit reizhañ an dra-se, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eclick here\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.ArcIMS │ │ │ │ │ - * Instances of OpenLayers.Layer.ArcIMS are used to display data from ESRI ArcIMS │ │ │ │ │ - * Mapping Services. Create a new ArcIMS layer with the <OpenLayers.Layer.ArcIMS> │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.ArcIMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + 'getLayerWarning': "N\'haller ket kargañ ar gwiskad ${layerType} ent reizh.\x3cbr\x3e\x3cbr\x3eEvit en em zizober eus ar c\'hemenn-mañ, dibabit ur BaseLayer nevez en diuzer gwiskadoù er c\'horn dehoù el laez.\x3cbr\x3e\x3cbr\x3eSur a-walc\'h eo peogwir n\'eo ket bet ensoc\'het mat al levraoueg ${layerLib}.\x3cbr\x3e\x3cbr\x3eDiorroerien : Evit gouzout penaos reizhañ an dra-se, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eclick here\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: DEFAULT_PARAMS │ │ │ │ │ - * {Object} Default query string parameters. │ │ │ │ │ - */ │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - ClientVersion: "9.2", │ │ │ │ │ - ServiceName: '' │ │ │ │ │ - }, │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Skeul = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: featureCoordSys │ │ │ │ │ - * {String} Code for feature coordinate system. Default is "4326". │ │ │ │ │ - */ │ │ │ │ │ - featureCoordSys: "4326", │ │ │ │ │ + 'W': "K", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: filterCoordSys │ │ │ │ │ - * {String} Code for filter coordinate system. Default is "4326". │ │ │ │ │ - */ │ │ │ │ │ - filterCoordSys: "4326", │ │ │ │ │ + 'E': "R", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: layers │ │ │ │ │ - * {Array} An array of objects with layer properties. │ │ │ │ │ - */ │ │ │ │ │ - layers: null, │ │ │ │ │ + 'N': "N", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: async │ │ │ │ │ - * {Boolean} Request images asynchronously. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - async: true, │ │ │ │ │ + 'S': "S", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: name │ │ │ │ │ - * {String} Layer name. Default is "ArcIMS". │ │ │ │ │ - */ │ │ │ │ │ - name: "ArcIMS", │ │ │ │ │ + 'reprojectDeprecated': "Emaoc\'h oc\'h implijout an dibarzh \'reproject\' war ar gwiskad ${layerName}. Dispredet eo an dibarzh-mañ : bet eo hag e talveze da ziskwel roadennoù war-c\'horre kartennoù diazez kenwerzhel, un dra hag a c\'haller ober bremañ gant an arc\'hwel dre skor banndres boullek Mercator. Muioc\'h a ditouroù a c\'haller da gaout war http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} The layer is a base layer. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + 'methodDeprecated': "Dispredet eo an daore-se ha tennet e vo kuit eus ar stumm 3.0. Grit gant ${newMethod} e plas." │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: DEFAULT_OPTIONS │ │ │ │ │ - * {Object} Default layers properties. │ │ │ │ │ - */ │ │ │ │ │ - DEFAULT_OPTIONS: { │ │ │ │ │ - tileSize: new OpenLayers.Size(512, 512), │ │ │ │ │ - featureCoordSys: "4326", │ │ │ │ │ - filterCoordSys: "4326", │ │ │ │ │ - layers: null, │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - async: true, │ │ │ │ │ - name: "ArcIMS" │ │ │ │ │ - }, │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/ia.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.ArcIMS │ │ │ │ │ - * Create a new ArcIMS layer object. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var arcims = new OpenLayers.Layer.ArcIMS( │ │ │ │ │ - * "Global Sample", │ │ │ │ │ - * "http://sample.avencia.com/servlet/com.esri.esrimap.Esrimap", │ │ │ │ │ - * { │ │ │ │ │ - * service: "OpenLayers_Sample", │ │ │ │ │ - * layers: [ │ │ │ │ │ - * // layers to manipulate │ │ │ │ │ - * {id: "1", visible: true} │ │ │ │ │ - * ] │ │ │ │ │ - * } │ │ │ │ │ - * ); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} A name for the layer │ │ │ │ │ - * url - {String} Base url for the ArcIMS server │ │ │ │ │ - * options - {Object} Optional object with properties to be set on the │ │ │ │ │ - * layer. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - McDutchie │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - this.tileSize = new OpenLayers.Size(512, 512); │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // parameters │ │ │ │ │ - this.params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - ServiceName: options.serviceName │ │ │ │ │ - }, │ │ │ │ │ - this.DEFAULT_PARAMS │ │ │ │ │ - ); │ │ │ │ │ - this.options = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options, this.DEFAULT_OPTIONS │ │ │ │ │ - ); │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["ia"] │ │ │ │ │ + * Dictionary for Interlingua. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["ia"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply( │ │ │ │ │ - this, [name, url, this.params, options] │ │ │ │ │ - ); │ │ │ │ │ + 'unhandledRequest': "Le responsa a un requesta non esseva maneate: ${statusText}", │ │ │ │ │ │ │ │ │ │ - //layer is transparent │ │ │ │ │ - if (this.transparent) { │ │ │ │ │ + 'Permalink': "Permaligamine", │ │ │ │ │ │ │ │ │ │ - // unless explicitly set in options, make layer an overlay │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - this.isBaseLayer = false; │ │ │ │ │ - } │ │ │ │ │ + 'Overlays': "Superpositiones", │ │ │ │ │ │ │ │ │ │ - // jpegs can never be transparent, so intelligently switch the │ │ │ │ │ - // format, depending on the browser's capabilities │ │ │ │ │ - if (this.format == "image/jpeg") { │ │ │ │ │ - this.format = OpenLayers.Util.alphaHack() ? "image/gif" : "image/png"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + 'Base Layer': "Strato de base", │ │ │ │ │ │ │ │ │ │ - // create an empty layer list if no layers specified in the options │ │ │ │ │ - if (this.options.layers === null) { │ │ │ │ │ - this.options.layers = []; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'noFID': "Non pote actualisar un elemento sin FID.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * Return an image url this layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox for the │ │ │ │ │ - * request. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the map image's url. │ │ │ │ │ - */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var url = ""; │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ + 'browserNotSupported': "Tu navigator non supporta le rendition de vectores. Le renditores actualmente supportate es:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - // create an arcxml request to generate the image │ │ │ │ │ - var axlReq = new OpenLayers.Format.ArcXML( │ │ │ │ │ - OpenLayers.Util.extend(this.options, { │ │ │ │ │ - requesttype: "image", │ │ │ │ │ - envelope: bounds.toArray(), │ │ │ │ │ - tileSize: this.tileSize │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ + 'minZoomLevelError': "Le proprietate minZoomLevel es solmente pro uso con le stratos descendente de FixedZoomLevels. Le facto que iste strato WFS verifica minZoomLevel es un reliquia del passato. Nonobstante, si nos lo remove immediatemente, nos pote rumper applicationes a base de OL que depende de illo. Ergo nos lo declara obsolete; le verification de minZoomLevel in basso essera removite in version 3.0. Per favor usa in su loco le configuration de resolutiones min/max como describite a: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - // create a synchronous ajax request to get an arcims image │ │ │ │ │ - var req = new OpenLayers.Request.POST({ │ │ │ │ │ - url: this.getFullRequestString(), │ │ │ │ │ - data: axlReq.write(), │ │ │ │ │ - async: false │ │ │ │ │ - }); │ │ │ │ │ + 'commitSuccess': "Transaction WFS: SUCCESSO ${response}", │ │ │ │ │ │ │ │ │ │ - // if the response exists │ │ │ │ │ - if (req != null) { │ │ │ │ │ - var doc = req.responseXML; │ │ │ │ │ + 'commitFailed': "Transaction WFS: FALLEVA ${response}", │ │ │ │ │ │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = req.responseText; │ │ │ │ │ - } │ │ │ │ │ + 'googleWarning': "Le strato Google non poteva esser cargate correctemente.\x3cbr\x3e\x3cbr\x3ePro disfacer te de iste message, selige un nove BaseLayer in le selector de strato in alto a dextra.\x3cbr\x3e\x3cbr\x3eMulto probabilemente, isto es proque le script del libreria de Google Maps non esseva includite o non contine le clave API correcte pro tu sito.\x3cbr\x3e\x3cbr\x3eDisveloppatores: Pro adjuta de corriger isto, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eclicca hic\x3c/a", │ │ │ │ │ │ │ │ │ │ - // create a new arcxml format to read the response │ │ │ │ │ - var axlResp = new OpenLayers.Format.ArcXML(); │ │ │ │ │ - var arcxml = axlResp.read(doc); │ │ │ │ │ - url = this.getUrlOrImage(arcxml.image.output); │ │ │ │ │ - } │ │ │ │ │ + 'getLayerWarning': "Le strato ${layerType} non poteva esser cargate correctemente.\x3cbr\x3e\x3cbr\x3ePro disfacer te de iste message, selige un nove BaseLayer in le selector de strato in alto a dextra.\x3cbr\x3e\x3cbr\x3eMulto probabilemente, isto es proque le script del libreria de ${layerLib} non esseva correctemente includite.\x3cbr\x3e\x3cbr\x3eDisveloppatores: Pro adjuta de corriger isto, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eclicca hic\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - return url; │ │ │ │ │ - }, │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Scala = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ + 'W': "W", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getURLasync │ │ │ │ │ - * Get an image url this layer asynchronously, and execute a callback │ │ │ │ │ - * when the image url is generated. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox for the │ │ │ │ │ - * request. │ │ │ │ │ - * callback - {Function} Function to call when image url is retrieved. │ │ │ │ │ - * scope - {Object} The scope of the callback method. │ │ │ │ │ - */ │ │ │ │ │ - getURLasync: function(bounds, callback, scope) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ + 'E': "E", │ │ │ │ │ │ │ │ │ │ - // create an arcxml request to generate the image │ │ │ │ │ - var axlReq = new OpenLayers.Format.ArcXML( │ │ │ │ │ - OpenLayers.Util.extend(this.options, { │ │ │ │ │ - requesttype: "image", │ │ │ │ │ - envelope: bounds.toArray(), │ │ │ │ │ - tileSize: this.tileSize │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ + 'N': "N", │ │ │ │ │ │ │ │ │ │ - // create an asynchronous ajax request to get an arcims image │ │ │ │ │ - OpenLayers.Request.POST({ │ │ │ │ │ - url: this.getFullRequestString(), │ │ │ │ │ - async: true, │ │ │ │ │ - data: axlReq.write(), │ │ │ │ │ - callback: function(req) { │ │ │ │ │ - // process the response from ArcIMS, and call the callback function │ │ │ │ │ - // to set the image URL │ │ │ │ │ - var doc = req.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = req.responseText; │ │ │ │ │ - } │ │ │ │ │ + 'S': "S", │ │ │ │ │ │ │ │ │ │ - // create a new arcxml format to read the response │ │ │ │ │ - var axlResp = new OpenLayers.Format.ArcXML(); │ │ │ │ │ - var arcxml = axlResp.read(doc); │ │ │ │ │ + 'reprojectDeprecated': "Tu usa le option \'reproject\' in le strato ${layerName} layer. Iste option es obsolescente: illo esseva pro poter monstrar datos super cartas de base commercial, ma iste functionalitate pote ora esser attingite con le uso de Spherical Mercator. Ulterior information es disponibile a http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ │ │ │ │ │ - callback.call(scope, this.getUrlOrImage(arcxml.image.output)); │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + 'methodDeprecated': "Iste methodo ha essite declarate obsolescente e essera removite in version 3.0. Per favor usa ${newMethod} in su loco." │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getUrlOrImage │ │ │ │ │ - * Extract a url or image from the ArcXML image output. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * output - {Object} The image.output property of the object returned from │ │ │ │ │ - * the ArcXML format read method. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A URL for an image (potentially with the data protocol). │ │ │ │ │ - */ │ │ │ │ │ - getUrlOrImage: function(output) { │ │ │ │ │ - var ret = ""; │ │ │ │ │ - if (output.url) { │ │ │ │ │ - // If the image response output url is a string, then the image │ │ │ │ │ - // data is not inline. │ │ │ │ │ - ret = output.url; │ │ │ │ │ - } else if (output.data) { │ │ │ │ │ - // The image data is inline and base64 encoded, create a data │ │ │ │ │ - // url for the image. This will only work for small images, │ │ │ │ │ - // due to browser url length limits. │ │ │ │ │ - ret = "data:image/" + output.type + │ │ │ │ │ - ";base64," + output.data; │ │ │ │ │ - } │ │ │ │ │ - return ret; │ │ │ │ │ - }, │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/fr.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setLayerQuery │ │ │ │ │ - * Set the query definition on this layer. Query definitions are used to │ │ │ │ │ - * render parts of the spatial data in an image, and can be used to │ │ │ │ │ - * filter features or layers in the ArcIMS service. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} The ArcIMS layer ID. │ │ │ │ │ - * querydef - {Object} The query definition to apply to this layer. │ │ │ │ │ - */ │ │ │ │ │ - setLayerQuery: function(id, querydef) { │ │ │ │ │ - // find the matching layer, if it exists │ │ │ │ │ - for (var lyr = 0; lyr < this.options.layers.length; lyr++) { │ │ │ │ │ - if (id == this.options.layers[lyr].id) { │ │ │ │ │ - // replace this layer definition │ │ │ │ │ - this.options.layers[lyr].query = querydef; │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Damouns │ │ │ │ │ + * - IAlex │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // no layer found, create a new definition │ │ │ │ │ - this.options.layers.push({ │ │ │ │ │ - id: id, │ │ │ │ │ - visible: true, │ │ │ │ │ - query: querydef │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getFeatureInfo │ │ │ │ │ - * Get feature information from ArcIMS. Using the applied geometry, apply │ │ │ │ │ - * the options to the query (buffer, area/envelope intersection), and │ │ │ │ │ - * query the ArcIMS service. │ │ │ │ │ - * │ │ │ │ │ - * A note about accuracy: │ │ │ │ │ - * ArcIMS interprets the accuracy attribute in feature requests to be │ │ │ │ │ - * something like the 'modulus' operator on feature coordinates, │ │ │ │ │ - * applied to the database geometry of the feature. It doesn't round, │ │ │ │ │ - * so your feature coordinates may be up to (1 x accuracy) offset from │ │ │ │ │ - * the actual feature coordinates. If the accuracy of the layer is not │ │ │ │ │ - * specified, the accuracy will be computed to be approximately 1 │ │ │ │ │ - * feature coordinate per screen pixel. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.LonLat>} or {<OpenLayers.Geometry.Polygon>} The │ │ │ │ │ - * geometry to use when making the query. This should be a closed │ │ │ │ │ - * polygon for behavior approximating a free selection. │ │ │ │ │ - * layer - {Object} The ArcIMS layer definition. This is an anonymous object │ │ │ │ │ - * that looks like: │ │ │ │ │ - * (code) │ │ │ │ │ - * { │ │ │ │ │ - * id: "ArcXML layer ID", // the ArcXML layer ID │ │ │ │ │ - * query: { │ │ │ │ │ - * where: "STATE = 'PA'", // the where clause of the query │ │ │ │ │ - * accuracy: 100 // the accuracy of the returned feature │ │ │ │ │ - * } │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ - * options - {Object} Object with non-default properties to set on the layer. │ │ │ │ │ - * Supported properties are buffer, callback, scope, and any other │ │ │ │ │ - * properties applicable to the ArcXML format. Set the 'callback' and │ │ │ │ │ - * 'scope' for an object and function to recieve the parsed features │ │ │ │ │ - * from ArcIMS. │ │ │ │ │ - */ │ │ │ │ │ - getFeatureInfo: function(geometry, layer, options) { │ │ │ │ │ - // set the buffer to 1 unit (dd/m/ft?) by default │ │ │ │ │ - var buffer = options.buffer || 1; │ │ │ │ │ - // empty callback by default │ │ │ │ │ - var callback = options.callback || function() {}; │ │ │ │ │ - // default scope is window (global) │ │ │ │ │ - var scope = options.scope || window; │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["fr"] │ │ │ │ │ + * Dictionary for Français. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["fr"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - // apply these option to the request options │ │ │ │ │ - var requestOptions = {}; │ │ │ │ │ - OpenLayers.Util.extend(requestOptions, this.options); │ │ │ │ │ + 'unhandledRequest': "Requête non gérée, retournant ${statusText}", │ │ │ │ │ │ │ │ │ │ - // this is a feature request │ │ │ │ │ - requestOptions.requesttype = "feature"; │ │ │ │ │ + 'Permalink': "Permalien", │ │ │ │ │ │ │ │ │ │ - if (geometry instanceof OpenLayers.LonLat) { │ │ │ │ │ - // create an envelope if the geometry is really a lon/lat │ │ │ │ │ - requestOptions.polygon = null; │ │ │ │ │ - requestOptions.envelope = [ │ │ │ │ │ - geometry.lon - buffer, │ │ │ │ │ - geometry.lat - buffer, │ │ │ │ │ - geometry.lon + buffer, │ │ │ │ │ - geometry.lat + buffer │ │ │ │ │ - ]; │ │ │ │ │ - } else if (geometry instanceof OpenLayers.Geometry.Polygon) { │ │ │ │ │ - // use the polygon assigned, and empty the envelope │ │ │ │ │ - requestOptions.envelope = null; │ │ │ │ │ - requestOptions.polygon = geometry; │ │ │ │ │ - } │ │ │ │ │ + 'Overlays': "Calques", │ │ │ │ │ │ │ │ │ │ - // create an arcxml request to get feature requests │ │ │ │ │ - var arcxml = new OpenLayers.Format.ArcXML(requestOptions); │ │ │ │ │ + 'Base Layer': "Calque de base", │ │ │ │ │ │ │ │ │ │ - // apply any get feature options to the arcxml request │ │ │ │ │ - OpenLayers.Util.extend(arcxml.request.get_feature, options); │ │ │ │ │ + 'noFID': "Impossible de mettre à jour un objet sans identifiant (fid).", │ │ │ │ │ │ │ │ │ │ - arcxml.request.get_feature.layer = layer.id; │ │ │ │ │ - if (typeof layer.query.accuracy == "number") { │ │ │ │ │ - // set the accuracy if it was specified │ │ │ │ │ - arcxml.request.get_feature.query.accuracy = layer.query.accuracy; │ │ │ │ │ - } else { │ │ │ │ │ - // guess that the accuracy is 1 per screen pixel │ │ │ │ │ - var mapCenter = this.map.getCenter(); │ │ │ │ │ - var viewPx = this.map.getViewPortPxFromLonLat(mapCenter); │ │ │ │ │ - viewPx.x++; │ │ │ │ │ - var mapOffCenter = this.map.getLonLatFromPixel(viewPx); │ │ │ │ │ - arcxml.request.get_feature.query.accuracy = mapOffCenter.lon - mapCenter.lon; │ │ │ │ │ - } │ │ │ │ │ + 'browserNotSupported': "Votre navigateur ne supporte pas le rendu vectoriel. Les renderers actuellement supportés sont : \n${renderers}", │ │ │ │ │ │ │ │ │ │ - // set the get_feature query to be the same as the layer passed in │ │ │ │ │ - arcxml.request.get_feature.query.where = layer.query.where; │ │ │ │ │ + 'minZoomLevelError': "La propriété minZoomLevel doit seulement être utilisée pour des couches FixedZoomLevels-descendent. Le fait que cette couche WFS vérifie la présence de minZoomLevel est une relique du passé. Nous ne pouvons toutefois la supprimer sans casser des applications qui pourraient en dépendre. C\'est pourquoi nous la déprécions -- la vérification du minZoomLevel sera supprimée en version 3.0. A la place, merci d\'utiliser les paramètres de résolutions min/max tel que décrit sur : http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - // use area_intersection │ │ │ │ │ - arcxml.request.get_feature.query.spatialfilter.relation = "area_intersection"; │ │ │ │ │ + 'commitSuccess': "Transaction WFS : SUCCES ${response}", │ │ │ │ │ │ │ │ │ │ - // create a new asynchronous request to get the feature info │ │ │ │ │ - OpenLayers.Request.POST({ │ │ │ │ │ - url: this.getFullRequestString({ │ │ │ │ │ - 'CustomService': 'Query' │ │ │ │ │ - }), │ │ │ │ │ - data: arcxml.write(), │ │ │ │ │ - callback: function(request) { │ │ │ │ │ - // parse the arcxml response │ │ │ │ │ - var response = arcxml.parseResponse(request.responseText); │ │ │ │ │ + 'commitFailed': "Transaction WFS : ECHEC ${response}", │ │ │ │ │ │ │ │ │ │ - if (!arcxml.iserror()) { │ │ │ │ │ - // if the arcxml is not an error, call the callback with the features parsed │ │ │ │ │ - callback.call(scope, response.features); │ │ │ │ │ - } else { │ │ │ │ │ - // if the arcxml is an error, return null features selected │ │ │ │ │ - callback.call(scope, null); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + 'googleWarning': "La couche Google n\'a pas été en mesure de se charger correctement.\x3cbr\x3e\x3cbr\x3ePour supprimer ce message, choisissez une nouvelle BaseLayer dans le sélecteur de couche en haut à droite.\x3cbr\x3e\x3cbr\x3eCela est possiblement causé par la non-inclusion de la librairie Google Maps, ou alors parce que la clé de l\'API ne correspond pas à votre site.\x3cbr\x3e\x3cbr\x3eDéveloppeurs : pour savoir comment corriger ceci, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3ecliquez ici\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.ArcIMS>} An exact clone of this layer │ │ │ │ │ - */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ + 'getLayerWarning': "La couche ${layerType} n\'est pas en mesure de se charger correctement.\x3cbr\x3e\x3cbr\x3ePour supprimer ce message, choisissez une nouvelle BaseLayer dans le sélecteur de couche en haut à droite.\x3cbr\x3e\x3cbr\x3eCela est possiblement causé par la non-inclusion de la librairie ${layerLib}.\x3cbr\x3e\x3cbr\x3eDéveloppeurs : pour savoir comment corriger ceci, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3ecliquez ici\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.ArcIMS(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Echelle ~ 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + 'W': "O", │ │ │ │ │ │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ + 'E': "E", │ │ │ │ │ │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ + 'N': "N", │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.ArcIMS" │ │ │ │ │ + 'S': "S", │ │ │ │ │ + │ │ │ │ │ + 'reprojectDeprecated': "Vous utilisez l\'option \'reproject\' sur la couche ${layerName}. Cette option est dépréciée : Son usage permettait d\'afficher des données au dessus de couches raster commerciales.Cette fonctionalité est maintenant supportée en utilisant le support de la projection Mercator Sphérique. Plus d\'information est disponible sur http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + │ │ │ │ │ + 'methodDeprecated': "Cette méthode est dépréciée, et sera supprimée à la version 3.0. Merci d\'utiliser ${newMethod} à la place." │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/OSM.js │ │ │ │ │ + OpenLayers/Lang/pl.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ +/* Translators: │ │ │ │ │ + * - Arkadiusz Grabka │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.OSM │ │ │ │ │ - * This layer allows accessing OpenStreetMap tiles. By default the OpenStreetMap │ │ │ │ │ - * hosted tile.openstreetmap.org Mapnik tileset is used. If you wish to use │ │ │ │ │ - * a different layer instead, you need to provide a different │ │ │ │ │ - * URL to the constructor. Here's an example for using OpenCycleMap: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * new OpenLayers.Layer.OSM("OpenCycleMap", │ │ │ │ │ - * ["http://a.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ - * "http://b.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ - * "http://c.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png"]); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.XYZ> │ │ │ │ │ + * Namespace: OpenLayers.Lang["pl"] │ │ │ │ │ + * Dictionary for Polish. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.OSM = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ +OpenLayers.Lang["pl"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: name │ │ │ │ │ - * {String} The layer name. Defaults to "OpenStreetMap" if the first │ │ │ │ │ - * argument to the constructor is null or undefined. │ │ │ │ │ - */ │ │ │ │ │ - name: "OpenStreetMap", │ │ │ │ │ + 'unhandledRequest': "Nieobsługiwane żądanie zwróciło ${statusText}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String} The tileset URL scheme. Defaults to │ │ │ │ │ - * : http://[a|b|c].tile.openstreetmap.org/${z}/${x}/${y}.png │ │ │ │ │ - * (the official OSM tileset) if the second argument to the constructor │ │ │ │ │ - * is null or undefined. To use another tileset you can have something │ │ │ │ │ - * like this: │ │ │ │ │ - * (code) │ │ │ │ │ - * new OpenLayers.Layer.OSM("OpenCycleMap", │ │ │ │ │ - * ["http://a.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ - * "http://b.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ - * "http://c.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png"]); │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ - url: [ │ │ │ │ │ - 'http://a.tile.openstreetmap.org/${z}/${x}/${y}.png', │ │ │ │ │ - 'http://b.tile.openstreetmap.org/${z}/${x}/${y}.png', │ │ │ │ │ - 'http://c.tile.openstreetmap.org/${z}/${x}/${y}.png' │ │ │ │ │ - ], │ │ │ │ │ + 'Permalink': "Permalink", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: attribution │ │ │ │ │ - * {String} The layer attribution. │ │ │ │ │ - */ │ │ │ │ │ - attribution: "© <a href='http://www.openstreetmap.org/copyright'>OpenStreetMap</a> contributors", │ │ │ │ │ + 'Overlays': "Nakładki", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: sphericalMercator │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - sphericalMercator: true, │ │ │ │ │ + 'Base Layer': "Warstwa podstawowa", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: wrapDateLine │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - wrapDateLine: true, │ │ │ │ │ + 'noFID': "Nie można zaktualizować funkcji, dla których nie ma FID.", │ │ │ │ │ │ │ │ │ │ - /** APIProperty: tileOptions │ │ │ │ │ - * {Object} optional configuration options for <OpenLayers.Tile> instances │ │ │ │ │ - * created by this Layer. Default is │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * {crossOriginKeyword: 'anonymous'} │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * When using OSM tilesets other than the default ones, it may be │ │ │ │ │ - * necessary to set this to │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * {crossOriginKeyword: null} │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * if the server does not send Access-Control-Allow-Origin headers. │ │ │ │ │ - */ │ │ │ │ │ - tileOptions: null, │ │ │ │ │ + 'browserNotSupported': "Twoja przeglądarka nie obsługuje renderowania wektorów. Obecnie obsługiwane renderowanie to:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.OSM │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} The layer name. │ │ │ │ │ - * url - {String} The tileset URL scheme. │ │ │ │ │ - * options - {Object} Configuration options for the layer. Any inherited │ │ │ │ │ - * layer option can be set in this object (e.g. │ │ │ │ │ - * <OpenLayers.Layer.Grid.buffer>). │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - crossOriginKeyword: 'anonymous' │ │ │ │ │ - }, this.options && this.options.tileOptions); │ │ │ │ │ - }, │ │ │ │ │ + // console message │ │ │ │ │ + 'minZoomLevelError': "Właściwość minZoomLevel jest przeznaczona tylko do użytku " + │ │ │ │ │ + "z warstwami FixedZoomLevels-descendent." + │ │ │ │ │ + "Warstwa wfs, która sprawdza minZoomLevel jest reliktem przeszłości." + │ │ │ │ │ + "Nie możemy jej jednak usunąc bez mozliwości łamania OL aplikacji, " + │ │ │ │ │ + "które mogą być od niej zależne. " + │ │ │ │ │ + "Dlatego jesteśmy za deprecjację -- minZoomLevel " + │ │ │ │ │ + "zostanie usunięta w wersji 3.0. W zamian prosze użyj " + │ │ │ │ │ + "min/max rozdzielczości w sposób opisany tutaj: " + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.OSM( │ │ │ │ │ - this.name, this.url, this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ + 'commitSuccess': "Transakcja WFS: SUKCES ${response}", │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.OSM" │ │ │ │ │ + 'commitFailed': "Transakcja WFS: FAILED ${response}", │ │ │ │ │ + │ │ │ │ │ + 'googleWarning': "Warstwa Google nie był w stanie załadować się poprawnie.<br><br>" + │ │ │ │ │ + "Aby pozbyć się tej wiadomości, wybierz nową Warstwe podstawową " + │ │ │ │ │ + "w przełączniku warstw w górnym prawym rogu mapy.<br><br>" + │ │ │ │ │ + "Najprawdopodobniej jest to spowodowane tym, że biblioteka Google Maps " + │ │ │ │ │ + "nie jest załadowana, lub nie zawiera poprawnego klucza do API dla twojej strony<br><br>" + │ │ │ │ │ + "Programisto: Aby uzyskać pomoc , " + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ + "target='_blank'>kliknij tutaj</a>", │ │ │ │ │ + │ │ │ │ │ + 'getLayerWarning': "Warstwa ${layerType} nie mogła zostać załadowana poprawnie.<br><br>" + │ │ │ │ │ + "Aby pozbyć się tej wiadomości, wybierz nową Warstwe podstawową " + │ │ │ │ │ + "w przełączniku warstw w górnym prawym rogu mapy.<br><br>" + │ │ │ │ │ + "Najprawdopodobniej jest to spowodowane tym, że biblioteka ${layerLib} " + │ │ │ │ │ + "nie jest załadowana, lub może(o ile biblioteka tego wymaga) " + │ │ │ │ │ + "byc potrzebny klucza do API dla twojej strony<br><br>" + │ │ │ │ │ + "Programisto: Aby uzyskać pomoc , " + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ + "target='_blank'>kliknij tutaj</a>", │ │ │ │ │ + │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Skala = 1 : ${scaleDenom}", │ │ │ │ │ + │ │ │ │ │ + //labels for the graticule control │ │ │ │ │ + 'W': 'ZACH', │ │ │ │ │ + 'E': 'WSCH', │ │ │ │ │ + 'N': 'PN', │ │ │ │ │ + 'S': 'PD', │ │ │ │ │ + 'Graticule': 'Siatka', │ │ │ │ │ + │ │ │ │ │ + // console message │ │ │ │ │ + 'reprojectDeprecated': "w warstwie ${layerName} używasz opcji 'reproject'. " + │ │ │ │ │ + "Ta opcja jest przestarzała: " + │ │ │ │ │ + "jej zastosowanie został zaprojektowany, aby wspierać wyświetlania danych przez komercyjne mapy, " + │ │ │ │ │ + "jednak obecnie ta funkcjonalność powinien zostać osiągnięty za pomocą Spherical Mercator " + │ │ │ │ │ + "its use was designed to support displaying data over commercial. Więcje informacji na ten temat możesz znaleźć na stronie " + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + │ │ │ │ │ + // console message │ │ │ │ │ + 'methodDeprecated': "Ta metoda jest przestarzała i będzie usunięta od wersji 3.0. " + │ │ │ │ │ + "W zamian użyj ${newMethod}." │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/MapGuide.js │ │ │ │ │ + OpenLayers/Lang/de.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Grille chompa │ │ │ │ │ + * - Nikiwaibel │ │ │ │ │ + * - Umherirrender │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.MapGuide │ │ │ │ │ - * Instances of OpenLayers.Layer.MapGuide are used to display │ │ │ │ │ - * data from a MapGuide OS instance. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ + * Namespace: OpenLayers.Lang["de"] │ │ │ │ │ + * Dictionary for Deutsch. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.MapGuide = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ +OpenLayers.Lang["de"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} Treat this layer as a base layer. Default is true. │ │ │ │ │ - **/ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + 'unhandledRequest': "Unbehandelte Anfragerückmeldung ${statusText}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: useHttpTile │ │ │ │ │ - * {Boolean} use a tile cache exposed directly via a webserver rather than the │ │ │ │ │ - * via mapguide server. This does require extra configuration on the Mapguide Server, │ │ │ │ │ - * and will only work when singleTile is false. The url for the layer must be set to the │ │ │ │ │ - * webserver path rather than the Mapguide mapagent. │ │ │ │ │ - * See http://trac.osgeo.org/mapguide/wiki/CodeSamples/Tiles/ServingTilesViaHttp │ │ │ │ │ - **/ │ │ │ │ │ - useHttpTile: false, │ │ │ │ │ + 'Permalink': "Permalink", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: singleTile │ │ │ │ │ - * {Boolean} use tile server or request single tile image. │ │ │ │ │ - **/ │ │ │ │ │ - singleTile: false, │ │ │ │ │ + 'Overlays': "Overlays", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: useOverlay │ │ │ │ │ - * {Boolean} flag to indicate if the layer should be retrieved using │ │ │ │ │ - * GETMAPIMAGE (default) or using GETDYNAMICOVERLAY requests. │ │ │ │ │ - **/ │ │ │ │ │ - useOverlay: false, │ │ │ │ │ + 'Base Layer': "Grundkarte", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: useAsyncOverlay │ │ │ │ │ - * {Boolean} indicates if the MapGuide site supports the asynchronous │ │ │ │ │ - * GETDYNAMICOVERLAY requests which is available in MapGuide Enterprise 2010 │ │ │ │ │ - * and MapGuide Open Source v2.0.3 or higher. The newer versions of MG │ │ │ │ │ - * is called asynchronously, allows selections to be drawn separately from │ │ │ │ │ - * the map and offers styling options. │ │ │ │ │ - * │ │ │ │ │ - * With older versions of MapGuide, set useAsyncOverlay=false. Note that in │ │ │ │ │ - * this case a synchronous AJAX call is issued and the mapname and session │ │ │ │ │ - * parameters must be used to initialize the layer, not the mapdefinition │ │ │ │ │ - * parameter. Also note that this will issue a synchronous AJAX request │ │ │ │ │ - * before the image request can be issued so the users browser may lock │ │ │ │ │ - * up if the MG Web tier does not respond in a timely fashion. │ │ │ │ │ - **/ │ │ │ │ │ - useAsyncOverlay: true, │ │ │ │ │ + 'noFID': "Ein Feature, für das keine FID existiert, kann nicht aktualisiert werden.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: TILE_PARAMS │ │ │ │ │ - * {Object} Hashtable of default parameter key/value pairs for tiled layer │ │ │ │ │ - */ │ │ │ │ │ - TILE_PARAMS: { │ │ │ │ │ - operation: 'GETTILEIMAGE', │ │ │ │ │ - version: '1.2.0' │ │ │ │ │ - }, │ │ │ │ │ + 'browserNotSupported': "Ihr Browser unterstützt keine Vektordarstellung. Aktuell unterstützte Renderer:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: SINGLE_TILE_PARAMS │ │ │ │ │ - * {Object} Hashtable of default parameter key/value pairs for untiled layer │ │ │ │ │ - */ │ │ │ │ │ - SINGLE_TILE_PARAMS: { │ │ │ │ │ - operation: 'GETMAPIMAGE', │ │ │ │ │ - format: 'PNG', │ │ │ │ │ - locale: 'en', │ │ │ │ │ - clip: '1', │ │ │ │ │ - version: '1.0.0' │ │ │ │ │ - }, │ │ │ │ │ + 'minZoomLevelError': "Die \x3ccode\x3eminZoomLevel\x3c/code\x3e-Eigenschaft ist nur für die Verwendung mit \x3ccode\x3eFixedZoomLevels\x3c/code\x3e-untergeordneten Layers vorgesehen. Das dieser \x3ctt\x3ewfs\x3c/tt\x3e-Layer die \x3ccode\x3eminZoomLevel\x3c/code\x3e-Eigenschaft überprüft ist ein Relikt der Vergangenheit. Wir können diese Überprüfung nicht entfernen, ohne das OL basierende Applikationen nicht mehr funktionieren. Daher markieren wir es als veraltet - die \x3ccode\x3eminZoomLevel\x3c/code\x3e-Überprüfung wird in Version 3.0 entfernt werden. Bitte verwenden Sie stattdessen die Min-/Max-Lösung, wie sie unter http://trac.openlayers.org/wiki/SettingZoomLevels beschrieben ist.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: OVERLAY_PARAMS │ │ │ │ │ - * {Object} Hashtable of default parameter key/value pairs for untiled layer │ │ │ │ │ - */ │ │ │ │ │ - OVERLAY_PARAMS: { │ │ │ │ │ - operation: 'GETDYNAMICMAPOVERLAYIMAGE', │ │ │ │ │ - format: 'PNG', │ │ │ │ │ - locale: 'en', │ │ │ │ │ - clip: '1', │ │ │ │ │ - version: '2.0.0' │ │ │ │ │ - }, │ │ │ │ │ + 'commitSuccess': "WFS-Transaktion: Erfolgreich ${response}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: FOLDER_PARAMS │ │ │ │ │ - * {Object} Hashtable of parameter key/value pairs which describe │ │ │ │ │ - * the folder structure for tiles as configured in the mapguide │ │ │ │ │ - * serverconfig.ini section [TileServiceProperties] │ │ │ │ │ - */ │ │ │ │ │ - FOLDER_PARAMS: { │ │ │ │ │ - tileColumnsPerFolder: 30, │ │ │ │ │ - tileRowsPerFolder: 30, │ │ │ │ │ - format: 'png', │ │ │ │ │ - querystring: null │ │ │ │ │ - }, │ │ │ │ │ + 'commitFailed': "WFS-Transaktion: Fehlgeschlagen ${response}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultSize │ │ │ │ │ - * {<OpenLayers.Size>} Tile size as produced by MapGuide server │ │ │ │ │ - **/ │ │ │ │ │ - defaultSize: new OpenLayers.Size(300, 300), │ │ │ │ │ + 'googleWarning': "Der Google-Layer konnte nicht korrekt geladen werden.\x3cbr\x3e\x3cbr\x3eUm diese Meldung nicht mehr zu erhalten, wählen Sie einen anderen Hintergrundlayer aus dem LayerSwitcher in der rechten oberen Ecke.\x3cbr\x3e\x3cbr\x3eSehr wahrscheinlich tritt dieser Fehler auf, weil das Skript der Google-Maps-Bibliothek nicht eingebunden wurde oder keinen gültigen API-Schlüssel für Ihre URL enthält.\x3cbr\x3e\x3cbr\x3eEntwickler: Besuche \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3edas Wiki\x3c/a\x3e für Hilfe zum korrekten Einbinden des Google-Layers", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: tileOriginCorner │ │ │ │ │ - * {String} MapGuide tile server uses top-left as tile origin │ │ │ │ │ - **/ │ │ │ │ │ - tileOriginCorner: "tl", │ │ │ │ │ + 'getLayerWarning': "Der ${layerType}-Layer konnte nicht korrekt geladen werden.\x3cbr\x3e\x3cbr\x3eUm diese Meldung nicht mehr zu erhalten, wählen Sie einen anderen Hintergrundlayer aus dem LayerSwitcher in der rechten oberen Ecke.\x3cbr\x3e\x3cbr\x3eSehr wahrscheinlich tritt dieser Fehler auf, weil das Skript der \'${layerLib}\'-Bibliothek nicht eingebunden wurde.\x3cbr\x3e\x3cbr\x3eEntwickler: Besuche \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3edas Wiki\x3c/a\x3e für Hilfe zum korrekten Einbinden von Layern", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.MapGuide │ │ │ │ │ - * Create a new Mapguide layer, either tiled or untiled. │ │ │ │ │ - * │ │ │ │ │ - * For tiled layers, the 'groupName' and 'mapDefinition' values │ │ │ │ │ - * must be specified as parameters in the constructor. │ │ │ │ │ - * │ │ │ │ │ - * For untiled base layers, specify either combination of 'mapName' and │ │ │ │ │ - * 'session', or 'mapDefinition' and 'locale'. │ │ │ │ │ - * │ │ │ │ │ - * For older versions of MapGuide and overlay layers, set useAsyncOverlay │ │ │ │ │ - * to false and in this case mapName and session are required parameters │ │ │ │ │ - * for the constructor. │ │ │ │ │ - * │ │ │ │ │ - * NOTE: MapGuide OS uses a DPI value and degrees to meters conversion │ │ │ │ │ - * factor that are different than the defaults used in OpenLayers, │ │ │ │ │ - * so these must be adjusted accordingly in your application. │ │ │ │ │ - * See the MapGuide example for how to set these values for MGOS. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} Name of the layer displayed in the interface │ │ │ │ │ - * url - {String} Location of the MapGuide mapagent executable │ │ │ │ │ - * (e.g. http://localhost:8008/mapguide/mapagent/mapagent.fcgi) │ │ │ │ │ - * params - {Object} hashtable of additional parameters to use. Some │ │ │ │ │ - * parameters may require additional code on the server. The ones that │ │ │ │ │ - * you may want to use are: │ │ │ │ │ - * - mapDefinition - {String} The MapGuide resource definition │ │ │ │ │ - * (e.g. Library://Samples/Gmap/Maps/gmapTiled.MapDefinition) │ │ │ │ │ - * - locale - Locale setting │ │ │ │ │ - * (for untiled overlays layers only) │ │ │ │ │ - * - mapName - {String} Name of the map as stored in the MapGuide session. │ │ │ │ │ - * (for untiled layers with a session parameter only) │ │ │ │ │ - * - session - { String} MapGuide session ID │ │ │ │ │ - * (for untiled overlays layers only) │ │ │ │ │ - * - basemaplayergroupname - {String} GroupName for tiled MapGuide layers only │ │ │ │ │ - * - format - Image format to be returned (for untiled overlay layers only) │ │ │ │ │ - * - showLayers - {String} A comma separated list of GUID's for the │ │ │ │ │ - * layers to display eg: 'cvc-xcv34,453-345-345sdf'. │ │ │ │ │ - * - hideLayers - {String} A comma separated list of GUID's for the │ │ │ │ │ - * layers to hide eg: 'cvc-xcv34,453-345-345sdf'. │ │ │ │ │ - * - showGroups - {String} A comma separated list of GUID's for the │ │ │ │ │ - * groups to display eg: 'cvc-xcv34,453-345-345sdf'. │ │ │ │ │ - * - hideGroups - {String} A comma separated list of GUID's for the │ │ │ │ │ - * groups to hide eg: 'cvc-xcv34,453-345-345sdf' │ │ │ │ │ - * - selectionXml - {String} A selection xml string Some server plumbing │ │ │ │ │ - * is required to read such a value. │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer; │ │ │ │ │ - * will vary depending if tiled or untiled maps are being requested │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Maßstab = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ + 'W': "W", │ │ │ │ │ │ │ │ │ │ - // unless explicitly set in options, if the layer is transparent, │ │ │ │ │ - // it will be an overlay │ │ │ │ │ - if (options == null || options.isBaseLayer == null) { │ │ │ │ │ - this.isBaseLayer = ((this.transparent != "true") && │ │ │ │ │ - (this.transparent != true)); │ │ │ │ │ - } │ │ │ │ │ + 'E': "O", │ │ │ │ │ │ │ │ │ │ - if (options && options.useOverlay != null) { │ │ │ │ │ - this.useOverlay = options.useOverlay; │ │ │ │ │ - } │ │ │ │ │ + 'N': "N", │ │ │ │ │ │ │ │ │ │ - //initialize for untiled layers │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - if (this.useOverlay) { │ │ │ │ │ - OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, │ │ │ │ │ - this.OVERLAY_PARAMS │ │ │ │ │ - ); │ │ │ │ │ - if (!this.useAsyncOverlay) { │ │ │ │ │ - this.params.version = "1.0.0"; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, │ │ │ │ │ - this.SINGLE_TILE_PARAMS │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - //initialize for tiled layers │ │ │ │ │ - if (this.useHttpTile) { │ │ │ │ │ - OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, │ │ │ │ │ - this.FOLDER_PARAMS │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, │ │ │ │ │ - this.TILE_PARAMS │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - this.setTileSize(this.defaultSize); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'S': "S", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.MapGuide>} An exact clone of this layer │ │ │ │ │ - */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.MapGuide(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.params, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + 'reprojectDeprecated': "Sie verwenden die „Reproject“-Option des Layers ${layerName}. Diese Option ist veraltet: Sie wurde entwickelt um die Anzeige von Daten auf kommerziellen Basiskarten zu unterstützen, aber diese Funktion sollte jetzt durch Unterstützung der „Spherical Mercator“ erreicht werden. Weitere Informationen sind unter http://trac.openlayers.org/wiki/SphericalMercator verfügbar.", │ │ │ │ │ │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ + 'methodDeprecated': "Die Methode ist veraltet und wird in 3.0 entfernt. Bitte verwende stattdessen ${newMethod}." │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * Return a query string for this layer │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox │ │ │ │ │ - * for the request │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also │ │ │ │ │ - * the passed-in bounds and appropriate tile size specified │ │ │ │ │ - * as parameters. │ │ │ │ │ - */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var url; │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var mapSize = this.map.getSize(); │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/gl.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - //set up the call for GETMAPIMAGE or GETDYNAMICMAPOVERLAY with │ │ │ │ │ - //dynamic map parameters │ │ │ │ │ - var params = { │ │ │ │ │ - setdisplaydpi: OpenLayers.DOTS_PER_INCH, │ │ │ │ │ - setdisplayheight: mapSize.h * this.ratio, │ │ │ │ │ - setdisplaywidth: mapSize.w * this.ratio, │ │ │ │ │ - setviewcenterx: center.lon, │ │ │ │ │ - setviewcentery: center.lat, │ │ │ │ │ - setviewscale: this.map.getScale() │ │ │ │ │ - }; │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Toliño │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (this.useOverlay && !this.useAsyncOverlay) { │ │ │ │ │ - //first we need to call GETVISIBLEMAPEXTENT to set the extent │ │ │ │ │ - var getVisParams = {}; │ │ │ │ │ - getVisParams = OpenLayers.Util.extend(getVisParams, params); │ │ │ │ │ - getVisParams.operation = "GETVISIBLEMAPEXTENT"; │ │ │ │ │ - getVisParams.version = "1.0.0"; │ │ │ │ │ - getVisParams.session = this.params.session; │ │ │ │ │ - getVisParams.mapName = this.params.mapName; │ │ │ │ │ - getVisParams.format = 'text/xml'; │ │ │ │ │ - url = this.getFullRequestString(getVisParams); │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - OpenLayers.Request.GET({ │ │ │ │ │ - url: url, │ │ │ │ │ - async: false │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - //construct the full URL │ │ │ │ │ - url = this.getFullRequestString(params); │ │ │ │ │ - } else { │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["gl"] │ │ │ │ │ + * Dictionary for Galego. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["gl"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - //tiled version │ │ │ │ │ - var currentRes = this.map.getResolution(); │ │ │ │ │ - var colidx = Math.floor((bounds.left - this.maxExtent.left) / currentRes); │ │ │ │ │ - colidx = Math.round(colidx / this.tileSize.w); │ │ │ │ │ - var rowidx = Math.floor((this.maxExtent.top - bounds.top) / currentRes); │ │ │ │ │ - rowidx = Math.round(rowidx / this.tileSize.h); │ │ │ │ │ + 'unhandledRequest': "Solicitude non xerada; a resposta foi: ${statusText}", │ │ │ │ │ │ │ │ │ │ - if (this.useHttpTile) { │ │ │ │ │ - url = this.getImageFilePath({ │ │ │ │ │ - tilecol: colidx, │ │ │ │ │ - tilerow: rowidx, │ │ │ │ │ - scaleindex: this.resolutions.length - this.map.zoom - 1 │ │ │ │ │ - }); │ │ │ │ │ + 'Permalink': "Ligazón permanente", │ │ │ │ │ │ │ │ │ │ - } else { │ │ │ │ │ - url = this.getFullRequestString({ │ │ │ │ │ - tilecol: colidx, │ │ │ │ │ - tilerow: rowidx, │ │ │ │ │ - scaleindex: this.resolutions.length - this.map.zoom - 1 │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return url; │ │ │ │ │ - }, │ │ │ │ │ + 'Overlays': "Capas superpostas", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getFullRequestString │ │ │ │ │ - * getFullRequestString on MapGuide layers is special, because we │ │ │ │ │ - * do a regular expression replace on ',' in parameters to '+'. │ │ │ │ │ - * This is why it is subclassed here. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * altUrl - {String} Alternative base URL to use. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url appropriately encoded for MapGuide │ │ │ │ │ - */ │ │ │ │ │ - getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ - // use layer's url unless altUrl passed in │ │ │ │ │ - var url = (altUrl == null) ? this.url : altUrl; │ │ │ │ │ + 'Base Layer': "Capa base", │ │ │ │ │ │ │ │ │ │ - // if url is not a string, it should be an array of strings, │ │ │ │ │ - // in which case we will randomly select one of them in order │ │ │ │ │ - // to evenly distribute requests to different urls. │ │ │ │ │ - if (typeof url == "object") { │ │ │ │ │ - url = url[Math.floor(Math.random() * url.length)]; │ │ │ │ │ - } │ │ │ │ │ - // requestString always starts with url │ │ │ │ │ - var requestString = url; │ │ │ │ │ + 'noFID': "Non se pode actualizar a funcionalidade para a que non hai FID.", │ │ │ │ │ │ │ │ │ │ - // create a new params hashtable with all the layer params and the │ │ │ │ │ - // new params together. then convert to string │ │ │ │ │ - var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ - allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ - // ignore parameters that are already in the url search string │ │ │ │ │ - var urlParams = OpenLayers.Util.upperCaseObject( │ │ │ │ │ - OpenLayers.Util.getParameters(url)); │ │ │ │ │ - for (var key in allParams) { │ │ │ │ │ - if (key.toUpperCase() in urlParams) { │ │ │ │ │ - delete allParams[key]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + 'browserNotSupported': "O seu navegador non soporta a renderización de vectores. Os renderizadores soportados actualmente son:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - /* MapGuide needs '+' seperating things like bounds/height/width. │ │ │ │ │ - Since typically this is URL encoded, we use a slight hack: we │ │ │ │ │ - depend on the list-like functionality of getParameterString to │ │ │ │ │ - leave ',' only in the case of list items (since otherwise it is │ │ │ │ │ - encoded) then do a regular expression replace on the , characters │ │ │ │ │ - to '+' */ │ │ │ │ │ - paramsString = paramsString.replace(/,/g, "+"); │ │ │ │ │ + 'minZoomLevelError': "A propiedade minZoomLevel é só para uso conxuntamente coas capas FixedZoomLevels-descendent. O feito de que esa capa wfs verifique o minZoomLevel é unha reliquia do pasado. Non podemos, con todo, eliminala sen a posibilidade de non romper as aplicacións baseadas en OL que poidan depender dela. Por iso a estamos deixando obsoleta (a comprobación minZoomLevel de embaixo será eliminada na versión 3.0). Por favor, no canto diso use o axuste de resolución mín/máx tal e como está descrito aquí: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - if (paramsString != "") { │ │ │ │ │ - var lastServerChar = url.charAt(url.length - 1); │ │ │ │ │ - if ((lastServerChar == "&") || (lastServerChar == "?")) { │ │ │ │ │ - requestString += paramsString; │ │ │ │ │ - } else { │ │ │ │ │ - if (url.indexOf('?') == -1) { │ │ │ │ │ - //serverPath has no ? -- add one │ │ │ │ │ - requestString += '?' + paramsString; │ │ │ │ │ - } else { │ │ │ │ │ - //serverPath contains ?, so must already have paramsString at the end │ │ │ │ │ - requestString += '&' + paramsString; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return requestString; │ │ │ │ │ - }, │ │ │ │ │ + 'commitSuccess': "Transacción WFS: ÉXITO ${response}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getImageFilePath │ │ │ │ │ - * special handler to request mapguide tiles from an http exposed tilecache │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * altUrl - {String} Alternative base URL to use. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the url for the tile image │ │ │ │ │ - */ │ │ │ │ │ - getImageFilePath: function(newParams, altUrl) { │ │ │ │ │ - // use layer's url unless altUrl passed in │ │ │ │ │ - var url = (altUrl == null) ? this.url : altUrl; │ │ │ │ │ + 'commitFailed': "Transacción WFS: FALLIDA ${response}", │ │ │ │ │ │ │ │ │ │ - // if url is not a string, it should be an array of strings, │ │ │ │ │ - // in which case we will randomly select one of them in order │ │ │ │ │ - // to evenly distribute requests to different urls. │ │ │ │ │ - if (typeof url == "object") { │ │ │ │ │ - url = url[Math.floor(Math.random() * url.length)]; │ │ │ │ │ - } │ │ │ │ │ - // requestString always starts with url │ │ │ │ │ - var requestString = url; │ │ │ │ │ + 'googleWarning': "A capa do Google non puido cargarse correctamente.\x3cbr\x3e\x3cbr\x3ePara evitar esta mensaxe, escolla unha nova capa base no seleccionador de capas na marxe superior dereita.\x3cbr\x3e\x3cbr\x3eProbablemente, isto acontece porque a escritura da libraría do Google Maps ou ben non foi incluída ou ben non contén a clave API correcta para o seu sitio.\x3cbr\x3e\x3cbr\x3eDesenvolvedores: para axudar a facer funcionar isto correctamente, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3epremede aquí\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - var tileRowGroup = ""; │ │ │ │ │ - var tileColGroup = ""; │ │ │ │ │ + 'getLayerWarning': "A capa ${layerType} foi incapaz de cargarse correctamente.\x3cbr\x3e\x3cbr\x3ePara evitar esta mensaxe, escolla unha nova capa base no seleccionador de capas na marxe superior dereita.\x3cbr\x3e\x3cbr\x3eProbablemente, isto acontece porque a escritura da libraría ${layerLib} non foi ben incluída.\x3cbr\x3e\x3cbr\x3eDesenvolvedores: para axudar a facer funcionar isto correctamente, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3epremede aquí\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - if (newParams.tilerow < 0) { │ │ │ │ │ - tileRowGroup = '-'; │ │ │ │ │ - } │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Escala = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - if (newParams.tilerow == 0) { │ │ │ │ │ - tileRowGroup += '0'; │ │ │ │ │ - } else { │ │ │ │ │ - tileRowGroup += Math.floor(Math.abs(newParams.tilerow / this.params.tileRowsPerFolder)) * this.params.tileRowsPerFolder; │ │ │ │ │ - } │ │ │ │ │ + 'W': "O", │ │ │ │ │ │ │ │ │ │ - if (newParams.tilecol < 0) { │ │ │ │ │ - tileColGroup = '-'; │ │ │ │ │ - } │ │ │ │ │ + 'E': "L", │ │ │ │ │ │ │ │ │ │ - if (newParams.tilecol == 0) { │ │ │ │ │ - tileColGroup += '0'; │ │ │ │ │ - } else { │ │ │ │ │ - tileColGroup += Math.floor(Math.abs(newParams.tilecol / this.params.tileColumnsPerFolder)) * this.params.tileColumnsPerFolder; │ │ │ │ │ - } │ │ │ │ │ + 'N': "N", │ │ │ │ │ │ │ │ │ │ - var tilePath = '/S' + Math.floor(newParams.scaleindex) + │ │ │ │ │ - '/' + this.params.basemaplayergroupname + │ │ │ │ │ - '/R' + tileRowGroup + │ │ │ │ │ - '/C' + tileColGroup + │ │ │ │ │ - '/' + (newParams.tilerow % this.params.tileRowsPerFolder) + │ │ │ │ │ - '_' + (newParams.tilecol % this.params.tileColumnsPerFolder) + │ │ │ │ │ - '.' + this.params.format; │ │ │ │ │ + 'S': "S", │ │ │ │ │ │ │ │ │ │ - if (this.params.querystring) { │ │ │ │ │ - tilePath += "?" + this.params.querystring; │ │ │ │ │ - } │ │ │ │ │ + 'reprojectDeprecated': "Está usando a opción \"reproject\" na capa ${layerName}. Esta opción está obsoleta: o seu uso foi deseñado para a visualización de datos sobre mapas base comerciais, pero esta funcionalidade debera agora ser obtida utilizando a proxección Spherical Mercator. Hai dispoñible máis información en http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ │ │ │ │ │ - requestString += tilePath; │ │ │ │ │ - return requestString; │ │ │ │ │ - }, │ │ │ │ │ + 'methodDeprecated': "Este método está obsoleto e será eliminado na versión 3.0. Por favor, no canto deste use ${newMethod}." │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.MapGuide" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/PointTrack.js │ │ │ │ │ + OpenLayers/Lang/es.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.PointTrack │ │ │ │ │ - * Vector layer to display ordered point features as a line, creating one │ │ │ │ │ - * LineString feature for each pair of two points. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Vector> │ │ │ │ │ + * Namespace: OpenLayers.Lang["es"] │ │ │ │ │ + * Dictionary for Spanish, UTF8 encoding. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.PointTrack = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ +OpenLayers.Lang.es = { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: dataFrom │ │ │ │ │ - * {<OpenLayers.Layer.PointTrack.TARGET_NODE>} or │ │ │ │ │ - * {<OpenLayers.Layer.PointTrack.SOURCE_NODE>} optional. If the lines │ │ │ │ │ - * should get the data/attributes from one of the two points it is │ │ │ │ │ - * composed of, which one should it be? │ │ │ │ │ - */ │ │ │ │ │ - dataFrom: null, │ │ │ │ │ + 'unhandledRequest': "Respuesta a petición no gestionada ${statusText}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: styleFrom │ │ │ │ │ - * {<OpenLayers.Layer.PointTrack.TARGET_NODE>} or │ │ │ │ │ - * {<OpenLayers.Layer.PointTrack.SOURCE_NODE>} optional. If the lines │ │ │ │ │ - * should get the style from one of the two points it is composed of, │ │ │ │ │ - * which one should it be? │ │ │ │ │ - */ │ │ │ │ │ - styleFrom: null, │ │ │ │ │ + 'Permalink': "Enlace permanente", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.PointTrack │ │ │ │ │ - * Constructor for a new OpenLayers.PointTrack instance. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} name of the layer │ │ │ │ │ - * options - {Object} Optional object with properties to tag onto the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ + 'Overlays': "Capas superpuestas", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: addNodes │ │ │ │ │ - * Adds point features that will be used to create lines from, using point │ │ │ │ │ - * pairs. The first point of a pair will be the source node, the second │ │ │ │ │ - * will be the target node. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pointFeatures - {Array(<OpenLayers.Feature>)} │ │ │ │ │ - * options - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Supported options: │ │ │ │ │ - * silent - {Boolean} true to suppress (before)feature(s)added events │ │ │ │ │ - */ │ │ │ │ │ - addNodes: function(pointFeatures, options) { │ │ │ │ │ - if (pointFeatures.length < 2) { │ │ │ │ │ - throw new Error("At least two point features have to be added to " + │ │ │ │ │ - "create a line from"); │ │ │ │ │ - } │ │ │ │ │ + 'Base Layer': "Capa Base", │ │ │ │ │ │ │ │ │ │ - var lines = new Array(pointFeatures.length - 1); │ │ │ │ │ + 'noFID': "No se puede actualizar un elemento para el que no existe FID.", │ │ │ │ │ │ │ │ │ │ - var pointFeature, startPoint, endPoint; │ │ │ │ │ - for (var i = 0, len = pointFeatures.length; i < len; i++) { │ │ │ │ │ - pointFeature = pointFeatures[i]; │ │ │ │ │ - endPoint = pointFeature.geometry; │ │ │ │ │ + 'browserNotSupported': "Su navegador no soporta renderización vectorial. Los renderizadores soportados actualmente son:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - if (!endPoint) { │ │ │ │ │ - var lonlat = pointFeature.lonlat; │ │ │ │ │ - endPoint = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ - } else if (endPoint.CLASS_NAME != "OpenLayers.Geometry.Point") { │ │ │ │ │ - throw new TypeError("Only features with point geometries are supported."); │ │ │ │ │ - } │ │ │ │ │ + // console message │ │ │ │ │ + 'minZoomLevelError': "La propiedad minZoomLevel debe sólo utilizarse " + │ │ │ │ │ + "con las capas que tienen FixedZoomLevels. El hecho de que " + │ │ │ │ │ + "una capa wfs compruebe minZoomLevel es una reliquia del " + │ │ │ │ │ + "pasado. Sin embargo, no podemos eliminarla sin discontinuar " + │ │ │ │ │ + "probablemente las aplicaciones OL que puedan depender de ello. " + │ │ │ │ │ + "Así pues estamos haciéndolo obsoleto --la comprobación " + │ │ │ │ │ + "minZoomLevel se eliminará en la versión 3.0. Utilice el ajuste " + │ │ │ │ │ + "de resolution min/max en su lugar, tal como se describe aquí: " + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - if (i > 0) { │ │ │ │ │ - var attributes = (this.dataFrom != null) ? │ │ │ │ │ - (pointFeatures[i + this.dataFrom].data || │ │ │ │ │ - pointFeatures[i + this.dataFrom].attributes) : │ │ │ │ │ - null; │ │ │ │ │ - var style = (this.styleFrom != null) ? │ │ │ │ │ - (pointFeatures[i + this.styleFrom].style) : │ │ │ │ │ - null; │ │ │ │ │ - var line = new OpenLayers.Geometry.LineString([startPoint, │ │ │ │ │ - endPoint │ │ │ │ │ - ]); │ │ │ │ │ + 'commitSuccess': "Transacción WFS: ÉXITO ${response}", │ │ │ │ │ │ │ │ │ │ - lines[i - 1] = new OpenLayers.Feature.Vector(line, attributes, │ │ │ │ │ - style); │ │ │ │ │ - } │ │ │ │ │ + 'commitFailed': "Transacción WFS: FALLÓ ${response}", │ │ │ │ │ │ │ │ │ │ - startPoint = endPoint; │ │ │ │ │ - } │ │ │ │ │ + 'googleWarning': "La capa Google no pudo ser cargada correctamente.<br><br>" + │ │ │ │ │ + "Para evitar este mensaje, seleccione una nueva Capa Base " + │ │ │ │ │ + "en el selector de capas en la esquina superior derecha.<br><br>" + │ │ │ │ │ + "Probablemente, esto se debe a que el script de la biblioteca de " + │ │ │ │ │ + "Google Maps no fue correctamente incluido en su página, o no " + │ │ │ │ │ + "contiene la clave del API correcta para su sitio.<br><br>" + │ │ │ │ │ + "Desarrolladores: Para ayudar a hacer funcionar esto correctamente, " + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ + "target='_blank'>haga clic aquí</a>", │ │ │ │ │ │ │ │ │ │ - this.addFeatures(lines, options); │ │ │ │ │ - }, │ │ │ │ │ + 'getLayerWarning': "La capa ${layerType} no pudo ser cargada correctamente.<br><br>" + │ │ │ │ │ + "Para evitar este mensaje, seleccione una nueva Capa Base " + │ │ │ │ │ + "en el selector de capas en la esquina superior derecha.<br><br>" + │ │ │ │ │ + "Probablemente, esto se debe a que el script de " + │ │ │ │ │ + "la biblioteca ${layerLib} " + │ │ │ │ │ + "no fue correctamente incluido en su página.<br><br>" + │ │ │ │ │ + "Desarrolladores: Para ayudar a hacer funcionar esto correctamente, " + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ + "target='_blank'>haga clic aquí</a>", │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.PointTrack" │ │ │ │ │ -}); │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Escala = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Layer.PointTrack.SOURCE_NODE │ │ │ │ │ - * {Number} value for <OpenLayers.Layer.PointTrack.dataFrom> and │ │ │ │ │ - * <OpenLayers.Layer.PointTrack.styleFrom> │ │ │ │ │ + //labels for the graticule control │ │ │ │ │ + 'W': 'O', │ │ │ │ │ + 'E': 'E', │ │ │ │ │ + 'N': 'N', │ │ │ │ │ + 'S': 'S', │ │ │ │ │ + 'Graticule': 'Retícula', │ │ │ │ │ + │ │ │ │ │ + // console message │ │ │ │ │ + 'reprojectDeprecated': "Está usando la opción 'reproject' en la capa " + │ │ │ │ │ + "${layerName}. Esta opción es obsoleta: su uso fue diseñado " + │ │ │ │ │ + "para soportar la visualización de datos sobre mapas base comerciales, " + │ │ │ │ │ + "pero ahora esa funcionalidad debería conseguirse mediante el soporte " + │ │ │ │ │ + "de la proyección Spherical Mercator. Más información disponible en " + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + │ │ │ │ │ + // console message │ │ │ │ │ + 'methodDeprecated': "Este método es obsoleto y se eliminará en la versión 3.0. " + │ │ │ │ │ + "Por favor utilice el método ${newMethod} en su lugar.", │ │ │ │ │ + │ │ │ │ │ + // **** end **** │ │ │ │ │ + 'end': '' │ │ │ │ │ + │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/bg.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - DCLXVI │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.PointTrack.SOURCE_NODE = -1; │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: OpenLayers.Layer.PointTrack.TARGET_NODE │ │ │ │ │ - * {Number} value for <OpenLayers.Layer.PointTrack.dataFrom> and │ │ │ │ │ - * <OpenLayers.Layer.PointTrack.styleFrom> │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.PointTrack.TARGET_NODE = 0; │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: OpenLayers.Layer.PointTrack.dataFrom │ │ │ │ │ - * {Object} with the following keys - *deprecated* │ │ │ │ │ - * - SOURCE_NODE: take data/attributes from the source node of the line │ │ │ │ │ - * - TARGET_NODE: take data/attributes from the target node of the line │ │ │ │ │ + * Namespace: OpenLayers.Lang["bg"] │ │ │ │ │ + * Dictionary for Български. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.PointTrack.dataFrom = { │ │ │ │ │ - 'SOURCE_NODE': -1, │ │ │ │ │ - 'TARGET_NODE': 0 │ │ │ │ │ -}; │ │ │ │ │ +OpenLayers.Lang["bg"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + │ │ │ │ │ + 'Permalink': "Постоянна препратка", │ │ │ │ │ + │ │ │ │ │ + 'Base Layer': "Основен слой", │ │ │ │ │ + │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Мащаб = 1 : ${scaleDenom}", │ │ │ │ │ + │ │ │ │ │ + 'methodDeprecated': "Този метод е остарял и ще бъде премахват в 3.0. Вместо него използвайте ${newMethod}." │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Boxes.js │ │ │ │ │ + OpenLayers/Lang/fur.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Klenje │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer.js │ │ │ │ │ - * @requires OpenLayers/Layer/Markers.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.Boxes │ │ │ │ │ - * Draw divs as 'boxes' on the layer. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Markers> │ │ │ │ │ + * Namespace: OpenLayers.Lang["fur"] │ │ │ │ │ + * Dictionary for Furlan. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Boxes = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ +OpenLayers.Lang["fur"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Boxes │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ - */ │ │ │ │ │ + 'Permalink': "Leam Permanent", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawMarker │ │ │ │ │ - * Calculate the pixel location for the marker, create it, and │ │ │ │ │ - * add it to the layer's div │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * marker - {<OpenLayers.Marker.Box>} │ │ │ │ │ - */ │ │ │ │ │ - drawMarker: function(marker) { │ │ │ │ │ - var topleft = this.map.getLayerPxFromLonLat({ │ │ │ │ │ - lon: marker.bounds.left, │ │ │ │ │ - lat: marker.bounds.top │ │ │ │ │ - }); │ │ │ │ │ - var botright = this.map.getLayerPxFromLonLat({ │ │ │ │ │ - lon: marker.bounds.right, │ │ │ │ │ - lat: marker.bounds.bottom │ │ │ │ │ - }); │ │ │ │ │ - if (botright == null || topleft == null) { │ │ │ │ │ - marker.display(false); │ │ │ │ │ - } else { │ │ │ │ │ - var markerDiv = marker.draw(topleft, { │ │ │ │ │ - w: Math.max(1, botright.x - topleft.x), │ │ │ │ │ - h: Math.max(1, botright.y - topleft.y) │ │ │ │ │ - }); │ │ │ │ │ - if (!marker.drawn) { │ │ │ │ │ - this.div.appendChild(markerDiv); │ │ │ │ │ - marker.drawn = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'Overlays': "Livei parsore", │ │ │ │ │ │ │ │ │ │ + 'Base Layer': "Livel di base", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: removeMarker │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * marker - {<OpenLayers.Marker.Box>} │ │ │ │ │ - */ │ │ │ │ │ - removeMarker: function(marker) { │ │ │ │ │ - OpenLayers.Util.removeItem(this.markers, marker); │ │ │ │ │ - if ((marker.div != null) && │ │ │ │ │ - (marker.div.parentNode == this.div)) { │ │ │ │ │ - this.div.removeChild(marker.div); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'browserNotSupported': "Il to sgarfadôr nol supuarte la renderizazion vetoriâl. Al moment a son supuartâts:\n${renderers}", │ │ │ │ │ + │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Scjale = 1 : ${scaleDenom}", │ │ │ │ │ + │ │ │ │ │ + 'W': "O", │ │ │ │ │ + │ │ │ │ │ + 'E': "E", │ │ │ │ │ + │ │ │ │ │ + 'N': "N", │ │ │ │ │ + │ │ │ │ │ + 'S': "S" │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Boxes" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Image.js │ │ │ │ │ + OpenLayers/Lang/is.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Ævar Arnfjörð Bjarmason │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer.js │ │ │ │ │ - * @requires OpenLayers/Tile/Image.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.Image │ │ │ │ │ - * Instances of OpenLayers.Layer.Image are used to display data from a web │ │ │ │ │ - * accessible image as a map layer. Create a new image layer with the │ │ │ │ │ - * <OpenLayers.Layer.Image> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer> │ │ │ │ │ + * Namespace: OpenLayers.Lang["is"] │ │ │ │ │ + * Dictionary for Íslenska. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Image = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ +OpenLayers.Lang["is"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: isBaseLayer │ │ │ │ │ - * {Boolean} The layer is a base layer. Default is true. Set this property │ │ │ │ │ - * in the layer options │ │ │ │ │ - */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + 'Permalink': "Varanlegur tengill", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: url │ │ │ │ │ - * {String} URL of the image to use │ │ │ │ │ - */ │ │ │ │ │ - url: null, │ │ │ │ │ + 'Overlays': "Þekjur", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: extent │ │ │ │ │ - * {<OpenLayers.Bounds>} The image bounds in map units. This extent will │ │ │ │ │ - * also be used as the default maxExtent for the layer. If you wish │ │ │ │ │ - * to have a maxExtent that is different than the image extent, set the │ │ │ │ │ - * maxExtent property of the options argument (as with any other layer). │ │ │ │ │ - */ │ │ │ │ │ - extent: null, │ │ │ │ │ + 'Base Layer': "Grunnlag", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: size │ │ │ │ │ - * {<OpenLayers.Size>} The image size in pixels │ │ │ │ │ - */ │ │ │ │ │ - size: null, │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Skali = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: tile │ │ │ │ │ - * {<OpenLayers.Tile.Image>} │ │ │ │ │ - */ │ │ │ │ │ - tile: null, │ │ │ │ │ + 'methodDeprecated': "Þetta fall hefur verið úrelt og verður fjarlægt í 3.0. Notaðu ${newMethod} í staðin." │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: aspectRatio │ │ │ │ │ - * {Float} The ratio of height/width represented by a single pixel in the │ │ │ │ │ - * graphic │ │ │ │ │ - */ │ │ │ │ │ - aspectRatio: null, │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/gsw.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Image │ │ │ │ │ - * Create a new image layer │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} A name for the layer. │ │ │ │ │ - * url - {String} Relative or absolute path to the image │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} The extent represented by the image │ │ │ │ │ - * size - {<OpenLayers.Size>} The size (in pixels) of the image │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(name, url, extent, size, options) { │ │ │ │ │ - this.url = url; │ │ │ │ │ - this.extent = extent; │ │ │ │ │ - this.maxExtent = extent; │ │ │ │ │ - this.size = size; │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Als-Holder │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - this.aspectRatio = (this.extent.getHeight() / this.size.h) / │ │ │ │ │ - (this.extent.getWidth() / this.size.w); │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Destroy this layer │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.tile) { │ │ │ │ │ - this.removeTileMonitoringHooks(this.tile); │ │ │ │ │ - this.tile.destroy(); │ │ │ │ │ - this.tile = null; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["gsw"] │ │ │ │ │ + * Dictionary for Alemannisch. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["gsw"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ - * │ │ │ │ │ - * Paramters: │ │ │ │ │ - * obj - {Object} An optional layer (is this ever used?) │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Image>} An exact copy of this layer │ │ │ │ │ - */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ + 'unhandledRequest': "Nit behandleti Aafrogsruckmäldig ${statusText}", │ │ │ │ │ │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Image(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.extent, │ │ │ │ │ - this.size, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ + 'Permalink': "Permalink", │ │ │ │ │ │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ + 'Overlays': "Iberlagerige", │ │ │ │ │ │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ + 'Base Layer': "Grundcharte", │ │ │ │ │ │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ + 'noFID': "E Feature, wu s kei FID derfir git, cha nit aktualisiert wäre.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setMap │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - /** │ │ │ │ │ - * If nothing to do with resolutions has been set, assume a single │ │ │ │ │ - * resolution determined by ratio*extent/size - if an image has a │ │ │ │ │ - * pixel aspect ratio different than one (as calculated above), the │ │ │ │ │ - * image will be stretched in one dimension only. │ │ │ │ │ - */ │ │ │ │ │ - if (this.options.maxResolution == null) { │ │ │ │ │ - this.options.maxResolution = this.aspectRatio * │ │ │ │ │ - this.extent.getWidth() / │ │ │ │ │ - this.size.w; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + 'browserNotSupported': "Dyy Browser unterstitzt kei Vektordarstellig. Aktuäll unterstitzti Renderer:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * Create the tile for the image or resize it for the new resolution │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * zoomChanged - {Boolean} │ │ │ │ │ - * dragging - {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + 'minZoomLevelError': "D minZoomLevel-Eigeschaft isch nume dänk fir d Layer, wu vu dr FixedZoomLevels abstamme. Ass dää wfs-Layer minZoomLevel prieft, scih e Relikt us dr Vergangeheit. Mir chenne s aber nit ändere ohni OL_basierti Aawändige villicht kaputt gehn, wu dervu abhänge. Us däm Grund het die Funktion d Eigeschaft \'deprecated\' iberchuu. D minZoomLevel-Priefig unte wird in dr Version 3.0 usegnuu. Bitte verwänd statt däm e min/max-Uflesig wie s do bschriben isch: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - var firstRendering = (this.tile == null); │ │ │ │ │ + 'commitSuccess': "WFS-Transaktion: ERFOLGRYCH ${response}", │ │ │ │ │ │ │ │ │ │ - if (zoomChanged || firstRendering) { │ │ │ │ │ + 'commitFailed': "WFS-Transaktion: FÄHLGSCHLAA ${response}", │ │ │ │ │ │ │ │ │ │ - //determine new tile size │ │ │ │ │ - this.setTileSize(); │ │ │ │ │ + 'googleWarning': "Dr Google-Layer het nit korräkt chenne glade wäre.\x3cbr\x3e\x3cbr\x3eGo die Mäldig nimi z kriege, wehl e andere Hintergrundlayer us em LayerSwitcher im rächte obere Ecke.\x3cbr\x3e\x3cbr\x3eDää Fähler git s seli hyfig, wel s Skript vu dr Google-Maps-Bibliothek nit yybunde woren isch oder wel s kei giltige API-Schlissel fir Dyy URL din het.\x3cbr\x3e\x3cbr\x3eEntwickler: Fir Hilf zum korräkte Yybinde vum Google-Layer \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3edoo drucke\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - //determine new position (upper left corner of new bounds) │ │ │ │ │ - var ulPx = this.map.getLayerPxFromLonLat({ │ │ │ │ │ - lon: this.extent.left, │ │ │ │ │ - lat: this.extent.top │ │ │ │ │ - }); │ │ │ │ │ + 'getLayerWarning': "Dr ${layerType}-Layer het nit korräkt chenne glade wäre.\x3cbr\x3e\x3cbr\x3eGo die Mäldig nimi z kriege, wehl e andere Hintergrundlayer us em LayerSwitcher im rächte obere Ecke.\x3cbr\x3e\x3cbr\x3eDää Fähler git s seli hyfig, wel s Skript vu dr \'${layerLib}\'-Bibliothek nit yybunde woren isch oder wel s kei giltige API-Schlissel fir Dyy URL din het.\x3cbr\x3e\x3cbr\x3eEntwickler: Fir Hilf zum korräkte Yybinde vu Layer \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3edoo drucke\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - if (firstRendering) { │ │ │ │ │ - //create the new tile │ │ │ │ │ - this.tile = new OpenLayers.Tile.Image(this, ulPx, this.extent, │ │ │ │ │ - null, this.tileSize); │ │ │ │ │ - this.addTileMonitoringHooks(this.tile); │ │ │ │ │ - } else { │ │ │ │ │ - //just resize the tile and set it's new position │ │ │ │ │ - this.tile.size = this.tileSize.clone(); │ │ │ │ │ - this.tile.position = ulPx.clone(); │ │ │ │ │ - } │ │ │ │ │ - this.tile.draw(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Maßstab = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Set the tile size based on the map size. │ │ │ │ │ - */ │ │ │ │ │ - setTileSize: function() { │ │ │ │ │ - var tileWidth = this.extent.getWidth() / this.map.getResolution(); │ │ │ │ │ - var tileHeight = this.extent.getHeight() / this.map.getResolution(); │ │ │ │ │ - this.tileSize = new OpenLayers.Size(tileWidth, tileHeight); │ │ │ │ │ - }, │ │ │ │ │ + 'W': "W", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: addTileMonitoringHooks │ │ │ │ │ - * This function takes a tile as input and adds the appropriate hooks to │ │ │ │ │ - * the tile so that the layer can keep track of the loading tiles. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * tile - {<OpenLayers.Tile>} │ │ │ │ │ - */ │ │ │ │ │ - addTileMonitoringHooks: function(tile) { │ │ │ │ │ - tile.onLoadStart = function() { │ │ │ │ │ - this.events.triggerEvent("loadstart"); │ │ │ │ │ - }; │ │ │ │ │ - tile.events.register("loadstart", this, tile.onLoadStart); │ │ │ │ │ + 'E': "O", │ │ │ │ │ │ │ │ │ │ - tile.onLoadEnd = function() { │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ - }; │ │ │ │ │ - tile.events.register("loadend", this, tile.onLoadEnd); │ │ │ │ │ - tile.events.register("unload", this, tile.onLoadEnd); │ │ │ │ │ - }, │ │ │ │ │ + 'N': "N", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeTileMonitoringHooks │ │ │ │ │ - * This function takes a tile as input and removes the tile hooks │ │ │ │ │ - * that were added in <addTileMonitoringHooks>. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * tile - {<OpenLayers.Tile>} │ │ │ │ │ - */ │ │ │ │ │ - removeTileMonitoringHooks: function(tile) { │ │ │ │ │ - tile.unload(); │ │ │ │ │ - tile.events.un({ │ │ │ │ │ - "loadstart": tile.onLoadStart, │ │ │ │ │ - "loadend": tile.onLoadEnd, │ │ │ │ │ - "unload": tile.onLoadEnd, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + 'S': "S", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setUrl │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newUrl - {String} │ │ │ │ │ - */ │ │ │ │ │ - setUrl: function(newUrl) { │ │ │ │ │ - this.url = newUrl; │ │ │ │ │ - this.tile.draw(); │ │ │ │ │ - }, │ │ │ │ │ + 'reprojectDeprecated': "Du bruchsch d \'reproject\'-Option bim ${layerName}-Layer. Die Option isch nimi giltig: si isch aagleit wore go Date iber kommerziälli Grundcharte lege, aber des sott mer jetz mache mit dr Unterstitzig vu Spherical Mercator. Meh Informatione git s uf http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getURL │ │ │ │ │ - * The url we return is always the same (the image itself never changes) │ │ │ │ │ - * so we can ignore the bounds parameter (it will always be the same, │ │ │ │ │ - * anyways) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - return this.url; │ │ │ │ │ - }, │ │ │ │ │ + 'methodDeprecated': "Die Methode isch veraltet un wird us dr Version 3.0 usegnuu. Bitte verwäbnd statt däm ${newMethod}." │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Image" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/PointGrid.js │ │ │ │ │ + OpenLayers/Lang/ja.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Fryed-peach │ │ │ │ │ + * - Mage Whopper │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.PointGrid │ │ │ │ │ - * A point grid layer dynamically generates a regularly spaced grid of point │ │ │ │ │ - * features. This is a specialty layer for cases where an application needs │ │ │ │ │ - * a regular grid of points. It can be used, for example, in an editing │ │ │ │ │ - * environment to snap to a grid. │ │ │ │ │ - * │ │ │ │ │ - * Create a new vector layer with the <OpenLayers.Layer.PointGrid> constructor. │ │ │ │ │ - * (code) │ │ │ │ │ - * // create a grid with points spaced at 10 map units │ │ │ │ │ - * var points = new OpenLayers.Layer.PointGrid({dx: 10, dy: 10}); │ │ │ │ │ - * │ │ │ │ │ - * // create a grid with different x/y spacing rotated 15 degrees clockwise. │ │ │ │ │ - * var points = new OpenLayers.Layer.PointGrid({dx: 5, dy: 10, rotation: 15}); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Vector> │ │ │ │ │ + * Namespace: OpenLayers.Lang["ja"] │ │ │ │ │ + * Dictionary for 日本語. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.PointGrid = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: dx │ │ │ │ │ - * {Number} Point grid spacing in the x-axis direction (map units). │ │ │ │ │ - * Read-only. Use the <setSpacing> method to modify this value. │ │ │ │ │ - */ │ │ │ │ │ - dx: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: dy │ │ │ │ │ - * {Number} Point grid spacing in the y-axis direction (map units). │ │ │ │ │ - * Read-only. Use the <setSpacing> method to modify this value. │ │ │ │ │ - */ │ │ │ │ │ - dy: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: ratio │ │ │ │ │ - * {Number} Ratio of the desired grid size to the map viewport size. │ │ │ │ │ - * Default is 1.5. Larger ratios mean the grid is recalculated less often │ │ │ │ │ - * while panning. The <maxFeatures> setting has precedence when determining │ │ │ │ │ - * grid size. Read-only. Use the <setRatio> method to modify this value. │ │ │ │ │ - */ │ │ │ │ │ - ratio: 1.5, │ │ │ │ │ +OpenLayers.Lang["ja"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: maxFeatures │ │ │ │ │ - * {Number} The maximum number of points to generate in the grid. Default │ │ │ │ │ - * is 250. Read-only. Use the <setMaxFeatures> method to modify this value. │ │ │ │ │ - */ │ │ │ │ │ - maxFeatures: 250, │ │ │ │ │ + 'unhandledRequest': "未処理の要求は ${statusText} を返します", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: rotation │ │ │ │ │ - * {Number} Grid rotation (in degrees clockwise from the positive x-axis). │ │ │ │ │ - * Default is 0. Read-only. Use the <setRotation> method to modify this │ │ │ │ │ - * value. │ │ │ │ │ - */ │ │ │ │ │ - rotation: 0, │ │ │ │ │ + 'Permalink': "パーマリンク", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: origin │ │ │ │ │ - * {<OpenLayers.LonLat>} Grid origin. The grid lattice will be aligned with │ │ │ │ │ - * the origin. If not set at construction, the center of the map's maximum │ │ │ │ │ - * extent is used. Read-only. Use the <setOrigin> method to modify this │ │ │ │ │ - * value. │ │ │ │ │ - */ │ │ │ │ │ - origin: null, │ │ │ │ │ + 'Overlays': "オーバーレイ", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: gridBounds │ │ │ │ │ - * {<OpenLayers.Bounds>} Internally cached grid bounds (with optional │ │ │ │ │ - * rotation applied). │ │ │ │ │ - */ │ │ │ │ │ - gridBounds: null, │ │ │ │ │ + 'Base Layer': "基底レイヤー", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.PointGrid │ │ │ │ │ - * Creates a new point grid layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} An object containing all configuration properties for │ │ │ │ │ - * the layer. The <dx> and <dy> properties are required to be set at │ │ │ │ │ - * construction. Any other layer properties may be set in this object. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ - config = config || {}; │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.initialize.apply(this, [config.name, config]); │ │ │ │ │ - }, │ │ │ │ │ + 'noFID': "FID のない地物は更新できません。", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * The layer has been added to the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ - map.events.register("moveend", this, this.onMoveEnd); │ │ │ │ │ - }, │ │ │ │ │ + 'browserNotSupported': "あなたのブラウザはベクターグラフィックスの描写に対応していません。現時点で対応しているソフトウェアは以下のものです。\n${renderers}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeMap │ │ │ │ │ - * The layer has been removed from the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - map.events.unregister("moveend", this, this.onMoveEnd); │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + 'minZoomLevelError': "minZoomLevel プロパティは FixedZoomLevels を継承するレイヤーでの使用のみを想定しています。この minZoomLevel に対する WFS レイヤーの検査は歴史的なものです。しかしながら、この検査を除去するとそれに依存する OpenLayers ベースのアプリケーションを破壊してしまう可能性があります。よって廃止が予定されており、この minZoomLevel 検査はバージョン3.0で除去されます。代わりに、http://trac.openlayers.org/wiki/SettingZoomLevels で解説されている、最小および最大解像度設定を使用してください。", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setRatio │ │ │ │ │ - * Set the grid <ratio> property and update the grid. Can only be called │ │ │ │ │ - * after the layer has been added to a map with a center/extent. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * ratio - {Number} │ │ │ │ │ - */ │ │ │ │ │ - setRatio: function(ratio) { │ │ │ │ │ - this.ratio = ratio; │ │ │ │ │ - this.updateGrid(true); │ │ │ │ │ - }, │ │ │ │ │ + 'commitSuccess': "WFS トランザクション: 成功 ${response}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setMaxFeatures │ │ │ │ │ - * Set the grid <maxFeatures> property and update the grid. Can only be │ │ │ │ │ - * called after the layer has been added to a map with a center/extent. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * maxFeatures - {Number} │ │ │ │ │ - */ │ │ │ │ │ - setMaxFeatures: function(maxFeatures) { │ │ │ │ │ - this.maxFeatures = maxFeatures; │ │ │ │ │ - this.updateGrid(true); │ │ │ │ │ - }, │ │ │ │ │ + 'commitFailed': "WFS トランザクション: 失敗 ${response}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setSpacing │ │ │ │ │ - * Set the grid <dx> and <dy> properties and update the grid. If only one │ │ │ │ │ - * argument is provided, it will be set as <dx> and <dy>. Can only be │ │ │ │ │ - * called after the layer has been added to a map with a center/extent. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * dx - {Number} │ │ │ │ │ - * dy - {Number} │ │ │ │ │ - */ │ │ │ │ │ - setSpacing: function(dx, dy) { │ │ │ │ │ - this.dx = dx; │ │ │ │ │ - this.dy = dy || dx; │ │ │ │ │ - this.updateGrid(true); │ │ │ │ │ - }, │ │ │ │ │ + 'googleWarning': "Google レイヤーが正しく読み込みを行えませんでした。\x3cbr\x3e\x3cbr\x3eこのメッセージを消すには、右上の隅にあるレイヤー切り替え部分で新しい基底レイヤーを選んでください。\x3cbr\x3e\x3cbr\x3eおそらく、これは Google マップ用ライブラリのスクリプトが組み込まれていないか、あなたのサイトに対応する正しい API キーが設定されていないためです。\x3cbr\x3e\x3cbr\x3e開発者の方へ: 正しい動作をさせるために\x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eこちらのウィキ\x3c/a\x3eを参照してください。", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setOrigin │ │ │ │ │ - * Set the grid <origin> property and update the grid. Can only be called │ │ │ │ │ - * after the layer has been added to a map with a center/extent. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * origin - {<OpenLayers.LonLat>} │ │ │ │ │ - */ │ │ │ │ │ - setOrigin: function(origin) { │ │ │ │ │ - this.origin = origin; │ │ │ │ │ - this.updateGrid(true); │ │ │ │ │ - }, │ │ │ │ │ + 'getLayerWarning': "${layerType} レイヤーが正しく読み込みを行えませんでした。\x3cbr\x3e\x3cbr\x3eこのメッセージを消すには、右上の隅にあるレイヤー切り替え部分で新しい基底レイヤーを選んでください。\x3cbr\x3e\x3cbr\x3eおそらく、これは ${layerLib} ライブラリのスクリプトが正しく組み込まれていないためです。\x3cbr\x3e\x3cbr\x3e開発者の方へ: 正しい動作をさせるために\x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eこちらのウィキ\x3c/a\x3eを参照してください。", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getOrigin │ │ │ │ │ - * Get the grid <origin> property. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.LonLat>} The grid origin. │ │ │ │ │ - */ │ │ │ │ │ - getOrigin: function() { │ │ │ │ │ - if (!this.origin) { │ │ │ │ │ - this.origin = this.map.getExtent().getCenterLonLat(); │ │ │ │ │ - } │ │ │ │ │ - return this.origin; │ │ │ │ │ - }, │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "縮尺 = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setRotation │ │ │ │ │ - * Set the grid <rotation> property and update the grid. Rotation values │ │ │ │ │ - * are in degrees clockwise from the positive x-axis (negative values │ │ │ │ │ - * for counter-clockwise rotation). Can only be called after the layer │ │ │ │ │ - * has been added to a map with a center/extent. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * rotation - {Number} Degrees clockwise from the positive x-axis. │ │ │ │ │ - */ │ │ │ │ │ - setRotation: function(rotation) { │ │ │ │ │ - this.rotation = rotation; │ │ │ │ │ - this.updateGrid(true); │ │ │ │ │ - }, │ │ │ │ │ + 'W': "西", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: onMoveEnd │ │ │ │ │ - * Listener for map "moveend" events. │ │ │ │ │ - */ │ │ │ │ │ - onMoveEnd: function() { │ │ │ │ │ - this.updateGrid(); │ │ │ │ │ - }, │ │ │ │ │ + 'E': "東", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getViewBounds │ │ │ │ │ - * Gets the (potentially rotated) view bounds for grid calculations. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} │ │ │ │ │ - */ │ │ │ │ │ - getViewBounds: function() { │ │ │ │ │ - var bounds = this.map.getExtent(); │ │ │ │ │ - if (this.rotation) { │ │ │ │ │ - var origin = this.getOrigin(); │ │ │ │ │ - var rotationOrigin = new OpenLayers.Geometry.Point(origin.lon, origin.lat); │ │ │ │ │ - var rect = bounds.toGeometry(); │ │ │ │ │ - rect.rotate(-this.rotation, rotationOrigin); │ │ │ │ │ - bounds = rect.getBounds(); │ │ │ │ │ - } │ │ │ │ │ - return bounds; │ │ │ │ │ - }, │ │ │ │ │ + 'N': "北", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateGrid │ │ │ │ │ - * Update the grid. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * force - {Boolean} Update the grid even if the previous bounds are still │ │ │ │ │ - * valid. │ │ │ │ │ - */ │ │ │ │ │ - updateGrid: function(force) { │ │ │ │ │ - if (force || this.invalidBounds()) { │ │ │ │ │ - var viewBounds = this.getViewBounds(); │ │ │ │ │ - var origin = this.getOrigin(); │ │ │ │ │ - var rotationOrigin = new OpenLayers.Geometry.Point(origin.lon, origin.lat); │ │ │ │ │ - var viewBoundsWidth = viewBounds.getWidth(); │ │ │ │ │ - var viewBoundsHeight = viewBounds.getHeight(); │ │ │ │ │ - var aspectRatio = viewBoundsWidth / viewBoundsHeight; │ │ │ │ │ - var maxHeight = Math.sqrt(this.dx * this.dy * this.maxFeatures / aspectRatio); │ │ │ │ │ - var maxWidth = maxHeight * aspectRatio; │ │ │ │ │ - var gridWidth = Math.min(viewBoundsWidth * this.ratio, maxWidth); │ │ │ │ │ - var gridHeight = Math.min(viewBoundsHeight * this.ratio, maxHeight); │ │ │ │ │ - var center = viewBounds.getCenterLonLat(); │ │ │ │ │ - this.gridBounds = new OpenLayers.Bounds( │ │ │ │ │ - center.lon - (gridWidth / 2), │ │ │ │ │ - center.lat - (gridHeight / 2), │ │ │ │ │ - center.lon + (gridWidth / 2), │ │ │ │ │ - center.lat + (gridHeight / 2) │ │ │ │ │ - ); │ │ │ │ │ - var rows = Math.floor(gridHeight / this.dy); │ │ │ │ │ - var cols = Math.floor(gridWidth / this.dx); │ │ │ │ │ - var gridLeft = origin.lon + (this.dx * Math.ceil((this.gridBounds.left - origin.lon) / this.dx)); │ │ │ │ │ - var gridBottom = origin.lat + (this.dy * Math.ceil((this.gridBounds.bottom - origin.lat) / this.dy)); │ │ │ │ │ - var features = new Array(rows * cols); │ │ │ │ │ - var x, y, point; │ │ │ │ │ - for (var i = 0; i < cols; ++i) { │ │ │ │ │ - x = gridLeft + (i * this.dx); │ │ │ │ │ - for (var j = 0; j < rows; ++j) { │ │ │ │ │ - y = gridBottom + (j * this.dy); │ │ │ │ │ - point = new OpenLayers.Geometry.Point(x, y); │ │ │ │ │ - if (this.rotation) { │ │ │ │ │ - point.rotate(this.rotation, rotationOrigin); │ │ │ │ │ - } │ │ │ │ │ - features[(i * rows) + j] = new OpenLayers.Feature.Vector(point); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.destroyFeatures(this.features, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.addFeatures(features, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'S': "南", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: invalidBounds │ │ │ │ │ - * Determine whether the previously generated point grid is invalid. │ │ │ │ │ - * This occurs when the map bounds extends beyond the previously │ │ │ │ │ - * generated grid bounds. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - invalidBounds: function() { │ │ │ │ │ - return !this.gridBounds || !this.gridBounds.containsBounds(this.getViewBounds()); │ │ │ │ │ - }, │ │ │ │ │ + 'reprojectDeprecated': "あなたは「${layerName}」レイヤーで reproject オプションを使っています。このオプションは商用の基底地図上に情報を表示する目的で設計されましたが、現在ではその機能は Spherical Mercator サポートを利用して実現されており、このオプションの使用は非推奨です。追加の情報は http://trac.openlayers.org/wiki/SphericalMercator で入手できます。", │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.PointGrid" │ │ │ │ │ + 'methodDeprecated': "このメソッドは廃止が予定されており、バージョン3.0で除去されます。代わりに ${newMethod} を使用してください。" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/GeoRSS.js │ │ │ │ │ + OpenLayers/Lang/km.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - វ័ណថារិទ្ធ │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Markers.js │ │ │ │ │ - * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.GeoRSS │ │ │ │ │ - * Add GeoRSS Point features to your map. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Markers> │ │ │ │ │ + * Namespace: OpenLayers.Lang["km"] │ │ │ │ │ + * Dictionary for ភាសាខ្មែរ. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.GeoRSS = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ +OpenLayers.Lang["km"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: location │ │ │ │ │ - * {String} store url of text file │ │ │ │ │ - */ │ │ │ │ │ - location: null, │ │ │ │ │ + 'Permalink': "តំណភ្ជាប់អចិន្ត្រៃយ៍", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Array(<OpenLayers.Feature>)} │ │ │ │ │ - */ │ │ │ │ │ - features: null, │ │ │ │ │ + 'Base Layer': "ស្រទាប់បាត​", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: formatOptions │ │ │ │ │ - * {Object} Hash of options which should be passed to the format when it is │ │ │ │ │ - * created. Must be passed in the constructor. │ │ │ │ │ - */ │ │ │ │ │ - formatOptions: null, │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "មាត្រដ្ឋាន = ១ ៖ ${scaleDenom}" │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: selectedFeature │ │ │ │ │ - * {<OpenLayers.Feature>} │ │ │ │ │ - */ │ │ │ │ │ - selectedFeature: null, │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/sv-SE.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: icon │ │ │ │ │ - * {<OpenLayers.Icon>}. This determines the Icon to be used on the map │ │ │ │ │ - * for this GeoRSS layer. │ │ │ │ │ - */ │ │ │ │ │ - icon: null, │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Sannab │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: popupSize │ │ │ │ │ - * {<OpenLayers.Size>} This determines the size of GeoRSS popups. If │ │ │ │ │ - * not provided, defaults to 250px by 120px. │ │ │ │ │ - */ │ │ │ │ │ - popupSize: null, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: useFeedTitle │ │ │ │ │ - * {Boolean} Set layer.name to the first <title> element in the feed. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - useFeedTitle: true, │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["sv"] │ │ │ │ │ + * Dictionary for Svenska. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["sv"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.GeoRSS │ │ │ │ │ - * Create a GeoRSS Layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * location - {String} │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(name, location, options) { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ - this.location = location; │ │ │ │ │ - this.features = []; │ │ │ │ │ - }, │ │ │ │ │ + 'unhandledRequest': "Ej hanterad fråga retur ${statusText}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - // Warning: Layer.Markers.destroy() must be called prior to calling │ │ │ │ │ - // clearFeatures() here, otherwise we leak memory. Indeed, if │ │ │ │ │ - // Layer.Markers.destroy() is called after clearFeatures(), it won't be │ │ │ │ │ - // able to remove the marker image elements from the layer's div since │ │ │ │ │ - // the markers will have been destroyed by clearFeatures(). │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.destroy.apply(this, arguments); │ │ │ │ │ - this.clearFeatures(); │ │ │ │ │ - this.features = null; │ │ │ │ │ - }, │ │ │ │ │ + 'Permalink': "Permalänk", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: loadRSS │ │ │ │ │ - * Start the load of the RSS data. Don't do this when we first add the layer, │ │ │ │ │ - * since we may not be visible at any point, and it would therefore be a waste. │ │ │ │ │ - */ │ │ │ │ │ - loadRSS: function() { │ │ │ │ │ - if (!this.loaded) { │ │ │ │ │ - this.events.triggerEvent("loadstart"); │ │ │ │ │ - OpenLayers.Request.GET({ │ │ │ │ │ - url: this.location, │ │ │ │ │ - success: this.parseData, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.loaded = true; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'Overlays': "Kartlager", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * If layer is visible and RSS has not been loaded, load RSS. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {Object} │ │ │ │ │ - * zoomChanged - {Object} │ │ │ │ │ - * minor - {Object} │ │ │ │ │ - */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, minor) { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - if (this.visibility && !this.loaded) { │ │ │ │ │ - this.loadRSS(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'Base Layer': "Bakgrundskarta", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseData │ │ │ │ │ - * Parse the data returned from the Events call. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * ajaxRequest - {<OpenLayers.Request.XMLHttpRequest>} │ │ │ │ │ - */ │ │ │ │ │ - parseData: function(ajaxRequest) { │ │ │ │ │ - var doc = ajaxRequest.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = OpenLayers.Format.XML.prototype.read(ajaxRequest.responseText); │ │ │ │ │ - } │ │ │ │ │ + 'noFID': "Kan ej uppdatera feature (objekt) för vilket FID saknas.", │ │ │ │ │ │ │ │ │ │ - if (this.useFeedTitle) { │ │ │ │ │ - var name = null; │ │ │ │ │ - try { │ │ │ │ │ - name = doc.getElementsByTagNameNS('*', 'title')[0].firstChild.nodeValue; │ │ │ │ │ - } catch (e) { │ │ │ │ │ - try { │ │ │ │ │ - name = doc.getElementsByTagName('title')[0].firstChild.nodeValue; │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - } │ │ │ │ │ - if (name) { │ │ │ │ │ - this.setName(name); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + 'browserNotSupported': "Din webbläsare stöder inte vektorvisning. För närvarande stöds följande visning:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - var options = {}; │ │ │ │ │ + 'minZoomLevelError': "Egenskapen minZoomLevel är endast avsedd att användas med lager med FixedZoomLevels. Att detta WFS-lager kontrollerar minZoomLevel är en relik från äldre versioner. Vi kan dock inte ta bort det utan att riskera att OL-baserade tillämpningar som använder detta slutar fungera. Därför är det satt som deprecated, minZoomLevel kommer att tas bort i version 3.0. Använd i stället inställning av min/max resolution som beskrivs här: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - OpenLayers.Util.extend(options, this.formatOptions); │ │ │ │ │ + 'commitSuccess': "WFS-transaktion: LYCKADES ${response}", │ │ │ │ │ │ │ │ │ │ - if (this.map && !this.projection.equals(this.map.getProjectionObject())) { │ │ │ │ │ - options.externalProjection = this.projection; │ │ │ │ │ - options.internalProjection = this.map.getProjectionObject(); │ │ │ │ │ - } │ │ │ │ │ + 'commitFailed': "WFS-transaktion: MISSLYCKADES ${response}", │ │ │ │ │ │ │ │ │ │ - var format = new OpenLayers.Format.GeoRSS(options); │ │ │ │ │ - var features = format.read(doc); │ │ │ │ │ + 'googleWarning': "Google-lagret kunde inte laddas korrekt.\x3cbr\x3e\x3cbr\x3eFör att slippa detta meddelande, välj en annan bakgrundskarta i lagerväljaren i övre högra hörnet.\x3cbr\x3e\x3cbr\x3eSannolikt beror felet på att Google Maps-biblioteket inte är inkluderat på webbsidan eller på att sidan inte anger korrekt API-nyckel för webbplatsen.\x3cbr\x3e\x3cbr\x3eUtvecklare: hjälp för att åtgärda detta, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eklicka här\x3c/a\x3e.", │ │ │ │ │ │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - var data = {}; │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ + 'getLayerWarning': "${layerType}-lagret kunde inte laddas korrekt.\x3cbr\x3e\x3cbr\x3eFör att slippa detta meddelande, välj en annan bakgrundskarta i lagerväljaren i övre högra hörnet.\x3cbr\x3e\x3cbr\x3eSannolikt beror felet på att ${layerLib}-biblioteket inte är inkluderat på webbsidan.\x3cbr\x3e\x3cbr\x3eUtvecklare: hjälp för att åtgärda detta, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eklicka här\x3c/a\x3e.", │ │ │ │ │ │ │ │ │ │ - // we don't support features with no geometry in the GeoRSS │ │ │ │ │ - // layer at this time. │ │ │ │ │ - if (!feature.geometry) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "\x3cstrong\x3eSkala\x3c/strong\x3e 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - var title = feature.attributes.title ? │ │ │ │ │ - feature.attributes.title : "Untitled"; │ │ │ │ │ + 'reprojectDeprecated': "Du använder inställningen \'reproject\' på lagret ${layerName}. Denna inställning markerad som deprecated: den var avsedd att användas för att stödja visning av kartdata på kommersiella bakgrundskartor, men nu bör man i stället använda Spherical Mercator-stöd för den funktionaliteten. Mer information finns på http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ │ │ │ │ │ - var description = feature.attributes.description ? │ │ │ │ │ - feature.attributes.description : "No description."; │ │ │ │ │ + 'methodDeprecated': "Denna metod är markerad som deprecated och kommer att tas bort i 3.0. Använd ${newMethod} i stället." │ │ │ │ │ │ │ │ │ │ - var link = feature.attributes.link ? feature.attributes.link : ""; │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/zh-CN.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - var location = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["zh-CN"] │ │ │ │ │ + * Dictionary for Simplified Chinese. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["zh-CN"] = { │ │ │ │ │ │ │ │ │ │ - data.icon = this.icon == null ? │ │ │ │ │ - OpenLayers.Marker.defaultIcon() : │ │ │ │ │ - this.icon.clone(); │ │ │ │ │ + 'unhandledRequest': "未处理的请求,返回值为 ${statusText}", │ │ │ │ │ │ │ │ │ │ - data.popupSize = this.popupSize ? │ │ │ │ │ - this.popupSize.clone() : │ │ │ │ │ - new OpenLayers.Size(250, 120); │ │ │ │ │ + 'Permalink': "永久链接", │ │ │ │ │ │ │ │ │ │ - if (title || description) { │ │ │ │ │ - // we have supplemental data, store them. │ │ │ │ │ - data.title = title; │ │ │ │ │ - data.description = description; │ │ │ │ │ + 'Overlays': "叠加层", │ │ │ │ │ │ │ │ │ │ - var contentHTML = '<div class="olLayerGeoRSSClose">[x]</div>'; │ │ │ │ │ - contentHTML += '<div class="olLayerGeoRSSTitle">'; │ │ │ │ │ - if (link) { │ │ │ │ │ - contentHTML += '<a class="link" href="' + link + '" target="_blank">'; │ │ │ │ │ - } │ │ │ │ │ - contentHTML += title; │ │ │ │ │ - if (link) { │ │ │ │ │ - contentHTML += '</a>'; │ │ │ │ │ - } │ │ │ │ │ - contentHTML += '</div>'; │ │ │ │ │ - contentHTML += '<div style="" class="olLayerGeoRSSDescription">'; │ │ │ │ │ - contentHTML += description; │ │ │ │ │ - contentHTML += '</div>'; │ │ │ │ │ - data['popupContentHTML'] = contentHTML; │ │ │ │ │ - } │ │ │ │ │ - var feature = new OpenLayers.Feature(this, location, data); │ │ │ │ │ - this.features.push(feature); │ │ │ │ │ - var marker = feature.createMarker(); │ │ │ │ │ - marker.events.register('click', feature, this.markerClick); │ │ │ │ │ - this.addMarker(marker); │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ - }, │ │ │ │ │ + 'Base Layer': "基础图层", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: markerClick │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - markerClick: function(evt) { │ │ │ │ │ - var sameMarkerClicked = (this == this.layer.selectedFeature); │ │ │ │ │ - this.layer.selectedFeature = (!sameMarkerClicked) ? this : null; │ │ │ │ │ - for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ - this.layer.map.removePopup(this.layer.map.popups[i]); │ │ │ │ │ - } │ │ │ │ │ - if (!sameMarkerClicked) { │ │ │ │ │ - var popup = this.createPopup(); │ │ │ │ │ - OpenLayers.Event.observe(popup.div, "click", │ │ │ │ │ - OpenLayers.Function.bind(function() { │ │ │ │ │ - for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ - this.layer.map.removePopup(this.layer.map.popups[i]); │ │ │ │ │ - } │ │ │ │ │ - }, this) │ │ │ │ │ - ); │ │ │ │ │ - this.layer.map.addPopup(popup); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - }, │ │ │ │ │ + 'noFID': "无法更新feature,缺少FID。", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clearFeatures │ │ │ │ │ - * Destroy all features in this layer. │ │ │ │ │ - */ │ │ │ │ │ - clearFeatures: function() { │ │ │ │ │ - if (this.features != null) { │ │ │ │ │ - while (this.features.length > 0) { │ │ │ │ │ - var feature = this.features[0]; │ │ │ │ │ - OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ - feature.destroy(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'browserNotSupported': "你使用的浏览器不支持矢量渲染。当前支持的渲染方式包括:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.GeoRSS" │ │ │ │ │ -}); │ │ │ │ │ + // console message │ │ │ │ │ + 'minZoomLevelError': "minZoomLevel属性仅适合用于" + │ │ │ │ │ + "使用了固定缩放级别的图层。这个 " + │ │ │ │ │ + "wfs 图层检查 minZoomLevel 是过去遗留下来的。" + │ │ │ │ │ + "然而,我们不能移除它," + │ │ │ │ │ + "而破坏依赖于它的基于OL的应用程序。" + │ │ │ │ │ + "因此,我们废除了它 -- minZoomLevel " + │ │ │ │ │ + "将会在3.0中被移除。请改用 " + │ │ │ │ │ + "min/max resolution 设置,参考:" + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + │ │ │ │ │ + 'commitSuccess': "WFS Transaction: 成功。 ${response}", │ │ │ │ │ + │ │ │ │ │ + 'commitFailed': "WFS Transaction: 失败。 ${response}", │ │ │ │ │ + │ │ │ │ │ + 'googleWarning': "Google图层不能正确加载。<br><br>" + │ │ │ │ │ + "要消除这个信息,请在右上角的" + │ │ │ │ │ + "图层控制面板中选择其他的基础图层。<br><br>" + │ │ │ │ │ + "这种情况很可能是没有正确的包含Google地图脚本库," + │ │ │ │ │ + "或者是没有包含在你的站点上" + │ │ │ │ │ + "使用的正确的Google Maps API密匙。<br><br>" + │ │ │ │ │ + "开发者:获取使其正确工作的帮助信息," + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ + "target='_blank'>点击这里</a>", │ │ │ │ │ + │ │ │ │ │ + 'getLayerWarning': "${layerType} 图层不能正确加载。<br><br>" + │ │ │ │ │ + "要消除这个信息,请在右上角的" + │ │ │ │ │ + "图层控制面板中选择其他的基础图层。<br><br>" + │ │ │ │ │ + "这种情况很可能是没有正确的包含" + │ │ │ │ │ + "${layerLib} 脚本库。<br><br>" + │ │ │ │ │ + "开发者:获取使其正确工作的帮助信息," + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ + "target='_blank'>点击这里</a>", │ │ │ │ │ + │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "比例尺 = 1 : ${scaleDenom}", │ │ │ │ │ + │ │ │ │ │ + // console message │ │ │ │ │ + 'reprojectDeprecated': "你正在使用 ${layerName} 图层上的'reproject'选项。" + │ │ │ │ │ + "这个选项已经不再使用:" + │ │ │ │ │ + "它是被设计用来支持显示商业的地图数据," + │ │ │ │ │ + "不过现在该功能可以通过使用Spherical Mercator来实现。" + │ │ │ │ │ + "更多信息可以参阅" + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + │ │ │ │ │ + // console message │ │ │ │ │ + 'methodDeprecated': "该方法已经不再被支持,并且将在3.0中被移除。" + │ │ │ │ │ + "请使用 ${newMethod} 方法来替代。", │ │ │ │ │ + │ │ │ │ │ + 'end': '' │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Google.js │ │ │ │ │ + OpenLayers/Lang/fi.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Nike │ │ │ │ │ + * - Str4nd │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["fi"] │ │ │ │ │ + * Dictionary for Suomi. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["fi"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + │ │ │ │ │ + 'Permalink': "Ikilinkki", │ │ │ │ │ + │ │ │ │ │ + 'Overlays': "Kerrokset", │ │ │ │ │ + │ │ │ │ │ + 'Base Layer': "Peruskerros", │ │ │ │ │ + │ │ │ │ │ + 'W': "L", │ │ │ │ │ + │ │ │ │ │ + 'E': "I", │ │ │ │ │ + │ │ │ │ │ + 'N': "P", │ │ │ │ │ + │ │ │ │ │ + 'S': "E" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/cs-CZ.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Mormegil │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/SphericalMercator.js │ │ │ │ │ - * @requires OpenLayers/Layer/EventPane.js │ │ │ │ │ - * @requires OpenLayers/Layer/FixedZoomLevels.js │ │ │ │ │ * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.Google │ │ │ │ │ - * │ │ │ │ │ - * Provides a wrapper for Google's Maps API │ │ │ │ │ - * Normally the Terms of Use for this API do not allow wrapping, but Google │ │ │ │ │ - * have provided written consent to OpenLayers for this - see email in │ │ │ │ │ - * http://osgeo-org.1560.n6.nabble.com/Google-Maps-API-Terms-of-Use-changes-tp4910013p4911981.html │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.SphericalMercator> │ │ │ │ │ - * - <OpenLayers.Layer.EventPane> │ │ │ │ │ - * - <OpenLayers.Layer.FixedZoomLevels> │ │ │ │ │ + * Namespace: OpenLayers.Lang["cs-CZ"] │ │ │ │ │ + * Dictionary for Česky. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Google = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Layer.EventPane, │ │ │ │ │ - OpenLayers.Layer.FixedZoomLevels, { │ │ │ │ │ +OpenLayers.Lang["cs-CZ"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: MIN_ZOOM_LEVEL │ │ │ │ │ - * {Integer} 0 │ │ │ │ │ - */ │ │ │ │ │ - MIN_ZOOM_LEVEL: 0, │ │ │ │ │ + 'unhandledRequest': "Nezpracovaná návratová hodnota ${statusText}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: MAX_ZOOM_LEVEL │ │ │ │ │ - * {Integer} 21 │ │ │ │ │ - */ │ │ │ │ │ - MAX_ZOOM_LEVEL: 21, │ │ │ │ │ + 'Permalink': "Trvalý odkaz", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: RESOLUTIONS │ │ │ │ │ - * {Array(Float)} Hardcode these resolutions so that they are more closely │ │ │ │ │ - * tied with the standard wms projection │ │ │ │ │ - */ │ │ │ │ │ - RESOLUTIONS: [ │ │ │ │ │ - 1.40625, │ │ │ │ │ - 0.703125, │ │ │ │ │ - 0.3515625, │ │ │ │ │ - 0.17578125, │ │ │ │ │ - 0.087890625, │ │ │ │ │ - 0.0439453125, │ │ │ │ │ - 0.02197265625, │ │ │ │ │ - 0.010986328125, │ │ │ │ │ - 0.0054931640625, │ │ │ │ │ - 0.00274658203125, │ │ │ │ │ - 0.001373291015625, │ │ │ │ │ - 0.0006866455078125, │ │ │ │ │ - 0.00034332275390625, │ │ │ │ │ - 0.000171661376953125, │ │ │ │ │ - 0.0000858306884765625, │ │ │ │ │ - 0.00004291534423828125, │ │ │ │ │ - 0.00002145767211914062, │ │ │ │ │ - 0.00001072883605957031, │ │ │ │ │ - 0.00000536441802978515, │ │ │ │ │ - 0.00000268220901489257, │ │ │ │ │ - 0.0000013411045074462891, │ │ │ │ │ - 0.00000067055225372314453 │ │ │ │ │ - ], │ │ │ │ │ + 'Overlays': "Překryvné vrstvy", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: type │ │ │ │ │ - * {GMapType} │ │ │ │ │ - */ │ │ │ │ │ - type: null, │ │ │ │ │ + 'Base Layer': "Podkladové vrstvy", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: wrapDateLine │ │ │ │ │ - * {Boolean} Allow user to pan forever east/west. Default is true. │ │ │ │ │ - * Setting this to false only restricts panning if │ │ │ │ │ - * <sphericalMercator> is true. │ │ │ │ │ - */ │ │ │ │ │ - wrapDateLine: true, │ │ │ │ │ + 'noFID': "Nelze aktualizovat prvek, pro který neexistuje FID.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: sphericalMercator │ │ │ │ │ - * {Boolean} Should the map act as a mercator-projected map? This will │ │ │ │ │ - * cause all interactions with the map to be in the actual map │ │ │ │ │ - * projection, which allows support for vector drawing, overlaying │ │ │ │ │ - * other maps, etc. │ │ │ │ │ - */ │ │ │ │ │ - sphericalMercator: false, │ │ │ │ │ + 'browserNotSupported': "Váš prohlížeč nepodporuje vykreslování vektorů. Momentálně podporované nástroje jsou::\n${renderers}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {Number} The version of the Google Maps API │ │ │ │ │ - */ │ │ │ │ │ - version: null, │ │ │ │ │ + 'minZoomLevelError': "Vlastnost minZoomLevel by se měla používat pouze s potomky FixedZoomLevels vrstvami. To znamená, že vrstva wfs kontroluje, zda-li minZoomLevel není zbytek z minulosti.Nelze to ovšem vyjmout bez možnosti, že bychom rozbili aplikace postavené na OL, které by na tom mohly záviset. Proto tuto vlastnost nedoporučujeme používat -- kontrola minZoomLevel bude odstraněna ve verzi 3.0. Použijte prosím raději nastavení min/max podle příkaldu popsaného na: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Google │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} A name for the layer. │ │ │ │ │ - * options - {Object} An optional object whose properties will be set │ │ │ │ │ - * on the layer. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - if (!options.version) { │ │ │ │ │ - options.version = typeof GMap2 === "function" ? "2" : "3"; │ │ │ │ │ - } │ │ │ │ │ - var mixin = OpenLayers.Layer.Google["v" + │ │ │ │ │ - options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (mixin) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, mixin); │ │ │ │ │ - } else { │ │ │ │ │ - throw "Unsupported Google Maps API version: " + options.version; │ │ │ │ │ - } │ │ │ │ │ + 'commitSuccess': "WFS Transaction: ÚSPĚCH ${response}", │ │ │ │ │ │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, mixin.DEFAULTS); │ │ │ │ │ - if (options.maxExtent) { │ │ │ │ │ - options.maxExtent = options.maxExtent.clone(); │ │ │ │ │ - } │ │ │ │ │ + 'commitFailed': "WFS Transaction: CHYBA ${response}", │ │ │ │ │ │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.initialize.apply(this, │ │ │ │ │ - [name, options]); │ │ │ │ │ - OpenLayers.Layer.FixedZoomLevels.prototype.initialize.apply(this, │ │ │ │ │ - [name, options]); │ │ │ │ │ + 'googleWarning': "Nepodařilo se správně načíst vrstvu Google.\x3cbr\x3e\x3cbr\x3eAbyste se zbavili této zprávy, zvolte jinou základní vrstvu v přepínači vrstev.\x3cbr\x3e\x3cbr\x3eTo se většinou stává, pokud nebyl načten skript, nebo neobsahuje správný klíč pro API pro tuto stránku.\x3cbr\x3e\x3cbr\x3eVývojáři: Pro pomoc, aby tohle fungovalo , \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eklikněte sem\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - OpenLayers.Util.extend(this, OpenLayers.Layer.SphericalMercator); │ │ │ │ │ - this.initMercatorParameters(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'getLayerWarning': "The ${layerType} Layer was unable to load correctly.\x3cbr\x3e\x3cbr\x3eTo get rid of this message, select a new BaseLayer in the layer switcher in the upper-right corner.\x3cbr\x3e\x3cbr\x3eMost likely, this is because the ${layerLib} library script was either not correctly included.\x3cbr\x3e\x3cbr\x3eDevelopers: For help getting this working correctly, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eclick here\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Google>} An exact clone of this layer │ │ │ │ │ - */ │ │ │ │ │ - clone: function() { │ │ │ │ │ - /** │ │ │ │ │ - * This method isn't intended to be called by a subclass and it │ │ │ │ │ - * doesn't call the same method on the superclass. We don't call │ │ │ │ │ - * the super's clone because we don't want properties that are set │ │ │ │ │ - * on this layer after initialize (i.e. this.mapObject etc.). │ │ │ │ │ - */ │ │ │ │ │ - return new OpenLayers.Layer.Google( │ │ │ │ │ - this.name, this.getOptions() │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Měřítko = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setVisibility │ │ │ │ │ - * Set the visibility flag for the layer and hide/show & redraw │ │ │ │ │ - * accordingly. Fire event unless otherwise specified │ │ │ │ │ - * │ │ │ │ │ - * Note that visibility is no longer simply whether or not the layer's │ │ │ │ │ - * style.display is set to "block". Now we store a 'visibility' state │ │ │ │ │ - * property on the layer class, this allows us to remember whether or │ │ │ │ │ - * not we *desire* for a layer to be visible. In the case where the │ │ │ │ │ - * map's resolution is out of the layer's range, this desire may be │ │ │ │ │ - * subverted. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * visible - {Boolean} Display the layer (if in range) │ │ │ │ │ - */ │ │ │ │ │ - setVisibility: function(visible) { │ │ │ │ │ - // sharing a map container, opacity has to be set per layer │ │ │ │ │ - var opacity = this.opacity == null ? 1 : this.opacity; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.setVisibility.apply(this, arguments); │ │ │ │ │ - this.setOpacity(opacity); │ │ │ │ │ - }, │ │ │ │ │ + 'reprojectDeprecated': "Použil jste volbu \'reproject\' ve vrstvě ${layerName}. Tato volba není doporučená: byla zde proto, aby bylo možno zobrazovat data z okomerčních serverů, ale tato funkce je nyní zajištěna pomocí podpory Spherical Mercator. Více informací naleznete na http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: display │ │ │ │ │ - * Hide or show the Layer │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * visible - {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - display: function(visible) { │ │ │ │ │ - if (!this._dragging) { │ │ │ │ │ - this.setGMapVisibility(visible); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.display.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + 'methodDeprecated': "Tato metoda je zavržená a bude ve verzi 3.0 odstraněna. Prosím, použijte raději ${newMethod}." │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * zoomChanged - {Boolean} Tells when zoom has changed, as layers have to │ │ │ │ │ - * do some init work in that case. │ │ │ │ │ - * dragging - {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - this._dragging = dragging; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - delete this._dragging; │ │ │ │ │ - }, │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/el.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setOpacity │ │ │ │ │ - * Sets the opacity for the entire layer (all images) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * opacity - {Float} │ │ │ │ │ - */ │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - if (opacity !== this.opacity) { │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "opacity" │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - this.opacity = opacity; │ │ │ │ │ - } │ │ │ │ │ - // Though this layer's opacity may not change, we're sharing a container │ │ │ │ │ - // and need to update the opacity for the entire container. │ │ │ │ │ - if (this.getVisibility()) { │ │ │ │ │ - var container = this.getMapContainer(); │ │ │ │ │ - OpenLayers.Util.modifyDOMElement( │ │ │ │ │ - container, null, null, null, null, null, null, opacity │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Omnipaedista │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up this layer. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - /** │ │ │ │ │ - * We have to override this method because the event pane destroy │ │ │ │ │ - * deletes the mapObject reference before removing this layer from │ │ │ │ │ - * the map. │ │ │ │ │ - */ │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.setGMapVisibility(false); │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache && cache.count <= 1) { │ │ │ │ │ - this.removeGMapElements(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeGMapElements │ │ │ │ │ - * Remove all elements added to the dom. This should only be called if │ │ │ │ │ - * this is the last of the Google layers for the given map. │ │ │ │ │ - */ │ │ │ │ │ - removeGMapElements: function() { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - // remove shared elements from dom │ │ │ │ │ - var container = this.mapObject && this.getMapContainer(); │ │ │ │ │ - if (container && container.parentNode) { │ │ │ │ │ - container.parentNode.removeChild(container); │ │ │ │ │ - } │ │ │ │ │ - var termsOfUse = cache.termsOfUse; │ │ │ │ │ - if (termsOfUse && termsOfUse.parentNode) { │ │ │ │ │ - termsOfUse.parentNode.removeChild(termsOfUse); │ │ │ │ │ - } │ │ │ │ │ - var poweredBy = cache.poweredBy; │ │ │ │ │ - if (poweredBy && poweredBy.parentNode) { │ │ │ │ │ - poweredBy.parentNode.removeChild(poweredBy); │ │ │ │ │ - } │ │ │ │ │ - if (this.mapObject && window.google && google.maps && │ │ │ │ │ - google.maps.event && google.maps.event.clearListeners) { │ │ │ │ │ - google.maps.event.clearListeners(this.mapObject, 'tilesloaded'); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["el"] │ │ │ │ │ + * Dictionary for Ελληνικά. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["el"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: removeMap │ │ │ │ │ - * On being removed from the map, also remove termsOfUse and poweredBy divs │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - // hide layer before removing │ │ │ │ │ - if (this.visibility && this.mapObject) { │ │ │ │ │ - this.setGMapVisibility(false); │ │ │ │ │ - } │ │ │ │ │ - // check to see if last Google layer in this map │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - if (cache.count <= 1) { │ │ │ │ │ - this.removeGMapElements(); │ │ │ │ │ - delete OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ - } else { │ │ │ │ │ - // decrement the layer count │ │ │ │ │ - --cache.count; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // remove references to gmap elements │ │ │ │ │ - delete this.termsOfUse; │ │ │ │ │ - delete this.poweredBy; │ │ │ │ │ - delete this.mapObject; │ │ │ │ │ - delete this.dragObject; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.removeMap.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Κλίμακα ~ 1 : ${scaleDenom}" │ │ │ │ │ │ │ │ │ │ - // │ │ │ │ │ - // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ - // │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/be-tarask.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getOLBoundsFromMapObjectBounds │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moBounds - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} An <OpenLayers.Bounds>, translated from the │ │ │ │ │ - * passed-in MapObject Bounds. │ │ │ │ │ - * Returns null if null value is passed in. │ │ │ │ │ - */ │ │ │ │ │ - getOLBoundsFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - var olBounds = null; │ │ │ │ │ - if (moBounds != null) { │ │ │ │ │ - var sw = moBounds.getSouthWest(); │ │ │ │ │ - var ne = moBounds.getNorthEast(); │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - sw = this.forwardMercator(sw.lng(), sw.lat()); │ │ │ │ │ - ne = this.forwardMercator(ne.lng(), ne.lat()); │ │ │ │ │ - } else { │ │ │ │ │ - sw = new OpenLayers.LonLat(sw.lng(), sw.lat()); │ │ │ │ │ - ne = new OpenLayers.LonLat(ne.lng(), ne.lat()); │ │ │ │ │ - } │ │ │ │ │ - olBounds = new OpenLayers.Bounds(sw.lon, │ │ │ │ │ - sw.lat, │ │ │ │ │ - ne.lon, │ │ │ │ │ - ne.lat); │ │ │ │ │ - } │ │ │ │ │ - return olBounds; │ │ │ │ │ - }, │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - EugeneZelenko │ │ │ │ │ + * - Jim-by │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getWarningHTML │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} String with information on why layer is broken, how to get │ │ │ │ │ - * it working. │ │ │ │ │ - */ │ │ │ │ │ - getWarningHTML: function() { │ │ │ │ │ - return OpenLayers.i18n("googleWarning"); │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["be-tarask"] │ │ │ │ │ + * Dictionary for Беларуская (тарашкевіца). Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["be-tarask"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Interface Controls * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ + 'unhandledRequest': "Неапрацаваны вынік запыту ${statusText}", │ │ │ │ │ │ │ │ │ │ + 'Permalink': "Сталая спасылка", │ │ │ │ │ │ │ │ │ │ - // Get&Set Center, Zoom │ │ │ │ │ + 'Overlays': "Слаі", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectCenter │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The mapObject's current center in Map Object format │ │ │ │ │ - */ │ │ │ │ │ - getMapObjectCenter: function() { │ │ │ │ │ - return this.mapObject.getCenter(); │ │ │ │ │ - }, │ │ │ │ │ + 'Base Layer': "Базавы слой", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectZoom │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} The mapObject's current zoom, in Map Object format │ │ │ │ │ - */ │ │ │ │ │ - getMapObjectZoom: function() { │ │ │ │ │ - return this.mapObject.getZoom(); │ │ │ │ │ - }, │ │ │ │ │ + 'noFID': "Немагчыма абнавіць магчымасьць, для якога не існуе FID.", │ │ │ │ │ │ │ │ │ │ + 'browserNotSupported': "Ваш браўзэр не падтрымлівае вэктарную графіку. У цяперашні момант падтрымліваюцца: ${renderers}", │ │ │ │ │ │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Primitives * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ + 'minZoomLevelError': "Уласьцівасьць minZoomLevel прызначана толькі для выкарыстаньня са слаямі вытворнымі ад FixedZoomLevels. Тое, што гэты wfs-слой правяраецца на minZoomLevel — рэха прошлага. Але мы ня можам выдаліць гэтую магчымасьць, таму што ад яе залежаць некаторыя заснаваныя на OL дастасаваньні. Тым ня менш, праверка minZoomLevel будзе выдаленая ў вэрсіі 3.0. Калі ласка, выкарыстоўваеце замест яе ўстаноўкі мінімальнага/максымальнага памераў, як апісана тут: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ + 'commitSuccess': "WFS-транзакцыя: ПОСЬПЕХ ${response}", │ │ │ │ │ │ │ │ │ │ - // LonLat │ │ │ │ │ + 'commitFailed': "WFS-транзакцыя: ПАМЫЛКА ${response}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getLongitudeFromMapObjectLonLat │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} Longitude of the given MapObject LonLat │ │ │ │ │ - */ │ │ │ │ │ - getLongitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - return this.sphericalMercator ? │ │ │ │ │ - this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lon : │ │ │ │ │ - moLonLat.lng(); │ │ │ │ │ - }, │ │ │ │ │ + 'googleWarning': "Не атрымалася загрузіць слой Google. \x3cbr\x3e\x3cbr\x3eКаб пазбавіцца гэтага паведамленьня, выберыце новы базавы слой у сьпісе ў верхнім правым куце.\x3cbr\x3e\x3cbr\x3e Хутчэй за ўсё, прычына ў тым, што скрыпт бібліятэкі Google Maps ня быў уключаныя альбо не ўтрымлівае слушны API-ключ для Вашага сайта.\x3cbr\x3e\x3cbr\x3eРаспрацоўшчыкам: Для таго, каб даведацца як зрабіць так, каб усё працавала, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eнацісьніце тут\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getLatitudeFromMapObjectLonLat │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} Latitude of the given MapObject LonLat │ │ │ │ │ - */ │ │ │ │ │ - getLatitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - var lat = this.sphericalMercator ? │ │ │ │ │ - this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lat : │ │ │ │ │ - moLonLat.lat(); │ │ │ │ │ - return lat; │ │ │ │ │ - }, │ │ │ │ │ + 'getLayerWarning': "Немагчыма загрузіць слой ${layerType}.\x3cbr\x3e\x3cbr\x3eКаб пазбавіцца гэтага паведамленьня, выберыце новы базавы слой у сьпісе ў верхнім правым куце.\x3cbr\x3e\x3cbr\x3eХутчэй за ўсё, прычына ў тым, што скрыпт бібліятэкі ${layerLib} ня быў слушна ўключаны.\x3cbr\x3e\x3cbr\x3eРаспрацоўшчыкам: Для таго, каб даведацца як зрабіць так, каб усё працавала, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eнацісьніце тут\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - // Pixel │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Маштаб = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getXFromMapObjectPixel │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moPixel - {Object} MapObject Pixel format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} X value of the MapObject Pixel │ │ │ │ │ - */ │ │ │ │ │ - getXFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return moPixel.x; │ │ │ │ │ - }, │ │ │ │ │ + 'W': "З", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getYFromMapObjectPixel │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moPixel - {Object} MapObject Pixel format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} Y value of the MapObject Pixel │ │ │ │ │ - */ │ │ │ │ │ - getYFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return moPixel.y; │ │ │ │ │ - }, │ │ │ │ │ + 'E': "У", │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Google" │ │ │ │ │ - }); │ │ │ │ │ + 'N': "Пн", │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Property: OpenLayers.Layer.Google.cache │ │ │ │ │ - * {Object} Cache for elements that should only be created once per map. │ │ │ │ │ + 'S': "Пд", │ │ │ │ │ + │ │ │ │ │ + 'reprojectDeprecated': "Вы выкарыстоўваеце ўстаноўку \'reproject\' для слоя ${layerName}. Гэтая ўстаноўка зьяўляецца састарэлай: яна выкарыстоўвалася для падтрымкі паказу зьвестак на камэрцыйных базавых мапах, але гэта функцыя цяпер рэалізаваная ў убудаванай падтрымцы сфэрычнай праекцыі Мэркатара. Дадатковая інфармацыя ёсьць на http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + │ │ │ │ │ + 'methodDeprecated': "Гэты мэтад састарэлы і будзе выдалены ў вэрсіі 3.0. Калі ласка, замест яго выкарыстоўвайце ${newMethod}." │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/vi.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Minh Nguyen │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Google.cache = {}; │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: OpenLayers.Layer.Google.v2 │ │ │ │ │ - * │ │ │ │ │ - * Mixin providing functionality specific to the Google Maps API v2. │ │ │ │ │ - * │ │ │ │ │ - * This API has been deprecated by Google. │ │ │ │ │ - * Developers are encouraged to migrate to v3 of the API; support for this │ │ │ │ │ - * is provided by <OpenLayers.Layer.Google.v3> │ │ │ │ │ + * Namespace: OpenLayers.Lang["vi"] │ │ │ │ │ + * Dictionary for Tiếng Việt. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Google.v2 = { │ │ │ │ │ +OpenLayers.Lang["vi"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: termsOfUse │ │ │ │ │ - * {DOMElement} Div for Google's copyright and terms of use link │ │ │ │ │ - */ │ │ │ │ │ - termsOfUse: null, │ │ │ │ │ + 'unhandledRequest': "Không xử lý được phản hồi ${statusText} cho yêu cầu", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: poweredBy │ │ │ │ │ - * {DOMElement} Div for Google's powered by logo and link │ │ │ │ │ - */ │ │ │ │ │ - poweredBy: null, │ │ │ │ │ + 'Permalink': "Liên kết thường trực", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: dragObject │ │ │ │ │ - * {GDraggableObject} Since 2.93, Google has exposed the ability to get │ │ │ │ │ - * the maps GDraggableObject. We can now use this for smooth panning │ │ │ │ │ - */ │ │ │ │ │ - dragObject: null, │ │ │ │ │ + 'Overlays': "Lấp bản đồ", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: loadMapObject │ │ │ │ │ - * Load the GMap and register appropriate event listeners. If we can't │ │ │ │ │ - * load GMap2, then display a warning message. │ │ │ │ │ - */ │ │ │ │ │ - loadMapObject: function() { │ │ │ │ │ - if (!this.type) { │ │ │ │ │ - this.type = G_NORMAL_MAP; │ │ │ │ │ - } │ │ │ │ │ - var mapObject, termsOfUse, poweredBy; │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - // there are already Google layers added to this map │ │ │ │ │ - mapObject = cache.mapObject; │ │ │ │ │ - termsOfUse = cache.termsOfUse; │ │ │ │ │ - poweredBy = cache.poweredBy; │ │ │ │ │ - // increment the layer count │ │ │ │ │ - ++cache.count; │ │ │ │ │ - } else { │ │ │ │ │ - // this is the first Google layer for this map │ │ │ │ │ + 'Base Layer': "Lớp nền", │ │ │ │ │ │ │ │ │ │ - var container = this.map.viewPortDiv; │ │ │ │ │ - var div = document.createElement("div"); │ │ │ │ │ - div.id = this.map.id + "_GMap2Container"; │ │ │ │ │ - div.style.position = "absolute"; │ │ │ │ │ - div.style.width = "100%"; │ │ │ │ │ - div.style.height = "100%"; │ │ │ │ │ - container.appendChild(div); │ │ │ │ │ + 'noFID': "Không thể cập nhật tính năng thiếu FID.", │ │ │ │ │ │ │ │ │ │ - // create GMap and shuffle elements │ │ │ │ │ - try { │ │ │ │ │ - mapObject = new GMap2(div); │ │ │ │ │ + 'browserNotSupported': "Trình duyệt của bạn không hỗ trợ chức năng vẽ bằng vectơ. Hiện hỗ trợ các bộ kết xuất:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - // move the ToS and branding stuff up to the container div │ │ │ │ │ - termsOfUse = div.lastChild; │ │ │ │ │ - container.appendChild(termsOfUse); │ │ │ │ │ - termsOfUse.style.zIndex = "1100"; │ │ │ │ │ - termsOfUse.style.right = ""; │ │ │ │ │ - termsOfUse.style.bottom = ""; │ │ │ │ │ - termsOfUse.className = "olLayerGoogleCopyright"; │ │ │ │ │ + 'minZoomLevelError': "Chỉ nên sử dụng thuộc tính minZoomLevel với các lớp FixedZoomLevels-descendent. Việc lớp wfs này tìm cho minZoomLevel là di tích còn lại từ xưa. Tuy nhiên, nếu chúng tôi dời nó thì sẽ vỡ các chương trình OpenLayers mà dựa trên nó. Bởi vậy chúng tôi phản đối sử dụng nó\x26nbsp;– bước tìm cho minZoomLevel sẽ được dời vào phiên bản 3.0. Xin sử dụng thiết lập độ phân tích tối thiểu / tối đa thay thế, theo hướng dẫn này: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - poweredBy = div.lastChild; │ │ │ │ │ - container.appendChild(poweredBy); │ │ │ │ │ - poweredBy.style.zIndex = "1100"; │ │ │ │ │ - poweredBy.style.right = ""; │ │ │ │ │ - poweredBy.style.bottom = ""; │ │ │ │ │ - poweredBy.className = "olLayerGooglePoweredBy gmnoprint"; │ │ │ │ │ + 'commitSuccess': "Giao dịch WFS: THÀNH CÔNG ${response}", │ │ │ │ │ │ │ │ │ │ - } catch (e) { │ │ │ │ │ - throw (e); │ │ │ │ │ - } │ │ │ │ │ - // cache elements for use by any other google layers added to │ │ │ │ │ - // this same map │ │ │ │ │ - OpenLayers.Layer.Google.cache[this.map.id] = { │ │ │ │ │ - mapObject: mapObject, │ │ │ │ │ - termsOfUse: termsOfUse, │ │ │ │ │ - poweredBy: poweredBy, │ │ │ │ │ - count: 1 │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ + 'commitFailed': "Giao dịch WFS: THẤT BẠI ${response}", │ │ │ │ │ │ │ │ │ │ - this.mapObject = mapObject; │ │ │ │ │ - this.termsOfUse = termsOfUse; │ │ │ │ │ - this.poweredBy = poweredBy; │ │ │ │ │ + 'googleWarning': "Không thể tải lớp Google đúng đắn.\x3cbr\x3e\x3cbr\x3eĐể tránh thông báo này lần sau, hãy chọn BaseLayer mới dùng điều khiển chọn lớp ở góc trên phải.\x3cbr\x3e\x3cbr\x3eChắc script thư viện Google Maps hoặc không được bao gồm hoặc không chứa khóa API hợp với website của bạn.\x3cbr\x3e\x3cbr\x3e\x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eTrợ giúp về tính năng này\x3c/a\x3e cho người phát triển.", │ │ │ │ │ │ │ │ │ │ - // ensure this layer type is one of the mapObject types │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.mapObject.getMapTypes(), │ │ │ │ │ - this.type) === -1) { │ │ │ │ │ - this.mapObject.addMapType(this.type); │ │ │ │ │ - } │ │ │ │ │ + 'getLayerWarning': "Không thể tải lớp ${layerType} đúng đắn.\x3cbr\x3e\x3cbr\x3eĐể tránh thông báo này lần sau, hãy chọn BaseLayer mới dùng điều khiển chọn lớp ở góc trên phải.\x3cbr\x3e\x3cbr\x3eChắc script thư viện ${layerLib} không được bao gồm đúng kiểu.\x3cbr\x3e\x3cbr\x3e\x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eTrợ giúp về tính năng này\x3c/a\x3e cho người phát triển.", │ │ │ │ │ │ │ │ │ │ - //since v 2.93 getDragObject is now available. │ │ │ │ │ - if (typeof mapObject.getDragObject == "function") { │ │ │ │ │ - this.dragObject = mapObject.getDragObject(); │ │ │ │ │ - } else { │ │ │ │ │ - this.dragPanMapObject = null; │ │ │ │ │ - } │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Tỷ lệ = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - if (this.isBaseLayer === false) { │ │ │ │ │ - this.setGMapVisibility(this.div.style.display !== "none"); │ │ │ │ │ - } │ │ │ │ │ + 'W': "T", │ │ │ │ │ │ │ │ │ │ - }, │ │ │ │ │ + 'E': "Đ", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: onMapResize │ │ │ │ │ - */ │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - // workaround for resizing of invisible or not yet fully loaded layers │ │ │ │ │ - // where GMap2.checkResize() does not work. We need to load the GMap │ │ │ │ │ - // for the old div size, then checkResize(), and then call │ │ │ │ │ - // layer.moveTo() to trigger GMap.setCenter() (which will finish │ │ │ │ │ - // the GMap initialization). │ │ │ │ │ - if (this.visibility && this.mapObject.isLoaded()) { │ │ │ │ │ - this.mapObject.checkResize(); │ │ │ │ │ - } else { │ │ │ │ │ - if (!this._resized) { │ │ │ │ │ - var layer = this; │ │ │ │ │ - var handle = GEvent.addListener(this.mapObject, "load", function() { │ │ │ │ │ - GEvent.removeListener(handle); │ │ │ │ │ - delete layer._resized; │ │ │ │ │ - layer.mapObject.checkResize(); │ │ │ │ │ - layer.moveTo(layer.map.getCenter(), layer.map.getZoom()); │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - this._resized = true; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'N': "B", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setGMapVisibility │ │ │ │ │ - * Display the GMap container and associated elements. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * visible - {Boolean} Display the GMap elements. │ │ │ │ │ - */ │ │ │ │ │ - setGMapVisibility: function(visible) { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - var container = this.mapObject.getContainer(); │ │ │ │ │ - if (visible === true) { │ │ │ │ │ - this.mapObject.setMapType(this.type); │ │ │ │ │ - container.style.display = ""; │ │ │ │ │ - this.termsOfUse.style.left = ""; │ │ │ │ │ - this.termsOfUse.style.display = ""; │ │ │ │ │ - this.poweredBy.style.display = ""; │ │ │ │ │ - cache.displayed = this.id; │ │ │ │ │ - } else { │ │ │ │ │ - if (cache.displayed === this.id) { │ │ │ │ │ - delete cache.displayed; │ │ │ │ │ - } │ │ │ │ │ - if (!cache.displayed) { │ │ │ │ │ - container.style.display = "none"; │ │ │ │ │ - this.termsOfUse.style.display = "none"; │ │ │ │ │ - // move ToU far to the left in addition to setting display │ │ │ │ │ - // to "none", because at the end of the GMap2 load │ │ │ │ │ - // sequence, display: none will be unset and ToU would be │ │ │ │ │ - // visible after loading a map with a google layer that is │ │ │ │ │ - // initially hidden. │ │ │ │ │ - this.termsOfUse.style.left = "-9999px"; │ │ │ │ │ - this.poweredBy.style.display = "none"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'S': "N", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getMapContainer │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} the GMap container's div │ │ │ │ │ - */ │ │ │ │ │ - getMapContainer: function() { │ │ │ │ │ - return this.mapObject.getContainer(); │ │ │ │ │ - }, │ │ │ │ │ + 'reprojectDeprecated': "Bạn đang áp dụng chế độ “reproject” vào lớp ${layerName}. Chế độ này đã bị phản đối: nó có mục đích hỗ trợ lấp dữ liệu trên các nền bản đồ thương mại; nên thực hiện hiệu ứng đó dùng tính năng Mercator Hình cầu. Có sẵn thêm chi tiết tại http://trac.openlayers.org/wiki/SphericalMercator .", │ │ │ │ │ │ │ │ │ │ - // │ │ │ │ │ - // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ - // │ │ │ │ │ + 'methodDeprecated': "Phương thức này đã bị phản đối và sẽ bị dời vào phiên bản 3.0. Xin hãy sử dụng ${newMethod} thay thế." │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectBoundsFromOLBounds │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * olBounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} A MapObject Bounds, translated from olBounds │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ - */ │ │ │ │ │ - getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ - var moBounds = null; │ │ │ │ │ - if (olBounds != null) { │ │ │ │ │ - var sw = this.sphericalMercator ? │ │ │ │ │ - this.inverseMercator(olBounds.bottom, olBounds.left) : │ │ │ │ │ - new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ - var ne = this.sphericalMercator ? │ │ │ │ │ - this.inverseMercator(olBounds.top, olBounds.right) : │ │ │ │ │ - new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ - moBounds = new GLatLngBounds(new GLatLng(sw.lat, sw.lon), │ │ │ │ │ - new GLatLng(ne.lat, ne.lon)); │ │ │ │ │ - } │ │ │ │ │ - return moBounds; │ │ │ │ │ - }, │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/it.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Interface Controls * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["it"] │ │ │ │ │ + * Dictionary for Italian. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang.it = { │ │ │ │ │ │ │ │ │ │ + 'unhandledRequest': "Codice di ritorno della richiesta ${statusText}", │ │ │ │ │ │ │ │ │ │ - // Get&Set Center, Zoom │ │ │ │ │ + 'Permalink': "Permalink", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setMapObjectCenter │ │ │ │ │ - * Set the mapObject to the specified center and zoom │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * center - {Object} MapObject LonLat format │ │ │ │ │ - * zoom - {int} MapObject zoom format │ │ │ │ │ - */ │ │ │ │ │ - setMapObjectCenter: function(center, zoom) { │ │ │ │ │ - this.mapObject.setCenter(center, zoom); │ │ │ │ │ - }, │ │ │ │ │ + 'Overlays': "Overlays", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: dragPanMapObject │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * dX - {Integer} │ │ │ │ │ - * dY - {Integer} │ │ │ │ │ - */ │ │ │ │ │ - dragPanMapObject: function(dX, dY) { │ │ │ │ │ - this.dragObject.moveBy(new GSize(-dX, dY)); │ │ │ │ │ - }, │ │ │ │ │ + 'Base Layer': "Livello base", │ │ │ │ │ │ │ │ │ │ + 'noFID': "Impossibile aggiornare un elemento grafico che non abbia il FID.", │ │ │ │ │ │ │ │ │ │ - // LonLat - Pixel Translation │ │ │ │ │ + 'browserNotSupported': "Il tuo browser non supporta il rendering vettoriale. I renderizzatore attualemnte supportati sono:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectLonLatFromMapObjectPixel │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moPixel - {Object} MapObject Pixel format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject LonLat translated from MapObject Pixel │ │ │ │ │ - */ │ │ │ │ │ - getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return this.mapObject.fromContainerPixelToLatLng(moPixel); │ │ │ │ │ - }, │ │ │ │ │ + // console message │ │ │ │ │ + 'minZoomLevelError': "La proprietà minZoomLevel è da utilizzare solamente " + │ │ │ │ │ + "con livelli che abbiano FixedZoomLevels. Il fatto che " + │ │ │ │ │ + "questo livello wfs controlli la proprietà minZoomLevel è " + │ │ │ │ │ + "un retaggio del passato. Non possiamo comunque rimuoverla " + │ │ │ │ │ + "senza rompere le vecchie applicazioni che dipendono su di essa." + │ │ │ │ │ + "Quindi siamo costretti a deprecarla -- minZoomLevel " + │ │ │ │ │ + "e sarà rimossa dalla vesione 3.0. Si prega di utilizzare i " + │ │ │ │ │ + "settaggi di risoluzione min/max come descritto qui: " + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectPixelFromMapObjectLonLat │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject Pixel transtlated from MapObject LonLat │ │ │ │ │ - */ │ │ │ │ │ - getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - return this.mapObject.fromLatLngToContainerPixel(moLonLat); │ │ │ │ │ - }, │ │ │ │ │ + 'commitSuccess': "Transazione WFS: SUCCESS ${response}", │ │ │ │ │ │ │ │ │ │ + 'commitFailed': "Transazione WFS: FAILED ${response}", │ │ │ │ │ │ │ │ │ │ - // Bounds │ │ │ │ │ + 'googleWarning': "Il livello Google non è riuscito a caricare correttamente.<br><br>" + │ │ │ │ │ + "Per evitare questo messaggio, seleziona un nuovo BaseLayer " + │ │ │ │ │ + "nel selettore di livelli nell'angolo in alto a destra.<br><br>" + │ │ │ │ │ + "Più precisamente, ciò accade perchè la libreria Google Maps " + │ │ │ │ │ + "non è stata inclusa nella pagina, oppure non contiene la " + │ │ │ │ │ + "corretta API key per il tuo sito.<br><br>" + │ │ │ │ │ + "Sviluppatori: Per aiuto su come farlo funzionare correttamente, " + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ + "target='_blank'>clicca qui</a>", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectZoomFromMapObjectBounds │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moBounds - {Object} MapObject Bounds format │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject Zoom for specified MapObject Bounds │ │ │ │ │ - */ │ │ │ │ │ - getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - return this.mapObject.getBoundsZoomLevel(moBounds); │ │ │ │ │ - }, │ │ │ │ │ + 'getLayerWarning': "Il livello ${layerType} non è riuscito a caricare correttamente.<br><br>" + │ │ │ │ │ + "Per evitare questo messaggio, seleziona un nuovo BaseLayer " + │ │ │ │ │ + "nel selettore di livelli nell'angolo in alto a destra.<br><br>" + │ │ │ │ │ + "Più precisamente, ciò accade perchè la libreria ${layerLib} " + │ │ │ │ │ + "non è stata inclusa nella pagina.<br><br>" + │ │ │ │ │ + "Sviluppatori: Per aiuto su come farlo funzionare correttamente, " + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ + "target='_blank'>clicca qui</a>", │ │ │ │ │ │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Primitives * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Scala = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ + // console message │ │ │ │ │ + 'reprojectDeprecated': "Stai utilizzando l'opzione 'reproject' sul livello ${layerName}. " + │ │ │ │ │ + "Questa opzione è deprecata: il suo utilizzo è stato introdotto per" + │ │ │ │ │ + "supportare il disegno dei dati sopra mappe commerciali, ma tale " + │ │ │ │ │ + "funzionalità dovrebbe essere ottenuta tramite l'utilizzo della proiezione " + │ │ │ │ │ + "Spherical Mercator. Per maggiori informazioni consultare qui " + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ │ │ │ │ │ - // LonLat │ │ │ │ │ + // console message │ │ │ │ │ + 'methodDeprecated': "Questo metodo è stato deprecato e sarà rimosso dalla versione 3.0. " + │ │ │ │ │ + "Si prega di utilizzare il metodo ${newMethod} in alternativa.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectLonLatFromLonLat │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * lon - {Float} │ │ │ │ │ - * lat - {Float} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject LonLat built from lon and lat params │ │ │ │ │ - */ │ │ │ │ │ - getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ - var gLatLng; │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ - gLatLng = new GLatLng(lonlat.lat, lonlat.lon); │ │ │ │ │ - } else { │ │ │ │ │ - gLatLng = new GLatLng(lat, lon); │ │ │ │ │ - } │ │ │ │ │ - return gLatLng; │ │ │ │ │ - }, │ │ │ │ │ + 'end': '' │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/pt-BR.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - // Pixel │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Luckas Blade │ │ │ │ │ + * - Rodrigo Avila │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectPixelFromXY │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {Integer} │ │ │ │ │ - * y - {Integer} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject Pixel from x and y parameters │ │ │ │ │ - */ │ │ │ │ │ - getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ - return new GPoint(x, y); │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ -}; │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["pt-br"] │ │ │ │ │ + * Dictionary for Português do Brasil. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["pt-BR"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + │ │ │ │ │ + 'unhandledRequest': "A requisição retornou um erro não tratado: ${statusText}", │ │ │ │ │ + │ │ │ │ │ + 'Permalink': "Link para essa página", │ │ │ │ │ + │ │ │ │ │ + 'Overlays': "Camadas de Sobreposição", │ │ │ │ │ + │ │ │ │ │ + 'Base Layer': "Camada Base", │ │ │ │ │ + │ │ │ │ │ + 'noFID': "Não é possível atualizar uma feição que não tenha um FID.", │ │ │ │ │ + │ │ │ │ │ + 'browserNotSupported': "Seu navegador não suporta renderização de vetores. Os renderizadores suportados atualmente são:\n${renderers}", │ │ │ │ │ + │ │ │ │ │ + 'minZoomLevelError': "A propriedade minZoomLevel é de uso restrito das camadas descendentes de FixedZoomLevels. A verificação dessa propriedade pelas camadas wfs é um resíduo do passado. Não podemos, entretanto não é possível removê-la sem possívelmente quebrar o funcionamento de aplicações OL que possuem depência com ela. Portanto estamos tornando seu uso obsoleto -- a verificação desse atributo será removida na versão 3.0. Ao invés, use as opções de resolução min/max como descrito em: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + │ │ │ │ │ + 'commitSuccess': "Transação WFS : SUCESSO ${response}", │ │ │ │ │ + │ │ │ │ │ + 'commitFailed': "Transação WFS : ERRO ${response}", │ │ │ │ │ + │ │ │ │ │ + 'googleWarning': "Não foi possível carregar a camada Google corretamente.\x3cbr\x3e\x3cbr\x3ePara se livrar dessa mensagem, selecione uma nova Camada Base, na ferramenta de alternação de camadas localização do canto superior direito.\x3cbr\x3e\x3cbr\x3eMuito provavelmente, isso foi causado porque o script da biblioteca do Google Maps não foi incluído, ou porque ele não contém a chave correta da API para o seu site.\x3cbr\x3e\x3cbr\x3eDesenvolvedores: Para obter ajuda em solucionar esse problema \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3ecliquem aqui\x3c/a\x3e", │ │ │ │ │ + │ │ │ │ │ + 'getLayerWarning': "Não foi possível carregar a camada ${layerType} corretamente.\x3cbr\x3e\x3cbr\x3ePara se livrar dessa mensagem, selecione uma nova Camada Base, na ferramenta de alternação de camadas localização do canto superior direito.\x3cbr\x3e\x3cbr\x3eMuito provavelmente, isso foi causado porque o script da biblioteca ${layerLib} não foi incluído corretamente.\x3cbr\x3e\x3cbr\x3eDesenvolvedores: Para obter ajuda em solucionar esse problema \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3ecliquem aqui\x3c/a\x3e", │ │ │ │ │ + │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Escala = 1 : ${scaleDenom}", │ │ │ │ │ + │ │ │ │ │ + 'W': "O", │ │ │ │ │ + │ │ │ │ │ + 'E': "L", │ │ │ │ │ + │ │ │ │ │ + 'N': "N", │ │ │ │ │ + │ │ │ │ │ + 'S': "S", │ │ │ │ │ + │ │ │ │ │ + 'reprojectDeprecated': "Você está usando a opção \'reproject\' na camada ${layerName}. Essa opção está obsoleta: seu uso foi projetado para suportar a visualização de dados sobre bases de mapas comerciais, entretanto essa funcionalidade deve agora ser alcançada usando o suporte à projeção Mercator. Mais informação está disponível em: http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + │ │ │ │ │ + 'methodDeprecated': "Esse método está obsoleto e será removido na versão 3.0. Ao invés, por favor use ${newMethod}." │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/KaMap.js │ │ │ │ │ + OpenLayers/Lang/id.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Irwangatot │ │ │ │ │ + * - IvanLanin │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.KaMap │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ + * Namespace: OpenLayers.Lang["id"] │ │ │ │ │ + * Dictionary for Bahasa Indonesia. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.KaMap = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ +OpenLayers.Lang["id"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} KaMap Layer is always a base layer │ │ │ │ │ - */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + 'unhandledRequest': "Permintaan yang tak tertangani menghasilkan ${statusText}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: DEFAULT_PARAMS │ │ │ │ │ - * {Object} parameters set by default. The default parameters set │ │ │ │ │ - * the format via the 'i' parameter to 'jpeg'. │ │ │ │ │ - */ │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - i: 'jpeg', │ │ │ │ │ - map: '' │ │ │ │ │ - }, │ │ │ │ │ + 'Permalink': "Pranala permanen", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.KaMap │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * url - {String} │ │ │ │ │ - * params - {Object} Parameters to be sent to the HTTP server in the │ │ │ │ │ - * query string for the tile. The format can be set via the 'i' │ │ │ │ │ - * parameter (defaults to jpg) , and the map should be set via │ │ │ │ │ - * the 'map' parameter. It has been reported that ka-Map may behave │ │ │ │ │ - * inconsistently if your format parameter does not match the format │ │ │ │ │ - * parameter configured in your config.php. (See ticket #327 for more │ │ │ │ │ - * information.) │ │ │ │ │ - * options - {Object} Additional options for the layer. Any of the │ │ │ │ │ - * APIProperties listed on this layer, and any layer types it │ │ │ │ │ - * extends, can be overridden through the options parameter. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, this.DEFAULT_PARAMS │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ + 'Overlays': "Hamparan", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as │ │ │ │ │ - * parameters │ │ │ │ │ - */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var mapRes = this.map.getResolution(); │ │ │ │ │ - var scale = Math.round((this.map.getScale() * 10000)) / 10000; │ │ │ │ │ - var pX = Math.round(bounds.left / mapRes); │ │ │ │ │ - var pY = -Math.round(bounds.top / mapRes); │ │ │ │ │ - return this.getFullRequestString({ │ │ │ │ │ - t: pY, │ │ │ │ │ - l: pX, │ │ │ │ │ - s: scale │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + 'Base Layer': "Lapisan Dasar", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateGridLayout │ │ │ │ │ - * ka-Map uses the center point of the map as an origin for │ │ │ │ │ - * its tiles. Override calculateGridLayout to center tiles │ │ │ │ │ - * correctly for this case. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bound>} │ │ │ │ │ - * origin - {<OpenLayers.LonLat>} │ │ │ │ │ - * resolution - {Number} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Object containing properties tilelon, tilelat, startcol, │ │ │ │ │ - * startrow │ │ │ │ │ - */ │ │ │ │ │ - calculateGridLayout: function(bounds, origin, resolution) { │ │ │ │ │ - var tilelon = resolution * this.tileSize.w; │ │ │ │ │ - var tilelat = resolution * this.tileSize.h; │ │ │ │ │ + 'noFID': "Tidak dapat memperbarui fitur yang tidak memiliki FID.", │ │ │ │ │ │ │ │ │ │ - var offsetlon = bounds.left; │ │ │ │ │ - var tilecol = Math.floor(offsetlon / tilelon) - this.buffer; │ │ │ │ │ + 'browserNotSupported': "Peramban Anda tidak mendukung penggambaran vektor. Penggambar yang didukung saat ini adalah:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - var offsetlat = bounds.top; │ │ │ │ │ - var tilerow = Math.floor(offsetlat / tilelat) + this.buffer; │ │ │ │ │ + 'minZoomLevelError': "Properti minZoomLevel hanya ditujukan bekerja dengan lapisan FixedZoomLevels-descendent. Pengecekan minZoomLevel oleh lapisan wfs adalah peninggalan masa lalu. Kami tidak dapat menghapusnya tanpa kemungkinan merusak aplikasi berbasis OL yang mungkin bergantung padanya. Karenanya, kami menganggapnya tidak berlaku -- Cek minZoomLevel di bawah ini akan dihapus pada 3.0. Silakan gunakan penyetelan resolusi min/maks seperti dijabarkan di sini: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - return { │ │ │ │ │ - tilelon: tilelon, │ │ │ │ │ - tilelat: tilelat, │ │ │ │ │ - startcol: tilecol, │ │ │ │ │ - startrow: tilerow │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ + 'commitSuccess': "WFS Transaksi: BERHASIL ${respon}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getTileBoundsForGridIndex │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * row - {Number} The row of the grid │ │ │ │ │ - * col - {Number} The column of the grid │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} The bounds for the tile at (row, col) │ │ │ │ │ - */ │ │ │ │ │ - getTileBoundsForGridIndex: function(row, col) { │ │ │ │ │ - var origin = this.getTileOrigin(); │ │ │ │ │ - var tileLayout = this.gridLayout; │ │ │ │ │ - var tilelon = tileLayout.tilelon; │ │ │ │ │ - var tilelat = tileLayout.tilelat; │ │ │ │ │ - var minX = (tileLayout.startcol + col) * tilelon; │ │ │ │ │ - var minY = (tileLayout.startrow - row) * tilelat; │ │ │ │ │ - return new OpenLayers.Bounds( │ │ │ │ │ - minX, minY, │ │ │ │ │ - minX + tilelon, minY + tilelat │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ + 'commitFailed': "WFS Transaksi: GAGAL ${respon}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Kamap>} An exact clone of this OpenLayers.Layer.KaMap │ │ │ │ │ - */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ + 'googleWarning': "Lapisan Google tidak dapat dimuat dengan benar.\x3cbr\x3e\x3cbr\x3eUntuk menghilangkan pesan ini, pilih suatu BaseLayer baru melalui penukar lapisan (layer switcher) di ujung kanan atas.\x3cbr\x3e\x3cbr\x3eKemungkinan besar ini karena pustaka skrip Google Maps tidak disertakan atau tidak mengandung kunci API yang tepat untuk situs Anda.\x3cbr\x3e\x3cbr\x3ePengembang: Untuk bantuan mengatasi masalah ini, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eklik di sini\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.KaMap(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.params, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ + 'getLayerWarning': "Lapisan ${layerType} tidak dapat dimuat dengan benar.\x3cbr\x3e\x3cbr\x3eUntuk menghilangkan pesan ini, pilih suatu BaseLayer baru melalui penukar lapisan (layer switcher) di ujung kanan atas.\x3cbr\x3e\x3cbr\x3eKemungkinan besar ini karena pustaka skrip Google Maps tidak disertakan dengan benar.\x3cbr\x3e\x3cbr\x3ePengembang: Untuk bantuan mengatasi masalah ini, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eklik di sini\x3c/a\x3e", │ │ │ │ │ │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Sekala = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - if (this.tileSize != null) { │ │ │ │ │ - obj.tileSize = this.tileSize.clone(); │ │ │ │ │ - } │ │ │ │ │ + 'W': "B", │ │ │ │ │ │ │ │ │ │ - // we do not want to copy reference to grid, so we make a new array │ │ │ │ │ - obj.grid = []; │ │ │ │ │ + 'E': "T", │ │ │ │ │ │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ + 'N': "U", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getTileBounds │ │ │ │ │ - * Returns The tile bounds for a layer given a pixel location. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * viewPortPx - {<OpenLayers.Pixel>} The location in the viewport. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} Bounds of the tile at the given pixel location. │ │ │ │ │ - */ │ │ │ │ │ - getTileBounds: function(viewPortPx) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var tileMapWidth = resolution * this.tileSize.w; │ │ │ │ │ - var tileMapHeight = resolution * this.tileSize.h; │ │ │ │ │ - var mapPoint = this.getLonLatFromViewPortPx(viewPortPx); │ │ │ │ │ - var tileLeft = tileMapWidth * Math.floor(mapPoint.lon / tileMapWidth); │ │ │ │ │ - var tileBottom = tileMapHeight * Math.floor(mapPoint.lat / tileMapHeight); │ │ │ │ │ - return new OpenLayers.Bounds(tileLeft, tileBottom, │ │ │ │ │ - tileLeft + tileMapWidth, │ │ │ │ │ - tileBottom + tileMapHeight); │ │ │ │ │ - }, │ │ │ │ │ + 'S': "S", │ │ │ │ │ + │ │ │ │ │ + 'reprojectDeprecated': "Anda menggunakan opsi \'reproject\' pada lapisan ${layerName}. Opsi ini telah ditinggalkan: penggunaannya dirancang untuk mendukung tampilan data melalui peta dasar komersial, tapi fungsionalitas tersebut saat ini harus dilakukan dengan menggunakan dukungan Spherical Mercator. Informasi lebih lanjut tersedia di http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + │ │ │ │ │ + 'methodDeprecated': "Metode ini telah usang dan akan dihapus di 3.0. Sebaliknya, harap gunakan ${newMethod}." │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.KaMap" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/Text.js │ │ │ │ │ + OpenLayers/Lang/io.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Malafaya │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.Text │ │ │ │ │ - * Read Text format. Create a new instance with the <OpenLayers.Format.Text> │ │ │ │ │ - * constructor. This reads text which is formatted like CSV text, using │ │ │ │ │ - * tabs as the seperator by default. It provides parsing of data originally │ │ │ │ │ - * used in the MapViewerService, described on the wiki. This Format is used │ │ │ │ │ - * by the <OpenLayers.Layer.Text> class. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format> │ │ │ │ │ + * Namespace: OpenLayers.Lang["io"] │ │ │ │ │ + * Dictionary for Ido. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.Text = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ +OpenLayers.Lang["io"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: defaultStyle │ │ │ │ │ - * defaultStyle allows one to control the default styling of the features. │ │ │ │ │ - * It should be a symbolizer hash. By default, this is set to match the │ │ │ │ │ - * Layer.Text behavior, which is to use the default OpenLayers Icon. │ │ │ │ │ - */ │ │ │ │ │ - defaultStyle: null, │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Skalo = 1 : ${scaleDenom}" │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: extractStyles │ │ │ │ │ - * set to true to extract styles from the TSV files, using information │ │ │ │ │ - * from the image or icon, iconSize and iconOffset fields. This will result │ │ │ │ │ - * in features with a symbolizer (style) property set, using the │ │ │ │ │ - * default symbolizer specified in <defaultStyle>. Set to false if you │ │ │ │ │ - * wish to use a styleMap or OpenLayers.Style options to style your │ │ │ │ │ - * layer instead. │ │ │ │ │ - */ │ │ │ │ │ - extractStyles: true, │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/en.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.Text │ │ │ │ │ - * Create a new parser for TSV Text. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (options.extractStyles !== false) { │ │ │ │ │ - options.defaultStyle = { │ │ │ │ │ - 'externalGraphic': OpenLayers.Util.getImageLocation("marker.png"), │ │ │ │ │ - 'graphicWidth': 21, │ │ │ │ │ - 'graphicHeight': 25, │ │ │ │ │ - 'graphicXOffset': -10.5, │ │ │ │ │ - 'graphicYOffset': -12.5 │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["en"] │ │ │ │ │ + * Dictionary for English. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang.en = { │ │ │ │ │ │ │ │ │ │ - OpenLayers.Format.prototype.initialize.apply(this, [options]); │ │ │ │ │ - }, │ │ │ │ │ + 'unhandledRequest': "Unhandled request return ${statusText}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Return a list of features from a Tab Seperated Values text string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * text - {String} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ - */ │ │ │ │ │ - read: function(text) { │ │ │ │ │ - var lines = text.split('\n'); │ │ │ │ │ - var columns; │ │ │ │ │ - var features = []; │ │ │ │ │ - // length - 1 to allow for trailing new line │ │ │ │ │ - for (var lcv = 0; lcv < (lines.length - 1); lcv++) { │ │ │ │ │ - var currLine = lines[lcv].replace(/^\s*/, '').replace(/\s*$/, ''); │ │ │ │ │ + 'Permalink': "Permalink", │ │ │ │ │ │ │ │ │ │ - if (currLine.charAt(0) != '#') { │ │ │ │ │ - /* not a comment */ │ │ │ │ │ + 'Overlays': "Overlays", │ │ │ │ │ │ │ │ │ │ - if (!columns) { │ │ │ │ │ - //First line is columns │ │ │ │ │ - columns = currLine.split('\t'); │ │ │ │ │ - } else { │ │ │ │ │ - var vals = currLine.split('\t'); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point(0, 0); │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - var style = this.defaultStyle ? │ │ │ │ │ - OpenLayers.Util.applyDefaults({}, this.defaultStyle) : │ │ │ │ │ - null; │ │ │ │ │ - var icon, iconSize, iconOffset, overflow; │ │ │ │ │ - var set = false; │ │ │ │ │ - for (var valIndex = 0; valIndex < vals.length; valIndex++) { │ │ │ │ │ - if (vals[valIndex]) { │ │ │ │ │ - if (columns[valIndex] == 'point') { │ │ │ │ │ - var coords = vals[valIndex].split(','); │ │ │ │ │ - geometry.y = parseFloat(coords[0]); │ │ │ │ │ - geometry.x = parseFloat(coords[1]); │ │ │ │ │ - set = true; │ │ │ │ │ - } else if (columns[valIndex] == 'lat') { │ │ │ │ │ - geometry.y = parseFloat(vals[valIndex]); │ │ │ │ │ - set = true; │ │ │ │ │ - } else if (columns[valIndex] == 'lon') { │ │ │ │ │ - geometry.x = parseFloat(vals[valIndex]); │ │ │ │ │ - set = true; │ │ │ │ │ - } else if (columns[valIndex] == 'title') │ │ │ │ │ - attributes['title'] = vals[valIndex]; │ │ │ │ │ - else if (columns[valIndex] == 'image' || │ │ │ │ │ - columns[valIndex] == 'icon' && style) { │ │ │ │ │ - style['externalGraphic'] = vals[valIndex]; │ │ │ │ │ - } else if (columns[valIndex] == 'iconSize' && style) { │ │ │ │ │ - var size = vals[valIndex].split(','); │ │ │ │ │ - style['graphicWidth'] = parseFloat(size[0]); │ │ │ │ │ - style['graphicHeight'] = parseFloat(size[1]); │ │ │ │ │ - } else if (columns[valIndex] == 'iconOffset' && style) { │ │ │ │ │ - var offset = vals[valIndex].split(','); │ │ │ │ │ - style['graphicXOffset'] = parseFloat(offset[0]); │ │ │ │ │ - style['graphicYOffset'] = parseFloat(offset[1]); │ │ │ │ │ - } else if (columns[valIndex] == 'description') { │ │ │ │ │ - attributes['description'] = vals[valIndex]; │ │ │ │ │ - } else if (columns[valIndex] == 'overflow') { │ │ │ │ │ - attributes['overflow'] = vals[valIndex]; │ │ │ │ │ - } else { │ │ │ │ │ - // For StyleMap filtering, allow additional │ │ │ │ │ - // columns to be stored as attributes. │ │ │ │ │ - attributes[columns[valIndex]] = vals[valIndex]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (set) { │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry.transform(this.externalProjection, │ │ │ │ │ - this.internalProjection); │ │ │ │ │ - } │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry, attributes, style); │ │ │ │ │ - features.push(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return features; │ │ │ │ │ - }, │ │ │ │ │ + 'Base Layer': "Base Layer", │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.Text" │ │ │ │ │ -}); │ │ │ │ │ + 'noFID': "Can't update a feature for which there is no FID.", │ │ │ │ │ + │ │ │ │ │ + 'browserNotSupported': "Your browser does not support vector rendering. Currently supported renderers are:\n${renderers}", │ │ │ │ │ + │ │ │ │ │ + // console message │ │ │ │ │ + 'minZoomLevelError': "The minZoomLevel property is only intended for use " + │ │ │ │ │ + "with the FixedZoomLevels-descendent layers. That this " + │ │ │ │ │ + "wfs layer checks for minZoomLevel is a relic of the" + │ │ │ │ │ + "past. We cannot, however, remove it without possibly " + │ │ │ │ │ + "breaking OL based applications that may depend on it." + │ │ │ │ │ + " Therefore we are deprecating it -- the minZoomLevel " + │ │ │ │ │ + "check below will be removed at 3.0. Please instead " + │ │ │ │ │ + "use min/max resolution setting as described here: " + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + │ │ │ │ │ + 'commitSuccess': "WFS Transaction: SUCCESS ${response}", │ │ │ │ │ + │ │ │ │ │ + 'commitFailed': "WFS Transaction: FAILED ${response}", │ │ │ │ │ + │ │ │ │ │ + 'googleWarning': "The Google Layer was unable to load correctly.<br><br>" + │ │ │ │ │ + "To get rid of this message, select a new BaseLayer " + │ │ │ │ │ + "in the layer switcher in the upper-right corner.<br><br>" + │ │ │ │ │ + "Most likely, this is because the Google Maps library " + │ │ │ │ │ + "script was either not included, or does not contain the " + │ │ │ │ │ + "correct API key for your site.<br><br>" + │ │ │ │ │ + "Developers: For help getting this working correctly, " + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ + "target='_blank'>click here</a>", │ │ │ │ │ + │ │ │ │ │ + 'getLayerWarning': "The ${layerType} Layer was unable to load correctly.<br><br>" + │ │ │ │ │ + "To get rid of this message, select a new BaseLayer " + │ │ │ │ │ + "in the layer switcher in the upper-right corner.<br><br>" + │ │ │ │ │ + "Most likely, this is because the ${layerLib} library " + │ │ │ │ │ + "script was not correctly included.<br><br>" + │ │ │ │ │ + "Developers: For help getting this working correctly, " + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ + "target='_blank'>click here</a>", │ │ │ │ │ + │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Scale = 1 : ${scaleDenom}", │ │ │ │ │ + │ │ │ │ │ + //labels for the graticule control │ │ │ │ │ + 'W': 'W', │ │ │ │ │ + 'E': 'E', │ │ │ │ │ + 'N': 'N', │ │ │ │ │ + 'S': 'S', │ │ │ │ │ + 'Graticule': 'Graticule', │ │ │ │ │ + │ │ │ │ │ + // console message │ │ │ │ │ + 'reprojectDeprecated': "You are using the 'reproject' option " + │ │ │ │ │ + "on the ${layerName} layer. This option is deprecated: " + │ │ │ │ │ + "its use was designed to support displaying data over commercial " + │ │ │ │ │ + "basemaps, but that functionality should now be achieved by using " + │ │ │ │ │ + "Spherical Mercator support. More information is available from " + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + │ │ │ │ │ + // console message │ │ │ │ │ + 'methodDeprecated': "This method has been deprecated and will be removed in 3.0. " + │ │ │ │ │ + "Please use ${newMethod} instead.", │ │ │ │ │ + │ │ │ │ │ + // **** end **** │ │ │ │ │ + 'end': '' │ │ │ │ │ + │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Text.js │ │ │ │ │ + OpenLayers/Lang/en-CA.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang/en.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["en-CA"] │ │ │ │ │ + * Dictionary for English-CA. This dictionary inherits from the standard │ │ │ │ │ + * English dictionary. Override only those entries with language specific │ │ │ │ │ + * to the CA region. │ │ │ │ │ + * │ │ │ │ │ + * Keys for entries are used in calls to <OpenLayers.Lang.translate>. Entry │ │ │ │ │ + * bodies are normal strings or strings formatted for use with │ │ │ │ │ + * <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang['en-CA'] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + │ │ │ │ │ + // add any entries specific for this region here │ │ │ │ │ + // e.g. │ │ │ │ │ + // "someKey": "Some regionally specific value" │ │ │ │ │ │ │ │ │ │ +}, OpenLayers.Lang["en"]); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/nb.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Markers.js │ │ │ │ │ - * @requires OpenLayers/Format/Text.js │ │ │ │ │ - * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.Text │ │ │ │ │ - * This layer creates markers given data in a text file. The <location> │ │ │ │ │ - * property of the layer (specified as a property of the options argument │ │ │ │ │ - * in the <OpenLayers.Layer.Text> constructor) points to a tab delimited │ │ │ │ │ - * file with data used to create markers. │ │ │ │ │ - * │ │ │ │ │ - * The first row of the data file should be a header line with the column names │ │ │ │ │ - * of the data. Each column should be delimited by a tab space. The │ │ │ │ │ - * possible columns are: │ │ │ │ │ - * - *point* lat,lon of the point where a marker is to be placed │ │ │ │ │ - * - *lat* Latitude of the point where a marker is to be placed │ │ │ │ │ - * - *lon* Longitude of the point where a marker is to be placed │ │ │ │ │ - * - *icon* or *image* URL of marker icon to use. │ │ │ │ │ - * - *iconSize* Size of Icon to use. │ │ │ │ │ - * - *iconOffset* Where the top-left corner of the icon is to be placed │ │ │ │ │ - * relative to the latitude and longitude of the point. │ │ │ │ │ - * - *title* The text of the 'title' is placed inside an 'h2' marker │ │ │ │ │ - * inside a popup, which opens when the marker is clicked. │ │ │ │ │ - * - *description* The text of the 'description' is placed below the h2 │ │ │ │ │ - * in the popup. this can be plain text or HTML. │ │ │ │ │ - * │ │ │ │ │ - * Example text file: │ │ │ │ │ - * (code) │ │ │ │ │ - * lat lon title description iconSize iconOffset icon │ │ │ │ │ - * 10 20 title description 21,25 -10,-25 http://www.openlayers.org/dev/img/marker.png │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Markers> │ │ │ │ │ + * Namespace: OpenLayers.Lang["nb"] │ │ │ │ │ + * Dictionary for norwegian bokmål (Norway). Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Text = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ +OpenLayers.Lang["nb"] = { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: location │ │ │ │ │ - * {String} URL of text file. Must be specified in the "options" argument │ │ │ │ │ - * of the constructor. Can not be changed once passed in. │ │ │ │ │ - */ │ │ │ │ │ - location: null, │ │ │ │ │ + 'unhandledRequest': "Ubehandlet forespørsel returnerte ${statusText}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Array(<OpenLayers.Feature>)} │ │ │ │ │ - */ │ │ │ │ │ - features: null, │ │ │ │ │ + 'Permalink': "Kobling til denne siden", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: formatOptions │ │ │ │ │ - * {Object} Hash of options which should be passed to the format when it is │ │ │ │ │ - * created. Must be passed in the constructor. │ │ │ │ │ - */ │ │ │ │ │ - formatOptions: null, │ │ │ │ │ + 'Overlays': "Kartlag", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: selectedFeature │ │ │ │ │ - * {<OpenLayers.Feature>} │ │ │ │ │ - */ │ │ │ │ │ - selectedFeature: null, │ │ │ │ │ + 'Base Layer': "Bakgrunnskart", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Text │ │ │ │ │ - * Create a text layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * options - {Object} Object with properties to be set on the layer. │ │ │ │ │ - * Must include <location> property. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.features = []; │ │ │ │ │ - }, │ │ │ │ │ + 'noFID': "Kan ikke oppdatere et feature (et objekt) som ikke har FID.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - // Warning: Layer.Markers.destroy() must be called prior to calling │ │ │ │ │ - // clearFeatures() here, otherwise we leak memory. Indeed, if │ │ │ │ │ - // Layer.Markers.destroy() is called after clearFeatures(), it won't be │ │ │ │ │ - // able to remove the marker image elements from the layer's div since │ │ │ │ │ - // the markers will have been destroyed by clearFeatures(). │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.destroy.apply(this, arguments); │ │ │ │ │ - this.clearFeatures(); │ │ │ │ │ - this.features = null; │ │ │ │ │ - }, │ │ │ │ │ + 'browserNotSupported': "Din nettleser støtter ikke vektortegning. Tegnemetodene som støttes er:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: loadText │ │ │ │ │ - * Start the load of the Text data. Don't do this when we first add the layer, │ │ │ │ │ - * since we may not be visible at any point, and it would therefore be a waste. │ │ │ │ │ - */ │ │ │ │ │ - loadText: function() { │ │ │ │ │ - if (!this.loaded) { │ │ │ │ │ - if (this.location != null) { │ │ │ │ │ + // console message │ │ │ │ │ + 'minZoomLevelError': "Egenskapen minZoomLevel er kun ment til bruk på lag " + │ │ │ │ │ + "basert på FixedZoomLevels. At dette wfs-laget sjekker " + │ │ │ │ │ + "minZoomLevel er en etterlevning fra tidligere versjoner. Det kan dog ikke " + │ │ │ │ │ + "tas bort uten å risikere at OL-baserte applikasjoner " + │ │ │ │ │ + "slutter å virke, så det er merket som foreldet: " + │ │ │ │ │ + "minZoomLevel i sjekken nedenfor vil fjernes i 3.0. " + │ │ │ │ │ + "Vennligst bruk innstillingene for min/maks oppløsning " + │ │ │ │ │ + "som er beskrevet her: " + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ │ │ │ │ │ - var onFail = function(e) { │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ - }; │ │ │ │ │ + 'commitSuccess': "WFS-transaksjon: LYKTES ${response}", │ │ │ │ │ │ │ │ │ │ - this.events.triggerEvent("loadstart"); │ │ │ │ │ - OpenLayers.Request.GET({ │ │ │ │ │ - url: this.location, │ │ │ │ │ - success: this.parseData, │ │ │ │ │ - failure: onFail, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.loaded = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'commitFailed': "WFS-transaksjon: MISLYKTES ${response}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * If layer is visible and Text has not been loaded, load Text. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {Object} │ │ │ │ │ - * zoomChanged - {Object} │ │ │ │ │ - * minor - {Object} │ │ │ │ │ - */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, minor) { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - if (this.visibility && !this.loaded) { │ │ │ │ │ - this.loadText(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'googleWarning': "Google-laget kunne ikke lastes.<br><br>" + │ │ │ │ │ + "Bytt til et annet bakgrunnslag i lagvelgeren i " + │ │ │ │ │ + "øvre høyre hjørne for å slippe denne meldingen.<br><br>" + │ │ │ │ │ + "Sannsynligvis forårsakes feilen av at Google Maps-biblioteket " + │ │ │ │ │ + "ikke er riktig inkludert på nettsiden, eller at det ikke er " + │ │ │ │ │ + "angitt riktig API-nøkkel for nettstedet.<br><br>" + │ │ │ │ │ + "Utviklere: For hjelp til å få dette til å virke se " + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ + "target='_blank'>her</a>.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseData │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * ajaxRequest - {<OpenLayers.Request.XMLHttpRequest>} │ │ │ │ │ - */ │ │ │ │ │ - parseData: function(ajaxRequest) { │ │ │ │ │ - var text = ajaxRequest.responseText; │ │ │ │ │ + 'getLayerWarning': "${layerType}-laget kunne ikke lastes.<br><br>" + │ │ │ │ │ + "Bytt til et annet bakgrunnslag i lagvelgeren i " + │ │ │ │ │ + "øvre høyre hjørne for å slippe denne meldingen.<br><br>" + │ │ │ │ │ + "Sannsynligvis forårsakes feilen av at " + │ │ │ │ │ + "${layerLib}-biblioteket ikke var riktig inkludert " + │ │ │ │ │ + "på nettsiden.<br><br>" + │ │ │ │ │ + "Utviklere: For hjelp til å få dette til å virke se " + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ + "target='_blank'>her</a>.", │ │ │ │ │ │ │ │ │ │ - var options = {}; │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "<strong>Skala</strong> 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - OpenLayers.Util.extend(options, this.formatOptions); │ │ │ │ │ + // console message │ │ │ │ │ + 'reprojectDeprecated': "Du bruker innstillingen 'reproject' på laget ${layerName}. " + │ │ │ │ │ + "Denne innstillingen er foreldet, den var ment for å støtte " + │ │ │ │ │ + "visning av kartdata over kommersielle bakgrunnskart, men det " + │ │ │ │ │ + "bør nå gjøres med støtten for Spherical Mercator. Mer informasjon " + │ │ │ │ │ + "finnes på http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ │ │ │ │ │ - if (this.map && !this.projection.equals(this.map.getProjectionObject())) { │ │ │ │ │ - options.externalProjection = this.projection; │ │ │ │ │ - options.internalProjection = this.map.getProjectionObject(); │ │ │ │ │ - } │ │ │ │ │ + // console message │ │ │ │ │ + 'methodDeprecated': "Denne metoden er markert som foreldet og vil bli fjernet i 3.0. " + │ │ │ │ │ + "Vennligst bruk ${newMethod} i stedet.", │ │ │ │ │ │ │ │ │ │ - var parser = new OpenLayers.Format.Text(options); │ │ │ │ │ - var features = parser.read(text); │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - var data = {}; │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - var location; │ │ │ │ │ - var iconSize, iconOffset; │ │ │ │ │ + 'end': '' │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ - location = new OpenLayers.LonLat(feature.geometry.x, │ │ │ │ │ - feature.geometry.y); │ │ │ │ │ +OpenLayers.Lang["no"] = OpenLayers.Lang["nb"]; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/sk.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - if (feature.style.graphicWidth && │ │ │ │ │ - feature.style.graphicHeight) { │ │ │ │ │ - iconSize = new OpenLayers.Size( │ │ │ │ │ - feature.style.graphicWidth, │ │ │ │ │ - feature.style.graphicHeight); │ │ │ │ │ - } │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Helix84 │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // FIXME: At the moment, we only use this if we have an │ │ │ │ │ - // externalGraphic, because icon has no setOffset API Method. │ │ │ │ │ - /** │ │ │ │ │ - * FIXME FIRST!! │ │ │ │ │ - * The Text format does all sorts of parseFloating │ │ │ │ │ - * The result of a parseFloat for a bogus string is NaN. That │ │ │ │ │ - * means the three possible values here are undefined, NaN, or a │ │ │ │ │ - * number. The previous check was an identity check for null. This │ │ │ │ │ - * means it was failing for all undefined or NaN. A slightly better │ │ │ │ │ - * check is for undefined. An even better check is to see if the │ │ │ │ │ - * value is a number (see #1441). │ │ │ │ │ - */ │ │ │ │ │ - if (feature.style.graphicXOffset !== undefined && │ │ │ │ │ - feature.style.graphicYOffset !== undefined) { │ │ │ │ │ - iconOffset = new OpenLayers.Pixel( │ │ │ │ │ - feature.style.graphicXOffset, │ │ │ │ │ - feature.style.graphicYOffset); │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (feature.style.externalGraphic != null) { │ │ │ │ │ - data.icon = new OpenLayers.Icon(feature.style.externalGraphic, │ │ │ │ │ - iconSize, │ │ │ │ │ - iconOffset); │ │ │ │ │ - } else { │ │ │ │ │ - data.icon = OpenLayers.Marker.defaultIcon(); │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["sk"] │ │ │ │ │ + * Dictionary for Slovenčina. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["sk"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - //allows for the case where the image url is not │ │ │ │ │ - // specified but the size is. use a default icon │ │ │ │ │ - // but change the size │ │ │ │ │ - if (iconSize != null) { │ │ │ │ │ - data.icon.setSize(iconSize); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + 'unhandledRequest': "Neobslúžené požiadavky vracajú ${statusText}", │ │ │ │ │ │ │ │ │ │ - if ((feature.attributes.title != null) && │ │ │ │ │ - (feature.attributes.description != null)) { │ │ │ │ │ - data['popupContentHTML'] = │ │ │ │ │ - '<h2>' + feature.attributes.title + '</h2>' + │ │ │ │ │ - '<p>' + feature.attributes.description + '</p>'; │ │ │ │ │ - } │ │ │ │ │ + 'Permalink': "Trvalý odkaz", │ │ │ │ │ │ │ │ │ │ - data['overflow'] = feature.attributes.overflow || "auto"; │ │ │ │ │ + 'Overlays': "Prekrytia", │ │ │ │ │ │ │ │ │ │ - var markerFeature = new OpenLayers.Feature(this, location, data); │ │ │ │ │ - this.features.push(markerFeature); │ │ │ │ │ - var marker = markerFeature.createMarker(); │ │ │ │ │ - if ((feature.attributes.title != null) && │ │ │ │ │ - (feature.attributes.description != null)) { │ │ │ │ │ - marker.events.register('click', markerFeature, this.markerClick); │ │ │ │ │ - } │ │ │ │ │ - this.addMarker(marker); │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ - }, │ │ │ │ │ + 'Base Layer': "Základná vrstva", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: markerClick │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Context: │ │ │ │ │ - * - {<OpenLayers.Feature>} │ │ │ │ │ - */ │ │ │ │ │ - markerClick: function(evt) { │ │ │ │ │ - var sameMarkerClicked = (this == this.layer.selectedFeature); │ │ │ │ │ - this.layer.selectedFeature = (!sameMarkerClicked) ? this : null; │ │ │ │ │ - for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ - this.layer.map.removePopup(this.layer.map.popups[i]); │ │ │ │ │ - } │ │ │ │ │ - if (!sameMarkerClicked) { │ │ │ │ │ - this.layer.map.addPopup(this.createPopup()); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - }, │ │ │ │ │ + 'noFID': "Nie je možné aktualizovať vlastnosť, pre ktorú neexistuje FID.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clearFeatures │ │ │ │ │ - */ │ │ │ │ │ - clearFeatures: function() { │ │ │ │ │ - if (this.features != null) { │ │ │ │ │ - while (this.features.length > 0) { │ │ │ │ │ - var feature = this.features[0]; │ │ │ │ │ - OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ - feature.destroy(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + 'browserNotSupported': "Váš prehliadač nepodporuje vykresľovanie vektorov. Momentálne podporované vykresľovače sú:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Text" │ │ │ │ │ + 'minZoomLevelError': "Vlastnosť minZoomLevel je určený iba na použitie s vrstvami odvodenými od FixedZoomLevels. To, že táto wfs vrstva kontroluje minZoomLevel je pozostatok z minulosti. Nemôžeme ho však odstrániť, aby sme sa vyhli možnému porušeniu aplikácií založených na Open Layers, ktoré na tomto môže závisieť. Preto ho označujeme ako zavrhovaný - dolu uvedená kontrola minZoomLevel bude odstránená vo verzii 3.0. Použite prosím namiesto toho kontrolu min./max. rozlíšenia podľa tu uvedeného popisu: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + │ │ │ │ │ + 'commitSuccess': "Transakcia WFS: ÚSPEŠNÁ ${response}", │ │ │ │ │ + │ │ │ │ │ + 'commitFailed': "Transakcia WFS: ZLYHALA ${response}", │ │ │ │ │ + │ │ │ │ │ + 'googleWarning': "Vrstvu Google nebolo možné správne načítať.\x3cbr\x3e\x3cbr\x3eAby ste sa tejto správy zbavili vyberte novú BaseLayer v prepínači vrstiev v pravom hornom rohu.\x3cbr\x3e\x3cbr\x3eToto sa stalo pravdepodobne preto, že skript knižnice Google Maps buď nebol načítaný alebo neobsahuje správny kľúč API pre vašu lokalitu.\x3cbr\x3e\x3cbr\x3eVývojári: Tu môžete získať \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3epomoc so sfunkčnením\x3c/a\x3e", │ │ │ │ │ + │ │ │ │ │ + 'getLayerWarning': "Vrstvu ${layerType} nebolo možné správne načítať.\x3cbr\x3e\x3cbr\x3eAby ste sa tejto správy zbavili vyberte novú BaseLayer v prepínači vrstiev v pravom hornom rohu.\x3cbr\x3e\x3cbr\x3eToto sa stalo pravdepodobne preto, že skript knižnice ${layerType} buď nebol načítaný alebo neobsahuje správny kľúč API pre vašu lokalitu.\x3cbr\x3e\x3cbr\x3eVývojári: Tu môžete získať \x3ca href=\'http://trac.openlayers.org/wiki/${layerType}\' target=\'_blank\'\x3epomoc so sfunkčnením\x3c/a\x3e", │ │ │ │ │ + │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Mierka = 1 : ${scaleDenom}", │ │ │ │ │ + │ │ │ │ │ + 'reprojectDeprecated': "Používate voľby „reproject“ vrstvy ${layerType}. Táto voľba je zzavrhovaná: jej použitie bolo navrhnuté na podporu zobrazovania údajov nad komerčnými základovými mapami, ale túto funkcionalitu je teraz možné dosiahnuť pomocou Spherical Mercator. Ďalšie informácie získate na stránke http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + │ │ │ │ │ + 'methodDeprecated': "Táto metóda je zavrhovaná a bude odstránená vo verzii 3.0. Použite prosím namiesto nej metódu ${newMethod}." │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Bing.js │ │ │ │ │ + OpenLayers/Lang/te.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Veeven │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.Bing │ │ │ │ │ - * Bing layer using direct tile access as provided by Bing Maps REST Services. │ │ │ │ │ - * See http://msdn.microsoft.com/en-us/library/ff701713.aspx for more │ │ │ │ │ - * information. Note: Terms of Service compliant use requires the map to be │ │ │ │ │ - * configured with an <OpenLayers.Control.Attribution> control and the │ │ │ │ │ - * attribution placed on or near the map. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.XYZ> │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["te"] │ │ │ │ │ + * Dictionary for తెలుగు. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Bing = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ +OpenLayers.Lang["te"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: key │ │ │ │ │ - * {String} API key for Bing maps, get your own key │ │ │ │ │ - * at http://bingmapsportal.com/ . │ │ │ │ │ - */ │ │ │ │ │ - key: null, │ │ │ │ │ + 'Permalink': "స్థిరలింకు", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: serverResolutions │ │ │ │ │ - * {Array} the resolutions provided by the Bing servers. │ │ │ │ │ - */ │ │ │ │ │ - serverResolutions: [ │ │ │ │ │ - 156543.03390625, 78271.516953125, 39135.7584765625, │ │ │ │ │ - 19567.87923828125, 9783.939619140625, 4891.9698095703125, │ │ │ │ │ - 2445.9849047851562, 1222.9924523925781, 611.4962261962891, │ │ │ │ │ - 305.74811309814453, 152.87405654907226, 76.43702827453613, │ │ │ │ │ - 38.218514137268066, 19.109257068634033, 9.554628534317017, │ │ │ │ │ - 4.777314267158508, 2.388657133579254, 1.194328566789627, │ │ │ │ │ - 0.5971642833948135, 0.29858214169740677, 0.14929107084870338, │ │ │ │ │ - 0.07464553542435169 │ │ │ │ │ - ], │ │ │ │ │ + 'W': "ప", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: attributionTemplate │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - attributionTemplate: '<span class="olBingAttribution ${type}">' + │ │ │ │ │ - '<div><a target="_blank" href="http://www.bing.com/maps/">' + │ │ │ │ │ - '<img src="${logo}" /></a></div>${copyrights}' + │ │ │ │ │ - '<a style="white-space: nowrap" target="_blank" ' + │ │ │ │ │ - 'href="http://www.microsoft.com/maps/product/terms.html">' + │ │ │ │ │ - 'Terms of Use</a></span>', │ │ │ │ │ + 'E': "తూ", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: metadata │ │ │ │ │ - * {Object} Metadata for this layer, as returned by the callback script │ │ │ │ │ - */ │ │ │ │ │ - metadata: null, │ │ │ │ │ + 'N': "ఉ", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: protocolRegex │ │ │ │ │ - * {RegExp} Regular expression to match and replace http: in bing urls │ │ │ │ │ - */ │ │ │ │ │ - protocolRegex: /^http:/i, │ │ │ │ │ + 'S': "ద" │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: type │ │ │ │ │ - * {String} The layer identifier. Any non-birdseye imageryType │ │ │ │ │ - * from http://msdn.microsoft.com/en-us/library/ff701716.aspx can be │ │ │ │ │ - * used. Default is "Road". │ │ │ │ │ - */ │ │ │ │ │ - type: "Road", │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/ksh.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: culture │ │ │ │ │ - * {String} The culture identifier. See http://msdn.microsoft.com/en-us/library/ff701709.aspx │ │ │ │ │ - * for the definition and the possible values. Default is "en-US". │ │ │ │ │ - */ │ │ │ │ │ - culture: "en-US", │ │ │ │ │ +/* Translators (2009 onwards): │ │ │ │ │ + * - Purodha │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: metadataParams │ │ │ │ │ - * {Object} Optional url parameters for the Get Imagery Metadata request │ │ │ │ │ - * as described here: http://msdn.microsoft.com/en-us/library/ff701716.aspx │ │ │ │ │ - */ │ │ │ │ │ - metadataParams: null, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** APIProperty: tileOptions │ │ │ │ │ - * {Object} optional configuration options for <OpenLayers.Tile> instances │ │ │ │ │ - * created by this Layer. Default is │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * {crossOriginKeyword: 'anonymous'} │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ - tileOptions: null, │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["ksh"] │ │ │ │ │ + * Dictionary for Ripoarisch. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang["ksh"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ │ │ │ │ │ - /** APIProperty: protocol │ │ │ │ │ - * {String} Protocol to use to fetch Imagery Metadata, tiles and bing logo │ │ │ │ │ - * Can be 'http:' 'https:' or '' │ │ │ │ │ - * │ │ │ │ │ - * Warning: tiles may not be available under both HTTP and HTTPS protocols. │ │ │ │ │ - * Microsoft approved use of both HTTP and HTTPS urls for tiles. However │ │ │ │ │ - * this is undocumented and the Imagery Metadata API always returns HTTP │ │ │ │ │ - * urls. │ │ │ │ │ - * │ │ │ │ │ - * Default is '', unless when executed from a file:/// uri, in which case │ │ │ │ │ - * it is 'http:'. │ │ │ │ │ - */ │ │ │ │ │ - protocol: ~window.location.href.indexOf('http') ? '' : 'http:', │ │ │ │ │ + 'unhandledRequest': "Met dä Antwoot op en Aanfrooch ham_mer nix aanjefange: ${statusText}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Bing │ │ │ │ │ - * Create a new Bing layer. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var road = new OpenLayers.Layer.Bing({ │ │ │ │ │ - * name: "My Bing Aerial Layer", │ │ │ │ │ - * type: "Aerial", │ │ │ │ │ - * key: "my-api-key-here", │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Configuration properties for the layer. │ │ │ │ │ - * │ │ │ │ │ - * Required configuration properties: │ │ │ │ │ - * key - {String} Bing Maps API key for your application. Get one at │ │ │ │ │ - * http://bingmapsportal.com/. │ │ │ │ │ - * type - {String} The layer identifier. Any non-birdseye imageryType │ │ │ │ │ - * from http://msdn.microsoft.com/en-us/library/ff701716.aspx can be │ │ │ │ │ - * used. │ │ │ │ │ - * │ │ │ │ │ - * Any other documented layer properties can be provided in the config object. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - sphericalMercator: true │ │ │ │ │ - }, options); │ │ │ │ │ - var name = options.name || "Bing " + (options.type || this.type); │ │ │ │ │ + 'Permalink': "Lengk op Duuer", │ │ │ │ │ │ │ │ │ │ - var newArgs = [name, null, options]; │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, newArgs); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - crossOriginKeyword: 'anonymous' │ │ │ │ │ - }, this.options.tileOptions); │ │ │ │ │ - this.loadMetadata(); │ │ │ │ │ - }, │ │ │ │ │ + 'Overlays': "Drövver jelaat", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: loadMetadata │ │ │ │ │ - */ │ │ │ │ │ - loadMetadata: function() { │ │ │ │ │ - this._callbackId = "_callback_" + this.id.replace(/\./g, "_"); │ │ │ │ │ - // link the processMetadata method to the global scope and bind it │ │ │ │ │ - // to this instance │ │ │ │ │ - window[this._callbackId] = OpenLayers.Function.bind( │ │ │ │ │ - OpenLayers.Layer.Bing.processMetadata, this │ │ │ │ │ - ); │ │ │ │ │ - var params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - key: this.key, │ │ │ │ │ - jsonp: this._callbackId, │ │ │ │ │ - include: "ImageryProviders" │ │ │ │ │ - }, this.metadataParams); │ │ │ │ │ - var url = this.protocol + "//dev.virtualearth.net/REST/v1/Imagery/Metadata/" + │ │ │ │ │ - this.type + "?" + OpenLayers.Util.getParameterString(params); │ │ │ │ │ - var script = document.createElement("script"); │ │ │ │ │ - script.type = "text/javascript"; │ │ │ │ │ - script.src = url; │ │ │ │ │ - script.id = this._callbackId; │ │ │ │ │ - document.getElementsByTagName("head")[0].appendChild(script); │ │ │ │ │ - }, │ │ │ │ │ + 'Base Layer': "Jrund-Nivoh", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: initLayer │ │ │ │ │ - * │ │ │ │ │ - * Sets layer properties according to the metadata provided by the API │ │ │ │ │ - */ │ │ │ │ │ - initLayer: function() { │ │ │ │ │ - var res = this.metadata.resourceSets[0].resources[0]; │ │ │ │ │ - var url = res.imageUrl.replace("{quadkey}", "${quadkey}"); │ │ │ │ │ - url = url.replace("{culture}", this.culture); │ │ │ │ │ - url = url.replace(this.protocolRegex, this.protocol); │ │ │ │ │ - this.url = []; │ │ │ │ │ - for (var i = 0; i < res.imageUrlSubdomains.length; ++i) { │ │ │ │ │ - this.url.push(url.replace("{subdomain}", res.imageUrlSubdomains[i])); │ │ │ │ │ - } │ │ │ │ │ - this.addOptions({ │ │ │ │ │ - maxResolution: Math.min( │ │ │ │ │ - this.serverResolutions[res.zoomMin], │ │ │ │ │ - this.maxResolution || Number.POSITIVE_INFINITY │ │ │ │ │ - ), │ │ │ │ │ - numZoomLevels: Math.min( │ │ │ │ │ - res.zoomMax + 1 - res.zoomMin, this.numZoomLevels │ │ │ │ │ - ) │ │ │ │ │ - }, true); │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - this.redraw(); │ │ │ │ │ - } │ │ │ │ │ - this.updateAttribution(); │ │ │ │ │ - }, │ │ │ │ │ + 'noFID': "En Saach, woh kein \x3ci lang=\"en\"\x3eFID\x3c/i\x3e för doh es, löht sesch nit ändere.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * │ │ │ │ │ - * Paramters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - if (!this.url) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var xyz = this.getXYZ(bounds), │ │ │ │ │ - x = xyz.x, │ │ │ │ │ - y = xyz.y, │ │ │ │ │ - z = xyz.z; │ │ │ │ │ - var quadDigits = []; │ │ │ │ │ - for (var i = z; i > 0; --i) { │ │ │ │ │ - var digit = '0'; │ │ │ │ │ - var mask = 1 << (i - 1); │ │ │ │ │ - if ((x & mask) != 0) { │ │ │ │ │ - digit++; │ │ │ │ │ - } │ │ │ │ │ - if ((y & mask) != 0) { │ │ │ │ │ - digit++; │ │ │ │ │ - digit++; │ │ │ │ │ - } │ │ │ │ │ - quadDigits.push(digit); │ │ │ │ │ - } │ │ │ │ │ - var quadKey = quadDigits.join(""); │ │ │ │ │ - var url = this.selectUrl('' + x + y + z, this.url); │ │ │ │ │ + 'browserNotSupported': "Dinge Brauser kann kein Väktore ußjävve. De Zoote Ußjaabe, di em Momang jon, sen:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - return OpenLayers.String.format(url, { │ │ │ │ │ - 'quadkey': quadKey │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + 'minZoomLevelError': "De Eijeschaff „\x3ccode lang=\"en\"\x3eminZoomLevel\x3c/code\x3e“ es bloß doför jedaach, dat mer se met dä Nivvohß bruch, di vun \x3ccode lang=\"en\"\x3eFixedZoomLevels\x3c/code\x3e affhange don. Dat dat \x3ci lang=\"en\"\x3eWFS\x3c/i\x3e-Nivvoh övverhoup de Eijeschaff „\x3ccode lang=\"en\"\x3eminZoomLevel\x3c/code\x3e“ pröhfe deiht, es noch övveresch vun fröhjer. Mer künne dat ävver jez nit fott lohße, oohne dat mer Jevaa loufe, dat Aanwendunge vun OpenLayers nit mieh loufe, di sesch doh velleijsch noch drop am verlohße sin. Dröm sare mer, dat mer et nit mieh han welle, un de „\x3ccode lang=\"en\"\x3eminZoomLevel\x3c/code\x3e“-Eijeschaff weed hee vun de Version 3.0 af nit mieh jeprööf wäde. Nemm doför de Enstellung för de hühßte un de kleinßte Oplöhsung, esu wi et en http://trac.openlayers.org/wiki/SettingZoomLevels opjeschrevve es.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateAttribution │ │ │ │ │ - * Updates the attribution according to the requirements outlined in │ │ │ │ │ - * http://gis.638310.n2.nabble.com/Bing-imagery-td5789168.html │ │ │ │ │ - */ │ │ │ │ │ - updateAttribution: function() { │ │ │ │ │ - var metadata = this.metadata; │ │ │ │ │ - if (!metadata.resourceSets || !this.map || !this.map.center) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var res = metadata.resourceSets[0].resources[0]; │ │ │ │ │ - var extent = this.map.getExtent().transform( │ │ │ │ │ - this.map.getProjectionObject(), │ │ │ │ │ - new OpenLayers.Projection("EPSG:4326") │ │ │ │ │ - ); │ │ │ │ │ - var providers = res.imageryProviders || [], │ │ │ │ │ - zoom = OpenLayers.Util.indexOf(this.serverResolutions, │ │ │ │ │ - this.getServerResolution()), │ │ │ │ │ - copyrights = "", │ │ │ │ │ - provider, i, ii, j, jj, bbox, coverage; │ │ │ │ │ - for (i = 0, ii = providers.length; i < ii; ++i) { │ │ │ │ │ - provider = providers[i]; │ │ │ │ │ - for (j = 0, jj = provider.coverageAreas.length; j < jj; ++j) { │ │ │ │ │ - coverage = provider.coverageAreas[j]; │ │ │ │ │ - // axis order provided is Y,X │ │ │ │ │ - bbox = OpenLayers.Bounds.fromArray(coverage.bbox, true); │ │ │ │ │ - if (extent.intersectsBounds(bbox) && │ │ │ │ │ - zoom <= coverage.zoomMax && zoom >= coverage.zoomMin) { │ │ │ │ │ - copyrights += provider.attribution + " "; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var logo = metadata.brandLogoUri.replace(this.protocolRegex, this.protocol); │ │ │ │ │ - this.attribution = OpenLayers.String.format(this.attributionTemplate, { │ │ │ │ │ - type: this.type.toLowerCase(), │ │ │ │ │ - logo: logo, │ │ │ │ │ - copyrights: copyrights │ │ │ │ │ - }); │ │ │ │ │ - this.map && this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "attribution" │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + 'commitSuccess': "Dä \x3ci lang=\"en\"\x3eWFS\x3c/i\x3e-Vörjang es joot jeloufe: ${response}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - */ │ │ │ │ │ - setMap: function() { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.register("moveend", this, this.updateAttribution); │ │ │ │ │ - }, │ │ │ │ │ + 'commitFailed': "Dä \x3ci lang=\"en\"\x3eWFS\x3c/i\x3e-Vörjang es scheif jejange: ${response}", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Bing>} An exact clone of this <OpenLayers.Layer.Bing> │ │ │ │ │ - */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Bing(this.options); │ │ │ │ │ - } │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ + 'googleWarning': "Dat Nivvoh \x3ccode lang=\"en\"\x3eGoogle\x3c/code\x3e kunnt nit reschtesch jelaade wääde.\x3cbr /\x3e\x3cbr /\x3eÖm hee di Nohreesch loß ze krijje, donn en ander Jrund-Nivvoh ußsöhke, rähß bovve en de Äk.\x3cbr /\x3e\x3cbr /\x3eWascheinlesch es dat wiel dat \x3ci lang=\"en\"\x3eGoogle-Maps\x3c/i\x3e-Skrepp entweeder nit reschtesch enjebonge wood, udder nit dä reschtejje \x3ci lang=\"en\"\x3eAPI\x3c/i\x3e-Schlößel för Ding Web-ßait scheke deiht.\x3cbr /\x3e\x3cbr /\x3eFör Projrammierer jidd_et Hölp do_drövver, \x3ca href=\"http://trac.openlayers.org/wiki/Google\" target=\"_blank\"\x3ewi mer dat aan et Loufe brengk\x3c/a\x3e.", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.map && │ │ │ │ │ - this.map.events.unregister("moveend", this, this.updateAttribution); │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + 'getLayerWarning': "Dat Nivvoh \x3ccode\x3e${layerType}\x3c/code\x3e kunnt nit reschtesch jelaade wääde.\x3cbr /\x3e\x3cbr /\x3eÖm hee di Nohreesch loß ze krijje, donn en ander Jrund-Nivvoh ußsöhkre, rähß bovve en de Äk.\x3cbr /\x3e\x3cbr /\x3eWascheinlesch es dat, wiel dat Skrepp \x3ccode\x3e${layerLib}\x3c/code\x3e nit reschtesch enjebonge wood.\x3cbr /\x3e\x3cbr /\x3eFör Projrammierer jidd_Et Hölp do_drövver, \x3ca href=\"http://trac.openlayers.org/wiki/${layerLib}\" target=\"_blank\"\x3ewi mer dat aan et Loufe brengk\x3c/a\x3e.", │ │ │ │ │ + │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Mohßshtaab = 1 : ${scaleDenom}", │ │ │ │ │ + │ │ │ │ │ + 'W': "W", │ │ │ │ │ + │ │ │ │ │ + 'E': "O", │ │ │ │ │ + │ │ │ │ │ + 'N': "N", │ │ │ │ │ + │ │ │ │ │ + 'S': "S", │ │ │ │ │ + │ │ │ │ │ + 'reprojectDeprecated': "Do bruchs de Ußwahl \x3ccode\x3ereproject\x3c/code\x3e op däm Nivvoh \x3ccode\x3e${layerName}\x3c/code\x3e. Di Ußwahl es nit mieh jähn jesinn. Se wohr doför jedaach, öm Date op jeschääfsmäßesch eruß jejovve Kaate bovve drop ze moole, wat ävver enzwesche besser met dä Öngershtözung för de ßfääresche Mäkaator Beldscher jeiht. Doh kanns De mieh drövver fenge op dä Sigg: http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + │ │ │ │ │ + 'methodDeprecated': "Hee di Metood es nim_mih aktoäll un et weed se en dä Version 3.0 nit mieh jävve. Nemm \x3ccode\x3e${newMethod}\x3c/code\x3e doföör." │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Bing" │ │ │ │ │ }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Lang/da-DK.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Function: OpenLayers.Layer.Bing.processMetadata │ │ │ │ │ - * This function will be bound to an instance, linked to the global scope with │ │ │ │ │ - * an id, and called by the JSONP script returned by the API. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * metadata - {Object} metadata as returned by the API │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Bing.processMetadata = function(metadata) { │ │ │ │ │ - this.metadata = metadata; │ │ │ │ │ - this.initLayer(); │ │ │ │ │ - var script = document.getElementById(this._callbackId); │ │ │ │ │ - script.parentNode.removeChild(script); │ │ │ │ │ - window[this._callbackId] = undefined; // cannot delete from window in IE │ │ │ │ │ - delete this._callbackId; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Namespace: OpenLayers.Lang["da-DK"] │ │ │ │ │ + * Dictionary for Danish. Keys for entries are used in calls to │ │ │ │ │ + * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ + * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Lang['da-DK'] = { │ │ │ │ │ + │ │ │ │ │ + 'unhandledRequest': "En ikke håndteret forespørgsel returnerede ${statusText}", │ │ │ │ │ + │ │ │ │ │ + 'Permalink': "Permalink", │ │ │ │ │ + │ │ │ │ │ + 'Overlays': "Kortlag", │ │ │ │ │ + │ │ │ │ │ + 'Base Layer': "Baggrundslag", │ │ │ │ │ + │ │ │ │ │ + 'noFID': "Kan ikke opdateret en feature (et objekt) der ikke har et FID.", │ │ │ │ │ + │ │ │ │ │ + 'browserNotSupported': "Din browser understøtter ikke vektor visning. Følgende vektor visninger understøttes:\n${renderers}", │ │ │ │ │ + │ │ │ │ │ + // console message │ │ │ │ │ + 'minZoomLevelError': "Egenskaben minZoomLevel er kun beregnet til brug " + │ │ │ │ │ + "med FixedZoomLevels. At dette WFS lag kontrollerer " + │ │ │ │ │ + "minZoomLevel egenskaben, er et levn fra en tidligere " + │ │ │ │ │ + "version. Vi kan desværre ikke fjerne dette uden at risikere " + │ │ │ │ │ + "at ødelægge eksisterende OL baserede programmer der " + │ │ │ │ │ + " benytter denne funktionalitet. " + │ │ │ │ │ + "Egenskaben bør derfor ikke anvendes, og minZoomLevel " + │ │ │ │ │ + "kontrollen herunder vil blive fjernet i version 3.0. " + │ │ │ │ │ + "Benyt istedet min/max opløsnings indstillingerne, som " + │ │ │ │ │ + "er beskrevet her: " + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + │ │ │ │ │ + 'commitSuccess': "WFS transaktion: LYKKEDES ${response}", │ │ │ │ │ + │ │ │ │ │ + 'commitFailed': "WFS transaktion: MISLYKKEDES ${response}", │ │ │ │ │ + │ │ │ │ │ + 'googleWarning': "Google laget kunne ikke indlæses.<br><br>" + │ │ │ │ │ + "For at fjerne denne besked, vælg et nyt bagrundskort i " + │ │ │ │ │ + "lagskifteren i øverste højre hjørne.<br><br>" + │ │ │ │ │ + "Fejlen skyldes formentlig at Google Maps bibliotekts " + │ │ │ │ │ + "scriptet ikke er inkluderet, eller ikke indeholder den " + │ │ │ │ │ + "korrkte API nøgle for dit site.<br><br>" + │ │ │ │ │ + "Udviklere: For hjælp til at få dette til at fungere, " + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ + "target='_blank'>klik her</a>", │ │ │ │ │ + │ │ │ │ │ + 'getLayerWarning': "${layerType}-laget kunne ikke indlæses.<br><br>" + │ │ │ │ │ + "For at fjerne denne besked, vælg et nyt bagrundskort i " + │ │ │ │ │ + "lagskifteren i øverste højre hjørne.<br><br>" + │ │ │ │ │ + "Fejlen skyldes formentlig at ${layerLib} bibliotekts " + │ │ │ │ │ + "scriptet ikke er inkluderet.<br><br>" + │ │ │ │ │ + "Udviklere: For hjælp til at få dette til at fungere, " + │ │ │ │ │ + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ + "target='_blank'>klik her</a>", │ │ │ │ │ + │ │ │ │ │ + 'Scale = 1 : ${scaleDenom}': "Målforhold = 1 : ${scaleDenom}", │ │ │ │ │ + │ │ │ │ │ + // console message │ │ │ │ │ + 'reprojectDeprecated': "Du anvender indstillingen 'reproject' på laget ${layerName}." + │ │ │ │ │ + "Denne indstilling bør ikke længere anvendes. Den var beregnet " + │ │ │ │ │ + "til at vise data ovenpå kommercielle grundkort, men den funktionalitet " + │ │ │ │ │ + "bør nu opnås ved at anvende Spherical Mercator understøttelsen. " + │ │ │ │ │ + "Mere information er tilgængelig her: " + │ │ │ │ │ + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + │ │ │ │ │ + // console message │ │ │ │ │ + 'methodDeprecated': "Denne funktion bør ikke længere anvendes, og vil blive fjernet i version 3.0. " + │ │ │ │ │ + "Anvend venligst funktionen ${newMethod} istedet." │ │ │ │ │ }; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Tile/UTFGrid.js │ │ │ │ │ + OpenLayers/Handler/Drag.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Tile.js │ │ │ │ │ - * @requires OpenLayers/Format/JSON.js │ │ │ │ │ - * @requires OpenLayers/Request.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Tile.UTFGrid │ │ │ │ │ - * Instances of OpenLayers.Tile.UTFGrid are used to manage │ │ │ │ │ - * UTFGrids. This is an unusual tile type in that it doesn't have a │ │ │ │ │ - * rendered image; only a 'hit grid' that can be used to │ │ │ │ │ - * look up feature attributes. │ │ │ │ │ + * Class: OpenLayers.Handler.Drag │ │ │ │ │ + * The drag handler is used to deal with sequences of browser events related │ │ │ │ │ + * to dragging. The handler is used by controls that want to know when │ │ │ │ │ + * a drag sequence begins, when a drag is happening, and when it has │ │ │ │ │ + * finished. │ │ │ │ │ * │ │ │ │ │ - * See the <OpenLayers.Tile.UTFGrid> constructor for details on constructing a │ │ │ │ │ - * new instance. │ │ │ │ │ + * Controls that use the drag handler typically construct it with callbacks │ │ │ │ │ + * for 'down', 'move', and 'done'. Callbacks for these keys are called │ │ │ │ │ + * when the drag begins, with each move, and when the drag is done. In │ │ │ │ │ + * addition, controls can have callbacks keyed to 'up' and 'out' if they │ │ │ │ │ + * care to differentiate between the types of events that correspond with │ │ │ │ │ + * the end of a drag sequence. If no drag actually occurs (no mouse move) │ │ │ │ │ + * the 'down' and 'up' callbacks will be called, but not the 'done' │ │ │ │ │ + * callback. │ │ │ │ │ + * │ │ │ │ │ + * Create a new drag handler with the <OpenLayers.Handler.Drag> constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Tile> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Tile.UTFGrid = OpenLayers.Class(OpenLayers.Tile, { │ │ │ │ │ +OpenLayers.Handler.Drag = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: url │ │ │ │ │ - * {String} │ │ │ │ │ - * The URL of the UTFGrid file being requested. Provided by the <getURL> │ │ │ │ │ - * method. │ │ │ │ │ + * Property: started │ │ │ │ │ + * {Boolean} When a mousedown or touchstart event is received, we want to │ │ │ │ │ + * record it, but not set 'dragging' until the mouse moves after starting. │ │ │ │ │ */ │ │ │ │ │ - url: null, │ │ │ │ │ + started: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: utfgridResolution │ │ │ │ │ - * {Number} │ │ │ │ │ - * Ratio of the pixel width to the width of a UTFGrid data point. If an │ │ │ │ │ - * entry in the grid represents a 4x4 block of pixels, the │ │ │ │ │ - * utfgridResolution would be 4. Default is 2. │ │ │ │ │ + * Property: stopDown │ │ │ │ │ + * {Boolean} Stop propagation of mousedown events from getting to listeners │ │ │ │ │ + * on the same element. Default is true. │ │ │ │ │ */ │ │ │ │ │ - utfgridResolution: 2, │ │ │ │ │ + stopDown: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: json │ │ │ │ │ - * {Object} │ │ │ │ │ - * Stores the parsed JSON tile data structure. │ │ │ │ │ + * Property: dragging │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - json: null, │ │ │ │ │ + dragging: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: format │ │ │ │ │ - * {OpenLayers.Format.JSON} │ │ │ │ │ - * Parser instance used to parse JSON for cross browser support. The native │ │ │ │ │ - * JSON.parse method will be used where available (all except IE<8). │ │ │ │ │ + * Property: last │ │ │ │ │ + * {<OpenLayers.Pixel>} The last pixel location of the drag. │ │ │ │ │ */ │ │ │ │ │ - format: null, │ │ │ │ │ + last: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Tile.UTFGrid │ │ │ │ │ - * Constructor for a new <OpenLayers.Tile.UTFGrid> instance. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer>} layer that the tile will go in. │ │ │ │ │ - * position - {<OpenLayers.Pixel>} │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * url - {<String>} Deprecated. Remove me in 3.0. │ │ │ │ │ - * size - {<OpenLayers.Size>} │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * Property: start │ │ │ │ │ + * {<OpenLayers.Pixel>} The first pixel location of the drag. │ │ │ │ │ */ │ │ │ │ │ + start: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up. │ │ │ │ │ + /** │ │ │ │ │ + * Property: lastMoveEvt │ │ │ │ │ + * {Object} The last mousemove event that occurred. Used to │ │ │ │ │ + * position the map correctly when our "delay drag" │ │ │ │ │ + * timeout expired. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.clear(); │ │ │ │ │ - OpenLayers.Tile.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + lastMoveEvt: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Check that a tile should be drawn, and draw it. │ │ │ │ │ - * In the case of UTFGrids, "drawing" it means fetching and │ │ │ │ │ - * parsing the json. │ │ │ │ │ + * Property: oldOnselectstart │ │ │ │ │ + * {Function} │ │ │ │ │ + */ │ │ │ │ │ + oldOnselectstart: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: interval │ │ │ │ │ + * {Integer} In order to increase performance, an interval (in │ │ │ │ │ + * milliseconds) can be set to reduce the number of drag events │ │ │ │ │ + * called. If set, a new drag event will not be set until the │ │ │ │ │ + * interval has passed. │ │ │ │ │ + * Defaults to 0, meaning no interval. │ │ │ │ │ + */ │ │ │ │ │ + interval: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: timeoutId │ │ │ │ │ + * {String} The id of the timeout used for the mousedown interval. │ │ │ │ │ + * This is "private", and should be left alone. │ │ │ │ │ + */ │ │ │ │ │ + timeoutId: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: documentDrag │ │ │ │ │ + * {Boolean} If set to true, the handler will also handle mouse moves when │ │ │ │ │ + * the cursor has moved out of the map viewport. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + documentDrag: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: documentEvents │ │ │ │ │ + * {Boolean} Are we currently observing document events? │ │ │ │ │ + */ │ │ │ │ │ + documentEvents: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Handler.Drag │ │ │ │ │ + * Returns OpenLayers.Handler.Drag │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Was a tile drawn? │ │ │ │ │ + * Parameters: │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ + * this handler. If a handler is being used without a control, the │ │ │ │ │ + * handlers setMap method must be overridden to deal properly with │ │ │ │ │ + * the map. │ │ │ │ │ + * callbacks - {Object} An object containing a single function to be │ │ │ │ │ + * called when the drag operation is finished. The callback should │ │ │ │ │ + * expect to recieve a single argument, the pixel location of the event. │ │ │ │ │ + * Callbacks for 'move' and 'done' are supported. You can also speficy │ │ │ │ │ + * callbacks for 'down', 'up', and 'out' to respond to those events. │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - var drawn = OpenLayers.Tile.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (drawn) { │ │ │ │ │ - if (this.isLoading) { │ │ │ │ │ - this.abortLoading(); │ │ │ │ │ - //if we're already loading, send 'reload' instead of 'loadstart'. │ │ │ │ │ - this.events.triggerEvent("reload"); │ │ │ │ │ - } else { │ │ │ │ │ - this.isLoading = true; │ │ │ │ │ - this.events.triggerEvent("loadstart"); │ │ │ │ │ - } │ │ │ │ │ - this.url = this.layer.getURL(this.bounds); │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - if (this.layer.useJSONP) { │ │ │ │ │ - // Use JSONP method to avoid xbrowser policy │ │ │ │ │ - var ols = new OpenLayers.Protocol.Script({ │ │ │ │ │ - url: this.url, │ │ │ │ │ - callback: function(response) { │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ - this.json = response.data; │ │ │ │ │ + if (this.documentDrag === true) { │ │ │ │ │ + var me = this; │ │ │ │ │ + this._docMove = function(evt) { │ │ │ │ │ + me.mousemove({ │ │ │ │ │ + xy: { │ │ │ │ │ + x: evt.clientX, │ │ │ │ │ + y: evt.clientY │ │ │ │ │ }, │ │ │ │ │ - scope: this │ │ │ │ │ + element: document │ │ │ │ │ }); │ │ │ │ │ - ols.read(); │ │ │ │ │ - this.request = ols; │ │ │ │ │ - } else { │ │ │ │ │ - // Use standard XHR │ │ │ │ │ - this.request = OpenLayers.Request.GET({ │ │ │ │ │ - url: this.url, │ │ │ │ │ - callback: function(response) { │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ - if (response.status === 200) { │ │ │ │ │ - this.parseData(response.responseText); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ + }; │ │ │ │ │ + this._docUp = function(evt) { │ │ │ │ │ + me.mouseup({ │ │ │ │ │ + xy: { │ │ │ │ │ + x: evt.clientX, │ │ │ │ │ + y: evt.clientY │ │ │ │ │ + } │ │ │ │ │ }); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.unload(); │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ - return drawn; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: abortLoading │ │ │ │ │ - * Cancel a pending request. │ │ │ │ │ - */ │ │ │ │ │ - abortLoading: function() { │ │ │ │ │ - if (this.request) { │ │ │ │ │ - this.request.abort(); │ │ │ │ │ - delete this.request; │ │ │ │ │ - } │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureInfo │ │ │ │ │ - * Get feature information associated with a pixel offset. If the pixel │ │ │ │ │ - * offset corresponds to a feature, the returned object will have id │ │ │ │ │ - * and data properties. Otherwise, null will be returned. │ │ │ │ │ - * │ │ │ │ │ + * Method: dragstart │ │ │ │ │ + * This private method is factorized from mousedown and touchstart methods │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * i - {Number} X-axis pixel offset (from top left of tile) │ │ │ │ │ - * j - {Number} Y-axis pixel offset (from top left of tile) │ │ │ │ │ + * evt - {Event} The event │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} Object with feature id and data properties corresponding to the │ │ │ │ │ - * given pixel offset. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - getFeatureInfo: function(i, j) { │ │ │ │ │ - var info = null; │ │ │ │ │ - if (this.json) { │ │ │ │ │ - var id = this.getFeatureId(i, j); │ │ │ │ │ - if (id !== null) { │ │ │ │ │ - info = { │ │ │ │ │ - id: id, │ │ │ │ │ - data: this.json.data[id] │ │ │ │ │ - }; │ │ │ │ │ + dragstart: function(evt) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + if (this.checkModifiers(evt) && │ │ │ │ │ + (OpenLayers.Event.isLeftClick(evt) || │ │ │ │ │ + OpenLayers.Event.isSingleTouch(evt))) { │ │ │ │ │ + this.started = true; │ │ │ │ │ + this.start = evt.xy; │ │ │ │ │ + this.last = evt.xy; │ │ │ │ │ + OpenLayers.Element.addClass( │ │ │ │ │ + this.map.viewPortDiv, "olDragDown" │ │ │ │ │ + ); │ │ │ │ │ + this.down(evt); │ │ │ │ │ + this.callback("down", [evt.xy]); │ │ │ │ │ + │ │ │ │ │ + // prevent document dragging │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + │ │ │ │ │ + if (!this.oldOnselectstart) { │ │ │ │ │ + this.oldOnselectstart = document.onselectstart ? │ │ │ │ │ + document.onselectstart : OpenLayers.Function.True; │ │ │ │ │ } │ │ │ │ │ + document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ + │ │ │ │ │ + propagate = !this.stopDown; │ │ │ │ │ + } else { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ } │ │ │ │ │ - return info; │ │ │ │ │ + return propagate; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureId │ │ │ │ │ - * Get the identifier for the feature associated with a pixel offset. │ │ │ │ │ + * Method: dragmove │ │ │ │ │ + * This private method is factorized from mousemove and touchmove methods │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * i - {Number} X-axis pixel offset (from top left of tile) │ │ │ │ │ - * j - {Number} Y-axis pixel offset (from top left of tile) │ │ │ │ │ + * evt - {Event} The event │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} The feature identifier corresponding to the given pixel offset. │ │ │ │ │ - * Returns null if pixel doesn't correspond to a feature. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - getFeatureId: function(i, j) { │ │ │ │ │ - var id = null; │ │ │ │ │ - if (this.json) { │ │ │ │ │ - var resolution = this.utfgridResolution; │ │ │ │ │ - var row = Math.floor(j / resolution); │ │ │ │ │ - var col = Math.floor(i / resolution); │ │ │ │ │ - var charCode = this.json.grid[row].charCodeAt(col); │ │ │ │ │ - var index = this.indexFromCharCode(charCode); │ │ │ │ │ - var keys = this.json.keys; │ │ │ │ │ - if (!isNaN(index) && (index in keys)) { │ │ │ │ │ - id = keys[index]; │ │ │ │ │ + dragmove: function(evt) { │ │ │ │ │ + this.lastMoveEvt = evt; │ │ │ │ │ + if (this.started && !this.timeoutId && (evt.xy.x != this.last.x || │ │ │ │ │ + evt.xy.y != this.last.y)) { │ │ │ │ │ + if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ + if (evt.element === document) { │ │ │ │ │ + this.adjustXY(evt); │ │ │ │ │ + // do setEvent manually because the documentEvents are not │ │ │ │ │ + // registered with the map │ │ │ │ │ + this.setEvent(evt); │ │ │ │ │ + } else { │ │ │ │ │ + this.removeDocumentEvents(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.interval > 0) { │ │ │ │ │ + this.timeoutId = setTimeout( │ │ │ │ │ + OpenLayers.Function.bind(this.removeTimeout, this), │ │ │ │ │ + this.interval); │ │ │ │ │ + } │ │ │ │ │ + this.dragging = true; │ │ │ │ │ + │ │ │ │ │ + this.move(evt); │ │ │ │ │ + this.callback("move", [evt.xy]); │ │ │ │ │ + if (!this.oldOnselectstart) { │ │ │ │ │ + this.oldOnselectstart = document.onselectstart; │ │ │ │ │ + document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ } │ │ │ │ │ + this.last = evt.xy; │ │ │ │ │ } │ │ │ │ │ - return id; │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: indexFromCharCode │ │ │ │ │ - * Given a character code for one of the UTFGrid "grid" characters, │ │ │ │ │ - * resolve the integer index for the feature id in the UTFGrid "keys" │ │ │ │ │ - * array. │ │ │ │ │ + * Method: dragend │ │ │ │ │ + * This private method is factorized from mouseup and touchend methods │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * charCode - {Integer} │ │ │ │ │ + * evt - {Event} The event │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Integer} Index for the feature id from the keys array. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - indexFromCharCode: function(charCode) { │ │ │ │ │ - if (charCode >= 93) { │ │ │ │ │ - charCode--; │ │ │ │ │ - } │ │ │ │ │ - if (charCode >= 35) { │ │ │ │ │ - charCode--; │ │ │ │ │ + dragend: function(evt) { │ │ │ │ │ + if (this.started) { │ │ │ │ │ + if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ + this.adjustXY(evt); │ │ │ │ │ + this.removeDocumentEvents(); │ │ │ │ │ + } │ │ │ │ │ + var dragged = (this.start != this.last); │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + OpenLayers.Element.removeClass( │ │ │ │ │ + this.map.viewPortDiv, "olDragDown" │ │ │ │ │ + ); │ │ │ │ │ + this.up(evt); │ │ │ │ │ + this.callback("up", [evt.xy]); │ │ │ │ │ + if (dragged) { │ │ │ │ │ + this.callback("done", [evt.xy]); │ │ │ │ │ + } │ │ │ │ │ + document.onselectstart = this.oldOnselectstart; │ │ │ │ │ } │ │ │ │ │ - return charCode - 32; │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseData │ │ │ │ │ - * Parse the JSON from a request │ │ │ │ │ + * The four methods below (down, move, up, and out) are used by subclasses │ │ │ │ │ + * to do their own processing related to these mouse events. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: down │ │ │ │ │ + * This method is called during the handling of the mouse down event. │ │ │ │ │ + * Subclasses can do their own processing here. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * str - {String} UTFGrid as a JSON string. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} parsed javascript data │ │ │ │ │ + * evt - {Event} The mouse down event │ │ │ │ │ */ │ │ │ │ │ - parseData: function(str) { │ │ │ │ │ - if (!this.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.JSON(); │ │ │ │ │ - } │ │ │ │ │ - this.json = this.format.read(str); │ │ │ │ │ - }, │ │ │ │ │ + down: function(evt) {}, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clear │ │ │ │ │ - * Delete data stored with this tile. │ │ │ │ │ + /** │ │ │ │ │ + * Method: move │ │ │ │ │ + * This method is called during the handling of the mouse move event. │ │ │ │ │ + * Subclasses can do their own processing here. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The mouse move event │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - clear: function() { │ │ │ │ │ - this.json = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Tile.UTFGrid" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/UTFGrid.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ - * @requires OpenLayers/Tile/UTFGrid.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.UTFGrid │ │ │ │ │ - * This Layer reads from UTFGrid tiled data sources. Since UTFGrids are │ │ │ │ │ - * essentially JSON-based ASCII art with attached attributes, they are not │ │ │ │ │ - * visibly rendered. In order to use them in the map, you must add a │ │ │ │ │ - * <OpenLayers.Control.UTFGrid> control as well. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * │ │ │ │ │ - * (start code) │ │ │ │ │ - * var world_utfgrid = new OpenLayers.Layer.UTFGrid({ │ │ │ │ │ - * url: "/tiles/world_utfgrid/${z}/${x}/${y}.json", │ │ │ │ │ - * utfgridResolution: 4, │ │ │ │ │ - * displayInLayerSwitcher: false │ │ │ │ │ - * ); │ │ │ │ │ - * map.addLayer(world_utfgrid); │ │ │ │ │ - * │ │ │ │ │ - * var control = new OpenLayers.Control.UTFGrid({ │ │ │ │ │ - * layers: [world_utfgrid], │ │ │ │ │ - * handlerMode: 'move', │ │ │ │ │ - * callback: function(dataLookup) { │ │ │ │ │ - * // do something with returned data │ │ │ │ │ - * } │ │ │ │ │ - * }) │ │ │ │ │ - * (end code) │ │ │ │ │ - * │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.XYZ> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.UTFGrid = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ + move: function(evt) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * Default is false, as UTFGrids are designed to be a transparent overlay layer. │ │ │ │ │ + * Method: up │ │ │ │ │ + * This method is called during the handling of the mouse up event. │ │ │ │ │ + * Subclasses can do their own processing here. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The mouse up event │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: false, │ │ │ │ │ + up: function(evt) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: projection │ │ │ │ │ - * {<OpenLayers.Projection>} │ │ │ │ │ - * Source projection for the UTFGrids. Default is "EPSG:900913". │ │ │ │ │ + * Method: out │ │ │ │ │ + * This method is called during the handling of the mouse out event. │ │ │ │ │ + * Subclasses can do their own processing here. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The mouse out event │ │ │ │ │ */ │ │ │ │ │ - projection: new OpenLayers.Projection("EPSG:900913"), │ │ │ │ │ + out: function(evt) {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: useJSONP │ │ │ │ │ - * {Boolean} │ │ │ │ │ - * Should we use a JSONP script approach instead of a standard AJAX call? │ │ │ │ │ - * │ │ │ │ │ - * Set to true for using utfgrids from another server. │ │ │ │ │ - * Avoids same-domain policy restrictions. │ │ │ │ │ - * Note that this only works if the server accepts │ │ │ │ │ - * the callback GET parameter and dynamically │ │ │ │ │ - * wraps the returned json in a function call. │ │ │ │ │ - * │ │ │ │ │ - * Default is false │ │ │ │ │ + * The methods below are part of the magic of event handling. Because │ │ │ │ │ + * they are named like browser events, they are registered as listeners │ │ │ │ │ + * for the events they represent. │ │ │ │ │ */ │ │ │ │ │ - useJSONP: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String} │ │ │ │ │ - * URL tempate for UTFGrid tiles. Include x, y, and z parameters. │ │ │ │ │ - * E.g. "/tiles/${z}/${x}/${y}.json" │ │ │ │ │ + * Method: mousedown │ │ │ │ │ + * Handle mousedown events │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + return this.dragstart(evt); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: utfgridResolution │ │ │ │ │ - * {Number} │ │ │ │ │ - * Ratio of the pixel width to the width of a UTFGrid data point. If an │ │ │ │ │ - * entry in the grid represents a 4x4 block of pixels, the │ │ │ │ │ - * utfgridResolution would be 4. Default is 2 (specified in │ │ │ │ │ - * <OpenLayers.Tile.UTFGrid>). │ │ │ │ │ + * Method: touchstart │ │ │ │ │ + * Handle touchstart events │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + return this.dragstart(evt); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: tileClass │ │ │ │ │ - * {<OpenLayers.Tile>} The tile class to use for this layer. │ │ │ │ │ - * Defaults is <OpenLayers.Tile.UTFGrid>. │ │ │ │ │ + * Method: mousemove │ │ │ │ │ + * Handle mousemove events │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - tileClass: OpenLayers.Tile.UTFGrid, │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + return this.dragmove(evt); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.UTFGrid │ │ │ │ │ - * Create a new UTFGrid layer. │ │ │ │ │ + * Method: touchmove │ │ │ │ │ + * Handle touchmove events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * config - {Object} Configuration properties for the layer. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ - * Required configuration properties: │ │ │ │ │ - * url - {String} The url template for UTFGrid tiles. See the <url> property. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply( │ │ │ │ │ - this, [options.name, options.url, {}, options] │ │ │ │ │ - ); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - utfgridResolution: this.utfgridResolution │ │ │ │ │ - }, this.tileOptions); │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + return this.dragmove(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createBackBuffer │ │ │ │ │ - * The UTFGrid cannot create a back buffer, so this method is overriden. │ │ │ │ │ + * Method: removeTimeout │ │ │ │ │ + * Private. Called by mousemove() to remove the drag timeout. │ │ │ │ │ */ │ │ │ │ │ - createBackBuffer: function() {}, │ │ │ │ │ + removeTimeout: function() { │ │ │ │ │ + this.timeoutId = null; │ │ │ │ │ + // if timeout expires while we're still dragging (mouseup │ │ │ │ │ + // hasn't occurred) then call mousemove to move to the │ │ │ │ │ + // correct position │ │ │ │ │ + if (this.dragging) { │ │ │ │ │ + this.mousemove(this.lastMoveEvt); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ + * Method: mouseup │ │ │ │ │ + * Handle mouseup events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {Object} Only used by a subclass of this layer. │ │ │ │ │ - * │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.UTFGrid>} An exact clone of this OpenLayers.Layer.UTFGrid │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.UTFGrid(this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + return this.dragend(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: getFeatureInfo │ │ │ │ │ - * Get details about a feature associated with a map location. The object │ │ │ │ │ - * returned will have id and data properties. If the given location │ │ │ │ │ - * doesn't correspond to a feature, null will be returned. │ │ │ │ │ + * Method: touchend │ │ │ │ │ + * Handle touchend events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * location - {<OpenLayers.LonLat>} map location │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} Object representing the feature id and UTFGrid data │ │ │ │ │ - * corresponding to the given map location. Returns null if the given │ │ │ │ │ - * location doesn't hit a feature. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - getFeatureInfo: function(location) { │ │ │ │ │ - var info = null; │ │ │ │ │ - var tileInfo = this.getTileData(location); │ │ │ │ │ - if (tileInfo && tileInfo.tile) { │ │ │ │ │ - info = tileInfo.tile.getFeatureInfo(tileInfo.i, tileInfo.j); │ │ │ │ │ - } │ │ │ │ │ - return info; │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + // override evt.xy with last position since touchend does not have │ │ │ │ │ + // any touch position │ │ │ │ │ + evt.xy = this.last; │ │ │ │ │ + return this.dragend(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getFeatureId │ │ │ │ │ - * Get the identifier for the feature associated with a map location. │ │ │ │ │ + * Method: mouseout │ │ │ │ │ + * Handle mouseout events │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * location - {<OpenLayers.LonLat>} map location │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} The feature identifier corresponding to the given map location. │ │ │ │ │ - * Returns null if the location doesn't hit a feature. │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - getFeatureId: function(location) { │ │ │ │ │ - var id = null; │ │ │ │ │ - var info = this.getTileData(location); │ │ │ │ │ - if (info.tile) { │ │ │ │ │ - id = info.tile.getFeatureId(info.i, info.j); │ │ │ │ │ + mouseout: function(evt) { │ │ │ │ │ + if (this.started && OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ + if (this.documentDrag === true) { │ │ │ │ │ + this.addDocumentEvents(); │ │ │ │ │ + } else { │ │ │ │ │ + var dragged = (this.start != this.last); │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + OpenLayers.Element.removeClass( │ │ │ │ │ + this.map.viewPortDiv, "olDragDown" │ │ │ │ │ + ); │ │ │ │ │ + this.out(evt); │ │ │ │ │ + this.callback("out", []); │ │ │ │ │ + if (dragged) { │ │ │ │ │ + this.callback("done", [evt.xy]); │ │ │ │ │ + } │ │ │ │ │ + if (document.onselectstart) { │ │ │ │ │ + document.onselectstart = this.oldOnselectstart; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return id; │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.UTFGrid" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/KaMapCache.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ - * @requires OpenLayers/Layer/KaMap.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.KaMapCache │ │ │ │ │ - * │ │ │ │ │ - * This class is designed to talk directly to a web-accessible ka-Map │ │ │ │ │ - * cache generated by the precache2.php script. │ │ │ │ │ - * │ │ │ │ │ - * To create a a new KaMapCache layer, you must indicate also the "i" parameter │ │ │ │ │ - * (that will be used to calculate the file extension), and another special │ │ │ │ │ - * parameter, object names "metaTileSize", with "h" (height) and "w" (width) │ │ │ │ │ - * properties. │ │ │ │ │ - * │ │ │ │ │ - * // Create a new kaMapCache layer. │ │ │ │ │ - * var kamap_base = new OpenLayers.Layer.KaMapCache( │ │ │ │ │ - * "Satellite", │ │ │ │ │ - * "http://www.example.org/web/acessible/cache", │ │ │ │ │ - * {g: "satellite", map: "world", i: 'png24', metaTileSize: {w: 5, h: 5} } │ │ │ │ │ - * ); │ │ │ │ │ - * │ │ │ │ │ - * // Create an kaMapCache overlay layer (using "isBaseLayer: false"). │ │ │ │ │ - * // Forces the output to be a "gif", using the "i" parameter. │ │ │ │ │ - * var kamap_overlay = new OpenLayers.Layer.KaMapCache( │ │ │ │ │ - * "Streets", │ │ │ │ │ - * "http://www.example.org/web/acessible/cache", │ │ │ │ │ - * {g: "streets", map: "world", i: "gif", metaTileSize: {w: 5, h: 5} }, │ │ │ │ │ - * {isBaseLayer: false} │ │ │ │ │ - * ); │ │ │ │ │ - * │ │ │ │ │ - * The cache URLs must look like: │ │ │ │ │ - * var/cache/World/50000/Group_Name/def/t-440320/l20480 │ │ │ │ │ - * │ │ │ │ │ - * This means that the cache generated via tile.php will *not* work with │ │ │ │ │ - * this class, and should instead use the KaMap layer. │ │ │ │ │ - * │ │ │ │ │ - * More information is available in Ticket #1518. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.KaMap> │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.KaMapCache = OpenLayers.Class(OpenLayers.Layer.KaMap, { │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Constant: IMAGE_EXTENSIONS │ │ │ │ │ - * {Object} Simple hash map to convert format to extension. │ │ │ │ │ + * Method: click │ │ │ │ │ + * The drag handler captures the click event. If something else registers │ │ │ │ │ + * for clicks on the same element, its listener will not be called │ │ │ │ │ + * after a drag. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ - IMAGE_EXTENSIONS: { │ │ │ │ │ - 'jpeg': 'jpg', │ │ │ │ │ - 'gif': 'gif', │ │ │ │ │ - 'png': 'png', │ │ │ │ │ - 'png8': 'png', │ │ │ │ │ - 'png24': 'png', │ │ │ │ │ - 'dithered': 'png' │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + // let the click event propagate only if the mouse moved │ │ │ │ │ + return (this.start == this.last); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: DEFAULT_FORMAT │ │ │ │ │ - * {Object} Simple hash map to convert format to extension. │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activate the handler. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was successfully activated. │ │ │ │ │ */ │ │ │ │ │ - DEFAULT_FORMAT: 'jpeg', │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + activated = true; │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.KaMapCache │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivate the handler. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * url - {String} │ │ │ │ │ - * params - {Object} Parameters to be sent to the HTTP server in the │ │ │ │ │ - * query string for the tile. The format can be set via the 'i' │ │ │ │ │ - * parameter (defaults to jpg) , and the map should be set via │ │ │ │ │ - * the 'map' parameter. It has been reported that ka-Map may behave │ │ │ │ │ - * inconsistently if your format parameter does not match the format │ │ │ │ │ - * parameter configured in your config.php. (See ticket #327 for more │ │ │ │ │ - * information.) │ │ │ │ │ - * options - {Object} Additional options for the layer. Any of the │ │ │ │ │ - * APIProperties listed on this layer, and any layer types it │ │ │ │ │ - * extends, can be overridden through the options parameter. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.KaMap.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.extension = this.IMAGE_EXTENSIONS[this.params.i.toLowerCase() || this.DEFAULT_FORMAT]; │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + deactivated = true; │ │ │ │ │ + OpenLayers.Element.removeClass( │ │ │ │ │ + this.map.viewPortDiv, "olDragDown" │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ + * Method: adjustXY │ │ │ │ │ + * Converts event coordinates that are relative to the document body to │ │ │ │ │ + * ones that are relative to the map viewport. The latter is the default in │ │ │ │ │ + * OpenLayers. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as │ │ │ │ │ - * parameters │ │ │ │ │ + * evt - {Object} │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var mapRes = this.map.getResolution(); │ │ │ │ │ - var scale = Math.round((this.map.getScale() * 10000)) / 10000; │ │ │ │ │ - var pX = Math.round(bounds.left / mapRes); │ │ │ │ │ - var pY = -Math.round(bounds.top / mapRes); │ │ │ │ │ - │ │ │ │ │ - var metaX = Math.floor(pX / this.tileSize.w / this.params.metaTileSize.w) * this.tileSize.w * this.params.metaTileSize.w; │ │ │ │ │ - var metaY = Math.floor(pY / this.tileSize.h / this.params.metaTileSize.h) * this.tileSize.h * this.params.metaTileSize.h; │ │ │ │ │ - │ │ │ │ │ - var components = [ │ │ │ │ │ - "/", │ │ │ │ │ - this.params.map, │ │ │ │ │ - "/", │ │ │ │ │ - scale, │ │ │ │ │ - "/", │ │ │ │ │ - this.params.g.replace(/\s/g, '_'), │ │ │ │ │ - "/def/t", │ │ │ │ │ - metaY, │ │ │ │ │ - "/l", │ │ │ │ │ - metaX, │ │ │ │ │ - "/t", │ │ │ │ │ - pY, │ │ │ │ │ - "l", │ │ │ │ │ - pX, │ │ │ │ │ - ".", │ │ │ │ │ - this.extension │ │ │ │ │ - ]; │ │ │ │ │ + adjustXY: function(evt) { │ │ │ │ │ + var pos = OpenLayers.Util.pagePosition(this.map.viewPortDiv); │ │ │ │ │ + evt.xy.x -= pos[0]; │ │ │ │ │ + evt.xy.y -= pos[1]; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var url = this.url; │ │ │ │ │ + /** │ │ │ │ │ + * Method: addDocumentEvents │ │ │ │ │ + * Start observing document events when documentDrag is true and the mouse │ │ │ │ │ + * cursor leaves the map viewport while dragging. │ │ │ │ │ + */ │ │ │ │ │ + addDocumentEvents: function() { │ │ │ │ │ + OpenLayers.Element.addClass(document.body, "olDragDown"); │ │ │ │ │ + this.documentEvents = true; │ │ │ │ │ + OpenLayers.Event.observe(document, "mousemove", this._docMove); │ │ │ │ │ + OpenLayers.Event.observe(document, "mouseup", this._docUp); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(components.join(''), url); │ │ │ │ │ - } │ │ │ │ │ - return url + components.join(""); │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeDocumentEvents │ │ │ │ │ + * Stops observing document events when documentDrag is true and the mouse │ │ │ │ │ + * cursor re-enters the map viewport while dragging. │ │ │ │ │ + */ │ │ │ │ │ + removeDocumentEvents: function() { │ │ │ │ │ + OpenLayers.Element.removeClass(document.body, "olDragDown"); │ │ │ │ │ + this.documentEvents = false; │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mousemove", this._docMove); │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mouseup", this._docUp); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.KaMapCache" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Drag" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/WMTS.js │ │ │ │ │ + OpenLayers/Handler/RegularPolygon.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/Handler/Drag.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.WMTS │ │ │ │ │ - * Instances of the WMTS class allow viewing of tiles from a service that │ │ │ │ │ - * implements the OGC WMTS specification version 1.0.0. │ │ │ │ │ + * Class: OpenLayers.Handler.RegularPolygon │ │ │ │ │ + * Handler to draw a regular polygon on the map. Polygon is displayed on mouse │ │ │ │ │ + * down, moves or is modified on mouse move, and is finished on mouse up. │ │ │ │ │ + * The handler triggers callbacks for 'done' and 'cancel'. Create a new │ │ │ │ │ + * instance with the <OpenLayers.Handler.RegularPolygon> constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ + * - <OpenLayers.Handler.Drag> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.WMTS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ +OpenLayers.Handler.RegularPolygon = OpenLayers.Class(OpenLayers.Handler.Drag, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} The layer will be considered a base layer. Default is true. │ │ │ │ │ + * APIProperty: sides │ │ │ │ │ + * {Integer} Number of sides for the regular polygon. Needs to be greater │ │ │ │ │ + * than 2. Defaults to 4. │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + sides: 4, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} WMTS version. Default is "1.0.0". │ │ │ │ │ + * APIProperty: radius │ │ │ │ │ + * {Float} Optional radius in map units of the regular polygon. If this is │ │ │ │ │ + * set to some non-zero value, a polygon with a fixed radius will be │ │ │ │ │ + * drawn and dragged with mose movements. If this property is not │ │ │ │ │ + * set, dragging changes the radius of the polygon. Set to null by │ │ │ │ │ + * default. │ │ │ │ │ */ │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ + radius: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: requestEncoding │ │ │ │ │ - * {String} Request encoding. Can be "REST" or "KVP". Default is "KVP". │ │ │ │ │ + * APIProperty: snapAngle │ │ │ │ │ + * {Float} If set to a non-zero value, the handler will snap the polygon │ │ │ │ │ + * rotation to multiples of the snapAngle. Value is an angle measured │ │ │ │ │ + * in degrees counterclockwise from the positive x-axis. │ │ │ │ │ */ │ │ │ │ │ - requestEncoding: "KVP", │ │ │ │ │ + snapAngle: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String|Array(String)} The base URL or request URL template for the WMTS │ │ │ │ │ - * service. Must be provided. Array is only supported for base URLs, not │ │ │ │ │ - * for request URL templates. URL templates are only supported for │ │ │ │ │ - * REST <requestEncoding>. │ │ │ │ │ + * APIProperty: snapToggle │ │ │ │ │ + * {String} If set, snapToggle is checked on mouse events and will set │ │ │ │ │ + * the snap mode to the opposite of what it currently is. To disallow │ │ │ │ │ + * toggling between snap and non-snap mode, set freehandToggle to │ │ │ │ │ + * null. Acceptable toggle values are 'shiftKey', 'ctrlKey', and │ │ │ │ │ + * 'altKey'. Snap mode is only possible if this.snapAngle is set to a │ │ │ │ │ + * non-zero value. │ │ │ │ │ */ │ │ │ │ │ - url: null, │ │ │ │ │ + snapToggle: 'shiftKey', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: layer │ │ │ │ │ - * {String} The layer identifier advertised by the WMTS service. Must be │ │ │ │ │ - * provided. │ │ │ │ │ - */ │ │ │ │ │ - layer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: matrixSet │ │ │ │ │ - * {String} One of the advertised matrix set identifiers. Must be provided. │ │ │ │ │ - */ │ │ │ │ │ - matrixSet: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: style │ │ │ │ │ - * {String} One of the advertised layer styles. Must be provided. │ │ │ │ │ + * Property: layerOptions │ │ │ │ │ + * {Object} Any optional properties to be set on the sketch layer. │ │ │ │ │ */ │ │ │ │ │ - style: null, │ │ │ │ │ + layerOptions: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: format │ │ │ │ │ - * {String} The image MIME type. Default is "image/jpeg". │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: persist │ │ │ │ │ + * {Boolean} Leave the feature rendered until clear is called. Default │ │ │ │ │ + * is false. If set to true, the feature remains rendered until │ │ │ │ │ + * clear is called, typically by deactivating the handler or starting │ │ │ │ │ + * another drawing. │ │ │ │ │ */ │ │ │ │ │ - format: "image/jpeg", │ │ │ │ │ + persist: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: tileOrigin │ │ │ │ │ - * {<OpenLayers.LonLat>} The top-left corner of the tile matrix in map │ │ │ │ │ - * units. If the tile origin for each matrix in a set is different, │ │ │ │ │ - * the <matrixIds> should include a topLeftCorner property. If │ │ │ │ │ - * not provided, the tile origin will default to the top left corner │ │ │ │ │ - * of the layer <maxExtent>. │ │ │ │ │ + * APIProperty: irregular │ │ │ │ │ + * {Boolean} Draw an irregular polygon instead of a regular polygon. │ │ │ │ │ + * Default is false. If true, the initial mouse down will represent │ │ │ │ │ + * one corner of the polygon bounds and with each mouse movement, the │ │ │ │ │ + * polygon will be stretched so the opposite corner of its bounds │ │ │ │ │ + * follows the mouse position. This property takes precedence over │ │ │ │ │ + * the radius property. If set to true, the radius property will │ │ │ │ │ + * be ignored. │ │ │ │ │ */ │ │ │ │ │ - tileOrigin: null, │ │ │ │ │ + irregular: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: tileFullExtent │ │ │ │ │ - * {<OpenLayers.Bounds>} The full extent of the tile set. If not supplied, │ │ │ │ │ - * the layer's <maxExtent> property will be used. │ │ │ │ │ + * APIProperty: citeCompliant │ │ │ │ │ + * {Boolean} If set to true, coordinates of features drawn in a map extent │ │ │ │ │ + * crossing the date line won't exceed the world bounds. Default is false. │ │ │ │ │ */ │ │ │ │ │ - tileFullExtent: null, │ │ │ │ │ + citeCompliant: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: formatSuffix │ │ │ │ │ - * {String} For REST request encoding, an image format suffix must be │ │ │ │ │ - * included in the request. If not provided, the suffix will be derived │ │ │ │ │ - * from the <format> property. │ │ │ │ │ + * Property: angle │ │ │ │ │ + * {Float} The angle from the origin (mouse down) to the current mouse │ │ │ │ │ + * position, in radians. This is measured counterclockwise from the │ │ │ │ │ + * positive x-axis. │ │ │ │ │ */ │ │ │ │ │ - formatSuffix: null, │ │ │ │ │ + angle: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: matrixIds │ │ │ │ │ - * {Array} A list of tile matrix identifiers. If not provided, the matrix │ │ │ │ │ - * identifiers will be assumed to be integers corresponding to the │ │ │ │ │ - * map zoom level. If a list of strings is provided, each item should │ │ │ │ │ - * be the matrix identifier that corresponds to the map zoom level. │ │ │ │ │ - * Additionally, a list of objects can be provided. Each object should │ │ │ │ │ - * describe the matrix as presented in the WMTS capabilities. These │ │ │ │ │ - * objects should have the propertes shown below. │ │ │ │ │ - * │ │ │ │ │ - * Matrix properties: │ │ │ │ │ - * identifier - {String} The matrix identifier (required). │ │ │ │ │ - * scaleDenominator - {Number} The matrix scale denominator. │ │ │ │ │ - * topLeftCorner - {<OpenLayers.LonLat>} The top left corner of the │ │ │ │ │ - * matrix. Must be provided if different than the layer <tileOrigin>. │ │ │ │ │ - * tileWidth - {Number} The tile width for the matrix. Must be provided │ │ │ │ │ - * if different than the width given in the layer <tileSize>. │ │ │ │ │ - * tileHeight - {Number} The tile height for the matrix. Must be provided │ │ │ │ │ - * if different than the height given in the layer <tileSize>. │ │ │ │ │ + * Property: fixedRadius │ │ │ │ │ + * {Boolean} The polygon has a fixed radius. True if a radius is set before │ │ │ │ │ + * drawing begins. False otherwise. │ │ │ │ │ */ │ │ │ │ │ - matrixIds: null, │ │ │ │ │ + fixedRadius: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: dimensions │ │ │ │ │ - * {Array} For RESTful request encoding, extra dimensions may be specified. │ │ │ │ │ - * Items in this list should be property names in the <params> object. │ │ │ │ │ - * Values of extra dimensions will be determined from the corresponding │ │ │ │ │ - * values in the <params> object. │ │ │ │ │ + * Property: feature │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} The currently drawn polygon feature │ │ │ │ │ */ │ │ │ │ │ - dimensions: null, │ │ │ │ │ + feature: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: params │ │ │ │ │ - * {Object} Extra parameters to include in tile requests. For KVP │ │ │ │ │ - * <requestEncoding>, these properties will be encoded in the request │ │ │ │ │ - * query string. For REST <requestEncoding>, these properties will │ │ │ │ │ - * become part of the request path, with order determined by the │ │ │ │ │ - * <dimensions> list. │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} The temporary drawing layer │ │ │ │ │ */ │ │ │ │ │ - params: null, │ │ │ │ │ + layer: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomOffset │ │ │ │ │ - * {Number} If your cache has more levels than you want to provide │ │ │ │ │ - * access to with this layer, supply a zoomOffset. This zoom offset │ │ │ │ │ - * is added to the current map zoom level to determine the level │ │ │ │ │ - * for a requested tile. For example, if you supply a zoomOffset │ │ │ │ │ - * of 3, when the map is at the zoom 0, tiles will be requested from │ │ │ │ │ - * level 3 of your cache. Default is 0 (assumes cache level and map │ │ │ │ │ - * zoom are equivalent). Additionally, if this layer is to be used │ │ │ │ │ - * as an overlay and the cache has fewer zoom levels than the base │ │ │ │ │ - * layer, you can supply a negative zoomOffset. For example, if a │ │ │ │ │ - * map zoom level of 1 corresponds to your cache level zero, you would │ │ │ │ │ - * supply a -1 zoomOffset (and set the maxResolution of the layer │ │ │ │ │ - * appropriately). The zoomOffset value has no effect if complete │ │ │ │ │ - * matrix definitions (including scaleDenominator) are supplied in │ │ │ │ │ - * the <matrixIds> property. Defaults to 0 (no zoom offset). │ │ │ │ │ + * Property: origin │ │ │ │ │ + * {<OpenLayers.Geometry.Point>} Location of the first mouse down │ │ │ │ │ */ │ │ │ │ │ - zoomOffset: 0, │ │ │ │ │ + origin: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: serverResolutions │ │ │ │ │ - * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ - * property if the map resolutions differ from the server. This │ │ │ │ │ - * property serves two purposes. (a) <serverResolutions> can include │ │ │ │ │ - * resolutions that the server supports and that you don't want to │ │ │ │ │ - * provide with this layer; you can also look at <zoomOffset>, which is │ │ │ │ │ - * an alternative to <serverResolutions> for that specific purpose. │ │ │ │ │ - * (b) The map can work with resolutions that aren't supported by │ │ │ │ │ - * the server, i.e. that aren't in <serverResolutions>. When the │ │ │ │ │ - * map is displayed in such a resolution data for the closest │ │ │ │ │ - * server-supported resolution is loaded and the layer div is │ │ │ │ │ - * stretched as necessary. │ │ │ │ │ + * Constructor: OpenLayers.Handler.RegularPolygon │ │ │ │ │ + * Create a new regular polygon handler. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ + * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ + * functions. Various callbacks described below. │ │ │ │ │ + * options - {Object} An object with properties to be set on the handler. │ │ │ │ │ + * If the options.sides property is not specified, the number of sides │ │ │ │ │ + * will default to 4. │ │ │ │ │ + * │ │ │ │ │ + * Named callbacks: │ │ │ │ │ + * create - Called when a sketch is first created. Callback called with │ │ │ │ │ + * the creation point geometry and sketch feature. │ │ │ │ │ + * done - Called when the sketch drawing is finished. The callback will │ │ │ │ │ + * recieve a single argument, the sketch geometry. │ │ │ │ │ + * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ + * cancel callback will receive a geometry. │ │ │ │ │ */ │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ + this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style['default'], {}); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Handler.Drag.prototype.initialize.apply(this, │ │ │ │ │ + [control, callbacks, options]); │ │ │ │ │ + this.options = (options) ? options : {}; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: formatSuffixMap │ │ │ │ │ - * {Object} a map between WMTS 'format' request parameter and tile image file suffix │ │ │ │ │ + * APIMethod: setOptions │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newOptions - {Object} │ │ │ │ │ */ │ │ │ │ │ - formatSuffixMap: { │ │ │ │ │ - "image/png": "png", │ │ │ │ │ - "image/png8": "png", │ │ │ │ │ - "image/png24": "png", │ │ │ │ │ - "image/png32": "png", │ │ │ │ │ - "png": "png", │ │ │ │ │ - "image/jpeg": "jpg", │ │ │ │ │ - "image/jpg": "jpg", │ │ │ │ │ - "jpeg": "jpg", │ │ │ │ │ - "jpg": "jpg" │ │ │ │ │ + setOptions: function(newOptions) { │ │ │ │ │ + OpenLayers.Util.extend(this.options, newOptions); │ │ │ │ │ + OpenLayers.Util.extend(this, newOptions); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: matrix │ │ │ │ │ - * {Object} Matrix definition for the current map resolution. Updated by │ │ │ │ │ - * the <updateMatrixProperties> method. │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Turn on the handler. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was successfully activated │ │ │ │ │ */ │ │ │ │ │ - matrix: null, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.Drag.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + // create temporary vector layer for rendering geometry sketch │ │ │ │ │ + var options = OpenLayers.Util.extend({ │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + // indicate that the temp vector layer will never be out of range │ │ │ │ │ + // without this, resolution properties must be specified at the │ │ │ │ │ + // map-level for this temporary layer to init its resolutions │ │ │ │ │ + // correctly │ │ │ │ │ + calculateInRange: OpenLayers.Function.True, │ │ │ │ │ + wrapDateLine: this.citeCompliant │ │ │ │ │ + }, this.layerOptions); │ │ │ │ │ + this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ + this.map.addLayer(this.layer); │ │ │ │ │ + activated = true; │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.WMTS │ │ │ │ │ - * Create a new WMTS layer. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var wmts = new OpenLayers.Layer.WMTS({ │ │ │ │ │ - * name: "My WMTS Layer", │ │ │ │ │ - * url: "http://example.com/wmts", │ │ │ │ │ - * layer: "layer_id", │ │ │ │ │ - * style: "default", │ │ │ │ │ - * matrixSet: "matrix_id" │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * config - {Object} Configuration properties for the layer. │ │ │ │ │ - * │ │ │ │ │ - * Required configuration properties: │ │ │ │ │ - * url - {String} The base url for the service. See the <url> property. │ │ │ │ │ - * layer - {String} The layer identifier. See the <layer> property. │ │ │ │ │ - * style - {String} The layer style identifier. See the <style> property. │ │ │ │ │ - * matrixSet - {String} The tile matrix set identifier. See the <matrixSet> │ │ │ │ │ - * property. │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Turn off the handler. │ │ │ │ │ * │ │ │ │ │ - * Any other documented layer properties can be provided in the config object. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was successfully deactivated │ │ │ │ │ */ │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ - │ │ │ │ │ - // confirm required properties are supplied │ │ │ │ │ - var required = { │ │ │ │ │ - url: true, │ │ │ │ │ - layer: true, │ │ │ │ │ - style: true, │ │ │ │ │ - matrixSet: true │ │ │ │ │ - }; │ │ │ │ │ - for (var prop in required) { │ │ │ │ │ - if (!(prop in config)) { │ │ │ │ │ - throw new Error("Missing property '" + prop + "' in layer configuration."); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.Drag.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + // call the cancel callback if mid-drawing │ │ │ │ │ + if (this.dragging) { │ │ │ │ │ + this.cancel(); │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - config.params = OpenLayers.Util.upperCaseObject(config.params); │ │ │ │ │ - var args = [config.name, config.url, config.params, config]; │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, args); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - // determine format suffix (for REST) │ │ │ │ │ - if (!this.formatSuffix) { │ │ │ │ │ - this.formatSuffix = this.formatSuffixMap[this.format] || this.format.split("/").pop(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // expand matrixIds (may be array of string or array of object) │ │ │ │ │ - if (this.matrixIds) { │ │ │ │ │ - var len = this.matrixIds.length; │ │ │ │ │ - if (len && typeof this.matrixIds[0] === "string") { │ │ │ │ │ - var ids = this.matrixIds; │ │ │ │ │ - this.matrixIds = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - this.matrixIds[i] = { │ │ │ │ │ - identifier: ids[i] │ │ │ │ │ - }; │ │ │ │ │ + // If a layer's map property is set to null, it means that that │ │ │ │ │ + // layer isn't added to the map. Since we ourself added the layer │ │ │ │ │ + // to the map in activate(), we can assume that if this.layer.map │ │ │ │ │ + // is null it means that the layer has been destroyed (as a result │ │ │ │ │ + // of map.destroy() for example. │ │ │ │ │ + if (this.layer.map != null) { │ │ │ │ │ + this.layer.destroy(false); │ │ │ │ │ + if (this.feature) { │ │ │ │ │ + this.feature.destroy(); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + this.layer = null; │ │ │ │ │ + this.feature = null; │ │ │ │ │ + deactivated = true; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ + * Method: down │ │ │ │ │ + * Start drawing a new feature │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The drag start event │ │ │ │ │ */ │ │ │ │ │ - setMap: function() { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + down: function(evt) { │ │ │ │ │ + this.fixedRadius = !!(this.radius); │ │ │ │ │ + var maploc = this.layer.getLonLatFromViewPortPx(evt.xy); │ │ │ │ │ + this.origin = new OpenLayers.Geometry.Point(maploc.lon, maploc.lat); │ │ │ │ │ + // create the new polygon │ │ │ │ │ + if (!this.fixedRadius || this.irregular) { │ │ │ │ │ + // smallest radius should not be less one pixel in map units │ │ │ │ │ + // VML doesn't behave well with smaller │ │ │ │ │ + this.radius = this.map.getResolution(); │ │ │ │ │ + } │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.clear(); │ │ │ │ │ + } │ │ │ │ │ + this.feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ + this.createGeometry(); │ │ │ │ │ + this.callback("create", [this.origin, this.feature]); │ │ │ │ │ + this.layer.addFeatures([this.feature], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.layer.drawFeature(this.feature, this.style); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: updateMatrixProperties │ │ │ │ │ - * Called when map resolution changes to update matrix related properties. │ │ │ │ │ + * Method: move │ │ │ │ │ + * Respond to drag move events │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Evt} The move event │ │ │ │ │ */ │ │ │ │ │ - updateMatrixProperties: function() { │ │ │ │ │ - this.matrix = this.getMatrix(); │ │ │ │ │ - if (this.matrix) { │ │ │ │ │ - if (this.matrix.topLeftCorner) { │ │ │ │ │ - this.tileOrigin = this.matrix.topLeftCorner; │ │ │ │ │ - } │ │ │ │ │ - if (this.matrix.tileWidth && this.matrix.tileHeight) { │ │ │ │ │ - this.tileSize = new OpenLayers.Size( │ │ │ │ │ - this.matrix.tileWidth, this.matrix.tileHeight │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (!this.tileOrigin) { │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat( │ │ │ │ │ - this.maxExtent.left, this.maxExtent.top │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (!this.tileFullExtent) { │ │ │ │ │ - this.tileFullExtent = this.maxExtent; │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + var maploc = this.layer.getLonLatFromViewPortPx(evt.xy); │ │ │ │ │ + var point = new OpenLayers.Geometry.Point(maploc.lon, maploc.lat); │ │ │ │ │ + if (this.irregular) { │ │ │ │ │ + var ry = Math.sqrt(2) * Math.abs(point.y - this.origin.y) / 2; │ │ │ │ │ + this.radius = Math.max(this.map.getResolution() / 2, ry); │ │ │ │ │ + } else if (this.fixedRadius) { │ │ │ │ │ + this.origin = point; │ │ │ │ │ + } else { │ │ │ │ │ + this.calculateAngle(point, evt); │ │ │ │ │ + this.radius = Math.max(this.map.getResolution() / 2, │ │ │ │ │ + point.distanceTo(this.origin)); │ │ │ │ │ + } │ │ │ │ │ + this.modifyGeometry(); │ │ │ │ │ + if (this.irregular) { │ │ │ │ │ + var dx = point.x - this.origin.x; │ │ │ │ │ + var dy = point.y - this.origin.y; │ │ │ │ │ + var ratio; │ │ │ │ │ + if (dy == 0) { │ │ │ │ │ + ratio = dx / (this.radius * Math.sqrt(2)); │ │ │ │ │ + } else { │ │ │ │ │ + ratio = dx / dy; │ │ │ │ │ } │ │ │ │ │ + this.feature.geometry.resize(1, this.origin, ratio); │ │ │ │ │ + this.feature.geometry.move(dx / 2, dy / 2); │ │ │ │ │ } │ │ │ │ │ + this.layer.drawFeature(this.feature, this.style); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveTo │ │ │ │ │ - * │ │ │ │ │ + * Method: up │ │ │ │ │ + * Finish drawing the feature │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * zoomChanged - {Boolean} Tells when zoom has changed, as layers have to │ │ │ │ │ - * do some init work in that case. │ │ │ │ │ - * dragging - {Boolean} │ │ │ │ │ + * evt - {Event} The mouse up event │ │ │ │ │ */ │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - if (zoomChanged || !this.matrix) { │ │ │ │ │ - this.updateMatrixProperties(); │ │ │ │ │ + up: function(evt) { │ │ │ │ │ + this.finalize(); │ │ │ │ │ + // the mouseup method of superclass doesn't call the │ │ │ │ │ + // "done" callback if there's been no move between │ │ │ │ │ + // down and up │ │ │ │ │ + if (this.start == this.last) { │ │ │ │ │ + this.callback("done", [evt.xy]); │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.moveTo.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ - * │ │ │ │ │ + * Method: out │ │ │ │ │ + * Finish drawing the feature. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.WMTS>} An exact clone of this <OpenLayers.Layer.WMTS> │ │ │ │ │ + * evt - {Event} The mouse out event │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.WMTS(this.options); │ │ │ │ │ - } │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - return obj; │ │ │ │ │ + out: function(evt) { │ │ │ │ │ + this.finalize(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getIdentifier │ │ │ │ │ - * Get the current index in the matrixIds array. │ │ │ │ │ + * Method: createGeometry │ │ │ │ │ + * Create the new polygon geometry. This is called at the start of the │ │ │ │ │ + * drag and at any point during the drag if the number of sides │ │ │ │ │ + * changes. │ │ │ │ │ */ │ │ │ │ │ - getIdentifier: function() { │ │ │ │ │ - return this.getServerZoom(); │ │ │ │ │ + createGeometry: function() { │ │ │ │ │ + this.angle = Math.PI * ((1 / this.sides) - (1 / 2)); │ │ │ │ │ + if (this.snapAngle) { │ │ │ │ │ + this.angle += this.snapAngle * (Math.PI / 180); │ │ │ │ │ + } │ │ │ │ │ + this.feature.geometry = OpenLayers.Geometry.Polygon.createRegularPolygon( │ │ │ │ │ + this.origin, this.radius, this.sides, this.snapAngle │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getMatrix │ │ │ │ │ - * Get the appropriate matrix definition for the current map resolution. │ │ │ │ │ + * Method: modifyGeometry │ │ │ │ │ + * Modify the polygon geometry in place. │ │ │ │ │ */ │ │ │ │ │ - getMatrix: function() { │ │ │ │ │ - var matrix; │ │ │ │ │ - if (!this.matrixIds || this.matrixIds.length === 0) { │ │ │ │ │ - matrix = { │ │ │ │ │ - identifier: this.getIdentifier() │ │ │ │ │ - }; │ │ │ │ │ - } else { │ │ │ │ │ - // get appropriate matrix given the map scale if possible │ │ │ │ │ - if ("scaleDenominator" in this.matrixIds[0]) { │ │ │ │ │ - // scale denominator calculation based on WMTS spec │ │ │ │ │ - var denom = │ │ │ │ │ - OpenLayers.METERS_PER_INCH * │ │ │ │ │ - OpenLayers.INCHES_PER_UNIT[this.units] * │ │ │ │ │ - this.getServerResolution() / 0.28E-3; │ │ │ │ │ - var diff = Number.POSITIVE_INFINITY; │ │ │ │ │ - var delta; │ │ │ │ │ - for (var i = 0, ii = this.matrixIds.length; i < ii; ++i) { │ │ │ │ │ - delta = Math.abs(1 - (this.matrixIds[i].scaleDenominator / denom)); │ │ │ │ │ - if (delta < diff) { │ │ │ │ │ - diff = delta; │ │ │ │ │ - matrix = this.matrixIds[i]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - // fall back on zoom as index │ │ │ │ │ - matrix = this.matrixIds[this.getIdentifier()]; │ │ │ │ │ - } │ │ │ │ │ + modifyGeometry: function() { │ │ │ │ │ + var angle, point; │ │ │ │ │ + var ring = this.feature.geometry.components[0]; │ │ │ │ │ + // if the number of sides ever changes, create a new geometry │ │ │ │ │ + if (ring.components.length != (this.sides + 1)) { │ │ │ │ │ + this.createGeometry(); │ │ │ │ │ + ring = this.feature.geometry.components[0]; │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0; i < this.sides; ++i) { │ │ │ │ │ + point = ring.components[i]; │ │ │ │ │ + angle = this.angle + (i * 2 * Math.PI / this.sides); │ │ │ │ │ + point.x = this.origin.x + (this.radius * Math.cos(angle)); │ │ │ │ │ + point.y = this.origin.y + (this.radius * Math.sin(angle)); │ │ │ │ │ + point.clearBounds(); │ │ │ │ │ } │ │ │ │ │ - return matrix; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getTileInfo │ │ │ │ │ - * Get tile information for a given location at the current map resolution. │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateAngle │ │ │ │ │ + * Calculate the angle based on settings. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * loc - {<OpenLayers.LonLat} A location in map coordinates. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object with "col", "row", "i", and "j" properties. The col │ │ │ │ │ - * and row values are zero based tile indexes from the top left. The │ │ │ │ │ - * i and j values are the number of pixels to the left and top │ │ │ │ │ - * (respectively) of the given location within the target tile. │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - getTileInfo: function(loc) { │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - │ │ │ │ │ - var fx = (loc.lon - this.tileOrigin.lon) / (res * this.tileSize.w); │ │ │ │ │ - var fy = (this.tileOrigin.lat - loc.lat) / (res * this.tileSize.h); │ │ │ │ │ - │ │ │ │ │ - var col = Math.floor(fx); │ │ │ │ │ - var row = Math.floor(fy); │ │ │ │ │ - │ │ │ │ │ - return { │ │ │ │ │ - col: col, │ │ │ │ │ - row: row, │ │ │ │ │ - i: Math.floor((fx - col) * this.tileSize.w), │ │ │ │ │ - j: Math.floor((fy - row) * this.tileSize.h) │ │ │ │ │ - }; │ │ │ │ │ + calculateAngle: function(point, evt) { │ │ │ │ │ + var alpha = Math.atan2(point.y - this.origin.y, │ │ │ │ │ + point.x - this.origin.x); │ │ │ │ │ + if (this.snapAngle && (this.snapToggle && !evt[this.snapToggle])) { │ │ │ │ │ + var snapAngleRad = (Math.PI / 180) * this.snapAngle; │ │ │ │ │ + this.angle = Math.round(alpha / snapAngleRad) * snapAngleRad; │ │ │ │ │ + } else { │ │ │ │ │ + this.angle = alpha; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A URL for the tile corresponding to the given bounds. │ │ │ │ │ + * APIMethod: cancel │ │ │ │ │ + * Finish the geometry and call the "cancel" callback. │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var url = ""; │ │ │ │ │ - if (!this.tileFullExtent || this.tileFullExtent.intersectsBounds(bounds)) { │ │ │ │ │ - │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var info = this.getTileInfo(center); │ │ │ │ │ - var matrixId = this.matrix.identifier; │ │ │ │ │ - var dimensions = this.dimensions, │ │ │ │ │ - params; │ │ │ │ │ - │ │ │ │ │ - if (OpenLayers.Util.isArray(this.url)) { │ │ │ │ │ - url = this.selectUrl([ │ │ │ │ │ - this.version, this.style, this.matrixSet, │ │ │ │ │ - this.matrix.identifier, info.row, info.col │ │ │ │ │ - ].join(","), this.url); │ │ │ │ │ - } else { │ │ │ │ │ - url = this.url; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.requestEncoding.toUpperCase() === "REST") { │ │ │ │ │ - params = this.params; │ │ │ │ │ - if (url.indexOf("{") !== -1) { │ │ │ │ │ - var template = url.replace(/\{/g, "${"); │ │ │ │ │ - var context = { │ │ │ │ │ - // spec does not make clear if capital S or not │ │ │ │ │ - style: this.style, │ │ │ │ │ - Style: this.style, │ │ │ │ │ - TileMatrixSet: this.matrixSet, │ │ │ │ │ - TileMatrix: this.matrix.identifier, │ │ │ │ │ - TileRow: info.row, │ │ │ │ │ - TileCol: info.col │ │ │ │ │ - }; │ │ │ │ │ - if (dimensions) { │ │ │ │ │ - var dimension, i; │ │ │ │ │ - for (i = dimensions.length - 1; i >= 0; --i) { │ │ │ │ │ - dimension = dimensions[i]; │ │ │ │ │ - context[dimension] = params[dimension.toUpperCase()]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - url = OpenLayers.String.format(template, context); │ │ │ │ │ - } else { │ │ │ │ │ - // include 'version', 'layer' and 'style' in tile resource url │ │ │ │ │ - var path = this.version + "/" + this.layer + "/" + this.style + "/"; │ │ │ │ │ - │ │ │ │ │ - // append optional dimension path elements │ │ │ │ │ - if (dimensions) { │ │ │ │ │ - for (var i = 0; i < dimensions.length; i++) { │ │ │ │ │ - if (params[dimensions[i]]) { │ │ │ │ │ - path = path + params[dimensions[i]] + "/"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // append other required path elements │ │ │ │ │ - path = path + this.matrixSet + "/" + this.matrix.identifier + │ │ │ │ │ - "/" + info.row + "/" + info.col + "." + this.formatSuffix; │ │ │ │ │ - │ │ │ │ │ - if (!url.match(/\/$/)) { │ │ │ │ │ - url = url + "/"; │ │ │ │ │ - } │ │ │ │ │ - url = url + path; │ │ │ │ │ - } │ │ │ │ │ - } else if (this.requestEncoding.toUpperCase() === "KVP") { │ │ │ │ │ + cancel: function() { │ │ │ │ │ + // the polygon geometry gets cloned in the callback method │ │ │ │ │ + this.callback("cancel", null); │ │ │ │ │ + this.finalize(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // assemble all required parameters │ │ │ │ │ - params = { │ │ │ │ │ - SERVICE: "WMTS", │ │ │ │ │ - REQUEST: "GetTile", │ │ │ │ │ - VERSION: this.version, │ │ │ │ │ - LAYER: this.layer, │ │ │ │ │ - STYLE: this.style, │ │ │ │ │ - TILEMATRIXSET: this.matrixSet, │ │ │ │ │ - TILEMATRIX: this.matrix.identifier, │ │ │ │ │ - TILEROW: info.row, │ │ │ │ │ - TILECOL: info.col, │ │ │ │ │ - FORMAT: this.format │ │ │ │ │ - }; │ │ │ │ │ - url = OpenLayers.Layer.Grid.prototype.getFullRequestString.apply(this, [params]); │ │ │ │ │ + /** │ │ │ │ │ + * Method: finalize │ │ │ │ │ + * Finish the geometry and call the "done" callback. │ │ │ │ │ + */ │ │ │ │ │ + finalize: function() { │ │ │ │ │ + this.origin = null; │ │ │ │ │ + this.radius = this.options.radius; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clear │ │ │ │ │ + * Clear any rendered features on the temporary layer. This is called │ │ │ │ │ + * when the handler is deactivated, canceled, or done (unless persist │ │ │ │ │ + * is true). │ │ │ │ │ + */ │ │ │ │ │ + clear: function() { │ │ │ │ │ + if (this.layer) { │ │ │ │ │ + this.layer.renderer.clear(); │ │ │ │ │ + this.layer.destroyFeatures(); │ │ │ │ │ } │ │ │ │ │ - return url; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: mergeNewParams │ │ │ │ │ - * Extend the existing layer <params> with new properties. Tiles will be │ │ │ │ │ - * reloaded with updated params in the request. │ │ │ │ │ - * │ │ │ │ │ + * Method: callback │ │ │ │ │ + * Trigger the control's named callback with the given arguments │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * newParams - {Object} Properties to extend to existing <params>. │ │ │ │ │ + * name - {String} The key for the callback that is one of the properties │ │ │ │ │ + * of the handler's callbacks object. │ │ │ │ │ + * args - {Array} An array of arguments with which to call the callback │ │ │ │ │ + * (defined by the control). │ │ │ │ │ */ │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - if (this.requestEncoding.toUpperCase() === "KVP") { │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply( │ │ │ │ │ - this, [OpenLayers.Util.upperCaseObject(newParams)] │ │ │ │ │ - ); │ │ │ │ │ + callback: function(name, args) { │ │ │ │ │ + // override the callback method to always send the polygon geometry │ │ │ │ │ + if (this.callbacks[name]) { │ │ │ │ │ + this.callbacks[name].apply(this.control, │ │ │ │ │ + [this.feature.geometry.clone()]); │ │ │ │ │ + } │ │ │ │ │ + // since sketch features are added to the temporary layer │ │ │ │ │ + // they must be cleared here if done or cancel │ │ │ │ │ + if (!this.persist && (name == "done" || name == "cancel")) { │ │ │ │ │ + this.clear(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.WMTS" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.RegularPolygon" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/ArcGIS93Rest.js │ │ │ │ │ + OpenLayers/Handler/Point.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.ArcGIS93Rest │ │ │ │ │ - * Instances of OpenLayers.Layer.ArcGIS93Rest are used to display data from │ │ │ │ │ - * ESRI ArcGIS Server 9.3 (and up?) Mapping Services using the REST API. │ │ │ │ │ - * Create a new ArcGIS93Rest layer with the <OpenLayers.Layer.ArcGIS93Rest> │ │ │ │ │ - * constructor. More detail on the REST API is available at │ │ │ │ │ - * http://sampleserver1.arcgisonline.com/ArcGIS/SDK/REST/index.html ; │ │ │ │ │ - * specifically, the URL provided to this layer should be an export service │ │ │ │ │ - * URL: http://sampleserver1.arcgisonline.com/ArcGIS/SDK/REST/export.html │ │ │ │ │ + * Class: OpenLayers.Handler.Point │ │ │ │ │ + * Handler to draw a point on the map. Point is displayed on activation, │ │ │ │ │ + * moves on mouse move, and is finished on mouse up. The handler triggers │ │ │ │ │ + * callbacks for 'done', 'cancel', and 'modify'. The modify callback is │ │ │ │ │ + * called with each change in the sketch and will receive the latest point │ │ │ │ │ + * drawn. Create a new instance with the <OpenLayers.Handler.Point> │ │ │ │ │ + * constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.ArcGIS93Rest = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ +OpenLayers.Handler.Point = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: DEFAULT_PARAMS │ │ │ │ │ - * {Object} Hashtable of default parameter key/value pairs │ │ │ │ │ + * Property: point │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} The currently drawn point │ │ │ │ │ */ │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - format: "png" │ │ │ │ │ - }, │ │ │ │ │ + point: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} Default is true for ArcGIS93Rest layer │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} The temporary drawing layer │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + layer: null, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: multi │ │ │ │ │ + * {Boolean} Cast features to multi-part geometries before passing to the │ │ │ │ │ + * layer. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + multi: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.ArcGIS93Rest │ │ │ │ │ - * Create a new ArcGIS93Rest layer object. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var arcims = new OpenLayers.Layer.ArcGIS93Rest("MyName", │ │ │ │ │ - * "http://sampleserver1.arcgisonline.com/ArcGIS/rest/services/Specialty/ESRI_StateCityHighway_USA/MapServer/export", │ │ │ │ │ - * { │ │ │ │ │ - * layers: "0,1,2" │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} A name for the layer │ │ │ │ │ - * url - {String} Base url for the ArcGIS server REST service │ │ │ │ │ - * options - {Object} An object with key/value pairs representing the │ │ │ │ │ - * options and option values. │ │ │ │ │ - * │ │ │ │ │ - * Valid Options: │ │ │ │ │ - * format - {String} MIME type of desired image type. │ │ │ │ │ - * layers - {String} Comma-separated list of layers to display. │ │ │ │ │ - * srs - {String} Projection ID. │ │ │ │ │ + * APIProperty: citeCompliant │ │ │ │ │ + * {Boolean} If set to true, coordinates of features drawn in a map extent │ │ │ │ │ + * crossing the date line won't exceed the world bounds. Default is false. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - //uppercase params │ │ │ │ │ - params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ - newArguments.push(name, url, params, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.params, │ │ │ │ │ - OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS) │ │ │ │ │ - ); │ │ │ │ │ + citeCompliant: false, │ │ │ │ │ │ │ │ │ │ - //layer is transparent │ │ │ │ │ - if (this.params.TRANSPARENT && │ │ │ │ │ - this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ + /** │ │ │ │ │ + * Property: mouseDown │ │ │ │ │ + * {Boolean} The mouse is down │ │ │ │ │ + */ │ │ │ │ │ + mouseDown: false, │ │ │ │ │ │ │ │ │ │ - // unless explicitly set in options, make layer an overlay │ │ │ │ │ - if ((options == null) || (!options.isBaseLayer)) { │ │ │ │ │ - this.isBaseLayer = false; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: stoppedDown │ │ │ │ │ + * {Boolean} Indicate whether the last mousedown stopped the event │ │ │ │ │ + * propagation. │ │ │ │ │ + */ │ │ │ │ │ + stoppedDown: null, │ │ │ │ │ │ │ │ │ │ - // jpegs can never be transparent, so intelligently switch the │ │ │ │ │ - // format, depending on the browser's capabilities │ │ │ │ │ - if (this.params.FORMAT == "jpg") { │ │ │ │ │ - this.params.FORMAT = OpenLayers.Util.alphaHack() ? "gif" : │ │ │ │ │ - "png"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: lastDown │ │ │ │ │ + * {<OpenLayers.Pixel>} Location of the last mouse down │ │ │ │ │ + */ │ │ │ │ │ + lastDown: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clone │ │ │ │ │ - * Create a clone of this layer │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.ArcGIS93Rest>} An exact clone of this layer │ │ │ │ │ + * Property: lastUp │ │ │ │ │ + * {<OpenLayers.Pixel>} │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ + lastUp: null, │ │ │ │ │ │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.ArcGIS93Rest(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.params, │ │ │ │ │ - this.getOptions()); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: persist │ │ │ │ │ + * {Boolean} Leave the feature rendered until destroyFeature is called. │ │ │ │ │ + * Default is false. If set to true, the feature remains rendered until │ │ │ │ │ + * destroyFeature is called, typically by deactivating the handler or │ │ │ │ │ + * starting another drawing. │ │ │ │ │ + */ │ │ │ │ │ + persist: false, │ │ │ │ │ │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: stopDown │ │ │ │ │ + * {Boolean} Stop event propagation on mousedown. Must be false to │ │ │ │ │ + * allow "pan while drawing". Defaults to false. │ │ │ │ │ + */ │ │ │ │ │ + stopDown: false, │ │ │ │ │ │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ + /** │ │ │ │ │ + * APIPropery: stopUp │ │ │ │ │ + * {Boolean} Stop event propagation on mouse. Must be false to │ │ │ │ │ + * allow "pan while dragging". Defaults to fase. │ │ │ │ │ + */ │ │ │ │ │ + stopUp: false, │ │ │ │ │ │ │ │ │ │ - return obj; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: layerOptions │ │ │ │ │ + * {Object} Any optional properties to be set on the sketch layer. │ │ │ │ │ + */ │ │ │ │ │ + layerOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: pixelTolerance │ │ │ │ │ + * {Number} Maximum number of pixels between down and up (mousedown │ │ │ │ │ + * and mouseup, or touchstart and touchend) for the handler to │ │ │ │ │ + * add a new point. If set to an integer value, if the │ │ │ │ │ + * displacement between down and up is great to this value │ │ │ │ │ + * no point will be added. Default value is 5. │ │ │ │ │ + */ │ │ │ │ │ + pixelTolerance: 5, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: lastTouchPx │ │ │ │ │ + * {<OpenLayers.Pixel>} The last pixel used to know the distance between │ │ │ │ │ + * two touches (for double touch). │ │ │ │ │ + */ │ │ │ │ │ + lastTouchPx: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * Return an image url this layer. │ │ │ │ │ + * Constructor: OpenLayers.Handler.Point │ │ │ │ │ + * Create a new point handler. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox for the │ │ │ │ │ - * request. │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ + * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ + * functions. Various callbacks described below. │ │ │ │ │ + * options - {Object} An optional object with properties to be set on the │ │ │ │ │ + * handler │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the map image's url. │ │ │ │ │ - */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - │ │ │ │ │ - // ArcGIS Server only wants the numeric portion of the projection ID. │ │ │ │ │ - var projWords = this.projection.getCode().split(":"); │ │ │ │ │ - var srid = projWords[projWords.length - 1]; │ │ │ │ │ + * Named callbacks: │ │ │ │ │ + * create - Called when a sketch is first created. Callback called with │ │ │ │ │ + * the creation point geometry and sketch feature. │ │ │ │ │ + * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ + * geometry and the sketch feature. │ │ │ │ │ + * done - Called when the point drawing is finished. The callback will │ │ │ │ │ + * recieve a single argument, the point geometry. │ │ │ │ │ + * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ + * cancel callback will receive a geometry. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ + this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style['default'], {}); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var imageSize = this.getImageSize(); │ │ │ │ │ - var newParams = { │ │ │ │ │ - 'BBOX': bounds.toBBOX(), │ │ │ │ │ - 'SIZE': imageSize.w + "," + imageSize.h, │ │ │ │ │ - // We always want image, the other options were json, image with a whole lotta html around it, etc. │ │ │ │ │ - 'F': "image", │ │ │ │ │ - 'BBOXSR': srid, │ │ │ │ │ - 'IMAGESR': srid │ │ │ │ │ - }; │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Now add the filter parameters. │ │ │ │ │ - if (this.layerDefs) { │ │ │ │ │ - var layerDefStrList = []; │ │ │ │ │ - var layerID; │ │ │ │ │ - for (layerID in this.layerDefs) { │ │ │ │ │ - if (this.layerDefs.hasOwnProperty(layerID)) { │ │ │ │ │ - if (this.layerDefs[layerID]) { │ │ │ │ │ - layerDefStrList.push(layerID); │ │ │ │ │ - layerDefStrList.push(":"); │ │ │ │ │ - layerDefStrList.push(this.layerDefs[layerID]); │ │ │ │ │ - layerDefStrList.push(";"); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (layerDefStrList.length > 0) { │ │ │ │ │ - newParams['LAYERDEFS'] = layerDefStrList.join(""); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * turn on the handler │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (!OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - var requestString = this.getFullRequestString(newParams); │ │ │ │ │ - return requestString; │ │ │ │ │ + // create temporary vector layer for rendering geometry sketch │ │ │ │ │ + // TBD: this could be moved to initialize/destroy - setting visibility here │ │ │ │ │ + var options = OpenLayers.Util.extend({ │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + // indicate that the temp vector layer will never be out of range │ │ │ │ │ + // without this, resolution properties must be specified at the │ │ │ │ │ + // map-level for this temporary layer to init its resolutions │ │ │ │ │ + // correctly │ │ │ │ │ + calculateInRange: OpenLayers.Function.True, │ │ │ │ │ + wrapDateLine: this.citeCompliant │ │ │ │ │ + }, this.layerOptions); │ │ │ │ │ + this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ + this.map.addLayer(this.layer); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setLayerFilter │ │ │ │ │ - * addTile creates a tile, initializes it, and adds it to the layer div. │ │ │ │ │ + * Method: createFeature │ │ │ │ │ + * Add temporary features │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * id - {String} The id of the layer to which the filter applies. │ │ │ │ │ - * queryDef - {String} A sql-ish query filter, for more detail see the ESRI │ │ │ │ │ - * documentation at http://sampleserver1.arcgisonline.com/ArcGIS/SDK/REST/export.html │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} A pixel location on the map. │ │ │ │ │ */ │ │ │ │ │ - setLayerFilter: function(id, queryDef) { │ │ │ │ │ - if (!this.layerDefs) { │ │ │ │ │ - this.layerDefs = {}; │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ + lonlat.lon, lonlat.lat │ │ │ │ │ + ); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.point]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * turn off the handler │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (!OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - if (queryDef) { │ │ │ │ │ - this.layerDefs[id] = queryDef; │ │ │ │ │ - } else { │ │ │ │ │ - delete this.layerDefs[id]; │ │ │ │ │ + this.cancel(); │ │ │ │ │ + // If a layer's map property is set to null, it means that that layer │ │ │ │ │ + // isn't added to the map. Since we ourself added the layer to the map │ │ │ │ │ + // in activate(), we can assume that if this.layer.map is null it means │ │ │ │ │ + // that the layer has been destroyed (as a result of map.destroy() for │ │ │ │ │ + // example. │ │ │ │ │ + if (this.layer.map != null) { │ │ │ │ │ + this.destroyFeature(true); │ │ │ │ │ + this.layer.destroy(false); │ │ │ │ │ } │ │ │ │ │ + this.layer = null; │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clearLayerFilter │ │ │ │ │ - * Clears layer filters, either from a specific layer, │ │ │ │ │ - * or all of them. │ │ │ │ │ + * Method: destroyFeature │ │ │ │ │ + * Destroy the temporary geometries │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * id - {String} The id of the layer from which to remove any │ │ │ │ │ - * filter. If unspecified/blank, all filters │ │ │ │ │ - * will be removed. │ │ │ │ │ + * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ */ │ │ │ │ │ - clearLayerFilter: function(id) { │ │ │ │ │ - if (id) { │ │ │ │ │ - delete this.layerDefs[id]; │ │ │ │ │ - } else { │ │ │ │ │ - delete this.layerDefs; │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + if (this.layer && (force || !this.persist)) { │ │ │ │ │ + this.layer.destroyFeatures(); │ │ │ │ │ } │ │ │ │ │ + this.point = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: mergeNewParams │ │ │ │ │ - * Catch changeParams and uppercase the new params to be merged in │ │ │ │ │ - * before calling changeParams on the super class. │ │ │ │ │ - * │ │ │ │ │ - * Once params have been changed, the tiles will be reloaded with │ │ │ │ │ - * the new parameters. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newParams - {Object} Hashtable of new params to use │ │ │ │ │ + * Method: destroyPersistedFeature │ │ │ │ │ + * Destroy the persisted feature. │ │ │ │ │ */ │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ - var newArguments = [upperParams]; │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, │ │ │ │ │ - newArguments); │ │ │ │ │ + destroyPersistedFeature: function() { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + if (layer && layer.features.length > 1) { │ │ │ │ │ + this.layer.features[0].destroy(); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.ArcGIS93Rest" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Zoomify.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/* │ │ │ │ │ - * Development supported by a R&D grant DC08P02OUK006 - Old Maps Online │ │ │ │ │ - * (www.oldmapsonline.org) from Ministry of Culture of the Czech Republic. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Layer.Zoomify │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Grid> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Layer.Zoomify = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Property: size │ │ │ │ │ - * {<OpenLayers.Size>} The Zoomify image size in pixels. │ │ │ │ │ + * Method: finalize │ │ │ │ │ + * Finish the geometry and call the "done" callback. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * cancel - {Boolean} Call cancel instead of done callback. Default │ │ │ │ │ + * is false. │ │ │ │ │ */ │ │ │ │ │ - size: null, │ │ │ │ │ + finalize: function(cancel) { │ │ │ │ │ + var key = cancel ? "cancel" : "done"; │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + this.lastDown = null; │ │ │ │ │ + this.lastUp = null; │ │ │ │ │ + this.lastTouchPx = null; │ │ │ │ │ + this.callback(key, [this.geometryClone()]); │ │ │ │ │ + this.destroyFeature(cancel); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: isBaseLayer │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * APIMethod: cancel │ │ │ │ │ + * Finish the geometry and call the "cancel" callback. │ │ │ │ │ */ │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ + cancel: function() { │ │ │ │ │ + this.finalize(true); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: standardTileSize │ │ │ │ │ - * {Integer} The size of a standard (non-border) square tile in pixels. │ │ │ │ │ + * Method: click │ │ │ │ │ + * Handle clicks. Clicks are stopped from propagating to other listeners │ │ │ │ │ + * on map.events or other dom elements. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - standardTileSize: 256, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: tileOriginCorner │ │ │ │ │ - * {String} This layer uses top-left as tile origin │ │ │ │ │ - **/ │ │ │ │ │ - tileOriginCorner: "tl", │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: numberOfTiers │ │ │ │ │ - * {Integer} Depth of the Zoomify pyramid, number of tiers (zoom levels) │ │ │ │ │ - * - filled during Zoomify pyramid initialization. │ │ │ │ │ + * Method: dblclick │ │ │ │ │ + * Handle double-clicks. Double-clicks are stopped from propagating to other │ │ │ │ │ + * listeners on map.events or other dom elements. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - numberOfTiers: 0, │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: tileCountUpToTier │ │ │ │ │ - * {Array(Integer)} Number of tiles up to the given tier of pyramid. │ │ │ │ │ - * - filled during Zoomify pyramid initialization. │ │ │ │ │ + * Method: modifyFeature │ │ │ │ │ + * Modify the existing geometry given a pixel location. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} A pixel location on the map. │ │ │ │ │ */ │ │ │ │ │ - tileCountUpToTier: null, │ │ │ │ │ + modifyFeature: function(pixel) { │ │ │ │ │ + if (!this.point) { │ │ │ │ │ + this.createFeature(pixel); │ │ │ │ │ + } │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point.geometry.x = lonlat.lon; │ │ │ │ │ + this.point.geometry.y = lonlat.lat; │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.point, false]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: tierSizeInTiles │ │ │ │ │ - * {Array(<OpenLayers.Size>)} Size (in tiles) for each tier of pyramid. │ │ │ │ │ - * - filled during Zoomify pyramid initialization. │ │ │ │ │ + * Method: drawFeature │ │ │ │ │ + * Render features on the temporary layer. │ │ │ │ │ */ │ │ │ │ │ - tierSizeInTiles: null, │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.point, this.style); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: tierImageSize │ │ │ │ │ - * {Array(<OpenLayers.Size>)} Image size in pixels for each pyramid tier. │ │ │ │ │ - * - filled during Zoomify pyramid initialization. │ │ │ │ │ + * Method: getGeometry │ │ │ │ │ + * Return the sketch geometry. If <multi> is true, this will return │ │ │ │ │ + * a multi-part geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Point>} │ │ │ │ │ */ │ │ │ │ │ - tierImageSize: null, │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.point && this.point.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiPoint([geometry]); │ │ │ │ │ + } │ │ │ │ │ + return geometry; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Zoomify │ │ │ │ │ + * Method: geometryClone │ │ │ │ │ + * Return a clone of the relevant geometry. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} A name for the layer. │ │ │ │ │ - * url - {String} - Relative or absolute path to the image or more │ │ │ │ │ - * precisly to the TileGroup[X] directories root. │ │ │ │ │ - * Flash plugin use the variable name "zoomifyImagePath" for this. │ │ │ │ │ - * size - {<OpenLayers.Size>} The size (in pixels) of the image. │ │ │ │ │ - * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry>} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(name, url, size, options) { │ │ │ │ │ - │ │ │ │ │ - // initilize the Zoomify pyramid for given size │ │ │ │ │ - this.initializeZoomify(size); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [ │ │ │ │ │ - name, url, size, {}, │ │ │ │ │ - options │ │ │ │ │ - ]); │ │ │ │ │ + geometryClone: function() { │ │ │ │ │ + var geom = this.getGeometry(); │ │ │ │ │ + return geom && geom.clone(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: initializeZoomify │ │ │ │ │ - * It generates constants for all tiers of the Zoomify pyramid │ │ │ │ │ - * │ │ │ │ │ + * Method: mousedown │ │ │ │ │ + * Handle mousedown. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>} The size of the image in pixels │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - initializeZoomify: function(size) { │ │ │ │ │ - │ │ │ │ │ - var imageSize = size.clone(); │ │ │ │ │ - this.size = size.clone(); │ │ │ │ │ - var tiles = new OpenLayers.Size( │ │ │ │ │ - Math.ceil(imageSize.w / this.standardTileSize), │ │ │ │ │ - Math.ceil(imageSize.h / this.standardTileSize) │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - this.tierSizeInTiles = [tiles]; │ │ │ │ │ - this.tierImageSize = [imageSize]; │ │ │ │ │ - │ │ │ │ │ - while (imageSize.w > this.standardTileSize || │ │ │ │ │ - imageSize.h > this.standardTileSize) { │ │ │ │ │ - │ │ │ │ │ - imageSize = new OpenLayers.Size( │ │ │ │ │ - Math.floor(imageSize.w / 2), │ │ │ │ │ - Math.floor(imageSize.h / 2) │ │ │ │ │ - ); │ │ │ │ │ - tiles = new OpenLayers.Size( │ │ │ │ │ - Math.ceil(imageSize.w / this.standardTileSize), │ │ │ │ │ - Math.ceil(imageSize.h / this.standardTileSize) │ │ │ │ │ - ); │ │ │ │ │ - this.tierSizeInTiles.push(tiles); │ │ │ │ │ - this.tierImageSize.push(imageSize); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.tierSizeInTiles.reverse(); │ │ │ │ │ - this.tierImageSize.reverse(); │ │ │ │ │ - │ │ │ │ │ - this.numberOfTiers = this.tierSizeInTiles.length; │ │ │ │ │ - var resolutions = [1]; │ │ │ │ │ - this.tileCountUpToTier = [0]; │ │ │ │ │ - for (var i = 1; i < this.numberOfTiers; i++) { │ │ │ │ │ - resolutions.unshift(Math.pow(2, i)); │ │ │ │ │ - this.tileCountUpToTier.push( │ │ │ │ │ - this.tierSizeInTiles[i - 1].w * this.tierSizeInTiles[i - 1].h + │ │ │ │ │ - this.tileCountUpToTier[i - 1] │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (!this.serverResolutions) { │ │ │ │ │ - this.serverResolutions = resolutions; │ │ │ │ │ - } │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + return this.down(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod:destroy │ │ │ │ │ + * Method: touchstart │ │ │ │ │ + * Handle touchstart. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - // for now, nothing special to do here. │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.destroy.apply(this, arguments); │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + this.lastTouchPx = evt.xy; │ │ │ │ │ + return this.down(evt); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Remove from memory the Zoomify pyramid - is that enough? │ │ │ │ │ - this.tileCountUpToTier.length = 0; │ │ │ │ │ - this.tierSizeInTiles.length = 0; │ │ │ │ │ - this.tierImageSize.length = 0; │ │ │ │ │ + /** │ │ │ │ │ + * Method: mousemove │ │ │ │ │ + * Handle mousemove. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ + */ │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + return this.move(evt); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Method: touchmove │ │ │ │ │ + * Handle touchmove. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ + */ │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + this.lastTouchPx = evt.xy; │ │ │ │ │ + return this.move(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clone │ │ │ │ │ + * Method: mouseup │ │ │ │ │ + * Handle mouseup. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ + */ │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + return this.up(evt); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: touchend │ │ │ │ │ + * Handle touchend. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {Object} │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Zoomify>} An exact clone of this <OpenLayers.Layer.Zoomify> │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + evt.xy = this.lastTouchPx; │ │ │ │ │ + return this.up(evt); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Zoomify(this.name, │ │ │ │ │ - this.url, │ │ │ │ │ - this.size, │ │ │ │ │ - this.options); │ │ │ │ │ + /** │ │ │ │ │ + * Method: down │ │ │ │ │ + * Handle mousedown and touchstart. Adjust the geometry and redraw. │ │ │ │ │ + * Return determines whether to propagate the event on the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ + */ │ │ │ │ │ + down: function(evt) { │ │ │ │ │ + this.mouseDown = true; │ │ │ │ │ + this.lastDown = evt.xy; │ │ │ │ │ + if (!this.touch) { // no point displayed until up on touch devices │ │ │ │ │ + this.modifyFeature(evt.xy); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - //get all additions from superclasses │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - │ │ │ │ │ - // copy/set any non-init, non-simple values here │ │ │ │ │ - │ │ │ │ │ - return obj; │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + return !this.stopDown; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getURL │ │ │ │ │ - * │ │ │ │ │ + * Method: move │ │ │ │ │ + * Handle mousemove and touchmove. Adjust the geometry and redraw. │ │ │ │ │ + * Return determines whether to propagate the event on the map. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A string with the layer's url and parameters and also the │ │ │ │ │ - * passed-in bounds and appropriate tile size specified as │ │ │ │ │ - * parameters │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((this.tileOrigin.lat - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getZoomForResolution(res); │ │ │ │ │ - │ │ │ │ │ - var tileIndex = x + y * this.tierSizeInTiles[z].w + this.tileCountUpToTier[z]; │ │ │ │ │ - var path = "TileGroup" + Math.floor((tileIndex) / 256) + │ │ │ │ │ - "/" + z + "-" + x + "-" + y + ".jpg"; │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(path, url); │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + if (!this.touch // no point displayed until up on touch devices │ │ │ │ │ + && │ │ │ │ │ + (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ + this.modifyFeature(evt.xy); │ │ │ │ │ } │ │ │ │ │ - return url + path; │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getImageSize │ │ │ │ │ - * getImageSize returns size for a particular tile. If bounds are given as │ │ │ │ │ - * first argument, size is calculated (bottom-right tiles are non square). │ │ │ │ │ + * Method: up │ │ │ │ │ + * Handle mouseup and touchend. Send the latest point in the geometry to the control. │ │ │ │ │ + * Return determines whether to propagate the event on the map. │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - getImageSize: function() { │ │ │ │ │ - if (arguments.length > 0) { │ │ │ │ │ - var bounds = this.adjustBounds(arguments[0]); │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((this.tileOrigin.lat - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getZoomForResolution(res); │ │ │ │ │ - var w = this.standardTileSize; │ │ │ │ │ - var h = this.standardTileSize; │ │ │ │ │ - if (x == this.tierSizeInTiles[z].w - 1) { │ │ │ │ │ - var w = this.tierImageSize[z].w % this.standardTileSize; │ │ │ │ │ + up: function(evt) { │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + │ │ │ │ │ + // check keyboard modifiers │ │ │ │ │ + if (!this.checkModifiers(evt)) { │ │ │ │ │ + return true; │ │ │ │ │ + } │ │ │ │ │ + // ignore double-clicks │ │ │ │ │ + if (this.lastUp && this.lastUp.equals(evt.xy)) { │ │ │ │ │ + return true; │ │ │ │ │ + } │ │ │ │ │ + if (this.lastDown && this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ + this.pixelTolerance)) { │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy); │ │ │ │ │ } │ │ │ │ │ - if (y == this.tierSizeInTiles[z].h - 1) { │ │ │ │ │ - var h = this.tierImageSize[z].h % this.standardTileSize; │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature(); │ │ │ │ │ } │ │ │ │ │ - return (new OpenLayers.Size(w, h)); │ │ │ │ │ + this.lastUp = evt.xy; │ │ │ │ │ + this.finalize(); │ │ │ │ │ + return !this.stopUp; │ │ │ │ │ } else { │ │ │ │ │ - return this.tileSize; │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setMap │ │ │ │ │ - * When the layer is added to a map, then we can fetch our origin │ │ │ │ │ - * (if we don't have one.) │ │ │ │ │ + * Method: mouseout │ │ │ │ │ + * Handle mouse out. For better user experience reset mouseDown │ │ │ │ │ + * and stoppedDown when the mouse leaves the map viewport. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.map.maxExtent.left, │ │ │ │ │ - this.map.maxExtent.top); │ │ │ │ │ + mouseout: function(evt) { │ │ │ │ │ + if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Zoomify" │ │ │ │ │ + /** │ │ │ │ │ + * Method: passesTolerance │ │ │ │ │ + * Determine whether the event is within the optional pixel tolerance. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The event is within the pixel tolerance (if specified). │ │ │ │ │ + */ │ │ │ │ │ + passesTolerance: function(pixel1, pixel2, tolerance) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + │ │ │ │ │ + if (tolerance != null && pixel1 && pixel2) { │ │ │ │ │ + var dist = pixel1.distanceTo(pixel2); │ │ │ │ │ + if (dist > tolerance) { │ │ │ │ │ + passes = false; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return passes; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Point" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Google/v3.js │ │ │ │ │ + OpenLayers/Handler/Path.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Google.js │ │ │ │ │ + * @requires OpenLayers/Handler/Point.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: OpenLayers.Layer.Google.v3 │ │ │ │ │ - * │ │ │ │ │ - * Mixin providing functionality specific to the Google Maps API v3. │ │ │ │ │ - * │ │ │ │ │ - * To use this layer, you must include the GMaps v3 API in your html. │ │ │ │ │ - * │ │ │ │ │ - * Note that this layer configures the google.maps.map object with the │ │ │ │ │ - * "disableDefaultUI" option set to true. Using UI controls that the Google │ │ │ │ │ - * Maps API provides is not supported by the OpenLayers API. │ │ │ │ │ + * Class: OpenLayers.Handler.Path │ │ │ │ │ + * Handler to draw a path on the map. Path is displayed on mouse down, │ │ │ │ │ + * moves on mouse move, and is finished on mouse up. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Handler.Point> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Google.v3 = { │ │ │ │ │ +OpenLayers.Handler.Path = OpenLayers.Class(OpenLayers.Handler.Point, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: DEFAULTS │ │ │ │ │ - * {Object} It is not recommended to change the properties set here. Note │ │ │ │ │ - * that Google.v3 layers only work when sphericalMercator is set to true. │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * { │ │ │ │ │ - * sphericalMercator: true, │ │ │ │ │ - * projection: "EPSG:900913" │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ + * Property: line │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - DEFAULTS: { │ │ │ │ │ - sphericalMercator: true, │ │ │ │ │ - projection: "EPSG:900913" │ │ │ │ │ + line: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxVertices │ │ │ │ │ + * {Number} The maximum number of vertices which can be drawn by this │ │ │ │ │ + * handler. When the number of vertices reaches maxVertices, the │ │ │ │ │ + * geometry is automatically finalized. Default is null. │ │ │ │ │ + */ │ │ │ │ │ + maxVertices: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: doubleTouchTolerance │ │ │ │ │ + * {Number} Maximum number of pixels between two touches for │ │ │ │ │ + * the gesture to be considered a "finalize feature" action. │ │ │ │ │ + * Default is 20. │ │ │ │ │ + */ │ │ │ │ │ + doubleTouchTolerance: 20, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: freehand │ │ │ │ │ + * {Boolean} In freehand mode, the handler starts the path on mouse down, │ │ │ │ │ + * adds a point for every mouse move, and finishes the path on mouse up. │ │ │ │ │ + * Outside of freehand mode, a point is added to the path on every mouse │ │ │ │ │ + * click and double-click finishes the path. │ │ │ │ │ + */ │ │ │ │ │ + freehand: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: freehandToggle │ │ │ │ │ + * {String} If set, freehandToggle is checked on mouse events and will set │ │ │ │ │ + * the freehand mode to the opposite of this.freehand. To disallow │ │ │ │ │ + * toggling between freehand and non-freehand mode, set freehandToggle to │ │ │ │ │ + * null. Acceptable toggle values are 'shiftKey', 'ctrlKey', and 'altKey'. │ │ │ │ │ + */ │ │ │ │ │ + freehandToggle: 'shiftKey', │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: timerId │ │ │ │ │ + * {Integer} The timer used to test the double touch. │ │ │ │ │ + */ │ │ │ │ │ + timerId: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: redoStack │ │ │ │ │ + * {Array} Stack containing points removed with <undo>. │ │ │ │ │ + */ │ │ │ │ │ + redoStack: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Handler.Path │ │ │ │ │ + * Create a new path hander │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ + * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ + * functions. Various callbacks described below. │ │ │ │ │ + * options - {Object} An optional object with properties to be set on the │ │ │ │ │ + * handler │ │ │ │ │ + * │ │ │ │ │ + * Named callbacks: │ │ │ │ │ + * create - Called when a sketch is first created. Callback called with │ │ │ │ │ + * the creation point geometry and sketch feature. │ │ │ │ │ + * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ + * geometry and the sketch feature. │ │ │ │ │ + * point - Called as each point is added. Receives the new point geometry. │ │ │ │ │ + * done - Called when the point drawing is finished. The callback will │ │ │ │ │ + * recieve a single argument, the linestring geometry. │ │ │ │ │ + * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ + * cancel callback will receive a geometry. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createFeature │ │ │ │ │ + * Add temporary geometries │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} The initial pixel location for the new │ │ │ │ │ + * feature. │ │ │ │ │ + */ │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ + lonlat.lon, lonlat.lat │ │ │ │ │ + ); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.line = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.LineString([this.point.geometry]) │ │ │ │ │ + ); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.line, this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: animationEnabled │ │ │ │ │ - * {Boolean} If set to true, the transition between zoom levels will be │ │ │ │ │ - * animated (if supported by the GMaps API for the device used). Set to │ │ │ │ │ - * false to match the zooming experience of other layer types. Default │ │ │ │ │ - * is true. Note that the GMaps API does not give us control over zoom │ │ │ │ │ - * animation, so if set to false, when zooming, this will make the │ │ │ │ │ - * layer temporarily invisible, wait until GMaps reports the map being │ │ │ │ │ - * idle, and make it visible again. The result will be a blank layer │ │ │ │ │ - * for a few moments while zooming. │ │ │ │ │ + * Method: destroyFeature │ │ │ │ │ + * Destroy temporary geometries │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ */ │ │ │ │ │ - animationEnabled: true, │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + OpenLayers.Handler.Point.prototype.destroyFeature.call( │ │ │ │ │ + this, force); │ │ │ │ │ + this.line = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: loadMapObject │ │ │ │ │ - * Load the GMap and register appropriate event listeners. │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroyPersistedFeature │ │ │ │ │ + * Destroy the persisted feature. │ │ │ │ │ */ │ │ │ │ │ - loadMapObject: function() { │ │ │ │ │ - if (!this.type) { │ │ │ │ │ - this.type = google.maps.MapTypeId.ROADMAP; │ │ │ │ │ + destroyPersistedFeature: function() { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + if (layer && layer.features.length > 2) { │ │ │ │ │ + this.layer.features[0].destroy(); │ │ │ │ │ } │ │ │ │ │ - var mapObject; │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - // there are already Google layers added to this map │ │ │ │ │ - mapObject = cache.mapObject; │ │ │ │ │ - // increment the layer count │ │ │ │ │ - ++cache.count; │ │ │ │ │ - } else { │ │ │ │ │ - // this is the first Google layer for this map │ │ │ │ │ - // create GMap │ │ │ │ │ - var center = this.map.getCenter(); │ │ │ │ │ - var container = document.createElement('div'); │ │ │ │ │ - container.className = "olForeignContainer"; │ │ │ │ │ - container.style.width = '100%'; │ │ │ │ │ - container.style.height = '100%'; │ │ │ │ │ - mapObject = new google.maps.Map(container, { │ │ │ │ │ - center: center ? │ │ │ │ │ - new google.maps.LatLng(center.lat, center.lon) : new google.maps.LatLng(0, 0), │ │ │ │ │ - zoom: this.map.getZoom() || 0, │ │ │ │ │ - mapTypeId: this.type, │ │ │ │ │ - disableDefaultUI: true, │ │ │ │ │ - keyboardShortcuts: false, │ │ │ │ │ - draggable: false, │ │ │ │ │ - disableDoubleClickZoom: true, │ │ │ │ │ - scrollwheel: false, │ │ │ │ │ - streetViewControl: false │ │ │ │ │ - }); │ │ │ │ │ - var googleControl = document.createElement('div'); │ │ │ │ │ - googleControl.style.width = '100%'; │ │ │ │ │ - googleControl.style.height = '100%'; │ │ │ │ │ - mapObject.controls[google.maps.ControlPosition.TOP_LEFT].push(googleControl); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // cache elements for use by any other google layers added to │ │ │ │ │ - // this same map │ │ │ │ │ - cache = { │ │ │ │ │ - googleControl: googleControl, │ │ │ │ │ - mapObject: mapObject, │ │ │ │ │ - count: 1 │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Layer.Google.cache[this.map.id] = cache; │ │ │ │ │ + /** │ │ │ │ │ + * Method: removePoint │ │ │ │ │ + * Destroy the temporary point. │ │ │ │ │ + */ │ │ │ │ │ + removePoint: function() { │ │ │ │ │ + if (this.point) { │ │ │ │ │ + this.layer.removeFeatures([this.point]); │ │ │ │ │ } │ │ │ │ │ - this.mapObject = mapObject; │ │ │ │ │ - this.setGMapVisibility(this.visibility); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: onMapResize │ │ │ │ │ + * Method: addPoint │ │ │ │ │ + * Add point to geometry. Send the point index to override │ │ │ │ │ + * the behavior of LinearRing that disregards adding duplicate points. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} The pixel location for the new point. │ │ │ │ │ */ │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - if (this.visibility) { │ │ │ │ │ - google.maps.event.trigger(this.mapObject, "resize"); │ │ │ │ │ - } │ │ │ │ │ + addPoint: function(pixel) { │ │ │ │ │ + this.layer.removeFeatures([this.point]); │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat) │ │ │ │ │ + ); │ │ │ │ │ + this.line.geometry.addComponent( │ │ │ │ │ + this.point.geometry, this.line.geometry.components.length │ │ │ │ │ + ); │ │ │ │ │ + this.layer.addFeatures([this.point]); │ │ │ │ │ + this.callback("point", [this.point.geometry, this.getGeometry()]); │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + delete this.redoStack; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setGMapVisibility │ │ │ │ │ - * Display the GMap container and associated elements. │ │ │ │ │ - * │ │ │ │ │ + * Method: insertXY │ │ │ │ │ + * Insert a point in the current sketch given x & y coordinates. The new │ │ │ │ │ + * point is inserted immediately before the most recently drawn point. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * visible - {Boolean} Display the GMap elements. │ │ │ │ │ + * x - {Number} The x-coordinate of the point. │ │ │ │ │ + * y - {Number} The y-coordinate of the point. │ │ │ │ │ */ │ │ │ │ │ - setGMapVisibility: function(visible) { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - var map = this.map; │ │ │ │ │ - if (cache) { │ │ │ │ │ - var type = this.type; │ │ │ │ │ - var layers = map.layers; │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = layers[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Google && │ │ │ │ │ - layer.visibility === true && layer.inRange === true) { │ │ │ │ │ - type = layer.type; │ │ │ │ │ - visible = true; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var container = this.mapObject.getDiv(); │ │ │ │ │ - if (visible === true) { │ │ │ │ │ - if (container.parentNode !== map.div) { │ │ │ │ │ - if (!cache.rendered) { │ │ │ │ │ - var me = this; │ │ │ │ │ - google.maps.event.addListenerOnce(this.mapObject, 'tilesloaded', function() { │ │ │ │ │ - cache.rendered = true; │ │ │ │ │ - me.setGMapVisibility(me.getVisibility()); │ │ │ │ │ - me.moveTo(me.map.getCenter()); │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - map.div.appendChild(container); │ │ │ │ │ - cache.googleControl.appendChild(map.viewPortDiv); │ │ │ │ │ - google.maps.event.trigger(this.mapObject, 'resize'); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.mapObject.setMapTypeId(type); │ │ │ │ │ - } else if (cache.googleControl.hasChildNodes()) { │ │ │ │ │ - map.div.appendChild(map.viewPortDiv); │ │ │ │ │ - map.div.removeChild(container); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + insertXY: function(x, y) { │ │ │ │ │ + this.line.geometry.addComponent( │ │ │ │ │ + new OpenLayers.Geometry.Point(x, y), │ │ │ │ │ + this.getCurrentPointIndex() │ │ │ │ │ + ); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + delete this.redoStack; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getMapContainer │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} the GMap container's div │ │ │ │ │ + * Method: insertDeltaXY │ │ │ │ │ + * Insert a point given offsets from the previously inserted point. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * dx - {Number} The x-coordinate offset of the point. │ │ │ │ │ + * dy - {Number} The y-coordinate offset of the point. │ │ │ │ │ */ │ │ │ │ │ - getMapContainer: function() { │ │ │ │ │ - return this.mapObject.getDiv(); │ │ │ │ │ + insertDeltaXY: function(dx, dy) { │ │ │ │ │ + var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ + var p0 = this.line.geometry.components[previousIndex]; │ │ │ │ │ + if (p0 && !isNaN(p0.x) && !isNaN(p0.y)) { │ │ │ │ │ + this.insertXY(p0.x + dx, p0.y + dy); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - // │ │ │ │ │ - // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ - // │ │ │ │ │ + /** │ │ │ │ │ + * Method: insertDirectionLength │ │ │ │ │ + * Insert a point in the current sketch given a direction and a length. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * direction - {Number} Degrees clockwise from the positive x-axis. │ │ │ │ │ + * length - {Number} Distance from the previously drawn point. │ │ │ │ │ + */ │ │ │ │ │ + insertDirectionLength: function(direction, length) { │ │ │ │ │ + direction *= Math.PI / 180; │ │ │ │ │ + var dx = length * Math.cos(direction); │ │ │ │ │ + var dy = length * Math.sin(direction); │ │ │ │ │ + this.insertDeltaXY(dx, dy); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectBoundsFromOLBounds │ │ │ │ │ - * │ │ │ │ │ + * Method: insertDeflectionLength │ │ │ │ │ + * Insert a point in the current sketch given a deflection and a length. │ │ │ │ │ + * The deflection should be degrees clockwise from the previously │ │ │ │ │ + * digitized segment. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * olBounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} A MapObject Bounds, translated from olBounds │ │ │ │ │ - * Returns null if null value is passed in │ │ │ │ │ + * deflection - {Number} Degrees clockwise from the previous segment. │ │ │ │ │ + * length - {Number} Distance from the previously drawn point. │ │ │ │ │ */ │ │ │ │ │ - getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ - var moBounds = null; │ │ │ │ │ - if (olBounds != null) { │ │ │ │ │ - var sw = this.sphericalMercator ? │ │ │ │ │ - this.inverseMercator(olBounds.bottom, olBounds.left) : │ │ │ │ │ - new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ - var ne = this.sphericalMercator ? │ │ │ │ │ - this.inverseMercator(olBounds.top, olBounds.right) : │ │ │ │ │ - new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ - moBounds = new google.maps.LatLngBounds( │ │ │ │ │ - new google.maps.LatLng(sw.lat, sw.lon), │ │ │ │ │ - new google.maps.LatLng(ne.lat, ne.lon) │ │ │ │ │ + insertDeflectionLength: function(deflection, length) { │ │ │ │ │ + var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ + if (previousIndex > 0) { │ │ │ │ │ + var p1 = this.line.geometry.components[previousIndex]; │ │ │ │ │ + var p0 = this.line.geometry.components[previousIndex - 1]; │ │ │ │ │ + var theta = Math.atan2(p1.y - p0.y, p1.x - p0.x); │ │ │ │ │ + this.insertDirectionLength( │ │ │ │ │ + (theta * 180 / Math.PI) + deflection, length │ │ │ │ │ ); │ │ │ │ │ } │ │ │ │ │ - return moBounds; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Interface Controls * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - // LonLat - Pixel Translation │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectLonLatFromMapObjectPixel │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moPixel - {Object} MapObject Pixel format │ │ │ │ │ + * Method: getCurrentPointIndex │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} MapObject LonLat translated from MapObject Pixel │ │ │ │ │ + * {Number} The index of the most recently drawn point. │ │ │ │ │ */ │ │ │ │ │ - getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - var size = this.map.getSize(); │ │ │ │ │ - var lon = this.getLongitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ - var lat = this.getLatitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ - var res = this.map.getResolution(); │ │ │ │ │ + getCurrentPointIndex: function() { │ │ │ │ │ + return this.line.geometry.components.length - 1; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var delta_x = moPixel.x - (size.w / 2); │ │ │ │ │ - var delta_y = moPixel.y - (size.h / 2); │ │ │ │ │ │ │ │ │ │ - var lonlat = new OpenLayers.LonLat( │ │ │ │ │ - lon + delta_x * res, │ │ │ │ │ - lat - delta_y * res │ │ │ │ │ - ); │ │ │ │ │ + /** │ │ │ │ │ + * Method: undo │ │ │ │ │ + * Remove the most recently added point in the sketch geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} A point was removed. │ │ │ │ │ + */ │ │ │ │ │ + undo: function() { │ │ │ │ │ + var geometry = this.line.geometry; │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var index = this.getCurrentPointIndex() - 1; │ │ │ │ │ + var target = components[index]; │ │ │ │ │ + var undone = geometry.removeComponent(target); │ │ │ │ │ + if (undone) { │ │ │ │ │ + // On touch devices, set the current ("mouse location") point to │ │ │ │ │ + // match the last digitized point. │ │ │ │ │ + if (this.touch && index > 0) { │ │ │ │ │ + components = geometry.components; // safety │ │ │ │ │ + var lastpt = components[index - 1]; │ │ │ │ │ + var curptidx = this.getCurrentPointIndex(); │ │ │ │ │ + var curpt = components[curptidx]; │ │ │ │ │ + curpt.x = lastpt.x; │ │ │ │ │ + curpt.y = lastpt.y; │ │ │ │ │ + } │ │ │ │ │ + if (!this.redoStack) { │ │ │ │ │ + this.redoStack = []; │ │ │ │ │ + } │ │ │ │ │ + this.redoStack.push(target); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + } │ │ │ │ │ + return undone; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - lonlat = lonlat.wrapDateLine(this.maxExtent); │ │ │ │ │ + /** │ │ │ │ │ + * Method: redo │ │ │ │ │ + * Reinsert the most recently removed point resulting from an <undo> call. │ │ │ │ │ + * The undo stack is deleted whenever a point is added by other means. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} A point was added. │ │ │ │ │ + */ │ │ │ │ │ + redo: function() { │ │ │ │ │ + var target = this.redoStack && this.redoStack.pop(); │ │ │ │ │ + if (target) { │ │ │ │ │ + this.line.geometry.addComponent(target, this.getCurrentPointIndex()); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ } │ │ │ │ │ - return this.getMapObjectLonLatFromLonLat(lonlat.lon, lonlat.lat); │ │ │ │ │ + return !!target; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectPixelFromMapObjectLonLat │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ - * │ │ │ │ │ + * Method: freehandMode │ │ │ │ │ + * Determine whether to behave in freehand mode or not. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} MapObject Pixel transtlated from MapObject LonLat │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - var res = this.map.getResolution(); │ │ │ │ │ - var extent = this.map.getExtent(); │ │ │ │ │ - return this.getMapObjectPixelFromXY((1 / res * (lon - extent.left)), │ │ │ │ │ - (1 / res * (extent.top - lat))); │ │ │ │ │ + freehandMode: function(evt) { │ │ │ │ │ + return (this.freehandToggle && evt[this.freehandToggle]) ? │ │ │ │ │ + !this.freehand : this.freehand; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setMapObjectCenter │ │ │ │ │ - * Set the mapObject to the specified center and zoom │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: modifyFeature │ │ │ │ │ + * Modify the existing geometry given the new point │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * center - {Object} MapObject LonLat format │ │ │ │ │ - * zoom - {int} MapObject zoom format │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} The updated pixel location for the latest │ │ │ │ │ + * point. │ │ │ │ │ + * drawing - {Boolean} Indicate if we're currently drawing. │ │ │ │ │ */ │ │ │ │ │ - setMapObjectCenter: function(center, zoom) { │ │ │ │ │ - if (this.animationEnabled === false && zoom != this.mapObject.zoom) { │ │ │ │ │ - var mapContainer = this.getMapContainer(); │ │ │ │ │ - google.maps.event.addListenerOnce( │ │ │ │ │ - this.mapObject, │ │ │ │ │ - "idle", │ │ │ │ │ - function() { │ │ │ │ │ - mapContainer.style.visibility = ""; │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - mapContainer.style.visibility = "hidden"; │ │ │ │ │ + modifyFeature: function(pixel, drawing) { │ │ │ │ │ + if (!this.line) { │ │ │ │ │ + this.createFeature(pixel); │ │ │ │ │ } │ │ │ │ │ - this.mapObject.setOptions({ │ │ │ │ │ - center: center, │ │ │ │ │ - zoom: zoom │ │ │ │ │ - }); │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point.geometry.x = lonlat.lon; │ │ │ │ │ + this.point.geometry.y = lonlat.lat; │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.getSketch(), drawing]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawFeature │ │ │ │ │ + * Render geometries on the temporary layer. │ │ │ │ │ + */ │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.line, this.style); │ │ │ │ │ + this.layer.drawFeature(this.point, this.style); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Bounds │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getMapObjectZoomFromMapObjectBounds │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * moBounds - {Object} MapObject Bounds format │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: getSketch │ │ │ │ │ + * Return the sketch feature. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} MapObject Zoom for specified MapObject Bounds │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - return this.mapObject.getBoundsZoomLevel(moBounds); │ │ │ │ │ + getSketch: function() { │ │ │ │ │ + return this.line; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /************************************ │ │ │ │ │ - * * │ │ │ │ │ - * MapObject Primitives * │ │ │ │ │ - * * │ │ │ │ │ - ************************************/ │ │ │ │ │ + /** │ │ │ │ │ + * Method: getGeometry │ │ │ │ │ + * Return the sketch geometry. If <multi> is true, this will return │ │ │ │ │ + * a multi-part geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.LineString>} │ │ │ │ │ + */ │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.line && this.line.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiLineString([geometry]); │ │ │ │ │ + } │ │ │ │ │ + return geometry; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * method: touchstart │ │ │ │ │ + * handle touchstart. │ │ │ │ │ + * │ │ │ │ │ + * parameters: │ │ │ │ │ + * evt - {event} the browser event │ │ │ │ │ + * │ │ │ │ │ + * returns: │ │ │ │ │ + * {boolean} allow event propagation │ │ │ │ │ + */ │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + if (this.timerId && │ │ │ │ │ + this.passesTolerance(this.lastTouchPx, evt.xy, │ │ │ │ │ + this.doubleTouchTolerance)) { │ │ │ │ │ + // double-tap, finalize the geometry │ │ │ │ │ + this.finishGeometry(); │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + return false; │ │ │ │ │ + } else { │ │ │ │ │ + if (this.timerId) { │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + } │ │ │ │ │ + this.timerId = window.setTimeout( │ │ │ │ │ + OpenLayers.Function.bind(function() { │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + }, this), 300); │ │ │ │ │ + return OpenLayers.Handler.Point.prototype.touchstart.call(this, evt); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // LonLat │ │ │ │ │ + /** │ │ │ │ │ + * Method: down │ │ │ │ │ + * Handle mousedown and touchstart. Add a new point to the geometry and │ │ │ │ │ + * render it. Return determines whether to propagate the event on the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ + */ │ │ │ │ │ + down: function(evt) { │ │ │ │ │ + var stopDown = this.stopDown; │ │ │ │ │ + if (this.freehandMode(evt)) { │ │ │ │ │ + stopDown = true; │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!this.touch && (!this.lastDown || │ │ │ │ │ + !this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ + this.pixelTolerance))) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ + } │ │ │ │ │ + this.mouseDown = true; │ │ │ │ │ + this.lastDown = evt.xy; │ │ │ │ │ + this.stoppedDown = stopDown; │ │ │ │ │ + return !stopDown; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectLonLatFromLonLat │ │ │ │ │ + * Method: move │ │ │ │ │ + * Handle mousemove and touchmove. Adjust the geometry and redraw. │ │ │ │ │ + * Return determines whether to propagate the event on the map. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * lon - {Float} │ │ │ │ │ - * lat - {Float} │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ + */ │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature(); │ │ │ │ │ + } │ │ │ │ │ + if (this.maxVertices && this.line && │ │ │ │ │ + this.line.geometry.components.length === this.maxVertices) { │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize(); │ │ │ │ │ + } else { │ │ │ │ │ + this.addPoint(evt.xy); │ │ │ │ │ + } │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: up │ │ │ │ │ + * Handle mouseup and touchend. Send the latest point in the geometry to │ │ │ │ │ + * the control. Return determines whether to propagate the event on the map. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject LonLat built from lon and lat params │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ - var gLatLng; │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ - gLatLng = new google.maps.LatLng(lonlat.lat, lonlat.lon); │ │ │ │ │ - } else { │ │ │ │ │ - gLatLng = new google.maps.LatLng(lat, lon); │ │ │ │ │ + up: function(evt) { │ │ │ │ │ + if (this.mouseDown && (!this.lastUp || !this.lastUp.equals(evt.xy))) { │ │ │ │ │ + if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature(); │ │ │ │ │ + } │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize(); │ │ │ │ │ + } else { │ │ │ │ │ + if (this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ + this.pixelTolerance)) { │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy); │ │ │ │ │ + } │ │ │ │ │ + if (this.lastUp == null && this.persist) { │ │ │ │ │ + this.destroyPersistedFeature(); │ │ │ │ │ + } │ │ │ │ │ + this.addPoint(evt.xy); │ │ │ │ │ + this.lastUp = evt.xy; │ │ │ │ │ + if (this.line.geometry.components.length === this.maxVertices + 1) { │ │ │ │ │ + this.finishGeometry(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return gLatLng; │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + return !this.stopUp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - // Pixel │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: finishGeometry │ │ │ │ │ + * Finish the geometry and send it back to the control. │ │ │ │ │ + */ │ │ │ │ │ + finishGeometry: function() { │ │ │ │ │ + var index = this.line.geometry.components.length - 1; │ │ │ │ │ + this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getMapObjectPixelFromXY │ │ │ │ │ + * Method: dblclick │ │ │ │ │ + * Handle double-clicks. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * x - {Integer} │ │ │ │ │ - * y - {Integer} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} MapObject Pixel from x and y parameters │ │ │ │ │ + * evt - {Event} The browser event │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow event propagation │ │ │ │ │ */ │ │ │ │ │ - getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ - return new google.maps.Point(x, y); │ │ │ │ │ - } │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + if (!this.freehandMode(evt)) { │ │ │ │ │ + this.finishGeometry(); │ │ │ │ │ + } │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -}; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Path" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Layer/Vector/RootContainer.js │ │ │ │ │ + OpenLayers/Handler/Polygon.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ + * @requires OpenLayers/Handler/Path.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Layer.Vector.RootContainer │ │ │ │ │ - * A special layer type to combine multiple vector layers inside a single │ │ │ │ │ - * renderer root container. This class is not supposed to be instantiated │ │ │ │ │ - * from user space, it is a helper class for controls that require event │ │ │ │ │ - * processing for multiple vector layers. │ │ │ │ │ + * Class: OpenLayers.Handler.Polygon │ │ │ │ │ + * Handler to draw a polygon on the map. Polygon is displayed on mouse down, │ │ │ │ │ + * moves on mouse move, and is finished on mouse up. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Layer.Vector> │ │ │ │ │ + * - <OpenLayers.Handler.Path> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Layer.Vector.RootContainer = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ +OpenLayers.Handler.Polygon = OpenLayers.Class(OpenLayers.Handler.Path, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: holeModifier │ │ │ │ │ + * {String} Key modifier to trigger hole digitizing. Acceptable values are │ │ │ │ │ + * "altKey", "shiftKey", or "ctrlKey". If not set, no hole digitizing │ │ │ │ │ + * will take place. Default is null. │ │ │ │ │ + */ │ │ │ │ │ + holeModifier: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: displayInLayerSwitcher │ │ │ │ │ - * Set to false for this layer type │ │ │ │ │ + * Property: drawingHole │ │ │ │ │ + * {Boolean} Currently drawing an interior ring. │ │ │ │ │ */ │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ + drawingHole: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: layers │ │ │ │ │ - * Layers that are attached to this container. Required config option. │ │ │ │ │ + * Property: polygon │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - layers: null, │ │ │ │ │ + polygon: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Layer.Vector.RootContainer │ │ │ │ │ - * Create a new root container for multiple vector layer. This constructor │ │ │ │ │ - * is not supposed to be used from user space, it is only to be used by │ │ │ │ │ - * controls that need feature selection across multiple vector layers. │ │ │ │ │ + * Constructor: OpenLayers.Handler.Polygon │ │ │ │ │ + * Create a Polygon Handler. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} A name for the layer │ │ │ │ │ - * options - {Object} Optional object with non-default properties to set on │ │ │ │ │ - * the layer. │ │ │ │ │ - * │ │ │ │ │ - * Required options properties: │ │ │ │ │ - * layers - {Array(<OpenLayers.Layer.Vector>)} The layers managed by this │ │ │ │ │ - * container │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ + * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ + * functions. Various callbacks described below. │ │ │ │ │ + * options - {Object} An optional object with properties to be set on the │ │ │ │ │ + * handler │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.Vector.RootContainer>} A new vector layer root │ │ │ │ │ - * container │ │ │ │ │ + * Named callbacks: │ │ │ │ │ + * create - Called when a sketch is first created. Callback called with │ │ │ │ │ + * the creation point geometry and sketch feature. │ │ │ │ │ + * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ + * geometry and the sketch feature. │ │ │ │ │ + * point - Called as each point is added. Receives the new point geometry. │ │ │ │ │ + * done - Called when the point drawing is finished. The callback will │ │ │ │ │ + * recieve a single argument, the polygon geometry. │ │ │ │ │ + * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ + * cancel callback will receive a geometry. │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: display │ │ │ │ │ + * Method: createFeature │ │ │ │ │ + * Add temporary geometries │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} The initial pixel location for the new │ │ │ │ │ + * feature. │ │ │ │ │ */ │ │ │ │ │ - display: function() {}, │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ + lonlat.lon, lonlat.lat │ │ │ │ │ + ); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.line = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.LinearRing([this.point.geometry]) │ │ │ │ │ + ); │ │ │ │ │ + this.polygon = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.Polygon([this.line.geometry]) │ │ │ │ │ + ); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.polygon, this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureFromEvent │ │ │ │ │ - * walk through the layers to find the feature returned by the event │ │ │ │ │ - * │ │ │ │ │ + * Method: addPoint │ │ │ │ │ + * Add point to geometry. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} event object with a feature property │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} The pixel location for the new point. │ │ │ │ │ */ │ │ │ │ │ - getFeatureFromEvent: function(evt) { │ │ │ │ │ - var layers = this.layers; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0; i < layers.length; i++) { │ │ │ │ │ - feature = layers[i].getFeatureFromEvent(evt); │ │ │ │ │ - if (feature) { │ │ │ │ │ - return feature; │ │ │ │ │ + addPoint: function(pixel) { │ │ │ │ │ + if (!this.drawingHole && this.holeModifier && │ │ │ │ │ + this.evt && this.evt[this.holeModifier]) { │ │ │ │ │ + var geometry = this.point.geometry; │ │ │ │ │ + var features = this.control.layer.features; │ │ │ │ │ + var candidate, polygon; │ │ │ │ │ + // look for intersections, last drawn gets priority │ │ │ │ │ + for (var i = features.length - 1; i >= 0; --i) { │ │ │ │ │ + candidate = features[i].geometry; │ │ │ │ │ + if ((candidate instanceof OpenLayers.Geometry.Polygon || │ │ │ │ │ + candidate instanceof OpenLayers.Geometry.MultiPolygon) && │ │ │ │ │ + candidate.intersects(geometry)) { │ │ │ │ │ + polygon = features[i]; │ │ │ │ │ + this.control.layer.removeFeatures([polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.control.layer.events.registerPriority( │ │ │ │ │ + "sketchcomplete", this, this.finalizeInteriorRing │ │ │ │ │ + ); │ │ │ │ │ + this.control.layer.events.registerPriority( │ │ │ │ │ + "sketchmodified", this, this.enforceTopology │ │ │ │ │ + ); │ │ │ │ │ + polygon.geometry.addComponent(this.line.geometry); │ │ │ │ │ + this.polygon = polygon; │ │ │ │ │ + this.drawingHole = true; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Handler.Path.prototype.addPoint.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ + * Method: getCurrentPointIndex │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number} The index of the most recently drawn point. │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.collectRoots(); │ │ │ │ │ - map.events.register("changelayer", this, this.handleChangeLayer); │ │ │ │ │ + getCurrentPointIndex: function() { │ │ │ │ │ + return this.line.geometry.components.length - 2; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removeMap │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * Method: enforceTopology │ │ │ │ │ + * Simple topology enforcement for drawing interior rings. Ensures vertices │ │ │ │ │ + * of interior rings are contained by exterior ring. Other topology │ │ │ │ │ + * rules are enforced in <finalizeInteriorRing> to allow drawing of │ │ │ │ │ + * rings that intersect only during the sketch (e.g. a "C" shaped ring │ │ │ │ │ + * that nearly encloses another ring). │ │ │ │ │ */ │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - map.events.unregister("changelayer", this, this.handleChangeLayer); │ │ │ │ │ - this.resetRoots(); │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments); │ │ │ │ │ + enforceTopology: function(event) { │ │ │ │ │ + var point = event.vertex; │ │ │ │ │ + var components = this.line.geometry.components; │ │ │ │ │ + // ensure that vertices of interior ring are contained by exterior ring │ │ │ │ │ + if (!this.polygon.geometry.intersects(point)) { │ │ │ │ │ + var last = components[components.length - 3]; │ │ │ │ │ + point.x = last.x; │ │ │ │ │ + point.y = last.y; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: collectRoots │ │ │ │ │ - * Collects the root nodes of all layers this control is configured with │ │ │ │ │ - * and moveswien the nodes to this control's layer │ │ │ │ │ + * Method: finishGeometry │ │ │ │ │ + * Finish the geometry and send it back to the control. │ │ │ │ │ */ │ │ │ │ │ - collectRoots: function() { │ │ │ │ │ - var layer; │ │ │ │ │ - // walk through all map layers, because we want to keep the order │ │ │ │ │ - for (var i = 0; i < this.map.layers.length; ++i) { │ │ │ │ │ - layer = this.map.layers[i]; │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ - layer.renderer.moveRoot(this.renderer); │ │ │ │ │ + finishGeometry: function() { │ │ │ │ │ + var index = this.line.geometry.components.length - 2; │ │ │ │ │ + this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: finalizeInteriorRing │ │ │ │ │ + * Enforces that new ring has some area and doesn't contain vertices of any │ │ │ │ │ + * other rings. │ │ │ │ │ + */ │ │ │ │ │ + finalizeInteriorRing: function() { │ │ │ │ │ + var ring = this.line.geometry; │ │ │ │ │ + // ensure that ring has some area │ │ │ │ │ + var modified = (ring.getArea() !== 0); │ │ │ │ │ + if (modified) { │ │ │ │ │ + // ensure that new ring doesn't intersect any other rings │ │ │ │ │ + var rings = this.polygon.geometry.components; │ │ │ │ │ + for (var i = rings.length - 2; i >= 0; --i) { │ │ │ │ │ + if (ring.intersects(rings[i])) { │ │ │ │ │ + modified = false; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (modified) { │ │ │ │ │ + // ensure that new ring doesn't contain any other rings │ │ │ │ │ + var target; │ │ │ │ │ + outer: for (var i = rings.length - 2; i > 0; --i) { │ │ │ │ │ + var points = rings[i].components; │ │ │ │ │ + for (var j = 0, jj = points.length; j < jj; ++j) { │ │ │ │ │ + if (ring.containsPoint(points[j])) { │ │ │ │ │ + modified = false; │ │ │ │ │ + break outer; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (modified) { │ │ │ │ │ + if (this.polygon.state !== OpenLayers.State.INSERT) { │ │ │ │ │ + this.polygon.state = OpenLayers.State.UPDATE; │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + this.polygon.geometry.removeComponent(ring); │ │ │ │ │ } │ │ │ │ │ + this.restoreFeature(); │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: resetRoots │ │ │ │ │ - * Resets the root nodes back into the layers they belong to. │ │ │ │ │ + * APIMethod: cancel │ │ │ │ │ + * Finish the geometry and call the "cancel" callback. │ │ │ │ │ */ │ │ │ │ │ - resetRoots: function() { │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = 0; i < this.layers.length; ++i) { │ │ │ │ │ - layer = this.layers[i]; │ │ │ │ │ - if (this.renderer && layer.renderer.getRenderLayerId() == this.id) { │ │ │ │ │ - this.renderer.moveRoot(layer.renderer); │ │ │ │ │ - } │ │ │ │ │ + cancel: function() { │ │ │ │ │ + if (this.drawingHole) { │ │ │ │ │ + this.polygon.geometry.removeComponent(this.line.geometry); │ │ │ │ │ + this.restoreFeature(true); │ │ │ │ │ } │ │ │ │ │ + return OpenLayers.Handler.Path.prototype.cancel.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleChangeLayer │ │ │ │ │ - * Event handler for the map's changelayer event. We need to rebuild │ │ │ │ │ - * this container's layer dom if order of one of its layers changes. │ │ │ │ │ - * This handler is added with the setMap method, and removed with the │ │ │ │ │ - * removeMap method. │ │ │ │ │ - * │ │ │ │ │ + * Method: restoreFeature │ │ │ │ │ + * Move the feature from the sketch layer to the target layer. │ │ │ │ │ + * │ │ │ │ │ + * Properties: │ │ │ │ │ + * cancel - {Boolean} Cancel drawing. If falsey, the "sketchcomplete" event │ │ │ │ │ + * will be fired. │ │ │ │ │ + */ │ │ │ │ │ + restoreFeature: function(cancel) { │ │ │ │ │ + this.control.layer.events.unregister( │ │ │ │ │ + "sketchcomplete", this, this.finalizeInteriorRing │ │ │ │ │ + ); │ │ │ │ │ + this.control.layer.events.unregister( │ │ │ │ │ + "sketchmodified", this, this.enforceTopology │ │ │ │ │ + ); │ │ │ │ │ + this.layer.removeFeatures([this.polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.control.layer.addFeatures([this.polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.drawingHole = false; │ │ │ │ │ + if (!cancel) { │ │ │ │ │ + // Re-trigger "sketchcomplete" so other listeners can do their │ │ │ │ │ + // business. While this is somewhat sloppy (if a listener is │ │ │ │ │ + // registered with registerPriority - not common - between the start │ │ │ │ │ + // and end of a single ring drawing - very uncommon - it will be │ │ │ │ │ + // called twice). │ │ │ │ │ + // TODO: In 3.0, collapse sketch handlers into geometry specific │ │ │ │ │ + // drawing controls. │ │ │ │ │ + this.control.layer.events.triggerEvent( │ │ │ │ │ + "sketchcomplete", { │ │ │ │ │ + feature: this.polygon │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroyFeature │ │ │ │ │ + * Destroy temporary geometries │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ + * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ */ │ │ │ │ │ - handleChangeLayer: function(evt) { │ │ │ │ │ - var layer = evt.layer; │ │ │ │ │ - if (evt.property == "order" && │ │ │ │ │ - OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ - this.resetRoots(); │ │ │ │ │ - this.collectRoots(); │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + OpenLayers.Handler.Path.prototype.destroyFeature.call( │ │ │ │ │ + this, force); │ │ │ │ │ + this.polygon = null; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawFeature │ │ │ │ │ + * Render geometries on the temporary layer. │ │ │ │ │ + */ │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.polygon, this.style); │ │ │ │ │ + this.layer.drawFeature(this.point, this.style); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getSketch │ │ │ │ │ + * Return the sketch feature. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + getSketch: function() { │ │ │ │ │ + return this.polygon; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getGeometry │ │ │ │ │ + * Return the sketch geometry. If <multi> is true, this will return │ │ │ │ │ + * a multi-part geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Polygon>} │ │ │ │ │ + */ │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.polygon && this.polygon.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiPolygon([geometry]); │ │ │ │ │ } │ │ │ │ │ + return geometry; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Vector.RootContainer" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Polygon" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Popup/Framed.js │ │ │ │ │ + OpenLayers/Handler/MouseWheel.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Popup/Anchored.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Popup.Framed │ │ │ │ │ + * Class: OpenLayers.Handler.MouseWheel │ │ │ │ │ + * Handler for wheel up/down events. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Popup.Anchored> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Popup.Framed = │ │ │ │ │ - OpenLayers.Class(OpenLayers.Popup.Anchored, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: imageSrc │ │ │ │ │ - * {String} location of the image to be used as the popup frame │ │ │ │ │ - */ │ │ │ │ │ - imageSrc: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: imageSize │ │ │ │ │ - * {<OpenLayers.Size>} Size (measured in pixels) of the image located │ │ │ │ │ - * by the 'imageSrc' property. │ │ │ │ │ - */ │ │ │ │ │ - imageSize: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isAlphaImage │ │ │ │ │ - * {Boolean} The image has some alpha and thus needs to use the alpha │ │ │ │ │ - * image hack. Note that setting this to true will have no noticeable │ │ │ │ │ - * effect in FF or IE7 browsers, but will all but crush the ie6 │ │ │ │ │ - * browser. │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - isAlphaImage: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: positionBlocks │ │ │ │ │ - * {Object} Hash of different position blocks (Object/Hashs). Each block │ │ │ │ │ - * will be keyed by a two-character 'relativePosition' │ │ │ │ │ - * code string (ie "tl", "tr", "bl", "br"). Block properties are │ │ │ │ │ - * 'offset', 'padding' (self-explanatory), and finally the 'blocks' │ │ │ │ │ - * parameter, which is an array of the block objects. │ │ │ │ │ - * │ │ │ │ │ - * Each block object must have 'size', 'anchor', and 'position' │ │ │ │ │ - * properties. │ │ │ │ │ - * │ │ │ │ │ - * Note that positionBlocks should never be modified at runtime. │ │ │ │ │ - */ │ │ │ │ │ - positionBlocks: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: blocks │ │ │ │ │ - * {Array[Object]} Array of objects, each of which is one "block" of the │ │ │ │ │ - * popup. Each block has a 'div' and an 'image' property, both of │ │ │ │ │ - * which are DOMElements, and the latter of which is appended to the │ │ │ │ │ - * former. These are reused as the popup goes changing positions for │ │ │ │ │ - * great economy and elegance. │ │ │ │ │ - */ │ │ │ │ │ - blocks: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: fixedRelativePosition │ │ │ │ │ - * {Boolean} We want the framed popup to work dynamically placed relative │ │ │ │ │ - * to its anchor but also in just one fixed position. A well designed │ │ │ │ │ - * framed popup will have the pixels and logic to display itself in │ │ │ │ │ - * any of the four relative positions, but (understandably), this will │ │ │ │ │ - * not be the case for all of them. By setting this property to 'true', │ │ │ │ │ - * framed popup will not recalculate for the best placement each time │ │ │ │ │ - * it's open, but will always open the same way. │ │ │ │ │ - * Note that if this is set to true, it is generally advisable to also │ │ │ │ │ - * set the 'panIntoView' property to true so that the popup can be │ │ │ │ │ - * scrolled into view (since it will often be offscreen on open) │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - fixedRelativePosition: false, │ │ │ │ │ +OpenLayers.Handler.MouseWheel = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + /** │ │ │ │ │ + * Property: wheelListener │ │ │ │ │ + * {function} │ │ │ │ │ + */ │ │ │ │ │ + wheelListener: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Popup.Framed │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ - * contentSize - {<OpenLayers.Size>} │ │ │ │ │ - * contentHTML - {String} │ │ │ │ │ - * anchor - {Object} Object to which we'll anchor the popup. Must expose │ │ │ │ │ - * a 'size' (<OpenLayers.Size>) and 'offset' (<OpenLayers.Pixel>) │ │ │ │ │ - * (Note that this is generally an <OpenLayers.Icon>). │ │ │ │ │ - * closeBox - {Boolean} │ │ │ │ │ - * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, │ │ │ │ │ - closeBoxCallback) { │ │ │ │ │ + /** │ │ │ │ │ + * Property: interval │ │ │ │ │ + * {Integer} In order to increase server performance, an interval (in │ │ │ │ │ + * milliseconds) can be set to reduce the number of up/down events │ │ │ │ │ + * called. If set, a new up/down event will not be set until the │ │ │ │ │ + * interval has passed. │ │ │ │ │ + * Defaults to 0, meaning no interval. │ │ │ │ │ + */ │ │ │ │ │ + interval: 0, │ │ │ │ │ │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.initialize.apply(this, arguments); │ │ │ │ │ + /** │ │ │ │ │ + * Property: maxDelta │ │ │ │ │ + * {Integer} Maximum delta to collect before breaking from the current │ │ │ │ │ + * interval. In cumulative mode, this also limits the maximum delta │ │ │ │ │ + * returned from the handler. Default is Number.POSITIVE_INFINITY. │ │ │ │ │ + */ │ │ │ │ │ + maxDelta: Number.POSITIVE_INFINITY, │ │ │ │ │ │ │ │ │ │ - if (this.fixedRelativePosition) { │ │ │ │ │ - //based on our decided relativePostion, set the current padding │ │ │ │ │ - // this keeps us from getting into trouble │ │ │ │ │ - this.updateRelativePosition(); │ │ │ │ │ + /** │ │ │ │ │ + * Property: delta │ │ │ │ │ + * {Integer} When interval is set, delta collects the mousewheel z-deltas │ │ │ │ │ + * of the events that occur within the interval. │ │ │ │ │ + * See also the cumulative option │ │ │ │ │ + */ │ │ │ │ │ + delta: 0, │ │ │ │ │ │ │ │ │ │ - //make calculateRelativePosition always return the specified │ │ │ │ │ - // fixed position. │ │ │ │ │ - this.calculateRelativePosition = function(px) { │ │ │ │ │ - return this.relativePosition; │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: cumulative │ │ │ │ │ + * {Boolean} When interval is set: true to collect all the mousewheel │ │ │ │ │ + * z-deltas, false to only record the delta direction (positive or │ │ │ │ │ + * negative) │ │ │ │ │ + */ │ │ │ │ │ + cumulative: true, │ │ │ │ │ │ │ │ │ │ - this.contentDiv.style.position = "absolute"; │ │ │ │ │ - this.contentDiv.style.zIndex = 1; │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Handler.MouseWheel │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * control - {<OpenLayers.Control>} │ │ │ │ │ + * callbacks - {Object} An object containing a single function to be │ │ │ │ │ + * called when the drag operation is finished. │ │ │ │ │ + * The callback should expect to recieve a single │ │ │ │ │ + * argument, the point geometry. │ │ │ │ │ + * options - {Object} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.wheelListener = OpenLayers.Function.bindAsEventListener( │ │ │ │ │ + this.onWheelEvent, this │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (closeBox) { │ │ │ │ │ - this.closeDiv.style.zIndex = 1; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ + this.wheelListener = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.groupDiv.style.position = "absolute"; │ │ │ │ │ - this.groupDiv.style.top = "0px"; │ │ │ │ │ - this.groupDiv.style.left = "0px"; │ │ │ │ │ - this.groupDiv.style.height = "100%"; │ │ │ │ │ - this.groupDiv.style.width = "100%"; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Mouse ScrollWheel code thanks to http://adomas.org/javascript-mouse-wheel/ │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.imageSrc = null; │ │ │ │ │ - this.imageSize = null; │ │ │ │ │ - this.isAlphaImage = null; │ │ │ │ │ + /** │ │ │ │ │ + * Method: onWheelEvent │ │ │ │ │ + * Catch the wheel event and handle it xbrowserly │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * e - {Event} │ │ │ │ │ + */ │ │ │ │ │ + onWheelEvent: function(e) { │ │ │ │ │ │ │ │ │ │ - this.fixedRelativePosition = false; │ │ │ │ │ - this.positionBlocks = null; │ │ │ │ │ + // make sure we have a map and check keyboard modifiers │ │ │ │ │ + if (!this.map || !this.checkModifiers(e)) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - //remove our blocks │ │ │ │ │ - for (var i = 0; i < this.blocks.length; i++) { │ │ │ │ │ - var block = this.blocks[i]; │ │ │ │ │ + // Ride up the element's DOM hierarchy to determine if it or any of │ │ │ │ │ + // its ancestors was: │ │ │ │ │ + // * specifically marked as scrollable (CSS overflow property) │ │ │ │ │ + // * one of our layer divs or a div marked as scrollable │ │ │ │ │ + // ('olScrollable' CSS class) │ │ │ │ │ + // * the map div │ │ │ │ │ + // │ │ │ │ │ + var overScrollableDiv = false; │ │ │ │ │ + var allowScroll = false; │ │ │ │ │ + var overMapDiv = false; │ │ │ │ │ │ │ │ │ │ - if (block.image) { │ │ │ │ │ - block.div.removeChild(block.image); │ │ │ │ │ - } │ │ │ │ │ - block.image = null; │ │ │ │ │ + var elem = OpenLayers.Event.element(e); │ │ │ │ │ + while ((elem != null) && !overMapDiv && !overScrollableDiv) { │ │ │ │ │ │ │ │ │ │ - if (block.div) { │ │ │ │ │ - this.groupDiv.removeChild(block.div); │ │ │ │ │ + if (!overScrollableDiv) { │ │ │ │ │ + try { │ │ │ │ │ + var overflow; │ │ │ │ │ + if (elem.currentStyle) { │ │ │ │ │ + overflow = elem.currentStyle["overflow"]; │ │ │ │ │ + } else { │ │ │ │ │ + var style = │ │ │ │ │ + document.defaultView.getComputedStyle(elem, null); │ │ │ │ │ + overflow = style.getPropertyValue("overflow"); │ │ │ │ │ + } │ │ │ │ │ + overScrollableDiv = (overflow && │ │ │ │ │ + (overflow == "auto") || (overflow == "scroll")); │ │ │ │ │ + } catch (err) { │ │ │ │ │ + //sometimes when scrolling in a popup, this causes │ │ │ │ │ + // obscure browser error │ │ │ │ │ } │ │ │ │ │ - block.div = null; │ │ │ │ │ } │ │ │ │ │ - this.blocks = null; │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setBackgroundColor │ │ │ │ │ - */ │ │ │ │ │ - setBackgroundColor: function(color) { │ │ │ │ │ - //does nothing since the framed popup's entire scheme is based on a │ │ │ │ │ - // an image -- changing the background color makes no sense. │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setBorder │ │ │ │ │ - */ │ │ │ │ │ - setBorder: function() { │ │ │ │ │ - //does nothing since the framed popup's entire scheme is based on a │ │ │ │ │ - // an image -- changing the popup's border makes no sense. │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setOpacity │ │ │ │ │ - * Sets the opacity of the popup. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * opacity - {float} A value between 0.0 (transparent) and 1.0 (solid). │ │ │ │ │ - */ │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - //does nothing since we suppose that we'll never apply an opacity │ │ │ │ │ - // to a framed popup │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setSize │ │ │ │ │ - * Overridden here, because we need to update the blocks whenever the size │ │ │ │ │ - * of the popup has changed. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * contentSize - {<OpenLayers.Size>} the new size for the popup's │ │ │ │ │ - * contents div (in pixels). │ │ │ │ │ - */ │ │ │ │ │ - setSize: function(contentSize) { │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.setSize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - this.updateBlocks(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateRelativePosition │ │ │ │ │ - * When the relative position changes, we need to set the new padding │ │ │ │ │ - * BBOX on the popup, reposition the close div, and update the blocks. │ │ │ │ │ - */ │ │ │ │ │ - updateRelativePosition: function() { │ │ │ │ │ │ │ │ │ │ - //update the padding │ │ │ │ │ - this.padding = this.positionBlocks[this.relativePosition].padding; │ │ │ │ │ - │ │ │ │ │ - //update the position of our close box to new padding │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - // use the content div's css padding to determine if we should │ │ │ │ │ - // padd the close div │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ - │ │ │ │ │ - this.closeDiv.style.right = contentDivPadding.right + │ │ │ │ │ - this.padding.right + "px"; │ │ │ │ │ - this.closeDiv.style.top = contentDivPadding.top + │ │ │ │ │ - this.padding.top + "px"; │ │ │ │ │ + if (!allowScroll) { │ │ │ │ │ + allowScroll = OpenLayers.Element.hasClass(elem, 'olScrollable'); │ │ │ │ │ + if (!allowScroll) { │ │ │ │ │ + for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ + // Are we in the layer div? Note that we have two cases │ │ │ │ │ + // here: one is to catch EventPane layers, which have a │ │ │ │ │ + // pane above the layer (layer.pane) │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if (elem == layer.div || elem == layer.pane) { │ │ │ │ │ + allowScroll = true; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + overMapDiv = (elem == this.map.div); │ │ │ │ │ │ │ │ │ │ - this.updateBlocks(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: calculateNewPx │ │ │ │ │ - * Besides the standard offset as determined by the Anchored class, our │ │ │ │ │ - * Framed popups have a special 'offset' property for each of their │ │ │ │ │ - * positions, which is used to offset the popup relative to its anchor. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Pixel>} The the new px position of the popup on the screen │ │ │ │ │ - * relative to the passed-in px. │ │ │ │ │ - */ │ │ │ │ │ - calculateNewPx: function(px) { │ │ │ │ │ - var newPx = OpenLayers.Popup.Anchored.prototype.calculateNewPx.apply( │ │ │ │ │ - this, arguments │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - newPx = newPx.offset(this.positionBlocks[this.relativePosition].offset); │ │ │ │ │ + elem = elem.parentNode; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - return newPx; │ │ │ │ │ - }, │ │ │ │ │ + // Logic below is the following: │ │ │ │ │ + // │ │ │ │ │ + // If we are over a scrollable div or not over the map div: │ │ │ │ │ + // * do nothing (let the browser handle scrolling) │ │ │ │ │ + // │ │ │ │ │ + // otherwise │ │ │ │ │ + // │ │ │ │ │ + // If we are over the layer div or a 'olScrollable' div: │ │ │ │ │ + // * zoom/in out │ │ │ │ │ + // then │ │ │ │ │ + // * kill event (so as not to also scroll the page after zooming) │ │ │ │ │ + // │ │ │ │ │ + // otherwise │ │ │ │ │ + // │ │ │ │ │ + // Kill the event (dont scroll the page if we wheel over the │ │ │ │ │ + // layerswitcher or the pan/zoom control) │ │ │ │ │ + // │ │ │ │ │ + if (!overScrollableDiv && overMapDiv) { │ │ │ │ │ + if (allowScroll) { │ │ │ │ │ + var delta = 0; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createBlocks │ │ │ │ │ - */ │ │ │ │ │ - createBlocks: function() { │ │ │ │ │ - this.blocks = []; │ │ │ │ │ + if (e.wheelDelta) { │ │ │ │ │ + delta = e.wheelDelta; │ │ │ │ │ + if (delta % 160 === 0) { │ │ │ │ │ + // opera have steps of 160 instead of 120 │ │ │ │ │ + delta = delta * 0.75; │ │ │ │ │ + } │ │ │ │ │ + delta = delta / 120; │ │ │ │ │ + } else if (e.detail) { │ │ │ │ │ + // detail in Firefox on OS X is 1/3 of Windows │ │ │ │ │ + // so force delta 1 / -1 │ │ │ │ │ + delta = -(e.detail / Math.abs(e.detail)); │ │ │ │ │ + } │ │ │ │ │ + this.delta += delta; │ │ │ │ │ │ │ │ │ │ - //since all positions contain the same number of blocks, we can │ │ │ │ │ - // just pick the first position and use its blocks array to create │ │ │ │ │ - // our blocks array │ │ │ │ │ - var firstPosition = null; │ │ │ │ │ - for (var key in this.positionBlocks) { │ │ │ │ │ - firstPosition = key; │ │ │ │ │ - break; │ │ │ │ │ + window.clearTimeout(this._timeoutId); │ │ │ │ │ + if (this.interval && Math.abs(this.delta) < this.maxDelta) { │ │ │ │ │ + // store e because window.event might change during delay │ │ │ │ │ + var evt = OpenLayers.Util.extend({}, e); │ │ │ │ │ + this._timeoutId = window.setTimeout( │ │ │ │ │ + OpenLayers.Function.bind(function() { │ │ │ │ │ + this.wheelZoom(evt); │ │ │ │ │ + }, this), │ │ │ │ │ + this.interval │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + this.wheelZoom(e); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Event.stop(e); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var position = this.positionBlocks[firstPosition]; │ │ │ │ │ - for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ - │ │ │ │ │ - var block = {}; │ │ │ │ │ - this.blocks.push(block); │ │ │ │ │ - │ │ │ │ │ - var divId = this.id + '_FrameDecorationDiv_' + i; │ │ │ │ │ - block.div = OpenLayers.Util.createDiv(divId, │ │ │ │ │ - null, null, null, "absolute", null, "hidden", null │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - var imgId = this.id + '_FrameDecorationImg_' + i; │ │ │ │ │ - var imageCreator = │ │ │ │ │ - (this.isAlphaImage) ? OpenLayers.Util.createAlphaImageDiv : │ │ │ │ │ - OpenLayers.Util.createImage; │ │ │ │ │ - │ │ │ │ │ - block.image = imageCreator(imgId, │ │ │ │ │ - null, this.imageSize, this.imageSrc, │ │ │ │ │ - "absolute", null, null, null │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - block.div.appendChild(block.image); │ │ │ │ │ - this.groupDiv.appendChild(block.div); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: wheelZoom │ │ │ │ │ + * Given the wheel event, we carry out the appropriate zooming in or out, │ │ │ │ │ + * based on the 'wheelDelta' or 'detail' property of the event. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * e - {Event} │ │ │ │ │ + */ │ │ │ │ │ + wheelZoom: function(e) { │ │ │ │ │ + var delta = this.delta; │ │ │ │ │ + this.delta = 0; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateBlocks │ │ │ │ │ - * Internal method, called on initialize and when the popup's relative │ │ │ │ │ - * position has changed. This function takes care of re-positioning │ │ │ │ │ - * the popup's blocks in their appropropriate places. │ │ │ │ │ - */ │ │ │ │ │ - updateBlocks: function() { │ │ │ │ │ - if (!this.blocks) { │ │ │ │ │ - this.createBlocks(); │ │ │ │ │ + if (delta) { │ │ │ │ │ + e.xy = this.map.events.getMousePosition(e); │ │ │ │ │ + if (delta < 0) { │ │ │ │ │ + this.callback("down", │ │ │ │ │ + [e, this.cumulative ? Math.max(-this.maxDelta, delta) : -1]); │ │ │ │ │ + } else { │ │ │ │ │ + this.callback("up", │ │ │ │ │ + [e, this.cumulative ? Math.min(this.maxDelta, delta) : 1]); │ │ │ │ │ } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (this.size && this.relativePosition) { │ │ │ │ │ - var position = this.positionBlocks[this.relativePosition]; │ │ │ │ │ - for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ - │ │ │ │ │ - var positionBlock = position.blocks[i]; │ │ │ │ │ - var block = this.blocks[i]; │ │ │ │ │ - │ │ │ │ │ - // adjust sizes │ │ │ │ │ - var l = positionBlock.anchor.left; │ │ │ │ │ - var b = positionBlock.anchor.bottom; │ │ │ │ │ - var r = positionBlock.anchor.right; │ │ │ │ │ - var t = positionBlock.anchor.top; │ │ │ │ │ - │ │ │ │ │ - //note that we use the isNaN() test here because if the │ │ │ │ │ - // size object is initialized with a "auto" parameter, the │ │ │ │ │ - // size constructor calls parseFloat() on the string, │ │ │ │ │ - // which will turn it into NaN │ │ │ │ │ - // │ │ │ │ │ - var w = (isNaN(positionBlock.size.w)) ? this.size.w - (r + l) : │ │ │ │ │ - positionBlock.size.w; │ │ │ │ │ - │ │ │ │ │ - var h = (isNaN(positionBlock.size.h)) ? this.size.h - (b + t) : │ │ │ │ │ - positionBlock.size.h; │ │ │ │ │ - │ │ │ │ │ - block.div.style.width = (w < 0 ? 0 : w) + 'px'; │ │ │ │ │ - block.div.style.height = (h < 0 ? 0 : h) + 'px'; │ │ │ │ │ - │ │ │ │ │ - block.div.style.left = (l != null) ? l + 'px' : ''; │ │ │ │ │ - block.div.style.bottom = (b != null) ? b + 'px' : ''; │ │ │ │ │ - block.div.style.right = (r != null) ? r + 'px' : ''; │ │ │ │ │ - block.div.style.top = (t != null) ? t + 'px' : ''; │ │ │ │ │ - │ │ │ │ │ - block.image.style.left = positionBlock.position.x + 'px'; │ │ │ │ │ - block.image.style.top = positionBlock.position.y + 'px'; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: activate │ │ │ │ │ + */ │ │ │ │ │ + activate: function(evt) { │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + //register mousewheel events specifically on the window and document │ │ │ │ │ + var wheelListener = this.wheelListener; │ │ │ │ │ + OpenLayers.Event.observe(window, "DOMMouseScroll", wheelListener); │ │ │ │ │ + OpenLayers.Event.observe(window, "mousewheel", wheelListener); │ │ │ │ │ + OpenLayers.Event.observe(document, "mousewheel", wheelListener); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.contentDiv.style.left = this.padding.left + "px"; │ │ │ │ │ - this.contentDiv.style.top = this.padding.top + "px"; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function(evt) { │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + // unregister mousewheel events specifically on the window and document │ │ │ │ │ + var wheelListener = this.wheelListener; │ │ │ │ │ + OpenLayers.Event.stopObserving(window, "DOMMouseScroll", wheelListener); │ │ │ │ │ + OpenLayers.Event.stopObserving(window, "mousewheel", wheelListener); │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mousewheel", wheelListener); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup.Framed" │ │ │ │ │ - }); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.MouseWheel" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Popup/FramedCloud.js │ │ │ │ │ + OpenLayers/Handler/Pinch.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Popup/Framed.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Bounds.js │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Pixel.js │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Size.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Popup.FramedCloud │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Popup.Framed> │ │ │ │ │ + * Class: OpenLayers.Handler.Pinch │ │ │ │ │ + * The pinch handler is used to deal with sequences of browser events related │ │ │ │ │ + * to pinch gestures. The handler is used by controls that want to know │ │ │ │ │ + * when a pinch sequence begins, when a pinch is happening, and when it has │ │ │ │ │ + * finished. │ │ │ │ │ + * │ │ │ │ │ + * Controls that use the pinch handler typically construct it with callbacks │ │ │ │ │ + * for 'start', 'move', and 'done'. Callbacks for these keys are │ │ │ │ │ + * called when the pinch begins, with each change, and when the pinch is │ │ │ │ │ + * done. │ │ │ │ │ + * │ │ │ │ │ + * Create a new pinch handler with the <OpenLayers.Handler.Pinch> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Popup.FramedCloud = │ │ │ │ │ - OpenLayers.Class(OpenLayers.Popup.Framed, { │ │ │ │ │ +OpenLayers.Handler.Pinch = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: contentDisplayClass │ │ │ │ │ - * {String} The CSS class of the popup content div. │ │ │ │ │ - */ │ │ │ │ │ - contentDisplayClass: "olFramedCloudPopupContent", │ │ │ │ │ + /** │ │ │ │ │ + * Property: started │ │ │ │ │ + * {Boolean} When a touchstart event is received, we want to record it, │ │ │ │ │ + * but not set 'pinching' until the touchmove get started after │ │ │ │ │ + * starting. │ │ │ │ │ + */ │ │ │ │ │ + started: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoSize │ │ │ │ │ - * {Boolean} Framed Cloud is autosizing by default. │ │ │ │ │ - */ │ │ │ │ │ - autoSize: true, │ │ │ │ │ + /** │ │ │ │ │ + * Property: stopDown │ │ │ │ │ + * {Boolean} Stop propagation of touchstart events from getting to │ │ │ │ │ + * listeners on the same element. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + stopDown: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: panMapIfOutOfView │ │ │ │ │ - * {Boolean} Framed Cloud does pan into view by default. │ │ │ │ │ - */ │ │ │ │ │ - panMapIfOutOfView: true, │ │ │ │ │ + /** │ │ │ │ │ + * Property: pinching │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + pinching: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: imageSize │ │ │ │ │ - * {<OpenLayers.Size>} │ │ │ │ │ - */ │ │ │ │ │ - imageSize: new OpenLayers.Size(1276, 736), │ │ │ │ │ + /** │ │ │ │ │ + * Property: last │ │ │ │ │ + * {Object} Object that store informations related to pinch last touch. │ │ │ │ │ + */ │ │ │ │ │ + last: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: isAlphaImage │ │ │ │ │ - * {Boolean} The FramedCloud does not use an alpha image (in honor of the │ │ │ │ │ - * good ie6 folk out there) │ │ │ │ │ - */ │ │ │ │ │ - isAlphaImage: false, │ │ │ │ │ + /** │ │ │ │ │ + * Property: start │ │ │ │ │ + * {Object} Object that store informations related to pinch touchstart. │ │ │ │ │ + */ │ │ │ │ │ + start: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: fixedRelativePosition │ │ │ │ │ - * {Boolean} The Framed Cloud popup works in just one fixed position. │ │ │ │ │ - */ │ │ │ │ │ - fixedRelativePosition: false, │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Handler.Pinch │ │ │ │ │ + * Returns OpenLayers.Handler.Pinch │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ + * this handler. If a handler is being used without a control, the │ │ │ │ │ + * handlers setMap method must be overridden to deal properly with │ │ │ │ │ + * the map. │ │ │ │ │ + * callbacks - {Object} An object containing functions to be called when │ │ │ │ │ + * the pinch operation start, change, or is finished. The callbacks │ │ │ │ │ + * should expect to receive an object argument, which contains │ │ │ │ │ + * information about scale, distance, and position of touch points. │ │ │ │ │ + * options - {Object} │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: positionBlocks │ │ │ │ │ - * {Object} Hash of differen position blocks, keyed by relativePosition │ │ │ │ │ - * two-character code string (ie "tl", "tr", "bl", "br") │ │ │ │ │ - */ │ │ │ │ │ - positionBlocks: { │ │ │ │ │ - "tl": { │ │ │ │ │ - 'offset': new OpenLayers.Pixel(44, 0), │ │ │ │ │ - 'padding': new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ - 'blocks': [{ // top-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { //top-right │ │ │ │ │ - size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { //bottom-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ - }, { //bottom-right │ │ │ │ │ - size: new OpenLayers.Size(22, 18), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -632) │ │ │ │ │ - }, { // stem │ │ │ │ │ - size: new OpenLayers.Size(81, 35), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -688) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - "tr": { │ │ │ │ │ - 'offset': new OpenLayers.Pixel(-45, 0), │ │ │ │ │ - 'padding': new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ - 'blocks': [{ // top-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { //top-right │ │ │ │ │ - size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { //bottom-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ - }, { //bottom-right │ │ │ │ │ - size: new OpenLayers.Size(22, 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -631) │ │ │ │ │ - }, { // stem │ │ │ │ │ - size: new OpenLayers.Size(81, 35), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, null, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-215, -687) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - "bl": { │ │ │ │ │ - 'offset': new OpenLayers.Pixel(45, 0), │ │ │ │ │ - 'padding': new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ - 'blocks': [{ // top-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { //top-right │ │ │ │ │ - size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { //bottom-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ - }, { //bottom-right │ │ │ │ │ - size: new OpenLayers.Size(22, 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ - }, { // stem │ │ │ │ │ - size: new OpenLayers.Size(81, 33), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, null, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-101, -674) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - "br": { │ │ │ │ │ - 'offset': new OpenLayers.Pixel(-44, 0), │ │ │ │ │ - 'padding': new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ - 'blocks': [{ // top-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { //top-right │ │ │ │ │ - size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { //bottom-left │ │ │ │ │ - size: new OpenLayers.Size('auto', 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ - }, { //bottom-right │ │ │ │ │ - size: new OpenLayers.Size(22, 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ - }, { // stem │ │ │ │ │ - size: new OpenLayers.Size(81, 33), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, null, null, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-311, -674) │ │ │ │ │ - }] │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: touchstart │ │ │ │ │ + * Handle touchstart events │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ + */ │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + if (OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ + this.started = true; │ │ │ │ │ + this.last = this.start = { │ │ │ │ │ + distance: this.getDistance(evt.touches), │ │ │ │ │ + delta: 0, │ │ │ │ │ + scale: 1 │ │ │ │ │ + }; │ │ │ │ │ + this.callback("start", [evt, this.start]); │ │ │ │ │ + propagate = !this.stopDown; │ │ │ │ │ + } else if (this.started) { │ │ │ │ │ + // Some webkit versions send fake single-touch events during │ │ │ │ │ + // multitouch, which cause the drag handler to trigger │ │ │ │ │ + return false; │ │ │ │ │ + } else { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + } │ │ │ │ │ + // prevent document dragging │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + return propagate; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: minSize │ │ │ │ │ - * {<OpenLayers.Size>} │ │ │ │ │ - */ │ │ │ │ │ - minSize: new OpenLayers.Size(105, 10), │ │ │ │ │ + /** │ │ │ │ │ + * Method: touchmove │ │ │ │ │ + * Handle touchmove events │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ + */ │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + if (this.started && OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ + this.pinching = true; │ │ │ │ │ + var current = this.getPinchData(evt); │ │ │ │ │ + this.callback("move", [evt, current]); │ │ │ │ │ + this.last = current; │ │ │ │ │ + // prevent document dragging │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + } else if (this.started) { │ │ │ │ │ + // Some webkit versions send fake single-touch events during │ │ │ │ │ + // multitouch, which cause the drag handler to trigger │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: maxSize │ │ │ │ │ - * {<OpenLayers.Size>} │ │ │ │ │ - */ │ │ │ │ │ - maxSize: new OpenLayers.Size(1200, 660), │ │ │ │ │ + /** │ │ │ │ │ + * Method: touchend │ │ │ │ │ + * Handle touchend events │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ + */ │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + if (this.started && !OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + this.callback("done", [evt, this.start, this.last]); │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Popup.FramedCloud │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ - * contentSize - {<OpenLayers.Size>} │ │ │ │ │ - * contentHTML - {String} │ │ │ │ │ - * anchor - {Object} Object to which we'll anchor the popup. Must expose │ │ │ │ │ - * a 'size' (<OpenLayers.Size>) and 'offset' (<OpenLayers.Pixel>) │ │ │ │ │ - * (Note that this is generally an <OpenLayers.Icon>). │ │ │ │ │ - * closeBox - {Boolean} │ │ │ │ │ - * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, │ │ │ │ │ - closeBoxCallback) { │ │ │ │ │ + /** │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activate the handler. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was successfully activated. │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + activated = true; │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.imageSrc = OpenLayers.Util.getImageLocation('cloud-popup-relative.png'); │ │ │ │ │ - OpenLayers.Popup.Framed.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.contentDiv.className = this.contentDisplayClass; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivate the handler. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The handler was successfully deactivated. │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + deactivated = true; │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getDistance │ │ │ │ │ + * Get the distance in pixels between two touches. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * touches - {Array(Object)} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number} The distance in pixels. │ │ │ │ │ + */ │ │ │ │ │ + getDistance: function(touches) { │ │ │ │ │ + var t0 = touches[0]; │ │ │ │ │ + var t1 = touches[1]; │ │ │ │ │ + return Math.sqrt( │ │ │ │ │ + Math.pow(t0.olClientX - t1.olClientX, 2) + │ │ │ │ │ + Math.pow(t0.olClientY - t1.olClientY, 2) │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getPinchData │ │ │ │ │ + * Get informations about the pinch event. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Object that contains data about the current pinch. │ │ │ │ │ + */ │ │ │ │ │ + getPinchData: function(evt) { │ │ │ │ │ + var distance = this.getDistance(evt.touches); │ │ │ │ │ + var scale = distance / this.start.distance; │ │ │ │ │ + return { │ │ │ │ │ + distance: distance, │ │ │ │ │ + delta: this.last.distance - distance, │ │ │ │ │ + scale: scale │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Pinch" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup.FramedCloud" │ │ │ │ │ - }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WFSCapabilities.js │ │ │ │ │ + OpenLayers/Handler/Box.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Handler/Drag.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WFSCapabilities │ │ │ │ │ - * Read WFS Capabilities. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Handler.Box │ │ │ │ │ + * Handler for dragging a rectangle across the map. Box is displayed │ │ │ │ │ + * on mouse down, moves on mouse move, and is finished on mouse up. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WFSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ +OpenLayers.Handler.Box = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: dragHandler │ │ │ │ │ + * {<OpenLayers.Handler.Drag>} │ │ │ │ │ + */ │ │ │ │ │ + dragHandler: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.1.0". │ │ │ │ │ + * APIProperty: boxDivClassName │ │ │ │ │ + * {String} The CSS class to use for drawing the box. Default is │ │ │ │ │ + * olHandlerBoxZoomBox │ │ │ │ │ */ │ │ │ │ │ - defaultVersion: "1.1.0", │ │ │ │ │ + boxDivClassName: 'olHandlerBoxZoomBox', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WFSCapabilities │ │ │ │ │ - * Create a new parser for WFS capabilities. │ │ │ │ │ + * Property: boxOffsets │ │ │ │ │ + * {Object} Caches box offsets from css. This is used by the getBoxOffsets │ │ │ │ │ + * method. │ │ │ │ │ + */ │ │ │ │ │ + boxOffsets: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Handler.Box │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * control - {<OpenLayers.Control>} │ │ │ │ │ + * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ + * functions. Various callbacks described below. │ │ │ │ │ + * options - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Named callbacks: │ │ │ │ │ + * start - Called when the box drag operation starts. │ │ │ │ │ + * done - Called when the box drag operation is finished. │ │ │ │ │ + * The callback should expect to receive a single argument, the box │ │ │ │ │ + * bounds or a pixel. If the box dragging didn't span more than a 5 │ │ │ │ │ + * pixel distance, a pixel will be returned instead of a bounds object. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.dragHandler = new OpenLayers.Handler.Drag( │ │ │ │ │ + this, { │ │ │ │ │ + down: this.startBox, │ │ │ │ │ + move: this.moveBox, │ │ │ │ │ + out: this.removeBox, │ │ │ │ │ + up: this.endBox │ │ │ │ │ + }, { │ │ │ │ │ + keyMask: this.keyMask │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ + if (this.dragHandler) { │ │ │ │ │ + this.dragHandler.destroy(); │ │ │ │ │ + this.dragHandler = null; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * Method: setMap │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Handler.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (this.dragHandler) { │ │ │ │ │ + this.dragHandler.setMap(map); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: startBox │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} List of named layers. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} │ │ │ │ │ */ │ │ │ │ │ + startBox: function(xy) { │ │ │ │ │ + this.callback("start", []); │ │ │ │ │ + this.zoomBox = OpenLayers.Util.createDiv('zoomBox', { │ │ │ │ │ + x: -9999, │ │ │ │ │ + y: -9999 │ │ │ │ │ + }); │ │ │ │ │ + this.zoomBox.className = this.boxDivClassName; │ │ │ │ │ + this.zoomBox.style.zIndex = this.map.Z_INDEX_BASE["Popup"] - 1; │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSCapabilities" │ │ │ │ │ + this.map.viewPortDiv.appendChild(this.zoomBox); │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WPSCapabilities.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + OpenLayers.Element.addClass( │ │ │ │ │ + this.map.viewPortDiv, "olDrawBox" │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveBox │ │ │ │ │ + */ │ │ │ │ │ + moveBox: function(xy) { │ │ │ │ │ + var startX = this.dragHandler.start.x; │ │ │ │ │ + var startY = this.dragHandler.start.y; │ │ │ │ │ + var deltaX = Math.abs(startX - xy.x); │ │ │ │ │ + var deltaY = Math.abs(startY - xy.y); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ - */ │ │ │ │ │ + var offset = this.getBoxOffsets(); │ │ │ │ │ + this.zoomBox.style.width = (deltaX + offset.width + 1) + "px"; │ │ │ │ │ + this.zoomBox.style.height = (deltaY + offset.height + 1) + "px"; │ │ │ │ │ + this.zoomBox.style.left = (xy.x < startX ? │ │ │ │ │ + startX - deltaX - offset.left : startX - offset.left) + "px"; │ │ │ │ │ + this.zoomBox.style.top = (xy.y < startY ? │ │ │ │ │ + startY - deltaY - offset.top : startY - offset.top) + "px"; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WPSCapabilities │ │ │ │ │ - * Read WPS Capabilities. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WPSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + /** │ │ │ │ │ + * Method: endBox │ │ │ │ │ + */ │ │ │ │ │ + endBox: function(end) { │ │ │ │ │ + var result; │ │ │ │ │ + if (Math.abs(this.dragHandler.start.x - end.x) > 5 || │ │ │ │ │ + Math.abs(this.dragHandler.start.y - end.y) > 5) { │ │ │ │ │ + var start = this.dragHandler.start; │ │ │ │ │ + var top = Math.min(start.y, end.y); │ │ │ │ │ + var bottom = Math.max(start.y, end.y); │ │ │ │ │ + var left = Math.min(start.x, end.x); │ │ │ │ │ + var right = Math.max(start.x, end.x); │ │ │ │ │ + result = new OpenLayers.Bounds(left, bottom, right, top); │ │ │ │ │ + } else { │ │ │ │ │ + result = this.dragHandler.start.clone(); // i.e. OL.Pixel │ │ │ │ │ + } │ │ │ │ │ + this.removeBox(); │ │ │ │ │ + │ │ │ │ │ + this.callback("done", [result]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.0.0". │ │ │ │ │ + * Method: removeBox │ │ │ │ │ + * Remove the zoombox from the screen and nullify our reference to it. │ │ │ │ │ */ │ │ │ │ │ - defaultVersion: "1.0.0", │ │ │ │ │ + removeBox: function() { │ │ │ │ │ + this.map.viewPortDiv.removeChild(this.zoomBox); │ │ │ │ │ + this.zoomBox = null; │ │ │ │ │ + this.boxOffsets = null; │ │ │ │ │ + OpenLayers.Element.removeClass( │ │ │ │ │ + this.map.viewPortDiv, "olDrawBox" │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WPSCapabilities │ │ │ │ │ - * Create a new parser for WPS Capabilities. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * Method: activate │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.dragHandler.activate(); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: deactivate │ │ │ │ │ */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + if (this.dragHandler.deactivate()) { │ │ │ │ │ + if (this.zoomBox) { │ │ │ │ │ + this.removeBox(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return information about │ │ │ │ │ - * the service. │ │ │ │ │ + * Method: getBoxOffsets │ │ │ │ │ + * Determines border offsets for a box, according to the box model. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} Info about the WPS │ │ │ │ │ + * {Object} an object with the following offsets: │ │ │ │ │ + * - left │ │ │ │ │ + * - right │ │ │ │ │ + * - top │ │ │ │ │ + * - bottom │ │ │ │ │ + * - width │ │ │ │ │ + * - height │ │ │ │ │ */ │ │ │ │ │ + getBoxOffsets: function() { │ │ │ │ │ + if (!this.boxOffsets) { │ │ │ │ │ + // Determine the box model. If the testDiv's clientWidth is 3, then │ │ │ │ │ + // the borders are outside and we are dealing with the w3c box │ │ │ │ │ + // model. Otherwise, the browser uses the traditional box model and │ │ │ │ │ + // the borders are inside the box bounds, leaving us with a │ │ │ │ │ + // clientWidth of 1. │ │ │ │ │ + var testDiv = document.createElement("div"); │ │ │ │ │ + //testDiv.style.visibility = "hidden"; │ │ │ │ │ + testDiv.style.position = "absolute"; │ │ │ │ │ + testDiv.style.border = "1px solid black"; │ │ │ │ │ + testDiv.style.width = "3px"; │ │ │ │ │ + document.body.appendChild(testDiv); │ │ │ │ │ + var w3cBoxModel = testDiv.clientWidth == 3; │ │ │ │ │ + document.body.removeChild(testDiv); │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WPSCapabilities" │ │ │ │ │ + var left = parseInt(OpenLayers.Element.getStyle(this.zoomBox, │ │ │ │ │ + "border-left-width")); │ │ │ │ │ + var right = parseInt(OpenLayers.Element.getStyle( │ │ │ │ │ + this.zoomBox, "border-right-width")); │ │ │ │ │ + var top = parseInt(OpenLayers.Element.getStyle(this.zoomBox, │ │ │ │ │ + "border-top-width")); │ │ │ │ │ + var bottom = parseInt(OpenLayers.Element.getStyle( │ │ │ │ │ + this.zoomBox, "border-bottom-width")); │ │ │ │ │ + this.boxOffsets = { │ │ │ │ │ + left: left, │ │ │ │ │ + right: right, │ │ │ │ │ + top: top, │ │ │ │ │ + bottom: bottom, │ │ │ │ │ + width: w3cBoxModel === false ? left + right : 0, │ │ │ │ │ + height: w3cBoxModel === false ? top + bottom : 0 │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + return this.boxOffsets; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Box" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/Atom.js │ │ │ │ │ + OpenLayers/Handler/Hover.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/GML/v3.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.Atom │ │ │ │ │ - * Read/write Atom feeds. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Format.AtomFeed> constructor. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Handler.Hover │ │ │ │ │ + * The hover handler is to be used to emulate mouseovers on objects │ │ │ │ │ + * on the map that aren't DOM elements. For example one can use │ │ │ │ │ + * this handler to send WMS/GetFeatureInfo requests as the user │ │ │ │ │ + * moves the mouve over the map. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.Atom = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Handler.Hover = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. Properties │ │ │ │ │ - * of this object should not be set individually. Read-only. All │ │ │ │ │ - * XML subclasses should have their own namespaces object. Use │ │ │ │ │ - * <setNamespace> to add or set a namespace alias after construction. │ │ │ │ │ + * APIProperty: delay │ │ │ │ │ + * {Integer} - Number of milliseconds between mousemoves before │ │ │ │ │ + * the event is considered a hover. Default is 500. │ │ │ │ │ */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - atom: "http://www.w3.org/2005/Atom", │ │ │ │ │ - georss: "http://www.georss.org/georss" │ │ │ │ │ - }, │ │ │ │ │ + delay: 500, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: feedTitle │ │ │ │ │ - * {String} Atom feed elements require a title. Default is "untitled". │ │ │ │ │ + * APIProperty: pixelTolerance │ │ │ │ │ + * {Integer} - Maximum number of pixels between mousemoves for │ │ │ │ │ + * an event to be considered a hover. Default is null. │ │ │ │ │ */ │ │ │ │ │ - feedTitle: "untitled", │ │ │ │ │ + pixelTolerance: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: defaultEntryTitle │ │ │ │ │ - * {String} Atom entry elements require a title. In cases where one is │ │ │ │ │ - * not provided in the feature attributes, this will be used. Default │ │ │ │ │ - * is "untitled". │ │ │ │ │ + * APIProperty: stopMove │ │ │ │ │ + * {Boolean} - Stop other listeners from being notified on mousemoves. │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - defaultEntryTitle: "untitled", │ │ │ │ │ + stopMove: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: gmlParse │ │ │ │ │ - * {Object} GML Format object for parsing features │ │ │ │ │ - * Non-API and only created if necessary │ │ │ │ │ + * Property: px │ │ │ │ │ + * {<OpenLayers.Pixel>} - The location of the last mousemove, expressed │ │ │ │ │ + * in pixels. │ │ │ │ │ */ │ │ │ │ │ - gmlParser: null, │ │ │ │ │ + px: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: xy │ │ │ │ │ - * {Boolean} Order of the GML coordinate: true:(x,y) or false:(y,x) │ │ │ │ │ - * For GeoRSS the default is (y,x), therefore: false │ │ │ │ │ + * Property: timerId │ │ │ │ │ + * {Number} - The id of the timer. │ │ │ │ │ */ │ │ │ │ │ - xy: false, │ │ │ │ │ + timerId: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.AtomEntry │ │ │ │ │ - * Create a new parser for Atom. │ │ │ │ │ + * Constructor: OpenLayers.Handler.Hover │ │ │ │ │ + * Construct a hover handler. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that initialized this │ │ │ │ │ + * handler. The control is assumed to have a valid map property; that │ │ │ │ │ + * map is used in the handler's own setMap method. │ │ │ │ │ + * callbacks - {Object} An object with keys corresponding to callbacks │ │ │ │ │ + * that will be called by the handler. The callbacks should │ │ │ │ │ + * expect to receive a single argument, the event. Callbacks for │ │ │ │ │ + * 'move', the mouse is moving, and 'pause', the mouse is pausing, │ │ │ │ │ + * are supported. │ │ │ │ │ * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * the handler. │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Return a list of features from an Atom feed or entry document. │ │ │ │ │ - │ │ │ │ │ + * Method: mousemove │ │ │ │ │ + * Called when the mouse moves on the map. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * doc - {Element} or {String} │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - read: function(doc) { │ │ │ │ │ - if (typeof doc == "string") { │ │ │ │ │ - doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]); │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + if (this.passesTolerance(evt.xy)) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.callback('move', [evt]); │ │ │ │ │ + this.px = evt.xy; │ │ │ │ │ + // clone the evt so original properties can be accessed even │ │ │ │ │ + // if the browser deletes them during the delay │ │ │ │ │ + evt = OpenLayers.Util.extend({}, evt); │ │ │ │ │ + this.timerId = window.setTimeout( │ │ │ │ │ + OpenLayers.Function.bind(this.delayedCall, this, evt), │ │ │ │ │ + this.delay │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - return this.parseFeatures(doc); │ │ │ │ │ + return !this.stopMove; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Serialize or more feature nodes to Atom documents. │ │ │ │ │ + * Method: mouseout │ │ │ │ │ + * Called when the mouse goes out of the map. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {<OpenLayers.Feature.Vector>} or Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} an Atom entry document if passed one feature node, or a feed │ │ │ │ │ - * document if passed an array of feature nodes. │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - write: function(features) { │ │ │ │ │ - var doc; │ │ │ │ │ - if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ - doc = this.createElementNSPlus("atom:feed"); │ │ │ │ │ - doc.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:title", { │ │ │ │ │ - value: this.feedTitle │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ - for (var i = 0, ii = features.length; i < ii; i++) { │ │ │ │ │ - doc.appendChild(this.buildEntryNode(features[i])); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - doc = this.buildEntryNode(features); │ │ │ │ │ + mouseout: function(evt) { │ │ │ │ │ + if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.callback('move', [evt]); │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [doc]); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildContentNode │ │ │ │ │ + * Method: passesTolerance │ │ │ │ │ + * Determine whether the mouse move is within the optional pixel tolerance. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * content - {Object} │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} an Atom content node. │ │ │ │ │ - * │ │ │ │ │ - * TODO: types other than text. │ │ │ │ │ + * {Boolean} The mouse move is within the pixel tolerance. │ │ │ │ │ */ │ │ │ │ │ - buildContentNode: function(content) { │ │ │ │ │ - var node = this.createElementNSPlus("atom:content", { │ │ │ │ │ - attributes: { │ │ │ │ │ - type: content.type || null │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (content.src) { │ │ │ │ │ - node.setAttribute("src", content.src); │ │ │ │ │ - } else { │ │ │ │ │ - if (content.type == "text" || content.type == null) { │ │ │ │ │ - node.appendChild( │ │ │ │ │ - this.createTextNode(content.value) │ │ │ │ │ - ); │ │ │ │ │ - } else if (content.type == "html") { │ │ │ │ │ - if (typeof content.value != "string") { │ │ │ │ │ - throw "HTML content must be in form of an escaped string"; │ │ │ │ │ - } │ │ │ │ │ - node.appendChild( │ │ │ │ │ - this.createTextNode(content.value) │ │ │ │ │ - ); │ │ │ │ │ - } else if (content.type == "xhtml") { │ │ │ │ │ - node.appendChild(content.value); │ │ │ │ │ - } else if (content.type == "xhtml" || │ │ │ │ │ - content.type.match(/(\+|\/)xml$/)) { │ │ │ │ │ - node.appendChild(content.value); │ │ │ │ │ - } else { // MUST be a valid Base64 encoding │ │ │ │ │ - node.appendChild( │ │ │ │ │ - this.createTextNode(content.value) │ │ │ │ │ - ); │ │ │ │ │ + passesTolerance: function(px) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + if (this.pixelTolerance && this.px) { │ │ │ │ │ + var dpx = Math.sqrt( │ │ │ │ │ + Math.pow(this.px.x - px.x, 2) + │ │ │ │ │ + Math.pow(this.px.y - px.y, 2) │ │ │ │ │ + ); │ │ │ │ │ + if (dpx < this.pixelTolerance) { │ │ │ │ │ + passes = false; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ + return passes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildEntryNode │ │ │ │ │ - * Build an Atom entry node from a feature object. │ │ │ │ │ + * Method: clearTimer │ │ │ │ │ + * Clear the timer and set <timerId> to null. │ │ │ │ │ + */ │ │ │ │ │ + clearTimer: function() { │ │ │ │ │ + if (this.timerId != null) { │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: delayedCall │ │ │ │ │ + * Triggers pause callback. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ + */ │ │ │ │ │ + delayedCall: function(evt) { │ │ │ │ │ + this.callback('pause', [evt]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the handler. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} an Atom entry node. │ │ │ │ │ - * │ │ │ │ │ - * These entries are geared for publication using AtomPub. │ │ │ │ │ - * │ │ │ │ │ - * TODO: support extension elements │ │ │ │ │ + * {Boolean} The handler was successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ - buildEntryNode: function(feature) { │ │ │ │ │ - var attrib = feature.attributes; │ │ │ │ │ - var atomAttrib = attrib.atom || {}; │ │ │ │ │ - var entryNode = this.createElementNSPlus("atom:entry"); │ │ │ │ │ - │ │ │ │ │ - // atom:author │ │ │ │ │ - if (atomAttrib.authors) { │ │ │ │ │ - var authors = OpenLayers.Util.isArray(atomAttrib.authors) ? │ │ │ │ │ - atomAttrib.authors : [atomAttrib.authors]; │ │ │ │ │ - for (var i = 0, ii = authors.length; i < ii; i++) { │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.buildPersonConstructNode( │ │ │ │ │ - "author", authors[i] │ │ │ │ │ - ) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + deactivated = true; │ │ │ │ │ } │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // atom:category │ │ │ │ │ - if (atomAttrib.categories) { │ │ │ │ │ - var categories = OpenLayers.Util.isArray(atomAttrib.categories) ? │ │ │ │ │ - atomAttrib.categories : [atomAttrib.categories]; │ │ │ │ │ - var category; │ │ │ │ │ - for (var i = 0, ii = categories.length; i < ii; i++) { │ │ │ │ │ - category = categories[i]; │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:category", { │ │ │ │ │ - attributes: { │ │ │ │ │ - term: category.term, │ │ │ │ │ - scheme: category.scheme || null, │ │ │ │ │ - label: category.label || null │ │ │ │ │ - } │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Hover" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Handler/Click.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - // atom:content │ │ │ │ │ - if (atomAttrib.content) { │ │ │ │ │ - entryNode.appendChild(this.buildContentNode(atomAttrib.content)); │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - // atom:contributor │ │ │ │ │ - if (atomAttrib.contributors) { │ │ │ │ │ - var contributors = OpenLayers.Util.isArray(atomAttrib.contributors) ? │ │ │ │ │ - atomAttrib.contributors : [atomAttrib.contributors]; │ │ │ │ │ - for (var i = 0, ii = contributors.length; i < ii; i++) { │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.buildPersonConstructNode( │ │ │ │ │ - "contributor", │ │ │ │ │ - contributors[i] │ │ │ │ │ - ) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // atom:id │ │ │ │ │ - if (feature.fid) { │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:id", { │ │ │ │ │ - value: feature.fid │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Handler.Click │ │ │ │ │ + * A handler for mouse clicks. The intention of this handler is to give │ │ │ │ │ + * controls more flexibility with handling clicks. Browsers trigger │ │ │ │ │ + * click events twice for a double-click. In addition, the mousedown, │ │ │ │ │ + * mousemove, mouseup sequence fires a click event. With this handler, │ │ │ │ │ + * controls can decide whether to ignore clicks associated with a double │ │ │ │ │ + * click. By setting a <pixelTolerance>, controls can also ignore clicks │ │ │ │ │ + * that include a drag. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Handler.Click> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Handler.Click = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: delay │ │ │ │ │ + * {Number} Number of milliseconds between clicks before the event is │ │ │ │ │ + * considered a double-click. │ │ │ │ │ + */ │ │ │ │ │ + delay: 300, │ │ │ │ │ │ │ │ │ │ - // atom:link │ │ │ │ │ - if (atomAttrib.links) { │ │ │ │ │ - var links = OpenLayers.Util.isArray(atomAttrib.links) ? │ │ │ │ │ - atomAttrib.links : [atomAttrib.links]; │ │ │ │ │ - var link; │ │ │ │ │ - for (var i = 0, ii = links.length; i < ii; i++) { │ │ │ │ │ - link = links[i]; │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:link", { │ │ │ │ │ - attributes: { │ │ │ │ │ - href: link.href, │ │ │ │ │ - rel: link.rel || null, │ │ │ │ │ - type: link.type || null, │ │ │ │ │ - hreflang: link.hreflang || null, │ │ │ │ │ - title: link.title || null, │ │ │ │ │ - length: link.length || null │ │ │ │ │ - } │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: single │ │ │ │ │ + * {Boolean} Handle single clicks. Default is true. If false, clicks │ │ │ │ │ + * will not be reported. If true, single-clicks will be reported. │ │ │ │ │ + */ │ │ │ │ │ + single: true, │ │ │ │ │ │ │ │ │ │ - // atom:published │ │ │ │ │ - if (atomAttrib.published) { │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:published", { │ │ │ │ │ - value: atomAttrib.published │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: double │ │ │ │ │ + * {Boolean} Handle double-clicks. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + 'double': false, │ │ │ │ │ │ │ │ │ │ - // atom:rights │ │ │ │ │ - if (atomAttrib.rights) { │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:rights", { │ │ │ │ │ - value: atomAttrib.rights │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: pixelTolerance │ │ │ │ │ + * {Number} Maximum number of pixels between mouseup and mousedown for an │ │ │ │ │ + * event to be considered a click. Default is 0. If set to an │ │ │ │ │ + * integer value, clicks with a drag greater than the value will be │ │ │ │ │ + * ignored. This property can only be set when the handler is │ │ │ │ │ + * constructed. │ │ │ │ │ + */ │ │ │ │ │ + pixelTolerance: 0, │ │ │ │ │ │ │ │ │ │ - // atom:source not implemented │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: dblclickTolerance │ │ │ │ │ + * {Number} Maximum distance in pixels between clicks for a sequence of │ │ │ │ │ + * events to be considered a double click. Default is 13. If the │ │ │ │ │ + * distance between two clicks is greater than this value, a double- │ │ │ │ │ + * click will not be fired. │ │ │ │ │ + */ │ │ │ │ │ + dblclickTolerance: 13, │ │ │ │ │ │ │ │ │ │ - // atom:summary │ │ │ │ │ - if (atomAttrib.summary || attrib.description) { │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:summary", { │ │ │ │ │ - value: atomAttrib.summary || attrib.description │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: stopSingle │ │ │ │ │ + * {Boolean} Stop other listeners from being notified of clicks. Default │ │ │ │ │ + * is false. If true, any listeners registered before this one for │ │ │ │ │ + * click or rightclick events will not be notified. │ │ │ │ │ + */ │ │ │ │ │ + stopSingle: false, │ │ │ │ │ │ │ │ │ │ - // atom:title │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:title", { │ │ │ │ │ - value: atomAttrib.title || attrib.title || this.defaultEntryTitle │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: stopDouble │ │ │ │ │ + * {Boolean} Stop other listeners from being notified of double-clicks. │ │ │ │ │ + * Default is false. If true, any click listeners registered before │ │ │ │ │ + * this one will not be notified of *any* double-click events. │ │ │ │ │ + * │ │ │ │ │ + * The one caveat with stopDouble is that given a map with two click │ │ │ │ │ + * handlers, one with stopDouble true and the other with stopSingle │ │ │ │ │ + * true, the stopSingle handler should be activated last to get │ │ │ │ │ + * uniform cross-browser performance. Since IE triggers one click │ │ │ │ │ + * with a dblclick and FF triggers two, if a stopSingle handler is │ │ │ │ │ + * activated first, all it gets in IE is a single click when the │ │ │ │ │ + * second handler stops propagation on the dblclick. │ │ │ │ │ + */ │ │ │ │ │ + stopDouble: false, │ │ │ │ │ │ │ │ │ │ - // atom:updated │ │ │ │ │ - if (atomAttrib.updated) { │ │ │ │ │ - entryNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:updated", { │ │ │ │ │ - value: atomAttrib.updated │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: timerId │ │ │ │ │ + * {Number} The id of the timeout waiting to clear the <delayedCall>. │ │ │ │ │ + */ │ │ │ │ │ + timerId: null, │ │ │ │ │ │ │ │ │ │ - // georss:where │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - var whereNode = this.createElementNSPlus("georss:where"); │ │ │ │ │ - whereNode.appendChild( │ │ │ │ │ - this.buildGeometryNode(feature.geometry) │ │ │ │ │ - ); │ │ │ │ │ - entryNode.appendChild(whereNode); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: down │ │ │ │ │ + * {Object} Object that store relevant information about the last │ │ │ │ │ + * mousedown or touchstart. Its 'xy' OpenLayers.Pixel property gives │ │ │ │ │ + * the average location of the mouse/touch event. Its 'touches' │ │ │ │ │ + * property records clientX/clientY of each touches. │ │ │ │ │ + */ │ │ │ │ │ + down: null, │ │ │ │ │ │ │ │ │ │ - return entryNode; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: last │ │ │ │ │ + * {Object} Object that store relevant information about the last │ │ │ │ │ + * mousemove or touchmove. Its 'xy' OpenLayers.Pixel property gives │ │ │ │ │ + * the average location of the mouse/touch event. Its 'touches' │ │ │ │ │ + * property records clientX/clientY of each touches. │ │ │ │ │ + */ │ │ │ │ │ + last: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: first │ │ │ │ │ + * {Object} When waiting for double clicks, this object will store │ │ │ │ │ + * information about the first click in a two click sequence. │ │ │ │ │ + */ │ │ │ │ │ + first: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: initGmlParser │ │ │ │ │ - * Creates a GML parser. │ │ │ │ │ + * Property: rightclickTimerId │ │ │ │ │ + * {Number} The id of the right mouse timeout waiting to clear the │ │ │ │ │ + * <delayedEvent>. │ │ │ │ │ */ │ │ │ │ │ - initGmlParser: function() { │ │ │ │ │ - this.gmlParser = new OpenLayers.Format.GML.v3({ │ │ │ │ │ - xy: this.xy, │ │ │ │ │ - featureNS: "http://example.com#feature", │ │ │ │ │ - internalProjection: this.internalProjection, │ │ │ │ │ - externalProjection: this.externalProjection │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + rightclickTimerId: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildGeometryNode │ │ │ │ │ - * builds a GeoRSS node with a given geometry │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Handler.Click │ │ │ │ │ + * Create a new click handler. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ + * this handler. If a handler is being used without a control, the │ │ │ │ │ + * handler's setMap method must be overridden to deal properly with │ │ │ │ │ + * the map. │ │ │ │ │ + * callbacks - {Object} An object with keys corresponding to callbacks │ │ │ │ │ + * that will be called by the handler. The callbacks should │ │ │ │ │ + * expect to recieve a single argument, the click event. │ │ │ │ │ + * Callbacks for 'click' and 'dblclick' are supported. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * handler. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: touchstart │ │ │ │ │ + * Handle touchstart. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} A gml node. │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - buildGeometryNode: function(geometry) { │ │ │ │ │ - if (!this.gmlParser) { │ │ │ │ │ - this.initGmlParser(); │ │ │ │ │ - } │ │ │ │ │ - var node = this.gmlParser.writeNode("feature:_geometry", geometry); │ │ │ │ │ - return node.firstChild; │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + this.down = this.getEventInfo(evt); │ │ │ │ │ + this.last = this.getEventInfo(evt); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildPersonConstructNode │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * name - {String} │ │ │ │ │ - * value - {Object} │ │ │ │ │ + * Method: touchmove │ │ │ │ │ + * Store position of last move, because touchend event can have │ │ │ │ │ + * an empty "touches" property. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} an Atom person construct node. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * >>> buildPersonConstructNode("author", {name: "John Smith"}) │ │ │ │ │ - * {<author><name>John Smith</name></author>} │ │ │ │ │ - * │ │ │ │ │ - * TODO: how to specify extension elements? Add to the oNames array? │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - buildPersonConstructNode: function(name, value) { │ │ │ │ │ - var oNames = ["uri", "email"]; │ │ │ │ │ - var personNode = this.createElementNSPlus("atom:" + name); │ │ │ │ │ - personNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:name", { │ │ │ │ │ - value: value.name │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ - for (var i = 0, ii = oNames.length; i < ii; i++) { │ │ │ │ │ - if (value[oNames[i]]) { │ │ │ │ │ - personNode.appendChild( │ │ │ │ │ - this.createElementNSPlus("atom:" + oNames[i], { │ │ │ │ │ - value: value[oNames[i]] │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return personNode; │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + this.last = this.getEventInfo(evt); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFirstChildValue │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * nsuri - {String} Child node namespace uri ("*" for any). │ │ │ │ │ - * name - {String} Child node name. │ │ │ │ │ - * def - {String} Optional string default to return if no child found. │ │ │ │ │ + * Method: touchend │ │ │ │ │ + * Correctly set event xy property, and add lastTouches to have │ │ │ │ │ + * touches property from last touchstart or touchmove │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} The value of the first child with the given tag name. Returns │ │ │ │ │ - * default value or empty string if none found. │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - getFirstChildValue: function(node, nsuri, name, def) { │ │ │ │ │ - var value; │ │ │ │ │ - var nodes = this.getElementsByTagNameNS(node, nsuri, name); │ │ │ │ │ - if (nodes && nodes.length > 0) { │ │ │ │ │ - value = this.getChildValue(nodes[0], def); │ │ │ │ │ - } else { │ │ │ │ │ - value = def; │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + // touchstart may not have been allowed to propagate │ │ │ │ │ + if (this.down) { │ │ │ │ │ + evt.xy = this.last.xy; │ │ │ │ │ + evt.lastTouches = this.last.touches; │ │ │ │ │ + this.handleSingle(evt); │ │ │ │ │ + this.down = null; │ │ │ │ │ } │ │ │ │ │ - return value; │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseFeature │ │ │ │ │ - * Parse feature from an Atom entry node.. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} An Atom entry or feed node. │ │ │ │ │ + * Method: mousedown │ │ │ │ │ + * Handle mousedown. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - parseFeature: function(node) { │ │ │ │ │ - var atomAttrib = {}; │ │ │ │ │ - var value = null; │ │ │ │ │ - var nodes = null; │ │ │ │ │ - var attval = null; │ │ │ │ │ - var atomns = this.namespaces.atom; │ │ │ │ │ - │ │ │ │ │ - // atomAuthor* │ │ │ │ │ - this.parsePersonConstructs(node, "author", atomAttrib); │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + this.down = this.getEventInfo(evt); │ │ │ │ │ + this.last = this.getEventInfo(evt); │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // atomCategory* │ │ │ │ │ - nodes = this.getElementsByTagNameNS(node, atomns, "category"); │ │ │ │ │ - if (nodes.length > 0) { │ │ │ │ │ - atomAttrib.categories = []; │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ - value = {}; │ │ │ │ │ - value.term = nodes[i].getAttribute("term"); │ │ │ │ │ - attval = nodes[i].getAttribute("scheme"); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value.scheme = attval; │ │ │ │ │ - } │ │ │ │ │ - attval = nodes[i].getAttribute("label"); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value.label = attval; │ │ │ │ │ - } │ │ │ │ │ - atomAttrib.categories.push(value); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: mouseup │ │ │ │ │ + * Handle mouseup. Installed to support collection of right mouse events. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ + */ │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ │ │ │ │ │ - // atomContent? │ │ │ │ │ - nodes = this.getElementsByTagNameNS(node, atomns, "content"); │ │ │ │ │ - if (nodes.length > 0) { │ │ │ │ │ - value = {}; │ │ │ │ │ - attval = nodes[0].getAttribute("type"); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value.type = attval; │ │ │ │ │ - } │ │ │ │ │ - attval = nodes[0].getAttribute("src"); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value.src = attval; │ │ │ │ │ - } else { │ │ │ │ │ - if (value.type == "text" || │ │ │ │ │ - value.type == "html" || │ │ │ │ │ - value.type == null) { │ │ │ │ │ - value.value = this.getFirstChildValue( │ │ │ │ │ - node, │ │ │ │ │ - atomns, │ │ │ │ │ - "content", │ │ │ │ │ - null │ │ │ │ │ - ); │ │ │ │ │ - } else if (value.type == "xhtml" || │ │ │ │ │ - value.type.match(/(\+|\/)xml$/)) { │ │ │ │ │ - value.value = this.getChildEl(nodes[0]); │ │ │ │ │ - } else { // MUST be base64 encoded │ │ │ │ │ - value.value = this.getFirstChildValue( │ │ │ │ │ - node, │ │ │ │ │ - atomns, │ │ │ │ │ - "content", │ │ │ │ │ - null │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - atomAttrib.content = value; │ │ │ │ │ - } │ │ │ │ │ + // Collect right mouse clicks from the mouseup │ │ │ │ │ + // IE - ignores the second right click in mousedown so using │ │ │ │ │ + // mouseup instead │ │ │ │ │ + if (this.checkModifiers(evt) && this.control.handleRightClicks && │ │ │ │ │ + OpenLayers.Event.isRightClick(evt)) { │ │ │ │ │ + propagate = this.rightclick(evt); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // atomContributor* │ │ │ │ │ - this.parsePersonConstructs(node, "contributor", atomAttrib); │ │ │ │ │ + return propagate; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // atomId │ │ │ │ │ - atomAttrib.id = this.getFirstChildValue(node, atomns, "id", null); │ │ │ │ │ + /** │ │ │ │ │ + * Method: rightclick │ │ │ │ │ + * Handle rightclick. For a dblrightclick, we get two clicks so we need │ │ │ │ │ + * to always register for dblrightclick to properly handle single │ │ │ │ │ + * clicks. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ + */ │ │ │ │ │ + rightclick: function(evt) { │ │ │ │ │ + if (this.passesTolerance(evt)) { │ │ │ │ │ + if (this.rightclickTimerId != null) { │ │ │ │ │ + //Second click received before timeout this must be │ │ │ │ │ + // a double click │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.callback('dblrightclick', [evt]); │ │ │ │ │ + return !this.stopDouble; │ │ │ │ │ + } else { │ │ │ │ │ + //Set the rightclickTimerId, send evt only if double is │ │ │ │ │ + // true else trigger single │ │ │ │ │ + var clickEvent = this['double'] ? │ │ │ │ │ + OpenLayers.Util.extend({}, evt) : │ │ │ │ │ + this.callback('rightclick', [evt]); │ │ │ │ │ │ │ │ │ │ - // atomLink* │ │ │ │ │ - nodes = this.getElementsByTagNameNS(node, atomns, "link"); │ │ │ │ │ - if (nodes.length > 0) { │ │ │ │ │ - atomAttrib.links = new Array(nodes.length); │ │ │ │ │ - } │ │ │ │ │ - var oAtts = ["rel", "type", "hreflang", "title", "length"]; │ │ │ │ │ - for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ - value = {}; │ │ │ │ │ - value.href = nodes[i].getAttribute("href"); │ │ │ │ │ - for (var j = 0, jj = oAtts.length; j < jj; j++) { │ │ │ │ │ - attval = nodes[i].getAttribute(oAtts[j]); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value[oAtts[j]] = attval; │ │ │ │ │ - } │ │ │ │ │ + var delayedRightCall = OpenLayers.Function.bind( │ │ │ │ │ + this.delayedRightCall, │ │ │ │ │ + this, │ │ │ │ │ + clickEvent │ │ │ │ │ + ); │ │ │ │ │ + this.rightclickTimerId = window.setTimeout( │ │ │ │ │ + delayedRightCall, this.delay │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - atomAttrib.links[i] = value; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // atomPublished? │ │ │ │ │ - value = this.getFirstChildValue(node, atomns, "published", null); │ │ │ │ │ - if (value) { │ │ │ │ │ - atomAttrib.published = value; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // atomRights? │ │ │ │ │ - value = this.getFirstChildValue(node, atomns, "rights", null); │ │ │ │ │ - if (value) { │ │ │ │ │ - atomAttrib.rights = value; │ │ │ │ │ } │ │ │ │ │ + return !this.stopSingle; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // atomSource? -- not implemented │ │ │ │ │ - │ │ │ │ │ - // atomSummary? │ │ │ │ │ - value = this.getFirstChildValue(node, atomns, "summary", null); │ │ │ │ │ - if (value) { │ │ │ │ │ - atomAttrib.summary = value; │ │ │ │ │ + /** │ │ │ │ │ + * Method: delayedRightCall │ │ │ │ │ + * Sets <rightclickTimerId> to null. And optionally triggers the │ │ │ │ │ + * rightclick callback if evt is set. │ │ │ │ │ + */ │ │ │ │ │ + delayedRightCall: function(evt) { │ │ │ │ │ + this.rightclickTimerId = null; │ │ │ │ │ + if (evt) { │ │ │ │ │ + this.callback('rightclick', [evt]); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // atomTitle │ │ │ │ │ - atomAttrib.title = this.getFirstChildValue( │ │ │ │ │ - node, atomns, "title", null │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - // atomUpdated │ │ │ │ │ - atomAttrib.updated = this.getFirstChildValue( │ │ │ │ │ - node, atomns, "updated", null │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - var featureAttrib = { │ │ │ │ │ - title: atomAttrib.title, │ │ │ │ │ - description: atomAttrib.summary, │ │ │ │ │ - atom: atomAttrib │ │ │ │ │ - }; │ │ │ │ │ - var geometry = this.parseLocations(node)[0]; │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry, featureAttrib); │ │ │ │ │ - feature.fid = atomAttrib.id; │ │ │ │ │ - return feature; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseFeatures │ │ │ │ │ - * Return features from an Atom entry or feed. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} An Atom entry or feed node. │ │ │ │ │ + * Method: click │ │ │ │ │ + * Handle click events from the browser. This is registered as a listener │ │ │ │ │ + * for click events and should not be called from other events in this │ │ │ │ │ + * handler. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - parseFeatures: function(node) { │ │ │ │ │ - var features = []; │ │ │ │ │ - var entries = this.getElementsByTagNameNS( │ │ │ │ │ - node, this.namespaces.atom, "entry" │ │ │ │ │ - ); │ │ │ │ │ - if (entries.length == 0) { │ │ │ │ │ - entries = [node]; │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, ii = entries.length; i < ii; i++) { │ │ │ │ │ - features.push(this.parseFeature(entries[i])); │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + if (!this.last) { │ │ │ │ │ + this.last = this.getEventInfo(evt); │ │ │ │ │ } │ │ │ │ │ - return features; │ │ │ │ │ + this.handleSingle(evt); │ │ │ │ │ + return !this.stopSingle; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseLocations │ │ │ │ │ - * Parse the locations from an Atom entry or feed. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} An Atom entry or feed node. │ │ │ │ │ - * │ │ │ │ │ + * Method: dblclick │ │ │ │ │ + * Handle dblclick. For a dblclick, we get two clicks in some browsers │ │ │ │ │ + * (FF) and one in others (IE). So we need to always register for │ │ │ │ │ + * dblclick to properly handle single clicks. This method is registered │ │ │ │ │ + * as a listener for the dblclick browser event. It should *not* be │ │ │ │ │ + * called by other methods in this handler. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * Array({<OpenLayers.Geometry>}) │ │ │ │ │ + * {Boolean} Continue propagating this event. │ │ │ │ │ */ │ │ │ │ │ - parseLocations: function(node) { │ │ │ │ │ - var georssns = this.namespaces.georss; │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + this.handleDouble(evt); │ │ │ │ │ + return !this.stopDouble; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var locations = { │ │ │ │ │ - components: [] │ │ │ │ │ - }; │ │ │ │ │ - var where = this.getElementsByTagNameNS(node, georssns, "where"); │ │ │ │ │ - if (where && where.length > 0) { │ │ │ │ │ - if (!this.gmlParser) { │ │ │ │ │ - this.initGmlParser(); │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, ii = where.length; i < ii; i++) { │ │ │ │ │ - this.gmlParser.readChildNodes(where[i], locations); │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleDouble │ │ │ │ │ + * Handle double-click sequence. │ │ │ │ │ + */ │ │ │ │ │ + handleDouble: function(evt) { │ │ │ │ │ + if (this.passesDblclickTolerance(evt)) { │ │ │ │ │ + if (this["double"]) { │ │ │ │ │ + this.callback("dblclick", [evt]); │ │ │ │ │ } │ │ │ │ │ + // to prevent a dblclick from firing the click callback in IE │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var components = locations.components; │ │ │ │ │ - var point = this.getElementsByTagNameNS(node, georssns, "point"); │ │ │ │ │ - if (point && point.length > 0) { │ │ │ │ │ - for (var i = 0, ii = point.length; i < ii; i++) { │ │ │ │ │ - var xy = OpenLayers.String.trim( │ │ │ │ │ - point[i].firstChild.nodeValue │ │ │ │ │ - ).split(/\s+/); │ │ │ │ │ - if (xy.length != 2) { │ │ │ │ │ - xy = OpenLayers.String.trim( │ │ │ │ │ - point[i].firstChild.nodeValue │ │ │ │ │ - ).split(/\s*,\s*/); │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleSingle │ │ │ │ │ + * Handle single click sequence. │ │ │ │ │ + */ │ │ │ │ │ + handleSingle: function(evt) { │ │ │ │ │ + if (this.passesTolerance(evt)) { │ │ │ │ │ + if (this.timerId != null) { │ │ │ │ │ + // already received a click │ │ │ │ │ + if (this.last.touches && this.last.touches.length === 1) { │ │ │ │ │ + // touch device, no dblclick event - this may be a double │ │ │ │ │ + if (this["double"]) { │ │ │ │ │ + // on Android don't let the browser zoom on the page │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + } │ │ │ │ │ + this.handleDouble(evt); │ │ │ │ │ } │ │ │ │ │ - components.push(new OpenLayers.Geometry.Point(xy[1], xy[0])); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var line = this.getElementsByTagNameNS(node, georssns, "line"); │ │ │ │ │ - if (line && line.length > 0) { │ │ │ │ │ - var coords; │ │ │ │ │ - var p; │ │ │ │ │ - var points; │ │ │ │ │ - for (var i = 0, ii = line.length; i < ii; i++) { │ │ │ │ │ - coords = OpenLayers.String.trim( │ │ │ │ │ - line[i].firstChild.nodeValue │ │ │ │ │ - ).split(/\s+/); │ │ │ │ │ - points = []; │ │ │ │ │ - for (var j = 0, jj = coords.length; j < jj; j += 2) { │ │ │ │ │ - p = new OpenLayers.Geometry.Point(coords[j + 1], coords[j]); │ │ │ │ │ - points.push(p); │ │ │ │ │ + // if we're not in a touch environment we clear the click timer │ │ │ │ │ + // if we've got a second touch, we'll get two touchend events │ │ │ │ │ + if (!this.last.touches || this.last.touches.length !== 2) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ } │ │ │ │ │ - components.push( │ │ │ │ │ - new OpenLayers.Geometry.LineString(points) │ │ │ │ │ - ); │ │ │ │ │ + } else { │ │ │ │ │ + // remember the first click info so we can compare to the second │ │ │ │ │ + this.first = this.getEventInfo(evt); │ │ │ │ │ + // set the timer, send evt only if single is true │ │ │ │ │ + //use a clone of the event object because it will no longer │ │ │ │ │ + //be a valid event object in IE in the timer callback │ │ │ │ │ + var clickEvent = this.single ? │ │ │ │ │ + OpenLayers.Util.extend({}, evt) : null; │ │ │ │ │ + this.queuePotentialClick(clickEvent); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var polygon = this.getElementsByTagNameNS(node, georssns, "polygon"); │ │ │ │ │ - if (polygon && polygon.length > 0) { │ │ │ │ │ - var coords; │ │ │ │ │ - var p; │ │ │ │ │ - var points; │ │ │ │ │ - for (var i = 0, ii = polygon.length; i < ii; i++) { │ │ │ │ │ - coords = OpenLayers.String.trim( │ │ │ │ │ - polygon[i].firstChild.nodeValue │ │ │ │ │ - ).split(/\s+/); │ │ │ │ │ - points = []; │ │ │ │ │ - for (var j = 0, jj = coords.length; j < jj; j += 2) { │ │ │ │ │ - p = new OpenLayers.Geometry.Point(coords[j + 1], coords[j]); │ │ │ │ │ - points.push(p); │ │ │ │ │ - } │ │ │ │ │ - components.push( │ │ │ │ │ - new OpenLayers.Geometry.Polygon( │ │ │ │ │ - [new OpenLayers.Geometry.LinearRing(points)] │ │ │ │ │ - ) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: queuePotentialClick │ │ │ │ │ + * This method is separated out largely to make testing easier (so we │ │ │ │ │ + * don't have to override window.setTimeout) │ │ │ │ │ + */ │ │ │ │ │ + queuePotentialClick: function(evt) { │ │ │ │ │ + this.timerId = window.setTimeout( │ │ │ │ │ + OpenLayers.Function.bind(this.delayedCall, this, evt), │ │ │ │ │ + this.delay │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - for (var i = 0, ii = components.length; i < ii; i++) { │ │ │ │ │ - if (components[i]) { │ │ │ │ │ - components[i].transform( │ │ │ │ │ - this.externalProjection, │ │ │ │ │ - this.internalProjection │ │ │ │ │ - ); │ │ │ │ │ + /** │ │ │ │ │ + * Method: passesTolerance │ │ │ │ │ + * Determine whether the event is within the optional pixel tolerance. Note │ │ │ │ │ + * that the pixel tolerance check only works if mousedown events get to │ │ │ │ │ + * the listeners registered here. If they are stopped by other elements, │ │ │ │ │ + * the <pixelTolerance> will have no effect here (this method will always │ │ │ │ │ + * return true). │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The click is within the pixel tolerance (if specified). │ │ │ │ │ + */ │ │ │ │ │ + passesTolerance: function(evt) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + if (this.pixelTolerance != null && this.down && this.down.xy) { │ │ │ │ │ + passes = this.pixelTolerance >= this.down.xy.distanceTo(evt.xy); │ │ │ │ │ + // for touch environments, we also enforce that all touches │ │ │ │ │ + // start and end within the given tolerance to be considered a click │ │ │ │ │ + if (passes && this.touch && │ │ │ │ │ + this.down.touches.length === this.last.touches.length) { │ │ │ │ │ + // the touchend event doesn't come with touches, so we check │ │ │ │ │ + // down and last │ │ │ │ │ + for (var i = 0, ii = this.down.touches.length; i < ii; ++i) { │ │ │ │ │ + if (this.getTouchDistance( │ │ │ │ │ + this.down.touches[i], │ │ │ │ │ + this.last.touches[i] │ │ │ │ │ + ) > this.pixelTolerance) { │ │ │ │ │ + passes = false; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return passes; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return components; │ │ │ │ │ + /** │ │ │ │ │ + * Method: getTouchDistance │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The pixel displacement between two touches. │ │ │ │ │ + */ │ │ │ │ │ + getTouchDistance: function(from, to) { │ │ │ │ │ + return Math.sqrt( │ │ │ │ │ + Math.pow(from.clientX - to.clientX, 2) + │ │ │ │ │ + Math.pow(from.clientY - to.clientY, 2) │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parsePersonConstruct │ │ │ │ │ - * Parse Atom person constructs from an Atom entry node. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} An Atom entry or feed node. │ │ │ │ │ - * name - {String} Construcy name ("author" or "contributor") │ │ │ │ │ - * data = {Object} Object in which to put parsed persons. │ │ │ │ │ + * Method: passesDblclickTolerance │ │ │ │ │ + * Determine whether the event is within the optional double-cick pixel │ │ │ │ │ + * tolerance. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * An {Object}. │ │ │ │ │ + * {Boolean} The click is within the double-click pixel tolerance. │ │ │ │ │ */ │ │ │ │ │ - parsePersonConstructs: function(node, name, data) { │ │ │ │ │ - var persons = []; │ │ │ │ │ - var atomns = this.namespaces.atom; │ │ │ │ │ - var nodes = this.getElementsByTagNameNS(node, atomns, name); │ │ │ │ │ - var oAtts = ["uri", "email"]; │ │ │ │ │ - for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ - var value = {}; │ │ │ │ │ - value.name = this.getFirstChildValue( │ │ │ │ │ - nodes[i], │ │ │ │ │ - atomns, │ │ │ │ │ - "name", │ │ │ │ │ - null │ │ │ │ │ - ); │ │ │ │ │ - for (var j = 0, jj = oAtts.length; j < jj; j++) { │ │ │ │ │ - var attval = this.getFirstChildValue( │ │ │ │ │ - nodes[i], │ │ │ │ │ - atomns, │ │ │ │ │ - oAtts[j], │ │ │ │ │ - null); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value[oAtts[j]] = attval; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - persons.push(value); │ │ │ │ │ - } │ │ │ │ │ - if (persons.length > 0) { │ │ │ │ │ - data[name + "s"] = persons; │ │ │ │ │ + passesDblclickTolerance: function(evt) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + if (this.down && this.first) { │ │ │ │ │ + passes = this.down.xy.distanceTo(this.first.xy) <= this.dblclickTolerance; │ │ │ │ │ } │ │ │ │ │ + return passes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.Atom" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/SOSCapabilities.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.SOSCapabilities │ │ │ │ │ - * Read SOS Capabilities. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.SOSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + /** │ │ │ │ │ + * Method: clearTimer │ │ │ │ │ + * Clear the timer and set <timerId> to null. │ │ │ │ │ + */ │ │ │ │ │ + clearTimer: function() { │ │ │ │ │ + if (this.timerId != null) { │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.rightclickTimerId != null) { │ │ │ │ │ + window.clearTimeout(this.rightclickTimerId); │ │ │ │ │ + this.rightclickTimerId = null; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.0.0". │ │ │ │ │ + * Method: delayedCall │ │ │ │ │ + * Sets <timerId> to null. And optionally triggers the click callback if │ │ │ │ │ + * evt is set. │ │ │ │ │ */ │ │ │ │ │ - defaultVersion: "1.0.0", │ │ │ │ │ + delayedCall: function(evt) { │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + if (evt) { │ │ │ │ │ + this.callback("click", [evt]); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.SOSCapabilities │ │ │ │ │ - * Create a new parser for SOS Capabilities. │ │ │ │ │ + * Method: getEventInfo │ │ │ │ │ + * This method allows us to store event information without storing the │ │ │ │ │ + * actual event. In touch devices (at least), the same event is │ │ │ │ │ + * modified between touchstart, touchmove, and touchend. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object with event related info. │ │ │ │ │ */ │ │ │ │ │ + getEventInfo: function(evt) { │ │ │ │ │ + var touches; │ │ │ │ │ + if (evt.touches) { │ │ │ │ │ + var len = evt.touches.length; │ │ │ │ │ + touches = new Array(len); │ │ │ │ │ + var touch; │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + touch = evt.touches[i]; │ │ │ │ │ + touches[i] = { │ │ │ │ │ + clientX: touch.olClientX, │ │ │ │ │ + clientY: touch.olClientY │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + xy: evt.xy, │ │ │ │ │ + touches: touches │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return information about │ │ │ │ │ - * the service (offering and observedProperty mostly). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the handler. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} Info about the SOS │ │ │ │ │ + * {Boolean} The handler was successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.down = null; │ │ │ │ │ + this.first = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + deactivated = true; │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SOSCapabilities" │ │ │ │ │ - │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Click" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/SOSGetFeatureOfInterest.js │ │ │ │ │ + OpenLayers/Handler/Feature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/GML/v3.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.SOSGetFeatureOfInterest │ │ │ │ │ - * Read and write SOS GetFeatureOfInterest. This is used to get to │ │ │ │ │ - * the location of the features (stations). The stations can have 1 or more │ │ │ │ │ - * sensors. │ │ │ │ │ + * Class: OpenLayers.Handler.Feature │ │ │ │ │ + * Handler to respond to mouse events related to a drawn feature. Callbacks │ │ │ │ │ + * with the following keys will be notified of the following events │ │ │ │ │ + * associated with features: click, clickout, over, out, and dblclick. │ │ │ │ │ + * │ │ │ │ │ + * This handler stops event propagation for mousedown and mouseup if those │ │ │ │ │ + * browser events target features that can be selected. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.SOSGetFeatureOfInterest = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.XML, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: VERSION │ │ │ │ │ - * {String} 1.0.0 │ │ │ │ │ - */ │ │ │ │ │ - VERSION: "1.0.0", │ │ │ │ │ +OpenLayers.Handler.Feature = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ - gml: "http://www.opengis.net/gml", │ │ │ │ │ - sa: "http://www.opengis.net/sampling/1.0", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + /** │ │ │ │ │ + * Property: EVENTMAP │ │ │ │ │ + * {Object} A object mapping the browser events to objects with callback │ │ │ │ │ + * keys for in and out. │ │ │ │ │ + */ │ │ │ │ │ + EVENTMAP: { │ │ │ │ │ + 'click': { │ │ │ │ │ + 'in': 'click', │ │ │ │ │ + 'out': 'clickout' │ │ │ │ │ + }, │ │ │ │ │ + 'mousemove': { │ │ │ │ │ + 'in': 'over', │ │ │ │ │ + 'out': 'out' │ │ │ │ │ }, │ │ │ │ │ + 'dblclick': { │ │ │ │ │ + 'in': 'dblclick', │ │ │ │ │ + 'out': null │ │ │ │ │ + }, │ │ │ │ │ + 'mousedown': { │ │ │ │ │ + 'in': null, │ │ │ │ │ + 'out': null │ │ │ │ │ + }, │ │ │ │ │ + 'mouseup': { │ │ │ │ │ + 'in': null, │ │ │ │ │ + 'out': null │ │ │ │ │ + }, │ │ │ │ │ + 'touchstart': { │ │ │ │ │ + 'in': 'click', │ │ │ │ │ + 'out': 'clickout' │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} Schema location │ │ │ │ │ - */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/sos/1.0 http://schemas.opengis.net/sos/1.0.0/sosAll.xsd", │ │ │ │ │ + /** │ │ │ │ │ + * Property: feature │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} The last feature that was hovered. │ │ │ │ │ + */ │ │ │ │ │ + feature: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - */ │ │ │ │ │ - defaultPrefix: "sos", │ │ │ │ │ + /** │ │ │ │ │ + * Property: lastFeature │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} The last feature that was handled. │ │ │ │ │ + */ │ │ │ │ │ + lastFeature: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ - */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: down │ │ │ │ │ + * {<OpenLayers.Pixel>} The location of the last mousedown. │ │ │ │ │ + */ │ │ │ │ │ + down: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.SOSGetFeatureOfInterest │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: up │ │ │ │ │ + * {<OpenLayers.Pixel>} The location of the last mouseup. │ │ │ │ │ + */ │ │ │ │ │ + up: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Parse a GetFeatureOfInterest response and return an array of features │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var info = { │ │ │ │ │ - features: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readNode(data, info); │ │ │ │ │ - │ │ │ │ │ - var features = []; │ │ │ │ │ - for (var i = 0, len = info.features.length; i < len; i++) { │ │ │ │ │ - var container = info.features[i]; │ │ │ │ │ - // reproject features if needed │ │ │ │ │ - if (this.internalProjection && this.externalProjection && │ │ │ │ │ - container.components[0]) { │ │ │ │ │ - container.components[0].transform( │ │ │ │ │ - this.externalProjection, this.internalProjection │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector( │ │ │ │ │ - container.components[0], container.attributes); │ │ │ │ │ - features.push(feature); │ │ │ │ │ - } │ │ │ │ │ - return features; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "sa": { │ │ │ │ │ - "SamplingPoint": function(node, obj) { │ │ │ │ │ - // sampling point can also be without a featureMember if │ │ │ │ │ - // there is only 1 │ │ │ │ │ - if (!obj.attributes) { │ │ │ │ │ - var feature = { │ │ │ │ │ - attributes: {} │ │ │ │ │ - }; │ │ │ │ │ - obj.features.push(feature); │ │ │ │ │ - obj = feature; │ │ │ │ │ - } │ │ │ │ │ - obj.attributes.id = this.getAttributeNS(node, │ │ │ │ │ - this.namespaces.gml, "id"); │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "position": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "FeatureCollection": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "featureMember": function(node, obj) { │ │ │ │ │ - var feature = { │ │ │ │ │ - attributes: {} │ │ │ │ │ - }; │ │ │ │ │ - obj.features.push(feature); │ │ │ │ │ - this.readChildNodes(node, feature); │ │ │ │ │ - }, │ │ │ │ │ - "name": function(node, obj) { │ │ │ │ │ - obj.attributes.name = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "pos": function(node, obj) { │ │ │ │ │ - // we need to parse the srsName to get to the │ │ │ │ │ - // externalProjection, that's why we cannot use │ │ │ │ │ - // GML v3 for this │ │ │ │ │ - if (!this.externalProjection) { │ │ │ │ │ - this.externalProjection = new OpenLayers.Projection( │ │ │ │ │ - node.getAttribute("srsName")); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Format.GML.v3.prototype.readers.gml.pos.apply( │ │ │ │ │ - this, [node, obj]); │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.GML.v3.prototype.readers.gml) │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ - */ │ │ │ │ │ - writers: { │ │ │ │ │ - "sos": { │ │ │ │ │ - "GetFeatureOfInterest": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("GetFeatureOfInterest", { │ │ │ │ │ - attributes: { │ │ │ │ │ - version: this.VERSION, │ │ │ │ │ - service: 'SOS', │ │ │ │ │ - "xsi:schemaLocation": this.schemaLocation │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - for (var i = 0, len = options.fois.length; i < len; i++) { │ │ │ │ │ - this.writeNode("FeatureOfInterestId", { │ │ │ │ │ - foi: options.fois[i] │ │ │ │ │ - }, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "FeatureOfInterestId": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("FeatureOfInterestId", { │ │ │ │ │ - value: options.foi │ │ │ │ │ - }); │ │ │ │ │ - return node; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SOSGetFeatureOfInterest" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSDescribeLayer.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: clickTolerance │ │ │ │ │ + * {Number} The number of pixels the mouse can move between mousedown │ │ │ │ │ + * and mouseup for the event to still be considered a click. │ │ │ │ │ + * Dragging the map should not trigger the click and clickout callbacks │ │ │ │ │ + * unless the map is moved by less than this tolerance. Defaults to 4. │ │ │ │ │ + */ │ │ │ │ │ + clickTolerance: 4, │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: geometryTypes │ │ │ │ │ + * To restrict dragging to a limited set of geometry types, send a list │ │ │ │ │ + * of strings corresponding to the geometry class names. │ │ │ │ │ + * │ │ │ │ │ + * @type Array(String) │ │ │ │ │ + */ │ │ │ │ │ + geometryTypes: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: stopClick │ │ │ │ │ + * {Boolean} If stopClick is set to true, handled clicks do not │ │ │ │ │ + * propagate to other click listeners. Otherwise, handled clicks │ │ │ │ │ + * do propagate. Unhandled clicks always propagate, whatever the │ │ │ │ │ + * value of stopClick. Defaults to true. │ │ │ │ │ + */ │ │ │ │ │ + stopClick: true, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WMSDescribeLayer │ │ │ │ │ - * Read SLD WMS DescribeLayer response │ │ │ │ │ - * DescribeLayer is meant to couple WMS to WFS and WCS │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WMSDescribeLayer = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + /** │ │ │ │ │ + * Property: stopDown │ │ │ │ │ + * {Boolean} If stopDown is set to true, handled mousedowns do not │ │ │ │ │ + * propagate to other mousedown listeners. Otherwise, handled │ │ │ │ │ + * mousedowns do propagate. Unhandled mousedowns always propagate, │ │ │ │ │ + * whatever the value of stopDown. Defaults to true. │ │ │ │ │ + */ │ │ │ │ │ + stopDown: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.1.1". │ │ │ │ │ + * Property: stopUp │ │ │ │ │ + * {Boolean} If stopUp is set to true, handled mouseups do not │ │ │ │ │ + * propagate to other mouseup listeners. Otherwise, handled mouseups │ │ │ │ │ + * do propagate. Unhandled mouseups always propagate, whatever the │ │ │ │ │ + * value of stopUp. Defaults to false. │ │ │ │ │ */ │ │ │ │ │ - defaultVersion: "1.1.1", │ │ │ │ │ + stopUp: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMSDescribeLayer │ │ │ │ │ - * Create a new parser for WMS DescribeLayer responses. │ │ │ │ │ + * Constructor: OpenLayers.Handler.Feature │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * control - {<OpenLayers.Control>} │ │ │ │ │ + * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ + * callbacks - {Object} An object with a 'over' property whos value is │ │ │ │ │ + * a function to be called when the mouse is over a feature. The │ │ │ │ │ + * callback should expect to recieve a single argument, the feature. │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ + initialize: function(control, layer, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read DescribeLayer data from a string, and return the response. │ │ │ │ │ - * The OGC currently defines 2 formats which are allowed for output, │ │ │ │ │ - * so we need to parse these 2 types │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * Method: touchstart │ │ │ │ │ + * Handle touchstart events │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array} Array of {<LayerDescription>} objects which have: │ │ │ │ │ - * - {String} owsType: WFS/WCS │ │ │ │ │ - * - {String} owsURL: the online resource │ │ │ │ │ - * - {String} typeName: the name of the typename on the service │ │ │ │ │ + * {Boolean} Let the event propagate. │ │ │ │ │ */ │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + return OpenLayers.Event.isMultiTouch(evt) ? │ │ │ │ │ + true : this.mousedown(evt); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSDescribeLayer" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/SOSGetObservation.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/SOSGetFeatureOfInterest.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.SOSGetObservation │ │ │ │ │ - * Read and write SOS GetObersation (to get the actual values from a sensor) │ │ │ │ │ - * version 1.0.0 │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.SOSGetObservation = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + /** │ │ │ │ │ + * Method: touchmove │ │ │ │ │ + * Handle touchmove events. We just prevent the browser default behavior, │ │ │ │ │ + * for Android Webkit not to select text when moving the finger after │ │ │ │ │ + * selecting a feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + * Method: mousedown │ │ │ │ │ + * Handle mouse down. Stop propagation if a feature is targeted by this │ │ │ │ │ + * event (stops map dragging during feature selection). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - ows: "http://www.opengis.net/ows", │ │ │ │ │ - gml: "http://www.opengis.net/gml", │ │ │ │ │ - sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ - ogc: "http://www.opengis.net/ogc", │ │ │ │ │ - om: "http://www.opengis.net/om/1.0", │ │ │ │ │ - sa: "http://www.opengis.net/sampling/1.0", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + // Feature selection is only done with a left click. Other handlers may stop the │ │ │ │ │ + // propagation of left-click mousedown events but not right-click mousedown events. │ │ │ │ │ + // This mismatch causes problems when comparing the location of the down and up │ │ │ │ │ + // events in the click function so it is important ignore right-clicks. │ │ │ │ │ + if (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ + this.down = evt.xy; │ │ │ │ │ + } │ │ │ │ │ + return this.handle(evt) ? !this.stopDown : true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ + * Method: mouseup │ │ │ │ │ + * Handle mouse up. Stop propagation if a feature is targeted by this │ │ │ │ │ + * event. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + this.up = evt.xy; │ │ │ │ │ + return this.handle(evt) ? !this.stopUp : true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: VERSION │ │ │ │ │ - * {String} 1.0.0 │ │ │ │ │ + * Method: click │ │ │ │ │ + * Handle click. Call the "click" callback if click on a feature, │ │ │ │ │ + * or the "clickout" callback if click outside any feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - VERSION: "1.0.0", │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + return this.handle(evt) ? !this.stopClick : true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} Schema location │ │ │ │ │ + * Method: mousemove │ │ │ │ │ + * Handle mouse moves. Call the "over" callback if moving in to a feature, │ │ │ │ │ + * or the "out" callback if moving out of a feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/sos/1.0 http://schemas.opengis.net/sos/1.0.0/sosGetObservation.xsd", │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + if (!this.callbacks['over'] && !this.callbacks['out']) { │ │ │ │ │ + return true; │ │ │ │ │ + } │ │ │ │ │ + this.handle(evt); │ │ │ │ │ + return true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ + * Method: dblclick │ │ │ │ │ + * Handle dblclick. Call the "dblclick" callback if dblclick on a feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - defaultPrefix: "sos", │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + return !this.handle(evt); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.SOSGetObservation │ │ │ │ │ + * Method: geometryTypeMatches │ │ │ │ │ + * Return true if the geometry type of the passed feature matches │ │ │ │ │ + * one of the geometry types in the geometryTypes array. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * feature - {<OpenLayers.Vector.Feature>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ + geometryTypeMatches: function(feature) { │ │ │ │ │ + return this.geometryTypes == null || │ │ │ │ │ + OpenLayers.Util.indexOf(this.geometryTypes, │ │ │ │ │ + feature.geometry.CLASS_NAME) > -1; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * Method: handle │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} An object containing the measurements │ │ │ │ │ + * {Boolean} The event occurred over a relevant feature. │ │ │ │ │ */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + handle: function(evt) { │ │ │ │ │ + if (this.feature && !this.feature.layer) { │ │ │ │ │ + // feature has been destroyed │ │ │ │ │ + this.feature = null; │ │ │ │ │ } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ + var type = evt.type; │ │ │ │ │ + var handled = false; │ │ │ │ │ + var previouslyIn = !!(this.feature); // previously in a feature │ │ │ │ │ + var click = (type == "click" || type == "dblclick" || type == "touchstart"); │ │ │ │ │ + this.feature = this.layer.getFeatureFromEvent(evt); │ │ │ │ │ + if (this.feature && !this.feature.layer) { │ │ │ │ │ + // feature has been destroyed │ │ │ │ │ + this.feature = null; │ │ │ │ │ } │ │ │ │ │ - var info = { │ │ │ │ │ - measurements: [], │ │ │ │ │ - observations: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readNode(data, info); │ │ │ │ │ - return info; │ │ │ │ │ + if (this.lastFeature && !this.lastFeature.layer) { │ │ │ │ │ + // last feature has been destroyed │ │ │ │ │ + this.lastFeature = null; │ │ │ │ │ + } │ │ │ │ │ + if (this.feature) { │ │ │ │ │ + if (type === "touchstart") { │ │ │ │ │ + // stop the event to prevent Android Webkit from │ │ │ │ │ + // "flashing" the map div │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + } │ │ │ │ │ + var inNew = (this.feature != this.lastFeature); │ │ │ │ │ + if (this.geometryTypeMatches(this.feature)) { │ │ │ │ │ + // in to a feature │ │ │ │ │ + if (previouslyIn && inNew) { │ │ │ │ │ + // out of last feature and in to another │ │ │ │ │ + if (this.lastFeature) { │ │ │ │ │ + this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ + } │ │ │ │ │ + this.triggerCallback(type, 'in', [this.feature]); │ │ │ │ │ + } else if (!previouslyIn || click) { │ │ │ │ │ + // in feature for the first time │ │ │ │ │ + this.triggerCallback(type, 'in', [this.feature]); │ │ │ │ │ + } │ │ │ │ │ + this.lastFeature = this.feature; │ │ │ │ │ + handled = true; │ │ │ │ │ + } else { │ │ │ │ │ + // not in to a feature │ │ │ │ │ + if (this.lastFeature && (previouslyIn && inNew || click)) { │ │ │ │ │ + // out of last feature for the first time │ │ │ │ │ + this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ + } │ │ │ │ │ + // next time the mouse goes in a feature whose geometry type │ │ │ │ │ + // doesn't match we don't want to call the 'out' callback │ │ │ │ │ + // again, so let's set this.feature to null so that │ │ │ │ │ + // previouslyIn will evaluate to false the next time │ │ │ │ │ + // we enter handle. Yes, a bit hackish... │ │ │ │ │ + this.feature = null; │ │ │ │ │ + } │ │ │ │ │ + } else if (this.lastFeature && (previouslyIn || click)) { │ │ │ │ │ + this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ + } │ │ │ │ │ + return handled; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: write │ │ │ │ │ + * Method: triggerCallback │ │ │ │ │ + * Call the callback keyed in the event map with the supplied arguments. │ │ │ │ │ + * For click and clickout, the <clickTolerance> is checked first. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object. │ │ │ │ │ + * type - {String} │ │ │ │ │ + */ │ │ │ │ │ + triggerCallback: function(type, mode, args) { │ │ │ │ │ + var key = this.EVENTMAP[type][mode]; │ │ │ │ │ + if (key) { │ │ │ │ │ + if (type == 'click' && this.up && this.down) { │ │ │ │ │ + // for click/clickout, only trigger callback if tolerance is met │ │ │ │ │ + var dpx = Math.sqrt( │ │ │ │ │ + Math.pow(this.up.x - this.down.x, 2) + │ │ │ │ │ + Math.pow(this.up.y - this.down.y, 2) │ │ │ │ │ + ); │ │ │ │ │ + if (dpx <= this.clickTolerance) { │ │ │ │ │ + this.callback(key, args); │ │ │ │ │ + } │ │ │ │ │ + // we're done with this set of events now: clear the cached │ │ │ │ │ + // positions so we can't trip over them later (this can occur │ │ │ │ │ + // if one of the up/down events gets eaten before it gets to us │ │ │ │ │ + // but we still get the click) │ │ │ │ │ + this.up = this.down = null; │ │ │ │ │ + } else { │ │ │ │ │ + this.callback(key, args); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Turn on the handler. Returns false if the handler was already active. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} An SOS GetObservation request XML string. │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - write: function(options) { │ │ │ │ │ - var node = this.writeNode("sos:GetObservation", options); │ │ │ │ │ - node.setAttribute("xmlns:om", this.namespaces.om); │ │ │ │ │ - node.setAttribute("xmlns:ogc", this.namespaces.ogc); │ │ │ │ │ - this.setAttributeNS( │ │ │ │ │ - node, this.namespaces.xsi, │ │ │ │ │ - "xsi:schemaLocation", this.schemaLocation │ │ │ │ │ - ); │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [node]); │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerToTop(); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + "removelayer": this.handleMapEvents, │ │ │ │ │ + "changelayer": this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + activated = true; │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Turn off the handler. Returns false if the handler was already active. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - readers: { │ │ │ │ │ - "om": { │ │ │ │ │ - "ObservationCollection": function(node, obj) { │ │ │ │ │ - obj.id = this.getAttributeNS(node, this.namespaces.gml, "id"); │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "member": function(node, observationCollection) { │ │ │ │ │ - this.readChildNodes(node, observationCollection); │ │ │ │ │ - }, │ │ │ │ │ - "Measurement": function(node, observationCollection) { │ │ │ │ │ - var measurement = {}; │ │ │ │ │ - observationCollection.measurements.push(measurement); │ │ │ │ │ - this.readChildNodes(node, measurement); │ │ │ │ │ - }, │ │ │ │ │ - "Observation": function(node, observationCollection) { │ │ │ │ │ - var observation = {}; │ │ │ │ │ - observationCollection.observations.push(observation); │ │ │ │ │ - this.readChildNodes(node, observation); │ │ │ │ │ - }, │ │ │ │ │ - "samplingTime": function(node, measurement) { │ │ │ │ │ - var samplingTime = {}; │ │ │ │ │ - measurement.samplingTime = samplingTime; │ │ │ │ │ - this.readChildNodes(node, samplingTime); │ │ │ │ │ - }, │ │ │ │ │ - "observedProperty": function(node, measurement) { │ │ │ │ │ - measurement.observedProperty = │ │ │ │ │ - this.getAttributeNS(node, this.namespaces.xlink, "href"); │ │ │ │ │ - this.readChildNodes(node, measurement); │ │ │ │ │ - }, │ │ │ │ │ - "procedure": function(node, measurement) { │ │ │ │ │ - measurement.procedure = │ │ │ │ │ - this.getAttributeNS(node, this.namespaces.xlink, "href"); │ │ │ │ │ - this.readChildNodes(node, measurement); │ │ │ │ │ - }, │ │ │ │ │ - "featureOfInterest": function(node, observation) { │ │ │ │ │ - var foi = { │ │ │ │ │ - features: [] │ │ │ │ │ - }; │ │ │ │ │ - observation.fois = []; │ │ │ │ │ - observation.fois.push(foi); │ │ │ │ │ - this.readChildNodes(node, foi); │ │ │ │ │ - // postprocessing to get actual features │ │ │ │ │ - var features = []; │ │ │ │ │ - for (var i = 0, len = foi.features.length; i < len; i++) { │ │ │ │ │ - var feature = foi.features[i]; │ │ │ │ │ - features.push(new OpenLayers.Feature.Vector( │ │ │ │ │ - feature.components[0], feature.attributes)); │ │ │ │ │ - } │ │ │ │ │ - foi.features = features; │ │ │ │ │ - }, │ │ │ │ │ - "result": function(node, measurement) { │ │ │ │ │ - var result = {}; │ │ │ │ │ - measurement.result = result; │ │ │ │ │ - if (this.getChildValue(node) !== '') { │ │ │ │ │ - result.value = this.getChildValue(node); │ │ │ │ │ - result.uom = node.getAttribute("uom"); │ │ │ │ │ - } else { │ │ │ │ │ - this.readChildNodes(node, result); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "sa": OpenLayers.Format.SOSGetFeatureOfInterest.prototype.readers.sa, │ │ │ │ │ - "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "TimeInstant": function(node, samplingTime) { │ │ │ │ │ - var timeInstant = {}; │ │ │ │ │ - samplingTime.timeInstant = timeInstant; │ │ │ │ │ - this.readChildNodes(node, timeInstant); │ │ │ │ │ - }, │ │ │ │ │ - "timePosition": function(node, timeInstant) { │ │ │ │ │ - timeInstant.timePosition = this.getChildValue(node); │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.SOSGetFeatureOfInterest.prototype.readers.gml) │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerBack(); │ │ │ │ │ + this.feature = null; │ │ │ │ │ + this.lastFeature = null; │ │ │ │ │ + this.down = null; │ │ │ │ │ + this.up = null; │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + "removelayer": this.handleMapEvents, │ │ │ │ │ + "changelayer": this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + deactivated = true; │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ + * Method: handleMapEvents │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} │ │ │ │ │ */ │ │ │ │ │ - writers: { │ │ │ │ │ - "sos": { │ │ │ │ │ - "GetObservation": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("GetObservation", { │ │ │ │ │ - attributes: { │ │ │ │ │ - version: this.VERSION, │ │ │ │ │ - service: 'SOS' │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - this.writeNode("offering", options, node); │ │ │ │ │ - if (options.eventTime) { │ │ │ │ │ - this.writeNode("eventTime", options, node); │ │ │ │ │ - } │ │ │ │ │ - for (var procedure in options.procedures) { │ │ │ │ │ - this.writeNode("procedure", options.procedures[procedure], node); │ │ │ │ │ - } │ │ │ │ │ - for (var observedProperty in options.observedProperties) { │ │ │ │ │ - this.writeNode("observedProperty", options.observedProperties[observedProperty], node); │ │ │ │ │ - } │ │ │ │ │ - if (options.foi) { │ │ │ │ │ - this.writeNode("featureOfInterest", options.foi, node); │ │ │ │ │ - } │ │ │ │ │ - this.writeNode("responseFormat", options, node); │ │ │ │ │ - if (options.resultModel) { │ │ │ │ │ - this.writeNode("resultModel", options, node); │ │ │ │ │ - } │ │ │ │ │ - if (options.responseMode) { │ │ │ │ │ - this.writeNode("responseMode", options, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "featureOfInterest": function(foi) { │ │ │ │ │ - var node = this.createElementNSPlus("featureOfInterest"); │ │ │ │ │ - this.writeNode("ObjectID", foi.objectId, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "ObjectID": function(options) { │ │ │ │ │ - return this.createElementNSPlus("ObjectID", { │ │ │ │ │ - value: options │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "responseFormat": function(options) { │ │ │ │ │ - return this.createElementNSPlus("responseFormat", { │ │ │ │ │ - value: options.responseFormat │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "procedure": function(procedure) { │ │ │ │ │ - return this.createElementNSPlus("procedure", { │ │ │ │ │ - value: procedure │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "offering": function(options) { │ │ │ │ │ - return this.createElementNSPlus("offering", { │ │ │ │ │ - value: options.offering │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "observedProperty": function(observedProperty) { │ │ │ │ │ - return this.createElementNSPlus("observedProperty", { │ │ │ │ │ - value: observedProperty │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "eventTime": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("eventTime"); │ │ │ │ │ - if (options.eventTime === 'latest') { │ │ │ │ │ - this.writeNode("ogc:TM_Equals", options, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "resultModel": function(options) { │ │ │ │ │ - return this.createElementNSPlus("resultModel", { │ │ │ │ │ - value: options.resultModel │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "responseMode": function(options) { │ │ │ │ │ - return this.createElementNSPlus("responseMode", { │ │ │ │ │ - value: options.responseMode │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "ogc": { │ │ │ │ │ - "TM_Equals": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:TM_Equals"); │ │ │ │ │ - this.writeNode("ogc:PropertyName", { │ │ │ │ │ - property: "urn:ogc:data:time:iso8601" │ │ │ │ │ - }, node); │ │ │ │ │ - if (options.eventTime === 'latest') { │ │ │ │ │ - this.writeNode("gml:TimeInstant", { │ │ │ │ │ - value: 'latest' │ │ │ │ │ - }, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "PropertyName": function(options) { │ │ │ │ │ - return this.createElementNSPlus("ogc:PropertyName", { │ │ │ │ │ - value: options.property │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "gml": { │ │ │ │ │ - "TimeInstant": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:TimeInstant"); │ │ │ │ │ - this.writeNode("gml:timePosition", options, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "timePosition": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:timePosition", { │ │ │ │ │ - value: options.value │ │ │ │ │ - }); │ │ │ │ │ - return node; │ │ │ │ │ - } │ │ │ │ │ + handleMapEvents: function(evt) { │ │ │ │ │ + if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ + this.moveLayerToTop(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SOSGetObservation" │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveLayerToTop │ │ │ │ │ + * Moves the layer for this handler to the top, so mouse events can reach │ │ │ │ │ + * it. │ │ │ │ │ + */ │ │ │ │ │ + moveLayerToTop: function() { │ │ │ │ │ + var index = Math.max(this.map.Z_INDEX_BASE['Feature'] - 1, │ │ │ │ │ + this.layer.getZIndex()) + 1; │ │ │ │ │ + this.layer.setZIndex(index); │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveLayerBack │ │ │ │ │ + * Moves the layer back to the position determined by the map's layers │ │ │ │ │ + * array. │ │ │ │ │ + */ │ │ │ │ │ + moveLayerBack: function() { │ │ │ │ │ + var index = this.layer.getZIndex() - 1; │ │ │ │ │ + if (index >= this.map.Z_INDEX_BASE['Feature']) { │ │ │ │ │ + this.layer.setZIndex(index); │ │ │ │ │ + } else { │ │ │ │ │ + this.map.setLayerZIndex(this.layer, │ │ │ │ │ + this.map.getLayerIndex(this.layer)); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Feature" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/Context.js │ │ │ │ │ + OpenLayers/Handler/Keyboard.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.Context │ │ │ │ │ - * Base class for both Format.WMC and Format.OWSContext │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.handler.Keyboard │ │ │ │ │ + * A handler for keyboard events. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Handler.Keyboard> constructor. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + * - <OpenLayers.Handler> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.Context = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ +OpenLayers.Handler.Keyboard = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: layerOptions │ │ │ │ │ - * {Object} Default options for layers created by the parser. These │ │ │ │ │ - * options are overridden by the options which are read from the │ │ │ │ │ - * capabilities document. │ │ │ │ │ + /* http://www.quirksmode.org/js/keys.html explains key x-browser │ │ │ │ │ + key handling quirks in pretty nice detail */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: KEY_EVENTS │ │ │ │ │ + * keydown, keypress, keyup │ │ │ │ │ */ │ │ │ │ │ - layerOptions: null, │ │ │ │ │ + KEY_EVENTS: ["keydown", "keyup"], │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: layerParams │ │ │ │ │ - * {Object} Default parameters for layers created by the parser. This │ │ │ │ │ - * can be used e.g. to override DEFAULT_PARAMS for │ │ │ │ │ - * OpenLayers.Layer.WMS. │ │ │ │ │ + /** │ │ │ │ │ + * Property: eventListener │ │ │ │ │ + * {Function} │ │ │ │ │ */ │ │ │ │ │ - layerParams: null, │ │ │ │ │ + eventListener: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.Context │ │ │ │ │ - * Create a new parser for Context documents. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * Property: observeElement │ │ │ │ │ + * {DOMElement|String} The DOM element on which we listen for │ │ │ │ │ + * key events. Default to the document. │ │ │ │ │ */ │ │ │ │ │ + observeElement: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read Context data from a string, and return an object with map │ │ │ │ │ - * properties and a list of layers. │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Handler.Keyboard │ │ │ │ │ + * Returns a new keyboard handler. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * options - {Object} The options object must contain a map property. If │ │ │ │ │ - * the map property is a string, it must be the id of a dom element │ │ │ │ │ - * where the new map will be placed. If the map property is an │ │ │ │ │ - * <OpenLayers.Map>, the layers from the context document will be added │ │ │ │ │ - * to the map. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Map>} A map based on the context. │ │ │ │ │ + * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ + * this handler. If a handler is being used without a control, the │ │ │ │ │ + * handlers setMap method must be overridden to deal properly with │ │ │ │ │ + * the map. │ │ │ │ │ + * callbacks - {Object} An object containing a single function to be │ │ │ │ │ + * called when the drag operation is finished. The callback should │ │ │ │ │ + * expect to recieve a single argument, the pixel location of the event. │ │ │ │ │ + * Callbacks for 'keydown', 'keypress', and 'keyup' are supported. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * handler. │ │ │ │ │ */ │ │ │ │ │ - read: function(data, options) { │ │ │ │ │ - var context = OpenLayers.Format.XML.VersionedOGC.prototype.read.apply(this, │ │ │ │ │ - arguments); │ │ │ │ │ - var map; │ │ │ │ │ - if (options && options.map) { │ │ │ │ │ - this.context = context; │ │ │ │ │ - if (options.map instanceof OpenLayers.Map) { │ │ │ │ │ - map = this.mergeContextToMap(context, options.map); │ │ │ │ │ - } else { │ │ │ │ │ - var mapOptions = options.map; │ │ │ │ │ - if (OpenLayers.Util.isElement(mapOptions) || │ │ │ │ │ - typeof mapOptions == "string") { │ │ │ │ │ - // we assume mapOptions references a div │ │ │ │ │ - // element │ │ │ │ │ - mapOptions = { │ │ │ │ │ - div: mapOptions │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - map = this.contextToMap(context, mapOptions); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - // not documented as part of the API, provided as a non-API option │ │ │ │ │ - map = context; │ │ │ │ │ - } │ │ │ │ │ - return map; │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + // cache the bound event listener method so it can be unobserved later │ │ │ │ │ + this.eventListener = OpenLayers.Function.bindAsEventListener( │ │ │ │ │ + this.handleKeyEvent, this │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getLayerFromContext │ │ │ │ │ - * Create a WMS layer from a layerContext object. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layerContext - {Object} An object representing a WMS layer. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.WMS>} A WMS layer. │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ - getLayerFromContext: function(layerContext) { │ │ │ │ │ - var i, len; │ │ │ │ │ - // fill initial options object from layerContext │ │ │ │ │ - var options = { │ │ │ │ │ - queryable: layerContext.queryable, //keep queryable for api compatibility │ │ │ │ │ - visibility: layerContext.visibility, │ │ │ │ │ - maxExtent: layerContext.maxExtent, │ │ │ │ │ - metadata: OpenLayers.Util.applyDefaults(layerContext.metadata, { │ │ │ │ │ - styles: layerContext.styles, │ │ │ │ │ - formats: layerContext.formats, │ │ │ │ │ - "abstract": layerContext["abstract"], │ │ │ │ │ - dataURL: layerContext.dataURL │ │ │ │ │ - }), │ │ │ │ │ - numZoomLevels: layerContext.numZoomLevels, │ │ │ │ │ - units: layerContext.units, │ │ │ │ │ - isBaseLayer: layerContext.isBaseLayer, │ │ │ │ │ - opacity: layerContext.opacity, │ │ │ │ │ - displayInLayerSwitcher: layerContext.displayInLayerSwitcher, │ │ │ │ │ - singleTile: layerContext.singleTile, │ │ │ │ │ - tileSize: (layerContext.tileSize) ? │ │ │ │ │ - new OpenLayers.Size( │ │ │ │ │ - layerContext.tileSize.width, │ │ │ │ │ - layerContext.tileSize.height │ │ │ │ │ - ) : undefined, │ │ │ │ │ - minScale: layerContext.minScale || layerContext.maxScaleDenominator, │ │ │ │ │ - maxScale: layerContext.maxScale || layerContext.minScaleDenominator, │ │ │ │ │ - srs: layerContext.srs, │ │ │ │ │ - dimensions: layerContext.dimensions, │ │ │ │ │ - metadataURL: layerContext.metadataURL │ │ │ │ │ - }; │ │ │ │ │ - if (this.layerOptions) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.layerOptions); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var params = { │ │ │ │ │ - layers: layerContext.name, │ │ │ │ │ - transparent: layerContext.transparent, │ │ │ │ │ - version: layerContext.version │ │ │ │ │ - }; │ │ │ │ │ - if (layerContext.formats && layerContext.formats.length > 0) { │ │ │ │ │ - // set default value for params if current attribute is not positionned │ │ │ │ │ - params.format = layerContext.formats[0].value; │ │ │ │ │ - for (i = 0, len = layerContext.formats.length; i < len; i++) { │ │ │ │ │ - var format = layerContext.formats[i]; │ │ │ │ │ - if (format.current == true) { │ │ │ │ │ - params.format = format.value; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (layerContext.styles && layerContext.styles.length > 0) { │ │ │ │ │ - for (i = 0, len = layerContext.styles.length; i < len; i++) { │ │ │ │ │ - var style = layerContext.styles[i]; │ │ │ │ │ - if (style.current == true) { │ │ │ │ │ - // three style types to consider │ │ │ │ │ - // 1) linked SLD │ │ │ │ │ - // 2) inline SLD │ │ │ │ │ - // 3) named style │ │ │ │ │ - if (style.href) { │ │ │ │ │ - params.sld = style.href; │ │ │ │ │ - } else if (style.body) { │ │ │ │ │ - params.sld_body = style.body; │ │ │ │ │ - } else { │ │ │ │ │ - params.styles = style.name; │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.layerParams) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(params, this.layerParams); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var layer = null; │ │ │ │ │ - var service = layerContext.service; │ │ │ │ │ - if (service == OpenLayers.Format.Context.serviceTypes.WFS) { │ │ │ │ │ - options.strategies = [new OpenLayers.Strategy.BBOX()]; │ │ │ │ │ - options.protocol = new OpenLayers.Protocol.WFS({ │ │ │ │ │ - url: layerContext.url, │ │ │ │ │ - // since we do not know featureNS, let the protocol │ │ │ │ │ - // determine it automagically using featurePrefix │ │ │ │ │ - featurePrefix: layerContext.name.split(":")[0], │ │ │ │ │ - featureType: layerContext.name.split(":").pop() │ │ │ │ │ - }); │ │ │ │ │ - layer = new OpenLayers.Layer.Vector( │ │ │ │ │ - layerContext.title || layerContext.name, │ │ │ │ │ - options │ │ │ │ │ - ); │ │ │ │ │ - } else if (service == OpenLayers.Format.Context.serviceTypes.KML) { │ │ │ │ │ - // use a vector layer with an HTTP Protcol and a Fixed strategy │ │ │ │ │ - options.strategies = [new OpenLayers.Strategy.Fixed()]; │ │ │ │ │ - options.protocol = new OpenLayers.Protocol.HTTP({ │ │ │ │ │ - url: layerContext.url, │ │ │ │ │ - format: new OpenLayers.Format.KML() │ │ │ │ │ - }); │ │ │ │ │ - layer = new OpenLayers.Layer.Vector( │ │ │ │ │ - layerContext.title || layerContext.name, │ │ │ │ │ - options │ │ │ │ │ - ); │ │ │ │ │ - } else if (service == OpenLayers.Format.Context.serviceTypes.GML) { │ │ │ │ │ - // use a vector layer with a HTTP Protocol and a Fixed strategy │ │ │ │ │ - options.strategies = [new OpenLayers.Strategy.Fixed()]; │ │ │ │ │ - options.protocol = new OpenLayers.Protocol.HTTP({ │ │ │ │ │ - url: layerContext.url, │ │ │ │ │ - format: new OpenLayers.Format.GML() │ │ │ │ │ - }); │ │ │ │ │ - layer = new OpenLayers.Layer.Vector( │ │ │ │ │ - layerContext.title || layerContext.name, │ │ │ │ │ - options │ │ │ │ │ - ); │ │ │ │ │ - } else if (layerContext.features) { │ │ │ │ │ - // inline GML or KML features │ │ │ │ │ - layer = new OpenLayers.Layer.Vector( │ │ │ │ │ - layerContext.title || layerContext.name, │ │ │ │ │ - options │ │ │ │ │ - ); │ │ │ │ │ - layer.addFeatures(layerContext.features); │ │ │ │ │ - } else if (layerContext.categoryLayer !== true) { │ │ │ │ │ - layer = new OpenLayers.Layer.WMS( │ │ │ │ │ - layerContext.title || layerContext.name, │ │ │ │ │ - layerContext.url, │ │ │ │ │ - params, │ │ │ │ │ - options │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return layer; │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.eventListener = null; │ │ │ │ │ + OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getLayersFromContext │ │ │ │ │ - * Create an array of layers from an array of layerContext objects. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layersContext - {Array(Object)} An array of objects representing layers. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Layer>)} An array of layers. │ │ │ │ │ + * Method: activate │ │ │ │ │ */ │ │ │ │ │ - getLayersFromContext: function(layersContext) { │ │ │ │ │ - var layers = []; │ │ │ │ │ - for (var i = 0, len = layersContext.length; i < len; i++) { │ │ │ │ │ - var layer = this.getLayerFromContext(layersContext[i]); │ │ │ │ │ - if (layer !== null) { │ │ │ │ │ - layers.push(layer); │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.observeElement = this.observeElement || document; │ │ │ │ │ + for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ + OpenLayers.Event.observe( │ │ │ │ │ + this.observeElement, this.KEY_EVENTS[i], this.eventListener); │ │ │ │ │ } │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - return layers; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: contextToMap │ │ │ │ │ - * Create a map given a context object. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * context - {Object} The context object. │ │ │ │ │ - * options - {Object} Default map options. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Map>} A map based on the context object. │ │ │ │ │ + * Method: deactivate │ │ │ │ │ */ │ │ │ │ │ - contextToMap: function(context, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - maxExtent: context.maxExtent, │ │ │ │ │ - projection: context.projection, │ │ │ │ │ - units: context.units │ │ │ │ │ - }, options); │ │ │ │ │ - │ │ │ │ │ - if (options.maxExtent) { │ │ │ │ │ - options.maxResolution = │ │ │ │ │ - options.maxExtent.getWidth() / OpenLayers.Map.TILE_WIDTH; │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ + OpenLayers.Event.stopObserving( │ │ │ │ │ + this.observeElement, this.KEY_EVENTS[i], this.eventListener); │ │ │ │ │ + } │ │ │ │ │ + deactivated = true; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var metadata = { │ │ │ │ │ - contactInformation: context.contactInformation, │ │ │ │ │ - "abstract": context["abstract"], │ │ │ │ │ - keywords: context.keywords, │ │ │ │ │ - logo: context.logo, │ │ │ │ │ - descriptionURL: context.descriptionURL │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - options.metadata = metadata; │ │ │ │ │ - │ │ │ │ │ - var map = new OpenLayers.Map(options); │ │ │ │ │ - map.addLayers(this.getLayersFromContext(context.layersContext)); │ │ │ │ │ - map.setCenter( │ │ │ │ │ - context.bounds.getCenterLonLat(), │ │ │ │ │ - map.getZoomForExtent(context.bounds, true) │ │ │ │ │ - ); │ │ │ │ │ - return map; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: mergeContextToMap │ │ │ │ │ - * Add layers from a context object to a map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * context - {Object} The context object. │ │ │ │ │ - * map - {<OpenLayers.Map>} The map. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Map>} The same map with layers added. │ │ │ │ │ - */ │ │ │ │ │ - mergeContextToMap: function(context, map) { │ │ │ │ │ - map.addLayers(this.getLayersFromContext(context.layersContext)); │ │ │ │ │ - return map; │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Write a context document given a map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {<OpenLayers.Map> | Object} A map or context object. │ │ │ │ │ - * options - {Object} Optional configuration object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A context document string. │ │ │ │ │ + * Method: handleKeyEvent │ │ │ │ │ */ │ │ │ │ │ - write: function(obj, options) { │ │ │ │ │ - obj = this.toContext(obj); │ │ │ │ │ - return OpenLayers.Format.XML.VersionedOGC.prototype.write.apply(this, │ │ │ │ │ - arguments); │ │ │ │ │ + handleKeyEvent: function(evt) { │ │ │ │ │ + if (this.checkModifiers(evt)) { │ │ │ │ │ + this.callback(evt.type, [evt]); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.Context" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Keyboard" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Format.Context.serviceTypes │ │ │ │ │ - * Enumeration for service types │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.Context.serviceTypes = { │ │ │ │ │ - "WMS": "urn:ogc:serviceType:WMS", │ │ │ │ │ - "WFS": "urn:ogc:serviceType:WFS", │ │ │ │ │ - "WCS": "urn:ogc:serviceType:WCS", │ │ │ │ │ - "GML": "urn:ogc:serviceType:GML", │ │ │ │ │ - "SLD": "urn:ogc:serviceType:SLD", │ │ │ │ │ - "FES": "urn:ogc:serviceType:FES", │ │ │ │ │ - "KML": "urn:ogc:serviceType:KML" │ │ │ │ │ -}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMC.js │ │ │ │ │ + OpenLayers/Renderer/Canvas.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/Context.js │ │ │ │ │ + * @requires OpenLayers/Renderer.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WMC │ │ │ │ │ - * Read and write Web Map Context documents. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.Context> │ │ │ │ │ + * Class: OpenLayers.Renderer.Canvas │ │ │ │ │ + * A renderer based on the 2D 'canvas' drawing element. │ │ │ │ │ + * │ │ │ │ │ + * Inherits: │ │ │ │ │ + * - <OpenLayers.Renderer> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WMC = OpenLayers.Class(OpenLayers.Format.Context, { │ │ │ │ │ +OpenLayers.Renderer.Canvas = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.1.0". │ │ │ │ │ + * APIProperty: hitDetection │ │ │ │ │ + * {Boolean} Allow for hit detection of features. Default is true. │ │ │ │ │ */ │ │ │ │ │ - defaultVersion: "1.1.0", │ │ │ │ │ + hitDetection: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMC │ │ │ │ │ - * Create a new parser for Web Map Context documents. │ │ │ │ │ + * Property: hitOverflow │ │ │ │ │ + * {Number} The method for converting feature identifiers to color values │ │ │ │ │ + * supports 16777215 sequential values. Two features cannot be │ │ │ │ │ + * predictably detected if their identifiers differ by more than this │ │ │ │ │ + * value. The hitOverflow allows for bigger numbers (but the │ │ │ │ │ + * difference in values is still limited). │ │ │ │ │ + */ │ │ │ │ │ + hitOverflow: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: canvas │ │ │ │ │ + * {Canvas} The canvas context object. │ │ │ │ │ + */ │ │ │ │ │ + canvas: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Object} Internal object of feature/style pairs for use in redrawing the layer. │ │ │ │ │ + */ │ │ │ │ │ + features: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: pendingRedraw │ │ │ │ │ + * {Boolean} The renderer needs a redraw call to render features added while │ │ │ │ │ + * the renderer was locked. │ │ │ │ │ + */ │ │ │ │ │ + pendingRedraw: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: cachedSymbolBounds │ │ │ │ │ + * {Object} Internal cache of calculated symbol extents. │ │ │ │ │ + */ │ │ │ │ │ + cachedSymbolBounds: {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Renderer.Canvas │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * containerID - {<String>} │ │ │ │ │ + * options - {Object} Optional properties to be set on the renderer. │ │ │ │ │ */ │ │ │ │ │ + initialize: function(containerID, options) { │ │ │ │ │ + OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.root = document.createElement("canvas"); │ │ │ │ │ + this.container.appendChild(this.root); │ │ │ │ │ + this.canvas = this.root.getContext("2d"); │ │ │ │ │ + this.features = {}; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitCanvas = document.createElement("canvas"); │ │ │ │ │ + this.hitContext = this.hitCanvas.getContext("2d"); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: layerToContext │ │ │ │ │ - * Create a layer context object given a wms layer object. │ │ │ │ │ + * Method: setExtent │ │ │ │ │ + * Set the visible part of the layer. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.WMS>} The layer. │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} │ │ │ │ │ + * resolutionChanged - {Boolean} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} A layer context object. │ │ │ │ │ + * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ + * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ + * False otherwise. │ │ │ │ │ */ │ │ │ │ │ - layerToContext: function(layer) { │ │ │ │ │ - var parser = this.getParser(); │ │ │ │ │ - var layerContext = { │ │ │ │ │ - queryable: layer.queryable, │ │ │ │ │ - visibility: layer.visibility, │ │ │ │ │ - name: layer.params["LAYERS"], │ │ │ │ │ - title: layer.name, │ │ │ │ │ - "abstract": layer.metadata["abstract"], │ │ │ │ │ - dataURL: layer.metadata.dataURL, │ │ │ │ │ - metadataURL: layer.metadataURL, │ │ │ │ │ - server: { │ │ │ │ │ - version: layer.params["VERSION"], │ │ │ │ │ - url: layer.url │ │ │ │ │ - }, │ │ │ │ │ - maxExtent: layer.maxExtent, │ │ │ │ │ - transparent: layer.params["TRANSPARENT"], │ │ │ │ │ - numZoomLevels: layer.numZoomLevels, │ │ │ │ │ - units: layer.units, │ │ │ │ │ - isBaseLayer: layer.isBaseLayer, │ │ │ │ │ - opacity: layer.opacity == 1 ? undefined : layer.opacity, │ │ │ │ │ - displayInLayerSwitcher: layer.displayInLayerSwitcher, │ │ │ │ │ - singleTile: layer.singleTile, │ │ │ │ │ - tileSize: (layer.singleTile || !layer.tileSize) ? │ │ │ │ │ - undefined : { │ │ │ │ │ - width: layer.tileSize.w, │ │ │ │ │ - height: layer.tileSize.h │ │ │ │ │ - }, │ │ │ │ │ - minScale: (layer.options.resolutions || │ │ │ │ │ - layer.options.scales || │ │ │ │ │ - layer.options.maxResolution || │ │ │ │ │ - layer.options.minScale) ? │ │ │ │ │ - layer.minScale : undefined, │ │ │ │ │ - maxScale: (layer.options.resolutions || │ │ │ │ │ - layer.options.scales || │ │ │ │ │ - layer.options.minResolution || │ │ │ │ │ - layer.options.maxScale) ? │ │ │ │ │ - layer.maxScale : undefined, │ │ │ │ │ - formats: [], │ │ │ │ │ - styles: [], │ │ │ │ │ - srs: layer.srs, │ │ │ │ │ - dimensions: layer.dimensions │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ + setExtent: function() { │ │ │ │ │ + OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + // always redraw features │ │ │ │ │ + return false; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (layer.metadata.servertitle) { │ │ │ │ │ - layerContext.server.title = layer.metadata.servertitle; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: eraseGeometry │ │ │ │ │ + * Erase a geometry from the renderer. Because the Canvas renderer has │ │ │ │ │ + * 'memory' of the features that it has drawn, we have to remove the │ │ │ │ │ + * feature so it doesn't redraw. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + eraseGeometry: function(geometry, featureId) { │ │ │ │ │ + this.eraseFeatures(this.features[featureId][0]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (layer.metadata.formats && layer.metadata.formats.length > 0) { │ │ │ │ │ - for (var i = 0, len = layer.metadata.formats.length; i < len; i++) { │ │ │ │ │ - var format = layer.metadata.formats[i]; │ │ │ │ │ - layerContext.formats.push({ │ │ │ │ │ - value: format.value, │ │ │ │ │ - current: (format.value == layer.params["FORMAT"]) │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - layerContext.formats.push({ │ │ │ │ │ - value: layer.params["FORMAT"], │ │ │ │ │ - current: true │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: supported │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the browser supports the renderer class │ │ │ │ │ + */ │ │ │ │ │ + supported: function() { │ │ │ │ │ + return OpenLayers.CANVAS_SUPPORTED; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (layer.metadata.styles && layer.metadata.styles.length > 0) { │ │ │ │ │ - for (var i = 0, len = layer.metadata.styles.length; i < len; i++) { │ │ │ │ │ - var style = layer.metadata.styles[i]; │ │ │ │ │ - if ((style.href == layer.params["SLD"]) || │ │ │ │ │ - (style.body == layer.params["SLD_BODY"]) || │ │ │ │ │ - (style.name == layer.params["STYLES"])) { │ │ │ │ │ - style.current = true; │ │ │ │ │ - } else { │ │ │ │ │ - style.current = false; │ │ │ │ │ - } │ │ │ │ │ - layerContext.styles.push(style); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - layerContext.styles.push({ │ │ │ │ │ - href: layer.params["SLD"], │ │ │ │ │ - body: layer.params["SLD_BODY"], │ │ │ │ │ - name: layer.params["STYLES"] || parser.defaultStyleName, │ │ │ │ │ - title: parser.defaultStyleTitle, │ │ │ │ │ - current: true │ │ │ │ │ - }); │ │ │ │ │ + /** │ │ │ │ │ + * Method: setSize │ │ │ │ │ + * Sets the size of the drawing surface. │ │ │ │ │ + * │ │ │ │ │ + * Once the size is updated, redraw the canvas. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * size - {<OpenLayers.Size>} │ │ │ │ │ + */ │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + this.size = size.clone(); │ │ │ │ │ + var root = this.root; │ │ │ │ │ + root.style.width = size.w + "px"; │ │ │ │ │ + root.style.height = size.h + "px"; │ │ │ │ │ + root.width = size.w; │ │ │ │ │ + root.height = size.h; │ │ │ │ │ + this.resolution = null; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + var hitCanvas = this.hitCanvas; │ │ │ │ │ + hitCanvas.style.width = size.w + "px"; │ │ │ │ │ + hitCanvas.style.height = size.h + "px"; │ │ │ │ │ + hitCanvas.width = size.w; │ │ │ │ │ + hitCanvas.height = size.h; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return layerContext; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: toContext │ │ │ │ │ - * Create a context object free from layer given a map or a │ │ │ │ │ - * context object. │ │ │ │ │ + * Method: drawFeature │ │ │ │ │ + * Draw the feature. Stores the feature in the features list, │ │ │ │ │ + * then redraws the layer. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {<OpenLayers.Map> | Object} The map or context. │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * style - {<Object>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} A context object. │ │ │ │ │ + * {Boolean} The feature has been drawn completely. If the feature has no │ │ │ │ │ + * geometry, undefined will be returned. If the feature is not rendered │ │ │ │ │ + * for other reasons, false will be returned. │ │ │ │ │ */ │ │ │ │ │ - toContext: function(obj) { │ │ │ │ │ - var context = {}; │ │ │ │ │ - var layers = obj.layers; │ │ │ │ │ - if (obj.CLASS_NAME == "OpenLayers.Map") { │ │ │ │ │ - var metadata = obj.metadata || {}; │ │ │ │ │ - context.size = obj.getSize(); │ │ │ │ │ - context.bounds = obj.getExtent(); │ │ │ │ │ - context.projection = obj.projection; │ │ │ │ │ - context.title = obj.title; │ │ │ │ │ - context.keywords = metadata.keywords; │ │ │ │ │ - context["abstract"] = metadata["abstract"]; │ │ │ │ │ - context.logo = metadata.logo; │ │ │ │ │ - context.descriptionURL = metadata.descriptionURL; │ │ │ │ │ - context.contactInformation = metadata.contactInformation; │ │ │ │ │ - context.maxExtent = obj.maxExtent; │ │ │ │ │ - } else { │ │ │ │ │ - // copy all obj properties except the "layers" property │ │ │ │ │ - OpenLayers.Util.applyDefaults(context, obj); │ │ │ │ │ - if (context.layers != undefined) { │ │ │ │ │ - delete(context.layers); │ │ │ │ │ + drawFeature: function(feature, style) { │ │ │ │ │ + var rendered; │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + style = this.applyDefaultSymbolizer(style || feature.style); │ │ │ │ │ + // don't render if display none or feature outside extent │ │ │ │ │ + var bounds = feature.geometry.getBounds(); │ │ │ │ │ + │ │ │ │ │ + var worldBounds; │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + worldBounds = this.map.getMaxExtent(); │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - if (context.layersContext == undefined) { │ │ │ │ │ - context.layersContext = []; │ │ │ │ │ + var intersects = bounds && bounds.intersectsBounds(this.extent, { │ │ │ │ │ + worldBounds: worldBounds │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + rendered = (style.display !== "none") && !!bounds && intersects; │ │ │ │ │ + if (rendered) { │ │ │ │ │ + // keep track of what we have rendered for redraw │ │ │ │ │ + this.features[feature.id] = [feature, style]; │ │ │ │ │ + } else { │ │ │ │ │ + // remove from features tracked for redraw │ │ │ │ │ + delete(this.features[feature.id]); │ │ │ │ │ + } │ │ │ │ │ + this.pendingRedraw = true; │ │ │ │ │ + } │ │ │ │ │ + if (this.pendingRedraw && !this.locked) { │ │ │ │ │ + this.redraw(); │ │ │ │ │ + this.pendingRedraw = false; │ │ │ │ │ } │ │ │ │ │ + return rendered; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // let's convert layers into layersContext object (if any) │ │ │ │ │ - if (layers != undefined && OpenLayers.Util.isArray(layers)) { │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - var layer = layers[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.WMS) { │ │ │ │ │ - context.layersContext.push(this.layerToContext(layer)); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawGeometry │ │ │ │ │ + * Used when looping (in redraw) over the features; draws │ │ │ │ │ + * the canvas. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + */ │ │ │ │ │ + drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ + var className = geometry.CLASS_NAME; │ │ │ │ │ + if ((className == "OpenLayers.Geometry.Collection") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ + (className == "OpenLayers.Geometry.MultiPolygon")) { │ │ │ │ │ + for (var i = 0; i < geometry.components.length; i++) { │ │ │ │ │ + this.drawGeometry(geometry.components[i], style, featureId); │ │ │ │ │ } │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + this.drawPoint(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + this.drawLineString(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + this.drawPolygon(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ - return context; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMC" │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawExternalGraphic │ │ │ │ │ + * Called to draw External graphics. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + drawExternalGraphic: function(geometry, style, featureId) { │ │ │ │ │ + var img = new Image(); │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/OSM.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + img.title = title; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ + var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ + style.graphicXOffset : -(0.5 * width); │ │ │ │ │ + var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ + style.graphicYOffset : -(0.5 * height); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ - * @requires OpenLayers/Projection.js │ │ │ │ │ - */ │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.OSM │ │ │ │ │ - * OSM parser. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Format.OSM> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.OSM = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + var onLoad = function() { │ │ │ │ │ + if (!this.features[featureId]) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ + if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ + var x = (p0 + xOffset) | 0; │ │ │ │ │ + var y = (p1 + yOffset) | 0; │ │ │ │ │ + var canvas = this.canvas; │ │ │ │ │ + canvas.globalAlpha = opacity; │ │ │ │ │ + var factor = OpenLayers.Renderer.Canvas.drawImageScaleFactor || │ │ │ │ │ + (OpenLayers.Renderer.Canvas.drawImageScaleFactor = │ │ │ │ │ + /android 2.1/.test(navigator.userAgent.toLowerCase()) ? │ │ │ │ │ + // 320 is the screen width of the G1 phone, for │ │ │ │ │ + // which drawImage works out of the box. │ │ │ │ │ + 320 / window.screen.width : 1 │ │ │ │ │ + ); │ │ │ │ │ + canvas.drawImage( │ │ │ │ │ + img, x * factor, y * factor, width * factor, height * factor │ │ │ │ │ + ); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId); │ │ │ │ │ + this.hitContext.fillRect(x, y, width, height); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: checkTags │ │ │ │ │ - * {Boolean} Should tags be checked to determine whether something │ │ │ │ │ - * should be treated as a seperate node. Will slow down parsing. │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - checkTags: false, │ │ │ │ │ + img.onload = OpenLayers.Function.bind(onLoad, this); │ │ │ │ │ + img.src = style.externalGraphic; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: interestingTagsExclude │ │ │ │ │ - * {Array} List of tags to exclude from 'interesting' checks on nodes. │ │ │ │ │ - * Must be set when creating the format. Will only be used if checkTags │ │ │ │ │ - * is set. │ │ │ │ │ + * Method: drawNamedSymbol │ │ │ │ │ + * Called to draw Well Known Graphic Symbol Name. │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ */ │ │ │ │ │ - interestingTagsExclude: null, │ │ │ │ │ + drawNamedSymbol: function(geometry, style, featureId) { │ │ │ │ │ + var x, y, cx, cy, i, symbolBounds, scaling, angle; │ │ │ │ │ + var unscaledStrokeWidth; │ │ │ │ │ + var deg2rad = Math.PI / 180.0; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: areaTags │ │ │ │ │ - * {Array} List of tags indicating that something is an area. │ │ │ │ │ - * Must be set when creating the format. Will only be used if │ │ │ │ │ - * checkTags is true. │ │ │ │ │ - */ │ │ │ │ │ - areaTags: null, │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[style.graphicName]; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.OSM │ │ │ │ │ - * Create a new parser for OSM. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - var layer_defaults = { │ │ │ │ │ - 'interestingTagsExclude': ['source', 'source_ref', │ │ │ │ │ - 'source:ref', 'history', 'attribution', 'created_by' │ │ │ │ │ - ], │ │ │ │ │ - 'areaTags': ['area', 'building', 'leisure', 'tourism', 'ruins', │ │ │ │ │ - 'historic', 'landuse', 'military', 'natural', 'sport' │ │ │ │ │ - ] │ │ │ │ │ - }; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(style.graphicName + ' is not a valid symbol name'); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - layer_defaults = OpenLayers.Util.extend(layer_defaults, options); │ │ │ │ │ + if (!symbol.length || symbol.length < 2) return; │ │ │ │ │ │ │ │ │ │ - var interesting = {}; │ │ │ │ │ - for (var i = 0; i < layer_defaults.interestingTagsExclude.length; i++) { │ │ │ │ │ - interesting[layer_defaults.interestingTagsExclude[i]] = true; │ │ │ │ │ - } │ │ │ │ │ - layer_defaults.interestingTagsExclude = interesting; │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ │ │ │ │ │ - var area = {}; │ │ │ │ │ - for (var i = 0; i < layer_defaults.areaTags.length; i++) { │ │ │ │ │ - area[layer_defaults.areaTags[i]] = true; │ │ │ │ │ - } │ │ │ │ │ - layer_defaults.areaTags = area; │ │ │ │ │ + if (isNaN(p0) || isNaN(p1)) return; │ │ │ │ │ │ │ │ │ │ - // OSM coordinates are always in longlat WGS84 │ │ │ │ │ - this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + // Use rounded line caps │ │ │ │ │ + this.canvas.lineCap = "round"; │ │ │ │ │ + this.canvas.lineJoin = "round"; │ │ │ │ │ │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [layer_defaults]); │ │ │ │ │ - }, │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.lineCap = "round"; │ │ │ │ │ + this.hitContext.lineJoin = "round"; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Return a list of features from a OSM doc │ │ │ │ │ - │ │ │ │ │ - * Parameters: │ │ │ │ │ - * doc - {Element} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ - */ │ │ │ │ │ - read: function(doc) { │ │ │ │ │ - if (typeof doc == "string") { │ │ │ │ │ - doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]); │ │ │ │ │ + // Scale and rotate symbols, using precalculated bounds whenever possible. │ │ │ │ │ + if (style.graphicName in this.cachedSymbolBounds) { │ │ │ │ │ + symbolBounds = this.cachedSymbolBounds[style.graphicName]; │ │ │ │ │ + } else { │ │ │ │ │ + symbolBounds = new OpenLayers.Bounds(); │ │ │ │ │ + for (i = 0; i < symbol.length; i += 2) { │ │ │ │ │ + symbolBounds.extend(new OpenLayers.LonLat(symbol[i], symbol[i + 1])); │ │ │ │ │ + } │ │ │ │ │ + this.cachedSymbolBounds[style.graphicName] = symbolBounds; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - var nodes = this.getNodes(doc); │ │ │ │ │ - var ways = this.getWays(doc); │ │ │ │ │ + // Push symbol scaling, translation and rotation onto the transformation stack in reverse order. │ │ │ │ │ + // Don't forget to apply all canvas transformations to the hitContext canvas as well(!) │ │ │ │ │ + this.canvas.save(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.save(); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // Geoms will contain at least ways.length entries. │ │ │ │ │ - var feat_list = new Array(ways.length); │ │ │ │ │ + // Step 3: place symbol at the desired location │ │ │ │ │ + this.canvas.translate(p0, p1); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.translate(p0, p1); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - for (var i = 0; i < ways.length; i++) { │ │ │ │ │ - // We know the minimal of this one ahead of time. (Could be -1 │ │ │ │ │ - // due to areas/polygons) │ │ │ │ │ - var point_list = new Array(ways[i].nodes.length); │ │ │ │ │ + // Step 2a. rotate the symbol if necessary │ │ │ │ │ + angle = deg2rad * style.rotation; // will be NaN when style.rotation is undefined. │ │ │ │ │ + if (!isNaN(angle)) { │ │ │ │ │ + this.canvas.rotate(angle); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.rotate(angle); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var poly = this.isWayArea(ways[i]) ? 1 : 0; │ │ │ │ │ - for (var j = 0; j < ways[i].nodes.length; j++) { │ │ │ │ │ - var node = nodes[ways[i].nodes[j]]; │ │ │ │ │ + // // Step 2: scale symbol such that pointRadius equals half the maximum symbol dimension. │ │ │ │ │ + scaling = 2.0 * style.pointRadius / Math.max(symbolBounds.getWidth(), symbolBounds.getHeight()); │ │ │ │ │ + this.canvas.scale(scaling, scaling); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.scale(scaling, scaling); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var point = new OpenLayers.Geometry.Point(node.lon, node.lat); │ │ │ │ │ + // Step 1: center the symbol at the origin │ │ │ │ │ + cx = symbolBounds.getCenterLonLat().lon; │ │ │ │ │ + cy = symbolBounds.getCenterLonLat().lat; │ │ │ │ │ + this.canvas.translate(-cx, -cy); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.translate(-cx, -cy); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // Since OSM is topological, we stash the node ID internally. │ │ │ │ │ - point.osm_id = parseInt(ways[i].nodes[j]); │ │ │ │ │ - point_list[j] = point; │ │ │ │ │ + // Don't forget to scale stroke widths, because they are affected by canvas scale transformations as well(!) │ │ │ │ │ + // Alternative: scale symbol coordinates manually, so stroke width scaling is not needed anymore. │ │ │ │ │ + unscaledStrokeWidth = style.strokeWidth; │ │ │ │ │ + style.strokeWidth = unscaledStrokeWidth / scaling; │ │ │ │ │ │ │ │ │ │ - // We don't display nodes if they're used inside other │ │ │ │ │ - // elements. │ │ │ │ │ - node.used = true; │ │ │ │ │ - } │ │ │ │ │ - var geometry = null; │ │ │ │ │ - if (poly) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.Polygon( │ │ │ │ │ - new OpenLayers.Geometry.LinearRing(point_list)); │ │ │ │ │ - } else { │ │ │ │ │ - geometry = new OpenLayers.Geometry.LineString(point_list); │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.canvas.lineTo(x, y); │ │ │ │ │ } │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry.transform(this.externalProjection, │ │ │ │ │ - this.internalProjection); │ │ │ │ │ + this.canvas.closePath(); │ │ │ │ │ + this.canvas.fill(); │ │ │ │ │ + │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.hitContext.lineTo(x, y); │ │ │ │ │ + } │ │ │ │ │ + this.hitContext.closePath(); │ │ │ │ │ + this.hitContext.fill(); │ │ │ │ │ } │ │ │ │ │ - var feat = new OpenLayers.Feature.Vector(geometry, │ │ │ │ │ - ways[i].tags); │ │ │ │ │ - feat.osm_id = parseInt(ways[i].id); │ │ │ │ │ - feat.fid = "way." + feat.osm_id; │ │ │ │ │ - feat_list[i] = feat; │ │ │ │ │ } │ │ │ │ │ - for (var node_id in nodes) { │ │ │ │ │ - var node = nodes[node_id]; │ │ │ │ │ - if (!node.used || this.checkTags) { │ │ │ │ │ - var tags = null; │ │ │ │ │ │ │ │ │ │ - if (this.checkTags) { │ │ │ │ │ - var result = this.getTags(node.node, true); │ │ │ │ │ - if (node.used && !result[1]) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - tags = result[0]; │ │ │ │ │ - } else { │ │ │ │ │ - tags = this.getTags(node.node); │ │ │ │ │ - } │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.canvas.lineTo(x, y); │ │ │ │ │ + } │ │ │ │ │ + this.canvas.closePath(); │ │ │ │ │ + this.canvas.stroke(); │ │ │ │ │ │ │ │ │ │ - var feat = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.Point(node['lon'], node['lat']), │ │ │ │ │ - tags); │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - feat.geometry.transform(this.externalProjection, │ │ │ │ │ - this.internalProjection); │ │ │ │ │ + │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style, scaling); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.hitContext.moveTo(x, y); │ │ │ │ │ + this.hitContext.lineTo(x, y); │ │ │ │ │ } │ │ │ │ │ - feat.osm_id = parseInt(node_id); │ │ │ │ │ - feat.fid = "node." + feat.osm_id; │ │ │ │ │ - feat_list.push(feat); │ │ │ │ │ + this.hitContext.closePath(); │ │ │ │ │ + this.hitContext.stroke(); │ │ │ │ │ } │ │ │ │ │ - // Memory cleanup │ │ │ │ │ - node.node = null; │ │ │ │ │ + │ │ │ │ │ } │ │ │ │ │ - return feat_list; │ │ │ │ │ + │ │ │ │ │ + style.strokeWidth = unscaledStrokeWidth; │ │ │ │ │ + this.canvas.restore(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.restore(); │ │ │ │ │ + } │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getNodes │ │ │ │ │ - * Return the node items from a doc. │ │ │ │ │ + * Method: setCanvasStyle │ │ │ │ │ + * Prepare the canvas for drawing by setting various global settings. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * doc - {DOMElement} node to parse tags from │ │ │ │ │ + * type - {String} one of 'stroke', 'fill', or 'reset' │ │ │ │ │ + * style - {Object} Symbolizer hash │ │ │ │ │ */ │ │ │ │ │ - getNodes: function(doc) { │ │ │ │ │ - var node_list = doc.getElementsByTagName("node"); │ │ │ │ │ - var nodes = {}; │ │ │ │ │ - for (var i = 0; i < node_list.length; i++) { │ │ │ │ │ - var node = node_list[i]; │ │ │ │ │ - var id = node.getAttribute("id"); │ │ │ │ │ - nodes[id] = { │ │ │ │ │ - 'lat': node.getAttribute("lat"), │ │ │ │ │ - 'lon': node.getAttribute("lon"), │ │ │ │ │ - 'node': node │ │ │ │ │ - }; │ │ │ │ │ + setCanvasStyle: function(type, style) { │ │ │ │ │ + if (type === "fill") { │ │ │ │ │ + this.canvas.globalAlpha = style['fillOpacity']; │ │ │ │ │ + this.canvas.fillStyle = style['fillColor']; │ │ │ │ │ + } else if (type === "stroke") { │ │ │ │ │ + this.canvas.globalAlpha = style['strokeOpacity']; │ │ │ │ │ + this.canvas.strokeStyle = style['strokeColor']; │ │ │ │ │ + this.canvas.lineWidth = style['strokeWidth']; │ │ │ │ │ + } else { │ │ │ │ │ + this.canvas.globalAlpha = 0; │ │ │ │ │ + this.canvas.lineWidth = 1; │ │ │ │ │ } │ │ │ │ │ - return nodes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getWays │ │ │ │ │ - * Return the way items from a doc. │ │ │ │ │ + * Method: featureIdToHex │ │ │ │ │ + * Convert a feature ID string into an RGB hex string. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * doc - {DOMElement} node to parse tags from │ │ │ │ │ + * featureId - {String} Feature id │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} RGB hex string. │ │ │ │ │ */ │ │ │ │ │ - getWays: function(doc) { │ │ │ │ │ - var way_list = doc.getElementsByTagName("way"); │ │ │ │ │ - var return_ways = []; │ │ │ │ │ - for (var i = 0; i < way_list.length; i++) { │ │ │ │ │ - var way = way_list[i]; │ │ │ │ │ - var way_object = { │ │ │ │ │ - id: way.getAttribute("id") │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - way_object.tags = this.getTags(way); │ │ │ │ │ - │ │ │ │ │ - var node_list = way.getElementsByTagName("nd"); │ │ │ │ │ - │ │ │ │ │ - way_object.nodes = new Array(node_list.length); │ │ │ │ │ - │ │ │ │ │ - for (var j = 0; j < node_list.length; j++) { │ │ │ │ │ - way_object.nodes[j] = node_list[j].getAttribute("ref"); │ │ │ │ │ - } │ │ │ │ │ - return_ways.push(way_object); │ │ │ │ │ + featureIdToHex: function(featureId) { │ │ │ │ │ + var id = Number(featureId.split("_").pop()) + 1; // zero for no feature │ │ │ │ │ + if (id >= 16777216) { │ │ │ │ │ + this.hitOverflow = id - 16777215; │ │ │ │ │ + id = id % 16777216 + 1; │ │ │ │ │ } │ │ │ │ │ - return return_ways; │ │ │ │ │ - │ │ │ │ │ + var hex = "000000" + id.toString(16); │ │ │ │ │ + var len = hex.length; │ │ │ │ │ + hex = "#" + hex.substring(len - 6, len); │ │ │ │ │ + return hex; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getTags │ │ │ │ │ - * Return the tags list attached to a specific DOM element. │ │ │ │ │ + * Method: setHitContextStyle │ │ │ │ │ + * Prepare the hit canvas for drawing by setting various global settings. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * dom_node - {DOMElement} node to parse tags from │ │ │ │ │ - * interesting_tags - {Boolean} whether the return from this function should │ │ │ │ │ - * return a boolean indicating that it has 'interesting tags' -- │ │ │ │ │ - * tags like attribution and source are ignored. (To change the list │ │ │ │ │ - * of tags, see interestingTagsExclude) │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * tags - {Object} hash of tags │ │ │ │ │ - * interesting - {Boolean} if interesting_tags is passed, returns │ │ │ │ │ - * whether there are any interesting tags on this element. │ │ │ │ │ + * type - {String} one of 'stroke', 'fill', or 'reset' │ │ │ │ │ + * featureId - {String} The feature id. │ │ │ │ │ + * symbolizer - {<OpenLayers.Symbolizer>} The symbolizer. │ │ │ │ │ */ │ │ │ │ │ - getTags: function(dom_node, interesting_tags) { │ │ │ │ │ - var tag_list = dom_node.getElementsByTagName("tag"); │ │ │ │ │ - var tags = {}; │ │ │ │ │ - var interesting = false; │ │ │ │ │ - for (var j = 0; j < tag_list.length; j++) { │ │ │ │ │ - var key = tag_list[j].getAttribute("k"); │ │ │ │ │ - tags[key] = tag_list[j].getAttribute("v"); │ │ │ │ │ - if (interesting_tags) { │ │ │ │ │ - if (!this.interestingTagsExclude[key]) { │ │ │ │ │ - interesting = true; │ │ │ │ │ + setHitContextStyle: function(type, featureId, symbolizer, strokeScaling) { │ │ │ │ │ + var hex = this.featureIdToHex(featureId); │ │ │ │ │ + if (type == "fill") { │ │ │ │ │ + this.hitContext.globalAlpha = 1.0; │ │ │ │ │ + this.hitContext.fillStyle = hex; │ │ │ │ │ + } else if (type == "stroke") { │ │ │ │ │ + this.hitContext.globalAlpha = 1.0; │ │ │ │ │ + this.hitContext.strokeStyle = hex; │ │ │ │ │ + // bump up stroke width to deal with antialiasing. If strokeScaling is defined, we're rendering a symbol │ │ │ │ │ + // on a transformed canvas, so the antialias width bump has to scale as well. │ │ │ │ │ + if (typeof strokeScaling === "undefined") { │ │ │ │ │ + this.hitContext.lineWidth = symbolizer.strokeWidth + 2; │ │ │ │ │ + } else { │ │ │ │ │ + if (!isNaN(strokeScaling)) { │ │ │ │ │ + this.hitContext.lineWidth = symbolizer.strokeWidth + 2.0 / strokeScaling; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + this.hitContext.globalAlpha = 0; │ │ │ │ │ + this.hitContext.lineWidth = 1; │ │ │ │ │ } │ │ │ │ │ - return interesting_tags ? [tags, interesting] : tags; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: isWayArea │ │ │ │ │ - * Given a way object from getWays, check whether the tags and geometry │ │ │ │ │ - * indicate something is an area. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawPoint │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ */ │ │ │ │ │ - isWayArea: function(way) { │ │ │ │ │ - var poly_shaped = false; │ │ │ │ │ - var poly_tags = false; │ │ │ │ │ + drawPoint: function(geometry, style, featureId) { │ │ │ │ │ + if (style.graphic !== false) { │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + this.drawExternalGraphic(geometry, style, featureId); │ │ │ │ │ + } else if (style.graphicName && (style.graphicName != "circle")) { │ │ │ │ │ + this.drawNamedSymbol(geometry, style, featureId); │ │ │ │ │ + } else { │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ + if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ + var twoPi = Math.PI * 2; │ │ │ │ │ + var radius = style.pointRadius; │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.canvas.fill(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.hitContext.fill(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (way.nodes[0] == way.nodes[way.nodes.length - 1]) { │ │ │ │ │ - poly_shaped = true; │ │ │ │ │ - } │ │ │ │ │ - if (this.checkTags) { │ │ │ │ │ - for (var key in way.tags) { │ │ │ │ │ - if (this.areaTags[key]) { │ │ │ │ │ - poly_tags = true; │ │ │ │ │ - break; │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.canvas.stroke(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.hitContext.stroke(); │ │ │ │ │ + } │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return poly_shaped && (this.checkTags ? poly_tags : true); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Takes a list of features, returns a serialized OSM format file for use │ │ │ │ │ - * in tools like JOSM. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * Method: drawLineString │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ */ │ │ │ │ │ - write: function(features) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ + drawLineString: function(geometry, style, featureId) { │ │ │ │ │ + style = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + fill: false │ │ │ │ │ + }, style); │ │ │ │ │ + this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.osm_id = 1; │ │ │ │ │ - this.created_nodes = {}; │ │ │ │ │ - var root_node = this.createElementNS(null, "osm"); │ │ │ │ │ - root_node.setAttribute("version", "0.5"); │ │ │ │ │ - root_node.setAttribute("generator", "OpenLayers " + OpenLayers.VERSION_NUMBER); │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawLinearRing │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + */ │ │ │ │ │ + drawLinearRing: function(geometry, style, featureId) { │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.renderPath(this.canvas, geometry, style, featureId, "fill"); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.renderPath(this.hitContext, geometry, style, featureId, "fill"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.renderPath(this.canvas, geometry, style, featureId, "stroke"); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ + this.renderPath(this.hitContext, geometry, style, featureId, "stroke"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Loop backwards, because the deserializer puts nodes last, and │ │ │ │ │ - // we want them first if possible │ │ │ │ │ - for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ - var nodes = this.createFeatureNodes(features[i]); │ │ │ │ │ - for (var j = 0; j < nodes.length; j++) { │ │ │ │ │ - root_node.appendChild(nodes[j]); │ │ │ │ │ + /** │ │ │ │ │ + * Method: renderPath │ │ │ │ │ + * Render a path with stroke and optional fill. │ │ │ │ │ + */ │ │ │ │ │ + renderPath: function(context, geometry, style, featureId, type) { │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + context.beginPath(); │ │ │ │ │ + var start = this.getLocalXY(components[0]); │ │ │ │ │ + var x = start[0]; │ │ │ │ │ + var y = start[1]; │ │ │ │ │ + if (!isNaN(x) && !isNaN(y)) { │ │ │ │ │ + context.moveTo(start[0], start[1]); │ │ │ │ │ + for (var i = 1; i < len; ++i) { │ │ │ │ │ + var pt = this.getLocalXY(components[i]); │ │ │ │ │ + context.lineTo(pt[0], pt[1]); │ │ │ │ │ + } │ │ │ │ │ + if (type === "fill") { │ │ │ │ │ + context.fill(); │ │ │ │ │ + } else { │ │ │ │ │ + context.stroke(); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [root_node]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createFeatureNodes │ │ │ │ │ - * Takes a feature, returns a list of nodes from size 0->n. │ │ │ │ │ - * Will include all pieces of the serialization that are required which │ │ │ │ │ - * have not already been created. Calls out to createXML based on geometry │ │ │ │ │ - * type. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * Method: drawPolygon │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * featureId - {String} │ │ │ │ │ */ │ │ │ │ │ - createFeatureNodes: function(feature) { │ │ │ │ │ - var nodes = []; │ │ │ │ │ - var className = feature.geometry.CLASS_NAME; │ │ │ │ │ - var type = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ - type = type.toLowerCase(); │ │ │ │ │ - var builder = this.createXML[type]; │ │ │ │ │ - if (builder) { │ │ │ │ │ - nodes = builder.apply(this, [feature]); │ │ │ │ │ + drawPolygon: function(geometry, style, featureId) { │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + this.drawLinearRing(components[0], style, featureId); │ │ │ │ │ + // erase inner rings │ │ │ │ │ + for (var i = 1; i < len; ++i) { │ │ │ │ │ + /** │ │ │ │ │ + * Note that this is overly agressive. Here we punch holes through │ │ │ │ │ + * all previously rendered features on the same canvas. A better │ │ │ │ │ + * solution for polygons with interior rings would be to draw the │ │ │ │ │ + * polygon on a sketch canvas first. We could erase all holes │ │ │ │ │ + * there and then copy the drawing to the layer canvas. │ │ │ │ │ + * TODO: http://trac.osgeo.org/openlayers/ticket/3130 │ │ │ │ │ + */ │ │ │ │ │ + this.canvas.globalCompositeOperation = "destination-out"; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.globalCompositeOperation = "destination-out"; │ │ │ │ │ + } │ │ │ │ │ + this.drawLinearRing( │ │ │ │ │ + components[i], │ │ │ │ │ + OpenLayers.Util.applyDefaults({ │ │ │ │ │ + stroke: false, │ │ │ │ │ + fillOpacity: 1.0 │ │ │ │ │ + }, style), │ │ │ │ │ + featureId │ │ │ │ │ + ); │ │ │ │ │ + this.canvas.globalCompositeOperation = "source-over"; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.globalCompositeOperation = "source-over"; │ │ │ │ │ + } │ │ │ │ │ + this.drawLinearRing( │ │ │ │ │ + components[i], │ │ │ │ │ + OpenLayers.Util.applyDefaults({ │ │ │ │ │ + fill: false │ │ │ │ │ + }, style), │ │ │ │ │ + featureId │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - return nodes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createXML │ │ │ │ │ - * Takes a feature, returns a list of nodes from size 0->n. │ │ │ │ │ - * Will include all pieces of the serialization that are required which │ │ │ │ │ - * have not already been created. │ │ │ │ │ + * Method: drawText │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * location - {<OpenLayers.Point>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ */ │ │ │ │ │ - createXML: { │ │ │ │ │ - 'point': function(point) { │ │ │ │ │ - var id = null; │ │ │ │ │ - var geometry = point.geometry ? point.geometry : point; │ │ │ │ │ + drawText: function(location, style) { │ │ │ │ │ + var pt = this.getLocalXY(location); │ │ │ │ │ │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry = geometry.clone(); │ │ │ │ │ - geometry.transform(this.internalProjection, │ │ │ │ │ - this.externalProjection); │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ + this.canvas.fillStyle = style.fontColor; │ │ │ │ │ + this.canvas.globalAlpha = style.fontOpacity || 1.0; │ │ │ │ │ + var fontStyle = [style.fontStyle ? style.fontStyle : "normal", │ │ │ │ │ + "normal", // "font-variant" not supported │ │ │ │ │ + style.fontWeight ? style.fontWeight : "normal", │ │ │ │ │ + style.fontSize ? style.fontSize : "1em", │ │ │ │ │ + style.fontFamily ? style.fontFamily : "sans-serif" │ │ │ │ │ + ].join(" "); │ │ │ │ │ + var labelRows = style.label.split('\n'); │ │ │ │ │ + var numRows = labelRows.length; │ │ │ │ │ + if (this.canvas.fillText) { │ │ │ │ │ + // HTML5 │ │ │ │ │ + this.canvas.font = fontStyle; │ │ │ │ │ + this.canvas.textAlign = │ │ │ │ │ + OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[0]] || │ │ │ │ │ + "center"; │ │ │ │ │ + this.canvas.textBaseline = │ │ │ │ │ + OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[1]] || │ │ │ │ │ + "middle"; │ │ │ │ │ + var vfactor = │ │ │ │ │ + OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var already_exists = false; // We don't return anything if the node │ │ │ │ │ - // has already been created │ │ │ │ │ - if (point.osm_id) { │ │ │ │ │ - id = point.osm_id; │ │ │ │ │ - if (this.created_nodes[id]) { │ │ │ │ │ - already_exists = true; │ │ │ │ │ + var lineHeight = │ │ │ │ │ + this.canvas.measureText('Mg').height || │ │ │ │ │ + this.canvas.measureText('xx').width; │ │ │ │ │ + pt[1] += lineHeight * vfactor * (numRows - 1); │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + if (style.labelOutlineWidth) { │ │ │ │ │ + this.canvas.save(); │ │ │ │ │ + this.canvas.globalAlpha = style.labelOutlineOpacity || style.fontOpacity || 1.0; │ │ │ │ │ + this.canvas.strokeStyle = style.labelOutlineColor; │ │ │ │ │ + this.canvas.lineWidth = style.labelOutlineWidth; │ │ │ │ │ + this.canvas.strokeText(labelRows[i], pt[0], pt[1] + (lineHeight * i) + 1); │ │ │ │ │ + this.canvas.restore(); │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - id = -this.osm_id; │ │ │ │ │ - this.osm_id++; │ │ │ │ │ - } │ │ │ │ │ - if (already_exists) { │ │ │ │ │ - node = this.created_nodes[id]; │ │ │ │ │ - } else { │ │ │ │ │ - var node = this.createElementNS(null, "node"); │ │ │ │ │ + this.canvas.fillText(labelRows[i], pt[0], pt[1] + (lineHeight * i)); │ │ │ │ │ } │ │ │ │ │ - this.created_nodes[id] = node; │ │ │ │ │ - node.setAttribute("id", id); │ │ │ │ │ - node.setAttribute("lon", geometry.x); │ │ │ │ │ - node.setAttribute("lat", geometry.y); │ │ │ │ │ - if (point.attributes) { │ │ │ │ │ - this.serializeTags(point, node); │ │ │ │ │ + } else if (this.canvas.mozDrawText) { │ │ │ │ │ + // Mozilla pre-Gecko1.9.1 (<FF3.1) │ │ │ │ │ + this.canvas.mozTextStyle = fontStyle; │ │ │ │ │ + // No built-in text alignment, so we measure and adjust the position │ │ │ │ │ + var hfactor = │ │ │ │ │ + OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[0]]; │ │ │ │ │ + if (hfactor == null) { │ │ │ │ │ + hfactor = -.5; │ │ │ │ │ } │ │ │ │ │ - this.setState(point, node); │ │ │ │ │ - return already_exists ? [] : [node]; │ │ │ │ │ - }, │ │ │ │ │ - linestring: function(feature) { │ │ │ │ │ - var id; │ │ │ │ │ - var nodes = []; │ │ │ │ │ - var geometry = feature.geometry; │ │ │ │ │ - if (feature.osm_id) { │ │ │ │ │ - id = feature.osm_id; │ │ │ │ │ - } else { │ │ │ │ │ - id = -this.osm_id; │ │ │ │ │ - this.osm_id++; │ │ │ │ │ + var vfactor = │ │ │ │ │ + OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5; │ │ │ │ │ } │ │ │ │ │ - var way = this.createElementNS(null, "way"); │ │ │ │ │ - way.setAttribute("id", id); │ │ │ │ │ - for (var i = 0; i < geometry.components.length; i++) { │ │ │ │ │ - var node = this.createXML['point'].apply(this, [geometry.components[i]]); │ │ │ │ │ - if (node.length) { │ │ │ │ │ - node = node[0]; │ │ │ │ │ - var node_ref = node.getAttribute("id"); │ │ │ │ │ - nodes.push(node); │ │ │ │ │ - } else { │ │ │ │ │ - node_ref = geometry.components[i].osm_id; │ │ │ │ │ - node = this.created_nodes[node_ref]; │ │ │ │ │ - } │ │ │ │ │ - this.setState(feature, node); │ │ │ │ │ - var nd_dom = this.createElementNS(null, "nd"); │ │ │ │ │ - nd_dom.setAttribute("ref", node_ref); │ │ │ │ │ - way.appendChild(nd_dom); │ │ │ │ │ + var lineHeight = this.canvas.mozMeasureText('xx'); │ │ │ │ │ + pt[1] += lineHeight * (1 + (vfactor * numRows)); │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + var x = pt[0] + (hfactor * this.canvas.mozMeasureText(labelRows[i])); │ │ │ │ │ + var y = pt[1] + (i * lineHeight); │ │ │ │ │ + this.canvas.translate(x, y); │ │ │ │ │ + this.canvas.mozDrawText(labelRows[i]); │ │ │ │ │ + this.canvas.translate(-x, -y); │ │ │ │ │ } │ │ │ │ │ - this.serializeTags(feature, way); │ │ │ │ │ - nodes.push(way); │ │ │ │ │ - │ │ │ │ │ - return nodes; │ │ │ │ │ - }, │ │ │ │ │ - polygon: function(feature) { │ │ │ │ │ - var attrs = OpenLayers.Util.extend({ │ │ │ │ │ - 'area': 'yes' │ │ │ │ │ - }, feature.attributes); │ │ │ │ │ - var feat = new OpenLayers.Feature.Vector(feature.geometry.components[0], attrs); │ │ │ │ │ - feat.osm_id = feature.osm_id; │ │ │ │ │ - return this.createXML['linestring'].apply(this, [feat]); │ │ │ │ │ } │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: serializeTags │ │ │ │ │ - * Given a feature, serialize the attributes onto the given node. │ │ │ │ │ + * Method: getLocalXY │ │ │ │ │ + * transform geographic xy into pixel xy │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * node - {DOMNode} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ */ │ │ │ │ │ - serializeTags: function(feature, node) { │ │ │ │ │ - for (var key in feature.attributes) { │ │ │ │ │ - var tag = this.createElementNS(null, "tag"); │ │ │ │ │ - tag.setAttribute("k", key); │ │ │ │ │ - tag.setAttribute("v", feature.attributes[key]); │ │ │ │ │ - node.appendChild(tag); │ │ │ │ │ + getLocalXY: function(point) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var extent = this.extent; │ │ │ │ │ + var x = ((point.x - this.featureDx) / resolution + (-extent.left / resolution)); │ │ │ │ │ + var y = ((extent.top / resolution) - point.y / resolution); │ │ │ │ │ + return [x, y]; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clear │ │ │ │ │ + * Clear all vectors from the renderer. │ │ │ │ │ + */ │ │ │ │ │ + clear: function() { │ │ │ │ │ + var height = this.root.height; │ │ │ │ │ + var width = this.root.width; │ │ │ │ │ + this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ + this.features = {}; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.clearRect(0, 0, width, height); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setState │ │ │ │ │ - * OpenStreetMap has a convention that 'state' is stored for modification or deletion. │ │ │ │ │ - * This allows the file to be uploaded via JOSM or the bulk uploader tool. │ │ │ │ │ + * Method: getFeatureIdFromEvent │ │ │ │ │ + * Returns a feature id from an event on the renderer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector} A feature or undefined. This method returns a │ │ │ │ │ + * feature instead of a feature id to avoid an unnecessary lookup on the │ │ │ │ │ + * layer. │ │ │ │ │ + */ │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var featureId, feature; │ │ │ │ │ + │ │ │ │ │ + if (this.hitDetection && this.root.style.display !== "none") { │ │ │ │ │ + // this dragging check should go in the feature handler │ │ │ │ │ + if (!this.map.dragging) { │ │ │ │ │ + var xy = evt.xy; │ │ │ │ │ + var x = xy.x | 0; │ │ │ │ │ + var y = xy.y | 0; │ │ │ │ │ + var data = this.hitContext.getImageData(x, y, 1, 1).data; │ │ │ │ │ + if (data[3] === 255) { // antialiased │ │ │ │ │ + var id = data[2] + (256 * (data[1] + (256 * data[0]))); │ │ │ │ │ + if (id) { │ │ │ │ │ + featureId = "OpenLayers_Feature_Vector_" + (id - 1 + this.hitOverflow); │ │ │ │ │ + try { │ │ │ │ │ + feature = this.features[featureId][0]; │ │ │ │ │ + } catch (err) { │ │ │ │ │ + // Because of antialiasing on the canvas, when the hit location is at a point where the edge of │ │ │ │ │ + // one symbol intersects the interior of another symbol, a wrong hit color (and therefore id) results. │ │ │ │ │ + // todo: set Antialiasing = 'off' on the hitContext as soon as browsers allow it. │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return feature; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: eraseFeatures │ │ │ │ │ + * This is called by the layer to erase features; removes the feature from │ │ │ │ │ + * the list, then redraws the layer. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * node - {DOMNode} │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ */ │ │ │ │ │ - setState: function(feature, node) { │ │ │ │ │ - if (feature.state) { │ │ │ │ │ - var state = null; │ │ │ │ │ - switch (feature.state) { │ │ │ │ │ - case OpenLayers.State.UPDATE: │ │ │ │ │ - state = "modify"; │ │ │ │ │ - case OpenLayers.State.DELETE: │ │ │ │ │ - state = "delete"; │ │ │ │ │ + eraseFeatures: function(features) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0; i < features.length; ++i) { │ │ │ │ │ + delete this.features[features[i].id]; │ │ │ │ │ + } │ │ │ │ │ + this.redraw(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: redraw │ │ │ │ │ + * The real 'meat' of the function: any time things have changed, │ │ │ │ │ + * redraw() can be called to loop over all the data and (you guessed │ │ │ │ │ + * it) redraw it. Unlike Elements-based Renderers, we can't interact │ │ │ │ │ + * with things once they're drawn, to remove them, for example, so │ │ │ │ │ + * instead we have to just clear everything and draw from scratch. │ │ │ │ │ + */ │ │ │ │ │ + redraw: function() { │ │ │ │ │ + if (!this.locked) { │ │ │ │ │ + var height = this.root.height; │ │ │ │ │ + var width = this.root.width; │ │ │ │ │ + this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.clearRect(0, 0, width, height); │ │ │ │ │ } │ │ │ │ │ - if (state) { │ │ │ │ │ - node.setAttribute("action", state); │ │ │ │ │ + var labelMap = []; │ │ │ │ │ + var feature, geometry, style; │ │ │ │ │ + var worldBounds = (this.map.baseLayer && this.map.baseLayer.wrapDateLine) && this.map.getMaxExtent(); │ │ │ │ │ + for (var id in this.features) { │ │ │ │ │ + if (!this.features.hasOwnProperty(id)) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + feature = this.features[id][0]; │ │ │ │ │ + geometry = feature.geometry; │ │ │ │ │ + this.calculateFeatureDx(geometry.getBounds(), worldBounds); │ │ │ │ │ + style = this.features[id][1]; │ │ │ │ │ + this.drawGeometry(geometry, style, feature.id); │ │ │ │ │ + if (style.label) { │ │ │ │ │ + labelMap.push([feature, style]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var item; │ │ │ │ │ + for (var i = 0, len = labelMap.length; i < len; ++i) { │ │ │ │ │ + item = labelMap[i]; │ │ │ │ │ + this.drawText(item[0].geometry.getCentroid(), item[1]); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.OSM" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.Canvas" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.Canvas.LABEL_ALIGN │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.Canvas.LABEL_ALIGN = { │ │ │ │ │ + "l": "left", │ │ │ │ │ + "r": "right", │ │ │ │ │ + "t": "top", │ │ │ │ │ + "b": "bottom" │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.Canvas.LABEL_FACTOR │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.Canvas.LABEL_FACTOR = { │ │ │ │ │ + "l": 0, │ │ │ │ │ + "r": -1, │ │ │ │ │ + "t": 0, │ │ │ │ │ + "b": -1 │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.Canvas.drawImageScaleFactor │ │ │ │ │ + * {Number} Scale factor to apply to the canvas drawImage arguments. This │ │ │ │ │ + * is always 1 except for Android 2.1 devices, to work around │ │ │ │ │ + * http://code.google.com/p/android/issues/detail?id=5141. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.Canvas.drawImageScaleFactor = null; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/OWSContext.js │ │ │ │ │ + OpenLayers/Renderer/VML.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/Context.js │ │ │ │ │ + * @requires OpenLayers/Renderer/Elements.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.OWSContext │ │ │ │ │ - * Read and write OWS Context documents. OWS Context documents are a │ │ │ │ │ - * preliminary OGC (Open Geospatial Consortium) standard for storing the │ │ │ │ │ - * state of a web mapping application. In a way it is the successor to │ │ │ │ │ - * Web Map Context (WMC), since it is more generic and more types of layers │ │ │ │ │ - * can be stored. Also, nesting of layers is supported since version 0.3.1. │ │ │ │ │ - * For more information see: http://www.ogcnetwork.net/context │ │ │ │ │ + * Class: OpenLayers.Renderer.VML │ │ │ │ │ + * Render vector features in browsers with VML capability. Construct a new │ │ │ │ │ + * VML renderer with the <OpenLayers.Renderer.VML> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Note that for all calculations in this class, we use (num | 0) to truncate a │ │ │ │ │ + * float value to an integer. This is done because it seems that VML doesn't │ │ │ │ │ + * support float values. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.Context> │ │ │ │ │ + * - <OpenLayers.Renderer.Elements> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.OWSContext = OpenLayers.Class(OpenLayers.Format.Context, { │ │ │ │ │ +OpenLayers.Renderer.VML = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "0.3.1". │ │ │ │ │ + * Property: xmlns │ │ │ │ │ + * {String} XML Namespace URN │ │ │ │ │ */ │ │ │ │ │ - defaultVersion: "0.3.1", │ │ │ │ │ + xmlns: "urn:schemas-microsoft-com:vml", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.OWSContext │ │ │ │ │ - * Create a new parser for OWS Context documents. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * Property: symbolCache │ │ │ │ │ + * {DOMElement} node holding symbols. This hash is keyed by symbol name, │ │ │ │ │ + * and each value is a hash with a "path" and an "extent" property. │ │ │ │ │ */ │ │ │ │ │ + symbolCache: {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getVersion │ │ │ │ │ - * Returns the version to use. Subclasses can override this function │ │ │ │ │ - * if a different version detection is needed. │ │ │ │ │ + * Property: offset │ │ │ │ │ + * {Object} Hash with "x" and "y" properties │ │ │ │ │ + */ │ │ │ │ │ + offset: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Renderer.VML │ │ │ │ │ + * Create a new VML renderer. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * root - {DOMElement} │ │ │ │ │ - * options - {Object} Optional configuration object. │ │ │ │ │ + * containerID - {String} The id for the element that contains the renderer │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(containerID) { │ │ │ │ │ + if (!this.supported()) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (!document.namespaces.olv) { │ │ │ │ │ + document.namespaces.add("olv", this.xmlns); │ │ │ │ │ + var style = document.createStyleSheet(); │ │ │ │ │ + var shapes = ['shape', 'rect', 'oval', 'fill', 'stroke', 'imagedata', 'group', 'textbox']; │ │ │ │ │ + for (var i = 0, len = shapes.length; i < len; i++) { │ │ │ │ │ + │ │ │ │ │ + style.addRule('olv\\:' + shapes[i], "behavior: url(#default#VML); " + │ │ │ │ │ + "position: absolute; display: inline-block;"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.initialize.apply(this, │ │ │ │ │ + arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: supported │ │ │ │ │ + * Determine whether a browser supports this renderer. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} The version to use. │ │ │ │ │ + * {Boolean} The browser supports the VML renderer │ │ │ │ │ */ │ │ │ │ │ - getVersion: function(root, options) { │ │ │ │ │ - var version = OpenLayers.Format.XML.VersionedOGC.prototype.getVersion.apply( │ │ │ │ │ - this, arguments); │ │ │ │ │ - // 0.3.1 is backwards compatible with 0.3.0 │ │ │ │ │ - if (version === "0.3.0") { │ │ │ │ │ - version = this.defaultVersion; │ │ │ │ │ - } │ │ │ │ │ - return version; │ │ │ │ │ + supported: function() { │ │ │ │ │ + return !!(document.namespaces); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: toContext │ │ │ │ │ - * Create a context object free from layer given a map or a │ │ │ │ │ - * context object. │ │ │ │ │ + * Method: setExtent │ │ │ │ │ + * Set the renderer's extent │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {<OpenLayers.Map> | Object} The map or context. │ │ │ │ │ - * │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} │ │ │ │ │ + * resolutionChanged - {Boolean} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} A context object. │ │ │ │ │ + * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ + * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ */ │ │ │ │ │ - toContext: function(obj) { │ │ │ │ │ - var context = {}; │ │ │ │ │ - if (obj.CLASS_NAME == "OpenLayers.Map") { │ │ │ │ │ - context.bounds = obj.getExtent(); │ │ │ │ │ - context.maxExtent = obj.maxExtent; │ │ │ │ │ - context.projection = obj.projection; │ │ │ │ │ - context.size = obj.getSize(); │ │ │ │ │ - context.layers = obj.layers; │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + │ │ │ │ │ + var left = (extent.left / resolution) | 0; │ │ │ │ │ + var top = (extent.top / resolution - this.size.h) | 0; │ │ │ │ │ + if (resolutionChanged || !this.offset) { │ │ │ │ │ + this.offset = { │ │ │ │ │ + x: left, │ │ │ │ │ + y: top │ │ │ │ │ + }; │ │ │ │ │ + left = 0; │ │ │ │ │ + top = 0; │ │ │ │ │ + } else { │ │ │ │ │ + left = left - this.offset.x; │ │ │ │ │ + top = top - this.offset.y; │ │ │ │ │ } │ │ │ │ │ - return context; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.OWSContext" │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/EncodedPolyline.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + var org = (left - this.xOffset) + " " + top; │ │ │ │ │ + this.root.coordorigin = org; │ │ │ │ │ + var roots = [this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ + var root; │ │ │ │ │ + for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ + root = roots[i]; │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + var size = this.size.w + " " + this.size.h; │ │ │ │ │ + root.coordsize = size; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - */ │ │ │ │ │ + } │ │ │ │ │ + // flip the VML display Y axis upside down so it │ │ │ │ │ + // matches the display Y axis of the map │ │ │ │ │ + this.root.style.flip = "y"; │ │ │ │ │ + │ │ │ │ │ + return coordSysUnchanged; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.EncodedPolyline │ │ │ │ │ - * Class for reading and writing encoded polylines. Create a new instance │ │ │ │ │ - * with the <OpenLayers.Format.EncodedPolyline> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.EncodedPolyline = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: geometryType │ │ │ │ │ - * {String} Geometry type to output. One of: linestring (default), │ │ │ │ │ - * linearring, point, multipoint or polygon. If the geometryType is │ │ │ │ │ - * point, only the first point of the string is returned. │ │ │ │ │ + * Method: setSize │ │ │ │ │ + * Set the size of the drawing surface │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * size - {<OpenLayers.Size>} the size of the drawing surface │ │ │ │ │ */ │ │ │ │ │ - geometryType: "linestring", │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + // setting width and height on all roots to avoid flicker which we │ │ │ │ │ + // would get with 100% width and height on child roots │ │ │ │ │ + var roots = [ │ │ │ │ │ + this.rendererRoot, │ │ │ │ │ + this.root, │ │ │ │ │ + this.vectorRoot, │ │ │ │ │ + this.textRoot │ │ │ │ │ + ]; │ │ │ │ │ + var w = this.size.w + "px"; │ │ │ │ │ + var h = this.size.h + "px"; │ │ │ │ │ + var root; │ │ │ │ │ + for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ + root = roots[i]; │ │ │ │ │ + root.style.width = w; │ │ │ │ │ + root.style.height = h; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.EncodedPolyline │ │ │ │ │ - * Create a new parser for encoded polylines │ │ │ │ │ + * Method: getNodeType │ │ │ │ │ + * Get the node type for a geometry and style │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Format.EncodedPolyline>} A new encoded polylines parser. │ │ │ │ │ + * {String} The corresponding node type for the specified geometry │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.prototype.initialize.apply(this, [options]); │ │ │ │ │ + getNodeType: function(geometry, style) { │ │ │ │ │ + var nodeType = null; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + nodeType = "olv:rect"; │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + nodeType = "olv:shape"; │ │ │ │ │ + } else { │ │ │ │ │ + nodeType = "olv:oval"; │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + nodeType = "olv:rect"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + case "OpenLayers.Geometry.Curve": │ │ │ │ │ + nodeType = "olv:shape"; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + return nodeType; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Deserialize an encoded polyline string and return a vector feature. │ │ │ │ │ + * Method: setStyle │ │ │ │ │ + * Use to set all the style attributes to a VML node. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * encoded - {String} An encoded polyline string │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A vector feature with a linestring. │ │ │ │ │ + * node - {DOMElement} An VML element to decorate │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * options - {Object} Currently supported options include │ │ │ │ │ + * 'isFilled' {Boolean} and │ │ │ │ │ + * 'isStroked' {Boolean} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ */ │ │ │ │ │ - read: function(encoded) { │ │ │ │ │ - var geomType; │ │ │ │ │ - if (this.geometryType == "linestring") │ │ │ │ │ - geomType = OpenLayers.Geometry.LineString; │ │ │ │ │ - else if (this.geometryType == "linearring") │ │ │ │ │ - geomType = OpenLayers.Geometry.LinearRing; │ │ │ │ │ - else if (this.geometryType == "multipoint") │ │ │ │ │ - geomType = OpenLayers.Geometry.MultiPoint; │ │ │ │ │ - else if (this.geometryType != "point" && this.geometryType != "polygon") │ │ │ │ │ - return null; │ │ │ │ │ - │ │ │ │ │ - var flatPoints = this.decodeDeltas(encoded, 2); │ │ │ │ │ - var flatPointsLength = flatPoints.length; │ │ │ │ │ + setStyle: function(node, style, options, geometry) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + options = options || node._options; │ │ │ │ │ + var fillColor = style.fillColor; │ │ │ │ │ │ │ │ │ │ - var pointGeometries = []; │ │ │ │ │ - for (var i = 0; i + 1 < flatPointsLength;) { │ │ │ │ │ - var y = flatPoints[i++], │ │ │ │ │ - x = flatPoints[i++]; │ │ │ │ │ - pointGeometries.push(new OpenLayers.Geometry.Point(x, y)); │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + node.title = title; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ + if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + options.isFilled = true; │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ │ │ │ │ │ - if (this.geometryType == "point") │ │ │ │ │ - return new OpenLayers.Feature.Vector( │ │ │ │ │ - pointGeometries[0] │ │ │ │ │ - ); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ + style.graphicXOffset : -(0.5 * width); │ │ │ │ │ + var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ + style.graphicYOffset : -(0.5 * height); │ │ │ │ │ │ │ │ │ │ - if (this.geometryType == "polygon") │ │ │ │ │ - return new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.Polygon([ │ │ │ │ │ - new OpenLayers.Geometry.LinearRing(pointGeometries) │ │ │ │ │ - ]) │ │ │ │ │ - ); │ │ │ │ │ + node.style.left = ((((geometry.x - this.featureDx) / resolution - this.offset.x) + xOffset) | 0) + "px"; │ │ │ │ │ + node.style.top = (((geometry.y / resolution - this.offset.y) - (yOffset + height)) | 0) + "px"; │ │ │ │ │ + node.style.width = width + "px"; │ │ │ │ │ + node.style.height = height + "px"; │ │ │ │ │ + node.style.flip = "y"; │ │ │ │ │ │ │ │ │ │ - return new OpenLayers.Feature.Vector( │ │ │ │ │ - new geomType(pointGeometries) │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ + // modify fillColor and options for stroke styling below │ │ │ │ │ + fillColor = "none"; │ │ │ │ │ + options.isStroked = false; │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + var cache = this.importSymbol(style.graphicName); │ │ │ │ │ + node.path = cache.path; │ │ │ │ │ + node.coordorigin = cache.left + "," + cache.bottom; │ │ │ │ │ + var size = cache.size; │ │ │ │ │ + node.coordsize = size + "," + size; │ │ │ │ │ + this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ + node.style.flip = "y"; │ │ │ │ │ + } else { │ │ │ │ │ + this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: decode │ │ │ │ │ - * Deserialize an encoded string and return an array of n-dimensional │ │ │ │ │ - * points. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * encoded - {String} An encoded string │ │ │ │ │ - * dims - {int} The dimension of the points that are returned │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(Array(int))} An array containing n-dimensional arrays of │ │ │ │ │ - * coordinates. │ │ │ │ │ - */ │ │ │ │ │ - decode: function(encoded, dims, opt_factor) { │ │ │ │ │ - var factor = opt_factor || 1e5; │ │ │ │ │ - var flatPoints = this.decodeDeltas(encoded, dims, factor); │ │ │ │ │ - var flatPointsLength = flatPoints.length; │ │ │ │ │ + // fill │ │ │ │ │ + if (options.isFilled) { │ │ │ │ │ + node.fillcolor = fillColor; │ │ │ │ │ + } else { │ │ │ │ │ + node.filled = "false"; │ │ │ │ │ + } │ │ │ │ │ + var fills = node.getElementsByTagName("fill"); │ │ │ │ │ + var fill = (fills.length == 0) ? null : fills[0]; │ │ │ │ │ + if (!options.isFilled) { │ │ │ │ │ + if (fill) { │ │ │ │ │ + node.removeChild(fill); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + if (!fill) { │ │ │ │ │ + fill = this.createNode('olv:fill', node.id + "_fill"); │ │ │ │ │ + } │ │ │ │ │ + fill.opacity = style.fillOpacity; │ │ │ │ │ │ │ │ │ │ - var points = []; │ │ │ │ │ - for (var i = 0; i + (dims - 1) < flatPointsLength;) { │ │ │ │ │ - var point = []; │ │ │ │ │ + if (node._geometryClass === "OpenLayers.Geometry.Point" && │ │ │ │ │ + style.externalGraphic) { │ │ │ │ │ │ │ │ │ │ - for (var dim = 0; dim < dims; ++dim) { │ │ │ │ │ - point.push(flatPoints[i++]) │ │ │ │ │ + // override fillOpacity │ │ │ │ │ + if (style.graphicOpacity) { │ │ │ │ │ + fill.opacity = style.graphicOpacity; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + fill.src = style.externalGraphic; │ │ │ │ │ + fill.type = "frame"; │ │ │ │ │ + │ │ │ │ │ + if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ + fill.aspect = "atmost"; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (fill.parentNode != node) { │ │ │ │ │ + node.appendChild(fill); │ │ │ │ │ } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - points.push(point); │ │ │ │ │ + // additional rendering for rotated graphics or symbols │ │ │ │ │ + var rotation = style.rotation; │ │ │ │ │ + if ((rotation !== undefined || node._rotation !== undefined)) { │ │ │ │ │ + node._rotation = rotation; │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ + // make the fill fully transparent, because we now have │ │ │ │ │ + // the graphic as imagedata element. We cannot just remove │ │ │ │ │ + // the fill, because this is part of the hack described │ │ │ │ │ + // in graphicRotate │ │ │ │ │ + fill.opacity = 0; │ │ │ │ │ + } else if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ + node.style.rotation = rotation || 0; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return points; │ │ │ │ │ + // stroke │ │ │ │ │ + var strokes = node.getElementsByTagName("stroke"); │ │ │ │ │ + var stroke = (strokes.length == 0) ? null : strokes[0]; │ │ │ │ │ + if (!options.isStroked) { │ │ │ │ │ + node.stroked = false; │ │ │ │ │ + if (stroke) { │ │ │ │ │ + stroke.on = false; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + if (!stroke) { │ │ │ │ │ + stroke = this.createNode('olv:stroke', node.id + "_stroke"); │ │ │ │ │ + node.appendChild(stroke); │ │ │ │ │ + } │ │ │ │ │ + stroke.on = true; │ │ │ │ │ + stroke.color = style.strokeColor; │ │ │ │ │ + stroke.weight = style.strokeWidth + "px"; │ │ │ │ │ + stroke.opacity = style.strokeOpacity; │ │ │ │ │ + stroke.endcap = style.strokeLinecap == 'butt' ? 'flat' : │ │ │ │ │ + (style.strokeLinecap || 'round'); │ │ │ │ │ + if (style.strokeDashstyle) { │ │ │ │ │ + stroke.dashstyle = this.dashStyle(style); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ + node.style.cursor = style.cursor; │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Serialize a feature or array of features into a WKT string. │ │ │ │ │ - * │ │ │ │ │ + * Method: graphicRotate │ │ │ │ │ + * If a point is to be styled with externalGraphic and rotation, VML fills │ │ │ │ │ + * cannot be used to display the graphic, because rotation of graphic │ │ │ │ │ + * fills is not supported by the VML implementation of Internet Explorer. │ │ │ │ │ + * This method creates a olv:imagedata element inside the VML node, │ │ │ │ │ + * DXImageTransform.Matrix and BasicImage filters for rotation and │ │ │ │ │ + * opacity, and a 3-step hack to remove rendering artefacts from the │ │ │ │ │ + * graphic and preserve the ability of graphics to trigger events. │ │ │ │ │ + * Finally, OpenLayers methods are used to determine the correct │ │ │ │ │ + * insertion point of the rotated image, because DXImageTransform.Matrix │ │ │ │ │ + * does the rotation without the ability to specify a rotation center │ │ │ │ │ + * point. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {<OpenLayers.Feature.Vector>|Array} A feature or array of │ │ │ │ │ - * features │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The WKT string representation of the input geometries │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * xOffset - {Number} rotation center relative to image, x coordinate │ │ │ │ │ + * yOffset - {Number} rotation center relative to image, y coordinate │ │ │ │ │ + * style - {Object} │ │ │ │ │ */ │ │ │ │ │ - write: function(features) { │ │ │ │ │ - var feature; │ │ │ │ │ - if (features.constructor == Array) │ │ │ │ │ - feature = features[0]; │ │ │ │ │ - else │ │ │ │ │ - feature = features; │ │ │ │ │ + graphicRotate: function(node, xOffset, yOffset, style) { │ │ │ │ │ + var style = style || node._style; │ │ │ │ │ + var rotation = style.rotation || 0; │ │ │ │ │ │ │ │ │ │ - var geometry = feature.geometry; │ │ │ │ │ - var type = geometry.CLASS_NAME.split('.')[2].toLowerCase(); │ │ │ │ │ + var aspectRatio, size; │ │ │ │ │ + if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ + // load the image to determine its size │ │ │ │ │ + var img = new Image(); │ │ │ │ │ + img.onreadystatechange = OpenLayers.Function.bind(function() { │ │ │ │ │ + if (img.readyState == "complete" || │ │ │ │ │ + img.readyState == "interactive") { │ │ │ │ │ + aspectRatio = img.width / img.height; │ │ │ │ │ + size = Math.max(style.pointRadius * 2, │ │ │ │ │ + style.graphicWidth || 0, │ │ │ │ │ + style.graphicHeight || 0); │ │ │ │ │ + xOffset = xOffset * aspectRatio; │ │ │ │ │ + style.graphicWidth = size * aspectRatio; │ │ │ │ │ + style.graphicHeight = size; │ │ │ │ │ + this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ + } │ │ │ │ │ + }, this); │ │ │ │ │ + img.src = style.externalGraphic; │ │ │ │ │ │ │ │ │ │ - var pointGeometries; │ │ │ │ │ - if (type == "point") │ │ │ │ │ - pointGeometries = new Array(geometry); │ │ │ │ │ - else if (type == "linestring" || │ │ │ │ │ - type == "linearring" || │ │ │ │ │ - type == "multipoint") │ │ │ │ │ - pointGeometries = geometry.components; │ │ │ │ │ - else if (type == "polygon") │ │ │ │ │ - pointGeometries = geometry.components[0].components; │ │ │ │ │ - else │ │ │ │ │ - return null; │ │ │ │ │ + // will be called again by the onreadystate handler │ │ │ │ │ + return; │ │ │ │ │ + } else { │ │ │ │ │ + size = Math.max(style.graphicWidth, style.graphicHeight); │ │ │ │ │ + aspectRatio = style.graphicWidth / style.graphicHeight; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var flatPoints = []; │ │ │ │ │ + var width = Math.round(style.graphicWidth || size * aspectRatio); │ │ │ │ │ + var height = Math.round(style.graphicHeight || size); │ │ │ │ │ + node.style.width = width + "px"; │ │ │ │ │ + node.style.height = height + "px"; │ │ │ │ │ │ │ │ │ │ - var pointGeometriesLength = pointGeometries.length; │ │ │ │ │ - for (var i = 0; i < pointGeometriesLength; ++i) { │ │ │ │ │ - var pointGeometry = pointGeometries[i]; │ │ │ │ │ - flatPoints.push(pointGeometry.y); │ │ │ │ │ - flatPoints.push(pointGeometry.x); │ │ │ │ │ + // Three steps are required to remove artefacts for images with │ │ │ │ │ + // transparent backgrounds (resulting from using DXImageTransform │ │ │ │ │ + // filters on svg objects), while preserving awareness for browser │ │ │ │ │ + // events on images: │ │ │ │ │ + // - Use the fill as usual (like for unrotated images) to handle │ │ │ │ │ + // events │ │ │ │ │ + // - specify an imagedata element with the same src as the fill │ │ │ │ │ + // - style the imagedata element with an AlphaImageLoader filter │ │ │ │ │ + // with empty src │ │ │ │ │ + var image = document.getElementById(node.id + "_image"); │ │ │ │ │ + if (!image) { │ │ │ │ │ + image = this.createNode("olv:imagedata", node.id + "_image"); │ │ │ │ │ + node.appendChild(image); │ │ │ │ │ } │ │ │ │ │ + image.style.width = width + "px"; │ │ │ │ │ + image.style.height = height + "px"; │ │ │ │ │ + image.src = style.externalGraphic; │ │ │ │ │ + image.style.filter = │ │ │ │ │ + "progid:DXImageTransform.Microsoft.AlphaImageLoader(" + │ │ │ │ │ + "src='', sizingMethod='scale')"; │ │ │ │ │ │ │ │ │ │ - return this.encodeDeltas(flatPoints, 2); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: encode │ │ │ │ │ - * Serialize an array of n-dimensional points and return an encoded string │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * points - {Array(Array(int))} An array containing n-dimensional │ │ │ │ │ - * arrays of coordinates │ │ │ │ │ - * dims - {int} The dimension of the points that should be read │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} An encoded string │ │ │ │ │ - */ │ │ │ │ │ - encode: function(points, dims, opt_factor) { │ │ │ │ │ - var factor = opt_factor || 1e5; │ │ │ │ │ - var flatPoints = []; │ │ │ │ │ + var rot = rotation * Math.PI / 180; │ │ │ │ │ + var sintheta = Math.sin(rot); │ │ │ │ │ + var costheta = Math.cos(rot); │ │ │ │ │ │ │ │ │ │ - var pointsLength = points.length; │ │ │ │ │ - for (var i = 0; i < pointsLength; ++i) { │ │ │ │ │ - var point = points[i]; │ │ │ │ │ + // do the rotation on the image │ │ │ │ │ + var filter = │ │ │ │ │ + "progid:DXImageTransform.Microsoft.Matrix(M11=" + costheta + │ │ │ │ │ + ",M12=" + (-sintheta) + ",M21=" + sintheta + ",M22=" + costheta + │ │ │ │ │ + ",SizingMethod='auto expand')\n"; │ │ │ │ │ │ │ │ │ │ - for (var dim = 0; dim < dims; ++dim) { │ │ │ │ │ - flatPoints.push(point[dim]); │ │ │ │ │ - } │ │ │ │ │ + // set the opacity (needed for the imagedata) │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ + if (opacity && opacity != 1) { │ │ │ │ │ + filter += │ │ │ │ │ + "progid:DXImageTransform.Microsoft.BasicImage(opacity=" + │ │ │ │ │ + opacity + ")\n"; │ │ │ │ │ } │ │ │ │ │ + node.style.filter = filter; │ │ │ │ │ │ │ │ │ │ - return this.encodeDeltas(flatPoints, dims, factor); │ │ │ │ │ + // do the rotation again on a box, so we know the insertion point │ │ │ │ │ + var centerPoint = new OpenLayers.Geometry.Point(-xOffset, -yOffset); │ │ │ │ │ + var imgBox = new OpenLayers.Bounds(0, 0, width, height).toGeometry(); │ │ │ │ │ + imgBox.rotate(style.rotation, centerPoint); │ │ │ │ │ + var imgBounds = imgBox.getBounds(); │ │ │ │ │ + │ │ │ │ │ + node.style.left = Math.round( │ │ │ │ │ + parseInt(node.style.left) + imgBounds.left) + "px"; │ │ │ │ │ + node.style.top = Math.round( │ │ │ │ │ + parseInt(node.style.top) - imgBounds.bottom) + "px"; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: encodeDeltas │ │ │ │ │ - * Encode a list of n-dimensional points and return an encoded string │ │ │ │ │ - * │ │ │ │ │ - * Attention: This function will modify the passed array! │ │ │ │ │ - * │ │ │ │ │ + * Method: postDraw │ │ │ │ │ + * Does some node postprocessing to work around browser issues: │ │ │ │ │ + * - Some versions of Internet Explorer seem to be unable to set fillcolor │ │ │ │ │ + * and strokecolor to "none" correctly before the fill node is appended │ │ │ │ │ + * to a visible vml node. This method takes care of that and sets │ │ │ │ │ + * fillcolor and strokecolor again if needed. │ │ │ │ │ + * - In some cases, a node won't become visible after being drawn. Setting │ │ │ │ │ + * style.visibility to "visible" works around that. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * numbers - {Array.<number>} A list of n-dimensional points. │ │ │ │ │ - * dimension - {number} The dimension of the points in the list. │ │ │ │ │ - * opt_factor - {number=} The factor by which the numbers will be │ │ │ │ │ - * multiplied. The remaining decimal places will get rounded away. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {string} The encoded string. │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - encodeDeltas: function(numbers, dimension, opt_factor) { │ │ │ │ │ - var factor = opt_factor || 1e5; │ │ │ │ │ - var d; │ │ │ │ │ - │ │ │ │ │ - var lastNumbers = new Array(dimension); │ │ │ │ │ - for (d = 0; d < dimension; ++d) { │ │ │ │ │ - lastNumbers[d] = 0; │ │ │ │ │ + postDraw: function(node) { │ │ │ │ │ + node.style.visibility = "visible"; │ │ │ │ │ + var fillColor = node._style.fillColor; │ │ │ │ │ + var strokeColor = node._style.strokeColor; │ │ │ │ │ + if (fillColor == "none" && │ │ │ │ │ + node.fillcolor != fillColor) { │ │ │ │ │ + node.fillcolor = fillColor; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var numbersLength = numbers.length; │ │ │ │ │ - for (var i = 0; i < numbersLength;) { │ │ │ │ │ - for (d = 0; d < dimension; ++d, ++i) { │ │ │ │ │ - var num = numbers[i]; │ │ │ │ │ - var delta = num - lastNumbers[d]; │ │ │ │ │ - lastNumbers[d] = num; │ │ │ │ │ - │ │ │ │ │ - numbers[i] = delta; │ │ │ │ │ - } │ │ │ │ │ + if (strokeColor == "none" && │ │ │ │ │ + node.strokecolor != strokeColor) { │ │ │ │ │ + node.strokecolor = strokeColor; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return this.encodeFloats(numbers, factor); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: decodeDeltas │ │ │ │ │ - * Decode a list of n-dimensional points from an encoded string │ │ │ │ │ - * │ │ │ │ │ + * Method: setNodeDimension │ │ │ │ │ + * Get the geometry's bounds, convert it to our vml coordinate system, │ │ │ │ │ + * then set the node's position, size, and local coordinate system. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * encoded - {string} An encoded string. │ │ │ │ │ - * dimension - {number} The dimension of the points in the encoded string. │ │ │ │ │ - * opt_factor - {number=} The factor by which the resulting numbers will │ │ │ │ │ - * be divided. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array.<number>} A list of n-dimensional points. │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ */ │ │ │ │ │ - decodeDeltas: function(encoded, dimension, opt_factor) { │ │ │ │ │ - var factor = opt_factor || 1e5; │ │ │ │ │ - var d; │ │ │ │ │ + setNodeDimension: function(node, geometry) { │ │ │ │ │ │ │ │ │ │ - var lastNumbers = new Array(dimension); │ │ │ │ │ - for (d = 0; d < dimension; ++d) { │ │ │ │ │ - lastNumbers[d] = 0; │ │ │ │ │ - } │ │ │ │ │ + var bbox = geometry.getBounds(); │ │ │ │ │ + if (bbox) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ │ │ │ │ │ - var numbers = this.decodeFloats(encoded, factor); │ │ │ │ │ + var scaledBox = │ │ │ │ │ + new OpenLayers.Bounds(((bbox.left - this.featureDx) / resolution - this.offset.x) | 0, │ │ │ │ │ + (bbox.bottom / resolution - this.offset.y) | 0, │ │ │ │ │ + ((bbox.right - this.featureDx) / resolution - this.offset.x) | 0, │ │ │ │ │ + (bbox.top / resolution - this.offset.y) | 0); │ │ │ │ │ │ │ │ │ │ - var numbersLength = numbers.length; │ │ │ │ │ - for (var i = 0; i < numbersLength;) { │ │ │ │ │ - for (d = 0; d < dimension; ++d, ++i) { │ │ │ │ │ - lastNumbers[d] += numbers[i]; │ │ │ │ │ + // Set the internal coordinate system to draw the path │ │ │ │ │ + node.style.left = scaledBox.left + "px"; │ │ │ │ │ + node.style.top = scaledBox.top + "px"; │ │ │ │ │ + node.style.width = scaledBox.getWidth() + "px"; │ │ │ │ │ + node.style.height = scaledBox.getHeight() + "px"; │ │ │ │ │ │ │ │ │ │ - numbers[i] = lastNumbers[d]; │ │ │ │ │ - } │ │ │ │ │ + node.coordorigin = scaledBox.left + " " + scaledBox.top; │ │ │ │ │ + node.coordsize = scaledBox.getWidth() + " " + scaledBox.getHeight(); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return numbers; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: encodeFloats │ │ │ │ │ - * Encode a list of floating point numbers and return an encoded string │ │ │ │ │ - * │ │ │ │ │ - * Attention: This function will modify the passed array! │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: dashStyle │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * numbers - {Array.<number>} A list of floating point numbers. │ │ │ │ │ - * opt_factor - {number=} The factor by which the numbers will be │ │ │ │ │ - * multiplied. The remaining decimal places will get rounded away. │ │ │ │ │ - * │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {string} The encoded string. │ │ │ │ │ + * {String} A VML compliant 'stroke-dasharray' value │ │ │ │ │ */ │ │ │ │ │ - encodeFloats: function(numbers, opt_factor) { │ │ │ │ │ - var factor = opt_factor || 1e5; │ │ │ │ │ - │ │ │ │ │ - var numbersLength = numbers.length; │ │ │ │ │ - for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ - numbers[i] = Math.round(numbers[i] * factor); │ │ │ │ │ + dashStyle: function(style) { │ │ │ │ │ + var dash = style.strokeDashstyle; │ │ │ │ │ + switch (dash) { │ │ │ │ │ + case 'solid': │ │ │ │ │ + case 'dot': │ │ │ │ │ + case 'dash': │ │ │ │ │ + case 'dashdot': │ │ │ │ │ + case 'longdash': │ │ │ │ │ + case 'longdashdot': │ │ │ │ │ + return dash; │ │ │ │ │ + default: │ │ │ │ │ + // very basic guessing of dash style patterns │ │ │ │ │ + var parts = dash.split(/[ ,]/); │ │ │ │ │ + if (parts.length == 2) { │ │ │ │ │ + if (1 * parts[0] >= 2 * parts[1]) { │ │ │ │ │ + return "longdash"; │ │ │ │ │ + } │ │ │ │ │ + return (parts[0] == 1 || parts[1] == 1) ? "dot" : "dash"; │ │ │ │ │ + } else if (parts.length == 4) { │ │ │ │ │ + return (1 * parts[0] >= 2 * parts[1]) ? "longdashdot" : │ │ │ │ │ + "dashdot"; │ │ │ │ │ + } │ │ │ │ │ + return "solid"; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return this.encodeSignedIntegers(numbers); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: decodeFloats │ │ │ │ │ - * Decode a list of floating point numbers from an encoded string │ │ │ │ │ + * Method: createNode │ │ │ │ │ + * Create a new node │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * encoded - {string} An encoded string. │ │ │ │ │ - * opt_factor - {number=} The factor by which the result will be divided. │ │ │ │ │ + * type - {String} Kind of node to draw │ │ │ │ │ + * id - {String} Id for node │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array.<number>} A list of floating point numbers. │ │ │ │ │ + * {DOMElement} A new node of the given type and id │ │ │ │ │ */ │ │ │ │ │ - decodeFloats: function(encoded, opt_factor) { │ │ │ │ │ - var factor = opt_factor || 1e5; │ │ │ │ │ - │ │ │ │ │ - var numbers = this.decodeSignedIntegers(encoded); │ │ │ │ │ - │ │ │ │ │ - var numbersLength = numbers.length; │ │ │ │ │ - for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ - numbers[i] /= factor; │ │ │ │ │ + createNode: function(type, id) { │ │ │ │ │ + var node = document.createElement(type); │ │ │ │ │ + if (id) { │ │ │ │ │ + node.id = id; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return numbers; │ │ │ │ │ - }, │ │ │ │ │ + // IE hack to make elements unselectable, to prevent 'blue flash' │ │ │ │ │ + // while dragging vectors; #1410 │ │ │ │ │ + node.unselectable = 'on'; │ │ │ │ │ + node.onselectstart = OpenLayers.Function.False; │ │ │ │ │ │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: encodeSignedIntegers │ │ │ │ │ - * Encode a list of signed integers and return an encoded string │ │ │ │ │ - * │ │ │ │ │ - * Attention: This function will modify the passed array! │ │ │ │ │ + * Method: nodeTypeCompare │ │ │ │ │ + * Determine whether a node is of a given type │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * numbers - {Array.<number>} A list of signed integers. │ │ │ │ │ + * node - {DOMElement} An VML element │ │ │ │ │ + * type - {String} Kind of node │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {string} The encoded string. │ │ │ │ │ + * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ */ │ │ │ │ │ - encodeSignedIntegers: function(numbers) { │ │ │ │ │ - var numbersLength = numbers.length; │ │ │ │ │ - for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ - var num = numbers[i]; │ │ │ │ │ + nodeTypeCompare: function(node, type) { │ │ │ │ │ │ │ │ │ │ - var signedNum = num << 1; │ │ │ │ │ - if (num < 0) { │ │ │ │ │ - signedNum = ~(signedNum); │ │ │ │ │ - } │ │ │ │ │ + //split type │ │ │ │ │ + var subType = type; │ │ │ │ │ + var splitIndex = subType.indexOf(":"); │ │ │ │ │ + if (splitIndex != -1) { │ │ │ │ │ + subType = subType.substr(splitIndex + 1); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - numbers[i] = signedNum; │ │ │ │ │ + //split nodeName │ │ │ │ │ + var nodeName = node.nodeName; │ │ │ │ │ + splitIndex = nodeName.indexOf(":"); │ │ │ │ │ + if (splitIndex != -1) { │ │ │ │ │ + nodeName = nodeName.substr(splitIndex + 1); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return this.encodeUnsignedIntegers(numbers); │ │ │ │ │ + return (subType == nodeName); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: decodeSignedIntegers │ │ │ │ │ - * Decode a list of signed integers from an encoded string │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * encoded - {string} An encoded string. │ │ │ │ │ + * Method: createRenderRoot │ │ │ │ │ + * Create the renderer root │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array.<number>} A list of signed integers. │ │ │ │ │ + * {DOMElement} The specific render engine's root element │ │ │ │ │ */ │ │ │ │ │ - decodeSignedIntegers: function(encoded) { │ │ │ │ │ - var numbers = this.decodeUnsignedIntegers(encoded); │ │ │ │ │ - │ │ │ │ │ - var numbersLength = numbers.length; │ │ │ │ │ - for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ - var num = numbers[i]; │ │ │ │ │ - numbers[i] = (num & 1) ? ~(num >> 1) : (num >> 1); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return numbers; │ │ │ │ │ + createRenderRoot: function() { │ │ │ │ │ + return this.nodeFactory(this.container.id + "_vmlRoot", "div"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: encodeUnsignedIntegers │ │ │ │ │ - * Encode a list of unsigned integers and return an encoded string │ │ │ │ │ - * │ │ │ │ │ + * Method: createRoot │ │ │ │ │ + * Create the main root element │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * numbers - {Array.<number>} A list of unsigned integers. │ │ │ │ │ + * suffix - {String} suffix to append to the id │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {string} The encoded string. │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - encodeUnsignedIntegers: function(numbers) { │ │ │ │ │ - var encoded = ''; │ │ │ │ │ - │ │ │ │ │ - var numbersLength = numbers.length; │ │ │ │ │ - for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ - encoded += this.encodeUnsignedInteger(numbers[i]); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return encoded; │ │ │ │ │ + createRoot: function(suffix) { │ │ │ │ │ + return this.nodeFactory(this.container.id + suffix, "olv:group"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + /************************************** │ │ │ │ │ + * * │ │ │ │ │ + * GEOMETRY DRAWING FUNCTIONS * │ │ │ │ │ + * * │ │ │ │ │ + **************************************/ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: decodeUnsignedIntegers │ │ │ │ │ - * Decode a list of unsigned integers from an encoded string │ │ │ │ │ - * │ │ │ │ │ + * Method: drawPoint │ │ │ │ │ + * Render a point │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * encoded - {string} An encoded string. │ │ │ │ │ - * │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array.<number>} A list of unsigned integers. │ │ │ │ │ + * {DOMElement} or false if the point could not be drawn │ │ │ │ │ */ │ │ │ │ │ - decodeUnsignedIntegers: function(encoded) { │ │ │ │ │ - var numbers = []; │ │ │ │ │ + drawPoint: function(node, geometry) { │ │ │ │ │ + return this.drawCircle(node, geometry, 1); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var current = 0; │ │ │ │ │ - var shift = 0; │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawCircle │ │ │ │ │ + * Render a circle. │ │ │ │ │ + * Size and Center a circle given geometry (x,y center) and radius │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * radius - {float} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the circle could not ne drawn │ │ │ │ │ + */ │ │ │ │ │ + drawCircle: function(node, geometry, radius) { │ │ │ │ │ + if (!isNaN(geometry.x) && !isNaN(geometry.y)) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ │ │ │ │ │ - var encodedLength = encoded.length; │ │ │ │ │ - for (var i = 0; i < encodedLength; ++i) { │ │ │ │ │ - var b = encoded.charCodeAt(i) - 63; │ │ │ │ │ + node.style.left = ((((geometry.x - this.featureDx) / resolution - this.offset.x) | 0) - radius) + "px"; │ │ │ │ │ + node.style.top = (((geometry.y / resolution - this.offset.y) | 0) - radius) + "px"; │ │ │ │ │ │ │ │ │ │ - current |= (b & 0x1f) << shift; │ │ │ │ │ + var diameter = radius * 2; │ │ │ │ │ │ │ │ │ │ - if (b < 0x20) { │ │ │ │ │ - numbers.push(current); │ │ │ │ │ - current = 0; │ │ │ │ │ - shift = 0; │ │ │ │ │ - } else { │ │ │ │ │ - shift += 5; │ │ │ │ │ - } │ │ │ │ │ + node.style.width = diameter + "px"; │ │ │ │ │ + node.style.height = diameter + "px"; │ │ │ │ │ + return node; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return numbers; │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: encodeFloat │ │ │ │ │ - * Encode one single floating point number and return an encoded string │ │ │ │ │ - * │ │ │ │ │ + * Method: drawLineString │ │ │ │ │ + * Render a linestring. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * num - {number} Floating point number that should be encoded. │ │ │ │ │ - * opt_factor - {number=} The factor by which num will be multiplied. │ │ │ │ │ - * The remaining decimal places will get rounded away. │ │ │ │ │ - * │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {string} The encoded string. │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - encodeFloat: function(num, opt_factor) { │ │ │ │ │ - num = Math.round(num * (opt_factor || 1e5)); │ │ │ │ │ - return this.encodeSignedInteger(num); │ │ │ │ │ + drawLineString: function(node, geometry) { │ │ │ │ │ + return this.drawLine(node, geometry, false); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: decodeFloat │ │ │ │ │ - * Decode one single floating point number from an encoded string │ │ │ │ │ - * │ │ │ │ │ + * Method: drawLinearRing │ │ │ │ │ + * Render a linearring │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * encoded - {string} An encoded string. │ │ │ │ │ - * opt_factor - {number=} The factor by which the result will be divided. │ │ │ │ │ - * │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {number} The decoded floating point number. │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - decodeFloat: function(encoded, opt_factor) { │ │ │ │ │ - var result = this.decodeSignedInteger(encoded); │ │ │ │ │ - return result / (opt_factor || 1e5); │ │ │ │ │ + drawLinearRing: function(node, geometry) { │ │ │ │ │ + return this.drawLine(node, geometry, true); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: encodeSignedInteger │ │ │ │ │ - * Encode one single signed integer and return an encoded string │ │ │ │ │ - * │ │ │ │ │ + * Method: DrawLine │ │ │ │ │ + * Render a line. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * num - {number} Signed integer that should be encoded. │ │ │ │ │ - * │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * closeLine - {Boolean} Close the line? (make it a ring?) │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {string} The encoded string. │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - encodeSignedInteger: function(num) { │ │ │ │ │ - var signedNum = num << 1; │ │ │ │ │ - if (num < 0) { │ │ │ │ │ - signedNum = ~(signedNum); │ │ │ │ │ - } │ │ │ │ │ + drawLine: function(node, geometry, closeLine) { │ │ │ │ │ │ │ │ │ │ - return this.encodeUnsignedInteger(signedNum); │ │ │ │ │ + this.setNodeDimension(node, geometry); │ │ │ │ │ + │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var numComponents = geometry.components.length; │ │ │ │ │ + var parts = new Array(numComponents); │ │ │ │ │ + │ │ │ │ │ + var comp, x, y; │ │ │ │ │ + for (var i = 0; i < numComponents; i++) { │ │ │ │ │ + comp = geometry.components[i]; │ │ │ │ │ + x = ((comp.x - this.featureDx) / resolution - this.offset.x) | 0; │ │ │ │ │ + y = (comp.y / resolution - this.offset.y) | 0; │ │ │ │ │ + parts[i] = " " + x + "," + y + " l "; │ │ │ │ │ + } │ │ │ │ │ + var end = (closeLine) ? " x e" : " e"; │ │ │ │ │ + node.path = "m" + parts.join("") + end; │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawPolygon │ │ │ │ │ + * Render a polygon │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + drawPolygon: function(node, geometry) { │ │ │ │ │ + this.setNodeDimension(node, geometry); │ │ │ │ │ + │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + │ │ │ │ │ + var path = []; │ │ │ │ │ + var j, jj, points, area, first, second, i, ii, comp, pathComp, x, y; │ │ │ │ │ + for (j = 0, jj = geometry.components.length; j < jj; j++) { │ │ │ │ │ + path.push("m"); │ │ │ │ │ + points = geometry.components[j].components; │ │ │ │ │ + // we only close paths of interior rings with area │ │ │ │ │ + area = (j === 0); │ │ │ │ │ + first = null; │ │ │ │ │ + second = null; │ │ │ │ │ + for (i = 0, ii = points.length; i < ii; i++) { │ │ │ │ │ + comp = points[i]; │ │ │ │ │ + x = ((comp.x - this.featureDx) / resolution - this.offset.x) | 0; │ │ │ │ │ + y = (comp.y / resolution - this.offset.y) | 0; │ │ │ │ │ + pathComp = " " + x + "," + y; │ │ │ │ │ + path.push(pathComp); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + path.push(" l"); │ │ │ │ │ + } │ │ │ │ │ + if (!area) { │ │ │ │ │ + // IE improperly renders sub-paths that have no area. │ │ │ │ │ + // Instead of checking the area of every ring, we confirm │ │ │ │ │ + // the ring has at least three distinct points. This does │ │ │ │ │ + // not catch all non-zero area cases, but it greatly improves │ │ │ │ │ + // interior ring digitizing and is a minor performance hit │ │ │ │ │ + // when rendering rings with many points. │ │ │ │ │ + if (!first) { │ │ │ │ │ + first = pathComp; │ │ │ │ │ + } else if (first != pathComp) { │ │ │ │ │ + if (!second) { │ │ │ │ │ + second = pathComp; │ │ │ │ │ + } else if (second != pathComp) { │ │ │ │ │ + // stop looking │ │ │ │ │ + area = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + path.push(area ? " x " : " "); │ │ │ │ │ + } │ │ │ │ │ + path.push("e"); │ │ │ │ │ + node.path = path.join(""); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: decodeSignedInteger │ │ │ │ │ - * Decode one single signed integer from an encoded string │ │ │ │ │ - * │ │ │ │ │ + * Method: drawRectangle │ │ │ │ │ + * Render a rectangle │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * encoded - {string} An encoded string. │ │ │ │ │ - * │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {number} The decoded signed integer. │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - decodeSignedInteger: function(encoded) { │ │ │ │ │ - var result = this.decodeUnsignedInteger(encoded); │ │ │ │ │ - return ((result & 1) ? ~(result >> 1) : (result >> 1)); │ │ │ │ │ + drawRectangle: function(node, geometry) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + │ │ │ │ │ + node.style.left = (((geometry.x - this.featureDx) / resolution - this.offset.x) | 0) + "px"; │ │ │ │ │ + node.style.top = ((geometry.y / resolution - this.offset.y) | 0) + "px"; │ │ │ │ │ + node.style.width = ((geometry.width / resolution) | 0) + "px"; │ │ │ │ │ + node.style.height = ((geometry.height / resolution) | 0) + "px"; │ │ │ │ │ + │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawText │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * style - │ │ │ │ │ + * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + */ │ │ │ │ │ + drawText: function(featureId, style, location) { │ │ │ │ │ + var label = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX, "olv:rect"); │ │ │ │ │ + var textbox = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX + "_textbox", "olv:textbox"); │ │ │ │ │ + │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + label.style.left = (((location.x - this.featureDx) / resolution - this.offset.x) | 0) + "px"; │ │ │ │ │ + label.style.top = ((location.y / resolution - this.offset.y) | 0) + "px"; │ │ │ │ │ + label.style.flip = "y"; │ │ │ │ │ + │ │ │ │ │ + textbox.innerText = style.label; │ │ │ │ │ + │ │ │ │ │ + if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ + textbox.style.cursor = style.cursor; │ │ │ │ │ + } │ │ │ │ │ + if (style.fontColor) { │ │ │ │ │ + textbox.style.color = style.fontColor; │ │ │ │ │ + } │ │ │ │ │ + if (style.fontOpacity) { │ │ │ │ │ + textbox.style.filter = 'alpha(opacity=' + (style.fontOpacity * 100) + ')'; │ │ │ │ │ + } │ │ │ │ │ + if (style.fontFamily) { │ │ │ │ │ + textbox.style.fontFamily = style.fontFamily; │ │ │ │ │ + } │ │ │ │ │ + if (style.fontSize) { │ │ │ │ │ + textbox.style.fontSize = style.fontSize; │ │ │ │ │ + } │ │ │ │ │ + if (style.fontWeight) { │ │ │ │ │ + textbox.style.fontWeight = style.fontWeight; │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStyle) { │ │ │ │ │ + textbox.style.fontStyle = style.fontStyle; │ │ │ │ │ + } │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + label._featureId = featureId; │ │ │ │ │ + textbox._featureId = featureId; │ │ │ │ │ + textbox._geometry = location; │ │ │ │ │ + textbox._geometryClass = location.CLASS_NAME; │ │ │ │ │ + } │ │ │ │ │ + textbox.style.whiteSpace = "nowrap"; │ │ │ │ │ + // fun with IE: IE7 in standards compliant mode does not display any │ │ │ │ │ + // text with a left inset of 0. So we set this to 1px and subtract one │ │ │ │ │ + // pixel later when we set label.style.left │ │ │ │ │ + textbox.inset = "1px,0px,0px,0px"; │ │ │ │ │ + │ │ │ │ │ + if (!label.parentNode) { │ │ │ │ │ + label.appendChild(textbox); │ │ │ │ │ + this.textRoot.appendChild(label); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var align = style.labelAlign || "cm"; │ │ │ │ │ + if (align.length == 1) { │ │ │ │ │ + align += "m"; │ │ │ │ │ + } │ │ │ │ │ + var xshift = textbox.clientWidth * │ │ │ │ │ + (OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(0, 1)]); │ │ │ │ │ + var yshift = textbox.clientHeight * │ │ │ │ │ + (OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(1, 1)]); │ │ │ │ │ + label.style.left = parseInt(label.style.left) - xshift - 1 + "px"; │ │ │ │ │ + label.style.top = parseInt(label.style.top) + yshift + "px"; │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: encodeUnsignedInteger │ │ │ │ │ - * Encode one single unsigned integer and return an encoded string │ │ │ │ │ - * │ │ │ │ │ + * Method: moveRoot │ │ │ │ │ + * moves this renderer's root to a different renderer. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * num - {number} Unsigned integer that should be encoded. │ │ │ │ │ - * │ │ │ │ │ + * renderer - {<OpenLayers.Renderer>} target renderer for the moved root │ │ │ │ │ + * root - {DOMElement} optional root node. To be used when this renderer │ │ │ │ │ + * holds roots from multiple layers to tell this method which one to │ │ │ │ │ + * detach │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {string} The encoded string. │ │ │ │ │ + * {Boolean} true if successful, false otherwise │ │ │ │ │ */ │ │ │ │ │ - encodeUnsignedInteger: function(num) { │ │ │ │ │ - var value, encoded = ''; │ │ │ │ │ - while (num >= 0x20) { │ │ │ │ │ - value = (0x20 | (num & 0x1f)) + 63; │ │ │ │ │ - encoded += (String.fromCharCode(value)); │ │ │ │ │ - num >>= 5; │ │ │ │ │ + moveRoot: function(renderer) { │ │ │ │ │ + var layer = this.map.getLayer(renderer.container.id); │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Vector.RootContainer) { │ │ │ │ │ + layer = this.map.getLayer(this.container.id); │ │ │ │ │ } │ │ │ │ │ - value = num + 63; │ │ │ │ │ - encoded += (String.fromCharCode(value)); │ │ │ │ │ - return encoded; │ │ │ │ │ + layer && layer.renderer.clear(); │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.moveRoot.apply(this, arguments); │ │ │ │ │ + layer && layer.redraw(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: decodeUnsignedInteger │ │ │ │ │ - * Decode one single unsigned integer from an encoded string │ │ │ │ │ - * │ │ │ │ │ + * Method: importSymbol │ │ │ │ │ + * add a new symbol definition from the rendererer's symbol hash │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * encoded - {string} An encoded string. │ │ │ │ │ - * │ │ │ │ │ + * graphicName - {String} name of the symbol to import │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {number} The decoded unsigned integer. │ │ │ │ │ + * {Object} - hash of {DOMElement} "symbol" and {Number} "size" │ │ │ │ │ */ │ │ │ │ │ - decodeUnsignedInteger: function(encoded) { │ │ │ │ │ - var result = 0; │ │ │ │ │ - var shift = 0; │ │ │ │ │ + importSymbol: function(graphicName) { │ │ │ │ │ + var id = this.container.id + "-" + graphicName; │ │ │ │ │ │ │ │ │ │ - var encodedLength = encoded.length; │ │ │ │ │ - for (var i = 0; i < encodedLength; ++i) { │ │ │ │ │ - var b = encoded.charCodeAt(i) - 63; │ │ │ │ │ + // check if symbol already exists in the cache │ │ │ │ │ + var cache = this.symbolCache[id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + return cache; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - result |= (b & 0x1f) << shift; │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(graphicName + ' is not a valid symbol name'); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (b < 0x20) │ │ │ │ │ - break; │ │ │ │ │ + var symbolExtent = new OpenLayers.Bounds( │ │ │ │ │ + Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ │ │ │ │ │ - shift += 5; │ │ │ │ │ + var pathitems = ["m"]; │ │ │ │ │ + for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + var x = symbol[i]; │ │ │ │ │ + var y = symbol[i + 1]; │ │ │ │ │ + symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ + symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ + symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ + symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ + │ │ │ │ │ + pathitems.push(x); │ │ │ │ │ + pathitems.push(y); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + pathitems.push("l"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + pathitems.push("x e"); │ │ │ │ │ + var path = pathitems.join(" "); │ │ │ │ │ + │ │ │ │ │ + var diff = (symbolExtent.getWidth() - symbolExtent.getHeight()) / 2; │ │ │ │ │ + if (diff > 0) { │ │ │ │ │ + symbolExtent.bottom = symbolExtent.bottom - diff; │ │ │ │ │ + symbolExtent.top = symbolExtent.top + diff; │ │ │ │ │ + } else { │ │ │ │ │ + symbolExtent.left = symbolExtent.left + diff; │ │ │ │ │ + symbolExtent.right = symbolExtent.right - diff; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return result; │ │ │ │ │ + cache = { │ │ │ │ │ + path: path, │ │ │ │ │ + size: symbolExtent.getWidth(), // equals getHeight() now │ │ │ │ │ + left: symbolExtent.left, │ │ │ │ │ + bottom: symbolExtent.bottom │ │ │ │ │ + }; │ │ │ │ │ + this.symbolCache[id] = cache; │ │ │ │ │ + │ │ │ │ │ + return cache; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.EncodedPolyline" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.VML" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.VML.LABEL_SHIFT │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.VML.LABEL_SHIFT = { │ │ │ │ │ + "l": 0, │ │ │ │ │ + "c": .5, │ │ │ │ │ + "r": 1, │ │ │ │ │ + "t": 0, │ │ │ │ │ + "m": .5, │ │ │ │ │ + "b": 1 │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSGetFeatureInfo.js │ │ │ │ │ + OpenLayers/Renderer/SVG.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Renderer/Elements.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WMSGetFeatureInfo │ │ │ │ │ - * Class to read GetFeatureInfo responses from Web Mapping Services │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * Class: OpenLayers.Renderer.SVG │ │ │ │ │ + * │ │ │ │ │ + * Inherits: │ │ │ │ │ + * - <OpenLayers.Renderer.Elements> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WMSGetFeatureInfo = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Renderer.SVG = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: layerIdentifier │ │ │ │ │ - * {String} All xml nodes containing this search criteria will populate an │ │ │ │ │ - * internal array of layer nodes. │ │ │ │ │ + /** │ │ │ │ │ + * Property: xmlns │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - layerIdentifier: '_layer', │ │ │ │ │ + xmlns: "http://www.w3.org/2000/svg", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: featureIdentifier │ │ │ │ │ - * {String} All xml nodes containing this search criteria will populate an │ │ │ │ │ - * internal array of feature nodes for each layer node found. │ │ │ │ │ + * Property: xlinkns │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - featureIdentifier: '_feature', │ │ │ │ │ + xlinkns: "http://www.w3.org/1999/xlink", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ + * Constant: MAX_PIXEL │ │ │ │ │ + * {Integer} Firefox has a limitation where values larger or smaller than │ │ │ │ │ + * about 15000 in an SVG document lock the browser up. This │ │ │ │ │ + * works around it. │ │ │ │ │ */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ - }, │ │ │ │ │ + MAX_PIXEL: 15000, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: gmlFormat │ │ │ │ │ - * {<OpenLayers.Format.GML>} internal GML format for parsing geometries │ │ │ │ │ - * in msGMLOutput │ │ │ │ │ + * Property: translationParameters │ │ │ │ │ + * {Object} Hash with "x" and "y" properties │ │ │ │ │ */ │ │ │ │ │ - gmlFormat: null, │ │ │ │ │ + translationParameters: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMSGetFeatureInfo │ │ │ │ │ - * Create a new parser for WMS GetFeatureInfo responses │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * Property: symbolMetrics │ │ │ │ │ + * {Object} Cache for symbol metrics according to their svg coordinate │ │ │ │ │ + * space. This is an object keyed by the symbol's id, and values are │ │ │ │ │ + * an array of [width, centerX, centerY]. │ │ │ │ │ */ │ │ │ │ │ + symbolMetrics: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read WMS GetFeatureInfo data from a string, and return an array of features │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Renderer.SVG │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ + * containerID - {String} │ │ │ │ │ */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - var result; │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - var root = data.documentElement; │ │ │ │ │ - if (root) { │ │ │ │ │ - var scope = this; │ │ │ │ │ - var read = this["read_" + root.nodeName]; │ │ │ │ │ - if (read) { │ │ │ │ │ - result = read.call(this, root); │ │ │ │ │ - } else { │ │ │ │ │ - // fall-back to GML since this is a common output format for WMS │ │ │ │ │ - // GetFeatureInfo responses │ │ │ │ │ - result = new OpenLayers.Format.GML((this.options ? this.options : {})).read(data); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - result = data; │ │ │ │ │ + initialize: function(containerID) { │ │ │ │ │ + if (!this.supported()) { │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ - return result; │ │ │ │ │ - }, │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.initialize.apply(this, │ │ │ │ │ + arguments); │ │ │ │ │ + this.translationParameters = { │ │ │ │ │ + x: 0, │ │ │ │ │ + y: 0 │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ + this.symbolMetrics = {}; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_msGMLOutput │ │ │ │ │ - * Parse msGMLOutput nodes. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: supported │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array} │ │ │ │ │ + * {Boolean} Whether or not the browser supports the SVG renderer │ │ │ │ │ */ │ │ │ │ │ - read_msGMLOutput: function(data) { │ │ │ │ │ - var response = []; │ │ │ │ │ - var layerNodes = this.getSiblingNodesByTagCriteria(data, │ │ │ │ │ - this.layerIdentifier); │ │ │ │ │ - if (layerNodes) { │ │ │ │ │ - for (var i = 0, len = layerNodes.length; i < len; ++i) { │ │ │ │ │ - var node = layerNodes[i]; │ │ │ │ │ - var layerName = node.nodeName; │ │ │ │ │ - if (node.prefix) { │ │ │ │ │ - layerName = layerName.split(':')[1]; │ │ │ │ │ - } │ │ │ │ │ - var layerName = layerName.replace(this.layerIdentifier, ''); │ │ │ │ │ - var featureNodes = this.getSiblingNodesByTagCriteria(node, │ │ │ │ │ - this.featureIdentifier); │ │ │ │ │ - if (featureNodes) { │ │ │ │ │ - for (var j = 0; j < featureNodes.length; j++) { │ │ │ │ │ - var featureNode = featureNodes[j]; │ │ │ │ │ - var geomInfo = this.parseGeometry(featureNode); │ │ │ │ │ - var attributes = this.parseAttributes(featureNode); │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geomInfo.geometry, │ │ │ │ │ - attributes, null); │ │ │ │ │ - feature.bounds = geomInfo.bounds; │ │ │ │ │ - feature.type = layerName; │ │ │ │ │ - response.push(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return response; │ │ │ │ │ + supported: function() { │ │ │ │ │ + var svgFeature = "http://www.w3.org/TR/SVG11/feature#"; │ │ │ │ │ + return (document.implementation && │ │ │ │ │ + (document.implementation.hasFeature("org.w3c.svg", "1.0") || │ │ │ │ │ + document.implementation.hasFeature(svgFeature + "SVG", "1.1") || │ │ │ │ │ + document.implementation.hasFeature(svgFeature + "BasicStructure", "1.1"))); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_FeatureInfoResponse │ │ │ │ │ - * Parse FeatureInfoResponse nodes. │ │ │ │ │ + * Method: inValidRange │ │ │ │ │ + * See #669 for more information │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ + * x - {Integer} │ │ │ │ │ + * y - {Integer} │ │ │ │ │ + * xyOnly - {Boolean} whether or not to just check for x and y, which means │ │ │ │ │ + * to not take the current translation parameters into account if true. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array} │ │ │ │ │ + * {Boolean} Whether or not the 'x' and 'y' coordinates are in the │ │ │ │ │ + * valid range. │ │ │ │ │ */ │ │ │ │ │ - read_FeatureInfoResponse: function(data) { │ │ │ │ │ - var response = []; │ │ │ │ │ - var featureNodes = this.getElementsByTagNameNS(data, '*', │ │ │ │ │ - 'FIELDS'); │ │ │ │ │ - │ │ │ │ │ - for (var i = 0, len = featureNodes.length; i < len; i++) { │ │ │ │ │ - var featureNode = featureNodes[i]; │ │ │ │ │ - var geom = null; │ │ │ │ │ - │ │ │ │ │ - // attributes can be actual attributes on the FIELDS tag, │ │ │ │ │ - // or FIELD children │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - var j; │ │ │ │ │ - var jlen = featureNode.attributes.length; │ │ │ │ │ - if (jlen > 0) { │ │ │ │ │ - for (j = 0; j < jlen; j++) { │ │ │ │ │ - var attribute = featureNode.attributes[j]; │ │ │ │ │ - attributes[attribute.nodeName] = attribute.nodeValue; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - var nodes = featureNode.childNodes; │ │ │ │ │ - for (j = 0, jlen = nodes.length; j < jlen; ++j) { │ │ │ │ │ - var node = nodes[j]; │ │ │ │ │ - if (node.nodeType != 3) { │ │ │ │ │ - attributes[node.getAttribute("name")] = │ │ │ │ │ - node.getAttribute("value"); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - response.push( │ │ │ │ │ - new OpenLayers.Feature.Vector(geom, attributes, null) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return response; │ │ │ │ │ + inValidRange: function(x, y, xyOnly) { │ │ │ │ │ + var left = x + (xyOnly ? 0 : this.translationParameters.x); │ │ │ │ │ + var top = y + (xyOnly ? 0 : this.translationParameters.y); │ │ │ │ │ + return (left >= -this.MAX_PIXEL && left <= this.MAX_PIXEL && │ │ │ │ │ + top >= -this.MAX_PIXEL && top <= this.MAX_PIXEL); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getSiblingNodesByTagCriteria │ │ │ │ │ - * Recursively searches passed xml node and all it's descendant levels for │ │ │ │ │ - * nodes whose tagName contains the passed search string. This returns an │ │ │ │ │ - * array of all sibling nodes which match the criteria from the highest │ │ │ │ │ - * hierarchial level from which a match is found. │ │ │ │ │ + * Method: setExtent │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} An xml node │ │ │ │ │ - * criteria - {String} Search string which will match some part of a tagName │ │ │ │ │ - * │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} │ │ │ │ │ + * resolutionChanged - {Boolean} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * Array({DOMElement}) An array of sibling xml nodes │ │ │ │ │ + * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ + * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ + * False otherwise. │ │ │ │ │ */ │ │ │ │ │ - getSiblingNodesByTagCriteria: function(node, criteria) { │ │ │ │ │ - var nodes = []; │ │ │ │ │ - var children, tagName, n, matchNodes, child; │ │ │ │ │ - if (node && node.hasChildNodes()) { │ │ │ │ │ - children = node.childNodes; │ │ │ │ │ - n = children.length; │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - for (var k = 0; k < n; k++) { │ │ │ │ │ - child = children[k]; │ │ │ │ │ - while (child && child.nodeType != 1) { │ │ │ │ │ - child = child.nextSibling; │ │ │ │ │ - k++; │ │ │ │ │ - } │ │ │ │ │ - tagName = (child ? child.nodeName : ''); │ │ │ │ │ - if (tagName.length > 0 && tagName.indexOf(criteria) > -1) { │ │ │ │ │ - nodes.push(child); │ │ │ │ │ - } else { │ │ │ │ │ - matchNodes = this.getSiblingNodesByTagCriteria( │ │ │ │ │ - child, criteria); │ │ │ │ │ + var resolution = this.getResolution(), │ │ │ │ │ + left = -extent.left / resolution, │ │ │ │ │ + top = extent.top / resolution; │ │ │ │ │ │ │ │ │ │ - if (matchNodes.length > 0) { │ │ │ │ │ - (nodes.length == 0) ? │ │ │ │ │ - nodes = matchNodes: nodes.push(matchNodes); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + // If the resolution has changed, start over changing the corner, because │ │ │ │ │ + // the features will redraw. │ │ │ │ │ + if (resolutionChanged) { │ │ │ │ │ + this.left = left; │ │ │ │ │ + this.top = top; │ │ │ │ │ + // Set the viewbox │ │ │ │ │ + var extentString = "0 0 " + this.size.w + " " + this.size.h; │ │ │ │ │ │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "viewBox", extentString); │ │ │ │ │ + this.translate(this.xOffset, 0); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + var inRange = this.translate(left - this.left + this.xOffset, top - this.top); │ │ │ │ │ + if (!inRange) { │ │ │ │ │ + // recenter the coordinate system │ │ │ │ │ + this.setExtent(extent, true); │ │ │ │ │ + } │ │ │ │ │ + return coordSysUnchanged && inRange; │ │ │ │ │ } │ │ │ │ │ - return nodes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseAttributes │ │ │ │ │ - * │ │ │ │ │ + * Method: translate │ │ │ │ │ + * Transforms the SVG coordinate system │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {<DOMElement>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An attributes object. │ │ │ │ │ + * x - {Float} │ │ │ │ │ + * y - {Float} │ │ │ │ │ * │ │ │ │ │ - * Notes: │ │ │ │ │ - * Assumes that attributes are direct child xml nodes of the passed node │ │ │ │ │ - * and contain only a single text node. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} true if the translation parameters are in the valid coordinates │ │ │ │ │ + * range, false otherwise. │ │ │ │ │ */ │ │ │ │ │ - parseAttributes: function(node) { │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - if (node.nodeType == 1) { │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - var n = children.length; │ │ │ │ │ - for (var i = 0; i < n; ++i) { │ │ │ │ │ - var child = children[i]; │ │ │ │ │ - if (child.nodeType == 1) { │ │ │ │ │ - var grandchildren = child.childNodes; │ │ │ │ │ - var name = (child.prefix) ? │ │ │ │ │ - child.nodeName.split(":")[1] : child.nodeName; │ │ │ │ │ - if (grandchildren.length == 0) { │ │ │ │ │ - attributes[name] = null; │ │ │ │ │ - } else if (grandchildren.length == 1) { │ │ │ │ │ - var grandchild = grandchildren[0]; │ │ │ │ │ - if (grandchild.nodeType == 3 || │ │ │ │ │ - grandchild.nodeType == 4) { │ │ │ │ │ - var value = grandchild.nodeValue.replace( │ │ │ │ │ - this.regExes.trimSpace, ""); │ │ │ │ │ - attributes[name] = value; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + translate: function(x, y) { │ │ │ │ │ + if (!this.inValidRange(x, y, true)) { │ │ │ │ │ + return false; │ │ │ │ │ + } else { │ │ │ │ │ + var transformString = ""; │ │ │ │ │ + if (x || y) { │ │ │ │ │ + transformString = "translate(" + x + "," + y + ")"; │ │ │ │ │ } │ │ │ │ │ + this.root.setAttributeNS(null, "transform", transformString); │ │ │ │ │ + this.translationParameters = { │ │ │ │ │ + x: x, │ │ │ │ │ + y: y │ │ │ │ │ + }; │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ - return attributes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseGeometry │ │ │ │ │ - * Parse the geometry and the feature bounds out of the node using │ │ │ │ │ - * Format.GML │ │ │ │ │ - * │ │ │ │ │ + * Method: setSize │ │ │ │ │ + * Sets the size of the drawing surface. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {<DOMElement>} │ │ │ │ │ - * │ │ │ │ │ + * size - {<OpenLayers.Size>} The size of the drawing surface │ │ │ │ │ + */ │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "width", this.size.w); │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "height", this.size.h); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getNodeType │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} An object containing the geometry and the feature bounds │ │ │ │ │ + * {String} The corresponding node type for the specified geometry │ │ │ │ │ */ │ │ │ │ │ - parseGeometry: function(node) { │ │ │ │ │ - // we need to use the old Format.GML parser since we do not know the │ │ │ │ │ - // geometry name │ │ │ │ │ - if (!this.gmlFormat) { │ │ │ │ │ - this.gmlFormat = new OpenLayers.Format.GML(); │ │ │ │ │ - } │ │ │ │ │ - var feature = this.gmlFormat.parseFeature(node); │ │ │ │ │ - var geometry, bounds = null; │ │ │ │ │ - if (feature) { │ │ │ │ │ - geometry = feature.geometry && feature.geometry.clone(); │ │ │ │ │ - bounds = feature.bounds && feature.bounds.clone(); │ │ │ │ │ - feature.destroy(); │ │ │ │ │ + getNodeType: function(geometry, style) { │ │ │ │ │ + var nodeType = null; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + nodeType = "image"; │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + nodeType = "svg"; │ │ │ │ │ + } else { │ │ │ │ │ + nodeType = "circle"; │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + nodeType = "rect"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + nodeType = "polyline"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + nodeType = "polygon"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + case "OpenLayers.Geometry.Curve": │ │ │ │ │ + nodeType = "path"; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ - return { │ │ │ │ │ - geometry: geometry, │ │ │ │ │ - bounds: bounds │ │ │ │ │ - }; │ │ │ │ │ + return nodeType; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSGetFeatureInfo" │ │ │ │ │ + /** │ │ │ │ │ + * Method: setStyle │ │ │ │ │ + * Use to set all the style attributes to a SVG node. │ │ │ │ │ + * │ │ │ │ │ + * Takes care to adjust stroke width and point radius to be │ │ │ │ │ + * resolution-relative │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {SVGDomElement} An SVG element to decorate │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * options - {Object} Currently supported options include │ │ │ │ │ + * 'isFilled' {Boolean} and │ │ │ │ │ + * 'isStroked' {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + setStyle: function(node, style, options) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + options = options || node._options; │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/CSWGetDomain.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + node.setAttributeNS(null, "title", title); │ │ │ │ │ + //Standards-conformant SVG │ │ │ │ │ + // Prevent duplicate nodes. See issue https://github.com/openlayers/openlayers/issues/92 │ │ │ │ │ + var titleNode = node.getElementsByTagName("title"); │ │ │ │ │ + if (titleNode.length > 0) { │ │ │ │ │ + titleNode[0].firstChild.textContent = title; │ │ │ │ │ + } else { │ │ │ │ │ + var label = this.nodeFactory(null, "title"); │ │ │ │ │ + label.textContent = title; │ │ │ │ │ + node.appendChild(label); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + var r = parseFloat(node.getAttributeNS(null, "r")); │ │ │ │ │ + var widthFactor = 1; │ │ │ │ │ + var pos; │ │ │ │ │ + if (node._geometryClass == "OpenLayers.Geometry.Point" && r) { │ │ │ │ │ + node.style.visibility = ""; │ │ │ │ │ + if (style.graphic === false) { │ │ │ │ │ + node.style.visibility = "hidden"; │ │ │ │ │ + } else if (style.externalGraphic) { │ │ │ │ │ + pos = this.getPosition(node); │ │ │ │ │ + if (style.graphicWidth && style.graphicHeight) { │ │ │ │ │ + node.setAttributeNS(null, "preserveAspectRatio", "none"); │ │ │ │ │ + } │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ + var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ + style.graphicXOffset : -(0.5 * width); │ │ │ │ │ + var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ + style.graphicYOffset : -(0.5 * height); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format.js │ │ │ │ │ - */ │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.CSWGetDomain │ │ │ │ │ - * Default version is 2.0.2. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Format>} A CSWGetDomain format of the given version. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.CSWGetDomain = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options, OpenLayers.Format.CSWGetDomain.DEFAULTS │ │ │ │ │ - ); │ │ │ │ │ - var cls = OpenLayers.Format.CSWGetDomain["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported CSWGetDomain version: " + options.version; │ │ │ │ │ - } │ │ │ │ │ - return new cls(options); │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: DEFAULTS │ │ │ │ │ - * {Object} Default properties for the CSWGetDomain format. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.CSWGetDomain.DEFAULTS = { │ │ │ │ │ - "version": "2.0.2" │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/QueryStringFilter.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Console.js │ │ │ │ │ - * @requires OpenLayers/Format.js │ │ │ │ │ - * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ - * @requires OpenLayers/Filter/Comparison.js │ │ │ │ │ - * @requires OpenLayers/Filter/Logical.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.QueryStringFilter │ │ │ │ │ - * Parser for reading a query string and creating a simple filter. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.QueryStringFilter = (function() { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Map the OpenLayers.Filter.Comparison types to the operation strings of │ │ │ │ │ - * the protocol. │ │ │ │ │ - */ │ │ │ │ │ - var cmpToStr = {}; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.EQUAL_TO] = "eq"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.NOT_EQUAL_TO] = "ne"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.LESS_THAN] = "lt"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO] = "lte"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.GREATER_THAN] = "gt"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO] = "gte"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.LIKE] = "ilike"; │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Function: regex2value │ │ │ │ │ - * Convert the value from a regular expression string to a LIKE/ILIKE │ │ │ │ │ - * string known to the web service. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * value - {String} The regex string. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The converted string. │ │ │ │ │ - */ │ │ │ │ │ - function regex2value(value) { │ │ │ │ │ - │ │ │ │ │ - // highly sensitive!! Do not change this without running the │ │ │ │ │ - // Protocol/HTTP.html unit tests │ │ │ │ │ - │ │ │ │ │ - // convert % to \% │ │ │ │ │ - value = value.replace(/%/g, "\\%"); │ │ │ │ │ - │ │ │ │ │ - // convert \\. to \\_ (\\.* occurences converted later) │ │ │ │ │ - value = value.replace(/\\\\\.(\*)?/g, function($0, $1) { │ │ │ │ │ - return $1 ? $0 : "\\\\_"; │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - // convert \\.* to \\% │ │ │ │ │ - value = value.replace(/\\\\\.\*/g, "\\\\%"); │ │ │ │ │ - │ │ │ │ │ - // convert . to _ (\. and .* occurences converted later) │ │ │ │ │ - value = value.replace(/(\\)?\.(\*)?/g, function($0, $1, $2) { │ │ │ │ │ - return $1 || $2 ? $0 : "_"; │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - // convert .* to % (\.* occurnces converted later) │ │ │ │ │ - value = value.replace(/(\\)?\.\*/g, function($0, $1) { │ │ │ │ │ - return $1 ? $0 : "%"; │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - // convert \. to . │ │ │ │ │ - value = value.replace(/\\\./g, "."); │ │ │ │ │ - │ │ │ │ │ - // replace \* with * (watching out for \\*) │ │ │ │ │ - value = value.replace(/(\\)?\\\*/g, function($0, $1) { │ │ │ │ │ - return $1 ? $0 : "*"; │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - return value; │ │ │ │ │ - } │ │ │ │ │ + node.setAttributeNS(null, "x", (pos.x + xOffset).toFixed()); │ │ │ │ │ + node.setAttributeNS(null, "y", (pos.y + yOffset).toFixed()); │ │ │ │ │ + node.setAttributeNS(null, "width", width); │ │ │ │ │ + node.setAttributeNS(null, "height", height); │ │ │ │ │ + node.setAttributeNS(this.xlinkns, "xlink:href", style.externalGraphic); │ │ │ │ │ + node.setAttributeNS(null, "style", "opacity: " + opacity); │ │ │ │ │ + node.onclick = OpenLayers.Event.preventDefault; │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + // the symbol viewBox is three times as large as the symbol │ │ │ │ │ + var offset = style.pointRadius * 3; │ │ │ │ │ + var size = offset * 2; │ │ │ │ │ + var src = this.importSymbol(style.graphicName); │ │ │ │ │ + pos = this.getPosition(node); │ │ │ │ │ + widthFactor = this.symbolMetrics[src.id][0] * 3 / size; │ │ │ │ │ │ │ │ │ │ - return OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ + // remove the node from the dom before we modify it. This │ │ │ │ │ + // prevents various rendering issues in Safari and FF │ │ │ │ │ + var parent = node.parentNode; │ │ │ │ │ + var nextSibling = node.nextSibling; │ │ │ │ │ + if (parent) { │ │ │ │ │ + parent.removeChild(node); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: wildcarded. │ │ │ │ │ - * {Boolean} If true percent signs are added around values │ │ │ │ │ - * read from LIKE filters, for example if the protocol │ │ │ │ │ - * read method is passed a LIKE filter whose property │ │ │ │ │ - * is "foo" and whose value is "bar" the string │ │ │ │ │ - * "foo__ilike=%bar%" will be sent in the query string; │ │ │ │ │ - * defaults to false. │ │ │ │ │ - */ │ │ │ │ │ - wildcarded: false, │ │ │ │ │ + // The more appropriate way to implement this would be use/defs, │ │ │ │ │ + // but due to various issues in several browsers, it is safer to │ │ │ │ │ + // copy the symbols instead of referencing them. │ │ │ │ │ + // See e.g. ticket http://trac.osgeo.org/openlayers/ticket/2985 │ │ │ │ │ + // and this email thread │ │ │ │ │ + // http://osgeo-org.1803224.n2.nabble.com/Select-Control-Ctrl-click-on-Feature-with-a-graphicName-opens-new-browser-window-tc5846039.html │ │ │ │ │ + node.firstChild && node.removeChild(node.firstChild); │ │ │ │ │ + node.appendChild(src.firstChild.cloneNode(true)); │ │ │ │ │ + node.setAttributeNS(null, "viewBox", src.getAttributeNS(null, "viewBox")); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: srsInBBOX │ │ │ │ │ - * {Boolean} Include the SRS identifier in BBOX query string parameter. │ │ │ │ │ - * Default is false. If true and the layer has a projection object set, │ │ │ │ │ - * any BBOX filter will be serialized with a fifth item identifying the │ │ │ │ │ - * projection. E.g. bbox=-1000,-1000,1000,1000,EPSG:900913 │ │ │ │ │ - */ │ │ │ │ │ - srsInBBOX: false, │ │ │ │ │ + node.setAttributeNS(null, "width", size); │ │ │ │ │ + node.setAttributeNS(null, "height", size); │ │ │ │ │ + node.setAttributeNS(null, "x", pos.x - offset); │ │ │ │ │ + node.setAttributeNS(null, "y", pos.y - offset); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Serialize an <OpenLayers.Filter> objects using the "simple" filter syntax for │ │ │ │ │ - * query string parameters. This function must be called as a method of │ │ │ │ │ - * a protocol instance. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} filter to convert. │ │ │ │ │ - * params - {Object} The parameters object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The resulting parameters object. │ │ │ │ │ - */ │ │ │ │ │ - write: function(filter, params) { │ │ │ │ │ - params = params || {}; │ │ │ │ │ - var className = filter.CLASS_NAME; │ │ │ │ │ - var filterType = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ - switch (filterType) { │ │ │ │ │ - case "Spatial": │ │ │ │ │ - switch (filter.type) { │ │ │ │ │ - case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ - params.bbox = filter.value.toArray(); │ │ │ │ │ - if (this.srsInBBOX && filter.projection) { │ │ │ │ │ - params.bbox.push(filter.projection.getCode()); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Filter.Spatial.DWITHIN: │ │ │ │ │ - params.tolerance = filter.distance; │ │ │ │ │ - // no break here │ │ │ │ │ - case OpenLayers.Filter.Spatial.WITHIN: │ │ │ │ │ - params.lon = filter.value.x; │ │ │ │ │ - params.lat = filter.value.y; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - OpenLayers.Console.warn( │ │ │ │ │ - "Unknown spatial filter type " + filter.type); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "Comparison": │ │ │ │ │ - var op = cmpToStr[filter.type]; │ │ │ │ │ - if (op !== undefined) { │ │ │ │ │ - var value = filter.value; │ │ │ │ │ - if (filter.type == OpenLayers.Filter.Comparison.LIKE) { │ │ │ │ │ - value = regex2value(value); │ │ │ │ │ - if (this.wildcarded) { │ │ │ │ │ - value = "%" + value + "%"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - params[filter.property + "__" + op] = value; │ │ │ │ │ - params.queryable = params.queryable || []; │ │ │ │ │ - params.queryable.push(filter.property); │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Console.warn( │ │ │ │ │ - "Unknown comparison filter type " + filter.type); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "Logical": │ │ │ │ │ - if (filter.type === OpenLayers.Filter.Logical.AND) { │ │ │ │ │ - for (var i = 0, len = filter.filters.length; i < len; i++) { │ │ │ │ │ - params = this.write(filter.filters[i], params); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Console.warn( │ │ │ │ │ - "Unsupported logical filter type " + filter.type); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - OpenLayers.Console.warn("Unknown filter type " + filterType); │ │ │ │ │ + // now that the node has all its new properties, insert it │ │ │ │ │ + // back into the dom where it was │ │ │ │ │ + if (nextSibling) { │ │ │ │ │ + parent.insertBefore(node, nextSibling); │ │ │ │ │ + } else if (parent) { │ │ │ │ │ + parent.appendChild(node); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "r", style.pointRadius); │ │ │ │ │ } │ │ │ │ │ - return params; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.QueryStringFilter" │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ + var rotation = style.rotation; │ │ │ │ │ │ │ │ │ │ + if ((rotation !== undefined || node._rotation !== undefined) && pos) { │ │ │ │ │ + node._rotation = rotation; │ │ │ │ │ + rotation |= 0; │ │ │ │ │ + if (node.nodeName !== "svg") { │ │ │ │ │ + node.setAttributeNS(null, "transform", │ │ │ │ │ + "rotate(" + rotation + " " + pos.x + " " + │ │ │ │ │ + pos.y + ")"); │ │ │ │ │ + } else { │ │ │ │ │ + var metrics = this.symbolMetrics[src.id]; │ │ │ │ │ + node.firstChild.setAttributeNS(null, "transform", "rotate(" + │ │ │ │ │ + rotation + " " + │ │ │ │ │ + metrics[1] + " " + │ │ │ │ │ + metrics[2] + ")"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -})(); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/KML.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + if (options.isFilled) { │ │ │ │ │ + node.setAttributeNS(null, "fill", style.fillColor); │ │ │ │ │ + node.setAttributeNS(null, "fill-opacity", style.fillOpacity); │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "fill", "none"); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + if (options.isStroked) { │ │ │ │ │ + node.setAttributeNS(null, "stroke", style.strokeColor); │ │ │ │ │ + node.setAttributeNS(null, "stroke-opacity", style.strokeOpacity); │ │ │ │ │ + node.setAttributeNS(null, "stroke-width", style.strokeWidth * widthFactor); │ │ │ │ │ + node.setAttributeNS(null, "stroke-linecap", style.strokeLinecap || "round"); │ │ │ │ │ + // Hard-coded linejoin for now, to make it look the same as in VML. │ │ │ │ │ + // There is no strokeLinejoin property yet for symbolizers. │ │ │ │ │ + node.setAttributeNS(null, "stroke-linejoin", "round"); │ │ │ │ │ + style.strokeDashstyle && node.setAttributeNS(null, │ │ │ │ │ + "stroke-dasharray", this.dashStyle(style, widthFactor)); │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "stroke", "none"); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/BaseTypes/Date.js │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Collection.js │ │ │ │ │ - * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ - * @requires OpenLayers/Projection.js │ │ │ │ │ - */ │ │ │ │ │ + if (style.pointerEvents) { │ │ │ │ │ + node.setAttributeNS(null, "pointer-events", style.pointerEvents); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.KML │ │ │ │ │ - * Read/Write KML. Create a new instance with the <OpenLayers.Format.KML> │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.KML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + if (style.cursor != null) { │ │ │ │ │ + node.setAttributeNS(null, "cursor", style.cursor); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - kml: "http://www.opengis.net/kml/2.2", │ │ │ │ │ - gx: "http://www.google.com/kml/ext/2.2" │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: kmlns │ │ │ │ │ - * {String} KML Namespace to use. Defaults to 2.0 namespace. │ │ │ │ │ - */ │ │ │ │ │ - kmlns: "http://earth.google.com/kml/2.0", │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: placemarksDesc │ │ │ │ │ - * {String} Name of the placemarks. Default is "No description available". │ │ │ │ │ + * Method: dashStyle │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * style - {Object} │ │ │ │ │ + * widthFactor - {Number} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A SVG compliant 'stroke-dasharray' value │ │ │ │ │ */ │ │ │ │ │ - placemarksDesc: "No description available", │ │ │ │ │ + dashStyle: function(style, widthFactor) { │ │ │ │ │ + var w = style.strokeWidth * widthFactor; │ │ │ │ │ + var str = style.strokeDashstyle; │ │ │ │ │ + switch (str) { │ │ │ │ │ + case 'solid': │ │ │ │ │ + return 'none'; │ │ │ │ │ + case 'dot': │ │ │ │ │ + return [1, 4 * w].join(); │ │ │ │ │ + case 'dash': │ │ │ │ │ + return [4 * w, 4 * w].join(); │ │ │ │ │ + case 'dashdot': │ │ │ │ │ + return [4 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ + case 'longdash': │ │ │ │ │ + return [8 * w, 4 * w].join(); │ │ │ │ │ + case 'longdashdot': │ │ │ │ │ + return [8 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ + default: │ │ │ │ │ + return OpenLayers.String.trim(str).replace(/\s+/g, ","); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: foldersName │ │ │ │ │ - * {String} Name of the folders. Default is "OpenLayers export". │ │ │ │ │ - * If set to null, no name element will be created. │ │ │ │ │ + * Method: createNode │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * type - {String} Kind of node to draw │ │ │ │ │ + * id - {String} Id for node │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A new node of the given type and id │ │ │ │ │ */ │ │ │ │ │ - foldersName: "OpenLayers export", │ │ │ │ │ + createNode: function(type, id) { │ │ │ │ │ + var node = document.createElementNS(this.xmlns, type); │ │ │ │ │ + if (id) { │ │ │ │ │ + node.setAttributeNS(null, "id", id); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: foldersDesc │ │ │ │ │ - * {String} Description of the folders. Default is "Exported on [date]." │ │ │ │ │ - * If set to null, no description element will be created. │ │ │ │ │ - */ │ │ │ │ │ - foldersDesc: "Exported on " + new Date(), │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: extractAttributes │ │ │ │ │ - * {Boolean} Extract attributes from KML. Default is true. │ │ │ │ │ - * Extracting styleUrls requires this to be set to true │ │ │ │ │ - * Note that currently only Data and SimpleData │ │ │ │ │ - * elements are handled. │ │ │ │ │ + * Method: nodeTypeCompare │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {SVGDomElement} An SVG element │ │ │ │ │ + * type - {String} Kind of node │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ */ │ │ │ │ │ - extractAttributes: true, │ │ │ │ │ + nodeTypeCompare: function(node, type) { │ │ │ │ │ + return (type == node.nodeName); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: kvpAttributes │ │ │ │ │ - * {Boolean} Only used if extractAttributes is true. │ │ │ │ │ - * If set to true, attributes will be simple │ │ │ │ │ - * key-value pairs, compatible with other formats, │ │ │ │ │ - * Any displayName elements will be ignored. │ │ │ │ │ - * If set to false, attributes will be objects, │ │ │ │ │ - * retaining any displayName elements, but not │ │ │ │ │ - * compatible with other formats. Any CDATA in │ │ │ │ │ - * displayName will be read in as a string value. │ │ │ │ │ - * Default is false. │ │ │ │ │ + * Method: createRenderRoot │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The specific render engine's root element │ │ │ │ │ */ │ │ │ │ │ - kvpAttributes: false, │ │ │ │ │ + createRenderRoot: function() { │ │ │ │ │ + var svg = this.nodeFactory(this.container.id + "_svgRoot", "svg"); │ │ │ │ │ + svg.style.display = "block"; │ │ │ │ │ + return svg; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: extractStyles │ │ │ │ │ - * {Boolean} Extract styles from KML. Default is false. │ │ │ │ │ - * Extracting styleUrls also requires extractAttributes to be │ │ │ │ │ - * set to true │ │ │ │ │ + * Method: createRoot │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * suffix - {String} suffix to append to the id │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - extractStyles: false, │ │ │ │ │ + createRoot: function(suffix) { │ │ │ │ │ + return this.nodeFactory(this.container.id + suffix, "g"); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: extractTracks │ │ │ │ │ - * {Boolean} Extract gx:Track elements from Placemark elements. Default │ │ │ │ │ - * is false. If true, features will be generated for all points in │ │ │ │ │ - * all gx:Track elements. Features will have a when (Date) attribute │ │ │ │ │ - * based on when elements in the track. If tracks include angle │ │ │ │ │ - * elements, features will have heading, tilt, and roll attributes. │ │ │ │ │ - * If track point coordinates have three values, features will have │ │ │ │ │ - * an altitude attribute with the third coordinate value. │ │ │ │ │ + * Method: createDefs │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The element to which we'll add the symbol definitions │ │ │ │ │ */ │ │ │ │ │ - extractTracks: false, │ │ │ │ │ + createDefs: function() { │ │ │ │ │ + var defs = this.nodeFactory(this.container.id + "_defs", "defs"); │ │ │ │ │ + this.rendererRoot.appendChild(defs); │ │ │ │ │ + return defs; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: trackAttributes │ │ │ │ │ - * {Array} If <extractTracks> is true, points within gx:Track elements will │ │ │ │ │ - * be parsed as features with when, heading, tilt, and roll attributes. │ │ │ │ │ - * Any additional attribute names can be provided in <trackAttributes>. │ │ │ │ │ - */ │ │ │ │ │ - trackAttributes: null, │ │ │ │ │ + /************************************** │ │ │ │ │ + * * │ │ │ │ │ + * GEOMETRY DRAWING FUNCTIONS * │ │ │ │ │ + * * │ │ │ │ │ + **************************************/ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: internalns │ │ │ │ │ - * {String} KML Namespace to use -- defaults to the namespace of the │ │ │ │ │ - * Placemark node being parsed, but falls back to kmlns. │ │ │ │ │ + * Method: drawPoint │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the point │ │ │ │ │ */ │ │ │ │ │ - internalns: null, │ │ │ │ │ + drawPoint: function(node, geometry) { │ │ │ │ │ + return this.drawCircle(node, geometry, 1); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Array} Array of features │ │ │ │ │ - * │ │ │ │ │ + * Method: drawCircle │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * radius - {Float} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the circle │ │ │ │ │ */ │ │ │ │ │ - features: null, │ │ │ │ │ + drawCircle: function(node, geometry, radius) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = ((geometry.x - this.featureDx) / resolution + this.left); │ │ │ │ │ + var y = (this.top - geometry.y / resolution); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: styles │ │ │ │ │ - * {Object} Storage of style objects │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - styles: null, │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + node.setAttributeNS(null, "cx", x); │ │ │ │ │ + node.setAttributeNS(null, "cy", y); │ │ │ │ │ + node.setAttributeNS(null, "r", radius); │ │ │ │ │ + return node; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: styleBaseUrl │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - styleBaseUrl: "", │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: fetched │ │ │ │ │ - * {Object} Storage of KML URLs that have been fetched before │ │ │ │ │ - * in order to prevent reloading them. │ │ │ │ │ + * Method: drawLineString │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components of │ │ │ │ │ + * the linestring, or false if nothing could be drawn │ │ │ │ │ */ │ │ │ │ │ - fetched: null, │ │ │ │ │ + drawLineString: function(node, geometry) { │ │ │ │ │ + var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ + if (componentsResult.path) { │ │ │ │ │ + node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ + return (componentsResult.complete ? node : null); │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: maxDepth │ │ │ │ │ - * {Integer} Maximum depth for recursive loading external KML URLs │ │ │ │ │ - * Defaults to 0: do no external fetching │ │ │ │ │ + * Method: drawLinearRing │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ + * of the linear ring, or false if nothing could be drawn │ │ │ │ │ */ │ │ │ │ │ - maxDepth: 0, │ │ │ │ │ + drawLinearRing: function(node, geometry) { │ │ │ │ │ + var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ + if (componentsResult.path) { │ │ │ │ │ + node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ + return (componentsResult.complete ? node : null); │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.KML │ │ │ │ │ - * Create a new parser for KML. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * Method: drawPolygon │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ + * of the polygon, or false if nothing could be drawn │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - // compile regular expressions once instead of every time they are used │ │ │ │ │ - this.regExes = { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g), │ │ │ │ │ - kmlColor: (/(\w{2})(\w{2})(\w{2})(\w{2})/), │ │ │ │ │ - kmlIconPalette: (/root:\/\/icons\/palette-(\d+)(\.\w+)/), │ │ │ │ │ - straightBracket: (/\$\[(.*?)\]/g) │ │ │ │ │ - }; │ │ │ │ │ - // KML coordinates are always in longlat WGS84 │ │ │ │ │ - this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + drawPolygon: function(node, geometry) { │ │ │ │ │ + var d = ""; │ │ │ │ │ + var draw = true; │ │ │ │ │ + var complete = true; │ │ │ │ │ + var linearRingResult, path; │ │ │ │ │ + for (var j = 0, len = geometry.components.length; j < len; j++) { │ │ │ │ │ + d += " M"; │ │ │ │ │ + linearRingResult = this.getComponentsString( │ │ │ │ │ + geometry.components[j].components, " "); │ │ │ │ │ + path = linearRingResult.path; │ │ │ │ │ + if (path) { │ │ │ │ │ + d += " " + path; │ │ │ │ │ + complete = linearRingResult.complete && complete; │ │ │ │ │ + } else { │ │ │ │ │ + draw = false; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + d += " z"; │ │ │ │ │ + if (draw) { │ │ │ │ │ + node.setAttributeNS(null, "d", d); │ │ │ │ │ + node.setAttributeNS(null, "fill-rule", "evenodd"); │ │ │ │ │ + return complete ? node : null; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read data from a string, and return a list of features. │ │ │ │ │ + * Method: drawRectangle │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} List of features. │ │ │ │ │ + * {DOMElement} or false if the renderer could not draw the rectangle │ │ │ │ │ */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - this.features = []; │ │ │ │ │ - this.styles = {}; │ │ │ │ │ - this.fetched = {}; │ │ │ │ │ - │ │ │ │ │ - // Set default options │ │ │ │ │ - var options = { │ │ │ │ │ - depth: 0, │ │ │ │ │ - styleBaseUrl: this.styleBaseUrl │ │ │ │ │ - }; │ │ │ │ │ + drawRectangle: function(node, geometry) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = ((geometry.x - this.featureDx) / resolution + this.left); │ │ │ │ │ + var y = (this.top - geometry.y / resolution); │ │ │ │ │ │ │ │ │ │ - return this.parseData(data, options); │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + node.setAttributeNS(null, "x", x); │ │ │ │ │ + node.setAttributeNS(null, "y", y); │ │ │ │ │ + node.setAttributeNS(null, "width", geometry.width / resolution); │ │ │ │ │ + node.setAttributeNS(null, "height", geometry.height / resolution); │ │ │ │ │ + return node; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseData │ │ │ │ │ - * Read data from a string, and return a list of features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * options - {Object} Hash of options │ │ │ │ │ + * Method: drawText │ │ │ │ │ + * This method is only called by the renderer itself. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} List of features. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * featureId - {String} │ │ │ │ │ + * style - │ │ │ │ │ + * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ */ │ │ │ │ │ - parseData: function(data, options) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + drawText: function(featureId, style, location) { │ │ │ │ │ + var drawOutline = (!!style.labelOutlineWidth); │ │ │ │ │ + // First draw text in halo color and size and overlay the │ │ │ │ │ + // normal text afterwards │ │ │ │ │ + if (drawOutline) { │ │ │ │ │ + var outlineStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ + outlineStyle.fontColor = outlineStyle.labelOutlineColor; │ │ │ │ │ + outlineStyle.fontStrokeColor = outlineStyle.labelOutlineColor; │ │ │ │ │ + outlineStyle.fontStrokeWidth = style.labelOutlineWidth; │ │ │ │ │ + if (style.labelOutlineOpacity) { │ │ │ │ │ + outlineStyle.fontOpacity = style.labelOutlineOpacity; │ │ │ │ │ + } │ │ │ │ │ + delete outlineStyle.labelOutlineWidth; │ │ │ │ │ + this.drawText(featureId, outlineStyle, location); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // Loop throught the following node types in this order and │ │ │ │ │ - // process the nodes found │ │ │ │ │ - var types = ["Link", "NetworkLink", "Style", "StyleMap", "Placemark"]; │ │ │ │ │ - for (var i = 0, len = types.length; i < len; ++i) { │ │ │ │ │ - var type = types[i]; │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ │ │ │ │ │ - var nodes = this.getElementsByTagNameNS(data, "*", type); │ │ │ │ │ + var x = ((location.x - this.featureDx) / resolution + this.left); │ │ │ │ │ + var y = (location.y / resolution - this.top); │ │ │ │ │ │ │ │ │ │ - // skip to next type if no nodes are found │ │ │ │ │ - if (nodes.length == 0) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ + var suffix = (drawOutline) ? this.LABEL_OUTLINE_SUFFIX : this.LABEL_ID_SUFFIX; │ │ │ │ │ + var label = this.nodeFactory(featureId + suffix, "text"); │ │ │ │ │ │ │ │ │ │ - switch (type.toLowerCase()) { │ │ │ │ │ + label.setAttributeNS(null, "x", x); │ │ │ │ │ + label.setAttributeNS(null, "y", -y); │ │ │ │ │ │ │ │ │ │ - // Fetch external links │ │ │ │ │ - case "link": │ │ │ │ │ - case "networklink": │ │ │ │ │ - this.parseLinks(nodes, options); │ │ │ │ │ - break; │ │ │ │ │ + if (style.fontColor) { │ │ │ │ │ + label.setAttributeNS(null, "fill", style.fontColor); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStrokeColor) { │ │ │ │ │ + label.setAttributeNS(null, "stroke", style.fontStrokeColor); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStrokeWidth) { │ │ │ │ │ + label.setAttributeNS(null, "stroke-width", style.fontStrokeWidth); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontOpacity) { │ │ │ │ │ + label.setAttributeNS(null, "opacity", style.fontOpacity); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontFamily) { │ │ │ │ │ + label.setAttributeNS(null, "font-family", style.fontFamily); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontSize) { │ │ │ │ │ + label.setAttributeNS(null, "font-size", style.fontSize); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontWeight) { │ │ │ │ │ + label.setAttributeNS(null, "font-weight", style.fontWeight); │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStyle) { │ │ │ │ │ + label.setAttributeNS(null, "font-style", style.fontStyle); │ │ │ │ │ + } │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + label.setAttributeNS(null, "pointer-events", "visible"); │ │ │ │ │ + label._featureId = featureId; │ │ │ │ │ + } else { │ │ │ │ │ + label.setAttributeNS(null, "pointer-events", "none"); │ │ │ │ │ + } │ │ │ │ │ + var align = style.labelAlign || OpenLayers.Renderer.defaultSymbolizer.labelAlign; │ │ │ │ │ + label.setAttributeNS(null, "text-anchor", │ │ │ │ │ + OpenLayers.Renderer.SVG.LABEL_ALIGN[align[0]] || "middle"); │ │ │ │ │ │ │ │ │ │ - // parse style information │ │ │ │ │ - case "style": │ │ │ │ │ - if (this.extractStyles) { │ │ │ │ │ - this.parseStyles(nodes, options); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "stylemap": │ │ │ │ │ - if (this.extractStyles) { │ │ │ │ │ - this.parseStyleMaps(nodes, options); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ + if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ + label.setAttributeNS(null, "dominant-baseline", │ │ │ │ │ + OpenLayers.Renderer.SVG.LABEL_ALIGN[align[1]] || "central"); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // parse features │ │ │ │ │ - case "placemark": │ │ │ │ │ - this.parseFeatures(nodes, options); │ │ │ │ │ - break; │ │ │ │ │ + var labelRows = style.label.split('\n'); │ │ │ │ │ + var numRows = labelRows.length; │ │ │ │ │ + while (label.childNodes.length > numRows) { │ │ │ │ │ + label.removeChild(label.lastChild); │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + var tspan = this.nodeFactory(featureId + suffix + "_tspan_" + i, "tspan"); │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + tspan._featureId = featureId; │ │ │ │ │ + tspan._geometry = location; │ │ │ │ │ + tspan._geometryClass = location.CLASS_NAME; │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.IS_GECKO === false) { │ │ │ │ │ + tspan.setAttributeNS(null, "baseline-shift", │ │ │ │ │ + OpenLayers.Renderer.SVG.LABEL_VSHIFT[align[1]] || "-35%"); │ │ │ │ │ + } │ │ │ │ │ + tspan.setAttribute("x", x); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + var vfactor = OpenLayers.Renderer.SVG.LABEL_VFACTOR[align[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5; │ │ │ │ │ + } │ │ │ │ │ + tspan.setAttribute("dy", (vfactor * (numRows - 1)) + "em"); │ │ │ │ │ + } else { │ │ │ │ │ + tspan.setAttribute("dy", "1em"); │ │ │ │ │ + } │ │ │ │ │ + tspan.textContent = (labelRows[i] === '') ? ' ' : labelRows[i]; │ │ │ │ │ + if (!tspan.parentNode) { │ │ │ │ │ + label.appendChild(tspan); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return this.features; │ │ │ │ │ + if (!label.parentNode) { │ │ │ │ │ + this.textRoot.appendChild(label); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseLinks │ │ │ │ │ - * Finds URLs of linked KML documents and fetches them │ │ │ │ │ + /** │ │ │ │ │ + * Method: getComponentString │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * nodes - {Array} of {DOMElement} data to read/parse. │ │ │ │ │ - * options - {Object} Hash of options │ │ │ │ │ + * Parameters: │ │ │ │ │ + * components - {Array(<OpenLayers.Geometry.Point>)} Array of points │ │ │ │ │ + * separator - {String} character between coordinate pairs. Defaults to "," │ │ │ │ │ * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} hash with properties "path" (the string created from the │ │ │ │ │ + * components and "complete" (false if the renderer was unable to │ │ │ │ │ + * draw all components) │ │ │ │ │ */ │ │ │ │ │ - parseLinks: function(nodes, options) { │ │ │ │ │ - │ │ │ │ │ - // Fetch external links <NetworkLink> and <Link> │ │ │ │ │ - // Don't do anything if we have reached our maximum depth for recursion │ │ │ │ │ - if (options.depth >= this.maxDepth) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // increase depth │ │ │ │ │ - var newOptions = OpenLayers.Util.extend({}, options); │ │ │ │ │ - newOptions.depth++; │ │ │ │ │ - │ │ │ │ │ - for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ - var href = this.parseProperty(nodes[i], "*", "href"); │ │ │ │ │ - if (href && !this.fetched[href]) { │ │ │ │ │ - this.fetched[href] = true; // prevent reloading the same urls │ │ │ │ │ - var data = this.fetchLink(href); │ │ │ │ │ - if (data) { │ │ │ │ │ - this.parseData(data, newOptions); │ │ │ │ │ + getComponentsString: function(components, separator) { │ │ │ │ │ + var renderCmp = []; │ │ │ │ │ + var complete = true; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + var strings = []; │ │ │ │ │ + var str, component; │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + component = components[i]; │ │ │ │ │ + renderCmp.push(component); │ │ │ │ │ + str = this.getShortString(component); │ │ │ │ │ + if (str) { │ │ │ │ │ + strings.push(str); │ │ │ │ │ + } else { │ │ │ │ │ + // The current component is outside the valid range. Let's │ │ │ │ │ + // see if the previous or next component is inside the range. │ │ │ │ │ + // If so, add the coordinate of the intersection with the │ │ │ │ │ + // valid range bounds. │ │ │ │ │ + if (i > 0) { │ │ │ │ │ + if (this.getShortString(components[i - 1])) { │ │ │ │ │ + strings.push(this.clipLine(components[i], │ │ │ │ │ + components[i - 1])); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (i < len - 1) { │ │ │ │ │ + if (this.getShortString(components[i + 1])) { │ │ │ │ │ + strings.push(this.clipLine(components[i], │ │ │ │ │ + components[i + 1])); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + complete = false; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ + return { │ │ │ │ │ + path: strings.join(separator || ","), │ │ │ │ │ + complete: complete │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: fetchLink │ │ │ │ │ - * Fetches a URL and returns the result │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * href - {String} url to be fetched │ │ │ │ │ + * Method: clipLine │ │ │ │ │ + * Given two points (one inside the valid range, and one outside), │ │ │ │ │ + * clips the line betweeen the two points so that the new points are both │ │ │ │ │ + * inside the valid range. │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * badComponent - {<OpenLayers.Geometry.Point>} original geometry of the │ │ │ │ │ + * invalid point │ │ │ │ │ + * goodComponent - {<OpenLayers.Geometry.Point>} original geometry of the │ │ │ │ │ + * valid point │ │ │ │ │ + * Returns │ │ │ │ │ + * {String} the SVG coordinate pair of the clipped point (like │ │ │ │ │ + * getShortString), or an empty string if both passed componets are at │ │ │ │ │ + * the same point. │ │ │ │ │ */ │ │ │ │ │ - fetchLink: function(href) { │ │ │ │ │ - var request = OpenLayers.Request.GET({ │ │ │ │ │ - url: href, │ │ │ │ │ - async: false │ │ │ │ │ - }); │ │ │ │ │ - if (request) { │ │ │ │ │ - return request.responseText; │ │ │ │ │ + clipLine: function(badComponent, goodComponent) { │ │ │ │ │ + if (goodComponent.equals(badComponent)) { │ │ │ │ │ + return ""; │ │ │ │ │ + } │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var maxX = this.MAX_PIXEL - this.translationParameters.x; │ │ │ │ │ + var maxY = this.MAX_PIXEL - this.translationParameters.y; │ │ │ │ │ + var x1 = (goodComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y1 = this.top - goodComponent.y / resolution; │ │ │ │ │ + var x2 = (badComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y2 = this.top - badComponent.y / resolution; │ │ │ │ │ + var k; │ │ │ │ │ + if (x2 < -maxX || x2 > maxX) { │ │ │ │ │ + k = (y2 - y1) / (x2 - x1); │ │ │ │ │ + x2 = x2 < 0 ? -maxX : maxX; │ │ │ │ │ + y2 = y1 + (x2 - x1) * k; │ │ │ │ │ + } │ │ │ │ │ + if (y2 < -maxY || y2 > maxY) { │ │ │ │ │ + k = (x2 - x1) / (y2 - y1); │ │ │ │ │ + y2 = y2 < 0 ? -maxY : maxY; │ │ │ │ │ + x2 = x1 + (y2 - y1) * k; │ │ │ │ │ } │ │ │ │ │ + return x2 + "," + y2; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseStyles │ │ │ │ │ - * Parses <Style> nodes │ │ │ │ │ + /** │ │ │ │ │ + * Method: getShortString │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * nodes - {Array} of {DOMElement} data to read/parse. │ │ │ │ │ - * options - {Object} Hash of options │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} or false if point is outside the valid range │ │ │ │ │ */ │ │ │ │ │ - parseStyles: function(nodes, options) { │ │ │ │ │ - for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ - var style = this.parseStyle(nodes[i]); │ │ │ │ │ - if (style) { │ │ │ │ │ - var styleName = (options.styleBaseUrl || "") + "#" + style.id; │ │ │ │ │ + getShortString: function(point) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = ((point.x - this.featureDx) / resolution + this.left); │ │ │ │ │ + var y = (this.top - point.y / resolution); │ │ │ │ │ │ │ │ │ │ - this.styles[styleName] = style; │ │ │ │ │ - } │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + return x + "," + y; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseKmlColor │ │ │ │ │ - * Parses a kml color (in 'aabbggrr' format) and returns the corresponding │ │ │ │ │ - * color and opacity or null if the color is invalid. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * kmlColor - {String} a kml formated color │ │ │ │ │ - * │ │ │ │ │ + * Method: getPosition │ │ │ │ │ + * Finds the position of an svg node. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} │ │ │ │ │ + * {Object} hash with x and y properties, representing the coordinates │ │ │ │ │ + * within the svg coordinate system │ │ │ │ │ */ │ │ │ │ │ - parseKmlColor: function(kmlColor) { │ │ │ │ │ - var color = null; │ │ │ │ │ - if (kmlColor) { │ │ │ │ │ - var matches = kmlColor.match(this.regExes.kmlColor); │ │ │ │ │ - if (matches) { │ │ │ │ │ - color = { │ │ │ │ │ - color: '#' + matches[4] + matches[3] + matches[2], │ │ │ │ │ - opacity: parseInt(matches[1], 16) / 255 │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return color; │ │ │ │ │ + getPosition: function(node) { │ │ │ │ │ + return ({ │ │ │ │ │ + x: parseFloat(node.getAttributeNS(null, "cx")), │ │ │ │ │ + y: parseFloat(node.getAttributeNS(null, "cy")) │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseStyle │ │ │ │ │ - * Parses the children of a <Style> node and builds the style hash │ │ │ │ │ - * accordingly │ │ │ │ │ + * Method: importSymbol │ │ │ │ │ + * add a new symbol definition from the rendererer's symbol hash │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} <Style> node │ │ │ │ │ + * Parameters: │ │ │ │ │ + * graphicName - {String} name of the symbol to import │ │ │ │ │ * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} - the imported symbol │ │ │ │ │ */ │ │ │ │ │ - parseStyle: function(node) { │ │ │ │ │ - var style = {}; │ │ │ │ │ + importSymbol: function(graphicName) { │ │ │ │ │ + if (!this.defs) { │ │ │ │ │ + // create svg defs tag │ │ │ │ │ + this.defs = this.createDefs(); │ │ │ │ │ + } │ │ │ │ │ + var id = this.container.id + "-" + graphicName; │ │ │ │ │ │ │ │ │ │ - var types = ["LineStyle", "PolyStyle", "IconStyle", "BalloonStyle", │ │ │ │ │ - "LabelStyle" │ │ │ │ │ - ]; │ │ │ │ │ - var type, styleTypeNode, nodeList, geometry, parser; │ │ │ │ │ - for (var i = 0, len = types.length; i < len; ++i) { │ │ │ │ │ - type = types[i]; │ │ │ │ │ - styleTypeNode = this.getElementsByTagNameNS(node, "*", type)[0]; │ │ │ │ │ - if (!styleTypeNode) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ + // check if symbol already exists in the defs │ │ │ │ │ + var existing = document.getElementById(id); │ │ │ │ │ + if (existing != null) { │ │ │ │ │ + return existing; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // only deal with first geometry of this type │ │ │ │ │ - switch (type.toLowerCase()) { │ │ │ │ │ - case "linestyle": │ │ │ │ │ - var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ - var color = this.parseKmlColor(kmlColor); │ │ │ │ │ - if (color) { │ │ │ │ │ - style["strokeColor"] = color.color; │ │ │ │ │ - style["strokeOpacity"] = color.opacity; │ │ │ │ │ - } │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(graphicName + ' is not a valid symbol name'); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - var width = this.parseProperty(styleTypeNode, "*", "width"); │ │ │ │ │ - if (width) { │ │ │ │ │ - style["strokeWidth"] = width; │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ + var symbolNode = this.nodeFactory(id, "symbol"); │ │ │ │ │ + var node = this.nodeFactory(null, "polygon"); │ │ │ │ │ + symbolNode.appendChild(node); │ │ │ │ │ + var symbolExtent = new OpenLayers.Bounds( │ │ │ │ │ + Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ │ │ │ │ │ - case "polystyle": │ │ │ │ │ - var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ - var color = this.parseKmlColor(kmlColor); │ │ │ │ │ - if (color) { │ │ │ │ │ - style["fillOpacity"] = color.opacity; │ │ │ │ │ - style["fillColor"] = color.color; │ │ │ │ │ - } │ │ │ │ │ - // Check if fill is disabled │ │ │ │ │ - var fill = this.parseProperty(styleTypeNode, "*", "fill"); │ │ │ │ │ - if (fill == "0") { │ │ │ │ │ - style["fillColor"] = "none"; │ │ │ │ │ - } │ │ │ │ │ - // Check if outline is disabled │ │ │ │ │ - var outline = this.parseProperty(styleTypeNode, "*", "outline"); │ │ │ │ │ - if (outline == "0") { │ │ │ │ │ - style["strokeWidth"] = "0"; │ │ │ │ │ - } │ │ │ │ │ + var points = []; │ │ │ │ │ + var x, y; │ │ │ │ │ + for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ + symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ + symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ + symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ + points.push(x, ",", y); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - break; │ │ │ │ │ + node.setAttributeNS(null, "points", points.join(" ")); │ │ │ │ │ │ │ │ │ │ - case "iconstyle": │ │ │ │ │ - // set scale │ │ │ │ │ - var scale = parseFloat(this.parseProperty(styleTypeNode, │ │ │ │ │ - "*", "scale") || 1); │ │ │ │ │ + var width = symbolExtent.getWidth(); │ │ │ │ │ + var height = symbolExtent.getHeight(); │ │ │ │ │ + // create a viewBox three times as large as the symbol itself, │ │ │ │ │ + // to allow for strokeWidth being displayed correctly at the corners. │ │ │ │ │ + var viewBox = [symbolExtent.left - width, │ │ │ │ │ + symbolExtent.bottom - height, width * 3, height * 3 │ │ │ │ │ + ]; │ │ │ │ │ + symbolNode.setAttributeNS(null, "viewBox", viewBox.join(" ")); │ │ │ │ │ + this.symbolMetrics[id] = [ │ │ │ │ │ + Math.max(width, height), │ │ │ │ │ + symbolExtent.getCenterLonLat().lon, │ │ │ │ │ + symbolExtent.getCenterLonLat().lat │ │ │ │ │ + ]; │ │ │ │ │ │ │ │ │ │ - // set default width and height of icon │ │ │ │ │ - var width = 32 * scale; │ │ │ │ │ - var height = 32 * scale; │ │ │ │ │ + this.defs.appendChild(symbolNode); │ │ │ │ │ + return symbolNode; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var iconNode = this.getElementsByTagNameNS(styleTypeNode, │ │ │ │ │ - "*", │ │ │ │ │ - "Icon")[0]; │ │ │ │ │ - if (iconNode) { │ │ │ │ │ - var href = this.parseProperty(iconNode, "*", "href"); │ │ │ │ │ - if (href) { │ │ │ │ │ + /** │ │ │ │ │ + * Method: getFeatureIdFromEvent │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} An <OpenLayers.Event> object │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A feature id or undefined. │ │ │ │ │ + */ │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var featureId = OpenLayers.Renderer.Elements.prototype.getFeatureIdFromEvent.apply(this, arguments); │ │ │ │ │ + if (!featureId) { │ │ │ │ │ + var target = evt.target; │ │ │ │ │ + featureId = target.parentNode && target != this.rendererRoot ? │ │ │ │ │ + target.parentNode._featureId : undefined; │ │ │ │ │ + } │ │ │ │ │ + return featureId; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var w = this.parseProperty(iconNode, "*", "w"); │ │ │ │ │ - var h = this.parseProperty(iconNode, "*", "h"); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.SVG" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ - // Settings for Google specific icons that are 64x64 │ │ │ │ │ - // We set the width and height to 64 and halve the │ │ │ │ │ - // scale to prevent icons from being too big │ │ │ │ │ - var google = "http://maps.google.com/mapfiles/kml"; │ │ │ │ │ - if (OpenLayers.String.startsWith( │ │ │ │ │ - href, google) && !w && !h) { │ │ │ │ │ - w = 64; │ │ │ │ │ - h = 64; │ │ │ │ │ - scale = scale / 2; │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.SVG.LABEL_ALIGN │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_ALIGN = { │ │ │ │ │ + "l": "start", │ │ │ │ │ + "r": "end", │ │ │ │ │ + "b": "bottom", │ │ │ │ │ + "t": "hanging" │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ - // if only dimension is defined, make sure the │ │ │ │ │ - // other one has the same value │ │ │ │ │ - w = w || h; │ │ │ │ │ - h = h || w; │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.SVG.LABEL_VSHIFT │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_VSHIFT = { │ │ │ │ │ + // according to │ │ │ │ │ + // http://www.w3.org/Graphics/SVG/Test/20061213/htmlObjectHarness/full-text-align-02-b.html │ │ │ │ │ + // a baseline-shift of -70% shifts the text exactly from the │ │ │ │ │ + // bottom to the top of the baseline, so -35% moves the text to │ │ │ │ │ + // the center of the baseline. │ │ │ │ │ + "t": "-70%", │ │ │ │ │ + "b": "0" │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ - if (w) { │ │ │ │ │ - width = parseInt(w) * scale; │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Renderer.SVG.LABEL_VFACTOR │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_VFACTOR = { │ │ │ │ │ + "t": 0, │ │ │ │ │ + "b": -1 │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ - if (h) { │ │ │ │ │ - height = parseInt(h) * scale; │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Function: OpenLayers.Renderer.SVG.preventDefault │ │ │ │ │ + * *Deprecated*. Use <OpenLayers.Event.preventDefault> method instead. │ │ │ │ │ + * Used to prevent default events (especially opening images in a new tab on │ │ │ │ │ + * ctrl-click) from being executed for externalGraphic symbols │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Renderer.SVG.preventDefault = function(e) { │ │ │ │ │ + OpenLayers.Event.preventDefault(e); │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Protocol/CSW.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - // support for internal icons │ │ │ │ │ - // (/root://icons/palette-x.png) │ │ │ │ │ - // x and y tell the position on the palette: │ │ │ │ │ - // - in pixels │ │ │ │ │ - // - starting from the left bottom │ │ │ │ │ - // We translate that to a position in the list │ │ │ │ │ - // and request the appropriate icon from the │ │ │ │ │ - // google maps website │ │ │ │ │ - var matches = href.match(this.regExes.kmlIconPalette); │ │ │ │ │ - if (matches) { │ │ │ │ │ - var palette = matches[1]; │ │ │ │ │ - var file_extension = matches[2]; │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - var x = this.parseProperty(iconNode, "*", "x"); │ │ │ │ │ - var y = this.parseProperty(iconNode, "*", "y"); │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Protocol.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - var posX = x ? x / 32 : 0; │ │ │ │ │ - var posY = y ? (7 - y / 32) : 7; │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.CSW │ │ │ │ │ + * Used to create a versioned CSW protocol. Default version is 2.0.2. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.CSW = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options, OpenLayers.Protocol.CSW.DEFAULTS │ │ │ │ │ + ); │ │ │ │ │ + var cls = OpenLayers.Protocol.CSW["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported CSW version: " + options.version; │ │ │ │ │ + } │ │ │ │ │ + return new cls(options); │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ - var pos = posY * 8 + posX; │ │ │ │ │ - href = "http://maps.google.com/mapfiles/kml/pal" + │ │ │ │ │ - palette + "/icon" + pos + file_extension; │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Protocol.CSW.DEFAULTS │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.CSW.DEFAULTS = { │ │ │ │ │ + "version": "2.0.2" │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Protocol/SOS.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - style["graphicOpacity"] = 1; // fully opaque │ │ │ │ │ - style["externalGraphic"] = href; │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Protocol.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * Function: OpenLayers.Protocol.SOS │ │ │ │ │ + * Used to create a versioned SOS protocol. Default version is 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol>} An SOS protocol for the given version. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.SOS = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options, OpenLayers.Protocol.SOS.DEFAULTS │ │ │ │ │ + ); │ │ │ │ │ + var cls = OpenLayers.Protocol.SOS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported SOS version: " + options.version; │ │ │ │ │ + } │ │ │ │ │ + return new cls(options); │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ - // hotSpots define the offset for an Icon │ │ │ │ │ - var hotSpotNode = this.getElementsByTagNameNS(styleTypeNode, │ │ │ │ │ - "*", │ │ │ │ │ - "hotSpot")[0]; │ │ │ │ │ - if (hotSpotNode) { │ │ │ │ │ - var x = parseFloat(hotSpotNode.getAttribute("x")); │ │ │ │ │ - var y = parseFloat(hotSpotNode.getAttribute("y")); │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Protocol.SOS.DEFAULTS │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.SOS.DEFAULTS = { │ │ │ │ │ + "version": "1.0.0" │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Protocol/WFS.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - var xUnits = hotSpotNode.getAttribute("xunits"); │ │ │ │ │ - if (xUnits == "pixels") { │ │ │ │ │ - style["graphicXOffset"] = -x * scale; │ │ │ │ │ - } else if (xUnits == "insetPixels") { │ │ │ │ │ - style["graphicXOffset"] = -width + (x * scale); │ │ │ │ │ - } else if (xUnits == "fraction") { │ │ │ │ │ - style["graphicXOffset"] = -width * x; │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - var yUnits = hotSpotNode.getAttribute("yunits"); │ │ │ │ │ - if (yUnits == "pixels") { │ │ │ │ │ - style["graphicYOffset"] = -height + (y * scale) + 1; │ │ │ │ │ - } else if (yUnits == "insetPixels") { │ │ │ │ │ - style["graphicYOffset"] = -(y * scale) + 1; │ │ │ │ │ - } else if (yUnits == "fraction") { │ │ │ │ │ - style["graphicYOffset"] = -height * (1 - y) + 1; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Protocol.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - style["graphicWidth"] = width; │ │ │ │ │ - style["graphicHeight"] = height; │ │ │ │ │ - break; │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.WFS │ │ │ │ │ + * Used to create a versioned WFS protocol. Default version is 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol>} A WFS protocol of the given version. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var protocol = new OpenLayers.Protocol.WFS({ │ │ │ │ │ + * version: "1.1.0", │ │ │ │ │ + * url: "http://demo.opengeo.org/geoserver/wfs", │ │ │ │ │ + * featureType: "tasmania_roads", │ │ │ │ │ + * featureNS: "http://www.openplans.org/topp", │ │ │ │ │ + * geometryName: "the_geom" │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * See the protocols for specific WFS versions for more detail. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.WFS = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options, OpenLayers.Protocol.WFS.DEFAULTS │ │ │ │ │ + ); │ │ │ │ │ + var cls = OpenLayers.Protocol.WFS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported WFS version: " + options.version; │ │ │ │ │ + } │ │ │ │ │ + return new cls(options); │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ - case "balloonstyle": │ │ │ │ │ - var balloonStyle = OpenLayers.Util.getXmlNodeValue( │ │ │ │ │ - styleTypeNode); │ │ │ │ │ - if (balloonStyle) { │ │ │ │ │ - style["balloonStyle"] = balloonStyle.replace( │ │ │ │ │ - this.regExes.straightBracket, "${$1}"); │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "labelstyle": │ │ │ │ │ - var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ - var color = this.parseKmlColor(kmlColor); │ │ │ │ │ - if (color) { │ │ │ │ │ - style["fontColor"] = color.color; │ │ │ │ │ - style["fontOpacity"] = color.opacity; │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ +/** │ │ │ │ │ + * Function: fromWMSLayer │ │ │ │ │ + * Convenience function to create a WFS protocol from a WMS layer. This makes │ │ │ │ │ + * the assumption that a WFS requests can be issued at the same URL as │ │ │ │ │ + * WMS requests and that a WFS featureType exists with the same name as the │ │ │ │ │ + * WMS layer. │ │ │ │ │ + * │ │ │ │ │ + * This function is designed to auto-configure <url>, <featureType>, │ │ │ │ │ + * <featurePrefix> and <srsName> for WFS <version> 1.1.0. Note that │ │ │ │ │ + * srsName matching with the WMS layer will not work with WFS 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer.WMS>} WMS layer that has a matching WFS │ │ │ │ │ + * FeatureType at the same server url with the same typename. │ │ │ │ │ + * options - {Object} Default properties to be set on the protocol. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.WFS>} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.WFS.fromWMSLayer = function(layer, options) { │ │ │ │ │ + var typeName, featurePrefix; │ │ │ │ │ + var param = layer.params["LAYERS"]; │ │ │ │ │ + var parts = (OpenLayers.Util.isArray(param) ? param[0] : param).split(":"); │ │ │ │ │ + if (parts.length > 1) { │ │ │ │ │ + featurePrefix = parts[0]; │ │ │ │ │ + } │ │ │ │ │ + typeName = parts.pop(); │ │ │ │ │ + var protocolOptions = { │ │ │ │ │ + url: layer.url, │ │ │ │ │ + featureType: typeName, │ │ │ │ │ + featurePrefix: featurePrefix, │ │ │ │ │ + srsName: layer.projection && layer.projection.getCode() || │ │ │ │ │ + layer.map && layer.map.getProjectionObject().getCode(), │ │ │ │ │ + version: "1.1.0" │ │ │ │ │ + }; │ │ │ │ │ + return new OpenLayers.Protocol.WFS(OpenLayers.Util.applyDefaults( │ │ │ │ │ + options, protocolOptions │ │ │ │ │ + )); │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ - default: │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Protocol.WFS.DEFAULTS │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.WFS.DEFAULTS = { │ │ │ │ │ + "version": "1.0.0" │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Protocol/HTTP.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - // Some polygons have no line color, so we use the fillColor for that │ │ │ │ │ - if (!style["strokeColor"] && style["fillColor"]) { │ │ │ │ │ - style["strokeColor"] = style["fillColor"]; │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - var id = node.getAttribute("id"); │ │ │ │ │ - if (id && style) { │ │ │ │ │ - style.id = id; │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Protocol.js │ │ │ │ │ + * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - return style; │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * if application uses the query string, for example, for BBOX parameters, │ │ │ │ │ + * OpenLayers/Format/QueryStringFilter.js should be included in the build config file │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.HTTP │ │ │ │ │ + * A basic HTTP protocol for vector layers. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Protocol.HTTP> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Protocol> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.HTTP = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseStyleMaps │ │ │ │ │ - * Parses <StyleMap> nodes, but only uses the 'normal' key │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * nodes - {Array} of {DOMElement} data to read/parse. │ │ │ │ │ - * options - {Object} Hash of options │ │ │ │ │ - * │ │ │ │ │ + * Property: url │ │ │ │ │ + * {String} Service URL, read-only, set through the options │ │ │ │ │ + * passed to constructor. │ │ │ │ │ */ │ │ │ │ │ - parseStyleMaps: function(nodes, options) { │ │ │ │ │ - // Only the default or "normal" part of the StyleMap is processed now │ │ │ │ │ - // To do the select or "highlight" bit, we'd need to change lots more │ │ │ │ │ + url: null, │ │ │ │ │ │ │ │ │ │ - for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ - var node = nodes[i]; │ │ │ │ │ - var pairs = this.getElementsByTagNameNS(node, "*", │ │ │ │ │ - "Pair"); │ │ │ │ │ + /** │ │ │ │ │ + * Property: headers │ │ │ │ │ + * {Object} HTTP request headers, read-only, set through the options │ │ │ │ │ + * passed to the constructor, │ │ │ │ │ + * Example: {'Content-Type': 'plain/text'} │ │ │ │ │ + */ │ │ │ │ │ + headers: null, │ │ │ │ │ │ │ │ │ │ - var id = node.getAttribute("id"); │ │ │ │ │ - for (var j = 0, jlen = pairs.length; j < jlen; j++) { │ │ │ │ │ - var pair = pairs[j]; │ │ │ │ │ - // Use the shortcut in the SLD format to quickly retrieve the │ │ │ │ │ - // value of a node. Maybe it's good to have a method in │ │ │ │ │ - // Format.XML to do this │ │ │ │ │ - var key = this.parseProperty(pair, "*", "key"); │ │ │ │ │ - var styleUrl = this.parseProperty(pair, "*", "styleUrl"); │ │ │ │ │ + /** │ │ │ │ │ + * Property: params │ │ │ │ │ + * {Object} Parameters of GET requests, read-only, set through the options │ │ │ │ │ + * passed to the constructor, │ │ │ │ │ + * Example: {'bbox': '5,5,5,5'} │ │ │ │ │ + */ │ │ │ │ │ + params: null, │ │ │ │ │ │ │ │ │ │ - if (styleUrl && key == "normal") { │ │ │ │ │ - this.styles[(options.styleBaseUrl || "") + "#" + id] = │ │ │ │ │ - this.styles[(options.styleBaseUrl || "") + styleUrl]; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: callback │ │ │ │ │ + * {Object} Function to be called when the <read>, <create>, │ │ │ │ │ + * <update>, <delete> or <commit> operation completes, read-only, │ │ │ │ │ + * set through the options passed to the constructor. │ │ │ │ │ + */ │ │ │ │ │ + callback: null, │ │ │ │ │ │ │ │ │ │ - // TODO: implement the "select" part │ │ │ │ │ - //if (styleUrl && key == "highlight") { │ │ │ │ │ - //} │ │ │ │ │ + /** │ │ │ │ │ + * Property: scope │ │ │ │ │ + * {Object} Callback execution scope, read-only, set through the │ │ │ │ │ + * options passed to the constructor. │ │ │ │ │ + */ │ │ │ │ │ + scope: null, │ │ │ │ │ │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: readWithPOST │ │ │ │ │ + * {Boolean} true if read operations are done with POST requests │ │ │ │ │ + * instead of GET, defaults to false. │ │ │ │ │ + */ │ │ │ │ │ + readWithPOST: false, │ │ │ │ │ │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: updateWithPOST │ │ │ │ │ + * {Boolean} true if update operations are done with POST requests │ │ │ │ │ + * defaults to false. │ │ │ │ │ + */ │ │ │ │ │ + updateWithPOST: false, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: deleteWithPOST │ │ │ │ │ + * {Boolean} true if delete operations are done with POST requests │ │ │ │ │ + * defaults to false. │ │ │ │ │ + * if true, POST data is set to output of format.write(). │ │ │ │ │ + */ │ │ │ │ │ + deleteWithPOST: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseFeatures │ │ │ │ │ - * Loop through all Placemark nodes and parse them. │ │ │ │ │ - * Will create a list of features │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * nodes - {Array} of {DOMElement} data to read/parse. │ │ │ │ │ - * options - {Object} Hash of options │ │ │ │ │ - * │ │ │ │ │ + * Property: wildcarded. │ │ │ │ │ + * {Boolean} If true percent signs are added around values │ │ │ │ │ + * read from LIKE filters, for example if the protocol │ │ │ │ │ + * read method is passed a LIKE filter whose property │ │ │ │ │ + * is "foo" and whose value is "bar" the string │ │ │ │ │ + * "foo__ilike=%bar%" will be sent in the query string; │ │ │ │ │ + * defaults to false. │ │ │ │ │ */ │ │ │ │ │ - parseFeatures: function(nodes, options) { │ │ │ │ │ - var features = []; │ │ │ │ │ - for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ - var featureNode = nodes[i]; │ │ │ │ │ - var feature = this.parseFeature.apply(this, [featureNode]); │ │ │ │ │ - if (feature) { │ │ │ │ │ + wildcarded: false, │ │ │ │ │ │ │ │ │ │ - // Create reference to styleUrl │ │ │ │ │ - if (this.extractStyles && feature.attributes && │ │ │ │ │ - feature.attributes.styleUrl) { │ │ │ │ │ - feature.style = this.getStyle(feature.attributes.styleUrl, options); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: srsInBBOX │ │ │ │ │ + * {Boolean} Include the SRS identifier in BBOX query string parameter. │ │ │ │ │ + * Default is false. If true and the layer has a projection object set, │ │ │ │ │ + * any BBOX filter will be serialized with a fifth item identifying the │ │ │ │ │ + * projection. E.g. bbox=-1000,-1000,1000,1000,EPSG:900913 │ │ │ │ │ + */ │ │ │ │ │ + srsInBBOX: false, │ │ │ │ │ │ │ │ │ │ - if (this.extractStyles) { │ │ │ │ │ - // Make sure that <Style> nodes within a placemark are │ │ │ │ │ - // processed as well │ │ │ │ │ - var inlineStyleNode = this.getElementsByTagNameNS(featureNode, │ │ │ │ │ - "*", │ │ │ │ │ - "Style")[0]; │ │ │ │ │ - if (inlineStyleNode) { │ │ │ │ │ - var inlineStyle = this.parseStyle(inlineStyleNode); │ │ │ │ │ - if (inlineStyle) { │ │ │ │ │ - feature.style = OpenLayers.Util.extend( │ │ │ │ │ - feature.style, inlineStyle │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Protocol.HTTP │ │ │ │ │ + * A class for giving layers generic HTTP protocol. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + * │ │ │ │ │ + * Valid options include: │ │ │ │ │ + * url - {String} │ │ │ │ │ + * headers - {Object} │ │ │ │ │ + * params - {Object} URL parameters for GET requests │ │ │ │ │ + * format - {<OpenLayers.Format>} │ │ │ │ │ + * callback - {Function} │ │ │ │ │ + * scope - {Object} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + this.params = {}; │ │ │ │ │ + this.headers = {}; │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - // check if gx:Track elements should be parsed │ │ │ │ │ - if (this.extractTracks) { │ │ │ │ │ - var tracks = this.getElementsByTagNameNS( │ │ │ │ │ - featureNode, this.namespaces.gx, "Track" │ │ │ │ │ - ); │ │ │ │ │ - if (tracks && tracks.length > 0) { │ │ │ │ │ - var track = tracks[0]; │ │ │ │ │ - var container = { │ │ │ │ │ - features: [], │ │ │ │ │ - feature: feature │ │ │ │ │ - }; │ │ │ │ │ - this.readNode(track, container); │ │ │ │ │ - if (container.features.length > 0) { │ │ │ │ │ - features.push.apply(features, container.features); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - // add feature to list of features │ │ │ │ │ - features.push(feature); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - throw "Bad Placemark: " + i; │ │ │ │ │ - } │ │ │ │ │ + if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ + var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ + wildcarded: this.wildcarded, │ │ │ │ │ + srsInBBOX: this.srsInBBOX │ │ │ │ │ + }); │ │ │ │ │ + this.filterToParams = function(filter, params) { │ │ │ │ │ + return format.write(filter, params); │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // add new features to existing feature list │ │ │ │ │ - this.features = this.features.concat(features); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up the protocol. │ │ │ │ │ */ │ │ │ │ │ - readers: { │ │ │ │ │ - "kml": { │ │ │ │ │ - "when": function(node, container) { │ │ │ │ │ - container.whens.push(OpenLayers.Date.parse( │ │ │ │ │ - this.getChildValue(node) │ │ │ │ │ - )); │ │ │ │ │ - }, │ │ │ │ │ - "_trackPointAttribute": function(node, container) { │ │ │ │ │ - var name = node.nodeName.split(":").pop(); │ │ │ │ │ - container.attributes[name].push(this.getChildValue(node)); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "gx": { │ │ │ │ │ - "Track": function(node, container) { │ │ │ │ │ - var obj = { │ │ │ │ │ - whens: [], │ │ │ │ │ - points: [], │ │ │ │ │ - angles: [] │ │ │ │ │ - }; │ │ │ │ │ - if (this.trackAttributes) { │ │ │ │ │ - var name; │ │ │ │ │ - obj.attributes = {}; │ │ │ │ │ - for (var i = 0, ii = this.trackAttributes.length; i < ii; ++i) { │ │ │ │ │ - name = this.trackAttributes[i]; │ │ │ │ │ - obj.attributes[name] = []; │ │ │ │ │ - if (!(name in this.readers.kml)) { │ │ │ │ │ - this.readers.kml[name] = this.readers.kml._trackPointAttribute; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - if (obj.whens.length !== obj.points.length) { │ │ │ │ │ - throw new Error("gx:Track with unequal number of when (" + │ │ │ │ │ - obj.whens.length + ") and gx:coord (" + │ │ │ │ │ - obj.points.length + ") elements."); │ │ │ │ │ - } │ │ │ │ │ - var hasAngles = obj.angles.length > 0; │ │ │ │ │ - if (hasAngles && obj.whens.length !== obj.angles.length) { │ │ │ │ │ - throw new Error("gx:Track with unequal number of when (" + │ │ │ │ │ - obj.whens.length + ") and gx:angles (" + │ │ │ │ │ - obj.angles.length + ") elements."); │ │ │ │ │ - } │ │ │ │ │ - var feature, point, angles; │ │ │ │ │ - for (var i = 0, ii = obj.whens.length; i < ii; ++i) { │ │ │ │ │ - feature = container.feature.clone(); │ │ │ │ │ - feature.fid = container.feature.fid || container.feature.id; │ │ │ │ │ - point = obj.points[i]; │ │ │ │ │ - feature.geometry = point; │ │ │ │ │ - if ("z" in point) { │ │ │ │ │ - feature.attributes.altitude = point.z; │ │ │ │ │ - } │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - feature.geometry.transform( │ │ │ │ │ - this.externalProjection, this.internalProjection │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (this.trackAttributes) { │ │ │ │ │ - for (var j = 0, jj = this.trackAttributes.length; j < jj; ++j) { │ │ │ │ │ - var name = this.trackAttributes[j]; │ │ │ │ │ - feature.attributes[name] = obj.attributes[name][i]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - feature.attributes.when = obj.whens[i]; │ │ │ │ │ - feature.attributes.trackId = container.feature.id; │ │ │ │ │ - if (hasAngles) { │ │ │ │ │ - angles = obj.angles[i]; │ │ │ │ │ - feature.attributes.heading = parseFloat(angles[0]); │ │ │ │ │ - feature.attributes.tilt = parseFloat(angles[1]); │ │ │ │ │ - feature.attributes.roll = parseFloat(angles[2]); │ │ │ │ │ - } │ │ │ │ │ - container.features.push(feature); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "coord": function(node, container) { │ │ │ │ │ - var str = this.getChildValue(node); │ │ │ │ │ - var coords = str.replace(this.regExes.trimSpace, "").split(/\s+/); │ │ │ │ │ - var point = new OpenLayers.Geometry.Point(coords[0], coords[1]); │ │ │ │ │ - if (coords.length > 2) { │ │ │ │ │ - point.z = parseFloat(coords[2]); │ │ │ │ │ - } │ │ │ │ │ - container.points.push(point); │ │ │ │ │ - }, │ │ │ │ │ - "angles": function(node, container) { │ │ │ │ │ - var str = this.getChildValue(node); │ │ │ │ │ - var parts = str.replace(this.regExes.trimSpace, "").split(/\s+/); │ │ │ │ │ - container.angles.push(parts); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.params = null; │ │ │ │ │ + this.headers = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseFeature │ │ │ │ │ - * This function is the core of the KML parsing code in OpenLayers. │ │ │ │ │ - * It creates the geometries that are then attached to the returned │ │ │ │ │ - * feature, and calls parseAttributes() to get attribute data out. │ │ │ │ │ + * APIMethod: filterToParams │ │ │ │ │ + * Optional method to translate an <OpenLayers.Filter> object into an object │ │ │ │ │ + * that can be serialized as request query string provided. If a custom │ │ │ │ │ + * method is not provided, the filter will be serialized using the │ │ │ │ │ + * <OpenLayers.Format.QueryStringFilter> class. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ + * filter - {<OpenLayers.Filter>} filter to convert. │ │ │ │ │ + * params - {Object} The parameters object. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A vector feature. │ │ │ │ │ + * {Object} The resulting parameters object. │ │ │ │ │ */ │ │ │ │ │ - parseFeature: function(node) { │ │ │ │ │ - // only accept one geometry per feature - look for highest "order" │ │ │ │ │ - var order = ["MultiGeometry", "Polygon", "LineString", "Point"]; │ │ │ │ │ - var type, nodeList, geometry, parser; │ │ │ │ │ - for (var i = 0, len = order.length; i < len; ++i) { │ │ │ │ │ - type = order[i]; │ │ │ │ │ - this.internalns = node.namespaceURI ? │ │ │ │ │ - node.namespaceURI : this.kmlns; │ │ │ │ │ - nodeList = this.getElementsByTagNameNS(node, │ │ │ │ │ - this.internalns, type); │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - // only deal with first geometry of this type │ │ │ │ │ - var parser = this.parseGeometry[type.toLowerCase()]; │ │ │ │ │ - if (parser) { │ │ │ │ │ - geometry = parser.apply(this, [nodeList[0]]); │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry.transform(this.externalProjection, │ │ │ │ │ - this.internalProjection); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - throw new TypeError("Unsupported geometry type: " + type); │ │ │ │ │ - } │ │ │ │ │ - // stop looking for different geometry types │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // construct feature (optionally with attributes) │ │ │ │ │ - var attributes; │ │ │ │ │ - if (this.extractAttributes) { │ │ │ │ │ - attributes = this.parseAttributes(node); │ │ │ │ │ - } │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry, attributes); │ │ │ │ │ - │ │ │ │ │ - var fid = node.getAttribute("id") || node.getAttribute("name"); │ │ │ │ │ - if (fid != null) { │ │ │ │ │ - feature.fid = fid; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return feature; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getStyle │ │ │ │ │ - * Retrieves a style from a style hash using styleUrl as the key │ │ │ │ │ - * If the styleUrl doesn't exist yet, we try to fetch it │ │ │ │ │ - * Internet │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * styleUrl - {String} URL of style │ │ │ │ │ - * options - {Object} Hash of options │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Construct a request for reading new features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * url - {String} Url for the request. │ │ │ │ │ + * params - {Object} Parameters to get serialized as a query string. │ │ │ │ │ + * headers - {Object} Headers to be set on the request. │ │ │ │ │ + * filter - {<OpenLayers.Filter>} Filter to get serialized as a │ │ │ │ │ + * query string. │ │ │ │ │ + * readWithPOST - {Boolean} If the request should be done with POST. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} - (reference to) Style hash │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} A response object, whose "priv" property │ │ │ │ │ + * references the HTTP request, this object is also passed to the │ │ │ │ │ + * callback function when the request completes, its "features" property │ │ │ │ │ + * is then populated with the features received from the server. │ │ │ │ │ */ │ │ │ │ │ - getStyle: function(styleUrl, options) { │ │ │ │ │ - │ │ │ │ │ - var styleBaseUrl = OpenLayers.Util.removeTail(styleUrl); │ │ │ │ │ - │ │ │ │ │ - var newOptions = OpenLayers.Util.extend({}, options); │ │ │ │ │ - newOptions.depth++; │ │ │ │ │ - newOptions.styleBaseUrl = styleBaseUrl; │ │ │ │ │ - │ │ │ │ │ - // Fetch remote Style URLs (if not fetched before) │ │ │ │ │ - if (!this.styles[styleUrl] && │ │ │ │ │ - !OpenLayers.String.startsWith(styleUrl, "#") && │ │ │ │ │ - newOptions.depth <= this.maxDepth && │ │ │ │ │ - !this.fetched[styleBaseUrl]) { │ │ │ │ │ - │ │ │ │ │ - var data = this.fetchLink(styleBaseUrl); │ │ │ │ │ - if (data) { │ │ │ │ │ - this.parseData(data, newOptions); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options.params, this.options.params); │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + if (options.filter && this.filterToParams) { │ │ │ │ │ + options.params = this.filterToParams( │ │ │ │ │ + options.filter, options.params │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // return requested style │ │ │ │ │ - var style = OpenLayers.Util.extend({}, this.styles[styleUrl]); │ │ │ │ │ - return style; │ │ │ │ │ + var readWithPOST = (options.readWithPOST !== undefined) ? │ │ │ │ │ + options.readWithPOST : this.readWithPOST; │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + if (readWithPOST) { │ │ │ │ │ + var headers = options.headers || {}; │ │ │ │ │ + headers["Content-Type"] = "application/x-www-form-urlencoded"; │ │ │ │ │ + resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ + data: OpenLayers.Util.getParameterString(options.params), │ │ │ │ │ + headers: headers │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + resp.priv = OpenLayers.Request.GET({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: parseGeometry │ │ │ │ │ - * Properties of this object are the functions that parse geometries based │ │ │ │ │ - * on their type. │ │ │ │ │ + * Method: handleRead │ │ │ │ │ + * Individual callbacks are created for read, create and update, should │ │ │ │ │ + * a subclass need to override each one separately. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * the user callback. │ │ │ │ │ + * options - {Object} The user options passed to the read call. │ │ │ │ │ */ │ │ │ │ │ - parseGeometry: { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseGeometry.point │ │ │ │ │ - * Given a KML node representing a point geometry, create an OpenLayers │ │ │ │ │ - * point geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A KML Point node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} A point geometry. │ │ │ │ │ - */ │ │ │ │ │ - point: function(node) { │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, this.internalns, │ │ │ │ │ - "coordinates"); │ │ │ │ │ - var coords = []; │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - var coordString = nodeList[0].firstChild.nodeValue; │ │ │ │ │ - coordString = coordString.replace(this.regExes.removeSpace, ""); │ │ │ │ │ - coords = coordString.split(","); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var point = null; │ │ │ │ │ - if (coords.length > 1) { │ │ │ │ │ - // preserve third dimension │ │ │ │ │ - if (coords.length == 2) { │ │ │ │ │ - coords[2] = null; │ │ │ │ │ - } │ │ │ │ │ - point = new OpenLayers.Geometry.Point(coords[0], coords[1], │ │ │ │ │ - coords[2]); │ │ │ │ │ - } else { │ │ │ │ │ - throw "Bad coordinate string: " + coordString; │ │ │ │ │ - } │ │ │ │ │ - return point; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseGeometry.linestring │ │ │ │ │ - * Given a KML node representing a linestring geometry, create an │ │ │ │ │ - * OpenLayers linestring geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A KML LineString node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.LineString>} A linestring geometry. │ │ │ │ │ - */ │ │ │ │ │ - linestring: function(node, ring) { │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, this.internalns, │ │ │ │ │ - "coordinates"); │ │ │ │ │ - var line = null; │ │ │ │ │ - if (nodeList.length > 0) { │ │ │ │ │ - var coordString = this.getChildValue(nodeList[0]); │ │ │ │ │ - │ │ │ │ │ - coordString = coordString.replace(this.regExes.trimSpace, │ │ │ │ │ - ""); │ │ │ │ │ - coordString = coordString.replace(this.regExes.trimComma, │ │ │ │ │ - ","); │ │ │ │ │ - var pointList = coordString.split(this.regExes.splitSpace); │ │ │ │ │ - var numPoints = pointList.length; │ │ │ │ │ - var points = new Array(numPoints); │ │ │ │ │ - var coords, numCoords; │ │ │ │ │ - for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ - coords = pointList[i].split(","); │ │ │ │ │ - numCoords = coords.length; │ │ │ │ │ - if (numCoords > 1) { │ │ │ │ │ - if (coords.length == 2) { │ │ │ │ │ - coords[2] = null; │ │ │ │ │ - } │ │ │ │ │ - points[i] = new OpenLayers.Geometry.Point(coords[0], │ │ │ │ │ - coords[1], │ │ │ │ │ - coords[2]); │ │ │ │ │ - } else { │ │ │ │ │ - throw "Bad LineString point coordinates: " + │ │ │ │ │ - pointList[i]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (numPoints) { │ │ │ │ │ - if (ring) { │ │ │ │ │ - line = new OpenLayers.Geometry.LinearRing(points); │ │ │ │ │ - } else { │ │ │ │ │ - line = new OpenLayers.Geometry.LineString(points); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - throw "Bad LineString coordinates: " + coordString; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return line; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseGeometry.polygon │ │ │ │ │ - * Given a KML node representing a polygon geometry, create an │ │ │ │ │ - * OpenLayers polygon geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A KML Polygon node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Polygon>} A polygon geometry. │ │ │ │ │ - */ │ │ │ │ │ - polygon: function(node) { │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, this.internalns, │ │ │ │ │ - "LinearRing"); │ │ │ │ │ - var numRings = nodeList.length; │ │ │ │ │ - var components = new Array(numRings); │ │ │ │ │ - if (numRings > 0) { │ │ │ │ │ - // this assumes exterior ring first, inner rings after │ │ │ │ │ - var ring; │ │ │ │ │ - for (var i = 0, len = nodeList.length; i < len; ++i) { │ │ │ │ │ - ring = this.parseGeometry.linestring.apply(this, │ │ │ │ │ - [nodeList[i], true]); │ │ │ │ │ - if (ring) { │ │ │ │ │ - components[i] = ring; │ │ │ │ │ - } else { │ │ │ │ │ - throw "Bad LinearRing geometry: " + i; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Geometry.Polygon(components); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseGeometry.multigeometry │ │ │ │ │ - * Given a KML node representing a multigeometry, create an │ │ │ │ │ - * OpenLayers geometry collection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A KML MultiGeometry node. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Collection>} A geometry collection. │ │ │ │ │ - */ │ │ │ │ │ - multigeometry: function(node) { │ │ │ │ │ - var child, parser; │ │ │ │ │ - var parts = []; │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ - child = children[i]; │ │ │ │ │ - if (child.nodeType == 1) { │ │ │ │ │ - var type = (child.prefix) ? │ │ │ │ │ - child.nodeName.split(":")[1] : │ │ │ │ │ - child.nodeName; │ │ │ │ │ - var parser = this.parseGeometry[type.toLowerCase()]; │ │ │ │ │ - if (parser) { │ │ │ │ │ - parts.push(parser.apply(this, [child])); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Geometry.Collection(parts); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ + handleRead: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseAttributes │ │ │ │ │ + * APIMethod: create │ │ │ │ │ + * Construct a request for writing newly created features. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ + * features - {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} An attributes object. │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, whose "priv" property references the HTTP request, this │ │ │ │ │ + * object is also passed to the callback function when the request │ │ │ │ │ + * completes, its "features" property is then populated with the │ │ │ │ │ + * the features received from the server. │ │ │ │ │ */ │ │ │ │ │ - parseAttributes: function(node) { │ │ │ │ │ - var attributes = {}; │ │ │ │ │ + create: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ │ │ │ │ │ - // Extended Data is parsed first. │ │ │ │ │ - var edNodes = node.getElementsByTagName("ExtendedData"); │ │ │ │ │ - if (edNodes.length) { │ │ │ │ │ - attributes = this.parseExtendedData(edNodes[0]); │ │ │ │ │ - } │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: features, │ │ │ │ │ + requestType: "create" │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - // assume attribute nodes are type 1 children with a type 3 or 4 child │ │ │ │ │ - var child, grandchildren, grandchild; │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ + resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleCreate, resp, options), │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(features) │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ - child = children[i]; │ │ │ │ │ - if (child.nodeType == 1) { │ │ │ │ │ - grandchildren = child.childNodes; │ │ │ │ │ - if (grandchildren.length >= 1 && grandchildren.length <= 3) { │ │ │ │ │ - var grandchild; │ │ │ │ │ - switch (grandchildren.length) { │ │ │ │ │ - case 1: │ │ │ │ │ - grandchild = grandchildren[0]; │ │ │ │ │ - break; │ │ │ │ │ - case 2: │ │ │ │ │ - var c1 = grandchildren[0]; │ │ │ │ │ - var c2 = grandchildren[1]; │ │ │ │ │ - grandchild = (c1.nodeType == 3 || c1.nodeType == 4) ? │ │ │ │ │ - c1 : c2; │ │ │ │ │ - break; │ │ │ │ │ - case 3: │ │ │ │ │ - default: │ │ │ │ │ - grandchild = grandchildren[1]; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - if (grandchild.nodeType == 3 || grandchild.nodeType == 4) { │ │ │ │ │ - var name = (child.prefix) ? │ │ │ │ │ - child.nodeName.split(":")[1] : │ │ │ │ │ - child.nodeName; │ │ │ │ │ - var value = OpenLayers.Util.getXmlNodeValue(grandchild); │ │ │ │ │ - if (value) { │ │ │ │ │ - value = value.replace(this.regExes.trimSpace, ""); │ │ │ │ │ - attributes[name] = value; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return attributes; │ │ │ │ │ + return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseExtendedData │ │ │ │ │ - * Parse ExtendedData from KML. Limited support for schemas/datatypes. │ │ │ │ │ - * See http://code.google.com/apis/kml/documentation/kmlreference.html#extendeddata │ │ │ │ │ - * for more information on extendeddata. │ │ │ │ │ + * Method: handleCreate │ │ │ │ │ + * Called the the request issued by <create> is complete. May be overridden │ │ │ │ │ + * by subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the create call. │ │ │ │ │ */ │ │ │ │ │ - parseExtendedData: function(node) { │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - var i, len, data, key; │ │ │ │ │ - var dataNodes = node.getElementsByTagName("Data"); │ │ │ │ │ - for (i = 0, len = dataNodes.length; i < len; i++) { │ │ │ │ │ - data = dataNodes[i]; │ │ │ │ │ - key = data.getAttribute("name"); │ │ │ │ │ - var ed = {}; │ │ │ │ │ - var valueNode = data.getElementsByTagName("value"); │ │ │ │ │ - if (valueNode.length) { │ │ │ │ │ - ed['value'] = this.getChildValue(valueNode[0]); │ │ │ │ │ - } │ │ │ │ │ - if (this.kvpAttributes) { │ │ │ │ │ - attributes[key] = ed['value']; │ │ │ │ │ - } else { │ │ │ │ │ - var nameNode = data.getElementsByTagName("displayName"); │ │ │ │ │ - if (nameNode.length) { │ │ │ │ │ - ed['displayName'] = this.getChildValue(nameNode[0]); │ │ │ │ │ - } │ │ │ │ │ - attributes[key] = ed; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var simpleDataNodes = node.getElementsByTagName("SimpleData"); │ │ │ │ │ - for (i = 0, len = simpleDataNodes.length; i < len; i++) { │ │ │ │ │ - var ed = {}; │ │ │ │ │ - data = simpleDataNodes[i]; │ │ │ │ │ - key = data.getAttribute("name"); │ │ │ │ │ - ed['value'] = this.getChildValue(data); │ │ │ │ │ - if (this.kvpAttributes) { │ │ │ │ │ - attributes[key] = ed['value']; │ │ │ │ │ - } else { │ │ │ │ │ - ed['displayName'] = key; │ │ │ │ │ - attributes[key] = ed; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return attributes; │ │ │ │ │ + handleCreate: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseProperty │ │ │ │ │ - * Convenience method to find a node and return its value │ │ │ │ │ + * APIMethod: update │ │ │ │ │ + * Construct a request updating modified feature. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * xmlNode - {<DOMElement>} │ │ │ │ │ - * namespace - {String} namespace of the node to find │ │ │ │ │ - * tagName - {String} name of the property to parse │ │ │ │ │ - * │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} The value for the requested property (defaults to null) │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, whose "priv" property references the HTTP request, this │ │ │ │ │ + * object is also passed to the callback function when the request │ │ │ │ │ + * completes, its "features" property is then populated with the │ │ │ │ │ + * the feature received from the server. │ │ │ │ │ */ │ │ │ │ │ - parseProperty: function(xmlNode, namespace, tagName) { │ │ │ │ │ - var value; │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(xmlNode, namespace, tagName); │ │ │ │ │ - try { │ │ │ │ │ - value = OpenLayers.Util.getXmlNodeValue(nodeList[0]); │ │ │ │ │ - } catch (e) { │ │ │ │ │ - value = null; │ │ │ │ │ - } │ │ │ │ │ + update: function(feature, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var url = options.url || │ │ │ │ │ + feature.url || │ │ │ │ │ + this.options.url + "/" + feature.fid; │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ │ │ │ │ │ - return value; │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: feature, │ │ │ │ │ + requestType: "update" │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + var method = this.updateWithPOST ? "POST" : "PUT"; │ │ │ │ │ + resp.priv = OpenLayers.Request[method]({ │ │ │ │ │ + url: url, │ │ │ │ │ + callback: this.createCallback(this.handleUpdate, resp, options), │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(feature) │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Accept Feature Collection, and return a string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ + * Method: handleUpdate │ │ │ │ │ + * Called the the request issued by <update> is complete. May be overridden │ │ │ │ │ + * by subclasses. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A KML string. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the update call. │ │ │ │ │ */ │ │ │ │ │ - write: function(features) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "kml"); │ │ │ │ │ - var folder = this.createFolderXML(); │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - folder.appendChild(this.createPlacemarkXML(features[i])); │ │ │ │ │ - } │ │ │ │ │ - kml.appendChild(folder); │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [kml]); │ │ │ │ │ + handleUpdate: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createFolderXML │ │ │ │ │ - * Creates and returns a KML folder node │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: delete │ │ │ │ │ + * Construct a request deleting a removed feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ + * object, whose "priv" property references the HTTP request, this │ │ │ │ │ + * object is also passed to the callback function when the request │ │ │ │ │ + * completes. │ │ │ │ │ */ │ │ │ │ │ - createFolderXML: function() { │ │ │ │ │ - // Folder │ │ │ │ │ - var folder = this.createElementNS(this.kmlns, "Folder"); │ │ │ │ │ + "delete": function(feature, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var url = options.url || │ │ │ │ │ + feature.url || │ │ │ │ │ + this.options.url + "/" + feature.fid; │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ │ │ │ │ │ - // Folder name │ │ │ │ │ - if (this.foldersName) { │ │ │ │ │ - var folderName = this.createElementNS(this.kmlns, "name"); │ │ │ │ │ - var folderNameText = this.createTextNode(this.foldersName); │ │ │ │ │ - folderName.appendChild(folderNameText); │ │ │ │ │ - folder.appendChild(folderName); │ │ │ │ │ - } │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: feature, │ │ │ │ │ + requestType: "delete" │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - // Folder description │ │ │ │ │ - if (this.foldersDesc) { │ │ │ │ │ - var folderDesc = this.createElementNS(this.kmlns, "description"); │ │ │ │ │ - var folderDescText = this.createTextNode(this.foldersDesc); │ │ │ │ │ - folderDesc.appendChild(folderDescText); │ │ │ │ │ - folder.appendChild(folderDesc); │ │ │ │ │ + var method = this.deleteWithPOST ? "POST" : "DELETE"; │ │ │ │ │ + var requestOptions = { │ │ │ │ │ + url: url, │ │ │ │ │ + callback: this.createCallback(this.handleDelete, resp, options), │ │ │ │ │ + headers: options.headers │ │ │ │ │ + }; │ │ │ │ │ + if (this.deleteWithPOST) { │ │ │ │ │ + requestOptions.data = this.format.write(feature); │ │ │ │ │ } │ │ │ │ │ + resp.priv = OpenLayers.Request[method](requestOptions); │ │ │ │ │ │ │ │ │ │ - return folder; │ │ │ │ │ + return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createPlacemarkXML │ │ │ │ │ - * Creates and returns a KML placemark node representing the given feature. │ │ │ │ │ - * │ │ │ │ │ + * Method: handleDelete │ │ │ │ │ + * Called the the request issued by <delete> is complete. May be overridden │ │ │ │ │ + * by subclasses. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the delete call. │ │ │ │ │ */ │ │ │ │ │ - createPlacemarkXML: function(feature) { │ │ │ │ │ - // Placemark name │ │ │ │ │ - var placemarkName = this.createElementNS(this.kmlns, "name"); │ │ │ │ │ - var label = (feature.style && feature.style.label) ? feature.style.label : feature.id; │ │ │ │ │ - var name = feature.attributes.name || label; │ │ │ │ │ - placemarkName.appendChild(this.createTextNode(name)); │ │ │ │ │ - │ │ │ │ │ - // Placemark description │ │ │ │ │ - var placemarkDesc = this.createElementNS(this.kmlns, "description"); │ │ │ │ │ - var desc = feature.attributes.description || this.placemarksDesc; │ │ │ │ │ - placemarkDesc.appendChild(this.createTextNode(desc)); │ │ │ │ │ - │ │ │ │ │ - // Placemark │ │ │ │ │ - var placemarkNode = this.createElementNS(this.kmlns, "Placemark"); │ │ │ │ │ - if (feature.fid != null) { │ │ │ │ │ - placemarkNode.setAttribute("id", feature.fid); │ │ │ │ │ - } │ │ │ │ │ - placemarkNode.appendChild(placemarkName); │ │ │ │ │ - placemarkNode.appendChild(placemarkDesc); │ │ │ │ │ - │ │ │ │ │ - // Geometry node (Point, LineString, etc. nodes) │ │ │ │ │ - var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ - placemarkNode.appendChild(geometryNode); │ │ │ │ │ + handleDelete: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // output attributes as extendedData │ │ │ │ │ - if (feature.attributes) { │ │ │ │ │ - var edNode = this.buildExtendedData(feature.attributes); │ │ │ │ │ - if (edNode) { │ │ │ │ │ - placemarkNode.appendChild(edNode); │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleResponse │ │ │ │ │ + * Called by CRUD specific handlers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the create, read, update, │ │ │ │ │ + * or delete call. │ │ │ │ │ + */ │ │ │ │ │ + handleResponse: function(resp, options) { │ │ │ │ │ + var request = resp.priv; │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + // success │ │ │ │ │ + if (resp.requestType != "delete") { │ │ │ │ │ + resp.features = this.parseFeatures(request); │ │ │ │ │ + } │ │ │ │ │ + resp.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + } else { │ │ │ │ │ + // failure │ │ │ │ │ + resp.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ } │ │ │ │ │ + options.callback.call(options.scope, resp); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return placemarkNode; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildGeometryNode │ │ │ │ │ - * Builds and returns a KML geometry node with the given geometry. │ │ │ │ │ - * │ │ │ │ │ + * Method: parseFeatures │ │ │ │ │ + * Read HTTP response body and return features. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * request - {XMLHttpRequest} The request object │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} Array of features or a single feature. │ │ │ │ │ */ │ │ │ │ │ - buildGeometryNode: function(geometry) { │ │ │ │ │ - var className = geometry.CLASS_NAME; │ │ │ │ │ - var type = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ - var builder = this.buildGeometry[type.toLowerCase()]; │ │ │ │ │ - var node = null; │ │ │ │ │ - if (builder) { │ │ │ │ │ - node = builder.apply(this, [geometry]); │ │ │ │ │ + parseFeatures: function(request) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + return this.format.read(doc); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: buildGeometry │ │ │ │ │ - * Object containing methods to do the actual geometry node building │ │ │ │ │ - * based on geometry type. │ │ │ │ │ + * APIMethod: commit │ │ │ │ │ + * Iterate over each feature and take action based on the feature state. │ │ │ │ │ + * Possible actions are create, update and delete. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array({<OpenLayers.Feature.Vector>})} │ │ │ │ │ + * options - {Object} Optional object for setting up intermediate commit │ │ │ │ │ + * callbacks. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * create - {Object} Optional object to be passed to the <create> method. │ │ │ │ │ + * update - {Object} Optional object to be passed to the <update> method. │ │ │ │ │ + * delete - {Object} Optional object to be passed to the <delete> method. │ │ │ │ │ + * callback - {Function} Optional function to be called when the commit │ │ │ │ │ + * is complete. │ │ │ │ │ + * scope - {Object} Optional object to be set as the scope of the callback. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Protocol.Response>)} An array of response objects, │ │ │ │ │ + * one per request made to the server, each object's "priv" property │ │ │ │ │ + * references the corresponding HTTP request. │ │ │ │ │ */ │ │ │ │ │ - buildGeometry: { │ │ │ │ │ - // TBD: Anybody care about namespace aliases here (these nodes have │ │ │ │ │ - // no prefixes)? │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.point │ │ │ │ │ - * Given an OpenLayers point geometry, create a KML point. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Point>} A point geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML point node. │ │ │ │ │ - */ │ │ │ │ │ - point: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "Point"); │ │ │ │ │ - kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - return kml; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.multipoint │ │ │ │ │ - * Given an OpenLayers multipoint geometry, create a KML │ │ │ │ │ - * GeometryCollection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Point>} A multipoint geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ - */ │ │ │ │ │ - multipoint: function(geometry) { │ │ │ │ │ - return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.linestring │ │ │ │ │ - * Given an OpenLayers linestring geometry, create a KML linestring. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.LineString>} A linestring geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML linestring node. │ │ │ │ │ - */ │ │ │ │ │ - linestring: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "LineString"); │ │ │ │ │ - kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - return kml; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.multilinestring │ │ │ │ │ - * Given an OpenLayers multilinestring geometry, create a KML │ │ │ │ │ - * GeometryCollection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Point>} A multilinestring geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ - */ │ │ │ │ │ - multilinestring: function(geometry) { │ │ │ │ │ - return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.linearring │ │ │ │ │ - * Given an OpenLayers linearring geometry, create a KML linearring. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.LinearRing>} A linearring geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML linearring node. │ │ │ │ │ - */ │ │ │ │ │ - linearring: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "LinearRing"); │ │ │ │ │ - kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - return kml; │ │ │ │ │ - }, │ │ │ │ │ + commit: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = [], │ │ │ │ │ + nResponses = 0; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.polygon │ │ │ │ │ - * Given an OpenLayers polygon geometry, create a KML polygon. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Polygon>} A polygon geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML polygon node. │ │ │ │ │ - */ │ │ │ │ │ - polygon: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "Polygon"); │ │ │ │ │ - var rings = geometry.components; │ │ │ │ │ - var ringMember, ringGeom, type; │ │ │ │ │ - for (var i = 0, len = rings.length; i < len; ++i) { │ │ │ │ │ - type = (i == 0) ? "outerBoundaryIs" : "innerBoundaryIs"; │ │ │ │ │ - ringMember = this.createElementNS(this.kmlns, type); │ │ │ │ │ - ringGeom = this.buildGeometry.linearring.apply(this, │ │ │ │ │ - [rings[i]]); │ │ │ │ │ - ringMember.appendChild(ringGeom); │ │ │ │ │ - kml.appendChild(ringMember); │ │ │ │ │ + // Divide up features before issuing any requests. This properly │ │ │ │ │ + // counts requests in the event that any responses come in before │ │ │ │ │ + // all requests have been issued. │ │ │ │ │ + var types = {}; │ │ │ │ │ + types[OpenLayers.State.INSERT] = []; │ │ │ │ │ + types[OpenLayers.State.UPDATE] = []; │ │ │ │ │ + types[OpenLayers.State.DELETE] = []; │ │ │ │ │ + var feature, list, requestFeatures = []; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + list = types[feature.state]; │ │ │ │ │ + if (list) { │ │ │ │ │ + list.push(feature); │ │ │ │ │ + requestFeatures.push(feature); │ │ │ │ │ } │ │ │ │ │ - return kml; │ │ │ │ │ - }, │ │ │ │ │ + } │ │ │ │ │ + // tally up number of requests │ │ │ │ │ + var nRequests = (types[OpenLayers.State.INSERT].length > 0 ? 1 : 0) + │ │ │ │ │ + types[OpenLayers.State.UPDATE].length + │ │ │ │ │ + types[OpenLayers.State.DELETE].length; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.multipolygon │ │ │ │ │ - * Given an OpenLayers multipolygon geometry, create a KML │ │ │ │ │ - * GeometryCollection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Point>} A multipolygon geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ - */ │ │ │ │ │ - multipolygon: function(geometry) { │ │ │ │ │ - return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ - }, │ │ │ │ │ + // This response will be sent to the final callback after all the others │ │ │ │ │ + // have been fired. │ │ │ │ │ + var success = true; │ │ │ │ │ + var finalResponse = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: requestFeatures │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometry.collection │ │ │ │ │ - * Given an OpenLayers geometry collection, create a KML MultiGeometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Collection>} A geometry collection. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A KML MultiGeometry node. │ │ │ │ │ - */ │ │ │ │ │ - collection: function(geometry) { │ │ │ │ │ - var kml = this.createElementNS(this.kmlns, "MultiGeometry"); │ │ │ │ │ - var child; │ │ │ │ │ - for (var i = 0, len = geometry.components.length; i < len; ++i) { │ │ │ │ │ - child = this.buildGeometryNode.apply(this, │ │ │ │ │ - [geometry.components[i]]); │ │ │ │ │ - if (child) { │ │ │ │ │ - kml.appendChild(child); │ │ │ │ │ - } │ │ │ │ │ + function insertCallback(response) { │ │ │ │ │ + var len = response.features ? response.features.length : 0; │ │ │ │ │ + var fids = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + fids[i] = response.features[i].fid; │ │ │ │ │ } │ │ │ │ │ - return kml; │ │ │ │ │ + finalResponse.insertIds = fids; │ │ │ │ │ + callback.apply(this, [response]); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildCoordinatesNode │ │ │ │ │ - * Builds and returns the KML coordinates node with the given geometry │ │ │ │ │ - * <coordinates>...</coordinates> │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - buildCoordinatesNode: function(geometry) { │ │ │ │ │ - var coordinatesNode = this.createElementNS(this.kmlns, "coordinates"); │ │ │ │ │ │ │ │ │ │ - var path; │ │ │ │ │ - var points = geometry.components; │ │ │ │ │ - if (points) { │ │ │ │ │ - // LineString or LinearRing │ │ │ │ │ - var point; │ │ │ │ │ - var numPoints = points.length; │ │ │ │ │ - var parts = new Array(numPoints); │ │ │ │ │ - for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ - point = points[i]; │ │ │ │ │ - parts[i] = this.buildCoordinates(point); │ │ │ │ │ + function callback(response) { │ │ │ │ │ + this.callUserCallback(response, options); │ │ │ │ │ + success = success && response.success(); │ │ │ │ │ + nResponses++; │ │ │ │ │ + if (nResponses >= nRequests) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + finalResponse.code = success ? │ │ │ │ │ + OpenLayers.Protocol.Response.SUCCESS : │ │ │ │ │ + OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + options.callback.apply(options.scope, [finalResponse]); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - path = parts.join(" "); │ │ │ │ │ - } else { │ │ │ │ │ - // Point │ │ │ │ │ - path = this.buildCoordinates(geometry); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - var txtNode = this.createTextNode(path); │ │ │ │ │ - coordinatesNode.appendChild(txtNode); │ │ │ │ │ - │ │ │ │ │ - return coordinatesNode; │ │ │ │ │ + // start issuing requests │ │ │ │ │ + var queue = types[OpenLayers.State.INSERT]; │ │ │ │ │ + if (queue.length > 0) { │ │ │ │ │ + resp.push(this.create( │ │ │ │ │ + queue, OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: insertCallback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options.create) │ │ │ │ │ + )); │ │ │ │ │ + } │ │ │ │ │ + queue = types[OpenLayers.State.UPDATE]; │ │ │ │ │ + for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ + resp.push(this.update( │ │ │ │ │ + queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: callback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options.update))); │ │ │ │ │ + } │ │ │ │ │ + queue = types[OpenLayers.State.DELETE]; │ │ │ │ │ + for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ + resp.push(this["delete"]( │ │ │ │ │ + queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: callback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options["delete"]))); │ │ │ │ │ + } │ │ │ │ │ + return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildCoordinates │ │ │ │ │ + * APIMethod: abort │ │ │ │ │ + * Abort an ongoing request, the response object passed to │ │ │ │ │ + * this method must come from this HTTP protocol (as a result │ │ │ │ │ + * of a create, read, update, delete or commit operation). │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ - * │ │ │ │ │ - * Returns │ │ │ │ │ - * {String} a coordinate pair │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ */ │ │ │ │ │ - buildCoordinates: function(point) { │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - point = point.clone(); │ │ │ │ │ - point.transform(this.internalProjection, │ │ │ │ │ - this.externalProjection); │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + response.priv.abort(); │ │ │ │ │ } │ │ │ │ │ - return point.x + "," + point.y; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildExtendedData │ │ │ │ │ + * Method: callUserCallback │ │ │ │ │ + * This method is used from within the commit method each time an │ │ │ │ │ + * an HTTP response is received from the server, it is responsible │ │ │ │ │ + * for calling the user-supplied callbacks. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * attributes - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns │ │ │ │ │ - * {DOMElement} A KML ExtendedData node or {null} if no attributes. │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} │ │ │ │ │ + * options - {Object} The map of options passed to the commit call. │ │ │ │ │ */ │ │ │ │ │ - buildExtendedData: function(attributes) { │ │ │ │ │ - var extendedData = this.createElementNS(this.kmlns, "ExtendedData"); │ │ │ │ │ - for (var attributeName in attributes) { │ │ │ │ │ - // empty, name, description, styleUrl attributes ignored │ │ │ │ │ - if (attributes[attributeName] && attributeName != "name" && attributeName != "description" && attributeName != "styleUrl") { │ │ │ │ │ - var data = this.createElementNS(this.kmlns, "Data"); │ │ │ │ │ - data.setAttribute("name", attributeName); │ │ │ │ │ - var value = this.createElementNS(this.kmlns, "value"); │ │ │ │ │ - if (typeof attributes[attributeName] == "object") { │ │ │ │ │ - // cater for object attributes with 'value' properties │ │ │ │ │ - // other object properties will output an empty node │ │ │ │ │ - if (attributes[attributeName].value) { │ │ │ │ │ - value.appendChild(this.createTextNode(attributes[attributeName].value)); │ │ │ │ │ - } │ │ │ │ │ - if (attributes[attributeName].displayName) { │ │ │ │ │ - var displayName = this.createElementNS(this.kmlns, "displayName"); │ │ │ │ │ - // displayName always written as CDATA │ │ │ │ │ - displayName.appendChild(this.getXMLDoc().createCDATASection(attributes[attributeName].displayName)); │ │ │ │ │ - data.appendChild(displayName); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - value.appendChild(this.createTextNode(attributes[attributeName])); │ │ │ │ │ - } │ │ │ │ │ - data.appendChild(value); │ │ │ │ │ - extendedData.appendChild(data); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.isSimpleContent(extendedData)) { │ │ │ │ │ - return null; │ │ │ │ │ - } else { │ │ │ │ │ - return extendedData; │ │ │ │ │ + callUserCallback: function(resp, options) { │ │ │ │ │ + var opt = options[resp.requestType]; │ │ │ │ │ + if (opt && opt.callback) { │ │ │ │ │ + opt.callback.call(opt.scope, resp); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.KML" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.HTTP" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WCSCapabilities.js │ │ │ │ │ + OpenLayers/Protocol/Script.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + * @requires OpenLayers/Protocol.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Format/GeoJSON.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WCSCapabilities │ │ │ │ │ - * Read WCS Capabilities. │ │ │ │ │ - * │ │ │ │ │ + * if application uses the query string, for example, for BBOX parameters, │ │ │ │ │ + * OpenLayers/Format/QueryStringFilter.js should be included in the build config file │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.Script │ │ │ │ │ + * A basic Script protocol for vector layers. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Protocol.Script> constructor. A script protocol is used to │ │ │ │ │ + * get around the same origin policy. It works with services that return │ │ │ │ │ + * JSONP - that is, JSON wrapped in a client-specified callback. The │ │ │ │ │ + * protocol handles fetching and parsing of feature data and sends parsed │ │ │ │ │ + * features to the <callback> configured with the protocol. The protocol │ │ │ │ │ + * expects features serialized as GeoJSON by default, but can be configured │ │ │ │ │ + * to work with other formats by setting the <format> property. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + * - <OpenLayers.Protocol> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WCSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ +OpenLayers.Protocol.Script = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.1.0". │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String} Service URL. The service is expected to return serialized │ │ │ │ │ + * features wrapped in a named callback (where the callback name is │ │ │ │ │ + * generated by this protocol). │ │ │ │ │ + * Read-only, set through the options passed to the constructor. │ │ │ │ │ */ │ │ │ │ │ - defaultVersion: "1.1.0", │ │ │ │ │ + url: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WCSCapabilities │ │ │ │ │ - * Create a new parser for WCS capabilities. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * APIProperty: params │ │ │ │ │ + * {Object} Query string parameters to be appended to the URL. │ │ │ │ │ + * Read-only, set through the options passed to the constructor. │ │ │ │ │ + * Example: {maxFeatures: 50} │ │ │ │ │ */ │ │ │ │ │ + params: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return a list of coverages. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} List of named coverages. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WCSCapabilities" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/GPX.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ - * @requires OpenLayers/Projection.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.GPX │ │ │ │ │ - * Read/write GPX parser. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Format.GPX> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.GPX = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: defaultDesc │ │ │ │ │ - * {String} Default description for the waypoints/tracks in the case │ │ │ │ │ - * where the feature has no "description" attribute. │ │ │ │ │ - * Default is "No description available". │ │ │ │ │ + * APIProperty: callback │ │ │ │ │ + * {Object} Function to be called when the <read> operation completes. │ │ │ │ │ */ │ │ │ │ │ - defaultDesc: "No description available", │ │ │ │ │ + callback: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: extractWaypoints │ │ │ │ │ - * {Boolean} Extract waypoints from GPX. (default: true) │ │ │ │ │ + * APIProperty: callbackTemplate │ │ │ │ │ + * {String} Template for creating a unique callback function name │ │ │ │ │ + * for the registry. Should include ${id}. The ${id} variable will be │ │ │ │ │ + * replaced with a string identifier prefixed with a "c" (e.g. c1, c2). │ │ │ │ │ + * Default is "OpenLayers.Protocol.Script.registry.${id}". │ │ │ │ │ */ │ │ │ │ │ - extractWaypoints: true, │ │ │ │ │ + callbackTemplate: "OpenLayers.Protocol.Script.registry.${id}", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: extractTracks │ │ │ │ │ - * {Boolean} Extract tracks from GPX. (default: true) │ │ │ │ │ + * APIProperty: callbackKey │ │ │ │ │ + * {String} The name of the query string parameter that the service │ │ │ │ │ + * recognizes as the callback identifier. Default is "callback". │ │ │ │ │ + * This key is used to generate the URL for the script. For example │ │ │ │ │ + * setting <callbackKey> to "myCallback" would result in a URL like │ │ │ │ │ + * http://example.com/?myCallback=... │ │ │ │ │ */ │ │ │ │ │ - extractTracks: true, │ │ │ │ │ + callbackKey: "callback", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: extractRoutes │ │ │ │ │ - * {Boolean} Extract routes from GPX. (default: true) │ │ │ │ │ + * APIProperty: callbackPrefix │ │ │ │ │ + * {String} Where a service requires that the callback query string │ │ │ │ │ + * parameter value is prefixed by some string, this value may be set. │ │ │ │ │ + * For example, setting <callbackPrefix> to "foo:" would result in a │ │ │ │ │ + * URL like http://example.com/?callback=foo:... Default is "". │ │ │ │ │ */ │ │ │ │ │ - extractRoutes: true, │ │ │ │ │ + callbackPrefix: "", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: extractAttributes │ │ │ │ │ - * {Boolean} Extract feature attributes from GPX. (default: true) │ │ │ │ │ - * NOTE: Attributes as part of extensions to the GPX standard may not │ │ │ │ │ - * be extracted. │ │ │ │ │ + * APIProperty: scope │ │ │ │ │ + * {Object} Optional ``this`` object for the callback. Read-only, set │ │ │ │ │ + * through the options passed to the constructor. │ │ │ │ │ */ │ │ │ │ │ - extractAttributes: true, │ │ │ │ │ + scope: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + * APIProperty: format │ │ │ │ │ + * {<OpenLayers.Format>} Format for parsing features. Default is an │ │ │ │ │ + * <OpenLayers.Format.GeoJSON> format. If an alternative is provided, │ │ │ │ │ + * the format's read method must take an object and return an array │ │ │ │ │ + * of features. │ │ │ │ │ */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - gpx: "http://www.topografix.com/GPX/1/1", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ - }, │ │ │ │ │ + format: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} Schema location. Defaults to │ │ │ │ │ - * "http://www.topografix.com/GPX/1/1 http://www.topografix.com/GPX/1/1/gpx.xsd" │ │ │ │ │ + * Property: pendingRequests │ │ │ │ │ + * {Object} References all pending requests. Property names are script │ │ │ │ │ + * identifiers and property values are script elements. │ │ │ │ │ */ │ │ │ │ │ - schemaLocation: "http://www.topografix.com/GPX/1/1 http://www.topografix.com/GPX/1/1/gpx.xsd", │ │ │ │ │ + pendingRequests: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: creator │ │ │ │ │ - * {String} The creator attribute to be added to the written GPX files. │ │ │ │ │ - * Defaults to "OpenLayers" │ │ │ │ │ + * APIProperty: srsInBBOX │ │ │ │ │ + * {Boolean} Include the SRS identifier in BBOX query string parameter. │ │ │ │ │ + * Setting this property has no effect if a custom filterToParams method │ │ │ │ │ + * is provided. Default is false. If true and the layer has a │ │ │ │ │ + * projection object set, any BBOX filter will be serialized with a │ │ │ │ │ + * fifth item identifying the projection. │ │ │ │ │ + * E.g. bbox=-1000,-1000,1000,1000,EPSG:900913 │ │ │ │ │ */ │ │ │ │ │ - creator: "OpenLayers", │ │ │ │ │ + srsInBBOX: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.GPX │ │ │ │ │ - * Create a new parser for GPX. │ │ │ │ │ + * Constructor: OpenLayers.Protocol.Script │ │ │ │ │ + * A class for giving layers generic Script protocol. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + * │ │ │ │ │ + * Valid options include: │ │ │ │ │ + * url - {String} │ │ │ │ │ + * params - {Object} │ │ │ │ │ + * callback - {Function} │ │ │ │ │ + * scope - {Object} │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - // GPX coordinates are always in longlat WGS84 │ │ │ │ │ - this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + options = options || {}; │ │ │ │ │ + this.params = {}; │ │ │ │ │ + this.pendingRequests = {}; │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (!this.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.GeoJSON(); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ + var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ + srsInBBOX: this.srsInBBOX │ │ │ │ │ + }); │ │ │ │ │ + this.filterToParams = function(filter, params) { │ │ │ │ │ + return format.write(filter, params); │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIMethod: read │ │ │ │ │ - * Return a list of features from a GPX doc │ │ │ │ │ + * Construct a request for reading new features. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * doc - {Element} │ │ │ │ │ + * options - {Object} Optional object for configuring the request. │ │ │ │ │ + * This object is modified and should not be reused. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * url - {String} Url for the request. │ │ │ │ │ + * params - {Object} Parameters to get serialized as a query string. │ │ │ │ │ + * filter - {<OpenLayers.Filter>} Filter to get serialized as a │ │ │ │ │ + * query string. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} A response object, whose "priv" property │ │ │ │ │ + * references the injected script. This object is also passed to the │ │ │ │ │ + * callback function when the request completes, its "features" property │ │ │ │ │ + * is then populated with the features received from the server. │ │ │ │ │ */ │ │ │ │ │ - read: function(doc) { │ │ │ │ │ - if (typeof doc == "string") { │ │ │ │ │ - doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]); │ │ │ │ │ - } │ │ │ │ │ - var features = []; │ │ │ │ │ - │ │ │ │ │ - if (this.extractTracks) { │ │ │ │ │ - var tracks = doc.getElementsByTagName("trk"); │ │ │ │ │ - for (var i = 0, len = tracks.length; i < len; i++) { │ │ │ │ │ - // Attributes are only in trk nodes, not trkseg nodes │ │ │ │ │ - var attrs = {}; │ │ │ │ │ - if (this.extractAttributes) { │ │ │ │ │ - attrs = this.parseAttributes(tracks[i]); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var segs = this.getElementsByTagNameNS(tracks[i], tracks[i].namespaceURI, "trkseg"); │ │ │ │ │ - for (var j = 0, seglen = segs.length; j < seglen; j++) { │ │ │ │ │ - // We don't yet support extraction of trkpt attributes │ │ │ │ │ - // All trksegs of a trk get that trk's attributes │ │ │ │ │ - var track = this.extractSegment(segs[j], "trkpt"); │ │ │ │ │ - features.push(new OpenLayers.Feature.Vector(track, attrs)); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.extractRoutes) { │ │ │ │ │ - var routes = doc.getElementsByTagName("rte"); │ │ │ │ │ - for (var k = 0, klen = routes.length; k < klen; k++) { │ │ │ │ │ - var attrs = {}; │ │ │ │ │ - if (this.extractAttributes) { │ │ │ │ │ - attrs = this.parseAttributes(routes[k]); │ │ │ │ │ - } │ │ │ │ │ - var route = this.extractSegment(routes[k], "rtept"); │ │ │ │ │ - features.push(new OpenLayers.Feature.Vector(route, attrs)); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.extractWaypoints) { │ │ │ │ │ - var waypoints = doc.getElementsByTagName("wpt"); │ │ │ │ │ - for (var l = 0, len = waypoints.length; l < len; l++) { │ │ │ │ │ - var attrs = {}; │ │ │ │ │ - if (this.extractAttributes) { │ │ │ │ │ - attrs = this.parseAttributes(waypoints[l]); │ │ │ │ │ - } │ │ │ │ │ - var wpt = new OpenLayers.Geometry.Point(waypoints[l].getAttribute("lon"), waypoints[l].getAttribute("lat")); │ │ │ │ │ - features.push(new OpenLayers.Feature.Vector(wpt, attrs)); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - for (var g = 0, featLength = features.length; g < featLength; g++) { │ │ │ │ │ - features[g].geometry.transform(this.externalProjection, │ │ │ │ │ - this.internalProjection); │ │ │ │ │ - } │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + options.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options.params, this.options.params │ │ │ │ │ + ); │ │ │ │ │ + if (options.filter && this.filterToParams) { │ │ │ │ │ + options.params = this.filterToParams( │ │ │ │ │ + options.filter, options.params │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return features; │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + var request = this.createRequest( │ │ │ │ │ + options.url, │ │ │ │ │ + options.params, │ │ │ │ │ + OpenLayers.Function.bind(function(data) { │ │ │ │ │ + response.data = data; │ │ │ │ │ + this.handleRead(response, options); │ │ │ │ │ + }, this) │ │ │ │ │ + ); │ │ │ │ │ + response.priv = request; │ │ │ │ │ + return response; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: extractSegment │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: filterToParams │ │ │ │ │ + * Optional method to translate an <OpenLayers.Filter> object into an object │ │ │ │ │ + * that can be serialized as request query string provided. If a custom │ │ │ │ │ + * method is not provided, any filter will not be serialized. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {<OpenLayers.Filter>} filter to convert. │ │ │ │ │ + * params - {Object} The parameters object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The resulting parameters object. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createRequest │ │ │ │ │ + * Issues a request for features by creating injecting a script in the │ │ │ │ │ + * document head. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * segment - {DOMElement} a trkseg or rte node to parse │ │ │ │ │ - * segmentType - {String} nodeName of waypoints that form the line │ │ │ │ │ + * url - {String} Service URL. │ │ │ │ │ + * params - {Object} Query string parameters. │ │ │ │ │ + * callback - {Function} Callback to be called with resulting data. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.LineString>} A linestring geometry │ │ │ │ │ + * {HTMLScriptElement} The script pending execution. │ │ │ │ │ */ │ │ │ │ │ - extractSegment: function(segment, segmentType) { │ │ │ │ │ - var points = this.getElementsByTagNameNS(segment, segment.namespaceURI, segmentType); │ │ │ │ │ - var point_features = []; │ │ │ │ │ - for (var i = 0, len = points.length; i < len; i++) { │ │ │ │ │ - point_features.push(new OpenLayers.Geometry.Point(points[i].getAttribute("lon"), points[i].getAttribute("lat"))); │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Geometry.LineString(point_features); │ │ │ │ │ + createRequest: function(url, params, callback) { │ │ │ │ │ + var id = OpenLayers.Protocol.Script.register(callback); │ │ │ │ │ + var name = OpenLayers.String.format(this.callbackTemplate, { │ │ │ │ │ + id: id │ │ │ │ │ + }); │ │ │ │ │ + params = OpenLayers.Util.extend({}, params); │ │ │ │ │ + params[this.callbackKey] = this.callbackPrefix + name; │ │ │ │ │ + url = OpenLayers.Util.urlAppend( │ │ │ │ │ + url, OpenLayers.Util.getParameterString(params) │ │ │ │ │ + ); │ │ │ │ │ + var script = document.createElement("script"); │ │ │ │ │ + script.type = "text/javascript"; │ │ │ │ │ + script.src = url; │ │ │ │ │ + script.id = "OpenLayers_Protocol_Script_" + id; │ │ │ │ │ + this.pendingRequests[script.id] = script; │ │ │ │ │ + var head = document.getElementsByTagName("head")[0]; │ │ │ │ │ + head.appendChild(script); │ │ │ │ │ + return script; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseAttributes │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroyRequest │ │ │ │ │ + * Remove a script node associated with a response from the document. Also │ │ │ │ │ + * unregisters the callback and removes the script from the │ │ │ │ │ + * <pendingRequests> object. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {<DOMElement>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An attributes object. │ │ │ │ │ + * script - {HTMLScriptElement} │ │ │ │ │ */ │ │ │ │ │ - parseAttributes: function(node) { │ │ │ │ │ - // node is either a wpt, trk or rte │ │ │ │ │ - // attributes are children of the form <attr>value</attr> │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - var attrNode = node.firstChild, │ │ │ │ │ - value, name; │ │ │ │ │ - while (attrNode) { │ │ │ │ │ - if (attrNode.nodeType == 1 && attrNode.firstChild) { │ │ │ │ │ - value = attrNode.firstChild; │ │ │ │ │ - if (value.nodeType == 3 || value.nodeType == 4) { │ │ │ │ │ - name = (attrNode.prefix) ? │ │ │ │ │ - attrNode.nodeName.split(":")[1] : │ │ │ │ │ - attrNode.nodeName; │ │ │ │ │ - if (name != "trkseg" && name != "rtept") { │ │ │ │ │ - attributes[name] = value.nodeValue; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - attrNode = attrNode.nextSibling; │ │ │ │ │ + destroyRequest: function(script) { │ │ │ │ │ + OpenLayers.Protocol.Script.unregister(script.id.split("_").pop()); │ │ │ │ │ + delete this.pendingRequests[script.id]; │ │ │ │ │ + if (script.parentNode) { │ │ │ │ │ + script.parentNode.removeChild(script); │ │ │ │ │ } │ │ │ │ │ - return attributes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Accepts Feature Collection, and returns a string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} List of features to serialize into a string. │ │ │ │ │ - * metadata - {Object} A key/value pairs object to build a metadata node to │ │ │ │ │ - * add to the gpx. Supported keys are 'name', 'desc', 'author'. │ │ │ │ │ + * Method: handleRead │ │ │ │ │ + * Individual callbacks are created for read, create and update, should │ │ │ │ │ + * a subclass need to override each one separately. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * the user callback. │ │ │ │ │ + * options - {Object} The user options passed to the read call. │ │ │ │ │ */ │ │ │ │ │ - write: function(features, metadata) { │ │ │ │ │ - features = OpenLayers.Util.isArray(features) ? │ │ │ │ │ - features : [features]; │ │ │ │ │ - var gpx = this.createElementNS(this.namespaces.gpx, "gpx"); │ │ │ │ │ - gpx.setAttribute("version", "1.1"); │ │ │ │ │ - gpx.setAttribute("creator", this.creator); │ │ │ │ │ - this.setAttributes(gpx, { │ │ │ │ │ - "xsi:schemaLocation": this.schemaLocation │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - if (metadata && typeof metadata == 'object') { │ │ │ │ │ - gpx.appendChild(this.buildMetadataNode(metadata)); │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - gpx.appendChild(this.buildFeatureNode(features[i])); │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [gpx]); │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + this.handleResponse(response, options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildMetadataNode │ │ │ │ │ - * Creates a "metadata" node. │ │ │ │ │ + * Method: handleResponse │ │ │ │ │ + * Called by CRUD specific handlers. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ + * any user callback. │ │ │ │ │ + * options - {Object} The user options passed to the create, read, update, │ │ │ │ │ + * or delete call. │ │ │ │ │ */ │ │ │ │ │ - buildMetadataNode: function(metadata) { │ │ │ │ │ - var types = ['name', 'desc', 'author'], │ │ │ │ │ - node = this.createElementNS(this.namespaces.gpx, 'metadata'); │ │ │ │ │ - for (var i = 0; i < types.length; i++) { │ │ │ │ │ - var type = types[i]; │ │ │ │ │ - if (metadata[type]) { │ │ │ │ │ - var n = this.createElementNS(this.namespaces.gpx, type); │ │ │ │ │ - n.appendChild(this.createTextNode(metadata[type])); │ │ │ │ │ - node.appendChild(n); │ │ │ │ │ + handleResponse: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + if (response.data) { │ │ │ │ │ + response.features = this.parseFeatures(response.data); │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ } │ │ │ │ │ + this.destroyRequest(response.priv); │ │ │ │ │ + options.callback.call(options.scope, response); │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildFeatureNode │ │ │ │ │ - * Accepts an <OpenLayers.Feature.Vector>, and builds a node for it. │ │ │ │ │ - * │ │ │ │ │ + * Method: parseFeatures │ │ │ │ │ + * Read Script response body and return features. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * data - {Object} The data sent to the callback function by the server. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} - The created node, either a 'wpt' or a 'trk'. │ │ │ │ │ + * {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} Array of features or a single feature. │ │ │ │ │ */ │ │ │ │ │ - buildFeatureNode: function(feature) { │ │ │ │ │ - var geometry = feature.geometry; │ │ │ │ │ - geometry = geometry.clone(); │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry.transform(this.internalProjection, │ │ │ │ │ - this.externalProjection); │ │ │ │ │ - } │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - var wpt = this.buildWptNode(geometry); │ │ │ │ │ - this.appendAttributesNode(wpt, feature); │ │ │ │ │ - return wpt; │ │ │ │ │ - } else { │ │ │ │ │ - var trkNode = this.createElementNS(this.namespaces.gpx, "trk"); │ │ │ │ │ - this.appendAttributesNode(trkNode, feature); │ │ │ │ │ - var trkSegNodes = this.buildTrkSegNode(geometry); │ │ │ │ │ - trkSegNodes = OpenLayers.Util.isArray(trkSegNodes) ? │ │ │ │ │ - trkSegNodes : [trkSegNodes]; │ │ │ │ │ - for (var i = 0, len = trkSegNodes.length; i < len; i++) { │ │ │ │ │ - trkNode.appendChild(trkSegNodes[i]); │ │ │ │ │ - } │ │ │ │ │ - return trkNode; │ │ │ │ │ - } │ │ │ │ │ + parseFeatures: function(data) { │ │ │ │ │ + return this.format.read(data); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildTrkSegNode │ │ │ │ │ - * Builds trkseg node(s) given a geometry │ │ │ │ │ + * APIMethod: abort │ │ │ │ │ + * Abort an ongoing request. If no response is provided, all pending │ │ │ │ │ + * requests will be aborted. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * trknode │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The response object returned │ │ │ │ │ + * from a <read> request. │ │ │ │ │ */ │ │ │ │ │ - buildTrkSegNode: function(geometry) { │ │ │ │ │ - var node, │ │ │ │ │ - i, │ │ │ │ │ - len, │ │ │ │ │ - point, │ │ │ │ │ - nodes; │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.LineString" || │ │ │ │ │ - geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ - node = this.createElementNS(this.namespaces.gpx, "trkseg"); │ │ │ │ │ - for (i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - point = geometry.components[i]; │ │ │ │ │ - node.appendChild(this.buildTrkPtNode(point)); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + this.destroyRequest(response.priv); │ │ │ │ │ } else { │ │ │ │ │ - nodes = []; │ │ │ │ │ - for (i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - nodes.push(this.buildTrkSegNode(geometry.components[i])); │ │ │ │ │ + for (var key in this.pendingRequests) { │ │ │ │ │ + this.destroyRequest(this.pendingRequests[key]); │ │ │ │ │ } │ │ │ │ │ - return nodes; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildTrkPtNode │ │ │ │ │ - * Builds a trkpt node given a point │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A trkpt node │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up the protocol. │ │ │ │ │ */ │ │ │ │ │ - buildTrkPtNode: function(point) { │ │ │ │ │ - var node = this.createElementNS(this.namespaces.gpx, "trkpt"); │ │ │ │ │ - node.setAttribute("lon", point.x); │ │ │ │ │ - node.setAttribute("lat", point.y); │ │ │ │ │ - return node; │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.abort(); │ │ │ │ │ + delete this.params; │ │ │ │ │ + delete this.format; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.Script" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +(function() { │ │ │ │ │ + var o = OpenLayers.Protocol.Script; │ │ │ │ │ + var counter = 0; │ │ │ │ │ + o.registry = {}; │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: buildWptNode │ │ │ │ │ - * Builds a wpt node given a point │ │ │ │ │ + * Function: OpenLayers.Protocol.Script.register │ │ │ │ │ + * Register a callback for a newly created script. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * callback - {Function} The callback to be executed when the newly added │ │ │ │ │ + * script loads. This callback will be called with a single argument │ │ │ │ │ + * that is the JSON returned by the service. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} A wpt node │ │ │ │ │ + * {Number} An identifier for retrieving the registered callback. │ │ │ │ │ */ │ │ │ │ │ - buildWptNode: function(geometry) { │ │ │ │ │ - var node = this.createElementNS(this.namespaces.gpx, "wpt"); │ │ │ │ │ - node.setAttribute("lon", geometry.x); │ │ │ │ │ - node.setAttribute("lat", geometry.y); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ + o.register = function(callback) { │ │ │ │ │ + var id = "c" + (++counter); │ │ │ │ │ + o.registry[id] = function() { │ │ │ │ │ + callback.apply(this, arguments); │ │ │ │ │ + }; │ │ │ │ │ + return id; │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: appendAttributesNode │ │ │ │ │ - * Adds some attributes node. │ │ │ │ │ + * Function: OpenLayers.Protocol.Script.unregister │ │ │ │ │ + * Unregister a callback previously registered with the register function. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} the node to append the attribute nodes to. │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * id - {Number} The identifer returned by the register function. │ │ │ │ │ */ │ │ │ │ │ - appendAttributesNode: function(node, feature) { │ │ │ │ │ - var name = this.createElementNS(this.namespaces.gpx, 'name'); │ │ │ │ │ - name.appendChild(this.createTextNode( │ │ │ │ │ - feature.attributes.name || feature.id)); │ │ │ │ │ - node.appendChild(name); │ │ │ │ │ - var desc = this.createElementNS(this.namespaces.gpx, 'desc'); │ │ │ │ │ - desc.appendChild(this.createTextNode( │ │ │ │ │ - feature.attributes.description || this.defaultDesc)); │ │ │ │ │ - node.appendChild(desc); │ │ │ │ │ - // TBD - deal with remaining (non name/description) attributes. │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.GPX" │ │ │ │ │ -}); │ │ │ │ │ + o.unregister = function(id) { │ │ │ │ │ + delete o.registry[id]; │ │ │ │ │ + }; │ │ │ │ │ +})(); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/GeoRSS.js │ │ │ │ │ + OpenLayers/Protocol/WFS/v1.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ + * @requires OpenLayers/Protocol/WFS.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.GeoRSS │ │ │ │ │ - * Read/write GeoRSS parser. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Format.GeoRSS> constructor. │ │ │ │ │ + * Class: OpenLayers.Protocol.WFS.v1 │ │ │ │ │ + * Abstract class for for v1.0.0 and v1.1.0 protocol. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * - <OpenLayers.Protocol> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.GeoRSS = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Protocol.WFS.v1 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: rssns │ │ │ │ │ - * {String} RSS namespace to use. Defaults to │ │ │ │ │ - * "http://backend.userland.com/rss2" │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} WFS version number. │ │ │ │ │ */ │ │ │ │ │ - rssns: "http://backend.userland.com/rss2", │ │ │ │ │ + version: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: featurens │ │ │ │ │ - * {String} Feature Attributes namespace. Defaults to │ │ │ │ │ - * "http://mapserver.gis.umn.edu/mapserver" │ │ │ │ │ + * Property: srsName │ │ │ │ │ + * {String} Name of spatial reference system. Default is "EPSG:4326". │ │ │ │ │ */ │ │ │ │ │ - featureNS: "http://mapserver.gis.umn.edu/mapserver", │ │ │ │ │ + srsName: "EPSG:4326", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: georssns │ │ │ │ │ - * {String} GeoRSS namespace to use. Defaults to │ │ │ │ │ - * "http://www.georss.org/georss" │ │ │ │ │ + * Property: featureType │ │ │ │ │ + * {String} Local feature typeName. │ │ │ │ │ */ │ │ │ │ │ - georssns: "http://www.georss.org/georss", │ │ │ │ │ + featureType: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: geons │ │ │ │ │ - * {String} W3C Geo namespace to use. Defaults to │ │ │ │ │ - * "http://www.w3.org/2003/01/geo/wgs84_pos#" │ │ │ │ │ + * Property: featureNS │ │ │ │ │ + * {String} Feature namespace. │ │ │ │ │ */ │ │ │ │ │ - geons: "http://www.w3.org/2003/01/geo/wgs84_pos#", │ │ │ │ │ + featureNS: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: featureTitle │ │ │ │ │ - * {String} Default title for features. Defaults to "Untitled" │ │ │ │ │ + * Property: geometryName │ │ │ │ │ + * {String} Name of the geometry attribute for features. Default is │ │ │ │ │ + * "the_geom" for WFS <version> 1.0, and null for higher versions. │ │ │ │ │ */ │ │ │ │ │ - featureTitle: "Untitled", │ │ │ │ │ + geometryName: "the_geom", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: featureDescription │ │ │ │ │ - * {String} Default description for features. Defaults to "No Description" │ │ │ │ │ + * Property: maxFeatures │ │ │ │ │ + * {Integer} Optional maximum number of features to retrieve. │ │ │ │ │ */ │ │ │ │ │ - featureDescription: "No Description", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: gmlParse │ │ │ │ │ - * {Object} GML Format object for parsing features │ │ │ │ │ - * Non-API and only created if necessary │ │ │ │ │ + * Property: schema │ │ │ │ │ + * {String} Optional schema location that will be included in the │ │ │ │ │ + * schemaLocation attribute value. Note that the feature type schema │ │ │ │ │ + * is required for a strict XML validator (on transactions with an │ │ │ │ │ + * insert for example), but is *not* required by the WFS specification │ │ │ │ │ + * (since the server is supposed to know about feature type schemas). │ │ │ │ │ */ │ │ │ │ │ - gmlParser: null, │ │ │ │ │ + schema: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: xy │ │ │ │ │ - * {Boolean} Order of the GML coordinate: true:(x,y) or false:(y,x) │ │ │ │ │ - * For GeoRSS the default is (y,x), therefore: false │ │ │ │ │ + * Property: featurePrefix │ │ │ │ │ + * {String} Namespace alias for feature type. Default is "feature". │ │ │ │ │ */ │ │ │ │ │ - xy: false, │ │ │ │ │ + featurePrefix: "feature", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.GeoRSS │ │ │ │ │ - * Create a new parser for GeoRSS. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * Property: formatOptions │ │ │ │ │ + * {Object} Optional options for the format. If a format is not provided, │ │ │ │ │ + * this property can be used to extend the default format options. │ │ │ │ │ + */ │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readFormat │ │ │ │ │ + * {<OpenLayers.Format>} For WFS requests it is possible to get a │ │ │ │ │ + * different output format than GML. In that case, we cannot parse │ │ │ │ │ + * the response with the default format (WFST) and we need a different │ │ │ │ │ + * format for reading. │ │ │ │ │ */ │ │ │ │ │ + readFormat: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createGeometryFromItem │ │ │ │ │ - * Return a geometry from a GeoRSS Item. │ │ │ │ │ + * Property: readOptions │ │ │ │ │ + * {Object} Optional object to pass to format's read. │ │ │ │ │ + */ │ │ │ │ │ + readOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Protocol.WFS │ │ │ │ │ + * A class for giving layers WFS protocol. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * item - {DOMElement} A GeoRSS item node. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} A geometry representing the node. │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * url - {String} URL to send requests to (required). │ │ │ │ │ + * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ + * featureNS - {String} Feature namespace (required, but can be autodetected │ │ │ │ │ + * during the first query if GML is used as readFormat and │ │ │ │ │ + * featurePrefix is provided and matches the prefix used by the server │ │ │ │ │ + * for this featureType). │ │ │ │ │ + * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ + * for writing if featureNS is provided). Default is 'feature'. │ │ │ │ │ + * geometryName - {String} Name of geometry attribute. The default is │ │ │ │ │ + * 'the_geom' for WFS <version> 1.0, and null for higher versions. If │ │ │ │ │ + * null, it will be set to the name of the first geometry found in the │ │ │ │ │ + * first read operation. │ │ │ │ │ + * multi - {Boolean} If set to true, geometries will be casted to Multi │ │ │ │ │ + * geometries before they are written in a transaction. No casting will │ │ │ │ │ + * be done when reading features. │ │ │ │ │ */ │ │ │ │ │ - createGeometryFromItem: function(item) { │ │ │ │ │ - var point = this.getElementsByTagNameNS(item, this.georssns, "point"); │ │ │ │ │ - var lat = this.getElementsByTagNameNS(item, this.geons, 'lat'); │ │ │ │ │ - var lon = this.getElementsByTagNameNS(item, this.geons, 'long'); │ │ │ │ │ - │ │ │ │ │ - var line = this.getElementsByTagNameNS(item, │ │ │ │ │ - this.georssns, │ │ │ │ │ - "line"); │ │ │ │ │ - var polygon = this.getElementsByTagNameNS(item, │ │ │ │ │ - this.georssns, │ │ │ │ │ - "polygon"); │ │ │ │ │ - var where = this.getElementsByTagNameNS(item, │ │ │ │ │ - this.georssns, │ │ │ │ │ - "where"); │ │ │ │ │ - var box = this.getElementsByTagNameNS(item, │ │ │ │ │ - this.georssns, │ │ │ │ │ - "box"); │ │ │ │ │ - │ │ │ │ │ - if (point.length > 0 || (lat.length > 0 && lon.length > 0)) { │ │ │ │ │ - var location; │ │ │ │ │ - if (point.length > 0) { │ │ │ │ │ - location = OpenLayers.String.trim( │ │ │ │ │ - point[0].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ - if (location.length != 2) { │ │ │ │ │ - location = OpenLayers.String.trim( │ │ │ │ │ - point[0].firstChild.nodeValue).split(/\s*,\s*/); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - location = [parseFloat(lat[0].firstChild.nodeValue), │ │ │ │ │ - parseFloat(lon[0].firstChild.nodeValue) │ │ │ │ │ - ]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point(location[1], location[0]); │ │ │ │ │ - │ │ │ │ │ - } else if (line.length > 0) { │ │ │ │ │ - var coords = OpenLayers.String.trim(this.getChildValue(line[0])).split(/\s+/); │ │ │ │ │ - var components = []; │ │ │ │ │ - var point; │ │ │ │ │ - for (var i = 0, len = coords.length; i < len; i += 2) { │ │ │ │ │ - point = new OpenLayers.Geometry.Point(coords[i + 1], coords[i]); │ │ │ │ │ - components.push(point); │ │ │ │ │ - } │ │ │ │ │ - geometry = new OpenLayers.Geometry.LineString(components); │ │ │ │ │ - } else if (polygon.length > 0) { │ │ │ │ │ - var coords = OpenLayers.String.trim(this.getChildValue(polygon[0])).split(/\s+/); │ │ │ │ │ - var components = []; │ │ │ │ │ - var point; │ │ │ │ │ - for (var i = 0, len = coords.length; i < len; i += 2) { │ │ │ │ │ - point = new OpenLayers.Geometry.Point(coords[i + 1], coords[i]); │ │ │ │ │ - components.push(point); │ │ │ │ │ - } │ │ │ │ │ - geometry = new OpenLayers.Geometry.Polygon([new OpenLayers.Geometry.LinearRing(components)]); │ │ │ │ │ - } else if (where.length > 0) { │ │ │ │ │ - if (!this.gmlParser) { │ │ │ │ │ - this.gmlParser = new OpenLayers.Format.GML({ │ │ │ │ │ - 'xy': this.xy │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - var feature = this.gmlParser.parseFeature(where[0]); │ │ │ │ │ - geometry = feature.geometry; │ │ │ │ │ - } else if (box.length > 0) { │ │ │ │ │ - var coords = OpenLayers.String.trim(box[0].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ - var components = []; │ │ │ │ │ - var point; │ │ │ │ │ - if (coords.length > 3) { │ │ │ │ │ - point = new OpenLayers.Geometry.Point(coords[1], coords[0]); │ │ │ │ │ - components.push(point); │ │ │ │ │ - point = new OpenLayers.Geometry.Point(coords[1], coords[2]); │ │ │ │ │ - components.push(point); │ │ │ │ │ - point = new OpenLayers.Geometry.Point(coords[3], coords[2]); │ │ │ │ │ - components.push(point); │ │ │ │ │ - point = new OpenLayers.Geometry.Point(coords[3], coords[0]); │ │ │ │ │ - components.push(point); │ │ │ │ │ - point = new OpenLayers.Geometry.Point(coords[1], coords[0]); │ │ │ │ │ - components.push(point); │ │ │ │ │ - } │ │ │ │ │ - geometry = new OpenLayers.Geometry.Polygon([new OpenLayers.Geometry.LinearRing(components)]); │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!options.format) { │ │ │ │ │ + this.format = OpenLayers.Format.WFST(OpenLayers.Util.extend({ │ │ │ │ │ + version: this.version, │ │ │ │ │ + featureType: this.featureType, │ │ │ │ │ + featureNS: this.featureNS, │ │ │ │ │ + featurePrefix: this.featurePrefix, │ │ │ │ │ + geometryName: this.geometryName, │ │ │ │ │ + srsName: this.srsName, │ │ │ │ │ + schema: this.schema │ │ │ │ │ + }, this.formatOptions)); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - if (geometry && this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry.transform(this.externalProjection, │ │ │ │ │ - this.internalProjection); │ │ │ │ │ + if (!options.geometryName && parseFloat(this.format.version) > 1.0) { │ │ │ │ │ + this.setGeometryName(null); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return geometry; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up the protocol. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.options && !this.options.format) { │ │ │ │ │ + this.format.destroy(); │ │ │ │ │ + } │ │ │ │ │ + this.format = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createFeatureFromItem │ │ │ │ │ - * Return a feature from a GeoRSS Item. │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Construct a request for reading new features. Since WFS splits the │ │ │ │ │ + * basic CRUD operations into GetFeature requests (for read) and │ │ │ │ │ + * Transactions (for all others), this method does not make use of the │ │ │ │ │ + * format's read method (that is only about reading transaction │ │ │ │ │ + * responses). │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * item - {DOMElement} A GeoRSS item node. │ │ │ │ │ + * options - {Object} Options for the read operation, in addition to the │ │ │ │ │ + * options set on the instance (options set here will take precedence). │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A feature representing the item. │ │ │ │ │ + * To use a configured protocol to get e.g. a WFS hit count, applications │ │ │ │ │ + * could do the following: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * protocol.read({ │ │ │ │ │ + * readOptions: {output: "object"}, │ │ │ │ │ + * resultType: "hits", │ │ │ │ │ + * maxFeatures: null, │ │ │ │ │ + * callback: function(resp) { │ │ │ │ │ + * // process resp.numberOfFeatures here │ │ │ │ │ + * } │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * To use a configured protocol to use WFS paging (if supported by the │ │ │ │ │ + * server), applications could do the following: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * protocol.read({ │ │ │ │ │ + * startIndex: 0, │ │ │ │ │ + * count: 50 │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * To limit the attributes returned by the GetFeature request, applications │ │ │ │ │ + * can use the propertyNames option to specify the properties to include in │ │ │ │ │ + * the response: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * protocol.read({ │ │ │ │ │ + * propertyNames: ["DURATION", "INTENSITY"] │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ */ │ │ │ │ │ - createFeatureFromItem: function(item) { │ │ │ │ │ - var geometry = this.createGeometryFromItem(item); │ │ │ │ │ - │ │ │ │ │ - /* Provide defaults for title and description */ │ │ │ │ │ - var title = this._getChildValue(item, "*", "title", this.featureTitle); │ │ │ │ │ - │ │ │ │ │ - /* First try RSS descriptions, then Atom summaries */ │ │ │ │ │ - var description = this._getChildValue( │ │ │ │ │ - item, "*", "description", │ │ │ │ │ - this._getChildValue(item, "*", "content", │ │ │ │ │ - this._getChildValue(item, "*", "summary", this.featureDescription))); │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - /* If no link URL is found in the first child node, try the │ │ │ │ │ - href attribute */ │ │ │ │ │ - var link = this._getChildValue(item, "*", "link"); │ │ │ │ │ - if (!link) { │ │ │ │ │ - try { │ │ │ │ │ - link = this.getElementsByTagNameNS(item, "*", "link")[0].getAttribute("href"); │ │ │ │ │ - } catch (e) { │ │ │ │ │ - link = null; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + var data = OpenLayers.Format.XML.prototype.write.apply( │ │ │ │ │ + this.format, [this.format.writeNode("wfs:GetFeature", options)] │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - var id = this._getChildValue(item, "*", "id", null); │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: data │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - var data = { │ │ │ │ │ - "title": title, │ │ │ │ │ - "description": description, │ │ │ │ │ - "link": link │ │ │ │ │ - }; │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry, data); │ │ │ │ │ - feature.fid = id; │ │ │ │ │ - return feature; │ │ │ │ │ + return response; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: _getChildValue │ │ │ │ │ + * APIMethod: setFeatureType │ │ │ │ │ + * Change the feature type on the fly. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * nsuri - {String} Child node namespace uri ("*" for any). │ │ │ │ │ - * name - {String} Child node name. │ │ │ │ │ - * def - {String} Optional string default to return if no child found. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The value of the first child with the given tag name. Returns │ │ │ │ │ - * default value or empty string if none found. │ │ │ │ │ + * featureType - {String} Local (without prefix) feature typeName. │ │ │ │ │ */ │ │ │ │ │ - _getChildValue: function(node, nsuri, name, def) { │ │ │ │ │ - var value; │ │ │ │ │ - var eles = this.getElementsByTagNameNS(node, nsuri, name); │ │ │ │ │ - if (eles && eles[0] && eles[0].firstChild && │ │ │ │ │ - eles[0].firstChild.nodeValue) { │ │ │ │ │ - value = this.getChildValue(eles[0]); │ │ │ │ │ - } else { │ │ │ │ │ - value = (def == undefined) ? "" : def; │ │ │ │ │ - } │ │ │ │ │ - return value; │ │ │ │ │ + setFeatureType: function(featureType) { │ │ │ │ │ + this.featureType = featureType; │ │ │ │ │ + this.format.featureType = featureType; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Return a list of features from a GeoRSS doc │ │ │ │ │ + * APIMethod: setGeometryName │ │ │ │ │ + * Sets the geometryName option after instantiation. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * doc - {Element} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * geometryName - {String} Name of geometry attribute. │ │ │ │ │ */ │ │ │ │ │ - read: function(doc) { │ │ │ │ │ - if (typeof doc == "string") { │ │ │ │ │ - doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - /* Try RSS items first, then Atom entries */ │ │ │ │ │ - var itemlist = null; │ │ │ │ │ - itemlist = this.getElementsByTagNameNS(doc, '*', 'item'); │ │ │ │ │ - if (itemlist.length == 0) { │ │ │ │ │ - itemlist = this.getElementsByTagNameNS(doc, '*', 'entry'); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var numItems = itemlist.length; │ │ │ │ │ - var features = new Array(numItems); │ │ │ │ │ - for (var i = 0; i < numItems; i++) { │ │ │ │ │ - features[i] = this.createFeatureFromItem(itemlist[i]); │ │ │ │ │ - } │ │ │ │ │ - return features; │ │ │ │ │ + setGeometryName: function(geometryName) { │ │ │ │ │ + this.geometryName = geometryName; │ │ │ │ │ + this.format.geometryName = geometryName; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Accept Feature Collection, and return a string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} List of features to serialize into a string. │ │ │ │ │ + * Method: handleRead │ │ │ │ │ + * Deal with response from the read request. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ + * to the user callback. │ │ │ │ │ + * options - {Object} The user options passed to the read call. │ │ │ │ │ */ │ │ │ │ │ - write: function(features) { │ │ │ │ │ - var georss; │ │ │ │ │ - if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ - georss = this.createElementNS(this.rssns, "rss"); │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - georss.appendChild(this.createFeatureXML(features[i])); │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + // success │ │ │ │ │ + var result = this.parseResponse(request, options.readOptions); │ │ │ │ │ + if (result && result.success !== false) { │ │ │ │ │ + if (options.readOptions && options.readOptions.output == "object") { │ │ │ │ │ + OpenLayers.Util.extend(response, result); │ │ │ │ │ + } else { │ │ │ │ │ + response.features = result; │ │ │ │ │ + } │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + } else { │ │ │ │ │ + // failure (service exception) │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + response.error = result; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // failure │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - georss = this.createFeatureXML(features); │ │ │ │ │ + options.callback.call(options.scope, response); │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [georss]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createFeatureXML │ │ │ │ │ - * Accept an <OpenLayers.Feature.Vector>, and build a geometry for it. │ │ │ │ │ - * │ │ │ │ │ + * Method: parseResponse │ │ │ │ │ + * Read HTTP response body and return features │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * request - {XMLHttpRequest} The request object │ │ │ │ │ + * options - {Object} Optional object to pass to format's read │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * {Object} or {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * An object with a features property, an array of features or a single │ │ │ │ │ + * feature. │ │ │ │ │ */ │ │ │ │ │ - createFeatureXML: function(feature) { │ │ │ │ │ - var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ - var featureNode = this.createElementNS(this.rssns, "item"); │ │ │ │ │ - var titleNode = this.createElementNS(this.rssns, "title"); │ │ │ │ │ - titleNode.appendChild(this.createTextNode(feature.attributes.title ? feature.attributes.title : "")); │ │ │ │ │ - var descNode = this.createElementNS(this.rssns, "description"); │ │ │ │ │ - descNode.appendChild(this.createTextNode(feature.attributes.description ? feature.attributes.description : "")); │ │ │ │ │ - featureNode.appendChild(titleNode); │ │ │ │ │ - featureNode.appendChild(descNode); │ │ │ │ │ - if (feature.attributes.link) { │ │ │ │ │ - var linkNode = this.createElementNS(this.rssns, "link"); │ │ │ │ │ - linkNode.appendChild(this.createTextNode(feature.attributes.link)); │ │ │ │ │ - featureNode.appendChild(linkNode); │ │ │ │ │ + parseResponse: function(request, options) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ } │ │ │ │ │ - for (var attr in feature.attributes) { │ │ │ │ │ - if (attr == "link" || attr == "title" || attr == "description") { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - var attrText = this.createTextNode(feature.attributes[attr]); │ │ │ │ │ - var nodename = attr; │ │ │ │ │ - if (attr.search(":") != -1) { │ │ │ │ │ - nodename = attr.split(":")[1]; │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + var result = (this.readFormat !== null) ? this.readFormat.read(doc) : │ │ │ │ │ + this.format.read(doc, options); │ │ │ │ │ + if (!this.featureNS) { │ │ │ │ │ + var format = this.readFormat || this.format; │ │ │ │ │ + this.featureNS = format.featureNS; │ │ │ │ │ + // no need to auto-configure again on subsequent reads │ │ │ │ │ + format.autoConfig = false; │ │ │ │ │ + if (!this.geometryName) { │ │ │ │ │ + this.setGeometryName(format.geometryName); │ │ │ │ │ } │ │ │ │ │ - var attrContainer = this.createElementNS(this.featureNS, "feature:" + nodename); │ │ │ │ │ - attrContainer.appendChild(attrText); │ │ │ │ │ - featureNode.appendChild(attrContainer); │ │ │ │ │ } │ │ │ │ │ - featureNode.appendChild(geometryNode); │ │ │ │ │ - return featureNode; │ │ │ │ │ + return result; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildGeometryNode │ │ │ │ │ - * builds a GeoRSS node with a given geometry │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: commit │ │ │ │ │ + * Given a list of feature, assemble a batch request for update, create, │ │ │ │ │ + * and delete transactions. A commit call on the prototype amounts │ │ │ │ │ + * to writing a WFS transaction - so the write method on the format │ │ │ │ │ + * is used. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * options - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * nativeElements - {Array({Object})} Array of objects with information for writing │ │ │ │ │ + * out <Native> elements, these objects have vendorId, safeToIgnore and │ │ │ │ │ + * value properties. The <Native> element is intended to allow access to │ │ │ │ │ + * vendor specific capabilities of any particular web feature server or │ │ │ │ │ + * datastore. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} A gml node. │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} A response object with a features │ │ │ │ │ + * property containing any insertIds and a priv property referencing │ │ │ │ │ + * the XMLHttpRequest object. │ │ │ │ │ */ │ │ │ │ │ - buildGeometryNode: function(geometry) { │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry = geometry.clone(); │ │ │ │ │ - geometry.transform(this.internalProjection, │ │ │ │ │ - this.externalProjection); │ │ │ │ │ - } │ │ │ │ │ - var node; │ │ │ │ │ - // match Polygon │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.Polygon") { │ │ │ │ │ - node = this.createElementNS(this.georssns, 'georss:polygon'); │ │ │ │ │ + commit: function(features, options) { │ │ │ │ │ │ │ │ │ │ - node.appendChild(this.buildCoordinatesNode(geometry.components[0])); │ │ │ │ │ - } │ │ │ │ │ - // match LineString │ │ │ │ │ - else if (geometry.CLASS_NAME == "OpenLayers.Geometry.LineString") { │ │ │ │ │ - node = this.createElementNS(this.georssns, 'georss:line'); │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ │ │ │ │ │ - node.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - } │ │ │ │ │ - // match Point │ │ │ │ │ - else if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - node = this.createElementNS(this.georssns, 'georss:point'); │ │ │ │ │ - node.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ - } else { │ │ │ │ │ - throw "Couldn't parse " + geometry.CLASS_NAME; │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "commit", │ │ │ │ │ + reqFeatures: features │ │ │ │ │ + }); │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(features, options), │ │ │ │ │ + callback: this.createCallback(this.handleCommit, response, options) │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + return response; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildCoordinatesNode │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleCommit │ │ │ │ │ + * Called when the commit request returns. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ + * to the user callback. │ │ │ │ │ + * options - {Object} The user options passed to the commit call. │ │ │ │ │ */ │ │ │ │ │ - buildCoordinatesNode: function(geometry) { │ │ │ │ │ - var points = null; │ │ │ │ │ + handleCommit: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ │ │ │ │ │ - if (geometry.components) { │ │ │ │ │ - points = geometry.components; │ │ │ │ │ + // ensure that we have an xml doc │ │ │ │ │ + var data = request.responseXML; │ │ │ │ │ + if (!data || !data.documentElement) { │ │ │ │ │ + data = request.responseText; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var obj = this.format.read(data) || {}; │ │ │ │ │ + │ │ │ │ │ + response.insertIds = obj.insertIds || []; │ │ │ │ │ + if (obj.success) { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + response.error = obj; │ │ │ │ │ + } │ │ │ │ │ + options.callback.call(options.scope, response); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var path; │ │ │ │ │ - if (points) { │ │ │ │ │ - var numPoints = points.length; │ │ │ │ │ - var parts = new Array(numPoints); │ │ │ │ │ - for (var i = 0; i < numPoints; i++) { │ │ │ │ │ - parts[i] = points[i].y + " " + points[i].x; │ │ │ │ │ + /** │ │ │ │ │ + * Method: filterDelete │ │ │ │ │ + * Send a request that deletes all features by their filter. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {<OpenLayers.Filter>} filter │ │ │ │ │ + */ │ │ │ │ │ + filterDelete: function(filter, options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "commit" │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + var root = this.format.createElementNSPlus("wfs:Transaction", { │ │ │ │ │ + attributes: { │ │ │ │ │ + service: "WFS", │ │ │ │ │ + version: this.version │ │ │ │ │ } │ │ │ │ │ - path = parts.join(" "); │ │ │ │ │ - } else { │ │ │ │ │ - path = geometry.y + " " + geometry.x; │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + var deleteNode = this.format.createElementNSPlus("wfs:Delete", { │ │ │ │ │ + attributes: { │ │ │ │ │ + typeName: (options.featureNS ? this.featurePrefix + ":" : "") + │ │ │ │ │ + options.featureType │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + if (options.featureNS) { │ │ │ │ │ + deleteNode.setAttribute("xmlns:" + this.featurePrefix, options.featureNS); │ │ │ │ │ } │ │ │ │ │ - return this.createTextNode(path); │ │ │ │ │ + var filterNode = this.format.writeNode("ogc:Filter", filter); │ │ │ │ │ + │ │ │ │ │ + deleteNode.appendChild(filterNode); │ │ │ │ │ + │ │ │ │ │ + root.appendChild(deleteNode); │ │ │ │ │ + │ │ │ │ │ + var data = OpenLayers.Format.XML.prototype.write.apply( │ │ │ │ │ + this.format, [root] │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Request.POST({ │ │ │ │ │ + url: this.url, │ │ │ │ │ + callback: options.callback || function() {}, │ │ │ │ │ + data: data │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.GeoRSS" │ │ │ │ │ + /** │ │ │ │ │ + * Method: abort │ │ │ │ │ + * Abort an ongoing request, the response object passed to │ │ │ │ │ + * this method must come from this protocol (as a result │ │ │ │ │ + * of a read, or commit operation). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ + */ │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + response.priv.abort(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.WFS.v1" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSCapabilities.js │ │ │ │ │ + OpenLayers/Protocol/WFS/v1_1_0.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + * @requires OpenLayers/Protocol/WFS/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/WFST/v1_1_0.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WMSCapabilities │ │ │ │ │ - * Read WMS Capabilities. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Protocol.WFS.v1_1_0 │ │ │ │ │ + * A WFS v1.1.0 protocol for vector layers. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Protocol.WFS.v1_1_0> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Differences from the v1.0.0 protocol: │ │ │ │ │ + * - uses Filter Encoding 1.1.0 instead of 1.0.0 │ │ │ │ │ + * - uses GML 3 instead of 2 if no format is provided │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + * - <OpenLayers.Protocol.WFS.v1> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WMSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.1.1". │ │ │ │ │ - */ │ │ │ │ │ - defaultVersion: "1.1.1", │ │ │ │ │ +OpenLayers.Protocol.WFS.v1_1_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: profile │ │ │ │ │ - * {String} If provided, use a custom profile. │ │ │ │ │ - * │ │ │ │ │ - * Currently supported profiles: │ │ │ │ │ - * - WMSC - parses vendor specific capabilities for WMS-C. │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} WFS version number. │ │ │ │ │ */ │ │ │ │ │ - profile: null, │ │ │ │ │ + version: "1.1.0", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMSCapabilities │ │ │ │ │ - * Create a new parser for WMS capabilities. │ │ │ │ │ + * Constructor: OpenLayers.Protocol.WFS.v1_1_0 │ │ │ │ │ + * A class for giving layers WFS v1.1.0 protocol. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} List of named layers. │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ + * featureNS - {String} Feature namespace (optional). │ │ │ │ │ + * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ + * if featureNS is provided). Default is 'feature'. │ │ │ │ │ + * geometryName - {String} Name of geometry attribute. Default is 'the_geom'. │ │ │ │ │ + * outputFormat - {String} Optional output format to use for WFS GetFeature │ │ │ │ │ + * requests. This can be any format advertized by the WFS's │ │ │ │ │ + * GetCapabilities response. If set, an appropriate readFormat also │ │ │ │ │ + * has to be provided, unless outputFormat is GML3, GML2 or JSON. │ │ │ │ │ + * readFormat - {<OpenLayers.Format>} An appropriate format parser if │ │ │ │ │ + * outputFormat is none of GML3, GML2 or JSON. │ │ │ │ │ */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Protocol.WFS.v1.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (this.outputFormat && !this.readFormat) { │ │ │ │ │ + if (this.outputFormat.toLowerCase() == "gml2") { │ │ │ │ │ + this.readFormat = new OpenLayers.Format.GML.v2({ │ │ │ │ │ + featureType: this.featureType, │ │ │ │ │ + featureNS: this.featureNS, │ │ │ │ │ + geometryName: this.geometryName │ │ │ │ │ + }); │ │ │ │ │ + } else if (this.outputFormat.toLowerCase() == "json") { │ │ │ │ │ + this.readFormat = new OpenLayers.Format.GeoJSON(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities" │ │ │ │ │ - │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.WFS.v1_1_0" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMTSCapabilities.js │ │ │ │ │ + OpenLayers/Format/GML/v2.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + * @requires OpenLayers/Format/GML/Base.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WMTSCapabilities │ │ │ │ │ - * Read WMTS Capabilities. │ │ │ │ │ + * Class: OpenLayers.Format.GML.v2 │ │ │ │ │ + * Parses GML version 2. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + * - <OpenLayers.Format.GML.Base> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WMTSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.0.0". │ │ │ │ │ - */ │ │ │ │ │ - defaultVersion: "1.0.0", │ │ │ │ │ +OpenLayers.Format.GML.v2 = OpenLayers.Class(OpenLayers.Format.GML.Base, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: yx │ │ │ │ │ - * {Object} Members in the yx object are used to determine if a CRS URN │ │ │ │ │ - * corresponds to a CRS with y,x axis order. Member names are CRS URNs │ │ │ │ │ - * and values are boolean. By default, the following CRS URN are │ │ │ │ │ - * assumed to correspond to a CRS with y,x axis order: │ │ │ │ │ - * │ │ │ │ │ - * * urn:ogc:def:crs:EPSG::4326 │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} Schema location for a particular minor version. │ │ │ │ │ */ │ │ │ │ │ - yx: { │ │ │ │ │ - "urn:ogc:def:crs:EPSG::4326": true │ │ │ │ │ - }, │ │ │ │ │ + schemaLocation: "http://www.opengis.net/gml http://schemas.opengis.net/gml/2.1.2/feature.xsd", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMTSCapabilities │ │ │ │ │ - * Create a new parser for WMTS capabilities. │ │ │ │ │ + * Constructor: OpenLayers.Format.GML.v2 │ │ │ │ │ + * Create a parser for GML v2. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * options - {Object} An optional object whose properties will be set on │ │ │ │ │ * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return information about │ │ │ │ │ - * the service (offering and observedProperty mostly). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Info about the WMTS Capabilities │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ + * featureNS - {String} Feature namespace (required). │ │ │ │ │ + * geometryName - {String} Geometry element name. │ │ │ │ │ */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: createLayer │ │ │ │ │ - * Create a WMTS layer given a capabilities object. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * capabilities - {Object} The object returned from a <read> call to this │ │ │ │ │ - * format. │ │ │ │ │ - * config - {Object} Configuration properties for the layer. Defaults for │ │ │ │ │ - * the layer will apply if not provided. │ │ │ │ │ - * │ │ │ │ │ - * Required config properties: │ │ │ │ │ - * layer - {String} The layer identifier. │ │ │ │ │ - * │ │ │ │ │ - * Optional config properties: │ │ │ │ │ - * matrixSet - {String} The matrix set identifier, required if there is │ │ │ │ │ - * more than one matrix set in the layer capabilities. │ │ │ │ │ - * style - {String} The name of the style │ │ │ │ │ - * format - {String} Image format for the layer. Default is the first │ │ │ │ │ - * format returned in the GetCapabilities response. │ │ │ │ │ - * param - {Object} The dimensions values eg: {"Year": "2012"} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.WMTS>} A properly configured WMTS layer. Throws an │ │ │ │ │ - * error if an incomplete config is provided. Returns undefined if no │ │ │ │ │ - * layer could be created with the provided config. │ │ │ │ │ - */ │ │ │ │ │ - createLayer: function(capabilities, config) { │ │ │ │ │ - var layer; │ │ │ │ │ - │ │ │ │ │ - // confirm required properties are supplied in config │ │ │ │ │ - if (!('layer' in config)) { │ │ │ │ │ - throw new Error("Missing property 'layer' in configuration."); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var contents = capabilities.contents; │ │ │ │ │ - │ │ │ │ │ - // find the layer definition with the given identifier │ │ │ │ │ - var layers = contents.layers; │ │ │ │ │ - var layerDef; │ │ │ │ │ - for (var i = 0, ii = contents.layers.length; i < ii; ++i) { │ │ │ │ │ - if (contents.layers[i].identifier === config.layer) { │ │ │ │ │ - layerDef = contents.layers[i]; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!layerDef) { │ │ │ │ │ - throw new Error("Layer not found"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var format = config.format; │ │ │ │ │ - if (!format && layerDef.formats && layerDef.formats.length) { │ │ │ │ │ - format = layerDef.formats[0]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // find the matrixSet definition │ │ │ │ │ - var matrixSet; │ │ │ │ │ - if (config.matrixSet) { │ │ │ │ │ - matrixSet = contents.tileMatrixSets[config.matrixSet]; │ │ │ │ │ - } else if (layerDef.tileMatrixSetLinks.length >= 1) { │ │ │ │ │ - matrixSet = contents.tileMatrixSets[ │ │ │ │ │ - layerDef.tileMatrixSetLinks[0].tileMatrixSet]; │ │ │ │ │ - } │ │ │ │ │ - if (!matrixSet) { │ │ │ │ │ - throw new Error("matrixSet not found"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // get the default style for the layer │ │ │ │ │ - var style; │ │ │ │ │ - for (var i = 0, ii = layerDef.styles.length; i < ii; ++i) { │ │ │ │ │ - style = layerDef.styles[i]; │ │ │ │ │ - if (style.isDefault) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var requestEncoding = config.requestEncoding; │ │ │ │ │ - if (!requestEncoding) { │ │ │ │ │ - requestEncoding = "KVP"; │ │ │ │ │ - if (capabilities.operationsMetadata.GetTile.dcp.http) { │ │ │ │ │ - var http = capabilities.operationsMetadata.GetTile.dcp.http; │ │ │ │ │ - // Get first get method │ │ │ │ │ - if (http.get[0].constraints) { │ │ │ │ │ - var constraints = http.get[0].constraints; │ │ │ │ │ - var allowedValues = constraints.GetEncoding.allowedValues; │ │ │ │ │ - │ │ │ │ │ - // The OGC documentation is not clear if we should use │ │ │ │ │ - // REST or RESTful, ArcGis use RESTful, │ │ │ │ │ - // and OpenLayers use REST. │ │ │ │ │ - if (!allowedValues.KVP && │ │ │ │ │ - (allowedValues.REST || allowedValues.RESTful)) { │ │ │ │ │ - requestEncoding = "REST"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var dimensions = []; │ │ │ │ │ - var params = config.params || {}; │ │ │ │ │ - // to don't overwrite the changes in the applyDefaults │ │ │ │ │ - delete config.params; │ │ │ │ │ - for (var id = 0, ld = layerDef.dimensions.length; id < ld; id++) { │ │ │ │ │ - var dimension = layerDef.dimensions[id]; │ │ │ │ │ - dimensions.push(dimension.identifier); │ │ │ │ │ - if (!params.hasOwnProperty(dimension.identifier)) { │ │ │ │ │ - params[dimension.identifier] = dimension['default']; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var projection = config.projection || matrixSet.supportedCRS.replace( │ │ │ │ │ - /urn:ogc:def:crs:(\w+):(.*:)?(\w+)$/, "$1:$3"); │ │ │ │ │ - var units = config.units || │ │ │ │ │ - (projection === "EPSG:4326" ? "degrees" : "m"); │ │ │ │ │ - │ │ │ │ │ - var resolutions = []; │ │ │ │ │ - for (var mid in matrixSet.matrixIds) { │ │ │ │ │ - if (matrixSet.matrixIds.hasOwnProperty(mid)) { │ │ │ │ │ - resolutions.push( │ │ │ │ │ - matrixSet.matrixIds[mid].scaleDenominator * 0.28E-3 / │ │ │ │ │ - OpenLayers.METERS_PER_INCH / │ │ │ │ │ - OpenLayers.INCHES_PER_UNIT[units]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var url; │ │ │ │ │ - if (requestEncoding === "REST" && layerDef.resourceUrls) { │ │ │ │ │ - url = []; │ │ │ │ │ - var resourceUrls = layerDef.resourceUrls, │ │ │ │ │ - resourceUrl; │ │ │ │ │ - for (var t = 0, tt = layerDef.resourceUrls.length; t < tt; ++t) { │ │ │ │ │ - resourceUrl = layerDef.resourceUrls[t]; │ │ │ │ │ - if (resourceUrl.format === format && resourceUrl.resourceType === "tile") { │ │ │ │ │ - url.push(resourceUrl.template); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - var httpGet = capabilities.operationsMetadata.GetTile.dcp.http.get; │ │ │ │ │ - url = []; │ │ │ │ │ - var constraint; │ │ │ │ │ - for (var i = 0, ii = httpGet.length; i < ii; i++) { │ │ │ │ │ - constraint = httpGet[i].constraints; │ │ │ │ │ - if (!constraint || (constraint && constraint.GetEncoding.allowedValues[requestEncoding])) { │ │ │ │ │ - url.push(httpGet[i].url); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return new OpenLayers.Layer.WMTS( │ │ │ │ │ - OpenLayers.Util.applyDefaults(config, { │ │ │ │ │ - url: url, │ │ │ │ │ - requestEncoding: requestEncoding, │ │ │ │ │ - name: layerDef.title, │ │ │ │ │ - style: style.identifier, │ │ │ │ │ - format: format, │ │ │ │ │ - matrixIds: matrixSet.matrixIds, │ │ │ │ │ - matrixSet: matrixSet.identifier, │ │ │ │ │ - projection: projection, │ │ │ │ │ - units: units, │ │ │ │ │ - resolutions: config.isBaseLayer === false ? undefined : resolutions, │ │ │ │ │ - serverResolutions: resolutions, │ │ │ │ │ - tileFullExtent: matrixSet.bounds, │ │ │ │ │ - dimensions: dimensions, │ │ │ │ │ - params: params │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMTSCapabilities" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/CSWGetRecords.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.CSWGetRecords │ │ │ │ │ - * Default version is 2.0.2. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Format>} A CSWGetRecords format of the given version. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.CSWGetRecords = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options, OpenLayers.Format.CSWGetRecords.DEFAULTS │ │ │ │ │ - ); │ │ │ │ │ - var cls = OpenLayers.Format.CSWGetRecords["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported CSWGetRecords version: " + options.version; │ │ │ │ │ - } │ │ │ │ │ - return new cls(options); │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: DEFAULTS │ │ │ │ │ - * {Object} Default properties for the CSWGetRecords format. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.CSWGetRecords.DEFAULTS = { │ │ │ │ │ - "version": "2.0.2" │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/CQL.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WKT.js │ │ │ │ │ - * @requires OpenLayers/Filter/Comparison.js │ │ │ │ │ - * @requires OpenLayers/Filter/Logical.js │ │ │ │ │ - * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.CQL │ │ │ │ │ - * Read CQL strings to get <OpenLayers.Filter> objects. Write │ │ │ │ │ - * <OpenLayers.Filter> objects to get CQL strings. Create a new parser with │ │ │ │ │ - * the <OpenLayers.Format.CQL> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.CQL = (function() { │ │ │ │ │ - │ │ │ │ │ - var tokens = [ │ │ │ │ │ - "PROPERTY", "COMPARISON", "VALUE", "LOGICAL" │ │ │ │ │ - ], │ │ │ │ │ - │ │ │ │ │ - patterns = { │ │ │ │ │ - PROPERTY: /^[_a-zA-Z]\w*/, │ │ │ │ │ - COMPARISON: /^(=|<>|<=|<|>=|>|LIKE)/i, │ │ │ │ │ - IS_NULL: /^IS NULL/i, │ │ │ │ │ - COMMA: /^,/, │ │ │ │ │ - LOGICAL: /^(AND|OR)/i, │ │ │ │ │ - VALUE: /^('([^']|'')*'|\d+(\.\d*)?|\.\d+)/, │ │ │ │ │ - LPAREN: /^\(/, │ │ │ │ │ - RPAREN: /^\)/, │ │ │ │ │ - SPATIAL: /^(BBOX|INTERSECTS|DWITHIN|WITHIN|CONTAINS)/i, │ │ │ │ │ - NOT: /^NOT/i, │ │ │ │ │ - BETWEEN: /^BETWEEN/i, │ │ │ │ │ - GEOMETRY: function(text) { │ │ │ │ │ - var type = /^(POINT|LINESTRING|POLYGON|MULTIPOINT|MULTILINESTRING|MULTIPOLYGON|GEOMETRYCOLLECTION)/.exec(text); │ │ │ │ │ - if (type) { │ │ │ │ │ - var len = text.length; │ │ │ │ │ - var idx = text.indexOf("(", type[0].length); │ │ │ │ │ - if (idx > -1) { │ │ │ │ │ - var depth = 1; │ │ │ │ │ - while (idx < len && depth > 0) { │ │ │ │ │ - idx++; │ │ │ │ │ - switch (text.charAt(idx)) { │ │ │ │ │ - case '(': │ │ │ │ │ - depth++; │ │ │ │ │ - break; │ │ │ │ │ - case ')': │ │ │ │ │ - depth--; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - // in default case, do nothing │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return [text.substr(0, idx + 1)]; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - END: /^$/ │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - follows = { │ │ │ │ │ - LPAREN: ['GEOMETRY', 'SPATIAL', 'PROPERTY', 'VALUE', 'LPAREN'], │ │ │ │ │ - RPAREN: ['NOT', 'LOGICAL', 'END', 'RPAREN'], │ │ │ │ │ - PROPERTY: ['COMPARISON', 'BETWEEN', 'COMMA', 'IS_NULL'], │ │ │ │ │ - BETWEEN: ['VALUE'], │ │ │ │ │ - IS_NULL: ['END'], │ │ │ │ │ - COMPARISON: ['VALUE'], │ │ │ │ │ - COMMA: ['GEOMETRY', 'VALUE', 'PROPERTY'], │ │ │ │ │ - VALUE: ['LOGICAL', 'COMMA', 'RPAREN', 'END'], │ │ │ │ │ - SPATIAL: ['LPAREN'], │ │ │ │ │ - LOGICAL: ['NOT', 'VALUE', 'SPATIAL', 'PROPERTY', 'LPAREN'], │ │ │ │ │ - NOT: ['PROPERTY', 'LPAREN'], │ │ │ │ │ - GEOMETRY: ['COMMA', 'RPAREN'] │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - operators = { │ │ │ │ │ - '=': OpenLayers.Filter.Comparison.EQUAL_TO, │ │ │ │ │ - '<>': OpenLayers.Filter.Comparison.NOT_EQUAL_TO, │ │ │ │ │ - '<': OpenLayers.Filter.Comparison.LESS_THAN, │ │ │ │ │ - '<=': OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO, │ │ │ │ │ - '>': OpenLayers.Filter.Comparison.GREATER_THAN, │ │ │ │ │ - '>=': OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO, │ │ │ │ │ - 'LIKE': OpenLayers.Filter.Comparison.LIKE, │ │ │ │ │ - 'BETWEEN': OpenLayers.Filter.Comparison.BETWEEN, │ │ │ │ │ - 'IS NULL': OpenLayers.Filter.Comparison.IS_NULL │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - operatorReverse = {}, │ │ │ │ │ - │ │ │ │ │ - logicals = { │ │ │ │ │ - 'AND': OpenLayers.Filter.Logical.AND, │ │ │ │ │ - 'OR': OpenLayers.Filter.Logical.OR │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - logicalReverse = {}, │ │ │ │ │ - │ │ │ │ │ - precedence = { │ │ │ │ │ - 'RPAREN': 3, │ │ │ │ │ - 'LOGICAL': 2, │ │ │ │ │ - 'COMPARISON': 1 │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - var i; │ │ │ │ │ - for (i in operators) { │ │ │ │ │ - if (operators.hasOwnProperty(i)) { │ │ │ │ │ - operatorReverse[operators[i]] = i; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - for (i in logicals) { │ │ │ │ │ - if (logicals.hasOwnProperty(i)) { │ │ │ │ │ - logicalReverse[logicals[i]] = i; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function tryToken(text, pattern) { │ │ │ │ │ - if (pattern instanceof RegExp) { │ │ │ │ │ - return pattern.exec(text); │ │ │ │ │ - } else { │ │ │ │ │ - return pattern(text); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function nextToken(text, tokens) { │ │ │ │ │ - var i, token, len = tokens.length; │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - token = tokens[i]; │ │ │ │ │ - var pat = patterns[token]; │ │ │ │ │ - var matches = tryToken(text, pat); │ │ │ │ │ - if (matches) { │ │ │ │ │ - var match = matches[0]; │ │ │ │ │ - var remainder = text.substr(match.length).replace(/^\s*/, ""); │ │ │ │ │ - return { │ │ │ │ │ - type: token, │ │ │ │ │ - text: match, │ │ │ │ │ - remainder: remainder │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var msg = "ERROR: In parsing: [" + text + "], expected one of: "; │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - token = tokens[i]; │ │ │ │ │ - msg += "\n " + token + ": " + patterns[token]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - throw new Error(msg); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function tokenize(text) { │ │ │ │ │ - var results = []; │ │ │ │ │ - var token, expect = ["NOT", "GEOMETRY", "SPATIAL", "PROPERTY", "LPAREN"]; │ │ │ │ │ - │ │ │ │ │ - do { │ │ │ │ │ - token = nextToken(text, expect); │ │ │ │ │ - text = token.remainder; │ │ │ │ │ - expect = follows[token.type]; │ │ │ │ │ - if (token.type != "END" && !expect) { │ │ │ │ │ - throw new Error("No follows list for " + token.type); │ │ │ │ │ - } │ │ │ │ │ - results.push(token); │ │ │ │ │ - } while (token.type != "END"); │ │ │ │ │ - │ │ │ │ │ - return results; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function buildAst(tokens) { │ │ │ │ │ - var operatorStack = [], │ │ │ │ │ - postfix = []; │ │ │ │ │ - │ │ │ │ │ - while (tokens.length) { │ │ │ │ │ - var tok = tokens.shift(); │ │ │ │ │ - switch (tok.type) { │ │ │ │ │ - case "PROPERTY": │ │ │ │ │ - case "GEOMETRY": │ │ │ │ │ - case "VALUE": │ │ │ │ │ - postfix.push(tok); │ │ │ │ │ - break; │ │ │ │ │ - case "COMPARISON": │ │ │ │ │ - case "BETWEEN": │ │ │ │ │ - case "IS_NULL": │ │ │ │ │ - case "LOGICAL": │ │ │ │ │ - var p = precedence[tok.type]; │ │ │ │ │ - │ │ │ │ │ - while (operatorStack.length > 0 && │ │ │ │ │ - (precedence[operatorStack[operatorStack.length - 1].type] <= p) │ │ │ │ │ - ) { │ │ │ │ │ - postfix.push(operatorStack.pop()); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - operatorStack.push(tok); │ │ │ │ │ - break; │ │ │ │ │ - case "SPATIAL": │ │ │ │ │ - case "NOT": │ │ │ │ │ - case "LPAREN": │ │ │ │ │ - operatorStack.push(tok); │ │ │ │ │ - break; │ │ │ │ │ - case "RPAREN": │ │ │ │ │ - while (operatorStack.length > 0 && │ │ │ │ │ - (operatorStack[operatorStack.length - 1].type != "LPAREN") │ │ │ │ │ - ) { │ │ │ │ │ - postfix.push(operatorStack.pop()); │ │ │ │ │ - } │ │ │ │ │ - operatorStack.pop(); // toss out the LPAREN │ │ │ │ │ - │ │ │ │ │ - if (operatorStack.length > 0 && │ │ │ │ │ - operatorStack[operatorStack.length - 1].type == "SPATIAL") { │ │ │ │ │ - postfix.push(operatorStack.pop()); │ │ │ │ │ - } │ │ │ │ │ - case "COMMA": │ │ │ │ │ - case "END": │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - throw new Error("Unknown token type " + tok.type); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - while (operatorStack.length > 0) { │ │ │ │ │ - postfix.push(operatorStack.pop()); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function buildTree() { │ │ │ │ │ - var tok = postfix.pop(); │ │ │ │ │ - switch (tok.type) { │ │ │ │ │ - case "LOGICAL": │ │ │ │ │ - var rhs = buildTree(), │ │ │ │ │ - lhs = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Logical({ │ │ │ │ │ - filters: [lhs, rhs], │ │ │ │ │ - type: logicals[tok.text.toUpperCase()] │ │ │ │ │ - }); │ │ │ │ │ - case "NOT": │ │ │ │ │ - var operand = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Logical({ │ │ │ │ │ - filters: [operand], │ │ │ │ │ - type: OpenLayers.Filter.Logical.NOT │ │ │ │ │ - }); │ │ │ │ │ - case "BETWEEN": │ │ │ │ │ - var min, max, property; │ │ │ │ │ - postfix.pop(); // unneeded AND token here │ │ │ │ │ - max = buildTree(); │ │ │ │ │ - min = buildTree(); │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Comparison({ │ │ │ │ │ - property: property, │ │ │ │ │ - lowerBoundary: min, │ │ │ │ │ - upperBoundary: max, │ │ │ │ │ - type: OpenLayers.Filter.Comparison.BETWEEN │ │ │ │ │ - }); │ │ │ │ │ - case "COMPARISON": │ │ │ │ │ - var value = buildTree(), │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Comparison({ │ │ │ │ │ - property: property, │ │ │ │ │ - value: value, │ │ │ │ │ - type: operators[tok.text.toUpperCase()] │ │ │ │ │ - }); │ │ │ │ │ - case "IS_NULL": │ │ │ │ │ - var property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Comparison({ │ │ │ │ │ - property: property, │ │ │ │ │ - type: operators[tok.text.toUpperCase()] │ │ │ │ │ - }); │ │ │ │ │ - case "VALUE": │ │ │ │ │ - var match = tok.text.match(/^'(.*)'$/); │ │ │ │ │ - if (match) { │ │ │ │ │ - return match[1].replace(/''/g, "'"); │ │ │ │ │ - } else { │ │ │ │ │ - return Number(tok.text); │ │ │ │ │ - } │ │ │ │ │ - case "SPATIAL": │ │ │ │ │ - switch (tok.text.toUpperCase()) { │ │ │ │ │ - case "BBOX": │ │ │ │ │ - var maxy = buildTree(), │ │ │ │ │ - maxx = buildTree(), │ │ │ │ │ - miny = buildTree(), │ │ │ │ │ - minx = buildTree(), │ │ │ │ │ - prop = buildTree(); │ │ │ │ │ - │ │ │ │ │ - return new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ - property: prop, │ │ │ │ │ - value: OpenLayers.Bounds.fromArray( │ │ │ │ │ - [minx, miny, maxx, maxy] │ │ │ │ │ - ) │ │ │ │ │ - }); │ │ │ │ │ - case "INTERSECTS": │ │ │ │ │ - var value = buildTree(), │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: property, │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - case "WITHIN": │ │ │ │ │ - var value = buildTree(), │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.WITHIN, │ │ │ │ │ - property: property, │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - case "CONTAINS": │ │ │ │ │ - var value = buildTree(), │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.CONTAINS, │ │ │ │ │ - property: property, │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - case "DWITHIN": │ │ │ │ │ - var distance = buildTree(), │ │ │ │ │ - value = buildTree(), │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ - value: value, │ │ │ │ │ - property: property, │ │ │ │ │ - distance: Number(distance) │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - case "GEOMETRY": │ │ │ │ │ - return OpenLayers.Geometry.fromWKT(tok.text); │ │ │ │ │ - default: │ │ │ │ │ - return tok.text; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var result = buildTree(); │ │ │ │ │ - if (postfix.length > 0) { │ │ │ │ │ - var msg = "Remaining tokens after building AST: \n"; │ │ │ │ │ - for (var i = postfix.length - 1; i >= 0; i--) { │ │ │ │ │ - msg += postfix[i].type + ": " + postfix[i].text + "\n"; │ │ │ │ │ - } │ │ │ │ │ - throw new Error(msg); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return result; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Generate a filter from a CQL string. │ │ │ │ │ - │ │ │ │ │ - * Parameters: │ │ │ │ │ - * text - {String} The CQL text. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Filter>} A filter based on the CQL text. │ │ │ │ │ - */ │ │ │ │ │ - read: function(text) { │ │ │ │ │ - var result = buildAst(tokenize(text)); │ │ │ │ │ - if (this.keepData) { │ │ │ │ │ - this.data = result; │ │ │ │ │ - } │ │ │ │ │ - return result; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Convert a filter into a CQL string. │ │ │ │ │ - │ │ │ │ │ - * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} The filter. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A CQL string based on the filter. │ │ │ │ │ - */ │ │ │ │ │ - write: function(filter) { │ │ │ │ │ - if (filter instanceof OpenLayers.Geometry) { │ │ │ │ │ - return filter.toString(); │ │ │ │ │ - } │ │ │ │ │ - switch (filter.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Filter.Spatial": │ │ │ │ │ - switch (filter.type) { │ │ │ │ │ - case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ - return "BBOX(" + │ │ │ │ │ - filter.property + "," + │ │ │ │ │ - filter.value.toBBOX() + │ │ │ │ │ - ")"; │ │ │ │ │ - case OpenLayers.Filter.Spatial.DWITHIN: │ │ │ │ │ - return "DWITHIN(" + │ │ │ │ │ - filter.property + ", " + │ │ │ │ │ - this.write(filter.value) + ", " + │ │ │ │ │ - filter.distance + ")"; │ │ │ │ │ - case OpenLayers.Filter.Spatial.WITHIN: │ │ │ │ │ - return "WITHIN(" + │ │ │ │ │ - filter.property + ", " + │ │ │ │ │ - this.write(filter.value) + ")"; │ │ │ │ │ - case OpenLayers.Filter.Spatial.INTERSECTS: │ │ │ │ │ - return "INTERSECTS(" + │ │ │ │ │ - filter.property + ", " + │ │ │ │ │ - this.write(filter.value) + ")"; │ │ │ │ │ - case OpenLayers.Filter.Spatial.CONTAINS: │ │ │ │ │ - return "CONTAINS(" + │ │ │ │ │ - filter.property + ", " + │ │ │ │ │ - this.write(filter.value) + ")"; │ │ │ │ │ - default: │ │ │ │ │ - throw new Error("Unknown spatial filter type: " + filter.type); │ │ │ │ │ - } │ │ │ │ │ - case "OpenLayers.Filter.Logical": │ │ │ │ │ - if (filter.type == OpenLayers.Filter.Logical.NOT) { │ │ │ │ │ - // TODO: deal with precedence of logical operators to │ │ │ │ │ - // avoid extra parentheses (not urgent) │ │ │ │ │ - return "NOT (" + this.write(filter.filters[0]) + ")"; │ │ │ │ │ - } else { │ │ │ │ │ - var res = "("; │ │ │ │ │ - var first = true; │ │ │ │ │ - for (var i = 0; i < filter.filters.length; i++) { │ │ │ │ │ - if (first) { │ │ │ │ │ - first = false; │ │ │ │ │ - } else { │ │ │ │ │ - res += ") " + logicalReverse[filter.type] + " ("; │ │ │ │ │ - } │ │ │ │ │ - res += this.write(filter.filters[i]); │ │ │ │ │ - } │ │ │ │ │ - return res + ")"; │ │ │ │ │ - } │ │ │ │ │ - case "OpenLayers.Filter.Comparison": │ │ │ │ │ - if (filter.type == OpenLayers.Filter.Comparison.BETWEEN) { │ │ │ │ │ - return filter.property + " BETWEEN " + │ │ │ │ │ - this.write(filter.lowerBoundary) + " AND " + │ │ │ │ │ - this.write(filter.upperBoundary); │ │ │ │ │ - } else { │ │ │ │ │ - return (filter.value !== null) ? filter.property + │ │ │ │ │ - " " + operatorReverse[filter.type] + " " + │ │ │ │ │ - this.write(filter.value) : filter.property + │ │ │ │ │ - " " + operatorReverse[filter.type]; │ │ │ │ │ - } │ │ │ │ │ - case undefined: │ │ │ │ │ - if (typeof filter === "string") { │ │ │ │ │ - return "'" + filter.replace(/'/g, "''") + "'"; │ │ │ │ │ - } else if (typeof filter === "number") { │ │ │ │ │ - return String(filter); │ │ │ │ │ - } │ │ │ │ │ - default: │ │ │ │ │ - throw new Error("Can't encode: " + filter.CLASS_NAME + " " + filter); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.CQL" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -})(); │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WFSDescribeFeatureType.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/OGCExceptionReport.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WFSDescribeFeatureType │ │ │ │ │ - * Read WFS DescribeFeatureType response │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WFSDescribeFeatureType = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.XML, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ - */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g) │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - xsd: "http://www.w3.org/2001/XMLSchema" │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WFSDescribeFeatureType │ │ │ │ │ - * Create a new parser for WFS DescribeFeatureType responses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "xsd": { │ │ │ │ │ - "schema": function(node, obj) { │ │ │ │ │ - var complexTypes = []; │ │ │ │ │ - var customTypes = {}; │ │ │ │ │ - var schema = { │ │ │ │ │ - complexTypes: complexTypes, │ │ │ │ │ - customTypes: customTypes │ │ │ │ │ - }; │ │ │ │ │ - var i, len; │ │ │ │ │ - │ │ │ │ │ - this.readChildNodes(node, schema); │ │ │ │ │ - │ │ │ │ │ - var attributes = node.attributes; │ │ │ │ │ - var attr, name; │ │ │ │ │ - for (i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ - attr = attributes[i]; │ │ │ │ │ - name = attr.name; │ │ │ │ │ - if (name.indexOf("xmlns") === 0) { │ │ │ │ │ - this.setNamespace(name.split(":")[1] || "", attr.value); │ │ │ │ │ - } else { │ │ │ │ │ - obj[name] = attr.value; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - obj.featureTypes = complexTypes; │ │ │ │ │ - obj.targetPrefix = this.namespaceAlias[obj.targetNamespace]; │ │ │ │ │ - │ │ │ │ │ - // map complexTypes to names of customTypes │ │ │ │ │ - var complexType, customType; │ │ │ │ │ - for (i = 0, len = complexTypes.length; i < len; ++i) { │ │ │ │ │ - complexType = complexTypes[i]; │ │ │ │ │ - customType = customTypes[complexType.typeName]; │ │ │ │ │ - if (customTypes[complexType.typeName]) { │ │ │ │ │ - complexType.typeName = customType.name; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "complexType": function(node, obj) { │ │ │ │ │ - var complexType = { │ │ │ │ │ - // this is a temporary typeName, it will be overwritten by │ │ │ │ │ - // the schema reader with the metadata found in the │ │ │ │ │ - // customTypes hash │ │ │ │ │ - "typeName": node.getAttribute("name") │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, complexType); │ │ │ │ │ - obj.complexTypes.push(complexType); │ │ │ │ │ - }, │ │ │ │ │ - "complexContent": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "extension": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "sequence": function(node, obj) { │ │ │ │ │ - var sequence = { │ │ │ │ │ - elements: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, sequence); │ │ │ │ │ - obj.properties = sequence.elements; │ │ │ │ │ - }, │ │ │ │ │ - "element": function(node, obj) { │ │ │ │ │ - var type; │ │ │ │ │ - if (obj.elements) { │ │ │ │ │ - var element = {}; │ │ │ │ │ - var attributes = node.attributes; │ │ │ │ │ - var attr; │ │ │ │ │ - for (var i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ - attr = attributes[i]; │ │ │ │ │ - element[attr.name] = attr.value; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - type = element.type; │ │ │ │ │ - if (!type) { │ │ │ │ │ - type = {}; │ │ │ │ │ - this.readChildNodes(node, type); │ │ │ │ │ - element.restriction = type; │ │ │ │ │ - element.type = type.base; │ │ │ │ │ - } │ │ │ │ │ - var fullType = type.base || type; │ │ │ │ │ - element.localType = fullType.split(":").pop(); │ │ │ │ │ - obj.elements.push(element); │ │ │ │ │ - this.readChildNodes(node, element); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (obj.complexTypes) { │ │ │ │ │ - type = node.getAttribute("type"); │ │ │ │ │ - var localType = type.split(":").pop(); │ │ │ │ │ - obj.customTypes[localType] = { │ │ │ │ │ - "name": node.getAttribute("name"), │ │ │ │ │ - "type": type │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "annotation": function(node, obj) { │ │ │ │ │ - obj.annotation = {}; │ │ │ │ │ - this.readChildNodes(node, obj.annotation); │ │ │ │ │ - }, │ │ │ │ │ - "appinfo": function(node, obj) { │ │ │ │ │ - if (!obj.appinfo) { │ │ │ │ │ - obj.appinfo = []; │ │ │ │ │ - } │ │ │ │ │ - obj.appinfo.push(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "documentation": function(node, obj) { │ │ │ │ │ - if (!obj.documentation) { │ │ │ │ │ - obj.documentation = []; │ │ │ │ │ - } │ │ │ │ │ - var value = this.getChildValue(node); │ │ │ │ │ - obj.documentation.push({ │ │ │ │ │ - lang: node.getAttribute("xml:lang"), │ │ │ │ │ - textContent: value.replace(this.regExes.trimSpace, "") │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "simpleType": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "restriction": function(node, obj) { │ │ │ │ │ - obj.base = node.getAttribute("base"); │ │ │ │ │ - this.readRestriction(node, obj); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: readRestriction │ │ │ │ │ - * Reads restriction defined in the child nodes of a restriction element │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} the node to parse │ │ │ │ │ - * obj - {Object} the object that receives the read result │ │ │ │ │ - */ │ │ │ │ │ - readRestriction: function(node, obj) { │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - var child, nodeName, value; │ │ │ │ │ - for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ - child = children[i]; │ │ │ │ │ - if (child.nodeType == 1) { │ │ │ │ │ - nodeName = child.nodeName.split(":").pop(); │ │ │ │ │ - value = child.getAttribute("value"); │ │ │ │ │ - if (!obj[nodeName]) { │ │ │ │ │ - obj[nodeName] = value; │ │ │ │ │ - } else { │ │ │ │ │ - if (typeof obj[nodeName] == "string") { │ │ │ │ │ - obj[nodeName] = [obj[nodeName]]; │ │ │ │ │ - } │ │ │ │ │ - obj[nodeName].push(value); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: read │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {DOMElement|String} A WFS DescribeFeatureType document. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object representing the WFS DescribeFeatureType response. │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ - } │ │ │ │ │ - var schema = {}; │ │ │ │ │ - if (data.nodeName.split(":").pop() === 'ExceptionReport') { │ │ │ │ │ - // an exception must have occurred, so parse it │ │ │ │ │ - var parser = new OpenLayers.Format.OGCExceptionReport(); │ │ │ │ │ - schema.error = parser.read(data); │ │ │ │ │ - } else { │ │ │ │ │ - this.readNode(data, schema); │ │ │ │ │ - } │ │ │ │ │ - return schema; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSDescribeFeatureType" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/SLD.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ - * @requires OpenLayers/Style.js │ │ │ │ │ - * @requires OpenLayers/Rule.js │ │ │ │ │ - * @requires OpenLayers/Filter/FeatureId.js │ │ │ │ │ - * @requires OpenLayers/Filter/Logical.js │ │ │ │ │ - * @requires OpenLayers/Filter/Comparison.js │ │ │ │ │ - * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.SLD │ │ │ │ │ - * Read/Write SLD. Create a new instance with the <OpenLayers.Format.SLD> │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.SLD = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: profile │ │ │ │ │ - * {String} If provided, use a custom profile. │ │ │ │ │ - * │ │ │ │ │ - * Currently supported profiles: │ │ │ │ │ - * - GeoServer - parses GeoServer vendor specific capabilities for SLD. │ │ │ │ │ - */ │ │ │ │ │ - profile: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.0.0". │ │ │ │ │ - */ │ │ │ │ │ - defaultVersion: "1.0.0", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: stringifyOutput │ │ │ │ │ - * {Boolean} If true, write will return a string otherwise a DOMElement. │ │ │ │ │ - * Default is true. │ │ │ │ │ - */ │ │ │ │ │ - stringifyOutput: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: namedLayersAsArray │ │ │ │ │ - * {Boolean} Generate a namedLayers array. If false, the namedLayers │ │ │ │ │ - * property value will be an object keyed by layer name. Default is │ │ │ │ │ - * false. │ │ │ │ │ - */ │ │ │ │ │ - namedLayersAsArray: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Write a SLD document given a list of styles. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * sld - {Object} An object representing the SLD. │ │ │ │ │ - * options - {Object} Optional configuration object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} An SLD document string. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read and SLD doc and return an object representing the SLD. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String | DOMElement} Data to read. │ │ │ │ │ - * options - {Object} Options for the reader. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object representing the SLD. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SLD" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WFS.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/GML.js │ │ │ │ │ - * @requires OpenLayers/Console.js │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WFS │ │ │ │ │ - * Read/Write WFS. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.GML> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WFS = OpenLayers.Class(OpenLayers.Format.GML, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer>} │ │ │ │ │ - */ │ │ │ │ │ - layer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: wfsns │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - wfsns: "http://www.opengis.net/wfs", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: ogcns │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - ogcns: "http://www.opengis.net/ogc", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WFS │ │ │ │ │ - * Create a WFS-T formatter. This requires a layer: that layer should │ │ │ │ │ - * have two properties: geometry_column and typename. The parser │ │ │ │ │ - * for this format is subclassed entirely from GML: There is a writer │ │ │ │ │ - * only, which uses most of the code from the GML layer, and wraps │ │ │ │ │ - * it in transactional elements. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ - * layer - {<OpenLayers.Layer>} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options, layer) { │ │ │ │ │ - OpenLayers.Format.GML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - if (this.layer.featureNS) { │ │ │ │ │ - this.featureNS = this.layer.featureNS; │ │ │ │ │ - } │ │ │ │ │ - if (this.layer.options.geometry_column) { │ │ │ │ │ - this.geometryName = this.layer.options.geometry_column; │ │ │ │ │ - } │ │ │ │ │ - if (this.layer.options.typename) { │ │ │ │ │ - this.featureName = this.layer.options.typename; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: write │ │ │ │ │ - * Takes a feature list, and generates a WFS-T Transaction │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ - */ │ │ │ │ │ - write: function(features) { │ │ │ │ │ - │ │ │ │ │ - var transaction = this.createElementNS(this.wfsns, 'wfs:Transaction'); │ │ │ │ │ - transaction.setAttribute("version", "1.0.0"); │ │ │ │ │ - transaction.setAttribute("service", "WFS"); │ │ │ │ │ - for (var i = 0; i < features.length; i++) { │ │ │ │ │ - switch (features[i].state) { │ │ │ │ │ - case OpenLayers.State.INSERT: │ │ │ │ │ - transaction.appendChild(this.insert(features[i])); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.State.UPDATE: │ │ │ │ │ - transaction.appendChild(this.update(features[i])); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.State.DELETE: │ │ │ │ │ - transaction.appendChild(this.remove(features[i])); │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [transaction]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createFeatureXML │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - createFeatureXML: function(feature) { │ │ │ │ │ - var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ - var geomContainer = this.createElementNS(this.featureNS, "feature:" + this.geometryName); │ │ │ │ │ - geomContainer.appendChild(geometryNode); │ │ │ │ │ - var featureContainer = this.createElementNS(this.featureNS, "feature:" + this.featureName); │ │ │ │ │ - featureContainer.appendChild(geomContainer); │ │ │ │ │ - for (var attr in feature.attributes) { │ │ │ │ │ - var attrText = this.createTextNode(feature.attributes[attr]); │ │ │ │ │ - var nodename = attr; │ │ │ │ │ - if (attr.search(":") != -1) { │ │ │ │ │ - nodename = attr.split(":")[1]; │ │ │ │ │ - } │ │ │ │ │ - var attrContainer = this.createElementNS(this.featureNS, "feature:" + nodename); │ │ │ │ │ - attrContainer.appendChild(attrText); │ │ │ │ │ - featureContainer.appendChild(attrContainer); │ │ │ │ │ - } │ │ │ │ │ - return featureContainer; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: insert │ │ │ │ │ - * Takes a feature, and generates a WFS-T Transaction "Insert" │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - insert: function(feature) { │ │ │ │ │ - var insertNode = this.createElementNS(this.wfsns, 'wfs:Insert'); │ │ │ │ │ - insertNode.appendChild(this.createFeatureXML(feature)); │ │ │ │ │ - return insertNode; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: update │ │ │ │ │ - * Takes a feature, and generates a WFS-T Transaction "Update" │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - update: function(feature) { │ │ │ │ │ - if (!feature.fid) { │ │ │ │ │ - OpenLayers.Console.userError(OpenLayers.i18n("noFID")); │ │ │ │ │ - } │ │ │ │ │ - var updateNode = this.createElementNS(this.wfsns, 'wfs:Update'); │ │ │ │ │ - updateNode.setAttribute("typeName", this.featurePrefix + ':' + this.featureName); │ │ │ │ │ - updateNode.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ - │ │ │ │ │ - var propertyNode = this.createElementNS(this.wfsns, 'wfs:Property'); │ │ │ │ │ - var nameNode = this.createElementNS(this.wfsns, 'wfs:Name'); │ │ │ │ │ - │ │ │ │ │ - var txtNode = this.createTextNode(this.geometryName); │ │ │ │ │ - nameNode.appendChild(txtNode); │ │ │ │ │ - propertyNode.appendChild(nameNode); │ │ │ │ │ - │ │ │ │ │ - var valueNode = this.createElementNS(this.wfsns, 'wfs:Value'); │ │ │ │ │ - │ │ │ │ │ - var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ - │ │ │ │ │ - if (feature.layer) { │ │ │ │ │ - geometryNode.setAttribute( │ │ │ │ │ - "srsName", feature.layer.projection.getCode() │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - valueNode.appendChild(geometryNode); │ │ │ │ │ - │ │ │ │ │ - propertyNode.appendChild(valueNode); │ │ │ │ │ - updateNode.appendChild(propertyNode); │ │ │ │ │ - │ │ │ │ │ - // add in attributes │ │ │ │ │ - for (var propName in feature.attributes) { │ │ │ │ │ - propertyNode = this.createElementNS(this.wfsns, 'wfs:Property'); │ │ │ │ │ - nameNode = this.createElementNS(this.wfsns, 'wfs:Name'); │ │ │ │ │ - nameNode.appendChild(this.createTextNode(propName)); │ │ │ │ │ - propertyNode.appendChild(nameNode); │ │ │ │ │ - valueNode = this.createElementNS(this.wfsns, 'wfs:Value'); │ │ │ │ │ - valueNode.appendChild(this.createTextNode(feature.attributes[propName])); │ │ │ │ │ - propertyNode.appendChild(valueNode); │ │ │ │ │ - updateNode.appendChild(propertyNode); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - var filterNode = this.createElementNS(this.ogcns, 'ogc:Filter'); │ │ │ │ │ - var filterIdNode = this.createElementNS(this.ogcns, 'ogc:FeatureId'); │ │ │ │ │ - filterIdNode.setAttribute("fid", feature.fid); │ │ │ │ │ - filterNode.appendChild(filterIdNode); │ │ │ │ │ - updateNode.appendChild(filterNode); │ │ │ │ │ - │ │ │ │ │ - return updateNode; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: remove │ │ │ │ │ - * Takes a feature, and generates a WFS-T Transaction "Delete" │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - remove: function(feature) { │ │ │ │ │ - if (!feature.fid) { │ │ │ │ │ - OpenLayers.Console.userError(OpenLayers.i18n("noFID")); │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - var deleteNode = this.createElementNS(this.wfsns, 'wfs:Delete'); │ │ │ │ │ - deleteNode.setAttribute("typeName", this.featurePrefix + ':' + this.featureName); │ │ │ │ │ - deleteNode.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ - │ │ │ │ │ - var filterNode = this.createElementNS(this.ogcns, 'ogc:Filter'); │ │ │ │ │ - var filterIdNode = this.createElementNS(this.ogcns, 'ogc:FeatureId'); │ │ │ │ │ - filterIdNode.setAttribute("fid", feature.fid); │ │ │ │ │ - filterNode.appendChild(filterIdNode); │ │ │ │ │ - deleteNode.appendChild(filterNode); │ │ │ │ │ - │ │ │ │ │ - return deleteNode; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Remove ciruclar ref to layer │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.layer = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFS" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/XLS.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.XLS │ │ │ │ │ - * Read/Write XLS (OpenLS). Create a new instance with the <OpenLayers.Format.XLS> │ │ │ │ │ - * constructor. Currently only implemented for Location Utility Services, more │ │ │ │ │ - * specifically only for Geocoding. No support for Reverse Geocoding as yet. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.XLS = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: defaultVersion │ │ │ │ │ - * {String} Version number to assume if none found. Default is "1.1.0". │ │ │ │ │ - */ │ │ │ │ │ - defaultVersion: "1.1.0", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: stringifyOutput │ │ │ │ │ - * {Boolean} If true, write will return a string otherwise a DOMElement. │ │ │ │ │ - * Default is true. │ │ │ │ │ - */ │ │ │ │ │ - stringifyOutput: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.XLS │ │ │ │ │ - * Create a new parser for XLS. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Write out an XLS request. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * request - {Object} An object representing the LUS request. │ │ │ │ │ - * options - {Object} Optional configuration object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} An XLS document string. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read an XLS doc and return an object representing the result. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String | DOMElement} Data to read. │ │ │ │ │ - * options - {Object} Options for the reader. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object representing the GeocodeResponse. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.XLS" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/GML/v2.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/GML/Base.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.GML.v2 │ │ │ │ │ - * Parses GML version 2. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.GML.Base> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.GML.v2 = OpenLayers.Class(OpenLayers.Format.GML.Base, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} Schema location for a particular minor version. │ │ │ │ │ - */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/gml http://schemas.opengis.net/gml/2.1.2/feature.xsd", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.GML.v2 │ │ │ │ │ - * Create a parser for GML v2. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ - * featureNS - {String} Feature namespace (required). │ │ │ │ │ - * geometryName - {String} Geometry element name. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.GML.Base.prototype.initialize.apply(this, [options]); │ │ │ │ │ - }, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.GML.Base.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Property: readers │ │ │ │ │ * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ * be applied when a namespaced node is found matching the function │ │ │ │ │ * name. The function will be applied in the scope of this parser │ │ │ │ │ * with two arguments: the node being read and a context object passed │ │ │ │ │ @@ -59882,14 +55281,626 @@ │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.Filter.v1_0_0" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WFST/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WFST/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/Filter/v1_0_0.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WFST.v1_0_0 │ │ │ │ │ + * A format for creating WFS v1.0.0 transactions. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Format.WFST.v1_0_0> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.Filter.v1_0_0> │ │ │ │ │ + * - <OpenLayers.Format.WFST.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WFST.v1_0_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.Filter.v1_0_0, OpenLayers.Format.WFST.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} WFS version number. │ │ │ │ │ + */ │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: srsNameInQuery │ │ │ │ │ + * {Boolean} If true the reference system is passed in Query requests │ │ │ │ │ + * via the "srsName" attribute to the "wfs:Query" element, this │ │ │ │ │ + * property defaults to false as it isn't WFS 1.0.0 compliant. │ │ │ │ │ + */ │ │ │ │ │ + srsNameInQuery: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocations │ │ │ │ │ + * {Object} Properties are namespace aliases, values are schema locations. │ │ │ │ │ + */ │ │ │ │ │ + schemaLocations: { │ │ │ │ │ + "wfs": "http://schemas.opengis.net/wfs/1.0.0/WFS-transaction.xsd" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WFST.v1_0_0 │ │ │ │ │ + * A class for parsing and generating WFS v1.0.0 transactions. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + * │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ + * featureNS - {String} Feature namespace (optional). │ │ │ │ │ + * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ + * if featureNS is provided). Default is 'feature'. │ │ │ │ │ + * geometryName - {String} Name of geometry attribute. Default is 'the_geom'. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.Filter.v1_0_0.prototype.initialize.apply(this, [options]); │ │ │ │ │ + OpenLayers.Format.WFST.v1.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: readNode │ │ │ │ │ + * Shorthand for applying one of the named readers given the node │ │ │ │ │ + * namespace and local name. Readers take two args (node, obj) and │ │ │ │ │ + * generally extend or modify the second. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} The node to be read (required). │ │ │ │ │ + * obj - {Object} The object to be modified (optional). │ │ │ │ │ + * first - {Boolean} Should be set to true for the first node read. This │ │ │ │ │ + * is usually the readNode call in the read method. Without this being │ │ │ │ │ + * set, auto-configured properties will stick on subsequent reads. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The input object, modified (or a new one if none was provided). │ │ │ │ │ + */ │ │ │ │ │ + readNode: function(node, obj, first) { │ │ │ │ │ + // Not the superclass, only the mixin classes inherit from │ │ │ │ │ + // Format.GML.v2. We need this because we don't want to get readNode │ │ │ │ │ + // from the superclass's superclass, which is OpenLayers.Format.XML. │ │ │ │ │ + return OpenLayers.Format.GML.v2.prototype.readNode.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "WFS_TransactionResponse": function(node, obj) { │ │ │ │ │ + obj.insertIds = []; │ │ │ │ │ + obj.success = false; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "InsertResult": function(node, container) { │ │ │ │ │ + var obj = { │ │ │ │ │ + fids: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + container.insertIds = container.insertIds.concat(obj.fids); │ │ │ │ │ + }, │ │ │ │ │ + "TransactionResult": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "Status": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "SUCCESS": function(node, obj) { │ │ │ │ │ + obj.success = true; │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WFST.v1.prototype.readers["wfs"]), │ │ │ │ │ + "gml": OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ + "feature": OpenLayers.Format.GML.v2.prototype.readers["feature"], │ │ │ │ │ + "ogc": OpenLayers.Format.Filter.v1_0_0.prototype.readers["ogc"] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ + */ │ │ │ │ │ + writers: { │ │ │ │ │ + "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "Query": function(options) { │ │ │ │ │ + options = OpenLayers.Util.extend({ │ │ │ │ │ + featureNS: this.featureNS, │ │ │ │ │ + featurePrefix: this.featurePrefix, │ │ │ │ │ + featureType: this.featureType, │ │ │ │ │ + srsName: this.srsName, │ │ │ │ │ + srsNameInQuery: this.srsNameInQuery │ │ │ │ │ + }, options); │ │ │ │ │ + var prefix = options.featurePrefix; │ │ │ │ │ + var node = this.createElementNSPlus("wfs:Query", { │ │ │ │ │ + attributes: { │ │ │ │ │ + typeName: (prefix ? prefix + ":" : "") + │ │ │ │ │ + options.featureType │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (options.srsNameInQuery && options.srsName) { │ │ │ │ │ + node.setAttribute("srsName", options.srsName); │ │ │ │ │ + } │ │ │ │ │ + if (options.featureNS) { │ │ │ │ │ + node.setAttribute("xmlns:" + prefix, options.featureNS); │ │ │ │ │ + } │ │ │ │ │ + if (options.propertyNames) { │ │ │ │ │ + for (var i = 0, len = options.propertyNames.length; i < len; i++) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "ogc:PropertyName", { │ │ │ │ │ + property: options.propertyNames[i] │ │ │ │ │ + }, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (options.filter) { │ │ │ │ │ + this.setFilterProperty(options.filter); │ │ │ │ │ + this.writeNode("ogc:Filter", options.filter, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WFST.v1.prototype.writers["wfs"]), │ │ │ │ │ + "gml": OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ + "feature": OpenLayers.Format.GML.v2.prototype.writers["feature"], │ │ │ │ │ + "ogc": OpenLayers.Format.Filter.v1_0_0.prototype.writers["ogc"] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFST.v1_0_0" │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Protocol/WFS/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Protocol/WFS/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/WFST/v1_0_0.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.WFS.v1_0_0 │ │ │ │ │ + * A WFS v1.0.0 protocol for vector layers. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Protocol.WFS.v1_0_0> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Protocol.WFS.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.WFS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} WFS version number. │ │ │ │ │ + */ │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Protocol.WFS.v1_0_0 │ │ │ │ │ + * A class for giving layers WFS v1.0.0 protocol. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + * │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ + * featureNS - {String} Feature namespace (optional). │ │ │ │ │ + * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ + * if featureNS is provided). Default is 'feature'. │ │ │ │ │ + * geometryName - {String} Name of geometry attribute. Default is 'the_geom'. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.WFS.v1_0_0" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/SOSGetFeatureOfInterest.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Format/GML/v3.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.SOSGetFeatureOfInterest │ │ │ │ │ + * Read and write SOS GetFeatureOfInterest. This is used to get to │ │ │ │ │ + * the location of the features (stations). The stations can have 1 or more │ │ │ │ │ + * sensors. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.SOSGetFeatureOfInterest = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constant: VERSION │ │ │ │ │ + * {String} 1.0.0 │ │ │ │ │ + */ │ │ │ │ │ + VERSION: "1.0.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ + gml: "http://www.opengis.net/gml", │ │ │ │ │ + sa: "http://www.opengis.net/sampling/1.0", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} Schema location │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: "http://www.opengis.net/sos/1.0 http://schemas.opengis.net/sos/1.0.0/sosAll.xsd", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: "sos", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ + */ │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.SOSGetFeatureOfInterest │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Parse a GetFeatureOfInterest response and return an array of features │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var info = { │ │ │ │ │ + features: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readNode(data, info); │ │ │ │ │ + │ │ │ │ │ + var features = []; │ │ │ │ │ + for (var i = 0, len = info.features.length; i < len; i++) { │ │ │ │ │ + var container = info.features[i]; │ │ │ │ │ + // reproject features if needed │ │ │ │ │ + if (this.internalProjection && this.externalProjection && │ │ │ │ │ + container.components[0]) { │ │ │ │ │ + container.components[0].transform( │ │ │ │ │ + this.externalProjection, this.internalProjection │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector( │ │ │ │ │ + container.components[0], container.attributes); │ │ │ │ │ + features.push(feature); │ │ │ │ │ + } │ │ │ │ │ + return features; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "sa": { │ │ │ │ │ + "SamplingPoint": function(node, obj) { │ │ │ │ │ + // sampling point can also be without a featureMember if │ │ │ │ │ + // there is only 1 │ │ │ │ │ + if (!obj.attributes) { │ │ │ │ │ + var feature = { │ │ │ │ │ + attributes: {} │ │ │ │ │ + }; │ │ │ │ │ + obj.features.push(feature); │ │ │ │ │ + obj = feature; │ │ │ │ │ + } │ │ │ │ │ + obj.attributes.id = this.getAttributeNS(node, │ │ │ │ │ + this.namespaces.gml, "id"); │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "position": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "FeatureCollection": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "featureMember": function(node, obj) { │ │ │ │ │ + var feature = { │ │ │ │ │ + attributes: {} │ │ │ │ │ + }; │ │ │ │ │ + obj.features.push(feature); │ │ │ │ │ + this.readChildNodes(node, feature); │ │ │ │ │ + }, │ │ │ │ │ + "name": function(node, obj) { │ │ │ │ │ + obj.attributes.name = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "pos": function(node, obj) { │ │ │ │ │ + // we need to parse the srsName to get to the │ │ │ │ │ + // externalProjection, that's why we cannot use │ │ │ │ │ + // GML v3 for this │ │ │ │ │ + if (!this.externalProjection) { │ │ │ │ │ + this.externalProjection = new OpenLayers.Projection( │ │ │ │ │ + node.getAttribute("srsName")); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Format.GML.v3.prototype.readers.gml.pos.apply( │ │ │ │ │ + this, [node, obj]); │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.GML.v3.prototype.readers.gml) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ + */ │ │ │ │ │ + writers: { │ │ │ │ │ + "sos": { │ │ │ │ │ + "GetFeatureOfInterest": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("GetFeatureOfInterest", { │ │ │ │ │ + attributes: { │ │ │ │ │ + version: this.VERSION, │ │ │ │ │ + service: 'SOS', │ │ │ │ │ + "xsi:schemaLocation": this.schemaLocation │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + for (var i = 0, len = options.fois.length; i < len; i++) { │ │ │ │ │ + this.writeNode("FeatureOfInterestId", { │ │ │ │ │ + foi: options.fois[i] │ │ │ │ │ + }, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "FeatureOfInterestId": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("FeatureOfInterestId", { │ │ │ │ │ + value: options.foi │ │ │ │ │ + }); │ │ │ │ │ + return node; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SOSGetFeatureOfInterest" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Protocol/SOS/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Protocol/SOS.js │ │ │ │ │ + * @requires OpenLayers/Format/SOSGetFeatureOfInterest.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Protocol.SOS.v1_0_0 │ │ │ │ │ + * An SOS v1.0.0 Protocol for vector layers. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Protocol.SOS.v1_0_0> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Protocol> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Protocol.SOS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: fois │ │ │ │ │ + * {Array(String)} Array of features of interest (foi) │ │ │ │ │ + */ │ │ │ │ │ + fois: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: formatOptions │ │ │ │ │ + * {Object} Optional options for the format. If a format is not provided, │ │ │ │ │ + * this property can be used to extend the default format options. │ │ │ │ │ + */ │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Protocol.SOS │ │ │ │ │ + * A class for giving layers an SOS protocol. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + * │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * url - {String} URL to send requests to (required). │ │ │ │ │ + * fois - {Array} The features of interest (required). │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!options.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.SOSGetFeatureOfInterest( │ │ │ │ │ + this.formatOptions); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up the protocol. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.options && !this.options.format) { │ │ │ │ │ + this.format.destroy(); │ │ │ │ │ + } │ │ │ │ │ + this.format = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Construct a request for reading new sensor positions. This is done by │ │ │ │ │ + * issuing one GetFeatureOfInterest request. │ │ │ │ │ + */ │ │ │ │ │ + read: function(options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + var format = this.format; │ │ │ │ │ + var data = OpenLayers.Format.XML.prototype.write.apply(format, │ │ │ │ │ + [format.writeNode("sos:GetFeatureOfInterest", { │ │ │ │ │ + fois: this.fois │ │ │ │ │ + })] │ │ │ │ │ + ); │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ + data: data │ │ │ │ │ + }); │ │ │ │ │ + return response; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleRead │ │ │ │ │ + * Deal with response from the read request. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ + * to the user callback. │ │ │ │ │ + * options - {Object} The user options passed to the read call. │ │ │ │ │ + */ │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + // success │ │ │ │ │ + response.features = this.parseFeatures(request); │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + } else { │ │ │ │ │ + // failure │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + } │ │ │ │ │ + options.callback.call(options.scope, response); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseFeatures │ │ │ │ │ + * Read HTTP response body and return features │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * request - {XMLHttpRequest} The request object │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array({<OpenLayers.Feature.Vector>})} Array of features │ │ │ │ │ + */ │ │ │ │ │ + parseFeatures: function(request) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ + } │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + return this.format.read(doc); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.SOS.v1_0_0" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/CSWGetRecords.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.CSWGetRecords │ │ │ │ │ + * Default version is 2.0.2. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Format>} A CSWGetRecords format of the given version. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.CSWGetRecords = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options, OpenLayers.Format.CSWGetRecords.DEFAULTS │ │ │ │ │ + ); │ │ │ │ │ + var cls = OpenLayers.Format.CSWGetRecords["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported CSWGetRecords version: " + options.version; │ │ │ │ │ + } │ │ │ │ │ + return new cls(options); │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: DEFAULTS │ │ │ │ │ + * {Object} Default properties for the CSWGetRecords format. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.CSWGetRecords.DEFAULTS = { │ │ │ │ │ + "version": "2.0.2" │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ OpenLayers/Format/CSWGetRecords/v2_0_2.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -60354,13899 +56365,15699 @@ │ │ │ │ │ }, │ │ │ │ │ "ogc": OpenLayers.Format.Filter.v1_1_0.prototype.writers["ogc"] │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.CSWGetRecords.v2_0_2" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/SLD/v1.js │ │ │ │ │ + OpenLayers/Protocol/CSW/v2_0_2.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Rule.js │ │ │ │ │ - * @requires OpenLayers/Format/SLD.js │ │ │ │ │ - * @requires OpenLayers/Format/Filter/v1_0_0.js │ │ │ │ │ - * @requires OpenLayers/Symbolizer/Point.js │ │ │ │ │ - * @requires OpenLayers/Symbolizer/Line.js │ │ │ │ │ - * @requires OpenLayers/Symbolizer/Polygon.js │ │ │ │ │ - * @requires OpenLayers/Symbolizer/Text.js │ │ │ │ │ - * @requires OpenLayers/Symbolizer/Raster.js │ │ │ │ │ + * @requires OpenLayers/Protocol/CSW.js │ │ │ │ │ + * @requires OpenLayers/Format/CSWGetRecords/v2_0_2.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.SLD.v1 │ │ │ │ │ - * Superclass for SLD version 1 parsers. │ │ │ │ │ + * Class: OpenLayers.Protocol.CSW.v2_0_2 │ │ │ │ │ + * CS-W (Catalogue services for the Web) version 2.0.2 protocol. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.Filter.v1_0_0> │ │ │ │ │ + * - <OpenLayers.Protocol> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.SLD.v1 = OpenLayers.Class(OpenLayers.Format.Filter.v1_0_0, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - sld: "http://www.opengis.net/sld", │ │ │ │ │ - ogc: "http://www.opengis.net/ogc", │ │ │ │ │ - gml: "http://www.opengis.net/gml", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - */ │ │ │ │ │ - defaultPrefix: "sld", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} Schema location for a particular minor version. │ │ │ │ │ - */ │ │ │ │ │ - schemaLocation: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: multipleSymbolizers │ │ │ │ │ - * {Boolean} Support multiple symbolizers per rule. Default is false. if │ │ │ │ │ - * true, an OpenLayers.Style2 instance will be created to represent │ │ │ │ │ - * user styles instead of an OpenLayers.Style instace. The │ │ │ │ │ - * OpenLayers.Style2 class allows collections of rules with multiple │ │ │ │ │ - * symbolizers, but is not currently useful for client side rendering. │ │ │ │ │ - * If multiple symbolizers is true, multiple FeatureTypeStyle elements │ │ │ │ │ - * are preserved in reading/writing by setting symbolizer zIndex values. │ │ │ │ │ - * In addition, the <defaultSymbolizer> property is ignored if │ │ │ │ │ - * multiple symbolizers are supported (defaults should be applied │ │ │ │ │ - * when rendering). │ │ │ │ │ - */ │ │ │ │ │ - multipleSymbolizers: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: featureTypeCounter │ │ │ │ │ - * {Number} Private counter for multiple feature type styles. │ │ │ │ │ - */ │ │ │ │ │ - featureTypeCounter: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: defaultSymbolizer. │ │ │ │ │ - * {Object} A symbolizer with the SLD defaults. │ │ │ │ │ - */ │ │ │ │ │ - defaultSymbolizer: { │ │ │ │ │ - fillColor: "#808080", │ │ │ │ │ - fillOpacity: 1, │ │ │ │ │ - strokeColor: "#000000", │ │ │ │ │ - strokeOpacity: 1, │ │ │ │ │ - strokeWidth: 1, │ │ │ │ │ - strokeDashstyle: "solid", │ │ │ │ │ - pointRadius: 3, │ │ │ │ │ - graphicName: "square" │ │ │ │ │ - }, │ │ │ │ │ +OpenLayers.Protocol.CSW.v2_0_2 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.SLD.v1 │ │ │ │ │ - * Instances of this class are not created directly. Use the │ │ │ │ │ - * <OpenLayers.Format.SLD> constructor instead. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * Property: formatOptions │ │ │ │ │ + * {Object} Optional options for the format. If a format is not provided, │ │ │ │ │ + * this property can be used to extend the default format options. │ │ │ │ │ */ │ │ │ │ │ + formatOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read │ │ │ │ │ + * Constructor: OpenLayers.Protocol.CSW.v2_0_2 │ │ │ │ │ + * A class for CSW version 2.0.2 protocol management. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {DOMElement} An SLD document element. │ │ │ │ │ - * options - {Object} Options for the reader. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * namedLayersAsArray - {Boolean} Generate a namedLayers array. If false, │ │ │ │ │ - * the namedLayers property value will be an object keyed by layer name. │ │ │ │ │ - * Default is false. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object representing the SLD. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ */ │ │ │ │ │ - read: function(data, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var sld = { │ │ │ │ │ - namedLayers: options.namedLayersAsArray === true ? [] : {} │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(data, sld); │ │ │ │ │ - return sld; │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!options.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.CSWGetRecords.v2_0_2(OpenLayers.Util.extend({}, this.formatOptions)); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up the protocol. │ │ │ │ │ */ │ │ │ │ │ - readers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "sld": { │ │ │ │ │ - "StyledLayerDescriptor": function(node, sld) { │ │ │ │ │ - sld.version = node.getAttribute("version"); │ │ │ │ │ - this.readChildNodes(node, sld); │ │ │ │ │ - }, │ │ │ │ │ - "Name": function(node, obj) { │ │ │ │ │ - obj.name = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Title": function(node, obj) { │ │ │ │ │ - obj.title = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Abstract": function(node, obj) { │ │ │ │ │ - obj.description = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "NamedLayer": function(node, sld) { │ │ │ │ │ - var layer = { │ │ │ │ │ - userStyles: [], │ │ │ │ │ - namedStyles: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, layer); │ │ │ │ │ - // give each of the user styles this layer name │ │ │ │ │ - for (var i = 0, len = layer.userStyles.length; i < len; ++i) { │ │ │ │ │ - layer.userStyles[i].layerName = layer.name; │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.Util.isArray(sld.namedLayers)) { │ │ │ │ │ - sld.namedLayers.push(layer); │ │ │ │ │ - } else { │ │ │ │ │ - sld.namedLayers[layer.name] = layer; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "NamedStyle": function(node, layer) { │ │ │ │ │ - layer.namedStyles.push( │ │ │ │ │ - this.getChildName(node.firstChild) │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - "UserStyle": function(node, layer) { │ │ │ │ │ - var obj = { │ │ │ │ │ - defaultsPerSymbolizer: true, │ │ │ │ │ - rules: [] │ │ │ │ │ - }; │ │ │ │ │ - this.featureTypeCounter = -1; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - var style; │ │ │ │ │ - if (this.multipleSymbolizers) { │ │ │ │ │ - delete obj.defaultsPerSymbolizer; │ │ │ │ │ - style = new OpenLayers.Style2(obj); │ │ │ │ │ - } else { │ │ │ │ │ - style = new OpenLayers.Style(this.defaultSymbolizer, obj); │ │ │ │ │ - } │ │ │ │ │ - layer.userStyles.push(style); │ │ │ │ │ - }, │ │ │ │ │ - "IsDefault": function(node, style) { │ │ │ │ │ - if (this.getChildValue(node) == "1") { │ │ │ │ │ - style.isDefault = true; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "FeatureTypeStyle": function(node, style) { │ │ │ │ │ - ++this.featureTypeCounter; │ │ │ │ │ - var obj = { │ │ │ │ │ - rules: this.multipleSymbolizers ? style.rules : [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - if (!this.multipleSymbolizers) { │ │ │ │ │ - style.rules = obj.rules; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Rule": function(node, obj) { │ │ │ │ │ - var config; │ │ │ │ │ - if (this.multipleSymbolizers) { │ │ │ │ │ - config = { │ │ │ │ │ - symbolizers: [] │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - var rule = new OpenLayers.Rule(config); │ │ │ │ │ - this.readChildNodes(node, rule); │ │ │ │ │ - obj.rules.push(rule); │ │ │ │ │ - }, │ │ │ │ │ - "ElseFilter": function(node, rule) { │ │ │ │ │ - rule.elseFilter = true; │ │ │ │ │ - }, │ │ │ │ │ - "MinScaleDenominator": function(node, rule) { │ │ │ │ │ - rule.minScaleDenominator = parseFloat(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "MaxScaleDenominator": function(node, rule) { │ │ │ │ │ - rule.maxScaleDenominator = parseFloat(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "TextSymbolizer": function(node, rule) { │ │ │ │ │ - var config = {}; │ │ │ │ │ - this.readChildNodes(node, config); │ │ │ │ │ - if (this.multipleSymbolizers) { │ │ │ │ │ - config.zIndex = this.featureTypeCounter; │ │ │ │ │ - rule.symbolizers.push( │ │ │ │ │ - new OpenLayers.Symbolizer.Text(config) │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - rule.symbolizer["Text"] = OpenLayers.Util.applyDefaults( │ │ │ │ │ - config, rule.symbolizer["Text"] │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "LabelPlacement": function(node, symbolizer) { │ │ │ │ │ - this.readChildNodes(node, symbolizer); │ │ │ │ │ - }, │ │ │ │ │ - "PointPlacement": function(node, symbolizer) { │ │ │ │ │ - var config = {}; │ │ │ │ │ - this.readChildNodes(node, config); │ │ │ │ │ - config.labelRotation = config.rotation; │ │ │ │ │ - delete config.rotation; │ │ │ │ │ - var labelAlign, │ │ │ │ │ - x = symbolizer.labelAnchorPointX, │ │ │ │ │ - y = symbolizer.labelAnchorPointY; │ │ │ │ │ - if (x <= 1 / 3) { │ │ │ │ │ - labelAlign = 'l'; │ │ │ │ │ - } else if (x > 1 / 3 && x < 2 / 3) { │ │ │ │ │ - labelAlign = 'c'; │ │ │ │ │ - } else if (x >= 2 / 3) { │ │ │ │ │ - labelAlign = 'r'; │ │ │ │ │ - } │ │ │ │ │ - if (y <= 1 / 3) { │ │ │ │ │ - labelAlign += 'b'; │ │ │ │ │ - } else if (y > 1 / 3 && y < 2 / 3) { │ │ │ │ │ - labelAlign += 'm'; │ │ │ │ │ - } else if (y >= 2 / 3) { │ │ │ │ │ - labelAlign += 't'; │ │ │ │ │ - } │ │ │ │ │ - config.labelAlign = labelAlign; │ │ │ │ │ - OpenLayers.Util.applyDefaults(symbolizer, config); │ │ │ │ │ - }, │ │ │ │ │ - "AnchorPoint": function(node, symbolizer) { │ │ │ │ │ - this.readChildNodes(node, symbolizer); │ │ │ │ │ - }, │ │ │ │ │ - "AnchorPointX": function(node, symbolizer) { │ │ │ │ │ - var labelAnchorPointX = this.readers.ogc._expression.call(this, node); │ │ │ │ │ - // always string, could be empty string │ │ │ │ │ - if (labelAnchorPointX) { │ │ │ │ │ - symbolizer.labelAnchorPointX = labelAnchorPointX; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "AnchorPointY": function(node, symbolizer) { │ │ │ │ │ - var labelAnchorPointY = this.readers.ogc._expression.call(this, node); │ │ │ │ │ - // always string, could be empty string │ │ │ │ │ - if (labelAnchorPointY) { │ │ │ │ │ - symbolizer.labelAnchorPointY = labelAnchorPointY; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Displacement": function(node, symbolizer) { │ │ │ │ │ - this.readChildNodes(node, symbolizer); │ │ │ │ │ - }, │ │ │ │ │ - "DisplacementX": function(node, symbolizer) { │ │ │ │ │ - var labelXOffset = this.readers.ogc._expression.call(this, node); │ │ │ │ │ - // always string, could be empty string │ │ │ │ │ - if (labelXOffset) { │ │ │ │ │ - symbolizer.labelXOffset = labelXOffset; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "DisplacementY": function(node, symbolizer) { │ │ │ │ │ - var labelYOffset = this.readers.ogc._expression.call(this, node); │ │ │ │ │ - // always string, could be empty string │ │ │ │ │ - if (labelYOffset) { │ │ │ │ │ - symbolizer.labelYOffset = labelYOffset; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "LinePlacement": function(node, symbolizer) { │ │ │ │ │ - this.readChildNodes(node, symbolizer); │ │ │ │ │ - }, │ │ │ │ │ - "PerpendicularOffset": function(node, symbolizer) { │ │ │ │ │ - var labelPerpendicularOffset = this.readers.ogc._expression.call(this, node); │ │ │ │ │ - // always string, could be empty string │ │ │ │ │ - if (labelPerpendicularOffset) { │ │ │ │ │ - symbolizer.labelPerpendicularOffset = labelPerpendicularOffset; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Label": function(node, symbolizer) { │ │ │ │ │ - var value = this.readers.ogc._expression.call(this, node); │ │ │ │ │ - if (value) { │ │ │ │ │ - symbolizer.label = value; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Font": function(node, symbolizer) { │ │ │ │ │ - this.readChildNodes(node, symbolizer); │ │ │ │ │ - }, │ │ │ │ │ - "Halo": function(node, symbolizer) { │ │ │ │ │ - // halo has a fill, so send fresh object │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - symbolizer.haloRadius = obj.haloRadius; │ │ │ │ │ - symbolizer.haloColor = obj.fillColor; │ │ │ │ │ - symbolizer.haloOpacity = obj.fillOpacity; │ │ │ │ │ - }, │ │ │ │ │ - "Radius": function(node, symbolizer) { │ │ │ │ │ - var radius = this.readers.ogc._expression.call(this, node); │ │ │ │ │ - if (radius != null) { │ │ │ │ │ - // radius is only used for halo │ │ │ │ │ - symbolizer.haloRadius = radius; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "RasterSymbolizer": function(node, rule) { │ │ │ │ │ - var config = {}; │ │ │ │ │ - this.readChildNodes(node, config); │ │ │ │ │ - if (this.multipleSymbolizers) { │ │ │ │ │ - config.zIndex = this.featureTypeCounter; │ │ │ │ │ - rule.symbolizers.push( │ │ │ │ │ - new OpenLayers.Symbolizer.Raster(config) │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - rule.symbolizer["Raster"] = OpenLayers.Util.applyDefaults( │ │ │ │ │ - config, rule.symbolizer["Raster"] │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Geometry": function(node, obj) { │ │ │ │ │ - obj.geometry = {}; │ │ │ │ │ - this.readChildNodes(node, obj.geometry); │ │ │ │ │ - }, │ │ │ │ │ - "ColorMap": function(node, symbolizer) { │ │ │ │ │ - symbolizer.colorMap = []; │ │ │ │ │ - this.readChildNodes(node, symbolizer.colorMap); │ │ │ │ │ - }, │ │ │ │ │ - "ColorMapEntry": function(node, colorMap) { │ │ │ │ │ - var q = node.getAttribute("quantity"); │ │ │ │ │ - var o = node.getAttribute("opacity"); │ │ │ │ │ - colorMap.push({ │ │ │ │ │ - color: node.getAttribute("color"), │ │ │ │ │ - quantity: q !== null ? parseFloat(q) : undefined, │ │ │ │ │ - label: node.getAttribute("label") || undefined, │ │ │ │ │ - opacity: o !== null ? parseFloat(o) : undefined │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "LineSymbolizer": function(node, rule) { │ │ │ │ │ - var config = {}; │ │ │ │ │ - this.readChildNodes(node, config); │ │ │ │ │ - if (this.multipleSymbolizers) { │ │ │ │ │ - config.zIndex = this.featureTypeCounter; │ │ │ │ │ - rule.symbolizers.push( │ │ │ │ │ - new OpenLayers.Symbolizer.Line(config) │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - rule.symbolizer["Line"] = OpenLayers.Util.applyDefaults( │ │ │ │ │ - config, rule.symbolizer["Line"] │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "PolygonSymbolizer": function(node, rule) { │ │ │ │ │ - var config = { │ │ │ │ │ - fill: false, │ │ │ │ │ - stroke: false │ │ │ │ │ - }; │ │ │ │ │ - if (!this.multipleSymbolizers) { │ │ │ │ │ - config = rule.symbolizer["Polygon"] || config; │ │ │ │ │ - } │ │ │ │ │ - this.readChildNodes(node, config); │ │ │ │ │ - if (this.multipleSymbolizers) { │ │ │ │ │ - config.zIndex = this.featureTypeCounter; │ │ │ │ │ - rule.symbolizers.push( │ │ │ │ │ - new OpenLayers.Symbolizer.Polygon(config) │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - rule.symbolizer["Polygon"] = config; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "PointSymbolizer": function(node, rule) { │ │ │ │ │ - var config = { │ │ │ │ │ - fill: false, │ │ │ │ │ - stroke: false, │ │ │ │ │ - graphic: false │ │ │ │ │ - }; │ │ │ │ │ - if (!this.multipleSymbolizers) { │ │ │ │ │ - config = rule.symbolizer["Point"] || config; │ │ │ │ │ - } │ │ │ │ │ - this.readChildNodes(node, config); │ │ │ │ │ - if (this.multipleSymbolizers) { │ │ │ │ │ - config.zIndex = this.featureTypeCounter; │ │ │ │ │ - rule.symbolizers.push( │ │ │ │ │ - new OpenLayers.Symbolizer.Point(config) │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - rule.symbolizer["Point"] = config; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Stroke": function(node, symbolizer) { │ │ │ │ │ - symbolizer.stroke = true; │ │ │ │ │ - this.readChildNodes(node, symbolizer); │ │ │ │ │ - }, │ │ │ │ │ - "Fill": function(node, symbolizer) { │ │ │ │ │ - symbolizer.fill = true; │ │ │ │ │ - this.readChildNodes(node, symbolizer); │ │ │ │ │ - }, │ │ │ │ │ - "CssParameter": function(node, symbolizer) { │ │ │ │ │ - var cssProperty = node.getAttribute("name"); │ │ │ │ │ - var symProperty = this.cssMap[cssProperty]; │ │ │ │ │ - // for labels, fill should map to fontColor and fill-opacity │ │ │ │ │ - // to fontOpacity │ │ │ │ │ - if (symbolizer.label) { │ │ │ │ │ - if (cssProperty === 'fill') { │ │ │ │ │ - symProperty = "fontColor"; │ │ │ │ │ - } else if (cssProperty === 'fill-opacity') { │ │ │ │ │ - symProperty = "fontOpacity"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (symProperty) { │ │ │ │ │ - // Limited support for parsing of OGC expressions │ │ │ │ │ - var value = this.readers.ogc._expression.call(this, node); │ │ │ │ │ - // always string, could be an empty string │ │ │ │ │ - if (value) { │ │ │ │ │ - symbolizer[symProperty] = value; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Graphic": function(node, symbolizer) { │ │ │ │ │ - symbolizer.graphic = true; │ │ │ │ │ - var graphic = {}; │ │ │ │ │ - // painter's order not respected here, clobber previous with next │ │ │ │ │ - this.readChildNodes(node, graphic); │ │ │ │ │ - // directly properties with names that match symbolizer properties │ │ │ │ │ - var properties = [ │ │ │ │ │ - "stroke", "strokeColor", "strokeWidth", "strokeOpacity", │ │ │ │ │ - "strokeLinecap", "fill", "fillColor", "fillOpacity", │ │ │ │ │ - "graphicName", "rotation", "graphicFormat" │ │ │ │ │ - ]; │ │ │ │ │ - var prop, value; │ │ │ │ │ - for (var i = 0, len = properties.length; i < len; ++i) { │ │ │ │ │ - prop = properties[i]; │ │ │ │ │ - value = graphic[prop]; │ │ │ │ │ - if (value != undefined) { │ │ │ │ │ - symbolizer[prop] = value; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // set other generic properties with specific graphic property names │ │ │ │ │ - if (graphic.opacity != undefined) { │ │ │ │ │ - symbolizer.graphicOpacity = graphic.opacity; │ │ │ │ │ - } │ │ │ │ │ - if (graphic.size != undefined) { │ │ │ │ │ - var pointRadius = graphic.size / 2; │ │ │ │ │ - if (isNaN(pointRadius)) { │ │ │ │ │ - // likely a property name │ │ │ │ │ - symbolizer.graphicWidth = graphic.size; │ │ │ │ │ - } else { │ │ │ │ │ - symbolizer.pointRadius = graphic.size / 2; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (graphic.href != undefined) { │ │ │ │ │ - symbolizer.externalGraphic = graphic.href; │ │ │ │ │ - } │ │ │ │ │ - if (graphic.rotation != undefined) { │ │ │ │ │ - symbolizer.rotation = graphic.rotation; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "ExternalGraphic": function(node, graphic) { │ │ │ │ │ - this.readChildNodes(node, graphic); │ │ │ │ │ - }, │ │ │ │ │ - "Mark": function(node, graphic) { │ │ │ │ │ - this.readChildNodes(node, graphic); │ │ │ │ │ - }, │ │ │ │ │ - "WellKnownName": function(node, graphic) { │ │ │ │ │ - graphic.graphicName = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Opacity": function(node, obj) { │ │ │ │ │ - var opacity = this.readers.ogc._expression.call(this, node); │ │ │ │ │ - // always string, could be empty string │ │ │ │ │ - if (opacity) { │ │ │ │ │ - obj.opacity = opacity; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Size": function(node, obj) { │ │ │ │ │ - var size = this.readers.ogc._expression.call(this, node); │ │ │ │ │ - // always string, could be empty string │ │ │ │ │ - if (size) { │ │ │ │ │ - obj.size = size; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Rotation": function(node, obj) { │ │ │ │ │ - var rotation = this.readers.ogc._expression.call(this, node); │ │ │ │ │ - // always string, could be empty string │ │ │ │ │ - if (rotation) { │ │ │ │ │ - obj.rotation = rotation; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "OnlineResource": function(node, obj) { │ │ │ │ │ - obj.href = this.getAttributeNS( │ │ │ │ │ - node, this.namespaces.xlink, "href" │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - "Format": function(node, graphic) { │ │ │ │ │ - graphic.graphicFormat = this.getChildValue(node); │ │ │ │ │ - } │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.options && !this.options.format) { │ │ │ │ │ + this.format.destroy(); │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.Filter.v1_0_0.prototype.readers), │ │ │ │ │ + this.format = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: cssMap │ │ │ │ │ - * {Object} Object mapping supported css property names to OpenLayers │ │ │ │ │ - * symbolizer property names. │ │ │ │ │ + * Method: read │ │ │ │ │ + * Construct a request for reading new records from the Catalogue. │ │ │ │ │ */ │ │ │ │ │ - cssMap: { │ │ │ │ │ - "stroke": "strokeColor", │ │ │ │ │ - "stroke-opacity": "strokeOpacity", │ │ │ │ │ - "stroke-width": "strokeWidth", │ │ │ │ │ - "stroke-linecap": "strokeLinecap", │ │ │ │ │ - "stroke-dasharray": "strokeDashstyle", │ │ │ │ │ - "fill": "fillColor", │ │ │ │ │ - "fill-opacity": "fillOpacity", │ │ │ │ │ - "font-family": "fontFamily", │ │ │ │ │ - "font-size": "fontSize", │ │ │ │ │ - "font-weight": "fontWeight", │ │ │ │ │ - "font-style": "fontStyle" │ │ │ │ │ + read: function(options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + var data = this.format.write(options.params || options); │ │ │ │ │ + │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: data │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + return response; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getCssProperty │ │ │ │ │ - * Given a symbolizer property, get the corresponding CSS property │ │ │ │ │ - * from the <cssMap>. │ │ │ │ │ + * Method: handleRead │ │ │ │ │ + * Deal with response from the read request. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * sym - {String} A symbolizer property name. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A CSS property name or null if none found. │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ + * to the user callback. │ │ │ │ │ + * This response is given a code property, and optionally a data property. │ │ │ │ │ + * The latter represents the CSW records as returned by the call to │ │ │ │ │ + * the CSW format read method. │ │ │ │ │ + * options - {Object} The user options passed to the read call. │ │ │ │ │ */ │ │ │ │ │ - getCssProperty: function(sym) { │ │ │ │ │ - var css = null; │ │ │ │ │ - for (var prop in this.cssMap) { │ │ │ │ │ - if (this.cssMap[prop] == sym) { │ │ │ │ │ - css = prop; │ │ │ │ │ - break; │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + // success │ │ │ │ │ + response.data = this.parseData(request); │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ + } else { │ │ │ │ │ + // failure │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ } │ │ │ │ │ + options.callback.call(options.scope, response); │ │ │ │ │ } │ │ │ │ │ - return css; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getGraphicFormat │ │ │ │ │ - * Given a href for an external graphic, try to determine the mime-type. │ │ │ │ │ - * This method doesn't try too hard, and will fall back to │ │ │ │ │ - * <defaultGraphicFormat> if one of the known <graphicFormats> is not │ │ │ │ │ - * the file extension of the provided href. │ │ │ │ │ + * Method: parseData │ │ │ │ │ + * Read HTTP response body and return records │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * href - {String} │ │ │ │ │ + * request - {XMLHttpRequest} The request object │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} The graphic format. │ │ │ │ │ + * {Object} The CSW records as returned by the call to the format read method. │ │ │ │ │ */ │ │ │ │ │ - getGraphicFormat: function(href) { │ │ │ │ │ - var format, regex; │ │ │ │ │ - for (var key in this.graphicFormats) { │ │ │ │ │ - if (this.graphicFormats[key].test(href)) { │ │ │ │ │ - format = key; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ + parseData: function(request) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ } │ │ │ │ │ - return format || this.defaultGraphicFormat; │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + return this.format.read(doc); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultGraphicFormat │ │ │ │ │ - * {String} If none other can be determined from <getGraphicFormat>, this │ │ │ │ │ - * default will be returned. │ │ │ │ │ - */ │ │ │ │ │ - defaultGraphicFormat: "image/png", │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.CSW.v2_0_2" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WCSCapabilities.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WCSCapabilities │ │ │ │ │ + * Read WCS Capabilities. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WCSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: graphicFormats │ │ │ │ │ - * {Object} Mapping of image mime-types to regular extensions matching │ │ │ │ │ - * well-known file extensions. │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.1.0". │ │ │ │ │ */ │ │ │ │ │ - graphicFormats: { │ │ │ │ │ - "image/jpeg": /\.jpe?g$/i, │ │ │ │ │ - "image/gif": /\.gif$/i, │ │ │ │ │ - "image/png": /\.png$/i │ │ │ │ │ - }, │ │ │ │ │ + defaultVersion: "1.1.0", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: write │ │ │ │ │ + * Constructor: OpenLayers.Format.WCSCapabilities │ │ │ │ │ + * Create a new parser for WCS capabilities. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * sld - {Object} An object representing the SLD. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The root of an SLD document. │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - write: function(sld) { │ │ │ │ │ - return this.writers.sld.StyledLayerDescriptor.apply(this, [sld]); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return a list of coverages. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} List of named coverages. │ │ │ │ │ */ │ │ │ │ │ - writers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "sld": { │ │ │ │ │ - "_OGCExpression": function(nodeName, value) { │ │ │ │ │ - // only the simplest of ogc:expression handled │ │ │ │ │ - // {label: "some text and a ${propertyName}"} │ │ │ │ │ - var node = this.createElementNSPlus(nodeName); │ │ │ │ │ - var tokens = typeof value == "string" ? │ │ │ │ │ - value.split("${") : [value]; │ │ │ │ │ - node.appendChild(this.createTextNode(tokens[0])); │ │ │ │ │ - var item, last; │ │ │ │ │ - for (var i = 1, len = tokens.length; i < len; i++) { │ │ │ │ │ - item = tokens[i]; │ │ │ │ │ - last = item.indexOf("}"); │ │ │ │ │ - if (last > 0) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "ogc:PropertyName", { │ │ │ │ │ - property: item.substring(0, last) │ │ │ │ │ - }, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - node.appendChild( │ │ │ │ │ - this.createTextNode(item.substring(++last)) │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - // no ending }, so this is a literal ${ │ │ │ │ │ - node.appendChild( │ │ │ │ │ - this.createTextNode("${" + item) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "StyledLayerDescriptor": function(sld) { │ │ │ │ │ - var root = this.createElementNSPlus( │ │ │ │ │ - "sld:StyledLayerDescriptor", { │ │ │ │ │ - attributes: { │ │ │ │ │ - "version": this.VERSION, │ │ │ │ │ - "xsi:schemaLocation": this.schemaLocation │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - // For ArcGIS Server it is necessary to define this │ │ │ │ │ - // at the root level (see ticket:2166). │ │ │ │ │ - root.setAttribute("xmlns:ogc", this.namespaces.ogc); │ │ │ │ │ - root.setAttribute("xmlns:gml", this.namespaces.gml); │ │ │ │ │ - │ │ │ │ │ - // add in optional name │ │ │ │ │ - if (sld.name) { │ │ │ │ │ - this.writeNode("Name", sld.name, root); │ │ │ │ │ - } │ │ │ │ │ - // add in optional title │ │ │ │ │ - if (sld.title) { │ │ │ │ │ - this.writeNode("Title", sld.title, root); │ │ │ │ │ - } │ │ │ │ │ - // add in optional description │ │ │ │ │ - if (sld.description) { │ │ │ │ │ - this.writeNode("Abstract", sld.description, root); │ │ │ │ │ - } │ │ │ │ │ - // add in named layers │ │ │ │ │ - // allow namedLayers to be an array │ │ │ │ │ - if (OpenLayers.Util.isArray(sld.namedLayers)) { │ │ │ │ │ - for (var i = 0, len = sld.namedLayers.length; i < len; ++i) { │ │ │ │ │ - this.writeNode("NamedLayer", sld.namedLayers[i], root); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - for (var name in sld.namedLayers) { │ │ │ │ │ - this.writeNode("NamedLayer", sld.namedLayers[name], root); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return root; │ │ │ │ │ - }, │ │ │ │ │ - "Name": function(name) { │ │ │ │ │ - return this.createElementNSPlus("sld:Name", { │ │ │ │ │ - value: name │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "Title": function(title) { │ │ │ │ │ - return this.createElementNSPlus("sld:Title", { │ │ │ │ │ - value: title │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "Abstract": function(description) { │ │ │ │ │ - return this.createElementNSPlus( │ │ │ │ │ - "sld:Abstract", { │ │ │ │ │ - value: description │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - "NamedLayer": function(layer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:NamedLayer"); │ │ │ │ │ - │ │ │ │ │ - // add in required name │ │ │ │ │ - this.writeNode("Name", layer.name, node); │ │ │ │ │ - │ │ │ │ │ - // optional sld:LayerFeatureConstraints here │ │ │ │ │ - │ │ │ │ │ - // add in named styles │ │ │ │ │ - if (layer.namedStyles) { │ │ │ │ │ - for (var i = 0, len = layer.namedStyles.length; i < len; ++i) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "NamedStyle", layer.namedStyles[i], node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // add in user styles │ │ │ │ │ - if (layer.userStyles) { │ │ │ │ │ - for (var i = 0, len = layer.userStyles.length; i < len; ++i) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "UserStyle", layer.userStyles[i], node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "NamedStyle": function(name) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:NamedStyle"); │ │ │ │ │ - this.writeNode("Name", name, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "UserStyle": function(style) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:UserStyle"); │ │ │ │ │ - │ │ │ │ │ - // add in optional name │ │ │ │ │ - if (style.name) { │ │ │ │ │ - this.writeNode("Name", style.name, node); │ │ │ │ │ - } │ │ │ │ │ - // add in optional title │ │ │ │ │ - if (style.title) { │ │ │ │ │ - this.writeNode("Title", style.title, node); │ │ │ │ │ - } │ │ │ │ │ - // add in optional description │ │ │ │ │ - if (style.description) { │ │ │ │ │ - this.writeNode("Abstract", style.description, node); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // add isdefault │ │ │ │ │ - if (style.isDefault) { │ │ │ │ │ - this.writeNode("IsDefault", style.isDefault, node); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // add FeatureTypeStyles │ │ │ │ │ - if (this.multipleSymbolizers && style.rules) { │ │ │ │ │ - // group style objects by symbolizer zIndex │ │ │ │ │ - var rulesByZ = { │ │ │ │ │ - 0: [] │ │ │ │ │ - }; │ │ │ │ │ - var zValues = [0]; │ │ │ │ │ - var rule, ruleMap, symbolizer, zIndex, clone; │ │ │ │ │ - for (var i = 0, ii = style.rules.length; i < ii; ++i) { │ │ │ │ │ - rule = style.rules[i]; │ │ │ │ │ - if (rule.symbolizers) { │ │ │ │ │ - ruleMap = {}; │ │ │ │ │ - for (var j = 0, jj = rule.symbolizers.length; j < jj; ++j) { │ │ │ │ │ - symbolizer = rule.symbolizers[j]; │ │ │ │ │ - zIndex = symbolizer.zIndex; │ │ │ │ │ - if (!(zIndex in ruleMap)) { │ │ │ │ │ - clone = rule.clone(); │ │ │ │ │ - clone.symbolizers = []; │ │ │ │ │ - ruleMap[zIndex] = clone; │ │ │ │ │ - } │ │ │ │ │ - ruleMap[zIndex].symbolizers.push(symbolizer.clone()); │ │ │ │ │ - } │ │ │ │ │ - for (zIndex in ruleMap) { │ │ │ │ │ - if (!(zIndex in rulesByZ)) { │ │ │ │ │ - zValues.push(zIndex); │ │ │ │ │ - rulesByZ[zIndex] = []; │ │ │ │ │ - } │ │ │ │ │ - rulesByZ[zIndex].push(ruleMap[zIndex]); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - // no symbolizers in rule │ │ │ │ │ - rulesByZ[0].push(rule.clone()); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // write one FeatureTypeStyle per zIndex │ │ │ │ │ - zValues.sort(); │ │ │ │ │ - var rules; │ │ │ │ │ - for (var i = 0, ii = zValues.length; i < ii; ++i) { │ │ │ │ │ - rules = rulesByZ[zValues[i]]; │ │ │ │ │ - if (rules.length > 0) { │ │ │ │ │ - clone = style.clone(); │ │ │ │ │ - clone.rules = rulesByZ[zValues[i]]; │ │ │ │ │ - this.writeNode("FeatureTypeStyle", clone, node); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.writeNode("FeatureTypeStyle", style, node); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "IsDefault": function(bool) { │ │ │ │ │ - return this.createElementNSPlus( │ │ │ │ │ - "sld:IsDefault", { │ │ │ │ │ - value: (bool) ? "1" : "0" │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - "FeatureTypeStyle": function(style) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:FeatureTypeStyle"); │ │ │ │ │ - │ │ │ │ │ - // OpenLayers currently stores no Name, Title, Abstract, │ │ │ │ │ - // FeatureTypeName, or SemanticTypeIdentifier information │ │ │ │ │ - // related to FeatureTypeStyle │ │ │ │ │ - │ │ │ │ │ - // add in rules │ │ │ │ │ - for (var i = 0, len = style.rules.length; i < len; ++i) { │ │ │ │ │ - this.writeNode("Rule", style.rules[i], node); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Rule": function(rule) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:Rule"); │ │ │ │ │ - │ │ │ │ │ - // add in optional name │ │ │ │ │ - if (rule.name) { │ │ │ │ │ - this.writeNode("Name", rule.name, node); │ │ │ │ │ - } │ │ │ │ │ - // add in optional title │ │ │ │ │ - if (rule.title) { │ │ │ │ │ - this.writeNode("Title", rule.title, node); │ │ │ │ │ - } │ │ │ │ │ - // add in optional description │ │ │ │ │ - if (rule.description) { │ │ │ │ │ - this.writeNode("Abstract", rule.description, node); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // add in LegendGraphic here │ │ │ │ │ - │ │ │ │ │ - // add in optional filters │ │ │ │ │ - if (rule.elseFilter) { │ │ │ │ │ - this.writeNode("ElseFilter", null, node); │ │ │ │ │ - } else if (rule.filter) { │ │ │ │ │ - this.writeNode("ogc:Filter", rule.filter, node); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // add in scale limits │ │ │ │ │ - if (rule.minScaleDenominator != undefined) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "MinScaleDenominator", rule.minScaleDenominator, node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (rule.maxScaleDenominator != undefined) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "MaxScaleDenominator", rule.maxScaleDenominator, node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var type, symbolizer; │ │ │ │ │ - if (this.multipleSymbolizers && rule.symbolizers) { │ │ │ │ │ - var symbolizer; │ │ │ │ │ - for (var i = 0, ii = rule.symbolizers.length; i < ii; ++i) { │ │ │ │ │ - symbolizer = rule.symbolizers[i]; │ │ │ │ │ - type = symbolizer.CLASS_NAME.split(".").pop(); │ │ │ │ │ - this.writeNode( │ │ │ │ │ - type + "Symbolizer", symbolizer, node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - // add in symbolizers (relies on geometry type keys) │ │ │ │ │ - var types = OpenLayers.Style.SYMBOLIZER_PREFIXES; │ │ │ │ │ - for (var i = 0, len = types.length; i < len; ++i) { │ │ │ │ │ - type = types[i]; │ │ │ │ │ - symbolizer = rule.symbolizer[type]; │ │ │ │ │ - if (symbolizer) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - type + "Symbolizer", symbolizer, node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - "ElseFilter": function() { │ │ │ │ │ - return this.createElementNSPlus("sld:ElseFilter"); │ │ │ │ │ - }, │ │ │ │ │ - "MinScaleDenominator": function(scale) { │ │ │ │ │ - return this.createElementNSPlus( │ │ │ │ │ - "sld:MinScaleDenominator", { │ │ │ │ │ - value: scale │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - "MaxScaleDenominator": function(scale) { │ │ │ │ │ - return this.createElementNSPlus( │ │ │ │ │ - "sld:MaxScaleDenominator", { │ │ │ │ │ - value: scale │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - "LineSymbolizer": function(symbolizer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:LineSymbolizer"); │ │ │ │ │ - this.writeNode("Stroke", symbolizer, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Stroke": function(symbolizer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:Stroke"); │ │ │ │ │ - │ │ │ │ │ - // GraphicFill here │ │ │ │ │ - // GraphicStroke here │ │ │ │ │ - │ │ │ │ │ - // add in CssParameters │ │ │ │ │ - if (symbolizer.strokeColor != undefined) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "CssParameter", { │ │ │ │ │ - symbolizer: symbolizer, │ │ │ │ │ - key: "strokeColor" │ │ │ │ │ - }, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.strokeOpacity != undefined) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "CssParameter", { │ │ │ │ │ - symbolizer: symbolizer, │ │ │ │ │ - key: "strokeOpacity" │ │ │ │ │ - }, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.strokeWidth != undefined) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "CssParameter", { │ │ │ │ │ - symbolizer: symbolizer, │ │ │ │ │ - key: "strokeWidth" │ │ │ │ │ - }, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.strokeDashstyle != undefined && symbolizer.strokeDashstyle !== "solid") { │ │ │ │ │ - // assumes valid stroke-dasharray value │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "CssParameter", { │ │ │ │ │ - symbolizer: symbolizer, │ │ │ │ │ - key: "strokeDashstyle" │ │ │ │ │ - }, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.strokeLinecap != undefined) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "CssParameter", { │ │ │ │ │ - symbolizer: symbolizer, │ │ │ │ │ - key: "strokeLinecap" │ │ │ │ │ - }, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "CssParameter": function(obj) { │ │ │ │ │ - // not handling ogc:expressions for now │ │ │ │ │ - return this.createElementNSPlus("sld:CssParameter", { │ │ │ │ │ - attributes: { │ │ │ │ │ - name: this.getCssProperty(obj.key) │ │ │ │ │ - }, │ │ │ │ │ - value: obj.symbolizer[obj.key] │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "TextSymbolizer": function(symbolizer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:TextSymbolizer"); │ │ │ │ │ - // add in optional Label │ │ │ │ │ - if (symbolizer.label != null) { │ │ │ │ │ - this.writeNode("Label", symbolizer.label, node); │ │ │ │ │ - } │ │ │ │ │ - // add in optional Font │ │ │ │ │ - if (symbolizer.fontFamily != null || │ │ │ │ │ - symbolizer.fontSize != null || │ │ │ │ │ - symbolizer.fontWeight != null || │ │ │ │ │ - symbolizer.fontStyle != null) { │ │ │ │ │ - this.writeNode("Font", symbolizer, node); │ │ │ │ │ - } │ │ │ │ │ - // add in optional LabelPlacement │ │ │ │ │ - if (symbolizer.labelAnchorPointX != null || │ │ │ │ │ - symbolizer.labelAnchorPointY != null || │ │ │ │ │ - symbolizer.labelAlign != null || │ │ │ │ │ - symbolizer.labelXOffset != null || │ │ │ │ │ - symbolizer.labelYOffset != null || │ │ │ │ │ - symbolizer.labelRotation != null || │ │ │ │ │ - symbolizer.labelPerpendicularOffset != null) { │ │ │ │ │ - this.writeNode("LabelPlacement", symbolizer, node); │ │ │ │ │ - } │ │ │ │ │ - // add in optional Halo │ │ │ │ │ - if (symbolizer.haloRadius != null || │ │ │ │ │ - symbolizer.haloColor != null || │ │ │ │ │ - symbolizer.haloOpacity != null) { │ │ │ │ │ - this.writeNode("Halo", symbolizer, node); │ │ │ │ │ - } │ │ │ │ │ - // add in optional Fill │ │ │ │ │ - if (symbolizer.fontColor != null || │ │ │ │ │ - symbolizer.fontOpacity != null) { │ │ │ │ │ - this.writeNode("Fill", { │ │ │ │ │ - fillColor: symbolizer.fontColor, │ │ │ │ │ - fillOpacity: symbolizer.fontOpacity │ │ │ │ │ - }, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "LabelPlacement": function(symbolizer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:LabelPlacement"); │ │ │ │ │ - // PointPlacement and LinePlacement are choices, so don't output both │ │ │ │ │ - if ((symbolizer.labelAnchorPointX != null || │ │ │ │ │ - symbolizer.labelAnchorPointY != null || │ │ │ │ │ - symbolizer.labelAlign != null || │ │ │ │ │ - symbolizer.labelXOffset != null || │ │ │ │ │ - symbolizer.labelYOffset != null || │ │ │ │ │ - symbolizer.labelRotation != null) && │ │ │ │ │ - symbolizer.labelPerpendicularOffset == null) { │ │ │ │ │ - this.writeNode("PointPlacement", symbolizer, node); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.labelPerpendicularOffset != null) { │ │ │ │ │ - this.writeNode("LinePlacement", symbolizer, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "LinePlacement": function(symbolizer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:LinePlacement"); │ │ │ │ │ - this.writeNode("PerpendicularOffset", symbolizer.labelPerpendicularOffset, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "PerpendicularOffset": function(value) { │ │ │ │ │ - return this.createElementNSPlus("sld:PerpendicularOffset", { │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "PointPlacement": function(symbolizer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:PointPlacement"); │ │ │ │ │ - if (symbolizer.labelAnchorPointX != null || │ │ │ │ │ - symbolizer.labelAnchorPointY != null || │ │ │ │ │ - symbolizer.labelAlign != null) { │ │ │ │ │ - this.writeNode("AnchorPoint", symbolizer, node); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.labelXOffset != null || │ │ │ │ │ - symbolizer.labelYOffset != null) { │ │ │ │ │ - this.writeNode("Displacement", symbolizer, node); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.labelRotation != null) { │ │ │ │ │ - this.writeNode("Rotation", symbolizer.labelRotation, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "AnchorPoint": function(symbolizer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:AnchorPoint"); │ │ │ │ │ - var x = symbolizer.labelAnchorPointX, │ │ │ │ │ - y = symbolizer.labelAnchorPointY; │ │ │ │ │ - if (x != null) { │ │ │ │ │ - this.writeNode("AnchorPointX", x, node); │ │ │ │ │ - } │ │ │ │ │ - if (y != null) { │ │ │ │ │ - this.writeNode("AnchorPointY", y, node); │ │ │ │ │ - } │ │ │ │ │ - if (x == null && y == null) { │ │ │ │ │ - var xAlign = symbolizer.labelAlign.substr(0, 1), │ │ │ │ │ - yAlign = symbolizer.labelAlign.substr(1, 1); │ │ │ │ │ - if (xAlign === "l") { │ │ │ │ │ - x = 0; │ │ │ │ │ - } else if (xAlign === "c") { │ │ │ │ │ - x = 0.5; │ │ │ │ │ - } else if (xAlign === "r") { │ │ │ │ │ - x = 1; │ │ │ │ │ - } │ │ │ │ │ - if (yAlign === "b") { │ │ │ │ │ - y = 0; │ │ │ │ │ - } else if (yAlign === "m") { │ │ │ │ │ - y = 0.5; │ │ │ │ │ - } else if (yAlign === "t") { │ │ │ │ │ - y = 1; │ │ │ │ │ - } │ │ │ │ │ - this.writeNode("AnchorPointX", x, node); │ │ │ │ │ - this.writeNode("AnchorPointY", y, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "AnchorPointX": function(value) { │ │ │ │ │ - return this.createElementNSPlus("sld:AnchorPointX", { │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "AnchorPointY": function(value) { │ │ │ │ │ - return this.createElementNSPlus("sld:AnchorPointY", { │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "Displacement": function(symbolizer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:Displacement"); │ │ │ │ │ - if (symbolizer.labelXOffset != null) { │ │ │ │ │ - this.writeNode("DisplacementX", symbolizer.labelXOffset, node); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.labelYOffset != null) { │ │ │ │ │ - this.writeNode("DisplacementY", symbolizer.labelYOffset, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "DisplacementX": function(value) { │ │ │ │ │ - return this.createElementNSPlus("sld:DisplacementX", { │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "DisplacementY": function(value) { │ │ │ │ │ - return this.createElementNSPlus("sld:DisplacementY", { │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "Font": function(symbolizer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:Font"); │ │ │ │ │ - // add in CssParameters │ │ │ │ │ - if (symbolizer.fontFamily) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "CssParameter", { │ │ │ │ │ - symbolizer: symbolizer, │ │ │ │ │ - key: "fontFamily" │ │ │ │ │ - }, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.fontSize) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "CssParameter", { │ │ │ │ │ - symbolizer: symbolizer, │ │ │ │ │ - key: "fontSize" │ │ │ │ │ - }, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.fontWeight) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "CssParameter", { │ │ │ │ │ - symbolizer: symbolizer, │ │ │ │ │ - key: "fontWeight" │ │ │ │ │ - }, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.fontStyle) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "CssParameter", { │ │ │ │ │ - symbolizer: symbolizer, │ │ │ │ │ - key: "fontStyle" │ │ │ │ │ - }, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Label": function(label) { │ │ │ │ │ - return this.writers.sld._OGCExpression.call( │ │ │ │ │ - this, "sld:Label", label │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - "Halo": function(symbolizer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:Halo"); │ │ │ │ │ - if (symbolizer.haloRadius) { │ │ │ │ │ - this.writeNode("Radius", symbolizer.haloRadius, node); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.haloColor || symbolizer.haloOpacity) { │ │ │ │ │ - this.writeNode("Fill", { │ │ │ │ │ - fillColor: symbolizer.haloColor, │ │ │ │ │ - fillOpacity: symbolizer.haloOpacity │ │ │ │ │ - }, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Radius": function(value) { │ │ │ │ │ - return this.createElementNSPlus("sld:Radius", { │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "RasterSymbolizer": function(symbolizer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:RasterSymbolizer"); │ │ │ │ │ - if (symbolizer.geometry) { │ │ │ │ │ - this.writeNode("Geometry", symbolizer.geometry, node); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.opacity) { │ │ │ │ │ - this.writeNode("Opacity", symbolizer.opacity, node); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.colorMap) { │ │ │ │ │ - this.writeNode("ColorMap", symbolizer.colorMap, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Geometry": function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:Geometry"); │ │ │ │ │ - if (geometry.property) { │ │ │ │ │ - this.writeNode("ogc:PropertyName", geometry, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "ColorMap": function(colorMap) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:ColorMap"); │ │ │ │ │ - for (var i = 0, len = colorMap.length; i < len; ++i) { │ │ │ │ │ - this.writeNode("ColorMapEntry", colorMap[i], node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "ColorMapEntry": function(colorMapEntry) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:ColorMapEntry"); │ │ │ │ │ - var a = colorMapEntry; │ │ │ │ │ - node.setAttribute("color", a.color); │ │ │ │ │ - a.opacity !== undefined && node.setAttribute("opacity", │ │ │ │ │ - parseFloat(a.opacity)); │ │ │ │ │ - a.quantity !== undefined && node.setAttribute("quantity", │ │ │ │ │ - parseFloat(a.quantity)); │ │ │ │ │ - a.label !== undefined && node.setAttribute("label", a.label); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "PolygonSymbolizer": function(symbolizer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:PolygonSymbolizer"); │ │ │ │ │ - if (symbolizer.fill !== false) { │ │ │ │ │ - this.writeNode("Fill", symbolizer, node); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.stroke !== false) { │ │ │ │ │ - this.writeNode("Stroke", symbolizer, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Fill": function(symbolizer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:Fill"); │ │ │ │ │ - │ │ │ │ │ - // GraphicFill here │ │ │ │ │ - │ │ │ │ │ - // add in CssParameters │ │ │ │ │ - if (symbolizer.fillColor) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "CssParameter", { │ │ │ │ │ - symbolizer: symbolizer, │ │ │ │ │ - key: "fillColor" │ │ │ │ │ - }, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.fillOpacity != null) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "CssParameter", { │ │ │ │ │ - symbolizer: symbolizer, │ │ │ │ │ - key: "fillOpacity" │ │ │ │ │ - }, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "PointSymbolizer": function(symbolizer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:PointSymbolizer"); │ │ │ │ │ - this.writeNode("Graphic", symbolizer, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Graphic": function(symbolizer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:Graphic"); │ │ │ │ │ - if (symbolizer.externalGraphic != undefined) { │ │ │ │ │ - this.writeNode("ExternalGraphic", symbolizer, node); │ │ │ │ │ - } else { │ │ │ │ │ - this.writeNode("Mark", symbolizer, node); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (symbolizer.graphicOpacity != undefined) { │ │ │ │ │ - this.writeNode("Opacity", symbolizer.graphicOpacity, node); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.pointRadius != undefined) { │ │ │ │ │ - this.writeNode("Size", symbolizer.pointRadius * 2, node); │ │ │ │ │ - } else if (symbolizer.graphicWidth != undefined) { │ │ │ │ │ - this.writeNode("Size", symbolizer.graphicWidth, node); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.rotation != undefined) { │ │ │ │ │ - this.writeNode("Rotation", symbolizer.rotation, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "ExternalGraphic": function(symbolizer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:ExternalGraphic"); │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "OnlineResource", symbolizer.externalGraphic, node │ │ │ │ │ - ); │ │ │ │ │ - var format = symbolizer.graphicFormat || │ │ │ │ │ - this.getGraphicFormat(symbolizer.externalGraphic); │ │ │ │ │ - this.writeNode("Format", format, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Mark": function(symbolizer) { │ │ │ │ │ - var node = this.createElementNSPlus("sld:Mark"); │ │ │ │ │ - if (symbolizer.graphicName) { │ │ │ │ │ - this.writeNode("WellKnownName", symbolizer.graphicName, node); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.fill !== false) { │ │ │ │ │ - this.writeNode("Fill", symbolizer, node); │ │ │ │ │ - } │ │ │ │ │ - if (symbolizer.stroke !== false) { │ │ │ │ │ - this.writeNode("Stroke", symbolizer, node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "WellKnownName": function(name) { │ │ │ │ │ - return this.createElementNSPlus("sld:WellKnownName", { │ │ │ │ │ - value: name │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "Opacity": function(value) { │ │ │ │ │ - return this.createElementNSPlus("sld:Opacity", { │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "Size": function(value) { │ │ │ │ │ - return this.writers.sld._OGCExpression.call( │ │ │ │ │ - this, "sld:Size", value │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - "Rotation": function(value) { │ │ │ │ │ - return this.createElementNSPlus("sld:Rotation", { │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "OnlineResource": function(href) { │ │ │ │ │ - return this.createElementNSPlus("sld:OnlineResource", { │ │ │ │ │ - attributes: { │ │ │ │ │ - "xlink:type": "simple", │ │ │ │ │ - "xlink:href": href │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "Format": function(format) { │ │ │ │ │ - return this.createElementNSPlus("sld:Format", { │ │ │ │ │ - value: format │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.Filter.v1_0_0.prototype.writers), │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SLD.v1" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WCSCapabilities" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/SLD/v1_0_0.js │ │ │ │ │ + OpenLayers/Format/WMSCapabilities.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/SLD/v1.js │ │ │ │ │ - * @requires OpenLayers/Format/Filter/v1_0_0.js │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.SLD.v1_0_0 │ │ │ │ │ - * Write SLD version 1.0.0. │ │ │ │ │ + * Class: OpenLayers.Format.WMSCapabilities │ │ │ │ │ + * Read WMS Capabilities. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.SLD.v1> │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.SLD.v1_0_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.SLD.v1, { │ │ │ │ │ +OpenLayers.Format.WMSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: VERSION │ │ │ │ │ - * {String} 1.0.0 │ │ │ │ │ - */ │ │ │ │ │ - VERSION: "1.0.0", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.1.1". │ │ │ │ │ + */ │ │ │ │ │ + defaultVersion: "1.1.1", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} http://www.opengis.net/sld │ │ │ │ │ - * http://schemas.opengis.net/sld/1.0.0/StyledLayerDescriptor.xsd │ │ │ │ │ - */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/sld http://schemas.opengis.net/sld/1.0.0/StyledLayerDescriptor.xsd", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: profile │ │ │ │ │ + * {String} If provided, use a custom profile. │ │ │ │ │ + * │ │ │ │ │ + * Currently supported profiles: │ │ │ │ │ + * - WMSC - parses vendor specific capabilities for WMS-C. │ │ │ │ │ + */ │ │ │ │ │ + profile: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.SLD.v1_0_0 │ │ │ │ │ - * Instances of this class are not created directly. Use the │ │ │ │ │ - * <OpenLayers.Format.SLD> constructor instead. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMSCapabilities │ │ │ │ │ + * Create a new parser for WMS capabilities. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SLD.v1_0_0" │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} List of named layers. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/OWSContext/v0_3_1.js │ │ │ │ │ + OpenLayers/Format/ArcXML.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/KML.js │ │ │ │ │ - * @requires OpenLayers/Format/GML.js │ │ │ │ │ - * @requires OpenLayers/Format/GML/v2.js │ │ │ │ │ - * @requires OpenLayers/Format/SLD/v1_0_0.js │ │ │ │ │ - * @requires OpenLayers/Format/OWSContext.js │ │ │ │ │ - * @requires OpenLayers/Format/OWSCommon/v1_0_0.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Geometry/MultiPolygon.js │ │ │ │ │ + * @requires OpenLayers/Geometry/LinearRing.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.OWSContext.v0_3_1 │ │ │ │ │ - * Read and write OWSContext version 0.3.1. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Format.ArcXML │ │ │ │ │ + * Read/Write ArcXML. Create a new instance with the <OpenLayers.Format.ArcXML> │ │ │ │ │ + * constructor. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Format.XML> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.OWSContext.v0_3_1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Format.ArcXML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + * Property: fontStyleKeys │ │ │ │ │ + * {Array} List of keys used in font styling. │ │ │ │ │ */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - owc: "http://www.opengis.net/ows-context", │ │ │ │ │ - gml: "http://www.opengis.net/gml", │ │ │ │ │ - kml: "http://www.opengis.net/kml/2.2", │ │ │ │ │ - ogc: "http://www.opengis.net/ogc", │ │ │ │ │ - ows: "http://www.opengis.net/ows", │ │ │ │ │ - sld: "http://www.opengis.net/sld", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ - }, │ │ │ │ │ + fontStyleKeys: [ │ │ │ │ │ + 'antialiasing', 'blockout', 'font', 'fontcolor', 'fontsize', 'fontstyle', │ │ │ │ │ + 'glowing', 'interval', 'outline', 'printmode', 'shadow', 'transparency' │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: VERSION │ │ │ │ │ - * {String} 0.3.1 │ │ │ │ │ + * Property: request │ │ │ │ │ + * A get_image request destined for an ArcIMS server. │ │ │ │ │ */ │ │ │ │ │ - VERSION: "0.3.1", │ │ │ │ │ + request: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} Schema location │ │ │ │ │ + * Property: response │ │ │ │ │ + * A parsed response from an ArcIMS server. │ │ │ │ │ */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/ows-context http://www.ogcnetwork.net/schemas/owc/0.3.1/owsContext.xsd", │ │ │ │ │ + response: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - * {String} Default namespace prefix to use. │ │ │ │ │ + * Constructor: OpenLayers.Format.ArcXML │ │ │ │ │ + * Create a new parser/writer for ArcXML. Create an instance of this class │ │ │ │ │ + * to begin authoring a request to an ArcIMS service. This is used │ │ │ │ │ + * primarily by the ArcIMS layer, but could be used to do other wild │ │ │ │ │ + * stuff, like geocoding. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - defaultPrefix: "owc", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.request = new OpenLayers.Format.ArcXML.Request(); │ │ │ │ │ + this.response = new OpenLayers.Format.ArcXML.Response(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: extractAttributes │ │ │ │ │ - * {Boolean} Extract attributes from GML. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - extractAttributes: true, │ │ │ │ │ + if (options) { │ │ │ │ │ + if (options.requesttype == "feature") { │ │ │ │ │ + this.request.get_image = null; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: xy │ │ │ │ │ - * {Boolean} Order of the GML coordinate true:(x,y) or false:(y,x) │ │ │ │ │ - * Changing is not recommended, a new Format should be instantiated. │ │ │ │ │ - */ │ │ │ │ │ - xy: true, │ │ │ │ │ + var qry = this.request.get_feature.query; │ │ │ │ │ + this.addCoordSys(qry.featurecoordsys, options.featureCoordSys); │ │ │ │ │ + this.addCoordSys(qry.filtercoordsys, options.filterCoordSys); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ - */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ - }, │ │ │ │ │ + if (options.polygon) { │ │ │ │ │ + qry.isspatial = true; │ │ │ │ │ + qry.spatialfilter.polygon = options.polygon; │ │ │ │ │ + } else if (options.envelope) { │ │ │ │ │ + qry.isspatial = true; │ │ │ │ │ + qry.spatialfilter.envelope = { │ │ │ │ │ + minx: 0, │ │ │ │ │ + miny: 0, │ │ │ │ │ + maxx: 0, │ │ │ │ │ + maxy: 0 │ │ │ │ │ + }; │ │ │ │ │ + this.parseEnvelope(qry.spatialfilter.envelope, options.envelope); │ │ │ │ │ + } │ │ │ │ │ + } else if (options.requesttype == "image") { │ │ │ │ │ + this.request.get_feature = null; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: featureNS │ │ │ │ │ - * {String} The namespace uri to use for writing InlineGeometry │ │ │ │ │ - */ │ │ │ │ │ - featureNS: "http://mapserver.gis.umn.edu/mapserver", │ │ │ │ │ + var props = this.request.get_image.properties; │ │ │ │ │ + this.parseEnvelope(props.envelope, options.envelope); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: featureType │ │ │ │ │ - * {String} The name to use as the feature type when writing out │ │ │ │ │ - * InlineGeometry │ │ │ │ │ - */ │ │ │ │ │ - featureType: 'vector', │ │ │ │ │ + this.addLayers(props.layerlist, options.layers); │ │ │ │ │ + this.addImageSize(props.imagesize, options.tileSize); │ │ │ │ │ + this.addCoordSys(props.featurecoordsys, options.featureCoordSys); │ │ │ │ │ + this.addCoordSys(props.filtercoordsys, options.filterCoordSys); │ │ │ │ │ + } else { │ │ │ │ │ + // if an arcxml object is being created with no request type, it is │ │ │ │ │ + // probably going to consume a response, so do not throw an error if │ │ │ │ │ + // the requesttype is not defined │ │ │ │ │ + this.request = null; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: geometryName │ │ │ │ │ - * {String} The name to use for the geometry attribute when writing out │ │ │ │ │ - * InlineGeometry │ │ │ │ │ - */ │ │ │ │ │ - geometryName: 'geometry', │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: nestingLayerLookup │ │ │ │ │ - * {Object} Hashtable lookup for nesting layer nodes. Used while writing │ │ │ │ │ - * the OWS context document. It is necessary to keep track of the │ │ │ │ │ - * nestingPaths for which nesting layer nodes have already been │ │ │ │ │ - * created, so (nesting) layer nodes are added to those nodes. │ │ │ │ │ + * Method: parseEnvelope │ │ │ │ │ + * Parse an array of coordinates into an ArcXML envelope structure. │ │ │ │ │ * │ │ │ │ │ - * For example: │ │ │ │ │ + * Parameters: │ │ │ │ │ + * env - {Object} An envelope object that will contain the parsed coordinates. │ │ │ │ │ + * arr - {Array(double)} An array of coordinates in the order: [ minx, miny, maxx, maxy ] │ │ │ │ │ + */ │ │ │ │ │ + parseEnvelope: function(env, arr) { │ │ │ │ │ + if (arr && arr.length == 4) { │ │ │ │ │ + env.minx = arr[0]; │ │ │ │ │ + env.miny = arr[1]; │ │ │ │ │ + env.maxx = arr[2]; │ │ │ │ │ + env.maxy = arr[3]; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: addLayers │ │ │ │ │ + * Add a collection of layers to another collection of layers. Each layer in the list is tuple of │ │ │ │ │ + * { id, visible }. These layer collections represent the │ │ │ │ │ + * /ARCXML/REQUEST/get_image/PROPERTIES/LAYERLIST/LAYERDEF items in ArcXML │ │ │ │ │ * │ │ │ │ │ - * If there are three layers with nestingPaths: │ │ │ │ │ - * layer1.metadata.nestingPath = "a/b/" │ │ │ │ │ - * layer2.metadata.nestingPath = "a/b/" │ │ │ │ │ - * layer2.metadata.nestingPath = "a/c" │ │ │ │ │ + * TODO: Add support for dynamic layer rendering. │ │ │ │ │ * │ │ │ │ │ - * then a nesting layer node "a" should be created once and added │ │ │ │ │ - * to the resource list, a nesting layer node "b" should be created │ │ │ │ │ - * once and added under "a", and a nesting layer node "c" should be │ │ │ │ │ - * created and added under "a". The lookup paths for these nodes │ │ │ │ │ - * will be "a", "a/b", and "a/c" respectively. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * ll - {Array({id,visible})} A list of layer definitions. │ │ │ │ │ + * lyrs - {Array({id,visible})} A list of layer definitions. │ │ │ │ │ */ │ │ │ │ │ - nestingLayerLookup: null, │ │ │ │ │ + addLayers: function(ll, lyrs) { │ │ │ │ │ + for (var lind = 0, len = lyrs.length; lind < len; lind++) { │ │ │ │ │ + ll.push(lyrs[lind]); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.OWSContext.v0_3_1 │ │ │ │ │ - * Instances of this class are not created directly. Use the │ │ │ │ │ - * <OpenLayers.Format.OWSContext> constructor instead. │ │ │ │ │ + * Method: addImageSize │ │ │ │ │ + * Set the size of the requested image. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ + * imsize - {Object} An ArcXML imagesize object. │ │ │ │ │ + * olsize - {<OpenLayers.Size>} The image size to set. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - OpenLayers.Format.GML.v2.prototype.setGeometryTypes.call(this); │ │ │ │ │ + addImageSize: function(imsize, olsize) { │ │ │ │ │ + if (olsize !== null) { │ │ │ │ │ + imsize.width = olsize.w; │ │ │ │ │ + imsize.height = olsize.h; │ │ │ │ │ + imsize.printwidth = olsize.w; │ │ │ │ │ + imsize.printheight = olsize.h; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setNestingPath │ │ │ │ │ - * Set the nestingPath property of the layer depending on the position │ │ │ │ │ - * of the layer in hierarchy of layers. │ │ │ │ │ + * Method: addCoordSys │ │ │ │ │ + * Add the coordinate system information to an object. The object may be │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * l - {Object} An object that may have a layersContext array property. │ │ │ │ │ - * │ │ │ │ │ + * featOrFilt - {Object} A featurecoordsys or filtercoordsys ArcXML structure. │ │ │ │ │ + * fsys - {String} or {<OpenLayers.Projection>} or {filtercoordsys} or │ │ │ │ │ + * {featurecoordsys} A projection representation. If it's a {String}, │ │ │ │ │ + * the value is assumed to be the SRID. If it's a {OpenLayers.Projection} │ │ │ │ │ + * AND Proj4js is available, the projection number and name are extracted │ │ │ │ │ + * from there. If it's a filter or feature ArcXML structure, it is copied. │ │ │ │ │ */ │ │ │ │ │ - setNestingPath: function(l) { │ │ │ │ │ - if (l.layersContext) { │ │ │ │ │ - for (var i = 0, len = l.layersContext.length; i < len; i++) { │ │ │ │ │ - var layerContext = l.layersContext[i]; │ │ │ │ │ - var nPath = []; │ │ │ │ │ - var nTitle = l.title || ""; │ │ │ │ │ - if (l.metadata && l.metadata.nestingPath) { │ │ │ │ │ - nPath = l.metadata.nestingPath.slice(); │ │ │ │ │ - } │ │ │ │ │ - if (nTitle != "") { │ │ │ │ │ - nPath.push(nTitle); │ │ │ │ │ - } │ │ │ │ │ - layerContext.metadata.nestingPath = nPath; │ │ │ │ │ - if (layerContext.layersContext) { │ │ │ │ │ - this.setNestingPath(layerContext); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + addCoordSys: function(featOrFilt, fsys) { │ │ │ │ │ + if (typeof fsys == "string") { │ │ │ │ │ + featOrFilt.id = parseInt(fsys); │ │ │ │ │ + featOrFilt.string = fsys; │ │ │ │ │ + } │ │ │ │ │ + // is this a proj4js instance? │ │ │ │ │ + else if (typeof fsys == "object" && fsys.proj !== null) { │ │ │ │ │ + featOrFilt.id = fsys.proj.srsProjNumber; │ │ │ │ │ + featOrFilt.string = fsys.proj.srsCode; │ │ │ │ │ + } else { │ │ │ │ │ + featOrFilt = fsys; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Function: decomposeNestingPath │ │ │ │ │ - * Takes a nestingPath like "a/b/c" and decomposes it into subpaths: │ │ │ │ │ - * "a", "a/b", "a/b/c" │ │ │ │ │ + * APIMethod: iserror │ │ │ │ │ + * Check to see if the response from the server was an error. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * nPath - {Array} the nesting path │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. If nothing is supplied, │ │ │ │ │ + * the current response is examined. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * Array({String}) Array with subpaths, or empty array if there is nothing │ │ │ │ │ - * to decompose │ │ │ │ │ + * {Boolean} true if the response was an error. │ │ │ │ │ */ │ │ │ │ │ - decomposeNestingPath: function(nPath) { │ │ │ │ │ - var a = []; │ │ │ │ │ - if (OpenLayers.Util.isArray(nPath)) { │ │ │ │ │ - var path = nPath.slice(); │ │ │ │ │ - while (path.length > 0) { │ │ │ │ │ - a.push(path.slice()); │ │ │ │ │ - path.pop(); │ │ │ │ │ - } │ │ │ │ │ - a.reverse(); │ │ │ │ │ + iserror: function(data) { │ │ │ │ │ + var ret = null; │ │ │ │ │ + │ │ │ │ │ + if (!data) { │ │ │ │ │ + ret = (this.response.error !== ''); │ │ │ │ │ + } else { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + var errorNodes = data.documentElement.getElementsByTagName("ERROR"); │ │ │ │ │ + ret = (errorNodes !== null && errorNodes.length > 0); │ │ │ │ │ } │ │ │ │ │ - return a; │ │ │ │ │ + │ │ │ │ │ + return ret; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIMethod: read │ │ │ │ │ - * Read OWS context data from a string or DOMElement, and return a list │ │ │ │ │ - * of layers. │ │ │ │ │ + * Read data from a string, and return an response. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ + * Parameters: │ │ │ │ │ * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} The context object with a flat layer list as a property named │ │ │ │ │ - * layersContext. │ │ │ │ │ + * {<OpenLayers.Format.ArcXML.Response>} An ArcXML response. Note that this response │ │ │ │ │ + * data may change in the future. │ │ │ │ │ */ │ │ │ │ │ read: function(data) { │ │ │ │ │ if (typeof data == "string") { │ │ │ │ │ data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ + │ │ │ │ │ + var arcNode = null; │ │ │ │ │ + if (data && data.documentElement) { │ │ │ │ │ + if (data.documentElement.nodeName == "ARCXML") { │ │ │ │ │ + arcNode = data.documentElement; │ │ │ │ │ + } else { │ │ │ │ │ + arcNode = data.documentElement.getElementsByTagName("ARCXML")[0]; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var context = {}; │ │ │ │ │ - this.readNode(data, context); │ │ │ │ │ - // since an OWSContext can be nested we need to go through this │ │ │ │ │ - // structure recursively │ │ │ │ │ - this.setNestingPath({ │ │ │ │ │ - layersContext: context.layersContext │ │ │ │ │ - }); │ │ │ │ │ - // after nesting path has been set, create a flat list of layers │ │ │ │ │ - var layers = []; │ │ │ │ │ - this.processLayer(layers, context); │ │ │ │ │ - delete context.layersContext; │ │ │ │ │ - context.layersContext = layers; │ │ │ │ │ - return context; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: processLayer │ │ │ │ │ - * Recursive function to get back a flat list of layers from the hierarchic │ │ │ │ │ - * layer structure. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layerArray - {Array({Object})} Array of layerContext objects │ │ │ │ │ - * layer - {Object} layerContext object │ │ │ │ │ - */ │ │ │ │ │ - processLayer: function(layerArray, layer) { │ │ │ │ │ - if (layer.layersContext) { │ │ │ │ │ - for (var i = 0, len = layer.layersContext.length; i < len; i++) { │ │ │ │ │ - var l = layer.layersContext[i]; │ │ │ │ │ - layerArray.push(l); │ │ │ │ │ - if (l.layersContext) { │ │ │ │ │ - this.processLayer(layerArray, l); │ │ │ │ │ - } │ │ │ │ │ + // in Safari, arcNode will be there but will have a child named │ │ │ │ │ + // parsererror │ │ │ │ │ + if (!arcNode || arcNode.firstChild.nodeName === 'parsererror') { │ │ │ │ │ + var error, source; │ │ │ │ │ + try { │ │ │ │ │ + error = data.firstChild.nodeValue; │ │ │ │ │ + source = data.firstChild.childNodes[1].firstChild.nodeValue; │ │ │ │ │ + } catch (err) { │ │ │ │ │ + // pass │ │ │ │ │ } │ │ │ │ │ + throw { │ │ │ │ │ + message: "Error parsing the ArcXML request", │ │ │ │ │ + error: error, │ │ │ │ │ + source: source │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + var response = this.parseResponse(arcNode); │ │ │ │ │ + return response; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIMethod: write │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * context - {Object} An object representing the map context. │ │ │ │ │ - * options - {Object} Optional object. │ │ │ │ │ - * │ │ │ │ │ + * Generate an ArcXml document string for sending to an ArcIMS server. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} An OWS Context document string. │ │ │ │ │ + * {String} A string representing the ArcXML document request. │ │ │ │ │ */ │ │ │ │ │ - write: function(context, options) { │ │ │ │ │ - var name = "OWSContext"; │ │ │ │ │ - this.nestingLayerLookup = {}; //start with empty lookup │ │ │ │ │ - options = options || {}; │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, context); │ │ │ │ │ - var root = this.writeNode(name, options); │ │ │ │ │ - this.nestingLayerLookup = null; //clear lookup │ │ │ │ │ - this.setAttributeNS( │ │ │ │ │ - root, this.namespaces["xsi"], │ │ │ │ │ - "xsi:schemaLocation", this.schemaLocation │ │ │ │ │ - ); │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [root]); │ │ │ │ │ - }, │ │ │ │ │ + write: function(request) { │ │ │ │ │ + if (!request) { │ │ │ │ │ + request = this.request; │ │ │ │ │ + } │ │ │ │ │ + var root = this.createElementNS("", "ARCXML"); │ │ │ │ │ + root.setAttribute("version", "1.1"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "kml": { │ │ │ │ │ - "Document": function(node, obj) { │ │ │ │ │ - obj.features = new OpenLayers.Format.KML({ │ │ │ │ │ - kmlns: this.namespaces.kml, │ │ │ │ │ - extractStyles: true │ │ │ │ │ - }).read(node); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "owc": { │ │ │ │ │ - "OWSContext": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "General": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "ResourceList": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "Layer": function(node, obj) { │ │ │ │ │ - var layerContext = { │ │ │ │ │ - metadata: {}, │ │ │ │ │ - visibility: (node.getAttribute("hidden") != "1"), │ │ │ │ │ - queryable: (node.getAttribute("queryable") == "1"), │ │ │ │ │ - opacity: ((node.getAttribute("opacity") != null) ? │ │ │ │ │ - parseFloat(node.getAttribute("opacity")) : null), │ │ │ │ │ - name: node.getAttribute("name"), │ │ │ │ │ - /* A category layer is a dummy layer meant for creating │ │ │ │ │ - hierarchies. It is not a physical layer in the │ │ │ │ │ - OpenLayers sense. The assumption we make here is that │ │ │ │ │ - category layers do not have a name attribute */ │ │ │ │ │ - categoryLayer: (node.getAttribute("name") == null), │ │ │ │ │ - formats: [], │ │ │ │ │ - styles: [] │ │ │ │ │ - }; │ │ │ │ │ - if (!obj.layersContext) { │ │ │ │ │ - obj.layersContext = []; │ │ │ │ │ - } │ │ │ │ │ - obj.layersContext.push(layerContext); │ │ │ │ │ - this.readChildNodes(node, layerContext); │ │ │ │ │ - }, │ │ │ │ │ - "InlineGeometry": function(node, obj) { │ │ │ │ │ - obj.features = []; │ │ │ │ │ - var elements = this.getElementsByTagNameNS(node, │ │ │ │ │ - this.namespaces.gml, "featureMember"); │ │ │ │ │ - var el; │ │ │ │ │ - if (elements.length >= 1) { │ │ │ │ │ - el = elements[0]; │ │ │ │ │ - } │ │ │ │ │ - if (el && el.firstChild) { │ │ │ │ │ - var featurenode = (el.firstChild.nextSibling) ? │ │ │ │ │ - el.firstChild.nextSibling : el.firstChild; │ │ │ │ │ - this.setNamespace("feature", featurenode.namespaceURI); │ │ │ │ │ - this.featureType = featurenode.localName || │ │ │ │ │ - featurenode.nodeName.split(":").pop(); │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ + var reqElem = this.createElementNS("", "REQUEST"); │ │ │ │ │ + │ │ │ │ │ + if (request.get_image != null) { │ │ │ │ │ + var getElem = this.createElementNS("", "GET_IMAGE"); │ │ │ │ │ + reqElem.appendChild(getElem); │ │ │ │ │ + │ │ │ │ │ + var propElem = this.createElementNS("", "PROPERTIES"); │ │ │ │ │ + getElem.appendChild(propElem); │ │ │ │ │ + │ │ │ │ │ + var props = request.get_image.properties; │ │ │ │ │ + if (props.featurecoordsys != null) { │ │ │ │ │ + var feat = this.createElementNS("", "FEATURECOORDSYS"); │ │ │ │ │ + propElem.appendChild(feat); │ │ │ │ │ + │ │ │ │ │ + if (props.featurecoordsys.id === 0) { │ │ │ │ │ + feat.setAttribute("string", props.featurecoordsys['string']); │ │ │ │ │ + } else { │ │ │ │ │ + feat.setAttribute("id", props.featurecoordsys.id); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "Server": function(node, obj) { │ │ │ │ │ - // when having multiple Server types, we prefer WMS │ │ │ │ │ - if ((!obj.service && !obj.version) || │ │ │ │ │ - (obj.service != │ │ │ │ │ - OpenLayers.Format.Context.serviceTypes.WMS)) { │ │ │ │ │ - obj.service = node.getAttribute("service"); │ │ │ │ │ - obj.version = node.getAttribute("version"); │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (props.filtercoordsys != null) { │ │ │ │ │ + var filt = this.createElementNS("", "FILTERCOORDSYS"); │ │ │ │ │ + propElem.appendChild(filt); │ │ │ │ │ + │ │ │ │ │ + if (props.filtercoordsys.id === 0) { │ │ │ │ │ + filt.setAttribute("string", props.filtercoordsys.string); │ │ │ │ │ + } else { │ │ │ │ │ + filt.setAttribute("id", props.filtercoordsys.id); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "Name": function(node, obj) { │ │ │ │ │ - obj.name = this.getChildValue(node); │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "Title": function(node, obj) { │ │ │ │ │ - obj.title = this.getChildValue(node); │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "StyleList": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj.styles); │ │ │ │ │ - }, │ │ │ │ │ - "Style": function(node, obj) { │ │ │ │ │ - var style = {}; │ │ │ │ │ - obj.push(style); │ │ │ │ │ - this.readChildNodes(node, style); │ │ │ │ │ - }, │ │ │ │ │ - "LegendURL": function(node, obj) { │ │ │ │ │ - var legend = {}; │ │ │ │ │ - obj.legend = legend; │ │ │ │ │ - this.readChildNodes(node, legend); │ │ │ │ │ - }, │ │ │ │ │ - "OnlineResource": function(node, obj) { │ │ │ │ │ - obj.url = this.getAttributeNS(node, this.namespaces.xlink, │ │ │ │ │ - "href"); │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "ows": OpenLayers.Format.OWSCommon.v1_0_0.prototype.readers.ows, │ │ │ │ │ - "gml": OpenLayers.Format.GML.v2.prototype.readers.gml, │ │ │ │ │ - "sld": OpenLayers.Format.SLD.v1_0_0.prototype.readers.sld, │ │ │ │ │ - "feature": OpenLayers.Format.GML.v2.prototype.readers.feature │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ - */ │ │ │ │ │ - writers: { │ │ │ │ │ - "owc": { │ │ │ │ │ - "OWSContext": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("OWSContext", { │ │ │ │ │ - attributes: { │ │ │ │ │ - version: this.VERSION, │ │ │ │ │ - id: options.id || OpenLayers.Util.createUniqueID("OpenLayers_OWSContext_") │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - this.writeNode("General", options, node); │ │ │ │ │ - this.writeNode("ResourceList", options, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "General": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("General"); │ │ │ │ │ - this.writeNode("ows:BoundingBox", options, node); │ │ │ │ │ - this.writeNode("ows:Title", options.title || 'OpenLayers OWSContext', node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "ResourceList": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("ResourceList"); │ │ │ │ │ - for (var i = 0, len = options.layers.length; i < len; i++) { │ │ │ │ │ - var layer = options.layers[i]; │ │ │ │ │ - var decomposedPath = this.decomposeNestingPath(layer.metadata.nestingPath); │ │ │ │ │ - this.writeNode("_Layer", { │ │ │ │ │ - layer: layer, │ │ │ │ │ - subPaths: decomposedPath │ │ │ │ │ - }, node); │ │ │ │ │ + if (props.envelope != null) { │ │ │ │ │ + var env = this.createElementNS("", "ENVELOPE"); │ │ │ │ │ + propElem.appendChild(env); │ │ │ │ │ + │ │ │ │ │ + env.setAttribute("minx", props.envelope.minx); │ │ │ │ │ + env.setAttribute("miny", props.envelope.miny); │ │ │ │ │ + env.setAttribute("maxx", props.envelope.maxx); │ │ │ │ │ + env.setAttribute("maxy", props.envelope.maxy); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var imagesz = this.createElementNS("", "IMAGESIZE"); │ │ │ │ │ + propElem.appendChild(imagesz); │ │ │ │ │ + │ │ │ │ │ + imagesz.setAttribute("height", props.imagesize.height); │ │ │ │ │ + imagesz.setAttribute("width", props.imagesize.width); │ │ │ │ │ + │ │ │ │ │ + if (props.imagesize.height != props.imagesize.printheight || │ │ │ │ │ + props.imagesize.width != props.imagesize.printwidth) { │ │ │ │ │ + imagesz.setAttribute("printheight", props.imagesize.printheight); │ │ │ │ │ + imagesz.setArrtibute("printwidth", props.imagesize.printwidth); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (props.background != null) { │ │ │ │ │ + var backgrnd = this.createElementNS("", "BACKGROUND"); │ │ │ │ │ + propElem.appendChild(backgrnd); │ │ │ │ │ + │ │ │ │ │ + backgrnd.setAttribute("color", │ │ │ │ │ + props.background.color.r + "," + │ │ │ │ │ + props.background.color.g + "," + │ │ │ │ │ + props.background.color.b); │ │ │ │ │ + │ │ │ │ │ + if (props.background.transcolor !== null) { │ │ │ │ │ + backgrnd.setAttribute("transcolor", │ │ │ │ │ + props.background.transcolor.r + "," + │ │ │ │ │ + props.background.transcolor.g + "," + │ │ │ │ │ + props.background.transcolor.b); │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Server": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("Server", { │ │ │ │ │ - attributes: { │ │ │ │ │ - version: options.version, │ │ │ │ │ - service: options.service │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (props.layerlist != null && props.layerlist.length > 0) { │ │ │ │ │ + var layerlst = this.createElementNS("", "LAYERLIST"); │ │ │ │ │ + propElem.appendChild(layerlst); │ │ │ │ │ + │ │ │ │ │ + for (var ld = 0; ld < props.layerlist.length; ld++) { │ │ │ │ │ + var ldef = this.createElementNS("", "LAYERDEF"); │ │ │ │ │ + layerlst.appendChild(ldef); │ │ │ │ │ + │ │ │ │ │ + ldef.setAttribute("id", props.layerlist[ld].id); │ │ │ │ │ + ldef.setAttribute("visible", props.layerlist[ld].visible); │ │ │ │ │ + │ │ │ │ │ + if (typeof props.layerlist[ld].query == "object") { │ │ │ │ │ + var query = props.layerlist[ld].query; │ │ │ │ │ + │ │ │ │ │ + if (query.where.length < 0) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var queryElem = null; │ │ │ │ │ + if (typeof query.spatialfilter == "boolean" && query.spatialfilter) { │ │ │ │ │ + // handle spatial filter madness │ │ │ │ │ + queryElem = this.createElementNS("", "SPATIALQUERY"); │ │ │ │ │ + } else { │ │ │ │ │ + queryElem = this.createElementNS("", "QUERY"); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + queryElem.setAttribute("where", query.where); │ │ │ │ │ + │ │ │ │ │ + if (typeof query.accuracy == "number" && query.accuracy > 0) { │ │ │ │ │ + queryElem.setAttribute("accuracy", query.accuracy); │ │ │ │ │ + } │ │ │ │ │ + if (typeof query.featurelimit == "number" && query.featurelimit < 2000) { │ │ │ │ │ + queryElem.setAttribute("featurelimit", query.featurelimit); │ │ │ │ │ + } │ │ │ │ │ + if (typeof query.subfields == "string" && query.subfields != "#ALL#") { │ │ │ │ │ + queryElem.setAttribute("subfields", query.subfields); │ │ │ │ │ + } │ │ │ │ │ + if (typeof query.joinexpression == "string" && query.joinexpression.length > 0) { │ │ │ │ │ + queryElem.setAttribute("joinexpression", query.joinexpression); │ │ │ │ │ + } │ │ │ │ │ + if (typeof query.jointables == "string" && query.jointables.length > 0) { │ │ │ │ │ + queryElem.setAttribute("jointables", query.jointables); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + ldef.appendChild(queryElem); │ │ │ │ │ } │ │ │ │ │ - }); │ │ │ │ │ - this.writeNode("OnlineResource", options, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "OnlineResource": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("OnlineResource", { │ │ │ │ │ - attributes: { │ │ │ │ │ - "xlink:href": options.url │ │ │ │ │ + │ │ │ │ │ + if (typeof props.layerlist[ld].renderer == "object") { │ │ │ │ │ + this.addRenderer(ldef, props.layerlist[ld].renderer); │ │ │ │ │ } │ │ │ │ │ - }); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "InlineGeometry": function(layer) { │ │ │ │ │ - var node = this.createElementNSPlus("InlineGeometry"), │ │ │ │ │ - dataExtent = layer.getDataExtent(); │ │ │ │ │ - if (dataExtent !== null) { │ │ │ │ │ - this.writeNode("gml:boundedBy", dataExtent, node); │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = layer.features.length; i < len; i++) { │ │ │ │ │ - this.writeNode("gml:featureMember", layer.features[i], node); │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "StyleList": function(styles) { │ │ │ │ │ - var node = this.createElementNSPlus("StyleList"); │ │ │ │ │ - for (var i = 0, len = styles.length; i < len; i++) { │ │ │ │ │ - this.writeNode("Style", styles[i], node); │ │ │ │ │ + } │ │ │ │ │ + } else if (request.get_feature != null) { │ │ │ │ │ + var getElem = this.createElementNS("", "GET_FEATURES"); │ │ │ │ │ + getElem.setAttribute("outputmode", "newxml"); │ │ │ │ │ + getElem.setAttribute("checkesc", "true"); │ │ │ │ │ + │ │ │ │ │ + if (request.get_feature.geometry) { │ │ │ │ │ + getElem.setAttribute("geometry", request.get_feature.geometry); │ │ │ │ │ + } else { │ │ │ │ │ + getElem.setAttribute("geometry", "false"); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (request.get_feature.compact) { │ │ │ │ │ + getElem.setAttribute("compact", request.get_feature.compact); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (request.get_feature.featurelimit == "number") { │ │ │ │ │ + getElem.setAttribute("featurelimit", request.get_feature.featurelimit); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + getElem.setAttribute("globalenvelope", "true"); │ │ │ │ │ + reqElem.appendChild(getElem); │ │ │ │ │ + │ │ │ │ │ + if (request.get_feature.layer != null && request.get_feature.layer.length > 0) { │ │ │ │ │ + var lyrElem = this.createElementNS("", "LAYER"); │ │ │ │ │ + lyrElem.setAttribute("id", request.get_feature.layer); │ │ │ │ │ + getElem.appendChild(lyrElem); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var fquery = request.get_feature.query; │ │ │ │ │ + if (fquery != null) { │ │ │ │ │ + var qElem = null; │ │ │ │ │ + if (fquery.isspatial) { │ │ │ │ │ + qElem = this.createElementNS("", "SPATIALQUERY"); │ │ │ │ │ + } else { │ │ │ │ │ + qElem = this.createElementNS("", "QUERY"); │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Style": function(style) { │ │ │ │ │ - var node = this.createElementNSPlus("Style"); │ │ │ │ │ - this.writeNode("Name", style, node); │ │ │ │ │ - this.writeNode("Title", style, node); │ │ │ │ │ - if (style.legend) { │ │ │ │ │ - this.writeNode("LegendURL", style, node); │ │ │ │ │ + getElem.appendChild(qElem); │ │ │ │ │ + │ │ │ │ │ + if (typeof fquery.accuracy == "number") { │ │ │ │ │ + qElem.setAttribute("accuracy", fquery.accuracy); │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Name": function(obj) { │ │ │ │ │ - var node = this.createElementNSPlus("Name", { │ │ │ │ │ - value: obj.name │ │ │ │ │ - }); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Title": function(obj) { │ │ │ │ │ - var node = this.createElementNSPlus("Title", { │ │ │ │ │ - value: obj.title │ │ │ │ │ - }); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "LegendURL": function(style) { │ │ │ │ │ - var node = this.createElementNSPlus("LegendURL"); │ │ │ │ │ - this.writeNode("OnlineResource", style.legend, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "_WMS": function(layer) { │ │ │ │ │ - var node = this.createElementNSPlus("Layer", { │ │ │ │ │ - attributes: { │ │ │ │ │ - name: layer.params.LAYERS, │ │ │ │ │ - queryable: layer.queryable ? "1" : "0", │ │ │ │ │ - hidden: layer.visibility ? "0" : "1", │ │ │ │ │ - opacity: layer.hasOwnProperty("opacity") ? layer.opacity : null │ │ │ │ │ + //qElem.setAttribute("featurelimit", "5"); │ │ │ │ │ + │ │ │ │ │ + if (fquery.featurecoordsys != null) { │ │ │ │ │ + var fcsElem1 = this.createElementNS("", "FEATURECOORDSYS"); │ │ │ │ │ + │ │ │ │ │ + if (fquery.featurecoordsys.id == 0) { │ │ │ │ │ + fcsElem1.setAttribute("string", fquery.featurecoordsys.string); │ │ │ │ │ + } else { │ │ │ │ │ + fcsElem1.setAttribute("id", fquery.featurecoordsys.id); │ │ │ │ │ } │ │ │ │ │ - }); │ │ │ │ │ - this.writeNode("ows:Title", layer.name, node); │ │ │ │ │ - this.writeNode("ows:OutputFormat", layer.params.FORMAT, node); │ │ │ │ │ - this.writeNode("Server", { │ │ │ │ │ - service: OpenLayers.Format.Context.serviceTypes.WMS, │ │ │ │ │ - version: layer.params.VERSION, │ │ │ │ │ - url: layer.url │ │ │ │ │ - }, node); │ │ │ │ │ - if (layer.metadata.styles && layer.metadata.styles.length > 0) { │ │ │ │ │ - this.writeNode("StyleList", layer.metadata.styles, node); │ │ │ │ │ + qElem.appendChild(fcsElem1); │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "_Layer": function(options) { │ │ │ │ │ - var layer, subPaths, node, title; │ │ │ │ │ - layer = options.layer; │ │ │ │ │ - subPaths = options.subPaths; │ │ │ │ │ - node = null; │ │ │ │ │ - title = null; │ │ │ │ │ - // subPaths is an array of an array │ │ │ │ │ - // recursively calling _Layer writer eats up subPaths, until a │ │ │ │ │ - // real writer is called and nodes are returned. │ │ │ │ │ - if (subPaths.length > 0) { │ │ │ │ │ - var path = subPaths[0].join("/"); │ │ │ │ │ - var index = path.lastIndexOf("/"); │ │ │ │ │ - node = this.nestingLayerLookup[path]; │ │ │ │ │ - title = (index > 0) ? path.substring(index + 1, path.length) : path; │ │ │ │ │ - if (!node) { │ │ │ │ │ - // category layer │ │ │ │ │ - node = this.createElementNSPlus("Layer"); │ │ │ │ │ - this.writeNode("ows:Title", title, node); │ │ │ │ │ - this.nestingLayerLookup[path] = node; │ │ │ │ │ + │ │ │ │ │ + if (fquery.filtercoordsys != null) { │ │ │ │ │ + var fcsElem2 = this.createElementNS("", "FILTERCOORDSYS"); │ │ │ │ │ + │ │ │ │ │ + if (fquery.filtercoordsys.id === 0) { │ │ │ │ │ + fcsElem2.setAttribute("string", fquery.filtercoordsys.string); │ │ │ │ │ + } else { │ │ │ │ │ + fcsElem2.setAttribute("id", fquery.filtercoordsys.id); │ │ │ │ │ } │ │ │ │ │ - options.subPaths.shift(); //remove a path after each call │ │ │ │ │ - this.writeNode("_Layer", options, node); │ │ │ │ │ - return node; │ │ │ │ │ - } else { │ │ │ │ │ - // write out the actual layer │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.WMS) { │ │ │ │ │ - node = this.writeNode("_WMS", layer); │ │ │ │ │ - } else if (layer instanceof OpenLayers.Layer.Vector) { │ │ │ │ │ - if (layer.protocol instanceof OpenLayers.Protocol.WFS.v1) { │ │ │ │ │ - node = this.writeNode("_WFS", layer); │ │ │ │ │ - } else if (layer.protocol instanceof OpenLayers.Protocol.HTTP) { │ │ │ │ │ - if (layer.protocol.format instanceof OpenLayers.Format.GML) { │ │ │ │ │ - layer.protocol.format.version = "2.1.2"; │ │ │ │ │ - node = this.writeNode("_GML", layer); │ │ │ │ │ - } else if (layer.protocol.format instanceof OpenLayers.Format.KML) { │ │ │ │ │ - layer.protocol.format.version = "2.2"; │ │ │ │ │ - node = this.writeNode("_KML", layer); │ │ │ │ │ + qElem.appendChild(fcsElem2); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (fquery.buffer > 0) { │ │ │ │ │ + var bufElem = this.createElementNS("", "BUFFER"); │ │ │ │ │ + bufElem.setAttribute("distance", fquery.buffer); │ │ │ │ │ + qElem.appendChild(bufElem); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (fquery.isspatial) { │ │ │ │ │ + var spfElem = this.createElementNS("", "SPATIALFILTER"); │ │ │ │ │ + spfElem.setAttribute("relation", fquery.spatialfilter.relation); │ │ │ │ │ + qElem.appendChild(spfElem); │ │ │ │ │ + │ │ │ │ │ + if (fquery.spatialfilter.envelope) { │ │ │ │ │ + var envElem = this.createElementNS("", "ENVELOPE"); │ │ │ │ │ + envElem.setAttribute("minx", fquery.spatialfilter.envelope.minx); │ │ │ │ │ + envElem.setAttribute("miny", fquery.spatialfilter.envelope.miny); │ │ │ │ │ + envElem.setAttribute("maxx", fquery.spatialfilter.envelope.maxx); │ │ │ │ │ + envElem.setAttribute("maxy", fquery.spatialfilter.envelope.maxy); │ │ │ │ │ + spfElem.appendChild(envElem); │ │ │ │ │ + } else if (typeof fquery.spatialfilter.polygon == "object") { │ │ │ │ │ + spfElem.appendChild(this.writePolygonGeometry(fquery.spatialfilter.polygon)); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (fquery.where != null && fquery.where.length > 0) { │ │ │ │ │ + qElem.setAttribute("where", fquery.where); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + root.appendChild(reqElem); │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [root]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + addGroupRenderer: function(ldef, toprenderer) { │ │ │ │ │ + var topRelem = this.createElementNS("", "GROUPRENDERER"); │ │ │ │ │ + ldef.appendChild(topRelem); │ │ │ │ │ + │ │ │ │ │ + for (var rind = 0; rind < toprenderer.length; rind++) { │ │ │ │ │ + var renderer = toprenderer[rind]; │ │ │ │ │ + this.addRenderer(topRelem, renderer); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + addRenderer: function(topRelem, renderer) { │ │ │ │ │ + if (OpenLayers.Util.isArray(renderer)) { │ │ │ │ │ + this.addGroupRenderer(topRelem, renderer); │ │ │ │ │ + } else { │ │ │ │ │ + var renderElem = this.createElementNS("", renderer.type.toUpperCase() + "RENDERER"); │ │ │ │ │ + topRelem.appendChild(renderElem); │ │ │ │ │ + │ │ │ │ │ + if (renderElem.tagName == "VALUEMAPRENDERER") { │ │ │ │ │ + this.addValueMapRenderer(renderElem, renderer); │ │ │ │ │ + } else if (renderElem.tagName == "VALUEMAPLABELRENDERER") { │ │ │ │ │ + this.addValueMapLabelRenderer(renderElem, renderer); │ │ │ │ │ + } else if (renderElem.tagName == "SIMPLELABELRENDERER") { │ │ │ │ │ + this.addSimpleLabelRenderer(renderElem, renderer); │ │ │ │ │ + } else if (renderElem.tagName == "SCALEDEPENDENTRENDERER") { │ │ │ │ │ + this.addScaleDependentRenderer(renderElem, renderer); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + addScaleDependentRenderer: function(renderElem, renderer) { │ │ │ │ │ + if (typeof renderer.lower == "string" || typeof renderer.lower == "number") { │ │ │ │ │ + renderElem.setAttribute("lower", renderer.lower); │ │ │ │ │ + } │ │ │ │ │ + if (typeof renderer.upper == "string" || typeof renderer.upper == "number") { │ │ │ │ │ + renderElem.setAttribute("upper", renderer.upper); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.addRenderer(renderElem, renderer.renderer); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + addValueMapLabelRenderer: function(renderElem, renderer) { │ │ │ │ │ + renderElem.setAttribute("lookupfield", renderer.lookupfield); │ │ │ │ │ + renderElem.setAttribute("labelfield", renderer.labelfield); │ │ │ │ │ + │ │ │ │ │ + if (typeof renderer.exacts == "object") { │ │ │ │ │ + for (var ext = 0, extlen = renderer.exacts.length; ext < extlen; ext++) { │ │ │ │ │ + var exact = renderer.exacts[ext]; │ │ │ │ │ + │ │ │ │ │ + var eelem = this.createElementNS("", "EXACT"); │ │ │ │ │ + │ │ │ │ │ + if (typeof exact.value == "string") { │ │ │ │ │ + eelem.setAttribute("value", exact.value); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.label == "string") { │ │ │ │ │ + eelem.setAttribute("label", exact.label); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.method == "string") { │ │ │ │ │ + eelem.setAttribute("method", exact.method); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + renderElem.appendChild(eelem); │ │ │ │ │ + │ │ │ │ │ + if (typeof exact.symbol == "object") { │ │ │ │ │ + var selem = null; │ │ │ │ │ + │ │ │ │ │ + if (exact.symbol.type == "text") { │ │ │ │ │ + selem = this.createElementNS("", "TEXTSYMBOL"); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (selem != null) { │ │ │ │ │ + var keys = this.fontStyleKeys; │ │ │ │ │ + for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ + var key = keys[i]; │ │ │ │ │ + if (exact.symbol[key]) { │ │ │ │ │ + selem.setAttribute(key, exact.symbol[key]); │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - // write out as inline GML since we have no idea │ │ │ │ │ - // about the original Format │ │ │ │ │ - this.setNamespace("feature", this.featureNS); │ │ │ │ │ - node = this.writeNode("_InlineGeometry", layer); │ │ │ │ │ } │ │ │ │ │ + eelem.appendChild(selem); │ │ │ │ │ } │ │ │ │ │ - if (layer.options.maxScale) { │ │ │ │ │ - this.writeNode("sld:MinScaleDenominator", │ │ │ │ │ - layer.options.maxScale, node); │ │ │ │ │ + } │ │ │ │ │ + } // for each exact │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + addValueMapRenderer: function(renderElem, renderer) { │ │ │ │ │ + renderElem.setAttribute("lookupfield", renderer.lookupfield); │ │ │ │ │ + │ │ │ │ │ + if (typeof renderer.ranges == "object") { │ │ │ │ │ + for (var rng = 0, rnglen = renderer.ranges.length; rng < rnglen; rng++) { │ │ │ │ │ + var range = renderer.ranges[rng]; │ │ │ │ │ + │ │ │ │ │ + var relem = this.createElementNS("", "RANGE"); │ │ │ │ │ + relem.setAttribute("lower", range.lower); │ │ │ │ │ + relem.setAttribute("upper", range.upper); │ │ │ │ │ + │ │ │ │ │ + renderElem.appendChild(relem); │ │ │ │ │ + │ │ │ │ │ + if (typeof range.symbol == "object") { │ │ │ │ │ + var selem = null; │ │ │ │ │ + │ │ │ │ │ + if (range.symbol.type == "simplepolygon") { │ │ │ │ │ + selem = this.createElementNS("", "SIMPLEPOLYGONSYMBOL"); │ │ │ │ │ } │ │ │ │ │ - if (layer.options.minScale) { │ │ │ │ │ - this.writeNode("sld:MaxScaleDenominator", │ │ │ │ │ - layer.options.minScale, node); │ │ │ │ │ + │ │ │ │ │ + if (selem != null) { │ │ │ │ │ + if (typeof range.symbol.boundarycolor == "string") { │ │ │ │ │ + selem.setAttribute("boundarycolor", range.symbol.boundarycolor); │ │ │ │ │ + } │ │ │ │ │ + if (typeof range.symbol.fillcolor == "string") { │ │ │ │ │ + selem.setAttribute("fillcolor", range.symbol.fillcolor); │ │ │ │ │ + } │ │ │ │ │ + if (typeof range.symbol.filltransparency == "number") { │ │ │ │ │ + selem.setAttribute("filltransparency", range.symbol.filltransparency); │ │ │ │ │ + } │ │ │ │ │ + relem.appendChild(selem); │ │ │ │ │ } │ │ │ │ │ - this.nestingLayerLookup[layer.name] = node; │ │ │ │ │ - return node; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "_WFS": function(layer) { │ │ │ │ │ - var node = this.createElementNSPlus("Layer", { │ │ │ │ │ - attributes: { │ │ │ │ │ - name: layer.protocol.featurePrefix + ":" + layer.protocol.featureType, │ │ │ │ │ - hidden: layer.visibility ? "0" : "1" │ │ │ │ │ + } // for each range │ │ │ │ │ + } else if (typeof renderer.exacts == "object") { │ │ │ │ │ + for (var ext = 0, extlen = renderer.exacts.length; ext < extlen; ext++) { │ │ │ │ │ + var exact = renderer.exacts[ext]; │ │ │ │ │ + │ │ │ │ │ + var eelem = this.createElementNS("", "EXACT"); │ │ │ │ │ + if (typeof exact.value == "string") { │ │ │ │ │ + eelem.setAttribute("value", exact.value); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.label == "string") { │ │ │ │ │ + eelem.setAttribute("label", exact.label); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.method == "string") { │ │ │ │ │ + eelem.setAttribute("method", exact.method); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + renderElem.appendChild(eelem); │ │ │ │ │ + │ │ │ │ │ + if (typeof exact.symbol == "object") { │ │ │ │ │ + var selem = null; │ │ │ │ │ + │ │ │ │ │ + if (exact.symbol.type == "simplemarker") { │ │ │ │ │ + selem = this.createElementNS("", "SIMPLEMARKERSYMBOL"); │ │ │ │ │ } │ │ │ │ │ - }); │ │ │ │ │ - this.writeNode("ows:Title", layer.name, node); │ │ │ │ │ - this.writeNode("Server", { │ │ │ │ │ - service: OpenLayers.Format.Context.serviceTypes.WFS, │ │ │ │ │ - version: layer.protocol.version, │ │ │ │ │ - url: layer.protocol.url │ │ │ │ │ - }, node); │ │ │ │ │ - return node; │ │ │ │ │ + │ │ │ │ │ + if (selem != null) { │ │ │ │ │ + if (typeof exact.symbol.antialiasing == "string") { │ │ │ │ │ + selem.setAttribute("antialiasing", exact.symbol.antialiasing); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.color == "string") { │ │ │ │ │ + selem.setAttribute("color", exact.symbol.color); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.outline == "string") { │ │ │ │ │ + selem.setAttribute("outline", exact.symbol.outline); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.overlap == "string") { │ │ │ │ │ + selem.setAttribute("overlap", exact.symbol.overlap); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.shadow == "string") { │ │ │ │ │ + selem.setAttribute("shadow", exact.symbol.shadow); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.transparency == "number") { │ │ │ │ │ + selem.setAttribute("transparency", exact.symbol.transparency); │ │ │ │ │ + } │ │ │ │ │ + //if (typeof exact.symbol.type == "string") │ │ │ │ │ + // selem.setAttribute("type", exact.symbol.type); │ │ │ │ │ + if (typeof exact.symbol.usecentroid == "string") { │ │ │ │ │ + selem.setAttribute("usecentroid", exact.symbol.usecentroid); │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.width == "number") { │ │ │ │ │ + selem.setAttribute("width", exact.symbol.width); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + eelem.appendChild(selem); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } // for each exact │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + addSimpleLabelRenderer: function(renderElem, renderer) { │ │ │ │ │ + renderElem.setAttribute("field", renderer.field); │ │ │ │ │ + var keys = ['featureweight', 'howmanylabels', 'labelbufferratio', │ │ │ │ │ + 'labelpriorities', 'labelweight', 'linelabelposition', │ │ │ │ │ + 'rotationalangles' │ │ │ │ │ + ]; │ │ │ │ │ + for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ + var key = keys[i]; │ │ │ │ │ + if (renderer[key]) { │ │ │ │ │ + renderElem.setAttribute(key, renderer[key]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (renderer.symbol.type == "text") { │ │ │ │ │ + var symbol = renderer.symbol; │ │ │ │ │ + var selem = this.createElementNS("", "TEXTSYMBOL"); │ │ │ │ │ + renderElem.appendChild(selem); │ │ │ │ │ + │ │ │ │ │ + var keys = this.fontStyleKeys; │ │ │ │ │ + for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ + var key = keys[i]; │ │ │ │ │ + if (symbol[key]) { │ │ │ │ │ + selem.setAttribute(key, renderer[key]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + writePolygonGeometry: function(polygon) { │ │ │ │ │ + if (!(polygon instanceof OpenLayers.Geometry.Polygon)) { │ │ │ │ │ + throw { │ │ │ │ │ + message: 'Cannot write polygon geometry to ArcXML with an ' + │ │ │ │ │ + polygon.CLASS_NAME + ' object.', │ │ │ │ │ + geometry: polygon │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var polyElem = this.createElementNS("", "POLYGON"); │ │ │ │ │ + │ │ │ │ │ + for (var ln = 0, lnlen = polygon.components.length; ln < lnlen; ln++) { │ │ │ │ │ + var ring = polygon.components[ln]; │ │ │ │ │ + var ringElem = this.createElementNS("", "RING"); │ │ │ │ │ + │ │ │ │ │ + for (var rn = 0, rnlen = ring.components.length; rn < rnlen; rn++) { │ │ │ │ │ + var point = ring.components[rn]; │ │ │ │ │ + var pointElem = this.createElementNS("", "POINT"); │ │ │ │ │ + │ │ │ │ │ + pointElem.setAttribute("x", point.x); │ │ │ │ │ + pointElem.setAttribute("y", point.y); │ │ │ │ │ + │ │ │ │ │ + ringElem.appendChild(pointElem); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + polyElem.appendChild(ringElem); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return polyElem; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseResponse │ │ │ │ │ + * Take an ArcXML response, and parse in into this object's internal properties. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} The ArcXML response, as either a string or the │ │ │ │ │ + * top level DOMElement of the response. │ │ │ │ │ + */ │ │ │ │ │ + parseResponse: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + var newData = new OpenLayers.Format.XML(); │ │ │ │ │ + data = newData.read(data); │ │ │ │ │ + } │ │ │ │ │ + var response = new OpenLayers.Format.ArcXML.Response(); │ │ │ │ │ + │ │ │ │ │ + var errorNode = data.getElementsByTagName("ERROR"); │ │ │ │ │ + │ │ │ │ │ + if (errorNode != null && errorNode.length > 0) { │ │ │ │ │ + response.error = this.getChildValue(errorNode, "Unknown error."); │ │ │ │ │ + } else { │ │ │ │ │ + var responseNode = data.getElementsByTagName("RESPONSE"); │ │ │ │ │ + │ │ │ │ │ + if (responseNode == null || responseNode.length == 0) { │ │ │ │ │ + response.error = "No RESPONSE tag found in ArcXML response."; │ │ │ │ │ + return response; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var rtype = responseNode[0].firstChild.nodeName; │ │ │ │ │ + if (rtype == "#text") { │ │ │ │ │ + rtype = responseNode[0].firstChild.nextSibling.nodeName; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (rtype == "IMAGE") { │ │ │ │ │ + var envelopeNode = data.getElementsByTagName("ENVELOPE"); │ │ │ │ │ + var outputNode = data.getElementsByTagName("OUTPUT"); │ │ │ │ │ + │ │ │ │ │ + if (envelopeNode == null || envelopeNode.length == 0) { │ │ │ │ │ + response.error = "No ENVELOPE tag found in ArcXML response."; │ │ │ │ │ + } else if (outputNode == null || outputNode.length == 0) { │ │ │ │ │ + response.error = "No OUTPUT tag found in ArcXML response."; │ │ │ │ │ + } else { │ │ │ │ │ + var envAttr = this.parseAttributes(envelopeNode[0]); │ │ │ │ │ + var outputAttr = this.parseAttributes(outputNode[0]); │ │ │ │ │ + │ │ │ │ │ + if (typeof outputAttr.type == "string") { │ │ │ │ │ + response.image = { │ │ │ │ │ + envelope: envAttr, │ │ │ │ │ + output: { │ │ │ │ │ + type: outputAttr.type, │ │ │ │ │ + data: this.getChildValue(outputNode[0]) │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + } else { │ │ │ │ │ + response.image = { │ │ │ │ │ + envelope: envAttr, │ │ │ │ │ + output: outputAttr │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else if (rtype == "FEATURES") { │ │ │ │ │ + var features = responseNode[0].getElementsByTagName("FEATURES"); │ │ │ │ │ + │ │ │ │ │ + // get the feature count │ │ │ │ │ + var featureCount = features[0].getElementsByTagName("FEATURECOUNT"); │ │ │ │ │ + response.features.featurecount = featureCount[0].getAttribute("count"); │ │ │ │ │ + │ │ │ │ │ + if (response.features.featurecount > 0) { │ │ │ │ │ + // get the feature envelope │ │ │ │ │ + var envelope = features[0].getElementsByTagName("ENVELOPE"); │ │ │ │ │ + response.features.envelope = this.parseAttributes(envelope[0], typeof(0)); │ │ │ │ │ + │ │ │ │ │ + // get the field values per feature │ │ │ │ │ + var featureList = features[0].getElementsByTagName("FEATURE"); │ │ │ │ │ + for (var fn = 0; fn < featureList.length; fn++) { │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ + var fields = featureList[fn].getElementsByTagName("FIELD"); │ │ │ │ │ + │ │ │ │ │ + for (var fdn = 0; fdn < fields.length; fdn++) { │ │ │ │ │ + var fieldName = fields[fdn].getAttribute("name"); │ │ │ │ │ + var fieldValue = fields[fdn].getAttribute("value"); │ │ │ │ │ + feature.attributes[fieldName] = fieldValue; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var geom = featureList[fn].getElementsByTagName("POLYGON"); │ │ │ │ │ + │ │ │ │ │ + if (geom.length > 0) { │ │ │ │ │ + // if there is a polygon, create an openlayers polygon, and assign │ │ │ │ │ + // it to the .geometry property of the feature │ │ │ │ │ + var ring = geom[0].getElementsByTagName("RING"); │ │ │ │ │ + │ │ │ │ │ + var polys = []; │ │ │ │ │ + for (var rn = 0; rn < ring.length; rn++) { │ │ │ │ │ + var linearRings = []; │ │ │ │ │ + linearRings.push(this.parsePointGeometry(ring[rn])); │ │ │ │ │ + │ │ │ │ │ + var holes = ring[rn].getElementsByTagName("HOLE"); │ │ │ │ │ + for (var hn = 0; hn < holes.length; hn++) { │ │ │ │ │ + linearRings.push(this.parsePointGeometry(holes[hn])); │ │ │ │ │ + } │ │ │ │ │ + holes = null; │ │ │ │ │ + polys.push(new OpenLayers.Geometry.Polygon(linearRings)); │ │ │ │ │ + linearRings = null; │ │ │ │ │ + } │ │ │ │ │ + ring = null; │ │ │ │ │ + │ │ │ │ │ + if (polys.length == 1) { │ │ │ │ │ + feature.geometry = polys[0]; │ │ │ │ │ + } else { │ │ │ │ │ + feature.geometry = new OpenLayers.Geometry.MultiPolygon(polys); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + response.features.feature.push(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + response.error = "Unidentified response type."; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return response; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseAttributes │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {<DOMElement>} An element to parse attributes from. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An attributes object, with properties set to attribute values. │ │ │ │ │ + */ │ │ │ │ │ + parseAttributes: function(node, type) { │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + for (var attr = 0; attr < node.attributes.length; attr++) { │ │ │ │ │ + if (type == "number") { │ │ │ │ │ + attributes[node.attributes[attr].nodeName] = parseFloat(node.attributes[attr].nodeValue); │ │ │ │ │ + } else { │ │ │ │ │ + attributes[node.attributes[attr].nodeName] = node.attributes[attr].nodeValue; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return attributes; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parsePointGeometry │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {<DOMElement>} An element to parse <COORDS> or <POINT> arcxml data from. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.LinearRing>} A linear ring represented by the node's points. │ │ │ │ │ + */ │ │ │ │ │ + parsePointGeometry: function(node) { │ │ │ │ │ + var ringPoints = []; │ │ │ │ │ + var coords = node.getElementsByTagName("COORDS"); │ │ │ │ │ + │ │ │ │ │ + if (coords.length > 0) { │ │ │ │ │ + // if coords is present, it's the only coords item │ │ │ │ │ + var coordArr = this.getChildValue(coords[0]); │ │ │ │ │ + coordArr = coordArr.split(/;/); │ │ │ │ │ + for (var cn = 0; cn < coordArr.length; cn++) { │ │ │ │ │ + var coordItems = coordArr[cn].split(/ /); │ │ │ │ │ + ringPoints.push(new OpenLayers.Geometry.Point(coordItems[0], coordItems[1])); │ │ │ │ │ + } │ │ │ │ │ + coords = null; │ │ │ │ │ + } else { │ │ │ │ │ + var point = node.getElementsByTagName("POINT"); │ │ │ │ │ + if (point.length > 0) { │ │ │ │ │ + for (var pn = 0; pn < point.length; pn++) { │ │ │ │ │ + ringPoints.push( │ │ │ │ │ + new OpenLayers.Geometry.Point( │ │ │ │ │ + parseFloat(point[pn].getAttribute("x")), │ │ │ │ │ + parseFloat(point[pn].getAttribute("y")) │ │ │ │ │ + ) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + point = null; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return new OpenLayers.Geometry.LinearRing(ringPoints); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.ArcXML" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Format.ArcXML.Request = OpenLayers.Class({ │ │ │ │ │ + initialize: function(params) { │ │ │ │ │ + var defaults = { │ │ │ │ │ + get_image: { │ │ │ │ │ + properties: { │ │ │ │ │ + background: null, │ │ │ │ │ + /*{ │ │ │ │ │ + color: { r:255, g:255, b:255 }, │ │ │ │ │ + transcolor: null │ │ │ │ │ + },*/ │ │ │ │ │ + draw: true, │ │ │ │ │ + envelope: { │ │ │ │ │ + minx: 0, │ │ │ │ │ + miny: 0, │ │ │ │ │ + maxx: 0, │ │ │ │ │ + maxy: 0 │ │ │ │ │ + }, │ │ │ │ │ + featurecoordsys: { │ │ │ │ │ + id: 0, │ │ │ │ │ + string: "", │ │ │ │ │ + datumtransformid: 0, │ │ │ │ │ + datumtransformstring: "" │ │ │ │ │ + }, │ │ │ │ │ + filtercoordsys: { │ │ │ │ │ + id: 0, │ │ │ │ │ + string: "", │ │ │ │ │ + datumtransformid: 0, │ │ │ │ │ + datumtransformstring: "" │ │ │ │ │ + }, │ │ │ │ │ + imagesize: { │ │ │ │ │ + height: 0, │ │ │ │ │ + width: 0, │ │ │ │ │ + dpi: 96, │ │ │ │ │ + printheight: 0, │ │ │ │ │ + printwidth: 0, │ │ │ │ │ + scalesymbols: false │ │ │ │ │ + }, │ │ │ │ │ + layerlist: [], │ │ │ │ │ + /* no support for legends */ │ │ │ │ │ + output: { │ │ │ │ │ + baseurl: "", │ │ │ │ │ + legendbaseurl: "", │ │ │ │ │ + legendname: "", │ │ │ │ │ + legendpath: "", │ │ │ │ │ + legendurl: "", │ │ │ │ │ + name: "", │ │ │ │ │ + path: "", │ │ │ │ │ + type: "jpg", │ │ │ │ │ + url: "" │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - "_InlineGeometry": function(layer) { │ │ │ │ │ - var node = this.createElementNSPlus("Layer", { │ │ │ │ │ - attributes: { │ │ │ │ │ - name: this.featureType, │ │ │ │ │ - hidden: layer.visibility ? "0" : "1" │ │ │ │ │ + │ │ │ │ │ + get_feature: { │ │ │ │ │ + layer: "", │ │ │ │ │ + query: { │ │ │ │ │ + isspatial: false, │ │ │ │ │ + featurecoordsys: { │ │ │ │ │ + id: 0, │ │ │ │ │ + string: "", │ │ │ │ │ + datumtransformid: 0, │ │ │ │ │ + datumtransformstring: "" │ │ │ │ │ + }, │ │ │ │ │ + filtercoordsys: { │ │ │ │ │ + id: 0, │ │ │ │ │ + string: "", │ │ │ │ │ + datumtransformid: 0, │ │ │ │ │ + datumtransformstring: "" │ │ │ │ │ + }, │ │ │ │ │ + buffer: 0, │ │ │ │ │ + where: "", │ │ │ │ │ + spatialfilter: { │ │ │ │ │ + relation: "envelope_intersection", │ │ │ │ │ + envelope: null │ │ │ │ │ } │ │ │ │ │ - }); │ │ │ │ │ - this.writeNode("ows:Title", layer.name, node); │ │ │ │ │ - this.writeNode("InlineGeometry", layer, node); │ │ │ │ │ - return node; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - "_GML": function(layer) { │ │ │ │ │ - var node = this.createElementNSPlus("Layer"); │ │ │ │ │ - this.writeNode("ows:Title", layer.name, node); │ │ │ │ │ - this.writeNode("Server", { │ │ │ │ │ - service: OpenLayers.Format.Context.serviceTypes.GML, │ │ │ │ │ - url: layer.protocol.url, │ │ │ │ │ - version: layer.protocol.format.version │ │ │ │ │ - }, node); │ │ │ │ │ - return node; │ │ │ │ │ + │ │ │ │ │ + environment: { │ │ │ │ │ + separators: { │ │ │ │ │ + cs: " ", │ │ │ │ │ + ts: ";" │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - "_KML": function(layer) { │ │ │ │ │ - var node = this.createElementNSPlus("Layer"); │ │ │ │ │ - this.writeNode("ows:Title", layer.name, node); │ │ │ │ │ - this.writeNode("Server", { │ │ │ │ │ - service: OpenLayers.Format.Context.serviceTypes.KML, │ │ │ │ │ - version: layer.protocol.format.version, │ │ │ │ │ - url: layer.protocol.url │ │ │ │ │ - }, node); │ │ │ │ │ - return node; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "boundedBy": function(bounds) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:boundedBy"); │ │ │ │ │ - this.writeNode("gml:Box", bounds, node); │ │ │ │ │ - return node; │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.GML.v2.prototype.writers.gml), │ │ │ │ │ - "ows": OpenLayers.Format.OWSCommon.v1_0_0.prototype.writers.ows, │ │ │ │ │ - "sld": OpenLayers.Format.SLD.v1_0_0.prototype.writers.sld, │ │ │ │ │ - "feature": OpenLayers.Format.GML.v2.prototype.writers.feature │ │ │ │ │ + │ │ │ │ │ + layer: [], │ │ │ │ │ + workspaces: [] │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Util.extend(this, defaults); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.OWSContext.v0_3_1" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.ArcXML.Request" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Format.ArcXML.Response = OpenLayers.Class({ │ │ │ │ │ + initialize: function(params) { │ │ │ │ │ + var defaults = { │ │ │ │ │ + image: { │ │ │ │ │ + envelope: null, │ │ │ │ │ + output: '' │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + features: { │ │ │ │ │ + featurecount: 0, │ │ │ │ │ + envelope: null, │ │ │ │ │ + feature: [] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + error: '' │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Util.extend(this, defaults); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.ArcXML.Response" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/XLS/v1.js │ │ │ │ │ + OpenLayers/Format/Text.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XLS.js │ │ │ │ │ - * @requires OpenLayers/Format/GML/v3.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.XLS.v1 │ │ │ │ │ - * Superclass for XLS version 1 parsers. Only supports GeocodeRequest for now. │ │ │ │ │ + * Class: OpenLayers.Format.Text │ │ │ │ │ + * Read Text format. Create a new instance with the <OpenLayers.Format.Text> │ │ │ │ │ + * constructor. This reads text which is formatted like CSV text, using │ │ │ │ │ + * tabs as the seperator by default. It provides parsing of data originally │ │ │ │ │ + * used in the MapViewerService, described on the wiki. This Format is used │ │ │ │ │ + * by the <OpenLayers.Layer.Text> class. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * - <OpenLayers.Format> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.XLS.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Format.Text = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + * APIProperty: defaultStyle │ │ │ │ │ + * defaultStyle allows one to control the default styling of the features. │ │ │ │ │ + * It should be a symbolizer hash. By default, this is set to match the │ │ │ │ │ + * Layer.Text behavior, which is to use the default OpenLayers Icon. │ │ │ │ │ */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - xls: "http://www.opengis.net/xls", │ │ │ │ │ - gml: "http://www.opengis.net/gml", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ - }, │ │ │ │ │ + defaultStyle: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ + * APIProperty: extractStyles │ │ │ │ │ + * set to true to extract styles from the TSV files, using information │ │ │ │ │ + * from the image or icon, iconSize and iconOffset fields. This will result │ │ │ │ │ + * in features with a symbolizer (style) property set, using the │ │ │ │ │ + * default symbolizer specified in <defaultStyle>. Set to false if you │ │ │ │ │ + * wish to use a styleMap or OpenLayers.Style options to style your │ │ │ │ │ + * layer instead. │ │ │ │ │ */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ + extractStyles: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.Text │ │ │ │ │ + * Create a new parser for TSV Text. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + │ │ │ │ │ + if (options.extractStyles !== false) { │ │ │ │ │ + options.defaultStyle = { │ │ │ │ │ + 'externalGraphic': OpenLayers.Util.getImageLocation("marker.png"), │ │ │ │ │ + 'graphicWidth': 21, │ │ │ │ │ + 'graphicHeight': 25, │ │ │ │ │ + 'graphicXOffset': -10.5, │ │ │ │ │ + 'graphicYOffset': -12.5 │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Format.prototype.initialize.apply(this, [options]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: xy │ │ │ │ │ - * {Boolean} Order of the GML coordinate true:(x,y) or false:(y,x) │ │ │ │ │ - * Changing is not recommended, a new Format should be instantiated. │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Return a list of features from a Tab Seperated Values text string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * text - {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ */ │ │ │ │ │ - xy: true, │ │ │ │ │ + read: function(text) { │ │ │ │ │ + var lines = text.split('\n'); │ │ │ │ │ + var columns; │ │ │ │ │ + var features = []; │ │ │ │ │ + // length - 1 to allow for trailing new line │ │ │ │ │ + for (var lcv = 0; lcv < (lines.length - 1); lcv++) { │ │ │ │ │ + var currLine = lines[lcv].replace(/^\s*/, '').replace(/\s*$/, ''); │ │ │ │ │ + │ │ │ │ │ + if (currLine.charAt(0) != '#') { │ │ │ │ │ + /* not a comment */ │ │ │ │ │ + │ │ │ │ │ + if (!columns) { │ │ │ │ │ + //First line is columns │ │ │ │ │ + columns = currLine.split('\t'); │ │ │ │ │ + } else { │ │ │ │ │ + var vals = currLine.split('\t'); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point(0, 0); │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + var style = this.defaultStyle ? │ │ │ │ │ + OpenLayers.Util.applyDefaults({}, this.defaultStyle) : │ │ │ │ │ + null; │ │ │ │ │ + var icon, iconSize, iconOffset, overflow; │ │ │ │ │ + var set = false; │ │ │ │ │ + for (var valIndex = 0; valIndex < vals.length; valIndex++) { │ │ │ │ │ + if (vals[valIndex]) { │ │ │ │ │ + if (columns[valIndex] == 'point') { │ │ │ │ │ + var coords = vals[valIndex].split(','); │ │ │ │ │ + geometry.y = parseFloat(coords[0]); │ │ │ │ │ + geometry.x = parseFloat(coords[1]); │ │ │ │ │ + set = true; │ │ │ │ │ + } else if (columns[valIndex] == 'lat') { │ │ │ │ │ + geometry.y = parseFloat(vals[valIndex]); │ │ │ │ │ + set = true; │ │ │ │ │ + } else if (columns[valIndex] == 'lon') { │ │ │ │ │ + geometry.x = parseFloat(vals[valIndex]); │ │ │ │ │ + set = true; │ │ │ │ │ + } else if (columns[valIndex] == 'title') │ │ │ │ │ + attributes['title'] = vals[valIndex]; │ │ │ │ │ + else if (columns[valIndex] == 'image' || │ │ │ │ │ + columns[valIndex] == 'icon' && style) { │ │ │ │ │ + style['externalGraphic'] = vals[valIndex]; │ │ │ │ │ + } else if (columns[valIndex] == 'iconSize' && style) { │ │ │ │ │ + var size = vals[valIndex].split(','); │ │ │ │ │ + style['graphicWidth'] = parseFloat(size[0]); │ │ │ │ │ + style['graphicHeight'] = parseFloat(size[1]); │ │ │ │ │ + } else if (columns[valIndex] == 'iconOffset' && style) { │ │ │ │ │ + var offset = vals[valIndex].split(','); │ │ │ │ │ + style['graphicXOffset'] = parseFloat(offset[0]); │ │ │ │ │ + style['graphicYOffset'] = parseFloat(offset[1]); │ │ │ │ │ + } else if (columns[valIndex] == 'description') { │ │ │ │ │ + attributes['description'] = vals[valIndex]; │ │ │ │ │ + } else if (columns[valIndex] == 'overflow') { │ │ │ │ │ + attributes['overflow'] = vals[valIndex]; │ │ │ │ │ + } else { │ │ │ │ │ + // For StyleMap filtering, allow additional │ │ │ │ │ + // columns to be stored as attributes. │ │ │ │ │ + attributes[columns[valIndex]] = vals[valIndex]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (set) { │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry.transform(this.externalProjection, │ │ │ │ │ + this.internalProjection); │ │ │ │ │ + } │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry, attributes, style); │ │ │ │ │ + features.push(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return features; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.Text" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/Context.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.Context │ │ │ │ │ + * Base class for both Format.WMC and Format.OWSContext │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.Context = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ + * Property: layerOptions │ │ │ │ │ + * {Object} Default options for layers created by the parser. These │ │ │ │ │ + * options are overridden by the options which are read from the │ │ │ │ │ + * capabilities document. │ │ │ │ │ */ │ │ │ │ │ - defaultPrefix: "xls", │ │ │ │ │ + layerOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} Schema location for a particular minor version. │ │ │ │ │ + * Property: layerParams │ │ │ │ │ + * {Object} Default parameters for layers created by the parser. This │ │ │ │ │ + * can be used e.g. to override DEFAULT_PARAMS for │ │ │ │ │ + * OpenLayers.Layer.WMS. │ │ │ │ │ */ │ │ │ │ │ - schemaLocation: null, │ │ │ │ │ + layerParams: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.XLS.v1 │ │ │ │ │ - * Instances of this class are not created directly. Use the │ │ │ │ │ - * <OpenLayers.Format.XLS> constructor instead. │ │ │ │ │ + * Constructor: OpenLayers.Format.Context │ │ │ │ │ + * Create a new parser for Context documents. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * options - {Object} An optional object whose properties will be set on │ │ │ │ │ * this instance. │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read Context data from a string, and return an object with map │ │ │ │ │ + * properties and a list of layers. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {DOMElement} An XLS document element. │ │ │ │ │ - * options - {Object} Options for the reader. │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * options - {Object} The options object must contain a map property. If │ │ │ │ │ + * the map property is a string, it must be the id of a dom element │ │ │ │ │ + * where the new map will be placed. If the map property is an │ │ │ │ │ + * <OpenLayers.Map>, the layers from the context document will be added │ │ │ │ │ + * to the map. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} An object representing the XLSResponse. │ │ │ │ │ + * {<OpenLayers.Map>} A map based on the context. │ │ │ │ │ */ │ │ │ │ │ read: function(data, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var xls = {}; │ │ │ │ │ - this.readChildNodes(data, xls); │ │ │ │ │ - return xls; │ │ │ │ │ + var context = OpenLayers.Format.XML.VersionedOGC.prototype.read.apply(this, │ │ │ │ │ + arguments); │ │ │ │ │ + var map; │ │ │ │ │ + if (options && options.map) { │ │ │ │ │ + this.context = context; │ │ │ │ │ + if (options.map instanceof OpenLayers.Map) { │ │ │ │ │ + map = this.mergeContextToMap(context, options.map); │ │ │ │ │ + } else { │ │ │ │ │ + var mapOptions = options.map; │ │ │ │ │ + if (OpenLayers.Util.isElement(mapOptions) || │ │ │ │ │ + typeof mapOptions == "string") { │ │ │ │ │ + // we assume mapOptions references a div │ │ │ │ │ + // element │ │ │ │ │ + mapOptions = { │ │ │ │ │ + div: mapOptions │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + map = this.contextToMap(context, mapOptions); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // not documented as part of the API, provided as a non-API option │ │ │ │ │ + map = context; │ │ │ │ │ + } │ │ │ │ │ + return map; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ + * Method: getLayerFromContext │ │ │ │ │ + * Create a WMS layer from a layerContext object. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layerContext - {Object} An object representing a WMS layer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.WMS>} A WMS layer. │ │ │ │ │ */ │ │ │ │ │ - readers: { │ │ │ │ │ - "xls": { │ │ │ │ │ - "XLS": function(node, xls) { │ │ │ │ │ - xls.version = node.getAttribute("version"); │ │ │ │ │ - this.readChildNodes(node, xls); │ │ │ │ │ - }, │ │ │ │ │ - "Response": function(node, xls) { │ │ │ │ │ - this.readChildNodes(node, xls); │ │ │ │ │ - }, │ │ │ │ │ - "GeocodeResponse": function(node, xls) { │ │ │ │ │ - xls.responseLists = []; │ │ │ │ │ - this.readChildNodes(node, xls); │ │ │ │ │ - }, │ │ │ │ │ - "GeocodeResponseList": function(node, xls) { │ │ │ │ │ - var responseList = { │ │ │ │ │ - features: [], │ │ │ │ │ - numberOfGeocodedAddresses: parseInt(node.getAttribute("numberOfGeocodedAddresses")) │ │ │ │ │ - }; │ │ │ │ │ - xls.responseLists.push(responseList); │ │ │ │ │ - this.readChildNodes(node, responseList); │ │ │ │ │ - }, │ │ │ │ │ - "GeocodedAddress": function(node, responseList) { │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ - responseList.features.push(feature); │ │ │ │ │ - this.readChildNodes(node, feature); │ │ │ │ │ - // post-process geometry │ │ │ │ │ - feature.geometry = feature.components[0]; │ │ │ │ │ - }, │ │ │ │ │ - "GeocodeMatchCode": function(node, feature) { │ │ │ │ │ - feature.attributes.matchCode = { │ │ │ │ │ - accuracy: parseFloat(node.getAttribute("accuracy")), │ │ │ │ │ - matchType: node.getAttribute("matchType") │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ - "Address": function(node, feature) { │ │ │ │ │ - var address = { │ │ │ │ │ - countryCode: node.getAttribute("countryCode"), │ │ │ │ │ - addressee: node.getAttribute("addressee"), │ │ │ │ │ - street: [], │ │ │ │ │ - place: [] │ │ │ │ │ - }; │ │ │ │ │ - feature.attributes.address = address; │ │ │ │ │ - this.readChildNodes(node, address); │ │ │ │ │ - }, │ │ │ │ │ - "freeFormAddress": function(node, address) { │ │ │ │ │ - address.freeFormAddress = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "StreetAddress": function(node, address) { │ │ │ │ │ - this.readChildNodes(node, address); │ │ │ │ │ - }, │ │ │ │ │ - "Building": function(node, address) { │ │ │ │ │ - address.building = { │ │ │ │ │ - 'number': node.getAttribute("number"), │ │ │ │ │ - subdivision: node.getAttribute("subdivision"), │ │ │ │ │ - buildingName: node.getAttribute("buildingName") │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ - "Street": function(node, address) { │ │ │ │ │ - // only support the built-in primitive type for now │ │ │ │ │ - address.street.push(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "Place": function(node, address) { │ │ │ │ │ - // type is one of CountrySubdivision, │ │ │ │ │ - // CountrySecondarySubdivision, Municipality or │ │ │ │ │ - // MunicipalitySubdivision │ │ │ │ │ - address.place[node.getAttribute("type")] = │ │ │ │ │ - this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "PostalCode": function(node, address) { │ │ │ │ │ - address.postalCode = this.getChildValue(node); │ │ │ │ │ + getLayerFromContext: function(layerContext) { │ │ │ │ │ + var i, len; │ │ │ │ │ + // fill initial options object from layerContext │ │ │ │ │ + var options = { │ │ │ │ │ + queryable: layerContext.queryable, //keep queryable for api compatibility │ │ │ │ │ + visibility: layerContext.visibility, │ │ │ │ │ + maxExtent: layerContext.maxExtent, │ │ │ │ │ + metadata: OpenLayers.Util.applyDefaults(layerContext.metadata, { │ │ │ │ │ + styles: layerContext.styles, │ │ │ │ │ + formats: layerContext.formats, │ │ │ │ │ + "abstract": layerContext["abstract"], │ │ │ │ │ + dataURL: layerContext.dataURL │ │ │ │ │ + }), │ │ │ │ │ + numZoomLevels: layerContext.numZoomLevels, │ │ │ │ │ + units: layerContext.units, │ │ │ │ │ + isBaseLayer: layerContext.isBaseLayer, │ │ │ │ │ + opacity: layerContext.opacity, │ │ │ │ │ + displayInLayerSwitcher: layerContext.displayInLayerSwitcher, │ │ │ │ │ + singleTile: layerContext.singleTile, │ │ │ │ │ + tileSize: (layerContext.tileSize) ? │ │ │ │ │ + new OpenLayers.Size( │ │ │ │ │ + layerContext.tileSize.width, │ │ │ │ │ + layerContext.tileSize.height │ │ │ │ │ + ) : undefined, │ │ │ │ │ + minScale: layerContext.minScale || layerContext.maxScaleDenominator, │ │ │ │ │ + maxScale: layerContext.maxScale || layerContext.minScaleDenominator, │ │ │ │ │ + srs: layerContext.srs, │ │ │ │ │ + dimensions: layerContext.dimensions, │ │ │ │ │ + metadataURL: layerContext.metadataURL │ │ │ │ │ + }; │ │ │ │ │ + if (this.layerOptions) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.layerOptions); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var params = { │ │ │ │ │ + layers: layerContext.name, │ │ │ │ │ + transparent: layerContext.transparent, │ │ │ │ │ + version: layerContext.version │ │ │ │ │ + }; │ │ │ │ │ + if (layerContext.formats && layerContext.formats.length > 0) { │ │ │ │ │ + // set default value for params if current attribute is not positionned │ │ │ │ │ + params.format = layerContext.formats[0].value; │ │ │ │ │ + for (i = 0, len = layerContext.formats.length; i < len; i++) { │ │ │ │ │ + var format = layerContext.formats[i]; │ │ │ │ │ + if (format.current == true) { │ │ │ │ │ + params.format = format.value; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "gml": OpenLayers.Format.GML.v3.prototype.readers.gml │ │ │ │ │ + } │ │ │ │ │ + if (layerContext.styles && layerContext.styles.length > 0) { │ │ │ │ │ + for (i = 0, len = layerContext.styles.length; i < len; i++) { │ │ │ │ │ + var style = layerContext.styles[i]; │ │ │ │ │ + if (style.current == true) { │ │ │ │ │ + // three style types to consider │ │ │ │ │ + // 1) linked SLD │ │ │ │ │ + // 2) inline SLD │ │ │ │ │ + // 3) named style │ │ │ │ │ + if (style.href) { │ │ │ │ │ + params.sld = style.href; │ │ │ │ │ + } else if (style.body) { │ │ │ │ │ + params.sld_body = style.body; │ │ │ │ │ + } else { │ │ │ │ │ + params.styles = style.name; │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.layerParams) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(params, this.layerParams); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var layer = null; │ │ │ │ │ + var service = layerContext.service; │ │ │ │ │ + if (service == OpenLayers.Format.Context.serviceTypes.WFS) { │ │ │ │ │ + options.strategies = [new OpenLayers.Strategy.BBOX()]; │ │ │ │ │ + options.protocol = new OpenLayers.Protocol.WFS({ │ │ │ │ │ + url: layerContext.url, │ │ │ │ │ + // since we do not know featureNS, let the protocol │ │ │ │ │ + // determine it automagically using featurePrefix │ │ │ │ │ + featurePrefix: layerContext.name.split(":")[0], │ │ │ │ │ + featureType: layerContext.name.split(":").pop() │ │ │ │ │ + }); │ │ │ │ │ + layer = new OpenLayers.Layer.Vector( │ │ │ │ │ + layerContext.title || layerContext.name, │ │ │ │ │ + options │ │ │ │ │ + ); │ │ │ │ │ + } else if (service == OpenLayers.Format.Context.serviceTypes.KML) { │ │ │ │ │ + // use a vector layer with an HTTP Protcol and a Fixed strategy │ │ │ │ │ + options.strategies = [new OpenLayers.Strategy.Fixed()]; │ │ │ │ │ + options.protocol = new OpenLayers.Protocol.HTTP({ │ │ │ │ │ + url: layerContext.url, │ │ │ │ │ + format: new OpenLayers.Format.KML() │ │ │ │ │ + }); │ │ │ │ │ + layer = new OpenLayers.Layer.Vector( │ │ │ │ │ + layerContext.title || layerContext.name, │ │ │ │ │ + options │ │ │ │ │ + ); │ │ │ │ │ + } else if (service == OpenLayers.Format.Context.serviceTypes.GML) { │ │ │ │ │ + // use a vector layer with a HTTP Protocol and a Fixed strategy │ │ │ │ │ + options.strategies = [new OpenLayers.Strategy.Fixed()]; │ │ │ │ │ + options.protocol = new OpenLayers.Protocol.HTTP({ │ │ │ │ │ + url: layerContext.url, │ │ │ │ │ + format: new OpenLayers.Format.GML() │ │ │ │ │ + }); │ │ │ │ │ + layer = new OpenLayers.Layer.Vector( │ │ │ │ │ + layerContext.title || layerContext.name, │ │ │ │ │ + options │ │ │ │ │ + ); │ │ │ │ │ + } else if (layerContext.features) { │ │ │ │ │ + // inline GML or KML features │ │ │ │ │ + layer = new OpenLayers.Layer.Vector( │ │ │ │ │ + layerContext.title || layerContext.name, │ │ │ │ │ + options │ │ │ │ │ + ); │ │ │ │ │ + layer.addFeatures(layerContext.features); │ │ │ │ │ + } else if (layerContext.categoryLayer !== true) { │ │ │ │ │ + layer = new OpenLayers.Layer.WMS( │ │ │ │ │ + layerContext.title || layerContext.name, │ │ │ │ │ + layerContext.url, │ │ │ │ │ + params, │ │ │ │ │ + options │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return layer; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: write │ │ │ │ │ + * Method: getLayersFromContext │ │ │ │ │ + * Create an array of layers from an array of layerContext objects. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * request - {Object} An object representing the geocode request. │ │ │ │ │ + * layersContext - {Array(Object)} An array of objects representing layers. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} The root of an XLS document. │ │ │ │ │ + * {Array(<OpenLayers.Layer>)} An array of layers. │ │ │ │ │ */ │ │ │ │ │ - write: function(request) { │ │ │ │ │ - return this.writers.xls.XLS.apply(this, [request]); │ │ │ │ │ + getLayersFromContext: function(layersContext) { │ │ │ │ │ + var layers = []; │ │ │ │ │ + for (var i = 0, len = layersContext.length; i < len; i++) { │ │ │ │ │ + var layer = this.getLayerFromContext(layersContext[i]); │ │ │ │ │ + if (layer !== null) { │ │ │ │ │ + layers.push(layer); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return layers; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ + * Method: contextToMap │ │ │ │ │ + * Create a map given a context object. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * context - {Object} The context object. │ │ │ │ │ + * options - {Object} Default map options. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Map>} A map based on the context object. │ │ │ │ │ */ │ │ │ │ │ - writers: { │ │ │ │ │ - "xls": { │ │ │ │ │ - "XLS": function(request) { │ │ │ │ │ - var root = this.createElementNSPlus( │ │ │ │ │ - "xls:XLS", { │ │ │ │ │ - attributes: { │ │ │ │ │ - "version": this.VERSION, │ │ │ │ │ - "xsi:schemaLocation": this.schemaLocation │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - this.writeNode("RequestHeader", request.header, root); │ │ │ │ │ - this.writeNode("Request", request, root); │ │ │ │ │ - return root; │ │ │ │ │ - }, │ │ │ │ │ - "RequestHeader": function(header) { │ │ │ │ │ - return this.createElementNSPlus("xls:RequestHeader"); │ │ │ │ │ - }, │ │ │ │ │ - "Request": function(request) { │ │ │ │ │ - var node = this.createElementNSPlus("xls:Request", { │ │ │ │ │ - attributes: { │ │ │ │ │ - methodName: "GeocodeRequest", │ │ │ │ │ - requestID: request.requestID || "", │ │ │ │ │ - version: this.VERSION │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - this.writeNode("GeocodeRequest", request.addresses, node); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "GeocodeRequest": function(addresses) { │ │ │ │ │ - var node = this.createElementNSPlus("xls:GeocodeRequest"); │ │ │ │ │ - for (var i = 0, len = addresses.length; i < len; i++) { │ │ │ │ │ - this.writeNode("Address", addresses[i], node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Address": function(address) { │ │ │ │ │ - var node = this.createElementNSPlus("xls:Address", { │ │ │ │ │ - attributes: { │ │ │ │ │ - countryCode: address.countryCode │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (address.freeFormAddress) { │ │ │ │ │ - this.writeNode("freeFormAddress", address.freeFormAddress, node); │ │ │ │ │ - } else { │ │ │ │ │ - if (address.street) { │ │ │ │ │ - this.writeNode("StreetAddress", address, node); │ │ │ │ │ - } │ │ │ │ │ - if (address.municipality) { │ │ │ │ │ - this.writeNode("Municipality", address.municipality, node); │ │ │ │ │ - } │ │ │ │ │ - if (address.countrySubdivision) { │ │ │ │ │ - this.writeNode("CountrySubdivision", address.countrySubdivision, node); │ │ │ │ │ - } │ │ │ │ │ - if (address.postalCode) { │ │ │ │ │ - this.writeNode("PostalCode", address.postalCode, node); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "freeFormAddress": function(freeFormAddress) { │ │ │ │ │ - return this.createElementNSPlus("freeFormAddress", { │ │ │ │ │ - value: freeFormAddress │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "StreetAddress": function(address) { │ │ │ │ │ - var node = this.createElementNSPlus("xls:StreetAddress"); │ │ │ │ │ - if (address.building) { │ │ │ │ │ - this.writeNode(node, "Building", address.building); │ │ │ │ │ - } │ │ │ │ │ - var street = address.street; │ │ │ │ │ - if (!(OpenLayers.Util.isArray(street))) { │ │ │ │ │ - street = [street]; │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = street.length; i < len; i++) { │ │ │ │ │ - this.writeNode("Street", street[i], node); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "Building": function(building) { │ │ │ │ │ - return this.createElementNSPlus("xls:Building", { │ │ │ │ │ - attributes: { │ │ │ │ │ - "number": building["number"], │ │ │ │ │ - "subdivision": building.subdivision, │ │ │ │ │ - "buildingName": building.buildingName │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "Street": function(street) { │ │ │ │ │ - return this.createElementNSPlus("xls:Street", { │ │ │ │ │ - value: street │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "Municipality": function(municipality) { │ │ │ │ │ - return this.createElementNSPlus("xls:Place", { │ │ │ │ │ - attributes: { │ │ │ │ │ - type: "Municipality" │ │ │ │ │ - }, │ │ │ │ │ - value: municipality │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "CountrySubdivision": function(countrySubdivision) { │ │ │ │ │ - return this.createElementNSPlus("xls:Place", { │ │ │ │ │ - attributes: { │ │ │ │ │ - type: "CountrySubdivision" │ │ │ │ │ - }, │ │ │ │ │ - value: countrySubdivision │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "PostalCode": function(postalCode) { │ │ │ │ │ - return this.createElementNSPlus("xls:PostalCode", { │ │ │ │ │ - value: postalCode │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ + contextToMap: function(context, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + maxExtent: context.maxExtent, │ │ │ │ │ + projection: context.projection, │ │ │ │ │ + units: context.units │ │ │ │ │ + }, options); │ │ │ │ │ + │ │ │ │ │ + if (options.maxExtent) { │ │ │ │ │ + options.maxResolution = │ │ │ │ │ + options.maxExtent.getWidth() / OpenLayers.Map.TILE_WIDTH; │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + var metadata = { │ │ │ │ │ + contactInformation: context.contactInformation, │ │ │ │ │ + "abstract": context["abstract"], │ │ │ │ │ + keywords: context.keywords, │ │ │ │ │ + logo: context.logo, │ │ │ │ │ + descriptionURL: context.descriptionURL │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + options.metadata = metadata; │ │ │ │ │ + │ │ │ │ │ + var map = new OpenLayers.Map(options); │ │ │ │ │ + map.addLayers(this.getLayersFromContext(context.layersContext)); │ │ │ │ │ + map.setCenter( │ │ │ │ │ + context.bounds.getCenterLonLat(), │ │ │ │ │ + map.getZoomForExtent(context.bounds, true) │ │ │ │ │ + ); │ │ │ │ │ + return map; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.XLS.v1" │ │ │ │ │ + /** │ │ │ │ │ + * Method: mergeContextToMap │ │ │ │ │ + * Add layers from a context object to a map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * context - {Object} The context object. │ │ │ │ │ + * map - {<OpenLayers.Map>} The map. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Map>} The same map with layers added. │ │ │ │ │ + */ │ │ │ │ │ + mergeContextToMap: function(context, map) { │ │ │ │ │ + map.addLayers(this.getLayersFromContext(context.layersContext)); │ │ │ │ │ + return map; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Write a context document given a map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {<OpenLayers.Map> | Object} A map or context object. │ │ │ │ │ + * options - {Object} Optional configuration object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A context document string. │ │ │ │ │ + */ │ │ │ │ │ + write: function(obj, options) { │ │ │ │ │ + obj = this.toContext(obj); │ │ │ │ │ + return OpenLayers.Format.XML.VersionedOGC.prototype.write.apply(this, │ │ │ │ │ + arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.Context" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Format.Context.serviceTypes │ │ │ │ │ + * Enumeration for service types │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.Context.serviceTypes = { │ │ │ │ │ + "WMS": "urn:ogc:serviceType:WMS", │ │ │ │ │ + "WFS": "urn:ogc:serviceType:WFS", │ │ │ │ │ + "WCS": "urn:ogc:serviceType:WCS", │ │ │ │ │ + "GML": "urn:ogc:serviceType:GML", │ │ │ │ │ + "SLD": "urn:ogc:serviceType:SLD", │ │ │ │ │ + "FES": "urn:ogc:serviceType:FES", │ │ │ │ │ + "KML": "urn:ogc:serviceType:KML" │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/XLS/v1_1_0.js │ │ │ │ │ + OpenLayers/Format/OWSContext.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/XLS/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/Context.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.XLS.v1_1_0 │ │ │ │ │ - * Read / write XLS version 1.1.0. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Format.OWSContext │ │ │ │ │ + * Read and write OWS Context documents. OWS Context documents are a │ │ │ │ │ + * preliminary OGC (Open Geospatial Consortium) standard for storing the │ │ │ │ │ + * state of a web mapping application. In a way it is the successor to │ │ │ │ │ + * Web Map Context (WMC), since it is more generic and more types of layers │ │ │ │ │ + * can be stored. Also, nesting of layers is supported since version 0.3.1. │ │ │ │ │ + * For more information see: http://www.ogcnetwork.net/context │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XLS.v1> │ │ │ │ │ + * - <OpenLayers.Format.Context> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.XLS.v1_1_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.XLS.v1, { │ │ │ │ │ +OpenLayers.Format.OWSContext = OpenLayers.Class(OpenLayers.Format.Context, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: VERSION │ │ │ │ │ - * {String} 1.1 │ │ │ │ │ - */ │ │ │ │ │ - VERSION: "1.1", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "0.3.1". │ │ │ │ │ + */ │ │ │ │ │ + defaultVersion: "0.3.1", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} http://www.opengis.net/xls │ │ │ │ │ - * http://schemas.opengis.net/ols/1.1.0/LocationUtilityService.xsd │ │ │ │ │ - */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/xls http://schemas.opengis.net/ols/1.1.0/LocationUtilityService.xsd", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.OWSContext │ │ │ │ │ + * Create a new parser for OWS Context documents. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.XLS.v1_1_0 │ │ │ │ │ - * Instances of this class are not created directly. Use the │ │ │ │ │ - * <OpenLayers.Format.XLS> constructor instead. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: getVersion │ │ │ │ │ + * Returns the version to use. Subclasses can override this function │ │ │ │ │ + * if a different version detection is needed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * root - {DOMElement} │ │ │ │ │ + * options - {Object} Optional configuration object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The version to use. │ │ │ │ │ + */ │ │ │ │ │ + getVersion: function(root, options) { │ │ │ │ │ + var version = OpenLayers.Format.XML.VersionedOGC.prototype.getVersion.apply( │ │ │ │ │ + this, arguments); │ │ │ │ │ + // 0.3.1 is backwards compatible with 0.3.0 │ │ │ │ │ + if (version === "0.3.0") { │ │ │ │ │ + version = this.defaultVersion; │ │ │ │ │ + } │ │ │ │ │ + return version; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.XLS.v1_1_0" │ │ │ │ │ + /** │ │ │ │ │ + * Method: toContext │ │ │ │ │ + * Create a context object free from layer given a map or a │ │ │ │ │ + * context object. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {<OpenLayers.Map> | Object} The map or context. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} A context object. │ │ │ │ │ + */ │ │ │ │ │ + toContext: function(obj) { │ │ │ │ │ + var context = {}; │ │ │ │ │ + if (obj.CLASS_NAME == "OpenLayers.Map") { │ │ │ │ │ + context.bounds = obj.getExtent(); │ │ │ │ │ + context.maxExtent = obj.maxExtent; │ │ │ │ │ + context.projection = obj.projection; │ │ │ │ │ + context.size = obj.getSize(); │ │ │ │ │ + context.layers = obj.layers; │ │ │ │ │ + } │ │ │ │ │ + return context; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.OWSContext" │ │ │ │ │ │ │ │ │ │ -// Support non standard implementation │ │ │ │ │ -OpenLayers.Format.XLS.v1_1 = OpenLayers.Format.XLS.v1_1_0; │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WCSCapabilities/v1.js │ │ │ │ │ + OpenLayers/Format/GPX.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/WCSCapabilities.js │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ + * @requires OpenLayers/Projection.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WCSCapabilities.v1 │ │ │ │ │ - * Abstract class not to be instantiated directly. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Format.GPX │ │ │ │ │ + * Read/write GPX parser. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Format.GPX> constructor. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Format.XML> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WCSCapabilities.v1 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Format.GPX = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - splitSpace: (/\s+/) │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - */ │ │ │ │ │ - defaultPrefix: "wcs", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultDesc │ │ │ │ │ + * {String} Default description for the waypoints/tracks in the case │ │ │ │ │ + * where the feature has no "description" attribute. │ │ │ │ │ + * Default is "No description available". │ │ │ │ │ + */ │ │ │ │ │ + defaultDesc: "No description available", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return a list of coverages. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} List of named coverages. │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - var raw = data; │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ - } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - return capabilities; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: extractWaypoints │ │ │ │ │ + * {Boolean} Extract waypoints from GPX. (default: true) │ │ │ │ │ + */ │ │ │ │ │ + extractWaypoints: true, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1" │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: extractTracks │ │ │ │ │ + * {Boolean} Extract tracks from GPX. (default: true) │ │ │ │ │ + */ │ │ │ │ │ + extractTracks: true, │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WCSCapabilities/v1_0_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: extractRoutes │ │ │ │ │ + * {Boolean} Extract routes from GPX. (default: true) │ │ │ │ │ + */ │ │ │ │ │ + extractRoutes: true, │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: extractAttributes │ │ │ │ │ + * {Boolean} Extract feature attributes from GPX. (default: true) │ │ │ │ │ + * NOTE: Attributes as part of extensions to the GPX standard may not │ │ │ │ │ + * be extracted. │ │ │ │ │ + */ │ │ │ │ │ + extractAttributes: true, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WCSCapabilities/v1.js │ │ │ │ │ - * @requires OpenLayers/Format/GML/v3.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + gpx: "http://www.topografix.com/GPX/1/1", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WCSCapabilities/v1_0_0 │ │ │ │ │ - * Read WCS Capabilities version 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WCSCapabilities.v1> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WCSCapabilities.v1_0_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WCSCapabilities.v1, { │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} Schema location. Defaults to │ │ │ │ │ + * "http://www.topografix.com/GPX/1/1 http://www.topografix.com/GPX/1/1/gpx.xsd" │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: "http://www.topografix.com/GPX/1/1 http://www.topografix.com/GPX/1/1/gpx.xsd", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WCSCapabilities.v1_0_0 │ │ │ │ │ - * Create a new parser for WCS capabilities version 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: creator │ │ │ │ │ + * {String} The creator attribute to be added to the written GPX files. │ │ │ │ │ + * Defaults to "OpenLayers" │ │ │ │ │ + */ │ │ │ │ │ + creator: "OpenLayers", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - wcs: "http://www.opengis.net/wcs", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - ows: "http://www.opengis.net/ows" │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.GPX │ │ │ │ │ + * Create a new parser for GPX. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + // GPX coordinates are always in longlat WGS84 │ │ │ │ │ + this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: errorProperty │ │ │ │ │ - * {String} Which property of the returned object to check for in order to │ │ │ │ │ - * determine whether or not parsing has failed. In the case that the │ │ │ │ │ - * errorProperty is undefined on the returned object, the document will be │ │ │ │ │ - * run through an OGCExceptionReport parser. │ │ │ │ │ - */ │ │ │ │ │ - errorProperty: "service", │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wcs": { │ │ │ │ │ - "WCS_Capabilities": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "Service": function(node, obj) { │ │ │ │ │ - obj.service = {}; │ │ │ │ │ - this.readChildNodes(node, obj.service); │ │ │ │ │ - }, │ │ │ │ │ - "name": function(node, service) { │ │ │ │ │ - service.name = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "label": function(node, service) { │ │ │ │ │ - service.label = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "keywords": function(node, service) { │ │ │ │ │ - service.keywords = []; │ │ │ │ │ - this.readChildNodes(node, service.keywords); │ │ │ │ │ - }, │ │ │ │ │ - "keyword": function(node, keywords) { │ │ │ │ │ - // Append the keyword to the keywords list │ │ │ │ │ - keywords.push(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "responsibleParty": function(node, service) { │ │ │ │ │ - service.responsibleParty = {}; │ │ │ │ │ - this.readChildNodes(node, service.responsibleParty); │ │ │ │ │ - }, │ │ │ │ │ - "individualName": function(node, responsibleParty) { │ │ │ │ │ - responsibleParty.individualName = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "organisationName": function(node, responsibleParty) { │ │ │ │ │ - responsibleParty.organisationName = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "positionName": function(node, responsibleParty) { │ │ │ │ │ - responsibleParty.positionName = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "contactInfo": function(node, responsibleParty) { │ │ │ │ │ - responsibleParty.contactInfo = {}; │ │ │ │ │ - this.readChildNodes(node, responsibleParty.contactInfo); │ │ │ │ │ - }, │ │ │ │ │ - "phone": function(node, contactInfo) { │ │ │ │ │ - contactInfo.phone = {}; │ │ │ │ │ - this.readChildNodes(node, contactInfo.phone); │ │ │ │ │ - }, │ │ │ │ │ - "voice": function(node, phone) { │ │ │ │ │ - phone.voice = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "facsimile": function(node, phone) { │ │ │ │ │ - phone.facsimile = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "address": function(node, contactInfo) { │ │ │ │ │ - contactInfo.address = {}; │ │ │ │ │ - this.readChildNodes(node, contactInfo.address); │ │ │ │ │ - }, │ │ │ │ │ - "deliveryPoint": function(node, address) { │ │ │ │ │ - address.deliveryPoint = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "city": function(node, address) { │ │ │ │ │ - address.city = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "postalCode": function(node, address) { │ │ │ │ │ - address.postalCode = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "country": function(node, address) { │ │ │ │ │ - address.country = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "electronicMailAddress": function(node, address) { │ │ │ │ │ - address.electronicMailAddress = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "fees": function(node, service) { │ │ │ │ │ - service.fees = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "accessConstraints": function(node, service) { │ │ │ │ │ - service.accessConstraints = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "ContentMetadata": function(node, obj) { │ │ │ │ │ - obj.contentMetadata = []; │ │ │ │ │ - this.readChildNodes(node, obj.contentMetadata); │ │ │ │ │ - }, │ │ │ │ │ - "CoverageOfferingBrief": function(node, contentMetadata) { │ │ │ │ │ - var coverageOfferingBrief = {}; │ │ │ │ │ - this.readChildNodes(node, coverageOfferingBrief); │ │ │ │ │ - contentMetadata.push(coverageOfferingBrief); │ │ │ │ │ - }, │ │ │ │ │ - "name": function(node, coverageOfferingBrief) { │ │ │ │ │ - coverageOfferingBrief.name = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "label": function(node, coverageOfferingBrief) { │ │ │ │ │ - coverageOfferingBrief.label = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "lonLatEnvelope": function(node, coverageOfferingBrief) { │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, "http://www.opengis.net/gml", "pos"); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Return a list of features from a GPX doc │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * doc - {Element} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ + */ │ │ │ │ │ + read: function(doc) { │ │ │ │ │ + if (typeof doc == "string") { │ │ │ │ │ + doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]); │ │ │ │ │ + } │ │ │ │ │ + var features = []; │ │ │ │ │ │ │ │ │ │ - // We expect two nodes here, to create the corners of a bounding box │ │ │ │ │ - if (nodeList.length == 2) { │ │ │ │ │ - var min = {}; │ │ │ │ │ - var max = {}; │ │ │ │ │ + if (this.extractTracks) { │ │ │ │ │ + var tracks = doc.getElementsByTagName("trk"); │ │ │ │ │ + for (var i = 0, len = tracks.length; i < len; i++) { │ │ │ │ │ + // Attributes are only in trk nodes, not trkseg nodes │ │ │ │ │ + var attrs = {}; │ │ │ │ │ + if (this.extractAttributes) { │ │ │ │ │ + attrs = this.parseAttributes(tracks[i]); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - OpenLayers.Format.GML.v3.prototype.readers["gml"].pos.apply(this, [nodeList[0], min]); │ │ │ │ │ - OpenLayers.Format.GML.v3.prototype.readers["gml"].pos.apply(this, [nodeList[1], max]); │ │ │ │ │ + var segs = this.getElementsByTagNameNS(tracks[i], tracks[i].namespaceURI, "trkseg"); │ │ │ │ │ + for (var j = 0, seglen = segs.length; j < seglen; j++) { │ │ │ │ │ + // We don't yet support extraction of trkpt attributes │ │ │ │ │ + // All trksegs of a trk get that trk's attributes │ │ │ │ │ + var track = this.extractSegment(segs[j], "trkpt"); │ │ │ │ │ + features.push(new OpenLayers.Feature.Vector(track, attrs)); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - coverageOfferingBrief.lonLatEnvelope = {}; │ │ │ │ │ - coverageOfferingBrief.lonLatEnvelope.srsName = node.getAttribute("srsName"); │ │ │ │ │ - coverageOfferingBrief.lonLatEnvelope.min = min.points[0]; │ │ │ │ │ - coverageOfferingBrief.lonLatEnvelope.max = max.points[0]; │ │ │ │ │ + if (this.extractRoutes) { │ │ │ │ │ + var routes = doc.getElementsByTagName("rte"); │ │ │ │ │ + for (var k = 0, klen = routes.length; k < klen; k++) { │ │ │ │ │ + var attrs = {}; │ │ │ │ │ + if (this.extractAttributes) { │ │ │ │ │ + attrs = this.parseAttributes(routes[k]); │ │ │ │ │ + } │ │ │ │ │ + var route = this.extractSegment(routes[k], "rtept"); │ │ │ │ │ + features.push(new OpenLayers.Feature.Vector(route, attrs)); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.extractWaypoints) { │ │ │ │ │ + var waypoints = doc.getElementsByTagName("wpt"); │ │ │ │ │ + for (var l = 0, len = waypoints.length; l < len; l++) { │ │ │ │ │ + var attrs = {}; │ │ │ │ │ + if (this.extractAttributes) { │ │ │ │ │ + attrs = this.parseAttributes(waypoints[l]); │ │ │ │ │ + } │ │ │ │ │ + var wpt = new OpenLayers.Geometry.Point(waypoints[l].getAttribute("lon"), waypoints[l].getAttribute("lat")); │ │ │ │ │ + features.push(new OpenLayers.Feature.Vector(wpt, attrs)); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + for (var g = 0, featLength = features.length; g < featLength; g++) { │ │ │ │ │ + features[g].geometry.transform(this.externalProjection, │ │ │ │ │ + this.internalProjection); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return features; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: extractSegment │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * segment - {DOMElement} a trkseg or rte node to parse │ │ │ │ │ + * segmentType - {String} nodeName of waypoints that form the line │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.LineString>} A linestring geometry │ │ │ │ │ + */ │ │ │ │ │ + extractSegment: function(segment, segmentType) { │ │ │ │ │ + var points = this.getElementsByTagNameNS(segment, segment.namespaceURI, segmentType); │ │ │ │ │ + var point_features = []; │ │ │ │ │ + for (var i = 0, len = points.length; i < len; i++) { │ │ │ │ │ + point_features.push(new OpenLayers.Geometry.Point(points[i].getAttribute("lon"), points[i].getAttribute("lat"))); │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.LineString(point_features); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseAttributes │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {<DOMElement>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An attributes object. │ │ │ │ │ + */ │ │ │ │ │ + parseAttributes: function(node) { │ │ │ │ │ + // node is either a wpt, trk or rte │ │ │ │ │ + // attributes are children of the form <attr>value</attr> │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + var attrNode = node.firstChild, │ │ │ │ │ + value, name; │ │ │ │ │ + while (attrNode) { │ │ │ │ │ + if (attrNode.nodeType == 1 && attrNode.firstChild) { │ │ │ │ │ + value = attrNode.firstChild; │ │ │ │ │ + if (value.nodeType == 3 || value.nodeType == 4) { │ │ │ │ │ + name = (attrNode.prefix) ? │ │ │ │ │ + attrNode.nodeName.split(":")[1] : │ │ │ │ │ + attrNode.nodeName; │ │ │ │ │ + if (name != "trkseg" && name != "rtept") { │ │ │ │ │ + attributes[name] = value.nodeValue; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ + attrNode = attrNode.nextSibling; │ │ │ │ │ + } │ │ │ │ │ + return attributes; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1_0_0" │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Accepts Feature Collection, and returns a string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} List of features to serialize into a string. │ │ │ │ │ + * metadata - {Object} A key/value pairs object to build a metadata node to │ │ │ │ │ + * add to the gpx. Supported keys are 'name', 'desc', 'author'. │ │ │ │ │ + */ │ │ │ │ │ + write: function(features, metadata) { │ │ │ │ │ + features = OpenLayers.Util.isArray(features) ? │ │ │ │ │ + features : [features]; │ │ │ │ │ + var gpx = this.createElementNS(this.namespaces.gpx, "gpx"); │ │ │ │ │ + gpx.setAttribute("version", "1.1"); │ │ │ │ │ + gpx.setAttribute("creator", this.creator); │ │ │ │ │ + this.setAttributes(gpx, { │ │ │ │ │ + "xsi:schemaLocation": this.schemaLocation │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ + if (metadata && typeof metadata == 'object') { │ │ │ │ │ + gpx.appendChild(this.buildMetadataNode(metadata)); │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + gpx.appendChild(this.buildFeatureNode(features[i])); │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [gpx]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildMetadataNode │ │ │ │ │ + * Creates a "metadata" node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + buildMetadataNode: function(metadata) { │ │ │ │ │ + var types = ['name', 'desc', 'author'], │ │ │ │ │ + node = this.createElementNS(this.namespaces.gpx, 'metadata'); │ │ │ │ │ + for (var i = 0; i < types.length; i++) { │ │ │ │ │ + var type = types[i]; │ │ │ │ │ + if (metadata[type]) { │ │ │ │ │ + var n = this.createElementNS(this.namespaces.gpx, type); │ │ │ │ │ + n.appendChild(this.createTextNode(metadata[type])); │ │ │ │ │ + node.appendChild(n); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildFeatureNode │ │ │ │ │ + * Accepts an <OpenLayers.Feature.Vector>, and builds a node for it. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} - The created node, either a 'wpt' or a 'trk'. │ │ │ │ │ + */ │ │ │ │ │ + buildFeatureNode: function(feature) { │ │ │ │ │ + var geometry = feature.geometry; │ │ │ │ │ + geometry = geometry.clone(); │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry.transform(this.internalProjection, │ │ │ │ │ + this.externalProjection); │ │ │ │ │ + } │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + var wpt = this.buildWptNode(geometry); │ │ │ │ │ + this.appendAttributesNode(wpt, feature); │ │ │ │ │ + return wpt; │ │ │ │ │ + } else { │ │ │ │ │ + var trkNode = this.createElementNS(this.namespaces.gpx, "trk"); │ │ │ │ │ + this.appendAttributesNode(trkNode, feature); │ │ │ │ │ + var trkSegNodes = this.buildTrkSegNode(geometry); │ │ │ │ │ + trkSegNodes = OpenLayers.Util.isArray(trkSegNodes) ? │ │ │ │ │ + trkSegNodes : [trkSegNodes]; │ │ │ │ │ + for (var i = 0, len = trkSegNodes.length; i < len; i++) { │ │ │ │ │ + trkNode.appendChild(trkSegNodes[i]); │ │ │ │ │ + } │ │ │ │ │ + return trkNode; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildTrkSegNode │ │ │ │ │ + * Builds trkseg node(s) given a geometry │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * trknode │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + */ │ │ │ │ │ + buildTrkSegNode: function(geometry) { │ │ │ │ │ + var node, │ │ │ │ │ + i, │ │ │ │ │ + len, │ │ │ │ │ + point, │ │ │ │ │ + nodes; │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.LineString" || │ │ │ │ │ + geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ + node = this.createElementNS(this.namespaces.gpx, "trkseg"); │ │ │ │ │ + for (i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + point = geometry.components[i]; │ │ │ │ │ + node.appendChild(this.buildTrkPtNode(point)); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + } else { │ │ │ │ │ + nodes = []; │ │ │ │ │ + for (i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + nodes.push(this.buildTrkSegNode(geometry.components[i])); │ │ │ │ │ + } │ │ │ │ │ + return nodes; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildTrkPtNode │ │ │ │ │ + * Builds a trkpt node given a point │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A trkpt node │ │ │ │ │ + */ │ │ │ │ │ + buildTrkPtNode: function(point) { │ │ │ │ │ + var node = this.createElementNS(this.namespaces.gpx, "trkpt"); │ │ │ │ │ + node.setAttribute("lon", point.x); │ │ │ │ │ + node.setAttribute("lat", point.y); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildWptNode │ │ │ │ │ + * Builds a wpt node given a point │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A wpt node │ │ │ │ │ + */ │ │ │ │ │ + buildWptNode: function(geometry) { │ │ │ │ │ + var node = this.createElementNS(this.namespaces.gpx, "wpt"); │ │ │ │ │ + node.setAttribute("lon", geometry.x); │ │ │ │ │ + node.setAttribute("lat", geometry.y); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: appendAttributesNode │ │ │ │ │ + * Adds some attributes node. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} the node to append the attribute nodes to. │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + appendAttributesNode: function(node, feature) { │ │ │ │ │ + var name = this.createElementNS(this.namespaces.gpx, 'name'); │ │ │ │ │ + name.appendChild(this.createTextNode( │ │ │ │ │ + feature.attributes.name || feature.id)); │ │ │ │ │ + node.appendChild(name); │ │ │ │ │ + var desc = this.createElementNS(this.namespaces.gpx, 'desc'); │ │ │ │ │ + desc.appendChild(this.createTextNode( │ │ │ │ │ + feature.attributes.description || this.defaultDesc)); │ │ │ │ │ + node.appendChild(desc); │ │ │ │ │ + // TBD - deal with remaining (non name/description) attributes. │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.GPX" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WCSCapabilities/v1_1_0.js │ │ │ │ │ + OpenLayers/Format/WMC.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/WCSCapabilities/v1.js │ │ │ │ │ - * @requires OpenLayers/Format/OWSCommon/v1_1_0.js │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Format/Context.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WCSCapabilities/v1_1_0 │ │ │ │ │ - * Read WCS Capabilities version 1.1.0. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Format.WMC │ │ │ │ │ + * Read and write Web Map Context documents. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WCSCapabilities.v1> │ │ │ │ │ + * - <OpenLayers.Format.Context> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WCSCapabilities.v1_1_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WCSCapabilities.v1, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - wcs: "http://www.opengis.net/wcs/1.1", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - ows: "http://www.opengis.net/ows/1.1" │ │ │ │ │ - }, │ │ │ │ │ +OpenLayers.Format.WMC = OpenLayers.Class(OpenLayers.Format.Context, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: errorProperty │ │ │ │ │ - * {String} Which property of the returned object to check for in order to │ │ │ │ │ - * determine whether or not parsing has failed. In the case that the │ │ │ │ │ - * errorProperty is undefined on the returned object, the document will be │ │ │ │ │ - * run through an OGCExceptionReport parser. │ │ │ │ │ - */ │ │ │ │ │ - errorProperty: "operationsMetadata", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.1.0". │ │ │ │ │ + */ │ │ │ │ │ + defaultVersion: "1.1.0", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WCSCapabilities.v1_1_0 │ │ │ │ │ - * Create a new parser for WCS capabilities version 1.1.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMC │ │ │ │ │ + * Create a new parser for Web Map Context documents. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wcs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - // In 1.0.0, this was WCS_Capabilties, in 1.1.0, it's Capabilities │ │ │ │ │ - "Capabilities": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "Contents": function(node, request) { │ │ │ │ │ - request.contentMetadata = []; │ │ │ │ │ - this.readChildNodes(node, request.contentMetadata); │ │ │ │ │ + /** │ │ │ │ │ + * Method: layerToContext │ │ │ │ │ + * Create a layer context object given a wms layer object. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer.WMS>} The layer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} A layer context object. │ │ │ │ │ + */ │ │ │ │ │ + layerToContext: function(layer) { │ │ │ │ │ + var parser = this.getParser(); │ │ │ │ │ + var layerContext = { │ │ │ │ │ + queryable: layer.queryable, │ │ │ │ │ + visibility: layer.visibility, │ │ │ │ │ + name: layer.params["LAYERS"], │ │ │ │ │ + title: layer.name, │ │ │ │ │ + "abstract": layer.metadata["abstract"], │ │ │ │ │ + dataURL: layer.metadata.dataURL, │ │ │ │ │ + metadataURL: layer.metadataURL, │ │ │ │ │ + server: { │ │ │ │ │ + version: layer.params["VERSION"], │ │ │ │ │ + url: layer.url │ │ │ │ │ + }, │ │ │ │ │ + maxExtent: layer.maxExtent, │ │ │ │ │ + transparent: layer.params["TRANSPARENT"], │ │ │ │ │ + numZoomLevels: layer.numZoomLevels, │ │ │ │ │ + units: layer.units, │ │ │ │ │ + isBaseLayer: layer.isBaseLayer, │ │ │ │ │ + opacity: layer.opacity == 1 ? undefined : layer.opacity, │ │ │ │ │ + displayInLayerSwitcher: layer.displayInLayerSwitcher, │ │ │ │ │ + singleTile: layer.singleTile, │ │ │ │ │ + tileSize: (layer.singleTile || !layer.tileSize) ? │ │ │ │ │ + undefined : { │ │ │ │ │ + width: layer.tileSize.w, │ │ │ │ │ + height: layer.tileSize.h │ │ │ │ │ }, │ │ │ │ │ - "CoverageSummary": function(node, contentMetadata) { │ │ │ │ │ - var coverageSummary = {}; │ │ │ │ │ - // Read the summary: │ │ │ │ │ - this.readChildNodes(node, coverageSummary); │ │ │ │ │ + minScale: (layer.options.resolutions || │ │ │ │ │ + layer.options.scales || │ │ │ │ │ + layer.options.maxResolution || │ │ │ │ │ + layer.options.minScale) ? │ │ │ │ │ + layer.minScale : undefined, │ │ │ │ │ + maxScale: (layer.options.resolutions || │ │ │ │ │ + layer.options.scales || │ │ │ │ │ + layer.options.minResolution || │ │ │ │ │ + layer.options.maxScale) ? │ │ │ │ │ + layer.maxScale : undefined, │ │ │ │ │ + formats: [], │ │ │ │ │ + styles: [], │ │ │ │ │ + srs: layer.srs, │ │ │ │ │ + dimensions: layer.dimensions │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - // Add it to the contentMetadata array: │ │ │ │ │ - contentMetadata.push(coverageSummary); │ │ │ │ │ - }, │ │ │ │ │ - "Identifier": function(node, coverageSummary) { │ │ │ │ │ - coverageSummary.identifier = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Title": function(node, coverageSummary) { │ │ │ │ │ - coverageSummary.title = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Abstract": function(node, coverageSummary) { │ │ │ │ │ - coverageSummary["abstract"] = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "SupportedCRS": function(node, coverageSummary) { │ │ │ │ │ - var crs = this.getChildValue(node); │ │ │ │ │ - if (crs) { │ │ │ │ │ - if (!coverageSummary.supportedCRS) { │ │ │ │ │ - coverageSummary.supportedCRS = []; │ │ │ │ │ - } │ │ │ │ │ - coverageSummary.supportedCRS.push(crs); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "SupportedFormat": function(node, coverageSummary) { │ │ │ │ │ - var format = this.getChildValue(node); │ │ │ │ │ - if (format) { │ │ │ │ │ - if (!coverageSummary.supportedFormat) { │ │ │ │ │ - coverageSummary.supportedFormat = []; │ │ │ │ │ - } │ │ │ │ │ - coverageSummary.supportedFormat.push(format); │ │ │ │ │ - } │ │ │ │ │ + │ │ │ │ │ + if (layer.metadata.servertitle) { │ │ │ │ │ + layerContext.server.title = layer.metadata.servertitle; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (layer.metadata.formats && layer.metadata.formats.length > 0) { │ │ │ │ │ + for (var i = 0, len = layer.metadata.formats.length; i < len; i++) { │ │ │ │ │ + var format = layer.metadata.formats[i]; │ │ │ │ │ + layerContext.formats.push({ │ │ │ │ │ + value: format.value, │ │ │ │ │ + current: (format.value == layer.params["FORMAT"]) │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + layerContext.formats.push({ │ │ │ │ │ + value: layer.params["FORMAT"], │ │ │ │ │ + current: true │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (layer.metadata.styles && layer.metadata.styles.length > 0) { │ │ │ │ │ + for (var i = 0, len = layer.metadata.styles.length; i < len; i++) { │ │ │ │ │ + var style = layer.metadata.styles[i]; │ │ │ │ │ + if ((style.href == layer.params["SLD"]) || │ │ │ │ │ + (style.body == layer.params["SLD_BODY"]) || │ │ │ │ │ + (style.name == layer.params["STYLES"])) { │ │ │ │ │ + style.current = true; │ │ │ │ │ + } else { │ │ │ │ │ + style.current = false; │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.WCSCapabilities.v1.prototype.readers["wcs"]), │ │ │ │ │ - "ows": OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ - }, │ │ │ │ │ + layerContext.styles.push(style); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + layerContext.styles.push({ │ │ │ │ │ + href: layer.params["SLD"], │ │ │ │ │ + body: layer.params["SLD_BODY"], │ │ │ │ │ + name: layer.params["STYLES"] || parser.defaultStyleName, │ │ │ │ │ + title: parser.defaultStyleTitle, │ │ │ │ │ + current: true │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1_1_0" │ │ │ │ │ + return layerContext; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ + /** │ │ │ │ │ + * Method: toContext │ │ │ │ │ + * Create a context object free from layer given a map or a │ │ │ │ │ + * context object. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {<OpenLayers.Map> | Object} The map or context. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} A context object. │ │ │ │ │ + */ │ │ │ │ │ + toContext: function(obj) { │ │ │ │ │ + var context = {}; │ │ │ │ │ + var layers = obj.layers; │ │ │ │ │ + if (obj.CLASS_NAME == "OpenLayers.Map") { │ │ │ │ │ + var metadata = obj.metadata || {}; │ │ │ │ │ + context.size = obj.getSize(); │ │ │ │ │ + context.bounds = obj.getExtent(); │ │ │ │ │ + context.projection = obj.projection; │ │ │ │ │ + context.title = obj.title; │ │ │ │ │ + context.keywords = metadata.keywords; │ │ │ │ │ + context["abstract"] = metadata["abstract"]; │ │ │ │ │ + context.logo = metadata.logo; │ │ │ │ │ + context.descriptionURL = metadata.descriptionURL; │ │ │ │ │ + context.contactInformation = metadata.contactInformation; │ │ │ │ │ + context.maxExtent = obj.maxExtent; │ │ │ │ │ + } else { │ │ │ │ │ + // copy all obj properties except the "layers" property │ │ │ │ │ + OpenLayers.Util.applyDefaults(context, obj); │ │ │ │ │ + if (context.layers != undefined) { │ │ │ │ │ + delete(context.layers); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (context.layersContext == undefined) { │ │ │ │ │ + context.layersContext = []; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // let's convert layers into layersContext object (if any) │ │ │ │ │ + if (layers != undefined && OpenLayers.Util.isArray(layers)) { │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + var layer = layers[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.WMS) { │ │ │ │ │ + context.layersContext.push(this.layerToContext(layer)); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return context; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMC" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/SLD/v1_0_0_GeoServer.js │ │ │ │ │ + OpenLayers/Format/WPSCapabilities.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/SLD/v1_0_0.js │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.SLD/v1_0_0_GeoServer │ │ │ │ │ - * Read and write SLD version 1.0.0 with GeoServer-specific enhanced options. │ │ │ │ │ - * See http://svn.osgeo.org/geotools/trunk/modules/extension/xsd/xsd-sld/src/main/resources/org/geotools/sld/bindings/StyledLayerDescriptor.xsd │ │ │ │ │ - * for more information. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Format.WPSCapabilities │ │ │ │ │ + * Read WPS Capabilities. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.SLD.v1_0_0> │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.SLD.v1_0_0_GeoServer = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.SLD.v1_0_0, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} The specific parser version. │ │ │ │ │ - */ │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: profile │ │ │ │ │ - * {String} The specific profile │ │ │ │ │ - */ │ │ │ │ │ - profile: "GeoServer", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.SLD.v1_0_0_GeoServer │ │ │ │ │ - * Create a new parser for GeoServer-enhanced SLD version 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ +OpenLayers.Format.WPSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "sld": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "Priority": function(node, obj) { │ │ │ │ │ - var value = this.readers.ogc._expression.call(this, node); │ │ │ │ │ - if (value) { │ │ │ │ │ - obj.priority = value; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "VendorOption": function(node, obj) { │ │ │ │ │ - if (!obj.vendorOptions) { │ │ │ │ │ - obj.vendorOptions = {}; │ │ │ │ │ - } │ │ │ │ │ - obj.vendorOptions[node.getAttribute("name")] = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "TextSymbolizer": function(node, rule) { │ │ │ │ │ - OpenLayers.Format.SLD.v1_0_0.prototype.readers.sld.TextSymbolizer.apply(this, arguments); │ │ │ │ │ - var symbolizer = this.multipleSymbolizers ? rule.symbolizers[rule.symbolizers.length - 1] : rule.symbolizer["Text"]; │ │ │ │ │ - if (symbolizer.graphic === undefined) { │ │ │ │ │ - symbolizer.graphic = false; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.SLD.v1_0_0.prototype.readers["sld"]) │ │ │ │ │ - }, OpenLayers.Format.SLD.v1_0_0.prototype.readers), │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.0.0". │ │ │ │ │ + */ │ │ │ │ │ + defaultVersion: "1.0.0", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ - */ │ │ │ │ │ - writers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "sld": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "Priority": function(priority) { │ │ │ │ │ - return this.writers.sld._OGCExpression.call( │ │ │ │ │ - this, "sld:Priority", priority │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - "VendorOption": function(option) { │ │ │ │ │ - return this.createElementNSPlus("sld:VendorOption", { │ │ │ │ │ - attributes: { │ │ │ │ │ - name: option.name │ │ │ │ │ - }, │ │ │ │ │ - value: option.value │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "TextSymbolizer": function(symbolizer) { │ │ │ │ │ - var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ - var node = writers["sld"]["TextSymbolizer"].apply(this, arguments); │ │ │ │ │ - if (symbolizer.graphic !== false && (symbolizer.externalGraphic || symbolizer.graphicName)) { │ │ │ │ │ - this.writeNode("Graphic", symbolizer, node); │ │ │ │ │ - } │ │ │ │ │ - if ("priority" in symbolizer) { │ │ │ │ │ - this.writeNode("Priority", symbolizer.priority, node); │ │ │ │ │ - } │ │ │ │ │ - return this.addVendorOptions(node, symbolizer); │ │ │ │ │ - }, │ │ │ │ │ - "PointSymbolizer": function(symbolizer) { │ │ │ │ │ - var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ - var node = writers["sld"]["PointSymbolizer"].apply(this, arguments); │ │ │ │ │ - return this.addVendorOptions(node, symbolizer); │ │ │ │ │ - }, │ │ │ │ │ - "LineSymbolizer": function(symbolizer) { │ │ │ │ │ - var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ - var node = writers["sld"]["LineSymbolizer"].apply(this, arguments); │ │ │ │ │ - return this.addVendorOptions(node, symbolizer); │ │ │ │ │ - }, │ │ │ │ │ - "PolygonSymbolizer": function(symbolizer) { │ │ │ │ │ - var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ - var node = writers["sld"]["PolygonSymbolizer"].apply(this, arguments); │ │ │ │ │ - return this.addVendorOptions(node, symbolizer); │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.SLD.v1_0_0.prototype.writers["sld"]) │ │ │ │ │ - }, OpenLayers.Format.SLD.v1_0_0.prototype.writers), │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WPSCapabilities │ │ │ │ │ + * Create a new parser for WPS Capabilities. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: addVendorOptions │ │ │ │ │ - * Add in the VendorOption tags and return the node again. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} A DOM node. │ │ │ │ │ - * symbolizer - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A DOM node. │ │ │ │ │ - */ │ │ │ │ │ - addVendorOptions: function(node, symbolizer) { │ │ │ │ │ - var options = symbolizer.vendorOptions; │ │ │ │ │ - if (options) { │ │ │ │ │ - for (var key in symbolizer.vendorOptions) { │ │ │ │ │ - this.writeNode("VendorOption", { │ │ │ │ │ - name: key, │ │ │ │ │ - value: symbolizer.vendorOptions[key] │ │ │ │ │ - }, node); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return information about │ │ │ │ │ + * the service. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Info about the WPS │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SLD.v1_0_0_GeoServer" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WPSCapabilities" │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/CSWGetDomain/v2_0_2.js │ │ │ │ │ + OpenLayers/Format/WMSGetFeatureInfo.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Format/XML.js │ │ │ │ │ - * @requires OpenLayers/Format/CSWGetDomain.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.CSWGetDomain.v2_0_2 │ │ │ │ │ - * A format for creating CSWGetDomain v2.0.2 transactions. │ │ │ │ │ - * Create a new instance with the │ │ │ │ │ - * <OpenLayers.Format.CSWGetDomain.v2_0_2> constructor. │ │ │ │ │ + * Class: OpenLayers.Format.WMSGetFeatureInfo │ │ │ │ │ + * Class to read GetFeatureInfo responses from Web Mapping Services │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Format.XML> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.CSWGetDomain.v2_0_2 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - csw: "http://www.opengis.net/cat/csw/2.0.2" │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - * {String} The default prefix (used by Format.XML). │ │ │ │ │ - */ │ │ │ │ │ - defaultPrefix: "csw", │ │ │ │ │ +OpenLayers.Format.WMSGetFeatureInfo = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} CSW version number. │ │ │ │ │ + * APIProperty: layerIdentifier │ │ │ │ │ + * {String} All xml nodes containing this search criteria will populate an │ │ │ │ │ + * internal array of layer nodes. │ │ │ │ │ */ │ │ │ │ │ - version: "2.0.2", │ │ │ │ │ + layerIdentifier: '_layer', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} http://www.opengis.net/cat/csw/2.0.2 │ │ │ │ │ - * http://schemas.opengis.net/csw/2.0.2/CSW-discovery.xsd │ │ │ │ │ + * APIProperty: featureIdentifier │ │ │ │ │ + * {String} All xml nodes containing this search criteria will populate an │ │ │ │ │ + * internal array of feature nodes for each layer node found. │ │ │ │ │ */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/cat/csw/2.0.2 http://schemas.opengis.net/csw/2.0.2/CSW-discovery.xsd", │ │ │ │ │ + featureIdentifier: '_feature', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: PropertyName │ │ │ │ │ - * {String} Value of the csw:PropertyName element, used when │ │ │ │ │ - * writing a GetDomain document. │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ */ │ │ │ │ │ - PropertyName: null, │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: ParameterName │ │ │ │ │ - * {String} Value of the csw:ParameterName element, used when │ │ │ │ │ - * writing a GetDomain document. │ │ │ │ │ + * Property: gmlFormat │ │ │ │ │ + * {<OpenLayers.Format.GML>} internal GML format for parsing geometries │ │ │ │ │ + * in msGMLOutput │ │ │ │ │ */ │ │ │ │ │ - ParameterName: null, │ │ │ │ │ + gmlFormat: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.CSWGetDomain.v2_0_2 │ │ │ │ │ - * A class for parsing and generating CSWGetDomain v2.0.2 transactions. │ │ │ │ │ + * Constructor: OpenLayers.Format.WMSGetFeatureInfo │ │ │ │ │ + * Create a new parser for WMS GetFeatureInfo responses │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * - PropertyName │ │ │ │ │ - * - ParameterName │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIMethod: read │ │ │ │ │ - * Parse the response from a GetDomain request. │ │ │ │ │ + * Read WMS GetFeatureInfo data from a string, and return an array of features │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ */ │ │ │ │ │ read: function(data) { │ │ │ │ │ + var result; │ │ │ │ │ if (typeof data == "string") { │ │ │ │ │ data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ + var root = data.documentElement; │ │ │ │ │ + if (root) { │ │ │ │ │ + var scope = this; │ │ │ │ │ + var read = this["read_" + root.nodeName]; │ │ │ │ │ + if (read) { │ │ │ │ │ + result = read.call(this, root); │ │ │ │ │ + } else { │ │ │ │ │ + // fall-back to GML since this is a common output format for WMS │ │ │ │ │ + // GetFeatureInfo responses │ │ │ │ │ + result = new OpenLayers.Format.GML((this.options ? this.options : {})).read(data); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + result = data; │ │ │ │ │ } │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readNode(data, obj); │ │ │ │ │ - return obj; │ │ │ │ │ + return result; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ + * Method: read_msGMLOutput │ │ │ │ │ + * Parse msGMLOutput nodes. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} │ │ │ │ │ */ │ │ │ │ │ - readers: { │ │ │ │ │ - "csw": { │ │ │ │ │ - "GetDomainResponse": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "DomainValues": function(node, obj) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(obj.DomainValues))) { │ │ │ │ │ - obj.DomainValues = []; │ │ │ │ │ + read_msGMLOutput: function(data) { │ │ │ │ │ + var response = []; │ │ │ │ │ + var layerNodes = this.getSiblingNodesByTagCriteria(data, │ │ │ │ │ + this.layerIdentifier); │ │ │ │ │ + if (layerNodes) { │ │ │ │ │ + for (var i = 0, len = layerNodes.length; i < len; ++i) { │ │ │ │ │ + var node = layerNodes[i]; │ │ │ │ │ + var layerName = node.nodeName; │ │ │ │ │ + if (node.prefix) { │ │ │ │ │ + layerName = layerName.split(':')[1]; │ │ │ │ │ } │ │ │ │ │ - var attrs = node.attributes; │ │ │ │ │ - var domainValue = {}; │ │ │ │ │ - for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - domainValue[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ + var layerName = layerName.replace(this.layerIdentifier, ''); │ │ │ │ │ + var featureNodes = this.getSiblingNodesByTagCriteria(node, │ │ │ │ │ + this.featureIdentifier); │ │ │ │ │ + if (featureNodes) { │ │ │ │ │ + for (var j = 0; j < featureNodes.length; j++) { │ │ │ │ │ + var featureNode = featureNodes[j]; │ │ │ │ │ + var geomInfo = this.parseGeometry(featureNode); │ │ │ │ │ + var attributes = this.parseAttributes(featureNode); │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geomInfo.geometry, │ │ │ │ │ + attributes, null); │ │ │ │ │ + feature.bounds = geomInfo.bounds; │ │ │ │ │ + feature.type = layerName; │ │ │ │ │ + response.push(feature); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.readChildNodes(node, domainValue); │ │ │ │ │ - obj.DomainValues.push(domainValue); │ │ │ │ │ - }, │ │ │ │ │ - "PropertyName": function(node, obj) { │ │ │ │ │ - obj.PropertyName = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "ParameterName": function(node, obj) { │ │ │ │ │ - obj.ParameterName = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "ListOfValues": function(node, obj) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(obj.ListOfValues))) { │ │ │ │ │ - obj.ListOfValues = []; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return response; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: read_FeatureInfoResponse │ │ │ │ │ + * Parse FeatureInfoResponse nodes. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} │ │ │ │ │ + */ │ │ │ │ │ + read_FeatureInfoResponse: function(data) { │ │ │ │ │ + var response = []; │ │ │ │ │ + var featureNodes = this.getElementsByTagNameNS(data, '*', │ │ │ │ │ + 'FIELDS'); │ │ │ │ │ + │ │ │ │ │ + for (var i = 0, len = featureNodes.length; i < len; i++) { │ │ │ │ │ + var featureNode = featureNodes[i]; │ │ │ │ │ + var geom = null; │ │ │ │ │ + │ │ │ │ │ + // attributes can be actual attributes on the FIELDS tag, │ │ │ │ │ + // or FIELD children │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + var j; │ │ │ │ │ + var jlen = featureNode.attributes.length; │ │ │ │ │ + if (jlen > 0) { │ │ │ │ │ + for (j = 0; j < jlen; j++) { │ │ │ │ │ + var attribute = featureNode.attributes[j]; │ │ │ │ │ + attributes[attribute.nodeName] = attribute.nodeValue; │ │ │ │ │ } │ │ │ │ │ - this.readChildNodes(node, obj.ListOfValues); │ │ │ │ │ - }, │ │ │ │ │ - "Value": function(node, obj) { │ │ │ │ │ - var attrs = node.attributes; │ │ │ │ │ - var value = {}; │ │ │ │ │ - for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - value[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ + } else { │ │ │ │ │ + var nodes = featureNode.childNodes; │ │ │ │ │ + for (j = 0, jlen = nodes.length; j < jlen; ++j) { │ │ │ │ │ + var node = nodes[j]; │ │ │ │ │ + if (node.nodeType != 3) { │ │ │ │ │ + attributes[node.getAttribute("name")] = │ │ │ │ │ + node.getAttribute("value"); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - value.value = this.getChildValue(node); │ │ │ │ │ - obj.push({ │ │ │ │ │ - Value: value │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - "ConceptualScheme": function(node, obj) { │ │ │ │ │ - obj.ConceptualScheme = {}; │ │ │ │ │ - this.readChildNodes(node, obj.ConceptualScheme); │ │ │ │ │ - }, │ │ │ │ │ - "Name": function(node, obj) { │ │ │ │ │ - obj.Name = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Document": function(node, obj) { │ │ │ │ │ - obj.Document = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Authority": function(node, obj) { │ │ │ │ │ - obj.Authority = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "RangeOfValues": function(node, obj) { │ │ │ │ │ - obj.RangeOfValues = {}; │ │ │ │ │ - this.readChildNodes(node, obj.RangeOfValues); │ │ │ │ │ - }, │ │ │ │ │ - "MinValue": function(node, obj) { │ │ │ │ │ - var attrs = node.attributes; │ │ │ │ │ - var value = {}; │ │ │ │ │ - for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - value[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + response.push( │ │ │ │ │ + new OpenLayers.Feature.Vector(geom, attributes, null) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return response; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getSiblingNodesByTagCriteria │ │ │ │ │ + * Recursively searches passed xml node and all it's descendant levels for │ │ │ │ │ + * nodes whose tagName contains the passed search string. This returns an │ │ │ │ │ + * array of all sibling nodes which match the criteria from the highest │ │ │ │ │ + * hierarchial level from which a match is found. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} An xml node │ │ │ │ │ + * criteria - {String} Search string which will match some part of a tagName │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * Array({DOMElement}) An array of sibling xml nodes │ │ │ │ │ + */ │ │ │ │ │ + getSiblingNodesByTagCriteria: function(node, criteria) { │ │ │ │ │ + var nodes = []; │ │ │ │ │ + var children, tagName, n, matchNodes, child; │ │ │ │ │ + if (node && node.hasChildNodes()) { │ │ │ │ │ + children = node.childNodes; │ │ │ │ │ + n = children.length; │ │ │ │ │ + │ │ │ │ │ + for (var k = 0; k < n; k++) { │ │ │ │ │ + child = children[k]; │ │ │ │ │ + while (child && child.nodeType != 1) { │ │ │ │ │ + child = child.nextSibling; │ │ │ │ │ + k++; │ │ │ │ │ } │ │ │ │ │ - value.value = this.getChildValue(node); │ │ │ │ │ - obj.MinValue = value; │ │ │ │ │ - }, │ │ │ │ │ - "MaxValue": function(node, obj) { │ │ │ │ │ - var attrs = node.attributes; │ │ │ │ │ - var value = {}; │ │ │ │ │ - for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - value[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ + tagName = (child ? child.nodeName : ''); │ │ │ │ │ + if (tagName.length > 0 && tagName.indexOf(criteria) > -1) { │ │ │ │ │ + nodes.push(child); │ │ │ │ │ + } else { │ │ │ │ │ + matchNodes = this.getSiblingNodesByTagCriteria( │ │ │ │ │ + child, criteria); │ │ │ │ │ + │ │ │ │ │ + if (matchNodes.length > 0) { │ │ │ │ │ + (nodes.length == 0) ? │ │ │ │ │ + nodes = matchNodes: nodes.push(matchNodes); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - value.value = this.getChildValue(node); │ │ │ │ │ - obj.MaxValue = value; │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ } │ │ │ │ │ + return nodes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: write │ │ │ │ │ - * Given an configuration js object, write a CSWGetDomain request. │ │ │ │ │ + * Method: parseAttributes │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} A object mapping the request. │ │ │ │ │ + * node - {<DOMElement>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A serialized CSWGetDomain request. │ │ │ │ │ + * {Object} An attributes object. │ │ │ │ │ + * │ │ │ │ │ + * Notes: │ │ │ │ │ + * Assumes that attributes are direct child xml nodes of the passed node │ │ │ │ │ + * and contain only a single text node. │ │ │ │ │ */ │ │ │ │ │ - write: function(options) { │ │ │ │ │ - var node = this.writeNode("csw:GetDomain", options); │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [node]); │ │ │ │ │ + parseAttributes: function(node) { │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + if (node.nodeType == 1) { │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + var n = children.length; │ │ │ │ │ + for (var i = 0; i < n; ++i) { │ │ │ │ │ + var child = children[i]; │ │ │ │ │ + if (child.nodeType == 1) { │ │ │ │ │ + var grandchildren = child.childNodes; │ │ │ │ │ + var name = (child.prefix) ? │ │ │ │ │ + child.nodeName.split(":")[1] : child.nodeName; │ │ │ │ │ + if (grandchildren.length == 0) { │ │ │ │ │ + attributes[name] = null; │ │ │ │ │ + } else if (grandchildren.length == 1) { │ │ │ │ │ + var grandchild = grandchildren[0]; │ │ │ │ │ + if (grandchild.nodeType == 3 || │ │ │ │ │ + grandchild.nodeType == 4) { │ │ │ │ │ + var value = grandchild.nodeValue.replace( │ │ │ │ │ + this.regExes.trimSpace, ""); │ │ │ │ │ + attributes[name] = value; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return attributes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ + * Method: parseGeometry │ │ │ │ │ + * Parse the geometry and the feature bounds out of the node using │ │ │ │ │ + * Format.GML │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {<DOMElement>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object containing the geometry and the feature bounds │ │ │ │ │ */ │ │ │ │ │ - writers: { │ │ │ │ │ - "csw": { │ │ │ │ │ - "GetDomain": function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:GetDomain", { │ │ │ │ │ - attributes: { │ │ │ │ │ - service: "CSW", │ │ │ │ │ - version: this.version │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (options.PropertyName || this.PropertyName) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "csw:PropertyName", │ │ │ │ │ - options.PropertyName || this.PropertyName, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } else if (options.ParameterName || this.ParameterName) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "csw:ParameterName", │ │ │ │ │ - options.ParameterName || this.ParameterName, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - this.readChildNodes(node, options); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "PropertyName": function(value) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:PropertyName", { │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - "ParameterName": function(value) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:ParameterName", { │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - return node; │ │ │ │ │ - } │ │ │ │ │ + parseGeometry: function(node) { │ │ │ │ │ + // we need to use the old Format.GML parser since we do not know the │ │ │ │ │ + // geometry name │ │ │ │ │ + if (!this.gmlFormat) { │ │ │ │ │ + this.gmlFormat = new OpenLayers.Format.GML(); │ │ │ │ │ + } │ │ │ │ │ + var feature = this.gmlFormat.parseFeature(node); │ │ │ │ │ + var geometry, bounds = null; │ │ │ │ │ + if (feature) { │ │ │ │ │ + geometry = feature.geometry && feature.geometry.clone(); │ │ │ │ │ + bounds = feature.bounds && feature.bounds.clone(); │ │ │ │ │ + feature.destroy(); │ │ │ │ │ } │ │ │ │ │ + return { │ │ │ │ │ + geometry: geometry, │ │ │ │ │ + bounds: bounds │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.CSWGetDomain.v2_0_2" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSGetFeatureInfo" │ │ │ │ │ + │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMTSCapabilities/v1_0_0.js │ │ │ │ │ + OpenLayers/Format/WFSDescribeFeatureType.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/WMTSCapabilities.js │ │ │ │ │ - * @requires OpenLayers/Format/OWSCommon/v1_1_0.js │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Format/OGCExceptionReport.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WMTSCapabilities.v1_0_0 │ │ │ │ │ - * Read WMTS Capabilities version 1.0.0. │ │ │ │ │ + * Class: OpenLayers.Format.WFSDescribeFeatureType │ │ │ │ │ + * Read WFS DescribeFeatureType response │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMTSCapabilities> │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WMTSCapabilities.v1_0_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.OWSCommon.v1_1_0, { │ │ │ │ │ +OpenLayers.Format.WFSDescribeFeatureType = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} The parser version ("1.0.0"). │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ */ │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g) │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Property: namespaces │ │ │ │ │ * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ */ │ │ │ │ │ namespaces: { │ │ │ │ │ - ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ - wmts: "http://www.opengis.net/wmts/1.0", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ + xsd: "http://www.w3.org/2001/XMLSchema" │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: yx │ │ │ │ │ - * {Object} Members in the yx object are used to determine if a CRS URN │ │ │ │ │ - * corresponds to a CRS with y,x axis order. Member names are CRS URNs │ │ │ │ │ - * and values are boolean. Defaults come from the │ │ │ │ │ - * <OpenLayers.Format.WMTSCapabilities> prototype. │ │ │ │ │ - */ │ │ │ │ │ - yx: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - * {String} The default namespace alias for creating element nodes. │ │ │ │ │ - */ │ │ │ │ │ - defaultPrefix: "wmts", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMTSCapabilities.v1_0_0 │ │ │ │ │ - * Create a new parser for WMTS capabilities version 1.0.0. │ │ │ │ │ + * Constructor: OpenLayers.Format.WFSDescribeFeatureType │ │ │ │ │ + * Create a new parser for WFS DescribeFeatureType responses. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * options - {Object} An optional object whose properties will be set on │ │ │ │ │ * this instance. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.options = options; │ │ │ │ │ - var yx = OpenLayers.Util.extend({}, OpenLayers.Format.WMTSCapabilities.prototype.yx); │ │ │ │ │ - this.yx = OpenLayers.Util.extend(yx, this.yx); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return info about the WMTS. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Information about the SOS service. │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ - } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - capabilities.version = this.version; │ │ │ │ │ - return capabilities; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Property: readers │ │ │ │ │ * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ * be applied when a namespaced node is found matching the function │ │ │ │ │ * name. The function will be applied in the scope of this parser │ │ │ │ │ * with two arguments: the node being read and a context object passed │ │ │ │ │ * from the parent. │ │ │ │ │ */ │ │ │ │ │ readers: { │ │ │ │ │ - "wmts": { │ │ │ │ │ - "Capabilities": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "Contents": function(node, obj) { │ │ │ │ │ - obj.contents = {}; │ │ │ │ │ - obj.contents.layers = []; │ │ │ │ │ - obj.contents.tileMatrixSets = {}; │ │ │ │ │ - this.readChildNodes(node, obj.contents); │ │ │ │ │ - }, │ │ │ │ │ - "Layer": function(node, obj) { │ │ │ │ │ - var layer = { │ │ │ │ │ - styles: [], │ │ │ │ │ - formats: [], │ │ │ │ │ - dimensions: [], │ │ │ │ │ - tileMatrixSetLinks: [] │ │ │ │ │ + "xsd": { │ │ │ │ │ + "schema": function(node, obj) { │ │ │ │ │ + var complexTypes = []; │ │ │ │ │ + var customTypes = {}; │ │ │ │ │ + var schema = { │ │ │ │ │ + complexTypes: complexTypes, │ │ │ │ │ + customTypes: customTypes │ │ │ │ │ }; │ │ │ │ │ - layer.layers = []; │ │ │ │ │ - this.readChildNodes(node, layer); │ │ │ │ │ - obj.layers.push(layer); │ │ │ │ │ - }, │ │ │ │ │ - "Style": function(node, obj) { │ │ │ │ │ - var style = {}; │ │ │ │ │ - style.isDefault = (node.getAttribute("isDefault") === "true"); │ │ │ │ │ - this.readChildNodes(node, style); │ │ │ │ │ - obj.styles.push(style); │ │ │ │ │ + var i, len; │ │ │ │ │ + │ │ │ │ │ + this.readChildNodes(node, schema); │ │ │ │ │ + │ │ │ │ │ + var attributes = node.attributes; │ │ │ │ │ + var attr, name; │ │ │ │ │ + for (i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ + attr = attributes[i]; │ │ │ │ │ + name = attr.name; │ │ │ │ │ + if (name.indexOf("xmlns") === 0) { │ │ │ │ │ + this.setNamespace(name.split(":")[1] || "", attr.value); │ │ │ │ │ + } else { │ │ │ │ │ + obj[name] = attr.value; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + obj.featureTypes = complexTypes; │ │ │ │ │ + obj.targetPrefix = this.namespaceAlias[obj.targetNamespace]; │ │ │ │ │ + │ │ │ │ │ + // map complexTypes to names of customTypes │ │ │ │ │ + var complexType, customType; │ │ │ │ │ + for (i = 0, len = complexTypes.length; i < len; ++i) { │ │ │ │ │ + complexType = complexTypes[i]; │ │ │ │ │ + customType = customTypes[complexType.typeName]; │ │ │ │ │ + if (customTypes[complexType.typeName]) { │ │ │ │ │ + complexType.typeName = customType.name; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - "Format": function(node, obj) { │ │ │ │ │ - obj.formats.push(this.getChildValue(node)); │ │ │ │ │ + "complexType": function(node, obj) { │ │ │ │ │ + var complexType = { │ │ │ │ │ + // this is a temporary typeName, it will be overwritten by │ │ │ │ │ + // the schema reader with the metadata found in the │ │ │ │ │ + // customTypes hash │ │ │ │ │ + "typeName": node.getAttribute("name") │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, complexType); │ │ │ │ │ + obj.complexTypes.push(complexType); │ │ │ │ │ }, │ │ │ │ │ - "TileMatrixSetLink": function(node, obj) { │ │ │ │ │ - var tileMatrixSetLink = {}; │ │ │ │ │ - this.readChildNodes(node, tileMatrixSetLink); │ │ │ │ │ - obj.tileMatrixSetLinks.push(tileMatrixSetLink); │ │ │ │ │ + "complexContent": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ }, │ │ │ │ │ - "TileMatrixSet": function(node, obj) { │ │ │ │ │ - // node could be child of wmts:Contents or wmts:TileMatrixSetLink │ │ │ │ │ - // duck type wmts:Contents by looking for layers │ │ │ │ │ - if (obj.layers) { │ │ │ │ │ - // TileMatrixSet as object type in schema │ │ │ │ │ - var tileMatrixSet = { │ │ │ │ │ - matrixIds: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, tileMatrixSet); │ │ │ │ │ - obj.tileMatrixSets[tileMatrixSet.identifier] = tileMatrixSet; │ │ │ │ │ - } else { │ │ │ │ │ - // TileMatrixSet as string type in schema │ │ │ │ │ - obj.tileMatrixSet = this.getChildValue(node); │ │ │ │ │ - } │ │ │ │ │ + "extension": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ }, │ │ │ │ │ - "TileMatrix": function(node, obj) { │ │ │ │ │ - var tileMatrix = { │ │ │ │ │ - supportedCRS: obj.supportedCRS │ │ │ │ │ + "sequence": function(node, obj) { │ │ │ │ │ + var sequence = { │ │ │ │ │ + elements: [] │ │ │ │ │ }; │ │ │ │ │ - this.readChildNodes(node, tileMatrix); │ │ │ │ │ - obj.matrixIds.push(tileMatrix); │ │ │ │ │ - }, │ │ │ │ │ - "ScaleDenominator": function(node, obj) { │ │ │ │ │ - obj.scaleDenominator = parseFloat(this.getChildValue(node)); │ │ │ │ │ + this.readChildNodes(node, sequence); │ │ │ │ │ + obj.properties = sequence.elements; │ │ │ │ │ }, │ │ │ │ │ - "TopLeftCorner": function(node, obj) { │ │ │ │ │ - var topLeftCorner = this.getChildValue(node); │ │ │ │ │ - var coords = topLeftCorner.split(" "); │ │ │ │ │ - // decide on axis order for the given CRS │ │ │ │ │ - var yx; │ │ │ │ │ - if (obj.supportedCRS) { │ │ │ │ │ - // extract out version from URN │ │ │ │ │ - var crs = obj.supportedCRS.replace( │ │ │ │ │ - /urn:ogc:def:crs:(\w+):.+:(\w+)$/, │ │ │ │ │ - "urn:ogc:def:crs:$1::$2" │ │ │ │ │ - ); │ │ │ │ │ - yx = !!this.yx[crs]; │ │ │ │ │ + "element": function(node, obj) { │ │ │ │ │ + var type; │ │ │ │ │ + if (obj.elements) { │ │ │ │ │ + var element = {}; │ │ │ │ │ + var attributes = node.attributes; │ │ │ │ │ + var attr; │ │ │ │ │ + for (var i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ + attr = attributes[i]; │ │ │ │ │ + element[attr.name] = attr.value; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + type = element.type; │ │ │ │ │ + if (!type) { │ │ │ │ │ + type = {}; │ │ │ │ │ + this.readChildNodes(node, type); │ │ │ │ │ + element.restriction = type; │ │ │ │ │ + element.type = type.base; │ │ │ │ │ + } │ │ │ │ │ + var fullType = type.base || type; │ │ │ │ │ + element.localType = fullType.split(":").pop(); │ │ │ │ │ + obj.elements.push(element); │ │ │ │ │ + this.readChildNodes(node, element); │ │ │ │ │ } │ │ │ │ │ - if (yx) { │ │ │ │ │ - obj.topLeftCorner = new OpenLayers.LonLat( │ │ │ │ │ - coords[1], coords[0] │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - obj.topLeftCorner = new OpenLayers.LonLat( │ │ │ │ │ - coords[0], coords[1] │ │ │ │ │ - ); │ │ │ │ │ + │ │ │ │ │ + if (obj.complexTypes) { │ │ │ │ │ + type = node.getAttribute("type"); │ │ │ │ │ + var localType = type.split(":").pop(); │ │ │ │ │ + obj.customTypes[localType] = { │ │ │ │ │ + "name": node.getAttribute("name"), │ │ │ │ │ + "type": type │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - "TileWidth": function(node, obj) { │ │ │ │ │ - obj.tileWidth = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "TileHeight": function(node, obj) { │ │ │ │ │ - obj.tileHeight = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "MatrixWidth": function(node, obj) { │ │ │ │ │ - obj.matrixWidth = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "MatrixHeight": function(node, obj) { │ │ │ │ │ - obj.matrixHeight = parseInt(this.getChildValue(node)); │ │ │ │ │ + "annotation": function(node, obj) { │ │ │ │ │ + obj.annotation = {}; │ │ │ │ │ + this.readChildNodes(node, obj.annotation); │ │ │ │ │ }, │ │ │ │ │ - "ResourceURL": function(node, obj) { │ │ │ │ │ - obj.resourceUrl = obj.resourceUrl || {}; │ │ │ │ │ - var resourceType = node.getAttribute("resourceType"); │ │ │ │ │ - if (!obj.resourceUrls) { │ │ │ │ │ - obj.resourceUrls = []; │ │ │ │ │ + "appinfo": function(node, obj) { │ │ │ │ │ + if (!obj.appinfo) { │ │ │ │ │ + obj.appinfo = []; │ │ │ │ │ } │ │ │ │ │ - var resourceUrl = obj.resourceUrl[resourceType] = { │ │ │ │ │ - format: node.getAttribute("format"), │ │ │ │ │ - template: node.getAttribute("template"), │ │ │ │ │ - resourceType: resourceType │ │ │ │ │ - }; │ │ │ │ │ - obj.resourceUrls.push(resourceUrl); │ │ │ │ │ - }, │ │ │ │ │ - // not used for now, can be added in the future though │ │ │ │ │ - /*"Themes": function(node, obj) { │ │ │ │ │ - obj.themes = []; │ │ │ │ │ - this.readChildNodes(node, obj.themes); │ │ │ │ │ - }, │ │ │ │ │ - "Theme": function(node, obj) { │ │ │ │ │ - var theme = {}; │ │ │ │ │ - this.readChildNodes(node, theme); │ │ │ │ │ - obj.push(theme); │ │ │ │ │ - },*/ │ │ │ │ │ - "WSDL": function(node, obj) { │ │ │ │ │ - obj.wsdl = {}; │ │ │ │ │ - obj.wsdl.href = node.getAttribute("xlink:href"); │ │ │ │ │ - // TODO: other attributes of <WSDL> element │ │ │ │ │ - }, │ │ │ │ │ - "ServiceMetadataURL": function(node, obj) { │ │ │ │ │ - obj.serviceMetadataUrl = {}; │ │ │ │ │ - obj.serviceMetadataUrl.href = node.getAttribute("xlink:href"); │ │ │ │ │ - // TODO: other attributes of <ServiceMetadataURL> element │ │ │ │ │ - }, │ │ │ │ │ - "LegendURL": function(node, obj) { │ │ │ │ │ - obj.legend = {}; │ │ │ │ │ - obj.legend.href = node.getAttribute("xlink:href"); │ │ │ │ │ - obj.legend.format = node.getAttribute("format"); │ │ │ │ │ + obj.appinfo.push(this.getChildValue(node)); │ │ │ │ │ }, │ │ │ │ │ - "Dimension": function(node, obj) { │ │ │ │ │ - var dimension = { │ │ │ │ │ - values: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, dimension); │ │ │ │ │ - obj.dimensions.push(dimension); │ │ │ │ │ + "documentation": function(node, obj) { │ │ │ │ │ + if (!obj.documentation) { │ │ │ │ │ + obj.documentation = []; │ │ │ │ │ + } │ │ │ │ │ + var value = this.getChildValue(node); │ │ │ │ │ + obj.documentation.push({ │ │ │ │ │ + lang: node.getAttribute("xml:lang"), │ │ │ │ │ + textContent: value.replace(this.regExes.trimSpace, "") │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ - "Default": function(node, obj) { │ │ │ │ │ - obj["default"] = this.getChildValue(node); │ │ │ │ │ + "simpleType": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ }, │ │ │ │ │ - "Value": function(node, obj) { │ │ │ │ │ - obj.values.push(this.getChildValue(node)); │ │ │ │ │ + "restriction": function(node, obj) { │ │ │ │ │ + obj.base = node.getAttribute("base"); │ │ │ │ │ + this.readRestriction(node, obj); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "ows": OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMTSCapabilities.v1_0_0" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WPSCapabilities/v1_0_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WPSCapabilities.js │ │ │ │ │ - * @requires OpenLayers/Format/OWSCommon/v1_1_0.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WPSCapabilities.v1_0_0 │ │ │ │ │ - * Read WPS Capabilities version 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WPSCapabilities.v1_0_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.XML, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ - wps: "http://www.opengis.net/wps/1.0.0", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ + * Method: readRestriction │ │ │ │ │ + * Reads restriction defined in the child nodes of a restriction element │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} the node to parse │ │ │ │ │ + * obj - {Object} the object that receives the read result │ │ │ │ │ */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ + readRestriction: function(node, obj) { │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + var child, nodeName, value; │ │ │ │ │ + for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ + child = children[i]; │ │ │ │ │ + if (child.nodeType == 1) { │ │ │ │ │ + nodeName = child.nodeName.split(":").pop(); │ │ │ │ │ + value = child.getAttribute("value"); │ │ │ │ │ + if (!obj[nodeName]) { │ │ │ │ │ + obj[nodeName] = value; │ │ │ │ │ + } else { │ │ │ │ │ + if (typeof obj[nodeName] == "string") { │ │ │ │ │ + obj[nodeName] = [obj[nodeName]]; │ │ │ │ │ + } │ │ │ │ │ + obj[nodeName].push(value); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WPSCapabilities.v1_0_0 │ │ │ │ │ - * Create a new parser for WPS capabilities version 1.0.0. │ │ │ │ │ + * Method: read │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return info about the WPS. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * data - {DOMElement|String} A WFS DescribeFeatureType document. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} Information about the WPS service. │ │ │ │ │ + * {Object} An object representing the WFS DescribeFeatureType response. │ │ │ │ │ */ │ │ │ │ │ read: function(data) { │ │ │ │ │ if (typeof data == "string") { │ │ │ │ │ data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ } │ │ │ │ │ if (data && data.nodeType == 9) { │ │ │ │ │ data = data.documentElement; │ │ │ │ │ } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - return capabilities; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wps": { │ │ │ │ │ - "Capabilities": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "ProcessOfferings": function(node, obj) { │ │ │ │ │ - obj.processOfferings = {}; │ │ │ │ │ - this.readChildNodes(node, obj.processOfferings); │ │ │ │ │ - }, │ │ │ │ │ - "Process": function(node, processOfferings) { │ │ │ │ │ - var processVersion = this.getAttributeNS(node, this.namespaces.wps, "processVersion"); │ │ │ │ │ - var process = { │ │ │ │ │ - processVersion: processVersion │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, process); │ │ │ │ │ - processOfferings[process.identifier] = process; │ │ │ │ │ - }, │ │ │ │ │ - "Languages": function(node, obj) { │ │ │ │ │ - obj.languages = []; │ │ │ │ │ - this.readChildNodes(node, obj.languages); │ │ │ │ │ - }, │ │ │ │ │ - "Default": function(node, languages) { │ │ │ │ │ - var language = { │ │ │ │ │ - isDefault: true │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, language); │ │ │ │ │ - languages.push(language); │ │ │ │ │ - }, │ │ │ │ │ - "Supported": function(node, languages) { │ │ │ │ │ - var language = {}; │ │ │ │ │ - this.readChildNodes(node, language); │ │ │ │ │ - languages.push(language); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "ows": OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ + var schema = {}; │ │ │ │ │ + if (data.nodeName.split(":").pop() === 'ExceptionReport') { │ │ │ │ │ + // an exception must have occurred, so parse it │ │ │ │ │ + var parser = new OpenLayers.Format.OGCExceptionReport(); │ │ │ │ │ + schema.error = parser.read(data); │ │ │ │ │ + } else { │ │ │ │ │ + this.readNode(data, schema); │ │ │ │ │ + } │ │ │ │ │ + return schema; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WPSCapabilities.v1_0_0" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSDescribeFeatureType" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSDescribeLayer/v1_1.js │ │ │ │ │ + OpenLayers/Format/CSWGetDomain.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/WMSDescribeLayer.js │ │ │ │ │ - * @requires OpenLayers/Format/OGCExceptionReport.js │ │ │ │ │ + * @requires OpenLayers/Format.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WMSDescribeLayer.v1_1_1 │ │ │ │ │ - * Read SLD WMS DescribeLayer response for WMS 1.1.X │ │ │ │ │ - * WMS 1.1.X is tightly coupled to SLD 1.0.0 │ │ │ │ │ - * │ │ │ │ │ - * Example DescribeLayer request: │ │ │ │ │ - * http://demo.opengeo.org/geoserver/wms?request=DescribeLayer&version=1.1.1&layers=topp:states │ │ │ │ │ + * Class: OpenLayers.Format.CSWGetDomain │ │ │ │ │ + * Default version is 2.0.2. │ │ │ │ │ * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMSDescribeLayer> │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Format>} A CSWGetDomain format of the given version. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WMSDescribeLayer.v1_1_1 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WMSDescribeLayer, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMSDescribeLayer │ │ │ │ │ - * Create a new parser for WMS DescribeLayer responses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.WMSDescribeLayer.prototype.initialize.apply(this, │ │ │ │ │ - [options]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read DescribeLayer data from a string, and return the response. │ │ │ │ │ - * The OGC defines 2 formats which are allowed for output, │ │ │ │ │ - * so we need to parse these 2 types for version 1.1.X │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Object with a layerDescriptions property, which holds an Array │ │ │ │ │ - * of {<LayerDescription>} objects which have: │ │ │ │ │ - * - {String} owsType: WFS/WCS │ │ │ │ │ - * - {String} owsURL: the online resource │ │ │ │ │ - * - {String} typeName: the name of the typename on the owsType service │ │ │ │ │ - * - {String} layerName: the name of the WMS layer we did a lookup for │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - var root = data.documentElement; │ │ │ │ │ - var children = root.childNodes; │ │ │ │ │ - var describelayer = { │ │ │ │ │ - layerDescriptions: [] │ │ │ │ │ - }; │ │ │ │ │ - var childNode, nodeName; │ │ │ │ │ - for (var i = 0; i < children.length; ++i) { │ │ │ │ │ - childNode = children[i]; │ │ │ │ │ - nodeName = childNode.nodeName; │ │ │ │ │ - if (nodeName == 'LayerDescription') { │ │ │ │ │ - var layerName = childNode.getAttribute('name'); │ │ │ │ │ - var owsType = ''; │ │ │ │ │ - var owsURL = ''; │ │ │ │ │ - var typeName = ''; │ │ │ │ │ - // check for owsType and owsURL attributes │ │ │ │ │ - if (childNode.getAttribute('owsType')) { │ │ │ │ │ - owsType = childNode.getAttribute('owsType'); │ │ │ │ │ - owsURL = childNode.getAttribute('owsURL'); │ │ │ │ │ - } else { │ │ │ │ │ - // look for wfs or wcs attribute │ │ │ │ │ - if (childNode.getAttribute('wfs') != '') { │ │ │ │ │ - owsType = 'WFS'; │ │ │ │ │ - owsURL = childNode.getAttribute('wfs'); │ │ │ │ │ - } else if (childNode.getAttribute('wcs') != '') { │ │ │ │ │ - owsType = 'WCS'; │ │ │ │ │ - owsURL = childNode.getAttribute('wcs'); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // look for Query child │ │ │ │ │ - var query = childNode.getElementsByTagName('Query'); │ │ │ │ │ - if (query.length > 0) { │ │ │ │ │ - typeName = query[0].getAttribute('typeName'); │ │ │ │ │ - if (!typeName) { │ │ │ │ │ - // because of Ionic bug │ │ │ │ │ - typeName = query[0].getAttribute('typename'); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var layerDescription = { │ │ │ │ │ - layerName: layerName, │ │ │ │ │ - owsType: owsType, │ │ │ │ │ - owsURL: owsURL, │ │ │ │ │ - typeName: typeName │ │ │ │ │ - }; │ │ │ │ │ - describelayer.layerDescriptions.push(layerDescription); │ │ │ │ │ - │ │ │ │ │ - //TODO do this in deprecated.js instead: │ │ │ │ │ - // array style index for backwards compatibility │ │ │ │ │ - describelayer.length = describelayer.layerDescriptions.length; │ │ │ │ │ - describelayer[describelayer.length - 1] = layerDescription; │ │ │ │ │ - │ │ │ │ │ - } else if (nodeName == 'ServiceException') { │ │ │ │ │ - // an exception must have occurred, so parse it │ │ │ │ │ - var parser = new OpenLayers.Format.OGCExceptionReport(); │ │ │ │ │ - return { │ │ │ │ │ - error: parser.read(data) │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return describelayer; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSDescribeLayer.v1_1_1" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ +OpenLayers.Format.CSWGetDomain = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options, OpenLayers.Format.CSWGetDomain.DEFAULTS │ │ │ │ │ + ); │ │ │ │ │ + var cls = OpenLayers.Format.CSWGetDomain["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported CSWGetDomain version: " + options.version; │ │ │ │ │ + } │ │ │ │ │ + return new cls(options); │ │ │ │ │ +}; │ │ │ │ │ │ │ │ │ │ -// Version alias - workaround for http://trac.osgeo.org/mapserver/ticket/2257 │ │ │ │ │ -OpenLayers.Format.WMSDescribeLayer.v1_1_0 = │ │ │ │ │ - OpenLayers.Format.WMSDescribeLayer.v1_1_1; │ │ │ │ │ +/** │ │ │ │ │ + * Constant: DEFAULTS │ │ │ │ │ + * {Object} Default properties for the CSWGetDomain format. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.CSWGetDomain.DEFAULTS = { │ │ │ │ │ + "version": "2.0.2" │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/ArcXML/Features.js │ │ │ │ │ + OpenLayers/Format/WMTSCapabilities.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/ArcXML.js │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.ArcXML.Features │ │ │ │ │ - * Read/Write ArcXML features. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Format.ArcXML.Features> constructor. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Format.WMTSCapabilities │ │ │ │ │ + * Read WMTS Capabilities. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.ArcXML.Features = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Format.WMTSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Format.ArcXML.Features │ │ │ │ │ - * Create a new parser/writer for ArcXML Features. Create an instance of this class │ │ │ │ │ - * to get a set of features from an ArcXML response. │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.0.0". │ │ │ │ │ + */ │ │ │ │ │ + defaultVersion: "1.0.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: yx │ │ │ │ │ + * {Object} Members in the yx object are used to determine if a CRS URN │ │ │ │ │ + * corresponds to a CRS with y,x axis order. Member names are CRS URNs │ │ │ │ │ + * and values are boolean. By default, the following CRS URN are │ │ │ │ │ + * assumed to correspond to a CRS with y,x axis order: │ │ │ │ │ + * │ │ │ │ │ + * * urn:ogc:def:crs:EPSG::4326 │ │ │ │ │ + */ │ │ │ │ │ + yx: { │ │ │ │ │ + "urn:ogc:def:crs:EPSG::4326": true │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMTSCapabilities │ │ │ │ │ + * Create a new parser for WMTS capabilities. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * options - {Object} An optional object whose properties will be set on │ │ │ │ │ * this instance. │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIMethod: read │ │ │ │ │ - * Read data from a string of ArcXML, and return a set of OpenLayers features. │ │ │ │ │ - * │ │ │ │ │ + * Read capabilities data from a string, and return information about │ │ │ │ │ + * the service (offering and observedProperty mostly). │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} A collection of features. │ │ │ │ │ + * {Object} Info about the WMTS Capabilities │ │ │ │ │ */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - var axl = new OpenLayers.Format.ArcXML(); │ │ │ │ │ - var parsed = axl.read(data); │ │ │ │ │ - │ │ │ │ │ - return parsed.features.feature; │ │ │ │ │ - } │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WFST/v1_0_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WFST/v1.js │ │ │ │ │ - * @requires OpenLayers/Format/Filter/v1_0_0.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: createLayer │ │ │ │ │ + * Create a WMTS layer given a capabilities object. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * capabilities - {Object} The object returned from a <read> call to this │ │ │ │ │ + * format. │ │ │ │ │ + * config - {Object} Configuration properties for the layer. Defaults for │ │ │ │ │ + * the layer will apply if not provided. │ │ │ │ │ + * │ │ │ │ │ + * Required config properties: │ │ │ │ │ + * layer - {String} The layer identifier. │ │ │ │ │ + * │ │ │ │ │ + * Optional config properties: │ │ │ │ │ + * matrixSet - {String} The matrix set identifier, required if there is │ │ │ │ │ + * more than one matrix set in the layer capabilities. │ │ │ │ │ + * style - {String} The name of the style │ │ │ │ │ + * format - {String} Image format for the layer. Default is the first │ │ │ │ │ + * format returned in the GetCapabilities response. │ │ │ │ │ + * param - {Object} The dimensions values eg: {"Year": "2012"} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.WMTS>} A properly configured WMTS layer. Throws an │ │ │ │ │ + * error if an incomplete config is provided. Returns undefined if no │ │ │ │ │ + * layer could be created with the provided config. │ │ │ │ │ + */ │ │ │ │ │ + createLayer: function(capabilities, config) { │ │ │ │ │ + var layer; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WFST.v1_0_0 │ │ │ │ │ - * A format for creating WFS v1.0.0 transactions. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Format.WFST.v1_0_0> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.Filter.v1_0_0> │ │ │ │ │ - * - <OpenLayers.Format.WFST.v1> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WFST.v1_0_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.Filter.v1_0_0, OpenLayers.Format.WFST.v1, { │ │ │ │ │ + // confirm required properties are supplied in config │ │ │ │ │ + if (!('layer' in config)) { │ │ │ │ │ + throw new Error("Missing property 'layer' in configuration."); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} WFS version number. │ │ │ │ │ - */ │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ + var contents = capabilities.contents; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: srsNameInQuery │ │ │ │ │ - * {Boolean} If true the reference system is passed in Query requests │ │ │ │ │ - * via the "srsName" attribute to the "wfs:Query" element, this │ │ │ │ │ - * property defaults to false as it isn't WFS 1.0.0 compliant. │ │ │ │ │ - */ │ │ │ │ │ - srsNameInQuery: false, │ │ │ │ │ + // find the layer definition with the given identifier │ │ │ │ │ + var layers = contents.layers; │ │ │ │ │ + var layerDef; │ │ │ │ │ + for (var i = 0, ii = contents.layers.length; i < ii; ++i) { │ │ │ │ │ + if (contents.layers[i].identifier === config.layer) { │ │ │ │ │ + layerDef = contents.layers[i]; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!layerDef) { │ │ │ │ │ + throw new Error("Layer not found"); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: schemaLocations │ │ │ │ │ - * {Object} Properties are namespace aliases, values are schema locations. │ │ │ │ │ - */ │ │ │ │ │ - schemaLocations: { │ │ │ │ │ - "wfs": "http://schemas.opengis.net/wfs/1.0.0/WFS-transaction.xsd" │ │ │ │ │ - }, │ │ │ │ │ + var format = config.format; │ │ │ │ │ + if (!format && layerDef.formats && layerDef.formats.length) { │ │ │ │ │ + format = layerDef.formats[0]; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WFST.v1_0_0 │ │ │ │ │ - * A class for parsing and generating WFS v1.0.0 transactions. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ - * featureNS - {String} Feature namespace (optional). │ │ │ │ │ - * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ - * if featureNS is provided). Default is 'feature'. │ │ │ │ │ - * geometryName - {String} Name of geometry attribute. Default is 'the_geom'. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.Filter.v1_0_0.prototype.initialize.apply(this, [options]); │ │ │ │ │ - OpenLayers.Format.WFST.v1.prototype.initialize.apply(this, [options]); │ │ │ │ │ - }, │ │ │ │ │ + // find the matrixSet definition │ │ │ │ │ + var matrixSet; │ │ │ │ │ + if (config.matrixSet) { │ │ │ │ │ + matrixSet = contents.tileMatrixSets[config.matrixSet]; │ │ │ │ │ + } else if (layerDef.tileMatrixSetLinks.length >= 1) { │ │ │ │ │ + matrixSet = contents.tileMatrixSets[ │ │ │ │ │ + layerDef.tileMatrixSetLinks[0].tileMatrixSet]; │ │ │ │ │ + } │ │ │ │ │ + if (!matrixSet) { │ │ │ │ │ + throw new Error("matrixSet not found"); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: readNode │ │ │ │ │ - * Shorthand for applying one of the named readers given the node │ │ │ │ │ - * namespace and local name. Readers take two args (node, obj) and │ │ │ │ │ - * generally extend or modify the second. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} The node to be read (required). │ │ │ │ │ - * obj - {Object} The object to be modified (optional). │ │ │ │ │ - * first - {Boolean} Should be set to true for the first node read. This │ │ │ │ │ - * is usually the readNode call in the read method. Without this being │ │ │ │ │ - * set, auto-configured properties will stick on subsequent reads. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The input object, modified (or a new one if none was provided). │ │ │ │ │ - */ │ │ │ │ │ - readNode: function(node, obj, first) { │ │ │ │ │ - // Not the superclass, only the mixin classes inherit from │ │ │ │ │ - // Format.GML.v2. We need this because we don't want to get readNode │ │ │ │ │ - // from the superclass's superclass, which is OpenLayers.Format.XML. │ │ │ │ │ - return OpenLayers.Format.GML.v2.prototype.readNode.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + // get the default style for the layer │ │ │ │ │ + var style; │ │ │ │ │ + for (var i = 0, ii = layerDef.styles.length; i < ii; ++i) { │ │ │ │ │ + style = layerDef.styles[i]; │ │ │ │ │ + if (style.isDefault) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "WFS_TransactionResponse": function(node, obj) { │ │ │ │ │ - obj.insertIds = []; │ │ │ │ │ - obj.success = false; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "InsertResult": function(node, container) { │ │ │ │ │ - var obj = { │ │ │ │ │ - fids: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - container.insertIds = container.insertIds.concat(obj.fids); │ │ │ │ │ - }, │ │ │ │ │ - "TransactionResult": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "Status": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "SUCCESS": function(node, obj) { │ │ │ │ │ - obj.success = true; │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WFST.v1.prototype.readers["wfs"]), │ │ │ │ │ - "gml": OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ - "feature": OpenLayers.Format.GML.v2.prototype.readers["feature"], │ │ │ │ │ - "ogc": OpenLayers.Format.Filter.v1_0_0.prototype.readers["ogc"] │ │ │ │ │ - }, │ │ │ │ │ + var requestEncoding = config.requestEncoding; │ │ │ │ │ + if (!requestEncoding) { │ │ │ │ │ + requestEncoding = "KVP"; │ │ │ │ │ + if (capabilities.operationsMetadata.GetTile.dcp.http) { │ │ │ │ │ + var http = capabilities.operationsMetadata.GetTile.dcp.http; │ │ │ │ │ + // Get first get method │ │ │ │ │ + if (http.get[0].constraints) { │ │ │ │ │ + var constraints = http.get[0].constraints; │ │ │ │ │ + var allowedValues = constraints.GetEncoding.allowedValues; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: writers │ │ │ │ │ - * As a compliment to the readers property, this structure contains public │ │ │ │ │ - * writing functions grouped by namespace alias and named like the │ │ │ │ │ - * node names they produce. │ │ │ │ │ - */ │ │ │ │ │ - writers: { │ │ │ │ │ - "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "Query": function(options) { │ │ │ │ │ - options = OpenLayers.Util.extend({ │ │ │ │ │ - featureNS: this.featureNS, │ │ │ │ │ - featurePrefix: this.featurePrefix, │ │ │ │ │ - featureType: this.featureType, │ │ │ │ │ - srsName: this.srsName, │ │ │ │ │ - srsNameInQuery: this.srsNameInQuery │ │ │ │ │ - }, options); │ │ │ │ │ - var prefix = options.featurePrefix; │ │ │ │ │ - var node = this.createElementNSPlus("wfs:Query", { │ │ │ │ │ - attributes: { │ │ │ │ │ - typeName: (prefix ? prefix + ":" : "") + │ │ │ │ │ - options.featureType │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (options.srsNameInQuery && options.srsName) { │ │ │ │ │ - node.setAttribute("srsName", options.srsName); │ │ │ │ │ - } │ │ │ │ │ - if (options.featureNS) { │ │ │ │ │ - node.setAttribute("xmlns:" + prefix, options.featureNS); │ │ │ │ │ - } │ │ │ │ │ - if (options.propertyNames) { │ │ │ │ │ - for (var i = 0, len = options.propertyNames.length; i < len; i++) { │ │ │ │ │ - this.writeNode( │ │ │ │ │ - "ogc:PropertyName", { │ │ │ │ │ - property: options.propertyNames[i] │ │ │ │ │ - }, │ │ │ │ │ - node │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (options.filter) { │ │ │ │ │ - this.setFilterProperty(options.filter); │ │ │ │ │ - this.writeNode("ogc:Filter", options.filter, node); │ │ │ │ │ + // The OGC documentation is not clear if we should use │ │ │ │ │ + // REST or RESTful, ArcGis use RESTful, │ │ │ │ │ + // and OpenLayers use REST. │ │ │ │ │ + if (!allowedValues.KVP && │ │ │ │ │ + (allowedValues.REST || allowedValues.RESTful)) { │ │ │ │ │ + requestEncoding = "REST"; │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.WFST.v1.prototype.writers["wfs"]), │ │ │ │ │ - "gml": OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ - "feature": OpenLayers.Format.GML.v2.prototype.writers["feature"], │ │ │ │ │ - "ogc": OpenLayers.Format.Filter.v1_0_0.prototype.writers["ogc"] │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFST.v1_0_0" │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/SOSCapabilities/v1_0_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/SOSCapabilities.js │ │ │ │ │ - * @requires OpenLayers/Format/OWSCommon/v1_1_0.js │ │ │ │ │ - * @requires OpenLayers/Format/GML/v3.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.SOSCapabilities.v1_0_0 │ │ │ │ │ - * Read SOS Capabilities version 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.SOSCapabilities> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.SOSCapabilities.v1_0_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.SOSCapabilities, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ - sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ - gml: "http://www.opengis.net/gml", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ - }, │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ - */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ - }, │ │ │ │ │ + var dimensions = []; │ │ │ │ │ + var params = config.params || {}; │ │ │ │ │ + // to don't overwrite the changes in the applyDefaults │ │ │ │ │ + delete config.params; │ │ │ │ │ + for (var id = 0, ld = layerDef.dimensions.length; id < ld; id++) { │ │ │ │ │ + var dimension = layerDef.dimensions[id]; │ │ │ │ │ + dimensions.push(dimension.identifier); │ │ │ │ │ + if (!params.hasOwnProperty(dimension.identifier)) { │ │ │ │ │ + params[dimension.identifier] = dimension['default']; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.SOSCapabilities.v1_0_0 │ │ │ │ │ - * Create a new parser for SOS capabilities version 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.options = options; │ │ │ │ │ - }, │ │ │ │ │ + var projection = config.projection || matrixSet.supportedCRS.replace( │ │ │ │ │ + /urn:ogc:def:crs:(\w+):(.*:)?(\w+)$/, "$1:$3"); │ │ │ │ │ + var units = config.units || │ │ │ │ │ + (projection === "EPSG:4326" ? "degrees" : "m"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return info about the SOS. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Information about the SOS service. │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ + var resolutions = []; │ │ │ │ │ + for (var mid in matrixSet.matrixIds) { │ │ │ │ │ + if (matrixSet.matrixIds.hasOwnProperty(mid)) { │ │ │ │ │ + resolutions.push( │ │ │ │ │ + matrixSet.matrixIds[mid].scaleDenominator * 0.28E-3 / │ │ │ │ │ + OpenLayers.METERS_PER_INCH / │ │ │ │ │ + OpenLayers.INCHES_PER_UNIT[units]); │ │ │ │ │ } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - return capabilities; │ │ │ │ │ - }, │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "name": function(node, obj) { │ │ │ │ │ - obj.name = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "TimePeriod": function(node, obj) { │ │ │ │ │ - obj.timePeriod = {}; │ │ │ │ │ - this.readChildNodes(node, obj.timePeriod); │ │ │ │ │ - }, │ │ │ │ │ - "beginPosition": function(node, timePeriod) { │ │ │ │ │ - timePeriod.beginPosition = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "endPosition": function(node, timePeriod) { │ │ │ │ │ - timePeriod.endPosition = this.getChildValue(node); │ │ │ │ │ + var url; │ │ │ │ │ + if (requestEncoding === "REST" && layerDef.resourceUrls) { │ │ │ │ │ + url = []; │ │ │ │ │ + var resourceUrls = layerDef.resourceUrls, │ │ │ │ │ + resourceUrl; │ │ │ │ │ + for (var t = 0, tt = layerDef.resourceUrls.length; t < tt; ++t) { │ │ │ │ │ + resourceUrl = layerDef.resourceUrls[t]; │ │ │ │ │ + if (resourceUrl.format === format && resourceUrl.resourceType === "tile") { │ │ │ │ │ + url.push(resourceUrl.template); │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.GML.v3.prototype.readers["gml"]), │ │ │ │ │ - "sos": { │ │ │ │ │ - "Capabilities": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "Contents": function(node, obj) { │ │ │ │ │ - obj.contents = {}; │ │ │ │ │ - this.readChildNodes(node, obj.contents); │ │ │ │ │ - }, │ │ │ │ │ - "ObservationOfferingList": function(node, contents) { │ │ │ │ │ - contents.offeringList = {}; │ │ │ │ │ - this.readChildNodes(node, contents.offeringList); │ │ │ │ │ - }, │ │ │ │ │ - "ObservationOffering": function(node, offeringList) { │ │ │ │ │ - var id = this.getAttributeNS(node, this.namespaces.gml, "id"); │ │ │ │ │ - offeringList[id] = { │ │ │ │ │ - procedures: [], │ │ │ │ │ - observedProperties: [], │ │ │ │ │ - featureOfInterestIds: [], │ │ │ │ │ - responseFormats: [], │ │ │ │ │ - resultModels: [], │ │ │ │ │ - responseModes: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, offeringList[id]); │ │ │ │ │ - }, │ │ │ │ │ - "time": function(node, offering) { │ │ │ │ │ - offering.time = {}; │ │ │ │ │ - this.readChildNodes(node, offering.time); │ │ │ │ │ - }, │ │ │ │ │ - "procedure": function(node, offering) { │ │ │ │ │ - offering.procedures.push(this.getAttributeNS(node, │ │ │ │ │ - this.namespaces.xlink, "href")); │ │ │ │ │ - }, │ │ │ │ │ - "observedProperty": function(node, offering) { │ │ │ │ │ - offering.observedProperties.push(this.getAttributeNS(node, │ │ │ │ │ - this.namespaces.xlink, "href")); │ │ │ │ │ - }, │ │ │ │ │ - "featureOfInterest": function(node, offering) { │ │ │ │ │ - offering.featureOfInterestIds.push(this.getAttributeNS(node, │ │ │ │ │ - this.namespaces.xlink, "href")); │ │ │ │ │ - }, │ │ │ │ │ - "responseFormat": function(node, offering) { │ │ │ │ │ - offering.responseFormats.push(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "resultModel": function(node, offering) { │ │ │ │ │ - offering.resultModels.push(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "responseMode": function(node, offering) { │ │ │ │ │ - offering.responseModes.push(this.getChildValue(node)); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + var httpGet = capabilities.operationsMetadata.GetTile.dcp.http.get; │ │ │ │ │ + url = []; │ │ │ │ │ + var constraint; │ │ │ │ │ + for (var i = 0, ii = httpGet.length; i < ii; i++) { │ │ │ │ │ + constraint = httpGet[i].constraints; │ │ │ │ │ + if (!constraint || (constraint && constraint.GetEncoding.allowedValues[requestEncoding])) { │ │ │ │ │ + url.push(httpGet[i].url); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "ows": OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ - }, │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SOSCapabilities.v1_0_0" │ │ │ │ │ + return new OpenLayers.Layer.WMTS( │ │ │ │ │ + OpenLayers.Util.applyDefaults(config, { │ │ │ │ │ + url: url, │ │ │ │ │ + requestEncoding: requestEncoding, │ │ │ │ │ + name: layerDef.title, │ │ │ │ │ + style: style.identifier, │ │ │ │ │ + format: format, │ │ │ │ │ + matrixIds: matrixSet.matrixIds, │ │ │ │ │ + matrixSet: matrixSet.identifier, │ │ │ │ │ + projection: projection, │ │ │ │ │ + units: units, │ │ │ │ │ + resolutions: config.isBaseLayer === false ? undefined : resolutions, │ │ │ │ │ + serverResolutions: resolutions, │ │ │ │ │ + tileFullExtent: matrixSet.bounds, │ │ │ │ │ + dimensions: dimensions, │ │ │ │ │ + params: params │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMTSCapabilities" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMC/v1.js │ │ │ │ │ + OpenLayers/Format/OSM.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/WMC.js │ │ │ │ │ * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ + * @requires OpenLayers/Projection.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WMC.v1 │ │ │ │ │ - * Superclass for WMC version 1 parsers. │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.OSM │ │ │ │ │ + * OSM parser. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Format.OSM> constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Format.XML> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WMC.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - ol: "http://openlayers.org/context", │ │ │ │ │ - wmc: "http://www.opengis.net/context", │ │ │ │ │ - sld: "http://www.opengis.net/sld", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ - }, │ │ │ │ │ +OpenLayers.Format.OSM = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} Schema location for a particular minor version. │ │ │ │ │ + * APIProperty: checkTags │ │ │ │ │ + * {Boolean} Should tags be checked to determine whether something │ │ │ │ │ + * should be treated as a seperate node. Will slow down parsing. │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - schemaLocation: "", │ │ │ │ │ + checkTags: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getNamespacePrefix │ │ │ │ │ - * Get the namespace prefix for a given uri from the <namespaces> object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A namespace prefix or null if none found. │ │ │ │ │ + * Property: interestingTagsExclude │ │ │ │ │ + * {Array} List of tags to exclude from 'interesting' checks on nodes. │ │ │ │ │ + * Must be set when creating the format. Will only be used if checkTags │ │ │ │ │ + * is set. │ │ │ │ │ */ │ │ │ │ │ - getNamespacePrefix: function(uri) { │ │ │ │ │ - var prefix = null; │ │ │ │ │ - if (uri == null) { │ │ │ │ │ - prefix = this.namespaces[this.defaultPrefix]; │ │ │ │ │ - } else { │ │ │ │ │ - for (prefix in this.namespaces) { │ │ │ │ │ - if (this.namespaces[prefix] == uri) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return prefix; │ │ │ │ │ - }, │ │ │ │ │ + interestingTagsExclude: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ + * APIProperty: areaTags │ │ │ │ │ + * {Array} List of tags indicating that something is an area. │ │ │ │ │ + * Must be set when creating the format. Will only be used if │ │ │ │ │ + * checkTags is true. │ │ │ │ │ */ │ │ │ │ │ - defaultPrefix: "wmc", │ │ │ │ │ + areaTags: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: rootPrefix │ │ │ │ │ - * {String} Prefix on the root node that maps to the context namespace URI. │ │ │ │ │ + * Constructor: OpenLayers.Format.OSM │ │ │ │ │ + * Create a new parser for OSM. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - rootPrefix: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + var layer_defaults = { │ │ │ │ │ + 'interestingTagsExclude': ['source', 'source_ref', │ │ │ │ │ + 'source:ref', 'history', 'attribution', 'created_by' │ │ │ │ │ + ], │ │ │ │ │ + 'areaTags': ['area', 'building', 'leisure', 'tourism', 'ruins', │ │ │ │ │ + 'historic', 'landuse', 'military', 'natural', 'sport' │ │ │ │ │ + ] │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultStyleName │ │ │ │ │ - * {String} Style name used if layer has no style param. Default is "". │ │ │ │ │ - */ │ │ │ │ │ - defaultStyleName: "", │ │ │ │ │ + layer_defaults = OpenLayers.Util.extend(layer_defaults, options); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultStyleTitle │ │ │ │ │ - * {String} Default style title. Default is "Default". │ │ │ │ │ - */ │ │ │ │ │ - defaultStyleTitle: "Default", │ │ │ │ │ + var interesting = {}; │ │ │ │ │ + for (var i = 0; i < layer_defaults.interestingTagsExclude.length; i++) { │ │ │ │ │ + interesting[layer_defaults.interestingTagsExclude[i]] = true; │ │ │ │ │ + } │ │ │ │ │ + layer_defaults.interestingTagsExclude = interesting; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMC.v1 │ │ │ │ │ - * Instances of this class are not created directly. Use the │ │ │ │ │ - * <OpenLayers.Format.WMC> constructor instead. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + var area = {}; │ │ │ │ │ + for (var i = 0; i < layer_defaults.areaTags.length; i++) { │ │ │ │ │ + area[layer_defaults.areaTags[i]] = true; │ │ │ │ │ + } │ │ │ │ │ + layer_defaults.areaTags = area; │ │ │ │ │ + │ │ │ │ │ + // OSM coordinates are always in longlat WGS84 │ │ │ │ │ + this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [layer_defaults]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read │ │ │ │ │ - * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Return a list of features from a OSM doc │ │ │ │ │ + │ │ │ │ │ + * Parameters: │ │ │ │ │ + * doc - {Element} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array} List of named layers. │ │ │ │ │ + * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + read: function(doc) { │ │ │ │ │ + if (typeof doc == "string") { │ │ │ │ │ + doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]); │ │ │ │ │ } │ │ │ │ │ - var root = data.documentElement; │ │ │ │ │ - this.rootPrefix = root.prefix; │ │ │ │ │ - var context = { │ │ │ │ │ - version: root.getAttribute("version") │ │ │ │ │ - }; │ │ │ │ │ - this.runChildNodes(context, root); │ │ │ │ │ - return context; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: runChildNodes │ │ │ │ │ - */ │ │ │ │ │ - runChildNodes: function(obj, node) { │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - var childNode, processor, prefix, local; │ │ │ │ │ - for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ - childNode = children[i]; │ │ │ │ │ - if (childNode.nodeType == 1) { │ │ │ │ │ - prefix = this.getNamespacePrefix(childNode.namespaceURI); │ │ │ │ │ - local = childNode.nodeName.split(":").pop(); │ │ │ │ │ - processor = this["read_" + prefix + "_" + local]; │ │ │ │ │ - if (processor) { │ │ │ │ │ - processor.apply(this, [obj, childNode]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + var nodes = this.getNodes(doc); │ │ │ │ │ + var ways = this.getWays(doc); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_General │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_General: function(context, node) { │ │ │ │ │ - this.runChildNodes(context, node); │ │ │ │ │ - }, │ │ │ │ │ + // Geoms will contain at least ways.length entries. │ │ │ │ │ + var feat_list = new Array(ways.length); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_BoundingBox │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_BoundingBox: function(context, node) { │ │ │ │ │ - context.projection = node.getAttribute("SRS"); │ │ │ │ │ - context.bounds = new OpenLayers.Bounds( │ │ │ │ │ - node.getAttribute("minx"), node.getAttribute("miny"), │ │ │ │ │ - node.getAttribute("maxx"), node.getAttribute("maxy") │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ + for (var i = 0; i < ways.length; i++) { │ │ │ │ │ + // We know the minimal of this one ahead of time. (Could be -1 │ │ │ │ │ + // due to areas/polygons) │ │ │ │ │ + var point_list = new Array(ways[i].nodes.length); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_LayerList │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_LayerList: function(context, node) { │ │ │ │ │ - // layersContext is an array containing info for each layer │ │ │ │ │ - context.layersContext = []; │ │ │ │ │ - this.runChildNodes(context, node); │ │ │ │ │ - }, │ │ │ │ │ + var poly = this.isWayArea(ways[i]) ? 1 : 0; │ │ │ │ │ + for (var j = 0; j < ways[i].nodes.length; j++) { │ │ │ │ │ + var node = nodes[ways[i].nodes[j]]; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_Layer │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_Layer: function(context, node) { │ │ │ │ │ - var layerContext = { │ │ │ │ │ - visibility: (node.getAttribute("hidden") != "1"), │ │ │ │ │ - queryable: (node.getAttribute("queryable") == "1"), │ │ │ │ │ - formats: [], │ │ │ │ │ - styles: [], │ │ │ │ │ - metadata: {} │ │ │ │ │ - }; │ │ │ │ │ + var point = new OpenLayers.Geometry.Point(node.lon, node.lat); │ │ │ │ │ │ │ │ │ │ - this.runChildNodes(layerContext, node); │ │ │ │ │ - // set properties common to multiple objects on layer options/params │ │ │ │ │ - context.layersContext.push(layerContext); │ │ │ │ │ - }, │ │ │ │ │ + // Since OSM is topological, we stash the node ID internally. │ │ │ │ │ + point.osm_id = parseInt(ways[i].nodes[j]); │ │ │ │ │ + point_list[j] = point; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_Extension │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_Extension: function(obj, node) { │ │ │ │ │ - this.runChildNodes(obj, node); │ │ │ │ │ - }, │ │ │ │ │ + // We don't display nodes if they're used inside other │ │ │ │ │ + // elements. │ │ │ │ │ + node.used = true; │ │ │ │ │ + } │ │ │ │ │ + var geometry = null; │ │ │ │ │ + if (poly) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.Polygon( │ │ │ │ │ + new OpenLayers.Geometry.LinearRing(point_list)); │ │ │ │ │ + } else { │ │ │ │ │ + geometry = new OpenLayers.Geometry.LineString(point_list); │ │ │ │ │ + } │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry.transform(this.externalProjection, │ │ │ │ │ + this.internalProjection); │ │ │ │ │ + } │ │ │ │ │ + var feat = new OpenLayers.Feature.Vector(geometry, │ │ │ │ │ + ways[i].tags); │ │ │ │ │ + feat.osm_id = parseInt(ways[i].id); │ │ │ │ │ + feat.fid = "way." + feat.osm_id; │ │ │ │ │ + feat_list[i] = feat; │ │ │ │ │ + } │ │ │ │ │ + for (var node_id in nodes) { │ │ │ │ │ + var node = nodes[node_id]; │ │ │ │ │ + if (!node.used || this.checkTags) { │ │ │ │ │ + var tags = null; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_ol_units │ │ │ │ │ - */ │ │ │ │ │ - read_ol_units: function(layerContext, node) { │ │ │ │ │ - layerContext.units = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ + if (this.checkTags) { │ │ │ │ │ + var result = this.getTags(node.node, true); │ │ │ │ │ + if (node.used && !result[1]) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + tags = result[0]; │ │ │ │ │ + } else { │ │ │ │ │ + tags = this.getTags(node.node); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_ol_maxExtent │ │ │ │ │ - */ │ │ │ │ │ - read_ol_maxExtent: function(obj, node) { │ │ │ │ │ - var bounds = new OpenLayers.Bounds( │ │ │ │ │ - node.getAttribute("minx"), node.getAttribute("miny"), │ │ │ │ │ - node.getAttribute("maxx"), node.getAttribute("maxy") │ │ │ │ │ - ); │ │ │ │ │ - obj.maxExtent = bounds; │ │ │ │ │ + var feat = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.Point(node['lon'], node['lat']), │ │ │ │ │ + tags); │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + feat.geometry.transform(this.externalProjection, │ │ │ │ │ + this.internalProjection); │ │ │ │ │ + } │ │ │ │ │ + feat.osm_id = parseInt(node_id); │ │ │ │ │ + feat.fid = "node." + feat.osm_id; │ │ │ │ │ + feat_list.push(feat); │ │ │ │ │ + } │ │ │ │ │ + // Memory cleanup │ │ │ │ │ + node.node = null; │ │ │ │ │ + } │ │ │ │ │ + return feat_list; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_ol_transparent │ │ │ │ │ + * Method: getNodes │ │ │ │ │ + * Return the node items from a doc. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * doc - {DOMElement} node to parse tags from │ │ │ │ │ */ │ │ │ │ │ - read_ol_transparent: function(layerContext, node) { │ │ │ │ │ - layerContext.transparent = this.getChildValue(node); │ │ │ │ │ + getNodes: function(doc) { │ │ │ │ │ + var node_list = doc.getElementsByTagName("node"); │ │ │ │ │ + var nodes = {}; │ │ │ │ │ + for (var i = 0; i < node_list.length; i++) { │ │ │ │ │ + var node = node_list[i]; │ │ │ │ │ + var id = node.getAttribute("id"); │ │ │ │ │ + nodes[id] = { │ │ │ │ │ + 'lat': node.getAttribute("lat"), │ │ │ │ │ + 'lon': node.getAttribute("lon"), │ │ │ │ │ + 'node': node │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + return nodes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_ol_numZoomLevels │ │ │ │ │ + * Method: getWays │ │ │ │ │ + * Return the way items from a doc. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * doc - {DOMElement} node to parse tags from │ │ │ │ │ */ │ │ │ │ │ - read_ol_numZoomLevels: function(layerContext, node) { │ │ │ │ │ - layerContext.numZoomLevels = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ + getWays: function(doc) { │ │ │ │ │ + var way_list = doc.getElementsByTagName("way"); │ │ │ │ │ + var return_ways = []; │ │ │ │ │ + for (var i = 0; i < way_list.length; i++) { │ │ │ │ │ + var way = way_list[i]; │ │ │ │ │ + var way_object = { │ │ │ │ │ + id: way.getAttribute("id") │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_ol_opacity │ │ │ │ │ - */ │ │ │ │ │ - read_ol_opacity: function(layerContext, node) { │ │ │ │ │ - layerContext.opacity = parseFloat(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ + way_object.tags = this.getTags(way); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_ol_singleTile │ │ │ │ │ - */ │ │ │ │ │ - read_ol_singleTile: function(layerContext, node) { │ │ │ │ │ - layerContext.singleTile = (this.getChildValue(node) == "true"); │ │ │ │ │ - }, │ │ │ │ │ + var node_list = way.getElementsByTagName("nd"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_ol_tileSize │ │ │ │ │ - */ │ │ │ │ │ - read_ol_tileSize: function(layerContext, node) { │ │ │ │ │ - var obj = { │ │ │ │ │ - "width": node.getAttribute("width"), │ │ │ │ │ - "height": node.getAttribute("height") │ │ │ │ │ - }; │ │ │ │ │ - layerContext.tileSize = obj; │ │ │ │ │ - }, │ │ │ │ │ + way_object.nodes = new Array(node_list.length); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_ol_isBaseLayer │ │ │ │ │ - */ │ │ │ │ │ - read_ol_isBaseLayer: function(layerContext, node) { │ │ │ │ │ - layerContext.isBaseLayer = (this.getChildValue(node) == "true"); │ │ │ │ │ - }, │ │ │ │ │ + for (var j = 0; j < node_list.length; j++) { │ │ │ │ │ + way_object.nodes[j] = node_list[j].getAttribute("ref"); │ │ │ │ │ + } │ │ │ │ │ + return_ways.push(way_object); │ │ │ │ │ + } │ │ │ │ │ + return return_ways; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_ol_displayInLayerSwitcher │ │ │ │ │ - */ │ │ │ │ │ - read_ol_displayInLayerSwitcher: function(layerContext, node) { │ │ │ │ │ - layerContext.displayInLayerSwitcher = (this.getChildValue(node) == "true"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_wmc_Server │ │ │ │ │ + * Method: getTags │ │ │ │ │ + * Return the tags list attached to a specific DOM element. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * dom_node - {DOMElement} node to parse tags from │ │ │ │ │ + * interesting_tags - {Boolean} whether the return from this function should │ │ │ │ │ + * return a boolean indicating that it has 'interesting tags' -- │ │ │ │ │ + * tags like attribution and source are ignored. (To change the list │ │ │ │ │ + * of tags, see interestingTagsExclude) │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * tags - {Object} hash of tags │ │ │ │ │ + * interesting - {Boolean} if interesting_tags is passed, returns │ │ │ │ │ + * whether there are any interesting tags on this element. │ │ │ │ │ */ │ │ │ │ │ - read_wmc_Server: function(layerContext, node) { │ │ │ │ │ - layerContext.version = node.getAttribute("version"); │ │ │ │ │ - layerContext.url = this.getOnlineResource_href(node); │ │ │ │ │ - layerContext.metadata.servertitle = node.getAttribute("title"); │ │ │ │ │ + getTags: function(dom_node, interesting_tags) { │ │ │ │ │ + var tag_list = dom_node.getElementsByTagName("tag"); │ │ │ │ │ + var tags = {}; │ │ │ │ │ + var interesting = false; │ │ │ │ │ + for (var j = 0; j < tag_list.length; j++) { │ │ │ │ │ + var key = tag_list[j].getAttribute("k"); │ │ │ │ │ + tags[key] = tag_list[j].getAttribute("v"); │ │ │ │ │ + if (interesting_tags) { │ │ │ │ │ + if (!this.interestingTagsExclude[key]) { │ │ │ │ │ + interesting = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return interesting_tags ? [tags, interesting] : tags; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_FormatList │ │ │ │ │ + /** │ │ │ │ │ + * Method: isWayArea │ │ │ │ │ + * Given a way object from getWays, check whether the tags and geometry │ │ │ │ │ + * indicate something is an area. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - read_wmc_FormatList: function(layerContext, node) { │ │ │ │ │ - this.runChildNodes(layerContext, node); │ │ │ │ │ - }, │ │ │ │ │ + isWayArea: function(way) { │ │ │ │ │ + var poly_shaped = false; │ │ │ │ │ + var poly_tags = false; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_Format │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_Format: function(layerContext, node) { │ │ │ │ │ - var format = { │ │ │ │ │ - value: this.getChildValue(node) │ │ │ │ │ - }; │ │ │ │ │ - if (node.getAttribute("current") == "1") { │ │ │ │ │ - format.current = true; │ │ │ │ │ + if (way.nodes[0] == way.nodes[way.nodes.length - 1]) { │ │ │ │ │ + poly_shaped = true; │ │ │ │ │ } │ │ │ │ │ - layerContext.formats.push(format); │ │ │ │ │ + if (this.checkTags) { │ │ │ │ │ + for (var key in way.tags) { │ │ │ │ │ + if (this.areaTags[key]) { │ │ │ │ │ + poly_tags = true; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return poly_shaped && (this.checkTags ? poly_tags : true); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_wmc_StyleList │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Takes a list of features, returns a serialized OSM format file for use │ │ │ │ │ + * in tools like JOSM. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ */ │ │ │ │ │ - read_wmc_StyleList: function(layerContext, node) { │ │ │ │ │ - this.runChildNodes(layerContext, node); │ │ │ │ │ - }, │ │ │ │ │ + write: function(features) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_Style │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_Style: function(layerContext, node) { │ │ │ │ │ - var style = {}; │ │ │ │ │ - this.runChildNodes(style, node); │ │ │ │ │ - if (node.getAttribute("current") == "1") { │ │ │ │ │ - style.current = true; │ │ │ │ │ + this.osm_id = 1; │ │ │ │ │ + this.created_nodes = {}; │ │ │ │ │ + var root_node = this.createElementNS(null, "osm"); │ │ │ │ │ + root_node.setAttribute("version", "0.5"); │ │ │ │ │ + root_node.setAttribute("generator", "OpenLayers " + OpenLayers.VERSION_NUMBER); │ │ │ │ │ + │ │ │ │ │ + // Loop backwards, because the deserializer puts nodes last, and │ │ │ │ │ + // we want them first if possible │ │ │ │ │ + for (var i = features.length - 1; i >= 0; i--) { │ │ │ │ │ + var nodes = this.createFeatureNodes(features[i]); │ │ │ │ │ + for (var j = 0; j < nodes.length; j++) { │ │ │ │ │ + root_node.appendChild(nodes[j]); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - layerContext.styles.push(style); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [root_node]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_wmc_SLD │ │ │ │ │ + * Method: createFeatureNodes │ │ │ │ │ + * Takes a feature, returns a list of nodes from size 0->n. │ │ │ │ │ + * Will include all pieces of the serialization that are required which │ │ │ │ │ + * have not already been created. Calls out to createXML based on geometry │ │ │ │ │ + * type. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - read_wmc_SLD: function(style, node) { │ │ │ │ │ - this.runChildNodes(style, node); │ │ │ │ │ - // style either comes back with an href or a body property │ │ │ │ │ + createFeatureNodes: function(feature) { │ │ │ │ │ + var nodes = []; │ │ │ │ │ + var className = feature.geometry.CLASS_NAME; │ │ │ │ │ + var type = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ + type = type.toLowerCase(); │ │ │ │ │ + var builder = this.createXML[type]; │ │ │ │ │ + if (builder) { │ │ │ │ │ + nodes = builder.apply(this, [feature]); │ │ │ │ │ + } │ │ │ │ │ + return nodes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_sld_StyledLayerDescriptor │ │ │ │ │ + * Method: createXML │ │ │ │ │ + * Takes a feature, returns a list of nodes from size 0->n. │ │ │ │ │ + * Will include all pieces of the serialization that are required which │ │ │ │ │ + * have not already been created. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - read_sld_StyledLayerDescriptor: function(sld, node) { │ │ │ │ │ - var xml = OpenLayers.Format.XML.prototype.write.apply(this, [node]); │ │ │ │ │ - sld.body = xml; │ │ │ │ │ - }, │ │ │ │ │ + createXML: { │ │ │ │ │ + 'point': function(point) { │ │ │ │ │ + var id = null; │ │ │ │ │ + var geometry = point.geometry ? point.geometry : point; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_sld_FeatureTypeStyle │ │ │ │ │ - */ │ │ │ │ │ - read_sld_FeatureTypeStyle: function(sld, node) { │ │ │ │ │ - var xml = OpenLayers.Format.XML.prototype.write.apply(this, [node]); │ │ │ │ │ - sld.body = xml; │ │ │ │ │ - }, │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry = geometry.clone(); │ │ │ │ │ + geometry.transform(this.internalProjection, │ │ │ │ │ + this.externalProjection); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_OnlineResource │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_OnlineResource: function(obj, node) { │ │ │ │ │ - obj.href = this.getAttributeNS( │ │ │ │ │ - node, this.namespaces.xlink, "href" │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ + var already_exists = false; // We don't return anything if the node │ │ │ │ │ + // has already been created │ │ │ │ │ + if (point.osm_id) { │ │ │ │ │ + id = point.osm_id; │ │ │ │ │ + if (this.created_nodes[id]) { │ │ │ │ │ + already_exists = true; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + id = -this.osm_id; │ │ │ │ │ + this.osm_id++; │ │ │ │ │ + } │ │ │ │ │ + if (already_exists) { │ │ │ │ │ + node = this.created_nodes[id]; │ │ │ │ │ + } else { │ │ │ │ │ + var node = this.createElementNS(null, "node"); │ │ │ │ │ + } │ │ │ │ │ + this.created_nodes[id] = node; │ │ │ │ │ + node.setAttribute("id", id); │ │ │ │ │ + node.setAttribute("lon", geometry.x); │ │ │ │ │ + node.setAttribute("lat", geometry.y); │ │ │ │ │ + if (point.attributes) { │ │ │ │ │ + this.serializeTags(point, node); │ │ │ │ │ + } │ │ │ │ │ + this.setState(point, node); │ │ │ │ │ + return already_exists ? [] : [node]; │ │ │ │ │ + }, │ │ │ │ │ + linestring: function(feature) { │ │ │ │ │ + var id; │ │ │ │ │ + var nodes = []; │ │ │ │ │ + var geometry = feature.geometry; │ │ │ │ │ + if (feature.osm_id) { │ │ │ │ │ + id = feature.osm_id; │ │ │ │ │ + } else { │ │ │ │ │ + id = -this.osm_id; │ │ │ │ │ + this.osm_id++; │ │ │ │ │ + } │ │ │ │ │ + var way = this.createElementNS(null, "way"); │ │ │ │ │ + way.setAttribute("id", id); │ │ │ │ │ + for (var i = 0; i < geometry.components.length; i++) { │ │ │ │ │ + var node = this.createXML['point'].apply(this, [geometry.components[i]]); │ │ │ │ │ + if (node.length) { │ │ │ │ │ + node = node[0]; │ │ │ │ │ + var node_ref = node.getAttribute("id"); │ │ │ │ │ + nodes.push(node); │ │ │ │ │ + } else { │ │ │ │ │ + node_ref = geometry.components[i].osm_id; │ │ │ │ │ + node = this.created_nodes[node_ref]; │ │ │ │ │ + } │ │ │ │ │ + this.setState(feature, node); │ │ │ │ │ + var nd_dom = this.createElementNS(null, "nd"); │ │ │ │ │ + nd_dom.setAttribute("ref", node_ref); │ │ │ │ │ + way.appendChild(nd_dom); │ │ │ │ │ + } │ │ │ │ │ + this.serializeTags(feature, way); │ │ │ │ │ + nodes.push(way); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_Name │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_Name: function(obj, node) { │ │ │ │ │ - var name = this.getChildValue(node); │ │ │ │ │ - if (name) { │ │ │ │ │ - obj.name = name; │ │ │ │ │ + return nodes; │ │ │ │ │ + }, │ │ │ │ │ + polygon: function(feature) { │ │ │ │ │ + var attrs = OpenLayers.Util.extend({ │ │ │ │ │ + 'area': 'yes' │ │ │ │ │ + }, feature.attributes); │ │ │ │ │ + var feat = new OpenLayers.Feature.Vector(feature.geometry.components[0], attrs); │ │ │ │ │ + feat.osm_id = feature.osm_id; │ │ │ │ │ + return this.createXML['linestring'].apply(this, [feat]); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_wmc_Title │ │ │ │ │ + * Method: serializeTags │ │ │ │ │ + * Given a feature, serialize the attributes onto the given node. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * node - {DOMNode} │ │ │ │ │ */ │ │ │ │ │ - read_wmc_Title: function(obj, node) { │ │ │ │ │ - var title = this.getChildValue(node); │ │ │ │ │ - if (title) { │ │ │ │ │ - obj.title = title; │ │ │ │ │ + serializeTags: function(feature, node) { │ │ │ │ │ + for (var key in feature.attributes) { │ │ │ │ │ + var tag = this.createElementNS(null, "tag"); │ │ │ │ │ + tag.setAttribute("k", key); │ │ │ │ │ + tag.setAttribute("v", feature.attributes[key]); │ │ │ │ │ + node.appendChild(tag); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_wmc_MetadataURL │ │ │ │ │ + * Method: setState │ │ │ │ │ + * OpenStreetMap has a convention that 'state' is stored for modification or deletion. │ │ │ │ │ + * This allows the file to be uploaded via JOSM or the bulk uploader tool. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * node - {DOMNode} │ │ │ │ │ */ │ │ │ │ │ - read_wmc_MetadataURL: function(layerContext, node) { │ │ │ │ │ - layerContext.metadataURL = this.getOnlineResource_href(node); │ │ │ │ │ + setState: function(feature, node) { │ │ │ │ │ + if (feature.state) { │ │ │ │ │ + var state = null; │ │ │ │ │ + switch (feature.state) { │ │ │ │ │ + case OpenLayers.State.UPDATE: │ │ │ │ │ + state = "modify"; │ │ │ │ │ + case OpenLayers.State.DELETE: │ │ │ │ │ + state = "delete"; │ │ │ │ │ + } │ │ │ │ │ + if (state) { │ │ │ │ │ + node.setAttribute("action", state); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_KeywordList │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_KeywordList: function(context, node) { │ │ │ │ │ - context.keywords = []; │ │ │ │ │ - this.runChildNodes(context.keywords, node); │ │ │ │ │ - }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.OSM" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WFS.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_Keyword │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_Keyword: function(keywords, node) { │ │ │ │ │ - keywords.push(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_Abstract │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_Abstract: function(obj, node) { │ │ │ │ │ - var abst = this.getChildValue(node); │ │ │ │ │ - if (abst) { │ │ │ │ │ - obj["abstract"] = abst; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/GML.js │ │ │ │ │ + * @requires OpenLayers/Console.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_LogoURL │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_LogoURL: function(context, node) { │ │ │ │ │ - context.logo = { │ │ │ │ │ - width: node.getAttribute("width"), │ │ │ │ │ - height: node.getAttribute("height"), │ │ │ │ │ - format: node.getAttribute("format"), │ │ │ │ │ - href: this.getOnlineResource_href(node) │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WFS │ │ │ │ │ + * Read/Write WFS. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.GML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WFS = OpenLayers.Class(OpenLayers.Format.GML, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_DescriptionURL │ │ │ │ │ + /** │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {<OpenLayers.Layer>} │ │ │ │ │ */ │ │ │ │ │ - read_wmc_DescriptionURL: function(context, node) { │ │ │ │ │ - context.descriptionURL = this.getOnlineResource_href(node); │ │ │ │ │ - }, │ │ │ │ │ + layer: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_wmc_ContactInformation │ │ │ │ │ + * APIProperty: wfsns │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - read_wmc_ContactInformation: function(obj, node) { │ │ │ │ │ - var contact = {}; │ │ │ │ │ - this.runChildNodes(contact, node); │ │ │ │ │ - obj.contactInformation = contact; │ │ │ │ │ - }, │ │ │ │ │ + wfsns: "http://www.opengis.net/wfs", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_wmc_ContactPersonPrimary │ │ │ │ │ + * Property: ogcns │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ - read_wmc_ContactPersonPrimary: function(contact, node) { │ │ │ │ │ - var personPrimary = {}; │ │ │ │ │ - this.runChildNodes(personPrimary, node); │ │ │ │ │ - contact.personPrimary = personPrimary; │ │ │ │ │ - }, │ │ │ │ │ + ogcns: "http://www.opengis.net/ogc", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_wmc_ContactPerson │ │ │ │ │ + * Constructor: OpenLayers.Format.WFS │ │ │ │ │ + * Create a WFS-T formatter. This requires a layer: that layer should │ │ │ │ │ + * have two properties: geometry_column and typename. The parser │ │ │ │ │ + * for this format is subclassed entirely from GML: There is a writer │ │ │ │ │ + * only, which uses most of the code from the GML layer, and wraps │ │ │ │ │ + * it in transactional elements. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} │ │ │ │ │ + * layer - {<OpenLayers.Layer>} │ │ │ │ │ */ │ │ │ │ │ - read_wmc_ContactPerson: function(primaryPerson, node) { │ │ │ │ │ - var person = this.getChildValue(node); │ │ │ │ │ - if (person) { │ │ │ │ │ - primaryPerson.person = person; │ │ │ │ │ + initialize: function(options, layer) { │ │ │ │ │ + OpenLayers.Format.GML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + if (this.layer.featureNS) { │ │ │ │ │ + this.featureNS = this.layer.featureNS; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_ContactOrganization │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_ContactOrganization: function(primaryPerson, node) { │ │ │ │ │ - var organization = this.getChildValue(node); │ │ │ │ │ - if (organization) { │ │ │ │ │ - primaryPerson.organization = organization; │ │ │ │ │ + if (this.layer.options.geometry_column) { │ │ │ │ │ + this.geometryName = this.layer.options.geometry_column; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_ContactPosition │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_ContactPosition: function(contact, node) { │ │ │ │ │ - var position = this.getChildValue(node); │ │ │ │ │ - if (position) { │ │ │ │ │ - contact.position = position; │ │ │ │ │ + if (this.layer.options.typename) { │ │ │ │ │ + this.featureName = this.layer.options.typename; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_wmc_ContactAddress │ │ │ │ │ + * Method: write │ │ │ │ │ + * Takes a feature list, and generates a WFS-T Transaction │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ */ │ │ │ │ │ - read_wmc_ContactAddress: function(contact, node) { │ │ │ │ │ - var contactAddress = {}; │ │ │ │ │ - this.runChildNodes(contactAddress, node); │ │ │ │ │ - contact.contactAddress = contactAddress; │ │ │ │ │ - }, │ │ │ │ │ + write: function(features) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_AddressType │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_AddressType: function(contactAddress, node) { │ │ │ │ │ - var type = this.getChildValue(node); │ │ │ │ │ - if (type) { │ │ │ │ │ - contactAddress.type = type; │ │ │ │ │ + var transaction = this.createElementNS(this.wfsns, 'wfs:Transaction'); │ │ │ │ │ + transaction.setAttribute("version", "1.0.0"); │ │ │ │ │ + transaction.setAttribute("service", "WFS"); │ │ │ │ │ + for (var i = 0; i < features.length; i++) { │ │ │ │ │ + switch (features[i].state) { │ │ │ │ │ + case OpenLayers.State.INSERT: │ │ │ │ │ + transaction.appendChild(this.insert(features[i])); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.State.UPDATE: │ │ │ │ │ + transaction.appendChild(this.update(features[i])); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.State.DELETE: │ │ │ │ │ + transaction.appendChild(this.remove(features[i])); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_Address │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_Address: function(contactAddress, node) { │ │ │ │ │ - var address = this.getChildValue(node); │ │ │ │ │ - if (address) { │ │ │ │ │ - contactAddress.address = address; │ │ │ │ │ - } │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [transaction]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_wmc_City │ │ │ │ │ + * Method: createFeatureXML │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - read_wmc_City: function(contactAddress, node) { │ │ │ │ │ - var city = this.getChildValue(node); │ │ │ │ │ - if (city) { │ │ │ │ │ - contactAddress.city = city; │ │ │ │ │ + createFeatureXML: function(feature) { │ │ │ │ │ + var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ + var geomContainer = this.createElementNS(this.featureNS, "feature:" + this.geometryName); │ │ │ │ │ + geomContainer.appendChild(geometryNode); │ │ │ │ │ + var featureContainer = this.createElementNS(this.featureNS, "feature:" + this.featureName); │ │ │ │ │ + featureContainer.appendChild(geomContainer); │ │ │ │ │ + for (var attr in feature.attributes) { │ │ │ │ │ + var attrText = this.createTextNode(feature.attributes[attr]); │ │ │ │ │ + var nodename = attr; │ │ │ │ │ + if (attr.search(":") != -1) { │ │ │ │ │ + nodename = attr.split(":")[1]; │ │ │ │ │ + } │ │ │ │ │ + var attrContainer = this.createElementNS(this.featureNS, "feature:" + nodename); │ │ │ │ │ + attrContainer.appendChild(attrText); │ │ │ │ │ + featureContainer.appendChild(attrContainer); │ │ │ │ │ } │ │ │ │ │ + return featureContainer; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_wmc_StateOrProvince │ │ │ │ │ + * Method: insert │ │ │ │ │ + * Takes a feature, and generates a WFS-T Transaction "Insert" │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - read_wmc_StateOrProvince: function(contactAddress, node) { │ │ │ │ │ - var stateOrProvince = this.getChildValue(node); │ │ │ │ │ - if (stateOrProvince) { │ │ │ │ │ - contactAddress.stateOrProvince = stateOrProvince; │ │ │ │ │ - } │ │ │ │ │ + insert: function(feature) { │ │ │ │ │ + var insertNode = this.createElementNS(this.wfsns, 'wfs:Insert'); │ │ │ │ │ + insertNode.appendChild(this.createFeatureXML(feature)); │ │ │ │ │ + return insertNode; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_wmc_PostCode │ │ │ │ │ + * Method: update │ │ │ │ │ + * Takes a feature, and generates a WFS-T Transaction "Update" │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - read_wmc_PostCode: function(contactAddress, node) { │ │ │ │ │ - var postcode = this.getChildValue(node); │ │ │ │ │ - if (postcode) { │ │ │ │ │ - contactAddress.postcode = postcode; │ │ │ │ │ + update: function(feature) { │ │ │ │ │ + if (!feature.fid) { │ │ │ │ │ + OpenLayers.Console.userError(OpenLayers.i18n("noFID")); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ + var updateNode = this.createElementNS(this.wfsns, 'wfs:Update'); │ │ │ │ │ + updateNode.setAttribute("typeName", this.featurePrefix + ':' + this.featureName); │ │ │ │ │ + updateNode.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_Country │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_Country: function(contactAddress, node) { │ │ │ │ │ - var country = this.getChildValue(node); │ │ │ │ │ - if (country) { │ │ │ │ │ - contactAddress.country = country; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + var propertyNode = this.createElementNS(this.wfsns, 'wfs:Property'); │ │ │ │ │ + var nameNode = this.createElementNS(this.wfsns, 'wfs:Name'); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_ContactVoiceTelephone │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_ContactVoiceTelephone: function(contact, node) { │ │ │ │ │ - var phone = this.getChildValue(node); │ │ │ │ │ - if (phone) { │ │ │ │ │ - contact.phone = phone; │ │ │ │ │ + var txtNode = this.createTextNode(this.geometryName); │ │ │ │ │ + nameNode.appendChild(txtNode); │ │ │ │ │ + propertyNode.appendChild(nameNode); │ │ │ │ │ + │ │ │ │ │ + var valueNode = this.createElementNS(this.wfsns, 'wfs:Value'); │ │ │ │ │ + │ │ │ │ │ + var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ + │ │ │ │ │ + if (feature.layer) { │ │ │ │ │ + geometryNode.setAttribute( │ │ │ │ │ + "srsName", feature.layer.projection.getCode() │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_ContactFacsimileTelephone │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_ContactFacsimileTelephone: function(contact, node) { │ │ │ │ │ - var fax = this.getChildValue(node); │ │ │ │ │ - if (fax) { │ │ │ │ │ - contact.fax = fax; │ │ │ │ │ + valueNode.appendChild(geometryNode); │ │ │ │ │ + │ │ │ │ │ + propertyNode.appendChild(valueNode); │ │ │ │ │ + updateNode.appendChild(propertyNode); │ │ │ │ │ + │ │ │ │ │ + // add in attributes │ │ │ │ │ + for (var propName in feature.attributes) { │ │ │ │ │ + propertyNode = this.createElementNS(this.wfsns, 'wfs:Property'); │ │ │ │ │ + nameNode = this.createElementNS(this.wfsns, 'wfs:Name'); │ │ │ │ │ + nameNode.appendChild(this.createTextNode(propName)); │ │ │ │ │ + propertyNode.appendChild(nameNode); │ │ │ │ │ + valueNode = this.createElementNS(this.wfsns, 'wfs:Value'); │ │ │ │ │ + valueNode.appendChild(this.createTextNode(feature.attributes[propName])); │ │ │ │ │ + propertyNode.appendChild(valueNode); │ │ │ │ │ + updateNode.appendChild(propertyNode); │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + var filterNode = this.createElementNS(this.ogcns, 'ogc:Filter'); │ │ │ │ │ + var filterIdNode = this.createElementNS(this.ogcns, 'ogc:FeatureId'); │ │ │ │ │ + filterIdNode.setAttribute("fid", feature.fid); │ │ │ │ │ + filterNode.appendChild(filterIdNode); │ │ │ │ │ + updateNode.appendChild(filterNode); │ │ │ │ │ + │ │ │ │ │ + return updateNode; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_wmc_ContactElectronicMailAddress │ │ │ │ │ + * Method: remove │ │ │ │ │ + * Takes a feature, and generates a WFS-T Transaction "Delete" │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - read_wmc_ContactElectronicMailAddress: function(contact, node) { │ │ │ │ │ - var email = this.getChildValue(node); │ │ │ │ │ - if (email) { │ │ │ │ │ - contact.email = email; │ │ │ │ │ + remove: function(feature) { │ │ │ │ │ + if (!feature.fid) { │ │ │ │ │ + OpenLayers.Console.userError(OpenLayers.i18n("noFID")); │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ + var deleteNode = this.createElementNS(this.wfsns, 'wfs:Delete'); │ │ │ │ │ + deleteNode.setAttribute("typeName", this.featurePrefix + ':' + this.featureName); │ │ │ │ │ + deleteNode.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ + │ │ │ │ │ + var filterNode = this.createElementNS(this.ogcns, 'ogc:Filter'); │ │ │ │ │ + var filterIdNode = this.createElementNS(this.ogcns, 'ogc:FeatureId'); │ │ │ │ │ + filterIdNode.setAttribute("fid", feature.fid); │ │ │ │ │ + filterNode.appendChild(filterIdNode); │ │ │ │ │ + deleteNode.appendChild(filterNode); │ │ │ │ │ + │ │ │ │ │ + return deleteNode; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_wmc_DataURL │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Remove ciruclar ref to layer │ │ │ │ │ */ │ │ │ │ │ - read_wmc_DataURL: function(layerContext, node) { │ │ │ │ │ - layerContext.dataURL = this.getOnlineResource_href(node); │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.layer = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFS" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/EncodedPolyline.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.EncodedPolyline │ │ │ │ │ + * Class for reading and writing encoded polylines. Create a new instance │ │ │ │ │ + * with the <OpenLayers.Format.EncodedPolyline> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.EncodedPolyline = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: read_wmc_LegendURL │ │ │ │ │ + * APIProperty: geometryType │ │ │ │ │ + * {String} Geometry type to output. One of: linestring (default), │ │ │ │ │ + * linearring, point, multipoint or polygon. If the geometryType is │ │ │ │ │ + * point, only the first point of the string is returned. │ │ │ │ │ */ │ │ │ │ │ - read_wmc_LegendURL: function(style, node) { │ │ │ │ │ - var legend = { │ │ │ │ │ - width: node.getAttribute('width'), │ │ │ │ │ - height: node.getAttribute('height'), │ │ │ │ │ - format: node.getAttribute('format'), │ │ │ │ │ - href: this.getOnlineResource_href(node) │ │ │ │ │ - }; │ │ │ │ │ - style.legend = legend; │ │ │ │ │ - }, │ │ │ │ │ + geometryType: "linestring", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: read_wmc_DimensionList │ │ │ │ │ + * Constructor: OpenLayers.Format.EncodedPolyline │ │ │ │ │ + * Create a new parser for encoded polylines │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Format.EncodedPolyline>} A new encoded polylines parser. │ │ │ │ │ */ │ │ │ │ │ - read_wmc_DimensionList: function(layerContext, node) { │ │ │ │ │ - layerContext.dimensions = {}; │ │ │ │ │ - this.runChildNodes(layerContext.dimensions, node); │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.prototype.initialize.apply(this, [options]); │ │ │ │ │ }, │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: read_wmc_Dimension │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Deserialize an encoded polyline string and return a vector feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * encoded - {String} An encoded polyline string │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A vector feature with a linestring. │ │ │ │ │ */ │ │ │ │ │ - read_wmc_Dimension: function(dimensions, node) { │ │ │ │ │ - var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ + read: function(encoded) { │ │ │ │ │ + var geomType; │ │ │ │ │ + if (this.geometryType == "linestring") │ │ │ │ │ + geomType = OpenLayers.Geometry.LineString; │ │ │ │ │ + else if (this.geometryType == "linearring") │ │ │ │ │ + geomType = OpenLayers.Geometry.LinearRing; │ │ │ │ │ + else if (this.geometryType == "multipoint") │ │ │ │ │ + geomType = OpenLayers.Geometry.MultiPoint; │ │ │ │ │ + else if (this.geometryType != "point" && this.geometryType != "polygon") │ │ │ │ │ + return null; │ │ │ │ │ │ │ │ │ │ - var dim = { │ │ │ │ │ - name: name, │ │ │ │ │ - units: node.getAttribute("units") || "", │ │ │ │ │ - unitSymbol: node.getAttribute("unitSymbol") || "", │ │ │ │ │ - userValue: node.getAttribute("userValue") || "", │ │ │ │ │ - nearestValue: node.getAttribute("nearestValue") === "1", │ │ │ │ │ - multipleValues: node.getAttribute("multipleValues") === "1", │ │ │ │ │ - current: node.getAttribute("current") === "1", │ │ │ │ │ - "default": node.getAttribute("default") || "" │ │ │ │ │ - }; │ │ │ │ │ - var values = this.getChildValue(node); │ │ │ │ │ - dim.values = values.split(","); │ │ │ │ │ + var flatPoints = this.decodeDeltas(encoded, 2); │ │ │ │ │ + var flatPointsLength = flatPoints.length; │ │ │ │ │ │ │ │ │ │ - dimensions[dim.name] = dim; │ │ │ │ │ + var pointGeometries = []; │ │ │ │ │ + for (var i = 0; i + 1 < flatPointsLength;) { │ │ │ │ │ + var y = flatPoints[i++], │ │ │ │ │ + x = flatPoints[i++]; │ │ │ │ │ + pointGeometries.push(new OpenLayers.Geometry.Point(x, y)); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + if (this.geometryType == "point") │ │ │ │ │ + return new OpenLayers.Feature.Vector( │ │ │ │ │ + pointGeometries[0] │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + if (this.geometryType == "polygon") │ │ │ │ │ + return new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.Polygon([ │ │ │ │ │ + new OpenLayers.Geometry.LinearRing(pointGeometries) │ │ │ │ │ + ]) │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + return new OpenLayers.Feature.Vector( │ │ │ │ │ + new geomType(pointGeometries) │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: write │ │ │ │ │ + * APIMethod: decode │ │ │ │ │ + * Deserialize an encoded string and return an array of n-dimensional │ │ │ │ │ + * points. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * context - {Object} An object representing the map context. │ │ │ │ │ - * options - {Object} Optional object. │ │ │ │ │ + * encoded - {String} An encoded string │ │ │ │ │ + * dims - {int} The dimension of the points that are returned │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A WMC document string. │ │ │ │ │ + * {Array(Array(int))} An array containing n-dimensional arrays of │ │ │ │ │ + * coordinates. │ │ │ │ │ */ │ │ │ │ │ - write: function(context, options) { │ │ │ │ │ - var root = this.createElementDefaultNS("ViewContext"); │ │ │ │ │ - this.setAttributes(root, { │ │ │ │ │ - version: this.VERSION, │ │ │ │ │ - id: (options && typeof options.id == "string") ? │ │ │ │ │ - options.id : OpenLayers.Util.createUniqueID("OpenLayers_Context_") │ │ │ │ │ - }); │ │ │ │ │ + decode: function(encoded, dims, opt_factor) { │ │ │ │ │ + var factor = opt_factor || 1e5; │ │ │ │ │ + var flatPoints = this.decodeDeltas(encoded, dims, factor); │ │ │ │ │ + var flatPointsLength = flatPoints.length; │ │ │ │ │ │ │ │ │ │ - // add schemaLocation attribute │ │ │ │ │ - this.setAttributeNS( │ │ │ │ │ - root, this.namespaces.xsi, │ │ │ │ │ - "xsi:schemaLocation", this.schemaLocation │ │ │ │ │ - ); │ │ │ │ │ + var points = []; │ │ │ │ │ + for (var i = 0; i + (dims - 1) < flatPointsLength;) { │ │ │ │ │ + var point = []; │ │ │ │ │ │ │ │ │ │ - // required General element │ │ │ │ │ - root.appendChild(this.write_wmc_General(context)); │ │ │ │ │ + for (var dim = 0; dim < dims; ++dim) { │ │ │ │ │ + point.push(flatPoints[i++]) │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // required LayerList element │ │ │ │ │ - root.appendChild(this.write_wmc_LayerList(context)); │ │ │ │ │ + points.push(point); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [root]); │ │ │ │ │ + return points; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createElementDefaultNS │ │ │ │ │ - * Shorthand for createElementNS with namespace from <defaultPrefix>. │ │ │ │ │ - * Can optionally be used to set attributes and a text child value. │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Serialize a feature or array of features into a WKT string. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} The qualified node name. │ │ │ │ │ - * childValue - {String} Optional value for text child node. │ │ │ │ │ - * attributes - {Object} Optional object representing attributes. │ │ │ │ │ + * features - {<OpenLayers.Feature.Vector>|Array} A feature or array of │ │ │ │ │ + * features │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Element} An element node. │ │ │ │ │ + * {String} The WKT string representation of the input geometries │ │ │ │ │ */ │ │ │ │ │ - createElementDefaultNS: function(name, childValue, attributes) { │ │ │ │ │ - var node = this.createElementNS( │ │ │ │ │ - this.namespaces[this.defaultPrefix], │ │ │ │ │ - name │ │ │ │ │ - ); │ │ │ │ │ - if (childValue) { │ │ │ │ │ - node.appendChild(this.createTextNode(childValue)); │ │ │ │ │ - } │ │ │ │ │ - if (attributes) { │ │ │ │ │ - this.setAttributes(node, attributes); │ │ │ │ │ + write: function(features) { │ │ │ │ │ + var feature; │ │ │ │ │ + if (features.constructor == Array) │ │ │ │ │ + feature = features[0]; │ │ │ │ │ + else │ │ │ │ │ + feature = features; │ │ │ │ │ + │ │ │ │ │ + var geometry = feature.geometry; │ │ │ │ │ + var type = geometry.CLASS_NAME.split('.')[2].toLowerCase(); │ │ │ │ │ + │ │ │ │ │ + var pointGeometries; │ │ │ │ │ + if (type == "point") │ │ │ │ │ + pointGeometries = new Array(geometry); │ │ │ │ │ + else if (type == "linestring" || │ │ │ │ │ + type == "linearring" || │ │ │ │ │ + type == "multipoint") │ │ │ │ │ + pointGeometries = geometry.components; │ │ │ │ │ + else if (type == "polygon") │ │ │ │ │ + pointGeometries = geometry.components[0].components; │ │ │ │ │ + else │ │ │ │ │ + return null; │ │ │ │ │ + │ │ │ │ │ + var flatPoints = []; │ │ │ │ │ + │ │ │ │ │ + var pointGeometriesLength = pointGeometries.length; │ │ │ │ │ + for (var i = 0; i < pointGeometriesLength; ++i) { │ │ │ │ │ + var pointGeometry = pointGeometries[i]; │ │ │ │ │ + flatPoints.push(pointGeometry.y); │ │ │ │ │ + flatPoints.push(pointGeometry.x); │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ + │ │ │ │ │ + return this.encodeDeltas(flatPoints, 2); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setAttributes │ │ │ │ │ - * Set multiple attributes given key value pairs from an object. │ │ │ │ │ + * APIMethod: encode │ │ │ │ │ + * Serialize an array of n-dimensional points and return an encoded string │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {Element} An element node. │ │ │ │ │ - * obj - {Object} An object whose properties represent attribute names and │ │ │ │ │ - * values represent attribute values. │ │ │ │ │ + * points - {Array(Array(int))} An array containing n-dimensional │ │ │ │ │ + * arrays of coordinates │ │ │ │ │ + * dims - {int} The dimension of the points that should be read │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} An encoded string │ │ │ │ │ */ │ │ │ │ │ - setAttributes: function(node, obj) { │ │ │ │ │ - var value; │ │ │ │ │ - for (var name in obj) { │ │ │ │ │ - value = obj[name].toString(); │ │ │ │ │ - if (value.match(/[A-Z]/)) { │ │ │ │ │ - // safari lowercases attributes with setAttribute │ │ │ │ │ - this.setAttributeNS(node, null, name, value); │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttribute(name, value); │ │ │ │ │ + encode: function(points, dims, opt_factor) { │ │ │ │ │ + var factor = opt_factor || 1e5; │ │ │ │ │ + var flatPoints = []; │ │ │ │ │ + │ │ │ │ │ + var pointsLength = points.length; │ │ │ │ │ + for (var i = 0; i < pointsLength; ++i) { │ │ │ │ │ + var point = points[i]; │ │ │ │ │ + │ │ │ │ │ + for (var dim = 0; dim < dims; ++dim) { │ │ │ │ │ + flatPoints.push(point[dim]); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + return this.encodeDeltas(flatPoints, dims, factor); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: write_wmc_General │ │ │ │ │ - * Create a General node given an context object. │ │ │ │ │ + * APIMethod: encodeDeltas │ │ │ │ │ + * Encode a list of n-dimensional points and return an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Attention: This function will modify the passed array! │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * context - {Object} Context object. │ │ │ │ │ + * numbers - {Array.<number>} A list of n-dimensional points. │ │ │ │ │ + * dimension - {number} The dimension of the points in the list. │ │ │ │ │ + * opt_factor - {number=} The factor by which the numbers will be │ │ │ │ │ + * multiplied. The remaining decimal places will get rounded away. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Element} A WMC General element node. │ │ │ │ │ + * {string} The encoded string. │ │ │ │ │ */ │ │ │ │ │ - write_wmc_General: function(context) { │ │ │ │ │ - var node = this.createElementDefaultNS("General"); │ │ │ │ │ + encodeDeltas: function(numbers, dimension, opt_factor) { │ │ │ │ │ + var factor = opt_factor || 1e5; │ │ │ │ │ + var d; │ │ │ │ │ │ │ │ │ │ - // optional Window element │ │ │ │ │ - if (context.size) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "Window", null, { │ │ │ │ │ - width: context.size.w, │ │ │ │ │ - height: context.size.h │ │ │ │ │ - } │ │ │ │ │ - )); │ │ │ │ │ + var lastNumbers = new Array(dimension); │ │ │ │ │ + for (d = 0; d < dimension; ++d) { │ │ │ │ │ + lastNumbers[d] = 0; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // required BoundingBox element │ │ │ │ │ - var bounds = context.bounds; │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "BoundingBox", null, { │ │ │ │ │ - minx: bounds.left.toPrecision(18), │ │ │ │ │ - miny: bounds.bottom.toPrecision(18), │ │ │ │ │ - maxx: bounds.right.toPrecision(18), │ │ │ │ │ - maxy: bounds.top.toPrecision(18), │ │ │ │ │ - SRS: context.projection │ │ │ │ │ + var numbersLength = numbers.length; │ │ │ │ │ + for (var i = 0; i < numbersLength;) { │ │ │ │ │ + for (d = 0; d < dimension; ++d, ++i) { │ │ │ │ │ + var num = numbers[i]; │ │ │ │ │ + var delta = num - lastNumbers[d]; │ │ │ │ │ + lastNumbers[d] = num; │ │ │ │ │ + │ │ │ │ │ + numbers[i] = delta; │ │ │ │ │ } │ │ │ │ │ - )); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // required Title element │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "Title", context.title │ │ │ │ │ - )); │ │ │ │ │ + return this.encodeFloats(numbers, factor); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // optional KeywordList element │ │ │ │ │ - if (context.keywords) { │ │ │ │ │ - node.appendChild(this.write_wmc_KeywordList(context.keywords)); │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - // optional Abstract element │ │ │ │ │ - if (context["abstract"]) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "Abstract", context["abstract"] │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: decodeDeltas │ │ │ │ │ + * Decode a list of n-dimensional points from an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * encoded - {string} An encoded string. │ │ │ │ │ + * dimension - {number} The dimension of the points in the encoded string. │ │ │ │ │ + * opt_factor - {number=} The factor by which the resulting numbers will │ │ │ │ │ + * be divided. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array.<number>} A list of n-dimensional points. │ │ │ │ │ + */ │ │ │ │ │ + decodeDeltas: function(encoded, dimension, opt_factor) { │ │ │ │ │ + var factor = opt_factor || 1e5; │ │ │ │ │ + var d; │ │ │ │ │ │ │ │ │ │ - // Optional LogoURL element │ │ │ │ │ - if (context.logo) { │ │ │ │ │ - node.appendChild(this.write_wmc_URLType("LogoURL", context.logo.href, context.logo)); │ │ │ │ │ + var lastNumbers = new Array(dimension); │ │ │ │ │ + for (d = 0; d < dimension; ++d) { │ │ │ │ │ + lastNumbers[d] = 0; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - // Optional DescriptionURL element │ │ │ │ │ - if (context.descriptionURL) { │ │ │ │ │ - node.appendChild(this.write_wmc_URLType("DescriptionURL", context.descriptionURL)); │ │ │ │ │ - } │ │ │ │ │ + var numbers = this.decodeFloats(encoded, factor); │ │ │ │ │ │ │ │ │ │ - // Optional ContactInformation element │ │ │ │ │ - if (context.contactInformation) { │ │ │ │ │ - node.appendChild(this.write_wmc_ContactInformation(context.contactInformation)); │ │ │ │ │ - } │ │ │ │ │ + var numbersLength = numbers.length; │ │ │ │ │ + for (var i = 0; i < numbersLength;) { │ │ │ │ │ + for (d = 0; d < dimension; ++d, ++i) { │ │ │ │ │ + lastNumbers[d] += numbers[i]; │ │ │ │ │ │ │ │ │ │ - // OpenLayers specific map properties │ │ │ │ │ - node.appendChild(this.write_ol_MapExtension(context)); │ │ │ │ │ + numbers[i] = lastNumbers[d]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - return node; │ │ │ │ │ + return numbers; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: write_wmc_KeywordList │ │ │ │ │ - */ │ │ │ │ │ - write_wmc_KeywordList: function(keywords) { │ │ │ │ │ - var node = this.createElementDefaultNS("KeywordList"); │ │ │ │ │ │ │ │ │ │ - for (var i = 0, len = keywords.length; i < len; i++) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "Keyword", keywords[i] │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ /** │ │ │ │ │ - * Method: write_wmc_ContactInformation │ │ │ │ │ + * APIMethod: encodeFloats │ │ │ │ │ + * Encode a list of floating point numbers and return an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Attention: This function will modify the passed array! │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * numbers - {Array.<number>} A list of floating point numbers. │ │ │ │ │ + * opt_factor - {number=} The factor by which the numbers will be │ │ │ │ │ + * multiplied. The remaining decimal places will get rounded away. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {string} The encoded string. │ │ │ │ │ */ │ │ │ │ │ - write_wmc_ContactInformation: function(contact) { │ │ │ │ │ - var node = this.createElementDefaultNS("ContactInformation"); │ │ │ │ │ + encodeFloats: function(numbers, opt_factor) { │ │ │ │ │ + var factor = opt_factor || 1e5; │ │ │ │ │ │ │ │ │ │ - if (contact.personPrimary) { │ │ │ │ │ - node.appendChild(this.write_wmc_ContactPersonPrimary(contact.personPrimary)); │ │ │ │ │ - } │ │ │ │ │ - if (contact.position) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "ContactPosition", contact.position │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - if (contact.contactAddress) { │ │ │ │ │ - node.appendChild(this.write_wmc_ContactAddress(contact.contactAddress)); │ │ │ │ │ - } │ │ │ │ │ - if (contact.phone) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "ContactVoiceTelephone", contact.phone │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - if (contact.fax) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "ContactFacsimileTelephone", contact.fax │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - if (contact.email) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "ContactElectronicMailAddress", contact.email │ │ │ │ │ - )); │ │ │ │ │ + var numbersLength = numbers.length; │ │ │ │ │ + for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ + numbers[i] = Math.round(numbers[i] * factor); │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: write_wmc_ContactPersonPrimary │ │ │ │ │ - */ │ │ │ │ │ - write_wmc_ContactPersonPrimary: function(personPrimary) { │ │ │ │ │ - var node = this.createElementDefaultNS("ContactPersonPrimary"); │ │ │ │ │ - if (personPrimary.person) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "ContactPerson", personPrimary.person │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - if (personPrimary.organization) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "ContactOrganization", personPrimary.organization │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ + return this.encodeSignedIntegers(numbers); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: write_wmc_ContactAddress │ │ │ │ │ - */ │ │ │ │ │ - write_wmc_ContactAddress: function(contactAddress) { │ │ │ │ │ - var node = this.createElementDefaultNS("ContactAddress"); │ │ │ │ │ - if (contactAddress.type) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "AddressType", contactAddress.type │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - if (contactAddress.address) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "Address", contactAddress.address │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - if (contactAddress.city) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "City", contactAddress.city │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - if (contactAddress.stateOrProvince) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "StateOrProvince", contactAddress.stateOrProvince │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - if (contactAddress.postcode) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "PostCode", contactAddress.postcode │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - if (contactAddress.country) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "Country", contactAddress.country │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: write_ol_MapExtension │ │ │ │ │ + * APIMethod: decodeFloats │ │ │ │ │ + * Decode a list of floating point numbers from an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * encoded - {string} An encoded string. │ │ │ │ │ + * opt_factor - {number=} The factor by which the result will be divided. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array.<number>} A list of floating point numbers. │ │ │ │ │ */ │ │ │ │ │ - write_ol_MapExtension: function(context) { │ │ │ │ │ - var node = this.createElementDefaultNS("Extension"); │ │ │ │ │ + decodeFloats: function(encoded, opt_factor) { │ │ │ │ │ + var factor = opt_factor || 1e5; │ │ │ │ │ │ │ │ │ │ - var bounds = context.maxExtent; │ │ │ │ │ - if (bounds) { │ │ │ │ │ - var maxExtent = this.createElementNS( │ │ │ │ │ - this.namespaces.ol, "ol:maxExtent" │ │ │ │ │ - ); │ │ │ │ │ - this.setAttributes(maxExtent, { │ │ │ │ │ - minx: bounds.left.toPrecision(18), │ │ │ │ │ - miny: bounds.bottom.toPrecision(18), │ │ │ │ │ - maxx: bounds.right.toPrecision(18), │ │ │ │ │ - maxy: bounds.top.toPrecision(18) │ │ │ │ │ - }); │ │ │ │ │ - node.appendChild(maxExtent); │ │ │ │ │ + var numbers = this.decodeSignedIntegers(encoded); │ │ │ │ │ + │ │ │ │ │ + var numbersLength = numbers.length; │ │ │ │ │ + for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ + numbers[i] /= factor; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return node; │ │ │ │ │ + return numbers; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: write_wmc_LayerList │ │ │ │ │ - * Create a LayerList node given an context object. │ │ │ │ │ + * APIMethod: encodeSignedIntegers │ │ │ │ │ + * Encode a list of signed integers and return an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Attention: This function will modify the passed array! │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * context - {Object} Context object. │ │ │ │ │ + * numbers - {Array.<number>} A list of signed integers. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Element} A WMC LayerList element node. │ │ │ │ │ + * {string} The encoded string. │ │ │ │ │ */ │ │ │ │ │ - write_wmc_LayerList: function(context) { │ │ │ │ │ - var list = this.createElementDefaultNS("LayerList"); │ │ │ │ │ + encodeSignedIntegers: function(numbers) { │ │ │ │ │ + var numbersLength = numbers.length; │ │ │ │ │ + for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ + var num = numbers[i]; │ │ │ │ │ │ │ │ │ │ - for (var i = 0, len = context.layersContext.length; i < len; ++i) { │ │ │ │ │ - list.appendChild(this.write_wmc_Layer(context.layersContext[i])); │ │ │ │ │ + var signedNum = num << 1; │ │ │ │ │ + if (num < 0) { │ │ │ │ │ + signedNum = ~(signedNum); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + numbers[i] = signedNum; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return list; │ │ │ │ │ + return this.encodeUnsignedIntegers(numbers); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: write_wmc_Layer │ │ │ │ │ - * Create a Layer node given a layer context object. │ │ │ │ │ + * APIMethod: decodeSignedIntegers │ │ │ │ │ + * Decode a list of signed integers from an encoded string │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * context - {Object} A layer context object.} │ │ │ │ │ + * encoded - {string} An encoded string. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Element} A WMC Layer element node. │ │ │ │ │ + * {Array.<number>} A list of signed integers. │ │ │ │ │ */ │ │ │ │ │ - write_wmc_Layer: function(context) { │ │ │ │ │ - var node = this.createElementDefaultNS( │ │ │ │ │ - "Layer", null, { │ │ │ │ │ - queryable: context.queryable ? "1" : "0", │ │ │ │ │ - hidden: context.visibility ? "0" : "1" │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - // required Server element │ │ │ │ │ - node.appendChild(this.write_wmc_Server(context)); │ │ │ │ │ + decodeSignedIntegers: function(encoded) { │ │ │ │ │ + var numbers = this.decodeUnsignedIntegers(encoded); │ │ │ │ │ │ │ │ │ │ - // required Name element │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "Name", context.name │ │ │ │ │ - )); │ │ │ │ │ + var numbersLength = numbers.length; │ │ │ │ │ + for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ + var num = numbers[i]; │ │ │ │ │ + numbers[i] = (num & 1) ? ~(num >> 1) : (num >> 1); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // required Title element │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "Title", context.title │ │ │ │ │ - )); │ │ │ │ │ + return numbers; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // optional Abstract element │ │ │ │ │ - if (context["abstract"]) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "Abstract", context["abstract"] │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - // optional DataURL element │ │ │ │ │ - if (context.dataURL) { │ │ │ │ │ - node.appendChild(this.write_wmc_URLType("DataURL", context.dataURL)); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: encodeUnsignedIntegers │ │ │ │ │ + * Encode a list of unsigned integers and return an encoded string │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * numbers - {Array.<number>} A list of unsigned integers. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {string} The encoded string. │ │ │ │ │ + */ │ │ │ │ │ + encodeUnsignedIntegers: function(numbers) { │ │ │ │ │ + var encoded = ''; │ │ │ │ │ │ │ │ │ │ - // optional MetadataURL element │ │ │ │ │ - if (context.metadataURL) { │ │ │ │ │ - node.appendChild(this.write_wmc_URLType("MetadataURL", context.metadataURL)); │ │ │ │ │ + var numbersLength = numbers.length; │ │ │ │ │ + for (var i = 0; i < numbersLength; ++i) { │ │ │ │ │ + encoded += this.encodeUnsignedInteger(numbers[i]); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return node; │ │ │ │ │ + return encoded; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: write_wmc_LayerExtension │ │ │ │ │ - * Add OpenLayers specific layer parameters to an Extension element. │ │ │ │ │ + * APIMethod: decodeUnsignedIntegers │ │ │ │ │ + * Decode a list of unsigned integers from an encoded string │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * context - {Object} A layer context object. │ │ │ │ │ + * encoded - {string} An encoded string. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Element} A WMC Extension element (for a layer). │ │ │ │ │ + * {Array.<number>} A list of unsigned integers. │ │ │ │ │ */ │ │ │ │ │ - write_wmc_LayerExtension: function(context) { │ │ │ │ │ - var node = this.createElementDefaultNS("Extension"); │ │ │ │ │ + decodeUnsignedIntegers: function(encoded) { │ │ │ │ │ + var numbers = []; │ │ │ │ │ │ │ │ │ │ - var bounds = context.maxExtent; │ │ │ │ │ - var maxExtent = this.createElementNS( │ │ │ │ │ - this.namespaces.ol, "ol:maxExtent" │ │ │ │ │ - ); │ │ │ │ │ - this.setAttributes(maxExtent, { │ │ │ │ │ - minx: bounds.left.toPrecision(18), │ │ │ │ │ - miny: bounds.bottom.toPrecision(18), │ │ │ │ │ - maxx: bounds.right.toPrecision(18), │ │ │ │ │ - maxy: bounds.top.toPrecision(18) │ │ │ │ │ - }); │ │ │ │ │ - node.appendChild(maxExtent); │ │ │ │ │ + var current = 0; │ │ │ │ │ + var shift = 0; │ │ │ │ │ │ │ │ │ │ - if (context.tileSize && !context.singleTile) { │ │ │ │ │ - var size = this.createElementNS( │ │ │ │ │ - this.namespaces.ol, "ol:tileSize" │ │ │ │ │ - ); │ │ │ │ │ - this.setAttributes(size, context.tileSize); │ │ │ │ │ - node.appendChild(size); │ │ │ │ │ - } │ │ │ │ │ + var encodedLength = encoded.length; │ │ │ │ │ + for (var i = 0; i < encodedLength; ++i) { │ │ │ │ │ + var b = encoded.charCodeAt(i) - 63; │ │ │ │ │ │ │ │ │ │ - var properties = [ │ │ │ │ │ - "transparent", "numZoomLevels", "units", "isBaseLayer", │ │ │ │ │ - "opacity", "displayInLayerSwitcher", "singleTile" │ │ │ │ │ - ]; │ │ │ │ │ - var child; │ │ │ │ │ - for (var i = 0, len = properties.length; i < len; ++i) { │ │ │ │ │ - child = this.createOLPropertyNode(context, properties[i]); │ │ │ │ │ - if (child) { │ │ │ │ │ - node.appendChild(child); │ │ │ │ │ + current |= (b & 0x1f) << shift; │ │ │ │ │ + │ │ │ │ │ + if (b < 0x20) { │ │ │ │ │ + numbers.push(current); │ │ │ │ │ + current = 0; │ │ │ │ │ + shift = 0; │ │ │ │ │ + } else { │ │ │ │ │ + shift += 5; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return node; │ │ │ │ │ + return numbers; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: createOLPropertyNode │ │ │ │ │ - * Create a node representing an OpenLayers property. If the property is │ │ │ │ │ - * null or undefined, null will be returned. │ │ │ │ │ + * Method: encodeFloat │ │ │ │ │ + * Encode one single floating point number and return an encoded string │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * obj - {Object} An object. │ │ │ │ │ - * prop - {String} A property. │ │ │ │ │ + * num - {number} Floating point number that should be encoded. │ │ │ │ │ + * opt_factor - {number=} The factor by which num will be multiplied. │ │ │ │ │ + * The remaining decimal places will get rounded away. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Element} A property node. │ │ │ │ │ + * {string} The encoded string. │ │ │ │ │ */ │ │ │ │ │ - createOLPropertyNode: function(obj, prop) { │ │ │ │ │ - var node = null; │ │ │ │ │ - if (obj[prop] != null) { │ │ │ │ │ - node = this.createElementNS(this.namespaces.ol, "ol:" + prop); │ │ │ │ │ - node.appendChild(this.createTextNode(obj[prop].toString())); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ + encodeFloat: function(num, opt_factor) { │ │ │ │ │ + num = Math.round(num * (opt_factor || 1e5)); │ │ │ │ │ + return this.encodeSignedInteger(num); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: write_wmc_Server │ │ │ │ │ - * Create a Server node given a layer context object. │ │ │ │ │ + * Method: decodeFloat │ │ │ │ │ + * Decode one single floating point number from an encoded string │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * context - {Object} Layer context object. │ │ │ │ │ + * encoded - {string} An encoded string. │ │ │ │ │ + * opt_factor - {number=} The factor by which the result will be divided. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Element} A WMC Server element node. │ │ │ │ │ + * {number} The decoded floating point number. │ │ │ │ │ */ │ │ │ │ │ - write_wmc_Server: function(context) { │ │ │ │ │ - var server = context.server; │ │ │ │ │ - var node = this.createElementDefaultNS("Server"); │ │ │ │ │ - var attributes = { │ │ │ │ │ - service: "OGC:WMS", │ │ │ │ │ - version: server.version │ │ │ │ │ - }; │ │ │ │ │ - if (server.title) { │ │ │ │ │ - attributes.title = server.title; │ │ │ │ │ - } │ │ │ │ │ - this.setAttributes(node, attributes); │ │ │ │ │ - │ │ │ │ │ - // required OnlineResource element │ │ │ │ │ - node.appendChild(this.write_wmc_OnlineResource(server.url)); │ │ │ │ │ - │ │ │ │ │ - return node; │ │ │ │ │ + decodeFloat: function(encoded, opt_factor) { │ │ │ │ │ + var result = this.decodeSignedInteger(encoded); │ │ │ │ │ + return result / (opt_factor || 1e5); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: write_wmc_URLType │ │ │ │ │ - * Create a LogoURL/DescriptionURL/MetadataURL/DataURL/LegendURL node given a object and elementName. │ │ │ │ │ + * Method: encodeSignedInteger │ │ │ │ │ + * Encode one single signed integer and return an encoded string │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * elName - {String} Name of element (LogoURL/DescriptionURL/MetadataURL/LegendURL) │ │ │ │ │ - * url - {String} URL string value │ │ │ │ │ - * attr - {Object} Optional attributes (width, height, format) │ │ │ │ │ + * num - {number} Signed integer that should be encoded. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Element} A WMC element node. │ │ │ │ │ + * {string} The encoded string. │ │ │ │ │ */ │ │ │ │ │ - write_wmc_URLType: function(elName, url, attr) { │ │ │ │ │ - var node = this.createElementDefaultNS(elName); │ │ │ │ │ - node.appendChild(this.write_wmc_OnlineResource(url)); │ │ │ │ │ - if (attr) { │ │ │ │ │ - var optionalAttributes = ["width", "height", "format"]; │ │ │ │ │ - for (var i = 0; i < optionalAttributes.length; i++) { │ │ │ │ │ - if (optionalAttributes[i] in attr) { │ │ │ │ │ - node.setAttribute(optionalAttributes[i], attr[optionalAttributes[i]]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + encodeSignedInteger: function(num) { │ │ │ │ │ + var signedNum = num << 1; │ │ │ │ │ + if (num < 0) { │ │ │ │ │ + signedNum = ~(signedNum); │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: write_wmc_DimensionList │ │ │ │ │ - */ │ │ │ │ │ - write_wmc_DimensionList: function(context) { │ │ │ │ │ - var node = this.createElementDefaultNS("DimensionList"); │ │ │ │ │ - var required_attributes = { │ │ │ │ │ - name: true, │ │ │ │ │ - units: true, │ │ │ │ │ - unitSymbol: true, │ │ │ │ │ - userValue: true │ │ │ │ │ - }; │ │ │ │ │ - for (var dim in context.dimensions) { │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - var dimension = context.dimensions[dim]; │ │ │ │ │ - for (var name in dimension) { │ │ │ │ │ - if (typeof dimension[name] == "boolean") { │ │ │ │ │ - attributes[name] = Number(dimension[name]); │ │ │ │ │ - } else { │ │ │ │ │ - attributes[name] = dimension[name]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var values = ""; │ │ │ │ │ - if (attributes.values) { │ │ │ │ │ - values = attributes.values.join(","); │ │ │ │ │ - delete attributes.values; │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "Dimension", values, attributes │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ + return this.encodeUnsignedInteger(signedNum); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: write_wmc_FormatList │ │ │ │ │ - * Create a FormatList node given a layer context. │ │ │ │ │ + * Method: decodeSignedInteger │ │ │ │ │ + * Decode one single signed integer from an encoded string │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * context - {Object} Layer context object. │ │ │ │ │ + * encoded - {string} An encoded string. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Element} A WMC FormatList element node. │ │ │ │ │ + * {number} The decoded signed integer. │ │ │ │ │ */ │ │ │ │ │ - write_wmc_FormatList: function(context) { │ │ │ │ │ - var node = this.createElementDefaultNS("FormatList"); │ │ │ │ │ - for (var i = 0, len = context.formats.length; i < len; i++) { │ │ │ │ │ - var format = context.formats[i]; │ │ │ │ │ - node.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "Format", │ │ │ │ │ - format.value, │ │ │ │ │ - (format.current && format.current == true) ? { │ │ │ │ │ - current: "1" │ │ │ │ │ - } : null │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return node; │ │ │ │ │ + decodeSignedInteger: function(encoded) { │ │ │ │ │ + var result = this.decodeUnsignedInteger(encoded); │ │ │ │ │ + return ((result & 1) ? ~(result >> 1) : (result >> 1)); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: write_wmc_StyleList │ │ │ │ │ - * Create a StyleList node given a layer context. │ │ │ │ │ + * Method: encodeUnsignedInteger │ │ │ │ │ + * Encode one single unsigned integer and return an encoded string │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layer - {Object} Layer context object. │ │ │ │ │ + * num - {number} Unsigned integer that should be encoded. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Element} A WMC StyleList element node. │ │ │ │ │ + * {string} The encoded string. │ │ │ │ │ */ │ │ │ │ │ - write_wmc_StyleList: function(layer) { │ │ │ │ │ - var node = this.createElementDefaultNS("StyleList"); │ │ │ │ │ - │ │ │ │ │ - var styles = layer.styles; │ │ │ │ │ - if (styles && OpenLayers.Util.isArray(styles)) { │ │ │ │ │ - var sld; │ │ │ │ │ - for (var i = 0, len = styles.length; i < len; i++) { │ │ │ │ │ - var s = styles[i]; │ │ │ │ │ - // three style types to consider │ │ │ │ │ - // [1] linked SLD │ │ │ │ │ - // [2] inline SLD │ │ │ │ │ - // [3] named style │ │ │ │ │ - // running child nodes always gets name, optionally gets href or body │ │ │ │ │ - var style = this.createElementDefaultNS( │ │ │ │ │ - "Style", │ │ │ │ │ - null, │ │ │ │ │ - (s.current && s.current == true) ? { │ │ │ │ │ - current: "1" │ │ │ │ │ - } : null │ │ │ │ │ - ); │ │ │ │ │ - if (s.href) { // [1] │ │ │ │ │ - sld = this.createElementDefaultNS("SLD"); │ │ │ │ │ - // Name is optional. │ │ │ │ │ - if (s.name) { │ │ │ │ │ - sld.appendChild(this.createElementDefaultNS("Name", s.name)); │ │ │ │ │ - } │ │ │ │ │ - // Title is optional. │ │ │ │ │ - if (s.title) { │ │ │ │ │ - sld.appendChild(this.createElementDefaultNS("Title", s.title)); │ │ │ │ │ - } │ │ │ │ │ - // LegendURL is optional │ │ │ │ │ - if (s.legend) { │ │ │ │ │ - sld.appendChild(this.write_wmc_URLType("LegendURL", s.legend.href, s.legend)); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var link = this.write_wmc_OnlineResource(s.href); │ │ │ │ │ - sld.appendChild(link); │ │ │ │ │ - style.appendChild(sld); │ │ │ │ │ - } else if (s.body) { // [2] │ │ │ │ │ - sld = this.createElementDefaultNS("SLD"); │ │ │ │ │ - // Name is optional. │ │ │ │ │ - if (s.name) { │ │ │ │ │ - sld.appendChild(this.createElementDefaultNS("Name", s.name)); │ │ │ │ │ - } │ │ │ │ │ - // Title is optional. │ │ │ │ │ - if (s.title) { │ │ │ │ │ - sld.appendChild(this.createElementDefaultNS("Title", s.title)); │ │ │ │ │ - } │ │ │ │ │ - // LegendURL is optional │ │ │ │ │ - if (s.legend) { │ │ │ │ │ - sld.appendChild(this.write_wmc_URLType("LegendURL", s.legend.href, s.legend)); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // read in body as xml doc - assume proper namespace declarations │ │ │ │ │ - var doc = OpenLayers.Format.XML.prototype.read.apply(this, [s.body]); │ │ │ │ │ - // append to StyledLayerDescriptor node │ │ │ │ │ - var imported = doc.documentElement; │ │ │ │ │ - if (sld.ownerDocument && sld.ownerDocument.importNode) { │ │ │ │ │ - imported = sld.ownerDocument.importNode(imported, true); │ │ │ │ │ - } │ │ │ │ │ - sld.appendChild(imported); │ │ │ │ │ - style.appendChild(sld); │ │ │ │ │ - } else { // [3] │ │ │ │ │ - // both Name and Title are required. │ │ │ │ │ - style.appendChild(this.createElementDefaultNS("Name", s.name)); │ │ │ │ │ - style.appendChild(this.createElementDefaultNS("Title", s.title)); │ │ │ │ │ - // Abstract is optional │ │ │ │ │ - if (s['abstract']) { // abstract is a js keyword │ │ │ │ │ - style.appendChild(this.createElementDefaultNS( │ │ │ │ │ - "Abstract", s['abstract'] │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - // LegendURL is optional │ │ │ │ │ - if (s.legend) { │ │ │ │ │ - style.appendChild(this.write_wmc_URLType("LegendURL", s.legend.href, s.legend)); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - node.appendChild(style); │ │ │ │ │ - } │ │ │ │ │ + encodeUnsignedInteger: function(num) { │ │ │ │ │ + var value, encoded = ''; │ │ │ │ │ + while (num >= 0x20) { │ │ │ │ │ + value = (0x20 | (num & 0x1f)) + 63; │ │ │ │ │ + encoded += (String.fromCharCode(value)); │ │ │ │ │ + num >>= 5; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return node; │ │ │ │ │ + value = num + 63; │ │ │ │ │ + encoded += (String.fromCharCode(value)); │ │ │ │ │ + return encoded; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: write_wmc_OnlineResource │ │ │ │ │ - * Create an OnlineResource node given a URL. │ │ │ │ │ + * Method: decodeUnsignedInteger │ │ │ │ │ + * Decode one single unsigned integer from an encoded string │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * href - {String} URL for the resource. │ │ │ │ │ + * encoded - {string} An encoded string. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Element} A WMC OnlineResource element node. │ │ │ │ │ + * {number} The decoded unsigned integer. │ │ │ │ │ */ │ │ │ │ │ - write_wmc_OnlineResource: function(href) { │ │ │ │ │ - var node = this.createElementDefaultNS("OnlineResource"); │ │ │ │ │ - this.setAttributeNS(node, this.namespaces.xlink, "xlink:type", "simple"); │ │ │ │ │ - this.setAttributeNS(node, this.namespaces.xlink, "xlink:href", href); │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ + decodeUnsignedInteger: function(encoded) { │ │ │ │ │ + var result = 0; │ │ │ │ │ + var shift = 0; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getOnlineResource_href │ │ │ │ │ - */ │ │ │ │ │ - getOnlineResource_href: function(node) { │ │ │ │ │ - var object = {}; │ │ │ │ │ - var links = node.getElementsByTagName("OnlineResource"); │ │ │ │ │ - if (links.length > 0) { │ │ │ │ │ - this.read_wmc_OnlineResource(object, links[0]); │ │ │ │ │ - } │ │ │ │ │ - return object.href; │ │ │ │ │ - }, │ │ │ │ │ + var encodedLength = encoded.length; │ │ │ │ │ + for (var i = 0; i < encodedLength; ++i) { │ │ │ │ │ + var b = encoded.charCodeAt(i) - 63; │ │ │ │ │ + │ │ │ │ │ + result |= (b & 0x1f) << shift; │ │ │ │ │ │ │ │ │ │ + if (b < 0x20) │ │ │ │ │ + break; │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMC.v1" │ │ │ │ │ + shift += 5; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return result; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.EncodedPolyline" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMC/v1_0_0.js │ │ │ │ │ + OpenLayers/Format/SOSCapabilities.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/WMC/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WMC.v1_0_0 │ │ │ │ │ - * Read and write WMC version 1.0.0. │ │ │ │ │ + * Class: OpenLayers.Format.SOSCapabilities │ │ │ │ │ + * Read SOS Capabilities. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMC.v1> │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WMC.v1_0_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WMC.v1, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: VERSION │ │ │ │ │ - * {String} 1.0.0 │ │ │ │ │ - */ │ │ │ │ │ - VERSION: "1.0.0", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} http://www.opengis.net/context │ │ │ │ │ - * http://schemas.opengis.net/context/1.0.0/context.xsd │ │ │ │ │ - */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/context http://schemas.opengis.net/context/1.0.0/context.xsd", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMC.v1_0_0 │ │ │ │ │ - * Instances of this class are not created directly. Use the │ │ │ │ │ - * <OpenLayers.Format.WMC> constructor instead. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.WMC.v1.prototype.initialize.apply( │ │ │ │ │ - this, [options] │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_SRS │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_SRS: function(layerContext, node) { │ │ │ │ │ - var srs = this.getChildValue(node); │ │ │ │ │ - if (typeof layerContext.projections != "object") { │ │ │ │ │ - layerContext.projections = {}; │ │ │ │ │ - } │ │ │ │ │ - var values = srs.split(/ +/); │ │ │ │ │ - for (var i = 0, len = values.length; i < len; i++) { │ │ │ │ │ - layerContext.projections[values[i]] = true; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: write_wmc_Layer │ │ │ │ │ - * Create a Layer node given a layer context object. This method adds │ │ │ │ │ - * elements specific to version 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * context - {Object} A layer context object.} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Element} A WMC Layer element node. │ │ │ │ │ - */ │ │ │ │ │ - write_wmc_Layer: function(context) { │ │ │ │ │ - var node = OpenLayers.Format.WMC.v1.prototype.write_wmc_Layer.apply( │ │ │ │ │ - this, [context] │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - // optional SRS element(s) │ │ │ │ │ - if (context.srs) { │ │ │ │ │ - var projections = []; │ │ │ │ │ - for (var name in context.srs) { │ │ │ │ │ - projections.push(name); │ │ │ │ │ - } │ │ │ │ │ - node.appendChild(this.createElementDefaultNS("SRS", projections.join(" "))); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // optional FormatList element │ │ │ │ │ - node.appendChild(this.write_wmc_FormatList(context)); │ │ │ │ │ +OpenLayers.Format.SOSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ │ │ │ │ │ - // optional StyleList element │ │ │ │ │ - node.appendChild(this.write_wmc_StyleList(context)); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.0.0". │ │ │ │ │ + */ │ │ │ │ │ + defaultVersion: "1.0.0", │ │ │ │ │ │ │ │ │ │ - // optional DimensionList element │ │ │ │ │ - if (context.dimensions) { │ │ │ │ │ - node.appendChild(this.write_wmc_DimensionList(context)); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.SOSCapabilities │ │ │ │ │ + * Create a new parser for SOS Capabilities. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // OpenLayers specific properties go in an Extension element │ │ │ │ │ - node.appendChild(this.write_wmc_LayerExtension(context)); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return information about │ │ │ │ │ + * the service (offering and observedProperty mostly). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Info about the SOS │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMC.v1_0_0" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SOSCapabilities" │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMC/v1_1_0.js │ │ │ │ │ + OpenLayers/Format/CQL.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/WMC/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/WKT.js │ │ │ │ │ + * @requires OpenLayers/Filter/Comparison.js │ │ │ │ │ + * @requires OpenLayers/Filter/Logical.js │ │ │ │ │ + * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WMC.v1_1_0 │ │ │ │ │ - * Read and write WMC version 1.1.0. │ │ │ │ │ + * Class: OpenLayers.Format.CQL │ │ │ │ │ + * Read CQL strings to get <OpenLayers.Filter> objects. Write │ │ │ │ │ + * <OpenLayers.Filter> objects to get CQL strings. Create a new parser with │ │ │ │ │ + * the <OpenLayers.Format.CQL> constructor. │ │ │ │ │ * │ │ │ │ │ - * Differences between 1.1.0 and 1.0.0: │ │ │ │ │ - * - 1.1.0 Layers have optional sld:MinScaleDenominator and │ │ │ │ │ - * sld:MaxScaleDenominator │ │ │ │ │ - * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMC.v1> │ │ │ │ │ + * - <OpenLayers.Format> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WMC.v1_1_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WMC.v1, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constant: VERSION │ │ │ │ │ - * {String} 1.1.0 │ │ │ │ │ - */ │ │ │ │ │ - VERSION: "1.1.0", │ │ │ │ │ +OpenLayers.Format.CQL = (function() { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: schemaLocation │ │ │ │ │ - * {String} http://www.opengis.net/context │ │ │ │ │ - * http://schemas.opengis.net/context/1.1.0/context.xsd │ │ │ │ │ - */ │ │ │ │ │ - schemaLocation: "http://www.opengis.net/context http://schemas.opengis.net/context/1.1.0/context.xsd", │ │ │ │ │ + var tokens = [ │ │ │ │ │ + "PROPERTY", "COMPARISON", "VALUE", "LOGICAL" │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMC.v1_1_0 │ │ │ │ │ - * Instances of this class are not created directly. Use the │ │ │ │ │ - * <OpenLayers.Format.WMC> constructor instead. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.WMC.v1.prototype.initialize.apply( │ │ │ │ │ - this, [options] │ │ │ │ │ - ); │ │ │ │ │ + patterns = { │ │ │ │ │ + PROPERTY: /^[_a-zA-Z]\w*/, │ │ │ │ │ + COMPARISON: /^(=|<>|<=|<|>=|>|LIKE)/i, │ │ │ │ │ + IS_NULL: /^IS NULL/i, │ │ │ │ │ + COMMA: /^,/, │ │ │ │ │ + LOGICAL: /^(AND|OR)/i, │ │ │ │ │ + VALUE: /^('([^']|'')*'|\d+(\.\d*)?|\.\d+)/, │ │ │ │ │ + LPAREN: /^\(/, │ │ │ │ │ + RPAREN: /^\)/, │ │ │ │ │ + SPATIAL: /^(BBOX|INTERSECTS|DWITHIN|WITHIN|CONTAINS)/i, │ │ │ │ │ + NOT: /^NOT/i, │ │ │ │ │ + BETWEEN: /^BETWEEN/i, │ │ │ │ │ + GEOMETRY: function(text) { │ │ │ │ │ + var type = /^(POINT|LINESTRING|POLYGON|MULTIPOINT|MULTILINESTRING|MULTIPOLYGON|GEOMETRYCOLLECTION)/.exec(text); │ │ │ │ │ + if (type) { │ │ │ │ │ + var len = text.length; │ │ │ │ │ + var idx = text.indexOf("(", type[0].length); │ │ │ │ │ + if (idx > -1) { │ │ │ │ │ + var depth = 1; │ │ │ │ │ + while (idx < len && depth > 0) { │ │ │ │ │ + idx++; │ │ │ │ │ + switch (text.charAt(idx)) { │ │ │ │ │ + case '(': │ │ │ │ │ + depth++; │ │ │ │ │ + break; │ │ │ │ │ + case ')': │ │ │ │ │ + depth--; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + // in default case, do nothing │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return [text.substr(0, idx + 1)]; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + END: /^$/ │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_sld_MinScaleDenominator │ │ │ │ │ - * Read a sld:MinScaleDenominator node. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layerContext - {Object} An object representing a layer. │ │ │ │ │ - * node - {Element} An element node. │ │ │ │ │ - */ │ │ │ │ │ - read_sld_MinScaleDenominator: function(layerContext, node) { │ │ │ │ │ - var minScaleDenominator = parseFloat(this.getChildValue(node)); │ │ │ │ │ - if (minScaleDenominator > 0) { │ │ │ │ │ - layerContext.maxScale = minScaleDenominator; │ │ │ │ │ - } │ │ │ │ │ + follows = { │ │ │ │ │ + LPAREN: ['GEOMETRY', 'SPATIAL', 'PROPERTY', 'VALUE', 'LPAREN'], │ │ │ │ │ + RPAREN: ['NOT', 'LOGICAL', 'END', 'RPAREN'], │ │ │ │ │ + PROPERTY: ['COMPARISON', 'BETWEEN', 'COMMA', 'IS_NULL'], │ │ │ │ │ + BETWEEN: ['VALUE'], │ │ │ │ │ + IS_NULL: ['END'], │ │ │ │ │ + COMPARISON: ['VALUE'], │ │ │ │ │ + COMMA: ['GEOMETRY', 'VALUE', 'PROPERTY'], │ │ │ │ │ + VALUE: ['LOGICAL', 'COMMA', 'RPAREN', 'END'], │ │ │ │ │ + SPATIAL: ['LPAREN'], │ │ │ │ │ + LOGICAL: ['NOT', 'VALUE', 'SPATIAL', 'PROPERTY', 'LPAREN'], │ │ │ │ │ + NOT: ['PROPERTY', 'LPAREN'], │ │ │ │ │ + GEOMETRY: ['COMMA', 'RPAREN'] │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_sld_MaxScaleDenominator │ │ │ │ │ - * Read a sld:MaxScaleDenominator node. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layerContext - {Object} An object representing a layer. │ │ │ │ │ - * node - {Element} An element node. │ │ │ │ │ - */ │ │ │ │ │ - read_sld_MaxScaleDenominator: function(layerContext, node) { │ │ │ │ │ - layerContext.minScale = parseFloat(this.getChildValue(node)); │ │ │ │ │ + operators = { │ │ │ │ │ + '=': OpenLayers.Filter.Comparison.EQUAL_TO, │ │ │ │ │ + '<>': OpenLayers.Filter.Comparison.NOT_EQUAL_TO, │ │ │ │ │ + '<': OpenLayers.Filter.Comparison.LESS_THAN, │ │ │ │ │ + '<=': OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO, │ │ │ │ │ + '>': OpenLayers.Filter.Comparison.GREATER_THAN, │ │ │ │ │ + '>=': OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO, │ │ │ │ │ + 'LIKE': OpenLayers.Filter.Comparison.LIKE, │ │ │ │ │ + 'BETWEEN': OpenLayers.Filter.Comparison.BETWEEN, │ │ │ │ │ + 'IS NULL': OpenLayers.Filter.Comparison.IS_NULL │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read_wmc_SRS │ │ │ │ │ - */ │ │ │ │ │ - read_wmc_SRS: function(layerContext, node) { │ │ │ │ │ - if (!("srs" in layerContext)) { │ │ │ │ │ - layerContext.srs = {}; │ │ │ │ │ - } │ │ │ │ │ - layerContext.srs[this.getChildValue(node)] = true; │ │ │ │ │ + operatorReverse = {}, │ │ │ │ │ + │ │ │ │ │ + logicals = { │ │ │ │ │ + 'AND': OpenLayers.Filter.Logical.AND, │ │ │ │ │ + 'OR': OpenLayers.Filter.Logical.OR │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: write_wmc_Layer │ │ │ │ │ - * Create a Layer node given a layer context object. This method adds │ │ │ │ │ - * elements specific to version 1.1.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * context - {Object} A layer context object.} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Element} A WMC Layer element node. │ │ │ │ │ - */ │ │ │ │ │ - write_wmc_Layer: function(context) { │ │ │ │ │ - var node = OpenLayers.Format.WMC.v1.prototype.write_wmc_Layer.apply( │ │ │ │ │ - this, [context] │ │ │ │ │ - ); │ │ │ │ │ + logicalReverse = {}, │ │ │ │ │ │ │ │ │ │ - // min/max scale denominator elements go before the 4th element in v1 │ │ │ │ │ - if (context.maxScale) { │ │ │ │ │ - var minSD = this.createElementNS( │ │ │ │ │ - this.namespaces.sld, "sld:MinScaleDenominator" │ │ │ │ │ - ); │ │ │ │ │ - minSD.appendChild(this.createTextNode(context.maxScale.toPrecision(16))); │ │ │ │ │ - node.appendChild(minSD); │ │ │ │ │ - } │ │ │ │ │ + precedence = { │ │ │ │ │ + 'RPAREN': 3, │ │ │ │ │ + 'LOGICAL': 2, │ │ │ │ │ + 'COMPARISON': 1 │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - if (context.minScale) { │ │ │ │ │ - var maxSD = this.createElementNS( │ │ │ │ │ - this.namespaces.sld, "sld:MaxScaleDenominator" │ │ │ │ │ - ); │ │ │ │ │ - maxSD.appendChild(this.createTextNode(context.minScale.toPrecision(16))); │ │ │ │ │ - node.appendChild(maxSD); │ │ │ │ │ - } │ │ │ │ │ + var i; │ │ │ │ │ + for (i in operators) { │ │ │ │ │ + if (operators.hasOwnProperty(i)) { │ │ │ │ │ + operatorReverse[operators[i]] = i; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // optional SRS element(s) │ │ │ │ │ - if (context.srs) { │ │ │ │ │ - for (var name in context.srs) { │ │ │ │ │ - node.appendChild(this.createElementDefaultNS("SRS", name)); │ │ │ │ │ - } │ │ │ │ │ + for (i in logicals) { │ │ │ │ │ + if (logicals.hasOwnProperty(i)) { │ │ │ │ │ + logicalReverse[logicals[i]] = i; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function tryToken(text, pattern) { │ │ │ │ │ + if (pattern instanceof RegExp) { │ │ │ │ │ + return pattern.exec(text); │ │ │ │ │ + } else { │ │ │ │ │ + return pattern(text); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function nextToken(text, tokens) { │ │ │ │ │ + var i, token, len = tokens.length; │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + token = tokens[i]; │ │ │ │ │ + var pat = patterns[token]; │ │ │ │ │ + var matches = tryToken(text, pat); │ │ │ │ │ + if (matches) { │ │ │ │ │ + var match = matches[0]; │ │ │ │ │ + var remainder = text.substr(match.length).replace(/^\s*/, ""); │ │ │ │ │ + return { │ │ │ │ │ + type: token, │ │ │ │ │ + text: match, │ │ │ │ │ + remainder: remainder │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // optional FormatList element │ │ │ │ │ - node.appendChild(this.write_wmc_FormatList(context)); │ │ │ │ │ + var msg = "ERROR: In parsing: [" + text + "], expected one of: "; │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + token = tokens[i]; │ │ │ │ │ + msg += "\n " + token + ": " + patterns[token]; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // optional StyleList element │ │ │ │ │ - node.appendChild(this.write_wmc_StyleList(context)); │ │ │ │ │ + throw new Error(msg); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // optional DimensionList element │ │ │ │ │ - if (context.dimensions) { │ │ │ │ │ - node.appendChild(this.write_wmc_DimensionList(context)); │ │ │ │ │ + function tokenize(text) { │ │ │ │ │ + var results = []; │ │ │ │ │ + var token, expect = ["NOT", "GEOMETRY", "SPATIAL", "PROPERTY", "LPAREN"]; │ │ │ │ │ + │ │ │ │ │ + do { │ │ │ │ │ + token = nextToken(text, expect); │ │ │ │ │ + text = token.remainder; │ │ │ │ │ + expect = follows[token.type]; │ │ │ │ │ + if (token.type != "END" && !expect) { │ │ │ │ │ + throw new Error("No follows list for " + token.type); │ │ │ │ │ } │ │ │ │ │ + results.push(token); │ │ │ │ │ + } while (token.type != "END"); │ │ │ │ │ │ │ │ │ │ - // OpenLayers specific properties go in an Extension element │ │ │ │ │ - node.appendChild(this.write_wmc_LayerExtension(context)); │ │ │ │ │ + return results; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - return node; │ │ │ │ │ + function buildAst(tokens) { │ │ │ │ │ + var operatorStack = [], │ │ │ │ │ + postfix = []; │ │ │ │ │ │ │ │ │ │ - }, │ │ │ │ │ + while (tokens.length) { │ │ │ │ │ + var tok = tokens.shift(); │ │ │ │ │ + switch (tok.type) { │ │ │ │ │ + case "PROPERTY": │ │ │ │ │ + case "GEOMETRY": │ │ │ │ │ + case "VALUE": │ │ │ │ │ + postfix.push(tok); │ │ │ │ │ + break; │ │ │ │ │ + case "COMPARISON": │ │ │ │ │ + case "BETWEEN": │ │ │ │ │ + case "IS_NULL": │ │ │ │ │ + case "LOGICAL": │ │ │ │ │ + var p = precedence[tok.type]; │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMC.v1_1_0" │ │ │ │ │ + while (operatorStack.length > 0 && │ │ │ │ │ + (precedence[operatorStack[operatorStack.length - 1].type] <= p) │ │ │ │ │ + ) { │ │ │ │ │ + postfix.push(operatorStack.pop()); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSCapabilities/v1.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + operatorStack.push(tok); │ │ │ │ │ + break; │ │ │ │ │ + case "SPATIAL": │ │ │ │ │ + case "NOT": │ │ │ │ │ + case "LPAREN": │ │ │ │ │ + operatorStack.push(tok); │ │ │ │ │ + break; │ │ │ │ │ + case "RPAREN": │ │ │ │ │ + while (operatorStack.length > 0 && │ │ │ │ │ + (operatorStack[operatorStack.length - 1].type != "LPAREN") │ │ │ │ │ + ) { │ │ │ │ │ + postfix.push(operatorStack.pop()); │ │ │ │ │ + } │ │ │ │ │ + operatorStack.pop(); // toss out the LPAREN │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + if (operatorStack.length > 0 && │ │ │ │ │ + operatorStack[operatorStack.length - 1].type == "SPATIAL") { │ │ │ │ │ + postfix.push(operatorStack.pop()); │ │ │ │ │ + } │ │ │ │ │ + case "COMMA": │ │ │ │ │ + case "END": │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + throw new Error("Unknown token type " + tok.type); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WMSCapabilities.js │ │ │ │ │ - * @requires OpenLayers/Format/OGCExceptionReport.js │ │ │ │ │ - * @requires OpenLayers/Format/XML.js │ │ │ │ │ - */ │ │ │ │ │ + while (operatorStack.length > 0) { │ │ │ │ │ + postfix.push(operatorStack.pop()); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WMSCapabilities.v1 │ │ │ │ │ - * Abstract class not to be instantiated directly. Creates │ │ │ │ │ - * the common parts for both WMS 1.1.X and WMS 1.3.X. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.XML, { │ │ │ │ │ + function buildTree() { │ │ │ │ │ + var tok = postfix.pop(); │ │ │ │ │ + switch (tok.type) { │ │ │ │ │ + case "LOGICAL": │ │ │ │ │ + var rhs = buildTree(), │ │ │ │ │ + lhs = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Logical({ │ │ │ │ │ + filters: [lhs, rhs], │ │ │ │ │ + type: logicals[tok.text.toUpperCase()] │ │ │ │ │ + }); │ │ │ │ │ + case "NOT": │ │ │ │ │ + var operand = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Logical({ │ │ │ │ │ + filters: [operand], │ │ │ │ │ + type: OpenLayers.Filter.Logical.NOT │ │ │ │ │ + }); │ │ │ │ │ + case "BETWEEN": │ │ │ │ │ + var min, max, property; │ │ │ │ │ + postfix.pop(); // unneeded AND token here │ │ │ │ │ + max = buildTree(); │ │ │ │ │ + min = buildTree(); │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Comparison({ │ │ │ │ │ + property: property, │ │ │ │ │ + lowerBoundary: min, │ │ │ │ │ + upperBoundary: max, │ │ │ │ │ + type: OpenLayers.Filter.Comparison.BETWEEN │ │ │ │ │ + }); │ │ │ │ │ + case "COMPARISON": │ │ │ │ │ + var value = buildTree(), │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Comparison({ │ │ │ │ │ + property: property, │ │ │ │ │ + value: value, │ │ │ │ │ + type: operators[tok.text.toUpperCase()] │ │ │ │ │ + }); │ │ │ │ │ + case "IS_NULL": │ │ │ │ │ + var property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Comparison({ │ │ │ │ │ + property: property, │ │ │ │ │ + type: operators[tok.text.toUpperCase()] │ │ │ │ │ + }); │ │ │ │ │ + case "VALUE": │ │ │ │ │ + var match = tok.text.match(/^'(.*)'$/); │ │ │ │ │ + if (match) { │ │ │ │ │ + return match[1].replace(/''/g, "'"); │ │ │ │ │ + } else { │ │ │ │ │ + return Number(tok.text); │ │ │ │ │ + } │ │ │ │ │ + case "SPATIAL": │ │ │ │ │ + switch (tok.text.toUpperCase()) { │ │ │ │ │ + case "BBOX": │ │ │ │ │ + var maxy = buildTree(), │ │ │ │ │ + maxx = buildTree(), │ │ │ │ │ + miny = buildTree(), │ │ │ │ │ + minx = buildTree(), │ │ │ │ │ + prop = buildTree(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - wms: "http://www.opengis.net/wms", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ - }, │ │ │ │ │ + return new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ + property: prop, │ │ │ │ │ + value: OpenLayers.Bounds.fromArray( │ │ │ │ │ + [minx, miny, maxx, maxy] │ │ │ │ │ + ) │ │ │ │ │ + }); │ │ │ │ │ + case "INTERSECTS": │ │ │ │ │ + var value = buildTree(), │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: property, │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + case "WITHIN": │ │ │ │ │ + var value = buildTree(), │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.WITHIN, │ │ │ │ │ + property: property, │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + case "CONTAINS": │ │ │ │ │ + var value = buildTree(), │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.CONTAINS, │ │ │ │ │ + property: property, │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + case "DWITHIN": │ │ │ │ │ + var distance = buildTree(), │ │ │ │ │ + value = buildTree(), │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ + value: value, │ │ │ │ │ + property: property, │ │ │ │ │ + distance: Number(distance) │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + case "GEOMETRY": │ │ │ │ │ + return OpenLayers.Geometry.fromWKT(tok.text); │ │ │ │ │ + default: │ │ │ │ │ + return tok.text; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - */ │ │ │ │ │ - defaultPrefix: "wms", │ │ │ │ │ + var result = buildTree(); │ │ │ │ │ + if (postfix.length > 0) { │ │ │ │ │ + var msg = "Remaining tokens after building AST: \n"; │ │ │ │ │ + for (var i = postfix.length - 1; i >= 0; i--) { │ │ │ │ │ + msg += postfix[i].type + ": " + postfix[i].text + "\n"; │ │ │ │ │ + } │ │ │ │ │ + throw new Error(msg); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMSCapabilities.v1 │ │ │ │ │ - * Create an instance of one of the subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + return result; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ + return OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ /** │ │ │ │ │ * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * Generate a filter from a CQL string. │ │ │ │ │ + │ │ │ │ │ + * Parameters: │ │ │ │ │ + * text - {String} The CQL text. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array} List of named layers. │ │ │ │ │ + * {<OpenLayers.Filter>} A filter based on the CQL text. │ │ │ │ │ */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - var raw = data; │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ - } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - if (capabilities.service === undefined) { │ │ │ │ │ - // an exception must have occurred, so parse it │ │ │ │ │ - var parser = new OpenLayers.Format.OGCExceptionReport(); │ │ │ │ │ - capabilities.error = parser.read(raw); │ │ │ │ │ + read: function(text) { │ │ │ │ │ + var result = buildAst(tokenize(text)); │ │ │ │ │ + if (this.keepData) { │ │ │ │ │ + this.data = result; │ │ │ │ │ } │ │ │ │ │ - return capabilities; │ │ │ │ │ + return result; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Convert a filter into a CQL string. │ │ │ │ │ + │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {<OpenLayers.Filter>} The filter. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A CQL string based on the filter. │ │ │ │ │ */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wms": { │ │ │ │ │ - "Service": function(node, obj) { │ │ │ │ │ - obj.service = {}; │ │ │ │ │ - this.readChildNodes(node, obj.service); │ │ │ │ │ - }, │ │ │ │ │ - "Name": function(node, obj) { │ │ │ │ │ - obj.name = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Title": function(node, obj) { │ │ │ │ │ - obj.title = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Abstract": function(node, obj) { │ │ │ │ │ - obj["abstract"] = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "BoundingBox": function(node, obj) { │ │ │ │ │ - var bbox = {}; │ │ │ │ │ - bbox.bbox = [ │ │ │ │ │ - parseFloat(node.getAttribute("minx")), │ │ │ │ │ - parseFloat(node.getAttribute("miny")), │ │ │ │ │ - parseFloat(node.getAttribute("maxx")), │ │ │ │ │ - parseFloat(node.getAttribute("maxy")) │ │ │ │ │ - ]; │ │ │ │ │ - var res = { │ │ │ │ │ - x: parseFloat(node.getAttribute("resx")), │ │ │ │ │ - y: parseFloat(node.getAttribute("resy")) │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - if (!(isNaN(res.x) && isNaN(res.y))) { │ │ │ │ │ - bbox.res = res; │ │ │ │ │ - } │ │ │ │ │ - // return the bbox so that descendant classes can set the │ │ │ │ │ - // CRS and SRS and add it to the obj │ │ │ │ │ - return bbox; │ │ │ │ │ - }, │ │ │ │ │ - "OnlineResource": function(node, obj) { │ │ │ │ │ - obj.href = this.getAttributeNS(node, this.namespaces.xlink, │ │ │ │ │ - "href"); │ │ │ │ │ - }, │ │ │ │ │ - "ContactInformation": function(node, obj) { │ │ │ │ │ - obj.contactInformation = {}; │ │ │ │ │ - this.readChildNodes(node, obj.contactInformation); │ │ │ │ │ - }, │ │ │ │ │ - "ContactPersonPrimary": function(node, obj) { │ │ │ │ │ - obj.personPrimary = {}; │ │ │ │ │ - this.readChildNodes(node, obj.personPrimary); │ │ │ │ │ - }, │ │ │ │ │ - "ContactPerson": function(node, obj) { │ │ │ │ │ - obj.person = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "ContactOrganization": function(node, obj) { │ │ │ │ │ - obj.organization = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "ContactPosition": function(node, obj) { │ │ │ │ │ - obj.position = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "ContactAddress": function(node, obj) { │ │ │ │ │ - obj.contactAddress = {}; │ │ │ │ │ - this.readChildNodes(node, obj.contactAddress); │ │ │ │ │ - }, │ │ │ │ │ - "AddressType": function(node, obj) { │ │ │ │ │ - obj.type = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Address": function(node, obj) { │ │ │ │ │ - obj.address = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "City": function(node, obj) { │ │ │ │ │ - obj.city = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "StateOrProvince": function(node, obj) { │ │ │ │ │ - obj.stateOrProvince = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "PostCode": function(node, obj) { │ │ │ │ │ - obj.postcode = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Country": function(node, obj) { │ │ │ │ │ - obj.country = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "ContactVoiceTelephone": function(node, obj) { │ │ │ │ │ - obj.phone = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "ContactFacsimileTelephone": function(node, obj) { │ │ │ │ │ - obj.fax = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "ContactElectronicMailAddress": function(node, obj) { │ │ │ │ │ - obj.email = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Fees": function(node, obj) { │ │ │ │ │ - var fees = this.getChildValue(node); │ │ │ │ │ - if (fees && fees.toLowerCase() != "none") { │ │ │ │ │ - obj.fees = fees; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "AccessConstraints": function(node, obj) { │ │ │ │ │ - var constraints = this.getChildValue(node); │ │ │ │ │ - if (constraints && constraints.toLowerCase() != "none") { │ │ │ │ │ - obj.accessConstraints = constraints; │ │ │ │ │ + write: function(filter) { │ │ │ │ │ + if (filter instanceof OpenLayers.Geometry) { │ │ │ │ │ + return filter.toString(); │ │ │ │ │ + } │ │ │ │ │ + switch (filter.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Filter.Spatial": │ │ │ │ │ + switch (filter.type) { │ │ │ │ │ + case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ + return "BBOX(" + │ │ │ │ │ + filter.property + "," + │ │ │ │ │ + filter.value.toBBOX() + │ │ │ │ │ + ")"; │ │ │ │ │ + case OpenLayers.Filter.Spatial.DWITHIN: │ │ │ │ │ + return "DWITHIN(" + │ │ │ │ │ + filter.property + ", " + │ │ │ │ │ + this.write(filter.value) + ", " + │ │ │ │ │ + filter.distance + ")"; │ │ │ │ │ + case OpenLayers.Filter.Spatial.WITHIN: │ │ │ │ │ + return "WITHIN(" + │ │ │ │ │ + filter.property + ", " + │ │ │ │ │ + this.write(filter.value) + ")"; │ │ │ │ │ + case OpenLayers.Filter.Spatial.INTERSECTS: │ │ │ │ │ + return "INTERSECTS(" + │ │ │ │ │ + filter.property + ", " + │ │ │ │ │ + this.write(filter.value) + ")"; │ │ │ │ │ + case OpenLayers.Filter.Spatial.CONTAINS: │ │ │ │ │ + return "CONTAINS(" + │ │ │ │ │ + filter.property + ", " + │ │ │ │ │ + this.write(filter.value) + ")"; │ │ │ │ │ + default: │ │ │ │ │ + throw new Error("Unknown spatial filter type: " + filter.type); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "Capability": function(node, obj) { │ │ │ │ │ - obj.capability = { │ │ │ │ │ - nestedLayers: [], │ │ │ │ │ - layers: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.capability); │ │ │ │ │ - }, │ │ │ │ │ - "Request": function(node, obj) { │ │ │ │ │ - obj.request = {}; │ │ │ │ │ - this.readChildNodes(node, obj.request); │ │ │ │ │ - }, │ │ │ │ │ - "GetCapabilities": function(node, obj) { │ │ │ │ │ - obj.getcapabilities = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getcapabilities); │ │ │ │ │ - }, │ │ │ │ │ - "Format": function(node, obj) { │ │ │ │ │ - if (OpenLayers.Util.isArray(obj.formats)) { │ │ │ │ │ - obj.formats.push(this.getChildValue(node)); │ │ │ │ │ + case "OpenLayers.Filter.Logical": │ │ │ │ │ + if (filter.type == OpenLayers.Filter.Logical.NOT) { │ │ │ │ │ + // TODO: deal with precedence of logical operators to │ │ │ │ │ + // avoid extra parentheses (not urgent) │ │ │ │ │ + return "NOT (" + this.write(filter.filters[0]) + ")"; │ │ │ │ │ } else { │ │ │ │ │ - obj.format = this.getChildValue(node); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "DCPType": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "HTTP": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "Get": function(node, obj) { │ │ │ │ │ - obj.get = {}; │ │ │ │ │ - this.readChildNodes(node, obj.get); │ │ │ │ │ - // backwards compatibility │ │ │ │ │ - if (!obj.href) { │ │ │ │ │ - obj.href = obj.get.href; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Post": function(node, obj) { │ │ │ │ │ - obj.post = {}; │ │ │ │ │ - this.readChildNodes(node, obj.post); │ │ │ │ │ - // backwards compatibility │ │ │ │ │ - if (!obj.href) { │ │ │ │ │ - obj.href = obj.get.href; │ │ │ │ │ + var res = "("; │ │ │ │ │ + var first = true; │ │ │ │ │ + for (var i = 0; i < filter.filters.length; i++) { │ │ │ │ │ + if (first) { │ │ │ │ │ + first = false; │ │ │ │ │ + } else { │ │ │ │ │ + res += ") " + logicalReverse[filter.type] + " ("; │ │ │ │ │ + } │ │ │ │ │ + res += this.write(filter.filters[i]); │ │ │ │ │ + } │ │ │ │ │ + return res + ")"; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "GetMap": function(node, obj) { │ │ │ │ │ - obj.getmap = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getmap); │ │ │ │ │ - }, │ │ │ │ │ - "GetFeatureInfo": function(node, obj) { │ │ │ │ │ - obj.getfeatureinfo = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getfeatureinfo); │ │ │ │ │ - }, │ │ │ │ │ - "Exception": function(node, obj) { │ │ │ │ │ - obj.exception = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.exception); │ │ │ │ │ - }, │ │ │ │ │ - "Layer": function(node, obj) { │ │ │ │ │ - var parentLayer, capability; │ │ │ │ │ - if (obj.capability) { │ │ │ │ │ - capability = obj.capability; │ │ │ │ │ - parentLayer = obj; │ │ │ │ │ + case "OpenLayers.Filter.Comparison": │ │ │ │ │ + if (filter.type == OpenLayers.Filter.Comparison.BETWEEN) { │ │ │ │ │ + return filter.property + " BETWEEN " + │ │ │ │ │ + this.write(filter.lowerBoundary) + " AND " + │ │ │ │ │ + this.write(filter.upperBoundary); │ │ │ │ │ } else { │ │ │ │ │ - capability = obj; │ │ │ │ │ + return (filter.value !== null) ? filter.property + │ │ │ │ │ + " " + operatorReverse[filter.type] + " " + │ │ │ │ │ + this.write(filter.value) : filter.property + │ │ │ │ │ + " " + operatorReverse[filter.type]; │ │ │ │ │ } │ │ │ │ │ - var attrNode = node.getAttributeNode("queryable"); │ │ │ │ │ - var queryable = (attrNode && attrNode.specified) ? │ │ │ │ │ - node.getAttribute("queryable") : null; │ │ │ │ │ - attrNode = node.getAttributeNode("cascaded"); │ │ │ │ │ - var cascaded = (attrNode && attrNode.specified) ? │ │ │ │ │ - node.getAttribute("cascaded") : null; │ │ │ │ │ - attrNode = node.getAttributeNode("opaque"); │ │ │ │ │ - var opaque = (attrNode && attrNode.specified) ? │ │ │ │ │ - node.getAttribute('opaque') : null; │ │ │ │ │ - var noSubsets = node.getAttribute('noSubsets'); │ │ │ │ │ - var fixedWidth = node.getAttribute('fixedWidth'); │ │ │ │ │ - var fixedHeight = node.getAttribute('fixedHeight'); │ │ │ │ │ - var parent = parentLayer || {}, │ │ │ │ │ - extend = OpenLayers.Util.extend; │ │ │ │ │ - var layer = { │ │ │ │ │ - nestedLayers: [], │ │ │ │ │ - styles: parentLayer ? [].concat(parentLayer.styles) : [], │ │ │ │ │ - srs: parentLayer ? extend({}, parent.srs) : {}, │ │ │ │ │ - metadataURLs: [], │ │ │ │ │ - bbox: parentLayer ? extend({}, parent.bbox) : {}, │ │ │ │ │ - llbbox: parent.llbbox, │ │ │ │ │ - dimensions: parentLayer ? extend({}, parent.dimensions) : {}, │ │ │ │ │ - authorityURLs: parentLayer ? extend({}, parent.authorityURLs) : {}, │ │ │ │ │ - identifiers: {}, │ │ │ │ │ - keywords: [], │ │ │ │ │ - queryable: (queryable && queryable !== "") ? │ │ │ │ │ - (queryable === "1" || queryable === "true") : (parent.queryable || false), │ │ │ │ │ - cascaded: (cascaded !== null) ? parseInt(cascaded) : (parent.cascaded || 0), │ │ │ │ │ - opaque: opaque ? │ │ │ │ │ - (opaque === "1" || opaque === "true") : (parent.opaque || false), │ │ │ │ │ - noSubsets: (noSubsets !== null) ? │ │ │ │ │ - (noSubsets === "1" || noSubsets === "true") : (parent.noSubsets || false), │ │ │ │ │ - fixedWidth: (fixedWidth != null) ? │ │ │ │ │ - parseInt(fixedWidth) : (parent.fixedWidth || 0), │ │ │ │ │ - fixedHeight: (fixedHeight != null) ? │ │ │ │ │ - parseInt(fixedHeight) : (parent.fixedHeight || 0), │ │ │ │ │ - minScale: parent.minScale, │ │ │ │ │ - maxScale: parent.maxScale, │ │ │ │ │ - attribution: parent.attribution │ │ │ │ │ - }; │ │ │ │ │ - obj.nestedLayers.push(layer); │ │ │ │ │ - layer.capability = capability; │ │ │ │ │ - this.readChildNodes(node, layer); │ │ │ │ │ - delete layer.capability; │ │ │ │ │ - if (layer.name) { │ │ │ │ │ - var parts = layer.name.split(":"), │ │ │ │ │ - request = capability.request, │ │ │ │ │ - gfi = request.getfeatureinfo; │ │ │ │ │ - if (parts.length > 0) { │ │ │ │ │ - layer.prefix = parts[0]; │ │ │ │ │ - } │ │ │ │ │ - capability.layers.push(layer); │ │ │ │ │ - if (layer.formats === undefined) { │ │ │ │ │ - layer.formats = request.getmap.formats; │ │ │ │ │ - } │ │ │ │ │ - if (layer.infoFormats === undefined && gfi) { │ │ │ │ │ - layer.infoFormats = gfi.formats; │ │ │ │ │ - } │ │ │ │ │ + case undefined: │ │ │ │ │ + if (typeof filter === "string") { │ │ │ │ │ + return "'" + filter.replace(/'/g, "''") + "'"; │ │ │ │ │ + } else if (typeof filter === "number") { │ │ │ │ │ + return String(filter); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "Attribution": function(node, obj) { │ │ │ │ │ - obj.attribution = {}; │ │ │ │ │ - this.readChildNodes(node, obj.attribution); │ │ │ │ │ - }, │ │ │ │ │ - "LogoURL": function(node, obj) { │ │ │ │ │ - obj.logo = { │ │ │ │ │ - width: node.getAttribute("width"), │ │ │ │ │ - height: node.getAttribute("height") │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.logo); │ │ │ │ │ - }, │ │ │ │ │ - "Style": function(node, obj) { │ │ │ │ │ - var style = {}; │ │ │ │ │ - obj.styles.push(style); │ │ │ │ │ - this.readChildNodes(node, style); │ │ │ │ │ - }, │ │ │ │ │ - "LegendURL": function(node, obj) { │ │ │ │ │ - var legend = { │ │ │ │ │ - width: node.getAttribute("width"), │ │ │ │ │ - height: node.getAttribute("height") │ │ │ │ │ - }; │ │ │ │ │ - obj.legend = legend; │ │ │ │ │ - this.readChildNodes(node, legend); │ │ │ │ │ - }, │ │ │ │ │ - "MetadataURL": function(node, obj) { │ │ │ │ │ - var metadataURL = { │ │ │ │ │ - type: node.getAttribute("type") │ │ │ │ │ - }; │ │ │ │ │ - obj.metadataURLs.push(metadataURL); │ │ │ │ │ - this.readChildNodes(node, metadataURL); │ │ │ │ │ - }, │ │ │ │ │ - "DataURL": function(node, obj) { │ │ │ │ │ - obj.dataURL = {}; │ │ │ │ │ - this.readChildNodes(node, obj.dataURL); │ │ │ │ │ - }, │ │ │ │ │ - "FeatureListURL": function(node, obj) { │ │ │ │ │ - obj.featureListURL = {}; │ │ │ │ │ - this.readChildNodes(node, obj.featureListURL); │ │ │ │ │ - }, │ │ │ │ │ - "AuthorityURL": function(node, obj) { │ │ │ │ │ - var name = node.getAttribute("name"); │ │ │ │ │ - var authority = {}; │ │ │ │ │ - this.readChildNodes(node, authority); │ │ │ │ │ - obj.authorityURLs[name] = authority.href; │ │ │ │ │ - }, │ │ │ │ │ - "Identifier": function(node, obj) { │ │ │ │ │ - var authority = node.getAttribute("authority"); │ │ │ │ │ - obj.identifiers[authority] = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "KeywordList": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "SRS": function(node, obj) { │ │ │ │ │ - obj.srs[this.getChildValue(node)] = true; │ │ │ │ │ - } │ │ │ │ │ + default: │ │ │ │ │ + throw new Error("Can't encode: " + filter.CLASS_NAME + " " + filter); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.CQL" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ +})(); │ │ │ │ │ + │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSCapabilities/v1_1.js │ │ │ │ │ + OpenLayers/Format/WMSDescribeLayer.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/WMSCapabilities/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WMSCapabilities.v1_1 │ │ │ │ │ - * Abstract class not to be instantiated directly. │ │ │ │ │ + * Class: OpenLayers.Format.WMSDescribeLayer │ │ │ │ │ + * Read SLD WMS DescribeLayer response │ │ │ │ │ + * DescribeLayer is meant to couple WMS to WFS and WCS │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMSCapabilities.v1> │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_1 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WMSCapabilities.v1, { │ │ │ │ │ +OpenLayers.Format.WMSDescribeLayer = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "WMT_MS_Capabilities": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "Keyword": function(node, obj) { │ │ │ │ │ - if (obj.keywords) { │ │ │ │ │ - obj.keywords.push(this.getChildValue(node)); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "DescribeLayer": function(node, obj) { │ │ │ │ │ - obj.describelayer = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.describelayer); │ │ │ │ │ - }, │ │ │ │ │ - "GetLegendGraphic": function(node, obj) { │ │ │ │ │ - obj.getlegendgraphic = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getlegendgraphic); │ │ │ │ │ - }, │ │ │ │ │ - "GetStyles": function(node, obj) { │ │ │ │ │ - obj.getstyles = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getstyles); │ │ │ │ │ - }, │ │ │ │ │ - "PutStyles": function(node, obj) { │ │ │ │ │ - obj.putstyles = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.putstyles); │ │ │ │ │ - }, │ │ │ │ │ - "UserDefinedSymbolization": function(node, obj) { │ │ │ │ │ - var userSymbols = { │ │ │ │ │ - supportSLD: parseInt(node.getAttribute("SupportSLD")) == 1, │ │ │ │ │ - userLayer: parseInt(node.getAttribute("UserLayer")) == 1, │ │ │ │ │ - userStyle: parseInt(node.getAttribute("UserStyle")) == 1, │ │ │ │ │ - remoteWFS: parseInt(node.getAttribute("RemoteWFS")) == 1 │ │ │ │ │ - }; │ │ │ │ │ - obj.userSymbols = userSymbols; │ │ │ │ │ - }, │ │ │ │ │ - "LatLonBoundingBox": function(node, obj) { │ │ │ │ │ - obj.llbbox = [ │ │ │ │ │ - parseFloat(node.getAttribute("minx")), │ │ │ │ │ - parseFloat(node.getAttribute("miny")), │ │ │ │ │ - parseFloat(node.getAttribute("maxx")), │ │ │ │ │ - parseFloat(node.getAttribute("maxy")) │ │ │ │ │ - ]; │ │ │ │ │ - }, │ │ │ │ │ - "BoundingBox": function(node, obj) { │ │ │ │ │ - var bbox = OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"].BoundingBox.apply(this, [node, obj]); │ │ │ │ │ - bbox.srs = node.getAttribute("SRS"); │ │ │ │ │ - obj.bbox[bbox.srs] = bbox; │ │ │ │ │ - }, │ │ │ │ │ - "ScaleHint": function(node, obj) { │ │ │ │ │ - var min = node.getAttribute("min"); │ │ │ │ │ - var max = node.getAttribute("max"); │ │ │ │ │ - var rad2 = Math.pow(2, 0.5); │ │ │ │ │ - var ipm = OpenLayers.INCHES_PER_UNIT["m"]; │ │ │ │ │ - if (min != 0) { │ │ │ │ │ - obj.maxScale = parseFloat( │ │ │ │ │ - ((min / rad2) * ipm * │ │ │ │ │ - OpenLayers.DOTS_PER_INCH).toPrecision(13) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (max != Number.POSITIVE_INFINITY) { │ │ │ │ │ - obj.minScale = parseFloat( │ │ │ │ │ - ((max / rad2) * ipm * │ │ │ │ │ - OpenLayers.DOTS_PER_INCH).toPrecision(13) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Dimension": function(node, obj) { │ │ │ │ │ - var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ - var dim = { │ │ │ │ │ - name: name, │ │ │ │ │ - units: node.getAttribute("units"), │ │ │ │ │ - unitsymbol: node.getAttribute("unitSymbol") │ │ │ │ │ - }; │ │ │ │ │ - obj.dimensions[dim.name] = dim; │ │ │ │ │ - }, │ │ │ │ │ - "Extent": function(node, obj) { │ │ │ │ │ - var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ - if (name in obj["dimensions"]) { │ │ │ │ │ - var extent = obj.dimensions[name]; │ │ │ │ │ - extent.nearestVal = │ │ │ │ │ - node.getAttribute("nearestValue") === "1"; │ │ │ │ │ - extent.multipleVal = │ │ │ │ │ - node.getAttribute("multipleValues") === "1"; │ │ │ │ │ - extent.current = node.getAttribute("current") === "1"; │ │ │ │ │ - extent["default"] = node.getAttribute("default") || ""; │ │ │ │ │ - var values = this.getChildValue(node); │ │ │ │ │ - extent.values = values.split(","); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"]) │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.1.1". │ │ │ │ │ + */ │ │ │ │ │ + defaultVersion: "1.1.1", │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1" │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMSDescribeLayer │ │ │ │ │ + * Create a new parser for WMS DescribeLayer responses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read DescribeLayer data from a string, and return the response. │ │ │ │ │ + * The OGC currently defines 2 formats which are allowed for output, │ │ │ │ │ + * so we need to parse these 2 types │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} Array of {<LayerDescription>} objects which have: │ │ │ │ │ + * - {String} owsType: WFS/WCS │ │ │ │ │ + * - {String} owsURL: the online resource │ │ │ │ │ + * - {String} typeName: the name of the typename on the service │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSDescribeLayer" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSCapabilities/v1_1_1.js │ │ │ │ │ + OpenLayers/Format/KML.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Format/WMSCapabilities/v1_1.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Date.js │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Collection.js │ │ │ │ │ + * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ + * @requires OpenLayers/Projection.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Format.WMSCapabilities/v1_1_1 │ │ │ │ │ - * Read WMS Capabilities version 1.1.1. │ │ │ │ │ - * │ │ │ │ │ - * Note on <ScaleHint> parsing: If the 'min' attribute is set to "0", no │ │ │ │ │ - * maxScale will be set on the layer object. If the 'max' attribute is set to │ │ │ │ │ - * "Infinity", no minScale will be set. This makes it easy to create proper │ │ │ │ │ - * {<OpenLayers.Layer.WMS>} configurations directly from the layer object │ │ │ │ │ - * literals returned by this format, because no minScale/maxScale modifications │ │ │ │ │ - * need to be made. │ │ │ │ │ + * Class: OpenLayers.Format.KML │ │ │ │ │ + * Read/Write KML. Create a new instance with the <OpenLayers.Format.KML> │ │ │ │ │ + * constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMSCapabilities.v1_1> │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_1_1 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WMSCapabilities.v1_1, { │ │ │ │ │ +OpenLayers.Format.KML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} The specific parser version. │ │ │ │ │ - */ │ │ │ │ │ - version: "1.1.1", │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + kml: "http://www.opengis.net/kml/2.2", │ │ │ │ │ + gx: "http://www.google.com/kml/ext/2.2" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMSCapabilities.v1_1_1 │ │ │ │ │ - * Create a new parser for WMS capabilities version 1.1.1. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: kmlns │ │ │ │ │ + * {String} KML Namespace to use. Defaults to 2.0 namespace. │ │ │ │ │ + */ │ │ │ │ │ + kmlns: "http://earth.google.com/kml/2.0", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "SRS": function(node, obj) { │ │ │ │ │ - obj.srs[this.getChildValue(node)] = true; │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1_1.prototype.readers["wms"]) │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: placemarksDesc │ │ │ │ │ + * {String} Name of the placemarks. Default is "No description available". │ │ │ │ │ + */ │ │ │ │ │ + placemarksDesc: "No description available", │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_1" │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: foldersName │ │ │ │ │ + * {String} Name of the folders. Default is "OpenLayers export". │ │ │ │ │ + * If set to null, no name element will be created. │ │ │ │ │ + */ │ │ │ │ │ + foldersName: "OpenLayers export", │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSCapabilities/v1_3.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: foldersDesc │ │ │ │ │ + * {String} Description of the folders. Default is "Exported on [date]." │ │ │ │ │ + * If set to null, no description element will be created. │ │ │ │ │ + */ │ │ │ │ │ + foldersDesc: "Exported on " + new Date(), │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: extractAttributes │ │ │ │ │ + * {Boolean} Extract attributes from KML. Default is true. │ │ │ │ │ + * Extracting styleUrls requires this to be set to true │ │ │ │ │ + * Note that currently only Data and SimpleData │ │ │ │ │ + * elements are handled. │ │ │ │ │ + */ │ │ │ │ │ + extractAttributes: true, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WMSCapabilities/v1.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: kvpAttributes │ │ │ │ │ + * {Boolean} Only used if extractAttributes is true. │ │ │ │ │ + * If set to true, attributes will be simple │ │ │ │ │ + * key-value pairs, compatible with other formats, │ │ │ │ │ + * Any displayName elements will be ignored. │ │ │ │ │ + * If set to false, attributes will be objects, │ │ │ │ │ + * retaining any displayName elements, but not │ │ │ │ │ + * compatible with other formats. Any CDATA in │ │ │ │ │ + * displayName will be read in as a string value. │ │ │ │ │ + * Default is false. │ │ │ │ │ + */ │ │ │ │ │ + kvpAttributes: false, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WMSCapabilities/v1_3 │ │ │ │ │ - * Abstract base class for WMS Capabilities version 1.3.X. │ │ │ │ │ - * SLD 1.1.0 adds in the extra operations DescribeLayer and GetLegendGraphic, │ │ │ │ │ - * see: http://schemas.opengis.net/sld/1.1.0/sld_capabilities.xsd │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMSCapabilities.v1> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_3 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WMSCapabilities.v1, { │ │ │ │ │ + /** │ │ │ │ │ + * Property: extractStyles │ │ │ │ │ + * {Boolean} Extract styles from KML. Default is false. │ │ │ │ │ + * Extracting styleUrls also requires extractAttributes to be │ │ │ │ │ + * set to true │ │ │ │ │ + */ │ │ │ │ │ + extractStyles: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "WMS_Capabilities": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "LayerLimit": function(node, obj) { │ │ │ │ │ - obj.layerLimit = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "MaxWidth": function(node, obj) { │ │ │ │ │ - obj.maxWidth = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "MaxHeight": function(node, obj) { │ │ │ │ │ - obj.maxHeight = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "BoundingBox": function(node, obj) { │ │ │ │ │ - var bbox = OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"].BoundingBox.apply(this, [node, obj]); │ │ │ │ │ - bbox.srs = node.getAttribute("CRS"); │ │ │ │ │ - obj.bbox[bbox.srs] = bbox; │ │ │ │ │ - }, │ │ │ │ │ - "CRS": function(node, obj) { │ │ │ │ │ - // CRS is the synonym of SRS │ │ │ │ │ - this.readers.wms.SRS.apply(this, [node, obj]); │ │ │ │ │ - }, │ │ │ │ │ - "EX_GeographicBoundingBox": function(node, obj) { │ │ │ │ │ - // replacement of LatLonBoundingBox │ │ │ │ │ - obj.llbbox = []; │ │ │ │ │ - this.readChildNodes(node, obj.llbbox); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: extractTracks │ │ │ │ │ + * {Boolean} Extract gx:Track elements from Placemark elements. Default │ │ │ │ │ + * is false. If true, features will be generated for all points in │ │ │ │ │ + * all gx:Track elements. Features will have a when (Date) attribute │ │ │ │ │ + * based on when elements in the track. If tracks include angle │ │ │ │ │ + * elements, features will have heading, tilt, and roll attributes. │ │ │ │ │ + * If track point coordinates have three values, features will have │ │ │ │ │ + * an altitude attribute with the third coordinate value. │ │ │ │ │ + */ │ │ │ │ │ + extractTracks: false, │ │ │ │ │ │ │ │ │ │ - }, │ │ │ │ │ - "westBoundLongitude": function(node, obj) { │ │ │ │ │ - obj[0] = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "eastBoundLongitude": function(node, obj) { │ │ │ │ │ - obj[2] = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "southBoundLatitude": function(node, obj) { │ │ │ │ │ - obj[1] = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "northBoundLatitude": function(node, obj) { │ │ │ │ │ - obj[3] = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "MinScaleDenominator": function(node, obj) { │ │ │ │ │ - obj.maxScale = parseFloat(this.getChildValue(node)).toPrecision(16); │ │ │ │ │ - }, │ │ │ │ │ - "MaxScaleDenominator": function(node, obj) { │ │ │ │ │ - obj.minScale = parseFloat(this.getChildValue(node)).toPrecision(16); │ │ │ │ │ - }, │ │ │ │ │ - "Dimension": function(node, obj) { │ │ │ │ │ - // dimension has extra attributes: default, multipleValues, │ │ │ │ │ - // nearestValue, current which used to be part of Extent. It now │ │ │ │ │ - // also contains the values. │ │ │ │ │ - var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ - var dim = { │ │ │ │ │ - name: name, │ │ │ │ │ - units: node.getAttribute("units"), │ │ │ │ │ - unitsymbol: node.getAttribute("unitSymbol"), │ │ │ │ │ - nearestVal: node.getAttribute("nearestValue") === "1", │ │ │ │ │ - multipleVal: node.getAttribute("multipleValues") === "1", │ │ │ │ │ - "default": node.getAttribute("default") || "", │ │ │ │ │ - current: node.getAttribute("current") === "1", │ │ │ │ │ - values: this.getChildValue(node).split(",") │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: trackAttributes │ │ │ │ │ + * {Array} If <extractTracks> is true, points within gx:Track elements will │ │ │ │ │ + * be parsed as features with when, heading, tilt, and roll attributes. │ │ │ │ │ + * Any additional attribute names can be provided in <trackAttributes>. │ │ │ │ │ + */ │ │ │ │ │ + trackAttributes: null, │ │ │ │ │ │ │ │ │ │ - }; │ │ │ │ │ - // Theoretically there can be more dimensions with the same │ │ │ │ │ - // name, but with a different unit. Until we meet such a case, │ │ │ │ │ - // let's just keep the same structure as the WMS 1.1 │ │ │ │ │ - // GetCapabilities parser uses. We will store the last │ │ │ │ │ - // one encountered. │ │ │ │ │ - obj.dimensions[dim.name] = dim; │ │ │ │ │ - }, │ │ │ │ │ - "Keyword": function(node, obj) { │ │ │ │ │ - // TODO: should we change the structure of keyword in v1.js? │ │ │ │ │ - // Make it an object with a value instead of a string? │ │ │ │ │ - var keyword = { │ │ │ │ │ - value: this.getChildValue(node), │ │ │ │ │ - vocabulary: node.getAttribute("vocabulary") │ │ │ │ │ - }; │ │ │ │ │ - if (obj.keywords) { │ │ │ │ │ - obj.keywords.push(keyword); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"]), │ │ │ │ │ - "sld": { │ │ │ │ │ - "UserDefinedSymbolization": function(node, obj) { │ │ │ │ │ - this.readers.wms.UserDefinedSymbolization.apply(this, [node, obj]); │ │ │ │ │ - // add the two extra attributes │ │ │ │ │ - obj.userSymbols.inlineFeature = parseInt(node.getAttribute("InlineFeature")) == 1; │ │ │ │ │ - obj.userSymbols.remoteWCS = parseInt(node.getAttribute("RemoteWCS")) == 1; │ │ │ │ │ - }, │ │ │ │ │ - "DescribeLayer": function(node, obj) { │ │ │ │ │ - this.readers.wms.DescribeLayer.apply(this, [node, obj]); │ │ │ │ │ - }, │ │ │ │ │ - "GetLegendGraphic": function(node, obj) { │ │ │ │ │ - this.readers.wms.GetLegendGraphic.apply(this, [node, obj]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: internalns │ │ │ │ │ + * {String} KML Namespace to use -- defaults to the namespace of the │ │ │ │ │ + * Placemark node being parsed, but falls back to kmlns. │ │ │ │ │ + */ │ │ │ │ │ + internalns: null, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_3" │ │ │ │ │ + /** │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Array} Array of features │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + features: null, │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSCapabilities/v1_3_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: styles │ │ │ │ │ + * {Object} Storage of style objects │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + styles: null, │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: styleBaseUrl │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + styleBaseUrl: "", │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WMSCapabilities/v1_3.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: fetched │ │ │ │ │ + * {Object} Storage of KML URLs that have been fetched before │ │ │ │ │ + * in order to prevent reloading them. │ │ │ │ │ + */ │ │ │ │ │ + fetched: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WMSCapabilities/v1_3_0 │ │ │ │ │ - * Read WMS Capabilities version 1.3.0. │ │ │ │ │ - * SLD 1.1.0 adds in the extra operations DescribeLayer and GetLegendGraphic, │ │ │ │ │ - * see: http://schemas.opengis.net/sld/1.1.0/sld_capabilities.xsd │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMSCapabilities.v1_3> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_3_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WMSCapabilities.v1_3, { │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxDepth │ │ │ │ │ + * {Integer} Maximum depth for recursive loading external KML URLs │ │ │ │ │ + * Defaults to 0: do no external fetching │ │ │ │ │ + */ │ │ │ │ │ + maxDepth: 0, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} The specific parser version. │ │ │ │ │ - */ │ │ │ │ │ - version: "1.3.0", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.KML │ │ │ │ │ + * Create a new parser for KML. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + // compile regular expressions once instead of every time they are used │ │ │ │ │ + this.regExes = { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g), │ │ │ │ │ + kmlColor: (/(\w{2})(\w{2})(\w{2})(\w{2})/), │ │ │ │ │ + kmlIconPalette: (/root:\/\/icons\/palette-(\d+)(\.\w+)/), │ │ │ │ │ + straightBracket: (/\$\[(.*?)\]/g) │ │ │ │ │ + }; │ │ │ │ │ + // KML coordinates are always in longlat WGS84 │ │ │ │ │ + this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_3_0" │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSCapabilities/v1_1_1_WMSC.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read data from a string, and return a list of features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} List of features. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + this.features = []; │ │ │ │ │ + this.styles = {}; │ │ │ │ │ + this.fetched = {}; │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + // Set default options │ │ │ │ │ + var options = { │ │ │ │ │ + depth: 0, │ │ │ │ │ + styleBaseUrl: this.styleBaseUrl │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WMSCapabilities/v1_1_1.js │ │ │ │ │ - */ │ │ │ │ │ + return this.parseData(data, options); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WMSCapabilities/v1_1_1_WMSC │ │ │ │ │ - * Read WMS-C Capabilities version 1.1.1. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMSCapabilities.v1_1_1> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_1_1_WMSC = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WMSCapabilities.v1_1_1, { │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseData │ │ │ │ │ + * Read data from a string, and return a list of features. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * options - {Object} Hash of options │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} List of features. │ │ │ │ │ + */ │ │ │ │ │ + parseData: function(data, options) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} The specific parser version. │ │ │ │ │ - */ │ │ │ │ │ - version: "1.1.1", │ │ │ │ │ + // Loop throught the following node types in this order and │ │ │ │ │ + // process the nodes found │ │ │ │ │ + var types = ["Link", "NetworkLink", "Style", "StyleMap", "Placemark"]; │ │ │ │ │ + for (var i = 0, len = types.length; i < len; ++i) { │ │ │ │ │ + var type = types[i]; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: profile │ │ │ │ │ - * {String} The specific profile │ │ │ │ │ - */ │ │ │ │ │ - profile: "WMSC", │ │ │ │ │ + var nodes = this.getElementsByTagNameNS(data, "*", type); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMSCapabilities.v1_1_1 │ │ │ │ │ - * Create a new parser for WMS-C capabilities version 1.1.1. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + // skip to next type if no nodes are found │ │ │ │ │ + if (nodes.length == 0) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "VendorSpecificCapabilities": function(node, obj) { │ │ │ │ │ - obj.vendorSpecific = { │ │ │ │ │ - tileSets: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.vendorSpecific); │ │ │ │ │ - }, │ │ │ │ │ - "TileSet": function(node, vendorSpecific) { │ │ │ │ │ - var tileset = { │ │ │ │ │ - srs: {}, │ │ │ │ │ - bbox: {}, │ │ │ │ │ - resolutions: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, tileset); │ │ │ │ │ - vendorSpecific.tileSets.push(tileset); │ │ │ │ │ - }, │ │ │ │ │ - "Resolutions": function(node, tileset) { │ │ │ │ │ - var res = this.getChildValue(node).split(" "); │ │ │ │ │ - for (var i = 0, len = res.length; i < len; i++) { │ │ │ │ │ - if (res[i] != "") { │ │ │ │ │ - tileset.resolutions.push(parseFloat(res[i])); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Width": function(node, tileset) { │ │ │ │ │ - tileset.width = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "Height": function(node, tileset) { │ │ │ │ │ - tileset.height = parseInt(this.getChildValue(node)); │ │ │ │ │ - }, │ │ │ │ │ - "Layers": function(node, tileset) { │ │ │ │ │ - tileset.layers = this.getChildValue(node); │ │ │ │ │ - }, │ │ │ │ │ - "Styles": function(node, tileset) { │ │ │ │ │ - tileset.styles = this.getChildValue(node); │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1_1_1.prototype.readers["wms"]) │ │ │ │ │ - }, │ │ │ │ │ + switch (type.toLowerCase()) { │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_1_WMSC" │ │ │ │ │ + // Fetch external links │ │ │ │ │ + case "link": │ │ │ │ │ + case "networklink": │ │ │ │ │ + this.parseLinks(nodes, options); │ │ │ │ │ + break; │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WMSCapabilities/v1_1_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + // parse style information │ │ │ │ │ + case "style": │ │ │ │ │ + if (this.extractStyles) { │ │ │ │ │ + this.parseStyles(nodes, options); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "stylemap": │ │ │ │ │ + if (this.extractStyles) { │ │ │ │ │ + this.parseStyleMaps(nodes, options); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + // parse features │ │ │ │ │ + case "placemark": │ │ │ │ │ + this.parseFeatures(nodes, options); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WMSCapabilities/v1_1.js │ │ │ │ │ - */ │ │ │ │ │ + return this.features; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WMSCapabilities/v1_1_0 │ │ │ │ │ - * Read WMS Capabilities version 1.1.0. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WMSCapabilities.v1_1> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_1_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WMSCapabilities.v1_1, { │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseLinks │ │ │ │ │ + * Finds URLs of linked KML documents and fetches them │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * nodes - {Array} of {DOMElement} data to read/parse. │ │ │ │ │ + * options - {Object} Hash of options │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + parseLinks: function(nodes, options) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} The specific parser version. │ │ │ │ │ - */ │ │ │ │ │ - version: "1.1.0", │ │ │ │ │ + // Fetch external links <NetworkLink> and <Link> │ │ │ │ │ + // Don't do anything if we have reached our maximum depth for recursion │ │ │ │ │ + if (options.depth >= this.maxDepth) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WMSCapabilities.v1_1_0 │ │ │ │ │ - * Create a new parser for WMS capabilities version 1.1.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + // increase depth │ │ │ │ │ + var newOptions = OpenLayers.Util.extend({}, options); │ │ │ │ │ + newOptions.depth++; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "SRS": function(node, obj) { │ │ │ │ │ - var srs = this.getChildValue(node); │ │ │ │ │ - var values = srs.split(/ +/); │ │ │ │ │ - for (var i = 0, len = values.length; i < len; i++) { │ │ │ │ │ - obj.srs[values[i]] = true; │ │ │ │ │ - } │ │ │ │ │ + for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ + var href = this.parseProperty(nodes[i], "*", "href"); │ │ │ │ │ + if (href && !this.fetched[href]) { │ │ │ │ │ + this.fetched[href] = true; // prevent reloading the same urls │ │ │ │ │ + var data = this.fetchLink(href); │ │ │ │ │ + if (data) { │ │ │ │ │ + this.parseData(data, newOptions); │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1_1.prototype.readers["wms"]) │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_0" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WFSCapabilities/v1.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WFSCapabilities.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: fetchLink │ │ │ │ │ + * Fetches a URL and returns the result │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * href - {String} url to be fetched │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + fetchLink: function(href) { │ │ │ │ │ + var request = OpenLayers.Request.GET({ │ │ │ │ │ + url: href, │ │ │ │ │ + async: false │ │ │ │ │ + }); │ │ │ │ │ + if (request) { │ │ │ │ │ + return request.responseText; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WFSCapabilities.v1 │ │ │ │ │ - * Abstract class not to be instantiated directly. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.XML> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WFSCapabilities.v1 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.XML, { │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseStyles │ │ │ │ │ + * Parses <Style> nodes │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * nodes - {Array} of {DOMElement} data to read/parse. │ │ │ │ │ + * options - {Object} Hash of options │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + parseStyles: function(nodes, options) { │ │ │ │ │ + for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ + var style = this.parseStyle(nodes[i]); │ │ │ │ │ + if (style) { │ │ │ │ │ + var styleName = (options.styleBaseUrl || "") + "#" + style.id; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: namespaces │ │ │ │ │ - * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ - */ │ │ │ │ │ - namespaces: { │ │ │ │ │ - wfs: "http://www.opengis.net/wfs", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - ows: "http://www.opengis.net/ows" │ │ │ │ │ - }, │ │ │ │ │ + this.styles[styleName] = style; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseKmlColor │ │ │ │ │ + * Parses a kml color (in 'aabbggrr' format) and returns the corresponding │ │ │ │ │ + * color and opacity or null if the color is invalid. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * kmlColor - {String} a kml formated color │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} │ │ │ │ │ + */ │ │ │ │ │ + parseKmlColor: function(kmlColor) { │ │ │ │ │ + var color = null; │ │ │ │ │ + if (kmlColor) { │ │ │ │ │ + var matches = kmlColor.match(this.regExes.kmlColor); │ │ │ │ │ + if (matches) { │ │ │ │ │ + color = { │ │ │ │ │ + color: '#' + matches[4] + matches[3] + matches[2], │ │ │ │ │ + opacity: parseInt(matches[1], 16) / 255 │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return color; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: errorProperty │ │ │ │ │ - * {String} Which property of the returned object to check for in order to │ │ │ │ │ - * determine whether or not parsing has failed. In the case that the │ │ │ │ │ - * errorProperty is undefined on the returned object, the document will be │ │ │ │ │ - * run through an OGCExceptionReport parser. │ │ │ │ │ - */ │ │ │ │ │ - errorProperty: "featureTypeList", │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseStyle │ │ │ │ │ + * Parses the children of a <Style> node and builds the style hash │ │ │ │ │ + * accordingly │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} <Style> node │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + parseStyle: function(node) { │ │ │ │ │ + var style = {}; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: defaultPrefix │ │ │ │ │ - */ │ │ │ │ │ - defaultPrefix: "wfs", │ │ │ │ │ + var types = ["LineStyle", "PolyStyle", "IconStyle", "BalloonStyle", │ │ │ │ │ + "LabelStyle" │ │ │ │ │ + ]; │ │ │ │ │ + var type, styleTypeNode, nodeList, geometry, parser; │ │ │ │ │ + for (var i = 0, len = types.length; i < len; ++i) { │ │ │ │ │ + type = types[i]; │ │ │ │ │ + styleTypeNode = this.getElementsByTagNameNS(node, "*", type)[0]; │ │ │ │ │ + if (!styleTypeNode) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WFSCapabilities.v1_1 │ │ │ │ │ - * Create an instance of one of the subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + // only deal with first geometry of this type │ │ │ │ │ + switch (type.toLowerCase()) { │ │ │ │ │ + case "linestyle": │ │ │ │ │ + var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ + var color = this.parseKmlColor(kmlColor); │ │ │ │ │ + if (color) { │ │ │ │ │ + style["strokeColor"] = color.color; │ │ │ │ │ + style["strokeOpacity"] = color.opacity; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} List of named layers. │ │ │ │ │ - */ │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - } │ │ │ │ │ - var raw = data; │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement; │ │ │ │ │ - } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - return capabilities; │ │ │ │ │ - }, │ │ │ │ │ + var width = this.parseProperty(styleTypeNode, "*", "width"); │ │ │ │ │ + if (width) { │ │ │ │ │ + style["strokeWidth"] = width; │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wfs": { │ │ │ │ │ - "WFS_Capabilities": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "FeatureTypeList": function(node, request) { │ │ │ │ │ - request.featureTypeList = { │ │ │ │ │ - featureTypes: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, request.featureTypeList); │ │ │ │ │ - }, │ │ │ │ │ - "FeatureType": function(node, featureTypeList) { │ │ │ │ │ - var featureType = {}; │ │ │ │ │ - this.readChildNodes(node, featureType); │ │ │ │ │ - featureTypeList.featureTypes.push(featureType); │ │ │ │ │ - }, │ │ │ │ │ - "Name": function(node, obj) { │ │ │ │ │ - var name = this.getChildValue(node); │ │ │ │ │ - if (name) { │ │ │ │ │ - var parts = name.split(":"); │ │ │ │ │ - obj.name = parts.pop(); │ │ │ │ │ - if (parts.length > 0) { │ │ │ │ │ - obj.featureNS = this.lookupNamespaceURI(node, parts[0]); │ │ │ │ │ - } │ │ │ │ │ + case "polystyle": │ │ │ │ │ + var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ + var color = this.parseKmlColor(kmlColor); │ │ │ │ │ + if (color) { │ │ │ │ │ + style["fillOpacity"] = color.opacity; │ │ │ │ │ + style["fillColor"] = color.color; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "Title": function(node, obj) { │ │ │ │ │ - var title = this.getChildValue(node); │ │ │ │ │ - if (title) { │ │ │ │ │ - obj.title = title; │ │ │ │ │ + // Check if fill is disabled │ │ │ │ │ + var fill = this.parseProperty(styleTypeNode, "*", "fill"); │ │ │ │ │ + if (fill == "0") { │ │ │ │ │ + style["fillColor"] = "none"; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - "Abstract": function(node, obj) { │ │ │ │ │ - var abst = this.getChildValue(node); │ │ │ │ │ - if (abst) { │ │ │ │ │ - obj["abstract"] = abst; │ │ │ │ │ + // Check if outline is disabled │ │ │ │ │ + var outline = this.parseProperty(styleTypeNode, "*", "outline"); │ │ │ │ │ + if (outline == "0") { │ │ │ │ │ + style["strokeWidth"] = "0"; │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1" │ │ │ │ │ + break; │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WFSCapabilities/v1_0_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + case "iconstyle": │ │ │ │ │ + // set scale │ │ │ │ │ + var scale = parseFloat(this.parseProperty(styleTypeNode, │ │ │ │ │ + "*", "scale") || 1); │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + // set default width and height of icon │ │ │ │ │ + var width = 32 * scale; │ │ │ │ │ + var height = 32 * scale; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WFSCapabilities/v1.js │ │ │ │ │ - */ │ │ │ │ │ + var iconNode = this.getElementsByTagNameNS(styleTypeNode, │ │ │ │ │ + "*", │ │ │ │ │ + "Icon")[0]; │ │ │ │ │ + if (iconNode) { │ │ │ │ │ + var href = this.parseProperty(iconNode, "*", "href"); │ │ │ │ │ + if (href) { │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WFSCapabilities/v1_0_0 │ │ │ │ │ - * Read WFS Capabilities version 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WFSCapabilities.v1> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WFSCapabilities.v1_0_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WFSCapabilities.v1, { │ │ │ │ │ + var w = this.parseProperty(iconNode, "*", "w"); │ │ │ │ │ + var h = this.parseProperty(iconNode, "*", "h"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WFSCapabilities.v1_0_0 │ │ │ │ │ - * Create a new parser for WFS capabilities version 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + // Settings for Google specific icons that are 64x64 │ │ │ │ │ + // We set the width and height to 64 and halve the │ │ │ │ │ + // scale to prevent icons from being too big │ │ │ │ │ + var google = "http://maps.google.com/mapfiles/kml"; │ │ │ │ │ + if (OpenLayers.String.startsWith( │ │ │ │ │ + href, google) && !w && !h) { │ │ │ │ │ + w = 64; │ │ │ │ │ + h = 64; │ │ │ │ │ + scale = scale / 2; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "Service": function(node, capabilities) { │ │ │ │ │ - capabilities.service = {}; │ │ │ │ │ - this.readChildNodes(node, capabilities.service); │ │ │ │ │ - }, │ │ │ │ │ - "Fees": function(node, service) { │ │ │ │ │ - var fees = this.getChildValue(node); │ │ │ │ │ - if (fees && fees.toLowerCase() != "none") { │ │ │ │ │ - service.fees = fees; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "AccessConstraints": function(node, service) { │ │ │ │ │ - var constraints = this.getChildValue(node); │ │ │ │ │ - if (constraints && constraints.toLowerCase() != "none") { │ │ │ │ │ - service.accessConstraints = constraints; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "OnlineResource": function(node, service) { │ │ │ │ │ - var onlineResource = this.getChildValue(node); │ │ │ │ │ - if (onlineResource && onlineResource.toLowerCase() != "none") { │ │ │ │ │ - service.onlineResource = onlineResource; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Keywords": function(node, service) { │ │ │ │ │ - var keywords = this.getChildValue(node); │ │ │ │ │ - if (keywords && keywords.toLowerCase() != "none") { │ │ │ │ │ - service.keywords = keywords.split(', '); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "Capability": function(node, capabilities) { │ │ │ │ │ - capabilities.capability = {}; │ │ │ │ │ - this.readChildNodes(node, capabilities.capability); │ │ │ │ │ - }, │ │ │ │ │ - "Request": function(node, obj) { │ │ │ │ │ - obj.request = {}; │ │ │ │ │ - this.readChildNodes(node, obj.request); │ │ │ │ │ - }, │ │ │ │ │ - "GetFeature": function(node, request) { │ │ │ │ │ - request.getfeature = { │ │ │ │ │ - href: {}, // DCPType │ │ │ │ │ - formats: [] // ResultFormat │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, request.getfeature); │ │ │ │ │ - }, │ │ │ │ │ - "ResultFormat": function(node, obj) { │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - var childNode; │ │ │ │ │ - for (var i = 0; i < children.length; i++) { │ │ │ │ │ - childNode = children[i]; │ │ │ │ │ - if (childNode.nodeType == 1) { │ │ │ │ │ - obj.formats.push(childNode.nodeName); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - "DCPType": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - }, │ │ │ │ │ - "HTTP": function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj.href); │ │ │ │ │ - }, │ │ │ │ │ - "Get": function(node, obj) { │ │ │ │ │ - obj.get = node.getAttribute("onlineResource"); │ │ │ │ │ - }, │ │ │ │ │ - "Post": function(node, obj) { │ │ │ │ │ - obj.post = node.getAttribute("onlineResource"); │ │ │ │ │ - }, │ │ │ │ │ - "SRS": function(node, obj) { │ │ │ │ │ - var srs = this.getChildValue(node); │ │ │ │ │ - if (srs) { │ │ │ │ │ - obj.srs = srs; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WFSCapabilities.v1.prototype.readers["wfs"]) │ │ │ │ │ - }, │ │ │ │ │ + // if only dimension is defined, make sure the │ │ │ │ │ + // other one has the same value │ │ │ │ │ + w = w || h; │ │ │ │ │ + h = h || w; │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1_0_0" │ │ │ │ │ + if (w) { │ │ │ │ │ + width = parseInt(w) * scale; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Format/WFSCapabilities/v1_1_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + if (h) { │ │ │ │ │ + height = parseInt(h) * scale; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + // support for internal icons │ │ │ │ │ + // (/root://icons/palette-x.png) │ │ │ │ │ + // x and y tell the position on the palette: │ │ │ │ │ + // - in pixels │ │ │ │ │ + // - starting from the left bottom │ │ │ │ │ + // We translate that to a position in the list │ │ │ │ │ + // and request the appropriate icon from the │ │ │ │ │ + // google maps website │ │ │ │ │ + var matches = href.match(this.regExes.kmlIconPalette); │ │ │ │ │ + if (matches) { │ │ │ │ │ + var palette = matches[1]; │ │ │ │ │ + var file_extension = matches[2]; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Format/WFSCapabilities/v1.js │ │ │ │ │ - * @requires OpenLayers/Format/OWSCommon/v1.js │ │ │ │ │ - */ │ │ │ │ │ + var x = this.parseProperty(iconNode, "*", "x"); │ │ │ │ │ + var y = this.parseProperty(iconNode, "*", "y"); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Format.WFSCapabilities/v1_1_0 │ │ │ │ │ - * Read WFS Capabilities version 1.1.0. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Format.WFSCapabilities> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Format.WFSCapabilities.v1_1_0 = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Format.WFSCapabilities.v1, { │ │ │ │ │ + var posX = x ? x / 32 : 0; │ │ │ │ │ + var posY = y ? (7 - y / 32) : 7; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: regExes │ │ │ │ │ - * Compiled regular expressions for manipulating strings. │ │ │ │ │ - */ │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ - removeSpace: (/\s*/g), │ │ │ │ │ - splitSpace: (/\s+/), │ │ │ │ │ - trimComma: (/\s*,\s*/g) │ │ │ │ │ - }, │ │ │ │ │ + var pos = posY * 8 + posX; │ │ │ │ │ + href = "http://maps.google.com/mapfiles/kml/pal" + │ │ │ │ │ + palette + "/icon" + pos + file_extension; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Format.WFSCapabilities.v1_1_0 │ │ │ │ │ - * Create a new parser for WFS capabilities version 1.1.0. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * this instance. │ │ │ │ │ - */ │ │ │ │ │ + style["graphicOpacity"] = 1; // fully opaque │ │ │ │ │ + style["externalGraphic"] = href; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readers │ │ │ │ │ - * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ - * be applied when a namespaced node is found matching the function │ │ │ │ │ - * name. The function will be applied in the scope of this parser │ │ │ │ │ - * with two arguments: the node being read and a context object passed │ │ │ │ │ - * from the parent. │ │ │ │ │ - */ │ │ │ │ │ - readers: { │ │ │ │ │ - "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "DefaultSRS": function(node, obj) { │ │ │ │ │ - var defaultSRS = this.getChildValue(node); │ │ │ │ │ - if (defaultSRS) { │ │ │ │ │ - obj.srs = defaultSRS; │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WFSCapabilities.v1.prototype.readers["wfs"]), │ │ │ │ │ - "ows": OpenLayers.Format.OWSCommon.v1.prototype.readers.ows │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1_1_0" │ │ │ │ │ - │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Events/featureclick.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ - */ │ │ │ │ │ + // hotSpots define the offset for an Icon │ │ │ │ │ + var hotSpotNode = this.getElementsByTagNameNS(styleTypeNode, │ │ │ │ │ + "*", │ │ │ │ │ + "hotSpot")[0]; │ │ │ │ │ + if (hotSpotNode) { │ │ │ │ │ + var x = parseFloat(hotSpotNode.getAttribute("x")); │ │ │ │ │ + var y = parseFloat(hotSpotNode.getAttribute("y")); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Events.featureclick │ │ │ │ │ - * │ │ │ │ │ - * Extension event type for handling feature click events, including overlapping │ │ │ │ │ - * features. │ │ │ │ │ - * │ │ │ │ │ - * Event types provided by this extension: │ │ │ │ │ - * - featureclick │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Events.featureclick = OpenLayers.Class({ │ │ │ │ │ + var xUnits = hotSpotNode.getAttribute("xunits"); │ │ │ │ │ + if (xUnits == "pixels") { │ │ │ │ │ + style["graphicXOffset"] = -x * scale; │ │ │ │ │ + } else if (xUnits == "insetPixels") { │ │ │ │ │ + style["graphicXOffset"] = -width + (x * scale); │ │ │ │ │ + } else if (xUnits == "fraction") { │ │ │ │ │ + style["graphicXOffset"] = -width * x; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: cache │ │ │ │ │ - * {Object} A cache of features under the mouse. │ │ │ │ │ - */ │ │ │ │ │ - cache: null, │ │ │ │ │ + var yUnits = hotSpotNode.getAttribute("yunits"); │ │ │ │ │ + if (yUnits == "pixels") { │ │ │ │ │ + style["graphicYOffset"] = -height + (y * scale) + 1; │ │ │ │ │ + } else if (yUnits == "insetPixels") { │ │ │ │ │ + style["graphicYOffset"] = -(y * scale) + 1; │ │ │ │ │ + } else if (yUnits == "fraction") { │ │ │ │ │ + style["graphicYOffset"] = -height * (1 - y) + 1; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: map │ │ │ │ │ - * {<OpenLayers.Map>} The map to register browser events on. │ │ │ │ │ - */ │ │ │ │ │ - map: null, │ │ │ │ │ + style["graphicWidth"] = width; │ │ │ │ │ + style["graphicHeight"] = height; │ │ │ │ │ + break; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: provides │ │ │ │ │ - * {Array(String)} The event types provided by this extension. │ │ │ │ │ - */ │ │ │ │ │ - provides: ["featureclick", "nofeatureclick", "featureover", "featureout"], │ │ │ │ │ + case "balloonstyle": │ │ │ │ │ + var balloonStyle = OpenLayers.Util.getXmlNodeValue( │ │ │ │ │ + styleTypeNode); │ │ │ │ │ + if (balloonStyle) { │ │ │ │ │ + style["balloonStyle"] = balloonStyle.replace( │ │ │ │ │ + this.regExes.straightBracket, "${$1}"); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "labelstyle": │ │ │ │ │ + var kmlColor = this.parseProperty(styleTypeNode, "*", "color"); │ │ │ │ │ + var color = this.parseKmlColor(kmlColor); │ │ │ │ │ + if (color) { │ │ │ │ │ + style["fontColor"] = color.color; │ │ │ │ │ + style["fontOpacity"] = color.opacity; │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Events.featureclick │ │ │ │ │ - * Create a new featureclick event type. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * target - {<OpenLayers.Events>} The events instance to create the events │ │ │ │ │ - * for. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(target) { │ │ │ │ │ - this.target = target; │ │ │ │ │ - if (target.object instanceof OpenLayers.Map) { │ │ │ │ │ - this.setMap(target.object); │ │ │ │ │ - } else if (target.object instanceof OpenLayers.Layer.Vector) { │ │ │ │ │ - if (target.object.map) { │ │ │ │ │ - this.setMap(target.object.map); │ │ │ │ │ - } else { │ │ │ │ │ - target.object.events.register("added", this, function(evt) { │ │ │ │ │ - this.setMap(target.object.map); │ │ │ │ │ - }); │ │ │ │ │ + default: │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - throw ("Listeners for '" + this.provides.join("', '") + │ │ │ │ │ - "' events can only be registered for OpenLayers.Layer.Vector " + │ │ │ │ │ - "or OpenLayers.Map instances"); │ │ │ │ │ } │ │ │ │ │ - for (var i = this.provides.length - 1; i >= 0; --i) { │ │ │ │ │ - target.extensions[this.provides[i]] = true; │ │ │ │ │ + │ │ │ │ │ + // Some polygons have no line color, so we use the fillColor for that │ │ │ │ │ + if (!style["strokeColor"] && style["fillColor"]) { │ │ │ │ │ + style["strokeColor"] = style["fillColor"]; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} The map to register browser events on. │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.map = map; │ │ │ │ │ - this.cache = {}; │ │ │ │ │ - map.events.register("mousedown", this, this.start, { │ │ │ │ │ - extension: true │ │ │ │ │ - }); │ │ │ │ │ - map.events.register("mouseup", this, this.onClick, { │ │ │ │ │ - extension: true │ │ │ │ │ - }); │ │ │ │ │ - map.events.register("touchstart", this, this.start, { │ │ │ │ │ - extension: true │ │ │ │ │ - }); │ │ │ │ │ - map.events.register("touchmove", this, this.cancel, { │ │ │ │ │ - extension: true │ │ │ │ │ - }); │ │ │ │ │ - map.events.register("touchend", this, this.onClick, { │ │ │ │ │ - extension: true │ │ │ │ │ - }); │ │ │ │ │ - map.events.register("mousemove", this, this.onMousemove, { │ │ │ │ │ - extension: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + var id = node.getAttribute("id"); │ │ │ │ │ + if (id && style) { │ │ │ │ │ + style.id = id; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: start │ │ │ │ │ - * Sets startEvt = evt. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - */ │ │ │ │ │ - start: function(evt) { │ │ │ │ │ - this.startEvt = evt; │ │ │ │ │ + return style; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: cancel │ │ │ │ │ - * Deletes the start event. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ + * Method: parseStyleMaps │ │ │ │ │ + * Parses <StyleMap> nodes, but only uses the 'normal' key │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * nodes - {Array} of {DOMElement} data to read/parse. │ │ │ │ │ + * options - {Object} Hash of options │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - cancel: function(evt) { │ │ │ │ │ - delete this.startEvt; │ │ │ │ │ - }, │ │ │ │ │ + parseStyleMaps: function(nodes, options) { │ │ │ │ │ + // Only the default or "normal" part of the StyleMap is processed now │ │ │ │ │ + // To do the select or "highlight" bit, we'd need to change lots more │ │ │ │ │ + │ │ │ │ │ + for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ + var node = nodes[i]; │ │ │ │ │ + var pairs = this.getElementsByTagNameNS(node, "*", │ │ │ │ │ + "Pair"); │ │ │ │ │ + │ │ │ │ │ + var id = node.getAttribute("id"); │ │ │ │ │ + for (var j = 0, jlen = pairs.length; j < jlen; j++) { │ │ │ │ │ + var pair = pairs[j]; │ │ │ │ │ + // Use the shortcut in the SLD format to quickly retrieve the │ │ │ │ │ + // value of a node. Maybe it's good to have a method in │ │ │ │ │ + // Format.XML to do this │ │ │ │ │ + var key = this.parseProperty(pair, "*", "key"); │ │ │ │ │ + var styleUrl = this.parseProperty(pair, "*", "styleUrl"); │ │ │ │ │ + │ │ │ │ │ + if (styleUrl && key == "normal") { │ │ │ │ │ + this.styles[(options.styleBaseUrl || "") + "#" + id] = │ │ │ │ │ + this.styles[(options.styleBaseUrl || "") + styleUrl]; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // TODO: implement the "select" part │ │ │ │ │ + //if (styleUrl && key == "highlight") { │ │ │ │ │ + //} │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: onClick │ │ │ │ │ - * Listener for the click event. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - */ │ │ │ │ │ - onClick: function(evt) { │ │ │ │ │ - if (!this.startEvt || evt.type !== "touchend" && │ │ │ │ │ - !OpenLayers.Event.isLeftClick(evt)) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var features = this.getFeatures(this.startEvt); │ │ │ │ │ - delete this.startEvt; │ │ │ │ │ - // fire featureclick events │ │ │ │ │ - var feature, layer, more, clicked = {}; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - layer = feature.layer; │ │ │ │ │ - clicked[layer.id] = true; │ │ │ │ │ - more = this.triggerEvent("featureclick", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - if (more === false) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // fire nofeatureclick events on all vector layers with no targets │ │ │ │ │ - for (i = 0, len = this.map.layers.length; i < len; ++i) { │ │ │ │ │ - layer = this.map.layers[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Vector && !clicked[layer.id]) { │ │ │ │ │ - this.triggerEvent("nofeatureclick", { │ │ │ │ │ - layer: layer │ │ │ │ │ - }); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: onMousemove │ │ │ │ │ - * Listener for the mousemove event. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ + * Method: parseFeatures │ │ │ │ │ + * Loop through all Placemark nodes and parse them. │ │ │ │ │ + * Will create a list of features │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * nodes - {Array} of {DOMElement} data to read/parse. │ │ │ │ │ + * options - {Object} Hash of options │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - onMousemove: function(evt) { │ │ │ │ │ - delete this.startEvt; │ │ │ │ │ - var features = this.getFeatures(evt); │ │ │ │ │ - var over = {}, │ │ │ │ │ - newly = [], │ │ │ │ │ - feature; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - over[feature.id] = feature; │ │ │ │ │ - if (!this.cache[feature.id]) { │ │ │ │ │ - newly.push(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // check if already over features │ │ │ │ │ - var out = []; │ │ │ │ │ - for (var id in this.cache) { │ │ │ │ │ - feature = this.cache[id]; │ │ │ │ │ - if (feature.layer && feature.layer.map) { │ │ │ │ │ - if (!over[feature.id]) { │ │ │ │ │ - out.push(feature); │ │ │ │ │ + parseFeatures: function(nodes, options) { │ │ │ │ │ + var features = []; │ │ │ │ │ + for (var i = 0, len = nodes.length; i < len; i++) { │ │ │ │ │ + var featureNode = nodes[i]; │ │ │ │ │ + var feature = this.parseFeature.apply(this, [featureNode]); │ │ │ │ │ + if (feature) { │ │ │ │ │ + │ │ │ │ │ + // Create reference to styleUrl │ │ │ │ │ + if (this.extractStyles && feature.attributes && │ │ │ │ │ + feature.attributes.styleUrl) { │ │ │ │ │ + feature.style = this.getStyle(feature.attributes.styleUrl, options); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.extractStyles) { │ │ │ │ │ + // Make sure that <Style> nodes within a placemark are │ │ │ │ │ + // processed as well │ │ │ │ │ + var inlineStyleNode = this.getElementsByTagNameNS(featureNode, │ │ │ │ │ + "*", │ │ │ │ │ + "Style")[0]; │ │ │ │ │ + if (inlineStyleNode) { │ │ │ │ │ + var inlineStyle = this.parseStyle(inlineStyleNode); │ │ │ │ │ + if (inlineStyle) { │ │ │ │ │ + feature.style = OpenLayers.Util.extend( │ │ │ │ │ + feature.style, inlineStyle │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // check if gx:Track elements should be parsed │ │ │ │ │ + if (this.extractTracks) { │ │ │ │ │ + var tracks = this.getElementsByTagNameNS( │ │ │ │ │ + featureNode, this.namespaces.gx, "Track" │ │ │ │ │ + ); │ │ │ │ │ + if (tracks && tracks.length > 0) { │ │ │ │ │ + var track = tracks[0]; │ │ │ │ │ + var container = { │ │ │ │ │ + features: [], │ │ │ │ │ + feature: feature │ │ │ │ │ + }; │ │ │ │ │ + this.readNode(track, container); │ │ │ │ │ + if (container.features.length > 0) { │ │ │ │ │ + features.push.apply(features, container.features); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // add feature to list of features │ │ │ │ │ + features.push(feature); │ │ │ │ │ } │ │ │ │ │ } else { │ │ │ │ │ - // removed │ │ │ │ │ - delete this.cache[id]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // fire featureover events │ │ │ │ │ - var more; │ │ │ │ │ - for (i = 0, len = newly.length; i < len; ++i) { │ │ │ │ │ - feature = newly[i]; │ │ │ │ │ - this.cache[feature.id] = feature; │ │ │ │ │ - more = this.triggerEvent("featureover", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - if (more === false) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // fire featureout events │ │ │ │ │ - for (i = 0, len = out.length; i < len; ++i) { │ │ │ │ │ - feature = out[i]; │ │ │ │ │ - delete this.cache[feature.id]; │ │ │ │ │ - more = this.triggerEvent("featureout", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - if (more === false) { │ │ │ │ │ - break; │ │ │ │ │ + throw "Bad Placemark: " + i; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + // add new features to existing feature list │ │ │ │ │ + this.features = this.features.concat(features); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: triggerEvent │ │ │ │ │ - * Determines where to trigger the event and triggers it. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * type - {String} The event type to trigger │ │ │ │ │ - * evt - {Object} The listener argument │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The last listener return. │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ */ │ │ │ │ │ - triggerEvent: function(type, evt) { │ │ │ │ │ - var layer = evt.feature ? evt.feature.layer : evt.layer, │ │ │ │ │ - object = this.target.object; │ │ │ │ │ - if (object instanceof OpenLayers.Map || object === layer) { │ │ │ │ │ - return this.target.triggerEvent(type, evt); │ │ │ │ │ + readers: { │ │ │ │ │ + "kml": { │ │ │ │ │ + "when": function(node, container) { │ │ │ │ │ + container.whens.push(OpenLayers.Date.parse( │ │ │ │ │ + this.getChildValue(node) │ │ │ │ │ + )); │ │ │ │ │ + }, │ │ │ │ │ + "_trackPointAttribute": function(node, container) { │ │ │ │ │ + var name = node.nodeName.split(":").pop(); │ │ │ │ │ + container.attributes[name].push(this.getChildValue(node)); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "gx": { │ │ │ │ │ + "Track": function(node, container) { │ │ │ │ │ + var obj = { │ │ │ │ │ + whens: [], │ │ │ │ │ + points: [], │ │ │ │ │ + angles: [] │ │ │ │ │ + }; │ │ │ │ │ + if (this.trackAttributes) { │ │ │ │ │ + var name; │ │ │ │ │ + obj.attributes = {}; │ │ │ │ │ + for (var i = 0, ii = this.trackAttributes.length; i < ii; ++i) { │ │ │ │ │ + name = this.trackAttributes[i]; │ │ │ │ │ + obj.attributes[name] = []; │ │ │ │ │ + if (!(name in this.readers.kml)) { │ │ │ │ │ + this.readers.kml[name] = this.readers.kml._trackPointAttribute; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + if (obj.whens.length !== obj.points.length) { │ │ │ │ │ + throw new Error("gx:Track with unequal number of when (" + │ │ │ │ │ + obj.whens.length + ") and gx:coord (" + │ │ │ │ │ + obj.points.length + ") elements."); │ │ │ │ │ + } │ │ │ │ │ + var hasAngles = obj.angles.length > 0; │ │ │ │ │ + if (hasAngles && obj.whens.length !== obj.angles.length) { │ │ │ │ │ + throw new Error("gx:Track with unequal number of when (" + │ │ │ │ │ + obj.whens.length + ") and gx:angles (" + │ │ │ │ │ + obj.angles.length + ") elements."); │ │ │ │ │ + } │ │ │ │ │ + var feature, point, angles; │ │ │ │ │ + for (var i = 0, ii = obj.whens.length; i < ii; ++i) { │ │ │ │ │ + feature = container.feature.clone(); │ │ │ │ │ + feature.fid = container.feature.fid || container.feature.id; │ │ │ │ │ + point = obj.points[i]; │ │ │ │ │ + feature.geometry = point; │ │ │ │ │ + if ("z" in point) { │ │ │ │ │ + feature.attributes.altitude = point.z; │ │ │ │ │ + } │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + feature.geometry.transform( │ │ │ │ │ + this.externalProjection, this.internalProjection │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (this.trackAttributes) { │ │ │ │ │ + for (var j = 0, jj = this.trackAttributes.length; j < jj; ++j) { │ │ │ │ │ + var name = this.trackAttributes[j]; │ │ │ │ │ + feature.attributes[name] = obj.attributes[name][i]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + feature.attributes.when = obj.whens[i]; │ │ │ │ │ + feature.attributes.trackId = container.feature.id; │ │ │ │ │ + if (hasAngles) { │ │ │ │ │ + angles = obj.angles[i]; │ │ │ │ │ + feature.attributes.heading = parseFloat(angles[0]); │ │ │ │ │ + feature.attributes.tilt = parseFloat(angles[1]); │ │ │ │ │ + feature.attributes.roll = parseFloat(angles[2]); │ │ │ │ │ + } │ │ │ │ │ + container.features.push(feature); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "coord": function(node, container) { │ │ │ │ │ + var str = this.getChildValue(node); │ │ │ │ │ + var coords = str.replace(this.regExes.trimSpace, "").split(/\s+/); │ │ │ │ │ + var point = new OpenLayers.Geometry.Point(coords[0], coords[1]); │ │ │ │ │ + if (coords.length > 2) { │ │ │ │ │ + point.z = parseFloat(coords[2]); │ │ │ │ │ + } │ │ │ │ │ + container.points.push(point); │ │ │ │ │ + }, │ │ │ │ │ + "angles": function(node, container) { │ │ │ │ │ + var str = this.getChildValue(node); │ │ │ │ │ + var parts = str.replace(this.regExes.trimSpace, "").split(/\s+/); │ │ │ │ │ + container.angles.push(parts); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatures │ │ │ │ │ - * Get all features at the given screen location. │ │ │ │ │ + * Method: parseFeature │ │ │ │ │ + * This function is the core of the KML parsing code in OpenLayers. │ │ │ │ │ + * It creates the geometries that are then attached to the returned │ │ │ │ │ + * feature, and calls parseAttributes() to get attribute data out. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} Event object. │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} List of features at the given point. │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A vector feature. │ │ │ │ │ */ │ │ │ │ │ - getFeatures: function(evt) { │ │ │ │ │ - var x = evt.clientX, │ │ │ │ │ - y = evt.clientY, │ │ │ │ │ - features = [], │ │ │ │ │ - targets = [], │ │ │ │ │ - layers = [], │ │ │ │ │ - layer, target, feature, i, len; │ │ │ │ │ - // go through all layers looking for targets │ │ │ │ │ - for (i = this.map.layers.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = this.map.layers[i]; │ │ │ │ │ - if (layer.div.style.display !== "none") { │ │ │ │ │ - if (layer.renderer instanceof OpenLayers.Renderer.Elements) { │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Vector) { │ │ │ │ │ - target = document.elementFromPoint(x, y); │ │ │ │ │ - while (target && target._featureId) { │ │ │ │ │ - feature = layer.getFeatureById(target._featureId); │ │ │ │ │ - if (feature) { │ │ │ │ │ - features.push(feature); │ │ │ │ │ - target.style.display = "none"; │ │ │ │ │ - targets.push(target); │ │ │ │ │ - target = document.elementFromPoint(x, y); │ │ │ │ │ - } else { │ │ │ │ │ - // sketch, all bets off │ │ │ │ │ - target = false; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - layers.push(layer); │ │ │ │ │ - layer.div.style.display = "none"; │ │ │ │ │ - } else if (layer.renderer instanceof OpenLayers.Renderer.Canvas) { │ │ │ │ │ - feature = layer.renderer.getFeatureIdFromEvent(evt); │ │ │ │ │ - if (feature) { │ │ │ │ │ - features.push(feature); │ │ │ │ │ - layers.push(layer); │ │ │ │ │ + parseFeature: function(node) { │ │ │ │ │ + // only accept one geometry per feature - look for highest "order" │ │ │ │ │ + var order = ["MultiGeometry", "Polygon", "LineString", "Point"]; │ │ │ │ │ + var type, nodeList, geometry, parser; │ │ │ │ │ + for (var i = 0, len = order.length; i < len; ++i) { │ │ │ │ │ + type = order[i]; │ │ │ │ │ + this.internalns = node.namespaceURI ? │ │ │ │ │ + node.namespaceURI : this.kmlns; │ │ │ │ │ + nodeList = this.getElementsByTagNameNS(node, │ │ │ │ │ + this.internalns, type); │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + // only deal with first geometry of this type │ │ │ │ │ + var parser = this.parseGeometry[type.toLowerCase()]; │ │ │ │ │ + if (parser) { │ │ │ │ │ + geometry = parser.apply(this, [nodeList[0]]); │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry.transform(this.externalProjection, │ │ │ │ │ + this.internalProjection); │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + throw new TypeError("Unsupported geometry type: " + type); │ │ │ │ │ } │ │ │ │ │ + // stop looking for different geometry types │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - // restore feature visibility │ │ │ │ │ - for (i = 0, len = targets.length; i < len; ++i) { │ │ │ │ │ - targets[i].style.display = ""; │ │ │ │ │ + │ │ │ │ │ + // construct feature (optionally with attributes) │ │ │ │ │ + var attributes; │ │ │ │ │ + if (this.extractAttributes) { │ │ │ │ │ + attributes = this.parseAttributes(node); │ │ │ │ │ } │ │ │ │ │ - // restore layer visibility │ │ │ │ │ - for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - layers[i].div.style.display = "block"; │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry, attributes); │ │ │ │ │ + │ │ │ │ │ + var fid = node.getAttribute("id") || node.getAttribute("name"); │ │ │ │ │ + if (fid != null) { │ │ │ │ │ + feature.fid = fid; │ │ │ │ │ } │ │ │ │ │ - return features; │ │ │ │ │ + │ │ │ │ │ + return feature; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up. │ │ │ │ │ + * Method: getStyle │ │ │ │ │ + * Retrieves a style from a style hash using styleUrl as the key │ │ │ │ │ + * If the styleUrl doesn't exist yet, we try to fetch it │ │ │ │ │ + * Internet │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * styleUrl - {String} URL of style │ │ │ │ │ + * options - {Object} Hash of options │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} - (reference to) Style hash │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i = this.provides.length - 1; i >= 0; --i) { │ │ │ │ │ - delete this.target.extensions[this.provides[i]]; │ │ │ │ │ - } │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - mousemove: this.onMousemove, │ │ │ │ │ - mousedown: this.start, │ │ │ │ │ - mouseup: this.onClick, │ │ │ │ │ - touchstart: this.start, │ │ │ │ │ - touchmove: this.cancel, │ │ │ │ │ - touchend: this.onClick, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - delete this.cache; │ │ │ │ │ - delete this.map; │ │ │ │ │ - delete this.target; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ + getStyle: function(styleUrl, options) { │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Events.nofeatureclick │ │ │ │ │ - * │ │ │ │ │ - * Extension event type for handling click events that do not hit a feature. │ │ │ │ │ - * │ │ │ │ │ - * Event types provided by this extension: │ │ │ │ │ - * - nofeatureclick │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Events.nofeatureclick = OpenLayers.Events.featureclick; │ │ │ │ │ + var styleBaseUrl = OpenLayers.Util.removeTail(styleUrl); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Events.featureover │ │ │ │ │ - * │ │ │ │ │ - * Extension event type for handling hovering over a feature. │ │ │ │ │ - * │ │ │ │ │ - * Event types provided by this extension: │ │ │ │ │ - * - featureover │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Events.featureover = OpenLayers.Events.featureclick; │ │ │ │ │ + var newOptions = OpenLayers.Util.extend({}, options); │ │ │ │ │ + newOptions.depth++; │ │ │ │ │ + newOptions.styleBaseUrl = styleBaseUrl; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Events.featureout │ │ │ │ │ - * │ │ │ │ │ - * Extension event type for handling leaving a feature. │ │ │ │ │ - * │ │ │ │ │ - * Event types provided by this extension: │ │ │ │ │ - * - featureout │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Events.featureout = OpenLayers.Events.featureclick; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Events/buttonclick.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + // Fetch remote Style URLs (if not fetched before) │ │ │ │ │ + if (!this.styles[styleUrl] && │ │ │ │ │ + !OpenLayers.String.startsWith(styleUrl, "#") && │ │ │ │ │ + newOptions.depth <= this.maxDepth && │ │ │ │ │ + !this.fetched[styleBaseUrl]) { │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + var data = this.fetchLink(styleBaseUrl); │ │ │ │ │ + if (data) { │ │ │ │ │ + this.parseData(data, newOptions); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ - */ │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Events.buttonclick │ │ │ │ │ - * Extension event type for handling buttons on top of a dom element. This │ │ │ │ │ - * event type fires "buttonclick" on its <target> when a button was │ │ │ │ │ - * clicked. Buttons are detected by the "olButton" class. │ │ │ │ │ - * │ │ │ │ │ - * This event type makes sure that button clicks do not interfere with other │ │ │ │ │ - * events that are registered on the same <element>. │ │ │ │ │ - * │ │ │ │ │ - * Event types provided by this extension: │ │ │ │ │ - * - *buttonclick* Triggered when a button is clicked. Listeners receive an │ │ │ │ │ - * object with a *buttonElement* property referencing the dom element of │ │ │ │ │ - * the clicked button, and an *buttonXY* property with the click position │ │ │ │ │ - * relative to the button. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Events.buttonclick = OpenLayers.Class({ │ │ │ │ │ + // return requested style │ │ │ │ │ + var style = OpenLayers.Util.extend({}, this.styles[styleUrl]); │ │ │ │ │ + return style; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: target │ │ │ │ │ - * {<OpenLayers.Events>} The events instance that the buttonclick event will │ │ │ │ │ - * be triggered on. │ │ │ │ │ + * Property: parseGeometry │ │ │ │ │ + * Properties of this object are the functions that parse geometries based │ │ │ │ │ + * on their type. │ │ │ │ │ */ │ │ │ │ │ - target: null, │ │ │ │ │ + parseGeometry: { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: events │ │ │ │ │ - * {Array} Events to observe and conditionally stop from propagating when │ │ │ │ │ - * an element with the olButton class (or its olAlphaImg child) is │ │ │ │ │ - * clicked. │ │ │ │ │ - */ │ │ │ │ │ - events: [ │ │ │ │ │ - 'mousedown', 'mouseup', 'click', 'dblclick', │ │ │ │ │ - 'touchstart', 'touchmove', 'touchend', 'keydown' │ │ │ │ │ - ], │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseGeometry.point │ │ │ │ │ + * Given a KML node representing a point geometry, create an OpenLayers │ │ │ │ │ + * point geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A KML Point node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Point>} A point geometry. │ │ │ │ │ + */ │ │ │ │ │ + point: function(node) { │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, this.internalns, │ │ │ │ │ + "coordinates"); │ │ │ │ │ + var coords = []; │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + var coordString = nodeList[0].firstChild.nodeValue; │ │ │ │ │ + coordString = coordString.replace(this.regExes.removeSpace, ""); │ │ │ │ │ + coords = coordString.split(","); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: startRegEx │ │ │ │ │ - * {RegExp} Regular expression to test Event.type for events that start │ │ │ │ │ - * a buttonclick sequence. │ │ │ │ │ - */ │ │ │ │ │ - startRegEx: /^mousedown|touchstart$/, │ │ │ │ │ + var point = null; │ │ │ │ │ + if (coords.length > 1) { │ │ │ │ │ + // preserve third dimension │ │ │ │ │ + if (coords.length == 2) { │ │ │ │ │ + coords[2] = null; │ │ │ │ │ + } │ │ │ │ │ + point = new OpenLayers.Geometry.Point(coords[0], coords[1], │ │ │ │ │ + coords[2]); │ │ │ │ │ + } else { │ │ │ │ │ + throw "Bad coordinate string: " + coordString; │ │ │ │ │ + } │ │ │ │ │ + return point; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: cancelRegEx │ │ │ │ │ - * {RegExp} Regular expression to test Event.type for events that cancel │ │ │ │ │ - * a buttonclick sequence. │ │ │ │ │ - */ │ │ │ │ │ - cancelRegEx: /^touchmove$/, │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseGeometry.linestring │ │ │ │ │ + * Given a KML node representing a linestring geometry, create an │ │ │ │ │ + * OpenLayers linestring geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A KML LineString node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.LineString>} A linestring geometry. │ │ │ │ │ + */ │ │ │ │ │ + linestring: function(node, ring) { │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, this.internalns, │ │ │ │ │ + "coordinates"); │ │ │ │ │ + var line = null; │ │ │ │ │ + if (nodeList.length > 0) { │ │ │ │ │ + var coordString = this.getChildValue(nodeList[0]); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: completeRegEx │ │ │ │ │ - * {RegExp} Regular expression to test Event.type for events that complete │ │ │ │ │ - * a buttonclick sequence. │ │ │ │ │ - */ │ │ │ │ │ - completeRegEx: /^mouseup|touchend$/, │ │ │ │ │ + coordString = coordString.replace(this.regExes.trimSpace, │ │ │ │ │ + ""); │ │ │ │ │ + coordString = coordString.replace(this.regExes.trimComma, │ │ │ │ │ + ","); │ │ │ │ │ + var pointList = coordString.split(this.regExes.splitSpace); │ │ │ │ │ + var numPoints = pointList.length; │ │ │ │ │ + var points = new Array(numPoints); │ │ │ │ │ + var coords, numCoords; │ │ │ │ │ + for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ + coords = pointList[i].split(","); │ │ │ │ │ + numCoords = coords.length; │ │ │ │ │ + if (numCoords > 1) { │ │ │ │ │ + if (coords.length == 2) { │ │ │ │ │ + coords[2] = null; │ │ │ │ │ + } │ │ │ │ │ + points[i] = new OpenLayers.Geometry.Point(coords[0], │ │ │ │ │ + coords[1], │ │ │ │ │ + coords[2]); │ │ │ │ │ + } else { │ │ │ │ │ + throw "Bad LineString point coordinates: " + │ │ │ │ │ + pointList[i]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (numPoints) { │ │ │ │ │ + if (ring) { │ │ │ │ │ + line = new OpenLayers.Geometry.LinearRing(points); │ │ │ │ │ + } else { │ │ │ │ │ + line = new OpenLayers.Geometry.LineString(points); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + throw "Bad LineString coordinates: " + coordString; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: startEvt │ │ │ │ │ - * {Event} The event that started the click sequence │ │ │ │ │ - */ │ │ │ │ │ + return line; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Events.buttonclick │ │ │ │ │ - * Construct a buttonclick event type. Applications are not supposed to │ │ │ │ │ - * create instances of this class - they are created on demand by │ │ │ │ │ - * <OpenLayers.Events> instances. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * target - {<OpenLayers.Events>} The events instance that the buttonclick │ │ │ │ │ - * event will be triggered on. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(target) { │ │ │ │ │ - this.target = target; │ │ │ │ │ - for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ - this.target.register(this.events[i], this, this.buttonClick, { │ │ │ │ │ - extension: true │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseGeometry.polygon │ │ │ │ │ + * Given a KML node representing a polygon geometry, create an │ │ │ │ │ + * OpenLayers polygon geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A KML Polygon node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Polygon>} A polygon geometry. │ │ │ │ │ + */ │ │ │ │ │ + polygon: function(node) { │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, this.internalns, │ │ │ │ │ + "LinearRing"); │ │ │ │ │ + var numRings = nodeList.length; │ │ │ │ │ + var components = new Array(numRings); │ │ │ │ │ + if (numRings > 0) { │ │ │ │ │ + // this assumes exterior ring first, inner rings after │ │ │ │ │ + var ring; │ │ │ │ │ + for (var i = 0, len = nodeList.length; i < len; ++i) { │ │ │ │ │ + ring = this.parseGeometry.linestring.apply(this, │ │ │ │ │ + [nodeList[i], true]); │ │ │ │ │ + if (ring) { │ │ │ │ │ + components[i] = ring; │ │ │ │ │ + } else { │ │ │ │ │ + throw "Bad LinearRing geometry: " + i; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.Polygon(components); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ - this.target.unregister(this.events[i], this, this.buttonClick); │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseGeometry.multigeometry │ │ │ │ │ + * Given a KML node representing a multigeometry, create an │ │ │ │ │ + * OpenLayers geometry collection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A KML MultiGeometry node. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Geometry.Collection>} A geometry collection. │ │ │ │ │ + */ │ │ │ │ │ + multigeometry: function(node) { │ │ │ │ │ + var child, parser; │ │ │ │ │ + var parts = []; │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ + child = children[i]; │ │ │ │ │ + if (child.nodeType == 1) { │ │ │ │ │ + var type = (child.prefix) ? │ │ │ │ │ + child.nodeName.split(":")[1] : │ │ │ │ │ + child.nodeName; │ │ │ │ │ + var parser = this.parseGeometry[type.toLowerCase()]; │ │ │ │ │ + if (parser) { │ │ │ │ │ + parts.push(parser.apply(this, [child])); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.Collection(parts); │ │ │ │ │ } │ │ │ │ │ - delete this.target; │ │ │ │ │ + │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getPressedButton │ │ │ │ │ - * Get the pressed button, if any. Returns undefined if no button │ │ │ │ │ - * was pressed. │ │ │ │ │ + * Method: parseAttributes │ │ │ │ │ * │ │ │ │ │ - * Arguments: │ │ │ │ │ - * element - {DOMElement} The event target. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} The button element, or undefined. │ │ │ │ │ + * {Object} An attributes object. │ │ │ │ │ */ │ │ │ │ │ - getPressedButton: function(element) { │ │ │ │ │ - var depth = 3, // limit the search depth │ │ │ │ │ - button; │ │ │ │ │ - do { │ │ │ │ │ - if (OpenLayers.Element.hasClass(element, "olButton")) { │ │ │ │ │ - // hit! │ │ │ │ │ - button = element; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - element = element.parentNode; │ │ │ │ │ - } while (--depth > 0 && element); │ │ │ │ │ - return button; │ │ │ │ │ - }, │ │ │ │ │ + parseAttributes: function(node) { │ │ │ │ │ + var attributes = {}; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: ignore │ │ │ │ │ - * Check for event target elements that should be ignored by OpenLayers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * element - {DOMElement} The event target. │ │ │ │ │ - */ │ │ │ │ │ - ignore: function(element) { │ │ │ │ │ - var depth = 3, │ │ │ │ │ - ignore = false; │ │ │ │ │ - do { │ │ │ │ │ - if (element.nodeName.toLowerCase() === 'a') { │ │ │ │ │ - ignore = true; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - element = element.parentNode; │ │ │ │ │ - } while (--depth > 0 && element); │ │ │ │ │ - return ignore; │ │ │ │ │ - }, │ │ │ │ │ + // Extended Data is parsed first. │ │ │ │ │ + var edNodes = node.getElementsByTagName("ExtendedData"); │ │ │ │ │ + if (edNodes.length) { │ │ │ │ │ + attributes = this.parseExtendedData(edNodes[0]); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: buttonClick │ │ │ │ │ - * Check if a button was clicked, and fire the buttonclick event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - buttonClick: function(evt) { │ │ │ │ │ - var propagate = true, │ │ │ │ │ - element = OpenLayers.Event.element(evt); │ │ │ │ │ - if (element && (OpenLayers.Event.isLeftClick(evt) || !~evt.type.indexOf("mouse"))) { │ │ │ │ │ - // was a button pressed? │ │ │ │ │ - var button = this.getPressedButton(element); │ │ │ │ │ - if (button) { │ │ │ │ │ - if (evt.type === "keydown") { │ │ │ │ │ - switch (evt.keyCode) { │ │ │ │ │ - case OpenLayers.Event.KEY_RETURN: │ │ │ │ │ - case OpenLayers.Event.KEY_SPACE: │ │ │ │ │ - this.target.triggerEvent("buttonclick", { │ │ │ │ │ - buttonElement: button │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } else if (this.startEvt) { │ │ │ │ │ - if (this.completeRegEx.test(evt.type)) { │ │ │ │ │ - var pos = OpenLayers.Util.pagePosition(button); │ │ │ │ │ - var viewportElement = OpenLayers.Util.getViewportElement(); │ │ │ │ │ - var scrollTop = window.pageYOffset || viewportElement.scrollTop; │ │ │ │ │ - var scrollLeft = window.pageXOffset || viewportElement.scrollLeft; │ │ │ │ │ - pos[0] = pos[0] - scrollLeft; │ │ │ │ │ - pos[1] = pos[1] - scrollTop; │ │ │ │ │ + // assume attribute nodes are type 1 children with a type 3 or 4 child │ │ │ │ │ + var child, grandchildren, grandchild; │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ │ │ │ │ │ - this.target.triggerEvent("buttonclick", { │ │ │ │ │ - buttonElement: button, │ │ │ │ │ - buttonXY: { │ │ │ │ │ - x: this.startEvt.clientX - pos[0], │ │ │ │ │ - y: this.startEvt.clientY - pos[1] │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ + for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ + child = children[i]; │ │ │ │ │ + if (child.nodeType == 1) { │ │ │ │ │ + grandchildren = child.childNodes; │ │ │ │ │ + if (grandchildren.length >= 1 && grandchildren.length <= 3) { │ │ │ │ │ + var grandchild; │ │ │ │ │ + switch (grandchildren.length) { │ │ │ │ │ + case 1: │ │ │ │ │ + grandchild = grandchildren[0]; │ │ │ │ │ + break; │ │ │ │ │ + case 2: │ │ │ │ │ + var c1 = grandchildren[0]; │ │ │ │ │ + var c2 = grandchildren[1]; │ │ │ │ │ + grandchild = (c1.nodeType == 3 || c1.nodeType == 4) ? │ │ │ │ │ + c1 : c2; │ │ │ │ │ + break; │ │ │ │ │ + case 3: │ │ │ │ │ + default: │ │ │ │ │ + grandchild = grandchildren[1]; │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ - if (this.cancelRegEx.test(evt.type)) { │ │ │ │ │ - delete this.startEvt; │ │ │ │ │ + if (grandchild.nodeType == 3 || grandchild.nodeType == 4) { │ │ │ │ │ + var name = (child.prefix) ? │ │ │ │ │ + child.nodeName.split(":")[1] : │ │ │ │ │ + child.nodeName; │ │ │ │ │ + var value = OpenLayers.Util.getXmlNodeValue(grandchild); │ │ │ │ │ + if (value) { │ │ │ │ │ + value = value.replace(this.regExes.trimSpace, ""); │ │ │ │ │ + attributes[name] = value; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false; │ │ │ │ │ - } │ │ │ │ │ - if (this.startRegEx.test(evt.type)) { │ │ │ │ │ - this.startEvt = evt; │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false; │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - propagate = !this.ignore(OpenLayers.Event.element(evt)); │ │ │ │ │ - delete this.startEvt; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return propagate; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Point.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Handler.Point │ │ │ │ │ - * Handler to draw a point on the map. Point is displayed on activation, │ │ │ │ │ - * moves on mouse move, and is finished on mouse up. The handler triggers │ │ │ │ │ - * callbacks for 'done', 'cancel', and 'modify'. The modify callback is │ │ │ │ │ - * called with each change in the sketch and will receive the latest point │ │ │ │ │ - * drawn. Create a new instance with the <OpenLayers.Handler.Point> │ │ │ │ │ - * constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.Point = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: point │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} The currently drawn point │ │ │ │ │ - */ │ │ │ │ │ - point: null, │ │ │ │ │ + return attributes; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} The temporary drawing layer │ │ │ │ │ + * Method: parseExtendedData │ │ │ │ │ + * Parse ExtendedData from KML. Limited support for schemas/datatypes. │ │ │ │ │ + * See http://code.google.com/apis/kml/documentation/kmlreference.html#extendeddata │ │ │ │ │ + * for more information on extendeddata. │ │ │ │ │ */ │ │ │ │ │ - layer: null, │ │ │ │ │ + parseExtendedData: function(node) { │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + var i, len, data, key; │ │ │ │ │ + var dataNodes = node.getElementsByTagName("Data"); │ │ │ │ │ + for (i = 0, len = dataNodes.length; i < len; i++) { │ │ │ │ │ + data = dataNodes[i]; │ │ │ │ │ + key = data.getAttribute("name"); │ │ │ │ │ + var ed = {}; │ │ │ │ │ + var valueNode = data.getElementsByTagName("value"); │ │ │ │ │ + if (valueNode.length) { │ │ │ │ │ + ed['value'] = this.getChildValue(valueNode[0]); │ │ │ │ │ + } │ │ │ │ │ + if (this.kvpAttributes) { │ │ │ │ │ + attributes[key] = ed['value']; │ │ │ │ │ + } else { │ │ │ │ │ + var nameNode = data.getElementsByTagName("displayName"); │ │ │ │ │ + if (nameNode.length) { │ │ │ │ │ + ed['displayName'] = this.getChildValue(nameNode[0]); │ │ │ │ │ + } │ │ │ │ │ + attributes[key] = ed; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var simpleDataNodes = node.getElementsByTagName("SimpleData"); │ │ │ │ │ + for (i = 0, len = simpleDataNodes.length; i < len; i++) { │ │ │ │ │ + var ed = {}; │ │ │ │ │ + data = simpleDataNodes[i]; │ │ │ │ │ + key = data.getAttribute("name"); │ │ │ │ │ + ed['value'] = this.getChildValue(data); │ │ │ │ │ + if (this.kvpAttributes) { │ │ │ │ │ + attributes[key] = ed['value']; │ │ │ │ │ + } else { │ │ │ │ │ + ed['displayName'] = key; │ │ │ │ │ + attributes[key] = ed; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: multi │ │ │ │ │ - * {Boolean} Cast features to multi-part geometries before passing to the │ │ │ │ │ - * layer. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - multi: false, │ │ │ │ │ + return attributes; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: citeCompliant │ │ │ │ │ - * {Boolean} If set to true, coordinates of features drawn in a map extent │ │ │ │ │ - * crossing the date line won't exceed the world bounds. Default is false. │ │ │ │ │ + * Method: parseProperty │ │ │ │ │ + * Convenience method to find a node and return its value │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * xmlNode - {<DOMElement>} │ │ │ │ │ + * namespace - {String} namespace of the node to find │ │ │ │ │ + * tagName - {String} name of the property to parse │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The value for the requested property (defaults to null) │ │ │ │ │ */ │ │ │ │ │ - citeCompliant: false, │ │ │ │ │ + parseProperty: function(xmlNode, namespace, tagName) { │ │ │ │ │ + var value; │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(xmlNode, namespace, tagName); │ │ │ │ │ + try { │ │ │ │ │ + value = OpenLayers.Util.getXmlNodeValue(nodeList[0]); │ │ │ │ │ + } catch (e) { │ │ │ │ │ + value = null; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: mouseDown │ │ │ │ │ - * {Boolean} The mouse is down │ │ │ │ │ - */ │ │ │ │ │ - mouseDown: false, │ │ │ │ │ + return value; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: stoppedDown │ │ │ │ │ - * {Boolean} Indicate whether the last mousedown stopped the event │ │ │ │ │ - * propagation. │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Accept Feature Collection, and return a string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} An array of features. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A KML string. │ │ │ │ │ */ │ │ │ │ │ - stoppedDown: null, │ │ │ │ │ + write: function(features) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ + } │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "kml"); │ │ │ │ │ + var folder = this.createFolderXML(); │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + folder.appendChild(this.createPlacemarkXML(features[i])); │ │ │ │ │ + } │ │ │ │ │ + kml.appendChild(folder); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [kml]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: lastDown │ │ │ │ │ - * {<OpenLayers.Pixel>} Location of the last mouse down │ │ │ │ │ + * Method: createFolderXML │ │ │ │ │ + * Creates and returns a KML folder node │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - lastDown: null, │ │ │ │ │ + createFolderXML: function() { │ │ │ │ │ + // Folder │ │ │ │ │ + var folder = this.createElementNS(this.kmlns, "Folder"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: lastUp │ │ │ │ │ - * {<OpenLayers.Pixel>} │ │ │ │ │ - */ │ │ │ │ │ - lastUp: null, │ │ │ │ │ + // Folder name │ │ │ │ │ + if (this.foldersName) { │ │ │ │ │ + var folderName = this.createElementNS(this.kmlns, "name"); │ │ │ │ │ + var folderNameText = this.createTextNode(this.foldersName); │ │ │ │ │ + folderName.appendChild(folderNameText); │ │ │ │ │ + folder.appendChild(folderName); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: persist │ │ │ │ │ - * {Boolean} Leave the feature rendered until destroyFeature is called. │ │ │ │ │ - * Default is false. If set to true, the feature remains rendered until │ │ │ │ │ - * destroyFeature is called, typically by deactivating the handler or │ │ │ │ │ - * starting another drawing. │ │ │ │ │ - */ │ │ │ │ │ - persist: false, │ │ │ │ │ + // Folder description │ │ │ │ │ + if (this.foldersDesc) { │ │ │ │ │ + var folderDesc = this.createElementNS(this.kmlns, "description"); │ │ │ │ │ + var folderDescText = this.createTextNode(this.foldersDesc); │ │ │ │ │ + folderDesc.appendChild(folderDescText); │ │ │ │ │ + folder.appendChild(folderDesc); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: stopDown │ │ │ │ │ - * {Boolean} Stop event propagation on mousedown. Must be false to │ │ │ │ │ - * allow "pan while drawing". Defaults to false. │ │ │ │ │ - */ │ │ │ │ │ - stopDown: false, │ │ │ │ │ + return folder; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIPropery: stopUp │ │ │ │ │ - * {Boolean} Stop event propagation on mouse. Must be false to │ │ │ │ │ - * allow "pan while dragging". Defaults to fase. │ │ │ │ │ + * Method: createPlacemarkXML │ │ │ │ │ + * Creates and returns a KML placemark node representing the given feature. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - stopUp: false, │ │ │ │ │ + createPlacemarkXML: function(feature) { │ │ │ │ │ + // Placemark name │ │ │ │ │ + var placemarkName = this.createElementNS(this.kmlns, "name"); │ │ │ │ │ + var label = (feature.style && feature.style.label) ? feature.style.label : feature.id; │ │ │ │ │ + var name = feature.attributes.name || label; │ │ │ │ │ + placemarkName.appendChild(this.createTextNode(name)); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: layerOptions │ │ │ │ │ - * {Object} Any optional properties to be set on the sketch layer. │ │ │ │ │ - */ │ │ │ │ │ - layerOptions: null, │ │ │ │ │ + // Placemark description │ │ │ │ │ + var placemarkDesc = this.createElementNS(this.kmlns, "description"); │ │ │ │ │ + var desc = feature.attributes.description || this.placemarksDesc; │ │ │ │ │ + placemarkDesc.appendChild(this.createTextNode(desc)); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: pixelTolerance │ │ │ │ │ - * {Number} Maximum number of pixels between down and up (mousedown │ │ │ │ │ - * and mouseup, or touchstart and touchend) for the handler to │ │ │ │ │ - * add a new point. If set to an integer value, if the │ │ │ │ │ - * displacement between down and up is great to this value │ │ │ │ │ - * no point will be added. Default value is 5. │ │ │ │ │ - */ │ │ │ │ │ - pixelTolerance: 5, │ │ │ │ │ + // Placemark │ │ │ │ │ + var placemarkNode = this.createElementNS(this.kmlns, "Placemark"); │ │ │ │ │ + if (feature.fid != null) { │ │ │ │ │ + placemarkNode.setAttribute("id", feature.fid); │ │ │ │ │ + } │ │ │ │ │ + placemarkNode.appendChild(placemarkName); │ │ │ │ │ + placemarkNode.appendChild(placemarkDesc); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: lastTouchPx │ │ │ │ │ - * {<OpenLayers.Pixel>} The last pixel used to know the distance between │ │ │ │ │ - * two touches (for double touch). │ │ │ │ │ - */ │ │ │ │ │ - lastTouchPx: null, │ │ │ │ │ + // Geometry node (Point, LineString, etc. nodes) │ │ │ │ │ + var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ + placemarkNode.appendChild(geometryNode); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Point │ │ │ │ │ - * Create a new point handler. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ - * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ - * functions. Various callbacks described below. │ │ │ │ │ - * options - {Object} An optional object with properties to be set on the │ │ │ │ │ - * handler │ │ │ │ │ - * │ │ │ │ │ - * Named callbacks: │ │ │ │ │ - * create - Called when a sketch is first created. Callback called with │ │ │ │ │ - * the creation point geometry and sketch feature. │ │ │ │ │ - * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ - * geometry and the sketch feature. │ │ │ │ │ - * done - Called when the point drawing is finished. The callback will │ │ │ │ │ - * recieve a single argument, the point geometry. │ │ │ │ │ - * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ - * cancel callback will receive a geometry. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ - this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style['default'], {}); │ │ │ │ │ + // output attributes as extendedData │ │ │ │ │ + if (feature.attributes) { │ │ │ │ │ + var edNode = this.buildExtendedData(feature.attributes); │ │ │ │ │ + if (edNode) { │ │ │ │ │ + placemarkNode.appendChild(edNode); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + return placemarkNode; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * turn on the handler │ │ │ │ │ + * Method: buildGeometryNode │ │ │ │ │ + * Builds and returns a KML geometry node with the given geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (!OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - return false; │ │ │ │ │ + buildGeometryNode: function(geometry) { │ │ │ │ │ + var className = geometry.CLASS_NAME; │ │ │ │ │ + var type = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ + var builder = this.buildGeometry[type.toLowerCase()]; │ │ │ │ │ + var node = null; │ │ │ │ │ + if (builder) { │ │ │ │ │ + node = builder.apply(this, [geometry]); │ │ │ │ │ } │ │ │ │ │ - // create temporary vector layer for rendering geometry sketch │ │ │ │ │ - // TBD: this could be moved to initialize/destroy - setting visibility here │ │ │ │ │ - var options = OpenLayers.Util.extend({ │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - // indicate that the temp vector layer will never be out of range │ │ │ │ │ - // without this, resolution properties must be specified at the │ │ │ │ │ - // map-level for this temporary layer to init its resolutions │ │ │ │ │ - // correctly │ │ │ │ │ - calculateInRange: OpenLayers.Function.True, │ │ │ │ │ - wrapDateLine: this.citeCompliant │ │ │ │ │ - }, this.layerOptions); │ │ │ │ │ - this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ - this.map.addLayer(this.layer); │ │ │ │ │ - return true; │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createFeature │ │ │ │ │ - * Add temporary features │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} A pixel location on the map. │ │ │ │ │ + * Property: buildGeometry │ │ │ │ │ + * Object containing methods to do the actual geometry node building │ │ │ │ │ + * based on geometry type. │ │ │ │ │ */ │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ - lonlat.lon, lonlat.lat │ │ │ │ │ - ); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.point]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.point], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + buildGeometry: { │ │ │ │ │ + // TBD: Anybody care about namespace aliases here (these nodes have │ │ │ │ │ + // no prefixes)? │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * turn off the handler │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (!OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - this.cancel(); │ │ │ │ │ - // If a layer's map property is set to null, it means that that layer │ │ │ │ │ - // isn't added to the map. Since we ourself added the layer to the map │ │ │ │ │ - // in activate(), we can assume that if this.layer.map is null it means │ │ │ │ │ - // that the layer has been destroyed (as a result of map.destroy() for │ │ │ │ │ - // example. │ │ │ │ │ - if (this.layer.map != null) { │ │ │ │ │ - this.destroyFeature(true); │ │ │ │ │ - this.layer.destroy(false); │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.point │ │ │ │ │ + * Given an OpenLayers point geometry, create a KML point. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Point>} A point geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML point node. │ │ │ │ │ + */ │ │ │ │ │ + point: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "Point"); │ │ │ │ │ + kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + return kml; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.multipoint │ │ │ │ │ + * Given an OpenLayers multipoint geometry, create a KML │ │ │ │ │ + * GeometryCollection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Point>} A multipoint geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ + */ │ │ │ │ │ + multipoint: function(geometry) { │ │ │ │ │ + return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.linestring │ │ │ │ │ + * Given an OpenLayers linestring geometry, create a KML linestring. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.LineString>} A linestring geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML linestring node. │ │ │ │ │ + */ │ │ │ │ │ + linestring: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "LineString"); │ │ │ │ │ + kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + return kml; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.multilinestring │ │ │ │ │ + * Given an OpenLayers multilinestring geometry, create a KML │ │ │ │ │ + * GeometryCollection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Point>} A multilinestring geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ + */ │ │ │ │ │ + multilinestring: function(geometry) { │ │ │ │ │ + return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.linearring │ │ │ │ │ + * Given an OpenLayers linearring geometry, create a KML linearring. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.LinearRing>} A linearring geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML linearring node. │ │ │ │ │ + */ │ │ │ │ │ + linearring: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "LinearRing"); │ │ │ │ │ + kml.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + return kml; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.polygon │ │ │ │ │ + * Given an OpenLayers polygon geometry, create a KML polygon. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Polygon>} A polygon geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML polygon node. │ │ │ │ │ + */ │ │ │ │ │ + polygon: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "Polygon"); │ │ │ │ │ + var rings = geometry.components; │ │ │ │ │ + var ringMember, ringGeom, type; │ │ │ │ │ + for (var i = 0, len = rings.length; i < len; ++i) { │ │ │ │ │ + type = (i == 0) ? "outerBoundaryIs" : "innerBoundaryIs"; │ │ │ │ │ + ringMember = this.createElementNS(this.kmlns, type); │ │ │ │ │ + ringGeom = this.buildGeometry.linearring.apply(this, │ │ │ │ │ + [rings[i]]); │ │ │ │ │ + ringMember.appendChild(ringGeom); │ │ │ │ │ + kml.appendChild(ringMember); │ │ │ │ │ + } │ │ │ │ │ + return kml; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.multipolygon │ │ │ │ │ + * Given an OpenLayers multipolygon geometry, create a KML │ │ │ │ │ + * GeometryCollection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Point>} A multipolygon geometry. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML GeometryCollection node. │ │ │ │ │ + */ │ │ │ │ │ + multipolygon: function(geometry) { │ │ │ │ │ + return this.buildGeometry.collection.apply(this, [geometry]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometry.collection │ │ │ │ │ + * Given an OpenLayers geometry collection, create a KML MultiGeometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry.Collection>} A geometry collection. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A KML MultiGeometry node. │ │ │ │ │ + */ │ │ │ │ │ + collection: function(geometry) { │ │ │ │ │ + var kml = this.createElementNS(this.kmlns, "MultiGeometry"); │ │ │ │ │ + var child; │ │ │ │ │ + for (var i = 0, len = geometry.components.length; i < len; ++i) { │ │ │ │ │ + child = this.buildGeometryNode.apply(this, │ │ │ │ │ + [geometry.components[i]]); │ │ │ │ │ + if (child) { │ │ │ │ │ + kml.appendChild(child); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return kml; │ │ │ │ │ } │ │ │ │ │ - this.layer = null; │ │ │ │ │ - return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroyFeature │ │ │ │ │ - * Destroy the temporary geometries │ │ │ │ │ - * │ │ │ │ │ + * Method: buildCoordinatesNode │ │ │ │ │ + * Builds and returns the KML coordinates node with the given geometry │ │ │ │ │ + * <coordinates>...</coordinates> │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - if (this.layer && (force || !this.persist)) { │ │ │ │ │ - this.layer.destroyFeatures(); │ │ │ │ │ - } │ │ │ │ │ - this.point = null; │ │ │ │ │ - }, │ │ │ │ │ + buildCoordinatesNode: function(geometry) { │ │ │ │ │ + var coordinatesNode = this.createElementNS(this.kmlns, "coordinates"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroyPersistedFeature │ │ │ │ │ - * Destroy the persisted feature. │ │ │ │ │ - */ │ │ │ │ │ - destroyPersistedFeature: function() { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - if (layer && layer.features.length > 1) { │ │ │ │ │ - this.layer.features[0].destroy(); │ │ │ │ │ + var path; │ │ │ │ │ + var points = geometry.components; │ │ │ │ │ + if (points) { │ │ │ │ │ + // LineString or LinearRing │ │ │ │ │ + var point; │ │ │ │ │ + var numPoints = points.length; │ │ │ │ │ + var parts = new Array(numPoints); │ │ │ │ │ + for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ + point = points[i]; │ │ │ │ │ + parts[i] = this.buildCoordinates(point); │ │ │ │ │ + } │ │ │ │ │ + path = parts.join(" "); │ │ │ │ │ + } else { │ │ │ │ │ + // Point │ │ │ │ │ + path = this.buildCoordinates(geometry); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: finalize │ │ │ │ │ - * Finish the geometry and call the "done" callback. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * cancel - {Boolean} Call cancel instead of done callback. Default │ │ │ │ │ - * is false. │ │ │ │ │ - */ │ │ │ │ │ - finalize: function(cancel) { │ │ │ │ │ - var key = cancel ? "cancel" : "done"; │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - this.lastDown = null; │ │ │ │ │ - this.lastUp = null; │ │ │ │ │ - this.lastTouchPx = null; │ │ │ │ │ - this.callback(key, [this.geometryClone()]); │ │ │ │ │ - this.destroyFeature(cancel); │ │ │ │ │ - }, │ │ │ │ │ + var txtNode = this.createTextNode(path); │ │ │ │ │ + coordinatesNode.appendChild(txtNode); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: cancel │ │ │ │ │ - * Finish the geometry and call the "cancel" callback. │ │ │ │ │ - */ │ │ │ │ │ - cancel: function() { │ │ │ │ │ - this.finalize(true); │ │ │ │ │ + return coordinatesNode; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: click │ │ │ │ │ - * Handle clicks. Clicks are stopped from propagating to other listeners │ │ │ │ │ - * on map.events or other dom elements. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ + * Method: buildCoordinates │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: dblclick │ │ │ │ │ - * Handle double-clicks. Double-clicks are stopped from propagating to other │ │ │ │ │ - * listeners on map.events or other dom elements. │ │ │ │ │ - * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * Returns │ │ │ │ │ + * {String} a coordinate pair │ │ │ │ │ */ │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - return false; │ │ │ │ │ + buildCoordinates: function(point) { │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + point = point.clone(); │ │ │ │ │ + point.transform(this.internalProjection, │ │ │ │ │ + this.externalProjection); │ │ │ │ │ + } │ │ │ │ │ + return point.x + "," + point.y; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: modifyFeature │ │ │ │ │ - * Modify the existing geometry given a pixel location. │ │ │ │ │ + * Method: buildExtendedData │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} A pixel location on the map. │ │ │ │ │ + * attributes - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns │ │ │ │ │ + * {DOMElement} A KML ExtendedData node or {null} if no attributes. │ │ │ │ │ */ │ │ │ │ │ - modifyFeature: function(pixel) { │ │ │ │ │ - if (!this.point) { │ │ │ │ │ - this.createFeature(pixel); │ │ │ │ │ + buildExtendedData: function(attributes) { │ │ │ │ │ + var extendedData = this.createElementNS(this.kmlns, "ExtendedData"); │ │ │ │ │ + for (var attributeName in attributes) { │ │ │ │ │ + // empty, name, description, styleUrl attributes ignored │ │ │ │ │ + if (attributes[attributeName] && attributeName != "name" && attributeName != "description" && attributeName != "styleUrl") { │ │ │ │ │ + var data = this.createElementNS(this.kmlns, "Data"); │ │ │ │ │ + data.setAttribute("name", attributeName); │ │ │ │ │ + var value = this.createElementNS(this.kmlns, "value"); │ │ │ │ │ + if (typeof attributes[attributeName] == "object") { │ │ │ │ │ + // cater for object attributes with 'value' properties │ │ │ │ │ + // other object properties will output an empty node │ │ │ │ │ + if (attributes[attributeName].value) { │ │ │ │ │ + value.appendChild(this.createTextNode(attributes[attributeName].value)); │ │ │ │ │ + } │ │ │ │ │ + if (attributes[attributeName].displayName) { │ │ │ │ │ + var displayName = this.createElementNS(this.kmlns, "displayName"); │ │ │ │ │ + // displayName always written as CDATA │ │ │ │ │ + displayName.appendChild(this.getXMLDoc().createCDATASection(attributes[attributeName].displayName)); │ │ │ │ │ + data.appendChild(displayName); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + value.appendChild(this.createTextNode(attributes[attributeName])); │ │ │ │ │ + } │ │ │ │ │ + data.appendChild(value); │ │ │ │ │ + extendedData.appendChild(data); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.isSimpleContent(extendedData)) { │ │ │ │ │ + return null; │ │ │ │ │ + } else { │ │ │ │ │ + return extendedData; │ │ │ │ │ } │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point.geometry.x = lonlat.lon; │ │ │ │ │ - this.point.geometry.y = lonlat.lat; │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.point, false]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawFeature │ │ │ │ │ - * Render features on the temporary layer. │ │ │ │ │ - */ │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.point, this.style); │ │ │ │ │ - }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.KML" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/SLD.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getGeometry │ │ │ │ │ - * Return the sketch geometry. If <multi> is true, this will return │ │ │ │ │ - * a multi-part geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} │ │ │ │ │ - */ │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.point && this.point.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiPoint([geometry]); │ │ │ │ │ - } │ │ │ │ │ - return geometry; │ │ │ │ │ - }, │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: geometryClone │ │ │ │ │ - * Return a clone of the relevant geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry>} │ │ │ │ │ - */ │ │ │ │ │ - geometryClone: function() { │ │ │ │ │ - var geom = this.getGeometry(); │ │ │ │ │ - return geom && geom.clone(); │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ + * @requires OpenLayers/Style.js │ │ │ │ │ + * @requires OpenLayers/Rule.js │ │ │ │ │ + * @requires OpenLayers/Filter/FeatureId.js │ │ │ │ │ + * @requires OpenLayers/Filter/Logical.js │ │ │ │ │ + * @requires OpenLayers/Filter/Comparison.js │ │ │ │ │ + * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: mousedown │ │ │ │ │ - * Handle mousedown. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - return this.down(evt); │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.SLD │ │ │ │ │ + * Read/Write SLD. Create a new instance with the <OpenLayers.Format.SLD> │ │ │ │ │ + * constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.SLD = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchstart │ │ │ │ │ - * Handle touchstart. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ + * APIProperty: profile │ │ │ │ │ + * {String} If provided, use a custom profile. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * Currently supported profiles: │ │ │ │ │ + * - GeoServer - parses GeoServer vendor specific capabilities for SLD. │ │ │ │ │ */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - this.lastTouchPx = evt.xy; │ │ │ │ │ - return this.down(evt); │ │ │ │ │ - }, │ │ │ │ │ + profile: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mousemove │ │ │ │ │ - * Handle mousemove. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.0.0". │ │ │ │ │ */ │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - return this.move(evt); │ │ │ │ │ - }, │ │ │ │ │ + defaultVersion: "1.0.0", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchmove │ │ │ │ │ - * Handle touchmove. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * APIProperty: stringifyOutput │ │ │ │ │ + * {Boolean} If true, write will return a string otherwise a DOMElement. │ │ │ │ │ + * Default is true. │ │ │ │ │ */ │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - this.lastTouchPx = evt.xy; │ │ │ │ │ - return this.move(evt); │ │ │ │ │ - }, │ │ │ │ │ + stringifyOutput: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mouseup │ │ │ │ │ - * Handle mouseup. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * APIProperty: namedLayersAsArray │ │ │ │ │ + * {Boolean} Generate a namedLayers array. If false, the namedLayers │ │ │ │ │ + * property value will be an object keyed by layer name. Default is │ │ │ │ │ + * false. │ │ │ │ │ */ │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - return this.up(evt); │ │ │ │ │ - }, │ │ │ │ │ + namedLayersAsArray: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchend │ │ │ │ │ - * Handle touchend. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Write a SLD document given a list of styles. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ + * sld - {Object} An object representing the SLD. │ │ │ │ │ + * options - {Object} Optional configuration object. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} An SLD document string. │ │ │ │ │ */ │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - evt.xy = this.lastTouchPx; │ │ │ │ │ - return this.up(evt); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: down │ │ │ │ │ - * Handle mousedown and touchstart. Adjust the geometry and redraw. │ │ │ │ │ - * Return determines whether to propagate the event on the map. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read and SLD doc and return an object representing the SLD. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ + * data - {String | DOMElement} Data to read. │ │ │ │ │ + * options - {Object} Options for the reader. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object representing the SLD. │ │ │ │ │ */ │ │ │ │ │ - down: function(evt) { │ │ │ │ │ - this.mouseDown = true; │ │ │ │ │ - this.lastDown = evt.xy; │ │ │ │ │ - if (!this.touch) { // no point displayed until up on touch devices │ │ │ │ │ - this.modifyFeature(evt.xy); │ │ │ │ │ - } │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - return !this.stopDown; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * Handle mousemove and touchmove. Adjust the geometry and redraw. │ │ │ │ │ - * Return determines whether to propagate the event on the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SLD" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/QueryStringFilter.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Console.js │ │ │ │ │ + * @requires OpenLayers/Format.js │ │ │ │ │ + * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ + * @requires OpenLayers/Filter/Comparison.js │ │ │ │ │ + * @requires OpenLayers/Filter/Logical.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.QueryStringFilter │ │ │ │ │ + * Parser for reading a query string and creating a simple filter. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.QueryStringFilter = (function() { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Map the OpenLayers.Filter.Comparison types to the operation strings of │ │ │ │ │ + * the protocol. │ │ │ │ │ */ │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - if (!this.touch // no point displayed until up on touch devices │ │ │ │ │ - && │ │ │ │ │ - (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ - this.modifyFeature(evt.xy); │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ + var cmpToStr = {}; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.EQUAL_TO] = "eq"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.NOT_EQUAL_TO] = "ne"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.LESS_THAN] = "lt"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO] = "lte"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.GREATER_THAN] = "gt"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO] = "gte"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.LIKE] = "ilike"; │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: up │ │ │ │ │ - * Handle mouseup and touchend. Send the latest point in the geometry to the control. │ │ │ │ │ - * Return determines whether to propagate the event on the map. │ │ │ │ │ + * Function: regex2value │ │ │ │ │ + * Convert the value from a regular expression string to a LIKE/ILIKE │ │ │ │ │ + * string known to the web service. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ + * value - {String} The regex string. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The converted string. │ │ │ │ │ */ │ │ │ │ │ - up: function(evt) { │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ + function regex2value(value) { │ │ │ │ │ │ │ │ │ │ - // check keyboard modifiers │ │ │ │ │ - if (!this.checkModifiers(evt)) { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - // ignore double-clicks │ │ │ │ │ - if (this.lastUp && this.lastUp.equals(evt.xy)) { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - if (this.lastDown && this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ - this.pixelTolerance)) { │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy); │ │ │ │ │ - } │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature(); │ │ │ │ │ - } │ │ │ │ │ - this.lastUp = evt.xy; │ │ │ │ │ - this.finalize(); │ │ │ │ │ - return !this.stopUp; │ │ │ │ │ - } else { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + // highly sensitive!! Do not change this without running the │ │ │ │ │ + // Protocol/HTTP.html unit tests │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: mouseout │ │ │ │ │ - * Handle mouse out. For better user experience reset mouseDown │ │ │ │ │ - * and stoppedDown when the mouse leaves the map viewport. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - */ │ │ │ │ │ - mouseout: function(evt) { │ │ │ │ │ - if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + // convert % to \% │ │ │ │ │ + value = value.replace(/%/g, "\\%"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: passesTolerance │ │ │ │ │ - * Determine whether the event is within the optional pixel tolerance. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The event is within the pixel tolerance (if specified). │ │ │ │ │ - */ │ │ │ │ │ - passesTolerance: function(pixel1, pixel2, tolerance) { │ │ │ │ │ - var passes = true; │ │ │ │ │ + // convert \\. to \\_ (\\.* occurences converted later) │ │ │ │ │ + value = value.replace(/\\\\\.(\*)?/g, function($0, $1) { │ │ │ │ │ + return $1 ? $0 : "\\\\_"; │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - if (tolerance != null && pixel1 && pixel2) { │ │ │ │ │ - var dist = pixel1.distanceTo(pixel2); │ │ │ │ │ - if (dist > tolerance) { │ │ │ │ │ - passes = false; │ │ │ │ │ + // convert \\.* to \\% │ │ │ │ │ + value = value.replace(/\\\\\.\*/g, "\\\\%"); │ │ │ │ │ + │ │ │ │ │ + // convert . to _ (\. and .* occurences converted later) │ │ │ │ │ + value = value.replace(/(\\)?\.(\*)?/g, function($0, $1, $2) { │ │ │ │ │ + return $1 || $2 ? $0 : "_"; │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + // convert .* to % (\.* occurnces converted later) │ │ │ │ │ + value = value.replace(/(\\)?\.\*/g, function($0, $1) { │ │ │ │ │ + return $1 ? $0 : "%"; │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + // convert \. to . │ │ │ │ │ + value = value.replace(/\\\./g, "."); │ │ │ │ │ + │ │ │ │ │ + // replace \* with * (watching out for \\*) │ │ │ │ │ + value = value.replace(/(\\)?\\\*/g, function($0, $1) { │ │ │ │ │ + return $1 ? $0 : "*"; │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + return value; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: wildcarded. │ │ │ │ │ + * {Boolean} If true percent signs are added around values │ │ │ │ │ + * read from LIKE filters, for example if the protocol │ │ │ │ │ + * read method is passed a LIKE filter whose property │ │ │ │ │ + * is "foo" and whose value is "bar" the string │ │ │ │ │ + * "foo__ilike=%bar%" will be sent in the query string; │ │ │ │ │ + * defaults to false. │ │ │ │ │ + */ │ │ │ │ │ + wildcarded: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: srsInBBOX │ │ │ │ │ + * {Boolean} Include the SRS identifier in BBOX query string parameter. │ │ │ │ │ + * Default is false. If true and the layer has a projection object set, │ │ │ │ │ + * any BBOX filter will be serialized with a fifth item identifying the │ │ │ │ │ + * projection. E.g. bbox=-1000,-1000,1000,1000,EPSG:900913 │ │ │ │ │ + */ │ │ │ │ │ + srsInBBOX: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Serialize an <OpenLayers.Filter> objects using the "simple" filter syntax for │ │ │ │ │ + * query string parameters. This function must be called as a method of │ │ │ │ │ + * a protocol instance. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {<OpenLayers.Filter>} filter to convert. │ │ │ │ │ + * params - {Object} The parameters object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The resulting parameters object. │ │ │ │ │ + */ │ │ │ │ │ + write: function(filter, params) { │ │ │ │ │ + params = params || {}; │ │ │ │ │ + var className = filter.CLASS_NAME; │ │ │ │ │ + var filterType = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ + switch (filterType) { │ │ │ │ │ + case "Spatial": │ │ │ │ │ + switch (filter.type) { │ │ │ │ │ + case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ + params.bbox = filter.value.toArray(); │ │ │ │ │ + if (this.srsInBBOX && filter.projection) { │ │ │ │ │ + params.bbox.push(filter.projection.getCode()); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Spatial.DWITHIN: │ │ │ │ │ + params.tolerance = filter.distance; │ │ │ │ │ + // no break here │ │ │ │ │ + case OpenLayers.Filter.Spatial.WITHIN: │ │ │ │ │ + params.lon = filter.value.x; │ │ │ │ │ + params.lat = filter.value.y; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + OpenLayers.Console.warn( │ │ │ │ │ + "Unknown spatial filter type " + filter.type); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "Comparison": │ │ │ │ │ + var op = cmpToStr[filter.type]; │ │ │ │ │ + if (op !== undefined) { │ │ │ │ │ + var value = filter.value; │ │ │ │ │ + if (filter.type == OpenLayers.Filter.Comparison.LIKE) { │ │ │ │ │ + value = regex2value(value); │ │ │ │ │ + if (this.wildcarded) { │ │ │ │ │ + value = "%" + value + "%"; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + params[filter.property + "__" + op] = value; │ │ │ │ │ + params.queryable = params.queryable || []; │ │ │ │ │ + params.queryable.push(filter.property); │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Console.warn( │ │ │ │ │ + "Unknown comparison filter type " + filter.type); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "Logical": │ │ │ │ │ + if (filter.type === OpenLayers.Filter.Logical.AND) { │ │ │ │ │ + for (var i = 0, len = filter.filters.length; i < len; i++) { │ │ │ │ │ + params = this.write(filter.filters[i], params); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Console.warn( │ │ │ │ │ + "Unsupported logical filter type " + filter.type); │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + OpenLayers.Console.warn("Unknown filter type " + filterType); │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return passes; │ │ │ │ │ - }, │ │ │ │ │ + return params; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Point" │ │ │ │ │ -}); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.QueryStringFilter" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +})(); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Feature.js │ │ │ │ │ + OpenLayers/Format/GeoRSS.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Handler.Feature │ │ │ │ │ - * Handler to respond to mouse events related to a drawn feature. Callbacks │ │ │ │ │ - * with the following keys will be notified of the following events │ │ │ │ │ - * associated with features: click, clickout, over, out, and dblclick. │ │ │ │ │ - * │ │ │ │ │ - * This handler stops event propagation for mousedown and mouseup if those │ │ │ │ │ - * browser events target features that can be selected. │ │ │ │ │ + * Class: OpenLayers.Format.GeoRSS │ │ │ │ │ + * Read/write GeoRSS parser. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Format.GeoRSS> constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.Feature = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: EVENTMAP │ │ │ │ │ - * {Object} A object mapping the browser events to objects with callback │ │ │ │ │ - * keys for in and out. │ │ │ │ │ - */ │ │ │ │ │ - EVENTMAP: { │ │ │ │ │ - 'click': { │ │ │ │ │ - 'in': 'click', │ │ │ │ │ - 'out': 'clickout' │ │ │ │ │ - }, │ │ │ │ │ - 'mousemove': { │ │ │ │ │ - 'in': 'over', │ │ │ │ │ - 'out': 'out' │ │ │ │ │ - }, │ │ │ │ │ - 'dblclick': { │ │ │ │ │ - 'in': 'dblclick', │ │ │ │ │ - 'out': null │ │ │ │ │ - }, │ │ │ │ │ - 'mousedown': { │ │ │ │ │ - 'in': null, │ │ │ │ │ - 'out': null │ │ │ │ │ - }, │ │ │ │ │ - 'mouseup': { │ │ │ │ │ - 'in': null, │ │ │ │ │ - 'out': null │ │ │ │ │ - }, │ │ │ │ │ - 'touchstart': { │ │ │ │ │ - 'in': 'click', │ │ │ │ │ - 'out': 'clickout' │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: feature │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} The last feature that was hovered. │ │ │ │ │ - */ │ │ │ │ │ - feature: null, │ │ │ │ │ +OpenLayers.Format.GeoRSS = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: lastFeature │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} The last feature that was handled. │ │ │ │ │ + * APIProperty: rssns │ │ │ │ │ + * {String} RSS namespace to use. Defaults to │ │ │ │ │ + * "http://backend.userland.com/rss2" │ │ │ │ │ */ │ │ │ │ │ - lastFeature: null, │ │ │ │ │ + rssns: "http://backend.userland.com/rss2", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: down │ │ │ │ │ - * {<OpenLayers.Pixel>} The location of the last mousedown. │ │ │ │ │ + * APIProperty: featurens │ │ │ │ │ + * {String} Feature Attributes namespace. Defaults to │ │ │ │ │ + * "http://mapserver.gis.umn.edu/mapserver" │ │ │ │ │ */ │ │ │ │ │ - down: null, │ │ │ │ │ + featureNS: "http://mapserver.gis.umn.edu/mapserver", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: up │ │ │ │ │ - * {<OpenLayers.Pixel>} The location of the last mouseup. │ │ │ │ │ + * APIProperty: georssns │ │ │ │ │ + * {String} GeoRSS namespace to use. Defaults to │ │ │ │ │ + * "http://www.georss.org/georss" │ │ │ │ │ */ │ │ │ │ │ - up: null, │ │ │ │ │ + georssns: "http://www.georss.org/georss", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: clickTolerance │ │ │ │ │ - * {Number} The number of pixels the mouse can move between mousedown │ │ │ │ │ - * and mouseup for the event to still be considered a click. │ │ │ │ │ - * Dragging the map should not trigger the click and clickout callbacks │ │ │ │ │ - * unless the map is moved by less than this tolerance. Defaults to 4. │ │ │ │ │ + * APIProperty: geons │ │ │ │ │ + * {String} W3C Geo namespace to use. Defaults to │ │ │ │ │ + * "http://www.w3.org/2003/01/geo/wgs84_pos#" │ │ │ │ │ */ │ │ │ │ │ - clickTolerance: 4, │ │ │ │ │ + geons: "http://www.w3.org/2003/01/geo/wgs84_pos#", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: geometryTypes │ │ │ │ │ - * To restrict dragging to a limited set of geometry types, send a list │ │ │ │ │ - * of strings corresponding to the geometry class names. │ │ │ │ │ - * │ │ │ │ │ - * @type Array(String) │ │ │ │ │ + * APIProperty: featureTitle │ │ │ │ │ + * {String} Default title for features. Defaults to "Untitled" │ │ │ │ │ */ │ │ │ │ │ - geometryTypes: null, │ │ │ │ │ + featureTitle: "Untitled", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: stopClick │ │ │ │ │ - * {Boolean} If stopClick is set to true, handled clicks do not │ │ │ │ │ - * propagate to other click listeners. Otherwise, handled clicks │ │ │ │ │ - * do propagate. Unhandled clicks always propagate, whatever the │ │ │ │ │ - * value of stopClick. Defaults to true. │ │ │ │ │ + * APIProperty: featureDescription │ │ │ │ │ + * {String} Default description for features. Defaults to "No Description" │ │ │ │ │ */ │ │ │ │ │ - stopClick: true, │ │ │ │ │ + featureDescription: "No Description", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: stopDown │ │ │ │ │ - * {Boolean} If stopDown is set to true, handled mousedowns do not │ │ │ │ │ - * propagate to other mousedown listeners. Otherwise, handled │ │ │ │ │ - * mousedowns do propagate. Unhandled mousedowns always propagate, │ │ │ │ │ - * whatever the value of stopDown. Defaults to true. │ │ │ │ │ + * Property: gmlParse │ │ │ │ │ + * {Object} GML Format object for parsing features │ │ │ │ │ + * Non-API and only created if necessary │ │ │ │ │ */ │ │ │ │ │ - stopDown: true, │ │ │ │ │ + gmlParser: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: stopUp │ │ │ │ │ - * {Boolean} If stopUp is set to true, handled mouseups do not │ │ │ │ │ - * propagate to other mouseup listeners. Otherwise, handled mouseups │ │ │ │ │ - * do propagate. Unhandled mouseups always propagate, whatever the │ │ │ │ │ - * value of stopUp. Defaults to false. │ │ │ │ │ + * APIProperty: xy │ │ │ │ │ + * {Boolean} Order of the GML coordinate: true:(x,y) or false:(y,x) │ │ │ │ │ + * For GeoRSS the default is (y,x), therefore: false │ │ │ │ │ */ │ │ │ │ │ - stopUp: false, │ │ │ │ │ + xy: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Feature │ │ │ │ │ + * Constructor: OpenLayers.Format.GeoRSS │ │ │ │ │ + * Create a new parser for GeoRSS. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} │ │ │ │ │ - * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ - * callbacks - {Object} An object with a 'over' property whos value is │ │ │ │ │ - * a function to be called when the mouse is over a feature. The │ │ │ │ │ - * callback should expect to recieve a single argument, the feature. │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(control, layer, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchstart │ │ │ │ │ - * Handle touchstart events │ │ │ │ │ + * Method: createGeometryFromItem │ │ │ │ │ + * Return a geometry from a GeoRSS Item. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * item - {DOMElement} A GeoRSS item node. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * {<OpenLayers.Geometry>} A geometry representing the node. │ │ │ │ │ */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - return OpenLayers.Event.isMultiTouch(evt) ? │ │ │ │ │ - true : this.mousedown(evt); │ │ │ │ │ - }, │ │ │ │ │ + createGeometryFromItem: function(item) { │ │ │ │ │ + var point = this.getElementsByTagNameNS(item, this.georssns, "point"); │ │ │ │ │ + var lat = this.getElementsByTagNameNS(item, this.geons, 'lat'); │ │ │ │ │ + var lon = this.getElementsByTagNameNS(item, this.geons, 'long'); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchmove │ │ │ │ │ - * Handle touchmove events. We just prevent the browser default behavior, │ │ │ │ │ - * for Android Webkit not to select text when moving the finger after │ │ │ │ │ - * selecting a feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ - }, │ │ │ │ │ + var line = this.getElementsByTagNameNS(item, │ │ │ │ │ + this.georssns, │ │ │ │ │ + "line"); │ │ │ │ │ + var polygon = this.getElementsByTagNameNS(item, │ │ │ │ │ + this.georssns, │ │ │ │ │ + "polygon"); │ │ │ │ │ + var where = this.getElementsByTagNameNS(item, │ │ │ │ │ + this.georssns, │ │ │ │ │ + "where"); │ │ │ │ │ + var box = this.getElementsByTagNameNS(item, │ │ │ │ │ + this.georssns, │ │ │ │ │ + "box"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: mousedown │ │ │ │ │ - * Handle mouse down. Stop propagation if a feature is targeted by this │ │ │ │ │ - * event (stops map dragging during feature selection). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - // Feature selection is only done with a left click. Other handlers may stop the │ │ │ │ │ - // propagation of left-click mousedown events but not right-click mousedown events. │ │ │ │ │ - // This mismatch causes problems when comparing the location of the down and up │ │ │ │ │ - // events in the click function so it is important ignore right-clicks. │ │ │ │ │ - if (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ - this.down = evt.xy; │ │ │ │ │ - } │ │ │ │ │ - return this.handle(evt) ? !this.stopDown : true; │ │ │ │ │ - }, │ │ │ │ │ + if (point.length > 0 || (lat.length > 0 && lon.length > 0)) { │ │ │ │ │ + var location; │ │ │ │ │ + if (point.length > 0) { │ │ │ │ │ + location = OpenLayers.String.trim( │ │ │ │ │ + point[0].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ + if (location.length != 2) { │ │ │ │ │ + location = OpenLayers.String.trim( │ │ │ │ │ + point[0].firstChild.nodeValue).split(/\s*,\s*/); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + location = [parseFloat(lat[0].firstChild.nodeValue), │ │ │ │ │ + parseFloat(lon[0].firstChild.nodeValue) │ │ │ │ │ + ]; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: mouseup │ │ │ │ │ - * Handle mouse up. Stop propagation if a feature is targeted by this │ │ │ │ │ - * event. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - this.up = evt.xy; │ │ │ │ │ - return this.handle(evt) ? !this.stopUp : true; │ │ │ │ │ - }, │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point(location[1], location[0]); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: click │ │ │ │ │ - * Handle click. Call the "click" callback if click on a feature, │ │ │ │ │ - * or the "clickout" callback if click outside any feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - return this.handle(evt) ? !this.stopClick : true; │ │ │ │ │ - }, │ │ │ │ │ + } else if (line.length > 0) { │ │ │ │ │ + var coords = OpenLayers.String.trim(this.getChildValue(line[0])).split(/\s+/); │ │ │ │ │ + var components = []; │ │ │ │ │ + var point; │ │ │ │ │ + for (var i = 0, len = coords.length; i < len; i += 2) { │ │ │ │ │ + point = new OpenLayers.Geometry.Point(coords[i + 1], coords[i]); │ │ │ │ │ + components.push(point); │ │ │ │ │ + } │ │ │ │ │ + geometry = new OpenLayers.Geometry.LineString(components); │ │ │ │ │ + } else if (polygon.length > 0) { │ │ │ │ │ + var coords = OpenLayers.String.trim(this.getChildValue(polygon[0])).split(/\s+/); │ │ │ │ │ + var components = []; │ │ │ │ │ + var point; │ │ │ │ │ + for (var i = 0, len = coords.length; i < len; i += 2) { │ │ │ │ │ + point = new OpenLayers.Geometry.Point(coords[i + 1], coords[i]); │ │ │ │ │ + components.push(point); │ │ │ │ │ + } │ │ │ │ │ + geometry = new OpenLayers.Geometry.Polygon([new OpenLayers.Geometry.LinearRing(components)]); │ │ │ │ │ + } else if (where.length > 0) { │ │ │ │ │ + if (!this.gmlParser) { │ │ │ │ │ + this.gmlParser = new OpenLayers.Format.GML({ │ │ │ │ │ + 'xy': this.xy │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + var feature = this.gmlParser.parseFeature(where[0]); │ │ │ │ │ + geometry = feature.geometry; │ │ │ │ │ + } else if (box.length > 0) { │ │ │ │ │ + var coords = OpenLayers.String.trim(box[0].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ + var components = []; │ │ │ │ │ + var point; │ │ │ │ │ + if (coords.length > 3) { │ │ │ │ │ + point = new OpenLayers.Geometry.Point(coords[1], coords[0]); │ │ │ │ │ + components.push(point); │ │ │ │ │ + point = new OpenLayers.Geometry.Point(coords[1], coords[2]); │ │ │ │ │ + components.push(point); │ │ │ │ │ + point = new OpenLayers.Geometry.Point(coords[3], coords[2]); │ │ │ │ │ + components.push(point); │ │ │ │ │ + point = new OpenLayers.Geometry.Point(coords[3], coords[0]); │ │ │ │ │ + components.push(point); │ │ │ │ │ + point = new OpenLayers.Geometry.Point(coords[1], coords[0]); │ │ │ │ │ + components.push(point); │ │ │ │ │ + } │ │ │ │ │ + geometry = new OpenLayers.Geometry.Polygon([new OpenLayers.Geometry.LinearRing(components)]); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: mousemove │ │ │ │ │ - * Handle mouse moves. Call the "over" callback if moving in to a feature, │ │ │ │ │ - * or the "out" callback if moving out of a feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - if (!this.callbacks['over'] && !this.callbacks['out']) { │ │ │ │ │ - return true; │ │ │ │ │ + if (geometry && this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry.transform(this.externalProjection, │ │ │ │ │ + this.internalProjection); │ │ │ │ │ } │ │ │ │ │ - this.handle(evt); │ │ │ │ │ - return true; │ │ │ │ │ + │ │ │ │ │ + return geometry; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: dblclick │ │ │ │ │ - * Handle dblclick. Call the "dblclick" callback if dblclick on a feature. │ │ │ │ │ + * Method: createFeatureFromItem │ │ │ │ │ + * Return a feature from a GeoRSS Item. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * item - {DOMElement} A GeoRSS item node. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A feature representing the item. │ │ │ │ │ */ │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - return !this.handle(evt); │ │ │ │ │ + createFeatureFromItem: function(item) { │ │ │ │ │ + var geometry = this.createGeometryFromItem(item); │ │ │ │ │ + │ │ │ │ │ + /* Provide defaults for title and description */ │ │ │ │ │ + var title = this._getChildValue(item, "*", "title", this.featureTitle); │ │ │ │ │ + │ │ │ │ │ + /* First try RSS descriptions, then Atom summaries */ │ │ │ │ │ + var description = this._getChildValue( │ │ │ │ │ + item, "*", "description", │ │ │ │ │ + this._getChildValue(item, "*", "content", │ │ │ │ │ + this._getChildValue(item, "*", "summary", this.featureDescription))); │ │ │ │ │ + │ │ │ │ │ + /* If no link URL is found in the first child node, try the │ │ │ │ │ + href attribute */ │ │ │ │ │ + var link = this._getChildValue(item, "*", "link"); │ │ │ │ │ + if (!link) { │ │ │ │ │ + try { │ │ │ │ │ + link = this.getElementsByTagNameNS(item, "*", "link")[0].getAttribute("href"); │ │ │ │ │ + } catch (e) { │ │ │ │ │ + link = null; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var id = this._getChildValue(item, "*", "id", null); │ │ │ │ │ + │ │ │ │ │ + var data = { │ │ │ │ │ + "title": title, │ │ │ │ │ + "description": description, │ │ │ │ │ + "link": link │ │ │ │ │ + }; │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry, data); │ │ │ │ │ + feature.fid = id; │ │ │ │ │ + return feature; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: geometryTypeMatches │ │ │ │ │ - * Return true if the geometry type of the passed feature matches │ │ │ │ │ - * one of the geometry types in the geometryTypes array. │ │ │ │ │ + * Method: _getChildValue │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Vector.Feature>} │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * nsuri - {String} Child node namespace uri ("*" for any). │ │ │ │ │ + * name - {String} Child node name. │ │ │ │ │ + * def - {String} Optional string default to return if no child found. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * {String} The value of the first child with the given tag name. Returns │ │ │ │ │ + * default value or empty string if none found. │ │ │ │ │ */ │ │ │ │ │ - geometryTypeMatches: function(feature) { │ │ │ │ │ - return this.geometryTypes == null || │ │ │ │ │ - OpenLayers.Util.indexOf(this.geometryTypes, │ │ │ │ │ - feature.geometry.CLASS_NAME) > -1; │ │ │ │ │ + _getChildValue: function(node, nsuri, name, def) { │ │ │ │ │ + var value; │ │ │ │ │ + var eles = this.getElementsByTagNameNS(node, nsuri, name); │ │ │ │ │ + if (eles && eles[0] && eles[0].firstChild && │ │ │ │ │ + eles[0].firstChild.nodeValue) { │ │ │ │ │ + value = this.getChildValue(eles[0]); │ │ │ │ │ + } else { │ │ │ │ │ + value = (def == undefined) ? "" : def; │ │ │ │ │ + } │ │ │ │ │ + return value; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handle │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Return a list of features from a GeoRSS doc │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * doc - {Element} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The event occurred over a relevant feature. │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ */ │ │ │ │ │ - handle: function(evt) { │ │ │ │ │ - if (this.feature && !this.feature.layer) { │ │ │ │ │ - // feature has been destroyed │ │ │ │ │ - this.feature = null; │ │ │ │ │ - } │ │ │ │ │ - var type = evt.type; │ │ │ │ │ - var handled = false; │ │ │ │ │ - var previouslyIn = !!(this.feature); // previously in a feature │ │ │ │ │ - var click = (type == "click" || type == "dblclick" || type == "touchstart"); │ │ │ │ │ - this.feature = this.layer.getFeatureFromEvent(evt); │ │ │ │ │ - if (this.feature && !this.feature.layer) { │ │ │ │ │ - // feature has been destroyed │ │ │ │ │ - this.feature = null; │ │ │ │ │ + read: function(doc) { │ │ │ │ │ + if (typeof doc == "string") { │ │ │ │ │ + doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]); │ │ │ │ │ } │ │ │ │ │ - if (this.lastFeature && !this.lastFeature.layer) { │ │ │ │ │ - // last feature has been destroyed │ │ │ │ │ - this.lastFeature = null; │ │ │ │ │ + │ │ │ │ │ + /* Try RSS items first, then Atom entries */ │ │ │ │ │ + var itemlist = null; │ │ │ │ │ + itemlist = this.getElementsByTagNameNS(doc, '*', 'item'); │ │ │ │ │ + if (itemlist.length == 0) { │ │ │ │ │ + itemlist = this.getElementsByTagNameNS(doc, '*', 'entry'); │ │ │ │ │ } │ │ │ │ │ - if (this.feature) { │ │ │ │ │ - if (type === "touchstart") { │ │ │ │ │ - // stop the event to prevent Android Webkit from │ │ │ │ │ - // "flashing" the map div │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ - } │ │ │ │ │ - var inNew = (this.feature != this.lastFeature); │ │ │ │ │ - if (this.geometryTypeMatches(this.feature)) { │ │ │ │ │ - // in to a feature │ │ │ │ │ - if (previouslyIn && inNew) { │ │ │ │ │ - // out of last feature and in to another │ │ │ │ │ - if (this.lastFeature) { │ │ │ │ │ - this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ - } │ │ │ │ │ - this.triggerCallback(type, 'in', [this.feature]); │ │ │ │ │ - } else if (!previouslyIn || click) { │ │ │ │ │ - // in feature for the first time │ │ │ │ │ - this.triggerCallback(type, 'in', [this.feature]); │ │ │ │ │ - } │ │ │ │ │ - this.lastFeature = this.feature; │ │ │ │ │ - handled = true; │ │ │ │ │ - } else { │ │ │ │ │ - // not in to a feature │ │ │ │ │ - if (this.lastFeature && (previouslyIn && inNew || click)) { │ │ │ │ │ - // out of last feature for the first time │ │ │ │ │ - this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ - } │ │ │ │ │ - // next time the mouse goes in a feature whose geometry type │ │ │ │ │ - // doesn't match we don't want to call the 'out' callback │ │ │ │ │ - // again, so let's set this.feature to null so that │ │ │ │ │ - // previouslyIn will evaluate to false the next time │ │ │ │ │ - // we enter handle. Yes, a bit hackish... │ │ │ │ │ - this.feature = null; │ │ │ │ │ - } │ │ │ │ │ - } else if (this.lastFeature && (previouslyIn || click)) { │ │ │ │ │ - this.triggerCallback(type, 'out', [this.lastFeature]); │ │ │ │ │ + │ │ │ │ │ + var numItems = itemlist.length; │ │ │ │ │ + var features = new Array(numItems); │ │ │ │ │ + for (var i = 0; i < numItems; i++) { │ │ │ │ │ + features[i] = this.createFeatureFromItem(itemlist[i]); │ │ │ │ │ } │ │ │ │ │ - return handled; │ │ │ │ │ + return features; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: triggerCallback │ │ │ │ │ - * Call the callback keyed in the event map with the supplied arguments. │ │ │ │ │ - * For click and clickout, the <clickTolerance> is checked first. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * type - {String} │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Accept Feature Collection, and return a string. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} List of features to serialize into a string. │ │ │ │ │ */ │ │ │ │ │ - triggerCallback: function(type, mode, args) { │ │ │ │ │ - var key = this.EVENTMAP[type][mode]; │ │ │ │ │ - if (key) { │ │ │ │ │ - if (type == 'click' && this.up && this.down) { │ │ │ │ │ - // for click/clickout, only trigger callback if tolerance is met │ │ │ │ │ - var dpx = Math.sqrt( │ │ │ │ │ - Math.pow(this.up.x - this.down.x, 2) + │ │ │ │ │ - Math.pow(this.up.y - this.down.y, 2) │ │ │ │ │ - ); │ │ │ │ │ - if (dpx <= this.clickTolerance) { │ │ │ │ │ - this.callback(key, args); │ │ │ │ │ - } │ │ │ │ │ - // we're done with this set of events now: clear the cached │ │ │ │ │ - // positions so we can't trip over them later (this can occur │ │ │ │ │ - // if one of the up/down events gets eaten before it gets to us │ │ │ │ │ - // but we still get the click) │ │ │ │ │ - this.up = this.down = null; │ │ │ │ │ - } else { │ │ │ │ │ - this.callback(key, args); │ │ │ │ │ + write: function(features) { │ │ │ │ │ + var georss; │ │ │ │ │ + if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ + georss = this.createElementNS(this.rssns, "rss"); │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + georss.appendChild(this.createFeatureXML(features[i])); │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + georss = this.createFeatureXML(features); │ │ │ │ │ } │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [georss]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Turn on the handler. Returns false if the handler was already active. │ │ │ │ │ + * Method: createFeatureXML │ │ │ │ │ + * Accept an <OpenLayers.Feature.Vector>, and build a geometry for it. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerToTop(); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - "removelayer": this.handleMapEvents, │ │ │ │ │ - "changelayer": this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - activated = true; │ │ │ │ │ + createFeatureXML: function(feature) { │ │ │ │ │ + var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ + var featureNode = this.createElementNS(this.rssns, "item"); │ │ │ │ │ + var titleNode = this.createElementNS(this.rssns, "title"); │ │ │ │ │ + titleNode.appendChild(this.createTextNode(feature.attributes.title ? feature.attributes.title : "")); │ │ │ │ │ + var descNode = this.createElementNS(this.rssns, "description"); │ │ │ │ │ + descNode.appendChild(this.createTextNode(feature.attributes.description ? feature.attributes.description : "")); │ │ │ │ │ + featureNode.appendChild(titleNode); │ │ │ │ │ + featureNode.appendChild(descNode); │ │ │ │ │ + if (feature.attributes.link) { │ │ │ │ │ + var linkNode = this.createElementNS(this.rssns, "link"); │ │ │ │ │ + linkNode.appendChild(this.createTextNode(feature.attributes.link)); │ │ │ │ │ + featureNode.appendChild(linkNode); │ │ │ │ │ } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Turn off the handler. Returns false if the handler was already active. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerBack(); │ │ │ │ │ - this.feature = null; │ │ │ │ │ - this.lastFeature = null; │ │ │ │ │ - this.down = null; │ │ │ │ │ - this.up = null; │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - "removelayer": this.handleMapEvents, │ │ │ │ │ - "changelayer": this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - deactivated = true; │ │ │ │ │ + for (var attr in feature.attributes) { │ │ │ │ │ + if (attr == "link" || attr == "title" || attr == "description") { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + var attrText = this.createTextNode(feature.attributes[attr]); │ │ │ │ │ + var nodename = attr; │ │ │ │ │ + if (attr.search(":") != -1) { │ │ │ │ │ + nodename = attr.split(":")[1]; │ │ │ │ │ + } │ │ │ │ │ + var attrContainer = this.createElementNS(this.featureNS, "feature:" + nodename); │ │ │ │ │ + attrContainer.appendChild(attrText); │ │ │ │ │ + featureNode.appendChild(attrContainer); │ │ │ │ │ } │ │ │ │ │ - return deactivated; │ │ │ │ │ + featureNode.appendChild(geometryNode); │ │ │ │ │ + return featureNode; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleMapEvents │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildGeometryNode │ │ │ │ │ + * builds a GeoRSS node with a given geometry │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A gml node. │ │ │ │ │ */ │ │ │ │ │ - handleMapEvents: function(evt) { │ │ │ │ │ - if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ - this.moveLayerToTop(); │ │ │ │ │ + buildGeometryNode: function(geometry) { │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry = geometry.clone(); │ │ │ │ │ + geometry.transform(this.internalProjection, │ │ │ │ │ + this.externalProjection); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ + var node; │ │ │ │ │ + // match Polygon │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.Polygon") { │ │ │ │ │ + node = this.createElementNS(this.georssns, 'georss:polygon'); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveLayerToTop │ │ │ │ │ - * Moves the layer for this handler to the top, so mouse events can reach │ │ │ │ │ - * it. │ │ │ │ │ - */ │ │ │ │ │ - moveLayerToTop: function() { │ │ │ │ │ - var index = Math.max(this.map.Z_INDEX_BASE['Feature'] - 1, │ │ │ │ │ - this.layer.getZIndex()) + 1; │ │ │ │ │ - this.layer.setZIndex(index); │ │ │ │ │ + node.appendChild(this.buildCoordinatesNode(geometry.components[0])); │ │ │ │ │ + } │ │ │ │ │ + // match LineString │ │ │ │ │ + else if (geometry.CLASS_NAME == "OpenLayers.Geometry.LineString") { │ │ │ │ │ + node = this.createElementNS(this.georssns, 'georss:line'); │ │ │ │ │ │ │ │ │ │ + node.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + } │ │ │ │ │ + // match Point │ │ │ │ │ + else if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + node = this.createElementNS(this.georssns, 'georss:point'); │ │ │ │ │ + node.appendChild(this.buildCoordinatesNode(geometry)); │ │ │ │ │ + } else { │ │ │ │ │ + throw "Couldn't parse " + geometry.CLASS_NAME; │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: moveLayerBack │ │ │ │ │ - * Moves the layer back to the position determined by the map's layers │ │ │ │ │ - * array. │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildCoordinatesNode │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ */ │ │ │ │ │ - moveLayerBack: function() { │ │ │ │ │ - var index = this.layer.getZIndex() - 1; │ │ │ │ │ - if (index >= this.map.Z_INDEX_BASE['Feature']) { │ │ │ │ │ - this.layer.setZIndex(index); │ │ │ │ │ + buildCoordinatesNode: function(geometry) { │ │ │ │ │ + var points = null; │ │ │ │ │ + │ │ │ │ │ + if (geometry.components) { │ │ │ │ │ + points = geometry.components; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var path; │ │ │ │ │ + if (points) { │ │ │ │ │ + var numPoints = points.length; │ │ │ │ │ + var parts = new Array(numPoints); │ │ │ │ │ + for (var i = 0; i < numPoints; i++) { │ │ │ │ │ + parts[i] = points[i].y + " " + points[i].x; │ │ │ │ │ + } │ │ │ │ │ + path = parts.join(" "); │ │ │ │ │ } else { │ │ │ │ │ - this.map.setLayerZIndex(this.layer, │ │ │ │ │ - this.map.getLayerIndex(this.layer)); │ │ │ │ │ + path = geometry.y + " " + geometry.x; │ │ │ │ │ } │ │ │ │ │ + return this.createTextNode(path); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Feature" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.GeoRSS" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Drag.js │ │ │ │ │ + OpenLayers/Format/WFSCapabilities.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Handler.Drag │ │ │ │ │ - * The drag handler is used to deal with sequences of browser events related │ │ │ │ │ - * to dragging. The handler is used by controls that want to know when │ │ │ │ │ - * a drag sequence begins, when a drag is happening, and when it has │ │ │ │ │ - * finished. │ │ │ │ │ - * │ │ │ │ │ - * Controls that use the drag handler typically construct it with callbacks │ │ │ │ │ - * for 'down', 'move', and 'done'. Callbacks for these keys are called │ │ │ │ │ - * when the drag begins, with each move, and when the drag is done. In │ │ │ │ │ - * addition, controls can have callbacks keyed to 'up' and 'out' if they │ │ │ │ │ - * care to differentiate between the types of events that correspond with │ │ │ │ │ - * the end of a drag sequence. If no drag actually occurs (no mouse move) │ │ │ │ │ - * the 'down' and 'up' callbacks will be called, but not the 'done' │ │ │ │ │ - * callback. │ │ │ │ │ - * │ │ │ │ │ - * Create a new drag handler with the <OpenLayers.Handler.Drag> constructor. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Format.WFSCapabilities │ │ │ │ │ + * Read WFS Capabilities. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.Drag = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ +OpenLayers.Format.WFSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: started │ │ │ │ │ - * {Boolean} When a mousedown or touchstart event is received, we want to │ │ │ │ │ - * record it, but not set 'dragging' until the mouse moves after starting. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.1.0". │ │ │ │ │ */ │ │ │ │ │ - started: false, │ │ │ │ │ + defaultVersion: "1.1.0", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: stopDown │ │ │ │ │ - * {Boolean} Stop propagation of mousedown events from getting to listeners │ │ │ │ │ - * on the same element. Default is true. │ │ │ │ │ + * Constructor: OpenLayers.Format.WFSCapabilities │ │ │ │ │ + * Create a new parser for WFS capabilities. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - stopDown: true, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: dragging │ │ │ │ │ - * {Boolean} │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} List of named layers. │ │ │ │ │ */ │ │ │ │ │ - dragging: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: last │ │ │ │ │ - * {<OpenLayers.Pixel>} The last pixel location of the drag. │ │ │ │ │ - */ │ │ │ │ │ - last: null, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSCapabilities" │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: start │ │ │ │ │ - * {<OpenLayers.Pixel>} The first pixel location of the drag. │ │ │ │ │ - */ │ │ │ │ │ - start: null, │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/Atom.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Format/GML/v3.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.Atom │ │ │ │ │ + * Read/write Atom feeds. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Format.AtomFeed> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.Atom = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: lastMoveEvt │ │ │ │ │ - * {Object} The last mousemove event that occurred. Used to │ │ │ │ │ - * position the map correctly when our "delay drag" │ │ │ │ │ - * timeout expired. │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. Properties │ │ │ │ │ + * of this object should not be set individually. Read-only. All │ │ │ │ │ + * XML subclasses should have their own namespaces object. Use │ │ │ │ │ + * <setNamespace> to add or set a namespace alias after construction. │ │ │ │ │ */ │ │ │ │ │ - lastMoveEvt: null, │ │ │ │ │ + namespaces: { │ │ │ │ │ + atom: "http://www.w3.org/2005/Atom", │ │ │ │ │ + georss: "http://www.georss.org/georss" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: oldOnselectstart │ │ │ │ │ - * {Function} │ │ │ │ │ + * APIProperty: feedTitle │ │ │ │ │ + * {String} Atom feed elements require a title. Default is "untitled". │ │ │ │ │ */ │ │ │ │ │ - oldOnselectstart: null, │ │ │ │ │ + feedTitle: "untitled", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: interval │ │ │ │ │ - * {Integer} In order to increase performance, an interval (in │ │ │ │ │ - * milliseconds) can be set to reduce the number of drag events │ │ │ │ │ - * called. If set, a new drag event will not be set until the │ │ │ │ │ - * interval has passed. │ │ │ │ │ - * Defaults to 0, meaning no interval. │ │ │ │ │ + * APIProperty: defaultEntryTitle │ │ │ │ │ + * {String} Atom entry elements require a title. In cases where one is │ │ │ │ │ + * not provided in the feature attributes, this will be used. Default │ │ │ │ │ + * is "untitled". │ │ │ │ │ */ │ │ │ │ │ - interval: 0, │ │ │ │ │ + defaultEntryTitle: "untitled", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: timeoutId │ │ │ │ │ - * {String} The id of the timeout used for the mousedown interval. │ │ │ │ │ - * This is "private", and should be left alone. │ │ │ │ │ + * Property: gmlParse │ │ │ │ │ + * {Object} GML Format object for parsing features │ │ │ │ │ + * Non-API and only created if necessary │ │ │ │ │ */ │ │ │ │ │ - timeoutId: null, │ │ │ │ │ + gmlParser: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: documentDrag │ │ │ │ │ - * {Boolean} If set to true, the handler will also handle mouse moves when │ │ │ │ │ - * the cursor has moved out of the map viewport. Default is false. │ │ │ │ │ + * APIProperty: xy │ │ │ │ │ + * {Boolean} Order of the GML coordinate: true:(x,y) or false:(y,x) │ │ │ │ │ + * For GeoRSS the default is (y,x), therefore: false │ │ │ │ │ */ │ │ │ │ │ - documentDrag: false, │ │ │ │ │ + xy: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: documentEvents │ │ │ │ │ - * {Boolean} Are we currently observing document events? │ │ │ │ │ + * Constructor: OpenLayers.Format.AtomEntry │ │ │ │ │ + * Create a new parser for Atom. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - documentEvents: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Drag │ │ │ │ │ - * Returns OpenLayers.Handler.Drag │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Return a list of features from an Atom feed or entry document. │ │ │ │ │ + │ │ │ │ │ * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ - * this handler. If a handler is being used without a control, the │ │ │ │ │ - * handlers setMap method must be overridden to deal properly with │ │ │ │ │ - * the map. │ │ │ │ │ - * callbacks - {Object} An object containing a single function to be │ │ │ │ │ - * called when the drag operation is finished. The callback should │ │ │ │ │ - * expect to recieve a single argument, the pixel location of the event. │ │ │ │ │ - * Callbacks for 'move' and 'done' are supported. You can also speficy │ │ │ │ │ - * callbacks for 'down', 'up', and 'out' to respond to those events. │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * doc - {Element} or {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - if (this.documentDrag === true) { │ │ │ │ │ - var me = this; │ │ │ │ │ - this._docMove = function(evt) { │ │ │ │ │ - me.mousemove({ │ │ │ │ │ - xy: { │ │ │ │ │ - x: evt.clientX, │ │ │ │ │ - y: evt.clientY │ │ │ │ │ - }, │ │ │ │ │ - element: document │ │ │ │ │ - }); │ │ │ │ │ - }; │ │ │ │ │ - this._docUp = function(evt) { │ │ │ │ │ - me.mouseup({ │ │ │ │ │ - xy: { │ │ │ │ │ - x: evt.clientX, │ │ │ │ │ - y: evt.clientY │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - }; │ │ │ │ │ + read: function(doc) { │ │ │ │ │ + if (typeof doc == "string") { │ │ │ │ │ + doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]); │ │ │ │ │ } │ │ │ │ │ + return this.parseFeatures(doc); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: dragstart │ │ │ │ │ - * This private method is factorized from mousedown and touchstart methods │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Serialize or more feature nodes to Atom documents. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} The event │ │ │ │ │ + * features - {<OpenLayers.Feature.Vector>} or Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * {String} an Atom entry document if passed one feature node, or a feed │ │ │ │ │ + * document if passed an array of feature nodes. │ │ │ │ │ */ │ │ │ │ │ - dragstart: function(evt) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - if (this.checkModifiers(evt) && │ │ │ │ │ - (OpenLayers.Event.isLeftClick(evt) || │ │ │ │ │ - OpenLayers.Event.isSingleTouch(evt))) { │ │ │ │ │ - this.started = true; │ │ │ │ │ - this.start = evt.xy; │ │ │ │ │ - this.last = evt.xy; │ │ │ │ │ - OpenLayers.Element.addClass( │ │ │ │ │ - this.map.viewPortDiv, "olDragDown" │ │ │ │ │ + write: function(features) { │ │ │ │ │ + var doc; │ │ │ │ │ + if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ + doc = this.createElementNSPlus("atom:feed"); │ │ │ │ │ + doc.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:title", { │ │ │ │ │ + value: this.feedTitle │ │ │ │ │ + }) │ │ │ │ │ ); │ │ │ │ │ - this.down(evt); │ │ │ │ │ - this.callback("down", [evt.xy]); │ │ │ │ │ - │ │ │ │ │ - // prevent document dragging │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ - │ │ │ │ │ - if (!this.oldOnselectstart) { │ │ │ │ │ - this.oldOnselectstart = document.onselectstart ? │ │ │ │ │ - document.onselectstart : OpenLayers.Function.True; │ │ │ │ │ + for (var i = 0, ii = features.length; i < ii; i++) { │ │ │ │ │ + doc.appendChild(this.buildEntryNode(features[i])); │ │ │ │ │ } │ │ │ │ │ - document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ - │ │ │ │ │ - propagate = !this.stopDown; │ │ │ │ │ } else { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ + doc = this.buildEntryNode(features); │ │ │ │ │ } │ │ │ │ │ - return propagate; │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [doc]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: dragmove │ │ │ │ │ - * This private method is factorized from mousemove and touchmove methods │ │ │ │ │ + * Method: buildContentNode │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} The event │ │ │ │ │ + * content - {Object} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * {DOMElement} an Atom content node. │ │ │ │ │ + * │ │ │ │ │ + * TODO: types other than text. │ │ │ │ │ */ │ │ │ │ │ - dragmove: function(evt) { │ │ │ │ │ - this.lastMoveEvt = evt; │ │ │ │ │ - if (this.started && !this.timeoutId && (evt.xy.x != this.last.x || │ │ │ │ │ - evt.xy.y != this.last.y)) { │ │ │ │ │ - if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ - if (evt.element === document) { │ │ │ │ │ - this.adjustXY(evt); │ │ │ │ │ - // do setEvent manually because the documentEvents are not │ │ │ │ │ - // registered with the map │ │ │ │ │ - this.setEvent(evt); │ │ │ │ │ - } else { │ │ │ │ │ - this.removeDocumentEvents(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.interval > 0) { │ │ │ │ │ - this.timeoutId = setTimeout( │ │ │ │ │ - OpenLayers.Function.bind(this.removeTimeout, this), │ │ │ │ │ - this.interval); │ │ │ │ │ + buildContentNode: function(content) { │ │ │ │ │ + var node = this.createElementNSPlus("atom:content", { │ │ │ │ │ + attributes: { │ │ │ │ │ + type: content.type || null │ │ │ │ │ } │ │ │ │ │ - this.dragging = true; │ │ │ │ │ - │ │ │ │ │ - this.move(evt); │ │ │ │ │ - this.callback("move", [evt.xy]); │ │ │ │ │ - if (!this.oldOnselectstart) { │ │ │ │ │ - this.oldOnselectstart = document.onselectstart; │ │ │ │ │ - document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ + }); │ │ │ │ │ + if (content.src) { │ │ │ │ │ + node.setAttribute("src", content.src); │ │ │ │ │ + } else { │ │ │ │ │ + if (content.type == "text" || content.type == null) { │ │ │ │ │ + node.appendChild( │ │ │ │ │ + this.createTextNode(content.value) │ │ │ │ │ + ); │ │ │ │ │ + } else if (content.type == "html") { │ │ │ │ │ + if (typeof content.value != "string") { │ │ │ │ │ + throw "HTML content must be in form of an escaped string"; │ │ │ │ │ + } │ │ │ │ │ + node.appendChild( │ │ │ │ │ + this.createTextNode(content.value) │ │ │ │ │ + ); │ │ │ │ │ + } else if (content.type == "xhtml") { │ │ │ │ │ + node.appendChild(content.value); │ │ │ │ │ + } else if (content.type == "xhtml" || │ │ │ │ │ + content.type.match(/(\+|\/)xml$/)) { │ │ │ │ │ + node.appendChild(content.value); │ │ │ │ │ + } else { // MUST be a valid Base64 encoding │ │ │ │ │ + node.appendChild( │ │ │ │ │ + this.createTextNode(content.value) │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - this.last = evt.xy; │ │ │ │ │ } │ │ │ │ │ - return true; │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: dragend │ │ │ │ │ - * This private method is factorized from mouseup and touchend methods │ │ │ │ │ + * Method: buildEntryNode │ │ │ │ │ + * Build an Atom entry node from a feature object. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} The event │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * {DOMElement} an Atom entry node. │ │ │ │ │ + * │ │ │ │ │ + * These entries are geared for publication using AtomPub. │ │ │ │ │ + * │ │ │ │ │ + * TODO: support extension elements │ │ │ │ │ */ │ │ │ │ │ - dragend: function(evt) { │ │ │ │ │ - if (this.started) { │ │ │ │ │ - if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ - this.adjustXY(evt); │ │ │ │ │ - this.removeDocumentEvents(); │ │ │ │ │ + buildEntryNode: function(feature) { │ │ │ │ │ + var attrib = feature.attributes; │ │ │ │ │ + var atomAttrib = attrib.atom || {}; │ │ │ │ │ + var entryNode = this.createElementNSPlus("atom:entry"); │ │ │ │ │ + │ │ │ │ │ + // atom:author │ │ │ │ │ + if (atomAttrib.authors) { │ │ │ │ │ + var authors = OpenLayers.Util.isArray(atomAttrib.authors) ? │ │ │ │ │ + atomAttrib.authors : [atomAttrib.authors]; │ │ │ │ │ + for (var i = 0, ii = authors.length; i < ii; i++) { │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.buildPersonConstructNode( │ │ │ │ │ + "author", authors[i] │ │ │ │ │ + ) │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - var dragged = (this.start != this.last); │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - OpenLayers.Element.removeClass( │ │ │ │ │ - this.map.viewPortDiv, "olDragDown" │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atom:category │ │ │ │ │ + if (atomAttrib.categories) { │ │ │ │ │ + var categories = OpenLayers.Util.isArray(atomAttrib.categories) ? │ │ │ │ │ + atomAttrib.categories : [atomAttrib.categories]; │ │ │ │ │ + var category; │ │ │ │ │ + for (var i = 0, ii = categories.length; i < ii; i++) { │ │ │ │ │ + category = categories[i]; │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:category", { │ │ │ │ │ + attributes: { │ │ │ │ │ + term: category.term, │ │ │ │ │ + scheme: category.scheme || null, │ │ │ │ │ + label: category.label || null │ │ │ │ │ + } │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atom:content │ │ │ │ │ + if (atomAttrib.content) { │ │ │ │ │ + entryNode.appendChild(this.buildContentNode(atomAttrib.content)); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atom:contributor │ │ │ │ │ + if (atomAttrib.contributors) { │ │ │ │ │ + var contributors = OpenLayers.Util.isArray(atomAttrib.contributors) ? │ │ │ │ │ + atomAttrib.contributors : [atomAttrib.contributors]; │ │ │ │ │ + for (var i = 0, ii = contributors.length; i < ii; i++) { │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.buildPersonConstructNode( │ │ │ │ │ + "contributor", │ │ │ │ │ + contributors[i] │ │ │ │ │ + ) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atom:id │ │ │ │ │ + if (feature.fid) { │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:id", { │ │ │ │ │ + value: feature.fid │ │ │ │ │ + }) │ │ │ │ │ ); │ │ │ │ │ - this.up(evt); │ │ │ │ │ - this.callback("up", [evt.xy]); │ │ │ │ │ - if (dragged) { │ │ │ │ │ - this.callback("done", [evt.xy]); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atom:link │ │ │ │ │ + if (atomAttrib.links) { │ │ │ │ │ + var links = OpenLayers.Util.isArray(atomAttrib.links) ? │ │ │ │ │ + atomAttrib.links : [atomAttrib.links]; │ │ │ │ │ + var link; │ │ │ │ │ + for (var i = 0, ii = links.length; i < ii; i++) { │ │ │ │ │ + link = links[i]; │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:link", { │ │ │ │ │ + attributes: { │ │ │ │ │ + href: link.href, │ │ │ │ │ + rel: link.rel || null, │ │ │ │ │ + type: link.type || null, │ │ │ │ │ + hreflang: link.hreflang || null, │ │ │ │ │ + title: link.title || null, │ │ │ │ │ + length: link.length || null │ │ │ │ │ + } │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - document.onselectstart = this.oldOnselectstart; │ │ │ │ │ } │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * The four methods below (down, move, up, and out) are used by subclasses │ │ │ │ │ - * to do their own processing related to these mouse events. │ │ │ │ │ - */ │ │ │ │ │ + // atom:published │ │ │ │ │ + if (atomAttrib.published) { │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:published", { │ │ │ │ │ + value: atomAttrib.published │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: down │ │ │ │ │ - * This method is called during the handling of the mouse down event. │ │ │ │ │ - * Subclasses can do their own processing here. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The mouse down event │ │ │ │ │ - */ │ │ │ │ │ - down: function(evt) {}, │ │ │ │ │ + // atom:rights │ │ │ │ │ + if (atomAttrib.rights) { │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:rights", { │ │ │ │ │ + value: atomAttrib.rights │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * This method is called during the handling of the mouse move event. │ │ │ │ │ - * Subclasses can do their own processing here. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The mouse move event │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - move: function(evt) {}, │ │ │ │ │ + // atom:source not implemented │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: up │ │ │ │ │ - * This method is called during the handling of the mouse up event. │ │ │ │ │ - * Subclasses can do their own processing here. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The mouse up event │ │ │ │ │ - */ │ │ │ │ │ - up: function(evt) {}, │ │ │ │ │ + // atom:summary │ │ │ │ │ + if (atomAttrib.summary || attrib.description) { │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:summary", { │ │ │ │ │ + value: atomAttrib.summary || attrib.description │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: out │ │ │ │ │ - * This method is called during the handling of the mouse out event. │ │ │ │ │ - * Subclasses can do their own processing here. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The mouse out event │ │ │ │ │ - */ │ │ │ │ │ - out: function(evt) {}, │ │ │ │ │ + // atom:title │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:title", { │ │ │ │ │ + value: atomAttrib.title || attrib.title || this.defaultEntryTitle │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + // atom:updated │ │ │ │ │ + if (atomAttrib.updated) { │ │ │ │ │ + entryNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:updated", { │ │ │ │ │ + value: atomAttrib.updated │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // georss:where │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + var whereNode = this.createElementNSPlus("georss:where"); │ │ │ │ │ + whereNode.appendChild( │ │ │ │ │ + this.buildGeometryNode(feature.geometry) │ │ │ │ │ + ); │ │ │ │ │ + entryNode.appendChild(whereNode); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return entryNode; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * The methods below are part of the magic of event handling. Because │ │ │ │ │ - * they are named like browser events, they are registered as listeners │ │ │ │ │ - * for the events they represent. │ │ │ │ │ + * Method: initGmlParser │ │ │ │ │ + * Creates a GML parser. │ │ │ │ │ */ │ │ │ │ │ + initGmlParser: function() { │ │ │ │ │ + this.gmlParser = new OpenLayers.Format.GML.v3({ │ │ │ │ │ + xy: this.xy, │ │ │ │ │ + featureNS: "http://example.com#feature", │ │ │ │ │ + internalProjection: this.internalProjection, │ │ │ │ │ + externalProjection: this.externalProjection │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mousedown │ │ │ │ │ - * Handle mousedown events │ │ │ │ │ + * Method: buildGeometryNode │ │ │ │ │ + * builds a GeoRSS node with a given geometry │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * {DOMElement} A gml node. │ │ │ │ │ */ │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - return this.dragstart(evt); │ │ │ │ │ + buildGeometryNode: function(geometry) { │ │ │ │ │ + if (!this.gmlParser) { │ │ │ │ │ + this.initGmlParser(); │ │ │ │ │ + } │ │ │ │ │ + var node = this.gmlParser.writeNode("feature:_geometry", geometry); │ │ │ │ │ + return node.firstChild; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchstart │ │ │ │ │ - * Handle touchstart events │ │ │ │ │ + * Method: buildPersonConstructNode │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * name - {String} │ │ │ │ │ + * value - {Object} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * {DOMElement} an Atom person construct node. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * >>> buildPersonConstructNode("author", {name: "John Smith"}) │ │ │ │ │ + * {<author><name>John Smith</name></author>} │ │ │ │ │ + * │ │ │ │ │ + * TODO: how to specify extension elements? Add to the oNames array? │ │ │ │ │ */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - return this.dragstart(evt); │ │ │ │ │ + buildPersonConstructNode: function(name, value) { │ │ │ │ │ + var oNames = ["uri", "email"]; │ │ │ │ │ + var personNode = this.createElementNSPlus("atom:" + name); │ │ │ │ │ + personNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:name", { │ │ │ │ │ + value: value.name │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + for (var i = 0, ii = oNames.length; i < ii; i++) { │ │ │ │ │ + if (value[oNames[i]]) { │ │ │ │ │ + personNode.appendChild( │ │ │ │ │ + this.createElementNSPlus("atom:" + oNames[i], { │ │ │ │ │ + value: value[oNames[i]] │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return personNode; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mousemove │ │ │ │ │ - * Handle mousemove events │ │ │ │ │ + * Method: getFirstChildValue │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * node - {DOMElement} │ │ │ │ │ + * nsuri - {String} Child node namespace uri ("*" for any). │ │ │ │ │ + * name - {String} Child node name. │ │ │ │ │ + * def - {String} Optional string default to return if no child found. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * {String} The value of the first child with the given tag name. Returns │ │ │ │ │ + * default value or empty string if none found. │ │ │ │ │ */ │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - return this.dragmove(evt); │ │ │ │ │ + getFirstChildValue: function(node, nsuri, name, def) { │ │ │ │ │ + var value; │ │ │ │ │ + var nodes = this.getElementsByTagNameNS(node, nsuri, name); │ │ │ │ │ + if (nodes && nodes.length > 0) { │ │ │ │ │ + value = this.getChildValue(nodes[0], def); │ │ │ │ │ + } else { │ │ │ │ │ + value = def; │ │ │ │ │ + } │ │ │ │ │ + return value; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchmove │ │ │ │ │ - * Handle touchmove events │ │ │ │ │ + * Method: parseFeature │ │ │ │ │ + * Parse feature from an Atom entry node.. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * node - {DOMElement} An Atom entry or feed node. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - return this.dragmove(evt); │ │ │ │ │ - }, │ │ │ │ │ + parseFeature: function(node) { │ │ │ │ │ + var atomAttrib = {}; │ │ │ │ │ + var value = null; │ │ │ │ │ + var nodes = null; │ │ │ │ │ + var attval = null; │ │ │ │ │ + var atomns = this.namespaces.atom; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeTimeout │ │ │ │ │ - * Private. Called by mousemove() to remove the drag timeout. │ │ │ │ │ - */ │ │ │ │ │ - removeTimeout: function() { │ │ │ │ │ - this.timeoutId = null; │ │ │ │ │ - // if timeout expires while we're still dragging (mouseup │ │ │ │ │ - // hasn't occurred) then call mousemove to move to the │ │ │ │ │ - // correct position │ │ │ │ │ - if (this.dragging) { │ │ │ │ │ - this.mousemove(this.lastMoveEvt); │ │ │ │ │ + // atomAuthor* │ │ │ │ │ + this.parsePersonConstructs(node, "author", atomAttrib); │ │ │ │ │ + │ │ │ │ │ + // atomCategory* │ │ │ │ │ + nodes = this.getElementsByTagNameNS(node, atomns, "category"); │ │ │ │ │ + if (nodes.length > 0) { │ │ │ │ │ + atomAttrib.categories = []; │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ + value = {}; │ │ │ │ │ + value.term = nodes[i].getAttribute("term"); │ │ │ │ │ + attval = nodes[i].getAttribute("scheme"); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value.scheme = attval; │ │ │ │ │ + } │ │ │ │ │ + attval = nodes[i].getAttribute("label"); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value.label = attval; │ │ │ │ │ + } │ │ │ │ │ + atomAttrib.categories.push(value); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atomContent? │ │ │ │ │ + nodes = this.getElementsByTagNameNS(node, atomns, "content"); │ │ │ │ │ + if (nodes.length > 0) { │ │ │ │ │ + value = {}; │ │ │ │ │ + attval = nodes[0].getAttribute("type"); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value.type = attval; │ │ │ │ │ + } │ │ │ │ │ + attval = nodes[0].getAttribute("src"); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value.src = attval; │ │ │ │ │ + } else { │ │ │ │ │ + if (value.type == "text" || │ │ │ │ │ + value.type == "html" || │ │ │ │ │ + value.type == null) { │ │ │ │ │ + value.value = this.getFirstChildValue( │ │ │ │ │ + node, │ │ │ │ │ + atomns, │ │ │ │ │ + "content", │ │ │ │ │ + null │ │ │ │ │ + ); │ │ │ │ │ + } else if (value.type == "xhtml" || │ │ │ │ │ + value.type.match(/(\+|\/)xml$/)) { │ │ │ │ │ + value.value = this.getChildEl(nodes[0]); │ │ │ │ │ + } else { // MUST be base64 encoded │ │ │ │ │ + value.value = this.getFirstChildValue( │ │ │ │ │ + node, │ │ │ │ │ + atomns, │ │ │ │ │ + "content", │ │ │ │ │ + null │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + atomAttrib.content = value; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atomContributor* │ │ │ │ │ + this.parsePersonConstructs(node, "contributor", atomAttrib); │ │ │ │ │ + │ │ │ │ │ + // atomId │ │ │ │ │ + atomAttrib.id = this.getFirstChildValue(node, atomns, "id", null); │ │ │ │ │ + │ │ │ │ │ + // atomLink* │ │ │ │ │ + nodes = this.getElementsByTagNameNS(node, atomns, "link"); │ │ │ │ │ + if (nodes.length > 0) { │ │ │ │ │ + atomAttrib.links = new Array(nodes.length); │ │ │ │ │ + } │ │ │ │ │ + var oAtts = ["rel", "type", "hreflang", "title", "length"]; │ │ │ │ │ + for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ + value = {}; │ │ │ │ │ + value.href = nodes[i].getAttribute("href"); │ │ │ │ │ + for (var j = 0, jj = oAtts.length; j < jj; j++) { │ │ │ │ │ + attval = nodes[i].getAttribute(oAtts[j]); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value[oAtts[j]] = attval; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + atomAttrib.links[i] = value; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atomPublished? │ │ │ │ │ + value = this.getFirstChildValue(node, atomns, "published", null); │ │ │ │ │ + if (value) { │ │ │ │ │ + atomAttrib.published = value; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atomRights? │ │ │ │ │ + value = this.getFirstChildValue(node, atomns, "rights", null); │ │ │ │ │ + if (value) { │ │ │ │ │ + atomAttrib.rights = value; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // atomSource? -- not implemented │ │ │ │ │ + │ │ │ │ │ + // atomSummary? │ │ │ │ │ + value = this.getFirstChildValue(node, atomns, "summary", null); │ │ │ │ │ + if (value) { │ │ │ │ │ + atomAttrib.summary = value; │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + // atomTitle │ │ │ │ │ + atomAttrib.title = this.getFirstChildValue( │ │ │ │ │ + node, atomns, "title", null │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + // atomUpdated │ │ │ │ │ + atomAttrib.updated = this.getFirstChildValue( │ │ │ │ │ + node, atomns, "updated", null │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + var featureAttrib = { │ │ │ │ │ + title: atomAttrib.title, │ │ │ │ │ + description: atomAttrib.summary, │ │ │ │ │ + atom: atomAttrib │ │ │ │ │ + }; │ │ │ │ │ + var geometry = this.parseLocations(node)[0]; │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry, featureAttrib); │ │ │ │ │ + feature.fid = atomAttrib.id; │ │ │ │ │ + return feature; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mouseup │ │ │ │ │ - * Handle mouseup events │ │ │ │ │ + * Method: parseFeatures │ │ │ │ │ + * Return features from an Atom entry or feed. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * node - {DOMElement} An Atom entry or feed node. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * Array({<OpenLayers.Feature.Vector>}) │ │ │ │ │ */ │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - return this.dragend(evt); │ │ │ │ │ + parseFeatures: function(node) { │ │ │ │ │ + var features = []; │ │ │ │ │ + var entries = this.getElementsByTagNameNS( │ │ │ │ │ + node, this.namespaces.atom, "entry" │ │ │ │ │ + ); │ │ │ │ │ + if (entries.length == 0) { │ │ │ │ │ + entries = [node]; │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, ii = entries.length; i < ii; i++) { │ │ │ │ │ + features.push(this.parseFeature(entries[i])); │ │ │ │ │ + } │ │ │ │ │ + return features; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchend │ │ │ │ │ - * Handle touchend events │ │ │ │ │ + * Method: parseLocations │ │ │ │ │ + * Parse the locations from an Atom entry or feed. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * node - {DOMElement} An Atom entry or feed node. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * Array({<OpenLayers.Geometry>}) │ │ │ │ │ */ │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - // override evt.xy with last position since touchend does not have │ │ │ │ │ - // any touch position │ │ │ │ │ - evt.xy = this.last; │ │ │ │ │ - return this.dragend(evt); │ │ │ │ │ + parseLocations: function(node) { │ │ │ │ │ + var georssns = this.namespaces.georss; │ │ │ │ │ + │ │ │ │ │ + var locations = { │ │ │ │ │ + components: [] │ │ │ │ │ + }; │ │ │ │ │ + var where = this.getElementsByTagNameNS(node, georssns, "where"); │ │ │ │ │ + if (where && where.length > 0) { │ │ │ │ │ + if (!this.gmlParser) { │ │ │ │ │ + this.initGmlParser(); │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, ii = where.length; i < ii; i++) { │ │ │ │ │ + this.gmlParser.readChildNodes(where[i], locations); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var components = locations.components; │ │ │ │ │ + var point = this.getElementsByTagNameNS(node, georssns, "point"); │ │ │ │ │ + if (point && point.length > 0) { │ │ │ │ │ + for (var i = 0, ii = point.length; i < ii; i++) { │ │ │ │ │ + var xy = OpenLayers.String.trim( │ │ │ │ │ + point[i].firstChild.nodeValue │ │ │ │ │ + ).split(/\s+/); │ │ │ │ │ + if (xy.length != 2) { │ │ │ │ │ + xy = OpenLayers.String.trim( │ │ │ │ │ + point[i].firstChild.nodeValue │ │ │ │ │ + ).split(/\s*,\s*/); │ │ │ │ │ + } │ │ │ │ │ + components.push(new OpenLayers.Geometry.Point(xy[1], xy[0])); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var line = this.getElementsByTagNameNS(node, georssns, "line"); │ │ │ │ │ + if (line && line.length > 0) { │ │ │ │ │ + var coords; │ │ │ │ │ + var p; │ │ │ │ │ + var points; │ │ │ │ │ + for (var i = 0, ii = line.length; i < ii; i++) { │ │ │ │ │ + coords = OpenLayers.String.trim( │ │ │ │ │ + line[i].firstChild.nodeValue │ │ │ │ │ + ).split(/\s+/); │ │ │ │ │ + points = []; │ │ │ │ │ + for (var j = 0, jj = coords.length; j < jj; j += 2) { │ │ │ │ │ + p = new OpenLayers.Geometry.Point(coords[j + 1], coords[j]); │ │ │ │ │ + points.push(p); │ │ │ │ │ + } │ │ │ │ │ + components.push( │ │ │ │ │ + new OpenLayers.Geometry.LineString(points) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var polygon = this.getElementsByTagNameNS(node, georssns, "polygon"); │ │ │ │ │ + if (polygon && polygon.length > 0) { │ │ │ │ │ + var coords; │ │ │ │ │ + var p; │ │ │ │ │ + var points; │ │ │ │ │ + for (var i = 0, ii = polygon.length; i < ii; i++) { │ │ │ │ │ + coords = OpenLayers.String.trim( │ │ │ │ │ + polygon[i].firstChild.nodeValue │ │ │ │ │ + ).split(/\s+/); │ │ │ │ │ + points = []; │ │ │ │ │ + for (var j = 0, jj = coords.length; j < jj; j += 2) { │ │ │ │ │ + p = new OpenLayers.Geometry.Point(coords[j + 1], coords[j]); │ │ │ │ │ + points.push(p); │ │ │ │ │ + } │ │ │ │ │ + components.push( │ │ │ │ │ + new OpenLayers.Geometry.Polygon( │ │ │ │ │ + [new OpenLayers.Geometry.LinearRing(points)] │ │ │ │ │ + ) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + for (var i = 0, ii = components.length; i < ii; i++) { │ │ │ │ │ + if (components[i]) { │ │ │ │ │ + components[i].transform( │ │ │ │ │ + this.externalProjection, │ │ │ │ │ + this.internalProjection │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return components; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mouseout │ │ │ │ │ - * Handle mouseout events │ │ │ │ │ + * Method: parsePersonConstruct │ │ │ │ │ + * Parse Atom person constructs from an Atom entry node. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * node - {DOMElement} An Atom entry or feed node. │ │ │ │ │ + * name - {String} Construcy name ("author" or "contributor") │ │ │ │ │ + * data = {Object} Object in which to put parsed persons. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * An {Object}. │ │ │ │ │ */ │ │ │ │ │ - mouseout: function(evt) { │ │ │ │ │ - if (this.started && OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ - if (this.documentDrag === true) { │ │ │ │ │ - this.addDocumentEvents(); │ │ │ │ │ - } else { │ │ │ │ │ - var dragged = (this.start != this.last); │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - OpenLayers.Element.removeClass( │ │ │ │ │ - this.map.viewPortDiv, "olDragDown" │ │ │ │ │ - ); │ │ │ │ │ - this.out(evt); │ │ │ │ │ - this.callback("out", []); │ │ │ │ │ - if (dragged) { │ │ │ │ │ - this.callback("done", [evt.xy]); │ │ │ │ │ - } │ │ │ │ │ - if (document.onselectstart) { │ │ │ │ │ - document.onselectstart = this.oldOnselectstart; │ │ │ │ │ + parsePersonConstructs: function(node, name, data) { │ │ │ │ │ + var persons = []; │ │ │ │ │ + var atomns = this.namespaces.atom; │ │ │ │ │ + var nodes = this.getElementsByTagNameNS(node, atomns, name); │ │ │ │ │ + var oAtts = ["uri", "email"]; │ │ │ │ │ + for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ + var value = {}; │ │ │ │ │ + value.name = this.getFirstChildValue( │ │ │ │ │ + nodes[i], │ │ │ │ │ + atomns, │ │ │ │ │ + "name", │ │ │ │ │ + null │ │ │ │ │ + ); │ │ │ │ │ + for (var j = 0, jj = oAtts.length; j < jj; j++) { │ │ │ │ │ + var attval = this.getFirstChildValue( │ │ │ │ │ + nodes[i], │ │ │ │ │ + atomns, │ │ │ │ │ + oAtts[j], │ │ │ │ │ + null); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value[oAtts[j]] = attval; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + persons.push(value); │ │ │ │ │ + } │ │ │ │ │ + if (persons.length > 0) { │ │ │ │ │ + data[name + "s"] = persons; │ │ │ │ │ } │ │ │ │ │ - return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.Atom" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/SOSGetObservation.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Format/SOSGetFeatureOfInterest.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.SOSGetObservation │ │ │ │ │ + * Read and write SOS GetObersation (to get the actual values from a sensor) │ │ │ │ │ + * version 1.0.0 │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.SOSGetObservation = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: click │ │ │ │ │ - * The drag handler captures the click event. If something else registers │ │ │ │ │ - * for clicks on the same element, its listener will not be called │ │ │ │ │ - * after a drag. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ */ │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - // let the click event propagate only if the mouse moved │ │ │ │ │ - return (this.start == this.last); │ │ │ │ │ + namespaces: { │ │ │ │ │ + ows: "http://www.opengis.net/ows", │ │ │ │ │ + gml: "http://www.opengis.net/gml", │ │ │ │ │ + sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ + ogc: "http://www.opengis.net/ogc", │ │ │ │ │ + om: "http://www.opengis.net/om/1.0", │ │ │ │ │ + sa: "http://www.opengis.net/sampling/1.0", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Activate the handler. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully activated. │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - activated = true; │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Deactivate the handler. │ │ │ │ │ + * Constant: VERSION │ │ │ │ │ + * {String} 1.0.0 │ │ │ │ │ + */ │ │ │ │ │ + VERSION: "1.0.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} Schema location │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: "http://www.opengis.net/sos/1.0 http://schemas.opengis.net/sos/1.0.0/sosGetObservation.xsd", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: "sos", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.SOSGetObservation │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: read │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully deactivated. │ │ │ │ │ + * {Object} An object containing the measurements │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - deactivated = true; │ │ │ │ │ - OpenLayers.Element.removeClass( │ │ │ │ │ - this.map.viewPortDiv, "olDragDown" │ │ │ │ │ - ); │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ } │ │ │ │ │ - return deactivated; │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + var info = { │ │ │ │ │ + measurements: [], │ │ │ │ │ + observations: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readNode(data, info); │ │ │ │ │ + return info; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: adjustXY │ │ │ │ │ - * Converts event coordinates that are relative to the document body to │ │ │ │ │ - * ones that are relative to the map viewport. The latter is the default in │ │ │ │ │ - * OpenLayers. │ │ │ │ │ - * │ │ │ │ │ + * Method: write │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ + * options - {Object} Optional object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} An SOS GetObservation request XML string. │ │ │ │ │ */ │ │ │ │ │ - adjustXY: function(evt) { │ │ │ │ │ - var pos = OpenLayers.Util.pagePosition(this.map.viewPortDiv); │ │ │ │ │ - evt.xy.x -= pos[0]; │ │ │ │ │ - evt.xy.y -= pos[1]; │ │ │ │ │ + write: function(options) { │ │ │ │ │ + var node = this.writeNode("sos:GetObservation", options); │ │ │ │ │ + node.setAttribute("xmlns:om", this.namespaces.om); │ │ │ │ │ + node.setAttribute("xmlns:ogc", this.namespaces.ogc); │ │ │ │ │ + this.setAttributeNS( │ │ │ │ │ + node, this.namespaces.xsi, │ │ │ │ │ + "xsi:schemaLocation", this.schemaLocation │ │ │ │ │ + ); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [node]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: addDocumentEvents │ │ │ │ │ - * Start observing document events when documentDrag is true and the mouse │ │ │ │ │ - * cursor leaves the map viewport while dragging. │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ */ │ │ │ │ │ - addDocumentEvents: function() { │ │ │ │ │ - OpenLayers.Element.addClass(document.body, "olDragDown"); │ │ │ │ │ - this.documentEvents = true; │ │ │ │ │ - OpenLayers.Event.observe(document, "mousemove", this._docMove); │ │ │ │ │ - OpenLayers.Event.observe(document, "mouseup", this._docUp); │ │ │ │ │ + readers: { │ │ │ │ │ + "om": { │ │ │ │ │ + "ObservationCollection": function(node, obj) { │ │ │ │ │ + obj.id = this.getAttributeNS(node, this.namespaces.gml, "id"); │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "member": function(node, observationCollection) { │ │ │ │ │ + this.readChildNodes(node, observationCollection); │ │ │ │ │ + }, │ │ │ │ │ + "Measurement": function(node, observationCollection) { │ │ │ │ │ + var measurement = {}; │ │ │ │ │ + observationCollection.measurements.push(measurement); │ │ │ │ │ + this.readChildNodes(node, measurement); │ │ │ │ │ + }, │ │ │ │ │ + "Observation": function(node, observationCollection) { │ │ │ │ │ + var observation = {}; │ │ │ │ │ + observationCollection.observations.push(observation); │ │ │ │ │ + this.readChildNodes(node, observation); │ │ │ │ │ + }, │ │ │ │ │ + "samplingTime": function(node, measurement) { │ │ │ │ │ + var samplingTime = {}; │ │ │ │ │ + measurement.samplingTime = samplingTime; │ │ │ │ │ + this.readChildNodes(node, samplingTime); │ │ │ │ │ + }, │ │ │ │ │ + "observedProperty": function(node, measurement) { │ │ │ │ │ + measurement.observedProperty = │ │ │ │ │ + this.getAttributeNS(node, this.namespaces.xlink, "href"); │ │ │ │ │ + this.readChildNodes(node, measurement); │ │ │ │ │ + }, │ │ │ │ │ + "procedure": function(node, measurement) { │ │ │ │ │ + measurement.procedure = │ │ │ │ │ + this.getAttributeNS(node, this.namespaces.xlink, "href"); │ │ │ │ │ + this.readChildNodes(node, measurement); │ │ │ │ │ + }, │ │ │ │ │ + "featureOfInterest": function(node, observation) { │ │ │ │ │ + var foi = { │ │ │ │ │ + features: [] │ │ │ │ │ + }; │ │ │ │ │ + observation.fois = []; │ │ │ │ │ + observation.fois.push(foi); │ │ │ │ │ + this.readChildNodes(node, foi); │ │ │ │ │ + // postprocessing to get actual features │ │ │ │ │ + var features = []; │ │ │ │ │ + for (var i = 0, len = foi.features.length; i < len; i++) { │ │ │ │ │ + var feature = foi.features[i]; │ │ │ │ │ + features.push(new OpenLayers.Feature.Vector( │ │ │ │ │ + feature.components[0], feature.attributes)); │ │ │ │ │ + } │ │ │ │ │ + foi.features = features; │ │ │ │ │ + }, │ │ │ │ │ + "result": function(node, measurement) { │ │ │ │ │ + var result = {}; │ │ │ │ │ + measurement.result = result; │ │ │ │ │ + if (this.getChildValue(node) !== '') { │ │ │ │ │ + result.value = this.getChildValue(node); │ │ │ │ │ + result.uom = node.getAttribute("uom"); │ │ │ │ │ + } else { │ │ │ │ │ + this.readChildNodes(node, result); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "sa": OpenLayers.Format.SOSGetFeatureOfInterest.prototype.readers.sa, │ │ │ │ │ + "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "TimeInstant": function(node, samplingTime) { │ │ │ │ │ + var timeInstant = {}; │ │ │ │ │ + samplingTime.timeInstant = timeInstant; │ │ │ │ │ + this.readChildNodes(node, timeInstant); │ │ │ │ │ + }, │ │ │ │ │ + "timePosition": function(node, timeInstant) { │ │ │ │ │ + timeInstant.timePosition = this.getChildValue(node); │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.SOSGetFeatureOfInterest.prototype.readers.gml) │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removeDocumentEvents │ │ │ │ │ - * Stops observing document events when documentDrag is true and the mouse │ │ │ │ │ - * cursor re-enters the map viewport while dragging. │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ */ │ │ │ │ │ - removeDocumentEvents: function() { │ │ │ │ │ - OpenLayers.Element.removeClass(document.body, "olDragDown"); │ │ │ │ │ - this.documentEvents = false; │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mousemove", this._docMove); │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mouseup", this._docUp); │ │ │ │ │ + writers: { │ │ │ │ │ + "sos": { │ │ │ │ │ + "GetObservation": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("GetObservation", { │ │ │ │ │ + attributes: { │ │ │ │ │ + version: this.VERSION, │ │ │ │ │ + service: 'SOS' │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + this.writeNode("offering", options, node); │ │ │ │ │ + if (options.eventTime) { │ │ │ │ │ + this.writeNode("eventTime", options, node); │ │ │ │ │ + } │ │ │ │ │ + for (var procedure in options.procedures) { │ │ │ │ │ + this.writeNode("procedure", options.procedures[procedure], node); │ │ │ │ │ + } │ │ │ │ │ + for (var observedProperty in options.observedProperties) { │ │ │ │ │ + this.writeNode("observedProperty", options.observedProperties[observedProperty], node); │ │ │ │ │ + } │ │ │ │ │ + if (options.foi) { │ │ │ │ │ + this.writeNode("featureOfInterest", options.foi, node); │ │ │ │ │ + } │ │ │ │ │ + this.writeNode("responseFormat", options, node); │ │ │ │ │ + if (options.resultModel) { │ │ │ │ │ + this.writeNode("resultModel", options, node); │ │ │ │ │ + } │ │ │ │ │ + if (options.responseMode) { │ │ │ │ │ + this.writeNode("responseMode", options, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "featureOfInterest": function(foi) { │ │ │ │ │ + var node = this.createElementNSPlus("featureOfInterest"); │ │ │ │ │ + this.writeNode("ObjectID", foi.objectId, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "ObjectID": function(options) { │ │ │ │ │ + return this.createElementNSPlus("ObjectID", { │ │ │ │ │ + value: options │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "responseFormat": function(options) { │ │ │ │ │ + return this.createElementNSPlus("responseFormat", { │ │ │ │ │ + value: options.responseFormat │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "procedure": function(procedure) { │ │ │ │ │ + return this.createElementNSPlus("procedure", { │ │ │ │ │ + value: procedure │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "offering": function(options) { │ │ │ │ │ + return this.createElementNSPlus("offering", { │ │ │ │ │ + value: options.offering │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "observedProperty": function(observedProperty) { │ │ │ │ │ + return this.createElementNSPlus("observedProperty", { │ │ │ │ │ + value: observedProperty │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "eventTime": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("eventTime"); │ │ │ │ │ + if (options.eventTime === 'latest') { │ │ │ │ │ + this.writeNode("ogc:TM_Equals", options, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "resultModel": function(options) { │ │ │ │ │ + return this.createElementNSPlus("resultModel", { │ │ │ │ │ + value: options.resultModel │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "responseMode": function(options) { │ │ │ │ │ + return this.createElementNSPlus("responseMode", { │ │ │ │ │ + value: options.responseMode │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "ogc": { │ │ │ │ │ + "TM_Equals": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:TM_Equals"); │ │ │ │ │ + this.writeNode("ogc:PropertyName", { │ │ │ │ │ + property: "urn:ogc:data:time:iso8601" │ │ │ │ │ + }, node); │ │ │ │ │ + if (options.eventTime === 'latest') { │ │ │ │ │ + this.writeNode("gml:TimeInstant", { │ │ │ │ │ + value: 'latest' │ │ │ │ │ + }, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "PropertyName": function(options) { │ │ │ │ │ + return this.createElementNSPlus("ogc:PropertyName", { │ │ │ │ │ + value: options.property │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "gml": { │ │ │ │ │ + "TimeInstant": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:TimeInstant"); │ │ │ │ │ + this.writeNode("gml:timePosition", options, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "timePosition": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:timePosition", { │ │ │ │ │ + value: options.value │ │ │ │ │ + }); │ │ │ │ │ + return node; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Drag" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SOSGetObservation" │ │ │ │ │ + │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/RegularPolygon.js │ │ │ │ │ + OpenLayers/Format/XLS.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler/Drag.js │ │ │ │ │ + * @requires OpenLayers/Format/XML/VersionedOGC.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Handler.RegularPolygon │ │ │ │ │ - * Handler to draw a regular polygon on the map. Polygon is displayed on mouse │ │ │ │ │ - * down, moves or is modified on mouse move, and is finished on mouse up. │ │ │ │ │ - * The handler triggers callbacks for 'done' and 'cancel'. Create a new │ │ │ │ │ - * instance with the <OpenLayers.Handler.RegularPolygon> constructor. │ │ │ │ │ + * Class: OpenLayers.Format.XLS │ │ │ │ │ + * Read/Write XLS (OpenLS). Create a new instance with the <OpenLayers.Format.XLS> │ │ │ │ │ + * constructor. Currently only implemented for Location Utility Services, more │ │ │ │ │ + * specifically only for Geocoding. No support for Reverse Geocoding as yet. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler.Drag> │ │ │ │ │ + * - <OpenLayers.Format.XML.VersionedOGC> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.RegularPolygon = OpenLayers.Class(OpenLayers.Handler.Drag, { │ │ │ │ │ +OpenLayers.Format.XLS = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: sides │ │ │ │ │ - * {Integer} Number of sides for the regular polygon. Needs to be greater │ │ │ │ │ - * than 2. Defaults to 4. │ │ │ │ │ + * APIProperty: defaultVersion │ │ │ │ │ + * {String} Version number to assume if none found. Default is "1.1.0". │ │ │ │ │ */ │ │ │ │ │ - sides: 4, │ │ │ │ │ + defaultVersion: "1.1.0", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: radius │ │ │ │ │ - * {Float} Optional radius in map units of the regular polygon. If this is │ │ │ │ │ - * set to some non-zero value, a polygon with a fixed radius will be │ │ │ │ │ - * drawn and dragged with mose movements. If this property is not │ │ │ │ │ - * set, dragging changes the radius of the polygon. Set to null by │ │ │ │ │ - * default. │ │ │ │ │ + * APIProperty: stringifyOutput │ │ │ │ │ + * {Boolean} If true, write will return a string otherwise a DOMElement. │ │ │ │ │ + * Default is true. │ │ │ │ │ */ │ │ │ │ │ - radius: null, │ │ │ │ │ + stringifyOutput: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: snapAngle │ │ │ │ │ - * {Float} If set to a non-zero value, the handler will snap the polygon │ │ │ │ │ - * rotation to multiples of the snapAngle. Value is an angle measured │ │ │ │ │ - * in degrees counterclockwise from the positive x-axis. │ │ │ │ │ + * Constructor: OpenLayers.Format.XLS │ │ │ │ │ + * Create a new parser for XLS. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - snapAngle: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: snapToggle │ │ │ │ │ - * {String} If set, snapToggle is checked on mouse events and will set │ │ │ │ │ - * the snap mode to the opposite of what it currently is. To disallow │ │ │ │ │ - * toggling between snap and non-snap mode, set freehandToggle to │ │ │ │ │ - * null. Acceptable toggle values are 'shiftKey', 'ctrlKey', and │ │ │ │ │ - * 'altKey'. Snap mode is only possible if this.snapAngle is set to a │ │ │ │ │ - * non-zero value. │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Write out an XLS request. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * request - {Object} An object representing the LUS request. │ │ │ │ │ + * options - {Object} Optional configuration object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} An XLS document string. │ │ │ │ │ */ │ │ │ │ │ - snapToggle: 'shiftKey', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: layerOptions │ │ │ │ │ - * {Object} Any optional properties to be set on the sketch layer. │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read an XLS doc and return an object representing the result. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String | DOMElement} Data to read. │ │ │ │ │ + * options - {Object} Options for the reader. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object representing the GeocodeResponse. │ │ │ │ │ */ │ │ │ │ │ - layerOptions: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: persist │ │ │ │ │ - * {Boolean} Leave the feature rendered until clear is called. Default │ │ │ │ │ - * is false. If set to true, the feature remains rendered until │ │ │ │ │ - * clear is called, typically by deactivating the handler or starting │ │ │ │ │ - * another drawing. │ │ │ │ │ - */ │ │ │ │ │ - persist: false, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.XLS" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/CSWGetDomain/v2_0_2.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: irregular │ │ │ │ │ - * {Boolean} Draw an irregular polygon instead of a regular polygon. │ │ │ │ │ - * Default is false. If true, the initial mouse down will represent │ │ │ │ │ - * one corner of the polygon bounds and with each mouse movement, the │ │ │ │ │ - * polygon will be stretched so the opposite corner of its bounds │ │ │ │ │ - * follows the mouse position. This property takes precedence over │ │ │ │ │ - * the radius property. If set to true, the radius property will │ │ │ │ │ - * be ignored. │ │ │ │ │ - */ │ │ │ │ │ - irregular: false, │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Format/CSWGetDomain.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.CSWGetDomain.v2_0_2 │ │ │ │ │ + * A format for creating CSWGetDomain v2.0.2 transactions. │ │ │ │ │ + * Create a new instance with the │ │ │ │ │ + * <OpenLayers.Format.CSWGetDomain.v2_0_2> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.CSWGetDomain.v2_0_2 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: citeCompliant │ │ │ │ │ - * {Boolean} If set to true, coordinates of features drawn in a map extent │ │ │ │ │ - * crossing the date line won't exceed the world bounds. Default is false. │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ */ │ │ │ │ │ - citeCompliant: false, │ │ │ │ │ + namespaces: { │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + csw: "http://www.opengis.net/cat/csw/2.0.2" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: angle │ │ │ │ │ - * {Float} The angle from the origin (mouse down) to the current mouse │ │ │ │ │ - * position, in radians. This is measured counterclockwise from the │ │ │ │ │ - * positive x-axis. │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + * {String} The default prefix (used by Format.XML). │ │ │ │ │ */ │ │ │ │ │ - angle: null, │ │ │ │ │ + defaultPrefix: "csw", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: fixedRadius │ │ │ │ │ - * {Boolean} The polygon has a fixed radius. True if a radius is set before │ │ │ │ │ - * drawing begins. False otherwise. │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} CSW version number. │ │ │ │ │ */ │ │ │ │ │ - fixedRadius: false, │ │ │ │ │ + version: "2.0.2", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: feature │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} The currently drawn polygon feature │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} http://www.opengis.net/cat/csw/2.0.2 │ │ │ │ │ + * http://schemas.opengis.net/csw/2.0.2/CSW-discovery.xsd │ │ │ │ │ */ │ │ │ │ │ - feature: null, │ │ │ │ │ + schemaLocation: "http://www.opengis.net/cat/csw/2.0.2 http://schemas.opengis.net/csw/2.0.2/CSW-discovery.xsd", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} The temporary drawing layer │ │ │ │ │ + * APIProperty: PropertyName │ │ │ │ │ + * {String} Value of the csw:PropertyName element, used when │ │ │ │ │ + * writing a GetDomain document. │ │ │ │ │ */ │ │ │ │ │ - layer: null, │ │ │ │ │ + PropertyName: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: origin │ │ │ │ │ - * {<OpenLayers.Geometry.Point>} Location of the first mouse down │ │ │ │ │ + * APIProperty: ParameterName │ │ │ │ │ + * {String} Value of the csw:ParameterName element, used when │ │ │ │ │ + * writing a GetDomain document. │ │ │ │ │ */ │ │ │ │ │ - origin: null, │ │ │ │ │ + ParameterName: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.RegularPolygon │ │ │ │ │ - * Create a new regular polygon handler. │ │ │ │ │ + * Constructor: OpenLayers.Format.CSWGetDomain.v2_0_2 │ │ │ │ │ + * A class for parsing and generating CSWGetDomain v2.0.2 transactions. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ - * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ - * functions. Various callbacks described below. │ │ │ │ │ - * options - {Object} An object with properties to be set on the handler. │ │ │ │ │ - * If the options.sides property is not specified, the number of sides │ │ │ │ │ - * will default to 4. │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ * │ │ │ │ │ - * Named callbacks: │ │ │ │ │ - * create - Called when a sketch is first created. Callback called with │ │ │ │ │ - * the creation point geometry and sketch feature. │ │ │ │ │ - * done - Called when the sketch drawing is finished. The callback will │ │ │ │ │ - * recieve a single argument, the sketch geometry. │ │ │ │ │ - * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ - * cancel callback will receive a geometry. │ │ │ │ │ + * Valid options properties: │ │ │ │ │ + * - PropertyName │ │ │ │ │ + * - ParameterName │ │ │ │ │ */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ - this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style['default'], {}); │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - OpenLayers.Handler.Drag.prototype.initialize.apply(this, │ │ │ │ │ - [control, callbacks, options]); │ │ │ │ │ - this.options = (options) ? options : {}; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Parse the response from a GetDomain request. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readNode(data, obj); │ │ │ │ │ + return obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setOptions │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * newOptions - {Object} │ │ │ │ │ - */ │ │ │ │ │ - setOptions: function(newOptions) { │ │ │ │ │ - OpenLayers.Util.extend(this.options, newOptions); │ │ │ │ │ - OpenLayers.Util.extend(this, newOptions); │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "csw": { │ │ │ │ │ + "GetDomainResponse": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "DomainValues": function(node, obj) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(obj.DomainValues))) { │ │ │ │ │ + obj.DomainValues = []; │ │ │ │ │ + } │ │ │ │ │ + var attrs = node.attributes; │ │ │ │ │ + var domainValue = {}; │ │ │ │ │ + for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ + domainValue[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ + } │ │ │ │ │ + this.readChildNodes(node, domainValue); │ │ │ │ │ + obj.DomainValues.push(domainValue); │ │ │ │ │ + }, │ │ │ │ │ + "PropertyName": function(node, obj) { │ │ │ │ │ + obj.PropertyName = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "ParameterName": function(node, obj) { │ │ │ │ │ + obj.ParameterName = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "ListOfValues": function(node, obj) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(obj.ListOfValues))) { │ │ │ │ │ + obj.ListOfValues = []; │ │ │ │ │ + } │ │ │ │ │ + this.readChildNodes(node, obj.ListOfValues); │ │ │ │ │ + }, │ │ │ │ │ + "Value": function(node, obj) { │ │ │ │ │ + var attrs = node.attributes; │ │ │ │ │ + var value = {}; │ │ │ │ │ + for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ + value[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ + } │ │ │ │ │ + value.value = this.getChildValue(node); │ │ │ │ │ + obj.push({ │ │ │ │ │ + Value: value │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "ConceptualScheme": function(node, obj) { │ │ │ │ │ + obj.ConceptualScheme = {}; │ │ │ │ │ + this.readChildNodes(node, obj.ConceptualScheme); │ │ │ │ │ + }, │ │ │ │ │ + "Name": function(node, obj) { │ │ │ │ │ + obj.Name = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Document": function(node, obj) { │ │ │ │ │ + obj.Document = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Authority": function(node, obj) { │ │ │ │ │ + obj.Authority = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "RangeOfValues": function(node, obj) { │ │ │ │ │ + obj.RangeOfValues = {}; │ │ │ │ │ + this.readChildNodes(node, obj.RangeOfValues); │ │ │ │ │ + }, │ │ │ │ │ + "MinValue": function(node, obj) { │ │ │ │ │ + var attrs = node.attributes; │ │ │ │ │ + var value = {}; │ │ │ │ │ + for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ + value[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ + } │ │ │ │ │ + value.value = this.getChildValue(node); │ │ │ │ │ + obj.MinValue = value; │ │ │ │ │ + }, │ │ │ │ │ + "MaxValue": function(node, obj) { │ │ │ │ │ + var attrs = node.attributes; │ │ │ │ │ + var value = {}; │ │ │ │ │ + for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ + value[attrs[i].name] = attrs[i].nodeValue; │ │ │ │ │ + } │ │ │ │ │ + value.value = this.getChildValue(node); │ │ │ │ │ + obj.MaxValue = value; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Turn on the handler. │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * Given an configuration js object, write a CSWGetDomain request. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} A object mapping the request. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully activated │ │ │ │ │ + * {String} A serialized CSWGetDomain request. │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.Drag.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - // create temporary vector layer for rendering geometry sketch │ │ │ │ │ - var options = OpenLayers.Util.extend({ │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - // indicate that the temp vector layer will never be out of range │ │ │ │ │ - // without this, resolution properties must be specified at the │ │ │ │ │ - // map-level for this temporary layer to init its resolutions │ │ │ │ │ - // correctly │ │ │ │ │ - calculateInRange: OpenLayers.Function.True, │ │ │ │ │ - wrapDateLine: this.citeCompliant │ │ │ │ │ - }, this.layerOptions); │ │ │ │ │ - this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ - this.map.addLayer(this.layer); │ │ │ │ │ - activated = true; │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ + write: function(options) { │ │ │ │ │ + var node = this.writeNode("csw:GetDomain", options); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [node]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Turn off the handler. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully deactivated │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.Drag.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - // call the cancel callback if mid-drawing │ │ │ │ │ - if (this.dragging) { │ │ │ │ │ - this.cancel(); │ │ │ │ │ - } │ │ │ │ │ - // If a layer's map property is set to null, it means that that │ │ │ │ │ - // layer isn't added to the map. Since we ourself added the layer │ │ │ │ │ - // to the map in activate(), we can assume that if this.layer.map │ │ │ │ │ - // is null it means that the layer has been destroyed (as a result │ │ │ │ │ - // of map.destroy() for example. │ │ │ │ │ - if (this.layer.map != null) { │ │ │ │ │ - this.layer.destroy(false); │ │ │ │ │ - if (this.feature) { │ │ │ │ │ - this.feature.destroy(); │ │ │ │ │ + writers: { │ │ │ │ │ + "csw": { │ │ │ │ │ + "GetDomain": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:GetDomain", { │ │ │ │ │ + attributes: { │ │ │ │ │ + service: "CSW", │ │ │ │ │ + version: this.version │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (options.PropertyName || this.PropertyName) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "csw:PropertyName", │ │ │ │ │ + options.PropertyName || this.PropertyName, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } else if (options.ParameterName || this.ParameterName) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "csw:ParameterName", │ │ │ │ │ + options.ParameterName || this.ParameterName, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ + this.readChildNodes(node, options); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "PropertyName": function(value) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:PropertyName", { │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "ParameterName": function(value) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:ParameterName", { │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + return node; │ │ │ │ │ } │ │ │ │ │ - this.layer = null; │ │ │ │ │ - this.feature = null; │ │ │ │ │ - deactivated = true; │ │ │ │ │ } │ │ │ │ │ - return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.CSWGetDomain.v2_0_2" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/SLD/v1.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Rule.js │ │ │ │ │ + * @requires OpenLayers/Format/SLD.js │ │ │ │ │ + * @requires OpenLayers/Format/Filter/v1_0_0.js │ │ │ │ │ + * @requires OpenLayers/Symbolizer/Point.js │ │ │ │ │ + * @requires OpenLayers/Symbolizer/Line.js │ │ │ │ │ + * @requires OpenLayers/Symbolizer/Polygon.js │ │ │ │ │ + * @requires OpenLayers/Symbolizer/Text.js │ │ │ │ │ + * @requires OpenLayers/Symbolizer/Raster.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.SLD.v1 │ │ │ │ │ + * Superclass for SLD version 1 parsers. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.Filter.v1_0_0> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.SLD.v1 = OpenLayers.Class(OpenLayers.Format.Filter.v1_0_0, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: down │ │ │ │ │ - * Start drawing a new feature │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The drag start event │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ */ │ │ │ │ │ - down: function(evt) { │ │ │ │ │ - this.fixedRadius = !!(this.radius); │ │ │ │ │ - var maploc = this.layer.getLonLatFromViewPortPx(evt.xy); │ │ │ │ │ - this.origin = new OpenLayers.Geometry.Point(maploc.lon, maploc.lat); │ │ │ │ │ - // create the new polygon │ │ │ │ │ - if (!this.fixedRadius || this.irregular) { │ │ │ │ │ - // smallest radius should not be less one pixel in map units │ │ │ │ │ - // VML doesn't behave well with smaller │ │ │ │ │ - this.radius = this.map.getResolution(); │ │ │ │ │ - } │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.clear(); │ │ │ │ │ - } │ │ │ │ │ - this.feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ - this.createGeometry(); │ │ │ │ │ - this.callback("create", [this.origin, this.feature]); │ │ │ │ │ - this.layer.addFeatures([this.feature], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.layer.drawFeature(this.feature, this.style); │ │ │ │ │ + namespaces: { │ │ │ │ │ + sld: "http://www.opengis.net/sld", │ │ │ │ │ + ogc: "http://www.opengis.net/ogc", │ │ │ │ │ + gml: "http://www.opengis.net/gml", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * Respond to drag move events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Evt} The move event │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ */ │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - var maploc = this.layer.getLonLatFromViewPortPx(evt.xy); │ │ │ │ │ - var point = new OpenLayers.Geometry.Point(maploc.lon, maploc.lat); │ │ │ │ │ - if (this.irregular) { │ │ │ │ │ - var ry = Math.sqrt(2) * Math.abs(point.y - this.origin.y) / 2; │ │ │ │ │ - this.radius = Math.max(this.map.getResolution() / 2, ry); │ │ │ │ │ - } else if (this.fixedRadius) { │ │ │ │ │ - this.origin = point; │ │ │ │ │ - } else { │ │ │ │ │ - this.calculateAngle(point, evt); │ │ │ │ │ - this.radius = Math.max(this.map.getResolution() / 2, │ │ │ │ │ - point.distanceTo(this.origin)); │ │ │ │ │ - } │ │ │ │ │ - this.modifyGeometry(); │ │ │ │ │ - if (this.irregular) { │ │ │ │ │ - var dx = point.x - this.origin.x; │ │ │ │ │ - var dy = point.y - this.origin.y; │ │ │ │ │ - var ratio; │ │ │ │ │ - if (dy == 0) { │ │ │ │ │ - ratio = dx / (this.radius * Math.sqrt(2)); │ │ │ │ │ - } else { │ │ │ │ │ - ratio = dx / dy; │ │ │ │ │ - } │ │ │ │ │ - this.feature.geometry.resize(1, this.origin, ratio); │ │ │ │ │ - this.feature.geometry.move(dx / 2, dy / 2); │ │ │ │ │ - } │ │ │ │ │ - this.layer.drawFeature(this.feature, this.style); │ │ │ │ │ + defaultPrefix: "sld", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} Schema location for a particular minor version. │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: multipleSymbolizers │ │ │ │ │ + * {Boolean} Support multiple symbolizers per rule. Default is false. if │ │ │ │ │ + * true, an OpenLayers.Style2 instance will be created to represent │ │ │ │ │ + * user styles instead of an OpenLayers.Style instace. The │ │ │ │ │ + * OpenLayers.Style2 class allows collections of rules with multiple │ │ │ │ │ + * symbolizers, but is not currently useful for client side rendering. │ │ │ │ │ + * If multiple symbolizers is true, multiple FeatureTypeStyle elements │ │ │ │ │ + * are preserved in reading/writing by setting symbolizer zIndex values. │ │ │ │ │ + * In addition, the <defaultSymbolizer> property is ignored if │ │ │ │ │ + * multiple symbolizers are supported (defaults should be applied │ │ │ │ │ + * when rendering). │ │ │ │ │ + */ │ │ │ │ │ + multipleSymbolizers: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: featureTypeCounter │ │ │ │ │ + * {Number} Private counter for multiple feature type styles. │ │ │ │ │ + */ │ │ │ │ │ + featureTypeCounter: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultSymbolizer. │ │ │ │ │ + * {Object} A symbolizer with the SLD defaults. │ │ │ │ │ + */ │ │ │ │ │ + defaultSymbolizer: { │ │ │ │ │ + fillColor: "#808080", │ │ │ │ │ + fillOpacity: 1, │ │ │ │ │ + strokeColor: "#000000", │ │ │ │ │ + strokeOpacity: 1, │ │ │ │ │ + strokeWidth: 1, │ │ │ │ │ + strokeDashstyle: "solid", │ │ │ │ │ + pointRadius: 3, │ │ │ │ │ + graphicName: "square" │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: up │ │ │ │ │ - * Finish drawing the feature │ │ │ │ │ + * Constructor: OpenLayers.Format.SLD.v1 │ │ │ │ │ + * Instances of this class are not created directly. Use the │ │ │ │ │ + * <OpenLayers.Format.SLD> constructor instead. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} The mouse up event │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - up: function(evt) { │ │ │ │ │ - this.finalize(); │ │ │ │ │ - // the mouseup method of superclass doesn't call the │ │ │ │ │ - // "done" callback if there's been no move between │ │ │ │ │ - // down and up │ │ │ │ │ - if (this.start == this.last) { │ │ │ │ │ - this.callback("done", [evt.xy]); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: out │ │ │ │ │ - * Finish drawing the feature. │ │ │ │ │ + * Method: read │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} The mouse out event │ │ │ │ │ + * data - {DOMElement} An SLD document element. │ │ │ │ │ + * options - {Object} Options for the reader. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * namedLayersAsArray - {Boolean} Generate a namedLayers array. If false, │ │ │ │ │ + * the namedLayers property value will be an object keyed by layer name. │ │ │ │ │ + * Default is false. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object representing the SLD. │ │ │ │ │ */ │ │ │ │ │ - out: function(evt) { │ │ │ │ │ - this.finalize(); │ │ │ │ │ + read: function(data, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var sld = { │ │ │ │ │ + namedLayers: options.namedLayersAsArray === true ? [] : {} │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(data, sld); │ │ │ │ │ + return sld; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createGeometry │ │ │ │ │ - * Create the new polygon geometry. This is called at the start of the │ │ │ │ │ - * drag and at any point during the drag if the number of sides │ │ │ │ │ - * changes. │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ */ │ │ │ │ │ - createGeometry: function() { │ │ │ │ │ - this.angle = Math.PI * ((1 / this.sides) - (1 / 2)); │ │ │ │ │ - if (this.snapAngle) { │ │ │ │ │ - this.angle += this.snapAngle * (Math.PI / 180); │ │ │ │ │ + readers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "sld": { │ │ │ │ │ + "StyledLayerDescriptor": function(node, sld) { │ │ │ │ │ + sld.version = node.getAttribute("version"); │ │ │ │ │ + this.readChildNodes(node, sld); │ │ │ │ │ + }, │ │ │ │ │ + "Name": function(node, obj) { │ │ │ │ │ + obj.name = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Title": function(node, obj) { │ │ │ │ │ + obj.title = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Abstract": function(node, obj) { │ │ │ │ │ + obj.description = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "NamedLayer": function(node, sld) { │ │ │ │ │ + var layer = { │ │ │ │ │ + userStyles: [], │ │ │ │ │ + namedStyles: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, layer); │ │ │ │ │ + // give each of the user styles this layer name │ │ │ │ │ + for (var i = 0, len = layer.userStyles.length; i < len; ++i) { │ │ │ │ │ + layer.userStyles[i].layerName = layer.name; │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.Util.isArray(sld.namedLayers)) { │ │ │ │ │ + sld.namedLayers.push(layer); │ │ │ │ │ + } else { │ │ │ │ │ + sld.namedLayers[layer.name] = layer; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "NamedStyle": function(node, layer) { │ │ │ │ │ + layer.namedStyles.push( │ │ │ │ │ + this.getChildName(node.firstChild) │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + "UserStyle": function(node, layer) { │ │ │ │ │ + var obj = { │ │ │ │ │ + defaultsPerSymbolizer: true, │ │ │ │ │ + rules: [] │ │ │ │ │ + }; │ │ │ │ │ + this.featureTypeCounter = -1; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + var style; │ │ │ │ │ + if (this.multipleSymbolizers) { │ │ │ │ │ + delete obj.defaultsPerSymbolizer; │ │ │ │ │ + style = new OpenLayers.Style2(obj); │ │ │ │ │ + } else { │ │ │ │ │ + style = new OpenLayers.Style(this.defaultSymbolizer, obj); │ │ │ │ │ + } │ │ │ │ │ + layer.userStyles.push(style); │ │ │ │ │ + }, │ │ │ │ │ + "IsDefault": function(node, style) { │ │ │ │ │ + if (this.getChildValue(node) == "1") { │ │ │ │ │ + style.isDefault = true; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "FeatureTypeStyle": function(node, style) { │ │ │ │ │ + ++this.featureTypeCounter; │ │ │ │ │ + var obj = { │ │ │ │ │ + rules: this.multipleSymbolizers ? style.rules : [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + if (!this.multipleSymbolizers) { │ │ │ │ │ + style.rules = obj.rules; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Rule": function(node, obj) { │ │ │ │ │ + var config; │ │ │ │ │ + if (this.multipleSymbolizers) { │ │ │ │ │ + config = { │ │ │ │ │ + symbolizers: [] │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + var rule = new OpenLayers.Rule(config); │ │ │ │ │ + this.readChildNodes(node, rule); │ │ │ │ │ + obj.rules.push(rule); │ │ │ │ │ + }, │ │ │ │ │ + "ElseFilter": function(node, rule) { │ │ │ │ │ + rule.elseFilter = true; │ │ │ │ │ + }, │ │ │ │ │ + "MinScaleDenominator": function(node, rule) { │ │ │ │ │ + rule.minScaleDenominator = parseFloat(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "MaxScaleDenominator": function(node, rule) { │ │ │ │ │ + rule.maxScaleDenominator = parseFloat(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "TextSymbolizer": function(node, rule) { │ │ │ │ │ + var config = {}; │ │ │ │ │ + this.readChildNodes(node, config); │ │ │ │ │ + if (this.multipleSymbolizers) { │ │ │ │ │ + config.zIndex = this.featureTypeCounter; │ │ │ │ │ + rule.symbolizers.push( │ │ │ │ │ + new OpenLayers.Symbolizer.Text(config) │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + rule.symbolizer["Text"] = OpenLayers.Util.applyDefaults( │ │ │ │ │ + config, rule.symbolizer["Text"] │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "LabelPlacement": function(node, symbolizer) { │ │ │ │ │ + this.readChildNodes(node, symbolizer); │ │ │ │ │ + }, │ │ │ │ │ + "PointPlacement": function(node, symbolizer) { │ │ │ │ │ + var config = {}; │ │ │ │ │ + this.readChildNodes(node, config); │ │ │ │ │ + config.labelRotation = config.rotation; │ │ │ │ │ + delete config.rotation; │ │ │ │ │ + var labelAlign, │ │ │ │ │ + x = symbolizer.labelAnchorPointX, │ │ │ │ │ + y = symbolizer.labelAnchorPointY; │ │ │ │ │ + if (x <= 1 / 3) { │ │ │ │ │ + labelAlign = 'l'; │ │ │ │ │ + } else if (x > 1 / 3 && x < 2 / 3) { │ │ │ │ │ + labelAlign = 'c'; │ │ │ │ │ + } else if (x >= 2 / 3) { │ │ │ │ │ + labelAlign = 'r'; │ │ │ │ │ + } │ │ │ │ │ + if (y <= 1 / 3) { │ │ │ │ │ + labelAlign += 'b'; │ │ │ │ │ + } else if (y > 1 / 3 && y < 2 / 3) { │ │ │ │ │ + labelAlign += 'm'; │ │ │ │ │ + } else if (y >= 2 / 3) { │ │ │ │ │ + labelAlign += 't'; │ │ │ │ │ + } │ │ │ │ │ + config.labelAlign = labelAlign; │ │ │ │ │ + OpenLayers.Util.applyDefaults(symbolizer, config); │ │ │ │ │ + }, │ │ │ │ │ + "AnchorPoint": function(node, symbolizer) { │ │ │ │ │ + this.readChildNodes(node, symbolizer); │ │ │ │ │ + }, │ │ │ │ │ + "AnchorPointX": function(node, symbolizer) { │ │ │ │ │ + var labelAnchorPointX = this.readers.ogc._expression.call(this, node); │ │ │ │ │ + // always string, could be empty string │ │ │ │ │ + if (labelAnchorPointX) { │ │ │ │ │ + symbolizer.labelAnchorPointX = labelAnchorPointX; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "AnchorPointY": function(node, symbolizer) { │ │ │ │ │ + var labelAnchorPointY = this.readers.ogc._expression.call(this, node); │ │ │ │ │ + // always string, could be empty string │ │ │ │ │ + if (labelAnchorPointY) { │ │ │ │ │ + symbolizer.labelAnchorPointY = labelAnchorPointY; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Displacement": function(node, symbolizer) { │ │ │ │ │ + this.readChildNodes(node, symbolizer); │ │ │ │ │ + }, │ │ │ │ │ + "DisplacementX": function(node, symbolizer) { │ │ │ │ │ + var labelXOffset = this.readers.ogc._expression.call(this, node); │ │ │ │ │ + // always string, could be empty string │ │ │ │ │ + if (labelXOffset) { │ │ │ │ │ + symbolizer.labelXOffset = labelXOffset; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "DisplacementY": function(node, symbolizer) { │ │ │ │ │ + var labelYOffset = this.readers.ogc._expression.call(this, node); │ │ │ │ │ + // always string, could be empty string │ │ │ │ │ + if (labelYOffset) { │ │ │ │ │ + symbolizer.labelYOffset = labelYOffset; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "LinePlacement": function(node, symbolizer) { │ │ │ │ │ + this.readChildNodes(node, symbolizer); │ │ │ │ │ + }, │ │ │ │ │ + "PerpendicularOffset": function(node, symbolizer) { │ │ │ │ │ + var labelPerpendicularOffset = this.readers.ogc._expression.call(this, node); │ │ │ │ │ + // always string, could be empty string │ │ │ │ │ + if (labelPerpendicularOffset) { │ │ │ │ │ + symbolizer.labelPerpendicularOffset = labelPerpendicularOffset; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Label": function(node, symbolizer) { │ │ │ │ │ + var value = this.readers.ogc._expression.call(this, node); │ │ │ │ │ + if (value) { │ │ │ │ │ + symbolizer.label = value; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Font": function(node, symbolizer) { │ │ │ │ │ + this.readChildNodes(node, symbolizer); │ │ │ │ │ + }, │ │ │ │ │ + "Halo": function(node, symbolizer) { │ │ │ │ │ + // halo has a fill, so send fresh object │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + symbolizer.haloRadius = obj.haloRadius; │ │ │ │ │ + symbolizer.haloColor = obj.fillColor; │ │ │ │ │ + symbolizer.haloOpacity = obj.fillOpacity; │ │ │ │ │ + }, │ │ │ │ │ + "Radius": function(node, symbolizer) { │ │ │ │ │ + var radius = this.readers.ogc._expression.call(this, node); │ │ │ │ │ + if (radius != null) { │ │ │ │ │ + // radius is only used for halo │ │ │ │ │ + symbolizer.haloRadius = radius; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "RasterSymbolizer": function(node, rule) { │ │ │ │ │ + var config = {}; │ │ │ │ │ + this.readChildNodes(node, config); │ │ │ │ │ + if (this.multipleSymbolizers) { │ │ │ │ │ + config.zIndex = this.featureTypeCounter; │ │ │ │ │ + rule.symbolizers.push( │ │ │ │ │ + new OpenLayers.Symbolizer.Raster(config) │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + rule.symbolizer["Raster"] = OpenLayers.Util.applyDefaults( │ │ │ │ │ + config, rule.symbolizer["Raster"] │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Geometry": function(node, obj) { │ │ │ │ │ + obj.geometry = {}; │ │ │ │ │ + this.readChildNodes(node, obj.geometry); │ │ │ │ │ + }, │ │ │ │ │ + "ColorMap": function(node, symbolizer) { │ │ │ │ │ + symbolizer.colorMap = []; │ │ │ │ │ + this.readChildNodes(node, symbolizer.colorMap); │ │ │ │ │ + }, │ │ │ │ │ + "ColorMapEntry": function(node, colorMap) { │ │ │ │ │ + var q = node.getAttribute("quantity"); │ │ │ │ │ + var o = node.getAttribute("opacity"); │ │ │ │ │ + colorMap.push({ │ │ │ │ │ + color: node.getAttribute("color"), │ │ │ │ │ + quantity: q !== null ? parseFloat(q) : undefined, │ │ │ │ │ + label: node.getAttribute("label") || undefined, │ │ │ │ │ + opacity: o !== null ? parseFloat(o) : undefined │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "LineSymbolizer": function(node, rule) { │ │ │ │ │ + var config = {}; │ │ │ │ │ + this.readChildNodes(node, config); │ │ │ │ │ + if (this.multipleSymbolizers) { │ │ │ │ │ + config.zIndex = this.featureTypeCounter; │ │ │ │ │ + rule.symbolizers.push( │ │ │ │ │ + new OpenLayers.Symbolizer.Line(config) │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + rule.symbolizer["Line"] = OpenLayers.Util.applyDefaults( │ │ │ │ │ + config, rule.symbolizer["Line"] │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "PolygonSymbolizer": function(node, rule) { │ │ │ │ │ + var config = { │ │ │ │ │ + fill: false, │ │ │ │ │ + stroke: false │ │ │ │ │ + }; │ │ │ │ │ + if (!this.multipleSymbolizers) { │ │ │ │ │ + config = rule.symbolizer["Polygon"] || config; │ │ │ │ │ + } │ │ │ │ │ + this.readChildNodes(node, config); │ │ │ │ │ + if (this.multipleSymbolizers) { │ │ │ │ │ + config.zIndex = this.featureTypeCounter; │ │ │ │ │ + rule.symbolizers.push( │ │ │ │ │ + new OpenLayers.Symbolizer.Polygon(config) │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + rule.symbolizer["Polygon"] = config; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "PointSymbolizer": function(node, rule) { │ │ │ │ │ + var config = { │ │ │ │ │ + fill: false, │ │ │ │ │ + stroke: false, │ │ │ │ │ + graphic: false │ │ │ │ │ + }; │ │ │ │ │ + if (!this.multipleSymbolizers) { │ │ │ │ │ + config = rule.symbolizer["Point"] || config; │ │ │ │ │ + } │ │ │ │ │ + this.readChildNodes(node, config); │ │ │ │ │ + if (this.multipleSymbolizers) { │ │ │ │ │ + config.zIndex = this.featureTypeCounter; │ │ │ │ │ + rule.symbolizers.push( │ │ │ │ │ + new OpenLayers.Symbolizer.Point(config) │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + rule.symbolizer["Point"] = config; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Stroke": function(node, symbolizer) { │ │ │ │ │ + symbolizer.stroke = true; │ │ │ │ │ + this.readChildNodes(node, symbolizer); │ │ │ │ │ + }, │ │ │ │ │ + "Fill": function(node, symbolizer) { │ │ │ │ │ + symbolizer.fill = true; │ │ │ │ │ + this.readChildNodes(node, symbolizer); │ │ │ │ │ + }, │ │ │ │ │ + "CssParameter": function(node, symbolizer) { │ │ │ │ │ + var cssProperty = node.getAttribute("name"); │ │ │ │ │ + var symProperty = this.cssMap[cssProperty]; │ │ │ │ │ + // for labels, fill should map to fontColor and fill-opacity │ │ │ │ │ + // to fontOpacity │ │ │ │ │ + if (symbolizer.label) { │ │ │ │ │ + if (cssProperty === 'fill') { │ │ │ │ │ + symProperty = "fontColor"; │ │ │ │ │ + } else if (cssProperty === 'fill-opacity') { │ │ │ │ │ + symProperty = "fontOpacity"; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (symProperty) { │ │ │ │ │ + // Limited support for parsing of OGC expressions │ │ │ │ │ + var value = this.readers.ogc._expression.call(this, node); │ │ │ │ │ + // always string, could be an empty string │ │ │ │ │ + if (value) { │ │ │ │ │ + symbolizer[symProperty] = value; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Graphic": function(node, symbolizer) { │ │ │ │ │ + symbolizer.graphic = true; │ │ │ │ │ + var graphic = {}; │ │ │ │ │ + // painter's order not respected here, clobber previous with next │ │ │ │ │ + this.readChildNodes(node, graphic); │ │ │ │ │ + // directly properties with names that match symbolizer properties │ │ │ │ │ + var properties = [ │ │ │ │ │ + "stroke", "strokeColor", "strokeWidth", "strokeOpacity", │ │ │ │ │ + "strokeLinecap", "fill", "fillColor", "fillOpacity", │ │ │ │ │ + "graphicName", "rotation", "graphicFormat" │ │ │ │ │ + ]; │ │ │ │ │ + var prop, value; │ │ │ │ │ + for (var i = 0, len = properties.length; i < len; ++i) { │ │ │ │ │ + prop = properties[i]; │ │ │ │ │ + value = graphic[prop]; │ │ │ │ │ + if (value != undefined) { │ │ │ │ │ + symbolizer[prop] = value; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // set other generic properties with specific graphic property names │ │ │ │ │ + if (graphic.opacity != undefined) { │ │ │ │ │ + symbolizer.graphicOpacity = graphic.opacity; │ │ │ │ │ + } │ │ │ │ │ + if (graphic.size != undefined) { │ │ │ │ │ + var pointRadius = graphic.size / 2; │ │ │ │ │ + if (isNaN(pointRadius)) { │ │ │ │ │ + // likely a property name │ │ │ │ │ + symbolizer.graphicWidth = graphic.size; │ │ │ │ │ + } else { │ │ │ │ │ + symbolizer.pointRadius = graphic.size / 2; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (graphic.href != undefined) { │ │ │ │ │ + symbolizer.externalGraphic = graphic.href; │ │ │ │ │ + } │ │ │ │ │ + if (graphic.rotation != undefined) { │ │ │ │ │ + symbolizer.rotation = graphic.rotation; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "ExternalGraphic": function(node, graphic) { │ │ │ │ │ + this.readChildNodes(node, graphic); │ │ │ │ │ + }, │ │ │ │ │ + "Mark": function(node, graphic) { │ │ │ │ │ + this.readChildNodes(node, graphic); │ │ │ │ │ + }, │ │ │ │ │ + "WellKnownName": function(node, graphic) { │ │ │ │ │ + graphic.graphicName = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Opacity": function(node, obj) { │ │ │ │ │ + var opacity = this.readers.ogc._expression.call(this, node); │ │ │ │ │ + // always string, could be empty string │ │ │ │ │ + if (opacity) { │ │ │ │ │ + obj.opacity = opacity; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Size": function(node, obj) { │ │ │ │ │ + var size = this.readers.ogc._expression.call(this, node); │ │ │ │ │ + // always string, could be empty string │ │ │ │ │ + if (size) { │ │ │ │ │ + obj.size = size; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Rotation": function(node, obj) { │ │ │ │ │ + var rotation = this.readers.ogc._expression.call(this, node); │ │ │ │ │ + // always string, could be empty string │ │ │ │ │ + if (rotation) { │ │ │ │ │ + obj.rotation = rotation; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "OnlineResource": function(node, obj) { │ │ │ │ │ + obj.href = this.getAttributeNS( │ │ │ │ │ + node, this.namespaces.xlink, "href" │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + "Format": function(node, graphic) { │ │ │ │ │ + graphic.graphicFormat = this.getChildValue(node); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.feature.geometry = OpenLayers.Geometry.Polygon.createRegularPolygon( │ │ │ │ │ - this.origin, this.radius, this.sides, this.snapAngle │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ + }, OpenLayers.Format.Filter.v1_0_0.prototype.readers), │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: modifyGeometry │ │ │ │ │ - * Modify the polygon geometry in place. │ │ │ │ │ + * Property: cssMap │ │ │ │ │ + * {Object} Object mapping supported css property names to OpenLayers │ │ │ │ │ + * symbolizer property names. │ │ │ │ │ */ │ │ │ │ │ - modifyGeometry: function() { │ │ │ │ │ - var angle, point; │ │ │ │ │ - var ring = this.feature.geometry.components[0]; │ │ │ │ │ - // if the number of sides ever changes, create a new geometry │ │ │ │ │ - if (ring.components.length != (this.sides + 1)) { │ │ │ │ │ - this.createGeometry(); │ │ │ │ │ - ring = this.feature.geometry.components[0]; │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0; i < this.sides; ++i) { │ │ │ │ │ - point = ring.components[i]; │ │ │ │ │ - angle = this.angle + (i * 2 * Math.PI / this.sides); │ │ │ │ │ - point.x = this.origin.x + (this.radius * Math.cos(angle)); │ │ │ │ │ - point.y = this.origin.y + (this.radius * Math.sin(angle)); │ │ │ │ │ - point.clearBounds(); │ │ │ │ │ - } │ │ │ │ │ + cssMap: { │ │ │ │ │ + "stroke": "strokeColor", │ │ │ │ │ + "stroke-opacity": "strokeOpacity", │ │ │ │ │ + "stroke-width": "strokeWidth", │ │ │ │ │ + "stroke-linecap": "strokeLinecap", │ │ │ │ │ + "stroke-dasharray": "strokeDashstyle", │ │ │ │ │ + "fill": "fillColor", │ │ │ │ │ + "fill-opacity": "fillOpacity", │ │ │ │ │ + "font-family": "fontFamily", │ │ │ │ │ + "font-size": "fontSize", │ │ │ │ │ + "font-weight": "fontWeight", │ │ │ │ │ + "font-style": "fontStyle" │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: calculateAngle │ │ │ │ │ - * Calculate the angle based on settings. │ │ │ │ │ + * Method: getCssProperty │ │ │ │ │ + * Given a symbolizer property, get the corresponding CSS property │ │ │ │ │ + * from the <cssMap>. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * sym - {String} A symbolizer property name. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A CSS property name or null if none found. │ │ │ │ │ */ │ │ │ │ │ - calculateAngle: function(point, evt) { │ │ │ │ │ - var alpha = Math.atan2(point.y - this.origin.y, │ │ │ │ │ - point.x - this.origin.x); │ │ │ │ │ - if (this.snapAngle && (this.snapToggle && !evt[this.snapToggle])) { │ │ │ │ │ - var snapAngleRad = (Math.PI / 180) * this.snapAngle; │ │ │ │ │ - this.angle = Math.round(alpha / snapAngleRad) * snapAngleRad; │ │ │ │ │ - } else { │ │ │ │ │ - this.angle = alpha; │ │ │ │ │ + getCssProperty: function(sym) { │ │ │ │ │ + var css = null; │ │ │ │ │ + for (var prop in this.cssMap) { │ │ │ │ │ + if (this.cssMap[prop] == sym) { │ │ │ │ │ + css = prop; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return css; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: cancel │ │ │ │ │ - * Finish the geometry and call the "cancel" callback. │ │ │ │ │ + * Method: getGraphicFormat │ │ │ │ │ + * Given a href for an external graphic, try to determine the mime-type. │ │ │ │ │ + * This method doesn't try too hard, and will fall back to │ │ │ │ │ + * <defaultGraphicFormat> if one of the known <graphicFormats> is not │ │ │ │ │ + * the file extension of the provided href. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * href - {String} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The graphic format. │ │ │ │ │ */ │ │ │ │ │ - cancel: function() { │ │ │ │ │ - // the polygon geometry gets cloned in the callback method │ │ │ │ │ - this.callback("cancel", null); │ │ │ │ │ - this.finalize(); │ │ │ │ │ + getGraphicFormat: function(href) { │ │ │ │ │ + var format, regex; │ │ │ │ │ + for (var key in this.graphicFormats) { │ │ │ │ │ + if (this.graphicFormats[key].test(href)) { │ │ │ │ │ + format = key; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return format || this.defaultGraphicFormat; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: finalize │ │ │ │ │ - * Finish the geometry and call the "done" callback. │ │ │ │ │ + * Property: defaultGraphicFormat │ │ │ │ │ + * {String} If none other can be determined from <getGraphicFormat>, this │ │ │ │ │ + * default will be returned. │ │ │ │ │ */ │ │ │ │ │ - finalize: function() { │ │ │ │ │ - this.origin = null; │ │ │ │ │ - this.radius = this.options.radius; │ │ │ │ │ - }, │ │ │ │ │ + defaultGraphicFormat: "image/png", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: clear │ │ │ │ │ - * Clear any rendered features on the temporary layer. This is called │ │ │ │ │ - * when the handler is deactivated, canceled, or done (unless persist │ │ │ │ │ - * is true). │ │ │ │ │ + * Property: graphicFormats │ │ │ │ │ + * {Object} Mapping of image mime-types to regular extensions matching │ │ │ │ │ + * well-known file extensions. │ │ │ │ │ */ │ │ │ │ │ - clear: function() { │ │ │ │ │ - if (this.layer) { │ │ │ │ │ - this.layer.renderer.clear(); │ │ │ │ │ - this.layer.destroyFeatures(); │ │ │ │ │ - } │ │ │ │ │ + graphicFormats: { │ │ │ │ │ + "image/jpeg": /\.jpe?g$/i, │ │ │ │ │ + "image/gif": /\.gif$/i, │ │ │ │ │ + "image/png": /\.png$/i │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: callback │ │ │ │ │ - * Trigger the control's named callback with the given arguments │ │ │ │ │ + * Method: write │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * name - {String} The key for the callback that is one of the properties │ │ │ │ │ - * of the handler's callbacks object. │ │ │ │ │ - * args - {Array} An array of arguments with which to call the callback │ │ │ │ │ - * (defined by the control). │ │ │ │ │ + * sld - {Object} An object representing the SLD. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The root of an SLD document. │ │ │ │ │ */ │ │ │ │ │ - callback: function(name, args) { │ │ │ │ │ - // override the callback method to always send the polygon geometry │ │ │ │ │ - if (this.callbacks[name]) { │ │ │ │ │ - this.callbacks[name].apply(this.control, │ │ │ │ │ - [this.feature.geometry.clone()]); │ │ │ │ │ - } │ │ │ │ │ - // since sketch features are added to the temporary layer │ │ │ │ │ - // they must be cleared here if done or cancel │ │ │ │ │ - if (!this.persist && (name == "done" || name == "cancel")) { │ │ │ │ │ - this.clear(); │ │ │ │ │ - } │ │ │ │ │ + write: function(sld) { │ │ │ │ │ + return this.writers.sld.StyledLayerDescriptor.apply(this, [sld]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.RegularPolygon" │ │ │ │ │ + /** │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ + */ │ │ │ │ │ + writers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "sld": { │ │ │ │ │ + "_OGCExpression": function(nodeName, value) { │ │ │ │ │ + // only the simplest of ogc:expression handled │ │ │ │ │ + // {label: "some text and a ${propertyName}"} │ │ │ │ │ + var node = this.createElementNSPlus(nodeName); │ │ │ │ │ + var tokens = typeof value == "string" ? │ │ │ │ │ + value.split("${") : [value]; │ │ │ │ │ + node.appendChild(this.createTextNode(tokens[0])); │ │ │ │ │ + var item, last; │ │ │ │ │ + for (var i = 1, len = tokens.length; i < len; i++) { │ │ │ │ │ + item = tokens[i]; │ │ │ │ │ + last = item.indexOf("}"); │ │ │ │ │ + if (last > 0) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "ogc:PropertyName", { │ │ │ │ │ + property: item.substring(0, last) │ │ │ │ │ + }, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + node.appendChild( │ │ │ │ │ + this.createTextNode(item.substring(++last)) │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + // no ending }, so this is a literal ${ │ │ │ │ │ + node.appendChild( │ │ │ │ │ + this.createTextNode("${" + item) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "StyledLayerDescriptor": function(sld) { │ │ │ │ │ + var root = this.createElementNSPlus( │ │ │ │ │ + "sld:StyledLayerDescriptor", { │ │ │ │ │ + attributes: { │ │ │ │ │ + "version": this.VERSION, │ │ │ │ │ + "xsi:schemaLocation": this.schemaLocation │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + // For ArcGIS Server it is necessary to define this │ │ │ │ │ + // at the root level (see ticket:2166). │ │ │ │ │ + root.setAttribute("xmlns:ogc", this.namespaces.ogc); │ │ │ │ │ + root.setAttribute("xmlns:gml", this.namespaces.gml); │ │ │ │ │ + │ │ │ │ │ + // add in optional name │ │ │ │ │ + if (sld.name) { │ │ │ │ │ + this.writeNode("Name", sld.name, root); │ │ │ │ │ + } │ │ │ │ │ + // add in optional title │ │ │ │ │ + if (sld.title) { │ │ │ │ │ + this.writeNode("Title", sld.title, root); │ │ │ │ │ + } │ │ │ │ │ + // add in optional description │ │ │ │ │ + if (sld.description) { │ │ │ │ │ + this.writeNode("Abstract", sld.description, root); │ │ │ │ │ + } │ │ │ │ │ + // add in named layers │ │ │ │ │ + // allow namedLayers to be an array │ │ │ │ │ + if (OpenLayers.Util.isArray(sld.namedLayers)) { │ │ │ │ │ + for (var i = 0, len = sld.namedLayers.length; i < len; ++i) { │ │ │ │ │ + this.writeNode("NamedLayer", sld.namedLayers[i], root); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + for (var name in sld.namedLayers) { │ │ │ │ │ + this.writeNode("NamedLayer", sld.namedLayers[name], root); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return root; │ │ │ │ │ + }, │ │ │ │ │ + "Name": function(name) { │ │ │ │ │ + return this.createElementNSPlus("sld:Name", { │ │ │ │ │ + value: name │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "Title": function(title) { │ │ │ │ │ + return this.createElementNSPlus("sld:Title", { │ │ │ │ │ + value: title │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "Abstract": function(description) { │ │ │ │ │ + return this.createElementNSPlus( │ │ │ │ │ + "sld:Abstract", { │ │ │ │ │ + value: description │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + "NamedLayer": function(layer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:NamedLayer"); │ │ │ │ │ + │ │ │ │ │ + // add in required name │ │ │ │ │ + this.writeNode("Name", layer.name, node); │ │ │ │ │ + │ │ │ │ │ + // optional sld:LayerFeatureConstraints here │ │ │ │ │ + │ │ │ │ │ + // add in named styles │ │ │ │ │ + if (layer.namedStyles) { │ │ │ │ │ + for (var i = 0, len = layer.namedStyles.length; i < len; ++i) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "NamedStyle", layer.namedStyles[i], node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // add in user styles │ │ │ │ │ + if (layer.userStyles) { │ │ │ │ │ + for (var i = 0, len = layer.userStyles.length; i < len; ++i) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "UserStyle", layer.userStyles[i], node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "NamedStyle": function(name) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:NamedStyle"); │ │ │ │ │ + this.writeNode("Name", name, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "UserStyle": function(style) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:UserStyle"); │ │ │ │ │ + │ │ │ │ │ + // add in optional name │ │ │ │ │ + if (style.name) { │ │ │ │ │ + this.writeNode("Name", style.name, node); │ │ │ │ │ + } │ │ │ │ │ + // add in optional title │ │ │ │ │ + if (style.title) { │ │ │ │ │ + this.writeNode("Title", style.title, node); │ │ │ │ │ + } │ │ │ │ │ + // add in optional description │ │ │ │ │ + if (style.description) { │ │ │ │ │ + this.writeNode("Abstract", style.description, node); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // add isdefault │ │ │ │ │ + if (style.isDefault) { │ │ │ │ │ + this.writeNode("IsDefault", style.isDefault, node); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // add FeatureTypeStyles │ │ │ │ │ + if (this.multipleSymbolizers && style.rules) { │ │ │ │ │ + // group style objects by symbolizer zIndex │ │ │ │ │ + var rulesByZ = { │ │ │ │ │ + 0: [] │ │ │ │ │ + }; │ │ │ │ │ + var zValues = [0]; │ │ │ │ │ + var rule, ruleMap, symbolizer, zIndex, clone; │ │ │ │ │ + for (var i = 0, ii = style.rules.length; i < ii; ++i) { │ │ │ │ │ + rule = style.rules[i]; │ │ │ │ │ + if (rule.symbolizers) { │ │ │ │ │ + ruleMap = {}; │ │ │ │ │ + for (var j = 0, jj = rule.symbolizers.length; j < jj; ++j) { │ │ │ │ │ + symbolizer = rule.symbolizers[j]; │ │ │ │ │ + zIndex = symbolizer.zIndex; │ │ │ │ │ + if (!(zIndex in ruleMap)) { │ │ │ │ │ + clone = rule.clone(); │ │ │ │ │ + clone.symbolizers = []; │ │ │ │ │ + ruleMap[zIndex] = clone; │ │ │ │ │ + } │ │ │ │ │ + ruleMap[zIndex].symbolizers.push(symbolizer.clone()); │ │ │ │ │ + } │ │ │ │ │ + for (zIndex in ruleMap) { │ │ │ │ │ + if (!(zIndex in rulesByZ)) { │ │ │ │ │ + zValues.push(zIndex); │ │ │ │ │ + rulesByZ[zIndex] = []; │ │ │ │ │ + } │ │ │ │ │ + rulesByZ[zIndex].push(ruleMap[zIndex]); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // no symbolizers in rule │ │ │ │ │ + rulesByZ[0].push(rule.clone()); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // write one FeatureTypeStyle per zIndex │ │ │ │ │ + zValues.sort(); │ │ │ │ │ + var rules; │ │ │ │ │ + for (var i = 0, ii = zValues.length; i < ii; ++i) { │ │ │ │ │ + rules = rulesByZ[zValues[i]]; │ │ │ │ │ + if (rules.length > 0) { │ │ │ │ │ + clone = style.clone(); │ │ │ │ │ + clone.rules = rulesByZ[zValues[i]]; │ │ │ │ │ + this.writeNode("FeatureTypeStyle", clone, node); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this.writeNode("FeatureTypeStyle", style, node); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "IsDefault": function(bool) { │ │ │ │ │ + return this.createElementNSPlus( │ │ │ │ │ + "sld:IsDefault", { │ │ │ │ │ + value: (bool) ? "1" : "0" │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + "FeatureTypeStyle": function(style) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:FeatureTypeStyle"); │ │ │ │ │ + │ │ │ │ │ + // OpenLayers currently stores no Name, Title, Abstract, │ │ │ │ │ + // FeatureTypeName, or SemanticTypeIdentifier information │ │ │ │ │ + // related to FeatureTypeStyle │ │ │ │ │ + │ │ │ │ │ + // add in rules │ │ │ │ │ + for (var i = 0, len = style.rules.length; i < len; ++i) { │ │ │ │ │ + this.writeNode("Rule", style.rules[i], node); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Rule": function(rule) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:Rule"); │ │ │ │ │ + │ │ │ │ │ + // add in optional name │ │ │ │ │ + if (rule.name) { │ │ │ │ │ + this.writeNode("Name", rule.name, node); │ │ │ │ │ + } │ │ │ │ │ + // add in optional title │ │ │ │ │ + if (rule.title) { │ │ │ │ │ + this.writeNode("Title", rule.title, node); │ │ │ │ │ + } │ │ │ │ │ + // add in optional description │ │ │ │ │ + if (rule.description) { │ │ │ │ │ + this.writeNode("Abstract", rule.description, node); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // add in LegendGraphic here │ │ │ │ │ + │ │ │ │ │ + // add in optional filters │ │ │ │ │ + if (rule.elseFilter) { │ │ │ │ │ + this.writeNode("ElseFilter", null, node); │ │ │ │ │ + } else if (rule.filter) { │ │ │ │ │ + this.writeNode("ogc:Filter", rule.filter, node); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // add in scale limits │ │ │ │ │ + if (rule.minScaleDenominator != undefined) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "MinScaleDenominator", rule.minScaleDenominator, node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (rule.maxScaleDenominator != undefined) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "MaxScaleDenominator", rule.maxScaleDenominator, node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var type, symbolizer; │ │ │ │ │ + if (this.multipleSymbolizers && rule.symbolizers) { │ │ │ │ │ + var symbolizer; │ │ │ │ │ + for (var i = 0, ii = rule.symbolizers.length; i < ii; ++i) { │ │ │ │ │ + symbolizer = rule.symbolizers[i]; │ │ │ │ │ + type = symbolizer.CLASS_NAME.split(".").pop(); │ │ │ │ │ + this.writeNode( │ │ │ │ │ + type + "Symbolizer", symbolizer, node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // add in symbolizers (relies on geometry type keys) │ │ │ │ │ + var types = OpenLayers.Style.SYMBOLIZER_PREFIXES; │ │ │ │ │ + for (var i = 0, len = types.length; i < len; ++i) { │ │ │ │ │ + type = types[i]; │ │ │ │ │ + symbolizer = rule.symbolizer[type]; │ │ │ │ │ + if (symbolizer) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + type + "Symbolizer", symbolizer, node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + "ElseFilter": function() { │ │ │ │ │ + return this.createElementNSPlus("sld:ElseFilter"); │ │ │ │ │ + }, │ │ │ │ │ + "MinScaleDenominator": function(scale) { │ │ │ │ │ + return this.createElementNSPlus( │ │ │ │ │ + "sld:MinScaleDenominator", { │ │ │ │ │ + value: scale │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + "MaxScaleDenominator": function(scale) { │ │ │ │ │ + return this.createElementNSPlus( │ │ │ │ │ + "sld:MaxScaleDenominator", { │ │ │ │ │ + value: scale │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + "LineSymbolizer": function(symbolizer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:LineSymbolizer"); │ │ │ │ │ + this.writeNode("Stroke", symbolizer, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Stroke": function(symbolizer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:Stroke"); │ │ │ │ │ + │ │ │ │ │ + // GraphicFill here │ │ │ │ │ + // GraphicStroke here │ │ │ │ │ + │ │ │ │ │ + // add in CssParameters │ │ │ │ │ + if (symbolizer.strokeColor != undefined) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "CssParameter", { │ │ │ │ │ + symbolizer: symbolizer, │ │ │ │ │ + key: "strokeColor" │ │ │ │ │ + }, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.strokeOpacity != undefined) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "CssParameter", { │ │ │ │ │ + symbolizer: symbolizer, │ │ │ │ │ + key: "strokeOpacity" │ │ │ │ │ + }, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.strokeWidth != undefined) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "CssParameter", { │ │ │ │ │ + symbolizer: symbolizer, │ │ │ │ │ + key: "strokeWidth" │ │ │ │ │ + }, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.strokeDashstyle != undefined && symbolizer.strokeDashstyle !== "solid") { │ │ │ │ │ + // assumes valid stroke-dasharray value │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "CssParameter", { │ │ │ │ │ + symbolizer: symbolizer, │ │ │ │ │ + key: "strokeDashstyle" │ │ │ │ │ + }, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.strokeLinecap != undefined) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "CssParameter", { │ │ │ │ │ + symbolizer: symbolizer, │ │ │ │ │ + key: "strokeLinecap" │ │ │ │ │ + }, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "CssParameter": function(obj) { │ │ │ │ │ + // not handling ogc:expressions for now │ │ │ │ │ + return this.createElementNSPlus("sld:CssParameter", { │ │ │ │ │ + attributes: { │ │ │ │ │ + name: this.getCssProperty(obj.key) │ │ │ │ │ + }, │ │ │ │ │ + value: obj.symbolizer[obj.key] │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "TextSymbolizer": function(symbolizer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:TextSymbolizer"); │ │ │ │ │ + // add in optional Label │ │ │ │ │ + if (symbolizer.label != null) { │ │ │ │ │ + this.writeNode("Label", symbolizer.label, node); │ │ │ │ │ + } │ │ │ │ │ + // add in optional Font │ │ │ │ │ + if (symbolizer.fontFamily != null || │ │ │ │ │ + symbolizer.fontSize != null || │ │ │ │ │ + symbolizer.fontWeight != null || │ │ │ │ │ + symbolizer.fontStyle != null) { │ │ │ │ │ + this.writeNode("Font", symbolizer, node); │ │ │ │ │ + } │ │ │ │ │ + // add in optional LabelPlacement │ │ │ │ │ + if (symbolizer.labelAnchorPointX != null || │ │ │ │ │ + symbolizer.labelAnchorPointY != null || │ │ │ │ │ + symbolizer.labelAlign != null || │ │ │ │ │ + symbolizer.labelXOffset != null || │ │ │ │ │ + symbolizer.labelYOffset != null || │ │ │ │ │ + symbolizer.labelRotation != null || │ │ │ │ │ + symbolizer.labelPerpendicularOffset != null) { │ │ │ │ │ + this.writeNode("LabelPlacement", symbolizer, node); │ │ │ │ │ + } │ │ │ │ │ + // add in optional Halo │ │ │ │ │ + if (symbolizer.haloRadius != null || │ │ │ │ │ + symbolizer.haloColor != null || │ │ │ │ │ + symbolizer.haloOpacity != null) { │ │ │ │ │ + this.writeNode("Halo", symbolizer, node); │ │ │ │ │ + } │ │ │ │ │ + // add in optional Fill │ │ │ │ │ + if (symbolizer.fontColor != null || │ │ │ │ │ + symbolizer.fontOpacity != null) { │ │ │ │ │ + this.writeNode("Fill", { │ │ │ │ │ + fillColor: symbolizer.fontColor, │ │ │ │ │ + fillOpacity: symbolizer.fontOpacity │ │ │ │ │ + }, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "LabelPlacement": function(symbolizer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:LabelPlacement"); │ │ │ │ │ + // PointPlacement and LinePlacement are choices, so don't output both │ │ │ │ │ + if ((symbolizer.labelAnchorPointX != null || │ │ │ │ │ + symbolizer.labelAnchorPointY != null || │ │ │ │ │ + symbolizer.labelAlign != null || │ │ │ │ │ + symbolizer.labelXOffset != null || │ │ │ │ │ + symbolizer.labelYOffset != null || │ │ │ │ │ + symbolizer.labelRotation != null) && │ │ │ │ │ + symbolizer.labelPerpendicularOffset == null) { │ │ │ │ │ + this.writeNode("PointPlacement", symbolizer, node); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.labelPerpendicularOffset != null) { │ │ │ │ │ + this.writeNode("LinePlacement", symbolizer, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "LinePlacement": function(symbolizer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:LinePlacement"); │ │ │ │ │ + this.writeNode("PerpendicularOffset", symbolizer.labelPerpendicularOffset, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "PerpendicularOffset": function(value) { │ │ │ │ │ + return this.createElementNSPlus("sld:PerpendicularOffset", { │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "PointPlacement": function(symbolizer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:PointPlacement"); │ │ │ │ │ + if (symbolizer.labelAnchorPointX != null || │ │ │ │ │ + symbolizer.labelAnchorPointY != null || │ │ │ │ │ + symbolizer.labelAlign != null) { │ │ │ │ │ + this.writeNode("AnchorPoint", symbolizer, node); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.labelXOffset != null || │ │ │ │ │ + symbolizer.labelYOffset != null) { │ │ │ │ │ + this.writeNode("Displacement", symbolizer, node); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.labelRotation != null) { │ │ │ │ │ + this.writeNode("Rotation", symbolizer.labelRotation, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "AnchorPoint": function(symbolizer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:AnchorPoint"); │ │ │ │ │ + var x = symbolizer.labelAnchorPointX, │ │ │ │ │ + y = symbolizer.labelAnchorPointY; │ │ │ │ │ + if (x != null) { │ │ │ │ │ + this.writeNode("AnchorPointX", x, node); │ │ │ │ │ + } │ │ │ │ │ + if (y != null) { │ │ │ │ │ + this.writeNode("AnchorPointY", y, node); │ │ │ │ │ + } │ │ │ │ │ + if (x == null && y == null) { │ │ │ │ │ + var xAlign = symbolizer.labelAlign.substr(0, 1), │ │ │ │ │ + yAlign = symbolizer.labelAlign.substr(1, 1); │ │ │ │ │ + if (xAlign === "l") { │ │ │ │ │ + x = 0; │ │ │ │ │ + } else if (xAlign === "c") { │ │ │ │ │ + x = 0.5; │ │ │ │ │ + } else if (xAlign === "r") { │ │ │ │ │ + x = 1; │ │ │ │ │ + } │ │ │ │ │ + if (yAlign === "b") { │ │ │ │ │ + y = 0; │ │ │ │ │ + } else if (yAlign === "m") { │ │ │ │ │ + y = 0.5; │ │ │ │ │ + } else if (yAlign === "t") { │ │ │ │ │ + y = 1; │ │ │ │ │ + } │ │ │ │ │ + this.writeNode("AnchorPointX", x, node); │ │ │ │ │ + this.writeNode("AnchorPointY", y, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "AnchorPointX": function(value) { │ │ │ │ │ + return this.createElementNSPlus("sld:AnchorPointX", { │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "AnchorPointY": function(value) { │ │ │ │ │ + return this.createElementNSPlus("sld:AnchorPointY", { │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "Displacement": function(symbolizer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:Displacement"); │ │ │ │ │ + if (symbolizer.labelXOffset != null) { │ │ │ │ │ + this.writeNode("DisplacementX", symbolizer.labelXOffset, node); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.labelYOffset != null) { │ │ │ │ │ + this.writeNode("DisplacementY", symbolizer.labelYOffset, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "DisplacementX": function(value) { │ │ │ │ │ + return this.createElementNSPlus("sld:DisplacementX", { │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "DisplacementY": function(value) { │ │ │ │ │ + return this.createElementNSPlus("sld:DisplacementY", { │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "Font": function(symbolizer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:Font"); │ │ │ │ │ + // add in CssParameters │ │ │ │ │ + if (symbolizer.fontFamily) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "CssParameter", { │ │ │ │ │ + symbolizer: symbolizer, │ │ │ │ │ + key: "fontFamily" │ │ │ │ │ + }, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.fontSize) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "CssParameter", { │ │ │ │ │ + symbolizer: symbolizer, │ │ │ │ │ + key: "fontSize" │ │ │ │ │ + }, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.fontWeight) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "CssParameter", { │ │ │ │ │ + symbolizer: symbolizer, │ │ │ │ │ + key: "fontWeight" │ │ │ │ │ + }, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.fontStyle) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "CssParameter", { │ │ │ │ │ + symbolizer: symbolizer, │ │ │ │ │ + key: "fontStyle" │ │ │ │ │ + }, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Label": function(label) { │ │ │ │ │ + return this.writers.sld._OGCExpression.call( │ │ │ │ │ + this, "sld:Label", label │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + "Halo": function(symbolizer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:Halo"); │ │ │ │ │ + if (symbolizer.haloRadius) { │ │ │ │ │ + this.writeNode("Radius", symbolizer.haloRadius, node); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.haloColor || symbolizer.haloOpacity) { │ │ │ │ │ + this.writeNode("Fill", { │ │ │ │ │ + fillColor: symbolizer.haloColor, │ │ │ │ │ + fillOpacity: symbolizer.haloOpacity │ │ │ │ │ + }, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Radius": function(value) { │ │ │ │ │ + return this.createElementNSPlus("sld:Radius", { │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "RasterSymbolizer": function(symbolizer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:RasterSymbolizer"); │ │ │ │ │ + if (symbolizer.geometry) { │ │ │ │ │ + this.writeNode("Geometry", symbolizer.geometry, node); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.opacity) { │ │ │ │ │ + this.writeNode("Opacity", symbolizer.opacity, node); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.colorMap) { │ │ │ │ │ + this.writeNode("ColorMap", symbolizer.colorMap, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Geometry": function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:Geometry"); │ │ │ │ │ + if (geometry.property) { │ │ │ │ │ + this.writeNode("ogc:PropertyName", geometry, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "ColorMap": function(colorMap) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:ColorMap"); │ │ │ │ │ + for (var i = 0, len = colorMap.length; i < len; ++i) { │ │ │ │ │ + this.writeNode("ColorMapEntry", colorMap[i], node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "ColorMapEntry": function(colorMapEntry) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:ColorMapEntry"); │ │ │ │ │ + var a = colorMapEntry; │ │ │ │ │ + node.setAttribute("color", a.color); │ │ │ │ │ + a.opacity !== undefined && node.setAttribute("opacity", │ │ │ │ │ + parseFloat(a.opacity)); │ │ │ │ │ + a.quantity !== undefined && node.setAttribute("quantity", │ │ │ │ │ + parseFloat(a.quantity)); │ │ │ │ │ + a.label !== undefined && node.setAttribute("label", a.label); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "PolygonSymbolizer": function(symbolizer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:PolygonSymbolizer"); │ │ │ │ │ + if (symbolizer.fill !== false) { │ │ │ │ │ + this.writeNode("Fill", symbolizer, node); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.stroke !== false) { │ │ │ │ │ + this.writeNode("Stroke", symbolizer, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Fill": function(symbolizer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:Fill"); │ │ │ │ │ + │ │ │ │ │ + // GraphicFill here │ │ │ │ │ + │ │ │ │ │ + // add in CssParameters │ │ │ │ │ + if (symbolizer.fillColor) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "CssParameter", { │ │ │ │ │ + symbolizer: symbolizer, │ │ │ │ │ + key: "fillColor" │ │ │ │ │ + }, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.fillOpacity != null) { │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "CssParameter", { │ │ │ │ │ + symbolizer: symbolizer, │ │ │ │ │ + key: "fillOpacity" │ │ │ │ │ + }, │ │ │ │ │ + node │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "PointSymbolizer": function(symbolizer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:PointSymbolizer"); │ │ │ │ │ + this.writeNode("Graphic", symbolizer, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Graphic": function(symbolizer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:Graphic"); │ │ │ │ │ + if (symbolizer.externalGraphic != undefined) { │ │ │ │ │ + this.writeNode("ExternalGraphic", symbolizer, node); │ │ │ │ │ + } else { │ │ │ │ │ + this.writeNode("Mark", symbolizer, node); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (symbolizer.graphicOpacity != undefined) { │ │ │ │ │ + this.writeNode("Opacity", symbolizer.graphicOpacity, node); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.pointRadius != undefined) { │ │ │ │ │ + this.writeNode("Size", symbolizer.pointRadius * 2, node); │ │ │ │ │ + } else if (symbolizer.graphicWidth != undefined) { │ │ │ │ │ + this.writeNode("Size", symbolizer.graphicWidth, node); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.rotation != undefined) { │ │ │ │ │ + this.writeNode("Rotation", symbolizer.rotation, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "ExternalGraphic": function(symbolizer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:ExternalGraphic"); │ │ │ │ │ + this.writeNode( │ │ │ │ │ + "OnlineResource", symbolizer.externalGraphic, node │ │ │ │ │ + ); │ │ │ │ │ + var format = symbolizer.graphicFormat || │ │ │ │ │ + this.getGraphicFormat(symbolizer.externalGraphic); │ │ │ │ │ + this.writeNode("Format", format, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Mark": function(symbolizer) { │ │ │ │ │ + var node = this.createElementNSPlus("sld:Mark"); │ │ │ │ │ + if (symbolizer.graphicName) { │ │ │ │ │ + this.writeNode("WellKnownName", symbolizer.graphicName, node); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.fill !== false) { │ │ │ │ │ + this.writeNode("Fill", symbolizer, node); │ │ │ │ │ + } │ │ │ │ │ + if (symbolizer.stroke !== false) { │ │ │ │ │ + this.writeNode("Stroke", symbolizer, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "WellKnownName": function(name) { │ │ │ │ │ + return this.createElementNSPlus("sld:WellKnownName", { │ │ │ │ │ + value: name │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "Opacity": function(value) { │ │ │ │ │ + return this.createElementNSPlus("sld:Opacity", { │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "Size": function(value) { │ │ │ │ │ + return this.writers.sld._OGCExpression.call( │ │ │ │ │ + this, "sld:Size", value │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + "Rotation": function(value) { │ │ │ │ │ + return this.createElementNSPlus("sld:Rotation", { │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "OnlineResource": function(href) { │ │ │ │ │ + return this.createElementNSPlus("sld:OnlineResource", { │ │ │ │ │ + attributes: { │ │ │ │ │ + "xlink:type": "simple", │ │ │ │ │ + "xlink:href": href │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "Format": function(format) { │ │ │ │ │ + return this.createElementNSPlus("sld:Format", { │ │ │ │ │ + value: format │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.Filter.v1_0_0.prototype.writers), │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SLD.v1" │ │ │ │ │ + │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Click.js │ │ │ │ │ + OpenLayers/Format/SLD/v1_0_0.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Format/SLD/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/Filter/v1_0_0.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Handler.Click │ │ │ │ │ - * A handler for mouse clicks. The intention of this handler is to give │ │ │ │ │ - * controls more flexibility with handling clicks. Browsers trigger │ │ │ │ │ - * click events twice for a double-click. In addition, the mousedown, │ │ │ │ │ - * mousemove, mouseup sequence fires a click event. With this handler, │ │ │ │ │ - * controls can decide whether to ignore clicks associated with a double │ │ │ │ │ - * click. By setting a <pixelTolerance>, controls can also ignore clicks │ │ │ │ │ - * that include a drag. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Handler.Click> constructor. │ │ │ │ │ + * Class: OpenLayers.Format.SLD.v1_0_0 │ │ │ │ │ + * Write SLD version 1.0.0. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ + * - <OpenLayers.Format.SLD.v1> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.Click = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: delay │ │ │ │ │ - * {Number} Number of milliseconds between clicks before the event is │ │ │ │ │ - * considered a double-click. │ │ │ │ │ - */ │ │ │ │ │ - delay: 300, │ │ │ │ │ +OpenLayers.Format.SLD.v1_0_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.SLD.v1, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: single │ │ │ │ │ - * {Boolean} Handle single clicks. Default is true. If false, clicks │ │ │ │ │ - * will not be reported. If true, single-clicks will be reported. │ │ │ │ │ - */ │ │ │ │ │ - single: true, │ │ │ │ │ + /** │ │ │ │ │ + * Constant: VERSION │ │ │ │ │ + * {String} 1.0.0 │ │ │ │ │ + */ │ │ │ │ │ + VERSION: "1.0.0", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: double │ │ │ │ │ - * {Boolean} Handle double-clicks. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - 'double': false, │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} http://www.opengis.net/sld │ │ │ │ │ + * http://schemas.opengis.net/sld/1.0.0/StyledLayerDescriptor.xsd │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: "http://www.opengis.net/sld http://schemas.opengis.net/sld/1.0.0/StyledLayerDescriptor.xsd", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: pixelTolerance │ │ │ │ │ - * {Number} Maximum number of pixels between mouseup and mousedown for an │ │ │ │ │ - * event to be considered a click. Default is 0. If set to an │ │ │ │ │ - * integer value, clicks with a drag greater than the value will be │ │ │ │ │ - * ignored. This property can only be set when the handler is │ │ │ │ │ - * constructed. │ │ │ │ │ - */ │ │ │ │ │ - pixelTolerance: 0, │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.SLD.v1_0_0 │ │ │ │ │ + * Instances of this class are not created directly. Use the │ │ │ │ │ + * <OpenLayers.Format.SLD> constructor instead. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: dblclickTolerance │ │ │ │ │ - * {Number} Maximum distance in pixels between clicks for a sequence of │ │ │ │ │ - * events to be considered a double click. Default is 13. If the │ │ │ │ │ - * distance between two clicks is greater than this value, a double- │ │ │ │ │ - * click will not be fired. │ │ │ │ │ - */ │ │ │ │ │ - dblclickTolerance: 13, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SLD.v1_0_0" │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: stopSingle │ │ │ │ │ - * {Boolean} Stop other listeners from being notified of clicks. Default │ │ │ │ │ - * is false. If true, any listeners registered before this one for │ │ │ │ │ - * click or rightclick events will not be notified. │ │ │ │ │ - */ │ │ │ │ │ - stopSingle: false, │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/SLD/v1_0_0_GeoServer.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: stopDouble │ │ │ │ │ - * {Boolean} Stop other listeners from being notified of double-clicks. │ │ │ │ │ - * Default is false. If true, any click listeners registered before │ │ │ │ │ - * this one will not be notified of *any* double-click events. │ │ │ │ │ - * │ │ │ │ │ - * The one caveat with stopDouble is that given a map with two click │ │ │ │ │ - * handlers, one with stopDouble true and the other with stopSingle │ │ │ │ │ - * true, the stopSingle handler should be activated last to get │ │ │ │ │ - * uniform cross-browser performance. Since IE triggers one click │ │ │ │ │ - * with a dblclick and FF triggers two, if a stopSingle handler is │ │ │ │ │ - * activated first, all it gets in IE is a single click when the │ │ │ │ │ - * second handler stops propagation on the dblclick. │ │ │ │ │ - */ │ │ │ │ │ - stopDouble: false, │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: timerId │ │ │ │ │ - * {Number} The id of the timeout waiting to clear the <delayedCall>. │ │ │ │ │ - */ │ │ │ │ │ - timerId: null, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/SLD/v1_0_0.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: down │ │ │ │ │ - * {Object} Object that store relevant information about the last │ │ │ │ │ - * mousedown or touchstart. Its 'xy' OpenLayers.Pixel property gives │ │ │ │ │ - * the average location of the mouse/touch event. Its 'touches' │ │ │ │ │ - * property records clientX/clientY of each touches. │ │ │ │ │ - */ │ │ │ │ │ - down: null, │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.SLD/v1_0_0_GeoServer │ │ │ │ │ + * Read and write SLD version 1.0.0 with GeoServer-specific enhanced options. │ │ │ │ │ + * See http://svn.osgeo.org/geotools/trunk/modules/extension/xsd/xsd-sld/src/main/resources/org/geotools/sld/bindings/StyledLayerDescriptor.xsd │ │ │ │ │ + * for more information. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.SLD.v1_0_0> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.SLD.v1_0_0_GeoServer = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.SLD.v1_0_0, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: last │ │ │ │ │ - * {Object} Object that store relevant information about the last │ │ │ │ │ - * mousemove or touchmove. Its 'xy' OpenLayers.Pixel property gives │ │ │ │ │ - * the average location of the mouse/touch event. Its 'touches' │ │ │ │ │ - * property records clientX/clientY of each touches. │ │ │ │ │ - */ │ │ │ │ │ - last: null, │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} The specific parser version. │ │ │ │ │ + */ │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: first │ │ │ │ │ - * {Object} When waiting for double clicks, this object will store │ │ │ │ │ - * information about the first click in a two click sequence. │ │ │ │ │ - */ │ │ │ │ │ - first: null, │ │ │ │ │ + /** │ │ │ │ │ + * Property: profile │ │ │ │ │ + * {String} The specific profile │ │ │ │ │ + */ │ │ │ │ │ + profile: "GeoServer", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: rightclickTimerId │ │ │ │ │ - * {Number} The id of the right mouse timeout waiting to clear the │ │ │ │ │ - * <delayedEvent>. │ │ │ │ │ - */ │ │ │ │ │ - rightclickTimerId: null, │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.SLD.v1_0_0_GeoServer │ │ │ │ │ + * Create a new parser for GeoServer-enhanced SLD version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Click │ │ │ │ │ - * Create a new click handler. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ - * this handler. If a handler is being used without a control, the │ │ │ │ │ - * handler's setMap method must be overridden to deal properly with │ │ │ │ │ - * the map. │ │ │ │ │ - * callbacks - {Object} An object with keys corresponding to callbacks │ │ │ │ │ - * that will be called by the handler. The callbacks should │ │ │ │ │ - * expect to recieve a single argument, the click event. │ │ │ │ │ - * Callbacks for 'click' and 'dblclick' are supported. │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * handler. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "sld": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "Priority": function(node, obj) { │ │ │ │ │ + var value = this.readers.ogc._expression.call(this, node); │ │ │ │ │ + if (value) { │ │ │ │ │ + obj.priority = value; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "VendorOption": function(node, obj) { │ │ │ │ │ + if (!obj.vendorOptions) { │ │ │ │ │ + obj.vendorOptions = {}; │ │ │ │ │ + } │ │ │ │ │ + obj.vendorOptions[node.getAttribute("name")] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "TextSymbolizer": function(node, rule) { │ │ │ │ │ + OpenLayers.Format.SLD.v1_0_0.prototype.readers.sld.TextSymbolizer.apply(this, arguments); │ │ │ │ │ + var symbolizer = this.multipleSymbolizers ? rule.symbolizers[rule.symbolizers.length - 1] : rule.symbolizer["Text"]; │ │ │ │ │ + if (symbolizer.graphic === undefined) { │ │ │ │ │ + symbolizer.graphic = false; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.SLD.v1_0_0.prototype.readers["sld"]) │ │ │ │ │ + }, OpenLayers.Format.SLD.v1_0_0.prototype.readers), │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchstart │ │ │ │ │ - * Handle touchstart. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ - */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - this.down = this.getEventInfo(evt); │ │ │ │ │ - this.last = this.getEventInfo(evt); │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ + */ │ │ │ │ │ + writers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "sld": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "Priority": function(priority) { │ │ │ │ │ + return this.writers.sld._OGCExpression.call( │ │ │ │ │ + this, "sld:Priority", priority │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ + "VendorOption": function(option) { │ │ │ │ │ + return this.createElementNSPlus("sld:VendorOption", { │ │ │ │ │ + attributes: { │ │ │ │ │ + name: option.name │ │ │ │ │ + }, │ │ │ │ │ + value: option.value │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "TextSymbolizer": function(symbolizer) { │ │ │ │ │ + var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ + var node = writers["sld"]["TextSymbolizer"].apply(this, arguments); │ │ │ │ │ + if (symbolizer.graphic !== false && (symbolizer.externalGraphic || symbolizer.graphicName)) { │ │ │ │ │ + this.writeNode("Graphic", symbolizer, node); │ │ │ │ │ + } │ │ │ │ │ + if ("priority" in symbolizer) { │ │ │ │ │ + this.writeNode("Priority", symbolizer.priority, node); │ │ │ │ │ + } │ │ │ │ │ + return this.addVendorOptions(node, symbolizer); │ │ │ │ │ + }, │ │ │ │ │ + "PointSymbolizer": function(symbolizer) { │ │ │ │ │ + var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ + var node = writers["sld"]["PointSymbolizer"].apply(this, arguments); │ │ │ │ │ + return this.addVendorOptions(node, symbolizer); │ │ │ │ │ + }, │ │ │ │ │ + "LineSymbolizer": function(symbolizer) { │ │ │ │ │ + var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ + var node = writers["sld"]["LineSymbolizer"].apply(this, arguments); │ │ │ │ │ + return this.addVendorOptions(node, symbolizer); │ │ │ │ │ + }, │ │ │ │ │ + "PolygonSymbolizer": function(symbolizer) { │ │ │ │ │ + var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ + var node = writers["sld"]["PolygonSymbolizer"].apply(this, arguments); │ │ │ │ │ + return this.addVendorOptions(node, symbolizer); │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.SLD.v1_0_0.prototype.writers["sld"]) │ │ │ │ │ + }, OpenLayers.Format.SLD.v1_0_0.prototype.writers), │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchmove │ │ │ │ │ - * Store position of last move, because touchend event can have │ │ │ │ │ - * an empty "touches" property. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ - */ │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - this.last = this.getEventInfo(evt); │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: addVendorOptions │ │ │ │ │ + * Add in the VendorOption tags and return the node again. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * node - {DOMElement} A DOM node. │ │ │ │ │ + * symbolizer - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A DOM node. │ │ │ │ │ + */ │ │ │ │ │ + addVendorOptions: function(node, symbolizer) { │ │ │ │ │ + var options = symbolizer.vendorOptions; │ │ │ │ │ + if (options) { │ │ │ │ │ + for (var key in symbolizer.vendorOptions) { │ │ │ │ │ + this.writeNode("VendorOption", { │ │ │ │ │ + name: key, │ │ │ │ │ + value: symbolizer.vendorOptions[key] │ │ │ │ │ + }, node); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: touchend │ │ │ │ │ - * Correctly set event xy property, and add lastTouches to have │ │ │ │ │ - * touches property from last touchstart or touchmove │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ - */ │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - // touchstart may not have been allowed to propagate │ │ │ │ │ - if (this.down) { │ │ │ │ │ - evt.xy = this.last.xy; │ │ │ │ │ - evt.lastTouches = this.last.touches; │ │ │ │ │ - this.handleSingle(evt); │ │ │ │ │ - this.down = null; │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SLD.v1_0_0_GeoServer" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/ArcXML/Features.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/ArcXML.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.ArcXML.Features │ │ │ │ │ + * Read/Write ArcXML features. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Format.ArcXML.Features> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.ArcXML.Features = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mousedown │ │ │ │ │ - * Handle mousedown. │ │ │ │ │ + * Constructor: OpenLayers.Format.ArcXML.Features │ │ │ │ │ + * Create a new parser/writer for ArcXML Features. Create an instance of this class │ │ │ │ │ + * to get a set of features from an ArcXML response. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - this.down = this.getEventInfo(evt); │ │ │ │ │ - this.last = this.getEventInfo(evt); │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mouseup │ │ │ │ │ - * Handle mouseup. Installed to support collection of right mouse events. │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read data from a string of ArcXML, and return a set of OpenLayers features. │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} A collection of features. │ │ │ │ │ */ │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ + read: function(data) { │ │ │ │ │ + var axl = new OpenLayers.Format.ArcXML(); │ │ │ │ │ + var parsed = axl.read(data); │ │ │ │ │ │ │ │ │ │ - // Collect right mouse clicks from the mouseup │ │ │ │ │ - // IE - ignores the second right click in mousedown so using │ │ │ │ │ - // mouseup instead │ │ │ │ │ - if (this.checkModifiers(evt) && this.control.handleRightClicks && │ │ │ │ │ - OpenLayers.Event.isRightClick(evt)) { │ │ │ │ │ - propagate = this.rightclick(evt); │ │ │ │ │ - } │ │ │ │ │ + return parsed.features.feature; │ │ │ │ │ + } │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WFSCapabilities/v1.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - return propagate; │ │ │ │ │ - }, │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: rightclick │ │ │ │ │ - * Handle rightclick. For a dblrightclick, we get two clicks so we need │ │ │ │ │ - * to always register for dblrightclick to properly handle single │ │ │ │ │ - * clicks. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ - */ │ │ │ │ │ - rightclick: function(evt) { │ │ │ │ │ - if (this.passesTolerance(evt)) { │ │ │ │ │ - if (this.rightclickTimerId != null) { │ │ │ │ │ - //Second click received before timeout this must be │ │ │ │ │ - // a double click │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.callback('dblrightclick', [evt]); │ │ │ │ │ - return !this.stopDouble; │ │ │ │ │ - } else { │ │ │ │ │ - //Set the rightclickTimerId, send evt only if double is │ │ │ │ │ - // true else trigger single │ │ │ │ │ - var clickEvent = this['double'] ? │ │ │ │ │ - OpenLayers.Util.extend({}, evt) : │ │ │ │ │ - this.callback('rightclick', [evt]); │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WFSCapabilities.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - var delayedRightCall = OpenLayers.Function.bind( │ │ │ │ │ - this.delayedRightCall, │ │ │ │ │ - this, │ │ │ │ │ - clickEvent │ │ │ │ │ - ); │ │ │ │ │ - this.rightclickTimerId = window.setTimeout( │ │ │ │ │ - delayedRightCall, this.delay │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return !this.stopSingle; │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WFSCapabilities.v1 │ │ │ │ │ + * Abstract class not to be instantiated directly. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WFSCapabilities.v1 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: delayedRightCall │ │ │ │ │ - * Sets <rightclickTimerId> to null. And optionally triggers the │ │ │ │ │ - * rightclick callback if evt is set. │ │ │ │ │ - */ │ │ │ │ │ - delayedRightCall: function(evt) { │ │ │ │ │ - this.rightclickTimerId = null; │ │ │ │ │ - if (evt) { │ │ │ │ │ - this.callback('rightclick', [evt]); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + wfs: "http://www.opengis.net/wfs", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + ows: "http://www.opengis.net/ows" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: click │ │ │ │ │ - * Handle click events from the browser. This is registered as a listener │ │ │ │ │ - * for click events and should not be called from other events in this │ │ │ │ │ - * handler. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ - */ │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - if (!this.last) { │ │ │ │ │ - this.last = this.getEventInfo(evt); │ │ │ │ │ - } │ │ │ │ │ - this.handleSingle(evt); │ │ │ │ │ - return !this.stopSingle; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: dblclick │ │ │ │ │ - * Handle dblclick. For a dblclick, we get two clicks in some browsers │ │ │ │ │ - * (FF) and one in others (IE). So we need to always register for │ │ │ │ │ - * dblclick to properly handle single clicks. This method is registered │ │ │ │ │ - * as a listener for the dblclick browser event. It should *not* be │ │ │ │ │ - * called by other methods in this handler. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ - */ │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - this.handleDouble(evt); │ │ │ │ │ - return !this.stopDouble; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: errorProperty │ │ │ │ │ + * {String} Which property of the returned object to check for in order to │ │ │ │ │ + * determine whether or not parsing has failed. In the case that the │ │ │ │ │ + * errorProperty is undefined on the returned object, the document will be │ │ │ │ │ + * run through an OGCExceptionReport parser. │ │ │ │ │ + */ │ │ │ │ │ + errorProperty: "featureTypeList", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleDouble │ │ │ │ │ - * Handle double-click sequence. │ │ │ │ │ - */ │ │ │ │ │ - handleDouble: function(evt) { │ │ │ │ │ - if (this.passesDblclickTolerance(evt)) { │ │ │ │ │ - if (this["double"]) { │ │ │ │ │ - this.callback("dblclick", [evt]); │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: "wfs", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WFSCapabilities.v1_1 │ │ │ │ │ + * Create an instance of one of the subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} List of named layers. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ } │ │ │ │ │ - // to prevent a dblclick from firing the click callback in IE │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + var raw = data; │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + return capabilities; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleSingle │ │ │ │ │ - * Handle single click sequence. │ │ │ │ │ - */ │ │ │ │ │ - handleSingle: function(evt) { │ │ │ │ │ - if (this.passesTolerance(evt)) { │ │ │ │ │ - if (this.timerId != null) { │ │ │ │ │ - // already received a click │ │ │ │ │ - if (this.last.touches && this.last.touches.length === 1) { │ │ │ │ │ - // touch device, no dblclick event - this may be a double │ │ │ │ │ - if (this["double"]) { │ │ │ │ │ - // on Android don't let the browser zoom on the page │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wfs": { │ │ │ │ │ + "WFS_Capabilities": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "FeatureTypeList": function(node, request) { │ │ │ │ │ + request.featureTypeList = { │ │ │ │ │ + featureTypes: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, request.featureTypeList); │ │ │ │ │ + }, │ │ │ │ │ + "FeatureType": function(node, featureTypeList) { │ │ │ │ │ + var featureType = {}; │ │ │ │ │ + this.readChildNodes(node, featureType); │ │ │ │ │ + featureTypeList.featureTypes.push(featureType); │ │ │ │ │ + }, │ │ │ │ │ + "Name": function(node, obj) { │ │ │ │ │ + var name = this.getChildValue(node); │ │ │ │ │ + if (name) { │ │ │ │ │ + var parts = name.split(":"); │ │ │ │ │ + obj.name = parts.pop(); │ │ │ │ │ + if (parts.length > 0) { │ │ │ │ │ + obj.featureNS = this.lookupNamespaceURI(node, parts[0]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Title": function(node, obj) { │ │ │ │ │ + var title = this.getChildValue(node); │ │ │ │ │ + if (title) { │ │ │ │ │ + obj.title = title; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Abstract": function(node, obj) { │ │ │ │ │ + var abst = this.getChildValue(node); │ │ │ │ │ + if (abst) { │ │ │ │ │ + obj["abstract"] = abst; │ │ │ │ │ } │ │ │ │ │ - this.handleDouble(evt); │ │ │ │ │ } │ │ │ │ │ - // if we're not in a touch environment we clear the click timer │ │ │ │ │ - // if we've got a second touch, we'll get two touchend events │ │ │ │ │ - if (!this.last.touches || this.last.touches.length !== 2) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WFSCapabilities/v1_1_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WFSCapabilities/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/OWSCommon/v1.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WFSCapabilities/v1_1_0 │ │ │ │ │ + * Read WFS Capabilities version 1.1.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WFSCapabilities> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WFSCapabilities.v1_1_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WFSCapabilities.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ + */ │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WFSCapabilities.v1_1_0 │ │ │ │ │ + * Create a new parser for WFS capabilities version 1.1.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "DefaultSRS": function(node, obj) { │ │ │ │ │ + var defaultSRS = this.getChildValue(node); │ │ │ │ │ + if (defaultSRS) { │ │ │ │ │ + obj.srs = defaultSRS; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - // remember the first click info so we can compare to the second │ │ │ │ │ - this.first = this.getEventInfo(evt); │ │ │ │ │ - // set the timer, send evt only if single is true │ │ │ │ │ - //use a clone of the event object because it will no longer │ │ │ │ │ - //be a valid event object in IE in the timer callback │ │ │ │ │ - var clickEvent = this.single ? │ │ │ │ │ - OpenLayers.Util.extend({}, evt) : null; │ │ │ │ │ - this.queuePotentialClick(clickEvent); │ │ │ │ │ + }, OpenLayers.Format.WFSCapabilities.v1.prototype.readers["wfs"]), │ │ │ │ │ + "ows": OpenLayers.Format.OWSCommon.v1.prototype.readers.ows │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1_1_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WFSCapabilities/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WFSCapabilities/v1.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WFSCapabilities/v1_0_0 │ │ │ │ │ + * Read WFS Capabilities version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WFSCapabilities.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WFSCapabilities.v1_0_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WFSCapabilities.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WFSCapabilities.v1_0_0 │ │ │ │ │ + * Create a new parser for WFS capabilities version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wfs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "Service": function(node, capabilities) { │ │ │ │ │ + capabilities.service = {}; │ │ │ │ │ + this.readChildNodes(node, capabilities.service); │ │ │ │ │ + }, │ │ │ │ │ + "Fees": function(node, service) { │ │ │ │ │ + var fees = this.getChildValue(node); │ │ │ │ │ + if (fees && fees.toLowerCase() != "none") { │ │ │ │ │ + service.fees = fees; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "AccessConstraints": function(node, service) { │ │ │ │ │ + var constraints = this.getChildValue(node); │ │ │ │ │ + if (constraints && constraints.toLowerCase() != "none") { │ │ │ │ │ + service.accessConstraints = constraints; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "OnlineResource": function(node, service) { │ │ │ │ │ + var onlineResource = this.getChildValue(node); │ │ │ │ │ + if (onlineResource && onlineResource.toLowerCase() != "none") { │ │ │ │ │ + service.onlineResource = onlineResource; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Keywords": function(node, service) { │ │ │ │ │ + var keywords = this.getChildValue(node); │ │ │ │ │ + if (keywords && keywords.toLowerCase() != "none") { │ │ │ │ │ + service.keywords = keywords.split(', '); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Capability": function(node, capabilities) { │ │ │ │ │ + capabilities.capability = {}; │ │ │ │ │ + this.readChildNodes(node, capabilities.capability); │ │ │ │ │ + }, │ │ │ │ │ + "Request": function(node, obj) { │ │ │ │ │ + obj.request = {}; │ │ │ │ │ + this.readChildNodes(node, obj.request); │ │ │ │ │ + }, │ │ │ │ │ + "GetFeature": function(node, request) { │ │ │ │ │ + request.getfeature = { │ │ │ │ │ + href: {}, // DCPType │ │ │ │ │ + formats: [] // ResultFormat │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, request.getfeature); │ │ │ │ │ + }, │ │ │ │ │ + "ResultFormat": function(node, obj) { │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + var childNode; │ │ │ │ │ + for (var i = 0; i < children.length; i++) { │ │ │ │ │ + childNode = children[i]; │ │ │ │ │ + if (childNode.nodeType == 1) { │ │ │ │ │ + obj.formats.push(childNode.nodeName); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "DCPType": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "HTTP": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj.href); │ │ │ │ │ + }, │ │ │ │ │ + "Get": function(node, obj) { │ │ │ │ │ + obj.get = node.getAttribute("onlineResource"); │ │ │ │ │ + }, │ │ │ │ │ + "Post": function(node, obj) { │ │ │ │ │ + obj.post = node.getAttribute("onlineResource"); │ │ │ │ │ + }, │ │ │ │ │ + "SRS": function(node, obj) { │ │ │ │ │ + var srs = this.getChildValue(node); │ │ │ │ │ + if (srs) { │ │ │ │ │ + obj.srs = srs; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WFSCapabilities.v1.prototype.readers["wfs"]) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1_0_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WMSCapabilities/v1.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WMSCapabilities.js │ │ │ │ │ + * @requires OpenLayers/Format/OGCExceptionReport.js │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WMSCapabilities.v1 │ │ │ │ │ + * Abstract class not to be instantiated directly. Creates │ │ │ │ │ + * the common parts for both WMS 1.1.X and WMS 1.3.X. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + wms: "http://www.opengis.net/wms", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: "wms", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMSCapabilities.v1 │ │ │ │ │ + * Create an instance of one of the subclasses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} List of named layers. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + var raw = data; │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + if (capabilities.service === undefined) { │ │ │ │ │ + // an exception must have occurred, so parse it │ │ │ │ │ + var parser = new OpenLayers.Format.OGCExceptionReport(); │ │ │ │ │ + capabilities.error = parser.read(raw); │ │ │ │ │ + } │ │ │ │ │ + return capabilities; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: queuePotentialClick │ │ │ │ │ - * This method is separated out largely to make testing easier (so we │ │ │ │ │ - * don't have to override window.setTimeout) │ │ │ │ │ - */ │ │ │ │ │ - queuePotentialClick: function(evt) { │ │ │ │ │ - this.timerId = window.setTimeout( │ │ │ │ │ - OpenLayers.Function.bind(this.delayedCall, this, evt), │ │ │ │ │ - this.delay │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wms": { │ │ │ │ │ + "Service": function(node, obj) { │ │ │ │ │ + obj.service = {}; │ │ │ │ │ + this.readChildNodes(node, obj.service); │ │ │ │ │ + }, │ │ │ │ │ + "Name": function(node, obj) { │ │ │ │ │ + obj.name = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Title": function(node, obj) { │ │ │ │ │ + obj.title = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Abstract": function(node, obj) { │ │ │ │ │ + obj["abstract"] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "BoundingBox": function(node, obj) { │ │ │ │ │ + var bbox = {}; │ │ │ │ │ + bbox.bbox = [ │ │ │ │ │ + parseFloat(node.getAttribute("minx")), │ │ │ │ │ + parseFloat(node.getAttribute("miny")), │ │ │ │ │ + parseFloat(node.getAttribute("maxx")), │ │ │ │ │ + parseFloat(node.getAttribute("maxy")) │ │ │ │ │ + ]; │ │ │ │ │ + var res = { │ │ │ │ │ + x: parseFloat(node.getAttribute("resx")), │ │ │ │ │ + y: parseFloat(node.getAttribute("resy")) │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: passesTolerance │ │ │ │ │ - * Determine whether the event is within the optional pixel tolerance. Note │ │ │ │ │ - * that the pixel tolerance check only works if mousedown events get to │ │ │ │ │ - * the listeners registered here. If they are stopped by other elements, │ │ │ │ │ - * the <pixelTolerance> will have no effect here (this method will always │ │ │ │ │ - * return true). │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The click is within the pixel tolerance (if specified). │ │ │ │ │ - */ │ │ │ │ │ - passesTolerance: function(evt) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - if (this.pixelTolerance != null && this.down && this.down.xy) { │ │ │ │ │ - passes = this.pixelTolerance >= this.down.xy.distanceTo(evt.xy); │ │ │ │ │ - // for touch environments, we also enforce that all touches │ │ │ │ │ - // start and end within the given tolerance to be considered a click │ │ │ │ │ - if (passes && this.touch && │ │ │ │ │ - this.down.touches.length === this.last.touches.length) { │ │ │ │ │ - // the touchend event doesn't come with touches, so we check │ │ │ │ │ - // down and last │ │ │ │ │ - for (var i = 0, ii = this.down.touches.length; i < ii; ++i) { │ │ │ │ │ - if (this.getTouchDistance( │ │ │ │ │ - this.down.touches[i], │ │ │ │ │ - this.last.touches[i] │ │ │ │ │ - ) > this.pixelTolerance) { │ │ │ │ │ - passes = false; │ │ │ │ │ - break; │ │ │ │ │ + if (!(isNaN(res.x) && isNaN(res.y))) { │ │ │ │ │ + bbox.res = res; │ │ │ │ │ + } │ │ │ │ │ + // return the bbox so that descendant classes can set the │ │ │ │ │ + // CRS and SRS and add it to the obj │ │ │ │ │ + return bbox; │ │ │ │ │ + }, │ │ │ │ │ + "OnlineResource": function(node, obj) { │ │ │ │ │ + obj.href = this.getAttributeNS(node, this.namespaces.xlink, │ │ │ │ │ + "href"); │ │ │ │ │ + }, │ │ │ │ │ + "ContactInformation": function(node, obj) { │ │ │ │ │ + obj.contactInformation = {}; │ │ │ │ │ + this.readChildNodes(node, obj.contactInformation); │ │ │ │ │ + }, │ │ │ │ │ + "ContactPersonPrimary": function(node, obj) { │ │ │ │ │ + obj.personPrimary = {}; │ │ │ │ │ + this.readChildNodes(node, obj.personPrimary); │ │ │ │ │ + }, │ │ │ │ │ + "ContactPerson": function(node, obj) { │ │ │ │ │ + obj.person = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "ContactOrganization": function(node, obj) { │ │ │ │ │ + obj.organization = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "ContactPosition": function(node, obj) { │ │ │ │ │ + obj.position = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "ContactAddress": function(node, obj) { │ │ │ │ │ + obj.contactAddress = {}; │ │ │ │ │ + this.readChildNodes(node, obj.contactAddress); │ │ │ │ │ + }, │ │ │ │ │ + "AddressType": function(node, obj) { │ │ │ │ │ + obj.type = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Address": function(node, obj) { │ │ │ │ │ + obj.address = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "City": function(node, obj) { │ │ │ │ │ + obj.city = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "StateOrProvince": function(node, obj) { │ │ │ │ │ + obj.stateOrProvince = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "PostCode": function(node, obj) { │ │ │ │ │ + obj.postcode = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Country": function(node, obj) { │ │ │ │ │ + obj.country = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "ContactVoiceTelephone": function(node, obj) { │ │ │ │ │ + obj.phone = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "ContactFacsimileTelephone": function(node, obj) { │ │ │ │ │ + obj.fax = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "ContactElectronicMailAddress": function(node, obj) { │ │ │ │ │ + obj.email = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Fees": function(node, obj) { │ │ │ │ │ + var fees = this.getChildValue(node); │ │ │ │ │ + if (fees && fees.toLowerCase() != "none") { │ │ │ │ │ + obj.fees = fees; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "AccessConstraints": function(node, obj) { │ │ │ │ │ + var constraints = this.getChildValue(node); │ │ │ │ │ + if (constraints && constraints.toLowerCase() != "none") { │ │ │ │ │ + obj.accessConstraints = constraints; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Capability": function(node, obj) { │ │ │ │ │ + obj.capability = { │ │ │ │ │ + nestedLayers: [], │ │ │ │ │ + layers: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.capability); │ │ │ │ │ + }, │ │ │ │ │ + "Request": function(node, obj) { │ │ │ │ │ + obj.request = {}; │ │ │ │ │ + this.readChildNodes(node, obj.request); │ │ │ │ │ + }, │ │ │ │ │ + "GetCapabilities": function(node, obj) { │ │ │ │ │ + obj.getcapabilities = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getcapabilities); │ │ │ │ │ + }, │ │ │ │ │ + "Format": function(node, obj) { │ │ │ │ │ + if (OpenLayers.Util.isArray(obj.formats)) { │ │ │ │ │ + obj.formats.push(this.getChildValue(node)); │ │ │ │ │ + } else { │ │ │ │ │ + obj.format = this.getChildValue(node); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "DCPType": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "HTTP": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "Get": function(node, obj) { │ │ │ │ │ + obj.get = {}; │ │ │ │ │ + this.readChildNodes(node, obj.get); │ │ │ │ │ + // backwards compatibility │ │ │ │ │ + if (!obj.href) { │ │ │ │ │ + obj.href = obj.get.href; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Post": function(node, obj) { │ │ │ │ │ + obj.post = {}; │ │ │ │ │ + this.readChildNodes(node, obj.post); │ │ │ │ │ + // backwards compatibility │ │ │ │ │ + if (!obj.href) { │ │ │ │ │ + obj.href = obj.get.href; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "GetMap": function(node, obj) { │ │ │ │ │ + obj.getmap = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getmap); │ │ │ │ │ + }, │ │ │ │ │ + "GetFeatureInfo": function(node, obj) { │ │ │ │ │ + obj.getfeatureinfo = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getfeatureinfo); │ │ │ │ │ + }, │ │ │ │ │ + "Exception": function(node, obj) { │ │ │ │ │ + obj.exception = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.exception); │ │ │ │ │ + }, │ │ │ │ │ + "Layer": function(node, obj) { │ │ │ │ │ + var parentLayer, capability; │ │ │ │ │ + if (obj.capability) { │ │ │ │ │ + capability = obj.capability; │ │ │ │ │ + parentLayer = obj; │ │ │ │ │ + } else { │ │ │ │ │ + capability = obj; │ │ │ │ │ + } │ │ │ │ │ + var attrNode = node.getAttributeNode("queryable"); │ │ │ │ │ + var queryable = (attrNode && attrNode.specified) ? │ │ │ │ │ + node.getAttribute("queryable") : null; │ │ │ │ │ + attrNode = node.getAttributeNode("cascaded"); │ │ │ │ │ + var cascaded = (attrNode && attrNode.specified) ? │ │ │ │ │ + node.getAttribute("cascaded") : null; │ │ │ │ │ + attrNode = node.getAttributeNode("opaque"); │ │ │ │ │ + var opaque = (attrNode && attrNode.specified) ? │ │ │ │ │ + node.getAttribute('opaque') : null; │ │ │ │ │ + var noSubsets = node.getAttribute('noSubsets'); │ │ │ │ │ + var fixedWidth = node.getAttribute('fixedWidth'); │ │ │ │ │ + var fixedHeight = node.getAttribute('fixedHeight'); │ │ │ │ │ + var parent = parentLayer || {}, │ │ │ │ │ + extend = OpenLayers.Util.extend; │ │ │ │ │ + var layer = { │ │ │ │ │ + nestedLayers: [], │ │ │ │ │ + styles: parentLayer ? [].concat(parentLayer.styles) : [], │ │ │ │ │ + srs: parentLayer ? extend({}, parent.srs) : {}, │ │ │ │ │ + metadataURLs: [], │ │ │ │ │ + bbox: parentLayer ? extend({}, parent.bbox) : {}, │ │ │ │ │ + llbbox: parent.llbbox, │ │ │ │ │ + dimensions: parentLayer ? extend({}, parent.dimensions) : {}, │ │ │ │ │ + authorityURLs: parentLayer ? extend({}, parent.authorityURLs) : {}, │ │ │ │ │ + identifiers: {}, │ │ │ │ │ + keywords: [], │ │ │ │ │ + queryable: (queryable && queryable !== "") ? │ │ │ │ │ + (queryable === "1" || queryable === "true") : (parent.queryable || false), │ │ │ │ │ + cascaded: (cascaded !== null) ? parseInt(cascaded) : (parent.cascaded || 0), │ │ │ │ │ + opaque: opaque ? │ │ │ │ │ + (opaque === "1" || opaque === "true") : (parent.opaque || false), │ │ │ │ │ + noSubsets: (noSubsets !== null) ? │ │ │ │ │ + (noSubsets === "1" || noSubsets === "true") : (parent.noSubsets || false), │ │ │ │ │ + fixedWidth: (fixedWidth != null) ? │ │ │ │ │ + parseInt(fixedWidth) : (parent.fixedWidth || 0), │ │ │ │ │ + fixedHeight: (fixedHeight != null) ? │ │ │ │ │ + parseInt(fixedHeight) : (parent.fixedHeight || 0), │ │ │ │ │ + minScale: parent.minScale, │ │ │ │ │ + maxScale: parent.maxScale, │ │ │ │ │ + attribution: parent.attribution │ │ │ │ │ + }; │ │ │ │ │ + obj.nestedLayers.push(layer); │ │ │ │ │ + layer.capability = capability; │ │ │ │ │ + this.readChildNodes(node, layer); │ │ │ │ │ + delete layer.capability; │ │ │ │ │ + if (layer.name) { │ │ │ │ │ + var parts = layer.name.split(":"), │ │ │ │ │ + request = capability.request, │ │ │ │ │ + gfi = request.getfeatureinfo; │ │ │ │ │ + if (parts.length > 0) { │ │ │ │ │ + layer.prefix = parts[0]; │ │ │ │ │ + } │ │ │ │ │ + capability.layers.push(layer); │ │ │ │ │ + if (layer.formats === undefined) { │ │ │ │ │ + layer.formats = request.getmap.formats; │ │ │ │ │ + } │ │ │ │ │ + if (layer.infoFormats === undefined && gfi) { │ │ │ │ │ + layer.infoFormats = gfi.formats; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ + "Attribution": function(node, obj) { │ │ │ │ │ + obj.attribution = {}; │ │ │ │ │ + this.readChildNodes(node, obj.attribution); │ │ │ │ │ + }, │ │ │ │ │ + "LogoURL": function(node, obj) { │ │ │ │ │ + obj.logo = { │ │ │ │ │ + width: node.getAttribute("width"), │ │ │ │ │ + height: node.getAttribute("height") │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.logo); │ │ │ │ │ + }, │ │ │ │ │ + "Style": function(node, obj) { │ │ │ │ │ + var style = {}; │ │ │ │ │ + obj.styles.push(style); │ │ │ │ │ + this.readChildNodes(node, style); │ │ │ │ │ + }, │ │ │ │ │ + "LegendURL": function(node, obj) { │ │ │ │ │ + var legend = { │ │ │ │ │ + width: node.getAttribute("width"), │ │ │ │ │ + height: node.getAttribute("height") │ │ │ │ │ + }; │ │ │ │ │ + obj.legend = legend; │ │ │ │ │ + this.readChildNodes(node, legend); │ │ │ │ │ + }, │ │ │ │ │ + "MetadataURL": function(node, obj) { │ │ │ │ │ + var metadataURL = { │ │ │ │ │ + type: node.getAttribute("type") │ │ │ │ │ + }; │ │ │ │ │ + obj.metadataURLs.push(metadataURL); │ │ │ │ │ + this.readChildNodes(node, metadataURL); │ │ │ │ │ + }, │ │ │ │ │ + "DataURL": function(node, obj) { │ │ │ │ │ + obj.dataURL = {}; │ │ │ │ │ + this.readChildNodes(node, obj.dataURL); │ │ │ │ │ + }, │ │ │ │ │ + "FeatureListURL": function(node, obj) { │ │ │ │ │ + obj.featureListURL = {}; │ │ │ │ │ + this.readChildNodes(node, obj.featureListURL); │ │ │ │ │ + }, │ │ │ │ │ + "AuthorityURL": function(node, obj) { │ │ │ │ │ + var name = node.getAttribute("name"); │ │ │ │ │ + var authority = {}; │ │ │ │ │ + this.readChildNodes(node, authority); │ │ │ │ │ + obj.authorityURLs[name] = authority.href; │ │ │ │ │ + }, │ │ │ │ │ + "Identifier": function(node, obj) { │ │ │ │ │ + var authority = node.getAttribute("authority"); │ │ │ │ │ + obj.identifiers[authority] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "KeywordList": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "SRS": function(node, obj) { │ │ │ │ │ + obj.srs[this.getChildValue(node)] = true; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return passes; │ │ │ │ │ - }, │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getTouchDistance │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The pixel displacement between two touches. │ │ │ │ │ - */ │ │ │ │ │ - getTouchDistance: function(from, to) { │ │ │ │ │ - return Math.sqrt( │ │ │ │ │ - Math.pow(from.clientX - to.clientX, 2) + │ │ │ │ │ - Math.pow(from.clientY - to.clientY, 2) │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1" │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: passesDblclickTolerance │ │ │ │ │ - * Determine whether the event is within the optional double-cick pixel │ │ │ │ │ - * tolerance. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The click is within the double-click pixel tolerance. │ │ │ │ │ - */ │ │ │ │ │ - passesDblclickTolerance: function(evt) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - if (this.down && this.first) { │ │ │ │ │ - passes = this.down.xy.distanceTo(this.first.xy) <= this.dblclickTolerance; │ │ │ │ │ - } │ │ │ │ │ - return passes; │ │ │ │ │ - }, │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WMSCapabilities/v1_1.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: clearTimer │ │ │ │ │ - * Clear the timer and set <timerId> to null. │ │ │ │ │ - */ │ │ │ │ │ - clearTimer: function() { │ │ │ │ │ - if (this.timerId != null) { │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - } │ │ │ │ │ - if (this.rightclickTimerId != null) { │ │ │ │ │ - window.clearTimeout(this.rightclickTimerId); │ │ │ │ │ - this.rightclickTimerId = null; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: delayedCall │ │ │ │ │ - * Sets <timerId> to null. And optionally triggers the click callback if │ │ │ │ │ - * evt is set. │ │ │ │ │ - */ │ │ │ │ │ - delayedCall: function(evt) { │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - if (evt) { │ │ │ │ │ - this.callback("click", [evt]); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WMSCapabilities/v1.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getEventInfo │ │ │ │ │ - * This method allows us to store event information without storing the │ │ │ │ │ - * actual event. In touch devices (at least), the same event is │ │ │ │ │ - * modified between touchstart, touchmove, and touchend. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object with event related info. │ │ │ │ │ - */ │ │ │ │ │ - getEventInfo: function(evt) { │ │ │ │ │ - var touches; │ │ │ │ │ - if (evt.touches) { │ │ │ │ │ - var len = evt.touches.length; │ │ │ │ │ - touches = new Array(len); │ │ │ │ │ - var touch; │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - touch = evt.touches[i]; │ │ │ │ │ - touches[i] = { │ │ │ │ │ - clientX: touch.olClientX, │ │ │ │ │ - clientY: touch.olClientY │ │ │ │ │ - }; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - xy: evt.xy, │ │ │ │ │ - touches: touches │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WMSCapabilities.v1_1 │ │ │ │ │ + * Abstract class not to be instantiated directly. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WMSCapabilities.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_1 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WMSCapabilities.v1, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the handler. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.down = null; │ │ │ │ │ - this.first = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - deactivated = true; │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "WMT_MS_Capabilities": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "Keyword": function(node, obj) { │ │ │ │ │ + if (obj.keywords) { │ │ │ │ │ + obj.keywords.push(this.getChildValue(node)); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "DescribeLayer": function(node, obj) { │ │ │ │ │ + obj.describelayer = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.describelayer); │ │ │ │ │ + }, │ │ │ │ │ + "GetLegendGraphic": function(node, obj) { │ │ │ │ │ + obj.getlegendgraphic = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getlegendgraphic); │ │ │ │ │ + }, │ │ │ │ │ + "GetStyles": function(node, obj) { │ │ │ │ │ + obj.getstyles = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getstyles); │ │ │ │ │ + }, │ │ │ │ │ + "PutStyles": function(node, obj) { │ │ │ │ │ + obj.putstyles = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.putstyles); │ │ │ │ │ + }, │ │ │ │ │ + "UserDefinedSymbolization": function(node, obj) { │ │ │ │ │ + var userSymbols = { │ │ │ │ │ + supportSLD: parseInt(node.getAttribute("SupportSLD")) == 1, │ │ │ │ │ + userLayer: parseInt(node.getAttribute("UserLayer")) == 1, │ │ │ │ │ + userStyle: parseInt(node.getAttribute("UserStyle")) == 1, │ │ │ │ │ + remoteWFS: parseInt(node.getAttribute("RemoteWFS")) == 1 │ │ │ │ │ + }; │ │ │ │ │ + obj.userSymbols = userSymbols; │ │ │ │ │ + }, │ │ │ │ │ + "LatLonBoundingBox": function(node, obj) { │ │ │ │ │ + obj.llbbox = [ │ │ │ │ │ + parseFloat(node.getAttribute("minx")), │ │ │ │ │ + parseFloat(node.getAttribute("miny")), │ │ │ │ │ + parseFloat(node.getAttribute("maxx")), │ │ │ │ │ + parseFloat(node.getAttribute("maxy")) │ │ │ │ │ + ]; │ │ │ │ │ + }, │ │ │ │ │ + "BoundingBox": function(node, obj) { │ │ │ │ │ + var bbox = OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"].BoundingBox.apply(this, [node, obj]); │ │ │ │ │ + bbox.srs = node.getAttribute("SRS"); │ │ │ │ │ + obj.bbox[bbox.srs] = bbox; │ │ │ │ │ + }, │ │ │ │ │ + "ScaleHint": function(node, obj) { │ │ │ │ │ + var min = node.getAttribute("min"); │ │ │ │ │ + var max = node.getAttribute("max"); │ │ │ │ │ + var rad2 = Math.pow(2, 0.5); │ │ │ │ │ + var ipm = OpenLayers.INCHES_PER_UNIT["m"]; │ │ │ │ │ + if (min != 0) { │ │ │ │ │ + obj.maxScale = parseFloat( │ │ │ │ │ + ((min / rad2) * ipm * │ │ │ │ │ + OpenLayers.DOTS_PER_INCH).toPrecision(13) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (max != Number.POSITIVE_INFINITY) { │ │ │ │ │ + obj.minScale = parseFloat( │ │ │ │ │ + ((max / rad2) * ipm * │ │ │ │ │ + OpenLayers.DOTS_PER_INCH).toPrecision(13) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Dimension": function(node, obj) { │ │ │ │ │ + var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ + var dim = { │ │ │ │ │ + name: name, │ │ │ │ │ + units: node.getAttribute("units"), │ │ │ │ │ + unitsymbol: node.getAttribute("unitSymbol") │ │ │ │ │ + }; │ │ │ │ │ + obj.dimensions[dim.name] = dim; │ │ │ │ │ + }, │ │ │ │ │ + "Extent": function(node, obj) { │ │ │ │ │ + var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ + if (name in obj["dimensions"]) { │ │ │ │ │ + var extent = obj.dimensions[name]; │ │ │ │ │ + extent.nearestVal = │ │ │ │ │ + node.getAttribute("nearestValue") === "1"; │ │ │ │ │ + extent.multipleVal = │ │ │ │ │ + node.getAttribute("multipleValues") === "1"; │ │ │ │ │ + extent.current = node.getAttribute("current") === "1"; │ │ │ │ │ + extent["default"] = node.getAttribute("default") || ""; │ │ │ │ │ + var values = this.getChildValue(node); │ │ │ │ │ + extent.values = values.split(","); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"]) │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Click" │ │ │ │ │ -}); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Path.js │ │ │ │ │ + OpenLayers/Format/WMSCapabilities/v1_1_0.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WMSCapabilities/v1_1.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Geometry/LineString.js │ │ │ │ │ + * Class: OpenLayers.Format.WMSCapabilities/v1_1_0 │ │ │ │ │ + * Read WMS Capabilities version 1.1.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WMSCapabilities.v1_1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_1_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WMSCapabilities.v1_1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} The specific parser version. │ │ │ │ │ + */ │ │ │ │ │ + version: "1.1.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMSCapabilities.v1_1_0 │ │ │ │ │ + * Create a new parser for WMS capabilities version 1.1.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "SRS": function(node, obj) { │ │ │ │ │ + var srs = this.getChildValue(node); │ │ │ │ │ + var values = srs.split(/ +/); │ │ │ │ │ + for (var i = 0, len = values.length; i < len; i++) { │ │ │ │ │ + obj.srs[values[i]] = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1_1.prototype.readers["wms"]) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WMSCapabilities/v1_1_1.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WMSCapabilities/v1_1.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Handler.Path │ │ │ │ │ - * Handler to draw a path on the map. Path is displayed on mouse down, │ │ │ │ │ - * moves on mouse move, and is finished on mouse up. │ │ │ │ │ + * Class: OpenLayers.Format.WMSCapabilities/v1_1_1 │ │ │ │ │ + * Read WMS Capabilities version 1.1.1. │ │ │ │ │ * │ │ │ │ │ + * Note on <ScaleHint> parsing: If the 'min' attribute is set to "0", no │ │ │ │ │ + * maxScale will be set on the layer object. If the 'max' attribute is set to │ │ │ │ │ + * "Infinity", no minScale will be set. This makes it easy to create proper │ │ │ │ │ + * {<OpenLayers.Layer.WMS>} configurations directly from the layer object │ │ │ │ │ + * literals returned by this format, because no minScale/maxScale modifications │ │ │ │ │ + * need to be made. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler.Point> │ │ │ │ │ + * - <OpenLayers.Format.WMSCapabilities.v1_1> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.Path = OpenLayers.Class(OpenLayers.Handler.Point, { │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_1_1 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WMSCapabilities.v1_1, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: line │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - line: null, │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} The specific parser version. │ │ │ │ │ + */ │ │ │ │ │ + version: "1.1.1", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: maxVertices │ │ │ │ │ - * {Number} The maximum number of vertices which can be drawn by this │ │ │ │ │ - * handler. When the number of vertices reaches maxVertices, the │ │ │ │ │ - * geometry is automatically finalized. Default is null. │ │ │ │ │ - */ │ │ │ │ │ - maxVertices: null, │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMSCapabilities.v1_1_1 │ │ │ │ │ + * Create a new parser for WMS capabilities version 1.1.1. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: doubleTouchTolerance │ │ │ │ │ - * {Number} Maximum number of pixels between two touches for │ │ │ │ │ - * the gesture to be considered a "finalize feature" action. │ │ │ │ │ - * Default is 20. │ │ │ │ │ - */ │ │ │ │ │ - doubleTouchTolerance: 20, │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "SRS": function(node, obj) { │ │ │ │ │ + obj.srs[this.getChildValue(node)] = true; │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1_1.prototype.readers["wms"]) │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: freehand │ │ │ │ │ - * {Boolean} In freehand mode, the handler starts the path on mouse down, │ │ │ │ │ - * adds a point for every mouse move, and finishes the path on mouse up. │ │ │ │ │ - * Outside of freehand mode, a point is added to the path on every mouse │ │ │ │ │ - * click and double-click finishes the path. │ │ │ │ │ - */ │ │ │ │ │ - freehand: false, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_1" │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: freehandToggle │ │ │ │ │ - * {String} If set, freehandToggle is checked on mouse events and will set │ │ │ │ │ - * the freehand mode to the opposite of this.freehand. To disallow │ │ │ │ │ - * toggling between freehand and non-freehand mode, set freehandToggle to │ │ │ │ │ - * null. Acceptable toggle values are 'shiftKey', 'ctrlKey', and 'altKey'. │ │ │ │ │ - */ │ │ │ │ │ - freehandToggle: 'shiftKey', │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WMSCapabilities/v1_1_1_WMSC.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: timerId │ │ │ │ │ - * {Integer} The timer used to test the double touch. │ │ │ │ │ - */ │ │ │ │ │ - timerId: null, │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: redoStack │ │ │ │ │ - * {Array} Stack containing points removed with <undo>. │ │ │ │ │ - */ │ │ │ │ │ - redoStack: null, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WMSCapabilities/v1_1_1.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Path │ │ │ │ │ - * Create a new path hander │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ - * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ - * functions. Various callbacks described below. │ │ │ │ │ - * options - {Object} An optional object with properties to be set on the │ │ │ │ │ - * handler │ │ │ │ │ - * │ │ │ │ │ - * Named callbacks: │ │ │ │ │ - * create - Called when a sketch is first created. Callback called with │ │ │ │ │ - * the creation point geometry and sketch feature. │ │ │ │ │ - * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ - * geometry and the sketch feature. │ │ │ │ │ - * point - Called as each point is added. Receives the new point geometry. │ │ │ │ │ - * done - Called when the point drawing is finished. The callback will │ │ │ │ │ - * recieve a single argument, the linestring geometry. │ │ │ │ │ - * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ - * cancel callback will receive a geometry. │ │ │ │ │ - */ │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WMSCapabilities/v1_1_1_WMSC │ │ │ │ │ + * Read WMS-C Capabilities version 1.1.1. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WMSCapabilities.v1_1_1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_1_1_WMSC = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WMSCapabilities.v1_1_1, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createFeature │ │ │ │ │ - * Add temporary geometries │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} The initial pixel location for the new │ │ │ │ │ - * feature. │ │ │ │ │ - */ │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ - lonlat.lon, lonlat.lat │ │ │ │ │ - ); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.line = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.LineString([this.point.geometry]) │ │ │ │ │ - ); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.line, this.point], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} The specific parser version. │ │ │ │ │ + */ │ │ │ │ │ + version: "1.1.1", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: profile │ │ │ │ │ + * {String} The specific profile │ │ │ │ │ + */ │ │ │ │ │ + profile: "WMSC", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMSCapabilities.v1_1_1 │ │ │ │ │ + * Create a new parser for WMS-C capabilities version 1.1.1. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "VendorSpecificCapabilities": function(node, obj) { │ │ │ │ │ + obj.vendorSpecific = { │ │ │ │ │ + tileSets: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.vendorSpecific); │ │ │ │ │ + }, │ │ │ │ │ + "TileSet": function(node, vendorSpecific) { │ │ │ │ │ + var tileset = { │ │ │ │ │ + srs: {}, │ │ │ │ │ + bbox: {}, │ │ │ │ │ + resolutions: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, tileset); │ │ │ │ │ + vendorSpecific.tileSets.push(tileset); │ │ │ │ │ + }, │ │ │ │ │ + "Resolutions": function(node, tileset) { │ │ │ │ │ + var res = this.getChildValue(node).split(" "); │ │ │ │ │ + for (var i = 0, len = res.length; i < len; i++) { │ │ │ │ │ + if (res[i] != "") { │ │ │ │ │ + tileset.resolutions.push(parseFloat(res[i])); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Width": function(node, tileset) { │ │ │ │ │ + tileset.width = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "Height": function(node, tileset) { │ │ │ │ │ + tileset.height = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "Layers": function(node, tileset) { │ │ │ │ │ + tileset.layers = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Styles": function(node, tileset) { │ │ │ │ │ + tileset.styles = this.getChildValue(node); │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1_1_1.prototype.readers["wms"]) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_1_WMSC" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WMSCapabilities/v1_3.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WMSCapabilities/v1.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WMSCapabilities/v1_3 │ │ │ │ │ + * Abstract base class for WMS Capabilities version 1.3.X. │ │ │ │ │ + * SLD 1.1.0 adds in the extra operations DescribeLayer and GetLegendGraphic, │ │ │ │ │ + * see: http://schemas.opengis.net/sld/1.1.0/sld_capabilities.xsd │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WMSCapabilities.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_3 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WMSCapabilities.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wms": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "WMS_Capabilities": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "LayerLimit": function(node, obj) { │ │ │ │ │ + obj.layerLimit = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "MaxWidth": function(node, obj) { │ │ │ │ │ + obj.maxWidth = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "MaxHeight": function(node, obj) { │ │ │ │ │ + obj.maxHeight = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "BoundingBox": function(node, obj) { │ │ │ │ │ + var bbox = OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"].BoundingBox.apply(this, [node, obj]); │ │ │ │ │ + bbox.srs = node.getAttribute("CRS"); │ │ │ │ │ + obj.bbox[bbox.srs] = bbox; │ │ │ │ │ + }, │ │ │ │ │ + "CRS": function(node, obj) { │ │ │ │ │ + // CRS is the synonym of SRS │ │ │ │ │ + this.readers.wms.SRS.apply(this, [node, obj]); │ │ │ │ │ + }, │ │ │ │ │ + "EX_GeographicBoundingBox": function(node, obj) { │ │ │ │ │ + // replacement of LatLonBoundingBox │ │ │ │ │ + obj.llbbox = []; │ │ │ │ │ + this.readChildNodes(node, obj.llbbox); │ │ │ │ │ + │ │ │ │ │ + }, │ │ │ │ │ + "westBoundLongitude": function(node, obj) { │ │ │ │ │ + obj[0] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "eastBoundLongitude": function(node, obj) { │ │ │ │ │ + obj[2] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "southBoundLatitude": function(node, obj) { │ │ │ │ │ + obj[1] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "northBoundLatitude": function(node, obj) { │ │ │ │ │ + obj[3] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "MinScaleDenominator": function(node, obj) { │ │ │ │ │ + obj.maxScale = parseFloat(this.getChildValue(node)).toPrecision(16); │ │ │ │ │ + }, │ │ │ │ │ + "MaxScaleDenominator": function(node, obj) { │ │ │ │ │ + obj.minScale = parseFloat(this.getChildValue(node)).toPrecision(16); │ │ │ │ │ + }, │ │ │ │ │ + "Dimension": function(node, obj) { │ │ │ │ │ + // dimension has extra attributes: default, multipleValues, │ │ │ │ │ + // nearestValue, current which used to be part of Extent. It now │ │ │ │ │ + // also contains the values. │ │ │ │ │ + var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ + var dim = { │ │ │ │ │ + name: name, │ │ │ │ │ + units: node.getAttribute("units"), │ │ │ │ │ + unitsymbol: node.getAttribute("unitSymbol"), │ │ │ │ │ + nearestVal: node.getAttribute("nearestValue") === "1", │ │ │ │ │ + multipleVal: node.getAttribute("multipleValues") === "1", │ │ │ │ │ + "default": node.getAttribute("default") || "", │ │ │ │ │ + current: node.getAttribute("current") === "1", │ │ │ │ │ + values: this.getChildValue(node).split(",") │ │ │ │ │ + │ │ │ │ │ + }; │ │ │ │ │ + // Theoretically there can be more dimensions with the same │ │ │ │ │ + // name, but with a different unit. Until we meet such a case, │ │ │ │ │ + // let's just keep the same structure as the WMS 1.1 │ │ │ │ │ + // GetCapabilities parser uses. We will store the last │ │ │ │ │ + // one encountered. │ │ │ │ │ + obj.dimensions[dim.name] = dim; │ │ │ │ │ + }, │ │ │ │ │ + "Keyword": function(node, obj) { │ │ │ │ │ + // TODO: should we change the structure of keyword in v1.js? │ │ │ │ │ + // Make it an object with a value instead of a string? │ │ │ │ │ + var keyword = { │ │ │ │ │ + value: this.getChildValue(node), │ │ │ │ │ + vocabulary: node.getAttribute("vocabulary") │ │ │ │ │ + }; │ │ │ │ │ + if (obj.keywords) { │ │ │ │ │ + obj.keywords.push(keyword); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"]), │ │ │ │ │ + "sld": { │ │ │ │ │ + "UserDefinedSymbolization": function(node, obj) { │ │ │ │ │ + this.readers.wms.UserDefinedSymbolization.apply(this, [node, obj]); │ │ │ │ │ + // add the two extra attributes │ │ │ │ │ + obj.userSymbols.inlineFeature = parseInt(node.getAttribute("InlineFeature")) == 1; │ │ │ │ │ + obj.userSymbols.remoteWCS = parseInt(node.getAttribute("RemoteWCS")) == 1; │ │ │ │ │ + }, │ │ │ │ │ + "DescribeLayer": function(node, obj) { │ │ │ │ │ + this.readers.wms.DescribeLayer.apply(this, [node, obj]); │ │ │ │ │ + }, │ │ │ │ │ + "GetLegendGraphic": function(node, obj) { │ │ │ │ │ + this.readers.wms.GetLegendGraphic.apply(this, [node, obj]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_3" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WMSCapabilities/v1_3_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WMSCapabilities/v1_3.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WMSCapabilities/v1_3_0 │ │ │ │ │ + * Read WMS Capabilities version 1.3.0. │ │ │ │ │ + * SLD 1.1.0 adds in the extra operations DescribeLayer and GetLegendGraphic, │ │ │ │ │ + * see: http://schemas.opengis.net/sld/1.1.0/sld_capabilities.xsd │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WMSCapabilities.v1_3> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_3_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WMSCapabilities.v1_3, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} The specific parser version. │ │ │ │ │ + */ │ │ │ │ │ + version: "1.3.0", │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_3_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WMTSCapabilities/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WMTSCapabilities.js │ │ │ │ │ + * @requires OpenLayers/Format/OWSCommon/v1_1_0.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WMTSCapabilities.v1_0_0 │ │ │ │ │ + * Read WMTS Capabilities version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WMTSCapabilities> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WMTSCapabilities.v1_0_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.OWSCommon.v1_1_0, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} The parser version ("1.0.0"). │ │ │ │ │ + */ │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ + wmts: "http://www.opengis.net/wmts/1.0", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: yx │ │ │ │ │ + * {Object} Members in the yx object are used to determine if a CRS URN │ │ │ │ │ + * corresponds to a CRS with y,x axis order. Member names are CRS URNs │ │ │ │ │ + * and values are boolean. Defaults come from the │ │ │ │ │ + * <OpenLayers.Format.WMTSCapabilities> prototype. │ │ │ │ │ + */ │ │ │ │ │ + yx: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + * {String} The default namespace alias for creating element nodes. │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: "wmts", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMTSCapabilities.v1_0_0 │ │ │ │ │ + * Create a new parser for WMTS capabilities version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.options = options; │ │ │ │ │ + var yx = OpenLayers.Util.extend({}, OpenLayers.Format.WMTSCapabilities.prototype.yx); │ │ │ │ │ + this.yx = OpenLayers.Util.extend(yx, this.yx); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return info about the WMTS. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Information about the SOS service. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + capabilities.version = this.version; │ │ │ │ │ + return capabilities; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wmts": { │ │ │ │ │ + "Capabilities": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "Contents": function(node, obj) { │ │ │ │ │ + obj.contents = {}; │ │ │ │ │ + obj.contents.layers = []; │ │ │ │ │ + obj.contents.tileMatrixSets = {}; │ │ │ │ │ + this.readChildNodes(node, obj.contents); │ │ │ │ │ + }, │ │ │ │ │ + "Layer": function(node, obj) { │ │ │ │ │ + var layer = { │ │ │ │ │ + styles: [], │ │ │ │ │ + formats: [], │ │ │ │ │ + dimensions: [], │ │ │ │ │ + tileMatrixSetLinks: [] │ │ │ │ │ + }; │ │ │ │ │ + layer.layers = []; │ │ │ │ │ + this.readChildNodes(node, layer); │ │ │ │ │ + obj.layers.push(layer); │ │ │ │ │ + }, │ │ │ │ │ + "Style": function(node, obj) { │ │ │ │ │ + var style = {}; │ │ │ │ │ + style.isDefault = (node.getAttribute("isDefault") === "true"); │ │ │ │ │ + this.readChildNodes(node, style); │ │ │ │ │ + obj.styles.push(style); │ │ │ │ │ + }, │ │ │ │ │ + "Format": function(node, obj) { │ │ │ │ │ + obj.formats.push(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "TileMatrixSetLink": function(node, obj) { │ │ │ │ │ + var tileMatrixSetLink = {}; │ │ │ │ │ + this.readChildNodes(node, tileMatrixSetLink); │ │ │ │ │ + obj.tileMatrixSetLinks.push(tileMatrixSetLink); │ │ │ │ │ + }, │ │ │ │ │ + "TileMatrixSet": function(node, obj) { │ │ │ │ │ + // node could be child of wmts:Contents or wmts:TileMatrixSetLink │ │ │ │ │ + // duck type wmts:Contents by looking for layers │ │ │ │ │ + if (obj.layers) { │ │ │ │ │ + // TileMatrixSet as object type in schema │ │ │ │ │ + var tileMatrixSet = { │ │ │ │ │ + matrixIds: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, tileMatrixSet); │ │ │ │ │ + obj.tileMatrixSets[tileMatrixSet.identifier] = tileMatrixSet; │ │ │ │ │ + } else { │ │ │ │ │ + // TileMatrixSet as string type in schema │ │ │ │ │ + obj.tileMatrixSet = this.getChildValue(node); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "TileMatrix": function(node, obj) { │ │ │ │ │ + var tileMatrix = { │ │ │ │ │ + supportedCRS: obj.supportedCRS │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, tileMatrix); │ │ │ │ │ + obj.matrixIds.push(tileMatrix); │ │ │ │ │ + }, │ │ │ │ │ + "ScaleDenominator": function(node, obj) { │ │ │ │ │ + obj.scaleDenominator = parseFloat(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "TopLeftCorner": function(node, obj) { │ │ │ │ │ + var topLeftCorner = this.getChildValue(node); │ │ │ │ │ + var coords = topLeftCorner.split(" "); │ │ │ │ │ + // decide on axis order for the given CRS │ │ │ │ │ + var yx; │ │ │ │ │ + if (obj.supportedCRS) { │ │ │ │ │ + // extract out version from URN │ │ │ │ │ + var crs = obj.supportedCRS.replace( │ │ │ │ │ + /urn:ogc:def:crs:(\w+):.+:(\w+)$/, │ │ │ │ │ + "urn:ogc:def:crs:$1::$2" │ │ │ │ │ + ); │ │ │ │ │ + yx = !!this.yx[crs]; │ │ │ │ │ + } │ │ │ │ │ + if (yx) { │ │ │ │ │ + obj.topLeftCorner = new OpenLayers.LonLat( │ │ │ │ │ + coords[1], coords[0] │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + obj.topLeftCorner = new OpenLayers.LonLat( │ │ │ │ │ + coords[0], coords[1] │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "TileWidth": function(node, obj) { │ │ │ │ │ + obj.tileWidth = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "TileHeight": function(node, obj) { │ │ │ │ │ + obj.tileHeight = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "MatrixWidth": function(node, obj) { │ │ │ │ │ + obj.matrixWidth = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "MatrixHeight": function(node, obj) { │ │ │ │ │ + obj.matrixHeight = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "ResourceURL": function(node, obj) { │ │ │ │ │ + obj.resourceUrl = obj.resourceUrl || {}; │ │ │ │ │ + var resourceType = node.getAttribute("resourceType"); │ │ │ │ │ + if (!obj.resourceUrls) { │ │ │ │ │ + obj.resourceUrls = []; │ │ │ │ │ + } │ │ │ │ │ + var resourceUrl = obj.resourceUrl[resourceType] = { │ │ │ │ │ + format: node.getAttribute("format"), │ │ │ │ │ + template: node.getAttribute("template"), │ │ │ │ │ + resourceType: resourceType │ │ │ │ │ + }; │ │ │ │ │ + obj.resourceUrls.push(resourceUrl); │ │ │ │ │ + }, │ │ │ │ │ + // not used for now, can be added in the future though │ │ │ │ │ + /*"Themes": function(node, obj) { │ │ │ │ │ + obj.themes = []; │ │ │ │ │ + this.readChildNodes(node, obj.themes); │ │ │ │ │ + }, │ │ │ │ │ + "Theme": function(node, obj) { │ │ │ │ │ + var theme = {}; │ │ │ │ │ + this.readChildNodes(node, theme); │ │ │ │ │ + obj.push(theme); │ │ │ │ │ + },*/ │ │ │ │ │ + "WSDL": function(node, obj) { │ │ │ │ │ + obj.wsdl = {}; │ │ │ │ │ + obj.wsdl.href = node.getAttribute("xlink:href"); │ │ │ │ │ + // TODO: other attributes of <WSDL> element │ │ │ │ │ + }, │ │ │ │ │ + "ServiceMetadataURL": function(node, obj) { │ │ │ │ │ + obj.serviceMetadataUrl = {}; │ │ │ │ │ + obj.serviceMetadataUrl.href = node.getAttribute("xlink:href"); │ │ │ │ │ + // TODO: other attributes of <ServiceMetadataURL> element │ │ │ │ │ + }, │ │ │ │ │ + "LegendURL": function(node, obj) { │ │ │ │ │ + obj.legend = {}; │ │ │ │ │ + obj.legend.href = node.getAttribute("xlink:href"); │ │ │ │ │ + obj.legend.format = node.getAttribute("format"); │ │ │ │ │ + }, │ │ │ │ │ + "Dimension": function(node, obj) { │ │ │ │ │ + var dimension = { │ │ │ │ │ + values: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, dimension); │ │ │ │ │ + obj.dimensions.push(dimension); │ │ │ │ │ + }, │ │ │ │ │ + "Default": function(node, obj) { │ │ │ │ │ + obj["default"] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Value": function(node, obj) { │ │ │ │ │ + obj.values.push(this.getChildValue(node)); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "ows": OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMTSCapabilities.v1_0_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WPSCapabilities/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WPSCapabilities.js │ │ │ │ │ + * @requires OpenLayers/Format/OWSCommon/v1_1_0.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WPSCapabilities.v1_0_0 │ │ │ │ │ + * Read WPS Capabilities version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WPSCapabilities.v1_0_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ + wps: "http://www.opengis.net/wps/1.0.0", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ + */ │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WPSCapabilities.v1_0_0 │ │ │ │ │ + * Create a new parser for WPS capabilities version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return info about the WPS. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Information about the WPS service. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + return capabilities; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wps": { │ │ │ │ │ + "Capabilities": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "ProcessOfferings": function(node, obj) { │ │ │ │ │ + obj.processOfferings = {}; │ │ │ │ │ + this.readChildNodes(node, obj.processOfferings); │ │ │ │ │ + }, │ │ │ │ │ + "Process": function(node, processOfferings) { │ │ │ │ │ + var processVersion = this.getAttributeNS(node, this.namespaces.wps, "processVersion"); │ │ │ │ │ + var process = { │ │ │ │ │ + processVersion: processVersion │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, process); │ │ │ │ │ + processOfferings[process.identifier] = process; │ │ │ │ │ + }, │ │ │ │ │ + "Languages": function(node, obj) { │ │ │ │ │ + obj.languages = []; │ │ │ │ │ + this.readChildNodes(node, obj.languages); │ │ │ │ │ + }, │ │ │ │ │ + "Default": function(node, languages) { │ │ │ │ │ + var language = { │ │ │ │ │ + isDefault: true │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, language); │ │ │ │ │ + languages.push(language); │ │ │ │ │ + }, │ │ │ │ │ + "Supported": function(node, languages) { │ │ │ │ │ + var language = {}; │ │ │ │ │ + this.readChildNodes(node, language); │ │ │ │ │ + languages.push(language); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "ows": OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WPSCapabilities.v1_0_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/OWSContext/v0_3_1.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ + * @requires OpenLayers/Format/KML.js │ │ │ │ │ + * @requires OpenLayers/Format/GML.js │ │ │ │ │ + * @requires OpenLayers/Format/GML/v2.js │ │ │ │ │ + * @requires OpenLayers/Format/SLD/v1_0_0.js │ │ │ │ │ + * @requires OpenLayers/Format/OWSContext.js │ │ │ │ │ + * @requires OpenLayers/Format/OWSCommon/v1_0_0.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.OWSContext.v0_3_1 │ │ │ │ │ + * Read and write OWSContext version 0.3.1. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.OWSContext.v0_3_1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroyFeature │ │ │ │ │ - * Destroy temporary geometries │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ */ │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - OpenLayers.Handler.Point.prototype.destroyFeature.call( │ │ │ │ │ - this, force); │ │ │ │ │ - this.line = null; │ │ │ │ │ + namespaces: { │ │ │ │ │ + owc: "http://www.opengis.net/ows-context", │ │ │ │ │ + gml: "http://www.opengis.net/gml", │ │ │ │ │ + kml: "http://www.opengis.net/kml/2.2", │ │ │ │ │ + ogc: "http://www.opengis.net/ogc", │ │ │ │ │ + ows: "http://www.opengis.net/ows", │ │ │ │ │ + sld: "http://www.opengis.net/sld", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroyPersistedFeature │ │ │ │ │ - * Destroy the persisted feature. │ │ │ │ │ + * Constant: VERSION │ │ │ │ │ + * {String} 0.3.1 │ │ │ │ │ */ │ │ │ │ │ - destroyPersistedFeature: function() { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - if (layer && layer.features.length > 2) { │ │ │ │ │ - this.layer.features[0].destroy(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + VERSION: "0.3.1", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removePoint │ │ │ │ │ - * Destroy the temporary point. │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} Schema location │ │ │ │ │ */ │ │ │ │ │ - removePoint: function() { │ │ │ │ │ - if (this.point) { │ │ │ │ │ - this.layer.removeFeatures([this.point]); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + schemaLocation: "http://www.opengis.net/ows-context http://www.ogcnetwork.net/schemas/owc/0.3.1/owsContext.xsd", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: addPoint │ │ │ │ │ - * Add point to geometry. Send the point index to override │ │ │ │ │ - * the behavior of LinearRing that disregards adding duplicate points. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} The pixel location for the new point. │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + * {String} Default namespace prefix to use. │ │ │ │ │ */ │ │ │ │ │ - addPoint: function(pixel) { │ │ │ │ │ - this.layer.removeFeatures([this.point]); │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat) │ │ │ │ │ - ); │ │ │ │ │ - this.line.geometry.addComponent( │ │ │ │ │ - this.point.geometry, this.line.geometry.components.length │ │ │ │ │ - ); │ │ │ │ │ - this.layer.addFeatures([this.point]); │ │ │ │ │ - this.callback("point", [this.point.geometry, this.getGeometry()]); │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - delete this.redoStack; │ │ │ │ │ - }, │ │ │ │ │ + defaultPrefix: "owc", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: insertXY │ │ │ │ │ - * Insert a point in the current sketch given x & y coordinates. The new │ │ │ │ │ - * point is inserted immediately before the most recently drawn point. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {Number} The x-coordinate of the point. │ │ │ │ │ - * y - {Number} The y-coordinate of the point. │ │ │ │ │ + * APIProperty: extractAttributes │ │ │ │ │ + * {Boolean} Extract attributes from GML. Default is true. │ │ │ │ │ */ │ │ │ │ │ - insertXY: function(x, y) { │ │ │ │ │ - this.line.geometry.addComponent( │ │ │ │ │ - new OpenLayers.Geometry.Point(x, y), │ │ │ │ │ - this.getCurrentPointIndex() │ │ │ │ │ - ); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - delete this.redoStack; │ │ │ │ │ - }, │ │ │ │ │ + extractAttributes: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: insertDeltaXY │ │ │ │ │ - * Insert a point given offsets from the previously inserted point. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * dx - {Number} The x-coordinate offset of the point. │ │ │ │ │ - * dy - {Number} The y-coordinate offset of the point. │ │ │ │ │ + * APIProperty: xy │ │ │ │ │ + * {Boolean} Order of the GML coordinate true:(x,y) or false:(y,x) │ │ │ │ │ + * Changing is not recommended, a new Format should be instantiated. │ │ │ │ │ */ │ │ │ │ │ - insertDeltaXY: function(dx, dy) { │ │ │ │ │ - var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ - var p0 = this.line.geometry.components[previousIndex]; │ │ │ │ │ - if (p0 && !isNaN(p0.x) && !isNaN(p0.y)) { │ │ │ │ │ - this.insertXY(p0.x + dx, p0.y + dy); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + xy: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: insertDirectionLength │ │ │ │ │ - * Insert a point in the current sketch given a direction and a length. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * direction - {Number} Degrees clockwise from the positive x-axis. │ │ │ │ │ - * length - {Number} Distance from the previously drawn point. │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ */ │ │ │ │ │ - insertDirectionLength: function(direction, length) { │ │ │ │ │ - direction *= Math.PI / 180; │ │ │ │ │ - var dx = length * Math.cos(direction); │ │ │ │ │ - var dy = length * Math.sin(direction); │ │ │ │ │ - this.insertDeltaXY(dx, dy); │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: insertDeflectionLength │ │ │ │ │ - * Insert a point in the current sketch given a deflection and a length. │ │ │ │ │ - * The deflection should be degrees clockwise from the previously │ │ │ │ │ - * digitized segment. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * deflection - {Number} Degrees clockwise from the previous segment. │ │ │ │ │ - * length - {Number} Distance from the previously drawn point. │ │ │ │ │ + * Property: featureNS │ │ │ │ │ + * {String} The namespace uri to use for writing InlineGeometry │ │ │ │ │ */ │ │ │ │ │ - insertDeflectionLength: function(deflection, length) { │ │ │ │ │ - var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ - if (previousIndex > 0) { │ │ │ │ │ - var p1 = this.line.geometry.components[previousIndex]; │ │ │ │ │ - var p0 = this.line.geometry.components[previousIndex - 1]; │ │ │ │ │ - var theta = Math.atan2(p1.y - p0.y, p1.x - p0.x); │ │ │ │ │ - this.insertDirectionLength( │ │ │ │ │ - (theta * 180 / Math.PI) + deflection, length │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + featureNS: "http://mapserver.gis.umn.edu/mapserver", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getCurrentPointIndex │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number} The index of the most recently drawn point. │ │ │ │ │ + * Property: featureType │ │ │ │ │ + * {String} The name to use as the feature type when writing out │ │ │ │ │ + * InlineGeometry │ │ │ │ │ */ │ │ │ │ │ - getCurrentPointIndex: function() { │ │ │ │ │ - return this.line.geometry.components.length - 1; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ + featureType: 'vector', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: undo │ │ │ │ │ - * Remove the most recently added point in the sketch geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} A point was removed. │ │ │ │ │ + * Property: geometryName │ │ │ │ │ + * {String} The name to use for the geometry attribute when writing out │ │ │ │ │ + * InlineGeometry │ │ │ │ │ */ │ │ │ │ │ - undo: function() { │ │ │ │ │ - var geometry = this.line.geometry; │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var index = this.getCurrentPointIndex() - 1; │ │ │ │ │ - var target = components[index]; │ │ │ │ │ - var undone = geometry.removeComponent(target); │ │ │ │ │ - if (undone) { │ │ │ │ │ - // On touch devices, set the current ("mouse location") point to │ │ │ │ │ - // match the last digitized point. │ │ │ │ │ - if (this.touch && index > 0) { │ │ │ │ │ - components = geometry.components; // safety │ │ │ │ │ - var lastpt = components[index - 1]; │ │ │ │ │ - var curptidx = this.getCurrentPointIndex(); │ │ │ │ │ - var curpt = components[curptidx]; │ │ │ │ │ - curpt.x = lastpt.x; │ │ │ │ │ - curpt.y = lastpt.y; │ │ │ │ │ - } │ │ │ │ │ - if (!this.redoStack) { │ │ │ │ │ - this.redoStack = []; │ │ │ │ │ - } │ │ │ │ │ - this.redoStack.push(target); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - } │ │ │ │ │ - return undone; │ │ │ │ │ - }, │ │ │ │ │ + geometryName: 'geometry', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: redo │ │ │ │ │ - * Reinsert the most recently removed point resulting from an <undo> call. │ │ │ │ │ - * The undo stack is deleted whenever a point is added by other means. │ │ │ │ │ + * Property: nestingLayerLookup │ │ │ │ │ + * {Object} Hashtable lookup for nesting layer nodes. Used while writing │ │ │ │ │ + * the OWS context document. It is necessary to keep track of the │ │ │ │ │ + * nestingPaths for which nesting layer nodes have already been │ │ │ │ │ + * created, so (nesting) layer nodes are added to those nodes. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} A point was added. │ │ │ │ │ + * For example: │ │ │ │ │ + * │ │ │ │ │ + * If there are three layers with nestingPaths: │ │ │ │ │ + * layer1.metadata.nestingPath = "a/b/" │ │ │ │ │ + * layer2.metadata.nestingPath = "a/b/" │ │ │ │ │ + * layer2.metadata.nestingPath = "a/c" │ │ │ │ │ + * │ │ │ │ │ + * then a nesting layer node "a" should be created once and added │ │ │ │ │ + * to the resource list, a nesting layer node "b" should be created │ │ │ │ │ + * once and added under "a", and a nesting layer node "c" should be │ │ │ │ │ + * created and added under "a". The lookup paths for these nodes │ │ │ │ │ + * will be "a", "a/b", and "a/c" respectively. │ │ │ │ │ */ │ │ │ │ │ - redo: function() { │ │ │ │ │ - var target = this.redoStack && this.redoStack.pop(); │ │ │ │ │ - if (target) { │ │ │ │ │ - this.line.geometry.addComponent(target, this.getCurrentPointIndex()); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - } │ │ │ │ │ - return !!target; │ │ │ │ │ - }, │ │ │ │ │ + nestingLayerLookup: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: freehandMode │ │ │ │ │ - * Determine whether to behave in freehand mode or not. │ │ │ │ │ + * Constructor: OpenLayers.Format.OWSContext.v0_3_1 │ │ │ │ │ + * Instances of this class are not created directly. Use the │ │ │ │ │ + * <OpenLayers.Format.OWSContext> constructor instead. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - freehandMode: function(evt) { │ │ │ │ │ - return (this.freehandToggle && evt[this.freehandToggle]) ? │ │ │ │ │ - !this.freehand : this.freehand; │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + OpenLayers.Format.GML.v2.prototype.setGeometryTypes.call(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: modifyFeature │ │ │ │ │ - * Modify the existing geometry given the new point │ │ │ │ │ + * Method: setNestingPath │ │ │ │ │ + * Set the nestingPath property of the layer depending on the position │ │ │ │ │ + * of the layer in hierarchy of layers. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} The updated pixel location for the latest │ │ │ │ │ - * point. │ │ │ │ │ - * drawing - {Boolean} Indicate if we're currently drawing. │ │ │ │ │ + * l - {Object} An object that may have a layersContext array property. │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - modifyFeature: function(pixel, drawing) { │ │ │ │ │ - if (!this.line) { │ │ │ │ │ - this.createFeature(pixel); │ │ │ │ │ + setNestingPath: function(l) { │ │ │ │ │ + if (l.layersContext) { │ │ │ │ │ + for (var i = 0, len = l.layersContext.length; i < len; i++) { │ │ │ │ │ + var layerContext = l.layersContext[i]; │ │ │ │ │ + var nPath = []; │ │ │ │ │ + var nTitle = l.title || ""; │ │ │ │ │ + if (l.metadata && l.metadata.nestingPath) { │ │ │ │ │ + nPath = l.metadata.nestingPath.slice(); │ │ │ │ │ + } │ │ │ │ │ + if (nTitle != "") { │ │ │ │ │ + nPath.push(nTitle); │ │ │ │ │ + } │ │ │ │ │ + layerContext.metadata.nestingPath = nPath; │ │ │ │ │ + if (layerContext.layersContext) { │ │ │ │ │ + this.setNestingPath(layerContext); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point.geometry.x = lonlat.lon; │ │ │ │ │ - this.point.geometry.y = lonlat.lat; │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.getSketch(), drawing]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawFeature │ │ │ │ │ - * Render geometries on the temporary layer. │ │ │ │ │ - */ │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.line, this.style); │ │ │ │ │ - this.layer.drawFeature(this.point, this.style); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getSketch │ │ │ │ │ - * Return the sketch feature. │ │ │ │ │ + * Function: decomposeNestingPath │ │ │ │ │ + * Takes a nestingPath like "a/b/c" and decomposes it into subpaths: │ │ │ │ │ + * "a", "a/b", "a/b/c" │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * nPath - {Array} the nesting path │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * Array({String}) Array with subpaths, or empty array if there is nothing │ │ │ │ │ + * to decompose │ │ │ │ │ */ │ │ │ │ │ - getSketch: function() { │ │ │ │ │ - return this.line; │ │ │ │ │ + decomposeNestingPath: function(nPath) { │ │ │ │ │ + var a = []; │ │ │ │ │ + if (OpenLayers.Util.isArray(nPath)) { │ │ │ │ │ + var path = nPath.slice(); │ │ │ │ │ + while (path.length > 0) { │ │ │ │ │ + a.push(path.slice()); │ │ │ │ │ + path.pop(); │ │ │ │ │ + } │ │ │ │ │ + a.reverse(); │ │ │ │ │ + } │ │ │ │ │ + return a; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getGeometry │ │ │ │ │ - * Return the sketch geometry. If <multi> is true, this will return │ │ │ │ │ - * a multi-part geometry. │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read OWS context data from a string or DOMElement, and return a list │ │ │ │ │ + * of layers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.LineString>} │ │ │ │ │ + * {Object} The context object with a flat layer list as a property named │ │ │ │ │ + * layersContext. │ │ │ │ │ */ │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.line && this.line.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiLineString([geometry]); │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ } │ │ │ │ │ - return geometry; │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + var context = {}; │ │ │ │ │ + this.readNode(data, context); │ │ │ │ │ + // since an OWSContext can be nested we need to go through this │ │ │ │ │ + // structure recursively │ │ │ │ │ + this.setNestingPath({ │ │ │ │ │ + layersContext: context.layersContext │ │ │ │ │ + }); │ │ │ │ │ + // after nesting path has been set, create a flat list of layers │ │ │ │ │ + var layers = []; │ │ │ │ │ + this.processLayer(layers, context); │ │ │ │ │ + delete context.layersContext; │ │ │ │ │ + context.layersContext = layers; │ │ │ │ │ + return context; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * method: touchstart │ │ │ │ │ - * handle touchstart. │ │ │ │ │ - * │ │ │ │ │ - * parameters: │ │ │ │ │ - * evt - {event} the browser event │ │ │ │ │ + * Method: processLayer │ │ │ │ │ + * Recursive function to get back a flat list of layers from the hierarchic │ │ │ │ │ + * layer structure. │ │ │ │ │ * │ │ │ │ │ - * returns: │ │ │ │ │ - * {boolean} allow event propagation │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layerArray - {Array({Object})} Array of layerContext objects │ │ │ │ │ + * layer - {Object} layerContext object │ │ │ │ │ */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - if (this.timerId && │ │ │ │ │ - this.passesTolerance(this.lastTouchPx, evt.xy, │ │ │ │ │ - this.doubleTouchTolerance)) { │ │ │ │ │ - // double-tap, finalize the geometry │ │ │ │ │ - this.finishGeometry(); │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - return false; │ │ │ │ │ - } else { │ │ │ │ │ - if (this.timerId) { │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ + processLayer: function(layerArray, layer) { │ │ │ │ │ + if (layer.layersContext) { │ │ │ │ │ + for (var i = 0, len = layer.layersContext.length; i < len; i++) { │ │ │ │ │ + var l = layer.layersContext[i]; │ │ │ │ │ + layerArray.push(l); │ │ │ │ │ + if (l.layersContext) { │ │ │ │ │ + this.processLayer(layerArray, l); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.timerId = window.setTimeout( │ │ │ │ │ - OpenLayers.Function.bind(function() { │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - }, this), 300); │ │ │ │ │ - return OpenLayers.Handler.Point.prototype.touchstart.call(this, evt); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: down │ │ │ │ │ - * Handle mousedown and touchstart. Add a new point to the geometry and │ │ │ │ │ - * render it. Return determines whether to propagate the event on the map. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: write │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ + * context - {Object} An object representing the map context. │ │ │ │ │ + * options - {Object} Optional object. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} An OWS Context document string. │ │ │ │ │ */ │ │ │ │ │ - down: function(evt) { │ │ │ │ │ - var stopDown = this.stopDown; │ │ │ │ │ - if (this.freehandMode(evt)) { │ │ │ │ │ - stopDown = true; │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!this.touch && (!this.lastDown || │ │ │ │ │ - !this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ - this.pixelTolerance))) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ - } │ │ │ │ │ - this.mouseDown = true; │ │ │ │ │ - this.lastDown = evt.xy; │ │ │ │ │ - this.stoppedDown = stopDown; │ │ │ │ │ - return !stopDown; │ │ │ │ │ + write: function(context, options) { │ │ │ │ │ + var name = "OWSContext"; │ │ │ │ │ + this.nestingLayerLookup = {}; //start with empty lookup │ │ │ │ │ + options = options || {}; │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, context); │ │ │ │ │ + var root = this.writeNode(name, options); │ │ │ │ │ + this.nestingLayerLookup = null; //clear lookup │ │ │ │ │ + this.setAttributeNS( │ │ │ │ │ + root, this.namespaces["xsi"], │ │ │ │ │ + "xsi:schemaLocation", this.schemaLocation │ │ │ │ │ + ); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [root]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: move │ │ │ │ │ - * Handle mousemove and touchmove. Adjust the geometry and redraw. │ │ │ │ │ - * Return determines whether to propagate the event on the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ */ │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature(); │ │ │ │ │ + readers: { │ │ │ │ │ + "kml": { │ │ │ │ │ + "Document": function(node, obj) { │ │ │ │ │ + obj.features = new OpenLayers.Format.KML({ │ │ │ │ │ + kmlns: this.namespaces.kml, │ │ │ │ │ + extractStyles: true │ │ │ │ │ + }).read(node); │ │ │ │ │ } │ │ │ │ │ - if (this.maxVertices && this.line && │ │ │ │ │ - this.line.geometry.components.length === this.maxVertices) { │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize(); │ │ │ │ │ - } else { │ │ │ │ │ - this.addPoint(evt.xy); │ │ │ │ │ + }, │ │ │ │ │ + "owc": { │ │ │ │ │ + "OWSContext": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "General": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "ResourceList": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "Layer": function(node, obj) { │ │ │ │ │ + var layerContext = { │ │ │ │ │ + metadata: {}, │ │ │ │ │ + visibility: (node.getAttribute("hidden") != "1"), │ │ │ │ │ + queryable: (node.getAttribute("queryable") == "1"), │ │ │ │ │ + opacity: ((node.getAttribute("opacity") != null) ? │ │ │ │ │ + parseFloat(node.getAttribute("opacity")) : null), │ │ │ │ │ + name: node.getAttribute("name"), │ │ │ │ │ + /* A category layer is a dummy layer meant for creating │ │ │ │ │ + hierarchies. It is not a physical layer in the │ │ │ │ │ + OpenLayers sense. The assumption we make here is that │ │ │ │ │ + category layers do not have a name attribute */ │ │ │ │ │ + categoryLayer: (node.getAttribute("name") == null), │ │ │ │ │ + formats: [], │ │ │ │ │ + styles: [] │ │ │ │ │ + }; │ │ │ │ │ + if (!obj.layersContext) { │ │ │ │ │ + obj.layersContext = []; │ │ │ │ │ + } │ │ │ │ │ + obj.layersContext.push(layerContext); │ │ │ │ │ + this.readChildNodes(node, layerContext); │ │ │ │ │ + }, │ │ │ │ │ + "InlineGeometry": function(node, obj) { │ │ │ │ │ + obj.features = []; │ │ │ │ │ + var elements = this.getElementsByTagNameNS(node, │ │ │ │ │ + this.namespaces.gml, "featureMember"); │ │ │ │ │ + var el; │ │ │ │ │ + if (elements.length >= 1) { │ │ │ │ │ + el = elements[0]; │ │ │ │ │ + } │ │ │ │ │ + if (el && el.firstChild) { │ │ │ │ │ + var featurenode = (el.firstChild.nextSibling) ? │ │ │ │ │ + el.firstChild.nextSibling : el.firstChild; │ │ │ │ │ + this.setNamespace("feature", featurenode.namespaceURI); │ │ │ │ │ + this.featureType = featurenode.localName || │ │ │ │ │ + featurenode.nodeName.split(":").pop(); │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Server": function(node, obj) { │ │ │ │ │ + // when having multiple Server types, we prefer WMS │ │ │ │ │ + if ((!obj.service && !obj.version) || │ │ │ │ │ + (obj.service != │ │ │ │ │ + OpenLayers.Format.Context.serviceTypes.WMS)) { │ │ │ │ │ + obj.service = node.getAttribute("service"); │ │ │ │ │ + obj.version = node.getAttribute("version"); │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "Name": function(node, obj) { │ │ │ │ │ + obj.name = this.getChildValue(node); │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "Title": function(node, obj) { │ │ │ │ │ + obj.title = this.getChildValue(node); │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "StyleList": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj.styles); │ │ │ │ │ + }, │ │ │ │ │ + "Style": function(node, obj) { │ │ │ │ │ + var style = {}; │ │ │ │ │ + obj.push(style); │ │ │ │ │ + this.readChildNodes(node, style); │ │ │ │ │ + }, │ │ │ │ │ + "LegendURL": function(node, obj) { │ │ │ │ │ + var legend = {}; │ │ │ │ │ + obj.legend = legend; │ │ │ │ │ + this.readChildNodes(node, legend); │ │ │ │ │ + }, │ │ │ │ │ + "OnlineResource": function(node, obj) { │ │ │ │ │ + obj.url = this.getAttributeNS(node, this.namespaces.xlink, │ │ │ │ │ + "href"); │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ } │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ + }, │ │ │ │ │ + "ows": OpenLayers.Format.OWSCommon.v1_0_0.prototype.readers.ows, │ │ │ │ │ + "gml": OpenLayers.Format.GML.v2.prototype.readers.gml, │ │ │ │ │ + "sld": OpenLayers.Format.SLD.v1_0_0.prototype.readers.sld, │ │ │ │ │ + "feature": OpenLayers.Format.GML.v2.prototype.readers.feature │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: up │ │ │ │ │ - * Handle mouseup and touchend. Send the latest point in the geometry to │ │ │ │ │ - * the control. Return determines whether to propagate the event on the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ */ │ │ │ │ │ - up: function(evt) { │ │ │ │ │ - if (this.mouseDown && (!this.lastUp || !this.lastUp.equals(evt.xy))) { │ │ │ │ │ - if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature(); │ │ │ │ │ + writers: { │ │ │ │ │ + "owc": { │ │ │ │ │ + "OWSContext": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("OWSContext", { │ │ │ │ │ + attributes: { │ │ │ │ │ + version: this.VERSION, │ │ │ │ │ + id: options.id || OpenLayers.Util.createUniqueID("OpenLayers_OWSContext_") │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + this.writeNode("General", options, node); │ │ │ │ │ + this.writeNode("ResourceList", options, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "General": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("General"); │ │ │ │ │ + this.writeNode("ows:BoundingBox", options, node); │ │ │ │ │ + this.writeNode("ows:Title", options.title || 'OpenLayers OWSContext', node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "ResourceList": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("ResourceList"); │ │ │ │ │ + for (var i = 0, len = options.layers.length; i < len; i++) { │ │ │ │ │ + var layer = options.layers[i]; │ │ │ │ │ + var decomposedPath = this.decomposeNestingPath(layer.metadata.nestingPath); │ │ │ │ │ + this.writeNode("_Layer", { │ │ │ │ │ + layer: layer, │ │ │ │ │ + subPaths: decomposedPath │ │ │ │ │ + }, node); │ │ │ │ │ } │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize(); │ │ │ │ │ - } else { │ │ │ │ │ - if (this.passesTolerance(this.lastDown, evt.xy, │ │ │ │ │ - this.pixelTolerance)) { │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Server": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("Server", { │ │ │ │ │ + attributes: { │ │ │ │ │ + version: options.version, │ │ │ │ │ + service: options.service │ │ │ │ │ } │ │ │ │ │ - if (this.lastUp == null && this.persist) { │ │ │ │ │ - this.destroyPersistedFeature(); │ │ │ │ │ + }); │ │ │ │ │ + this.writeNode("OnlineResource", options, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "OnlineResource": function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("OnlineResource", { │ │ │ │ │ + attributes: { │ │ │ │ │ + "xlink:href": options.url │ │ │ │ │ } │ │ │ │ │ - this.addPoint(evt.xy); │ │ │ │ │ - this.lastUp = evt.xy; │ │ │ │ │ - if (this.line.geometry.components.length === this.maxVertices + 1) { │ │ │ │ │ - this.finishGeometry(); │ │ │ │ │ + }); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "InlineGeometry": function(layer) { │ │ │ │ │ + var node = this.createElementNSPlus("InlineGeometry"), │ │ │ │ │ + dataExtent = layer.getDataExtent(); │ │ │ │ │ + if (dataExtent !== null) { │ │ │ │ │ + this.writeNode("gml:boundedBy", dataExtent, node); │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, len = layer.features.length; i < len; i++) { │ │ │ │ │ + this.writeNode("gml:featureMember", layer.features[i], node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "StyleList": function(styles) { │ │ │ │ │ + var node = this.createElementNSPlus("StyleList"); │ │ │ │ │ + for (var i = 0, len = styles.length; i < len; i++) { │ │ │ │ │ + this.writeNode("Style", styles[i], node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Style": function(style) { │ │ │ │ │ + var node = this.createElementNSPlus("Style"); │ │ │ │ │ + this.writeNode("Name", style, node); │ │ │ │ │ + this.writeNode("Title", style, node); │ │ │ │ │ + if (style.legend) { │ │ │ │ │ + this.writeNode("LegendURL", style, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Name": function(obj) { │ │ │ │ │ + var node = this.createElementNSPlus("Name", { │ │ │ │ │ + value: obj.name │ │ │ │ │ + }); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Title": function(obj) { │ │ │ │ │ + var node = this.createElementNSPlus("Title", { │ │ │ │ │ + value: obj.title │ │ │ │ │ + }); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "LegendURL": function(style) { │ │ │ │ │ + var node = this.createElementNSPlus("LegendURL"); │ │ │ │ │ + this.writeNode("OnlineResource", style.legend, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "_WMS": function(layer) { │ │ │ │ │ + var node = this.createElementNSPlus("Layer", { │ │ │ │ │ + attributes: { │ │ │ │ │ + name: layer.params.LAYERS, │ │ │ │ │ + queryable: layer.queryable ? "1" : "0", │ │ │ │ │ + hidden: layer.visibility ? "0" : "1", │ │ │ │ │ + opacity: layer.hasOwnProperty("opacity") ? layer.opacity : null │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + this.writeNode("ows:Title", layer.name, node); │ │ │ │ │ + this.writeNode("ows:OutputFormat", layer.params.FORMAT, node); │ │ │ │ │ + this.writeNode("Server", { │ │ │ │ │ + service: OpenLayers.Format.Context.serviceTypes.WMS, │ │ │ │ │ + version: layer.params.VERSION, │ │ │ │ │ + url: layer.url │ │ │ │ │ + }, node); │ │ │ │ │ + if (layer.metadata.styles && layer.metadata.styles.length > 0) { │ │ │ │ │ + this.writeNode("StyleList", layer.metadata.styles, node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "_Layer": function(options) { │ │ │ │ │ + var layer, subPaths, node, title; │ │ │ │ │ + layer = options.layer; │ │ │ │ │ + subPaths = options.subPaths; │ │ │ │ │ + node = null; │ │ │ │ │ + title = null; │ │ │ │ │ + // subPaths is an array of an array │ │ │ │ │ + // recursively calling _Layer writer eats up subPaths, until a │ │ │ │ │ + // real writer is called and nodes are returned. │ │ │ │ │ + if (subPaths.length > 0) { │ │ │ │ │ + var path = subPaths[0].join("/"); │ │ │ │ │ + var index = path.lastIndexOf("/"); │ │ │ │ │ + node = this.nestingLayerLookup[path]; │ │ │ │ │ + title = (index > 0) ? path.substring(index + 1, path.length) : path; │ │ │ │ │ + if (!node) { │ │ │ │ │ + // category layer │ │ │ │ │ + node = this.createElementNSPlus("Layer"); │ │ │ │ │ + this.writeNode("ows:Title", title, node); │ │ │ │ │ + this.nestingLayerLookup[path] = node; │ │ │ │ │ } │ │ │ │ │ + options.subPaths.shift(); //remove a path after each call │ │ │ │ │ + this.writeNode("_Layer", options, node); │ │ │ │ │ + return node; │ │ │ │ │ + } else { │ │ │ │ │ + // write out the actual layer │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.WMS) { │ │ │ │ │ + node = this.writeNode("_WMS", layer); │ │ │ │ │ + } else if (layer instanceof OpenLayers.Layer.Vector) { │ │ │ │ │ + if (layer.protocol instanceof OpenLayers.Protocol.WFS.v1) { │ │ │ │ │ + node = this.writeNode("_WFS", layer); │ │ │ │ │ + } else if (layer.protocol instanceof OpenLayers.Protocol.HTTP) { │ │ │ │ │ + if (layer.protocol.format instanceof OpenLayers.Format.GML) { │ │ │ │ │ + layer.protocol.format.version = "2.1.2"; │ │ │ │ │ + node = this.writeNode("_GML", layer); │ │ │ │ │ + } else if (layer.protocol.format instanceof OpenLayers.Format.KML) { │ │ │ │ │ + layer.protocol.format.version = "2.2"; │ │ │ │ │ + node = this.writeNode("_KML", layer); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // write out as inline GML since we have no idea │ │ │ │ │ + // about the original Format │ │ │ │ │ + this.setNamespace("feature", this.featureNS); │ │ │ │ │ + node = this.writeNode("_InlineGeometry", layer); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (layer.options.maxScale) { │ │ │ │ │ + this.writeNode("sld:MinScaleDenominator", │ │ │ │ │ + layer.options.maxScale, node); │ │ │ │ │ + } │ │ │ │ │ + if (layer.options.minScale) { │ │ │ │ │ + this.writeNode("sld:MaxScaleDenominator", │ │ │ │ │ + layer.options.minScale, node); │ │ │ │ │ + } │ │ │ │ │ + this.nestingLayerLookup[layer.name] = node; │ │ │ │ │ + return node; │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ + "_WFS": function(layer) { │ │ │ │ │ + var node = this.createElementNSPlus("Layer", { │ │ │ │ │ + attributes: { │ │ │ │ │ + name: layer.protocol.featurePrefix + ":" + layer.protocol.featureType, │ │ │ │ │ + hidden: layer.visibility ? "0" : "1" │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + this.writeNode("ows:Title", layer.name, node); │ │ │ │ │ + this.writeNode("Server", { │ │ │ │ │ + service: OpenLayers.Format.Context.serviceTypes.WFS, │ │ │ │ │ + version: layer.protocol.version, │ │ │ │ │ + url: layer.protocol.url │ │ │ │ │ + }, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "_InlineGeometry": function(layer) { │ │ │ │ │ + var node = this.createElementNSPlus("Layer", { │ │ │ │ │ + attributes: { │ │ │ │ │ + name: this.featureType, │ │ │ │ │ + hidden: layer.visibility ? "0" : "1" │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + this.writeNode("ows:Title", layer.name, node); │ │ │ │ │ + this.writeNode("InlineGeometry", layer, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "_GML": function(layer) { │ │ │ │ │ + var node = this.createElementNSPlus("Layer"); │ │ │ │ │ + this.writeNode("ows:Title", layer.name, node); │ │ │ │ │ + this.writeNode("Server", { │ │ │ │ │ + service: OpenLayers.Format.Context.serviceTypes.GML, │ │ │ │ │ + url: layer.protocol.url, │ │ │ │ │ + version: layer.protocol.format.version │ │ │ │ │ + }, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "_KML": function(layer) { │ │ │ │ │ + var node = this.createElementNSPlus("Layer"); │ │ │ │ │ + this.writeNode("ows:Title", layer.name, node); │ │ │ │ │ + this.writeNode("Server", { │ │ │ │ │ + service: OpenLayers.Format.Context.serviceTypes.KML, │ │ │ │ │ + version: layer.protocol.format.version, │ │ │ │ │ + url: layer.protocol.url │ │ │ │ │ + }, node); │ │ │ │ │ + return node; │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - return !this.stopUp; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: finishGeometry │ │ │ │ │ - * Finish the geometry and send it back to the control. │ │ │ │ │ - */ │ │ │ │ │ - finishGeometry: function() { │ │ │ │ │ - var index = this.line.geometry.components.length - 1; │ │ │ │ │ - this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize(); │ │ │ │ │ + }, │ │ │ │ │ + "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "boundedBy": function(bounds) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:boundedBy"); │ │ │ │ │ + this.writeNode("gml:Box", bounds, node); │ │ │ │ │ + return node; │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.GML.v2.prototype.writers.gml), │ │ │ │ │ + "ows": OpenLayers.Format.OWSCommon.v1_0_0.prototype.writers.ows, │ │ │ │ │ + "sld": OpenLayers.Format.SLD.v1_0_0.prototype.writers.sld, │ │ │ │ │ + "feature": OpenLayers.Format.GML.v2.prototype.writers.feature │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: dblclick │ │ │ │ │ - * Handle double-clicks. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} The browser event │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow event propagation │ │ │ │ │ - */ │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - if (!this.freehandMode(evt)) { │ │ │ │ │ - this.finishGeometry(); │ │ │ │ │ - } │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.OWSContext.v0_3_1" │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Path" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Polygon.js │ │ │ │ │ + OpenLayers/Format/WMSDescribeLayer/v1_1.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler/Path.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ + * @requires OpenLayers/Format/WMSDescribeLayer.js │ │ │ │ │ + * @requires OpenLayers/Format/OGCExceptionReport.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Handler.Polygon │ │ │ │ │ - * Handler to draw a polygon on the map. Polygon is displayed on mouse down, │ │ │ │ │ - * moves on mouse move, and is finished on mouse up. │ │ │ │ │ + * Class: OpenLayers.Format.WMSDescribeLayer.v1_1_1 │ │ │ │ │ + * Read SLD WMS DescribeLayer response for WMS 1.1.X │ │ │ │ │ + * WMS 1.1.X is tightly coupled to SLD 1.0.0 │ │ │ │ │ + * │ │ │ │ │ + * Example DescribeLayer request: │ │ │ │ │ + * http://demo.opengeo.org/geoserver/wms?request=DescribeLayer&version=1.1.1&layers=topp:states │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler.Path> │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ + * - <OpenLayers.Format.WMSDescribeLayer> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.Polygon = OpenLayers.Class(OpenLayers.Handler.Path, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: holeModifier │ │ │ │ │ - * {String} Key modifier to trigger hole digitizing. Acceptable values are │ │ │ │ │ - * "altKey", "shiftKey", or "ctrlKey". If not set, no hole digitizing │ │ │ │ │ - * will take place. Default is null. │ │ │ │ │ - */ │ │ │ │ │ - holeModifier: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: drawingHole │ │ │ │ │ - * {Boolean} Currently drawing an interior ring. │ │ │ │ │ - */ │ │ │ │ │ - drawingHole: false, │ │ │ │ │ +OpenLayers.Format.WMSDescribeLayer.v1_1_1 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WMSDescribeLayer, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: polygon │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - polygon: null, │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMSDescribeLayer │ │ │ │ │ + * Create a new parser for WMS DescribeLayer responses. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.WMSDescribeLayer.prototype.initialize.apply(this, │ │ │ │ │ + [options]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Polygon │ │ │ │ │ - * Create a Polygon Handler. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that owns this handler │ │ │ │ │ - * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ - * functions. Various callbacks described below. │ │ │ │ │ - * options - {Object} An optional object with properties to be set on the │ │ │ │ │ - * handler │ │ │ │ │ - * │ │ │ │ │ - * Named callbacks: │ │ │ │ │ - * create - Called when a sketch is first created. Callback called with │ │ │ │ │ - * the creation point geometry and sketch feature. │ │ │ │ │ - * modify - Called with each move of a vertex with the vertex (point) │ │ │ │ │ - * geometry and the sketch feature. │ │ │ │ │ - * point - Called as each point is added. Receives the new point geometry. │ │ │ │ │ - * done - Called when the point drawing is finished. The callback will │ │ │ │ │ - * recieve a single argument, the polygon geometry. │ │ │ │ │ - * cancel - Called when the handler is deactivated while drawing. The │ │ │ │ │ - * cancel callback will receive a geometry. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read DescribeLayer data from a string, and return the response. │ │ │ │ │ + * The OGC defines 2 formats which are allowed for output, │ │ │ │ │ + * so we need to parse these 2 types for version 1.1.X │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Object with a layerDescriptions property, which holds an Array │ │ │ │ │ + * of {<LayerDescription>} objects which have: │ │ │ │ │ + * - {String} owsType: WFS/WCS │ │ │ │ │ + * - {String} owsURL: the online resource │ │ │ │ │ + * - {String} typeName: the name of the typename on the owsType service │ │ │ │ │ + * - {String} layerName: the name of the WMS layer we did a lookup for │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + } │ │ │ │ │ + var root = data.documentElement; │ │ │ │ │ + var children = root.childNodes; │ │ │ │ │ + var describelayer = { │ │ │ │ │ + layerDescriptions: [] │ │ │ │ │ + }; │ │ │ │ │ + var childNode, nodeName; │ │ │ │ │ + for (var i = 0; i < children.length; ++i) { │ │ │ │ │ + childNode = children[i]; │ │ │ │ │ + nodeName = childNode.nodeName; │ │ │ │ │ + if (nodeName == 'LayerDescription') { │ │ │ │ │ + var layerName = childNode.getAttribute('name'); │ │ │ │ │ + var owsType = ''; │ │ │ │ │ + var owsURL = ''; │ │ │ │ │ + var typeName = ''; │ │ │ │ │ + // check for owsType and owsURL attributes │ │ │ │ │ + if (childNode.getAttribute('owsType')) { │ │ │ │ │ + owsType = childNode.getAttribute('owsType'); │ │ │ │ │ + owsURL = childNode.getAttribute('owsURL'); │ │ │ │ │ + } else { │ │ │ │ │ + // look for wfs or wcs attribute │ │ │ │ │ + if (childNode.getAttribute('wfs') != '') { │ │ │ │ │ + owsType = 'WFS'; │ │ │ │ │ + owsURL = childNode.getAttribute('wfs'); │ │ │ │ │ + } else if (childNode.getAttribute('wcs') != '') { │ │ │ │ │ + owsType = 'WCS'; │ │ │ │ │ + owsURL = childNode.getAttribute('wcs'); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // look for Query child │ │ │ │ │ + var query = childNode.getElementsByTagName('Query'); │ │ │ │ │ + if (query.length > 0) { │ │ │ │ │ + typeName = query[0].getAttribute('typeName'); │ │ │ │ │ + if (!typeName) { │ │ │ │ │ + // because of Ionic bug │ │ │ │ │ + typeName = query[0].getAttribute('typename'); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var layerDescription = { │ │ │ │ │ + layerName: layerName, │ │ │ │ │ + owsType: owsType, │ │ │ │ │ + owsURL: owsURL, │ │ │ │ │ + typeName: typeName │ │ │ │ │ + }; │ │ │ │ │ + describelayer.layerDescriptions.push(layerDescription); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createFeature │ │ │ │ │ - * Add temporary geometries │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} The initial pixel location for the new │ │ │ │ │ - * feature. │ │ │ │ │ - */ │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ - lonlat.lon, lonlat.lat │ │ │ │ │ - ); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.line = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.LinearRing([this.point.geometry]) │ │ │ │ │ - ); │ │ │ │ │ - this.polygon = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.Polygon([this.line.geometry]) │ │ │ │ │ - ); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.polygon, this.point], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + //TODO do this in deprecated.js instead: │ │ │ │ │ + // array style index for backwards compatibility │ │ │ │ │ + describelayer.length = describelayer.layerDescriptions.length; │ │ │ │ │ + describelayer[describelayer.length - 1] = layerDescription; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: addPoint │ │ │ │ │ - * Add point to geometry. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} The pixel location for the new point. │ │ │ │ │ - */ │ │ │ │ │ - addPoint: function(pixel) { │ │ │ │ │ - if (!this.drawingHole && this.holeModifier && │ │ │ │ │ - this.evt && this.evt[this.holeModifier]) { │ │ │ │ │ - var geometry = this.point.geometry; │ │ │ │ │ - var features = this.control.layer.features; │ │ │ │ │ - var candidate, polygon; │ │ │ │ │ - // look for intersections, last drawn gets priority │ │ │ │ │ - for (var i = features.length - 1; i >= 0; --i) { │ │ │ │ │ - candidate = features[i].geometry; │ │ │ │ │ - if ((candidate instanceof OpenLayers.Geometry.Polygon || │ │ │ │ │ - candidate instanceof OpenLayers.Geometry.MultiPolygon) && │ │ │ │ │ - candidate.intersects(geometry)) { │ │ │ │ │ - polygon = features[i]; │ │ │ │ │ - this.control.layer.removeFeatures([polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.control.layer.events.registerPriority( │ │ │ │ │ - "sketchcomplete", this, this.finalizeInteriorRing │ │ │ │ │ - ); │ │ │ │ │ - this.control.layer.events.registerPriority( │ │ │ │ │ - "sketchmodified", this, this.enforceTopology │ │ │ │ │ - ); │ │ │ │ │ - polygon.geometry.addComponent(this.line.geometry); │ │ │ │ │ - this.polygon = polygon; │ │ │ │ │ - this.drawingHole = true; │ │ │ │ │ - break; │ │ │ │ │ + } else if (nodeName == 'ServiceException') { │ │ │ │ │ + // an exception must have occurred, so parse it │ │ │ │ │ + var parser = new OpenLayers.Format.OGCExceptionReport(); │ │ │ │ │ + return { │ │ │ │ │ + error: parser.read(data) │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Handler.Path.prototype.addPoint.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + return describelayer; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getCurrentPointIndex │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number} The index of the most recently drawn point. │ │ │ │ │ - */ │ │ │ │ │ - getCurrentPointIndex: function() { │ │ │ │ │ - return this.line.geometry.components.length - 2; │ │ │ │ │ - }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSDescribeLayer.v1_1_1" │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: enforceTopology │ │ │ │ │ - * Simple topology enforcement for drawing interior rings. Ensures vertices │ │ │ │ │ - * of interior rings are contained by exterior ring. Other topology │ │ │ │ │ - * rules are enforced in <finalizeInteriorRing> to allow drawing of │ │ │ │ │ - * rings that intersect only during the sketch (e.g. a "C" shaped ring │ │ │ │ │ - * that nearly encloses another ring). │ │ │ │ │ - */ │ │ │ │ │ - enforceTopology: function(event) { │ │ │ │ │ - var point = event.vertex; │ │ │ │ │ - var components = this.line.geometry.components; │ │ │ │ │ - // ensure that vertices of interior ring are contained by exterior ring │ │ │ │ │ - if (!this.polygon.geometry.intersects(point)) { │ │ │ │ │ - var last = components[components.length - 3]; │ │ │ │ │ - point.x = last.x; │ │ │ │ │ - point.y = last.y; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: finishGeometry │ │ │ │ │ - * Finish the geometry and send it back to the control. │ │ │ │ │ - */ │ │ │ │ │ - finishGeometry: function() { │ │ │ │ │ - var index = this.line.geometry.components.length - 2; │ │ │ │ │ - this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize(); │ │ │ │ │ - }, │ │ │ │ │ +// Version alias - workaround for http://trac.osgeo.org/mapserver/ticket/2257 │ │ │ │ │ +OpenLayers.Format.WMSDescribeLayer.v1_1_0 = │ │ │ │ │ + OpenLayers.Format.WMSDescribeLayer.v1_1_1; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WCSCapabilities/v1.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: finalizeInteriorRing │ │ │ │ │ - * Enforces that new ring has some area and doesn't contain vertices of any │ │ │ │ │ - * other rings. │ │ │ │ │ - */ │ │ │ │ │ - finalizeInteriorRing: function() { │ │ │ │ │ - var ring = this.line.geometry; │ │ │ │ │ - // ensure that ring has some area │ │ │ │ │ - var modified = (ring.getArea() !== 0); │ │ │ │ │ - if (modified) { │ │ │ │ │ - // ensure that new ring doesn't intersect any other rings │ │ │ │ │ - var rings = this.polygon.geometry.components; │ │ │ │ │ - for (var i = rings.length - 2; i >= 0; --i) { │ │ │ │ │ - if (ring.intersects(rings[i])) { │ │ │ │ │ - modified = false; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WCSCapabilities.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WCSCapabilities.v1 │ │ │ │ │ + * Abstract class not to be instantiated directly. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WCSCapabilities.v1 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + splitSpace: (/\s+/) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ + */ │ │ │ │ │ + defaultPrefix: "wcs", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return a list of coverages. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} List of named coverages. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ } │ │ │ │ │ - if (modified) { │ │ │ │ │ - // ensure that new ring doesn't contain any other rings │ │ │ │ │ - var target; │ │ │ │ │ - outer: for (var i = rings.length - 2; i > 0; --i) { │ │ │ │ │ - var points = rings[i].components; │ │ │ │ │ - for (var j = 0, jj = points.length; j < jj; ++j) { │ │ │ │ │ - if (ring.containsPoint(points[j])) { │ │ │ │ │ - modified = false; │ │ │ │ │ - break outer; │ │ │ │ │ + var raw = data; │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + return capabilities; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WCSCapabilities/v1_1_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WCSCapabilities/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/OWSCommon/v1_1_0.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WCSCapabilities/v1_1_0 │ │ │ │ │ + * Read WCS Capabilities version 1.1.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WCSCapabilities.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WCSCapabilities.v1_1_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WCSCapabilities.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + wcs: "http://www.opengis.net/wcs/1.1", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + ows: "http://www.opengis.net/ows/1.1" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: errorProperty │ │ │ │ │ + * {String} Which property of the returned object to check for in order to │ │ │ │ │ + * determine whether or not parsing has failed. In the case that the │ │ │ │ │ + * errorProperty is undefined on the returned object, the document will be │ │ │ │ │ + * run through an OGCExceptionReport parser. │ │ │ │ │ + */ │ │ │ │ │ + errorProperty: "operationsMetadata", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WCSCapabilities.v1_1_0 │ │ │ │ │ + * Create a new parser for WCS capabilities version 1.1.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wcs": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + // In 1.0.0, this was WCS_Capabilties, in 1.1.0, it's Capabilities │ │ │ │ │ + "Capabilities": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "Contents": function(node, request) { │ │ │ │ │ + request.contentMetadata = []; │ │ │ │ │ + this.readChildNodes(node, request.contentMetadata); │ │ │ │ │ + }, │ │ │ │ │ + "CoverageSummary": function(node, contentMetadata) { │ │ │ │ │ + var coverageSummary = {}; │ │ │ │ │ + // Read the summary: │ │ │ │ │ + this.readChildNodes(node, coverageSummary); │ │ │ │ │ + │ │ │ │ │ + // Add it to the contentMetadata array: │ │ │ │ │ + contentMetadata.push(coverageSummary); │ │ │ │ │ + }, │ │ │ │ │ + "Identifier": function(node, coverageSummary) { │ │ │ │ │ + coverageSummary.identifier = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Title": function(node, coverageSummary) { │ │ │ │ │ + coverageSummary.title = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "Abstract": function(node, coverageSummary) { │ │ │ │ │ + coverageSummary["abstract"] = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "SupportedCRS": function(node, coverageSummary) { │ │ │ │ │ + var crs = this.getChildValue(node); │ │ │ │ │ + if (crs) { │ │ │ │ │ + if (!coverageSummary.supportedCRS) { │ │ │ │ │ + coverageSummary.supportedCRS = []; │ │ │ │ │ } │ │ │ │ │ + coverageSummary.supportedCRS.push(crs); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "SupportedFormat": function(node, coverageSummary) { │ │ │ │ │ + var format = this.getChildValue(node); │ │ │ │ │ + if (format) { │ │ │ │ │ + if (!coverageSummary.supportedFormat) { │ │ │ │ │ + coverageSummary.supportedFormat = []; │ │ │ │ │ + } │ │ │ │ │ + coverageSummary.supportedFormat.push(format); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (modified) { │ │ │ │ │ - if (this.polygon.state !== OpenLayers.State.INSERT) { │ │ │ │ │ - this.polygon.state = OpenLayers.State.UPDATE; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.polygon.geometry.removeComponent(ring); │ │ │ │ │ - } │ │ │ │ │ - this.restoreFeature(); │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ + }, OpenLayers.Format.WCSCapabilities.v1.prototype.readers["wcs"]), │ │ │ │ │ + "ows": OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: cancel │ │ │ │ │ - * Finish the geometry and call the "cancel" callback. │ │ │ │ │ - */ │ │ │ │ │ - cancel: function() { │ │ │ │ │ - if (this.drawingHole) { │ │ │ │ │ - this.polygon.geometry.removeComponent(this.line.geometry); │ │ │ │ │ - this.restoreFeature(true); │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Handler.Path.prototype.cancel.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1_1_0" │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: restoreFeature │ │ │ │ │ - * Move the feature from the sketch layer to the target layer. │ │ │ │ │ - * │ │ │ │ │ - * Properties: │ │ │ │ │ - * cancel - {Boolean} Cancel drawing. If falsey, the "sketchcomplete" event │ │ │ │ │ - * will be fired. │ │ │ │ │ - */ │ │ │ │ │ - restoreFeature: function(cancel) { │ │ │ │ │ - this.control.layer.events.unregister( │ │ │ │ │ - "sketchcomplete", this, this.finalizeInteriorRing │ │ │ │ │ - ); │ │ │ │ │ - this.control.layer.events.unregister( │ │ │ │ │ - "sketchmodified", this, this.enforceTopology │ │ │ │ │ - ); │ │ │ │ │ - this.layer.removeFeatures([this.polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.control.layer.addFeatures([this.polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.drawingHole = false; │ │ │ │ │ - if (!cancel) { │ │ │ │ │ - // Re-trigger "sketchcomplete" so other listeners can do their │ │ │ │ │ - // business. While this is somewhat sloppy (if a listener is │ │ │ │ │ - // registered with registerPriority - not common - between the start │ │ │ │ │ - // and end of a single ring drawing - very uncommon - it will be │ │ │ │ │ - // called twice). │ │ │ │ │ - // TODO: In 3.0, collapse sketch handlers into geometry specific │ │ │ │ │ - // drawing controls. │ │ │ │ │ - this.control.layer.events.triggerEvent( │ │ │ │ │ - "sketchcomplete", { │ │ │ │ │ - feature: this.polygon │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/WCSCapabilities/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroyFeature │ │ │ │ │ - * Destroy temporary geometries │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * force - {Boolean} Destroy even if persist is true. │ │ │ │ │ - */ │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - OpenLayers.Handler.Path.prototype.destroyFeature.call( │ │ │ │ │ - this, force); │ │ │ │ │ - this.polygon = null; │ │ │ │ │ - }, │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawFeature │ │ │ │ │ - * Render geometries on the temporary layer. │ │ │ │ │ - */ │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.polygon, this.style); │ │ │ │ │ - this.layer.drawFeature(this.point, this.style); │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/WCSCapabilities/v1.js │ │ │ │ │ + * @requires OpenLayers/Format/GML/v3.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getSketch │ │ │ │ │ - * Return the sketch feature. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - getSketch: function() { │ │ │ │ │ - return this.polygon; │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.WCSCapabilities/v1_0_0 │ │ │ │ │ + * Read WCS Capabilities version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.WCSCapabilities.v1> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.WCSCapabilities.v1_0_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WCSCapabilities.v1, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WCSCapabilities.v1_0_0 │ │ │ │ │ + * Create a new parser for WCS capabilities version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + wcs: "http://www.opengis.net/wcs", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + ows: "http://www.opengis.net/ows" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: errorProperty │ │ │ │ │ + * {String} Which property of the returned object to check for in order to │ │ │ │ │ + * determine whether or not parsing has failed. In the case that the │ │ │ │ │ + * errorProperty is undefined on the returned object, the document will be │ │ │ │ │ + * run through an OGCExceptionReport parser. │ │ │ │ │ + */ │ │ │ │ │ + errorProperty: "service", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "wcs": { │ │ │ │ │ + "WCS_Capabilities": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "Service": function(node, obj) { │ │ │ │ │ + obj.service = {}; │ │ │ │ │ + this.readChildNodes(node, obj.service); │ │ │ │ │ + }, │ │ │ │ │ + "name": function(node, service) { │ │ │ │ │ + service.name = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "label": function(node, service) { │ │ │ │ │ + service.label = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "keywords": function(node, service) { │ │ │ │ │ + service.keywords = []; │ │ │ │ │ + this.readChildNodes(node, service.keywords); │ │ │ │ │ + }, │ │ │ │ │ + "keyword": function(node, keywords) { │ │ │ │ │ + // Append the keyword to the keywords list │ │ │ │ │ + keywords.push(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "responsibleParty": function(node, service) { │ │ │ │ │ + service.responsibleParty = {}; │ │ │ │ │ + this.readChildNodes(node, service.responsibleParty); │ │ │ │ │ + }, │ │ │ │ │ + "individualName": function(node, responsibleParty) { │ │ │ │ │ + responsibleParty.individualName = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "organisationName": function(node, responsibleParty) { │ │ │ │ │ + responsibleParty.organisationName = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "positionName": function(node, responsibleParty) { │ │ │ │ │ + responsibleParty.positionName = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "contactInfo": function(node, responsibleParty) { │ │ │ │ │ + responsibleParty.contactInfo = {}; │ │ │ │ │ + this.readChildNodes(node, responsibleParty.contactInfo); │ │ │ │ │ + }, │ │ │ │ │ + "phone": function(node, contactInfo) { │ │ │ │ │ + contactInfo.phone = {}; │ │ │ │ │ + this.readChildNodes(node, contactInfo.phone); │ │ │ │ │ + }, │ │ │ │ │ + "voice": function(node, phone) { │ │ │ │ │ + phone.voice = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "facsimile": function(node, phone) { │ │ │ │ │ + phone.facsimile = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "address": function(node, contactInfo) { │ │ │ │ │ + contactInfo.address = {}; │ │ │ │ │ + this.readChildNodes(node, contactInfo.address); │ │ │ │ │ + }, │ │ │ │ │ + "deliveryPoint": function(node, address) { │ │ │ │ │ + address.deliveryPoint = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "city": function(node, address) { │ │ │ │ │ + address.city = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "postalCode": function(node, address) { │ │ │ │ │ + address.postalCode = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "country": function(node, address) { │ │ │ │ │ + address.country = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "electronicMailAddress": function(node, address) { │ │ │ │ │ + address.electronicMailAddress = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "fees": function(node, service) { │ │ │ │ │ + service.fees = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "accessConstraints": function(node, service) { │ │ │ │ │ + service.accessConstraints = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "ContentMetadata": function(node, obj) { │ │ │ │ │ + obj.contentMetadata = []; │ │ │ │ │ + this.readChildNodes(node, obj.contentMetadata); │ │ │ │ │ + }, │ │ │ │ │ + "CoverageOfferingBrief": function(node, contentMetadata) { │ │ │ │ │ + var coverageOfferingBrief = {}; │ │ │ │ │ + this.readChildNodes(node, coverageOfferingBrief); │ │ │ │ │ + contentMetadata.push(coverageOfferingBrief); │ │ │ │ │ + }, │ │ │ │ │ + "name": function(node, coverageOfferingBrief) { │ │ │ │ │ + coverageOfferingBrief.name = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "label": function(node, coverageOfferingBrief) { │ │ │ │ │ + coverageOfferingBrief.label = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "lonLatEnvelope": function(node, coverageOfferingBrief) { │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, "http://www.opengis.net/gml", "pos"); │ │ │ │ │ + │ │ │ │ │ + // We expect two nodes here, to create the corners of a bounding box │ │ │ │ │ + if (nodeList.length == 2) { │ │ │ │ │ + var min = {}; │ │ │ │ │ + var max = {}; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getGeometry │ │ │ │ │ - * Return the sketch geometry. If <multi> is true, this will return │ │ │ │ │ - * a multi-part geometry. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Geometry.Polygon>} │ │ │ │ │ - */ │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.polygon && this.polygon.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiPolygon([geometry]); │ │ │ │ │ - } │ │ │ │ │ - return geometry; │ │ │ │ │ - }, │ │ │ │ │ + OpenLayers.Format.GML.v3.prototype.readers["gml"].pos.apply(this, [nodeList[0], min]); │ │ │ │ │ + OpenLayers.Format.GML.v3.prototype.readers["gml"].pos.apply(this, [nodeList[1], max]); │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Polygon" │ │ │ │ │ -}); │ │ │ │ │ + coverageOfferingBrief.lonLatEnvelope = {}; │ │ │ │ │ + coverageOfferingBrief.lonLatEnvelope.srsName = node.getAttribute("srsName"); │ │ │ │ │ + coverageOfferingBrief.lonLatEnvelope.min = min.points[0]; │ │ │ │ │ + coverageOfferingBrief.lonLatEnvelope.max = max.points[0]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1_0_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/MouseWheel.js │ │ │ │ │ + OpenLayers/Format/XLS/v1.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Format/XLS.js │ │ │ │ │ + * @requires OpenLayers/Format/GML/v3.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Handler.MouseWheel │ │ │ │ │ - * Handler for wheel up/down events. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Format.XLS.v1 │ │ │ │ │ + * Superclass for XLS version 1 parsers. Only supports GeocodeRequest for now. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.MouseWheel = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - /** │ │ │ │ │ - * Property: wheelListener │ │ │ │ │ - * {function} │ │ │ │ │ +OpenLayers.Format.XLS.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ */ │ │ │ │ │ - wheelListener: null, │ │ │ │ │ + namespaces: { │ │ │ │ │ + xls: "http://www.opengis.net/xls", │ │ │ │ │ + gml: "http://www.opengis.net/gml", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: interval │ │ │ │ │ - * {Integer} In order to increase server performance, an interval (in │ │ │ │ │ - * milliseconds) can be set to reduce the number of up/down events │ │ │ │ │ - * called. If set, a new up/down event will not be set until the │ │ │ │ │ - * interval has passed. │ │ │ │ │ - * Defaults to 0, meaning no interval. │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ */ │ │ │ │ │ - interval: 0, │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: maxDelta │ │ │ │ │ - * {Integer} Maximum delta to collect before breaking from the current │ │ │ │ │ - * interval. In cumulative mode, this also limits the maximum delta │ │ │ │ │ - * returned from the handler. Default is Number.POSITIVE_INFINITY. │ │ │ │ │ + * APIProperty: xy │ │ │ │ │ + * {Boolean} Order of the GML coordinate true:(x,y) or false:(y,x) │ │ │ │ │ + * Changing is not recommended, a new Format should be instantiated. │ │ │ │ │ */ │ │ │ │ │ - maxDelta: Number.POSITIVE_INFINITY, │ │ │ │ │ + xy: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: delta │ │ │ │ │ - * {Integer} When interval is set, delta collects the mousewheel z-deltas │ │ │ │ │ - * of the events that occur within the interval. │ │ │ │ │ - * See also the cumulative option │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ */ │ │ │ │ │ - delta: 0, │ │ │ │ │ + defaultPrefix: "xls", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: cumulative │ │ │ │ │ - * {Boolean} When interval is set: true to collect all the mousewheel │ │ │ │ │ - * z-deltas, false to only record the delta direction (positive or │ │ │ │ │ - * negative) │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} Schema location for a particular minor version. │ │ │ │ │ */ │ │ │ │ │ - cumulative: true, │ │ │ │ │ + schemaLocation: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.MouseWheel │ │ │ │ │ + * Constructor: OpenLayers.Format.XLS.v1 │ │ │ │ │ + * Instances of this class are not created directly. Use the │ │ │ │ │ + * <OpenLayers.Format.XLS> constructor instead. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} │ │ │ │ │ - * callbacks - {Object} An object containing a single function to be │ │ │ │ │ - * called when the drag operation is finished. │ │ │ │ │ - * The callback should expect to recieve a single │ │ │ │ │ - * argument, the point geometry. │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.wheelListener = OpenLayers.Function.bindAsEventListener( │ │ │ │ │ - this.onWheelEvent, this │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * Method: read │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {DOMElement} An XLS document element. │ │ │ │ │ + * options - {Object} Options for the reader. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object representing the XLSResponse. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ - this.wheelListener = null; │ │ │ │ │ + read: function(data, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var xls = {}; │ │ │ │ │ + this.readChildNodes(data, xls); │ │ │ │ │ + return xls; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Mouse ScrollWheel code thanks to http://adomas.org/javascript-mouse-wheel/ │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ */ │ │ │ │ │ + readers: { │ │ │ │ │ + "xls": { │ │ │ │ │ + "XLS": function(node, xls) { │ │ │ │ │ + xls.version = node.getAttribute("version"); │ │ │ │ │ + this.readChildNodes(node, xls); │ │ │ │ │ + }, │ │ │ │ │ + "Response": function(node, xls) { │ │ │ │ │ + this.readChildNodes(node, xls); │ │ │ │ │ + }, │ │ │ │ │ + "GeocodeResponse": function(node, xls) { │ │ │ │ │ + xls.responseLists = []; │ │ │ │ │ + this.readChildNodes(node, xls); │ │ │ │ │ + }, │ │ │ │ │ + "GeocodeResponseList": function(node, xls) { │ │ │ │ │ + var responseList = { │ │ │ │ │ + features: [], │ │ │ │ │ + numberOfGeocodedAddresses: parseInt(node.getAttribute("numberOfGeocodedAddresses")) │ │ │ │ │ + }; │ │ │ │ │ + xls.responseLists.push(responseList); │ │ │ │ │ + this.readChildNodes(node, responseList); │ │ │ │ │ + }, │ │ │ │ │ + "GeocodedAddress": function(node, responseList) { │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(); │ │ │ │ │ + responseList.features.push(feature); │ │ │ │ │ + this.readChildNodes(node, feature); │ │ │ │ │ + // post-process geometry │ │ │ │ │ + feature.geometry = feature.components[0]; │ │ │ │ │ + }, │ │ │ │ │ + "GeocodeMatchCode": function(node, feature) { │ │ │ │ │ + feature.attributes.matchCode = { │ │ │ │ │ + accuracy: parseFloat(node.getAttribute("accuracy")), │ │ │ │ │ + matchType: node.getAttribute("matchType") │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ + "Address": function(node, feature) { │ │ │ │ │ + var address = { │ │ │ │ │ + countryCode: node.getAttribute("countryCode"), │ │ │ │ │ + addressee: node.getAttribute("addressee"), │ │ │ │ │ + street: [], │ │ │ │ │ + place: [] │ │ │ │ │ + }; │ │ │ │ │ + feature.attributes.address = address; │ │ │ │ │ + this.readChildNodes(node, address); │ │ │ │ │ + }, │ │ │ │ │ + "freeFormAddress": function(node, address) { │ │ │ │ │ + address.freeFormAddress = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "StreetAddress": function(node, address) { │ │ │ │ │ + this.readChildNodes(node, address); │ │ │ │ │ + }, │ │ │ │ │ + "Building": function(node, address) { │ │ │ │ │ + address.building = { │ │ │ │ │ + 'number': node.getAttribute("number"), │ │ │ │ │ + subdivision: node.getAttribute("subdivision"), │ │ │ │ │ + buildingName: node.getAttribute("buildingName") │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ + "Street": function(node, address) { │ │ │ │ │ + // only support the built-in primitive type for now │ │ │ │ │ + address.street.push(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "Place": function(node, address) { │ │ │ │ │ + // type is one of CountrySubdivision, │ │ │ │ │ + // CountrySecondarySubdivision, Municipality or │ │ │ │ │ + // MunicipalitySubdivision │ │ │ │ │ + address.place[node.getAttribute("type")] = │ │ │ │ │ + this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "PostalCode": function(node, address) { │ │ │ │ │ + address.postalCode = this.getChildValue(node); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "gml": OpenLayers.Format.GML.v3.prototype.readers.gml │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: onWheelEvent │ │ │ │ │ - * Catch the wheel event and handle it xbrowserly │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: write │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * e - {Event} │ │ │ │ │ + * request - {Object} An object representing the geocode request. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The root of an XLS document. │ │ │ │ │ */ │ │ │ │ │ - onWheelEvent: function(e) { │ │ │ │ │ - │ │ │ │ │ - // make sure we have a map and check keyboard modifiers │ │ │ │ │ - if (!this.map || !this.checkModifiers(e)) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // Ride up the element's DOM hierarchy to determine if it or any of │ │ │ │ │ - // its ancestors was: │ │ │ │ │ - // * specifically marked as scrollable (CSS overflow property) │ │ │ │ │ - // * one of our layer divs or a div marked as scrollable │ │ │ │ │ - // ('olScrollable' CSS class) │ │ │ │ │ - // * the map div │ │ │ │ │ - // │ │ │ │ │ - var overScrollableDiv = false; │ │ │ │ │ - var allowScroll = false; │ │ │ │ │ - var overMapDiv = false; │ │ │ │ │ - │ │ │ │ │ - var elem = OpenLayers.Event.element(e); │ │ │ │ │ - while ((elem != null) && !overMapDiv && !overScrollableDiv) { │ │ │ │ │ - │ │ │ │ │ - if (!overScrollableDiv) { │ │ │ │ │ - try { │ │ │ │ │ - var overflow; │ │ │ │ │ - if (elem.currentStyle) { │ │ │ │ │ - overflow = elem.currentStyle["overflow"]; │ │ │ │ │ - } else { │ │ │ │ │ - var style = │ │ │ │ │ - document.defaultView.getComputedStyle(elem, null); │ │ │ │ │ - overflow = style.getPropertyValue("overflow"); │ │ │ │ │ - } │ │ │ │ │ - overScrollableDiv = (overflow && │ │ │ │ │ - (overflow == "auto") || (overflow == "scroll")); │ │ │ │ │ - } catch (err) { │ │ │ │ │ - //sometimes when scrolling in a popup, this causes │ │ │ │ │ - // obscure browser error │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + write: function(request) { │ │ │ │ │ + return this.writers.xls.XLS.apply(this, [request]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (!allowScroll) { │ │ │ │ │ - allowScroll = OpenLayers.Element.hasClass(elem, 'olScrollable'); │ │ │ │ │ - if (!allowScroll) { │ │ │ │ │ - for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ - // Are we in the layer div? Note that we have two cases │ │ │ │ │ - // here: one is to catch EventPane layers, which have a │ │ │ │ │ - // pane above the layer (layer.pane) │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if (elem == layer.div || elem == layer.pane) { │ │ │ │ │ - allowScroll = true; │ │ │ │ │ - break; │ │ │ │ │ + /** │ │ │ │ │ + * Property: writers │ │ │ │ │ + * As a compliment to the readers property, this structure contains public │ │ │ │ │ + * writing functions grouped by namespace alias and named like the │ │ │ │ │ + * node names they produce. │ │ │ │ │ + */ │ │ │ │ │ + writers: { │ │ │ │ │ + "xls": { │ │ │ │ │ + "XLS": function(request) { │ │ │ │ │ + var root = this.createElementNSPlus( │ │ │ │ │ + "xls:XLS", { │ │ │ │ │ + attributes: { │ │ │ │ │ + "version": this.VERSION, │ │ │ │ │ + "xsi:schemaLocation": this.schemaLocation │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - overMapDiv = (elem == this.map.div); │ │ │ │ │ - │ │ │ │ │ - elem = elem.parentNode; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // Logic below is the following: │ │ │ │ │ - // │ │ │ │ │ - // If we are over a scrollable div or not over the map div: │ │ │ │ │ - // * do nothing (let the browser handle scrolling) │ │ │ │ │ - // │ │ │ │ │ - // otherwise │ │ │ │ │ - // │ │ │ │ │ - // If we are over the layer div or a 'olScrollable' div: │ │ │ │ │ - // * zoom/in out │ │ │ │ │ - // then │ │ │ │ │ - // * kill event (so as not to also scroll the page after zooming) │ │ │ │ │ - // │ │ │ │ │ - // otherwise │ │ │ │ │ - // │ │ │ │ │ - // Kill the event (dont scroll the page if we wheel over the │ │ │ │ │ - // layerswitcher or the pan/zoom control) │ │ │ │ │ - // │ │ │ │ │ - if (!overScrollableDiv && overMapDiv) { │ │ │ │ │ - if (allowScroll) { │ │ │ │ │ - var delta = 0; │ │ │ │ │ - │ │ │ │ │ - if (e.wheelDelta) { │ │ │ │ │ - delta = e.wheelDelta; │ │ │ │ │ - if (delta % 160 === 0) { │ │ │ │ │ - // opera have steps of 160 instead of 120 │ │ │ │ │ - delta = delta * 0.75; │ │ │ │ │ + ); │ │ │ │ │ + this.writeNode("RequestHeader", request.header, root); │ │ │ │ │ + this.writeNode("Request", request, root); │ │ │ │ │ + return root; │ │ │ │ │ + }, │ │ │ │ │ + "RequestHeader": function(header) { │ │ │ │ │ + return this.createElementNSPlus("xls:RequestHeader"); │ │ │ │ │ + }, │ │ │ │ │ + "Request": function(request) { │ │ │ │ │ + var node = this.createElementNSPlus("xls:Request", { │ │ │ │ │ + attributes: { │ │ │ │ │ + methodName: "GeocodeRequest", │ │ │ │ │ + requestID: request.requestID || "", │ │ │ │ │ + version: this.VERSION │ │ │ │ │ } │ │ │ │ │ - delta = delta / 120; │ │ │ │ │ - } else if (e.detail) { │ │ │ │ │ - // detail in Firefox on OS X is 1/3 of Windows │ │ │ │ │ - // so force delta 1 / -1 │ │ │ │ │ - delta = -(e.detail / Math.abs(e.detail)); │ │ │ │ │ + }); │ │ │ │ │ + this.writeNode("GeocodeRequest", request.addresses, node); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "GeocodeRequest": function(addresses) { │ │ │ │ │ + var node = this.createElementNSPlus("xls:GeocodeRequest"); │ │ │ │ │ + for (var i = 0, len = addresses.length; i < len; i++) { │ │ │ │ │ + this.writeNode("Address", addresses[i], node); │ │ │ │ │ } │ │ │ │ │ - this.delta += delta; │ │ │ │ │ - │ │ │ │ │ - window.clearTimeout(this._timeoutId); │ │ │ │ │ - if (this.interval && Math.abs(this.delta) < this.maxDelta) { │ │ │ │ │ - // store e because window.event might change during delay │ │ │ │ │ - var evt = OpenLayers.Util.extend({}, e); │ │ │ │ │ - this._timeoutId = window.setTimeout( │ │ │ │ │ - OpenLayers.Function.bind(function() { │ │ │ │ │ - this.wheelZoom(evt); │ │ │ │ │ - }, this), │ │ │ │ │ - this.interval │ │ │ │ │ - ); │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Address": function(address) { │ │ │ │ │ + var node = this.createElementNSPlus("xls:Address", { │ │ │ │ │ + attributes: { │ │ │ │ │ + countryCode: address.countryCode │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (address.freeFormAddress) { │ │ │ │ │ + this.writeNode("freeFormAddress", address.freeFormAddress, node); │ │ │ │ │ } else { │ │ │ │ │ - this.wheelZoom(e); │ │ │ │ │ + if (address.street) { │ │ │ │ │ + this.writeNode("StreetAddress", address, node); │ │ │ │ │ + } │ │ │ │ │ + if (address.municipality) { │ │ │ │ │ + this.writeNode("Municipality", address.municipality, node); │ │ │ │ │ + } │ │ │ │ │ + if (address.countrySubdivision) { │ │ │ │ │ + this.writeNode("CountrySubdivision", address.countrySubdivision, node); │ │ │ │ │ + } │ │ │ │ │ + if (address.postalCode) { │ │ │ │ │ + this.writeNode("PostalCode", address.postalCode, node); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "freeFormAddress": function(freeFormAddress) { │ │ │ │ │ + return this.createElementNSPlus("freeFormAddress", { │ │ │ │ │ + value: freeFormAddress │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "StreetAddress": function(address) { │ │ │ │ │ + var node = this.createElementNSPlus("xls:StreetAddress"); │ │ │ │ │ + if (address.building) { │ │ │ │ │ + this.writeNode(node, "Building", address.building); │ │ │ │ │ + } │ │ │ │ │ + var street = address.street; │ │ │ │ │ + if (!(OpenLayers.Util.isArray(street))) { │ │ │ │ │ + street = [street]; │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, len = street.length; i < len; i++) { │ │ │ │ │ + this.writeNode("Street", street[i], node); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ + "Building": function(building) { │ │ │ │ │ + return this.createElementNSPlus("xls:Building", { │ │ │ │ │ + attributes: { │ │ │ │ │ + "number": building["number"], │ │ │ │ │ + "subdivision": building.subdivision, │ │ │ │ │ + "buildingName": building.buildingName │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "Street": function(street) { │ │ │ │ │ + return this.createElementNSPlus("xls:Street", { │ │ │ │ │ + value: street │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "Municipality": function(municipality) { │ │ │ │ │ + return this.createElementNSPlus("xls:Place", { │ │ │ │ │ + attributes: { │ │ │ │ │ + type: "Municipality" │ │ │ │ │ + }, │ │ │ │ │ + value: municipality │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "CountrySubdivision": function(countrySubdivision) { │ │ │ │ │ + return this.createElementNSPlus("xls:Place", { │ │ │ │ │ + attributes: { │ │ │ │ │ + type: "CountrySubdivision" │ │ │ │ │ + }, │ │ │ │ │ + value: countrySubdivision │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + "PostalCode": function(postalCode) { │ │ │ │ │ + return this.createElementNSPlus("xls:PostalCode", { │ │ │ │ │ + value: postalCode │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Event.stop(e); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: wheelZoom │ │ │ │ │ - * Given the wheel event, we carry out the appropriate zooming in or out, │ │ │ │ │ - * based on the 'wheelDelta' or 'detail' property of the event. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * e - {Event} │ │ │ │ │ - */ │ │ │ │ │ - wheelZoom: function(e) { │ │ │ │ │ - var delta = this.delta; │ │ │ │ │ - this.delta = 0; │ │ │ │ │ - │ │ │ │ │ - if (delta) { │ │ │ │ │ - e.xy = this.map.events.getMousePosition(e); │ │ │ │ │ - if (delta < 0) { │ │ │ │ │ - this.callback("down", │ │ │ │ │ - [e, this.cumulative ? Math.max(-this.maxDelta, delta) : -1]); │ │ │ │ │ - } else { │ │ │ │ │ - this.callback("up", │ │ │ │ │ - [e, this.cumulative ? Math.min(this.maxDelta, delta) : 1]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - */ │ │ │ │ │ - activate: function(evt) { │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - //register mousewheel events specifically on the window and document │ │ │ │ │ - var wheelListener = this.wheelListener; │ │ │ │ │ - OpenLayers.Event.observe(window, "DOMMouseScroll", wheelListener); │ │ │ │ │ - OpenLayers.Event.observe(window, "mousewheel", wheelListener); │ │ │ │ │ - OpenLayers.Event.observe(document, "mousewheel", wheelListener); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function(evt) { │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - // unregister mousewheel events specifically on the window and document │ │ │ │ │ - var wheelListener = this.wheelListener; │ │ │ │ │ - OpenLayers.Event.stopObserving(window, "DOMMouseScroll", wheelListener); │ │ │ │ │ - OpenLayers.Event.stopObserving(window, "mousewheel", wheelListener); │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mousewheel", wheelListener); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.XLS.v1" │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.MouseWheel" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Keyboard.js │ │ │ │ │ + OpenLayers/Format/XLS/v1_1_0.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ + * @requires OpenLayers/Format/XLS/v1.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.handler.Keyboard │ │ │ │ │ - * A handler for keyboard events. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Handler.Keyboard> constructor. │ │ │ │ │ + * Class: OpenLayers.Format.XLS.v1_1_0 │ │ │ │ │ + * Read / write XLS version 1.1.0. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ + * - <OpenLayers.Format.XLS.v1> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.Keyboard = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ +OpenLayers.Format.XLS.v1_1_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.XLS.v1, { │ │ │ │ │ │ │ │ │ │ - /* http://www.quirksmode.org/js/keys.html explains key x-browser │ │ │ │ │ - key handling quirks in pretty nice detail */ │ │ │ │ │ + /** │ │ │ │ │ + * Constant: VERSION │ │ │ │ │ + * {String} 1.1 │ │ │ │ │ + */ │ │ │ │ │ + VERSION: "1.1", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constant: KEY_EVENTS │ │ │ │ │ - * keydown, keypress, keyup │ │ │ │ │ - */ │ │ │ │ │ - KEY_EVENTS: ["keydown", "keyup"], │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} http://www.opengis.net/xls │ │ │ │ │ + * http://schemas.opengis.net/ols/1.1.0/LocationUtilityService.xsd │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: "http://www.opengis.net/xls http://schemas.opengis.net/ols/1.1.0/LocationUtilityService.xsd", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: eventListener │ │ │ │ │ - * {Function} │ │ │ │ │ - */ │ │ │ │ │ - eventListener: null, │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.XLS.v1_1_0 │ │ │ │ │ + * Instances of this class are not created directly. Use the │ │ │ │ │ + * <OpenLayers.Format.XLS> constructor instead. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: observeElement │ │ │ │ │ - * {DOMElement|String} The DOM element on which we listen for │ │ │ │ │ - * key events. Default to the document. │ │ │ │ │ - */ │ │ │ │ │ - observeElement: null, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.XLS.v1_1_0" │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Keyboard │ │ │ │ │ - * Returns a new keyboard handler. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ - * this handler. If a handler is being used without a control, the │ │ │ │ │ - * handlers setMap method must be overridden to deal properly with │ │ │ │ │ - * the map. │ │ │ │ │ - * callbacks - {Object} An object containing a single function to be │ │ │ │ │ - * called when the drag operation is finished. The callback should │ │ │ │ │ - * expect to recieve a single argument, the pixel location of the event. │ │ │ │ │ - * Callbacks for 'keydown', 'keypress', and 'keyup' are supported. │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * handler. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - // cache the bound event listener method so it can be unobserved later │ │ │ │ │ - this.eventListener = OpenLayers.Function.bindAsEventListener( │ │ │ │ │ - this.handleKeyEvent, this │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.eventListener = null; │ │ │ │ │ - OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ +// Support non standard implementation │ │ │ │ │ +OpenLayers.Format.XLS.v1_1 = OpenLayers.Format.XLS.v1_1_0; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Format/SOSCapabilities/v1_0_0.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.observeElement = this.observeElement || document; │ │ │ │ │ - for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ - OpenLayers.Event.observe( │ │ │ │ │ - this.observeElement, this.KEY_EVENTS[i], this.eventListener); │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ - OpenLayers.Event.stopObserving( │ │ │ │ │ - this.observeElement, this.KEY_EVENTS[i], this.eventListener); │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Format/SOSCapabilities.js │ │ │ │ │ + * @requires OpenLayers/Format/OWSCommon/v1_1_0.js │ │ │ │ │ + * @requires OpenLayers/Format/GML/v3.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Format.SOSCapabilities.v1_0_0 │ │ │ │ │ + * Read SOS Capabilities version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Format.SOSCapabilities> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Format.SOSCapabilities.v1_0_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.SOSCapabilities, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ + */ │ │ │ │ │ + namespaces: { │ │ │ │ │ + ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ + sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ + gml: "http://www.opengis.net/gml", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: regExes │ │ │ │ │ + * Compiled regular expressions for manipulating strings. │ │ │ │ │ + */ │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: (/^\s*|\s*$/g), │ │ │ │ │ + removeSpace: (/\s*/g), │ │ │ │ │ + splitSpace: (/\s+/), │ │ │ │ │ + trimComma: (/\s*,\s*/g) │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.SOSCapabilities.v1_0_0 │ │ │ │ │ + * Create a new parser for SOS capabilities version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.options = options; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: read │ │ │ │ │ + * Read capabilities data from a string, and return info about the SOS. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Information about the SOS service. │ │ │ │ │ + */ │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ } │ │ │ │ │ - deactivated = true; │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement; │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + return capabilities; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleKeyEvent │ │ │ │ │ - */ │ │ │ │ │ - handleKeyEvent: function(evt) { │ │ │ │ │ - if (this.checkModifiers(evt)) { │ │ │ │ │ - this.callback(evt.type, [evt]); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: readers │ │ │ │ │ + * Contains public functions, grouped by namespace prefix, that will │ │ │ │ │ + * be applied when a namespaced node is found matching the function │ │ │ │ │ + * name. The function will be applied in the scope of this parser │ │ │ │ │ + * with two arguments: the node being read and a context object passed │ │ │ │ │ + * from the parent. │ │ │ │ │ + */ │ │ │ │ │ + readers: { │ │ │ │ │ + "gml": OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "name": function(node, obj) { │ │ │ │ │ + obj.name = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "TimePeriod": function(node, obj) { │ │ │ │ │ + obj.timePeriod = {}; │ │ │ │ │ + this.readChildNodes(node, obj.timePeriod); │ │ │ │ │ + }, │ │ │ │ │ + "beginPosition": function(node, timePeriod) { │ │ │ │ │ + timePeriod.beginPosition = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ + "endPosition": function(node, timePeriod) { │ │ │ │ │ + timePeriod.endPosition = this.getChildValue(node); │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.GML.v3.prototype.readers["gml"]), │ │ │ │ │ + "sos": { │ │ │ │ │ + "Capabilities": function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + }, │ │ │ │ │ + "Contents": function(node, obj) { │ │ │ │ │ + obj.contents = {}; │ │ │ │ │ + this.readChildNodes(node, obj.contents); │ │ │ │ │ + }, │ │ │ │ │ + "ObservationOfferingList": function(node, contents) { │ │ │ │ │ + contents.offeringList = {}; │ │ │ │ │ + this.readChildNodes(node, contents.offeringList); │ │ │ │ │ + }, │ │ │ │ │ + "ObservationOffering": function(node, offeringList) { │ │ │ │ │ + var id = this.getAttributeNS(node, this.namespaces.gml, "id"); │ │ │ │ │ + offeringList[id] = { │ │ │ │ │ + procedures: [], │ │ │ │ │ + observedProperties: [], │ │ │ │ │ + featureOfInterestIds: [], │ │ │ │ │ + responseFormats: [], │ │ │ │ │ + resultModels: [], │ │ │ │ │ + responseModes: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, offeringList[id]); │ │ │ │ │ + }, │ │ │ │ │ + "time": function(node, offering) { │ │ │ │ │ + offering.time = {}; │ │ │ │ │ + this.readChildNodes(node, offering.time); │ │ │ │ │ + }, │ │ │ │ │ + "procedure": function(node, offering) { │ │ │ │ │ + offering.procedures.push(this.getAttributeNS(node, │ │ │ │ │ + this.namespaces.xlink, "href")); │ │ │ │ │ + }, │ │ │ │ │ + "observedProperty": function(node, offering) { │ │ │ │ │ + offering.observedProperties.push(this.getAttributeNS(node, │ │ │ │ │ + this.namespaces.xlink, "href")); │ │ │ │ │ + }, │ │ │ │ │ + "featureOfInterest": function(node, offering) { │ │ │ │ │ + offering.featureOfInterestIds.push(this.getAttributeNS(node, │ │ │ │ │ + this.namespaces.xlink, "href")); │ │ │ │ │ + }, │ │ │ │ │ + "responseFormat": function(node, offering) { │ │ │ │ │ + offering.responseFormats.push(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "resultModel": function(node, offering) { │ │ │ │ │ + offering.resultModels.push(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ + "responseMode": function(node, offering) { │ │ │ │ │ + offering.responseModes.push(this.getChildValue(node)); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + "ows": OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Keyboard" │ │ │ │ │ -}); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SOSCapabilities.v1_0_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Hover.js │ │ │ │ │ + OpenLayers/Format/WMC/v1.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ + * @requires OpenLayers/Format/WMC.js │ │ │ │ │ + * @requires OpenLayers/Format/XML.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Handler.Hover │ │ │ │ │ - * The hover handler is to be used to emulate mouseovers on objects │ │ │ │ │ - * on the map that aren't DOM elements. For example one can use │ │ │ │ │ - * this handler to send WMS/GetFeatureInfo requests as the user │ │ │ │ │ - * moves the mouve over the map. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Format.WMC.v1 │ │ │ │ │ + * Superclass for WMC version 1 parsers. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ + * - <OpenLayers.Format.XML> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Handler.Hover = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ +OpenLayers.Format.WMC.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: delay │ │ │ │ │ - * {Integer} - Number of milliseconds between mousemoves before │ │ │ │ │ - * the event is considered a hover. Default is 500. │ │ │ │ │ + * Property: namespaces │ │ │ │ │ + * {Object} Mapping of namespace aliases to namespace URIs. │ │ │ │ │ */ │ │ │ │ │ - delay: 500, │ │ │ │ │ + namespaces: { │ │ │ │ │ + ol: "http://openlayers.org/context", │ │ │ │ │ + wmc: "http://www.opengis.net/context", │ │ │ │ │ + sld: "http://www.opengis.net/sld", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: pixelTolerance │ │ │ │ │ - * {Integer} - Maximum number of pixels between mousemoves for │ │ │ │ │ - * an event to be considered a hover. Default is null. │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} Schema location for a particular minor version. │ │ │ │ │ */ │ │ │ │ │ - pixelTolerance: null, │ │ │ │ │ + schemaLocation: "", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: stopMove │ │ │ │ │ - * {Boolean} - Stop other listeners from being notified on mousemoves. │ │ │ │ │ - * Default is false. │ │ │ │ │ + * Method: getNamespacePrefix │ │ │ │ │ + * Get the namespace prefix for a given uri from the <namespaces> object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A namespace prefix or null if none found. │ │ │ │ │ */ │ │ │ │ │ - stopMove: false, │ │ │ │ │ + getNamespacePrefix: function(uri) { │ │ │ │ │ + var prefix = null; │ │ │ │ │ + if (uri == null) { │ │ │ │ │ + prefix = this.namespaces[this.defaultPrefix]; │ │ │ │ │ + } else { │ │ │ │ │ + for (prefix in this.namespaces) { │ │ │ │ │ + if (this.namespaces[prefix] == uri) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return prefix; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: px │ │ │ │ │ - * {<OpenLayers.Pixel>} - The location of the last mousemove, expressed │ │ │ │ │ - * in pixels. │ │ │ │ │ + * Property: defaultPrefix │ │ │ │ │ */ │ │ │ │ │ - px: null, │ │ │ │ │ + defaultPrefix: "wmc", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: timerId │ │ │ │ │ - * {Number} - The id of the timer. │ │ │ │ │ + * Property: rootPrefix │ │ │ │ │ + * {String} Prefix on the root node that maps to the context namespace URI. │ │ │ │ │ */ │ │ │ │ │ - timerId: null, │ │ │ │ │ + rootPrefix: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Hover │ │ │ │ │ - * Construct a hover handler. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that initialized this │ │ │ │ │ - * handler. The control is assumed to have a valid map property; that │ │ │ │ │ - * map is used in the handler's own setMap method. │ │ │ │ │ - * callbacks - {Object} An object with keys corresponding to callbacks │ │ │ │ │ - * that will be called by the handler. The callbacks should │ │ │ │ │ - * expect to receive a single argument, the event. Callbacks for │ │ │ │ │ - * 'move', the mouse is moving, and 'pause', the mouse is pausing, │ │ │ │ │ - * are supported. │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * the handler. │ │ │ │ │ + * Property: defaultStyleName │ │ │ │ │ + * {String} Style name used if layer has no style param. Default is "". │ │ │ │ │ */ │ │ │ │ │ + defaultStyleName: "", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mousemove │ │ │ │ │ - * Called when the mouse moves on the map. │ │ │ │ │ + * Property: defaultStyleTitle │ │ │ │ │ + * {String} Default style title. Default is "Default". │ │ │ │ │ + */ │ │ │ │ │ + defaultStyleTitle: "Default", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMC.v1 │ │ │ │ │ + * Instances of this class are not created directly. Use the │ │ │ │ │ + * <OpenLayers.Format.WMC> constructor instead. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ */ │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - if (this.passesTolerance(evt.xy)) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.callback('move', [evt]); │ │ │ │ │ - this.px = evt.xy; │ │ │ │ │ - // clone the evt so original properties can be accessed even │ │ │ │ │ - // if the browser deletes them during the delay │ │ │ │ │ - evt = OpenLayers.Util.extend({}, evt); │ │ │ │ │ - this.timerId = window.setTimeout( │ │ │ │ │ - OpenLayers.Function.bind(this.delayedCall, this, evt), │ │ │ │ │ - this.delay │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return !this.stopMove; │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: mouseout │ │ │ │ │ - * Called when the mouse goes out of the map. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ + * Method: read │ │ │ │ │ + * Read capabilities data from a string, and return a list of layers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * data - {String} or {DOMElement} data to read/parse. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} Continue propagating this event. │ │ │ │ │ + * {Array} List of named layers. │ │ │ │ │ */ │ │ │ │ │ - mouseout: function(evt) { │ │ │ │ │ - if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.callback('move', [evt]); │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ } │ │ │ │ │ - return true; │ │ │ │ │ + var root = data.documentElement; │ │ │ │ │ + this.rootPrefix = root.prefix; │ │ │ │ │ + var context = { │ │ │ │ │ + version: root.getAttribute("version") │ │ │ │ │ + }; │ │ │ │ │ + this.runChildNodes(context, root); │ │ │ │ │ + return context; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: passesTolerance │ │ │ │ │ - * Determine whether the mouse move is within the optional pixel tolerance. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The mouse move is within the pixel tolerance. │ │ │ │ │ + * Method: runChildNodes │ │ │ │ │ */ │ │ │ │ │ - passesTolerance: function(px) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - if (this.pixelTolerance && this.px) { │ │ │ │ │ - var dpx = Math.sqrt( │ │ │ │ │ - Math.pow(this.px.x - px.x, 2) + │ │ │ │ │ - Math.pow(this.px.y - px.y, 2) │ │ │ │ │ - ); │ │ │ │ │ - if (dpx < this.pixelTolerance) { │ │ │ │ │ - passes = false; │ │ │ │ │ + runChildNodes: function(obj, node) { │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + var childNode, processor, prefix, local; │ │ │ │ │ + for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ + childNode = children[i]; │ │ │ │ │ + if (childNode.nodeType == 1) { │ │ │ │ │ + prefix = this.getNamespacePrefix(childNode.namespaceURI); │ │ │ │ │ + local = childNode.nodeName.split(":").pop(); │ │ │ │ │ + processor = this["read_" + prefix + "_" + local]; │ │ │ │ │ + if (processor) { │ │ │ │ │ + processor.apply(this, [obj, childNode]); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return passes; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clearTimer │ │ │ │ │ - * Clear the timer and set <timerId> to null. │ │ │ │ │ + * Method: read_wmc_General │ │ │ │ │ */ │ │ │ │ │ - clearTimer: function() { │ │ │ │ │ - if (this.timerId != null) { │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - } │ │ │ │ │ + read_wmc_General: function(context, node) { │ │ │ │ │ + this.runChildNodes(context, node); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: delayedCall │ │ │ │ │ - * Triggers pause callback. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ + * Method: read_wmc_BoundingBox │ │ │ │ │ */ │ │ │ │ │ - delayedCall: function(evt) { │ │ │ │ │ - this.callback('pause', [evt]); │ │ │ │ │ + read_wmc_BoundingBox: function(context, node) { │ │ │ │ │ + context.projection = node.getAttribute("SRS"); │ │ │ │ │ + context.bounds = new OpenLayers.Bounds( │ │ │ │ │ + node.getAttribute("minx"), node.getAttribute("miny"), │ │ │ │ │ + node.getAttribute("maxx"), node.getAttribute("maxy") │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the handler. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully deactivated. │ │ │ │ │ + * Method: read_wmc_LayerList │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - deactivated = true; │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ + read_wmc_LayerList: function(context, node) { │ │ │ │ │ + // layersContext is an array containing info for each layer │ │ │ │ │ + context.layersContext = []; │ │ │ │ │ + this.runChildNodes(context, node); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Hover" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Pinch.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: read_wmc_Layer │ │ │ │ │ + */ │ │ │ │ │ + read_wmc_Layer: function(context, node) { │ │ │ │ │ + var layerContext = { │ │ │ │ │ + visibility: (node.getAttribute("hidden") != "1"), │ │ │ │ │ + queryable: (node.getAttribute("queryable") == "1"), │ │ │ │ │ + formats: [], │ │ │ │ │ + styles: [], │ │ │ │ │ + metadata: {} │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Handler.Pinch │ │ │ │ │ - * The pinch handler is used to deal with sequences of browser events related │ │ │ │ │ - * to pinch gestures. The handler is used by controls that want to know │ │ │ │ │ - * when a pinch sequence begins, when a pinch is happening, and when it has │ │ │ │ │ - * finished. │ │ │ │ │ - * │ │ │ │ │ - * Controls that use the pinch handler typically construct it with callbacks │ │ │ │ │ - * for 'start', 'move', and 'done'. Callbacks for these keys are │ │ │ │ │ - * called when the pinch begins, with each change, and when the pinch is │ │ │ │ │ - * done. │ │ │ │ │ - * │ │ │ │ │ - * Create a new pinch handler with the <OpenLayers.Handler.Pinch> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.Pinch = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + this.runChildNodes(layerContext, node); │ │ │ │ │ + // set properties common to multiple objects on layer options/params │ │ │ │ │ + context.layersContext.push(layerContext); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: started │ │ │ │ │ - * {Boolean} When a touchstart event is received, we want to record it, │ │ │ │ │ - * but not set 'pinching' until the touchmove get started after │ │ │ │ │ - * starting. │ │ │ │ │ + * Method: read_wmc_Extension │ │ │ │ │ */ │ │ │ │ │ - started: false, │ │ │ │ │ + read_wmc_Extension: function(obj, node) { │ │ │ │ │ + this.runChildNodes(obj, node); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: stopDown │ │ │ │ │ - * {Boolean} Stop propagation of touchstart events from getting to │ │ │ │ │ - * listeners on the same element. Default is false. │ │ │ │ │ + * Method: read_ol_units │ │ │ │ │ */ │ │ │ │ │ - stopDown: false, │ │ │ │ │ + read_ol_units: function(layerContext, node) { │ │ │ │ │ + layerContext.units = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: pinching │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * Method: read_ol_maxExtent │ │ │ │ │ */ │ │ │ │ │ - pinching: false, │ │ │ │ │ + read_ol_maxExtent: function(obj, node) { │ │ │ │ │ + var bounds = new OpenLayers.Bounds( │ │ │ │ │ + node.getAttribute("minx"), node.getAttribute("miny"), │ │ │ │ │ + node.getAttribute("maxx"), node.getAttribute("maxy") │ │ │ │ │ + ); │ │ │ │ │ + obj.maxExtent = bounds; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: last │ │ │ │ │ - * {Object} Object that store informations related to pinch last touch. │ │ │ │ │ + * Method: read_ol_transparent │ │ │ │ │ */ │ │ │ │ │ - last: null, │ │ │ │ │ + read_ol_transparent: function(layerContext, node) { │ │ │ │ │ + layerContext.transparent = this.getChildValue(node); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: start │ │ │ │ │ - * {Object} Object that store informations related to pinch touchstart. │ │ │ │ │ + * Method: read_ol_numZoomLevels │ │ │ │ │ */ │ │ │ │ │ - start: null, │ │ │ │ │ + read_ol_numZoomLevels: function(layerContext, node) { │ │ │ │ │ + layerContext.numZoomLevels = parseInt(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Pinch │ │ │ │ │ - * Returns OpenLayers.Handler.Pinch │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control that is making use of │ │ │ │ │ - * this handler. If a handler is being used without a control, the │ │ │ │ │ - * handlers setMap method must be overridden to deal properly with │ │ │ │ │ - * the map. │ │ │ │ │ - * callbacks - {Object} An object containing functions to be called when │ │ │ │ │ - * the pinch operation start, change, or is finished. The callbacks │ │ │ │ │ - * should expect to receive an object argument, which contains │ │ │ │ │ - * information about scale, distance, and position of touch points. │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * Method: read_ol_opacity │ │ │ │ │ */ │ │ │ │ │ + read_ol_opacity: function(layerContext, node) { │ │ │ │ │ + layerContext.opacity = parseFloat(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchstart │ │ │ │ │ - * Handle touchstart events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * Method: read_ol_singleTile │ │ │ │ │ */ │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - if (OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ - this.started = true; │ │ │ │ │ - this.last = this.start = { │ │ │ │ │ - distance: this.getDistance(evt.touches), │ │ │ │ │ - delta: 0, │ │ │ │ │ - scale: 1 │ │ │ │ │ - }; │ │ │ │ │ - this.callback("start", [evt, this.start]); │ │ │ │ │ - propagate = !this.stopDown; │ │ │ │ │ - } else if (this.started) { │ │ │ │ │ - // Some webkit versions send fake single-touch events during │ │ │ │ │ - // multitouch, which cause the drag handler to trigger │ │ │ │ │ - return false; │ │ │ │ │ - } else { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - } │ │ │ │ │ - // prevent document dragging │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ - return propagate; │ │ │ │ │ + read_ol_singleTile: function(layerContext, node) { │ │ │ │ │ + layerContext.singleTile = (this.getChildValue(node) == "true"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchmove │ │ │ │ │ - * Handle touchmove events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * Method: read_ol_tileSize │ │ │ │ │ */ │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - if (this.started && OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ - this.pinching = true; │ │ │ │ │ - var current = this.getPinchData(evt); │ │ │ │ │ - this.callback("move", [evt, current]); │ │ │ │ │ - this.last = current; │ │ │ │ │ - // prevent document dragging │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - } else if (this.started) { │ │ │ │ │ - // Some webkit versions send fake single-touch events during │ │ │ │ │ - // multitouch, which cause the drag handler to trigger │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ + read_ol_tileSize: function(layerContext, node) { │ │ │ │ │ + var obj = { │ │ │ │ │ + "width": node.getAttribute("width"), │ │ │ │ │ + "height": node.getAttribute("height") │ │ │ │ │ + }; │ │ │ │ │ + layerContext.tileSize = obj; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: touchend │ │ │ │ │ - * Handle touchend events │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Let the event propagate. │ │ │ │ │ + * Method: read_ol_isBaseLayer │ │ │ │ │ */ │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - if (this.started && !OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - this.callback("done", [evt, this.start, this.last]); │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ + read_ol_isBaseLayer: function(layerContext, node) { │ │ │ │ │ + layerContext.isBaseLayer = (this.getChildValue(node) == "true"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Activate the handler. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully activated. │ │ │ │ │ + * Method: read_ol_displayInLayerSwitcher │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - activated = true; │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ + read_ol_displayInLayerSwitcher: function(layerContext, node) { │ │ │ │ │ + layerContext.displayInLayerSwitcher = (this.getChildValue(node) == "true"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Deactivate the handler. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The handler was successfully deactivated. │ │ │ │ │ + * Method: read_wmc_Server │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - deactivated = true; │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ + read_wmc_Server: function(layerContext, node) { │ │ │ │ │ + layerContext.version = node.getAttribute("version"); │ │ │ │ │ + layerContext.url = this.getOnlineResource_href(node); │ │ │ │ │ + layerContext.metadata.servertitle = node.getAttribute("title"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getDistance │ │ │ │ │ - * Get the distance in pixels between two touches. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * touches - {Array(Object)} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number} The distance in pixels. │ │ │ │ │ + * Method: read_wmc_FormatList │ │ │ │ │ */ │ │ │ │ │ - getDistance: function(touches) { │ │ │ │ │ - var t0 = touches[0]; │ │ │ │ │ - var t1 = touches[1]; │ │ │ │ │ - return Math.sqrt( │ │ │ │ │ - Math.pow(t0.olClientX - t1.olClientX, 2) + │ │ │ │ │ - Math.pow(t0.olClientY - t1.olClientY, 2) │ │ │ │ │ - ); │ │ │ │ │ + read_wmc_FormatList: function(layerContext, node) { │ │ │ │ │ + this.runChildNodes(layerContext, node); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: getPinchData │ │ │ │ │ - * Get informations about the pinch event. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Object that contains data about the current pinch. │ │ │ │ │ + * Method: read_wmc_Format │ │ │ │ │ */ │ │ │ │ │ - getPinchData: function(evt) { │ │ │ │ │ - var distance = this.getDistance(evt.touches); │ │ │ │ │ - var scale = distance / this.start.distance; │ │ │ │ │ - return { │ │ │ │ │ - distance: distance, │ │ │ │ │ - delta: this.last.distance - distance, │ │ │ │ │ - scale: scale │ │ │ │ │ + read_wmc_Format: function(layerContext, node) { │ │ │ │ │ + var format = { │ │ │ │ │ + value: this.getChildValue(node) │ │ │ │ │ }; │ │ │ │ │ + if (node.getAttribute("current") == "1") { │ │ │ │ │ + format.current = true; │ │ │ │ │ + } │ │ │ │ │ + layerContext.formats.push(format); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Pinch" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Handler/Box.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Handler.js │ │ │ │ │ - * @requires OpenLayers/Handler/Drag.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Handler.Box │ │ │ │ │ - * Handler for dragging a rectangle across the map. Box is displayed │ │ │ │ │ - * on mouse down, moves on mouse move, and is finished on mouse up. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Handler> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Handler.Box = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: dragHandler │ │ │ │ │ - * {<OpenLayers.Handler.Drag>} │ │ │ │ │ - */ │ │ │ │ │ - dragHandler: null, │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: boxDivClassName │ │ │ │ │ - * {String} The CSS class to use for drawing the box. Default is │ │ │ │ │ - * olHandlerBoxZoomBox │ │ │ │ │ + * Method: read_wmc_StyleList │ │ │ │ │ */ │ │ │ │ │ - boxDivClassName: 'olHandlerBoxZoomBox', │ │ │ │ │ + read_wmc_StyleList: function(layerContext, node) { │ │ │ │ │ + this.runChildNodes(layerContext, node); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: boxOffsets │ │ │ │ │ - * {Object} Caches box offsets from css. This is used by the getBoxOffsets │ │ │ │ │ - * method. │ │ │ │ │ + * Method: read_wmc_Style │ │ │ │ │ */ │ │ │ │ │ - boxOffsets: null, │ │ │ │ │ + read_wmc_Style: function(layerContext, node) { │ │ │ │ │ + var style = {}; │ │ │ │ │ + this.runChildNodes(style, node); │ │ │ │ │ + if (node.getAttribute("current") == "1") { │ │ │ │ │ + style.current = true; │ │ │ │ │ + } │ │ │ │ │ + layerContext.styles.push(style); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Handler.Box │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} │ │ │ │ │ - * callbacks - {Object} An object with a properties whose values are │ │ │ │ │ - * functions. Various callbacks described below. │ │ │ │ │ - * options - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Named callbacks: │ │ │ │ │ - * start - Called when the box drag operation starts. │ │ │ │ │ - * done - Called when the box drag operation is finished. │ │ │ │ │ - * The callback should expect to receive a single argument, the box │ │ │ │ │ - * bounds or a pixel. If the box dragging didn't span more than a 5 │ │ │ │ │ - * pixel distance, a pixel will be returned instead of a bounds object. │ │ │ │ │ + * Method: read_wmc_SLD │ │ │ │ │ */ │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.dragHandler = new OpenLayers.Handler.Drag( │ │ │ │ │ - this, { │ │ │ │ │ - down: this.startBox, │ │ │ │ │ - move: this.moveBox, │ │ │ │ │ - out: this.removeBox, │ │ │ │ │ - up: this.endBox │ │ │ │ │ - }, { │ │ │ │ │ - keyMask: this.keyMask │ │ │ │ │ - } │ │ │ │ │ - ); │ │ │ │ │ + read_wmc_SLD: function(style, node) { │ │ │ │ │ + this.runChildNodes(style, node); │ │ │ │ │ + // style either comes back with an href or a body property │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * Method: read_sld_StyledLayerDescriptor │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ - if (this.dragHandler) { │ │ │ │ │ - this.dragHandler.destroy(); │ │ │ │ │ - this.dragHandler = null; │ │ │ │ │ - } │ │ │ │ │ + read_sld_StyledLayerDescriptor: function(sld, node) { │ │ │ │ │ + var xml = OpenLayers.Format.XML.prototype.write.apply(this, [node]); │ │ │ │ │ + sld.body = xml; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ + * Method: read_sld_FeatureTypeStyle │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Handler.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (this.dragHandler) { │ │ │ │ │ - this.dragHandler.setMap(map); │ │ │ │ │ - } │ │ │ │ │ + read_sld_FeatureTypeStyle: function(sld, node) { │ │ │ │ │ + var xml = OpenLayers.Format.XML.prototype.write.apply(this, [node]); │ │ │ │ │ + sld.body = xml; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: startBox │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} │ │ │ │ │ + * Method: read_wmc_OnlineResource │ │ │ │ │ */ │ │ │ │ │ - startBox: function(xy) { │ │ │ │ │ - this.callback("start", []); │ │ │ │ │ - this.zoomBox = OpenLayers.Util.createDiv('zoomBox', { │ │ │ │ │ - x: -9999, │ │ │ │ │ - y: -9999 │ │ │ │ │ - }); │ │ │ │ │ - this.zoomBox.className = this.boxDivClassName; │ │ │ │ │ - this.zoomBox.style.zIndex = this.map.Z_INDEX_BASE["Popup"] - 1; │ │ │ │ │ - │ │ │ │ │ - this.map.viewPortDiv.appendChild(this.zoomBox); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Element.addClass( │ │ │ │ │ - this.map.viewPortDiv, "olDrawBox" │ │ │ │ │ + read_wmc_OnlineResource: function(obj, node) { │ │ │ │ │ + obj.href = this.getAttributeNS( │ │ │ │ │ + node, this.namespaces.xlink, "href" │ │ │ │ │ ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveBox │ │ │ │ │ + * Method: read_wmc_Name │ │ │ │ │ */ │ │ │ │ │ - moveBox: function(xy) { │ │ │ │ │ - var startX = this.dragHandler.start.x; │ │ │ │ │ - var startY = this.dragHandler.start.y; │ │ │ │ │ - var deltaX = Math.abs(startX - xy.x); │ │ │ │ │ - var deltaY = Math.abs(startY - xy.y); │ │ │ │ │ - │ │ │ │ │ - var offset = this.getBoxOffsets(); │ │ │ │ │ - this.zoomBox.style.width = (deltaX + offset.width + 1) + "px"; │ │ │ │ │ - this.zoomBox.style.height = (deltaY + offset.height + 1) + "px"; │ │ │ │ │ - this.zoomBox.style.left = (xy.x < startX ? │ │ │ │ │ - startX - deltaX - offset.left : startX - offset.left) + "px"; │ │ │ │ │ - this.zoomBox.style.top = (xy.y < startY ? │ │ │ │ │ - startY - deltaY - offset.top : startY - offset.top) + "px"; │ │ │ │ │ + read_wmc_Name: function(obj, node) { │ │ │ │ │ + var name = this.getChildValue(node); │ │ │ │ │ + if (name) { │ │ │ │ │ + obj.name = name; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: endBox │ │ │ │ │ + * Method: read_wmc_Title │ │ │ │ │ */ │ │ │ │ │ - endBox: function(end) { │ │ │ │ │ - var result; │ │ │ │ │ - if (Math.abs(this.dragHandler.start.x - end.x) > 5 || │ │ │ │ │ - Math.abs(this.dragHandler.start.y - end.y) > 5) { │ │ │ │ │ - var start = this.dragHandler.start; │ │ │ │ │ - var top = Math.min(start.y, end.y); │ │ │ │ │ - var bottom = Math.max(start.y, end.y); │ │ │ │ │ - var left = Math.min(start.x, end.x); │ │ │ │ │ - var right = Math.max(start.x, end.x); │ │ │ │ │ - result = new OpenLayers.Bounds(left, bottom, right, top); │ │ │ │ │ - } else { │ │ │ │ │ - result = this.dragHandler.start.clone(); // i.e. OL.Pixel │ │ │ │ │ + read_wmc_Title: function(obj, node) { │ │ │ │ │ + var title = this.getChildValue(node); │ │ │ │ │ + if (title) { │ │ │ │ │ + obj.title = title; │ │ │ │ │ } │ │ │ │ │ - this.removeBox(); │ │ │ │ │ - │ │ │ │ │ - this.callback("done", [result]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removeBox │ │ │ │ │ - * Remove the zoombox from the screen and nullify our reference to it. │ │ │ │ │ + * Method: read_wmc_MetadataURL │ │ │ │ │ */ │ │ │ │ │ - removeBox: function() { │ │ │ │ │ - this.map.viewPortDiv.removeChild(this.zoomBox); │ │ │ │ │ - this.zoomBox = null; │ │ │ │ │ - this.boxOffsets = null; │ │ │ │ │ - OpenLayers.Element.removeClass( │ │ │ │ │ - this.map.viewPortDiv, "olDrawBox" │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ + read_wmc_MetadataURL: function(layerContext, node) { │ │ │ │ │ + layerContext.metadataURL = this.getOnlineResource_href(node); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: activate │ │ │ │ │ + * Method: read_wmc_KeywordList │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.dragHandler.activate(); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ + read_wmc_KeywordList: function(context, node) { │ │ │ │ │ + context.keywords = []; │ │ │ │ │ + this.runChildNodes(context.keywords, node); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ + * Method: read_wmc_Keyword │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - if (this.dragHandler.deactivate()) { │ │ │ │ │ - if (this.zoomBox) { │ │ │ │ │ - this.removeBox(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ + read_wmc_Keyword: function(keywords, node) { │ │ │ │ │ + keywords.push(this.getChildValue(node)); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getBoxOffsets │ │ │ │ │ - * Determines border offsets for a box, according to the box model. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} an object with the following offsets: │ │ │ │ │ - * - left │ │ │ │ │ - * - right │ │ │ │ │ - * - top │ │ │ │ │ - * - bottom │ │ │ │ │ - * - width │ │ │ │ │ - * - height │ │ │ │ │ + * Method: read_wmc_Abstract │ │ │ │ │ */ │ │ │ │ │ - getBoxOffsets: function() { │ │ │ │ │ - if (!this.boxOffsets) { │ │ │ │ │ - // Determine the box model. If the testDiv's clientWidth is 3, then │ │ │ │ │ - // the borders are outside and we are dealing with the w3c box │ │ │ │ │ - // model. Otherwise, the browser uses the traditional box model and │ │ │ │ │ - // the borders are inside the box bounds, leaving us with a │ │ │ │ │ - // clientWidth of 1. │ │ │ │ │ - var testDiv = document.createElement("div"); │ │ │ │ │ - //testDiv.style.visibility = "hidden"; │ │ │ │ │ - testDiv.style.position = "absolute"; │ │ │ │ │ - testDiv.style.border = "1px solid black"; │ │ │ │ │ - testDiv.style.width = "3px"; │ │ │ │ │ - document.body.appendChild(testDiv); │ │ │ │ │ - var w3cBoxModel = testDiv.clientWidth == 3; │ │ │ │ │ - document.body.removeChild(testDiv); │ │ │ │ │ - │ │ │ │ │ - var left = parseInt(OpenLayers.Element.getStyle(this.zoomBox, │ │ │ │ │ - "border-left-width")); │ │ │ │ │ - var right = parseInt(OpenLayers.Element.getStyle( │ │ │ │ │ - this.zoomBox, "border-right-width")); │ │ │ │ │ - var top = parseInt(OpenLayers.Element.getStyle(this.zoomBox, │ │ │ │ │ - "border-top-width")); │ │ │ │ │ - var bottom = parseInt(OpenLayers.Element.getStyle( │ │ │ │ │ - this.zoomBox, "border-bottom-width")); │ │ │ │ │ - this.boxOffsets = { │ │ │ │ │ - left: left, │ │ │ │ │ - right: right, │ │ │ │ │ - top: top, │ │ │ │ │ - bottom: bottom, │ │ │ │ │ - width: w3cBoxModel === false ? left + right : 0, │ │ │ │ │ - height: w3cBoxModel === false ? top + bottom : 0 │ │ │ │ │ - }; │ │ │ │ │ + read_wmc_Abstract: function(obj, node) { │ │ │ │ │ + var abst = this.getChildValue(node); │ │ │ │ │ + if (abst) { │ │ │ │ │ + obj["abstract"] = abst; │ │ │ │ │ } │ │ │ │ │ - return this.boxOffsets; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Box" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Marker/Box.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Marker.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Marker.Box │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Marker> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Marker.Box = OpenLayers.Class(OpenLayers.Marker, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: bounds │ │ │ │ │ - * {<OpenLayers.Bounds>} │ │ │ │ │ + /** │ │ │ │ │ + * Method: read_wmc_LogoURL │ │ │ │ │ */ │ │ │ │ │ - bounds: null, │ │ │ │ │ + read_wmc_LogoURL: function(context, node) { │ │ │ │ │ + context.logo = { │ │ │ │ │ + width: node.getAttribute("width"), │ │ │ │ │ + height: node.getAttribute("height"), │ │ │ │ │ + format: node.getAttribute("format"), │ │ │ │ │ + href: this.getOnlineResource_href(node) │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: div │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + /** │ │ │ │ │ + * Method: read_wmc_DescriptionURL │ │ │ │ │ */ │ │ │ │ │ - div: null, │ │ │ │ │ + read_wmc_DescriptionURL: function(context, node) { │ │ │ │ │ + context.descriptionURL = this.getOnlineResource_href(node); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Marker.Box │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ - * borderColor - {String} │ │ │ │ │ - * borderWidth - {int} │ │ │ │ │ + /** │ │ │ │ │ + * Method: read_wmc_ContactInformation │ │ │ │ │ */ │ │ │ │ │ - initialize: function(bounds, borderColor, borderWidth) { │ │ │ │ │ - this.bounds = bounds; │ │ │ │ │ - this.div = OpenLayers.Util.createDiv(); │ │ │ │ │ - this.div.style.overflow = 'hidden'; │ │ │ │ │ - this.events = new OpenLayers.Events(this, this.div); │ │ │ │ │ - this.setBorder(borderColor, borderWidth); │ │ │ │ │ + read_wmc_ContactInformation: function(obj, node) { │ │ │ │ │ + var contact = {}; │ │ │ │ │ + this.runChildNodes(contact, node); │ │ │ │ │ + obj.contactInformation = contact; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * Method: read_wmc_ContactPersonPrimary │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - │ │ │ │ │ - this.bounds = null; │ │ │ │ │ - this.div = null; │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Marker.prototype.destroy.apply(this, arguments); │ │ │ │ │ + read_wmc_ContactPersonPrimary: function(contact, node) { │ │ │ │ │ + var personPrimary = {}; │ │ │ │ │ + this.runChildNodes(personPrimary, node); │ │ │ │ │ + contact.personPrimary = personPrimary; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setBorder │ │ │ │ │ - * Allow the user to change the box's color and border width │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * color - {String} Default is "red" │ │ │ │ │ - * width - {int} Default is 2 │ │ │ │ │ + /** │ │ │ │ │ + * Method: read_wmc_ContactPerson │ │ │ │ │ */ │ │ │ │ │ - setBorder: function(color, width) { │ │ │ │ │ - if (!color) { │ │ │ │ │ - color = "red"; │ │ │ │ │ - } │ │ │ │ │ - if (!width) { │ │ │ │ │ - width = 2; │ │ │ │ │ + read_wmc_ContactPerson: function(primaryPerson, node) { │ │ │ │ │ + var person = this.getChildValue(node); │ │ │ │ │ + if (person) { │ │ │ │ │ + primaryPerson.person = person; │ │ │ │ │ } │ │ │ │ │ - this.div.style.border = width + "px solid " + color; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - * sz - {<OpenLayers.Size>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A new DOM Image with this marker's icon set at the │ │ │ │ │ - * location passed-in │ │ │ │ │ + /** │ │ │ │ │ + * Method: read_wmc_ContactOrganization │ │ │ │ │ */ │ │ │ │ │ - draw: function(px, sz) { │ │ │ │ │ - OpenLayers.Util.modifyDOMElement(this.div, null, px, sz); │ │ │ │ │ - return this.div; │ │ │ │ │ + read_wmc_ContactOrganization: function(primaryPerson, node) { │ │ │ │ │ + var organization = this.getChildValue(node); │ │ │ │ │ + if (organization) { │ │ │ │ │ + primaryPerson.organization = organization; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onScreen │ │ │ │ │ - * │ │ │ │ │ - * Rreturn: │ │ │ │ │ - * {Boolean} Whether or not the marker is currently visible on screen. │ │ │ │ │ + * Method: read_wmc_ContactPosition │ │ │ │ │ */ │ │ │ │ │ - onScreen: function() { │ │ │ │ │ - var onScreen = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - var screenBounds = this.map.getExtent(); │ │ │ │ │ - onScreen = screenBounds.containsBounds(this.bounds, true, true); │ │ │ │ │ + read_wmc_ContactPosition: function(contact, node) { │ │ │ │ │ + var position = this.getChildValue(node); │ │ │ │ │ + if (position) { │ │ │ │ │ + contact.position = position; │ │ │ │ │ } │ │ │ │ │ - return onScreen; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: display │ │ │ │ │ - * Hide or show the icon │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * display - {Boolean} │ │ │ │ │ + * Method: read_wmc_ContactAddress │ │ │ │ │ */ │ │ │ │ │ - display: function(display) { │ │ │ │ │ - this.div.style.display = (display) ? "" : "none"; │ │ │ │ │ + read_wmc_ContactAddress: function(contact, node) { │ │ │ │ │ + var contactAddress = {}; │ │ │ │ │ + this.runChildNodes(contactAddress, node); │ │ │ │ │ + contact.contactAddress = contactAddress; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Marker.Box" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy/Cluster.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Strategy.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Strategy.Cluster │ │ │ │ │ - * Strategy for vector feature clustering. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Strategy> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Strategy.Cluster = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: distance │ │ │ │ │ - * {Integer} Pixel distance between features that should be considered a │ │ │ │ │ - * single cluster. Default is 20 pixels. │ │ │ │ │ + * Method: read_wmc_AddressType │ │ │ │ │ */ │ │ │ │ │ - distance: 20, │ │ │ │ │ + read_wmc_AddressType: function(contactAddress, node) { │ │ │ │ │ + var type = this.getChildValue(node); │ │ │ │ │ + if (type) { │ │ │ │ │ + contactAddress.type = type; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: threshold │ │ │ │ │ - * {Integer} Optional threshold below which original features will be │ │ │ │ │ - * added to the layer instead of clusters. For example, a threshold │ │ │ │ │ - * of 3 would mean that any time there are 2 or fewer features in │ │ │ │ │ - * a cluster, those features will be added directly to the layer instead │ │ │ │ │ - * of a cluster representing those features. Default is null (which is │ │ │ │ │ - * equivalent to 1 - meaning that clusters may contain just one feature). │ │ │ │ │ + * Method: read_wmc_Address │ │ │ │ │ */ │ │ │ │ │ - threshold: null, │ │ │ │ │ + read_wmc_Address: function(contactAddress, node) { │ │ │ │ │ + var address = this.getChildValue(node); │ │ │ │ │ + if (address) { │ │ │ │ │ + contactAddress.address = address; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} Cached features. │ │ │ │ │ + * Method: read_wmc_City │ │ │ │ │ */ │ │ │ │ │ - features: null, │ │ │ │ │ + read_wmc_City: function(contactAddress, node) { │ │ │ │ │ + var city = this.getChildValue(node); │ │ │ │ │ + if (city) { │ │ │ │ │ + contactAddress.city = city; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: clusters │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} Calculated clusters. │ │ │ │ │ + * Method: read_wmc_StateOrProvince │ │ │ │ │ */ │ │ │ │ │ - clusters: null, │ │ │ │ │ + read_wmc_StateOrProvince: function(contactAddress, node) { │ │ │ │ │ + var stateOrProvince = this.getChildValue(node); │ │ │ │ │ + if (stateOrProvince) { │ │ │ │ │ + contactAddress.stateOrProvince = stateOrProvince; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: clustering │ │ │ │ │ - * {Boolean} The strategy is currently clustering features. │ │ │ │ │ + * Method: read_wmc_PostCode │ │ │ │ │ */ │ │ │ │ │ - clustering: false, │ │ │ │ │ + read_wmc_PostCode: function(contactAddress, node) { │ │ │ │ │ + var postcode = this.getChildValue(node); │ │ │ │ │ + if (postcode) { │ │ │ │ │ + contactAddress.postcode = postcode; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: resolution │ │ │ │ │ - * {Float} The resolution (map units per pixel) of the current cluster set. │ │ │ │ │ + * Method: read_wmc_Country │ │ │ │ │ */ │ │ │ │ │ - resolution: null, │ │ │ │ │ + read_wmc_Country: function(contactAddress, node) { │ │ │ │ │ + var country = this.getChildValue(node); │ │ │ │ │ + if (country) { │ │ │ │ │ + contactAddress.country = country; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy.Cluster │ │ │ │ │ - * Create a new clustering strategy. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * Method: read_wmc_ContactVoiceTelephone │ │ │ │ │ */ │ │ │ │ │ + read_wmc_ContactVoiceTelephone: function(contact, node) { │ │ │ │ │ + var phone = this.getChildValue(node); │ │ │ │ │ + if (phone) { │ │ │ │ │ + contact.phone = phone; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully activated. │ │ │ │ │ + * Method: read_wmc_ContactFacsimileTelephone │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "beforefeaturesadded": this.cacheFeatures, │ │ │ │ │ - "featuresremoved": this.clearCache, │ │ │ │ │ - "moveend": this.cluster, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + read_wmc_ContactFacsimileTelephone: function(contact, node) { │ │ │ │ │ + var fax = this.getChildValue(node); │ │ │ │ │ + if (fax) { │ │ │ │ │ + contact.fax = fax; │ │ │ │ │ } │ │ │ │ │ - return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ - * tear-down. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ + * Method: read_wmc_ContactElectronicMailAddress │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.clearCache(); │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - "beforefeaturesadded": this.cacheFeatures, │ │ │ │ │ - "featuresremoved": this.clearCache, │ │ │ │ │ - "moveend": this.cluster, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + read_wmc_ContactElectronicMailAddress: function(contact, node) { │ │ │ │ │ + var email = this.getChildValue(node); │ │ │ │ │ + if (email) { │ │ │ │ │ + contact.email = email; │ │ │ │ │ } │ │ │ │ │ - return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: cacheFeatures │ │ │ │ │ - * Cache features before they are added to the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * event - {Object} The event that this was listening for. This will come │ │ │ │ │ - * with a batch of features to be clustered. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} False to stop features from being added to the layer. │ │ │ │ │ + * Method: read_wmc_DataURL │ │ │ │ │ */ │ │ │ │ │ - cacheFeatures: function(event) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - if (!this.clustering) { │ │ │ │ │ - this.clearCache(); │ │ │ │ │ - this.features = event.features; │ │ │ │ │ - this.cluster(); │ │ │ │ │ - propagate = false; │ │ │ │ │ - } │ │ │ │ │ - return propagate; │ │ │ │ │ + read_wmc_DataURL: function(layerContext, node) { │ │ │ │ │ + layerContext.dataURL = this.getOnlineResource_href(node); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clearCache │ │ │ │ │ - * Clear out the cached features. │ │ │ │ │ + * Method: read_wmc_LegendURL │ │ │ │ │ */ │ │ │ │ │ - clearCache: function() { │ │ │ │ │ - if (!this.clustering) { │ │ │ │ │ - this.features = null; │ │ │ │ │ - } │ │ │ │ │ + read_wmc_LegendURL: function(style, node) { │ │ │ │ │ + var legend = { │ │ │ │ │ + width: node.getAttribute('width'), │ │ │ │ │ + height: node.getAttribute('height'), │ │ │ │ │ + format: node.getAttribute('format'), │ │ │ │ │ + href: this.getOnlineResource_href(node) │ │ │ │ │ + }; │ │ │ │ │ + style.legend = legend; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: cluster │ │ │ │ │ - * Cluster features based on some threshold distance. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * event - {Object} The event received when cluster is called as a │ │ │ │ │ - * result of a moveend event. │ │ │ │ │ + * Method: read_wmc_DimensionList │ │ │ │ │ */ │ │ │ │ │ - cluster: function(event) { │ │ │ │ │ - if ((!event || event.zoomChanged) && this.features) { │ │ │ │ │ - var resolution = this.layer.map.getResolution(); │ │ │ │ │ - if (resolution != this.resolution || !this.clustersExist()) { │ │ │ │ │ - this.resolution = resolution; │ │ │ │ │ - var clusters = []; │ │ │ │ │ - var feature, clustered, cluster; │ │ │ │ │ - for (var i = 0; i < this.features.length; ++i) { │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - clustered = false; │ │ │ │ │ - for (var j = clusters.length - 1; j >= 0; --j) { │ │ │ │ │ - cluster = clusters[j]; │ │ │ │ │ - if (this.shouldCluster(cluster, feature)) { │ │ │ │ │ - this.addToCluster(cluster, feature); │ │ │ │ │ - clustered = true; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!clustered) { │ │ │ │ │ - clusters.push(this.createCluster(this.features[i])); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.clustering = true; │ │ │ │ │ - this.layer.removeAllFeatures(); │ │ │ │ │ - this.clustering = false; │ │ │ │ │ - if (clusters.length > 0) { │ │ │ │ │ - if (this.threshold > 1) { │ │ │ │ │ - var clone = clusters.slice(); │ │ │ │ │ - clusters = []; │ │ │ │ │ - var candidate; │ │ │ │ │ - for (var i = 0, len = clone.length; i < len; ++i) { │ │ │ │ │ - candidate = clone[i]; │ │ │ │ │ - if (candidate.attributes.count < this.threshold) { │ │ │ │ │ - Array.prototype.push.apply(clusters, candidate.cluster); │ │ │ │ │ - } else { │ │ │ │ │ - clusters.push(candidate); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.clustering = true; │ │ │ │ │ - // A legitimate feature addition could occur during this │ │ │ │ │ - // addFeatures call. For clustering to behave well, features │ │ │ │ │ - // should be removed from a layer before requesting a new batch. │ │ │ │ │ - this.layer.addFeatures(clusters); │ │ │ │ │ - this.clustering = false; │ │ │ │ │ - } │ │ │ │ │ - this.clusters = clusters; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + read_wmc_DimensionList: function(layerContext, node) { │ │ │ │ │ + layerContext.dimensions = {}; │ │ │ │ │ + this.runChildNodes(layerContext.dimensions, node); │ │ │ │ │ + }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: read_wmc_Dimension │ │ │ │ │ + */ │ │ │ │ │ + read_wmc_Dimension: function(dimensions, node) { │ │ │ │ │ + var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ + │ │ │ │ │ + var dim = { │ │ │ │ │ + name: name, │ │ │ │ │ + units: node.getAttribute("units") || "", │ │ │ │ │ + unitSymbol: node.getAttribute("unitSymbol") || "", │ │ │ │ │ + userValue: node.getAttribute("userValue") || "", │ │ │ │ │ + nearestValue: node.getAttribute("nearestValue") === "1", │ │ │ │ │ + multipleValues: node.getAttribute("multipleValues") === "1", │ │ │ │ │ + current: node.getAttribute("current") === "1", │ │ │ │ │ + "default": node.getAttribute("default") || "" │ │ │ │ │ + }; │ │ │ │ │ + var values = this.getChildValue(node); │ │ │ │ │ + dim.values = values.split(","); │ │ │ │ │ + │ │ │ │ │ + dimensions[dim.name] = dim; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clustersExist │ │ │ │ │ - * Determine whether calculated clusters are already on the layer. │ │ │ │ │ + * Method: write │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * context - {Object} An object representing the map context. │ │ │ │ │ + * options - {Object} Optional object. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The calculated clusters are already on the layer. │ │ │ │ │ + * {String} A WMC document string. │ │ │ │ │ */ │ │ │ │ │ - clustersExist: function() { │ │ │ │ │ - var exist = false; │ │ │ │ │ - if (this.clusters && this.clusters.length > 0 && │ │ │ │ │ - this.clusters.length == this.layer.features.length) { │ │ │ │ │ - exist = true; │ │ │ │ │ - for (var i = 0; i < this.clusters.length; ++i) { │ │ │ │ │ - if (this.clusters[i] != this.layer.features[i]) { │ │ │ │ │ - exist = false; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return exist; │ │ │ │ │ + write: function(context, options) { │ │ │ │ │ + var root = this.createElementDefaultNS("ViewContext"); │ │ │ │ │ + this.setAttributes(root, { │ │ │ │ │ + version: this.VERSION, │ │ │ │ │ + id: (options && typeof options.id == "string") ? │ │ │ │ │ + options.id : OpenLayers.Util.createUniqueID("OpenLayers_Context_") │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + // add schemaLocation attribute │ │ │ │ │ + this.setAttributeNS( │ │ │ │ │ + root, this.namespaces.xsi, │ │ │ │ │ + "xsi:schemaLocation", this.schemaLocation │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + // required General element │ │ │ │ │ + root.appendChild(this.write_wmc_General(context)); │ │ │ │ │ + │ │ │ │ │ + // required LayerList element │ │ │ │ │ + root.appendChild(this.write_wmc_LayerList(context)); │ │ │ │ │ + │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [root]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: shouldCluster │ │ │ │ │ - * Determine whether to include a feature in a given cluster. │ │ │ │ │ + * Method: createElementDefaultNS │ │ │ │ │ + * Shorthand for createElementNS with namespace from <defaultPrefix>. │ │ │ │ │ + * Can optionally be used to set attributes and a text child value. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * cluster - {<OpenLayers.Feature.Vector>} A cluster. │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} A feature. │ │ │ │ │ + * name - {String} The qualified node name. │ │ │ │ │ + * childValue - {String} Optional value for text child node. │ │ │ │ │ + * attributes - {Object} Optional object representing attributes. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} The feature should be included in the cluster. │ │ │ │ │ + * {Element} An element node. │ │ │ │ │ */ │ │ │ │ │ - shouldCluster: function(cluster, feature) { │ │ │ │ │ - var cc = cluster.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - var fc = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - var distance = ( │ │ │ │ │ - Math.sqrt( │ │ │ │ │ - Math.pow((cc.lon - fc.lon), 2) + Math.pow((cc.lat - fc.lat), 2) │ │ │ │ │ - ) / this.resolution │ │ │ │ │ + createElementDefaultNS: function(name, childValue, attributes) { │ │ │ │ │ + var node = this.createElementNS( │ │ │ │ │ + this.namespaces[this.defaultPrefix], │ │ │ │ │ + name │ │ │ │ │ ); │ │ │ │ │ - return (distance <= this.distance); │ │ │ │ │ + if (childValue) { │ │ │ │ │ + node.appendChild(this.createTextNode(childValue)); │ │ │ │ │ + } │ │ │ │ │ + if (attributes) { │ │ │ │ │ + this.setAttributes(node, attributes); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: addToCluster │ │ │ │ │ - * Add a feature to a cluster. │ │ │ │ │ + * Method: setAttributes │ │ │ │ │ + * Set multiple attributes given key value pairs from an object. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * cluster - {<OpenLayers.Feature.Vector>} A cluster. │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} A feature. │ │ │ │ │ + * node - {Element} An element node. │ │ │ │ │ + * obj - {Object} An object whose properties represent attribute names and │ │ │ │ │ + * values represent attribute values. │ │ │ │ │ */ │ │ │ │ │ - addToCluster: function(cluster, feature) { │ │ │ │ │ - cluster.cluster.push(feature); │ │ │ │ │ - cluster.attributes.count += 1; │ │ │ │ │ + setAttributes: function(node, obj) { │ │ │ │ │ + var value; │ │ │ │ │ + for (var name in obj) { │ │ │ │ │ + value = obj[name].toString(); │ │ │ │ │ + if (value.match(/[A-Z]/)) { │ │ │ │ │ + // safari lowercases attributes with setAttribute │ │ │ │ │ + this.setAttributeNS(node, null, name, value); │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttribute(name, value); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createCluster │ │ │ │ │ - * Given a feature, create a cluster. │ │ │ │ │ + * Method: write_wmc_General │ │ │ │ │ + * Create a General node given an context object. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * context - {Object} Context object. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} A cluster. │ │ │ │ │ + * {Element} A WMC General element node. │ │ │ │ │ */ │ │ │ │ │ - createCluster: function(feature) { │ │ │ │ │ - var center = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - var cluster = new OpenLayers.Feature.Vector( │ │ │ │ │ - new OpenLayers.Geometry.Point(center.lon, center.lat), { │ │ │ │ │ - count: 1 │ │ │ │ │ + write_wmc_General: function(context) { │ │ │ │ │ + var node = this.createElementDefaultNS("General"); │ │ │ │ │ + │ │ │ │ │ + // optional Window element │ │ │ │ │ + if (context.size) { │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "Window", null, { │ │ │ │ │ + width: context.size.w, │ │ │ │ │ + height: context.size.h │ │ │ │ │ + } │ │ │ │ │ + )); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // required BoundingBox element │ │ │ │ │ + var bounds = context.bounds; │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "BoundingBox", null, { │ │ │ │ │ + minx: bounds.left.toPrecision(18), │ │ │ │ │ + miny: bounds.bottom.toPrecision(18), │ │ │ │ │ + maxx: bounds.right.toPrecision(18), │ │ │ │ │ + maxy: bounds.top.toPrecision(18), │ │ │ │ │ + SRS: context.projection │ │ │ │ │ } │ │ │ │ │ - ); │ │ │ │ │ - cluster.cluster = [feature]; │ │ │ │ │ - return cluster; │ │ │ │ │ - }, │ │ │ │ │ + )); │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Cluster" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy/Paging.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + // required Title element │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "Title", context.title │ │ │ │ │ + )); │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + // optional KeywordList element │ │ │ │ │ + if (context.keywords) { │ │ │ │ │ + node.appendChild(this.write_wmc_KeywordList(context.keywords)); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Strategy.js │ │ │ │ │ - */ │ │ │ │ │ + // optional Abstract element │ │ │ │ │ + if (context["abstract"]) { │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "Abstract", context["abstract"] │ │ │ │ │ + )); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Strategy.Paging │ │ │ │ │ - * Strategy for vector feature paging │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Strategy> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Strategy.Paging = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + // Optional LogoURL element │ │ │ │ │ + if (context.logo) { │ │ │ │ │ + node.appendChild(this.write_wmc_URLType("LogoURL", context.logo.href, context.logo)); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} Cached features. │ │ │ │ │ - */ │ │ │ │ │ - features: null, │ │ │ │ │ + // Optional DescriptionURL element │ │ │ │ │ + if (context.descriptionURL) { │ │ │ │ │ + node.appendChild(this.write_wmc_URLType("DescriptionURL", context.descriptionURL)); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: length │ │ │ │ │ - * {Integer} Number of features per page. Default is 10. │ │ │ │ │ - */ │ │ │ │ │ - length: 10, │ │ │ │ │ + // Optional ContactInformation element │ │ │ │ │ + if (context.contactInformation) { │ │ │ │ │ + node.appendChild(this.write_wmc_ContactInformation(context.contactInformation)); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: num │ │ │ │ │ - * {Integer} The currently displayed page number. │ │ │ │ │ - */ │ │ │ │ │ - num: null, │ │ │ │ │ + // OpenLayers specific map properties │ │ │ │ │ + node.appendChild(this.write_ol_MapExtension(context)); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: paging │ │ │ │ │ - * {Boolean} The strategy is currently changing pages. │ │ │ │ │ - */ │ │ │ │ │ - paging: false, │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy.Paging │ │ │ │ │ - * Create a new paging strategy. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * Method: write_wmc_KeywordList │ │ │ │ │ */ │ │ │ │ │ + write_wmc_KeywordList: function(keywords) { │ │ │ │ │ + var node = this.createElementDefaultNS("KeywordList"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "beforefeaturesadded": this.cacheFeatures, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + for (var i = 0, len = keywords.length; i < len; i++) { │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "Keyword", keywords[i] │ │ │ │ │ + )); │ │ │ │ │ } │ │ │ │ │ - return activated; │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ - * tear-down. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ + * Method: write_wmc_ContactInformation │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.clearCache(); │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - "beforefeaturesadded": this.cacheFeatures, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + write_wmc_ContactInformation: function(contact) { │ │ │ │ │ + var node = this.createElementDefaultNS("ContactInformation"); │ │ │ │ │ + │ │ │ │ │ + if (contact.personPrimary) { │ │ │ │ │ + node.appendChild(this.write_wmc_ContactPersonPrimary(contact.personPrimary)); │ │ │ │ │ } │ │ │ │ │ - return deactivated; │ │ │ │ │ + if (contact.position) { │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "ContactPosition", contact.position │ │ │ │ │ + )); │ │ │ │ │ + } │ │ │ │ │ + if (contact.contactAddress) { │ │ │ │ │ + node.appendChild(this.write_wmc_ContactAddress(contact.contactAddress)); │ │ │ │ │ + } │ │ │ │ │ + if (contact.phone) { │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "ContactVoiceTelephone", contact.phone │ │ │ │ │ + )); │ │ │ │ │ + } │ │ │ │ │ + if (contact.fax) { │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "ContactFacsimileTelephone", contact.fax │ │ │ │ │ + )); │ │ │ │ │ + } │ │ │ │ │ + if (contact.email) { │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "ContactElectronicMailAddress", contact.email │ │ │ │ │ + )); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: cacheFeatures │ │ │ │ │ - * Cache features before they are added to the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * event - {Object} The event that this was listening for. This will come │ │ │ │ │ - * with a batch of features to be paged. │ │ │ │ │ + * Method: write_wmc_ContactPersonPrimary │ │ │ │ │ */ │ │ │ │ │ - cacheFeatures: function(event) { │ │ │ │ │ - if (!this.paging) { │ │ │ │ │ - this.clearCache(); │ │ │ │ │ - this.features = event.features; │ │ │ │ │ - this.pageNext(event); │ │ │ │ │ + write_wmc_ContactPersonPrimary: function(personPrimary) { │ │ │ │ │ + var node = this.createElementDefaultNS("ContactPersonPrimary"); │ │ │ │ │ + if (personPrimary.person) { │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "ContactPerson", personPrimary.person │ │ │ │ │ + )); │ │ │ │ │ + } │ │ │ │ │ + if (personPrimary.organization) { │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "ContactOrganization", personPrimary.organization │ │ │ │ │ + )); │ │ │ │ │ } │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clearCache │ │ │ │ │ - * Clear out the cached features. This destroys features, assuming │ │ │ │ │ - * nothing else has a reference. │ │ │ │ │ + * Method: write_wmc_ContactAddress │ │ │ │ │ */ │ │ │ │ │ - clearCache: function() { │ │ │ │ │ - if (this.features) { │ │ │ │ │ - for (var i = 0; i < this.features.length; ++i) { │ │ │ │ │ - this.features[i].destroy(); │ │ │ │ │ - } │ │ │ │ │ + write_wmc_ContactAddress: function(contactAddress) { │ │ │ │ │ + var node = this.createElementDefaultNS("ContactAddress"); │ │ │ │ │ + if (contactAddress.type) { │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "AddressType", contactAddress.type │ │ │ │ │ + )); │ │ │ │ │ } │ │ │ │ │ - this.features = null; │ │ │ │ │ - this.num = null; │ │ │ │ │ + if (contactAddress.address) { │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "Address", contactAddress.address │ │ │ │ │ + )); │ │ │ │ │ + } │ │ │ │ │ + if (contactAddress.city) { │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "City", contactAddress.city │ │ │ │ │ + )); │ │ │ │ │ + } │ │ │ │ │ + if (contactAddress.stateOrProvince) { │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "StateOrProvince", contactAddress.stateOrProvince │ │ │ │ │ + )); │ │ │ │ │ + } │ │ │ │ │ + if (contactAddress.postcode) { │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "PostCode", contactAddress.postcode │ │ │ │ │ + )); │ │ │ │ │ + } │ │ │ │ │ + if (contactAddress.country) { │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "Country", contactAddress.country │ │ │ │ │ + )); │ │ │ │ │ + } │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: pageCount │ │ │ │ │ - * Get the total count of pages given the current cache of features. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} The page count. │ │ │ │ │ + * Method: write_ol_MapExtension │ │ │ │ │ */ │ │ │ │ │ - pageCount: function() { │ │ │ │ │ - var numFeatures = this.features ? this.features.length : 0; │ │ │ │ │ - return Math.ceil(numFeatures / this.length); │ │ │ │ │ - }, │ │ │ │ │ + write_ol_MapExtension: function(context) { │ │ │ │ │ + var node = this.createElementDefaultNS("Extension"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: pageNum │ │ │ │ │ - * Get the zero based page number. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Integer} The current page number being displayed. │ │ │ │ │ - */ │ │ │ │ │ - pageNum: function() { │ │ │ │ │ - return this.num; │ │ │ │ │ + var bounds = context.maxExtent; │ │ │ │ │ + if (bounds) { │ │ │ │ │ + var maxExtent = this.createElementNS( │ │ │ │ │ + this.namespaces.ol, "ol:maxExtent" │ │ │ │ │ + ); │ │ │ │ │ + this.setAttributes(maxExtent, { │ │ │ │ │ + minx: bounds.left.toPrecision(18), │ │ │ │ │ + miny: bounds.bottom.toPrecision(18), │ │ │ │ │ + maxx: bounds.right.toPrecision(18), │ │ │ │ │ + maxy: bounds.top.toPrecision(18) │ │ │ │ │ + }); │ │ │ │ │ + node.appendChild(maxExtent); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: pageLength │ │ │ │ │ - * Gets or sets page length. │ │ │ │ │ + * Method: write_wmc_LayerList │ │ │ │ │ + * Create a LayerList node given an context object. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * newLength - {Integer} Optional length to be set. │ │ │ │ │ + * context - {Object} Context object. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Integer} The length of a page (number of features per page). │ │ │ │ │ + * {Element} A WMC LayerList element node. │ │ │ │ │ */ │ │ │ │ │ - pageLength: function(newLength) { │ │ │ │ │ - if (newLength && newLength > 0) { │ │ │ │ │ - this.length = newLength; │ │ │ │ │ + write_wmc_LayerList: function(context) { │ │ │ │ │ + var list = this.createElementDefaultNS("LayerList"); │ │ │ │ │ + │ │ │ │ │ + for (var i = 0, len = context.layersContext.length; i < len; ++i) { │ │ │ │ │ + list.appendChild(this.write_wmc_Layer(context.layersContext[i])); │ │ │ │ │ } │ │ │ │ │ - return this.length; │ │ │ │ │ + │ │ │ │ │ + return list; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: pageNext │ │ │ │ │ - * Display the next page of features. │ │ │ │ │ + * Method: write_wmc_Layer │ │ │ │ │ + * Create a Layer node given a layer context object. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * context - {Object} A layer context object.} │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} A new page was displayed. │ │ │ │ │ + * {Element} A WMC Layer element node. │ │ │ │ │ */ │ │ │ │ │ - pageNext: function(event) { │ │ │ │ │ - var changed = false; │ │ │ │ │ - if (this.features) { │ │ │ │ │ - if (this.num === null) { │ │ │ │ │ - this.num = -1; │ │ │ │ │ + write_wmc_Layer: function(context) { │ │ │ │ │ + var node = this.createElementDefaultNS( │ │ │ │ │ + "Layer", null, { │ │ │ │ │ + queryable: context.queryable ? "1" : "0", │ │ │ │ │ + hidden: context.visibility ? "0" : "1" │ │ │ │ │ } │ │ │ │ │ - var start = (this.num + 1) * this.length; │ │ │ │ │ - changed = this.page(start, event); │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + // required Server element │ │ │ │ │ + node.appendChild(this.write_wmc_Server(context)); │ │ │ │ │ + │ │ │ │ │ + // required Name element │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "Name", context.name │ │ │ │ │ + )); │ │ │ │ │ + │ │ │ │ │ + // required Title element │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "Title", context.title │ │ │ │ │ + )); │ │ │ │ │ + │ │ │ │ │ + // optional Abstract element │ │ │ │ │ + if (context["abstract"]) { │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "Abstract", context["abstract"] │ │ │ │ │ + )); │ │ │ │ │ } │ │ │ │ │ - return changed; │ │ │ │ │ + │ │ │ │ │ + // optional DataURL element │ │ │ │ │ + if (context.dataURL) { │ │ │ │ │ + node.appendChild(this.write_wmc_URLType("DataURL", context.dataURL)); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // optional MetadataURL element │ │ │ │ │ + if (context.metadataURL) { │ │ │ │ │ + node.appendChild(this.write_wmc_URLType("MetadataURL", context.metadataURL)); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: pagePrevious │ │ │ │ │ - * Display the previous page of features. │ │ │ │ │ + * Method: write_wmc_LayerExtension │ │ │ │ │ + * Add OpenLayers specific layer parameters to an Extension element. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * context - {Object} A layer context object. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} A new page was displayed. │ │ │ │ │ + * {Element} A WMC Extension element (for a layer). │ │ │ │ │ */ │ │ │ │ │ - pagePrevious: function() { │ │ │ │ │ - var changed = false; │ │ │ │ │ - if (this.features) { │ │ │ │ │ - if (this.num === null) { │ │ │ │ │ - this.num = this.pageCount(); │ │ │ │ │ + write_wmc_LayerExtension: function(context) { │ │ │ │ │ + var node = this.createElementDefaultNS("Extension"); │ │ │ │ │ + │ │ │ │ │ + var bounds = context.maxExtent; │ │ │ │ │ + var maxExtent = this.createElementNS( │ │ │ │ │ + this.namespaces.ol, "ol:maxExtent" │ │ │ │ │ + ); │ │ │ │ │ + this.setAttributes(maxExtent, { │ │ │ │ │ + minx: bounds.left.toPrecision(18), │ │ │ │ │ + miny: bounds.bottom.toPrecision(18), │ │ │ │ │ + maxx: bounds.right.toPrecision(18), │ │ │ │ │ + maxy: bounds.top.toPrecision(18) │ │ │ │ │ + }); │ │ │ │ │ + node.appendChild(maxExtent); │ │ │ │ │ + │ │ │ │ │ + if (context.tileSize && !context.singleTile) { │ │ │ │ │ + var size = this.createElementNS( │ │ │ │ │ + this.namespaces.ol, "ol:tileSize" │ │ │ │ │ + ); │ │ │ │ │ + this.setAttributes(size, context.tileSize); │ │ │ │ │ + node.appendChild(size); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var properties = [ │ │ │ │ │ + "transparent", "numZoomLevels", "units", "isBaseLayer", │ │ │ │ │ + "opacity", "displayInLayerSwitcher", "singleTile" │ │ │ │ │ + ]; │ │ │ │ │ + var child; │ │ │ │ │ + for (var i = 0, len = properties.length; i < len; ++i) { │ │ │ │ │ + child = this.createOLPropertyNode(context, properties[i]); │ │ │ │ │ + if (child) { │ │ │ │ │ + node.appendChild(child); │ │ │ │ │ } │ │ │ │ │ - var start = (this.num - 1) * this.length; │ │ │ │ │ - changed = this.page(start); │ │ │ │ │ } │ │ │ │ │ - return changed; │ │ │ │ │ + │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: page │ │ │ │ │ - * Display the page starting at the given index from the cache. │ │ │ │ │ + * Method: createOLPropertyNode │ │ │ │ │ + * Create a node representing an OpenLayers property. If the property is │ │ │ │ │ + * null or undefined, null will be returned. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} An object. │ │ │ │ │ + * prop - {String} A property. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} A new page was displayed. │ │ │ │ │ + * {Element} A property node. │ │ │ │ │ */ │ │ │ │ │ - page: function(start, event) { │ │ │ │ │ - var changed = false; │ │ │ │ │ - if (this.features) { │ │ │ │ │ - if (start >= 0 && start < this.features.length) { │ │ │ │ │ - var num = Math.floor(start / this.length); │ │ │ │ │ - if (num != this.num) { │ │ │ │ │ - this.paging = true; │ │ │ │ │ - var features = this.features.slice(start, start + this.length); │ │ │ │ │ - this.layer.removeFeatures(this.layer.features); │ │ │ │ │ - this.num = num; │ │ │ │ │ - // modify the event if any │ │ │ │ │ - if (event && event.features) { │ │ │ │ │ - // this.was called by an event listener │ │ │ │ │ - event.features = features; │ │ │ │ │ - } else { │ │ │ │ │ - // this was called directly on the strategy │ │ │ │ │ - this.layer.addFeatures(features); │ │ │ │ │ - } │ │ │ │ │ - this.paging = false; │ │ │ │ │ - changed = true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + createOLPropertyNode: function(obj, prop) { │ │ │ │ │ + var node = null; │ │ │ │ │ + if (obj[prop] != null) { │ │ │ │ │ + node = this.createElementNS(this.namespaces.ol, "ol:" + prop); │ │ │ │ │ + node.appendChild(this.createTextNode(obj[prop].toString())); │ │ │ │ │ } │ │ │ │ │ - return changed; │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Paging" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy/Filter.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Strategy.js │ │ │ │ │ - * @requires OpenLayers/Filter.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Strategy.Filter │ │ │ │ │ - * Strategy for limiting features that get added to a layer by │ │ │ │ │ - * evaluating a filter. The strategy maintains a cache of │ │ │ │ │ - * all features until removeFeatures is called on the layer. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Strategy> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Strategy.Filter = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: filter │ │ │ │ │ - * {<OpenLayers.Filter>} Filter for limiting features sent to the layer. │ │ │ │ │ - * Use the <setFilter> method to update this filter after construction. │ │ │ │ │ - */ │ │ │ │ │ - filter: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: cache │ │ │ │ │ - * {Array(<OpenLayers.Feature.Vector>)} List of currently cached │ │ │ │ │ - * features. │ │ │ │ │ - */ │ │ │ │ │ - cache: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: caching │ │ │ │ │ - * {Boolean} The filter is currently caching features. │ │ │ │ │ - */ │ │ │ │ │ - caching: false, │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy.Filter │ │ │ │ │ - * Create a new filter strategy. │ │ │ │ │ + * Method: write_wmc_Server │ │ │ │ │ + * Create a Server node given a layer context object. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ - * By default, this strategy automatically activates itself when a layer │ │ │ │ │ - * is added to a map. │ │ │ │ │ + * context - {Object} Layer context object. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} True if the strategy was successfully activated or false if │ │ │ │ │ - * the strategy was already active. │ │ │ │ │ + * {Element} A WMC Server element node. │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.cache = []; │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "beforefeaturesadded": this.handleAdd, │ │ │ │ │ - "beforefeaturesremoved": this.handleRemove, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + write_wmc_Server: function(context) { │ │ │ │ │ + var server = context.server; │ │ │ │ │ + var node = this.createElementDefaultNS("Server"); │ │ │ │ │ + var attributes = { │ │ │ │ │ + service: "OGC:WMS", │ │ │ │ │ + version: server.version │ │ │ │ │ + }; │ │ │ │ │ + if (server.title) { │ │ │ │ │ + attributes.title = server.title; │ │ │ │ │ } │ │ │ │ │ - return activated; │ │ │ │ │ + this.setAttributes(node, attributes); │ │ │ │ │ + │ │ │ │ │ + // required OnlineResource element │ │ │ │ │ + node.appendChild(this.write_wmc_OnlineResource(server.url)); │ │ │ │ │ + │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the strategy. Clear the feature cache. │ │ │ │ │ + * Method: write_wmc_URLType │ │ │ │ │ + * Create a LogoURL/DescriptionURL/MetadataURL/DataURL/LegendURL node given a object and elementName. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * elName - {String} Name of element (LogoURL/DescriptionURL/MetadataURL/LegendURL) │ │ │ │ │ + * url - {String} URL string value │ │ │ │ │ + * attr - {Object} Optional attributes (width, height, format) │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} True if the strategy was successfully deactivated or false if │ │ │ │ │ - * the strategy was already inactive. │ │ │ │ │ + * {Element} A WMC element node. │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - this.cache = null; │ │ │ │ │ - if (this.layer && this.layer.events) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - "beforefeaturesadded": this.handleAdd, │ │ │ │ │ - "beforefeaturesremoved": this.handleRemove, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + write_wmc_URLType: function(elName, url, attr) { │ │ │ │ │ + var node = this.createElementDefaultNS(elName); │ │ │ │ │ + node.appendChild(this.write_wmc_OnlineResource(url)); │ │ │ │ │ + if (attr) { │ │ │ │ │ + var optionalAttributes = ["width", "height", "format"]; │ │ │ │ │ + for (var i = 0; i < optionalAttributes.length; i++) { │ │ │ │ │ + if (optionalAttributes[i] in attr) { │ │ │ │ │ + node.setAttribute(optionalAttributes[i], attr[optionalAttributes[i]]); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Strategy.prototype.deactivate.apply(this, arguments); │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleAdd │ │ │ │ │ + * Method: write_wmc_DimensionList │ │ │ │ │ */ │ │ │ │ │ - handleAdd: function(event) { │ │ │ │ │ - if (!this.caching && this.filter) { │ │ │ │ │ - var features = event.features; │ │ │ │ │ - event.features = []; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0, ii = features.length; i < ii; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (this.filter.evaluate(feature)) { │ │ │ │ │ - event.features.push(feature); │ │ │ │ │ + write_wmc_DimensionList: function(context) { │ │ │ │ │ + var node = this.createElementDefaultNS("DimensionList"); │ │ │ │ │ + var required_attributes = { │ │ │ │ │ + name: true, │ │ │ │ │ + units: true, │ │ │ │ │ + unitSymbol: true, │ │ │ │ │ + userValue: true │ │ │ │ │ + }; │ │ │ │ │ + for (var dim in context.dimensions) { │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + var dimension = context.dimensions[dim]; │ │ │ │ │ + for (var name in dimension) { │ │ │ │ │ + if (typeof dimension[name] == "boolean") { │ │ │ │ │ + attributes[name] = Number(dimension[name]); │ │ │ │ │ } else { │ │ │ │ │ - this.cache.push(feature); │ │ │ │ │ + attributes[name] = dimension[name]; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + var values = ""; │ │ │ │ │ + if (attributes.values) { │ │ │ │ │ + values = attributes.values.join(","); │ │ │ │ │ + delete attributes.values; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleRemove │ │ │ │ │ - */ │ │ │ │ │ - handleRemove: function(event) { │ │ │ │ │ - if (!this.caching) { │ │ │ │ │ - this.cache = []; │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "Dimension", values, attributes │ │ │ │ │ + )); │ │ │ │ │ } │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setFilter │ │ │ │ │ - * Update the filter for this strategy. This will re-evaluate │ │ │ │ │ - * any features on the layer and in the cache. Only features │ │ │ │ │ - * for which filter.evalute(feature) returns true will be │ │ │ │ │ - * added to the layer. Others will be cached by the strategy. │ │ │ │ │ + /** │ │ │ │ │ + * Method: write_wmc_FormatList │ │ │ │ │ + * Create a FormatList node given a layer context. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} A filter for evaluating features. │ │ │ │ │ + * context - {Object} Layer context object. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Element} A WMC FormatList element node. │ │ │ │ │ */ │ │ │ │ │ - setFilter: function(filter) { │ │ │ │ │ - this.filter = filter; │ │ │ │ │ - var previousCache = this.cache; │ │ │ │ │ - this.cache = []; │ │ │ │ │ - // look through layer for features to remove from layer │ │ │ │ │ - this.handleAdd({ │ │ │ │ │ - features: this.layer.features │ │ │ │ │ - }); │ │ │ │ │ - // cache now contains features to remove from layer │ │ │ │ │ - if (this.cache.length > 0) { │ │ │ │ │ - this.caching = true; │ │ │ │ │ - this.layer.removeFeatures(this.cache.slice()); │ │ │ │ │ - this.caching = false; │ │ │ │ │ - } │ │ │ │ │ - // now look through previous cache for features to add to layer │ │ │ │ │ - if (previousCache.length > 0) { │ │ │ │ │ - var event = { │ │ │ │ │ - features: previousCache │ │ │ │ │ - }; │ │ │ │ │ - this.handleAdd(event); │ │ │ │ │ - if (event.features.length > 0) { │ │ │ │ │ - // event has features to add to layer │ │ │ │ │ - this.caching = true; │ │ │ │ │ - this.layer.addFeatures(event.features); │ │ │ │ │ - this.caching = false; │ │ │ │ │ - } │ │ │ │ │ + write_wmc_FormatList: function(context) { │ │ │ │ │ + var node = this.createElementDefaultNS("FormatList"); │ │ │ │ │ + for (var i = 0, len = context.formats.length; i < len; i++) { │ │ │ │ │ + var format = context.formats[i]; │ │ │ │ │ + node.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "Format", │ │ │ │ │ + format.value, │ │ │ │ │ + (format.current && format.current == true) ? { │ │ │ │ │ + current: "1" │ │ │ │ │ + } : null │ │ │ │ │ + )); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Filter" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy/Refresh.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Strategy.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Strategy.Refresh │ │ │ │ │ - * A strategy that refreshes the layer. By default the strategy waits for a │ │ │ │ │ - * call to <refresh> before refreshing. By configuring the strategy with │ │ │ │ │ - * the <interval> option, refreshing can take place automatically. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Strategy> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Strategy.Refresh = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: force │ │ │ │ │ - * {Boolean} Force a refresh on the layer. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - force: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: interval │ │ │ │ │ - * {Number} Auto-refresh. Default is 0. If > 0, layer will be refreshed │ │ │ │ │ - * every N milliseconds. │ │ │ │ │ - */ │ │ │ │ │ - interval: 0, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: timer │ │ │ │ │ - * {Number} The id of the timer. │ │ │ │ │ - */ │ │ │ │ │ - timer: null, │ │ │ │ │ + return node; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy.Refresh │ │ │ │ │ - * Create a new Refresh strategy. │ │ │ │ │ + * Method: write_wmc_StyleList │ │ │ │ │ + * Create a StyleList node given a layer context. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ - * │ │ │ │ │ + * layer - {Object} Layer context object. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} True if the strategy was successfully activated. │ │ │ │ │ + * {Element} A WMC StyleList element node. │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - if (this.layer.visibility === true) { │ │ │ │ │ - this.start(); │ │ │ │ │ + write_wmc_StyleList: function(layer) { │ │ │ │ │ + var node = this.createElementDefaultNS("StyleList"); │ │ │ │ │ + │ │ │ │ │ + var styles = layer.styles; │ │ │ │ │ + if (styles && OpenLayers.Util.isArray(styles)) { │ │ │ │ │ + var sld; │ │ │ │ │ + for (var i = 0, len = styles.length; i < len; i++) { │ │ │ │ │ + var s = styles[i]; │ │ │ │ │ + // three style types to consider │ │ │ │ │ + // [1] linked SLD │ │ │ │ │ + // [2] inline SLD │ │ │ │ │ + // [3] named style │ │ │ │ │ + // running child nodes always gets name, optionally gets href or body │ │ │ │ │ + var style = this.createElementDefaultNS( │ │ │ │ │ + "Style", │ │ │ │ │ + null, │ │ │ │ │ + (s.current && s.current == true) ? { │ │ │ │ │ + current: "1" │ │ │ │ │ + } : null │ │ │ │ │ + ); │ │ │ │ │ + if (s.href) { // [1] │ │ │ │ │ + sld = this.createElementDefaultNS("SLD"); │ │ │ │ │ + // Name is optional. │ │ │ │ │ + if (s.name) { │ │ │ │ │ + sld.appendChild(this.createElementDefaultNS("Name", s.name)); │ │ │ │ │ + } │ │ │ │ │ + // Title is optional. │ │ │ │ │ + if (s.title) { │ │ │ │ │ + sld.appendChild(this.createElementDefaultNS("Title", s.title)); │ │ │ │ │ + } │ │ │ │ │ + // LegendURL is optional │ │ │ │ │ + if (s.legend) { │ │ │ │ │ + sld.appendChild(this.write_wmc_URLType("LegendURL", s.legend.href, s.legend)); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var link = this.write_wmc_OnlineResource(s.href); │ │ │ │ │ + sld.appendChild(link); │ │ │ │ │ + style.appendChild(sld); │ │ │ │ │ + } else if (s.body) { // [2] │ │ │ │ │ + sld = this.createElementDefaultNS("SLD"); │ │ │ │ │ + // Name is optional. │ │ │ │ │ + if (s.name) { │ │ │ │ │ + sld.appendChild(this.createElementDefaultNS("Name", s.name)); │ │ │ │ │ + } │ │ │ │ │ + // Title is optional. │ │ │ │ │ + if (s.title) { │ │ │ │ │ + sld.appendChild(this.createElementDefaultNS("Title", s.title)); │ │ │ │ │ + } │ │ │ │ │ + // LegendURL is optional │ │ │ │ │ + if (s.legend) { │ │ │ │ │ + sld.appendChild(this.write_wmc_URLType("LegendURL", s.legend.href, s.legend)); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // read in body as xml doc - assume proper namespace declarations │ │ │ │ │ + var doc = OpenLayers.Format.XML.prototype.read.apply(this, [s.body]); │ │ │ │ │ + // append to StyledLayerDescriptor node │ │ │ │ │ + var imported = doc.documentElement; │ │ │ │ │ + if (sld.ownerDocument && sld.ownerDocument.importNode) { │ │ │ │ │ + imported = sld.ownerDocument.importNode(imported, true); │ │ │ │ │ + } │ │ │ │ │ + sld.appendChild(imported); │ │ │ │ │ + style.appendChild(sld); │ │ │ │ │ + } else { // [3] │ │ │ │ │ + // both Name and Title are required. │ │ │ │ │ + style.appendChild(this.createElementDefaultNS("Name", s.name)); │ │ │ │ │ + style.appendChild(this.createElementDefaultNS("Title", s.title)); │ │ │ │ │ + // Abstract is optional │ │ │ │ │ + if (s['abstract']) { // abstract is a js keyword │ │ │ │ │ + style.appendChild(this.createElementDefaultNS( │ │ │ │ │ + "Abstract", s['abstract'] │ │ │ │ │ + )); │ │ │ │ │ + } │ │ │ │ │ + // LegendURL is optional │ │ │ │ │ + if (s.legend) { │ │ │ │ │ + style.appendChild(this.write_wmc_URLType("LegendURL", s.legend.href, s.legend)); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + node.appendChild(style); │ │ │ │ │ } │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "visibilitychanged": this.reset, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ - * tear-down. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} True if the strategy was successfully deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.stop(); │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - "visibilitychanged": this.reset, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: reset │ │ │ │ │ - * Start or cancel the refresh interval depending on the visibility of │ │ │ │ │ - * the layer. │ │ │ │ │ + * Method: write_wmc_OnlineResource │ │ │ │ │ + * Create an OnlineResource node given a URL. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * href - {String} URL for the resource. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Element} A WMC OnlineResource element node. │ │ │ │ │ */ │ │ │ │ │ - reset: function() { │ │ │ │ │ - if (this.layer.visibility === true) { │ │ │ │ │ - this.start(); │ │ │ │ │ - } else { │ │ │ │ │ - this.stop(); │ │ │ │ │ - } │ │ │ │ │ + write_wmc_OnlineResource: function(href) { │ │ │ │ │ + var node = this.createElementDefaultNS("OnlineResource"); │ │ │ │ │ + this.setAttributeNS(node, this.namespaces.xlink, "xlink:type", "simple"); │ │ │ │ │ + this.setAttributeNS(node, this.namespaces.xlink, "xlink:href", href); │ │ │ │ │ + return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: start │ │ │ │ │ - * Start the refresh interval. │ │ │ │ │ + * Method: getOnlineResource_href │ │ │ │ │ */ │ │ │ │ │ - start: function() { │ │ │ │ │ - if (this.interval && typeof this.interval === "number" && │ │ │ │ │ - this.interval > 0) { │ │ │ │ │ - │ │ │ │ │ - this.timer = window.setInterval( │ │ │ │ │ - OpenLayers.Function.bind(this.refresh, this), │ │ │ │ │ - this.interval); │ │ │ │ │ + getOnlineResource_href: function(node) { │ │ │ │ │ + var object = {}; │ │ │ │ │ + var links = node.getElementsByTagName("OnlineResource"); │ │ │ │ │ + if (links.length > 0) { │ │ │ │ │ + this.read_wmc_OnlineResource(object, links[0]); │ │ │ │ │ } │ │ │ │ │ + return object.href; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: refresh │ │ │ │ │ - * Tell the strategy to refresh which will refresh the layer. │ │ │ │ │ - */ │ │ │ │ │ - refresh: function() { │ │ │ │ │ - if (this.layer && this.layer.refresh && │ │ │ │ │ - typeof this.layer.refresh == "function") { │ │ │ │ │ - │ │ │ │ │ - this.layer.refresh({ │ │ │ │ │ - force: this.force │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: stop │ │ │ │ │ - * Cancels the refresh interval. │ │ │ │ │ - */ │ │ │ │ │ - stop: function() { │ │ │ │ │ - if (this.timer !== null) { │ │ │ │ │ - window.clearInterval(this.timer); │ │ │ │ │ - this.timer = null; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMC.v1" │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Refresh" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy/Save.js │ │ │ │ │ + OpenLayers/Format/WMC/v1_1_0.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Strategy.js │ │ │ │ │ + * @requires OpenLayers/Format/WMC/v1.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Strategy.Save │ │ │ │ │ - * A strategy that commits newly created or modified features. By default │ │ │ │ │ - * the strategy waits for a call to <save> before persisting changes. By │ │ │ │ │ - * configuring the strategy with the <auto> option, changes can be saved │ │ │ │ │ - * automatically. │ │ │ │ │ + * Class: OpenLayers.Format.WMC.v1_1_0 │ │ │ │ │ + * Read and write WMC version 1.1.0. │ │ │ │ │ * │ │ │ │ │ + * Differences between 1.1.0 and 1.0.0: │ │ │ │ │ + * - 1.1.0 Layers have optional sld:MinScaleDenominator and │ │ │ │ │ + * sld:MaxScaleDenominator │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Strategy> │ │ │ │ │ + * - <OpenLayers.Format.WMC.v1> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Strategy.Save = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ +OpenLayers.Format.WMC.v1_1_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WMC.v1, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} An events object that handles all │ │ │ │ │ - * events on the strategy object. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * strategy.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Supported event types: │ │ │ │ │ - * start - Triggered before saving │ │ │ │ │ - * success - Triggered after a successful transaction │ │ │ │ │ - * fail - Triggered after a failed transaction │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Constant: VERSION │ │ │ │ │ + * {String} 1.1.0 │ │ │ │ │ + */ │ │ │ │ │ + VERSION: "1.1.0", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for triggering this protocol │ │ │ │ │ - * events. │ │ │ │ │ - */ │ │ │ │ │ - events: null, │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} http://www.opengis.net/context │ │ │ │ │ + * http://schemas.opengis.net/context/1.1.0/context.xsd │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: "http://www.opengis.net/context http://schemas.opengis.net/context/1.1.0/context.xsd", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: auto │ │ │ │ │ - * {Boolean | Number} Auto-save. Default is false. If true, features will be │ │ │ │ │ - * saved immediately after being added to the layer and with each │ │ │ │ │ - * modification or deletion. If auto is a number, features will be │ │ │ │ │ - * saved on an interval provided by the value (in seconds). │ │ │ │ │ - */ │ │ │ │ │ - auto: false, │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMC.v1_1_0 │ │ │ │ │ + * Instances of this class are not created directly. Use the │ │ │ │ │ + * <OpenLayers.Format.WMC> constructor instead. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.WMC.v1.prototype.initialize.apply( │ │ │ │ │ + this, [options] │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: timer │ │ │ │ │ - * {Number} The id of the timer. │ │ │ │ │ - */ │ │ │ │ │ - timer: null, │ │ │ │ │ + /** │ │ │ │ │ + * Method: read_sld_MinScaleDenominator │ │ │ │ │ + * Read a sld:MinScaleDenominator node. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layerContext - {Object} An object representing a layer. │ │ │ │ │ + * node - {Element} An element node. │ │ │ │ │ + */ │ │ │ │ │ + read_sld_MinScaleDenominator: function(layerContext, node) { │ │ │ │ │ + var minScaleDenominator = parseFloat(this.getChildValue(node)); │ │ │ │ │ + if (minScaleDenominator > 0) { │ │ │ │ │ + layerContext.maxScale = minScaleDenominator; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy.Save │ │ │ │ │ - * Create a new Save strategy. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Strategy.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.events = new OpenLayers.Events(this); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: read_sld_MaxScaleDenominator │ │ │ │ │ + * Read a sld:MaxScaleDenominator node. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layerContext - {Object} An object representing a layer. │ │ │ │ │ + * node - {Element} An element node. │ │ │ │ │ + */ │ │ │ │ │ + read_sld_MaxScaleDenominator: function(layerContext, node) { │ │ │ │ │ + layerContext.minScale = parseFloat(this.getChildValue(node)); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - if (this.auto) { │ │ │ │ │ - if (typeof this.auto === "number") { │ │ │ │ │ - this.timer = window.setInterval( │ │ │ │ │ - OpenLayers.Function.bind(this.save, this), │ │ │ │ │ - this.auto * 1000 │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "featureadded": this.triggerSave, │ │ │ │ │ - "afterfeaturemodified": this.triggerSave, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: read_wmc_SRS │ │ │ │ │ + */ │ │ │ │ │ + read_wmc_SRS: function(layerContext, node) { │ │ │ │ │ + if (!("srs" in layerContext)) { │ │ │ │ │ + layerContext.srs = {}; │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ + layerContext.srs[this.getChildValue(node)] = true; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ - * tear-down. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - if (this.auto) { │ │ │ │ │ - if (typeof this.auto === "number") { │ │ │ │ │ - window.clearInterval(this.timer); │ │ │ │ │ - } else { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - "featureadded": this.triggerSave, │ │ │ │ │ - "afterfeaturemodified": this.triggerSave, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: write_wmc_Layer │ │ │ │ │ + * Create a Layer node given a layer context object. This method adds │ │ │ │ │ + * elements specific to version 1.1.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * context - {Object} A layer context object.} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Element} A WMC Layer element node. │ │ │ │ │ + */ │ │ │ │ │ + write_wmc_Layer: function(context) { │ │ │ │ │ + var node = OpenLayers.Format.WMC.v1.prototype.write_wmc_Layer.apply( │ │ │ │ │ + this, [context] │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: triggerSave │ │ │ │ │ - * Registered as a listener. Calls save if a feature has insert, update, │ │ │ │ │ - * or delete state. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * event - {Object} The event this function is listening for. │ │ │ │ │ - */ │ │ │ │ │ - triggerSave: function(event) { │ │ │ │ │ - var feature = event.feature; │ │ │ │ │ - if (feature.state === OpenLayers.State.INSERT || │ │ │ │ │ - feature.state === OpenLayers.State.UPDATE || │ │ │ │ │ - feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ - this.save([event.feature]); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + // min/max scale denominator elements go before the 4th element in v1 │ │ │ │ │ + if (context.maxScale) { │ │ │ │ │ + var minSD = this.createElementNS( │ │ │ │ │ + this.namespaces.sld, "sld:MinScaleDenominator" │ │ │ │ │ + ); │ │ │ │ │ + minSD.appendChild(this.createTextNode(context.maxScale.toPrecision(16))); │ │ │ │ │ + node.appendChild(minSD); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: save │ │ │ │ │ - * Tell the layer protocol to commit unsaved features. If the layer │ │ │ │ │ - * projection differs from the map projection, features will be │ │ │ │ │ - * transformed into the layer projection before being committed. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array} Features to be saved. If null, then default is all │ │ │ │ │ - * features in the layer. Features are assumed to be in the map │ │ │ │ │ - * projection. │ │ │ │ │ - */ │ │ │ │ │ - save: function(features) { │ │ │ │ │ - if (!features) { │ │ │ │ │ - features = this.layer.features; │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("start", { │ │ │ │ │ - features: features │ │ │ │ │ - }); │ │ │ │ │ - var remote = this.layer.projection; │ │ │ │ │ - var local = this.layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var len = features.length; │ │ │ │ │ - var clones = new Array(len); │ │ │ │ │ - var orig, clone; │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - orig = features[i]; │ │ │ │ │ - clone = orig.clone(); │ │ │ │ │ - clone.fid = orig.fid; │ │ │ │ │ - clone.state = orig.state; │ │ │ │ │ - if (orig.url) { │ │ │ │ │ - clone.url = orig.url; │ │ │ │ │ - } │ │ │ │ │ - clone._original = orig; │ │ │ │ │ - clone.geometry.transform(local, remote); │ │ │ │ │ - clones[i] = clone; │ │ │ │ │ + if (context.minScale) { │ │ │ │ │ + var maxSD = this.createElementNS( │ │ │ │ │ + this.namespaces.sld, "sld:MaxScaleDenominator" │ │ │ │ │ + ); │ │ │ │ │ + maxSD.appendChild(this.createTextNode(context.minScale.toPrecision(16))); │ │ │ │ │ + node.appendChild(maxSD); │ │ │ │ │ } │ │ │ │ │ - features = clones; │ │ │ │ │ - } │ │ │ │ │ - this.layer.protocol.commit(features, { │ │ │ │ │ - callback: this.onCommit, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: onCommit │ │ │ │ │ - * Called after protocol commit. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} A response object. │ │ │ │ │ - */ │ │ │ │ │ - onCommit: function(response) { │ │ │ │ │ - var evt = { │ │ │ │ │ - "response": response │ │ │ │ │ - }; │ │ │ │ │ - if (response.success()) { │ │ │ │ │ - var features = response.reqFeatures; │ │ │ │ │ - // deal with inserts, updates, and deletes │ │ │ │ │ - var state, feature; │ │ │ │ │ - var destroys = []; │ │ │ │ │ - var insertIds = response.insertIds || []; │ │ │ │ │ - var j = 0; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - // if projection was different, we may be dealing with clones │ │ │ │ │ - feature = feature._original || feature; │ │ │ │ │ - state = feature.state; │ │ │ │ │ - if (state) { │ │ │ │ │ - if (state == OpenLayers.State.DELETE) { │ │ │ │ │ - destroys.push(feature); │ │ │ │ │ - } else if (state == OpenLayers.State.INSERT) { │ │ │ │ │ - feature.fid = insertIds[j]; │ │ │ │ │ - ++j; │ │ │ │ │ - } │ │ │ │ │ - feature.state = null; │ │ │ │ │ + // optional SRS element(s) │ │ │ │ │ + if (context.srs) { │ │ │ │ │ + for (var name in context.srs) { │ │ │ │ │ + node.appendChild(this.createElementDefaultNS("SRS", name)); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - if (destroys.length > 0) { │ │ │ │ │ - this.layer.destroyFeatures(destroys); │ │ │ │ │ + // optional FormatList element │ │ │ │ │ + node.appendChild(this.write_wmc_FormatList(context)); │ │ │ │ │ + │ │ │ │ │ + // optional StyleList element │ │ │ │ │ + node.appendChild(this.write_wmc_StyleList(context)); │ │ │ │ │ + │ │ │ │ │ + // optional DimensionList element │ │ │ │ │ + if (context.dimensions) { │ │ │ │ │ + node.appendChild(this.write_wmc_DimensionList(context)); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - this.events.triggerEvent("success", evt); │ │ │ │ │ + // OpenLayers specific properties go in an Extension element │ │ │ │ │ + node.appendChild(this.write_wmc_LayerExtension(context)); │ │ │ │ │ │ │ │ │ │ - } else { │ │ │ │ │ - this.events.triggerEvent("fail", evt); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + return node; │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Save" │ │ │ │ │ -}); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMC.v1_1_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy/Fixed.js │ │ │ │ │ + OpenLayers/Format/WMC/v1_0_0.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Strategy.js │ │ │ │ │ + * @requires OpenLayers/Format/WMC/v1.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Strategy.Fixed │ │ │ │ │ - * A simple strategy that requests features once and never requests new data. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Format.WMC.v1_0_0 │ │ │ │ │ + * Read and write WMC version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Strategy> │ │ │ │ │ + * - <OpenLayers.Format.WMC.v1> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Strategy.Fixed = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ +OpenLayers.Format.WMC.v1_0_0 = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Format.WMC.v1, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: preload │ │ │ │ │ - * {Boolean} Load data before layer made visible. Enabling this may result │ │ │ │ │ - * in considerable overhead if your application loads many data layers │ │ │ │ │ - * that are not visible by default. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - preload: false, │ │ │ │ │ + /** │ │ │ │ │ + * Constant: VERSION │ │ │ │ │ + * {String} 1.0.0 │ │ │ │ │ + */ │ │ │ │ │ + VERSION: "1.0.0", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy.Fixed │ │ │ │ │ - * Create a new Fixed strategy. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: schemaLocation │ │ │ │ │ + * {String} http://www.opengis.net/context │ │ │ │ │ + * http://schemas.opengis.net/context/1.0.0/context.xsd │ │ │ │ │ + */ │ │ │ │ │ + schemaLocation: "http://www.opengis.net/context http://schemas.opengis.net/context/1.0.0/context.xsd", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Activate the strategy: load data or add listener to load when visible │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} True if the strategy was successfully activated or false if │ │ │ │ │ - * the strategy was already active. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "refresh": this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - if (this.layer.visibility == true || this.preload) { │ │ │ │ │ - this.load(); │ │ │ │ │ - } else { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "visibilitychanged": this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Format.WMC.v1_0_0 │ │ │ │ │ + * Instances of this class are not created directly. Use the │ │ │ │ │ + * <OpenLayers.Format.WMC> constructor instead. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * this instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.WMC.v1.prototype.initialize.apply( │ │ │ │ │ + this, [options] │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Deactivate the strategy. Undo what is done in <activate>. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - "refresh": this.load, │ │ │ │ │ - "visibilitychanged": this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: read_wmc_SRS │ │ │ │ │ + */ │ │ │ │ │ + read_wmc_SRS: function(layerContext, node) { │ │ │ │ │ + var srs = this.getChildValue(node); │ │ │ │ │ + if (typeof layerContext.projections != "object") { │ │ │ │ │ + layerContext.projections = {}; │ │ │ │ │ + } │ │ │ │ │ + var values = srs.split(/ +/); │ │ │ │ │ + for (var i = 0, len = values.length; i < len; i++) { │ │ │ │ │ + layerContext.projections[values[i]] = true; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: load │ │ │ │ │ - * Tells protocol to load data and unhooks the visibilitychanged event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} options to pass to protocol read. │ │ │ │ │ - */ │ │ │ │ │ - load: function(options) { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - layer.events.triggerEvent("loadstart", { │ │ │ │ │ - filter: layer.filter │ │ │ │ │ - }); │ │ │ │ │ - layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: this.merge, │ │ │ │ │ - filter: layer.filter, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options)); │ │ │ │ │ - layer.events.un({ │ │ │ │ │ - "visibilitychanged": this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: write_wmc_Layer │ │ │ │ │ + * Create a Layer node given a layer context object. This method adds │ │ │ │ │ + * elements specific to version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * context - {Object} A layer context object.} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Element} A WMC Layer element node. │ │ │ │ │ + */ │ │ │ │ │ + write_wmc_Layer: function(context) { │ │ │ │ │ + var node = OpenLayers.Format.WMC.v1.prototype.write_wmc_Layer.apply( │ │ │ │ │ + this, [context] │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: merge │ │ │ │ │ - * Add all features to the layer. │ │ │ │ │ - * If the layer projection differs from the map projection, features │ │ │ │ │ - * will be transformed from the layer projection to the map projection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object passed │ │ │ │ │ - * by the protocol. │ │ │ │ │ - */ │ │ │ │ │ - merge: function(resp) { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - layer.destroyFeatures(); │ │ │ │ │ - var features = resp.features; │ │ │ │ │ - if (features && features.length > 0) { │ │ │ │ │ - var remote = layer.projection; │ │ │ │ │ - var local = layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var geom; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - geom = features[i].geometry; │ │ │ │ │ - if (geom) { │ │ │ │ │ - geom.transform(remote, local); │ │ │ │ │ - } │ │ │ │ │ + // optional SRS element(s) │ │ │ │ │ + if (context.srs) { │ │ │ │ │ + var projections = []; │ │ │ │ │ + for (var name in context.srs) { │ │ │ │ │ + projections.push(name); │ │ │ │ │ } │ │ │ │ │ + node.appendChild(this.createElementDefaultNS("SRS", projections.join(" "))); │ │ │ │ │ } │ │ │ │ │ - layer.addFeatures(features); │ │ │ │ │ - } │ │ │ │ │ - layer.events.triggerEvent("loadend", { │ │ │ │ │ - response: resp │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Fixed" │ │ │ │ │ -}); │ │ │ │ │ + // optional FormatList element │ │ │ │ │ + node.appendChild(this.write_wmc_FormatList(context)); │ │ │ │ │ + │ │ │ │ │ + // optional StyleList element │ │ │ │ │ + node.appendChild(this.write_wmc_StyleList(context)); │ │ │ │ │ + │ │ │ │ │ + // optional DimensionList element │ │ │ │ │ + if (context.dimensions) { │ │ │ │ │ + node.appendChild(this.write_wmc_DimensionList(context)); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // OpenLayers specific properties go in an Extension element │ │ │ │ │ + node.appendChild(this.write_wmc_LayerExtension(context)); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMC.v1_0_0" │ │ │ │ │ + │ │ │ │ │ + }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Strategy/BBOX.js │ │ │ │ │ + OpenLayers/Events/buttonclick.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Strategy.js │ │ │ │ │ - * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Strategy.BBOX │ │ │ │ │ - * A simple strategy that reads new features when the viewport invalidates │ │ │ │ │ - * some bounds. │ │ │ │ │ + * Class: OpenLayers.Events.buttonclick │ │ │ │ │ + * Extension event type for handling buttons on top of a dom element. This │ │ │ │ │ + * event type fires "buttonclick" on its <target> when a button was │ │ │ │ │ + * clicked. Buttons are detected by the "olButton" class. │ │ │ │ │ * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Strategy> │ │ │ │ │ + * This event type makes sure that button clicks do not interfere with other │ │ │ │ │ + * events that are registered on the same <element>. │ │ │ │ │ + * │ │ │ │ │ + * Event types provided by this extension: │ │ │ │ │ + * - *buttonclick* Triggered when a button is clicked. Listeners receive an │ │ │ │ │ + * object with a *buttonElement* property referencing the dom element of │ │ │ │ │ + * the clicked button, and an *buttonXY* property with the click position │ │ │ │ │ + * relative to the button. │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Strategy.BBOX = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ +OpenLayers.Events.buttonclick = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: bounds │ │ │ │ │ - * {<OpenLayers.Bounds>} The current data bounds (in the same projection │ │ │ │ │ - * as the layer - not always the same projection as the map). │ │ │ │ │ - */ │ │ │ │ │ - bounds: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: resolution │ │ │ │ │ - * {Float} The current data resolution. │ │ │ │ │ + * Property: target │ │ │ │ │ + * {<OpenLayers.Events>} The events instance that the buttonclick event will │ │ │ │ │ + * be triggered on. │ │ │ │ │ */ │ │ │ │ │ - resolution: null, │ │ │ │ │ + target: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: ratio │ │ │ │ │ - * {Float} The ratio of the data bounds to the viewport bounds (in each │ │ │ │ │ - * dimension). Default is 2. │ │ │ │ │ - */ │ │ │ │ │ - ratio: 2, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: resFactor │ │ │ │ │ - * {Float} Optional factor used to determine when previously requested │ │ │ │ │ - * features are invalid. If set, the resFactor will be compared to the │ │ │ │ │ - * resolution of the previous request to the current map resolution. │ │ │ │ │ - * If resFactor > (old / new) and 1/resFactor < (old / new). If you │ │ │ │ │ - * set a resFactor of 1, data will be requested every time the │ │ │ │ │ - * resolution changes. If you set a resFactor of 3, data will be │ │ │ │ │ - * requested if the old resolution is 3 times the new, or if the new is │ │ │ │ │ - * 3 times the old. If the old bounds do not contain the new bounds │ │ │ │ │ - * new data will always be requested (with or without considering │ │ │ │ │ - * resFactor). │ │ │ │ │ + * Property: events │ │ │ │ │ + * {Array} Events to observe and conditionally stop from propagating when │ │ │ │ │ + * an element with the olButton class (or its olAlphaImg child) is │ │ │ │ │ + * clicked. │ │ │ │ │ */ │ │ │ │ │ - resFactor: null, │ │ │ │ │ + events: [ │ │ │ │ │ + 'mousedown', 'mouseup', 'click', 'dblclick', │ │ │ │ │ + 'touchstart', 'touchmove', 'touchend', 'keydown' │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: response │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} The protocol response object returned │ │ │ │ │ - * by the layer protocol. │ │ │ │ │ + * Property: startRegEx │ │ │ │ │ + * {RegExp} Regular expression to test Event.type for events that start │ │ │ │ │ + * a buttonclick sequence. │ │ │ │ │ */ │ │ │ │ │ - response: null, │ │ │ │ │ + startRegEx: /^mousedown|touchstart$/, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Strategy.BBOX │ │ │ │ │ - * Create a new BBOX strategy. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * Property: cancelRegEx │ │ │ │ │ + * {RegExp} Regular expression to test Event.type for events that cancel │ │ │ │ │ + * a buttonclick sequence. │ │ │ │ │ */ │ │ │ │ │ + cancelRegEx: /^touchmove$/, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Set up strategy with regard to reading new batches of remote data. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully activated. │ │ │ │ │ + * Property: completeRegEx │ │ │ │ │ + * {RegExp} Regular expression to test Event.type for events that complete │ │ │ │ │ + * a buttonclick sequence. │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - "moveend": this.update, │ │ │ │ │ - "refresh": this.update, │ │ │ │ │ - "visibilitychanged": this.update, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.update(); │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ + completeRegEx: /^mouseup|touchend$/, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Tear down strategy with regard to reading new batches of remote data. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ + * Property: startEvt │ │ │ │ │ + * {Event} The event that started the click sequence │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - "moveend": this.update, │ │ │ │ │ - "refresh": this.update, │ │ │ │ │ - "visibilitychanged": this.update, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: update │ │ │ │ │ - * Callback function called on "moveend" or "refresh" layer events. │ │ │ │ │ + * Constructor: OpenLayers.Events.buttonclick │ │ │ │ │ + * Construct a buttonclick event type. Applications are not supposed to │ │ │ │ │ + * create instances of this class - they are created on demand by │ │ │ │ │ + * <OpenLayers.Events> instances. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will determine │ │ │ │ │ - * the behaviour of this Strategy │ │ │ │ │ - * │ │ │ │ │ - * Valid options include: │ │ │ │ │ - * force - {Boolean} if true, new data must be unconditionally read. │ │ │ │ │ - * noAbort - {Boolean} if true, do not abort previous requests. │ │ │ │ │ + * target - {<OpenLayers.Events>} The events instance that the buttonclick │ │ │ │ │ + * event will be triggered on. │ │ │ │ │ */ │ │ │ │ │ - update: function(options) { │ │ │ │ │ - var mapBounds = this.getMapBounds(); │ │ │ │ │ - if (mapBounds !== null && ((options && options.force) || │ │ │ │ │ - (this.layer.visibility && this.layer.calculateInRange() && this.invalidBounds(mapBounds)))) { │ │ │ │ │ - this.calculateBounds(mapBounds); │ │ │ │ │ - this.resolution = this.layer.map.getResolution(); │ │ │ │ │ - this.triggerRead(options); │ │ │ │ │ + initialize: function(target) { │ │ │ │ │ + this.target = target; │ │ │ │ │ + for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ + this.target.register(this.events[i], this, this.buttonClick, { │ │ │ │ │ + extension: true │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getMapBounds │ │ │ │ │ - * Get the map bounds expressed in the same projection as this layer. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Bounds>} Map bounds in the projection of the layer. │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ - getMapBounds: function() { │ │ │ │ │ - if (this.layer.map === null) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - var bounds = this.layer.map.getExtent(); │ │ │ │ │ - if (bounds && !this.layer.projection.equals( │ │ │ │ │ - this.layer.map.getProjectionObject())) { │ │ │ │ │ - bounds = bounds.clone().transform( │ │ │ │ │ - this.layer.map.getProjectionObject(), this.layer.projection │ │ │ │ │ - ); │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ + this.target.unregister(this.events[i], this, this.buttonClick); │ │ │ │ │ } │ │ │ │ │ - return bounds; │ │ │ │ │ + delete this.target; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: invalidBounds │ │ │ │ │ - * Determine whether the previously requested set of features is invalid. │ │ │ │ │ - * This occurs when the new map bounds do not contain the previously │ │ │ │ │ - * requested bounds. In addition, if <resFactor> is set, it will be │ │ │ │ │ - * considered. │ │ │ │ │ + * Method: getPressedButton │ │ │ │ │ + * Get the pressed button, if any. Returns undefined if no button │ │ │ │ │ + * was pressed. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * mapBounds - {<OpenLayers.Bounds>} the current map extent, will be │ │ │ │ │ - * retrieved from the map object if not provided │ │ │ │ │ + * Arguments: │ │ │ │ │ + * element - {DOMElement} The event target. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Boolean} │ │ │ │ │ + * {DOMElement} The button element, or undefined. │ │ │ │ │ */ │ │ │ │ │ - invalidBounds: function(mapBounds) { │ │ │ │ │ - if (!mapBounds) { │ │ │ │ │ - mapBounds = this.getMapBounds(); │ │ │ │ │ - } │ │ │ │ │ - var invalid = !this.bounds || !this.bounds.containsBounds(mapBounds); │ │ │ │ │ - if (!invalid && this.resFactor) { │ │ │ │ │ - var ratio = this.resolution / this.layer.map.getResolution(); │ │ │ │ │ - invalid = (ratio >= this.resFactor || ratio <= (1 / this.resFactor)); │ │ │ │ │ - } │ │ │ │ │ - return invalid; │ │ │ │ │ + getPressedButton: function(element) { │ │ │ │ │ + var depth = 3, // limit the search depth │ │ │ │ │ + button; │ │ │ │ │ + do { │ │ │ │ │ + if (OpenLayers.Element.hasClass(element, "olButton")) { │ │ │ │ │ + // hit! │ │ │ │ │ + button = element; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + element = element.parentNode; │ │ │ │ │ + } while (--depth > 0 && element); │ │ │ │ │ + return button; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: calculateBounds │ │ │ │ │ + * Method: ignore │ │ │ │ │ + * Check for event target elements that should be ignored by OpenLayers. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * mapBounds - {<OpenLayers.Bounds>} the current map extent, will be │ │ │ │ │ - * retrieved from the map object if not provided │ │ │ │ │ + * element - {DOMElement} The event target. │ │ │ │ │ */ │ │ │ │ │ - calculateBounds: function(mapBounds) { │ │ │ │ │ - if (!mapBounds) { │ │ │ │ │ - mapBounds = this.getMapBounds(); │ │ │ │ │ - } │ │ │ │ │ - var center = mapBounds.getCenterLonLat(); │ │ │ │ │ - var dataWidth = mapBounds.getWidth() * this.ratio; │ │ │ │ │ - var dataHeight = mapBounds.getHeight() * this.ratio; │ │ │ │ │ - this.bounds = new OpenLayers.Bounds( │ │ │ │ │ - center.lon - (dataWidth / 2), │ │ │ │ │ - center.lat - (dataHeight / 2), │ │ │ │ │ - center.lon + (dataWidth / 2), │ │ │ │ │ - center.lat + (dataHeight / 2) │ │ │ │ │ - ); │ │ │ │ │ + ignore: function(element) { │ │ │ │ │ + var depth = 3, │ │ │ │ │ + ignore = false; │ │ │ │ │ + do { │ │ │ │ │ + if (element.nodeName.toLowerCase() === 'a') { │ │ │ │ │ + ignore = true; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + element = element.parentNode; │ │ │ │ │ + } while (--depth > 0 && element); │ │ │ │ │ + return ignore; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: triggerRead │ │ │ │ │ + * Method: buttonClick │ │ │ │ │ + * Check if a button was clicked, and fire the buttonclick event │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Additional options for the protocol's read method │ │ │ │ │ - * (optional) │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} The protocol response object │ │ │ │ │ - * returned by the layer protocol. │ │ │ │ │ - */ │ │ │ │ │ - triggerRead: function(options) { │ │ │ │ │ - if (this.response && !(options && options.noAbort === true)) { │ │ │ │ │ - this.layer.protocol.abort(this.response); │ │ │ │ │ - this.layer.events.triggerEvent("loadend"); │ │ │ │ │ - } │ │ │ │ │ - var evt = { │ │ │ │ │ - filter: this.createFilter() │ │ │ │ │ - }; │ │ │ │ │ - this.layer.events.triggerEvent("loadstart", evt); │ │ │ │ │ - this.response = this.layer.protocol.read( │ │ │ │ │ - OpenLayers.Util.applyDefaults({ │ │ │ │ │ - filter: evt.filter, │ │ │ │ │ - callback: this.merge, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options)); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createFilter │ │ │ │ │ - * Creates a spatial BBOX filter. If the layer that this strategy belongs │ │ │ │ │ - * to has a filter property, this filter will be combined with the BBOX │ │ │ │ │ - * filter. │ │ │ │ │ - * │ │ │ │ │ - * Returns │ │ │ │ │ - * {<OpenLayers.Filter>} The filter object. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - createFilter: function() { │ │ │ │ │ - var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ - value: this.bounds, │ │ │ │ │ - projection: this.layer.projection │ │ │ │ │ - }); │ │ │ │ │ - if (this.layer.filter) { │ │ │ │ │ - filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ - type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ - filters: [this.layer.filter, filter] │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return filter; │ │ │ │ │ - }, │ │ │ │ │ + buttonClick: function(evt) { │ │ │ │ │ + var propagate = true, │ │ │ │ │ + element = OpenLayers.Event.element(evt); │ │ │ │ │ + if (element && (OpenLayers.Event.isLeftClick(evt) || !~evt.type.indexOf("mouse"))) { │ │ │ │ │ + // was a button pressed? │ │ │ │ │ + var button = this.getPressedButton(element); │ │ │ │ │ + if (button) { │ │ │ │ │ + if (evt.type === "keydown") { │ │ │ │ │ + switch (evt.keyCode) { │ │ │ │ │ + case OpenLayers.Event.KEY_RETURN: │ │ │ │ │ + case OpenLayers.Event.KEY_SPACE: │ │ │ │ │ + this.target.triggerEvent("buttonclick", { │ │ │ │ │ + buttonElement: button │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } else if (this.startEvt) { │ │ │ │ │ + if (this.completeRegEx.test(evt.type)) { │ │ │ │ │ + var pos = OpenLayers.Util.pagePosition(button); │ │ │ │ │ + var viewportElement = OpenLayers.Util.getViewportElement(); │ │ │ │ │ + var scrollTop = window.pageYOffset || viewportElement.scrollTop; │ │ │ │ │ + var scrollLeft = window.pageXOffset || viewportElement.scrollLeft; │ │ │ │ │ + pos[0] = pos[0] - scrollLeft; │ │ │ │ │ + pos[1] = pos[1] - scrollTop; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: merge │ │ │ │ │ - * Given a list of features, determine which ones to add to the layer. │ │ │ │ │ - * If the layer projection differs from the map projection, features │ │ │ │ │ - * will be transformed from the layer projection to the map projection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object passed │ │ │ │ │ - * by the protocol. │ │ │ │ │ - */ │ │ │ │ │ - merge: function(resp) { │ │ │ │ │ - this.layer.destroyFeatures(); │ │ │ │ │ - if (resp.success()) { │ │ │ │ │ - var features = resp.features; │ │ │ │ │ - if (features && features.length > 0) { │ │ │ │ │ - var remote = this.layer.projection; │ │ │ │ │ - var local = this.layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var geom; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - geom = features[i].geometry; │ │ │ │ │ - if (geom) { │ │ │ │ │ - geom.transform(remote, local); │ │ │ │ │ - } │ │ │ │ │ + this.target.triggerEvent("buttonclick", { │ │ │ │ │ + buttonElement: button, │ │ │ │ │ + buttonXY: { │ │ │ │ │ + x: this.startEvt.clientX - pos[0], │ │ │ │ │ + y: this.startEvt.clientY - pos[1] │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ + if (this.cancelRegEx.test(evt.type)) { │ │ │ │ │ + delete this.startEvt; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false; │ │ │ │ │ } │ │ │ │ │ - this.layer.addFeatures(features); │ │ │ │ │ + if (this.startRegEx.test(evt.type)) { │ │ │ │ │ + this.startEvt = evt; │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + propagate = !this.ignore(OpenLayers.Event.element(evt)); │ │ │ │ │ + delete this.startEvt; │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - this.bounds = null; │ │ │ │ │ } │ │ │ │ │ - this.response = null; │ │ │ │ │ - this.layer.events.triggerEvent("loadend", { │ │ │ │ │ - response: resp │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + return propagate; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.BBOX" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Panel.js │ │ │ │ │ + OpenLayers/Events/featureclick.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.Panel │ │ │ │ │ - * The Panel control is a container for other controls. With it toolbars │ │ │ │ │ - * may be composed. │ │ │ │ │ + * Class: OpenLayers.Events.featureclick │ │ │ │ │ * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ + * Extension event type for handling feature click events, including overlapping │ │ │ │ │ + * features. │ │ │ │ │ + * │ │ │ │ │ + * Event types provided by this extension: │ │ │ │ │ + * - featureclick │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.Panel = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - /** │ │ │ │ │ - * Property: controls │ │ │ │ │ - * {Array(<OpenLayers.Control>)} │ │ │ │ │ - */ │ │ │ │ │ - controls: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ - */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: defaultControl │ │ │ │ │ - * {<OpenLayers.Control>} The control which is activated when the control is │ │ │ │ │ - * activated (turned on), which also happens at instantiation. │ │ │ │ │ - * If <saveState> is true, <defaultControl> will be nullified after the │ │ │ │ │ - * first activation of the panel. │ │ │ │ │ - */ │ │ │ │ │ - defaultControl: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: saveState │ │ │ │ │ - * {Boolean} If set to true, the active state of this panel's controls will │ │ │ │ │ - * be stored on panel deactivation, and restored on reactivation. Default │ │ │ │ │ - * is false. │ │ │ │ │ - */ │ │ │ │ │ - saveState: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: allowDepress │ │ │ │ │ - * {Boolean} If is true the <OpenLayers.Control.TYPE_TOOL> controls can │ │ │ │ │ - * be deactivated by clicking the icon that represents them. Default │ │ │ │ │ - * is false. │ │ │ │ │ - */ │ │ │ │ │ - allowDepress: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: activeState │ │ │ │ │ - * {Object} stores the active state of this panel's controls. │ │ │ │ │ - */ │ │ │ │ │ - activeState: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Panel │ │ │ │ │ - * Create a new control panel. │ │ │ │ │ - * │ │ │ │ │ - * Each control in the panel is represented by an icon. When clicking │ │ │ │ │ - * on an icon, the <activateControl> method is called. │ │ │ │ │ - * │ │ │ │ │ - * Specific properties for controls on a panel: │ │ │ │ │ - * type - {Number} One of <OpenLayers.Control.TYPE_TOOL>, │ │ │ │ │ - * <OpenLayers.Control.TYPE_TOGGLE>, <OpenLayers.Control.TYPE_BUTTON>. │ │ │ │ │ - * If not provided, <OpenLayers.Control.TYPE_TOOL> is assumed. │ │ │ │ │ - * title - {string} Text displayed when mouse is over the icon that │ │ │ │ │ - * represents the control. │ │ │ │ │ - * │ │ │ │ │ - * The <OpenLayers.Control.type> of a control determines the behavior when │ │ │ │ │ - * clicking its icon: │ │ │ │ │ - * <OpenLayers.Control.TYPE_TOOL> - The control is activated and other │ │ │ │ │ - * controls of this type in the same panel are deactivated. This is │ │ │ │ │ - * the default type. │ │ │ │ │ - * <OpenLayers.Control.TYPE_TOGGLE> - The active state of the control is │ │ │ │ │ - * toggled. │ │ │ │ │ - * <OpenLayers.Control.TYPE_BUTTON> - The │ │ │ │ │ - * <OpenLayers.Control.Button.trigger> method of the control is called, │ │ │ │ │ - * but its active state is not changed. │ │ │ │ │ - * │ │ │ │ │ - * If a control is <OpenLayers.Control.active>, it will be drawn with the │ │ │ │ │ - * olControl[Name]ItemActive class, otherwise with the │ │ │ │ │ - * olControl[Name]ItemInactive class. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be used │ │ │ │ │ - * to extend the control. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.controls = []; │ │ │ │ │ - this.activeState = {}; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - for (var ctl, i = this.controls.length - 1; i >= 0; i--) { │ │ │ │ │ - ctl = this.controls[i]; │ │ │ │ │ - if (ctl.events) { │ │ │ │ │ - ctl.events.un({ │ │ │ │ │ - activate: this.iconOn, │ │ │ │ │ - deactivate: this.iconOff │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - ctl.panel_div = null; │ │ │ │ │ - } │ │ │ │ │ - this.activeState = null; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - control = this.controls[i]; │ │ │ │ │ - if (control === this.defaultControl || │ │ │ │ │ - (this.saveState && this.activeState[control.id])) { │ │ │ │ │ - control.activate(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.saveState === true) { │ │ │ │ │ - this.defaultControl = null; │ │ │ │ │ - } │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - control = this.controls[i]; │ │ │ │ │ - this.activeState[control.id] = control.deactivate(); │ │ │ │ │ - } │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +OpenLayers.Events.featureclick = OpenLayers.Class({ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Property: cache │ │ │ │ │ + * {Object} A cache of features under the mouse. │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (this.outsideViewport) { │ │ │ │ │ - this.events.attachToElement(this.div); │ │ │ │ │ - this.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ - } else { │ │ │ │ │ - this.map.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ - } │ │ │ │ │ - this.addControlsToMap(this.controls); │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ + cache: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: redraw │ │ │ │ │ + * Property: map │ │ │ │ │ + * {<OpenLayers.Map>} The map to register browser events on. │ │ │ │ │ */ │ │ │ │ │ - redraw: function() { │ │ │ │ │ - for (var l = this.div.childNodes.length, i = l - 1; i >= 0; i--) { │ │ │ │ │ - this.div.removeChild(this.div.childNodes[i]); │ │ │ │ │ - } │ │ │ │ │ - this.div.innerHTML = ""; │ │ │ │ │ - if (this.active) { │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - this.div.appendChild(this.controls[i].panel_div); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + map: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: activateControl │ │ │ │ │ - * This method is called when the user click on the icon representing a │ │ │ │ │ - * control in the panel. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} │ │ │ │ │ + * Property: provides │ │ │ │ │ + * {Array(String)} The event types provided by this extension. │ │ │ │ │ */ │ │ │ │ │ - activateControl: function(control) { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - if (control.type == OpenLayers.Control.TYPE_BUTTON) { │ │ │ │ │ - control.trigger(); │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (control.type == OpenLayers.Control.TYPE_TOGGLE) { │ │ │ │ │ - if (control.active) { │ │ │ │ │ - control.deactivate(); │ │ │ │ │ - } else { │ │ │ │ │ - control.activate(); │ │ │ │ │ - } │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (this.allowDepress && control.active) { │ │ │ │ │ - control.deactivate(); │ │ │ │ │ - } else { │ │ │ │ │ - var c; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - c = this.controls[i]; │ │ │ │ │ - if (c != control && │ │ │ │ │ - (c.type === OpenLayers.Control.TYPE_TOOL || c.type == null)) { │ │ │ │ │ - c.deactivate(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - control.activate(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + provides: ["featureclick", "nofeatureclick", "featureover", "featureout"], │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: addControls │ │ │ │ │ - * To build a toolbar, you add a set of controls to it. addControls │ │ │ │ │ - * lets you add a single control or a list of controls to the │ │ │ │ │ - * Control Panel. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * controls - {<OpenLayers.Control>} Controls to add in the panel. │ │ │ │ │ - */ │ │ │ │ │ - addControls: function(controls) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(controls))) { │ │ │ │ │ - controls = [controls]; │ │ │ │ │ - } │ │ │ │ │ - this.controls = this.controls.concat(controls); │ │ │ │ │ - │ │ │ │ │ - for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ - var control = controls[i], │ │ │ │ │ - element = this.createControlMarkup(control); │ │ │ │ │ - OpenLayers.Element.addClass(element, │ │ │ │ │ - control.displayClass + "ItemInactive"); │ │ │ │ │ - OpenLayers.Element.addClass(element, "olButton"); │ │ │ │ │ - if (control.title != "" && !element.title) { │ │ │ │ │ - element.title = control.title; │ │ │ │ │ + * Constructor: OpenLayers.Events.featureclick │ │ │ │ │ + * Create a new featureclick event type. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * target - {<OpenLayers.Events>} The events instance to create the events │ │ │ │ │ + * for. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(target) { │ │ │ │ │ + this.target = target; │ │ │ │ │ + if (target.object instanceof OpenLayers.Map) { │ │ │ │ │ + this.setMap(target.object); │ │ │ │ │ + } else if (target.object instanceof OpenLayers.Layer.Vector) { │ │ │ │ │ + if (target.object.map) { │ │ │ │ │ + this.setMap(target.object.map); │ │ │ │ │ + } else { │ │ │ │ │ + target.object.events.register("added", this, function(evt) { │ │ │ │ │ + this.setMap(target.object.map); │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - control.panel_div = element; │ │ │ │ │ + } else { │ │ │ │ │ + throw ("Listeners for '" + this.provides.join("', '") + │ │ │ │ │ + "' events can only be registered for OpenLayers.Layer.Vector " + │ │ │ │ │ + "or OpenLayers.Map instances"); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - if (this.map) { // map.addControl() has already been called on the panel │ │ │ │ │ - this.addControlsToMap(controls); │ │ │ │ │ - this.redraw(); │ │ │ │ │ + for (var i = this.provides.length - 1; i >= 0; --i) { │ │ │ │ │ + target.extensions[this.provides[i]] = true; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: createControlMarkup │ │ │ │ │ - * This function just creates a div for the control. If specific HTML │ │ │ │ │ - * markup is needed this function can be overridden in specific classes, │ │ │ │ │ - * or at panel instantiation time: │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var panel = new OpenLayers.Control.Panel({ │ │ │ │ │ - * defaultControl: control, │ │ │ │ │ - * // ovverride createControlMarkup to create actual buttons │ │ │ │ │ - * // including texts wrapped into span elements. │ │ │ │ │ - * createControlMarkup: function(control) { │ │ │ │ │ - * var button = document.createElement('button'), │ │ │ │ │ - * span = document.createElement('span'); │ │ │ │ │ - * if (control.text) { │ │ │ │ │ - * span.innerHTML = control.text; │ │ │ │ │ - * } │ │ │ │ │ - * return button; │ │ │ │ │ - * } │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ + * Method: setMap │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * control - {<OpenLayers.Control>} The control to create the HTML │ │ │ │ │ - * markup for. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The markup. │ │ │ │ │ + * map - {<OpenLayers.Map>} The map to register browser events on. │ │ │ │ │ */ │ │ │ │ │ - createControlMarkup: function(control) { │ │ │ │ │ - return document.createElement("div"); │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.map = map; │ │ │ │ │ + this.cache = {}; │ │ │ │ │ + map.events.register("mousedown", this, this.start, { │ │ │ │ │ + extension: true │ │ │ │ │ + }); │ │ │ │ │ + map.events.register("mouseup", this, this.onClick, { │ │ │ │ │ + extension: true │ │ │ │ │ + }); │ │ │ │ │ + map.events.register("touchstart", this, this.start, { │ │ │ │ │ + extension: true │ │ │ │ │ + }); │ │ │ │ │ + map.events.register("touchmove", this, this.cancel, { │ │ │ │ │ + extension: true │ │ │ │ │ + }); │ │ │ │ │ + map.events.register("touchend", this, this.onClick, { │ │ │ │ │ + extension: true │ │ │ │ │ + }); │ │ │ │ │ + map.events.register("mousemove", this, this.onMousemove, { │ │ │ │ │ + extension: true │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: addControlsToMap │ │ │ │ │ - * Only for internal use in draw() and addControls() methods. │ │ │ │ │ + * Method: start │ │ │ │ │ + * Sets startEvt = evt. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * controls - {Array(<OpenLayers.Control>)} Controls to add into map. │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - addControlsToMap: function(controls) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ - control = controls[i]; │ │ │ │ │ - if (control.autoActivate === true) { │ │ │ │ │ - control.autoActivate = false; │ │ │ │ │ - this.map.addControl(control); │ │ │ │ │ - control.autoActivate = true; │ │ │ │ │ - } else { │ │ │ │ │ - this.map.addControl(control); │ │ │ │ │ - control.deactivate(); │ │ │ │ │ - } │ │ │ │ │ - control.events.on({ │ │ │ │ │ - activate: this.iconOn, │ │ │ │ │ - deactivate: this.iconOff │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ + start: function(evt) { │ │ │ │ │ + this.startEvt = evt; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: iconOn │ │ │ │ │ - * Internal use, for use only with "controls[i].events.on/un". │ │ │ │ │ + * Method: cancel │ │ │ │ │ + * Deletes the start event. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - iconOn: function() { │ │ │ │ │ - var d = this.panel_div; // "this" refers to a control on panel! │ │ │ │ │ - var re = new RegExp("\\b(" + this.displayClass + "Item)Inactive\\b"); │ │ │ │ │ - d.className = d.className.replace(re, "$1Active"); │ │ │ │ │ + cancel: function(evt) { │ │ │ │ │ + delete this.startEvt; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: iconOff │ │ │ │ │ - * Internal use, for use only with "controls[i].events.on/un". │ │ │ │ │ + * Method: onClick │ │ │ │ │ + * Listener for the click event. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - iconOff: function() { │ │ │ │ │ - var d = this.panel_div; // "this" refers to a control on panel! │ │ │ │ │ - var re = new RegExp("\\b(" + this.displayClass + "Item)Active\\b"); │ │ │ │ │ - d.className = d.className.replace(re, "$1Inactive"); │ │ │ │ │ + onClick: function(evt) { │ │ │ │ │ + if (!this.startEvt || evt.type !== "touchend" && │ │ │ │ │ + !OpenLayers.Event.isLeftClick(evt)) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var features = this.getFeatures(this.startEvt); │ │ │ │ │ + delete this.startEvt; │ │ │ │ │ + // fire featureclick events │ │ │ │ │ + var feature, layer, more, clicked = {}; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + layer = feature.layer; │ │ │ │ │ + clicked[layer.id] = true; │ │ │ │ │ + more = this.triggerEvent("featureclick", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + if (more === false) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // fire nofeatureclick events on all vector layers with no targets │ │ │ │ │ + for (i = 0, len = this.map.layers.length; i < len; ++i) { │ │ │ │ │ + layer = this.map.layers[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Vector && !clicked[layer.id]) { │ │ │ │ │ + this.triggerEvent("nofeatureclick", { │ │ │ │ │ + layer: layer │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onButtonClick │ │ │ │ │ + * Method: onMousemove │ │ │ │ │ + * Listener for the mousemove event. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - var controls = this.controls, │ │ │ │ │ - button = evt.buttonElement; │ │ │ │ │ - for (var i = controls.length - 1; i >= 0; --i) { │ │ │ │ │ - if (controls[i].panel_div === button) { │ │ │ │ │ - this.activateControl(controls[i]); │ │ │ │ │ + onMousemove: function(evt) { │ │ │ │ │ + delete this.startEvt; │ │ │ │ │ + var features = this.getFeatures(evt); │ │ │ │ │ + var over = {}, │ │ │ │ │ + newly = [], │ │ │ │ │ + feature; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + over[feature.id] = feature; │ │ │ │ │ + if (!this.cache[feature.id]) { │ │ │ │ │ + newly.push(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // check if already over features │ │ │ │ │ + var out = []; │ │ │ │ │ + for (var id in this.cache) { │ │ │ │ │ + feature = this.cache[id]; │ │ │ │ │ + if (feature.layer && feature.layer.map) { │ │ │ │ │ + if (!over[feature.id]) { │ │ │ │ │ + out.push(feature); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // removed │ │ │ │ │ + delete this.cache[id]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // fire featureover events │ │ │ │ │ + var more; │ │ │ │ │ + for (i = 0, len = newly.length; i < len; ++i) { │ │ │ │ │ + feature = newly[i]; │ │ │ │ │ + this.cache[feature.id] = feature; │ │ │ │ │ + more = this.triggerEvent("featureover", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + if (more === false) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // fire featureout events │ │ │ │ │ + for (i = 0, len = out.length; i < len; ++i) { │ │ │ │ │ + feature = out[i]; │ │ │ │ │ + delete this.cache[feature.id]; │ │ │ │ │ + more = this.triggerEvent("featureout", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + if (more === false) { │ │ │ │ │ break; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getControlsBy │ │ │ │ │ - * Get a list of controls with properties matching the given criteria. │ │ │ │ │ + * Method: triggerEvent │ │ │ │ │ + * Determines where to trigger the event and triggers it. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * property - {String} A control property to be matched. │ │ │ │ │ - * match - {String | Object} A string to match. Can also be a regular │ │ │ │ │ - * expression literal or object. In addition, it can be any object │ │ │ │ │ - * with a method named test. For reqular expressions or other, if │ │ │ │ │ - * match.test(control[property]) evaluates to true, the control will be │ │ │ │ │ - * included in the array returned. If no controls are found, an empty │ │ │ │ │ - * array is returned. │ │ │ │ │ + * type - {String} The event type to trigger │ │ │ │ │ + * evt - {Object} The listener argument │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Control>)} A list of controls matching the given criteria. │ │ │ │ │ - * An empty array is returned if no matches are found. │ │ │ │ │ + * {Boolean} The last listener return. │ │ │ │ │ */ │ │ │ │ │ - getControlsBy: function(property, match) { │ │ │ │ │ - var test = (typeof match.test == "function"); │ │ │ │ │ - var found = OpenLayers.Array.filter(this.controls, function(item) { │ │ │ │ │ - return item[property] == match || (test && match.test(item[property])); │ │ │ │ │ - }); │ │ │ │ │ - return found; │ │ │ │ │ + triggerEvent: function(type, evt) { │ │ │ │ │ + var layer = evt.feature ? evt.feature.layer : evt.layer, │ │ │ │ │ + object = this.target.object; │ │ │ │ │ + if (object instanceof OpenLayers.Map || object === layer) { │ │ │ │ │ + return this.target.triggerEvent(type, evt); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getControlsByName │ │ │ │ │ - * Get a list of contorls with names matching the given name. │ │ │ │ │ + * Method: getFeatures │ │ │ │ │ + * Get all features at the given screen location. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * match - {String | Object} A control name. The name can also be a regular │ │ │ │ │ - * expression literal or object. In addition, it can be any object │ │ │ │ │ - * with a method named test. For reqular expressions or other, if │ │ │ │ │ - * name.test(control.name) evaluates to true, the control will be included │ │ │ │ │ - * in the list of controls returned. If no controls are found, an empty │ │ │ │ │ - * array is returned. │ │ │ │ │ + * evt - {Object} Event object. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Control>)} A list of controls matching the given name. │ │ │ │ │ - * An empty array is returned if no matches are found. │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} List of features at the given point. │ │ │ │ │ */ │ │ │ │ │ - getControlsByName: function(match) { │ │ │ │ │ - return this.getControlsBy("name", match); │ │ │ │ │ + getFeatures: function(evt) { │ │ │ │ │ + var x = evt.clientX, │ │ │ │ │ + y = evt.clientY, │ │ │ │ │ + features = [], │ │ │ │ │ + targets = [], │ │ │ │ │ + layers = [], │ │ │ │ │ + layer, target, feature, i, len; │ │ │ │ │ + // go through all layers looking for targets │ │ │ │ │ + for (i = this.map.layers.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = this.map.layers[i]; │ │ │ │ │ + if (layer.div.style.display !== "none") { │ │ │ │ │ + if (layer.renderer instanceof OpenLayers.Renderer.Elements) { │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Vector) { │ │ │ │ │ + target = document.elementFromPoint(x, y); │ │ │ │ │ + while (target && target._featureId) { │ │ │ │ │ + feature = layer.getFeatureById(target._featureId); │ │ │ │ │ + if (feature) { │ │ │ │ │ + features.push(feature); │ │ │ │ │ + target.style.display = "none"; │ │ │ │ │ + targets.push(target); │ │ │ │ │ + target = document.elementFromPoint(x, y); │ │ │ │ │ + } else { │ │ │ │ │ + // sketch, all bets off │ │ │ │ │ + target = false; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + layers.push(layer); │ │ │ │ │ + layer.div.style.display = "none"; │ │ │ │ │ + } else if (layer.renderer instanceof OpenLayers.Renderer.Canvas) { │ │ │ │ │ + feature = layer.renderer.getFeatureIdFromEvent(evt); │ │ │ │ │ + if (feature) { │ │ │ │ │ + features.push(feature); │ │ │ │ │ + layers.push(layer); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // restore feature visibility │ │ │ │ │ + for (i = 0, len = targets.length; i < len; ++i) { │ │ │ │ │ + targets[i].style.display = ""; │ │ │ │ │ + } │ │ │ │ │ + // restore layer visibility │ │ │ │ │ + for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + layers[i].div.style.display = "block"; │ │ │ │ │ + } │ │ │ │ │ + return features; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: getControlsByClass │ │ │ │ │ - * Get a list of controls of a given type (CLASS_NAME). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * match - {String | Object} A control class name. The type can also be a │ │ │ │ │ - * regular expression literal or object. In addition, it can be any │ │ │ │ │ - * object with a method named test. For reqular expressions or other, │ │ │ │ │ - * if type.test(control.CLASS_NAME) evaluates to true, the control will │ │ │ │ │ - * be included in the list of controls returned. If no controls are │ │ │ │ │ - * found, an empty array is returned. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Control>)} A list of controls matching the given type. │ │ │ │ │ - * An empty array is returned if no matches are found. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up. │ │ │ │ │ */ │ │ │ │ │ - getControlsByClass: function(match) { │ │ │ │ │ - return this.getControlsBy("CLASS_NAME", match); │ │ │ │ │ - }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i = this.provides.length - 1; i >= 0; --i) { │ │ │ │ │ + delete this.target.extensions[this.provides[i]]; │ │ │ │ │ + } │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + mousemove: this.onMousemove, │ │ │ │ │ + mousedown: this.start, │ │ │ │ │ + mouseup: this.onClick, │ │ │ │ │ + touchstart: this.start, │ │ │ │ │ + touchmove: this.cancel, │ │ │ │ │ + touchend: this.onClick, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + delete this.cache; │ │ │ │ │ + delete this.map; │ │ │ │ │ + delete this.target; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Panel" │ │ │ │ │ }); │ │ │ │ │ │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Events.nofeatureclick │ │ │ │ │ + * │ │ │ │ │ + * Extension event type for handling click events that do not hit a feature. │ │ │ │ │ + * │ │ │ │ │ + * Event types provided by this extension: │ │ │ │ │ + * - nofeatureclick │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Events.nofeatureclick = OpenLayers.Events.featureclick; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Events.featureover │ │ │ │ │ + * │ │ │ │ │ + * Extension event type for handling hovering over a feature. │ │ │ │ │ + * │ │ │ │ │ + * Event types provided by this extension: │ │ │ │ │ + * - featureover │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Events.featureover = OpenLayers.Events.featureclick; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Events.featureout │ │ │ │ │ + * │ │ │ │ │ + * Extension event type for handling leaving a feature. │ │ │ │ │ + * │ │ │ │ │ + * Event types provided by this extension: │ │ │ │ │ + * - featureout │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Events.featureout = OpenLayers.Events.featureclick; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Control/ZoomBox.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ @@ -74547,130 +72358,360 @@ │ │ │ │ │ this.panned = false; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.DragPan" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Navigation.js │ │ │ │ │ + OpenLayers/Control/Scale.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Control/ZoomBox.js │ │ │ │ │ - * @requires OpenLayers/Control/DragPan.js │ │ │ │ │ - * @requires OpenLayers/Handler/MouseWheel.js │ │ │ │ │ - * @requires OpenLayers/Handler/Click.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.Navigation │ │ │ │ │ - * The navigation control handles map browsing with mouse events (dragging, │ │ │ │ │ - * double-clicking, and scrolling the wheel). Create a new navigation │ │ │ │ │ - * control with the <OpenLayers.Control.Navigation> control. │ │ │ │ │ - * │ │ │ │ │ - * Note that this control is added to the map by default (if no controls │ │ │ │ │ - * array is sent in the options object to the <OpenLayers.Map> │ │ │ │ │ - * constructor). │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Control.Scale │ │ │ │ │ + * The Scale control displays the current map scale as a ratio (e.g. Scale = │ │ │ │ │ + * 1:1M). By default it is displayed in the lower right corner of the map. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.Scale = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: element │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + element: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: geodesic │ │ │ │ │ + * {Boolean} Use geodesic measurement. Default is false. The recommended │ │ │ │ │ + * setting for maps in EPSG:4326 is false, and true EPSG:900913. If set to │ │ │ │ │ + * true, the scale will be calculated based on the horizontal size of the │ │ │ │ │ + * pixel in the center of the map viewport. │ │ │ │ │ + */ │ │ │ │ │ + geodesic: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.Scale │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * element - {DOMElement} │ │ │ │ │ + * options - {Object} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(element, options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.element = OpenLayers.Util.getElement(element); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (!this.element) { │ │ │ │ │ + this.element = document.createElement("div"); │ │ │ │ │ + this.div.appendChild(this.element); │ │ │ │ │ + } │ │ │ │ │ + this.map.events.register('moveend', this, this.updateScale); │ │ │ │ │ + this.updateScale(); │ │ │ │ │ + return this.div; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateScale │ │ │ │ │ + */ │ │ │ │ │ + updateScale: function() { │ │ │ │ │ + var scale; │ │ │ │ │ + if (this.geodesic === true) { │ │ │ │ │ + var units = this.map.getUnits(); │ │ │ │ │ + if (!units) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var inches = OpenLayers.INCHES_PER_UNIT; │ │ │ │ │ + scale = (this.map.getGeodesicPixelSize().w || 0.000001) * │ │ │ │ │ + inches["km"] * OpenLayers.DOTS_PER_INCH; │ │ │ │ │ + } else { │ │ │ │ │ + scale = this.map.getScale(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (!scale) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (scale >= 9500 && scale <= 950000) { │ │ │ │ │ + scale = Math.round(scale / 1000) + "K"; │ │ │ │ │ + } else if (scale >= 950000) { │ │ │ │ │ + scale = Math.round(scale / 1000000) + "M"; │ │ │ │ │ + } else { │ │ │ │ │ + scale = Math.round(scale); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.element.innerHTML = OpenLayers.i18n("Scale = 1 : ${scaleDenom}", { │ │ │ │ │ + 'scaleDenom': scale │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Scale" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/PinchZoom.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Handler/Pinch.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.PinchZoom │ │ │ │ │ + * │ │ │ │ │ * Inherits: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.Navigation = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Control.PinchZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: dragPan │ │ │ │ │ - * {<OpenLayers.Control.DragPan>} │ │ │ │ │ + * Property: type │ │ │ │ │ + * {OpenLayers.Control.TYPES} │ │ │ │ │ */ │ │ │ │ │ - dragPan: null, │ │ │ │ │ + type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: dragPanOptions │ │ │ │ │ - * {Object} Options passed to the DragPan control. │ │ │ │ │ + * Property: pinchOrigin │ │ │ │ │ + * {Object} Cached object representing the pinch start (in pixels). │ │ │ │ │ */ │ │ │ │ │ - dragPanOptions: null, │ │ │ │ │ + pinchOrigin: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: pinchZoom │ │ │ │ │ - * {<OpenLayers.Control.PinchZoom>} │ │ │ │ │ + * Property: currentCenter │ │ │ │ │ + * {Object} Cached object representing the latest pinch center (in pixels). │ │ │ │ │ */ │ │ │ │ │ - pinchZoom: null, │ │ │ │ │ + currentCenter: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: pinchZoomOptions │ │ │ │ │ - * {Object} Options passed to the PinchZoom control. │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ - pinchZoomOptions: null, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: documentDrag │ │ │ │ │ - * {Boolean} Allow panning of the map by dragging outside map viewport. │ │ │ │ │ - * Default is false. │ │ │ │ │ + * APIProperty: preserveCenter │ │ │ │ │ + * {Boolean} Set this to true if you don't want the map center to change │ │ │ │ │ + * while pinching. For example you may want to set preserveCenter to │ │ │ │ │ + * true when the user location is being watched and you want to preserve │ │ │ │ │ + * the user location at the center of the map even if he zooms in or │ │ │ │ │ + * out using pinch. This property's value can be changed any time on an │ │ │ │ │ + * existing instance. Default is false. │ │ │ │ │ */ │ │ │ │ │ - documentDrag: false, │ │ │ │ │ + preserveCenter: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: zoomBox │ │ │ │ │ - * {<OpenLayers.Control.ZoomBox>} │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: handlerOptions │ │ │ │ │ + * {Object} Used to set non-default properties on the pinch handler │ │ │ │ │ */ │ │ │ │ │ - zoomBox: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomBoxEnabled │ │ │ │ │ - * {Boolean} Whether the user can draw a box to zoom │ │ │ │ │ + * Constructor: OpenLayers.Control.PinchZoom │ │ │ │ │ + * Create a control for zooming with pinch gestures. This works on devices │ │ │ │ │ + * with multi-touch support. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} An optional object whose properties will be set on │ │ │ │ │ + * the control │ │ │ │ │ */ │ │ │ │ │ - zoomBoxEnabled: true, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.handler = new OpenLayers.Handler.Pinch(this, { │ │ │ │ │ + start: this.pinchStart, │ │ │ │ │ + move: this.pinchMove, │ │ │ │ │ + done: this.pinchDone │ │ │ │ │ + }, this.handlerOptions); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomWheelEnabled │ │ │ │ │ - * {Boolean} Whether the mousewheel should zoom the map │ │ │ │ │ + * Method: pinchStart │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * pinchData - {Object} pinch data object related to the current touchmove │ │ │ │ │ + * of the pinch gesture. This give us the current scale of the pinch. │ │ │ │ │ */ │ │ │ │ │ - zoomWheelEnabled: true, │ │ │ │ │ + pinchStart: function(evt, pinchData) { │ │ │ │ │ + var xy = (this.preserveCenter) ? │ │ │ │ │ + this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ + this.pinchOrigin = xy; │ │ │ │ │ + this.currentCenter = xy; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: mouseWheelOptions │ │ │ │ │ - * {Object} Options passed to the MouseWheel control (only useful if │ │ │ │ │ - * <zoomWheelEnabled> is set to true). Default is no options for maps │ │ │ │ │ - * with fractionalZoom set to true, otherwise │ │ │ │ │ - * {cumulative: false, interval: 50, maxDelta: 6} │ │ │ │ │ + * Method: pinchMove │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * pinchData - {Object} pinch data object related to the current touchmove │ │ │ │ │ + * of the pinch gesture. This give us the current scale of the pinch. │ │ │ │ │ */ │ │ │ │ │ - mouseWheelOptions: null, │ │ │ │ │ + pinchMove: function(evt, pinchData) { │ │ │ │ │ + var scale = pinchData.scale; │ │ │ │ │ + var containerOrigin = this.map.layerContainerOriginPx; │ │ │ │ │ + var pinchOrigin = this.pinchOrigin; │ │ │ │ │ + var current = (this.preserveCenter) ? │ │ │ │ │ + this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ + │ │ │ │ │ + var dx = Math.round((containerOrigin.x + current.x - pinchOrigin.x) + (scale - 1) * (containerOrigin.x - pinchOrigin.x)); │ │ │ │ │ + var dy = Math.round((containerOrigin.y + current.y - pinchOrigin.y) + (scale - 1) * (containerOrigin.y - pinchOrigin.y)); │ │ │ │ │ + │ │ │ │ │ + this.map.applyTransform(dx, dy, scale); │ │ │ │ │ + this.currentCenter = current; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: handleRightClicks │ │ │ │ │ - * {Boolean} Whether or not to handle right clicks. Default is false. │ │ │ │ │ + * Method: pinchDone │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * start - {Object} pinch data object related to the touchstart event that │ │ │ │ │ + * started the pinch gesture. │ │ │ │ │ + * last - {Object} pinch data object related to the last touchmove event │ │ │ │ │ + * of the pinch gesture. This give us the final scale of the pinch. │ │ │ │ │ */ │ │ │ │ │ - handleRightClicks: false, │ │ │ │ │ + pinchDone: function(evt, start, last) { │ │ │ │ │ + this.map.applyTransform(); │ │ │ │ │ + var zoom = this.map.getZoomForResolution(this.map.getResolution() / last.scale, true); │ │ │ │ │ + if (zoom !== this.map.getZoom() || !this.currentCenter.equals(this.pinchOrigin)) { │ │ │ │ │ + var resolution = this.map.getResolutionForZoom(zoom); │ │ │ │ │ + │ │ │ │ │ + var location = this.map.getLonLatFromPixel(this.pinchOrigin); │ │ │ │ │ + var zoomPixel = this.currentCenter; │ │ │ │ │ + var size = this.map.getSize(); │ │ │ │ │ + │ │ │ │ │ + location.lon += resolution * ((size.w / 2) - zoomPixel.x); │ │ │ │ │ + location.lat -= resolution * ((size.h / 2) - zoomPixel.y); │ │ │ │ │ + │ │ │ │ │ + // Force a reflow before calling setCenter. This is to work │ │ │ │ │ + // around an issue occuring in iOS. │ │ │ │ │ + // │ │ │ │ │ + // See https://github.com/openlayers/openlayers/pull/351. │ │ │ │ │ + // │ │ │ │ │ + // Without a reflow setting the layer container div's top left │ │ │ │ │ + // style properties to "0px" - as done in Map.moveTo when zoom │ │ │ │ │ + // is changed - won't actually correctly reposition the layer │ │ │ │ │ + // container div. │ │ │ │ │ + // │ │ │ │ │ + // Also, we need to use a statement that the Google Closure │ │ │ │ │ + // compiler won't optimize away. │ │ │ │ │ + this.map.div.clientWidth = this.map.div.clientWidth; │ │ │ │ │ + │ │ │ │ │ + this.map.setCenter(location, zoom); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.PinchZoom" │ │ │ │ │ + │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/TouchNavigation.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control/DragPan.js │ │ │ │ │ + * @requires OpenLayers/Control/PinchZoom.js │ │ │ │ │ + * @requires OpenLayers/Handler/Click.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.TouchNavigation │ │ │ │ │ + * The navigation control handles map browsing with touch events (dragging, │ │ │ │ │ + * double-tapping, tap with two fingers, and pinch zoom). Create a new │ │ │ │ │ + * control with the <OpenLayers.Control.TouchNavigation> constructor. │ │ │ │ │ + * │ │ │ │ │ + * If you’re only targeting touch enabled devices with your mapping application, │ │ │ │ │ + * you can create a map with only a TouchNavigation control. The │ │ │ │ │ + * <OpenLayers.Control.Navigation> control is mobile ready by default, but │ │ │ │ │ + * you can generate a smaller build of the library by only including this │ │ │ │ │ + * touch navigation control if you aren't concerned about mouse interaction. │ │ │ │ │ + * │ │ │ │ │ + * Inherits: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.TouchNavigation = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomBoxKeyMask │ │ │ │ │ - * {Integer} <OpenLayers.Handler> key code of the key, which has to be │ │ │ │ │ - * pressed, while drawing the zoom box with the mouse on the screen. │ │ │ │ │ - * You should probably set handleRightClicks to true if you use this │ │ │ │ │ - * with MOD_CTRL, to disable the context menu for machines which use │ │ │ │ │ - * CTRL-Click as a right click. │ │ │ │ │ - * Default: <OpenLayers.Handler.MOD_SHIFT> │ │ │ │ │ + * Property: dragPan │ │ │ │ │ + * {<OpenLayers.Control.DragPan>} │ │ │ │ │ */ │ │ │ │ │ - zoomBoxKeyMask: OpenLayers.Handler.MOD_SHIFT, │ │ │ │ │ + dragPan: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: dragPanOptions │ │ │ │ │ + * {Object} Options passed to the DragPan control. │ │ │ │ │ + */ │ │ │ │ │ + dragPanOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: pinchZoom │ │ │ │ │ + * {<OpenLayers.Control.PinchZoom>} │ │ │ │ │ + */ │ │ │ │ │ + pinchZoom: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: pinchZoomOptions │ │ │ │ │ + * {Object} Options passed to the PinchZoom control. │ │ │ │ │ + */ │ │ │ │ │ + pinchZoomOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: clickHandlerOptions │ │ │ │ │ + * {Object} Options passed to the Click handler. │ │ │ │ │ + */ │ │ │ │ │ + clickHandlerOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: documentDrag │ │ │ │ │ + * {Boolean} Allow panning of the map by dragging outside map viewport. │ │ │ │ │ + * Default is false. │ │ │ │ │ + */ │ │ │ │ │ + documentDrag: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIProperty: autoActivate │ │ │ │ │ * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ * true. │ │ │ │ │ */ │ │ │ │ │ autoActivate: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Navigation │ │ │ │ │ + * Constructor: OpenLayers.Control.TouchNavigation │ │ │ │ │ * Create a new navigation control │ │ │ │ │ - * │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ * options - {Object} An optional object whose properties will be set on │ │ │ │ │ * the control │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ this.handlers = {}; │ │ │ │ │ OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ @@ -74680,116 +72721,79 @@ │ │ │ │ │ * Method: destroy │ │ │ │ │ * The destroy method is used to perform any clean up before the control │ │ │ │ │ * is dereferenced. Typically this is where event listeners are removed │ │ │ │ │ * to prevent memory leaks. │ │ │ │ │ */ │ │ │ │ │ destroy: function() { │ │ │ │ │ this.deactivate(); │ │ │ │ │ - │ │ │ │ │ if (this.dragPan) { │ │ │ │ │ this.dragPan.destroy(); │ │ │ │ │ } │ │ │ │ │ this.dragPan = null; │ │ │ │ │ - │ │ │ │ │ - if (this.zoomBox) { │ │ │ │ │ - this.zoomBox.destroy(); │ │ │ │ │ - } │ │ │ │ │ - this.zoomBox = null; │ │ │ │ │ - │ │ │ │ │ if (this.pinchZoom) { │ │ │ │ │ this.pinchZoom.destroy(); │ │ │ │ │ + delete this.pinchZoom; │ │ │ │ │ } │ │ │ │ │ - this.pinchZoom = null; │ │ │ │ │ - │ │ │ │ │ OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Method: activate │ │ │ │ │ */ │ │ │ │ │ activate: function() { │ │ │ │ │ - this.dragPan.activate(); │ │ │ │ │ - if (this.zoomWheelEnabled) { │ │ │ │ │ - this.handlers.wheel.activate(); │ │ │ │ │ - } │ │ │ │ │ - this.handlers.click.activate(); │ │ │ │ │ - if (this.zoomBoxEnabled) { │ │ │ │ │ - this.zoomBox.activate(); │ │ │ │ │ - } │ │ │ │ │ - if (this.pinchZoom) { │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.dragPan.activate(); │ │ │ │ │ + this.handlers.click.activate(); │ │ │ │ │ this.pinchZoom.activate(); │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Control.prototype.activate.apply(this, arguments); │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Method: deactivate │ │ │ │ │ */ │ │ │ │ │ deactivate: function() { │ │ │ │ │ - if (this.pinchZoom) { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.dragPan.deactivate(); │ │ │ │ │ + this.handlers.click.deactivate(); │ │ │ │ │ this.pinchZoom.deactivate(); │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ - this.zoomBox.deactivate(); │ │ │ │ │ - this.dragPan.deactivate(); │ │ │ │ │ - this.handlers.click.deactivate(); │ │ │ │ │ - this.handlers.wheel.deactivate(); │ │ │ │ │ - return OpenLayers.Control.prototype.deactivate.apply(this, arguments); │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Method: draw │ │ │ │ │ */ │ │ │ │ │ draw: function() { │ │ │ │ │ - // disable right mouse context menu for support of right click events │ │ │ │ │ - if (this.handleRightClicks) { │ │ │ │ │ - this.map.viewPortDiv.oncontextmenu = OpenLayers.Function.False; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ var clickCallbacks = { │ │ │ │ │ - 'click': this.defaultClick, │ │ │ │ │ - 'dblclick': this.defaultDblClick, │ │ │ │ │ - 'dblrightclick': this.defaultDblRightClick │ │ │ │ │ - }; │ │ │ │ │ - var clickOptions = { │ │ │ │ │ - 'double': true, │ │ │ │ │ - 'stopDouble': true │ │ │ │ │ + click: this.defaultClick, │ │ │ │ │ + dblclick: this.defaultDblClick │ │ │ │ │ }; │ │ │ │ │ + var clickOptions = OpenLayers.Util.extend({ │ │ │ │ │ + "double": true, │ │ │ │ │ + stopDouble: true, │ │ │ │ │ + pixelTolerance: 2 │ │ │ │ │ + }, this.clickHandlerOptions); │ │ │ │ │ this.handlers.click = new OpenLayers.Handler.Click( │ │ │ │ │ this, clickCallbacks, clickOptions │ │ │ │ │ ); │ │ │ │ │ this.dragPan = new OpenLayers.Control.DragPan( │ │ │ │ │ OpenLayers.Util.extend({ │ │ │ │ │ map: this.map, │ │ │ │ │ documentDrag: this.documentDrag │ │ │ │ │ }, this.dragPanOptions) │ │ │ │ │ ); │ │ │ │ │ - this.zoomBox = new OpenLayers.Control.ZoomBox({ │ │ │ │ │ - map: this.map, │ │ │ │ │ - keyMask: this.zoomBoxKeyMask │ │ │ │ │ - }); │ │ │ │ │ this.dragPan.draw(); │ │ │ │ │ - this.zoomBox.draw(); │ │ │ │ │ - var wheelOptions = this.map.fractionalZoom ? {} : { │ │ │ │ │ - cumulative: false, │ │ │ │ │ - interval: 50, │ │ │ │ │ - maxDelta: 6 │ │ │ │ │ - }; │ │ │ │ │ - this.handlers.wheel = new OpenLayers.Handler.MouseWheel( │ │ │ │ │ - this, { │ │ │ │ │ - up: this.wheelUp, │ │ │ │ │ - down: this.wheelDown │ │ │ │ │ - }, │ │ │ │ │ - OpenLayers.Util.extend(wheelOptions, this.mouseWheelOptions) │ │ │ │ │ + this.pinchZoom = new OpenLayers.Control.PinchZoom( │ │ │ │ │ + OpenLayers.Util.extend({ │ │ │ │ │ + map: this.map │ │ │ │ │ + }, this.pinchZoomOptions) │ │ │ │ │ ); │ │ │ │ │ - if (OpenLayers.Control.PinchZoom) { │ │ │ │ │ - this.pinchZoom = new OpenLayers.Control.PinchZoom( │ │ │ │ │ - OpenLayers.Util.extend({ │ │ │ │ │ - map: this.map │ │ │ │ │ - }, this.pinchZoomOptions)); │ │ │ │ │ - } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Method: defaultClick │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * evt - {Event} │ │ │ │ │ @@ -74797,342 +72801,2242 @@ │ │ │ │ │ defaultClick: function(evt) { │ │ │ │ │ if (evt.lastTouches && evt.lastTouches.length == 2) { │ │ │ │ │ this.map.zoomOut(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: defaultDblClick │ │ │ │ │ - * │ │ │ │ │ + * Method: defaultDblClick │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ defaultDblClick: function(evt) { │ │ │ │ │ this.map.zoomTo(this.map.zoom + 1, evt.xy); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.TouchNavigation" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/KeyboardDefaults.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Handler/Keyboard.js │ │ │ │ │ + * @requires OpenLayers/Events.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.KeyboardDefaults │ │ │ │ │ + * The KeyboardDefaults control adds panning and zooming functions, controlled │ │ │ │ │ + * with the keyboard. By default arrow keys pan, +/- keys zoom & Page Up/Page │ │ │ │ │ + * Down/Home/End scroll by three quarters of a page. │ │ │ │ │ + * │ │ │ │ │ + * This control has no visible appearance. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.KeyboardDefaults = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ + */ │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: slideFactor │ │ │ │ │ + * Pixels to slide by. │ │ │ │ │ + */ │ │ │ │ │ + slideFactor: 75, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: observeElement │ │ │ │ │ + * {DOMelement|String} The DOM element to handle keys for. You │ │ │ │ │ + * can use the map div here, to have the navigation keys │ │ │ │ │ + * work when the map div has the focus. If undefined the │ │ │ │ │ + * document is used. │ │ │ │ │ + */ │ │ │ │ │ + observeElement: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.KeyboardDefaults │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Create handler. │ │ │ │ │ + */ │ │ │ │ │ + draw: function() { │ │ │ │ │ + var observeElement = this.observeElement || document; │ │ │ │ │ + this.handler = new OpenLayers.Handler.Keyboard(this, { │ │ │ │ │ + "keydown": this.defaultKeyPress │ │ │ │ │ + }, { │ │ │ │ │ + observeElement: observeElement │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: defaultDblRightClick │ │ │ │ │ + * Method: defaultKeyPress │ │ │ │ │ + * When handling the key event, we only use evt.keyCode. This holds │ │ │ │ │ + * some drawbacks, though we get around them below. When interpretting │ │ │ │ │ + * the keycodes below (including the comments associated with them), │ │ │ │ │ + * consult the URL below. For instance, the Safari browser returns │ │ │ │ │ + * "IE keycodes", and so is supported by any keycode labeled "IE". │ │ │ │ │ * │ │ │ │ │ + * Very informative URL: │ │ │ │ │ + * http://unixpapa.com/js/key.html │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - defaultDblRightClick: function(evt) { │ │ │ │ │ - this.map.zoomTo(this.map.zoom - 1, evt.xy); │ │ │ │ │ + defaultKeyPress: function(evt) { │ │ │ │ │ + var size, handled = true; │ │ │ │ │ + │ │ │ │ │ + var target = OpenLayers.Event.element(evt); │ │ │ │ │ + if (target && │ │ │ │ │ + (target.tagName == 'INPUT' || │ │ │ │ │ + target.tagName == 'TEXTAREA' || │ │ │ │ │ + target.tagName == 'SELECT')) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + switch (evt.keyCode) { │ │ │ │ │ + case OpenLayers.Event.KEY_LEFT: │ │ │ │ │ + this.map.pan(-this.slideFactor, 0); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Event.KEY_RIGHT: │ │ │ │ │ + this.map.pan(this.slideFactor, 0); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Event.KEY_UP: │ │ │ │ │ + this.map.pan(0, -this.slideFactor); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Event.KEY_DOWN: │ │ │ │ │ + this.map.pan(0, this.slideFactor); │ │ │ │ │ + break; │ │ │ │ │ + │ │ │ │ │ + case 33: // Page Up. Same in all browsers. │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + this.map.pan(0, -0.75 * size.h); │ │ │ │ │ + break; │ │ │ │ │ + case 34: // Page Down. Same in all browsers. │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + this.map.pan(0, 0.75 * size.h); │ │ │ │ │ + break; │ │ │ │ │ + case 35: // End. Same in all browsers. │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + this.map.pan(0.75 * size.w, 0); │ │ │ │ │ + break; │ │ │ │ │ + case 36: // Home. Same in all browsers. │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + this.map.pan(-0.75 * size.w, 0); │ │ │ │ │ + break; │ │ │ │ │ + │ │ │ │ │ + case 43: // +/= (ASCII), keypad + (ASCII, Opera) │ │ │ │ │ + case 61: // +/= (Mozilla, Opera, some ASCII) │ │ │ │ │ + case 187: // +/= (IE) │ │ │ │ │ + case 107: // keypad + (IE, Mozilla) │ │ │ │ │ + this.map.zoomIn(); │ │ │ │ │ + break; │ │ │ │ │ + case 45: // -/_ (ASCII, Opera), keypad - (ASCII, Opera) │ │ │ │ │ + case 109: // -/_ (Mozilla), keypad - (Mozilla, IE) │ │ │ │ │ + case 189: // -/_ (IE) │ │ │ │ │ + case 95: // -/_ (some ASCII) │ │ │ │ │ + this.map.zoomOut(); │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + handled = false; │ │ │ │ │ + } │ │ │ │ │ + if (handled) { │ │ │ │ │ + // prevent browser default not to move the page │ │ │ │ │ + // when moving the page with the keyboard │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.KeyboardDefaults" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/ArgParser.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.ArgParser │ │ │ │ │ + * The ArgParser control adds location bar query string parsing functionality │ │ │ │ │ + * to an OpenLayers Map. │ │ │ │ │ + * When added to a Map control, on a page load/refresh, the Map will │ │ │ │ │ + * automatically take the href string and parse it for lon, lat, zoom, and │ │ │ │ │ + * layers information. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.ArgParser = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: wheelChange │ │ │ │ │ + * Property: center │ │ │ │ │ + * {<OpenLayers.LonLat>} │ │ │ │ │ + */ │ │ │ │ │ + center: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: zoom │ │ │ │ │ + * {int} │ │ │ │ │ + */ │ │ │ │ │ + zoom: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: layers │ │ │ │ │ + * {String} Each character represents the state of the corresponding layer │ │ │ │ │ + * on the map. │ │ │ │ │ + */ │ │ │ │ │ + layers: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: displayProjection │ │ │ │ │ + * {<OpenLayers.Projection>} Requires proj4js support. │ │ │ │ │ + * Projection used when reading the coordinates from the URL. This will │ │ │ │ │ + * reproject the map coordinates from the URL into the map's │ │ │ │ │ + * projection. │ │ │ │ │ + * │ │ │ │ │ + * If you are using this functionality, be aware that any permalink │ │ │ │ │ + * which is added to the map will determine the coordinate type which │ │ │ │ │ + * is read from the URL, which means you should not add permalinks with │ │ │ │ │ + * different displayProjections to the same map. │ │ │ │ │ + */ │ │ │ │ │ + displayProjection: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.ArgParser │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * deltaZ - {Integer} │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - wheelChange: function(evt, deltaZ) { │ │ │ │ │ - if (!this.map.fractionalZoom) { │ │ │ │ │ - deltaZ = Math.round(deltaZ); │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getParameters │ │ │ │ │ + */ │ │ │ │ │ + getParameters: function(url) { │ │ │ │ │ + url = url || window.location.href; │ │ │ │ │ + var parameters = OpenLayers.Util.getParameters(url); │ │ │ │ │ + │ │ │ │ │ + // If we have an anchor in the url use it to split the url │ │ │ │ │ + var index = url.indexOf('#'); │ │ │ │ │ + if (index > 0) { │ │ │ │ │ + // create an url to parse on the getParameters │ │ │ │ │ + url = '?' + url.substring(index + 1, url.length); │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Util.extend(parameters, │ │ │ │ │ + OpenLayers.Util.getParameters(url)); │ │ │ │ │ } │ │ │ │ │ - var currentZoom = this.map.getZoom(), │ │ │ │ │ - newZoom = currentZoom + deltaZ; │ │ │ │ │ - newZoom = Math.max(newZoom, 0); │ │ │ │ │ - newZoom = Math.min(newZoom, this.map.getNumZoomLevels()); │ │ │ │ │ - if (newZoom === currentZoom) { │ │ │ │ │ - return; │ │ │ │ │ + return parameters; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the control. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + //make sure we dont already have an arg parser attached │ │ │ │ │ + for (var i = 0, len = this.map.controls.length; i < len; i++) { │ │ │ │ │ + var control = this.map.controls[i]; │ │ │ │ │ + if ((control != this) && │ │ │ │ │ + (control.CLASS_NAME == "OpenLayers.Control.ArgParser")) { │ │ │ │ │ + │ │ │ │ │ + // If a second argparser is added to the map, then we │ │ │ │ │ + // override the displayProjection to be the one added to the │ │ │ │ │ + // map. │ │ │ │ │ + if (control.displayProjection != this.displayProjection) { │ │ │ │ │ + this.displayProjection = control.displayProjection; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (i == this.map.controls.length) { │ │ │ │ │ + │ │ │ │ │ + var args = this.getParameters(); │ │ │ │ │ + // Be careful to set layer first, to not trigger unnecessary layer loads │ │ │ │ │ + if (args.layers) { │ │ │ │ │ + this.layers = args.layers; │ │ │ │ │ + │ │ │ │ │ + // when we add a new layer, set its visibility │ │ │ │ │ + this.map.events.register('addlayer', this, │ │ │ │ │ + this.configureLayers); │ │ │ │ │ + this.configureLayers(); │ │ │ │ │ + } │ │ │ │ │ + if (args.lat && args.lon) { │ │ │ │ │ + this.center = new OpenLayers.LonLat(parseFloat(args.lon), │ │ │ │ │ + parseFloat(args.lat)); │ │ │ │ │ + if (args.zoom) { │ │ │ │ │ + this.zoom = parseFloat(args.zoom); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // when we add a new baselayer to see when we can set the center │ │ │ │ │ + this.map.events.register('changebaselayer', this, │ │ │ │ │ + this.setCenter); │ │ │ │ │ + this.setCenter(); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.map.zoomTo(newZoom, evt.xy); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: wheelUp │ │ │ │ │ - * User spun scroll wheel up │ │ │ │ │ + * Method: setCenter │ │ │ │ │ + * As soon as a baseLayer has been loaded, we center and zoom │ │ │ │ │ + * ...and remove the handler. │ │ │ │ │ + */ │ │ │ │ │ + setCenter: function() { │ │ │ │ │ + │ │ │ │ │ + if (this.map.baseLayer) { │ │ │ │ │ + //dont need to listen for this one anymore │ │ │ │ │ + this.map.events.unregister('changebaselayer', this, │ │ │ │ │ + this.setCenter); │ │ │ │ │ + │ │ │ │ │ + if (this.displayProjection) { │ │ │ │ │ + this.center.transform(this.displayProjection, │ │ │ │ │ + this.map.getProjectionObject()); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.map.setCenter(this.center, this.zoom); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: configureLayers │ │ │ │ │ + * As soon as all the layers are loaded, cycle through them and │ │ │ │ │ + * hide or show them. │ │ │ │ │ + */ │ │ │ │ │ + configureLayers: function() { │ │ │ │ │ + │ │ │ │ │ + if (this.layers.length == this.map.layers.length) { │ │ │ │ │ + this.map.events.unregister('addlayer', this, this.configureLayers); │ │ │ │ │ + │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + var c = this.layers.charAt(i); │ │ │ │ │ + │ │ │ │ │ + if (c == "B") { │ │ │ │ │ + this.map.setBaseLayer(layer); │ │ │ │ │ + } else if ((c == "T") || (c == "F")) { │ │ │ │ │ + layer.setVisibility(c == "T"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ArgParser" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/WMTSGetFeatureInfo.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Handler/Click.js │ │ │ │ │ + * @requires OpenLayers/Handler/Hover.js │ │ │ │ │ + * @requires OpenLayers/Request.js │ │ │ │ │ + * @requires OpenLayers/Format/WMSGetFeatureInfo.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.WMTSGetFeatureInfo │ │ │ │ │ + * The WMTSGetFeatureInfo control uses a WMTS query to get information about a │ │ │ │ │ + * point on the map. The information may be in a display-friendly format │ │ │ │ │ + * such as HTML, or a machine-friendly format such as GML, depending on the │ │ │ │ │ + * server's capabilities and the client's configuration. This control │ │ │ │ │ + * handles click or hover events, attempts to parse the results using an │ │ │ │ │ + * OpenLayers.Format, and fires a 'getfeatureinfo' event for each layer │ │ │ │ │ + * queried. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.WMTSGetFeatureInfo = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: hover │ │ │ │ │ + * {Boolean} Send GetFeatureInfo requests when mouse stops moving. │ │ │ │ │ + * Default is false. │ │ │ │ │ + */ │ │ │ │ │ + hover: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: requestEncoding │ │ │ │ │ + * {String} One of "KVP" or "REST". Only KVP encoding is supported at this │ │ │ │ │ + * time. │ │ │ │ │ + */ │ │ │ │ │ + requestEncoding: "KVP", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: drillDown │ │ │ │ │ + * {Boolean} Drill down over all WMTS layers in the map. When │ │ │ │ │ + * using drillDown mode, hover is not possible. A getfeatureinfo event │ │ │ │ │ + * will be fired for each layer queried. │ │ │ │ │ + */ │ │ │ │ │ + drillDown: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxFeatures │ │ │ │ │ + * {Integer} Maximum number of features to return from a WMTS query. This │ │ │ │ │ + * sets the feature_count parameter on WMTS GetFeatureInfo │ │ │ │ │ + * requests. │ │ │ │ │ + */ │ │ │ │ │ + maxFeatures: 10, │ │ │ │ │ + │ │ │ │ │ + /** APIProperty: clickCallback │ │ │ │ │ + * {String} The click callback to register in the │ │ │ │ │ + * {<OpenLayers.Handler.Click>} object created when the hover │ │ │ │ │ + * option is set to false. Default is "click". │ │ │ │ │ + */ │ │ │ │ │ + clickCallback: "click", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: layers │ │ │ │ │ + * {Array(<OpenLayers.Layer.WMTS>)} The layers to query for feature info. │ │ │ │ │ + * If omitted, all map WMTS layers will be considered. │ │ │ │ │ + */ │ │ │ │ │ + layers: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: queryVisible │ │ │ │ │ + * {Boolean} Filter out hidden layers when searching the map for layers to │ │ │ │ │ + * query. Default is true. │ │ │ │ │ + */ │ │ │ │ │ + queryVisible: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: infoFormat │ │ │ │ │ + * {String} The mimetype to request from the server │ │ │ │ │ + */ │ │ │ │ │ + infoFormat: 'text/html', │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: vendorParams │ │ │ │ │ + * {Object} Additional parameters that will be added to the request, for │ │ │ │ │ + * WMTS implementations that support them. This could e.g. look like │ │ │ │ │ + * (start code) │ │ │ │ │ + * { │ │ │ │ │ + * radius: 5 │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ + vendorParams: {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: format │ │ │ │ │ + * {<OpenLayers.Format>} A format for parsing GetFeatureInfo responses. │ │ │ │ │ + * Default is <OpenLayers.Format.WMSGetFeatureInfo>. │ │ │ │ │ + */ │ │ │ │ │ + format: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: formatOptions │ │ │ │ │ + * {Object} Optional properties to set on the format (if one is not provided │ │ │ │ │ + * in the <format> property. │ │ │ │ │ + */ │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: handlerOptions │ │ │ │ │ + * {Object} Additional options for the handlers used by this control, e.g. │ │ │ │ │ + * (start code) │ │ │ │ │ + * { │ │ │ │ │ + * "click": {delay: 100}, │ │ │ │ │ + * "hover": {delay: 300} │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: handler │ │ │ │ │ + * {Object} Reference to the <OpenLayers.Handler> for this control │ │ │ │ │ + */ │ │ │ │ │ + handler: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: hoverRequest │ │ │ │ │ + * {<OpenLayers.Request>} contains the currently running hover request │ │ │ │ │ + * (if any). │ │ │ │ │ + */ │ │ │ │ │ + hoverRequest: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ + * beforegetfeatureinfo - Triggered before each request is sent. │ │ │ │ │ + * The event object has an *xy* property with the position of the │ │ │ │ │ + * mouse click or hover event that triggers the request and a *layer* │ │ │ │ │ + * property referencing the layer about to be queried. If a listener │ │ │ │ │ + * returns false, the request will not be issued. │ │ │ │ │ + * getfeatureinfo - Triggered when a GetFeatureInfo response is received. │ │ │ │ │ + * The event object has a *text* property with the body of the │ │ │ │ │ + * response (String), a *features* property with an array of the │ │ │ │ │ + * parsed features, an *xy* property with the position of the mouse │ │ │ │ │ + * click or hover event that triggered the request, a *layer* property │ │ │ │ │ + * referencing the layer queried and a *request* property with the │ │ │ │ │ + * request itself. If drillDown is set to true, one event will be fired │ │ │ │ │ + * for each layer queried. │ │ │ │ │ + * exception - Triggered when a GetFeatureInfo request fails (with a │ │ │ │ │ + * status other than 200) or whenparsing fails. Listeners will receive │ │ │ │ │ + * an event with *request*, *xy*, and *layer* properties. In the case │ │ │ │ │ + * of a parsing error, the event will also contain an *error* property. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: pending │ │ │ │ │ + * {Number} The number of pending requests. │ │ │ │ │ + */ │ │ │ │ │ + pending: 0, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: <OpenLayers.Control.WMTSGetFeatureInfo> │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + │ │ │ │ │ + if (!this.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.WMSGetFeatureInfo( │ │ │ │ │ + options.formatOptions │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.drillDown === true) { │ │ │ │ │ + this.hover = false; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Hover( │ │ │ │ │ + this, { │ │ │ │ │ + move: this.cancelHover, │ │ │ │ │ + pause: this.getInfoForHover │ │ │ │ │ + }, │ │ │ │ │ + OpenLayers.Util.extend( │ │ │ │ │ + this.handlerOptions.hover || {}, { │ │ │ │ │ + delay: 250 │ │ │ │ │ + } │ │ │ │ │ + ) │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + var callbacks = {}; │ │ │ │ │ + callbacks[this.clickCallback] = this.getInfoForClick; │ │ │ │ │ + this.handler = new OpenLayers.Handler.Click( │ │ │ │ │ + this, callbacks, this.handlerOptions.click || {} │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getInfoForClick │ │ │ │ │ + * Called on click │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ + */ │ │ │ │ │ + getInfoForClick: function(evt) { │ │ │ │ │ + this.request(evt.xy, {}); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getInfoForHover │ │ │ │ │ + * Pause callback for the hover handler │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} │ │ │ │ │ + */ │ │ │ │ │ + getInfoForHover: function(evt) { │ │ │ │ │ + this.request(evt.xy, { │ │ │ │ │ + hover: true │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: cancelHover │ │ │ │ │ + * Cancel callback for the hover handler │ │ │ │ │ + */ │ │ │ │ │ + cancelHover: function() { │ │ │ │ │ + if (this.hoverRequest) { │ │ │ │ │ + --this.pending; │ │ │ │ │ + if (this.pending <= 0) { │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + this.pending = 0; │ │ │ │ │ + } │ │ │ │ │ + this.hoverRequest.abort(); │ │ │ │ │ + this.hoverRequest = null; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: findLayers │ │ │ │ │ + * Internal method to get the layers, independent of whether we are │ │ │ │ │ + * inspecting the map or using a client-provided array │ │ │ │ │ + */ │ │ │ │ │ + findLayers: function() { │ │ │ │ │ + var candidates = this.layers || this.map.layers; │ │ │ │ │ + var layers = []; │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = candidates[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.WMTS && │ │ │ │ │ + layer.requestEncoding === this.requestEncoding && │ │ │ │ │ + (!this.queryVisible || layer.getVisibility())) { │ │ │ │ │ + layers.push(layer); │ │ │ │ │ + if (!this.drillDown || this.hover) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return layers; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: buildRequestOptions │ │ │ │ │ + * Build an object with the relevant options for the GetFeatureInfo request. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer.WMTS>} A WMTS layer. │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} The position on the map where the │ │ │ │ │ + * mouse event occurred. │ │ │ │ │ + */ │ │ │ │ │ + buildRequestOptions: function(layer, xy) { │ │ │ │ │ + var loc = this.map.getLonLatFromPixel(xy); │ │ │ │ │ + var getTileUrl = layer.getURL( │ │ │ │ │ + new OpenLayers.Bounds(loc.lon, loc.lat, loc.lon, loc.lat) │ │ │ │ │ + ); │ │ │ │ │ + var params = OpenLayers.Util.getParameters(getTileUrl); │ │ │ │ │ + var tileInfo = layer.getTileInfo(loc); │ │ │ │ │ + OpenLayers.Util.extend(params, { │ │ │ │ │ + service: "WMTS", │ │ │ │ │ + version: layer.version, │ │ │ │ │ + request: "GetFeatureInfo", │ │ │ │ │ + infoFormat: this.infoFormat, │ │ │ │ │ + i: tileInfo.i, │ │ │ │ │ + j: tileInfo.j │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Util.applyDefaults(params, this.vendorParams); │ │ │ │ │ + return { │ │ │ │ │ + url: OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url, │ │ │ │ │ + params: OpenLayers.Util.upperCaseObject(params), │ │ │ │ │ + callback: function(request) { │ │ │ │ │ + this.handleResponse(xy, request, layer); │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: request │ │ │ │ │ + * Sends a GetFeatureInfo request to the WMTS │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * delta - {Integer} │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} The position on the map where the mouse event │ │ │ │ │ + * occurred. │ │ │ │ │ + * options - {Object} additional options for this method. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * - *hover* {Boolean} true if we do the request for the hover handler │ │ │ │ │ */ │ │ │ │ │ - wheelUp: function(evt, delta) { │ │ │ │ │ - this.wheelChange(evt, delta || 1); │ │ │ │ │ + request: function(xy, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var layers = this.findLayers(); │ │ │ │ │ + if (layers.length > 0) { │ │ │ │ │ + var issue, layer; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + layer = layers[i]; │ │ │ │ │ + issue = this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ + xy: xy, │ │ │ │ │ + layer: layer │ │ │ │ │ + }); │ │ │ │ │ + if (issue !== false) { │ │ │ │ │ + ++this.pending; │ │ │ │ │ + var requestOptions = this.buildRequestOptions(layer, xy); │ │ │ │ │ + var request = OpenLayers.Request.GET(requestOptions); │ │ │ │ │ + if (options.hover === true) { │ │ │ │ │ + this.hoverRequest = request; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.pending > 0) { │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleResponse │ │ │ │ │ + * Handler for the GetFeatureInfo response. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} The position on the map where the mouse event │ │ │ │ │ + * occurred. │ │ │ │ │ + * request - {XMLHttpRequest} The request object. │ │ │ │ │ + * layer - {<OpenLayers.Layer.WMTS>} The queried layer. │ │ │ │ │ + */ │ │ │ │ │ + handleResponse: function(xy, request, layer) { │ │ │ │ │ + --this.pending; │ │ │ │ │ + if (this.pending <= 0) { │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + this.pending = 0; │ │ │ │ │ + } │ │ │ │ │ + if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ + this.events.triggerEvent("exception", { │ │ │ │ │ + xy: xy, │ │ │ │ │ + request: request, │ │ │ │ │ + layer: layer │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ + } │ │ │ │ │ + var features, except; │ │ │ │ │ + try { │ │ │ │ │ + features = this.format.read(doc); │ │ │ │ │ + } catch (error) { │ │ │ │ │ + except = true; │ │ │ │ │ + this.events.triggerEvent("exception", { │ │ │ │ │ + xy: xy, │ │ │ │ │ + request: request, │ │ │ │ │ + error: error, │ │ │ │ │ + layer: layer │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + if (!except) { │ │ │ │ │ + this.events.triggerEvent("getfeatureinfo", { │ │ │ │ │ + text: request.responseText, │ │ │ │ │ + features: features, │ │ │ │ │ + request: request, │ │ │ │ │ + xy: xy, │ │ │ │ │ + layer: layer │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.WMTSGetFeatureInfo" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/Button.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.Button │ │ │ │ │ + * The Button control is a very simple push-button, for use with │ │ │ │ │ + * <OpenLayers.Control.Panel>. │ │ │ │ │ + * When clicked, the function trigger() is executed. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + * │ │ │ │ │ + * Use: │ │ │ │ │ + * (code) │ │ │ │ │ + * var button = new OpenLayers.Control.Button({ │ │ │ │ │ + * displayClass: "MyButton", trigger: myFunction │ │ │ │ │ + * }); │ │ │ │ │ + * panel.addControls([button]); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Will create a button with CSS class MyButtonItemInactive, that │ │ │ │ │ + * will call the function MyFunction() when clicked. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.Button = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + /** │ │ │ │ │ + * Property: type │ │ │ │ │ + * {Integer} OpenLayers.Control.TYPE_BUTTON. │ │ │ │ │ + */ │ │ │ │ │ + type: OpenLayers.Control.TYPE_BUTTON, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: trigger │ │ │ │ │ + * Called by a control panel when the button is clicked. │ │ │ │ │ + */ │ │ │ │ │ + trigger: function() {}, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Button" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/ZoomOut.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control/Button.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.ZoomOut │ │ │ │ │ + * The ZoomOut control is a button to decrease the zoom level of a map. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.ZoomOut = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: trigger │ │ │ │ │ + */ │ │ │ │ │ + trigger: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.zoomOut(); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomOut" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/SLDSelect.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Layer/WMS.js │ │ │ │ │ + * @requires OpenLayers/Handler/RegularPolygon.js │ │ │ │ │ + * @requires OpenLayers/Handler/Polygon.js │ │ │ │ │ + * @requires OpenLayers/Handler/Path.js │ │ │ │ │ + * @requires OpenLayers/Handler/Click.js │ │ │ │ │ + * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ + * @requires OpenLayers/Format/SLD/v1_0_0.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.SLDSelect │ │ │ │ │ + * Perform selections on WMS layers using Styled Layer Descriptor (SLD) │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.SLDSelect = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: wheelDown │ │ │ │ │ - * User spun scroll wheel down │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ + * selected - Triggered when a selection occurs. Listeners receive an │ │ │ │ │ + * event with *filters* and *layer* properties. Filters will be an │ │ │ │ │ + * array of OpenLayers.Filter objects created in order to perform │ │ │ │ │ + * the particular selection. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: clearOnDeactivate │ │ │ │ │ + * {Boolean} Should the selection be cleared when the control is │ │ │ │ │ + * deactivated. Default value is false. │ │ │ │ │ + */ │ │ │ │ │ + clearOnDeactivate: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layers │ │ │ │ │ + * {Array(<OpenLayers.Layer.WMS>)} The WMS layers this control will work │ │ │ │ │ + * on. │ │ │ │ │ + */ │ │ │ │ │ + layers: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: callbacks │ │ │ │ │ + * {Object} The functions that are sent to the handler for callback │ │ │ │ │ + */ │ │ │ │ │ + callbacks: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: selectionSymbolizer │ │ │ │ │ + * {Object} Determines the styling of the selected objects. Default is │ │ │ │ │ + * a selection in red. │ │ │ │ │ + */ │ │ │ │ │ + selectionSymbolizer: { │ │ │ │ │ + 'Polygon': { │ │ │ │ │ + fillColor: '#FF0000', │ │ │ │ │ + stroke: false │ │ │ │ │ + }, │ │ │ │ │ + 'Line': { │ │ │ │ │ + strokeColor: '#FF0000', │ │ │ │ │ + strokeWidth: 2 │ │ │ │ │ + }, │ │ │ │ │ + 'Point': { │ │ │ │ │ + graphicName: 'square', │ │ │ │ │ + fillColor: '#FF0000', │ │ │ │ │ + pointRadius: 5 │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layerOptions │ │ │ │ │ + * {Object} The options to apply to the selection layer, by default the │ │ │ │ │ + * selection layer will be kept out of the layer switcher. │ │ │ │ │ + */ │ │ │ │ │ + layerOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: handlerOptions │ │ │ │ │ + * {Object} Used to set non-default properties on the control's handler │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: sketchStyle │ │ │ │ │ + * {<OpenLayers.Style>|Object} Style or symbolizer to use for the sketch │ │ │ │ │ + * handler. The recommended way of styling the sketch layer, however, is │ │ │ │ │ + * to configure an <OpenLayers.StyleMap> in the layerOptions of the │ │ │ │ │ + * <handlerOptions>: │ │ │ │ │ * │ │ │ │ │ + * (code) │ │ │ │ │ + * new OpenLayers.Control.SLDSelect(OpenLayers.Handler.Path, { │ │ │ │ │ + * handlerOptions: { │ │ │ │ │ + * layerOptions: { │ │ │ │ │ + * styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ + * "default": {strokeColor: "yellow"} │ │ │ │ │ + * }) │ │ │ │ │ + * } │ │ │ │ │ + * } │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ + sketchStyle: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: wfsCache │ │ │ │ │ + * {Object} Cache to use for storing parsed results from │ │ │ │ │ + * <OpenLayers.Format.WFSDescribeFeatureType.read>. If not provided, │ │ │ │ │ + * these will be cached on the prototype. │ │ │ │ │ + */ │ │ │ │ │ + wfsCache: {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layerCache │ │ │ │ │ + * {Object} Cache to use for storing references to the selection layers. │ │ │ │ │ + * Normally each source layer will have exactly 1 selection layer of │ │ │ │ │ + * type OpenLayers.Layer.WMS. If not provided, layers will │ │ │ │ │ + * be cached on the prototype. Note that if <clearOnDeactivate> is │ │ │ │ │ + * true, the layer will no longer be cached after deactivating the │ │ │ │ │ + * control. │ │ │ │ │ + */ │ │ │ │ │ + layerCache: {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.SLDSelect │ │ │ │ │ + * Create a new control for selecting features in WMS layers using │ │ │ │ │ + * Styled Layer Descriptor (SLD). │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * delta - {Integer} │ │ │ │ │ + * handler - {<OpenLayers.Class>} A sketch handler class. This determines │ │ │ │ │ + * the type of selection, e.g. box (<OpenLayers.Handler.Box>), point │ │ │ │ │ + * (<OpenLayers.Handler.Point>), path (<OpenLayers.Handler.Path>) or │ │ │ │ │ + * polygon (<OpenLayers.Handler.Polygon>) selection. To use circle │ │ │ │ │ + * type selection, use <OpenLayers.Handler.RegularPolygon> and pass │ │ │ │ │ + * the number of desired sides (e.g. 40) as "sides" property to the │ │ │ │ │ + * <handlerOptions>. │ │ │ │ │ + * options - {Object} An object containing all configuration properties for │ │ │ │ │ + * the control. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * layers - Array({<OpenLayers.Layer.WMS>}) The layers to perform the │ │ │ │ │ + * selection on. │ │ │ │ │ */ │ │ │ │ │ - wheelDown: function(evt, delta) { │ │ │ │ │ - this.wheelChange(evt, delta || -1); │ │ │ │ │ + initialize: function(handler, options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + │ │ │ │ │ + this.callbacks = OpenLayers.Util.extend({ │ │ │ │ │ + done: this.select, │ │ │ │ │ + click: this.select │ │ │ │ │ + }, this.callbacks); │ │ │ │ │ + this.handlerOptions = this.handlerOptions || {}; │ │ │ │ │ + this.layerOptions = OpenLayers.Util.applyDefaults(this.layerOptions, { │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + tileOptions: { │ │ │ │ │ + maxGetUrlLength: 2048 │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (this.sketchStyle) { │ │ │ │ │ + this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.handlerOptions.layerOptions, { │ │ │ │ │ + styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ + "default": this.sketchStyle │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + this.handler = new handler(this, this.callbacks, this.handlerOptions); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: disableZoomBox │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Take care of things that are not handled in superclass. │ │ │ │ │ */ │ │ │ │ │ - disableZoomBox: function() { │ │ │ │ │ - this.zoomBoxEnabled = false; │ │ │ │ │ - this.zoomBox.deactivate(); │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var key in this.layerCache) { │ │ │ │ │ + delete this.layerCache[key]; │ │ │ │ │ + } │ │ │ │ │ + for (var key in this.wfsCache) { │ │ │ │ │ + delete this.wfsCache[key]; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: enableZoomBox │ │ │ │ │ + * Method: coupleLayerVisiblity │ │ │ │ │ + * Couple the selection layer and the source layer with respect to │ │ │ │ │ + * layer visibility. So if the source layer is turned off, the │ │ │ │ │ + * selection layer is also turned off. │ │ │ │ │ + * │ │ │ │ │ + * Context: │ │ │ │ │ + * - {<OpenLayers.Layer>} │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} │ │ │ │ │ */ │ │ │ │ │ - enableZoomBox: function() { │ │ │ │ │ - this.zoomBoxEnabled = true; │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.zoomBox.activate(); │ │ │ │ │ + coupleLayerVisiblity: function(evt) { │ │ │ │ │ + this.setVisibility(evt.object.getVisibility()); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createSelectionLayer │ │ │ │ │ + * Creates a "clone" from the source layer in which the selection can │ │ │ │ │ + * be drawn. This ensures both the source layer and the selection are │ │ │ │ │ + * visible and not only the selection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * source - {<OpenLayers.Layer.WMS>} The source layer on which the selection │ │ │ │ │ + * is performed. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.WMS>} A WMS layer with maxGetUrlLength configured to 2048 │ │ │ │ │ + * since SLD selections can easily get quite long. │ │ │ │ │ + */ │ │ │ │ │ + createSelectionLayer: function(source) { │ │ │ │ │ + // check if we already have a selection layer for the source layer │ │ │ │ │ + var selectionLayer; │ │ │ │ │ + if (!this.layerCache[source.id]) { │ │ │ │ │ + selectionLayer = new OpenLayers.Layer.WMS(source.name, │ │ │ │ │ + source.url, source.params, │ │ │ │ │ + OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.layerOptions, │ │ │ │ │ + source.getOptions()) │ │ │ │ │ + ); │ │ │ │ │ + this.layerCache[source.id] = selectionLayer; │ │ │ │ │ + // make sure the layers are coupled wrt visibility, but only │ │ │ │ │ + // if they are not displayed in the layer switcher, because in │ │ │ │ │ + // that case the user cannot control visibility. │ │ │ │ │ + if (this.layerOptions.displayInLayerSwitcher === false) { │ │ │ │ │ + source.events.on({ │ │ │ │ │ + "visibilitychanged": this.coupleLayerVisiblity, │ │ │ │ │ + scope: selectionLayer │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + this.map.addLayer(selectionLayer); │ │ │ │ │ + } else { │ │ │ │ │ + selectionLayer = this.layerCache[source.id]; │ │ │ │ │ } │ │ │ │ │ + return selectionLayer; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: disableZoomWheel │ │ │ │ │ + * Method: createSLD │ │ │ │ │ + * Create the SLD document for the layer using the supplied filters. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer.WMS>} │ │ │ │ │ + * filters - Array({<OpenLayers.Filter>}) The filters to be applied. │ │ │ │ │ + * geometryAttributes - Array({Object}) The geometry attributes of the │ │ │ │ │ + * layer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The SLD document generated as a string. │ │ │ │ │ */ │ │ │ │ │ + createSLD: function(layer, filters, geometryAttributes) { │ │ │ │ │ + var sld = { │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + namedLayers: {} │ │ │ │ │ + }; │ │ │ │ │ + var layerNames = [layer.params.LAYERS].join(",").split(","); │ │ │ │ │ + for (var i = 0, len = layerNames.length; i < len; i++) { │ │ │ │ │ + var name = layerNames[i]; │ │ │ │ │ + sld.namedLayers[name] = { │ │ │ │ │ + name: name, │ │ │ │ │ + userStyles: [] │ │ │ │ │ + }; │ │ │ │ │ + var symbolizer = this.selectionSymbolizer; │ │ │ │ │ + var geometryAttribute = geometryAttributes[i]; │ │ │ │ │ + if (geometryAttribute.type.indexOf('Polygon') >= 0) { │ │ │ │ │ + symbolizer = { │ │ │ │ │ + Polygon: this.selectionSymbolizer['Polygon'] │ │ │ │ │ + }; │ │ │ │ │ + } else if (geometryAttribute.type.indexOf('LineString') >= 0) { │ │ │ │ │ + symbolizer = { │ │ │ │ │ + Line: this.selectionSymbolizer['Line'] │ │ │ │ │ + }; │ │ │ │ │ + } else if (geometryAttribute.type.indexOf('Point') >= 0) { │ │ │ │ │ + symbolizer = { │ │ │ │ │ + Point: this.selectionSymbolizer['Point'] │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + var filter = filters[i]; │ │ │ │ │ + sld.namedLayers[name].userStyles.push({ │ │ │ │ │ + name: 'default', │ │ │ │ │ + rules: [ │ │ │ │ │ + new OpenLayers.Rule({ │ │ │ │ │ + symbolizer: symbolizer, │ │ │ │ │ + filter: filter, │ │ │ │ │ + maxScaleDenominator: layer.options.minScale │ │ │ │ │ + }) │ │ │ │ │ + ] │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Format.SLD({ │ │ │ │ │ + srsName: this.map.getProjection() │ │ │ │ │ + }).write(sld); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - disableZoomWheel: function() { │ │ │ │ │ - this.zoomWheelEnabled = false; │ │ │ │ │ - this.handlers.wheel.deactivate(); │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseDescribeLayer │ │ │ │ │ + * Parse the SLD WMS DescribeLayer response and issue the corresponding │ │ │ │ │ + * WFS DescribeFeatureType request │ │ │ │ │ + * │ │ │ │ │ + * request - {XMLHttpRequest} The request object. │ │ │ │ │ + */ │ │ │ │ │ + parseDescribeLayer: function(request) { │ │ │ │ │ + var format = new OpenLayers.Format.WMSDescribeLayer(); │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ + } │ │ │ │ │ + var describeLayer = format.read(doc); │ │ │ │ │ + var typeNames = []; │ │ │ │ │ + var url = null; │ │ │ │ │ + for (var i = 0, len = describeLayer.length; i < len; i++) { │ │ │ │ │ + // perform a WFS DescribeFeatureType request │ │ │ │ │ + if (describeLayer[i].owsType == "WFS") { │ │ │ │ │ + typeNames.push(describeLayer[i].typeName); │ │ │ │ │ + url = describeLayer[i].owsURL; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var options = { │ │ │ │ │ + url: url, │ │ │ │ │ + params: { │ │ │ │ │ + SERVICE: "WFS", │ │ │ │ │ + TYPENAME: typeNames.toString(), │ │ │ │ │ + REQUEST: "DescribeFeatureType", │ │ │ │ │ + VERSION: "1.0.0" │ │ │ │ │ + }, │ │ │ │ │ + callback: function(request) { │ │ │ │ │ + var format = new OpenLayers.Format.WFSDescribeFeatureType(); │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ + } │ │ │ │ │ + var describeFeatureType = format.read(doc); │ │ │ │ │ + this.control.wfsCache[this.layer.id] = describeFeatureType; │ │ │ │ │ + this.control._queue && this.control.applySelection(); │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Request.GET(options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: enableZoomWheel │ │ │ │ │ + * Method: getGeometryAttributes │ │ │ │ │ + * Look up the geometry attributes from the WFS DescribeFeatureType response │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer.WMS>} The layer for which to look up the │ │ │ │ │ + * geometry attributes. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * Array({Object}) Array of geometry attributes │ │ │ │ │ */ │ │ │ │ │ + getGeometryAttributes: function(layer) { │ │ │ │ │ + var result = []; │ │ │ │ │ + var cache = this.wfsCache[layer.id]; │ │ │ │ │ + for (var i = 0, len = cache.featureTypes.length; i < len; i++) { │ │ │ │ │ + var typeName = cache.featureTypes[i]; │ │ │ │ │ + var properties = typeName.properties; │ │ │ │ │ + for (var j = 0, lenj = properties.length; j < lenj; j++) { │ │ │ │ │ + var property = properties[j]; │ │ │ │ │ + var type = property.type; │ │ │ │ │ + if ((type.indexOf('LineString') >= 0) || │ │ │ │ │ + (type.indexOf('GeometryAssociationType') >= 0) || │ │ │ │ │ + (type.indexOf('GeometryPropertyType') >= 0) || │ │ │ │ │ + (type.indexOf('Point') >= 0) || │ │ │ │ │ + (type.indexOf('Polygon') >= 0)) { │ │ │ │ │ + result.push(property); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return result; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - enableZoomWheel: function() { │ │ │ │ │ - this.zoomWheelEnabled = true; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Activate the control. Activating the control will perform a SLD WMS │ │ │ │ │ + * DescribeLayer request followed by a WFS DescribeFeatureType request │ │ │ │ │ + * so that the proper symbolizers can be chosen based on the geometry │ │ │ │ │ + * type. │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Control.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.layers[i]; │ │ │ │ │ + if (layer && !this.wfsCache[layer.id]) { │ │ │ │ │ + var options = { │ │ │ │ │ + url: layer.url, │ │ │ │ │ + params: { │ │ │ │ │ + SERVICE: "WMS", │ │ │ │ │ + VERSION: layer.params.VERSION, │ │ │ │ │ + LAYERS: layer.params.LAYERS, │ │ │ │ │ + REQUEST: "DescribeLayer" │ │ │ │ │ + }, │ │ │ │ │ + callback: this.parseDescribeLayer, │ │ │ │ │ + scope: { │ │ │ │ │ + layer: layer, │ │ │ │ │ + control: this │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Request.GET(options); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the control. If clearOnDeactivate is true, remove the │ │ │ │ │ + * selection layer(s). │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Control.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.layers[i]; │ │ │ │ │ + if (layer && this.clearOnDeactivate === true) { │ │ │ │ │ + var layerCache = this.layerCache; │ │ │ │ │ + var selectionLayer = layerCache[layer.id]; │ │ │ │ │ + if (selectionLayer) { │ │ │ │ │ + layer.events.un({ │ │ │ │ │ + "visibilitychanged": this.coupleLayerVisiblity, │ │ │ │ │ + scope: selectionLayer │ │ │ │ │ + }); │ │ │ │ │ + selectionLayer.destroy(); │ │ │ │ │ + delete layerCache[layer.id]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setLayers │ │ │ │ │ + * Set the layers on which the selection should be performed. Call the │ │ │ │ │ + * setLayers method if the layer(s) to be used change and the same │ │ │ │ │ + * control should be used on a new set of layers. │ │ │ │ │ + * If the control is already active, it will be active after the new │ │ │ │ │ + * set of layers is set. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layers - {Array(<OpenLayers.Layer.WMS>)} The new set of layers on which │ │ │ │ │ + * the selection should be performed. │ │ │ │ │ + */ │ │ │ │ │ + setLayers: function(layers) { │ │ │ │ │ if (this.active) { │ │ │ │ │ - this.handlers.wheel.activate(); │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.layers = layers; │ │ │ │ │ + this.activate(); │ │ │ │ │ + } else { │ │ │ │ │ + this.layers = layers; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Navigation" │ │ │ │ │ + /** │ │ │ │ │ + * Function: createFilter │ │ │ │ │ + * Create the filter to be used in the SLD. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometryAttribute - {Object} Used to get the name of the geometry │ │ │ │ │ + * attribute which is needed for constructing the spatial filter. │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} The geometry to use. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Filter.Spatial>} The spatial filter created. │ │ │ │ │ + */ │ │ │ │ │ + createFilter: function(geometryAttribute, geometry) { │ │ │ │ │ + var filter = null; │ │ │ │ │ + if (this.handler instanceof OpenLayers.Handler.RegularPolygon) { │ │ │ │ │ + // box │ │ │ │ │ + if (this.handler.irregular === true) { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry.getBounds() │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } else if (this.handler instanceof OpenLayers.Handler.Polygon) { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry │ │ │ │ │ + }); │ │ │ │ │ + } else if (this.handler instanceof OpenLayers.Handler.Path) { │ │ │ │ │ + // if source layer is point based, use DWITHIN instead │ │ │ │ │ + if (geometryAttribute.type.indexOf('Point') >= 0) { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + distance: this.map.getExtent().getWidth() * 0.01, │ │ │ │ │ + distanceUnits: this.map.getUnits(), │ │ │ │ │ + value: geometry │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } else if (this.handler instanceof OpenLayers.Handler.Click) { │ │ │ │ │ + if (geometryAttribute.type.indexOf('Polygon') >= 0) { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + distance: this.map.getExtent().getWidth() * 0.01, │ │ │ │ │ + distanceUnits: this.map.getUnits(), │ │ │ │ │ + value: geometry │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return filter; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: select │ │ │ │ │ + * When the handler is done, use SLD_BODY on the selection layer to │ │ │ │ │ + * display the selection in the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {Object} or {<OpenLayers.Geometry>} │ │ │ │ │ + */ │ │ │ │ │ + select: function(geometry) { │ │ │ │ │ + this._queue = function() { │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.layers[i]; │ │ │ │ │ + var geometryAttributes = this.getGeometryAttributes(layer); │ │ │ │ │ + var filters = []; │ │ │ │ │ + for (var j = 0, lenj = geometryAttributes.length; j < lenj; j++) { │ │ │ │ │ + var geometryAttribute = geometryAttributes[j]; │ │ │ │ │ + if (geometryAttribute !== null) { │ │ │ │ │ + // from the click handler we will not get an actual │ │ │ │ │ + // geometry so transform │ │ │ │ │ + if (!(geometry instanceof OpenLayers.Geometry)) { │ │ │ │ │ + var point = this.map.getLonLatFromPixel( │ │ │ │ │ + geometry.xy); │ │ │ │ │ + geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ + point.lon, point.lat); │ │ │ │ │ + } │ │ │ │ │ + var filter = this.createFilter(geometryAttribute, │ │ │ │ │ + geometry); │ │ │ │ │ + if (filter !== null) { │ │ │ │ │ + filters.push(filter); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var selectionLayer = this.createSelectionLayer(layer); │ │ │ │ │ + │ │ │ │ │ + this.events.triggerEvent("selected", { │ │ │ │ │ + layer: layer, │ │ │ │ │ + filters: filters │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + var sld = this.createSLD(layer, filters, geometryAttributes); │ │ │ │ │ + │ │ │ │ │ + selectionLayer.mergeNewParams({ │ │ │ │ │ + SLD_BODY: sld │ │ │ │ │ + }); │ │ │ │ │ + delete this._queue; │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + this.applySelection(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: applySelection │ │ │ │ │ + * Checks if all required wfs data is cached, and applies the selection │ │ │ │ │ + */ │ │ │ │ │ + applySelection: function() { │ │ │ │ │ + var canApply = true; │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + if (!this.wfsCache[this.layers[i].id]) { │ │ │ │ │ + canApply = false; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + canApply && this._queue.call(this); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.SLDSelect" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/NavToolbar.js │ │ │ │ │ + OpenLayers/Control/PanZoom.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Control/Panel.js │ │ │ │ │ - * @requires OpenLayers/Control/Navigation.js │ │ │ │ │ - * @requires OpenLayers/Control/ZoomBox.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.NavToolbar │ │ │ │ │ - * This Toolbar is an alternative to the Navigation control that displays │ │ │ │ │ - * the state of the control, and provides a UI for changing state to │ │ │ │ │ - * use the zoomBox via a Panel control. │ │ │ │ │ + * Class: OpenLayers.Control.PanZoom │ │ │ │ │ + * The PanZoom is a visible control, composed of a │ │ │ │ │ + * <OpenLayers.Control.PanPanel> and a <OpenLayers.Control.ZoomPanel>. By │ │ │ │ │ + * default it is drawn in the upper left corner of the map. │ │ │ │ │ * │ │ │ │ │ - * If you wish to change the properties of the Navigation control used │ │ │ │ │ - * in the NavToolbar, see: │ │ │ │ │ - * http://trac.openlayers.org/wiki/Toolbars#SubclassingNavToolbar │ │ │ │ │ - * │ │ │ │ │ - * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control.Panel> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.NavToolbar = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ +OpenLayers.Control.PanZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: slideFactor │ │ │ │ │ + * {Integer} Number of pixels by which we'll pan the map in any direction │ │ │ │ │ + * on clicking the arrow buttons. If you want to pan by some ratio │ │ │ │ │ + * of the map dimensions, use <slideRatio> instead. │ │ │ │ │ + */ │ │ │ │ │ + slideFactor: 50, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: slideRatio │ │ │ │ │ + * {Number} The fraction of map width/height by which we'll pan the map │ │ │ │ │ + * on clicking the arrow buttons. Default is null. If set, will │ │ │ │ │ + * override <slideFactor>. E.g. if slideRatio is .5, then the Pan Up │ │ │ │ │ + * button will pan up half the map height. │ │ │ │ │ + */ │ │ │ │ │ + slideRatio: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: buttons │ │ │ │ │ + * {Array(DOMElement)} Array of Button Divs │ │ │ │ │ + */ │ │ │ │ │ + buttons: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: position │ │ │ │ │ + * {<OpenLayers.Pixel>} │ │ │ │ │ + */ │ │ │ │ │ + position: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.NavToolbar │ │ │ │ │ - * Add our two mousedefaults controls. │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Control.PanZoom │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be used │ │ │ │ │ - * to extend the control. │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.addControls([ │ │ │ │ │ - new OpenLayers.Control.Navigation(), │ │ │ │ │ - new OpenLayers.Control.ZoomBox() │ │ │ │ │ - ]); │ │ │ │ │ + this.position = new OpenLayers.Pixel(OpenLayers.Control.PanZoom.X, │ │ │ │ │ + OpenLayers.Control.PanZoom.Y); │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * calls the default draw, and then activates mouse defaults. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - var div = OpenLayers.Control.Panel.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (this.defaultControl === null) { │ │ │ │ │ - this.defaultControl = this.controls[0]; │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ } │ │ │ │ │ - return div; │ │ │ │ │ + this.removeButtons(); │ │ │ │ │ + this.buttons = null; │ │ │ │ │ + this.position = null; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.NavToolbar" │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * │ │ │ │ │ + * Properties: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A reference to the container div for the PanZoom control. │ │ │ │ │ + */ │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + // initialize our internal div │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + px = this.position; │ │ │ │ │ + │ │ │ │ │ + // place the controls │ │ │ │ │ + this.buttons = []; │ │ │ │ │ + │ │ │ │ │ + var sz = { │ │ │ │ │ + w: 18, │ │ │ │ │ + h: 18 │ │ │ │ │ + }; │ │ │ │ │ + var centered = new OpenLayers.Pixel(px.x + sz.w / 2, px.y); │ │ │ │ │ + │ │ │ │ │ + this._addButton("panup", "north-mini.png", centered, sz); │ │ │ │ │ + px.y = centered.y + sz.h; │ │ │ │ │ + this._addButton("panleft", "west-mini.png", px, sz); │ │ │ │ │ + this._addButton("panright", "east-mini.png", px.add(sz.w, 0), sz); │ │ │ │ │ + this._addButton("pandown", "south-mini.png", │ │ │ │ │ + centered.add(0, sz.h * 2), sz); │ │ │ │ │ + this._addButton("zoomin", "zoom-plus-mini.png", │ │ │ │ │ + centered.add(0, sz.h * 3 + 5), sz); │ │ │ │ │ + this._addButton("zoomworld", "zoom-world-mini.png", │ │ │ │ │ + centered.add(0, sz.h * 4 + 5), sz); │ │ │ │ │ + this._addButton("zoomout", "zoom-minus-mini.png", │ │ │ │ │ + centered.add(0, sz.h * 5 + 5), sz); │ │ │ │ │ + return this.div; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: _addButton │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} │ │ │ │ │ + * img - {String} │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} │ │ │ │ │ + * sz - {<OpenLayers.Size>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A Div (an alphaImageDiv, to be precise) that contains the │ │ │ │ │ + * image of the button, and has all the proper event handlers set. │ │ │ │ │ + */ │ │ │ │ │ + _addButton: function(id, img, xy, sz) { │ │ │ │ │ + var imgLocation = OpenLayers.Util.getImageLocation(img); │ │ │ │ │ + var btn = OpenLayers.Util.createAlphaImageDiv( │ │ │ │ │ + this.id + "_" + id, │ │ │ │ │ + xy, sz, imgLocation, "absolute"); │ │ │ │ │ + btn.style.cursor = "pointer"; │ │ │ │ │ + //we want to add the outer div │ │ │ │ │ + this.div.appendChild(btn); │ │ │ │ │ + btn.action = id; │ │ │ │ │ + btn.className = "olButton"; │ │ │ │ │ + │ │ │ │ │ + //we want to remember/reference the outer div │ │ │ │ │ + this.buttons.push(btn); │ │ │ │ │ + return btn; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: _removeButton │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * btn - {Object} │ │ │ │ │ + */ │ │ │ │ │ + _removeButton: function(btn) { │ │ │ │ │ + this.div.removeChild(btn); │ │ │ │ │ + OpenLayers.Util.removeItem(this.buttons, btn); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeButtons │ │ │ │ │ + */ │ │ │ │ │ + removeButtons: function() { │ │ │ │ │ + for (var i = this.buttons.length - 1; i >= 0; --i) { │ │ │ │ │ + this._removeButton(this.buttons[i]); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: onButtonClick │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + var btn = evt.buttonElement; │ │ │ │ │ + switch (btn.action) { │ │ │ │ │ + case "panup": │ │ │ │ │ + this.map.pan(0, -this.getSlideFactor("h")); │ │ │ │ │ + break; │ │ │ │ │ + case "pandown": │ │ │ │ │ + this.map.pan(0, this.getSlideFactor("h")); │ │ │ │ │ + break; │ │ │ │ │ + case "panleft": │ │ │ │ │ + this.map.pan(-this.getSlideFactor("w"), 0); │ │ │ │ │ + break; │ │ │ │ │ + case "panright": │ │ │ │ │ + this.map.pan(this.getSlideFactor("w"), 0); │ │ │ │ │ + break; │ │ │ │ │ + case "zoomin": │ │ │ │ │ + this.map.zoomIn(); │ │ │ │ │ + break; │ │ │ │ │ + case "zoomout": │ │ │ │ │ + this.map.zoomOut(); │ │ │ │ │ + break; │ │ │ │ │ + case "zoomworld": │ │ │ │ │ + this.map.zoomToMaxExtent(); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getSlideFactor │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * dim - {String} "w" or "h" (for width or height). │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Number} The slide factor for panning in the requested direction. │ │ │ │ │ + */ │ │ │ │ │ + getSlideFactor: function(dim) { │ │ │ │ │ + return this.slideRatio ? │ │ │ │ │ + this.map.getSize()[dim] * this.slideRatio : │ │ │ │ │ + this.slideFactor; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.PanZoom" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: X │ │ │ │ │ + * {Integer} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.PanZoom.X = 4; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: Y │ │ │ │ │ + * {Integer} │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.PanZoom.Y = 4; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/CacheRead.js │ │ │ │ │ + OpenLayers/Control/WMSGetFeatureInfo.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Handler/Click.js │ │ │ │ │ + * @requires OpenLayers/Handler/Hover.js │ │ │ │ │ + * @requires OpenLayers/Request.js │ │ │ │ │ + * @requires OpenLayers/Format/WMSGetFeatureInfo.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.CacheRead │ │ │ │ │ - * A control for using image tiles cached with <OpenLayers.Control.CacheWrite> │ │ │ │ │ - * from the browser's local storage. │ │ │ │ │ + * Class: OpenLayers.Control.WMSGetFeatureInfo │ │ │ │ │ + * The WMSGetFeatureInfo control uses a WMS query to get information about a point on the map. The │ │ │ │ │ + * information may be in a display-friendly format such as HTML, or a machine-friendly format such │ │ │ │ │ + * as GML, depending on the server's capabilities and the client's configuration. This control │ │ │ │ │ + * handles click or hover events, attempts to parse the results using an OpenLayers.Format, and │ │ │ │ │ + * fires a 'getfeatureinfo' event with the click position, the raw body of the response, and an │ │ │ │ │ + * array of features if it successfully read the response. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.CacheRead = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Control.WMSGetFeatureInfo = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: fetchEvent │ │ │ │ │ - * {String} The layer event to listen to for replacing remote resource tile │ │ │ │ │ - * URLs with cached data URIs. Supported values are "tileerror" (try │ │ │ │ │ - * remote first, fall back to cached) and "tileloadstart" (try cache │ │ │ │ │ - * first, fall back to remote). Default is "tileloadstart". │ │ │ │ │ - * │ │ │ │ │ - * Note that "tileerror" will not work for CORS enabled images (see │ │ │ │ │ - * https://developer.mozilla.org/en/CORS_Enabled_Image), i.e. layers │ │ │ │ │ - * configured with a <OpenLayers.Tile.Image.crossOriginKeyword> in │ │ │ │ │ - * <OpenLayers.Layer.Grid.tileOptions>. │ │ │ │ │ + * APIProperty: hover │ │ │ │ │ + * {Boolean} Send GetFeatureInfo requests when mouse stops moving. │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - fetchEvent: "tileloadstart", │ │ │ │ │ + hover: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: drillDown │ │ │ │ │ + * {Boolean} Drill down over all WMS layers in the map. When │ │ │ │ │ + * using drillDown mode, hover is not possible, and an infoFormat that │ │ │ │ │ + * returns parseable features is required. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + drillDown: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxFeatures │ │ │ │ │ + * {Integer} Maximum number of features to return from a WMS query. This │ │ │ │ │ + * sets the feature_count parameter on WMS GetFeatureInfo │ │ │ │ │ + * requests. │ │ │ │ │ + */ │ │ │ │ │ + maxFeatures: 10, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: clickCallback │ │ │ │ │ + * {String} The click callback to register in the │ │ │ │ │ + * {<OpenLayers.Handler.Click>} object created when the hover │ │ │ │ │ + * option is set to false. Default is "click". │ │ │ │ │ + */ │ │ │ │ │ + clickCallback: "click", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: output │ │ │ │ │ + * {String} Either "features" or "object". When triggering a getfeatureinfo │ │ │ │ │ + * request should we pass on an array of features or an object with with │ │ │ │ │ + * a "features" property and other properties (such as the url of the │ │ │ │ │ + * WMS). Default is "features". │ │ │ │ │ + */ │ │ │ │ │ + output: "features", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIProperty: layers │ │ │ │ │ - * {Array(<OpenLayers.Layer.Grid>)}. Optional. If provided, only these │ │ │ │ │ - * layers will receive tiles from the cache. │ │ │ │ │ + * {Array(<OpenLayers.Layer.WMS>)} The layers to query for feature info. │ │ │ │ │ + * If omitted, all map WMS layers with a url that matches this <url> or │ │ │ │ │ + * <layerUrls> will be considered. │ │ │ │ │ */ │ │ │ │ │ layers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ + * APIProperty: queryVisible │ │ │ │ │ + * {Boolean} If true, filter out hidden layers when searching the map for │ │ │ │ │ + * layers to query. Default is false. │ │ │ │ │ */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ + queryVisible: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.CacheRead │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String} The URL of the WMS service to use. If not provided, the url │ │ │ │ │ + * of the first eligible layer will be used. │ │ │ │ │ + */ │ │ │ │ │ + url: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layerUrls │ │ │ │ │ + * {Array(String)} Optional list of urls for layers that should be queried. │ │ │ │ │ + * This can be used when the layer url differs from the url used for │ │ │ │ │ + * making GetFeatureInfo requests (in the case of a layer using cached │ │ │ │ │ + * tiles). │ │ │ │ │ + */ │ │ │ │ │ + layerUrls: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: infoFormat │ │ │ │ │ + * {String} The mimetype to request from the server. If you are using │ │ │ │ │ + * drillDown mode and have multiple servers that do not share a common │ │ │ │ │ + * infoFormat, you can override the control's infoFormat by providing an │ │ │ │ │ + * INFO_FORMAT parameter in your <OpenLayers.Layer.WMS> instance(s). │ │ │ │ │ + */ │ │ │ │ │ + infoFormat: 'text/html', │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: vendorParams │ │ │ │ │ + * {Object} Additional parameters that will be added to the request, for │ │ │ │ │ + * WMS implementations that support them. This could e.g. look like │ │ │ │ │ + * (start code) │ │ │ │ │ + * { │ │ │ │ │ + * radius: 5 │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ + vendorParams: {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: format │ │ │ │ │ + * {<OpenLayers.Format>} A format for parsing GetFeatureInfo responses. │ │ │ │ │ + * Default is <OpenLayers.Format.WMSGetFeatureInfo>. │ │ │ │ │ + */ │ │ │ │ │ + format: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: formatOptions │ │ │ │ │ + * {Object} Optional properties to set on the format (if one is not provided │ │ │ │ │ + * in the <format> property. │ │ │ │ │ + */ │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: handlerOptions │ │ │ │ │ + * {Object} Additional options for the handlers used by this control, e.g. │ │ │ │ │ + * (start code) │ │ │ │ │ + * { │ │ │ │ │ + * "click": {delay: 100}, │ │ │ │ │ + * "hover": {delay: 300} │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: handler │ │ │ │ │ + * {Object} Reference to the <OpenLayers.Handler> for this control │ │ │ │ │ + */ │ │ │ │ │ + handler: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: hoverRequest │ │ │ │ │ + * {<OpenLayers.Request>} contains the currently running hover request │ │ │ │ │ + * (if any). │ │ │ │ │ + */ │ │ │ │ │ + hoverRequest: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ + * beforegetfeatureinfo - Triggered before the request is sent. │ │ │ │ │ + * The event object has an *xy* property with the position of the │ │ │ │ │ + * mouse click or hover event that triggers the request. │ │ │ │ │ + * nogetfeatureinfo - no queryable layers were found. │ │ │ │ │ + * getfeatureinfo - Triggered when a GetFeatureInfo response is received. │ │ │ │ │ + * The event object has a *text* property with the body of the │ │ │ │ │ + * response (String), a *features* property with an array of the │ │ │ │ │ + * parsed features, an *xy* property with the position of the mouse │ │ │ │ │ + * click or hover event that triggered the request, and a *request* │ │ │ │ │ + * property with the request itself. If drillDown is set to true and │ │ │ │ │ + * multiple requests were issued to collect feature info from all │ │ │ │ │ + * layers, *text* and *request* will only contain the response body │ │ │ │ │ + * and request object of the last request. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: <OpenLayers.Control.WMSGetFeatureInfo> │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Object with API properties for this control │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the control. │ │ │ │ │ - * │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + │ │ │ │ │ + if (!this.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.WMSGetFeatureInfo( │ │ │ │ │ + options.formatOptions │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.drillDown === true) { │ │ │ │ │ + this.hover = false; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Hover( │ │ │ │ │ + this, { │ │ │ │ │ + 'move': this.cancelHover, │ │ │ │ │ + 'pause': this.getInfoForHover │ │ │ │ │ + }, │ │ │ │ │ + OpenLayers.Util.extend(this.handlerOptions.hover || {}, { │ │ │ │ │ + 'delay': 250 │ │ │ │ │ + })); │ │ │ │ │ + } else { │ │ │ │ │ + var callbacks = {}; │ │ │ │ │ + callbacks[this.clickCallback] = this.getInfoForClick; │ │ │ │ │ + this.handler = new OpenLayers.Handler.Click( │ │ │ │ │ + this, callbacks, this.handlerOptions.click || {}); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getInfoForClick │ │ │ │ │ + * Called on click │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - var i, layers = this.layers || map.layers; │ │ │ │ │ - for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - this.addLayer({ │ │ │ │ │ - layer: layers[i] │ │ │ │ │ - }); │ │ │ │ │ + getInfoForClick: function(evt) { │ │ │ │ │ + this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ + xy: evt.xy │ │ │ │ │ + }); │ │ │ │ │ + // Set the cursor to "wait" to tell the user we're working on their │ │ │ │ │ + // click. │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + this.request(evt.xy, {}); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getInfoForHover │ │ │ │ │ + * Pause callback for the hover handler │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} │ │ │ │ │ + */ │ │ │ │ │ + getInfoForHover: function(evt) { │ │ │ │ │ + this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ + xy: evt.xy │ │ │ │ │ + }); │ │ │ │ │ + this.request(evt.xy, { │ │ │ │ │ + hover: true │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: cancelHover │ │ │ │ │ + * Cancel callback for the hover handler │ │ │ │ │ + */ │ │ │ │ │ + cancelHover: function() { │ │ │ │ │ + if (this.hoverRequest) { │ │ │ │ │ + this.hoverRequest.abort(); │ │ │ │ │ + this.hoverRequest = null; │ │ │ │ │ } │ │ │ │ │ - if (!this.layers) { │ │ │ │ │ - map.events.on({ │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - removeLayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: findLayers │ │ │ │ │ + * Internal method to get the layers, independent of whether we are │ │ │ │ │ + * inspecting the map or using a client-provided array │ │ │ │ │ + */ │ │ │ │ │ + findLayers: function() { │ │ │ │ │ + │ │ │ │ │ + var candidates = this.layers || this.map.layers; │ │ │ │ │ + var layers = []; │ │ │ │ │ + var layer, url; │ │ │ │ │ + for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = candidates[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.WMS && │ │ │ │ │ + (!this.queryVisible || layer.getVisibility())) { │ │ │ │ │ + url = OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url; │ │ │ │ │ + // if the control was not configured with a url, set it │ │ │ │ │ + // to the first layer url │ │ │ │ │ + if (this.drillDown === false && !this.url) { │ │ │ │ │ + this.url = url; │ │ │ │ │ + } │ │ │ │ │ + if (this.drillDown === true || this.urlMatches(url)) { │ │ │ │ │ + layers.push(layer); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return layers; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: addLayer │ │ │ │ │ - * Adds a layer to the control. Once added, tiles requested for this layer │ │ │ │ │ - * will be cached. │ │ │ │ │ + * Method: urlMatches │ │ │ │ │ + * Test to see if the provided url matches either the control <url> or one │ │ │ │ │ + * of the <layerUrls>. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} Object with a layer property referencing an │ │ │ │ │ - * <OpenLayers.Layer> instance │ │ │ │ │ + * url - {String} The url to test. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The provided url matches the control <url> or one of the │ │ │ │ │ + * <layerUrls>. │ │ │ │ │ */ │ │ │ │ │ - addLayer: function(evt) { │ │ │ │ │ - evt.layer.events.register(this.fetchEvent, this, this.fetch); │ │ │ │ │ + urlMatches: function(url) { │ │ │ │ │ + var matches = OpenLayers.Util.isEquivalentUrl(this.url, url); │ │ │ │ │ + if (!matches && this.layerUrls) { │ │ │ │ │ + for (var i = 0, len = this.layerUrls.length; i < len; ++i) { │ │ │ │ │ + if (OpenLayers.Util.isEquivalentUrl(this.layerUrls[i], url)) { │ │ │ │ │ + matches = true; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return matches; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: removeLayer │ │ │ │ │ - * Removes a layer from the control. Once removed, tiles requested for this │ │ │ │ │ - * layer will no longer be cached. │ │ │ │ │ + * Method: buildWMSOptions │ │ │ │ │ + * Build an object with the relevant WMS options for the GetFeatureInfo request │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} Object with a layer property referencing an │ │ │ │ │ - * <OpenLayers.Layer> instance │ │ │ │ │ + * url - {String} The url to be used for sending the request │ │ │ │ │ + * layers - {Array(<OpenLayers.Layer.WMS)} An array of layers │ │ │ │ │ + * clickPosition - {<OpenLayers.Pixel>} The position on the map where the mouse │ │ │ │ │ + * event occurred. │ │ │ │ │ + * format - {String} The format from the corresponding GetMap request │ │ │ │ │ */ │ │ │ │ │ - removeLayer: function(evt) { │ │ │ │ │ - evt.layer.events.unregister(this.fetchEvent, this, this.fetch); │ │ │ │ │ + buildWMSOptions: function(url, layers, clickPosition, format) { │ │ │ │ │ + var layerNames = [], │ │ │ │ │ + styleNames = []; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + if (layers[i].params.LAYERS != null) { │ │ │ │ │ + layerNames = layerNames.concat(layers[i].params.LAYERS); │ │ │ │ │ + styleNames = styleNames.concat(this.getStyleNames(layers[i])); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var firstLayer = layers[0]; │ │ │ │ │ + // use the firstLayer's projection if it matches the map projection - │ │ │ │ │ + // this assumes that all layers will be available in this projection │ │ │ │ │ + var projection = this.map.getProjection(); │ │ │ │ │ + var layerProj = firstLayer.projection; │ │ │ │ │ + if (layerProj && layerProj.equals(this.map.getProjectionObject())) { │ │ │ │ │ + projection = layerProj.getCode(); │ │ │ │ │ + } │ │ │ │ │ + var params = OpenLayers.Util.extend({ │ │ │ │ │ + service: "WMS", │ │ │ │ │ + version: firstLayer.params.VERSION, │ │ │ │ │ + request: "GetFeatureInfo", │ │ │ │ │ + exceptions: firstLayer.params.EXCEPTIONS, │ │ │ │ │ + bbox: this.map.getExtent().toBBOX(null, │ │ │ │ │ + firstLayer.reverseAxisOrder()), │ │ │ │ │ + feature_count: this.maxFeatures, │ │ │ │ │ + height: this.map.getSize().h, │ │ │ │ │ + width: this.map.getSize().w, │ │ │ │ │ + format: format, │ │ │ │ │ + info_format: firstLayer.params.INFO_FORMAT || this.infoFormat │ │ │ │ │ + }, (parseFloat(firstLayer.params.VERSION) >= 1.3) ? { │ │ │ │ │ + crs: projection, │ │ │ │ │ + i: parseInt(clickPosition.x), │ │ │ │ │ + j: parseInt(clickPosition.y) │ │ │ │ │ + } : { │ │ │ │ │ + srs: projection, │ │ │ │ │ + x: parseInt(clickPosition.x), │ │ │ │ │ + y: parseInt(clickPosition.y) │ │ │ │ │ + }); │ │ │ │ │ + if (layerNames.length != 0) { │ │ │ │ │ + params = OpenLayers.Util.extend({ │ │ │ │ │ + layers: layerNames, │ │ │ │ │ + query_layers: layerNames, │ │ │ │ │ + styles: styleNames │ │ │ │ │ + }, params); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.applyDefaults(params, this.vendorParams); │ │ │ │ │ + return { │ │ │ │ │ + url: url, │ │ │ │ │ + params: OpenLayers.Util.upperCaseObject(params), │ │ │ │ │ + callback: function(request) { │ │ │ │ │ + this.handleResponse(clickPosition, request, url); │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: fetch │ │ │ │ │ - * Listener to the <fetchEvent> event. Replaces a tile's url with a data │ │ │ │ │ - * URI from the cache. │ │ │ │ │ + * Method: getStyleNames │ │ │ │ │ + * Gets the STYLES parameter for the layer. Make sure the STYLES parameter │ │ │ │ │ + * matches the LAYERS parameter │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} Event object with a tile property. │ │ │ │ │ + * layer - {<OpenLayers.Layer.WMS>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(String)} The STYLES parameter │ │ │ │ │ */ │ │ │ │ │ - fetch: function(evt) { │ │ │ │ │ - if (this.active && window.localStorage && │ │ │ │ │ - evt.tile instanceof OpenLayers.Tile.Image) { │ │ │ │ │ - var tile = evt.tile, │ │ │ │ │ - url = tile.url; │ │ │ │ │ - // deal with modified tile urls when both CacheWrite and CacheRead │ │ │ │ │ - // are active │ │ │ │ │ - if (!tile.layer.crossOriginKeyword && OpenLayers.ProxyHost && │ │ │ │ │ - url.indexOf(OpenLayers.ProxyHost) === 0) { │ │ │ │ │ - url = OpenLayers.Control.CacheWrite.urlMap[url]; │ │ │ │ │ + getStyleNames: function(layer) { │ │ │ │ │ + // in the event of a WMS layer bundling multiple layers but not │ │ │ │ │ + // specifying styles,we need the same number of commas to specify │ │ │ │ │ + // the default style for each of the layers. We can't just leave it │ │ │ │ │ + // blank as we may be including other layers that do specify styles. │ │ │ │ │ + var styleNames; │ │ │ │ │ + if (layer.params.STYLES) { │ │ │ │ │ + styleNames = layer.params.STYLES; │ │ │ │ │ + } else { │ │ │ │ │ + if (OpenLayers.Util.isArray(layer.params.LAYERS)) { │ │ │ │ │ + styleNames = new Array(layer.params.LAYERS.length); │ │ │ │ │ + } else { // Assume it's a String │ │ │ │ │ + styleNames = layer.params.LAYERS.replace(/[^,]/g, ""); │ │ │ │ │ } │ │ │ │ │ - var dataURI = window.localStorage.getItem("olCache_" + url); │ │ │ │ │ - if (dataURI) { │ │ │ │ │ - tile.url = dataURI; │ │ │ │ │ - if (evt.type === "tileerror") { │ │ │ │ │ - tile.setImgSrc(dataURI); │ │ │ │ │ + } │ │ │ │ │ + return styleNames; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: request │ │ │ │ │ + * Sends a GetFeatureInfo request to the WMS │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * clickPosition - {<OpenLayers.Pixel>} The position on the map where the │ │ │ │ │ + * mouse event occurred. │ │ │ │ │ + * options - {Object} additional options for this method. │ │ │ │ │ + * │ │ │ │ │ + * Valid options: │ │ │ │ │ + * - *hover* {Boolean} true if we do the request for the hover handler │ │ │ │ │ + */ │ │ │ │ │ + request: function(clickPosition, options) { │ │ │ │ │ + var layers = this.findLayers(); │ │ │ │ │ + if (layers.length == 0) { │ │ │ │ │ + this.events.triggerEvent("nogetfeatureinfo"); │ │ │ │ │ + // Reset the cursor. │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + options = options || {}; │ │ │ │ │ + if (this.drillDown === false) { │ │ │ │ │ + var wmsOptions = this.buildWMSOptions(this.url, layers, │ │ │ │ │ + clickPosition, layers[0].params.FORMAT); │ │ │ │ │ + var request = OpenLayers.Request.GET(wmsOptions); │ │ │ │ │ + │ │ │ │ │ + if (options.hover === true) { │ │ │ │ │ + this.hoverRequest = request; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this._requestCount = 0; │ │ │ │ │ + this._numRequests = 0; │ │ │ │ │ + this.features = []; │ │ │ │ │ + // group according to service url to combine requests │ │ │ │ │ + var services = {}, │ │ │ │ │ + url; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + var layer = layers[i]; │ │ │ │ │ + var service, found = false; │ │ │ │ │ + url = OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url; │ │ │ │ │ + if (url in services) { │ │ │ │ │ + services[url].push(layer); │ │ │ │ │ + } else { │ │ │ │ │ + this._numRequests++; │ │ │ │ │ + services[url] = [layer]; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + var layers; │ │ │ │ │ + for (var url in services) { │ │ │ │ │ + layers = services[url]; │ │ │ │ │ + var wmsOptions = this.buildWMSOptions(url, layers, │ │ │ │ │ + clickPosition, layers[0].params.FORMAT); │ │ │ │ │ + OpenLayers.Request.GET(wmsOptions); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * The destroy method is used to perform any clean up before the control │ │ │ │ │ - * is dereferenced. Typically this is where event listeners are removed │ │ │ │ │ - * to prevent memory leaks. │ │ │ │ │ + * Method: triggerGetFeatureInfo │ │ │ │ │ + * Trigger the getfeatureinfo event when all is done │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * request - {XMLHttpRequest} The request object │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} The position on the map where the │ │ │ │ │ + * mouse event occurred. │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} or │ │ │ │ │ + * {Array({Object}) when output is "object". The object has a url and a │ │ │ │ │ + * features property which contains an array of features. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.layers || this.map) { │ │ │ │ │ - var i, layers = this.layers || this.map.layers; │ │ │ │ │ - for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - this.removeLayer({ │ │ │ │ │ - layer: layers[i] │ │ │ │ │ + triggerGetFeatureInfo: function(request, xy, features) { │ │ │ │ │ + this.events.triggerEvent("getfeatureinfo", { │ │ │ │ │ + text: request.responseText, │ │ │ │ │ + features: features, │ │ │ │ │ + request: request, │ │ │ │ │ + xy: xy │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + // Reset the cursor. │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleResponse │ │ │ │ │ + * Handler for the GetFeatureInfo response. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * xy - {<OpenLayers.Pixel>} The position on the map where the │ │ │ │ │ + * mouse event occurred. │ │ │ │ │ + * request - {XMLHttpRequest} The request object. │ │ │ │ │ + * url - {String} The url which was used for this request. │ │ │ │ │ + */ │ │ │ │ │ + handleResponse: function(xy, request, url) { │ │ │ │ │ + │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText; │ │ │ │ │ + } │ │ │ │ │ + var features = this.format.read(doc); │ │ │ │ │ + if (this.drillDown === false) { │ │ │ │ │ + this.triggerGetFeatureInfo(request, xy, features); │ │ │ │ │ + } else { │ │ │ │ │ + this._requestCount++; │ │ │ │ │ + if (this.output === "object") { │ │ │ │ │ + this._features = (this._features || []).concat({ │ │ │ │ │ + url: url, │ │ │ │ │ + features: features │ │ │ │ │ }); │ │ │ │ │ + } else { │ │ │ │ │ + this._features = (this._features || []).concat(features); │ │ │ │ │ + } │ │ │ │ │ + if (this._requestCount === this._numRequests) { │ │ │ │ │ + this.triggerGetFeatureInfo(request, xy, this._features.concat()); │ │ │ │ │ + delete this._features; │ │ │ │ │ + delete this._requestCount; │ │ │ │ │ + delete this._numRequests; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - removeLayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.CacheRead" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.WMSGetFeatureInfo" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Control/ModifyFeature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ @@ -76005,14 +75909,190 @@ │ │ │ │ │ OpenLayers.Control.ModifyFeature.ROTATE = 4; │ │ │ │ │ /** │ │ │ │ │ * Constant: DRAG │ │ │ │ │ * {Integer} Constant used to make the control work in drag mode │ │ │ │ │ */ │ │ │ │ │ OpenLayers.Control.ModifyFeature.DRAG = 8; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ + OpenLayers/Control/ZoomIn.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control/Button.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.ZoomIn │ │ │ │ │ + * The ZoomIn control is a button to increase the zoom level of a map. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.ZoomIn = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: trigger │ │ │ │ │ + */ │ │ │ │ │ + trigger: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.zoomIn(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomIn" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/Zoom.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.Zoom │ │ │ │ │ + * The Zoom control is a pair of +/- links for zooming in and out. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.Zoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomInText │ │ │ │ │ + * {String} │ │ │ │ │ + * Text for zoom-in link. Default is "+". │ │ │ │ │ + */ │ │ │ │ │ + zoomInText: "+", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomInId │ │ │ │ │ + * {String} │ │ │ │ │ + * Instead of having the control create a zoom in link, you can provide │ │ │ │ │ + * the identifier for an anchor element already added to the document. │ │ │ │ │ + * By default, an element with id "olZoomInLink" will be searched for │ │ │ │ │ + * and used if it exists. │ │ │ │ │ + */ │ │ │ │ │ + zoomInId: "olZoomInLink", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomOutText │ │ │ │ │ + * {String} │ │ │ │ │ + * Text for zoom-out link. Default is "\u2212". │ │ │ │ │ + */ │ │ │ │ │ + zoomOutText: "\u2212", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomOutId │ │ │ │ │ + * {String} │ │ │ │ │ + * Instead of having the control create a zoom out link, you can provide │ │ │ │ │ + * the identifier for an anchor element already added to the document. │ │ │ │ │ + * By default, an element with id "olZoomOutLink" will be searched for │ │ │ │ │ + * and used if it exists. │ │ │ │ │ + */ │ │ │ │ │ + zoomOutId: "olZoomOutLink", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A reference to the DOMElement containing the zoom links. │ │ │ │ │ + */ │ │ │ │ │ + draw: function() { │ │ │ │ │ + var div = OpenLayers.Control.prototype.draw.apply(this), │ │ │ │ │ + links = this.getOrCreateLinks(div), │ │ │ │ │ + zoomIn = links.zoomIn, │ │ │ │ │ + zoomOut = links.zoomOut, │ │ │ │ │ + eventsInstance = this.map.events; │ │ │ │ │ + │ │ │ │ │ + if (zoomOut.parentNode !== div) { │ │ │ │ │ + eventsInstance = this.events; │ │ │ │ │ + eventsInstance.attachToElement(zoomOut.parentNode); │ │ │ │ │ + } │ │ │ │ │ + eventsInstance.register("buttonclick", this, this.onZoomClick); │ │ │ │ │ + │ │ │ │ │ + this.zoomInLink = zoomIn; │ │ │ │ │ + this.zoomOutLink = zoomOut; │ │ │ │ │ + return div; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getOrCreateLinks │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * el - {DOMElement} │ │ │ │ │ + * │ │ │ │ │ + * Return: │ │ │ │ │ + * {Object} Object with zoomIn and zoomOut properties referencing links. │ │ │ │ │ + */ │ │ │ │ │ + getOrCreateLinks: function(el) { │ │ │ │ │ + var zoomIn = document.getElementById(this.zoomInId), │ │ │ │ │ + zoomOut = document.getElementById(this.zoomOutId); │ │ │ │ │ + if (!zoomIn) { │ │ │ │ │ + zoomIn = document.createElement("a"); │ │ │ │ │ + zoomIn.href = "#zoomIn"; │ │ │ │ │ + zoomIn.appendChild(document.createTextNode(this.zoomInText)); │ │ │ │ │ + zoomIn.className = "olControlZoomIn"; │ │ │ │ │ + el.appendChild(zoomIn); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.addClass(zoomIn, "olButton"); │ │ │ │ │ + if (!zoomOut) { │ │ │ │ │ + zoomOut = document.createElement("a"); │ │ │ │ │ + zoomOut.href = "#zoomOut"; │ │ │ │ │ + zoomOut.appendChild(document.createTextNode(this.zoomOutText)); │ │ │ │ │ + zoomOut.className = "olControlZoomOut"; │ │ │ │ │ + el.appendChild(zoomOut); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.addClass(zoomOut, "olButton"); │ │ │ │ │ + return { │ │ │ │ │ + zoomIn: zoomIn, │ │ │ │ │ + zoomOut: zoomOut │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: onZoomClick │ │ │ │ │ + * Called when zoomin/out link is clicked. │ │ │ │ │ + */ │ │ │ │ │ + onZoomClick: function(evt) { │ │ │ │ │ + var button = evt.buttonElement; │ │ │ │ │ + if (button === this.zoomInLink) { │ │ │ │ │ + this.map.zoomIn(); │ │ │ │ │ + } else if (button === this.zoomOutLink) { │ │ │ │ │ + this.map.zoomOut(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Clean up. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onZoomClick); │ │ │ │ │ + } │ │ │ │ │ + delete this.zoomInLink; │ │ │ │ │ + delete this.zoomOutLink; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Zoom" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ OpenLayers/Control/DrawFeature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -76251,957 +76331,274 @@ │ │ │ │ │ cancel: function() { │ │ │ │ │ this.handler.cancel(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.DrawFeature" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/ArgParser.js │ │ │ │ │ + OpenLayers/Control/Permalink.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Control/ArgParser.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.ArgParser │ │ │ │ │ - * The ArgParser control adds location bar query string parsing functionality │ │ │ │ │ - * to an OpenLayers Map. │ │ │ │ │ - * When added to a Map control, on a page load/refresh, the Map will │ │ │ │ │ - * automatically take the href string and parse it for lon, lat, zoom, and │ │ │ │ │ - * layers information. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Control.Permalink │ │ │ │ │ + * The Permalink control is hyperlink that will return the user to the │ │ │ │ │ + * current map view. By default it is drawn in the lower right corner of the │ │ │ │ │ + * map. The href is updated as the map is zoomed, panned and whilst layers │ │ │ │ │ + * are switched. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.ArgParser = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: center │ │ │ │ │ - * {<OpenLayers.LonLat>} │ │ │ │ │ - */ │ │ │ │ │ - center: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: zoom │ │ │ │ │ - * {int} │ │ │ │ │ - */ │ │ │ │ │ - zoom: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layers │ │ │ │ │ - * {String} Each character represents the state of the corresponding layer │ │ │ │ │ - * on the map. │ │ │ │ │ - */ │ │ │ │ │ - layers: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: displayProjection │ │ │ │ │ - * {<OpenLayers.Projection>} Requires proj4js support. │ │ │ │ │ - * Projection used when reading the coordinates from the URL. This will │ │ │ │ │ - * reproject the map coordinates from the URL into the map's │ │ │ │ │ - * projection. │ │ │ │ │ - * │ │ │ │ │ - * If you are using this functionality, be aware that any permalink │ │ │ │ │ - * which is added to the map will determine the coordinate type which │ │ │ │ │ - * is read from the URL, which means you should not add permalinks with │ │ │ │ │ - * different displayProjections to the same map. │ │ │ │ │ - */ │ │ │ │ │ - displayProjection: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.ArgParser │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getParameters │ │ │ │ │ - */ │ │ │ │ │ - getParameters: function(url) { │ │ │ │ │ - url = url || window.location.href; │ │ │ │ │ - var parameters = OpenLayers.Util.getParameters(url); │ │ │ │ │ - │ │ │ │ │ - // If we have an anchor in the url use it to split the url │ │ │ │ │ - var index = url.indexOf('#'); │ │ │ │ │ - if (index > 0) { │ │ │ │ │ - // create an url to parse on the getParameters │ │ │ │ │ - url = '?' + url.substring(index + 1, url.length); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Util.extend(parameters, │ │ │ │ │ - OpenLayers.Util.getParameters(url)); │ │ │ │ │ - } │ │ │ │ │ - return parameters; │ │ │ │ │ - }, │ │ │ │ │ +OpenLayers.Control.Permalink = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the control. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - //make sure we dont already have an arg parser attached │ │ │ │ │ - for (var i = 0, len = this.map.controls.length; i < len; i++) { │ │ │ │ │ - var control = this.map.controls[i]; │ │ │ │ │ - if ((control != this) && │ │ │ │ │ - (control.CLASS_NAME == "OpenLayers.Control.ArgParser")) { │ │ │ │ │ - │ │ │ │ │ - // If a second argparser is added to the map, then we │ │ │ │ │ - // override the displayProjection to be the one added to the │ │ │ │ │ - // map. │ │ │ │ │ - if (control.displayProjection != this.displayProjection) { │ │ │ │ │ - this.displayProjection = control.displayProjection; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (i == this.map.controls.length) { │ │ │ │ │ - │ │ │ │ │ - var args = this.getParameters(); │ │ │ │ │ - // Be careful to set layer first, to not trigger unnecessary layer loads │ │ │ │ │ - if (args.layers) { │ │ │ │ │ - this.layers = args.layers; │ │ │ │ │ - │ │ │ │ │ - // when we add a new layer, set its visibility │ │ │ │ │ - this.map.events.register('addlayer', this, │ │ │ │ │ - this.configureLayers); │ │ │ │ │ - this.configureLayers(); │ │ │ │ │ - } │ │ │ │ │ - if (args.lat && args.lon) { │ │ │ │ │ - this.center = new OpenLayers.LonLat(parseFloat(args.lon), │ │ │ │ │ - parseFloat(args.lat)); │ │ │ │ │ - if (args.zoom) { │ │ │ │ │ - this.zoom = parseFloat(args.zoom); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // when we add a new baselayer to see when we can set the center │ │ │ │ │ - this.map.events.register('changebaselayer', this, │ │ │ │ │ - this.setCenter); │ │ │ │ │ - this.setCenter(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setCenter │ │ │ │ │ - * As soon as a baseLayer has been loaded, we center and zoom │ │ │ │ │ - * ...and remove the handler. │ │ │ │ │ + * APIProperty: argParserClass │ │ │ │ │ + * {Class} The ArgParser control class (not instance) to use with this │ │ │ │ │ + * control. │ │ │ │ │ */ │ │ │ │ │ - setCenter: function() { │ │ │ │ │ - │ │ │ │ │ - if (this.map.baseLayer) { │ │ │ │ │ - //dont need to listen for this one anymore │ │ │ │ │ - this.map.events.unregister('changebaselayer', this, │ │ │ │ │ - this.setCenter); │ │ │ │ │ - │ │ │ │ │ - if (this.displayProjection) { │ │ │ │ │ - this.center.transform(this.displayProjection, │ │ │ │ │ - this.map.getProjectionObject()); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this.map.setCenter(this.center, this.zoom); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + argParserClass: OpenLayers.Control.ArgParser, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: configureLayers │ │ │ │ │ - * As soon as all the layers are loaded, cycle through them and │ │ │ │ │ - * hide or show them. │ │ │ │ │ + * Property: element │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - configureLayers: function() { │ │ │ │ │ - │ │ │ │ │ - if (this.layers.length == this.map.layers.length) { │ │ │ │ │ - this.map.events.unregister('addlayer', this, this.configureLayers); │ │ │ │ │ - │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - var c = this.layers.charAt(i); │ │ │ │ │ - │ │ │ │ │ - if (c == "B") { │ │ │ │ │ - this.map.setBaseLayer(layer); │ │ │ │ │ - } else if ((c == "T") || (c == "F")) { │ │ │ │ │ - layer.setVisibility(c == "T"); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ArgParser" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Geolocate.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ - * @requires OpenLayers/Projection.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Geolocate │ │ │ │ │ - * The Geolocate control wraps w3c geolocation API into control that can be │ │ │ │ │ - * bound to a map, and generate events on location update │ │ │ │ │ - * │ │ │ │ │ - * To use this control requires to load the proj4js library if the projection │ │ │ │ │ - * of the map is not EPSG:4326 or EPSG:900913. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Geolocate = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + element: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * locationupdated - Triggered when browser return a new position. Listeners will │ │ │ │ │ - * receive an object with a 'position' property which is the browser.geolocation.position │ │ │ │ │ - * native object, as well as a 'point' property which is the location transformed in the │ │ │ │ │ - * current map projection. │ │ │ │ │ - * locationfailed - Triggered when geolocation has failed │ │ │ │ │ - * locationuncapable - Triggered when control is activated on a browser │ │ │ │ │ - * which doesn't support geolocation │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: geolocation │ │ │ │ │ - * {Object} The geolocation engine, as a property to be possibly mocked. │ │ │ │ │ - * This is set lazily to avoid a memory leak in IE9. │ │ │ │ │ - */ │ │ │ │ │ - geolocation: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: available │ │ │ │ │ - * {Boolean} The navigator.geolocation object is available. │ │ │ │ │ - */ │ │ │ │ │ - available: ('geolocation' in navigator), │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: bind │ │ │ │ │ - * {Boolean} If true, map center will be set on location update. │ │ │ │ │ - */ │ │ │ │ │ - bind: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: watch │ │ │ │ │ - * {Boolean} If true, position will be update regularly. │ │ │ │ │ - */ │ │ │ │ │ - watch: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geolocationOptions │ │ │ │ │ - * {Object} Options to pass to the navigator's geolocation API. See │ │ │ │ │ - * <http://dev.w3.org/geo/api/spec-source.html>. No specific │ │ │ │ │ - * option is passed to the geolocation API by default. │ │ │ │ │ - */ │ │ │ │ │ - geolocationOptions: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Geolocate │ │ │ │ │ - * Create a new control to deal with browser geolocation API │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Activates the control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The control was effectively activated. │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (this.available && !this.geolocation) { │ │ │ │ │ - // set lazily to avoid IE9 memory leak │ │ │ │ │ - this.geolocation = navigator.geolocation; │ │ │ │ │ - } │ │ │ │ │ - if (!this.geolocation) { │ │ │ │ │ - this.events.triggerEvent("locationuncapable"); │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - if (this.watch) { │ │ │ │ │ - this.watchId = this.geolocation.watchPosition( │ │ │ │ │ - OpenLayers.Function.bind(this.geolocate, this), │ │ │ │ │ - OpenLayers.Function.bind(this.failure, this), │ │ │ │ │ - this.geolocationOptions │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - this.getCurrentLocation(); │ │ │ │ │ - } │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Deactivates the control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The control was effectively deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active && this.watchId !== null) { │ │ │ │ │ - this.geolocation.clearWatch(this.watchId); │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Control.prototype.deactivate.apply( │ │ │ │ │ - this, arguments │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: geolocate │ │ │ │ │ - * Activates the control. │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - geolocate: function(position) { │ │ │ │ │ - var center = new OpenLayers.LonLat( │ │ │ │ │ - position.coords.longitude, │ │ │ │ │ - position.coords.latitude │ │ │ │ │ - ).transform( │ │ │ │ │ - new OpenLayers.Projection("EPSG:4326"), │ │ │ │ │ - this.map.getProjectionObject() │ │ │ │ │ - ); │ │ │ │ │ - if (this.bind) { │ │ │ │ │ - this.map.setCenter(center); │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("locationupdated", { │ │ │ │ │ - position: position, │ │ │ │ │ - point: new OpenLayers.Geometry.Point( │ │ │ │ │ - center.lon, center.lat │ │ │ │ │ - ) │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: getCurrentLocation │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Returns true if a event will be fired (successfull │ │ │ │ │ - * registration) │ │ │ │ │ - */ │ │ │ │ │ - getCurrentLocation: function() { │ │ │ │ │ - if (!this.active || this.watch) { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - this.geolocation.getCurrentPosition( │ │ │ │ │ - OpenLayers.Function.bind(this.geolocate, this), │ │ │ │ │ - OpenLayers.Function.bind(this.failure, this), │ │ │ │ │ - this.geolocationOptions │ │ │ │ │ - ); │ │ │ │ │ - return true; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: failure │ │ │ │ │ - * method called on browser's geolocation failure │ │ │ │ │ - * │ │ │ │ │ + * APIProperty: anchor │ │ │ │ │ + * {Boolean} This option changes 3 things: │ │ │ │ │ + * the character '#' is used in place of the character '?', │ │ │ │ │ + * the window.href is updated if no element is provided. │ │ │ │ │ + * When this option is set to true it's not recommend to provide │ │ │ │ │ + * a base without provide an element. │ │ │ │ │ */ │ │ │ │ │ - failure: function(error) { │ │ │ │ │ - this.events.triggerEvent("locationfailed", { │ │ │ │ │ - error: error │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Geolocate" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/CacheWrite.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Request.js │ │ │ │ │ - * @requires OpenLayers/Console.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.CacheWrite │ │ │ │ │ - * A control for caching image tiles in the browser's local storage. The │ │ │ │ │ - * <OpenLayers.Control.CacheRead> control is used to fetch and use the cached │ │ │ │ │ - * tile images. │ │ │ │ │ - * │ │ │ │ │ - * Note: Before using this control on any layer that is not your own, make sure │ │ │ │ │ - * that the terms of service of the tile provider allow local storage of tiles. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.CacheWrite = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + anchor: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * To register events in the constructor, configure <eventListeners>. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * cachefull - Triggered when the cache is full. Listeners receive an │ │ │ │ │ - * object with a tile property as first argument. The tile references │ │ │ │ │ - * the tile that couldn't be cached. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: eventListeners │ │ │ │ │ - * {Object} Object with event listeners, keyed by event name. An optional │ │ │ │ │ - * scope property defines the scope that listeners will be executed in. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: layers │ │ │ │ │ - * {Array(<OpenLayers.Layer.Grid>)}. Optional. If provided, caching │ │ │ │ │ - * will be enabled for these layers only, otherwise for all cacheable │ │ │ │ │ - * layers. │ │ │ │ │ - */ │ │ │ │ │ - layers: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: imageFormat │ │ │ │ │ - * {String} The image format used for caching. The default is "image/png". │ │ │ │ │ - * Supported formats depend on the user agent. If an unsupported │ │ │ │ │ - * <imageFormat> is provided, "image/png" will be used. For aerial │ │ │ │ │ - * imagery, "image/jpeg" is recommended. │ │ │ │ │ - */ │ │ │ │ │ - imageFormat: "image/png", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: quotaRegEx │ │ │ │ │ - * {RegExp} │ │ │ │ │ - */ │ │ │ │ │ - quotaRegEx: (/quota/i), │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.CacheWrite │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Object with API properties for this control. │ │ │ │ │ + * APIProperty: base │ │ │ │ │ + * {String} │ │ │ │ │ */ │ │ │ │ │ + base: '', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the control. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - var i, layers = this.layers || map.layers; │ │ │ │ │ - for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - this.addLayer({ │ │ │ │ │ - layer: layers[i] │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - if (!this.layers) { │ │ │ │ │ - map.events.on({ │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - removeLayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: addLayer │ │ │ │ │ - * Adds a layer to the control. Once added, tiles requested for this layer │ │ │ │ │ - * will be cached. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} Object with a layer property referencing an │ │ │ │ │ - * <OpenLayers.Layer> instance │ │ │ │ │ - */ │ │ │ │ │ - addLayer: function(evt) { │ │ │ │ │ - evt.layer.events.on({ │ │ │ │ │ - tileloadstart: this.makeSameOrigin, │ │ │ │ │ - tileloaded: this.onTileLoaded, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeLayer │ │ │ │ │ - * Removes a layer from the control. Once removed, tiles requested for this │ │ │ │ │ - * layer will no longer be cached. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} Object with a layer property referencing an │ │ │ │ │ - * <OpenLayers.Layer> instance │ │ │ │ │ - */ │ │ │ │ │ - removeLayer: function(evt) { │ │ │ │ │ - evt.layer.events.un({ │ │ │ │ │ - tileloadstart: this.makeSameOrigin, │ │ │ │ │ - tileloaded: this.onTileLoaded, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: makeSameOrigin │ │ │ │ │ - * If the tile does not have CORS image loading enabled and is from a │ │ │ │ │ - * different origin, use OpenLayers.ProxyHost to make it a same origin url. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ + * APIProperty: displayProjection │ │ │ │ │ + * {<OpenLayers.Projection>} Requires proj4js support. Projection used │ │ │ │ │ + * when creating the coordinates in the link. This will reproject the │ │ │ │ │ + * map coordinates into display coordinates. If you are using this │ │ │ │ │ + * functionality, the permalink which is last added to the map will │ │ │ │ │ + * determine the coordinate type which is read from the URL, which │ │ │ │ │ + * means you should not add permalinks with different │ │ │ │ │ + * displayProjections to the same map. │ │ │ │ │ */ │ │ │ │ │ - makeSameOrigin: function(evt) { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - var tile = evt.tile; │ │ │ │ │ - if (tile instanceof OpenLayers.Tile.Image && │ │ │ │ │ - !tile.crossOriginKeyword && │ │ │ │ │ - tile.url.substr(0, 5) !== "data:") { │ │ │ │ │ - var sameOriginUrl = OpenLayers.Request.makeSameOrigin( │ │ │ │ │ - tile.url, OpenLayers.ProxyHost │ │ │ │ │ - ); │ │ │ │ │ - OpenLayers.Control.CacheWrite.urlMap[sameOriginUrl] = tile.url; │ │ │ │ │ - tile.url = sameOriginUrl; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + displayProjection: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onTileLoaded │ │ │ │ │ - * Decides whether a tile can be cached and calls the cache method. │ │ │ │ │ + * Constructor: OpenLayers.Control.Permalink │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - onTileLoaded: function(evt) { │ │ │ │ │ - if (this.active && !evt.aborted && │ │ │ │ │ - evt.tile instanceof OpenLayers.Tile.Image && │ │ │ │ │ - evt.tile.url.substr(0, 5) !== 'data:') { │ │ │ │ │ - this.cache({ │ │ │ │ │ - tile: evt.tile │ │ │ │ │ - }); │ │ │ │ │ - delete OpenLayers.Control.CacheWrite.urlMap[evt.tile.url]; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: cache │ │ │ │ │ - * Adds a tile to the cache. When the cache is full, the "cachefull" event │ │ │ │ │ - * is triggered. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * element - {DOMElement} │ │ │ │ │ + * base - {String} │ │ │ │ │ + * options - {Object} options to the control. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * obj - {Object} Object with a tile property, tile being the │ │ │ │ │ - * <OpenLayers.Tile.Image> with the data to add to the cache │ │ │ │ │ + * Or for anchor: │ │ │ │ │ + * options - {Object} options to the control. │ │ │ │ │ */ │ │ │ │ │ - cache: function(obj) { │ │ │ │ │ - if (window.localStorage) { │ │ │ │ │ - var tile = obj.tile; │ │ │ │ │ - try { │ │ │ │ │ - var canvasContext = tile.getCanvasContext(); │ │ │ │ │ - if (canvasContext) { │ │ │ │ │ - var urlMap = OpenLayers.Control.CacheWrite.urlMap; │ │ │ │ │ - var url = urlMap[tile.url] || tile.url; │ │ │ │ │ - window.localStorage.setItem( │ │ │ │ │ - "olCache_" + url, │ │ │ │ │ - canvasContext.canvas.toDataURL(this.imageFormat) │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - } catch (e) { │ │ │ │ │ - // local storage full or CORS violation │ │ │ │ │ - var reason = e.name || e.message; │ │ │ │ │ - if (reason && this.quotaRegEx.test(reason)) { │ │ │ │ │ - this.events.triggerEvent("cachefull", { │ │ │ │ │ - tile: tile │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Console.error(e.toString()); │ │ │ │ │ - } │ │ │ │ │ + initialize: function(element, base, options) { │ │ │ │ │ + if (element !== null && typeof element == 'object' && !OpenLayers.Util.isElement(element)) { │ │ │ │ │ + options = element; │ │ │ │ │ + this.base = document.location.href; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (this.element != null) { │ │ │ │ │ + this.element = OpenLayers.Util.getElement(this.element); │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.element = OpenLayers.Util.getElement(element); │ │ │ │ │ + this.base = base || document.location.href; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * The destroy method is used to perform any clean up before the control │ │ │ │ │ - * is dereferenced. Typically this is where event listeners are removed │ │ │ │ │ - * to prevent memory leaks. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ */ │ │ │ │ │ destroy: function() { │ │ │ │ │ - if (this.layers || this.map) { │ │ │ │ │ - var i, layers = this.layers || this.map.layers; │ │ │ │ │ - for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - this.removeLayer({ │ │ │ │ │ - layer: layers[i] │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - removeLayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.CacheWrite" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * APIFunction: OpenLayers.Control.CacheWrite.clearCache │ │ │ │ │ - * Clears all tiles cached with <OpenLayers.Control.CacheWrite> from the cache. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.CacheWrite.clearCache = function() { │ │ │ │ │ - if (!window.localStorage) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var i, key; │ │ │ │ │ - for (i = window.localStorage.length - 1; i >= 0; --i) { │ │ │ │ │ - key = window.localStorage.key(i); │ │ │ │ │ - if (key.substr(0, 8) === "olCache_") { │ │ │ │ │ - window.localStorage.removeItem(key); │ │ │ │ │ + if (this.element && this.element.parentNode == this.div) { │ │ │ │ │ + this.div.removeChild(this.element); │ │ │ │ │ + this.element = null; │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Property: OpenLayers.Control.CacheWrite.urlMap │ │ │ │ │ - * {Object} Mapping of same origin urls to cache url keys. Entries will be │ │ │ │ │ - * deleted as soon as a tile was cached. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.CacheWrite.urlMap = {}; │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Button.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Button │ │ │ │ │ - * The Button control is a very simple push-button, for use with │ │ │ │ │ - * <OpenLayers.Control.Panel>. │ │ │ │ │ - * When clicked, the function trigger() is executed. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - * │ │ │ │ │ - * Use: │ │ │ │ │ - * (code) │ │ │ │ │ - * var button = new OpenLayers.Control.Button({ │ │ │ │ │ - * displayClass: "MyButton", trigger: myFunction │ │ │ │ │ - * }); │ │ │ │ │ - * panel.addControls([button]); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Will create a button with CSS class MyButtonItemInactive, that │ │ │ │ │ - * will call the function MyFunction() when clicked. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Button = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - /** │ │ │ │ │ - * Property: type │ │ │ │ │ - * {Integer} OpenLayers.Control.TYPE_BUTTON. │ │ │ │ │ - */ │ │ │ │ │ - type: OpenLayers.Control.TYPE_BUTTON, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: trigger │ │ │ │ │ - * Called by a control panel when the button is clicked. │ │ │ │ │ - */ │ │ │ │ │ - trigger: function() {}, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Button" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Pan.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control/Button.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Pan │ │ │ │ │ - * The Pan control is a single button to pan the map in one direction. For │ │ │ │ │ - * a more complete control see <OpenLayers.Control.PanPanel>. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Pan = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: slideFactor │ │ │ │ │ - * {Integer} Number of pixels by which we'll pan the map in any direction │ │ │ │ │ - * on clicking the arrow buttons, defaults to 50. If you want to pan │ │ │ │ │ - * by some ratio of the map dimensions, use <slideRatio> instead. │ │ │ │ │ - */ │ │ │ │ │ - slideFactor: 50, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: slideRatio │ │ │ │ │ - * {Number} The fraction of map width/height by which we'll pan the map │ │ │ │ │ - * on clicking the arrow buttons. Default is null. If set, will │ │ │ │ │ - * override <slideFactor>. E.g. if slideRatio is .5, then Pan Up will │ │ │ │ │ - * pan up half the map height. │ │ │ │ │ - */ │ │ │ │ │ - slideRatio: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: direction │ │ │ │ │ - * {String} in {'North', 'South', 'East', 'West'} │ │ │ │ │ - */ │ │ │ │ │ - direction: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Pan │ │ │ │ │ - * Control which handles the panning (in any of the cardinal directions) │ │ │ │ │ - * of the map by a set px distance. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * direction - {String} The direction this button should pan. │ │ │ │ │ - * options - {Object} An optional object whose properties will be used │ │ │ │ │ - * to extend the control. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(direction, options) { │ │ │ │ │ - │ │ │ │ │ - this.direction = direction; │ │ │ │ │ - this.CLASS_NAME += this.direction; │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: trigger │ │ │ │ │ - */ │ │ │ │ │ - trigger: function() { │ │ │ │ │ if (this.map) { │ │ │ │ │ - var getSlideFactor = OpenLayers.Function.bind(function(dim) { │ │ │ │ │ - return this.slideRatio ? │ │ │ │ │ - this.map.getSize()[dim] * this.slideRatio : │ │ │ │ │ - this.slideFactor; │ │ │ │ │ - }, this); │ │ │ │ │ - │ │ │ │ │ - switch (this.direction) { │ │ │ │ │ - case OpenLayers.Control.Pan.NORTH: │ │ │ │ │ - this.map.pan(0, -getSlideFactor("h")); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Control.Pan.SOUTH: │ │ │ │ │ - this.map.pan(0, getSlideFactor("h")); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Control.Pan.WEST: │ │ │ │ │ - this.map.pan(-getSlideFactor("w"), 0); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Control.Pan.EAST: │ │ │ │ │ - this.map.pan(getSlideFactor("w"), 0); │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ + this.map.events.unregister('moveend', this, this.updateLink); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Pan" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -OpenLayers.Control.Pan.NORTH = "North"; │ │ │ │ │ -OpenLayers.Control.Pan.SOUTH = "South"; │ │ │ │ │ -OpenLayers.Control.Pan.EAST = "East"; │ │ │ │ │ -OpenLayers.Control.Pan.WEST = "West"; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/PanPanel.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control/Panel.js │ │ │ │ │ - * @requires OpenLayers/Control/Pan.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.PanPanel │ │ │ │ │ - * The PanPanel is visible control for panning the map North, South, East or │ │ │ │ │ - * West in small steps. By default it is drawn in the top left corner of the │ │ │ │ │ - * map. │ │ │ │ │ - * │ │ │ │ │ - * Note: │ │ │ │ │ - * If you wish to use this class with the default images and you want │ │ │ │ │ - * it to look nice in ie6, you should add the following, conditionally │ │ │ │ │ - * added css stylesheet to your HTML file: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * <!--[if lte IE 6]> │ │ │ │ │ - * <link rel="stylesheet" href="../theme/default/ie6-style.css" type="text/css" /> │ │ │ │ │ - * <![endif]--> │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control.Panel> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.PanPanel = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: slideFactor │ │ │ │ │ - * {Integer} Number of pixels by which we'll pan the map in any direction │ │ │ │ │ - * on clicking the arrow buttons, defaults to 50. If you want to pan │ │ │ │ │ - * by some ratio of the map dimensions, use <slideRatio> instead. │ │ │ │ │ - */ │ │ │ │ │ - slideFactor: 50, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: slideRatio │ │ │ │ │ - * {Number} The fraction of map width/height by which we'll pan the map │ │ │ │ │ - * on clicking the arrow buttons. Default is null. If set, will │ │ │ │ │ - * override <slideFactor>. E.g. if slideRatio is .5, then Pan Up will │ │ │ │ │ - * pan up half the map height. │ │ │ │ │ - */ │ │ │ │ │ - slideRatio: null, │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.PanPanel │ │ │ │ │ - * Add the four directional pan buttons. │ │ │ │ │ - * │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the control. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be used │ │ │ │ │ - * to extend the control. │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ - var options = { │ │ │ │ │ - slideFactor: this.slideFactor, │ │ │ │ │ - slideRatio: this.slideRatio │ │ │ │ │ - }; │ │ │ │ │ - this.addControls([ │ │ │ │ │ - new OpenLayers.Control.Pan(OpenLayers.Control.Pan.NORTH, options), │ │ │ │ │ - new OpenLayers.Control.Pan(OpenLayers.Control.Pan.SOUTH, options), │ │ │ │ │ - new OpenLayers.Control.Pan(OpenLayers.Control.Pan.EAST, options), │ │ │ │ │ - new OpenLayers.Control.Pan(OpenLayers.Control.Pan.WEST, options) │ │ │ │ │ - ]); │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + //make sure we have an arg parser attached │ │ │ │ │ + for (var i = 0, len = this.map.controls.length; i < len; i++) { │ │ │ │ │ + var control = this.map.controls[i]; │ │ │ │ │ + if (control.CLASS_NAME == this.argParserClass.CLASS_NAME) { │ │ │ │ │ + │ │ │ │ │ + // If a permalink is added to the map, and an ArgParser already │ │ │ │ │ + // exists, we override the displayProjection to be the one │ │ │ │ │ + // on the permalink. │ │ │ │ │ + if (control.displayProjection != this.displayProjection) { │ │ │ │ │ + this.displayProjection = control.displayProjection; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (i == this.map.controls.length) { │ │ │ │ │ + this.map.addControl(new this.argParserClass({ │ │ │ │ │ + 'displayProjection': this.displayProjection │ │ │ │ │ + })); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.PanPanel" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/ZoomIn.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + if (!this.element && !this.anchor) { │ │ │ │ │ + this.element = document.createElement("a"); │ │ │ │ │ + this.element.innerHTML = OpenLayers.i18n("Permalink"); │ │ │ │ │ + this.element.href = ""; │ │ │ │ │ + this.div.appendChild(this.element); │ │ │ │ │ + } │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + 'moveend': this.updateLink, │ │ │ │ │ + 'changelayer': this.updateLink, │ │ │ │ │ + 'changebaselayer': this.updateLink, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control/Button.js │ │ │ │ │ - */ │ │ │ │ │ + // Make it so there is at least a link even though the map may not have │ │ │ │ │ + // moved yet. │ │ │ │ │ + this.updateLink(); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.ZoomIn │ │ │ │ │ - * The ZoomIn control is a button to increase the zoom level of a map. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ZoomIn = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ + return this.div; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: trigger │ │ │ │ │ + * Method: updateLink │ │ │ │ │ */ │ │ │ │ │ - trigger: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.zoomIn(); │ │ │ │ │ + updateLink: function() { │ │ │ │ │ + var separator = this.anchor ? '#' : '?'; │ │ │ │ │ + var href = this.base; │ │ │ │ │ + var anchor = null; │ │ │ │ │ + if (href.indexOf("#") != -1 && this.anchor == false) { │ │ │ │ │ + anchor = href.substring(href.indexOf("#"), href.length); │ │ │ │ │ + } │ │ │ │ │ + if (href.indexOf(separator) != -1) { │ │ │ │ │ + href = href.substring(0, href.indexOf(separator)); │ │ │ │ │ + } │ │ │ │ │ + var splits = href.split("#"); │ │ │ │ │ + href = splits[0] + separator + OpenLayers.Util.getParameterString(this.createParams()); │ │ │ │ │ + if (anchor) { │ │ │ │ │ + href += anchor; │ │ │ │ │ + } │ │ │ │ │ + if (this.anchor && !this.element) { │ │ │ │ │ + window.location.href = href; │ │ │ │ │ + } else { │ │ │ │ │ + this.element.href = href; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomIn" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/ZoomOut.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: createParams │ │ │ │ │ + * Creates the parameters that need to be encoded into the permalink url. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * center - {<OpenLayers.LonLat>} center to encode in the permalink. │ │ │ │ │ + * Defaults to the current map center. │ │ │ │ │ + * zoom - {Integer} zoom level to encode in the permalink. Defaults to the │ │ │ │ │ + * current map zoom level. │ │ │ │ │ + * layers - {Array(<OpenLayers.Layer>)} layers to encode in the permalink. │ │ │ │ │ + * Defaults to the current map layers. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Hash of parameters that will be url-encoded into the │ │ │ │ │ + * permalink. │ │ │ │ │ + */ │ │ │ │ │ + createParams: function(center, zoom, layers) { │ │ │ │ │ + center = center || this.map.getCenter(); │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + var params = OpenLayers.Util.getParameters(this.base); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control/Button.js │ │ │ │ │ - */ │ │ │ │ │ + // If there's still no center, map is not initialized yet. │ │ │ │ │ + // Break out of this function, and simply return the params from the │ │ │ │ │ + // base link. │ │ │ │ │ + if (center) { │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.ZoomOut │ │ │ │ │ - * The ZoomOut control is a button to decrease the zoom level of a map. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ZoomOut = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ + //zoom │ │ │ │ │ + params.zoom = zoom || this.map.getZoom(); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: trigger │ │ │ │ │ - */ │ │ │ │ │ - trigger: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.zoomOut(); │ │ │ │ │ + //lon,lat │ │ │ │ │ + var lat = center.lat; │ │ │ │ │ + var lon = center.lon; │ │ │ │ │ + │ │ │ │ │ + if (this.displayProjection) { │ │ │ │ │ + var mapPosition = OpenLayers.Projection.transform({ │ │ │ │ │ + x: lon, │ │ │ │ │ + y: lat │ │ │ │ │ + }, │ │ │ │ │ + this.map.getProjectionObject(), │ │ │ │ │ + this.displayProjection); │ │ │ │ │ + lon = mapPosition.x; │ │ │ │ │ + lat = mapPosition.y; │ │ │ │ │ + } │ │ │ │ │ + params.lat = Math.round(lat * 100000) / 100000; │ │ │ │ │ + params.lon = Math.round(lon * 100000) / 100000; │ │ │ │ │ + │ │ │ │ │ + //layers │ │ │ │ │ + layers = layers || this.map.layers; │ │ │ │ │ + params.layers = ''; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + var layer = layers[i]; │ │ │ │ │ + │ │ │ │ │ + if (layer.isBaseLayer) { │ │ │ │ │ + params.layers += (layer == this.map.baseLayer) ? "B" : "0"; │ │ │ │ │ + } else { │ │ │ │ │ + params.layers += (layer.getVisibility()) ? "T" : "F"; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + │ │ │ │ │ + return params; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomOut" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Permalink" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Control/ZoomToMaxExtent.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ @@ -77235,691 +76632,549 @@ │ │ │ │ │ this.map.zoomToMaxExtent(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.ZoomToMaxExtent" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/ZoomPanel.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control/Panel.js │ │ │ │ │ - * @requires OpenLayers/Control/ZoomIn.js │ │ │ │ │ - * @requires OpenLayers/Control/ZoomOut.js │ │ │ │ │ - * @requires OpenLayers/Control/ZoomToMaxExtent.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.ZoomPanel │ │ │ │ │ - * The ZoomPanel control is a compact collecton of 3 zoom controls: a │ │ │ │ │ - * <OpenLayers.Control.ZoomIn>, a <OpenLayers.Control.ZoomToMaxExtent>, and a │ │ │ │ │ - * <OpenLayers.Control.ZoomOut>. By default it is drawn in the upper left │ │ │ │ │ - * corner of the map. │ │ │ │ │ - * │ │ │ │ │ - * Note: │ │ │ │ │ - * If you wish to use this class with the default images and you want │ │ │ │ │ - * it to look nice in ie6, you should add the following, conditionally │ │ │ │ │ - * added css stylesheet to your HTML file: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * <!--[if lte IE 6]> │ │ │ │ │ - * <link rel="stylesheet" href="../theme/default/ie6-style.css" type="text/css" /> │ │ │ │ │ - * <![endif]--> │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control.Panel> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ZoomPanel = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.ZoomPanel │ │ │ │ │ - * Add the three zooming controls. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be used │ │ │ │ │ - * to extend the control. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.addControls([ │ │ │ │ │ - new OpenLayers.Control.ZoomIn(), │ │ │ │ │ - new OpenLayers.Control.ZoomToMaxExtent(), │ │ │ │ │ - new OpenLayers.Control.ZoomOut() │ │ │ │ │ - ]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomPanel" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/GetFeature.js │ │ │ │ │ + OpenLayers/Control/Panel.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Handler/Click.js │ │ │ │ │ - * @requires OpenLayers/Handler/Box.js │ │ │ │ │ - * @requires OpenLayers/Handler/Hover.js │ │ │ │ │ - * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ + * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.GetFeature │ │ │ │ │ - * Gets vector features for locations underneath the mouse cursor. Can be │ │ │ │ │ - * configured to act on click, hover or dragged boxes. Uses an │ │ │ │ │ - * <OpenLayers.Protocol> that supports spatial filters to retrieve │ │ │ │ │ - * features from a server and fires events that notify applications of the │ │ │ │ │ - * selected features. │ │ │ │ │ + * Class: OpenLayers.Control.Panel │ │ │ │ │ + * The Panel control is a container for other controls. With it toolbars │ │ │ │ │ + * may be composed. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.GetFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: protocol │ │ │ │ │ - * {<OpenLayers.Protocol>} Required. The protocol used for fetching │ │ │ │ │ - * features. │ │ │ │ │ - */ │ │ │ │ │ - protocol: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: multipleKey │ │ │ │ │ - * {String} An event modifier ('altKey' or 'shiftKey') that temporarily sets │ │ │ │ │ - * the <multiple> property to true. Default is null. │ │ │ │ │ - */ │ │ │ │ │ - multipleKey: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: toggleKey │ │ │ │ │ - * {String} An event modifier ('altKey' or 'shiftKey') that temporarily sets │ │ │ │ │ - * the <toggle> property to true. Default is null. │ │ │ │ │ - */ │ │ │ │ │ - toggleKey: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: modifiers │ │ │ │ │ - * {Object} The event modifiers to use, according to the current event │ │ │ │ │ - * being handled by this control's handlers │ │ │ │ │ - */ │ │ │ │ │ - modifiers: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: multiple │ │ │ │ │ - * {Boolean} Allow selection of multiple geometries. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - multiple: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: click │ │ │ │ │ - * {Boolean} Use a click handler for selecting/unselecting features. If │ │ │ │ │ - * both <click> and <box> are set to true, the click handler takes │ │ │ │ │ - * precedence over the box handler if a box with zero extent was │ │ │ │ │ - * selected. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - click: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: single │ │ │ │ │ - * {Boolean} Tells whether select by click should select a single │ │ │ │ │ - * feature. If set to false, all matching features are selected. │ │ │ │ │ - * If set to true, only the best matching feature is selected. │ │ │ │ │ - * This option has an effect only of the <click> option is set │ │ │ │ │ - * to true. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - single: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: clickout │ │ │ │ │ - * {Boolean} Unselect features when clicking outside any feature. │ │ │ │ │ - * Applies only if <click> is true. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - clickout: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: toggle │ │ │ │ │ - * {Boolean} Unselect a selected feature on click. Applies only if │ │ │ │ │ - * <click> is true. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - toggle: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: clickTolerance │ │ │ │ │ - * {Integer} Tolerance for the filter query in pixels. This has the │ │ │ │ │ - * same effect as the tolerance parameter on WMS GetFeatureInfo │ │ │ │ │ - * requests. Will be ignored for box selections. Applies only if │ │ │ │ │ - * <click> or <hover> is true. Default is 5. Note that this not │ │ │ │ │ - * only affects requests on click, but also on hover. │ │ │ │ │ - */ │ │ │ │ │ - clickTolerance: 5, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: hover │ │ │ │ │ - * {Boolean} Send feature requests on mouse moves. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - hover: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: box │ │ │ │ │ - * {Boolean} Allow feature selection by drawing a box. If set to │ │ │ │ │ - * true set <click> to false to disable the click handler and │ │ │ │ │ - * rely on the box handler only, even for "zero extent" boxes. │ │ │ │ │ - * See the description of the <click> option for additional │ │ │ │ │ - * information. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - box: false, │ │ │ │ │ - │ │ │ │ │ +OpenLayers.Control.Panel = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: maxFeatures │ │ │ │ │ - * {Integer} Maximum number of features to return from a query in single mode │ │ │ │ │ - * if supported by the <protocol>. This set of features is then used to │ │ │ │ │ - * determine the best match client-side. Default is 10. │ │ │ │ │ + * Property: controls │ │ │ │ │ + * {Array(<OpenLayers.Control>)} │ │ │ │ │ */ │ │ │ │ │ - maxFeatures: 10, │ │ │ │ │ + controls: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Object} Hash of {<OpenLayers.Feature.Vector>}, keyed by fid, holding │ │ │ │ │ - * the currently selected features │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ - features: null, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Proeprty: hoverFeature │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} The feature currently selected by the │ │ │ │ │ - * hover handler │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: defaultControl │ │ │ │ │ + * {<OpenLayers.Control>} The control which is activated when the control is │ │ │ │ │ + * activated (turned on), which also happens at instantiation. │ │ │ │ │ + * If <saveState> is true, <defaultControl> will be nullified after the │ │ │ │ │ + * first activation of the panel. │ │ │ │ │ */ │ │ │ │ │ - hoverFeature: null, │ │ │ │ │ + defaultControl: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: handlerOptions │ │ │ │ │ - * {Object} Additional options for the handlers used by this control. This │ │ │ │ │ - * is a hash with the keys "click", "box" and "hover". │ │ │ │ │ + * APIProperty: saveState │ │ │ │ │ + * {Boolean} If set to true, the active state of this panel's controls will │ │ │ │ │ + * be stored on panel deactivation, and restored on reactivation. Default │ │ │ │ │ + * is false. │ │ │ │ │ */ │ │ │ │ │ + saveState: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: handlers │ │ │ │ │ - * {Object} Object with references to multiple <OpenLayers.Handler> │ │ │ │ │ - * instances. │ │ │ │ │ + * APIProperty: allowDepress │ │ │ │ │ + * {Boolean} If is true the <OpenLayers.Control.TYPE_TOOL> controls can │ │ │ │ │ + * be deactivated by clicking the icon that represents them. Default │ │ │ │ │ + * is false. │ │ │ │ │ */ │ │ │ │ │ - handlers: null, │ │ │ │ │ + allowDepress: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: hoverResponse │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} The response object associated with │ │ │ │ │ - * the currently running hover request (if any). │ │ │ │ │ + * Property: activeState │ │ │ │ │ + * {Object} stores the active state of this panel's controls. │ │ │ │ │ */ │ │ │ │ │ - hoverResponse: null, │ │ │ │ │ + activeState: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: filterType │ │ │ │ │ - * {<String>} The type of filter to use when sending off a request. │ │ │ │ │ - * Possible values: │ │ │ │ │ - * OpenLayers.Filter.Spatial.<BBOX|INTERSECTS|WITHIN|CONTAINS> │ │ │ │ │ - * Defaults to: OpenLayers.Filter.Spatial.BBOX │ │ │ │ │ - */ │ │ │ │ │ - filterType: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ + * Constructor: OpenLayers.Control.Panel │ │ │ │ │ + * Create a new control panel. │ │ │ │ │ * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ + * Each control in the panel is represented by an icon. When clicking │ │ │ │ │ + * on an icon, the <activateControl> method is called. │ │ │ │ │ * │ │ │ │ │ - * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * beforefeatureselected - Triggered when <click> is true before a │ │ │ │ │ - * feature is selected. The event object has a feature property with │ │ │ │ │ - * the feature about to select │ │ │ │ │ - * featureselected - Triggered when <click> is true and a feature is │ │ │ │ │ - * selected. The event object has a feature property with the │ │ │ │ │ - * selected feature │ │ │ │ │ - * beforefeaturesselected - Triggered when <click> is true before a │ │ │ │ │ - * set of features is selected. The event object is an array of │ │ │ │ │ - * feature properties with the features about to be selected. │ │ │ │ │ - * Return false after receiving this event to discontinue processing │ │ │ │ │ - * of all featureselected events and the featuresselected event. │ │ │ │ │ - * featuresselected - Triggered when <click> is true and a set of │ │ │ │ │ - * features is selected. The event object is an array of feature │ │ │ │ │ - * properties of the selected features │ │ │ │ │ - * featureunselected - Triggered when <click> is true and a feature is │ │ │ │ │ - * unselected. The event object has a feature property with the │ │ │ │ │ - * unselected feature │ │ │ │ │ - * clickout - Triggered when when <click> is true and no feature was │ │ │ │ │ - * selected. │ │ │ │ │ - * hoverfeature - Triggered when <hover> is true and the mouse has │ │ │ │ │ - * stopped over a feature │ │ │ │ │ - * outfeature - Triggered when <hover> is true and the mouse moves │ │ │ │ │ - * moved away from a hover-selected feature │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.GetFeature │ │ │ │ │ - * Create a new control for fetching remote features. │ │ │ │ │ + * Specific properties for controls on a panel: │ │ │ │ │ + * type - {Number} One of <OpenLayers.Control.TYPE_TOOL>, │ │ │ │ │ + * <OpenLayers.Control.TYPE_TOGGLE>, <OpenLayers.Control.TYPE_BUTTON>. │ │ │ │ │ + * If not provided, <OpenLayers.Control.TYPE_TOOL> is assumed. │ │ │ │ │ + * title - {string} Text displayed when mouse is over the icon that │ │ │ │ │ + * represents the control. │ │ │ │ │ + * │ │ │ │ │ + * The <OpenLayers.Control.type> of a control determines the behavior when │ │ │ │ │ + * clicking its icon: │ │ │ │ │ + * <OpenLayers.Control.TYPE_TOOL> - The control is activated and other │ │ │ │ │ + * controls of this type in the same panel are deactivated. This is │ │ │ │ │ + * the default type. │ │ │ │ │ + * <OpenLayers.Control.TYPE_TOGGLE> - The active state of the control is │ │ │ │ │ + * toggled. │ │ │ │ │ + * <OpenLayers.Control.TYPE_BUTTON> - The │ │ │ │ │ + * <OpenLayers.Control.Button.trigger> method of the control is called, │ │ │ │ │ + * but its active state is not changed. │ │ │ │ │ + * │ │ │ │ │ + * If a control is <OpenLayers.Control.active>, it will be drawn with the │ │ │ │ │ + * olControl[Name]ItemActive class, otherwise with the │ │ │ │ │ + * olControl[Name]ItemInactive class. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} A configuration object which at least has to contain │ │ │ │ │ - * a <protocol> property (if not, it has to be set before a request is │ │ │ │ │ - * made) │ │ │ │ │ + * options - {Object} An optional object whose properties will be used │ │ │ │ │ + * to extend the control. │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ - │ │ │ │ │ OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.controls = []; │ │ │ │ │ + this.activeState = {}; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.features = {}; │ │ │ │ │ - │ │ │ │ │ - this.handlers = {}; │ │ │ │ │ - │ │ │ │ │ - if (this.click) { │ │ │ │ │ - this.handlers.click = new OpenLayers.Handler.Click(this, { │ │ │ │ │ - click: this.selectClick │ │ │ │ │ - }, this.handlerOptions.click || {}); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.box) { │ │ │ │ │ - this.handlers.box = new OpenLayers.Handler.Box( │ │ │ │ │ - this, { │ │ │ │ │ - done: this.selectBox │ │ │ │ │ - }, │ │ │ │ │ - OpenLayers.Util.extend(this.handlerOptions.box, { │ │ │ │ │ - boxDivClassName: "olHandlerBoxSelectFeature" │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - this.handlers.hover = new OpenLayers.Handler.Hover( │ │ │ │ │ - this, { │ │ │ │ │ - 'move': this.cancelHover, │ │ │ │ │ - 'pause': this.selectHover │ │ │ │ │ - }, │ │ │ │ │ - OpenLayers.Util.extend(this.handlerOptions.hover, { │ │ │ │ │ - 'delay': 250, │ │ │ │ │ - 'pixelTolerance': 2 │ │ │ │ │ - }) │ │ │ │ │ - ); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + for (var ctl, i = this.controls.length - 1; i >= 0; i--) { │ │ │ │ │ + ctl = this.controls[i]; │ │ │ │ │ + if (ctl.events) { │ │ │ │ │ + ctl.events.un({ │ │ │ │ │ + activate: this.iconOn, │ │ │ │ │ + deactivate: this.iconOff │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + ctl.panel_div = null; │ │ │ │ │ } │ │ │ │ │ + this.activeState = null; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - * Activates the control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The control was effectively activated. │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ */ │ │ │ │ │ activate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - for (var i in this.handlers) { │ │ │ │ │ - this.handlers[i].activate(); │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + control = this.controls[i]; │ │ │ │ │ + if (control === this.defaultControl || │ │ │ │ │ + (this.saveState && this.activeState[control.id])) { │ │ │ │ │ + control.activate(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.saveState === true) { │ │ │ │ │ + this.defaultControl = null; │ │ │ │ │ } │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Control.prototype.activate.apply( │ │ │ │ │ - this, arguments │ │ │ │ │ - ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - * Deactivates the control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The control was effectively deactivated. │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ */ │ │ │ │ │ deactivate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - for (var i in this.handlers) { │ │ │ │ │ - this.handlers[i].deactivate(); │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + control = this.controls[i]; │ │ │ │ │ + this.activeState[control.id] = control.deactivate(); │ │ │ │ │ } │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Control.prototype.deactivate.apply( │ │ │ │ │ - this, arguments │ │ │ │ │ - ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: selectClick │ │ │ │ │ - * Called on click │ │ │ │ │ + * Method: draw │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - selectClick: function(evt) { │ │ │ │ │ - var bounds = this.pixelToBounds(evt.xy); │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (this.outsideViewport) { │ │ │ │ │ + this.events.attachToElement(this.div); │ │ │ │ │ + this.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ + } else { │ │ │ │ │ + this.map.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ + } │ │ │ │ │ + this.addControlsToMap(this.controls); │ │ │ │ │ + return this.div; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.setModifiers(evt); │ │ │ │ │ - this.request(bounds, { │ │ │ │ │ - single: this.single │ │ │ │ │ - }); │ │ │ │ │ + /** │ │ │ │ │ + * Method: redraw │ │ │ │ │ + */ │ │ │ │ │ + redraw: function() { │ │ │ │ │ + for (var l = this.div.childNodes.length, i = l - 1; i >= 0; i--) { │ │ │ │ │ + this.div.removeChild(this.div.childNodes[i]); │ │ │ │ │ + } │ │ │ │ │ + this.div.innerHTML = ""; │ │ │ │ │ + if (this.active) { │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + this.div.appendChild(this.controls[i].panel_div); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: selectBox │ │ │ │ │ - * Callback from the handlers.box set up when <box> selection is on │ │ │ │ │ + * APIMethod: activateControl │ │ │ │ │ + * This method is called when the user click on the icon representing a │ │ │ │ │ + * control in the panel. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * position - {<OpenLayers.Bounds>|Object} An OpenLayers.Bounds or │ │ │ │ │ - * an object with a 'left', 'bottom', 'right' and 'top' properties. │ │ │ │ │ + * control - {<OpenLayers.Control>} │ │ │ │ │ */ │ │ │ │ │ - selectBox: function(position) { │ │ │ │ │ - var bounds; │ │ │ │ │ - if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ - var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.left, │ │ │ │ │ - y: position.bottom │ │ │ │ │ - }); │ │ │ │ │ - var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.right, │ │ │ │ │ - y: position.top │ │ │ │ │ - }); │ │ │ │ │ - bounds = new OpenLayers.Bounds( │ │ │ │ │ - minXY.lon, minXY.lat, maxXY.lon, maxXY.lat │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ + activateControl: function(control) { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + if (control.type == OpenLayers.Control.TYPE_BUTTON) { │ │ │ │ │ + control.trigger(); │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (control.type == OpenLayers.Control.TYPE_TOGGLE) { │ │ │ │ │ + if (control.active) { │ │ │ │ │ + control.deactivate(); │ │ │ │ │ + } else { │ │ │ │ │ + control.activate(); │ │ │ │ │ + } │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (this.allowDepress && control.active) { │ │ │ │ │ + control.deactivate(); │ │ │ │ │ } else { │ │ │ │ │ - if (this.click) { │ │ │ │ │ - // box without extent - let the click handler take care of it │ │ │ │ │ - return; │ │ │ │ │ + var c; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + c = this.controls[i]; │ │ │ │ │ + if (c != control && │ │ │ │ │ + (c.type === OpenLayers.Control.TYPE_TOOL || c.type == null)) { │ │ │ │ │ + c.deactivate(); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - bounds = this.pixelToBounds(position); │ │ │ │ │ + control.activate(); │ │ │ │ │ } │ │ │ │ │ - this.setModifiers(this.handlers.box.dragHandler.evt); │ │ │ │ │ - this.request(bounds); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: selectHover │ │ │ │ │ - * Callback from the handlers.hover set up when <hover> selection is on │ │ │ │ │ + * APIMethod: addControls │ │ │ │ │ + * To build a toolbar, you add a set of controls to it. addControls │ │ │ │ │ + * lets you add a single control or a list of controls to the │ │ │ │ │ + * Control Panel. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Object} event object with an xy property │ │ │ │ │ + * controls - {<OpenLayers.Control>} Controls to add in the panel. │ │ │ │ │ */ │ │ │ │ │ - selectHover: function(evt) { │ │ │ │ │ - var bounds = this.pixelToBounds(evt.xy); │ │ │ │ │ - this.request(bounds, { │ │ │ │ │ - single: true, │ │ │ │ │ - hover: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + addControls: function(controls) { │ │ │ │ │ + if (!(OpenLayers.Util.isArray(controls))) { │ │ │ │ │ + controls = [controls]; │ │ │ │ │ + } │ │ │ │ │ + this.controls = this.controls.concat(controls); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: cancelHover │ │ │ │ │ - * Callback from the handlers.hover set up when <hover> selection is on │ │ │ │ │ - */ │ │ │ │ │ - cancelHover: function() { │ │ │ │ │ - if (this.hoverResponse) { │ │ │ │ │ - this.protocol.abort(this.hoverResponse); │ │ │ │ │ - this.hoverResponse = null; │ │ │ │ │ + for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ + var control = controls[i], │ │ │ │ │ + element = this.createControlMarkup(control); │ │ │ │ │ + OpenLayers.Element.addClass(element, │ │ │ │ │ + control.displayClass + "ItemInactive"); │ │ │ │ │ + OpenLayers.Element.addClass(element, "olButton"); │ │ │ │ │ + if (control.title != "" && !element.title) { │ │ │ │ │ + element.title = control.title; │ │ │ │ │ + } │ │ │ │ │ + control.panel_div = element; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + if (this.map) { // map.addControl() has already been called on the panel │ │ │ │ │ + this.addControlsToMap(controls); │ │ │ │ │ + this.redraw(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: request │ │ │ │ │ - * Sends a GetFeature request to the WFS │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: createControlMarkup │ │ │ │ │ + * This function just creates a div for the control. If specific HTML │ │ │ │ │ + * markup is needed this function can be overridden in specific classes, │ │ │ │ │ + * or at panel instantiation time: │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var panel = new OpenLayers.Control.Panel({ │ │ │ │ │ + * defaultControl: control, │ │ │ │ │ + * // ovverride createControlMarkup to create actual buttons │ │ │ │ │ + * // including texts wrapped into span elements. │ │ │ │ │ + * createControlMarkup: function(control) { │ │ │ │ │ + * var button = document.createElement('button'), │ │ │ │ │ + * span = document.createElement('span'); │ │ │ │ │ + * if (control.text) { │ │ │ │ │ + * span.innerHTML = control.text; │ │ │ │ │ + * } │ │ │ │ │ + * return button; │ │ │ │ │ + * } │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * bounds - {<OpenLayers.Bounds>} bounds for the request's BBOX filter │ │ │ │ │ - * options - {Object} additional options for this method. │ │ │ │ │ - * │ │ │ │ │ - * Supported options include: │ │ │ │ │ - * single - {Boolean} A single feature should be returned. │ │ │ │ │ - * Note that this will be ignored if the protocol does not │ │ │ │ │ - * return the geometries of the features. │ │ │ │ │ - * hover - {Boolean} Do the request for the hover handler. │ │ │ │ │ + * control - {<OpenLayers.Control>} The control to create the HTML │ │ │ │ │ + * markup for. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} The markup. │ │ │ │ │ */ │ │ │ │ │ - request: function(bounds, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: this.filterType, │ │ │ │ │ - value: bounds │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - // Set the cursor to "wait" to tell the user we're working. │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - │ │ │ │ │ - var response = this.protocol.read({ │ │ │ │ │ - maxFeatures: options.single == true ? this.maxFeatures : undefined, │ │ │ │ │ - filter: filter, │ │ │ │ │ - callback: function(result) { │ │ │ │ │ - if (result.success()) { │ │ │ │ │ - if (result.features.length) { │ │ │ │ │ - if (options.single == true) { │ │ │ │ │ - this.selectBestFeature(result.features, │ │ │ │ │ - bounds.getCenterLonLat(), options); │ │ │ │ │ - } else { │ │ │ │ │ - this.select(result.features); │ │ │ │ │ - } │ │ │ │ │ - } else if (options.hover) { │ │ │ │ │ - this.hoverSelect(); │ │ │ │ │ - } else { │ │ │ │ │ - this.events.triggerEvent("clickout"); │ │ │ │ │ - if (this.clickout) { │ │ │ │ │ - this.unselectAll(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // Reset the cursor. │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - if (options.hover == true) { │ │ │ │ │ - this.hoverResponse = response; │ │ │ │ │ - } │ │ │ │ │ + createControlMarkup: function(control) { │ │ │ │ │ + return document.createElement("div"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: selectBestFeature │ │ │ │ │ - * Selects the feature from an array of features that is the best match │ │ │ │ │ - * for the click position. │ │ │ │ │ - * │ │ │ │ │ + * Method: addControlsToMap │ │ │ │ │ + * Only for internal use in draw() and addControls() methods. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ - * clickPosition - {<OpenLayers.LonLat>} │ │ │ │ │ - * options - {Object} additional options for this method │ │ │ │ │ - * │ │ │ │ │ - * Supported options include: │ │ │ │ │ - * hover - {Boolean} Do the selection for the hover handler. │ │ │ │ │ + * controls - {Array(<OpenLayers.Control>)} Controls to add into map. │ │ │ │ │ */ │ │ │ │ │ - selectBestFeature: function(features, clickPosition, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - if (features.length) { │ │ │ │ │ - var point = new OpenLayers.Geometry.Point(clickPosition.lon, │ │ │ │ │ - clickPosition.lat); │ │ │ │ │ - var feature, resultFeature, dist; │ │ │ │ │ - var minDist = Number.MAX_VALUE; │ │ │ │ │ - for (var i = 0; i < features.length; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - dist = point.distanceTo(feature.geometry, { │ │ │ │ │ - edge: false │ │ │ │ │ - }); │ │ │ │ │ - if (dist < minDist) { │ │ │ │ │ - minDist = dist; │ │ │ │ │ - resultFeature = feature; │ │ │ │ │ - if (minDist == 0) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (options.hover == true) { │ │ │ │ │ - this.hoverSelect(resultFeature); │ │ │ │ │ + addControlsToMap: function(controls) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ + control = controls[i]; │ │ │ │ │ + if (control.autoActivate === true) { │ │ │ │ │ + control.autoActivate = false; │ │ │ │ │ + this.map.addControl(control); │ │ │ │ │ + control.autoActivate = true; │ │ │ │ │ } else { │ │ │ │ │ - this.select(resultFeature || features); │ │ │ │ │ + this.map.addControl(control); │ │ │ │ │ + control.deactivate(); │ │ │ │ │ } │ │ │ │ │ + control.events.on({ │ │ │ │ │ + activate: this.iconOn, │ │ │ │ │ + deactivate: this.iconOff │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setModifiers │ │ │ │ │ - * Sets the multiple and toggle modifiers according to the current event │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ + * Method: iconOn │ │ │ │ │ + * Internal use, for use only with "controls[i].events.on/un". │ │ │ │ │ */ │ │ │ │ │ - setModifiers: function(evt) { │ │ │ │ │ - this.modifiers = { │ │ │ │ │ - multiple: this.multiple || (this.multipleKey && evt[this.multipleKey]), │ │ │ │ │ - toggle: this.toggle || (this.toggleKey && evt[this.toggleKey]) │ │ │ │ │ - }; │ │ │ │ │ + iconOn: function() { │ │ │ │ │ + var d = this.panel_div; // "this" refers to a control on panel! │ │ │ │ │ + var re = new RegExp("\\b(" + this.displayClass + "Item)Inactive\\b"); │ │ │ │ │ + d.className = d.className.replace(re, "$1Active"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: select │ │ │ │ │ - * Add feature to the hash of selected features and trigger the │ │ │ │ │ - * featureselected and featuresselected events. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {<OpenLayers.Feature.Vector>} or an array of features │ │ │ │ │ + * Method: iconOff │ │ │ │ │ + * Internal use, for use only with "controls[i].events.on/un". │ │ │ │ │ */ │ │ │ │ │ - select: function(features) { │ │ │ │ │ - if (!this.modifiers.multiple && !this.modifiers.toggle) { │ │ │ │ │ - this.unselectAll(); │ │ │ │ │ - } │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var cont = this.events.triggerEvent("beforefeaturesselected", { │ │ │ │ │ - features: features │ │ │ │ │ - }); │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - var selectedFeatures = []; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (this.features[feature.fid || feature.id]) { │ │ │ │ │ - if (this.modifiers.toggle) { │ │ │ │ │ - this.unselect(this.features[feature.fid || feature.id]); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - cont = this.events.triggerEvent("beforefeatureselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - this.features[feature.fid || feature.id] = feature; │ │ │ │ │ - selectedFeatures.push(feature); │ │ │ │ │ + iconOff: function() { │ │ │ │ │ + var d = this.panel_div; // "this" refers to a control on panel! │ │ │ │ │ + var re = new RegExp("\\b(" + this.displayClass + "Item)Active\\b"); │ │ │ │ │ + d.className = d.className.replace(re, "$1Inactive"); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.events.triggerEvent("featureselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: onButtonClick │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + var controls = this.controls, │ │ │ │ │ + button = evt.buttonElement; │ │ │ │ │ + for (var i = controls.length - 1; i >= 0; --i) { │ │ │ │ │ + if (controls[i].panel_div === button) { │ │ │ │ │ + this.activateControl(controls[i]); │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ - this.events.triggerEvent("featuresselected", { │ │ │ │ │ - features: selectedFeatures │ │ │ │ │ - }); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: hoverSelect │ │ │ │ │ - * Sets/unsets the <hoverFeature> │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getControlsBy │ │ │ │ │ + * Get a list of controls with properties matching the given criteria. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} the feature to hover-select. │ │ │ │ │ - * If none is provided, the current <hoverFeature> will be nulled and │ │ │ │ │ - * the outfeature event will be triggered. │ │ │ │ │ + * property - {String} A control property to be matched. │ │ │ │ │ + * match - {String | Object} A string to match. Can also be a regular │ │ │ │ │ + * expression literal or object. In addition, it can be any object │ │ │ │ │ + * with a method named test. For reqular expressions or other, if │ │ │ │ │ + * match.test(control[property]) evaluates to true, the control will be │ │ │ │ │ + * included in the array returned. If no controls are found, an empty │ │ │ │ │ + * array is returned. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Control>)} A list of controls matching the given criteria. │ │ │ │ │ + * An empty array is returned if no matches are found. │ │ │ │ │ */ │ │ │ │ │ - hoverSelect: function(feature) { │ │ │ │ │ - var fid = feature ? feature.fid || feature.id : null; │ │ │ │ │ - var hfid = this.hoverFeature ? │ │ │ │ │ - this.hoverFeature.fid || this.hoverFeature.id : null; │ │ │ │ │ - │ │ │ │ │ - if (hfid && hfid != fid) { │ │ │ │ │ - this.events.triggerEvent("outfeature", { │ │ │ │ │ - feature: this.hoverFeature │ │ │ │ │ - }); │ │ │ │ │ - this.hoverFeature = null; │ │ │ │ │ - } │ │ │ │ │ - if (fid && fid != hfid) { │ │ │ │ │ - this.events.triggerEvent("hoverfeature", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - this.hoverFeature = feature; │ │ │ │ │ - } │ │ │ │ │ + getControlsBy: function(property, match) { │ │ │ │ │ + var test = (typeof match.test == "function"); │ │ │ │ │ + var found = OpenLayers.Array.filter(this.controls, function(item) { │ │ │ │ │ + return item[property] == match || (test && match.test(item[property])); │ │ │ │ │ + }); │ │ │ │ │ + return found; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: unselect │ │ │ │ │ - * Remove feature from the hash of selected features and trigger the │ │ │ │ │ - * featureunselected event. │ │ │ │ │ + * APIMethod: getControlsByName │ │ │ │ │ + * Get a list of contorls with names matching the given name. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * match - {String | Object} A control name. The name can also be a regular │ │ │ │ │ + * expression literal or object. In addition, it can be any object │ │ │ │ │ + * with a method named test. For reqular expressions or other, if │ │ │ │ │ + * name.test(control.name) evaluates to true, the control will be included │ │ │ │ │ + * in the list of controls returned. If no controls are found, an empty │ │ │ │ │ + * array is returned. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Control>)} A list of controls matching the given name. │ │ │ │ │ + * An empty array is returned if no matches are found. │ │ │ │ │ */ │ │ │ │ │ - unselect: function(feature) { │ │ │ │ │ - delete this.features[feature.fid || feature.id]; │ │ │ │ │ - this.events.triggerEvent("featureunselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ + getControlsByName: function(match) { │ │ │ │ │ + return this.getControlsBy("name", match); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: unselectAll │ │ │ │ │ - * Unselect all selected features. │ │ │ │ │ + * APIMethod: getControlsByClass │ │ │ │ │ + * Get a list of controls of a given type (CLASS_NAME). │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * match - {String | Object} A control class name. The type can also be a │ │ │ │ │ + * regular expression literal or object. In addition, it can be any │ │ │ │ │ + * object with a method named test. For reqular expressions or other, │ │ │ │ │ + * if type.test(control.CLASS_NAME) evaluates to true, the control will │ │ │ │ │ + * be included in the list of controls returned. If no controls are │ │ │ │ │ + * found, an empty array is returned. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array(<OpenLayers.Control>)} A list of controls matching the given type. │ │ │ │ │ + * An empty array is returned if no matches are found. │ │ │ │ │ */ │ │ │ │ │ - unselectAll: function() { │ │ │ │ │ - // we'll want an option to supress notification here │ │ │ │ │ - for (var fid in this.features) { │ │ │ │ │ - this.unselect(this.features[fid]); │ │ │ │ │ - } │ │ │ │ │ + getControlsByClass: function(match) { │ │ │ │ │ + return this.getControlsBy("CLASS_NAME", match); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Panel" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/Pan.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control/Button.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.Pan │ │ │ │ │ + * The Pan control is a single button to pan the map in one direction. For │ │ │ │ │ + * a more complete control see <OpenLayers.Control.PanPanel>. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.Pan = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the control. │ │ │ │ │ - * │ │ │ │ │ + * APIProperty: slideFactor │ │ │ │ │ + * {Integer} Number of pixels by which we'll pan the map in any direction │ │ │ │ │ + * on clicking the arrow buttons, defaults to 50. If you want to pan │ │ │ │ │ + * by some ratio of the map dimensions, use <slideRatio> instead. │ │ │ │ │ + */ │ │ │ │ │ + slideFactor: 50, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: slideRatio │ │ │ │ │ + * {Number} The fraction of map width/height by which we'll pan the map │ │ │ │ │ + * on clicking the arrow buttons. Default is null. If set, will │ │ │ │ │ + * override <slideFactor>. E.g. if slideRatio is .5, then Pan Up will │ │ │ │ │ + * pan up half the map height. │ │ │ │ │ + */ │ │ │ │ │ + slideRatio: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: direction │ │ │ │ │ + * {String} in {'North', 'South', 'East', 'West'} │ │ │ │ │ + */ │ │ │ │ │ + direction: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.Pan │ │ │ │ │ + * Control which handles the panning (in any of the cardinal directions) │ │ │ │ │ + * of the map by a set px distance. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * direction - {String} The direction this button should pan. │ │ │ │ │ + * options - {Object} An optional object whose properties will be used │ │ │ │ │ + * to extend the control. │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - for (var i in this.handlers) { │ │ │ │ │ - this.handlers[i].setMap(map); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + initialize: function(direction, options) { │ │ │ │ │ + │ │ │ │ │ + this.direction = direction; │ │ │ │ │ + this.CLASS_NAME += this.direction; │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: pixelToBounds │ │ │ │ │ - * Takes a pixel as argument and creates bounds after adding the │ │ │ │ │ - * <clickTolerance>. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * pixel - {<OpenLayers.Pixel>} │ │ │ │ │ + * Method: trigger │ │ │ │ │ */ │ │ │ │ │ - pixelToBounds: function(pixel) { │ │ │ │ │ - var llPx = pixel.add(-this.clickTolerance / 2, this.clickTolerance / 2); │ │ │ │ │ - var urPx = pixel.add(this.clickTolerance / 2, -this.clickTolerance / 2); │ │ │ │ │ - var ll = this.map.getLonLatFromPixel(llPx); │ │ │ │ │ - var ur = this.map.getLonLatFromPixel(urPx); │ │ │ │ │ - return new OpenLayers.Bounds(ll.lon, ll.lat, ur.lon, ur.lat); │ │ │ │ │ + trigger: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + var getSlideFactor = OpenLayers.Function.bind(function(dim) { │ │ │ │ │ + return this.slideRatio ? │ │ │ │ │ + this.map.getSize()[dim] * this.slideRatio : │ │ │ │ │ + this.slideFactor; │ │ │ │ │ + }, this); │ │ │ │ │ + │ │ │ │ │ + switch (this.direction) { │ │ │ │ │ + case OpenLayers.Control.Pan.NORTH: │ │ │ │ │ + this.map.pan(0, -getSlideFactor("h")); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Control.Pan.SOUTH: │ │ │ │ │ + this.map.pan(0, getSlideFactor("h")); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Control.Pan.WEST: │ │ │ │ │ + this.map.pan(-getSlideFactor("w"), 0); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Control.Pan.EAST: │ │ │ │ │ + this.map.pan(getSlideFactor("w"), 0); │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.GetFeature" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Pan" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ +OpenLayers.Control.Pan.NORTH = "North"; │ │ │ │ │ +OpenLayers.Control.Pan.SOUTH = "South"; │ │ │ │ │ +OpenLayers.Control.Pan.EAST = "East"; │ │ │ │ │ +OpenLayers.Control.Pan.WEST = "West"; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Control/Snapping.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ @@ -78493,581 +77748,1574 @@ │ │ │ │ │ delete this.targets; │ │ │ │ │ OpenLayers.Control.prototype.destroy.call(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.Snapping" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/WMTSGetFeatureInfo.js │ │ │ │ │ + OpenLayers/Control/CacheWrite.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Handler/Click.js │ │ │ │ │ - * @requires OpenLayers/Handler/Hover.js │ │ │ │ │ * @requires OpenLayers/Request.js │ │ │ │ │ - * @requires OpenLayers/Format/WMSGetFeatureInfo.js │ │ │ │ │ + * @requires OpenLayers/Console.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.WMTSGetFeatureInfo │ │ │ │ │ - * The WMTSGetFeatureInfo control uses a WMTS query to get information about a │ │ │ │ │ - * point on the map. The information may be in a display-friendly format │ │ │ │ │ - * such as HTML, or a machine-friendly format such as GML, depending on the │ │ │ │ │ - * server's capabilities and the client's configuration. This control │ │ │ │ │ - * handles click or hover events, attempts to parse the results using an │ │ │ │ │ - * OpenLayers.Format, and fires a 'getfeatureinfo' event for each layer │ │ │ │ │ - * queried. │ │ │ │ │ + * Class: OpenLayers.Control.CacheWrite │ │ │ │ │ + * A control for caching image tiles in the browser's local storage. The │ │ │ │ │ + * <OpenLayers.Control.CacheRead> control is used to fetch and use the cached │ │ │ │ │ + * tile images. │ │ │ │ │ + * │ │ │ │ │ + * Note: Before using this control on any layer that is not your own, make sure │ │ │ │ │ + * that the terms of service of the tile provider allow local storage of tiles. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.WMTSGetFeatureInfo = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Control.CacheWrite = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: hover │ │ │ │ │ - * {Boolean} Send GetFeatureInfo requests when mouse stops moving. │ │ │ │ │ - * Default is false. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ + * │ │ │ │ │ + * To register events in the constructor, configure <eventListeners>. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ + * cachefull - Triggered when the cache is full. Listeners receive an │ │ │ │ │ + * object with a tile property as first argument. The tile references │ │ │ │ │ + * the tile that couldn't be cached. │ │ │ │ │ */ │ │ │ │ │ - hover: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: requestEncoding │ │ │ │ │ - * {String} One of "KVP" or "REST". Only KVP encoding is supported at this │ │ │ │ │ - * time. │ │ │ │ │ + * APIProperty: eventListeners │ │ │ │ │ + * {Object} Object with event listeners, keyed by event name. An optional │ │ │ │ │ + * scope property defines the scope that listeners will be executed in. │ │ │ │ │ */ │ │ │ │ │ - requestEncoding: "KVP", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: drillDown │ │ │ │ │ - * {Boolean} Drill down over all WMTS layers in the map. When │ │ │ │ │ - * using drillDown mode, hover is not possible. A getfeatureinfo event │ │ │ │ │ - * will be fired for each layer queried. │ │ │ │ │ + * APIProperty: layers │ │ │ │ │ + * {Array(<OpenLayers.Layer.Grid>)}. Optional. If provided, caching │ │ │ │ │ + * will be enabled for these layers only, otherwise for all cacheable │ │ │ │ │ + * layers. │ │ │ │ │ */ │ │ │ │ │ - drillDown: false, │ │ │ │ │ + layers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: maxFeatures │ │ │ │ │ - * {Integer} Maximum number of features to return from a WMTS query. This │ │ │ │ │ - * sets the feature_count parameter on WMTS GetFeatureInfo │ │ │ │ │ - * requests. │ │ │ │ │ - */ │ │ │ │ │ - maxFeatures: 10, │ │ │ │ │ - │ │ │ │ │ - /** APIProperty: clickCallback │ │ │ │ │ - * {String} The click callback to register in the │ │ │ │ │ - * {<OpenLayers.Handler.Click>} object created when the hover │ │ │ │ │ - * option is set to false. Default is "click". │ │ │ │ │ + * APIProperty: imageFormat │ │ │ │ │ + * {String} The image format used for caching. The default is "image/png". │ │ │ │ │ + * Supported formats depend on the user agent. If an unsupported │ │ │ │ │ + * <imageFormat> is provided, "image/png" will be used. For aerial │ │ │ │ │ + * imagery, "image/jpeg" is recommended. │ │ │ │ │ */ │ │ │ │ │ - clickCallback: "click", │ │ │ │ │ + imageFormat: "image/png", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: layers │ │ │ │ │ - * {Array(<OpenLayers.Layer.WMTS>)} The layers to query for feature info. │ │ │ │ │ - * If omitted, all map WMTS layers will be considered. │ │ │ │ │ + * Property: quotaRegEx │ │ │ │ │ + * {RegExp} │ │ │ │ │ */ │ │ │ │ │ - layers: null, │ │ │ │ │ + quotaRegEx: (/quota/i), │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: queryVisible │ │ │ │ │ - * {Boolean} Filter out hidden layers when searching the map for layers to │ │ │ │ │ - * query. Default is true. │ │ │ │ │ + * Constructor: OpenLayers.Control.CacheWrite │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Object with API properties for this control. │ │ │ │ │ */ │ │ │ │ │ - queryVisible: true, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: infoFormat │ │ │ │ │ - * {String} The mimetype to request from the server │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the control. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - infoFormat: 'text/html', │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + var i, layers = this.layers || map.layers; │ │ │ │ │ + for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + this.addLayer({ │ │ │ │ │ + layer: layers[i] │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + if (!this.layers) { │ │ │ │ │ + map.events.on({ │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + removeLayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: vendorParams │ │ │ │ │ - * {Object} Additional parameters that will be added to the request, for │ │ │ │ │ - * WMTS implementations that support them. This could e.g. look like │ │ │ │ │ - * (start code) │ │ │ │ │ - * { │ │ │ │ │ - * radius: 5 │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ + * Method: addLayer │ │ │ │ │ + * Adds a layer to the control. Once added, tiles requested for this layer │ │ │ │ │ + * will be cached. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} Object with a layer property referencing an │ │ │ │ │ + * <OpenLayers.Layer> instance │ │ │ │ │ */ │ │ │ │ │ - vendorParams: {}, │ │ │ │ │ + addLayer: function(evt) { │ │ │ │ │ + evt.layer.events.on({ │ │ │ │ │ + tileloadstart: this.makeSameOrigin, │ │ │ │ │ + tileloaded: this.onTileLoaded, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: format │ │ │ │ │ - * {<OpenLayers.Format>} A format for parsing GetFeatureInfo responses. │ │ │ │ │ - * Default is <OpenLayers.Format.WMSGetFeatureInfo>. │ │ │ │ │ + * Method: removeLayer │ │ │ │ │ + * Removes a layer from the control. Once removed, tiles requested for this │ │ │ │ │ + * layer will no longer be cached. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} Object with a layer property referencing an │ │ │ │ │ + * <OpenLayers.Layer> instance │ │ │ │ │ */ │ │ │ │ │ - format: null, │ │ │ │ │ + removeLayer: function(evt) { │ │ │ │ │ + evt.layer.events.un({ │ │ │ │ │ + tileloadstart: this.makeSameOrigin, │ │ │ │ │ + tileloaded: this.onTileLoaded, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: formatOptions │ │ │ │ │ - * {Object} Optional properties to set on the format (if one is not provided │ │ │ │ │ - * in the <format> property. │ │ │ │ │ + * Method: makeSameOrigin │ │ │ │ │ + * If the tile does not have CORS image loading enabled and is from a │ │ │ │ │ + * different origin, use OpenLayers.ProxyHost to make it a same origin url. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - formatOptions: null, │ │ │ │ │ + makeSameOrigin: function(evt) { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + var tile = evt.tile; │ │ │ │ │ + if (tile instanceof OpenLayers.Tile.Image && │ │ │ │ │ + !tile.crossOriginKeyword && │ │ │ │ │ + tile.url.substr(0, 5) !== "data:") { │ │ │ │ │ + var sameOriginUrl = OpenLayers.Request.makeSameOrigin( │ │ │ │ │ + tile.url, OpenLayers.ProxyHost │ │ │ │ │ + ); │ │ │ │ │ + OpenLayers.Control.CacheWrite.urlMap[sameOriginUrl] = tile.url; │ │ │ │ │ + tile.url = sameOriginUrl; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: handlerOptions │ │ │ │ │ - * {Object} Additional options for the handlers used by this control, e.g. │ │ │ │ │ - * (start code) │ │ │ │ │ - * { │ │ │ │ │ - * "click": {delay: 100}, │ │ │ │ │ - * "hover": {delay: 300} │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ + * Method: onTileLoaded │ │ │ │ │ + * Decides whether a tile can be cached and calls the cache method. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ + onTileLoaded: function(evt) { │ │ │ │ │ + if (this.active && !evt.aborted && │ │ │ │ │ + evt.tile instanceof OpenLayers.Tile.Image && │ │ │ │ │ + evt.tile.url.substr(0, 5) !== 'data:') { │ │ │ │ │ + this.cache({ │ │ │ │ │ + tile: evt.tile │ │ │ │ │ + }); │ │ │ │ │ + delete OpenLayers.Control.CacheWrite.urlMap[evt.tile.url]; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: handler │ │ │ │ │ - * {Object} Reference to the <OpenLayers.Handler> for this control │ │ │ │ │ + * Method: cache │ │ │ │ │ + * Adds a tile to the cache. When the cache is full, the "cachefull" event │ │ │ │ │ + * is triggered. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} Object with a tile property, tile being the │ │ │ │ │ + * <OpenLayers.Tile.Image> with the data to add to the cache │ │ │ │ │ */ │ │ │ │ │ - handler: null, │ │ │ │ │ + cache: function(obj) { │ │ │ │ │ + if (window.localStorage) { │ │ │ │ │ + var tile = obj.tile; │ │ │ │ │ + try { │ │ │ │ │ + var canvasContext = tile.getCanvasContext(); │ │ │ │ │ + if (canvasContext) { │ │ │ │ │ + var urlMap = OpenLayers.Control.CacheWrite.urlMap; │ │ │ │ │ + var url = urlMap[tile.url] || tile.url; │ │ │ │ │ + window.localStorage.setItem( │ │ │ │ │ + "olCache_" + url, │ │ │ │ │ + canvasContext.canvas.toDataURL(this.imageFormat) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } catch (e) { │ │ │ │ │ + // local storage full or CORS violation │ │ │ │ │ + var reason = e.name || e.message; │ │ │ │ │ + if (reason && this.quotaRegEx.test(reason)) { │ │ │ │ │ + this.events.triggerEvent("cachefull", { │ │ │ │ │ + tile: tile │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Console.error(e.toString()); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: hoverRequest │ │ │ │ │ - * {<OpenLayers.Request>} contains the currently running hover request │ │ │ │ │ - * (if any). │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * The destroy method is used to perform any clean up before the control │ │ │ │ │ + * is dereferenced. Typically this is where event listeners are removed │ │ │ │ │ + * to prevent memory leaks. │ │ │ │ │ */ │ │ │ │ │ - hoverRequest: null, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.layers || this.map) { │ │ │ │ │ + var i, layers = this.layers || this.map.layers; │ │ │ │ │ + for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + this.removeLayer({ │ │ │ │ │ + layer: layers[i] │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + removeLayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.CacheWrite" │ │ │ │ │ +}); │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * APIFunction: OpenLayers.Control.CacheWrite.clearCache │ │ │ │ │ + * Clears all tiles cached with <OpenLayers.Control.CacheWrite> from the cache. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.CacheWrite.clearCache = function() { │ │ │ │ │ + if (!window.localStorage) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var i, key; │ │ │ │ │ + for (i = window.localStorage.length - 1; i >= 0; --i) { │ │ │ │ │ + key = window.localStorage.key(i); │ │ │ │ │ + if (key.substr(0, 8) === "olCache_") { │ │ │ │ │ + window.localStorage.removeItem(key); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Property: OpenLayers.Control.CacheWrite.urlMap │ │ │ │ │ + * {Object} Mapping of same origin urls to cache url keys. Entries will be │ │ │ │ │ + * deleted as soon as a tile was cached. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.CacheWrite.urlMap = {}; │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/PanPanel.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control/Panel.js │ │ │ │ │ + * @requires OpenLayers/Control/Pan.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.PanPanel │ │ │ │ │ + * The PanPanel is visible control for panning the map North, South, East or │ │ │ │ │ + * West in small steps. By default it is drawn in the top left corner of the │ │ │ │ │ + * map. │ │ │ │ │ + * │ │ │ │ │ + * Note: │ │ │ │ │ + * If you wish to use this class with the default images and you want │ │ │ │ │ + * it to look nice in ie6, you should add the following, conditionally │ │ │ │ │ + * added css stylesheet to your HTML file: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * <!--[if lte IE 6]> │ │ │ │ │ + * <link rel="stylesheet" href="../theme/default/ie6-style.css" type="text/css" /> │ │ │ │ │ + * <![endif]--> │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control.Panel> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.PanPanel = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * beforegetfeatureinfo - Triggered before each request is sent. │ │ │ │ │ - * The event object has an *xy* property with the position of the │ │ │ │ │ - * mouse click or hover event that triggers the request and a *layer* │ │ │ │ │ - * property referencing the layer about to be queried. If a listener │ │ │ │ │ - * returns false, the request will not be issued. │ │ │ │ │ - * getfeatureinfo - Triggered when a GetFeatureInfo response is received. │ │ │ │ │ - * The event object has a *text* property with the body of the │ │ │ │ │ - * response (String), a *features* property with an array of the │ │ │ │ │ - * parsed features, an *xy* property with the position of the mouse │ │ │ │ │ - * click or hover event that triggered the request, a *layer* property │ │ │ │ │ - * referencing the layer queried and a *request* property with the │ │ │ │ │ - * request itself. If drillDown is set to true, one event will be fired │ │ │ │ │ - * for each layer queried. │ │ │ │ │ - * exception - Triggered when a GetFeatureInfo request fails (with a │ │ │ │ │ - * status other than 200) or whenparsing fails. Listeners will receive │ │ │ │ │ - * an event with *request*, *xy*, and *layer* properties. In the case │ │ │ │ │ - * of a parsing error, the event will also contain an *error* property. │ │ │ │ │ + * APIProperty: slideFactor │ │ │ │ │ + * {Integer} Number of pixels by which we'll pan the map in any direction │ │ │ │ │ + * on clicking the arrow buttons, defaults to 50. If you want to pan │ │ │ │ │ + * by some ratio of the map dimensions, use <slideRatio> instead. │ │ │ │ │ */ │ │ │ │ │ + slideFactor: 50, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: pending │ │ │ │ │ - * {Number} The number of pending requests. │ │ │ │ │ + * APIProperty: slideRatio │ │ │ │ │ + * {Number} The fraction of map width/height by which we'll pan the map │ │ │ │ │ + * on clicking the arrow buttons. Default is null. If set, will │ │ │ │ │ + * override <slideFactor>. E.g. if slideRatio is .5, then Pan Up will │ │ │ │ │ + * pan up half the map height. │ │ │ │ │ */ │ │ │ │ │ - pending: 0, │ │ │ │ │ + slideRatio: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: <OpenLayers.Control.WMTSGetFeatureInfo> │ │ │ │ │ + * Constructor: OpenLayers.Control.PanPanel │ │ │ │ │ + * Add the four directional pan buttons. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * options - {Object} An optional object whose properties will be used │ │ │ │ │ + * to extend the control. │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ + OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ + var options = { │ │ │ │ │ + slideFactor: this.slideFactor, │ │ │ │ │ + slideRatio: this.slideRatio │ │ │ │ │ + }; │ │ │ │ │ + this.addControls([ │ │ │ │ │ + new OpenLayers.Control.Pan(OpenLayers.Control.Pan.NORTH, options), │ │ │ │ │ + new OpenLayers.Control.Pan(OpenLayers.Control.Pan.SOUTH, options), │ │ │ │ │ + new OpenLayers.Control.Pan(OpenLayers.Control.Pan.EAST, options), │ │ │ │ │ + new OpenLayers.Control.Pan(OpenLayers.Control.Pan.WEST, options) │ │ │ │ │ + ]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.PanPanel" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/MousePosition.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - if (!this.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.WMSGetFeatureInfo( │ │ │ │ │ - options.formatOptions │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - if (this.drillDown === true) { │ │ │ │ │ - this.hover = false; │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Hover( │ │ │ │ │ - this, { │ │ │ │ │ - move: this.cancelHover, │ │ │ │ │ - pause: this.getInfoForHover │ │ │ │ │ - }, │ │ │ │ │ - OpenLayers.Util.extend( │ │ │ │ │ - this.handlerOptions.hover || {}, { │ │ │ │ │ - delay: 250 │ │ │ │ │ - } │ │ │ │ │ - ) │ │ │ │ │ - ); │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.MousePosition │ │ │ │ │ + * The MousePosition control displays geographic coordinates of the mouse │ │ │ │ │ + * pointer, as it is moved about the map. │ │ │ │ │ + * │ │ │ │ │ + * You can use the <prefix>- or <suffix>-properties to provide more information │ │ │ │ │ + * about the displayed coordinates to the user: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * var mousePositionCtrl = new OpenLayers.Control.MousePosition({ │ │ │ │ │ + * prefix: '<a target="_blank" ' + │ │ │ │ │ + * 'href="http://spatialreference.org/ref/epsg/4326/">' + │ │ │ │ │ + * 'EPSG:4326</a> coordinates: ' │ │ │ │ │ + * } │ │ │ │ │ + * ); │ │ │ │ │ + * (end code) │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.MousePosition = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ + */ │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: element │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + element: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: prefix │ │ │ │ │ + * {String} A string to be prepended to the current pointers coordinates │ │ │ │ │ + * when it is rendered. Defaults to the empty string ''. │ │ │ │ │ + */ │ │ │ │ │ + prefix: '', │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: separator │ │ │ │ │ + * {String} A string to be used to seperate the two coordinates from each │ │ │ │ │ + * other. Defaults to the string ', ', which will result in a │ │ │ │ │ + * rendered coordinate of e.g. '42.12, 21.22'. │ │ │ │ │ + */ │ │ │ │ │ + separator: ', ', │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: suffix │ │ │ │ │ + * {String} A string to be appended to the current pointers coordinates │ │ │ │ │ + * when it is rendered. Defaults to the empty string ''. │ │ │ │ │ + */ │ │ │ │ │ + suffix: '', │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: numDigits │ │ │ │ │ + * {Integer} The number of digits each coordinate shall have when being │ │ │ │ │ + * rendered, Defaults to 5. │ │ │ │ │ + */ │ │ │ │ │ + numDigits: 5, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: granularity │ │ │ │ │ + * {Integer} │ │ │ │ │ + */ │ │ │ │ │ + granularity: 10, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: emptyString │ │ │ │ │ + * {String} Set this to some value to set when the mouse is outside the │ │ │ │ │ + * map. │ │ │ │ │ + */ │ │ │ │ │ + emptyString: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: lastXy │ │ │ │ │ + * {<OpenLayers.Pixel>} │ │ │ │ │ + */ │ │ │ │ │ + lastXy: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: displayProjection │ │ │ │ │ + * {<OpenLayers.Projection>} The projection in which the mouse position is │ │ │ │ │ + * displayed. │ │ │ │ │ + */ │ │ │ │ │ + displayProjection: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.MousePosition │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Options for control. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.map.events.register('mousemove', this, this.redraw); │ │ │ │ │ + this.map.events.register('mouseout', this, this.reset); │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return true; │ │ │ │ │ } else { │ │ │ │ │ - var callbacks = {}; │ │ │ │ │ - callbacks[this.clickCallback] = this.getInfoForClick; │ │ │ │ │ - this.handler = new OpenLayers.Handler.Click( │ │ │ │ │ - this, callbacks, this.handlerOptions.click || {} │ │ │ │ │ - ); │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getInfoForClick │ │ │ │ │ - * Called on click │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ */ │ │ │ │ │ - getInfoForClick: function(evt) { │ │ │ │ │ - this.request(evt.xy, {}); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.map.events.unregister('mousemove', this, this.redraw); │ │ │ │ │ + this.map.events.unregister('mouseout', this, this.reset); │ │ │ │ │ + this.element.innerHTML = ""; │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getInfoForHover │ │ │ │ │ - * Pause callback for the hover handler │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ + * Method: draw │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - getInfoForHover: function(evt) { │ │ │ │ │ - this.request(evt.xy, { │ │ │ │ │ - hover: true │ │ │ │ │ - }); │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + if (!this.element) { │ │ │ │ │ + this.div.left = ""; │ │ │ │ │ + this.div.top = ""; │ │ │ │ │ + this.element = this.div; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: cancelHover │ │ │ │ │ - * Cancel callback for the hover handler │ │ │ │ │ + * Method: redraw │ │ │ │ │ */ │ │ │ │ │ - cancelHover: function() { │ │ │ │ │ - if (this.hoverRequest) { │ │ │ │ │ - --this.pending; │ │ │ │ │ - if (this.pending <= 0) { │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - this.pending = 0; │ │ │ │ │ + redraw: function(evt) { │ │ │ │ │ + │ │ │ │ │ + var lonLat; │ │ │ │ │ + │ │ │ │ │ + if (evt == null) { │ │ │ │ │ + this.reset(); │ │ │ │ │ + return; │ │ │ │ │ + } else { │ │ │ │ │ + if (this.lastXy == null || │ │ │ │ │ + Math.abs(evt.xy.x - this.lastXy.x) > this.granularity || │ │ │ │ │ + Math.abs(evt.xy.y - this.lastXy.y) > this.granularity) { │ │ │ │ │ + this.lastXy = evt.xy; │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ - this.hoverRequest.abort(); │ │ │ │ │ - this.hoverRequest = null; │ │ │ │ │ + │ │ │ │ │ + lonLat = this.map.getLonLatFromPixel(evt.xy); │ │ │ │ │ + if (!lonLat) { │ │ │ │ │ + // map has not yet been properly initialized │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + if (this.displayProjection) { │ │ │ │ │ + lonLat.transform(this.map.getProjectionObject(), │ │ │ │ │ + this.displayProjection); │ │ │ │ │ + } │ │ │ │ │ + this.lastXy = evt.xy; │ │ │ │ │ + │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var newHtml = this.formatOutput(lonLat); │ │ │ │ │ + │ │ │ │ │ + if (newHtml != this.element.innerHTML) { │ │ │ │ │ + this.element.innerHTML = newHtml; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: findLayers │ │ │ │ │ - * Internal method to get the layers, independent of whether we are │ │ │ │ │ - * inspecting the map or using a client-provided array │ │ │ │ │ + * Method: reset │ │ │ │ │ */ │ │ │ │ │ - findLayers: function() { │ │ │ │ │ - var candidates = this.layers || this.map.layers; │ │ │ │ │ - var layers = []; │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = candidates[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.WMTS && │ │ │ │ │ - layer.requestEncoding === this.requestEncoding && │ │ │ │ │ - (!this.queryVisible || layer.getVisibility())) { │ │ │ │ │ - layers.push(layer); │ │ │ │ │ - if (!this.drillDown || this.hover) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + reset: function(evt) { │ │ │ │ │ + if (this.emptyString != null) { │ │ │ │ │ + this.element.innerHTML = this.emptyString; │ │ │ │ │ } │ │ │ │ │ - return layers; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: buildRequestOptions │ │ │ │ │ - * Build an object with the relevant options for the GetFeatureInfo request. │ │ │ │ │ + * Method: formatOutput │ │ │ │ │ + * Override to provide custom display output │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.WMTS>} A WMTS layer. │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} The position on the map where the │ │ │ │ │ - * mouse event occurred. │ │ │ │ │ + * lonLat - {<OpenLayers.LonLat>} Location to display │ │ │ │ │ */ │ │ │ │ │ - buildRequestOptions: function(layer, xy) { │ │ │ │ │ - var loc = this.map.getLonLatFromPixel(xy); │ │ │ │ │ - var getTileUrl = layer.getURL( │ │ │ │ │ - new OpenLayers.Bounds(loc.lon, loc.lat, loc.lon, loc.lat) │ │ │ │ │ - ); │ │ │ │ │ - var params = OpenLayers.Util.getParameters(getTileUrl); │ │ │ │ │ - var tileInfo = layer.getTileInfo(loc); │ │ │ │ │ - OpenLayers.Util.extend(params, { │ │ │ │ │ - service: "WMTS", │ │ │ │ │ - version: layer.version, │ │ │ │ │ - request: "GetFeatureInfo", │ │ │ │ │ - infoFormat: this.infoFormat, │ │ │ │ │ - i: tileInfo.i, │ │ │ │ │ - j: tileInfo.j │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Util.applyDefaults(params, this.vendorParams); │ │ │ │ │ - return { │ │ │ │ │ - url: OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url, │ │ │ │ │ - params: OpenLayers.Util.upperCaseObject(params), │ │ │ │ │ - callback: function(request) { │ │ │ │ │ - this.handleResponse(xy, request, layer); │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }; │ │ │ │ │ + formatOutput: function(lonLat) { │ │ │ │ │ + var digits = parseInt(this.numDigits); │ │ │ │ │ + var newHtml = │ │ │ │ │ + this.prefix + │ │ │ │ │ + lonLat.lon.toFixed(digits) + │ │ │ │ │ + this.separator + │ │ │ │ │ + lonLat.lat.toFixed(digits) + │ │ │ │ │ + this.suffix; │ │ │ │ │ + return newHtml; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.MousePosition" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/Graticule.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Rule.js │ │ │ │ │ + * @requires OpenLayers/StyleMap.js │ │ │ │ │ + * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.Graticule │ │ │ │ │ + * The Graticule displays a grid of latitude/longitude lines reprojected on │ │ │ │ │ + * the map. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.Graticule = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: request │ │ │ │ │ - * Sends a GetFeatureInfo request to the WMTS │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ + */ │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: intervals │ │ │ │ │ + * {Array(Float)} A list of possible graticule widths in degrees. │ │ │ │ │ + */ │ │ │ │ │ + intervals: [45, 30, 20, 10, 5, 2, 1, │ │ │ │ │ + 0.5, 0.2, 0.1, 0.05, 0.01, │ │ │ │ │ + 0.005, 0.002, 0.001 │ │ │ │ │ + ], │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: displayInLayerSwitcher │ │ │ │ │ + * {Boolean} Allows the Graticule control to be switched on and off by │ │ │ │ │ + * LayerSwitcher control. Defaults is true. │ │ │ │ │ + */ │ │ │ │ │ + displayInLayerSwitcher: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: visible │ │ │ │ │ + * {Boolean} should the graticule be initially visible (default=true) │ │ │ │ │ + */ │ │ │ │ │ + visible: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: numPoints │ │ │ │ │ + * {Integer} The number of points to use in each graticule line. Higher │ │ │ │ │ + * numbers result in a smoother curve for projected maps │ │ │ │ │ + */ │ │ │ │ │ + numPoints: 50, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: targetSize │ │ │ │ │ + * {Integer} The maximum size of the grid in pixels on the map │ │ │ │ │ + */ │ │ │ │ │ + targetSize: 200, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layerName │ │ │ │ │ + * {String} The name to be displayed in the layer switcher, default is set │ │ │ │ │ + * by {<OpenLayers.Lang>}. │ │ │ │ │ + */ │ │ │ │ │ + layerName: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: labelled │ │ │ │ │ + * {Boolean} Should the graticule lines be labelled?. default=true │ │ │ │ │ + */ │ │ │ │ │ + labelled: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: labelFormat │ │ │ │ │ + * {String} the format of the labels, default = 'dm'. See │ │ │ │ │ + * <OpenLayers.Util.getFormattedLonLat> for other options. │ │ │ │ │ + */ │ │ │ │ │ + labelFormat: 'dm', │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: lineSymbolizer │ │ │ │ │ + * {symbolizer} the symbolizer used to render lines │ │ │ │ │ + */ │ │ │ │ │ + lineSymbolizer: { │ │ │ │ │ + strokeColor: "#333", │ │ │ │ │ + strokeWidth: 1, │ │ │ │ │ + strokeOpacity: 0.5 │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: labelSymbolizer │ │ │ │ │ + * {symbolizer} the symbolizer used to render labels │ │ │ │ │ + */ │ │ │ │ │ + labelSymbolizer: {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: gratLayer │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} vector layer used to draw the graticule on │ │ │ │ │ + */ │ │ │ │ │ + gratLayer: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.Graticule │ │ │ │ │ + * Create a new graticule control to display a grid of latitude longitude │ │ │ │ │ + * lines. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} The position on the map where the mouse event │ │ │ │ │ - * occurred. │ │ │ │ │ - * options - {Object} additional options for this method. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * - *hover* {Boolean} true if we do the request for the hover handler │ │ │ │ │ + * options - {Object} An optional object whose properties will be used │ │ │ │ │ + * to extend the control. │ │ │ │ │ */ │ │ │ │ │ - request: function(xy, options) { │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ options = options || {}; │ │ │ │ │ - var layers = this.findLayers(); │ │ │ │ │ - if (layers.length > 0) { │ │ │ │ │ - var issue, layer; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - layer = layers[i]; │ │ │ │ │ - issue = this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ - xy: xy, │ │ │ │ │ - layer: layer │ │ │ │ │ - }); │ │ │ │ │ - if (issue !== false) { │ │ │ │ │ - ++this.pending; │ │ │ │ │ - var requestOptions = this.buildRequestOptions(layer, xy); │ │ │ │ │ - var request = OpenLayers.Request.GET(requestOptions); │ │ │ │ │ - if (options.hover === true) { │ │ │ │ │ - this.hoverRequest = request; │ │ │ │ │ + options.layerName = options.layerName || OpenLayers.i18n("Graticule"); │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + │ │ │ │ │ + this.labelSymbolizer.stroke = false; │ │ │ │ │ + this.labelSymbolizer.fill = false; │ │ │ │ │ + this.labelSymbolizer.label = "${label}"; │ │ │ │ │ + this.labelSymbolizer.labelAlign = "${labelAlign}"; │ │ │ │ │ + this.labelSymbolizer.labelXOffset = "${xOffset}"; │ │ │ │ │ + this.labelSymbolizer.labelYOffset = "${yOffset}"; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + if (this.gratLayer) { │ │ │ │ │ + this.gratLayer.destroy(); │ │ │ │ │ + this.gratLayer = null; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * │ │ │ │ │ + * initializes the graticule layer and does the initial update │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (!this.gratLayer) { │ │ │ │ │ + var gratStyle = new OpenLayers.Style({}, { │ │ │ │ │ + rules: [new OpenLayers.Rule({ │ │ │ │ │ + 'symbolizer': { │ │ │ │ │ + "Point": this.labelSymbolizer, │ │ │ │ │ + "Line": this.lineSymbolizer │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.pending > 0) { │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - } │ │ │ │ │ + })] │ │ │ │ │ + }); │ │ │ │ │ + this.gratLayer = new OpenLayers.Layer.Vector(this.layerName, { │ │ │ │ │ + styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ + 'default': gratStyle │ │ │ │ │ + }), │ │ │ │ │ + visibility: this.visible, │ │ │ │ │ + displayInLayerSwitcher: this.displayInLayerSwitcher │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ + return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleResponse │ │ │ │ │ - * Handler for the GetFeatureInfo response. │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.map.addLayer(this.gratLayer); │ │ │ │ │ + this.map.events.register('moveend', this, this.update); │ │ │ │ │ + this.update(); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.map.events.unregister('moveend', this, this.update); │ │ │ │ │ + this.map.removeLayer(this.gratLayer); │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + /** │ │ │ │ │ + * Method: update │ │ │ │ │ + * │ │ │ │ │ + * calculates the grid to be displayed and actually draws it │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} The position on the map where the mouse event │ │ │ │ │ - * occurred. │ │ │ │ │ - * request - {XMLHttpRequest} The request object. │ │ │ │ │ - * layer - {<OpenLayers.Layer.WMTS>} The queried layer. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - handleResponse: function(xy, request, layer) { │ │ │ │ │ - --this.pending; │ │ │ │ │ - if (this.pending <= 0) { │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - this.pending = 0; │ │ │ │ │ + update: function() { │ │ │ │ │ + //wait for the map to be initialized before proceeding │ │ │ │ │ + var mapBounds = this.map.getExtent(); │ │ │ │ │ + if (!mapBounds) { │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ - if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ - this.events.triggerEvent("exception", { │ │ │ │ │ - xy: xy, │ │ │ │ │ - request: request, │ │ │ │ │ - layer: layer │ │ │ │ │ + │ │ │ │ │ + //clear out the old grid │ │ │ │ │ + this.gratLayer.destroyFeatures(); │ │ │ │ │ + │ │ │ │ │ + //get the projection objects required │ │ │ │ │ + var llProj = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + var mapProj = this.map.getProjectionObject(); │ │ │ │ │ + var mapRes = this.map.getResolution(); │ │ │ │ │ + │ │ │ │ │ + //if the map is in lon/lat, then the lines are straight and only one │ │ │ │ │ + //point is required │ │ │ │ │ + if (mapProj.proj && mapProj.proj.projName == "longlat") { │ │ │ │ │ + this.numPoints = 1; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //get the map center in EPSG:4326 │ │ │ │ │ + var mapCenter = this.map.getCenter(); //lon and lat here are really map x and y │ │ │ │ │ + var mapCenterLL = new OpenLayers.Pixel(mapCenter.lon, mapCenter.lat); │ │ │ │ │ + OpenLayers.Projection.transform(mapCenterLL, mapProj, llProj); │ │ │ │ │ + │ │ │ │ │ + /* This block of code determines the lon/lat interval to use for the │ │ │ │ │ + * grid by calculating the diagonal size of one grid cell at the map │ │ │ │ │ + * center. Iterates through the intervals array until the diagonal │ │ │ │ │ + * length is less than the targetSize option. │ │ │ │ │ + */ │ │ │ │ │ + //find lat/lon interval that results in a grid of less than the target size │ │ │ │ │ + var testSq = this.targetSize * mapRes; │ │ │ │ │ + testSq *= testSq; //compare squares rather than doing a square root to save time │ │ │ │ │ + var llInterval; │ │ │ │ │ + for (var i = 0; i < this.intervals.length; ++i) { │ │ │ │ │ + llInterval = this.intervals[i]; //could do this for both x and y?? │ │ │ │ │ + var delta = llInterval / 2; │ │ │ │ │ + var p1 = mapCenterLL.offset({ │ │ │ │ │ + x: -delta, │ │ │ │ │ + y: -delta │ │ │ │ │ + }); //test coords in EPSG:4326 space │ │ │ │ │ + var p2 = mapCenterLL.offset({ │ │ │ │ │ + x: delta, │ │ │ │ │ + y: delta │ │ │ │ │ }); │ │ │ │ │ - } else { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ + OpenLayers.Projection.transform(p1, llProj, mapProj); // convert them back to map projection │ │ │ │ │ + OpenLayers.Projection.transform(p2, llProj, mapProj); │ │ │ │ │ + var distSq = (p1.x - p2.x) * (p1.x - p2.x) + (p1.y - p2.y) * (p1.y - p2.y); │ │ │ │ │ + if (distSq <= testSq) { │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ - var features, except; │ │ │ │ │ - try { │ │ │ │ │ - features = this.format.read(doc); │ │ │ │ │ - } catch (error) { │ │ │ │ │ - except = true; │ │ │ │ │ - this.events.triggerEvent("exception", { │ │ │ │ │ - xy: xy, │ │ │ │ │ - request: request, │ │ │ │ │ - error: error, │ │ │ │ │ - layer: layer │ │ │ │ │ - }); │ │ │ │ │ + } │ │ │ │ │ + //alert(llInterval); │ │ │ │ │ + │ │ │ │ │ + //round the LL center to an even number based on the interval │ │ │ │ │ + mapCenterLL.x = Math.floor(mapCenterLL.x / llInterval) * llInterval; │ │ │ │ │ + mapCenterLL.y = Math.floor(mapCenterLL.y / llInterval) * llInterval; │ │ │ │ │ + //TODO adjust for minutses/seconds? │ │ │ │ │ + │ │ │ │ │ + /* The following 2 blocks calculate the nodes of the grid along a │ │ │ │ │ + * line of constant longitude (then latitiude) running through the │ │ │ │ │ + * center of the map until it reaches the map edge. The calculation │ │ │ │ │ + * goes from the center in both directions to the edge. │ │ │ │ │ + */ │ │ │ │ │ + //get the central longitude line, increment the latitude │ │ │ │ │ + var iter = 0; │ │ │ │ │ + var centerLonPoints = [mapCenterLL.clone()]; │ │ │ │ │ + var newPoint = mapCenterLL.clone(); │ │ │ │ │ + var mapXY; │ │ │ │ │ + do { │ │ │ │ │ + newPoint = newPoint.offset({ │ │ │ │ │ + x: 0, │ │ │ │ │ + y: llInterval │ │ │ │ │ + }); │ │ │ │ │ + mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ + centerLonPoints.unshift(newPoint); │ │ │ │ │ + } while (mapBounds.containsPixel(mapXY) && ++iter < 1000); │ │ │ │ │ + newPoint = mapCenterLL.clone(); │ │ │ │ │ + do { │ │ │ │ │ + newPoint = newPoint.offset({ │ │ │ │ │ + x: 0, │ │ │ │ │ + y: -llInterval │ │ │ │ │ + }); │ │ │ │ │ + mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ + centerLonPoints.push(newPoint); │ │ │ │ │ + } while (mapBounds.containsPixel(mapXY) && ++iter < 1000); │ │ │ │ │ + │ │ │ │ │ + //get the central latitude line, increment the longitude │ │ │ │ │ + iter = 0; │ │ │ │ │ + var centerLatPoints = [mapCenterLL.clone()]; │ │ │ │ │ + newPoint = mapCenterLL.clone(); │ │ │ │ │ + do { │ │ │ │ │ + newPoint = newPoint.offset({ │ │ │ │ │ + x: -llInterval, │ │ │ │ │ + y: 0 │ │ │ │ │ + }); │ │ │ │ │ + mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ + centerLatPoints.unshift(newPoint); │ │ │ │ │ + } while (mapBounds.containsPixel(mapXY) && ++iter < 1000); │ │ │ │ │ + newPoint = mapCenterLL.clone(); │ │ │ │ │ + do { │ │ │ │ │ + newPoint = newPoint.offset({ │ │ │ │ │ + x: llInterval, │ │ │ │ │ + y: 0 │ │ │ │ │ + }); │ │ │ │ │ + mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ + centerLatPoints.push(newPoint); │ │ │ │ │ + } while (mapBounds.containsPixel(mapXY) && ++iter < 1000); │ │ │ │ │ + │ │ │ │ │ + //now generate a line for each node in the central lat and lon lines │ │ │ │ │ + //first loop over constant longitude │ │ │ │ │ + var lines = []; │ │ │ │ │ + for (var i = 0; i < centerLatPoints.length; ++i) { │ │ │ │ │ + var lon = centerLatPoints[i].x; │ │ │ │ │ + var pointList = []; │ │ │ │ │ + var labelPoint = null; │ │ │ │ │ + var latEnd = Math.min(centerLonPoints[0].y, 90); │ │ │ │ │ + var latStart = Math.max(centerLonPoints[centerLonPoints.length - 1].y, -90); │ │ │ │ │ + var latDelta = (latEnd - latStart) / this.numPoints; │ │ │ │ │ + var lat = latStart; │ │ │ │ │ + for (var j = 0; j <= this.numPoints; ++j) { │ │ │ │ │ + var gridPoint = new OpenLayers.Geometry.Point(lon, lat); │ │ │ │ │ + gridPoint.transform(llProj, mapProj); │ │ │ │ │ + pointList.push(gridPoint); │ │ │ │ │ + lat += latDelta; │ │ │ │ │ + if (gridPoint.y >= mapBounds.bottom && !labelPoint) { │ │ │ │ │ + labelPoint = gridPoint; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (!except) { │ │ │ │ │ - this.events.triggerEvent("getfeatureinfo", { │ │ │ │ │ - text: request.responseText, │ │ │ │ │ - features: features, │ │ │ │ │ - request: request, │ │ │ │ │ - xy: xy, │ │ │ │ │ - layer: layer │ │ │ │ │ - }); │ │ │ │ │ + if (this.labelled) { │ │ │ │ │ + //keep track of when this grid line crosses the map bounds to set │ │ │ │ │ + //the label position │ │ │ │ │ + //labels along the bottom, add 10 pixel offset up into the map │ │ │ │ │ + //TODO add option for labels on top │ │ │ │ │ + var labelPos = new OpenLayers.Geometry.Point(labelPoint.x, mapBounds.bottom); │ │ │ │ │ + var labelAttrs = { │ │ │ │ │ + value: lon, │ │ │ │ │ + label: this.labelled ? OpenLayers.Util.getFormattedLonLat(lon, "lon", this.labelFormat) : "", │ │ │ │ │ + labelAlign: "cb", │ │ │ │ │ + xOffset: 0, │ │ │ │ │ + yOffset: 2 │ │ │ │ │ + }; │ │ │ │ │ + this.gratLayer.addFeatures(new OpenLayers.Feature.Vector(labelPos, labelAttrs)); │ │ │ │ │ + } │ │ │ │ │ + var geom = new OpenLayers.Geometry.LineString(pointList); │ │ │ │ │ + lines.push(new OpenLayers.Feature.Vector(geom)); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + //now draw the lines of constant latitude │ │ │ │ │ + for (var j = 0; j < centerLonPoints.length; ++j) { │ │ │ │ │ + lat = centerLonPoints[j].y; │ │ │ │ │ + if (lat < -90 || lat > 90) { //latitudes only valid between -90 and 90 │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ + var pointList = []; │ │ │ │ │ + var lonStart = centerLatPoints[0].x; │ │ │ │ │ + var lonEnd = centerLatPoints[centerLatPoints.length - 1].x; │ │ │ │ │ + var lonDelta = (lonEnd - lonStart) / this.numPoints; │ │ │ │ │ + var lon = lonStart; │ │ │ │ │ + var labelPoint = null; │ │ │ │ │ + for (var i = 0; i <= this.numPoints; ++i) { │ │ │ │ │ + var gridPoint = new OpenLayers.Geometry.Point(lon, lat); │ │ │ │ │ + gridPoint.transform(llProj, mapProj); │ │ │ │ │ + pointList.push(gridPoint); │ │ │ │ │ + lon += lonDelta; │ │ │ │ │ + if (gridPoint.x < mapBounds.right) { │ │ │ │ │ + labelPoint = gridPoint; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + if (this.labelled) { │ │ │ │ │ + //keep track of when this grid line crosses the map bounds to set │ │ │ │ │ + //the label position │ │ │ │ │ + //labels along the right, 30 pixel offset left into the map │ │ │ │ │ + //TODO add option for labels on left │ │ │ │ │ + var labelPos = new OpenLayers.Geometry.Point(mapBounds.right, labelPoint.y); │ │ │ │ │ + var labelAttrs = { │ │ │ │ │ + value: lat, │ │ │ │ │ + label: this.labelled ? OpenLayers.Util.getFormattedLonLat(lat, "lat", this.labelFormat) : "", │ │ │ │ │ + labelAlign: "rb", │ │ │ │ │ + xOffset: -2, │ │ │ │ │ + yOffset: 2 │ │ │ │ │ + }; │ │ │ │ │ + this.gratLayer.addFeatures(new OpenLayers.Feature.Vector(labelPos, labelAttrs)); │ │ │ │ │ + } │ │ │ │ │ + var geom = new OpenLayers.Geometry.LineString(pointList); │ │ │ │ │ + lines.push(new OpenLayers.Feature.Vector(geom)); │ │ │ │ │ } │ │ │ │ │ + this.gratLayer.addFeatures(lines); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.WMTSGetFeatureInfo" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Graticule" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/PinchZoom.js │ │ │ │ │ + OpenLayers/Control/Navigation.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Handler/Pinch.js │ │ │ │ │ + * @requires OpenLayers/Control/ZoomBox.js │ │ │ │ │ + * @requires OpenLayers/Control/DragPan.js │ │ │ │ │ + * @requires OpenLayers/Handler/MouseWheel.js │ │ │ │ │ + * @requires OpenLayers/Handler/Click.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.PinchZoom │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Control.Navigation │ │ │ │ │ + * The navigation control handles map browsing with mouse events (dragging, │ │ │ │ │ + * double-clicking, and scrolling the wheel). Create a new navigation │ │ │ │ │ + * control with the <OpenLayers.Control.Navigation> control. │ │ │ │ │ + * │ │ │ │ │ + * Note that this control is added to the map by default (if no controls │ │ │ │ │ + * array is sent in the options object to the <OpenLayers.Map> │ │ │ │ │ + * constructor). │ │ │ │ │ + * │ │ │ │ │ * Inherits: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.PinchZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Control.Navigation = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: type │ │ │ │ │ - * {OpenLayers.Control.TYPES} │ │ │ │ │ + * Property: dragPan │ │ │ │ │ + * {<OpenLayers.Control.DragPan>} │ │ │ │ │ */ │ │ │ │ │ - type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ + dragPan: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: pinchOrigin │ │ │ │ │ - * {Object} Cached object representing the pinch start (in pixels). │ │ │ │ │ + * APIProperty: dragPanOptions │ │ │ │ │ + * {Object} Options passed to the DragPan control. │ │ │ │ │ */ │ │ │ │ │ - pinchOrigin: null, │ │ │ │ │ + dragPanOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: currentCenter │ │ │ │ │ - * {Object} Cached object representing the latest pinch center (in pixels). │ │ │ │ │ + * Property: pinchZoom │ │ │ │ │ + * {<OpenLayers.Control.PinchZoom>} │ │ │ │ │ */ │ │ │ │ │ - currentCenter: null, │ │ │ │ │ + pinchZoom: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ + * APIProperty: pinchZoomOptions │ │ │ │ │ + * {Object} Options passed to the PinchZoom control. │ │ │ │ │ */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ + pinchZoomOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: preserveCenter │ │ │ │ │ - * {Boolean} Set this to true if you don't want the map center to change │ │ │ │ │ - * while pinching. For example you may want to set preserveCenter to │ │ │ │ │ - * true when the user location is being watched and you want to preserve │ │ │ │ │ - * the user location at the center of the map even if he zooms in or │ │ │ │ │ - * out using pinch. This property's value can be changed any time on an │ │ │ │ │ - * existing instance. Default is false. │ │ │ │ │ + * APIProperty: documentDrag │ │ │ │ │ + * {Boolean} Allow panning of the map by dragging outside map viewport. │ │ │ │ │ + * Default is false. │ │ │ │ │ */ │ │ │ │ │ - preserveCenter: false, │ │ │ │ │ + documentDrag: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: zoomBox │ │ │ │ │ + * {<OpenLayers.Control.ZoomBox>} │ │ │ │ │ + */ │ │ │ │ │ + zoomBox: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: handlerOptions │ │ │ │ │ - * {Object} Used to set non-default properties on the pinch handler │ │ │ │ │ + * APIProperty: zoomBoxEnabled │ │ │ │ │ + * {Boolean} Whether the user can draw a box to zoom │ │ │ │ │ */ │ │ │ │ │ + zoomBoxEnabled: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.PinchZoom │ │ │ │ │ - * Create a control for zooming with pinch gestures. This works on devices │ │ │ │ │ - * with multi-touch support. │ │ │ │ │ - * │ │ │ │ │ + * APIProperty: zoomWheelEnabled │ │ │ │ │ + * {Boolean} Whether the mousewheel should zoom the map │ │ │ │ │ + */ │ │ │ │ │ + zoomWheelEnabled: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: mouseWheelOptions │ │ │ │ │ + * {Object} Options passed to the MouseWheel control (only useful if │ │ │ │ │ + * <zoomWheelEnabled> is set to true). Default is no options for maps │ │ │ │ │ + * with fractionalZoom set to true, otherwise │ │ │ │ │ + * {cumulative: false, interval: 50, maxDelta: 6} │ │ │ │ │ + */ │ │ │ │ │ + mouseWheelOptions: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: handleRightClicks │ │ │ │ │ + * {Boolean} Whether or not to handle right clicks. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + handleRightClicks: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomBoxKeyMask │ │ │ │ │ + * {Integer} <OpenLayers.Handler> key code of the key, which has to be │ │ │ │ │ + * pressed, while drawing the zoom box with the mouse on the screen. │ │ │ │ │ + * You should probably set handleRightClicks to true if you use this │ │ │ │ │ + * with MOD_CTRL, to disable the context menu for machines which use │ │ │ │ │ + * CTRL-Click as a right click. │ │ │ │ │ + * Default: <OpenLayers.Handler.MOD_SHIFT> │ │ │ │ │ + */ │ │ │ │ │ + zoomBoxKeyMask: OpenLayers.Handler.MOD_SHIFT, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ + */ │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.Navigation │ │ │ │ │ + * Create a new navigation control │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ * options - {Object} An optional object whose properties will be set on │ │ │ │ │ * the control │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ + this.handlers = {}; │ │ │ │ │ OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.handler = new OpenLayers.Handler.Pinch(this, { │ │ │ │ │ - start: this.pinchStart, │ │ │ │ │ - move: this.pinchMove, │ │ │ │ │ - done: this.pinchDone │ │ │ │ │ - }, this.handlerOptions); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: pinchStart │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * The destroy method is used to perform any clean up before the control │ │ │ │ │ + * is dereferenced. Typically this is where event listeners are removed │ │ │ │ │ + * to prevent memory leaks. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + │ │ │ │ │ + if (this.dragPan) { │ │ │ │ │ + this.dragPan.destroy(); │ │ │ │ │ + } │ │ │ │ │ + this.dragPan = null; │ │ │ │ │ + │ │ │ │ │ + if (this.zoomBox) { │ │ │ │ │ + this.zoomBox.destroy(); │ │ │ │ │ + } │ │ │ │ │ + this.zoomBox = null; │ │ │ │ │ + │ │ │ │ │ + if (this.pinchZoom) { │ │ │ │ │ + this.pinchZoom.destroy(); │ │ │ │ │ + } │ │ │ │ │ + this.pinchZoom = null; │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: activate │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + this.dragPan.activate(); │ │ │ │ │ + if (this.zoomWheelEnabled) { │ │ │ │ │ + this.handlers.wheel.activate(); │ │ │ │ │ + } │ │ │ │ │ + this.handlers.click.activate(); │ │ │ │ │ + if (this.zoomBoxEnabled) { │ │ │ │ │ + this.zoomBox.activate(); │ │ │ │ │ + } │ │ │ │ │ + if (this.pinchZoom) { │ │ │ │ │ + this.pinchZoom.activate(); │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Control.prototype.activate.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.pinchZoom) { │ │ │ │ │ + this.pinchZoom.deactivate(); │ │ │ │ │ + } │ │ │ │ │ + this.zoomBox.deactivate(); │ │ │ │ │ + this.dragPan.deactivate(); │ │ │ │ │ + this.handlers.click.deactivate(); │ │ │ │ │ + this.handlers.wheel.deactivate(); │ │ │ │ │ + return OpenLayers.Control.prototype.deactivate.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + */ │ │ │ │ │ + draw: function() { │ │ │ │ │ + // disable right mouse context menu for support of right click events │ │ │ │ │ + if (this.handleRightClicks) { │ │ │ │ │ + this.map.viewPortDiv.oncontextmenu = OpenLayers.Function.False; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var clickCallbacks = { │ │ │ │ │ + 'click': this.defaultClick, │ │ │ │ │ + 'dblclick': this.defaultDblClick, │ │ │ │ │ + 'dblrightclick': this.defaultDblRightClick │ │ │ │ │ + }; │ │ │ │ │ + var clickOptions = { │ │ │ │ │ + 'double': true, │ │ │ │ │ + 'stopDouble': true │ │ │ │ │ + }; │ │ │ │ │ + this.handlers.click = new OpenLayers.Handler.Click( │ │ │ │ │ + this, clickCallbacks, clickOptions │ │ │ │ │ + ); │ │ │ │ │ + this.dragPan = new OpenLayers.Control.DragPan( │ │ │ │ │ + OpenLayers.Util.extend({ │ │ │ │ │ + map: this.map, │ │ │ │ │ + documentDrag: this.documentDrag │ │ │ │ │ + }, this.dragPanOptions) │ │ │ │ │ + ); │ │ │ │ │ + this.zoomBox = new OpenLayers.Control.ZoomBox({ │ │ │ │ │ + map: this.map, │ │ │ │ │ + keyMask: this.zoomBoxKeyMask │ │ │ │ │ + }); │ │ │ │ │ + this.dragPan.draw(); │ │ │ │ │ + this.zoomBox.draw(); │ │ │ │ │ + var wheelOptions = this.map.fractionalZoom ? {} : { │ │ │ │ │ + cumulative: false, │ │ │ │ │ + interval: 50, │ │ │ │ │ + maxDelta: 6 │ │ │ │ │ + }; │ │ │ │ │ + this.handlers.wheel = new OpenLayers.Handler.MouseWheel( │ │ │ │ │ + this, { │ │ │ │ │ + up: this.wheelUp, │ │ │ │ │ + down: this.wheelDown │ │ │ │ │ + }, │ │ │ │ │ + OpenLayers.Util.extend(wheelOptions, this.mouseWheelOptions) │ │ │ │ │ + ); │ │ │ │ │ + if (OpenLayers.Control.PinchZoom) { │ │ │ │ │ + this.pinchZoom = new OpenLayers.Control.PinchZoom( │ │ │ │ │ + OpenLayers.Util.extend({ │ │ │ │ │ + map: this.map │ │ │ │ │ + }, this.pinchZoomOptions)); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: defaultClick │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * evt - {Event} │ │ │ │ │ - * pinchData - {Object} pinch data object related to the current touchmove │ │ │ │ │ - * of the pinch gesture. This give us the current scale of the pinch. │ │ │ │ │ */ │ │ │ │ │ - pinchStart: function(evt, pinchData) { │ │ │ │ │ - var xy = (this.preserveCenter) ? │ │ │ │ │ - this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ - this.pinchOrigin = xy; │ │ │ │ │ - this.currentCenter = xy; │ │ │ │ │ + defaultClick: function(evt) { │ │ │ │ │ + if (evt.lastTouches && evt.lastTouches.length == 2) { │ │ │ │ │ + this.map.zoomOut(); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: pinchMove │ │ │ │ │ + * Method: defaultDblClick │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + defaultDblClick: function(evt) { │ │ │ │ │ + this.map.zoomTo(this.map.zoom + 1, evt.xy); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: defaultDblRightClick │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + */ │ │ │ │ │ + defaultDblRightClick: function(evt) { │ │ │ │ │ + this.map.zoomTo(this.map.zoom - 1, evt.xy); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: wheelChange │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * evt - {Event} │ │ │ │ │ - * pinchData - {Object} pinch data object related to the current touchmove │ │ │ │ │ - * of the pinch gesture. This give us the current scale of the pinch. │ │ │ │ │ + * deltaZ - {Integer} │ │ │ │ │ */ │ │ │ │ │ - pinchMove: function(evt, pinchData) { │ │ │ │ │ - var scale = pinchData.scale; │ │ │ │ │ - var containerOrigin = this.map.layerContainerOriginPx; │ │ │ │ │ - var pinchOrigin = this.pinchOrigin; │ │ │ │ │ - var current = (this.preserveCenter) ? │ │ │ │ │ - this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ + wheelChange: function(evt, deltaZ) { │ │ │ │ │ + if (!this.map.fractionalZoom) { │ │ │ │ │ + deltaZ = Math.round(deltaZ); │ │ │ │ │ + } │ │ │ │ │ + var currentZoom = this.map.getZoom(), │ │ │ │ │ + newZoom = currentZoom + deltaZ; │ │ │ │ │ + newZoom = Math.max(newZoom, 0); │ │ │ │ │ + newZoom = Math.min(newZoom, this.map.getNumZoomLevels()); │ │ │ │ │ + if (newZoom === currentZoom) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + this.map.zoomTo(newZoom, evt.xy); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var dx = Math.round((containerOrigin.x + current.x - pinchOrigin.x) + (scale - 1) * (containerOrigin.x - pinchOrigin.x)); │ │ │ │ │ - var dy = Math.round((containerOrigin.y + current.y - pinchOrigin.y) + (scale - 1) * (containerOrigin.y - pinchOrigin.y)); │ │ │ │ │ + /** │ │ │ │ │ + * Method: wheelUp │ │ │ │ │ + * User spun scroll wheel up │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * delta - {Integer} │ │ │ │ │ + */ │ │ │ │ │ + wheelUp: function(evt, delta) { │ │ │ │ │ + this.wheelChange(evt, delta || 1); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - this.map.applyTransform(dx, dy, scale); │ │ │ │ │ - this.currentCenter = current; │ │ │ │ │ + /** │ │ │ │ │ + * Method: wheelDown │ │ │ │ │ + * User spun scroll wheel down │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * delta - {Integer} │ │ │ │ │ + */ │ │ │ │ │ + wheelDown: function(evt, delta) { │ │ │ │ │ + this.wheelChange(evt, delta || -1); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: pinchDone │ │ │ │ │ + * Method: disableZoomBox │ │ │ │ │ + */ │ │ │ │ │ + disableZoomBox: function() { │ │ │ │ │ + this.zoomBoxEnabled = false; │ │ │ │ │ + this.zoomBox.deactivate(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: enableZoomBox │ │ │ │ │ + */ │ │ │ │ │ + enableZoomBox: function() { │ │ │ │ │ + this.zoomBoxEnabled = true; │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.zoomBox.activate(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: disableZoomWheel │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + disableZoomWheel: function() { │ │ │ │ │ + this.zoomWheelEnabled = false; │ │ │ │ │ + this.handlers.wheel.deactivate(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: enableZoomWheel │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + enableZoomWheel: function() { │ │ │ │ │ + this.zoomWheelEnabled = true; │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.handlers.wheel.activate(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Navigation" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/NavToolbar.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control/Panel.js │ │ │ │ │ + * @requires OpenLayers/Control/Navigation.js │ │ │ │ │ + * @requires OpenLayers/Control/ZoomBox.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.NavToolbar │ │ │ │ │ + * This Toolbar is an alternative to the Navigation control that displays │ │ │ │ │ + * the state of the control, and provides a UI for changing state to │ │ │ │ │ + * use the zoomBox via a Panel control. │ │ │ │ │ + * │ │ │ │ │ + * If you wish to change the properties of the Navigation control used │ │ │ │ │ + * in the NavToolbar, see: │ │ │ │ │ + * http://trac.openlayers.org/wiki/Toolbars#SubclassingNavToolbar │ │ │ │ │ + * │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control.Panel> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.NavToolbar = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.NavToolbar │ │ │ │ │ + * Add our two mousedefaults controls. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - * start - {Object} pinch data object related to the touchstart event that │ │ │ │ │ - * started the pinch gesture. │ │ │ │ │ - * last - {Object} pinch data object related to the last touchmove event │ │ │ │ │ - * of the pinch gesture. This give us the final scale of the pinch. │ │ │ │ │ + * options - {Object} An optional object whose properties will be used │ │ │ │ │ + * to extend the control. │ │ │ │ │ */ │ │ │ │ │ - pinchDone: function(evt, start, last) { │ │ │ │ │ - this.map.applyTransform(); │ │ │ │ │ - var zoom = this.map.getZoomForResolution(this.map.getResolution() / last.scale, true); │ │ │ │ │ - if (zoom !== this.map.getZoom() || !this.currentCenter.equals(this.pinchOrigin)) { │ │ │ │ │ - var resolution = this.map.getResolutionForZoom(zoom); │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.addControls([ │ │ │ │ │ + new OpenLayers.Control.Navigation(), │ │ │ │ │ + new OpenLayers.Control.ZoomBox() │ │ │ │ │ + ]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var location = this.map.getLonLatFromPixel(this.pinchOrigin); │ │ │ │ │ - var zoomPixel = this.currentCenter; │ │ │ │ │ - var size = this.map.getSize(); │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * calls the default draw, and then activates mouse defaults. │ │ │ │ │ + */ │ │ │ │ │ + draw: function() { │ │ │ │ │ + var div = OpenLayers.Control.Panel.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (this.defaultControl === null) { │ │ │ │ │ + this.defaultControl = this.controls[0]; │ │ │ │ │ + } │ │ │ │ │ + return div; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - location.lon += resolution * ((size.w / 2) - zoomPixel.x); │ │ │ │ │ - location.lat -= resolution * ((size.h / 2) - zoomPixel.y); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.NavToolbar" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/CacheRead.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - // Force a reflow before calling setCenter. This is to work │ │ │ │ │ - // around an issue occuring in iOS. │ │ │ │ │ - // │ │ │ │ │ - // See https://github.com/openlayers/openlayers/pull/351. │ │ │ │ │ - // │ │ │ │ │ - // Without a reflow setting the layer container div's top left │ │ │ │ │ - // style properties to "0px" - as done in Map.moveTo when zoom │ │ │ │ │ - // is changed - won't actually correctly reposition the layer │ │ │ │ │ - // container div. │ │ │ │ │ - // │ │ │ │ │ - // Also, we need to use a statement that the Google Closure │ │ │ │ │ - // compiler won't optimize away. │ │ │ │ │ - this.map.div.clientWidth = this.map.div.clientWidth; │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - this.map.setCenter(location, zoom); │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.CacheRead │ │ │ │ │ + * A control for using image tiles cached with <OpenLayers.Control.CacheWrite> │ │ │ │ │ + * from the browser's local storage. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.CacheRead = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: fetchEvent │ │ │ │ │ + * {String} The layer event to listen to for replacing remote resource tile │ │ │ │ │ + * URLs with cached data URIs. Supported values are "tileerror" (try │ │ │ │ │ + * remote first, fall back to cached) and "tileloadstart" (try cache │ │ │ │ │ + * first, fall back to remote). Default is "tileloadstart". │ │ │ │ │ + * │ │ │ │ │ + * Note that "tileerror" will not work for CORS enabled images (see │ │ │ │ │ + * https://developer.mozilla.org/en/CORS_Enabled_Image), i.e. layers │ │ │ │ │ + * configured with a <OpenLayers.Tile.Image.crossOriginKeyword> in │ │ │ │ │ + * <OpenLayers.Layer.Grid.tileOptions>. │ │ │ │ │ + */ │ │ │ │ │ + fetchEvent: "tileloadstart", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layers │ │ │ │ │ + * {Array(<OpenLayers.Layer.Grid>)}. Optional. If provided, only these │ │ │ │ │ + * layers will receive tiles from the cache. │ │ │ │ │ + */ │ │ │ │ │ + layers: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ + */ │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.CacheRead │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Object with API properties for this control │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the control. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + var i, layers = this.layers || map.layers; │ │ │ │ │ + for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + this.addLayer({ │ │ │ │ │ + layer: layers[i] │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + if (!this.layers) { │ │ │ │ │ + map.events.on({ │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + removeLayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.PinchZoom" │ │ │ │ │ + /** │ │ │ │ │ + * Method: addLayer │ │ │ │ │ + * Adds a layer to the control. Once added, tiles requested for this layer │ │ │ │ │ + * will be cached. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} Object with a layer property referencing an │ │ │ │ │ + * <OpenLayers.Layer> instance │ │ │ │ │ + */ │ │ │ │ │ + addLayer: function(evt) { │ │ │ │ │ + evt.layer.events.register(this.fetchEvent, this, this.fetch); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeLayer │ │ │ │ │ + * Removes a layer from the control. Once removed, tiles requested for this │ │ │ │ │ + * layer will no longer be cached. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} Object with a layer property referencing an │ │ │ │ │ + * <OpenLayers.Layer> instance │ │ │ │ │ + */ │ │ │ │ │ + removeLayer: function(evt) { │ │ │ │ │ + evt.layer.events.unregister(this.fetchEvent, this, this.fetch); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: fetch │ │ │ │ │ + * Listener to the <fetchEvent> event. Replaces a tile's url with a data │ │ │ │ │ + * URI from the cache. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} Event object with a tile property. │ │ │ │ │ + */ │ │ │ │ │ + fetch: function(evt) { │ │ │ │ │ + if (this.active && window.localStorage && │ │ │ │ │ + evt.tile instanceof OpenLayers.Tile.Image) { │ │ │ │ │ + var tile = evt.tile, │ │ │ │ │ + url = tile.url; │ │ │ │ │ + // deal with modified tile urls when both CacheWrite and CacheRead │ │ │ │ │ + // are active │ │ │ │ │ + if (!tile.layer.crossOriginKeyword && OpenLayers.ProxyHost && │ │ │ │ │ + url.indexOf(OpenLayers.ProxyHost) === 0) { │ │ │ │ │ + url = OpenLayers.Control.CacheWrite.urlMap[url]; │ │ │ │ │ + } │ │ │ │ │ + var dataURI = window.localStorage.getItem("olCache_" + url); │ │ │ │ │ + if (dataURI) { │ │ │ │ │ + tile.url = dataURI; │ │ │ │ │ + if (evt.type === "tileerror") { │ │ │ │ │ + tile.setImgSrc(dataURI); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * The destroy method is used to perform any clean up before the control │ │ │ │ │ + * is dereferenced. Typically this is where event listeners are removed │ │ │ │ │ + * to prevent memory leaks. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.layers || this.map) { │ │ │ │ │ + var i, layers = this.layers || this.map.layers; │ │ │ │ │ + for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + this.removeLayer({ │ │ │ │ │ + layer: layers[i] │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + removeLayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.CacheRead" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Control/OverviewMap.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ @@ -79827,1255 +80075,539 @@ │ │ │ │ │ }; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: 'OpenLayers.Control.OverviewMap' │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/WMSGetFeatureInfo.js │ │ │ │ │ + OpenLayers/Control/LayerSwitcher.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Handler/Click.js │ │ │ │ │ - * @requires OpenLayers/Handler/Hover.js │ │ │ │ │ - * @requires OpenLayers/Request.js │ │ │ │ │ - * @requires OpenLayers/Format/WMSGetFeatureInfo.js │ │ │ │ │ + * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.WMSGetFeatureInfo │ │ │ │ │ - * The WMSGetFeatureInfo control uses a WMS query to get information about a point on the map. The │ │ │ │ │ - * information may be in a display-friendly format such as HTML, or a machine-friendly format such │ │ │ │ │ - * as GML, depending on the server's capabilities and the client's configuration. This control │ │ │ │ │ - * handles click or hover events, attempts to parse the results using an OpenLayers.Format, and │ │ │ │ │ - * fires a 'getfeatureinfo' event with the click position, the raw body of the response, and an │ │ │ │ │ - * array of features if it successfully read the response. │ │ │ │ │ + * Class: OpenLayers.Control.LayerSwitcher │ │ │ │ │ + * The LayerSwitcher control displays a table of contents for the map. This │ │ │ │ │ + * allows the user interface to switch between BaseLasyers and to show or hide │ │ │ │ │ + * Overlays. By default the switcher is shown minimized on the right edge of │ │ │ │ │ + * the map, the user may expand it by clicking on the handle. │ │ │ │ │ + * │ │ │ │ │ + * To create the LayerSwitcher outside of the map, pass the Id of a html div │ │ │ │ │ + * as the first argument to the constructor. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.WMSGetFeatureInfo = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: hover │ │ │ │ │ - * {Boolean} Send GetFeatureInfo requests when mouse stops moving. │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - hover: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: drillDown │ │ │ │ │ - * {Boolean} Drill down over all WMS layers in the map. When │ │ │ │ │ - * using drillDown mode, hover is not possible, and an infoFormat that │ │ │ │ │ - * returns parseable features is required. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - drillDown: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: maxFeatures │ │ │ │ │ - * {Integer} Maximum number of features to return from a WMS query. This │ │ │ │ │ - * sets the feature_count parameter on WMS GetFeatureInfo │ │ │ │ │ - * requests. │ │ │ │ │ - */ │ │ │ │ │ - maxFeatures: 10, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: clickCallback │ │ │ │ │ - * {String} The click callback to register in the │ │ │ │ │ - * {<OpenLayers.Handler.Click>} object created when the hover │ │ │ │ │ - * option is set to false. Default is "click". │ │ │ │ │ - */ │ │ │ │ │ - clickCallback: "click", │ │ │ │ │ +OpenLayers.Control.LayerSwitcher = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: output │ │ │ │ │ - * {String} Either "features" or "object". When triggering a getfeatureinfo │ │ │ │ │ - * request should we pass on an array of features or an object with with │ │ │ │ │ - * a "features" property and other properties (such as the url of the │ │ │ │ │ - * WMS). Default is "features". │ │ │ │ │ + /** │ │ │ │ │ + * Property: layerStates │ │ │ │ │ + * {Array(Object)} Basically a copy of the "state" of the map's layers │ │ │ │ │ + * the last time the control was drawn. We have this in order to avoid │ │ │ │ │ + * unnecessarily redrawing the control. │ │ │ │ │ */ │ │ │ │ │ - output: "features", │ │ │ │ │ + layerStates: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: layers │ │ │ │ │ - * {Array(<OpenLayers.Layer.WMS>)} The layers to query for feature info. │ │ │ │ │ - * If omitted, all map WMS layers with a url that matches this <url> or │ │ │ │ │ - * <layerUrls> will be considered. │ │ │ │ │ - */ │ │ │ │ │ - layers: null, │ │ │ │ │ + // DOM Elements │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: queryVisible │ │ │ │ │ - * {Boolean} If true, filter out hidden layers when searching the map for │ │ │ │ │ - * layers to query. Default is false. │ │ │ │ │ + * Property: layersDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - queryVisible: false, │ │ │ │ │ + layersDiv: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String} The URL of the WMS service to use. If not provided, the url │ │ │ │ │ - * of the first eligible layer will be used. │ │ │ │ │ + * Property: baseLayersDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - url: null, │ │ │ │ │ + baseLayersDiv: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: layerUrls │ │ │ │ │ - * {Array(String)} Optional list of urls for layers that should be queried. │ │ │ │ │ - * This can be used when the layer url differs from the url used for │ │ │ │ │ - * making GetFeatureInfo requests (in the case of a layer using cached │ │ │ │ │ - * tiles). │ │ │ │ │ + * Property: baseLayers │ │ │ │ │ + * {Array(Object)} │ │ │ │ │ */ │ │ │ │ │ - layerUrls: null, │ │ │ │ │ + baseLayers: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: infoFormat │ │ │ │ │ - * {String} The mimetype to request from the server. If you are using │ │ │ │ │ - * drillDown mode and have multiple servers that do not share a common │ │ │ │ │ - * infoFormat, you can override the control's infoFormat by providing an │ │ │ │ │ - * INFO_FORMAT parameter in your <OpenLayers.Layer.WMS> instance(s). │ │ │ │ │ - */ │ │ │ │ │ - infoFormat: 'text/html', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: vendorParams │ │ │ │ │ - * {Object} Additional parameters that will be added to the request, for │ │ │ │ │ - * WMS implementations that support them. This could e.g. look like │ │ │ │ │ - * (start code) │ │ │ │ │ - * { │ │ │ │ │ - * radius: 5 │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ + * Property: dataLbl │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - vendorParams: {}, │ │ │ │ │ + dataLbl: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: format │ │ │ │ │ - * {<OpenLayers.Format>} A format for parsing GetFeatureInfo responses. │ │ │ │ │ - * Default is <OpenLayers.Format.WMSGetFeatureInfo>. │ │ │ │ │ + * Property: dataLayersDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - format: null, │ │ │ │ │ + dataLayersDiv: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: formatOptions │ │ │ │ │ - * {Object} Optional properties to set on the format (if one is not provided │ │ │ │ │ - * in the <format> property. │ │ │ │ │ + * Property: dataLayers │ │ │ │ │ + * {Array(Object)} │ │ │ │ │ */ │ │ │ │ │ - formatOptions: null, │ │ │ │ │ + dataLayers: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: handlerOptions │ │ │ │ │ - * {Object} Additional options for the handlers used by this control, e.g. │ │ │ │ │ - * (start code) │ │ │ │ │ - * { │ │ │ │ │ - * "click": {delay: 100}, │ │ │ │ │ - * "hover": {delay: 300} │ │ │ │ │ - * } │ │ │ │ │ - * (end) │ │ │ │ │ - */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: handler │ │ │ │ │ - * {Object} Reference to the <OpenLayers.Handler> for this control │ │ │ │ │ + * Property: minimizeDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - handler: null, │ │ │ │ │ + minimizeDiv: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: hoverRequest │ │ │ │ │ - * {<OpenLayers.Request>} contains the currently running hover request │ │ │ │ │ - * (if any). │ │ │ │ │ + * Property: maximizeDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - hoverRequest: null, │ │ │ │ │ + maximizeDiv: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * beforegetfeatureinfo - Triggered before the request is sent. │ │ │ │ │ - * The event object has an *xy* property with the position of the │ │ │ │ │ - * mouse click or hover event that triggers the request. │ │ │ │ │ - * nogetfeatureinfo - no queryable layers were found. │ │ │ │ │ - * getfeatureinfo - Triggered when a GetFeatureInfo response is received. │ │ │ │ │ - * The event object has a *text* property with the body of the │ │ │ │ │ - * response (String), a *features* property with an array of the │ │ │ │ │ - * parsed features, an *xy* property with the position of the mouse │ │ │ │ │ - * click or hover event that triggered the request, and a *request* │ │ │ │ │ - * property with the request itself. If drillDown is set to true and │ │ │ │ │ - * multiple requests were issued to collect feature info from all │ │ │ │ │ - * layers, *text* and *request* will only contain the response body │ │ │ │ │ - * and request object of the last request. │ │ │ │ │ + * APIProperty: ascending │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ + ascending: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: <OpenLayers.Control.WMSGetFeatureInfo> │ │ │ │ │ + * Constructor: OpenLayers.Control.LayerSwitcher │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ * options - {Object} │ │ │ │ │ */ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - │ │ │ │ │ - if (!this.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.WMSGetFeatureInfo( │ │ │ │ │ - options.formatOptions │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.drillDown === true) { │ │ │ │ │ - this.hover = false; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Hover( │ │ │ │ │ - this, { │ │ │ │ │ - 'move': this.cancelHover, │ │ │ │ │ - 'pause': this.getInfoForHover │ │ │ │ │ - }, │ │ │ │ │ - OpenLayers.Util.extend(this.handlerOptions.hover || {}, { │ │ │ │ │ - 'delay': 250 │ │ │ │ │ - })); │ │ │ │ │ - } else { │ │ │ │ │ - var callbacks = {}; │ │ │ │ │ - callbacks[this.clickCallback] = this.getInfoForClick; │ │ │ │ │ - this.handler = new OpenLayers.Handler.Click( │ │ │ │ │ - this, callbacks, this.handlerOptions.click || {}); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getInfoForClick │ │ │ │ │ - * Called on click │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - */ │ │ │ │ │ - getInfoForClick: function(evt) { │ │ │ │ │ - this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ - xy: evt.xy │ │ │ │ │ - }); │ │ │ │ │ - // Set the cursor to "wait" to tell the user we're working on their │ │ │ │ │ - // click. │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - this.request(evt.xy, {}); │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.layerStates = []; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getInfoForHover │ │ │ │ │ - * Pause callback for the hover handler │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ */ │ │ │ │ │ - getInfoForHover: function(evt) { │ │ │ │ │ - this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ - xy: evt.xy │ │ │ │ │ - }); │ │ │ │ │ - this.request(evt.xy, { │ │ │ │ │ - hover: true │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: cancelHover │ │ │ │ │ - * Cancel callback for the hover handler │ │ │ │ │ - */ │ │ │ │ │ - cancelHover: function() { │ │ │ │ │ - if (this.hoverRequest) { │ │ │ │ │ - this.hoverRequest.abort(); │ │ │ │ │ - this.hoverRequest = null; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + //clear out layers info and unregister their events │ │ │ │ │ + this.clearLayersArray("base"); │ │ │ │ │ + this.clearLayersArray("data"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: findLayers │ │ │ │ │ - * Internal method to get the layers, independent of whether we are │ │ │ │ │ - * inspecting the map or using a client-provided array │ │ │ │ │ - */ │ │ │ │ │ - findLayers: function() { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + buttonclick: this.onButtonClick, │ │ │ │ │ + addlayer: this.redraw, │ │ │ │ │ + changelayer: this.redraw, │ │ │ │ │ + removelayer: this.redraw, │ │ │ │ │ + changebaselayer: this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ │ │ │ │ │ - var candidates = this.layers || this.map.layers; │ │ │ │ │ - var layers = []; │ │ │ │ │ - var layer, url; │ │ │ │ │ - for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = candidates[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.WMS && │ │ │ │ │ - (!this.queryVisible || layer.getVisibility())) { │ │ │ │ │ - url = OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url; │ │ │ │ │ - // if the control was not configured with a url, set it │ │ │ │ │ - // to the first layer url │ │ │ │ │ - if (this.drillDown === false && !this.url) { │ │ │ │ │ - this.url = url; │ │ │ │ │ - } │ │ │ │ │ - if (this.drillDown === true || this.urlMatches(url)) { │ │ │ │ │ - layers.push(layer); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return layers; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: urlMatches │ │ │ │ │ - * Test to see if the provided url matches either the control <url> or one │ │ │ │ │ - * of the <layerUrls>. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * url - {String} The url to test. │ │ │ │ │ + * Method: setMap │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The provided url matches the control <url> or one of the │ │ │ │ │ - * <layerUrls>. │ │ │ │ │ + * Properties: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - urlMatches: function(url) { │ │ │ │ │ - var matches = OpenLayers.Util.isEquivalentUrl(this.url, url); │ │ │ │ │ - if (!matches && this.layerUrls) { │ │ │ │ │ - for (var i = 0, len = this.layerUrls.length; i < len; ++i) { │ │ │ │ │ - if (OpenLayers.Util.isEquivalentUrl(this.layerUrls[i], url)) { │ │ │ │ │ - matches = true; │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return matches; │ │ │ │ │ - }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: buildWMSOptions │ │ │ │ │ - * Build an object with the relevant WMS options for the GetFeatureInfo request │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * url - {String} The url to be used for sending the request │ │ │ │ │ - * layers - {Array(<OpenLayers.Layer.WMS)} An array of layers │ │ │ │ │ - * clickPosition - {<OpenLayers.Pixel>} The position on the map where the mouse │ │ │ │ │ - * event occurred. │ │ │ │ │ - * format - {String} The format from the corresponding GetMap request │ │ │ │ │ - */ │ │ │ │ │ - buildWMSOptions: function(url, layers, clickPosition, format) { │ │ │ │ │ - var layerNames = [], │ │ │ │ │ - styleNames = []; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - if (layers[i].params.LAYERS != null) { │ │ │ │ │ - layerNames = layerNames.concat(layers[i].params.LAYERS); │ │ │ │ │ - styleNames = styleNames.concat(this.getStyleNames(layers[i])); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var firstLayer = layers[0]; │ │ │ │ │ - // use the firstLayer's projection if it matches the map projection - │ │ │ │ │ - // this assumes that all layers will be available in this projection │ │ │ │ │ - var projection = this.map.getProjection(); │ │ │ │ │ - var layerProj = firstLayer.projection; │ │ │ │ │ - if (layerProj && layerProj.equals(this.map.getProjectionObject())) { │ │ │ │ │ - projection = layerProj.getCode(); │ │ │ │ │ - } │ │ │ │ │ - var params = OpenLayers.Util.extend({ │ │ │ │ │ - service: "WMS", │ │ │ │ │ - version: firstLayer.params.VERSION, │ │ │ │ │ - request: "GetFeatureInfo", │ │ │ │ │ - exceptions: firstLayer.params.EXCEPTIONS, │ │ │ │ │ - bbox: this.map.getExtent().toBBOX(null, │ │ │ │ │ - firstLayer.reverseAxisOrder()), │ │ │ │ │ - feature_count: this.maxFeatures, │ │ │ │ │ - height: this.map.getSize().h, │ │ │ │ │ - width: this.map.getSize().w, │ │ │ │ │ - format: format, │ │ │ │ │ - info_format: firstLayer.params.INFO_FORMAT || this.infoFormat │ │ │ │ │ - }, (parseFloat(firstLayer.params.VERSION) >= 1.3) ? { │ │ │ │ │ - crs: projection, │ │ │ │ │ - i: parseInt(clickPosition.x), │ │ │ │ │ - j: parseInt(clickPosition.y) │ │ │ │ │ - } : { │ │ │ │ │ - srs: projection, │ │ │ │ │ - x: parseInt(clickPosition.x), │ │ │ │ │ - y: parseInt(clickPosition.y) │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + addlayer: this.redraw, │ │ │ │ │ + changelayer: this.redraw, │ │ │ │ │ + removelayer: this.redraw, │ │ │ │ │ + changebaselayer: this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ }); │ │ │ │ │ - if (layerNames.length != 0) { │ │ │ │ │ - params = OpenLayers.Util.extend({ │ │ │ │ │ - layers: layerNames, │ │ │ │ │ - query_layers: layerNames, │ │ │ │ │ - styles: styleNames │ │ │ │ │ - }, params); │ │ │ │ │ + if (this.outsideViewport) { │ │ │ │ │ + this.events.attachToElement(this.div); │ │ │ │ │ + this.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ + } else { │ │ │ │ │ + this.map.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Util.applyDefaults(params, this.vendorParams); │ │ │ │ │ - return { │ │ │ │ │ - url: url, │ │ │ │ │ - params: OpenLayers.Util.upperCaseObject(params), │ │ │ │ │ - callback: function(request) { │ │ │ │ │ - this.handleResponse(clickPosition, request, url); │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getStyleNames │ │ │ │ │ - * Gets the STYLES parameter for the layer. Make sure the STYLES parameter │ │ │ │ │ - * matches the LAYERS parameter │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.WMS>} │ │ │ │ │ + * Method: draw │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array(String)} The STYLES parameter │ │ │ │ │ + * {DOMElement} A reference to the DIV DOMElement containing the │ │ │ │ │ + * switcher tabs. │ │ │ │ │ */ │ │ │ │ │ - getStyleNames: function(layer) { │ │ │ │ │ - // in the event of a WMS layer bundling multiple layers but not │ │ │ │ │ - // specifying styles,we need the same number of commas to specify │ │ │ │ │ - // the default style for each of the layers. We can't just leave it │ │ │ │ │ - // blank as we may be including other layers that do specify styles. │ │ │ │ │ - var styleNames; │ │ │ │ │ - if (layer.params.STYLES) { │ │ │ │ │ - styleNames = layer.params.STYLES; │ │ │ │ │ - } else { │ │ │ │ │ - if (OpenLayers.Util.isArray(layer.params.LAYERS)) { │ │ │ │ │ - styleNames = new Array(layer.params.LAYERS.length); │ │ │ │ │ - } else { // Assume it's a String │ │ │ │ │ - styleNames = layer.params.LAYERS.replace(/[^,]/g, ""); │ │ │ │ │ - } │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this); │ │ │ │ │ + │ │ │ │ │ + // create layout divs │ │ │ │ │ + this.loadContents(); │ │ │ │ │ + │ │ │ │ │ + // set mode to minimize │ │ │ │ │ + if (!this.outsideViewport) { │ │ │ │ │ + this.minimizeControl(); │ │ │ │ │ } │ │ │ │ │ - return styleNames; │ │ │ │ │ + │ │ │ │ │ + // populate div with current info │ │ │ │ │ + this.redraw(); │ │ │ │ │ + │ │ │ │ │ + return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: request │ │ │ │ │ - * Sends a GetFeatureInfo request to the WMS │ │ │ │ │ + * Method: onButtonClick │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * clickPosition - {<OpenLayers.Pixel>} The position on the map where the │ │ │ │ │ - * mouse event occurred. │ │ │ │ │ - * options - {Object} additional options for this method. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * - *hover* {Boolean} true if we do the request for the hover handler │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - request: function(clickPosition, options) { │ │ │ │ │ - var layers = this.findLayers(); │ │ │ │ │ - if (layers.length == 0) { │ │ │ │ │ - this.events.triggerEvent("nogetfeatureinfo"); │ │ │ │ │ - // Reset the cursor. │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - options = options || {}; │ │ │ │ │ - if (this.drillDown === false) { │ │ │ │ │ - var wmsOptions = this.buildWMSOptions(this.url, layers, │ │ │ │ │ - clickPosition, layers[0].params.FORMAT); │ │ │ │ │ - var request = OpenLayers.Request.GET(wmsOptions); │ │ │ │ │ - │ │ │ │ │ - if (options.hover === true) { │ │ │ │ │ - this.hoverRequest = request; │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + var button = evt.buttonElement; │ │ │ │ │ + if (button === this.minimizeDiv) { │ │ │ │ │ + this.minimizeControl(); │ │ │ │ │ + } else if (button === this.maximizeDiv) { │ │ │ │ │ + this.maximizeControl(); │ │ │ │ │ + } else if (button._layerSwitcher === this.id) { │ │ │ │ │ + if (button["for"]) { │ │ │ │ │ + button = document.getElementById(button["for"]); │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - this._requestCount = 0; │ │ │ │ │ - this._numRequests = 0; │ │ │ │ │ - this.features = []; │ │ │ │ │ - // group according to service url to combine requests │ │ │ │ │ - var services = {}, │ │ │ │ │ - url; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - var layer = layers[i]; │ │ │ │ │ - var service, found = false; │ │ │ │ │ - url = OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url; │ │ │ │ │ - if (url in services) { │ │ │ │ │ - services[url].push(layer); │ │ │ │ │ + if (!button.disabled) { │ │ │ │ │ + if (button.type == "radio") { │ │ │ │ │ + button.checked = true; │ │ │ │ │ + this.map.setBaseLayer(this.map.getLayer(button._layer)); │ │ │ │ │ } else { │ │ │ │ │ - this._numRequests++; │ │ │ │ │ - services[url] = [layer]; │ │ │ │ │ + button.checked = !button.checked; │ │ │ │ │ + this.updateMap(); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var layers; │ │ │ │ │ - for (var url in services) { │ │ │ │ │ - layers = services[url]; │ │ │ │ │ - var wmsOptions = this.buildWMSOptions(url, layers, │ │ │ │ │ - clickPosition, layers[0].params.FORMAT); │ │ │ │ │ - OpenLayers.Request.GET(wmsOptions); │ │ │ │ │ - } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: triggerGetFeatureInfo │ │ │ │ │ - * Trigger the getfeatureinfo event when all is done │ │ │ │ │ + * Method: clearLayersArray │ │ │ │ │ + * User specifies either "base" or "data". we then clear all the │ │ │ │ │ + * corresponding listeners, the div, and reinitialize a new array. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * request - {XMLHttpRequest} The request object │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} The position on the map where the │ │ │ │ │ - * mouse event occurred. │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} or │ │ │ │ │ - * {Array({Object}) when output is "object". The object has a url and a │ │ │ │ │ - * features property which contains an array of features. │ │ │ │ │ + * layersType - {String} │ │ │ │ │ */ │ │ │ │ │ - triggerGetFeatureInfo: function(request, xy, features) { │ │ │ │ │ - this.events.triggerEvent("getfeatureinfo", { │ │ │ │ │ - text: request.responseText, │ │ │ │ │ - features: features, │ │ │ │ │ - request: request, │ │ │ │ │ - xy: xy │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - // Reset the cursor. │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + clearLayersArray: function(layersType) { │ │ │ │ │ + this[layersType + "LayersDiv"].innerHTML = ""; │ │ │ │ │ + this[layersType + "Layers"] = []; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: handleResponse │ │ │ │ │ - * Handler for the GetFeatureInfo response. │ │ │ │ │ + * Method: checkRedraw │ │ │ │ │ + * Checks if the layer state has changed since the last redraw() call. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} The position on the map where the │ │ │ │ │ - * mouse event occurred. │ │ │ │ │ - * request - {XMLHttpRequest} The request object. │ │ │ │ │ - * url - {String} The url which was used for this request. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The layer state changed since the last redraw() call. │ │ │ │ │ */ │ │ │ │ │ - handleResponse: function(xy, request, url) { │ │ │ │ │ - │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ - } │ │ │ │ │ - var features = this.format.read(doc); │ │ │ │ │ - if (this.drillDown === false) { │ │ │ │ │ - this.triggerGetFeatureInfo(request, xy, features); │ │ │ │ │ - } else { │ │ │ │ │ - this._requestCount++; │ │ │ │ │ - if (this.output === "object") { │ │ │ │ │ - this._features = (this._features || []).concat({ │ │ │ │ │ - url: url, │ │ │ │ │ - features: features │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - this._features = (this._features || []).concat(features); │ │ │ │ │ - } │ │ │ │ │ - if (this._requestCount === this._numRequests) { │ │ │ │ │ - this.triggerGetFeatureInfo(request, xy, this._features.concat()); │ │ │ │ │ - delete this._features; │ │ │ │ │ - delete this._requestCount; │ │ │ │ │ - delete this._numRequests; │ │ │ │ │ - } │ │ │ │ │ + checkRedraw: function() { │ │ │ │ │ + if (!this.layerStates.length || │ │ │ │ │ + (this.map.layers.length != this.layerStates.length)) { │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.WMSGetFeatureInfo" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/EditingToolbar.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control/Panel.js │ │ │ │ │ - * @requires OpenLayers/Control/Navigation.js │ │ │ │ │ - * @requires OpenLayers/Control/DrawFeature.js │ │ │ │ │ - * @requires OpenLayers/Handler/Point.js │ │ │ │ │ - * @requires OpenLayers/Handler/Path.js │ │ │ │ │ - * @requires OpenLayers/Handler/Polygon.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.EditingToolbar │ │ │ │ │ - * The EditingToolbar is a panel of 4 controls to draw polygons, lines, │ │ │ │ │ - * points, or to navigate the map by panning. By default it appears in the │ │ │ │ │ - * upper right corner of the map. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control.Panel> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.EditingToolbar = OpenLayers.Class( │ │ │ │ │ - OpenLayers.Control.Panel, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: citeCompliant │ │ │ │ │ - * {Boolean} If set to true, coordinates of features drawn in a map extent │ │ │ │ │ - * crossing the date line won't exceed the world bounds. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - citeCompliant: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.EditingToolbar │ │ │ │ │ - * Create an editing toolbar for a given layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(layer, options) { │ │ │ │ │ - OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ - │ │ │ │ │ - this.addControls( │ │ │ │ │ - [new OpenLayers.Control.Navigation()] │ │ │ │ │ - ); │ │ │ │ │ - var controls = [ │ │ │ │ │ - new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Point, { │ │ │ │ │ - displayClass: 'olControlDrawFeaturePoint', │ │ │ │ │ - handlerOptions: { │ │ │ │ │ - citeCompliant: this.citeCompliant │ │ │ │ │ - } │ │ │ │ │ - }), │ │ │ │ │ - new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Path, { │ │ │ │ │ - displayClass: 'olControlDrawFeaturePath', │ │ │ │ │ - handlerOptions: { │ │ │ │ │ - citeCompliant: this.citeCompliant │ │ │ │ │ - } │ │ │ │ │ - }), │ │ │ │ │ - new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Polygon, { │ │ │ │ │ - displayClass: 'olControlDrawFeaturePolygon', │ │ │ │ │ - handlerOptions: { │ │ │ │ │ - citeCompliant: this.citeCompliant │ │ │ │ │ - } │ │ │ │ │ - }) │ │ │ │ │ - ]; │ │ │ │ │ - this.addControls(controls); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * calls the default draw, and then activates mouse defaults. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - var div = OpenLayers.Control.Panel.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (this.defaultControl === null) { │ │ │ │ │ - this.defaultControl = this.controls[0]; │ │ │ │ │ - } │ │ │ │ │ - return div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.EditingToolbar" │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Attribution.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Attribution │ │ │ │ │ - * The attribution control adds attribution from layers to the map display. │ │ │ │ │ - * It uses 'attribution' property of each layer. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Attribution = │ │ │ │ │ - OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: separator │ │ │ │ │ - * {String} String used to separate layers. │ │ │ │ │ - */ │ │ │ │ │ - separator: ", ", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: template │ │ │ │ │ - * {String} Template for the attribution. This has to include the substring │ │ │ │ │ - * "${layers}", which will be replaced by the layer specific │ │ │ │ │ - * attributions, separated by <separator>. The default is "${layers}". │ │ │ │ │ - */ │ │ │ │ │ - template: "${layers}", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Attribution │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Options for control. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Destroy control. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - "removelayer": this.updateAttribution, │ │ │ │ │ - "addlayer": this.updateAttribution, │ │ │ │ │ - "changelayer": this.updateAttribution, │ │ │ │ │ - "changebaselayer": this.updateAttribution, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Initialize control. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - 'changebaselayer': this.updateAttribution, │ │ │ │ │ - 'changelayer': this.updateAttribution, │ │ │ │ │ - 'addlayer': this.updateAttribution, │ │ │ │ │ - 'removelayer': this.updateAttribution, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.updateAttribution(); │ │ │ │ │ │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateAttribution │ │ │ │ │ - * Update attribution string. │ │ │ │ │ - */ │ │ │ │ │ - updateAttribution: function() { │ │ │ │ │ - var attributions = []; │ │ │ │ │ - if (this.map && this.map.layers) { │ │ │ │ │ - for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if (layer.attribution && layer.getVisibility()) { │ │ │ │ │ - // add attribution only if attribution text is unique │ │ │ │ │ - if (OpenLayers.Util.indexOf( │ │ │ │ │ - attributions, layer.attribution) === -1) { │ │ │ │ │ - attributions.push(layer.attribution); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.div.innerHTML = OpenLayers.String.format(this.template, { │ │ │ │ │ - layers: attributions.join(this.separator) │ │ │ │ │ - }); │ │ │ │ │ + for (var i = 0, len = this.layerStates.length; i < len; i++) { │ │ │ │ │ + var layerState = this.layerStates[i]; │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if ((layerState.name != layer.name) || │ │ │ │ │ + (layerState.inRange != layer.inRange) || │ │ │ │ │ + (layerState.id != layer.id) || │ │ │ │ │ + (layerState.visibility != layer.visibility)) { │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Attribution" │ │ │ │ │ - }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/PanZoom.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.PanZoom │ │ │ │ │ - * The PanZoom is a visible control, composed of a │ │ │ │ │ - * <OpenLayers.Control.PanPanel> and a <OpenLayers.Control.ZoomPanel>. By │ │ │ │ │ - * default it is drawn in the upper left corner of the map. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.PanZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: slideFactor │ │ │ │ │ - * {Integer} Number of pixels by which we'll pan the map in any direction │ │ │ │ │ - * on clicking the arrow buttons. If you want to pan by some ratio │ │ │ │ │ - * of the map dimensions, use <slideRatio> instead. │ │ │ │ │ - */ │ │ │ │ │ - slideFactor: 50, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: slideRatio │ │ │ │ │ - * {Number} The fraction of map width/height by which we'll pan the map │ │ │ │ │ - * on clicking the arrow buttons. Default is null. If set, will │ │ │ │ │ - * override <slideFactor>. E.g. if slideRatio is .5, then the Pan Up │ │ │ │ │ - * button will pan up half the map height. │ │ │ │ │ - */ │ │ │ │ │ - slideRatio: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: buttons │ │ │ │ │ - * {Array(DOMElement)} Array of Button Divs │ │ │ │ │ - */ │ │ │ │ │ - buttons: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: position │ │ │ │ │ - * {<OpenLayers.Pixel>} │ │ │ │ │ - */ │ │ │ │ │ - position: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.PanZoom │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.position = new OpenLayers.Pixel(OpenLayers.Control.PanZoom.X, │ │ │ │ │ - OpenLayers.Control.PanZoom.Y); │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ } │ │ │ │ │ - this.removeButtons(); │ │ │ │ │ - this.buttons = null; │ │ │ │ │ - this.position = null; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * │ │ │ │ │ - * Properties: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ + * Method: redraw │ │ │ │ │ + * Goes through and takes the current state of the Map and rebuilds the │ │ │ │ │ + * control to display that state. Groups base layers into a │ │ │ │ │ + * radio-button group and lists each data layer with a checkbox. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A reference to the container div for the PanZoom control. │ │ │ │ │ - */ │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - // initialize our internal div │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - px = this.position; │ │ │ │ │ - │ │ │ │ │ - // place the controls │ │ │ │ │ - this.buttons = []; │ │ │ │ │ - │ │ │ │ │ - var sz = { │ │ │ │ │ - w: 18, │ │ │ │ │ - h: 18 │ │ │ │ │ - }; │ │ │ │ │ - var centered = new OpenLayers.Pixel(px.x + sz.w / 2, px.y); │ │ │ │ │ - │ │ │ │ │ - this._addButton("panup", "north-mini.png", centered, sz); │ │ │ │ │ - px.y = centered.y + sz.h; │ │ │ │ │ - this._addButton("panleft", "west-mini.png", px, sz); │ │ │ │ │ - this._addButton("panright", "east-mini.png", px.add(sz.w, 0), sz); │ │ │ │ │ - this._addButton("pandown", "south-mini.png", │ │ │ │ │ - centered.add(0, sz.h * 2), sz); │ │ │ │ │ - this._addButton("zoomin", "zoom-plus-mini.png", │ │ │ │ │ - centered.add(0, sz.h * 3 + 5), sz); │ │ │ │ │ - this._addButton("zoomworld", "zoom-world-mini.png", │ │ │ │ │ - centered.add(0, sz.h * 4 + 5), sz); │ │ │ │ │ - this._addButton("zoomout", "zoom-minus-mini.png", │ │ │ │ │ - centered.add(0, sz.h * 5 + 5), sz); │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: _addButton │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {String} │ │ │ │ │ - * img - {String} │ │ │ │ │ - * xy - {<OpenLayers.Pixel>} │ │ │ │ │ - * sz - {<OpenLayers.Size>} │ │ │ │ │ - * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} A Div (an alphaImageDiv, to be precise) that contains the │ │ │ │ │ - * image of the button, and has all the proper event handlers set. │ │ │ │ │ + * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ */ │ │ │ │ │ - _addButton: function(id, img, xy, sz) { │ │ │ │ │ - var imgLocation = OpenLayers.Util.getImageLocation(img); │ │ │ │ │ - var btn = OpenLayers.Util.createAlphaImageDiv( │ │ │ │ │ - this.id + "_" + id, │ │ │ │ │ - xy, sz, imgLocation, "absolute"); │ │ │ │ │ - btn.style.cursor = "pointer"; │ │ │ │ │ - //we want to add the outer div │ │ │ │ │ - this.div.appendChild(btn); │ │ │ │ │ - btn.action = id; │ │ │ │ │ - btn.className = "olButton"; │ │ │ │ │ + redraw: function() { │ │ │ │ │ + //if the state hasn't changed since last redraw, no need │ │ │ │ │ + // to do anything. Just return the existing div. │ │ │ │ │ + if (!this.checkRedraw()) { │ │ │ │ │ + return this.div; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - //we want to remember/reference the outer div │ │ │ │ │ - this.buttons.push(btn); │ │ │ │ │ - return btn; │ │ │ │ │ - }, │ │ │ │ │ + //clear out previous layers │ │ │ │ │ + this.clearLayersArray("base"); │ │ │ │ │ + this.clearLayersArray("data"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: _removeButton │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * btn - {Object} │ │ │ │ │ - */ │ │ │ │ │ - _removeButton: function(btn) { │ │ │ │ │ - this.div.removeChild(btn); │ │ │ │ │ - OpenLayers.Util.removeItem(this.buttons, btn); │ │ │ │ │ - }, │ │ │ │ │ + var containsOverlays = false; │ │ │ │ │ + var containsBaseLayers = false; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: removeButtons │ │ │ │ │ - */ │ │ │ │ │ - removeButtons: function() { │ │ │ │ │ - for (var i = this.buttons.length - 1; i >= 0; --i) { │ │ │ │ │ - this._removeButton(this.buttons[i]); │ │ │ │ │ + // Save state -- for checking layer if the map state changed. │ │ │ │ │ + // We save this before redrawing, because in the process of redrawing │ │ │ │ │ + // we will trigger more visibility changes, and we want to not redraw │ │ │ │ │ + // and enter an infinite loop. │ │ │ │ │ + var len = this.map.layers.length; │ │ │ │ │ + this.layerStates = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + this.layerStates[i] = { │ │ │ │ │ + 'name': layer.name, │ │ │ │ │ + 'visibility': layer.visibility, │ │ │ │ │ + 'inRange': layer.inRange, │ │ │ │ │ + 'id': layer.id │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: onButtonClick │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - var btn = evt.buttonElement; │ │ │ │ │ - switch (btn.action) { │ │ │ │ │ - case "panup": │ │ │ │ │ - this.map.pan(0, -this.getSlideFactor("h")); │ │ │ │ │ - break; │ │ │ │ │ - case "pandown": │ │ │ │ │ - this.map.pan(0, this.getSlideFactor("h")); │ │ │ │ │ - break; │ │ │ │ │ - case "panleft": │ │ │ │ │ - this.map.pan(-this.getSlideFactor("w"), 0); │ │ │ │ │ - break; │ │ │ │ │ - case "panright": │ │ │ │ │ - this.map.pan(this.getSlideFactor("w"), 0); │ │ │ │ │ - break; │ │ │ │ │ - case "zoomin": │ │ │ │ │ - this.map.zoomIn(); │ │ │ │ │ - break; │ │ │ │ │ - case "zoomout": │ │ │ │ │ - this.map.zoomOut(); │ │ │ │ │ - break; │ │ │ │ │ - case "zoomworld": │ │ │ │ │ - this.map.zoomToMaxExtent(); │ │ │ │ │ - break; │ │ │ │ │ + var layers = this.map.layers.slice(); │ │ │ │ │ + if (!this.ascending) { │ │ │ │ │ + layers.reverse(); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + var layer = layers[i]; │ │ │ │ │ + var baseLayer = layer.isBaseLayer; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getSlideFactor │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * dim - {String} "w" or "h" (for width or height). │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number} The slide factor for panning in the requested direction. │ │ │ │ │ - */ │ │ │ │ │ - getSlideFactor: function(dim) { │ │ │ │ │ - return this.slideRatio ? │ │ │ │ │ - this.map.getSize()[dim] * this.slideRatio : │ │ │ │ │ - this.slideFactor; │ │ │ │ │ - }, │ │ │ │ │ + if (layer.displayInLayerSwitcher) { │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.PanZoom" │ │ │ │ │ -}); │ │ │ │ │ + if (baseLayer) { │ │ │ │ │ + containsBaseLayers = true; │ │ │ │ │ + } else { │ │ │ │ │ + containsOverlays = true; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: X │ │ │ │ │ - * {Integer} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.PanZoom.X = 4; │ │ │ │ │ + // only check a baselayer if it is *the* baselayer, check data │ │ │ │ │ + // layers if they are visible │ │ │ │ │ + var checked = (baseLayer) ? (layer == this.map.baseLayer) : │ │ │ │ │ + layer.getVisibility(); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: Y │ │ │ │ │ - * {Integer} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.PanZoom.Y = 4; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Permalink.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + // create input element │ │ │ │ │ + var inputElem = document.createElement("input"), │ │ │ │ │ + // The input shall have an id attribute so we can use │ │ │ │ │ + // labels to interact with them. │ │ │ │ │ + inputId = OpenLayers.Util.createUniqueID( │ │ │ │ │ + this.id + "_input_" │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + inputElem.id = inputId; │ │ │ │ │ + inputElem.name = (baseLayer) ? this.id + "_baseLayers" : layer.name; │ │ │ │ │ + inputElem.type = (baseLayer) ? "radio" : "checkbox"; │ │ │ │ │ + inputElem.value = layer.name; │ │ │ │ │ + inputElem.checked = checked; │ │ │ │ │ + inputElem.defaultChecked = checked; │ │ │ │ │ + inputElem.className = "olButton"; │ │ │ │ │ + inputElem._layer = layer.id; │ │ │ │ │ + inputElem._layerSwitcher = this.id; │ │ │ │ │ │ │ │ │ │ + if (!baseLayer && !layer.inRange) { │ │ │ │ │ + inputElem.disabled = true; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Control/ArgParser.js │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ + // create span │ │ │ │ │ + var labelSpan = document.createElement("label"); │ │ │ │ │ + // this isn't the DOM attribute 'for', but an arbitrary name we │ │ │ │ │ + // use to find the appropriate input element in <onButtonClick> │ │ │ │ │ + labelSpan["for"] = inputElem.id; │ │ │ │ │ + OpenLayers.Element.addClass(labelSpan, "labelSpan olButton"); │ │ │ │ │ + labelSpan._layer = layer.id; │ │ │ │ │ + labelSpan._layerSwitcher = this.id; │ │ │ │ │ + if (!baseLayer && !layer.inRange) { │ │ │ │ │ + labelSpan.style.color = "gray"; │ │ │ │ │ + } │ │ │ │ │ + labelSpan.innerHTML = layer.name; │ │ │ │ │ + labelSpan.style.verticalAlign = (baseLayer) ? "bottom" : │ │ │ │ │ + "baseline"; │ │ │ │ │ + // create line break │ │ │ │ │ + var br = document.createElement("br"); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Permalink │ │ │ │ │ - * The Permalink control is hyperlink that will return the user to the │ │ │ │ │ - * current map view. By default it is drawn in the lower right corner of the │ │ │ │ │ - * map. The href is updated as the map is zoomed, panned and whilst layers │ │ │ │ │ - * are switched. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Permalink = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: argParserClass │ │ │ │ │ - * {Class} The ArgParser control class (not instance) to use with this │ │ │ │ │ - * control. │ │ │ │ │ - */ │ │ │ │ │ - argParserClass: OpenLayers.Control.ArgParser, │ │ │ │ │ + var groupArray = (baseLayer) ? this.baseLayers : │ │ │ │ │ + this.dataLayers; │ │ │ │ │ + groupArray.push({ │ │ │ │ │ + 'layer': layer, │ │ │ │ │ + 'inputElem': inputElem, │ │ │ │ │ + 'labelSpan': labelSpan │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: element │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - element: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: anchor │ │ │ │ │ - * {Boolean} This option changes 3 things: │ │ │ │ │ - * the character '#' is used in place of the character '?', │ │ │ │ │ - * the window.href is updated if no element is provided. │ │ │ │ │ - * When this option is set to true it's not recommend to provide │ │ │ │ │ - * a base without provide an element. │ │ │ │ │ - */ │ │ │ │ │ - anchor: false, │ │ │ │ │ + var groupDiv = (baseLayer) ? this.baseLayersDiv : │ │ │ │ │ + this.dataLayersDiv; │ │ │ │ │ + groupDiv.appendChild(inputElem); │ │ │ │ │ + groupDiv.appendChild(labelSpan); │ │ │ │ │ + groupDiv.appendChild(br); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: base │ │ │ │ │ - * {String} │ │ │ │ │ - */ │ │ │ │ │ - base: '', │ │ │ │ │ + // if no overlays, dont display the overlay label │ │ │ │ │ + this.dataLbl.style.display = (containsOverlays) ? "" : "none"; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: displayProjection │ │ │ │ │ - * {<OpenLayers.Projection>} Requires proj4js support. Projection used │ │ │ │ │ - * when creating the coordinates in the link. This will reproject the │ │ │ │ │ - * map coordinates into display coordinates. If you are using this │ │ │ │ │ - * functionality, the permalink which is last added to the map will │ │ │ │ │ - * determine the coordinate type which is read from the URL, which │ │ │ │ │ - * means you should not add permalinks with different │ │ │ │ │ - * displayProjections to the same map. │ │ │ │ │ - */ │ │ │ │ │ - displayProjection: null, │ │ │ │ │ + // if no baselayers, dont display the baselayer label │ │ │ │ │ + this.baseLbl.style.display = (containsBaseLayers) ? "" : "none"; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Permalink │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * element - {DOMElement} │ │ │ │ │ - * base - {String} │ │ │ │ │ - * options - {Object} options to the control. │ │ │ │ │ - * │ │ │ │ │ - * Or for anchor: │ │ │ │ │ - * options - {Object} options to the control. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(element, base, options) { │ │ │ │ │ - if (element !== null && typeof element == 'object' && !OpenLayers.Util.isElement(element)) { │ │ │ │ │ - options = element; │ │ │ │ │ - this.base = document.location.href; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (this.element != null) { │ │ │ │ │ - this.element = OpenLayers.Util.getElement(this.element); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.element = OpenLayers.Util.getElement(element); │ │ │ │ │ - this.base = base || document.location.href; │ │ │ │ │ - } │ │ │ │ │ + return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ + * Method: updateMap │ │ │ │ │ + * Cycles through the loaded data and base layer input arrays and makes │ │ │ │ │ + * the necessary calls to the Map object such that that the map's │ │ │ │ │ + * visual state corresponds to what the user has selected in │ │ │ │ │ + * the control. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.element && this.element.parentNode == this.div) { │ │ │ │ │ - this.div.removeChild(this.element); │ │ │ │ │ - this.element = null; │ │ │ │ │ + updateMap: function() { │ │ │ │ │ + │ │ │ │ │ + // set the newly selected base layer │ │ │ │ │ + for (var i = 0, len = this.baseLayers.length; i < len; i++) { │ │ │ │ │ + var layerEntry = this.baseLayers[i]; │ │ │ │ │ + if (layerEntry.inputElem.checked) { │ │ │ │ │ + this.map.setBaseLayer(layerEntry.layer, false); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister('moveend', this, this.updateLink); │ │ │ │ │ + │ │ │ │ │ + // set the correct visibilities for the overlays │ │ │ │ │ + for (var i = 0, len = this.dataLayers.length; i < len; i++) { │ │ │ │ │ + var layerEntry = this.dataLayers[i]; │ │ │ │ │ + layerEntry.layer.setVisibility(layerEntry.inputElem.checked); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the control. │ │ │ │ │ - * │ │ │ │ │ + * Method: maximizeControl │ │ │ │ │ + * Set up the labels and divs for the control │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ + * e - {Event} │ │ │ │ │ */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + maximizeControl: function(e) { │ │ │ │ │ │ │ │ │ │ - //make sure we have an arg parser attached │ │ │ │ │ - for (var i = 0, len = this.map.controls.length; i < len; i++) { │ │ │ │ │ - var control = this.map.controls[i]; │ │ │ │ │ - if (control.CLASS_NAME == this.argParserClass.CLASS_NAME) { │ │ │ │ │ + // set the div's width and height to empty values, so │ │ │ │ │ + // the div dimensions can be controlled by CSS │ │ │ │ │ + this.div.style.width = ""; │ │ │ │ │ + this.div.style.height = ""; │ │ │ │ │ │ │ │ │ │ - // If a permalink is added to the map, and an ArgParser already │ │ │ │ │ - // exists, we override the displayProjection to be the one │ │ │ │ │ - // on the permalink. │ │ │ │ │ - if (control.displayProjection != this.displayProjection) { │ │ │ │ │ - this.displayProjection = control.displayProjection; │ │ │ │ │ - } │ │ │ │ │ + this.showControls(false); │ │ │ │ │ │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (i == this.map.controls.length) { │ │ │ │ │ - this.map.addControl(new this.argParserClass({ │ │ │ │ │ - 'displayProjection': this.displayProjection │ │ │ │ │ - })); │ │ │ │ │ + if (e != null) { │ │ │ │ │ + OpenLayers.Event.stop(e); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ + * Method: minimizeControl │ │ │ │ │ + * Hide all the contents of the control, shrink the size, │ │ │ │ │ + * add the maximize icon │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * e - {Event} │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + minimizeControl: function(e) { │ │ │ │ │ │ │ │ │ │ - if (!this.element && !this.anchor) { │ │ │ │ │ - this.element = document.createElement("a"); │ │ │ │ │ - this.element.innerHTML = OpenLayers.i18n("Permalink"); │ │ │ │ │ - this.element.href = ""; │ │ │ │ │ - this.div.appendChild(this.element); │ │ │ │ │ - } │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - 'moveend': this.updateLink, │ │ │ │ │ - 'changelayer': this.updateLink, │ │ │ │ │ - 'changebaselayer': this.updateLink, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ + // to minimize the control we set its div's width │ │ │ │ │ + // and height to 0px, we cannot just set "display" │ │ │ │ │ + // to "none" because it would hide the maximize │ │ │ │ │ + // div │ │ │ │ │ + this.div.style.width = "0px"; │ │ │ │ │ + this.div.style.height = "0px"; │ │ │ │ │ │ │ │ │ │ - // Make it so there is at least a link even though the map may not have │ │ │ │ │ - // moved yet. │ │ │ │ │ - this.updateLink(); │ │ │ │ │ + this.showControls(true); │ │ │ │ │ │ │ │ │ │ - return this.div; │ │ │ │ │ + if (e != null) { │ │ │ │ │ + OpenLayers.Event.stop(e); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: updateLink │ │ │ │ │ + * Method: showControls │ │ │ │ │ + * Hide/Show all LayerSwitcher controls depending on whether we are │ │ │ │ │ + * minimized or not │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * minimize - {Boolean} │ │ │ │ │ */ │ │ │ │ │ - updateLink: function() { │ │ │ │ │ - var separator = this.anchor ? '#' : '?'; │ │ │ │ │ - var href = this.base; │ │ │ │ │ - var anchor = null; │ │ │ │ │ - if (href.indexOf("#") != -1 && this.anchor == false) { │ │ │ │ │ - anchor = href.substring(href.indexOf("#"), href.length); │ │ │ │ │ - } │ │ │ │ │ - if (href.indexOf(separator) != -1) { │ │ │ │ │ - href = href.substring(0, href.indexOf(separator)); │ │ │ │ │ - } │ │ │ │ │ - var splits = href.split("#"); │ │ │ │ │ - href = splits[0] + separator + OpenLayers.Util.getParameterString(this.createParams()); │ │ │ │ │ - if (anchor) { │ │ │ │ │ - href += anchor; │ │ │ │ │ - } │ │ │ │ │ - if (this.anchor && !this.element) { │ │ │ │ │ - window.location.href = href; │ │ │ │ │ - } else { │ │ │ │ │ - this.element.href = href; │ │ │ │ │ - } │ │ │ │ │ + showControls: function(minimize) { │ │ │ │ │ + │ │ │ │ │ + this.maximizeDiv.style.display = minimize ? "" : "none"; │ │ │ │ │ + this.minimizeDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ + │ │ │ │ │ + this.layersDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: createParams │ │ │ │ │ - * Creates the parameters that need to be encoded into the permalink url. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * center - {<OpenLayers.LonLat>} center to encode in the permalink. │ │ │ │ │ - * Defaults to the current map center. │ │ │ │ │ - * zoom - {Integer} zoom level to encode in the permalink. Defaults to the │ │ │ │ │ - * current map zoom level. │ │ │ │ │ - * layers - {Array(<OpenLayers.Layer>)} layers to encode in the permalink. │ │ │ │ │ - * Defaults to the current map layers. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Hash of parameters that will be url-encoded into the │ │ │ │ │ - * permalink. │ │ │ │ │ + * Method: loadContents │ │ │ │ │ + * Set up the labels and divs for the control │ │ │ │ │ */ │ │ │ │ │ - createParams: function(center, zoom, layers) { │ │ │ │ │ - center = center || this.map.getCenter(); │ │ │ │ │ - │ │ │ │ │ - var params = OpenLayers.Util.getParameters(this.base); │ │ │ │ │ + loadContents: function() { │ │ │ │ │ │ │ │ │ │ - // If there's still no center, map is not initialized yet. │ │ │ │ │ - // Break out of this function, and simply return the params from the │ │ │ │ │ - // base link. │ │ │ │ │ - if (center) { │ │ │ │ │ + // layers list div │ │ │ │ │ + this.layersDiv = document.createElement("div"); │ │ │ │ │ + this.layersDiv.id = this.id + "_layersDiv"; │ │ │ │ │ + OpenLayers.Element.addClass(this.layersDiv, "layersDiv"); │ │ │ │ │ │ │ │ │ │ - //zoom │ │ │ │ │ - params.zoom = zoom || this.map.getZoom(); │ │ │ │ │ + this.baseLbl = document.createElement("div"); │ │ │ │ │ + this.baseLbl.innerHTML = OpenLayers.i18n("Base Layer"); │ │ │ │ │ + OpenLayers.Element.addClass(this.baseLbl, "baseLbl"); │ │ │ │ │ │ │ │ │ │ - //lon,lat │ │ │ │ │ - var lat = center.lat; │ │ │ │ │ - var lon = center.lon; │ │ │ │ │ + this.baseLayersDiv = document.createElement("div"); │ │ │ │ │ + OpenLayers.Element.addClass(this.baseLayersDiv, "baseLayersDiv"); │ │ │ │ │ │ │ │ │ │ - if (this.displayProjection) { │ │ │ │ │ - var mapPosition = OpenLayers.Projection.transform({ │ │ │ │ │ - x: lon, │ │ │ │ │ - y: lat │ │ │ │ │ - }, │ │ │ │ │ - this.map.getProjectionObject(), │ │ │ │ │ - this.displayProjection); │ │ │ │ │ - lon = mapPosition.x; │ │ │ │ │ - lat = mapPosition.y; │ │ │ │ │ - } │ │ │ │ │ - params.lat = Math.round(lat * 100000) / 100000; │ │ │ │ │ - params.lon = Math.round(lon * 100000) / 100000; │ │ │ │ │ + this.dataLbl = document.createElement("div"); │ │ │ │ │ + this.dataLbl.innerHTML = OpenLayers.i18n("Overlays"); │ │ │ │ │ + OpenLayers.Element.addClass(this.dataLbl, "dataLbl"); │ │ │ │ │ │ │ │ │ │ - //layers │ │ │ │ │ - layers = layers || this.map.layers; │ │ │ │ │ - params.layers = ''; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - var layer = layers[i]; │ │ │ │ │ + this.dataLayersDiv = document.createElement("div"); │ │ │ │ │ + OpenLayers.Element.addClass(this.dataLayersDiv, "dataLayersDiv"); │ │ │ │ │ │ │ │ │ │ - if (layer.isBaseLayer) { │ │ │ │ │ - params.layers += (layer == this.map.baseLayer) ? "B" : "0"; │ │ │ │ │ - } else { │ │ │ │ │ - params.layers += (layer.getVisibility()) ? "T" : "F"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + if (this.ascending) { │ │ │ │ │ + this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ + } else { │ │ │ │ │ + this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - return params; │ │ │ │ │ + this.div.appendChild(this.layersDiv); │ │ │ │ │ + │ │ │ │ │ + // maximize button div │ │ │ │ │ + var img = OpenLayers.Util.getImageLocation('layer-switcher-maximize.png'); │ │ │ │ │ + this.maximizeDiv = OpenLayers.Util.createAlphaImageDiv( │ │ │ │ │ + "OpenLayers_Control_MaximizeDiv", │ │ │ │ │ + null, │ │ │ │ │ + null, │ │ │ │ │ + img, │ │ │ │ │ + "absolute"); │ │ │ │ │ + OpenLayers.Element.addClass(this.maximizeDiv, "maximizeDiv olButton"); │ │ │ │ │ + this.maximizeDiv.style.display = "none"; │ │ │ │ │ + │ │ │ │ │ + this.div.appendChild(this.maximizeDiv); │ │ │ │ │ + │ │ │ │ │ + // minimize button div │ │ │ │ │ + var img = OpenLayers.Util.getImageLocation('layer-switcher-minimize.png'); │ │ │ │ │ + this.minimizeDiv = OpenLayers.Util.createAlphaImageDiv( │ │ │ │ │ + "OpenLayers_Control_MinimizeDiv", │ │ │ │ │ + null, │ │ │ │ │ + null, │ │ │ │ │ + img, │ │ │ │ │ + "absolute"); │ │ │ │ │ + OpenLayers.Element.addClass(this.minimizeDiv, "minimizeDiv olButton"); │ │ │ │ │ + this.minimizeDiv.style.display = "none"; │ │ │ │ │ + │ │ │ │ │ + this.div.appendChild(this.minimizeDiv); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Permalink" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.LayerSwitcher" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Control/Split.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ @@ -81579,433 +81111,14 @@ │ │ │ │ │ } │ │ │ │ │ OpenLayers.Control.prototype.destroy.call(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.Split" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/PanZoomBar.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control/PanZoom.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.PanZoomBar │ │ │ │ │ - * The PanZoomBar is a visible control composed of a │ │ │ │ │ - * <OpenLayers.Control.PanPanel> and a <OpenLayers.Control.ZoomBar>. │ │ │ │ │ - * By default it is displayed in the upper left corner of the map as 4 │ │ │ │ │ - * directional arrows above a vertical slider. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control.PanZoom> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.PanZoomBar = OpenLayers.Class(OpenLayers.Control.PanZoom, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomStopWidth │ │ │ │ │ - */ │ │ │ │ │ - zoomStopWidth: 18, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomStopHeight │ │ │ │ │ - */ │ │ │ │ │ - zoomStopHeight: 11, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: slider │ │ │ │ │ - */ │ │ │ │ │ - slider: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: sliderEvents │ │ │ │ │ - * {<OpenLayers.Events>} │ │ │ │ │ - */ │ │ │ │ │ - sliderEvents: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: zoombarDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - zoombarDiv: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: zoomWorldIcon │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - zoomWorldIcon: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: panIcons │ │ │ │ │ - * {Boolean} Set this property to false not to display the pan icons. If │ │ │ │ │ - * false the zoom world icon is placed under the zoom bar. Defaults to │ │ │ │ │ - * true. │ │ │ │ │ - */ │ │ │ │ │ - panIcons: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: forceFixedZoomLevel │ │ │ │ │ - * {Boolean} Force a fixed zoom level even though the map has │ │ │ │ │ - * fractionalZoom │ │ │ │ │ - */ │ │ │ │ │ - forceFixedZoomLevel: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: mouseDragStart │ │ │ │ │ - * {<OpenLayers.Pixel>} │ │ │ │ │ - */ │ │ │ │ │ - mouseDragStart: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: deltaY │ │ │ │ │ - * {Number} The cumulative vertical pixel offset during a zoom bar drag. │ │ │ │ │ - */ │ │ │ │ │ - deltaY: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: zoomStart │ │ │ │ │ - * {<OpenLayers.Pixel>} │ │ │ │ │ - */ │ │ │ │ │ - zoomStart: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.PanZoomBar │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - │ │ │ │ │ - this._removeZoomBar(); │ │ │ │ │ - │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - "changebaselayer": this.redraw, │ │ │ │ │ - "updatesize": this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Control.PanZoom.prototype.destroy.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - delete this.mouseDragStart; │ │ │ │ │ - delete this.zoomStart; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.PanZoom.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - "changebaselayer": this.redraw, │ │ │ │ │ - "updatesize": this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: redraw │ │ │ │ │ - * clear the div and start over. │ │ │ │ │ - */ │ │ │ │ │ - redraw: function() { │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.removeButtons(); │ │ │ │ │ - this._removeZoomBar(); │ │ │ │ │ - } │ │ │ │ │ - this.draw(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * px - {<OpenLayers.Pixel>} │ │ │ │ │ - */ │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - // initialize our internal div │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - px = this.position.clone(); │ │ │ │ │ - │ │ │ │ │ - // place the controls │ │ │ │ │ - this.buttons = []; │ │ │ │ │ - │ │ │ │ │ - var sz = { │ │ │ │ │ - w: 18, │ │ │ │ │ - h: 18 │ │ │ │ │ - }; │ │ │ │ │ - if (this.panIcons) { │ │ │ │ │ - var centered = new OpenLayers.Pixel(px.x + sz.w / 2, px.y); │ │ │ │ │ - var wposition = sz.w; │ │ │ │ │ - │ │ │ │ │ - if (this.zoomWorldIcon) { │ │ │ │ │ - centered = new OpenLayers.Pixel(px.x + sz.w, px.y); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - this._addButton("panup", "north-mini.png", centered, sz); │ │ │ │ │ - px.y = centered.y + sz.h; │ │ │ │ │ - this._addButton("panleft", "west-mini.png", px, sz); │ │ │ │ │ - if (this.zoomWorldIcon) { │ │ │ │ │ - this._addButton("zoomworld", "zoom-world-mini.png", px.add(sz.w, 0), sz); │ │ │ │ │ - │ │ │ │ │ - wposition *= 2; │ │ │ │ │ - } │ │ │ │ │ - this._addButton("panright", "east-mini.png", px.add(wposition, 0), sz); │ │ │ │ │ - this._addButton("pandown", "south-mini.png", centered.add(0, sz.h * 2), sz); │ │ │ │ │ - this._addButton("zoomin", "zoom-plus-mini.png", centered.add(0, sz.h * 3 + 5), sz); │ │ │ │ │ - centered = this._addZoomBar(centered.add(0, sz.h * 4 + 5)); │ │ │ │ │ - this._addButton("zoomout", "zoom-minus-mini.png", centered, sz); │ │ │ │ │ - } else { │ │ │ │ │ - this._addButton("zoomin", "zoom-plus-mini.png", px, sz); │ │ │ │ │ - centered = this._addZoomBar(px.add(0, sz.h)); │ │ │ │ │ - this._addButton("zoomout", "zoom-minus-mini.png", centered, sz); │ │ │ │ │ - if (this.zoomWorldIcon) { │ │ │ │ │ - centered = centered.add(0, sz.h + 3); │ │ │ │ │ - this._addButton("zoomworld", "zoom-world-mini.png", centered, sz); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: _addZoomBar │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * centered - {<OpenLayers.Pixel>} where zoombar drawing is to start. │ │ │ │ │ - */ │ │ │ │ │ - _addZoomBar: function(centered) { │ │ │ │ │ - var imgLocation = OpenLayers.Util.getImageLocation("slider.png"); │ │ │ │ │ - var id = this.id + "_" + this.map.id; │ │ │ │ │ - var minZoom = this.map.getMinZoom(); │ │ │ │ │ - var zoomsToEnd = this.map.getNumZoomLevels() - 1 - this.map.getZoom(); │ │ │ │ │ - var slider = OpenLayers.Util.createAlphaImageDiv(id, │ │ │ │ │ - centered.add(-1, zoomsToEnd * this.zoomStopHeight), { │ │ │ │ │ - w: 20, │ │ │ │ │ - h: 9 │ │ │ │ │ - }, │ │ │ │ │ - imgLocation, │ │ │ │ │ - "absolute"); │ │ │ │ │ - slider.style.cursor = "move"; │ │ │ │ │ - this.slider = slider; │ │ │ │ │ - │ │ │ │ │ - this.sliderEvents = new OpenLayers.Events(this, slider, null, true, { │ │ │ │ │ - includeXY: true │ │ │ │ │ - }); │ │ │ │ │ - this.sliderEvents.on({ │ │ │ │ │ - "touchstart": this.zoomBarDown, │ │ │ │ │ - "touchmove": this.zoomBarDrag, │ │ │ │ │ - "touchend": this.zoomBarUp, │ │ │ │ │ - "mousedown": this.zoomBarDown, │ │ │ │ │ - "mousemove": this.zoomBarDrag, │ │ │ │ │ - "mouseup": this.zoomBarUp │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - var sz = { │ │ │ │ │ - w: this.zoomStopWidth, │ │ │ │ │ - h: this.zoomStopHeight * (this.map.getNumZoomLevels() - minZoom) │ │ │ │ │ - }; │ │ │ │ │ - var imgLocation = OpenLayers.Util.getImageLocation("zoombar.png"); │ │ │ │ │ - var div = null; │ │ │ │ │ - │ │ │ │ │ - if (OpenLayers.Util.alphaHack()) { │ │ │ │ │ - var id = this.id + "_" + this.map.id; │ │ │ │ │ - div = OpenLayers.Util.createAlphaImageDiv(id, centered, { │ │ │ │ │ - w: sz.w, │ │ │ │ │ - h: this.zoomStopHeight │ │ │ │ │ - }, │ │ │ │ │ - imgLocation, │ │ │ │ │ - "absolute", null, "crop"); │ │ │ │ │ - div.style.height = sz.h + "px"; │ │ │ │ │ - } else { │ │ │ │ │ - div = OpenLayers.Util.createDiv( │ │ │ │ │ - 'OpenLayers_Control_PanZoomBar_Zoombar' + this.map.id, │ │ │ │ │ - centered, │ │ │ │ │ - sz, │ │ │ │ │ - imgLocation); │ │ │ │ │ - } │ │ │ │ │ - div.style.cursor = "pointer"; │ │ │ │ │ - div.className = "olButton"; │ │ │ │ │ - this.zoombarDiv = div; │ │ │ │ │ - │ │ │ │ │ - this.div.appendChild(div); │ │ │ │ │ - │ │ │ │ │ - this.startTop = parseInt(div.style.top); │ │ │ │ │ - this.div.appendChild(slider); │ │ │ │ │ - │ │ │ │ │ - this.map.events.register("zoomend", this, this.moveZoomBar); │ │ │ │ │ - │ │ │ │ │ - centered = centered.add(0, │ │ │ │ │ - this.zoomStopHeight * (this.map.getNumZoomLevels() - minZoom)); │ │ │ │ │ - return centered; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: _removeZoomBar │ │ │ │ │ - */ │ │ │ │ │ - _removeZoomBar: function() { │ │ │ │ │ - this.sliderEvents.un({ │ │ │ │ │ - "touchstart": this.zoomBarDown, │ │ │ │ │ - "touchmove": this.zoomBarDrag, │ │ │ │ │ - "touchend": this.zoomBarUp, │ │ │ │ │ - "mousedown": this.zoomBarDown, │ │ │ │ │ - "mousemove": this.zoomBarDrag, │ │ │ │ │ - "mouseup": this.zoomBarUp │ │ │ │ │ - }); │ │ │ │ │ - this.sliderEvents.destroy(); │ │ │ │ │ - │ │ │ │ │ - this.div.removeChild(this.zoombarDiv); │ │ │ │ │ - this.zoombarDiv = null; │ │ │ │ │ - this.div.removeChild(this.slider); │ │ │ │ │ - this.slider = null; │ │ │ │ │ - │ │ │ │ │ - this.map.events.unregister("zoomend", this, this.moveZoomBar); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onButtonClick │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - OpenLayers.Control.PanZoom.prototype.onButtonClick.apply(this, arguments); │ │ │ │ │ - if (evt.buttonElement === this.zoombarDiv) { │ │ │ │ │ - var levels = evt.buttonXY.y / this.zoomStopHeight; │ │ │ │ │ - if (this.forceFixedZoomLevel || !this.map.fractionalZoom) { │ │ │ │ │ - levels = Math.floor(levels); │ │ │ │ │ - } │ │ │ │ │ - var zoom = (this.map.getNumZoomLevels() - 1) - levels; │ │ │ │ │ - zoom = Math.min(Math.max(zoom, 0), this.map.getNumZoomLevels() - 1); │ │ │ │ │ - this.map.zoomTo(zoom); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: passEventToSlider │ │ │ │ │ - * This function is used to pass events that happen on the div, or the map, │ │ │ │ │ - * through to the slider, which then does its moving thing. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - */ │ │ │ │ │ - passEventToSlider: function(evt) { │ │ │ │ │ - this.sliderEvents.handleBrowserEvent(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /* │ │ │ │ │ - * Method: zoomBarDown │ │ │ │ │ - * event listener for clicks on the slider │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - */ │ │ │ │ │ - zoomBarDown: function(evt) { │ │ │ │ │ - if (!OpenLayers.Event.isLeftClick(evt) && !OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - "touchmove": this.passEventToSlider, │ │ │ │ │ - "mousemove": this.passEventToSlider, │ │ │ │ │ - "mouseup": this.passEventToSlider, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.mouseDragStart = evt.xy.clone(); │ │ │ │ │ - this.zoomStart = evt.xy.clone(); │ │ │ │ │ - this.div.style.cursor = "move"; │ │ │ │ │ - // reset the div offsets just in case the div moved │ │ │ │ │ - this.zoombarDiv.offsets = null; │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /* │ │ │ │ │ - * Method: zoomBarDrag │ │ │ │ │ - * This is what happens when a click has occurred, and the client is │ │ │ │ │ - * dragging. Here we must ensure that the slider doesn't go beyond the │ │ │ │ │ - * bottom/top of the zoombar div, as well as moving the slider to its new │ │ │ │ │ - * visual location │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - */ │ │ │ │ │ - zoomBarDrag: function(evt) { │ │ │ │ │ - if (this.mouseDragStart != null) { │ │ │ │ │ - var deltaY = this.mouseDragStart.y - evt.xy.y; │ │ │ │ │ - var offsets = OpenLayers.Util.pagePosition(this.zoombarDiv); │ │ │ │ │ - if ((evt.clientY - offsets[1]) > 0 && │ │ │ │ │ - (evt.clientY - offsets[1]) < parseInt(this.zoombarDiv.style.height) - 2) { │ │ │ │ │ - var newTop = parseInt(this.slider.style.top) - deltaY; │ │ │ │ │ - this.slider.style.top = newTop + "px"; │ │ │ │ │ - this.mouseDragStart = evt.xy.clone(); │ │ │ │ │ - } │ │ │ │ │ - // set cumulative displacement │ │ │ │ │ - this.deltaY = this.zoomStart.y - evt.xy.y; │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /* │ │ │ │ │ - * Method: zoomBarUp │ │ │ │ │ - * Perform cleanup when a mouseup event is received -- discover new zoom │ │ │ │ │ - * level and switch to it. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - */ │ │ │ │ │ - zoomBarUp: function(evt) { │ │ │ │ │ - if (!OpenLayers.Event.isLeftClick(evt) && evt.type !== "touchend") { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (this.mouseDragStart) { │ │ │ │ │ - this.div.style.cursor = ""; │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - "touchmove": this.passEventToSlider, │ │ │ │ │ - "mouseup": this.passEventToSlider, │ │ │ │ │ - "mousemove": this.passEventToSlider, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - var zoomLevel = this.map.zoom; │ │ │ │ │ - if (!this.forceFixedZoomLevel && this.map.fractionalZoom) { │ │ │ │ │ - zoomLevel += this.deltaY / this.zoomStopHeight; │ │ │ │ │ - zoomLevel = Math.min(Math.max(zoomLevel, 0), │ │ │ │ │ - this.map.getNumZoomLevels() - 1); │ │ │ │ │ - } else { │ │ │ │ │ - zoomLevel += this.deltaY / this.zoomStopHeight; │ │ │ │ │ - zoomLevel = Math.max(Math.round(zoomLevel), 0); │ │ │ │ │ - } │ │ │ │ │ - this.map.zoomTo(zoomLevel); │ │ │ │ │ - this.mouseDragStart = null; │ │ │ │ │ - this.zoomStart = null; │ │ │ │ │ - this.deltaY = 0; │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /* │ │ │ │ │ - * Method: moveZoomBar │ │ │ │ │ - * Change the location of the slider to match the current zoom level. │ │ │ │ │ - */ │ │ │ │ │ - moveZoomBar: function() { │ │ │ │ │ - var newTop = │ │ │ │ │ - ((this.map.getNumZoomLevels() - 1) - this.map.getZoom()) * │ │ │ │ │ - this.zoomStopHeight + this.startTop + 1; │ │ │ │ │ - this.slider.style.top = newTop + "px"; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.PanZoomBar" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ OpenLayers/Control/DragFeature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -82369,238 +81482,14 @@ │ │ │ │ │ this.handlers.feature.setMap(map); │ │ │ │ │ OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.DragFeature" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/ScaleLine.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.ScaleLine │ │ │ │ │ - * The ScaleLine displays a small line indicator representing the current │ │ │ │ │ - * map scale on the map. By default it is drawn in the lower left corner of │ │ │ │ │ - * the map. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - * │ │ │ │ │ - * Is a very close copy of: │ │ │ │ │ - * - <OpenLayers.Control.Scale> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.ScaleLine = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: maxWidth │ │ │ │ │ - * {Integer} Maximum width of the scale line in pixels. Default is 100. │ │ │ │ │ - */ │ │ │ │ │ - maxWidth: 100, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: topOutUnits │ │ │ │ │ - * {String} Units for zoomed out on top bar. Default is km. │ │ │ │ │ - */ │ │ │ │ │ - topOutUnits: "km", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: topInUnits │ │ │ │ │ - * {String} Units for zoomed in on top bar. Default is m. │ │ │ │ │ - */ │ │ │ │ │ - topInUnits: "m", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: bottomOutUnits │ │ │ │ │ - * {String} Units for zoomed out on bottom bar. Default is mi. │ │ │ │ │ - */ │ │ │ │ │ - bottomOutUnits: "mi", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: bottomInUnits │ │ │ │ │ - * {String} Units for zoomed in on bottom bar. Default is ft. │ │ │ │ │ - */ │ │ │ │ │ - bottomInUnits: "ft", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: eTop │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - eTop: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: eBottom │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - eBottom: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geodesic │ │ │ │ │ - * {Boolean} Use geodesic measurement. Default is false. The recommended │ │ │ │ │ - * setting for maps in EPSG:4326 is false, and true EPSG:900913. If set to │ │ │ │ │ - * true, the scale will be calculated based on the horizontal size of the │ │ │ │ │ - * pixel in the center of the map viewport. │ │ │ │ │ - */ │ │ │ │ │ - geodesic: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.ScaleLine │ │ │ │ │ - * Create a new scale line control. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be used │ │ │ │ │ - * to extend the control. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (!this.eTop) { │ │ │ │ │ - // stick in the top bar │ │ │ │ │ - this.eTop = document.createElement("div"); │ │ │ │ │ - this.eTop.className = this.displayClass + "Top"; │ │ │ │ │ - var theLen = this.topInUnits.length; │ │ │ │ │ - this.div.appendChild(this.eTop); │ │ │ │ │ - if ((this.topOutUnits == "") || (this.topInUnits == "")) { │ │ │ │ │ - this.eTop.style.visibility = "hidden"; │ │ │ │ │ - } else { │ │ │ │ │ - this.eTop.style.visibility = "visible"; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // and the bottom bar │ │ │ │ │ - this.eBottom = document.createElement("div"); │ │ │ │ │ - this.eBottom.className = this.displayClass + "Bottom"; │ │ │ │ │ - this.div.appendChild(this.eBottom); │ │ │ │ │ - if ((this.bottomOutUnits == "") || (this.bottomInUnits == "")) { │ │ │ │ │ - this.eBottom.style.visibility = "hidden"; │ │ │ │ │ - } else { │ │ │ │ │ - this.eBottom.style.visibility = "visible"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.map.events.register('moveend', this, this.update); │ │ │ │ │ - this.update(); │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getBarLen │ │ │ │ │ - * Given a number, round it down to the nearest 1,2,5 times a power of 10. │ │ │ │ │ - * That seems a fairly useful set of number groups to use. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * maxLen - {float} the number we're rounding down from │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} the rounded number (less than or equal to maxLen) │ │ │ │ │ - */ │ │ │ │ │ - getBarLen: function(maxLen) { │ │ │ │ │ - // nearest power of 10 lower than maxLen │ │ │ │ │ - var digits = parseInt(Math.log(maxLen) / Math.log(10)); │ │ │ │ │ - var pow10 = Math.pow(10, digits); │ │ │ │ │ - │ │ │ │ │ - // ok, find first character │ │ │ │ │ - var firstChar = parseInt(maxLen / pow10); │ │ │ │ │ - │ │ │ │ │ - // right, put it into the correct bracket │ │ │ │ │ - var barLen; │ │ │ │ │ - if (firstChar > 5) { │ │ │ │ │ - barLen = 5; │ │ │ │ │ - } else if (firstChar > 2) { │ │ │ │ │ - barLen = 2; │ │ │ │ │ - } else { │ │ │ │ │ - barLen = 1; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // scale it up the correct power of 10 │ │ │ │ │ - return barLen * pow10; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: update │ │ │ │ │ - * Update the size of the bars, and the labels they contain. │ │ │ │ │ - */ │ │ │ │ │ - update: function() { │ │ │ │ │ - var res = this.map.getResolution(); │ │ │ │ │ - if (!res) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var curMapUnits = this.map.getUnits(); │ │ │ │ │ - var inches = OpenLayers.INCHES_PER_UNIT; │ │ │ │ │ - │ │ │ │ │ - // convert maxWidth to map units │ │ │ │ │ - var maxSizeData = this.maxWidth * res * inches[curMapUnits]; │ │ │ │ │ - var geodesicRatio = 1; │ │ │ │ │ - if (this.geodesic === true) { │ │ │ │ │ - var maxSizeGeodesic = (this.map.getGeodesicPixelSize().w || │ │ │ │ │ - 0.000001) * this.maxWidth; │ │ │ │ │ - var maxSizeKilometers = maxSizeData / inches["km"]; │ │ │ │ │ - geodesicRatio = maxSizeGeodesic / maxSizeKilometers; │ │ │ │ │ - maxSizeData *= geodesicRatio; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // decide whether to use large or small scale units │ │ │ │ │ - var topUnits; │ │ │ │ │ - var bottomUnits; │ │ │ │ │ - if (maxSizeData > 100000) { │ │ │ │ │ - topUnits = this.topOutUnits; │ │ │ │ │ - bottomUnits = this.bottomOutUnits; │ │ │ │ │ - } else { │ │ │ │ │ - topUnits = this.topInUnits; │ │ │ │ │ - bottomUnits = this.bottomInUnits; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // and to map units units │ │ │ │ │ - var topMax = maxSizeData / inches[topUnits]; │ │ │ │ │ - var bottomMax = maxSizeData / inches[bottomUnits]; │ │ │ │ │ - │ │ │ │ │ - // now trim this down to useful block length │ │ │ │ │ - var topRounded = this.getBarLen(topMax); │ │ │ │ │ - var bottomRounded = this.getBarLen(bottomMax); │ │ │ │ │ - │ │ │ │ │ - // and back to display units │ │ │ │ │ - topMax = topRounded / inches[curMapUnits] * inches[topUnits]; │ │ │ │ │ - bottomMax = bottomRounded / inches[curMapUnits] * inches[bottomUnits]; │ │ │ │ │ - │ │ │ │ │ - // and to pixel units │ │ │ │ │ - var topPx = topMax / res / geodesicRatio; │ │ │ │ │ - var bottomPx = bottomMax / res / geodesicRatio; │ │ │ │ │ - │ │ │ │ │ - // now set the pixel widths │ │ │ │ │ - // and the values inside them │ │ │ │ │ - │ │ │ │ │ - if (this.eBottom.style.visibility == "visible") { │ │ │ │ │ - this.eBottom.style.width = Math.round(bottomPx) + "px"; │ │ │ │ │ - this.eBottom.innerHTML = bottomRounded + " " + bottomUnits; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.eTop.style.visibility == "visible") { │ │ │ │ │ - this.eTop.style.width = Math.round(topPx) + "px"; │ │ │ │ │ - this.eTop.innerHTML = topRounded + " " + topUnits; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ScaleLine" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ OpenLayers/Control/TransformFeature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ @@ -83235,1053 +82124,283 @@ │ │ │ │ │ this.dragControl = null; │ │ │ │ │ OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.TransformFeature" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/KeyboardDefaults.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Handler/Keyboard.js │ │ │ │ │ - * @requires OpenLayers/Events.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.KeyboardDefaults │ │ │ │ │ - * The KeyboardDefaults control adds panning and zooming functions, controlled │ │ │ │ │ - * with the keyboard. By default arrow keys pan, +/- keys zoom & Page Up/Page │ │ │ │ │ - * Down/Home/End scroll by three quarters of a page. │ │ │ │ │ - * │ │ │ │ │ - * This control has no visible appearance. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.KeyboardDefaults = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ - */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: slideFactor │ │ │ │ │ - * Pixels to slide by. │ │ │ │ │ - */ │ │ │ │ │ - slideFactor: 75, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: observeElement │ │ │ │ │ - * {DOMelement|String} The DOM element to handle keys for. You │ │ │ │ │ - * can use the map div here, to have the navigation keys │ │ │ │ │ - * work when the map div has the focus. If undefined the │ │ │ │ │ - * document is used. │ │ │ │ │ - */ │ │ │ │ │ - observeElement: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.KeyboardDefaults │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Create handler. │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - var observeElement = this.observeElement || document; │ │ │ │ │ - this.handler = new OpenLayers.Handler.Keyboard(this, { │ │ │ │ │ - "keydown": this.defaultKeyPress │ │ │ │ │ - }, { │ │ │ │ │ - observeElement: observeElement │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: defaultKeyPress │ │ │ │ │ - * When handling the key event, we only use evt.keyCode. This holds │ │ │ │ │ - * some drawbacks, though we get around them below. When interpretting │ │ │ │ │ - * the keycodes below (including the comments associated with them), │ │ │ │ │ - * consult the URL below. For instance, the Safari browser returns │ │ │ │ │ - * "IE keycodes", and so is supported by any keycode labeled "IE". │ │ │ │ │ - * │ │ │ │ │ - * Very informative URL: │ │ │ │ │ - * http://unixpapa.com/js/key.html │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - defaultKeyPress: function(evt) { │ │ │ │ │ - var size, handled = true; │ │ │ │ │ - │ │ │ │ │ - var target = OpenLayers.Event.element(evt); │ │ │ │ │ - if (target && │ │ │ │ │ - (target.tagName == 'INPUT' || │ │ │ │ │ - target.tagName == 'TEXTAREA' || │ │ │ │ │ - target.tagName == 'SELECT')) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - switch (evt.keyCode) { │ │ │ │ │ - case OpenLayers.Event.KEY_LEFT: │ │ │ │ │ - this.map.pan(-this.slideFactor, 0); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Event.KEY_RIGHT: │ │ │ │ │ - this.map.pan(this.slideFactor, 0); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Event.KEY_UP: │ │ │ │ │ - this.map.pan(0, -this.slideFactor); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Event.KEY_DOWN: │ │ │ │ │ - this.map.pan(0, this.slideFactor); │ │ │ │ │ - break; │ │ │ │ │ - │ │ │ │ │ - case 33: // Page Up. Same in all browsers. │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - this.map.pan(0, -0.75 * size.h); │ │ │ │ │ - break; │ │ │ │ │ - case 34: // Page Down. Same in all browsers. │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - this.map.pan(0, 0.75 * size.h); │ │ │ │ │ - break; │ │ │ │ │ - case 35: // End. Same in all browsers. │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - this.map.pan(0.75 * size.w, 0); │ │ │ │ │ - break; │ │ │ │ │ - case 36: // Home. Same in all browsers. │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - this.map.pan(-0.75 * size.w, 0); │ │ │ │ │ - break; │ │ │ │ │ - │ │ │ │ │ - case 43: // +/= (ASCII), keypad + (ASCII, Opera) │ │ │ │ │ - case 61: // +/= (Mozilla, Opera, some ASCII) │ │ │ │ │ - case 187: // +/= (IE) │ │ │ │ │ - case 107: // keypad + (IE, Mozilla) │ │ │ │ │ - this.map.zoomIn(); │ │ │ │ │ - break; │ │ │ │ │ - case 45: // -/_ (ASCII, Opera), keypad - (ASCII, Opera) │ │ │ │ │ - case 109: // -/_ (Mozilla), keypad - (Mozilla, IE) │ │ │ │ │ - case 189: // -/_ (IE) │ │ │ │ │ - case 95: // -/_ (some ASCII) │ │ │ │ │ - this.map.zoomOut(); │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - handled = false; │ │ │ │ │ - } │ │ │ │ │ - if (handled) { │ │ │ │ │ - // prevent browser default not to move the page │ │ │ │ │ - // when moving the page with the keyboard │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.KeyboardDefaults" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/UTFGrid.js │ │ │ │ │ + OpenLayers/Control/GetFeature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Handler/Hover.js │ │ │ │ │ * @requires OpenLayers/Handler/Click.js │ │ │ │ │ + * @requires OpenLayers/Handler/Box.js │ │ │ │ │ + * @requires OpenLayers/Handler/Hover.js │ │ │ │ │ + * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.UTFGrid │ │ │ │ │ - * │ │ │ │ │ - * This Control provides behavior associated with UTFGrid Layers. │ │ │ │ │ - * These 'hit grids' provide underlying feature attributes without │ │ │ │ │ - * calling the server (again). This control allows Mousemove, Hovering │ │ │ │ │ - * and Click events to trigger callbacks that use the attributes in │ │ │ │ │ - * whatever way you need. │ │ │ │ │ - * │ │ │ │ │ - * The most common example may be a UTFGrid layer containing feature │ │ │ │ │ - * attributes that are displayed in a div as you mouseover. │ │ │ │ │ - * │ │ │ │ │ - * Example Code: │ │ │ │ │ - * │ │ │ │ │ - * (start code) │ │ │ │ │ - * var world_utfgrid = new OpenLayers.Layer.UTFGrid( │ │ │ │ │ - * 'UTFGrid Layer', │ │ │ │ │ - * "http://tiles/world_utfgrid/${z}/${x}/${y}.json" │ │ │ │ │ - * ); │ │ │ │ │ - * map.addLayer(world_utfgrid); │ │ │ │ │ - * │ │ │ │ │ - * var control = new OpenLayers.Control.UTFGrid({ │ │ │ │ │ - * layers: [world_utfgrid], │ │ │ │ │ - * handlerMode: 'move', │ │ │ │ │ - * callback: function(infoLookup) { │ │ │ │ │ - * // do something with returned data │ │ │ │ │ - * │ │ │ │ │ - * } │ │ │ │ │ - * }) │ │ │ │ │ - * (end code) │ │ │ │ │ - * │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.UTFGrid = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ - */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: Layers │ │ │ │ │ - * List of layers to consider. Must be Layer.UTFGrids │ │ │ │ │ - * `null` is the default indicating all UTFGrid Layers are queried. │ │ │ │ │ - * {Array} <OpenLayers.Layer.UTFGrid> │ │ │ │ │ - */ │ │ │ │ │ - layers: null, │ │ │ │ │ - │ │ │ │ │ - /* Property: defaultHandlerOptions │ │ │ │ │ - * The default opts passed to the handler constructors │ │ │ │ │ - */ │ │ │ │ │ - defaultHandlerOptions: { │ │ │ │ │ - 'delay': 300, │ │ │ │ │ - 'pixelTolerance': 4, │ │ │ │ │ - 'stopMove': false, │ │ │ │ │ - 'single': true, │ │ │ │ │ - 'double': false, │ │ │ │ │ - 'stopSingle': false, │ │ │ │ │ - 'stopDouble': false │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /* APIProperty: handlerMode │ │ │ │ │ - * Defaults to 'click'. Can be 'hover' or 'move'. │ │ │ │ │ - */ │ │ │ │ │ - handlerMode: 'click', │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: setHandler │ │ │ │ │ - * sets this.handlerMode and calls resetHandler() │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * hm - {String} Handler Mode string; 'click', 'hover' or 'move'. │ │ │ │ │ - */ │ │ │ │ │ - setHandler: function(hm) { │ │ │ │ │ - this.handlerMode = hm; │ │ │ │ │ - this.resetHandler(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: resetHandler │ │ │ │ │ - * Deactivates the old hanlder and creates a new │ │ │ │ │ - * <OpenLayers.Handler> based on the mode specified in │ │ │ │ │ - * this.handlerMode │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ - resetHandler: function() { │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.deactivate(); │ │ │ │ │ - this.handler.destroy(); │ │ │ │ │ - this.handler = null; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.handlerMode == 'hover') { │ │ │ │ │ - // Handle this event on hover │ │ │ │ │ - this.handler = new OpenLayers.Handler.Hover( │ │ │ │ │ - this, { │ │ │ │ │ - 'pause': this.handleEvent, │ │ │ │ │ - 'move': this.reset │ │ │ │ │ - }, │ │ │ │ │ - this.handlerOptions │ │ │ │ │ - ); │ │ │ │ │ - } else if (this.handlerMode == 'click') { │ │ │ │ │ - // Handle this event on click │ │ │ │ │ - this.handler = new OpenLayers.Handler.Click( │ │ │ │ │ - this, { │ │ │ │ │ - 'click': this.handleEvent │ │ │ │ │ - }, this.handlerOptions │ │ │ │ │ - ); │ │ │ │ │ - } else if (this.handlerMode == 'move') { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Hover( │ │ │ │ │ - this, │ │ │ │ │ - // Handle this event while hovering OR moving │ │ │ │ │ - { │ │ │ │ │ - 'pause': this.handleEvent, │ │ │ │ │ - 'move': this.handleEvent │ │ │ │ │ - }, │ │ │ │ │ - this.handlerOptions │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: <OpenLayers.Control.UTFGrid> │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.handlerOptions = options.handlerOptions || this.defaultHandlerOptions; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.resetHandler(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleEvent │ │ │ │ │ - * Internal method called when specified event is triggered. │ │ │ │ │ - * │ │ │ │ │ - * This method does several things: │ │ │ │ │ - * │ │ │ │ │ - * Gets the lonLat of the event. │ │ │ │ │ - * │ │ │ │ │ - * Loops through the appropriate hit grid layers and gathers the attributes. │ │ │ │ │ - * │ │ │ │ │ - * Passes the attributes to the callback │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ - */ │ │ │ │ │ - handleEvent: function(evt) { │ │ │ │ │ - if (evt == null) { │ │ │ │ │ - this.reset(); │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var lonLat = this.map.getLonLatFromPixel(evt.xy); │ │ │ │ │ - if (!lonLat) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var layers = this.findLayers(); │ │ │ │ │ - if (layers.length > 0) { │ │ │ │ │ - var infoLookup = {}; │ │ │ │ │ - var layer, idx; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - layer = layers[i]; │ │ │ │ │ - idx = OpenLayers.Util.indexOf(this.map.layers, layer); │ │ │ │ │ - infoLookup[idx] = layer.getFeatureInfo(lonLat); │ │ │ │ │ - } │ │ │ │ │ - this.callback(infoLookup, lonLat, evt.xy); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: callback │ │ │ │ │ - * Function to be called when a mouse event corresponds with a location that │ │ │ │ │ - * includes data in one of the configured UTFGrid layers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * infoLookup - {Object} Keys of this object are layer indexes and can be │ │ │ │ │ - * used to resolve a layer in the map.layers array. The structure of │ │ │ │ │ - * the property values depend on the data included in the underlying │ │ │ │ │ - * UTFGrid and may be any valid JSON type. │ │ │ │ │ - */ │ │ │ │ │ - callback: function(infoLookup) { │ │ │ │ │ - // to be provided in the constructor │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: reset │ │ │ │ │ - * Calls the callback with null. │ │ │ │ │ - */ │ │ │ │ │ - reset: function(evt) { │ │ │ │ │ - this.callback(null); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: findLayers │ │ │ │ │ - * Internal method to get the layers, independent of whether we are │ │ │ │ │ - * inspecting the map or using a client-provided array │ │ │ │ │ - * │ │ │ │ │ - * The default value of this.layers is null; this causes the │ │ │ │ │ - * findLayers method to return ALL UTFGrid layers encountered. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * None │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array} Layers to handle on each event │ │ │ │ │ - */ │ │ │ │ │ - findLayers: function() { │ │ │ │ │ - var candidates = this.layers || this.map.layers; │ │ │ │ │ - var layers = []; │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = candidates[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.UTFGrid) { │ │ │ │ │ - layers.push(layer); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return layers; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.UTFGrid" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/MousePosition.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.MousePosition │ │ │ │ │ - * The MousePosition control displays geographic coordinates of the mouse │ │ │ │ │ - * pointer, as it is moved about the map. │ │ │ │ │ - * │ │ │ │ │ - * You can use the <prefix>- or <suffix>-properties to provide more information │ │ │ │ │ - * about the displayed coordinates to the user: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * var mousePositionCtrl = new OpenLayers.Control.MousePosition({ │ │ │ │ │ - * prefix: '<a target="_blank" ' + │ │ │ │ │ - * 'href="http://spatialreference.org/ref/epsg/4326/">' + │ │ │ │ │ - * 'EPSG:4326</a> coordinates: ' │ │ │ │ │ - * } │ │ │ │ │ - * ); │ │ │ │ │ - * (end code) │ │ │ │ │ + * Class: OpenLayers.Control.GetFeature │ │ │ │ │ + * Gets vector features for locations underneath the mouse cursor. Can be │ │ │ │ │ + * configured to act on click, hover or dragged boxes. Uses an │ │ │ │ │ + * <OpenLayers.Protocol> that supports spatial filters to retrieve │ │ │ │ │ + * features from a server and fires events that notify applications of the │ │ │ │ │ + * selected features. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.MousePosition = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ - */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: element │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - element: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: prefix │ │ │ │ │ - * {String} A string to be prepended to the current pointers coordinates │ │ │ │ │ - * when it is rendered. Defaults to the empty string ''. │ │ │ │ │ - */ │ │ │ │ │ - prefix: '', │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: separator │ │ │ │ │ - * {String} A string to be used to seperate the two coordinates from each │ │ │ │ │ - * other. Defaults to the string ', ', which will result in a │ │ │ │ │ - * rendered coordinate of e.g. '42.12, 21.22'. │ │ │ │ │ - */ │ │ │ │ │ - separator: ', ', │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: suffix │ │ │ │ │ - * {String} A string to be appended to the current pointers coordinates │ │ │ │ │ - * when it is rendered. Defaults to the empty string ''. │ │ │ │ │ - */ │ │ │ │ │ - suffix: '', │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: numDigits │ │ │ │ │ - * {Integer} The number of digits each coordinate shall have when being │ │ │ │ │ - * rendered, Defaults to 5. │ │ │ │ │ - */ │ │ │ │ │ - numDigits: 5, │ │ │ │ │ +OpenLayers.Control.GetFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: granularity │ │ │ │ │ - * {Integer} │ │ │ │ │ + * APIProperty: protocol │ │ │ │ │ + * {<OpenLayers.Protocol>} Required. The protocol used for fetching │ │ │ │ │ + * features. │ │ │ │ │ */ │ │ │ │ │ - granularity: 10, │ │ │ │ │ + protocol: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: emptyString │ │ │ │ │ - * {String} Set this to some value to set when the mouse is outside the │ │ │ │ │ - * map. │ │ │ │ │ + * APIProperty: multipleKey │ │ │ │ │ + * {String} An event modifier ('altKey' or 'shiftKey') that temporarily sets │ │ │ │ │ + * the <multiple> property to true. Default is null. │ │ │ │ │ */ │ │ │ │ │ - emptyString: null, │ │ │ │ │ + multipleKey: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: lastXy │ │ │ │ │ - * {<OpenLayers.Pixel>} │ │ │ │ │ + * APIProperty: toggleKey │ │ │ │ │ + * {String} An event modifier ('altKey' or 'shiftKey') that temporarily sets │ │ │ │ │ + * the <toggle> property to true. Default is null. │ │ │ │ │ */ │ │ │ │ │ - lastXy: null, │ │ │ │ │ + toggleKey: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: displayProjection │ │ │ │ │ - * {<OpenLayers.Projection>} The projection in which the mouse position is │ │ │ │ │ - * displayed. │ │ │ │ │ + * Property: modifiers │ │ │ │ │ + * {Object} The event modifiers to use, according to the current event │ │ │ │ │ + * being handled by this control's handlers │ │ │ │ │ */ │ │ │ │ │ - displayProjection: null, │ │ │ │ │ + modifiers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.MousePosition │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Options for control. │ │ │ │ │ + * APIProperty: multiple │ │ │ │ │ + * {Boolean} Allow selection of multiple geometries. Default is false. │ │ │ │ │ */ │ │ │ │ │ + multiple: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ + * APIProperty: click │ │ │ │ │ + * {Boolean} Use a click handler for selecting/unselecting features. If │ │ │ │ │ + * both <click> and <box> are set to true, the click handler takes │ │ │ │ │ + * precedence over the box handler if a box with zero extent was │ │ │ │ │ + * selected. Default is true. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + click: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ + * APIProperty: single │ │ │ │ │ + * {Boolean} Tells whether select by click should select a single │ │ │ │ │ + * feature. If set to false, all matching features are selected. │ │ │ │ │ + * If set to true, only the best matching feature is selected. │ │ │ │ │ + * This option has an effect only of the <click> option is set │ │ │ │ │ + * to true. Default is true. │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.map.events.register('mousemove', this, this.redraw); │ │ │ │ │ - this.map.events.register('mouseout', this, this.reset); │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + single: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ + * APIProperty: clickout │ │ │ │ │ + * {Boolean} Unselect features when clicking outside any feature. │ │ │ │ │ + * Applies only if <click> is true. Default is true. │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.map.events.unregister('mousemove', this, this.redraw); │ │ │ │ │ - this.map.events.unregister('mouseout', this, this.reset); │ │ │ │ │ - this.element.innerHTML = ""; │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + clickout: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * APIProperty: toggle │ │ │ │ │ + * {Boolean} Unselect a selected feature on click. Applies only if │ │ │ │ │ + * <click> is true. Default is false. │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - if (!this.element) { │ │ │ │ │ - this.div.left = ""; │ │ │ │ │ - this.div.top = ""; │ │ │ │ │ - this.element = this.div; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ + toggle: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: redraw │ │ │ │ │ + * APIProperty: clickTolerance │ │ │ │ │ + * {Integer} Tolerance for the filter query in pixels. This has the │ │ │ │ │ + * same effect as the tolerance parameter on WMS GetFeatureInfo │ │ │ │ │ + * requests. Will be ignored for box selections. Applies only if │ │ │ │ │ + * <click> or <hover> is true. Default is 5. Note that this not │ │ │ │ │ + * only affects requests on click, but also on hover. │ │ │ │ │ */ │ │ │ │ │ - redraw: function(evt) { │ │ │ │ │ - │ │ │ │ │ - var lonLat; │ │ │ │ │ - │ │ │ │ │ - if (evt == null) { │ │ │ │ │ - this.reset(); │ │ │ │ │ - return; │ │ │ │ │ - } else { │ │ │ │ │ - if (this.lastXy == null || │ │ │ │ │ - Math.abs(evt.xy.x - this.lastXy.x) > this.granularity || │ │ │ │ │ - Math.abs(evt.xy.y - this.lastXy.y) > this.granularity) { │ │ │ │ │ - this.lastXy = evt.xy; │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - lonLat = this.map.getLonLatFromPixel(evt.xy); │ │ │ │ │ - if (!lonLat) { │ │ │ │ │ - // map has not yet been properly initialized │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (this.displayProjection) { │ │ │ │ │ - lonLat.transform(this.map.getProjectionObject(), │ │ │ │ │ - this.displayProjection); │ │ │ │ │ - } │ │ │ │ │ - this.lastXy = evt.xy; │ │ │ │ │ - │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var newHtml = this.formatOutput(lonLat); │ │ │ │ │ - │ │ │ │ │ - if (newHtml != this.element.innerHTML) { │ │ │ │ │ - this.element.innerHTML = newHtml; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + clickTolerance: 5, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: reset │ │ │ │ │ + * APIProperty: hover │ │ │ │ │ + * {Boolean} Send feature requests on mouse moves. Default is false. │ │ │ │ │ */ │ │ │ │ │ - reset: function(evt) { │ │ │ │ │ - if (this.emptyString != null) { │ │ │ │ │ - this.element.innerHTML = this.emptyString; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + hover: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: formatOutput │ │ │ │ │ - * Override to provide custom display output │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * lonLat - {<OpenLayers.LonLat>} Location to display │ │ │ │ │ + * APIProperty: box │ │ │ │ │ + * {Boolean} Allow feature selection by drawing a box. If set to │ │ │ │ │ + * true set <click> to false to disable the click handler and │ │ │ │ │ + * rely on the box handler only, even for "zero extent" boxes. │ │ │ │ │ + * See the description of the <click> option for additional │ │ │ │ │ + * information. Default is false. │ │ │ │ │ */ │ │ │ │ │ - formatOutput: function(lonLat) { │ │ │ │ │ - var digits = parseInt(this.numDigits); │ │ │ │ │ - var newHtml = │ │ │ │ │ - this.prefix + │ │ │ │ │ - lonLat.lon.toFixed(digits) + │ │ │ │ │ - this.separator + │ │ │ │ │ - lonLat.lat.toFixed(digits) + │ │ │ │ │ - this.suffix; │ │ │ │ │ - return newHtml; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.MousePosition" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Zoom.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Zoom │ │ │ │ │ - * The Zoom control is a pair of +/- links for zooming in and out. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Zoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + box: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomInText │ │ │ │ │ - * {String} │ │ │ │ │ - * Text for zoom-in link. Default is "+". │ │ │ │ │ + * APIProperty: maxFeatures │ │ │ │ │ + * {Integer} Maximum number of features to return from a query in single mode │ │ │ │ │ + * if supported by the <protocol>. This set of features is then used to │ │ │ │ │ + * determine the best match client-side. Default is 10. │ │ │ │ │ */ │ │ │ │ │ - zoomInText: "+", │ │ │ │ │ + maxFeatures: 10, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomInId │ │ │ │ │ - * {String} │ │ │ │ │ - * Instead of having the control create a zoom in link, you can provide │ │ │ │ │ - * the identifier for an anchor element already added to the document. │ │ │ │ │ - * By default, an element with id "olZoomInLink" will be searched for │ │ │ │ │ - * and used if it exists. │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Object} Hash of {<OpenLayers.Feature.Vector>}, keyed by fid, holding │ │ │ │ │ + * the currently selected features │ │ │ │ │ */ │ │ │ │ │ - zoomInId: "olZoomInLink", │ │ │ │ │ + features: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomOutText │ │ │ │ │ - * {String} │ │ │ │ │ - * Text for zoom-out link. Default is "\u2212". │ │ │ │ │ + * Proeprty: hoverFeature │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} The feature currently selected by the │ │ │ │ │ + * hover handler │ │ │ │ │ */ │ │ │ │ │ - zoomOutText: "\u2212", │ │ │ │ │ + hoverFeature: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: zoomOutId │ │ │ │ │ - * {String} │ │ │ │ │ - * Instead of having the control create a zoom out link, you can provide │ │ │ │ │ - * the identifier for an anchor element already added to the document. │ │ │ │ │ - * By default, an element with id "olZoomOutLink" will be searched for │ │ │ │ │ - * and used if it exists. │ │ │ │ │ + * APIProperty: handlerOptions │ │ │ │ │ + * {Object} Additional options for the handlers used by this control. This │ │ │ │ │ + * is a hash with the keys "click", "box" and "hover". │ │ │ │ │ */ │ │ │ │ │ - zoomOutId: "olZoomOutLink", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A reference to the DOMElement containing the zoom links. │ │ │ │ │ + * Property: handlers │ │ │ │ │ + * {Object} Object with references to multiple <OpenLayers.Handler> │ │ │ │ │ + * instances. │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - var div = OpenLayers.Control.prototype.draw.apply(this), │ │ │ │ │ - links = this.getOrCreateLinks(div), │ │ │ │ │ - zoomIn = links.zoomIn, │ │ │ │ │ - zoomOut = links.zoomOut, │ │ │ │ │ - eventsInstance = this.map.events; │ │ │ │ │ - │ │ │ │ │ - if (zoomOut.parentNode !== div) { │ │ │ │ │ - eventsInstance = this.events; │ │ │ │ │ - eventsInstance.attachToElement(zoomOut.parentNode); │ │ │ │ │ - } │ │ │ │ │ - eventsInstance.register("buttonclick", this, this.onZoomClick); │ │ │ │ │ - │ │ │ │ │ - this.zoomInLink = zoomIn; │ │ │ │ │ - this.zoomOutLink = zoomOut; │ │ │ │ │ - return div; │ │ │ │ │ - }, │ │ │ │ │ + handlers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getOrCreateLinks │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * el - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ - * Return: │ │ │ │ │ - * {Object} Object with zoomIn and zoomOut properties referencing links. │ │ │ │ │ + * Property: hoverResponse │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} The response object associated with │ │ │ │ │ + * the currently running hover request (if any). │ │ │ │ │ */ │ │ │ │ │ - getOrCreateLinks: function(el) { │ │ │ │ │ - var zoomIn = document.getElementById(this.zoomInId), │ │ │ │ │ - zoomOut = document.getElementById(this.zoomOutId); │ │ │ │ │ - if (!zoomIn) { │ │ │ │ │ - zoomIn = document.createElement("a"); │ │ │ │ │ - zoomIn.href = "#zoomIn"; │ │ │ │ │ - zoomIn.appendChild(document.createTextNode(this.zoomInText)); │ │ │ │ │ - zoomIn.className = "olControlZoomIn"; │ │ │ │ │ - el.appendChild(zoomIn); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Element.addClass(zoomIn, "olButton"); │ │ │ │ │ - if (!zoomOut) { │ │ │ │ │ - zoomOut = document.createElement("a"); │ │ │ │ │ - zoomOut.href = "#zoomOut"; │ │ │ │ │ - zoomOut.appendChild(document.createTextNode(this.zoomOutText)); │ │ │ │ │ - zoomOut.className = "olControlZoomOut"; │ │ │ │ │ - el.appendChild(zoomOut); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Element.addClass(zoomOut, "olButton"); │ │ │ │ │ - return { │ │ │ │ │ - zoomIn: zoomIn, │ │ │ │ │ - zoomOut: zoomOut │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ + hoverResponse: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onZoomClick │ │ │ │ │ - * Called when zoomin/out link is clicked. │ │ │ │ │ - */ │ │ │ │ │ - onZoomClick: function(evt) { │ │ │ │ │ - var button = evt.buttonElement; │ │ │ │ │ - if (button === this.zoomInLink) { │ │ │ │ │ - this.map.zoomIn(); │ │ │ │ │ - } else if (button === this.zoomOutLink) { │ │ │ │ │ - this.map.zoomOut(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * Clean up. │ │ │ │ │ + * Property: filterType │ │ │ │ │ + * {<String>} The type of filter to use when sending off a request. │ │ │ │ │ + * Possible values: │ │ │ │ │ + * OpenLayers.Filter.Spatial.<BBOX|INTERSECTS|WITHIN|CONTAINS> │ │ │ │ │ + * Defaults to: OpenLayers.Filter.Spatial.BBOX │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onZoomClick); │ │ │ │ │ - } │ │ │ │ │ - delete this.zoomInLink; │ │ │ │ │ - delete this.zoomOutLink; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Zoom" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/SelectFeature.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Handler/Feature.js │ │ │ │ │ - * @requires OpenLayers/Layer/Vector/RootContainer.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.SelectFeature │ │ │ │ │ - * The SelectFeature control selects vector features from a given layer on │ │ │ │ │ - * click or hover. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.SelectFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + filterType: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIProperty: events │ │ │ │ │ * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ * control specific events. │ │ │ │ │ * │ │ │ │ │ * Register a listener for a particular event with the following syntax: │ │ │ │ │ * (code) │ │ │ │ │ * control.events.register(type, obj, listener); │ │ │ │ │ * (end) │ │ │ │ │ * │ │ │ │ │ * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * beforefeaturehighlighted - Triggered before a feature is highlighted │ │ │ │ │ - * featurehighlighted - Triggered when a feature is highlighted │ │ │ │ │ - * featureunhighlighted - Triggered when a feature is unhighlighted │ │ │ │ │ - * boxselectionstart - Triggered before box selection starts │ │ │ │ │ - * boxselectionend - Triggered after box selection ends │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: multipleKey │ │ │ │ │ - * {String} An event modifier ('altKey' or 'shiftKey') that temporarily sets │ │ │ │ │ - * the <multiple> property to true. Default is null. │ │ │ │ │ - */ │ │ │ │ │ - multipleKey: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: toggleKey │ │ │ │ │ - * {String} An event modifier ('altKey' or 'shiftKey') that temporarily sets │ │ │ │ │ - * the <toggle> property to true. Default is null. │ │ │ │ │ - */ │ │ │ │ │ - toggleKey: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: multiple │ │ │ │ │ - * {Boolean} Allow selection of multiple geometries. Default is false. │ │ │ │ │ - */ │ │ │ │ │ - multiple: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: clickout │ │ │ │ │ - * {Boolean} Unselect features when clicking outside any feature. │ │ │ │ │ - * Default is true. │ │ │ │ │ - */ │ │ │ │ │ - clickout: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: toggle │ │ │ │ │ - * {Boolean} Unselect a selected feature on click. Default is false. Only │ │ │ │ │ - * has meaning if hover is false. │ │ │ │ │ - */ │ │ │ │ │ - toggle: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: hover │ │ │ │ │ - * {Boolean} Select on mouse over and deselect on mouse out. If true, this │ │ │ │ │ - * ignores clicks and only listens to mouse moves. │ │ │ │ │ - */ │ │ │ │ │ - hover: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: highlightOnly │ │ │ │ │ - * {Boolean} If true do not actually select features (that is place them in │ │ │ │ │ - * the layer's selected features array), just highlight them. This property │ │ │ │ │ - * has no effect if hover is false. Defaults to false. │ │ │ │ │ - */ │ │ │ │ │ - highlightOnly: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: box │ │ │ │ │ - * {Boolean} Allow feature selection by drawing a box. │ │ │ │ │ - */ │ │ │ │ │ - box: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: onBeforeSelect │ │ │ │ │ - * {Function} Optional function to be called before a feature is selected. │ │ │ │ │ - * The function should expect to be called with a feature. │ │ │ │ │ - */ │ │ │ │ │ - onBeforeSelect: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: onSelect │ │ │ │ │ - * {Function} Optional function to be called when a feature is selected. │ │ │ │ │ - * The function should expect to be called with a feature. │ │ │ │ │ - */ │ │ │ │ │ - onSelect: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: onUnselect │ │ │ │ │ - * {Function} Optional function to be called when a feature is unselected. │ │ │ │ │ - * The function should expect to be called with a feature. │ │ │ │ │ - */ │ │ │ │ │ - onUnselect: function() {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: scope │ │ │ │ │ - * {Object} The scope to use with the onBeforeSelect, onSelect, onUnselect │ │ │ │ │ - * callbacks. If null the scope will be this control. │ │ │ │ │ - */ │ │ │ │ │ - scope: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geometryTypes │ │ │ │ │ - * {Array(String)} To restrict selecting to a limited set of geometry types, │ │ │ │ │ - * send a list of strings corresponding to the geometry class names. │ │ │ │ │ - */ │ │ │ │ │ - geometryTypes: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layer │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} The vector layer with a common renderer │ │ │ │ │ - * root for all layers this control is configured with (if an array of │ │ │ │ │ - * layers was passed to the constructor), or the vector layer the control │ │ │ │ │ - * was configured with (if a single layer was passed to the constructor). │ │ │ │ │ - */ │ │ │ │ │ - layer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layers │ │ │ │ │ - * {Array(<OpenLayers.Layer.Vector>)} The layers this control will work on, │ │ │ │ │ - * or null if the control was configured with a single layer │ │ │ │ │ - */ │ │ │ │ │ - layers: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: callbacks │ │ │ │ │ - * {Object} The functions that are sent to the handlers.feature for callback │ │ │ │ │ - */ │ │ │ │ │ - callbacks: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: selectStyle │ │ │ │ │ - * {Object} Hash of styles │ │ │ │ │ - */ │ │ │ │ │ - selectStyle: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: renderIntent │ │ │ │ │ - * {String} key used to retrieve the select style from the layer's │ │ │ │ │ - * style map. │ │ │ │ │ - */ │ │ │ │ │ - renderIntent: "select", │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: handlers │ │ │ │ │ - * {Object} Object with references to multiple <OpenLayers.Handler> │ │ │ │ │ - * instances. │ │ │ │ │ + * beforefeatureselected - Triggered when <click> is true before a │ │ │ │ │ + * feature is selected. The event object has a feature property with │ │ │ │ │ + * the feature about to select │ │ │ │ │ + * featureselected - Triggered when <click> is true and a feature is │ │ │ │ │ + * selected. The event object has a feature property with the │ │ │ │ │ + * selected feature │ │ │ │ │ + * beforefeaturesselected - Triggered when <click> is true before a │ │ │ │ │ + * set of features is selected. The event object is an array of │ │ │ │ │ + * feature properties with the features about to be selected. │ │ │ │ │ + * Return false after receiving this event to discontinue processing │ │ │ │ │ + * of all featureselected events and the featuresselected event. │ │ │ │ │ + * featuresselected - Triggered when <click> is true and a set of │ │ │ │ │ + * features is selected. The event object is an array of feature │ │ │ │ │ + * properties of the selected features │ │ │ │ │ + * featureunselected - Triggered when <click> is true and a feature is │ │ │ │ │ + * unselected. The event object has a feature property with the │ │ │ │ │ + * unselected feature │ │ │ │ │ + * clickout - Triggered when when <click> is true and no feature was │ │ │ │ │ + * selected. │ │ │ │ │ + * hoverfeature - Triggered when <hover> is true and the mouse has │ │ │ │ │ + * stopped over a feature │ │ │ │ │ + * outfeature - Triggered when <hover> is true and the mouse moves │ │ │ │ │ + * moved away from a hover-selected feature │ │ │ │ │ */ │ │ │ │ │ - handlers: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.SelectFeature │ │ │ │ │ - * Create a new control for selecting features. │ │ │ │ │ + * Constructor: OpenLayers.Control.GetFeature │ │ │ │ │ + * Create a new control for fetching remote features. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layers - {<OpenLayers.Layer.Vector>}, or an array of vector layers. The │ │ │ │ │ - * layer(s) this control will select features from. │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * options - {Object} A configuration object which at least has to contain │ │ │ │ │ + * a <protocol> property (if not, it has to be set before a request is │ │ │ │ │ + * made) │ │ │ │ │ */ │ │ │ │ │ - initialize: function(layers, options) { │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ + │ │ │ │ │ OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ │ │ │ │ │ - if (this.scope === null) { │ │ │ │ │ - this.scope = this; │ │ │ │ │ - } │ │ │ │ │ - this.initLayer(layers); │ │ │ │ │ - var callbacks = { │ │ │ │ │ - click: this.clickFeature, │ │ │ │ │ - clickout: this.clickoutFeature │ │ │ │ │ - }; │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - callbacks.over = this.overFeature; │ │ │ │ │ - callbacks.out = this.outFeature; │ │ │ │ │ - } │ │ │ │ │ + this.features = {}; │ │ │ │ │ │ │ │ │ │ - this.callbacks = OpenLayers.Util.extend(callbacks, this.callbacks); │ │ │ │ │ - this.handlers = { │ │ │ │ │ - feature: new OpenLayers.Handler.Feature( │ │ │ │ │ - this, this.layer, this.callbacks, { │ │ │ │ │ - geometryTypes: this.geometryTypes │ │ │ │ │ - } │ │ │ │ │ - ) │ │ │ │ │ - }; │ │ │ │ │ + this.handlers = {}; │ │ │ │ │ + │ │ │ │ │ + if (this.click) { │ │ │ │ │ + this.handlers.click = new OpenLayers.Handler.Click(this, { │ │ │ │ │ + click: this.selectClick │ │ │ │ │ + }, this.handlerOptions.click || {}); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ if (this.box) { │ │ │ │ │ this.handlers.box = new OpenLayers.Handler.Box( │ │ │ │ │ this, { │ │ │ │ │ done: this.selectBox │ │ │ │ │ - }, { │ │ │ │ │ + }, │ │ │ │ │ + OpenLayers.Util.extend(this.handlerOptions.box, { │ │ │ │ │ boxDivClassName: "olHandlerBoxSelectFeature" │ │ │ │ │ - } │ │ │ │ │ + }) │ │ │ │ │ ); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: initLayer │ │ │ │ │ - * Assign the layer property. If layers is an array, we need to use │ │ │ │ │ - * a RootContainer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layers - {<OpenLayers.Layer.Vector>}, or an array of vector layers. │ │ │ │ │ - */ │ │ │ │ │ - initLayer: function(layers) { │ │ │ │ │ - if (OpenLayers.Util.isArray(layers)) { │ │ │ │ │ - this.layers = layers; │ │ │ │ │ - this.layer = new OpenLayers.Layer.Vector.RootContainer( │ │ │ │ │ - this.id + "_container", { │ │ │ │ │ - layers: layers │ │ │ │ │ - } │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + this.handlers.hover = new OpenLayers.Handler.Hover( │ │ │ │ │ + this, { │ │ │ │ │ + 'move': this.cancelHover, │ │ │ │ │ + 'pause': this.selectHover │ │ │ │ │ + }, │ │ │ │ │ + OpenLayers.Util.extend(this.handlerOptions.hover, { │ │ │ │ │ + 'delay': 250, │ │ │ │ │ + 'pixelTolerance': 2 │ │ │ │ │ + }) │ │ │ │ │ ); │ │ │ │ │ - } else { │ │ │ │ │ - this.layer = layers; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.active && this.layers) { │ │ │ │ │ - this.map.removeLayer(this.layer); │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - if (this.layers) { │ │ │ │ │ - this.layer.destroy(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Method: activate │ │ │ │ │ * Activates the control. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ * {Boolean} The control was effectively activated. │ │ │ │ │ */ │ │ │ │ │ activate: function() { │ │ │ │ │ if (!this.active) { │ │ │ │ │ - if (this.layers) { │ │ │ │ │ - this.map.addLayer(this.layer); │ │ │ │ │ - } │ │ │ │ │ - this.handlers.feature.activate(); │ │ │ │ │ - if (this.box && this.handlers.box) { │ │ │ │ │ - this.handlers.box.activate(); │ │ │ │ │ + for (var i in this.handlers) { │ │ │ │ │ + this.handlers[i].activate(); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ return OpenLayers.Control.prototype.activate.apply( │ │ │ │ │ this, arguments │ │ │ │ │ ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ @@ -84290,5995 +82409,3591 @@ │ │ │ │ │ * Deactivates the control. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ * {Boolean} The control was effectively deactivated. │ │ │ │ │ */ │ │ │ │ │ deactivate: function() { │ │ │ │ │ if (this.active) { │ │ │ │ │ - this.handlers.feature.deactivate(); │ │ │ │ │ - if (this.handlers.box) { │ │ │ │ │ - this.handlers.box.deactivate(); │ │ │ │ │ - } │ │ │ │ │ - if (this.layers) { │ │ │ │ │ - this.map.removeLayer(this.layer); │ │ │ │ │ + for (var i in this.handlers) { │ │ │ │ │ + this.handlers[i].deactivate(); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ return OpenLayers.Control.prototype.deactivate.apply( │ │ │ │ │ this, arguments │ │ │ │ │ ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: unselectAll │ │ │ │ │ - * Unselect all selected features. To unselect all except for a single │ │ │ │ │ - * feature, set the options.except property to the feature. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional configuration object. │ │ │ │ │ - */ │ │ │ │ │ - unselectAll: function(options) { │ │ │ │ │ - // we'll want an option to supress notification here │ │ │ │ │ - var layers = this.layers || [this.layer], │ │ │ │ │ - layer, feature, l, numExcept; │ │ │ │ │ - for (l = 0; l < layers.length; ++l) { │ │ │ │ │ - layer = layers[l]; │ │ │ │ │ - numExcept = 0; │ │ │ │ │ - //layer.selectedFeatures is null when layer is destroyed and │ │ │ │ │ - //one of it's preremovelayer listener calls setLayer │ │ │ │ │ - //with another layer on this control │ │ │ │ │ - if (layer.selectedFeatures != null) { │ │ │ │ │ - while (layer.selectedFeatures.length > numExcept) { │ │ │ │ │ - feature = layer.selectedFeatures[numExcept]; │ │ │ │ │ - if (!options || options.except != feature) { │ │ │ │ │ - this.unselect(feature); │ │ │ │ │ - } else { │ │ │ │ │ - ++numExcept; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clickFeature │ │ │ │ │ - * Called on click in a feature │ │ │ │ │ - * Only responds if this.hover is false. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - clickFeature: function(feature) { │ │ │ │ │ - if (!this.hover) { │ │ │ │ │ - var selected = (OpenLayers.Util.indexOf( │ │ │ │ │ - feature.layer.selectedFeatures, feature) > -1); │ │ │ │ │ - if (selected) { │ │ │ │ │ - if (this.toggleSelect()) { │ │ │ │ │ - this.unselect(feature); │ │ │ │ │ - } else if (!this.multipleSelect()) { │ │ │ │ │ - this.unselectAll({ │ │ │ │ │ - except: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - if (!this.multipleSelect()) { │ │ │ │ │ - this.unselectAll({ │ │ │ │ │ - except: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - this.select(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: multipleSelect │ │ │ │ │ - * Allow for multiple selected features based on <multiple> property and │ │ │ │ │ - * <multipleKey> event modifier. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Allow for multiple selected features. │ │ │ │ │ - */ │ │ │ │ │ - multipleSelect: function() { │ │ │ │ │ - return this.multiple || (this.handlers.feature.evt && │ │ │ │ │ - this.handlers.feature.evt[this.multipleKey]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: toggleSelect │ │ │ │ │ - * Event should toggle the selected state of a feature based on <toggle> │ │ │ │ │ - * property and <toggleKey> event modifier. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Toggle the selected state of a feature. │ │ │ │ │ - */ │ │ │ │ │ - toggleSelect: function() { │ │ │ │ │ - return this.toggle || (this.handlers.feature.evt && │ │ │ │ │ - this.handlers.feature.evt[this.toggleKey]); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clickoutFeature │ │ │ │ │ - * Called on click outside a previously clicked (selected) feature. │ │ │ │ │ - * Only responds if this.hover is false. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Vector.Feature>} │ │ │ │ │ - */ │ │ │ │ │ - clickoutFeature: function(feature) { │ │ │ │ │ - if (!this.hover && this.clickout) { │ │ │ │ │ - this.unselectAll(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: overFeature │ │ │ │ │ - * Called on over a feature. │ │ │ │ │ - * Only responds if this.hover is true. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - overFeature: function(feature) { │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - if (this.highlightOnly) { │ │ │ │ │ - this.highlight(feature); │ │ │ │ │ - } else if (OpenLayers.Util.indexOf( │ │ │ │ │ - layer.selectedFeatures, feature) == -1) { │ │ │ │ │ - this.select(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: outFeature │ │ │ │ │ - * Called on out of a selected feature. │ │ │ │ │ - * Only responds if this.hover is true. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - outFeature: function(feature) { │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - if (this.highlightOnly) { │ │ │ │ │ - // we do nothing if we're not the last highlighter of the │ │ │ │ │ - // feature │ │ │ │ │ - if (feature._lastHighlighter == this.id) { │ │ │ │ │ - // if another select control had highlighted the feature before │ │ │ │ │ - // we did it ourself then we use that control to highlight the │ │ │ │ │ - // feature as it was before we highlighted it, else we just │ │ │ │ │ - // unhighlight it │ │ │ │ │ - if (feature._prevHighlighter && │ │ │ │ │ - feature._prevHighlighter != this.id) { │ │ │ │ │ - delete feature._lastHighlighter; │ │ │ │ │ - var control = this.map.getControl( │ │ │ │ │ - feature._prevHighlighter); │ │ │ │ │ - if (control) { │ │ │ │ │ - control.highlight(feature); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.unhighlight(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.unselect(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: highlight │ │ │ │ │ - * Redraw feature with the select style. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - highlight: function(feature) { │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - var cont = this.events.triggerEvent("beforefeaturehighlighted", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - feature._prevHighlighter = feature._lastHighlighter; │ │ │ │ │ - feature._lastHighlighter = this.id; │ │ │ │ │ - var style = this.selectStyle || this.renderIntent; │ │ │ │ │ - layer.drawFeature(feature, style); │ │ │ │ │ - this.events.triggerEvent("featurehighlighted", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: unhighlight │ │ │ │ │ - * Redraw feature with the "default" style │ │ │ │ │ + * Method: selectClick │ │ │ │ │ + * Called on click │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - unhighlight: function(feature) { │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - // three cases: │ │ │ │ │ - // 1. there's no other highlighter, in that case _prev is undefined, │ │ │ │ │ - // and we just need to undef _last │ │ │ │ │ - // 2. another control highlighted the feature after we did it, in │ │ │ │ │ - // that case _last references this other control, and we just │ │ │ │ │ - // need to undef _prev │ │ │ │ │ - // 3. another control highlighted the feature before we did it, in │ │ │ │ │ - // that case _prev references this other control, and we need to │ │ │ │ │ - // set _last to _prev and undef _prev │ │ │ │ │ - if (feature._prevHighlighter == undefined) { │ │ │ │ │ - delete feature._lastHighlighter; │ │ │ │ │ - } else if (feature._prevHighlighter == this.id) { │ │ │ │ │ - delete feature._prevHighlighter; │ │ │ │ │ - } else { │ │ │ │ │ - feature._lastHighlighter = feature._prevHighlighter; │ │ │ │ │ - delete feature._prevHighlighter; │ │ │ │ │ - } │ │ │ │ │ - layer.drawFeature(feature, feature.style || feature.layer.style || │ │ │ │ │ - "default"); │ │ │ │ │ - this.events.triggerEvent("featureunhighlighted", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: select │ │ │ │ │ - * Add feature to the layer's selectedFeature array, render the feature as │ │ │ │ │ - * selected, and call the onSelect function. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - select: function(feature) { │ │ │ │ │ - var cont = this.onBeforeSelect.call(this.scope, feature); │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - cont = layer.events.triggerEvent("beforefeatureselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - layer.selectedFeatures.push(feature); │ │ │ │ │ - this.highlight(feature); │ │ │ │ │ - // if the feature handler isn't involved in the feature │ │ │ │ │ - // selection (because the box handler is used or the │ │ │ │ │ - // feature is selected programatically) we fake the │ │ │ │ │ - // feature handler to allow unselecting on click │ │ │ │ │ - if (!this.handlers.feature.lastFeature) { │ │ │ │ │ - this.handlers.feature.lastFeature = layer.selectedFeatures[0]; │ │ │ │ │ - } │ │ │ │ │ - layer.events.triggerEvent("featureselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - this.onSelect.call(this.scope, feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + selectClick: function(evt) { │ │ │ │ │ + var bounds = this.pixelToBounds(evt.xy); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: unselect │ │ │ │ │ - * Remove feature from the layer's selectedFeature array, render the feature as │ │ │ │ │ - * normal, and call the onUnselect function. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - unselect: function(feature) { │ │ │ │ │ - var layer = feature.layer; │ │ │ │ │ - // Store feature style for restoration later │ │ │ │ │ - this.unhighlight(feature); │ │ │ │ │ - OpenLayers.Util.removeItem(layer.selectedFeatures, feature); │ │ │ │ │ - layer.events.triggerEvent("featureunselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ + this.setModifiers(evt); │ │ │ │ │ + this.request(bounds, { │ │ │ │ │ + single: this.single │ │ │ │ │ }); │ │ │ │ │ - this.onUnselect.call(this.scope, feature); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Method: selectBox │ │ │ │ │ - * Callback from the handlers.box set up when <box> selection is true │ │ │ │ │ - * on. │ │ │ │ │ + * Callback from the handlers.box set up when <box> selection is on │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * position - {<OpenLayers.Bounds> || <OpenLayers.Pixel> } │ │ │ │ │ + * position - {<OpenLayers.Bounds>|Object} An OpenLayers.Bounds or │ │ │ │ │ + * an object with a 'left', 'bottom', 'right' and 'top' properties. │ │ │ │ │ */ │ │ │ │ │ selectBox: function(position) { │ │ │ │ │ + var bounds; │ │ │ │ │ if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ x: position.left, │ │ │ │ │ y: position.bottom │ │ │ │ │ }); │ │ │ │ │ var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ x: position.right, │ │ │ │ │ y: position.top │ │ │ │ │ }); │ │ │ │ │ - var bounds = new OpenLayers.Bounds( │ │ │ │ │ + bounds = new OpenLayers.Bounds( │ │ │ │ │ minXY.lon, minXY.lat, maxXY.lon, maxXY.lat │ │ │ │ │ ); │ │ │ │ │ │ │ │ │ │ - // if multiple is false, first deselect currently selected features │ │ │ │ │ - if (!this.multipleSelect()) { │ │ │ │ │ - this.unselectAll(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // because we're using a box, we consider we want multiple selection │ │ │ │ │ - var prevMultiple = this.multiple; │ │ │ │ │ - this.multiple = true; │ │ │ │ │ - var layers = this.layers || [this.layer]; │ │ │ │ │ - this.events.triggerEvent("boxselectionstart", { │ │ │ │ │ - layers: layers │ │ │ │ │ - }); │ │ │ │ │ - var layer; │ │ │ │ │ - for (var l = 0; l < layers.length; ++l) { │ │ │ │ │ - layer = layers[l]; │ │ │ │ │ - for (var i = 0, len = layer.features.length; i < len; ++i) { │ │ │ │ │ - var feature = layer.features[i]; │ │ │ │ │ - // check if the feature is displayed │ │ │ │ │ - if (!feature.getVisibility()) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (this.geometryTypes == null || OpenLayers.Util.indexOf( │ │ │ │ │ - this.geometryTypes, feature.geometry.CLASS_NAME) > -1) { │ │ │ │ │ - if (bounds.toGeometry().intersects(feature.geometry)) { │ │ │ │ │ - if (OpenLayers.Util.indexOf(layer.selectedFeatures, feature) == -1) { │ │ │ │ │ - this.select(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + } else { │ │ │ │ │ + if (this.click) { │ │ │ │ │ + // box without extent - let the click handler take care of it │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ - this.multiple = prevMultiple; │ │ │ │ │ - this.events.triggerEvent("boxselectionend", { │ │ │ │ │ - layers: layers │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the control. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.handlers.feature.setMap(map); │ │ │ │ │ - if (this.box) { │ │ │ │ │ - this.handlers.box.setMap(map); │ │ │ │ │ + bounds = this.pixelToBounds(position); │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.setModifiers(this.handlers.box.dragHandler.evt); │ │ │ │ │ + this.request(bounds); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setLayer │ │ │ │ │ - * Attach a new layer to the control, overriding any existing layers. │ │ │ │ │ + * Method: selectHover │ │ │ │ │ + * Callback from the handlers.hover set up when <hover> selection is on │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layers - Array of {<OpenLayers.Layer.Vector>} or a single │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} │ │ │ │ │ - */ │ │ │ │ │ - setLayer: function(layers) { │ │ │ │ │ - var isActive = this.active; │ │ │ │ │ - this.unselectAll(); │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - if (this.layers) { │ │ │ │ │ - this.layer.destroy(); │ │ │ │ │ - this.layers = null; │ │ │ │ │ - } │ │ │ │ │ - this.initLayer(layers); │ │ │ │ │ - this.handlers.feature.layer = this.layer; │ │ │ │ │ - if (isActive) { │ │ │ │ │ - this.activate(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.SelectFeature" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/NavigationHistory.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Control/Button.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.NavigationHistory │ │ │ │ │ - * A navigation history control. This is a meta-control, that creates two │ │ │ │ │ - * dependent controls: <previous> and <next>. Call the trigger method │ │ │ │ │ - * on the <previous> and <next> controls to restore previous and next │ │ │ │ │ - * history states. The previous and next controls will become active │ │ │ │ │ - * when there are available states to restore and will become deactive │ │ │ │ │ - * when there are no states to restore. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.NavigationHistory = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: type │ │ │ │ │ - * {String} Note that this control is not intended to be added directly │ │ │ │ │ - * to a control panel. Instead, add the sub-controls previous and │ │ │ │ │ - * next. These sub-controls are button type controls that activate │ │ │ │ │ - * and deactivate themselves. If this parent control is added to │ │ │ │ │ - * a panel, it will act as a toggle. │ │ │ │ │ - */ │ │ │ │ │ - type: OpenLayers.Control.TYPE_TOGGLE, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: previous │ │ │ │ │ - * {<OpenLayers.Control>} A button type control whose trigger method restores │ │ │ │ │ - * the previous state managed by this control. │ │ │ │ │ - */ │ │ │ │ │ - previous: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: previousOptions │ │ │ │ │ - * {Object} Set this property on the options argument of the constructor │ │ │ │ │ - * to set optional properties on the <previous> control. │ │ │ │ │ - */ │ │ │ │ │ - previousOptions: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: next │ │ │ │ │ - * {<OpenLayers.Control>} A button type control whose trigger method restores │ │ │ │ │ - * the next state managed by this control. │ │ │ │ │ - */ │ │ │ │ │ - next: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: nextOptions │ │ │ │ │ - * {Object} Set this property on the options argument of the constructor │ │ │ │ │ - * to set optional properties on the <next> control. │ │ │ │ │ - */ │ │ │ │ │ - nextOptions: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: limit │ │ │ │ │ - * {Integer} Optional limit on the number of history items to retain. If │ │ │ │ │ - * null, there is no limit. Default is 50. │ │ │ │ │ - */ │ │ │ │ │ - limit: 50, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ - */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: clearOnDeactivate │ │ │ │ │ - * {Boolean} Clear the history when the control is deactivated. Default │ │ │ │ │ - * is false. │ │ │ │ │ - */ │ │ │ │ │ - clearOnDeactivate: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: registry │ │ │ │ │ - * {Object} An object with keys corresponding to event types. Values │ │ │ │ │ - * are functions that return an object representing the current state. │ │ │ │ │ - */ │ │ │ │ │ - registry: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: nextStack │ │ │ │ │ - * {Array} Array of items in the history. │ │ │ │ │ - */ │ │ │ │ │ - nextStack: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: previousStack │ │ │ │ │ - * {Array} List of items in the history. First item represents the current │ │ │ │ │ - * state. │ │ │ │ │ - */ │ │ │ │ │ - previousStack: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: listeners │ │ │ │ │ - * {Object} An object containing properties corresponding to event types. │ │ │ │ │ - * This object is used to configure the control and is modified on │ │ │ │ │ - * construction. │ │ │ │ │ - */ │ │ │ │ │ - listeners: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: restoring │ │ │ │ │ - * {Boolean} Currently restoring a history state. This is set to true │ │ │ │ │ - * before calling restore and set to false after restore returns. │ │ │ │ │ - */ │ │ │ │ │ - restoring: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.NavigationHistory │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be used │ │ │ │ │ - * to extend the control. │ │ │ │ │ + * evt - {Object} event object with an xy property │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - │ │ │ │ │ - this.registry = OpenLayers.Util.extend({ │ │ │ │ │ - "moveend": this.getState │ │ │ │ │ - }, this.registry); │ │ │ │ │ - │ │ │ │ │ - var previousOptions = { │ │ │ │ │ - trigger: OpenLayers.Function.bind(this.previousTrigger, this), │ │ │ │ │ - displayClass: this.displayClass + " " + this.displayClass + "Previous" │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Util.extend(previousOptions, this.previousOptions); │ │ │ │ │ - this.previous = new OpenLayers.Control.Button(previousOptions); │ │ │ │ │ - │ │ │ │ │ - var nextOptions = { │ │ │ │ │ - trigger: OpenLayers.Function.bind(this.nextTrigger, this), │ │ │ │ │ - displayClass: this.displayClass + " " + this.displayClass + "Next" │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Util.extend(nextOptions, this.nextOptions); │ │ │ │ │ - this.next = new OpenLayers.Control.Button(nextOptions); │ │ │ │ │ - │ │ │ │ │ - this.clear(); │ │ │ │ │ + selectHover: function(evt) { │ │ │ │ │ + var bounds = this.pixelToBounds(evt.xy); │ │ │ │ │ + this.request(bounds, { │ │ │ │ │ + single: true, │ │ │ │ │ + hover: true │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: onPreviousChange │ │ │ │ │ - * Called when the previous history stack changes. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * state - {Object} An object representing the state to be restored │ │ │ │ │ - * if previous is triggered again or null if no previous states remain. │ │ │ │ │ - * length - {Integer} The number of remaining previous states that can │ │ │ │ │ - * be restored. │ │ │ │ │ + * Method: cancelHover │ │ │ │ │ + * Callback from the handlers.hover set up when <hover> selection is on │ │ │ │ │ */ │ │ │ │ │ - onPreviousChange: function(state, length) { │ │ │ │ │ - if (state && !this.previous.active) { │ │ │ │ │ - this.previous.activate(); │ │ │ │ │ - } else if (!state && this.previous.active) { │ │ │ │ │ - this.previous.deactivate(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + cancelHover: function() { │ │ │ │ │ + if (this.hoverResponse) { │ │ │ │ │ + this.protocol.abort(this.hoverResponse); │ │ │ │ │ + this.hoverResponse = null; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: onNextChange │ │ │ │ │ - * Called when the next history stack changes. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * state - {Object} An object representing the state to be restored │ │ │ │ │ - * if next is triggered again or null if no next states remain. │ │ │ │ │ - * length - {Integer} The number of remaining next states that can │ │ │ │ │ - * be restored. │ │ │ │ │ - */ │ │ │ │ │ - onNextChange: function(state, length) { │ │ │ │ │ - if (state && !this.next.active) { │ │ │ │ │ - this.next.activate(); │ │ │ │ │ - } else if (!state && this.next.active) { │ │ │ │ │ - this.next.deactivate(); │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Destroy the control. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this); │ │ │ │ │ - this.previous.destroy(); │ │ │ │ │ - this.next.destroy(); │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - for (var prop in this) { │ │ │ │ │ - this[prop] = null; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * Set the map property for the control and <previous> and <next> child │ │ │ │ │ - * controls. │ │ │ │ │ - * │ │ │ │ │ + * Method: request │ │ │ │ │ + * Sends a GetFeature request to the WFS │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.map = map; │ │ │ │ │ - this.next.setMap(map); │ │ │ │ │ - this.previous.setMap(map); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * Called when the control is added to the map. │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - this.next.draw(); │ │ │ │ │ - this.previous.draw(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: previousTrigger │ │ │ │ │ - * Restore the previous state. If no items are in the previous history │ │ │ │ │ - * stack, this has no effect. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Item representing state that was restored. Undefined if no │ │ │ │ │ - * items are in the previous history stack. │ │ │ │ │ - */ │ │ │ │ │ - previousTrigger: function() { │ │ │ │ │ - var current = this.previousStack.shift(); │ │ │ │ │ - var state = this.previousStack.shift(); │ │ │ │ │ - if (state != undefined) { │ │ │ │ │ - this.nextStack.unshift(current); │ │ │ │ │ - this.previousStack.unshift(state); │ │ │ │ │ - this.restoring = true; │ │ │ │ │ - this.restore(state); │ │ │ │ │ - this.restoring = false; │ │ │ │ │ - this.onNextChange(this.nextStack[0], this.nextStack.length); │ │ │ │ │ - this.onPreviousChange( │ │ │ │ │ - this.previousStack[1], this.previousStack.length - 1 │ │ │ │ │ - ); │ │ │ │ │ - } else { │ │ │ │ │ - this.previousStack.unshift(current); │ │ │ │ │ - } │ │ │ │ │ - return state; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: nextTrigger │ │ │ │ │ - * Restore the next state. If no items are in the next history │ │ │ │ │ - * stack, this has no effect. The next history stack is populated │ │ │ │ │ - * as states are restored from the previous history stack. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} Item representing state that was restored. Undefined if no │ │ │ │ │ - * items are in the next history stack. │ │ │ │ │ - */ │ │ │ │ │ - nextTrigger: function() { │ │ │ │ │ - var state = this.nextStack.shift(); │ │ │ │ │ - if (state != undefined) { │ │ │ │ │ - this.previousStack.unshift(state); │ │ │ │ │ - this.restoring = true; │ │ │ │ │ - this.restore(state); │ │ │ │ │ - this.restoring = false; │ │ │ │ │ - this.onNextChange(this.nextStack[0], this.nextStack.length); │ │ │ │ │ - this.onPreviousChange( │ │ │ │ │ - this.previousStack[1], this.previousStack.length - 1 │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - return state; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: clear │ │ │ │ │ - * Clear history. │ │ │ │ │ - */ │ │ │ │ │ - clear: function() { │ │ │ │ │ - this.previousStack = []; │ │ │ │ │ - this.previous.deactivate(); │ │ │ │ │ - this.nextStack = []; │ │ │ │ │ - this.next.deactivate(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: getState │ │ │ │ │ - * Get the current state and return it. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} An object representing the current state. │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} bounds for the request's BBOX filter │ │ │ │ │ + * options - {Object} additional options for this method. │ │ │ │ │ + * │ │ │ │ │ + * Supported options include: │ │ │ │ │ + * single - {Boolean} A single feature should be returned. │ │ │ │ │ + * Note that this will be ignored if the protocol does not │ │ │ │ │ + * return the geometries of the features. │ │ │ │ │ + * hover - {Boolean} Do the request for the hover handler. │ │ │ │ │ */ │ │ │ │ │ - getState: function() { │ │ │ │ │ - return { │ │ │ │ │ - center: this.map.getCenter(), │ │ │ │ │ - resolution: this.map.getResolution(), │ │ │ │ │ - projection: this.map.getProjectionObject(), │ │ │ │ │ - units: this.map.getProjectionObject().getUnits() || │ │ │ │ │ - this.map.units || this.map.baseLayer.units │ │ │ │ │ - }; │ │ │ │ │ - }, │ │ │ │ │ + request: function(bounds, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: this.filterType, │ │ │ │ │ + value: bounds │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: restore │ │ │ │ │ - * Update the state with the given object. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * state - {Object} An object representing the state to restore. │ │ │ │ │ - */ │ │ │ │ │ - restore: function(state) { │ │ │ │ │ - var center, zoom; │ │ │ │ │ - if (this.map.getProjectionObject() == state.projection) { │ │ │ │ │ - zoom = this.map.getZoomForResolution(state.resolution); │ │ │ │ │ - center = state.center; │ │ │ │ │ - } else { │ │ │ │ │ - center = state.center.clone(); │ │ │ │ │ - center.transform(state.projection, this.map.getProjectionObject()); │ │ │ │ │ - var sourceUnits = state.units; │ │ │ │ │ - var targetUnits = this.map.getProjectionObject().getUnits() || │ │ │ │ │ - this.map.units || this.map.baseLayer.units; │ │ │ │ │ - var resolutionFactor = sourceUnits && targetUnits ? │ │ │ │ │ - OpenLayers.INCHES_PER_UNIT[sourceUnits] / OpenLayers.INCHES_PER_UNIT[targetUnits] : 1; │ │ │ │ │ - zoom = this.map.getZoomForResolution(resolutionFactor * state.resolution); │ │ │ │ │ - } │ │ │ │ │ - this.map.setCenter(center, zoom); │ │ │ │ │ - }, │ │ │ │ │ + // Set the cursor to "wait" to tell the user we're working. │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setListeners │ │ │ │ │ - * Sets functions to be registered in the listeners object. │ │ │ │ │ - */ │ │ │ │ │ - setListeners: function() { │ │ │ │ │ - this.listeners = {}; │ │ │ │ │ - for (var type in this.registry) { │ │ │ │ │ - this.listeners[type] = OpenLayers.Function.bind(function() { │ │ │ │ │ - if (!this.restoring) { │ │ │ │ │ - var state = this.registry[type].apply(this, arguments); │ │ │ │ │ - this.previousStack.unshift(state); │ │ │ │ │ - if (this.previousStack.length > 1) { │ │ │ │ │ - this.onPreviousChange( │ │ │ │ │ - this.previousStack[1], this.previousStack.length - 1 │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (this.previousStack.length > (this.limit + 1)) { │ │ │ │ │ - this.previousStack.pop(); │ │ │ │ │ - } │ │ │ │ │ - if (this.nextStack.length > 0) { │ │ │ │ │ - this.nextStack = []; │ │ │ │ │ - this.onNextChange(null, 0); │ │ │ │ │ + var response = this.protocol.read({ │ │ │ │ │ + maxFeatures: options.single == true ? this.maxFeatures : undefined, │ │ │ │ │ + filter: filter, │ │ │ │ │ + callback: function(result) { │ │ │ │ │ + if (result.success()) { │ │ │ │ │ + if (result.features.length) { │ │ │ │ │ + if (options.single == true) { │ │ │ │ │ + this.selectBestFeature(result.features, │ │ │ │ │ + bounds.getCenterLonLat(), options); │ │ │ │ │ + } else { │ │ │ │ │ + this.select(result.features); │ │ │ │ │ + } │ │ │ │ │ + } else if (options.hover) { │ │ │ │ │ + this.hoverSelect(); │ │ │ │ │ + } else { │ │ │ │ │ + this.events.triggerEvent("clickout"); │ │ │ │ │ + if (this.clickout) { │ │ │ │ │ + this.unselectAll(); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return true; │ │ │ │ │ - }, this); │ │ │ │ │ + // Reset the cursor. │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + if (options.hover == true) { │ │ │ │ │ + this.hoverResponse = response; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Activate the control. This registers any listeners. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Control successfully activated. │ │ │ │ │ + * Method: selectBestFeature │ │ │ │ │ + * Selects the feature from an array of features that is the best match │ │ │ │ │ + * for the click position. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * clickPosition - {<OpenLayers.LonLat>} │ │ │ │ │ + * options - {Object} additional options for this method │ │ │ │ │ + * │ │ │ │ │ + * Supported options include: │ │ │ │ │ + * hover - {Boolean} Do the selection for the hover handler. │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this)) { │ │ │ │ │ - if (this.listeners == null) { │ │ │ │ │ - this.setListeners(); │ │ │ │ │ - } │ │ │ │ │ - for (var type in this.listeners) { │ │ │ │ │ - this.map.events.register(type, this, this.listeners[type]); │ │ │ │ │ - } │ │ │ │ │ - activated = true; │ │ │ │ │ - if (this.previousStack.length == 0) { │ │ │ │ │ - this.initStack(); │ │ │ │ │ + selectBestFeature: function(features, clickPosition, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + if (features.length) { │ │ │ │ │ + var point = new OpenLayers.Geometry.Point(clickPosition.lon, │ │ │ │ │ + clickPosition.lat); │ │ │ │ │ + var feature, resultFeature, dist; │ │ │ │ │ + var minDist = Number.MAX_VALUE; │ │ │ │ │ + for (var i = 0; i < features.length; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + dist = point.distanceTo(feature.geometry, { │ │ │ │ │ + edge: false │ │ │ │ │ + }); │ │ │ │ │ + if (dist < minDist) { │ │ │ │ │ + minDist = dist; │ │ │ │ │ + resultFeature = feature; │ │ │ │ │ + if (minDist == 0) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: initStack │ │ │ │ │ - * Called after the control is activated if the previous history stack is │ │ │ │ │ - * empty. │ │ │ │ │ - */ │ │ │ │ │ - initStack: function() { │ │ │ │ │ - if (this.map.getCenter()) { │ │ │ │ │ - this.listeners.moveend(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the control. This unregisters any listeners. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Control successfully deactivated. │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this)) { │ │ │ │ │ - for (var type in this.listeners) { │ │ │ │ │ - this.map.events.unregister( │ │ │ │ │ - type, this, this.listeners[type] │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - if (this.clearOnDeactivate) { │ │ │ │ │ - this.clear(); │ │ │ │ │ - } │ │ │ │ │ - deactivated = true; │ │ │ │ │ + if (options.hover == true) { │ │ │ │ │ + this.hoverSelect(resultFeature); │ │ │ │ │ + } else { │ │ │ │ │ + this.select(resultFeature || features); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.NavigationHistory" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/TouchNavigation.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control/DragPan.js │ │ │ │ │ - * @requires OpenLayers/Control/PinchZoom.js │ │ │ │ │ - * @requires OpenLayers/Handler/Click.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.TouchNavigation │ │ │ │ │ - * The navigation control handles map browsing with touch events (dragging, │ │ │ │ │ - * double-tapping, tap with two fingers, and pinch zoom). Create a new │ │ │ │ │ - * control with the <OpenLayers.Control.TouchNavigation> constructor. │ │ │ │ │ - * │ │ │ │ │ - * If you’re only targeting touch enabled devices with your mapping application, │ │ │ │ │ - * you can create a map with only a TouchNavigation control. The │ │ │ │ │ - * <OpenLayers.Control.Navigation> control is mobile ready by default, but │ │ │ │ │ - * you can generate a smaller build of the library by only including this │ │ │ │ │ - * touch navigation control if you aren't concerned about mouse interaction. │ │ │ │ │ - * │ │ │ │ │ - * Inherits: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.TouchNavigation = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: dragPan │ │ │ │ │ - * {<OpenLayers.Control.DragPan>} │ │ │ │ │ - */ │ │ │ │ │ - dragPan: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: dragPanOptions │ │ │ │ │ - * {Object} Options passed to the DragPan control. │ │ │ │ │ - */ │ │ │ │ │ - dragPanOptions: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: pinchZoom │ │ │ │ │ - * {<OpenLayers.Control.PinchZoom>} │ │ │ │ │ - */ │ │ │ │ │ - pinchZoom: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: pinchZoomOptions │ │ │ │ │ - * {Object} Options passed to the PinchZoom control. │ │ │ │ │ - */ │ │ │ │ │ - pinchZoomOptions: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: clickHandlerOptions │ │ │ │ │ - * {Object} Options passed to the Click handler. │ │ │ │ │ - */ │ │ │ │ │ - clickHandlerOptions: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: documentDrag │ │ │ │ │ - * {Boolean} Allow panning of the map by dragging outside map viewport. │ │ │ │ │ - * Default is false. │ │ │ │ │ - */ │ │ │ │ │ - documentDrag: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ - */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.TouchNavigation │ │ │ │ │ - * Create a new navigation control │ │ │ │ │ - * │ │ │ │ │ + * Method: setModifiers │ │ │ │ │ + * Sets the multiple and toggle modifiers according to the current event │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be set on │ │ │ │ │ - * the control │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.handlers = {}; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroy │ │ │ │ │ - * The destroy method is used to perform any clean up before the control │ │ │ │ │ - * is dereferenced. Typically this is where event listeners are removed │ │ │ │ │ - * to prevent memory leaks. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - if (this.dragPan) { │ │ │ │ │ - this.dragPan.destroy(); │ │ │ │ │ - } │ │ │ │ │ - this.dragPan = null; │ │ │ │ │ - if (this.pinchZoom) { │ │ │ │ │ - this.pinchZoom.destroy(); │ │ │ │ │ - delete this.pinchZoom; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: activate │ │ │ │ │ - */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.dragPan.activate(); │ │ │ │ │ - this.handlers.click.activate(); │ │ │ │ │ - this.pinchZoom.activate(); │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: deactivate │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.dragPan.deactivate(); │ │ │ │ │ - this.handlers.click.deactivate(); │ │ │ │ │ - this.pinchZoom.deactivate(); │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - var clickCallbacks = { │ │ │ │ │ - click: this.defaultClick, │ │ │ │ │ - dblclick: this.defaultDblClick │ │ │ │ │ + setModifiers: function(evt) { │ │ │ │ │ + this.modifiers = { │ │ │ │ │ + multiple: this.multiple || (this.multipleKey && evt[this.multipleKey]), │ │ │ │ │ + toggle: this.toggle || (this.toggleKey && evt[this.toggleKey]) │ │ │ │ │ }; │ │ │ │ │ - var clickOptions = OpenLayers.Util.extend({ │ │ │ │ │ - "double": true, │ │ │ │ │ - stopDouble: true, │ │ │ │ │ - pixelTolerance: 2 │ │ │ │ │ - }, this.clickHandlerOptions); │ │ │ │ │ - this.handlers.click = new OpenLayers.Handler.Click( │ │ │ │ │ - this, clickCallbacks, clickOptions │ │ │ │ │ - ); │ │ │ │ │ - this.dragPan = new OpenLayers.Control.DragPan( │ │ │ │ │ - OpenLayers.Util.extend({ │ │ │ │ │ - map: this.map, │ │ │ │ │ - documentDrag: this.documentDrag │ │ │ │ │ - }, this.dragPanOptions) │ │ │ │ │ - ); │ │ │ │ │ - this.dragPan.draw(); │ │ │ │ │ - this.pinchZoom = new OpenLayers.Control.PinchZoom( │ │ │ │ │ - OpenLayers.Util.extend({ │ │ │ │ │ - map: this.map │ │ │ │ │ - }, this.pinchZoomOptions) │ │ │ │ │ - ); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: defaultClick │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - defaultClick: function(evt) { │ │ │ │ │ - if (evt.lastTouches && evt.lastTouches.length == 2) { │ │ │ │ │ - this.map.zoomOut(); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: defaultDblClick │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - defaultDblClick: function(evt) { │ │ │ │ │ - this.map.zoomTo(this.map.zoom + 1, evt.xy); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.TouchNavigation" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/LayerSwitcher.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - * @requires OpenLayers/Util.js │ │ │ │ │ - * @requires OpenLayers/Events/buttonclick.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.LayerSwitcher │ │ │ │ │ - * The LayerSwitcher control displays a table of contents for the map. This │ │ │ │ │ - * allows the user interface to switch between BaseLasyers and to show or hide │ │ │ │ │ - * Overlays. By default the switcher is shown minimized on the right edge of │ │ │ │ │ - * the map, the user may expand it by clicking on the handle. │ │ │ │ │ - * │ │ │ │ │ - * To create the LayerSwitcher outside of the map, pass the Id of a html div │ │ │ │ │ - * as the first argument to the constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.LayerSwitcher = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layerStates │ │ │ │ │ - * {Array(Object)} Basically a copy of the "state" of the map's layers │ │ │ │ │ - * the last time the control was drawn. We have this in order to avoid │ │ │ │ │ - * unnecessarily redrawing the control. │ │ │ │ │ - */ │ │ │ │ │ - layerStates: null, │ │ │ │ │ - │ │ │ │ │ - // DOM Elements │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: layersDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - layersDiv: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: baseLayersDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - baseLayersDiv: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: baseLayers │ │ │ │ │ - * {Array(Object)} │ │ │ │ │ - */ │ │ │ │ │ - baseLayers: null, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: dataLbl │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - dataLbl: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: dataLayersDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - dataLayersDiv: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: dataLayers │ │ │ │ │ - * {Array(Object)} │ │ │ │ │ - */ │ │ │ │ │ - dataLayers: null, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: minimizeDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - minimizeDiv: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: maximizeDiv │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - maximizeDiv: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: ascending │ │ │ │ │ - * {Boolean} │ │ │ │ │ - */ │ │ │ │ │ - ascending: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.LayerSwitcher │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.layerStates = []; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - │ │ │ │ │ - //clear out layers info and unregister their events │ │ │ │ │ - this.clearLayersArray("base"); │ │ │ │ │ - this.clearLayersArray("data"); │ │ │ │ │ - │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - buttonclick: this.onButtonClick, │ │ │ │ │ - addlayer: this.redraw, │ │ │ │ │ - changelayer: this.redraw, │ │ │ │ │ - removelayer: this.redraw, │ │ │ │ │ - changebaselayer: this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ - │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: setMap │ │ │ │ │ - * │ │ │ │ │ - * Properties: │ │ │ │ │ - * map - {<OpenLayers.Map>} │ │ │ │ │ - */ │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - addlayer: this.redraw, │ │ │ │ │ - changelayer: this.redraw, │ │ │ │ │ - removelayer: this.redraw, │ │ │ │ │ - changebaselayer: this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - if (this.outsideViewport) { │ │ │ │ │ - this.events.attachToElement(this.div); │ │ │ │ │ - this.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ - } else { │ │ │ │ │ - this.map.events.register("buttonclick", this, this.onButtonClick); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A reference to the DIV DOMElement containing the │ │ │ │ │ - * switcher tabs. │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this); │ │ │ │ │ - │ │ │ │ │ - // create layout divs │ │ │ │ │ - this.loadContents(); │ │ │ │ │ - │ │ │ │ │ - // set mode to minimize │ │ │ │ │ - if (!this.outsideViewport) { │ │ │ │ │ - this.minimizeControl(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // populate div with current info │ │ │ │ │ - this.redraw(); │ │ │ │ │ - │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: onButtonClick │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Event} │ │ │ │ │ - */ │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - var button = evt.buttonElement; │ │ │ │ │ - if (button === this.minimizeDiv) { │ │ │ │ │ - this.minimizeControl(); │ │ │ │ │ - } else if (button === this.maximizeDiv) { │ │ │ │ │ - this.maximizeControl(); │ │ │ │ │ - } else if (button._layerSwitcher === this.id) { │ │ │ │ │ - if (button["for"]) { │ │ │ │ │ - button = document.getElementById(button["for"]); │ │ │ │ │ - } │ │ │ │ │ - if (!button.disabled) { │ │ │ │ │ - if (button.type == "radio") { │ │ │ │ │ - button.checked = true; │ │ │ │ │ - this.map.setBaseLayer(this.map.getLayer(button._layer)); │ │ │ │ │ - } else { │ │ │ │ │ - button.checked = !button.checked; │ │ │ │ │ - this.updateMap(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clearLayersArray │ │ │ │ │ - * User specifies either "base" or "data". we then clear all the │ │ │ │ │ - * corresponding listeners, the div, and reinitialize a new array. │ │ │ │ │ - * │ │ │ │ │ + * Method: select │ │ │ │ │ + * Add feature to the hash of selected features and trigger the │ │ │ │ │ + * featureselected and featuresselected events. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layersType - {String} │ │ │ │ │ - */ │ │ │ │ │ - clearLayersArray: function(layersType) { │ │ │ │ │ - this[layersType + "LayersDiv"].innerHTML = ""; │ │ │ │ │ - this[layersType + "Layers"] = []; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: checkRedraw │ │ │ │ │ - * Checks if the layer state has changed since the last redraw() call. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The layer state changed since the last redraw() call. │ │ │ │ │ - */ │ │ │ │ │ - checkRedraw: function() { │ │ │ │ │ - if (!this.layerStates.length || │ │ │ │ │ - (this.map.layers.length != this.layerStates.length)) { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - for (var i = 0, len = this.layerStates.length; i < len; i++) { │ │ │ │ │ - var layerState = this.layerStates[i]; │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if ((layerState.name != layer.name) || │ │ │ │ │ - (layerState.inRange != layer.inRange) || │ │ │ │ │ - (layerState.id != layer.id) || │ │ │ │ │ - (layerState.visibility != layer.visibility)) { │ │ │ │ │ - return true; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: redraw │ │ │ │ │ - * Goes through and takes the current state of the Map and rebuilds the │ │ │ │ │ - * control to display that state. Groups base layers into a │ │ │ │ │ - * radio-button group and lists each data layer with a checkbox. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ + * features - {<OpenLayers.Feature.Vector>} or an array of features │ │ │ │ │ */ │ │ │ │ │ - redraw: function() { │ │ │ │ │ - //if the state hasn't changed since last redraw, no need │ │ │ │ │ - // to do anything. Just return the existing div. │ │ │ │ │ - if (!this.checkRedraw()) { │ │ │ │ │ - return this.div; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //clear out previous layers │ │ │ │ │ - this.clearLayersArray("base"); │ │ │ │ │ - this.clearLayersArray("data"); │ │ │ │ │ - │ │ │ │ │ - var containsOverlays = false; │ │ │ │ │ - var containsBaseLayers = false; │ │ │ │ │ - │ │ │ │ │ - // Save state -- for checking layer if the map state changed. │ │ │ │ │ - // We save this before redrawing, because in the process of redrawing │ │ │ │ │ - // we will trigger more visibility changes, and we want to not redraw │ │ │ │ │ - // and enter an infinite loop. │ │ │ │ │ - var len = this.map.layers.length; │ │ │ │ │ - this.layerStates = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - this.layerStates[i] = { │ │ │ │ │ - 'name': layer.name, │ │ │ │ │ - 'visibility': layer.visibility, │ │ │ │ │ - 'inRange': layer.inRange, │ │ │ │ │ - 'id': layer.id │ │ │ │ │ - }; │ │ │ │ │ + select: function(features) { │ │ │ │ │ + if (!this.modifiers.multiple && !this.modifiers.toggle) { │ │ │ │ │ + this.unselectAll(); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var layers = this.map.layers.slice(); │ │ │ │ │ - if (!this.ascending) { │ │ │ │ │ - layers.reverse(); │ │ │ │ │ + if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ + features = [features]; │ │ │ │ │ } │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - var layer = layers[i]; │ │ │ │ │ - var baseLayer = layer.isBaseLayer; │ │ │ │ │ - │ │ │ │ │ - if (layer.displayInLayerSwitcher) { │ │ │ │ │ │ │ │ │ │ - if (baseLayer) { │ │ │ │ │ - containsBaseLayers = true; │ │ │ │ │ + var cont = this.events.triggerEvent("beforefeaturesselected", { │ │ │ │ │ + features: features │ │ │ │ │ + }); │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + var selectedFeatures = []; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (this.features[feature.fid || feature.id]) { │ │ │ │ │ + if (this.modifiers.toggle) { │ │ │ │ │ + this.unselect(this.features[feature.fid || feature.id]); │ │ │ │ │ + } │ │ │ │ │ } else { │ │ │ │ │ - containsOverlays = true; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // only check a baselayer if it is *the* baselayer, check data │ │ │ │ │ - // layers if they are visible │ │ │ │ │ - var checked = (baseLayer) ? (layer == this.map.baseLayer) : │ │ │ │ │ - layer.getVisibility(); │ │ │ │ │ - │ │ │ │ │ - // create input element │ │ │ │ │ - var inputElem = document.createElement("input"), │ │ │ │ │ - // The input shall have an id attribute so we can use │ │ │ │ │ - // labels to interact with them. │ │ │ │ │ - inputId = OpenLayers.Util.createUniqueID( │ │ │ │ │ - this.id + "_input_" │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - inputElem.id = inputId; │ │ │ │ │ - inputElem.name = (baseLayer) ? this.id + "_baseLayers" : layer.name; │ │ │ │ │ - inputElem.type = (baseLayer) ? "radio" : "checkbox"; │ │ │ │ │ - inputElem.value = layer.name; │ │ │ │ │ - inputElem.checked = checked; │ │ │ │ │ - inputElem.defaultChecked = checked; │ │ │ │ │ - inputElem.className = "olButton"; │ │ │ │ │ - inputElem._layer = layer.id; │ │ │ │ │ - inputElem._layerSwitcher = this.id; │ │ │ │ │ - │ │ │ │ │ - if (!baseLayer && !layer.inRange) { │ │ │ │ │ - inputElem.disabled = true; │ │ │ │ │ - } │ │ │ │ │ + cont = this.events.triggerEvent("beforefeatureselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + this.features[feature.fid || feature.id] = feature; │ │ │ │ │ + selectedFeatures.push(feature); │ │ │ │ │ │ │ │ │ │ - // create span │ │ │ │ │ - var labelSpan = document.createElement("label"); │ │ │ │ │ - // this isn't the DOM attribute 'for', but an arbitrary name we │ │ │ │ │ - // use to find the appropriate input element in <onButtonClick> │ │ │ │ │ - labelSpan["for"] = inputElem.id; │ │ │ │ │ - OpenLayers.Element.addClass(labelSpan, "labelSpan olButton"); │ │ │ │ │ - labelSpan._layer = layer.id; │ │ │ │ │ - labelSpan._layerSwitcher = this.id; │ │ │ │ │ - if (!baseLayer && !layer.inRange) { │ │ │ │ │ - labelSpan.style.color = "gray"; │ │ │ │ │ + this.events.triggerEvent("featureselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - labelSpan.innerHTML = layer.name; │ │ │ │ │ - labelSpan.style.verticalAlign = (baseLayer) ? "bottom" : │ │ │ │ │ - "baseline"; │ │ │ │ │ - // create line break │ │ │ │ │ - var br = document.createElement("br"); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - var groupArray = (baseLayer) ? this.baseLayers : │ │ │ │ │ - this.dataLayers; │ │ │ │ │ - groupArray.push({ │ │ │ │ │ - 'layer': layer, │ │ │ │ │ - 'inputElem': inputElem, │ │ │ │ │ - 'labelSpan': labelSpan │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - │ │ │ │ │ - var groupDiv = (baseLayer) ? this.baseLayersDiv : │ │ │ │ │ - this.dataLayersDiv; │ │ │ │ │ - groupDiv.appendChild(inputElem); │ │ │ │ │ - groupDiv.appendChild(labelSpan); │ │ │ │ │ - groupDiv.appendChild(br); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // if no overlays, dont display the overlay label │ │ │ │ │ - this.dataLbl.style.display = (containsOverlays) ? "" : "none"; │ │ │ │ │ - │ │ │ │ │ - // if no baselayers, dont display the baselayer label │ │ │ │ │ - this.baseLbl.style.display = (containsBaseLayers) ? "" : "none"; │ │ │ │ │ - │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateMap │ │ │ │ │ - * Cycles through the loaded data and base layer input arrays and makes │ │ │ │ │ - * the necessary calls to the Map object such that that the map's │ │ │ │ │ - * visual state corresponds to what the user has selected in │ │ │ │ │ - * the control. │ │ │ │ │ - */ │ │ │ │ │ - updateMap: function() { │ │ │ │ │ - │ │ │ │ │ - // set the newly selected base layer │ │ │ │ │ - for (var i = 0, len = this.baseLayers.length; i < len; i++) { │ │ │ │ │ - var layerEntry = this.baseLayers[i]; │ │ │ │ │ - if (layerEntry.inputElem.checked) { │ │ │ │ │ - this.map.setBaseLayer(layerEntry.layer, false); │ │ │ │ │ } │ │ │ │ │ + this.events.triggerEvent("featuresselected", { │ │ │ │ │ + features: selectedFeatures │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // set the correct visibilities for the overlays │ │ │ │ │ - for (var i = 0, len = this.dataLayers.length; i < len; i++) { │ │ │ │ │ - var layerEntry = this.dataLayers[i]; │ │ │ │ │ - layerEntry.layer.setVisibility(layerEntry.inputElem.checked); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: maximizeControl │ │ │ │ │ - * Set up the labels and divs for the control │ │ │ │ │ - * │ │ │ │ │ + * Method: hoverSelect │ │ │ │ │ + * Sets/unsets the <hoverFeature> │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * e - {Event} │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} the feature to hover-select. │ │ │ │ │ + * If none is provided, the current <hoverFeature> will be nulled and │ │ │ │ │ + * the outfeature event will be triggered. │ │ │ │ │ */ │ │ │ │ │ - maximizeControl: function(e) { │ │ │ │ │ - │ │ │ │ │ - // set the div's width and height to empty values, so │ │ │ │ │ - // the div dimensions can be controlled by CSS │ │ │ │ │ - this.div.style.width = ""; │ │ │ │ │ - this.div.style.height = ""; │ │ │ │ │ - │ │ │ │ │ - this.showControls(false); │ │ │ │ │ + hoverSelect: function(feature) { │ │ │ │ │ + var fid = feature ? feature.fid || feature.id : null; │ │ │ │ │ + var hfid = this.hoverFeature ? │ │ │ │ │ + this.hoverFeature.fid || this.hoverFeature.id : null; │ │ │ │ │ │ │ │ │ │ - if (e != null) { │ │ │ │ │ - OpenLayers.Event.stop(e); │ │ │ │ │ + if (hfid && hfid != fid) { │ │ │ │ │ + this.events.triggerEvent("outfeature", { │ │ │ │ │ + feature: this.hoverFeature │ │ │ │ │ + }); │ │ │ │ │ + this.hoverFeature = null; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: minimizeControl │ │ │ │ │ - * Hide all the contents of the control, shrink the size, │ │ │ │ │ - * add the maximize icon │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * e - {Event} │ │ │ │ │ - */ │ │ │ │ │ - minimizeControl: function(e) { │ │ │ │ │ - │ │ │ │ │ - // to minimize the control we set its div's width │ │ │ │ │ - // and height to 0px, we cannot just set "display" │ │ │ │ │ - // to "none" because it would hide the maximize │ │ │ │ │ - // div │ │ │ │ │ - this.div.style.width = "0px"; │ │ │ │ │ - this.div.style.height = "0px"; │ │ │ │ │ - │ │ │ │ │ - this.showControls(true); │ │ │ │ │ - │ │ │ │ │ - if (e != null) { │ │ │ │ │ - OpenLayers.Event.stop(e); │ │ │ │ │ + if (fid && fid != hfid) { │ │ │ │ │ + this.events.triggerEvent("hoverfeature", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.hoverFeature = feature; │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: showControls │ │ │ │ │ - * Hide/Show all LayerSwitcher controls depending on whether we are │ │ │ │ │ - * minimized or not │ │ │ │ │ + * Method: unselect │ │ │ │ │ + * Remove feature from the hash of selected features and trigger the │ │ │ │ │ + * featureunselected event. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * minimize - {Boolean} │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - showControls: function(minimize) { │ │ │ │ │ - │ │ │ │ │ - this.maximizeDiv.style.display = minimize ? "" : "none"; │ │ │ │ │ - this.minimizeDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ - │ │ │ │ │ - this.layersDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ + unselect: function(feature) { │ │ │ │ │ + delete this.features[feature.fid || feature.id]; │ │ │ │ │ + this.events.triggerEvent("featureunselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: loadContents │ │ │ │ │ - * Set up the labels and divs for the control │ │ │ │ │ + * Method: unselectAll │ │ │ │ │ + * Unselect all selected features. │ │ │ │ │ */ │ │ │ │ │ - loadContents: function() { │ │ │ │ │ - │ │ │ │ │ - // layers list div │ │ │ │ │ - this.layersDiv = document.createElement("div"); │ │ │ │ │ - this.layersDiv.id = this.id + "_layersDiv"; │ │ │ │ │ - OpenLayers.Element.addClass(this.layersDiv, "layersDiv"); │ │ │ │ │ - │ │ │ │ │ - this.baseLbl = document.createElement("div"); │ │ │ │ │ - this.baseLbl.innerHTML = OpenLayers.i18n("Base Layer"); │ │ │ │ │ - OpenLayers.Element.addClass(this.baseLbl, "baseLbl"); │ │ │ │ │ - │ │ │ │ │ - this.baseLayersDiv = document.createElement("div"); │ │ │ │ │ - OpenLayers.Element.addClass(this.baseLayersDiv, "baseLayersDiv"); │ │ │ │ │ - │ │ │ │ │ - this.dataLbl = document.createElement("div"); │ │ │ │ │ - this.dataLbl.innerHTML = OpenLayers.i18n("Overlays"); │ │ │ │ │ - OpenLayers.Element.addClass(this.dataLbl, "dataLbl"); │ │ │ │ │ - │ │ │ │ │ - this.dataLayersDiv = document.createElement("div"); │ │ │ │ │ - OpenLayers.Element.addClass(this.dataLayersDiv, "dataLayersDiv"); │ │ │ │ │ - │ │ │ │ │ - if (this.ascending) { │ │ │ │ │ - this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ - } else { │ │ │ │ │ - this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ + unselectAll: function() { │ │ │ │ │ + // we'll want an option to supress notification here │ │ │ │ │ + for (var fid in this.features) { │ │ │ │ │ + this.unselect(this.features[fid]); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - this.div.appendChild(this.layersDiv); │ │ │ │ │ - │ │ │ │ │ - // maximize button div │ │ │ │ │ - var img = OpenLayers.Util.getImageLocation('layer-switcher-maximize.png'); │ │ │ │ │ - this.maximizeDiv = OpenLayers.Util.createAlphaImageDiv( │ │ │ │ │ - "OpenLayers_Control_MaximizeDiv", │ │ │ │ │ - null, │ │ │ │ │ - null, │ │ │ │ │ - img, │ │ │ │ │ - "absolute"); │ │ │ │ │ - OpenLayers.Element.addClass(this.maximizeDiv, "maximizeDiv olButton"); │ │ │ │ │ - this.maximizeDiv.style.display = "none"; │ │ │ │ │ - │ │ │ │ │ - this.div.appendChild(this.maximizeDiv); │ │ │ │ │ - │ │ │ │ │ - // minimize button div │ │ │ │ │ - var img = OpenLayers.Util.getImageLocation('layer-switcher-minimize.png'); │ │ │ │ │ - this.minimizeDiv = OpenLayers.Util.createAlphaImageDiv( │ │ │ │ │ - "OpenLayers_Control_MinimizeDiv", │ │ │ │ │ - null, │ │ │ │ │ - null, │ │ │ │ │ - img, │ │ │ │ │ - "absolute"); │ │ │ │ │ - OpenLayers.Element.addClass(this.minimizeDiv, "minimizeDiv olButton"); │ │ │ │ │ - this.minimizeDiv.style.display = "none"; │ │ │ │ │ - │ │ │ │ │ - this.div.appendChild(this.minimizeDiv); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.LayerSwitcher" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Graticule.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - * @requires OpenLayers/Rule.js │ │ │ │ │ - * @requires OpenLayers/StyleMap.js │ │ │ │ │ - * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Control.Graticule │ │ │ │ │ - * The Graticule displays a grid of latitude/longitude lines reprojected on │ │ │ │ │ - * the map. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ - * │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Control.Graticule = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: autoActivate │ │ │ │ │ - * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ - * true. │ │ │ │ │ - */ │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: intervals │ │ │ │ │ - * {Array(Float)} A list of possible graticule widths in degrees. │ │ │ │ │ - */ │ │ │ │ │ - intervals: [45, 30, 20, 10, 5, 2, 1, │ │ │ │ │ - 0.5, 0.2, 0.1, 0.05, 0.01, │ │ │ │ │ - 0.005, 0.002, 0.001 │ │ │ │ │ - ], │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: displayInLayerSwitcher │ │ │ │ │ - * {Boolean} Allows the Graticule control to be switched on and off by │ │ │ │ │ - * LayerSwitcher control. Defaults is true. │ │ │ │ │ - */ │ │ │ │ │ - displayInLayerSwitcher: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: visible │ │ │ │ │ - * {Boolean} should the graticule be initially visible (default=true) │ │ │ │ │ - */ │ │ │ │ │ - visible: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: numPoints │ │ │ │ │ - * {Integer} The number of points to use in each graticule line. Higher │ │ │ │ │ - * numbers result in a smoother curve for projected maps │ │ │ │ │ - */ │ │ │ │ │ - numPoints: 50, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: targetSize │ │ │ │ │ - * {Integer} The maximum size of the grid in pixels on the map │ │ │ │ │ - */ │ │ │ │ │ - targetSize: 200, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: layerName │ │ │ │ │ - * {String} The name to be displayed in the layer switcher, default is set │ │ │ │ │ - * by {<OpenLayers.Lang>}. │ │ │ │ │ - */ │ │ │ │ │ - layerName: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: labelled │ │ │ │ │ - * {Boolean} Should the graticule lines be labelled?. default=true │ │ │ │ │ - */ │ │ │ │ │ - labelled: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: labelFormat │ │ │ │ │ - * {String} the format of the labels, default = 'dm'. See │ │ │ │ │ - * <OpenLayers.Util.getFormattedLonLat> for other options. │ │ │ │ │ - */ │ │ │ │ │ - labelFormat: 'dm', │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: lineSymbolizer │ │ │ │ │ - * {symbolizer} the symbolizer used to render lines │ │ │ │ │ - */ │ │ │ │ │ - lineSymbolizer: { │ │ │ │ │ - strokeColor: "#333", │ │ │ │ │ - strokeWidth: 1, │ │ │ │ │ - strokeOpacity: 0.5 │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: labelSymbolizer │ │ │ │ │ - * {symbolizer} the symbolizer used to render labels │ │ │ │ │ - */ │ │ │ │ │ - labelSymbolizer: {}, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: gratLayer │ │ │ │ │ - * {<OpenLayers.Layer.Vector>} vector layer used to draw the graticule on │ │ │ │ │ - */ │ │ │ │ │ - gratLayer: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Graticule │ │ │ │ │ - * Create a new graticule control to display a grid of latitude longitude │ │ │ │ │ - * lines. │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the control. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} An optional object whose properties will be used │ │ │ │ │ - * to extend the control. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.layerName = options.layerName || OpenLayers.i18n("Graticule"); │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - │ │ │ │ │ - this.labelSymbolizer.stroke = false; │ │ │ │ │ - this.labelSymbolizer.fill = false; │ │ │ │ │ - this.labelSymbolizer.label = "${label}"; │ │ │ │ │ - this.labelSymbolizer.labelAlign = "${labelAlign}"; │ │ │ │ │ - this.labelSymbolizer.labelXOffset = "${xOffset}"; │ │ │ │ │ - this.labelSymbolizer.labelYOffset = "${yOffset}"; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - if (this.gratLayer) { │ │ │ │ │ - this.gratLayer.destroy(); │ │ │ │ │ - this.gratLayer = null; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: draw │ │ │ │ │ - * │ │ │ │ │ - * initializes the graticule layer and does the initial update │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (!this.gratLayer) { │ │ │ │ │ - var gratStyle = new OpenLayers.Style({}, { │ │ │ │ │ - rules: [new OpenLayers.Rule({ │ │ │ │ │ - 'symbolizer': { │ │ │ │ │ - "Point": this.labelSymbolizer, │ │ │ │ │ - "Line": this.lineSymbolizer │ │ │ │ │ - } │ │ │ │ │ - })] │ │ │ │ │ - }); │ │ │ │ │ - this.gratLayer = new OpenLayers.Layer.Vector(this.layerName, { │ │ │ │ │ - styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ - 'default': gratStyle │ │ │ │ │ - }), │ │ │ │ │ - visibility: this.visible, │ │ │ │ │ - displayInLayerSwitcher: this.displayInLayerSwitcher │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - return this.div; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.map.addLayer(this.gratLayer); │ │ │ │ │ - this.map.events.register('moveend', this, this.update); │ │ │ │ │ - this.update(); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + for (var i in this.handlers) { │ │ │ │ │ + this.handlers[i].setMap(map); │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.map.events.unregister('moveend', this, this.update); │ │ │ │ │ - this.map.removeLayer(this.gratLayer); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - /** │ │ │ │ │ - * Method: update │ │ │ │ │ - * │ │ │ │ │ - * calculates the grid to be displayed and actually draws it │ │ │ │ │ + * Method: pixelToBounds │ │ │ │ │ + * Takes a pixel as argument and creates bounds after adding the │ │ │ │ │ + * <clickTolerance>. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pixel - {<OpenLayers.Pixel>} │ │ │ │ │ */ │ │ │ │ │ - update: function() { │ │ │ │ │ - //wait for the map to be initialized before proceeding │ │ │ │ │ - var mapBounds = this.map.getExtent(); │ │ │ │ │ - if (!mapBounds) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //clear out the old grid │ │ │ │ │ - this.gratLayer.destroyFeatures(); │ │ │ │ │ - │ │ │ │ │ - //get the projection objects required │ │ │ │ │ - var llProj = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - var mapProj = this.map.getProjectionObject(); │ │ │ │ │ - var mapRes = this.map.getResolution(); │ │ │ │ │ - │ │ │ │ │ - //if the map is in lon/lat, then the lines are straight and only one │ │ │ │ │ - //point is required │ │ │ │ │ - if (mapProj.proj && mapProj.proj.projName == "longlat") { │ │ │ │ │ - this.numPoints = 1; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //get the map center in EPSG:4326 │ │ │ │ │ - var mapCenter = this.map.getCenter(); //lon and lat here are really map x and y │ │ │ │ │ - var mapCenterLL = new OpenLayers.Pixel(mapCenter.lon, mapCenter.lat); │ │ │ │ │ - OpenLayers.Projection.transform(mapCenterLL, mapProj, llProj); │ │ │ │ │ - │ │ │ │ │ - /* This block of code determines the lon/lat interval to use for the │ │ │ │ │ - * grid by calculating the diagonal size of one grid cell at the map │ │ │ │ │ - * center. Iterates through the intervals array until the diagonal │ │ │ │ │ - * length is less than the targetSize option. │ │ │ │ │ - */ │ │ │ │ │ - //find lat/lon interval that results in a grid of less than the target size │ │ │ │ │ - var testSq = this.targetSize * mapRes; │ │ │ │ │ - testSq *= testSq; //compare squares rather than doing a square root to save time │ │ │ │ │ - var llInterval; │ │ │ │ │ - for (var i = 0; i < this.intervals.length; ++i) { │ │ │ │ │ - llInterval = this.intervals[i]; //could do this for both x and y?? │ │ │ │ │ - var delta = llInterval / 2; │ │ │ │ │ - var p1 = mapCenterLL.offset({ │ │ │ │ │ - x: -delta, │ │ │ │ │ - y: -delta │ │ │ │ │ - }); //test coords in EPSG:4326 space │ │ │ │ │ - var p2 = mapCenterLL.offset({ │ │ │ │ │ - x: delta, │ │ │ │ │ - y: delta │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Projection.transform(p1, llProj, mapProj); // convert them back to map projection │ │ │ │ │ - OpenLayers.Projection.transform(p2, llProj, mapProj); │ │ │ │ │ - var distSq = (p1.x - p2.x) * (p1.x - p2.x) + (p1.y - p2.y) * (p1.y - p2.y); │ │ │ │ │ - if (distSq <= testSq) { │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - //alert(llInterval); │ │ │ │ │ - │ │ │ │ │ - //round the LL center to an even number based on the interval │ │ │ │ │ - mapCenterLL.x = Math.floor(mapCenterLL.x / llInterval) * llInterval; │ │ │ │ │ - mapCenterLL.y = Math.floor(mapCenterLL.y / llInterval) * llInterval; │ │ │ │ │ - //TODO adjust for minutses/seconds? │ │ │ │ │ - │ │ │ │ │ - /* The following 2 blocks calculate the nodes of the grid along a │ │ │ │ │ - * line of constant longitude (then latitiude) running through the │ │ │ │ │ - * center of the map until it reaches the map edge. The calculation │ │ │ │ │ - * goes from the center in both directions to the edge. │ │ │ │ │ - */ │ │ │ │ │ - //get the central longitude line, increment the latitude │ │ │ │ │ - var iter = 0; │ │ │ │ │ - var centerLonPoints = [mapCenterLL.clone()]; │ │ │ │ │ - var newPoint = mapCenterLL.clone(); │ │ │ │ │ - var mapXY; │ │ │ │ │ - do { │ │ │ │ │ - newPoint = newPoint.offset({ │ │ │ │ │ - x: 0, │ │ │ │ │ - y: llInterval │ │ │ │ │ - }); │ │ │ │ │ - mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ - centerLonPoints.unshift(newPoint); │ │ │ │ │ - } while (mapBounds.containsPixel(mapXY) && ++iter < 1000); │ │ │ │ │ - newPoint = mapCenterLL.clone(); │ │ │ │ │ - do { │ │ │ │ │ - newPoint = newPoint.offset({ │ │ │ │ │ - x: 0, │ │ │ │ │ - y: -llInterval │ │ │ │ │ - }); │ │ │ │ │ - mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ - centerLonPoints.push(newPoint); │ │ │ │ │ - } while (mapBounds.containsPixel(mapXY) && ++iter < 1000); │ │ │ │ │ - │ │ │ │ │ - //get the central latitude line, increment the longitude │ │ │ │ │ - iter = 0; │ │ │ │ │ - var centerLatPoints = [mapCenterLL.clone()]; │ │ │ │ │ - newPoint = mapCenterLL.clone(); │ │ │ │ │ - do { │ │ │ │ │ - newPoint = newPoint.offset({ │ │ │ │ │ - x: -llInterval, │ │ │ │ │ - y: 0 │ │ │ │ │ - }); │ │ │ │ │ - mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ - centerLatPoints.unshift(newPoint); │ │ │ │ │ - } while (mapBounds.containsPixel(mapXY) && ++iter < 1000); │ │ │ │ │ - newPoint = mapCenterLL.clone(); │ │ │ │ │ - do { │ │ │ │ │ - newPoint = newPoint.offset({ │ │ │ │ │ - x: llInterval, │ │ │ │ │ - y: 0 │ │ │ │ │ - }); │ │ │ │ │ - mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ - centerLatPoints.push(newPoint); │ │ │ │ │ - } while (mapBounds.containsPixel(mapXY) && ++iter < 1000); │ │ │ │ │ - │ │ │ │ │ - //now generate a line for each node in the central lat and lon lines │ │ │ │ │ - //first loop over constant longitude │ │ │ │ │ - var lines = []; │ │ │ │ │ - for (var i = 0; i < centerLatPoints.length; ++i) { │ │ │ │ │ - var lon = centerLatPoints[i].x; │ │ │ │ │ - var pointList = []; │ │ │ │ │ - var labelPoint = null; │ │ │ │ │ - var latEnd = Math.min(centerLonPoints[0].y, 90); │ │ │ │ │ - var latStart = Math.max(centerLonPoints[centerLonPoints.length - 1].y, -90); │ │ │ │ │ - var latDelta = (latEnd - latStart) / this.numPoints; │ │ │ │ │ - var lat = latStart; │ │ │ │ │ - for (var j = 0; j <= this.numPoints; ++j) { │ │ │ │ │ - var gridPoint = new OpenLayers.Geometry.Point(lon, lat); │ │ │ │ │ - gridPoint.transform(llProj, mapProj); │ │ │ │ │ - pointList.push(gridPoint); │ │ │ │ │ - lat += latDelta; │ │ │ │ │ - if (gridPoint.y >= mapBounds.bottom && !labelPoint) { │ │ │ │ │ - labelPoint = gridPoint; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.labelled) { │ │ │ │ │ - //keep track of when this grid line crosses the map bounds to set │ │ │ │ │ - //the label position │ │ │ │ │ - //labels along the bottom, add 10 pixel offset up into the map │ │ │ │ │ - //TODO add option for labels on top │ │ │ │ │ - var labelPos = new OpenLayers.Geometry.Point(labelPoint.x, mapBounds.bottom); │ │ │ │ │ - var labelAttrs = { │ │ │ │ │ - value: lon, │ │ │ │ │ - label: this.labelled ? OpenLayers.Util.getFormattedLonLat(lon, "lon", this.labelFormat) : "", │ │ │ │ │ - labelAlign: "cb", │ │ │ │ │ - xOffset: 0, │ │ │ │ │ - yOffset: 2 │ │ │ │ │ - }; │ │ │ │ │ - this.gratLayer.addFeatures(new OpenLayers.Feature.Vector(labelPos, labelAttrs)); │ │ │ │ │ - } │ │ │ │ │ - var geom = new OpenLayers.Geometry.LineString(pointList); │ │ │ │ │ - lines.push(new OpenLayers.Feature.Vector(geom)); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //now draw the lines of constant latitude │ │ │ │ │ - for (var j = 0; j < centerLonPoints.length; ++j) { │ │ │ │ │ - lat = centerLonPoints[j].y; │ │ │ │ │ - if (lat < -90 || lat > 90) { //latitudes only valid between -90 and 90 │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - var pointList = []; │ │ │ │ │ - var lonStart = centerLatPoints[0].x; │ │ │ │ │ - var lonEnd = centerLatPoints[centerLatPoints.length - 1].x; │ │ │ │ │ - var lonDelta = (lonEnd - lonStart) / this.numPoints; │ │ │ │ │ - var lon = lonStart; │ │ │ │ │ - var labelPoint = null; │ │ │ │ │ - for (var i = 0; i <= this.numPoints; ++i) { │ │ │ │ │ - var gridPoint = new OpenLayers.Geometry.Point(lon, lat); │ │ │ │ │ - gridPoint.transform(llProj, mapProj); │ │ │ │ │ - pointList.push(gridPoint); │ │ │ │ │ - lon += lonDelta; │ │ │ │ │ - if (gridPoint.x < mapBounds.right) { │ │ │ │ │ - labelPoint = gridPoint; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.labelled) { │ │ │ │ │ - //keep track of when this grid line crosses the map bounds to set │ │ │ │ │ - //the label position │ │ │ │ │ - //labels along the right, 30 pixel offset left into the map │ │ │ │ │ - //TODO add option for labels on left │ │ │ │ │ - var labelPos = new OpenLayers.Geometry.Point(mapBounds.right, labelPoint.y); │ │ │ │ │ - var labelAttrs = { │ │ │ │ │ - value: lat, │ │ │ │ │ - label: this.labelled ? OpenLayers.Util.getFormattedLonLat(lat, "lat", this.labelFormat) : "", │ │ │ │ │ - labelAlign: "rb", │ │ │ │ │ - xOffset: -2, │ │ │ │ │ - yOffset: 2 │ │ │ │ │ - }; │ │ │ │ │ - this.gratLayer.addFeatures(new OpenLayers.Feature.Vector(labelPos, labelAttrs)); │ │ │ │ │ - } │ │ │ │ │ - var geom = new OpenLayers.Geometry.LineString(pointList); │ │ │ │ │ - lines.push(new OpenLayers.Feature.Vector(geom)); │ │ │ │ │ - } │ │ │ │ │ - this.gratLayer.addFeatures(lines); │ │ │ │ │ + pixelToBounds: function(pixel) { │ │ │ │ │ + var llPx = pixel.add(-this.clickTolerance / 2, this.clickTolerance / 2); │ │ │ │ │ + var urPx = pixel.add(this.clickTolerance / 2, -this.clickTolerance / 2); │ │ │ │ │ + var ll = this.map.getLonLatFromPixel(llPx); │ │ │ │ │ + var ur = this.map.getLonLatFromPixel(urPx); │ │ │ │ │ + return new OpenLayers.Bounds(ll.lon, ll.lat, ur.lon, ur.lat); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Graticule" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.GetFeature" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Measure.js │ │ │ │ │ + OpenLayers/Control/PanZoomBar.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Control/PanZoom.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.Measure │ │ │ │ │ - * Allows for drawing of features for measurements. │ │ │ │ │ + * Class: OpenLayers.Control.PanZoomBar │ │ │ │ │ + * The PanZoomBar is a visible control composed of a │ │ │ │ │ + * <OpenLayers.Control.PanPanel> and a <OpenLayers.Control.ZoomBar>. │ │ │ │ │ + * By default it is displayed in the upper left corner of the map as 4 │ │ │ │ │ + * directional arrows above a vertical slider. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Control> │ │ │ │ │ + * - <OpenLayers.Control.PanZoom> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.Measure = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: events │ │ │ │ │ - * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ - * control specific events. │ │ │ │ │ - * │ │ │ │ │ - * Register a listener for a particular event with the following syntax: │ │ │ │ │ - * (code) │ │ │ │ │ - * control.events.register(type, obj, listener); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * measure - Triggered when a measurement sketch is complete. Listeners │ │ │ │ │ - * will receive an event with measure, units, order, and geometry │ │ │ │ │ - * properties. │ │ │ │ │ - * measurepartial - Triggered when a new point is added to the │ │ │ │ │ - * measurement sketch or if the <immediate> property is true and the │ │ │ │ │ - * measurement sketch is modified. Listeners receive an event with measure, │ │ │ │ │ - * units, order, and geometry. │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: handlerOptions │ │ │ │ │ - * {Object} Used to set non-default properties on the control's handler │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: callbacks │ │ │ │ │ - * {Object} The functions that are sent to the handler for callback │ │ │ │ │ - */ │ │ │ │ │ - callbacks: null, │ │ │ │ │ +OpenLayers.Control.PanZoomBar = OpenLayers.Class(OpenLayers.Control.PanZoom, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: displaySystem │ │ │ │ │ - * {String} Display system for output measurements. Supported values │ │ │ │ │ - * are 'english', 'metric', and 'geographic'. Default is 'metric'. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomStopWidth │ │ │ │ │ */ │ │ │ │ │ - displaySystem: 'metric', │ │ │ │ │ + zoomStopWidth: 18, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: geodesic │ │ │ │ │ - * {Boolean} Calculate geodesic metrics instead of planar metrics. This │ │ │ │ │ - * requires that geometries can be transformed into Geographic/WGS84 │ │ │ │ │ - * (if that is not already the map projection). Default is false. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomStopHeight │ │ │ │ │ */ │ │ │ │ │ - geodesic: false, │ │ │ │ │ + zoomStopHeight: 11, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: displaySystemUnits │ │ │ │ │ - * {Object} Units for various measurement systems. Values are arrays │ │ │ │ │ - * of unit abbreviations (from OpenLayers.INCHES_PER_UNIT) in decreasing │ │ │ │ │ - * order of length. │ │ │ │ │ + /** │ │ │ │ │ + * Property: slider │ │ │ │ │ */ │ │ │ │ │ - displaySystemUnits: { │ │ │ │ │ - geographic: ['dd'], │ │ │ │ │ - english: ['mi', 'ft', 'in'], │ │ │ │ │ - metric: ['km', 'm'] │ │ │ │ │ - }, │ │ │ │ │ + slider: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: delay │ │ │ │ │ - * {Number} Number of milliseconds between clicks before the event is │ │ │ │ │ - * considered a double-click. The "measurepartial" event will not │ │ │ │ │ - * be triggered if the sketch is completed within this time. This │ │ │ │ │ - * is required for IE where creating a browser reflow (if a listener │ │ │ │ │ - * is modifying the DOM by displaying the measurement values) messes │ │ │ │ │ - * with the dblclick listener in the sketch handler. │ │ │ │ │ + /** │ │ │ │ │ + * Property: sliderEvents │ │ │ │ │ + * {<OpenLayers.Events>} │ │ │ │ │ */ │ │ │ │ │ - partialDelay: 300, │ │ │ │ │ + sliderEvents: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: delayedTrigger │ │ │ │ │ - * {Number} Timeout id of trigger for measurepartial. │ │ │ │ │ + /** │ │ │ │ │ + * Property: zoombarDiv │ │ │ │ │ + * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - delayedTrigger: null, │ │ │ │ │ + zoombarDiv: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: persist │ │ │ │ │ - * {Boolean} Keep the temporary measurement sketch drawn after the │ │ │ │ │ - * measurement is complete. The geometry will persist until a new │ │ │ │ │ - * measurement is started, the control is deactivated, or <cancel> is │ │ │ │ │ - * called. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomWorldIcon │ │ │ │ │ + * {Boolean} │ │ │ │ │ */ │ │ │ │ │ - persist: false, │ │ │ │ │ + zoomWorldIcon: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: immediate │ │ │ │ │ - * {Boolean} Activates the immediate measurement so that the "measurepartial" │ │ │ │ │ - * event is also fired once the measurement sketch is modified. │ │ │ │ │ - * Default is false. │ │ │ │ │ + * APIProperty: panIcons │ │ │ │ │ + * {Boolean} Set this property to false not to display the pan icons. If │ │ │ │ │ + * false the zoom world icon is placed under the zoom bar. Defaults to │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ - immediate: false, │ │ │ │ │ + panIcons: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Measure │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * handler - {<OpenLayers.Handler>} │ │ │ │ │ - * options - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(handler, options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - var callbacks = { │ │ │ │ │ - done: this.measureComplete, │ │ │ │ │ - point: this.measurePartial │ │ │ │ │ - }; │ │ │ │ │ - if (this.immediate) { │ │ │ │ │ - callbacks.modify = this.measureImmediate; │ │ │ │ │ - } │ │ │ │ │ - this.callbacks = OpenLayers.Util.extend(callbacks, this.callbacks); │ │ │ │ │ + * APIProperty: forceFixedZoomLevel │ │ │ │ │ + * {Boolean} Force a fixed zoom level even though the map has │ │ │ │ │ + * fractionalZoom │ │ │ │ │ + */ │ │ │ │ │ + forceFixedZoomLevel: false, │ │ │ │ │ │ │ │ │ │ - // let the handler options override, so old code that passes 'persist' │ │ │ │ │ - // directly to the handler does not need an update │ │ │ │ │ - this.handlerOptions = OpenLayers.Util.extend({ │ │ │ │ │ - persist: this.persist │ │ │ │ │ - }, this.handlerOptions); │ │ │ │ │ - this.handler = new handler(this, this.callbacks, this.handlerOptions); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: mouseDragStart │ │ │ │ │ + * {<OpenLayers.Pixel>} │ │ │ │ │ + */ │ │ │ │ │ + mouseDragStart: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ + * Property: deltaY │ │ │ │ │ + * {Number} The cumulative vertical pixel offset during a zoom bar drag. │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - this.cancelDelay(); │ │ │ │ │ - return OpenLayers.Control.prototype.deactivate.apply(this, arguments); │ │ │ │ │ - }, │ │ │ │ │ + deltaY: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: cancel │ │ │ │ │ - * Stop the control from measuring. If <persist> is true, the temporary │ │ │ │ │ - * sketch will be erased. │ │ │ │ │ + * Property: zoomStart │ │ │ │ │ + * {<OpenLayers.Pixel>} │ │ │ │ │ */ │ │ │ │ │ - cancel: function() { │ │ │ │ │ - this.cancelDelay(); │ │ │ │ │ - this.handler.cancel(); │ │ │ │ │ - }, │ │ │ │ │ + zoomStart: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setImmediate │ │ │ │ │ - * Sets the <immediate> property. Changes the activity of immediate │ │ │ │ │ - * measurement. │ │ │ │ │ + * Constructor: OpenLayers.Control.PanZoomBar │ │ │ │ │ */ │ │ │ │ │ - setImmediate: function(immediate) { │ │ │ │ │ - this.immediate = immediate; │ │ │ │ │ - if (this.immediate) { │ │ │ │ │ - this.callbacks.modify = this.measureImmediate; │ │ │ │ │ - } else { │ │ │ │ │ - delete this.callbacks.modify; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: updateHandler │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * handler - {Function} One of the sketch handler constructors. │ │ │ │ │ - * options - {Object} Options for the handler. │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ */ │ │ │ │ │ - updateHandler: function(handler, options) { │ │ │ │ │ - var active = this.active; │ │ │ │ │ - if (active) { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - } │ │ │ │ │ - this.handler = new handler(this, this.callbacks, options); │ │ │ │ │ - if (active) { │ │ │ │ │ - this.activate(); │ │ │ │ │ - } │ │ │ │ │ + destroy: function() { │ │ │ │ │ + │ │ │ │ │ + this._removeZoomBar(); │ │ │ │ │ + │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + "changebaselayer": this.redraw, │ │ │ │ │ + "updatesize": this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + OpenLayers.Control.PanZoom.prototype.destroy.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + delete this.mouseDragStart; │ │ │ │ │ + delete this.zoomStart; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: measureComplete │ │ │ │ │ - * Called when the measurement sketch is done. │ │ │ │ │ - * │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - measureComplete: function(geometry) { │ │ │ │ │ - this.cancelDelay(); │ │ │ │ │ - this.measure(geometry, "measure"); │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.PanZoom.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + "changebaselayer": this.redraw, │ │ │ │ │ + "updatesize": this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: measurePartial │ │ │ │ │ - * Called each time a new point is added to the measurement sketch. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} The last point added. │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} The sketch geometry. │ │ │ │ │ + /** │ │ │ │ │ + * Method: redraw │ │ │ │ │ + * clear the div and start over. │ │ │ │ │ */ │ │ │ │ │ - measurePartial: function(point, geometry) { │ │ │ │ │ - this.cancelDelay(); │ │ │ │ │ - geometry = geometry.clone(); │ │ │ │ │ - // when we're wating for a dblclick, we have to trigger measurepartial │ │ │ │ │ - // after some delay to deal with reflow issues in IE │ │ │ │ │ - if (this.handler.freehandMode(this.handler.evt)) { │ │ │ │ │ - // no dblclick in freehand mode │ │ │ │ │ - this.measure(geometry, "measurepartial"); │ │ │ │ │ - } else { │ │ │ │ │ - this.delayedTrigger = window.setTimeout( │ │ │ │ │ - OpenLayers.Function.bind(function() { │ │ │ │ │ - this.delayedTrigger = null; │ │ │ │ │ - this.measure(geometry, "measurepartial"); │ │ │ │ │ - }, this), │ │ │ │ │ - this.partialDelay │ │ │ │ │ - ); │ │ │ │ │ + redraw: function() { │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.removeButtons(); │ │ │ │ │ + this._removeZoomBar(); │ │ │ │ │ } │ │ │ │ │ + this.draw(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: measureImmediate │ │ │ │ │ - * Called each time the measurement sketch is modified. │ │ │ │ │ + * Method: draw │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} The point at the mouse position. │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} The sketch feature. │ │ │ │ │ - * drawing - {Boolean} Indicates whether we're currently drawing. │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ */ │ │ │ │ │ - measureImmediate: function(point, feature, drawing) { │ │ │ │ │ - if (drawing && !this.handler.freehandMode(this.handler.evt)) { │ │ │ │ │ - this.cancelDelay(); │ │ │ │ │ - this.measure(feature.geometry, "measurepartial"); │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + // initialize our internal div │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + px = this.position.clone(); │ │ │ │ │ + │ │ │ │ │ + // place the controls │ │ │ │ │ + this.buttons = []; │ │ │ │ │ + │ │ │ │ │ + var sz = { │ │ │ │ │ + w: 18, │ │ │ │ │ + h: 18 │ │ │ │ │ + }; │ │ │ │ │ + if (this.panIcons) { │ │ │ │ │ + var centered = new OpenLayers.Pixel(px.x + sz.w / 2, px.y); │ │ │ │ │ + var wposition = sz.w; │ │ │ │ │ + │ │ │ │ │ + if (this.zoomWorldIcon) { │ │ │ │ │ + centered = new OpenLayers.Pixel(px.x + sz.w, px.y); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this._addButton("panup", "north-mini.png", centered, sz); │ │ │ │ │ + px.y = centered.y + sz.h; │ │ │ │ │ + this._addButton("panleft", "west-mini.png", px, sz); │ │ │ │ │ + if (this.zoomWorldIcon) { │ │ │ │ │ + this._addButton("zoomworld", "zoom-world-mini.png", px.add(sz.w, 0), sz); │ │ │ │ │ + │ │ │ │ │ + wposition *= 2; │ │ │ │ │ + } │ │ │ │ │ + this._addButton("panright", "east-mini.png", px.add(wposition, 0), sz); │ │ │ │ │ + this._addButton("pandown", "south-mini.png", centered.add(0, sz.h * 2), sz); │ │ │ │ │ + this._addButton("zoomin", "zoom-plus-mini.png", centered.add(0, sz.h * 3 + 5), sz); │ │ │ │ │ + centered = this._addZoomBar(centered.add(0, sz.h * 4 + 5)); │ │ │ │ │ + this._addButton("zoomout", "zoom-minus-mini.png", centered, sz); │ │ │ │ │ + } else { │ │ │ │ │ + this._addButton("zoomin", "zoom-plus-mini.png", px, sz); │ │ │ │ │ + centered = this._addZoomBar(px.add(0, sz.h)); │ │ │ │ │ + this._addButton("zoomout", "zoom-minus-mini.png", centered, sz); │ │ │ │ │ + if (this.zoomWorldIcon) { │ │ │ │ │ + centered = centered.add(0, sz.h + 3); │ │ │ │ │ + this._addButton("zoomworld", "zoom-world-mini.png", centered, sz); │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: cancelDelay │ │ │ │ │ - * Cancels the delay measurement that measurePartial began. │ │ │ │ │ + /** │ │ │ │ │ + * Method: _addZoomBar │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * centered - {<OpenLayers.Pixel>} where zoombar drawing is to start. │ │ │ │ │ */ │ │ │ │ │ - cancelDelay: function() { │ │ │ │ │ - if (this.delayedTrigger !== null) { │ │ │ │ │ - window.clearTimeout(this.delayedTrigger); │ │ │ │ │ - this.delayedTrigger = null; │ │ │ │ │ + _addZoomBar: function(centered) { │ │ │ │ │ + var imgLocation = OpenLayers.Util.getImageLocation("slider.png"); │ │ │ │ │ + var id = this.id + "_" + this.map.id; │ │ │ │ │ + var minZoom = this.map.getMinZoom(); │ │ │ │ │ + var zoomsToEnd = this.map.getNumZoomLevels() - 1 - this.map.getZoom(); │ │ │ │ │ + var slider = OpenLayers.Util.createAlphaImageDiv(id, │ │ │ │ │ + centered.add(-1, zoomsToEnd * this.zoomStopHeight), { │ │ │ │ │ + w: 20, │ │ │ │ │ + h: 9 │ │ │ │ │ + }, │ │ │ │ │ + imgLocation, │ │ │ │ │ + "absolute"); │ │ │ │ │ + slider.style.cursor = "move"; │ │ │ │ │ + this.slider = slider; │ │ │ │ │ + │ │ │ │ │ + this.sliderEvents = new OpenLayers.Events(this, slider, null, true, { │ │ │ │ │ + includeXY: true │ │ │ │ │ + }); │ │ │ │ │ + this.sliderEvents.on({ │ │ │ │ │ + "touchstart": this.zoomBarDown, │ │ │ │ │ + "touchmove": this.zoomBarDrag, │ │ │ │ │ + "touchend": this.zoomBarUp, │ │ │ │ │ + "mousedown": this.zoomBarDown, │ │ │ │ │ + "mousemove": this.zoomBarDrag, │ │ │ │ │ + "mouseup": this.zoomBarUp │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + var sz = { │ │ │ │ │ + w: this.zoomStopWidth, │ │ │ │ │ + h: this.zoomStopHeight * (this.map.getNumZoomLevels() - minZoom) │ │ │ │ │ + }; │ │ │ │ │ + var imgLocation = OpenLayers.Util.getImageLocation("zoombar.png"); │ │ │ │ │ + var div = null; │ │ │ │ │ + │ │ │ │ │ + if (OpenLayers.Util.alphaHack()) { │ │ │ │ │ + var id = this.id + "_" + this.map.id; │ │ │ │ │ + div = OpenLayers.Util.createAlphaImageDiv(id, centered, { │ │ │ │ │ + w: sz.w, │ │ │ │ │ + h: this.zoomStopHeight │ │ │ │ │ + }, │ │ │ │ │ + imgLocation, │ │ │ │ │ + "absolute", null, "crop"); │ │ │ │ │ + div.style.height = sz.h + "px"; │ │ │ │ │ + } else { │ │ │ │ │ + div = OpenLayers.Util.createDiv( │ │ │ │ │ + 'OpenLayers_Control_PanZoomBar_Zoombar' + this.map.id, │ │ │ │ │ + centered, │ │ │ │ │ + sz, │ │ │ │ │ + imgLocation); │ │ │ │ │ } │ │ │ │ │ + div.style.cursor = "pointer"; │ │ │ │ │ + div.className = "olButton"; │ │ │ │ │ + this.zoombarDiv = div; │ │ │ │ │ + │ │ │ │ │ + this.div.appendChild(div); │ │ │ │ │ + │ │ │ │ │ + this.startTop = parseInt(div.style.top); │ │ │ │ │ + this.div.appendChild(slider); │ │ │ │ │ + │ │ │ │ │ + this.map.events.register("zoomend", this, this.moveZoomBar); │ │ │ │ │ + │ │ │ │ │ + centered = centered.add(0, │ │ │ │ │ + this.zoomStopHeight * (this.map.getNumZoomLevels() - minZoom)); │ │ │ │ │ + return centered; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: measure │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * eventType - {String} │ │ │ │ │ + * Method: _removeZoomBar │ │ │ │ │ */ │ │ │ │ │ - measure: function(geometry, eventType) { │ │ │ │ │ - var stat, order; │ │ │ │ │ - if (geometry.CLASS_NAME.indexOf('LineString') > -1) { │ │ │ │ │ - stat = this.getBestLength(geometry); │ │ │ │ │ - order = 1; │ │ │ │ │ - } else { │ │ │ │ │ - stat = this.getBestArea(geometry); │ │ │ │ │ - order = 2; │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent(eventType, { │ │ │ │ │ - measure: stat[0], │ │ │ │ │ - units: stat[1], │ │ │ │ │ - order: order, │ │ │ │ │ - geometry: geometry │ │ │ │ │ + _removeZoomBar: function() { │ │ │ │ │ + this.sliderEvents.un({ │ │ │ │ │ + "touchstart": this.zoomBarDown, │ │ │ │ │ + "touchmove": this.zoomBarDrag, │ │ │ │ │ + "touchend": this.zoomBarUp, │ │ │ │ │ + "mousedown": this.zoomBarDown, │ │ │ │ │ + "mousemove": this.zoomBarDrag, │ │ │ │ │ + "mouseup": this.zoomBarUp │ │ │ │ │ }); │ │ │ │ │ + this.sliderEvents.destroy(); │ │ │ │ │ + │ │ │ │ │ + this.div.removeChild(this.zoombarDiv); │ │ │ │ │ + this.zoombarDiv = null; │ │ │ │ │ + this.div.removeChild(this.slider); │ │ │ │ │ + this.slider = null; │ │ │ │ │ + │ │ │ │ │ + this.map.events.unregister("zoomend", this, this.moveZoomBar); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getBestArea │ │ │ │ │ - * Based on the <displaySystem> returns the area of a geometry. │ │ │ │ │ + * Method: onButtonClick │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array([Float, String])} Returns a two item array containing the │ │ │ │ │ - * area and the units abbreviation. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - getBestArea: function(geometry) { │ │ │ │ │ - var units = this.displaySystemUnits[this.displaySystem]; │ │ │ │ │ - var unit, area; │ │ │ │ │ - for (var i = 0, len = units.length; i < len; ++i) { │ │ │ │ │ - unit = units[i]; │ │ │ │ │ - area = this.getArea(geometry, unit); │ │ │ │ │ - if (area > 1) { │ │ │ │ │ - break; │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + OpenLayers.Control.PanZoom.prototype.onButtonClick.apply(this, arguments); │ │ │ │ │ + if (evt.buttonElement === this.zoombarDiv) { │ │ │ │ │ + var levels = evt.buttonXY.y / this.zoomStopHeight; │ │ │ │ │ + if (this.forceFixedZoomLevel || !this.map.fractionalZoom) { │ │ │ │ │ + levels = Math.floor(levels); │ │ │ │ │ } │ │ │ │ │ + var zoom = (this.map.getNumZoomLevels() - 1) - levels; │ │ │ │ │ + zoom = Math.min(Math.max(zoom, 0), this.map.getNumZoomLevels() - 1); │ │ │ │ │ + this.map.zoomTo(zoom); │ │ │ │ │ } │ │ │ │ │ - return [area, unit]; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getArea │ │ │ │ │ + * Method: passEventToSlider │ │ │ │ │ + * This function is used to pass events that happen on the div, or the map, │ │ │ │ │ + * through to the slider, which then does its moving thing. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * units - {String} Unit abbreviation │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ + */ │ │ │ │ │ + passEventToSlider: function(evt) { │ │ │ │ │ + this.sliderEvents.handleBrowserEvent(evt); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /* │ │ │ │ │ + * Method: zoomBarDown │ │ │ │ │ + * event listener for clicks on the slider │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The geometry area in the given units. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - getArea: function(geometry, units) { │ │ │ │ │ - var area, geomUnits; │ │ │ │ │ - if (this.geodesic) { │ │ │ │ │ - area = geometry.getGeodesicArea(this.map.getProjectionObject()); │ │ │ │ │ - geomUnits = "m"; │ │ │ │ │ - } else { │ │ │ │ │ - area = geometry.getArea(); │ │ │ │ │ - geomUnits = this.map.getUnits(); │ │ │ │ │ - } │ │ │ │ │ - var inPerDisplayUnit = OpenLayers.INCHES_PER_UNIT[units]; │ │ │ │ │ - if (inPerDisplayUnit) { │ │ │ │ │ - var inPerMapUnit = OpenLayers.INCHES_PER_UNIT[geomUnits]; │ │ │ │ │ - area *= Math.pow((inPerMapUnit / inPerDisplayUnit), 2); │ │ │ │ │ + zoomBarDown: function(evt) { │ │ │ │ │ + if (!OpenLayers.Event.isLeftClick(evt) && !OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ - return area; │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + "touchmove": this.passEventToSlider, │ │ │ │ │ + "mousemove": this.passEventToSlider, │ │ │ │ │ + "mouseup": this.passEventToSlider, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.mouseDragStart = evt.xy.clone(); │ │ │ │ │ + this.zoomStart = evt.xy.clone(); │ │ │ │ │ + this.div.style.cursor = "move"; │ │ │ │ │ + // reset the div offsets just in case the div moved │ │ │ │ │ + this.zoombarDiv.offsets = null; │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getBestLength │ │ │ │ │ - * Based on the <displaySystem> returns the length of a geometry. │ │ │ │ │ + /* │ │ │ │ │ + * Method: zoomBarDrag │ │ │ │ │ + * This is what happens when a click has occurred, and the client is │ │ │ │ │ + * dragging. Here we must ensure that the slider doesn't go beyond the │ │ │ │ │ + * bottom/top of the zoombar div, as well as moving the slider to its new │ │ │ │ │ + * visual location │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array([Float, String])} Returns a two item array containing the │ │ │ │ │ - * length and the units abbreviation. │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - getBestLength: function(geometry) { │ │ │ │ │ - var units = this.displaySystemUnits[this.displaySystem]; │ │ │ │ │ - var unit, length; │ │ │ │ │ - for (var i = 0, len = units.length; i < len; ++i) { │ │ │ │ │ - unit = units[i]; │ │ │ │ │ - length = this.getLength(geometry, unit); │ │ │ │ │ - if (length > 1) { │ │ │ │ │ - break; │ │ │ │ │ + zoomBarDrag: function(evt) { │ │ │ │ │ + if (this.mouseDragStart != null) { │ │ │ │ │ + var deltaY = this.mouseDragStart.y - evt.xy.y; │ │ │ │ │ + var offsets = OpenLayers.Util.pagePosition(this.zoombarDiv); │ │ │ │ │ + if ((evt.clientY - offsets[1]) > 0 && │ │ │ │ │ + (evt.clientY - offsets[1]) < parseInt(this.zoombarDiv.style.height) - 2) { │ │ │ │ │ + var newTop = parseInt(this.slider.style.top) - deltaY; │ │ │ │ │ + this.slider.style.top = newTop + "px"; │ │ │ │ │ + this.mouseDragStart = evt.xy.clone(); │ │ │ │ │ } │ │ │ │ │ + // set cumulative displacement │ │ │ │ │ + this.deltaY = this.zoomStart.y - evt.xy.y; │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ } │ │ │ │ │ - return [length, unit]; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getLength │ │ │ │ │ + /* │ │ │ │ │ + * Method: zoomBarUp │ │ │ │ │ + * Perform cleanup when a mouseup event is received -- discover new zoom │ │ │ │ │ + * level and switch to it. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * units - {String} Unit abbreviation │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Float} The geometry length in the given units. │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - getLength: function(geometry, units) { │ │ │ │ │ - var length, geomUnits; │ │ │ │ │ - if (this.geodesic) { │ │ │ │ │ - length = geometry.getGeodesicLength(this.map.getProjectionObject()); │ │ │ │ │ - geomUnits = "m"; │ │ │ │ │ - } else { │ │ │ │ │ - length = geometry.getLength(); │ │ │ │ │ - geomUnits = this.map.getUnits(); │ │ │ │ │ + zoomBarUp: function(evt) { │ │ │ │ │ + if (!OpenLayers.Event.isLeftClick(evt) && evt.type !== "touchend") { │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ - var inPerDisplayUnit = OpenLayers.INCHES_PER_UNIT[units]; │ │ │ │ │ - if (inPerDisplayUnit) { │ │ │ │ │ - var inPerMapUnit = OpenLayers.INCHES_PER_UNIT[geomUnits]; │ │ │ │ │ - length *= (inPerMapUnit / inPerDisplayUnit); │ │ │ │ │ + if (this.mouseDragStart) { │ │ │ │ │ + this.div.style.cursor = ""; │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + "touchmove": this.passEventToSlider, │ │ │ │ │ + "mouseup": this.passEventToSlider, │ │ │ │ │ + "mousemove": this.passEventToSlider, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + var zoomLevel = this.map.zoom; │ │ │ │ │ + if (!this.forceFixedZoomLevel && this.map.fractionalZoom) { │ │ │ │ │ + zoomLevel += this.deltaY / this.zoomStopHeight; │ │ │ │ │ + zoomLevel = Math.min(Math.max(zoomLevel, 0), │ │ │ │ │ + this.map.getNumZoomLevels() - 1); │ │ │ │ │ + } else { │ │ │ │ │ + zoomLevel += this.deltaY / this.zoomStopHeight; │ │ │ │ │ + zoomLevel = Math.max(Math.round(zoomLevel), 0); │ │ │ │ │ + } │ │ │ │ │ + this.map.zoomTo(zoomLevel); │ │ │ │ │ + this.mouseDragStart = null; │ │ │ │ │ + this.zoomStart = null; │ │ │ │ │ + this.deltaY = 0; │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ } │ │ │ │ │ - return length; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Measure" │ │ │ │ │ + /* │ │ │ │ │ + * Method: moveZoomBar │ │ │ │ │ + * Change the location of the slider to match the current zoom level. │ │ │ │ │ + */ │ │ │ │ │ + moveZoomBar: function() { │ │ │ │ │ + var newTop = │ │ │ │ │ + ((this.map.getNumZoomLevels() - 1) - this.map.getZoom()) * │ │ │ │ │ + this.zoomStopHeight + this.startTop + 1; │ │ │ │ │ + this.slider.style.top = newTop + "px"; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.PanZoomBar" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/Scale.js │ │ │ │ │ + OpenLayers/Control/ScaleLine.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.Scale │ │ │ │ │ - * The Scale control displays the current map scale as a ratio (e.g. Scale = │ │ │ │ │ - * 1:1M). By default it is displayed in the lower right corner of the map. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Control.ScaleLine │ │ │ │ │ + * The ScaleLine displays a small line indicator representing the current │ │ │ │ │ + * map scale on the map. By default it is drawn in the lower left corner of │ │ │ │ │ + * the map. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ + * │ │ │ │ │ + * Is a very close copy of: │ │ │ │ │ + * - <OpenLayers.Control.Scale> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.Scale = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Control.ScaleLine = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: element │ │ │ │ │ + * Property: maxWidth │ │ │ │ │ + * {Integer} Maximum width of the scale line in pixels. Default is 100. │ │ │ │ │ + */ │ │ │ │ │ + maxWidth: 100, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: topOutUnits │ │ │ │ │ + * {String} Units for zoomed out on top bar. Default is km. │ │ │ │ │ + */ │ │ │ │ │ + topOutUnits: "km", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: topInUnits │ │ │ │ │ + * {String} Units for zoomed in on top bar. Default is m. │ │ │ │ │ + */ │ │ │ │ │ + topInUnits: "m", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: bottomOutUnits │ │ │ │ │ + * {String} Units for zoomed out on bottom bar. Default is mi. │ │ │ │ │ + */ │ │ │ │ │ + bottomOutUnits: "mi", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: bottomInUnits │ │ │ │ │ + * {String} Units for zoomed in on bottom bar. Default is ft. │ │ │ │ │ + */ │ │ │ │ │ + bottomInUnits: "ft", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: eTop │ │ │ │ │ * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ - element: null, │ │ │ │ │ + eTop: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: eBottom │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + eBottom: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIProperty: geodesic │ │ │ │ │ * {Boolean} Use geodesic measurement. Default is false. The recommended │ │ │ │ │ * setting for maps in EPSG:4326 is false, and true EPSG:900913. If set to │ │ │ │ │ * true, the scale will be calculated based on the horizontal size of the │ │ │ │ │ * pixel in the center of the map viewport. │ │ │ │ │ */ │ │ │ │ │ geodesic: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.Scale │ │ │ │ │ + * Constructor: OpenLayers.Control.ScaleLine │ │ │ │ │ + * Create a new scale line control. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * element - {DOMElement} │ │ │ │ │ - * options - {Object} │ │ │ │ │ + * options - {Object} An optional object whose properties will be used │ │ │ │ │ + * to extend the control. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(element, options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.element = OpenLayers.Util.getElement(element); │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * Method: draw │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ * {DOMElement} │ │ │ │ │ */ │ │ │ │ │ draw: function() { │ │ │ │ │ OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (!this.element) { │ │ │ │ │ - this.element = document.createElement("div"); │ │ │ │ │ - this.div.appendChild(this.element); │ │ │ │ │ + if (!this.eTop) { │ │ │ │ │ + // stick in the top bar │ │ │ │ │ + this.eTop = document.createElement("div"); │ │ │ │ │ + this.eTop.className = this.displayClass + "Top"; │ │ │ │ │ + var theLen = this.topInUnits.length; │ │ │ │ │ + this.div.appendChild(this.eTop); │ │ │ │ │ + if ((this.topOutUnits == "") || (this.topInUnits == "")) { │ │ │ │ │ + this.eTop.style.visibility = "hidden"; │ │ │ │ │ + } else { │ │ │ │ │ + this.eTop.style.visibility = "visible"; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // and the bottom bar │ │ │ │ │ + this.eBottom = document.createElement("div"); │ │ │ │ │ + this.eBottom.className = this.displayClass + "Bottom"; │ │ │ │ │ + this.div.appendChild(this.eBottom); │ │ │ │ │ + if ((this.bottomOutUnits == "") || (this.bottomInUnits == "")) { │ │ │ │ │ + this.eBottom.style.visibility = "hidden"; │ │ │ │ │ + } else { │ │ │ │ │ + this.eBottom.style.visibility = "visible"; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.map.events.register('moveend', this, this.updateScale); │ │ │ │ │ - this.updateScale(); │ │ │ │ │ + this.map.events.register('moveend', this, this.update); │ │ │ │ │ + this.update(); │ │ │ │ │ return this.div; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: updateScale │ │ │ │ │ + /** │ │ │ │ │ + * Method: getBarLen │ │ │ │ │ + * Given a number, round it down to the nearest 1,2,5 times a power of 10. │ │ │ │ │ + * That seems a fairly useful set of number groups to use. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * maxLen - {float} the number we're rounding down from │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} the rounded number (less than or equal to maxLen) │ │ │ │ │ */ │ │ │ │ │ - updateScale: function() { │ │ │ │ │ - var scale; │ │ │ │ │ - if (this.geodesic === true) { │ │ │ │ │ - var units = this.map.getUnits(); │ │ │ │ │ - if (!units) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var inches = OpenLayers.INCHES_PER_UNIT; │ │ │ │ │ - scale = (this.map.getGeodesicPixelSize().w || 0.000001) * │ │ │ │ │ - inches["km"] * OpenLayers.DOTS_PER_INCH; │ │ │ │ │ + getBarLen: function(maxLen) { │ │ │ │ │ + // nearest power of 10 lower than maxLen │ │ │ │ │ + var digits = parseInt(Math.log(maxLen) / Math.log(10)); │ │ │ │ │ + var pow10 = Math.pow(10, digits); │ │ │ │ │ + │ │ │ │ │ + // ok, find first character │ │ │ │ │ + var firstChar = parseInt(maxLen / pow10); │ │ │ │ │ + │ │ │ │ │ + // right, put it into the correct bracket │ │ │ │ │ + var barLen; │ │ │ │ │ + if (firstChar > 5) { │ │ │ │ │ + barLen = 5; │ │ │ │ │ + } else if (firstChar > 2) { │ │ │ │ │ + barLen = 2; │ │ │ │ │ } else { │ │ │ │ │ - scale = this.map.getScale(); │ │ │ │ │ + barLen = 1; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - if (!scale) { │ │ │ │ │ + // scale it up the correct power of 10 │ │ │ │ │ + return barLen * pow10; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: update │ │ │ │ │ + * Update the size of the bars, and the labels they contain. │ │ │ │ │ + */ │ │ │ │ │ + update: function() { │ │ │ │ │ + var res = this.map.getResolution(); │ │ │ │ │ + if (!res) { │ │ │ │ │ return; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - if (scale >= 9500 && scale <= 950000) { │ │ │ │ │ - scale = Math.round(scale / 1000) + "K"; │ │ │ │ │ - } else if (scale >= 950000) { │ │ │ │ │ - scale = Math.round(scale / 1000000) + "M"; │ │ │ │ │ + var curMapUnits = this.map.getUnits(); │ │ │ │ │ + var inches = OpenLayers.INCHES_PER_UNIT; │ │ │ │ │ + │ │ │ │ │ + // convert maxWidth to map units │ │ │ │ │ + var maxSizeData = this.maxWidth * res * inches[curMapUnits]; │ │ │ │ │ + var geodesicRatio = 1; │ │ │ │ │ + if (this.geodesic === true) { │ │ │ │ │ + var maxSizeGeodesic = (this.map.getGeodesicPixelSize().w || │ │ │ │ │ + 0.000001) * this.maxWidth; │ │ │ │ │ + var maxSizeKilometers = maxSizeData / inches["km"]; │ │ │ │ │ + geodesicRatio = maxSizeGeodesic / maxSizeKilometers; │ │ │ │ │ + maxSizeData *= geodesicRatio; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // decide whether to use large or small scale units │ │ │ │ │ + var topUnits; │ │ │ │ │ + var bottomUnits; │ │ │ │ │ + if (maxSizeData > 100000) { │ │ │ │ │ + topUnits = this.topOutUnits; │ │ │ │ │ + bottomUnits = this.bottomOutUnits; │ │ │ │ │ } else { │ │ │ │ │ - scale = Math.round(scale); │ │ │ │ │ + topUnits = this.topInUnits; │ │ │ │ │ + bottomUnits = this.bottomInUnits; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // and to map units units │ │ │ │ │ + var topMax = maxSizeData / inches[topUnits]; │ │ │ │ │ + var bottomMax = maxSizeData / inches[bottomUnits]; │ │ │ │ │ + │ │ │ │ │ + // now trim this down to useful block length │ │ │ │ │ + var topRounded = this.getBarLen(topMax); │ │ │ │ │ + var bottomRounded = this.getBarLen(bottomMax); │ │ │ │ │ + │ │ │ │ │ + // and back to display units │ │ │ │ │ + topMax = topRounded / inches[curMapUnits] * inches[topUnits]; │ │ │ │ │ + bottomMax = bottomRounded / inches[curMapUnits] * inches[bottomUnits]; │ │ │ │ │ + │ │ │ │ │ + // and to pixel units │ │ │ │ │ + var topPx = topMax / res / geodesicRatio; │ │ │ │ │ + var bottomPx = bottomMax / res / geodesicRatio; │ │ │ │ │ + │ │ │ │ │ + // now set the pixel widths │ │ │ │ │ + // and the values inside them │ │ │ │ │ + │ │ │ │ │ + if (this.eBottom.style.visibility == "visible") { │ │ │ │ │ + this.eBottom.style.width = Math.round(bottomPx) + "px"; │ │ │ │ │ + this.eBottom.innerHTML = bottomRounded + " " + bottomUnits; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.eTop.style.visibility == "visible") { │ │ │ │ │ + this.eTop.style.width = Math.round(topPx) + "px"; │ │ │ │ │ + this.eTop.innerHTML = topRounded + " " + topUnits; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - this.element.innerHTML = OpenLayers.i18n("Scale = 1 : ${scaleDenom}", { │ │ │ │ │ - 'scaleDenom': scale │ │ │ │ │ - }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Scale" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ScaleLine" │ │ │ │ │ }); │ │ │ │ │ │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Control/SLDSelect.js │ │ │ │ │ + OpenLayers/Layer/Vector/RootContainer.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.Vector.RootContainer │ │ │ │ │ + * A special layer type to combine multiple vector layers inside a single │ │ │ │ │ + * renderer root container. This class is not supposed to be instantiated │ │ │ │ │ + * from user space, it is a helper class for controls that require event │ │ │ │ │ + * processing for multiple vector layers. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Vector> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Vector.RootContainer = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: displayInLayerSwitcher │ │ │ │ │ + * Set to false for this layer type │ │ │ │ │ + */ │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layers │ │ │ │ │ + * Layers that are attached to this container. Required config option. │ │ │ │ │ + */ │ │ │ │ │ + layers: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.Vector.RootContainer │ │ │ │ │ + * Create a new root container for multiple vector layer. This constructor │ │ │ │ │ + * is not supposed to be used from user space, it is only to be used by │ │ │ │ │ + * controls that need feature selection across multiple vector layers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} A name for the layer │ │ │ │ │ + * options - {Object} Optional object with non-default properties to set on │ │ │ │ │ + * the layer. │ │ │ │ │ + * │ │ │ │ │ + * Required options properties: │ │ │ │ │ + * layers - {Array(<OpenLayers.Layer.Vector>)} The layers managed by this │ │ │ │ │ + * container │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Vector.RootContainer>} A new vector layer root │ │ │ │ │ + * container │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: display │ │ │ │ │ + */ │ │ │ │ │ + display: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: getFeatureFromEvent │ │ │ │ │ + * walk through the layers to find the feature returned by the event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} event object with a feature property │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} │ │ │ │ │ + */ │ │ │ │ │ + getFeatureFromEvent: function(evt) { │ │ │ │ │ + var layers = this.layers; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0; i < layers.length; i++) { │ │ │ │ │ + feature = layers[i].getFeatureFromEvent(evt); │ │ │ │ │ + if (feature) { │ │ │ │ │ + return feature; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.collectRoots(); │ │ │ │ │ + map.events.register("changelayer", this, this.handleChangeLayer); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeMap │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + map.events.unregister("changelayer", this, this.handleChangeLayer); │ │ │ │ │ + this.resetRoots(); │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: collectRoots │ │ │ │ │ + * Collects the root nodes of all layers this control is configured with │ │ │ │ │ + * and moveswien the nodes to this control's layer │ │ │ │ │ + */ │ │ │ │ │ + collectRoots: function() { │ │ │ │ │ + var layer; │ │ │ │ │ + // walk through all map layers, because we want to keep the order │ │ │ │ │ + for (var i = 0; i < this.map.layers.length; ++i) { │ │ │ │ │ + layer = this.map.layers[i]; │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ + layer.renderer.moveRoot(this.renderer); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: resetRoots │ │ │ │ │ + * Resets the root nodes back into the layers they belong to. │ │ │ │ │ + */ │ │ │ │ │ + resetRoots: function() { │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = 0; i < this.layers.length; ++i) { │ │ │ │ │ + layer = this.layers[i]; │ │ │ │ │ + if (this.renderer && layer.renderer.getRenderLayerId() == this.id) { │ │ │ │ │ + this.renderer.moveRoot(layer.renderer); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleChangeLayer │ │ │ │ │ + * Event handler for the map's changelayer event. We need to rebuild │ │ │ │ │ + * this container's layer dom if order of one of its layers changes. │ │ │ │ │ + * This handler is added with the setMap method, and removed with the │ │ │ │ │ + * removeMap method. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Object} │ │ │ │ │ + */ │ │ │ │ │ + handleChangeLayer: function(evt) { │ │ │ │ │ + var layer = evt.layer; │ │ │ │ │ + if (evt.property == "order" && │ │ │ │ │ + OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ + this.resetRoots(); │ │ │ │ │ + this.collectRoots(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Vector.RootContainer" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/SelectFeature.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ * @requires OpenLayers/Control.js │ │ │ │ │ - * @requires OpenLayers/Layer/WMS.js │ │ │ │ │ - * @requires OpenLayers/Handler/RegularPolygon.js │ │ │ │ │ - * @requires OpenLayers/Handler/Polygon.js │ │ │ │ │ - * @requires OpenLayers/Handler/Path.js │ │ │ │ │ - * @requires OpenLayers/Handler/Click.js │ │ │ │ │ - * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ - * @requires OpenLayers/Format/SLD/v1_0_0.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ + * @requires OpenLayers/Handler/Feature.js │ │ │ │ │ + * @requires OpenLayers/Layer/Vector/RootContainer.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Control.SLDSelect │ │ │ │ │ - * Perform selections on WMS layers using Styled Layer Descriptor (SLD) │ │ │ │ │ + * Class: OpenLayers.Control.SelectFeature │ │ │ │ │ + * The SelectFeature control selects vector features from a given layer on │ │ │ │ │ + * click or hover. │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Control.SLDSelect = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ +OpenLayers.Control.SelectFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ * APIProperty: events │ │ │ │ │ * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ * control specific events. │ │ │ │ │ * │ │ │ │ │ * Register a listener for a particular event with the following syntax: │ │ │ │ │ * (code) │ │ │ │ │ * control.events.register(type, obj, listener); │ │ │ │ │ * (end) │ │ │ │ │ * │ │ │ │ │ * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ - * selected - Triggered when a selection occurs. Listeners receive an │ │ │ │ │ - * event with *filters* and *layer* properties. Filters will be an │ │ │ │ │ - * array of OpenLayers.Filter objects created in order to perform │ │ │ │ │ - * the particular selection. │ │ │ │ │ + * beforefeaturehighlighted - Triggered before a feature is highlighted │ │ │ │ │ + * featurehighlighted - Triggered when a feature is highlighted │ │ │ │ │ + * featureunhighlighted - Triggered when a feature is unhighlighted │ │ │ │ │ + * boxselectionstart - Triggered before box selection starts │ │ │ │ │ + * boxselectionend - Triggered after box selection ends │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: clearOnDeactivate │ │ │ │ │ - * {Boolean} Should the selection be cleared when the control is │ │ │ │ │ - * deactivated. Default value is false. │ │ │ │ │ + * Property: multipleKey │ │ │ │ │ + * {String} An event modifier ('altKey' or 'shiftKey') that temporarily sets │ │ │ │ │ + * the <multiple> property to true. Default is null. │ │ │ │ │ */ │ │ │ │ │ - clearOnDeactivate: false, │ │ │ │ │ + multipleKey: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: layers │ │ │ │ │ - * {Array(<OpenLayers.Layer.WMS>)} The WMS layers this control will work │ │ │ │ │ - * on. │ │ │ │ │ + * Property: toggleKey │ │ │ │ │ + * {String} An event modifier ('altKey' or 'shiftKey') that temporarily sets │ │ │ │ │ + * the <toggle> property to true. Default is null. │ │ │ │ │ */ │ │ │ │ │ - layers: null, │ │ │ │ │ + toggleKey: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: callbacks │ │ │ │ │ - * {Object} The functions that are sent to the handler for callback │ │ │ │ │ + * APIProperty: multiple │ │ │ │ │ + * {Boolean} Allow selection of multiple geometries. Default is false. │ │ │ │ │ */ │ │ │ │ │ - callbacks: null, │ │ │ │ │ + multiple: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: selectionSymbolizer │ │ │ │ │ - * {Object} Determines the styling of the selected objects. Default is │ │ │ │ │ - * a selection in red. │ │ │ │ │ + * APIProperty: clickout │ │ │ │ │ + * {Boolean} Unselect features when clicking outside any feature. │ │ │ │ │ + * Default is true. │ │ │ │ │ */ │ │ │ │ │ - selectionSymbolizer: { │ │ │ │ │ - 'Polygon': { │ │ │ │ │ - fillColor: '#FF0000', │ │ │ │ │ - stroke: false │ │ │ │ │ - }, │ │ │ │ │ - 'Line': { │ │ │ │ │ - strokeColor: '#FF0000', │ │ │ │ │ - strokeWidth: 2 │ │ │ │ │ - }, │ │ │ │ │ - 'Point': { │ │ │ │ │ - graphicName: 'square', │ │ │ │ │ - fillColor: '#FF0000', │ │ │ │ │ - pointRadius: 5 │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + clickout: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: layerOptions │ │ │ │ │ - * {Object} The options to apply to the selection layer, by default the │ │ │ │ │ - * selection layer will be kept out of the layer switcher. │ │ │ │ │ + * APIProperty: toggle │ │ │ │ │ + * {Boolean} Unselect a selected feature on click. Default is false. Only │ │ │ │ │ + * has meaning if hover is false. │ │ │ │ │ */ │ │ │ │ │ - layerOptions: null, │ │ │ │ │ + toggle: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: handlerOptions │ │ │ │ │ - * {Object} Used to set non-default properties on the control's handler │ │ │ │ │ + * APIProperty: hover │ │ │ │ │ + * {Boolean} Select on mouse over and deselect on mouse out. If true, this │ │ │ │ │ + * ignores clicks and only listens to mouse moves. │ │ │ │ │ */ │ │ │ │ │ + hover: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: sketchStyle │ │ │ │ │ - * {<OpenLayers.Style>|Object} Style or symbolizer to use for the sketch │ │ │ │ │ - * handler. The recommended way of styling the sketch layer, however, is │ │ │ │ │ - * to configure an <OpenLayers.StyleMap> in the layerOptions of the │ │ │ │ │ - * <handlerOptions>: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * new OpenLayers.Control.SLDSelect(OpenLayers.Handler.Path, { │ │ │ │ │ - * handlerOptions: { │ │ │ │ │ - * layerOptions: { │ │ │ │ │ - * styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ - * "default": {strokeColor: "yellow"} │ │ │ │ │ - * }) │ │ │ │ │ - * } │ │ │ │ │ - * } │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ + * APIProperty: highlightOnly │ │ │ │ │ + * {Boolean} If true do not actually select features (that is place them in │ │ │ │ │ + * the layer's selected features array), just highlight them. This property │ │ │ │ │ + * has no effect if hover is false. Defaults to false. │ │ │ │ │ */ │ │ │ │ │ - sketchStyle: null, │ │ │ │ │ + highlightOnly: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: wfsCache │ │ │ │ │ - * {Object} Cache to use for storing parsed results from │ │ │ │ │ - * <OpenLayers.Format.WFSDescribeFeatureType.read>. If not provided, │ │ │ │ │ - * these will be cached on the prototype. │ │ │ │ │ + * APIProperty: box │ │ │ │ │ + * {Boolean} Allow feature selection by drawing a box. │ │ │ │ │ */ │ │ │ │ │ - wfsCache: {}, │ │ │ │ │ + box: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: layerCache │ │ │ │ │ - * {Object} Cache to use for storing references to the selection layers. │ │ │ │ │ - * Normally each source layer will have exactly 1 selection layer of │ │ │ │ │ - * type OpenLayers.Layer.WMS. If not provided, layers will │ │ │ │ │ - * be cached on the prototype. Note that if <clearOnDeactivate> is │ │ │ │ │ - * true, the layer will no longer be cached after deactivating the │ │ │ │ │ - * control. │ │ │ │ │ + * Property: onBeforeSelect │ │ │ │ │ + * {Function} Optional function to be called before a feature is selected. │ │ │ │ │ + * The function should expect to be called with a feature. │ │ │ │ │ */ │ │ │ │ │ - layerCache: {}, │ │ │ │ │ + onBeforeSelect: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Control.SLDSelect │ │ │ │ │ - * Create a new control for selecting features in WMS layers using │ │ │ │ │ - * Styled Layer Descriptor (SLD). │ │ │ │ │ + * APIProperty: onSelect │ │ │ │ │ + * {Function} Optional function to be called when a feature is selected. │ │ │ │ │ + * The function should expect to be called with a feature. │ │ │ │ │ + */ │ │ │ │ │ + onSelect: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: onUnselect │ │ │ │ │ + * {Function} Optional function to be called when a feature is unselected. │ │ │ │ │ + * The function should expect to be called with a feature. │ │ │ │ │ + */ │ │ │ │ │ + onUnselect: function() {}, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: scope │ │ │ │ │ + * {Object} The scope to use with the onBeforeSelect, onSelect, onUnselect │ │ │ │ │ + * callbacks. If null the scope will be this control. │ │ │ │ │ + */ │ │ │ │ │ + scope: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: geometryTypes │ │ │ │ │ + * {Array(String)} To restrict selecting to a limited set of geometry types, │ │ │ │ │ + * send a list of strings corresponding to the geometry class names. │ │ │ │ │ + */ │ │ │ │ │ + geometryTypes: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: layer │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} The vector layer with a common renderer │ │ │ │ │ + * root for all layers this control is configured with (if an array of │ │ │ │ │ + * layers was passed to the constructor), or the vector layer the control │ │ │ │ │ + * was configured with (if a single layer was passed to the constructor). │ │ │ │ │ + */ │ │ │ │ │ + layer: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: layers │ │ │ │ │ + * {Array(<OpenLayers.Layer.Vector>)} The layers this control will work on, │ │ │ │ │ + * or null if the control was configured with a single layer │ │ │ │ │ + */ │ │ │ │ │ + layers: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: callbacks │ │ │ │ │ + * {Object} The functions that are sent to the handlers.feature for callback │ │ │ │ │ + */ │ │ │ │ │ + callbacks: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: selectStyle │ │ │ │ │ + * {Object} Hash of styles │ │ │ │ │ + */ │ │ │ │ │ + selectStyle: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: renderIntent │ │ │ │ │ + * {String} key used to retrieve the select style from the layer's │ │ │ │ │ + * style map. │ │ │ │ │ + */ │ │ │ │ │ + renderIntent: "select", │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: handlers │ │ │ │ │ + * {Object} Object with references to multiple <OpenLayers.Handler> │ │ │ │ │ + * instances. │ │ │ │ │ + */ │ │ │ │ │ + handlers: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.SelectFeature │ │ │ │ │ + * Create a new control for selecting features. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * handler - {<OpenLayers.Class>} A sketch handler class. This determines │ │ │ │ │ - * the type of selection, e.g. box (<OpenLayers.Handler.Box>), point │ │ │ │ │ - * (<OpenLayers.Handler.Point>), path (<OpenLayers.Handler.Path>) or │ │ │ │ │ - * polygon (<OpenLayers.Handler.Polygon>) selection. To use circle │ │ │ │ │ - * type selection, use <OpenLayers.Handler.RegularPolygon> and pass │ │ │ │ │ - * the number of desired sides (e.g. 40) as "sides" property to the │ │ │ │ │ - * <handlerOptions>. │ │ │ │ │ - * options - {Object} An object containing all configuration properties for │ │ │ │ │ - * the control. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * layers - Array({<OpenLayers.Layer.WMS>}) The layers to perform the │ │ │ │ │ - * selection on. │ │ │ │ │ + * layers - {<OpenLayers.Layer.Vector>}, or an array of vector layers. The │ │ │ │ │ + * layer(s) this control will select features from. │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(handler, options) { │ │ │ │ │ + initialize: function(layers, options) { │ │ │ │ │ OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ │ │ │ │ │ - this.callbacks = OpenLayers.Util.extend({ │ │ │ │ │ - done: this.select, │ │ │ │ │ - click: this.select │ │ │ │ │ - }, this.callbacks); │ │ │ │ │ - this.handlerOptions = this.handlerOptions || {}; │ │ │ │ │ - this.layerOptions = OpenLayers.Util.applyDefaults(this.layerOptions, { │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - tileOptions: { │ │ │ │ │ - maxGetUrlLength: 2048 │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (this.sketchStyle) { │ │ │ │ │ - this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.handlerOptions.layerOptions, { │ │ │ │ │ - styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ - "default": this.sketchStyle │ │ │ │ │ - }) │ │ │ │ │ + if (this.scope === null) { │ │ │ │ │ + this.scope = this; │ │ │ │ │ + } │ │ │ │ │ + this.initLayer(layers); │ │ │ │ │ + var callbacks = { │ │ │ │ │ + click: this.clickFeature, │ │ │ │ │ + clickout: this.clickoutFeature │ │ │ │ │ + }; │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + callbacks.over = this.overFeature; │ │ │ │ │ + callbacks.out = this.outFeature; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.callbacks = OpenLayers.Util.extend(callbacks, this.callbacks); │ │ │ │ │ + this.handlers = { │ │ │ │ │ + feature: new OpenLayers.Handler.Feature( │ │ │ │ │ + this, this.layer, this.callbacks, { │ │ │ │ │ + geometryTypes: this.geometryTypes │ │ │ │ │ + } │ │ │ │ │ + ) │ │ │ │ │ + }; │ │ │ │ │ + │ │ │ │ │ + if (this.box) { │ │ │ │ │ + this.handlers.box = new OpenLayers.Handler.Box( │ │ │ │ │ + this, { │ │ │ │ │ + done: this.selectBox │ │ │ │ │ + }, { │ │ │ │ │ + boxDivClassName: "olHandlerBoxSelectFeature" │ │ │ │ │ } │ │ │ │ │ ); │ │ │ │ │ } │ │ │ │ │ - this.handler = new handler(this, this.callbacks, this.handlerOptions); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Take care of things that are not handled in superclass. │ │ │ │ │ + * Method: initLayer │ │ │ │ │ + * Assign the layer property. If layers is an array, we need to use │ │ │ │ │ + * a RootContainer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layers - {<OpenLayers.Layer.Vector>}, or an array of vector layers. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var key in this.layerCache) { │ │ │ │ │ - delete this.layerCache[key]; │ │ │ │ │ - } │ │ │ │ │ - for (var key in this.wfsCache) { │ │ │ │ │ - delete this.wfsCache[key]; │ │ │ │ │ + initLayer: function(layers) { │ │ │ │ │ + if (OpenLayers.Util.isArray(layers)) { │ │ │ │ │ + this.layers = layers; │ │ │ │ │ + this.layer = new OpenLayers.Layer.Vector.RootContainer( │ │ │ │ │ + this.id + "_container", { │ │ │ │ │ + layers: layers │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + this.layer = layers; │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: coupleLayerVisiblity │ │ │ │ │ - * Couple the selection layer and the source layer with respect to │ │ │ │ │ - * layer visibility. So if the source layer is turned off, the │ │ │ │ │ - * selection layer is also turned off. │ │ │ │ │ - * │ │ │ │ │ - * Context: │ │ │ │ │ - * - {<OpenLayers.Layer>} │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ - coupleLayerVisiblity: function(evt) { │ │ │ │ │ - this.setVisibility(evt.object.getVisibility()); │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.active && this.layers) { │ │ │ │ │ + this.map.removeLayer(this.layer); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + if (this.layers) { │ │ │ │ │ + this.layer.destroy(); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createSelectionLayer │ │ │ │ │ - * Creates a "clone" from the source layer in which the selection can │ │ │ │ │ - * be drawn. This ensures both the source layer and the selection are │ │ │ │ │ - * visible and not only the selection. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * source - {<OpenLayers.Layer.WMS>} The source layer on which the selection │ │ │ │ │ - * is performed. │ │ │ │ │ - * │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activates the control. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Layer.WMS>} A WMS layer with maxGetUrlLength configured to 2048 │ │ │ │ │ - * since SLD selections can easily get quite long. │ │ │ │ │ + * {Boolean} The control was effectively activated. │ │ │ │ │ */ │ │ │ │ │ - createSelectionLayer: function(source) { │ │ │ │ │ - // check if we already have a selection layer for the source layer │ │ │ │ │ - var selectionLayer; │ │ │ │ │ - if (!this.layerCache[source.id]) { │ │ │ │ │ - selectionLayer = new OpenLayers.Layer.WMS(source.name, │ │ │ │ │ - source.url, source.params, │ │ │ │ │ - OpenLayers.Util.applyDefaults( │ │ │ │ │ - this.layerOptions, │ │ │ │ │ - source.getOptions()) │ │ │ │ │ - ); │ │ │ │ │ - this.layerCache[source.id] = selectionLayer; │ │ │ │ │ - // make sure the layers are coupled wrt visibility, but only │ │ │ │ │ - // if they are not displayed in the layer switcher, because in │ │ │ │ │ - // that case the user cannot control visibility. │ │ │ │ │ - if (this.layerOptions.displayInLayerSwitcher === false) { │ │ │ │ │ - source.events.on({ │ │ │ │ │ - "visibilitychanged": this.coupleLayerVisiblity, │ │ │ │ │ - scope: selectionLayer │ │ │ │ │ - }); │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + if (this.layers) { │ │ │ │ │ + this.map.addLayer(this.layer); │ │ │ │ │ + } │ │ │ │ │ + this.handlers.feature.activate(); │ │ │ │ │ + if (this.box && this.handlers.box) { │ │ │ │ │ + this.handlers.box.activate(); │ │ │ │ │ } │ │ │ │ │ - this.map.addLayer(selectionLayer); │ │ │ │ │ - } else { │ │ │ │ │ - selectionLayer = this.layerCache[source.id]; │ │ │ │ │ } │ │ │ │ │ - return selectionLayer; │ │ │ │ │ + return OpenLayers.Control.prototype.activate.apply( │ │ │ │ │ + this, arguments │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createSLD │ │ │ │ │ - * Create the SLD document for the layer using the supplied filters. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.WMS>} │ │ │ │ │ - * filters - Array({<OpenLayers.Filter>}) The filters to be applied. │ │ │ │ │ - * geometryAttributes - Array({Object}) The geometry attributes of the │ │ │ │ │ - * layer. │ │ │ │ │ - * │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivates the control. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} The SLD document generated as a string. │ │ │ │ │ + * {Boolean} The control was effectively deactivated. │ │ │ │ │ */ │ │ │ │ │ - createSLD: function(layer, filters, geometryAttributes) { │ │ │ │ │ - var sld = { │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - namedLayers: {} │ │ │ │ │ - }; │ │ │ │ │ - var layerNames = [layer.params.LAYERS].join(",").split(","); │ │ │ │ │ - for (var i = 0, len = layerNames.length; i < len; i++) { │ │ │ │ │ - var name = layerNames[i]; │ │ │ │ │ - sld.namedLayers[name] = { │ │ │ │ │ - name: name, │ │ │ │ │ - userStyles: [] │ │ │ │ │ - }; │ │ │ │ │ - var symbolizer = this.selectionSymbolizer; │ │ │ │ │ - var geometryAttribute = geometryAttributes[i]; │ │ │ │ │ - if (geometryAttribute.type.indexOf('Polygon') >= 0) { │ │ │ │ │ - symbolizer = { │ │ │ │ │ - Polygon: this.selectionSymbolizer['Polygon'] │ │ │ │ │ - }; │ │ │ │ │ - } else if (geometryAttribute.type.indexOf('LineString') >= 0) { │ │ │ │ │ - symbolizer = { │ │ │ │ │ - Line: this.selectionSymbolizer['Line'] │ │ │ │ │ - }; │ │ │ │ │ - } else if (geometryAttribute.type.indexOf('Point') >= 0) { │ │ │ │ │ - symbolizer = { │ │ │ │ │ - Point: this.selectionSymbolizer['Point'] │ │ │ │ │ - }; │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.handlers.feature.deactivate(); │ │ │ │ │ + if (this.handlers.box) { │ │ │ │ │ + this.handlers.box.deactivate(); │ │ │ │ │ + } │ │ │ │ │ + if (this.layers) { │ │ │ │ │ + this.map.removeLayer(this.layer); │ │ │ │ │ } │ │ │ │ │ - var filter = filters[i]; │ │ │ │ │ - sld.namedLayers[name].userStyles.push({ │ │ │ │ │ - name: 'default', │ │ │ │ │ - rules: [ │ │ │ │ │ - new OpenLayers.Rule({ │ │ │ │ │ - symbolizer: symbolizer, │ │ │ │ │ - filter: filter, │ │ │ │ │ - maxScaleDenominator: layer.options.minScale │ │ │ │ │ - }) │ │ │ │ │ - ] │ │ │ │ │ - }); │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Format.SLD({ │ │ │ │ │ - srsName: this.map.getProjection() │ │ │ │ │ - }).write(sld); │ │ │ │ │ + return OpenLayers.Control.prototype.deactivate.apply( │ │ │ │ │ + this, arguments │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseDescribeLayer │ │ │ │ │ - * Parse the SLD WMS DescribeLayer response and issue the corresponding │ │ │ │ │ - * WFS DescribeFeatureType request │ │ │ │ │ + * Method: unselectAll │ │ │ │ │ + * Unselect all selected features. To unselect all except for a single │ │ │ │ │ + * feature, set the options.except property to the feature. │ │ │ │ │ * │ │ │ │ │ - * request - {XMLHttpRequest} The request object. │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional configuration object. │ │ │ │ │ */ │ │ │ │ │ - parseDescribeLayer: function(request) { │ │ │ │ │ - var format = new OpenLayers.Format.WMSDescribeLayer(); │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ - } │ │ │ │ │ - var describeLayer = format.read(doc); │ │ │ │ │ - var typeNames = []; │ │ │ │ │ - var url = null; │ │ │ │ │ - for (var i = 0, len = describeLayer.length; i < len; i++) { │ │ │ │ │ - // perform a WFS DescribeFeatureType request │ │ │ │ │ - if (describeLayer[i].owsType == "WFS") { │ │ │ │ │ - typeNames.push(describeLayer[i].typeName); │ │ │ │ │ - url = describeLayer[i].owsURL; │ │ │ │ │ + unselectAll: function(options) { │ │ │ │ │ + // we'll want an option to supress notification here │ │ │ │ │ + var layers = this.layers || [this.layer], │ │ │ │ │ + layer, feature, l, numExcept; │ │ │ │ │ + for (l = 0; l < layers.length; ++l) { │ │ │ │ │ + layer = layers[l]; │ │ │ │ │ + numExcept = 0; │ │ │ │ │ + //layer.selectedFeatures is null when layer is destroyed and │ │ │ │ │ + //one of it's preremovelayer listener calls setLayer │ │ │ │ │ + //with another layer on this control │ │ │ │ │ + if (layer.selectedFeatures != null) { │ │ │ │ │ + while (layer.selectedFeatures.length > numExcept) { │ │ │ │ │ + feature = layer.selectedFeatures[numExcept]; │ │ │ │ │ + if (!options || options.except != feature) { │ │ │ │ │ + this.unselect(feature); │ │ │ │ │ + } else { │ │ │ │ │ + ++numExcept; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var options = { │ │ │ │ │ - url: url, │ │ │ │ │ - params: { │ │ │ │ │ - SERVICE: "WFS", │ │ │ │ │ - TYPENAME: typeNames.toString(), │ │ │ │ │ - REQUEST: "DescribeFeatureType", │ │ │ │ │ - VERSION: "1.0.0" │ │ │ │ │ - }, │ │ │ │ │ - callback: function(request) { │ │ │ │ │ - var format = new OpenLayers.Format.WFSDescribeFeatureType(); │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ - } │ │ │ │ │ - var describeFeatureType = format.read(doc); │ │ │ │ │ - this.control.wfsCache[this.layer.id] = describeFeatureType; │ │ │ │ │ - this.control._queue && this.control.applySelection(); │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Request.GET(options); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getGeometryAttributes │ │ │ │ │ - * Look up the geometry attributes from the WFS DescribeFeatureType response │ │ │ │ │ + * Method: clickFeature │ │ │ │ │ + * Called on click in a feature │ │ │ │ │ + * Only responds if this.hover is false. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.WMS>} The layer for which to look up the │ │ │ │ │ - * geometry attributes. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * Array({Object}) Array of geometry attributes │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - getGeometryAttributes: function(layer) { │ │ │ │ │ - var result = []; │ │ │ │ │ - var cache = this.wfsCache[layer.id]; │ │ │ │ │ - for (var i = 0, len = cache.featureTypes.length; i < len; i++) { │ │ │ │ │ - var typeName = cache.featureTypes[i]; │ │ │ │ │ - var properties = typeName.properties; │ │ │ │ │ - for (var j = 0, lenj = properties.length; j < lenj; j++) { │ │ │ │ │ - var property = properties[j]; │ │ │ │ │ - var type = property.type; │ │ │ │ │ - if ((type.indexOf('LineString') >= 0) || │ │ │ │ │ - (type.indexOf('GeometryAssociationType') >= 0) || │ │ │ │ │ - (type.indexOf('GeometryPropertyType') >= 0) || │ │ │ │ │ - (type.indexOf('Point') >= 0) || │ │ │ │ │ - (type.indexOf('Polygon') >= 0)) { │ │ │ │ │ - result.push(property); │ │ │ │ │ + clickFeature: function(feature) { │ │ │ │ │ + if (!this.hover) { │ │ │ │ │ + var selected = (OpenLayers.Util.indexOf( │ │ │ │ │ + feature.layer.selectedFeatures, feature) > -1); │ │ │ │ │ + if (selected) { │ │ │ │ │ + if (this.toggleSelect()) { │ │ │ │ │ + this.unselect(feature); │ │ │ │ │ + } else if (!this.multipleSelect()) { │ │ │ │ │ + this.unselectAll({ │ │ │ │ │ + except: feature │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + if (!this.multipleSelect()) { │ │ │ │ │ + this.unselectAll({ │ │ │ │ │ + except: feature │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ + this.select(feature); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return result; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: activate │ │ │ │ │ - * Activate the control. Activating the control will perform a SLD WMS │ │ │ │ │ - * DescribeLayer request followed by a WFS DescribeFeatureType request │ │ │ │ │ - * so that the proper symbolizers can be chosen based on the geometry │ │ │ │ │ - * type. │ │ │ │ │ + * Method: multipleSelect │ │ │ │ │ + * Allow for multiple selected features based on <multiple> property and │ │ │ │ │ + * <multipleKey> event modifier. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Allow for multiple selected features. │ │ │ │ │ */ │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Control.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.layers[i]; │ │ │ │ │ - if (layer && !this.wfsCache[layer.id]) { │ │ │ │ │ - var options = { │ │ │ │ │ - url: layer.url, │ │ │ │ │ - params: { │ │ │ │ │ - SERVICE: "WMS", │ │ │ │ │ - VERSION: layer.params.VERSION, │ │ │ │ │ - LAYERS: layer.params.LAYERS, │ │ │ │ │ - REQUEST: "DescribeLayer" │ │ │ │ │ - }, │ │ │ │ │ - callback: this.parseDescribeLayer, │ │ │ │ │ - scope: { │ │ │ │ │ - layer: layer, │ │ │ │ │ - control: this │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Request.GET(options); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return activated; │ │ │ │ │ + multipleSelect: function() { │ │ │ │ │ + return this.multiple || (this.handlers.feature.evt && │ │ │ │ │ + this.handlers.feature.evt[this.multipleKey]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: deactivate │ │ │ │ │ - * Deactivate the control. If clearOnDeactivate is true, remove the │ │ │ │ │ - * selection layer(s). │ │ │ │ │ + * Method: toggleSelect │ │ │ │ │ + * Event should toggle the selected state of a feature based on <toggle> │ │ │ │ │ + * property and <toggleKey> event modifier. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Toggle the selected state of a feature. │ │ │ │ │ */ │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Control.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.layers[i]; │ │ │ │ │ - if (layer && this.clearOnDeactivate === true) { │ │ │ │ │ - var layerCache = this.layerCache; │ │ │ │ │ - var selectionLayer = layerCache[layer.id]; │ │ │ │ │ - if (selectionLayer) { │ │ │ │ │ - layer.events.un({ │ │ │ │ │ - "visibilitychanged": this.coupleLayerVisiblity, │ │ │ │ │ - scope: selectionLayer │ │ │ │ │ - }); │ │ │ │ │ - selectionLayer.destroy(); │ │ │ │ │ - delete layerCache[layer.id]; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return deactivated; │ │ │ │ │ + toggleSelect: function() { │ │ │ │ │ + return this.toggle || (this.handlers.feature.evt && │ │ │ │ │ + this.handlers.feature.evt[this.toggleKey]); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setLayers │ │ │ │ │ - * Set the layers on which the selection should be performed. Call the │ │ │ │ │ - * setLayers method if the layer(s) to be used change and the same │ │ │ │ │ - * control should be used on a new set of layers. │ │ │ │ │ - * If the control is already active, it will be active after the new │ │ │ │ │ - * set of layers is set. │ │ │ │ │ + * Method: clickoutFeature │ │ │ │ │ + * Called on click outside a previously clicked (selected) feature. │ │ │ │ │ + * Only responds if this.hover is false. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * layers - {Array(<OpenLayers.Layer.WMS>)} The new set of layers on which │ │ │ │ │ - * the selection should be performed. │ │ │ │ │ + * feature - {<OpenLayers.Vector.Feature>} │ │ │ │ │ */ │ │ │ │ │ - setLayers: function(layers) { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.layers = layers; │ │ │ │ │ - this.activate(); │ │ │ │ │ - } else { │ │ │ │ │ - this.layers = layers; │ │ │ │ │ + clickoutFeature: function(feature) { │ │ │ │ │ + if (!this.hover && this.clickout) { │ │ │ │ │ + this.unselectAll(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Function: createFilter │ │ │ │ │ - * Create the filter to be used in the SLD. │ │ │ │ │ + * Method: overFeature │ │ │ │ │ + * Called on over a feature. │ │ │ │ │ + * Only responds if this.hover is true. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometryAttribute - {Object} Used to get the name of the geometry │ │ │ │ │ - * attribute which is needed for constructing the spatial filter. │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} The geometry to use. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Filter.Spatial>} The spatial filter created. │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - createFilter: function(geometryAttribute, geometry) { │ │ │ │ │ - var filter = null; │ │ │ │ │ - if (this.handler instanceof OpenLayers.Handler.RegularPolygon) { │ │ │ │ │ - // box │ │ │ │ │ - if (this.handler.irregular === true) { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry.getBounds() │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } else if (this.handler instanceof OpenLayers.Handler.Polygon) { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry │ │ │ │ │ - }); │ │ │ │ │ - } else if (this.handler instanceof OpenLayers.Handler.Path) { │ │ │ │ │ - // if source layer is point based, use DWITHIN instead │ │ │ │ │ - if (geometryAttribute.type.indexOf('Point') >= 0) { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - distance: this.map.getExtent().getWidth() * 0.01, │ │ │ │ │ - distanceUnits: this.map.getUnits(), │ │ │ │ │ - value: geometry │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry │ │ │ │ │ - }); │ │ │ │ │ - } │ │ │ │ │ - } else if (this.handler instanceof OpenLayers.Handler.Click) { │ │ │ │ │ - if (geometryAttribute.type.indexOf('Polygon') >= 0) { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry │ │ │ │ │ - }); │ │ │ │ │ - } else { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - distance: this.map.getExtent().getWidth() * 0.01, │ │ │ │ │ - distanceUnits: this.map.getUnits(), │ │ │ │ │ - value: geometry │ │ │ │ │ - }); │ │ │ │ │ + overFeature: function(feature) { │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + if (this.highlightOnly) { │ │ │ │ │ + this.highlight(feature); │ │ │ │ │ + } else if (OpenLayers.Util.indexOf( │ │ │ │ │ + layer.selectedFeatures, feature) == -1) { │ │ │ │ │ + this.select(feature); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return filter; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: select │ │ │ │ │ - * When the handler is done, use SLD_BODY on the selection layer to │ │ │ │ │ - * display the selection in the map. │ │ │ │ │ + * Method: outFeature │ │ │ │ │ + * Called on out of a selected feature. │ │ │ │ │ + * Only responds if this.hover is true. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {Object} or {<OpenLayers.Geometry>} │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - select: function(geometry) { │ │ │ │ │ - this._queue = function() { │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.layers[i]; │ │ │ │ │ - var geometryAttributes = this.getGeometryAttributes(layer); │ │ │ │ │ - var filters = []; │ │ │ │ │ - for (var j = 0, lenj = geometryAttributes.length; j < lenj; j++) { │ │ │ │ │ - var geometryAttribute = geometryAttributes[j]; │ │ │ │ │ - if (geometryAttribute !== null) { │ │ │ │ │ - // from the click handler we will not get an actual │ │ │ │ │ - // geometry so transform │ │ │ │ │ - if (!(geometry instanceof OpenLayers.Geometry)) { │ │ │ │ │ - var point = this.map.getLonLatFromPixel( │ │ │ │ │ - geometry.xy); │ │ │ │ │ - geometry = new OpenLayers.Geometry.Point( │ │ │ │ │ - point.lon, point.lat); │ │ │ │ │ - } │ │ │ │ │ - var filter = this.createFilter(geometryAttribute, │ │ │ │ │ - geometry); │ │ │ │ │ - if (filter !== null) { │ │ │ │ │ - filters.push(filter); │ │ │ │ │ + outFeature: function(feature) { │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + if (this.highlightOnly) { │ │ │ │ │ + // we do nothing if we're not the last highlighter of the │ │ │ │ │ + // feature │ │ │ │ │ + if (feature._lastHighlighter == this.id) { │ │ │ │ │ + // if another select control had highlighted the feature before │ │ │ │ │ + // we did it ourself then we use that control to highlight the │ │ │ │ │ + // feature as it was before we highlighted it, else we just │ │ │ │ │ + // unhighlight it │ │ │ │ │ + if (feature._prevHighlighter && │ │ │ │ │ + feature._prevHighlighter != this.id) { │ │ │ │ │ + delete feature._lastHighlighter; │ │ │ │ │ + var control = this.map.getControl( │ │ │ │ │ + feature._prevHighlighter); │ │ │ │ │ + if (control) { │ │ │ │ │ + control.highlight(feature); │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + this.unhighlight(feature); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var selectionLayer = this.createSelectionLayer(layer); │ │ │ │ │ - │ │ │ │ │ - this.events.triggerEvent("selected", { │ │ │ │ │ - layer: layer, │ │ │ │ │ - filters: filters │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - var sld = this.createSLD(layer, filters, geometryAttributes); │ │ │ │ │ - │ │ │ │ │ - selectionLayer.mergeNewParams({ │ │ │ │ │ - SLD_BODY: sld │ │ │ │ │ - }); │ │ │ │ │ - delete this._queue; │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - this.applySelection(); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: applySelection │ │ │ │ │ - * Checks if all required wfs data is cached, and applies the selection │ │ │ │ │ - */ │ │ │ │ │ - applySelection: function() { │ │ │ │ │ - var canApply = true; │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - if (!this.wfsCache[this.layers[i].id]) { │ │ │ │ │ - canApply = false; │ │ │ │ │ - break; │ │ │ │ │ + } else { │ │ │ │ │ + this.unselect(feature); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - canApply && this._queue.call(this); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.SLDSelect" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Renderer/Canvas.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Renderer.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Renderer.Canvas │ │ │ │ │ - * A renderer based on the 2D 'canvas' drawing element. │ │ │ │ │ - * │ │ │ │ │ - * Inherits: │ │ │ │ │ - * - <OpenLayers.Renderer> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.Canvas = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: hitDetection │ │ │ │ │ - * {Boolean} Allow for hit detection of features. Default is true. │ │ │ │ │ - */ │ │ │ │ │ - hitDetection: true, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: hitOverflow │ │ │ │ │ - * {Number} The method for converting feature identifiers to color values │ │ │ │ │ - * supports 16777215 sequential values. Two features cannot be │ │ │ │ │ - * predictably detected if their identifiers differ by more than this │ │ │ │ │ - * value. The hitOverflow allows for bigger numbers (but the │ │ │ │ │ - * difference in values is still limited). │ │ │ │ │ - */ │ │ │ │ │ - hitOverflow: 0, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: canvas │ │ │ │ │ - * {Canvas} The canvas context object. │ │ │ │ │ - */ │ │ │ │ │ - canvas: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: features │ │ │ │ │ - * {Object} Internal object of feature/style pairs for use in redrawing the layer. │ │ │ │ │ - */ │ │ │ │ │ - features: null, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: pendingRedraw │ │ │ │ │ - * {Boolean} The renderer needs a redraw call to render features added while │ │ │ │ │ - * the renderer was locked. │ │ │ │ │ - */ │ │ │ │ │ - pendingRedraw: false, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Property: cachedSymbolBounds │ │ │ │ │ - * {Object} Internal cache of calculated symbol extents. │ │ │ │ │ - */ │ │ │ │ │ - cachedSymbolBounds: {}, │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Renderer.Canvas │ │ │ │ │ + * Method: highlight │ │ │ │ │ + * Redraw feature with the select style. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * containerID - {<String>} │ │ │ │ │ - * options - {Object} Optional properties to be set on the renderer. │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(containerID, options) { │ │ │ │ │ - OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.root = document.createElement("canvas"); │ │ │ │ │ - this.container.appendChild(this.root); │ │ │ │ │ - this.canvas = this.root.getContext("2d"); │ │ │ │ │ - this.features = {}; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitCanvas = document.createElement("canvas"); │ │ │ │ │ - this.hitContext = this.hitCanvas.getContext("2d"); │ │ │ │ │ + highlight: function(feature) { │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + var cont = this.events.triggerEvent("beforefeaturehighlighted", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + feature._prevHighlighter = feature._lastHighlighter; │ │ │ │ │ + feature._lastHighlighter = this.id; │ │ │ │ │ + var style = this.selectStyle || this.renderIntent; │ │ │ │ │ + layer.drawFeature(feature, style); │ │ │ │ │ + this.events.triggerEvent("featurehighlighted", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setExtent │ │ │ │ │ - * Set the visible part of the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} │ │ │ │ │ - * resolutionChanged - {Boolean} │ │ │ │ │ + * Method: unhighlight │ │ │ │ │ + * Redraw feature with the "default" style │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ - * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ - * False otherwise. │ │ │ │ │ - */ │ │ │ │ │ - setExtent: function() { │ │ │ │ │ - OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - // always redraw features │ │ │ │ │ - return false; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: eraseGeometry │ │ │ │ │ - * Erase a geometry from the renderer. Because the Canvas renderer has │ │ │ │ │ - * 'memory' of the features that it has drawn, we have to remove the │ │ │ │ │ - * feature so it doesn't redraw. │ │ │ │ │ - * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - eraseGeometry: function(geometry, featureId) { │ │ │ │ │ - this.eraseFeatures(this.features[featureId][0]); │ │ │ │ │ + unhighlight: function(feature) { │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + // three cases: │ │ │ │ │ + // 1. there's no other highlighter, in that case _prev is undefined, │ │ │ │ │ + // and we just need to undef _last │ │ │ │ │ + // 2. another control highlighted the feature after we did it, in │ │ │ │ │ + // that case _last references this other control, and we just │ │ │ │ │ + // need to undef _prev │ │ │ │ │ + // 3. another control highlighted the feature before we did it, in │ │ │ │ │ + // that case _prev references this other control, and we need to │ │ │ │ │ + // set _last to _prev and undef _prev │ │ │ │ │ + if (feature._prevHighlighter == undefined) { │ │ │ │ │ + delete feature._lastHighlighter; │ │ │ │ │ + } else if (feature._prevHighlighter == this.id) { │ │ │ │ │ + delete feature._prevHighlighter; │ │ │ │ │ + } else { │ │ │ │ │ + feature._lastHighlighter = feature._prevHighlighter; │ │ │ │ │ + delete feature._prevHighlighter; │ │ │ │ │ + } │ │ │ │ │ + layer.drawFeature(feature, feature.style || feature.layer.style || │ │ │ │ │ + "default"); │ │ │ │ │ + this.events.triggerEvent("featureunhighlighted", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: supported │ │ │ │ │ + * Method: select │ │ │ │ │ + * Add feature to the layer's selectedFeature array, render the feature as │ │ │ │ │ + * selected, and call the onSelect function. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the browser supports the renderer class │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - supported: function() { │ │ │ │ │ - return OpenLayers.CANVAS_SUPPORTED; │ │ │ │ │ + select: function(feature) { │ │ │ │ │ + var cont = this.onBeforeSelect.call(this.scope, feature); │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + cont = layer.events.triggerEvent("beforefeatureselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + layer.selectedFeatures.push(feature); │ │ │ │ │ + this.highlight(feature); │ │ │ │ │ + // if the feature handler isn't involved in the feature │ │ │ │ │ + // selection (because the box handler is used or the │ │ │ │ │ + // feature is selected programatically) we fake the │ │ │ │ │ + // feature handler to allow unselecting on click │ │ │ │ │ + if (!this.handlers.feature.lastFeature) { │ │ │ │ │ + this.handlers.feature.lastFeature = layer.selectedFeatures[0]; │ │ │ │ │ + } │ │ │ │ │ + layer.events.triggerEvent("featureselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.onSelect.call(this.scope, feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setSize │ │ │ │ │ - * Sets the size of the drawing surface. │ │ │ │ │ - * │ │ │ │ │ - * Once the size is updated, redraw the canvas. │ │ │ │ │ + * Method: unselect │ │ │ │ │ + * Remove feature from the layer's selectedFeature array, render the feature as │ │ │ │ │ + * normal, and call the onUnselect function. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>} │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ */ │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - this.size = size.clone(); │ │ │ │ │ - var root = this.root; │ │ │ │ │ - root.style.width = size.w + "px"; │ │ │ │ │ - root.style.height = size.h + "px"; │ │ │ │ │ - root.width = size.w; │ │ │ │ │ - root.height = size.h; │ │ │ │ │ - this.resolution = null; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - var hitCanvas = this.hitCanvas; │ │ │ │ │ - hitCanvas.style.width = size.w + "px"; │ │ │ │ │ - hitCanvas.style.height = size.h + "px"; │ │ │ │ │ - hitCanvas.width = size.w; │ │ │ │ │ - hitCanvas.height = size.h; │ │ │ │ │ - } │ │ │ │ │ + unselect: function(feature) { │ │ │ │ │ + var layer = feature.layer; │ │ │ │ │ + // Store feature style for restoration later │ │ │ │ │ + this.unhighlight(feature); │ │ │ │ │ + OpenLayers.Util.removeItem(layer.selectedFeatures, feature); │ │ │ │ │ + layer.events.triggerEvent("featureunselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.onUnselect.call(this.scope, feature); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawFeature │ │ │ │ │ - * Draw the feature. Stores the feature in the features list, │ │ │ │ │ - * then redraws the layer. │ │ │ │ │ + * Method: selectBox │ │ │ │ │ + * Callback from the handlers.box set up when <box> selection is true │ │ │ │ │ + * on. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * style - {<Object>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The feature has been drawn completely. If the feature has no │ │ │ │ │ - * geometry, undefined will be returned. If the feature is not rendered │ │ │ │ │ - * for other reasons, false will be returned. │ │ │ │ │ + * position - {<OpenLayers.Bounds> || <OpenLayers.Pixel> } │ │ │ │ │ */ │ │ │ │ │ - drawFeature: function(feature, style) { │ │ │ │ │ - var rendered; │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - style = this.applyDefaultSymbolizer(style || feature.style); │ │ │ │ │ - // don't render if display none or feature outside extent │ │ │ │ │ - var bounds = feature.geometry.getBounds(); │ │ │ │ │ + selectBox: function(position) { │ │ │ │ │ + if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ + var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.left, │ │ │ │ │ + y: position.bottom │ │ │ │ │ + }); │ │ │ │ │ + var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.right, │ │ │ │ │ + y: position.top │ │ │ │ │ + }); │ │ │ │ │ + var bounds = new OpenLayers.Bounds( │ │ │ │ │ + minXY.lon, minXY.lat, maxXY.lon, maxXY.lat │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - var worldBounds; │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - worldBounds = this.map.getMaxExtent(); │ │ │ │ │ + // if multiple is false, first deselect currently selected features │ │ │ │ │ + if (!this.multipleSelect()) { │ │ │ │ │ + this.unselectAll(); │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - var intersects = bounds && bounds.intersectsBounds(this.extent, { │ │ │ │ │ - worldBounds: worldBounds │ │ │ │ │ + // because we're using a box, we consider we want multiple selection │ │ │ │ │ + var prevMultiple = this.multiple; │ │ │ │ │ + this.multiple = true; │ │ │ │ │ + var layers = this.layers || [this.layer]; │ │ │ │ │ + this.events.triggerEvent("boxselectionstart", { │ │ │ │ │ + layers: layers │ │ │ │ │ }); │ │ │ │ │ + var layer; │ │ │ │ │ + for (var l = 0; l < layers.length; ++l) { │ │ │ │ │ + layer = layers[l]; │ │ │ │ │ + for (var i = 0, len = layer.features.length; i < len; ++i) { │ │ │ │ │ + var feature = layer.features[i]; │ │ │ │ │ + // check if the feature is displayed │ │ │ │ │ + if (!feature.getVisibility()) { │ │ │ │ │ + continue; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - rendered = (style.display !== "none") && !!bounds && intersects; │ │ │ │ │ - if (rendered) { │ │ │ │ │ - // keep track of what we have rendered for redraw │ │ │ │ │ - this.features[feature.id] = [feature, style]; │ │ │ │ │ - } else { │ │ │ │ │ - // remove from features tracked for redraw │ │ │ │ │ - delete(this.features[feature.id]); │ │ │ │ │ + if (this.geometryTypes == null || OpenLayers.Util.indexOf( │ │ │ │ │ + this.geometryTypes, feature.geometry.CLASS_NAME) > -1) { │ │ │ │ │ + if (bounds.toGeometry().intersects(feature.geometry)) { │ │ │ │ │ + if (OpenLayers.Util.indexOf(layer.selectedFeatures, feature) == -1) { │ │ │ │ │ + this.select(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.pendingRedraw = true; │ │ │ │ │ - } │ │ │ │ │ - if (this.pendingRedraw && !this.locked) { │ │ │ │ │ - this.redraw(); │ │ │ │ │ - this.pendingRedraw = false; │ │ │ │ │ + this.multiple = prevMultiple; │ │ │ │ │ + this.events.triggerEvent("boxselectionend", { │ │ │ │ │ + layers: layers │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - return rendered; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawGeometry │ │ │ │ │ - * Used when looping (in redraw) over the features; draws │ │ │ │ │ - * the canvas. │ │ │ │ │ - * │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the control. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ - var className = geometry.CLASS_NAME; │ │ │ │ │ - if ((className == "OpenLayers.Geometry.Collection") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiPoint") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiLineString") || │ │ │ │ │ - (className == "OpenLayers.Geometry.MultiPolygon")) { │ │ │ │ │ - for (var i = 0; i < geometry.components.length; i++) { │ │ │ │ │ - this.drawGeometry(geometry.components[i], style, featureId); │ │ │ │ │ - } │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - this.drawPoint(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - this.drawLineString(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - this.drawPolygon(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break; │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.handlers.feature.setMap(map); │ │ │ │ │ + if (this.box) { │ │ │ │ │ + this.handlers.box.setMap(map); │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawExternalGraphic │ │ │ │ │ - * Called to draw External graphics. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * APIMethod: setLayer │ │ │ │ │ + * Attach a new layer to the control, overriding any existing layers. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layers - Array of {<OpenLayers.Layer.Vector>} or a single │ │ │ │ │ + * {<OpenLayers.Layer.Vector>} │ │ │ │ │ */ │ │ │ │ │ - drawExternalGraphic: function(geometry, style, featureId) { │ │ │ │ │ - var img = new Image(); │ │ │ │ │ - │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - img.title = title; │ │ │ │ │ + setLayer: function(layers) { │ │ │ │ │ + var isActive = this.active; │ │ │ │ │ + this.unselectAll(); │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + if (this.layers) { │ │ │ │ │ + this.layer.destroy(); │ │ │ │ │ + this.layers = null; │ │ │ │ │ + } │ │ │ │ │ + this.initLayer(layers); │ │ │ │ │ + this.handlers.feature.layer = this.layer; │ │ │ │ │ + if (isActive) { │ │ │ │ │ + this.activate(); │ │ │ │ │ } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ - style.graphicXOffset : -(0.5 * width); │ │ │ │ │ - var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ - style.graphicYOffset : -(0.5 * height); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.SelectFeature" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/EditingToolbar.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - var onLoad = function() { │ │ │ │ │ - if (!this.features[featureId]) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ - if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ - var x = (p0 + xOffset) | 0; │ │ │ │ │ - var y = (p1 + yOffset) | 0; │ │ │ │ │ - var canvas = this.canvas; │ │ │ │ │ - canvas.globalAlpha = opacity; │ │ │ │ │ - var factor = OpenLayers.Renderer.Canvas.drawImageScaleFactor || │ │ │ │ │ - (OpenLayers.Renderer.Canvas.drawImageScaleFactor = │ │ │ │ │ - /android 2.1/.test(navigator.userAgent.toLowerCase()) ? │ │ │ │ │ - // 320 is the screen width of the G1 phone, for │ │ │ │ │ - // which drawImage works out of the box. │ │ │ │ │ - 320 / window.screen.width : 1 │ │ │ │ │ - ); │ │ │ │ │ - canvas.drawImage( │ │ │ │ │ - img, x * factor, y * factor, width * factor, height * factor │ │ │ │ │ - ); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId); │ │ │ │ │ - this.hitContext.fillRect(x, y, width, height); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control/Panel.js │ │ │ │ │ + * @requires OpenLayers/Control/Navigation.js │ │ │ │ │ + * @requires OpenLayers/Control/DrawFeature.js │ │ │ │ │ + * @requires OpenLayers/Handler/Point.js │ │ │ │ │ + * @requires OpenLayers/Handler/Path.js │ │ │ │ │ + * @requires OpenLayers/Handler/Polygon.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - img.onload = OpenLayers.Function.bind(onLoad, this); │ │ │ │ │ - img.src = style.externalGraphic; │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.EditingToolbar │ │ │ │ │ + * The EditingToolbar is a panel of 4 controls to draw polygons, lines, │ │ │ │ │ + * points, or to navigate the map by panning. By default it appears in the │ │ │ │ │ + * upper right corner of the map. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control.Panel> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.EditingToolbar = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Control.Panel, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawNamedSymbol │ │ │ │ │ - * Called to draw Well Known Graphic Symbol Name. │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - */ │ │ │ │ │ - drawNamedSymbol: function(geometry, style, featureId) { │ │ │ │ │ - var x, y, cx, cy, i, symbolBounds, scaling, angle; │ │ │ │ │ - var unscaledStrokeWidth; │ │ │ │ │ - var deg2rad = Math.PI / 180.0; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: citeCompliant │ │ │ │ │ + * {Boolean} If set to true, coordinates of features drawn in a map extent │ │ │ │ │ + * crossing the date line won't exceed the world bounds. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + citeCompliant: false, │ │ │ │ │ │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[style.graphicName]; │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.EditingToolbar │ │ │ │ │ + * Create an editing toolbar for a given layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * layer - {<OpenLayers.Layer.Vector>} │ │ │ │ │ + * options - {Object} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(layer, options) { │ │ │ │ │ + OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(style.graphicName + ' is not a valid symbol name'); │ │ │ │ │ - } │ │ │ │ │ + this.addControls( │ │ │ │ │ + [new OpenLayers.Control.Navigation()] │ │ │ │ │ + ); │ │ │ │ │ + var controls = [ │ │ │ │ │ + new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Point, { │ │ │ │ │ + displayClass: 'olControlDrawFeaturePoint', │ │ │ │ │ + handlerOptions: { │ │ │ │ │ + citeCompliant: this.citeCompliant │ │ │ │ │ + } │ │ │ │ │ + }), │ │ │ │ │ + new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Path, { │ │ │ │ │ + displayClass: 'olControlDrawFeaturePath', │ │ │ │ │ + handlerOptions: { │ │ │ │ │ + citeCompliant: this.citeCompliant │ │ │ │ │ + } │ │ │ │ │ + }), │ │ │ │ │ + new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Polygon, { │ │ │ │ │ + displayClass: 'olControlDrawFeaturePolygon', │ │ │ │ │ + handlerOptions: { │ │ │ │ │ + citeCompliant: this.citeCompliant │ │ │ │ │ + } │ │ │ │ │ + }) │ │ │ │ │ + ]; │ │ │ │ │ + this.addControls(controls); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (!symbol.length || symbol.length < 2) return; │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * calls the default draw, and then activates mouse defaults. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + draw: function() { │ │ │ │ │ + var div = OpenLayers.Control.Panel.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (this.defaultControl === null) { │ │ │ │ │ + this.defaultControl = this.controls[0]; │ │ │ │ │ + } │ │ │ │ │ + return div; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.EditingToolbar" │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/Attribution.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - if (isNaN(p0) || isNaN(p1)) return; │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - // Use rounded line caps │ │ │ │ │ - this.canvas.lineCap = "round"; │ │ │ │ │ - this.canvas.lineJoin = "round"; │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.lineCap = "round"; │ │ │ │ │ - this.hitContext.lineJoin = "round"; │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.Attribution │ │ │ │ │ + * The attribution control adds attribution from layers to the map display. │ │ │ │ │ + * It uses 'attribution' property of each layer. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.Attribution = │ │ │ │ │ + OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - // Scale and rotate symbols, using precalculated bounds whenever possible. │ │ │ │ │ - if (style.graphicName in this.cachedSymbolBounds) { │ │ │ │ │ - symbolBounds = this.cachedSymbolBounds[style.graphicName]; │ │ │ │ │ - } else { │ │ │ │ │ - symbolBounds = new OpenLayers.Bounds(); │ │ │ │ │ - for (i = 0; i < symbol.length; i += 2) { │ │ │ │ │ - symbolBounds.extend(new OpenLayers.LonLat(symbol[i], symbol[i + 1])); │ │ │ │ │ - } │ │ │ │ │ - this.cachedSymbolBounds[style.graphicName] = symbolBounds; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: separator │ │ │ │ │ + * {String} String used to separate layers. │ │ │ │ │ + */ │ │ │ │ │ + separator: ", ", │ │ │ │ │ │ │ │ │ │ - // Push symbol scaling, translation and rotation onto the transformation stack in reverse order. │ │ │ │ │ - // Don't forget to apply all canvas transformations to the hitContext canvas as well(!) │ │ │ │ │ - this.canvas.save(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.save(); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: template │ │ │ │ │ + * {String} Template for the attribution. This has to include the substring │ │ │ │ │ + * "${layers}", which will be replaced by the layer specific │ │ │ │ │ + * attributions, separated by <separator>. The default is "${layers}". │ │ │ │ │ + */ │ │ │ │ │ + template: "${layers}", │ │ │ │ │ │ │ │ │ │ - // Step 3: place symbol at the desired location │ │ │ │ │ - this.canvas.translate(p0, p1); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.translate(p0, p1); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Control.Attribution │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Options for control. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // Step 2a. rotate the symbol if necessary │ │ │ │ │ - angle = deg2rad * style.rotation; // will be NaN when style.rotation is undefined. │ │ │ │ │ - if (!isNaN(angle)) { │ │ │ │ │ - this.canvas.rotate(angle); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.rotate(angle); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Destroy control. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + "removelayer": this.updateAttribution, │ │ │ │ │ + "addlayer": this.updateAttribution, │ │ │ │ │ + "changelayer": this.updateAttribution, │ │ │ │ │ + "changebaselayer": this.updateAttribution, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - // // Step 2: scale symbol such that pointRadius equals half the maximum symbol dimension. │ │ │ │ │ - scaling = 2.0 * style.pointRadius / Math.max(symbolBounds.getWidth(), symbolBounds.getHeight()); │ │ │ │ │ - this.canvas.scale(scaling, scaling); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.scale(scaling, scaling); │ │ │ │ │ - } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // Step 1: center the symbol at the origin │ │ │ │ │ - cx = symbolBounds.getCenterLonLat().lon; │ │ │ │ │ - cy = symbolBounds.getCenterLonLat().lat; │ │ │ │ │ - this.canvas.translate(-cx, -cy); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.translate(-cx, -cy); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Initialize control. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A reference to the DIV DOMElement containing the control │ │ │ │ │ + */ │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - // Don't forget to scale stroke widths, because they are affected by canvas scale transformations as well(!) │ │ │ │ │ - // Alternative: scale symbol coordinates manually, so stroke width scaling is not needed anymore. │ │ │ │ │ - unscaledStrokeWidth = style.strokeWidth; │ │ │ │ │ - style.strokeWidth = unscaledStrokeWidth / scaling; │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + 'changebaselayer': this.updateAttribution, │ │ │ │ │ + 'changelayer': this.updateAttribution, │ │ │ │ │ + 'addlayer': this.updateAttribution, │ │ │ │ │ + 'removelayer': this.updateAttribution, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.updateAttribution(); │ │ │ │ │ │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.canvas.lineTo(x, y); │ │ │ │ │ - } │ │ │ │ │ - this.canvas.closePath(); │ │ │ │ │ - this.canvas.fill(); │ │ │ │ │ + return this.div; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.hitContext.lineTo(x, y); │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateAttribution │ │ │ │ │ + * Update attribution string. │ │ │ │ │ + */ │ │ │ │ │ + updateAttribution: function() { │ │ │ │ │ + var attributions = []; │ │ │ │ │ + if (this.map && this.map.layers) { │ │ │ │ │ + for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if (layer.attribution && layer.getVisibility()) { │ │ │ │ │ + // add attribution only if attribution text is unique │ │ │ │ │ + if (OpenLayers.Util.indexOf( │ │ │ │ │ + attributions, layer.attribution) === -1) { │ │ │ │ │ + attributions.push(layer.attribution); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.hitContext.closePath(); │ │ │ │ │ - this.hitContext.fill(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.canvas.lineTo(x, y); │ │ │ │ │ + this.div.innerHTML = OpenLayers.String.format(this.template, { │ │ │ │ │ + layers: attributions.join(this.separator) │ │ │ │ │ + }); │ │ │ │ │ } │ │ │ │ │ - this.canvas.closePath(); │ │ │ │ │ - this.canvas.stroke(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Attribution" │ │ │ │ │ + }); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/Geolocate.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style, scaling); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.hitContext.moveTo(x, y); │ │ │ │ │ - this.hitContext.lineTo(x, y); │ │ │ │ │ - } │ │ │ │ │ - this.hitContext.closePath(); │ │ │ │ │ - this.hitContext.stroke(); │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Point.js │ │ │ │ │ + * @requires OpenLayers/Projection.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - style.strokeWidth = unscaledStrokeWidth; │ │ │ │ │ - this.canvas.restore(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.restore(); │ │ │ │ │ - } │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ - }, │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.Geolocate │ │ │ │ │ + * The Geolocate control wraps w3c geolocation API into control that can be │ │ │ │ │ + * bound to a map, and generate events on location update │ │ │ │ │ + * │ │ │ │ │ + * To use this control requires to load the proj4js library if the projection │ │ │ │ │ + * of the map is not EPSG:4326 or EPSG:900913. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.Geolocate = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setCanvasStyle │ │ │ │ │ - * Prepare the canvas for drawing by setting various global settings. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * type - {String} one of 'stroke', 'fill', or 'reset' │ │ │ │ │ - * style - {Object} Symbolizer hash │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ + * locationupdated - Triggered when browser return a new position. Listeners will │ │ │ │ │ + * receive an object with a 'position' property which is the browser.geolocation.position │ │ │ │ │ + * native object, as well as a 'point' property which is the location transformed in the │ │ │ │ │ + * current map projection. │ │ │ │ │ + * locationfailed - Triggered when geolocation has failed │ │ │ │ │ + * locationuncapable - Triggered when control is activated on a browser │ │ │ │ │ + * which doesn't support geolocation │ │ │ │ │ */ │ │ │ │ │ - setCanvasStyle: function(type, style) { │ │ │ │ │ - if (type === "fill") { │ │ │ │ │ - this.canvas.globalAlpha = style['fillOpacity']; │ │ │ │ │ - this.canvas.fillStyle = style['fillColor']; │ │ │ │ │ - } else if (type === "stroke") { │ │ │ │ │ - this.canvas.globalAlpha = style['strokeOpacity']; │ │ │ │ │ - this.canvas.strokeStyle = style['strokeColor']; │ │ │ │ │ - this.canvas.lineWidth = style['strokeWidth']; │ │ │ │ │ - } else { │ │ │ │ │ - this.canvas.globalAlpha = 0; │ │ │ │ │ - this.canvas.lineWidth = 1; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: featureIdToHex │ │ │ │ │ - * Convert a feature ID string into an RGB hex string. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureId - {String} Feature id │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} RGB hex string. │ │ │ │ │ + * Property: geolocation │ │ │ │ │ + * {Object} The geolocation engine, as a property to be possibly mocked. │ │ │ │ │ + * This is set lazily to avoid a memory leak in IE9. │ │ │ │ │ */ │ │ │ │ │ - featureIdToHex: function(featureId) { │ │ │ │ │ - var id = Number(featureId.split("_").pop()) + 1; // zero for no feature │ │ │ │ │ - if (id >= 16777216) { │ │ │ │ │ - this.hitOverflow = id - 16777215; │ │ │ │ │ - id = id % 16777216 + 1; │ │ │ │ │ - } │ │ │ │ │ - var hex = "000000" + id.toString(16); │ │ │ │ │ - var len = hex.length; │ │ │ │ │ - hex = "#" + hex.substring(len - 6, len); │ │ │ │ │ - return hex; │ │ │ │ │ - }, │ │ │ │ │ + geolocation: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setHitContextStyle │ │ │ │ │ - * Prepare the hit canvas for drawing by setting various global settings. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * type - {String} one of 'stroke', 'fill', or 'reset' │ │ │ │ │ - * featureId - {String} The feature id. │ │ │ │ │ - * symbolizer - {<OpenLayers.Symbolizer>} The symbolizer. │ │ │ │ │ + * Property: available │ │ │ │ │ + * {Boolean} The navigator.geolocation object is available. │ │ │ │ │ */ │ │ │ │ │ - setHitContextStyle: function(type, featureId, symbolizer, strokeScaling) { │ │ │ │ │ - var hex = this.featureIdToHex(featureId); │ │ │ │ │ - if (type == "fill") { │ │ │ │ │ - this.hitContext.globalAlpha = 1.0; │ │ │ │ │ - this.hitContext.fillStyle = hex; │ │ │ │ │ - } else if (type == "stroke") { │ │ │ │ │ - this.hitContext.globalAlpha = 1.0; │ │ │ │ │ - this.hitContext.strokeStyle = hex; │ │ │ │ │ - // bump up stroke width to deal with antialiasing. If strokeScaling is defined, we're rendering a symbol │ │ │ │ │ - // on a transformed canvas, so the antialias width bump has to scale as well. │ │ │ │ │ - if (typeof strokeScaling === "undefined") { │ │ │ │ │ - this.hitContext.lineWidth = symbolizer.strokeWidth + 2; │ │ │ │ │ - } else { │ │ │ │ │ - if (!isNaN(strokeScaling)) { │ │ │ │ │ - this.hitContext.lineWidth = symbolizer.strokeWidth + 2.0 / strokeScaling; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.hitContext.globalAlpha = 0; │ │ │ │ │ - this.hitContext.lineWidth = 1; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + available: ('geolocation' in navigator), │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPoint │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * APIProperty: bind │ │ │ │ │ + * {Boolean} If true, map center will be set on location update. │ │ │ │ │ */ │ │ │ │ │ - drawPoint: function(geometry, style, featureId) { │ │ │ │ │ - if (style.graphic !== false) { │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - this.drawExternalGraphic(geometry, style, featureId); │ │ │ │ │ - } else if (style.graphicName && (style.graphicName != "circle")) { │ │ │ │ │ - this.drawNamedSymbol(geometry, style, featureId); │ │ │ │ │ - } else { │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ - if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ - var twoPi = Math.PI * 2; │ │ │ │ │ - var radius = style.pointRadius; │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.canvas.fill(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.hitContext.fill(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.canvas.stroke(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.hitContext.stroke(); │ │ │ │ │ - } │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + bind: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLineString │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * APIProperty: watch │ │ │ │ │ + * {Boolean} If true, position will be update regularly. │ │ │ │ │ */ │ │ │ │ │ - drawLineString: function(geometry, style, featureId) { │ │ │ │ │ - style = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - fill: false │ │ │ │ │ - }, style); │ │ │ │ │ - this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ - }, │ │ │ │ │ + watch: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLinearRing │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * APIProperty: geolocationOptions │ │ │ │ │ + * {Object} Options to pass to the navigator's geolocation API. See │ │ │ │ │ + * <http://dev.w3.org/geo/api/spec-source.html>. No specific │ │ │ │ │ + * option is passed to the geolocation API by default. │ │ │ │ │ */ │ │ │ │ │ - drawLinearRing: function(geometry, style, featureId) { │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.renderPath(this.canvas, geometry, style, featureId, "fill"); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.renderPath(this.hitContext, geometry, style, featureId, "fill"); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.renderPath(this.canvas, geometry, style, featureId, "stroke"); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ - this.renderPath(this.hitContext, geometry, style, featureId, "stroke"); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ - }, │ │ │ │ │ + geolocationOptions: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: renderPath │ │ │ │ │ - * Render a path with stroke and optional fill. │ │ │ │ │ + * Constructor: OpenLayers.Control.Geolocate │ │ │ │ │ + * Create a new control to deal with browser geolocation API │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - renderPath: function(context, geometry, style, featureId, type) { │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - context.beginPath(); │ │ │ │ │ - var start = this.getLocalXY(components[0]); │ │ │ │ │ - var x = start[0]; │ │ │ │ │ - var y = start[1]; │ │ │ │ │ - if (!isNaN(x) && !isNaN(y)) { │ │ │ │ │ - context.moveTo(start[0], start[1]); │ │ │ │ │ - for (var i = 1; i < len; ++i) { │ │ │ │ │ - var pt = this.getLocalXY(components[i]); │ │ │ │ │ - context.lineTo(pt[0], pt[1]); │ │ │ │ │ - } │ │ │ │ │ - if (type === "fill") { │ │ │ │ │ - context.fill(); │ │ │ │ │ - } else { │ │ │ │ │ - context.stroke(); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPolygon │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * featureId - {String} │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ - drawPolygon: function(geometry, style, featureId) { │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - this.drawLinearRing(components[0], style, featureId); │ │ │ │ │ - // erase inner rings │ │ │ │ │ - for (var i = 1; i < len; ++i) { │ │ │ │ │ - /** │ │ │ │ │ - * Note that this is overly agressive. Here we punch holes through │ │ │ │ │ - * all previously rendered features on the same canvas. A better │ │ │ │ │ - * solution for polygons with interior rings would be to draw the │ │ │ │ │ - * polygon on a sketch canvas first. We could erase all holes │ │ │ │ │ - * there and then copy the drawing to the layer canvas. │ │ │ │ │ - * TODO: http://trac.osgeo.org/openlayers/ticket/3130 │ │ │ │ │ - */ │ │ │ │ │ - this.canvas.globalCompositeOperation = "destination-out"; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.globalCompositeOperation = "destination-out"; │ │ │ │ │ - } │ │ │ │ │ - this.drawLinearRing( │ │ │ │ │ - components[i], │ │ │ │ │ - OpenLayers.Util.applyDefaults({ │ │ │ │ │ - stroke: false, │ │ │ │ │ - fillOpacity: 1.0 │ │ │ │ │ - }, style), │ │ │ │ │ - featureId │ │ │ │ │ - ); │ │ │ │ │ - this.canvas.globalCompositeOperation = "source-over"; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.globalCompositeOperation = "source-over"; │ │ │ │ │ - } │ │ │ │ │ - this.drawLinearRing( │ │ │ │ │ - components[i], │ │ │ │ │ - OpenLayers.Util.applyDefaults({ │ │ │ │ │ - fill: false │ │ │ │ │ - }, style), │ │ │ │ │ - featureId │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawText │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activates the control. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * location - {<OpenLayers.Point>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The control was effectively activated. │ │ │ │ │ */ │ │ │ │ │ - drawText: function(location, style) { │ │ │ │ │ - var pt = this.getLocalXY(location); │ │ │ │ │ - │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ - this.canvas.fillStyle = style.fontColor; │ │ │ │ │ - this.canvas.globalAlpha = style.fontOpacity || 1.0; │ │ │ │ │ - var fontStyle = [style.fontStyle ? style.fontStyle : "normal", │ │ │ │ │ - "normal", // "font-variant" not supported │ │ │ │ │ - style.fontWeight ? style.fontWeight : "normal", │ │ │ │ │ - style.fontSize ? style.fontSize : "1em", │ │ │ │ │ - style.fontFamily ? style.fontFamily : "sans-serif" │ │ │ │ │ - ].join(" "); │ │ │ │ │ - var labelRows = style.label.split('\n'); │ │ │ │ │ - var numRows = labelRows.length; │ │ │ │ │ - if (this.canvas.fillText) { │ │ │ │ │ - // HTML5 │ │ │ │ │ - this.canvas.font = fontStyle; │ │ │ │ │ - this.canvas.textAlign = │ │ │ │ │ - OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[0]] || │ │ │ │ │ - "center"; │ │ │ │ │ - this.canvas.textBaseline = │ │ │ │ │ - OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[1]] || │ │ │ │ │ - "middle"; │ │ │ │ │ - var vfactor = │ │ │ │ │ - OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5; │ │ │ │ │ - } │ │ │ │ │ - var lineHeight = │ │ │ │ │ - this.canvas.measureText('Mg').height || │ │ │ │ │ - this.canvas.measureText('xx').width; │ │ │ │ │ - pt[1] += lineHeight * vfactor * (numRows - 1); │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - if (style.labelOutlineWidth) { │ │ │ │ │ - this.canvas.save(); │ │ │ │ │ - this.canvas.globalAlpha = style.labelOutlineOpacity || style.fontOpacity || 1.0; │ │ │ │ │ - this.canvas.strokeStyle = style.labelOutlineColor; │ │ │ │ │ - this.canvas.lineWidth = style.labelOutlineWidth; │ │ │ │ │ - this.canvas.strokeText(labelRows[i], pt[0], pt[1] + (lineHeight * i) + 1); │ │ │ │ │ - this.canvas.restore(); │ │ │ │ │ - } │ │ │ │ │ - this.canvas.fillText(labelRows[i], pt[0], pt[1] + (lineHeight * i)); │ │ │ │ │ - } │ │ │ │ │ - } else if (this.canvas.mozDrawText) { │ │ │ │ │ - // Mozilla pre-Gecko1.9.1 (<FF3.1) │ │ │ │ │ - this.canvas.mozTextStyle = fontStyle; │ │ │ │ │ - // No built-in text alignment, so we measure and adjust the position │ │ │ │ │ - var hfactor = │ │ │ │ │ - OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[0]]; │ │ │ │ │ - if (hfactor == null) { │ │ │ │ │ - hfactor = -.5; │ │ │ │ │ - } │ │ │ │ │ - var vfactor = │ │ │ │ │ - OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5; │ │ │ │ │ - } │ │ │ │ │ - var lineHeight = this.canvas.mozMeasureText('xx'); │ │ │ │ │ - pt[1] += lineHeight * (1 + (vfactor * numRows)); │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - var x = pt[0] + (hfactor * this.canvas.mozMeasureText(labelRows[i])); │ │ │ │ │ - var y = pt[1] + (i * lineHeight); │ │ │ │ │ - this.canvas.translate(x, y); │ │ │ │ │ - this.canvas.mozDrawText(labelRows[i]); │ │ │ │ │ - this.canvas.translate(-x, -y); │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (this.available && !this.geolocation) { │ │ │ │ │ + // set lazily to avoid IE9 memory leak │ │ │ │ │ + this.geolocation = navigator.geolocation; │ │ │ │ │ + } │ │ │ │ │ + if (!this.geolocation) { │ │ │ │ │ + this.events.triggerEvent("locationuncapable"); │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + if (this.watch) { │ │ │ │ │ + this.watchId = this.geolocation.watchPosition( │ │ │ │ │ + OpenLayers.Function.bind(this.geolocate, this), │ │ │ │ │ + OpenLayers.Function.bind(this.failure, this), │ │ │ │ │ + this.geolocationOptions │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + this.getCurrentLocation(); │ │ │ │ │ } │ │ │ │ │ + return true; │ │ │ │ │ } │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ + return false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getLocalXY │ │ │ │ │ - * transform geographic xy into pixel xy │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivates the control. │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ - */ │ │ │ │ │ - getLocalXY: function(point) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var extent = this.extent; │ │ │ │ │ - var x = ((point.x - this.featureDx) / resolution + (-extent.left / resolution)); │ │ │ │ │ - var y = ((extent.top / resolution) - point.y / resolution); │ │ │ │ │ - return [x, y]; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: clear │ │ │ │ │ - * Clear all vectors from the renderer. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The control was effectively deactivated. │ │ │ │ │ */ │ │ │ │ │ - clear: function() { │ │ │ │ │ - var height = this.root.height; │ │ │ │ │ - var width = this.root.width; │ │ │ │ │ - this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ - this.features = {}; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.clearRect(0, 0, width, height); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active && this.watchId !== null) { │ │ │ │ │ + this.geolocation.clearWatch(this.watchId); │ │ │ │ │ } │ │ │ │ │ + return OpenLayers.Control.prototype.deactivate.apply( │ │ │ │ │ + this, arguments │ │ │ │ │ + ); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureIdFromEvent │ │ │ │ │ - * Returns a feature id from an event on the renderer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {<OpenLayers.Event>} │ │ │ │ │ + * Method: geolocate │ │ │ │ │ + * Activates the control. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Feature.Vector} A feature or undefined. This method returns a │ │ │ │ │ - * feature instead of a feature id to avoid an unnecessary lookup on the │ │ │ │ │ - * layer. │ │ │ │ │ */ │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var featureId, feature; │ │ │ │ │ - │ │ │ │ │ - if (this.hitDetection && this.root.style.display !== "none") { │ │ │ │ │ - // this dragging check should go in the feature handler │ │ │ │ │ - if (!this.map.dragging) { │ │ │ │ │ - var xy = evt.xy; │ │ │ │ │ - var x = xy.x | 0; │ │ │ │ │ - var y = xy.y | 0; │ │ │ │ │ - var data = this.hitContext.getImageData(x, y, 1, 1).data; │ │ │ │ │ - if (data[3] === 255) { // antialiased │ │ │ │ │ - var id = data[2] + (256 * (data[1] + (256 * data[0]))); │ │ │ │ │ - if (id) { │ │ │ │ │ - featureId = "OpenLayers_Feature_Vector_" + (id - 1 + this.hitOverflow); │ │ │ │ │ - try { │ │ │ │ │ - feature = this.features[featureId][0]; │ │ │ │ │ - } catch (err) { │ │ │ │ │ - // Because of antialiasing on the canvas, when the hit location is at a point where the edge of │ │ │ │ │ - // one symbol intersects the interior of another symbol, a wrong hit color (and therefore id) results. │ │ │ │ │ - // todo: set Antialiasing = 'off' on the hitContext as soon as browsers allow it. │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + geolocate: function(position) { │ │ │ │ │ + var center = new OpenLayers.LonLat( │ │ │ │ │ + position.coords.longitude, │ │ │ │ │ + position.coords.latitude │ │ │ │ │ + ).transform( │ │ │ │ │ + new OpenLayers.Projection("EPSG:4326"), │ │ │ │ │ + this.map.getProjectionObject() │ │ │ │ │ + ); │ │ │ │ │ + if (this.bind) { │ │ │ │ │ + this.map.setCenter(center); │ │ │ │ │ } │ │ │ │ │ - return feature; │ │ │ │ │ + this.events.triggerEvent("locationupdated", { │ │ │ │ │ + position: position, │ │ │ │ │ + point: new OpenLayers.Geometry.Point( │ │ │ │ │ + center.lon, center.lat │ │ │ │ │ + ) │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: eraseFeatures │ │ │ │ │ - * This is called by the layer to erase features; removes the feature from │ │ │ │ │ - * the list, then redraws the layer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ + * APIMethod: getCurrentLocation │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} Returns true if a event will be fired (successfull │ │ │ │ │ + * registration) │ │ │ │ │ */ │ │ │ │ │ - eraseFeatures: function(features) { │ │ │ │ │ - if (!(OpenLayers.Util.isArray(features))) { │ │ │ │ │ - features = [features]; │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0; i < features.length; ++i) { │ │ │ │ │ - delete this.features[features[i].id]; │ │ │ │ │ + getCurrentLocation: function() { │ │ │ │ │ + if (!this.active || this.watch) { │ │ │ │ │ + return false; │ │ │ │ │ } │ │ │ │ │ - this.redraw(); │ │ │ │ │ + this.geolocation.getCurrentPosition( │ │ │ │ │ + OpenLayers.Function.bind(this.geolocate, this), │ │ │ │ │ + OpenLayers.Function.bind(this.failure, this), │ │ │ │ │ + this.geolocationOptions │ │ │ │ │ + ); │ │ │ │ │ + return true; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: redraw │ │ │ │ │ - * The real 'meat' of the function: any time things have changed, │ │ │ │ │ - * redraw() can be called to loop over all the data and (you guessed │ │ │ │ │ - * it) redraw it. Unlike Elements-based Renderers, we can't interact │ │ │ │ │ - * with things once they're drawn, to remove them, for example, so │ │ │ │ │ - * instead we have to just clear everything and draw from scratch. │ │ │ │ │ + * Method: failure │ │ │ │ │ + * method called on browser's geolocation failure │ │ │ │ │ + * │ │ │ │ │ */ │ │ │ │ │ - redraw: function() { │ │ │ │ │ - if (!this.locked) { │ │ │ │ │ - var height = this.root.height; │ │ │ │ │ - var width = this.root.width; │ │ │ │ │ - this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.clearRect(0, 0, width, height); │ │ │ │ │ - } │ │ │ │ │ - var labelMap = []; │ │ │ │ │ - var feature, geometry, style; │ │ │ │ │ - var worldBounds = (this.map.baseLayer && this.map.baseLayer.wrapDateLine) && this.map.getMaxExtent(); │ │ │ │ │ - for (var id in this.features) { │ │ │ │ │ - if (!this.features.hasOwnProperty(id)) { │ │ │ │ │ - continue; │ │ │ │ │ - } │ │ │ │ │ - feature = this.features[id][0]; │ │ │ │ │ - geometry = feature.geometry; │ │ │ │ │ - this.calculateFeatureDx(geometry.getBounds(), worldBounds); │ │ │ │ │ - style = this.features[id][1]; │ │ │ │ │ - this.drawGeometry(geometry, style, feature.id); │ │ │ │ │ - if (style.label) { │ │ │ │ │ - labelMap.push([feature, style]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var item; │ │ │ │ │ - for (var i = 0, len = labelMap.length; i < len; ++i) { │ │ │ │ │ - item = labelMap[i]; │ │ │ │ │ - this.drawText(item[0].geometry.getCentroid(), item[1]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + failure: function(error) { │ │ │ │ │ + this.events.triggerEvent("locationfailed", { │ │ │ │ │ + error: error │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.Canvas" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Geolocate" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.Canvas.LABEL_ALIGN │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.Canvas.LABEL_ALIGN = { │ │ │ │ │ - "l": "left", │ │ │ │ │ - "r": "right", │ │ │ │ │ - "t": "top", │ │ │ │ │ - "b": "bottom" │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.Canvas.LABEL_FACTOR │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.Canvas.LABEL_FACTOR = { │ │ │ │ │ - "l": 0, │ │ │ │ │ - "r": -1, │ │ │ │ │ - "t": 0, │ │ │ │ │ - "b": -1 │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.Canvas.drawImageScaleFactor │ │ │ │ │ - * {Number} Scale factor to apply to the canvas drawImage arguments. This │ │ │ │ │ - * is always 1 except for Android 2.1 devices, to work around │ │ │ │ │ - * http://code.google.com/p/android/issues/detail?id=5141. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.Canvas.drawImageScaleFactor = null; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Renderer/VML.js │ │ │ │ │ + OpenLayers/Control/UTFGrid.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Renderer/Elements.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Handler/Hover.js │ │ │ │ │ + * @requires OpenLayers/Handler/Click.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Renderer.VML │ │ │ │ │ - * Render vector features in browsers with VML capability. Construct a new │ │ │ │ │ - * VML renderer with the <OpenLayers.Renderer.VML> constructor. │ │ │ │ │ + * Class: OpenLayers.Control.UTFGrid │ │ │ │ │ + * │ │ │ │ │ + * This Control provides behavior associated with UTFGrid Layers. │ │ │ │ │ + * These 'hit grids' provide underlying feature attributes without │ │ │ │ │ + * calling the server (again). This control allows Mousemove, Hovering │ │ │ │ │ + * and Click events to trigger callbacks that use the attributes in │ │ │ │ │ + * whatever way you need. │ │ │ │ │ + * │ │ │ │ │ + * The most common example may be a UTFGrid layer containing feature │ │ │ │ │ + * attributes that are displayed in a div as you mouseover. │ │ │ │ │ + * │ │ │ │ │ + * Example Code: │ │ │ │ │ + * │ │ │ │ │ + * (start code) │ │ │ │ │ + * var world_utfgrid = new OpenLayers.Layer.UTFGrid( │ │ │ │ │ + * 'UTFGrid Layer', │ │ │ │ │ + * "http://tiles/world_utfgrid/${z}/${x}/${y}.json" │ │ │ │ │ + * ); │ │ │ │ │ + * map.addLayer(world_utfgrid); │ │ │ │ │ * │ │ │ │ │ - * Note that for all calculations in this class, we use (num | 0) to truncate a │ │ │ │ │ - * float value to an integer. This is done because it seems that VML doesn't │ │ │ │ │ - * support float values. │ │ │ │ │ + * var control = new OpenLayers.Control.UTFGrid({ │ │ │ │ │ + * layers: [world_utfgrid], │ │ │ │ │ + * handlerMode: 'move', │ │ │ │ │ + * callback: function(infoLookup) { │ │ │ │ │ + * // do something with returned data │ │ │ │ │ + * │ │ │ │ │ + * } │ │ │ │ │ + * }) │ │ │ │ │ + * (end code) │ │ │ │ │ + * │ │ │ │ │ * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Renderer.Elements> │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Renderer.VML = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ +OpenLayers.Control.UTFGrid = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: xmlns │ │ │ │ │ - * {String} XML Namespace URN │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ */ │ │ │ │ │ - xmlns: "urn:schemas-microsoft-com:vml", │ │ │ │ │ + autoActivate: true, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: symbolCache │ │ │ │ │ - * {DOMElement} node holding symbols. This hash is keyed by symbol name, │ │ │ │ │ - * and each value is a hash with a "path" and an "extent" property. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: Layers │ │ │ │ │ + * List of layers to consider. Must be Layer.UTFGrids │ │ │ │ │ + * `null` is the default indicating all UTFGrid Layers are queried. │ │ │ │ │ + * {Array} <OpenLayers.Layer.UTFGrid> │ │ │ │ │ */ │ │ │ │ │ - symbolCache: {}, │ │ │ │ │ + layers: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: offset │ │ │ │ │ - * {Object} Hash with "x" and "y" properties │ │ │ │ │ + /* Property: defaultHandlerOptions │ │ │ │ │ + * The default opts passed to the handler constructors │ │ │ │ │ */ │ │ │ │ │ - offset: null, │ │ │ │ │ + defaultHandlerOptions: { │ │ │ │ │ + 'delay': 300, │ │ │ │ │ + 'pixelTolerance': 4, │ │ │ │ │ + 'stopMove': false, │ │ │ │ │ + 'single': true, │ │ │ │ │ + 'double': false, │ │ │ │ │ + 'stopSingle': false, │ │ │ │ │ + 'stopDouble': false │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /* APIProperty: handlerMode │ │ │ │ │ + * Defaults to 'click'. Can be 'hover' or 'move'. │ │ │ │ │ + */ │ │ │ │ │ + handlerMode: 'click', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Renderer.VML │ │ │ │ │ - * Create a new VML renderer. │ │ │ │ │ + * APIMethod: setHandler │ │ │ │ │ + * sets this.handlerMode and calls resetHandler() │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * containerID - {String} The id for the element that contains the renderer │ │ │ │ │ + * hm - {String} Handler Mode string; 'click', 'hover' or 'move'. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(containerID) { │ │ │ │ │ - if (!this.supported()) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - if (!document.namespaces.olv) { │ │ │ │ │ - document.namespaces.add("olv", this.xmlns); │ │ │ │ │ - var style = document.createStyleSheet(); │ │ │ │ │ - var shapes = ['shape', 'rect', 'oval', 'fill', 'stroke', 'imagedata', 'group', 'textbox']; │ │ │ │ │ - for (var i = 0, len = shapes.length; i < len; i++) { │ │ │ │ │ + setHandler: function(hm) { │ │ │ │ │ + this.handlerMode = hm; │ │ │ │ │ + this.resetHandler(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - style.addRule('olv\\:' + shapes[i], "behavior: url(#default#VML); " + │ │ │ │ │ - "position: absolute; display: inline-block;"); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Method: resetHandler │ │ │ │ │ + * Deactivates the old hanlder and creates a new │ │ │ │ │ + * <OpenLayers.Handler> based on the mode specified in │ │ │ │ │ + * this.handlerMode │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + resetHandler: function() { │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.deactivate(); │ │ │ │ │ + this.handler.destroy(); │ │ │ │ │ + this.handler = null; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.initialize.apply(this, │ │ │ │ │ - arguments); │ │ │ │ │ + if (this.handlerMode == 'hover') { │ │ │ │ │ + // Handle this event on hover │ │ │ │ │ + this.handler = new OpenLayers.Handler.Hover( │ │ │ │ │ + this, { │ │ │ │ │ + 'pause': this.handleEvent, │ │ │ │ │ + 'move': this.reset │ │ │ │ │ + }, │ │ │ │ │ + this.handlerOptions │ │ │ │ │ + ); │ │ │ │ │ + } else if (this.handlerMode == 'click') { │ │ │ │ │ + // Handle this event on click │ │ │ │ │ + this.handler = new OpenLayers.Handler.Click( │ │ │ │ │ + this, { │ │ │ │ │ + 'click': this.handleEvent │ │ │ │ │ + }, this.handlerOptions │ │ │ │ │ + ); │ │ │ │ │ + } else if (this.handlerMode == 'move') { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Hover( │ │ │ │ │ + this, │ │ │ │ │ + // Handle this event while hovering OR moving │ │ │ │ │ + { │ │ │ │ │ + 'pause': this.handleEvent, │ │ │ │ │ + 'move': this.handleEvent │ │ │ │ │ + }, │ │ │ │ │ + this.handlerOptions │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + return true; │ │ │ │ │ + } else { │ │ │ │ │ + return false; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: supported │ │ │ │ │ - * Determine whether a browser supports this renderer. │ │ │ │ │ + * Constructor: <OpenLayers.Control.UTFGrid> │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} The browser supports the VML renderer │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - supported: function() { │ │ │ │ │ - return !!(document.namespaces); │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.handlerOptions = options.handlerOptions || this.defaultHandlerOptions; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.resetHandler(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setExtent │ │ │ │ │ - * Set the renderer's extent │ │ │ │ │ + * Method: handleEvent │ │ │ │ │ + * Internal method called when specified event is triggered. │ │ │ │ │ + * │ │ │ │ │ + * This method does several things: │ │ │ │ │ + * │ │ │ │ │ + * Gets the lonLat of the event. │ │ │ │ │ + * │ │ │ │ │ + * Loops through the appropriate hit grid layers and gathers the attributes. │ │ │ │ │ + * │ │ │ │ │ + * Passes the attributes to the callback │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} │ │ │ │ │ - * resolutionChanged - {Boolean} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ - * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ + * evt - {<OpenLayers.Event>} │ │ │ │ │ */ │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - │ │ │ │ │ - var left = (extent.left / resolution) | 0; │ │ │ │ │ - var top = (extent.top / resolution - this.size.h) | 0; │ │ │ │ │ - if (resolutionChanged || !this.offset) { │ │ │ │ │ - this.offset = { │ │ │ │ │ - x: left, │ │ │ │ │ - y: top │ │ │ │ │ - }; │ │ │ │ │ - left = 0; │ │ │ │ │ - top = 0; │ │ │ │ │ - } else { │ │ │ │ │ - left = left - this.offset.x; │ │ │ │ │ - top = top - this.offset.y; │ │ │ │ │ + handleEvent: function(evt) { │ │ │ │ │ + if (evt == null) { │ │ │ │ │ + this.reset(); │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ - var org = (left - this.xOffset) + " " + top; │ │ │ │ │ - this.root.coordorigin = org; │ │ │ │ │ - var roots = [this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ - var root; │ │ │ │ │ - for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ - root = roots[i]; │ │ │ │ │ - │ │ │ │ │ - var size = this.size.w + " " + this.size.h; │ │ │ │ │ - root.coordsize = size; │ │ │ │ │ - │ │ │ │ │ + var lonLat = this.map.getLonLatFromPixel(evt.xy); │ │ │ │ │ + if (!lonLat) { │ │ │ │ │ + return; │ │ │ │ │ } │ │ │ │ │ - // flip the VML display Y axis upside down so it │ │ │ │ │ - // matches the display Y axis of the map │ │ │ │ │ - this.root.style.flip = "y"; │ │ │ │ │ │ │ │ │ │ - return coordSysUnchanged; │ │ │ │ │ + var layers = this.findLayers(); │ │ │ │ │ + if (layers.length > 0) { │ │ │ │ │ + var infoLookup = {}; │ │ │ │ │ + var layer, idx; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + layer = layers[i]; │ │ │ │ │ + idx = OpenLayers.Util.indexOf(this.map.layers, layer); │ │ │ │ │ + infoLookup[idx] = layer.getFeatureInfo(lonLat); │ │ │ │ │ + } │ │ │ │ │ + this.callback(infoLookup, lonLat, evt.xy); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: setSize │ │ │ │ │ - * Set the size of the drawing surface │ │ │ │ │ + * APIMethod: callback │ │ │ │ │ + * Function to be called when a mouse event corresponds with a location that │ │ │ │ │ + * includes data in one of the configured UTFGrid layers. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>} the size of the drawing surface │ │ │ │ │ + * infoLookup - {Object} Keys of this object are layer indexes and can be │ │ │ │ │ + * used to resolve a layer in the map.layers array. The structure of │ │ │ │ │ + * the property values depend on the data included in the underlying │ │ │ │ │ + * UTFGrid and may be any valid JSON type. │ │ │ │ │ */ │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - // setting width and height on all roots to avoid flicker which we │ │ │ │ │ - // would get with 100% width and height on child roots │ │ │ │ │ - var roots = [ │ │ │ │ │ - this.rendererRoot, │ │ │ │ │ - this.root, │ │ │ │ │ - this.vectorRoot, │ │ │ │ │ - this.textRoot │ │ │ │ │ - ]; │ │ │ │ │ - var w = this.size.w + "px"; │ │ │ │ │ - var h = this.size.h + "px"; │ │ │ │ │ - var root; │ │ │ │ │ - for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ - root = roots[i]; │ │ │ │ │ - root.style.width = w; │ │ │ │ │ - root.style.height = h; │ │ │ │ │ - } │ │ │ │ │ + callback: function(infoLookup) { │ │ │ │ │ + // to be provided in the constructor │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getNodeType │ │ │ │ │ - * Get the node type for a geometry and style │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The corresponding node type for the specified geometry │ │ │ │ │ + * Method: reset │ │ │ │ │ + * Calls the callback with null. │ │ │ │ │ */ │ │ │ │ │ - getNodeType: function(geometry, style) { │ │ │ │ │ - var nodeType = null; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - nodeType = "olv:rect"; │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - nodeType = "olv:shape"; │ │ │ │ │ - } else { │ │ │ │ │ - nodeType = "olv:oval"; │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - nodeType = "olv:rect"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - case "OpenLayers.Geometry.Curve": │ │ │ │ │ - nodeType = "olv:shape"; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - return nodeType; │ │ │ │ │ + reset: function(evt) { │ │ │ │ │ + this.callback(null); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setStyle │ │ │ │ │ - * Use to set all the style attributes to a VML node. │ │ │ │ │ + * Method: findLayers │ │ │ │ │ + * Internal method to get the layers, independent of whether we are │ │ │ │ │ + * inspecting the map or using a client-provided array │ │ │ │ │ + * │ │ │ │ │ + * The default value of this.layers is null; this causes the │ │ │ │ │ + * findLayers method to return ALL UTFGrid layers encountered. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} An VML element to decorate │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * options - {Object} Currently supported options include │ │ │ │ │ - * 'isFilled' {Boolean} and │ │ │ │ │ - * 'isStroked' {Boolean} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * None │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Array} Layers to handle on each event │ │ │ │ │ */ │ │ │ │ │ - setStyle: function(node, style, options, geometry) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - options = options || node._options; │ │ │ │ │ - var fillColor = style.fillColor; │ │ │ │ │ - │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - node.title = title; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - options.isFilled = true; │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ - style.graphicXOffset : -(0.5 * width); │ │ │ │ │ - var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ - style.graphicYOffset : -(0.5 * height); │ │ │ │ │ - │ │ │ │ │ - node.style.left = ((((geometry.x - this.featureDx) / resolution - this.offset.x) + xOffset) | 0) + "px"; │ │ │ │ │ - node.style.top = (((geometry.y / resolution - this.offset.y) - (yOffset + height)) | 0) + "px"; │ │ │ │ │ - node.style.width = width + "px"; │ │ │ │ │ - node.style.height = height + "px"; │ │ │ │ │ - node.style.flip = "y"; │ │ │ │ │ - │ │ │ │ │ - // modify fillColor and options for stroke styling below │ │ │ │ │ - fillColor = "none"; │ │ │ │ │ - options.isStroked = false; │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - var cache = this.importSymbol(style.graphicName); │ │ │ │ │ - node.path = cache.path; │ │ │ │ │ - node.coordorigin = cache.left + "," + cache.bottom; │ │ │ │ │ - var size = cache.size; │ │ │ │ │ - node.coordsize = size + "," + size; │ │ │ │ │ - this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ - node.style.flip = "y"; │ │ │ │ │ - } else { │ │ │ │ │ - this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ + findLayers: function() { │ │ │ │ │ + var candidates = this.layers || this.map.layers; │ │ │ │ │ + var layers = []; │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = candidates[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.UTFGrid) { │ │ │ │ │ + layers.push(layer); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return layers; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // fill │ │ │ │ │ - if (options.isFilled) { │ │ │ │ │ - node.fillcolor = fillColor; │ │ │ │ │ - } else { │ │ │ │ │ - node.filled = "false"; │ │ │ │ │ - } │ │ │ │ │ - var fills = node.getElementsByTagName("fill"); │ │ │ │ │ - var fill = (fills.length == 0) ? null : fills[0]; │ │ │ │ │ - if (!options.isFilled) { │ │ │ │ │ - if (fill) { │ │ │ │ │ - node.removeChild(fill); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - if (!fill) { │ │ │ │ │ - fill = this.createNode('olv:fill', node.id + "_fill"); │ │ │ │ │ - } │ │ │ │ │ - fill.opacity = style.fillOpacity; │ │ │ │ │ - │ │ │ │ │ - if (node._geometryClass === "OpenLayers.Geometry.Point" && │ │ │ │ │ - style.externalGraphic) { │ │ │ │ │ - │ │ │ │ │ - // override fillOpacity │ │ │ │ │ - if (style.graphicOpacity) { │ │ │ │ │ - fill.opacity = style.graphicOpacity; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - fill.src = style.externalGraphic; │ │ │ │ │ - fill.type = "frame"; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.UTFGrid" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/NavigationHistory.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ - fill.aspect = "atmost"; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (fill.parentNode != node) { │ │ │ │ │ - node.appendChild(fill); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - // additional rendering for rotated graphics or symbols │ │ │ │ │ - var rotation = style.rotation; │ │ │ │ │ - if ((rotation !== undefined || node._rotation !== undefined)) { │ │ │ │ │ - node._rotation = rotation; │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ - // make the fill fully transparent, because we now have │ │ │ │ │ - // the graphic as imagedata element. We cannot just remove │ │ │ │ │ - // the fill, because this is part of the hack described │ │ │ │ │ - // in graphicRotate │ │ │ │ │ - fill.opacity = 0; │ │ │ │ │ - } else if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ - node.style.rotation = rotation || 0; │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Control/Button.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // stroke │ │ │ │ │ - var strokes = node.getElementsByTagName("stroke"); │ │ │ │ │ - var stroke = (strokes.length == 0) ? null : strokes[0]; │ │ │ │ │ - if (!options.isStroked) { │ │ │ │ │ - node.stroked = false; │ │ │ │ │ - if (stroke) { │ │ │ │ │ - stroke.on = false; │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - if (!stroke) { │ │ │ │ │ - stroke = this.createNode('olv:stroke', node.id + "_stroke"); │ │ │ │ │ - node.appendChild(stroke); │ │ │ │ │ - } │ │ │ │ │ - stroke.on = true; │ │ │ │ │ - stroke.color = style.strokeColor; │ │ │ │ │ - stroke.weight = style.strokeWidth + "px"; │ │ │ │ │ - stroke.opacity = style.strokeOpacity; │ │ │ │ │ - stroke.endcap = style.strokeLinecap == 'butt' ? 'flat' : │ │ │ │ │ - (style.strokeLinecap || 'round'); │ │ │ │ │ - if (style.strokeDashstyle) { │ │ │ │ │ - stroke.dashstyle = this.dashStyle(style); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.NavigationHistory │ │ │ │ │ + * A navigation history control. This is a meta-control, that creates two │ │ │ │ │ + * dependent controls: <previous> and <next>. Call the trigger method │ │ │ │ │ + * on the <previous> and <next> controls to restore previous and next │ │ │ │ │ + * history states. The previous and next controls will become active │ │ │ │ │ + * when there are available states to restore and will become deactive │ │ │ │ │ + * when there are no states to restore. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.NavigationHistory = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ - node.style.cursor = style.cursor; │ │ │ │ │ - } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: type │ │ │ │ │ + * {String} Note that this control is not intended to be added directly │ │ │ │ │ + * to a control panel. Instead, add the sub-controls previous and │ │ │ │ │ + * next. These sub-controls are button type controls that activate │ │ │ │ │ + * and deactivate themselves. If this parent control is added to │ │ │ │ │ + * a panel, it will act as a toggle. │ │ │ │ │ + */ │ │ │ │ │ + type: OpenLayers.Control.TYPE_TOGGLE, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: graphicRotate │ │ │ │ │ - * If a point is to be styled with externalGraphic and rotation, VML fills │ │ │ │ │ - * cannot be used to display the graphic, because rotation of graphic │ │ │ │ │ - * fills is not supported by the VML implementation of Internet Explorer. │ │ │ │ │ - * This method creates a olv:imagedata element inside the VML node, │ │ │ │ │ - * DXImageTransform.Matrix and BasicImage filters for rotation and │ │ │ │ │ - * opacity, and a 3-step hack to remove rendering artefacts from the │ │ │ │ │ - * graphic and preserve the ability of graphics to trigger events. │ │ │ │ │ - * Finally, OpenLayers methods are used to determine the correct │ │ │ │ │ - * insertion point of the rotated image, because DXImageTransform.Matrix │ │ │ │ │ - * does the rotation without the ability to specify a rotation center │ │ │ │ │ - * point. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * xOffset - {Number} rotation center relative to image, x coordinate │ │ │ │ │ - * yOffset - {Number} rotation center relative to image, y coordinate │ │ │ │ │ - * style - {Object} │ │ │ │ │ + * APIProperty: previous │ │ │ │ │ + * {<OpenLayers.Control>} A button type control whose trigger method restores │ │ │ │ │ + * the previous state managed by this control. │ │ │ │ │ */ │ │ │ │ │ - graphicRotate: function(node, xOffset, yOffset, style) { │ │ │ │ │ - var style = style || node._style; │ │ │ │ │ - var rotation = style.rotation || 0; │ │ │ │ │ + previous: null, │ │ │ │ │ │ │ │ │ │ - var aspectRatio, size; │ │ │ │ │ - if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ - // load the image to determine its size │ │ │ │ │ - var img = new Image(); │ │ │ │ │ - img.onreadystatechange = OpenLayers.Function.bind(function() { │ │ │ │ │ - if (img.readyState == "complete" || │ │ │ │ │ - img.readyState == "interactive") { │ │ │ │ │ - aspectRatio = img.width / img.height; │ │ │ │ │ - size = Math.max(style.pointRadius * 2, │ │ │ │ │ - style.graphicWidth || 0, │ │ │ │ │ - style.graphicHeight || 0); │ │ │ │ │ - xOffset = xOffset * aspectRatio; │ │ │ │ │ - style.graphicWidth = size * aspectRatio; │ │ │ │ │ - style.graphicHeight = size; │ │ │ │ │ - this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ - } │ │ │ │ │ - }, this); │ │ │ │ │ - img.src = style.externalGraphic; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: previousOptions │ │ │ │ │ + * {Object} Set this property on the options argument of the constructor │ │ │ │ │ + * to set optional properties on the <previous> control. │ │ │ │ │ + */ │ │ │ │ │ + previousOptions: null, │ │ │ │ │ │ │ │ │ │ - // will be called again by the onreadystate handler │ │ │ │ │ - return; │ │ │ │ │ - } else { │ │ │ │ │ - size = Math.max(style.graphicWidth, style.graphicHeight); │ │ │ │ │ - aspectRatio = style.graphicWidth / style.graphicHeight; │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: next │ │ │ │ │ + * {<OpenLayers.Control>} A button type control whose trigger method restores │ │ │ │ │ + * the next state managed by this control. │ │ │ │ │ + */ │ │ │ │ │ + next: null, │ │ │ │ │ │ │ │ │ │ - var width = Math.round(style.graphicWidth || size * aspectRatio); │ │ │ │ │ - var height = Math.round(style.graphicHeight || size); │ │ │ │ │ - node.style.width = width + "px"; │ │ │ │ │ - node.style.height = height + "px"; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: nextOptions │ │ │ │ │ + * {Object} Set this property on the options argument of the constructor │ │ │ │ │ + * to set optional properties on the <next> control. │ │ │ │ │ + */ │ │ │ │ │ + nextOptions: null, │ │ │ │ │ │ │ │ │ │ - // Three steps are required to remove artefacts for images with │ │ │ │ │ - // transparent backgrounds (resulting from using DXImageTransform │ │ │ │ │ - // filters on svg objects), while preserving awareness for browser │ │ │ │ │ - // events on images: │ │ │ │ │ - // - Use the fill as usual (like for unrotated images) to handle │ │ │ │ │ - // events │ │ │ │ │ - // - specify an imagedata element with the same src as the fill │ │ │ │ │ - // - style the imagedata element with an AlphaImageLoader filter │ │ │ │ │ - // with empty src │ │ │ │ │ - var image = document.getElementById(node.id + "_image"); │ │ │ │ │ - if (!image) { │ │ │ │ │ - image = this.createNode("olv:imagedata", node.id + "_image"); │ │ │ │ │ - node.appendChild(image); │ │ │ │ │ - } │ │ │ │ │ - image.style.width = width + "px"; │ │ │ │ │ - image.style.height = height + "px"; │ │ │ │ │ - image.src = style.externalGraphic; │ │ │ │ │ - image.style.filter = │ │ │ │ │ - "progid:DXImageTransform.Microsoft.AlphaImageLoader(" + │ │ │ │ │ - "src='', sizingMethod='scale')"; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: limit │ │ │ │ │ + * {Integer} Optional limit on the number of history items to retain. If │ │ │ │ │ + * null, there is no limit. Default is 50. │ │ │ │ │ + */ │ │ │ │ │ + limit: 50, │ │ │ │ │ │ │ │ │ │ - var rot = rotation * Math.PI / 180; │ │ │ │ │ - var sintheta = Math.sin(rot); │ │ │ │ │ - var costheta = Math.cos(rot); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: autoActivate │ │ │ │ │ + * {Boolean} Activate the control when it is added to a map. Default is │ │ │ │ │ + * true. │ │ │ │ │ + */ │ │ │ │ │ + autoActivate: true, │ │ │ │ │ │ │ │ │ │ - // do the rotation on the image │ │ │ │ │ - var filter = │ │ │ │ │ - "progid:DXImageTransform.Microsoft.Matrix(M11=" + costheta + │ │ │ │ │ - ",M12=" + (-sintheta) + ",M21=" + sintheta + ",M22=" + costheta + │ │ │ │ │ - ",SizingMethod='auto expand')\n"; │ │ │ │ │ + /** │ │ │ │ │ + * Property: clearOnDeactivate │ │ │ │ │ + * {Boolean} Clear the history when the control is deactivated. Default │ │ │ │ │ + * is false. │ │ │ │ │ + */ │ │ │ │ │ + clearOnDeactivate: false, │ │ │ │ │ │ │ │ │ │ - // set the opacity (needed for the imagedata) │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ - if (opacity && opacity != 1) { │ │ │ │ │ - filter += │ │ │ │ │ - "progid:DXImageTransform.Microsoft.BasicImage(opacity=" + │ │ │ │ │ - opacity + ")\n"; │ │ │ │ │ - } │ │ │ │ │ - node.style.filter = filter; │ │ │ │ │ + /** │ │ │ │ │ + * Property: registry │ │ │ │ │ + * {Object} An object with keys corresponding to event types. Values │ │ │ │ │ + * are functions that return an object representing the current state. │ │ │ │ │ + */ │ │ │ │ │ + registry: null, │ │ │ │ │ │ │ │ │ │ - // do the rotation again on a box, so we know the insertion point │ │ │ │ │ - var centerPoint = new OpenLayers.Geometry.Point(-xOffset, -yOffset); │ │ │ │ │ - var imgBox = new OpenLayers.Bounds(0, 0, width, height).toGeometry(); │ │ │ │ │ - imgBox.rotate(style.rotation, centerPoint); │ │ │ │ │ - var imgBounds = imgBox.getBounds(); │ │ │ │ │ + /** │ │ │ │ │ + * Property: nextStack │ │ │ │ │ + * {Array} Array of items in the history. │ │ │ │ │ + */ │ │ │ │ │ + nextStack: null, │ │ │ │ │ │ │ │ │ │ - node.style.left = Math.round( │ │ │ │ │ - parseInt(node.style.left) + imgBounds.left) + "px"; │ │ │ │ │ - node.style.top = Math.round( │ │ │ │ │ - parseInt(node.style.top) - imgBounds.bottom) + "px"; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: previousStack │ │ │ │ │ + * {Array} List of items in the history. First item represents the current │ │ │ │ │ + * state. │ │ │ │ │ + */ │ │ │ │ │ + previousStack: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: postDraw │ │ │ │ │ - * Does some node postprocessing to work around browser issues: │ │ │ │ │ - * - Some versions of Internet Explorer seem to be unable to set fillcolor │ │ │ │ │ - * and strokecolor to "none" correctly before the fill node is appended │ │ │ │ │ - * to a visible vml node. This method takes care of that and sets │ │ │ │ │ - * fillcolor and strokecolor again if needed. │ │ │ │ │ - * - In some cases, a node won't become visible after being drawn. Setting │ │ │ │ │ - * style.visibility to "visible" works around that. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ + * Property: listeners │ │ │ │ │ + * {Object} An object containing properties corresponding to event types. │ │ │ │ │ + * This object is used to configure the control and is modified on │ │ │ │ │ + * construction. │ │ │ │ │ */ │ │ │ │ │ - postDraw: function(node) { │ │ │ │ │ - node.style.visibility = "visible"; │ │ │ │ │ - var fillColor = node._style.fillColor; │ │ │ │ │ - var strokeColor = node._style.strokeColor; │ │ │ │ │ - if (fillColor == "none" && │ │ │ │ │ - node.fillcolor != fillColor) { │ │ │ │ │ - node.fillcolor = fillColor; │ │ │ │ │ - } │ │ │ │ │ - if (strokeColor == "none" && │ │ │ │ │ - node.strokecolor != strokeColor) { │ │ │ │ │ - node.strokecolor = strokeColor; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + listeners: null, │ │ │ │ │ │ │ │ │ │ + /** │ │ │ │ │ + * Property: restoring │ │ │ │ │ + * {Boolean} Currently restoring a history state. This is set to true │ │ │ │ │ + * before calling restore and set to false after restore returns. │ │ │ │ │ + */ │ │ │ │ │ + restoring: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setNodeDimension │ │ │ │ │ - * Get the geometry's bounds, convert it to our vml coordinate system, │ │ │ │ │ - * then set the node's position, size, and local coordinate system. │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Control.NavigationHistory │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ + * options - {Object} An optional object whose properties will be used │ │ │ │ │ + * to extend the control. │ │ │ │ │ */ │ │ │ │ │ - setNodeDimension: function(node, geometry) { │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ │ │ │ │ │ - var bbox = geometry.getBounds(); │ │ │ │ │ - if (bbox) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ + this.registry = OpenLayers.Util.extend({ │ │ │ │ │ + "moveend": this.getState │ │ │ │ │ + }, this.registry); │ │ │ │ │ │ │ │ │ │ - var scaledBox = │ │ │ │ │ - new OpenLayers.Bounds(((bbox.left - this.featureDx) / resolution - this.offset.x) | 0, │ │ │ │ │ - (bbox.bottom / resolution - this.offset.y) | 0, │ │ │ │ │ - ((bbox.right - this.featureDx) / resolution - this.offset.x) | 0, │ │ │ │ │ - (bbox.top / resolution - this.offset.y) | 0); │ │ │ │ │ + var previousOptions = { │ │ │ │ │ + trigger: OpenLayers.Function.bind(this.previousTrigger, this), │ │ │ │ │ + displayClass: this.displayClass + " " + this.displayClass + "Previous" │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Util.extend(previousOptions, this.previousOptions); │ │ │ │ │ + this.previous = new OpenLayers.Control.Button(previousOptions); │ │ │ │ │ │ │ │ │ │ - // Set the internal coordinate system to draw the path │ │ │ │ │ - node.style.left = scaledBox.left + "px"; │ │ │ │ │ - node.style.top = scaledBox.top + "px"; │ │ │ │ │ - node.style.width = scaledBox.getWidth() + "px"; │ │ │ │ │ - node.style.height = scaledBox.getHeight() + "px"; │ │ │ │ │ + var nextOptions = { │ │ │ │ │ + trigger: OpenLayers.Function.bind(this.nextTrigger, this), │ │ │ │ │ + displayClass: this.displayClass + " " + this.displayClass + "Next" │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Util.extend(nextOptions, this.nextOptions); │ │ │ │ │ + this.next = new OpenLayers.Control.Button(nextOptions); │ │ │ │ │ │ │ │ │ │ - node.coordorigin = scaledBox.left + " " + scaledBox.top; │ │ │ │ │ - node.coordsize = scaledBox.getWidth() + " " + scaledBox.getHeight(); │ │ │ │ │ - } │ │ │ │ │ + this.clear(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: dashStyle │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: onPreviousChange │ │ │ │ │ + * Called when the previous history stack changes. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A VML compliant 'stroke-dasharray' value │ │ │ │ │ + * state - {Object} An object representing the state to be restored │ │ │ │ │ + * if previous is triggered again or null if no previous states remain. │ │ │ │ │ + * length - {Integer} The number of remaining previous states that can │ │ │ │ │ + * be restored. │ │ │ │ │ */ │ │ │ │ │ - dashStyle: function(style) { │ │ │ │ │ - var dash = style.strokeDashstyle; │ │ │ │ │ - switch (dash) { │ │ │ │ │ - case 'solid': │ │ │ │ │ - case 'dot': │ │ │ │ │ - case 'dash': │ │ │ │ │ - case 'dashdot': │ │ │ │ │ - case 'longdash': │ │ │ │ │ - case 'longdashdot': │ │ │ │ │ - return dash; │ │ │ │ │ - default: │ │ │ │ │ - // very basic guessing of dash style patterns │ │ │ │ │ - var parts = dash.split(/[ ,]/); │ │ │ │ │ - if (parts.length == 2) { │ │ │ │ │ - if (1 * parts[0] >= 2 * parts[1]) { │ │ │ │ │ - return "longdash"; │ │ │ │ │ - } │ │ │ │ │ - return (parts[0] == 1 || parts[1] == 1) ? "dot" : "dash"; │ │ │ │ │ - } else if (parts.length == 4) { │ │ │ │ │ - return (1 * parts[0] >= 2 * parts[1]) ? "longdashdot" : │ │ │ │ │ - "dashdot"; │ │ │ │ │ - } │ │ │ │ │ - return "solid"; │ │ │ │ │ + onPreviousChange: function(state, length) { │ │ │ │ │ + if (state && !this.previous.active) { │ │ │ │ │ + this.previous.activate(); │ │ │ │ │ + } else if (!state && this.previous.active) { │ │ │ │ │ + this.previous.deactivate(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createNode │ │ │ │ │ - * Create a new node │ │ │ │ │ + * Method: onNextChange │ │ │ │ │ + * Called when the next history stack changes. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * type - {String} Kind of node to draw │ │ │ │ │ - * id - {String} Id for node │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A new node of the given type and id │ │ │ │ │ + * state - {Object} An object representing the state to be restored │ │ │ │ │ + * if next is triggered again or null if no next states remain. │ │ │ │ │ + * length - {Integer} The number of remaining next states that can │ │ │ │ │ + * be restored. │ │ │ │ │ */ │ │ │ │ │ - createNode: function(type, id) { │ │ │ │ │ - var node = document.createElement(type); │ │ │ │ │ - if (id) { │ │ │ │ │ - node.id = id; │ │ │ │ │ + onNextChange: function(state, length) { │ │ │ │ │ + if (state && !this.next.active) { │ │ │ │ │ + this.next.activate(); │ │ │ │ │ + } else if (!state && this.next.active) { │ │ │ │ │ + this.next.deactivate(); │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - // IE hack to make elements unselectable, to prevent 'blue flash' │ │ │ │ │ - // while dragging vectors; #1410 │ │ │ │ │ - node.unselectable = 'on'; │ │ │ │ │ - node.onselectstart = OpenLayers.Function.False; │ │ │ │ │ - │ │ │ │ │ - return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: nodeTypeCompare │ │ │ │ │ - * Determine whether a node is of a given type │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} An VML element │ │ │ │ │ - * type - {String} Kind of node │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Destroy the control. │ │ │ │ │ */ │ │ │ │ │ - nodeTypeCompare: function(node, type) { │ │ │ │ │ - │ │ │ │ │ - //split type │ │ │ │ │ - var subType = type; │ │ │ │ │ - var splitIndex = subType.indexOf(":"); │ │ │ │ │ - if (splitIndex != -1) { │ │ │ │ │ - subType = subType.substr(splitIndex + 1); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - //split nodeName │ │ │ │ │ - var nodeName = node.nodeName; │ │ │ │ │ - splitIndex = nodeName.indexOf(":"); │ │ │ │ │ - if (splitIndex != -1) { │ │ │ │ │ - nodeName = nodeName.substr(splitIndex + 1); │ │ │ │ │ + destroy: function() { │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this); │ │ │ │ │ + this.previous.destroy(); │ │ │ │ │ + this.next.destroy(); │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + for (var prop in this) { │ │ │ │ │ + this[prop] = null; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return (subType == nodeName); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createRenderRoot │ │ │ │ │ - * Create the renderer root │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * Set the map property for the control and <previous> and <next> child │ │ │ │ │ + * controls. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The specific render engine's root element │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ */ │ │ │ │ │ - createRenderRoot: function() { │ │ │ │ │ - return this.nodeFactory(this.container.id + "_vmlRoot", "div"); │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.map = map; │ │ │ │ │ + this.next.setMap(map); │ │ │ │ │ + this.previous.setMap(map); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createRoot │ │ │ │ │ - * Create the main root element │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * suffix - {String} suffix to append to the id │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Called when the control is added to the map. │ │ │ │ │ */ │ │ │ │ │ - createRoot: function(suffix) { │ │ │ │ │ - return this.nodeFactory(this.container.id + suffix, "olv:group"); │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + this.next.draw(); │ │ │ │ │ + this.previous.draw(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /************************************** │ │ │ │ │ - * * │ │ │ │ │ - * GEOMETRY DRAWING FUNCTIONS * │ │ │ │ │ - * * │ │ │ │ │ - **************************************/ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPoint │ │ │ │ │ - * Render a point │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * Method: previousTrigger │ │ │ │ │ + * Restore the previous state. If no items are in the previous history │ │ │ │ │ + * stack, this has no effect. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or false if the point could not be drawn │ │ │ │ │ + * {Object} Item representing state that was restored. Undefined if no │ │ │ │ │ + * items are in the previous history stack. │ │ │ │ │ */ │ │ │ │ │ - drawPoint: function(node, geometry) { │ │ │ │ │ - return this.drawCircle(node, geometry, 1); │ │ │ │ │ + previousTrigger: function() { │ │ │ │ │ + var current = this.previousStack.shift(); │ │ │ │ │ + var state = this.previousStack.shift(); │ │ │ │ │ + if (state != undefined) { │ │ │ │ │ + this.nextStack.unshift(current); │ │ │ │ │ + this.previousStack.unshift(state); │ │ │ │ │ + this.restoring = true; │ │ │ │ │ + this.restore(state); │ │ │ │ │ + this.restoring = false; │ │ │ │ │ + this.onNextChange(this.nextStack[0], this.nextStack.length); │ │ │ │ │ + this.onPreviousChange( │ │ │ │ │ + this.previousStack[1], this.previousStack.length - 1 │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + this.previousStack.unshift(current); │ │ │ │ │ + } │ │ │ │ │ + return state; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawCircle │ │ │ │ │ - * Render a circle. │ │ │ │ │ - * Size and Center a circle given geometry (x,y center) and radius │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * radius - {float} │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: nextTrigger │ │ │ │ │ + * Restore the next state. If no items are in the next history │ │ │ │ │ + * stack, this has no effect. The next history stack is populated │ │ │ │ │ + * as states are restored from the previous history stack. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or false if the circle could not ne drawn │ │ │ │ │ + * {Object} Item representing state that was restored. Undefined if no │ │ │ │ │ + * items are in the next history stack. │ │ │ │ │ */ │ │ │ │ │ - drawCircle: function(node, geometry, radius) { │ │ │ │ │ - if (!isNaN(geometry.x) && !isNaN(geometry.y)) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - │ │ │ │ │ - node.style.left = ((((geometry.x - this.featureDx) / resolution - this.offset.x) | 0) - radius) + "px"; │ │ │ │ │ - node.style.top = (((geometry.y / resolution - this.offset.y) | 0) - radius) + "px"; │ │ │ │ │ - │ │ │ │ │ - var diameter = radius * 2; │ │ │ │ │ - │ │ │ │ │ - node.style.width = diameter + "px"; │ │ │ │ │ - node.style.height = diameter + "px"; │ │ │ │ │ - return node; │ │ │ │ │ + nextTrigger: function() { │ │ │ │ │ + var state = this.nextStack.shift(); │ │ │ │ │ + if (state != undefined) { │ │ │ │ │ + this.previousStack.unshift(state); │ │ │ │ │ + this.restoring = true; │ │ │ │ │ + this.restore(state); │ │ │ │ │ + this.restoring = false; │ │ │ │ │ + this.onNextChange(this.nextStack[0], this.nextStack.length); │ │ │ │ │ + this.onPreviousChange( │ │ │ │ │ + this.previousStack[1], this.previousStack.length - 1 │ │ │ │ │ + ); │ │ │ │ │ } │ │ │ │ │ - return false; │ │ │ │ │ + return state; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLineString │ │ │ │ │ - * Render a linestring. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * APIMethod: clear │ │ │ │ │ + * Clear history. │ │ │ │ │ */ │ │ │ │ │ - drawLineString: function(node, geometry) { │ │ │ │ │ - return this.drawLine(node, geometry, false); │ │ │ │ │ + clear: function() { │ │ │ │ │ + this.previousStack = []; │ │ │ │ │ + this.previous.deactivate(); │ │ │ │ │ + this.nextStack = []; │ │ │ │ │ + this.next.deactivate(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLinearRing │ │ │ │ │ - * Render a linearring │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * Method: getState │ │ │ │ │ + * Get the current state and return it. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * {Object} An object representing the current state. │ │ │ │ │ */ │ │ │ │ │ - drawLinearRing: function(node, geometry) { │ │ │ │ │ - return this.drawLine(node, geometry, true); │ │ │ │ │ + getState: function() { │ │ │ │ │ + return { │ │ │ │ │ + center: this.map.getCenter(), │ │ │ │ │ + resolution: this.map.getResolution(), │ │ │ │ │ + projection: this.map.getProjectionObject(), │ │ │ │ │ + units: this.map.getProjectionObject().getUnits() || │ │ │ │ │ + this.map.units || this.map.baseLayer.units │ │ │ │ │ + }; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: DrawLine │ │ │ │ │ - * Render a line. │ │ │ │ │ - * │ │ │ │ │ + * Method: restore │ │ │ │ │ + * Update the state with the given object. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * closeLine - {Boolean} Close the line? (make it a ring?) │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * state - {Object} An object representing the state to restore. │ │ │ │ │ */ │ │ │ │ │ - drawLine: function(node, geometry, closeLine) { │ │ │ │ │ - │ │ │ │ │ - this.setNodeDimension(node, geometry); │ │ │ │ │ - │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var numComponents = geometry.components.length; │ │ │ │ │ - var parts = new Array(numComponents); │ │ │ │ │ - │ │ │ │ │ - var comp, x, y; │ │ │ │ │ - for (var i = 0; i < numComponents; i++) { │ │ │ │ │ - comp = geometry.components[i]; │ │ │ │ │ - x = ((comp.x - this.featureDx) / resolution - this.offset.x) | 0; │ │ │ │ │ - y = (comp.y / resolution - this.offset.y) | 0; │ │ │ │ │ - parts[i] = " " + x + "," + y + " l "; │ │ │ │ │ + restore: function(state) { │ │ │ │ │ + var center, zoom; │ │ │ │ │ + if (this.map.getProjectionObject() == state.projection) { │ │ │ │ │ + zoom = this.map.getZoomForResolution(state.resolution); │ │ │ │ │ + center = state.center; │ │ │ │ │ + } else { │ │ │ │ │ + center = state.center.clone(); │ │ │ │ │ + center.transform(state.projection, this.map.getProjectionObject()); │ │ │ │ │ + var sourceUnits = state.units; │ │ │ │ │ + var targetUnits = this.map.getProjectionObject().getUnits() || │ │ │ │ │ + this.map.units || this.map.baseLayer.units; │ │ │ │ │ + var resolutionFactor = sourceUnits && targetUnits ? │ │ │ │ │ + OpenLayers.INCHES_PER_UNIT[sourceUnits] / OpenLayers.INCHES_PER_UNIT[targetUnits] : 1; │ │ │ │ │ + zoom = this.map.getZoomForResolution(resolutionFactor * state.resolution); │ │ │ │ │ } │ │ │ │ │ - var end = (closeLine) ? " x e" : " e"; │ │ │ │ │ - node.path = "m" + parts.join("") + end; │ │ │ │ │ - return node; │ │ │ │ │ + this.map.setCenter(center, zoom); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPolygon │ │ │ │ │ - * Render a polygon │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * Method: setListeners │ │ │ │ │ + * Sets functions to be registered in the listeners object. │ │ │ │ │ */ │ │ │ │ │ - drawPolygon: function(node, geometry) { │ │ │ │ │ - this.setNodeDimension(node, geometry); │ │ │ │ │ - │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - │ │ │ │ │ - var path = []; │ │ │ │ │ - var j, jj, points, area, first, second, i, ii, comp, pathComp, x, y; │ │ │ │ │ - for (j = 0, jj = geometry.components.length; j < jj; j++) { │ │ │ │ │ - path.push("m"); │ │ │ │ │ - points = geometry.components[j].components; │ │ │ │ │ - // we only close paths of interior rings with area │ │ │ │ │ - area = (j === 0); │ │ │ │ │ - first = null; │ │ │ │ │ - second = null; │ │ │ │ │ - for (i = 0, ii = points.length; i < ii; i++) { │ │ │ │ │ - comp = points[i]; │ │ │ │ │ - x = ((comp.x - this.featureDx) / resolution - this.offset.x) | 0; │ │ │ │ │ - y = (comp.y / resolution - this.offset.y) | 0; │ │ │ │ │ - pathComp = " " + x + "," + y; │ │ │ │ │ - path.push(pathComp); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - path.push(" l"); │ │ │ │ │ - } │ │ │ │ │ - if (!area) { │ │ │ │ │ - // IE improperly renders sub-paths that have no area. │ │ │ │ │ - // Instead of checking the area of every ring, we confirm │ │ │ │ │ - // the ring has at least three distinct points. This does │ │ │ │ │ - // not catch all non-zero area cases, but it greatly improves │ │ │ │ │ - // interior ring digitizing and is a minor performance hit │ │ │ │ │ - // when rendering rings with many points. │ │ │ │ │ - if (!first) { │ │ │ │ │ - first = pathComp; │ │ │ │ │ - } else if (first != pathComp) { │ │ │ │ │ - if (!second) { │ │ │ │ │ - second = pathComp; │ │ │ │ │ - } else if (second != pathComp) { │ │ │ │ │ - // stop looking │ │ │ │ │ - area = true; │ │ │ │ │ - } │ │ │ │ │ + setListeners: function() { │ │ │ │ │ + this.listeners = {}; │ │ │ │ │ + for (var type in this.registry) { │ │ │ │ │ + this.listeners[type] = OpenLayers.Function.bind(function() { │ │ │ │ │ + if (!this.restoring) { │ │ │ │ │ + var state = this.registry[type].apply(this, arguments); │ │ │ │ │ + this.previousStack.unshift(state); │ │ │ │ │ + if (this.previousStack.length > 1) { │ │ │ │ │ + this.onPreviousChange( │ │ │ │ │ + this.previousStack[1], this.previousStack.length - 1 │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (this.previousStack.length > (this.limit + 1)) { │ │ │ │ │ + this.previousStack.pop(); │ │ │ │ │ + } │ │ │ │ │ + if (this.nextStack.length > 0) { │ │ │ │ │ + this.nextStack = []; │ │ │ │ │ + this.onNextChange(null, 0); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - path.push(area ? " x " : " "); │ │ │ │ │ + return true; │ │ │ │ │ + }, this); │ │ │ │ │ } │ │ │ │ │ - path.push("e"); │ │ │ │ │ - node.path = path.join(""); │ │ │ │ │ - return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawRectangle │ │ │ │ │ - * Render a rectangle │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Activate the control. This registers any listeners. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ - */ │ │ │ │ │ - drawRectangle: function(node, geometry) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - │ │ │ │ │ - node.style.left = (((geometry.x - this.featureDx) / resolution - this.offset.x) | 0) + "px"; │ │ │ │ │ - node.style.top = ((geometry.y / resolution - this.offset.y) | 0) + "px"; │ │ │ │ │ - node.style.width = ((geometry.width / resolution) | 0) + "px"; │ │ │ │ │ - node.style.height = ((geometry.height / resolution) | 0) + "px"; │ │ │ │ │ - │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: drawText │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * style - │ │ │ │ │ - * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * {Boolean} Control successfully activated. │ │ │ │ │ */ │ │ │ │ │ - drawText: function(featureId, style, location) { │ │ │ │ │ - var label = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX, "olv:rect"); │ │ │ │ │ - var textbox = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX + "_textbox", "olv:textbox"); │ │ │ │ │ - │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - label.style.left = (((location.x - this.featureDx) / resolution - this.offset.x) | 0) + "px"; │ │ │ │ │ - label.style.top = ((location.y / resolution - this.offset.y) | 0) + "px"; │ │ │ │ │ - label.style.flip = "y"; │ │ │ │ │ - │ │ │ │ │ - textbox.innerText = style.label; │ │ │ │ │ - │ │ │ │ │ - if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ - textbox.style.cursor = style.cursor; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontColor) { │ │ │ │ │ - textbox.style.color = style.fontColor; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontOpacity) { │ │ │ │ │ - textbox.style.filter = 'alpha(opacity=' + (style.fontOpacity * 100) + ')'; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontFamily) { │ │ │ │ │ - textbox.style.fontFamily = style.fontFamily; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontSize) { │ │ │ │ │ - textbox.style.fontSize = style.fontSize; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontWeight) { │ │ │ │ │ - textbox.style.fontWeight = style.fontWeight; │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStyle) { │ │ │ │ │ - textbox.style.fontStyle = style.fontStyle; │ │ │ │ │ - } │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - label._featureId = featureId; │ │ │ │ │ - textbox._featureId = featureId; │ │ │ │ │ - textbox._geometry = location; │ │ │ │ │ - textbox._geometryClass = location.CLASS_NAME; │ │ │ │ │ - } │ │ │ │ │ - textbox.style.whiteSpace = "nowrap"; │ │ │ │ │ - // fun with IE: IE7 in standards compliant mode does not display any │ │ │ │ │ - // text with a left inset of 0. So we set this to 1px and subtract one │ │ │ │ │ - // pixel later when we set label.style.left │ │ │ │ │ - textbox.inset = "1px,0px,0px,0px"; │ │ │ │ │ - │ │ │ │ │ - if (!label.parentNode) { │ │ │ │ │ - label.appendChild(textbox); │ │ │ │ │ - this.textRoot.appendChild(label); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var align = style.labelAlign || "cm"; │ │ │ │ │ - if (align.length == 1) { │ │ │ │ │ - align += "m"; │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this)) { │ │ │ │ │ + if (this.listeners == null) { │ │ │ │ │ + this.setListeners(); │ │ │ │ │ + } │ │ │ │ │ + for (var type in this.listeners) { │ │ │ │ │ + this.map.events.register(type, this, this.listeners[type]); │ │ │ │ │ + } │ │ │ │ │ + activated = true; │ │ │ │ │ + if (this.previousStack.length == 0) { │ │ │ │ │ + this.initStack(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var xshift = textbox.clientWidth * │ │ │ │ │ - (OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(0, 1)]); │ │ │ │ │ - var yshift = textbox.clientHeight * │ │ │ │ │ - (OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(1, 1)]); │ │ │ │ │ - label.style.left = parseInt(label.style.left) - xshift - 1 + "px"; │ │ │ │ │ - label.style.top = parseInt(label.style.top) + yshift + "px"; │ │ │ │ │ - │ │ │ │ │ + return activated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: moveRoot │ │ │ │ │ - * moves this renderer's root to a different renderer. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * renderer - {<OpenLayers.Renderer>} target renderer for the moved root │ │ │ │ │ - * root - {DOMElement} optional root node. To be used when this renderer │ │ │ │ │ - * holds roots from multiple layers to tell this method which one to │ │ │ │ │ - * detach │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true if successful, false otherwise │ │ │ │ │ + * Method: initStack │ │ │ │ │ + * Called after the control is activated if the previous history stack is │ │ │ │ │ + * empty. │ │ │ │ │ */ │ │ │ │ │ - moveRoot: function(renderer) { │ │ │ │ │ - var layer = this.map.getLayer(renderer.container.id); │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Vector.RootContainer) { │ │ │ │ │ - layer = this.map.getLayer(this.container.id); │ │ │ │ │ + initStack: function() { │ │ │ │ │ + if (this.map.getCenter()) { │ │ │ │ │ + this.listeners.moveend(); │ │ │ │ │ } │ │ │ │ │ - layer && layer.renderer.clear(); │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.moveRoot.apply(this, arguments); │ │ │ │ │ - layer && layer.redraw(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: importSymbol │ │ │ │ │ - * add a new symbol definition from the rendererer's symbol hash │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * graphicName - {String} name of the symbol to import │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the control. This unregisters any listeners. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} - hash of {DOMElement} "symbol" and {Number} "size" │ │ │ │ │ + * {Boolean} Control successfully deactivated. │ │ │ │ │ */ │ │ │ │ │ - importSymbol: function(graphicName) { │ │ │ │ │ - var id = this.container.id + "-" + graphicName; │ │ │ │ │ - │ │ │ │ │ - // check if symbol already exists in the cache │ │ │ │ │ - var cache = this.symbolCache[id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - return cache; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(graphicName + ' is not a valid symbol name'); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var symbolExtent = new OpenLayers.Bounds( │ │ │ │ │ - Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ - │ │ │ │ │ - var pathitems = ["m"]; │ │ │ │ │ - for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - var x = symbol[i]; │ │ │ │ │ - var y = symbol[i + 1]; │ │ │ │ │ - symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ - symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ - symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ - symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ - │ │ │ │ │ - pathitems.push(x); │ │ │ │ │ - pathitems.push(y); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - pathitems.push("l"); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this)) { │ │ │ │ │ + for (var type in this.listeners) { │ │ │ │ │ + this.map.events.unregister( │ │ │ │ │ + type, this, this.listeners[type] │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (this.clearOnDeactivate) { │ │ │ │ │ + this.clear(); │ │ │ │ │ + } │ │ │ │ │ + deactivated = true; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - pathitems.push("x e"); │ │ │ │ │ - var path = pathitems.join(" "); │ │ │ │ │ - │ │ │ │ │ - var diff = (symbolExtent.getWidth() - symbolExtent.getHeight()) / 2; │ │ │ │ │ - if (diff > 0) { │ │ │ │ │ - symbolExtent.bottom = symbolExtent.bottom - diff; │ │ │ │ │ - symbolExtent.top = symbolExtent.top + diff; │ │ │ │ │ - } else { │ │ │ │ │ - symbolExtent.left = symbolExtent.left + diff; │ │ │ │ │ - symbolExtent.right = symbolExtent.right - diff; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - cache = { │ │ │ │ │ - path: path, │ │ │ │ │ - size: symbolExtent.getWidth(), // equals getHeight() now │ │ │ │ │ - left: symbolExtent.left, │ │ │ │ │ - bottom: symbolExtent.bottom │ │ │ │ │ - }; │ │ │ │ │ - this.symbolCache[id] = cache; │ │ │ │ │ - │ │ │ │ │ - return cache; │ │ │ │ │ + return deactivated; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.VML" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.NavigationHistory" │ │ │ │ │ }); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.VML.LABEL_SHIFT │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.VML.LABEL_SHIFT = { │ │ │ │ │ - "l": 0, │ │ │ │ │ - "c": .5, │ │ │ │ │ - "r": 1, │ │ │ │ │ - "t": 0, │ │ │ │ │ - "m": .5, │ │ │ │ │ - "b": 1 │ │ │ │ │ -}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Renderer/SVG.js │ │ │ │ │ + OpenLayers/Control/Measure.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Renderer/Elements.js │ │ │ │ │ + * @requires OpenLayers/Control.js │ │ │ │ │ + * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Renderer.SVG │ │ │ │ │ - * │ │ │ │ │ - * Inherits: │ │ │ │ │ - * - <OpenLayers.Renderer.Elements> │ │ │ │ │ + * Class: OpenLayers.Control.Measure │ │ │ │ │ + * Allows for drawing of features for measurements. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Renderer.SVG = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ +OpenLayers.Control.Measure = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: xmlns │ │ │ │ │ - * {String} │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for listeners and triggering │ │ │ │ │ + * control specific events. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * control.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types (in addition to those from <OpenLayers.Control.events>): │ │ │ │ │ + * measure - Triggered when a measurement sketch is complete. Listeners │ │ │ │ │ + * will receive an event with measure, units, order, and geometry │ │ │ │ │ + * properties. │ │ │ │ │ + * measurepartial - Triggered when a new point is added to the │ │ │ │ │ + * measurement sketch or if the <immediate> property is true and the │ │ │ │ │ + * measurement sketch is modified. Listeners receive an event with measure, │ │ │ │ │ + * units, order, and geometry. │ │ │ │ │ */ │ │ │ │ │ - xmlns: "http://www.w3.org/2000/svg", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: xlinkns │ │ │ │ │ - * {String} │ │ │ │ │ + * APIProperty: handlerOptions │ │ │ │ │ + * {Object} Used to set non-default properties on the control's handler │ │ │ │ │ */ │ │ │ │ │ - xlinkns: "http://www.w3.org/1999/xlink", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constant: MAX_PIXEL │ │ │ │ │ - * {Integer} Firefox has a limitation where values larger or smaller than │ │ │ │ │ - * about 15000 in an SVG document lock the browser up. This │ │ │ │ │ - * works around it. │ │ │ │ │ + * Property: callbacks │ │ │ │ │ + * {Object} The functions that are sent to the handler for callback │ │ │ │ │ */ │ │ │ │ │ - MAX_PIXEL: 15000, │ │ │ │ │ + callbacks: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: translationParameters │ │ │ │ │ - * {Object} Hash with "x" and "y" properties │ │ │ │ │ + * APIProperty: displaySystem │ │ │ │ │ + * {String} Display system for output measurements. Supported values │ │ │ │ │ + * are 'english', 'metric', and 'geographic'. Default is 'metric'. │ │ │ │ │ */ │ │ │ │ │ - translationParameters: null, │ │ │ │ │ + displaySystem: 'metric', │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: symbolMetrics │ │ │ │ │ - * {Object} Cache for symbol metrics according to their svg coordinate │ │ │ │ │ - * space. This is an object keyed by the symbol's id, and values are │ │ │ │ │ - * an array of [width, centerX, centerY]. │ │ │ │ │ + * APIProperty: geodesic │ │ │ │ │ + * {Boolean} Calculate geodesic metrics instead of planar metrics. This │ │ │ │ │ + * requires that geometries can be transformed into Geographic/WGS84 │ │ │ │ │ + * (if that is not already the map projection). Default is false. │ │ │ │ │ */ │ │ │ │ │ - symbolMetrics: null, │ │ │ │ │ + geodesic: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Renderer.SVG │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * containerID - {String} │ │ │ │ │ + * Property: displaySystemUnits │ │ │ │ │ + * {Object} Units for various measurement systems. Values are arrays │ │ │ │ │ + * of unit abbreviations (from OpenLayers.INCHES_PER_UNIT) in decreasing │ │ │ │ │ + * order of length. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(containerID) { │ │ │ │ │ - if (!this.supported()) { │ │ │ │ │ - return; │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.initialize.apply(this, │ │ │ │ │ - arguments); │ │ │ │ │ - this.translationParameters = { │ │ │ │ │ - x: 0, │ │ │ │ │ - y: 0 │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - this.symbolMetrics = {}; │ │ │ │ │ + displaySystemUnits: { │ │ │ │ │ + geographic: ['dd'], │ │ │ │ │ + english: ['mi', 'ft', 'in'], │ │ │ │ │ + metric: ['km', 'm'] │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: supported │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the browser supports the SVG renderer │ │ │ │ │ + * Property: delay │ │ │ │ │ + * {Number} Number of milliseconds between clicks before the event is │ │ │ │ │ + * considered a double-click. The "measurepartial" event will not │ │ │ │ │ + * be triggered if the sketch is completed within this time. This │ │ │ │ │ + * is required for IE where creating a browser reflow (if a listener │ │ │ │ │ + * is modifying the DOM by displaying the measurement values) messes │ │ │ │ │ + * with the dblclick listener in the sketch handler. │ │ │ │ │ */ │ │ │ │ │ - supported: function() { │ │ │ │ │ - var svgFeature = "http://www.w3.org/TR/SVG11/feature#"; │ │ │ │ │ - return (document.implementation && │ │ │ │ │ - (document.implementation.hasFeature("org.w3c.svg", "1.0") || │ │ │ │ │ - document.implementation.hasFeature(svgFeature + "SVG", "1.1") || │ │ │ │ │ - document.implementation.hasFeature(svgFeature + "BasicStructure", "1.1"))); │ │ │ │ │ - }, │ │ │ │ │ + partialDelay: 300, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: inValidRange │ │ │ │ │ - * See #669 for more information │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * x - {Integer} │ │ │ │ │ - * y - {Integer} │ │ │ │ │ - * xyOnly - {Boolean} whether or not to just check for x and y, which means │ │ │ │ │ - * to not take the current translation parameters into account if true. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the 'x' and 'y' coordinates are in the │ │ │ │ │ - * valid range. │ │ │ │ │ + * Property: delayedTrigger │ │ │ │ │ + * {Number} Timeout id of trigger for measurepartial. │ │ │ │ │ */ │ │ │ │ │ - inValidRange: function(x, y, xyOnly) { │ │ │ │ │ - var left = x + (xyOnly ? 0 : this.translationParameters.x); │ │ │ │ │ - var top = y + (xyOnly ? 0 : this.translationParameters.y); │ │ │ │ │ - return (left >= -this.MAX_PIXEL && left <= this.MAX_PIXEL && │ │ │ │ │ - top >= -this.MAX_PIXEL && top <= this.MAX_PIXEL); │ │ │ │ │ - }, │ │ │ │ │ + delayedTrigger: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setExtent │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * extent - {<OpenLayers.Bounds>} │ │ │ │ │ - * resolutionChanged - {Boolean} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true to notify the layer that the new extent does not exceed │ │ │ │ │ - * the coordinate range, and the features will not need to be redrawn. │ │ │ │ │ - * False otherwise. │ │ │ │ │ + * APIProperty: persist │ │ │ │ │ + * {Boolean} Keep the temporary measurement sketch drawn after the │ │ │ │ │ + * measurement is complete. The geometry will persist until a new │ │ │ │ │ + * measurement is started, the control is deactivated, or <cancel> is │ │ │ │ │ + * called. │ │ │ │ │ */ │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - var resolution = this.getResolution(), │ │ │ │ │ - left = -extent.left / resolution, │ │ │ │ │ - top = extent.top / resolution; │ │ │ │ │ - │ │ │ │ │ - // If the resolution has changed, start over changing the corner, because │ │ │ │ │ - // the features will redraw. │ │ │ │ │ - if (resolutionChanged) { │ │ │ │ │ - this.left = left; │ │ │ │ │ - this.top = top; │ │ │ │ │ - // Set the viewbox │ │ │ │ │ - var extentString = "0 0 " + this.size.w + " " + this.size.h; │ │ │ │ │ + persist: false, │ │ │ │ │ │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "viewBox", extentString); │ │ │ │ │ - this.translate(this.xOffset, 0); │ │ │ │ │ - return true; │ │ │ │ │ - } else { │ │ │ │ │ - var inRange = this.translate(left - this.left + this.xOffset, top - this.top); │ │ │ │ │ - if (!inRange) { │ │ │ │ │ - // recenter the coordinate system │ │ │ │ │ - this.setExtent(extent, true); │ │ │ │ │ - } │ │ │ │ │ - return coordSysUnchanged && inRange; │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: immediate │ │ │ │ │ + * {Boolean} Activates the immediate measurement so that the "measurepartial" │ │ │ │ │ + * event is also fired once the measurement sketch is modified. │ │ │ │ │ + * Default is false. │ │ │ │ │ + */ │ │ │ │ │ + immediate: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: translate │ │ │ │ │ - * Transforms the SVG coordinate system │ │ │ │ │ - * │ │ │ │ │ + * Constructor: OpenLayers.Control.Measure │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * x - {Float} │ │ │ │ │ - * y - {Float} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} true if the translation parameters are in the valid coordinates │ │ │ │ │ - * range, false otherwise. │ │ │ │ │ + * handler - {<OpenLayers.Handler>} │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - translate: function(x, y) { │ │ │ │ │ - if (!this.inValidRange(x, y, true)) { │ │ │ │ │ - return false; │ │ │ │ │ - } else { │ │ │ │ │ - var transformString = ""; │ │ │ │ │ - if (x || y) { │ │ │ │ │ - transformString = "translate(" + x + "," + y + ")"; │ │ │ │ │ - } │ │ │ │ │ - this.root.setAttributeNS(null, "transform", transformString); │ │ │ │ │ - this.translationParameters = { │ │ │ │ │ - x: x, │ │ │ │ │ - y: y │ │ │ │ │ - }; │ │ │ │ │ - return true; │ │ │ │ │ + initialize: function(handler, options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + var callbacks = { │ │ │ │ │ + done: this.measureComplete, │ │ │ │ │ + point: this.measurePartial │ │ │ │ │ + }; │ │ │ │ │ + if (this.immediate) { │ │ │ │ │ + callbacks.modify = this.measureImmediate; │ │ │ │ │ } │ │ │ │ │ + this.callbacks = OpenLayers.Util.extend(callbacks, this.callbacks); │ │ │ │ │ + │ │ │ │ │ + // let the handler options override, so old code that passes 'persist' │ │ │ │ │ + // directly to the handler does not need an update │ │ │ │ │ + this.handlerOptions = OpenLayers.Util.extend({ │ │ │ │ │ + persist: this.persist │ │ │ │ │ + }, this.handlerOptions); │ │ │ │ │ + this.handler = new handler(this, this.callbacks, this.handlerOptions); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: setSize │ │ │ │ │ - * Sets the size of the drawing surface. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * size - {<OpenLayers.Size>} The size of the drawing surface │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ */ │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ - │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "width", this.size.w); │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "height", this.size.h); │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + this.cancelDelay(); │ │ │ │ │ + return OpenLayers.Control.prototype.deactivate.apply(this, arguments); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getNodeType │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} The corresponding node type for the specified geometry │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: cancel │ │ │ │ │ + * Stop the control from measuring. If <persist> is true, the temporary │ │ │ │ │ + * sketch will be erased. │ │ │ │ │ */ │ │ │ │ │ - getNodeType: function(geometry, style) { │ │ │ │ │ - var nodeType = null; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - nodeType = "image"; │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - nodeType = "svg"; │ │ │ │ │ - } else { │ │ │ │ │ - nodeType = "circle"; │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - nodeType = "rect"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - nodeType = "polyline"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - nodeType = "polygon"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - case "OpenLayers.Geometry.Curve": │ │ │ │ │ - nodeType = "path"; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break; │ │ │ │ │ - } │ │ │ │ │ - return nodeType; │ │ │ │ │ + cancel: function() { │ │ │ │ │ + this.cancelDelay(); │ │ │ │ │ + this.handler.cancel(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: setStyle │ │ │ │ │ - * Use to set all the style attributes to a SVG node. │ │ │ │ │ - * │ │ │ │ │ - * Takes care to adjust stroke width and point radius to be │ │ │ │ │ - * resolution-relative │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {SVGDomElement} An SVG element to decorate │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * options - {Object} Currently supported options include │ │ │ │ │ - * 'isFilled' {Boolean} and │ │ │ │ │ - * 'isStroked' {Boolean} │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setImmediate │ │ │ │ │ + * Sets the <immediate> property. Changes the activity of immediate │ │ │ │ │ + * measurement. │ │ │ │ │ */ │ │ │ │ │ - setStyle: function(node, style, options) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - options = options || node._options; │ │ │ │ │ - │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - node.setAttributeNS(null, "title", title); │ │ │ │ │ - //Standards-conformant SVG │ │ │ │ │ - // Prevent duplicate nodes. See issue https://github.com/openlayers/openlayers/issues/92 │ │ │ │ │ - var titleNode = node.getElementsByTagName("title"); │ │ │ │ │ - if (titleNode.length > 0) { │ │ │ │ │ - titleNode[0].firstChild.textContent = title; │ │ │ │ │ - } else { │ │ │ │ │ - var label = this.nodeFactory(null, "title"); │ │ │ │ │ - label.textContent = title; │ │ │ │ │ - node.appendChild(label); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var r = parseFloat(node.getAttributeNS(null, "r")); │ │ │ │ │ - var widthFactor = 1; │ │ │ │ │ - var pos; │ │ │ │ │ - if (node._geometryClass == "OpenLayers.Geometry.Point" && r) { │ │ │ │ │ - node.style.visibility = ""; │ │ │ │ │ - if (style.graphic === false) { │ │ │ │ │ - node.style.visibility = "hidden"; │ │ │ │ │ - } else if (style.externalGraphic) { │ │ │ │ │ - pos = this.getPosition(node); │ │ │ │ │ - if (style.graphicWidth && style.graphicHeight) { │ │ │ │ │ - node.setAttributeNS(null, "preserveAspectRatio", "none"); │ │ │ │ │ - } │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - var xOffset = (style.graphicXOffset != undefined) ? │ │ │ │ │ - style.graphicXOffset : -(0.5 * width); │ │ │ │ │ - var yOffset = (style.graphicYOffset != undefined) ? │ │ │ │ │ - style.graphicYOffset : -(0.5 * height); │ │ │ │ │ - │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ - │ │ │ │ │ - node.setAttributeNS(null, "x", (pos.x + xOffset).toFixed()); │ │ │ │ │ - node.setAttributeNS(null, "y", (pos.y + yOffset).toFixed()); │ │ │ │ │ - node.setAttributeNS(null, "width", width); │ │ │ │ │ - node.setAttributeNS(null, "height", height); │ │ │ │ │ - node.setAttributeNS(this.xlinkns, "xlink:href", style.externalGraphic); │ │ │ │ │ - node.setAttributeNS(null, "style", "opacity: " + opacity); │ │ │ │ │ - node.onclick = OpenLayers.Event.preventDefault; │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - // the symbol viewBox is three times as large as the symbol │ │ │ │ │ - var offset = style.pointRadius * 3; │ │ │ │ │ - var size = offset * 2; │ │ │ │ │ - var src = this.importSymbol(style.graphicName); │ │ │ │ │ - pos = this.getPosition(node); │ │ │ │ │ - widthFactor = this.symbolMetrics[src.id][0] * 3 / size; │ │ │ │ │ - │ │ │ │ │ - // remove the node from the dom before we modify it. This │ │ │ │ │ - // prevents various rendering issues in Safari and FF │ │ │ │ │ - var parent = node.parentNode; │ │ │ │ │ - var nextSibling = node.nextSibling; │ │ │ │ │ - if (parent) { │ │ │ │ │ - parent.removeChild(node); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - // The more appropriate way to implement this would be use/defs, │ │ │ │ │ - // but due to various issues in several browsers, it is safer to │ │ │ │ │ - // copy the symbols instead of referencing them. │ │ │ │ │ - // See e.g. ticket http://trac.osgeo.org/openlayers/ticket/2985 │ │ │ │ │ - // and this email thread │ │ │ │ │ - // http://osgeo-org.1803224.n2.nabble.com/Select-Control-Ctrl-click-on-Feature-with-a-graphicName-opens-new-browser-window-tc5846039.html │ │ │ │ │ - node.firstChild && node.removeChild(node.firstChild); │ │ │ │ │ - node.appendChild(src.firstChild.cloneNode(true)); │ │ │ │ │ - node.setAttributeNS(null, "viewBox", src.getAttributeNS(null, "viewBox")); │ │ │ │ │ - │ │ │ │ │ - node.setAttributeNS(null, "width", size); │ │ │ │ │ - node.setAttributeNS(null, "height", size); │ │ │ │ │ - node.setAttributeNS(null, "x", pos.x - offset); │ │ │ │ │ - node.setAttributeNS(null, "y", pos.y - offset); │ │ │ │ │ - │ │ │ │ │ - // now that the node has all its new properties, insert it │ │ │ │ │ - // back into the dom where it was │ │ │ │ │ - if (nextSibling) { │ │ │ │ │ - parent.insertBefore(node, nextSibling); │ │ │ │ │ - } else if (parent) { │ │ │ │ │ - parent.appendChild(node); │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "r", style.pointRadius); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var rotation = style.rotation; │ │ │ │ │ - │ │ │ │ │ - if ((rotation !== undefined || node._rotation !== undefined) && pos) { │ │ │ │ │ - node._rotation = rotation; │ │ │ │ │ - rotation |= 0; │ │ │ │ │ - if (node.nodeName !== "svg") { │ │ │ │ │ - node.setAttributeNS(null, "transform", │ │ │ │ │ - "rotate(" + rotation + " " + pos.x + " " + │ │ │ │ │ - pos.y + ")"); │ │ │ │ │ - } else { │ │ │ │ │ - var metrics = this.symbolMetrics[src.id]; │ │ │ │ │ - node.firstChild.setAttributeNS(null, "transform", "rotate(" + │ │ │ │ │ - rotation + " " + │ │ │ │ │ - metrics[1] + " " + │ │ │ │ │ - metrics[2] + ")"); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (options.isFilled) { │ │ │ │ │ - node.setAttributeNS(null, "fill", style.fillColor); │ │ │ │ │ - node.setAttributeNS(null, "fill-opacity", style.fillOpacity); │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "fill", "none"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (options.isStroked) { │ │ │ │ │ - node.setAttributeNS(null, "stroke", style.strokeColor); │ │ │ │ │ - node.setAttributeNS(null, "stroke-opacity", style.strokeOpacity); │ │ │ │ │ - node.setAttributeNS(null, "stroke-width", style.strokeWidth * widthFactor); │ │ │ │ │ - node.setAttributeNS(null, "stroke-linecap", style.strokeLinecap || "round"); │ │ │ │ │ - // Hard-coded linejoin for now, to make it look the same as in VML. │ │ │ │ │ - // There is no strokeLinejoin property yet for symbolizers. │ │ │ │ │ - node.setAttributeNS(null, "stroke-linejoin", "round"); │ │ │ │ │ - style.strokeDashstyle && node.setAttributeNS(null, │ │ │ │ │ - "stroke-dasharray", this.dashStyle(style, widthFactor)); │ │ │ │ │ + setImmediate: function(immediate) { │ │ │ │ │ + this.immediate = immediate; │ │ │ │ │ + if (this.immediate) { │ │ │ │ │ + this.callbacks.modify = this.measureImmediate; │ │ │ │ │ } else { │ │ │ │ │ - node.setAttributeNS(null, "stroke", "none"); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (style.pointerEvents) { │ │ │ │ │ - node.setAttributeNS(null, "pointer-events", style.pointerEvents); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (style.cursor != null) { │ │ │ │ │ - node.setAttributeNS(null, "cursor", style.cursor); │ │ │ │ │ + delete this.callbacks.modify; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - return node; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: dashStyle │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateHandler │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * style - {Object} │ │ │ │ │ - * widthFactor - {Number} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {String} A SVG compliant 'stroke-dasharray' value │ │ │ │ │ + * handler - {Function} One of the sketch handler constructors. │ │ │ │ │ + * options - {Object} Options for the handler. │ │ │ │ │ */ │ │ │ │ │ - dashStyle: function(style, widthFactor) { │ │ │ │ │ - var w = style.strokeWidth * widthFactor; │ │ │ │ │ - var str = style.strokeDashstyle; │ │ │ │ │ - switch (str) { │ │ │ │ │ - case 'solid': │ │ │ │ │ - return 'none'; │ │ │ │ │ - case 'dot': │ │ │ │ │ - return [1, 4 * w].join(); │ │ │ │ │ - case 'dash': │ │ │ │ │ - return [4 * w, 4 * w].join(); │ │ │ │ │ - case 'dashdot': │ │ │ │ │ - return [4 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ - case 'longdash': │ │ │ │ │ - return [8 * w, 4 * w].join(); │ │ │ │ │ - case 'longdashdot': │ │ │ │ │ - return [8 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ - default: │ │ │ │ │ - return OpenLayers.String.trim(str).replace(/\s+/g, ","); │ │ │ │ │ + updateHandler: function(handler, options) { │ │ │ │ │ + var active = this.active; │ │ │ │ │ + if (active) { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: createNode │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * type - {String} Kind of node to draw │ │ │ │ │ - * id - {String} Id for node │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} A new node of the given type and id │ │ │ │ │ - */ │ │ │ │ │ - createNode: function(type, id) { │ │ │ │ │ - var node = document.createElementNS(this.xmlns, type); │ │ │ │ │ - if (id) { │ │ │ │ │ - node.setAttributeNS(null, "id", id); │ │ │ │ │ + this.handler = new handler(this, this.callbacks, options); │ │ │ │ │ + if (active) { │ │ │ │ │ + this.activate(); │ │ │ │ │ } │ │ │ │ │ - return node; │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Method: nodeTypeCompare │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {SVGDomElement} An SVG element │ │ │ │ │ - * type - {String} Kind of node │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Boolean} Whether or not the specified node is of the specified type │ │ │ │ │ - */ │ │ │ │ │ - nodeTypeCompare: function(node, type) { │ │ │ │ │ - return (type == node.nodeName); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createRenderRoot │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The specific render engine's root element │ │ │ │ │ + * Method: measureComplete │ │ │ │ │ + * Called when the measurement sketch is done. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ */ │ │ │ │ │ - createRenderRoot: function() { │ │ │ │ │ - var svg = this.nodeFactory(this.container.id + "_svgRoot", "svg"); │ │ │ │ │ - svg.style.display = "block"; │ │ │ │ │ - return svg; │ │ │ │ │ + measureComplete: function(geometry) { │ │ │ │ │ + this.cancelDelay(); │ │ │ │ │ + this.measure(geometry, "measure"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createRoot │ │ │ │ │ - * │ │ │ │ │ + * Method: measurePartial │ │ │ │ │ + * Called each time a new point is added to the measurement sketch. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * suffix - {String} suffix to append to the id │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} The last point added. │ │ │ │ │ + * geometry - {<OpenLayers.Geometry>} The sketch geometry. │ │ │ │ │ */ │ │ │ │ │ - createRoot: function(suffix) { │ │ │ │ │ - return this.nodeFactory(this.container.id + suffix, "g"); │ │ │ │ │ + measurePartial: function(point, geometry) { │ │ │ │ │ + this.cancelDelay(); │ │ │ │ │ + geometry = geometry.clone(); │ │ │ │ │ + // when we're wating for a dblclick, we have to trigger measurepartial │ │ │ │ │ + // after some delay to deal with reflow issues in IE │ │ │ │ │ + if (this.handler.freehandMode(this.handler.evt)) { │ │ │ │ │ + // no dblclick in freehand mode │ │ │ │ │ + this.measure(geometry, "measurepartial"); │ │ │ │ │ + } else { │ │ │ │ │ + this.delayedTrigger = window.setTimeout( │ │ │ │ │ + OpenLayers.Function.bind(function() { │ │ │ │ │ + this.delayedTrigger = null; │ │ │ │ │ + this.measure(geometry, "measurepartial"); │ │ │ │ │ + }, this), │ │ │ │ │ + this.partialDelay │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: createDefs │ │ │ │ │ + * Method: measureImmediate │ │ │ │ │ + * Called each time the measurement sketch is modified. │ │ │ │ │ * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} The element to which we'll add the symbol definitions │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} The point at the mouse position. │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} The sketch feature. │ │ │ │ │ + * drawing - {Boolean} Indicates whether we're currently drawing. │ │ │ │ │ */ │ │ │ │ │ - createDefs: function() { │ │ │ │ │ - var defs = this.nodeFactory(this.container.id + "_defs", "defs"); │ │ │ │ │ - this.rendererRoot.appendChild(defs); │ │ │ │ │ - return defs; │ │ │ │ │ + measureImmediate: function(point, feature, drawing) { │ │ │ │ │ + if (drawing && !this.handler.freehandMode(this.handler.evt)) { │ │ │ │ │ + this.cancelDelay(); │ │ │ │ │ + this.measure(feature.geometry, "measurepartial"); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /************************************** │ │ │ │ │ - * * │ │ │ │ │ - * GEOMETRY DRAWING FUNCTIONS * │ │ │ │ │ - * * │ │ │ │ │ - **************************************/ │ │ │ │ │ - │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPoint │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the point │ │ │ │ │ + * Method: cancelDelay │ │ │ │ │ + * Cancels the delay measurement that measurePartial began. │ │ │ │ │ */ │ │ │ │ │ - drawPoint: function(node, geometry) { │ │ │ │ │ - return this.drawCircle(node, geometry, 1); │ │ │ │ │ + cancelDelay: function() { │ │ │ │ │ + if (this.delayedTrigger !== null) { │ │ │ │ │ + window.clearTimeout(this.delayedTrigger); │ │ │ │ │ + this.delayedTrigger = null; │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawCircle │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ + * Method: measure │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * radius - {Float} │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the circle │ │ │ │ │ + * eventType - {String} │ │ │ │ │ */ │ │ │ │ │ - drawCircle: function(node, geometry, radius) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = ((geometry.x - this.featureDx) / resolution + this.left); │ │ │ │ │ - var y = (this.top - geometry.y / resolution); │ │ │ │ │ - │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - node.setAttributeNS(null, "cx", x); │ │ │ │ │ - node.setAttributeNS(null, "cy", y); │ │ │ │ │ - node.setAttributeNS(null, "r", radius); │ │ │ │ │ - return node; │ │ │ │ │ + measure: function(geometry, eventType) { │ │ │ │ │ + var stat, order; │ │ │ │ │ + if (geometry.CLASS_NAME.indexOf('LineString') > -1) { │ │ │ │ │ + stat = this.getBestLength(geometry); │ │ │ │ │ + order = 1; │ │ │ │ │ } else { │ │ │ │ │ - return false; │ │ │ │ │ + stat = this.getBestArea(geometry); │ │ │ │ │ + order = 2; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ + this.events.triggerEvent(eventType, { │ │ │ │ │ + measure: stat[0], │ │ │ │ │ + units: stat[1], │ │ │ │ │ + order: order, │ │ │ │ │ + geometry: geometry │ │ │ │ │ + }); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLineString │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ + * Method: getBestArea │ │ │ │ │ + * Based on the <displaySystem> returns the area of a geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components of │ │ │ │ │ - * the linestring, or false if nothing could be drawn │ │ │ │ │ + * {Array([Float, String])} Returns a two item array containing the │ │ │ │ │ + * area and the units abbreviation. │ │ │ │ │ */ │ │ │ │ │ - drawLineString: function(node, geometry) { │ │ │ │ │ - var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ - if (componentsResult.path) { │ │ │ │ │ - node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ - return (componentsResult.complete ? node : null); │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ + getBestArea: function(geometry) { │ │ │ │ │ + var units = this.displaySystemUnits[this.displaySystem]; │ │ │ │ │ + var unit, area; │ │ │ │ │ + for (var i = 0, len = units.length; i < len; ++i) { │ │ │ │ │ + unit = units[i]; │ │ │ │ │ + area = this.getArea(geometry, unit); │ │ │ │ │ + if (area > 1) { │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return [area, unit]; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawLinearRing │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ + * Method: getArea │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * units - {String} Unit abbreviation │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ - * of the linear ring, or false if nothing could be drawn │ │ │ │ │ + * {Float} The geometry area in the given units. │ │ │ │ │ */ │ │ │ │ │ - drawLinearRing: function(node, geometry) { │ │ │ │ │ - var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ - if (componentsResult.path) { │ │ │ │ │ - node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ - return (componentsResult.complete ? node : null); │ │ │ │ │ + getArea: function(geometry, units) { │ │ │ │ │ + var area, geomUnits; │ │ │ │ │ + if (this.geodesic) { │ │ │ │ │ + area = geometry.getGeodesicArea(this.map.getProjectionObject()); │ │ │ │ │ + geomUnits = "m"; │ │ │ │ │ } else { │ │ │ │ │ - return false; │ │ │ │ │ + area = geometry.getArea(); │ │ │ │ │ + geomUnits = this.map.getUnits(); │ │ │ │ │ + } │ │ │ │ │ + var inPerDisplayUnit = OpenLayers.INCHES_PER_UNIT[units]; │ │ │ │ │ + if (inPerDisplayUnit) { │ │ │ │ │ + var inPerMapUnit = OpenLayers.INCHES_PER_UNIT[geomUnits]; │ │ │ │ │ + area *= Math.pow((inPerMapUnit / inPerDisplayUnit), 2); │ │ │ │ │ } │ │ │ │ │ + return area; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawPolygon │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ + * Method: getBestLength │ │ │ │ │ + * Based on the <displaySystem> returns the length of a geometry. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or null if the renderer could not draw all components │ │ │ │ │ - * of the polygon, or false if nothing could be drawn │ │ │ │ │ + * {Array([Float, String])} Returns a two item array containing the │ │ │ │ │ + * length and the units abbreviation. │ │ │ │ │ */ │ │ │ │ │ - drawPolygon: function(node, geometry) { │ │ │ │ │ - var d = ""; │ │ │ │ │ - var draw = true; │ │ │ │ │ - var complete = true; │ │ │ │ │ - var linearRingResult, path; │ │ │ │ │ - for (var j = 0, len = geometry.components.length; j < len; j++) { │ │ │ │ │ - d += " M"; │ │ │ │ │ - linearRingResult = this.getComponentsString( │ │ │ │ │ - geometry.components[j].components, " "); │ │ │ │ │ - path = linearRingResult.path; │ │ │ │ │ - if (path) { │ │ │ │ │ - d += " " + path; │ │ │ │ │ - complete = linearRingResult.complete && complete; │ │ │ │ │ - } else { │ │ │ │ │ - draw = false; │ │ │ │ │ + getBestLength: function(geometry) { │ │ │ │ │ + var units = this.displaySystemUnits[this.displaySystem]; │ │ │ │ │ + var unit, length; │ │ │ │ │ + for (var i = 0, len = units.length; i < len; ++i) { │ │ │ │ │ + unit = units[i]; │ │ │ │ │ + length = this.getLength(geometry, unit); │ │ │ │ │ + if (length > 1) { │ │ │ │ │ + break; │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - d += " z"; │ │ │ │ │ - if (draw) { │ │ │ │ │ - node.setAttributeNS(null, "d", d); │ │ │ │ │ - node.setAttributeNS(null, "fill-rule", "evenodd"); │ │ │ │ │ - return complete ? node : null; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ - } │ │ │ │ │ + return [length, unit]; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: drawRectangle │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ + * Method: getLength │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ * geometry - {<OpenLayers.Geometry>} │ │ │ │ │ - * │ │ │ │ │ + * units - {String} Unit abbreviation │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} or false if the renderer could not draw the rectangle │ │ │ │ │ + * {Float} The geometry length in the given units. │ │ │ │ │ */ │ │ │ │ │ - drawRectangle: function(node, geometry) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = ((geometry.x - this.featureDx) / resolution + this.left); │ │ │ │ │ - var y = (this.top - geometry.y / resolution); │ │ │ │ │ - │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - node.setAttributeNS(null, "x", x); │ │ │ │ │ - node.setAttributeNS(null, "y", y); │ │ │ │ │ - node.setAttributeNS(null, "width", geometry.width / resolution); │ │ │ │ │ - node.setAttributeNS(null, "height", geometry.height / resolution); │ │ │ │ │ - return node; │ │ │ │ │ + getLength: function(geometry, units) { │ │ │ │ │ + var length, geomUnits; │ │ │ │ │ + if (this.geodesic) { │ │ │ │ │ + length = geometry.getGeodesicLength(this.map.getProjectionObject()); │ │ │ │ │ + geomUnits = "m"; │ │ │ │ │ } else { │ │ │ │ │ - return false; │ │ │ │ │ + length = geometry.getLength(); │ │ │ │ │ + geomUnits = this.map.getUnits(); │ │ │ │ │ + } │ │ │ │ │ + var inPerDisplayUnit = OpenLayers.INCHES_PER_UNIT[units]; │ │ │ │ │ + if (inPerDisplayUnit) { │ │ │ │ │ + var inPerMapUnit = OpenLayers.INCHES_PER_UNIT[geomUnits]; │ │ │ │ │ + length *= (inPerMapUnit / inPerDisplayUnit); │ │ │ │ │ } │ │ │ │ │ + return length; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Measure" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Control/ZoomPanel.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Control/Panel.js │ │ │ │ │ + * @requires OpenLayers/Control/ZoomIn.js │ │ │ │ │ + * @requires OpenLayers/Control/ZoomOut.js │ │ │ │ │ + * @requires OpenLayers/Control/ZoomToMaxExtent.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Control.ZoomPanel │ │ │ │ │ + * The ZoomPanel control is a compact collecton of 3 zoom controls: a │ │ │ │ │ + * <OpenLayers.Control.ZoomIn>, a <OpenLayers.Control.ZoomToMaxExtent>, and a │ │ │ │ │ + * <OpenLayers.Control.ZoomOut>. By default it is drawn in the upper left │ │ │ │ │ + * corner of the map. │ │ │ │ │ + * │ │ │ │ │ + * Note: │ │ │ │ │ + * If you wish to use this class with the default images and you want │ │ │ │ │ + * it to look nice in ie6, you should add the following, conditionally │ │ │ │ │ + * added css stylesheet to your HTML file: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * <!--[if lte IE 6]> │ │ │ │ │ + * <link rel="stylesheet" href="../theme/default/ie6-style.css" type="text/css" /> │ │ │ │ │ + * <![endif]--> │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Control.Panel> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Control.ZoomPanel = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: drawText │ │ │ │ │ - * This method is only called by the renderer itself. │ │ │ │ │ + * Constructor: OpenLayers.Control.ZoomPanel │ │ │ │ │ + * Add the three zooming controls. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * featureId - {String} │ │ │ │ │ - * style - │ │ │ │ │ - * location - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * options - {Object} An optional object whose properties will be used │ │ │ │ │ + * to extend the control. │ │ │ │ │ */ │ │ │ │ │ - drawText: function(featureId, style, location) { │ │ │ │ │ - var drawOutline = (!!style.labelOutlineWidth); │ │ │ │ │ - // First draw text in halo color and size and overlay the │ │ │ │ │ - // normal text afterwards │ │ │ │ │ - if (drawOutline) { │ │ │ │ │ - var outlineStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ - outlineStyle.fontColor = outlineStyle.labelOutlineColor; │ │ │ │ │ - outlineStyle.fontStrokeColor = outlineStyle.labelOutlineColor; │ │ │ │ │ - outlineStyle.fontStrokeWidth = style.labelOutlineWidth; │ │ │ │ │ - if (style.labelOutlineOpacity) { │ │ │ │ │ - outlineStyle.fontOpacity = style.labelOutlineOpacity; │ │ │ │ │ - } │ │ │ │ │ - delete outlineStyle.labelOutlineWidth; │ │ │ │ │ - this.drawText(featureId, outlineStyle, location); │ │ │ │ │ - } │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.addControls([ │ │ │ │ │ + new OpenLayers.Control.ZoomIn(), │ │ │ │ │ + new OpenLayers.Control.ZoomToMaxExtent(), │ │ │ │ │ + new OpenLayers.Control.ZoomOut() │ │ │ │ │ + ]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomPanel" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Tile/UTFGrid.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - var x = ((location.x - this.featureDx) / resolution + this.left); │ │ │ │ │ - var y = (location.y / resolution - this.top); │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - var suffix = (drawOutline) ? this.LABEL_OUTLINE_SUFFIX : this.LABEL_ID_SUFFIX; │ │ │ │ │ - var label = this.nodeFactory(featureId + suffix, "text"); │ │ │ │ │ │ │ │ │ │ - label.setAttributeNS(null, "x", x); │ │ │ │ │ - label.setAttributeNS(null, "y", -y); │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Tile.js │ │ │ │ │ + * @requires OpenLayers/Format/JSON.js │ │ │ │ │ + * @requires OpenLayers/Request.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - if (style.fontColor) { │ │ │ │ │ - label.setAttributeNS(null, "fill", style.fontColor); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStrokeColor) { │ │ │ │ │ - label.setAttributeNS(null, "stroke", style.fontStrokeColor); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStrokeWidth) { │ │ │ │ │ - label.setAttributeNS(null, "stroke-width", style.fontStrokeWidth); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontOpacity) { │ │ │ │ │ - label.setAttributeNS(null, "opacity", style.fontOpacity); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontFamily) { │ │ │ │ │ - label.setAttributeNS(null, "font-family", style.fontFamily); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontSize) { │ │ │ │ │ - label.setAttributeNS(null, "font-size", style.fontSize); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontWeight) { │ │ │ │ │ - label.setAttributeNS(null, "font-weight", style.fontWeight); │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStyle) { │ │ │ │ │ - label.setAttributeNS(null, "font-style", style.fontStyle); │ │ │ │ │ - } │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - label.setAttributeNS(null, "pointer-events", "visible"); │ │ │ │ │ - label._featureId = featureId; │ │ │ │ │ - } else { │ │ │ │ │ - label.setAttributeNS(null, "pointer-events", "none"); │ │ │ │ │ - } │ │ │ │ │ - var align = style.labelAlign || OpenLayers.Renderer.defaultSymbolizer.labelAlign; │ │ │ │ │ - label.setAttributeNS(null, "text-anchor", │ │ │ │ │ - OpenLayers.Renderer.SVG.LABEL_ALIGN[align[0]] || "middle"); │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Tile.UTFGrid │ │ │ │ │ + * Instances of OpenLayers.Tile.UTFGrid are used to manage │ │ │ │ │ + * UTFGrids. This is an unusual tile type in that it doesn't have a │ │ │ │ │ + * rendered image; only a 'hit grid' that can be used to │ │ │ │ │ + * look up feature attributes. │ │ │ │ │ + * │ │ │ │ │ + * See the <OpenLayers.Tile.UTFGrid> constructor for details on constructing a │ │ │ │ │ + * new instance. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Tile> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Tile.UTFGrid = OpenLayers.Class(OpenLayers.Tile, { │ │ │ │ │ │ │ │ │ │ - if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ - label.setAttributeNS(null, "dominant-baseline", │ │ │ │ │ - OpenLayers.Renderer.SVG.LABEL_ALIGN[align[1]] || "central"); │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: url │ │ │ │ │ + * {String} │ │ │ │ │ + * The URL of the UTFGrid file being requested. Provided by the <getURL> │ │ │ │ │ + * method. │ │ │ │ │ + */ │ │ │ │ │ + url: null, │ │ │ │ │ │ │ │ │ │ - var labelRows = style.label.split('\n'); │ │ │ │ │ - var numRows = labelRows.length; │ │ │ │ │ - while (label.childNodes.length > numRows) { │ │ │ │ │ - label.removeChild(label.lastChild); │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - var tspan = this.nodeFactory(featureId + suffix + "_tspan_" + i, "tspan"); │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - tspan._featureId = featureId; │ │ │ │ │ - tspan._geometry = location; │ │ │ │ │ - tspan._geometryClass = location.CLASS_NAME; │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.IS_GECKO === false) { │ │ │ │ │ - tspan.setAttributeNS(null, "baseline-shift", │ │ │ │ │ - OpenLayers.Renderer.SVG.LABEL_VSHIFT[align[1]] || "-35%"); │ │ │ │ │ - } │ │ │ │ │ - tspan.setAttribute("x", x); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - var vfactor = OpenLayers.Renderer.SVG.LABEL_VFACTOR[align[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5; │ │ │ │ │ - } │ │ │ │ │ - tspan.setAttribute("dy", (vfactor * (numRows - 1)) + "em"); │ │ │ │ │ - } else { │ │ │ │ │ - tspan.setAttribute("dy", "1em"); │ │ │ │ │ - } │ │ │ │ │ - tspan.textContent = (labelRows[i] === '') ? ' ' : labelRows[i]; │ │ │ │ │ - if (!tspan.parentNode) { │ │ │ │ │ - label.appendChild(tspan); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * Property: utfgridResolution │ │ │ │ │ + * {Number} │ │ │ │ │ + * Ratio of the pixel width to the width of a UTFGrid data point. If an │ │ │ │ │ + * entry in the grid represents a 4x4 block of pixels, the │ │ │ │ │ + * utfgridResolution would be 4. Default is 2. │ │ │ │ │ + */ │ │ │ │ │ + utfgridResolution: 2, │ │ │ │ │ │ │ │ │ │ - if (!label.parentNode) { │ │ │ │ │ - this.textRoot.appendChild(label); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * Property: json │ │ │ │ │ + * {Object} │ │ │ │ │ + * Stores the parsed JSON tile data structure. │ │ │ │ │ + */ │ │ │ │ │ + json: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getComponentString │ │ │ │ │ + * Property: format │ │ │ │ │ + * {OpenLayers.Format.JSON} │ │ │ │ │ + * Parser instance used to parse JSON for cross browser support. The native │ │ │ │ │ + * JSON.parse method will be used where available (all except IE<8). │ │ │ │ │ + */ │ │ │ │ │ + format: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Tile.UTFGrid │ │ │ │ │ + * Constructor for a new <OpenLayers.Tile.UTFGrid> instance. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * components - {Array(<OpenLayers.Geometry.Point>)} Array of points │ │ │ │ │ - * separator - {String} character between coordinate pairs. Defaults to "," │ │ │ │ │ + * layer - {<OpenLayers.Layer>} layer that the tile will go in. │ │ │ │ │ + * position - {<OpenLayers.Pixel>} │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * url - {<String>} Deprecated. Remove me in 3.0. │ │ │ │ │ + * size - {<OpenLayers.Size>} │ │ │ │ │ + * options - {Object} │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.clear(); │ │ │ │ │ + OpenLayers.Tile.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * Check that a tile should be drawn, and draw it. │ │ │ │ │ + * In the case of UTFGrids, "drawing" it means fetching and │ │ │ │ │ + * parsing the json. │ │ │ │ │ * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} hash with properties "path" (the string created from the │ │ │ │ │ - * components and "complete" (false if the renderer was unable to │ │ │ │ │ - * draw all components) │ │ │ │ │ + * {Boolean} Was a tile drawn? │ │ │ │ │ */ │ │ │ │ │ - getComponentsString: function(components, separator) { │ │ │ │ │ - var renderCmp = []; │ │ │ │ │ - var complete = true; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - var strings = []; │ │ │ │ │ - var str, component; │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - component = components[i]; │ │ │ │ │ - renderCmp.push(component); │ │ │ │ │ - str = this.getShortString(component); │ │ │ │ │ - if (str) { │ │ │ │ │ - strings.push(str); │ │ │ │ │ + draw: function() { │ │ │ │ │ + var drawn = OpenLayers.Tile.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (drawn) { │ │ │ │ │ + if (this.isLoading) { │ │ │ │ │ + this.abortLoading(); │ │ │ │ │ + //if we're already loading, send 'reload' instead of 'loadstart'. │ │ │ │ │ + this.events.triggerEvent("reload"); │ │ │ │ │ } else { │ │ │ │ │ - // The current component is outside the valid range. Let's │ │ │ │ │ - // see if the previous or next component is inside the range. │ │ │ │ │ - // If so, add the coordinate of the intersection with the │ │ │ │ │ - // valid range bounds. │ │ │ │ │ - if (i > 0) { │ │ │ │ │ - if (this.getShortString(components[i - 1])) { │ │ │ │ │ - strings.push(this.clipLine(components[i], │ │ │ │ │ - components[i - 1])); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (i < len - 1) { │ │ │ │ │ - if (this.getShortString(components[i + 1])) { │ │ │ │ │ - strings.push(this.clipLine(components[i], │ │ │ │ │ - components[i + 1])); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - complete = false; │ │ │ │ │ + this.isLoading = true; │ │ │ │ │ + this.events.triggerEvent("loadstart"); │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ + this.url = this.layer.getURL(this.bounds); │ │ │ │ │ │ │ │ │ │ - return { │ │ │ │ │ - path: strings.join(separator || ","), │ │ │ │ │ - complete: complete │ │ │ │ │ - }; │ │ │ │ │ + if (this.layer.useJSONP) { │ │ │ │ │ + // Use JSONP method to avoid xbrowser policy │ │ │ │ │ + var ols = new OpenLayers.Protocol.Script({ │ │ │ │ │ + url: this.url, │ │ │ │ │ + callback: function(response) { │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + this.json = response.data; │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + ols.read(); │ │ │ │ │ + this.request = ols; │ │ │ │ │ + } else { │ │ │ │ │ + // Use standard XHR │ │ │ │ │ + this.request = OpenLayers.Request.GET({ │ │ │ │ │ + url: this.url, │ │ │ │ │ + callback: function(response) { │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + if (response.status === 200) { │ │ │ │ │ + this.parseData(response.responseText); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this.unload(); │ │ │ │ │ + } │ │ │ │ │ + return drawn; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: clipLine │ │ │ │ │ - * Given two points (one inside the valid range, and one outside), │ │ │ │ │ - * clips the line betweeen the two points so that the new points are both │ │ │ │ │ - * inside the valid range. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * badComponent - {<OpenLayers.Geometry.Point>} original geometry of the │ │ │ │ │ - * invalid point │ │ │ │ │ - * goodComponent - {<OpenLayers.Geometry.Point>} original geometry of the │ │ │ │ │ - * valid point │ │ │ │ │ - * Returns │ │ │ │ │ - * {String} the SVG coordinate pair of the clipped point (like │ │ │ │ │ - * getShortString), or an empty string if both passed componets are at │ │ │ │ │ - * the same point. │ │ │ │ │ + * Method: abortLoading │ │ │ │ │ + * Cancel a pending request. │ │ │ │ │ */ │ │ │ │ │ - clipLine: function(badComponent, goodComponent) { │ │ │ │ │ - if (goodComponent.equals(badComponent)) { │ │ │ │ │ - return ""; │ │ │ │ │ - } │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var maxX = this.MAX_PIXEL - this.translationParameters.x; │ │ │ │ │ - var maxY = this.MAX_PIXEL - this.translationParameters.y; │ │ │ │ │ - var x1 = (goodComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y1 = this.top - goodComponent.y / resolution; │ │ │ │ │ - var x2 = (badComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y2 = this.top - badComponent.y / resolution; │ │ │ │ │ - var k; │ │ │ │ │ - if (x2 < -maxX || x2 > maxX) { │ │ │ │ │ - k = (y2 - y1) / (x2 - x1); │ │ │ │ │ - x2 = x2 < 0 ? -maxX : maxX; │ │ │ │ │ - y2 = y1 + (x2 - x1) * k; │ │ │ │ │ - } │ │ │ │ │ - if (y2 < -maxY || y2 > maxY) { │ │ │ │ │ - k = (x2 - x1) / (y2 - y1); │ │ │ │ │ - y2 = y2 < 0 ? -maxY : maxY; │ │ │ │ │ - x2 = x1 + (y2 - y1) * k; │ │ │ │ │ + abortLoading: function() { │ │ │ │ │ + if (this.request) { │ │ │ │ │ + this.request.abort(); │ │ │ │ │ + delete this.request; │ │ │ │ │ } │ │ │ │ │ - return x2 + "," + y2; │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: getShortString │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * Method: getFeatureInfo │ │ │ │ │ + * Get feature information associated with a pixel offset. If the pixel │ │ │ │ │ + * offset corresponds to a feature, the returned object will have id │ │ │ │ │ + * and data properties. Otherwise, null will be returned. │ │ │ │ │ + * │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ - * │ │ │ │ │ + * i - {Number} X-axis pixel offset (from top left of tile) │ │ │ │ │ + * j - {Number} Y-axis pixel offset (from top left of tile) │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} or false if point is outside the valid range │ │ │ │ │ + * {Object} Object with feature id and data properties corresponding to the │ │ │ │ │ + * given pixel offset. │ │ │ │ │ */ │ │ │ │ │ - getShortString: function(point) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = ((point.x - this.featureDx) / resolution + this.left); │ │ │ │ │ - var y = (this.top - point.y / resolution); │ │ │ │ │ - │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - return x + "," + y; │ │ │ │ │ - } else { │ │ │ │ │ - return false; │ │ │ │ │ + getFeatureInfo: function(i, j) { │ │ │ │ │ + var info = null; │ │ │ │ │ + if (this.json) { │ │ │ │ │ + var id = this.getFeatureId(i, j); │ │ │ │ │ + if (id !== null) { │ │ │ │ │ + info = { │ │ │ │ │ + id: id, │ │ │ │ │ + data: this.json.data[id] │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return info; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getPosition │ │ │ │ │ - * Finds the position of an svg node. │ │ │ │ │ - * │ │ │ │ │ + * Method: getFeatureId │ │ │ │ │ + * Get the identifier for the feature associated with a pixel offset. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * node - {DOMElement} │ │ │ │ │ - * │ │ │ │ │ + * i - {Number} X-axis pixel offset (from top left of tile) │ │ │ │ │ + * j - {Number} Y-axis pixel offset (from top left of tile) │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} hash with x and y properties, representing the coordinates │ │ │ │ │ - * within the svg coordinate system │ │ │ │ │ + * {Object} The feature identifier corresponding to the given pixel offset. │ │ │ │ │ + * Returns null if pixel doesn't correspond to a feature. │ │ │ │ │ */ │ │ │ │ │ - getPosition: function(node) { │ │ │ │ │ - return ({ │ │ │ │ │ - x: parseFloat(node.getAttributeNS(null, "cx")), │ │ │ │ │ - y: parseFloat(node.getAttributeNS(null, "cy")) │ │ │ │ │ - }); │ │ │ │ │ + getFeatureId: function(i, j) { │ │ │ │ │ + var id = null; │ │ │ │ │ + if (this.json) { │ │ │ │ │ + var resolution = this.utfgridResolution; │ │ │ │ │ + var row = Math.floor(j / resolution); │ │ │ │ │ + var col = Math.floor(i / resolution); │ │ │ │ │ + var charCode = this.json.grid[row].charCodeAt(col); │ │ │ │ │ + var index = this.indexFromCharCode(charCode); │ │ │ │ │ + var keys = this.json.keys; │ │ │ │ │ + if (!isNaN(index) && (index in keys)) { │ │ │ │ │ + id = keys[index]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return id; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: importSymbol │ │ │ │ │ - * add a new symbol definition from the rendererer's symbol hash │ │ │ │ │ - * │ │ │ │ │ + * Method: indexFromCharCode │ │ │ │ │ + * Given a character code for one of the UTFGrid "grid" characters, │ │ │ │ │ + * resolve the integer index for the feature id in the UTFGrid "keys" │ │ │ │ │ + * array. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * graphicName - {String} name of the symbol to import │ │ │ │ │ - * │ │ │ │ │ + * charCode - {Integer} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {DOMElement} - the imported symbol │ │ │ │ │ + * {Integer} Index for the feature id from the keys array. │ │ │ │ │ */ │ │ │ │ │ - importSymbol: function(graphicName) { │ │ │ │ │ - if (!this.defs) { │ │ │ │ │ - // create svg defs tag │ │ │ │ │ - this.defs = this.createDefs(); │ │ │ │ │ - } │ │ │ │ │ - var id = this.container.id + "-" + graphicName; │ │ │ │ │ - │ │ │ │ │ - // check if symbol already exists in the defs │ │ │ │ │ - var existing = document.getElementById(id); │ │ │ │ │ - if (existing != null) { │ │ │ │ │ - return existing; │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(graphicName + ' is not a valid symbol name'); │ │ │ │ │ + indexFromCharCode: function(charCode) { │ │ │ │ │ + if (charCode >= 93) { │ │ │ │ │ + charCode--; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - var symbolNode = this.nodeFactory(id, "symbol"); │ │ │ │ │ - var node = this.nodeFactory(null, "polygon"); │ │ │ │ │ - symbolNode.appendChild(node); │ │ │ │ │ - var symbolExtent = new OpenLayers.Bounds( │ │ │ │ │ - Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ - │ │ │ │ │ - var points = []; │ │ │ │ │ - var x, y; │ │ │ │ │ - for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ - symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ - symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ - symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ - points.push(x, ",", y); │ │ │ │ │ + if (charCode >= 35) { │ │ │ │ │ + charCode--; │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - node.setAttributeNS(null, "points", points.join(" ")); │ │ │ │ │ - │ │ │ │ │ - var width = symbolExtent.getWidth(); │ │ │ │ │ - var height = symbolExtent.getHeight(); │ │ │ │ │ - // create a viewBox three times as large as the symbol itself, │ │ │ │ │ - // to allow for strokeWidth being displayed correctly at the corners. │ │ │ │ │ - var viewBox = [symbolExtent.left - width, │ │ │ │ │ - symbolExtent.bottom - height, width * 3, height * 3 │ │ │ │ │ - ]; │ │ │ │ │ - symbolNode.setAttributeNS(null, "viewBox", viewBox.join(" ")); │ │ │ │ │ - this.symbolMetrics[id] = [ │ │ │ │ │ - Math.max(width, height), │ │ │ │ │ - symbolExtent.getCenterLonLat().lon, │ │ │ │ │ - symbolExtent.getCenterLonLat().lat │ │ │ │ │ - ]; │ │ │ │ │ - │ │ │ │ │ - this.defs.appendChild(symbolNode); │ │ │ │ │ - return symbolNode; │ │ │ │ │ + return charCode - 32; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: getFeatureIdFromEvent │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * evt - {Object} An <OpenLayers.Event> object │ │ │ │ │ + * Method: parseData │ │ │ │ │ + * Parse the JSON from a request │ │ │ │ │ * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * str - {String} UTFGrid as a JSON string. │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {String} A feature id or undefined. │ │ │ │ │ + * {Object} parsed javascript data │ │ │ │ │ */ │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var featureId = OpenLayers.Renderer.Elements.prototype.getFeatureIdFromEvent.apply(this, arguments); │ │ │ │ │ - if (!featureId) { │ │ │ │ │ - var target = evt.target; │ │ │ │ │ - featureId = target.parentNode && target != this.rendererRoot ? │ │ │ │ │ - target.parentNode._featureId : undefined; │ │ │ │ │ + parseData: function(str) { │ │ │ │ │ + if (!this.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.JSON(); │ │ │ │ │ } │ │ │ │ │ - return featureId; │ │ │ │ │ + this.json = this.format.read(str); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.SVG" │ │ │ │ │ -}); │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.SVG.LABEL_ALIGN │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_ALIGN = { │ │ │ │ │ - "l": "start", │ │ │ │ │ - "r": "end", │ │ │ │ │ - "b": "bottom", │ │ │ │ │ - "t": "hanging" │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.SVG.LABEL_VSHIFT │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_VSHIFT = { │ │ │ │ │ - // according to │ │ │ │ │ - // http://www.w3.org/Graphics/SVG/Test/20061213/htmlObjectHarness/full-text-align-02-b.html │ │ │ │ │ - // a baseline-shift of -70% shifts the text exactly from the │ │ │ │ │ - // bottom to the top of the baseline, so -35% moves the text to │ │ │ │ │ - // the center of the baseline. │ │ │ │ │ - "t": "-70%", │ │ │ │ │ - "b": "0" │ │ │ │ │ -}; │ │ │ │ │ + /** │ │ │ │ │ + * Method: clear │ │ │ │ │ + * Delete data stored with this tile. │ │ │ │ │ + */ │ │ │ │ │ + clear: function() { │ │ │ │ │ + this.json = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Renderer.SVG.LABEL_VFACTOR │ │ │ │ │ - * {Object} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_VFACTOR = { │ │ │ │ │ - "t": 0, │ │ │ │ │ - "b": -1 │ │ │ │ │ -}; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Tile.UTFGrid" │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Function: OpenLayers.Renderer.SVG.preventDefault │ │ │ │ │ - * *Deprecated*. Use <OpenLayers.Event.preventDefault> method instead. │ │ │ │ │ - * Used to prevent default events (especially opening images in a new tab on │ │ │ │ │ - * ctrl-click) from being executed for externalGraphic symbols │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Renderer.SVG.preventDefault = function(e) { │ │ │ │ │ - OpenLayers.Event.preventDefault(e); │ │ │ │ │ -}; │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ OpenLayers/Tile/Image/IFrame.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ @@ -90509,4439 +86224,8724 @@ │ │ │ │ │ if (this.useIFrame === false) { │ │ │ │ │ backBuffer = OpenLayers.Tile.Image.prototype.createBackBuffer.call(this); │ │ │ │ │ } │ │ │ │ │ return backBuffer; │ │ │ │ │ } │ │ │ │ │ }; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/cs-CZ.js │ │ │ │ │ + OpenLayers/Strategy/BBOX.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Mormegil │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Strategy.js │ │ │ │ │ + * @requires OpenLayers/Filter/Spatial.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["cs-CZ"] │ │ │ │ │ - * Dictionary for Česky. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Strategy.BBOX │ │ │ │ │ + * A simple strategy that reads new features when the viewport invalidates │ │ │ │ │ + * some bounds. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Strategy> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["cs-CZ"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ +OpenLayers.Strategy.BBOX = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "Nezpracovaná návratová hodnota ${statusText}", │ │ │ │ │ + /** │ │ │ │ │ + * Property: bounds │ │ │ │ │ + * {<OpenLayers.Bounds>} The current data bounds (in the same projection │ │ │ │ │ + * as the layer - not always the same projection as the map). │ │ │ │ │ + */ │ │ │ │ │ + bounds: null, │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Trvalý odkaz", │ │ │ │ │ + /** │ │ │ │ │ + * Property: resolution │ │ │ │ │ + * {Float} The current data resolution. │ │ │ │ │ + */ │ │ │ │ │ + resolution: null, │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Překryvné vrstvy", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: ratio │ │ │ │ │ + * {Float} The ratio of the data bounds to the viewport bounds (in each │ │ │ │ │ + * dimension). Default is 2. │ │ │ │ │ + */ │ │ │ │ │ + ratio: 2, │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Podkladové vrstvy", │ │ │ │ │ + /** │ │ │ │ │ + * Property: resFactor │ │ │ │ │ + * {Float} Optional factor used to determine when previously requested │ │ │ │ │ + * features are invalid. If set, the resFactor will be compared to the │ │ │ │ │ + * resolution of the previous request to the current map resolution. │ │ │ │ │ + * If resFactor > (old / new) and 1/resFactor < (old / new). If you │ │ │ │ │ + * set a resFactor of 1, data will be requested every time the │ │ │ │ │ + * resolution changes. If you set a resFactor of 3, data will be │ │ │ │ │ + * requested if the old resolution is 3 times the new, or if the new is │ │ │ │ │ + * 3 times the old. If the old bounds do not contain the new bounds │ │ │ │ │ + * new data will always be requested (with or without considering │ │ │ │ │ + * resFactor). │ │ │ │ │ + */ │ │ │ │ │ + resFactor: null, │ │ │ │ │ │ │ │ │ │ - 'noFID': "Nelze aktualizovat prvek, pro který neexistuje FID.", │ │ │ │ │ + /** │ │ │ │ │ + * Property: response │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} The protocol response object returned │ │ │ │ │ + * by the layer protocol. │ │ │ │ │ + */ │ │ │ │ │ + response: null, │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Váš prohlížeč nepodporuje vykreslování vektorů. Momentálně podporované nástroje jsou::\n${renderers}", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Strategy.BBOX │ │ │ │ │ + * Create a new BBOX strategy. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - 'minZoomLevelError': "Vlastnost minZoomLevel by se měla používat pouze s potomky FixedZoomLevels vrstvami. To znamená, že vrstva wfs kontroluje, zda-li minZoomLevel není zbytek z minulosti.Nelze to ovšem vyjmout bez možnosti, že bychom rozbili aplikace postavené na OL, které by na tom mohly záviset. Proto tuto vlastnost nedoporučujeme používat -- kontrola minZoomLevel bude odstraněna ve verzi 3.0. Použijte prosím raději nastavení min/max podle příkaldu popsaného na: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + /** │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Set up strategy with regard to reading new batches of remote data. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The strategy was successfully activated. │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "moveend": this.update, │ │ │ │ │ + "refresh": this.update, │ │ │ │ │ + "visibilitychanged": this.update, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.update(); │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "WFS Transaction: ÚSPĚCH ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Tear down strategy with regard to reading new batches of remote data. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + "moveend": this.update, │ │ │ │ │ + "refresh": this.update, │ │ │ │ │ + "visibilitychanged": this.update, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "WFS Transaction: CHYBA ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: update │ │ │ │ │ + * Callback function called on "moveend" or "refresh" layer events. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will determine │ │ │ │ │ + * the behaviour of this Strategy │ │ │ │ │ + * │ │ │ │ │ + * Valid options include: │ │ │ │ │ + * force - {Boolean} if true, new data must be unconditionally read. │ │ │ │ │ + * noAbort - {Boolean} if true, do not abort previous requests. │ │ │ │ │ + */ │ │ │ │ │ + update: function(options) { │ │ │ │ │ + var mapBounds = this.getMapBounds(); │ │ │ │ │ + if (mapBounds !== null && ((options && options.force) || │ │ │ │ │ + (this.layer.visibility && this.layer.calculateInRange() && this.invalidBounds(mapBounds)))) { │ │ │ │ │ + this.calculateBounds(mapBounds); │ │ │ │ │ + this.resolution = this.layer.map.getResolution(); │ │ │ │ │ + this.triggerRead(options); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Nepodařilo se správně načíst vrstvu Google.\x3cbr\x3e\x3cbr\x3eAbyste se zbavili této zprávy, zvolte jinou základní vrstvu v přepínači vrstev.\x3cbr\x3e\x3cbr\x3eTo se většinou stává, pokud nebyl načten skript, nebo neobsahuje správný klíč pro API pro tuto stránku.\x3cbr\x3e\x3cbr\x3eVývojáři: Pro pomoc, aby tohle fungovalo , \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eklikněte sem\x3c/a\x3e", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getMapBounds │ │ │ │ │ + * Get the map bounds expressed in the same projection as this layer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} Map bounds in the projection of the layer. │ │ │ │ │ + */ │ │ │ │ │ + getMapBounds: function() { │ │ │ │ │ + if (this.layer.map === null) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + var bounds = this.layer.map.getExtent(); │ │ │ │ │ + if (bounds && !this.layer.projection.equals( │ │ │ │ │ + this.layer.map.getProjectionObject())) { │ │ │ │ │ + bounds = bounds.clone().transform( │ │ │ │ │ + this.layer.map.getProjectionObject(), this.layer.projection │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return bounds; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "The ${layerType} Layer was unable to load correctly.\x3cbr\x3e\x3cbr\x3eTo get rid of this message, select a new BaseLayer in the layer switcher in the upper-right corner.\x3cbr\x3e\x3cbr\x3eMost likely, this is because the ${layerLib} library script was either not correctly included.\x3cbr\x3e\x3cbr\x3eDevelopers: For help getting this working correctly, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eclick here\x3c/a\x3e", │ │ │ │ │ + /** │ │ │ │ │ + * Method: invalidBounds │ │ │ │ │ + * Determine whether the previously requested set of features is invalid. │ │ │ │ │ + * This occurs when the new map bounds do not contain the previously │ │ │ │ │ + * requested bounds. In addition, if <resFactor> is set, it will be │ │ │ │ │ + * considered. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * mapBounds - {<OpenLayers.Bounds>} the current map extent, will be │ │ │ │ │ + * retrieved from the map object if not provided │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + invalidBounds: function(mapBounds) { │ │ │ │ │ + if (!mapBounds) { │ │ │ │ │ + mapBounds = this.getMapBounds(); │ │ │ │ │ + } │ │ │ │ │ + var invalid = !this.bounds || !this.bounds.containsBounds(mapBounds); │ │ │ │ │ + if (!invalid && this.resFactor) { │ │ │ │ │ + var ratio = this.resolution / this.layer.map.getResolution(); │ │ │ │ │ + invalid = (ratio >= this.resFactor || ratio <= (1 / this.resFactor)); │ │ │ │ │ + } │ │ │ │ │ + return invalid; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Měřítko = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateBounds │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * mapBounds - {<OpenLayers.Bounds>} the current map extent, will be │ │ │ │ │ + * retrieved from the map object if not provided │ │ │ │ │ + */ │ │ │ │ │ + calculateBounds: function(mapBounds) { │ │ │ │ │ + if (!mapBounds) { │ │ │ │ │ + mapBounds = this.getMapBounds(); │ │ │ │ │ + } │ │ │ │ │ + var center = mapBounds.getCenterLonLat(); │ │ │ │ │ + var dataWidth = mapBounds.getWidth() * this.ratio; │ │ │ │ │ + var dataHeight = mapBounds.getHeight() * this.ratio; │ │ │ │ │ + this.bounds = new OpenLayers.Bounds( │ │ │ │ │ + center.lon - (dataWidth / 2), │ │ │ │ │ + center.lat - (dataHeight / 2), │ │ │ │ │ + center.lon + (dataWidth / 2), │ │ │ │ │ + center.lat + (dataHeight / 2) │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Použil jste volbu \'reproject\' ve vrstvě ${layerName}. Tato volba není doporučená: byla zde proto, aby bylo možno zobrazovat data z okomerčních serverů, ale tato funkce je nyní zajištěna pomocí podpory Spherical Mercator. Více informací naleznete na http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + /** │ │ │ │ │ + * Method: triggerRead │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Additional options for the protocol's read method │ │ │ │ │ + * (optional) │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Protocol.Response>} The protocol response object │ │ │ │ │ + * returned by the layer protocol. │ │ │ │ │ + */ │ │ │ │ │ + triggerRead: function(options) { │ │ │ │ │ + if (this.response && !(options && options.noAbort === true)) { │ │ │ │ │ + this.layer.protocol.abort(this.response); │ │ │ │ │ + this.layer.events.triggerEvent("loadend"); │ │ │ │ │ + } │ │ │ │ │ + var evt = { │ │ │ │ │ + filter: this.createFilter() │ │ │ │ │ + }; │ │ │ │ │ + this.layer.events.triggerEvent("loadstart", evt); │ │ │ │ │ + this.response = this.layer.protocol.read( │ │ │ │ │ + OpenLayers.Util.applyDefaults({ │ │ │ │ │ + filter: evt.filter, │ │ │ │ │ + callback: this.merge, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options)); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Tato metoda je zavržená a bude ve verzi 3.0 odstraněna. Prosím, použijte raději ${newMethod}." │ │ │ │ │ + /** │ │ │ │ │ + * Method: createFilter │ │ │ │ │ + * Creates a spatial BBOX filter. If the layer that this strategy belongs │ │ │ │ │ + * to has a filter property, this filter will be combined with the BBOX │ │ │ │ │ + * filter. │ │ │ │ │ + * │ │ │ │ │ + * Returns │ │ │ │ │ + * {<OpenLayers.Filter>} The filter object. │ │ │ │ │ + */ │ │ │ │ │ + createFilter: function() { │ │ │ │ │ + var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ + value: this.bounds, │ │ │ │ │ + projection: this.layer.projection │ │ │ │ │ + }); │ │ │ │ │ + if (this.layer.filter) { │ │ │ │ │ + filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ + type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ + filters: [this.layer.filter, filter] │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return filter; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/ro.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: merge │ │ │ │ │ + * Given a list of features, determine which ones to add to the layer. │ │ │ │ │ + * If the layer projection differs from the map projection, features │ │ │ │ │ + * will be transformed from the layer projection to the map projection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object passed │ │ │ │ │ + * by the protocol. │ │ │ │ │ + */ │ │ │ │ │ + merge: function(resp) { │ │ │ │ │ + this.layer.destroyFeatures(); │ │ │ │ │ + if (resp.success()) { │ │ │ │ │ + var features = resp.features; │ │ │ │ │ + if (features && features.length > 0) { │ │ │ │ │ + var remote = this.layer.projection; │ │ │ │ │ + var local = this.layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var geom; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + geom = features[i].geometry; │ │ │ │ │ + if (geom) { │ │ │ │ │ + geom.transform(remote, local); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.layer.addFeatures(features); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this.bounds = null; │ │ │ │ │ + } │ │ │ │ │ + this.response = null; │ │ │ │ │ + this.layer.events.triggerEvent("loadend", { │ │ │ │ │ + response: resp │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["ro"] │ │ │ │ │ - * Dictionary for Romanian. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Lang.ro = { │ │ │ │ │ - 'unhandledRequest': "Cerere nesoluționată return ${statusText}", │ │ │ │ │ - 'Permalink': "Legatură permanentă", │ │ │ │ │ - 'Overlays': "Straturi vector", │ │ │ │ │ - 'Base Layer': "Straturi de bază", │ │ │ │ │ - 'noFID': "Nu pot actualiza un feature pentru care nu există FID.", │ │ │ │ │ - 'browserNotSupported': "Browserul tău nu suportă afișarea vectorilor. Supoetul curent pentru randare:\n${renderers}", │ │ │ │ │ - // console message │ │ │ │ │ - 'minZoomLevelError': "Proprietatea minZoomLevel este doar pentru a fi folosită " + │ │ │ │ │ - "cu straturile FixedZoomLevels-descendent. De aceea acest " + │ │ │ │ │ - "strat wfs verifică dacă minZoomLevel este o relicvă" + │ │ │ │ │ - ". Nu îl putem , oricum, înlătura fără " + │ │ │ │ │ - "a afecta aplicațiile Openlayers care depind de ea." + │ │ │ │ │ - " De aceea considerăm depreciat -- minZoomLevel " + │ │ │ │ │ - "și îl vom înlătura în 3.0. Folosește " + │ │ │ │ │ - "min/max resolution cum este descrisă aici: " + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - 'commitSuccess': "Tranzacție WFS: SUCCES ${response}", │ │ │ │ │ - 'commitFailed': "Tranzacție WFS : EȘEC ${response}", │ │ │ │ │ - 'googleWarning': "Stratul Google nu a putut fi încărcat corect.<br><br>" + │ │ │ │ │ - "Pentru a elimina acest mesaj, selectează un nou strat de bază " + │ │ │ │ │ - "în Layerswitcher din colțul dreata-sus.<br><br>" + │ │ │ │ │ - "Asta datorită, faptului că Google Maps library " + │ │ │ │ │ - "script nu este inclus, sau nu conține " + │ │ │ │ │ - "cheia API corectă pentru situl tău.<br><br>" + │ │ │ │ │ - "Developeri: Pentru ajutor, " + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ - "target='_blank'>apăsați aici</a>", │ │ │ │ │ - 'getLayerWarning': "Stratul ${layerType} nu a putut fi încărcat corect.<br><br>" + │ │ │ │ │ - "pentru a înlătura acest mesaj, selectează un nou strat de bază " + │ │ │ │ │ - "Acesta eroare apare de obicei când ${layerLib} library " + │ │ │ │ │ - "script nu a fost încărcat corect.<br><br>" + │ │ │ │ │ - "Developeri: Pentru ajutor privind utilizarea corectă, " + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ - "target='_blank'>apasă aici</a>", │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Scara = 1 : ${scaleDenom}", │ │ │ │ │ - //labels for the graticule control │ │ │ │ │ - 'W': 'V', │ │ │ │ │ - 'E': 'E', │ │ │ │ │ - 'N': 'N', │ │ │ │ │ - 'S': 'S', │ │ │ │ │ - 'Graticule': 'Graticule', │ │ │ │ │ - // console message │ │ │ │ │ - 'reprojectDeprecated': "folosești opțiunea 'reproject' " + │ │ │ │ │ - "pentru stratul ${layerName} . Această opțiune este depreciată: " + │ │ │ │ │ - "a fost utilizată pentru afișarea straturilor de bază comerciale " + │ │ │ │ │ - "Mai multe informații despre proiecția Mercator sunt disponibile aici " + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - // console message │ │ │ │ │ - 'methodDeprecated': "Această metodă este depreciată și va fi înlăturată in versiunea 3.0. " + │ │ │ │ │ - "folosește metoda ${newMethod}.", │ │ │ │ │ - // **** end **** │ │ │ │ │ - 'end': '' │ │ │ │ │ -}; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.BBOX" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/pl.js │ │ │ │ │ + OpenLayers/Strategy/Filter.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators: │ │ │ │ │ - * - Arkadiusz Grabka │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Strategy.js │ │ │ │ │ + * @requires OpenLayers/Filter.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["pl"] │ │ │ │ │ - * Dictionary for Polish. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Strategy.Filter │ │ │ │ │ + * Strategy for limiting features that get added to a layer by │ │ │ │ │ + * evaluating a filter. The strategy maintains a cache of │ │ │ │ │ + * all features until removeFeatures is called on the layer. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Strategy> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["pl"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - │ │ │ │ │ - 'unhandledRequest': "Nieobsługiwane żądanie zwróciło ${statusText}", │ │ │ │ │ - │ │ │ │ │ - 'Permalink': "Permalink", │ │ │ │ │ - │ │ │ │ │ - 'Overlays': "Nakładki", │ │ │ │ │ - │ │ │ │ │ - 'Base Layer': "Warstwa podstawowa", │ │ │ │ │ +OpenLayers.Strategy.Filter = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ │ │ │ │ │ - 'noFID': "Nie można zaktualizować funkcji, dla których nie ma FID.", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: filter │ │ │ │ │ + * {<OpenLayers.Filter>} Filter for limiting features sent to the layer. │ │ │ │ │ + * Use the <setFilter> method to update this filter after construction. │ │ │ │ │ + */ │ │ │ │ │ + filter: null, │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Twoja przeglądarka nie obsługuje renderowania wektorów. Obecnie obsługiwane renderowanie to:\n${renderers}", │ │ │ │ │ + /** │ │ │ │ │ + * Property: cache │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} List of currently cached │ │ │ │ │ + * features. │ │ │ │ │ + */ │ │ │ │ │ + cache: null, │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'minZoomLevelError': "Właściwość minZoomLevel jest przeznaczona tylko do użytku " + │ │ │ │ │ - "z warstwami FixedZoomLevels-descendent." + │ │ │ │ │ - "Warstwa wfs, która sprawdza minZoomLevel jest reliktem przeszłości." + │ │ │ │ │ - "Nie możemy jej jednak usunąc bez mozliwości łamania OL aplikacji, " + │ │ │ │ │ - "które mogą być od niej zależne. " + │ │ │ │ │ - "Dlatego jesteśmy za deprecjację -- minZoomLevel " + │ │ │ │ │ - "zostanie usunięta w wersji 3.0. W zamian prosze użyj " + │ │ │ │ │ - "min/max rozdzielczości w sposób opisany tutaj: " + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + /** │ │ │ │ │ + * Property: caching │ │ │ │ │ + * {Boolean} The filter is currently caching features. │ │ │ │ │ + */ │ │ │ │ │ + caching: false, │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "Transakcja WFS: SUKCES ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Strategy.Filter │ │ │ │ │ + * Create a new filter strategy. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "Transakcja WFS: FAILED ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ + * By default, this strategy automatically activates itself when a layer │ │ │ │ │ + * is added to a map. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} True if the strategy was successfully activated or false if │ │ │ │ │ + * the strategy was already active. │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.cache = []; │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "beforefeaturesadded": this.handleAdd, │ │ │ │ │ + "beforefeaturesremoved": this.handleRemove, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Warstwa Google nie był w stanie załadować się poprawnie.<br><br>" + │ │ │ │ │ - "Aby pozbyć się tej wiadomości, wybierz nową Warstwe podstawową " + │ │ │ │ │ - "w przełączniku warstw w górnym prawym rogu mapy.<br><br>" + │ │ │ │ │ - "Najprawdopodobniej jest to spowodowane tym, że biblioteka Google Maps " + │ │ │ │ │ - "nie jest załadowana, lub nie zawiera poprawnego klucza do API dla twojej strony<br><br>" + │ │ │ │ │ - "Programisto: Aby uzyskać pomoc , " + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ - "target='_blank'>kliknij tutaj</a>", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the strategy. Clear the feature cache. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} True if the strategy was successfully deactivated or false if │ │ │ │ │ + * the strategy was already inactive. │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + this.cache = null; │ │ │ │ │ + if (this.layer && this.layer.events) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + "beforefeaturesadded": this.handleAdd, │ │ │ │ │ + "beforefeaturesremoved": this.handleRemove, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Strategy.prototype.deactivate.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "Warstwa ${layerType} nie mogła zostać załadowana poprawnie.<br><br>" + │ │ │ │ │ - "Aby pozbyć się tej wiadomości, wybierz nową Warstwe podstawową " + │ │ │ │ │ - "w przełączniku warstw w górnym prawym rogu mapy.<br><br>" + │ │ │ │ │ - "Najprawdopodobniej jest to spowodowane tym, że biblioteka ${layerLib} " + │ │ │ │ │ - "nie jest załadowana, lub może(o ile biblioteka tego wymaga) " + │ │ │ │ │ - "byc potrzebny klucza do API dla twojej strony<br><br>" + │ │ │ │ │ - "Programisto: Aby uzyskać pomoc , " + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ - "target='_blank'>kliknij tutaj</a>", │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleAdd │ │ │ │ │ + */ │ │ │ │ │ + handleAdd: function(event) { │ │ │ │ │ + if (!this.caching && this.filter) { │ │ │ │ │ + var features = event.features; │ │ │ │ │ + event.features = []; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0, ii = features.length; i < ii; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (this.filter.evaluate(feature)) { │ │ │ │ │ + event.features.push(feature); │ │ │ │ │ + } else { │ │ │ │ │ + this.cache.push(feature); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Skala = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: handleRemove │ │ │ │ │ + */ │ │ │ │ │ + handleRemove: function(event) { │ │ │ │ │ + if (!this.caching) { │ │ │ │ │ + this.cache = []; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - //labels for the graticule control │ │ │ │ │ - 'W': 'ZACH', │ │ │ │ │ - 'E': 'WSCH', │ │ │ │ │ - 'N': 'PN', │ │ │ │ │ - 'S': 'PD', │ │ │ │ │ - 'Graticule': 'Siatka', │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setFilter │ │ │ │ │ + * Update the filter for this strategy. This will re-evaluate │ │ │ │ │ + * any features on the layer and in the cache. Only features │ │ │ │ │ + * for which filter.evalute(feature) returns true will be │ │ │ │ │ + * added to the layer. Others will be cached by the strategy. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * filter - {<OpenLayers.Filter>} A filter for evaluating features. │ │ │ │ │ + */ │ │ │ │ │ + setFilter: function(filter) { │ │ │ │ │ + this.filter = filter; │ │ │ │ │ + var previousCache = this.cache; │ │ │ │ │ + this.cache = []; │ │ │ │ │ + // look through layer for features to remove from layer │ │ │ │ │ + this.handleAdd({ │ │ │ │ │ + features: this.layer.features │ │ │ │ │ + }); │ │ │ │ │ + // cache now contains features to remove from layer │ │ │ │ │ + if (this.cache.length > 0) { │ │ │ │ │ + this.caching = true; │ │ │ │ │ + this.layer.removeFeatures(this.cache.slice()); │ │ │ │ │ + this.caching = false; │ │ │ │ │ + } │ │ │ │ │ + // now look through previous cache for features to add to layer │ │ │ │ │ + if (previousCache.length > 0) { │ │ │ │ │ + var event = { │ │ │ │ │ + features: previousCache │ │ │ │ │ + }; │ │ │ │ │ + this.handleAdd(event); │ │ │ │ │ + if (event.features.length > 0) { │ │ │ │ │ + // event has features to add to layer │ │ │ │ │ + this.caching = true; │ │ │ │ │ + this.layer.addFeatures(event.features); │ │ │ │ │ + this.caching = false; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'reprojectDeprecated': "w warstwie ${layerName} używasz opcji 'reproject'. " + │ │ │ │ │ - "Ta opcja jest przestarzała: " + │ │ │ │ │ - "jej zastosowanie został zaprojektowany, aby wspierać wyświetlania danych przez komercyjne mapy, " + │ │ │ │ │ - "jednak obecnie ta funkcjonalność powinien zostać osiągnięty za pomocą Spherical Mercator " + │ │ │ │ │ - "its use was designed to support displaying data over commercial. Więcje informacji na ten temat możesz znaleźć na stronie " + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Filter" │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'methodDeprecated': "Ta metoda jest przestarzała i będzie usunięta od wersji 3.0. " + │ │ │ │ │ - "W zamian użyj ${newMethod}." │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/nb.js │ │ │ │ │ + OpenLayers/Strategy/Refresh.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Strategy.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["nb"] │ │ │ │ │ - * Dictionary for norwegian bokmål (Norway). Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Strategy.Refresh │ │ │ │ │ + * A strategy that refreshes the layer. By default the strategy waits for a │ │ │ │ │ + * call to <refresh> before refreshing. By configuring the strategy with │ │ │ │ │ + * the <interval> option, refreshing can take place automatically. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Strategy> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["nb"] = { │ │ │ │ │ - │ │ │ │ │ - 'unhandledRequest': "Ubehandlet forespørsel returnerte ${statusText}", │ │ │ │ │ - │ │ │ │ │ - 'Permalink': "Kobling til denne siden", │ │ │ │ │ - │ │ │ │ │ - 'Overlays': "Kartlag", │ │ │ │ │ +OpenLayers.Strategy.Refresh = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Bakgrunnskart", │ │ │ │ │ + /** │ │ │ │ │ + * Property: force │ │ │ │ │ + * {Boolean} Force a refresh on the layer. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + force: false, │ │ │ │ │ │ │ │ │ │ - 'noFID': "Kan ikke oppdatere et feature (et objekt) som ikke har FID.", │ │ │ │ │ + /** │ │ │ │ │ + * Property: interval │ │ │ │ │ + * {Number} Auto-refresh. Default is 0. If > 0, layer will be refreshed │ │ │ │ │ + * every N milliseconds. │ │ │ │ │ + */ │ │ │ │ │ + interval: 0, │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Din nettleser støtter ikke vektortegning. Tegnemetodene som støttes er:\n${renderers}", │ │ │ │ │ + /** │ │ │ │ │ + * Property: timer │ │ │ │ │ + * {Number} The id of the timer. │ │ │ │ │ + */ │ │ │ │ │ + timer: null, │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'minZoomLevelError': "Egenskapen minZoomLevel er kun ment til bruk på lag " + │ │ │ │ │ - "basert på FixedZoomLevels. At dette wfs-laget sjekker " + │ │ │ │ │ - "minZoomLevel er en etterlevning fra tidligere versjoner. Det kan dog ikke " + │ │ │ │ │ - "tas bort uten å risikere at OL-baserte applikasjoner " + │ │ │ │ │ - "slutter å virke, så det er merket som foreldet: " + │ │ │ │ │ - "minZoomLevel i sjekken nedenfor vil fjernes i 3.0. " + │ │ │ │ │ - "Vennligst bruk innstillingene for min/maks oppløsning " + │ │ │ │ │ - "som er beskrevet her: " + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Strategy.Refresh │ │ │ │ │ + * Create a new Refresh strategy. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "WFS-transaksjon: LYKTES ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} True if the strategy was successfully activated. │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + if (this.layer.visibility === true) { │ │ │ │ │ + this.start(); │ │ │ │ │ + } │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "visibilitychanged": this.reset, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "WFS-transaksjon: MISLYKTES ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ + * tear-down. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} True if the strategy was successfully deactivated. │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.stop(); │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + "visibilitychanged": this.reset, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Google-laget kunne ikke lastes.<br><br>" + │ │ │ │ │ - "Bytt til et annet bakgrunnslag i lagvelgeren i " + │ │ │ │ │ - "øvre høyre hjørne for å slippe denne meldingen.<br><br>" + │ │ │ │ │ - "Sannsynligvis forårsakes feilen av at Google Maps-biblioteket " + │ │ │ │ │ - "ikke er riktig inkludert på nettsiden, eller at det ikke er " + │ │ │ │ │ - "angitt riktig API-nøkkel for nettstedet.<br><br>" + │ │ │ │ │ - "Utviklere: For hjelp til å få dette til å virke se " + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ - "target='_blank'>her</a>.", │ │ │ │ │ + /** │ │ │ │ │ + * Method: reset │ │ │ │ │ + * Start or cancel the refresh interval depending on the visibility of │ │ │ │ │ + * the layer. │ │ │ │ │ + */ │ │ │ │ │ + reset: function() { │ │ │ │ │ + if (this.layer.visibility === true) { │ │ │ │ │ + this.start(); │ │ │ │ │ + } else { │ │ │ │ │ + this.stop(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "${layerType}-laget kunne ikke lastes.<br><br>" + │ │ │ │ │ - "Bytt til et annet bakgrunnslag i lagvelgeren i " + │ │ │ │ │ - "øvre høyre hjørne for å slippe denne meldingen.<br><br>" + │ │ │ │ │ - "Sannsynligvis forårsakes feilen av at " + │ │ │ │ │ - "${layerLib}-biblioteket ikke var riktig inkludert " + │ │ │ │ │ - "på nettsiden.<br><br>" + │ │ │ │ │ - "Utviklere: For hjelp til å få dette til å virke se " + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ - "target='_blank'>her</a>.", │ │ │ │ │ + /** │ │ │ │ │ + * Method: start │ │ │ │ │ + * Start the refresh interval. │ │ │ │ │ + */ │ │ │ │ │ + start: function() { │ │ │ │ │ + if (this.interval && typeof this.interval === "number" && │ │ │ │ │ + this.interval > 0) { │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "<strong>Skala</strong> 1 : ${scaleDenom}", │ │ │ │ │ + this.timer = window.setInterval( │ │ │ │ │ + OpenLayers.Function.bind(this.refresh, this), │ │ │ │ │ + this.interval); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'reprojectDeprecated': "Du bruker innstillingen 'reproject' på laget ${layerName}. " + │ │ │ │ │ - "Denne innstillingen er foreldet, den var ment for å støtte " + │ │ │ │ │ - "visning av kartdata over kommersielle bakgrunnskart, men det " + │ │ │ │ │ - "bør nå gjøres med støtten for Spherical Mercator. Mer informasjon " + │ │ │ │ │ - "finnes på http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: refresh │ │ │ │ │ + * Tell the strategy to refresh which will refresh the layer. │ │ │ │ │ + */ │ │ │ │ │ + refresh: function() { │ │ │ │ │ + if (this.layer && this.layer.refresh && │ │ │ │ │ + typeof this.layer.refresh == "function") { │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'methodDeprecated': "Denne metoden er markert som foreldet og vil bli fjernet i 3.0. " + │ │ │ │ │ - "Vennligst bruk ${newMethod} i stedet.", │ │ │ │ │ + this.layer.refresh({ │ │ │ │ │ + force: this.force │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'end': '' │ │ │ │ │ -}; │ │ │ │ │ + /** │ │ │ │ │ + * Method: stop │ │ │ │ │ + * Cancels the refresh interval. │ │ │ │ │ + */ │ │ │ │ │ + stop: function() { │ │ │ │ │ + if (this.timer !== null) { │ │ │ │ │ + window.clearInterval(this.timer); │ │ │ │ │ + this.timer = null; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -OpenLayers.Lang["no"] = OpenLayers.Lang["nb"]; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Refresh" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/is.js │ │ │ │ │ + OpenLayers/Strategy/Cluster.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Ævar Arnfjörð Bjarmason │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Strategy.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["is"] │ │ │ │ │ - * Dictionary for Íslenska. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Strategy.Cluster │ │ │ │ │ + * Strategy for vector feature clustering. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Strategy> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["is"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ +OpenLayers.Strategy.Cluster = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Varanlegur tengill", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: distance │ │ │ │ │ + * {Integer} Pixel distance between features that should be considered a │ │ │ │ │ + * single cluster. Default is 20 pixels. │ │ │ │ │ + */ │ │ │ │ │ + distance: 20, │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Þekjur", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: threshold │ │ │ │ │ + * {Integer} Optional threshold below which original features will be │ │ │ │ │ + * added to the layer instead of clusters. For example, a threshold │ │ │ │ │ + * of 3 would mean that any time there are 2 or fewer features in │ │ │ │ │ + * a cluster, those features will be added directly to the layer instead │ │ │ │ │ + * of a cluster representing those features. Default is null (which is │ │ │ │ │ + * equivalent to 1 - meaning that clusters may contain just one feature). │ │ │ │ │ + */ │ │ │ │ │ + threshold: null, │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Grunnlag", │ │ │ │ │ + /** │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} Cached features. │ │ │ │ │ + */ │ │ │ │ │ + features: null, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Skali = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * Property: clusters │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} Calculated clusters. │ │ │ │ │ + */ │ │ │ │ │ + clusters: null, │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Þetta fall hefur verið úrelt og verður fjarlægt í 3.0. Notaðu ${newMethod} í staðin." │ │ │ │ │ + /** │ │ │ │ │ + * Property: clustering │ │ │ │ │ + * {Boolean} The strategy is currently clustering features. │ │ │ │ │ + */ │ │ │ │ │ + clustering: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: resolution │ │ │ │ │ + * {Float} The resolution (map units per pixel) of the current cluster set. │ │ │ │ │ + */ │ │ │ │ │ + resolution: null, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Strategy.Cluster │ │ │ │ │ + * Create a new clustering strategy. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The strategy was successfully activated. │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "beforefeaturesadded": this.cacheFeatures, │ │ │ │ │ + "featuresremoved": this.clearCache, │ │ │ │ │ + "moveend": this.cluster, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ + * tear-down. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.clearCache(); │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + "beforefeaturesadded": this.cacheFeatures, │ │ │ │ │ + "featuresremoved": this.clearCache, │ │ │ │ │ + "moveend": this.cluster, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: cacheFeatures │ │ │ │ │ + * Cache features before they are added to the layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * event - {Object} The event that this was listening for. This will come │ │ │ │ │ + * with a batch of features to be clustered. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} False to stop features from being added to the layer. │ │ │ │ │ + */ │ │ │ │ │ + cacheFeatures: function(event) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + if (!this.clustering) { │ │ │ │ │ + this.clearCache(); │ │ │ │ │ + this.features = event.features; │ │ │ │ │ + this.cluster(); │ │ │ │ │ + propagate = false; │ │ │ │ │ + } │ │ │ │ │ + return propagate; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clearCache │ │ │ │ │ + * Clear out the cached features. │ │ │ │ │ + */ │ │ │ │ │ + clearCache: function() { │ │ │ │ │ + if (!this.clustering) { │ │ │ │ │ + this.features = null; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: cluster │ │ │ │ │ + * Cluster features based on some threshold distance. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * event - {Object} The event received when cluster is called as a │ │ │ │ │ + * result of a moveend event. │ │ │ │ │ + */ │ │ │ │ │ + cluster: function(event) { │ │ │ │ │ + if ((!event || event.zoomChanged) && this.features) { │ │ │ │ │ + var resolution = this.layer.map.getResolution(); │ │ │ │ │ + if (resolution != this.resolution || !this.clustersExist()) { │ │ │ │ │ + this.resolution = resolution; │ │ │ │ │ + var clusters = []; │ │ │ │ │ + var feature, clustered, cluster; │ │ │ │ │ + for (var i = 0; i < this.features.length; ++i) { │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + clustered = false; │ │ │ │ │ + for (var j = clusters.length - 1; j >= 0; --j) { │ │ │ │ │ + cluster = clusters[j]; │ │ │ │ │ + if (this.shouldCluster(cluster, feature)) { │ │ │ │ │ + this.addToCluster(cluster, feature); │ │ │ │ │ + clustered = true; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!clustered) { │ │ │ │ │ + clusters.push(this.createCluster(this.features[i])); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.clustering = true; │ │ │ │ │ + this.layer.removeAllFeatures(); │ │ │ │ │ + this.clustering = false; │ │ │ │ │ + if (clusters.length > 0) { │ │ │ │ │ + if (this.threshold > 1) { │ │ │ │ │ + var clone = clusters.slice(); │ │ │ │ │ + clusters = []; │ │ │ │ │ + var candidate; │ │ │ │ │ + for (var i = 0, len = clone.length; i < len; ++i) { │ │ │ │ │ + candidate = clone[i]; │ │ │ │ │ + if (candidate.attributes.count < this.threshold) { │ │ │ │ │ + Array.prototype.push.apply(clusters, candidate.cluster); │ │ │ │ │ + } else { │ │ │ │ │ + clusters.push(candidate); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.clustering = true; │ │ │ │ │ + // A legitimate feature addition could occur during this │ │ │ │ │ + // addFeatures call. For clustering to behave well, features │ │ │ │ │ + // should be removed from a layer before requesting a new batch. │ │ │ │ │ + this.layer.addFeatures(clusters); │ │ │ │ │ + this.clustering = false; │ │ │ │ │ + } │ │ │ │ │ + this.clusters = clusters; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clustersExist │ │ │ │ │ + * Determine whether calculated clusters are already on the layer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The calculated clusters are already on the layer. │ │ │ │ │ + */ │ │ │ │ │ + clustersExist: function() { │ │ │ │ │ + var exist = false; │ │ │ │ │ + if (this.clusters && this.clusters.length > 0 && │ │ │ │ │ + this.clusters.length == this.layer.features.length) { │ │ │ │ │ + exist = true; │ │ │ │ │ + for (var i = 0; i < this.clusters.length; ++i) { │ │ │ │ │ + if (this.clusters[i] != this.layer.features[i]) { │ │ │ │ │ + exist = false; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return exist; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: shouldCluster │ │ │ │ │ + * Determine whether to include a feature in a given cluster. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * cluster - {<OpenLayers.Feature.Vector>} A cluster. │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} A feature. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The feature should be included in the cluster. │ │ │ │ │ + */ │ │ │ │ │ + shouldCluster: function(cluster, feature) { │ │ │ │ │ + var cc = cluster.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + var fc = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + var distance = ( │ │ │ │ │ + Math.sqrt( │ │ │ │ │ + Math.pow((cc.lon - fc.lon), 2) + Math.pow((cc.lat - fc.lat), 2) │ │ │ │ │ + ) / this.resolution │ │ │ │ │ + ); │ │ │ │ │ + return (distance <= this.distance); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: addToCluster │ │ │ │ │ + * Add a feature to a cluster. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * cluster - {<OpenLayers.Feature.Vector>} A cluster. │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} A feature. │ │ │ │ │ + */ │ │ │ │ │ + addToCluster: function(cluster, feature) { │ │ │ │ │ + cluster.cluster.push(feature); │ │ │ │ │ + cluster.attributes.count += 1; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createCluster │ │ │ │ │ + * Given a feature, create a cluster. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Feature.Vector>} A cluster. │ │ │ │ │ + */ │ │ │ │ │ + createCluster: function(feature) { │ │ │ │ │ + var center = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + var cluster = new OpenLayers.Feature.Vector( │ │ │ │ │ + new OpenLayers.Geometry.Point(center.lon, center.lat), { │ │ │ │ │ + count: 1 │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + cluster.cluster = [feature]; │ │ │ │ │ + return cluster; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Cluster" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/de.js │ │ │ │ │ + OpenLayers/Strategy/Save.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Grille chompa │ │ │ │ │ - * - Nikiwaibel │ │ │ │ │ - * - Umherirrender │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Strategy.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["de"] │ │ │ │ │ - * Dictionary for Deutsch. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Strategy.Save │ │ │ │ │ + * A strategy that commits newly created or modified features. By default │ │ │ │ │ + * the strategy waits for a call to <save> before persisting changes. By │ │ │ │ │ + * configuring the strategy with the <auto> option, changes can be saved │ │ │ │ │ + * automatically. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Strategy> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["de"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - │ │ │ │ │ - 'unhandledRequest': "Unbehandelte Anfragerückmeldung ${statusText}", │ │ │ │ │ - │ │ │ │ │ - 'Permalink': "Permalink", │ │ │ │ │ - │ │ │ │ │ - 'Overlays': "Overlays", │ │ │ │ │ - │ │ │ │ │ - 'Base Layer': "Grundkarte", │ │ │ │ │ - │ │ │ │ │ - 'noFID': "Ein Feature, für das keine FID existiert, kann nicht aktualisiert werden.", │ │ │ │ │ +OpenLayers.Strategy.Save = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Ihr Browser unterstützt keine Vektordarstellung. Aktuell unterstützte Renderer:\n${renderers}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: events │ │ │ │ │ + * {<OpenLayers.Events>} An events object that handles all │ │ │ │ │ + * events on the strategy object. │ │ │ │ │ + * │ │ │ │ │ + * Register a listener for a particular event with the following syntax: │ │ │ │ │ + * (code) │ │ │ │ │ + * strategy.events.register(type, obj, listener); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Supported event types: │ │ │ │ │ + * start - Triggered before saving │ │ │ │ │ + * success - Triggered after a successful transaction │ │ │ │ │ + * fail - Triggered after a failed transaction │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - 'minZoomLevelError': "Die \x3ccode\x3eminZoomLevel\x3c/code\x3e-Eigenschaft ist nur für die Verwendung mit \x3ccode\x3eFixedZoomLevels\x3c/code\x3e-untergeordneten Layers vorgesehen. Das dieser \x3ctt\x3ewfs\x3c/tt\x3e-Layer die \x3ccode\x3eminZoomLevel\x3c/code\x3e-Eigenschaft überprüft ist ein Relikt der Vergangenheit. Wir können diese Überprüfung nicht entfernen, ohne das OL basierende Applikationen nicht mehr funktionieren. Daher markieren wir es als veraltet - die \x3ccode\x3eminZoomLevel\x3c/code\x3e-Überprüfung wird in Version 3.0 entfernt werden. Bitte verwenden Sie stattdessen die Min-/Max-Lösung, wie sie unter http://trac.openlayers.org/wiki/SettingZoomLevels beschrieben ist.", │ │ │ │ │ + /** │ │ │ │ │ + * Property: events │ │ │ │ │ + * {<OpenLayers.Events>} Events instance for triggering this protocol │ │ │ │ │ + * events. │ │ │ │ │ + */ │ │ │ │ │ + events: null, │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "WFS-Transaktion: Erfolgreich ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: auto │ │ │ │ │ + * {Boolean | Number} Auto-save. Default is false. If true, features will be │ │ │ │ │ + * saved immediately after being added to the layer and with each │ │ │ │ │ + * modification or deletion. If auto is a number, features will be │ │ │ │ │ + * saved on an interval provided by the value (in seconds). │ │ │ │ │ + */ │ │ │ │ │ + auto: false, │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "WFS-Transaktion: Fehlgeschlagen ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Property: timer │ │ │ │ │ + * {Number} The id of the timer. │ │ │ │ │ + */ │ │ │ │ │ + timer: null, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Der Google-Layer konnte nicht korrekt geladen werden.\x3cbr\x3e\x3cbr\x3eUm diese Meldung nicht mehr zu erhalten, wählen Sie einen anderen Hintergrundlayer aus dem LayerSwitcher in der rechten oberen Ecke.\x3cbr\x3e\x3cbr\x3eSehr wahrscheinlich tritt dieser Fehler auf, weil das Skript der Google-Maps-Bibliothek nicht eingebunden wurde oder keinen gültigen API-Schlüssel für Ihre URL enthält.\x3cbr\x3e\x3cbr\x3eEntwickler: Besuche \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3edas Wiki\x3c/a\x3e für Hilfe zum korrekten Einbinden des Google-Layers", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Strategy.Save │ │ │ │ │ + * Create a new Save strategy. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Strategy.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.events = new OpenLayers.Events(this); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "Der ${layerType}-Layer konnte nicht korrekt geladen werden.\x3cbr\x3e\x3cbr\x3eUm diese Meldung nicht mehr zu erhalten, wählen Sie einen anderen Hintergrundlayer aus dem LayerSwitcher in der rechten oberen Ecke.\x3cbr\x3e\x3cbr\x3eSehr wahrscheinlich tritt dieser Fehler auf, weil das Skript der \'${layerLib}\'-Bibliothek nicht eingebunden wurde.\x3cbr\x3e\x3cbr\x3eEntwickler: Besuche \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3edas Wiki\x3c/a\x3e für Hilfe zum korrekten Einbinden von Layern", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The strategy was successfully activated. │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + if (this.auto) { │ │ │ │ │ + if (typeof this.auto === "number") { │ │ │ │ │ + this.timer = window.setInterval( │ │ │ │ │ + OpenLayers.Function.bind(this.save, this), │ │ │ │ │ + this.auto * 1000 │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "featureadded": this.triggerSave, │ │ │ │ │ + "afterfeaturemodified": this.triggerSave, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Maßstab = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ + * tear-down. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + if (this.auto) { │ │ │ │ │ + if (typeof this.auto === "number") { │ │ │ │ │ + window.clearInterval(this.timer); │ │ │ │ │ + } else { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + "featureadded": this.triggerSave, │ │ │ │ │ + "afterfeaturemodified": this.triggerSave, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'W': "W", │ │ │ │ │ + /** │ │ │ │ │ + * Method: triggerSave │ │ │ │ │ + * Registered as a listener. Calls save if a feature has insert, update, │ │ │ │ │ + * or delete state. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * event - {Object} The event this function is listening for. │ │ │ │ │ + */ │ │ │ │ │ + triggerSave: function(event) { │ │ │ │ │ + var feature = event.feature; │ │ │ │ │ + if (feature.state === OpenLayers.State.INSERT || │ │ │ │ │ + feature.state === OpenLayers.State.UPDATE || │ │ │ │ │ + feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ + this.save([event.feature]); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'E': "O", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: save │ │ │ │ │ + * Tell the layer protocol to commit unsaved features. If the layer │ │ │ │ │ + * projection differs from the map projection, features will be │ │ │ │ │ + * transformed into the layer projection before being committed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * features - {Array} Features to be saved. If null, then default is all │ │ │ │ │ + * features in the layer. Features are assumed to be in the map │ │ │ │ │ + * projection. │ │ │ │ │ + */ │ │ │ │ │ + save: function(features) { │ │ │ │ │ + if (!features) { │ │ │ │ │ + features = this.layer.features; │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("start", { │ │ │ │ │ + features: features │ │ │ │ │ + }); │ │ │ │ │ + var remote = this.layer.projection; │ │ │ │ │ + var local = this.layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var len = features.length; │ │ │ │ │ + var clones = new Array(len); │ │ │ │ │ + var orig, clone; │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + orig = features[i]; │ │ │ │ │ + clone = orig.clone(); │ │ │ │ │ + clone.fid = orig.fid; │ │ │ │ │ + clone.state = orig.state; │ │ │ │ │ + if (orig.url) { │ │ │ │ │ + clone.url = orig.url; │ │ │ │ │ + } │ │ │ │ │ + clone._original = orig; │ │ │ │ │ + clone.geometry.transform(local, remote); │ │ │ │ │ + clones[i] = clone; │ │ │ │ │ + } │ │ │ │ │ + features = clones; │ │ │ │ │ + } │ │ │ │ │ + this.layer.protocol.commit(features, { │ │ │ │ │ + callback: this.onCommit, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'N': "N", │ │ │ │ │ + /** │ │ │ │ │ + * Method: onCommit │ │ │ │ │ + * Called after protocol commit. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * response - {<OpenLayers.Protocol.Response>} A response object. │ │ │ │ │ + */ │ │ │ │ │ + onCommit: function(response) { │ │ │ │ │ + var evt = { │ │ │ │ │ + "response": response │ │ │ │ │ + }; │ │ │ │ │ + if (response.success()) { │ │ │ │ │ + var features = response.reqFeatures; │ │ │ │ │ + // deal with inserts, updates, and deletes │ │ │ │ │ + var state, feature; │ │ │ │ │ + var destroys = []; │ │ │ │ │ + var insertIds = response.insertIds || []; │ │ │ │ │ + var j = 0; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + // if projection was different, we may be dealing with clones │ │ │ │ │ + feature = feature._original || feature; │ │ │ │ │ + state = feature.state; │ │ │ │ │ + if (state) { │ │ │ │ │ + if (state == OpenLayers.State.DELETE) { │ │ │ │ │ + destroys.push(feature); │ │ │ │ │ + } else if (state == OpenLayers.State.INSERT) { │ │ │ │ │ + feature.fid = insertIds[j]; │ │ │ │ │ + ++j; │ │ │ │ │ + } │ │ │ │ │ + feature.state = null; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'S': "S", │ │ │ │ │ + if (destroys.length > 0) { │ │ │ │ │ + this.layer.destroyFeatures(destroys); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Sie verwenden die „Reproject“-Option des Layers ${layerName}. Diese Option ist veraltet: Sie wurde entwickelt um die Anzeige von Daten auf kommerziellen Basiskarten zu unterstützen, aber diese Funktion sollte jetzt durch Unterstützung der „Spherical Mercator“ erreicht werden. Weitere Informationen sind unter http://trac.openlayers.org/wiki/SphericalMercator verfügbar.", │ │ │ │ │ + this.events.triggerEvent("success", evt); │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Die Methode ist veraltet und wird in 3.0 entfernt. Bitte verwende stattdessen ${newMethod}." │ │ │ │ │ + } else { │ │ │ │ │ + this.events.triggerEvent("fail", evt); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Save" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/en.js │ │ │ │ │ + OpenLayers/Strategy/Paging.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Strategy.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["en"] │ │ │ │ │ - * Dictionary for English. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Strategy.Paging │ │ │ │ │ + * Strategy for vector feature paging │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Strategy> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang.en = { │ │ │ │ │ - │ │ │ │ │ - 'unhandledRequest': "Unhandled request return ${statusText}", │ │ │ │ │ +OpenLayers.Strategy.Paging = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Permalink", │ │ │ │ │ + /** │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Array(<OpenLayers.Feature.Vector>)} Cached features. │ │ │ │ │ + */ │ │ │ │ │ + features: null, │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Overlays", │ │ │ │ │ + /** │ │ │ │ │ + * Property: length │ │ │ │ │ + * {Integer} Number of features per page. Default is 10. │ │ │ │ │ + */ │ │ │ │ │ + length: 10, │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Base Layer", │ │ │ │ │ + /** │ │ │ │ │ + * Property: num │ │ │ │ │ + * {Integer} The currently displayed page number. │ │ │ │ │ + */ │ │ │ │ │ + num: null, │ │ │ │ │ │ │ │ │ │ - 'noFID': "Can't update a feature for which there is no FID.", │ │ │ │ │ + /** │ │ │ │ │ + * Property: paging │ │ │ │ │ + * {Boolean} The strategy is currently changing pages. │ │ │ │ │ + */ │ │ │ │ │ + paging: false, │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Your browser does not support vector rendering. Currently supported renderers are:\n${renderers}", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Strategy.Paging │ │ │ │ │ + * Create a new paging strategy. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'minZoomLevelError': "The minZoomLevel property is only intended for use " + │ │ │ │ │ - "with the FixedZoomLevels-descendent layers. That this " + │ │ │ │ │ - "wfs layer checks for minZoomLevel is a relic of the" + │ │ │ │ │ - "past. We cannot, however, remove it without possibly " + │ │ │ │ │ - "breaking OL based applications that may depend on it." + │ │ │ │ │ - " Therefore we are deprecating it -- the minZoomLevel " + │ │ │ │ │ - "check below will be removed at 3.0. Please instead " + │ │ │ │ │ - "use min/max resolution setting as described here: " + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: activate │ │ │ │ │ + * Activate the strategy. Register any listeners, do appropriate setup. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The strategy was successfully activated. │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "beforefeaturesadded": this.cacheFeatures, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "WFS Transaction: SUCCESS ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: deactivate │ │ │ │ │ + * Deactivate the strategy. Unregister any listeners, do appropriate │ │ │ │ │ + * tear-down. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.clearCache(); │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + "beforefeaturesadded": this.cacheFeatures, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "WFS Transaction: FAILED ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: cacheFeatures │ │ │ │ │ + * Cache features before they are added to the layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * event - {Object} The event that this was listening for. This will come │ │ │ │ │ + * with a batch of features to be paged. │ │ │ │ │ + */ │ │ │ │ │ + cacheFeatures: function(event) { │ │ │ │ │ + if (!this.paging) { │ │ │ │ │ + this.clearCache(); │ │ │ │ │ + this.features = event.features; │ │ │ │ │ + this.pageNext(event); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "The Google Layer was unable to load correctly.<br><br>" + │ │ │ │ │ - "To get rid of this message, select a new BaseLayer " + │ │ │ │ │ - "in the layer switcher in the upper-right corner.<br><br>" + │ │ │ │ │ - "Most likely, this is because the Google Maps library " + │ │ │ │ │ - "script was either not included, or does not contain the " + │ │ │ │ │ - "correct API key for your site.<br><br>" + │ │ │ │ │ - "Developers: For help getting this working correctly, " + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ - "target='_blank'>click here</a>", │ │ │ │ │ + /** │ │ │ │ │ + * Method: clearCache │ │ │ │ │ + * Clear out the cached features. This destroys features, assuming │ │ │ │ │ + * nothing else has a reference. │ │ │ │ │ + */ │ │ │ │ │ + clearCache: function() { │ │ │ │ │ + if (this.features) { │ │ │ │ │ + for (var i = 0; i < this.features.length; ++i) { │ │ │ │ │ + this.features[i].destroy(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.features = null; │ │ │ │ │ + this.num = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "The ${layerType} Layer was unable to load correctly.<br><br>" + │ │ │ │ │ - "To get rid of this message, select a new BaseLayer " + │ │ │ │ │ - "in the layer switcher in the upper-right corner.<br><br>" + │ │ │ │ │ - "Most likely, this is because the ${layerLib} library " + │ │ │ │ │ - "script was not correctly included.<br><br>" + │ │ │ │ │ - "Developers: For help getting this working correctly, " + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ - "target='_blank'>click here</a>", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: pageCount │ │ │ │ │ + * Get the total count of pages given the current cache of features. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} The page count. │ │ │ │ │ + */ │ │ │ │ │ + pageCount: function() { │ │ │ │ │ + var numFeatures = this.features ? this.features.length : 0; │ │ │ │ │ + return Math.ceil(numFeatures / this.length); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Scale = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: pageNum │ │ │ │ │ + * Get the zero based page number. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} The current page number being displayed. │ │ │ │ │ + */ │ │ │ │ │ + pageNum: function() { │ │ │ │ │ + return this.num; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - //labels for the graticule control │ │ │ │ │ - 'W': 'W', │ │ │ │ │ - 'E': 'E', │ │ │ │ │ - 'N': 'N', │ │ │ │ │ - 'S': 'S', │ │ │ │ │ - 'Graticule': 'Graticule', │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: pageLength │ │ │ │ │ + * Gets or sets page length. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newLength - {Integer} Optional length to be set. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} The length of a page (number of features per page). │ │ │ │ │ + */ │ │ │ │ │ + pageLength: function(newLength) { │ │ │ │ │ + if (newLength && newLength > 0) { │ │ │ │ │ + this.length = newLength; │ │ │ │ │ + } │ │ │ │ │ + return this.length; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'reprojectDeprecated': "You are using the 'reproject' option " + │ │ │ │ │ - "on the ${layerName} layer. This option is deprecated: " + │ │ │ │ │ - "its use was designed to support displaying data over commercial " + │ │ │ │ │ - "basemaps, but that functionality should now be achieved by using " + │ │ │ │ │ - "Spherical Mercator support. More information is available from " + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: pageNext │ │ │ │ │ + * Display the next page of features. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} A new page was displayed. │ │ │ │ │ + */ │ │ │ │ │ + pageNext: function(event) { │ │ │ │ │ + var changed = false; │ │ │ │ │ + if (this.features) { │ │ │ │ │ + if (this.num === null) { │ │ │ │ │ + this.num = -1; │ │ │ │ │ + } │ │ │ │ │ + var start = (this.num + 1) * this.length; │ │ │ │ │ + changed = this.page(start, event); │ │ │ │ │ + } │ │ │ │ │ + return changed; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'methodDeprecated': "This method has been deprecated and will be removed in 3.0. " + │ │ │ │ │ - "Please use ${newMethod} instead.", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: pagePrevious │ │ │ │ │ + * Display the previous page of features. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} A new page was displayed. │ │ │ │ │ + */ │ │ │ │ │ + pagePrevious: function() { │ │ │ │ │ + var changed = false; │ │ │ │ │ + if (this.features) { │ │ │ │ │ + if (this.num === null) { │ │ │ │ │ + this.num = this.pageCount(); │ │ │ │ │ + } │ │ │ │ │ + var start = (this.num - 1) * this.length; │ │ │ │ │ + changed = this.page(start); │ │ │ │ │ + } │ │ │ │ │ + return changed; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // **** end **** │ │ │ │ │ - 'end': '' │ │ │ │ │ + /** │ │ │ │ │ + * Method: page │ │ │ │ │ + * Display the page starting at the given index from the cache. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} A new page was displayed. │ │ │ │ │ + */ │ │ │ │ │ + page: function(start, event) { │ │ │ │ │ + var changed = false; │ │ │ │ │ + if (this.features) { │ │ │ │ │ + if (start >= 0 && start < this.features.length) { │ │ │ │ │ + var num = Math.floor(start / this.length); │ │ │ │ │ + if (num != this.num) { │ │ │ │ │ + this.paging = true; │ │ │ │ │ + var features = this.features.slice(start, start + this.length); │ │ │ │ │ + this.layer.removeFeatures(this.layer.features); │ │ │ │ │ + this.num = num; │ │ │ │ │ + // modify the event if any │ │ │ │ │ + if (event && event.features) { │ │ │ │ │ + // this.was called by an event listener │ │ │ │ │ + event.features = features; │ │ │ │ │ + } else { │ │ │ │ │ + // this was called directly on the strategy │ │ │ │ │ + this.layer.addFeatures(features); │ │ │ │ │ + } │ │ │ │ │ + this.paging = false; │ │ │ │ │ + changed = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return changed; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -}; │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Paging" │ │ │ │ │ +}); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/en-CA.js │ │ │ │ │ + OpenLayers/Strategy/Fixed.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang/en.js │ │ │ │ │ + * @requires OpenLayers/Strategy.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["en-CA"] │ │ │ │ │ - * Dictionary for English-CA. This dictionary inherits from the standard │ │ │ │ │ - * English dictionary. Override only those entries with language specific │ │ │ │ │ - * to the CA region. │ │ │ │ │ - * │ │ │ │ │ - * Keys for entries are used in calls to <OpenLayers.Lang.translate>. Entry │ │ │ │ │ - * bodies are normal strings or strings formatted for use with │ │ │ │ │ - * <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Strategy.Fixed │ │ │ │ │ + * A simple strategy that requests features once and never requests new data. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Strategy> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang['en-CA'] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ +OpenLayers.Strategy.Fixed = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ │ │ │ │ │ - // add any entries specific for this region here │ │ │ │ │ - // e.g. │ │ │ │ │ - // "someKey": "Some regionally specific value" │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: preload │ │ │ │ │ + * {Boolean} Load data before layer made visible. Enabling this may result │ │ │ │ │ + * in considerable overhead if your application loads many data layers │ │ │ │ │ + * that are not visible by default. Default is false. │ │ │ │ │ + */ │ │ │ │ │ + preload: false, │ │ │ │ │ │ │ │ │ │ -}, OpenLayers.Lang["en"]); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/el.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Strategy.Fixed │ │ │ │ │ + * Create a new Fixed strategy. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ + * instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Omnipaedista │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: activate │ │ │ │ │ + * Activate the strategy: load data or add listener to load when visible │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} True if the strategy was successfully activated or false if │ │ │ │ │ + * the strategy was already active. │ │ │ │ │ + */ │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "refresh": this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + if (this.layer.visibility == true || this.preload) { │ │ │ │ │ + this.load(); │ │ │ │ │ + } else { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + "visibilitychanged": this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return activated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: deactivate │ │ │ │ │ + * Deactivate the strategy. Undo what is done in <activate>. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} The strategy was successfully deactivated. │ │ │ │ │ + */ │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + "refresh": this.load, │ │ │ │ │ + "visibilitychanged": this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + return deactivated; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["el"] │ │ │ │ │ - * Dictionary for Ελληνικά. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Lang["el"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + /** │ │ │ │ │ + * Method: load │ │ │ │ │ + * Tells protocol to load data and unhooks the visibilitychanged event │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} options to pass to protocol read. │ │ │ │ │ + */ │ │ │ │ │ + load: function(options) { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + layer.events.triggerEvent("loadstart", { │ │ │ │ │ + filter: layer.filter │ │ │ │ │ + }); │ │ │ │ │ + layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: this.merge, │ │ │ │ │ + filter: layer.filter, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options)); │ │ │ │ │ + layer.events.un({ │ │ │ │ │ + "visibilitychanged": this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Κλίμακα ~ 1 : ${scaleDenom}" │ │ │ │ │ + /** │ │ │ │ │ + * Method: merge │ │ │ │ │ + * Add all features to the layer. │ │ │ │ │ + * If the layer projection differs from the map projection, features │ │ │ │ │ + * will be transformed from the layer projection to the map projection. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * resp - {<OpenLayers.Protocol.Response>} The response object passed │ │ │ │ │ + * by the protocol. │ │ │ │ │ + */ │ │ │ │ │ + merge: function(resp) { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + layer.destroyFeatures(); │ │ │ │ │ + var features = resp.features; │ │ │ │ │ + if (features && features.length > 0) { │ │ │ │ │ + var remote = layer.projection; │ │ │ │ │ + var local = layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var geom; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + geom = features[i].geometry; │ │ │ │ │ + if (geom) { │ │ │ │ │ + geom.transform(remote, local); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + layer.addFeatures(features); │ │ │ │ │ + } │ │ │ │ │ + layer.events.triggerEvent("loadend", { │ │ │ │ │ + response: resp │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Fixed" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/oc.js │ │ │ │ │ + OpenLayers/Marker/Box.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Cedric31 │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Marker.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["oc"] │ │ │ │ │ - * Dictionary for Occitan. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Marker.Box │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Marker> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["oc"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - │ │ │ │ │ - 'unhandledRequest': "Requèsta pas gerida, retorna ${statusText}", │ │ │ │ │ - │ │ │ │ │ - 'Permalink': "Permaligam", │ │ │ │ │ - │ │ │ │ │ - 'Overlays': "Calques", │ │ │ │ │ - │ │ │ │ │ - 'Base Layer': "Calc de basa", │ │ │ │ │ - │ │ │ │ │ - 'noFID': "Impossible de metre a jorn un objècte sens identificant (fid).", │ │ │ │ │ - │ │ │ │ │ - 'browserNotSupported': "Vòstre navegidor supòrta pas lo rendut vectorial. Los renderers actualament suportats son : \n${renderers}", │ │ │ │ │ - │ │ │ │ │ - 'minZoomLevelError': "La proprietat minZoomLevel deu èsser utilizada solament per de jaces FixedZoomLevels-descendent. Lo fach qu\'aqueste jaç WFS verifique la preséncia de minZoomLevel es una relica del passat. Çaquelà, la podèm suprimir sens copar d\'aplicacions que ne poirián dependre. Es per aquò que la depreciam -- la verificacion del minZoomLevel serà suprimida en version 3.0. A la plaça, mercés d\'utilizar los paramètres de resolucions min/max tal coma descrich sus : http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - │ │ │ │ │ - 'commitSuccess': "Transaccion WFS : SUCCES ${response}", │ │ │ │ │ +OpenLayers.Marker.Box = OpenLayers.Class(OpenLayers.Marker, { │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "Transaccion WFS : FRACAS ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Property: bounds │ │ │ │ │ + * {<OpenLayers.Bounds>} │ │ │ │ │ + */ │ │ │ │ │ + bounds: null, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Lo jaç Google es pas estat en mesura de se cargar corrèctament.\x3cbr\x3e\x3cbr\x3ePer suprimir aqueste messatge, causissètz una BaseLayer novèla dins lo selector de jaç en naut a drecha.\x3cbr\x3e\x3cbr\x3eAquò es possiblament causat par la non-inclusion de la librariá Google Maps, o alara perque que la clau de l\'API correspond pas a vòstre site.\x3cbr\x3e\x3cbr\x3eDesvolopaires : per saber cossí corregir aquò, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eclicatz aicí\x3c/a\x3e", │ │ │ │ │ + /** │ │ │ │ │ + * Property: div │ │ │ │ │ + * {DOMElement} │ │ │ │ │ + */ │ │ │ │ │ + div: null, │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "Lo jaç ${layerType} es pas en mesura de se cargar corrèctament.\x3cbr\x3e\x3cbr\x3ePer suprimir aqueste messatge, causissètz una BaseLayer novèla dins lo selector de jaç en naut a drecha.\x3cbr\x3e\x3cbr\x3eAquò es possiblament causat per la non-inclusion de la librariá ${layerLib}.\x3cbr\x3e\x3cbr\x3eDesvolopaires : per saber cossí corregir aquí, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eclicatz aicí\x3c/a\x3e", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Marker.Box │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * borderColor - {String} │ │ │ │ │ + * borderWidth - {int} │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(bounds, borderColor, borderWidth) { │ │ │ │ │ + this.bounds = bounds; │ │ │ │ │ + this.div = OpenLayers.Util.createDiv(); │ │ │ │ │ + this.div.style.overflow = 'hidden'; │ │ │ │ │ + this.events = new OpenLayers.Events(this, this.div); │ │ │ │ │ + this.setBorder(borderColor, borderWidth); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Escala ~ 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ │ │ │ │ │ - 'W': "O", │ │ │ │ │ + this.bounds = null; │ │ │ │ │ + this.div = null; │ │ │ │ │ │ │ │ │ │ - 'E': "È", │ │ │ │ │ + OpenLayers.Marker.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'N': "N", │ │ │ │ │ + /** │ │ │ │ │ + * Method: setBorder │ │ │ │ │ + * Allow the user to change the box's color and border width │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * color - {String} Default is "red" │ │ │ │ │ + * width - {int} Default is 2 │ │ │ │ │ + */ │ │ │ │ │ + setBorder: function(color, width) { │ │ │ │ │ + if (!color) { │ │ │ │ │ + color = "red"; │ │ │ │ │ + } │ │ │ │ │ + if (!width) { │ │ │ │ │ + width = 2; │ │ │ │ │ + } │ │ │ │ │ + this.div.style.border = width + "px solid " + color; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'S': "S", │ │ │ │ │ + /** │ │ │ │ │ + * Method: draw │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ + * sz - {<OpenLayers.Size>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {DOMElement} A new DOM Image with this marker's icon set at the │ │ │ │ │ + * location passed-in │ │ │ │ │ + */ │ │ │ │ │ + draw: function(px, sz) { │ │ │ │ │ + OpenLayers.Util.modifyDOMElement(this.div, null, px, sz); │ │ │ │ │ + return this.div; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Utilizatz l\'opcion \'reproject\' sul jaç ${layerName}. Aquesta opcion es despreciada : Son usatge permetiá d\'afichar de donadas al dessús de jaces raster comercials. Aquesta foncionalitat ara es suportada en utilizant lo supòrt de la projeccion Mercator Esferica. Mai d\'informacion es disponibla sus http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + /** │ │ │ │ │ + * Method: onScreen │ │ │ │ │ + * │ │ │ │ │ + * Rreturn: │ │ │ │ │ + * {Boolean} Whether or not the marker is currently visible on screen. │ │ │ │ │ + */ │ │ │ │ │ + onScreen: function() { │ │ │ │ │ + var onScreen = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + var screenBounds = this.map.getExtent(); │ │ │ │ │ + onScreen = screenBounds.containsBounds(this.bounds, true, true); │ │ │ │ │ + } │ │ │ │ │ + return onScreen; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Aqueste metòde es despreciada, e serà suprimida a la version 3.0. Mercés d\'utilizar ${newMethod} a la plaça." │ │ │ │ │ + /** │ │ │ │ │ + * Method: display │ │ │ │ │ + * Hide or show the icon │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * display - {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + display: function(display) { │ │ │ │ │ + this.div.style.display = (display) ? "" : "none"; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Marker.Box" │ │ │ │ │ }); │ │ │ │ │ + │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/lt.js │ │ │ │ │ + OpenLayers/Layer/XYZ.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["lt"] │ │ │ │ │ - * Dictionary for Lithuanian. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.XYZ │ │ │ │ │ + * The XYZ class is designed to make it easier for people who have tiles │ │ │ │ │ + * arranged by a standard XYZ grid. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang['lt'] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ +OpenLayers.Layer.XYZ = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "Neapdorota užklausa gražino ${statusText}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * Default is true, as this is designed to be a base tile source. │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Pastovi nuoroda", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: sphericalMercator │ │ │ │ │ + * Whether the tile extents should be set to the defaults for │ │ │ │ │ + * spherical mercator. Useful for things like OpenStreetMap. │ │ │ │ │ + * Default is false, except for the OSM subclass. │ │ │ │ │ + */ │ │ │ │ │ + sphericalMercator: false, │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Papildomi sluoksniai", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomOffset │ │ │ │ │ + * {Number} If your cache has more zoom levels than you want to provide │ │ │ │ │ + * access to with this layer, supply a zoomOffset. This zoom offset │ │ │ │ │ + * is added to the current map zoom level to determine the level │ │ │ │ │ + * for a requested tile. For example, if you supply a zoomOffset │ │ │ │ │ + * of 3, when the map is at the zoom 0, tiles will be requested from │ │ │ │ │ + * level 3 of your cache. Default is 0 (assumes cache level and map │ │ │ │ │ + * zoom are equivalent). Using <zoomOffset> is an alternative to │ │ │ │ │ + * setting <serverResolutions> if you only want to expose a subset │ │ │ │ │ + * of the server resolutions. │ │ │ │ │ + */ │ │ │ │ │ + zoomOffset: 0, │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Pagrindinis sluoksnis", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: serverResolutions │ │ │ │ │ + * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ + * property if the map resolutions differ from the server. This │ │ │ │ │ + * property serves two purposes. (a) <serverResolutions> can include │ │ │ │ │ + * resolutions that the server supports and that you don't want to │ │ │ │ │ + * provide with this layer; you can also look at <zoomOffset>, which is │ │ │ │ │ + * an alternative to <serverResolutions> for that specific purpose. │ │ │ │ │ + * (b) The map can work with resolutions that aren't supported by │ │ │ │ │ + * the server, i.e. that aren't in <serverResolutions>. When the │ │ │ │ │ + * map is displayed in such a resolution data for the closest │ │ │ │ │ + * server-supported resolution is loaded and the layer div is │ │ │ │ │ + * stretched as necessary. │ │ │ │ │ + */ │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ │ │ │ │ │ - 'noFID': "Negaliu atnaujinti objekto, kuris neturi FID.", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.XYZ │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} │ │ │ │ │ + * url - {String} │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + if (options && options.sphericalMercator || this.sphericalMercator) { │ │ │ │ │ + options = OpenLayers.Util.extend({ │ │ │ │ │ + projection: "EPSG:900913", │ │ │ │ │ + numZoomLevels: 19 │ │ │ │ │ + }, options); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [ │ │ │ │ │ + name || this.name, url || this.url, {}, │ │ │ │ │ + options │ │ │ │ │ + ]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Jūsų naršyklė nemoka parodyti vektorių. Šiuo metu galima naudotis tokiais rodymo varikliais:\n{renderers}", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} Is this ever used? │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.XYZ>} An exact clone of this OpenLayers.Layer.XYZ │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "WFS Tranzakcija: PAVYKO ${response}", │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.XYZ(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "WFS Tranzakcija: ŽLUGO ${response}", │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Mastelis = 1 : ${scaleDenom}", │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - //labels for the graticule control │ │ │ │ │ - 'W': 'V', │ │ │ │ │ - 'E': 'R', │ │ │ │ │ - 'N': 'Š', │ │ │ │ │ - 'S': 'P', │ │ │ │ │ - 'Graticule': 'Tinklelis', │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as │ │ │ │ │ + * parameters │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var xyz = this.getXYZ(bounds); │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + var s = '' + xyz.x + xyz.y + xyz.z; │ │ │ │ │ + url = this.selectUrl(s, url); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'methodDeprecated': "Šis metodas yra pasenęs ir 3.0 versijoje bus pašalintas. " + │ │ │ │ │ - "Prašome naudoti ${newMethod}.", │ │ │ │ │ + return OpenLayers.String.format(url, xyz); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // **** end **** │ │ │ │ │ - 'end': '' │ │ │ │ │ + /** │ │ │ │ │ + * Method: getXYZ │ │ │ │ │ + * Calculates x, y and z for the given bounds. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} - an object with x, y and z properties. │ │ │ │ │ + */ │ │ │ │ │ + getXYZ: function(bounds) { │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.maxExtent.left) / │ │ │ │ │ + (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((this.maxExtent.top - bounds.top) / │ │ │ │ │ + (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getServerZoom(); │ │ │ │ │ + │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + var limit = Math.pow(2, z); │ │ │ │ │ + x = ((x % limit) + limit) % limit; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + return { │ │ │ │ │ + 'x': x, │ │ │ │ │ + 'y': y, │ │ │ │ │ + 'z': z │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /* APIMethod: setMap │ │ │ │ │ + * When the layer is added to a map, then we can fetch our origin │ │ │ │ │ + * (if we don't have one.) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (!this.tileOrigin) { │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, │ │ │ │ │ + this.maxExtent.bottom); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.XYZ" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/sv-SE.js │ │ │ │ │ + OpenLayers/Layer/Text.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Sannab │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Layer/Markers.js │ │ │ │ │ + * @requires OpenLayers/Format/Text.js │ │ │ │ │ + * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["sv"] │ │ │ │ │ - * Dictionary for Svenska. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Layer.Text │ │ │ │ │ + * This layer creates markers given data in a text file. The <location> │ │ │ │ │ + * property of the layer (specified as a property of the options argument │ │ │ │ │ + * in the <OpenLayers.Layer.Text> constructor) points to a tab delimited │ │ │ │ │ + * file with data used to create markers. │ │ │ │ │ + * │ │ │ │ │ + * The first row of the data file should be a header line with the column names │ │ │ │ │ + * of the data. Each column should be delimited by a tab space. The │ │ │ │ │ + * possible columns are: │ │ │ │ │ + * - *point* lat,lon of the point where a marker is to be placed │ │ │ │ │ + * - *lat* Latitude of the point where a marker is to be placed │ │ │ │ │ + * - *lon* Longitude of the point where a marker is to be placed │ │ │ │ │ + * - *icon* or *image* URL of marker icon to use. │ │ │ │ │ + * - *iconSize* Size of Icon to use. │ │ │ │ │ + * - *iconOffset* Where the top-left corner of the icon is to be placed │ │ │ │ │ + * relative to the latitude and longitude of the point. │ │ │ │ │ + * - *title* The text of the 'title' is placed inside an 'h2' marker │ │ │ │ │ + * inside a popup, which opens when the marker is clicked. │ │ │ │ │ + * - *description* The text of the 'description' is placed below the h2 │ │ │ │ │ + * in the popup. this can be plain text or HTML. │ │ │ │ │ + * │ │ │ │ │ + * Example text file: │ │ │ │ │ + * (code) │ │ │ │ │ + * lat lon title description iconSize iconOffset icon │ │ │ │ │ + * 10 20 title description 21,25 -10,-25 http://www.openlayers.org/dev/img/marker.png │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Markers> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["sv"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ +OpenLayers.Layer.Text = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "Ej hanterad fråga retur ${statusText}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: location │ │ │ │ │ + * {String} URL of text file. Must be specified in the "options" argument │ │ │ │ │ + * of the constructor. Can not be changed once passed in. │ │ │ │ │ + */ │ │ │ │ │ + location: null, │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Permalänk", │ │ │ │ │ + /** │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Array(<OpenLayers.Feature>)} │ │ │ │ │ + */ │ │ │ │ │ + features: null, │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Kartlager", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: formatOptions │ │ │ │ │ + * {Object} Hash of options which should be passed to the format when it is │ │ │ │ │ + * created. Must be passed in the constructor. │ │ │ │ │ + */ │ │ │ │ │ + formatOptions: null, │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Bakgrundskarta", │ │ │ │ │ + /** │ │ │ │ │ + * Property: selectedFeature │ │ │ │ │ + * {<OpenLayers.Feature>} │ │ │ │ │ + */ │ │ │ │ │ + selectedFeature: null, │ │ │ │ │ │ │ │ │ │ - 'noFID': "Kan ej uppdatera feature (objekt) för vilket FID saknas.", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.Text │ │ │ │ │ + * Create a text layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} │ │ │ │ │ + * options - {Object} Object with properties to be set on the layer. │ │ │ │ │ + * Must include <location> property. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.features = []; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Din webbläsare stöder inte vektorvisning. För närvarande stöds följande visning:\n${renderers}", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + // Warning: Layer.Markers.destroy() must be called prior to calling │ │ │ │ │ + // clearFeatures() here, otherwise we leak memory. Indeed, if │ │ │ │ │ + // Layer.Markers.destroy() is called after clearFeatures(), it won't be │ │ │ │ │ + // able to remove the marker image elements from the layer's div since │ │ │ │ │ + // the markers will have been destroyed by clearFeatures(). │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.destroy.apply(this, arguments); │ │ │ │ │ + this.clearFeatures(); │ │ │ │ │ + this.features = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'minZoomLevelError': "Egenskapen minZoomLevel är endast avsedd att användas med lager med FixedZoomLevels. Att detta WFS-lager kontrollerar minZoomLevel är en relik från äldre versioner. Vi kan dock inte ta bort det utan att riskera att OL-baserade tillämpningar som använder detta slutar fungera. Därför är det satt som deprecated, minZoomLevel kommer att tas bort i version 3.0. Använd i stället inställning av min/max resolution som beskrivs här: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + /** │ │ │ │ │ + * Method: loadText │ │ │ │ │ + * Start the load of the Text data. Don't do this when we first add the layer, │ │ │ │ │ + * since we may not be visible at any point, and it would therefore be a waste. │ │ │ │ │ + */ │ │ │ │ │ + loadText: function() { │ │ │ │ │ + if (!this.loaded) { │ │ │ │ │ + if (this.location != null) { │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "WFS-transaktion: LYCKADES ${response}", │ │ │ │ │ + var onFail = function(e) { │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "WFS-transaktion: MISSLYCKADES ${response}", │ │ │ │ │ + this.events.triggerEvent("loadstart"); │ │ │ │ │ + OpenLayers.Request.GET({ │ │ │ │ │ + url: this.location, │ │ │ │ │ + success: this.parseData, │ │ │ │ │ + failure: onFail, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.loaded = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Google-lagret kunde inte laddas korrekt.\x3cbr\x3e\x3cbr\x3eFör att slippa detta meddelande, välj en annan bakgrundskarta i lagerväljaren i övre högra hörnet.\x3cbr\x3e\x3cbr\x3eSannolikt beror felet på att Google Maps-biblioteket inte är inkluderat på webbsidan eller på att sidan inte anger korrekt API-nyckel för webbplatsen.\x3cbr\x3e\x3cbr\x3eUtvecklare: hjälp för att åtgärda detta, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eklicka här\x3c/a\x3e.", │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * If layer is visible and Text has not been loaded, load Text. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {Object} │ │ │ │ │ + * zoomChanged - {Object} │ │ │ │ │ + * minor - {Object} │ │ │ │ │ + */ │ │ │ │ │ + moveTo: function(bounds, zoomChanged, minor) { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + if (this.visibility && !this.loaded) { │ │ │ │ │ + this.loadText(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "${layerType}-lagret kunde inte laddas korrekt.\x3cbr\x3e\x3cbr\x3eFör att slippa detta meddelande, välj en annan bakgrundskarta i lagerväljaren i övre högra hörnet.\x3cbr\x3e\x3cbr\x3eSannolikt beror felet på att ${layerLib}-biblioteket inte är inkluderat på webbsidan.\x3cbr\x3e\x3cbr\x3eUtvecklare: hjälp för att åtgärda detta, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eklicka här\x3c/a\x3e.", │ │ │ │ │ + /** │ │ │ │ │ + * Method: parseData │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * ajaxRequest - {<OpenLayers.Request.XMLHttpRequest>} │ │ │ │ │ + */ │ │ │ │ │ + parseData: function(ajaxRequest) { │ │ │ │ │ + var text = ajaxRequest.responseText; │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "\x3cstrong\x3eSkala\x3c/strong\x3e 1 : ${scaleDenom}", │ │ │ │ │ + var options = {}; │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Du använder inställningen \'reproject\' på lagret ${layerName}. Denna inställning markerad som deprecated: den var avsedd att användas för att stödja visning av kartdata på kommersiella bakgrundskartor, men nu bör man i stället använda Spherical Mercator-stöd för den funktionaliteten. Mer information finns på http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + OpenLayers.Util.extend(options, this.formatOptions); │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Denna metod är markerad som deprecated och kommer att tas bort i 3.0. Använd ${newMethod} i stället." │ │ │ │ │ + if (this.map && !this.projection.equals(this.map.getProjectionObject())) { │ │ │ │ │ + options.externalProjection = this.projection; │ │ │ │ │ + options.internalProjection = this.map.getProjectionObject(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var parser = new OpenLayers.Format.Text(options); │ │ │ │ │ + var features = parser.read(text); │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + var data = {}; │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + var location; │ │ │ │ │ + var iconSize, iconOffset; │ │ │ │ │ + │ │ │ │ │ + location = new OpenLayers.LonLat(feature.geometry.x, │ │ │ │ │ + feature.geometry.y); │ │ │ │ │ + │ │ │ │ │ + if (feature.style.graphicWidth && │ │ │ │ │ + feature.style.graphicHeight) { │ │ │ │ │ + iconSize = new OpenLayers.Size( │ │ │ │ │ + feature.style.graphicWidth, │ │ │ │ │ + feature.style.graphicHeight); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + // FIXME: At the moment, we only use this if we have an │ │ │ │ │ + // externalGraphic, because icon has no setOffset API Method. │ │ │ │ │ + /** │ │ │ │ │ + * FIXME FIRST!! │ │ │ │ │ + * The Text format does all sorts of parseFloating │ │ │ │ │ + * The result of a parseFloat for a bogus string is NaN. That │ │ │ │ │ + * means the three possible values here are undefined, NaN, or a │ │ │ │ │ + * number. The previous check was an identity check for null. This │ │ │ │ │ + * means it was failing for all undefined or NaN. A slightly better │ │ │ │ │ + * check is for undefined. An even better check is to see if the │ │ │ │ │ + * value is a number (see #1441). │ │ │ │ │ + */ │ │ │ │ │ + if (feature.style.graphicXOffset !== undefined && │ │ │ │ │ + feature.style.graphicYOffset !== undefined) { │ │ │ │ │ + iconOffset = new OpenLayers.Pixel( │ │ │ │ │ + feature.style.graphicXOffset, │ │ │ │ │ + feature.style.graphicYOffset); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (feature.style.externalGraphic != null) { │ │ │ │ │ + data.icon = new OpenLayers.Icon(feature.style.externalGraphic, │ │ │ │ │ + iconSize, │ │ │ │ │ + iconOffset); │ │ │ │ │ + } else { │ │ │ │ │ + data.icon = OpenLayers.Marker.defaultIcon(); │ │ │ │ │ + │ │ │ │ │ + //allows for the case where the image url is not │ │ │ │ │ + // specified but the size is. use a default icon │ │ │ │ │ + // but change the size │ │ │ │ │ + if (iconSize != null) { │ │ │ │ │ + data.icon.setSize(iconSize); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if ((feature.attributes.title != null) && │ │ │ │ │ + (feature.attributes.description != null)) { │ │ │ │ │ + data['popupContentHTML'] = │ │ │ │ │ + '<h2>' + feature.attributes.title + '</h2>' + │ │ │ │ │ + '<p>' + feature.attributes.description + '</p>'; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + data['overflow'] = feature.attributes.overflow || "auto"; │ │ │ │ │ + │ │ │ │ │ + var markerFeature = new OpenLayers.Feature(this, location, data); │ │ │ │ │ + this.features.push(markerFeature); │ │ │ │ │ + var marker = markerFeature.createMarker(); │ │ │ │ │ + if ((feature.attributes.title != null) && │ │ │ │ │ + (feature.attributes.description != null)) { │ │ │ │ │ + marker.events.register('click', markerFeature, this.markerClick); │ │ │ │ │ + } │ │ │ │ │ + this.addMarker(marker); │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: markerClick │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * evt - {Event} │ │ │ │ │ + * │ │ │ │ │ + * Context: │ │ │ │ │ + * - {<OpenLayers.Feature>} │ │ │ │ │ + */ │ │ │ │ │ + markerClick: function(evt) { │ │ │ │ │ + var sameMarkerClicked = (this == this.layer.selectedFeature); │ │ │ │ │ + this.layer.selectedFeature = (!sameMarkerClicked) ? this : null; │ │ │ │ │ + for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ + this.layer.map.removePopup(this.layer.map.popups[i]); │ │ │ │ │ + } │ │ │ │ │ + if (!sameMarkerClicked) { │ │ │ │ │ + this.layer.map.addPopup(this.createPopup()); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clearFeatures │ │ │ │ │ + */ │ │ │ │ │ + clearFeatures: function() { │ │ │ │ │ + if (this.features != null) { │ │ │ │ │ + while (this.features.length > 0) { │ │ │ │ │ + var feature = this.features[0]; │ │ │ │ │ + OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ + feature.destroy(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Text" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/es.js │ │ │ │ │ + OpenLayers/Layer/ArcGISCache.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["es"] │ │ │ │ │ - * Dictionary for Spanish, UTF8 encoding. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang.es = { │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "Respuesta a petición no gestionada ${statusText}", │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.ArcGISCache │ │ │ │ │ + * Layer for accessing cached map tiles from an ArcGIS Server style mapcache. │ │ │ │ │ + * Tile must already be cached for this layer to access it. This does not require │ │ │ │ │ + * ArcGIS Server itself. │ │ │ │ │ + * │ │ │ │ │ + * A few attempts have been made at this kind of layer before. See │ │ │ │ │ + * http://trac.osgeo.org/openlayers/ticket/1967 │ │ │ │ │ + * and │ │ │ │ │ + * http://trac.osgeo.org/openlayers/browser/sandbox/tschaub/arcgiscache/lib/OpenLayers/Layer/ArcGISCache.js │ │ │ │ │ + * │ │ │ │ │ + * Typically the problem encountered is that the tiles seem to "jump around". │ │ │ │ │ + * This is due to the fact that the actual max extent for the tiles on AGS layers │ │ │ │ │ + * changes at each zoom level due to the way these caches are constructed. │ │ │ │ │ + * We have attempted to use the resolutions, tile size, and tile origin │ │ │ │ │ + * from the cache meta data to make the appropriate changes to the max extent │ │ │ │ │ + * of the tile to compensate for this behavior. This must be done as zoom levels change │ │ │ │ │ + * and before tiles are requested, which is why methods from base classes are overridden. │ │ │ │ │ + * │ │ │ │ │ + * For reference, you can access mapcache meta data in two ways. For accessing a │ │ │ │ │ + * mapcache through ArcGIS Server, you can simply go to the landing page for the │ │ │ │ │ + * layer. (ie. http://services.arcgisonline.com/ArcGIS/rest/services/World_Street_Map/MapServer) │ │ │ │ │ + * For accessing it directly through HTTP, there should always be a conf.xml file │ │ │ │ │ + * in the root directory. │ │ │ │ │ + * (ie. http://serverx.esri.com/arcgiscache/DG_County_roads_yesA_backgroundDark/Layers/conf.xml) │ │ │ │ │ + * │ │ │ │ │ + *Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.XYZ> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.ArcGISCache = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Enlace permanente", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String | Array} The base URL for the layer cache. You can also │ │ │ │ │ + * provide a list of URL strings for the layer if your cache is │ │ │ │ │ + * available from multiple origins. This must be set before the layer │ │ │ │ │ + * is drawn. │ │ │ │ │ + */ │ │ │ │ │ + url: null, │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Capas superpuestas", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: tileOrigin │ │ │ │ │ + * {<OpenLayers.LonLat>} The location of the tile origin for the cache. │ │ │ │ │ + * An ArcGIS cache has it's origin at the upper-left (lowest x value │ │ │ │ │ + * and highest y value of the coordinate system). The units for the │ │ │ │ │ + * tile origin should be the same as the units for the cached data. │ │ │ │ │ + */ │ │ │ │ │ + tileOrigin: null, │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Capa Base", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: tileSize │ │ │ │ │ + * {<OpenLayers.Size>} This size of each tile. Defaults to 256 by 256 pixels. │ │ │ │ │ + */ │ │ │ │ │ + tileSize: new OpenLayers.Size(256, 256), │ │ │ │ │ │ │ │ │ │ - 'noFID': "No se puede actualizar un elemento para el que no existe FID.", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: useAGS │ │ │ │ │ + * {Boolean} Indicates if we are going to be accessing the ArcGIS Server (AGS) │ │ │ │ │ + * cache via an AGS MapServer or directly through HTTP. When accessing via │ │ │ │ │ + * AGS the path structure uses a standard z/y/x structure. But AGS actually │ │ │ │ │ + * stores the tile images on disk using a hex based folder structure that looks │ │ │ │ │ + * like "http://example.com/mylayer/L00/R00000000/C00000000.png". Learn more │ │ │ │ │ + * about this here: │ │ │ │ │ + * http://blogs.esri.com/Support/blogs/mappingcenter/archive/2010/08/20/Checking-Your-Local-Cache-Folders.aspx │ │ │ │ │ + * Defaults to true; │ │ │ │ │ + */ │ │ │ │ │ + useArcGISServer: true, │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Su navegador no soporta renderización vectorial. Los renderizadores soportados actualmente son:\n${renderers}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: type │ │ │ │ │ + * {String} Image type for the layer. This becomes the filename extension │ │ │ │ │ + * in tile requests. Default is "png" (generating a url like │ │ │ │ │ + * "http://example.com/mylayer/L00/R00000000/C00000000.png"). │ │ │ │ │ + */ │ │ │ │ │ + type: 'png', │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'minZoomLevelError': "La propiedad minZoomLevel debe sólo utilizarse " + │ │ │ │ │ - "con las capas que tienen FixedZoomLevels. El hecho de que " + │ │ │ │ │ - "una capa wfs compruebe minZoomLevel es una reliquia del " + │ │ │ │ │ - "pasado. Sin embargo, no podemos eliminarla sin discontinuar " + │ │ │ │ │ - "probablemente las aplicaciones OL que puedan depender de ello. " + │ │ │ │ │ - "Así pues estamos haciéndolo obsoleto --la comprobación " + │ │ │ │ │ - "minZoomLevel se eliminará en la versión 3.0. Utilice el ajuste " + │ │ │ │ │ - "de resolution min/max en su lugar, tal como se describe aquí: " + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: useScales │ │ │ │ │ + * {Boolean} Optional override to indicate that the layer should use 'scale' information │ │ │ │ │ + * returned from the server capabilities object instead of 'resolution' information. │ │ │ │ │ + * This can be important if your tile server uses an unusual DPI for the tiles. │ │ │ │ │ + */ │ │ │ │ │ + useScales: false, │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "Transacción WFS: ÉXITO ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: overrideDPI │ │ │ │ │ + * {Boolean} Optional override to change the OpenLayers.DOTS_PER_INCH setting based │ │ │ │ │ + * on the tile information in the server capabilities object. This can be useful │ │ │ │ │ + * if your server has a non-standard DPI setting on its tiles, and you're only using │ │ │ │ │ + * tiles with that DPI. This value is used while OpenLayers is calculating resolution │ │ │ │ │ + * using scales, and is not necessary if you have resolution information. (This is │ │ │ │ │ + * typically the case) Regardless, this setting can be useful, but is dangerous │ │ │ │ │ + * because it will impact other layers while calculating resolution. Only use this │ │ │ │ │ + * if you know what you are doing. (See OpenLayers.Util.getResolutionFromScale) │ │ │ │ │ + */ │ │ │ │ │ + overrideDPI: false, │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "Transacción WFS: FALLÓ ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.ArcGISCache │ │ │ │ │ + * Creates a new instance of this class │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} │ │ │ │ │ + * url - {String} │ │ │ │ │ + * options - {Object} extra layer options │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "La capa Google no pudo ser cargada correctamente.<br><br>" + │ │ │ │ │ - "Para evitar este mensaje, seleccione una nueva Capa Base " + │ │ │ │ │ - "en el selector de capas en la esquina superior derecha.<br><br>" + │ │ │ │ │ - "Probablemente, esto se debe a que el script de la biblioteca de " + │ │ │ │ │ - "Google Maps no fue correctamente incluido en su página, o no " + │ │ │ │ │ - "contiene la clave del API correcta para su sitio.<br><br>" + │ │ │ │ │ - "Desarrolladores: Para ayudar a hacer funcionar esto correctamente, " + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ - "target='_blank'>haga clic aquí</a>", │ │ │ │ │ + if (this.resolutions) { │ │ │ │ │ + this.serverResolutions = this.resolutions; │ │ │ │ │ + this.maxExtent = this.getMaxExtentForResolution(this.resolutions[0]); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "La capa ${layerType} no pudo ser cargada correctamente.<br><br>" + │ │ │ │ │ - "Para evitar este mensaje, seleccione una nueva Capa Base " + │ │ │ │ │ - "en el selector de capas en la esquina superior derecha.<br><br>" + │ │ │ │ │ - "Probablemente, esto se debe a que el script de " + │ │ │ │ │ - "la biblioteca ${layerLib} " + │ │ │ │ │ - "no fue correctamente incluido en su página.<br><br>" + │ │ │ │ │ - "Desarrolladores: Para ayudar a hacer funcionar esto correctamente, " + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ - "target='_blank'>haga clic aquí</a>", │ │ │ │ │ + // this block steps through translating the values from the server layer JSON │ │ │ │ │ + // capabilities object into values that we can use. This is also a helpful │ │ │ │ │ + // reference when configuring this layer directly. │ │ │ │ │ + if (this.layerInfo) { │ │ │ │ │ + // alias the object │ │ │ │ │ + var info = this.layerInfo; │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Escala = 1 : ${scaleDenom}", │ │ │ │ │ + // build our extents │ │ │ │ │ + var startingTileExtent = new OpenLayers.Bounds( │ │ │ │ │ + info.fullExtent.xmin, │ │ │ │ │ + info.fullExtent.ymin, │ │ │ │ │ + info.fullExtent.xmax, │ │ │ │ │ + info.fullExtent.ymax │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - //labels for the graticule control │ │ │ │ │ - 'W': 'O', │ │ │ │ │ - 'E': 'E', │ │ │ │ │ - 'N': 'N', │ │ │ │ │ - 'S': 'S', │ │ │ │ │ - 'Graticule': 'Retícula', │ │ │ │ │ + // set our projection based on the given spatial reference. │ │ │ │ │ + // esri uses slightly different IDs, so this may not be comprehensive │ │ │ │ │ + this.projection = 'EPSG:' + info.spatialReference.wkid; │ │ │ │ │ + this.sphericalMercator = (info.spatialReference.wkid == 102100); │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'reprojectDeprecated': "Está usando la opción 'reproject' en la capa " + │ │ │ │ │ - "${layerName}. Esta opción es obsoleta: su uso fue diseñado " + │ │ │ │ │ - "para soportar la visualización de datos sobre mapas base comerciales, " + │ │ │ │ │ - "pero ahora esa funcionalidad debería conseguirse mediante el soporte " + │ │ │ │ │ - "de la proyección Spherical Mercator. Más información disponible en " + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + // convert esri units into openlayers units (basic feet or meters only) │ │ │ │ │ + this.units = (info.units == "esriFeet") ? 'ft' : 'm'; │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'methodDeprecated': "Este método es obsoleto y se eliminará en la versión 3.0. " + │ │ │ │ │ - "Por favor utilice el método ${newMethod} en su lugar.", │ │ │ │ │ + // optional extended section based on whether or not the server returned │ │ │ │ │ + // specific tile information │ │ │ │ │ + if (!!info.tileInfo) { │ │ │ │ │ + // either set the tiles based on rows/columns, or specific width/height │ │ │ │ │ + this.tileSize = new OpenLayers.Size( │ │ │ │ │ + info.tileInfo.width || info.tileInfo.cols, │ │ │ │ │ + info.tileInfo.height || info.tileInfo.rows │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - // **** end **** │ │ │ │ │ - 'end': '' │ │ │ │ │ + // this must be set when manually configuring this layer │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat( │ │ │ │ │ + info.tileInfo.origin.x, │ │ │ │ │ + info.tileInfo.origin.y │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/vi.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + var upperLeft = new OpenLayers.Geometry.Point( │ │ │ │ │ + startingTileExtent.left, │ │ │ │ │ + startingTileExtent.top │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Minh Nguyen │ │ │ │ │ - */ │ │ │ │ │ + var bottomRight = new OpenLayers.Geometry.Point( │ │ │ │ │ + startingTileExtent.right, │ │ │ │ │ + startingTileExtent.bottom │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ + if (this.useScales) { │ │ │ │ │ + this.scales = []; │ │ │ │ │ + } else { │ │ │ │ │ + this.resolutions = []; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["vi"] │ │ │ │ │ - * Dictionary for Tiếng Việt. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Lang["vi"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + this.lods = []; │ │ │ │ │ + for (var key in info.tileInfo.lods) { │ │ │ │ │ + if (info.tileInfo.lods.hasOwnProperty(key)) { │ │ │ │ │ + var lod = info.tileInfo.lods[key]; │ │ │ │ │ + if (this.useScales) { │ │ │ │ │ + this.scales.push(lod.scale); │ │ │ │ │ + } else { │ │ │ │ │ + this.resolutions.push(lod.resolution); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "Không xử lý được phản hồi ${statusText} cho yêu cầu", │ │ │ │ │ + var start = this.getContainingTileCoords(upperLeft, lod.resolution); │ │ │ │ │ + lod.startTileCol = start.x; │ │ │ │ │ + lod.startTileRow = start.y; │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Liên kết thường trực", │ │ │ │ │ + var end = this.getContainingTileCoords(bottomRight, lod.resolution); │ │ │ │ │ + lod.endTileCol = end.x; │ │ │ │ │ + lod.endTileRow = end.y; │ │ │ │ │ + this.lods.push(lod); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Lấp bản đồ", │ │ │ │ │ + this.maxExtent = this.calculateMaxExtentWithLOD(this.lods[0]); │ │ │ │ │ + this.serverResolutions = this.resolutions; │ │ │ │ │ + if (this.overrideDPI && info.tileInfo.dpi) { │ │ │ │ │ + // see comment above for 'overrideDPI' │ │ │ │ │ + OpenLayers.DOTS_PER_INCH = info.tileInfo.dpi; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Lớp nền", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getContainingTileCoords │ │ │ │ │ + * Calculates the x/y pixel corresponding to the position of the tile │ │ │ │ │ + * that contains the given point and for the for the given resolution. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * point - {<OpenLayers.Geometry.Point>} │ │ │ │ │ + * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Pixel>} The x/y pixel corresponding to the position │ │ │ │ │ + * of the upper left tile for the given resolution. │ │ │ │ │ + */ │ │ │ │ │ + getContainingTileCoords: function(point, res) { │ │ │ │ │ + return new OpenLayers.Pixel( │ │ │ │ │ + Math.max(Math.floor((point.x - this.tileOrigin.lon) / (this.tileSize.w * res)), 0), │ │ │ │ │ + Math.max(Math.floor((this.tileOrigin.lat - point.y) / (this.tileSize.h * res)), 0) │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'noFID': "Không thể cập nhật tính năng thiếu FID.", │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateMaxExtentWithLOD │ │ │ │ │ + * Given a Level of Detail object from the server, this function │ │ │ │ │ + * calculates the actual max extent │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * lod - {Object} a Level of Detail Object from the server capabilities object │ │ │ │ │ + representing a particular zoom level │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} The actual extent of the tiles for the given zoom level │ │ │ │ │ + */ │ │ │ │ │ + calculateMaxExtentWithLOD: function(lod) { │ │ │ │ │ + // the max extent we're provided with just overlaps some tiles │ │ │ │ │ + // our real extent is the bounds of all the tiles we touch │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Trình duyệt của bạn không hỗ trợ chức năng vẽ bằng vectơ. Hiện hỗ trợ các bộ kết xuất:\n${renderers}", │ │ │ │ │ + var numTileCols = (lod.endTileCol - lod.startTileCol) + 1; │ │ │ │ │ + var numTileRows = (lod.endTileRow - lod.startTileRow) + 1; │ │ │ │ │ │ │ │ │ │ - 'minZoomLevelError': "Chỉ nên sử dụng thuộc tính minZoomLevel với các lớp FixedZoomLevels-descendent. Việc lớp wfs này tìm cho minZoomLevel là di tích còn lại từ xưa. Tuy nhiên, nếu chúng tôi dời nó thì sẽ vỡ các chương trình OpenLayers mà dựa trên nó. Bởi vậy chúng tôi phản đối sử dụng nó\x26nbsp;– bước tìm cho minZoomLevel sẽ được dời vào phiên bản 3.0. Xin sử dụng thiết lập độ phân tích tối thiểu / tối đa thay thế, theo hướng dẫn này: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + var minX = this.tileOrigin.lon + (lod.startTileCol * this.tileSize.w * lod.resolution); │ │ │ │ │ + var maxX = minX + (numTileCols * this.tileSize.w * lod.resolution); │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "Giao dịch WFS: THÀNH CÔNG ${response}", │ │ │ │ │ + var maxY = this.tileOrigin.lat - (lod.startTileRow * this.tileSize.h * lod.resolution); │ │ │ │ │ + var minY = maxY - (numTileRows * this.tileSize.h * lod.resolution); │ │ │ │ │ + return new OpenLayers.Bounds(minX, minY, maxX, maxY); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "Giao dịch WFS: THẤT BẠI ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateMaxExtentWithExtent │ │ │ │ │ + * Given a 'suggested' max extent from the server, this function uses │ │ │ │ │ + * information about the actual tile sizes to determine the actual │ │ │ │ │ + * extent of the layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} The 'suggested' extent for the layer │ │ │ │ │ + * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} The actual extent of the tiles for the given zoom level │ │ │ │ │ + */ │ │ │ │ │ + calculateMaxExtentWithExtent: function(extent, res) { │ │ │ │ │ + var upperLeft = new OpenLayers.Geometry.Point(extent.left, extent.top); │ │ │ │ │ + var bottomRight = new OpenLayers.Geometry.Point(extent.right, extent.bottom); │ │ │ │ │ + var start = this.getContainingTileCoords(upperLeft, res); │ │ │ │ │ + var end = this.getContainingTileCoords(bottomRight, res); │ │ │ │ │ + var lod = { │ │ │ │ │ + resolution: res, │ │ │ │ │ + startTileCol: start.x, │ │ │ │ │ + startTileRow: start.y, │ │ │ │ │ + endTileCol: end.x, │ │ │ │ │ + endTileRow: end.y │ │ │ │ │ + }; │ │ │ │ │ + return this.calculateMaxExtentWithLOD(lod); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Không thể tải lớp Google đúng đắn.\x3cbr\x3e\x3cbr\x3eĐể tránh thông báo này lần sau, hãy chọn BaseLayer mới dùng điều khiển chọn lớp ở góc trên phải.\x3cbr\x3e\x3cbr\x3eChắc script thư viện Google Maps hoặc không được bao gồm hoặc không chứa khóa API hợp với website của bạn.\x3cbr\x3e\x3cbr\x3e\x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eTrợ giúp về tính năng này\x3c/a\x3e cho người phát triển.", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getUpperLeftTileCoord │ │ │ │ │ + * Calculates the x/y pixel corresponding to the position │ │ │ │ │ + * of the upper left tile for the given resolution. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Pixel>} The x/y pixel corresponding to the position │ │ │ │ │ + * of the upper left tile for the given resolution. │ │ │ │ │ + */ │ │ │ │ │ + getUpperLeftTileCoord: function(res) { │ │ │ │ │ + var upperLeft = new OpenLayers.Geometry.Point( │ │ │ │ │ + this.maxExtent.left, │ │ │ │ │ + this.maxExtent.top); │ │ │ │ │ + return this.getContainingTileCoords(upperLeft, res); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "Không thể tải lớp ${layerType} đúng đắn.\x3cbr\x3e\x3cbr\x3eĐể tránh thông báo này lần sau, hãy chọn BaseLayer mới dùng điều khiển chọn lớp ở góc trên phải.\x3cbr\x3e\x3cbr\x3eChắc script thư viện ${layerLib} không được bao gồm đúng kiểu.\x3cbr\x3e\x3cbr\x3e\x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eTrợ giúp về tính năng này\x3c/a\x3e cho người phát triển.", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getLowerRightTileCoord │ │ │ │ │ + * Calculates the x/y pixel corresponding to the position │ │ │ │ │ + * of the lower right tile for the given resolution. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Pixel>} The x/y pixel corresponding to the position │ │ │ │ │ + * of the lower right tile for the given resolution. │ │ │ │ │ + */ │ │ │ │ │ + getLowerRightTileCoord: function(res) { │ │ │ │ │ + var bottomRight = new OpenLayers.Geometry.Point( │ │ │ │ │ + this.maxExtent.right, │ │ │ │ │ + this.maxExtent.bottom); │ │ │ │ │ + return this.getContainingTileCoords(bottomRight, res); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Tỷ lệ = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getMaxExtentForResolution │ │ │ │ │ + * Since the max extent of a set of tiles can change from zoom level │ │ │ │ │ + * to zoom level, we need to be able to calculate that max extent │ │ │ │ │ + * for a given resolution. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * res - {Float} The resolution for which to compute the extent. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} The extent for this resolution │ │ │ │ │ + */ │ │ │ │ │ + getMaxExtentForResolution: function(res) { │ │ │ │ │ + var start = this.getUpperLeftTileCoord(res); │ │ │ │ │ + var end = this.getLowerRightTileCoord(res); │ │ │ │ │ │ │ │ │ │ - 'W': "T", │ │ │ │ │ + var numTileCols = (end.x - start.x) + 1; │ │ │ │ │ + var numTileRows = (end.y - start.y) + 1; │ │ │ │ │ │ │ │ │ │ - 'E': "Đ", │ │ │ │ │ + var minX = this.tileOrigin.lon + (start.x * this.tileSize.w * res); │ │ │ │ │ + var maxX = minX + (numTileCols * this.tileSize.w * res); │ │ │ │ │ │ │ │ │ │ - 'N': "B", │ │ │ │ │ + var maxY = this.tileOrigin.lat - (start.y * this.tileSize.h * res); │ │ │ │ │ + var minY = maxY - (numTileRows * this.tileSize.h * res); │ │ │ │ │ + return new OpenLayers.Bounds(minX, minY, maxX, maxY); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'S': "N", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Returns an exact clone of this OpenLayers.Layer.ArcGISCache │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * [obj] - {Object} optional object to assign the cloned instance to. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.ArcGISCache>} clone of this instance │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.ArcGISCache(this.name, this.url, this.options); │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Bạn đang áp dụng chế độ “reproject” vào lớp ${layerName}. Chế độ này đã bị phản đối: nó có mục đích hỗ trợ lấp dữ liệu trên các nền bản đồ thương mại; nên thực hiện hiệu ứng đó dùng tính năng Mercator Hình cầu. Có sẵn thêm chi tiết tại http://trac.openlayers.org/wiki/SphericalMercator .", │ │ │ │ │ + /** │ │ │ │ │ + * Method: initGriddedTiles │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + */ │ │ │ │ │ + initGriddedTiles: function(bounds) { │ │ │ │ │ + delete this._tileOrigin; │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initGriddedTiles.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Phương thức này đã bị phản đối và sẽ bị dời vào phiên bản 3.0. Xin hãy sử dụng ${newMethod} thay thế." │ │ │ │ │ + /** │ │ │ │ │ + * Method: getMaxExtent │ │ │ │ │ + * Get this layer's maximum extent. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} │ │ │ │ │ + */ │ │ │ │ │ + getMaxExtent: function() { │ │ │ │ │ + var resolution = this.map.getResolution(); │ │ │ │ │ + return this.maxExtent = this.getMaxExtentForResolution(resolution); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/io.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: getTileOrigin │ │ │ │ │ + * Determine the origin for aligning the grid of tiles. │ │ │ │ │ + * The origin will be derived from the layer's <maxExtent> property. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.LonLat>} The tile origin. │ │ │ │ │ + */ │ │ │ │ │ + getTileOrigin: function() { │ │ │ │ │ + if (!this._tileOrigin) { │ │ │ │ │ + var extent = this.getMaxExtent(); │ │ │ │ │ + this._tileOrigin = new OpenLayers.LonLat(extent.left, extent.bottom); │ │ │ │ │ + } │ │ │ │ │ + return this._tileOrigin; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Malafaya │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * Determine the URL for a tile given the tile bounds. This is should support │ │ │ │ │ + * urls that access tiles through an ArcGIS Server MapServer or directly through │ │ │ │ │ + * the hex folder structure using HTTP. Just be sure to set the useArcGISServer │ │ │ │ │ + * property appropriately! This is basically the same as │ │ │ │ │ + * 'OpenLayers.Layer.TMS.getURL', but with the addition of hex addressing, │ │ │ │ │ + * and tile rounding. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The URL for a tile based on given bounds. │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var res = this.getResolution(); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ + // tile center │ │ │ │ │ + var originTileX = (this.tileOrigin.lon + (res * this.tileSize.w / 2)); │ │ │ │ │ + var originTileY = (this.tileOrigin.lat - (res * this.tileSize.h / 2)); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["io"] │ │ │ │ │ - * Dictionary for Ido. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Lang["io"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var point = { │ │ │ │ │ + x: center.lon, │ │ │ │ │ + y: center.lat │ │ │ │ │ + }; │ │ │ │ │ + var x = (Math.round(Math.abs((center.lon - originTileX) / (res * this.tileSize.w)))); │ │ │ │ │ + var y = (Math.round(Math.abs((originTileY - center.lat) / (res * this.tileSize.h)))); │ │ │ │ │ + var z = this.map.getZoom(); │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Skalo = 1 : ${scaleDenom}" │ │ │ │ │ + // this prevents us from getting pink tiles (non-existant tiles) │ │ │ │ │ + if (this.lods) { │ │ │ │ │ + var lod = this.lods[this.map.getZoom()]; │ │ │ │ │ + if ((x < lod.startTileCol || x > lod.endTileCol) || │ │ │ │ │ + (y < lod.startTileRow || y > lod.endTileRow)) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + var start = this.getUpperLeftTileCoord(res); │ │ │ │ │ + var end = this.getLowerRightTileCoord(res); │ │ │ │ │ + if ((x < start.x || x >= end.x) || │ │ │ │ │ + (y < start.y || y >= end.y)) { │ │ │ │ │ + return null; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/nn.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + // Construct the url string │ │ │ │ │ + var url = this.url; │ │ │ │ │ + var s = '' + x + y + z; │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Harald Khan │ │ │ │ │ - */ │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(s, url); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ + // Accessing tiles through ArcGIS Server uses a different path │ │ │ │ │ + // structure than direct access via the folder structure. │ │ │ │ │ + if (this.useArcGISServer) { │ │ │ │ │ + // AGS MapServers have pretty url access to tiles │ │ │ │ │ + url = url + '/tile/${z}/${y}/${x}'; │ │ │ │ │ + } else { │ │ │ │ │ + // The tile images are stored using hex values on disk. │ │ │ │ │ + x = 'C' + OpenLayers.Number.zeroPad(x, 8, 16); │ │ │ │ │ + y = 'R' + OpenLayers.Number.zeroPad(y, 8, 16); │ │ │ │ │ + z = 'L' + OpenLayers.Number.zeroPad(z, 2, 10); │ │ │ │ │ + url = url + '/${z}/${y}/${x}.' + this.type; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["nn"] │ │ │ │ │ - * Dictionary for ‪Norsk (nynorsk)‬. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Lang["nn"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + // Write the values into our formatted url │ │ │ │ │ + url = OpenLayers.String.format(url, { │ │ │ │ │ + 'x': x, │ │ │ │ │ + 'y': y, │ │ │ │ │ + 'z': z │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Skala = 1 : ${scaleDenom}" │ │ │ │ │ + return OpenLayers.Util.urlAppend( │ │ │ │ │ + url, OpenLayers.Util.getParameterString(this.params) │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: 'OpenLayers.Layer.ArcGISCache' │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/sk.js │ │ │ │ │ + OpenLayers/Layer/ArcIMS.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Helix84 │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/Format/ArcXML.js │ │ │ │ │ + * @requires OpenLayers/Request.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["sk"] │ │ │ │ │ - * Dictionary for Slovenčina. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Layer.ArcIMS │ │ │ │ │ + * Instances of OpenLayers.Layer.ArcIMS are used to display data from ESRI ArcIMS │ │ │ │ │ + * Mapping Services. Create a new ArcIMS layer with the <OpenLayers.Layer.ArcIMS> │ │ │ │ │ + * constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["sk"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - │ │ │ │ │ - 'unhandledRequest': "Neobslúžené požiadavky vracajú ${statusText}", │ │ │ │ │ - │ │ │ │ │ - 'Permalink': "Trvalý odkaz", │ │ │ │ │ - │ │ │ │ │ - 'Overlays': "Prekrytia", │ │ │ │ │ - │ │ │ │ │ - 'Base Layer': "Základná vrstva", │ │ │ │ │ - │ │ │ │ │ - 'noFID': "Nie je možné aktualizovať vlastnosť, pre ktorú neexistuje FID.", │ │ │ │ │ - │ │ │ │ │ - 'browserNotSupported': "Váš prehliadač nepodporuje vykresľovanie vektorov. Momentálne podporované vykresľovače sú:\n${renderers}", │ │ │ │ │ - │ │ │ │ │ - 'minZoomLevelError': "Vlastnosť minZoomLevel je určený iba na použitie s vrstvami odvodenými od FixedZoomLevels. To, že táto wfs vrstva kontroluje minZoomLevel je pozostatok z minulosti. Nemôžeme ho však odstrániť, aby sme sa vyhli možnému porušeniu aplikácií založených na Open Layers, ktoré na tomto môže závisieť. Preto ho označujeme ako zavrhovaný - dolu uvedená kontrola minZoomLevel bude odstránená vo verzii 3.0. Použite prosím namiesto toho kontrolu min./max. rozlíšenia podľa tu uvedeného popisu: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - │ │ │ │ │ - 'commitSuccess': "Transakcia WFS: ÚSPEŠNÁ ${response}", │ │ │ │ │ - │ │ │ │ │ - 'commitFailed': "Transakcia WFS: ZLYHALA ${response}", │ │ │ │ │ +OpenLayers.Layer.ArcIMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Vrstvu Google nebolo možné správne načítať.\x3cbr\x3e\x3cbr\x3eAby ste sa tejto správy zbavili vyberte novú BaseLayer v prepínači vrstiev v pravom hornom rohu.\x3cbr\x3e\x3cbr\x3eToto sa stalo pravdepodobne preto, že skript knižnice Google Maps buď nebol načítaný alebo neobsahuje správny kľúč API pre vašu lokalitu.\x3cbr\x3e\x3cbr\x3eVývojári: Tu môžete získať \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3epomoc so sfunkčnením\x3c/a\x3e", │ │ │ │ │ + /** │ │ │ │ │ + * Constant: DEFAULT_PARAMS │ │ │ │ │ + * {Object} Default query string parameters. │ │ │ │ │ + */ │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + ClientVersion: "9.2", │ │ │ │ │ + ServiceName: '' │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "Vrstvu ${layerType} nebolo možné správne načítať.\x3cbr\x3e\x3cbr\x3eAby ste sa tejto správy zbavili vyberte novú BaseLayer v prepínači vrstiev v pravom hornom rohu.\x3cbr\x3e\x3cbr\x3eToto sa stalo pravdepodobne preto, že skript knižnice ${layerType} buď nebol načítaný alebo neobsahuje správny kľúč API pre vašu lokalitu.\x3cbr\x3e\x3cbr\x3eVývojári: Tu môžete získať \x3ca href=\'http://trac.openlayers.org/wiki/${layerType}\' target=\'_blank\'\x3epomoc so sfunkčnením\x3c/a\x3e", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: featureCoordSys │ │ │ │ │ + * {String} Code for feature coordinate system. Default is "4326". │ │ │ │ │ + */ │ │ │ │ │ + featureCoordSys: "4326", │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Mierka = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: filterCoordSys │ │ │ │ │ + * {String} Code for filter coordinate system. Default is "4326". │ │ │ │ │ + */ │ │ │ │ │ + filterCoordSys: "4326", │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Používate voľby „reproject“ vrstvy ${layerType}. Táto voľba je zzavrhovaná: jej použitie bolo navrhnuté na podporu zobrazovania údajov nad komerčnými základovými mapami, ale túto funkcionalitu je teraz možné dosiahnuť pomocou Spherical Mercator. Ďalšie informácie získate na stránke http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layers │ │ │ │ │ + * {Array} An array of objects with layer properties. │ │ │ │ │ + */ │ │ │ │ │ + layers: null, │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Táto metóda je zavrhovaná a bude odstránená vo verzii 3.0. Použite prosím namiesto nej metódu ${newMethod}." │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/gsw.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: async │ │ │ │ │ + * {Boolean} Request images asynchronously. Default is true. │ │ │ │ │ + */ │ │ │ │ │ + async: true, │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Als-Holder │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: name │ │ │ │ │ + * {String} Layer name. Default is "ArcIMS". │ │ │ │ │ + */ │ │ │ │ │ + name: "ArcIMS", │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} The layer is a base layer. Default is true. │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["gsw"] │ │ │ │ │ - * Dictionary for Alemannisch. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Lang["gsw"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + /** │ │ │ │ │ + * Constant: DEFAULT_OPTIONS │ │ │ │ │ + * {Object} Default layers properties. │ │ │ │ │ + */ │ │ │ │ │ + DEFAULT_OPTIONS: { │ │ │ │ │ + tileSize: new OpenLayers.Size(512, 512), │ │ │ │ │ + featureCoordSys: "4326", │ │ │ │ │ + filterCoordSys: "4326", │ │ │ │ │ + layers: null, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + async: true, │ │ │ │ │ + name: "ArcIMS" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "Nit behandleti Aafrogsruckmäldig ${statusText}", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.ArcIMS │ │ │ │ │ + * Create a new ArcIMS layer object. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var arcims = new OpenLayers.Layer.ArcIMS( │ │ │ │ │ + * "Global Sample", │ │ │ │ │ + * "http://sample.avencia.com/servlet/com.esri.esrimap.Esrimap", │ │ │ │ │ + * { │ │ │ │ │ + * service: "OpenLayers_Sample", │ │ │ │ │ + * layers: [ │ │ │ │ │ + * // layers to manipulate │ │ │ │ │ + * {id: "1", visible: true} │ │ │ │ │ + * ] │ │ │ │ │ + * } │ │ │ │ │ + * ); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} A name for the layer │ │ │ │ │ + * url - {String} Base url for the ArcIMS server │ │ │ │ │ + * options - {Object} Optional object with properties to be set on the │ │ │ │ │ + * layer. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Permalink", │ │ │ │ │ + this.tileSize = new OpenLayers.Size(512, 512); │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Iberlagerige", │ │ │ │ │ + // parameters │ │ │ │ │ + this.params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + ServiceName: options.serviceName │ │ │ │ │ + }, │ │ │ │ │ + this.DEFAULT_PARAMS │ │ │ │ │ + ); │ │ │ │ │ + this.options = OpenLayers.Util.applyDefaults( │ │ │ │ │ + options, this.DEFAULT_OPTIONS │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Grundcharte", │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply( │ │ │ │ │ + this, [name, url, this.params, options] │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - 'noFID': "E Feature, wu s kei FID derfir git, cha nit aktualisiert wäre.", │ │ │ │ │ + //layer is transparent │ │ │ │ │ + if (this.transparent) { │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Dyy Browser unterstitzt kei Vektordarstellig. Aktuäll unterstitzti Renderer:\n${renderers}", │ │ │ │ │ + // unless explicitly set in options, make layer an overlay │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + this.isBaseLayer = false; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'minZoomLevelError': "D minZoomLevel-Eigeschaft isch nume dänk fir d Layer, wu vu dr FixedZoomLevels abstamme. Ass dää wfs-Layer minZoomLevel prieft, scih e Relikt us dr Vergangeheit. Mir chenne s aber nit ändere ohni OL_basierti Aawändige villicht kaputt gehn, wu dervu abhänge. Us däm Grund het die Funktion d Eigeschaft \'deprecated\' iberchuu. D minZoomLevel-Priefig unte wird in dr Version 3.0 usegnuu. Bitte verwänd statt däm e min/max-Uflesig wie s do bschriben isch: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + // jpegs can never be transparent, so intelligently switch the │ │ │ │ │ + // format, depending on the browser's capabilities │ │ │ │ │ + if (this.format == "image/jpeg") { │ │ │ │ │ + this.format = OpenLayers.Util.alphaHack() ? "image/gif" : "image/png"; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "WFS-Transaktion: ERFOLGRYCH ${response}", │ │ │ │ │ + // create an empty layer list if no layers specified in the options │ │ │ │ │ + if (this.options.layers === null) { │ │ │ │ │ + this.options.layers = []; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "WFS-Transaktion: FÄHLGSCHLAA ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * Return an image url this layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox for the │ │ │ │ │ + * request. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the map image's url. │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var url = ""; │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Dr Google-Layer het nit korräkt chenne glade wäre.\x3cbr\x3e\x3cbr\x3eGo die Mäldig nimi z kriege, wehl e andere Hintergrundlayer us em LayerSwitcher im rächte obere Ecke.\x3cbr\x3e\x3cbr\x3eDää Fähler git s seli hyfig, wel s Skript vu dr Google-Maps-Bibliothek nit yybunde woren isch oder wel s kei giltige API-Schlissel fir Dyy URL din het.\x3cbr\x3e\x3cbr\x3eEntwickler: Fir Hilf zum korräkte Yybinde vum Google-Layer \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3edoo drucke\x3c/a\x3e", │ │ │ │ │ + // create an arcxml request to generate the image │ │ │ │ │ + var axlReq = new OpenLayers.Format.ArcXML( │ │ │ │ │ + OpenLayers.Util.extend(this.options, { │ │ │ │ │ + requesttype: "image", │ │ │ │ │ + envelope: bounds.toArray(), │ │ │ │ │ + tileSize: this.tileSize │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "Dr ${layerType}-Layer het nit korräkt chenne glade wäre.\x3cbr\x3e\x3cbr\x3eGo die Mäldig nimi z kriege, wehl e andere Hintergrundlayer us em LayerSwitcher im rächte obere Ecke.\x3cbr\x3e\x3cbr\x3eDää Fähler git s seli hyfig, wel s Skript vu dr \'${layerLib}\'-Bibliothek nit yybunde woren isch oder wel s kei giltige API-Schlissel fir Dyy URL din het.\x3cbr\x3e\x3cbr\x3eEntwickler: Fir Hilf zum korräkte Yybinde vu Layer \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3edoo drucke\x3c/a\x3e", │ │ │ │ │ + // create a synchronous ajax request to get an arcims image │ │ │ │ │ + var req = new OpenLayers.Request.POST({ │ │ │ │ │ + url: this.getFullRequestString(), │ │ │ │ │ + data: axlReq.write(), │ │ │ │ │ + async: false │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Maßstab = 1 : ${scaleDenom}", │ │ │ │ │ + // if the response exists │ │ │ │ │ + if (req != null) { │ │ │ │ │ + var doc = req.responseXML; │ │ │ │ │ │ │ │ │ │ - 'W': "W", │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = req.responseText; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'E': "O", │ │ │ │ │ + // create a new arcxml format to read the response │ │ │ │ │ + var axlResp = new OpenLayers.Format.ArcXML(); │ │ │ │ │ + var arcxml = axlResp.read(doc); │ │ │ │ │ + url = this.getUrlOrImage(arcxml.image.output); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'N': "N", │ │ │ │ │ + return url; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'S': "S", │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Du bruchsch d \'reproject\'-Option bim ${layerName}-Layer. Die Option isch nimi giltig: si isch aagleit wore go Date iber kommerziälli Grundcharte lege, aber des sott mer jetz mache mit dr Unterstitzig vu Spherical Mercator. Meh Informatione git s uf http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURLasync │ │ │ │ │ + * Get an image url this layer asynchronously, and execute a callback │ │ │ │ │ + * when the image url is generated. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox for the │ │ │ │ │ + * request. │ │ │ │ │ + * callback - {Function} Function to call when image url is retrieved. │ │ │ │ │ + * scope - {Object} The scope of the callback method. │ │ │ │ │ + */ │ │ │ │ │ + getURLasync: function(bounds, callback, scope) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Die Methode isch veraltet un wird us dr Version 3.0 usegnuu. Bitte verwäbnd statt däm ${newMethod}." │ │ │ │ │ + // create an arcxml request to generate the image │ │ │ │ │ + var axlReq = new OpenLayers.Format.ArcXML( │ │ │ │ │ + OpenLayers.Util.extend(this.options, { │ │ │ │ │ + requesttype: "image", │ │ │ │ │ + envelope: bounds.toArray(), │ │ │ │ │ + tileSize: this.tileSize │ │ │ │ │ + }) │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/br.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + // create an asynchronous ajax request to get an arcims image │ │ │ │ │ + OpenLayers.Request.POST({ │ │ │ │ │ + url: this.getFullRequestString(), │ │ │ │ │ + async: true, │ │ │ │ │ + data: axlReq.write(), │ │ │ │ │ + callback: function(req) { │ │ │ │ │ + // process the response from ArcIMS, and call the callback function │ │ │ │ │ + // to set the image URL │ │ │ │ │ + var doc = req.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = req.responseText; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Fulup │ │ │ │ │ - */ │ │ │ │ │ + // create a new arcxml format to read the response │ │ │ │ │ + var axlResp = new OpenLayers.Format.ArcXML(); │ │ │ │ │ + var arcxml = axlResp.read(doc); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ + callback.call(scope, this.getUrlOrImage(arcxml.image.output)); │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["br"] │ │ │ │ │ - * Dictionary for Brezhoneg. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Lang["br"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + /** │ │ │ │ │ + * Method: getUrlOrImage │ │ │ │ │ + * Extract a url or image from the ArcXML image output. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * output - {Object} The image.output property of the object returned from │ │ │ │ │ + * the ArcXML format read method. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A URL for an image (potentially with the data protocol). │ │ │ │ │ + */ │ │ │ │ │ + getUrlOrImage: function(output) { │ │ │ │ │ + var ret = ""; │ │ │ │ │ + if (output.url) { │ │ │ │ │ + // If the image response output url is a string, then the image │ │ │ │ │ + // data is not inline. │ │ │ │ │ + ret = output.url; │ │ │ │ │ + } else if (output.data) { │ │ │ │ │ + // The image data is inline and base64 encoded, create a data │ │ │ │ │ + // url for the image. This will only work for small images, │ │ │ │ │ + // due to browser url length limits. │ │ │ │ │ + ret = "data:image/" + output.type + │ │ │ │ │ + ";base64," + output.data; │ │ │ │ │ + } │ │ │ │ │ + return ret; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "Distro evel reked anveret ${statusText}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: setLayerQuery │ │ │ │ │ + * Set the query definition on this layer. Query definitions are used to │ │ │ │ │ + * render parts of the spatial data in an image, and can be used to │ │ │ │ │ + * filter features or layers in the ArcIMS service. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} The ArcIMS layer ID. │ │ │ │ │ + * querydef - {Object} The query definition to apply to this layer. │ │ │ │ │ + */ │ │ │ │ │ + setLayerQuery: function(id, querydef) { │ │ │ │ │ + // find the matching layer, if it exists │ │ │ │ │ + for (var lyr = 0; lyr < this.options.layers.length; lyr++) { │ │ │ │ │ + if (id == this.options.layers[lyr].id) { │ │ │ │ │ + // replace this layer definition │ │ │ │ │ + this.options.layers[lyr].query = querydef; │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Peurliamm", │ │ │ │ │ + // no layer found, create a new definition │ │ │ │ │ + this.options.layers.push({ │ │ │ │ │ + id: id, │ │ │ │ │ + visible: true, │ │ │ │ │ + query: querydef │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Gwiskadoù", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getFeatureInfo │ │ │ │ │ + * Get feature information from ArcIMS. Using the applied geometry, apply │ │ │ │ │ + * the options to the query (buffer, area/envelope intersection), and │ │ │ │ │ + * query the ArcIMS service. │ │ │ │ │ + * │ │ │ │ │ + * A note about accuracy: │ │ │ │ │ + * ArcIMS interprets the accuracy attribute in feature requests to be │ │ │ │ │ + * something like the 'modulus' operator on feature coordinates, │ │ │ │ │ + * applied to the database geometry of the feature. It doesn't round, │ │ │ │ │ + * so your feature coordinates may be up to (1 x accuracy) offset from │ │ │ │ │ + * the actual feature coordinates. If the accuracy of the layer is not │ │ │ │ │ + * specified, the accuracy will be computed to be approximately 1 │ │ │ │ │ + * feature coordinate per screen pixel. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * geometry - {<OpenLayers.LonLat>} or {<OpenLayers.Geometry.Polygon>} The │ │ │ │ │ + * geometry to use when making the query. This should be a closed │ │ │ │ │ + * polygon for behavior approximating a free selection. │ │ │ │ │ + * layer - {Object} The ArcIMS layer definition. This is an anonymous object │ │ │ │ │ + * that looks like: │ │ │ │ │ + * (code) │ │ │ │ │ + * { │ │ │ │ │ + * id: "ArcXML layer ID", // the ArcXML layer ID │ │ │ │ │ + * query: { │ │ │ │ │ + * where: "STATE = 'PA'", // the where clause of the query │ │ │ │ │ + * accuracy: 100 // the accuracy of the returned feature │ │ │ │ │ + * } │ │ │ │ │ + * } │ │ │ │ │ + * (end) │ │ │ │ │ + * options - {Object} Object with non-default properties to set on the layer. │ │ │ │ │ + * Supported properties are buffer, callback, scope, and any other │ │ │ │ │ + * properties applicable to the ArcXML format. Set the 'callback' and │ │ │ │ │ + * 'scope' for an object and function to recieve the parsed features │ │ │ │ │ + * from ArcIMS. │ │ │ │ │ + */ │ │ │ │ │ + getFeatureInfo: function(geometry, layer, options) { │ │ │ │ │ + // set the buffer to 1 unit (dd/m/ft?) by default │ │ │ │ │ + var buffer = options.buffer || 1; │ │ │ │ │ + // empty callback by default │ │ │ │ │ + var callback = options.callback || function() {}; │ │ │ │ │ + // default scope is window (global) │ │ │ │ │ + var scope = options.scope || window; │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Gwiskad diazez", │ │ │ │ │ + // apply these option to the request options │ │ │ │ │ + var requestOptions = {}; │ │ │ │ │ + OpenLayers.Util.extend(requestOptions, this.options); │ │ │ │ │ │ │ │ │ │ - 'noFID': "N\'haller ket hizivaat un elfenn ma n\'eus ket a niverenn-anaout (FID) eviti.", │ │ │ │ │ + // this is a feature request │ │ │ │ │ + requestOptions.requesttype = "feature"; │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "N\'eo ket skoret an daskor vektorel gant ho merdeer. Setu aze an daskorerioù skoret evit ar poent :\n${renderers}", │ │ │ │ │ + if (geometry instanceof OpenLayers.LonLat) { │ │ │ │ │ + // create an envelope if the geometry is really a lon/lat │ │ │ │ │ + requestOptions.polygon = null; │ │ │ │ │ + requestOptions.envelope = [ │ │ │ │ │ + geometry.lon - buffer, │ │ │ │ │ + geometry.lat - buffer, │ │ │ │ │ + geometry.lon + buffer, │ │ │ │ │ + geometry.lat + buffer │ │ │ │ │ + ]; │ │ │ │ │ + } else if (geometry instanceof OpenLayers.Geometry.Polygon) { │ │ │ │ │ + // use the polygon assigned, and empty the envelope │ │ │ │ │ + requestOptions.envelope = null; │ │ │ │ │ + requestOptions.polygon = geometry; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'minZoomLevelError': "Ne zleer implijout ar perzh minZoomLevel nemet evit gwiskadoù FixedZoomLevels-descendent. Ar fed ma wiria ar gwiskad WHS-se hag-eñ ez eus eus minZoomLevel zo un aspadenn gozh. Koulskoude n\'omp ket evit e ziverkañ kuit da derriñ arloadoù diazezet war OL a c\'hallfe bezañ stag outañ. Setu perak eo dispredet -- Lamet kuit e vo ar gwiriañ minZoomLevel a-is er stumm 3.0. Ober gant an arventennoù bihanañ/brasañ evel deskrivet amañ e plas : http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + // create an arcxml request to get feature requests │ │ │ │ │ + var arcxml = new OpenLayers.Format.ArcXML(requestOptions); │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "Treuzgread WFS : MAT EO ${response}", │ │ │ │ │ + // apply any get feature options to the arcxml request │ │ │ │ │ + OpenLayers.Util.extend(arcxml.request.get_feature, options); │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "Treuzgread WFS Transaction: C\'HWITET ${response}", │ │ │ │ │ + arcxml.request.get_feature.layer = layer.id; │ │ │ │ │ + if (typeof layer.query.accuracy == "number") { │ │ │ │ │ + // set the accuracy if it was specified │ │ │ │ │ + arcxml.request.get_feature.query.accuracy = layer.query.accuracy; │ │ │ │ │ + } else { │ │ │ │ │ + // guess that the accuracy is 1 per screen pixel │ │ │ │ │ + var mapCenter = this.map.getCenter(); │ │ │ │ │ + var viewPx = this.map.getViewPortPxFromLonLat(mapCenter); │ │ │ │ │ + viewPx.x++; │ │ │ │ │ + var mapOffCenter = this.map.getLonLatFromPixel(viewPx); │ │ │ │ │ + arcxml.request.get_feature.query.accuracy = mapOffCenter.lon - mapCenter.lon; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "N\'eus ket bet gallet kargañ ar gwiskad Google ent reizh.\x3cbr\x3e\x3cbr\x3eEvit en em zizober eus ar c\'hemenn-mañ, dibabit ur BaseLayer nevez en diuzer gwiskadoù er c\'horn dehoù el laez.\x3cbr\x3e\x3cbr\x3eSur a-walc\'h eo peogwir n\'eo ket bet ensoc\'het levraoueg Google Maps pe neuze ne glot ket an alc\'hwez API gant ho lec\'hienn.\x3cbr\x3e\x3cbr\x3eDiorroerien : Evit reizhañ an dra-se, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eclick here\x3c/a\x3e", │ │ │ │ │ + // set the get_feature query to be the same as the layer passed in │ │ │ │ │ + arcxml.request.get_feature.query.where = layer.query.where; │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "N\'haller ket kargañ ar gwiskad ${layerType} ent reizh.\x3cbr\x3e\x3cbr\x3eEvit en em zizober eus ar c\'hemenn-mañ, dibabit ur BaseLayer nevez en diuzer gwiskadoù er c\'horn dehoù el laez.\x3cbr\x3e\x3cbr\x3eSur a-walc\'h eo peogwir n\'eo ket bet ensoc\'het mat al levraoueg ${layerLib}.\x3cbr\x3e\x3cbr\x3eDiorroerien : Evit gouzout penaos reizhañ an dra-se, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eclick here\x3c/a\x3e", │ │ │ │ │ + // use area_intersection │ │ │ │ │ + arcxml.request.get_feature.query.spatialfilter.relation = "area_intersection"; │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Skeul = 1 : ${scaleDenom}", │ │ │ │ │ + // create a new asynchronous request to get the feature info │ │ │ │ │ + OpenLayers.Request.POST({ │ │ │ │ │ + url: this.getFullRequestString({ │ │ │ │ │ + 'CustomService': 'Query' │ │ │ │ │ + }), │ │ │ │ │ + data: arcxml.write(), │ │ │ │ │ + callback: function(request) { │ │ │ │ │ + // parse the arcxml response │ │ │ │ │ + var response = arcxml.parseResponse(request.responseText); │ │ │ │ │ │ │ │ │ │ - 'W': "K", │ │ │ │ │ + if (!arcxml.iserror()) { │ │ │ │ │ + // if the arcxml is not an error, call the callback with the features parsed │ │ │ │ │ + callback.call(scope, response.features); │ │ │ │ │ + } else { │ │ │ │ │ + // if the arcxml is an error, return null features selected │ │ │ │ │ + callback.call(scope, null); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'E': "R", │ │ │ │ │ + /** │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.ArcIMS>} An exact clone of this layer │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ │ │ │ │ │ - 'N': "N", │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.ArcIMS(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'S': "S", │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Emaoc\'h oc\'h implijout an dibarzh \'reproject\' war ar gwiskad ${layerName}. Dispredet eo an dibarzh-mañ : bet eo hag e talveze da ziskwel roadennoù war-c\'horre kartennoù diazez kenwerzhel, un dra hag a c\'haller ober bremañ gant an arc\'hwel dre skor banndres boullek Mercator. Muioc\'h a ditouroù a c\'haller da gaout war http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Dispredet eo an daore-se ha tennet e vo kuit eus ar stumm 3.0. Grit gant ${newMethod} e plas." │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.ArcIMS" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/da-DK.js │ │ │ │ │ + OpenLayers/Layer/PointGrid.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ + * @requires OpenLayers/Geometry/Polygon.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["da-DK"] │ │ │ │ │ - * Dictionary for Danish. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Layer.PointGrid │ │ │ │ │ + * A point grid layer dynamically generates a regularly spaced grid of point │ │ │ │ │ + * features. This is a specialty layer for cases where an application needs │ │ │ │ │ + * a regular grid of points. It can be used, for example, in an editing │ │ │ │ │ + * environment to snap to a grid. │ │ │ │ │ + * │ │ │ │ │ + * Create a new vector layer with the <OpenLayers.Layer.PointGrid> constructor. │ │ │ │ │ + * (code) │ │ │ │ │ + * // create a grid with points spaced at 10 map units │ │ │ │ │ + * var points = new OpenLayers.Layer.PointGrid({dx: 10, dy: 10}); │ │ │ │ │ + * │ │ │ │ │ + * // create a grid with different x/y spacing rotated 15 degrees clockwise. │ │ │ │ │ + * var points = new OpenLayers.Layer.PointGrid({dx: 5, dy: 10, rotation: 15}); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Vector> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang['da-DK'] = { │ │ │ │ │ +OpenLayers.Layer.PointGrid = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "En ikke håndteret forespørgsel returnerede ${statusText}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: dx │ │ │ │ │ + * {Number} Point grid spacing in the x-axis direction (map units). │ │ │ │ │ + * Read-only. Use the <setSpacing> method to modify this value. │ │ │ │ │ + */ │ │ │ │ │ + dx: null, │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Permalink", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: dy │ │ │ │ │ + * {Number} Point grid spacing in the y-axis direction (map units). │ │ │ │ │ + * Read-only. Use the <setSpacing> method to modify this value. │ │ │ │ │ + */ │ │ │ │ │ + dy: null, │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Kortlag", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: ratio │ │ │ │ │ + * {Number} Ratio of the desired grid size to the map viewport size. │ │ │ │ │ + * Default is 1.5. Larger ratios mean the grid is recalculated less often │ │ │ │ │ + * while panning. The <maxFeatures> setting has precedence when determining │ │ │ │ │ + * grid size. Read-only. Use the <setRatio> method to modify this value. │ │ │ │ │ + */ │ │ │ │ │ + ratio: 1.5, │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Baggrundslag", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxFeatures │ │ │ │ │ + * {Number} The maximum number of points to generate in the grid. Default │ │ │ │ │ + * is 250. Read-only. Use the <setMaxFeatures> method to modify this value. │ │ │ │ │ + */ │ │ │ │ │ + maxFeatures: 250, │ │ │ │ │ │ │ │ │ │ - 'noFID': "Kan ikke opdateret en feature (et objekt) der ikke har et FID.", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: rotation │ │ │ │ │ + * {Number} Grid rotation (in degrees clockwise from the positive x-axis). │ │ │ │ │ + * Default is 0. Read-only. Use the <setRotation> method to modify this │ │ │ │ │ + * value. │ │ │ │ │ + */ │ │ │ │ │ + rotation: 0, │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Din browser understøtter ikke vektor visning. Følgende vektor visninger understøttes:\n${renderers}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: origin │ │ │ │ │ + * {<OpenLayers.LonLat>} Grid origin. The grid lattice will be aligned with │ │ │ │ │ + * the origin. If not set at construction, the center of the map's maximum │ │ │ │ │ + * extent is used. Read-only. Use the <setOrigin> method to modify this │ │ │ │ │ + * value. │ │ │ │ │ + */ │ │ │ │ │ + origin: null, │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'minZoomLevelError': "Egenskaben minZoomLevel er kun beregnet til brug " + │ │ │ │ │ - "med FixedZoomLevels. At dette WFS lag kontrollerer " + │ │ │ │ │ - "minZoomLevel egenskaben, er et levn fra en tidligere " + │ │ │ │ │ - "version. Vi kan desværre ikke fjerne dette uden at risikere " + │ │ │ │ │ - "at ødelægge eksisterende OL baserede programmer der " + │ │ │ │ │ - " benytter denne funktionalitet. " + │ │ │ │ │ - "Egenskaben bør derfor ikke anvendes, og minZoomLevel " + │ │ │ │ │ - "kontrollen herunder vil blive fjernet i version 3.0. " + │ │ │ │ │ - "Benyt istedet min/max opløsnings indstillingerne, som " + │ │ │ │ │ - "er beskrevet her: " + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + /** │ │ │ │ │ + * Property: gridBounds │ │ │ │ │ + * {<OpenLayers.Bounds>} Internally cached grid bounds (with optional │ │ │ │ │ + * rotation applied). │ │ │ │ │ + */ │ │ │ │ │ + gridBounds: null, │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "WFS transaktion: LYKKEDES ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.PointGrid │ │ │ │ │ + * Creates a new point grid layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} An object containing all configuration properties for │ │ │ │ │ + * the layer. The <dx> and <dy> properties are required to be set at │ │ │ │ │ + * construction. Any other layer properties may be set in this object. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ + config = config || {}; │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.initialize.apply(this, [config.name, config]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "WFS transaktion: MISLYKKEDES ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * The layer has been added to the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ + map.events.register("moveend", this, this.onMoveEnd); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Google laget kunne ikke indlæses.<br><br>" + │ │ │ │ │ - "For at fjerne denne besked, vælg et nyt bagrundskort i " + │ │ │ │ │ - "lagskifteren i øverste højre hjørne.<br><br>" + │ │ │ │ │ - "Fejlen skyldes formentlig at Google Maps bibliotekts " + │ │ │ │ │ - "scriptet ikke er inkluderet, eller ikke indeholder den " + │ │ │ │ │ - "korrkte API nøgle for dit site.<br><br>" + │ │ │ │ │ - "Udviklere: For hjælp til at få dette til at fungere, " + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ - "target='_blank'>klik her</a>", │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeMap │ │ │ │ │ + * The layer has been removed from the map. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + map.events.unregister("moveend", this, this.onMoveEnd); │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "${layerType}-laget kunne ikke indlæses.<br><br>" + │ │ │ │ │ - "For at fjerne denne besked, vælg et nyt bagrundskort i " + │ │ │ │ │ - "lagskifteren i øverste højre hjørne.<br><br>" + │ │ │ │ │ - "Fejlen skyldes formentlig at ${layerLib} bibliotekts " + │ │ │ │ │ - "scriptet ikke er inkluderet.<br><br>" + │ │ │ │ │ - "Udviklere: For hjælp til at få dette til at fungere, " + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ - "target='_blank'>klik her</a>", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setRatio │ │ │ │ │ + * Set the grid <ratio> property and update the grid. Can only be called │ │ │ │ │ + * after the layer has been added to a map with a center/extent. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * ratio - {Number} │ │ │ │ │ + */ │ │ │ │ │ + setRatio: function(ratio) { │ │ │ │ │ + this.ratio = ratio; │ │ │ │ │ + this.updateGrid(true); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Målforhold = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setMaxFeatures │ │ │ │ │ + * Set the grid <maxFeatures> property and update the grid. Can only be │ │ │ │ │ + * called after the layer has been added to a map with a center/extent. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * maxFeatures - {Number} │ │ │ │ │ + */ │ │ │ │ │ + setMaxFeatures: function(maxFeatures) { │ │ │ │ │ + this.maxFeatures = maxFeatures; │ │ │ │ │ + this.updateGrid(true); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'reprojectDeprecated': "Du anvender indstillingen 'reproject' på laget ${layerName}." + │ │ │ │ │ - "Denne indstilling bør ikke længere anvendes. Den var beregnet " + │ │ │ │ │ - "til at vise data ovenpå kommercielle grundkort, men den funktionalitet " + │ │ │ │ │ - "bør nu opnås ved at anvende Spherical Mercator understøttelsen. " + │ │ │ │ │ - "Mere information er tilgængelig her: " + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setSpacing │ │ │ │ │ + * Set the grid <dx> and <dy> properties and update the grid. If only one │ │ │ │ │ + * argument is provided, it will be set as <dx> and <dy>. Can only be │ │ │ │ │ + * called after the layer has been added to a map with a center/extent. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * dx - {Number} │ │ │ │ │ + * dy - {Number} │ │ │ │ │ + */ │ │ │ │ │ + setSpacing: function(dx, dy) { │ │ │ │ │ + this.dx = dx; │ │ │ │ │ + this.dy = dy || dx; │ │ │ │ │ + this.updateGrid(true); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'methodDeprecated': "Denne funktion bør ikke længere anvendes, og vil blive fjernet i version 3.0. " + │ │ │ │ │ - "Anvend venligst funktionen ${newMethod} istedet." │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/km.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setOrigin │ │ │ │ │ + * Set the grid <origin> property and update the grid. Can only be called │ │ │ │ │ + * after the layer has been added to a map with a center/extent. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * origin - {<OpenLayers.LonLat>} │ │ │ │ │ + */ │ │ │ │ │ + setOrigin: function(origin) { │ │ │ │ │ + this.origin = origin; │ │ │ │ │ + this.updateGrid(true); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - វ័ណថារិទ្ធ │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getOrigin │ │ │ │ │ + * Get the grid <origin> property. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.LonLat>} The grid origin. │ │ │ │ │ + */ │ │ │ │ │ + getOrigin: function() { │ │ │ │ │ + if (!this.origin) { │ │ │ │ │ + this.origin = this.map.getExtent().getCenterLonLat(); │ │ │ │ │ + } │ │ │ │ │ + return this.origin; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setRotation │ │ │ │ │ + * Set the grid <rotation> property and update the grid. Rotation values │ │ │ │ │ + * are in degrees clockwise from the positive x-axis (negative values │ │ │ │ │ + * for counter-clockwise rotation). Can only be called after the layer │ │ │ │ │ + * has been added to a map with a center/extent. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * rotation - {Number} Degrees clockwise from the positive x-axis. │ │ │ │ │ + */ │ │ │ │ │ + setRotation: function(rotation) { │ │ │ │ │ + this.rotation = rotation; │ │ │ │ │ + this.updateGrid(true); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["km"] │ │ │ │ │ - * Dictionary for ភាសាខ្មែរ. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Lang["km"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + /** │ │ │ │ │ + * Method: onMoveEnd │ │ │ │ │ + * Listener for map "moveend" events. │ │ │ │ │ + */ │ │ │ │ │ + onMoveEnd: function() { │ │ │ │ │ + this.updateGrid(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Permalink': "តំណភ្ជាប់អចិន្ត្រៃយ៍", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getViewBounds │ │ │ │ │ + * Gets the (potentially rotated) view bounds for grid calculations. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} │ │ │ │ │ + */ │ │ │ │ │ + getViewBounds: function() { │ │ │ │ │ + var bounds = this.map.getExtent(); │ │ │ │ │ + if (this.rotation) { │ │ │ │ │ + var origin = this.getOrigin(); │ │ │ │ │ + var rotationOrigin = new OpenLayers.Geometry.Point(origin.lon, origin.lat); │ │ │ │ │ + var rect = bounds.toGeometry(); │ │ │ │ │ + rect.rotate(-this.rotation, rotationOrigin); │ │ │ │ │ + bounds = rect.getBounds(); │ │ │ │ │ + } │ │ │ │ │ + return bounds; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "ស្រទាប់បាត​", │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateGrid │ │ │ │ │ + * Update the grid. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * force - {Boolean} Update the grid even if the previous bounds are still │ │ │ │ │ + * valid. │ │ │ │ │ + */ │ │ │ │ │ + updateGrid: function(force) { │ │ │ │ │ + if (force || this.invalidBounds()) { │ │ │ │ │ + var viewBounds = this.getViewBounds(); │ │ │ │ │ + var origin = this.getOrigin(); │ │ │ │ │ + var rotationOrigin = new OpenLayers.Geometry.Point(origin.lon, origin.lat); │ │ │ │ │ + var viewBoundsWidth = viewBounds.getWidth(); │ │ │ │ │ + var viewBoundsHeight = viewBounds.getHeight(); │ │ │ │ │ + var aspectRatio = viewBoundsWidth / viewBoundsHeight; │ │ │ │ │ + var maxHeight = Math.sqrt(this.dx * this.dy * this.maxFeatures / aspectRatio); │ │ │ │ │ + var maxWidth = maxHeight * aspectRatio; │ │ │ │ │ + var gridWidth = Math.min(viewBoundsWidth * this.ratio, maxWidth); │ │ │ │ │ + var gridHeight = Math.min(viewBoundsHeight * this.ratio, maxHeight); │ │ │ │ │ + var center = viewBounds.getCenterLonLat(); │ │ │ │ │ + this.gridBounds = new OpenLayers.Bounds( │ │ │ │ │ + center.lon - (gridWidth / 2), │ │ │ │ │ + center.lat - (gridHeight / 2), │ │ │ │ │ + center.lon + (gridWidth / 2), │ │ │ │ │ + center.lat + (gridHeight / 2) │ │ │ │ │ + ); │ │ │ │ │ + var rows = Math.floor(gridHeight / this.dy); │ │ │ │ │ + var cols = Math.floor(gridWidth / this.dx); │ │ │ │ │ + var gridLeft = origin.lon + (this.dx * Math.ceil((this.gridBounds.left - origin.lon) / this.dx)); │ │ │ │ │ + var gridBottom = origin.lat + (this.dy * Math.ceil((this.gridBounds.bottom - origin.lat) / this.dy)); │ │ │ │ │ + var features = new Array(rows * cols); │ │ │ │ │ + var x, y, point; │ │ │ │ │ + for (var i = 0; i < cols; ++i) { │ │ │ │ │ + x = gridLeft + (i * this.dx); │ │ │ │ │ + for (var j = 0; j < rows; ++j) { │ │ │ │ │ + y = gridBottom + (j * this.dy); │ │ │ │ │ + point = new OpenLayers.Geometry.Point(x, y); │ │ │ │ │ + if (this.rotation) { │ │ │ │ │ + point.rotate(this.rotation, rotationOrigin); │ │ │ │ │ + } │ │ │ │ │ + features[(i * rows) + j] = new OpenLayers.Feature.Vector(point); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.destroyFeatures(this.features, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.addFeatures(features, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "មាត្រដ្ឋាន = ១ ៖ ${scaleDenom}" │ │ │ │ │ + /** │ │ │ │ │ + * Method: invalidBounds │ │ │ │ │ + * Determine whether the previously generated point grid is invalid. │ │ │ │ │ + * This occurs when the map bounds extends beyond the previously │ │ │ │ │ + * generated grid bounds. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + invalidBounds: function() { │ │ │ │ │ + return !this.gridBounds || !this.gridBounds.containsBounds(this.getViewBounds()); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.PointGrid" │ │ │ │ │ │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/nds.js │ │ │ │ │ + OpenLayers/Layer/TileCache.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Slomox │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["nds"] │ │ │ │ │ - * Dictionary for Plattdüütsch. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Layer.TileCache │ │ │ │ │ + * A read only TileCache layer. Used to requests tiles cached by TileCache in │ │ │ │ │ + * a web accessible cache. This means that you have to pre-populate your │ │ │ │ │ + * cache before this layer can be used. It is meant only to read tiles │ │ │ │ │ + * created by TileCache, and not to make calls to TileCache for tile │ │ │ │ │ + * creation. Create a new instance with the │ │ │ │ │ + * <OpenLayers.Layer.TileCache> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["nds"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ +OpenLayers.Layer.TileCache = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "Unbehannelt Trüchmellels för de Anfraag ${statusText}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} Treat this layer as a base layer. Default is true. │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Permalink", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: format │ │ │ │ │ + * {String} Mime type of the images returned. Default is image/png. │ │ │ │ │ + */ │ │ │ │ │ + format: 'image/png', │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Overlays", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: serverResolutions │ │ │ │ │ + * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ + * property if the map resolutions differ from the server. This │ │ │ │ │ + * property serves two purposes. (a) <serverResolutions> can include │ │ │ │ │ + * resolutions that the server supports and that you don't want to │ │ │ │ │ + * provide with this layer. (b) The map can work with resolutions │ │ │ │ │ + * that aren't supported by the server, i.e. that aren't in │ │ │ │ │ + * <serverResolutions>. When the map is displayed in such a resolution │ │ │ │ │ + * data for the closest server-supported resolution is loaded and the │ │ │ │ │ + * layer div is stretched as necessary. │ │ │ │ │ + */ │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Achtergrundkoort", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.TileCache │ │ │ │ │ + * Create a new read only TileCache layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} Name of the layer displayed in the interface │ │ │ │ │ + * url - {String} Location of the web accessible cache (not the location of │ │ │ │ │ + * your tilecache script!) │ │ │ │ │ + * layername - {String} Layer name as defined in the TileCache │ │ │ │ │ + * configuration │ │ │ │ │ + * options - {Object} Optional object with properties to be set on the │ │ │ │ │ + * layer. Note that you should speficy your resolutions to match │ │ │ │ │ + * your TileCache configuration. This can be done by setting │ │ │ │ │ + * the resolutions array directly (here or on the map), by setting │ │ │ │ │ + * maxResolution and numZoomLevels, or by using scale based properties. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, layername, options) { │ │ │ │ │ + this.layername = layername; │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, │ │ │ │ │ + [name, url, {}, options]); │ │ │ │ │ + this.extension = this.format.split('/')[1].toLowerCase(); │ │ │ │ │ + this.extension = (this.extension == 'jpg') ? 'jpeg' : this.extension; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'noFID': "En Feature, dat keen FID hett, kann nich aktuell maakt warrn.", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * obj - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.TileCache>} An exact clone of this │ │ │ │ │ + * <OpenLayers.Layer.TileCache> │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Dien Browser ünnerstütt keen Vektorbiller. Ünnerstütt Renderers:\n${renderers}", │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.TileCache(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.layername, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "WFS-Transakschoon: hett klappt ${response}", │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "WFS-Transakschoon: hett nich klappt ${response}", │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Skaal = 1 : ${scaleDenom}", │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Disse Methood is oold un schall dat in 3.0 nich mehr geven. Bruuk dor man beter ${newMethod} för." │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as parameters. │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var bbox = this.maxExtent; │ │ │ │ │ + var size = this.tileSize; │ │ │ │ │ + var tileX = Math.round((bounds.left - bbox.left) / (res * size.w)); │ │ │ │ │ + var tileY = Math.round((bounds.bottom - bbox.bottom) / (res * size.h)); │ │ │ │ │ + var tileZ = this.serverResolutions != null ? │ │ │ │ │ + OpenLayers.Util.indexOf(this.serverResolutions, res) : │ │ │ │ │ + this.map.getZoom(); │ │ │ │ │ + │ │ │ │ │ + var components = [ │ │ │ │ │ + this.layername, │ │ │ │ │ + OpenLayers.Number.zeroPad(tileZ, 2), │ │ │ │ │ + OpenLayers.Number.zeroPad(parseInt(tileX / 1000000), 3), │ │ │ │ │ + OpenLayers.Number.zeroPad((parseInt(tileX / 1000) % 1000), 3), │ │ │ │ │ + OpenLayers.Number.zeroPad((parseInt(tileX) % 1000), 3), │ │ │ │ │ + OpenLayers.Number.zeroPad(parseInt(tileY / 1000000), 3), │ │ │ │ │ + OpenLayers.Number.zeroPad((parseInt(tileY / 1000) % 1000), 3), │ │ │ │ │ + OpenLayers.Number.zeroPad((parseInt(tileY) % 1000), 3) + '.' + this.extension │ │ │ │ │ + ]; │ │ │ │ │ + var path = components.join('/'); │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(path, url); │ │ │ │ │ + } │ │ │ │ │ + url = (url.charAt(url.length - 1) == '/') ? url : url + '/'; │ │ │ │ │ + return url + path; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.TileCache" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/zh-TW.js │ │ │ │ │ + OpenLayers/Layer/WMTS.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["zh-TW"] │ │ │ │ │ - * Dictionary for Traditional Chinese. (Used Mainly in Taiwan) │ │ │ │ │ - * Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Layer.WMTS │ │ │ │ │ + * Instances of the WMTS class allow viewing of tiles from a service that │ │ │ │ │ + * implements the OGC WMTS specification version 1.0.0. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["zh-TW"] = { │ │ │ │ │ +OpenLayers.Layer.WMTS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "未處理的請求,傳回值為 ${statusText}。", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} The layer will be considered a base layer. Default is true. │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ - 'Permalink': "永久連結", │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {String} WMTS version. Default is "1.0.0". │ │ │ │ │ + */ │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ │ │ │ │ │ - 'Overlays': "額外圖層", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: requestEncoding │ │ │ │ │ + * {String} Request encoding. Can be "REST" or "KVP". Default is "KVP". │ │ │ │ │ + */ │ │ │ │ │ + requestEncoding: "KVP", │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "基礎圖層", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String|Array(String)} The base URL or request URL template for the WMTS │ │ │ │ │ + * service. Must be provided. Array is only supported for base URLs, not │ │ │ │ │ + * for request URL templates. URL templates are only supported for │ │ │ │ │ + * REST <requestEncoding>. │ │ │ │ │ + */ │ │ │ │ │ + url: null, │ │ │ │ │ │ │ │ │ │ - 'noFID': "因為沒有 FID 所以無法更新 feature。", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layer │ │ │ │ │ + * {String} The layer identifier advertised by the WMTS service. Must be │ │ │ │ │ + * provided. │ │ │ │ │ + */ │ │ │ │ │ + layer: null, │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "您的瀏覽器未支援向量渲染. 目前支援的渲染方式是:\n${renderers}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: matrixSet │ │ │ │ │ + * {String} One of the advertised matrix set identifiers. Must be provided. │ │ │ │ │ + */ │ │ │ │ │ + matrixSet: null, │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'minZoomLevelError': "minZoomLevel 屬性僅適合用在 " + │ │ │ │ │ - "FixedZoomLevels-descendent 類型的圖層. 這個" + │ │ │ │ │ - "wfs layer 的 minZoomLevel 是過去所遺留下來的," + │ │ │ │ │ - "然而我們不能移除它而不讓它將" + │ │ │ │ │ - "過去的程式相容性給破壞掉。" + │ │ │ │ │ - "因此我們將會迴避使用它 -- minZoomLevel " + │ │ │ │ │ - "會在3.0被移除,請改" + │ │ │ │ │ - "用在這邊描述的 min/max resolution 設定: " + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: style │ │ │ │ │ + * {String} One of the advertised layer styles. Must be provided. │ │ │ │ │ + */ │ │ │ │ │ + style: null, │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "WFS Transaction: 成功 ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: format │ │ │ │ │ + * {String} The image MIME type. Default is "image/jpeg". │ │ │ │ │ + */ │ │ │ │ │ + format: "image/jpeg", │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "WFS Transaction: 失敗 ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: tileOrigin │ │ │ │ │ + * {<OpenLayers.LonLat>} The top-left corner of the tile matrix in map │ │ │ │ │ + * units. If the tile origin for each matrix in a set is different, │ │ │ │ │ + * the <matrixIds> should include a topLeftCorner property. If │ │ │ │ │ + * not provided, the tile origin will default to the top left corner │ │ │ │ │ + * of the layer <maxExtent>. │ │ │ │ │ + */ │ │ │ │ │ + tileOrigin: null, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "The Google Layer 圖層無法被正確的載入。<br><br>" + │ │ │ │ │ - "要迴避這個訊息, 請在右上角的圖層改變器裡," + │ │ │ │ │ - "選一個新的基礎圖層。<br><br>" + │ │ │ │ │ - "很有可能是因為 Google Maps 的函式庫" + │ │ │ │ │ - "腳本沒有被正確的置入,或沒有包含 " + │ │ │ │ │ - "您網站上正確的 API key <br><br>" + │ │ │ │ │ - "開發者: 要幫助這個行為正確完成," + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ - "target='_blank'>請按這裡</a>", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: tileFullExtent │ │ │ │ │ + * {<OpenLayers.Bounds>} The full extent of the tile set. If not supplied, │ │ │ │ │ + * the layer's <maxExtent> property will be used. │ │ │ │ │ + */ │ │ │ │ │ + tileFullExtent: null, │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "${layerType} 圖層無法被正確的載入。<br><br>" + │ │ │ │ │ - "要迴避這個訊息, 請在右上角的圖層改變器裡," + │ │ │ │ │ - "選一個新的基礎圖層。<br><br>" + │ │ │ │ │ - "很有可能是因為 ${layerLib} 的函式庫" + │ │ │ │ │ - "腳本沒有被正確的置入。<br><br>" + │ │ │ │ │ - "開發者: 要幫助這個行為正確完成," + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ - "target='_blank'>請按這裡</a>", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: formatSuffix │ │ │ │ │ + * {String} For REST request encoding, an image format suffix must be │ │ │ │ │ + * included in the request. If not provided, the suffix will be derived │ │ │ │ │ + * from the <format> property. │ │ │ │ │ + */ │ │ │ │ │ + formatSuffix: null, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Scale = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: matrixIds │ │ │ │ │ + * {Array} A list of tile matrix identifiers. If not provided, the matrix │ │ │ │ │ + * identifiers will be assumed to be integers corresponding to the │ │ │ │ │ + * map zoom level. If a list of strings is provided, each item should │ │ │ │ │ + * be the matrix identifier that corresponds to the map zoom level. │ │ │ │ │ + * Additionally, a list of objects can be provided. Each object should │ │ │ │ │ + * describe the matrix as presented in the WMTS capabilities. These │ │ │ │ │ + * objects should have the propertes shown below. │ │ │ │ │ + * │ │ │ │ │ + * Matrix properties: │ │ │ │ │ + * identifier - {String} The matrix identifier (required). │ │ │ │ │ + * scaleDenominator - {Number} The matrix scale denominator. │ │ │ │ │ + * topLeftCorner - {<OpenLayers.LonLat>} The top left corner of the │ │ │ │ │ + * matrix. Must be provided if different than the layer <tileOrigin>. │ │ │ │ │ + * tileWidth - {Number} The tile width for the matrix. Must be provided │ │ │ │ │ + * if different than the width given in the layer <tileSize>. │ │ │ │ │ + * tileHeight - {Number} The tile height for the matrix. Must be provided │ │ │ │ │ + * if different than the height given in the layer <tileSize>. │ │ │ │ │ + */ │ │ │ │ │ + matrixIds: null, │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'reprojectDeprecated': "你正使用 'reproject' 這個選項 " + │ │ │ │ │ - "在 ${layerName} 層。這個選項已經不再使用:" + │ │ │ │ │ - "它的使用原本是設計用來支援在商業地圖上秀出資料," + │ │ │ │ │ - "但這個功能已經被" + │ │ │ │ │ - "Spherical Mercator所取代。更多的資訊可以在 " + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SphericalMercator 找到。", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: dimensions │ │ │ │ │ + * {Array} For RESTful request encoding, extra dimensions may be specified. │ │ │ │ │ + * Items in this list should be property names in the <params> object. │ │ │ │ │ + * Values of extra dimensions will be determined from the corresponding │ │ │ │ │ + * values in the <params> object. │ │ │ │ │ + */ │ │ │ │ │ + dimensions: null, │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'methodDeprecated': "這個方法已經不再使用且在3.0將會被移除," + │ │ │ │ │ - "請使用 ${newMethod} 來代替。", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: params │ │ │ │ │ + * {Object} Extra parameters to include in tile requests. For KVP │ │ │ │ │ + * <requestEncoding>, these properties will be encoded in the request │ │ │ │ │ + * query string. For REST <requestEncoding>, these properties will │ │ │ │ │ + * become part of the request path, with order determined by the │ │ │ │ │ + * <dimensions> list. │ │ │ │ │ + */ │ │ │ │ │ + params: null, │ │ │ │ │ │ │ │ │ │ - 'end': '' │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/ja.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomOffset │ │ │ │ │ + * {Number} If your cache has more levels than you want to provide │ │ │ │ │ + * access to with this layer, supply a zoomOffset. This zoom offset │ │ │ │ │ + * is added to the current map zoom level to determine the level │ │ │ │ │ + * for a requested tile. For example, if you supply a zoomOffset │ │ │ │ │ + * of 3, when the map is at the zoom 0, tiles will be requested from │ │ │ │ │ + * level 3 of your cache. Default is 0 (assumes cache level and map │ │ │ │ │ + * zoom are equivalent). Additionally, if this layer is to be used │ │ │ │ │ + * as an overlay and the cache has fewer zoom levels than the base │ │ │ │ │ + * layer, you can supply a negative zoomOffset. For example, if a │ │ │ │ │ + * map zoom level of 1 corresponds to your cache level zero, you would │ │ │ │ │ + * supply a -1 zoomOffset (and set the maxResolution of the layer │ │ │ │ │ + * appropriately). The zoomOffset value has no effect if complete │ │ │ │ │ + * matrix definitions (including scaleDenominator) are supplied in │ │ │ │ │ + * the <matrixIds> property. Defaults to 0 (no zoom offset). │ │ │ │ │ + */ │ │ │ │ │ + zoomOffset: 0, │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Fryed-peach │ │ │ │ │ - * - Mage Whopper │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: serverResolutions │ │ │ │ │ + * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ + * property if the map resolutions differ from the server. This │ │ │ │ │ + * property serves two purposes. (a) <serverResolutions> can include │ │ │ │ │ + * resolutions that the server supports and that you don't want to │ │ │ │ │ + * provide with this layer; you can also look at <zoomOffset>, which is │ │ │ │ │ + * an alternative to <serverResolutions> for that specific purpose. │ │ │ │ │ + * (b) The map can work with resolutions that aren't supported by │ │ │ │ │ + * the server, i.e. that aren't in <serverResolutions>. When the │ │ │ │ │ + * map is displayed in such a resolution data for the closest │ │ │ │ │ + * server-supported resolution is loaded and the layer div is │ │ │ │ │ + * stretched as necessary. │ │ │ │ │ + */ │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: formatSuffixMap │ │ │ │ │ + * {Object} a map between WMTS 'format' request parameter and tile image file suffix │ │ │ │ │ + */ │ │ │ │ │ + formatSuffixMap: { │ │ │ │ │ + "image/png": "png", │ │ │ │ │ + "image/png8": "png", │ │ │ │ │ + "image/png24": "png", │ │ │ │ │ + "image/png32": "png", │ │ │ │ │ + "png": "png", │ │ │ │ │ + "image/jpeg": "jpg", │ │ │ │ │ + "image/jpg": "jpg", │ │ │ │ │ + "jpeg": "jpg", │ │ │ │ │ + "jpg": "jpg" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["ja"] │ │ │ │ │ - * Dictionary for 日本語. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Lang["ja"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + /** │ │ │ │ │ + * Property: matrix │ │ │ │ │ + * {Object} Matrix definition for the current map resolution. Updated by │ │ │ │ │ + * the <updateMatrixProperties> method. │ │ │ │ │ + */ │ │ │ │ │ + matrix: null, │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "未処理の要求は ${statusText} を返します", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.WMTS │ │ │ │ │ + * Create a new WMTS layer. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var wmts = new OpenLayers.Layer.WMTS({ │ │ │ │ │ + * name: "My WMTS Layer", │ │ │ │ │ + * url: "http://example.com/wmts", │ │ │ │ │ + * layer: "layer_id", │ │ │ │ │ + * style: "default", │ │ │ │ │ + * matrixSet: "matrix_id" │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Configuration properties for the layer. │ │ │ │ │ + * │ │ │ │ │ + * Required configuration properties: │ │ │ │ │ + * url - {String} The base url for the service. See the <url> property. │ │ │ │ │ + * layer - {String} The layer identifier. See the <layer> property. │ │ │ │ │ + * style - {String} The layer style identifier. See the <style> property. │ │ │ │ │ + * matrixSet - {String} The tile matrix set identifier. See the <matrixSet> │ │ │ │ │ + * property. │ │ │ │ │ + * │ │ │ │ │ + * Any other documented layer properties can be provided in the config object. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ │ │ │ │ │ - 'Permalink': "パーマリンク", │ │ │ │ │ + // confirm required properties are supplied │ │ │ │ │ + var required = { │ │ │ │ │ + url: true, │ │ │ │ │ + layer: true, │ │ │ │ │ + style: true, │ │ │ │ │ + matrixSet: true │ │ │ │ │ + }; │ │ │ │ │ + for (var prop in required) { │ │ │ │ │ + if (!(prop in config)) { │ │ │ │ │ + throw new Error("Missing property '" + prop + "' in layer configuration."); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'Overlays': "オーバーレイ", │ │ │ │ │ + config.params = OpenLayers.Util.upperCaseObject(config.params); │ │ │ │ │ + var args = [config.name, config.url, config.params, config]; │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, args); │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "基底レイヤー", │ │ │ │ │ │ │ │ │ │ - 'noFID': "FID のない地物は更新できません。", │ │ │ │ │ + // determine format suffix (for REST) │ │ │ │ │ + if (!this.formatSuffix) { │ │ │ │ │ + this.formatSuffix = this.formatSuffixMap[this.format] || this.format.split("/").pop(); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "あなたのブラウザはベクターグラフィックスの描写に対応していません。現時点で対応しているソフトウェアは以下のものです。\n${renderers}", │ │ │ │ │ + // expand matrixIds (may be array of string or array of object) │ │ │ │ │ + if (this.matrixIds) { │ │ │ │ │ + var len = this.matrixIds.length; │ │ │ │ │ + if (len && typeof this.matrixIds[0] === "string") { │ │ │ │ │ + var ids = this.matrixIds; │ │ │ │ │ + this.matrixIds = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + this.matrixIds[i] = { │ │ │ │ │ + identifier: ids[i] │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'minZoomLevelError': "minZoomLevel プロパティは FixedZoomLevels を継承するレイヤーでの使用のみを想定しています。この minZoomLevel に対する WFS レイヤーの検査は歴史的なものです。しかしながら、この検査を除去するとそれに依存する OpenLayers ベースのアプリケーションを破壊してしまう可能性があります。よって廃止が予定されており、この minZoomLevel 検査はバージョン3.0で除去されます。代わりに、http://trac.openlayers.org/wiki/SettingZoomLevels で解説されている、最小および最大解像度設定を使用してください。", │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "WFS トランザクション: 成功 ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + */ │ │ │ │ │ + setMap: function() { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "WFS トランザクション: 失敗 ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateMatrixProperties │ │ │ │ │ + * Called when map resolution changes to update matrix related properties. │ │ │ │ │ + */ │ │ │ │ │ + updateMatrixProperties: function() { │ │ │ │ │ + this.matrix = this.getMatrix(); │ │ │ │ │ + if (this.matrix) { │ │ │ │ │ + if (this.matrix.topLeftCorner) { │ │ │ │ │ + this.tileOrigin = this.matrix.topLeftCorner; │ │ │ │ │ + } │ │ │ │ │ + if (this.matrix.tileWidth && this.matrix.tileHeight) { │ │ │ │ │ + this.tileSize = new OpenLayers.Size( │ │ │ │ │ + this.matrix.tileWidth, this.matrix.tileHeight │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (!this.tileOrigin) { │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat( │ │ │ │ │ + this.maxExtent.left, this.maxExtent.top │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (!this.tileFullExtent) { │ │ │ │ │ + this.tileFullExtent = this.maxExtent; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Google レイヤーが正しく読み込みを行えませんでした。\x3cbr\x3e\x3cbr\x3eこのメッセージを消すには、右上の隅にあるレイヤー切り替え部分で新しい基底レイヤーを選んでください。\x3cbr\x3e\x3cbr\x3eおそらく、これは Google マップ用ライブラリのスクリプトが組み込まれていないか、あなたのサイトに対応する正しい API キーが設定されていないためです。\x3cbr\x3e\x3cbr\x3e開発者の方へ: 正しい動作をさせるために\x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eこちらのウィキ\x3c/a\x3eを参照してください。", │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * zoomChanged - {Boolean} Tells when zoom has changed, as layers have to │ │ │ │ │ + * do some init work in that case. │ │ │ │ │ + * dragging - {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + if (zoomChanged || !this.matrix) { │ │ │ │ │ + this.updateMatrixProperties(); │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "${layerType} レイヤーが正しく読み込みを行えませんでした。\x3cbr\x3e\x3cbr\x3eこのメッセージを消すには、右上の隅にあるレイヤー切り替え部分で新しい基底レイヤーを選んでください。\x3cbr\x3e\x3cbr\x3eおそらく、これは ${layerLib} ライブラリのスクリプトが正しく組み込まれていないためです。\x3cbr\x3e\x3cbr\x3e開発者の方へ: 正しい動作をさせるために\x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eこちらのウィキ\x3c/a\x3eを参照してください。", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.WMTS>} An exact clone of this <OpenLayers.Layer.WMTS> │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.WMTS(this.options); │ │ │ │ │ + } │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "縮尺 = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getIdentifier │ │ │ │ │ + * Get the current index in the matrixIds array. │ │ │ │ │ + */ │ │ │ │ │ + getIdentifier: function() { │ │ │ │ │ + return this.getServerZoom(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'W': "西", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getMatrix │ │ │ │ │ + * Get the appropriate matrix definition for the current map resolution. │ │ │ │ │ + */ │ │ │ │ │ + getMatrix: function() { │ │ │ │ │ + var matrix; │ │ │ │ │ + if (!this.matrixIds || this.matrixIds.length === 0) { │ │ │ │ │ + matrix = { │ │ │ │ │ + identifier: this.getIdentifier() │ │ │ │ │ + }; │ │ │ │ │ + } else { │ │ │ │ │ + // get appropriate matrix given the map scale if possible │ │ │ │ │ + if ("scaleDenominator" in this.matrixIds[0]) { │ │ │ │ │ + // scale denominator calculation based on WMTS spec │ │ │ │ │ + var denom = │ │ │ │ │ + OpenLayers.METERS_PER_INCH * │ │ │ │ │ + OpenLayers.INCHES_PER_UNIT[this.units] * │ │ │ │ │ + this.getServerResolution() / 0.28E-3; │ │ │ │ │ + var diff = Number.POSITIVE_INFINITY; │ │ │ │ │ + var delta; │ │ │ │ │ + for (var i = 0, ii = this.matrixIds.length; i < ii; ++i) { │ │ │ │ │ + delta = Math.abs(1 - (this.matrixIds[i].scaleDenominator / denom)); │ │ │ │ │ + if (delta < diff) { │ │ │ │ │ + diff = delta; │ │ │ │ │ + matrix = this.matrixIds[i]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + // fall back on zoom as index │ │ │ │ │ + matrix = this.matrixIds[this.getIdentifier()]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return matrix; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'E': "東", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getTileInfo │ │ │ │ │ + * Get tile information for a given location at the current map resolution. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * loc - {<OpenLayers.LonLat} A location in map coordinates. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} An object with "col", "row", "i", and "j" properties. The col │ │ │ │ │ + * and row values are zero based tile indexes from the top left. The │ │ │ │ │ + * i and j values are the number of pixels to the left and top │ │ │ │ │ + * (respectively) of the given location within the target tile. │ │ │ │ │ + */ │ │ │ │ │ + getTileInfo: function(loc) { │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ │ │ │ │ │ - 'N': "北", │ │ │ │ │ + var fx = (loc.lon - this.tileOrigin.lon) / (res * this.tileSize.w); │ │ │ │ │ + var fy = (this.tileOrigin.lat - loc.lat) / (res * this.tileSize.h); │ │ │ │ │ │ │ │ │ │ - 'S': "南", │ │ │ │ │ + var col = Math.floor(fx); │ │ │ │ │ + var row = Math.floor(fy); │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "あなたは「${layerName}」レイヤーで reproject オプションを使っています。このオプションは商用の基底地図上に情報を表示する目的で設計されましたが、現在ではその機能は Spherical Mercator サポートを利用して実現されており、このオプションの使用は非推奨です。追加の情報は http://trac.openlayers.org/wiki/SphericalMercator で入手できます。", │ │ │ │ │ + return { │ │ │ │ │ + col: col, │ │ │ │ │ + row: row, │ │ │ │ │ + i: Math.floor((fx - col) * this.tileSize.w), │ │ │ │ │ + j: Math.floor((fy - row) * this.tileSize.h) │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "このメソッドは廃止が予定されており、バージョン3.0で除去されます。代わりに ${newMethod} を使用してください。" │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A URL for the tile corresponding to the given bounds. │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var url = ""; │ │ │ │ │ + if (!this.tileFullExtent || this.tileFullExtent.intersectsBounds(bounds)) { │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/bg.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var info = this.getTileInfo(center); │ │ │ │ │ + var matrixId = this.matrix.identifier; │ │ │ │ │ + var dimensions = this.dimensions, │ │ │ │ │ + params; │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - DCLXVI │ │ │ │ │ - */ │ │ │ │ │ + if (OpenLayers.Util.isArray(this.url)) { │ │ │ │ │ + url = this.selectUrl([ │ │ │ │ │ + this.version, this.style, this.matrixSet, │ │ │ │ │ + this.matrix.identifier, info.row, info.col │ │ │ │ │ + ].join(","), this.url); │ │ │ │ │ + } else { │ │ │ │ │ + url = this.url; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ + if (this.requestEncoding.toUpperCase() === "REST") { │ │ │ │ │ + params = this.params; │ │ │ │ │ + if (url.indexOf("{") !== -1) { │ │ │ │ │ + var template = url.replace(/\{/g, "${"); │ │ │ │ │ + var context = { │ │ │ │ │ + // spec does not make clear if capital S or not │ │ │ │ │ + style: this.style, │ │ │ │ │ + Style: this.style, │ │ │ │ │ + TileMatrixSet: this.matrixSet, │ │ │ │ │ + TileMatrix: this.matrix.identifier, │ │ │ │ │ + TileRow: info.row, │ │ │ │ │ + TileCol: info.col │ │ │ │ │ + }; │ │ │ │ │ + if (dimensions) { │ │ │ │ │ + var dimension, i; │ │ │ │ │ + for (i = dimensions.length - 1; i >= 0; --i) { │ │ │ │ │ + dimension = dimensions[i]; │ │ │ │ │ + context[dimension] = params[dimension.toUpperCase()]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + url = OpenLayers.String.format(template, context); │ │ │ │ │ + } else { │ │ │ │ │ + // include 'version', 'layer' and 'style' in tile resource url │ │ │ │ │ + var path = this.version + "/" + this.layer + "/" + this.style + "/"; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["bg"] │ │ │ │ │ - * Dictionary for Български. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Lang["bg"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + // append optional dimension path elements │ │ │ │ │ + if (dimensions) { │ │ │ │ │ + for (var i = 0; i < dimensions.length; i++) { │ │ │ │ │ + if (params[dimensions[i]]) { │ │ │ │ │ + path = path + params[dimensions[i]] + "/"; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Постоянна препратка", │ │ │ │ │ + // append other required path elements │ │ │ │ │ + path = path + this.matrixSet + "/" + this.matrix.identifier + │ │ │ │ │ + "/" + info.row + "/" + info.col + "." + this.formatSuffix; │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Основен слой", │ │ │ │ │ + if (!url.match(/\/$/)) { │ │ │ │ │ + url = url + "/"; │ │ │ │ │ + } │ │ │ │ │ + url = url + path; │ │ │ │ │ + } │ │ │ │ │ + } else if (this.requestEncoding.toUpperCase() === "KVP") { │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Мащаб = 1 : ${scaleDenom}", │ │ │ │ │ + // assemble all required parameters │ │ │ │ │ + params = { │ │ │ │ │ + SERVICE: "WMTS", │ │ │ │ │ + REQUEST: "GetTile", │ │ │ │ │ + VERSION: this.version, │ │ │ │ │ + LAYER: this.layer, │ │ │ │ │ + STYLE: this.style, │ │ │ │ │ + TILEMATRIXSET: this.matrixSet, │ │ │ │ │ + TILEMATRIX: this.matrix.identifier, │ │ │ │ │ + TILEROW: info.row, │ │ │ │ │ + TILECOL: info.col, │ │ │ │ │ + FORMAT: this.format │ │ │ │ │ + }; │ │ │ │ │ + url = OpenLayers.Layer.Grid.prototype.getFullRequestString.apply(this, [params]); │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Този метод е остарял и ще бъде премахват в 3.0. Вместо него използвайте ${newMethod}." │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return url; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: mergeNewParams │ │ │ │ │ + * Extend the existing layer <params> with new properties. Tiles will be │ │ │ │ │ + * reloaded with updated params in the request. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newParams - {Object} Properties to extend to existing <params>. │ │ │ │ │ + */ │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + if (this.requestEncoding.toUpperCase() === "KVP") { │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply( │ │ │ │ │ + this, [OpenLayers.Util.upperCaseObject(newParams)] │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.WMTS" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/be-tarask.js │ │ │ │ │ + OpenLayers/Layer/PointTrack.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - EugeneZelenko │ │ │ │ │ - * - Jim-by │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Layer/Vector.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["be-tarask"] │ │ │ │ │ - * Dictionary for Беларуская (тарашкевіца). Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Layer.PointTrack │ │ │ │ │ + * Vector layer to display ordered point features as a line, creating one │ │ │ │ │ + * LineString feature for each pair of two points. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Vector> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["be-tarask"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - │ │ │ │ │ - 'unhandledRequest': "Неапрацаваны вынік запыту ${statusText}", │ │ │ │ │ - │ │ │ │ │ - 'Permalink': "Сталая спасылка", │ │ │ │ │ - │ │ │ │ │ - 'Overlays': "Слаі", │ │ │ │ │ - │ │ │ │ │ - 'Base Layer': "Базавы слой", │ │ │ │ │ - │ │ │ │ │ - 'noFID': "Немагчыма абнавіць магчымасьць, для якога не існуе FID.", │ │ │ │ │ - │ │ │ │ │ - 'browserNotSupported': "Ваш браўзэр не падтрымлівае вэктарную графіку. У цяперашні момант падтрымліваюцца: ${renderers}", │ │ │ │ │ - │ │ │ │ │ - 'minZoomLevelError': "Уласьцівасьць minZoomLevel прызначана толькі для выкарыстаньня са слаямі вытворнымі ад FixedZoomLevels. Тое, што гэты wfs-слой правяраецца на minZoomLevel — рэха прошлага. Але мы ня можам выдаліць гэтую магчымасьць, таму што ад яе залежаць некаторыя заснаваныя на OL дастасаваньні. Тым ня менш, праверка minZoomLevel будзе выдаленая ў вэрсіі 3.0. Калі ласка, выкарыстоўваеце замест яе ўстаноўкі мінімальнага/максымальнага памераў, як апісана тут: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ +OpenLayers.Layer.PointTrack = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "WFS-транзакцыя: ПОСЬПЕХ ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: dataFrom │ │ │ │ │ + * {<OpenLayers.Layer.PointTrack.TARGET_NODE>} or │ │ │ │ │ + * {<OpenLayers.Layer.PointTrack.SOURCE_NODE>} optional. If the lines │ │ │ │ │ + * should get the data/attributes from one of the two points it is │ │ │ │ │ + * composed of, which one should it be? │ │ │ │ │ + */ │ │ │ │ │ + dataFrom: null, │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "WFS-транзакцыя: ПАМЫЛКА ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: styleFrom │ │ │ │ │ + * {<OpenLayers.Layer.PointTrack.TARGET_NODE>} or │ │ │ │ │ + * {<OpenLayers.Layer.PointTrack.SOURCE_NODE>} optional. If the lines │ │ │ │ │ + * should get the style from one of the two points it is composed of, │ │ │ │ │ + * which one should it be? │ │ │ │ │ + */ │ │ │ │ │ + styleFrom: null, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Не атрымалася загрузіць слой Google. \x3cbr\x3e\x3cbr\x3eКаб пазбавіцца гэтага паведамленьня, выберыце новы базавы слой у сьпісе ў верхнім правым куце.\x3cbr\x3e\x3cbr\x3e Хутчэй за ўсё, прычына ў тым, што скрыпт бібліятэкі Google Maps ня быў уключаныя альбо не ўтрымлівае слушны API-ключ для Вашага сайта.\x3cbr\x3e\x3cbr\x3eРаспрацоўшчыкам: Для таго, каб даведацца як зрабіць так, каб усё працавала, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eнацісьніце тут\x3c/a\x3e", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.PointTrack │ │ │ │ │ + * Constructor for a new OpenLayers.PointTrack instance. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} name of the layer │ │ │ │ │ + * options - {Object} Optional object with properties to tag onto the │ │ │ │ │ + * instance. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "Немагчыма загрузіць слой ${layerType}.\x3cbr\x3e\x3cbr\x3eКаб пазбавіцца гэтага паведамленьня, выберыце новы базавы слой у сьпісе ў верхнім правым куце.\x3cbr\x3e\x3cbr\x3eХутчэй за ўсё, прычына ў тым, што скрыпт бібліятэкі ${layerLib} ня быў слушна ўключаны.\x3cbr\x3e\x3cbr\x3eРаспрацоўшчыкам: Для таго, каб даведацца як зрабіць так, каб усё працавала, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eнацісьніце тут\x3c/a\x3e", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: addNodes │ │ │ │ │ + * Adds point features that will be used to create lines from, using point │ │ │ │ │ + * pairs. The first point of a pair will be the source node, the second │ │ │ │ │ + * will be the target node. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * pointFeatures - {Array(<OpenLayers.Feature>)} │ │ │ │ │ + * options - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Supported options: │ │ │ │ │ + * silent - {Boolean} true to suppress (before)feature(s)added events │ │ │ │ │ + */ │ │ │ │ │ + addNodes: function(pointFeatures, options) { │ │ │ │ │ + if (pointFeatures.length < 2) { │ │ │ │ │ + throw new Error("At least two point features have to be added to " + │ │ │ │ │ + "create a line from"); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Маштаб = 1 : ${scaleDenom}", │ │ │ │ │ + var lines = new Array(pointFeatures.length - 1); │ │ │ │ │ │ │ │ │ │ - 'W': "З", │ │ │ │ │ + var pointFeature, startPoint, endPoint; │ │ │ │ │ + for (var i = 0, len = pointFeatures.length; i < len; i++) { │ │ │ │ │ + pointFeature = pointFeatures[i]; │ │ │ │ │ + endPoint = pointFeature.geometry; │ │ │ │ │ │ │ │ │ │ - 'E': "У", │ │ │ │ │ + if (!endPoint) { │ │ │ │ │ + var lonlat = pointFeature.lonlat; │ │ │ │ │ + endPoint = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ + } else if (endPoint.CLASS_NAME != "OpenLayers.Geometry.Point") { │ │ │ │ │ + throw new TypeError("Only features with point geometries are supported."); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'N': "Пн", │ │ │ │ │ + if (i > 0) { │ │ │ │ │ + var attributes = (this.dataFrom != null) ? │ │ │ │ │ + (pointFeatures[i + this.dataFrom].data || │ │ │ │ │ + pointFeatures[i + this.dataFrom].attributes) : │ │ │ │ │ + null; │ │ │ │ │ + var style = (this.styleFrom != null) ? │ │ │ │ │ + (pointFeatures[i + this.styleFrom].style) : │ │ │ │ │ + null; │ │ │ │ │ + var line = new OpenLayers.Geometry.LineString([startPoint, │ │ │ │ │ + endPoint │ │ │ │ │ + ]); │ │ │ │ │ │ │ │ │ │ - 'S': "Пд", │ │ │ │ │ + lines[i - 1] = new OpenLayers.Feature.Vector(line, attributes, │ │ │ │ │ + style); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Вы выкарыстоўваеце ўстаноўку \'reproject\' для слоя ${layerName}. Гэтая ўстаноўка зьяўляецца састарэлай: яна выкарыстоўвалася для падтрымкі паказу зьвестак на камэрцыйных базавых мапах, але гэта функцыя цяпер рэалізаваная ў убудаванай падтрымцы сфэрычнай праекцыі Мэркатара. Дадатковая інфармацыя ёсьць на http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + startPoint = endPoint; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Гэты мэтад састарэлы і будзе выдалены ў вэрсіі 3.0. Калі ласка, замест яго выкарыстоўвайце ${newMethod}." │ │ │ │ │ + this.addFeatures(lines, options); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.PointTrack" │ │ │ │ │ }); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/ca.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * Constant: OpenLayers.Layer.PointTrack.SOURCE_NODE │ │ │ │ │ + * {Number} value for <OpenLayers.Layer.PointTrack.dataFrom> and │ │ │ │ │ + * <OpenLayers.Layer.PointTrack.styleFrom> │ │ │ │ │ */ │ │ │ │ │ +OpenLayers.Layer.PointTrack.SOURCE_NODE = -1; │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["ca"] │ │ │ │ │ - * Dictionary for Catalan, UTF8 encoding. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Constant: OpenLayers.Layer.PointTrack.TARGET_NODE │ │ │ │ │ + * {Number} value for <OpenLayers.Layer.PointTrack.dataFrom> and │ │ │ │ │ + * <OpenLayers.Layer.PointTrack.styleFrom> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang.ca = { │ │ │ │ │ - │ │ │ │ │ - 'unhandledRequest': "Resposta a petició no gestionada ${statusText}", │ │ │ │ │ - │ │ │ │ │ - 'Permalink': "Enllaç permanent", │ │ │ │ │ +OpenLayers.Layer.PointTrack.TARGET_NODE = 0; │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Capes addicionals", │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Layer.PointTrack.dataFrom │ │ │ │ │ + * {Object} with the following keys - *deprecated* │ │ │ │ │ + * - SOURCE_NODE: take data/attributes from the source node of the line │ │ │ │ │ + * - TARGET_NODE: take data/attributes from the target node of the line │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.PointTrack.dataFrom = { │ │ │ │ │ + 'SOURCE_NODE': -1, │ │ │ │ │ + 'TARGET_NODE': 0 │ │ │ │ │ +}; │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/Zoomify.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Capa Base", │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ - 'noFID': "No es pot actualitzar un element per al que no existeix FID.", │ │ │ │ │ +/* │ │ │ │ │ + * Development supported by a R&D grant DC08P02OUK006 - Old Maps Online │ │ │ │ │ + * (www.oldmapsonline.org) from Ministry of Culture of the Czech Republic. │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "El seu navegador no suporta renderització vectorial. Els renderitzadors suportats actualment són:\n${renderers}", │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'minZoomLevelError': "La propietat minZoomLevel s'ha d'utilitzar només " + │ │ │ │ │ - "amb les capes que tenen FixedZoomLevels. El fet que " + │ │ │ │ │ - "una capa wfs comprovi minZoomLevel és una relíquia del " + │ │ │ │ │ - "passat. No podem, però, eliminar-la sense trencar " + │ │ │ │ │ - "les aplicacions d'OpenLayers que en puguin dependre. " + │ │ │ │ │ - "Així doncs estem fent-la obsoleta -- la comprovació " + │ │ │ │ │ - "minZoomLevel s'eliminarà a la versió 3.0. Feu servir " + │ │ │ │ │ - "els paràmetres min/max resolution en substitució, tal com es descriu aquí: " + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "Transacció WFS: CORRECTA ${response}", │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.Zoomify │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Zoomify = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "Transacció WFS: HA FALLAT ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Property: size │ │ │ │ │ + * {<OpenLayers.Size>} The Zoomify image size in pixels. │ │ │ │ │ + */ │ │ │ │ │ + size: null, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "La capa Google no s'ha pogut carregar correctament.<br><br>" + │ │ │ │ │ - "Per evitar aquest missatge, seleccioneu una nova Capa Base " + │ │ │ │ │ - "al gestor de capes de la cantonada superior dreta.<br><br>" + │ │ │ │ │ - "Probablement això és degut a que l'script de la biblioteca de " + │ │ │ │ │ - "Google Maps no ha estat inclòs a la vostra pàgina, o no " + │ │ │ │ │ - "conté la clau de l'API correcta per a la vostra adreça.<br><br>" + │ │ │ │ │ - "Desenvolupadors: Per obtenir consells sobre com fer anar això, " + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ - "target='_blank'>féu clic aquí</a>", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "Per evitar aquest missatge, seleccioneu una nova Capa Base " + │ │ │ │ │ - "al gestor de capes de la cantonada superior dreta.<br><br>" + │ │ │ │ │ - "Probablement això és degut a que l'script de la biblioteca " + │ │ │ │ │ - "${layerLib} " + │ │ │ │ │ - "no ha estat inclòs a la vostra pàgina.<br><br>" + │ │ │ │ │ - "Desenvolupadors: Per obtenir consells sobre com fer anar això, " + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ - "target='_blank'>féu clic aquí</a>", │ │ │ │ │ + /** │ │ │ │ │ + * Property: standardTileSize │ │ │ │ │ + * {Integer} The size of a standard (non-border) square tile in pixels. │ │ │ │ │ + */ │ │ │ │ │ + standardTileSize: 256, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Escala = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * Property: tileOriginCorner │ │ │ │ │ + * {String} This layer uses top-left as tile origin │ │ │ │ │ + **/ │ │ │ │ │ + tileOriginCorner: "tl", │ │ │ │ │ │ │ │ │ │ - //labels for the graticule control │ │ │ │ │ - 'W': 'O', │ │ │ │ │ - 'E': 'E', │ │ │ │ │ - 'N': 'N', │ │ │ │ │ - 'S': 'S', │ │ │ │ │ - 'Graticule': 'Retícula', │ │ │ │ │ + /** │ │ │ │ │ + * Property: numberOfTiers │ │ │ │ │ + * {Integer} Depth of the Zoomify pyramid, number of tiers (zoom levels) │ │ │ │ │ + * - filled during Zoomify pyramid initialization. │ │ │ │ │ + */ │ │ │ │ │ + numberOfTiers: 0, │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'reprojectDeprecated': "Esteu fent servir l'opció 'reproject' a la capa " + │ │ │ │ │ - "${layerName}. Aquesta opció és obsoleta: el seu ús fou concebut " + │ │ │ │ │ - "per suportar la visualització de dades sobre mapes base comercials, " + │ │ │ │ │ - "però ara aquesta funcionalitat s'hauria d'assolir mitjançant el suport " + │ │ │ │ │ - "de la projecció Spherical Mercator. Més informació disponible a " + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + /** │ │ │ │ │ + * Property: tileCountUpToTier │ │ │ │ │ + * {Array(Integer)} Number of tiles up to the given tier of pyramid. │ │ │ │ │ + * - filled during Zoomify pyramid initialization. │ │ │ │ │ + */ │ │ │ │ │ + tileCountUpToTier: null, │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'methodDeprecated': "Aquest mètode és obsolet i s'eliminarà a la versió 3.0. " + │ │ │ │ │ - "Si us plau feu servir em mètode alternatiu ${newMethod}.", │ │ │ │ │ + /** │ │ │ │ │ + * Property: tierSizeInTiles │ │ │ │ │ + * {Array(<OpenLayers.Size>)} Size (in tiles) for each tier of pyramid. │ │ │ │ │ + * - filled during Zoomify pyramid initialization. │ │ │ │ │ + */ │ │ │ │ │ + tierSizeInTiles: null, │ │ │ │ │ │ │ │ │ │ - // **** end **** │ │ │ │ │ - 'end': '' │ │ │ │ │ + /** │ │ │ │ │ + * Property: tierImageSize │ │ │ │ │ + * {Array(<OpenLayers.Size>)} Image size in pixels for each pyramid tier. │ │ │ │ │ + * - filled during Zoomify pyramid initialization. │ │ │ │ │ + */ │ │ │ │ │ + tierImageSize: null, │ │ │ │ │ │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/ksh.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.Zoomify │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} A name for the layer. │ │ │ │ │ + * url - {String} - Relative or absolute path to the image or more │ │ │ │ │ + * precisly to the TileGroup[X] directories root. │ │ │ │ │ + * Flash plugin use the variable name "zoomifyImagePath" for this. │ │ │ │ │ + * size - {<OpenLayers.Size>} The size (in pixels) of the image. │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, size, options) { │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Purodha │ │ │ │ │ - */ │ │ │ │ │ + // initilize the Zoomify pyramid for given size │ │ │ │ │ + this.initializeZoomify(size); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [ │ │ │ │ │ + name, url, size, {}, │ │ │ │ │ + options │ │ │ │ │ + ]); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["ksh"] │ │ │ │ │ - * Dictionary for Ripoarisch. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Lang["ksh"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + /** │ │ │ │ │ + * Method: initializeZoomify │ │ │ │ │ + * It generates constants for all tiers of the Zoomify pyramid │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * size - {<OpenLayers.Size>} The size of the image in pixels │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + initializeZoomify: function(size) { │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "Met dä Antwoot op en Aanfrooch ham_mer nix aanjefange: ${statusText}", │ │ │ │ │ + var imageSize = size.clone(); │ │ │ │ │ + this.size = size.clone(); │ │ │ │ │ + var tiles = new OpenLayers.Size( │ │ │ │ │ + Math.ceil(imageSize.w / this.standardTileSize), │ │ │ │ │ + Math.ceil(imageSize.h / this.standardTileSize) │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Lengk op Duuer", │ │ │ │ │ + this.tierSizeInTiles = [tiles]; │ │ │ │ │ + this.tierImageSize = [imageSize]; │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Drövver jelaat", │ │ │ │ │ + while (imageSize.w > this.standardTileSize || │ │ │ │ │ + imageSize.h > this.standardTileSize) { │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Jrund-Nivoh", │ │ │ │ │ + imageSize = new OpenLayers.Size( │ │ │ │ │ + Math.floor(imageSize.w / 2), │ │ │ │ │ + Math.floor(imageSize.h / 2) │ │ │ │ │ + ); │ │ │ │ │ + tiles = new OpenLayers.Size( │ │ │ │ │ + Math.ceil(imageSize.w / this.standardTileSize), │ │ │ │ │ + Math.ceil(imageSize.h / this.standardTileSize) │ │ │ │ │ + ); │ │ │ │ │ + this.tierSizeInTiles.push(tiles); │ │ │ │ │ + this.tierImageSize.push(imageSize); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'noFID': "En Saach, woh kein \x3ci lang=\"en\"\x3eFID\x3c/i\x3e för doh es, löht sesch nit ändere.", │ │ │ │ │ + this.tierSizeInTiles.reverse(); │ │ │ │ │ + this.tierImageSize.reverse(); │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Dinge Brauser kann kein Väktore ußjävve. De Zoote Ußjaabe, di em Momang jon, sen:\n${renderers}", │ │ │ │ │ + this.numberOfTiers = this.tierSizeInTiles.length; │ │ │ │ │ + var resolutions = [1]; │ │ │ │ │ + this.tileCountUpToTier = [0]; │ │ │ │ │ + for (var i = 1; i < this.numberOfTiers; i++) { │ │ │ │ │ + resolutions.unshift(Math.pow(2, i)); │ │ │ │ │ + this.tileCountUpToTier.push( │ │ │ │ │ + this.tierSizeInTiles[i - 1].w * this.tierSizeInTiles[i - 1].h + │ │ │ │ │ + this.tileCountUpToTier[i - 1] │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + if (!this.serverResolutions) { │ │ │ │ │ + this.serverResolutions = resolutions; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'minZoomLevelError': "De Eijeschaff „\x3ccode lang=\"en\"\x3eminZoomLevel\x3c/code\x3e“ es bloß doför jedaach, dat mer se met dä Nivvohß bruch, di vun \x3ccode lang=\"en\"\x3eFixedZoomLevels\x3c/code\x3e affhange don. Dat dat \x3ci lang=\"en\"\x3eWFS\x3c/i\x3e-Nivvoh övverhoup de Eijeschaff „\x3ccode lang=\"en\"\x3eminZoomLevel\x3c/code\x3e“ pröhfe deiht, es noch övveresch vun fröhjer. Mer künne dat ävver jez nit fott lohße, oohne dat mer Jevaa loufe, dat Aanwendunge vun OpenLayers nit mieh loufe, di sesch doh velleijsch noch drop am verlohße sin. Dröm sare mer, dat mer et nit mieh han welle, un de „\x3ccode lang=\"en\"\x3eminZoomLevel\x3c/code\x3e“-Eijeschaff weed hee vun de Version 3.0 af nit mieh jeprööf wäde. Nemm doför de Enstellung för de hühßte un de kleinßte Oplöhsung, esu wi et en http://trac.openlayers.org/wiki/SettingZoomLevels opjeschrevve es.", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod:destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + // for now, nothing special to do here. │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.destroy.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "Dä \x3ci lang=\"en\"\x3eWFS\x3c/i\x3e-Vörjang es joot jeloufe: ${response}", │ │ │ │ │ + // Remove from memory the Zoomify pyramid - is that enough? │ │ │ │ │ + this.tileCountUpToTier.length = 0; │ │ │ │ │ + this.tierSizeInTiles.length = 0; │ │ │ │ │ + this.tierImageSize.length = 0; │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "Dä \x3ci lang=\"en\"\x3eWFS\x3c/i\x3e-Vörjang es scheif jejange: ${response}", │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Dat Nivvoh \x3ccode lang=\"en\"\x3eGoogle\x3c/code\x3e kunnt nit reschtesch jelaade wääde.\x3cbr /\x3e\x3cbr /\x3eÖm hee di Nohreesch loß ze krijje, donn en ander Jrund-Nivvoh ußsöhke, rähß bovve en de Äk.\x3cbr /\x3e\x3cbr /\x3eWascheinlesch es dat wiel dat \x3ci lang=\"en\"\x3eGoogle-Maps\x3c/i\x3e-Skrepp entweeder nit reschtesch enjebonge wood, udder nit dä reschtejje \x3ci lang=\"en\"\x3eAPI\x3c/i\x3e-Schlößel för Ding Web-ßait scheke deiht.\x3cbr /\x3e\x3cbr /\x3eFör Projrammierer jidd_et Hölp do_drövver, \x3ca href=\"http://trac.openlayers.org/wiki/Google\" target=\"_blank\"\x3ewi mer dat aan et Loufe brengk\x3c/a\x3e.", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Zoomify>} An exact clone of this <OpenLayers.Layer.Zoomify> │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "Dat Nivvoh \x3ccode\x3e${layerType}\x3c/code\x3e kunnt nit reschtesch jelaade wääde.\x3cbr /\x3e\x3cbr /\x3eÖm hee di Nohreesch loß ze krijje, donn en ander Jrund-Nivvoh ußsöhkre, rähß bovve en de Äk.\x3cbr /\x3e\x3cbr /\x3eWascheinlesch es dat, wiel dat Skrepp \x3ccode\x3e${layerLib}\x3c/code\x3e nit reschtesch enjebonge wood.\x3cbr /\x3e\x3cbr /\x3eFör Projrammierer jidd_Et Hölp do_drövver, \x3ca href=\"http://trac.openlayers.org/wiki/${layerLib}\" target=\"_blank\"\x3ewi mer dat aan et Loufe brengk\x3c/a\x3e.", │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Zoomify(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.size, │ │ │ │ │ + this.options); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Mohßshtaab = 1 : ${scaleDenom}", │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ │ │ │ │ │ - 'W': "W", │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ │ │ │ │ │ - 'E': "O", │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'N': "N", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as │ │ │ │ │ + * parameters │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((this.tileOrigin.lat - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getZoomForResolution(res); │ │ │ │ │ │ │ │ │ │ - 'S': "S", │ │ │ │ │ + var tileIndex = x + y * this.tierSizeInTiles[z].w + this.tileCountUpToTier[z]; │ │ │ │ │ + var path = "TileGroup" + Math.floor((tileIndex) / 256) + │ │ │ │ │ + "/" + z + "-" + x + "-" + y + ".jpg"; │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(path, url); │ │ │ │ │ + } │ │ │ │ │ + return url + path; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Do bruchs de Ußwahl \x3ccode\x3ereproject\x3c/code\x3e op däm Nivvoh \x3ccode\x3e${layerName}\x3c/code\x3e. Di Ußwahl es nit mieh jähn jesinn. Se wohr doför jedaach, öm Date op jeschääfsmäßesch eruß jejovve Kaate bovve drop ze moole, wat ävver enzwesche besser met dä Öngershtözung för de ßfääresche Mäkaator Beldscher jeiht. Doh kanns De mieh drövver fenge op dä Sigg: http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getImageSize │ │ │ │ │ + * getImageSize returns size for a particular tile. If bounds are given as │ │ │ │ │ + * first argument, size is calculated (bottom-right tiles are non square). │ │ │ │ │ + * │ │ │ │ │ + */ │ │ │ │ │ + getImageSize: function() { │ │ │ │ │ + if (arguments.length > 0) { │ │ │ │ │ + var bounds = this.adjustBounds(arguments[0]); │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((this.tileOrigin.lat - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getZoomForResolution(res); │ │ │ │ │ + var w = this.standardTileSize; │ │ │ │ │ + var h = this.standardTileSize; │ │ │ │ │ + if (x == this.tierSizeInTiles[z].w - 1) { │ │ │ │ │ + var w = this.tierImageSize[z].w % this.standardTileSize; │ │ │ │ │ + } │ │ │ │ │ + if (y == this.tierSizeInTiles[z].h - 1) { │ │ │ │ │ + var h = this.tierImageSize[z].h % this.standardTileSize; │ │ │ │ │ + } │ │ │ │ │ + return (new OpenLayers.Size(w, h)); │ │ │ │ │ + } else { │ │ │ │ │ + return this.tileSize; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Hee di Metood es nim_mih aktoäll un et weed se en dä Version 3.0 nit mieh jävve. Nemm \x3ccode\x3e${newMethod}\x3c/code\x3e doföör." │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setMap │ │ │ │ │ + * When the layer is added to a map, then we can fetch our origin │ │ │ │ │ + * (if we don't have one.) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.map.maxExtent.left, │ │ │ │ │ + this.map.maxExtent.top); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Zoomify" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/hr.js │ │ │ │ │ + OpenLayers/Layer/Boxes.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Mvrban │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Layer.js │ │ │ │ │ + * @requires OpenLayers/Layer/Markers.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["hr"] │ │ │ │ │ - * Dictionary for Hrvatski. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Layer.Boxes │ │ │ │ │ + * Draw divs as 'boxes' on the layer. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Markers> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["hr"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - │ │ │ │ │ - 'unhandledRequest': "Nepodržani zahtjev ${statusText}", │ │ │ │ │ - │ │ │ │ │ - 'Permalink': "Permalink", │ │ │ │ │ - │ │ │ │ │ - 'Overlays': "Overlays", │ │ │ │ │ - │ │ │ │ │ - 'Base Layer': "Osnovna karta", │ │ │ │ │ - │ │ │ │ │ - 'noFID': "Ne mogu ažurirati značajku za koju ne postoji FID.", │ │ │ │ │ - │ │ │ │ │ - 'browserNotSupported': "Vaš preglednik ne podržava vektorsko renderiranje. Trenutno podržani rendereri su: ${renderers}", │ │ │ │ │ +OpenLayers.Layer.Boxes = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "WFS Transakcija: USPJEŠNA ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.Boxes │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + */ │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "WFS Transakcija: NEUSPJEŠNA ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: drawMarker │ │ │ │ │ + * Calculate the pixel location for the marker, create it, and │ │ │ │ │ + * add it to the layer's div │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * marker - {<OpenLayers.Marker.Box>} │ │ │ │ │ + */ │ │ │ │ │ + drawMarker: function(marker) { │ │ │ │ │ + var topleft = this.map.getLayerPxFromLonLat({ │ │ │ │ │ + lon: marker.bounds.left, │ │ │ │ │ + lat: marker.bounds.top │ │ │ │ │ + }); │ │ │ │ │ + var botright = this.map.getLayerPxFromLonLat({ │ │ │ │ │ + lon: marker.bounds.right, │ │ │ │ │ + lat: marker.bounds.bottom │ │ │ │ │ + }); │ │ │ │ │ + if (botright == null || topleft == null) { │ │ │ │ │ + marker.display(false); │ │ │ │ │ + } else { │ │ │ │ │ + var markerDiv = marker.draw(topleft, { │ │ │ │ │ + w: Math.max(1, botright.x - topleft.x), │ │ │ │ │ + h: Math.max(1, botright.y - topleft.y) │ │ │ │ │ + }); │ │ │ │ │ + if (!marker.drawn) { │ │ │ │ │ + this.div.appendChild(markerDiv); │ │ │ │ │ + marker.drawn = true; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Mjerilo = 1 : ${scaleDenom}", │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Ova metoda nije odobrena i biti će maknuta u 3.0. Koristite ${newMethod}." │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: removeMarker │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * marker - {<OpenLayers.Marker.Box>} │ │ │ │ │ + */ │ │ │ │ │ + removeMarker: function(marker) { │ │ │ │ │ + OpenLayers.Util.removeItem(this.markers, marker); │ │ │ │ │ + if ((marker.div != null) && │ │ │ │ │ + (marker.div.parentNode == this.div)) { │ │ │ │ │ + this.div.removeChild(marker.div); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Boxes" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/hu.js │ │ │ │ │ + OpenLayers/Layer/Bing.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - City-busz │ │ │ │ │ - * - Glanthor Reviol │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["hu"] │ │ │ │ │ - * Dictionary for Magyar. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.Bing │ │ │ │ │ + * Bing layer using direct tile access as provided by Bing Maps REST Services. │ │ │ │ │ + * See http://msdn.microsoft.com/en-us/library/ff701713.aspx for more │ │ │ │ │ + * information. Note: Terms of Service compliant use requires the map to be │ │ │ │ │ + * configured with an <OpenLayers.Control.Attribution> control and the │ │ │ │ │ + * attribution placed on or near the map. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.XYZ> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["hu"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ +OpenLayers.Layer.Bing = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "Nem kezelt kérés visszatérése ${statusText}", │ │ │ │ │ + /** │ │ │ │ │ + * Property: key │ │ │ │ │ + * {String} API key for Bing maps, get your own key │ │ │ │ │ + * at http://bingmapsportal.com/ . │ │ │ │ │ + */ │ │ │ │ │ + key: null, │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Permalink", │ │ │ │ │ + /** │ │ │ │ │ + * Property: serverResolutions │ │ │ │ │ + * {Array} the resolutions provided by the Bing servers. │ │ │ │ │ + */ │ │ │ │ │ + serverResolutions: [ │ │ │ │ │ + 156543.03390625, 78271.516953125, 39135.7584765625, │ │ │ │ │ + 19567.87923828125, 9783.939619140625, 4891.9698095703125, │ │ │ │ │ + 2445.9849047851562, 1222.9924523925781, 611.4962261962891, │ │ │ │ │ + 305.74811309814453, 152.87405654907226, 76.43702827453613, │ │ │ │ │ + 38.218514137268066, 19.109257068634033, 9.554628534317017, │ │ │ │ │ + 4.777314267158508, 2.388657133579254, 1.194328566789627, │ │ │ │ │ + 0.5971642833948135, 0.29858214169740677, 0.14929107084870338, │ │ │ │ │ + 0.07464553542435169 │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Rávetítések", │ │ │ │ │ + /** │ │ │ │ │ + * Property: attributionTemplate │ │ │ │ │ + * {String} │ │ │ │ │ + */ │ │ │ │ │ + attributionTemplate: '<span class="olBingAttribution ${type}">' + │ │ │ │ │ + '<div><a target="_blank" href="http://www.bing.com/maps/">' + │ │ │ │ │ + '<img src="${logo}" /></a></div>${copyrights}' + │ │ │ │ │ + '<a style="white-space: nowrap" target="_blank" ' + │ │ │ │ │ + 'href="http://www.microsoft.com/maps/product/terms.html">' + │ │ │ │ │ + 'Terms of Use</a></span>', │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Alapréteg", │ │ │ │ │ + /** │ │ │ │ │ + * Property: metadata │ │ │ │ │ + * {Object} Metadata for this layer, as returned by the callback script │ │ │ │ │ + */ │ │ │ │ │ + metadata: null, │ │ │ │ │ │ │ │ │ │ - 'noFID': "Nem frissíthető olyan jellemző, amely nem rendelkezik FID-del.", │ │ │ │ │ + /** │ │ │ │ │ + * Property: protocolRegex │ │ │ │ │ + * {RegExp} Regular expression to match and replace http: in bing urls │ │ │ │ │ + */ │ │ │ │ │ + protocolRegex: /^http:/i, │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "A böngészője nem támogatja a vektoros renderelést. A jelenleg támogatott renderelők:\n${renderers}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: type │ │ │ │ │ + * {String} The layer identifier. Any non-birdseye imageryType │ │ │ │ │ + * from http://msdn.microsoft.com/en-us/library/ff701716.aspx can be │ │ │ │ │ + * used. Default is "Road". │ │ │ │ │ + */ │ │ │ │ │ + type: "Road", │ │ │ │ │ │ │ │ │ │ - 'minZoomLevelError': "A minZoomLevel tulajdonságot csak a következővel való használatra szánták: FixedZoomLevels-leszármazott fóliák. Ez azt jelenti, hogy a minZoomLevel wfs fólia jelölőnégyzetei már a múlté. Mi azonban nem távolíthatjuk el annak a veszélye nélkül, hogy az esetlegesen ettől függő OL alapú alkalmazásokat tönkretennénk. Ezért ezt érvénytelenítjük -- a minZoomLevel az alul levő jelölőnégyzet a 3.0-s verzióból el lesz távolítva. Kérjük, helyette használja a min/max felbontás beállítást, amelyről az alábbi helyen talál leírást: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: culture │ │ │ │ │ + * {String} The culture identifier. See http://msdn.microsoft.com/en-us/library/ff701709.aspx │ │ │ │ │ + * for the definition and the possible values. Default is "en-US". │ │ │ │ │ + */ │ │ │ │ │ + culture: "en-US", │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "WFS tranzakció: SIKERES ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: metadataParams │ │ │ │ │ + * {Object} Optional url parameters for the Get Imagery Metadata request │ │ │ │ │ + * as described here: http://msdn.microsoft.com/en-us/library/ff701716.aspx │ │ │ │ │ + */ │ │ │ │ │ + metadataParams: null, │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "WFS tranzakció: SIKERTELEN ${response}", │ │ │ │ │ + /** APIProperty: tileOptions │ │ │ │ │ + * {Object} optional configuration options for <OpenLayers.Tile> instances │ │ │ │ │ + * created by this Layer. Default is │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * {crossOriginKeyword: 'anonymous'} │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ + tileOptions: null, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "A Google fólia betöltése sikertelen.\x3cbr\x3e\x3cbr\x3eAhhoz, hogy ez az üzenet eltűnjön, válasszon egy új BaseLayer fóliát a jobb felső sarokban található fóliakapcsoló segítségével.\x3cbr\x3e\x3cbr\x3eNagy valószínűséggel ez azért van, mert a Google Maps könyvtár parancsfájlja nem található, vagy nem tartalmazza az Ön oldalához tartozó megfelelő API-kulcsot.\x3cbr\x3e\x3cbr\x3eFejlesztőknek: A helyes működtetésre vonatkozó segítség az alábbi helyen érhető el, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3ekattintson ide\x3c/a\x3e", │ │ │ │ │ + /** APIProperty: protocol │ │ │ │ │ + * {String} Protocol to use to fetch Imagery Metadata, tiles and bing logo │ │ │ │ │ + * Can be 'http:' 'https:' or '' │ │ │ │ │ + * │ │ │ │ │ + * Warning: tiles may not be available under both HTTP and HTTPS protocols. │ │ │ │ │ + * Microsoft approved use of both HTTP and HTTPS urls for tiles. However │ │ │ │ │ + * this is undocumented and the Imagery Metadata API always returns HTTP │ │ │ │ │ + * urls. │ │ │ │ │ + * │ │ │ │ │ + * Default is '', unless when executed from a file:/// uri, in which case │ │ │ │ │ + * it is 'http:'. │ │ │ │ │ + */ │ │ │ │ │ + protocol: ~window.location.href.indexOf('http') ? '' : 'http:', │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "A(z) ${layerType} fólia nem töltődött be helyesen.\x3cbr\x3e\x3cbr\x3eAhhoz, hogy ez az üzenet eltűnjön, válasszon egy új BaseLayer fóliát a jobb felső sarokban található fóliakapcsoló segítségével.\x3cbr\x3e\x3cbr\x3eNagy valószínűséggel ez azért van, mert a(z) ${layerLib} könyvtár parancsfájlja helytelen.\x3cbr\x3e\x3cbr\x3eFejlesztőknek: A helyes működtetésre vonatkozó segítség az alábbi helyen érhető el, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3ekattintson ide\x3c/a\x3e", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.Bing │ │ │ │ │ + * Create a new Bing layer. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var road = new OpenLayers.Layer.Bing({ │ │ │ │ │ + * name: "My Bing Aerial Layer", │ │ │ │ │ + * type: "Aerial", │ │ │ │ │ + * key: "my-api-key-here", │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * options - {Object} Configuration properties for the layer. │ │ │ │ │ + * │ │ │ │ │ + * Required configuration properties: │ │ │ │ │ + * key - {String} Bing Maps API key for your application. Get one at │ │ │ │ │ + * http://bingmapsportal.com/. │ │ │ │ │ + * type - {String} The layer identifier. Any non-birdseye imageryType │ │ │ │ │ + * from http://msdn.microsoft.com/en-us/library/ff701716.aspx can be │ │ │ │ │ + * used. │ │ │ │ │ + * │ │ │ │ │ + * Any other documented layer properties can be provided in the config object. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + sphericalMercator: true │ │ │ │ │ + }, options); │ │ │ │ │ + var name = options.name || "Bing " + (options.type || this.type); │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Lépték = 1 : ${scaleDenom}", │ │ │ │ │ + var newArgs = [name, null, options]; │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, newArgs); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + crossOriginKeyword: 'anonymous' │ │ │ │ │ + }, this.options.tileOptions); │ │ │ │ │ + this.loadMetadata(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'W': "Ny", │ │ │ │ │ + /** │ │ │ │ │ + * Method: loadMetadata │ │ │ │ │ + */ │ │ │ │ │ + loadMetadata: function() { │ │ │ │ │ + this._callbackId = "_callback_" + this.id.replace(/\./g, "_"); │ │ │ │ │ + // link the processMetadata method to the global scope and bind it │ │ │ │ │ + // to this instance │ │ │ │ │ + window[this._callbackId] = OpenLayers.Function.bind( │ │ │ │ │ + OpenLayers.Layer.Bing.processMetadata, this │ │ │ │ │ + ); │ │ │ │ │ + var params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + key: this.key, │ │ │ │ │ + jsonp: this._callbackId, │ │ │ │ │ + include: "ImageryProviders" │ │ │ │ │ + }, this.metadataParams); │ │ │ │ │ + var url = this.protocol + "//dev.virtualearth.net/REST/v1/Imagery/Metadata/" + │ │ │ │ │ + this.type + "?" + OpenLayers.Util.getParameterString(params); │ │ │ │ │ + var script = document.createElement("script"); │ │ │ │ │ + script.type = "text/javascript"; │ │ │ │ │ + script.src = url; │ │ │ │ │ + script.id = this._callbackId; │ │ │ │ │ + document.getElementsByTagName("head")[0].appendChild(script); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'E': "K", │ │ │ │ │ + /** │ │ │ │ │ + * Method: initLayer │ │ │ │ │ + * │ │ │ │ │ + * Sets layer properties according to the metadata provided by the API │ │ │ │ │ + */ │ │ │ │ │ + initLayer: function() { │ │ │ │ │ + var res = this.metadata.resourceSets[0].resources[0]; │ │ │ │ │ + var url = res.imageUrl.replace("{quadkey}", "${quadkey}"); │ │ │ │ │ + url = url.replace("{culture}", this.culture); │ │ │ │ │ + url = url.replace(this.protocolRegex, this.protocol); │ │ │ │ │ + this.url = []; │ │ │ │ │ + for (var i = 0; i < res.imageUrlSubdomains.length; ++i) { │ │ │ │ │ + this.url.push(url.replace("{subdomain}", res.imageUrlSubdomains[i])); │ │ │ │ │ + } │ │ │ │ │ + this.addOptions({ │ │ │ │ │ + maxResolution: Math.min( │ │ │ │ │ + this.serverResolutions[res.zoomMin], │ │ │ │ │ + this.maxResolution || Number.POSITIVE_INFINITY │ │ │ │ │ + ), │ │ │ │ │ + numZoomLevels: Math.min( │ │ │ │ │ + res.zoomMax + 1 - res.zoomMin, this.numZoomLevels │ │ │ │ │ + ) │ │ │ │ │ + }, true); │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + this.redraw(); │ │ │ │ │ + } │ │ │ │ │ + this.updateAttribution(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'N': "É", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * │ │ │ │ │ + * Paramters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + if (!this.url) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var xyz = this.getXYZ(bounds), │ │ │ │ │ + x = xyz.x, │ │ │ │ │ + y = xyz.y, │ │ │ │ │ + z = xyz.z; │ │ │ │ │ + var quadDigits = []; │ │ │ │ │ + for (var i = z; i > 0; --i) { │ │ │ │ │ + var digit = '0'; │ │ │ │ │ + var mask = 1 << (i - 1); │ │ │ │ │ + if ((x & mask) != 0) { │ │ │ │ │ + digit++; │ │ │ │ │ + } │ │ │ │ │ + if ((y & mask) != 0) { │ │ │ │ │ + digit++; │ │ │ │ │ + digit++; │ │ │ │ │ + } │ │ │ │ │ + quadDigits.push(digit); │ │ │ │ │ + } │ │ │ │ │ + var quadKey = quadDigits.join(""); │ │ │ │ │ + var url = this.selectUrl('' + x + y + z, this.url); │ │ │ │ │ │ │ │ │ │ - 'S': "D", │ │ │ │ │ + return OpenLayers.String.format(url, { │ │ │ │ │ + 'quadkey': quadKey │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Ön a \'reproject\' beállítást használja a(z) ${layerName} fólián. Ez a beállítás érvénytelen: használata az üzleti alaptérképek fölötti adatok megjelenítésének támogatására szolgált, de ezt a funkció ezentúl a Gömbi Mercator használatával érhető el. További információ az alábbi helyen érhető el: http://trac.openlayers.org/wiki/SphericalMercator", │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateAttribution │ │ │ │ │ + * Updates the attribution according to the requirements outlined in │ │ │ │ │ + * http://gis.638310.n2.nabble.com/Bing-imagery-td5789168.html │ │ │ │ │ + */ │ │ │ │ │ + updateAttribution: function() { │ │ │ │ │ + var metadata = this.metadata; │ │ │ │ │ + if (!metadata.resourceSets || !this.map || !this.map.center) { │ │ │ │ │ + return; │ │ │ │ │ + } │ │ │ │ │ + var res = metadata.resourceSets[0].resources[0]; │ │ │ │ │ + var extent = this.map.getExtent().transform( │ │ │ │ │ + this.map.getProjectionObject(), │ │ │ │ │ + new OpenLayers.Projection("EPSG:4326") │ │ │ │ │ + ); │ │ │ │ │ + var providers = res.imageryProviders || [], │ │ │ │ │ + zoom = OpenLayers.Util.indexOf(this.serverResolutions, │ │ │ │ │ + this.getServerResolution()), │ │ │ │ │ + copyrights = "", │ │ │ │ │ + provider, i, ii, j, jj, bbox, coverage; │ │ │ │ │ + for (i = 0, ii = providers.length; i < ii; ++i) { │ │ │ │ │ + provider = providers[i]; │ │ │ │ │ + for (j = 0, jj = provider.coverageAreas.length; j < jj; ++j) { │ │ │ │ │ + coverage = provider.coverageAreas[j]; │ │ │ │ │ + // axis order provided is Y,X │ │ │ │ │ + bbox = OpenLayers.Bounds.fromArray(coverage.bbox, true); │ │ │ │ │ + if (extent.intersectsBounds(bbox) && │ │ │ │ │ + zoom <= coverage.zoomMax && zoom >= coverage.zoomMin) { │ │ │ │ │ + copyrights += provider.attribution + " "; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var logo = metadata.brandLogoUri.replace(this.protocolRegex, this.protocol); │ │ │ │ │ + this.attribution = OpenLayers.String.format(this.attributionTemplate, { │ │ │ │ │ + type: this.type.toLowerCase(), │ │ │ │ │ + logo: logo, │ │ │ │ │ + copyrights: copyrights │ │ │ │ │ + }); │ │ │ │ │ + this.map && this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "attribution" │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Ez a módszer érvénytelenítve lett és a 3.0-s verzióból el lesz távolítva. Használja a(z) ${newMethod} módszert helyette." │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + */ │ │ │ │ │ + setMap: function() { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.register("moveend", this, this.updateAttribution); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/te.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Bing>} An exact clone of this <OpenLayers.Layer.Bing> │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Bing(this.options); │ │ │ │ │ + } │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Veeven │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.map && │ │ │ │ │ + this.map.events.unregister("moveend", this, this.updateAttribution); │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Bing" │ │ │ │ │ +}); │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["te"] │ │ │ │ │ - * Dictionary for తెలుగు. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Function: OpenLayers.Layer.Bing.processMetadata │ │ │ │ │ + * This function will be bound to an instance, linked to the global scope with │ │ │ │ │ + * an id, and called by the JSONP script returned by the API. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * metadata - {Object} metadata as returned by the API │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["te"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - │ │ │ │ │ - 'Permalink': "స్థిరలింకు", │ │ │ │ │ - │ │ │ │ │ - 'W': "ప", │ │ │ │ │ - │ │ │ │ │ - 'E': "తూ", │ │ │ │ │ - │ │ │ │ │ - 'N': "ఉ", │ │ │ │ │ - │ │ │ │ │ - 'S': "ద" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ +OpenLayers.Layer.Bing.processMetadata = function(metadata) { │ │ │ │ │ + this.metadata = metadata; │ │ │ │ │ + this.initLayer(); │ │ │ │ │ + var script = document.getElementById(this._callbackId); │ │ │ │ │ + script.parentNode.removeChild(script); │ │ │ │ │ + window[this._callbackId] = undefined; // cannot delete from window in IE │ │ │ │ │ + delete this._callbackId; │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/ru.js │ │ │ │ │ + OpenLayers/Layer/MapGuide.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Ferrer │ │ │ │ │ - * - Komzpa │ │ │ │ │ - * - Lockal │ │ │ │ │ - * - Александр Сигачёв │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["ru"] │ │ │ │ │ - * Dictionary for Русский. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Layer.MapGuide │ │ │ │ │ + * Instances of OpenLayers.Layer.MapGuide are used to display │ │ │ │ │ + * data from a MapGuide OS instance. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["ru"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ +OpenLayers.Layer.MapGuide = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "Необработанный запрос вернул ${statusText}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} Treat this layer as a base layer. Default is true. │ │ │ │ │ + **/ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Постоянная ссылка", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: useHttpTile │ │ │ │ │ + * {Boolean} use a tile cache exposed directly via a webserver rather than the │ │ │ │ │ + * via mapguide server. This does require extra configuration on the Mapguide Server, │ │ │ │ │ + * and will only work when singleTile is false. The url for the layer must be set to the │ │ │ │ │ + * webserver path rather than the Mapguide mapagent. │ │ │ │ │ + * See http://trac.osgeo.org/mapguide/wiki/CodeSamples/Tiles/ServingTilesViaHttp │ │ │ │ │ + **/ │ │ │ │ │ + useHttpTile: false, │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Слои", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: singleTile │ │ │ │ │ + * {Boolean} use tile server or request single tile image. │ │ │ │ │ + **/ │ │ │ │ │ + singleTile: false, │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Основной слой", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: useOverlay │ │ │ │ │ + * {Boolean} flag to indicate if the layer should be retrieved using │ │ │ │ │ + * GETMAPIMAGE (default) or using GETDYNAMICOVERLAY requests. │ │ │ │ │ + **/ │ │ │ │ │ + useOverlay: false, │ │ │ │ │ │ │ │ │ │ - 'noFID': "Невозможно обновить объект, для которого нет FID.", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: useAsyncOverlay │ │ │ │ │ + * {Boolean} indicates if the MapGuide site supports the asynchronous │ │ │ │ │ + * GETDYNAMICOVERLAY requests which is available in MapGuide Enterprise 2010 │ │ │ │ │ + * and MapGuide Open Source v2.0.3 or higher. The newer versions of MG │ │ │ │ │ + * is called asynchronously, allows selections to be drawn separately from │ │ │ │ │ + * the map and offers styling options. │ │ │ │ │ + * │ │ │ │ │ + * With older versions of MapGuide, set useAsyncOverlay=false. Note that in │ │ │ │ │ + * this case a synchronous AJAX call is issued and the mapname and session │ │ │ │ │ + * parameters must be used to initialize the layer, not the mapdefinition │ │ │ │ │ + * parameter. Also note that this will issue a synchronous AJAX request │ │ │ │ │ + * before the image request can be issued so the users browser may lock │ │ │ │ │ + * up if the MG Web tier does not respond in a timely fashion. │ │ │ │ │ + **/ │ │ │ │ │ + useAsyncOverlay: true, │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Ваш браузер не поддерживает векторную графику. На данный момент поддерживаются:\n${renderers}", │ │ │ │ │ + /** │ │ │ │ │ + * Constant: TILE_PARAMS │ │ │ │ │ + * {Object} Hashtable of default parameter key/value pairs for tiled layer │ │ │ │ │ + */ │ │ │ │ │ + TILE_PARAMS: { │ │ │ │ │ + operation: 'GETTILEIMAGE', │ │ │ │ │ + version: '1.2.0' │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'minZoomLevelError': "Свойство minZoomLevel предназначено только для использования со слоями, являющимися потомками FixedZoomLevels. То, что этот WFS-слой проверяется на minZoomLevel — реликт прошлого. Однако мы не можем удалить эту функцию, так как, возможно, от неё зависят некоторые основанные на OpenLayers приложения. Функция объявлена устаревшей — проверка minZoomLevel будет удалена в 3.0. Пожалуйста, используйте вместо неё настройку мин/макс разрешения, описанную здесь: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + /** │ │ │ │ │ + * Constant: SINGLE_TILE_PARAMS │ │ │ │ │ + * {Object} Hashtable of default parameter key/value pairs for untiled layer │ │ │ │ │ + */ │ │ │ │ │ + SINGLE_TILE_PARAMS: { │ │ │ │ │ + operation: 'GETMAPIMAGE', │ │ │ │ │ + format: 'PNG', │ │ │ │ │ + locale: 'en', │ │ │ │ │ + clip: '1', │ │ │ │ │ + version: '1.0.0' │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "Транзакция WFS: УСПЕШНО ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Constant: OVERLAY_PARAMS │ │ │ │ │ + * {Object} Hashtable of default parameter key/value pairs for untiled layer │ │ │ │ │ + */ │ │ │ │ │ + OVERLAY_PARAMS: { │ │ │ │ │ + operation: 'GETDYNAMICMAPOVERLAYIMAGE', │ │ │ │ │ + format: 'PNG', │ │ │ │ │ + locale: 'en', │ │ │ │ │ + clip: '1', │ │ │ │ │ + version: '2.0.0' │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "Транзакция WFS: ОШИБКА ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Constant: FOLDER_PARAMS │ │ │ │ │ + * {Object} Hashtable of parameter key/value pairs which describe │ │ │ │ │ + * the folder structure for tiles as configured in the mapguide │ │ │ │ │ + * serverconfig.ini section [TileServiceProperties] │ │ │ │ │ + */ │ │ │ │ │ + FOLDER_PARAMS: { │ │ │ │ │ + tileColumnsPerFolder: 30, │ │ │ │ │ + tileRowsPerFolder: 30, │ │ │ │ │ + format: 'png', │ │ │ │ │ + querystring: null │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Слой Google не удалось нормально загрузить.\x3cbr\x3e\x3cbr\x3eЧтобы избавиться от этого сообщения, выбите другой основной слой в переключателе в правом верхнем углу.\x3cbr\x3e\x3cbr\x3eСкорее всего, причина в том, что библиотека Google Maps не была включена или не содержит корректного API-ключа для вашего сайта.\x3cbr\x3e\x3cbr\x3eРазработчикам: чтобы узнать, как сделать, чтобы всё заработало, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eщёлкните тут\x3c/a\x3e", │ │ │ │ │ + /** │ │ │ │ │ + * Property: defaultSize │ │ │ │ │ + * {<OpenLayers.Size>} Tile size as produced by MapGuide server │ │ │ │ │ + **/ │ │ │ │ │ + defaultSize: new OpenLayers.Size(300, 300), │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "Слой ${layerType} не удалось нормально загрузить. \x3cbr\x3e\x3cbr\x3eЧтобы избавиться от этого сообщения, выбите другой основной слой в переключателе в правом верхнем углу.\x3cbr\x3e\x3cbr\x3eСкорее всего, причина в том, что библиотека ${layerLib} не была включена или была включена некорректно.\x3cbr\x3e\x3cbr\x3eРазработчикам: чтобы узнать, как сделать, чтобы всё заработало, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eщёлкните тут\x3c/a\x3e", │ │ │ │ │ + /** │ │ │ │ │ + * Property: tileOriginCorner │ │ │ │ │ + * {String} MapGuide tile server uses top-left as tile origin │ │ │ │ │ + **/ │ │ │ │ │ + tileOriginCorner: "tl", │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Масштаб = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.MapGuide │ │ │ │ │ + * Create a new Mapguide layer, either tiled or untiled. │ │ │ │ │ + * │ │ │ │ │ + * For tiled layers, the 'groupName' and 'mapDefinition' values │ │ │ │ │ + * must be specified as parameters in the constructor. │ │ │ │ │ + * │ │ │ │ │ + * For untiled base layers, specify either combination of 'mapName' and │ │ │ │ │ + * 'session', or 'mapDefinition' and 'locale'. │ │ │ │ │ + * │ │ │ │ │ + * For older versions of MapGuide and overlay layers, set useAsyncOverlay │ │ │ │ │ + * to false and in this case mapName and session are required parameters │ │ │ │ │ + * for the constructor. │ │ │ │ │ + * │ │ │ │ │ + * NOTE: MapGuide OS uses a DPI value and degrees to meters conversion │ │ │ │ │ + * factor that are different than the defaults used in OpenLayers, │ │ │ │ │ + * so these must be adjusted accordingly in your application. │ │ │ │ │ + * See the MapGuide example for how to set these values for MGOS. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} Name of the layer displayed in the interface │ │ │ │ │ + * url - {String} Location of the MapGuide mapagent executable │ │ │ │ │ + * (e.g. http://localhost:8008/mapguide/mapagent/mapagent.fcgi) │ │ │ │ │ + * params - {Object} hashtable of additional parameters to use. Some │ │ │ │ │ + * parameters may require additional code on the server. The ones that │ │ │ │ │ + * you may want to use are: │ │ │ │ │ + * - mapDefinition - {String} The MapGuide resource definition │ │ │ │ │ + * (e.g. Library://Samples/Gmap/Maps/gmapTiled.MapDefinition) │ │ │ │ │ + * - locale - Locale setting │ │ │ │ │ + * (for untiled overlays layers only) │ │ │ │ │ + * - mapName - {String} Name of the map as stored in the MapGuide session. │ │ │ │ │ + * (for untiled layers with a session parameter only) │ │ │ │ │ + * - session - { String} MapGuide session ID │ │ │ │ │ + * (for untiled overlays layers only) │ │ │ │ │ + * - basemaplayergroupname - {String} GroupName for tiled MapGuide layers only │ │ │ │ │ + * - format - Image format to be returned (for untiled overlay layers only) │ │ │ │ │ + * - showLayers - {String} A comma separated list of GUID's for the │ │ │ │ │ + * layers to display eg: 'cvc-xcv34,453-345-345sdf'. │ │ │ │ │ + * - hideLayers - {String} A comma separated list of GUID's for the │ │ │ │ │ + * layers to hide eg: 'cvc-xcv34,453-345-345sdf'. │ │ │ │ │ + * - showGroups - {String} A comma separated list of GUID's for the │ │ │ │ │ + * groups to display eg: 'cvc-xcv34,453-345-345sdf'. │ │ │ │ │ + * - hideGroups - {String} A comma separated list of GUID's for the │ │ │ │ │ + * groups to hide eg: 'cvc-xcv34,453-345-345sdf' │ │ │ │ │ + * - selectionXml - {String} A selection xml string Some server plumbing │ │ │ │ │ + * is required to read such a value. │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer; │ │ │ │ │ + * will vary depending if tiled or untiled maps are being requested │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ │ │ │ │ │ - 'W': "З", │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - 'E': "В", │ │ │ │ │ + // unless explicitly set in options, if the layer is transparent, │ │ │ │ │ + // it will be an overlay │ │ │ │ │ + if (options == null || options.isBaseLayer == null) { │ │ │ │ │ + this.isBaseLayer = ((this.transparent != "true") && │ │ │ │ │ + (this.transparent != true)); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'N': "С", │ │ │ │ │ + if (options && options.useOverlay != null) { │ │ │ │ │ + this.useOverlay = options.useOverlay; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'S': "Ю", │ │ │ │ │ + //initialize for untiled layers │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + if (this.useOverlay) { │ │ │ │ │ + OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, │ │ │ │ │ + this.OVERLAY_PARAMS │ │ │ │ │ + ); │ │ │ │ │ + if (!this.useAsyncOverlay) { │ │ │ │ │ + this.params.version = "1.0.0"; │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, │ │ │ │ │ + this.SINGLE_TILE_PARAMS │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + //initialize for tiled layers │ │ │ │ │ + if (this.useHttpTile) { │ │ │ │ │ + OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, │ │ │ │ │ + this.FOLDER_PARAMS │ │ │ │ │ + ); │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, │ │ │ │ │ + this.TILE_PARAMS │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + this.setTileSize(this.defaultSize); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Вы используете опцию \'reproject\' для слоя ${layerName}. Эта опция является устаревшей: ее использование предполагалось для поддержки показа данных поверх коммерческих базовых карт, но теперь этот функционал несёт встроенная поддержка сферической проекции Меркатора. Больше сведений доступно на http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + /** │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.MapGuide>} An exact clone of this layer │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.MapGuide(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.params, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Этот метод считается устаревшим и будет удалён в версии 3.0. Пожалуйста, пользуйтесь ${newMethod}." │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/gl.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * Return a query string for this layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox │ │ │ │ │ + * for the request │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the layer's url and parameters and also │ │ │ │ │ + * the passed-in bounds and appropriate tile size specified │ │ │ │ │ + * as parameters. │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var url; │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var mapSize = this.map.getSize(); │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Toliño │ │ │ │ │ - */ │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + //set up the call for GETMAPIMAGE or GETDYNAMICMAPOVERLAY with │ │ │ │ │ + //dynamic map parameters │ │ │ │ │ + var params = { │ │ │ │ │ + setdisplaydpi: OpenLayers.DOTS_PER_INCH, │ │ │ │ │ + setdisplayheight: mapSize.h * this.ratio, │ │ │ │ │ + setdisplaywidth: mapSize.w * this.ratio, │ │ │ │ │ + setviewcenterx: center.lon, │ │ │ │ │ + setviewcentery: center.lat, │ │ │ │ │ + setviewscale: this.map.getScale() │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ + if (this.useOverlay && !this.useAsyncOverlay) { │ │ │ │ │ + //first we need to call GETVISIBLEMAPEXTENT to set the extent │ │ │ │ │ + var getVisParams = {}; │ │ │ │ │ + getVisParams = OpenLayers.Util.extend(getVisParams, params); │ │ │ │ │ + getVisParams.operation = "GETVISIBLEMAPEXTENT"; │ │ │ │ │ + getVisParams.version = "1.0.0"; │ │ │ │ │ + getVisParams.session = this.params.session; │ │ │ │ │ + getVisParams.mapName = this.params.mapName; │ │ │ │ │ + getVisParams.format = 'text/xml'; │ │ │ │ │ + url = this.getFullRequestString(getVisParams); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["gl"] │ │ │ │ │ - * Dictionary for Galego. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Lang["gl"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + OpenLayers.Request.GET({ │ │ │ │ │ + url: url, │ │ │ │ │ + async: false │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + //construct the full URL │ │ │ │ │ + url = this.getFullRequestString(params); │ │ │ │ │ + } else { │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "Solicitude non xerada; a resposta foi: ${statusText}", │ │ │ │ │ + //tiled version │ │ │ │ │ + var currentRes = this.map.getResolution(); │ │ │ │ │ + var colidx = Math.floor((bounds.left - this.maxExtent.left) / currentRes); │ │ │ │ │ + colidx = Math.round(colidx / this.tileSize.w); │ │ │ │ │ + var rowidx = Math.floor((this.maxExtent.top - bounds.top) / currentRes); │ │ │ │ │ + rowidx = Math.round(rowidx / this.tileSize.h); │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Ligazón permanente", │ │ │ │ │ + if (this.useHttpTile) { │ │ │ │ │ + url = this.getImageFilePath({ │ │ │ │ │ + tilecol: colidx, │ │ │ │ │ + tilerow: rowidx, │ │ │ │ │ + scaleindex: this.resolutions.length - this.map.zoom - 1 │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Capas superpostas", │ │ │ │ │ + } else { │ │ │ │ │ + url = this.getFullRequestString({ │ │ │ │ │ + tilecol: colidx, │ │ │ │ │ + tilerow: rowidx, │ │ │ │ │ + scaleindex: this.resolutions.length - this.map.zoom - 1 │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return url; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Capa base", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getFullRequestString │ │ │ │ │ + * getFullRequestString on MapGuide layers is special, because we │ │ │ │ │ + * do a regular expression replace on ',' in parameters to '+'. │ │ │ │ │ + * This is why it is subclassed here. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * altUrl - {String} Alternative base URL to use. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the layer's url appropriately encoded for MapGuide │ │ │ │ │ + */ │ │ │ │ │ + getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ + // use layer's url unless altUrl passed in │ │ │ │ │ + var url = (altUrl == null) ? this.url : altUrl; │ │ │ │ │ │ │ │ │ │ - 'noFID': "Non se pode actualizar a funcionalidade para a que non hai FID.", │ │ │ │ │ + // if url is not a string, it should be an array of strings, │ │ │ │ │ + // in which case we will randomly select one of them in order │ │ │ │ │ + // to evenly distribute requests to different urls. │ │ │ │ │ + if (typeof url == "object") { │ │ │ │ │ + url = url[Math.floor(Math.random() * url.length)]; │ │ │ │ │ + } │ │ │ │ │ + // requestString always starts with url │ │ │ │ │ + var requestString = url; │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "O seu navegador non soporta a renderización de vectores. Os renderizadores soportados actualmente son:\n${renderers}", │ │ │ │ │ + // create a new params hashtable with all the layer params and the │ │ │ │ │ + // new params together. then convert to string │ │ │ │ │ + var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ + allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ + // ignore parameters that are already in the url search string │ │ │ │ │ + var urlParams = OpenLayers.Util.upperCaseObject( │ │ │ │ │ + OpenLayers.Util.getParameters(url)); │ │ │ │ │ + for (var key in allParams) { │ │ │ │ │ + if (key.toUpperCase() in urlParams) { │ │ │ │ │ + delete allParams[key]; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ │ │ │ │ │ - 'minZoomLevelError': "A propiedade minZoomLevel é só para uso conxuntamente coas capas FixedZoomLevels-descendent. O feito de que esa capa wfs verifique o minZoomLevel é unha reliquia do pasado. Non podemos, con todo, eliminala sen a posibilidade de non romper as aplicacións baseadas en OL que poidan depender dela. Por iso a estamos deixando obsoleta (a comprobación minZoomLevel de embaixo será eliminada na versión 3.0). Por favor, no canto diso use o axuste de resolución mín/máx tal e como está descrito aquí: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + /* MapGuide needs '+' seperating things like bounds/height/width. │ │ │ │ │ + Since typically this is URL encoded, we use a slight hack: we │ │ │ │ │ + depend on the list-like functionality of getParameterString to │ │ │ │ │ + leave ',' only in the case of list items (since otherwise it is │ │ │ │ │ + encoded) then do a regular expression replace on the , characters │ │ │ │ │ + to '+' */ │ │ │ │ │ + paramsString = paramsString.replace(/,/g, "+"); │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "Transacción WFS: ÉXITO ${response}", │ │ │ │ │ + if (paramsString != "") { │ │ │ │ │ + var lastServerChar = url.charAt(url.length - 1); │ │ │ │ │ + if ((lastServerChar == "&") || (lastServerChar == "?")) { │ │ │ │ │ + requestString += paramsString; │ │ │ │ │ + } else { │ │ │ │ │ + if (url.indexOf('?') == -1) { │ │ │ │ │ + //serverPath has no ? -- add one │ │ │ │ │ + requestString += '?' + paramsString; │ │ │ │ │ + } else { │ │ │ │ │ + //serverPath contains ?, so must already have paramsString at the end │ │ │ │ │ + requestString += '&' + paramsString; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return requestString; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "Transacción WFS: FALLIDA ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getImageFilePath │ │ │ │ │ + * special handler to request mapguide tiles from an http exposed tilecache │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * altUrl - {String} Alternative base URL to use. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the url for the tile image │ │ │ │ │ + */ │ │ │ │ │ + getImageFilePath: function(newParams, altUrl) { │ │ │ │ │ + // use layer's url unless altUrl passed in │ │ │ │ │ + var url = (altUrl == null) ? this.url : altUrl; │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "A capa do Google non puido cargarse correctamente.\x3cbr\x3e\x3cbr\x3ePara evitar esta mensaxe, escolla unha nova capa base no seleccionador de capas na marxe superior dereita.\x3cbr\x3e\x3cbr\x3eProbablemente, isto acontece porque a escritura da libraría do Google Maps ou ben non foi incluída ou ben non contén a clave API correcta para o seu sitio.\x3cbr\x3e\x3cbr\x3eDesenvolvedores: para axudar a facer funcionar isto correctamente, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3epremede aquí\x3c/a\x3e", │ │ │ │ │ + // if url is not a string, it should be an array of strings, │ │ │ │ │ + // in which case we will randomly select one of them in order │ │ │ │ │ + // to evenly distribute requests to different urls. │ │ │ │ │ + if (typeof url == "object") { │ │ │ │ │ + url = url[Math.floor(Math.random() * url.length)]; │ │ │ │ │ + } │ │ │ │ │ + // requestString always starts with url │ │ │ │ │ + var requestString = url; │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "A capa ${layerType} foi incapaz de cargarse correctamente.\x3cbr\x3e\x3cbr\x3ePara evitar esta mensaxe, escolla unha nova capa base no seleccionador de capas na marxe superior dereita.\x3cbr\x3e\x3cbr\x3eProbablemente, isto acontece porque a escritura da libraría ${layerLib} non foi ben incluída.\x3cbr\x3e\x3cbr\x3eDesenvolvedores: para axudar a facer funcionar isto correctamente, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3epremede aquí\x3c/a\x3e", │ │ │ │ │ + var tileRowGroup = ""; │ │ │ │ │ + var tileColGroup = ""; │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Escala = 1 : ${scaleDenom}", │ │ │ │ │ + if (newParams.tilerow < 0) { │ │ │ │ │ + tileRowGroup = '-'; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'W': "O", │ │ │ │ │ + if (newParams.tilerow == 0) { │ │ │ │ │ + tileRowGroup += '0'; │ │ │ │ │ + } else { │ │ │ │ │ + tileRowGroup += Math.floor(Math.abs(newParams.tilerow / this.params.tileRowsPerFolder)) * this.params.tileRowsPerFolder; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'E': "L", │ │ │ │ │ + if (newParams.tilecol < 0) { │ │ │ │ │ + tileColGroup = '-'; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'N': "N", │ │ │ │ │ + if (newParams.tilecol == 0) { │ │ │ │ │ + tileColGroup += '0'; │ │ │ │ │ + } else { │ │ │ │ │ + tileColGroup += Math.floor(Math.abs(newParams.tilecol / this.params.tileColumnsPerFolder)) * this.params.tileColumnsPerFolder; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'S': "S", │ │ │ │ │ + var tilePath = '/S' + Math.floor(newParams.scaleindex) + │ │ │ │ │ + '/' + this.params.basemaplayergroupname + │ │ │ │ │ + '/R' + tileRowGroup + │ │ │ │ │ + '/C' + tileColGroup + │ │ │ │ │ + '/' + (newParams.tilerow % this.params.tileRowsPerFolder) + │ │ │ │ │ + '_' + (newParams.tilecol % this.params.tileColumnsPerFolder) + │ │ │ │ │ + '.' + this.params.format; │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Está usando a opción \"reproject\" na capa ${layerName}. Esta opción está obsoleta: o seu uso foi deseñado para a visualización de datos sobre mapas base comerciais, pero esta funcionalidade debera agora ser obtida utilizando a proxección Spherical Mercator. Hai dispoñible máis información en http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + if (this.params.querystring) { │ │ │ │ │ + tilePath += "?" + this.params.querystring; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Este método está obsoleto e será eliminado na versión 3.0. Por favor, no canto deste use ${newMethod}." │ │ │ │ │ + requestString += tilePath; │ │ │ │ │ + return requestString; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.MapGuide" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/pt.js │ │ │ │ │ + OpenLayers/Layer/TMS.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Hamilton Abreu │ │ │ │ │ - * - Malafaya │ │ │ │ │ - * - Waldir │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["pt"] │ │ │ │ │ - * Dictionary for Português. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Layer.TMS │ │ │ │ │ + * Create a layer for accessing tiles from services that conform with the │ │ │ │ │ + * Tile Map Service Specification │ │ │ │ │ + * (http://wiki.osgeo.org/wiki/Tile_Map_Service_Specification). │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var layer = new OpenLayers.Layer.TMS( │ │ │ │ │ + * "My Layer", // name for display in LayerSwitcher │ │ │ │ │ + * "http://tilecache.osgeo.org/wms-c/Basic.py/", // service endpoint │ │ │ │ │ + * {layername: "basic", type: "png"} // required properties │ │ │ │ │ + * ); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["pt"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - │ │ │ │ │ - 'unhandledRequest': "Servidor devolveu erro não contemplado ${statusText}", │ │ │ │ │ - │ │ │ │ │ - 'Permalink': "Ligação permanente", │ │ │ │ │ - │ │ │ │ │ - 'Overlays': "Sobreposições", │ │ │ │ │ +OpenLayers.Layer.TMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Camada Base", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: serviceVersion │ │ │ │ │ + * {String} Service version for tile requests. Default is "1.0.0". │ │ │ │ │ + */ │ │ │ │ │ + serviceVersion: "1.0.0", │ │ │ │ │ │ │ │ │ │ - 'noFID': "Não é possível atualizar um elemento para a qual não há FID.", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: layername │ │ │ │ │ + * {String} The identifier for the <TileMap> as advertised by the service. │ │ │ │ │ + * For example, if the service advertises a <TileMap> with │ │ │ │ │ + * 'href="http://tms.osgeo.org/1.0.0/vmap0"', the <layername> property │ │ │ │ │ + * would be set to "vmap0". │ │ │ │ │ + */ │ │ │ │ │ + layername: null, │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "O seu navegador não suporta renderização vetorial. Actualmente os renderizadores suportados são:\n${renderers}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: type │ │ │ │ │ + * {String} The format extension corresponding to the requested tile image │ │ │ │ │ + * type. This is advertised in a <TileFormat> element as the │ │ │ │ │ + * "extension" attribute. For example, if the service advertises a │ │ │ │ │ + * <TileMap> with <TileFormat width="256" height="256" mime-type="image/jpeg" extension="jpg" />, │ │ │ │ │ + * the <type> property would be set to "jpg". │ │ │ │ │ + */ │ │ │ │ │ + type: null, │ │ │ │ │ │ │ │ │ │ - 'minZoomLevelError': "A propriedade minZoomLevel só deve ser usada com as camadas descendentes da FixedZoomLevels. A verificação da propriedade por esta camada wfs é uma relíquia do passado. No entanto, não podemos removê-la sem correr o risco de afectar aplicações OL que dependam dela. Portanto, estamos a torná-la obsoleta -- a verificação minZoomLevel será removida na versão 3.0. Em vez dela, por favor, use as opções de resolução min/max descritas aqui: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} Make this layer a base layer. Default is true. Set false to │ │ │ │ │ + * use the layer as an overlay. │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "Transacção WFS: SUCESSO ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: tileOrigin │ │ │ │ │ + * {<OpenLayers.LonLat>} Optional origin for aligning the grid of tiles. │ │ │ │ │ + * If provided, requests for tiles at all resolutions will be aligned │ │ │ │ │ + * with this location (no tiles shall overlap this location). If │ │ │ │ │ + * not provided, the grid of tiles will be aligned with the bottom-left │ │ │ │ │ + * corner of the map's <maxExtent>. Default is ``null``. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var layer = new OpenLayers.Layer.TMS( │ │ │ │ │ + * "My Layer", │ │ │ │ │ + * "http://tilecache.osgeo.org/wms-c/Basic.py/", │ │ │ │ │ + * { │ │ │ │ │ + * layername: "basic", │ │ │ │ │ + * type: "png", │ │ │ │ │ + * // set if different than the bottom left of map.maxExtent │ │ │ │ │ + * tileOrigin: new OpenLayers.LonLat(-180, -90) │ │ │ │ │ + * } │ │ │ │ │ + * ); │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ + tileOrigin: null, │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "Transacção WFS: FALHOU ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: serverResolutions │ │ │ │ │ + * {Array} A list of all resolutions available on the server. Only set this │ │ │ │ │ + * property if the map resolutions differ from the server. This │ │ │ │ │ + * property serves two purposes. (a) <serverResolutions> can include │ │ │ │ │ + * resolutions that the server supports and that you don't want to │ │ │ │ │ + * provide with this layer; you can also look at <zoomOffset>, which is │ │ │ │ │ + * an alternative to <serverResolutions> for that specific purpose. │ │ │ │ │ + * (b) The map can work with resolutions that aren't supported by │ │ │ │ │ + * the server, i.e. that aren't in <serverResolutions>. When the │ │ │ │ │ + * map is displayed in such a resolution data for the closest │ │ │ │ │ + * server-supported resolution is loaded and the layer div is │ │ │ │ │ + * stretched as necessary. │ │ │ │ │ + */ │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "A Camada Google não foi correctamente carregada.\x3cbr\x3e\x3cbr\x3ePara deixar de receber esta mensagem, seleccione uma nova Camada-Base no \'\'switcher\'\' de camadas no canto superior direito.\x3cbr\x3e\x3cbr\x3eProvavelmente, isto acontece porque o \'\'script\'\' da biblioteca do Google Maps não foi incluído ou não contém a chave API correcta para o seu sítio.\x3cbr\x3e\x3cbr\x3eProgramadores: Para ajuda sobre como solucionar o problema \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eclique aqui\x3c/a\x3e .", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomOffset │ │ │ │ │ + * {Number} If your cache has more zoom levels than you want to provide │ │ │ │ │ + * access to with this layer, supply a zoomOffset. This zoom offset │ │ │ │ │ + * is added to the current map zoom level to determine the level │ │ │ │ │ + * for a requested tile. For example, if you supply a zoomOffset │ │ │ │ │ + * of 3, when the map is at the zoom 0, tiles will be requested from │ │ │ │ │ + * level 3 of your cache. Default is 0 (assumes cache level and map │ │ │ │ │ + * zoom are equivalent). Using <zoomOffset> is an alternative to │ │ │ │ │ + * setting <serverResolutions> if you only want to expose a subset │ │ │ │ │ + * of the server resolutions. │ │ │ │ │ + */ │ │ │ │ │ + zoomOffset: 0, │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "A camada ${layerType} não foi correctamente carregada.\x3cbr\x3e\x3cbr\x3ePara desactivar esta mensagem, seleccione uma nova Camada-Base no \'\'switcher\'\' de camadas no canto superior direito.\x3cbr\x3e\x3cbr\x3eProvavelmente, isto acontece porque o \'\'script\'\' da biblioteca ${layerLib} não foi incluído correctamente.\x3cbr\x3e\x3cbr\x3eProgramadores: Para ajuda sobre como solucionar o problema \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eclique aqui\x3c/a\x3e .", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.TMS │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} Title to be displayed in a <OpenLayers.Control.LayerSwitcher> │ │ │ │ │ + * url - {String} Service endpoint (without the version number). E.g. │ │ │ │ │ + * "http://tms.osgeo.org/". │ │ │ │ │ + * options - {Object} Additional properties to be set on the layer. The │ │ │ │ │ + * <layername> and <type> properties must be set here. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + newArguments.push(name, url, {}, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Escala = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Create a complete copy of this layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} Should only be provided by subclasses that call this │ │ │ │ │ + * method. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.TMS>} An exact clone of this <OpenLayers.Layer.TMS> │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ │ │ │ │ │ - 'W': "O", │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.TMS(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'E': "E", │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ │ │ │ │ │ - 'N': "N", │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ │ │ │ │ │ - 'S': "S", │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Está usando a opção \'reproject\' na camada ${layerName}. Esta opção é obsoleta: foi concebida para permitir a apresentação de dados sobre mapas-base comerciais, mas esta funcionalidade é agora suportada pelo Mercator Esférico. Mais informação está disponível em http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as │ │ │ │ │ + * parameters │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((bounds.bottom - this.tileOrigin.lat) / (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getServerZoom(); │ │ │ │ │ + var path = this.serviceVersion + "/" + this.layername + "/" + z + "/" + x + "/" + y + "." + this.type; │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(path, url); │ │ │ │ │ + } │ │ │ │ │ + return url + path; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Este método foi declarado obsoleto e será removido na versão 3.0. Por favor, use ${newMethod} em vez disso." │ │ │ │ │ + /** │ │ │ │ │ + * Method: setMap │ │ │ │ │ + * When the layer is added to a map, then we can fetch our origin │ │ │ │ │ + * (if we don't have one.) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (!this.tileOrigin) { │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.map.maxExtent.left, │ │ │ │ │ + this.map.maxExtent.bottom); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.TMS" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/ar.js │ │ │ │ │ + OpenLayers/Layer/OSM.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Meno25 │ │ │ │ │ - * - Mutarjem horr │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["ar"] │ │ │ │ │ - * Dictionary for العربية. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Layer.OSM │ │ │ │ │ + * This layer allows accessing OpenStreetMap tiles. By default the OpenStreetMap │ │ │ │ │ + * hosted tile.openstreetmap.org Mapnik tileset is used. If you wish to use │ │ │ │ │ + * a different layer instead, you need to provide a different │ │ │ │ │ + * URL to the constructor. Here's an example for using OpenCycleMap: │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * new OpenLayers.Layer.OSM("OpenCycleMap", │ │ │ │ │ + * ["http://a.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ + * "http://b.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ + * "http://c.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png"]); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.XYZ> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["ar"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ +OpenLayers.Layer.OSM = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ │ │ │ │ │ - 'Permalink': "وصلة دائمة", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: name │ │ │ │ │ + * {String} The layer name. Defaults to "OpenStreetMap" if the first │ │ │ │ │ + * argument to the constructor is null or undefined. │ │ │ │ │ + */ │ │ │ │ │ + name: "OpenStreetMap", │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "الطبقة الاساسية", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String} The tileset URL scheme. Defaults to │ │ │ │ │ + * : http://[a|b|c].tile.openstreetmap.org/${z}/${x}/${y}.png │ │ │ │ │ + * (the official OSM tileset) if the second argument to the constructor │ │ │ │ │ + * is null or undefined. To use another tileset you can have something │ │ │ │ │ + * like this: │ │ │ │ │ + * (code) │ │ │ │ │ + * new OpenLayers.Layer.OSM("OpenCycleMap", │ │ │ │ │ + * ["http://a.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ + * "http://b.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png", │ │ │ │ │ + * "http://c.tile.opencyclemap.org/cycle/${z}/${x}/${y}.png"]); │ │ │ │ │ + * (end) │ │ │ │ │ + */ │ │ │ │ │ + url: [ │ │ │ │ │ + 'http://a.tile.openstreetmap.org/${z}/${x}/${y}.png', │ │ │ │ │ + 'http://b.tile.openstreetmap.org/${z}/${x}/${y}.png', │ │ │ │ │ + 'http://c.tile.openstreetmap.org/${z}/${x}/${y}.png' │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "النسبة = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * Property: attribution │ │ │ │ │ + * {String} The layer attribution. │ │ │ │ │ + */ │ │ │ │ │ + attribution: "© <a href='http://www.openstreetmap.org/copyright'>OpenStreetMap</a> contributors", │ │ │ │ │ │ │ │ │ │ - 'W': "غ", │ │ │ │ │ + /** │ │ │ │ │ + * Property: sphericalMercator │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + sphericalMercator: true, │ │ │ │ │ │ │ │ │ │ - 'E': "شر", │ │ │ │ │ + /** │ │ │ │ │ + * Property: wrapDateLine │ │ │ │ │ + * {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + wrapDateLine: true, │ │ │ │ │ │ │ │ │ │ - 'N': "شم", │ │ │ │ │ + /** APIProperty: tileOptions │ │ │ │ │ + * {Object} optional configuration options for <OpenLayers.Tile> instances │ │ │ │ │ + * created by this Layer. Default is │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * {crossOriginKeyword: 'anonymous'} │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * When using OSM tilesets other than the default ones, it may be │ │ │ │ │ + * necessary to set this to │ │ │ │ │ + * │ │ │ │ │ + * (code) │ │ │ │ │ + * {crossOriginKeyword: null} │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * if the server does not send Access-Control-Allow-Origin headers. │ │ │ │ │ + */ │ │ │ │ │ + tileOptions: null, │ │ │ │ │ │ │ │ │ │ - 'S': "ج" │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.OSM │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} The layer name. │ │ │ │ │ + * url - {String} The tileset URL scheme. │ │ │ │ │ + * options - {Object} Configuration options for the layer. Any inherited │ │ │ │ │ + * layer option can be set in this object (e.g. │ │ │ │ │ + * <OpenLayers.Layer.Grid.buffer>). │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + crossOriginKeyword: 'anonymous' │ │ │ │ │ + }, this.options && this.options.tileOptions); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clone │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.OSM( │ │ │ │ │ + this.name, this.url, this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.OSM" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/ia.js │ │ │ │ │ + OpenLayers/Layer/WorldWind.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - McDutchie │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["ia"] │ │ │ │ │ - * Dictionary for Interlingua. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Layer.WorldWind │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["ia"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - │ │ │ │ │ - 'unhandledRequest': "Le responsa a un requesta non esseva maneate: ${statusText}", │ │ │ │ │ - │ │ │ │ │ - 'Permalink': "Permaligamine", │ │ │ │ │ - │ │ │ │ │ - 'Overlays': "Superpositiones", │ │ │ │ │ - │ │ │ │ │ - 'Base Layer': "Strato de base", │ │ │ │ │ - │ │ │ │ │ - 'noFID': "Non pote actualisar un elemento sin FID.", │ │ │ │ │ - │ │ │ │ │ - 'browserNotSupported': "Tu navigator non supporta le rendition de vectores. Le renditores actualmente supportate es:\n${renderers}", │ │ │ │ │ - │ │ │ │ │ - 'minZoomLevelError': "Le proprietate minZoomLevel es solmente pro uso con le stratos descendente de FixedZoomLevels. Le facto que iste strato WFS verifica minZoomLevel es un reliquia del passato. Nonobstante, si nos lo remove immediatemente, nos pote rumper applicationes a base de OL que depende de illo. Ergo nos lo declara obsolete; le verification de minZoomLevel in basso essera removite in version 3.0. Per favor usa in su loco le configuration de resolutiones min/max como describite a: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - │ │ │ │ │ - 'commitSuccess': "Transaction WFS: SUCCESSO ${response}", │ │ │ │ │ - │ │ │ │ │ - 'commitFailed': "Transaction WFS: FALLEVA ${response}", │ │ │ │ │ - │ │ │ │ │ - 'googleWarning': "Le strato Google non poteva esser cargate correctemente.\x3cbr\x3e\x3cbr\x3ePro disfacer te de iste message, selige un nove BaseLayer in le selector de strato in alto a dextra.\x3cbr\x3e\x3cbr\x3eMulto probabilemente, isto es proque le script del libreria de Google Maps non esseva includite o non contine le clave API correcte pro tu sito.\x3cbr\x3e\x3cbr\x3eDisveloppatores: Pro adjuta de corriger isto, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eclicca hic\x3c/a", │ │ │ │ │ +OpenLayers.Layer.WorldWind = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "Le strato ${layerType} non poteva esser cargate correctemente.\x3cbr\x3e\x3cbr\x3ePro disfacer te de iste message, selige un nove BaseLayer in le selector de strato in alto a dextra.\x3cbr\x3e\x3cbr\x3eMulto probabilemente, isto es proque le script del libreria de ${layerLib} non esseva correctemente includite.\x3cbr\x3e\x3cbr\x3eDisveloppatores: Pro adjuta de corriger isto, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eclicca hic\x3c/a\x3e", │ │ │ │ │ + DEFAULT_PARAMS: {}, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Scala = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} WorldWind layer is a base layer by default. │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ - 'W': "W", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: lzd │ │ │ │ │ + * {Float} LevelZeroTileSizeDegrees │ │ │ │ │ + */ │ │ │ │ │ + lzd: null, │ │ │ │ │ │ │ │ │ │ - 'E': "E", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: zoomLevels │ │ │ │ │ + * {Integer} Number of zoom levels. │ │ │ │ │ + */ │ │ │ │ │ + zoomLevels: null, │ │ │ │ │ │ │ │ │ │ - 'N': "N", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.WorldWind │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} Name of Layer │ │ │ │ │ + * url - {String} Base URL │ │ │ │ │ + * lzd - {Float} Level zero tile size degrees │ │ │ │ │ + * zoomLevels - {Integer} number of zoom levels │ │ │ │ │ + * params - {Object} additional parameters │ │ │ │ │ + * options - {Object} additional options │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, lzd, zoomLevels, params, options) { │ │ │ │ │ + this.lzd = lzd; │ │ │ │ │ + this.zoomLevels = zoomLevels; │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + newArguments.push(name, url, params, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + this.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, this.DEFAULT_PARAMS │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'S': "S", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getZoom │ │ │ │ │ + * Convert map zoom to WW zoom. │ │ │ │ │ + */ │ │ │ │ │ + getZoom: function() { │ │ │ │ │ + var zoom = this.map.getZoom(); │ │ │ │ │ + var extent = this.map.getMaxExtent(); │ │ │ │ │ + zoom = zoom - Math.log(this.maxResolution / (this.lzd / 512)) / Math.log(2); │ │ │ │ │ + return zoom; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Tu usa le option \'reproject\' in le strato ${layerName} layer. Iste option es obsolescente: illo esseva pro poter monstrar datos super cartas de base commercial, ma iste functionalitate pote ora esser attingite con le uso de Spherical Mercator. Ulterior information es disponibile a http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as │ │ │ │ │ + * parameters │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var zoom = this.getZoom(); │ │ │ │ │ + var extent = this.map.getMaxExtent(); │ │ │ │ │ + var deg = this.lzd / Math.pow(2, this.getZoom()); │ │ │ │ │ + var x = Math.floor((bounds.left - extent.left) / deg); │ │ │ │ │ + var y = Math.floor((bounds.bottom - extent.bottom) / deg); │ │ │ │ │ + if (this.map.getResolution() <= (this.lzd / 512) && │ │ │ │ │ + this.getZoom() <= this.zoomLevels) { │ │ │ │ │ + return this.getFullRequestString({ │ │ │ │ │ + L: zoom, │ │ │ │ │ + X: x, │ │ │ │ │ + Y: y │ │ │ │ │ + }); │ │ │ │ │ + } else { │ │ │ │ │ + return OpenLayers.Util.getImageLocation("blank.gif"); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Iste methodo ha essite declarate obsolescente e essera removite in version 3.0. Per favor usa ${newMethod} in su loco." │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.WorldWind" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/zh-CN.js │ │ │ │ │ + OpenLayers/Layer/KaMap.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["zh-CN"] │ │ │ │ │ - * Dictionary for Simplified Chinese. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Lang["zh-CN"] = { │ │ │ │ │ - │ │ │ │ │ - 'unhandledRequest': "未处理的请求,返回值为 ${statusText}", │ │ │ │ │ - │ │ │ │ │ - 'Permalink': "永久链接", │ │ │ │ │ - │ │ │ │ │ - 'Overlays': "叠加层", │ │ │ │ │ - │ │ │ │ │ - 'Base Layer': "基础图层", │ │ │ │ │ - │ │ │ │ │ - 'noFID': "无法更新feature,缺少FID。", │ │ │ │ │ - │ │ │ │ │ - 'browserNotSupported': "你使用的浏览器不支持矢量渲染。当前支持的渲染方式包括:\n${renderers}", │ │ │ │ │ - │ │ │ │ │ - // console message │ │ │ │ │ - 'minZoomLevelError': "minZoomLevel属性仅适合用于" + │ │ │ │ │ - "使用了固定缩放级别的图层。这个 " + │ │ │ │ │ - "wfs 图层检查 minZoomLevel 是过去遗留下来的。" + │ │ │ │ │ - "然而,我们不能移除它," + │ │ │ │ │ - "而破坏依赖于它的基于OL的应用程序。" + │ │ │ │ │ - "因此,我们废除了它 -- minZoomLevel " + │ │ │ │ │ - "将会在3.0中被移除。请改用 " + │ │ │ │ │ - "min/max resolution 设置,参考:" + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - │ │ │ │ │ - 'commitSuccess': "WFS Transaction: 成功。 ${response}", │ │ │ │ │ - │ │ │ │ │ - 'commitFailed': "WFS Transaction: 失败。 ${response}", │ │ │ │ │ - │ │ │ │ │ - 'googleWarning': "Google图层不能正确加载。<br><br>" + │ │ │ │ │ - "要消除这个信息,请在右上角的" + │ │ │ │ │ - "图层控制面板中选择其他的基础图层。<br><br>" + │ │ │ │ │ - "这种情况很可能是没有正确的包含Google地图脚本库," + │ │ │ │ │ - "或者是没有包含在你的站点上" + │ │ │ │ │ - "使用的正确的Google Maps API密匙。<br><br>" + │ │ │ │ │ - "开发者:获取使其正确工作的帮助信息," + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ - "target='_blank'>点击这里</a>", │ │ │ │ │ - │ │ │ │ │ - 'getLayerWarning': "${layerType} 图层不能正确加载。<br><br>" + │ │ │ │ │ - "要消除这个信息,请在右上角的" + │ │ │ │ │ - "图层控制面板中选择其他的基础图层。<br><br>" + │ │ │ │ │ - "这种情况很可能是没有正确的包含" + │ │ │ │ │ - "${layerLib} 脚本库。<br><br>" + │ │ │ │ │ - "开发者:获取使其正确工作的帮助信息," + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ - "target='_blank'>点击这里</a>", │ │ │ │ │ - │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "比例尺 = 1 : ${scaleDenom}", │ │ │ │ │ - │ │ │ │ │ - // console message │ │ │ │ │ - 'reprojectDeprecated': "你正在使用 ${layerName} 图层上的'reproject'选项。" + │ │ │ │ │ - "这个选项已经不再使用:" + │ │ │ │ │ - "它是被设计用来支持显示商业的地图数据," + │ │ │ │ │ - "不过现在该功能可以通过使用Spherical Mercator来实现。" + │ │ │ │ │ - "更多信息可以参阅" + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - │ │ │ │ │ - // console message │ │ │ │ │ - 'methodDeprecated': "该方法已经不再被支持,并且将在3.0中被移除。" + │ │ │ │ │ - "请使用 ${newMethod} 方法来替代。", │ │ │ │ │ - │ │ │ │ │ - 'end': '' │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/pt-BR.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Luckas Blade │ │ │ │ │ - * - Rodrigo Avila │ │ │ │ │ - */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["pt-br"] │ │ │ │ │ - * Dictionary for Português do Brasil. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Layer.KaMap │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["pt-BR"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - │ │ │ │ │ - 'unhandledRequest': "A requisição retornou um erro não tratado: ${statusText}", │ │ │ │ │ - │ │ │ │ │ - 'Permalink': "Link para essa página", │ │ │ │ │ +OpenLayers.Layer.KaMap = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Camadas de Sobreposição", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} KaMap Layer is always a base layer │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Camada Base", │ │ │ │ │ + /** │ │ │ │ │ + * Constant: DEFAULT_PARAMS │ │ │ │ │ + * {Object} parameters set by default. The default parameters set │ │ │ │ │ + * the format via the 'i' parameter to 'jpeg'. │ │ │ │ │ + */ │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + i: 'jpeg', │ │ │ │ │ + map: '' │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'noFID': "Não é possível atualizar uma feição que não tenha um FID.", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.KaMap │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} │ │ │ │ │ + * url - {String} │ │ │ │ │ + * params - {Object} Parameters to be sent to the HTTP server in the │ │ │ │ │ + * query string for the tile. The format can be set via the 'i' │ │ │ │ │ + * parameter (defaults to jpg) , and the map should be set via │ │ │ │ │ + * the 'map' parameter. It has been reported that ka-Map may behave │ │ │ │ │ + * inconsistently if your format parameter does not match the format │ │ │ │ │ + * parameter configured in your config.php. (See ticket #327 for more │ │ │ │ │ + * information.) │ │ │ │ │ + * options - {Object} Additional options for the layer. Any of the │ │ │ │ │ + * APIProperties listed on this layer, and any layer types it │ │ │ │ │ + * extends, can be overridden through the options parameter. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, this.DEFAULT_PARAMS │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Seu navegador não suporta renderização de vetores. Os renderizadores suportados atualmente são:\n${renderers}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as │ │ │ │ │ + * parameters │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var mapRes = this.map.getResolution(); │ │ │ │ │ + var scale = Math.round((this.map.getScale() * 10000)) / 10000; │ │ │ │ │ + var pX = Math.round(bounds.left / mapRes); │ │ │ │ │ + var pY = -Math.round(bounds.top / mapRes); │ │ │ │ │ + return this.getFullRequestString({ │ │ │ │ │ + t: pY, │ │ │ │ │ + l: pX, │ │ │ │ │ + s: scale │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'minZoomLevelError': "A propriedade minZoomLevel é de uso restrito das camadas descendentes de FixedZoomLevels. A verificação dessa propriedade pelas camadas wfs é um resíduo do passado. Não podemos, entretanto não é possível removê-la sem possívelmente quebrar o funcionamento de aplicações OL que possuem depência com ela. Portanto estamos tornando seu uso obsoleto -- a verificação desse atributo será removida na versão 3.0. Ao invés, use as opções de resolução min/max como descrito em: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateGridLayout │ │ │ │ │ + * ka-Map uses the center point of the map as an origin for │ │ │ │ │ + * its tiles. Override calculateGridLayout to center tiles │ │ │ │ │ + * correctly for this case. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bound>} │ │ │ │ │ + * origin - {<OpenLayers.LonLat>} │ │ │ │ │ + * resolution - {Number} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Object containing properties tilelon, tilelat, startcol, │ │ │ │ │ + * startrow │ │ │ │ │ + */ │ │ │ │ │ + calculateGridLayout: function(bounds, origin, resolution) { │ │ │ │ │ + var tilelon = resolution * this.tileSize.w; │ │ │ │ │ + var tilelat = resolution * this.tileSize.h; │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "Transação WFS : SUCESSO ${response}", │ │ │ │ │ + var offsetlon = bounds.left; │ │ │ │ │ + var tilecol = Math.floor(offsetlon / tilelon) - this.buffer; │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "Transação WFS : ERRO ${response}", │ │ │ │ │ + var offsetlat = bounds.top; │ │ │ │ │ + var tilerow = Math.floor(offsetlat / tilelat) + this.buffer; │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Não foi possível carregar a camada Google corretamente.\x3cbr\x3e\x3cbr\x3ePara se livrar dessa mensagem, selecione uma nova Camada Base, na ferramenta de alternação de camadas localização do canto superior direito.\x3cbr\x3e\x3cbr\x3eMuito provavelmente, isso foi causado porque o script da biblioteca do Google Maps não foi incluído, ou porque ele não contém a chave correta da API para o seu site.\x3cbr\x3e\x3cbr\x3eDesenvolvedores: Para obter ajuda em solucionar esse problema \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3ecliquem aqui\x3c/a\x3e", │ │ │ │ │ + return { │ │ │ │ │ + tilelon: tilelon, │ │ │ │ │ + tilelat: tilelat, │ │ │ │ │ + startcol: tilecol, │ │ │ │ │ + startrow: tilerow │ │ │ │ │ + }; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "Não foi possível carregar a camada ${layerType} corretamente.\x3cbr\x3e\x3cbr\x3ePara se livrar dessa mensagem, selecione uma nova Camada Base, na ferramenta de alternação de camadas localização do canto superior direito.\x3cbr\x3e\x3cbr\x3eMuito provavelmente, isso foi causado porque o script da biblioteca ${layerLib} não foi incluído corretamente.\x3cbr\x3e\x3cbr\x3eDesenvolvedores: Para obter ajuda em solucionar esse problema \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3ecliquem aqui\x3c/a\x3e", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getTileBoundsForGridIndex │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * row - {Number} The row of the grid │ │ │ │ │ + * col - {Number} The column of the grid │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} The bounds for the tile at (row, col) │ │ │ │ │ + */ │ │ │ │ │ + getTileBoundsForGridIndex: function(row, col) { │ │ │ │ │ + var origin = this.getTileOrigin(); │ │ │ │ │ + var tileLayout = this.gridLayout; │ │ │ │ │ + var tilelon = tileLayout.tilelon; │ │ │ │ │ + var tilelat = tileLayout.tilelat; │ │ │ │ │ + var minX = (tileLayout.startcol + col) * tilelon; │ │ │ │ │ + var minY = (tileLayout.startrow - row) * tilelat; │ │ │ │ │ + return new OpenLayers.Bounds( │ │ │ │ │ + minX, minY, │ │ │ │ │ + minX + tilelon, minY + tilelat │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Escala = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Kamap>} An exact clone of this OpenLayers.Layer.KaMap │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ │ │ │ │ │ - 'W': "O", │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.KaMap(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.params, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'E': "L", │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ │ │ │ │ │ - 'N': "N", │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ + if (this.tileSize != null) { │ │ │ │ │ + obj.tileSize = this.tileSize.clone(); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'S': "S", │ │ │ │ │ + // we do not want to copy reference to grid, so we make a new array │ │ │ │ │ + obj.grid = []; │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Você está usando a opção \'reproject\' na camada ${layerName}. Essa opção está obsoleta: seu uso foi projetado para suportar a visualização de dados sobre bases de mapas comerciais, entretanto essa funcionalidade deve agora ser alcançada usando o suporte à projeção Mercator. Mais informação está disponível em: http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Esse método está obsoleto e será removido na versão 3.0. Ao invés, por favor use ${newMethod}." │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getTileBounds │ │ │ │ │ + * Returns The tile bounds for a layer given a pixel location. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * viewPortPx - {<OpenLayers.Pixel>} The location in the viewport. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} Bounds of the tile at the given pixel location. │ │ │ │ │ + */ │ │ │ │ │ + getTileBounds: function(viewPortPx) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var tileMapWidth = resolution * this.tileSize.w; │ │ │ │ │ + var tileMapHeight = resolution * this.tileSize.h; │ │ │ │ │ + var mapPoint = this.getLonLatFromViewPortPx(viewPortPx); │ │ │ │ │ + var tileLeft = tileMapWidth * Math.floor(mapPoint.lon / tileMapWidth); │ │ │ │ │ + var tileBottom = tileMapHeight * Math.floor(mapPoint.lat / tileMapHeight); │ │ │ │ │ + return new OpenLayers.Bounds(tileLeft, tileBottom, │ │ │ │ │ + tileLeft + tileMapWidth, │ │ │ │ │ + tileBottom + tileMapHeight); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.KaMap" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/fi.js │ │ │ │ │ + OpenLayers/Layer/KaMapCache.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Nike │ │ │ │ │ - * - Str4nd │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["fi"] │ │ │ │ │ - * Dictionary for Suomi. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Lang["fi"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - │ │ │ │ │ - 'Permalink': "Ikilinkki", │ │ │ │ │ - │ │ │ │ │ - 'Overlays': "Kerrokset", │ │ │ │ │ - │ │ │ │ │ - 'Base Layer': "Peruskerros", │ │ │ │ │ - │ │ │ │ │ - 'W': "L", │ │ │ │ │ - │ │ │ │ │ - 'E': "I", │ │ │ │ │ - │ │ │ │ │ - 'N': "P", │ │ │ │ │ - │ │ │ │ │ - 'S': "E" │ │ │ │ │ - │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/nl.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Siebrand │ │ │ │ │ - */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ + * @requires OpenLayers/Layer/KaMap.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["nl"] │ │ │ │ │ - * Dictionary for Nederlands. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Layer.KaMapCache │ │ │ │ │ + * │ │ │ │ │ + * This class is designed to talk directly to a web-accessible ka-Map │ │ │ │ │ + * cache generated by the precache2.php script. │ │ │ │ │ + * │ │ │ │ │ + * To create a a new KaMapCache layer, you must indicate also the "i" parameter │ │ │ │ │ + * (that will be used to calculate the file extension), and another special │ │ │ │ │ + * parameter, object names "metaTileSize", with "h" (height) and "w" (width) │ │ │ │ │ + * properties. │ │ │ │ │ + * │ │ │ │ │ + * // Create a new kaMapCache layer. │ │ │ │ │ + * var kamap_base = new OpenLayers.Layer.KaMapCache( │ │ │ │ │ + * "Satellite", │ │ │ │ │ + * "http://www.example.org/web/acessible/cache", │ │ │ │ │ + * {g: "satellite", map: "world", i: 'png24', metaTileSize: {w: 5, h: 5} } │ │ │ │ │ + * ); │ │ │ │ │ + * │ │ │ │ │ + * // Create an kaMapCache overlay layer (using "isBaseLayer: false"). │ │ │ │ │ + * // Forces the output to be a "gif", using the "i" parameter. │ │ │ │ │ + * var kamap_overlay = new OpenLayers.Layer.KaMapCache( │ │ │ │ │ + * "Streets", │ │ │ │ │ + * "http://www.example.org/web/acessible/cache", │ │ │ │ │ + * {g: "streets", map: "world", i: "gif", metaTileSize: {w: 5, h: 5} }, │ │ │ │ │ + * {isBaseLayer: false} │ │ │ │ │ + * ); │ │ │ │ │ + * │ │ │ │ │ + * The cache URLs must look like: │ │ │ │ │ + * var/cache/World/50000/Group_Name/def/t-440320/l20480 │ │ │ │ │ + * │ │ │ │ │ + * This means that the cache generated via tile.php will *not* work with │ │ │ │ │ + * this class, and should instead use the KaMap layer. │ │ │ │ │ + * │ │ │ │ │ + * More information is available in Ticket #1518. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.KaMap> │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["nl"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - │ │ │ │ │ - 'unhandledRequest': "Het verzoek is niet afgehandeld met de volgende melding: ${statusText}", │ │ │ │ │ - │ │ │ │ │ - 'Permalink': "Permanente verwijzing", │ │ │ │ │ - │ │ │ │ │ - 'Overlays': "Overlays", │ │ │ │ │ - │ │ │ │ │ - 'Base Layer': "Achtergrondkaart", │ │ │ │ │ - │ │ │ │ │ - 'noFID': "Een optie die geen FID heeft kan niet bijgewerkt worden.", │ │ │ │ │ - │ │ │ │ │ - 'browserNotSupported': "Uw browser ondersteunt het weergeven van vectoren niet.\nMomenteel ondersteunde weergavemogelijkheden:\n${renderers}", │ │ │ │ │ - │ │ │ │ │ - 'minZoomLevelError': "De eigenschap minZoomLevel is alleen bedoeld voor gebruik lagen met die afstammen van FixedZoomLevels-lagen.\nDat deze WFS-laag minZoomLevel controleert, is een overblijfsel uit het verleden.\nWe kunnen deze controle echter niet verwijderen zonder op OL gebaseerde applicaties die hervan afhankelijk zijn stuk te maken.\nDaarom heeft deze functionaliteit de eigenschap \'deprecated\' gekregen - de minZoomLevel wordt verwijderd in versie 3.0.\nGebruik in plaats van deze functie de mogelijkheid om min/max voor resolutie in te stellen zoals op de volgende pagina wordt beschreven:\nhttp://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - │ │ │ │ │ - 'commitSuccess': "WFS-transactie: succesvol ${response}", │ │ │ │ │ - │ │ │ │ │ - 'commitFailed': "WFS-transactie: mislukt ${response}", │ │ │ │ │ - │ │ │ │ │ - 'googleWarning': "De Google-Layer kon niet correct geladen worden.\x3cbr /\x3e\x3cbr /\x3e\nOm deze melding niet meer te krijgen, moet u een andere achtergrondkaart kiezen in de laagwisselaar in de rechterbovenhoek.\x3cbr /\x3e\x3cbr /\x3e\nDit komt waarschijnlijk doordat de bibliotheek ${layerLib} niet correct ingevoegd is.\x3cbr /\x3e\x3cbr /\x3e\nOntwikkelaars: \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eklik hier\x3c/a\x3e om dit werkend te krijgen.", │ │ │ │ │ +OpenLayers.Layer.KaMapCache = OpenLayers.Class(OpenLayers.Layer.KaMap, { │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "De laag ${layerType} kon niet goed geladen worden.\x3cbr /\x3e\x3cbr /\x3e\nOm deze melding niet meer te krijgen, moet u een andere achtergrondkaart kiezen in de laagwisselaar in de rechterbovenhoek.\x3cbr /\x3e\x3cbr /\x3e\nDit komt waarschijnlijk doordat de bibliotheek ${layerLib} niet correct is ingevoegd.\x3cbr /\x3e\x3cbr /\x3e\nOntwikkelaars: \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eklik hier\x3c/a\x3e om dit werkend te krijgen.", │ │ │ │ │ + /** │ │ │ │ │ + * Constant: IMAGE_EXTENSIONS │ │ │ │ │ + * {Object} Simple hash map to convert format to extension. │ │ │ │ │ + */ │ │ │ │ │ + IMAGE_EXTENSIONS: { │ │ │ │ │ + 'jpeg': 'jpg', │ │ │ │ │ + 'gif': 'gif', │ │ │ │ │ + 'png': 'png', │ │ │ │ │ + 'png8': 'png', │ │ │ │ │ + 'png24': 'png', │ │ │ │ │ + 'dithered': 'png' │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Schaal = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * Constant: DEFAULT_FORMAT │ │ │ │ │ + * {Object} Simple hash map to convert format to extension. │ │ │ │ │ + */ │ │ │ │ │ + DEFAULT_FORMAT: 'jpeg', │ │ │ │ │ │ │ │ │ │ - 'W': "W", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.KaMapCache │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} │ │ │ │ │ + * url - {String} │ │ │ │ │ + * params - {Object} Parameters to be sent to the HTTP server in the │ │ │ │ │ + * query string for the tile. The format can be set via the 'i' │ │ │ │ │ + * parameter (defaults to jpg) , and the map should be set via │ │ │ │ │ + * the 'map' parameter. It has been reported that ka-Map may behave │ │ │ │ │ + * inconsistently if your format parameter does not match the format │ │ │ │ │ + * parameter configured in your config.php. (See ticket #327 for more │ │ │ │ │ + * information.) │ │ │ │ │ + * options - {Object} Additional options for the layer. Any of the │ │ │ │ │ + * APIProperties listed on this layer, and any layer types it │ │ │ │ │ + * extends, can be overridden through the options parameter. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.KaMap.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.extension = this.IMAGE_EXTENSIONS[this.params.i.toLowerCase() || this.DEFAULT_FORMAT]; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'E': "O", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the layer's url and parameters and also the │ │ │ │ │ + * passed-in bounds and appropriate tile size specified as │ │ │ │ │ + * parameters │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var mapRes = this.map.getResolution(); │ │ │ │ │ + var scale = Math.round((this.map.getScale() * 10000)) / 10000; │ │ │ │ │ + var pX = Math.round(bounds.left / mapRes); │ │ │ │ │ + var pY = -Math.round(bounds.top / mapRes); │ │ │ │ │ │ │ │ │ │ - 'N': "N", │ │ │ │ │ + var metaX = Math.floor(pX / this.tileSize.w / this.params.metaTileSize.w) * this.tileSize.w * this.params.metaTileSize.w; │ │ │ │ │ + var metaY = Math.floor(pY / this.tileSize.h / this.params.metaTileSize.h) * this.tileSize.h * this.params.metaTileSize.h; │ │ │ │ │ │ │ │ │ │ - 'S': "Z", │ │ │ │ │ + var components = [ │ │ │ │ │ + "/", │ │ │ │ │ + this.params.map, │ │ │ │ │ + "/", │ │ │ │ │ + scale, │ │ │ │ │ + "/", │ │ │ │ │ + this.params.g.replace(/\s/g, '_'), │ │ │ │ │ + "/def/t", │ │ │ │ │ + metaY, │ │ │ │ │ + "/l", │ │ │ │ │ + metaX, │ │ │ │ │ + "/t", │ │ │ │ │ + pY, │ │ │ │ │ + "l", │ │ │ │ │ + pX, │ │ │ │ │ + ".", │ │ │ │ │ + this.extension │ │ │ │ │ + ]; │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "U gebruikt de optie \'reproject\' op de laag ${layerName}.\nDeze optie is vervallen: deze optie was ontwikkeld om gegevens over commerciële basiskaarten weer te geven, maar deze functionaliteit wordt nu bereikt door ondersteuning van Spherical Mercator.\nMeer informatie is beschikbaar op http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + var url = this.url; │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Deze methode is verouderd en wordt verwijderd in versie 3.0.\nGebruik ${newMethod}." │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(components.join(''), url); │ │ │ │ │ + } │ │ │ │ │ + return url + components.join(""); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.KaMapCache" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/fr.js │ │ │ │ │ + OpenLayers/Layer/ArcGIS93Rest.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Damouns │ │ │ │ │ - * - IAlex │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Layer/Grid.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["fr"] │ │ │ │ │ - * Dictionary for Français. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Layer.ArcGIS93Rest │ │ │ │ │ + * Instances of OpenLayers.Layer.ArcGIS93Rest are used to display data from │ │ │ │ │ + * ESRI ArcGIS Server 9.3 (and up?) Mapping Services using the REST API. │ │ │ │ │ + * Create a new ArcGIS93Rest layer with the <OpenLayers.Layer.ArcGIS93Rest> │ │ │ │ │ + * constructor. More detail on the REST API is available at │ │ │ │ │ + * http://sampleserver1.arcgisonline.com/ArcGIS/SDK/REST/index.html ; │ │ │ │ │ + * specifically, the URL provided to this layer should be an export service │ │ │ │ │ + * URL: http://sampleserver1.arcgisonline.com/ArcGIS/SDK/REST/export.html │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.Grid> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["fr"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ +OpenLayers.Layer.ArcGIS93Rest = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "Requête non gérée, retournant ${statusText}", │ │ │ │ │ + /** │ │ │ │ │ + * Constant: DEFAULT_PARAMS │ │ │ │ │ + * {Object} Hashtable of default parameter key/value pairs │ │ │ │ │ + */ │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + format: "png" │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Permalien", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * {Boolean} Default is true for ArcGIS93Rest layer │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Calques", │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Calque de base", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.ArcGIS93Rest │ │ │ │ │ + * Create a new ArcGIS93Rest layer object. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * (code) │ │ │ │ │ + * var arcims = new OpenLayers.Layer.ArcGIS93Rest("MyName", │ │ │ │ │ + * "http://sampleserver1.arcgisonline.com/ArcGIS/rest/services/Specialty/ESRI_StateCityHighway_USA/MapServer/export", │ │ │ │ │ + * { │ │ │ │ │ + * layers: "0,1,2" │ │ │ │ │ + * }); │ │ │ │ │ + * (end) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} A name for the layer │ │ │ │ │ + * url - {String} Base url for the ArcGIS server REST service │ │ │ │ │ + * options - {Object} An object with key/value pairs representing the │ │ │ │ │ + * options and option values. │ │ │ │ │ + * │ │ │ │ │ + * Valid Options: │ │ │ │ │ + * format - {String} MIME type of desired image type. │ │ │ │ │ + * layers - {String} Comma-separated list of layers to display. │ │ │ │ │ + * srs - {String} Projection ID. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + //uppercase params │ │ │ │ │ + params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ + newArguments.push(name, url, params, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + OpenLayers.Util.applyDefaults( │ │ │ │ │ + this.params, │ │ │ │ │ + OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS) │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - 'noFID': "Impossible de mettre à jour un objet sans identifiant (fid).", │ │ │ │ │ + //layer is transparent │ │ │ │ │ + if (this.params.TRANSPARENT && │ │ │ │ │ + this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Votre navigateur ne supporte pas le rendu vectoriel. Les renderers actuellement supportés sont : \n${renderers}", │ │ │ │ │ + // unless explicitly set in options, make layer an overlay │ │ │ │ │ + if ((options == null) || (!options.isBaseLayer)) { │ │ │ │ │ + this.isBaseLayer = false; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'minZoomLevelError': "La propriété minZoomLevel doit seulement être utilisée pour des couches FixedZoomLevels-descendent. Le fait que cette couche WFS vérifie la présence de minZoomLevel est une relique du passé. Nous ne pouvons toutefois la supprimer sans casser des applications qui pourraient en dépendre. C\'est pourquoi nous la déprécions -- la vérification du minZoomLevel sera supprimée en version 3.0. A la place, merci d\'utiliser les paramètres de résolutions min/max tel que décrit sur : http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + // jpegs can never be transparent, so intelligently switch the │ │ │ │ │ + // format, depending on the browser's capabilities │ │ │ │ │ + if (this.params.FORMAT == "jpg") { │ │ │ │ │ + this.params.FORMAT = OpenLayers.Util.alphaHack() ? "gif" : │ │ │ │ │ + "png"; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "Transaction WFS : SUCCES ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.ArcGIS93Rest>} An exact clone of this layer │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "Transaction WFS : ECHEC ${response}", │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.ArcGIS93Rest(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.params, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "La couche Google n\'a pas été en mesure de se charger correctement.\x3cbr\x3e\x3cbr\x3ePour supprimer ce message, choisissez une nouvelle BaseLayer dans le sélecteur de couche en haut à droite.\x3cbr\x3e\x3cbr\x3eCela est possiblement causé par la non-inclusion de la librairie Google Maps, ou alors parce que la clé de l\'API ne correspond pas à votre site.\x3cbr\x3e\x3cbr\x3eDéveloppeurs : pour savoir comment corriger ceci, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3ecliquez ici\x3c/a\x3e", │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "La couche ${layerType} n\'est pas en mesure de se charger correctement.\x3cbr\x3e\x3cbr\x3ePour supprimer ce message, choisissez une nouvelle BaseLayer dans le sélecteur de couche en haut à droite.\x3cbr\x3e\x3cbr\x3eCela est possiblement causé par la non-inclusion de la librairie ${layerLib}.\x3cbr\x3e\x3cbr\x3eDéveloppeurs : pour savoir comment corriger ceci, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3ecliquez ici\x3c/a\x3e", │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Echelle ~ 1 : ${scaleDenom}", │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'W': "O", │ │ │ │ │ │ │ │ │ │ - 'E': "E", │ │ │ │ │ + /** │ │ │ │ │ + * Method: getURL │ │ │ │ │ + * Return an image url this layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} A bounds representing the bbox for the │ │ │ │ │ + * request. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} A string with the map image's url. │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ │ │ │ │ │ - 'N': "N", │ │ │ │ │ + // ArcGIS Server only wants the numeric portion of the projection ID. │ │ │ │ │ + var projWords = this.projection.getCode().split(":"); │ │ │ │ │ + var srid = projWords[projWords.length - 1]; │ │ │ │ │ │ │ │ │ │ - 'S': "S", │ │ │ │ │ + var imageSize = this.getImageSize(); │ │ │ │ │ + var newParams = { │ │ │ │ │ + 'BBOX': bounds.toBBOX(), │ │ │ │ │ + 'SIZE': imageSize.w + "," + imageSize.h, │ │ │ │ │ + // We always want image, the other options were json, image with a whole lotta html around it, etc. │ │ │ │ │ + 'F': "image", │ │ │ │ │ + 'BBOXSR': srid, │ │ │ │ │ + 'IMAGESR': srid │ │ │ │ │ + }; │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Vous utilisez l\'option \'reproject\' sur la couche ${layerName}. Cette option est dépréciée : Son usage permettait d\'afficher des données au dessus de couches raster commerciales.Cette fonctionalité est maintenant supportée en utilisant le support de la projection Mercator Sphérique. Plus d\'information est disponible sur http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + // Now add the filter parameters. │ │ │ │ │ + if (this.layerDefs) { │ │ │ │ │ + var layerDefStrList = []; │ │ │ │ │ + var layerID; │ │ │ │ │ + for (layerID in this.layerDefs) { │ │ │ │ │ + if (this.layerDefs.hasOwnProperty(layerID)) { │ │ │ │ │ + if (this.layerDefs[layerID]) { │ │ │ │ │ + layerDefStrList.push(layerID); │ │ │ │ │ + layerDefStrList.push(":"); │ │ │ │ │ + layerDefStrList.push(this.layerDefs[layerID]); │ │ │ │ │ + layerDefStrList.push(";"); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (layerDefStrList.length > 0) { │ │ │ │ │ + newParams['LAYERDEFS'] = layerDefStrList.join(""); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var requestString = this.getFullRequestString(newParams); │ │ │ │ │ + return requestString; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Cette méthode est dépréciée, et sera supprimée à la version 3.0. Merci d\'utiliser ${newMethod} à la place." │ │ │ │ │ + /** │ │ │ │ │ + * Method: setLayerFilter │ │ │ │ │ + * addTile creates a tile, initializes it, and adds it to the layer div. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} The id of the layer to which the filter applies. │ │ │ │ │ + * queryDef - {String} A sql-ish query filter, for more detail see the ESRI │ │ │ │ │ + * documentation at http://sampleserver1.arcgisonline.com/ArcGIS/SDK/REST/export.html │ │ │ │ │ + */ │ │ │ │ │ + setLayerFilter: function(id, queryDef) { │ │ │ │ │ + if (!this.layerDefs) { │ │ │ │ │ + this.layerDefs = {}; │ │ │ │ │ + } │ │ │ │ │ + if (queryDef) { │ │ │ │ │ + this.layerDefs[id] = queryDef; │ │ │ │ │ + } else { │ │ │ │ │ + delete this.layerDefs[id]; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: clearLayerFilter │ │ │ │ │ + * Clears layer filters, either from a specific layer, │ │ │ │ │ + * or all of them. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} The id of the layer from which to remove any │ │ │ │ │ + * filter. If unspecified/blank, all filters │ │ │ │ │ + * will be removed. │ │ │ │ │ + */ │ │ │ │ │ + clearLayerFilter: function(id) { │ │ │ │ │ + if (id) { │ │ │ │ │ + delete this.layerDefs[id]; │ │ │ │ │ + } else { │ │ │ │ │ + delete this.layerDefs; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: mergeNewParams │ │ │ │ │ + * Catch changeParams and uppercase the new params to be merged in │ │ │ │ │ + * before calling changeParams on the super class. │ │ │ │ │ + * │ │ │ │ │ + * Once params have been changed, the tiles will be reloaded with │ │ │ │ │ + * the new parameters. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newParams - {Object} Hashtable of new params to use │ │ │ │ │ + */ │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ + var newArguments = [upperParams]; │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, │ │ │ │ │ + newArguments); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.ArcGIS93Rest" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/fur.js │ │ │ │ │ + OpenLayers/Layer/Image.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Klenje │ │ │ │ │ - */ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ + * @requires OpenLayers/Layer.js │ │ │ │ │ + * @requires OpenLayers/Tile/Image.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["fur"] │ │ │ │ │ - * Dictionary for Furlan. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Layer.Image │ │ │ │ │ + * Instances of OpenLayers.Layer.Image are used to display data from a web │ │ │ │ │ + * accessible image as a map layer. Create a new image layer with the │ │ │ │ │ + * <OpenLayers.Layer.Image> constructor. │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang["fur"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - │ │ │ │ │ - 'Permalink': "Leam Permanent", │ │ │ │ │ - │ │ │ │ │ - 'Overlays': "Livei parsore", │ │ │ │ │ - │ │ │ │ │ - 'Base Layer': "Livel di base", │ │ │ │ │ - │ │ │ │ │ - 'browserNotSupported': "Il to sgarfadôr nol supuarte la renderizazion vetoriâl. Al moment a son supuartâts:\n${renderers}", │ │ │ │ │ +OpenLayers.Layer.Image = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Scjale = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * Property: isBaseLayer │ │ │ │ │ + * {Boolean} The layer is a base layer. Default is true. Set this property │ │ │ │ │ + * in the layer options │ │ │ │ │ + */ │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ │ │ │ │ │ - 'W': "O", │ │ │ │ │ + /** │ │ │ │ │ + * Property: url │ │ │ │ │ + * {String} URL of the image to use │ │ │ │ │ + */ │ │ │ │ │ + url: null, │ │ │ │ │ │ │ │ │ │ - 'E': "E", │ │ │ │ │ + /** │ │ │ │ │ + * Property: extent │ │ │ │ │ + * {<OpenLayers.Bounds>} The image bounds in map units. This extent will │ │ │ │ │ + * also be used as the default maxExtent for the layer. If you wish │ │ │ │ │ + * to have a maxExtent that is different than the image extent, set the │ │ │ │ │ + * maxExtent property of the options argument (as with any other layer). │ │ │ │ │ + */ │ │ │ │ │ + extent: null, │ │ │ │ │ │ │ │ │ │ - 'N': "N", │ │ │ │ │ + /** │ │ │ │ │ + * Property: size │ │ │ │ │ + * {<OpenLayers.Size>} The image size in pixels │ │ │ │ │ + */ │ │ │ │ │ + size: null, │ │ │ │ │ │ │ │ │ │ - 'S': "S" │ │ │ │ │ + /** │ │ │ │ │ + * Property: tile │ │ │ │ │ + * {<OpenLayers.Tile.Image>} │ │ │ │ │ + */ │ │ │ │ │ + tile: null, │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/id.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: aspectRatio │ │ │ │ │ + * {Float} The ratio of height/width represented by a single pixel in the │ │ │ │ │ + * graphic │ │ │ │ │ + */ │ │ │ │ │ + aspectRatio: null, │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Irwangatot │ │ │ │ │ - * - IvanLanin │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.Image │ │ │ │ │ + * Create a new image layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} A name for the layer. │ │ │ │ │ + * url - {String} Relative or absolute path to the image │ │ │ │ │ + * extent - {<OpenLayers.Bounds>} The extent represented by the image │ │ │ │ │ + * size - {<OpenLayers.Size>} The size (in pixels) of the image │ │ │ │ │ + * options - {Object} Hashtable of extra options to tag onto the layer │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, url, extent, size, options) { │ │ │ │ │ + this.url = url; │ │ │ │ │ + this.extent = extent; │ │ │ │ │ + this.maxExtent = extent; │ │ │ │ │ + this.size = size; │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ + this.aspectRatio = (this.extent.getHeight() / this.size.h) / │ │ │ │ │ + (this.extent.getWidth() / this.size.w); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["id"] │ │ │ │ │ - * Dictionary for Bahasa Indonesia. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Lang["id"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + /** │ │ │ │ │ + * Method: destroy │ │ │ │ │ + * Destroy this layer │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.tile) { │ │ │ │ │ + this.removeTileMonitoringHooks(this.tile); │ │ │ │ │ + this.tile.destroy(); │ │ │ │ │ + this.tile = null; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "Permintaan yang tak tertangani menghasilkan ${statusText}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ + * │ │ │ │ │ + * Paramters: │ │ │ │ │ + * obj - {Object} An optional layer (is this ever used?) │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Image>} An exact copy of this layer │ │ │ │ │ + */ │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Pranala permanen", │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Image(this.name, │ │ │ │ │ + this.url, │ │ │ │ │ + this.extent, │ │ │ │ │ + this.size, │ │ │ │ │ + this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Hamparan", │ │ │ │ │ + //get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Lapisan Dasar", │ │ │ │ │ + // copy/set any non-init, non-simple values here │ │ │ │ │ │ │ │ │ │ - 'noFID': "Tidak dapat memperbarui fitur yang tidak memiliki FID.", │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Peramban Anda tidak mendukung penggambaran vektor. Penggambar yang didukung saat ini adalah:\n${renderers}", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setMap │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + /** │ │ │ │ │ + * If nothing to do with resolutions has been set, assume a single │ │ │ │ │ + * resolution determined by ratio*extent/size - if an image has a │ │ │ │ │ + * pixel aspect ratio different than one (as calculated above), the │ │ │ │ │ + * image will be stretched in one dimension only. │ │ │ │ │ + */ │ │ │ │ │ + if (this.options.maxResolution == null) { │ │ │ │ │ + this.options.maxResolution = this.aspectRatio * │ │ │ │ │ + this.extent.getWidth() / │ │ │ │ │ + this.size.w; │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.prototype.setMap.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'minZoomLevelError': "Properti minZoomLevel hanya ditujukan bekerja dengan lapisan FixedZoomLevels-descendent. Pengecekan minZoomLevel oleh lapisan wfs adalah peninggalan masa lalu. Kami tidak dapat menghapusnya tanpa kemungkinan merusak aplikasi berbasis OL yang mungkin bergantung padanya. Karenanya, kami menganggapnya tidak berlaku -- Cek minZoomLevel di bawah ini akan dihapus pada 3.0. Silakan gunakan penyetelan resolusi min/maks seperti dijabarkan di sini: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * Create the tile for the image or resize it for the new resolution │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * zoomChanged - {Boolean} │ │ │ │ │ + * dragging - {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "WFS Transaksi: BERHASIL ${respon}", │ │ │ │ │ + var firstRendering = (this.tile == null); │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "WFS Transaksi: GAGAL ${respon}", │ │ │ │ │ + if (zoomChanged || firstRendering) { │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Lapisan Google tidak dapat dimuat dengan benar.\x3cbr\x3e\x3cbr\x3eUntuk menghilangkan pesan ini, pilih suatu BaseLayer baru melalui penukar lapisan (layer switcher) di ujung kanan atas.\x3cbr\x3e\x3cbr\x3eKemungkinan besar ini karena pustaka skrip Google Maps tidak disertakan atau tidak mengandung kunci API yang tepat untuk situs Anda.\x3cbr\x3e\x3cbr\x3ePengembang: Untuk bantuan mengatasi masalah ini, \x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3eklik di sini\x3c/a\x3e", │ │ │ │ │ + //determine new tile size │ │ │ │ │ + this.setTileSize(); │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "Lapisan ${layerType} tidak dapat dimuat dengan benar.\x3cbr\x3e\x3cbr\x3eUntuk menghilangkan pesan ini, pilih suatu BaseLayer baru melalui penukar lapisan (layer switcher) di ujung kanan atas.\x3cbr\x3e\x3cbr\x3eKemungkinan besar ini karena pustaka skrip Google Maps tidak disertakan dengan benar.\x3cbr\x3e\x3cbr\x3ePengembang: Untuk bantuan mengatasi masalah ini, \x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3eklik di sini\x3c/a\x3e", │ │ │ │ │ + //determine new position (upper left corner of new bounds) │ │ │ │ │ + var ulPx = this.map.getLayerPxFromLonLat({ │ │ │ │ │ + lon: this.extent.left, │ │ │ │ │ + lat: this.extent.top │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Sekala = 1 : ${scaleDenom}", │ │ │ │ │ + if (firstRendering) { │ │ │ │ │ + //create the new tile │ │ │ │ │ + this.tile = new OpenLayers.Tile.Image(this, ulPx, this.extent, │ │ │ │ │ + null, this.tileSize); │ │ │ │ │ + this.addTileMonitoringHooks(this.tile); │ │ │ │ │ + } else { │ │ │ │ │ + //just resize the tile and set it's new position │ │ │ │ │ + this.tile.size = this.tileSize.clone(); │ │ │ │ │ + this.tile.position = ulPx.clone(); │ │ │ │ │ + } │ │ │ │ │ + this.tile.draw(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'W': "B", │ │ │ │ │ + /** │ │ │ │ │ + * Set the tile size based on the map size. │ │ │ │ │ + */ │ │ │ │ │ + setTileSize: function() { │ │ │ │ │ + var tileWidth = this.extent.getWidth() / this.map.getResolution(); │ │ │ │ │ + var tileHeight = this.extent.getHeight() / this.map.getResolution(); │ │ │ │ │ + this.tileSize = new OpenLayers.Size(tileWidth, tileHeight); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'E': "T", │ │ │ │ │ + /** │ │ │ │ │ + * Method: addTileMonitoringHooks │ │ │ │ │ + * This function takes a tile as input and adds the appropriate hooks to │ │ │ │ │ + * the tile so that the layer can keep track of the loading tiles. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * tile - {<OpenLayers.Tile>} │ │ │ │ │ + */ │ │ │ │ │ + addTileMonitoringHooks: function(tile) { │ │ │ │ │ + tile.onLoadStart = function() { │ │ │ │ │ + this.events.triggerEvent("loadstart"); │ │ │ │ │ + }; │ │ │ │ │ + tile.events.register("loadstart", this, tile.onLoadStart); │ │ │ │ │ │ │ │ │ │ - 'N': "U", │ │ │ │ │ + tile.onLoadEnd = function() { │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + }; │ │ │ │ │ + tile.events.register("loadend", this, tile.onLoadEnd); │ │ │ │ │ + tile.events.register("unload", this, tile.onLoadEnd); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'S': "S", │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeTileMonitoringHooks │ │ │ │ │ + * This function takes a tile as input and removes the tile hooks │ │ │ │ │ + * that were added in <addTileMonitoringHooks>. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * tile - {<OpenLayers.Tile>} │ │ │ │ │ + */ │ │ │ │ │ + removeTileMonitoringHooks: function(tile) { │ │ │ │ │ + tile.unload(); │ │ │ │ │ + tile.events.un({ │ │ │ │ │ + "loadstart": tile.onLoadStart, │ │ │ │ │ + "loadend": tile.onLoadEnd, │ │ │ │ │ + "unload": tile.onLoadEnd, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Anda menggunakan opsi \'reproject\' pada lapisan ${layerName}. Opsi ini telah ditinggalkan: penggunaannya dirancang untuk mendukung tampilan data melalui peta dasar komersial, tapi fungsionalitas tersebut saat ini harus dilakukan dengan menggunakan dukungan Spherical Mercator. Informasi lebih lanjut tersedia di http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setUrl │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * newUrl - {String} │ │ │ │ │ + */ │ │ │ │ │ + setUrl: function(newUrl) { │ │ │ │ │ + this.url = newUrl; │ │ │ │ │ + this.tile.draw(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Metode ini telah usang dan akan dihapus di 3.0. Sebaliknya, harap gunakan ${newMethod}." │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getURL │ │ │ │ │ + * The url we return is always the same (the image itself never changes) │ │ │ │ │ + * so we can ignore the bounds parameter (it will always be the same, │ │ │ │ │ + * anyways) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + */ │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + return this.url; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Image" │ │ │ │ │ }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/it.js │ │ │ │ │ + OpenLayers/Layer/Google.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ + * @requires OpenLayers/Layer/SphericalMercator.js │ │ │ │ │ + * @requires OpenLayers/Layer/EventPane.js │ │ │ │ │ + * @requires OpenLayers/Layer/FixedZoomLevels.js │ │ │ │ │ * @requires OpenLayers/Lang.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Namespace: OpenLayers.Lang["it"] │ │ │ │ │ - * Dictionary for Italian. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ + * Class: OpenLayers.Layer.Google │ │ │ │ │ + * │ │ │ │ │ + * Provides a wrapper for Google's Maps API │ │ │ │ │ + * Normally the Terms of Use for this API do not allow wrapping, but Google │ │ │ │ │ + * have provided written consent to OpenLayers for this - see email in │ │ │ │ │ + * http://osgeo-org.1560.n6.nabble.com/Google-Maps-API-Terms-of-Use-changes-tp4910013p4911981.html │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Layer.SphericalMercator> │ │ │ │ │ + * - <OpenLayers.Layer.EventPane> │ │ │ │ │ + * - <OpenLayers.Layer.FixedZoomLevels> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Lang.it = { │ │ │ │ │ - │ │ │ │ │ - 'unhandledRequest': "Codice di ritorno della richiesta ${statusText}", │ │ │ │ │ - │ │ │ │ │ - 'Permalink': "Permalink", │ │ │ │ │ - │ │ │ │ │ - 'Overlays': "Overlays", │ │ │ │ │ - │ │ │ │ │ - 'Base Layer': "Livello base", │ │ │ │ │ - │ │ │ │ │ - 'noFID': "Impossibile aggiornare un elemento grafico che non abbia il FID.", │ │ │ │ │ - │ │ │ │ │ - 'browserNotSupported': "Il tuo browser non supporta il rendering vettoriale. I renderizzatore attualemnte supportati sono:\n${renderers}", │ │ │ │ │ - │ │ │ │ │ - // console message │ │ │ │ │ - 'minZoomLevelError': "La proprietà minZoomLevel è da utilizzare solamente " + │ │ │ │ │ - "con livelli che abbiano FixedZoomLevels. Il fatto che " + │ │ │ │ │ - "questo livello wfs controlli la proprietà minZoomLevel è " + │ │ │ │ │ - "un retaggio del passato. Non possiamo comunque rimuoverla " + │ │ │ │ │ - "senza rompere le vecchie applicazioni che dipendono su di essa." + │ │ │ │ │ - "Quindi siamo costretti a deprecarla -- minZoomLevel " + │ │ │ │ │ - "e sarà rimossa dalla vesione 3.0. Si prega di utilizzare i " + │ │ │ │ │ - "settaggi di risoluzione min/max come descritto qui: " + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - │ │ │ │ │ - 'commitSuccess': "Transazione WFS: SUCCESS ${response}", │ │ │ │ │ +OpenLayers.Layer.Google = OpenLayers.Class( │ │ │ │ │ + OpenLayers.Layer.EventPane, │ │ │ │ │ + OpenLayers.Layer.FixedZoomLevels, { │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "Transazione WFS: FAILED ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Constant: MIN_ZOOM_LEVEL │ │ │ │ │ + * {Integer} 0 │ │ │ │ │ + */ │ │ │ │ │ + MIN_ZOOM_LEVEL: 0, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Il livello Google non è riuscito a caricare correttamente.<br><br>" + │ │ │ │ │ - "Per evitare questo messaggio, seleziona un nuovo BaseLayer " + │ │ │ │ │ - "nel selettore di livelli nell'angolo in alto a destra.<br><br>" + │ │ │ │ │ - "Più precisamente, ciò accade perchè la libreria Google Maps " + │ │ │ │ │ - "non è stata inclusa nella pagina, oppure non contiene la " + │ │ │ │ │ - "corretta API key per il tuo sito.<br><br>" + │ │ │ │ │ - "Sviluppatori: Per aiuto su come farlo funzionare correttamente, " + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/Google' " + │ │ │ │ │ - "target='_blank'>clicca qui</a>", │ │ │ │ │ + /** │ │ │ │ │ + * Constant: MAX_ZOOM_LEVEL │ │ │ │ │ + * {Integer} 21 │ │ │ │ │ + */ │ │ │ │ │ + MAX_ZOOM_LEVEL: 21, │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "Il livello ${layerType} non è riuscito a caricare correttamente.<br><br>" + │ │ │ │ │ - "Per evitare questo messaggio, seleziona un nuovo BaseLayer " + │ │ │ │ │ - "nel selettore di livelli nell'angolo in alto a destra.<br><br>" + │ │ │ │ │ - "Più precisamente, ciò accade perchè la libreria ${layerLib} " + │ │ │ │ │ - "non è stata inclusa nella pagina.<br><br>" + │ │ │ │ │ - "Sviluppatori: Per aiuto su come farlo funzionare correttamente, " + │ │ │ │ │ - "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + │ │ │ │ │ - "target='_blank'>clicca qui</a>", │ │ │ │ │ + /** │ │ │ │ │ + * Constant: RESOLUTIONS │ │ │ │ │ + * {Array(Float)} Hardcode these resolutions so that they are more closely │ │ │ │ │ + * tied with the standard wms projection │ │ │ │ │ + */ │ │ │ │ │ + RESOLUTIONS: [ │ │ │ │ │ + 1.40625, │ │ │ │ │ + 0.703125, │ │ │ │ │ + 0.3515625, │ │ │ │ │ + 0.17578125, │ │ │ │ │ + 0.087890625, │ │ │ │ │ + 0.0439453125, │ │ │ │ │ + 0.02197265625, │ │ │ │ │ + 0.010986328125, │ │ │ │ │ + 0.0054931640625, │ │ │ │ │ + 0.00274658203125, │ │ │ │ │ + 0.001373291015625, │ │ │ │ │ + 0.0006866455078125, │ │ │ │ │ + 0.00034332275390625, │ │ │ │ │ + 0.000171661376953125, │ │ │ │ │ + 0.0000858306884765625, │ │ │ │ │ + 0.00004291534423828125, │ │ │ │ │ + 0.00002145767211914062, │ │ │ │ │ + 0.00001072883605957031, │ │ │ │ │ + 0.00000536441802978515, │ │ │ │ │ + 0.00000268220901489257, │ │ │ │ │ + 0.0000013411045074462891, │ │ │ │ │ + 0.00000067055225372314453 │ │ │ │ │ + ], │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Scala = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: type │ │ │ │ │ + * {GMapType} │ │ │ │ │ + */ │ │ │ │ │ + type: null, │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'reprojectDeprecated': "Stai utilizzando l'opzione 'reproject' sul livello ${layerName}. " + │ │ │ │ │ - "Questa opzione è deprecata: il suo utilizzo è stato introdotto per" + │ │ │ │ │ - "supportare il disegno dei dati sopra mappe commerciali, ma tale " + │ │ │ │ │ - "funzionalità dovrebbe essere ottenuta tramite l'utilizzo della proiezione " + │ │ │ │ │ - "Spherical Mercator. Per maggiori informazioni consultare qui " + │ │ │ │ │ - "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: wrapDateLine │ │ │ │ │ + * {Boolean} Allow user to pan forever east/west. Default is true. │ │ │ │ │ + * Setting this to false only restricts panning if │ │ │ │ │ + * <sphericalMercator> is true. │ │ │ │ │ + */ │ │ │ │ │ + wrapDateLine: true, │ │ │ │ │ │ │ │ │ │ - // console message │ │ │ │ │ - 'methodDeprecated': "Questo metodo è stato deprecato e sarà rimosso dalla versione 3.0. " + │ │ │ │ │ - "Si prega di utilizzare il metodo ${newMethod} in alternativa.", │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: sphericalMercator │ │ │ │ │ + * {Boolean} Should the map act as a mercator-projected map? This will │ │ │ │ │ + * cause all interactions with the map to be in the actual map │ │ │ │ │ + * projection, which allows support for vector drawing, overlaying │ │ │ │ │ + * other maps, etc. │ │ │ │ │ + */ │ │ │ │ │ + sphericalMercator: false, │ │ │ │ │ │ │ │ │ │ - 'end': '' │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Lang/hsb.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: version │ │ │ │ │ + * {Number} The version of the Google Maps API │ │ │ │ │ + */ │ │ │ │ │ + version: null, │ │ │ │ │ │ │ │ │ │ -/* Translators (2009 onwards): │ │ │ │ │ - * - Michawiki │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Layer.Google │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} A name for the layer. │ │ │ │ │ + * options - {Object} An optional object whose properties will be set │ │ │ │ │ + * on the layer. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + if (!options.version) { │ │ │ │ │ + options.version = typeof GMap2 === "function" ? "2" : "3"; │ │ │ │ │ + } │ │ │ │ │ + var mixin = OpenLayers.Layer.Google["v" + │ │ │ │ │ + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (mixin) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, mixin); │ │ │ │ │ + } else { │ │ │ │ │ + throw "Unsupported Google Maps API version: " + options.version; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Lang.js │ │ │ │ │ - */ │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, mixin.DEFAULTS); │ │ │ │ │ + if (options.maxExtent) { │ │ │ │ │ + options.maxExtent = options.maxExtent.clone(); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Namespace: OpenLayers.Lang["hsb"] │ │ │ │ │ - * Dictionary for Hornjoserbsce. Keys for entries are used in calls to │ │ │ │ │ - * <OpenLayers.Lang.translate>. Entry bodies are normal strings or │ │ │ │ │ - * strings formatted for use with <OpenLayers.String.format> calls. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Lang["hsb"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.initialize.apply(this, │ │ │ │ │ + [name, options]); │ │ │ │ │ + OpenLayers.Layer.FixedZoomLevels.prototype.initialize.apply(this, │ │ │ │ │ + [name, options]); │ │ │ │ │ │ │ │ │ │ - 'unhandledRequest': "Wotmołwa njewobdźěłaneho naprašowanja ${statusText}", │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + OpenLayers.Util.extend(this, OpenLayers.Layer.SphericalMercator); │ │ │ │ │ + this.initMercatorParameters(); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Permalink': "Trajny wotkaz", │ │ │ │ │ + /** │ │ │ │ │ + * Method: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.Google>} An exact clone of this layer │ │ │ │ │ + */ │ │ │ │ │ + clone: function() { │ │ │ │ │ + /** │ │ │ │ │ + * This method isn't intended to be called by a subclass and it │ │ │ │ │ + * doesn't call the same method on the superclass. We don't call │ │ │ │ │ + * the super's clone because we don't want properties that are set │ │ │ │ │ + * on this layer after initialize (i.e. this.mapObject etc.). │ │ │ │ │ + */ │ │ │ │ │ + return new OpenLayers.Layer.Google( │ │ │ │ │ + this.name, this.getOptions() │ │ │ │ │ + ); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Overlays': "Naworštowanja", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setVisibility │ │ │ │ │ + * Set the visibility flag for the layer and hide/show & redraw │ │ │ │ │ + * accordingly. Fire event unless otherwise specified │ │ │ │ │ + * │ │ │ │ │ + * Note that visibility is no longer simply whether or not the layer's │ │ │ │ │ + * style.display is set to "block". Now we store a 'visibility' state │ │ │ │ │ + * property on the layer class, this allows us to remember whether or │ │ │ │ │ + * not we *desire* for a layer to be visible. In the case where the │ │ │ │ │ + * map's resolution is out of the layer's range, this desire may be │ │ │ │ │ + * subverted. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * visible - {Boolean} Display the layer (if in range) │ │ │ │ │ + */ │ │ │ │ │ + setVisibility: function(visible) { │ │ │ │ │ + // sharing a map container, opacity has to be set per layer │ │ │ │ │ + var opacity = this.opacity == null ? 1 : this.opacity; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.setVisibility.apply(this, arguments); │ │ │ │ │ + this.setOpacity(opacity); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Base Layer': "Zakładna runina", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: display │ │ │ │ │ + * Hide or show the Layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * visible - {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + display: function(visible) { │ │ │ │ │ + if (!this._dragging) { │ │ │ │ │ + this.setGMapVisibility(visible); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.display.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'noFID': "Funkcija, za kotruž FID njeje, njeda so aktualizować.", │ │ │ │ │ + /** │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * bounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * zoomChanged - {Boolean} Tells when zoom has changed, as layers have to │ │ │ │ │ + * do some init work in that case. │ │ │ │ │ + * dragging - {Boolean} │ │ │ │ │ + */ │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + this._dragging = dragging; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + delete this._dragging; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'browserNotSupported': "Twój wobhladowak wektorowe rysowanje njepodpěruje. Tuchwilu podpěrowane rysowaki su:\n${renderers}", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setOpacity │ │ │ │ │ + * Sets the opacity for the entire layer (all images) │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * opacity - {Float} │ │ │ │ │ + */ │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + if (opacity !== this.opacity) { │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "opacity" │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + this.opacity = opacity; │ │ │ │ │ + } │ │ │ │ │ + // Though this layer's opacity may not change, we're sharing a container │ │ │ │ │ + // and need to update the opacity for the entire container. │ │ │ │ │ + if (this.getVisibility()) { │ │ │ │ │ + var container = this.getMapContainer(); │ │ │ │ │ + OpenLayers.Util.modifyDOMElement( │ │ │ │ │ + container, null, null, null, null, null, null, opacity │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'minZoomLevelError': "Kajkosć minZoomLevel je jenož za wužiwanje z worštami myslena, kotrež wot FixedZoomLevels pochadźeja. Zo tuta woršta wfs za minZoomLevel přepruwuje, je relikt zańdźenosće. Njemóžemy wšak ju wotstronić, bjeztoho zo aplikacije, kotrež na OpenLayers bazěruja a snano tutu kajkosć wužiwaja, hižo njefunguja. Tohodla smy ju jako zestarjenu woznamjenili -- přepruwowanje za minZoomLevel budu so we wersiji 3.0 wotstronjeć. Prošu wužij město toho nastajenje min/max, kaž je tu wopisane: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + * Clean up this layer. │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + /** │ │ │ │ │ + * We have to override this method because the event pane destroy │ │ │ │ │ + * deletes the mapObject reference before removing this layer from │ │ │ │ │ + * the map. │ │ │ │ │ + */ │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.setGMapVisibility(false); │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache && cache.count <= 1) { │ │ │ │ │ + this.removeGMapElements(); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'commitSuccess': "WFS-Transakcija: WUSPĚŠNA ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * Method: removeGMapElements │ │ │ │ │ + * Remove all elements added to the dom. This should only be called if │ │ │ │ │ + * this is the last of the Google layers for the given map. │ │ │ │ │ + */ │ │ │ │ │ + removeGMapElements: function() { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + // remove shared elements from dom │ │ │ │ │ + var container = this.mapObject && this.getMapContainer(); │ │ │ │ │ + if (container && container.parentNode) { │ │ │ │ │ + container.parentNode.removeChild(container); │ │ │ │ │ + } │ │ │ │ │ + var termsOfUse = cache.termsOfUse; │ │ │ │ │ + if (termsOfUse && termsOfUse.parentNode) { │ │ │ │ │ + termsOfUse.parentNode.removeChild(termsOfUse); │ │ │ │ │ + } │ │ │ │ │ + var poweredBy = cache.poweredBy; │ │ │ │ │ + if (poweredBy && poweredBy.parentNode) { │ │ │ │ │ + poweredBy.parentNode.removeChild(poweredBy); │ │ │ │ │ + } │ │ │ │ │ + if (this.mapObject && window.google && google.maps && │ │ │ │ │ + google.maps.event && google.maps.event.clearListeners) { │ │ │ │ │ + google.maps.event.clearListeners(this.mapObject, 'tilesloaded'); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'commitFailed': "WFS-Transakcija: NJEPORADŹENA ${response}", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: removeMap │ │ │ │ │ + * On being removed from the map, also remove termsOfUse and poweredBy divs │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * map - {<OpenLayers.Map>} │ │ │ │ │ + */ │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + // hide layer before removing │ │ │ │ │ + if (this.visibility && this.mapObject) { │ │ │ │ │ + this.setGMapVisibility(false); │ │ │ │ │ + } │ │ │ │ │ + // check to see if last Google layer in this map │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + if (cache.count <= 1) { │ │ │ │ │ + this.removeGMapElements(); │ │ │ │ │ + delete OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ + } else { │ │ │ │ │ + // decrement the layer count │ │ │ │ │ + --cache.count; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + // remove references to gmap elements │ │ │ │ │ + delete this.termsOfUse; │ │ │ │ │ + delete this.poweredBy; │ │ │ │ │ + delete this.mapObject; │ │ │ │ │ + delete this.dragObject; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.removeMap.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'googleWarning': "Woršta Google njemóžeše so korektnje začitać.\x3cbr\x3e\x3cbr\x3eZo by tutu zdźělenku wotbył, wubjer nowy BaseLayer z wuběra worštow horjeka naprawo.\x3cbr\x3e\x3cbr\x3eNajskerje so to stawa, dokelž skript biblioteki Google Maps pak njebu zapřijaty pak njewobsahuje korektny kluč API za twoje sydło.\x3cbr\x3e\x3cbr\x3eWuwiwarjo: Za pomoc ke korektnemu fungowanju worštow\n\x3ca href=\'http://trac.openlayers.org/wiki/Google\' target=\'_blank\'\x3etu kliknyć\x3c/a\x3e", │ │ │ │ │ + // │ │ │ │ │ + // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ + // │ │ │ │ │ │ │ │ │ │ - 'getLayerWarning': "Woršta ${layerType} njemóžeše so korektnje začitać.\x3cbr\x3e\x3cbr\x3eZo by tutu zdźělenku wotbył, wubjer nowy BaseLayer z wuběra worštow horjeka naprawo.\x3cbr\x3e\x3cbr\x3eNajskerje so to stawa, dokelž skript biblioteki ${layerLib} njebu korektnje zapřijaty.\x3cbr\x3e\x3cbr\x3eWuwiwarjo: Za pomoc ke korektnemu fungowanju worštow\n\x3ca href=\'http://trac.openlayers.org/wiki/${layerLib}\' target=\'_blank\'\x3etu kliknyć\x3c/a\x3e", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getOLBoundsFromMapObjectBounds │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moBounds - {Object} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Bounds>} An <OpenLayers.Bounds>, translated from the │ │ │ │ │ + * passed-in MapObject Bounds. │ │ │ │ │ + * Returns null if null value is passed in. │ │ │ │ │ + */ │ │ │ │ │ + getOLBoundsFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + var olBounds = null; │ │ │ │ │ + if (moBounds != null) { │ │ │ │ │ + var sw = moBounds.getSouthWest(); │ │ │ │ │ + var ne = moBounds.getNorthEast(); │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + sw = this.forwardMercator(sw.lng(), sw.lat()); │ │ │ │ │ + ne = this.forwardMercator(ne.lng(), ne.lat()); │ │ │ │ │ + } else { │ │ │ │ │ + sw = new OpenLayers.LonLat(sw.lng(), sw.lat()); │ │ │ │ │ + ne = new OpenLayers.LonLat(ne.lng(), ne.lat()); │ │ │ │ │ + } │ │ │ │ │ + olBounds = new OpenLayers.Bounds(sw.lon, │ │ │ │ │ + sw.lat, │ │ │ │ │ + ne.lon, │ │ │ │ │ + ne.lat); │ │ │ │ │ + } │ │ │ │ │ + return olBounds; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'Scale = 1 : ${scaleDenom}': "Měritko = 1 : ${scaleDenom}", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getWarningHTML │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} String with information on why layer is broken, how to get │ │ │ │ │ + * it working. │ │ │ │ │ + */ │ │ │ │ │ + getWarningHTML: function() { │ │ │ │ │ + return OpenLayers.i18n("googleWarning"); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'W': "Z", │ │ │ │ │ │ │ │ │ │ - 'E': "W", │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Interface Controls * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ │ │ │ │ │ - 'N': "S", │ │ │ │ │ │ │ │ │ │ - 'S': "J", │ │ │ │ │ + // Get&Set Center, Zoom │ │ │ │ │ │ │ │ │ │ - 'reprojectDeprecated': "Wužiwaš opciju \"reproject\" wořšty ${layerName}. Tuta opcija je zestarjena: jeje wužiwanje bě myslene, zo by zwobraznjenje datow nad komercielnymi bazowymi kartami podpěrało, ale funkcionalnosć měła so nětko z pomocu Sperical Mercator docpěć. Dalše informacije steja na http://trac.openlayers.org/wiki/SphericalMercator k dispoziciji.", │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectCenter │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} The mapObject's current center in Map Object format │ │ │ │ │ + */ │ │ │ │ │ + getMapObjectCenter: function() { │ │ │ │ │ + return this.mapObject.getCenter(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - 'methodDeprecated': "Tuta metoda je so njeschwaliła a budźe so w 3.0 wotstronjeć. Prošu wužij ${newMethod} město toho." │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectZoom │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} The mapObject's current zoom, in Map Object format │ │ │ │ │ + */ │ │ │ │ │ + getMapObjectZoom: function() { │ │ │ │ │ + return this.mapObject.getZoom(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/SOS.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Primitives * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Protocol.js │ │ │ │ │ - */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Function: OpenLayers.Protocol.SOS │ │ │ │ │ - * Used to create a versioned SOS protocol. Default version is 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol>} An SOS protocol for the given version. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol.SOS = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options, OpenLayers.Protocol.SOS.DEFAULTS │ │ │ │ │ - ); │ │ │ │ │ - var cls = OpenLayers.Protocol.SOS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported SOS version: " + options.version; │ │ │ │ │ - } │ │ │ │ │ - return new cls(options); │ │ │ │ │ -}; │ │ │ │ │ + // LonLat │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Protocol.SOS.DEFAULTS │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol.SOS.DEFAULTS = { │ │ │ │ │ - "version": "1.0.0" │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/CSW.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getLongitudeFromMapObjectLonLat │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} Longitude of the given MapObject LonLat │ │ │ │ │ + */ │ │ │ │ │ + getLongitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + return this.sphericalMercator ? │ │ │ │ │ + this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lon : │ │ │ │ │ + moLonLat.lng(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getLatitudeFromMapObjectLonLat │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Float} Latitude of the given MapObject LonLat │ │ │ │ │ + */ │ │ │ │ │ + getLatitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + var lat = this.sphericalMercator ? │ │ │ │ │ + this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lat : │ │ │ │ │ + moLonLat.lat(); │ │ │ │ │ + return lat; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Protocol.js │ │ │ │ │ - */ │ │ │ │ │ + // Pixel │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Protocol.CSW │ │ │ │ │ - * Used to create a versioned CSW protocol. Default version is 2.0.2. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol.CSW = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options, OpenLayers.Protocol.CSW.DEFAULTS │ │ │ │ │ - ); │ │ │ │ │ - var cls = OpenLayers.Protocol.CSW["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported CSW version: " + options.version; │ │ │ │ │ - } │ │ │ │ │ - return new cls(options); │ │ │ │ │ -}; │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getXFromMapObjectPixel │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moPixel - {Object} MapObject Pixel format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} X value of the MapObject Pixel │ │ │ │ │ + */ │ │ │ │ │ + getXFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return moPixel.x; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Protocol.CSW.DEFAULTS │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol.CSW.DEFAULTS = { │ │ │ │ │ - "version": "2.0.2" │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/WFS.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getYFromMapObjectPixel │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moPixel - {Object} MapObject Pixel format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Integer} Y value of the MapObject Pixel │ │ │ │ │ + */ │ │ │ │ │ + getYFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return moPixel.y; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Google" │ │ │ │ │ + }); │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Protocol.js │ │ │ │ │ + * Property: OpenLayers.Layer.Google.cache │ │ │ │ │ + * {Object} Cache for elements that should only be created once per map. │ │ │ │ │ */ │ │ │ │ │ +OpenLayers.Layer.Google.cache = {}; │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Protocol.WFS │ │ │ │ │ - * Used to create a versioned WFS protocol. Default version is 1.0.0. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol>} A WFS protocol of the given version. │ │ │ │ │ - * │ │ │ │ │ - * Example: │ │ │ │ │ - * (code) │ │ │ │ │ - * var protocol = new OpenLayers.Protocol.WFS({ │ │ │ │ │ - * version: "1.1.0", │ │ │ │ │ - * url: "http://demo.opengeo.org/geoserver/wfs", │ │ │ │ │ - * featureType: "tasmania_roads", │ │ │ │ │ - * featureNS: "http://www.openplans.org/topp", │ │ │ │ │ - * geometryName: "the_geom" │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * See the protocols for specific WFS versions for more detail. │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol.WFS = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options, OpenLayers.Protocol.WFS.DEFAULTS │ │ │ │ │ - ); │ │ │ │ │ - var cls = OpenLayers.Protocol.WFS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported WFS version: " + options.version; │ │ │ │ │ - } │ │ │ │ │ - return new cls(options); │ │ │ │ │ -}; │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Function: fromWMSLayer │ │ │ │ │ - * Convenience function to create a WFS protocol from a WMS layer. This makes │ │ │ │ │ - * the assumption that a WFS requests can be issued at the same URL as │ │ │ │ │ - * WMS requests and that a WFS featureType exists with the same name as the │ │ │ │ │ - * WMS layer. │ │ │ │ │ - * │ │ │ │ │ - * This function is designed to auto-configure <url>, <featureType>, │ │ │ │ │ - * <featurePrefix> and <srsName> for WFS <version> 1.1.0. Note that │ │ │ │ │ - * srsName matching with the WMS layer will not work with WFS 1.0.0. │ │ │ │ │ + * Constant: OpenLayers.Layer.Google.v2 │ │ │ │ │ * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * layer - {<OpenLayers.Layer.WMS>} WMS layer that has a matching WFS │ │ │ │ │ - * FeatureType at the same server url with the same typename. │ │ │ │ │ - * options - {Object} Default properties to be set on the protocol. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.WFS>} │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol.WFS.fromWMSLayer = function(layer, options) { │ │ │ │ │ - var typeName, featurePrefix; │ │ │ │ │ - var param = layer.params["LAYERS"]; │ │ │ │ │ - var parts = (OpenLayers.Util.isArray(param) ? param[0] : param).split(":"); │ │ │ │ │ - if (parts.length > 1) { │ │ │ │ │ - featurePrefix = parts[0]; │ │ │ │ │ - } │ │ │ │ │ - typeName = parts.pop(); │ │ │ │ │ - var protocolOptions = { │ │ │ │ │ - url: layer.url, │ │ │ │ │ - featureType: typeName, │ │ │ │ │ - featurePrefix: featurePrefix, │ │ │ │ │ - srsName: layer.projection && layer.projection.getCode() || │ │ │ │ │ - layer.map && layer.map.getProjectionObject().getCode(), │ │ │ │ │ - version: "1.1.0" │ │ │ │ │ - }; │ │ │ │ │ - return new OpenLayers.Protocol.WFS(OpenLayers.Util.applyDefaults( │ │ │ │ │ - options, protocolOptions │ │ │ │ │ - )); │ │ │ │ │ -}; │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Constant: OpenLayers.Protocol.WFS.DEFAULTS │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol.WFS.DEFAULTS = { │ │ │ │ │ - "version": "1.0.0" │ │ │ │ │ -}; │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/HTTP.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ - │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Protocol.js │ │ │ │ │ - * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * if application uses the query string, for example, for BBOX parameters, │ │ │ │ │ - * OpenLayers/Format/QueryStringFilter.js should be included in the build config file │ │ │ │ │ - */ │ │ │ │ │ - │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Protocol.HTTP │ │ │ │ │ - * A basic HTTP protocol for vector layers. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Protocol.HTTP> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol> │ │ │ │ │ + * Mixin providing functionality specific to the Google Maps API v2. │ │ │ │ │ + * │ │ │ │ │ + * This API has been deprecated by Google. │ │ │ │ │ + * Developers are encouraged to migrate to v3 of the API; support for this │ │ │ │ │ + * is provided by <OpenLayers.Layer.Google.v3> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.HTTP = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ +OpenLayers.Layer.Google.v2 = { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: url │ │ │ │ │ - * {String} Service URL, read-only, set through the options │ │ │ │ │ - * passed to constructor. │ │ │ │ │ + * Property: termsOfUse │ │ │ │ │ + * {DOMElement} Div for Google's copyright and terms of use link │ │ │ │ │ */ │ │ │ │ │ - url: null, │ │ │ │ │ + termsOfUse: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: headers │ │ │ │ │ - * {Object} HTTP request headers, read-only, set through the options │ │ │ │ │ - * passed to the constructor, │ │ │ │ │ - * Example: {'Content-Type': 'plain/text'} │ │ │ │ │ + * Property: poweredBy │ │ │ │ │ + * {DOMElement} Div for Google's powered by logo and link │ │ │ │ │ */ │ │ │ │ │ - headers: null, │ │ │ │ │ + poweredBy: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: params │ │ │ │ │ - * {Object} Parameters of GET requests, read-only, set through the options │ │ │ │ │ - * passed to the constructor, │ │ │ │ │ - * Example: {'bbox': '5,5,5,5'} │ │ │ │ │ + * Property: dragObject │ │ │ │ │ + * {GDraggableObject} Since 2.93, Google has exposed the ability to get │ │ │ │ │ + * the maps GDraggableObject. We can now use this for smooth panning │ │ │ │ │ */ │ │ │ │ │ - params: null, │ │ │ │ │ + dragObject: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: callback │ │ │ │ │ - * {Object} Function to be called when the <read>, <create>, │ │ │ │ │ - * <update>, <delete> or <commit> operation completes, read-only, │ │ │ │ │ - * set through the options passed to the constructor. │ │ │ │ │ + /** │ │ │ │ │ + * Method: loadMapObject │ │ │ │ │ + * Load the GMap and register appropriate event listeners. If we can't │ │ │ │ │ + * load GMap2, then display a warning message. │ │ │ │ │ */ │ │ │ │ │ - callback: null, │ │ │ │ │ + loadMapObject: function() { │ │ │ │ │ + if (!this.type) { │ │ │ │ │ + this.type = G_NORMAL_MAP; │ │ │ │ │ + } │ │ │ │ │ + var mapObject, termsOfUse, poweredBy; │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + // there are already Google layers added to this map │ │ │ │ │ + mapObject = cache.mapObject; │ │ │ │ │ + termsOfUse = cache.termsOfUse; │ │ │ │ │ + poweredBy = cache.poweredBy; │ │ │ │ │ + // increment the layer count │ │ │ │ │ + ++cache.count; │ │ │ │ │ + } else { │ │ │ │ │ + // this is the first Google layer for this map │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: scope │ │ │ │ │ - * {Object} Callback execution scope, read-only, set through the │ │ │ │ │ - * options passed to the constructor. │ │ │ │ │ - */ │ │ │ │ │ - scope: null, │ │ │ │ │ + var container = this.map.viewPortDiv; │ │ │ │ │ + var div = document.createElement("div"); │ │ │ │ │ + div.id = this.map.id + "_GMap2Container"; │ │ │ │ │ + div.style.position = "absolute"; │ │ │ │ │ + div.style.width = "100%"; │ │ │ │ │ + div.style.height = "100%"; │ │ │ │ │ + container.appendChild(div); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: readWithPOST │ │ │ │ │ - * {Boolean} true if read operations are done with POST requests │ │ │ │ │ - * instead of GET, defaults to false. │ │ │ │ │ - */ │ │ │ │ │ - readWithPOST: false, │ │ │ │ │ + // create GMap and shuffle elements │ │ │ │ │ + try { │ │ │ │ │ + mapObject = new GMap2(div); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: updateWithPOST │ │ │ │ │ - * {Boolean} true if update operations are done with POST requests │ │ │ │ │ - * defaults to false. │ │ │ │ │ - */ │ │ │ │ │ - updateWithPOST: false, │ │ │ │ │ + // move the ToS and branding stuff up to the container div │ │ │ │ │ + termsOfUse = div.lastChild; │ │ │ │ │ + container.appendChild(termsOfUse); │ │ │ │ │ + termsOfUse.style.zIndex = "1100"; │ │ │ │ │ + termsOfUse.style.right = ""; │ │ │ │ │ + termsOfUse.style.bottom = ""; │ │ │ │ │ + termsOfUse.className = "olLayerGoogleCopyright"; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: deleteWithPOST │ │ │ │ │ - * {Boolean} true if delete operations are done with POST requests │ │ │ │ │ - * defaults to false. │ │ │ │ │ - * if true, POST data is set to output of format.write(). │ │ │ │ │ - */ │ │ │ │ │ - deleteWithPOST: false, │ │ │ │ │ + poweredBy = div.lastChild; │ │ │ │ │ + container.appendChild(poweredBy); │ │ │ │ │ + poweredBy.style.zIndex = "1100"; │ │ │ │ │ + poweredBy.style.right = ""; │ │ │ │ │ + poweredBy.style.bottom = ""; │ │ │ │ │ + poweredBy.className = "olLayerGooglePoweredBy gmnoprint"; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: wildcarded. │ │ │ │ │ - * {Boolean} If true percent signs are added around values │ │ │ │ │ - * read from LIKE filters, for example if the protocol │ │ │ │ │ - * read method is passed a LIKE filter whose property │ │ │ │ │ - * is "foo" and whose value is "bar" the string │ │ │ │ │ - * "foo__ilike=%bar%" will be sent in the query string; │ │ │ │ │ - * defaults to false. │ │ │ │ │ - */ │ │ │ │ │ - wildcarded: false, │ │ │ │ │ + } catch (e) { │ │ │ │ │ + throw (e); │ │ │ │ │ + } │ │ │ │ │ + // cache elements for use by any other google layers added to │ │ │ │ │ + // this same map │ │ │ │ │ + OpenLayers.Layer.Google.cache[this.map.id] = { │ │ │ │ │ + mapObject: mapObject, │ │ │ │ │ + termsOfUse: termsOfUse, │ │ │ │ │ + poweredBy: poweredBy, │ │ │ │ │ + count: 1 │ │ │ │ │ + }; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: srsInBBOX │ │ │ │ │ - * {Boolean} Include the SRS identifier in BBOX query string parameter. │ │ │ │ │ - * Default is false. If true and the layer has a projection object set, │ │ │ │ │ - * any BBOX filter will be serialized with a fifth item identifying the │ │ │ │ │ - * projection. E.g. bbox=-1000,-1000,1000,1000,EPSG:900913 │ │ │ │ │ - */ │ │ │ │ │ - srsInBBOX: false, │ │ │ │ │ + this.mapObject = mapObject; │ │ │ │ │ + this.termsOfUse = termsOfUse; │ │ │ │ │ + this.poweredBy = poweredBy; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.HTTP │ │ │ │ │ - * A class for giving layers generic HTTP protocol. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - * │ │ │ │ │ - * Valid options include: │ │ │ │ │ - * url - {String} │ │ │ │ │ - * headers - {Object} │ │ │ │ │ - * params - {Object} URL parameters for GET requests │ │ │ │ │ - * format - {<OpenLayers.Format>} │ │ │ │ │ - * callback - {Function} │ │ │ │ │ - * scope - {Object} │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - this.params = {}; │ │ │ │ │ - this.headers = {}; │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ + // ensure this layer type is one of the mapObject types │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.mapObject.getMapTypes(), │ │ │ │ │ + this.type) === -1) { │ │ │ │ │ + this.mapObject.addMapType(this.type); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ - var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ - wildcarded: this.wildcarded, │ │ │ │ │ - srsInBBOX: this.srsInBBOX │ │ │ │ │ - }); │ │ │ │ │ - this.filterToParams = function(filter, params) { │ │ │ │ │ - return format.write(filter, params); │ │ │ │ │ - }; │ │ │ │ │ + //since v 2.93 getDragObject is now available. │ │ │ │ │ + if (typeof mapObject.getDragObject == "function") { │ │ │ │ │ + this.dragObject = mapObject.getDragObject(); │ │ │ │ │ + } else { │ │ │ │ │ + this.dragPanMapObject = null; │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up the protocol. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.params = null; │ │ │ │ │ - this.headers = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ - }, │ │ │ │ │ + if (this.isBaseLayer === false) { │ │ │ │ │ + this.setGMapVisibility(this.div.style.display !== "none"); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: filterToParams │ │ │ │ │ - * Optional method to translate an <OpenLayers.Filter> object into an object │ │ │ │ │ - * that can be serialized as request query string provided. If a custom │ │ │ │ │ - * method is not provided, the filter will be serialized using the │ │ │ │ │ - * <OpenLayers.Format.QueryStringFilter> class. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} filter to convert. │ │ │ │ │ - * params - {Object} The parameters object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The resulting parameters object. │ │ │ │ │ - */ │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Construct a request for reading new features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * url - {String} Url for the request. │ │ │ │ │ - * params - {Object} Parameters to get serialized as a query string. │ │ │ │ │ - * headers - {Object} Headers to be set on the request. │ │ │ │ │ - * filter - {<OpenLayers.Filter>} Filter to get serialized as a │ │ │ │ │ - * query string. │ │ │ │ │ - * readWithPOST - {Boolean} If the request should be done with POST. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} A response object, whose "priv" property │ │ │ │ │ - * references the HTTP request, this object is also passed to the │ │ │ │ │ - * callback function when the request completes, its "features" property │ │ │ │ │ - * is then populated with the features received from the server. │ │ │ │ │ + * APIMethod: onMapResize │ │ │ │ │ */ │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options.params, this.options.params); │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - if (options.filter && this.filterToParams) { │ │ │ │ │ - options.params = this.filterToParams( │ │ │ │ │ - options.filter, options.params │ │ │ │ │ - ); │ │ │ │ │ - } │ │ │ │ │ - var readWithPOST = (options.readWithPOST !== undefined) ? │ │ │ │ │ - options.readWithPOST : this.readWithPOST; │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - if (readWithPOST) { │ │ │ │ │ - var headers = options.headers || {}; │ │ │ │ │ - headers["Content-Type"] = "application/x-www-form-urlencoded"; │ │ │ │ │ - resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ - data: OpenLayers.Util.getParameterString(options.params), │ │ │ │ │ - headers: headers │ │ │ │ │ - }); │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + // workaround for resizing of invisible or not yet fully loaded layers │ │ │ │ │ + // where GMap2.checkResize() does not work. We need to load the GMap │ │ │ │ │ + // for the old div size, then checkResize(), and then call │ │ │ │ │ + // layer.moveTo() to trigger GMap.setCenter() (which will finish │ │ │ │ │ + // the GMap initialization). │ │ │ │ │ + if (this.visibility && this.mapObject.isLoaded()) { │ │ │ │ │ + this.mapObject.checkResize(); │ │ │ │ │ } else { │ │ │ │ │ - resp.priv = OpenLayers.Request.GET({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers │ │ │ │ │ - }); │ │ │ │ │ + if (!this._resized) { │ │ │ │ │ + var layer = this; │ │ │ │ │ + var handle = GEvent.addListener(this.mapObject, "load", function() { │ │ │ │ │ + GEvent.removeListener(handle); │ │ │ │ │ + delete layer._resized; │ │ │ │ │ + layer.mapObject.checkResize(); │ │ │ │ │ + layer.moveTo(layer.map.getCenter(), layer.map.getZoom()); │ │ │ │ │ + }); │ │ │ │ │ + } │ │ │ │ │ + this._resized = true; │ │ │ │ │ } │ │ │ │ │ - return resp; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleRead │ │ │ │ │ - * Individual callbacks are created for read, create and update, should │ │ │ │ │ - * a subclass need to override each one separately. │ │ │ │ │ - * │ │ │ │ │ + * Method: setGMapVisibility │ │ │ │ │ + * Display the GMap container and associated elements. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * the user callback. │ │ │ │ │ - * options - {Object} The user options passed to the read call. │ │ │ │ │ + * visible - {Boolean} Display the GMap elements. │ │ │ │ │ */ │ │ │ │ │ - handleRead: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options); │ │ │ │ │ + setGMapVisibility: function(visible) { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + var container = this.mapObject.getContainer(); │ │ │ │ │ + if (visible === true) { │ │ │ │ │ + this.mapObject.setMapType(this.type); │ │ │ │ │ + container.style.display = ""; │ │ │ │ │ + this.termsOfUse.style.left = ""; │ │ │ │ │ + this.termsOfUse.style.display = ""; │ │ │ │ │ + this.poweredBy.style.display = ""; │ │ │ │ │ + cache.displayed = this.id; │ │ │ │ │ + } else { │ │ │ │ │ + if (cache.displayed === this.id) { │ │ │ │ │ + delete cache.displayed; │ │ │ │ │ + } │ │ │ │ │ + if (!cache.displayed) { │ │ │ │ │ + container.style.display = "none"; │ │ │ │ │ + this.termsOfUse.style.display = "none"; │ │ │ │ │ + // move ToU far to the left in addition to setting display │ │ │ │ │ + // to "none", because at the end of the GMap2 load │ │ │ │ │ + // sequence, display: none will be unset and ToU would be │ │ │ │ │ + // visible after loading a map with a google layer that is │ │ │ │ │ + // initially hidden. │ │ │ │ │ + this.termsOfUse.style.left = "-9999px"; │ │ │ │ │ + this.poweredBy.style.display = "none"; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: create │ │ │ │ │ - * Construct a request for writing newly created features. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * features - {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ + * Method: getMapContainer │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, whose "priv" property references the HTTP request, this │ │ │ │ │ - * object is also passed to the callback function when the request │ │ │ │ │ - * completes, its "features" property is then populated with the │ │ │ │ │ - * the features received from the server. │ │ │ │ │ + * {DOMElement} the GMap container's div │ │ │ │ │ */ │ │ │ │ │ - create: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: features, │ │ │ │ │ - requestType: "create" │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleCreate, resp, options), │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(features) │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - return resp; │ │ │ │ │ + getMapContainer: function() { │ │ │ │ │ + return this.mapObject.getContainer(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleCreate │ │ │ │ │ - * Called the the request issued by <create> is complete. May be overridden │ │ │ │ │ - * by subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the create call. │ │ │ │ │ - */ │ │ │ │ │ - handleCreate: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options); │ │ │ │ │ - }, │ │ │ │ │ + // │ │ │ │ │ + // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ + // │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: update │ │ │ │ │ - * Construct a request updating modified feature. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getMapObjectBoundsFromOLBounds │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ + * olBounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, whose "priv" property references the HTTP request, this │ │ │ │ │ - * object is also passed to the callback function when the request │ │ │ │ │ - * completes, its "features" property is then populated with the │ │ │ │ │ - * the feature received from the server. │ │ │ │ │ + * {Object} A MapObject Bounds, translated from olBounds │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ */ │ │ │ │ │ - update: function(feature, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var url = options.url || │ │ │ │ │ - feature.url || │ │ │ │ │ - this.options.url + "/" + feature.fid; │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ + var moBounds = null; │ │ │ │ │ + if (olBounds != null) { │ │ │ │ │ + var sw = this.sphericalMercator ? │ │ │ │ │ + this.inverseMercator(olBounds.bottom, olBounds.left) : │ │ │ │ │ + new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ + var ne = this.sphericalMercator ? │ │ │ │ │ + this.inverseMercator(olBounds.top, olBounds.right) : │ │ │ │ │ + new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ + moBounds = new GLatLngBounds(new GLatLng(sw.lat, sw.lon), │ │ │ │ │ + new GLatLng(ne.lat, ne.lon)); │ │ │ │ │ + } │ │ │ │ │ + return moBounds; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: feature, │ │ │ │ │ - requestType: "update" │ │ │ │ │ - }); │ │ │ │ │ │ │ │ │ │ - var method = this.updateWithPOST ? "POST" : "PUT"; │ │ │ │ │ - resp.priv = OpenLayers.Request[method]({ │ │ │ │ │ - url: url, │ │ │ │ │ - callback: this.createCallback(this.handleUpdate, resp, options), │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(feature) │ │ │ │ │ - }); │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Interface Controls * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ │ │ │ │ │ - return resp; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleUpdate │ │ │ │ │ - * Called the the request issued by <update> is complete. May be overridden │ │ │ │ │ - * by subclasses. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the update call. │ │ │ │ │ - */ │ │ │ │ │ - handleUpdate: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options); │ │ │ │ │ - }, │ │ │ │ │ + // Get&Set Center, Zoom │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: delete │ │ │ │ │ - * Construct a request deleting a removed feature. │ │ │ │ │ - * │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setMapObjectCenter │ │ │ │ │ + * Set the mapObject to the specified center and zoom │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * feature - {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} An <OpenLayers.Protocol.Response> │ │ │ │ │ - * object, whose "priv" property references the HTTP request, this │ │ │ │ │ - * object is also passed to the callback function when the request │ │ │ │ │ - * completes. │ │ │ │ │ + * center - {Object} MapObject LonLat format │ │ │ │ │ + * zoom - {int} MapObject zoom format │ │ │ │ │ */ │ │ │ │ │ - "delete": function(feature, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var url = options.url || │ │ │ │ │ - feature.url || │ │ │ │ │ - this.options.url + "/" + feature.fid; │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: feature, │ │ │ │ │ - requestType: "delete" │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - var method = this.deleteWithPOST ? "POST" : "DELETE"; │ │ │ │ │ - var requestOptions = { │ │ │ │ │ - url: url, │ │ │ │ │ - callback: this.createCallback(this.handleDelete, resp, options), │ │ │ │ │ - headers: options.headers │ │ │ │ │ - }; │ │ │ │ │ - if (this.deleteWithPOST) { │ │ │ │ │ - requestOptions.data = this.format.write(feature); │ │ │ │ │ - } │ │ │ │ │ - resp.priv = OpenLayers.Request[method](requestOptions); │ │ │ │ │ - │ │ │ │ │ - return resp; │ │ │ │ │ + setMapObjectCenter: function(center, zoom) { │ │ │ │ │ + this.mapObject.setCenter(center, zoom); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleDelete │ │ │ │ │ - * Called the the request issued by <delete> is complete. May be overridden │ │ │ │ │ - * by subclasses. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: dragPanMapObject │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the delete call. │ │ │ │ │ + * dX - {Integer} │ │ │ │ │ + * dY - {Integer} │ │ │ │ │ */ │ │ │ │ │ - handleDelete: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options); │ │ │ │ │ + dragPanMapObject: function(dX, dY) { │ │ │ │ │ + this.dragObject.moveBy(new GSize(-dX, dY)); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + │ │ │ │ │ + // LonLat - Pixel Translation │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: handleResponse │ │ │ │ │ - * Called by CRUD specific handlers. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getMapObjectLonLatFromMapObjectPixel │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the create, read, update, │ │ │ │ │ - * or delete call. │ │ │ │ │ + * moPixel - {Object} MapObject Pixel format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject LonLat translated from MapObject Pixel │ │ │ │ │ */ │ │ │ │ │ - handleResponse: function(resp, options) { │ │ │ │ │ - var request = resp.priv; │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - // success │ │ │ │ │ - if (resp.requestType != "delete") { │ │ │ │ │ - resp.features = this.parseFeatures(request); │ │ │ │ │ - } │ │ │ │ │ - resp.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ - } else { │ │ │ │ │ - // failure │ │ │ │ │ - resp.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - } │ │ │ │ │ - options.callback.call(options.scope, resp); │ │ │ │ │ - } │ │ │ │ │ + getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return this.mapObject.fromContainerPixelToLatLng(moPixel); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseFeatures │ │ │ │ │ - * Read HTTP response body and return features. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getMapObjectPixelFromMapObjectLonLat │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * request - {XMLHttpRequest} The request object │ │ │ │ │ - * │ │ │ │ │ + * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} Array of features or a single feature. │ │ │ │ │ + * {Object} MapObject Pixel transtlated from MapObject LonLat │ │ │ │ │ */ │ │ │ │ │ - parseFeatures: function(request) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - return this.format.read(doc); │ │ │ │ │ + getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + return this.mapObject.fromLatLngToContainerPixel(moLonLat); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: commit │ │ │ │ │ - * Iterate over each feature and take action based on the feature state. │ │ │ │ │ - * Possible actions are create, update and delete. │ │ │ │ │ - * │ │ │ │ │ + │ │ │ │ │ + // Bounds │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectZoomFromMapObjectBounds │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array({<OpenLayers.Feature.Vector>})} │ │ │ │ │ - * options - {Object} Optional object for setting up intermediate commit │ │ │ │ │ - * callbacks. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * create - {Object} Optional object to be passed to the <create> method. │ │ │ │ │ - * update - {Object} Optional object to be passed to the <update> method. │ │ │ │ │ - * delete - {Object} Optional object to be passed to the <delete> method. │ │ │ │ │ - * callback - {Function} Optional function to be called when the commit │ │ │ │ │ - * is complete. │ │ │ │ │ - * scope - {Object} Optional object to be set as the scope of the callback. │ │ │ │ │ - * │ │ │ │ │ + * moBounds - {Object} MapObject Bounds format │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Array(<OpenLayers.Protocol.Response>)} An array of response objects, │ │ │ │ │ - * one per request made to the server, each object's "priv" property │ │ │ │ │ - * references the corresponding HTTP request. │ │ │ │ │ + * {Object} MapObject Zoom for specified MapObject Bounds │ │ │ │ │ */ │ │ │ │ │ - commit: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = [], │ │ │ │ │ - nResponses = 0; │ │ │ │ │ - │ │ │ │ │ - // Divide up features before issuing any requests. This properly │ │ │ │ │ - // counts requests in the event that any responses come in before │ │ │ │ │ - // all requests have been issued. │ │ │ │ │ - var types = {}; │ │ │ │ │ - types[OpenLayers.State.INSERT] = []; │ │ │ │ │ - types[OpenLayers.State.UPDATE] = []; │ │ │ │ │ - types[OpenLayers.State.DELETE] = []; │ │ │ │ │ - var feature, list, requestFeatures = []; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - list = types[feature.state]; │ │ │ │ │ - if (list) { │ │ │ │ │ - list.push(feature); │ │ │ │ │ - requestFeatures.push(feature); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - // tally up number of requests │ │ │ │ │ - var nRequests = (types[OpenLayers.State.INSERT].length > 0 ? 1 : 0) + │ │ │ │ │ - types[OpenLayers.State.UPDATE].length + │ │ │ │ │ - types[OpenLayers.State.DELETE].length; │ │ │ │ │ - │ │ │ │ │ - // This response will be sent to the final callback after all the others │ │ │ │ │ - // have been fired. │ │ │ │ │ - var success = true; │ │ │ │ │ - var finalResponse = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: requestFeatures │ │ │ │ │ - }); │ │ │ │ │ + getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + return this.mapObject.getBoundsZoomLevel(moBounds); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - function insertCallback(response) { │ │ │ │ │ - var len = response.features ? response.features.length : 0; │ │ │ │ │ - var fids = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - fids[i] = response.features[i].fid; │ │ │ │ │ - } │ │ │ │ │ - finalResponse.insertIds = fids; │ │ │ │ │ - callback.apply(this, [response]); │ │ │ │ │ - } │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Primitives * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ │ │ │ │ │ - function callback(response) { │ │ │ │ │ - this.callUserCallback(response, options); │ │ │ │ │ - success = success && response.success(); │ │ │ │ │ - nResponses++; │ │ │ │ │ - if (nResponses >= nRequests) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - finalResponse.code = success ? │ │ │ │ │ - OpenLayers.Protocol.Response.SUCCESS : │ │ │ │ │ - OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - options.callback.apply(options.scope, [finalResponse]); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ │ │ │ │ │ - // start issuing requests │ │ │ │ │ - var queue = types[OpenLayers.State.INSERT]; │ │ │ │ │ - if (queue.length > 0) { │ │ │ │ │ - resp.push(this.create( │ │ │ │ │ - queue, OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: insertCallback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options.create) │ │ │ │ │ - )); │ │ │ │ │ - } │ │ │ │ │ - queue = types[OpenLayers.State.UPDATE]; │ │ │ │ │ - for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ - resp.push(this.update( │ │ │ │ │ - queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: callback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options.update))); │ │ │ │ │ - } │ │ │ │ │ - queue = types[OpenLayers.State.DELETE]; │ │ │ │ │ - for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ - resp.push(this["delete"]( │ │ │ │ │ - queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: callback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options["delete"]))); │ │ │ │ │ - } │ │ │ │ │ - return resp; │ │ │ │ │ - }, │ │ │ │ │ + // LonLat │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: abort │ │ │ │ │ - * Abort an ongoing request, the response object passed to │ │ │ │ │ - * this method must come from this HTTP protocol (as a result │ │ │ │ │ - * of a create, read, update, delete or commit operation). │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getMapObjectLonLatFromLonLat │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ + * lon - {Float} │ │ │ │ │ + * lat - {Float} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject LonLat built from lon and lat params │ │ │ │ │ */ │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - response.priv.abort(); │ │ │ │ │ + getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ + var gLatLng; │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ + gLatLng = new GLatLng(lonlat.lat, lonlat.lon); │ │ │ │ │ + } else { │ │ │ │ │ + gLatLng = new GLatLng(lat, lon); │ │ │ │ │ } │ │ │ │ │ + return gLatLng; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + // Pixel │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: callUserCallback │ │ │ │ │ - * This method is used from within the commit method each time an │ │ │ │ │ - * an HTTP response is received from the server, it is responsible │ │ │ │ │ - * for calling the user-supplied callbacks. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getMapObjectPixelFromXY │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * resp - {<OpenLayers.Protocol.Response>} │ │ │ │ │ - * options - {Object} The map of options passed to the commit call. │ │ │ │ │ + * x - {Integer} │ │ │ │ │ + * y - {Integer} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject Pixel from x and y parameters │ │ │ │ │ */ │ │ │ │ │ - callUserCallback: function(resp, options) { │ │ │ │ │ - var opt = options[resp.requestType]; │ │ │ │ │ - if (opt && opt.callback) { │ │ │ │ │ - opt.callback.call(opt.scope, resp); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ + return new GPoint(x, y); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.HTTP" │ │ │ │ │ -}); │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/Script.js │ │ │ │ │ + OpenLayers/Layer/GeoRSS.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Protocol.js │ │ │ │ │ - * @requires OpenLayers/Feature/Vector.js │ │ │ │ │ - * @requires OpenLayers/Format/GeoJSON.js │ │ │ │ │ - */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * if application uses the query string, for example, for BBOX parameters, │ │ │ │ │ - * OpenLayers/Format/QueryStringFilter.js should be included in the build config file │ │ │ │ │ + * @requires OpenLayers/Layer/Markers.js │ │ │ │ │ + * @requires OpenLayers/Request/XMLHttpRequest.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Protocol.Script │ │ │ │ │ - * A basic Script protocol for vector layers. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Protocol.Script> constructor. A script protocol is used to │ │ │ │ │ - * get around the same origin policy. It works with services that return │ │ │ │ │ - * JSONP - that is, JSON wrapped in a client-specified callback. The │ │ │ │ │ - * protocol handles fetching and parsing of feature data and sends parsed │ │ │ │ │ - * features to the <callback> configured with the protocol. The protocol │ │ │ │ │ - * expects features serialized as GeoJSON by default, but can be configured │ │ │ │ │ - * to work with other formats by setting the <format> property. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Layer.GeoRSS │ │ │ │ │ + * Add GeoRSS Point features to your map. │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol> │ │ │ │ │ + * - <OpenLayers.Layer.Markers> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.Script = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ +OpenLayers.Layer.GeoRSS = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: url │ │ │ │ │ - * {String} Service URL. The service is expected to return serialized │ │ │ │ │ - * features wrapped in a named callback (where the callback name is │ │ │ │ │ - * generated by this protocol). │ │ │ │ │ - * Read-only, set through the options passed to the constructor. │ │ │ │ │ + /** │ │ │ │ │ + * Property: location │ │ │ │ │ + * {String} store url of text file │ │ │ │ │ */ │ │ │ │ │ - url: null, │ │ │ │ │ + location: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: params │ │ │ │ │ - * {Object} Query string parameters to be appended to the URL. │ │ │ │ │ - * Read-only, set through the options passed to the constructor. │ │ │ │ │ - * Example: {maxFeatures: 50} │ │ │ │ │ + /** │ │ │ │ │ + * Property: features │ │ │ │ │ + * {Array(<OpenLayers.Feature>)} │ │ │ │ │ */ │ │ │ │ │ - params: null, │ │ │ │ │ + features: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: callback │ │ │ │ │ - * {Object} Function to be called when the <read> operation completes. │ │ │ │ │ + * APIProperty: formatOptions │ │ │ │ │ + * {Object} Hash of options which should be passed to the format when it is │ │ │ │ │ + * created. Must be passed in the constructor. │ │ │ │ │ */ │ │ │ │ │ - callback: null, │ │ │ │ │ + formatOptions: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: callbackTemplate │ │ │ │ │ - * {String} Template for creating a unique callback function name │ │ │ │ │ - * for the registry. Should include ${id}. The ${id} variable will be │ │ │ │ │ - * replaced with a string identifier prefixed with a "c" (e.g. c1, c2). │ │ │ │ │ - * Default is "OpenLayers.Protocol.Script.registry.${id}". │ │ │ │ │ + /** │ │ │ │ │ + * Property: selectedFeature │ │ │ │ │ + * {<OpenLayers.Feature>} │ │ │ │ │ */ │ │ │ │ │ - callbackTemplate: "OpenLayers.Protocol.Script.registry.${id}", │ │ │ │ │ + selectedFeature: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: callbackKey │ │ │ │ │ - * {String} The name of the query string parameter that the service │ │ │ │ │ - * recognizes as the callback identifier. Default is "callback". │ │ │ │ │ - * This key is used to generate the URL for the script. For example │ │ │ │ │ - * setting <callbackKey> to "myCallback" would result in a URL like │ │ │ │ │ - * http://example.com/?myCallback=... │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: icon │ │ │ │ │ + * {<OpenLayers.Icon>}. This determines the Icon to be used on the map │ │ │ │ │ + * for this GeoRSS layer. │ │ │ │ │ */ │ │ │ │ │ - callbackKey: "callback", │ │ │ │ │ + icon: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: callbackPrefix │ │ │ │ │ - * {String} Where a service requires that the callback query string │ │ │ │ │ - * parameter value is prefixed by some string, this value may be set. │ │ │ │ │ - * For example, setting <callbackPrefix> to "foo:" would result in a │ │ │ │ │ - * URL like http://example.com/?callback=foo:... Default is "". │ │ │ │ │ + * APIProperty: popupSize │ │ │ │ │ + * {<OpenLayers.Size>} This determines the size of GeoRSS popups. If │ │ │ │ │ + * not provided, defaults to 250px by 120px. │ │ │ │ │ */ │ │ │ │ │ - callbackPrefix: "", │ │ │ │ │ + popupSize: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: scope │ │ │ │ │ - * {Object} Optional ``this`` object for the callback. Read-only, set │ │ │ │ │ - * through the options passed to the constructor. │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: useFeedTitle │ │ │ │ │ + * {Boolean} Set layer.name to the first <title> element in the feed. Default is true. │ │ │ │ │ */ │ │ │ │ │ - scope: null, │ │ │ │ │ + useFeedTitle: true, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: format │ │ │ │ │ - * {<OpenLayers.Format>} Format for parsing features. Default is an │ │ │ │ │ - * <OpenLayers.Format.GeoJSON> format. If an alternative is provided, │ │ │ │ │ - * the format's read method must take an object and return an array │ │ │ │ │ - * of features. │ │ │ │ │ + * Constructor: OpenLayers.Layer.GeoRSS │ │ │ │ │ + * Create a GeoRSS Layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * name - {String} │ │ │ │ │ + * location - {String} │ │ │ │ │ + * options - {Object} │ │ │ │ │ */ │ │ │ │ │ - format: null, │ │ │ │ │ + initialize: function(name, location, options) { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ + this.location = location; │ │ │ │ │ + this.features = []; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: pendingRequests │ │ │ │ │ - * {Object} References all pending requests. Property names are script │ │ │ │ │ - * identifiers and property values are script elements. │ │ │ │ │ + * Method: destroy │ │ │ │ │ */ │ │ │ │ │ - pendingRequests: null, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + // Warning: Layer.Markers.destroy() must be called prior to calling │ │ │ │ │ + // clearFeatures() here, otherwise we leak memory. Indeed, if │ │ │ │ │ + // Layer.Markers.destroy() is called after clearFeatures(), it won't be │ │ │ │ │ + // able to remove the marker image elements from the layer's div since │ │ │ │ │ + // the markers will have been destroyed by clearFeatures(). │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.destroy.apply(this, arguments); │ │ │ │ │ + this.clearFeatures(); │ │ │ │ │ + this.features = null; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIProperty: srsInBBOX │ │ │ │ │ - * {Boolean} Include the SRS identifier in BBOX query string parameter. │ │ │ │ │ - * Setting this property has no effect if a custom filterToParams method │ │ │ │ │ - * is provided. Default is false. If true and the layer has a │ │ │ │ │ - * projection object set, any BBOX filter will be serialized with a │ │ │ │ │ - * fifth item identifying the projection. │ │ │ │ │ - * E.g. bbox=-1000,-1000,1000,1000,EPSG:900913 │ │ │ │ │ + * Method: loadRSS │ │ │ │ │ + * Start the load of the RSS data. Don't do this when we first add the layer, │ │ │ │ │ + * since we may not be visible at any point, and it would therefore be a waste. │ │ │ │ │ */ │ │ │ │ │ - srsInBBOX: false, │ │ │ │ │ + loadRSS: function() { │ │ │ │ │ + if (!this.loaded) { │ │ │ │ │ + this.events.triggerEvent("loadstart"); │ │ │ │ │ + OpenLayers.Request.GET({ │ │ │ │ │ + url: this.location, │ │ │ │ │ + success: this.parseData, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.loaded = true; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.Script │ │ │ │ │ - * A class for giving layers generic Script protocol. │ │ │ │ │ - * │ │ │ │ │ + * Method: moveTo │ │ │ │ │ + * If layer is visible and RSS has not been loaded, load RSS. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - * │ │ │ │ │ - * Valid options include: │ │ │ │ │ - * url - {String} │ │ │ │ │ - * params - {Object} │ │ │ │ │ - * callback - {Function} │ │ │ │ │ - * scope - {Object} │ │ │ │ │ + * bounds - {Object} │ │ │ │ │ + * zoomChanged - {Object} │ │ │ │ │ + * minor - {Object} │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - this.params = {}; │ │ │ │ │ - this.pendingRequests = {}; │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (!this.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.GeoJSON(); │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ - var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ - srsInBBOX: this.srsInBBOX │ │ │ │ │ - }); │ │ │ │ │ - this.filterToParams = function(filter, params) { │ │ │ │ │ - return format.write(filter, params); │ │ │ │ │ - }; │ │ │ │ │ + moveTo: function(bounds, zoomChanged, minor) { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + if (this.visibility && !this.loaded) { │ │ │ │ │ + this.loadRSS(); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Construct a request for reading new features. │ │ │ │ │ + * Method: parseData │ │ │ │ │ + * Parse the data returned from the Events call. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object for configuring the request. │ │ │ │ │ - * This object is modified and should not be reused. │ │ │ │ │ - * │ │ │ │ │ - * Valid options: │ │ │ │ │ - * url - {String} Url for the request. │ │ │ │ │ - * params - {Object} Parameters to get serialized as a query string. │ │ │ │ │ - * filter - {<OpenLayers.Filter>} Filter to get serialized as a │ │ │ │ │ - * query string. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} A response object, whose "priv" property │ │ │ │ │ - * references the injected script. This object is also passed to the │ │ │ │ │ - * callback function when the request completes, its "features" property │ │ │ │ │ - * is then populated with the features received from the server. │ │ │ │ │ + * ajaxRequest - {<OpenLayers.Request.XMLHttpRequest>} │ │ │ │ │ */ │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - options.params = OpenLayers.Util.applyDefaults( │ │ │ │ │ - options.params, this.options.params │ │ │ │ │ - ); │ │ │ │ │ - if (options.filter && this.filterToParams) { │ │ │ │ │ - options.params = this.filterToParams( │ │ │ │ │ - options.filter, options.params │ │ │ │ │ - ); │ │ │ │ │ + parseData: function(ajaxRequest) { │ │ │ │ │ + var doc = ajaxRequest.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = OpenLayers.Format.XML.prototype.read(ajaxRequest.responseText); │ │ │ │ │ } │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - var request = this.createRequest( │ │ │ │ │ - options.url, │ │ │ │ │ - options.params, │ │ │ │ │ - OpenLayers.Function.bind(function(data) { │ │ │ │ │ - response.data = data; │ │ │ │ │ - this.handleRead(response, options); │ │ │ │ │ - }, this) │ │ │ │ │ - ); │ │ │ │ │ - response.priv = request; │ │ │ │ │ - return response; │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: filterToParams │ │ │ │ │ - * Optional method to translate an <OpenLayers.Filter> object into an object │ │ │ │ │ - * that can be serialized as request query string provided. If a custom │ │ │ │ │ - * method is not provided, any filter will not be serialized. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} filter to convert. │ │ │ │ │ - * params - {Object} The parameters object. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The resulting parameters object. │ │ │ │ │ - */ │ │ │ │ │ + if (this.useFeedTitle) { │ │ │ │ │ + var name = null; │ │ │ │ │ + try { │ │ │ │ │ + name = doc.getElementsByTagNameNS('*', 'title')[0].firstChild.nodeValue; │ │ │ │ │ + } catch (e) { │ │ │ │ │ + try { │ │ │ │ │ + name = doc.getElementsByTagName('title')[0].firstChild.nodeValue; │ │ │ │ │ + } catch (e) {} │ │ │ │ │ + } │ │ │ │ │ + if (name) { │ │ │ │ │ + this.setName(name); │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: createRequest │ │ │ │ │ - * Issues a request for features by creating injecting a script in the │ │ │ │ │ - * document head. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * url - {String} Service URL. │ │ │ │ │ - * params - {Object} Query string parameters. │ │ │ │ │ - * callback - {Function} Callback to be called with resulting data. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {HTMLScriptElement} The script pending execution. │ │ │ │ │ - */ │ │ │ │ │ - createRequest: function(url, params, callback) { │ │ │ │ │ - var id = OpenLayers.Protocol.Script.register(callback); │ │ │ │ │ - var name = OpenLayers.String.format(this.callbackTemplate, { │ │ │ │ │ - id: id │ │ │ │ │ - }); │ │ │ │ │ - params = OpenLayers.Util.extend({}, params); │ │ │ │ │ - params[this.callbackKey] = this.callbackPrefix + name; │ │ │ │ │ - url = OpenLayers.Util.urlAppend( │ │ │ │ │ - url, OpenLayers.Util.getParameterString(params) │ │ │ │ │ - ); │ │ │ │ │ - var script = document.createElement("script"); │ │ │ │ │ - script.type = "text/javascript"; │ │ │ │ │ - script.src = url; │ │ │ │ │ - script.id = "OpenLayers_Protocol_Script_" + id; │ │ │ │ │ - this.pendingRequests[script.id] = script; │ │ │ │ │ - var head = document.getElementsByTagName("head")[0]; │ │ │ │ │ - head.appendChild(script); │ │ │ │ │ - return script; │ │ │ │ │ - }, │ │ │ │ │ + var options = {}; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: destroyRequest │ │ │ │ │ - * Remove a script node associated with a response from the document. Also │ │ │ │ │ - * unregisters the callback and removes the script from the │ │ │ │ │ - * <pendingRequests> object. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * script - {HTMLScriptElement} │ │ │ │ │ - */ │ │ │ │ │ - destroyRequest: function(script) { │ │ │ │ │ - OpenLayers.Protocol.Script.unregister(script.id.split("_").pop()); │ │ │ │ │ - delete this.pendingRequests[script.id]; │ │ │ │ │ - if (script.parentNode) { │ │ │ │ │ - script.parentNode.removeChild(script); │ │ │ │ │ + OpenLayers.Util.extend(options, this.formatOptions); │ │ │ │ │ + │ │ │ │ │ + if (this.map && !this.projection.equals(this.map.getProjectionObject())) { │ │ │ │ │ + options.externalProjection = this.projection; │ │ │ │ │ + options.internalProjection = this.map.getProjectionObject(); │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleRead │ │ │ │ │ - * Individual callbacks are created for read, create and update, should │ │ │ │ │ - * a subclass need to override each one separately. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * the user callback. │ │ │ │ │ - * options - {Object} The user options passed to the read call. │ │ │ │ │ - */ │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - this.handleResponse(response, options); │ │ │ │ │ - }, │ │ │ │ │ + var format = new OpenLayers.Format.GeoRSS(options); │ │ │ │ │ + var features = format.read(doc); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleResponse │ │ │ │ │ - * Called by CRUD specific handlers. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The response object to pass to │ │ │ │ │ - * any user callback. │ │ │ │ │ - * options - {Object} The user options passed to the create, read, update, │ │ │ │ │ - * or delete call. │ │ │ │ │ - */ │ │ │ │ │ - handleResponse: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - if (response.data) { │ │ │ │ │ - response.features = this.parseFeatures(response.data); │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + var data = {}; │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + │ │ │ │ │ + // we don't support features with no geometry in the GeoRSS │ │ │ │ │ + // layer at this time. │ │ │ │ │ + if (!feature.geometry) { │ │ │ │ │ + continue; │ │ │ │ │ } │ │ │ │ │ - this.destroyRequest(response.priv); │ │ │ │ │ - options.callback.call(options.scope, response); │ │ │ │ │ + │ │ │ │ │ + var title = feature.attributes.title ? │ │ │ │ │ + feature.attributes.title : "Untitled"; │ │ │ │ │ + │ │ │ │ │ + var description = feature.attributes.description ? │ │ │ │ │ + feature.attributes.description : "No description."; │ │ │ │ │ + │ │ │ │ │ + var link = feature.attributes.link ? feature.attributes.link : ""; │ │ │ │ │ + │ │ │ │ │ + var location = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ + data.icon = this.icon == null ? │ │ │ │ │ + OpenLayers.Marker.defaultIcon() : │ │ │ │ │ + this.icon.clone(); │ │ │ │ │ + │ │ │ │ │ + data.popupSize = this.popupSize ? │ │ │ │ │ + this.popupSize.clone() : │ │ │ │ │ + new OpenLayers.Size(250, 120); │ │ │ │ │ + │ │ │ │ │ + if (title || description) { │ │ │ │ │ + // we have supplemental data, store them. │ │ │ │ │ + data.title = title; │ │ │ │ │ + data.description = description; │ │ │ │ │ + │ │ │ │ │ + var contentHTML = '<div class="olLayerGeoRSSClose">[x]</div>'; │ │ │ │ │ + contentHTML += '<div class="olLayerGeoRSSTitle">'; │ │ │ │ │ + if (link) { │ │ │ │ │ + contentHTML += '<a class="link" href="' + link + '" target="_blank">'; │ │ │ │ │ + } │ │ │ │ │ + contentHTML += title; │ │ │ │ │ + if (link) { │ │ │ │ │ + contentHTML += '</a>'; │ │ │ │ │ + } │ │ │ │ │ + contentHTML += '</div>'; │ │ │ │ │ + contentHTML += '<div style="" class="olLayerGeoRSSDescription">'; │ │ │ │ │ + contentHTML += description; │ │ │ │ │ + contentHTML += '</div>'; │ │ │ │ │ + data['popupContentHTML'] = contentHTML; │ │ │ │ │ + } │ │ │ │ │ + var feature = new OpenLayers.Feature(this, location, data); │ │ │ │ │ + this.features.push(feature); │ │ │ │ │ + var marker = feature.createMarker(); │ │ │ │ │ + marker.events.register('click', feature, this.markerClick); │ │ │ │ │ + this.addMarker(marker); │ │ │ │ │ } │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseFeatures │ │ │ │ │ - * Read Script response body and return features. │ │ │ │ │ + * Method: markerClick │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * data - {Object} The data sent to the callback function by the server. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} Array of features or a single feature. │ │ │ │ │ + * evt - {Event} │ │ │ │ │ */ │ │ │ │ │ - parseFeatures: function(data) { │ │ │ │ │ - return this.format.read(data); │ │ │ │ │ + markerClick: function(evt) { │ │ │ │ │ + var sameMarkerClicked = (this == this.layer.selectedFeature); │ │ │ │ │ + this.layer.selectedFeature = (!sameMarkerClicked) ? this : null; │ │ │ │ │ + for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ + this.layer.map.removePopup(this.layer.map.popups[i]); │ │ │ │ │ + } │ │ │ │ │ + if (!sameMarkerClicked) { │ │ │ │ │ + var popup = this.createPopup(); │ │ │ │ │ + OpenLayers.Event.observe(popup.div, "click", │ │ │ │ │ + OpenLayers.Function.bind(function() { │ │ │ │ │ + for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ + this.layer.map.removePopup(this.layer.map.popups[i]); │ │ │ │ │ + } │ │ │ │ │ + }, this) │ │ │ │ │ + ); │ │ │ │ │ + this.layer.map.addPopup(popup); │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: abort │ │ │ │ │ - * Abort an ongoing request. If no response is provided, all pending │ │ │ │ │ - * requests will be aborted. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The response object returned │ │ │ │ │ - * from a <read> request. │ │ │ │ │ + * Method: clearFeatures │ │ │ │ │ + * Destroy all features in this layer. │ │ │ │ │ */ │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - this.destroyRequest(response.priv); │ │ │ │ │ - } else { │ │ │ │ │ - for (var key in this.pendingRequests) { │ │ │ │ │ - this.destroyRequest(this.pendingRequests[key]); │ │ │ │ │ + clearFeatures: function() { │ │ │ │ │ + if (this.features != null) { │ │ │ │ │ + while (this.features.length > 0) { │ │ │ │ │ + var feature = this.features[0]; │ │ │ │ │ + OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ + feature.destroy(); │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up the protocol. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.abort(); │ │ │ │ │ - delete this.params; │ │ │ │ │ - delete this.format; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ - }, │ │ │ │ │ - │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.Script" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.GeoRSS" │ │ │ │ │ }); │ │ │ │ │ - │ │ │ │ │ -(function() { │ │ │ │ │ - var o = OpenLayers.Protocol.Script; │ │ │ │ │ - var counter = 0; │ │ │ │ │ - o.registry = {}; │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Function: OpenLayers.Protocol.Script.register │ │ │ │ │ - * Register a callback for a newly created script. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * callback - {Function} The callback to be executed when the newly added │ │ │ │ │ - * script loads. This callback will be called with a single argument │ │ │ │ │ - * that is the JSON returned by the service. │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Number} An identifier for retrieving the registered callback. │ │ │ │ │ - */ │ │ │ │ │ - o.register = function(callback) { │ │ │ │ │ - var id = "c" + (++counter); │ │ │ │ │ - o.registry[id] = function() { │ │ │ │ │ - callback.apply(this, arguments); │ │ │ │ │ - }; │ │ │ │ │ - return id; │ │ │ │ │ - }; │ │ │ │ │ - │ │ │ │ │ - /** │ │ │ │ │ - * Function: OpenLayers.Protocol.Script.unregister │ │ │ │ │ - * Unregister a callback previously registered with the register function. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * id - {Number} The identifer returned by the register function. │ │ │ │ │ - */ │ │ │ │ │ - o.unregister = function(id) { │ │ │ │ │ - delete o.registry[id]; │ │ │ │ │ - }; │ │ │ │ │ -})(); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/WFS/v1.js │ │ │ │ │ + OpenLayers/Layer/UTFGrid.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Protocol/WFS.js │ │ │ │ │ + * @requires OpenLayers/Layer/XYZ.js │ │ │ │ │ + * @requires OpenLayers/Tile/UTFGrid.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Protocol.WFS.v1 │ │ │ │ │ - * Abstract class for for v1.0.0 and v1.1.0 protocol. │ │ │ │ │ +/** │ │ │ │ │ + * Class: OpenLayers.Layer.UTFGrid │ │ │ │ │ + * This Layer reads from UTFGrid tiled data sources. Since UTFGrids are │ │ │ │ │ + * essentially JSON-based ASCII art with attached attributes, they are not │ │ │ │ │ + * visibly rendered. In order to use them in the map, you must add a │ │ │ │ │ + * <OpenLayers.Control.UTFGrid> control as well. │ │ │ │ │ + * │ │ │ │ │ + * Example: │ │ │ │ │ + * │ │ │ │ │ + * (start code) │ │ │ │ │ + * var world_utfgrid = new OpenLayers.Layer.UTFGrid({ │ │ │ │ │ + * url: "/tiles/world_utfgrid/${z}/${x}/${y}.json", │ │ │ │ │ + * utfgridResolution: 4, │ │ │ │ │ + * displayInLayerSwitcher: false │ │ │ │ │ + * ); │ │ │ │ │ + * map.addLayer(world_utfgrid); │ │ │ │ │ + * │ │ │ │ │ + * var control = new OpenLayers.Control.UTFGrid({ │ │ │ │ │ + * layers: [world_utfgrid], │ │ │ │ │ + * handlerMode: 'move', │ │ │ │ │ + * callback: function(dataLookup) { │ │ │ │ │ + * // do something with returned data │ │ │ │ │ + * } │ │ │ │ │ + * }) │ │ │ │ │ + * (end code) │ │ │ │ │ * │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol> │ │ │ │ │ + * - <OpenLayers.Layer.XYZ> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.WFS.v1 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ +OpenLayers.Layer.UTFGrid = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} WFS version number. │ │ │ │ │ + * APIProperty: isBaseLayer │ │ │ │ │ + * Default is false, as UTFGrids are designed to be a transparent overlay layer. │ │ │ │ │ */ │ │ │ │ │ - version: null, │ │ │ │ │ + isBaseLayer: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: srsName │ │ │ │ │ - * {String} Name of spatial reference system. Default is "EPSG:4326". │ │ │ │ │ + * APIProperty: projection │ │ │ │ │ + * {<OpenLayers.Projection>} │ │ │ │ │ + * Source projection for the UTFGrids. Default is "EPSG:900913". │ │ │ │ │ */ │ │ │ │ │ - srsName: "EPSG:4326", │ │ │ │ │ + projection: new OpenLayers.Projection("EPSG:900913"), │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: featureType │ │ │ │ │ - * {String} Local feature typeName. │ │ │ │ │ + * Property: useJSONP │ │ │ │ │ + * {Boolean} │ │ │ │ │ + * Should we use a JSONP script approach instead of a standard AJAX call? │ │ │ │ │ + * │ │ │ │ │ + * Set to true for using utfgrids from another server. │ │ │ │ │ + * Avoids same-domain policy restrictions. │ │ │ │ │ + * Note that this only works if the server accepts │ │ │ │ │ + * the callback GET parameter and dynamically │ │ │ │ │ + * wraps the returned json in a function call. │ │ │ │ │ + * │ │ │ │ │ + * Default is false │ │ │ │ │ */ │ │ │ │ │ - featureType: null, │ │ │ │ │ + useJSONP: false, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: featureNS │ │ │ │ │ - * {String} Feature namespace. │ │ │ │ │ + * APIProperty: url │ │ │ │ │ + * {String} │ │ │ │ │ + * URL tempate for UTFGrid tiles. Include x, y, and z parameters. │ │ │ │ │ + * E.g. "/tiles/${z}/${x}/${y}.json" │ │ │ │ │ */ │ │ │ │ │ - featureNS: null, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: geometryName │ │ │ │ │ - * {String} Name of the geometry attribute for features. Default is │ │ │ │ │ - * "the_geom" for WFS <version> 1.0, and null for higher versions. │ │ │ │ │ + * APIProperty: utfgridResolution │ │ │ │ │ + * {Number} │ │ │ │ │ + * Ratio of the pixel width to the width of a UTFGrid data point. If an │ │ │ │ │ + * entry in the grid represents a 4x4 block of pixels, the │ │ │ │ │ + * utfgridResolution would be 4. Default is 2 (specified in │ │ │ │ │ + * <OpenLayers.Tile.UTFGrid>). │ │ │ │ │ */ │ │ │ │ │ - geometryName: "the_geom", │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: maxFeatures │ │ │ │ │ - * {Integer} Optional maximum number of features to retrieve. │ │ │ │ │ + * Property: tileClass │ │ │ │ │ + * {<OpenLayers.Tile>} The tile class to use for this layer. │ │ │ │ │ + * Defaults is <OpenLayers.Tile.UTFGrid>. │ │ │ │ │ */ │ │ │ │ │ + tileClass: OpenLayers.Tile.UTFGrid, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: schema │ │ │ │ │ - * {String} Optional schema location that will be included in the │ │ │ │ │ - * schemaLocation attribute value. Note that the feature type schema │ │ │ │ │ - * is required for a strict XML validator (on transactions with an │ │ │ │ │ - * insert for example), but is *not* required by the WFS specification │ │ │ │ │ - * (since the server is supposed to know about feature type schemas). │ │ │ │ │ + * Constructor: OpenLayers.Layer.UTFGrid │ │ │ │ │ + * Create a new UTFGrid layer. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * config - {Object} Configuration properties for the layer. │ │ │ │ │ + * │ │ │ │ │ + * Required configuration properties: │ │ │ │ │ + * url - {String} The url template for UTFGrid tiles. See the <url> property. │ │ │ │ │ */ │ │ │ │ │ - schema: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply( │ │ │ │ │ + this, [options.name, options.url, {}, options] │ │ │ │ │ + ); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + utfgridResolution: this.utfgridResolution │ │ │ │ │ + }, this.tileOptions); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: featurePrefix │ │ │ │ │ - * {String} Namespace alias for feature type. Default is "feature". │ │ │ │ │ + * Method: createBackBuffer │ │ │ │ │ + * The UTFGrid cannot create a back buffer, so this method is overriden. │ │ │ │ │ */ │ │ │ │ │ - featurePrefix: "feature", │ │ │ │ │ + createBackBuffer: function() {}, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Property: formatOptions │ │ │ │ │ - * {Object} Optional options for the format. If a format is not provided, │ │ │ │ │ - * this property can be used to extend the default format options. │ │ │ │ │ + * APIMethod: clone │ │ │ │ │ + * Create a clone of this layer │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * obj - {Object} Only used by a subclass of this layer. │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Layer.UTFGrid>} An exact clone of this OpenLayers.Layer.UTFGrid │ │ │ │ │ */ │ │ │ │ │ - formatOptions: null, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.UTFGrid(this.getOptions()); │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readFormat │ │ │ │ │ - * {<OpenLayers.Format>} For WFS requests it is possible to get a │ │ │ │ │ - * different output format than GML. In that case, we cannot parse │ │ │ │ │ - * the response with the default format (WFST) and we need a different │ │ │ │ │ - * format for reading. │ │ │ │ │ - */ │ │ │ │ │ - readFormat: null, │ │ │ │ │ + // get all additions from superclasses │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: readOptions │ │ │ │ │ - * {Object} Optional object to pass to format's read. │ │ │ │ │ - */ │ │ │ │ │ - readOptions: null, │ │ │ │ │ + return obj; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.WFS │ │ │ │ │ - * A class for giving layers WFS protocol. │ │ │ │ │ + * APIProperty: getFeatureInfo │ │ │ │ │ + * Get details about a feature associated with a map location. The object │ │ │ │ │ + * returned will have id and data properties. If the given location │ │ │ │ │ + * doesn't correspond to a feature, null will be returned. │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ + * location - {<OpenLayers.LonLat>} map location │ │ │ │ │ * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * url - {String} URL to send requests to (required). │ │ │ │ │ - * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ - * featureNS - {String} Feature namespace (required, but can be autodetected │ │ │ │ │ - * during the first query if GML is used as readFormat and │ │ │ │ │ - * featurePrefix is provided and matches the prefix used by the server │ │ │ │ │ - * for this featureType). │ │ │ │ │ - * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ - * for writing if featureNS is provided). Default is 'feature'. │ │ │ │ │ - * geometryName - {String} Name of geometry attribute. The default is │ │ │ │ │ - * 'the_geom' for WFS <version> 1.0, and null for higher versions. If │ │ │ │ │ - * null, it will be set to the name of the first geometry found in the │ │ │ │ │ - * first read operation. │ │ │ │ │ - * multi - {Boolean} If set to true, geometries will be casted to Multi │ │ │ │ │ - * geometries before they are written in a transaction. No casting will │ │ │ │ │ - * be done when reading features. │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} Object representing the feature id and UTFGrid data │ │ │ │ │ + * corresponding to the given map location. Returns null if the given │ │ │ │ │ + * location doesn't hit a feature. │ │ │ │ │ */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!options.format) { │ │ │ │ │ - this.format = OpenLayers.Format.WFST(OpenLayers.Util.extend({ │ │ │ │ │ - version: this.version, │ │ │ │ │ - featureType: this.featureType, │ │ │ │ │ - featureNS: this.featureNS, │ │ │ │ │ - featurePrefix: this.featurePrefix, │ │ │ │ │ - geometryName: this.geometryName, │ │ │ │ │ - srsName: this.srsName, │ │ │ │ │ - schema: this.schema │ │ │ │ │ - }, this.formatOptions)); │ │ │ │ │ - } │ │ │ │ │ - if (!options.geometryName && parseFloat(this.format.version) > 1.0) { │ │ │ │ │ - this.setGeometryName(null); │ │ │ │ │ + getFeatureInfo: function(location) { │ │ │ │ │ + var info = null; │ │ │ │ │ + var tileInfo = this.getTileData(location); │ │ │ │ │ + if (tileInfo && tileInfo.tile) { │ │ │ │ │ + info = tileInfo.tile.getFeatureInfo(tileInfo.i, tileInfo.j); │ │ │ │ │ } │ │ │ │ │ + return info; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up the protocol. │ │ │ │ │ + * APIMethod: getFeatureId │ │ │ │ │ + * Get the identifier for the feature associated with a map location. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * location - {<OpenLayers.LonLat>} map location │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {String} The feature identifier corresponding to the given map location. │ │ │ │ │ + * Returns null if the location doesn't hit a feature. │ │ │ │ │ */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.options && !this.options.format) { │ │ │ │ │ - this.format.destroy(); │ │ │ │ │ + getFeatureId: function(location) { │ │ │ │ │ + var id = null; │ │ │ │ │ + var info = this.getTileData(location); │ │ │ │ │ + if (info.tile) { │ │ │ │ │ + id = info.tile.getFeatureId(info.i, info.j); │ │ │ │ │ } │ │ │ │ │ - this.format = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ + return id; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.UTFGrid" │ │ │ │ │ +}); │ │ │ │ │ +/* ====================================================================== │ │ │ │ │ + OpenLayers/Layer/Google/v3.js │ │ │ │ │ + ====================================================================== */ │ │ │ │ │ + │ │ │ │ │ +/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ + * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ + * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ + * full text of the license. */ │ │ │ │ │ + │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * @requires OpenLayers/Layer/Google.js │ │ │ │ │ + */ │ │ │ │ │ + │ │ │ │ │ +/** │ │ │ │ │ + * Constant: OpenLayers.Layer.Google.v3 │ │ │ │ │ + * │ │ │ │ │ + * Mixin providing functionality specific to the Google Maps API v3. │ │ │ │ │ + * │ │ │ │ │ + * To use this layer, you must include the GMaps v3 API in your html. │ │ │ │ │ + * │ │ │ │ │ + * Note that this layer configures the google.maps.map object with the │ │ │ │ │ + * "disableDefaultUI" option set to true. Using UI controls that the Google │ │ │ │ │ + * Maps API provides is not supported by the OpenLayers API. │ │ │ │ │ + */ │ │ │ │ │ +OpenLayers.Layer.Google.v3 = { │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Construct a request for reading new features. Since WFS splits the │ │ │ │ │ - * basic CRUD operations into GetFeature requests (for read) and │ │ │ │ │ - * Transactions (for all others), this method does not make use of the │ │ │ │ │ - * format's read method (that is only about reading transaction │ │ │ │ │ - * responses). │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Options for the read operation, in addition to the │ │ │ │ │ - * options set on the instance (options set here will take precedence). │ │ │ │ │ - * │ │ │ │ │ - * To use a configured protocol to get e.g. a WFS hit count, applications │ │ │ │ │ - * could do the following: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * protocol.read({ │ │ │ │ │ - * readOptions: {output: "object"}, │ │ │ │ │ - * resultType: "hits", │ │ │ │ │ - * maxFeatures: null, │ │ │ │ │ - * callback: function(resp) { │ │ │ │ │ - * // process resp.numberOfFeatures here │ │ │ │ │ - * } │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * To use a configured protocol to use WFS paging (if supported by the │ │ │ │ │ - * server), applications could do the following: │ │ │ │ │ - * │ │ │ │ │ - * (code) │ │ │ │ │ - * protocol.read({ │ │ │ │ │ - * startIndex: 0, │ │ │ │ │ - * count: 50 │ │ │ │ │ - * }); │ │ │ │ │ - * (end) │ │ │ │ │ - * │ │ │ │ │ - * To limit the attributes returned by the GetFeature request, applications │ │ │ │ │ - * can use the propertyNames option to specify the properties to include in │ │ │ │ │ - * the response: │ │ │ │ │ - * │ │ │ │ │ + * Constant: DEFAULTS │ │ │ │ │ + * {Object} It is not recommended to change the properties set here. Note │ │ │ │ │ + * that Google.v3 layers only work when sphericalMercator is set to true. │ │ │ │ │ + * │ │ │ │ │ * (code) │ │ │ │ │ - * protocol.read({ │ │ │ │ │ - * propertyNames: ["DURATION", "INTENSITY"] │ │ │ │ │ - * }); │ │ │ │ │ + * { │ │ │ │ │ + * sphericalMercator: true, │ │ │ │ │ + * projection: "EPSG:900913" │ │ │ │ │ + * } │ │ │ │ │ * (end) │ │ │ │ │ */ │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - var data = OpenLayers.Format.XML.prototype.write.apply( │ │ │ │ │ - this.format, [this.format.writeNode("wfs:GetFeature", options)] │ │ │ │ │ - ); │ │ │ │ │ - │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: data │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - return response; │ │ │ │ │ + DEFAULTS: { │ │ │ │ │ + sphericalMercator: true, │ │ │ │ │ + projection: "EPSG:900913" │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setFeatureType │ │ │ │ │ - * Change the feature type on the fly. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * featureType - {String} Local (without prefix) feature typeName. │ │ │ │ │ + * APIProperty: animationEnabled │ │ │ │ │ + * {Boolean} If set to true, the transition between zoom levels will be │ │ │ │ │ + * animated (if supported by the GMaps API for the device used). Set to │ │ │ │ │ + * false to match the zooming experience of other layer types. Default │ │ │ │ │ + * is true. Note that the GMaps API does not give us control over zoom │ │ │ │ │ + * animation, so if set to false, when zooming, this will make the │ │ │ │ │ + * layer temporarily invisible, wait until GMaps reports the map being │ │ │ │ │ + * idle, and make it visible again. The result will be a blank layer │ │ │ │ │ + * for a few moments while zooming. │ │ │ │ │ */ │ │ │ │ │ - setFeatureType: function(featureType) { │ │ │ │ │ - this.featureType = featureType; │ │ │ │ │ - this.format.featureType = featureType; │ │ │ │ │ + animationEnabled: true, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: loadMapObject │ │ │ │ │ + * Load the GMap and register appropriate event listeners. │ │ │ │ │ + */ │ │ │ │ │ + loadMapObject: function() { │ │ │ │ │ + if (!this.type) { │ │ │ │ │ + this.type = google.maps.MapTypeId.ROADMAP; │ │ │ │ │ + } │ │ │ │ │ + var mapObject; │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + // there are already Google layers added to this map │ │ │ │ │ + mapObject = cache.mapObject; │ │ │ │ │ + // increment the layer count │ │ │ │ │ + ++cache.count; │ │ │ │ │ + } else { │ │ │ │ │ + // this is the first Google layer for this map │ │ │ │ │ + // create GMap │ │ │ │ │ + var center = this.map.getCenter(); │ │ │ │ │ + var container = document.createElement('div'); │ │ │ │ │ + container.className = "olForeignContainer"; │ │ │ │ │ + container.style.width = '100%'; │ │ │ │ │ + container.style.height = '100%'; │ │ │ │ │ + mapObject = new google.maps.Map(container, { │ │ │ │ │ + center: center ? │ │ │ │ │ + new google.maps.LatLng(center.lat, center.lon) : new google.maps.LatLng(0, 0), │ │ │ │ │ + zoom: this.map.getZoom() || 0, │ │ │ │ │ + mapTypeId: this.type, │ │ │ │ │ + disableDefaultUI: true, │ │ │ │ │ + keyboardShortcuts: false, │ │ │ │ │ + draggable: false, │ │ │ │ │ + disableDoubleClickZoom: true, │ │ │ │ │ + scrollwheel: false, │ │ │ │ │ + streetViewControl: false │ │ │ │ │ + }); │ │ │ │ │ + var googleControl = document.createElement('div'); │ │ │ │ │ + googleControl.style.width = '100%'; │ │ │ │ │ + googleControl.style.height = '100%'; │ │ │ │ │ + mapObject.controls[google.maps.ControlPosition.TOP_LEFT].push(googleControl); │ │ │ │ │ + │ │ │ │ │ + // cache elements for use by any other google layers added to │ │ │ │ │ + // this same map │ │ │ │ │ + cache = { │ │ │ │ │ + googleControl: googleControl, │ │ │ │ │ + mapObject: mapObject, │ │ │ │ │ + count: 1 │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Layer.Google.cache[this.map.id] = cache; │ │ │ │ │ + } │ │ │ │ │ + this.mapObject = mapObject; │ │ │ │ │ + this.setGMapVisibility(this.visibility); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * APIMethod: setGeometryName │ │ │ │ │ - * Sets the geometryName option after instantiation. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * geometryName - {String} Name of geometry attribute. │ │ │ │ │ + * APIMethod: onMapResize │ │ │ │ │ */ │ │ │ │ │ - setGeometryName: function(geometryName) { │ │ │ │ │ - this.geometryName = geometryName; │ │ │ │ │ - this.format.geometryName = geometryName; │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + if (this.visibility) { │ │ │ │ │ + google.maps.event.trigger(this.mapObject, "resize"); │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleRead │ │ │ │ │ - * Deal with response from the read request. │ │ │ │ │ - * │ │ │ │ │ + * Method: setGMapVisibility │ │ │ │ │ + * Display the GMap container and associated elements. │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ - * to the user callback. │ │ │ │ │ - * options - {Object} The user options passed to the read call. │ │ │ │ │ + * visible - {Boolean} Display the GMap elements. │ │ │ │ │ */ │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - // success │ │ │ │ │ - var result = this.parseResponse(request, options.readOptions); │ │ │ │ │ - if (result && result.success !== false) { │ │ │ │ │ - if (options.readOptions && options.readOptions.output == "object") { │ │ │ │ │ - OpenLayers.Util.extend(response, result); │ │ │ │ │ + setGMapVisibility: function(visible) { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + var map = this.map; │ │ │ │ │ + if (cache) { │ │ │ │ │ + var type = this.type; │ │ │ │ │ + var layers = map.layers; │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = layers[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Google && │ │ │ │ │ + layer.visibility === true && layer.inRange === true) { │ │ │ │ │ + type = layer.type; │ │ │ │ │ + visible = true; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var container = this.mapObject.getDiv(); │ │ │ │ │ + if (visible === true) { │ │ │ │ │ + if (container.parentNode !== map.div) { │ │ │ │ │ + if (!cache.rendered) { │ │ │ │ │ + var me = this; │ │ │ │ │ + google.maps.event.addListenerOnce(this.mapObject, 'tilesloaded', function() { │ │ │ │ │ + cache.rendered = true; │ │ │ │ │ + me.setGMapVisibility(me.getVisibility()); │ │ │ │ │ + me.moveTo(me.map.getCenter()); │ │ │ │ │ + }); │ │ │ │ │ } else { │ │ │ │ │ - response.features = result; │ │ │ │ │ + map.div.appendChild(container); │ │ │ │ │ + cache.googleControl.appendChild(map.viewPortDiv); │ │ │ │ │ + google.maps.event.trigger(this.mapObject, 'resize'); │ │ │ │ │ } │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ - } else { │ │ │ │ │ - // failure (service exception) │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - response.error = result; │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - // failure │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + this.mapObject.setMapTypeId(type); │ │ │ │ │ + } else if (cache.googleControl.hasChildNodes()) { │ │ │ │ │ + map.div.appendChild(map.viewPortDiv); │ │ │ │ │ + map.div.removeChild(container); │ │ │ │ │ } │ │ │ │ │ - options.callback.call(options.scope, response); │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: parseResponse │ │ │ │ │ - * Read HTTP response body and return features │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * request - {XMLHttpRequest} The request object │ │ │ │ │ - * options - {Object} Optional object to pass to format's read │ │ │ │ │ - * │ │ │ │ │ + * Method: getMapContainer │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {Object} or {Array({<OpenLayers.Feature.Vector>})} or │ │ │ │ │ - * {<OpenLayers.Feature.Vector>} │ │ │ │ │ - * An object with a features property, an array of features or a single │ │ │ │ │ - * feature. │ │ │ │ │ + * {DOMElement} the GMap container's div │ │ │ │ │ */ │ │ │ │ │ - parseResponse: function(request, options) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - var result = (this.readFormat !== null) ? this.readFormat.read(doc) : │ │ │ │ │ - this.format.read(doc, options); │ │ │ │ │ - if (!this.featureNS) { │ │ │ │ │ - var format = this.readFormat || this.format; │ │ │ │ │ - this.featureNS = format.featureNS; │ │ │ │ │ - // no need to auto-configure again on subsequent reads │ │ │ │ │ - format.autoConfig = false; │ │ │ │ │ - if (!this.geometryName) { │ │ │ │ │ - this.setGeometryName(format.geometryName); │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return result; │ │ │ │ │ + getMapContainer: function() { │ │ │ │ │ + return this.mapObject.getDiv(); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ + // │ │ │ │ │ + // TRANSLATION: MapObject Bounds <-> OpenLayers.Bounds │ │ │ │ │ + // │ │ │ │ │ + │ │ │ │ │ /** │ │ │ │ │ - * Method: commit │ │ │ │ │ - * Given a list of feature, assemble a batch request for update, create, │ │ │ │ │ - * and delete transactions. A commit call on the prototype amounts │ │ │ │ │ - * to writing a WFS transaction - so the write method on the format │ │ │ │ │ - * is used. │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getMapObjectBoundsFromOLBounds │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * features - {Array(<OpenLayers.Feature.Vector>)} │ │ │ │ │ - * options - {Object} │ │ │ │ │ - * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * nativeElements - {Array({Object})} Array of objects with information for writing │ │ │ │ │ - * out <Native> elements, these objects have vendorId, safeToIgnore and │ │ │ │ │ - * value properties. The <Native> element is intended to allow access to │ │ │ │ │ - * vendor specific capabilities of any particular web feature server or │ │ │ │ │ - * datastore. │ │ │ │ │ - * │ │ │ │ │ + * olBounds - {<OpenLayers.Bounds>} │ │ │ │ │ + * │ │ │ │ │ * Returns: │ │ │ │ │ - * {<OpenLayers.Protocol.Response>} A response object with a features │ │ │ │ │ - * property containing any insertIds and a priv property referencing │ │ │ │ │ - * the XMLHttpRequest object. │ │ │ │ │ + * {Object} A MapObject Bounds, translated from olBounds │ │ │ │ │ + * Returns null if null value is passed in │ │ │ │ │ */ │ │ │ │ │ - commit: function(features, options) { │ │ │ │ │ + getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ + var moBounds = null; │ │ │ │ │ + if (olBounds != null) { │ │ │ │ │ + var sw = this.sphericalMercator ? │ │ │ │ │ + this.inverseMercator(olBounds.bottom, olBounds.left) : │ │ │ │ │ + new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ + var ne = this.sphericalMercator ? │ │ │ │ │ + this.inverseMercator(olBounds.top, olBounds.right) : │ │ │ │ │ + new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ + moBounds = new google.maps.LatLngBounds( │ │ │ │ │ + new google.maps.LatLng(sw.lat, sw.lon), │ │ │ │ │ + new google.maps.LatLng(ne.lat, ne.lon) │ │ │ │ │ + ); │ │ │ │ │ + } │ │ │ │ │ + return moBounds; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "commit", │ │ │ │ │ - reqFeatures: features │ │ │ │ │ - }); │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(features, options), │ │ │ │ │ - callback: this.createCallback(this.handleCommit, response, options) │ │ │ │ │ - }); │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Interface Controls * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ │ │ │ │ │ - return response; │ │ │ │ │ - }, │ │ │ │ │ + │ │ │ │ │ + // LonLat - Pixel Translation │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: handleCommit │ │ │ │ │ - * Called when the commit request returns. │ │ │ │ │ + * APIMethod: getMapObjectLonLatFromMapObjectPixel │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ - * to the user callback. │ │ │ │ │ - * options - {Object} The user options passed to the commit call. │ │ │ │ │ + * moPixel - {Object} MapObject Pixel format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject LonLat translated from MapObject Pixel │ │ │ │ │ */ │ │ │ │ │ - handleCommit: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ + getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + var size = this.map.getSize(); │ │ │ │ │ + var lon = this.getLongitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ + var lat = this.getLatitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ + var res = this.map.getResolution(); │ │ │ │ │ │ │ │ │ │ - // ensure that we have an xml doc │ │ │ │ │ - var data = request.responseXML; │ │ │ │ │ - if (!data || !data.documentElement) { │ │ │ │ │ - data = request.responseText; │ │ │ │ │ - } │ │ │ │ │ + var delta_x = moPixel.x - (size.w / 2); │ │ │ │ │ + var delta_y = moPixel.y - (size.h / 2); │ │ │ │ │ │ │ │ │ │ - var obj = this.format.read(data) || {}; │ │ │ │ │ + var lonlat = new OpenLayers.LonLat( │ │ │ │ │ + lon + delta_x * res, │ │ │ │ │ + lat - delta_y * res │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ - response.insertIds = obj.insertIds || []; │ │ │ │ │ - if (obj.success) { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - response.error = obj; │ │ │ │ │ - } │ │ │ │ │ - options.callback.call(options.scope, response); │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + lonlat = lonlat.wrapDateLine(this.maxExtent); │ │ │ │ │ } │ │ │ │ │ + return this.getMapObjectLonLatFromLonLat(lonlat.lon, lonlat.lat); │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: filterDelete │ │ │ │ │ - * Send a request that deletes all features by their filter. │ │ │ │ │ + * APIMethod: getMapObjectPixelFromMapObjectLonLat │ │ │ │ │ * │ │ │ │ │ * Parameters: │ │ │ │ │ - * filter - {<OpenLayers.Filter>} filter │ │ │ │ │ + * moLonLat - {Object} MapObject LonLat format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject Pixel transtlated from MapObject LonLat │ │ │ │ │ */ │ │ │ │ │ - filterDelete: function(filter, options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "commit" │ │ │ │ │ - }); │ │ │ │ │ + getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + var res = this.map.getResolution(); │ │ │ │ │ + var extent = this.map.getExtent(); │ │ │ │ │ + return this.getMapObjectPixelFromXY((1 / res * (lon - extent.left)), │ │ │ │ │ + (1 / res * (extent.top - lat))); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var root = this.format.createElementNSPlus("wfs:Transaction", { │ │ │ │ │ - attributes: { │ │ │ │ │ - service: "WFS", │ │ │ │ │ - version: this.version │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ │ │ │ │ │ - var deleteNode = this.format.createElementNSPlus("wfs:Delete", { │ │ │ │ │ - attributes: { │ │ │ │ │ - typeName: (options.featureNS ? this.featurePrefix + ":" : "") + │ │ │ │ │ - options.featureType │ │ │ │ │ - } │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setMapObjectCenter │ │ │ │ │ + * Set the mapObject to the specified center and zoom │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * center - {Object} MapObject LonLat format │ │ │ │ │ + * zoom - {int} MapObject zoom format │ │ │ │ │ + */ │ │ │ │ │ + setMapObjectCenter: function(center, zoom) { │ │ │ │ │ + if (this.animationEnabled === false && zoom != this.mapObject.zoom) { │ │ │ │ │ + var mapContainer = this.getMapContainer(); │ │ │ │ │ + google.maps.event.addListenerOnce( │ │ │ │ │ + this.mapObject, │ │ │ │ │ + "idle", │ │ │ │ │ + function() { │ │ │ │ │ + mapContainer.style.visibility = ""; │ │ │ │ │ + } │ │ │ │ │ + ); │ │ │ │ │ + mapContainer.style.visibility = "hidden"; │ │ │ │ │ + } │ │ │ │ │ + this.mapObject.setOptions({ │ │ │ │ │ + center: center, │ │ │ │ │ + zoom: zoom │ │ │ │ │ }); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - if (options.featureNS) { │ │ │ │ │ - deleteNode.setAttribute("xmlns:" + this.featurePrefix, options.featureNS); │ │ │ │ │ - } │ │ │ │ │ - var filterNode = this.format.writeNode("ogc:Filter", filter); │ │ │ │ │ │ │ │ │ │ - deleteNode.appendChild(filterNode); │ │ │ │ │ + // Bounds │ │ │ │ │ │ │ │ │ │ - root.appendChild(deleteNode); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectZoomFromMapObjectBounds │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * moBounds - {Object} MapObject Bounds format │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject Zoom for specified MapObject Bounds │ │ │ │ │ + */ │ │ │ │ │ + getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + return this.mapObject.getBoundsZoomLevel(moBounds); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - var data = OpenLayers.Format.XML.prototype.write.apply( │ │ │ │ │ - this.format, [root] │ │ │ │ │ - ); │ │ │ │ │ + /************************************ │ │ │ │ │ + * * │ │ │ │ │ + * MapObject Primitives * │ │ │ │ │ + * * │ │ │ │ │ + ************************************/ │ │ │ │ │ │ │ │ │ │ - return OpenLayers.Request.POST({ │ │ │ │ │ - url: this.url, │ │ │ │ │ - callback: options.callback || function() {}, │ │ │ │ │ - data: data │ │ │ │ │ - }); │ │ │ │ │ │ │ │ │ │ - }, │ │ │ │ │ + // LonLat │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Method: abort │ │ │ │ │ - * Abort an ongoing request, the response object passed to │ │ │ │ │ - * this method must come from this protocol (as a result │ │ │ │ │ - * of a read, or commit operation). │ │ │ │ │ - * │ │ │ │ │ + * APIMethod: getMapObjectLonLatFromLonLat │ │ │ │ │ + * │ │ │ │ │ * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} │ │ │ │ │ + * lon - {Float} │ │ │ │ │ + * lat - {Float} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject LonLat built from lon and lat params │ │ │ │ │ */ │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - response.priv.abort(); │ │ │ │ │ + getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ + var gLatLng; │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ + gLatLng = new google.maps.LatLng(lonlat.lat, lonlat.lon); │ │ │ │ │ + } else { │ │ │ │ │ + gLatLng = new google.maps.LatLng(lat, lon); │ │ │ │ │ } │ │ │ │ │ + return gLatLng; │ │ │ │ │ }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.WFS.v1" │ │ │ │ │ -}); │ │ │ │ │ + // Pixel │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: getMapObjectPixelFromXY │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * x - {Integer} │ │ │ │ │ + * y - {Integer} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {Object} MapObject Pixel from x and y parameters │ │ │ │ │ + */ │ │ │ │ │ + getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ + return new google.maps.Point(x, y); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ +}; │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/WFS/v1_0_0.js │ │ │ │ │ + OpenLayers/Popup/Framed.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Protocol/WFS/v1.js │ │ │ │ │ - * @requires OpenLayers/Format/WFST/v1_0_0.js │ │ │ │ │ + * @requires OpenLayers/Popup/Anchored.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Protocol.WFS.v1_0_0 │ │ │ │ │ - * A WFS v1.0.0 protocol for vector layers. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Protocol.WFS.v1_0_0> constructor. │ │ │ │ │ - * │ │ │ │ │ + * Class: OpenLayers.Popup.Framed │ │ │ │ │ + * │ │ │ │ │ * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol.WFS.v1> │ │ │ │ │ + * - <OpenLayers.Popup.Anchored> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.WFS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ +OpenLayers.Popup.Framed = │ │ │ │ │ + OpenLayers.Class(OpenLayers.Popup.Anchored, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} WFS version number. │ │ │ │ │ - */ │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ + /** │ │ │ │ │ + * Property: imageSrc │ │ │ │ │ + * {String} location of the image to be used as the popup frame │ │ │ │ │ + */ │ │ │ │ │ + imageSrc: null, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.WFS.v1_0_0 │ │ │ │ │ - * A class for giving layers WFS v1.0.0 protocol. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ - * featureNS - {String} Feature namespace (optional). │ │ │ │ │ - * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ - * if featureNS is provided). Default is 'feature'. │ │ │ │ │ - * geometryName - {String} Name of geometry attribute. Default is 'the_geom'. │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: imageSize │ │ │ │ │ + * {<OpenLayers.Size>} Size (measured in pixels) of the image located │ │ │ │ │ + * by the 'imageSrc' property. │ │ │ │ │ + */ │ │ │ │ │ + imageSize: null, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.WFS.v1_0_0" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/WFS/v1_1_0.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isAlphaImage │ │ │ │ │ + * {Boolean} The image has some alpha and thus needs to use the alpha │ │ │ │ │ + * image hack. Note that setting this to true will have no noticeable │ │ │ │ │ + * effect in FF or IE7 browsers, but will all but crush the ie6 │ │ │ │ │ + * browser. │ │ │ │ │ + * Default is false. │ │ │ │ │ + */ │ │ │ │ │ + isAlphaImage: false, │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: positionBlocks │ │ │ │ │ + * {Object} Hash of different position blocks (Object/Hashs). Each block │ │ │ │ │ + * will be keyed by a two-character 'relativePosition' │ │ │ │ │ + * code string (ie "tl", "tr", "bl", "br"). Block properties are │ │ │ │ │ + * 'offset', 'padding' (self-explanatory), and finally the 'blocks' │ │ │ │ │ + * parameter, which is an array of the block objects. │ │ │ │ │ + * │ │ │ │ │ + * Each block object must have 'size', 'anchor', and 'position' │ │ │ │ │ + * properties. │ │ │ │ │ + * │ │ │ │ │ + * Note that positionBlocks should never be modified at runtime. │ │ │ │ │ + */ │ │ │ │ │ + positionBlocks: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Protocol/WFS/v1.js │ │ │ │ │ - * @requires OpenLayers/Format/WFST/v1_1_0.js │ │ │ │ │ - */ │ │ │ │ │ + /** │ │ │ │ │ + * Property: blocks │ │ │ │ │ + * {Array[Object]} Array of objects, each of which is one "block" of the │ │ │ │ │ + * popup. Each block has a 'div' and an 'image' property, both of │ │ │ │ │ + * which are DOMElements, and the latter of which is appended to the │ │ │ │ │ + * former. These are reused as the popup goes changing positions for │ │ │ │ │ + * great economy and elegance. │ │ │ │ │ + */ │ │ │ │ │ + blocks: null, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Protocol.WFS.v1_1_0 │ │ │ │ │ - * A WFS v1.1.0 protocol for vector layers. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Protocol.WFS.v1_1_0> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Differences from the v1.0.0 protocol: │ │ │ │ │ - * - uses Filter Encoding 1.1.0 instead of 1.0.0 │ │ │ │ │ - * - uses GML 3 instead of 2 if no format is provided │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol.WFS.v1> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol.WFS.v1_1_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: fixedRelativePosition │ │ │ │ │ + * {Boolean} We want the framed popup to work dynamically placed relative │ │ │ │ │ + * to its anchor but also in just one fixed position. A well designed │ │ │ │ │ + * framed popup will have the pixels and logic to display itself in │ │ │ │ │ + * any of the four relative positions, but (understandably), this will │ │ │ │ │ + * not be the case for all of them. By setting this property to 'true', │ │ │ │ │ + * framed popup will not recalculate for the best placement each time │ │ │ │ │ + * it's open, but will always open the same way. │ │ │ │ │ + * Note that if this is set to true, it is generally advisable to also │ │ │ │ │ + * set the 'panIntoView' property to true so that the popup can be │ │ │ │ │ + * scrolled into view (since it will often be offscreen on open) │ │ │ │ │ + * Default is false. │ │ │ │ │ + */ │ │ │ │ │ + fixedRelativePosition: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: version │ │ │ │ │ - * {String} WFS version number. │ │ │ │ │ - */ │ │ │ │ │ - version: "1.1.0", │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Popup.Framed │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ + * contentSize - {<OpenLayers.Size>} │ │ │ │ │ + * contentHTML - {String} │ │ │ │ │ + * anchor - {Object} Object to which we'll anchor the popup. Must expose │ │ │ │ │ + * a 'size' (<OpenLayers.Size>) and 'offset' (<OpenLayers.Pixel>) │ │ │ │ │ + * (Note that this is generally an <OpenLayers.Icon>). │ │ │ │ │ + * closeBox - {Boolean} │ │ │ │ │ + * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, │ │ │ │ │ + closeBoxCallback) { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.WFS.v1_1_0 │ │ │ │ │ - * A class for giving layers WFS v1.1.0 protocol. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * featureType - {String} Local (without prefix) feature typeName (required). │ │ │ │ │ - * featureNS - {String} Feature namespace (optional). │ │ │ │ │ - * featurePrefix - {String} Feature namespace alias (optional - only used │ │ │ │ │ - * if featureNS is provided). Default is 'feature'. │ │ │ │ │ - * geometryName - {String} Name of geometry attribute. Default is 'the_geom'. │ │ │ │ │ - * outputFormat - {String} Optional output format to use for WFS GetFeature │ │ │ │ │ - * requests. This can be any format advertized by the WFS's │ │ │ │ │ - * GetCapabilities response. If set, an appropriate readFormat also │ │ │ │ │ - * has to be provided, unless outputFormat is GML3, GML2 or JSON. │ │ │ │ │ - * readFormat - {<OpenLayers.Format>} An appropriate format parser if │ │ │ │ │ - * outputFormat is none of GML3, GML2 or JSON. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.WFS.v1.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (this.outputFormat && !this.readFormat) { │ │ │ │ │ - if (this.outputFormat.toLowerCase() == "gml2") { │ │ │ │ │ - this.readFormat = new OpenLayers.Format.GML.v2({ │ │ │ │ │ - featureType: this.featureType, │ │ │ │ │ - featureNS: this.featureNS, │ │ │ │ │ - geometryName: this.geometryName │ │ │ │ │ - }); │ │ │ │ │ - } else if (this.outputFormat.toLowerCase() == "json") { │ │ │ │ │ - this.readFormat = new OpenLayers.Format.GeoJSON(); │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.initialize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + if (this.fixedRelativePosition) { │ │ │ │ │ + //based on our decided relativePostion, set the current padding │ │ │ │ │ + // this keeps us from getting into trouble │ │ │ │ │ + this.updateRelativePosition(); │ │ │ │ │ + │ │ │ │ │ + //make calculateRelativePosition always return the specified │ │ │ │ │ + // fixed position. │ │ │ │ │ + this.calculateRelativePosition = function(px) { │ │ │ │ │ + return this.relativePosition; │ │ │ │ │ + }; │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.WFS.v1_1_0" │ │ │ │ │ -}); │ │ │ │ │ -/* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/CSW/v2_0_2.js │ │ │ │ │ - ====================================================================== */ │ │ │ │ │ + this.contentDiv.style.position = "absolute"; │ │ │ │ │ + this.contentDiv.style.zIndex = 1; │ │ │ │ │ │ │ │ │ │ -/* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ - * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ - * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ - * full text of the license. */ │ │ │ │ │ + if (closeBox) { │ │ │ │ │ + this.closeDiv.style.zIndex = 1; │ │ │ │ │ + } │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * @requires OpenLayers/Protocol/CSW.js │ │ │ │ │ - * @requires OpenLayers/Format/CSWGetRecords/v2_0_2.js │ │ │ │ │ - */ │ │ │ │ │ + this.groupDiv.style.position = "absolute"; │ │ │ │ │ + this.groupDiv.style.top = "0px"; │ │ │ │ │ + this.groupDiv.style.left = "0px"; │ │ │ │ │ + this.groupDiv.style.height = "100%"; │ │ │ │ │ + this.groupDiv.style.width = "100%"; │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ -/** │ │ │ │ │ - * Class: OpenLayers.Protocol.CSW.v2_0_2 │ │ │ │ │ - * CS-W (Catalogue services for the Web) version 2.0.2 protocol. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol> │ │ │ │ │ - */ │ │ │ │ │ -OpenLayers.Protocol.CSW.v2_0_2 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: destroy │ │ │ │ │ + */ │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.imageSrc = null; │ │ │ │ │ + this.imageSize = null; │ │ │ │ │ + this.isAlphaImage = null; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: formatOptions │ │ │ │ │ - * {Object} Optional options for the format. If a format is not provided, │ │ │ │ │ - * this property can be used to extend the default format options. │ │ │ │ │ - */ │ │ │ │ │ - formatOptions: null, │ │ │ │ │ + this.fixedRelativePosition = false; │ │ │ │ │ + this.positionBlocks = null; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.CSW.v2_0_2 │ │ │ │ │ - * A class for CSW version 2.0.2 protocol management. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!options.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.CSWGetRecords.v2_0_2(OpenLayers.Util.extend({}, this.formatOptions)); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + //remove our blocks │ │ │ │ │ + for (var i = 0; i < this.blocks.length; i++) { │ │ │ │ │ + var block = this.blocks[i]; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up the protocol. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.options && !this.options.format) { │ │ │ │ │ - this.format.destroy(); │ │ │ │ │ - } │ │ │ │ │ - this.format = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ - }, │ │ │ │ │ + if (block.image) { │ │ │ │ │ + block.div.removeChild(block.image); │ │ │ │ │ + } │ │ │ │ │ + block.image = null; │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: read │ │ │ │ │ - * Construct a request for reading new records from the Catalogue. │ │ │ │ │ - */ │ │ │ │ │ - read: function(options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ + if (block.div) { │ │ │ │ │ + this.groupDiv.removeChild(block.div); │ │ │ │ │ + } │ │ │ │ │ + block.div = null; │ │ │ │ │ + } │ │ │ │ │ + this.blocks = null; │ │ │ │ │ │ │ │ │ │ - var data = this.format.write(options.params || options); │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.destroy.apply(this, arguments); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: data │ │ │ │ │ - }); │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setBackgroundColor │ │ │ │ │ + */ │ │ │ │ │ + setBackgroundColor: function(color) { │ │ │ │ │ + //does nothing since the framed popup's entire scheme is based on a │ │ │ │ │ + // an image -- changing the background color makes no sense. │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - return response; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setBorder │ │ │ │ │ + */ │ │ │ │ │ + setBorder: function() { │ │ │ │ │ + //does nothing since the framed popup's entire scheme is based on a │ │ │ │ │ + // an image -- changing the popup's border makes no sense. │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleRead │ │ │ │ │ - * Deal with response from the read request. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ - * to the user callback. │ │ │ │ │ - * This response is given a code property, and optionally a data property. │ │ │ │ │ - * The latter represents the CSW records as returned by the call to │ │ │ │ │ - * the CSW format read method. │ │ │ │ │ - * options - {Object} The user options passed to the read call. │ │ │ │ │ - */ │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - // success │ │ │ │ │ - response.data = this.parseData(request); │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ - } else { │ │ │ │ │ - // failure │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + /** │ │ │ │ │ + * Method: setOpacity │ │ │ │ │ + * Sets the opacity of the popup. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * opacity - {float} A value between 0.0 (transparent) and 1.0 (solid). │ │ │ │ │ + */ │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + //does nothing since we suppose that we'll never apply an opacity │ │ │ │ │ + // to a framed popup │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * APIMethod: setSize │ │ │ │ │ + * Overridden here, because we need to update the blocks whenever the size │ │ │ │ │ + * of the popup has changed. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * contentSize - {<OpenLayers.Size>} the new size for the popup's │ │ │ │ │ + * contents div (in pixels). │ │ │ │ │ + */ │ │ │ │ │ + setSize: function(contentSize) { │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.setSize.apply(this, arguments); │ │ │ │ │ + │ │ │ │ │ + this.updateBlocks(); │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateRelativePosition │ │ │ │ │ + * When the relative position changes, we need to set the new padding │ │ │ │ │ + * BBOX on the popup, reposition the close div, and update the blocks. │ │ │ │ │ + */ │ │ │ │ │ + updateRelativePosition: function() { │ │ │ │ │ + │ │ │ │ │ + //update the padding │ │ │ │ │ + this.padding = this.positionBlocks[this.relativePosition].padding; │ │ │ │ │ + │ │ │ │ │ + //update the position of our close box to new padding │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + // use the content div's css padding to determine if we should │ │ │ │ │ + // padd the close div │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ + │ │ │ │ │ + this.closeDiv.style.right = contentDivPadding.right + │ │ │ │ │ + this.padding.right + "px"; │ │ │ │ │ + this.closeDiv.style.top = contentDivPadding.top + │ │ │ │ │ + this.padding.top + "px"; │ │ │ │ │ } │ │ │ │ │ - options.callback.call(options.scope, response); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseData │ │ │ │ │ - * Read HTTP response body and return records │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * request - {XMLHttpRequest} The request object │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Object} The CSW records as returned by the call to the format read method. │ │ │ │ │ - */ │ │ │ │ │ - parseData: function(request) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - return this.format.read(doc); │ │ │ │ │ - }, │ │ │ │ │ + this.updateBlocks(); │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.CSW.v2_0_2" │ │ │ │ │ + /** │ │ │ │ │ + * Method: calculateNewPx │ │ │ │ │ + * Besides the standard offset as determined by the Anchored class, our │ │ │ │ │ + * Framed popups have a special 'offset' property for each of their │ │ │ │ │ + * positions, which is used to offset the popup relative to its anchor. │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * px - {<OpenLayers.Pixel>} │ │ │ │ │ + * │ │ │ │ │ + * Returns: │ │ │ │ │ + * {<OpenLayers.Pixel>} The the new px position of the popup on the screen │ │ │ │ │ + * relative to the passed-in px. │ │ │ │ │ + */ │ │ │ │ │ + calculateNewPx: function(px) { │ │ │ │ │ + var newPx = OpenLayers.Popup.Anchored.prototype.calculateNewPx.apply( │ │ │ │ │ + this, arguments │ │ │ │ │ + ); │ │ │ │ │ │ │ │ │ │ -}); │ │ │ │ │ + newPx = newPx.offset(this.positionBlocks[this.relativePosition].offset); │ │ │ │ │ + │ │ │ │ │ + return newPx; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: createBlocks │ │ │ │ │ + */ │ │ │ │ │ + createBlocks: function() { │ │ │ │ │ + this.blocks = []; │ │ │ │ │ + │ │ │ │ │ + //since all positions contain the same number of blocks, we can │ │ │ │ │ + // just pick the first position and use its blocks array to create │ │ │ │ │ + // our blocks array │ │ │ │ │ + var firstPosition = null; │ │ │ │ │ + for (var key in this.positionBlocks) { │ │ │ │ │ + firstPosition = key; │ │ │ │ │ + break; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + var position = this.positionBlocks[firstPosition]; │ │ │ │ │ + for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ + │ │ │ │ │ + var block = {}; │ │ │ │ │ + this.blocks.push(block); │ │ │ │ │ + │ │ │ │ │ + var divId = this.id + '_FrameDecorationDiv_' + i; │ │ │ │ │ + block.div = OpenLayers.Util.createDiv(divId, │ │ │ │ │ + null, null, null, "absolute", null, "hidden", null │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + var imgId = this.id + '_FrameDecorationImg_' + i; │ │ │ │ │ + var imageCreator = │ │ │ │ │ + (this.isAlphaImage) ? OpenLayers.Util.createAlphaImageDiv : │ │ │ │ │ + OpenLayers.Util.createImage; │ │ │ │ │ + │ │ │ │ │ + block.image = imageCreator(imgId, │ │ │ │ │ + null, this.imageSize, this.imageSrc, │ │ │ │ │ + "absolute", null, null, null │ │ │ │ │ + ); │ │ │ │ │ + │ │ │ │ │ + block.div.appendChild(block.image); │ │ │ │ │ + this.groupDiv.appendChild(block.div); │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Method: updateBlocks │ │ │ │ │ + * Internal method, called on initialize and when the popup's relative │ │ │ │ │ + * position has changed. This function takes care of re-positioning │ │ │ │ │ + * the popup's blocks in their appropropriate places. │ │ │ │ │ + */ │ │ │ │ │ + updateBlocks: function() { │ │ │ │ │ + if (!this.blocks) { │ │ │ │ │ + this.createBlocks(); │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + if (this.size && this.relativePosition) { │ │ │ │ │ + var position = this.positionBlocks[this.relativePosition]; │ │ │ │ │ + for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ + │ │ │ │ │ + var positionBlock = position.blocks[i]; │ │ │ │ │ + var block = this.blocks[i]; │ │ │ │ │ + │ │ │ │ │ + // adjust sizes │ │ │ │ │ + var l = positionBlock.anchor.left; │ │ │ │ │ + var b = positionBlock.anchor.bottom; │ │ │ │ │ + var r = positionBlock.anchor.right; │ │ │ │ │ + var t = positionBlock.anchor.top; │ │ │ │ │ + │ │ │ │ │ + //note that we use the isNaN() test here because if the │ │ │ │ │ + // size object is initialized with a "auto" parameter, the │ │ │ │ │ + // size constructor calls parseFloat() on the string, │ │ │ │ │ + // which will turn it into NaN │ │ │ │ │ + // │ │ │ │ │ + var w = (isNaN(positionBlock.size.w)) ? this.size.w - (r + l) : │ │ │ │ │ + positionBlock.size.w; │ │ │ │ │ + │ │ │ │ │ + var h = (isNaN(positionBlock.size.h)) ? this.size.h - (b + t) : │ │ │ │ │ + positionBlock.size.h; │ │ │ │ │ + │ │ │ │ │ + block.div.style.width = (w < 0 ? 0 : w) + 'px'; │ │ │ │ │ + block.div.style.height = (h < 0 ? 0 : h) + 'px'; │ │ │ │ │ + │ │ │ │ │ + block.div.style.left = (l != null) ? l + 'px' : ''; │ │ │ │ │ + block.div.style.bottom = (b != null) ? b + 'px' : ''; │ │ │ │ │ + block.div.style.right = (r != null) ? r + 'px' : ''; │ │ │ │ │ + block.div.style.top = (t != null) ? t + 'px' : ''; │ │ │ │ │ + │ │ │ │ │ + block.image.style.left = positionBlock.position.x + 'px'; │ │ │ │ │ + block.image.style.top = positionBlock.position.y + 'px'; │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + this.contentDiv.style.left = this.padding.left + "px"; │ │ │ │ │ + this.contentDiv.style.top = this.padding.top + "px"; │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup.Framed" │ │ │ │ │ + }); │ │ │ │ │ /* ====================================================================== │ │ │ │ │ - OpenLayers/Protocol/SOS/v1_0_0.js │ │ │ │ │ + OpenLayers/Popup/FramedCloud.js │ │ │ │ │ ====================================================================== */ │ │ │ │ │ │ │ │ │ │ /* Copyright (c) 2006-2013 by OpenLayers Contributors (see authors.txt for │ │ │ │ │ * full list of contributors). Published under the 2-clause BSD license. │ │ │ │ │ * See license.txt in the OpenLayers distribution or repository for the │ │ │ │ │ * full text of the license. */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * @requires OpenLayers/Protocol/SOS.js │ │ │ │ │ - * @requires OpenLayers/Format/SOSGetFeatureOfInterest.js │ │ │ │ │ + * @requires OpenLayers/Popup/Framed.js │ │ │ │ │ + * @requires OpenLayers/Util.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Bounds.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Pixel.js │ │ │ │ │ + * @requires OpenLayers/BaseTypes/Size.js │ │ │ │ │ */ │ │ │ │ │ │ │ │ │ │ /** │ │ │ │ │ - * Class: OpenLayers.Protocol.SOS.v1_0_0 │ │ │ │ │ - * An SOS v1.0.0 Protocol for vector layers. Create a new instance with the │ │ │ │ │ - * <OpenLayers.Protocol.SOS.v1_0_0> constructor. │ │ │ │ │ - * │ │ │ │ │ - * Inherits from: │ │ │ │ │ - * - <OpenLayers.Protocol> │ │ │ │ │ + * Class: OpenLayers.Popup.FramedCloud │ │ │ │ │ + * │ │ │ │ │ + * Inherits from: │ │ │ │ │ + * - <OpenLayers.Popup.Framed> │ │ │ │ │ */ │ │ │ │ │ -OpenLayers.Protocol.SOS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ +OpenLayers.Popup.FramedCloud = │ │ │ │ │ + OpenLayers.Class(OpenLayers.Popup.Framed, { │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIProperty: fois │ │ │ │ │ - * {Array(String)} Array of features of interest (foi) │ │ │ │ │ - */ │ │ │ │ │ - fois: null, │ │ │ │ │ + /** │ │ │ │ │ + * Property: contentDisplayClass │ │ │ │ │ + * {String} The CSS class of the popup content div. │ │ │ │ │ + */ │ │ │ │ │ + contentDisplayClass: "olFramedCloudPopupContent", │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Property: formatOptions │ │ │ │ │ - * {Object} Optional options for the format. If a format is not provided, │ │ │ │ │ - * this property can be used to extend the default format options. │ │ │ │ │ - */ │ │ │ │ │ - formatOptions: null, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: autoSize │ │ │ │ │ + * {Boolean} Framed Cloud is autosizing by default. │ │ │ │ │ + */ │ │ │ │ │ + autoSize: true, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Constructor: OpenLayers.Protocol.SOS │ │ │ │ │ - * A class for giving layers an SOS protocol. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * options - {Object} Optional object whose properties will be set on the │ │ │ │ │ - * instance. │ │ │ │ │ - * │ │ │ │ │ - * Valid options properties: │ │ │ │ │ - * url - {String} URL to send requests to (required). │ │ │ │ │ - * fois - {Array} The features of interest (required). │ │ │ │ │ - */ │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!options.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.SOSGetFeatureOfInterest( │ │ │ │ │ - this.formatOptions); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: panMapIfOutOfView │ │ │ │ │ + * {Boolean} Framed Cloud does pan into view by default. │ │ │ │ │ + */ │ │ │ │ │ + panMapIfOutOfView: true, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: destroy │ │ │ │ │ - * Clean up the protocol. │ │ │ │ │ - */ │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.options && !this.options.format) { │ │ │ │ │ - this.format.destroy(); │ │ │ │ │ - } │ │ │ │ │ - this.format = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: imageSize │ │ │ │ │ + * {<OpenLayers.Size>} │ │ │ │ │ + */ │ │ │ │ │ + imageSize: new OpenLayers.Size(1276, 736), │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * APIMethod: read │ │ │ │ │ - * Construct a request for reading new sensor positions. This is done by │ │ │ │ │ - * issuing one GetFeatureOfInterest request. │ │ │ │ │ - */ │ │ │ │ │ - read: function(options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - var format = this.format; │ │ │ │ │ - var data = OpenLayers.Format.XML.prototype.write.apply(format, │ │ │ │ │ - [format.writeNode("sos:GetFeatureOfInterest", { │ │ │ │ │ - fois: this.fois │ │ │ │ │ - })] │ │ │ │ │ - ); │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ - data: data │ │ │ │ │ - }); │ │ │ │ │ - return response; │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: isAlphaImage │ │ │ │ │ + * {Boolean} The FramedCloud does not use an alpha image (in honor of the │ │ │ │ │ + * good ie6 folk out there) │ │ │ │ │ + */ │ │ │ │ │ + isAlphaImage: false, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: handleRead │ │ │ │ │ - * Deal with response from the read request. │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * response - {<OpenLayers.Protocol.Response>} The response object to pass │ │ │ │ │ - * to the user callback. │ │ │ │ │ - * options - {Object} The user options passed to the read call. │ │ │ │ │ - */ │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - // success │ │ │ │ │ - response.features = this.parseFeatures(request); │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS; │ │ │ │ │ - } else { │ │ │ │ │ - // failure │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: fixedRelativePosition │ │ │ │ │ + * {Boolean} The Framed Cloud popup works in just one fixed position. │ │ │ │ │ + */ │ │ │ │ │ + fixedRelativePosition: false, │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Property: positionBlocks │ │ │ │ │ + * {Object} Hash of differen position blocks, keyed by relativePosition │ │ │ │ │ + * two-character code string (ie "tl", "tr", "bl", "br") │ │ │ │ │ + */ │ │ │ │ │ + positionBlocks: { │ │ │ │ │ + "tl": { │ │ │ │ │ + 'offset': new OpenLayers.Pixel(44, 0), │ │ │ │ │ + 'padding': new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ + 'blocks': [{ // top-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { //top-right │ │ │ │ │ + size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { //bottom-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ + }, { //bottom-right │ │ │ │ │ + size: new OpenLayers.Size(22, 18), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -632) │ │ │ │ │ + }, { // stem │ │ │ │ │ + size: new OpenLayers.Size(81, 35), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -688) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + "tr": { │ │ │ │ │ + 'offset': new OpenLayers.Pixel(-45, 0), │ │ │ │ │ + 'padding': new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ + 'blocks': [{ // top-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { //top-right │ │ │ │ │ + size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { //bottom-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ + }, { //bottom-right │ │ │ │ │ + size: new OpenLayers.Size(22, 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -631) │ │ │ │ │ + }, { // stem │ │ │ │ │ + size: new OpenLayers.Size(81, 35), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, null, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-215, -687) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + "bl": { │ │ │ │ │ + 'offset': new OpenLayers.Pixel(45, 0), │ │ │ │ │ + 'padding': new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ + 'blocks': [{ // top-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { //top-right │ │ │ │ │ + size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { //bottom-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ + }, { //bottom-right │ │ │ │ │ + size: new OpenLayers.Size(22, 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ + }, { // stem │ │ │ │ │ + size: new OpenLayers.Size(81, 33), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, null, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-101, -674) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + "br": { │ │ │ │ │ + 'offset': new OpenLayers.Pixel(-44, 0), │ │ │ │ │ + 'padding': new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ + 'blocks': [{ // top-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { //top-right │ │ │ │ │ + size: new OpenLayers.Size(22, 'auto'), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { //bottom-left │ │ │ │ │ + size: new OpenLayers.Size('auto', 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ + }, { //bottom-right │ │ │ │ │ + size: new OpenLayers.Size(22, 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ + }, { // stem │ │ │ │ │ + size: new OpenLayers.Size(81, 33), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, null, null, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-311, -674) │ │ │ │ │ + }] │ │ │ │ │ } │ │ │ │ │ - options.callback.call(options.scope, response); │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ + }, │ │ │ │ │ │ │ │ │ │ - /** │ │ │ │ │ - * Method: parseFeatures │ │ │ │ │ - * Read HTTP response body and return features │ │ │ │ │ - * │ │ │ │ │ - * Parameters: │ │ │ │ │ - * request - {XMLHttpRequest} The request object │ │ │ │ │ - * │ │ │ │ │ - * Returns: │ │ │ │ │ - * {Array({<OpenLayers.Feature.Vector>})} Array of features │ │ │ │ │ - */ │ │ │ │ │ - parseFeatures: function(request) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText; │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null; │ │ │ │ │ - } │ │ │ │ │ - return this.format.read(doc); │ │ │ │ │ - }, │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: minSize │ │ │ │ │ + * {<OpenLayers.Size>} │ │ │ │ │ + */ │ │ │ │ │ + minSize: new OpenLayers.Size(105, 10), │ │ │ │ │ │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.SOS.v1_0_0" │ │ │ │ │ -}); │ │ │ │ │ + /** │ │ │ │ │ + * APIProperty: maxSize │ │ │ │ │ + * {<OpenLayers.Size>} │ │ │ │ │ + */ │ │ │ │ │ + maxSize: new OpenLayers.Size(1200, 660), │ │ │ │ │ + │ │ │ │ │ + /** │ │ │ │ │ + * Constructor: OpenLayers.Popup.FramedCloud │ │ │ │ │ + * │ │ │ │ │ + * Parameters: │ │ │ │ │ + * id - {String} │ │ │ │ │ + * lonlat - {<OpenLayers.LonLat>} │ │ │ │ │ + * contentSize - {<OpenLayers.Size>} │ │ │ │ │ + * contentHTML - {String} │ │ │ │ │ + * anchor - {Object} Object to which we'll anchor the popup. Must expose │ │ │ │ │ + * a 'size' (<OpenLayers.Size>) and 'offset' (<OpenLayers.Pixel>) │ │ │ │ │ + * (Note that this is generally an <OpenLayers.Icon>). │ │ │ │ │ + * closeBox - {Boolean} │ │ │ │ │ + * closeBoxCallback - {Function} Function to be called on closeBox click. │ │ │ │ │ + */ │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, │ │ │ │ │ + closeBoxCallback) { │ │ │ │ │ + │ │ │ │ │ + this.imageSrc = OpenLayers.Util.getImageLocation('cloud-popup-relative.png'); │ │ │ │ │ + OpenLayers.Popup.Framed.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.contentDiv.className = this.contentDisplayClass; │ │ │ │ │ + }, │ │ │ │ │ + │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup.FramedCloud" │ │ │ │ │ + }); │ │ │ ├── ./usr/share/javascript/openlayers/OpenLayers.tests.min.js │ │ │ │ ├── js-beautify {} │ │ │ │ │ @@ -13318,356 +13318,14 @@ │ │ │ │ │ var corner = OpenLayers.Bounds.oppositeQuadrant(this.relativePosition); │ │ │ │ │ OpenLayers.Util.removeItem(corners, corner); │ │ │ │ │ return corners.join(" ") │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Popup.AnchoredBubble" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Popup.AnchoredBubble.CORNER_SIZE = 5; │ │ │ │ │ -OpenLayers.Kinetic = OpenLayers.Class({ │ │ │ │ │ - threshold: 0, │ │ │ │ │ - deceleration: .0035, │ │ │ │ │ - nbPoints: 100, │ │ │ │ │ - delay: 200, │ │ │ │ │ - points: undefined, │ │ │ │ │ - timerId: undefined, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options) │ │ │ │ │ - }, │ │ │ │ │ - begin: function() { │ │ │ │ │ - OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ - this.timerId = undefined; │ │ │ │ │ - this.points = [] │ │ │ │ │ - }, │ │ │ │ │ - update: function(xy) { │ │ │ │ │ - this.points.unshift({ │ │ │ │ │ - xy: xy, │ │ │ │ │ - tick: (new Date).getTime() │ │ │ │ │ - }); │ │ │ │ │ - if (this.points.length > this.nbPoints) { │ │ │ │ │ - this.points.pop() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - end: function(xy) { │ │ │ │ │ - var last, now = (new Date).getTime(); │ │ │ │ │ - for (var i = 0, l = this.points.length, point; i < l; i++) { │ │ │ │ │ - point = this.points[i]; │ │ │ │ │ - if (now - point.tick > this.delay) { │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - last = point │ │ │ │ │ - } │ │ │ │ │ - if (!last) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var time = (new Date).getTime() - last.tick; │ │ │ │ │ - var dist = Math.sqrt(Math.pow(xy.x - last.xy.x, 2) + Math.pow(xy.y - last.xy.y, 2)); │ │ │ │ │ - var speed = dist / time; │ │ │ │ │ - if (speed == 0 || speed < this.threshold) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var theta = Math.asin((xy.y - last.xy.y) / dist); │ │ │ │ │ - if (last.xy.x <= xy.x) { │ │ │ │ │ - theta = Math.PI - theta │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - speed: speed, │ │ │ │ │ - theta: theta │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - move: function(info, callback) { │ │ │ │ │ - var v0 = info.speed; │ │ │ │ │ - var fx = Math.cos(info.theta); │ │ │ │ │ - var fy = -Math.sin(info.theta); │ │ │ │ │ - var initialTime = (new Date).getTime(); │ │ │ │ │ - var lastX = 0; │ │ │ │ │ - var lastY = 0; │ │ │ │ │ - var timerCallback = function() { │ │ │ │ │ - if (this.timerId == null) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var t = (new Date).getTime() - initialTime; │ │ │ │ │ - var p = -this.deceleration * Math.pow(t, 2) / 2 + v0 * t; │ │ │ │ │ - var x = p * fx; │ │ │ │ │ - var y = p * fy; │ │ │ │ │ - var args = {}; │ │ │ │ │ - args.end = false; │ │ │ │ │ - var v = -this.deceleration * t + v0; │ │ │ │ │ - if (v <= 0) { │ │ │ │ │ - OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - args.end = true │ │ │ │ │ - } │ │ │ │ │ - args.x = x - lastX; │ │ │ │ │ - args.y = y - lastY; │ │ │ │ │ - lastX = x; │ │ │ │ │ - lastY = y; │ │ │ │ │ - callback(args.x, args.y, args.end) │ │ │ │ │ - }; │ │ │ │ │ - this.timerId = OpenLayers.Animation.start(OpenLayers.Function.bind(timerCallback, this)) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Kinetic" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Icon = OpenLayers.Class({ │ │ │ │ │ - url: null, │ │ │ │ │ - size: null, │ │ │ │ │ - offset: null, │ │ │ │ │ - calculateOffset: null, │ │ │ │ │ - imageDiv: null, │ │ │ │ │ - px: null, │ │ │ │ │ - initialize: function(url, size, offset, calculateOffset) { │ │ │ │ │ - this.url = url; │ │ │ │ │ - this.size = size || { │ │ │ │ │ - w: 20, │ │ │ │ │ - h: 20 │ │ │ │ │ - }; │ │ │ │ │ - this.offset = offset || { │ │ │ │ │ - x: -(this.size.w / 2), │ │ │ │ │ - y: -(this.size.h / 2) │ │ │ │ │ - }; │ │ │ │ │ - this.calculateOffset = calculateOffset; │ │ │ │ │ - var id = OpenLayers.Util.createUniqueID("OL_Icon_"); │ │ │ │ │ - this.imageDiv = OpenLayers.Util.createAlphaImageDiv(id) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.erase(); │ │ │ │ │ - OpenLayers.Event.stopObservingElement(this.imageDiv.firstChild); │ │ │ │ │ - this.imageDiv.innerHTML = ""; │ │ │ │ │ - this.imageDiv = null │ │ │ │ │ - }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - return new OpenLayers.Icon(this.url, this.size, this.offset, this.calculateOffset) │ │ │ │ │ - }, │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - if (size != null) { │ │ │ │ │ - this.size = size │ │ │ │ │ - } │ │ │ │ │ - this.draw() │ │ │ │ │ - }, │ │ │ │ │ - setUrl: function(url) { │ │ │ │ │ - if (url != null) { │ │ │ │ │ - this.url = url │ │ │ │ │ - } │ │ │ │ │ - this.draw() │ │ │ │ │ - }, │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, this.size, this.url, "absolute"); │ │ │ │ │ - this.moveTo(px); │ │ │ │ │ - return this.imageDiv │ │ │ │ │ - }, │ │ │ │ │ - erase: function() { │ │ │ │ │ - if (this.imageDiv != null && this.imageDiv.parentNode != null) { │ │ │ │ │ - OpenLayers.Element.remove(this.imageDiv) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, null, null, null, null, null, opacity) │ │ │ │ │ - }, │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - if (px != null) { │ │ │ │ │ - this.px = px │ │ │ │ │ - } │ │ │ │ │ - if (this.imageDiv != null) { │ │ │ │ │ - if (this.px == null) { │ │ │ │ │ - this.display(false) │ │ │ │ │ - } else { │ │ │ │ │ - if (this.calculateOffset) { │ │ │ │ │ - this.offset = this.calculateOffset(this.size) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, { │ │ │ │ │ - x: this.px.x + this.offset.x, │ │ │ │ │ - y: this.px.y + this.offset.y │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - display: function(display) { │ │ │ │ │ - this.imageDiv.style.display = display ? "" : "none" │ │ │ │ │ - }, │ │ │ │ │ - isDrawn: function() { │ │ │ │ │ - var isDrawn = this.imageDiv && this.imageDiv.parentNode && this.imageDiv.parentNode.nodeType != 11; │ │ │ │ │ - return isDrawn │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Icon" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Marker = OpenLayers.Class({ │ │ │ │ │ - icon: null, │ │ │ │ │ - lonlat: null, │ │ │ │ │ - events: null, │ │ │ │ │ - map: null, │ │ │ │ │ - initialize: function(lonlat, icon) { │ │ │ │ │ - this.lonlat = lonlat; │ │ │ │ │ - var newIcon = icon ? icon : OpenLayers.Marker.defaultIcon(); │ │ │ │ │ - if (this.icon == null) { │ │ │ │ │ - this.icon = newIcon │ │ │ │ │ - } else { │ │ │ │ │ - this.icon.url = newIcon.url; │ │ │ │ │ - this.icon.size = newIcon.size; │ │ │ │ │ - this.icon.offset = newIcon.offset; │ │ │ │ │ - this.icon.calculateOffset = newIcon.calculateOffset │ │ │ │ │ - } │ │ │ │ │ - this.events = new OpenLayers.Events(this, this.icon.imageDiv) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.erase(); │ │ │ │ │ - this.map = null; │ │ │ │ │ - this.events.destroy(); │ │ │ │ │ - this.events = null; │ │ │ │ │ - if (this.icon != null) { │ │ │ │ │ - this.icon.destroy(); │ │ │ │ │ - this.icon = null │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - return this.icon.draw(px) │ │ │ │ │ - }, │ │ │ │ │ - erase: function() { │ │ │ │ │ - if (this.icon != null) { │ │ │ │ │ - this.icon.erase() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - moveTo: function(px) { │ │ │ │ │ - if (px != null && this.icon != null) { │ │ │ │ │ - this.icon.moveTo(px) │ │ │ │ │ - } │ │ │ │ │ - this.lonlat = this.map.getLonLatFromLayerPx(px) │ │ │ │ │ - }, │ │ │ │ │ - isDrawn: function() { │ │ │ │ │ - var isDrawn = this.icon && this.icon.isDrawn(); │ │ │ │ │ - return isDrawn │ │ │ │ │ - }, │ │ │ │ │ - onScreen: function() { │ │ │ │ │ - var onScreen = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - var screenBounds = this.map.getExtent(); │ │ │ │ │ - onScreen = screenBounds.containsLonLat(this.lonlat) │ │ │ │ │ - } │ │ │ │ │ - return onScreen │ │ │ │ │ - }, │ │ │ │ │ - inflate: function(inflate) { │ │ │ │ │ - if (this.icon) { │ │ │ │ │ - this.icon.setSize({ │ │ │ │ │ - w: this.icon.size.w * inflate, │ │ │ │ │ - h: this.icon.size.h * inflate │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - this.icon.setOpacity(opacity) │ │ │ │ │ - }, │ │ │ │ │ - setUrl: function(url) { │ │ │ │ │ - this.icon.setUrl(url) │ │ │ │ │ - }, │ │ │ │ │ - display: function(display) { │ │ │ │ │ - this.icon.display(display) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Marker" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Marker.defaultIcon = function() { │ │ │ │ │ - return new OpenLayers.Icon(OpenLayers.Util.getImageLocation("marker.png"), { │ │ │ │ │ - w: 21, │ │ │ │ │ - h: 25 │ │ │ │ │ - }, { │ │ │ │ │ - x: -10.5, │ │ │ │ │ - y: -25 │ │ │ │ │ - }) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Handler = OpenLayers.Class({ │ │ │ │ │ - id: null, │ │ │ │ │ - control: null, │ │ │ │ │ - map: null, │ │ │ │ │ - keyMask: null, │ │ │ │ │ - active: false, │ │ │ │ │ - evt: null, │ │ │ │ │ - touch: false, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.control = control; │ │ │ │ │ - this.callbacks = callbacks; │ │ │ │ │ - var map = this.map || control.map; │ │ │ │ │ - if (map) { │ │ │ │ │ - this.setMap(map) │ │ │ │ │ - } │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.map = map │ │ │ │ │ - }, │ │ │ │ │ - checkModifiers: function(evt) { │ │ │ │ │ - if (this.keyMask == null) { │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - var keyModifiers = (evt.shiftKey ? OpenLayers.Handler.MOD_SHIFT : 0) | (evt.ctrlKey ? OpenLayers.Handler.MOD_CTRL : 0) | (evt.altKey ? OpenLayers.Handler.MOD_ALT : 0) | (evt.metaKey ? OpenLayers.Handler.MOD_META : 0); │ │ │ │ │ - return keyModifiers == this.keyMask │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.register(events[i], this[events[i]]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.active = true; │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.unregister(events[i], this[events[i]]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.touch = false; │ │ │ │ │ - this.active = false; │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - startTouch: function() { │ │ │ │ │ - if (!this.touch) { │ │ │ │ │ - this.touch = true; │ │ │ │ │ - var events = ["mousedown", "mouseup", "mousemove", "click", "dblclick", "mouseout"]; │ │ │ │ │ - for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ - if (this[events[i]]) { │ │ │ │ │ - this.unregister(events[i], this[events[i]]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - callback: function(name, args) { │ │ │ │ │ - if (name && this.callbacks[name]) { │ │ │ │ │ - this.callbacks[name].apply(this.control, args) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - register: function(name, method) { │ │ │ │ │ - this.map.events.registerPriority(name, this, method); │ │ │ │ │ - this.map.events.registerPriority(name, this, this.setEvent) │ │ │ │ │ - }, │ │ │ │ │ - unregister: function(name, method) { │ │ │ │ │ - this.map.events.unregister(name, this, method); │ │ │ │ │ - this.map.events.unregister(name, this, this.setEvent) │ │ │ │ │ - }, │ │ │ │ │ - setEvent: function(evt) { │ │ │ │ │ - this.evt = evt; │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.control = this.map = null │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.MOD_NONE = 0; │ │ │ │ │ -OpenLayers.Handler.MOD_SHIFT = 1; │ │ │ │ │ -OpenLayers.Handler.MOD_CTRL = 2; │ │ │ │ │ -OpenLayers.Handler.MOD_ALT = 4; │ │ │ │ │ -OpenLayers.Handler.MOD_META = 8; │ │ │ │ │ OpenLayers.Filter = OpenLayers.Class({ │ │ │ │ │ initialize: function(options) { │ │ │ │ │ OpenLayers.Util.extend(this, options) │ │ │ │ │ }, │ │ │ │ │ destroy: function() {}, │ │ │ │ │ evaluate: function(context) { │ │ │ │ │ return true │ │ │ │ │ @@ -13682,354 +13340,14 @@ │ │ │ │ │ } else { │ │ │ │ │ string = Object.prototype.toString.call(this) │ │ │ │ │ } │ │ │ │ │ return string │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Filter" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.TileManager = OpenLayers.Class({ │ │ │ │ │ - cacheSize: 256, │ │ │ │ │ - tilesPerFrame: 2, │ │ │ │ │ - frameDelay: 16, │ │ │ │ │ - moveDelay: 100, │ │ │ │ │ - zoomDelay: 200, │ │ │ │ │ - maps: null, │ │ │ │ │ - tileQueueId: null, │ │ │ │ │ - tileQueue: null, │ │ │ │ │ - tileCache: null, │ │ │ │ │ - tileCacheIndex: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - this.maps = []; │ │ │ │ │ - this.tileQueueId = {}; │ │ │ │ │ - this.tileQueue = {}; │ │ │ │ │ - this.tileCache = {}; │ │ │ │ │ - this.tileCacheIndex = [] │ │ │ │ │ - }, │ │ │ │ │ - addMap: function(map) { │ │ │ │ │ - if (this._destroyed || !OpenLayers.Layer.Grid) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - this.maps.push(map); │ │ │ │ │ - this.tileQueue[map.id] = []; │ │ │ │ │ - for (var i = 0, ii = map.layers.length; i < ii; ++i) { │ │ │ │ │ - this.addLayer({ │ │ │ │ │ - layer: map.layers[i] │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - map.events.on({ │ │ │ │ │ - move: this.move, │ │ │ │ │ - zoomend: this.zoomEnd, │ │ │ │ │ - changelayer: this.changeLayer, │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - preremovelayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - if (this._destroyed || !OpenLayers.Layer.Grid) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - window.clearTimeout(this.tileQueueId[map.id]); │ │ │ │ │ - if (map.layers) { │ │ │ │ │ - for (var i = 0, ii = map.layers.length; i < ii; ++i) { │ │ │ │ │ - this.removeLayer({ │ │ │ │ │ - layer: map.layers[i] │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (map.events) { │ │ │ │ │ - map.events.un({ │ │ │ │ │ - move: this.move, │ │ │ │ │ - zoomend: this.zoomEnd, │ │ │ │ │ - changelayer: this.changeLayer, │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - preremovelayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - delete this.tileQueue[map.id]; │ │ │ │ │ - delete this.tileQueueId[map.id]; │ │ │ │ │ - OpenLayers.Util.removeItem(this.maps, map) │ │ │ │ │ - }, │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - this.updateTimeout(evt.object, this.moveDelay, true) │ │ │ │ │ - }, │ │ │ │ │ - zoomEnd: function(evt) { │ │ │ │ │ - this.updateTimeout(evt.object, this.zoomDelay) │ │ │ │ │ - }, │ │ │ │ │ - changeLayer: function(evt) { │ │ │ │ │ - if (evt.property === "visibility" || evt.property === "params") { │ │ │ │ │ - this.updateTimeout(evt.object, 0) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - addLayer: function(evt) { │ │ │ │ │ - var layer = evt.layer; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ - layer.events.on({ │ │ │ │ │ - addtile: this.addTile, │ │ │ │ │ - retile: this.clearTileQueue, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - var i, j, tile; │ │ │ │ │ - for (i = layer.grid.length - 1; i >= 0; --i) { │ │ │ │ │ - for (j = layer.grid[i].length - 1; j >= 0; --j) { │ │ │ │ │ - tile = layer.grid[i][j]; │ │ │ │ │ - this.addTile({ │ │ │ │ │ - tile: tile │ │ │ │ │ - }); │ │ │ │ │ - if (tile.url && !tile.imgDiv) { │ │ │ │ │ - this.manageTileCache({ │ │ │ │ │ - object: tile │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - removeLayer: function(evt) { │ │ │ │ │ - var layer = evt.layer; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ - this.clearTileQueue({ │ │ │ │ │ - object: layer │ │ │ │ │ - }); │ │ │ │ │ - if (layer.events) { │ │ │ │ │ - layer.events.un({ │ │ │ │ │ - addtile: this.addTile, │ │ │ │ │ - retile: this.clearTileQueue, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - if (layer.grid) { │ │ │ │ │ - var i, j, tile; │ │ │ │ │ - for (i = layer.grid.length - 1; i >= 0; --i) { │ │ │ │ │ - for (j = layer.grid[i].length - 1; j >= 0; --j) { │ │ │ │ │ - tile = layer.grid[i][j]; │ │ │ │ │ - this.unloadTile({ │ │ │ │ │ - object: tile │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - updateTimeout: function(map, delay, nice) { │ │ │ │ │ - window.clearTimeout(this.tileQueueId[map.id]); │ │ │ │ │ - var tileQueue = this.tileQueue[map.id]; │ │ │ │ │ - if (!nice || tileQueue.length) { │ │ │ │ │ - this.tileQueueId[map.id] = window.setTimeout(OpenLayers.Function.bind(function() { │ │ │ │ │ - this.drawTilesFromQueue(map); │ │ │ │ │ - if (tileQueue.length) { │ │ │ │ │ - this.updateTimeout(map, this.frameDelay) │ │ │ │ │ - } │ │ │ │ │ - }, this), delay) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - addTile: function(evt) { │ │ │ │ │ - if (evt.tile instanceof OpenLayers.Tile.Image) { │ │ │ │ │ - evt.tile.events.on({ │ │ │ │ │ - beforedraw: this.queueTileDraw, │ │ │ │ │ - beforeload: this.manageTileCache, │ │ │ │ │ - loadend: this.addToCache, │ │ │ │ │ - unload: this.unloadTile, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - this.removeLayer({ │ │ │ │ │ - layer: evt.tile.layer │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - unloadTile: function(evt) { │ │ │ │ │ - var tile = evt.object; │ │ │ │ │ - tile.events.un({ │ │ │ │ │ - beforedraw: this.queueTileDraw, │ │ │ │ │ - beforeload: this.manageTileCache, │ │ │ │ │ - loadend: this.addToCache, │ │ │ │ │ - unload: this.unloadTile, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Util.removeItem(this.tileQueue[tile.layer.map.id], tile) │ │ │ │ │ - }, │ │ │ │ │ - queueTileDraw: function(evt) { │ │ │ │ │ - var tile = evt.object; │ │ │ │ │ - var queued = false; │ │ │ │ │ - var layer = tile.layer; │ │ │ │ │ - var url = layer.getURL(tile.bounds); │ │ │ │ │ - var img = this.tileCache[url]; │ │ │ │ │ - if (img && img.className !== "olTileImage") { │ │ │ │ │ - delete this.tileCache[url]; │ │ │ │ │ - OpenLayers.Util.removeItem(this.tileCacheIndex, url); │ │ │ │ │ - img = null │ │ │ │ │ - } │ │ │ │ │ - if (layer.url && (layer.async || !img)) { │ │ │ │ │ - var tileQueue = this.tileQueue[layer.map.id]; │ │ │ │ │ - if (!~OpenLayers.Util.indexOf(tileQueue, tile)) { │ │ │ │ │ - tileQueue.push(tile) │ │ │ │ │ - } │ │ │ │ │ - queued = true │ │ │ │ │ - } │ │ │ │ │ - return !queued │ │ │ │ │ - }, │ │ │ │ │ - drawTilesFromQueue: function(map) { │ │ │ │ │ - var tileQueue = this.tileQueue[map.id]; │ │ │ │ │ - var limit = this.tilesPerFrame; │ │ │ │ │ - var animating = map.zoomTween && map.zoomTween.playing; │ │ │ │ │ - while (!animating && tileQueue.length && limit) { │ │ │ │ │ - tileQueue.shift().draw(true); │ │ │ │ │ - --limit │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - manageTileCache: function(evt) { │ │ │ │ │ - var tile = evt.object; │ │ │ │ │ - var img = this.tileCache[tile.url]; │ │ │ │ │ - if (img) { │ │ │ │ │ - if (img.parentNode && OpenLayers.Element.hasClass(img.parentNode, "olBackBuffer")) { │ │ │ │ │ - img.parentNode.removeChild(img); │ │ │ │ │ - img.id = null │ │ │ │ │ - } │ │ │ │ │ - if (!img.parentNode) { │ │ │ │ │ - img.style.visibility = "hidden"; │ │ │ │ │ - img.style.opacity = 0; │ │ │ │ │ - tile.setImage(img); │ │ │ │ │ - OpenLayers.Util.removeItem(this.tileCacheIndex, tile.url); │ │ │ │ │ - this.tileCacheIndex.push(tile.url) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - addToCache: function(evt) { │ │ │ │ │ - var tile = evt.object; │ │ │ │ │ - if (!this.tileCache[tile.url]) { │ │ │ │ │ - if (!OpenLayers.Element.hasClass(tile.imgDiv, "olImageLoadError")) { │ │ │ │ │ - if (this.tileCacheIndex.length >= this.cacheSize) { │ │ │ │ │ - delete this.tileCache[this.tileCacheIndex[0]]; │ │ │ │ │ - this.tileCacheIndex.shift() │ │ │ │ │ - } │ │ │ │ │ - this.tileCache[tile.url] = tile.imgDiv; │ │ │ │ │ - this.tileCacheIndex.push(tile.url) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - clearTileQueue: function(evt) { │ │ │ │ │ - var layer = evt.object; │ │ │ │ │ - var tileQueue = this.tileQueue[layer.map.id]; │ │ │ │ │ - for (var i = tileQueue.length - 1; i >= 0; --i) { │ │ │ │ │ - if (tileQueue[i].layer === layer) { │ │ │ │ │ - tileQueue.splice(i, 1) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i = this.maps.length - 1; i >= 0; --i) { │ │ │ │ │ - this.removeMap(this.maps[i]) │ │ │ │ │ - } │ │ │ │ │ - this.maps = null; │ │ │ │ │ - this.tileQueue = null; │ │ │ │ │ - this.tileQueueId = null; │ │ │ │ │ - this.tileCache = null; │ │ │ │ │ - this.tileCacheIndex = null; │ │ │ │ │ - this._destroyed = true │ │ │ │ │ - } │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Symbolizer = OpenLayers.Class({ │ │ │ │ │ - zIndex: 0, │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ - OpenLayers.Util.extend(this, config) │ │ │ │ │ - }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - var Type = eval(this.CLASS_NAME); │ │ │ │ │ - return new Type(OpenLayers.Util.extend({}, this)) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Symbolizer" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Rule = OpenLayers.Class({ │ │ │ │ │ - id: null, │ │ │ │ │ - name: null, │ │ │ │ │ - title: null, │ │ │ │ │ - description: null, │ │ │ │ │ - context: null, │ │ │ │ │ - filter: null, │ │ │ │ │ - elseFilter: false, │ │ │ │ │ - symbolizer: null, │ │ │ │ │ - symbolizers: null, │ │ │ │ │ - minScaleDenominator: null, │ │ │ │ │ - maxScaleDenominator: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.symbolizer = {}; │ │ │ │ │ - OpenLayers.Util.extend(this, options); │ │ │ │ │ - if (this.symbolizers) { │ │ │ │ │ - delete this.symbolizer │ │ │ │ │ - } │ │ │ │ │ - this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i in this.symbolizer) { │ │ │ │ │ - this.symbolizer[i] = null │ │ │ │ │ - } │ │ │ │ │ - this.symbolizer = null; │ │ │ │ │ - delete this.symbolizers │ │ │ │ │ - }, │ │ │ │ │ - evaluate: function(feature) { │ │ │ │ │ - var context = this.getContext(feature); │ │ │ │ │ - var applies = true; │ │ │ │ │ - if (this.minScaleDenominator || this.maxScaleDenominator) { │ │ │ │ │ - var scale = feature.layer.map.getScale() │ │ │ │ │ - } │ │ │ │ │ - if (this.minScaleDenominator) { │ │ │ │ │ - applies = scale >= OpenLayers.Style.createLiteral(this.minScaleDenominator, context) │ │ │ │ │ - } │ │ │ │ │ - if (applies && this.maxScaleDenominator) { │ │ │ │ │ - applies = scale < OpenLayers.Style.createLiteral(this.maxScaleDenominator, context) │ │ │ │ │ - } │ │ │ │ │ - if (applies && this.filter) { │ │ │ │ │ - if (this.filter.CLASS_NAME == "OpenLayers.Filter.FeatureId") { │ │ │ │ │ - applies = this.filter.evaluate(feature) │ │ │ │ │ - } else { │ │ │ │ │ - applies = this.filter.evaluate(context) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return applies │ │ │ │ │ - }, │ │ │ │ │ - getContext: function(feature) { │ │ │ │ │ - var context = this.context; │ │ │ │ │ - if (!context) { │ │ │ │ │ - context = feature.attributes || feature.data │ │ │ │ │ - } │ │ │ │ │ - if (typeof this.context == "function") { │ │ │ │ │ - context = this.context(feature) │ │ │ │ │ - } │ │ │ │ │ - return context │ │ │ │ │ - }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - var options = OpenLayers.Util.extend({}, this); │ │ │ │ │ - if (this.symbolizers) { │ │ │ │ │ - var len = this.symbolizers.length; │ │ │ │ │ - options.symbolizers = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - options.symbolizers[i] = this.symbolizers[i].clone() │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - options.symbolizer = {}; │ │ │ │ │ - var value, type; │ │ │ │ │ - for (var key in this.symbolizer) { │ │ │ │ │ - value = this.symbolizer[key]; │ │ │ │ │ - type = typeof value; │ │ │ │ │ - if (type === "object") { │ │ │ │ │ - options.symbolizer[key] = OpenLayers.Util.extend({}, value) │ │ │ │ │ - } else if (type === "string") { │ │ │ │ │ - options.symbolizer[key] = value │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - options.filter = this.filter && this.filter.clone(); │ │ │ │ │ - options.context = this.context && OpenLayers.Util.extend({}, this.context); │ │ │ │ │ - return new OpenLayers.Rule(options) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Rule" │ │ │ │ │ -}); │ │ │ │ │ OpenLayers.Format.GeoJSON = OpenLayers.Class(OpenLayers.Format.JSON, { │ │ │ │ │ ignoreExtraDims: false, │ │ │ │ │ read: function(json, type, filter) { │ │ │ │ │ type = type ? type : "FeatureCollection"; │ │ │ │ │ var results = null; │ │ │ │ │ var obj = null; │ │ │ │ │ if (typeof json == "string") { │ │ │ │ │ @@ -17860,14 +17178,178 @@ │ │ │ │ │ process: null, │ │ │ │ │ output: null, │ │ │ │ │ initialize: function(options) { │ │ │ │ │ OpenLayers.Util.extend(this, options) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.WPSProcess.ChainLink" │ │ │ │ │ }); │ │ │ │ │ +OpenLayers.Icon = OpenLayers.Class({ │ │ │ │ │ + url: null, │ │ │ │ │ + size: null, │ │ │ │ │ + offset: null, │ │ │ │ │ + calculateOffset: null, │ │ │ │ │ + imageDiv: null, │ │ │ │ │ + px: null, │ │ │ │ │ + initialize: function(url, size, offset, calculateOffset) { │ │ │ │ │ + this.url = url; │ │ │ │ │ + this.size = size || { │ │ │ │ │ + w: 20, │ │ │ │ │ + h: 20 │ │ │ │ │ + }; │ │ │ │ │ + this.offset = offset || { │ │ │ │ │ + x: -(this.size.w / 2), │ │ │ │ │ + y: -(this.size.h / 2) │ │ │ │ │ + }; │ │ │ │ │ + this.calculateOffset = calculateOffset; │ │ │ │ │ + var id = OpenLayers.Util.createUniqueID("OL_Icon_"); │ │ │ │ │ + this.imageDiv = OpenLayers.Util.createAlphaImageDiv(id) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.erase(); │ │ │ │ │ + OpenLayers.Event.stopObservingElement(this.imageDiv.firstChild); │ │ │ │ │ + this.imageDiv.innerHTML = ""; │ │ │ │ │ + this.imageDiv = null │ │ │ │ │ + }, │ │ │ │ │ + clone: function() { │ │ │ │ │ + return new OpenLayers.Icon(this.url, this.size, this.offset, this.calculateOffset) │ │ │ │ │ + }, │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + if (size != null) { │ │ │ │ │ + this.size = size │ │ │ │ │ + } │ │ │ │ │ + this.draw() │ │ │ │ │ + }, │ │ │ │ │ + setUrl: function(url) { │ │ │ │ │ + if (url != null) { │ │ │ │ │ + this.url = url │ │ │ │ │ + } │ │ │ │ │ + this.draw() │ │ │ │ │ + }, │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, this.size, this.url, "absolute"); │ │ │ │ │ + this.moveTo(px); │ │ │ │ │ + return this.imageDiv │ │ │ │ │ + }, │ │ │ │ │ + erase: function() { │ │ │ │ │ + if (this.imageDiv != null && this.imageDiv.parentNode != null) { │ │ │ │ │ + OpenLayers.Element.remove(this.imageDiv) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, null, null, null, null, null, null, opacity) │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + if (px != null) { │ │ │ │ │ + this.px = px │ │ │ │ │ + } │ │ │ │ │ + if (this.imageDiv != null) { │ │ │ │ │ + if (this.px == null) { │ │ │ │ │ + this.display(false) │ │ │ │ │ + } else { │ │ │ │ │ + if (this.calculateOffset) { │ │ │ │ │ + this.offset = this.calculateOffset(this.size) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.modifyAlphaImageDiv(this.imageDiv, null, { │ │ │ │ │ + x: this.px.x + this.offset.x, │ │ │ │ │ + y: this.px.y + this.offset.y │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + display: function(display) { │ │ │ │ │ + this.imageDiv.style.display = display ? "" : "none" │ │ │ │ │ + }, │ │ │ │ │ + isDrawn: function() { │ │ │ │ │ + var isDrawn = this.imageDiv && this.imageDiv.parentNode && this.imageDiv.parentNode.nodeType != 11; │ │ │ │ │ + return isDrawn │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Icon" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Marker = OpenLayers.Class({ │ │ │ │ │ + icon: null, │ │ │ │ │ + lonlat: null, │ │ │ │ │ + events: null, │ │ │ │ │ + map: null, │ │ │ │ │ + initialize: function(lonlat, icon) { │ │ │ │ │ + this.lonlat = lonlat; │ │ │ │ │ + var newIcon = icon ? icon : OpenLayers.Marker.defaultIcon(); │ │ │ │ │ + if (this.icon == null) { │ │ │ │ │ + this.icon = newIcon │ │ │ │ │ + } else { │ │ │ │ │ + this.icon.url = newIcon.url; │ │ │ │ │ + this.icon.size = newIcon.size; │ │ │ │ │ + this.icon.offset = newIcon.offset; │ │ │ │ │ + this.icon.calculateOffset = newIcon.calculateOffset │ │ │ │ │ + } │ │ │ │ │ + this.events = new OpenLayers.Events(this, this.icon.imageDiv) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.erase(); │ │ │ │ │ + this.map = null; │ │ │ │ │ + this.events.destroy(); │ │ │ │ │ + this.events = null; │ │ │ │ │ + if (this.icon != null) { │ │ │ │ │ + this.icon.destroy(); │ │ │ │ │ + this.icon = null │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + return this.icon.draw(px) │ │ │ │ │ + }, │ │ │ │ │ + erase: function() { │ │ │ │ │ + if (this.icon != null) { │ │ │ │ │ + this.icon.erase() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(px) { │ │ │ │ │ + if (px != null && this.icon != null) { │ │ │ │ │ + this.icon.moveTo(px) │ │ │ │ │ + } │ │ │ │ │ + this.lonlat = this.map.getLonLatFromLayerPx(px) │ │ │ │ │ + }, │ │ │ │ │ + isDrawn: function() { │ │ │ │ │ + var isDrawn = this.icon && this.icon.isDrawn(); │ │ │ │ │ + return isDrawn │ │ │ │ │ + }, │ │ │ │ │ + onScreen: function() { │ │ │ │ │ + var onScreen = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + var screenBounds = this.map.getExtent(); │ │ │ │ │ + onScreen = screenBounds.containsLonLat(this.lonlat) │ │ │ │ │ + } │ │ │ │ │ + return onScreen │ │ │ │ │ + }, │ │ │ │ │ + inflate: function(inflate) { │ │ │ │ │ + if (this.icon) { │ │ │ │ │ + this.icon.setSize({ │ │ │ │ │ + w: this.icon.size.w * inflate, │ │ │ │ │ + h: this.icon.size.h * inflate │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + this.icon.setOpacity(opacity) │ │ │ │ │ + }, │ │ │ │ │ + setUrl: function(url) { │ │ │ │ │ + this.icon.setUrl(url) │ │ │ │ │ + }, │ │ │ │ │ + display: function(display) { │ │ │ │ │ + this.icon.display(display) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Marker" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Marker.defaultIcon = function() { │ │ │ │ │ + return new OpenLayers.Icon(OpenLayers.Util.getImageLocation("marker.png"), { │ │ │ │ │ + w: 21, │ │ │ │ │ + h: 25 │ │ │ │ │ + }, { │ │ │ │ │ + x: -10.5, │ │ │ │ │ + y: -25 │ │ │ │ │ + }) │ │ │ │ │ +}; │ │ │ │ │ OpenLayers.Strategy = OpenLayers.Class({ │ │ │ │ │ layer: null, │ │ │ │ │ options: null, │ │ │ │ │ active: null, │ │ │ │ │ autoActivate: true, │ │ │ │ │ autoDestroy: true, │ │ │ │ │ initialize: function(options) { │ │ │ │ │ @@ -17895,14 +17377,427 @@ │ │ │ │ │ this.active = false; │ │ │ │ │ return true │ │ │ │ │ } │ │ │ │ │ return false │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Strategy" │ │ │ │ │ }); │ │ │ │ │ +OpenLayers.Kinetic = OpenLayers.Class({ │ │ │ │ │ + threshold: 0, │ │ │ │ │ + deceleration: .0035, │ │ │ │ │ + nbPoints: 100, │ │ │ │ │ + delay: 200, │ │ │ │ │ + points: undefined, │ │ │ │ │ + timerId: undefined, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options) │ │ │ │ │ + }, │ │ │ │ │ + begin: function() { │ │ │ │ │ + OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ + this.timerId = undefined; │ │ │ │ │ + this.points = [] │ │ │ │ │ + }, │ │ │ │ │ + update: function(xy) { │ │ │ │ │ + this.points.unshift({ │ │ │ │ │ + xy: xy, │ │ │ │ │ + tick: (new Date).getTime() │ │ │ │ │ + }); │ │ │ │ │ + if (this.points.length > this.nbPoints) { │ │ │ │ │ + this.points.pop() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + end: function(xy) { │ │ │ │ │ + var last, now = (new Date).getTime(); │ │ │ │ │ + for (var i = 0, l = this.points.length, point; i < l; i++) { │ │ │ │ │ + point = this.points[i]; │ │ │ │ │ + if (now - point.tick > this.delay) { │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + last = point │ │ │ │ │ + } │ │ │ │ │ + if (!last) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var time = (new Date).getTime() - last.tick; │ │ │ │ │ + var dist = Math.sqrt(Math.pow(xy.x - last.xy.x, 2) + Math.pow(xy.y - last.xy.y, 2)); │ │ │ │ │ + var speed = dist / time; │ │ │ │ │ + if (speed == 0 || speed < this.threshold) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var theta = Math.asin((xy.y - last.xy.y) / dist); │ │ │ │ │ + if (last.xy.x <= xy.x) { │ │ │ │ │ + theta = Math.PI - theta │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + speed: speed, │ │ │ │ │ + theta: theta │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + move: function(info, callback) { │ │ │ │ │ + var v0 = info.speed; │ │ │ │ │ + var fx = Math.cos(info.theta); │ │ │ │ │ + var fy = -Math.sin(info.theta); │ │ │ │ │ + var initialTime = (new Date).getTime(); │ │ │ │ │ + var lastX = 0; │ │ │ │ │ + var lastY = 0; │ │ │ │ │ + var timerCallback = function() { │ │ │ │ │ + if (this.timerId == null) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var t = (new Date).getTime() - initialTime; │ │ │ │ │ + var p = -this.deceleration * Math.pow(t, 2) / 2 + v0 * t; │ │ │ │ │ + var x = p * fx; │ │ │ │ │ + var y = p * fy; │ │ │ │ │ + var args = {}; │ │ │ │ │ + args.end = false; │ │ │ │ │ + var v = -this.deceleration * t + v0; │ │ │ │ │ + if (v <= 0) { │ │ │ │ │ + OpenLayers.Animation.stop(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + args.end = true │ │ │ │ │ + } │ │ │ │ │ + args.x = x - lastX; │ │ │ │ │ + args.y = y - lastY; │ │ │ │ │ + lastX = x; │ │ │ │ │ + lastY = y; │ │ │ │ │ + callback(args.x, args.y, args.end) │ │ │ │ │ + }; │ │ │ │ │ + this.timerId = OpenLayers.Animation.start(OpenLayers.Function.bind(timerCallback, this)) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Kinetic" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.TileManager = OpenLayers.Class({ │ │ │ │ │ + cacheSize: 256, │ │ │ │ │ + tilesPerFrame: 2, │ │ │ │ │ + frameDelay: 16, │ │ │ │ │ + moveDelay: 100, │ │ │ │ │ + zoomDelay: 200, │ │ │ │ │ + maps: null, │ │ │ │ │ + tileQueueId: null, │ │ │ │ │ + tileQueue: null, │ │ │ │ │ + tileCache: null, │ │ │ │ │ + tileCacheIndex: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.maps = []; │ │ │ │ │ + this.tileQueueId = {}; │ │ │ │ │ + this.tileQueue = {}; │ │ │ │ │ + this.tileCache = {}; │ │ │ │ │ + this.tileCacheIndex = [] │ │ │ │ │ + }, │ │ │ │ │ + addMap: function(map) { │ │ │ │ │ + if (this._destroyed || !OpenLayers.Layer.Grid) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + this.maps.push(map); │ │ │ │ │ + this.tileQueue[map.id] = []; │ │ │ │ │ + for (var i = 0, ii = map.layers.length; i < ii; ++i) { │ │ │ │ │ + this.addLayer({ │ │ │ │ │ + layer: map.layers[i] │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + map.events.on({ │ │ │ │ │ + move: this.move, │ │ │ │ │ + zoomend: this.zoomEnd, │ │ │ │ │ + changelayer: this.changeLayer, │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + preremovelayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + if (this._destroyed || !OpenLayers.Layer.Grid) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + window.clearTimeout(this.tileQueueId[map.id]); │ │ │ │ │ + if (map.layers) { │ │ │ │ │ + for (var i = 0, ii = map.layers.length; i < ii; ++i) { │ │ │ │ │ + this.removeLayer({ │ │ │ │ │ + layer: map.layers[i] │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (map.events) { │ │ │ │ │ + map.events.un({ │ │ │ │ │ + move: this.move, │ │ │ │ │ + zoomend: this.zoomEnd, │ │ │ │ │ + changelayer: this.changeLayer, │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + preremovelayer: this.removeLayer, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + delete this.tileQueue[map.id]; │ │ │ │ │ + delete this.tileQueueId[map.id]; │ │ │ │ │ + OpenLayers.Util.removeItem(this.maps, map) │ │ │ │ │ + }, │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + this.updateTimeout(evt.object, this.moveDelay, true) │ │ │ │ │ + }, │ │ │ │ │ + zoomEnd: function(evt) { │ │ │ │ │ + this.updateTimeout(evt.object, this.zoomDelay) │ │ │ │ │ + }, │ │ │ │ │ + changeLayer: function(evt) { │ │ │ │ │ + if (evt.property === "visibility" || evt.property === "params") { │ │ │ │ │ + this.updateTimeout(evt.object, 0) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addLayer: function(evt) { │ │ │ │ │ + var layer = evt.layer; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ + layer.events.on({ │ │ │ │ │ + addtile: this.addTile, │ │ │ │ │ + retile: this.clearTileQueue, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + var i, j, tile; │ │ │ │ │ + for (i = layer.grid.length - 1; i >= 0; --i) { │ │ │ │ │ + for (j = layer.grid[i].length - 1; j >= 0; --j) { │ │ │ │ │ + tile = layer.grid[i][j]; │ │ │ │ │ + this.addTile({ │ │ │ │ │ + tile: tile │ │ │ │ │ + }); │ │ │ │ │ + if (tile.url && !tile.imgDiv) { │ │ │ │ │ + this.manageTileCache({ │ │ │ │ │ + object: tile │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + removeLayer: function(evt) { │ │ │ │ │ + var layer = evt.layer; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Grid) { │ │ │ │ │ + this.clearTileQueue({ │ │ │ │ │ + object: layer │ │ │ │ │ + }); │ │ │ │ │ + if (layer.events) { │ │ │ │ │ + layer.events.un({ │ │ │ │ │ + addtile: this.addTile, │ │ │ │ │ + retile: this.clearTileQueue, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + if (layer.grid) { │ │ │ │ │ + var i, j, tile; │ │ │ │ │ + for (i = layer.grid.length - 1; i >= 0; --i) { │ │ │ │ │ + for (j = layer.grid[i].length - 1; j >= 0; --j) { │ │ │ │ │ + tile = layer.grid[i][j]; │ │ │ │ │ + this.unloadTile({ │ │ │ │ │ + object: tile │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + updateTimeout: function(map, delay, nice) { │ │ │ │ │ + window.clearTimeout(this.tileQueueId[map.id]); │ │ │ │ │ + var tileQueue = this.tileQueue[map.id]; │ │ │ │ │ + if (!nice || tileQueue.length) { │ │ │ │ │ + this.tileQueueId[map.id] = window.setTimeout(OpenLayers.Function.bind(function() { │ │ │ │ │ + this.drawTilesFromQueue(map); │ │ │ │ │ + if (tileQueue.length) { │ │ │ │ │ + this.updateTimeout(map, this.frameDelay) │ │ │ │ │ + } │ │ │ │ │ + }, this), delay) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addTile: function(evt) { │ │ │ │ │ + if (evt.tile instanceof OpenLayers.Tile.Image) { │ │ │ │ │ + evt.tile.events.on({ │ │ │ │ │ + beforedraw: this.queueTileDraw, │ │ │ │ │ + beforeload: this.manageTileCache, │ │ │ │ │ + loadend: this.addToCache, │ │ │ │ │ + unload: this.unloadTile, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + this.removeLayer({ │ │ │ │ │ + layer: evt.tile.layer │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + unloadTile: function(evt) { │ │ │ │ │ + var tile = evt.object; │ │ │ │ │ + tile.events.un({ │ │ │ │ │ + beforedraw: this.queueTileDraw, │ │ │ │ │ + beforeload: this.manageTileCache, │ │ │ │ │ + loadend: this.addToCache, │ │ │ │ │ + unload: this.unloadTile, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Util.removeItem(this.tileQueue[tile.layer.map.id], tile) │ │ │ │ │ + }, │ │ │ │ │ + queueTileDraw: function(evt) { │ │ │ │ │ + var tile = evt.object; │ │ │ │ │ + var queued = false; │ │ │ │ │ + var layer = tile.layer; │ │ │ │ │ + var url = layer.getURL(tile.bounds); │ │ │ │ │ + var img = this.tileCache[url]; │ │ │ │ │ + if (img && img.className !== "olTileImage") { │ │ │ │ │ + delete this.tileCache[url]; │ │ │ │ │ + OpenLayers.Util.removeItem(this.tileCacheIndex, url); │ │ │ │ │ + img = null │ │ │ │ │ + } │ │ │ │ │ + if (layer.url && (layer.async || !img)) { │ │ │ │ │ + var tileQueue = this.tileQueue[layer.map.id]; │ │ │ │ │ + if (!~OpenLayers.Util.indexOf(tileQueue, tile)) { │ │ │ │ │ + tileQueue.push(tile) │ │ │ │ │ + } │ │ │ │ │ + queued = true │ │ │ │ │ + } │ │ │ │ │ + return !queued │ │ │ │ │ + }, │ │ │ │ │ + drawTilesFromQueue: function(map) { │ │ │ │ │ + var tileQueue = this.tileQueue[map.id]; │ │ │ │ │ + var limit = this.tilesPerFrame; │ │ │ │ │ + var animating = map.zoomTween && map.zoomTween.playing; │ │ │ │ │ + while (!animating && tileQueue.length && limit) { │ │ │ │ │ + tileQueue.shift().draw(true); │ │ │ │ │ + --limit │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + manageTileCache: function(evt) { │ │ │ │ │ + var tile = evt.object; │ │ │ │ │ + var img = this.tileCache[tile.url]; │ │ │ │ │ + if (img) { │ │ │ │ │ + if (img.parentNode && OpenLayers.Element.hasClass(img.parentNode, "olBackBuffer")) { │ │ │ │ │ + img.parentNode.removeChild(img); │ │ │ │ │ + img.id = null │ │ │ │ │ + } │ │ │ │ │ + if (!img.parentNode) { │ │ │ │ │ + img.style.visibility = "hidden"; │ │ │ │ │ + img.style.opacity = 0; │ │ │ │ │ + tile.setImage(img); │ │ │ │ │ + OpenLayers.Util.removeItem(this.tileCacheIndex, tile.url); │ │ │ │ │ + this.tileCacheIndex.push(tile.url) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addToCache: function(evt) { │ │ │ │ │ + var tile = evt.object; │ │ │ │ │ + if (!this.tileCache[tile.url]) { │ │ │ │ │ + if (!OpenLayers.Element.hasClass(tile.imgDiv, "olImageLoadError")) { │ │ │ │ │ + if (this.tileCacheIndex.length >= this.cacheSize) { │ │ │ │ │ + delete this.tileCache[this.tileCacheIndex[0]]; │ │ │ │ │ + this.tileCacheIndex.shift() │ │ │ │ │ + } │ │ │ │ │ + this.tileCache[tile.url] = tile.imgDiv; │ │ │ │ │ + this.tileCacheIndex.push(tile.url) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + clearTileQueue: function(evt) { │ │ │ │ │ + var layer = evt.object; │ │ │ │ │ + var tileQueue = this.tileQueue[layer.map.id]; │ │ │ │ │ + for (var i = tileQueue.length - 1; i >= 0; --i) { │ │ │ │ │ + if (tileQueue[i].layer === layer) { │ │ │ │ │ + tileQueue.splice(i, 1) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i = this.maps.length - 1; i >= 0; --i) { │ │ │ │ │ + this.removeMap(this.maps[i]) │ │ │ │ │ + } │ │ │ │ │ + this.maps = null; │ │ │ │ │ + this.tileQueue = null; │ │ │ │ │ + this.tileQueueId = null; │ │ │ │ │ + this.tileCache = null; │ │ │ │ │ + this.tileCacheIndex = null; │ │ │ │ │ + this._destroyed = true │ │ │ │ │ + } │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Rule = OpenLayers.Class({ │ │ │ │ │ + id: null, │ │ │ │ │ + name: null, │ │ │ │ │ + title: null, │ │ │ │ │ + description: null, │ │ │ │ │ + context: null, │ │ │ │ │ + filter: null, │ │ │ │ │ + elseFilter: false, │ │ │ │ │ + symbolizer: null, │ │ │ │ │ + symbolizers: null, │ │ │ │ │ + minScaleDenominator: null, │ │ │ │ │ + maxScaleDenominator: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.symbolizer = {}; │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + if (this.symbolizers) { │ │ │ │ │ + delete this.symbolizer │ │ │ │ │ + } │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i in this.symbolizer) { │ │ │ │ │ + this.symbolizer[i] = null │ │ │ │ │ + } │ │ │ │ │ + this.symbolizer = null; │ │ │ │ │ + delete this.symbolizers │ │ │ │ │ + }, │ │ │ │ │ + evaluate: function(feature) { │ │ │ │ │ + var context = this.getContext(feature); │ │ │ │ │ + var applies = true; │ │ │ │ │ + if (this.minScaleDenominator || this.maxScaleDenominator) { │ │ │ │ │ + var scale = feature.layer.map.getScale() │ │ │ │ │ + } │ │ │ │ │ + if (this.minScaleDenominator) { │ │ │ │ │ + applies = scale >= OpenLayers.Style.createLiteral(this.minScaleDenominator, context) │ │ │ │ │ + } │ │ │ │ │ + if (applies && this.maxScaleDenominator) { │ │ │ │ │ + applies = scale < OpenLayers.Style.createLiteral(this.maxScaleDenominator, context) │ │ │ │ │ + } │ │ │ │ │ + if (applies && this.filter) { │ │ │ │ │ + if (this.filter.CLASS_NAME == "OpenLayers.Filter.FeatureId") { │ │ │ │ │ + applies = this.filter.evaluate(feature) │ │ │ │ │ + } else { │ │ │ │ │ + applies = this.filter.evaluate(context) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return applies │ │ │ │ │ + }, │ │ │ │ │ + getContext: function(feature) { │ │ │ │ │ + var context = this.context; │ │ │ │ │ + if (!context) { │ │ │ │ │ + context = feature.attributes || feature.data │ │ │ │ │ + } │ │ │ │ │ + if (typeof this.context == "function") { │ │ │ │ │ + context = this.context(feature) │ │ │ │ │ + } │ │ │ │ │ + return context │ │ │ │ │ + }, │ │ │ │ │ + clone: function() { │ │ │ │ │ + var options = OpenLayers.Util.extend({}, this); │ │ │ │ │ + if (this.symbolizers) { │ │ │ │ │ + var len = this.symbolizers.length; │ │ │ │ │ + options.symbolizers = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + options.symbolizers[i] = this.symbolizers[i].clone() │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + options.symbolizer = {}; │ │ │ │ │ + var value, type; │ │ │ │ │ + for (var key in this.symbolizer) { │ │ │ │ │ + value = this.symbolizer[key]; │ │ │ │ │ + type = typeof value; │ │ │ │ │ + if (type === "object") { │ │ │ │ │ + options.symbolizer[key] = OpenLayers.Util.extend({}, value) │ │ │ │ │ + } else if (type === "string") { │ │ │ │ │ + options.symbolizer[key] = value │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + options.filter = this.filter && this.filter.clone(); │ │ │ │ │ + options.context = this.context && OpenLayers.Util.extend({}, this.context); │ │ │ │ │ + return new OpenLayers.Rule(options) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Rule" │ │ │ │ │ +}); │ │ │ │ │ OpenLayers.Format.WPSDescribeProcess = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ VERSION: "1.0.0", │ │ │ │ │ namespaces: { │ │ │ │ │ wps: "http://www.opengis.net/wps/1.0.0", │ │ │ │ │ ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ }, │ │ │ │ │ @@ -18087,28 +17982,25 @@ │ │ │ │ │ destroy: function() { │ │ │ │ │ this.events.destroy(); │ │ │ │ │ this.events = null; │ │ │ │ │ this.servers = null │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.WPSClient" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Spherical = OpenLayers.Spherical || {}; │ │ │ │ │ -OpenLayers.Spherical.DEFAULT_RADIUS = 6378137; │ │ │ │ │ -OpenLayers.Spherical.computeDistanceBetween = function(from, to, radius) { │ │ │ │ │ - var R = radius || OpenLayers.Spherical.DEFAULT_RADIUS; │ │ │ │ │ - var sinHalfDeltaLon = Math.sin(Math.PI * (to.lon - from.lon) / 360); │ │ │ │ │ - var sinHalfDeltaLat = Math.sin(Math.PI * (to.lat - from.lat) / 360); │ │ │ │ │ - var a = sinHalfDeltaLat * sinHalfDeltaLat + sinHalfDeltaLon * sinHalfDeltaLon * Math.cos(Math.PI * from.lat / 180) * Math.cos(Math.PI * to.lat / 180); │ │ │ │ │ - return 2 * R * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a)) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Spherical.computeHeading = function(from, to) { │ │ │ │ │ - var y = Math.sin(Math.PI * (from.lon - to.lon) / 180) * Math.cos(Math.PI * to.lat / 180); │ │ │ │ │ - var x = Math.cos(Math.PI * from.lat / 180) * Math.sin(Math.PI * to.lat / 180) - Math.sin(Math.PI * from.lat / 180) * Math.cos(Math.PI * to.lat / 180) * Math.cos(Math.PI * (from.lon - to.lon) / 180); │ │ │ │ │ - return 180 * Math.atan2(y, x) / Math.PI │ │ │ │ │ -}; │ │ │ │ │ +OpenLayers.Symbolizer = OpenLayers.Class({ │ │ │ │ │ + zIndex: 0, │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ + OpenLayers.Util.extend(this, config) │ │ │ │ │ + }, │ │ │ │ │ + clone: function() { │ │ │ │ │ + var Type = eval(this.CLASS_NAME); │ │ │ │ │ + return new Type(OpenLayers.Util.extend({}, this)) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Symbolizer" │ │ │ │ │ +}); │ │ │ │ │ OpenLayers.Symbolizer.Point = OpenLayers.Class(OpenLayers.Symbolizer, { │ │ │ │ │ initialize: function(config) { │ │ │ │ │ OpenLayers.Symbolizer.prototype.initialize.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Symbolizer.Point" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Symbolizer.Line = OpenLayers.Class(OpenLayers.Symbolizer, { │ │ │ │ │ @@ -18161,3857 +18053,4991 @@ │ │ │ │ │ config.rules.push(this.rules[i].clone()) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ return new OpenLayers.Style2(config) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Style2" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.TileCache = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - format: "image/png", │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ - initialize: function(name, url, layername, options) { │ │ │ │ │ - this.layername = layername; │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name, url, {}, options]); │ │ │ │ │ - this.extension = this.format.split("/")[1].toLowerCase(); │ │ │ │ │ - this.extension = this.extension == "jpg" ? "jpeg" : this.extension │ │ │ │ │ +OpenLayers.Handler = OpenLayers.Class({ │ │ │ │ │ + id: null, │ │ │ │ │ + control: null, │ │ │ │ │ + map: null, │ │ │ │ │ + keyMask: null, │ │ │ │ │ + active: false, │ │ │ │ │ + evt: null, │ │ │ │ │ + touch: false, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Util.extend(this, options); │ │ │ │ │ + this.control = control; │ │ │ │ │ + this.callbacks = callbacks; │ │ │ │ │ + var map = this.map || control.map; │ │ │ │ │ + if (map) { │ │ │ │ │ + this.setMap(map) │ │ │ │ │ + } │ │ │ │ │ + this.id = OpenLayers.Util.createUniqueID(this.CLASS_NAME + "_") │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.TileCache(this.name, this.url, this.layername, this.getOptions()) │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.map = map │ │ │ │ │ + }, │ │ │ │ │ + checkModifiers: function(evt) { │ │ │ │ │ + if (this.keyMask == null) { │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ + var keyModifiers = (evt.shiftKey ? OpenLayers.Handler.MOD_SHIFT : 0) | (evt.ctrlKey ? OpenLayers.Handler.MOD_CTRL : 0) | (evt.altKey ? OpenLayers.Handler.MOD_ALT : 0) | (evt.metaKey ? OpenLayers.Handler.MOD_META : 0); │ │ │ │ │ + return keyModifiers == this.keyMask │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var bbox = this.maxExtent; │ │ │ │ │ - var size = this.tileSize; │ │ │ │ │ - var tileX = Math.round((bounds.left - bbox.left) / (res * size.w)); │ │ │ │ │ - var tileY = Math.round((bounds.bottom - bbox.bottom) / (res * size.h)); │ │ │ │ │ - var tileZ = this.serverResolutions != null ? OpenLayers.Util.indexOf(this.serverResolutions, res) : this.map.getZoom(); │ │ │ │ │ - var components = [this.layername, OpenLayers.Number.zeroPad(tileZ, 2), OpenLayers.Number.zeroPad(parseInt(tileX / 1e6), 3), OpenLayers.Number.zeroPad(parseInt(tileX / 1e3) % 1e3, 3), OpenLayers.Number.zeroPad(parseInt(tileX) % 1e3, 3), OpenLayers.Number.zeroPad(parseInt(tileY / 1e6), 3), OpenLayers.Number.zeroPad(parseInt(tileY / 1e3) % 1e3, 3), OpenLayers.Number.zeroPad(parseInt(tileY) % 1e3, 3) + "." + this.extension]; │ │ │ │ │ - var path = components.join("/"); │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(path, url) │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - url = url.charAt(url.length - 1) == "/" ? url : url + "/"; │ │ │ │ │ - return url + path │ │ │ │ │ + var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.register(events[i], this[events[i]]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.active = true; │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.TileCache" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.WorldWind = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - DEFAULT_PARAMS: {}, │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - lzd: null, │ │ │ │ │ - zoomLevels: null, │ │ │ │ │ - initialize: function(name, url, lzd, zoomLevels, params, options) { │ │ │ │ │ - this.lzd = lzd; │ │ │ │ │ - this.zoomLevels = zoomLevels; │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - newArguments.push(name, url, params, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - this.params = OpenLayers.Util.applyDefaults(this.params, this.DEFAULT_PARAMS) │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + var events = OpenLayers.Events.prototype.BROWSER_EVENTS; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.unregister(events[i], this[events[i]]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.touch = false; │ │ │ │ │ + this.active = false; │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - getZoom: function() { │ │ │ │ │ - var zoom = this.map.getZoom(); │ │ │ │ │ - var extent = this.map.getMaxExtent(); │ │ │ │ │ - zoom = zoom - Math.log(this.maxResolution / (this.lzd / 512)) / Math.log(2); │ │ │ │ │ - return zoom │ │ │ │ │ + startTouch: function() { │ │ │ │ │ + if (!this.touch) { │ │ │ │ │ + this.touch = true; │ │ │ │ │ + var events = ["mousedown", "mouseup", "mousemove", "click", "dblclick", "mouseout"]; │ │ │ │ │ + for (var i = 0, len = events.length; i < len; i++) { │ │ │ │ │ + if (this[events[i]]) { │ │ │ │ │ + this.unregister(events[i], this[events[i]]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var zoom = this.getZoom(); │ │ │ │ │ - var extent = this.map.getMaxExtent(); │ │ │ │ │ - var deg = this.lzd / Math.pow(2, this.getZoom()); │ │ │ │ │ - var x = Math.floor((bounds.left - extent.left) / deg); │ │ │ │ │ - var y = Math.floor((bounds.bottom - extent.bottom) / deg); │ │ │ │ │ - if (this.map.getResolution() <= this.lzd / 512 && this.getZoom() <= this.zoomLevels) { │ │ │ │ │ - return this.getFullRequestString({ │ │ │ │ │ - L: zoom, │ │ │ │ │ - X: x, │ │ │ │ │ - Y: y │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - return OpenLayers.Util.getImageLocation("blank.gif") │ │ │ │ │ + callback: function(name, args) { │ │ │ │ │ + if (name && this.callbacks[name]) { │ │ │ │ │ + this.callbacks[name].apply(this.control, args) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.WorldWind" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.TMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - serviceVersion: "1.0.0", │ │ │ │ │ - layername: null, │ │ │ │ │ - type: null, │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - tileOrigin: null, │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ - zoomOffset: 0, │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - newArguments.push(name, url, {}, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments) │ │ │ │ │ + register: function(name, method) { │ │ │ │ │ + this.map.events.registerPriority(name, this, method); │ │ │ │ │ + this.map.events.registerPriority(name, this, this.setEvent) │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.TMS(this.name, this.url, this.getOptions()) │ │ │ │ │ - } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ + unregister: function(name, method) { │ │ │ │ │ + this.map.events.unregister(name, this, method); │ │ │ │ │ + this.map.events.unregister(name, this, this.setEvent) │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((bounds.bottom - this.tileOrigin.lat) / (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getServerZoom(); │ │ │ │ │ - var path = this.serviceVersion + "/" + this.layername + "/" + z + "/" + x + "/" + y + "." + this.type; │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(path, url) │ │ │ │ │ - } │ │ │ │ │ - return url + path │ │ │ │ │ + setEvent: function(evt) { │ │ │ │ │ + this.evt = evt; │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (!this.tileOrigin) { │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.map.maxExtent.left, this.map.maxExtent.bottom) │ │ │ │ │ - } │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.control = this.map = null │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.TMS" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.XYZ = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - sphericalMercator: false, │ │ │ │ │ - zoomOffset: 0, │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - if (options && options.sphericalMercator || this.sphericalMercator) { │ │ │ │ │ - options = OpenLayers.Util.extend({ │ │ │ │ │ - projection: "EPSG:900913", │ │ │ │ │ - numZoomLevels: 19 │ │ │ │ │ - }, options) │ │ │ │ │ +OpenLayers.Handler.MOD_NONE = 0; │ │ │ │ │ +OpenLayers.Handler.MOD_SHIFT = 1; │ │ │ │ │ +OpenLayers.Handler.MOD_CTRL = 2; │ │ │ │ │ +OpenLayers.Handler.MOD_ALT = 4; │ │ │ │ │ +OpenLayers.Handler.MOD_META = 8; │ │ │ │ │ +OpenLayers.Spherical = OpenLayers.Spherical || {}; │ │ │ │ │ +OpenLayers.Spherical.DEFAULT_RADIUS = 6378137; │ │ │ │ │ +OpenLayers.Spherical.computeDistanceBetween = function(from, to, radius) { │ │ │ │ │ + var R = radius || OpenLayers.Spherical.DEFAULT_RADIUS; │ │ │ │ │ + var sinHalfDeltaLon = Math.sin(Math.PI * (to.lon - from.lon) / 360); │ │ │ │ │ + var sinHalfDeltaLat = Math.sin(Math.PI * (to.lat - from.lat) / 360); │ │ │ │ │ + var a = sinHalfDeltaLat * sinHalfDeltaLat + sinHalfDeltaLon * sinHalfDeltaLon * Math.cos(Math.PI * from.lat / 180) * Math.cos(Math.PI * to.lat / 180); │ │ │ │ │ + return 2 * R * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a)) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Spherical.computeHeading = function(from, to) { │ │ │ │ │ + var y = Math.sin(Math.PI * (from.lon - to.lon) / 180) * Math.cos(Math.PI * to.lat / 180); │ │ │ │ │ + var x = Math.cos(Math.PI * from.lat / 180) * Math.sin(Math.PI * to.lat / 180) - Math.sin(Math.PI * from.lat / 180) * Math.cos(Math.PI * to.lat / 180) * Math.cos(Math.PI * (from.lon - to.lon) / 180); │ │ │ │ │ + return 180 * Math.atan2(y, x) / Math.PI │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Lang["hsb"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Wotmołwa njewobdźěłaneho naprašowanja ${statusText}", │ │ │ │ │ + Permalink: "Trajny wotkaz", │ │ │ │ │ + Overlays: "Naworštowanja", │ │ │ │ │ + "Base Layer": "Zakładna runina", │ │ │ │ │ + noFID: "Funkcija, za kotruž FID njeje, njeda so aktualizować.", │ │ │ │ │ + browserNotSupported: "Twój wobhladowak wektorowe rysowanje njepodpěruje. Tuchwilu podpěrowane rysowaki su:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "Kajkosć minZoomLevel je jenož za wužiwanje z worštami myslena, kotrež wot FixedZoomLevels pochadźeja. Zo tuta woršta wfs za minZoomLevel přepruwuje, je relikt zańdźenosće. Njemóžemy wšak ju wotstronić, bjeztoho zo aplikacije, kotrež na OpenLayers bazěruja a snano tutu kajkosć wužiwaja, hižo njefunguja. Tohodla smy ju jako zestarjenu woznamjenili -- přepruwowanje za minZoomLevel budu so we wersiji 3.0 wotstronjeć. Prošu wužij město toho nastajenje min/max, kaž je tu wopisane: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "WFS-Transakcija: WUSPĚŠNA ${response}", │ │ │ │ │ + commitFailed: "WFS-Transakcija: NJEPORADŹENA ${response}", │ │ │ │ │ + googleWarning: "Woršta Google njemóžeše so korektnje začitać.<br><br>Zo by tutu zdźělenku wotbył, wubjer nowy BaseLayer z wuběra worštow horjeka naprawo.<br><br>Najskerje so to stawa, dokelž skript biblioteki Google Maps pak njebu zapřijaty pak njewobsahuje korektny kluč API za twoje sydło.<br><br>Wuwiwarjo: Za pomoc ke korektnemu fungowanju worštow\n<a href='http://trac.openlayers.org/wiki/Google' target='_blank'>tu kliknyć</a>", │ │ │ │ │ + getLayerWarning: "Woršta ${layerType} njemóžeše so korektnje začitać.<br><br>Zo by tutu zdźělenku wotbył, wubjer nowy BaseLayer z wuběra worštow horjeka naprawo.<br><br>Najskerje so to stawa, dokelž skript biblioteki ${layerLib} njebu korektnje zapřijaty.<br><br>Wuwiwarjo: Za pomoc ke korektnemu fungowanju worštow\n<a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>tu kliknyć</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Měritko = 1 : ${scaleDenom}", │ │ │ │ │ + W: "Z", │ │ │ │ │ + E: "W", │ │ │ │ │ + N: "S", │ │ │ │ │ + S: "J", │ │ │ │ │ + reprojectDeprecated: 'Wužiwaš opciju "reproject" wořšty ${layerName}. Tuta opcija je zestarjena: jeje wužiwanje bě myslene, zo by zwobraznjenje datow nad komercielnymi bazowymi kartami podpěrało, ale funkcionalnosć měła so nětko z pomocu Sperical Mercator docpěć. Dalše informacije steja na http://trac.openlayers.org/wiki/SphericalMercator k dispoziciji.', │ │ │ │ │ + methodDeprecated: "Tuta metoda je so njeschwaliła a budźe so w 3.0 wotstronjeć. Prošu wužij ${newMethod} město toho." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["nn"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Skala = 1 : ${scaleDenom}" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang.ca = { │ │ │ │ │ + unhandledRequest: "Resposta a petició no gestionada ${statusText}", │ │ │ │ │ + Permalink: "Enllaç permanent", │ │ │ │ │ + Overlays: "Capes addicionals", │ │ │ │ │ + "Base Layer": "Capa Base", │ │ │ │ │ + noFID: "No es pot actualitzar un element per al que no existeix FID.", │ │ │ │ │ + browserNotSupported: "El seu navegador no suporta renderització vectorial. Els renderitzadors suportats actualment són:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "La propietat minZoomLevel s'ha d'utilitzar només " + "amb les capes que tenen FixedZoomLevels. El fet que " + "una capa wfs comprovi minZoomLevel és una relíquia del " + "passat. No podem, però, eliminar-la sense trencar " + "les aplicacions d'OpenLayers que en puguin dependre. " + "Així doncs estem fent-la obsoleta -- la comprovació " + "minZoomLevel s'eliminarà a la versió 3.0. Feu servir " + "els paràmetres min/max resolution en substitució, tal com es descriu aquí: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "Transacció WFS: CORRECTA ${response}", │ │ │ │ │ + commitFailed: "Transacció WFS: HA FALLAT ${response}", │ │ │ │ │ + googleWarning: "La capa Google no s'ha pogut carregar correctament.<br><br>" + "Per evitar aquest missatge, seleccioneu una nova Capa Base " + "al gestor de capes de la cantonada superior dreta.<br><br>" + "Probablement això és degut a que l'script de la biblioteca de " + "Google Maps no ha estat inclòs a la vostra pàgina, o no " + "conté la clau de l'API correcta per a la vostra adreça.<br><br>" + "Desenvolupadors: Per obtenir consells sobre com fer anar això, " + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>féu clic aquí</a>", │ │ │ │ │ + getLayerWarning: "Per evitar aquest missatge, seleccioneu una nova Capa Base " + "al gestor de capes de la cantonada superior dreta.<br><br>" + "Probablement això és degut a que l'script de la biblioteca " + "${layerLib} " + "no ha estat inclòs a la vostra pàgina.<br><br>" + "Desenvolupadors: Per obtenir consells sobre com fer anar això, " + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>féu clic aquí</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Escala = 1 : ${scaleDenom}", │ │ │ │ │ + W: "O", │ │ │ │ │ + E: "E", │ │ │ │ │ + N: "N", │ │ │ │ │ + S: "S", │ │ │ │ │ + Graticule: "Retícula", │ │ │ │ │ + reprojectDeprecated: "Esteu fent servir l'opció 'reproject' a la capa " + "${layerName}. Aquesta opció és obsoleta: el seu ús fou concebut " + "per suportar la visualització de dades sobre mapes base comercials, " + "però ara aquesta funcionalitat s'hauria d'assolir mitjançant el suport " + "de la projecció Spherical Mercator. Més informació disponible a " + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Aquest mètode és obsolet i s'eliminarà a la versió 3.0. " + "Si us plau feu servir em mètode alternatiu ${newMethod}.", │ │ │ │ │ + end: "" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Lang["oc"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Requèsta pas gerida, retorna ${statusText}", │ │ │ │ │ + Permalink: "Permaligam", │ │ │ │ │ + Overlays: "Calques", │ │ │ │ │ + "Base Layer": "Calc de basa", │ │ │ │ │ + noFID: "Impossible de metre a jorn un objècte sens identificant (fid).", │ │ │ │ │ + browserNotSupported: "Vòstre navegidor supòrta pas lo rendut vectorial. Los renderers actualament suportats son : \n${renderers}", │ │ │ │ │ + minZoomLevelError: "La proprietat minZoomLevel deu èsser utilizada solament per de jaces FixedZoomLevels-descendent. Lo fach qu'aqueste jaç WFS verifique la preséncia de minZoomLevel es una relica del passat. Çaquelà, la podèm suprimir sens copar d'aplicacions que ne poirián dependre. Es per aquò que la depreciam -- la verificacion del minZoomLevel serà suprimida en version 3.0. A la plaça, mercés d'utilizar los paramètres de resolucions min/max tal coma descrich sus : http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "Transaccion WFS : SUCCES ${response}", │ │ │ │ │ + commitFailed: "Transaccion WFS : FRACAS ${response}", │ │ │ │ │ + googleWarning: "Lo jaç Google es pas estat en mesura de se cargar corrèctament.<br><br>Per suprimir aqueste messatge, causissètz una BaseLayer novèla dins lo selector de jaç en naut a drecha.<br><br>Aquò es possiblament causat par la non-inclusion de la librariá Google Maps, o alara perque que la clau de l'API correspond pas a vòstre site.<br><br>Desvolopaires : per saber cossí corregir aquò, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>clicatz aicí</a>", │ │ │ │ │ + getLayerWarning: "Lo jaç ${layerType} es pas en mesura de se cargar corrèctament.<br><br>Per suprimir aqueste messatge, causissètz una BaseLayer novèla dins lo selector de jaç en naut a drecha.<br><br>Aquò es possiblament causat per la non-inclusion de la librariá ${layerLib}.<br><br>Desvolopaires : per saber cossí corregir aquí, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>clicatz aicí</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Escala ~ 1 : ${scaleDenom}", │ │ │ │ │ + W: "O", │ │ │ │ │ + E: "È", │ │ │ │ │ + N: "N", │ │ │ │ │ + S: "S", │ │ │ │ │ + reprojectDeprecated: "Utilizatz l'opcion 'reproject' sul jaç ${layerName}. Aquesta opcion es despreciada : Son usatge permetiá d'afichar de donadas al dessús de jaces raster comercials. Aquesta foncionalitat ara es suportada en utilizant lo supòrt de la projeccion Mercator Esferica. Mai d'informacion es disponibla sus http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Aqueste metòde es despreciada, e serà suprimida a la version 3.0. Mercés d'utilizar ${newMethod} a la plaça." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["nl"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Het verzoek is niet afgehandeld met de volgende melding: ${statusText}", │ │ │ │ │ + Permalink: "Permanente verwijzing", │ │ │ │ │ + Overlays: "Overlays", │ │ │ │ │ + "Base Layer": "Achtergrondkaart", │ │ │ │ │ + noFID: "Een optie die geen FID heeft kan niet bijgewerkt worden.", │ │ │ │ │ + browserNotSupported: "Uw browser ondersteunt het weergeven van vectoren niet.\nMomenteel ondersteunde weergavemogelijkheden:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "De eigenschap minZoomLevel is alleen bedoeld voor gebruik lagen met die afstammen van FixedZoomLevels-lagen.\nDat deze WFS-laag minZoomLevel controleert, is een overblijfsel uit het verleden.\nWe kunnen deze controle echter niet verwijderen zonder op OL gebaseerde applicaties die hervan afhankelijk zijn stuk te maken.\nDaarom heeft deze functionaliteit de eigenschap 'deprecated' gekregen - de minZoomLevel wordt verwijderd in versie 3.0.\nGebruik in plaats van deze functie de mogelijkheid om min/max voor resolutie in te stellen zoals op de volgende pagina wordt beschreven:\nhttp://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "WFS-transactie: succesvol ${response}", │ │ │ │ │ + commitFailed: "WFS-transactie: mislukt ${response}", │ │ │ │ │ + googleWarning: "De Google-Layer kon niet correct geladen worden.<br /><br />\nOm deze melding niet meer te krijgen, moet u een andere achtergrondkaart kiezen in de laagwisselaar in de rechterbovenhoek.<br /><br />\nDit komt waarschijnlijk doordat de bibliotheek ${layerLib} niet correct ingevoegd is.<br /><br />\nOntwikkelaars: <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>klik hier</a> om dit werkend te krijgen.", │ │ │ │ │ + getLayerWarning: "De laag ${layerType} kon niet goed geladen worden.<br /><br />\nOm deze melding niet meer te krijgen, moet u een andere achtergrondkaart kiezen in de laagwisselaar in de rechterbovenhoek.<br /><br />\nDit komt waarschijnlijk doordat de bibliotheek ${layerLib} niet correct is ingevoegd.<br /><br />\nOntwikkelaars: <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>klik hier</a> om dit werkend te krijgen.", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Schaal = 1 : ${scaleDenom}", │ │ │ │ │ + W: "W", │ │ │ │ │ + E: "O", │ │ │ │ │ + N: "N", │ │ │ │ │ + S: "Z", │ │ │ │ │ + reprojectDeprecated: "U gebruikt de optie 'reproject' op de laag ${layerName}.\nDeze optie is vervallen: deze optie was ontwikkeld om gegevens over commerciële basiskaarten weer te geven, maar deze functionaliteit wordt nu bereikt door ondersteuning van Spherical Mercator.\nMeer informatie is beschikbaar op http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Deze methode is verouderd en wordt verwijderd in versie 3.0.\nGebruik ${newMethod}." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["zh-TW"] = { │ │ │ │ │ + unhandledRequest: "未處理的請求,傳回值為 ${statusText}。", │ │ │ │ │ + Permalink: "永久連結", │ │ │ │ │ + Overlays: "額外圖層", │ │ │ │ │ + "Base Layer": "基礎圖層", │ │ │ │ │ + noFID: "因為沒有 FID 所以無法更新 feature。", │ │ │ │ │ + browserNotSupported: "您的瀏覽器未支援向量渲染. 目前支援的渲染方式是:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "minZoomLevel 屬性僅適合用在 " + "FixedZoomLevels-descendent 類型的圖層. 這個" + "wfs layer 的 minZoomLevel 是過去所遺留下來的," + "然而我們不能移除它而不讓它將" + "過去的程式相容性給破壞掉。" + "因此我們將會迴避使用它 -- minZoomLevel " + "會在3.0被移除,請改" + "用在這邊描述的 min/max resolution 設定: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "WFS Transaction: 成功 ${response}", │ │ │ │ │ + commitFailed: "WFS Transaction: 失敗 ${response}", │ │ │ │ │ + googleWarning: "The Google Layer 圖層無法被正確的載入。<br><br>" + "要迴避這個訊息, 請在右上角的圖層改變器裡," + "選一個新的基礎圖層。<br><br>" + "很有可能是因為 Google Maps 的函式庫" + "腳本沒有被正確的置入,或沒有包含 " + "您網站上正確的 API key <br><br>" + "開發者: 要幫助這個行為正確完成," + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>請按這裡</a>", │ │ │ │ │ + getLayerWarning: "${layerType} 圖層無法被正確的載入。<br><br>" + "要迴避這個訊息, 請在右上角的圖層改變器裡," + "選一個新的基礎圖層。<br><br>" + "很有可能是因為 ${layerLib} 的函式庫" + "腳本沒有被正確的置入。<br><br>" + "開發者: 要幫助這個行為正確完成," + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>請按這裡</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Scale = 1 : ${scaleDenom}", │ │ │ │ │ + reprojectDeprecated: "你正使用 'reproject' 這個選項 " + "在 ${layerName} 層。這個選項已經不再使用:" + "它的使用原本是設計用來支援在商業地圖上秀出資料," + "但這個功能已經被" + "Spherical Mercator所取代。更多的資訊可以在 " + "http://trac.openlayers.org/wiki/SphericalMercator 找到。", │ │ │ │ │ + methodDeprecated: "這個方法已經不再使用且在3.0將會被移除," + "請使用 ${newMethod} 來代替。", │ │ │ │ │ + end: "" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Lang["hu"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Nem kezelt kérés visszatérése ${statusText}", │ │ │ │ │ + Permalink: "Permalink", │ │ │ │ │ + Overlays: "Rávetítések", │ │ │ │ │ + "Base Layer": "Alapréteg", │ │ │ │ │ + noFID: "Nem frissíthető olyan jellemző, amely nem rendelkezik FID-del.", │ │ │ │ │ + browserNotSupported: "A böngészője nem támogatja a vektoros renderelést. A jelenleg támogatott renderelők:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "A minZoomLevel tulajdonságot csak a következővel való használatra szánták: FixedZoomLevels-leszármazott fóliák. Ez azt jelenti, hogy a minZoomLevel wfs fólia jelölőnégyzetei már a múlté. Mi azonban nem távolíthatjuk el annak a veszélye nélkül, hogy az esetlegesen ettől függő OL alapú alkalmazásokat tönkretennénk. Ezért ezt érvénytelenítjük -- a minZoomLevel az alul levő jelölőnégyzet a 3.0-s verzióból el lesz távolítva. Kérjük, helyette használja a min/max felbontás beállítást, amelyről az alábbi helyen talál leírást: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "WFS tranzakció: SIKERES ${response}", │ │ │ │ │ + commitFailed: "WFS tranzakció: SIKERTELEN ${response}", │ │ │ │ │ + googleWarning: "A Google fólia betöltése sikertelen.<br><br>Ahhoz, hogy ez az üzenet eltűnjön, válasszon egy új BaseLayer fóliát a jobb felső sarokban található fóliakapcsoló segítségével.<br><br>Nagy valószínűséggel ez azért van, mert a Google Maps könyvtár parancsfájlja nem található, vagy nem tartalmazza az Ön oldalához tartozó megfelelő API-kulcsot.<br><br>Fejlesztőknek: A helyes működtetésre vonatkozó segítség az alábbi helyen érhető el, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>kattintson ide</a>", │ │ │ │ │ + getLayerWarning: "A(z) ${layerType} fólia nem töltődött be helyesen.<br><br>Ahhoz, hogy ez az üzenet eltűnjön, válasszon egy új BaseLayer fóliát a jobb felső sarokban található fóliakapcsoló segítségével.<br><br>Nagy valószínűséggel ez azért van, mert a(z) ${layerLib} könyvtár parancsfájlja helytelen.<br><br>Fejlesztőknek: A helyes működtetésre vonatkozó segítség az alábbi helyen érhető el, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>kattintson ide</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Lépték = 1 : ${scaleDenom}", │ │ │ │ │ + W: "Ny", │ │ │ │ │ + E: "K", │ │ │ │ │ + N: "É", │ │ │ │ │ + S: "D", │ │ │ │ │ + reprojectDeprecated: "Ön a 'reproject' beállítást használja a(z) ${layerName} fólián. Ez a beállítás érvénytelen: használata az üzleti alaptérképek fölötti adatok megjelenítésének támogatására szolgált, de ezt a funkció ezentúl a Gömbi Mercator használatával érhető el. További információ az alábbi helyen érhető el: http://trac.openlayers.org/wiki/SphericalMercator", │ │ │ │ │ + methodDeprecated: "Ez a módszer érvénytelenítve lett és a 3.0-s verzióból el lesz távolítva. Használja a(z) ${newMethod} módszert helyette." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["pt"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Servidor devolveu erro não contemplado ${statusText}", │ │ │ │ │ + Permalink: "Ligação permanente", │ │ │ │ │ + Overlays: "Sobreposições", │ │ │ │ │ + "Base Layer": "Camada Base", │ │ │ │ │ + noFID: "Não é possível atualizar um elemento para a qual não há FID.", │ │ │ │ │ + browserNotSupported: "O seu navegador não suporta renderização vetorial. Actualmente os renderizadores suportados são:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "A propriedade minZoomLevel só deve ser usada com as camadas descendentes da FixedZoomLevels. A verificação da propriedade por esta camada wfs é uma relíquia do passado. No entanto, não podemos removê-la sem correr o risco de afectar aplicações OL que dependam dela. Portanto, estamos a torná-la obsoleta -- a verificação minZoomLevel será removida na versão 3.0. Em vez dela, por favor, use as opções de resolução min/max descritas aqui: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "Transacção WFS: SUCESSO ${response}", │ │ │ │ │ + commitFailed: "Transacção WFS: FALHOU ${response}", │ │ │ │ │ + googleWarning: "A Camada Google não foi correctamente carregada.<br><br>Para deixar de receber esta mensagem, seleccione uma nova Camada-Base no ''switcher'' de camadas no canto superior direito.<br><br>Provavelmente, isto acontece porque o ''script'' da biblioteca do Google Maps não foi incluído ou não contém a chave API correcta para o seu sítio.<br><br>Programadores: Para ajuda sobre como solucionar o problema <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>clique aqui</a> .", │ │ │ │ │ + getLayerWarning: "A camada ${layerType} não foi correctamente carregada.<br><br>Para desactivar esta mensagem, seleccione uma nova Camada-Base no ''switcher'' de camadas no canto superior direito.<br><br>Provavelmente, isto acontece porque o ''script'' da biblioteca ${layerLib} não foi incluído correctamente.<br><br>Programadores: Para ajuda sobre como solucionar o problema <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>clique aqui</a> .", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Escala = 1 : ${scaleDenom}", │ │ │ │ │ + W: "O", │ │ │ │ │ + E: "E", │ │ │ │ │ + N: "N", │ │ │ │ │ + S: "S", │ │ │ │ │ + reprojectDeprecated: "Está usando a opção 'reproject' na camada ${layerName}. Esta opção é obsoleta: foi concebida para permitir a apresentação de dados sobre mapas-base comerciais, mas esta funcionalidade é agora suportada pelo Mercator Esférico. Mais informação está disponível em http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Este método foi declarado obsoleto e será removido na versão 3.0. Por favor, use ${newMethod} em vez disso." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["hr"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Nepodržani zahtjev ${statusText}", │ │ │ │ │ + Permalink: "Permalink", │ │ │ │ │ + Overlays: "Overlays", │ │ │ │ │ + "Base Layer": "Osnovna karta", │ │ │ │ │ + noFID: "Ne mogu ažurirati značajku za koju ne postoji FID.", │ │ │ │ │ + browserNotSupported: "Vaš preglednik ne podržava vektorsko renderiranje. Trenutno podržani rendereri su: ${renderers}", │ │ │ │ │ + commitSuccess: "WFS Transakcija: USPJEŠNA ${response}", │ │ │ │ │ + commitFailed: "WFS Transakcija: NEUSPJEŠNA ${response}", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Mjerilo = 1 : ${scaleDenom}", │ │ │ │ │ + methodDeprecated: "Ova metoda nije odobrena i biti će maknuta u 3.0. Koristite ${newMethod}." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang.ro = { │ │ │ │ │ + unhandledRequest: "Cerere nesoluționată return ${statusText}", │ │ │ │ │ + Permalink: "Legatură permanentă", │ │ │ │ │ + Overlays: "Straturi vector", │ │ │ │ │ + "Base Layer": "Straturi de bază", │ │ │ │ │ + noFID: "Nu pot actualiza un feature pentru care nu există FID.", │ │ │ │ │ + browserNotSupported: "Browserul tău nu suportă afișarea vectorilor. Supoetul curent pentru randare:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "Proprietatea minZoomLevel este doar pentru a fi folosită " + "cu straturile FixedZoomLevels-descendent. De aceea acest " + "strat wfs verifică dacă minZoomLevel este o relicvă" + ". Nu îl putem , oricum, înlătura fără " + "a afecta aplicațiile Openlayers care depind de ea." + " De aceea considerăm depreciat -- minZoomLevel " + "și îl vom înlătura în 3.0. Folosește " + "min/max resolution cum este descrisă aici: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "Tranzacție WFS: SUCCES ${response}", │ │ │ │ │ + commitFailed: "Tranzacție WFS : EȘEC ${response}", │ │ │ │ │ + googleWarning: "Stratul Google nu a putut fi încărcat corect.<br><br>" + "Pentru a elimina acest mesaj, selectează un nou strat de bază " + "în Layerswitcher din colțul dreata-sus.<br><br>" + "Asta datorită, faptului că Google Maps library " + "script nu este inclus, sau nu conține " + "cheia API corectă pentru situl tău.<br><br>" + "Developeri: Pentru ajutor, " + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>apăsați aici</a>", │ │ │ │ │ + getLayerWarning: "Stratul ${layerType} nu a putut fi încărcat corect.<br><br>" + "pentru a înlătura acest mesaj, selectează un nou strat de bază " + "Acesta eroare apare de obicei când ${layerLib} library " + "script nu a fost încărcat corect.<br><br>" + "Developeri: Pentru ajutor privind utilizarea corectă, " + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>apasă aici</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Scara = 1 : ${scaleDenom}", │ │ │ │ │ + W: "V", │ │ │ │ │ + E: "E", │ │ │ │ │ + N: "N", │ │ │ │ │ + S: "S", │ │ │ │ │ + Graticule: "Graticule", │ │ │ │ │ + reprojectDeprecated: "folosești opțiunea 'reproject' " + "pentru stratul ${layerName} . Această opțiune este depreciată: " + "a fost utilizată pentru afișarea straturilor de bază comerciale " + "Mai multe informații despre proiecția Mercator sunt disponibile aici " + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Această metodă este depreciată și va fi înlăturată in versiunea 3.0. " + "folosește metoda ${newMethod}.", │ │ │ │ │ + end: "" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Lang["ar"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Permalink: "وصلة دائمة", │ │ │ │ │ + "Base Layer": "الطبقة الاساسية", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "النسبة = 1 : ${scaleDenom}", │ │ │ │ │ + W: "غ", │ │ │ │ │ + E: "شر", │ │ │ │ │ + N: "شم", │ │ │ │ │ + S: "ج" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["ru"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Необработанный запрос вернул ${statusText}", │ │ │ │ │ + Permalink: "Постоянная ссылка", │ │ │ │ │ + Overlays: "Слои", │ │ │ │ │ + "Base Layer": "Основной слой", │ │ │ │ │ + noFID: "Невозможно обновить объект, для которого нет FID.", │ │ │ │ │ + browserNotSupported: "Ваш браузер не поддерживает векторную графику. На данный момент поддерживаются:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "Свойство minZoomLevel предназначено только для использования со слоями, являющимися потомками FixedZoomLevels. То, что этот WFS-слой проверяется на minZoomLevel — реликт прошлого. Однако мы не можем удалить эту функцию, так как, возможно, от неё зависят некоторые основанные на OpenLayers приложения. Функция объявлена устаревшей — проверка minZoomLevel будет удалена в 3.0. Пожалуйста, используйте вместо неё настройку мин/макс разрешения, описанную здесь: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "Транзакция WFS: УСПЕШНО ${response}", │ │ │ │ │ + commitFailed: "Транзакция WFS: ОШИБКА ${response}", │ │ │ │ │ + googleWarning: "Слой Google не удалось нормально загрузить.<br><br>Чтобы избавиться от этого сообщения, выбите другой основной слой в переключателе в правом верхнем углу.<br><br>Скорее всего, причина в том, что библиотека Google Maps не была включена или не содержит корректного API-ключа для вашего сайта.<br><br>Разработчикам: чтобы узнать, как сделать, чтобы всё заработало, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>щёлкните тут</a>", │ │ │ │ │ + getLayerWarning: "Слой ${layerType} не удалось нормально загрузить. <br><br>Чтобы избавиться от этого сообщения, выбите другой основной слой в переключателе в правом верхнем углу.<br><br>Скорее всего, причина в том, что библиотека ${layerLib} не была включена или была включена некорректно.<br><br>Разработчикам: чтобы узнать, как сделать, чтобы всё заработало, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>щёлкните тут</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Масштаб = 1 : ${scaleDenom}", │ │ │ │ │ + W: "З", │ │ │ │ │ + E: "В", │ │ │ │ │ + N: "С", │ │ │ │ │ + S: "Ю", │ │ │ │ │ + reprojectDeprecated: "Вы используете опцию 'reproject' для слоя ${layerName}. Эта опция является устаревшей: ее использование предполагалось для поддержки показа данных поверх коммерческих базовых карт, но теперь этот функционал несёт встроенная поддержка сферической проекции Меркатора. Больше сведений доступно на http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Этот метод считается устаревшим и будет удалён в версии 3.0. Пожалуйста, пользуйтесь ${newMethod}." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["lt"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Neapdorota užklausa gražino ${statusText}", │ │ │ │ │ + Permalink: "Pastovi nuoroda", │ │ │ │ │ + Overlays: "Papildomi sluoksniai", │ │ │ │ │ + "Base Layer": "Pagrindinis sluoksnis", │ │ │ │ │ + noFID: "Negaliu atnaujinti objekto, kuris neturi FID.", │ │ │ │ │ + browserNotSupported: "Jūsų naršyklė nemoka parodyti vektorių. Šiuo metu galima naudotis tokiais rodymo varikliais:\n{renderers}", │ │ │ │ │ + commitSuccess: "WFS Tranzakcija: PAVYKO ${response}", │ │ │ │ │ + commitFailed: "WFS Tranzakcija: ŽLUGO ${response}", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Mastelis = 1 : ${scaleDenom}", │ │ │ │ │ + W: "V", │ │ │ │ │ + E: "R", │ │ │ │ │ + N: "Š", │ │ │ │ │ + S: "P", │ │ │ │ │ + Graticule: "Tinklelis", │ │ │ │ │ + methodDeprecated: "Šis metodas yra pasenęs ir 3.0 versijoje bus pašalintas. " + "Prašome naudoti ${newMethod}.", │ │ │ │ │ + end: "" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["nds"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Unbehannelt Trüchmellels för de Anfraag ${statusText}", │ │ │ │ │ + Permalink: "Permalink", │ │ │ │ │ + Overlays: "Overlays", │ │ │ │ │ + "Base Layer": "Achtergrundkoort", │ │ │ │ │ + noFID: "En Feature, dat keen FID hett, kann nich aktuell maakt warrn.", │ │ │ │ │ + browserNotSupported: "Dien Browser ünnerstütt keen Vektorbiller. Ünnerstütt Renderers:\n${renderers}", │ │ │ │ │ + commitSuccess: "WFS-Transakschoon: hett klappt ${response}", │ │ │ │ │ + commitFailed: "WFS-Transakschoon: hett nich klappt ${response}", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Skaal = 1 : ${scaleDenom}", │ │ │ │ │ + methodDeprecated: "Disse Methood is oold un schall dat in 3.0 nich mehr geven. Bruuk dor man beter ${newMethod} för." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["br"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Distro evel reked anveret ${statusText}", │ │ │ │ │ + Permalink: "Peurliamm", │ │ │ │ │ + Overlays: "Gwiskadoù", │ │ │ │ │ + "Base Layer": "Gwiskad diazez", │ │ │ │ │ + noFID: "N'haller ket hizivaat un elfenn ma n'eus ket a niverenn-anaout (FID) eviti.", │ │ │ │ │ + browserNotSupported: "N'eo ket skoret an daskor vektorel gant ho merdeer. Setu aze an daskorerioù skoret evit ar poent :\n${renderers}", │ │ │ │ │ + minZoomLevelError: "Ne zleer implijout ar perzh minZoomLevel nemet evit gwiskadoù FixedZoomLevels-descendent. Ar fed ma wiria ar gwiskad WHS-se hag-eñ ez eus eus minZoomLevel zo un aspadenn gozh. Koulskoude n'omp ket evit e ziverkañ kuit da derriñ arloadoù diazezet war OL a c'hallfe bezañ stag outañ. Setu perak eo dispredet -- Lamet kuit e vo ar gwiriañ minZoomLevel a-is er stumm 3.0. Ober gant an arventennoù bihanañ/brasañ evel deskrivet amañ e plas : http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "Treuzgread WFS : MAT EO ${response}", │ │ │ │ │ + commitFailed: "Treuzgread WFS Transaction: C'HWITET ${response}", │ │ │ │ │ + googleWarning: "N'eus ket bet gallet kargañ ar gwiskad Google ent reizh.<br><br>Evit en em zizober eus ar c'hemenn-mañ, dibabit ur BaseLayer nevez en diuzer gwiskadoù er c'horn dehoù el laez.<br><br>Sur a-walc'h eo peogwir n'eo ket bet ensoc'het levraoueg Google Maps pe neuze ne glot ket an alc'hwez API gant ho lec'hienn.<br><br>Diorroerien : Evit reizhañ an dra-se, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>click here</a>", │ │ │ │ │ + getLayerWarning: "N'haller ket kargañ ar gwiskad ${layerType} ent reizh.<br><br>Evit en em zizober eus ar c'hemenn-mañ, dibabit ur BaseLayer nevez en diuzer gwiskadoù er c'horn dehoù el laez.<br><br>Sur a-walc'h eo peogwir n'eo ket bet ensoc'het mat al levraoueg ${layerLib}.<br><br>Diorroerien : Evit gouzout penaos reizhañ an dra-se, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>click here</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Skeul = 1 : ${scaleDenom}", │ │ │ │ │ + W: "K", │ │ │ │ │ + E: "R", │ │ │ │ │ + N: "N", │ │ │ │ │ + S: "S", │ │ │ │ │ + reprojectDeprecated: "Emaoc'h oc'h implijout an dibarzh 'reproject' war ar gwiskad ${layerName}. Dispredet eo an dibarzh-mañ : bet eo hag e talveze da ziskwel roadennoù war-c'horre kartennoù diazez kenwerzhel, un dra hag a c'haller ober bremañ gant an arc'hwel dre skor banndres boullek Mercator. Muioc'h a ditouroù a c'haller da gaout war http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Dispredet eo an daore-se ha tennet e vo kuit eus ar stumm 3.0. Grit gant ${newMethod} e plas." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["ia"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Le responsa a un requesta non esseva maneate: ${statusText}", │ │ │ │ │ + Permalink: "Permaligamine", │ │ │ │ │ + Overlays: "Superpositiones", │ │ │ │ │ + "Base Layer": "Strato de base", │ │ │ │ │ + noFID: "Non pote actualisar un elemento sin FID.", │ │ │ │ │ + browserNotSupported: "Tu navigator non supporta le rendition de vectores. Le renditores actualmente supportate es:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "Le proprietate minZoomLevel es solmente pro uso con le stratos descendente de FixedZoomLevels. Le facto que iste strato WFS verifica minZoomLevel es un reliquia del passato. Nonobstante, si nos lo remove immediatemente, nos pote rumper applicationes a base de OL que depende de illo. Ergo nos lo declara obsolete; le verification de minZoomLevel in basso essera removite in version 3.0. Per favor usa in su loco le configuration de resolutiones min/max como describite a: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "Transaction WFS: SUCCESSO ${response}", │ │ │ │ │ + commitFailed: "Transaction WFS: FALLEVA ${response}", │ │ │ │ │ + googleWarning: "Le strato Google non poteva esser cargate correctemente.<br><br>Pro disfacer te de iste message, selige un nove BaseLayer in le selector de strato in alto a dextra.<br><br>Multo probabilemente, isto es proque le script del libreria de Google Maps non esseva includite o non contine le clave API correcte pro tu sito.<br><br>Disveloppatores: Pro adjuta de corriger isto, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>clicca hic</a", │ │ │ │ │ + getLayerWarning: "Le strato ${layerType} non poteva esser cargate correctemente.<br><br>Pro disfacer te de iste message, selige un nove BaseLayer in le selector de strato in alto a dextra.<br><br>Multo probabilemente, isto es proque le script del libreria de ${layerLib} non esseva correctemente includite.<br><br>Disveloppatores: Pro adjuta de corriger isto, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>clicca hic</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Scala = 1 : ${scaleDenom}", │ │ │ │ │ + W: "W", │ │ │ │ │ + E: "E", │ │ │ │ │ + N: "N", │ │ │ │ │ + S: "S", │ │ │ │ │ + reprojectDeprecated: "Tu usa le option 'reproject' in le strato ${layerName} layer. Iste option es obsolescente: illo esseva pro poter monstrar datos super cartas de base commercial, ma iste functionalitate pote ora esser attingite con le uso de Spherical Mercator. Ulterior information es disponibile a http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Iste methodo ha essite declarate obsolescente e essera removite in version 3.0. Per favor usa ${newMethod} in su loco." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["fr"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Requête non gérée, retournant ${statusText}", │ │ │ │ │ + Permalink: "Permalien", │ │ │ │ │ + Overlays: "Calques", │ │ │ │ │ + "Base Layer": "Calque de base", │ │ │ │ │ + noFID: "Impossible de mettre à jour un objet sans identifiant (fid).", │ │ │ │ │ + browserNotSupported: "Votre navigateur ne supporte pas le rendu vectoriel. Les renderers actuellement supportés sont : \n${renderers}", │ │ │ │ │ + minZoomLevelError: "La propriété minZoomLevel doit seulement être utilisée pour des couches FixedZoomLevels-descendent. Le fait que cette couche WFS vérifie la présence de minZoomLevel est une relique du passé. Nous ne pouvons toutefois la supprimer sans casser des applications qui pourraient en dépendre. C'est pourquoi nous la déprécions -- la vérification du minZoomLevel sera supprimée en version 3.0. A la place, merci d'utiliser les paramètres de résolutions min/max tel que décrit sur : http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "Transaction WFS : SUCCES ${response}", │ │ │ │ │ + commitFailed: "Transaction WFS : ECHEC ${response}", │ │ │ │ │ + googleWarning: "La couche Google n'a pas été en mesure de se charger correctement.<br><br>Pour supprimer ce message, choisissez une nouvelle BaseLayer dans le sélecteur de couche en haut à droite.<br><br>Cela est possiblement causé par la non-inclusion de la librairie Google Maps, ou alors parce que la clé de l'API ne correspond pas à votre site.<br><br>Développeurs : pour savoir comment corriger ceci, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>cliquez ici</a>", │ │ │ │ │ + getLayerWarning: "La couche ${layerType} n'est pas en mesure de se charger correctement.<br><br>Pour supprimer ce message, choisissez une nouvelle BaseLayer dans le sélecteur de couche en haut à droite.<br><br>Cela est possiblement causé par la non-inclusion de la librairie ${layerLib}.<br><br>Développeurs : pour savoir comment corriger ceci, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>cliquez ici</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Echelle ~ 1 : ${scaleDenom}", │ │ │ │ │ + W: "O", │ │ │ │ │ + E: "E", │ │ │ │ │ + N: "N", │ │ │ │ │ + S: "S", │ │ │ │ │ + reprojectDeprecated: "Vous utilisez l'option 'reproject' sur la couche ${layerName}. Cette option est dépréciée : Son usage permettait d'afficher des données au dessus de couches raster commerciales.Cette fonctionalité est maintenant supportée en utilisant le support de la projection Mercator Sphérique. Plus d'information est disponible sur http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Cette méthode est dépréciée, et sera supprimée à la version 3.0. Merci d'utiliser ${newMethod} à la place." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["pl"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Nieobsługiwane żądanie zwróciło ${statusText}", │ │ │ │ │ + Permalink: "Permalink", │ │ │ │ │ + Overlays: "Nakładki", │ │ │ │ │ + "Base Layer": "Warstwa podstawowa", │ │ │ │ │ + noFID: "Nie można zaktualizować funkcji, dla których nie ma FID.", │ │ │ │ │ + browserNotSupported: "Twoja przeglądarka nie obsługuje renderowania wektorów. Obecnie obsługiwane renderowanie to:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "Właściwość minZoomLevel jest przeznaczona tylko do użytku " + "z warstwami FixedZoomLevels-descendent." + "Warstwa wfs, która sprawdza minZoomLevel jest reliktem przeszłości." + "Nie możemy jej jednak usunąc bez mozliwości łamania OL aplikacji, " + "które mogą być od niej zależne. " + "Dlatego jesteśmy za deprecjację -- minZoomLevel " + "zostanie usunięta w wersji 3.0. W zamian prosze użyj " + "min/max rozdzielczości w sposób opisany tutaj: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "Transakcja WFS: SUKCES ${response}", │ │ │ │ │ + commitFailed: "Transakcja WFS: FAILED ${response}", │ │ │ │ │ + googleWarning: "Warstwa Google nie był w stanie załadować się poprawnie.<br><br>" + "Aby pozbyć się tej wiadomości, wybierz nową Warstwe podstawową " + "w przełączniku warstw w górnym prawym rogu mapy.<br><br>" + "Najprawdopodobniej jest to spowodowane tym, że biblioteka Google Maps " + "nie jest załadowana, lub nie zawiera poprawnego klucza do API dla twojej strony<br><br>" + "Programisto: Aby uzyskać pomoc , " + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>kliknij tutaj</a>", │ │ │ │ │ + getLayerWarning: "Warstwa ${layerType} nie mogła zostać załadowana poprawnie.<br><br>" + "Aby pozbyć się tej wiadomości, wybierz nową Warstwe podstawową " + "w przełączniku warstw w górnym prawym rogu mapy.<br><br>" + "Najprawdopodobniej jest to spowodowane tym, że biblioteka ${layerLib} " + "nie jest załadowana, lub może(o ile biblioteka tego wymaga) " + "byc potrzebny klucza do API dla twojej strony<br><br>" + "Programisto: Aby uzyskać pomoc , " + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>kliknij tutaj</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Skala = 1 : ${scaleDenom}", │ │ │ │ │ + W: "ZACH", │ │ │ │ │ + E: "WSCH", │ │ │ │ │ + N: "PN", │ │ │ │ │ + S: "PD", │ │ │ │ │ + Graticule: "Siatka", │ │ │ │ │ + reprojectDeprecated: "w warstwie ${layerName} używasz opcji 'reproject'. " + "Ta opcja jest przestarzała: " + "jej zastosowanie został zaprojektowany, aby wspierać wyświetlania danych przez komercyjne mapy, " + "jednak obecnie ta funkcjonalność powinien zostać osiągnięty za pomocą Spherical Mercator " + "its use was designed to support displaying data over commercial. Więcje informacji na ten temat możesz znaleźć na stronie " + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Ta metoda jest przestarzała i będzie usunięta od wersji 3.0. " + "W zamian użyj ${newMethod}." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["de"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Unbehandelte Anfragerückmeldung ${statusText}", │ │ │ │ │ + Permalink: "Permalink", │ │ │ │ │ + Overlays: "Overlays", │ │ │ │ │ + "Base Layer": "Grundkarte", │ │ │ │ │ + noFID: "Ein Feature, für das keine FID existiert, kann nicht aktualisiert werden.", │ │ │ │ │ + browserNotSupported: "Ihr Browser unterstützt keine Vektordarstellung. Aktuell unterstützte Renderer:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "Die <code>minZoomLevel</code>-Eigenschaft ist nur für die Verwendung mit <code>FixedZoomLevels</code>-untergeordneten Layers vorgesehen. Das dieser <tt>wfs</tt>-Layer die <code>minZoomLevel</code>-Eigenschaft überprüft ist ein Relikt der Vergangenheit. Wir können diese Überprüfung nicht entfernen, ohne das OL basierende Applikationen nicht mehr funktionieren. Daher markieren wir es als veraltet - die <code>minZoomLevel</code>-Überprüfung wird in Version 3.0 entfernt werden. Bitte verwenden Sie stattdessen die Min-/Max-Lösung, wie sie unter http://trac.openlayers.org/wiki/SettingZoomLevels beschrieben ist.", │ │ │ │ │ + commitSuccess: "WFS-Transaktion: Erfolgreich ${response}", │ │ │ │ │ + commitFailed: "WFS-Transaktion: Fehlgeschlagen ${response}", │ │ │ │ │ + googleWarning: "Der Google-Layer konnte nicht korrekt geladen werden.<br><br>Um diese Meldung nicht mehr zu erhalten, wählen Sie einen anderen Hintergrundlayer aus dem LayerSwitcher in der rechten oberen Ecke.<br><br>Sehr wahrscheinlich tritt dieser Fehler auf, weil das Skript der Google-Maps-Bibliothek nicht eingebunden wurde oder keinen gültigen API-Schlüssel für Ihre URL enthält.<br><br>Entwickler: Besuche <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>das Wiki</a> für Hilfe zum korrekten Einbinden des Google-Layers", │ │ │ │ │ + getLayerWarning: "Der ${layerType}-Layer konnte nicht korrekt geladen werden.<br><br>Um diese Meldung nicht mehr zu erhalten, wählen Sie einen anderen Hintergrundlayer aus dem LayerSwitcher in der rechten oberen Ecke.<br><br>Sehr wahrscheinlich tritt dieser Fehler auf, weil das Skript der '${layerLib}'-Bibliothek nicht eingebunden wurde.<br><br>Entwickler: Besuche <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>das Wiki</a> für Hilfe zum korrekten Einbinden von Layern", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Maßstab = 1 : ${scaleDenom}", │ │ │ │ │ + W: "W", │ │ │ │ │ + E: "O", │ │ │ │ │ + N: "N", │ │ │ │ │ + S: "S", │ │ │ │ │ + reprojectDeprecated: "Sie verwenden die „Reproject“-Option des Layers ${layerName}. Diese Option ist veraltet: Sie wurde entwickelt um die Anzeige von Daten auf kommerziellen Basiskarten zu unterstützen, aber diese Funktion sollte jetzt durch Unterstützung der „Spherical Mercator“ erreicht werden. Weitere Informationen sind unter http://trac.openlayers.org/wiki/SphericalMercator verfügbar.", │ │ │ │ │ + methodDeprecated: "Die Methode ist veraltet und wird in 3.0 entfernt. Bitte verwende stattdessen ${newMethod}." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["gl"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Solicitude non xerada; a resposta foi: ${statusText}", │ │ │ │ │ + Permalink: "Ligazón permanente", │ │ │ │ │ + Overlays: "Capas superpostas", │ │ │ │ │ + "Base Layer": "Capa base", │ │ │ │ │ + noFID: "Non se pode actualizar a funcionalidade para a que non hai FID.", │ │ │ │ │ + browserNotSupported: "O seu navegador non soporta a renderización de vectores. Os renderizadores soportados actualmente son:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "A propiedade minZoomLevel é só para uso conxuntamente coas capas FixedZoomLevels-descendent. O feito de que esa capa wfs verifique o minZoomLevel é unha reliquia do pasado. Non podemos, con todo, eliminala sen a posibilidade de non romper as aplicacións baseadas en OL que poidan depender dela. Por iso a estamos deixando obsoleta (a comprobación minZoomLevel de embaixo será eliminada na versión 3.0). Por favor, no canto diso use o axuste de resolución mín/máx tal e como está descrito aquí: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "Transacción WFS: ÉXITO ${response}", │ │ │ │ │ + commitFailed: "Transacción WFS: FALLIDA ${response}", │ │ │ │ │ + googleWarning: "A capa do Google non puido cargarse correctamente.<br><br>Para evitar esta mensaxe, escolla unha nova capa base no seleccionador de capas na marxe superior dereita.<br><br>Probablemente, isto acontece porque a escritura da libraría do Google Maps ou ben non foi incluída ou ben non contén a clave API correcta para o seu sitio.<br><br>Desenvolvedores: para axudar a facer funcionar isto correctamente, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>premede aquí</a>", │ │ │ │ │ + getLayerWarning: "A capa ${layerType} foi incapaz de cargarse correctamente.<br><br>Para evitar esta mensaxe, escolla unha nova capa base no seleccionador de capas na marxe superior dereita.<br><br>Probablemente, isto acontece porque a escritura da libraría ${layerLib} non foi ben incluída.<br><br>Desenvolvedores: para axudar a facer funcionar isto correctamente, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>premede aquí</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Escala = 1 : ${scaleDenom}", │ │ │ │ │ + W: "O", │ │ │ │ │ + E: "L", │ │ │ │ │ + N: "N", │ │ │ │ │ + S: "S", │ │ │ │ │ + reprojectDeprecated: 'Está usando a opción "reproject" na capa ${layerName}. Esta opción está obsoleta: o seu uso foi deseñado para a visualización de datos sobre mapas base comerciais, pero esta funcionalidade debera agora ser obtida utilizando a proxección Spherical Mercator. Hai dispoñible máis información en http://trac.openlayers.org/wiki/SphericalMercator.', │ │ │ │ │ + methodDeprecated: "Este método está obsoleto e será eliminado na versión 3.0. Por favor, no canto deste use ${newMethod}." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang.es = { │ │ │ │ │ + unhandledRequest: "Respuesta a petición no gestionada ${statusText}", │ │ │ │ │ + Permalink: "Enlace permanente", │ │ │ │ │ + Overlays: "Capas superpuestas", │ │ │ │ │ + "Base Layer": "Capa Base", │ │ │ │ │ + noFID: "No se puede actualizar un elemento para el que no existe FID.", │ │ │ │ │ + browserNotSupported: "Su navegador no soporta renderización vectorial. Los renderizadores soportados actualmente son:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "La propiedad minZoomLevel debe sólo utilizarse " + "con las capas que tienen FixedZoomLevels. El hecho de que " + "una capa wfs compruebe minZoomLevel es una reliquia del " + "pasado. Sin embargo, no podemos eliminarla sin discontinuar " + "probablemente las aplicaciones OL que puedan depender de ello. " + "Así pues estamos haciéndolo obsoleto --la comprobación " + "minZoomLevel se eliminará en la versión 3.0. Utilice el ajuste " + "de resolution min/max en su lugar, tal como se describe aquí: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "Transacción WFS: ÉXITO ${response}", │ │ │ │ │ + commitFailed: "Transacción WFS: FALLÓ ${response}", │ │ │ │ │ + googleWarning: "La capa Google no pudo ser cargada correctamente.<br><br>" + "Para evitar este mensaje, seleccione una nueva Capa Base " + "en el selector de capas en la esquina superior derecha.<br><br>" + "Probablemente, esto se debe a que el script de la biblioteca de " + "Google Maps no fue correctamente incluido en su página, o no " + "contiene la clave del API correcta para su sitio.<br><br>" + "Desarrolladores: Para ayudar a hacer funcionar esto correctamente, " + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>haga clic aquí</a>", │ │ │ │ │ + getLayerWarning: "La capa ${layerType} no pudo ser cargada correctamente.<br><br>" + "Para evitar este mensaje, seleccione una nueva Capa Base " + "en el selector de capas en la esquina superior derecha.<br><br>" + "Probablemente, esto se debe a que el script de " + "la biblioteca ${layerLib} " + "no fue correctamente incluido en su página.<br><br>" + "Desarrolladores: Para ayudar a hacer funcionar esto correctamente, " + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>haga clic aquí</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Escala = 1 : ${scaleDenom}", │ │ │ │ │ + W: "O", │ │ │ │ │ + E: "E", │ │ │ │ │ + N: "N", │ │ │ │ │ + S: "S", │ │ │ │ │ + Graticule: "Retícula", │ │ │ │ │ + reprojectDeprecated: "Está usando la opción 'reproject' en la capa " + "${layerName}. Esta opción es obsoleta: su uso fue diseñado " + "para soportar la visualización de datos sobre mapas base comerciales, " + "pero ahora esa funcionalidad debería conseguirse mediante el soporte " + "de la proyección Spherical Mercator. Más información disponible en " + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Este método es obsoleto y se eliminará en la versión 3.0. " + "Por favor utilice el método ${newMethod} en su lugar.", │ │ │ │ │ + end: "" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Lang["bg"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Permalink: "Постоянна препратка", │ │ │ │ │ + "Base Layer": "Основен слой", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Мащаб = 1 : ${scaleDenom}", │ │ │ │ │ + methodDeprecated: "Този метод е остарял и ще бъде премахват в 3.0. Вместо него използвайте ${newMethod}." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["fur"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Permalink: "Leam Permanent", │ │ │ │ │ + Overlays: "Livei parsore", │ │ │ │ │ + "Base Layer": "Livel di base", │ │ │ │ │ + browserNotSupported: "Il to sgarfadôr nol supuarte la renderizazion vetoriâl. Al moment a son supuartâts:\n${renderers}", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Scjale = 1 : ${scaleDenom}", │ │ │ │ │ + W: "O", │ │ │ │ │ + E: "E", │ │ │ │ │ + N: "N", │ │ │ │ │ + S: "S" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["is"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Permalink: "Varanlegur tengill", │ │ │ │ │ + Overlays: "Þekjur", │ │ │ │ │ + "Base Layer": "Grunnlag", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Skali = 1 : ${scaleDenom}", │ │ │ │ │ + methodDeprecated: "Þetta fall hefur verið úrelt og verður fjarlægt í 3.0. Notaðu ${newMethod} í staðin." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["gsw"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Nit behandleti Aafrogsruckmäldig ${statusText}", │ │ │ │ │ + Permalink: "Permalink", │ │ │ │ │ + Overlays: "Iberlagerige", │ │ │ │ │ + "Base Layer": "Grundcharte", │ │ │ │ │ + noFID: "E Feature, wu s kei FID derfir git, cha nit aktualisiert wäre.", │ │ │ │ │ + browserNotSupported: "Dyy Browser unterstitzt kei Vektordarstellig. Aktuäll unterstitzti Renderer:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "D minZoomLevel-Eigeschaft isch nume dänk fir d Layer, wu vu dr FixedZoomLevels abstamme. Ass dää wfs-Layer minZoomLevel prieft, scih e Relikt us dr Vergangeheit. Mir chenne s aber nit ändere ohni OL_basierti Aawändige villicht kaputt gehn, wu dervu abhänge. Us däm Grund het die Funktion d Eigeschaft 'deprecated' iberchuu. D minZoomLevel-Priefig unte wird in dr Version 3.0 usegnuu. Bitte verwänd statt däm e min/max-Uflesig wie s do bschriben isch: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "WFS-Transaktion: ERFOLGRYCH ${response}", │ │ │ │ │ + commitFailed: "WFS-Transaktion: FÄHLGSCHLAA ${response}", │ │ │ │ │ + googleWarning: "Dr Google-Layer het nit korräkt chenne glade wäre.<br><br>Go die Mäldig nimi z kriege, wehl e andere Hintergrundlayer us em LayerSwitcher im rächte obere Ecke.<br><br>Dää Fähler git s seli hyfig, wel s Skript vu dr Google-Maps-Bibliothek nit yybunde woren isch oder wel s kei giltige API-Schlissel fir Dyy URL din het.<br><br>Entwickler: Fir Hilf zum korräkte Yybinde vum Google-Layer <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>doo drucke</a>", │ │ │ │ │ + getLayerWarning: "Dr ${layerType}-Layer het nit korräkt chenne glade wäre.<br><br>Go die Mäldig nimi z kriege, wehl e andere Hintergrundlayer us em LayerSwitcher im rächte obere Ecke.<br><br>Dää Fähler git s seli hyfig, wel s Skript vu dr '${layerLib}'-Bibliothek nit yybunde woren isch oder wel s kei giltige API-Schlissel fir Dyy URL din het.<br><br>Entwickler: Fir Hilf zum korräkte Yybinde vu Layer <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>doo drucke</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Maßstab = 1 : ${scaleDenom}", │ │ │ │ │ + W: "W", │ │ │ │ │ + E: "O", │ │ │ │ │ + N: "N", │ │ │ │ │ + S: "S", │ │ │ │ │ + reprojectDeprecated: "Du bruchsch d 'reproject'-Option bim ${layerName}-Layer. Die Option isch nimi giltig: si isch aagleit wore go Date iber kommerziälli Grundcharte lege, aber des sott mer jetz mache mit dr Unterstitzig vu Spherical Mercator. Meh Informatione git s uf http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Die Methode isch veraltet un wird us dr Version 3.0 usegnuu. Bitte verwäbnd statt däm ${newMethod}." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["ja"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "未処理の要求は ${statusText} を返します", │ │ │ │ │ + Permalink: "パーマリンク", │ │ │ │ │ + Overlays: "オーバーレイ", │ │ │ │ │ + "Base Layer": "基底レイヤー", │ │ │ │ │ + noFID: "FID のない地物は更新できません。", │ │ │ │ │ + browserNotSupported: "あなたのブラウザはベクターグラフィックスの描写に対応していません。現時点で対応しているソフトウェアは以下のものです。\n${renderers}", │ │ │ │ │ + minZoomLevelError: "minZoomLevel プロパティは FixedZoomLevels を継承するレイヤーでの使用のみを想定しています。この minZoomLevel に対する WFS レイヤーの検査は歴史的なものです。しかしながら、この検査を除去するとそれに依存する OpenLayers ベースのアプリケーションを破壊してしまう可能性があります。よって廃止が予定されており、この minZoomLevel 検査はバージョン3.0で除去されます。代わりに、http://trac.openlayers.org/wiki/SettingZoomLevels で解説されている、最小および最大解像度設定を使用してください。", │ │ │ │ │ + commitSuccess: "WFS トランザクション: 成功 ${response}", │ │ │ │ │ + commitFailed: "WFS トランザクション: 失敗 ${response}", │ │ │ │ │ + googleWarning: "Google レイヤーが正しく読み込みを行えませんでした。<br><br>このメッセージを消すには、右上の隅にあるレイヤー切り替え部分で新しい基底レイヤーを選んでください。<br><br>おそらく、これは Google マップ用ライブラリのスクリプトが組み込まれていないか、あなたのサイトに対応する正しい API キーが設定されていないためです。<br><br>開発者の方へ: 正しい動作をさせるために<a href='http://trac.openlayers.org/wiki/Google' target='_blank'>こちらのウィキ</a>を参照してください。", │ │ │ │ │ + getLayerWarning: "${layerType} レイヤーが正しく読み込みを行えませんでした。<br><br>このメッセージを消すには、右上の隅にあるレイヤー切り替え部分で新しい基底レイヤーを選んでください。<br><br>おそらく、これは ${layerLib} ライブラリのスクリプトが正しく組み込まれていないためです。<br><br>開発者の方へ: 正しい動作をさせるために<a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>こちらのウィキ</a>を参照してください。", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "縮尺 = 1 : ${scaleDenom}", │ │ │ │ │ + W: "西", │ │ │ │ │ + E: "東", │ │ │ │ │ + N: "北", │ │ │ │ │ + S: "南", │ │ │ │ │ + reprojectDeprecated: "あなたは「${layerName}」レイヤーで reproject オプションを使っています。このオプションは商用の基底地図上に情報を表示する目的で設計されましたが、現在ではその機能は Spherical Mercator サポートを利用して実現されており、このオプションの使用は非推奨です。追加の情報は http://trac.openlayers.org/wiki/SphericalMercator で入手できます。", │ │ │ │ │ + methodDeprecated: "このメソッドは廃止が予定されており、バージョン3.0で除去されます。代わりに ${newMethod} を使用してください。" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["km"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Permalink: "តំណភ្ជាប់អចិន្ត្រៃយ៍", │ │ │ │ │ + "Base Layer": "ស្រទាប់បាត​", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "មាត្រដ្ឋាន = ១ ៖ ${scaleDenom}" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["sv"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Ej hanterad fråga retur ${statusText}", │ │ │ │ │ + Permalink: "Permalänk", │ │ │ │ │ + Overlays: "Kartlager", │ │ │ │ │ + "Base Layer": "Bakgrundskarta", │ │ │ │ │ + noFID: "Kan ej uppdatera feature (objekt) för vilket FID saknas.", │ │ │ │ │ + browserNotSupported: "Din webbläsare stöder inte vektorvisning. För närvarande stöds följande visning:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "Egenskapen minZoomLevel är endast avsedd att användas med lager med FixedZoomLevels. Att detta WFS-lager kontrollerar minZoomLevel är en relik från äldre versioner. Vi kan dock inte ta bort det utan att riskera att OL-baserade tillämpningar som använder detta slutar fungera. Därför är det satt som deprecated, minZoomLevel kommer att tas bort i version 3.0. Använd i stället inställning av min/max resolution som beskrivs här: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "WFS-transaktion: LYCKADES ${response}", │ │ │ │ │ + commitFailed: "WFS-transaktion: MISSLYCKADES ${response}", │ │ │ │ │ + googleWarning: "Google-lagret kunde inte laddas korrekt.<br><br>För att slippa detta meddelande, välj en annan bakgrundskarta i lagerväljaren i övre högra hörnet.<br><br>Sannolikt beror felet på att Google Maps-biblioteket inte är inkluderat på webbsidan eller på att sidan inte anger korrekt API-nyckel för webbplatsen.<br><br>Utvecklare: hjälp för att åtgärda detta, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>klicka här</a>.", │ │ │ │ │ + getLayerWarning: "${layerType}-lagret kunde inte laddas korrekt.<br><br>För att slippa detta meddelande, välj en annan bakgrundskarta i lagerväljaren i övre högra hörnet.<br><br>Sannolikt beror felet på att ${layerLib}-biblioteket inte är inkluderat på webbsidan.<br><br>Utvecklare: hjälp för att åtgärda detta, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>klicka här</a>.", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "<strong>Skala</strong> 1 : ${scaleDenom}", │ │ │ │ │ + reprojectDeprecated: "Du använder inställningen 'reproject' på lagret ${layerName}. Denna inställning markerad som deprecated: den var avsedd att användas för att stödja visning av kartdata på kommersiella bakgrundskartor, men nu bör man i stället använda Spherical Mercator-stöd för den funktionaliteten. Mer information finns på http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Denna metod är markerad som deprecated och kommer att tas bort i 3.0. Använd ${newMethod} i stället." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["zh-CN"] = { │ │ │ │ │ + unhandledRequest: "未处理的请求,返回值为 ${statusText}", │ │ │ │ │ + Permalink: "永久链接", │ │ │ │ │ + Overlays: "叠加层", │ │ │ │ │ + "Base Layer": "基础图层", │ │ │ │ │ + noFID: "无法更新feature,缺少FID。", │ │ │ │ │ + browserNotSupported: "你使用的浏览器不支持矢量渲染。当前支持的渲染方式包括:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "minZoomLevel属性仅适合用于" + "使用了固定缩放级别的图层。这个 " + "wfs 图层检查 minZoomLevel 是过去遗留下来的。" + "然而,我们不能移除它," + "而破坏依赖于它的基于OL的应用程序。" + "因此,我们废除了它 -- minZoomLevel " + "将会在3.0中被移除。请改用 " + "min/max resolution 设置,参考:" + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "WFS Transaction: 成功。 ${response}", │ │ │ │ │ + commitFailed: "WFS Transaction: 失败。 ${response}", │ │ │ │ │ + googleWarning: "Google图层不能正确加载。<br><br>" + "要消除这个信息,请在右上角的" + "图层控制面板中选择其他的基础图层。<br><br>" + "这种情况很可能是没有正确的包含Google地图脚本库," + "或者是没有包含在你的站点上" + "使用的正确的Google Maps API密匙。<br><br>" + "开发者:获取使其正确工作的帮助信息," + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>点击这里</a>", │ │ │ │ │ + getLayerWarning: "${layerType} 图层不能正确加载。<br><br>" + "要消除这个信息,请在右上角的" + "图层控制面板中选择其他的基础图层。<br><br>" + "这种情况很可能是没有正确的包含" + "${layerLib} 脚本库。<br><br>" + "开发者:获取使其正确工作的帮助信息," + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>点击这里</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "比例尺 = 1 : ${scaleDenom}", │ │ │ │ │ + reprojectDeprecated: "你正在使用 ${layerName} 图层上的'reproject'选项。" + "这个选项已经不再使用:" + "它是被设计用来支持显示商业的地图数据," + "不过现在该功能可以通过使用Spherical Mercator来实现。" + "更多信息可以参阅" + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "该方法已经不再被支持,并且将在3.0中被移除。" + "请使用 ${newMethod} 方法来替代。", │ │ │ │ │ + end: "" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Lang["fi"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Permalink: "Ikilinkki", │ │ │ │ │ + Overlays: "Kerrokset", │ │ │ │ │ + "Base Layer": "Peruskerros", │ │ │ │ │ + W: "L", │ │ │ │ │ + E: "I", │ │ │ │ │ + N: "P", │ │ │ │ │ + S: "E" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["cs-CZ"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Nezpracovaná návratová hodnota ${statusText}", │ │ │ │ │ + Permalink: "Trvalý odkaz", │ │ │ │ │ + Overlays: "Překryvné vrstvy", │ │ │ │ │ + "Base Layer": "Podkladové vrstvy", │ │ │ │ │ + noFID: "Nelze aktualizovat prvek, pro který neexistuje FID.", │ │ │ │ │ + browserNotSupported: "Váš prohlížeč nepodporuje vykreslování vektorů. Momentálně podporované nástroje jsou::\n${renderers}", │ │ │ │ │ + minZoomLevelError: "Vlastnost minZoomLevel by se měla používat pouze s potomky FixedZoomLevels vrstvami. To znamená, že vrstva wfs kontroluje, zda-li minZoomLevel není zbytek z minulosti.Nelze to ovšem vyjmout bez možnosti, že bychom rozbili aplikace postavené na OL, které by na tom mohly záviset. Proto tuto vlastnost nedoporučujeme používat -- kontrola minZoomLevel bude odstraněna ve verzi 3.0. Použijte prosím raději nastavení min/max podle příkaldu popsaného na: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "WFS Transaction: ÚSPĚCH ${response}", │ │ │ │ │ + commitFailed: "WFS Transaction: CHYBA ${response}", │ │ │ │ │ + googleWarning: "Nepodařilo se správně načíst vrstvu Google.<br><br>Abyste se zbavili této zprávy, zvolte jinou základní vrstvu v přepínači vrstev.<br><br>To se většinou stává, pokud nebyl načten skript, nebo neobsahuje správný klíč pro API pro tuto stránku.<br><br>Vývojáři: Pro pomoc, aby tohle fungovalo , <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>klikněte sem</a>", │ │ │ │ │ + getLayerWarning: "The ${layerType} Layer was unable to load correctly.<br><br>To get rid of this message, select a new BaseLayer in the layer switcher in the upper-right corner.<br><br>Most likely, this is because the ${layerLib} library script was either not correctly included.<br><br>Developers: For help getting this working correctly, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>click here</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Měřítko = 1 : ${scaleDenom}", │ │ │ │ │ + reprojectDeprecated: "Použil jste volbu 'reproject' ve vrstvě ${layerName}. Tato volba není doporučená: byla zde proto, aby bylo možno zobrazovat data z okomerčních serverů, ale tato funkce je nyní zajištěna pomocí podpory Spherical Mercator. Více informací naleznete na http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Tato metoda je zavržená a bude ve verzi 3.0 odstraněna. Prosím, použijte raději ${newMethod}." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["el"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Κλίμακα ~ 1 : ${scaleDenom}" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["be-tarask"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Неапрацаваны вынік запыту ${statusText}", │ │ │ │ │ + Permalink: "Сталая спасылка", │ │ │ │ │ + Overlays: "Слаі", │ │ │ │ │ + "Base Layer": "Базавы слой", │ │ │ │ │ + noFID: "Немагчыма абнавіць магчымасьць, для якога не існуе FID.", │ │ │ │ │ + browserNotSupported: "Ваш браўзэр не падтрымлівае вэктарную графіку. У цяперашні момант падтрымліваюцца: ${renderers}", │ │ │ │ │ + minZoomLevelError: "Уласьцівасьць minZoomLevel прызначана толькі для выкарыстаньня са слаямі вытворнымі ад FixedZoomLevels. Тое, што гэты wfs-слой правяраецца на minZoomLevel — рэха прошлага. Але мы ня можам выдаліць гэтую магчымасьць, таму што ад яе залежаць некаторыя заснаваныя на OL дастасаваньні. Тым ня менш, праверка minZoomLevel будзе выдаленая ў вэрсіі 3.0. Калі ласка, выкарыстоўваеце замест яе ўстаноўкі мінімальнага/максымальнага памераў, як апісана тут: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "WFS-транзакцыя: ПОСЬПЕХ ${response}", │ │ │ │ │ + commitFailed: "WFS-транзакцыя: ПАМЫЛКА ${response}", │ │ │ │ │ + googleWarning: "Не атрымалася загрузіць слой Google. <br><br>Каб пазбавіцца гэтага паведамленьня, выберыце новы базавы слой у сьпісе ў верхнім правым куце.<br><br> Хутчэй за ўсё, прычына ў тым, што скрыпт бібліятэкі Google Maps ня быў уключаныя альбо не ўтрымлівае слушны API-ключ для Вашага сайта.<br><br>Распрацоўшчыкам: Для таго, каб даведацца як зрабіць так, каб усё працавала, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>націсьніце тут</a>", │ │ │ │ │ + getLayerWarning: "Немагчыма загрузіць слой ${layerType}.<br><br>Каб пазбавіцца гэтага паведамленьня, выберыце новы базавы слой у сьпісе ў верхнім правым куце.<br><br>Хутчэй за ўсё, прычына ў тым, што скрыпт бібліятэкі ${layerLib} ня быў слушна ўключаны.<br><br>Распрацоўшчыкам: Для таго, каб даведацца як зрабіць так, каб усё працавала, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>націсьніце тут</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Маштаб = 1 : ${scaleDenom}", │ │ │ │ │ + W: "З", │ │ │ │ │ + E: "У", │ │ │ │ │ + N: "Пн", │ │ │ │ │ + S: "Пд", │ │ │ │ │ + reprojectDeprecated: "Вы выкарыстоўваеце ўстаноўку 'reproject' для слоя ${layerName}. Гэтая ўстаноўка зьяўляецца састарэлай: яна выкарыстоўвалася для падтрымкі паказу зьвестак на камэрцыйных базавых мапах, але гэта функцыя цяпер рэалізаваная ў убудаванай падтрымцы сфэрычнай праекцыі Мэркатара. Дадатковая інфармацыя ёсьць на http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Гэты мэтад састарэлы і будзе выдалены ў вэрсіі 3.0. Калі ласка, замест яго выкарыстоўвайце ${newMethod}." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["vi"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Không xử lý được phản hồi ${statusText} cho yêu cầu", │ │ │ │ │ + Permalink: "Liên kết thường trực", │ │ │ │ │ + Overlays: "Lấp bản đồ", │ │ │ │ │ + "Base Layer": "Lớp nền", │ │ │ │ │ + noFID: "Không thể cập nhật tính năng thiếu FID.", │ │ │ │ │ + browserNotSupported: "Trình duyệt của bạn không hỗ trợ chức năng vẽ bằng vectơ. Hiện hỗ trợ các bộ kết xuất:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "Chỉ nên sử dụng thuộc tính minZoomLevel với các lớp FixedZoomLevels-descendent. Việc lớp wfs này tìm cho minZoomLevel là di tích còn lại từ xưa. Tuy nhiên, nếu chúng tôi dời nó thì sẽ vỡ các chương trình OpenLayers mà dựa trên nó. Bởi vậy chúng tôi phản đối sử dụng nó – bước tìm cho minZoomLevel sẽ được dời vào phiên bản 3.0. Xin sử dụng thiết lập độ phân tích tối thiểu / tối đa thay thế, theo hướng dẫn này: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "Giao dịch WFS: THÀNH CÔNG ${response}", │ │ │ │ │ + commitFailed: "Giao dịch WFS: THẤT BẠI ${response}", │ │ │ │ │ + googleWarning: "Không thể tải lớp Google đúng đắn.<br><br>Để tránh thông báo này lần sau, hãy chọn BaseLayer mới dùng điều khiển chọn lớp ở góc trên phải.<br><br>Chắc script thư viện Google Maps hoặc không được bao gồm hoặc không chứa khóa API hợp với website của bạn.<br><br><a href='http://trac.openlayers.org/wiki/Google' target='_blank'>Trợ giúp về tính năng này</a> cho người phát triển.", │ │ │ │ │ + getLayerWarning: "Không thể tải lớp ${layerType} đúng đắn.<br><br>Để tránh thông báo này lần sau, hãy chọn BaseLayer mới dùng điều khiển chọn lớp ở góc trên phải.<br><br>Chắc script thư viện ${layerLib} không được bao gồm đúng kiểu.<br><br><a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>Trợ giúp về tính năng này</a> cho người phát triển.", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Tỷ lệ = 1 : ${scaleDenom}", │ │ │ │ │ + W: "T", │ │ │ │ │ + E: "Đ", │ │ │ │ │ + N: "B", │ │ │ │ │ + S: "N", │ │ │ │ │ + reprojectDeprecated: "Bạn đang áp dụng chế độ “reproject” vào lớp ${layerName}. Chế độ này đã bị phản đối: nó có mục đích hỗ trợ lấp dữ liệu trên các nền bản đồ thương mại; nên thực hiện hiệu ứng đó dùng tính năng Mercator Hình cầu. Có sẵn thêm chi tiết tại http://trac.openlayers.org/wiki/SphericalMercator .", │ │ │ │ │ + methodDeprecated: "Phương thức này đã bị phản đối và sẽ bị dời vào phiên bản 3.0. Xin hãy sử dụng ${newMethod} thay thế." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang.it = { │ │ │ │ │ + unhandledRequest: "Codice di ritorno della richiesta ${statusText}", │ │ │ │ │ + Permalink: "Permalink", │ │ │ │ │ + Overlays: "Overlays", │ │ │ │ │ + "Base Layer": "Livello base", │ │ │ │ │ + noFID: "Impossibile aggiornare un elemento grafico che non abbia il FID.", │ │ │ │ │ + browserNotSupported: "Il tuo browser non supporta il rendering vettoriale. I renderizzatore attualemnte supportati sono:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "La proprietà minZoomLevel è da utilizzare solamente " + "con livelli che abbiano FixedZoomLevels. Il fatto che " + "questo livello wfs controlli la proprietà minZoomLevel è " + "un retaggio del passato. Non possiamo comunque rimuoverla " + "senza rompere le vecchie applicazioni che dipendono su di essa." + "Quindi siamo costretti a deprecarla -- minZoomLevel " + "e sarà rimossa dalla vesione 3.0. Si prega di utilizzare i " + "settaggi di risoluzione min/max come descritto qui: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "Transazione WFS: SUCCESS ${response}", │ │ │ │ │ + commitFailed: "Transazione WFS: FAILED ${response}", │ │ │ │ │ + googleWarning: "Il livello Google non è riuscito a caricare correttamente.<br><br>" + "Per evitare questo messaggio, seleziona un nuovo BaseLayer " + "nel selettore di livelli nell'angolo in alto a destra.<br><br>" + "Più precisamente, ciò accade perchè la libreria Google Maps " + "non è stata inclusa nella pagina, oppure non contiene la " + "corretta API key per il tuo sito.<br><br>" + "Sviluppatori: Per aiuto su come farlo funzionare correttamente, " + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>clicca qui</a>", │ │ │ │ │ + getLayerWarning: "Il livello ${layerType} non è riuscito a caricare correttamente.<br><br>" + "Per evitare questo messaggio, seleziona un nuovo BaseLayer " + "nel selettore di livelli nell'angolo in alto a destra.<br><br>" + "Più precisamente, ciò accade perchè la libreria ${layerLib} " + "non è stata inclusa nella pagina.<br><br>" + "Sviluppatori: Per aiuto su come farlo funzionare correttamente, " + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>clicca qui</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Scala = 1 : ${scaleDenom}", │ │ │ │ │ + reprojectDeprecated: "Stai utilizzando l'opzione 'reproject' sul livello ${layerName}. " + "Questa opzione è deprecata: il suo utilizzo è stato introdotto per" + "supportare il disegno dei dati sopra mappe commerciali, ma tale " + "funzionalità dovrebbe essere ottenuta tramite l'utilizzo della proiezione " + "Spherical Mercator. Per maggiori informazioni consultare qui " + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Questo metodo è stato deprecato e sarà rimosso dalla versione 3.0. " + "Si prega di utilizzare il metodo ${newMethod} in alternativa.", │ │ │ │ │ + end: "" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Lang["pt-BR"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "A requisição retornou um erro não tratado: ${statusText}", │ │ │ │ │ + Permalink: "Link para essa página", │ │ │ │ │ + Overlays: "Camadas de Sobreposição", │ │ │ │ │ + "Base Layer": "Camada Base", │ │ │ │ │ + noFID: "Não é possível atualizar uma feição que não tenha um FID.", │ │ │ │ │ + browserNotSupported: "Seu navegador não suporta renderização de vetores. Os renderizadores suportados atualmente são:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "A propriedade minZoomLevel é de uso restrito das camadas descendentes de FixedZoomLevels. A verificação dessa propriedade pelas camadas wfs é um resíduo do passado. Não podemos, entretanto não é possível removê-la sem possívelmente quebrar o funcionamento de aplicações OL que possuem depência com ela. Portanto estamos tornando seu uso obsoleto -- a verificação desse atributo será removida na versão 3.0. Ao invés, use as opções de resolução min/max como descrito em: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "Transação WFS : SUCESSO ${response}", │ │ │ │ │ + commitFailed: "Transação WFS : ERRO ${response}", │ │ │ │ │ + googleWarning: "Não foi possível carregar a camada Google corretamente.<br><br>Para se livrar dessa mensagem, selecione uma nova Camada Base, na ferramenta de alternação de camadas localização do canto superior direito.<br><br>Muito provavelmente, isso foi causado porque o script da biblioteca do Google Maps não foi incluído, ou porque ele não contém a chave correta da API para o seu site.<br><br>Desenvolvedores: Para obter ajuda em solucionar esse problema <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>cliquem aqui</a>", │ │ │ │ │ + getLayerWarning: "Não foi possível carregar a camada ${layerType} corretamente.<br><br>Para se livrar dessa mensagem, selecione uma nova Camada Base, na ferramenta de alternação de camadas localização do canto superior direito.<br><br>Muito provavelmente, isso foi causado porque o script da biblioteca ${layerLib} não foi incluído corretamente.<br><br>Desenvolvedores: Para obter ajuda em solucionar esse problema <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>cliquem aqui</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Escala = 1 : ${scaleDenom}", │ │ │ │ │ + W: "O", │ │ │ │ │ + E: "L", │ │ │ │ │ + N: "N", │ │ │ │ │ + S: "S", │ │ │ │ │ + reprojectDeprecated: "Você está usando a opção 'reproject' na camada ${layerName}. Essa opção está obsoleta: seu uso foi projetado para suportar a visualização de dados sobre bases de mapas comerciais, entretanto essa funcionalidade deve agora ser alcançada usando o suporte à projeção Mercator. Mais informação está disponível em: http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Esse método está obsoleto e será removido na versão 3.0. Ao invés, por favor use ${newMethod}." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["id"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Permintaan yang tak tertangani menghasilkan ${statusText}", │ │ │ │ │ + Permalink: "Pranala permanen", │ │ │ │ │ + Overlays: "Hamparan", │ │ │ │ │ + "Base Layer": "Lapisan Dasar", │ │ │ │ │ + noFID: "Tidak dapat memperbarui fitur yang tidak memiliki FID.", │ │ │ │ │ + browserNotSupported: "Peramban Anda tidak mendukung penggambaran vektor. Penggambar yang didukung saat ini adalah:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "Properti minZoomLevel hanya ditujukan bekerja dengan lapisan FixedZoomLevels-descendent. Pengecekan minZoomLevel oleh lapisan wfs adalah peninggalan masa lalu. Kami tidak dapat menghapusnya tanpa kemungkinan merusak aplikasi berbasis OL yang mungkin bergantung padanya. Karenanya, kami menganggapnya tidak berlaku -- Cek minZoomLevel di bawah ini akan dihapus pada 3.0. Silakan gunakan penyetelan resolusi min/maks seperti dijabarkan di sini: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "WFS Transaksi: BERHASIL ${respon}", │ │ │ │ │ + commitFailed: "WFS Transaksi: GAGAL ${respon}", │ │ │ │ │ + googleWarning: "Lapisan Google tidak dapat dimuat dengan benar.<br><br>Untuk menghilangkan pesan ini, pilih suatu BaseLayer baru melalui penukar lapisan (layer switcher) di ujung kanan atas.<br><br>Kemungkinan besar ini karena pustaka skrip Google Maps tidak disertakan atau tidak mengandung kunci API yang tepat untuk situs Anda.<br><br>Pengembang: Untuk bantuan mengatasi masalah ini, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>klik di sini</a>", │ │ │ │ │ + getLayerWarning: "Lapisan ${layerType} tidak dapat dimuat dengan benar.<br><br>Untuk menghilangkan pesan ini, pilih suatu BaseLayer baru melalui penukar lapisan (layer switcher) di ujung kanan atas.<br><br>Kemungkinan besar ini karena pustaka skrip Google Maps tidak disertakan dengan benar.<br><br>Pengembang: Untuk bantuan mengatasi masalah ini, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>klik di sini</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Sekala = 1 : ${scaleDenom}", │ │ │ │ │ + W: "B", │ │ │ │ │ + E: "T", │ │ │ │ │ + N: "U", │ │ │ │ │ + S: "S", │ │ │ │ │ + reprojectDeprecated: "Anda menggunakan opsi 'reproject' pada lapisan ${layerName}. Opsi ini telah ditinggalkan: penggunaannya dirancang untuk mendukung tampilan data melalui peta dasar komersial, tapi fungsionalitas tersebut saat ini harus dilakukan dengan menggunakan dukungan Spherical Mercator. Informasi lebih lanjut tersedia di http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Metode ini telah usang dan akan dihapus di 3.0. Sebaliknya, harap gunakan ${newMethod}." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["io"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Skalo = 1 : ${scaleDenom}" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang.en = { │ │ │ │ │ + unhandledRequest: "Unhandled request return ${statusText}", │ │ │ │ │ + Permalink: "Permalink", │ │ │ │ │ + Overlays: "Overlays", │ │ │ │ │ + "Base Layer": "Base Layer", │ │ │ │ │ + noFID: "Can't update a feature for which there is no FID.", │ │ │ │ │ + browserNotSupported: "Your browser does not support vector rendering. Currently supported renderers are:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "The minZoomLevel property is only intended for use " + "with the FixedZoomLevels-descendent layers. That this " + "wfs layer checks for minZoomLevel is a relic of the" + "past. We cannot, however, remove it without possibly " + "breaking OL based applications that may depend on it." + " Therefore we are deprecating it -- the minZoomLevel " + "check below will be removed at 3.0. Please instead " + "use min/max resolution setting as described here: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "WFS Transaction: SUCCESS ${response}", │ │ │ │ │ + commitFailed: "WFS Transaction: FAILED ${response}", │ │ │ │ │ + googleWarning: "The Google Layer was unable to load correctly.<br><br>" + "To get rid of this message, select a new BaseLayer " + "in the layer switcher in the upper-right corner.<br><br>" + "Most likely, this is because the Google Maps library " + "script was either not included, or does not contain the " + "correct API key for your site.<br><br>" + "Developers: For help getting this working correctly, " + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>click here</a>", │ │ │ │ │ + getLayerWarning: "The ${layerType} Layer was unable to load correctly.<br><br>" + "To get rid of this message, select a new BaseLayer " + "in the layer switcher in the upper-right corner.<br><br>" + "Most likely, this is because the ${layerLib} library " + "script was not correctly included.<br><br>" + "Developers: For help getting this working correctly, " + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>click here</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Scale = 1 : ${scaleDenom}", │ │ │ │ │ + W: "W", │ │ │ │ │ + E: "E", │ │ │ │ │ + N: "N", │ │ │ │ │ + S: "S", │ │ │ │ │ + Graticule: "Graticule", │ │ │ │ │ + reprojectDeprecated: "You are using the 'reproject' option " + "on the ${layerName} layer. This option is deprecated: " + "its use was designed to support displaying data over commercial " + "basemaps, but that functionality should now be achieved by using " + "Spherical Mercator support. More information is available from " + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "This method has been deprecated and will be removed in 3.0. " + "Please use ${newMethod} instead.", │ │ │ │ │ + end: "" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Lang["en-CA"] = OpenLayers.Util.applyDefaults({}, OpenLayers.Lang["en"]); │ │ │ │ │ +OpenLayers.Lang["nb"] = { │ │ │ │ │ + unhandledRequest: "Ubehandlet forespørsel returnerte ${statusText}", │ │ │ │ │ + Permalink: "Kobling til denne siden", │ │ │ │ │ + Overlays: "Kartlag", │ │ │ │ │ + "Base Layer": "Bakgrunnskart", │ │ │ │ │ + noFID: "Kan ikke oppdatere et feature (et objekt) som ikke har FID.", │ │ │ │ │ + browserNotSupported: "Din nettleser støtter ikke vektortegning. Tegnemetodene som støttes er:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "Egenskapen minZoomLevel er kun ment til bruk på lag " + "basert på FixedZoomLevels. At dette wfs-laget sjekker " + "minZoomLevel er en etterlevning fra tidligere versjoner. Det kan dog ikke " + "tas bort uten å risikere at OL-baserte applikasjoner " + "slutter å virke, så det er merket som foreldet: " + "minZoomLevel i sjekken nedenfor vil fjernes i 3.0. " + "Vennligst bruk innstillingene for min/maks oppløsning " + "som er beskrevet her: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "WFS-transaksjon: LYKTES ${response}", │ │ │ │ │ + commitFailed: "WFS-transaksjon: MISLYKTES ${response}", │ │ │ │ │ + googleWarning: "Google-laget kunne ikke lastes.<br><br>" + "Bytt til et annet bakgrunnslag i lagvelgeren i " + "øvre høyre hjørne for å slippe denne meldingen.<br><br>" + "Sannsynligvis forårsakes feilen av at Google Maps-biblioteket " + "ikke er riktig inkludert på nettsiden, eller at det ikke er " + "angitt riktig API-nøkkel for nettstedet.<br><br>" + "Utviklere: For hjelp til å få dette til å virke se " + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>her</a>.", │ │ │ │ │ + getLayerWarning: "${layerType}-laget kunne ikke lastes.<br><br>" + "Bytt til et annet bakgrunnslag i lagvelgeren i " + "øvre høyre hjørne for å slippe denne meldingen.<br><br>" + "Sannsynligvis forårsakes feilen av at " + "${layerLib}-biblioteket ikke var riktig inkludert " + "på nettsiden.<br><br>" + "Utviklere: For hjelp til å få dette til å virke se " + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>her</a>.", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "<strong>Skala</strong> 1 : ${scaleDenom}", │ │ │ │ │ + reprojectDeprecated: "Du bruker innstillingen 'reproject' på laget ${layerName}. " + "Denne innstillingen er foreldet, den var ment for å støtte " + "visning av kartdata over kommersielle bakgrunnskart, men det " + "bør nå gjøres med støtten for Spherical Mercator. Mer informasjon " + "finnes på http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Denne metoden er markert som foreldet og vil bli fjernet i 3.0. " + "Vennligst bruk ${newMethod} i stedet.", │ │ │ │ │ + end: "" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Lang["no"] = OpenLayers.Lang["nb"]; │ │ │ │ │ +OpenLayers.Lang["sk"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Neobslúžené požiadavky vracajú ${statusText}", │ │ │ │ │ + Permalink: "Trvalý odkaz", │ │ │ │ │ + Overlays: "Prekrytia", │ │ │ │ │ + "Base Layer": "Základná vrstva", │ │ │ │ │ + noFID: "Nie je možné aktualizovať vlastnosť, pre ktorú neexistuje FID.", │ │ │ │ │ + browserNotSupported: "Váš prehliadač nepodporuje vykresľovanie vektorov. Momentálne podporované vykresľovače sú:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "Vlastnosť minZoomLevel je určený iba na použitie s vrstvami odvodenými od FixedZoomLevels. To, že táto wfs vrstva kontroluje minZoomLevel je pozostatok z minulosti. Nemôžeme ho však odstrániť, aby sme sa vyhli možnému porušeniu aplikácií založených na Open Layers, ktoré na tomto môže závisieť. Preto ho označujeme ako zavrhovaný - dolu uvedená kontrola minZoomLevel bude odstránená vo verzii 3.0. Použite prosím namiesto toho kontrolu min./max. rozlíšenia podľa tu uvedeného popisu: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "Transakcia WFS: ÚSPEŠNÁ ${response}", │ │ │ │ │ + commitFailed: "Transakcia WFS: ZLYHALA ${response}", │ │ │ │ │ + googleWarning: "Vrstvu Google nebolo možné správne načítať.<br><br>Aby ste sa tejto správy zbavili vyberte novú BaseLayer v prepínači vrstiev v pravom hornom rohu.<br><br>Toto sa stalo pravdepodobne preto, že skript knižnice Google Maps buď nebol načítaný alebo neobsahuje správny kľúč API pre vašu lokalitu.<br><br>Vývojári: Tu môžete získať <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>pomoc so sfunkčnením</a>", │ │ │ │ │ + getLayerWarning: "Vrstvu ${layerType} nebolo možné správne načítať.<br><br>Aby ste sa tejto správy zbavili vyberte novú BaseLayer v prepínači vrstiev v pravom hornom rohu.<br><br>Toto sa stalo pravdepodobne preto, že skript knižnice ${layerType} buď nebol načítaný alebo neobsahuje správny kľúč API pre vašu lokalitu.<br><br>Vývojári: Tu môžete získať <a href='http://trac.openlayers.org/wiki/${layerType}' target='_blank'>pomoc so sfunkčnením</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Mierka = 1 : ${scaleDenom}", │ │ │ │ │ + reprojectDeprecated: "Používate voľby „reproject“ vrstvy ${layerType}. Táto voľba je zzavrhovaná: jej použitie bolo navrhnuté na podporu zobrazovania údajov nad komerčnými základovými mapami, ale túto funkcionalitu je teraz možné dosiahnuť pomocou Spherical Mercator. Ďalšie informácie získate na stránke http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Táto metóda je zavrhovaná a bude odstránená vo verzii 3.0. Použite prosím namiesto nej metódu ${newMethod}." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["te"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Permalink: "స్థిరలింకు", │ │ │ │ │ + W: "ప", │ │ │ │ │ + E: "తూ", │ │ │ │ │ + N: "ఉ", │ │ │ │ │ + S: "ద" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["ksh"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + unhandledRequest: "Met dä Antwoot op en Aanfrooch ham_mer nix aanjefange: ${statusText}", │ │ │ │ │ + Permalink: "Lengk op Duuer", │ │ │ │ │ + Overlays: "Drövver jelaat", │ │ │ │ │ + "Base Layer": "Jrund-Nivoh", │ │ │ │ │ + noFID: 'En Saach, woh kein <i lang="en">FID</i> för doh es, löht sesch nit ändere.', │ │ │ │ │ + browserNotSupported: "Dinge Brauser kann kein Väktore ußjävve. De Zoote Ußjaabe, di em Momang jon, sen:\n${renderers}", │ │ │ │ │ + minZoomLevelError: 'De Eijeschaff „<code lang="en">minZoomLevel</code>“ es bloß doför jedaach, dat mer se met dä Nivvohß bruch, di vun <code lang="en">FixedZoomLevels</code> affhange don. Dat dat <i lang="en">WFS</i>-Nivvoh övverhoup de Eijeschaff „<code lang="en">minZoomLevel</code>“ pröhfe deiht, es noch övveresch vun fröhjer. Mer künne dat ävver jez nit fott lohße, oohne dat mer Jevaa loufe, dat Aanwendunge vun OpenLayers nit mieh loufe, di sesch doh velleijsch noch drop am verlohße sin. Dröm sare mer, dat mer et nit mieh han welle, un de „<code lang="en">minZoomLevel</code>“-Eijeschaff weed hee vun de Version 3.0 af nit mieh jeprööf wäde. Nemm doför de Enstellung för de hühßte un de kleinßte Oplöhsung, esu wi et en http://trac.openlayers.org/wiki/SettingZoomLevels opjeschrevve es.', │ │ │ │ │ + commitSuccess: 'Dä <i lang="en">WFS</i>-Vörjang es joot jeloufe: ${response}', │ │ │ │ │ + commitFailed: 'Dä <i lang="en">WFS</i>-Vörjang es scheif jejange: ${response}', │ │ │ │ │ + googleWarning: 'Dat Nivvoh <code lang="en">Google</code> kunnt nit reschtesch jelaade wääde.<br /><br />Öm hee di Nohreesch loß ze krijje, donn en ander Jrund-Nivvoh ußsöhke, rähß bovve en de Äk.<br /><br />Wascheinlesch es dat wiel dat <i lang="en">Google-Maps</i>-Skrepp entweeder nit reschtesch enjebonge wood, udder nit dä reschtejje <i lang="en">API</i>-Schlößel för Ding Web-ßait scheke deiht.<br /><br />För Projrammierer jidd_et Hölp do_drövver, <a href="http://trac.openlayers.org/wiki/Google" target="_blank">wi mer dat aan et Loufe brengk</a>.', │ │ │ │ │ + getLayerWarning: 'Dat Nivvoh <code>${layerType}</code> kunnt nit reschtesch jelaade wääde.<br /><br />Öm hee di Nohreesch loß ze krijje, donn en ander Jrund-Nivvoh ußsöhkre, rähß bovve en de Äk.<br /><br />Wascheinlesch es dat, wiel dat Skrepp <code>${layerLib}</code> nit reschtesch enjebonge wood.<br /><br />För Projrammierer jidd_Et Hölp do_drövver, <a href="http://trac.openlayers.org/wiki/${layerLib}" target="_blank">wi mer dat aan et Loufe brengk</a>.', │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Mohßshtaab = 1 : ${scaleDenom}", │ │ │ │ │ + W: "W", │ │ │ │ │ + E: "O", │ │ │ │ │ + N: "N", │ │ │ │ │ + S: "S", │ │ │ │ │ + reprojectDeprecated: "Do bruchs de Ußwahl <code>reproject</code> op däm Nivvoh <code>${layerName}</code>. Di Ußwahl es nit mieh jähn jesinn. Se wohr doför jedaach, öm Date op jeschääfsmäßesch eruß jejovve Kaate bovve drop ze moole, wat ävver enzwesche besser met dä Öngershtözung för de ßfääresche Mäkaator Beldscher jeiht. Doh kanns De mieh drövver fenge op dä Sigg: http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Hee di Metood es nim_mih aktoäll un et weed se en dä Version 3.0 nit mieh jävve. Nemm <code>${newMethod}</code> doföör." │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Lang["da-DK"] = { │ │ │ │ │ + unhandledRequest: "En ikke håndteret forespørgsel returnerede ${statusText}", │ │ │ │ │ + Permalink: "Permalink", │ │ │ │ │ + Overlays: "Kortlag", │ │ │ │ │ + "Base Layer": "Baggrundslag", │ │ │ │ │ + noFID: "Kan ikke opdateret en feature (et objekt) der ikke har et FID.", │ │ │ │ │ + browserNotSupported: "Din browser understøtter ikke vektor visning. Følgende vektor visninger understøttes:\n${renderers}", │ │ │ │ │ + minZoomLevelError: "Egenskaben minZoomLevel er kun beregnet til brug " + "med FixedZoomLevels. At dette WFS lag kontrollerer " + "minZoomLevel egenskaben, er et levn fra en tidligere " + "version. Vi kan desværre ikke fjerne dette uden at risikere " + "at ødelægge eksisterende OL baserede programmer der " + " benytter denne funktionalitet. " + "Egenskaben bør derfor ikke anvendes, og minZoomLevel " + "kontrollen herunder vil blive fjernet i version 3.0. " + "Benyt istedet min/max opløsnings indstillingerne, som " + "er beskrevet her: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ + commitSuccess: "WFS transaktion: LYKKEDES ${response}", │ │ │ │ │ + commitFailed: "WFS transaktion: MISLYKKEDES ${response}", │ │ │ │ │ + googleWarning: "Google laget kunne ikke indlæses.<br><br>" + "For at fjerne denne besked, vælg et nyt bagrundskort i " + "lagskifteren i øverste højre hjørne.<br><br>" + "Fejlen skyldes formentlig at Google Maps bibliotekts " + "scriptet ikke er inkluderet, eller ikke indeholder den " + "korrkte API nøgle for dit site.<br><br>" + "Udviklere: For hjælp til at få dette til at fungere, " + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>klik her</a>", │ │ │ │ │ + getLayerWarning: "${layerType}-laget kunne ikke indlæses.<br><br>" + "For at fjerne denne besked, vælg et nyt bagrundskort i " + "lagskifteren i øverste højre hjørne.<br><br>" + "Fejlen skyldes formentlig at ${layerLib} bibliotekts " + "scriptet ikke er inkluderet.<br><br>" + "Udviklere: For hjælp til at få dette til at fungere, " + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>klik her</a>", │ │ │ │ │ + "Scale = 1 : ${scaleDenom}": "Målforhold = 1 : ${scaleDenom}", │ │ │ │ │ + reprojectDeprecated: "Du anvender indstillingen 'reproject' på laget ${layerName}." + "Denne indstilling bør ikke længere anvendes. Den var beregnet " + "til at vise data ovenpå kommercielle grundkort, men den funktionalitet " + "bør nu opnås ved at anvende Spherical Mercator understøttelsen. " + "Mere information er tilgængelig her: " + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ + methodDeprecated: "Denne funktion bør ikke længere anvendes, og vil blive fjernet i version 3.0. " + "Anvend venligst funktionen ${newMethod} istedet." │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Handler.Drag = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + started: false, │ │ │ │ │ + stopDown: true, │ │ │ │ │ + dragging: false, │ │ │ │ │ + last: null, │ │ │ │ │ + start: null, │ │ │ │ │ + lastMoveEvt: null, │ │ │ │ │ + oldOnselectstart: null, │ │ │ │ │ + interval: 0, │ │ │ │ │ + timeoutId: null, │ │ │ │ │ + documentDrag: false, │ │ │ │ │ + documentEvents: null, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (this.documentDrag === true) { │ │ │ │ │ + var me = this; │ │ │ │ │ + this._docMove = function(evt) { │ │ │ │ │ + me.mousemove({ │ │ │ │ │ + xy: { │ │ │ │ │ + x: evt.clientX, │ │ │ │ │ + y: evt.clientY │ │ │ │ │ + }, │ │ │ │ │ + element: document │ │ │ │ │ + }) │ │ │ │ │ + }; │ │ │ │ │ + this._docUp = function(evt) { │ │ │ │ │ + me.mouseup({ │ │ │ │ │ + xy: { │ │ │ │ │ + x: evt.clientX, │ │ │ │ │ + y: evt.clientY │ │ │ │ │ + } │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name || this.name, url || this.url, {}, options]) │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.XYZ(this.name, this.url, this.getOptions()) │ │ │ │ │ + dragstart: function(evt) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + if (this.checkModifiers(evt) && (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt))) { │ │ │ │ │ + this.started = true; │ │ │ │ │ + this.start = evt.xy; │ │ │ │ │ + this.last = evt.xy; │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ + this.down(evt); │ │ │ │ │ + this.callback("down", [evt.xy]); │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + if (!this.oldOnselectstart) { │ │ │ │ │ + this.oldOnselectstart = document.onselectstart ? document.onselectstart : OpenLayers.Function.True │ │ │ │ │ + } │ │ │ │ │ + document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ + propagate = !this.stopDown │ │ │ │ │ + } else { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ + return propagate │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var xyz = this.getXYZ(bounds); │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - var s = "" + xyz.x + xyz.y + xyz.z; │ │ │ │ │ - url = this.selectUrl(s, url) │ │ │ │ │ + dragmove: function(evt) { │ │ │ │ │ + this.lastMoveEvt = evt; │ │ │ │ │ + if (this.started && !this.timeoutId && (evt.xy.x != this.last.x || evt.xy.y != this.last.y)) { │ │ │ │ │ + if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ + if (evt.element === document) { │ │ │ │ │ + this.adjustXY(evt); │ │ │ │ │ + this.setEvent(evt) │ │ │ │ │ + } else { │ │ │ │ │ + this.removeDocumentEvents() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.interval > 0) { │ │ │ │ │ + this.timeoutId = setTimeout(OpenLayers.Function.bind(this.removeTimeout, this), this.interval) │ │ │ │ │ + } │ │ │ │ │ + this.dragging = true; │ │ │ │ │ + this.move(evt); │ │ │ │ │ + this.callback("move", [evt.xy]); │ │ │ │ │ + if (!this.oldOnselectstart) { │ │ │ │ │ + this.oldOnselectstart = document.onselectstart; │ │ │ │ │ + document.onselectstart = OpenLayers.Function.False │ │ │ │ │ + } │ │ │ │ │ + this.last = evt.xy │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.String.format(url, xyz) │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - getXYZ: function(bounds) { │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.maxExtent.left) / (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((this.maxExtent.top - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getServerZoom(); │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - var limit = Math.pow(2, z); │ │ │ │ │ - x = (x % limit + limit) % limit │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - x: x, │ │ │ │ │ - y: y, │ │ │ │ │ - z: z │ │ │ │ │ + dragend: function(evt) { │ │ │ │ │ + if (this.started) { │ │ │ │ │ + if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ + this.adjustXY(evt); │ │ │ │ │ + this.removeDocumentEvents() │ │ │ │ │ + } │ │ │ │ │ + var dragged = this.start != this.last; │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ + this.up(evt); │ │ │ │ │ + this.callback("up", [evt.xy]); │ │ │ │ │ + if (dragged) { │ │ │ │ │ + this.callback("done", [evt.xy]) │ │ │ │ │ + } │ │ │ │ │ + document.onselectstart = this.oldOnselectstart │ │ │ │ │ } │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (!this.tileOrigin) { │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, this.maxExtent.bottom) │ │ │ │ │ - } │ │ │ │ │ + down: function(evt) {}, │ │ │ │ │ + move: function(evt) {}, │ │ │ │ │ + up: function(evt) {}, │ │ │ │ │ + out: function(evt) {}, │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + return this.dragstart(evt) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.XYZ" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.ArcGISCache = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ - url: null, │ │ │ │ │ - tileOrigin: null, │ │ │ │ │ - tileSize: new OpenLayers.Size(256, 256), │ │ │ │ │ - useArcGISServer: true, │ │ │ │ │ - type: "png", │ │ │ │ │ - useScales: false, │ │ │ │ │ - overrideDPI: false, │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (this.resolutions) { │ │ │ │ │ - this.serverResolutions = this.resolutions; │ │ │ │ │ - this.maxExtent = this.getMaxExtentForResolution(this.resolutions[0]) │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + return this.dragstart(evt) │ │ │ │ │ + }, │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + return this.dragmove(evt) │ │ │ │ │ + }, │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + return this.dragmove(evt) │ │ │ │ │ + }, │ │ │ │ │ + removeTimeout: function() { │ │ │ │ │ + this.timeoutId = null; │ │ │ │ │ + if (this.dragging) { │ │ │ │ │ + this.mousemove(this.lastMoveEvt) │ │ │ │ │ } │ │ │ │ │ - if (this.layerInfo) { │ │ │ │ │ - var info = this.layerInfo; │ │ │ │ │ - var startingTileExtent = new OpenLayers.Bounds(info.fullExtent.xmin, info.fullExtent.ymin, info.fullExtent.xmax, info.fullExtent.ymax); │ │ │ │ │ - this.projection = "EPSG:" + info.spatialReference.wkid; │ │ │ │ │ - this.sphericalMercator = info.spatialReference.wkid == 102100; │ │ │ │ │ - this.units = info.units == "esriFeet" ? "ft" : "m"; │ │ │ │ │ - if (!!info.tileInfo) { │ │ │ │ │ - this.tileSize = new OpenLayers.Size(info.tileInfo.width || info.tileInfo.cols, info.tileInfo.height || info.tileInfo.rows); │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(info.tileInfo.origin.x, info.tileInfo.origin.y); │ │ │ │ │ - var upperLeft = new OpenLayers.Geometry.Point(startingTileExtent.left, startingTileExtent.top); │ │ │ │ │ - var bottomRight = new OpenLayers.Geometry.Point(startingTileExtent.right, startingTileExtent.bottom); │ │ │ │ │ - if (this.useScales) { │ │ │ │ │ - this.scales = [] │ │ │ │ │ - } else { │ │ │ │ │ - this.resolutions = [] │ │ │ │ │ - } │ │ │ │ │ - this.lods = []; │ │ │ │ │ - for (var key in info.tileInfo.lods) { │ │ │ │ │ - if (info.tileInfo.lods.hasOwnProperty(key)) { │ │ │ │ │ - var lod = info.tileInfo.lods[key]; │ │ │ │ │ - if (this.useScales) { │ │ │ │ │ - this.scales.push(lod.scale) │ │ │ │ │ - } else { │ │ │ │ │ - this.resolutions.push(lod.resolution) │ │ │ │ │ - } │ │ │ │ │ - var start = this.getContainingTileCoords(upperLeft, lod.resolution); │ │ │ │ │ - lod.startTileCol = start.x; │ │ │ │ │ - lod.startTileRow = start.y; │ │ │ │ │ - var end = this.getContainingTileCoords(bottomRight, lod.resolution); │ │ │ │ │ - lod.endTileCol = end.x; │ │ │ │ │ - lod.endTileRow = end.y; │ │ │ │ │ - this.lods.push(lod) │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + return this.dragend(evt) │ │ │ │ │ + }, │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + evt.xy = this.last; │ │ │ │ │ + return this.dragend(evt) │ │ │ │ │ + }, │ │ │ │ │ + mouseout: function(evt) { │ │ │ │ │ + if (this.started && OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ + if (this.documentDrag === true) { │ │ │ │ │ + this.addDocumentEvents() │ │ │ │ │ + } else { │ │ │ │ │ + var dragged = this.start != this.last; │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ + this.out(evt); │ │ │ │ │ + this.callback("out", []); │ │ │ │ │ + if (dragged) { │ │ │ │ │ + this.callback("done", [evt.xy]) │ │ │ │ │ } │ │ │ │ │ - this.maxExtent = this.calculateMaxExtentWithLOD(this.lods[0]); │ │ │ │ │ - this.serverResolutions = this.resolutions; │ │ │ │ │ - if (this.overrideDPI && info.tileInfo.dpi) { │ │ │ │ │ - OpenLayers.DOTS_PER_INCH = info.tileInfo.dpi │ │ │ │ │ + if (document.onselectstart) { │ │ │ │ │ + document.onselectstart = this.oldOnselectstart │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - getContainingTileCoords: function(point, res) { │ │ │ │ │ - return new OpenLayers.Pixel(Math.max(Math.floor((point.x - this.tileOrigin.lon) / (this.tileSize.w * res)), 0), Math.max(Math.floor((this.tileOrigin.lat - point.y) / (this.tileSize.h * res)), 0)) │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + return this.start == this.last │ │ │ │ │ }, │ │ │ │ │ - calculateMaxExtentWithLOD: function(lod) { │ │ │ │ │ - var numTileCols = lod.endTileCol - lod.startTileCol + 1; │ │ │ │ │ - var numTileRows = lod.endTileRow - lod.startTileRow + 1; │ │ │ │ │ - var minX = this.tileOrigin.lon + lod.startTileCol * this.tileSize.w * lod.resolution; │ │ │ │ │ - var maxX = minX + numTileCols * this.tileSize.w * lod.resolution; │ │ │ │ │ - var maxY = this.tileOrigin.lat - lod.startTileRow * this.tileSize.h * lod.resolution; │ │ │ │ │ - var minY = maxY - numTileRows * this.tileSize.h * lod.resolution; │ │ │ │ │ - return new OpenLayers.Bounds(minX, minY, maxX, maxY) │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + activated = true │ │ │ │ │ + } │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - calculateMaxExtentWithExtent: function(extent, res) { │ │ │ │ │ - var upperLeft = new OpenLayers.Geometry.Point(extent.left, extent.top); │ │ │ │ │ - var bottomRight = new OpenLayers.Geometry.Point(extent.right, extent.bottom); │ │ │ │ │ - var start = this.getContainingTileCoords(upperLeft, res); │ │ │ │ │ - var end = this.getContainingTileCoords(bottomRight, res); │ │ │ │ │ - var lod = { │ │ │ │ │ - resolution: res, │ │ │ │ │ - startTileCol: start.x, │ │ │ │ │ - startTileRow: start.y, │ │ │ │ │ - endTileCol: end.x, │ │ │ │ │ - endTileRow: end.y │ │ │ │ │ - }; │ │ │ │ │ - return this.calculateMaxExtentWithLOD(lod) │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.dragging = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + deactivated = true; │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown") │ │ │ │ │ + } │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - getUpperLeftTileCoord: function(res) { │ │ │ │ │ - var upperLeft = new OpenLayers.Geometry.Point(this.maxExtent.left, this.maxExtent.top); │ │ │ │ │ - return this.getContainingTileCoords(upperLeft, res) │ │ │ │ │ + adjustXY: function(evt) { │ │ │ │ │ + var pos = OpenLayers.Util.pagePosition(this.map.viewPortDiv); │ │ │ │ │ + evt.xy.x -= pos[0]; │ │ │ │ │ + evt.xy.y -= pos[1] │ │ │ │ │ }, │ │ │ │ │ - getLowerRightTileCoord: function(res) { │ │ │ │ │ - var bottomRight = new OpenLayers.Geometry.Point(this.maxExtent.right, this.maxExtent.bottom); │ │ │ │ │ - return this.getContainingTileCoords(bottomRight, res) │ │ │ │ │ + addDocumentEvents: function() { │ │ │ │ │ + OpenLayers.Element.addClass(document.body, "olDragDown"); │ │ │ │ │ + this.documentEvents = true; │ │ │ │ │ + OpenLayers.Event.observe(document, "mousemove", this._docMove); │ │ │ │ │ + OpenLayers.Event.observe(document, "mouseup", this._docUp) │ │ │ │ │ }, │ │ │ │ │ - getMaxExtentForResolution: function(res) { │ │ │ │ │ - var start = this.getUpperLeftTileCoord(res); │ │ │ │ │ - var end = this.getLowerRightTileCoord(res); │ │ │ │ │ - var numTileCols = end.x - start.x + 1; │ │ │ │ │ - var numTileRows = end.y - start.y + 1; │ │ │ │ │ - var minX = this.tileOrigin.lon + start.x * this.tileSize.w * res; │ │ │ │ │ - var maxX = minX + numTileCols * this.tileSize.w * res; │ │ │ │ │ - var maxY = this.tileOrigin.lat - start.y * this.tileSize.h * res; │ │ │ │ │ - var minY = maxY - numTileRows * this.tileSize.h * res; │ │ │ │ │ - return new OpenLayers.Bounds(minX, minY, maxX, maxY) │ │ │ │ │ + removeDocumentEvents: function() { │ │ │ │ │ + OpenLayers.Element.removeClass(document.body, "olDragDown"); │ │ │ │ │ + this.documentEvents = false; │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mousemove", this._docMove); │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mouseup", this._docUp) │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.ArcGISCache(this.name, this.url, this.options) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Drag" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Handler.RegularPolygon = OpenLayers.Class(OpenLayers.Handler.Drag, { │ │ │ │ │ + sides: 4, │ │ │ │ │ + radius: null, │ │ │ │ │ + snapAngle: null, │ │ │ │ │ + snapToggle: "shiftKey", │ │ │ │ │ + layerOptions: null, │ │ │ │ │ + persist: false, │ │ │ │ │ + irregular: false, │ │ │ │ │ + citeCompliant: false, │ │ │ │ │ + angle: null, │ │ │ │ │ + fixedRadius: false, │ │ │ │ │ + feature: null, │ │ │ │ │ + layer: null, │ │ │ │ │ + origin: null, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ + this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style["default"], {}) │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]) │ │ │ │ │ - }, │ │ │ │ │ - initGriddedTiles: function(bounds) { │ │ │ │ │ - delete this._tileOrigin; │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initGriddedTiles.apply(this, arguments) │ │ │ │ │ + OpenLayers.Handler.Drag.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ + this.options = options ? options : {} │ │ │ │ │ }, │ │ │ │ │ - getMaxExtent: function() { │ │ │ │ │ - var resolution = this.map.getResolution(); │ │ │ │ │ - return this.maxExtent = this.getMaxExtentForResolution(resolution) │ │ │ │ │ + setOptions: function(newOptions) { │ │ │ │ │ + OpenLayers.Util.extend(this.options, newOptions); │ │ │ │ │ + OpenLayers.Util.extend(this, newOptions) │ │ │ │ │ }, │ │ │ │ │ - getTileOrigin: function() { │ │ │ │ │ - if (!this._tileOrigin) { │ │ │ │ │ - var extent = this.getMaxExtent(); │ │ │ │ │ - this._tileOrigin = new OpenLayers.LonLat(extent.left, extent.bottom) │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.Drag.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + var options = OpenLayers.Util.extend({ │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + calculateInRange: OpenLayers.Function.True, │ │ │ │ │ + wrapDateLine: this.citeCompliant │ │ │ │ │ + }, this.layerOptions); │ │ │ │ │ + this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ + this.map.addLayer(this.layer); │ │ │ │ │ + activated = true │ │ │ │ │ } │ │ │ │ │ - return this._tileOrigin │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var res = this.getResolution(); │ │ │ │ │ - var originTileX = this.tileOrigin.lon + res * this.tileSize.w / 2; │ │ │ │ │ - var originTileY = this.tileOrigin.lat - res * this.tileSize.h / 2; │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var point = { │ │ │ │ │ - x: center.lon, │ │ │ │ │ - y: center.lat │ │ │ │ │ - }; │ │ │ │ │ - var x = Math.round(Math.abs((center.lon - originTileX) / (res * this.tileSize.w))); │ │ │ │ │ - var y = Math.round(Math.abs((originTileY - center.lat) / (res * this.tileSize.h))); │ │ │ │ │ - var z = this.map.getZoom(); │ │ │ │ │ - if (this.lods) { │ │ │ │ │ - var lod = this.lods[this.map.getZoom()]; │ │ │ │ │ - if (x < lod.startTileCol || x > lod.endTileCol || (y < lod.startTileRow || y > lod.endTileRow)) { │ │ │ │ │ - return null │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.Drag.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + if (this.dragging) { │ │ │ │ │ + this.cancel() │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - var start = this.getUpperLeftTileCoord(res); │ │ │ │ │ - var end = this.getLowerRightTileCoord(res); │ │ │ │ │ - if (x < start.x || x >= end.x || (y < start.y || y >= end.y)) { │ │ │ │ │ - return null │ │ │ │ │ + if (this.layer.map != null) { │ │ │ │ │ + this.layer.destroy(false); │ │ │ │ │ + if (this.feature) { │ │ │ │ │ + this.feature.destroy() │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + this.layer = null; │ │ │ │ │ + this.feature = null; │ │ │ │ │ + deactivated = true │ │ │ │ │ } │ │ │ │ │ - var url = this.url; │ │ │ │ │ - var s = "" + x + y + z; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(s, url) │ │ │ │ │ + return deactivated │ │ │ │ │ + }, │ │ │ │ │ + down: function(evt) { │ │ │ │ │ + this.fixedRadius = !!this.radius; │ │ │ │ │ + var maploc = this.layer.getLonLatFromViewPortPx(evt.xy); │ │ │ │ │ + this.origin = new OpenLayers.Geometry.Point(maploc.lon, maploc.lat); │ │ │ │ │ + if (!this.fixedRadius || this.irregular) { │ │ │ │ │ + this.radius = this.map.getResolution() │ │ │ │ │ } │ │ │ │ │ - if (this.useArcGISServer) { │ │ │ │ │ - url = url + "/tile/${z}/${y}/${x}" │ │ │ │ │ - } else { │ │ │ │ │ - x = "C" + OpenLayers.Number.zeroPad(x, 8, 16); │ │ │ │ │ - y = "R" + OpenLayers.Number.zeroPad(y, 8, 16); │ │ │ │ │ - z = "L" + OpenLayers.Number.zeroPad(z, 2, 10); │ │ │ │ │ - url = url + "/${z}/${y}/${x}." + this.type │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.clear() │ │ │ │ │ } │ │ │ │ │ - url = OpenLayers.String.format(url, { │ │ │ │ │ - x: x, │ │ │ │ │ - y: y, │ │ │ │ │ - z: z │ │ │ │ │ + this.feature = new OpenLayers.Feature.Vector; │ │ │ │ │ + this.createGeometry(); │ │ │ │ │ + this.callback("create", [this.origin, this.feature]); │ │ │ │ │ + this.layer.addFeatures([this.feature], { │ │ │ │ │ + silent: true │ │ │ │ │ }); │ │ │ │ │ - return OpenLayers.Util.urlAppend(url, OpenLayers.Util.getParameterString(this.params)) │ │ │ │ │ + this.layer.drawFeature(this.feature, this.style) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.ArcGISCache" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.ArcXML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - fontStyleKeys: ["antialiasing", "blockout", "font", "fontcolor", "fontsize", "fontstyle", "glowing", "interval", "outline", "printmode", "shadow", "transparency"], │ │ │ │ │ - request: null, │ │ │ │ │ - response: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.request = new OpenLayers.Format.ArcXML.Request; │ │ │ │ │ - this.response = new OpenLayers.Format.ArcXML.Response; │ │ │ │ │ - if (options) { │ │ │ │ │ - if (options.requesttype == "feature") { │ │ │ │ │ - this.request.get_image = null; │ │ │ │ │ - var qry = this.request.get_feature.query; │ │ │ │ │ - this.addCoordSys(qry.featurecoordsys, options.featureCoordSys); │ │ │ │ │ - this.addCoordSys(qry.filtercoordsys, options.filterCoordSys); │ │ │ │ │ - if (options.polygon) { │ │ │ │ │ - qry.isspatial = true; │ │ │ │ │ - qry.spatialfilter.polygon = options.polygon │ │ │ │ │ - } else if (options.envelope) { │ │ │ │ │ - qry.isspatial = true; │ │ │ │ │ - qry.spatialfilter.envelope = { │ │ │ │ │ - minx: 0, │ │ │ │ │ - miny: 0, │ │ │ │ │ - maxx: 0, │ │ │ │ │ - maxy: 0 │ │ │ │ │ - }; │ │ │ │ │ - this.parseEnvelope(qry.spatialfilter.envelope, options.envelope) │ │ │ │ │ - } │ │ │ │ │ - } else if (options.requesttype == "image") { │ │ │ │ │ - this.request.get_feature = null; │ │ │ │ │ - var props = this.request.get_image.properties; │ │ │ │ │ - this.parseEnvelope(props.envelope, options.envelope); │ │ │ │ │ - this.addLayers(props.layerlist, options.layers); │ │ │ │ │ - this.addImageSize(props.imagesize, options.tileSize); │ │ │ │ │ - this.addCoordSys(props.featurecoordsys, options.featureCoordSys); │ │ │ │ │ - this.addCoordSys(props.filtercoordsys, options.filterCoordSys) │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + var maploc = this.layer.getLonLatFromViewPortPx(evt.xy); │ │ │ │ │ + var point = new OpenLayers.Geometry.Point(maploc.lon, maploc.lat); │ │ │ │ │ + if (this.irregular) { │ │ │ │ │ + var ry = Math.sqrt(2) * Math.abs(point.y - this.origin.y) / 2; │ │ │ │ │ + this.radius = Math.max(this.map.getResolution() / 2, ry) │ │ │ │ │ + } else if (this.fixedRadius) { │ │ │ │ │ + this.origin = point │ │ │ │ │ + } else { │ │ │ │ │ + this.calculateAngle(point, evt); │ │ │ │ │ + this.radius = Math.max(this.map.getResolution() / 2, point.distanceTo(this.origin)) │ │ │ │ │ + } │ │ │ │ │ + this.modifyGeometry(); │ │ │ │ │ + if (this.irregular) { │ │ │ │ │ + var dx = point.x - this.origin.x; │ │ │ │ │ + var dy = point.y - this.origin.y; │ │ │ │ │ + var ratio; │ │ │ │ │ + if (dy == 0) { │ │ │ │ │ + ratio = dx / (this.radius * Math.sqrt(2)) │ │ │ │ │ } else { │ │ │ │ │ - this.request = null │ │ │ │ │ + ratio = dx / dy │ │ │ │ │ } │ │ │ │ │ + this.feature.geometry.resize(1, this.origin, ratio); │ │ │ │ │ + this.feature.geometry.move(dx / 2, dy / 2) │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ + this.layer.drawFeature(this.feature, this.style) │ │ │ │ │ }, │ │ │ │ │ - parseEnvelope: function(env, arr) { │ │ │ │ │ - if (arr && arr.length == 4) { │ │ │ │ │ - env.minx = arr[0]; │ │ │ │ │ - env.miny = arr[1]; │ │ │ │ │ - env.maxx = arr[2]; │ │ │ │ │ - env.maxy = arr[3] │ │ │ │ │ + up: function(evt) { │ │ │ │ │ + this.finalize(); │ │ │ │ │ + if (this.start == this.last) { │ │ │ │ │ + this.callback("done", [evt.xy]) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - addLayers: function(ll, lyrs) { │ │ │ │ │ - for (var lind = 0, len = lyrs.length; lind < len; lind++) { │ │ │ │ │ - ll.push(lyrs[lind]) │ │ │ │ │ - } │ │ │ │ │ + out: function(evt) { │ │ │ │ │ + this.finalize() │ │ │ │ │ }, │ │ │ │ │ - addImageSize: function(imsize, olsize) { │ │ │ │ │ - if (olsize !== null) { │ │ │ │ │ - imsize.width = olsize.w; │ │ │ │ │ - imsize.height = olsize.h; │ │ │ │ │ - imsize.printwidth = olsize.w; │ │ │ │ │ - imsize.printheight = olsize.h │ │ │ │ │ + createGeometry: function() { │ │ │ │ │ + this.angle = Math.PI * (1 / this.sides - 1 / 2); │ │ │ │ │ + if (this.snapAngle) { │ │ │ │ │ + this.angle += this.snapAngle * (Math.PI / 180) │ │ │ │ │ } │ │ │ │ │ + this.feature.geometry = OpenLayers.Geometry.Polygon.createRegularPolygon(this.origin, this.radius, this.sides, this.snapAngle) │ │ │ │ │ }, │ │ │ │ │ - addCoordSys: function(featOrFilt, fsys) { │ │ │ │ │ - if (typeof fsys == "string") { │ │ │ │ │ - featOrFilt.id = parseInt(fsys); │ │ │ │ │ - featOrFilt.string = fsys │ │ │ │ │ - } else if (typeof fsys == "object" && fsys.proj !== null) { │ │ │ │ │ - featOrFilt.id = fsys.proj.srsProjNumber; │ │ │ │ │ - featOrFilt.string = fsys.proj.srsCode │ │ │ │ │ - } else { │ │ │ │ │ - featOrFilt = fsys │ │ │ │ │ + modifyGeometry: function() { │ │ │ │ │ + var angle, point; │ │ │ │ │ + var ring = this.feature.geometry.components[0]; │ │ │ │ │ + if (ring.components.length != this.sides + 1) { │ │ │ │ │ + this.createGeometry(); │ │ │ │ │ + ring = this.feature.geometry.components[0] │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0; i < this.sides; ++i) { │ │ │ │ │ + point = ring.components[i]; │ │ │ │ │ + angle = this.angle + i * 2 * Math.PI / this.sides; │ │ │ │ │ + point.x = this.origin.x + this.radius * Math.cos(angle); │ │ │ │ │ + point.y = this.origin.y + this.radius * Math.sin(angle); │ │ │ │ │ + point.clearBounds() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - iserror: function(data) { │ │ │ │ │ - var ret = null; │ │ │ │ │ - if (!data) { │ │ │ │ │ - ret = this.response.error !== "" │ │ │ │ │ + calculateAngle: function(point, evt) { │ │ │ │ │ + var alpha = Math.atan2(point.y - this.origin.y, point.x - this.origin.x); │ │ │ │ │ + if (this.snapAngle && (this.snapToggle && !evt[this.snapToggle])) { │ │ │ │ │ + var snapAngleRad = Math.PI / 180 * this.snapAngle; │ │ │ │ │ + this.angle = Math.round(alpha / snapAngleRad) * snapAngleRad │ │ │ │ │ } else { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ - var errorNodes = data.documentElement.getElementsByTagName("ERROR"); │ │ │ │ │ - ret = errorNodes !== null && errorNodes.length > 0 │ │ │ │ │ + this.angle = alpha │ │ │ │ │ } │ │ │ │ │ - return ret │ │ │ │ │ }, │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ - } │ │ │ │ │ - var arcNode = null; │ │ │ │ │ - if (data && data.documentElement) { │ │ │ │ │ - if (data.documentElement.nodeName == "ARCXML") { │ │ │ │ │ - arcNode = data.documentElement │ │ │ │ │ - } else { │ │ │ │ │ - arcNode = data.documentElement.getElementsByTagName("ARCXML")[0] │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!arcNode || arcNode.firstChild.nodeName === "parsererror") { │ │ │ │ │ - var error, source; │ │ │ │ │ - try { │ │ │ │ │ - error = data.firstChild.nodeValue; │ │ │ │ │ - source = data.firstChild.childNodes[1].firstChild.nodeValue │ │ │ │ │ - } catch (err) {} │ │ │ │ │ - throw { │ │ │ │ │ - message: "Error parsing the ArcXML request", │ │ │ │ │ - error: error, │ │ │ │ │ - source: source │ │ │ │ │ - } │ │ │ │ │ + cancel: function() { │ │ │ │ │ + this.callback("cancel", null); │ │ │ │ │ + this.finalize() │ │ │ │ │ + }, │ │ │ │ │ + finalize: function() { │ │ │ │ │ + this.origin = null; │ │ │ │ │ + this.radius = this.options.radius │ │ │ │ │ + }, │ │ │ │ │ + clear: function() { │ │ │ │ │ + if (this.layer) { │ │ │ │ │ + this.layer.renderer.clear(); │ │ │ │ │ + this.layer.destroyFeatures() │ │ │ │ │ } │ │ │ │ │ - var response = this.parseResponse(arcNode); │ │ │ │ │ - return response │ │ │ │ │ }, │ │ │ │ │ - write: function(request) { │ │ │ │ │ - if (!request) { │ │ │ │ │ - request = this.request │ │ │ │ │ + callback: function(name, args) { │ │ │ │ │ + if (this.callbacks[name]) { │ │ │ │ │ + this.callbacks[name].apply(this.control, [this.feature.geometry.clone()]) │ │ │ │ │ } │ │ │ │ │ - var root = this.createElementNS("", "ARCXML"); │ │ │ │ │ - root.setAttribute("version", "1.1"); │ │ │ │ │ - var reqElem = this.createElementNS("", "REQUEST"); │ │ │ │ │ - if (request.get_image != null) { │ │ │ │ │ - var getElem = this.createElementNS("", "GET_IMAGE"); │ │ │ │ │ - reqElem.appendChild(getElem); │ │ │ │ │ - var propElem = this.createElementNS("", "PROPERTIES"); │ │ │ │ │ - getElem.appendChild(propElem); │ │ │ │ │ - var props = request.get_image.properties; │ │ │ │ │ - if (props.featurecoordsys != null) { │ │ │ │ │ - var feat = this.createElementNS("", "FEATURECOORDSYS"); │ │ │ │ │ - propElem.appendChild(feat); │ │ │ │ │ - if (props.featurecoordsys.id === 0) { │ │ │ │ │ - feat.setAttribute("string", props.featurecoordsys["string"]) │ │ │ │ │ - } else { │ │ │ │ │ - feat.setAttribute("id", props.featurecoordsys.id) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (props.filtercoordsys != null) { │ │ │ │ │ - var filt = this.createElementNS("", "FILTERCOORDSYS"); │ │ │ │ │ - propElem.appendChild(filt); │ │ │ │ │ - if (props.filtercoordsys.id === 0) { │ │ │ │ │ - filt.setAttribute("string", props.filtercoordsys.string) │ │ │ │ │ - } else { │ │ │ │ │ - filt.setAttribute("id", props.filtercoordsys.id) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (props.envelope != null) { │ │ │ │ │ - var env = this.createElementNS("", "ENVELOPE"); │ │ │ │ │ - propElem.appendChild(env); │ │ │ │ │ - env.setAttribute("minx", props.envelope.minx); │ │ │ │ │ - env.setAttribute("miny", props.envelope.miny); │ │ │ │ │ - env.setAttribute("maxx", props.envelope.maxx); │ │ │ │ │ - env.setAttribute("maxy", props.envelope.maxy) │ │ │ │ │ - } │ │ │ │ │ - var imagesz = this.createElementNS("", "IMAGESIZE"); │ │ │ │ │ - propElem.appendChild(imagesz); │ │ │ │ │ - imagesz.setAttribute("height", props.imagesize.height); │ │ │ │ │ - imagesz.setAttribute("width", props.imagesize.width); │ │ │ │ │ - if (props.imagesize.height != props.imagesize.printheight || props.imagesize.width != props.imagesize.printwidth) { │ │ │ │ │ - imagesz.setAttribute("printheight", props.imagesize.printheight); │ │ │ │ │ - imagesz.setArrtibute("printwidth", props.imagesize.printwidth) │ │ │ │ │ - } │ │ │ │ │ - if (props.background != null) { │ │ │ │ │ - var backgrnd = this.createElementNS("", "BACKGROUND"); │ │ │ │ │ - propElem.appendChild(backgrnd); │ │ │ │ │ - backgrnd.setAttribute("color", props.background.color.r + "," + props.background.color.g + "," + props.background.color.b); │ │ │ │ │ - if (props.background.transcolor !== null) { │ │ │ │ │ - backgrnd.setAttribute("transcolor", props.background.transcolor.r + "," + props.background.transcolor.g + "," + props.background.transcolor.b) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (props.layerlist != null && props.layerlist.length > 0) { │ │ │ │ │ - var layerlst = this.createElementNS("", "LAYERLIST"); │ │ │ │ │ - propElem.appendChild(layerlst); │ │ │ │ │ - for (var ld = 0; ld < props.layerlist.length; ld++) { │ │ │ │ │ - var ldef = this.createElementNS("", "LAYERDEF"); │ │ │ │ │ - layerlst.appendChild(ldef); │ │ │ │ │ - ldef.setAttribute("id", props.layerlist[ld].id); │ │ │ │ │ - ldef.setAttribute("visible", props.layerlist[ld].visible); │ │ │ │ │ - if (typeof props.layerlist[ld].query == "object") { │ │ │ │ │ - var query = props.layerlist[ld].query; │ │ │ │ │ - if (query.where.length < 0) { │ │ │ │ │ - continue │ │ │ │ │ - } │ │ │ │ │ - var queryElem = null; │ │ │ │ │ - if (typeof query.spatialfilter == "boolean" && query.spatialfilter) { │ │ │ │ │ - queryElem = this.createElementNS("", "SPATIALQUERY") │ │ │ │ │ - } else { │ │ │ │ │ - queryElem = this.createElementNS("", "QUERY") │ │ │ │ │ - } │ │ │ │ │ - queryElem.setAttribute("where", query.where); │ │ │ │ │ - if (typeof query.accuracy == "number" && query.accuracy > 0) { │ │ │ │ │ - queryElem.setAttribute("accuracy", query.accuracy) │ │ │ │ │ - } │ │ │ │ │ - if (typeof query.featurelimit == "number" && query.featurelimit < 2e3) { │ │ │ │ │ - queryElem.setAttribute("featurelimit", query.featurelimit) │ │ │ │ │ - } │ │ │ │ │ - if (typeof query.subfields == "string" && query.subfields != "#ALL#") { │ │ │ │ │ - queryElem.setAttribute("subfields", query.subfields) │ │ │ │ │ - } │ │ │ │ │ - if (typeof query.joinexpression == "string" && query.joinexpression.length > 0) { │ │ │ │ │ - queryElem.setAttribute("joinexpression", query.joinexpression) │ │ │ │ │ - } │ │ │ │ │ - if (typeof query.jointables == "string" && query.jointables.length > 0) { │ │ │ │ │ - queryElem.setAttribute("jointables", query.jointables) │ │ │ │ │ - } │ │ │ │ │ - ldef.appendChild(queryElem) │ │ │ │ │ - } │ │ │ │ │ - if (typeof props.layerlist[ld].renderer == "object") { │ │ │ │ │ - this.addRenderer(ldef, props.layerlist[ld].renderer) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else if (request.get_feature != null) { │ │ │ │ │ - var getElem = this.createElementNS("", "GET_FEATURES"); │ │ │ │ │ - getElem.setAttribute("outputmode", "newxml"); │ │ │ │ │ - getElem.setAttribute("checkesc", "true"); │ │ │ │ │ - if (request.get_feature.geometry) { │ │ │ │ │ - getElem.setAttribute("geometry", request.get_feature.geometry) │ │ │ │ │ - } else { │ │ │ │ │ - getElem.setAttribute("geometry", "false") │ │ │ │ │ - } │ │ │ │ │ - if (request.get_feature.compact) { │ │ │ │ │ - getElem.setAttribute("compact", request.get_feature.compact) │ │ │ │ │ - } │ │ │ │ │ - if (request.get_feature.featurelimit == "number") { │ │ │ │ │ - getElem.setAttribute("featurelimit", request.get_feature.featurelimit) │ │ │ │ │ - } │ │ │ │ │ - getElem.setAttribute("globalenvelope", "true"); │ │ │ │ │ - reqElem.appendChild(getElem); │ │ │ │ │ - if (request.get_feature.layer != null && request.get_feature.layer.length > 0) { │ │ │ │ │ - var lyrElem = this.createElementNS("", "LAYER"); │ │ │ │ │ - lyrElem.setAttribute("id", request.get_feature.layer); │ │ │ │ │ - getElem.appendChild(lyrElem) │ │ │ │ │ - } │ │ │ │ │ - var fquery = request.get_feature.query; │ │ │ │ │ - if (fquery != null) { │ │ │ │ │ - var qElem = null; │ │ │ │ │ - if (fquery.isspatial) { │ │ │ │ │ - qElem = this.createElementNS("", "SPATIALQUERY") │ │ │ │ │ - } else { │ │ │ │ │ - qElem = this.createElementNS("", "QUERY") │ │ │ │ │ - } │ │ │ │ │ - getElem.appendChild(qElem); │ │ │ │ │ - if (typeof fquery.accuracy == "number") { │ │ │ │ │ - qElem.setAttribute("accuracy", fquery.accuracy) │ │ │ │ │ - } │ │ │ │ │ - if (fquery.featurecoordsys != null) { │ │ │ │ │ - var fcsElem1 = this.createElementNS("", "FEATURECOORDSYS"); │ │ │ │ │ - if (fquery.featurecoordsys.id == 0) { │ │ │ │ │ - fcsElem1.setAttribute("string", fquery.featurecoordsys.string) │ │ │ │ │ - } else { │ │ │ │ │ - fcsElem1.setAttribute("id", fquery.featurecoordsys.id) │ │ │ │ │ - } │ │ │ │ │ - qElem.appendChild(fcsElem1) │ │ │ │ │ - } │ │ │ │ │ - if (fquery.filtercoordsys != null) { │ │ │ │ │ - var fcsElem2 = this.createElementNS("", "FILTERCOORDSYS"); │ │ │ │ │ - if (fquery.filtercoordsys.id === 0) { │ │ │ │ │ - fcsElem2.setAttribute("string", fquery.filtercoordsys.string) │ │ │ │ │ - } else { │ │ │ │ │ - fcsElem2.setAttribute("id", fquery.filtercoordsys.id) │ │ │ │ │ - } │ │ │ │ │ - qElem.appendChild(fcsElem2) │ │ │ │ │ - } │ │ │ │ │ - if (fquery.buffer > 0) { │ │ │ │ │ - var bufElem = this.createElementNS("", "BUFFER"); │ │ │ │ │ - bufElem.setAttribute("distance", fquery.buffer); │ │ │ │ │ - qElem.appendChild(bufElem) │ │ │ │ │ - } │ │ │ │ │ - if (fquery.isspatial) { │ │ │ │ │ - var spfElem = this.createElementNS("", "SPATIALFILTER"); │ │ │ │ │ - spfElem.setAttribute("relation", fquery.spatialfilter.relation); │ │ │ │ │ - qElem.appendChild(spfElem); │ │ │ │ │ - if (fquery.spatialfilter.envelope) { │ │ │ │ │ - var envElem = this.createElementNS("", "ENVELOPE"); │ │ │ │ │ - envElem.setAttribute("minx", fquery.spatialfilter.envelope.minx); │ │ │ │ │ - envElem.setAttribute("miny", fquery.spatialfilter.envelope.miny); │ │ │ │ │ - envElem.setAttribute("maxx", fquery.spatialfilter.envelope.maxx); │ │ │ │ │ - envElem.setAttribute("maxy", fquery.spatialfilter.envelope.maxy); │ │ │ │ │ - spfElem.appendChild(envElem) │ │ │ │ │ - } else if (typeof fquery.spatialfilter.polygon == "object") { │ │ │ │ │ - spfElem.appendChild(this.writePolygonGeometry(fquery.spatialfilter.polygon)) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (fquery.where != null && fquery.where.length > 0) { │ │ │ │ │ - qElem.setAttribute("where", fquery.where) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + if (!this.persist && (name == "done" || name == "cancel")) { │ │ │ │ │ + this.clear() │ │ │ │ │ } │ │ │ │ │ - root.appendChild(reqElem); │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [root]) │ │ │ │ │ }, │ │ │ │ │ - addGroupRenderer: function(ldef, toprenderer) { │ │ │ │ │ - var topRelem = this.createElementNS("", "GROUPRENDERER"); │ │ │ │ │ - ldef.appendChild(topRelem); │ │ │ │ │ - for (var rind = 0; rind < toprenderer.length; rind++) { │ │ │ │ │ - var renderer = toprenderer[rind]; │ │ │ │ │ - this.addRenderer(topRelem, renderer) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.RegularPolygon" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Handler.Point = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + point: null, │ │ │ │ │ + layer: null, │ │ │ │ │ + multi: false, │ │ │ │ │ + citeCompliant: false, │ │ │ │ │ + mouseDown: false, │ │ │ │ │ + stoppedDown: null, │ │ │ │ │ + lastDown: null, │ │ │ │ │ + lastUp: null, │ │ │ │ │ + persist: false, │ │ │ │ │ + stopDown: false, │ │ │ │ │ + stopUp: false, │ │ │ │ │ + layerOptions: null, │ │ │ │ │ + pixelTolerance: 5, │ │ │ │ │ + lastTouchPx: null, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ + this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style["default"], {}) │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - addRenderer: function(topRelem, renderer) { │ │ │ │ │ - if (OpenLayers.Util.isArray(renderer)) { │ │ │ │ │ - this.addGroupRenderer(topRelem, renderer) │ │ │ │ │ - } else { │ │ │ │ │ - var renderElem = this.createElementNS("", renderer.type.toUpperCase() + "RENDERER"); │ │ │ │ │ - topRelem.appendChild(renderElem); │ │ │ │ │ - if (renderElem.tagName == "VALUEMAPRENDERER") { │ │ │ │ │ - this.addValueMapRenderer(renderElem, renderer) │ │ │ │ │ - } else if (renderElem.tagName == "VALUEMAPLABELRENDERER") { │ │ │ │ │ - this.addValueMapLabelRenderer(renderElem, renderer) │ │ │ │ │ - } else if (renderElem.tagName == "SIMPLELABELRENDERER") { │ │ │ │ │ - this.addSimpleLabelRenderer(renderElem, renderer) │ │ │ │ │ - } else if (renderElem.tagName == "SCALEDEPENDENTRENDERER") { │ │ │ │ │ - this.addScaleDependentRenderer(renderElem, renderer) │ │ │ │ │ - } │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (!OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ + var options = OpenLayers.Util.extend({ │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + calculateInRange: OpenLayers.Function.True, │ │ │ │ │ + wrapDateLine: this.citeCompliant │ │ │ │ │ + }, this.layerOptions); │ │ │ │ │ + this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ + this.map.addLayer(this.layer); │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - addScaleDependentRenderer: function(renderElem, renderer) { │ │ │ │ │ - if (typeof renderer.lower == "string" || typeof renderer.lower == "number") { │ │ │ │ │ - renderElem.setAttribute("lower", renderer.lower) │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.point]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (!OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - if (typeof renderer.upper == "string" || typeof renderer.upper == "number") { │ │ │ │ │ - renderElem.setAttribute("upper", renderer.upper) │ │ │ │ │ + this.cancel(); │ │ │ │ │ + if (this.layer.map != null) { │ │ │ │ │ + this.destroyFeature(true); │ │ │ │ │ + this.layer.destroy(false) │ │ │ │ │ } │ │ │ │ │ - this.addRenderer(renderElem, renderer.renderer) │ │ │ │ │ + this.layer = null; │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - addValueMapLabelRenderer: function(renderElem, renderer) { │ │ │ │ │ - renderElem.setAttribute("lookupfield", renderer.lookupfield); │ │ │ │ │ - renderElem.setAttribute("labelfield", renderer.labelfield); │ │ │ │ │ - if (typeof renderer.exacts == "object") { │ │ │ │ │ - for (var ext = 0, extlen = renderer.exacts.length; ext < extlen; ext++) { │ │ │ │ │ - var exact = renderer.exacts[ext]; │ │ │ │ │ - var eelem = this.createElementNS("", "EXACT"); │ │ │ │ │ - if (typeof exact.value == "string") { │ │ │ │ │ - eelem.setAttribute("value", exact.value) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.label == "string") { │ │ │ │ │ - eelem.setAttribute("label", exact.label) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.method == "string") { │ │ │ │ │ - eelem.setAttribute("method", exact.method) │ │ │ │ │ - } │ │ │ │ │ - renderElem.appendChild(eelem); │ │ │ │ │ - if (typeof exact.symbol == "object") { │ │ │ │ │ - var selem = null; │ │ │ │ │ - if (exact.symbol.type == "text") { │ │ │ │ │ - selem = this.createElementNS("", "TEXTSYMBOL") │ │ │ │ │ - } │ │ │ │ │ - if (selem != null) { │ │ │ │ │ - var keys = this.fontStyleKeys; │ │ │ │ │ - for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ - var key = keys[i]; │ │ │ │ │ - if (exact.symbol[key]) { │ │ │ │ │ - selem.setAttribute(key, exact.symbol[key]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - eelem.appendChild(selem) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + if (this.layer && (force || !this.persist)) { │ │ │ │ │ + this.layer.destroyFeatures() │ │ │ │ │ } │ │ │ │ │ + this.point = null │ │ │ │ │ }, │ │ │ │ │ - addValueMapRenderer: function(renderElem, renderer) { │ │ │ │ │ - renderElem.setAttribute("lookupfield", renderer.lookupfield); │ │ │ │ │ - if (typeof renderer.ranges == "object") { │ │ │ │ │ - for (var rng = 0, rnglen = renderer.ranges.length; rng < rnglen; rng++) { │ │ │ │ │ - var range = renderer.ranges[rng]; │ │ │ │ │ - var relem = this.createElementNS("", "RANGE"); │ │ │ │ │ - relem.setAttribute("lower", range.lower); │ │ │ │ │ - relem.setAttribute("upper", range.upper); │ │ │ │ │ - renderElem.appendChild(relem); │ │ │ │ │ - if (typeof range.symbol == "object") { │ │ │ │ │ - var selem = null; │ │ │ │ │ - if (range.symbol.type == "simplepolygon") { │ │ │ │ │ - selem = this.createElementNS("", "SIMPLEPOLYGONSYMBOL") │ │ │ │ │ - } │ │ │ │ │ - if (selem != null) { │ │ │ │ │ - if (typeof range.symbol.boundarycolor == "string") { │ │ │ │ │ - selem.setAttribute("boundarycolor", range.symbol.boundarycolor) │ │ │ │ │ - } │ │ │ │ │ - if (typeof range.symbol.fillcolor == "string") { │ │ │ │ │ - selem.setAttribute("fillcolor", range.symbol.fillcolor) │ │ │ │ │ - } │ │ │ │ │ - if (typeof range.symbol.filltransparency == "number") { │ │ │ │ │ - selem.setAttribute("filltransparency", range.symbol.filltransparency) │ │ │ │ │ - } │ │ │ │ │ - relem.appendChild(selem) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else if (typeof renderer.exacts == "object") { │ │ │ │ │ - for (var ext = 0, extlen = renderer.exacts.length; ext < extlen; ext++) { │ │ │ │ │ - var exact = renderer.exacts[ext]; │ │ │ │ │ - var eelem = this.createElementNS("", "EXACT"); │ │ │ │ │ - if (typeof exact.value == "string") { │ │ │ │ │ - eelem.setAttribute("value", exact.value) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.label == "string") { │ │ │ │ │ - eelem.setAttribute("label", exact.label) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.method == "string") { │ │ │ │ │ - eelem.setAttribute("method", exact.method) │ │ │ │ │ - } │ │ │ │ │ - renderElem.appendChild(eelem); │ │ │ │ │ - if (typeof exact.symbol == "object") { │ │ │ │ │ - var selem = null; │ │ │ │ │ - if (exact.symbol.type == "simplemarker") { │ │ │ │ │ - selem = this.createElementNS("", "SIMPLEMARKERSYMBOL") │ │ │ │ │ - } │ │ │ │ │ - if (selem != null) { │ │ │ │ │ - if (typeof exact.symbol.antialiasing == "string") { │ │ │ │ │ - selem.setAttribute("antialiasing", exact.symbol.antialiasing) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.color == "string") { │ │ │ │ │ - selem.setAttribute("color", exact.symbol.color) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.outline == "string") { │ │ │ │ │ - selem.setAttribute("outline", exact.symbol.outline) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.overlap == "string") { │ │ │ │ │ - selem.setAttribute("overlap", exact.symbol.overlap) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.shadow == "string") { │ │ │ │ │ - selem.setAttribute("shadow", exact.symbol.shadow) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.transparency == "number") { │ │ │ │ │ - selem.setAttribute("transparency", exact.symbol.transparency) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.usecentroid == "string") { │ │ │ │ │ - selem.setAttribute("usecentroid", exact.symbol.usecentroid) │ │ │ │ │ - } │ │ │ │ │ - if (typeof exact.symbol.width == "number") { │ │ │ │ │ - selem.setAttribute("width", exact.symbol.width) │ │ │ │ │ - } │ │ │ │ │ - eelem.appendChild(selem) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + destroyPersistedFeature: function() { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + if (layer && layer.features.length > 1) { │ │ │ │ │ + this.layer.features[0].destroy() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - addSimpleLabelRenderer: function(renderElem, renderer) { │ │ │ │ │ - renderElem.setAttribute("field", renderer.field); │ │ │ │ │ - var keys = ["featureweight", "howmanylabels", "labelbufferratio", "labelpriorities", "labelweight", "linelabelposition", "rotationalangles"]; │ │ │ │ │ - for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ - var key = keys[i]; │ │ │ │ │ - if (renderer[key]) { │ │ │ │ │ - renderElem.setAttribute(key, renderer[key]) │ │ │ │ │ - } │ │ │ │ │ + finalize: function(cancel) { │ │ │ │ │ + var key = cancel ? "cancel" : "done"; │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + this.lastDown = null; │ │ │ │ │ + this.lastUp = null; │ │ │ │ │ + this.lastTouchPx = null; │ │ │ │ │ + this.callback(key, [this.geometryClone()]); │ │ │ │ │ + this.destroyFeature(cancel) │ │ │ │ │ + }, │ │ │ │ │ + cancel: function() { │ │ │ │ │ + this.finalize(true) │ │ │ │ │ + }, │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + modifyFeature: function(pixel) { │ │ │ │ │ + if (!this.point) { │ │ │ │ │ + this.createFeature(pixel) │ │ │ │ │ } │ │ │ │ │ - if (renderer.symbol.type == "text") { │ │ │ │ │ - var symbol = renderer.symbol; │ │ │ │ │ - var selem = this.createElementNS("", "TEXTSYMBOL"); │ │ │ │ │ - renderElem.appendChild(selem); │ │ │ │ │ - var keys = this.fontStyleKeys; │ │ │ │ │ - for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ - var key = keys[i]; │ │ │ │ │ - if (symbol[key]) { │ │ │ │ │ - selem.setAttribute(key, renderer[key]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point.geometry.x = lonlat.lon; │ │ │ │ │ + this.point.geometry.y = lonlat.lat; │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.point, false]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.drawFeature() │ │ │ │ │ + }, │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.point, this.style) │ │ │ │ │ + }, │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.point && this.point.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiPoint([geometry]) │ │ │ │ │ } │ │ │ │ │ + return geometry │ │ │ │ │ }, │ │ │ │ │ - writePolygonGeometry: function(polygon) { │ │ │ │ │ - if (!(polygon instanceof OpenLayers.Geometry.Polygon)) { │ │ │ │ │ - throw { │ │ │ │ │ - message: "Cannot write polygon geometry to ArcXML with an " + polygon.CLASS_NAME + " object.", │ │ │ │ │ - geometry: polygon │ │ │ │ │ - } │ │ │ │ │ + geometryClone: function() { │ │ │ │ │ + var geom = this.getGeometry(); │ │ │ │ │ + return geom && geom.clone() │ │ │ │ │ + }, │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + return this.down(evt) │ │ │ │ │ + }, │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + this.lastTouchPx = evt.xy; │ │ │ │ │ + return this.down(evt) │ │ │ │ │ + }, │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + return this.move(evt) │ │ │ │ │ + }, │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + this.lastTouchPx = evt.xy; │ │ │ │ │ + return this.move(evt) │ │ │ │ │ + }, │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + return this.up(evt) │ │ │ │ │ + }, │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + evt.xy = this.lastTouchPx; │ │ │ │ │ + return this.up(evt) │ │ │ │ │ + }, │ │ │ │ │ + down: function(evt) { │ │ │ │ │ + this.mouseDown = true; │ │ │ │ │ + this.lastDown = evt.xy; │ │ │ │ │ + if (!this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy) │ │ │ │ │ } │ │ │ │ │ - var polyElem = this.createElementNS("", "POLYGON"); │ │ │ │ │ - for (var ln = 0, lnlen = polygon.components.length; ln < lnlen; ln++) { │ │ │ │ │ - var ring = polygon.components[ln]; │ │ │ │ │ - var ringElem = this.createElementNS("", "RING"); │ │ │ │ │ - for (var rn = 0, rnlen = ring.components.length; rn < rnlen; rn++) { │ │ │ │ │ - var point = ring.components[rn]; │ │ │ │ │ - var pointElem = this.createElementNS("", "POINT"); │ │ │ │ │ - pointElem.setAttribute("x", point.x); │ │ │ │ │ - pointElem.setAttribute("y", point.y); │ │ │ │ │ - ringElem.appendChild(pointElem) │ │ │ │ │ - } │ │ │ │ │ - polyElem.appendChild(ringElem) │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + return !this.stopDown │ │ │ │ │ + }, │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ + this.modifyFeature(evt.xy) │ │ │ │ │ } │ │ │ │ │ - return polyElem │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - parseResponse: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - var newData = new OpenLayers.Format.XML; │ │ │ │ │ - data = newData.read(data) │ │ │ │ │ + up: function(evt) { │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + if (!this.checkModifiers(evt)) { │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ - var response = new OpenLayers.Format.ArcXML.Response; │ │ │ │ │ - var errorNode = data.getElementsByTagName("ERROR"); │ │ │ │ │ - if (errorNode != null && errorNode.length > 0) { │ │ │ │ │ - response.error = this.getChildValue(errorNode, "Unknown error.") │ │ │ │ │ - } else { │ │ │ │ │ - var responseNode = data.getElementsByTagName("RESPONSE"); │ │ │ │ │ - if (responseNode == null || responseNode.length == 0) { │ │ │ │ │ - response.error = "No RESPONSE tag found in ArcXML response."; │ │ │ │ │ - return response │ │ │ │ │ - } │ │ │ │ │ - var rtype = responseNode[0].firstChild.nodeName; │ │ │ │ │ - if (rtype == "#text") { │ │ │ │ │ - rtype = responseNode[0].firstChild.nextSibling.nodeName │ │ │ │ │ + if (this.lastUp && this.lastUp.equals(evt.xy)) { │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + if (this.lastDown && this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance)) { │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy) │ │ │ │ │ } │ │ │ │ │ - if (rtype == "IMAGE") { │ │ │ │ │ - var envelopeNode = data.getElementsByTagName("ENVELOPE"); │ │ │ │ │ - var outputNode = data.getElementsByTagName("OUTPUT"); │ │ │ │ │ - if (envelopeNode == null || envelopeNode.length == 0) { │ │ │ │ │ - response.error = "No ENVELOPE tag found in ArcXML response." │ │ │ │ │ - } else if (outputNode == null || outputNode.length == 0) { │ │ │ │ │ - response.error = "No OUTPUT tag found in ArcXML response." │ │ │ │ │ - } else { │ │ │ │ │ - var envAttr = this.parseAttributes(envelopeNode[0]); │ │ │ │ │ - var outputAttr = this.parseAttributes(outputNode[0]); │ │ │ │ │ - if (typeof outputAttr.type == "string") { │ │ │ │ │ - response.image = { │ │ │ │ │ - envelope: envAttr, │ │ │ │ │ - output: { │ │ │ │ │ - type: outputAttr.type, │ │ │ │ │ - data: this.getChildValue(outputNode[0]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - response.image = { │ │ │ │ │ - envelope: envAttr, │ │ │ │ │ - output: outputAttr │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else if (rtype == "FEATURES") { │ │ │ │ │ - var features = responseNode[0].getElementsByTagName("FEATURES"); │ │ │ │ │ - var featureCount = features[0].getElementsByTagName("FEATURECOUNT"); │ │ │ │ │ - response.features.featurecount = featureCount[0].getAttribute("count"); │ │ │ │ │ - if (response.features.featurecount > 0) { │ │ │ │ │ - var envelope = features[0].getElementsByTagName("ENVELOPE"); │ │ │ │ │ - response.features.envelope = this.parseAttributes(envelope[0], typeof 0); │ │ │ │ │ - var featureList = features[0].getElementsByTagName("FEATURE"); │ │ │ │ │ - for (var fn = 0; fn < featureList.length; fn++) { │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector; │ │ │ │ │ - var fields = featureList[fn].getElementsByTagName("FIELD"); │ │ │ │ │ - for (var fdn = 0; fdn < fields.length; fdn++) { │ │ │ │ │ - var fieldName = fields[fdn].getAttribute("name"); │ │ │ │ │ - var fieldValue = fields[fdn].getAttribute("value"); │ │ │ │ │ - feature.attributes[fieldName] = fieldValue │ │ │ │ │ - } │ │ │ │ │ - var geom = featureList[fn].getElementsByTagName("POLYGON"); │ │ │ │ │ - if (geom.length > 0) { │ │ │ │ │ - var ring = geom[0].getElementsByTagName("RING"); │ │ │ │ │ - var polys = []; │ │ │ │ │ - for (var rn = 0; rn < ring.length; rn++) { │ │ │ │ │ - var linearRings = []; │ │ │ │ │ - linearRings.push(this.parsePointGeometry(ring[rn])); │ │ │ │ │ - var holes = ring[rn].getElementsByTagName("HOLE"); │ │ │ │ │ - for (var hn = 0; hn < holes.length; hn++) { │ │ │ │ │ - linearRings.push(this.parsePointGeometry(holes[hn])) │ │ │ │ │ - } │ │ │ │ │ - holes = null; │ │ │ │ │ - polys.push(new OpenLayers.Geometry.Polygon(linearRings)); │ │ │ │ │ - linearRings = null │ │ │ │ │ - } │ │ │ │ │ - ring = null; │ │ │ │ │ - if (polys.length == 1) { │ │ │ │ │ - feature.geometry = polys[0] │ │ │ │ │ - } else { │ │ │ │ │ - feature.geometry = new OpenLayers.Geometry.MultiPolygon(polys) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - response.features.feature.push(feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - response.error = "Unidentified response type." │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature() │ │ │ │ │ } │ │ │ │ │ + this.lastUp = evt.xy; │ │ │ │ │ + this.finalize(); │ │ │ │ │ + return !this.stopUp │ │ │ │ │ + } else { │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ - return response │ │ │ │ │ }, │ │ │ │ │ - parseAttributes: function(node, type) { │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - for (var attr = 0; attr < node.attributes.length; attr++) { │ │ │ │ │ - if (type == "number") { │ │ │ │ │ - attributes[node.attributes[attr].nodeName] = parseFloat(node.attributes[attr].nodeValue) │ │ │ │ │ - } else { │ │ │ │ │ - attributes[node.attributes[attr].nodeName] = node.attributes[attr].nodeValue │ │ │ │ │ - } │ │ │ │ │ + mouseout: function(evt) { │ │ │ │ │ + if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + this.mouseDown = false │ │ │ │ │ } │ │ │ │ │ - return attributes │ │ │ │ │ }, │ │ │ │ │ - parsePointGeometry: function(node) { │ │ │ │ │ - var ringPoints = []; │ │ │ │ │ - var coords = node.getElementsByTagName("COORDS"); │ │ │ │ │ - if (coords.length > 0) { │ │ │ │ │ - var coordArr = this.getChildValue(coords[0]); │ │ │ │ │ - coordArr = coordArr.split(/;/); │ │ │ │ │ - for (var cn = 0; cn < coordArr.length; cn++) { │ │ │ │ │ - var coordItems = coordArr[cn].split(/ /); │ │ │ │ │ - ringPoints.push(new OpenLayers.Geometry.Point(coordItems[0], coordItems[1])) │ │ │ │ │ - } │ │ │ │ │ - coords = null │ │ │ │ │ - } else { │ │ │ │ │ - var point = node.getElementsByTagName("POINT"); │ │ │ │ │ - if (point.length > 0) { │ │ │ │ │ - for (var pn = 0; pn < point.length; pn++) { │ │ │ │ │ - ringPoints.push(new OpenLayers.Geometry.Point(parseFloat(point[pn].getAttribute("x")), parseFloat(point[pn].getAttribute("y")))) │ │ │ │ │ - } │ │ │ │ │ + passesTolerance: function(pixel1, pixel2, tolerance) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + if (tolerance != null && pixel1 && pixel2) { │ │ │ │ │ + var dist = pixel1.distanceTo(pixel2); │ │ │ │ │ + if (dist > tolerance) { │ │ │ │ │ + passes = false │ │ │ │ │ } │ │ │ │ │ - point = null │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Geometry.LinearRing(ringPoints) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.ArcXML" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.ArcXML.Request = OpenLayers.Class({ │ │ │ │ │ - initialize: function(params) { │ │ │ │ │ - var defaults = { │ │ │ │ │ - get_image: { │ │ │ │ │ - properties: { │ │ │ │ │ - background: null, │ │ │ │ │ - draw: true, │ │ │ │ │ - envelope: { │ │ │ │ │ - minx: 0, │ │ │ │ │ - miny: 0, │ │ │ │ │ - maxx: 0, │ │ │ │ │ - maxy: 0 │ │ │ │ │ - }, │ │ │ │ │ - featurecoordsys: { │ │ │ │ │ - id: 0, │ │ │ │ │ - string: "", │ │ │ │ │ - datumtransformid: 0, │ │ │ │ │ - datumtransformstring: "" │ │ │ │ │ - }, │ │ │ │ │ - filtercoordsys: { │ │ │ │ │ - id: 0, │ │ │ │ │ - string: "", │ │ │ │ │ - datumtransformid: 0, │ │ │ │ │ - datumtransformstring: "" │ │ │ │ │ - }, │ │ │ │ │ - imagesize: { │ │ │ │ │ - height: 0, │ │ │ │ │ - width: 0, │ │ │ │ │ - dpi: 96, │ │ │ │ │ - printheight: 0, │ │ │ │ │ - printwidth: 0, │ │ │ │ │ - scalesymbols: false │ │ │ │ │ - }, │ │ │ │ │ - layerlist: [], │ │ │ │ │ - output: { │ │ │ │ │ - baseurl: "", │ │ │ │ │ - legendbaseurl: "", │ │ │ │ │ - legendname: "", │ │ │ │ │ - legendpath: "", │ │ │ │ │ - legendurl: "", │ │ │ │ │ - name: "", │ │ │ │ │ - path: "", │ │ │ │ │ - type: "jpg", │ │ │ │ │ - url: "" │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - get_feature: { │ │ │ │ │ - layer: "", │ │ │ │ │ - query: { │ │ │ │ │ - isspatial: false, │ │ │ │ │ - featurecoordsys: { │ │ │ │ │ - id: 0, │ │ │ │ │ - string: "", │ │ │ │ │ - datumtransformid: 0, │ │ │ │ │ - datumtransformstring: "" │ │ │ │ │ - }, │ │ │ │ │ - filtercoordsys: { │ │ │ │ │ - id: 0, │ │ │ │ │ - string: "", │ │ │ │ │ - datumtransformid: 0, │ │ │ │ │ - datumtransformstring: "" │ │ │ │ │ - }, │ │ │ │ │ - buffer: 0, │ │ │ │ │ - where: "", │ │ │ │ │ - spatialfilter: { │ │ │ │ │ - relation: "envelope_intersection", │ │ │ │ │ - envelope: null │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - environment: { │ │ │ │ │ - separators: { │ │ │ │ │ - cs: " ", │ │ │ │ │ - ts: ";" │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - layer: [], │ │ │ │ │ - workspaces: [] │ │ │ │ │ - }; │ │ │ │ │ - return OpenLayers.Util.extend(this, defaults) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.ArcXML.Request" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.ArcXML.Response = OpenLayers.Class({ │ │ │ │ │ - initialize: function(params) { │ │ │ │ │ - var defaults = { │ │ │ │ │ - image: { │ │ │ │ │ - envelope: null, │ │ │ │ │ - output: "" │ │ │ │ │ - }, │ │ │ │ │ - features: { │ │ │ │ │ - featurecount: 0, │ │ │ │ │ - envelope: null, │ │ │ │ │ - feature: [] │ │ │ │ │ - }, │ │ │ │ │ - error: "" │ │ │ │ │ - }; │ │ │ │ │ - return OpenLayers.Util.extend(this, defaults) │ │ │ │ │ + return passes │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.ArcXML.Response" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Point" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.ArcIMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - ClientVersion: "9.2", │ │ │ │ │ - ServiceName: "" │ │ │ │ │ +OpenLayers.Handler.Path = OpenLayers.Class(OpenLayers.Handler.Point, { │ │ │ │ │ + line: null, │ │ │ │ │ + maxVertices: null, │ │ │ │ │ + doubleTouchTolerance: 20, │ │ │ │ │ + freehand: false, │ │ │ │ │ + freehandToggle: "shiftKey", │ │ │ │ │ + timerId: null, │ │ │ │ │ + redoStack: null, │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.line = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.LineString([this.point.geometry])); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.line, this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - featureCoordSys: "4326", │ │ │ │ │ - filterCoordSys: "4326", │ │ │ │ │ - layers: null, │ │ │ │ │ - async: true, │ │ │ │ │ - name: "ArcIMS", │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - DEFAULT_OPTIONS: { │ │ │ │ │ - tileSize: new OpenLayers.Size(512, 512), │ │ │ │ │ - featureCoordSys: "4326", │ │ │ │ │ - filterCoordSys: "4326", │ │ │ │ │ - layers: null, │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - async: true, │ │ │ │ │ - name: "ArcIMS" │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + OpenLayers.Handler.Point.prototype.destroyFeature.call(this, force); │ │ │ │ │ + this.line = null │ │ │ │ │ }, │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - this.tileSize = new OpenLayers.Size(512, 512); │ │ │ │ │ - this.params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - ServiceName: options.serviceName │ │ │ │ │ - }, this.DEFAULT_PARAMS); │ │ │ │ │ - this.options = OpenLayers.Util.applyDefaults(options, this.DEFAULT_OPTIONS); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name, url, this.params, options]); │ │ │ │ │ - if (this.transparent) { │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - this.isBaseLayer = false │ │ │ │ │ - } │ │ │ │ │ - if (this.format == "image/jpeg") { │ │ │ │ │ - this.format = OpenLayers.Util.alphaHack() ? "image/gif" : "image/png" │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.options.layers === null) { │ │ │ │ │ - this.options.layers = [] │ │ │ │ │ + destroyPersistedFeature: function() { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + if (layer && layer.features.length > 2) { │ │ │ │ │ + this.layer.features[0].destroy() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var url = ""; │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var axlReq = new OpenLayers.Format.ArcXML(OpenLayers.Util.extend(this.options, { │ │ │ │ │ - requesttype: "image", │ │ │ │ │ - envelope: bounds.toArray(), │ │ │ │ │ - tileSize: this.tileSize │ │ │ │ │ - })); │ │ │ │ │ - var req = new OpenLayers.Request.POST({ │ │ │ │ │ - url: this.getFullRequestString(), │ │ │ │ │ - data: axlReq.write(), │ │ │ │ │ - async: false │ │ │ │ │ - }); │ │ │ │ │ - if (req != null) { │ │ │ │ │ - var doc = req.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = req.responseText │ │ │ │ │ - } │ │ │ │ │ - var axlResp = new OpenLayers.Format.ArcXML; │ │ │ │ │ - var arcxml = axlResp.read(doc); │ │ │ │ │ - url = this.getUrlOrImage(arcxml.image.output) │ │ │ │ │ + removePoint: function() { │ │ │ │ │ + if (this.point) { │ │ │ │ │ + this.layer.removeFeatures([this.point]) │ │ │ │ │ } │ │ │ │ │ - return url │ │ │ │ │ }, │ │ │ │ │ - getURLasync: function(bounds, callback, scope) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var axlReq = new OpenLayers.Format.ArcXML(OpenLayers.Util.extend(this.options, { │ │ │ │ │ - requesttype: "image", │ │ │ │ │ - envelope: bounds.toArray(), │ │ │ │ │ - tileSize: this.tileSize │ │ │ │ │ - })); │ │ │ │ │ - OpenLayers.Request.POST({ │ │ │ │ │ - url: this.getFullRequestString(), │ │ │ │ │ - async: true, │ │ │ │ │ - data: axlReq.write(), │ │ │ │ │ - callback: function(req) { │ │ │ │ │ - var doc = req.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = req.responseText │ │ │ │ │ - } │ │ │ │ │ - var axlResp = new OpenLayers.Format.ArcXML; │ │ │ │ │ - var arcxml = axlResp.read(doc); │ │ │ │ │ - callback.call(scope, this.getUrlOrImage(arcxml.image.output)) │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ + addPoint: function(pixel) { │ │ │ │ │ + this.layer.removeFeatures([this.point]); │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat)); │ │ │ │ │ + this.line.geometry.addComponent(this.point.geometry, this.line.geometry.components.length); │ │ │ │ │ + this.layer.addFeatures([this.point]); │ │ │ │ │ + this.callback("point", [this.point.geometry, this.getGeometry()]); │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + delete this.redoStack │ │ │ │ │ }, │ │ │ │ │ - getUrlOrImage: function(output) { │ │ │ │ │ - var ret = ""; │ │ │ │ │ - if (output.url) { │ │ │ │ │ - ret = output.url │ │ │ │ │ - } else if (output.data) { │ │ │ │ │ - ret = "data:image/" + output.type + ";base64," + output.data │ │ │ │ │ - } │ │ │ │ │ - return ret │ │ │ │ │ + insertXY: function(x, y) { │ │ │ │ │ + this.line.geometry.addComponent(new OpenLayers.Geometry.Point(x, y), this.getCurrentPointIndex()); │ │ │ │ │ + this.drawFeature(); │ │ │ │ │ + delete this.redoStack │ │ │ │ │ }, │ │ │ │ │ - setLayerQuery: function(id, querydef) { │ │ │ │ │ - for (var lyr = 0; lyr < this.options.layers.length; lyr++) { │ │ │ │ │ - if (id == this.options.layers[lyr].id) { │ │ │ │ │ - this.options.layers[lyr].query = querydef; │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ + insertDeltaXY: function(dx, dy) { │ │ │ │ │ + var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ + var p0 = this.line.geometry.components[previousIndex]; │ │ │ │ │ + if (p0 && !isNaN(p0.x) && !isNaN(p0.y)) { │ │ │ │ │ + this.insertXY(p0.x + dx, p0.y + dy) │ │ │ │ │ } │ │ │ │ │ - this.options.layers.push({ │ │ │ │ │ - id: id, │ │ │ │ │ - visible: true, │ │ │ │ │ - query: querydef │ │ │ │ │ - }) │ │ │ │ │ }, │ │ │ │ │ - getFeatureInfo: function(geometry, layer, options) { │ │ │ │ │ - var buffer = options.buffer || 1; │ │ │ │ │ - var callback = options.callback || function() {}; │ │ │ │ │ - var scope = options.scope || window; │ │ │ │ │ - var requestOptions = {}; │ │ │ │ │ - OpenLayers.Util.extend(requestOptions, this.options); │ │ │ │ │ - requestOptions.requesttype = "feature"; │ │ │ │ │ - if (geometry instanceof OpenLayers.LonLat) { │ │ │ │ │ - requestOptions.polygon = null; │ │ │ │ │ - requestOptions.envelope = [geometry.lon - buffer, geometry.lat - buffer, geometry.lon + buffer, geometry.lat + buffer] │ │ │ │ │ - } else if (geometry instanceof OpenLayers.Geometry.Polygon) { │ │ │ │ │ - requestOptions.envelope = null; │ │ │ │ │ - requestOptions.polygon = geometry │ │ │ │ │ - } │ │ │ │ │ - var arcxml = new OpenLayers.Format.ArcXML(requestOptions); │ │ │ │ │ - OpenLayers.Util.extend(arcxml.request.get_feature, options); │ │ │ │ │ - arcxml.request.get_feature.layer = layer.id; │ │ │ │ │ - if (typeof layer.query.accuracy == "number") { │ │ │ │ │ - arcxml.request.get_feature.query.accuracy = layer.query.accuracy │ │ │ │ │ - } else { │ │ │ │ │ - var mapCenter = this.map.getCenter(); │ │ │ │ │ - var viewPx = this.map.getViewPortPxFromLonLat(mapCenter); │ │ │ │ │ - viewPx.x++; │ │ │ │ │ - var mapOffCenter = this.map.getLonLatFromPixel(viewPx); │ │ │ │ │ - arcxml.request.get_feature.query.accuracy = mapOffCenter.lon - mapCenter.lon │ │ │ │ │ - } │ │ │ │ │ - arcxml.request.get_feature.query.where = layer.query.where; │ │ │ │ │ - arcxml.request.get_feature.query.spatialfilter.relation = "area_intersection"; │ │ │ │ │ - OpenLayers.Request.POST({ │ │ │ │ │ - url: this.getFullRequestString({ │ │ │ │ │ - CustomService: "Query" │ │ │ │ │ - }), │ │ │ │ │ - data: arcxml.write(), │ │ │ │ │ - callback: function(request) { │ │ │ │ │ - var response = arcxml.parseResponse(request.responseText); │ │ │ │ │ - if (!arcxml.iserror()) { │ │ │ │ │ - callback.call(scope, response.features) │ │ │ │ │ - } else { │ │ │ │ │ - callback.call(scope, null) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }) │ │ │ │ │ + insertDirectionLength: function(direction, length) { │ │ │ │ │ + direction *= Math.PI / 180; │ │ │ │ │ + var dx = length * Math.cos(direction); │ │ │ │ │ + var dy = length * Math.sin(direction); │ │ │ │ │ + this.insertDeltaXY(dx, dy) │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.ArcIMS(this.name, this.url, this.getOptions()) │ │ │ │ │ + insertDeflectionLength: function(deflection, length) { │ │ │ │ │ + var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ + if (previousIndex > 0) { │ │ │ │ │ + var p1 = this.line.geometry.components[previousIndex]; │ │ │ │ │ + var p0 = this.line.geometry.components[previousIndex - 1]; │ │ │ │ │ + var theta = Math.atan2(p1.y - p0.y, p1.x - p0.x); │ │ │ │ │ + this.insertDirectionLength(theta * 180 / Math.PI + deflection, length) │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.ArcIMS" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.OSM = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ - name: "OpenStreetMap", │ │ │ │ │ - url: ["http://a.tile.openstreetmap.org/${z}/${x}/${y}.png", "http://b.tile.openstreetmap.org/${z}/${x}/${y}.png", "http://c.tile.openstreetmap.org/${z}/${x}/${y}.png"], │ │ │ │ │ - attribution: "© <a href='http://www.openstreetmap.org/copyright'>OpenStreetMap</a> contributors", │ │ │ │ │ - sphericalMercator: true, │ │ │ │ │ - wrapDateLine: true, │ │ │ │ │ - tileOptions: null, │ │ │ │ │ - initialize: function(name, url, options) { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - crossOriginKeyword: "anonymous" │ │ │ │ │ - }, this.options && this.options.tileOptions) │ │ │ │ │ + getCurrentPointIndex: function() { │ │ │ │ │ + return this.line.geometry.components.length - 1 │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.OSM(this.name, this.url, this.getOptions()) │ │ │ │ │ + undo: function() { │ │ │ │ │ + var geometry = this.line.geometry; │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var index = this.getCurrentPointIndex() - 1; │ │ │ │ │ + var target = components[index]; │ │ │ │ │ + var undone = geometry.removeComponent(target); │ │ │ │ │ + if (undone) { │ │ │ │ │ + if (this.touch && index > 0) { │ │ │ │ │ + components = geometry.components; │ │ │ │ │ + var lastpt = components[index - 1]; │ │ │ │ │ + var curptidx = this.getCurrentPointIndex(); │ │ │ │ │ + var curpt = components[curptidx]; │ │ │ │ │ + curpt.x = lastpt.x; │ │ │ │ │ + curpt.y = lastpt.y │ │ │ │ │ + } │ │ │ │ │ + if (!this.redoStack) { │ │ │ │ │ + this.redoStack = [] │ │ │ │ │ + } │ │ │ │ │ + this.redoStack.push(target); │ │ │ │ │ + this.drawFeature() │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ + return undone │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.OSM" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.MapGuide = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - useHttpTile: false, │ │ │ │ │ - singleTile: false, │ │ │ │ │ - useOverlay: false, │ │ │ │ │ - useAsyncOverlay: true, │ │ │ │ │ - TILE_PARAMS: { │ │ │ │ │ - operation: "GETTILEIMAGE", │ │ │ │ │ - version: "1.2.0" │ │ │ │ │ + redo: function() { │ │ │ │ │ + var target = this.redoStack && this.redoStack.pop(); │ │ │ │ │ + if (target) { │ │ │ │ │ + this.line.geometry.addComponent(target, this.getCurrentPointIndex()); │ │ │ │ │ + this.drawFeature() │ │ │ │ │ + } │ │ │ │ │ + return !!target │ │ │ │ │ }, │ │ │ │ │ - SINGLE_TILE_PARAMS: { │ │ │ │ │ - operation: "GETMAPIMAGE", │ │ │ │ │ - format: "PNG", │ │ │ │ │ - locale: "en", │ │ │ │ │ - clip: "1", │ │ │ │ │ - version: "1.0.0" │ │ │ │ │ + freehandMode: function(evt) { │ │ │ │ │ + return this.freehandToggle && evt[this.freehandToggle] ? !this.freehand : this.freehand │ │ │ │ │ }, │ │ │ │ │ - OVERLAY_PARAMS: { │ │ │ │ │ - operation: "GETDYNAMICMAPOVERLAYIMAGE", │ │ │ │ │ - format: "PNG", │ │ │ │ │ - locale: "en", │ │ │ │ │ - clip: "1", │ │ │ │ │ - version: "2.0.0" │ │ │ │ │ + modifyFeature: function(pixel, drawing) { │ │ │ │ │ + if (!this.line) { │ │ │ │ │ + this.createFeature(pixel) │ │ │ │ │ + } │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + this.point.geometry.x = lonlat.lon; │ │ │ │ │ + this.point.geometry.y = lonlat.lat; │ │ │ │ │ + this.callback("modify", [this.point.geometry, this.getSketch(), drawing]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.drawFeature() │ │ │ │ │ }, │ │ │ │ │ - FOLDER_PARAMS: { │ │ │ │ │ - tileColumnsPerFolder: 30, │ │ │ │ │ - tileRowsPerFolder: 30, │ │ │ │ │ - format: "png", │ │ │ │ │ - querystring: null │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.line, this.style); │ │ │ │ │ + this.layer.drawFeature(this.point, this.style) │ │ │ │ │ }, │ │ │ │ │ - defaultSize: new OpenLayers.Size(300, 300), │ │ │ │ │ - tileOriginCorner: "tl", │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (options == null || options.isBaseLayer == null) { │ │ │ │ │ - this.isBaseLayer = this.transparent != "true" && this.transparent != true │ │ │ │ │ - } │ │ │ │ │ - if (options && options.useOverlay != null) { │ │ │ │ │ - this.useOverlay = options.useOverlay │ │ │ │ │ + getSketch: function() { │ │ │ │ │ + return this.line │ │ │ │ │ + }, │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.line && this.line.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiLineString([geometry]) │ │ │ │ │ } │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - if (this.useOverlay) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(this.params, this.OVERLAY_PARAMS); │ │ │ │ │ - if (!this.useAsyncOverlay) { │ │ │ │ │ - this.params.version = "1.0.0" │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Util.applyDefaults(this.params, this.SINGLE_TILE_PARAMS) │ │ │ │ │ - } │ │ │ │ │ + return geometry │ │ │ │ │ + }, │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + if (this.timerId && this.passesTolerance(this.lastTouchPx, evt.xy, this.doubleTouchTolerance)) { │ │ │ │ │ + this.finishGeometry(); │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + return false │ │ │ │ │ } else { │ │ │ │ │ - if (this.useHttpTile) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(this.params, this.FOLDER_PARAMS) │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Util.applyDefaults(this.params, this.TILE_PARAMS) │ │ │ │ │ + if (this.timerId) { │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null │ │ │ │ │ } │ │ │ │ │ - this.setTileSize(this.defaultSize) │ │ │ │ │ + this.timerId = window.setTimeout(OpenLayers.Function.bind(function() { │ │ │ │ │ + this.timerId = null │ │ │ │ │ + }, this), 300); │ │ │ │ │ + return OpenLayers.Handler.Point.prototype.touchstart.call(this, evt) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.MapGuide(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ + down: function(evt) { │ │ │ │ │ + var stopDown = this.stopDown; │ │ │ │ │ + if (this.freehandMode(evt)) { │ │ │ │ │ + stopDown = true; │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ + if (!this.touch && (!this.lastDown || !this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance))) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp) │ │ │ │ │ + } │ │ │ │ │ + this.mouseDown = true; │ │ │ │ │ + this.lastDown = evt.xy; │ │ │ │ │ + this.stoppedDown = stopDown; │ │ │ │ │ + return !stopDown │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - var url; │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var mapSize = this.map.getSize(); │ │ │ │ │ - if (this.singleTile) { │ │ │ │ │ - var params = { │ │ │ │ │ - setdisplaydpi: OpenLayers.DOTS_PER_INCH, │ │ │ │ │ - setdisplayheight: mapSize.h * this.ratio, │ │ │ │ │ - setdisplaywidth: mapSize.w * this.ratio, │ │ │ │ │ - setviewcenterx: center.lon, │ │ │ │ │ - setviewcentery: center.lat, │ │ │ │ │ - setviewscale: this.map.getScale() │ │ │ │ │ - }; │ │ │ │ │ - if (this.useOverlay && !this.useAsyncOverlay) { │ │ │ │ │ - var getVisParams = {}; │ │ │ │ │ - getVisParams = OpenLayers.Util.extend(getVisParams, params); │ │ │ │ │ - getVisParams.operation = "GETVISIBLEMAPEXTENT"; │ │ │ │ │ - getVisParams.version = "1.0.0"; │ │ │ │ │ - getVisParams.session = this.params.session; │ │ │ │ │ - getVisParams.mapName = this.params.mapName; │ │ │ │ │ - getVisParams.format = "text/xml"; │ │ │ │ │ - url = this.getFullRequestString(getVisParams); │ │ │ │ │ - OpenLayers.Request.GET({ │ │ │ │ │ - url: url, │ │ │ │ │ - async: false │ │ │ │ │ - }) │ │ │ │ │ + move: function(evt) { │ │ │ │ │ + if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature() │ │ │ │ │ } │ │ │ │ │ - url = this.getFullRequestString(params) │ │ │ │ │ - } else { │ │ │ │ │ - var currentRes = this.map.getResolution(); │ │ │ │ │ - var colidx = Math.floor((bounds.left - this.maxExtent.left) / currentRes); │ │ │ │ │ - colidx = Math.round(colidx / this.tileSize.w); │ │ │ │ │ - var rowidx = Math.floor((this.maxExtent.top - bounds.top) / currentRes); │ │ │ │ │ - rowidx = Math.round(rowidx / this.tileSize.h); │ │ │ │ │ - if (this.useHttpTile) { │ │ │ │ │ - url = this.getImageFilePath({ │ │ │ │ │ - tilecol: colidx, │ │ │ │ │ - tilerow: rowidx, │ │ │ │ │ - scaleindex: this.resolutions.length - this.map.zoom - 1 │ │ │ │ │ - }) │ │ │ │ │ + if (this.maxVertices && this.line && this.line.geometry.components.length === this.maxVertices) { │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize() │ │ │ │ │ } else { │ │ │ │ │ - url = this.getFullRequestString({ │ │ │ │ │ - tilecol: colidx, │ │ │ │ │ - tilerow: rowidx, │ │ │ │ │ - scaleindex: this.resolutions.length - this.map.zoom - 1 │ │ │ │ │ - }) │ │ │ │ │ + this.addPoint(evt.xy) │ │ │ │ │ } │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return url │ │ │ │ │ - }, │ │ │ │ │ - getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ - var url = altUrl == null ? this.url : altUrl; │ │ │ │ │ - if (typeof url == "object") { │ │ │ │ │ - url = url[Math.floor(Math.random() * url.length)] │ │ │ │ │ - } │ │ │ │ │ - var requestString = url; │ │ │ │ │ - var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ - allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ - var urlParams = OpenLayers.Util.upperCaseObject(OpenLayers.Util.getParameters(url)); │ │ │ │ │ - for (var key in allParams) { │ │ │ │ │ - if (key.toUpperCase() in urlParams) { │ │ │ │ │ - delete allParams[key] │ │ │ │ │ - } │ │ │ │ │ + if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ + this.modifyFeature(evt.xy, !!this.lastUp) │ │ │ │ │ } │ │ │ │ │ - var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ - paramsString = paramsString.replace(/,/g, "+"); │ │ │ │ │ - if (paramsString != "") { │ │ │ │ │ - var lastServerChar = url.charAt(url.length - 1); │ │ │ │ │ - if (lastServerChar == "&" || lastServerChar == "?") { │ │ │ │ │ - requestString += paramsString │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + up: function(evt) { │ │ │ │ │ + if (this.mouseDown && (!this.lastUp || !this.lastUp.equals(evt.xy))) { │ │ │ │ │ + if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ + if (this.persist) { │ │ │ │ │ + this.destroyPersistedFeature() │ │ │ │ │ + } │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize() │ │ │ │ │ } else { │ │ │ │ │ - if (url.indexOf("?") == -1) { │ │ │ │ │ - requestString += "?" + paramsString │ │ │ │ │ - } else { │ │ │ │ │ - requestString += "&" + paramsString │ │ │ │ │ + if (this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance)) { │ │ │ │ │ + if (this.touch) { │ │ │ │ │ + this.modifyFeature(evt.xy) │ │ │ │ │ + } │ │ │ │ │ + if (this.lastUp == null && this.persist) { │ │ │ │ │ + this.destroyPersistedFeature() │ │ │ │ │ + } │ │ │ │ │ + this.addPoint(evt.xy); │ │ │ │ │ + this.lastUp = evt.xy; │ │ │ │ │ + if (this.line.geometry.components.length === this.maxVertices + 1) { │ │ │ │ │ + this.finishGeometry() │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return requestString │ │ │ │ │ + this.stoppedDown = this.stopDown; │ │ │ │ │ + this.mouseDown = false; │ │ │ │ │ + return !this.stopUp │ │ │ │ │ }, │ │ │ │ │ - getImageFilePath: function(newParams, altUrl) { │ │ │ │ │ - var url = altUrl == null ? this.url : altUrl; │ │ │ │ │ - if (typeof url == "object") { │ │ │ │ │ - url = url[Math.floor(Math.random() * url.length)] │ │ │ │ │ + finishGeometry: function() { │ │ │ │ │ + var index = this.line.geometry.components.length - 1; │ │ │ │ │ + this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize() │ │ │ │ │ + }, │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + if (!this.freehandMode(evt)) { │ │ │ │ │ + this.finishGeometry() │ │ │ │ │ } │ │ │ │ │ - var requestString = url; │ │ │ │ │ - var tileRowGroup = ""; │ │ │ │ │ - var tileColGroup = ""; │ │ │ │ │ - if (newParams.tilerow < 0) { │ │ │ │ │ - tileRowGroup = "-" │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Path" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Handler.Polygon = OpenLayers.Class(OpenLayers.Handler.Path, { │ │ │ │ │ + holeModifier: null, │ │ │ │ │ + drawingHole: false, │ │ │ │ │ + polygon: null, │ │ │ │ │ + createFeature: function(pixel) { │ │ │ │ │ + var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ + this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + this.line = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.LinearRing([this.point.geometry])); │ │ │ │ │ + this.polygon = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Polygon([this.line.geometry])); │ │ │ │ │ + this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ + this.point.geometry.clearBounds(); │ │ │ │ │ + this.layer.addFeatures([this.polygon, this.point], { │ │ │ │ │ + silent: true │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + addPoint: function(pixel) { │ │ │ │ │ + if (!this.drawingHole && this.holeModifier && this.evt && this.evt[this.holeModifier]) { │ │ │ │ │ + var geometry = this.point.geometry; │ │ │ │ │ + var features = this.control.layer.features; │ │ │ │ │ + var candidate, polygon; │ │ │ │ │ + for (var i = features.length - 1; i >= 0; --i) { │ │ │ │ │ + candidate = features[i].geometry; │ │ │ │ │ + if ((candidate instanceof OpenLayers.Geometry.Polygon || candidate instanceof OpenLayers.Geometry.MultiPolygon) && candidate.intersects(geometry)) { │ │ │ │ │ + polygon = features[i]; │ │ │ │ │ + this.control.layer.removeFeatures([polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.control.layer.events.registerPriority("sketchcomplete", this, this.finalizeInteriorRing); │ │ │ │ │ + this.control.layer.events.registerPriority("sketchmodified", this, this.enforceTopology); │ │ │ │ │ + polygon.geometry.addComponent(this.line.geometry); │ │ │ │ │ + this.polygon = polygon; │ │ │ │ │ + this.drawingHole = true; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (newParams.tilerow == 0) { │ │ │ │ │ - tileRowGroup += "0" │ │ │ │ │ - } else { │ │ │ │ │ - tileRowGroup += Math.floor(Math.abs(newParams.tilerow / this.params.tileRowsPerFolder)) * this.params.tileRowsPerFolder │ │ │ │ │ + OpenLayers.Handler.Path.prototype.addPoint.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + getCurrentPointIndex: function() { │ │ │ │ │ + return this.line.geometry.components.length - 2 │ │ │ │ │ + }, │ │ │ │ │ + enforceTopology: function(event) { │ │ │ │ │ + var point = event.vertex; │ │ │ │ │ + var components = this.line.geometry.components; │ │ │ │ │ + if (!this.polygon.geometry.intersects(point)) { │ │ │ │ │ + var last = components[components.length - 3]; │ │ │ │ │ + point.x = last.x; │ │ │ │ │ + point.y = last.y │ │ │ │ │ } │ │ │ │ │ - if (newParams.tilecol < 0) { │ │ │ │ │ - tileColGroup = "-" │ │ │ │ │ + }, │ │ │ │ │ + finishGeometry: function() { │ │ │ │ │ + var index = this.line.geometry.components.length - 2; │ │ │ │ │ + this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ + this.removePoint(); │ │ │ │ │ + this.finalize() │ │ │ │ │ + }, │ │ │ │ │ + finalizeInteriorRing: function() { │ │ │ │ │ + var ring = this.line.geometry; │ │ │ │ │ + var modified = ring.getArea() !== 0; │ │ │ │ │ + if (modified) { │ │ │ │ │ + var rings = this.polygon.geometry.components; │ │ │ │ │ + for (var i = rings.length - 2; i >= 0; --i) { │ │ │ │ │ + if (ring.intersects(rings[i])) { │ │ │ │ │ + modified = false; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (modified) { │ │ │ │ │ + var target; │ │ │ │ │ + outer: for (var i = rings.length - 2; i > 0; --i) { │ │ │ │ │ + var points = rings[i].components; │ │ │ │ │ + for (var j = 0, jj = points.length; j < jj; ++j) { │ │ │ │ │ + if (ring.containsPoint(points[j])) { │ │ │ │ │ + modified = false; │ │ │ │ │ + break outer │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (newParams.tilecol == 0) { │ │ │ │ │ - tileColGroup += "0" │ │ │ │ │ + if (modified) { │ │ │ │ │ + if (this.polygon.state !== OpenLayers.State.INSERT) { │ │ │ │ │ + this.polygon.state = OpenLayers.State.UPDATE │ │ │ │ │ + } │ │ │ │ │ } else { │ │ │ │ │ - tileColGroup += Math.floor(Math.abs(newParams.tilecol / this.params.tileColumnsPerFolder)) * this.params.tileColumnsPerFolder │ │ │ │ │ - } │ │ │ │ │ - var tilePath = "/S" + Math.floor(newParams.scaleindex) + "/" + this.params.basemaplayergroupname + "/R" + tileRowGroup + "/C" + tileColGroup + "/" + newParams.tilerow % this.params.tileRowsPerFolder + "_" + newParams.tilecol % this.params.tileColumnsPerFolder + "." + this.params.format; │ │ │ │ │ - if (this.params.querystring) { │ │ │ │ │ - tilePath += "?" + this.params.querystring │ │ │ │ │ + this.polygon.geometry.removeComponent(ring) │ │ │ │ │ } │ │ │ │ │ - requestString += tilePath; │ │ │ │ │ - return requestString │ │ │ │ │ + this.restoreFeature(); │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.MapGuide" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.PointTrack = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ - dataFrom: null, │ │ │ │ │ - styleFrom: null, │ │ │ │ │ - addNodes: function(pointFeatures, options) { │ │ │ │ │ - if (pointFeatures.length < 2) { │ │ │ │ │ - throw new Error("At least two point features have to be added to " + "create a line from") │ │ │ │ │ - } │ │ │ │ │ - var lines = new Array(pointFeatures.length - 1); │ │ │ │ │ - var pointFeature, startPoint, endPoint; │ │ │ │ │ - for (var i = 0, len = pointFeatures.length; i < len; i++) { │ │ │ │ │ - pointFeature = pointFeatures[i]; │ │ │ │ │ - endPoint = pointFeature.geometry; │ │ │ │ │ - if (!endPoint) { │ │ │ │ │ - var lonlat = pointFeature.lonlat; │ │ │ │ │ - endPoint = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat) │ │ │ │ │ - } else if (endPoint.CLASS_NAME != "OpenLayers.Geometry.Point") { │ │ │ │ │ - throw new TypeError("Only features with point geometries are supported.") │ │ │ │ │ - } │ │ │ │ │ - if (i > 0) { │ │ │ │ │ - var attributes = this.dataFrom != null ? pointFeatures[i + this.dataFrom].data || pointFeatures[i + this.dataFrom].attributes : null; │ │ │ │ │ - var style = this.styleFrom != null ? pointFeatures[i + this.styleFrom].style : null; │ │ │ │ │ - var line = new OpenLayers.Geometry.LineString([startPoint, endPoint]); │ │ │ │ │ - lines[i - 1] = new OpenLayers.Feature.Vector(line, attributes, style) │ │ │ │ │ - } │ │ │ │ │ - startPoint = endPoint │ │ │ │ │ + cancel: function() { │ │ │ │ │ + if (this.drawingHole) { │ │ │ │ │ + this.polygon.geometry.removeComponent(this.line.geometry); │ │ │ │ │ + this.restoreFeature(true) │ │ │ │ │ } │ │ │ │ │ - this.addFeatures(lines, options) │ │ │ │ │ + return OpenLayers.Handler.Path.prototype.cancel.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.PointTrack" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.PointTrack.SOURCE_NODE = -1; │ │ │ │ │ -OpenLayers.Layer.PointTrack.TARGET_NODE = 0; │ │ │ │ │ -OpenLayers.Layer.PointTrack.dataFrom = { │ │ │ │ │ - SOURCE_NODE: -1, │ │ │ │ │ - TARGET_NODE: 0 │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Layer.Boxes = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ - drawMarker: function(marker) { │ │ │ │ │ - var topleft = this.map.getLayerPxFromLonLat({ │ │ │ │ │ - lon: marker.bounds.left, │ │ │ │ │ - lat: marker.bounds.top │ │ │ │ │ + restoreFeature: function(cancel) { │ │ │ │ │ + this.control.layer.events.unregister("sketchcomplete", this, this.finalizeInteriorRing); │ │ │ │ │ + this.control.layer.events.unregister("sketchmodified", this, this.enforceTopology); │ │ │ │ │ + this.layer.removeFeatures([this.polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ }); │ │ │ │ │ - var botright = this.map.getLayerPxFromLonLat({ │ │ │ │ │ - lon: marker.bounds.right, │ │ │ │ │ - lat: marker.bounds.bottom │ │ │ │ │ + this.control.layer.addFeatures([this.polygon], { │ │ │ │ │ + silent: true │ │ │ │ │ }); │ │ │ │ │ - if (botright == null || topleft == null) { │ │ │ │ │ - marker.display(false) │ │ │ │ │ - } else { │ │ │ │ │ - var markerDiv = marker.draw(topleft, { │ │ │ │ │ - w: Math.max(1, botright.x - topleft.x), │ │ │ │ │ - h: Math.max(1, botright.y - topleft.y) │ │ │ │ │ - }); │ │ │ │ │ - if (!marker.drawn) { │ │ │ │ │ - this.div.appendChild(markerDiv); │ │ │ │ │ - marker.drawn = true │ │ │ │ │ - } │ │ │ │ │ + this.drawingHole = false; │ │ │ │ │ + if (!cancel) { │ │ │ │ │ + this.control.layer.events.triggerEvent("sketchcomplete", { │ │ │ │ │ + feature: this.polygon │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - removeMarker: function(marker) { │ │ │ │ │ - OpenLayers.Util.removeItem(this.markers, marker); │ │ │ │ │ - if (marker.div != null && marker.div.parentNode == this.div) { │ │ │ │ │ - this.div.removeChild(marker.div) │ │ │ │ │ + destroyFeature: function(force) { │ │ │ │ │ + OpenLayers.Handler.Path.prototype.destroyFeature.call(this, force); │ │ │ │ │ + this.polygon = null │ │ │ │ │ + }, │ │ │ │ │ + drawFeature: function() { │ │ │ │ │ + this.layer.drawFeature(this.polygon, this.style); │ │ │ │ │ + this.layer.drawFeature(this.point, this.style) │ │ │ │ │ + }, │ │ │ │ │ + getSketch: function() { │ │ │ │ │ + return this.polygon │ │ │ │ │ + }, │ │ │ │ │ + getGeometry: function() { │ │ │ │ │ + var geometry = this.polygon && this.polygon.geometry; │ │ │ │ │ + if (geometry && this.multi) { │ │ │ │ │ + geometry = new OpenLayers.Geometry.MultiPolygon([geometry]) │ │ │ │ │ } │ │ │ │ │ + return geometry │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Boxes" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Polygon" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.Image = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - url: null, │ │ │ │ │ - extent: null, │ │ │ │ │ - size: null, │ │ │ │ │ - tile: null, │ │ │ │ │ - aspectRatio: null, │ │ │ │ │ - initialize: function(name, url, extent, size, options) { │ │ │ │ │ - this.url = url; │ │ │ │ │ - this.extent = extent; │ │ │ │ │ - this.maxExtent = extent; │ │ │ │ │ - this.size = size; │ │ │ │ │ - OpenLayers.Layer.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ - this.aspectRatio = this.extent.getHeight() / this.size.h / (this.extent.getWidth() / this.size.w) │ │ │ │ │ +OpenLayers.Handler.MouseWheel = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + wheelListener: null, │ │ │ │ │ + interval: 0, │ │ │ │ │ + maxDelta: Number.POSITIVE_INFINITY, │ │ │ │ │ + delta: 0, │ │ │ │ │ + cumulative: true, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.wheelListener = OpenLayers.Function.bindAsEventListener(this.onWheelEvent, this) │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - if (this.tile) { │ │ │ │ │ - this.removeTileMonitoringHooks(this.tile); │ │ │ │ │ - this.tile.destroy(); │ │ │ │ │ - this.tile = null │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ + OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ + this.wheelListener = null │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Image(this.name, this.url, this.extent, this.size, this.getOptions()) │ │ │ │ │ + onWheelEvent: function(e) { │ │ │ │ │ + if (!this.map || !this.checkModifiers(e)) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - if (this.options.maxResolution == null) { │ │ │ │ │ - this.options.maxResolution = this.aspectRatio * this.extent.getWidth() / this.size.w │ │ │ │ │ + var overScrollableDiv = false; │ │ │ │ │ + var allowScroll = false; │ │ │ │ │ + var overMapDiv = false; │ │ │ │ │ + var elem = OpenLayers.Event.element(e); │ │ │ │ │ + while (elem != null && !overMapDiv && !overScrollableDiv) { │ │ │ │ │ + if (!overScrollableDiv) { │ │ │ │ │ + try { │ │ │ │ │ + var overflow; │ │ │ │ │ + if (elem.currentStyle) { │ │ │ │ │ + overflow = elem.currentStyle["overflow"] │ │ │ │ │ + } else { │ │ │ │ │ + var style = document.defaultView.getComputedStyle(elem, null); │ │ │ │ │ + overflow = style.getPropertyValue("overflow") │ │ │ │ │ + } │ │ │ │ │ + overScrollableDiv = overflow && overflow == "auto" || overflow == "scroll" │ │ │ │ │ + } catch (err) {} │ │ │ │ │ + } │ │ │ │ │ + if (!allowScroll) { │ │ │ │ │ + allowScroll = OpenLayers.Element.hasClass(elem, "olScrollable"); │ │ │ │ │ + if (!allowScroll) { │ │ │ │ │ + for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if (elem == layer.div || elem == layer.pane) { │ │ │ │ │ + allowScroll = true; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + overMapDiv = elem == this.map.div; │ │ │ │ │ + elem = elem.parentNode │ │ │ │ │ + } │ │ │ │ │ + if (!overScrollableDiv && overMapDiv) { │ │ │ │ │ + if (allowScroll) { │ │ │ │ │ + var delta = 0; │ │ │ │ │ + if (e.wheelDelta) { │ │ │ │ │ + delta = e.wheelDelta; │ │ │ │ │ + if (delta % 160 === 0) { │ │ │ │ │ + delta = delta * .75 │ │ │ │ │ + } │ │ │ │ │ + delta = delta / 120 │ │ │ │ │ + } else if (e.detail) { │ │ │ │ │ + delta = -(e.detail / Math.abs(e.detail)) │ │ │ │ │ + } │ │ │ │ │ + this.delta += delta; │ │ │ │ │ + window.clearTimeout(this._timeoutId); │ │ │ │ │ + if (this.interval && Math.abs(this.delta) < this.maxDelta) { │ │ │ │ │ + var evt = OpenLayers.Util.extend({}, e); │ │ │ │ │ + this._timeoutId = window.setTimeout(OpenLayers.Function.bind(function() { │ │ │ │ │ + this.wheelZoom(evt) │ │ │ │ │ + }, this), this.interval) │ │ │ │ │ + } else { │ │ │ │ │ + this.wheelZoom(e) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stop(e) │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.prototype.setMap.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - var firstRendering = this.tile == null; │ │ │ │ │ - if (zoomChanged || firstRendering) { │ │ │ │ │ - this.setTileSize(); │ │ │ │ │ - var ulPx = this.map.getLayerPxFromLonLat({ │ │ │ │ │ - lon: this.extent.left, │ │ │ │ │ - lat: this.extent.top │ │ │ │ │ - }); │ │ │ │ │ - if (firstRendering) { │ │ │ │ │ - this.tile = new OpenLayers.Tile.Image(this, ulPx, this.extent, null, this.tileSize); │ │ │ │ │ - this.addTileMonitoringHooks(this.tile) │ │ │ │ │ + wheelZoom: function(e) { │ │ │ │ │ + var delta = this.delta; │ │ │ │ │ + this.delta = 0; │ │ │ │ │ + if (delta) { │ │ │ │ │ + e.xy = this.map.events.getMousePosition(e); │ │ │ │ │ + if (delta < 0) { │ │ │ │ │ + this.callback("down", [e, this.cumulative ? Math.max(-this.maxDelta, delta) : -1]) │ │ │ │ │ } else { │ │ │ │ │ - this.tile.size = this.tileSize.clone(); │ │ │ │ │ - this.tile.position = ulPx.clone() │ │ │ │ │ + this.callback("up", [e, this.cumulative ? Math.min(this.maxDelta, delta) : 1]) │ │ │ │ │ } │ │ │ │ │ - this.tile.draw() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - setTileSize: function() { │ │ │ │ │ - var tileWidth = this.extent.getWidth() / this.map.getResolution(); │ │ │ │ │ - var tileHeight = this.extent.getHeight() / this.map.getResolution(); │ │ │ │ │ - this.tileSize = new OpenLayers.Size(tileWidth, tileHeight) │ │ │ │ │ - }, │ │ │ │ │ - addTileMonitoringHooks: function(tile) { │ │ │ │ │ - tile.onLoadStart = function() { │ │ │ │ │ - this.events.triggerEvent("loadstart") │ │ │ │ │ - }; │ │ │ │ │ - tile.events.register("loadstart", this, tile.onLoadStart); │ │ │ │ │ - tile.onLoadEnd = function() { │ │ │ │ │ - this.events.triggerEvent("loadend") │ │ │ │ │ - }; │ │ │ │ │ - tile.events.register("loadend", this, tile.onLoadEnd); │ │ │ │ │ - tile.events.register("unload", this, tile.onLoadEnd) │ │ │ │ │ - }, │ │ │ │ │ - removeTileMonitoringHooks: function(tile) { │ │ │ │ │ - tile.unload(); │ │ │ │ │ - tile.events.un({ │ │ │ │ │ - loadstart: tile.onLoadStart, │ │ │ │ │ - loadend: tile.onLoadEnd, │ │ │ │ │ - unload: tile.onLoadEnd, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - setUrl: function(newUrl) { │ │ │ │ │ - this.url = newUrl; │ │ │ │ │ - this.tile.draw() │ │ │ │ │ + activate: function(evt) { │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + var wheelListener = this.wheelListener; │ │ │ │ │ + OpenLayers.Event.observe(window, "DOMMouseScroll", wheelListener); │ │ │ │ │ + OpenLayers.Event.observe(window, "mousewheel", wheelListener); │ │ │ │ │ + OpenLayers.Event.observe(document, "mousewheel", wheelListener); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - return this.url │ │ │ │ │ + deactivate: function(evt) { │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + var wheelListener = this.wheelListener; │ │ │ │ │ + OpenLayers.Event.stopObserving(window, "DOMMouseScroll", wheelListener); │ │ │ │ │ + OpenLayers.Event.stopObserving(window, "mousewheel", wheelListener); │ │ │ │ │ + OpenLayers.Event.stopObserving(document, "mousewheel", wheelListener); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Image" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.MouseWheel" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.PointGrid = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ - dx: null, │ │ │ │ │ - dy: null, │ │ │ │ │ - ratio: 1.5, │ │ │ │ │ - maxFeatures: 250, │ │ │ │ │ - rotation: 0, │ │ │ │ │ - origin: null, │ │ │ │ │ - gridBounds: null, │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ - config = config || {}; │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.initialize.apply(this, [config.name, config]) │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ - map.events.register("moveend", this, this.onMoveEnd) │ │ │ │ │ +OpenLayers.Handler.Pinch = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + started: false, │ │ │ │ │ + stopDown: false, │ │ │ │ │ + pinching: false, │ │ │ │ │ + last: null, │ │ │ │ │ + start: null, │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + if (OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ + this.started = true; │ │ │ │ │ + this.last = this.start = { │ │ │ │ │ + distance: this.getDistance(evt.touches), │ │ │ │ │ + delta: 0, │ │ │ │ │ + scale: 1 │ │ │ │ │ + }; │ │ │ │ │ + this.callback("start", [evt, this.start]); │ │ │ │ │ + propagate = !this.stopDown │ │ │ │ │ + } else if (this.started) { │ │ │ │ │ + return false │ │ │ │ │ + } else { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.preventDefault(evt); │ │ │ │ │ + return propagate │ │ │ │ │ }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - map.events.unregister("moveend", this, this.onMoveEnd); │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments) │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + if (this.started && OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ + this.pinching = true; │ │ │ │ │ + var current = this.getPinchData(evt); │ │ │ │ │ + this.callback("move", [evt, current]); │ │ │ │ │ + this.last = current; │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ + } else if (this.started) { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - setRatio: function(ratio) { │ │ │ │ │ - this.ratio = ratio; │ │ │ │ │ - this.updateGrid(true) │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + if (this.started && !OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + this.callback("done", [evt, this.start, this.last]); │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - setMaxFeatures: function(maxFeatures) { │ │ │ │ │ - this.maxFeatures = maxFeatures; │ │ │ │ │ - this.updateGrid(true) │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + activated = true │ │ │ │ │ + } │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - setSpacing: function(dx, dy) { │ │ │ │ │ - this.dx = dx; │ │ │ │ │ - this.dy = dy || dx; │ │ │ │ │ - this.updateGrid(true) │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.started = false; │ │ │ │ │ + this.pinching = false; │ │ │ │ │ + this.start = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + deactivated = true │ │ │ │ │ + } │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - setOrigin: function(origin) { │ │ │ │ │ - this.origin = origin; │ │ │ │ │ - this.updateGrid(true) │ │ │ │ │ + getDistance: function(touches) { │ │ │ │ │ + var t0 = touches[0]; │ │ │ │ │ + var t1 = touches[1]; │ │ │ │ │ + return Math.sqrt(Math.pow(t0.olClientX - t1.olClientX, 2) + Math.pow(t0.olClientY - t1.olClientY, 2)) │ │ │ │ │ }, │ │ │ │ │ - getOrigin: function() { │ │ │ │ │ - if (!this.origin) { │ │ │ │ │ - this.origin = this.map.getExtent().getCenterLonLat() │ │ │ │ │ + getPinchData: function(evt) { │ │ │ │ │ + var distance = this.getDistance(evt.touches); │ │ │ │ │ + var scale = distance / this.start.distance; │ │ │ │ │ + return { │ │ │ │ │ + distance: distance, │ │ │ │ │ + delta: this.last.distance - distance, │ │ │ │ │ + scale: scale │ │ │ │ │ } │ │ │ │ │ - return this.origin │ │ │ │ │ - }, │ │ │ │ │ - setRotation: function(rotation) { │ │ │ │ │ - this.rotation = rotation; │ │ │ │ │ - this.updateGrid(true) │ │ │ │ │ }, │ │ │ │ │ - onMoveEnd: function() { │ │ │ │ │ - this.updateGrid() │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Pinch" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Handler.Box = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + dragHandler: null, │ │ │ │ │ + boxDivClassName: "olHandlerBoxZoomBox", │ │ │ │ │ + boxOffsets: null, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.dragHandler = new OpenLayers.Handler.Drag(this, { │ │ │ │ │ + down: this.startBox, │ │ │ │ │ + move: this.moveBox, │ │ │ │ │ + out: this.removeBox, │ │ │ │ │ + up: this.endBox │ │ │ │ │ + }, { │ │ │ │ │ + keyMask: this.keyMask │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - getViewBounds: function() { │ │ │ │ │ - var bounds = this.map.getExtent(); │ │ │ │ │ - if (this.rotation) { │ │ │ │ │ - var origin = this.getOrigin(); │ │ │ │ │ - var rotationOrigin = new OpenLayers.Geometry.Point(origin.lon, origin.lat); │ │ │ │ │ - var rect = bounds.toGeometry(); │ │ │ │ │ - rect.rotate(-this.rotation, rotationOrigin); │ │ │ │ │ - bounds = rect.getBounds() │ │ │ │ │ + destroy: function() { │ │ │ │ │ + OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ + if (this.dragHandler) { │ │ │ │ │ + this.dragHandler.destroy(); │ │ │ │ │ + this.dragHandler = null │ │ │ │ │ } │ │ │ │ │ - return bounds │ │ │ │ │ }, │ │ │ │ │ - updateGrid: function(force) { │ │ │ │ │ - if (force || this.invalidBounds()) { │ │ │ │ │ - var viewBounds = this.getViewBounds(); │ │ │ │ │ - var origin = this.getOrigin(); │ │ │ │ │ - var rotationOrigin = new OpenLayers.Geometry.Point(origin.lon, origin.lat); │ │ │ │ │ - var viewBoundsWidth = viewBounds.getWidth(); │ │ │ │ │ - var viewBoundsHeight = viewBounds.getHeight(); │ │ │ │ │ - var aspectRatio = viewBoundsWidth / viewBoundsHeight; │ │ │ │ │ - var maxHeight = Math.sqrt(this.dx * this.dy * this.maxFeatures / aspectRatio); │ │ │ │ │ - var maxWidth = maxHeight * aspectRatio; │ │ │ │ │ - var gridWidth = Math.min(viewBoundsWidth * this.ratio, maxWidth); │ │ │ │ │ - var gridHeight = Math.min(viewBoundsHeight * this.ratio, maxHeight); │ │ │ │ │ - var center = viewBounds.getCenterLonLat(); │ │ │ │ │ - this.gridBounds = new OpenLayers.Bounds(center.lon - gridWidth / 2, center.lat - gridHeight / 2, center.lon + gridWidth / 2, center.lat + gridHeight / 2); │ │ │ │ │ - var rows = Math.floor(gridHeight / this.dy); │ │ │ │ │ - var cols = Math.floor(gridWidth / this.dx); │ │ │ │ │ - var gridLeft = origin.lon + this.dx * Math.ceil((this.gridBounds.left - origin.lon) / this.dx); │ │ │ │ │ - var gridBottom = origin.lat + this.dy * Math.ceil((this.gridBounds.bottom - origin.lat) / this.dy); │ │ │ │ │ - var features = new Array(rows * cols); │ │ │ │ │ - var x, y, point; │ │ │ │ │ - for (var i = 0; i < cols; ++i) { │ │ │ │ │ - x = gridLeft + i * this.dx; │ │ │ │ │ - for (var j = 0; j < rows; ++j) { │ │ │ │ │ - y = gridBottom + j * this.dy; │ │ │ │ │ - point = new OpenLayers.Geometry.Point(x, y); │ │ │ │ │ - if (this.rotation) { │ │ │ │ │ - point.rotate(this.rotation, rotationOrigin) │ │ │ │ │ - } │ │ │ │ │ - features[i * rows + j] = new OpenLayers.Feature.Vector(point) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.destroyFeatures(this.features, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.addFeatures(features, { │ │ │ │ │ - silent: true │ │ │ │ │ - }) │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Handler.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (this.dragHandler) { │ │ │ │ │ + this.dragHandler.setMap(map) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - invalidBounds: function() { │ │ │ │ │ - return !this.gridBounds || !this.gridBounds.containsBounds(this.getViewBounds()) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.PointGrid" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.GeoRSS = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ - location: null, │ │ │ │ │ - features: null, │ │ │ │ │ - formatOptions: null, │ │ │ │ │ - selectedFeature: null, │ │ │ │ │ - icon: null, │ │ │ │ │ - popupSize: null, │ │ │ │ │ - useFeedTitle: true, │ │ │ │ │ - initialize: function(name, location, options) { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ - this.location = location; │ │ │ │ │ - this.features = [] │ │ │ │ │ + startBox: function(xy) { │ │ │ │ │ + this.callback("start", []); │ │ │ │ │ + this.zoomBox = OpenLayers.Util.createDiv("zoomBox", { │ │ │ │ │ + x: -9999, │ │ │ │ │ + y: -9999 │ │ │ │ │ + }); │ │ │ │ │ + this.zoomBox.className = this.boxDivClassName; │ │ │ │ │ + this.zoomBox.style.zIndex = this.map.Z_INDEX_BASE["Popup"] - 1; │ │ │ │ │ + this.map.viewPortDiv.appendChild(this.zoomBox); │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, "olDrawBox") │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.destroy.apply(this, arguments); │ │ │ │ │ - this.clearFeatures(); │ │ │ │ │ - this.features = null │ │ │ │ │ + moveBox: function(xy) { │ │ │ │ │ + var startX = this.dragHandler.start.x; │ │ │ │ │ + var startY = this.dragHandler.start.y; │ │ │ │ │ + var deltaX = Math.abs(startX - xy.x); │ │ │ │ │ + var deltaY = Math.abs(startY - xy.y); │ │ │ │ │ + var offset = this.getBoxOffsets(); │ │ │ │ │ + this.zoomBox.style.width = deltaX + offset.width + 1 + "px"; │ │ │ │ │ + this.zoomBox.style.height = deltaY + offset.height + 1 + "px"; │ │ │ │ │ + this.zoomBox.style.left = (xy.x < startX ? startX - deltaX - offset.left : startX - offset.left) + "px"; │ │ │ │ │ + this.zoomBox.style.top = (xy.y < startY ? startY - deltaY - offset.top : startY - offset.top) + "px" │ │ │ │ │ }, │ │ │ │ │ - loadRSS: function() { │ │ │ │ │ - if (!this.loaded) { │ │ │ │ │ - this.events.triggerEvent("loadstart"); │ │ │ │ │ - OpenLayers.Request.GET({ │ │ │ │ │ - url: this.location, │ │ │ │ │ - success: this.parseData, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.loaded = true │ │ │ │ │ + endBox: function(end) { │ │ │ │ │ + var result; │ │ │ │ │ + if (Math.abs(this.dragHandler.start.x - end.x) > 5 || Math.abs(this.dragHandler.start.y - end.y) > 5) { │ │ │ │ │ + var start = this.dragHandler.start; │ │ │ │ │ + var top = Math.min(start.y, end.y); │ │ │ │ │ + var bottom = Math.max(start.y, end.y); │ │ │ │ │ + var left = Math.min(start.x, end.x); │ │ │ │ │ + var right = Math.max(start.x, end.x); │ │ │ │ │ + result = new OpenLayers.Bounds(left, bottom, right, top) │ │ │ │ │ + } else { │ │ │ │ │ + result = this.dragHandler.start.clone() │ │ │ │ │ } │ │ │ │ │ + this.removeBox(); │ │ │ │ │ + this.callback("done", [result]) │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, minor) { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - if (this.visibility && !this.loaded) { │ │ │ │ │ - this.loadRSS() │ │ │ │ │ - } │ │ │ │ │ + removeBox: function() { │ │ │ │ │ + this.map.viewPortDiv.removeChild(this.zoomBox); │ │ │ │ │ + this.zoomBox = null; │ │ │ │ │ + this.boxOffsets = null; │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDrawBox") │ │ │ │ │ }, │ │ │ │ │ - parseData: function(ajaxRequest) { │ │ │ │ │ - var doc = ajaxRequest.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = OpenLayers.Format.XML.prototype.read(ajaxRequest.responseText) │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.dragHandler.activate(); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - if (this.useFeedTitle) { │ │ │ │ │ - var name = null; │ │ │ │ │ - try { │ │ │ │ │ - name = doc.getElementsByTagNameNS("*", "title")[0].firstChild.nodeValue │ │ │ │ │ - } catch (e) { │ │ │ │ │ - try { │ │ │ │ │ - name = doc.getElementsByTagName("title")[0].firstChild.nodeValue │ │ │ │ │ - } catch (e) {} │ │ │ │ │ - } │ │ │ │ │ - if (name) { │ │ │ │ │ - this.setName(name) │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + if (this.dragHandler.deactivate()) { │ │ │ │ │ + if (this.zoomBox) { │ │ │ │ │ + this.removeBox() │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - 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 format = new OpenLayers.Format.GeoRSS(options); │ │ │ │ │ - var features = format.read(doc); │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - var data = {}; │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - if (!feature.geometry) { │ │ │ │ │ - continue │ │ │ │ │ - } │ │ │ │ │ - var title = feature.attributes.title ? feature.attributes.title : "Untitled"; │ │ │ │ │ - var description = feature.attributes.description ? feature.attributes.description : "No description."; │ │ │ │ │ - var link = feature.attributes.link ? feature.attributes.link : ""; │ │ │ │ │ - var location = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - data.icon = this.icon == null ? OpenLayers.Marker.defaultIcon() : this.icon.clone(); │ │ │ │ │ - data.popupSize = this.popupSize ? this.popupSize.clone() : new OpenLayers.Size(250, 120); │ │ │ │ │ - if (title || description) { │ │ │ │ │ - data.title = title; │ │ │ │ │ - data.description = description; │ │ │ │ │ - var contentHTML = '<div class="olLayerGeoRSSClose">[x]</div>'; │ │ │ │ │ - contentHTML += '<div class="olLayerGeoRSSTitle">'; │ │ │ │ │ - if (link) { │ │ │ │ │ - contentHTML += '<a class="link" href="' + link + '" target="_blank">' │ │ │ │ │ - } │ │ │ │ │ - contentHTML += title; │ │ │ │ │ - if (link) { │ │ │ │ │ - contentHTML += "</a>" │ │ │ │ │ - } │ │ │ │ │ - contentHTML += "</div>"; │ │ │ │ │ - contentHTML += '<div style="" class="olLayerGeoRSSDescription">'; │ │ │ │ │ - contentHTML += description; │ │ │ │ │ - contentHTML += "</div>"; │ │ │ │ │ - data["popupContentHTML"] = contentHTML │ │ │ │ │ + }, │ │ │ │ │ + getBoxOffsets: function() { │ │ │ │ │ + if (!this.boxOffsets) { │ │ │ │ │ + var testDiv = document.createElement("div"); │ │ │ │ │ + testDiv.style.position = "absolute"; │ │ │ │ │ + testDiv.style.border = "1px solid black"; │ │ │ │ │ + testDiv.style.width = "3px"; │ │ │ │ │ + document.body.appendChild(testDiv); │ │ │ │ │ + var w3cBoxModel = testDiv.clientWidth == 3; │ │ │ │ │ + document.body.removeChild(testDiv); │ │ │ │ │ + var left = parseInt(OpenLayers.Element.getStyle(this.zoomBox, "border-left-width")); │ │ │ │ │ + var right = parseInt(OpenLayers.Element.getStyle(this.zoomBox, "border-right-width")); │ │ │ │ │ + var top = parseInt(OpenLayers.Element.getStyle(this.zoomBox, "border-top-width")); │ │ │ │ │ + var bottom = parseInt(OpenLayers.Element.getStyle(this.zoomBox, "border-bottom-width")); │ │ │ │ │ + this.boxOffsets = { │ │ │ │ │ + left: left, │ │ │ │ │ + right: right, │ │ │ │ │ + top: top, │ │ │ │ │ + bottom: bottom, │ │ │ │ │ + width: w3cBoxModel === false ? left + right : 0, │ │ │ │ │ + height: w3cBoxModel === false ? top + bottom : 0 │ │ │ │ │ } │ │ │ │ │ - var feature = new OpenLayers.Feature(this, location, data); │ │ │ │ │ - this.features.push(feature); │ │ │ │ │ - var marker = feature.createMarker(); │ │ │ │ │ - marker.events.register("click", feature, this.markerClick); │ │ │ │ │ - this.addMarker(marker) │ │ │ │ │ } │ │ │ │ │ - this.events.triggerEvent("loadend") │ │ │ │ │ + return this.boxOffsets │ │ │ │ │ }, │ │ │ │ │ - markerClick: function(evt) { │ │ │ │ │ - var sameMarkerClicked = this == this.layer.selectedFeature; │ │ │ │ │ - this.layer.selectedFeature = !sameMarkerClicked ? this : null; │ │ │ │ │ - for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ - this.layer.map.removePopup(this.layer.map.popups[i]) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Box" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Handler.Hover = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + delay: 500, │ │ │ │ │ + pixelTolerance: null, │ │ │ │ │ + stopMove: false, │ │ │ │ │ + px: null, │ │ │ │ │ + timerId: null, │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + if (this.passesTolerance(evt.xy)) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.callback("move", [evt]); │ │ │ │ │ + this.px = evt.xy; │ │ │ │ │ + evt = OpenLayers.Util.extend({}, evt); │ │ │ │ │ + this.timerId = window.setTimeout(OpenLayers.Function.bind(this.delayedCall, this, evt), this.delay) │ │ │ │ │ } │ │ │ │ │ - if (!sameMarkerClicked) { │ │ │ │ │ - var popup = this.createPopup(); │ │ │ │ │ - OpenLayers.Event.observe(popup.div, "click", OpenLayers.Function.bind(function() { │ │ │ │ │ - for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ - this.layer.map.removePopup(this.layer.map.popups[i]) │ │ │ │ │ - } │ │ │ │ │ - }, this)); │ │ │ │ │ - this.layer.map.addPopup(popup) │ │ │ │ │ + return !this.stopMove │ │ │ │ │ + }, │ │ │ │ │ + mouseout: function(evt) { │ │ │ │ │ + if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.callback("move", [evt]) │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - clearFeatures: function() { │ │ │ │ │ - if (this.features != null) { │ │ │ │ │ - while (this.features.length > 0) { │ │ │ │ │ - var feature = this.features[0]; │ │ │ │ │ - OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ - feature.destroy() │ │ │ │ │ + passesTolerance: function(px) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + if (this.pixelTolerance && this.px) { │ │ │ │ │ + var dpx = Math.sqrt(Math.pow(this.px.x - px.x, 2) + Math.pow(this.px.y - px.y, 2)); │ │ │ │ │ + if (dpx < this.pixelTolerance) { │ │ │ │ │ + passes = false │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return passes │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.GeoRSS" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Google = OpenLayers.Class(OpenLayers.Layer.EventPane, OpenLayers.Layer.FixedZoomLevels, { │ │ │ │ │ - MIN_ZOOM_LEVEL: 0, │ │ │ │ │ - MAX_ZOOM_LEVEL: 21, │ │ │ │ │ - RESOLUTIONS: [1.40625, .703125, .3515625, .17578125, .087890625, .0439453125, .02197265625, .010986328125, .0054931640625, .00274658203125, .001373291015625, .0006866455078125, .00034332275390625, .000171661376953125, 858306884765625e-19, 4291534423828125e-20, 2145767211914062e-20, 1072883605957031e-20, 536441802978515e-20, 268220901489257e-20, 1341104507446289e-21, 6.705522537231445e-7], │ │ │ │ │ - type: null, │ │ │ │ │ - wrapDateLine: true, │ │ │ │ │ - sphericalMercator: false, │ │ │ │ │ - version: null, │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - if (!options.version) { │ │ │ │ │ - options.version = typeof GMap2 === "function" ? "2" : "3" │ │ │ │ │ - } │ │ │ │ │ - var mixin = OpenLayers.Layer.Google["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (mixin) { │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, mixin) │ │ │ │ │ - } else { │ │ │ │ │ - throw "Unsupported Google Maps API version: " + options.version │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, mixin.DEFAULTS); │ │ │ │ │ - if (options.maxExtent) { │ │ │ │ │ - options.maxExtent = options.maxExtent.clone() │ │ │ │ │ + clearTimer: function() { │ │ │ │ │ + if (this.timerId != null) { │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ - OpenLayers.Layer.FixedZoomLevels.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - OpenLayers.Util.extend(this, OpenLayers.Layer.SphericalMercator); │ │ │ │ │ - this.initMercatorParameters() │ │ │ │ │ + }, │ │ │ │ │ + delayedCall: function(evt) { │ │ │ │ │ + this.callback("pause", [evt]) │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + deactivated = true │ │ │ │ │ } │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - clone: function() { │ │ │ │ │ - return new OpenLayers.Layer.Google(this.name, this.getOptions()) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Hover" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Handler.Click = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + delay: 300, │ │ │ │ │ + single: true, │ │ │ │ │ + double: false, │ │ │ │ │ + pixelTolerance: 0, │ │ │ │ │ + dblclickTolerance: 13, │ │ │ │ │ + stopSingle: false, │ │ │ │ │ + stopDouble: false, │ │ │ │ │ + timerId: null, │ │ │ │ │ + down: null, │ │ │ │ │ + last: null, │ │ │ │ │ + first: null, │ │ │ │ │ + rightclickTimerId: null, │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + this.down = this.getEventInfo(evt); │ │ │ │ │ + this.last = this.getEventInfo(evt); │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - setVisibility: function(visible) { │ │ │ │ │ - var opacity = this.opacity == null ? 1 : this.opacity; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.setVisibility.apply(this, arguments); │ │ │ │ │ - this.setOpacity(opacity) │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + this.last = this.getEventInfo(evt); │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - display: function(visible) { │ │ │ │ │ - if (!this._dragging) { │ │ │ │ │ - this.setGMapVisibility(visible) │ │ │ │ │ + touchend: function(evt) { │ │ │ │ │ + if (this.down) { │ │ │ │ │ + evt.xy = this.last.xy; │ │ │ │ │ + evt.lastTouches = this.last.touches; │ │ │ │ │ + this.handleSingle(evt); │ │ │ │ │ + this.down = null │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.display.apply(this, arguments) │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - this._dragging = dragging; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - delete this._dragging │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + this.down = this.getEventInfo(evt); │ │ │ │ │ + this.last = this.getEventInfo(evt); │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - setOpacity: function(opacity) { │ │ │ │ │ - if (opacity !== this.opacity) { │ │ │ │ │ - if (this.map != null) { │ │ │ │ │ - this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "opacity" │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.opacity = opacity │ │ │ │ │ - } │ │ │ │ │ - if (this.getVisibility()) { │ │ │ │ │ - var container = this.getMapContainer(); │ │ │ │ │ - OpenLayers.Util.modifyDOMElement(container, null, null, null, null, null, null, opacity) │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + if (this.checkModifiers(evt) && this.control.handleRightClicks && OpenLayers.Event.isRightClick(evt)) { │ │ │ │ │ + propagate = this.rightclick(evt) │ │ │ │ │ } │ │ │ │ │ + return propagate │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.setGMapVisibility(false); │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache && cache.count <= 1) { │ │ │ │ │ - this.removeGMapElements() │ │ │ │ │ + rightclick: function(evt) { │ │ │ │ │ + if (this.passesTolerance(evt)) { │ │ │ │ │ + if (this.rightclickTimerId != null) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.callback("dblrightclick", [evt]); │ │ │ │ │ + return !this.stopDouble │ │ │ │ │ + } else { │ │ │ │ │ + var clickEvent = this["double"] ? OpenLayers.Util.extend({}, evt) : this.callback("rightclick", [evt]); │ │ │ │ │ + var delayedRightCall = OpenLayers.Function.bind(this.delayedRightCall, this, clickEvent); │ │ │ │ │ + this.rightclickTimerId = window.setTimeout(delayedRightCall, this.delay) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.destroy.apply(this, arguments) │ │ │ │ │ + return !this.stopSingle │ │ │ │ │ }, │ │ │ │ │ - removeGMapElements: function() { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - var container = this.mapObject && this.getMapContainer(); │ │ │ │ │ - if (container && container.parentNode) { │ │ │ │ │ - container.parentNode.removeChild(container) │ │ │ │ │ - } │ │ │ │ │ - var termsOfUse = cache.termsOfUse; │ │ │ │ │ - if (termsOfUse && termsOfUse.parentNode) { │ │ │ │ │ - termsOfUse.parentNode.removeChild(termsOfUse) │ │ │ │ │ - } │ │ │ │ │ - var poweredBy = cache.poweredBy; │ │ │ │ │ - if (poweredBy && poweredBy.parentNode) { │ │ │ │ │ - poweredBy.parentNode.removeChild(poweredBy) │ │ │ │ │ - } │ │ │ │ │ - if (this.mapObject && window.google && google.maps && google.maps.event && google.maps.event.clearListeners) { │ │ │ │ │ - google.maps.event.clearListeners(this.mapObject, "tilesloaded") │ │ │ │ │ - } │ │ │ │ │ + delayedRightCall: function(evt) { │ │ │ │ │ + this.rightclickTimerId = null; │ │ │ │ │ + if (evt) { │ │ │ │ │ + this.callback("rightclick", [evt]) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - if (this.visibility && this.mapObject) { │ │ │ │ │ - this.setGMapVisibility(false) │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + if (!this.last) { │ │ │ │ │ + this.last = this.getEventInfo(evt) │ │ │ │ │ } │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - if (cache.count <= 1) { │ │ │ │ │ - this.removeGMapElements(); │ │ │ │ │ - delete OpenLayers.Layer.Google.cache[map.id] │ │ │ │ │ - } else { │ │ │ │ │ - --cache.count │ │ │ │ │ + this.handleSingle(evt); │ │ │ │ │ + return !this.stopSingle │ │ │ │ │ + }, │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + this.handleDouble(evt); │ │ │ │ │ + return !this.stopDouble │ │ │ │ │ + }, │ │ │ │ │ + handleDouble: function(evt) { │ │ │ │ │ + if (this.passesDblclickTolerance(evt)) { │ │ │ │ │ + if (this["double"]) { │ │ │ │ │ + this.callback("dblclick", [evt]) │ │ │ │ │ } │ │ │ │ │ + this.clearTimer() │ │ │ │ │ } │ │ │ │ │ - delete this.termsOfUse; │ │ │ │ │ - delete this.poweredBy; │ │ │ │ │ - delete this.mapObject; │ │ │ │ │ - delete this.dragObject; │ │ │ │ │ - OpenLayers.Layer.EventPane.prototype.removeMap.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - getOLBoundsFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - var olBounds = null; │ │ │ │ │ - if (moBounds != null) { │ │ │ │ │ - var sw = moBounds.getSouthWest(); │ │ │ │ │ - var ne = moBounds.getNorthEast(); │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - sw = this.forwardMercator(sw.lng(), sw.lat()); │ │ │ │ │ - ne = this.forwardMercator(ne.lng(), ne.lat()) │ │ │ │ │ + handleSingle: function(evt) { │ │ │ │ │ + if (this.passesTolerance(evt)) { │ │ │ │ │ + if (this.timerId != null) { │ │ │ │ │ + if (this.last.touches && this.last.touches.length === 1) { │ │ │ │ │ + if (this["double"]) { │ │ │ │ │ + OpenLayers.Event.preventDefault(evt) │ │ │ │ │ + } │ │ │ │ │ + this.handleDouble(evt) │ │ │ │ │ + } │ │ │ │ │ + if (!this.last.touches || this.last.touches.length !== 2) { │ │ │ │ │ + this.clearTimer() │ │ │ │ │ + } │ │ │ │ │ } else { │ │ │ │ │ - sw = new OpenLayers.LonLat(sw.lng(), sw.lat()); │ │ │ │ │ - ne = new OpenLayers.LonLat(ne.lng(), ne.lat()) │ │ │ │ │ + this.first = this.getEventInfo(evt); │ │ │ │ │ + var clickEvent = this.single ? OpenLayers.Util.extend({}, evt) : null; │ │ │ │ │ + this.queuePotentialClick(clickEvent) │ │ │ │ │ } │ │ │ │ │ - olBounds = new OpenLayers.Bounds(sw.lon, sw.lat, ne.lon, ne.lat) │ │ │ │ │ } │ │ │ │ │ - return olBounds │ │ │ │ │ - }, │ │ │ │ │ - getWarningHTML: function() { │ │ │ │ │ - return OpenLayers.i18n("googleWarning") │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectCenter: function() { │ │ │ │ │ - return this.mapObject.getCenter() │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectZoom: function() { │ │ │ │ │ - return this.mapObject.getZoom() │ │ │ │ │ - }, │ │ │ │ │ - getLongitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - return this.sphericalMercator ? this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lon : moLonLat.lng() │ │ │ │ │ - }, │ │ │ │ │ - getLatitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - var lat = this.sphericalMercator ? this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lat : moLonLat.lat(); │ │ │ │ │ - return lat │ │ │ │ │ }, │ │ │ │ │ - getXFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return moPixel.x │ │ │ │ │ - }, │ │ │ │ │ - getYFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return moPixel.y │ │ │ │ │ + queuePotentialClick: function(evt) { │ │ │ │ │ + this.timerId = window.setTimeout(OpenLayers.Function.bind(this.delayedCall, this, evt), this.delay) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Google" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Google.cache = {}; │ │ │ │ │ -OpenLayers.Layer.Google.v2 = { │ │ │ │ │ - termsOfUse: null, │ │ │ │ │ - poweredBy: null, │ │ │ │ │ - dragObject: null, │ │ │ │ │ - loadMapObject: function() { │ │ │ │ │ - if (!this.type) { │ │ │ │ │ - this.type = G_NORMAL_MAP │ │ │ │ │ - } │ │ │ │ │ - var mapObject, termsOfUse, poweredBy; │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - mapObject = cache.mapObject; │ │ │ │ │ - termsOfUse = cache.termsOfUse; │ │ │ │ │ - poweredBy = cache.poweredBy; │ │ │ │ │ - ++cache.count │ │ │ │ │ - } else { │ │ │ │ │ - var container = this.map.viewPortDiv; │ │ │ │ │ - var div = document.createElement("div"); │ │ │ │ │ - div.id = this.map.id + "_GMap2Container"; │ │ │ │ │ - div.style.position = "absolute"; │ │ │ │ │ - div.style.width = "100%"; │ │ │ │ │ - div.style.height = "100%"; │ │ │ │ │ - container.appendChild(div); │ │ │ │ │ - try { │ │ │ │ │ - mapObject = new GMap2(div); │ │ │ │ │ - termsOfUse = div.lastChild; │ │ │ │ │ - container.appendChild(termsOfUse); │ │ │ │ │ - termsOfUse.style.zIndex = "1100"; │ │ │ │ │ - termsOfUse.style.right = ""; │ │ │ │ │ - termsOfUse.style.bottom = ""; │ │ │ │ │ - termsOfUse.className = "olLayerGoogleCopyright"; │ │ │ │ │ - poweredBy = div.lastChild; │ │ │ │ │ - container.appendChild(poweredBy); │ │ │ │ │ - poweredBy.style.zIndex = "1100"; │ │ │ │ │ - poweredBy.style.right = ""; │ │ │ │ │ - poweredBy.style.bottom = ""; │ │ │ │ │ - poweredBy.className = "olLayerGooglePoweredBy gmnoprint" │ │ │ │ │ - } catch (e) { │ │ │ │ │ - throw e │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Layer.Google.cache[this.map.id] = { │ │ │ │ │ - mapObject: mapObject, │ │ │ │ │ - termsOfUse: termsOfUse, │ │ │ │ │ - poweredBy: poweredBy, │ │ │ │ │ - count: 1 │ │ │ │ │ + passesTolerance: function(evt) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + if (this.pixelTolerance != null && this.down && this.down.xy) { │ │ │ │ │ + passes = this.pixelTolerance >= this.down.xy.distanceTo(evt.xy); │ │ │ │ │ + if (passes && this.touch && this.down.touches.length === this.last.touches.length) { │ │ │ │ │ + for (var i = 0, ii = this.down.touches.length; i < ii; ++i) { │ │ │ │ │ + if (this.getTouchDistance(this.down.touches[i], this.last.touches[i]) > this.pixelTolerance) { │ │ │ │ │ + passes = false; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.mapObject = mapObject; │ │ │ │ │ - this.termsOfUse = termsOfUse; │ │ │ │ │ - this.poweredBy = poweredBy; │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.mapObject.getMapTypes(), this.type) === -1) { │ │ │ │ │ - this.mapObject.addMapType(this.type) │ │ │ │ │ + return passes │ │ │ │ │ + }, │ │ │ │ │ + getTouchDistance: function(from, to) { │ │ │ │ │ + return Math.sqrt(Math.pow(from.clientX - to.clientX, 2) + Math.pow(from.clientY - to.clientY, 2)) │ │ │ │ │ + }, │ │ │ │ │ + passesDblclickTolerance: function(evt) { │ │ │ │ │ + var passes = true; │ │ │ │ │ + if (this.down && this.first) { │ │ │ │ │ + passes = this.down.xy.distanceTo(this.first.xy) <= this.dblclickTolerance │ │ │ │ │ } │ │ │ │ │ - if (typeof mapObject.getDragObject == "function") { │ │ │ │ │ - this.dragObject = mapObject.getDragObject() │ │ │ │ │ - } else { │ │ │ │ │ - this.dragPanMapObject = null │ │ │ │ │ + return passes │ │ │ │ │ + }, │ │ │ │ │ + clearTimer: function() { │ │ │ │ │ + if (this.timerId != null) { │ │ │ │ │ + window.clearTimeout(this.timerId); │ │ │ │ │ + this.timerId = null │ │ │ │ │ } │ │ │ │ │ - if (this.isBaseLayer === false) { │ │ │ │ │ - this.setGMapVisibility(this.div.style.display !== "none") │ │ │ │ │ + if (this.rightclickTimerId != null) { │ │ │ │ │ + window.clearTimeout(this.rightclickTimerId); │ │ │ │ │ + this.rightclickTimerId = null │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - if (this.visibility && this.mapObject.isLoaded()) { │ │ │ │ │ - this.mapObject.checkResize() │ │ │ │ │ - } else { │ │ │ │ │ - if (!this._resized) { │ │ │ │ │ - var layer = this; │ │ │ │ │ - var handle = GEvent.addListener(this.mapObject, "load", function() { │ │ │ │ │ - GEvent.removeListener(handle); │ │ │ │ │ - delete layer._resized; │ │ │ │ │ - layer.mapObject.checkResize(); │ │ │ │ │ - layer.moveTo(layer.map.getCenter(), layer.map.getZoom()) │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this._resized = true │ │ │ │ │ + delayedCall: function(evt) { │ │ │ │ │ + this.timerId = null; │ │ │ │ │ + if (evt) { │ │ │ │ │ + this.callback("click", [evt]) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - setGMapVisibility: function(visible) { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - var container = this.mapObject.getContainer(); │ │ │ │ │ - if (visible === true) { │ │ │ │ │ - this.mapObject.setMapType(this.type); │ │ │ │ │ - container.style.display = ""; │ │ │ │ │ - this.termsOfUse.style.left = ""; │ │ │ │ │ - this.termsOfUse.style.display = ""; │ │ │ │ │ - this.poweredBy.style.display = ""; │ │ │ │ │ - cache.displayed = this.id │ │ │ │ │ - } else { │ │ │ │ │ - if (cache.displayed === this.id) { │ │ │ │ │ - delete cache.displayed │ │ │ │ │ - } │ │ │ │ │ - if (!cache.displayed) { │ │ │ │ │ - container.style.display = "none"; │ │ │ │ │ - this.termsOfUse.style.display = "none"; │ │ │ │ │ - this.termsOfUse.style.left = "-9999px"; │ │ │ │ │ - this.poweredBy.style.display = "none" │ │ │ │ │ + getEventInfo: function(evt) { │ │ │ │ │ + var touches; │ │ │ │ │ + if (evt.touches) { │ │ │ │ │ + var len = evt.touches.length; │ │ │ │ │ + touches = new Array(len); │ │ │ │ │ + var touch; │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + touch = evt.touches[i]; │ │ │ │ │ + touches[i] = { │ │ │ │ │ + clientX: touch.olClientX, │ │ │ │ │ + clientY: touch.olClientY │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return { │ │ │ │ │ + xy: evt.xy, │ │ │ │ │ + touches: touches │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - getMapContainer: function() { │ │ │ │ │ - return this.mapObject.getContainer() │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.clearTimer(); │ │ │ │ │ + this.down = null; │ │ │ │ │ + this.first = null; │ │ │ │ │ + this.last = null; │ │ │ │ │ + deactivated = true │ │ │ │ │ + } │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ - var moBounds = null; │ │ │ │ │ - if (olBounds != null) { │ │ │ │ │ - var sw = this.sphericalMercator ? this.inverseMercator(olBounds.bottom, olBounds.left) : new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ - var ne = this.sphericalMercator ? this.inverseMercator(olBounds.top, olBounds.right) : new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ - moBounds = new GLatLngBounds(new GLatLng(sw.lat, sw.lon), new GLatLng(ne.lat, ne.lon)) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Click" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Handler.Feature = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + EVENTMAP: { │ │ │ │ │ + click: { │ │ │ │ │ + in: "click", │ │ │ │ │ + out: "clickout" │ │ │ │ │ + }, │ │ │ │ │ + mousemove: { │ │ │ │ │ + in: "over", │ │ │ │ │ + out: "out" │ │ │ │ │ + }, │ │ │ │ │ + dblclick: { │ │ │ │ │ + in: "dblclick", │ │ │ │ │ + out: null │ │ │ │ │ + }, │ │ │ │ │ + mousedown: { │ │ │ │ │ + in: null, │ │ │ │ │ + out: null │ │ │ │ │ + }, │ │ │ │ │ + mouseup: { │ │ │ │ │ + in: null, │ │ │ │ │ + out: null │ │ │ │ │ + }, │ │ │ │ │ + touchstart: { │ │ │ │ │ + in: "click", │ │ │ │ │ + out: "clickout" │ │ │ │ │ } │ │ │ │ │ - return moBounds │ │ │ │ │ }, │ │ │ │ │ - setMapObjectCenter: function(center, zoom) { │ │ │ │ │ - this.mapObject.setCenter(center, zoom) │ │ │ │ │ + feature: null, │ │ │ │ │ + lastFeature: null, │ │ │ │ │ + down: null, │ │ │ │ │ + up: null, │ │ │ │ │ + clickTolerance: 4, │ │ │ │ │ + geometryTypes: null, │ │ │ │ │ + stopClick: true, │ │ │ │ │ + stopDown: true, │ │ │ │ │ + stopUp: false, │ │ │ │ │ + initialize: function(control, layer, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ + this.layer = layer │ │ │ │ │ }, │ │ │ │ │ - dragPanMapObject: function(dX, dY) { │ │ │ │ │ - this.dragObject.moveBy(new GSize(-dX, dY)) │ │ │ │ │ + touchstart: function(evt) { │ │ │ │ │ + this.startTouch(); │ │ │ │ │ + return OpenLayers.Event.isMultiTouch(evt) ? true : this.mousedown(evt) │ │ │ │ │ }, │ │ │ │ │ - getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - return this.mapObject.fromContainerPixelToLatLng(moPixel) │ │ │ │ │ + touchmove: function(evt) { │ │ │ │ │ + OpenLayers.Event.preventDefault(evt) │ │ │ │ │ }, │ │ │ │ │ - getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - return this.mapObject.fromLatLngToContainerPixel(moLonLat) │ │ │ │ │ + mousedown: function(evt) { │ │ │ │ │ + if (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ + this.down = evt.xy │ │ │ │ │ + } │ │ │ │ │ + return this.handle(evt) ? !this.stopDown : true │ │ │ │ │ }, │ │ │ │ │ - getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - return this.mapObject.getBoundsZoomLevel(moBounds) │ │ │ │ │ + mouseup: function(evt) { │ │ │ │ │ + this.up = evt.xy; │ │ │ │ │ + return this.handle(evt) ? !this.stopUp : true │ │ │ │ │ }, │ │ │ │ │ - getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ - var gLatLng; │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ - gLatLng = new GLatLng(lonlat.lat, lonlat.lon) │ │ │ │ │ - } else { │ │ │ │ │ - gLatLng = new GLatLng(lat, lon) │ │ │ │ │ + click: function(evt) { │ │ │ │ │ + return this.handle(evt) ? !this.stopClick : true │ │ │ │ │ + }, │ │ │ │ │ + mousemove: function(evt) { │ │ │ │ │ + if (!this.callbacks["over"] && !this.callbacks["out"]) { │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ - return gLatLng │ │ │ │ │ + this.handle(evt); │ │ │ │ │ + return true │ │ │ │ │ }, │ │ │ │ │ - getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ - return new GPoint(x, y) │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Layer.KaMap = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - i: "jpeg", │ │ │ │ │ - map: "" │ │ │ │ │ + dblclick: function(evt) { │ │ │ │ │ + return !this.handle(evt) │ │ │ │ │ }, │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.params = OpenLayers.Util.applyDefaults(this.params, this.DEFAULT_PARAMS) │ │ │ │ │ + geometryTypeMatches: function(feature) { │ │ │ │ │ + return this.geometryTypes == null || OpenLayers.Util.indexOf(this.geometryTypes, feature.geometry.CLASS_NAME) > -1 │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var mapRes = this.map.getResolution(); │ │ │ │ │ - var scale = Math.round(this.map.getScale() * 1e4) / 1e4; │ │ │ │ │ - var pX = Math.round(bounds.left / mapRes); │ │ │ │ │ - var pY = -Math.round(bounds.top / mapRes); │ │ │ │ │ - return this.getFullRequestString({ │ │ │ │ │ - t: pY, │ │ │ │ │ - l: pX, │ │ │ │ │ - s: scale │ │ │ │ │ - }) │ │ │ │ │ + handle: function(evt) { │ │ │ │ │ + if (this.feature && !this.feature.layer) { │ │ │ │ │ + this.feature = null │ │ │ │ │ + } │ │ │ │ │ + var type = evt.type; │ │ │ │ │ + var handled = false; │ │ │ │ │ + var previouslyIn = !!this.feature; │ │ │ │ │ + var click = type == "click" || type == "dblclick" || type == "touchstart"; │ │ │ │ │ + this.feature = this.layer.getFeatureFromEvent(evt); │ │ │ │ │ + if (this.feature && !this.feature.layer) { │ │ │ │ │ + this.feature = null │ │ │ │ │ + } │ │ │ │ │ + if (this.lastFeature && !this.lastFeature.layer) { │ │ │ │ │ + this.lastFeature = null │ │ │ │ │ + } │ │ │ │ │ + if (this.feature) { │ │ │ │ │ + if (type === "touchstart") { │ │ │ │ │ + OpenLayers.Event.preventDefault(evt) │ │ │ │ │ + } │ │ │ │ │ + var inNew = this.feature != this.lastFeature; │ │ │ │ │ + if (this.geometryTypeMatches(this.feature)) { │ │ │ │ │ + if (previouslyIn && inNew) { │ │ │ │ │ + if (this.lastFeature) { │ │ │ │ │ + this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ + } │ │ │ │ │ + this.triggerCallback(type, "in", [this.feature]) │ │ │ │ │ + } else if (!previouslyIn || click) { │ │ │ │ │ + this.triggerCallback(type, "in", [this.feature]) │ │ │ │ │ + } │ │ │ │ │ + this.lastFeature = this.feature; │ │ │ │ │ + handled = true │ │ │ │ │ + } else { │ │ │ │ │ + if (this.lastFeature && (previouslyIn && inNew || click)) { │ │ │ │ │ + this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ + } │ │ │ │ │ + this.feature = null │ │ │ │ │ + } │ │ │ │ │ + } else if (this.lastFeature && (previouslyIn || click)) { │ │ │ │ │ + this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ + } │ │ │ │ │ + return handled │ │ │ │ │ }, │ │ │ │ │ - calculateGridLayout: function(bounds, origin, resolution) { │ │ │ │ │ - var tilelon = resolution * this.tileSize.w; │ │ │ │ │ - var tilelat = resolution * this.tileSize.h; │ │ │ │ │ - var offsetlon = bounds.left; │ │ │ │ │ - var tilecol = Math.floor(offsetlon / tilelon) - this.buffer; │ │ │ │ │ - var offsetlat = bounds.top; │ │ │ │ │ - var tilerow = Math.floor(offsetlat / tilelat) + this.buffer; │ │ │ │ │ - return { │ │ │ │ │ - tilelon: tilelon, │ │ │ │ │ - tilelat: tilelat, │ │ │ │ │ - startcol: tilecol, │ │ │ │ │ - startrow: tilerow │ │ │ │ │ + triggerCallback: function(type, mode, args) { │ │ │ │ │ + var key = this.EVENTMAP[type][mode]; │ │ │ │ │ + if (key) { │ │ │ │ │ + if (type == "click" && this.up && this.down) { │ │ │ │ │ + var dpx = Math.sqrt(Math.pow(this.up.x - this.down.x, 2) + Math.pow(this.up.y - this.down.y, 2)); │ │ │ │ │ + if (dpx <= this.clickTolerance) { │ │ │ │ │ + this.callback(key, args) │ │ │ │ │ + } │ │ │ │ │ + this.up = this.down = null │ │ │ │ │ + } else { │ │ │ │ │ + this.callback(key, args) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - getTileBoundsForGridIndex: function(row, col) { │ │ │ │ │ - var origin = this.getTileOrigin(); │ │ │ │ │ - var tileLayout = this.gridLayout; │ │ │ │ │ - var tilelon = tileLayout.tilelon; │ │ │ │ │ - var tilelat = tileLayout.tilelat; │ │ │ │ │ - var minX = (tileLayout.startcol + col) * tilelon; │ │ │ │ │ - var minY = (tileLayout.startrow - row) * tilelat; │ │ │ │ │ - return new OpenLayers.Bounds(minX, minY, minX + tilelon, minY + tilelat) │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerToTop(); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + removelayer: this.handleMapEvents, │ │ │ │ │ + changelayer: this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + activated = true │ │ │ │ │ + } │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.KaMap(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerBack(); │ │ │ │ │ + this.feature = null; │ │ │ │ │ + this.lastFeature = null; │ │ │ │ │ + this.down = null; │ │ │ │ │ + this.up = null; │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + removelayer: this.handleMapEvents, │ │ │ │ │ + changelayer: this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + deactivated = true │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - if (this.tileSize != null) { │ │ │ │ │ - obj.tileSize = this.tileSize.clone() │ │ │ │ │ + return deactivated │ │ │ │ │ + }, │ │ │ │ │ + handleMapEvents: function(evt) { │ │ │ │ │ + if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ + this.moveLayerToTop() │ │ │ │ │ } │ │ │ │ │ - obj.grid = []; │ │ │ │ │ - return obj │ │ │ │ │ }, │ │ │ │ │ - getTileBounds: function(viewPortPx) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var tileMapWidth = resolution * this.tileSize.w; │ │ │ │ │ - var tileMapHeight = resolution * this.tileSize.h; │ │ │ │ │ - var mapPoint = this.getLonLatFromViewPortPx(viewPortPx); │ │ │ │ │ - var tileLeft = tileMapWidth * Math.floor(mapPoint.lon / tileMapWidth); │ │ │ │ │ - var tileBottom = tileMapHeight * Math.floor(mapPoint.lat / tileMapHeight); │ │ │ │ │ - return new OpenLayers.Bounds(tileLeft, tileBottom, tileLeft + tileMapWidth, tileBottom + tileMapHeight) │ │ │ │ │ + moveLayerToTop: function() { │ │ │ │ │ + var index = Math.max(this.map.Z_INDEX_BASE["Feature"] - 1, this.layer.getZIndex()) + 1; │ │ │ │ │ + this.layer.setZIndex(index) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.KaMap" │ │ │ │ │ + moveLayerBack: function() { │ │ │ │ │ + var index = this.layer.getZIndex() - 1; │ │ │ │ │ + if (index >= this.map.Z_INDEX_BASE["Feature"]) { │ │ │ │ │ + this.layer.setZIndex(index) │ │ │ │ │ + } else { │ │ │ │ │ + this.map.setLayerZIndex(this.layer, this.map.getLayerIndex(this.layer)) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Feature" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.Text = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ - defaultStyle: null, │ │ │ │ │ - extractStyles: true, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - if (options.extractStyles !== false) { │ │ │ │ │ - options.defaultStyle = { │ │ │ │ │ - externalGraphic: OpenLayers.Util.getImageLocation("marker.png"), │ │ │ │ │ - graphicWidth: 21, │ │ │ │ │ - graphicHeight: 25, │ │ │ │ │ - graphicXOffset: -10.5, │ │ │ │ │ - graphicYOffset: -12.5 │ │ │ │ │ +OpenLayers.Handler.Keyboard = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ + KEY_EVENTS: ["keydown", "keyup"], │ │ │ │ │ + eventListener: null, │ │ │ │ │ + observeElement: null, │ │ │ │ │ + initialize: function(control, callbacks, options) { │ │ │ │ │ + OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.eventListener = OpenLayers.Function.bindAsEventListener(this.handleKeyEvent, this) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.eventListener = null; │ │ │ │ │ + OpenLayers.Handler.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.observeElement = this.observeElement || document; │ │ │ │ │ + for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ + OpenLayers.Event.observe(this.observeElement, this.KEY_EVENTS[i], this.eventListener) │ │ │ │ │ } │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Format.prototype.initialize.apply(this, [options]) │ │ │ │ │ }, │ │ │ │ │ - read: function(text) { │ │ │ │ │ - var lines = text.split("\n"); │ │ │ │ │ - var columns; │ │ │ │ │ - var features = []; │ │ │ │ │ - for (var lcv = 0; lcv < lines.length - 1; lcv++) { │ │ │ │ │ - var currLine = lines[lcv].replace(/^\s*/, "").replace(/\s*$/, ""); │ │ │ │ │ - if (currLine.charAt(0) != "#") { │ │ │ │ │ - if (!columns) { │ │ │ │ │ - columns = currLine.split("\t") │ │ │ │ │ - } else { │ │ │ │ │ - var vals = currLine.split("\t"); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point(0, 0); │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - var style = this.defaultStyle ? OpenLayers.Util.applyDefaults({}, this.defaultStyle) : null; │ │ │ │ │ - var icon, iconSize, iconOffset, overflow; │ │ │ │ │ - var set = false; │ │ │ │ │ - for (var valIndex = 0; valIndex < vals.length; valIndex++) { │ │ │ │ │ - if (vals[valIndex]) { │ │ │ │ │ - if (columns[valIndex] == "point") { │ │ │ │ │ - var coords = vals[valIndex].split(","); │ │ │ │ │ - geometry.y = parseFloat(coords[0]); │ │ │ │ │ - geometry.x = parseFloat(coords[1]); │ │ │ │ │ - set = true │ │ │ │ │ - } else if (columns[valIndex] == "lat") { │ │ │ │ │ - geometry.y = parseFloat(vals[valIndex]); │ │ │ │ │ - set = true │ │ │ │ │ - } else if (columns[valIndex] == "lon") { │ │ │ │ │ - geometry.x = parseFloat(vals[valIndex]); │ │ │ │ │ - set = true │ │ │ │ │ - } else if (columns[valIndex] == "title") attributes["title"] = vals[valIndex]; │ │ │ │ │ - else if (columns[valIndex] == "image" || columns[valIndex] == "icon" && style) { │ │ │ │ │ - style["externalGraphic"] = vals[valIndex] │ │ │ │ │ - } else if (columns[valIndex] == "iconSize" && style) { │ │ │ │ │ - var size = vals[valIndex].split(","); │ │ │ │ │ - style["graphicWidth"] = parseFloat(size[0]); │ │ │ │ │ - style["graphicHeight"] = parseFloat(size[1]) │ │ │ │ │ - } else if (columns[valIndex] == "iconOffset" && style) { │ │ │ │ │ - var offset = vals[valIndex].split(","); │ │ │ │ │ - style["graphicXOffset"] = parseFloat(offset[0]); │ │ │ │ │ - style["graphicYOffset"] = parseFloat(offset[1]) │ │ │ │ │ - } else if (columns[valIndex] == "description") { │ │ │ │ │ - attributes["description"] = vals[valIndex] │ │ │ │ │ - } else if (columns[valIndex] == "overflow") { │ │ │ │ │ - attributes["overflow"] = vals[valIndex] │ │ │ │ │ - } else { │ │ │ │ │ - attributes[columns[valIndex]] = vals[valIndex] │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (set) { │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ - } │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry, attributes, style); │ │ │ │ │ - features.push(feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = false; │ │ │ │ │ + if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ + OpenLayers.Event.stopObserving(this.observeElement, this.KEY_EVENTS[i], this.eventListener) │ │ │ │ │ } │ │ │ │ │ + deactivated = true │ │ │ │ │ } │ │ │ │ │ - return features │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.Text" │ │ │ │ │ + handleKeyEvent: function(evt) { │ │ │ │ │ + if (this.checkModifiers(evt)) { │ │ │ │ │ + this.callback(evt.type, [evt]) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Handler.Keyboard" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Layer.Text = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ - location: null, │ │ │ │ │ +OpenLayers.Renderer.Canvas = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ + hitDetection: true, │ │ │ │ │ + hitOverflow: 0, │ │ │ │ │ + canvas: null, │ │ │ │ │ features: null, │ │ │ │ │ - formatOptions: null, │ │ │ │ │ - selectedFeature: null, │ │ │ │ │ - initialize: function(name, options) { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.features = [] │ │ │ │ │ + pendingRedraw: false, │ │ │ │ │ + cachedSymbolBounds: {}, │ │ │ │ │ + initialize: function(containerID, options) { │ │ │ │ │ + OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.root = document.createElement("canvas"); │ │ │ │ │ + this.container.appendChild(this.root); │ │ │ │ │ + this.canvas = this.root.getContext("2d"); │ │ │ │ │ + this.features = {}; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitCanvas = document.createElement("canvas"); │ │ │ │ │ + this.hitContext = this.hitCanvas.getContext("2d") │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.destroy.apply(this, arguments); │ │ │ │ │ - this.clearFeatures(); │ │ │ │ │ - this.features = null │ │ │ │ │ + setExtent: function() { │ │ │ │ │ + OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - loadText: function() { │ │ │ │ │ - if (!this.loaded) { │ │ │ │ │ - if (this.location != null) { │ │ │ │ │ - var onFail = function(e) { │ │ │ │ │ - this.events.triggerEvent("loadend") │ │ │ │ │ - }; │ │ │ │ │ - this.events.triggerEvent("loadstart"); │ │ │ │ │ - OpenLayers.Request.GET({ │ │ │ │ │ - url: this.location, │ │ │ │ │ - success: this.parseData, │ │ │ │ │ - failure: onFail, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.loaded = true │ │ │ │ │ + eraseGeometry: function(geometry, featureId) { │ │ │ │ │ + this.eraseFeatures(this.features[featureId][0]) │ │ │ │ │ + }, │ │ │ │ │ + supported: function() { │ │ │ │ │ + return OpenLayers.CANVAS_SUPPORTED │ │ │ │ │ + }, │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + this.size = size.clone(); │ │ │ │ │ + var root = this.root; │ │ │ │ │ + root.style.width = size.w + "px"; │ │ │ │ │ + root.style.height = size.h + "px"; │ │ │ │ │ + root.width = size.w; │ │ │ │ │ + root.height = size.h; │ │ │ │ │ + this.resolution = null; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + var hitCanvas = this.hitCanvas; │ │ │ │ │ + hitCanvas.style.width = size.w + "px"; │ │ │ │ │ + hitCanvas.style.height = size.h + "px"; │ │ │ │ │ + hitCanvas.width = size.w; │ │ │ │ │ + hitCanvas.height = size.h │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + drawFeature: function(feature, style) { │ │ │ │ │ + var rendered; │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + style = this.applyDefaultSymbolizer(style || feature.style); │ │ │ │ │ + var bounds = feature.geometry.getBounds(); │ │ │ │ │ + var worldBounds; │ │ │ │ │ + if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ + worldBounds = this.map.getMaxExtent() │ │ │ │ │ + } │ │ │ │ │ + var intersects = bounds && bounds.intersectsBounds(this.extent, { │ │ │ │ │ + worldBounds: worldBounds │ │ │ │ │ + }); │ │ │ │ │ + rendered = style.display !== "none" && !!bounds && intersects; │ │ │ │ │ + if (rendered) { │ │ │ │ │ + this.features[feature.id] = [feature, style] │ │ │ │ │ + } else { │ │ │ │ │ + delete this.features[feature.id] │ │ │ │ │ } │ │ │ │ │ + this.pendingRedraw = true │ │ │ │ │ + } │ │ │ │ │ + if (this.pendingRedraw && !this.locked) { │ │ │ │ │ + this.redraw(); │ │ │ │ │ + this.pendingRedraw = false │ │ │ │ │ } │ │ │ │ │ + return rendered │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, minor) { │ │ │ │ │ - OpenLayers.Layer.Markers.prototype.moveTo.apply(this, arguments); │ │ │ │ │ - if (this.visibility && !this.loaded) { │ │ │ │ │ - this.loadText() │ │ │ │ │ + drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ + var className = geometry.CLASS_NAME; │ │ │ │ │ + if (className == "OpenLayers.Geometry.Collection" || className == "OpenLayers.Geometry.MultiPoint" || className == "OpenLayers.Geometry.MultiLineString" || className == "OpenLayers.Geometry.MultiPolygon") { │ │ │ │ │ + for (var i = 0; i < geometry.components.length; i++) { │ │ │ │ │ + this.drawGeometry(geometry.components[i], style, featureId) │ │ │ │ │ + } │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + this.drawPoint(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + this.drawLineString(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + this.drawPolygon(geometry, style, featureId); │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - parseData: function(ajaxRequest) { │ │ │ │ │ - var text = ajaxRequest.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() │ │ │ │ │ + drawExternalGraphic: function(geometry, style, featureId) { │ │ │ │ │ + var img = new Image; │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + img.title = title │ │ │ │ │ } │ │ │ │ │ - var parser = new OpenLayers.Format.Text(options); │ │ │ │ │ - var features = parser.read(text); │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - var data = {}; │ │ │ │ │ - var feature = features[i]; │ │ │ │ │ - var location; │ │ │ │ │ - var iconSize, iconOffset; │ │ │ │ │ - location = new OpenLayers.LonLat(feature.geometry.x, feature.geometry.y); │ │ │ │ │ - if (feature.style.graphicWidth && feature.style.graphicHeight) { │ │ │ │ │ - iconSize = new OpenLayers.Size(feature.style.graphicWidth, feature.style.graphicHeight) │ │ │ │ │ - } │ │ │ │ │ - if (feature.style.graphicXOffset !== undefined && feature.style.graphicYOffset !== undefined) { │ │ │ │ │ - iconOffset = new OpenLayers.Pixel(feature.style.graphicXOffset, feature.style.graphicYOffset) │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ + var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ + var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ + var onLoad = function() { │ │ │ │ │ + if (!this.features[featureId]) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - if (feature.style.externalGraphic != null) { │ │ │ │ │ - data.icon = new OpenLayers.Icon(feature.style.externalGraphic, iconSize, iconOffset) │ │ │ │ │ - } else { │ │ │ │ │ - data.icon = OpenLayers.Marker.defaultIcon(); │ │ │ │ │ - if (iconSize != null) { │ │ │ │ │ - data.icon.setSize(iconSize) │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ + if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ + var x = p0 + xOffset | 0; │ │ │ │ │ + var y = p1 + yOffset | 0; │ │ │ │ │ + var canvas = this.canvas; │ │ │ │ │ + canvas.globalAlpha = opacity; │ │ │ │ │ + var factor = OpenLayers.Renderer.Canvas.drawImageScaleFactor || (OpenLayers.Renderer.Canvas.drawImageScaleFactor = /android 2.1/.test(navigator.userAgent.toLowerCase()) ? 320 / window.screen.width : 1); │ │ │ │ │ + canvas.drawImage(img, x * factor, y * factor, width * factor, height * factor); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId); │ │ │ │ │ + this.hitContext.fillRect(x, y, width, height) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (feature.attributes.title != null && feature.attributes.description != null) { │ │ │ │ │ - data["popupContentHTML"] = "<h2>" + feature.attributes.title + "</h2>" + "<p>" + feature.attributes.description + "</p>" │ │ │ │ │ - } │ │ │ │ │ - data["overflow"] = feature.attributes.overflow || "auto"; │ │ │ │ │ - var markerFeature = new OpenLayers.Feature(this, location, data); │ │ │ │ │ - this.features.push(markerFeature); │ │ │ │ │ - var marker = markerFeature.createMarker(); │ │ │ │ │ - if (feature.attributes.title != null && feature.attributes.description != null) { │ │ │ │ │ - marker.events.register("click", markerFeature, this.markerClick) │ │ │ │ │ - } │ │ │ │ │ - this.addMarker(marker) │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("loadend") │ │ │ │ │ + }; │ │ │ │ │ + img.onload = OpenLayers.Function.bind(onLoad, this); │ │ │ │ │ + img.src = style.externalGraphic │ │ │ │ │ }, │ │ │ │ │ - markerClick: function(evt) { │ │ │ │ │ - var sameMarkerClicked = this == this.layer.selectedFeature; │ │ │ │ │ - this.layer.selectedFeature = !sameMarkerClicked ? this : null; │ │ │ │ │ - for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ - this.layer.map.removePopup(this.layer.map.popups[i]) │ │ │ │ │ + drawNamedSymbol: function(geometry, style, featureId) { │ │ │ │ │ + var x, y, cx, cy, i, symbolBounds, scaling, angle; │ │ │ │ │ + var unscaledStrokeWidth; │ │ │ │ │ + var deg2rad = Math.PI / 180; │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[style.graphicName]; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(style.graphicName + " is not a valid symbol name") │ │ │ │ │ } │ │ │ │ │ - if (!sameMarkerClicked) { │ │ │ │ │ - this.layer.map.addPopup(this.createPopup()) │ │ │ │ │ + if (!symbol.length || symbol.length < 2) return; │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ + if (isNaN(p0) || isNaN(p1)) return; │ │ │ │ │ + this.canvas.lineCap = "round"; │ │ │ │ │ + this.canvas.lineJoin = "round"; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.lineCap = "round"; │ │ │ │ │ + this.hitContext.lineJoin = "round" │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ - }, │ │ │ │ │ - clearFeatures: function() { │ │ │ │ │ - if (this.features != null) { │ │ │ │ │ - while (this.features.length > 0) { │ │ │ │ │ - var feature = this.features[0]; │ │ │ │ │ - OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ - feature.destroy() │ │ │ │ │ + if (style.graphicName in this.cachedSymbolBounds) { │ │ │ │ │ + symbolBounds = this.cachedSymbolBounds[style.graphicName] │ │ │ │ │ + } else { │ │ │ │ │ + symbolBounds = new OpenLayers.Bounds; │ │ │ │ │ + for (i = 0; i < symbol.length; i += 2) { │ │ │ │ │ + symbolBounds.extend(new OpenLayers.LonLat(symbol[i], symbol[i + 1])) │ │ │ │ │ } │ │ │ │ │ + this.cachedSymbolBounds[style.graphicName] = symbolBounds │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Text" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Bing = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ - key: null, │ │ │ │ │ - serverResolutions: [156543.03390625, 78271.516953125, 39135.7584765625, 19567.87923828125, 9783.939619140625, 4891.9698095703125, 2445.9849047851562, 1222.9924523925781, 611.4962261962891, 305.74811309814453, 152.87405654907226, 76.43702827453613, 38.218514137268066, 19.109257068634033, 9.554628534317017, 4.777314267158508, 2.388657133579254, 1.194328566789627, .5971642833948135, .29858214169740677, .14929107084870338, .07464553542435169], │ │ │ │ │ - attributionTemplate: '<span class="olBingAttribution ${type}">' + '<div><a target="_blank" href="http://www.bing.com/maps/">' + '<img src="${logo}" /></a></div>${copyrights}' + '<a style="white-space: nowrap" target="_blank" ' + 'href="http://www.microsoft.com/maps/product/terms.html">' + "Terms of Use</a></span>", │ │ │ │ │ - metadata: null, │ │ │ │ │ - protocolRegex: /^http:/i, │ │ │ │ │ - type: "Road", │ │ │ │ │ - culture: "en-US", │ │ │ │ │ - metadataParams: null, │ │ │ │ │ - tileOptions: null, │ │ │ │ │ - protocol: ~window.location.href.indexOf("http") ? "" : "http:", │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - sphericalMercator: true │ │ │ │ │ - }, options); │ │ │ │ │ - var name = options.name || "Bing " + (options.type || this.type); │ │ │ │ │ - var newArgs = [name, null, options]; │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.initialize.apply(this, newArgs); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - crossOriginKeyword: "anonymous" │ │ │ │ │ - }, this.options.tileOptions); │ │ │ │ │ - this.loadMetadata() │ │ │ │ │ - }, │ │ │ │ │ - loadMetadata: function() { │ │ │ │ │ - this._callbackId = "_callback_" + this.id.replace(/\./g, "_"); │ │ │ │ │ - window[this._callbackId] = OpenLayers.Function.bind(OpenLayers.Layer.Bing.processMetadata, this); │ │ │ │ │ - var params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - key: this.key, │ │ │ │ │ - jsonp: this._callbackId, │ │ │ │ │ - include: "ImageryProviders" │ │ │ │ │ - }, this.metadataParams); │ │ │ │ │ - var url = this.protocol + "//dev.virtualearth.net/REST/v1/Imagery/Metadata/" + this.type + "?" + OpenLayers.Util.getParameterString(params); │ │ │ │ │ - var script = document.createElement("script"); │ │ │ │ │ - script.type = "text/javascript"; │ │ │ │ │ - script.src = url; │ │ │ │ │ - script.id = this._callbackId; │ │ │ │ │ - document.getElementsByTagName("head")[0].appendChild(script) │ │ │ │ │ - }, │ │ │ │ │ - initLayer: function() { │ │ │ │ │ - var res = this.metadata.resourceSets[0].resources[0]; │ │ │ │ │ - var url = res.imageUrl.replace("{quadkey}", "${quadkey}"); │ │ │ │ │ - url = url.replace("{culture}", this.culture); │ │ │ │ │ - url = url.replace(this.protocolRegex, this.protocol); │ │ │ │ │ - this.url = []; │ │ │ │ │ - for (var i = 0; i < res.imageUrlSubdomains.length; ++i) { │ │ │ │ │ - this.url.push(url.replace("{subdomain}", res.imageUrlSubdomains[i])) │ │ │ │ │ + this.canvas.save(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.save() │ │ │ │ │ } │ │ │ │ │ - this.addOptions({ │ │ │ │ │ - maxResolution: Math.min(this.serverResolutions[res.zoomMin], this.maxResolution || Number.POSITIVE_INFINITY), │ │ │ │ │ - numZoomLevels: Math.min(res.zoomMax + 1 - res.zoomMin, this.numZoomLevels) │ │ │ │ │ - }, true); │ │ │ │ │ - if (!this.isBaseLayer) { │ │ │ │ │ - this.redraw() │ │ │ │ │ + this.canvas.translate(p0, p1); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.translate(p0, p1) │ │ │ │ │ } │ │ │ │ │ - this.updateAttribution() │ │ │ │ │ - }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - if (!this.url) { │ │ │ │ │ - return │ │ │ │ │ + angle = deg2rad * style.rotation; │ │ │ │ │ + if (!isNaN(angle)) { │ │ │ │ │ + this.canvas.rotate(angle); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.rotate(angle) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var xyz = this.getXYZ(bounds), │ │ │ │ │ - x = xyz.x, │ │ │ │ │ - y = xyz.y, │ │ │ │ │ - z = xyz.z; │ │ │ │ │ - var quadDigits = []; │ │ │ │ │ - for (var i = z; i > 0; --i) { │ │ │ │ │ - var digit = "0"; │ │ │ │ │ - var mask = 1 << i - 1; │ │ │ │ │ - if ((x & mask) != 0) { │ │ │ │ │ - digit++ │ │ │ │ │ + scaling = 2 * style.pointRadius / Math.max(symbolBounds.getWidth(), symbolBounds.getHeight()); │ │ │ │ │ + this.canvas.scale(scaling, scaling); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.scale(scaling, scaling) │ │ │ │ │ + } │ │ │ │ │ + cx = symbolBounds.getCenterLonLat().lon; │ │ │ │ │ + cy = symbolBounds.getCenterLonLat().lat; │ │ │ │ │ + this.canvas.translate(-cx, -cy); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.translate(-cx, -cy) │ │ │ │ │ + } │ │ │ │ │ + unscaledStrokeWidth = style.strokeWidth; │ │ │ │ │ + style.strokeWidth = unscaledStrokeWidth / scaling; │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.canvas.lineTo(x, y) │ │ │ │ │ } │ │ │ │ │ - if ((y & mask) != 0) { │ │ │ │ │ - digit++; │ │ │ │ │ - digit++ │ │ │ │ │ + this.canvas.closePath(); │ │ │ │ │ + this.canvas.fill(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.hitContext.lineTo(x, y) │ │ │ │ │ + } │ │ │ │ │ + this.hitContext.closePath(); │ │ │ │ │ + this.hitContext.fill() │ │ │ │ │ } │ │ │ │ │ - quadDigits.push(digit) │ │ │ │ │ - } │ │ │ │ │ - var quadKey = quadDigits.join(""); │ │ │ │ │ - var url = this.selectUrl("" + x + y + z, this.url); │ │ │ │ │ - return OpenLayers.String.format(url, { │ │ │ │ │ - quadkey: quadKey │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - updateAttribution: function() { │ │ │ │ │ - var metadata = this.metadata; │ │ │ │ │ - if (!metadata.resourceSets || !this.map || !this.map.center) { │ │ │ │ │ - return │ │ │ │ │ } │ │ │ │ │ - var res = metadata.resourceSets[0].resources[0]; │ │ │ │ │ - var extent = this.map.getExtent().transform(this.map.getProjectionObject(), new OpenLayers.Projection("EPSG:4326")); │ │ │ │ │ - var providers = res.imageryProviders || [], │ │ │ │ │ - zoom = OpenLayers.Util.indexOf(this.serverResolutions, this.getServerResolution()), │ │ │ │ │ - copyrights = "", │ │ │ │ │ - provider, i, ii, j, jj, bbox, coverage; │ │ │ │ │ - for (i = 0, ii = providers.length; i < ii; ++i) { │ │ │ │ │ - provider = providers[i]; │ │ │ │ │ - for (j = 0, jj = provider.coverageAreas.length; j < jj; ++j) { │ │ │ │ │ - coverage = provider.coverageAreas[j]; │ │ │ │ │ - bbox = OpenLayers.Bounds.fromArray(coverage.bbox, true); │ │ │ │ │ - if (extent.intersectsBounds(bbox) && zoom <= coverage.zoomMax && zoom >= coverage.zoomMin) { │ │ │ │ │ - copyrights += provider.attribution + " " │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ + this.canvas.lineTo(x, y) │ │ │ │ │ + } │ │ │ │ │ + this.canvas.closePath(); │ │ │ │ │ + this.canvas.stroke(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style, scaling); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + if (i == 0) this.hitContext.moveTo(x, y); │ │ │ │ │ + this.hitContext.lineTo(x, y) │ │ │ │ │ } │ │ │ │ │ + this.hitContext.closePath(); │ │ │ │ │ + this.hitContext.stroke() │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var logo = metadata.brandLogoUri.replace(this.protocolRegex, this.protocol); │ │ │ │ │ - this.attribution = OpenLayers.String.format(this.attributionTemplate, { │ │ │ │ │ - type: this.type.toLowerCase(), │ │ │ │ │ - logo: logo, │ │ │ │ │ - copyrights: copyrights │ │ │ │ │ - }); │ │ │ │ │ - this.map && this.map.events.triggerEvent("changelayer", { │ │ │ │ │ - layer: this, │ │ │ │ │ - property: "attribution" │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - setMap: function() { │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.register("moveend", this, this.updateAttribution) │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Bing(this.options) │ │ │ │ │ + style.strokeWidth = unscaledStrokeWidth; │ │ │ │ │ + this.canvas.restore(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.restore() │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.map && this.map.events.unregister("moveend", this, this.updateAttribution); │ │ │ │ │ - OpenLayers.Layer.XYZ.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Bing" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Bing.processMetadata = function(metadata) { │ │ │ │ │ - this.metadata = metadata; │ │ │ │ │ - this.initLayer(); │ │ │ │ │ - var script = document.getElementById(this._callbackId); │ │ │ │ │ - script.parentNode.removeChild(script); │ │ │ │ │ - window[this._callbackId] = undefined; │ │ │ │ │ - delete this._callbackId │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Tile.UTFGrid = OpenLayers.Class(OpenLayers.Tile, { │ │ │ │ │ - url: null, │ │ │ │ │ - utfgridResolution: 2, │ │ │ │ │ - json: null, │ │ │ │ │ - format: null, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.clear(); │ │ │ │ │ - OpenLayers.Tile.prototype.destroy.apply(this, arguments) │ │ │ │ │ + this.setCanvasStyle("reset") │ │ │ │ │ }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - var drawn = OpenLayers.Tile.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (drawn) { │ │ │ │ │ - if (this.isLoading) { │ │ │ │ │ - this.abortLoading(); │ │ │ │ │ - this.events.triggerEvent("reload") │ │ │ │ │ - } else { │ │ │ │ │ - this.isLoading = true; │ │ │ │ │ - this.events.triggerEvent("loadstart") │ │ │ │ │ - } │ │ │ │ │ - this.url = this.layer.getURL(this.bounds); │ │ │ │ │ - if (this.layer.useJSONP) { │ │ │ │ │ - var ols = new OpenLayers.Protocol.Script({ │ │ │ │ │ - url: this.url, │ │ │ │ │ - callback: function(response) { │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ - this.json = response.data │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - ols.read(); │ │ │ │ │ - this.request = ols │ │ │ │ │ - } else { │ │ │ │ │ - this.request = OpenLayers.Request.GET({ │ │ │ │ │ - url: this.url, │ │ │ │ │ - callback: function(response) { │ │ │ │ │ - this.isLoading = false; │ │ │ │ │ - this.events.triggerEvent("loadend"); │ │ │ │ │ - if (response.status === 200) { │ │ │ │ │ - this.parseData(response.responseText) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ + setCanvasStyle: function(type, style) { │ │ │ │ │ + if (type === "fill") { │ │ │ │ │ + this.canvas.globalAlpha = style["fillOpacity"]; │ │ │ │ │ + this.canvas.fillStyle = style["fillColor"] │ │ │ │ │ + } else if (type === "stroke") { │ │ │ │ │ + this.canvas.globalAlpha = style["strokeOpacity"]; │ │ │ │ │ + this.canvas.strokeStyle = style["strokeColor"]; │ │ │ │ │ + this.canvas.lineWidth = style["strokeWidth"] │ │ │ │ │ } else { │ │ │ │ │ - this.unload() │ │ │ │ │ + this.canvas.globalAlpha = 0; │ │ │ │ │ + this.canvas.lineWidth = 1 │ │ │ │ │ } │ │ │ │ │ - return drawn │ │ │ │ │ }, │ │ │ │ │ - abortLoading: function() { │ │ │ │ │ - if (this.request) { │ │ │ │ │ - this.request.abort(); │ │ │ │ │ - delete this.request │ │ │ │ │ + featureIdToHex: function(featureId) { │ │ │ │ │ + var id = Number(featureId.split("_").pop()) + 1; │ │ │ │ │ + if (id >= 16777216) { │ │ │ │ │ + this.hitOverflow = id - 16777215; │ │ │ │ │ + id = id % 16777216 + 1 │ │ │ │ │ } │ │ │ │ │ - this.isLoading = false │ │ │ │ │ + var hex = "000000" + id.toString(16); │ │ │ │ │ + var len = hex.length; │ │ │ │ │ + hex = "#" + hex.substring(len - 6, len); │ │ │ │ │ + return hex │ │ │ │ │ }, │ │ │ │ │ - getFeatureInfo: function(i, j) { │ │ │ │ │ - var info = null; │ │ │ │ │ - if (this.json) { │ │ │ │ │ - var id = this.getFeatureId(i, j); │ │ │ │ │ - if (id !== null) { │ │ │ │ │ - info = { │ │ │ │ │ - id: id, │ │ │ │ │ - data: this.json.data[id] │ │ │ │ │ + setHitContextStyle: function(type, featureId, symbolizer, strokeScaling) { │ │ │ │ │ + var hex = this.featureIdToHex(featureId); │ │ │ │ │ + if (type == "fill") { │ │ │ │ │ + this.hitContext.globalAlpha = 1; │ │ │ │ │ + this.hitContext.fillStyle = hex │ │ │ │ │ + } else if (type == "stroke") { │ │ │ │ │ + this.hitContext.globalAlpha = 1; │ │ │ │ │ + this.hitContext.strokeStyle = hex; │ │ │ │ │ + if (typeof strokeScaling === "undefined") { │ │ │ │ │ + this.hitContext.lineWidth = symbolizer.strokeWidth + 2 │ │ │ │ │ + } else { │ │ │ │ │ + if (!isNaN(strokeScaling)) { │ │ │ │ │ + this.hitContext.lineWidth = symbolizer.strokeWidth + 2 / strokeScaling │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + this.hitContext.globalAlpha = 0; │ │ │ │ │ + this.hitContext.lineWidth = 1 │ │ │ │ │ } │ │ │ │ │ - return info │ │ │ │ │ }, │ │ │ │ │ - getFeatureId: function(i, j) { │ │ │ │ │ - var id = null; │ │ │ │ │ - if (this.json) { │ │ │ │ │ - var resolution = this.utfgridResolution; │ │ │ │ │ - var row = Math.floor(j / resolution); │ │ │ │ │ - var col = Math.floor(i / resolution); │ │ │ │ │ - var charCode = this.json.grid[row].charCodeAt(col); │ │ │ │ │ - var index = this.indexFromCharCode(charCode); │ │ │ │ │ - var keys = this.json.keys; │ │ │ │ │ - if (!isNaN(index) && index in keys) { │ │ │ │ │ - id = keys[index] │ │ │ │ │ + drawPoint: function(geometry, style, featureId) { │ │ │ │ │ + if (style.graphic !== false) { │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + this.drawExternalGraphic(geometry, style, featureId) │ │ │ │ │ + } else if (style.graphicName && style.graphicName != "circle") { │ │ │ │ │ + this.drawNamedSymbol(geometry, style, featureId) │ │ │ │ │ + } else { │ │ │ │ │ + var pt = this.getLocalXY(geometry); │ │ │ │ │ + var p0 = pt[0]; │ │ │ │ │ + var p1 = pt[1]; │ │ │ │ │ + if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ + var twoPi = Math.PI * 2; │ │ │ │ │ + var radius = style.pointRadius; │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.canvas.fill(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.hitContext.fill() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.canvas.beginPath(); │ │ │ │ │ + this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.canvas.stroke(); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ + this.hitContext.beginPath(); │ │ │ │ │ + this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ + this.hitContext.stroke() │ │ │ │ │ + } │ │ │ │ │ + this.setCanvasStyle("reset") │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return id │ │ │ │ │ }, │ │ │ │ │ - indexFromCharCode: function(charCode) { │ │ │ │ │ - if (charCode >= 93) { │ │ │ │ │ - charCode-- │ │ │ │ │ - } │ │ │ │ │ - if (charCode >= 35) { │ │ │ │ │ - charCode-- │ │ │ │ │ - } │ │ │ │ │ - return charCode - 32 │ │ │ │ │ + drawLineString: function(geometry, style, featureId) { │ │ │ │ │ + style = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + fill: false │ │ │ │ │ + }, style); │ │ │ │ │ + this.drawLinearRing(geometry, style, featureId) │ │ │ │ │ }, │ │ │ │ │ - parseData: function(str) { │ │ │ │ │ - if (!this.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.JSON │ │ │ │ │ + drawLinearRing: function(geometry, style, featureId) { │ │ │ │ │ + if (style.fill !== false) { │ │ │ │ │ + this.setCanvasStyle("fill", style); │ │ │ │ │ + this.renderPath(this.canvas, geometry, style, featureId, "fill"); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ + this.renderPath(this.hitContext, geometry, style, featureId, "fill") │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.json = this.format.read(str) │ │ │ │ │ - }, │ │ │ │ │ - clear: function() { │ │ │ │ │ - this.json = null │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Tile.UTFGrid" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.UTFGrid = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ - isBaseLayer: false, │ │ │ │ │ - projection: new OpenLayers.Projection("EPSG:900913"), │ │ │ │ │ - useJSONP: false, │ │ │ │ │ - tileClass: OpenLayers.Tile.UTFGrid, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [options.name, options.url, {}, options]); │ │ │ │ │ - this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ - utfgridResolution: this.utfgridResolution │ │ │ │ │ - }, this.tileOptions) │ │ │ │ │ - }, │ │ │ │ │ - createBackBuffer: function() {}, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.UTFGrid(this.getOptions()) │ │ │ │ │ + if (style.stroke !== false) { │ │ │ │ │ + this.setCanvasStyle("stroke", style); │ │ │ │ │ + this.renderPath(this.canvas, geometry, style, featureId, "stroke"); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ + this.renderPath(this.hitContext, geometry, style, featureId, "stroke") │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ + this.setCanvasStyle("reset") │ │ │ │ │ }, │ │ │ │ │ - getFeatureInfo: function(location) { │ │ │ │ │ - var info = null; │ │ │ │ │ - var tileInfo = this.getTileData(location); │ │ │ │ │ - if (tileInfo && tileInfo.tile) { │ │ │ │ │ - info = tileInfo.tile.getFeatureInfo(tileInfo.i, tileInfo.j) │ │ │ │ │ + renderPath: function(context, geometry, style, featureId, type) { │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + context.beginPath(); │ │ │ │ │ + var start = this.getLocalXY(components[0]); │ │ │ │ │ + var x = start[0]; │ │ │ │ │ + var y = start[1]; │ │ │ │ │ + if (!isNaN(x) && !isNaN(y)) { │ │ │ │ │ + context.moveTo(start[0], start[1]); │ │ │ │ │ + for (var i = 1; i < len; ++i) { │ │ │ │ │ + var pt = this.getLocalXY(components[i]); │ │ │ │ │ + context.lineTo(pt[0], pt[1]) │ │ │ │ │ + } │ │ │ │ │ + if (type === "fill") { │ │ │ │ │ + context.fill() │ │ │ │ │ + } else { │ │ │ │ │ + context.stroke() │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return info │ │ │ │ │ }, │ │ │ │ │ - getFeatureId: function(location) { │ │ │ │ │ - var id = null; │ │ │ │ │ - var info = this.getTileData(location); │ │ │ │ │ - if (info.tile) { │ │ │ │ │ - id = info.tile.getFeatureId(info.i, info.j) │ │ │ │ │ + drawPolygon: function(geometry, style, featureId) { │ │ │ │ │ + var components = geometry.components; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + this.drawLinearRing(components[0], style, featureId); │ │ │ │ │ + for (var i = 1; i < len; ++i) { │ │ │ │ │ + this.canvas.globalCompositeOperation = "destination-out"; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.globalCompositeOperation = "destination-out" │ │ │ │ │ + } │ │ │ │ │ + this.drawLinearRing(components[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + stroke: false, │ │ │ │ │ + fillOpacity: 1 │ │ │ │ │ + }, style), featureId); │ │ │ │ │ + this.canvas.globalCompositeOperation = "source-over"; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.globalCompositeOperation = "source-over" │ │ │ │ │ + } │ │ │ │ │ + this.drawLinearRing(components[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + fill: false │ │ │ │ │ + }, style), featureId) │ │ │ │ │ } │ │ │ │ │ - return id │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.UTFGrid" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.KaMapCache = OpenLayers.Class(OpenLayers.Layer.KaMap, { │ │ │ │ │ - IMAGE_EXTENSIONS: { │ │ │ │ │ - jpeg: "jpg", │ │ │ │ │ - gif: "gif", │ │ │ │ │ - png: "png", │ │ │ │ │ - png8: "png", │ │ │ │ │ - png24: "png", │ │ │ │ │ - dithered: "png" │ │ │ │ │ - }, │ │ │ │ │ - DEFAULT_FORMAT: "jpeg", │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - OpenLayers.Layer.KaMap.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.extension = this.IMAGE_EXTENSIONS[this.params.i.toLowerCase() || this.DEFAULT_FORMAT] │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var mapRes = this.map.getResolution(); │ │ │ │ │ - var scale = Math.round(this.map.getScale() * 1e4) / 1e4; │ │ │ │ │ - var pX = Math.round(bounds.left / mapRes); │ │ │ │ │ - var pY = -Math.round(bounds.top / mapRes); │ │ │ │ │ - var metaX = Math.floor(pX / this.tileSize.w / this.params.metaTileSize.w) * this.tileSize.w * this.params.metaTileSize.w; │ │ │ │ │ - var metaY = Math.floor(pY / this.tileSize.h / this.params.metaTileSize.h) * this.tileSize.h * this.params.metaTileSize.h; │ │ │ │ │ - var components = ["/", this.params.map, "/", scale, "/", this.params.g.replace(/\s/g, "_"), "/def/t", metaY, "/l", metaX, "/t", pY, "l", pX, ".", this.extension]; │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(components.join(""), url) │ │ │ │ │ + drawText: function(location, style) { │ │ │ │ │ + var pt = this.getLocalXY(location); │ │ │ │ │ + this.setCanvasStyle("reset"); │ │ │ │ │ + this.canvas.fillStyle = style.fontColor; │ │ │ │ │ + this.canvas.globalAlpha = style.fontOpacity || 1; │ │ │ │ │ + var fontStyle = [style.fontStyle ? style.fontStyle : "normal", "normal", style.fontWeight ? style.fontWeight : "normal", style.fontSize ? style.fontSize : "1em", style.fontFamily ? style.fontFamily : "sans-serif"].join(" "); │ │ │ │ │ + var labelRows = style.label.split("\n"); │ │ │ │ │ + var numRows = labelRows.length; │ │ │ │ │ + if (this.canvas.fillText) { │ │ │ │ │ + this.canvas.font = fontStyle; │ │ │ │ │ + this.canvas.textAlign = OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[0]] || "center"; │ │ │ │ │ + this.canvas.textBaseline = OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[1]] || "middle"; │ │ │ │ │ + var vfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5 │ │ │ │ │ + } │ │ │ │ │ + var lineHeight = this.canvas.measureText("Mg").height || this.canvas.measureText("xx").width; │ │ │ │ │ + pt[1] += lineHeight * vfactor * (numRows - 1); │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + if (style.labelOutlineWidth) { │ │ │ │ │ + this.canvas.save(); │ │ │ │ │ + this.canvas.globalAlpha = style.labelOutlineOpacity || style.fontOpacity || 1; │ │ │ │ │ + this.canvas.strokeStyle = style.labelOutlineColor; │ │ │ │ │ + this.canvas.lineWidth = style.labelOutlineWidth; │ │ │ │ │ + this.canvas.strokeText(labelRows[i], pt[0], pt[1] + lineHeight * i + 1); │ │ │ │ │ + this.canvas.restore() │ │ │ │ │ + } │ │ │ │ │ + this.canvas.fillText(labelRows[i], pt[0], pt[1] + lineHeight * i) │ │ │ │ │ + } │ │ │ │ │ + } else if (this.canvas.mozDrawText) { │ │ │ │ │ + this.canvas.mozTextStyle = fontStyle; │ │ │ │ │ + var hfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[0]]; │ │ │ │ │ + if (hfactor == null) { │ │ │ │ │ + hfactor = -.5 │ │ │ │ │ + } │ │ │ │ │ + var vfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5 │ │ │ │ │ + } │ │ │ │ │ + var lineHeight = this.canvas.mozMeasureText("xx"); │ │ │ │ │ + pt[1] += lineHeight * (1 + vfactor * numRows); │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + var x = pt[0] + hfactor * this.canvas.mozMeasureText(labelRows[i]); │ │ │ │ │ + var y = pt[1] + i * lineHeight; │ │ │ │ │ + this.canvas.translate(x, y); │ │ │ │ │ + this.canvas.mozDrawText(labelRows[i]); │ │ │ │ │ + this.canvas.translate(-x, -y) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return url + components.join("") │ │ │ │ │ + this.setCanvasStyle("reset") │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.KaMapCache" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.WMTS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - requestEncoding: "KVP", │ │ │ │ │ - url: null, │ │ │ │ │ - layer: null, │ │ │ │ │ - matrixSet: null, │ │ │ │ │ - style: null, │ │ │ │ │ - format: "image/jpeg", │ │ │ │ │ - tileOrigin: null, │ │ │ │ │ - tileFullExtent: null, │ │ │ │ │ - formatSuffix: null, │ │ │ │ │ - matrixIds: null, │ │ │ │ │ - dimensions: null, │ │ │ │ │ - params: null, │ │ │ │ │ - zoomOffset: 0, │ │ │ │ │ - serverResolutions: null, │ │ │ │ │ - formatSuffixMap: { │ │ │ │ │ - "image/png": "png", │ │ │ │ │ - "image/png8": "png", │ │ │ │ │ - "image/png24": "png", │ │ │ │ │ - "image/png32": "png", │ │ │ │ │ - png: "png", │ │ │ │ │ - "image/jpeg": "jpg", │ │ │ │ │ - "image/jpg": "jpg", │ │ │ │ │ - jpeg: "jpg", │ │ │ │ │ - jpg: "jpg" │ │ │ │ │ + getLocalXY: function(point) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var extent = this.extent; │ │ │ │ │ + var x = (point.x - this.featureDx) / resolution + -extent.left / resolution; │ │ │ │ │ + var y = extent.top / resolution - point.y / resolution; │ │ │ │ │ + return [x, y] │ │ │ │ │ }, │ │ │ │ │ - matrix: null, │ │ │ │ │ - initialize: function(config) { │ │ │ │ │ - var required = { │ │ │ │ │ - url: true, │ │ │ │ │ - layer: true, │ │ │ │ │ - style: true, │ │ │ │ │ - matrixSet: true │ │ │ │ │ - }; │ │ │ │ │ - for (var prop in required) { │ │ │ │ │ - if (!(prop in config)) { │ │ │ │ │ - throw new Error("Missing property '" + prop + "' in layer configuration.") │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - config.params = OpenLayers.Util.upperCaseObject(config.params); │ │ │ │ │ - var args = [config.name, config.url, config.params, config]; │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, args); │ │ │ │ │ - if (!this.formatSuffix) { │ │ │ │ │ - this.formatSuffix = this.formatSuffixMap[this.format] || this.format.split("/").pop() │ │ │ │ │ + clear: function() { │ │ │ │ │ + var height = this.root.height; │ │ │ │ │ + var width = this.root.width; │ │ │ │ │ + this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ + this.features = {}; │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.clearRect(0, 0, width, height) │ │ │ │ │ } │ │ │ │ │ - if (this.matrixIds) { │ │ │ │ │ - var len = this.matrixIds.length; │ │ │ │ │ - if (len && typeof this.matrixIds[0] === "string") { │ │ │ │ │ - var ids = this.matrixIds; │ │ │ │ │ - this.matrixIds = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - this.matrixIds[i] = { │ │ │ │ │ - identifier: ids[i] │ │ │ │ │ + }, │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var featureId, feature; │ │ │ │ │ + if (this.hitDetection && this.root.style.display !== "none") { │ │ │ │ │ + if (!this.map.dragging) { │ │ │ │ │ + var xy = evt.xy; │ │ │ │ │ + var x = xy.x | 0; │ │ │ │ │ + var y = xy.y | 0; │ │ │ │ │ + var data = this.hitContext.getImageData(x, y, 1, 1).data; │ │ │ │ │ + if (data[3] === 255) { │ │ │ │ │ + var id = data[2] + 256 * (data[1] + 256 * data[0]); │ │ │ │ │ + if (id) { │ │ │ │ │ + featureId = "OpenLayers_Feature_Vector_" + (id - 1 + this.hitOverflow); │ │ │ │ │ + try { │ │ │ │ │ + feature = this.features[featureId][0] │ │ │ │ │ + } catch (err) {} │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return feature │ │ │ │ │ }, │ │ │ │ │ - setMap: function() { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments) │ │ │ │ │ + eraseFeatures: function(features) { │ │ │ │ │ + if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ + features = [features] │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0; i < features.length; ++i) { │ │ │ │ │ + delete this.features[features[i].id] │ │ │ │ │ + } │ │ │ │ │ + this.redraw() │ │ │ │ │ }, │ │ │ │ │ - updateMatrixProperties: function() { │ │ │ │ │ - this.matrix = this.getMatrix(); │ │ │ │ │ - if (this.matrix) { │ │ │ │ │ - if (this.matrix.topLeftCorner) { │ │ │ │ │ - this.tileOrigin = this.matrix.topLeftCorner │ │ │ │ │ - } │ │ │ │ │ - if (this.matrix.tileWidth && this.matrix.tileHeight) { │ │ │ │ │ - this.tileSize = new OpenLayers.Size(this.matrix.tileWidth, this.matrix.tileHeight) │ │ │ │ │ + redraw: function() { │ │ │ │ │ + if (!this.locked) { │ │ │ │ │ + var height = this.root.height; │ │ │ │ │ + var width = this.root.width; │ │ │ │ │ + this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ + if (this.hitDetection) { │ │ │ │ │ + this.hitContext.clearRect(0, 0, width, height) │ │ │ │ │ } │ │ │ │ │ - if (!this.tileOrigin) { │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, this.maxExtent.top) │ │ │ │ │ + var labelMap = []; │ │ │ │ │ + var feature, geometry, style; │ │ │ │ │ + var worldBounds = this.map.baseLayer && this.map.baseLayer.wrapDateLine && this.map.getMaxExtent(); │ │ │ │ │ + for (var id in this.features) { │ │ │ │ │ + if (!this.features.hasOwnProperty(id)) { │ │ │ │ │ + continue │ │ │ │ │ + } │ │ │ │ │ + feature = this.features[id][0]; │ │ │ │ │ + geometry = feature.geometry; │ │ │ │ │ + this.calculateFeatureDx(geometry.getBounds(), worldBounds); │ │ │ │ │ + style = this.features[id][1]; │ │ │ │ │ + this.drawGeometry(geometry, style, feature.id); │ │ │ │ │ + if (style.label) { │ │ │ │ │ + labelMap.push([feature, style]) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (!this.tileFullExtent) { │ │ │ │ │ - this.tileFullExtent = this.maxExtent │ │ │ │ │ + var item; │ │ │ │ │ + for (var i = 0, len = labelMap.length; i < len; ++i) { │ │ │ │ │ + item = labelMap[i]; │ │ │ │ │ + this.drawText(item[0].geometry.getCentroid(), item[1]) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ - if (zoomChanged || !this.matrix) { │ │ │ │ │ - this.updateMatrixProperties() │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.Canvas" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Renderer.Canvas.LABEL_ALIGN = { │ │ │ │ │ + l: "left", │ │ │ │ │ + r: "right", │ │ │ │ │ + t: "top", │ │ │ │ │ + b: "bottom" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.Canvas.LABEL_FACTOR = { │ │ │ │ │ + l: 0, │ │ │ │ │ + r: -1, │ │ │ │ │ + t: 0, │ │ │ │ │ + b: -1 │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.Canvas.drawImageScaleFactor = null; │ │ │ │ │ +OpenLayers.Renderer.VML = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ + xmlns: "urn:schemas-microsoft-com:vml", │ │ │ │ │ + symbolCache: {}, │ │ │ │ │ + offset: null, │ │ │ │ │ + initialize: function(containerID) { │ │ │ │ │ + if (!this.supported()) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.moveTo.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.WMTS(this.options) │ │ │ │ │ + if (!document.namespaces.olv) { │ │ │ │ │ + document.namespaces.add("olv", this.xmlns); │ │ │ │ │ + var style = document.createStyleSheet(); │ │ │ │ │ + var shapes = ["shape", "rect", "oval", "fill", "stroke", "imagedata", "group", "textbox"]; │ │ │ │ │ + for (var i = 0, len = shapes.length; i < len; i++) { │ │ │ │ │ + style.addRule("olv\\:" + shapes[i], "behavior: url(#default#VML); " + "position: absolute; display: inline-block;") │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.initialize.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - getIdentifier: function() { │ │ │ │ │ - return this.getServerZoom() │ │ │ │ │ + supported: function() { │ │ │ │ │ + return !!document.namespaces │ │ │ │ │ }, │ │ │ │ │ - getMatrix: function() { │ │ │ │ │ - var matrix; │ │ │ │ │ - if (!this.matrixIds || this.matrixIds.length === 0) { │ │ │ │ │ - matrix = { │ │ │ │ │ - identifier: this.getIdentifier() │ │ │ │ │ - } │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var left = extent.left / resolution | 0; │ │ │ │ │ + var top = extent.top / resolution - this.size.h | 0; │ │ │ │ │ + if (resolutionChanged || !this.offset) { │ │ │ │ │ + this.offset = { │ │ │ │ │ + x: left, │ │ │ │ │ + y: top │ │ │ │ │ + }; │ │ │ │ │ + left = 0; │ │ │ │ │ + top = 0 │ │ │ │ │ } else { │ │ │ │ │ - if ("scaleDenominator" in this.matrixIds[0]) { │ │ │ │ │ - var denom = OpenLayers.METERS_PER_INCH * OpenLayers.INCHES_PER_UNIT[this.units] * this.getServerResolution() / 28e-5; │ │ │ │ │ - var diff = Number.POSITIVE_INFINITY; │ │ │ │ │ - var delta; │ │ │ │ │ - for (var i = 0, ii = this.matrixIds.length; i < ii; ++i) { │ │ │ │ │ - delta = Math.abs(1 - this.matrixIds[i].scaleDenominator / denom); │ │ │ │ │ - if (delta < diff) { │ │ │ │ │ - diff = delta; │ │ │ │ │ - matrix = this.matrixIds[i] │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - matrix = this.matrixIds[this.getIdentifier()] │ │ │ │ │ - } │ │ │ │ │ + left = left - this.offset.x; │ │ │ │ │ + top = top - this.offset.y │ │ │ │ │ } │ │ │ │ │ - return matrix │ │ │ │ │ + var org = left - this.xOffset + " " + top; │ │ │ │ │ + this.root.coordorigin = org; │ │ │ │ │ + var roots = [this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ + var root; │ │ │ │ │ + for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ + root = roots[i]; │ │ │ │ │ + var size = this.size.w + " " + this.size.h; │ │ │ │ │ + root.coordsize = size │ │ │ │ │ + } │ │ │ │ │ + this.root.style.flip = "y"; │ │ │ │ │ + return coordSysUnchanged │ │ │ │ │ }, │ │ │ │ │ - getTileInfo: function(loc) { │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var fx = (loc.lon - this.tileOrigin.lon) / (res * this.tileSize.w); │ │ │ │ │ - var fy = (this.tileOrigin.lat - loc.lat) / (res * this.tileSize.h); │ │ │ │ │ - var col = Math.floor(fx); │ │ │ │ │ - var row = Math.floor(fy); │ │ │ │ │ - return { │ │ │ │ │ - col: col, │ │ │ │ │ - row: row, │ │ │ │ │ - i: Math.floor((fx - col) * this.tileSize.w), │ │ │ │ │ - j: Math.floor((fy - row) * this.tileSize.h) │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ + var roots = [this.rendererRoot, this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ + var w = this.size.w + "px"; │ │ │ │ │ + var h = this.size.h + "px"; │ │ │ │ │ + var root; │ │ │ │ │ + for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ + root = roots[i]; │ │ │ │ │ + root.style.width = w; │ │ │ │ │ + root.style.height = h │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var url = ""; │ │ │ │ │ - if (!this.tileFullExtent || this.tileFullExtent.intersectsBounds(bounds)) { │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var info = this.getTileInfo(center); │ │ │ │ │ - var matrixId = this.matrix.identifier; │ │ │ │ │ - var dimensions = this.dimensions, │ │ │ │ │ - params; │ │ │ │ │ - if (OpenLayers.Util.isArray(this.url)) { │ │ │ │ │ - url = this.selectUrl([this.version, this.style, this.matrixSet, this.matrix.identifier, info.row, info.col].join(","), this.url) │ │ │ │ │ - } else { │ │ │ │ │ - url = this.url │ │ │ │ │ - } │ │ │ │ │ - if (this.requestEncoding.toUpperCase() === "REST") { │ │ │ │ │ - params = this.params; │ │ │ │ │ - if (url.indexOf("{") !== -1) { │ │ │ │ │ - var template = url.replace(/\{/g, "${"); │ │ │ │ │ - var context = { │ │ │ │ │ - style: this.style, │ │ │ │ │ - Style: this.style, │ │ │ │ │ - TileMatrixSet: this.matrixSet, │ │ │ │ │ - TileMatrix: this.matrix.identifier, │ │ │ │ │ - TileRow: info.row, │ │ │ │ │ - TileCol: info.col │ │ │ │ │ - }; │ │ │ │ │ - if (dimensions) { │ │ │ │ │ - var dimension, i; │ │ │ │ │ - for (i = dimensions.length - 1; i >= 0; --i) { │ │ │ │ │ - dimension = dimensions[i]; │ │ │ │ │ - context[dimension] = params[dimension.toUpperCase()] │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - url = OpenLayers.String.format(template, context) │ │ │ │ │ + getNodeType: function(geometry, style) { │ │ │ │ │ + var nodeType = null; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + nodeType = "olv:rect" │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + nodeType = "olv:shape" │ │ │ │ │ } else { │ │ │ │ │ - var path = this.version + "/" + this.layer + "/" + this.style + "/"; │ │ │ │ │ - if (dimensions) { │ │ │ │ │ - for (var i = 0; i < dimensions.length; i++) { │ │ │ │ │ - if (params[dimensions[i]]) { │ │ │ │ │ - path = path + params[dimensions[i]] + "/" │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - path = path + this.matrixSet + "/" + this.matrix.identifier + "/" + info.row + "/" + info.col + "." + this.formatSuffix; │ │ │ │ │ - if (!url.match(/\/$/)) { │ │ │ │ │ - url = url + "/" │ │ │ │ │ - } │ │ │ │ │ - url = url + path │ │ │ │ │ + nodeType = "olv:oval" │ │ │ │ │ } │ │ │ │ │ - } else if (this.requestEncoding.toUpperCase() === "KVP") { │ │ │ │ │ - params = { │ │ │ │ │ - SERVICE: "WMTS", │ │ │ │ │ - REQUEST: "GetTile", │ │ │ │ │ - VERSION: this.version, │ │ │ │ │ - LAYER: this.layer, │ │ │ │ │ - STYLE: this.style, │ │ │ │ │ - TILEMATRIXSET: this.matrixSet, │ │ │ │ │ - TILEMATRIX: this.matrix.identifier, │ │ │ │ │ - TILEROW: info.row, │ │ │ │ │ - TILECOL: info.col, │ │ │ │ │ - FORMAT: this.format │ │ │ │ │ - }; │ │ │ │ │ - url = OpenLayers.Layer.Grid.prototype.getFullRequestString.apply(this, [params]) │ │ │ │ │ - } │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + nodeType = "olv:rect"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + case "OpenLayers.Geometry.Curve": │ │ │ │ │ + nodeType = "olv:shape"; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - return url │ │ │ │ │ + return nodeType │ │ │ │ │ }, │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - if (this.requestEncoding.toUpperCase() === "KVP") { │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, [OpenLayers.Util.upperCaseObject(newParams)]) │ │ │ │ │ + setStyle: function(node, style, options, geometry) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + options = options || node._options; │ │ │ │ │ + var fillColor = style.fillColor; │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + node.title = title │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.WMTS" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.ArcGIS93Rest = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - DEFAULT_PARAMS: { │ │ │ │ │ - format: "png" │ │ │ │ │ - }, │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - initialize: function(name, url, params, options) { │ │ │ │ │ - var newArguments = []; │ │ │ │ │ - params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ - newArguments.push(name, url, params, options); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ - OpenLayers.Util.applyDefaults(this.params, OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS)); │ │ │ │ │ - if (this.params.TRANSPARENT && this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ - if (options == null || !options.isBaseLayer) { │ │ │ │ │ - this.isBaseLayer = false │ │ │ │ │ - } │ │ │ │ │ - if (this.params.FORMAT == "jpg") { │ │ │ │ │ - this.params.FORMAT = OpenLayers.Util.alphaHack() ? "gif" : "png" │ │ │ │ │ + if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + options.isFilled = true; │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ + var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ + node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x + xOffset | 0) + "px"; │ │ │ │ │ + node.style.top = (geometry.y / resolution - this.offset.y - (yOffset + height) | 0) + "px"; │ │ │ │ │ + node.style.width = width + "px"; │ │ │ │ │ + node.style.height = height + "px"; │ │ │ │ │ + node.style.flip = "y"; │ │ │ │ │ + fillColor = "none"; │ │ │ │ │ + options.isStroked = false │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + var cache = this.importSymbol(style.graphicName); │ │ │ │ │ + node.path = cache.path; │ │ │ │ │ + node.coordorigin = cache.left + "," + cache.bottom; │ │ │ │ │ + var size = cache.size; │ │ │ │ │ + node.coordsize = size + "," + size; │ │ │ │ │ + this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ + node.style.flip = "y" │ │ │ │ │ + } else { │ │ │ │ │ + this.drawCircle(node, geometry, style.pointRadius) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.ArcGIS93Rest(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ + if (options.isFilled) { │ │ │ │ │ + node.fillcolor = fillColor │ │ │ │ │ + } else { │ │ │ │ │ + node.filled = "false" │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var projWords = this.projection.getCode().split(":"); │ │ │ │ │ - var srid = projWords[projWords.length - 1]; │ │ │ │ │ - var imageSize = this.getImageSize(); │ │ │ │ │ - var newParams = { │ │ │ │ │ - BBOX: bounds.toBBOX(), │ │ │ │ │ - SIZE: imageSize.w + "," + imageSize.h, │ │ │ │ │ - F: "image", │ │ │ │ │ - BBOXSR: srid, │ │ │ │ │ - IMAGESR: srid │ │ │ │ │ - }; │ │ │ │ │ - if (this.layerDefs) { │ │ │ │ │ - var layerDefStrList = []; │ │ │ │ │ - var layerID; │ │ │ │ │ - for (layerID in this.layerDefs) { │ │ │ │ │ - if (this.layerDefs.hasOwnProperty(layerID)) { │ │ │ │ │ - if (this.layerDefs[layerID]) { │ │ │ │ │ - layerDefStrList.push(layerID); │ │ │ │ │ - layerDefStrList.push(":"); │ │ │ │ │ - layerDefStrList.push(this.layerDefs[layerID]); │ │ │ │ │ - layerDefStrList.push(";") │ │ │ │ │ - } │ │ │ │ │ + var fills = node.getElementsByTagName("fill"); │ │ │ │ │ + var fill = fills.length == 0 ? null : fills[0]; │ │ │ │ │ + if (!options.isFilled) { │ │ │ │ │ + if (fill) { │ │ │ │ │ + node.removeChild(fill) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + if (!fill) { │ │ │ │ │ + fill = this.createNode("olv:fill", node.id + "_fill") │ │ │ │ │ + } │ │ │ │ │ + fill.opacity = style.fillOpacity; │ │ │ │ │ + if (node._geometryClass === "OpenLayers.Geometry.Point" && style.externalGraphic) { │ │ │ │ │ + if (style.graphicOpacity) { │ │ │ │ │ + fill.opacity = style.graphicOpacity │ │ │ │ │ + } │ │ │ │ │ + fill.src = style.externalGraphic; │ │ │ │ │ + fill.type = "frame"; │ │ │ │ │ + if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ + fill.aspect = "atmost" │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (layerDefStrList.length > 0) { │ │ │ │ │ - newParams["LAYERDEFS"] = layerDefStrList.join("") │ │ │ │ │ + if (fill.parentNode != node) { │ │ │ │ │ + node.appendChild(fill) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var requestString = this.getFullRequestString(newParams); │ │ │ │ │ - return requestString │ │ │ │ │ - }, │ │ │ │ │ - setLayerFilter: function(id, queryDef) { │ │ │ │ │ - if (!this.layerDefs) { │ │ │ │ │ - this.layerDefs = {} │ │ │ │ │ + var rotation = style.rotation; │ │ │ │ │ + if (rotation !== undefined || node._rotation !== undefined) { │ │ │ │ │ + node._rotation = rotation; │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ + fill.opacity = 0 │ │ │ │ │ + } else if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ + node.style.rotation = rotation || 0 │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (queryDef) { │ │ │ │ │ - this.layerDefs[id] = queryDef │ │ │ │ │ + var strokes = node.getElementsByTagName("stroke"); │ │ │ │ │ + var stroke = strokes.length == 0 ? null : strokes[0]; │ │ │ │ │ + if (!options.isStroked) { │ │ │ │ │ + node.stroked = false; │ │ │ │ │ + if (stroke) { │ │ │ │ │ + stroke.on = false │ │ │ │ │ + } │ │ │ │ │ } else { │ │ │ │ │ - delete this.layerDefs[id] │ │ │ │ │ + if (!stroke) { │ │ │ │ │ + stroke = this.createNode("olv:stroke", node.id + "_stroke"); │ │ │ │ │ + node.appendChild(stroke) │ │ │ │ │ + } │ │ │ │ │ + stroke.on = true; │ │ │ │ │ + stroke.color = style.strokeColor; │ │ │ │ │ + stroke.weight = style.strokeWidth + "px"; │ │ │ │ │ + stroke.opacity = style.strokeOpacity; │ │ │ │ │ + stroke.endcap = style.strokeLinecap == "butt" ? "flat" : style.strokeLinecap || "round"; │ │ │ │ │ + if (style.strokeDashstyle) { │ │ │ │ │ + stroke.dashstyle = this.dashStyle(style) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ + node.style.cursor = style.cursor │ │ │ │ │ } │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - clearLayerFilter: function(id) { │ │ │ │ │ - if (id) { │ │ │ │ │ - delete this.layerDefs[id] │ │ │ │ │ + graphicRotate: function(node, xOffset, yOffset, style) { │ │ │ │ │ + var style = style || node._style; │ │ │ │ │ + var rotation = style.rotation || 0; │ │ │ │ │ + var aspectRatio, size; │ │ │ │ │ + if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ + var img = new Image; │ │ │ │ │ + img.onreadystatechange = OpenLayers.Function.bind(function() { │ │ │ │ │ + if (img.readyState == "complete" || img.readyState == "interactive") { │ │ │ │ │ + aspectRatio = img.width / img.height; │ │ │ │ │ + size = Math.max(style.pointRadius * 2, style.graphicWidth || 0, style.graphicHeight || 0); │ │ │ │ │ + xOffset = xOffset * aspectRatio; │ │ │ │ │ + style.graphicWidth = size * aspectRatio; │ │ │ │ │ + style.graphicHeight = size; │ │ │ │ │ + this.graphicRotate(node, xOffset, yOffset, style) │ │ │ │ │ + } │ │ │ │ │ + }, this); │ │ │ │ │ + img.src = style.externalGraphic; │ │ │ │ │ + return │ │ │ │ │ } else { │ │ │ │ │ - delete this.layerDefs │ │ │ │ │ + size = Math.max(style.graphicWidth, style.graphicHeight); │ │ │ │ │ + aspectRatio = style.graphicWidth / style.graphicHeight │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - mergeNewParams: function(newParams) { │ │ │ │ │ - var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ - var newArguments = [upperParams]; │ │ │ │ │ - return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, newArguments) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.ArcGIS93Rest" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Zoomify = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ - size: null, │ │ │ │ │ - isBaseLayer: true, │ │ │ │ │ - standardTileSize: 256, │ │ │ │ │ - tileOriginCorner: "tl", │ │ │ │ │ - numberOfTiers: 0, │ │ │ │ │ - tileCountUpToTier: null, │ │ │ │ │ - tierSizeInTiles: null, │ │ │ │ │ - tierImageSize: null, │ │ │ │ │ - initialize: function(name, url, size, options) { │ │ │ │ │ - this.initializeZoomify(size); │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name, url, size, {}, options]) │ │ │ │ │ - }, │ │ │ │ │ - initializeZoomify: function(size) { │ │ │ │ │ - var imageSize = size.clone(); │ │ │ │ │ - this.size = size.clone(); │ │ │ │ │ - var tiles = new OpenLayers.Size(Math.ceil(imageSize.w / this.standardTileSize), Math.ceil(imageSize.h / this.standardTileSize)); │ │ │ │ │ - this.tierSizeInTiles = [tiles]; │ │ │ │ │ - this.tierImageSize = [imageSize]; │ │ │ │ │ - while (imageSize.w > this.standardTileSize || imageSize.h > this.standardTileSize) { │ │ │ │ │ - imageSize = new OpenLayers.Size(Math.floor(imageSize.w / 2), Math.floor(imageSize.h / 2)); │ │ │ │ │ - tiles = new OpenLayers.Size(Math.ceil(imageSize.w / this.standardTileSize), Math.ceil(imageSize.h / this.standardTileSize)); │ │ │ │ │ - this.tierSizeInTiles.push(tiles); │ │ │ │ │ - this.tierImageSize.push(imageSize) │ │ │ │ │ + var width = Math.round(style.graphicWidth || size * aspectRatio); │ │ │ │ │ + var height = Math.round(style.graphicHeight || size); │ │ │ │ │ + node.style.width = width + "px"; │ │ │ │ │ + node.style.height = height + "px"; │ │ │ │ │ + var image = document.getElementById(node.id + "_image"); │ │ │ │ │ + if (!image) { │ │ │ │ │ + image = this.createNode("olv:imagedata", node.id + "_image"); │ │ │ │ │ + node.appendChild(image) │ │ │ │ │ } │ │ │ │ │ - this.tierSizeInTiles.reverse(); │ │ │ │ │ - this.tierImageSize.reverse(); │ │ │ │ │ - this.numberOfTiers = this.tierSizeInTiles.length; │ │ │ │ │ - var resolutions = [1]; │ │ │ │ │ - this.tileCountUpToTier = [0]; │ │ │ │ │ - for (var i = 1; i < this.numberOfTiers; i++) { │ │ │ │ │ - resolutions.unshift(Math.pow(2, i)); │ │ │ │ │ - this.tileCountUpToTier.push(this.tierSizeInTiles[i - 1].w * this.tierSizeInTiles[i - 1].h + this.tileCountUpToTier[i - 1]) │ │ │ │ │ + image.style.width = width + "px"; │ │ │ │ │ + image.style.height = height + "px"; │ │ │ │ │ + image.src = style.externalGraphic; │ │ │ │ │ + image.style.filter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(" + "src='', sizingMethod='scale')"; │ │ │ │ │ + var rot = rotation * Math.PI / 180; │ │ │ │ │ + var sintheta = Math.sin(rot); │ │ │ │ │ + var costheta = Math.cos(rot); │ │ │ │ │ + var filter = "progid:DXImageTransform.Microsoft.Matrix(M11=" + costheta + ",M12=" + -sintheta + ",M21=" + sintheta + ",M22=" + costheta + ",SizingMethod='auto expand')\n"; │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ + if (opacity && opacity != 1) { │ │ │ │ │ + filter += "progid:DXImageTransform.Microsoft.BasicImage(opacity=" + opacity + ")\n" │ │ │ │ │ } │ │ │ │ │ - if (!this.serverResolutions) { │ │ │ │ │ - this.serverResolutions = resolutions │ │ │ │ │ + node.style.filter = filter; │ │ │ │ │ + var centerPoint = new OpenLayers.Geometry.Point(-xOffset, -yOffset); │ │ │ │ │ + var imgBox = new OpenLayers.Bounds(0, 0, width, height).toGeometry(); │ │ │ │ │ + imgBox.rotate(style.rotation, centerPoint); │ │ │ │ │ + var imgBounds = imgBox.getBounds(); │ │ │ │ │ + node.style.left = Math.round(parseInt(node.style.left) + imgBounds.left) + "px"; │ │ │ │ │ + node.style.top = Math.round(parseInt(node.style.top) - imgBounds.bottom) + "px" │ │ │ │ │ + }, │ │ │ │ │ + postDraw: function(node) { │ │ │ │ │ + node.style.visibility = "visible"; │ │ │ │ │ + var fillColor = node._style.fillColor; │ │ │ │ │ + var strokeColor = node._style.strokeColor; │ │ │ │ │ + if (fillColor == "none" && node.fillcolor != fillColor) { │ │ │ │ │ + node.fillcolor = fillColor │ │ │ │ │ + } │ │ │ │ │ + if (strokeColor == "none" && node.strokecolor != strokeColor) { │ │ │ │ │ + node.strokecolor = strokeColor │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.destroy.apply(this, arguments); │ │ │ │ │ - this.tileCountUpToTier.length = 0; │ │ │ │ │ - this.tierSizeInTiles.length = 0; │ │ │ │ │ - this.tierImageSize.length = 0 │ │ │ │ │ + setNodeDimension: function(node, geometry) { │ │ │ │ │ + var bbox = geometry.getBounds(); │ │ │ │ │ + if (bbox) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var scaledBox = new OpenLayers.Bounds((bbox.left - this.featureDx) / resolution - this.offset.x | 0, bbox.bottom / resolution - this.offset.y | 0, (bbox.right - this.featureDx) / resolution - this.offset.x | 0, bbox.top / resolution - this.offset.y | 0); │ │ │ │ │ + node.style.left = scaledBox.left + "px"; │ │ │ │ │ + node.style.top = scaledBox.top + "px"; │ │ │ │ │ + node.style.width = scaledBox.getWidth() + "px"; │ │ │ │ │ + node.style.height = scaledBox.getHeight() + "px"; │ │ │ │ │ + node.coordorigin = scaledBox.left + " " + scaledBox.top; │ │ │ │ │ + node.coordsize = scaledBox.getWidth() + " " + scaledBox.getHeight() │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - clone: function(obj) { │ │ │ │ │ - if (obj == null) { │ │ │ │ │ - obj = new OpenLayers.Layer.Zoomify(this.name, this.url, this.size, this.options) │ │ │ │ │ + dashStyle: function(style) { │ │ │ │ │ + var dash = style.strokeDashstyle; │ │ │ │ │ + switch (dash) { │ │ │ │ │ + case "solid": │ │ │ │ │ + case "dot": │ │ │ │ │ + case "dash": │ │ │ │ │ + case "dashdot": │ │ │ │ │ + case "longdash": │ │ │ │ │ + case "longdashdot": │ │ │ │ │ + return dash; │ │ │ │ │ + default: │ │ │ │ │ + var parts = dash.split(/[ ,]/); │ │ │ │ │ + if (parts.length == 2) { │ │ │ │ │ + if (1 * parts[0] >= 2 * parts[1]) { │ │ │ │ │ + return "longdash" │ │ │ │ │ + } │ │ │ │ │ + return parts[0] == 1 || parts[1] == 1 ? "dot" : "dash" │ │ │ │ │ + } else if (parts.length == 4) { │ │ │ │ │ + return 1 * parts[0] >= 2 * parts[1] ? "longdashdot" : "dashdot" │ │ │ │ │ + } │ │ │ │ │ + return "solid" │ │ │ │ │ } │ │ │ │ │ - obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ - return obj │ │ │ │ │ }, │ │ │ │ │ - getURL: function(bounds) { │ │ │ │ │ - bounds = this.adjustBounds(bounds); │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((this.tileOrigin.lat - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getZoomForResolution(res); │ │ │ │ │ - var tileIndex = x + y * this.tierSizeInTiles[z].w + this.tileCountUpToTier[z]; │ │ │ │ │ - var path = "TileGroup" + Math.floor(tileIndex / 256) + "/" + z + "-" + x + "-" + y + ".jpg"; │ │ │ │ │ - var url = this.url; │ │ │ │ │ - if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ - url = this.selectUrl(path, url) │ │ │ │ │ + createNode: function(type, id) { │ │ │ │ │ + var node = document.createElement(type); │ │ │ │ │ + if (id) { │ │ │ │ │ + node.id = id │ │ │ │ │ } │ │ │ │ │ - return url + path │ │ │ │ │ + node.unselectable = "on"; │ │ │ │ │ + node.onselectstart = OpenLayers.Function.False; │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - getImageSize: function() { │ │ │ │ │ - if (arguments.length > 0) { │ │ │ │ │ - var bounds = this.adjustBounds(arguments[0]); │ │ │ │ │ - var res = this.getServerResolution(); │ │ │ │ │ - var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ - var y = Math.round((this.tileOrigin.lat - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ - var z = this.getZoomForResolution(res); │ │ │ │ │ - var w = this.standardTileSize; │ │ │ │ │ - var h = this.standardTileSize; │ │ │ │ │ - if (x == this.tierSizeInTiles[z].w - 1) { │ │ │ │ │ - var w = this.tierImageSize[z].w % this.standardTileSize │ │ │ │ │ - } │ │ │ │ │ - if (y == this.tierSizeInTiles[z].h - 1) { │ │ │ │ │ - var h = this.tierImageSize[z].h % this.standardTileSize │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Size(w, h) │ │ │ │ │ - } else { │ │ │ │ │ - return this.tileSize │ │ │ │ │ + nodeTypeCompare: function(node, type) { │ │ │ │ │ + var subType = type; │ │ │ │ │ + var splitIndex = subType.indexOf(":"); │ │ │ │ │ + if (splitIndex != -1) { │ │ │ │ │ + subType = subType.substr(splitIndex + 1) │ │ │ │ │ + } │ │ │ │ │ + var nodeName = node.nodeName; │ │ │ │ │ + splitIndex = nodeName.indexOf(":"); │ │ │ │ │ + if (splitIndex != -1) { │ │ │ │ │ + nodeName = nodeName.substr(splitIndex + 1) │ │ │ │ │ } │ │ │ │ │ + return subType == nodeName │ │ │ │ │ }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.tileOrigin = new OpenLayers.LonLat(this.map.maxExtent.left, this.map.maxExtent.top) │ │ │ │ │ + createRenderRoot: function() { │ │ │ │ │ + return this.nodeFactory(this.container.id + "_vmlRoot", "div") │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Zoomify" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Layer.Google.v3 = { │ │ │ │ │ - DEFAULTS: { │ │ │ │ │ - sphericalMercator: true, │ │ │ │ │ - projection: "EPSG:900913" │ │ │ │ │ + createRoot: function(suffix) { │ │ │ │ │ + return this.nodeFactory(this.container.id + suffix, "olv:group") │ │ │ │ │ }, │ │ │ │ │ - animationEnabled: true, │ │ │ │ │ - loadMapObject: function() { │ │ │ │ │ - if (!this.type) { │ │ │ │ │ - this.type = google.maps.MapTypeId.ROADMAP │ │ │ │ │ - } │ │ │ │ │ - var mapObject; │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - mapObject = cache.mapObject; │ │ │ │ │ - ++cache.count │ │ │ │ │ - } else { │ │ │ │ │ - var center = this.map.getCenter(); │ │ │ │ │ - var container = document.createElement("div"); │ │ │ │ │ - container.className = "olForeignContainer"; │ │ │ │ │ - container.style.width = "100%"; │ │ │ │ │ - container.style.height = "100%"; │ │ │ │ │ - mapObject = new google.maps.Map(container, { │ │ │ │ │ - center: center ? new google.maps.LatLng(center.lat, center.lon) : new google.maps.LatLng(0, 0), │ │ │ │ │ - zoom: this.map.getZoom() || 0, │ │ │ │ │ - mapTypeId: this.type, │ │ │ │ │ - disableDefaultUI: true, │ │ │ │ │ - keyboardShortcuts: false, │ │ │ │ │ - draggable: false, │ │ │ │ │ - disableDoubleClickZoom: true, │ │ │ │ │ - scrollwheel: false, │ │ │ │ │ - streetViewControl: false │ │ │ │ │ - }); │ │ │ │ │ - var googleControl = document.createElement("div"); │ │ │ │ │ - googleControl.style.width = "100%"; │ │ │ │ │ - googleControl.style.height = "100%"; │ │ │ │ │ - mapObject.controls[google.maps.ControlPosition.TOP_LEFT].push(googleControl); │ │ │ │ │ - cache = { │ │ │ │ │ - googleControl: googleControl, │ │ │ │ │ - mapObject: mapObject, │ │ │ │ │ - count: 1 │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Layer.Google.cache[this.map.id] = cache │ │ │ │ │ + drawPoint: function(node, geometry) { │ │ │ │ │ + return this.drawCircle(node, geometry, 1) │ │ │ │ │ + }, │ │ │ │ │ + drawCircle: function(node, geometry, radius) { │ │ │ │ │ + if (!isNaN(geometry.x) && !isNaN(geometry.y)) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x | 0) - radius + "px"; │ │ │ │ │ + node.style.top = (geometry.y / resolution - this.offset.y | 0) - radius + "px"; │ │ │ │ │ + var diameter = radius * 2; │ │ │ │ │ + node.style.width = diameter + "px"; │ │ │ │ │ + node.style.height = diameter + "px"; │ │ │ │ │ + return node │ │ │ │ │ } │ │ │ │ │ - this.mapObject = mapObject; │ │ │ │ │ - this.setGMapVisibility(this.visibility) │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - onMapResize: function() { │ │ │ │ │ - if (this.visibility) { │ │ │ │ │ - google.maps.event.trigger(this.mapObject, "resize") │ │ │ │ │ + drawLineString: function(node, geometry) { │ │ │ │ │ + return this.drawLine(node, geometry, false) │ │ │ │ │ + }, │ │ │ │ │ + drawLinearRing: function(node, geometry) { │ │ │ │ │ + return this.drawLine(node, geometry, true) │ │ │ │ │ + }, │ │ │ │ │ + drawLine: function(node, geometry, closeLine) { │ │ │ │ │ + this.setNodeDimension(node, geometry); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var numComponents = geometry.components.length; │ │ │ │ │ + var parts = new Array(numComponents); │ │ │ │ │ + var comp, x, y; │ │ │ │ │ + for (var i = 0; i < numComponents; i++) { │ │ │ │ │ + comp = geometry.components[i]; │ │ │ │ │ + x = (comp.x - this.featureDx) / resolution - this.offset.x | 0; │ │ │ │ │ + y = comp.y / resolution - this.offset.y | 0; │ │ │ │ │ + parts[i] = " " + x + "," + y + " l " │ │ │ │ │ } │ │ │ │ │ + var end = closeLine ? " x e" : " e"; │ │ │ │ │ + node.path = "m" + parts.join("") + end; │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - setGMapVisibility: function(visible) { │ │ │ │ │ - var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ - var map = this.map; │ │ │ │ │ - if (cache) { │ │ │ │ │ - var type = this.type; │ │ │ │ │ - var layers = map.layers; │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = layers[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Google && layer.visibility === true && layer.inRange === true) { │ │ │ │ │ - type = layer.type; │ │ │ │ │ - visible = true; │ │ │ │ │ - break │ │ │ │ │ + drawPolygon: function(node, geometry) { │ │ │ │ │ + this.setNodeDimension(node, geometry); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var path = []; │ │ │ │ │ + var j, jj, points, area, first, second, i, ii, comp, pathComp, x, y; │ │ │ │ │ + for (j = 0, jj = geometry.components.length; j < jj; j++) { │ │ │ │ │ + path.push("m"); │ │ │ │ │ + points = geometry.components[j].components; │ │ │ │ │ + area = j === 0; │ │ │ │ │ + first = null; │ │ │ │ │ + second = null; │ │ │ │ │ + for (i = 0, ii = points.length; i < ii; i++) { │ │ │ │ │ + comp = points[i]; │ │ │ │ │ + x = (comp.x - this.featureDx) / resolution - this.offset.x | 0; │ │ │ │ │ + y = comp.y / resolution - this.offset.y | 0; │ │ │ │ │ + pathComp = " " + x + "," + y; │ │ │ │ │ + path.push(pathComp); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + path.push(" l") │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - var container = this.mapObject.getDiv(); │ │ │ │ │ - if (visible === true) { │ │ │ │ │ - if (container.parentNode !== map.div) { │ │ │ │ │ - if (!cache.rendered) { │ │ │ │ │ - var me = this; │ │ │ │ │ - google.maps.event.addListenerOnce(this.mapObject, "tilesloaded", function() { │ │ │ │ │ - cache.rendered = true; │ │ │ │ │ - me.setGMapVisibility(me.getVisibility()); │ │ │ │ │ - me.moveTo(me.map.getCenter()) │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - map.div.appendChild(container); │ │ │ │ │ - cache.googleControl.appendChild(map.viewPortDiv); │ │ │ │ │ - google.maps.event.trigger(this.mapObject, "resize") │ │ │ │ │ + if (!area) { │ │ │ │ │ + if (!first) { │ │ │ │ │ + first = pathComp │ │ │ │ │ + } else if (first != pathComp) { │ │ │ │ │ + if (!second) { │ │ │ │ │ + second = pathComp │ │ │ │ │ + } else if (second != pathComp) { │ │ │ │ │ + area = true │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.mapObject.setMapTypeId(type) │ │ │ │ │ - } else if (cache.googleControl.hasChildNodes()) { │ │ │ │ │ - map.div.appendChild(map.viewPortDiv); │ │ │ │ │ - map.div.removeChild(container) │ │ │ │ │ } │ │ │ │ │ + path.push(area ? " x " : " ") │ │ │ │ │ } │ │ │ │ │ + path.push("e"); │ │ │ │ │ + node.path = path.join(""); │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - getMapContainer: function() { │ │ │ │ │ - return this.mapObject.getDiv() │ │ │ │ │ + drawRectangle: function(node, geometry) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x | 0) + "px"; │ │ │ │ │ + node.style.top = (geometry.y / resolution - this.offset.y | 0) + "px"; │ │ │ │ │ + node.style.width = (geometry.width / resolution | 0) + "px"; │ │ │ │ │ + node.style.height = (geometry.height / resolution | 0) + "px"; │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ - var moBounds = null; │ │ │ │ │ - if (olBounds != null) { │ │ │ │ │ - var sw = this.sphericalMercator ? this.inverseMercator(olBounds.bottom, olBounds.left) : new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ - var ne = this.sphericalMercator ? this.inverseMercator(olBounds.top, olBounds.right) : new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ - moBounds = new google.maps.LatLngBounds(new google.maps.LatLng(sw.lat, sw.lon), new google.maps.LatLng(ne.lat, ne.lon)) │ │ │ │ │ + drawText: function(featureId, style, location) { │ │ │ │ │ + var label = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX, "olv:rect"); │ │ │ │ │ + var textbox = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX + "_textbox", "olv:textbox"); │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + label.style.left = ((location.x - this.featureDx) / resolution - this.offset.x | 0) + "px"; │ │ │ │ │ + label.style.top = (location.y / resolution - this.offset.y | 0) + "px"; │ │ │ │ │ + label.style.flip = "y"; │ │ │ │ │ + textbox.innerText = style.label; │ │ │ │ │ + if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ + textbox.style.cursor = style.cursor │ │ │ │ │ } │ │ │ │ │ - return moBounds │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ - var size = this.map.getSize(); │ │ │ │ │ - var lon = this.getLongitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ - var lat = this.getLatitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ - var res = this.map.getResolution(); │ │ │ │ │ - var delta_x = moPixel.x - size.w / 2; │ │ │ │ │ - var delta_y = moPixel.y - size.h / 2; │ │ │ │ │ - var lonlat = new OpenLayers.LonLat(lon + delta_x * res, lat - delta_y * res); │ │ │ │ │ - if (this.wrapDateLine) { │ │ │ │ │ - lonlat = lonlat.wrapDateLine(this.maxExtent) │ │ │ │ │ + if (style.fontColor) { │ │ │ │ │ + textbox.style.color = style.fontColor │ │ │ │ │ } │ │ │ │ │ - return this.getMapObjectLonLatFromLonLat(lonlat.lon, lonlat.lat) │ │ │ │ │ - }, │ │ │ │ │ - getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ - var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ - var res = this.map.getResolution(); │ │ │ │ │ - var extent = this.map.getExtent(); │ │ │ │ │ - return this.getMapObjectPixelFromXY(1 / res * (lon - extent.left), 1 / res * (extent.top - lat)) │ │ │ │ │ - }, │ │ │ │ │ - setMapObjectCenter: function(center, zoom) { │ │ │ │ │ - if (this.animationEnabled === false && zoom != this.mapObject.zoom) { │ │ │ │ │ - var mapContainer = this.getMapContainer(); │ │ │ │ │ - google.maps.event.addListenerOnce(this.mapObject, "idle", function() { │ │ │ │ │ - mapContainer.style.visibility = "" │ │ │ │ │ - }); │ │ │ │ │ - mapContainer.style.visibility = "hidden" │ │ │ │ │ + if (style.fontOpacity) { │ │ │ │ │ + textbox.style.filter = "alpha(opacity=" + style.fontOpacity * 100 + ")" │ │ │ │ │ } │ │ │ │ │ - this.mapObject.setOptions({ │ │ │ │ │ - center: center, │ │ │ │ │ - zoom: zoom │ │ │ │ │ - }) │ │ │ │ │ + if (style.fontFamily) { │ │ │ │ │ + textbox.style.fontFamily = style.fontFamily │ │ │ │ │ + } │ │ │ │ │ + if (style.fontSize) { │ │ │ │ │ + textbox.style.fontSize = style.fontSize │ │ │ │ │ + } │ │ │ │ │ + if (style.fontWeight) { │ │ │ │ │ + textbox.style.fontWeight = style.fontWeight │ │ │ │ │ + } │ │ │ │ │ + if (style.fontStyle) { │ │ │ │ │ + textbox.style.fontStyle = style.fontStyle │ │ │ │ │ + } │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + label._featureId = featureId; │ │ │ │ │ + textbox._featureId = featureId; │ │ │ │ │ + textbox._geometry = location; │ │ │ │ │ + textbox._geometryClass = location.CLASS_NAME │ │ │ │ │ + } │ │ │ │ │ + textbox.style.whiteSpace = "nowrap"; │ │ │ │ │ + textbox.inset = "1px,0px,0px,0px"; │ │ │ │ │ + if (!label.parentNode) { │ │ │ │ │ + label.appendChild(textbox); │ │ │ │ │ + this.textRoot.appendChild(label) │ │ │ │ │ + } │ │ │ │ │ + var align = style.labelAlign || "cm"; │ │ │ │ │ + if (align.length == 1) { │ │ │ │ │ + align += "m" │ │ │ │ │ + } │ │ │ │ │ + var xshift = textbox.clientWidth * OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(0, 1)]; │ │ │ │ │ + var yshift = textbox.clientHeight * OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(1, 1)]; │ │ │ │ │ + label.style.left = parseInt(label.style.left) - xshift - 1 + "px"; │ │ │ │ │ + label.style.top = parseInt(label.style.top) + yshift + "px" │ │ │ │ │ }, │ │ │ │ │ - getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ - return this.mapObject.getBoundsZoomLevel(moBounds) │ │ │ │ │ + moveRoot: function(renderer) { │ │ │ │ │ + var layer = this.map.getLayer(renderer.container.id); │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Vector.RootContainer) { │ │ │ │ │ + layer = this.map.getLayer(this.container.id) │ │ │ │ │ + } │ │ │ │ │ + layer && layer.renderer.clear(); │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.moveRoot.apply(this, arguments); │ │ │ │ │ + layer && layer.redraw() │ │ │ │ │ }, │ │ │ │ │ - getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ - var gLatLng; │ │ │ │ │ - if (this.sphericalMercator) { │ │ │ │ │ - var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ - gLatLng = new google.maps.LatLng(lonlat.lat, lonlat.lon) │ │ │ │ │ + importSymbol: function(graphicName) { │ │ │ │ │ + var id = this.container.id + "-" + graphicName; │ │ │ │ │ + var cache = this.symbolCache[id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + return cache │ │ │ │ │ + } │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(graphicName + " is not a valid symbol name") │ │ │ │ │ + } │ │ │ │ │ + var symbolExtent = new OpenLayers.Bounds(Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ + var pathitems = ["m"]; │ │ │ │ │ + for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + var x = symbol[i]; │ │ │ │ │ + var y = symbol[i + 1]; │ │ │ │ │ + symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ + symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ + symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ + symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ + pathitems.push(x); │ │ │ │ │ + pathitems.push(y); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + pathitems.push("l") │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + pathitems.push("x e"); │ │ │ │ │ + var path = pathitems.join(" "); │ │ │ │ │ + var diff = (symbolExtent.getWidth() - symbolExtent.getHeight()) / 2; │ │ │ │ │ + if (diff > 0) { │ │ │ │ │ + symbolExtent.bottom = symbolExtent.bottom - diff; │ │ │ │ │ + symbolExtent.top = symbolExtent.top + diff │ │ │ │ │ } else { │ │ │ │ │ - gLatLng = new google.maps.LatLng(lat, lon) │ │ │ │ │ + symbolExtent.left = symbolExtent.left + diff; │ │ │ │ │ + symbolExtent.right = symbolExtent.right - diff │ │ │ │ │ } │ │ │ │ │ - return gLatLng │ │ │ │ │ + cache = { │ │ │ │ │ + path: path, │ │ │ │ │ + size: symbolExtent.getWidth(), │ │ │ │ │ + left: symbolExtent.left, │ │ │ │ │ + bottom: symbolExtent.bottom │ │ │ │ │ + }; │ │ │ │ │ + this.symbolCache[id] = cache; │ │ │ │ │ + return cache │ │ │ │ │ }, │ │ │ │ │ - getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ - return new google.maps.Point(x, y) │ │ │ │ │ - } │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.VML" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Renderer.VML.LABEL_SHIFT = { │ │ │ │ │ + l: 0, │ │ │ │ │ + c: .5, │ │ │ │ │ + r: 1, │ │ │ │ │ + t: 0, │ │ │ │ │ + m: .5, │ │ │ │ │ + b: 1 │ │ │ │ │ }; │ │ │ │ │ -OpenLayers.Layer.Vector.RootContainer = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - layers: null, │ │ │ │ │ - display: function() {}, │ │ │ │ │ - getFeatureFromEvent: function(evt) { │ │ │ │ │ - var layers = this.layers; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0; i < layers.length; i++) { │ │ │ │ │ - feature = layers[i].getFeatureFromEvent(evt); │ │ │ │ │ - if (feature) { │ │ │ │ │ - return feature │ │ │ │ │ - } │ │ │ │ │ +OpenLayers.Renderer.SVG = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ + xmlns: "http://www.w3.org/2000/svg", │ │ │ │ │ + xlinkns: "http://www.w3.org/1999/xlink", │ │ │ │ │ + MAX_PIXEL: 15e3, │ │ │ │ │ + translationParameters: null, │ │ │ │ │ + symbolMetrics: null, │ │ │ │ │ + initialize: function(containerID) { │ │ │ │ │ + if (!this.supported()) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Renderer.Elements.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.translationParameters = { │ │ │ │ │ + x: 0, │ │ │ │ │ + y: 0 │ │ │ │ │ + }; │ │ │ │ │ + this.symbolMetrics = {} │ │ │ │ │ }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.collectRoots(); │ │ │ │ │ - map.events.register("changelayer", this, this.handleChangeLayer) │ │ │ │ │ + supported: function() { │ │ │ │ │ + var svgFeature = "http://www.w3.org/TR/SVG11/feature#"; │ │ │ │ │ + return document.implementation && (document.implementation.hasFeature("org.w3c.svg", "1.0") || document.implementation.hasFeature(svgFeature + "SVG", "1.1") || document.implementation.hasFeature(svgFeature + "BasicStructure", "1.1")) │ │ │ │ │ }, │ │ │ │ │ - removeMap: function(map) { │ │ │ │ │ - map.events.unregister("changelayer", this, this.handleChangeLayer); │ │ │ │ │ - this.resetRoots(); │ │ │ │ │ - OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments) │ │ │ │ │ + inValidRange: function(x, y, xyOnly) { │ │ │ │ │ + var left = x + (xyOnly ? 0 : this.translationParameters.x); │ │ │ │ │ + var top = y + (xyOnly ? 0 : this.translationParameters.y); │ │ │ │ │ + return left >= -this.MAX_PIXEL && left <= this.MAX_PIXEL && top >= -this.MAX_PIXEL && top <= this.MAX_PIXEL │ │ │ │ │ }, │ │ │ │ │ - collectRoots: function() { │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = 0; i < this.map.layers.length; ++i) { │ │ │ │ │ - layer = this.map.layers[i]; │ │ │ │ │ - if (OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ - layer.renderer.moveRoot(this.renderer) │ │ │ │ │ + setExtent: function(extent, resolutionChanged) { │ │ │ │ │ + var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ + var resolution = this.getResolution(), │ │ │ │ │ + left = -extent.left / resolution, │ │ │ │ │ + top = extent.top / resolution; │ │ │ │ │ + if (resolutionChanged) { │ │ │ │ │ + this.left = left; │ │ │ │ │ + this.top = top; │ │ │ │ │ + var extentString = "0 0 " + this.size.w + " " + this.size.h; │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "viewBox", extentString); │ │ │ │ │ + this.translate(this.xOffset, 0); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + var inRange = this.translate(left - this.left + this.xOffset, top - this.top); │ │ │ │ │ + if (!inRange) { │ │ │ │ │ + this.setExtent(extent, true) │ │ │ │ │ } │ │ │ │ │ + return coordSysUnchanged && inRange │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - resetRoots: function() { │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = 0; i < this.layers.length; ++i) { │ │ │ │ │ - layer = this.layers[i]; │ │ │ │ │ - if (this.renderer && layer.renderer.getRenderLayerId() == this.id) { │ │ │ │ │ - this.renderer.moveRoot(layer.renderer) │ │ │ │ │ + translate: function(x, y) { │ │ │ │ │ + if (!this.inValidRange(x, y, true)) { │ │ │ │ │ + return false │ │ │ │ │ + } else { │ │ │ │ │ + var transformString = ""; │ │ │ │ │ + if (x || y) { │ │ │ │ │ + transformString = "translate(" + x + "," + y + ")" │ │ │ │ │ } │ │ │ │ │ + this.root.setAttributeNS(null, "transform", transformString); │ │ │ │ │ + this.translationParameters = { │ │ │ │ │ + x: x, │ │ │ │ │ + y: y │ │ │ │ │ + }; │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - handleChangeLayer: function(evt) { │ │ │ │ │ - var layer = evt.layer; │ │ │ │ │ - if (evt.property == "order" && OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ - this.resetRoots(); │ │ │ │ │ - this.collectRoots() │ │ │ │ │ + setSize: function(size) { │ │ │ │ │ + OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "width", this.size.w); │ │ │ │ │ + this.rendererRoot.setAttributeNS(null, "height", this.size.h) │ │ │ │ │ + }, │ │ │ │ │ + getNodeType: function(geometry, style) { │ │ │ │ │ + var nodeType = null; │ │ │ │ │ + switch (geometry.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Geometry.Point": │ │ │ │ │ + if (style.externalGraphic) { │ │ │ │ │ + nodeType = "image" │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + nodeType = "svg" │ │ │ │ │ + } else { │ │ │ │ │ + nodeType = "circle" │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ + nodeType = "rect"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LineString": │ │ │ │ │ + nodeType = "polyline"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ + nodeType = "polygon"; │ │ │ │ │ + break; │ │ │ │ │ + case "OpenLayers.Geometry.Polygon": │ │ │ │ │ + case "OpenLayers.Geometry.Curve": │ │ │ │ │ + nodeType = "path"; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ + return nodeType │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Layer.Vector.RootContainer" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Popup.Framed = OpenLayers.Class(OpenLayers.Popup.Anchored, { │ │ │ │ │ - imageSrc: null, │ │ │ │ │ - imageSize: null, │ │ │ │ │ - isAlphaImage: false, │ │ │ │ │ - positionBlocks: null, │ │ │ │ │ - blocks: null, │ │ │ │ │ - fixedRelativePosition: false, │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, closeBoxCallback) { │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (this.fixedRelativePosition) { │ │ │ │ │ - this.updateRelativePosition(); │ │ │ │ │ - this.calculateRelativePosition = function(px) { │ │ │ │ │ - return this.relativePosition │ │ │ │ │ + setStyle: function(node, style, options) { │ │ │ │ │ + style = style || node._style; │ │ │ │ │ + options = options || node._options; │ │ │ │ │ + var title = style.title || style.graphicTitle; │ │ │ │ │ + if (title) { │ │ │ │ │ + node.setAttributeNS(null, "title", title); │ │ │ │ │ + var titleNode = node.getElementsByTagName("title"); │ │ │ │ │ + if (titleNode.length > 0) { │ │ │ │ │ + titleNode[0].firstChild.textContent = title │ │ │ │ │ + } else { │ │ │ │ │ + var label = this.nodeFactory(null, "title"); │ │ │ │ │ + label.textContent = title; │ │ │ │ │ + node.appendChild(label) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.contentDiv.style.position = "absolute"; │ │ │ │ │ - this.contentDiv.style.zIndex = 1; │ │ │ │ │ - if (closeBox) { │ │ │ │ │ - this.closeDiv.style.zIndex = 1 │ │ │ │ │ - } │ │ │ │ │ - this.groupDiv.style.position = "absolute"; │ │ │ │ │ - this.groupDiv.style.top = "0px"; │ │ │ │ │ - this.groupDiv.style.left = "0px"; │ │ │ │ │ - this.groupDiv.style.height = "100%"; │ │ │ │ │ - this.groupDiv.style.width = "100%" │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.imageSrc = null; │ │ │ │ │ - this.imageSize = null; │ │ │ │ │ - this.isAlphaImage = null; │ │ │ │ │ - this.fixedRelativePosition = false; │ │ │ │ │ - this.positionBlocks = null; │ │ │ │ │ - for (var i = 0; i < this.blocks.length; i++) { │ │ │ │ │ - var block = this.blocks[i]; │ │ │ │ │ - if (block.image) { │ │ │ │ │ - block.div.removeChild(block.image) │ │ │ │ │ + var r = parseFloat(node.getAttributeNS(null, "r")); │ │ │ │ │ + var widthFactor = 1; │ │ │ │ │ + var pos; │ │ │ │ │ + if (node._geometryClass == "OpenLayers.Geometry.Point" && r) { │ │ │ │ │ + node.style.visibility = ""; │ │ │ │ │ + if (style.graphic === false) { │ │ │ │ │ + node.style.visibility = "hidden" │ │ │ │ │ + } else if (style.externalGraphic) { │ │ │ │ │ + pos = this.getPosition(node); │ │ │ │ │ + if (style.graphicWidth && style.graphicHeight) { │ │ │ │ │ + node.setAttributeNS(null, "preserveAspectRatio", "none") │ │ │ │ │ + } │ │ │ │ │ + var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ + var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ + width = width ? width : style.pointRadius * 2; │ │ │ │ │ + height = height ? height : style.pointRadius * 2; │ │ │ │ │ + var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ + var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ + var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ + node.setAttributeNS(null, "x", (pos.x + xOffset).toFixed()); │ │ │ │ │ + node.setAttributeNS(null, "y", (pos.y + yOffset).toFixed()); │ │ │ │ │ + node.setAttributeNS(null, "width", width); │ │ │ │ │ + node.setAttributeNS(null, "height", height); │ │ │ │ │ + node.setAttributeNS(this.xlinkns, "xlink:href", style.externalGraphic); │ │ │ │ │ + node.setAttributeNS(null, "style", "opacity: " + opacity); │ │ │ │ │ + node.onclick = OpenLayers.Event.preventDefault │ │ │ │ │ + } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ + var offset = style.pointRadius * 3; │ │ │ │ │ + var size = offset * 2; │ │ │ │ │ + var src = this.importSymbol(style.graphicName); │ │ │ │ │ + pos = this.getPosition(node); │ │ │ │ │ + widthFactor = this.symbolMetrics[src.id][0] * 3 / size; │ │ │ │ │ + var parent = node.parentNode; │ │ │ │ │ + var nextSibling = node.nextSibling; │ │ │ │ │ + if (parent) { │ │ │ │ │ + parent.removeChild(node) │ │ │ │ │ + } │ │ │ │ │ + node.firstChild && node.removeChild(node.firstChild); │ │ │ │ │ + node.appendChild(src.firstChild.cloneNode(true)); │ │ │ │ │ + node.setAttributeNS(null, "viewBox", src.getAttributeNS(null, "viewBox")); │ │ │ │ │ + node.setAttributeNS(null, "width", size); │ │ │ │ │ + node.setAttributeNS(null, "height", size); │ │ │ │ │ + node.setAttributeNS(null, "x", pos.x - offset); │ │ │ │ │ + node.setAttributeNS(null, "y", pos.y - offset); │ │ │ │ │ + if (nextSibling) { │ │ │ │ │ + parent.insertBefore(node, nextSibling) │ │ │ │ │ + } else if (parent) { │ │ │ │ │ + parent.appendChild(node) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "r", style.pointRadius) │ │ │ │ │ } │ │ │ │ │ - block.image = null; │ │ │ │ │ - if (block.div) { │ │ │ │ │ - this.groupDiv.removeChild(block.div) │ │ │ │ │ + var rotation = style.rotation; │ │ │ │ │ + if ((rotation !== undefined || node._rotation !== undefined) && pos) { │ │ │ │ │ + node._rotation = rotation; │ │ │ │ │ + rotation |= 0; │ │ │ │ │ + if (node.nodeName !== "svg") { │ │ │ │ │ + node.setAttributeNS(null, "transform", "rotate(" + rotation + " " + pos.x + " " + pos.y + ")") │ │ │ │ │ + } else { │ │ │ │ │ + var metrics = this.symbolMetrics[src.id]; │ │ │ │ │ + node.firstChild.setAttributeNS(null, "transform", "rotate(" + rotation + " " + metrics[1] + " " + metrics[2] + ")") │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - block.div = null │ │ │ │ │ } │ │ │ │ │ - this.blocks = null; │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - setBackgroundColor: function(color) {}, │ │ │ │ │ - setBorder: function() {}, │ │ │ │ │ - setOpacity: function(opacity) {}, │ │ │ │ │ - setSize: function(contentSize) { │ │ │ │ │ - OpenLayers.Popup.Anchored.prototype.setSize.apply(this, arguments); │ │ │ │ │ - this.updateBlocks() │ │ │ │ │ - }, │ │ │ │ │ - updateRelativePosition: function() { │ │ │ │ │ - this.padding = this.positionBlocks[this.relativePosition].padding; │ │ │ │ │ - if (this.closeDiv) { │ │ │ │ │ - var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ - this.closeDiv.style.right = contentDivPadding.right + this.padding.right + "px"; │ │ │ │ │ - this.closeDiv.style.top = contentDivPadding.top + this.padding.top + "px" │ │ │ │ │ + if (options.isFilled) { │ │ │ │ │ + node.setAttributeNS(null, "fill", style.fillColor); │ │ │ │ │ + node.setAttributeNS(null, "fill-opacity", style.fillOpacity) │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "fill", "none") │ │ │ │ │ } │ │ │ │ │ - this.updateBlocks() │ │ │ │ │ - }, │ │ │ │ │ - calculateNewPx: function(px) { │ │ │ │ │ - var newPx = OpenLayers.Popup.Anchored.prototype.calculateNewPx.apply(this, arguments); │ │ │ │ │ - newPx = newPx.offset(this.positionBlocks[this.relativePosition].offset); │ │ │ │ │ - return newPx │ │ │ │ │ - }, │ │ │ │ │ - createBlocks: function() { │ │ │ │ │ - this.blocks = []; │ │ │ │ │ - var firstPosition = null; │ │ │ │ │ - for (var key in this.positionBlocks) { │ │ │ │ │ - firstPosition = key; │ │ │ │ │ - break │ │ │ │ │ + if (options.isStroked) { │ │ │ │ │ + node.setAttributeNS(null, "stroke", style.strokeColor); │ │ │ │ │ + node.setAttributeNS(null, "stroke-opacity", style.strokeOpacity); │ │ │ │ │ + node.setAttributeNS(null, "stroke-width", style.strokeWidth * widthFactor); │ │ │ │ │ + node.setAttributeNS(null, "stroke-linecap", style.strokeLinecap || "round"); │ │ │ │ │ + node.setAttributeNS(null, "stroke-linejoin", "round"); │ │ │ │ │ + style.strokeDashstyle && node.setAttributeNS(null, "stroke-dasharray", this.dashStyle(style, widthFactor)) │ │ │ │ │ + } else { │ │ │ │ │ + node.setAttributeNS(null, "stroke", "none") │ │ │ │ │ } │ │ │ │ │ - var position = this.positionBlocks[firstPosition]; │ │ │ │ │ - for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ - var block = {}; │ │ │ │ │ - this.blocks.push(block); │ │ │ │ │ - var divId = this.id + "_FrameDecorationDiv_" + i; │ │ │ │ │ - block.div = OpenLayers.Util.createDiv(divId, null, null, null, "absolute", null, "hidden", null); │ │ │ │ │ - var imgId = this.id + "_FrameDecorationImg_" + i; │ │ │ │ │ - var imageCreator = this.isAlphaImage ? OpenLayers.Util.createAlphaImageDiv : OpenLayers.Util.createImage; │ │ │ │ │ - block.image = imageCreator(imgId, null, this.imageSize, this.imageSrc, "absolute", null, null, null); │ │ │ │ │ - block.div.appendChild(block.image); │ │ │ │ │ - this.groupDiv.appendChild(block.div) │ │ │ │ │ + if (style.pointerEvents) { │ │ │ │ │ + node.setAttributeNS(null, "pointer-events", style.pointerEvents) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - updateBlocks: function() { │ │ │ │ │ - if (!this.blocks) { │ │ │ │ │ - this.createBlocks() │ │ │ │ │ + if (style.cursor != null) { │ │ │ │ │ + node.setAttributeNS(null, "cursor", style.cursor) │ │ │ │ │ } │ │ │ │ │ - if (this.size && this.relativePosition) { │ │ │ │ │ - var position = this.positionBlocks[this.relativePosition]; │ │ │ │ │ - for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ - var positionBlock = position.blocks[i]; │ │ │ │ │ - var block = this.blocks[i]; │ │ │ │ │ - var l = positionBlock.anchor.left; │ │ │ │ │ - var b = positionBlock.anchor.bottom; │ │ │ │ │ - var r = positionBlock.anchor.right; │ │ │ │ │ - var t = positionBlock.anchor.top; │ │ │ │ │ - var w = isNaN(positionBlock.size.w) ? this.size.w - (r + l) : positionBlock.size.w; │ │ │ │ │ - var h = isNaN(positionBlock.size.h) ? this.size.h - (b + t) : positionBlock.size.h; │ │ │ │ │ - block.div.style.width = (w < 0 ? 0 : w) + "px"; │ │ │ │ │ - block.div.style.height = (h < 0 ? 0 : h) + "px"; │ │ │ │ │ - block.div.style.left = l != null ? l + "px" : ""; │ │ │ │ │ - block.div.style.bottom = b != null ? b + "px" : ""; │ │ │ │ │ - block.div.style.right = r != null ? r + "px" : ""; │ │ │ │ │ - block.div.style.top = t != null ? t + "px" : ""; │ │ │ │ │ - block.image.style.left = positionBlock.position.x + "px"; │ │ │ │ │ - block.image.style.top = positionBlock.position.y + "px" │ │ │ │ │ - } │ │ │ │ │ - this.contentDiv.style.left = this.padding.left + "px"; │ │ │ │ │ - this.contentDiv.style.top = this.padding.top + "px" │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + dashStyle: function(style, widthFactor) { │ │ │ │ │ + var w = style.strokeWidth * widthFactor; │ │ │ │ │ + var str = style.strokeDashstyle; │ │ │ │ │ + switch (str) { │ │ │ │ │ + case "solid": │ │ │ │ │ + return "none"; │ │ │ │ │ + case "dot": │ │ │ │ │ + return [1, 4 * w].join(); │ │ │ │ │ + case "dash": │ │ │ │ │ + return [4 * w, 4 * w].join(); │ │ │ │ │ + case "dashdot": │ │ │ │ │ + return [4 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ + case "longdash": │ │ │ │ │ + return [8 * w, 4 * w].join(); │ │ │ │ │ + case "longdashdot": │ │ │ │ │ + return [8 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ + default: │ │ │ │ │ + return OpenLayers.String.trim(str).replace(/\s+/g, ",") │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup.Framed" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Popup.FramedCloud = OpenLayers.Class(OpenLayers.Popup.Framed, { │ │ │ │ │ - contentDisplayClass: "olFramedCloudPopupContent", │ │ │ │ │ - autoSize: true, │ │ │ │ │ - panMapIfOutOfView: true, │ │ │ │ │ - imageSize: new OpenLayers.Size(1276, 736), │ │ │ │ │ - isAlphaImage: false, │ │ │ │ │ - fixedRelativePosition: false, │ │ │ │ │ - positionBlocks: { │ │ │ │ │ - tl: { │ │ │ │ │ - offset: new OpenLayers.Pixel(44, 0), │ │ │ │ │ - padding: new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ - blocks: [{ │ │ │ │ │ - size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size("auto", 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, 18), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -632) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(81, 35), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -688) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - tr: { │ │ │ │ │ - offset: new OpenLayers.Pixel(-45, 0), │ │ │ │ │ - padding: new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ - blocks: [{ │ │ │ │ │ - size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size("auto", 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, 19), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -631) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(81, 35), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, null, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-215, -687) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - bl: { │ │ │ │ │ - offset: new OpenLayers.Pixel(45, 0), │ │ │ │ │ - padding: new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ - blocks: [{ │ │ │ │ │ - size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size("auto", 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(81, 33), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, null, 0, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-101, -674) │ │ │ │ │ - }] │ │ │ │ │ - }, │ │ │ │ │ - br: { │ │ │ │ │ - offset: new OpenLayers.Pixel(-44, 0), │ │ │ │ │ - padding: new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ - blocks: [{ │ │ │ │ │ - size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size("auto", 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ - position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(22, 21), │ │ │ │ │ - anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ - position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ - }, { │ │ │ │ │ - size: new OpenLayers.Size(81, 33), │ │ │ │ │ - anchor: new OpenLayers.Bounds(0, null, null, 0), │ │ │ │ │ - position: new OpenLayers.Pixel(-311, -674) │ │ │ │ │ - }] │ │ │ │ │ + createNode: function(type, id) { │ │ │ │ │ + var node = document.createElementNS(this.xmlns, type); │ │ │ │ │ + if (id) { │ │ │ │ │ + node.setAttributeNS(null, "id", id) │ │ │ │ │ } │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - minSize: new OpenLayers.Size(105, 10), │ │ │ │ │ - maxSize: new OpenLayers.Size(1200, 660), │ │ │ │ │ - initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, closeBoxCallback) { │ │ │ │ │ - this.imageSrc = OpenLayers.Util.getImageLocation("cloud-popup-relative.png"); │ │ │ │ │ - OpenLayers.Popup.Framed.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.contentDiv.className = this.contentDisplayClass │ │ │ │ │ + nodeTypeCompare: function(node, type) { │ │ │ │ │ + return type == node.nodeName │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Popup.FramedCloud" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WFSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - defaultVersion: "1.1.0", │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSCapabilities" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WPSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - defaultVersion: "1.0.0", │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WPSCapabilities" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.Atom = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - namespaces: { │ │ │ │ │ - atom: "http://www.w3.org/2005/Atom", │ │ │ │ │ - georss: "http://www.georss.org/georss" │ │ │ │ │ + createRenderRoot: function() { │ │ │ │ │ + var svg = this.nodeFactory(this.container.id + "_svgRoot", "svg"); │ │ │ │ │ + svg.style.display = "block"; │ │ │ │ │ + return svg │ │ │ │ │ }, │ │ │ │ │ - feedTitle: "untitled", │ │ │ │ │ - defaultEntryTitle: "untitled", │ │ │ │ │ - gmlParser: null, │ │ │ │ │ - xy: false, │ │ │ │ │ - read: function(doc) { │ │ │ │ │ - if (typeof doc == "string") { │ │ │ │ │ - doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]) │ │ │ │ │ + createRoot: function(suffix) { │ │ │ │ │ + return this.nodeFactory(this.container.id + suffix, "g") │ │ │ │ │ + }, │ │ │ │ │ + createDefs: function() { │ │ │ │ │ + var defs = this.nodeFactory(this.container.id + "_defs", "defs"); │ │ │ │ │ + this.rendererRoot.appendChild(defs); │ │ │ │ │ + return defs │ │ │ │ │ + }, │ │ │ │ │ + drawPoint: function(node, geometry) { │ │ │ │ │ + return this.drawCircle(node, geometry, 1) │ │ │ │ │ + }, │ │ │ │ │ + drawCircle: function(node, geometry, radius) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = (geometry.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y = this.top - geometry.y / resolution; │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + node.setAttributeNS(null, "cx", x); │ │ │ │ │ + node.setAttributeNS(null, "cy", y); │ │ │ │ │ + node.setAttributeNS(null, "r", radius); │ │ │ │ │ + return node │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return this.parseFeatures(doc) │ │ │ │ │ }, │ │ │ │ │ - write: function(features) { │ │ │ │ │ - var doc; │ │ │ │ │ - if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ - doc = this.createElementNSPlus("atom:feed"); │ │ │ │ │ - doc.appendChild(this.createElementNSPlus("atom:title", { │ │ │ │ │ - value: this.feedTitle │ │ │ │ │ - })); │ │ │ │ │ - for (var i = 0, ii = features.length; i < ii; i++) { │ │ │ │ │ - doc.appendChild(this.buildEntryNode(features[i])) │ │ │ │ │ - } │ │ │ │ │ + drawLineString: function(node, geometry) { │ │ │ │ │ + var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ + if (componentsResult.path) { │ │ │ │ │ + node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ + return componentsResult.complete ? node : null │ │ │ │ │ } else { │ │ │ │ │ - doc = this.buildEntryNode(features) │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [doc]) │ │ │ │ │ }, │ │ │ │ │ - buildContentNode: function(content) { │ │ │ │ │ - var node = this.createElementNSPlus("atom:content", { │ │ │ │ │ - attributes: { │ │ │ │ │ - type: content.type || null │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (content.src) { │ │ │ │ │ - node.setAttribute("src", content.src) │ │ │ │ │ + drawLinearRing: function(node, geometry) { │ │ │ │ │ + var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ + if (componentsResult.path) { │ │ │ │ │ + node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ + return componentsResult.complete ? node : null │ │ │ │ │ } else { │ │ │ │ │ - if (content.type == "text" || content.type == null) { │ │ │ │ │ - node.appendChild(this.createTextNode(content.value)) │ │ │ │ │ - } else if (content.type == "html") { │ │ │ │ │ - if (typeof content.value != "string") { │ │ │ │ │ - throw "HTML content must be in form of an escaped string" │ │ │ │ │ - } │ │ │ │ │ - node.appendChild(this.createTextNode(content.value)) │ │ │ │ │ - } else if (content.type == "xhtml") { │ │ │ │ │ - node.appendChild(content.value) │ │ │ │ │ - } else if (content.type == "xhtml" || content.type.match(/(\+|\/)xml$/)) { │ │ │ │ │ - node.appendChild(content.value) │ │ │ │ │ - } else { │ │ │ │ │ - node.appendChild(this.createTextNode(content.value)) │ │ │ │ │ - } │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ }, │ │ │ │ │ - buildEntryNode: function(feature) { │ │ │ │ │ - var attrib = feature.attributes; │ │ │ │ │ - var atomAttrib = attrib.atom || {}; │ │ │ │ │ - var entryNode = this.createElementNSPlus("atom:entry"); │ │ │ │ │ - if (atomAttrib.authors) { │ │ │ │ │ - var authors = OpenLayers.Util.isArray(atomAttrib.authors) ? atomAttrib.authors : [atomAttrib.authors]; │ │ │ │ │ - for (var i = 0, ii = authors.length; i < ii; i++) { │ │ │ │ │ - entryNode.appendChild(this.buildPersonConstructNode("author", authors[i])) │ │ │ │ │ + drawPolygon: function(node, geometry) { │ │ │ │ │ + var d = ""; │ │ │ │ │ + var draw = true; │ │ │ │ │ + var complete = true; │ │ │ │ │ + var linearRingResult, path; │ │ │ │ │ + for (var j = 0, len = geometry.components.length; j < len; j++) { │ │ │ │ │ + d += " M"; │ │ │ │ │ + linearRingResult = this.getComponentsString(geometry.components[j].components, " "); │ │ │ │ │ + path = linearRingResult.path; │ │ │ │ │ + if (path) { │ │ │ │ │ + d += " " + path; │ │ │ │ │ + complete = linearRingResult.complete && complete │ │ │ │ │ + } else { │ │ │ │ │ + draw = false │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (atomAttrib.categories) { │ │ │ │ │ - var categories = OpenLayers.Util.isArray(atomAttrib.categories) ? atomAttrib.categories : [atomAttrib.categories]; │ │ │ │ │ - var category; │ │ │ │ │ - for (var i = 0, ii = categories.length; i < ii; i++) { │ │ │ │ │ - category = categories[i]; │ │ │ │ │ - entryNode.appendChild(this.createElementNSPlus("atom:category", { │ │ │ │ │ - attributes: { │ │ │ │ │ - term: category.term, │ │ │ │ │ - scheme: category.scheme || null, │ │ │ │ │ - label: category.label || null │ │ │ │ │ - } │ │ │ │ │ - })) │ │ │ │ │ - } │ │ │ │ │ + d += " z"; │ │ │ │ │ + if (draw) { │ │ │ │ │ + node.setAttributeNS(null, "d", d); │ │ │ │ │ + node.setAttributeNS(null, "fill-rule", "evenodd"); │ │ │ │ │ + return complete ? node : null │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - if (atomAttrib.content) { │ │ │ │ │ - entryNode.appendChild(this.buildContentNode(atomAttrib.content)) │ │ │ │ │ + }, │ │ │ │ │ + drawRectangle: function(node, geometry) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = (geometry.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y = this.top - geometry.y / resolution; │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + node.setAttributeNS(null, "x", x); │ │ │ │ │ + node.setAttributeNS(null, "y", y); │ │ │ │ │ + node.setAttributeNS(null, "width", geometry.width / resolution); │ │ │ │ │ + node.setAttributeNS(null, "height", geometry.height / resolution); │ │ │ │ │ + return node │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - if (atomAttrib.contributors) { │ │ │ │ │ - var contributors = OpenLayers.Util.isArray(atomAttrib.contributors) ? atomAttrib.contributors : [atomAttrib.contributors]; │ │ │ │ │ - for (var i = 0, ii = contributors.length; i < ii; i++) { │ │ │ │ │ - entryNode.appendChild(this.buildPersonConstructNode("contributor", contributors[i])) │ │ │ │ │ + }, │ │ │ │ │ + drawText: function(featureId, style, location) { │ │ │ │ │ + var drawOutline = !!style.labelOutlineWidth; │ │ │ │ │ + if (drawOutline) { │ │ │ │ │ + var outlineStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ + outlineStyle.fontColor = outlineStyle.labelOutlineColor; │ │ │ │ │ + outlineStyle.fontStrokeColor = outlineStyle.labelOutlineColor; │ │ │ │ │ + outlineStyle.fontStrokeWidth = style.labelOutlineWidth; │ │ │ │ │ + if (style.labelOutlineOpacity) { │ │ │ │ │ + outlineStyle.fontOpacity = style.labelOutlineOpacity │ │ │ │ │ } │ │ │ │ │ + delete outlineStyle.labelOutlineWidth; │ │ │ │ │ + this.drawText(featureId, outlineStyle, location) │ │ │ │ │ } │ │ │ │ │ - if (feature.fid) { │ │ │ │ │ - entryNode.appendChild(this.createElementNSPlus("atom:id", { │ │ │ │ │ - value: feature.fid │ │ │ │ │ - })) │ │ │ │ │ - } │ │ │ │ │ - if (atomAttrib.links) { │ │ │ │ │ - var links = OpenLayers.Util.isArray(atomAttrib.links) ? atomAttrib.links : [atomAttrib.links]; │ │ │ │ │ - var link; │ │ │ │ │ - for (var i = 0, ii = links.length; i < ii; i++) { │ │ │ │ │ - link = links[i]; │ │ │ │ │ - entryNode.appendChild(this.createElementNSPlus("atom:link", { │ │ │ │ │ - attributes: { │ │ │ │ │ - href: link.href, │ │ │ │ │ - rel: link.rel || null, │ │ │ │ │ - type: link.type || null, │ │ │ │ │ - hreflang: link.hreflang || null, │ │ │ │ │ - title: link.title || null, │ │ │ │ │ - length: link.length || null │ │ │ │ │ - } │ │ │ │ │ - })) │ │ │ │ │ - } │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = (location.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y = location.y / resolution - this.top; │ │ │ │ │ + var suffix = drawOutline ? this.LABEL_OUTLINE_SUFFIX : this.LABEL_ID_SUFFIX; │ │ │ │ │ + var label = this.nodeFactory(featureId + suffix, "text"); │ │ │ │ │ + label.setAttributeNS(null, "x", x); │ │ │ │ │ + label.setAttributeNS(null, "y", -y); │ │ │ │ │ + if (style.fontColor) { │ │ │ │ │ + label.setAttributeNS(null, "fill", style.fontColor) │ │ │ │ │ } │ │ │ │ │ - if (atomAttrib.published) { │ │ │ │ │ - entryNode.appendChild(this.createElementNSPlus("atom:published", { │ │ │ │ │ - value: atomAttrib.published │ │ │ │ │ - })) │ │ │ │ │ + if (style.fontStrokeColor) { │ │ │ │ │ + label.setAttributeNS(null, "stroke", style.fontStrokeColor) │ │ │ │ │ } │ │ │ │ │ - if (atomAttrib.rights) { │ │ │ │ │ - entryNode.appendChild(this.createElementNSPlus("atom:rights", { │ │ │ │ │ - value: atomAttrib.rights │ │ │ │ │ - })) │ │ │ │ │ + if (style.fontStrokeWidth) { │ │ │ │ │ + label.setAttributeNS(null, "stroke-width", style.fontStrokeWidth) │ │ │ │ │ } │ │ │ │ │ - if (atomAttrib.summary || attrib.description) { │ │ │ │ │ - entryNode.appendChild(this.createElementNSPlus("atom:summary", { │ │ │ │ │ - value: atomAttrib.summary || attrib.description │ │ │ │ │ - })) │ │ │ │ │ + if (style.fontOpacity) { │ │ │ │ │ + label.setAttributeNS(null, "opacity", style.fontOpacity) │ │ │ │ │ } │ │ │ │ │ - entryNode.appendChild(this.createElementNSPlus("atom:title", { │ │ │ │ │ - value: atomAttrib.title || attrib.title || this.defaultEntryTitle │ │ │ │ │ - })); │ │ │ │ │ - if (atomAttrib.updated) { │ │ │ │ │ - entryNode.appendChild(this.createElementNSPlus("atom:updated", { │ │ │ │ │ - value: atomAttrib.updated │ │ │ │ │ - })) │ │ │ │ │ + if (style.fontFamily) { │ │ │ │ │ + label.setAttributeNS(null, "font-family", style.fontFamily) │ │ │ │ │ } │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - var whereNode = this.createElementNSPlus("georss:where"); │ │ │ │ │ - whereNode.appendChild(this.buildGeometryNode(feature.geometry)); │ │ │ │ │ - entryNode.appendChild(whereNode) │ │ │ │ │ + if (style.fontSize) { │ │ │ │ │ + label.setAttributeNS(null, "font-size", style.fontSize) │ │ │ │ │ } │ │ │ │ │ - return entryNode │ │ │ │ │ - }, │ │ │ │ │ - initGmlParser: function() { │ │ │ │ │ - this.gmlParser = new OpenLayers.Format.GML.v3({ │ │ │ │ │ - xy: this.xy, │ │ │ │ │ - featureNS: "http://example.com#feature", │ │ │ │ │ - internalProjection: this.internalProjection, │ │ │ │ │ - externalProjection: this.externalProjection │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - buildGeometryNode: function(geometry) { │ │ │ │ │ - if (!this.gmlParser) { │ │ │ │ │ - this.initGmlParser() │ │ │ │ │ + if (style.fontWeight) { │ │ │ │ │ + label.setAttributeNS(null, "font-weight", style.fontWeight) │ │ │ │ │ } │ │ │ │ │ - var node = this.gmlParser.writeNode("feature:_geometry", geometry); │ │ │ │ │ - return node.firstChild │ │ │ │ │ - }, │ │ │ │ │ - buildPersonConstructNode: function(name, value) { │ │ │ │ │ - var oNames = ["uri", "email"]; │ │ │ │ │ - var personNode = this.createElementNSPlus("atom:" + name); │ │ │ │ │ - personNode.appendChild(this.createElementNSPlus("atom:name", { │ │ │ │ │ - value: value.name │ │ │ │ │ - })); │ │ │ │ │ - for (var i = 0, ii = oNames.length; i < ii; i++) { │ │ │ │ │ - if (value[oNames[i]]) { │ │ │ │ │ - personNode.appendChild(this.createElementNSPlus("atom:" + oNames[i], { │ │ │ │ │ - value: value[oNames[i]] │ │ │ │ │ - })) │ │ │ │ │ - } │ │ │ │ │ + if (style.fontStyle) { │ │ │ │ │ + label.setAttributeNS(null, "font-style", style.fontStyle) │ │ │ │ │ } │ │ │ │ │ - return personNode │ │ │ │ │ - }, │ │ │ │ │ - getFirstChildValue: function(node, nsuri, name, def) { │ │ │ │ │ - var value; │ │ │ │ │ - var nodes = this.getElementsByTagNameNS(node, nsuri, name); │ │ │ │ │ - if (nodes && nodes.length > 0) { │ │ │ │ │ - value = this.getChildValue(nodes[0], def) │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + label.setAttributeNS(null, "pointer-events", "visible"); │ │ │ │ │ + label._featureId = featureId │ │ │ │ │ } else { │ │ │ │ │ - value = def │ │ │ │ │ + label.setAttributeNS(null, "pointer-events", "none") │ │ │ │ │ } │ │ │ │ │ - return value │ │ │ │ │ - }, │ │ │ │ │ - parseFeature: function(node) { │ │ │ │ │ - var atomAttrib = {}; │ │ │ │ │ - var value = null; │ │ │ │ │ - var nodes = null; │ │ │ │ │ - var attval = null; │ │ │ │ │ - var atomns = this.namespaces.atom; │ │ │ │ │ - this.parsePersonConstructs(node, "author", atomAttrib); │ │ │ │ │ - nodes = this.getElementsByTagNameNS(node, atomns, "category"); │ │ │ │ │ - if (nodes.length > 0) { │ │ │ │ │ - atomAttrib.categories = [] │ │ │ │ │ + var align = style.labelAlign || OpenLayers.Renderer.defaultSymbolizer.labelAlign; │ │ │ │ │ + label.setAttributeNS(null, "text-anchor", OpenLayers.Renderer.SVG.LABEL_ALIGN[align[0]] || "middle"); │ │ │ │ │ + if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ + label.setAttributeNS(null, "dominant-baseline", OpenLayers.Renderer.SVG.LABEL_ALIGN[align[1]] || "central") │ │ │ │ │ } │ │ │ │ │ - for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ - value = {}; │ │ │ │ │ - value.term = nodes[i].getAttribute("term"); │ │ │ │ │ - attval = nodes[i].getAttribute("scheme"); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value.scheme = attval │ │ │ │ │ - } │ │ │ │ │ - attval = nodes[i].getAttribute("label"); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value.label = attval │ │ │ │ │ - } │ │ │ │ │ - atomAttrib.categories.push(value) │ │ │ │ │ + var labelRows = style.label.split("\n"); │ │ │ │ │ + var numRows = labelRows.length; │ │ │ │ │ + while (label.childNodes.length > numRows) { │ │ │ │ │ + label.removeChild(label.lastChild) │ │ │ │ │ } │ │ │ │ │ - nodes = this.getElementsByTagNameNS(node, atomns, "content"); │ │ │ │ │ - if (nodes.length > 0) { │ │ │ │ │ - value = {}; │ │ │ │ │ - attval = nodes[0].getAttribute("type"); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value.type = attval │ │ │ │ │ + for (var i = 0; i < numRows; i++) { │ │ │ │ │ + var tspan = this.nodeFactory(featureId + suffix + "_tspan_" + i, "tspan"); │ │ │ │ │ + if (style.labelSelect === true) { │ │ │ │ │ + tspan._featureId = featureId; │ │ │ │ │ + tspan._geometry = location; │ │ │ │ │ + tspan._geometryClass = location.CLASS_NAME │ │ │ │ │ } │ │ │ │ │ - attval = nodes[0].getAttribute("src"); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value.src = attval │ │ │ │ │ - } else { │ │ │ │ │ - if (value.type == "text" || value.type == "html" || value.type == null) { │ │ │ │ │ - value.value = this.getFirstChildValue(node, atomns, "content", null) │ │ │ │ │ - } else if (value.type == "xhtml" || value.type.match(/(\+|\/)xml$/)) { │ │ │ │ │ - value.value = this.getChildEl(nodes[0]) │ │ │ │ │ - } else { │ │ │ │ │ - value.value = this.getFirstChildValue(node, atomns, "content", null) │ │ │ │ │ + if (OpenLayers.IS_GECKO === false) { │ │ │ │ │ + tspan.setAttributeNS(null, "baseline-shift", OpenLayers.Renderer.SVG.LABEL_VSHIFT[align[1]] || "-35%") │ │ │ │ │ + } │ │ │ │ │ + tspan.setAttribute("x", x); │ │ │ │ │ + if (i == 0) { │ │ │ │ │ + var vfactor = OpenLayers.Renderer.SVG.LABEL_VFACTOR[align[1]]; │ │ │ │ │ + if (vfactor == null) { │ │ │ │ │ + vfactor = -.5 │ │ │ │ │ } │ │ │ │ │ - atomAttrib.content = value │ │ │ │ │ + tspan.setAttribute("dy", vfactor * (numRows - 1) + "em") │ │ │ │ │ + } else { │ │ │ │ │ + tspan.setAttribute("dy", "1em") │ │ │ │ │ + } │ │ │ │ │ + tspan.textContent = labelRows[i] === "" ? " " : labelRows[i]; │ │ │ │ │ + if (!tspan.parentNode) { │ │ │ │ │ + label.appendChild(tspan) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.parsePersonConstructs(node, "contributor", atomAttrib); │ │ │ │ │ - atomAttrib.id = this.getFirstChildValue(node, atomns, "id", null); │ │ │ │ │ - nodes = this.getElementsByTagNameNS(node, atomns, "link"); │ │ │ │ │ - if (nodes.length > 0) { │ │ │ │ │ - atomAttrib.links = new Array(nodes.length) │ │ │ │ │ + if (!label.parentNode) { │ │ │ │ │ + this.textRoot.appendChild(label) │ │ │ │ │ } │ │ │ │ │ - var oAtts = ["rel", "type", "hreflang", "title", "length"]; │ │ │ │ │ - for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ - value = {}; │ │ │ │ │ - value.href = nodes[i].getAttribute("href"); │ │ │ │ │ - for (var j = 0, jj = oAtts.length; j < jj; j++) { │ │ │ │ │ - attval = nodes[i].getAttribute(oAtts[j]); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value[oAtts[j]] = attval │ │ │ │ │ + }, │ │ │ │ │ + getComponentsString: function(components, separator) { │ │ │ │ │ + var renderCmp = []; │ │ │ │ │ + var complete = true; │ │ │ │ │ + var len = components.length; │ │ │ │ │ + var strings = []; │ │ │ │ │ + var str, component; │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + component = components[i]; │ │ │ │ │ + renderCmp.push(component); │ │ │ │ │ + str = this.getShortString(component); │ │ │ │ │ + if (str) { │ │ │ │ │ + strings.push(str) │ │ │ │ │ + } else { │ │ │ │ │ + if (i > 0) { │ │ │ │ │ + if (this.getShortString(components[i - 1])) { │ │ │ │ │ + strings.push(this.clipLine(components[i], components[i - 1])) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (i < len - 1) { │ │ │ │ │ + if (this.getShortString(components[i + 1])) { │ │ │ │ │ + strings.push(this.clipLine(components[i], components[i + 1])) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + complete = false │ │ │ │ │ } │ │ │ │ │ - atomAttrib.links[i] = value │ │ │ │ │ } │ │ │ │ │ - value = this.getFirstChildValue(node, atomns, "published", null); │ │ │ │ │ - if (value) { │ │ │ │ │ - atomAttrib.published = value │ │ │ │ │ + return { │ │ │ │ │ + path: strings.join(separator || ","), │ │ │ │ │ + complete: complete │ │ │ │ │ } │ │ │ │ │ - value = this.getFirstChildValue(node, atomns, "rights", null); │ │ │ │ │ - if (value) { │ │ │ │ │ - atomAttrib.rights = value │ │ │ │ │ + }, │ │ │ │ │ + clipLine: function(badComponent, goodComponent) { │ │ │ │ │ + if (goodComponent.equals(badComponent)) { │ │ │ │ │ + return "" │ │ │ │ │ } │ │ │ │ │ - value = this.getFirstChildValue(node, atomns, "summary", null); │ │ │ │ │ - if (value) { │ │ │ │ │ - atomAttrib.summary = value │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var maxX = this.MAX_PIXEL - this.translationParameters.x; │ │ │ │ │ + var maxY = this.MAX_PIXEL - this.translationParameters.y; │ │ │ │ │ + var x1 = (goodComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y1 = this.top - goodComponent.y / resolution; │ │ │ │ │ + var x2 = (badComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y2 = this.top - badComponent.y / resolution; │ │ │ │ │ + var k; │ │ │ │ │ + if (x2 < -maxX || x2 > maxX) { │ │ │ │ │ + k = (y2 - y1) / (x2 - x1); │ │ │ │ │ + x2 = x2 < 0 ? -maxX : maxX; │ │ │ │ │ + y2 = y1 + (x2 - x1) * k │ │ │ │ │ } │ │ │ │ │ - atomAttrib.title = this.getFirstChildValue(node, atomns, "title", null); │ │ │ │ │ - atomAttrib.updated = this.getFirstChildValue(node, atomns, "updated", null); │ │ │ │ │ - var featureAttrib = { │ │ │ │ │ - title: atomAttrib.title, │ │ │ │ │ - description: atomAttrib.summary, │ │ │ │ │ - atom: atomAttrib │ │ │ │ │ - }; │ │ │ │ │ - var geometry = this.parseLocations(node)[0]; │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry, featureAttrib); │ │ │ │ │ - feature.fid = atomAttrib.id; │ │ │ │ │ - return feature │ │ │ │ │ + if (y2 < -maxY || y2 > maxY) { │ │ │ │ │ + k = (x2 - x1) / (y2 - y1); │ │ │ │ │ + y2 = y2 < 0 ? -maxY : maxY; │ │ │ │ │ + x2 = x1 + (y2 - y1) * k │ │ │ │ │ + } │ │ │ │ │ + return x2 + "," + y2 │ │ │ │ │ }, │ │ │ │ │ - parseFeatures: function(node) { │ │ │ │ │ - var features = []; │ │ │ │ │ - var entries = this.getElementsByTagNameNS(node, this.namespaces.atom, "entry"); │ │ │ │ │ - if (entries.length == 0) { │ │ │ │ │ - entries = [node] │ │ │ │ │ + getShortString: function(point) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var x = (point.x - this.featureDx) / resolution + this.left; │ │ │ │ │ + var y = this.top - point.y / resolution; │ │ │ │ │ + if (this.inValidRange(x, y)) { │ │ │ │ │ + return x + "," + y │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - for (var i = 0, ii = entries.length; i < ii; i++) { │ │ │ │ │ - features.push(this.parseFeature(entries[i])) │ │ │ │ │ + }, │ │ │ │ │ + getPosition: function(node) { │ │ │ │ │ + return { │ │ │ │ │ + x: parseFloat(node.getAttributeNS(null, "cx")), │ │ │ │ │ + y: parseFloat(node.getAttributeNS(null, "cy")) │ │ │ │ │ } │ │ │ │ │ - return features │ │ │ │ │ }, │ │ │ │ │ - parseLocations: function(node) { │ │ │ │ │ - var georssns = this.namespaces.georss; │ │ │ │ │ - var locations = { │ │ │ │ │ - components: [] │ │ │ │ │ - }; │ │ │ │ │ - var where = this.getElementsByTagNameNS(node, georssns, "where"); │ │ │ │ │ - if (where && where.length > 0) { │ │ │ │ │ - if (!this.gmlParser) { │ │ │ │ │ - this.initGmlParser() │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, ii = where.length; i < ii; i++) { │ │ │ │ │ - this.gmlParser.readChildNodes(where[i], locations) │ │ │ │ │ - } │ │ │ │ │ + importSymbol: function(graphicName) { │ │ │ │ │ + if (!this.defs) { │ │ │ │ │ + this.defs = this.createDefs() │ │ │ │ │ } │ │ │ │ │ - var components = locations.components; │ │ │ │ │ - var point = this.getElementsByTagNameNS(node, georssns, "point"); │ │ │ │ │ - if (point && point.length > 0) { │ │ │ │ │ - for (var i = 0, ii = point.length; i < ii; i++) { │ │ │ │ │ - var xy = OpenLayers.String.trim(point[i].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ - if (xy.length != 2) { │ │ │ │ │ - xy = OpenLayers.String.trim(point[i].firstChild.nodeValue).split(/\s*,\s*/) │ │ │ │ │ - } │ │ │ │ │ - components.push(new OpenLayers.Geometry.Point(xy[1], xy[0])) │ │ │ │ │ + var id = this.container.id + "-" + graphicName; │ │ │ │ │ + var existing = document.getElementById(id); │ │ │ │ │ + if (existing != null) { │ │ │ │ │ + return existing │ │ │ │ │ + } │ │ │ │ │ + var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ + if (!symbol) { │ │ │ │ │ + throw new Error(graphicName + " is not a valid symbol name") │ │ │ │ │ + } │ │ │ │ │ + var symbolNode = this.nodeFactory(id, "symbol"); │ │ │ │ │ + var node = this.nodeFactory(null, "polygon"); │ │ │ │ │ + symbolNode.appendChild(node); │ │ │ │ │ + var symbolExtent = new OpenLayers.Bounds(Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ + var points = []; │ │ │ │ │ + var x, y; │ │ │ │ │ + for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ + x = symbol[i]; │ │ │ │ │ + y = symbol[i + 1]; │ │ │ │ │ + symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ + symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ + symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ + symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ + points.push(x, ",", y) │ │ │ │ │ + } │ │ │ │ │ + node.setAttributeNS(null, "points", points.join(" ")); │ │ │ │ │ + var width = symbolExtent.getWidth(); │ │ │ │ │ + var height = symbolExtent.getHeight(); │ │ │ │ │ + var viewBox = [symbolExtent.left - width, symbolExtent.bottom - height, width * 3, height * 3]; │ │ │ │ │ + symbolNode.setAttributeNS(null, "viewBox", viewBox.join(" ")); │ │ │ │ │ + this.symbolMetrics[id] = [Math.max(width, height), symbolExtent.getCenterLonLat().lon, symbolExtent.getCenterLonLat().lat]; │ │ │ │ │ + this.defs.appendChild(symbolNode); │ │ │ │ │ + return symbolNode │ │ │ │ │ + }, │ │ │ │ │ + getFeatureIdFromEvent: function(evt) { │ │ │ │ │ + var featureId = OpenLayers.Renderer.Elements.prototype.getFeatureIdFromEvent.apply(this, arguments); │ │ │ │ │ + if (!featureId) { │ │ │ │ │ + var target = evt.target; │ │ │ │ │ + featureId = target.parentNode && target != this.rendererRoot ? target.parentNode._featureId : undefined │ │ │ │ │ + } │ │ │ │ │ + return featureId │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Renderer.SVG" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_ALIGN = { │ │ │ │ │ + l: "start", │ │ │ │ │ + r: "end", │ │ │ │ │ + b: "bottom", │ │ │ │ │ + t: "hanging" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_VSHIFT = { │ │ │ │ │ + t: "-70%", │ │ │ │ │ + b: "0" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.SVG.LABEL_VFACTOR = { │ │ │ │ │ + t: 0, │ │ │ │ │ + b: -1 │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Renderer.SVG.preventDefault = function(e) { │ │ │ │ │ + OpenLayers.Event.preventDefault(e) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol.CSW = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, OpenLayers.Protocol.CSW.DEFAULTS); │ │ │ │ │ + var cls = OpenLayers.Protocol.CSW["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported CSW version: " + options.version │ │ │ │ │ + } │ │ │ │ │ + return new cls(options) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol.CSW.DEFAULTS = { │ │ │ │ │ + version: "2.0.2" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol.SOS = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, OpenLayers.Protocol.SOS.DEFAULTS); │ │ │ │ │ + var cls = OpenLayers.Protocol.SOS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported SOS version: " + options.version │ │ │ │ │ + } │ │ │ │ │ + return new cls(options) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol.SOS.DEFAULTS = { │ │ │ │ │ + version: "1.0.0" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol.WFS = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, OpenLayers.Protocol.WFS.DEFAULTS); │ │ │ │ │ + var cls = OpenLayers.Protocol.WFS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported WFS version: " + options.version │ │ │ │ │ + } │ │ │ │ │ + return new cls(options) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol.WFS.fromWMSLayer = function(layer, options) { │ │ │ │ │ + var typeName, featurePrefix; │ │ │ │ │ + var param = layer.params["LAYERS"]; │ │ │ │ │ + var parts = (OpenLayers.Util.isArray(param) ? param[0] : param).split(":"); │ │ │ │ │ + if (parts.length > 1) { │ │ │ │ │ + featurePrefix = parts[0] │ │ │ │ │ + } │ │ │ │ │ + typeName = parts.pop(); │ │ │ │ │ + var protocolOptions = { │ │ │ │ │ + url: layer.url, │ │ │ │ │ + featureType: typeName, │ │ │ │ │ + featurePrefix: featurePrefix, │ │ │ │ │ + srsName: layer.projection && layer.projection.getCode() || layer.map && layer.map.getProjectionObject().getCode(), │ │ │ │ │ + version: "1.1.0" │ │ │ │ │ + }; │ │ │ │ │ + return new OpenLayers.Protocol.WFS(OpenLayers.Util.applyDefaults(options, protocolOptions)) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol.WFS.DEFAULTS = { │ │ │ │ │ + version: "1.0.0" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Protocol.HTTP = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + url: null, │ │ │ │ │ + headers: null, │ │ │ │ │ + params: null, │ │ │ │ │ + callback: null, │ │ │ │ │ + scope: null, │ │ │ │ │ + readWithPOST: false, │ │ │ │ │ + updateWithPOST: false, │ │ │ │ │ + deleteWithPOST: false, │ │ │ │ │ + wildcarded: false, │ │ │ │ │ + srsInBBOX: false, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + this.params = {}; │ │ │ │ │ + this.headers = {}; │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ + var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ + wildcarded: this.wildcarded, │ │ │ │ │ + srsInBBOX: this.srsInBBOX │ │ │ │ │ + }); │ │ │ │ │ + this.filterToParams = function(filter, params) { │ │ │ │ │ + return format.write(filter, params) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var line = this.getElementsByTagNameNS(node, georssns, "line"); │ │ │ │ │ - if (line && line.length > 0) { │ │ │ │ │ - var coords; │ │ │ │ │ - var p; │ │ │ │ │ - var points; │ │ │ │ │ - for (var i = 0, ii = line.length; i < ii; i++) { │ │ │ │ │ - coords = OpenLayers.String.trim(line[i].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ - points = []; │ │ │ │ │ - for (var j = 0, jj = coords.length; j < jj; j += 2) { │ │ │ │ │ - p = new OpenLayers.Geometry.Point(coords[j + 1], coords[j]); │ │ │ │ │ - points.push(p) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.params = null; │ │ │ │ │ + this.headers = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ + }, │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.params = OpenLayers.Util.applyDefaults(options.params, this.options.params); │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + if (options.filter && this.filterToParams) { │ │ │ │ │ + options.params = this.filterToParams(options.filter, options.params) │ │ │ │ │ + } │ │ │ │ │ + var readWithPOST = options.readWithPOST !== undefined ? options.readWithPOST : this.readWithPOST; │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + if (readWithPOST) { │ │ │ │ │ + var headers = options.headers || {}; │ │ │ │ │ + headers["Content-Type"] = "application/x-www-form-urlencoded"; │ │ │ │ │ + resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ + data: OpenLayers.Util.getParameterString(options.params), │ │ │ │ │ + headers: headers │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + resp.priv = OpenLayers.Request.GET({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + return resp │ │ │ │ │ + }, │ │ │ │ │ + handleRead: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options) │ │ │ │ │ + }, │ │ │ │ │ + create: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: features, │ │ │ │ │ + requestType: "create" │ │ │ │ │ + }); │ │ │ │ │ + resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleCreate, resp, options), │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(features) │ │ │ │ │ + }); │ │ │ │ │ + return resp │ │ │ │ │ + }, │ │ │ │ │ + handleCreate: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options) │ │ │ │ │ + }, │ │ │ │ │ + update: function(feature, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var url = options.url || feature.url || this.options.url + "/" + feature.fid; │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: feature, │ │ │ │ │ + requestType: "update" │ │ │ │ │ + }); │ │ │ │ │ + var method = this.updateWithPOST ? "POST" : "PUT"; │ │ │ │ │ + resp.priv = OpenLayers.Request[method]({ │ │ │ │ │ + url: url, │ │ │ │ │ + callback: this.createCallback(this.handleUpdate, resp, options), │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(feature) │ │ │ │ │ + }); │ │ │ │ │ + return resp │ │ │ │ │ + }, │ │ │ │ │ + handleUpdate: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options) │ │ │ │ │ + }, │ │ │ │ │ + delete: function(feature, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var url = options.url || feature.url || this.options.url + "/" + feature.fid; │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: feature, │ │ │ │ │ + requestType: "delete" │ │ │ │ │ + }); │ │ │ │ │ + var method = this.deleteWithPOST ? "POST" : "DELETE"; │ │ │ │ │ + var requestOptions = { │ │ │ │ │ + url: url, │ │ │ │ │ + callback: this.createCallback(this.handleDelete, resp, options), │ │ │ │ │ + headers: options.headers │ │ │ │ │ + }; │ │ │ │ │ + if (this.deleteWithPOST) { │ │ │ │ │ + requestOptions.data = this.format.write(feature) │ │ │ │ │ + } │ │ │ │ │ + resp.priv = OpenLayers.Request[method](requestOptions); │ │ │ │ │ + return resp │ │ │ │ │ + }, │ │ │ │ │ + handleDelete: function(resp, options) { │ │ │ │ │ + this.handleResponse(resp, options) │ │ │ │ │ + }, │ │ │ │ │ + handleResponse: function(resp, options) { │ │ │ │ │ + var request = resp.priv; │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + if (resp.requestType != "delete") { │ │ │ │ │ + resp.features = this.parseFeatures(request) │ │ │ │ │ } │ │ │ │ │ - components.push(new OpenLayers.Geometry.LineString(points)) │ │ │ │ │ + resp.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ + } else { │ │ │ │ │ + resp.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ + } │ │ │ │ │ + options.callback.call(options.scope, resp) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + parseFeatures: function(request) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ + } │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ + return this.format.read(doc) │ │ │ │ │ + }, │ │ │ │ │ + commit: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var resp = [], │ │ │ │ │ + nResponses = 0; │ │ │ │ │ + var types = {}; │ │ │ │ │ + types[OpenLayers.State.INSERT] = []; │ │ │ │ │ + types[OpenLayers.State.UPDATE] = []; │ │ │ │ │ + types[OpenLayers.State.DELETE] = []; │ │ │ │ │ + var feature, list, requestFeatures = []; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + list = types[feature.state]; │ │ │ │ │ + if (list) { │ │ │ │ │ + list.push(feature); │ │ │ │ │ + requestFeatures.push(feature) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var polygon = this.getElementsByTagNameNS(node, georssns, "polygon"); │ │ │ │ │ - if (polygon && polygon.length > 0) { │ │ │ │ │ - var coords; │ │ │ │ │ - var p; │ │ │ │ │ - var points; │ │ │ │ │ - for (var i = 0, ii = polygon.length; i < ii; i++) { │ │ │ │ │ - coords = OpenLayers.String.trim(polygon[i].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ - points = []; │ │ │ │ │ - for (var j = 0, jj = coords.length; j < jj; j += 2) { │ │ │ │ │ - p = new OpenLayers.Geometry.Point(coords[j + 1], coords[j]); │ │ │ │ │ - points.push(p) │ │ │ │ │ + var nRequests = (types[OpenLayers.State.INSERT].length > 0 ? 1 : 0) + types[OpenLayers.State.UPDATE].length + types[OpenLayers.State.DELETE].length; │ │ │ │ │ + var success = true; │ │ │ │ │ + var finalResponse = new OpenLayers.Protocol.Response({ │ │ │ │ │ + reqFeatures: requestFeatures │ │ │ │ │ + }); │ │ │ │ │ + │ │ │ │ │ + function insertCallback(response) { │ │ │ │ │ + var len = response.features ? response.features.length : 0; │ │ │ │ │ + var fids = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + fids[i] = response.features[i].fid │ │ │ │ │ + } │ │ │ │ │ + finalResponse.insertIds = fids; │ │ │ │ │ + callback.apply(this, [response]) │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function callback(response) { │ │ │ │ │ + this.callUserCallback(response, options); │ │ │ │ │ + success = success && response.success(); │ │ │ │ │ + nResponses++; │ │ │ │ │ + if (nResponses >= nRequests) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + finalResponse.code = success ? OpenLayers.Protocol.Response.SUCCESS : OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + options.callback.apply(options.scope, [finalResponse]) │ │ │ │ │ } │ │ │ │ │ - components.push(new OpenLayers.Geometry.Polygon([new OpenLayers.Geometry.LinearRing(points)])) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - for (var i = 0, ii = components.length; i < ii; i++) { │ │ │ │ │ - if (components[i]) { │ │ │ │ │ - components[i].transform(this.externalProjection, this.internalProjection) │ │ │ │ │ + var queue = types[OpenLayers.State.INSERT]; │ │ │ │ │ + if (queue.length > 0) { │ │ │ │ │ + resp.push(this.create(queue, OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: insertCallback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options.create))) │ │ │ │ │ + } │ │ │ │ │ + queue = types[OpenLayers.State.UPDATE]; │ │ │ │ │ + for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ + resp.push(this.update(queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: callback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options.update))) │ │ │ │ │ + } │ │ │ │ │ + queue = types[OpenLayers.State.DELETE]; │ │ │ │ │ + for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ + resp.push(this["delete"](queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: callback, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options["delete"]))) │ │ │ │ │ + } │ │ │ │ │ + return resp │ │ │ │ │ + }, │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + response.priv.abort() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + callUserCallback: function(resp, options) { │ │ │ │ │ + var opt = options[resp.requestType]; │ │ │ │ │ + if (opt && opt.callback) { │ │ │ │ │ + opt.callback.call(opt.scope, resp) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.HTTP" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Protocol.Script = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + url: null, │ │ │ │ │ + params: null, │ │ │ │ │ + callback: null, │ │ │ │ │ + callbackTemplate: "OpenLayers.Protocol.Script.registry.${id}", │ │ │ │ │ + callbackKey: "callback", │ │ │ │ │ + callbackPrefix: "", │ │ │ │ │ + scope: null, │ │ │ │ │ + format: null, │ │ │ │ │ + pendingRequests: null, │ │ │ │ │ + srsInBBOX: false, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + this.params = {}; │ │ │ │ │ + this.pendingRequests = {}; │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (!this.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.GeoJSON │ │ │ │ │ + } │ │ │ │ │ + if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ + var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ + srsInBBOX: this.srsInBBOX │ │ │ │ │ + }); │ │ │ │ │ + this.filterToParams = function(filter, params) { │ │ │ │ │ + return format.write(filter, params) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + options.params = OpenLayers.Util.applyDefaults(options.params, this.options.params); │ │ │ │ │ + if (options.filter && this.filterToParams) { │ │ │ │ │ + options.params = this.filterToParams(options.filter, options.params) │ │ │ │ │ + } │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + var request = this.createRequest(options.url, options.params, OpenLayers.Function.bind(function(data) { │ │ │ │ │ + response.data = data; │ │ │ │ │ + this.handleRead(response, options) │ │ │ │ │ + }, this)); │ │ │ │ │ + response.priv = request; │ │ │ │ │ + return response │ │ │ │ │ + }, │ │ │ │ │ + createRequest: function(url, params, callback) { │ │ │ │ │ + var id = OpenLayers.Protocol.Script.register(callback); │ │ │ │ │ + var name = OpenLayers.String.format(this.callbackTemplate, { │ │ │ │ │ + id: id │ │ │ │ │ + }); │ │ │ │ │ + params = OpenLayers.Util.extend({}, params); │ │ │ │ │ + params[this.callbackKey] = this.callbackPrefix + name; │ │ │ │ │ + url = OpenLayers.Util.urlAppend(url, OpenLayers.Util.getParameterString(params)); │ │ │ │ │ + var script = document.createElement("script"); │ │ │ │ │ + script.type = "text/javascript"; │ │ │ │ │ + script.src = url; │ │ │ │ │ + script.id = "OpenLayers_Protocol_Script_" + id; │ │ │ │ │ + this.pendingRequests[script.id] = script; │ │ │ │ │ + var head = document.getElementsByTagName("head")[0]; │ │ │ │ │ + head.appendChild(script); │ │ │ │ │ + return script │ │ │ │ │ + }, │ │ │ │ │ + destroyRequest: function(script) { │ │ │ │ │ + OpenLayers.Protocol.Script.unregister(script.id.split("_").pop()); │ │ │ │ │ + delete this.pendingRequests[script.id]; │ │ │ │ │ + if (script.parentNode) { │ │ │ │ │ + script.parentNode.removeChild(script) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + this.handleResponse(response, options) │ │ │ │ │ + }, │ │ │ │ │ + handleResponse: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + if (response.data) { │ │ │ │ │ + response.features = this.parseFeatures(response.data); │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ + } │ │ │ │ │ + this.destroyRequest(response.priv); │ │ │ │ │ + options.callback.call(options.scope, response) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + parseFeatures: function(data) { │ │ │ │ │ + return this.format.read(data) │ │ │ │ │ + }, │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + this.destroyRequest(response.priv) │ │ │ │ │ + } else { │ │ │ │ │ + for (var key in this.pendingRequests) { │ │ │ │ │ + this.destroyRequest(this.pendingRequests[key]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.abort(); │ │ │ │ │ + delete this.params; │ │ │ │ │ + delete this.format; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.Script" │ │ │ │ │ +}); │ │ │ │ │ +(function() { │ │ │ │ │ + var o = OpenLayers.Protocol.Script; │ │ │ │ │ + var counter = 0; │ │ │ │ │ + o.registry = {}; │ │ │ │ │ + o.register = function(callback) { │ │ │ │ │ + var id = "c" + ++counter; │ │ │ │ │ + o.registry[id] = function() { │ │ │ │ │ + callback.apply(this, arguments) │ │ │ │ │ + }; │ │ │ │ │ + return id │ │ │ │ │ + }; │ │ │ │ │ + o.unregister = function(id) { │ │ │ │ │ + delete o.registry[id] │ │ │ │ │ + } │ │ │ │ │ +})(); │ │ │ │ │ +OpenLayers.Protocol.WFS.v1 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + version: null, │ │ │ │ │ + srsName: "EPSG:4326", │ │ │ │ │ + featureType: null, │ │ │ │ │ + featureNS: null, │ │ │ │ │ + geometryName: "the_geom", │ │ │ │ │ + schema: null, │ │ │ │ │ + featurePrefix: "feature", │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + readFormat: null, │ │ │ │ │ + readOptions: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!options.format) { │ │ │ │ │ + this.format = OpenLayers.Format.WFST(OpenLayers.Util.extend({ │ │ │ │ │ + version: this.version, │ │ │ │ │ + featureType: this.featureType, │ │ │ │ │ + featureNS: this.featureNS, │ │ │ │ │ + featurePrefix: this.featurePrefix, │ │ │ │ │ + geometryName: this.geometryName, │ │ │ │ │ + srsName: this.srsName, │ │ │ │ │ + schema: this.schema │ │ │ │ │ + }, this.formatOptions)) │ │ │ │ │ + } │ │ │ │ │ + if (!options.geometryName && parseFloat(this.format.version) > 1) { │ │ │ │ │ + this.setGeometryName(null) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.options && !this.options.format) { │ │ │ │ │ + this.format.destroy() │ │ │ │ │ + } │ │ │ │ │ + this.format = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ + }, │ │ │ │ │ + read: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + var data = OpenLayers.Format.XML.prototype.write.apply(this.format, [this.format.writeNode("wfs:GetFeature", options)]); │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: data │ │ │ │ │ + }); │ │ │ │ │ + return response │ │ │ │ │ + }, │ │ │ │ │ + setFeatureType: function(featureType) { │ │ │ │ │ + this.featureType = featureType; │ │ │ │ │ + this.format.featureType = featureType │ │ │ │ │ + }, │ │ │ │ │ + setGeometryName: function(geometryName) { │ │ │ │ │ + this.geometryName = geometryName; │ │ │ │ │ + this.format.geometryName = geometryName │ │ │ │ │ + }, │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + var result = this.parseResponse(request, options.readOptions); │ │ │ │ │ + if (result && result.success !== false) { │ │ │ │ │ + if (options.readOptions && options.readOptions.output == "object") { │ │ │ │ │ + OpenLayers.Util.extend(response, result) │ │ │ │ │ + } else { │ │ │ │ │ + response.features = result │ │ │ │ │ + } │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + response.error = result │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ } │ │ │ │ │ + options.callback.call(options.scope, response) │ │ │ │ │ } │ │ │ │ │ - return components │ │ │ │ │ }, │ │ │ │ │ - parsePersonConstructs: function(node, name, data) { │ │ │ │ │ - var persons = []; │ │ │ │ │ - var atomns = this.namespaces.atom; │ │ │ │ │ - var nodes = this.getElementsByTagNameNS(node, atomns, name); │ │ │ │ │ - var oAtts = ["uri", "email"]; │ │ │ │ │ - for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ - var value = {}; │ │ │ │ │ - value.name = this.getFirstChildValue(nodes[i], atomns, "name", null); │ │ │ │ │ - for (var j = 0, jj = oAtts.length; j < jj; j++) { │ │ │ │ │ - var attval = this.getFirstChildValue(nodes[i], atomns, oAtts[j], null); │ │ │ │ │ - if (attval) { │ │ │ │ │ - value[oAtts[j]] = attval │ │ │ │ │ + parseResponse: function(request, options) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ + } │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ + var result = this.readFormat !== null ? this.readFormat.read(doc) : this.format.read(doc, options); │ │ │ │ │ + if (!this.featureNS) { │ │ │ │ │ + var format = this.readFormat || this.format; │ │ │ │ │ + this.featureNS = format.featureNS; │ │ │ │ │ + format.autoConfig = false; │ │ │ │ │ + if (!this.geometryName) { │ │ │ │ │ + this.setGeometryName(format.geometryName) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return result │ │ │ │ │ + }, │ │ │ │ │ + commit: function(features, options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "commit", │ │ │ │ │ + reqFeatures: features │ │ │ │ │ + }); │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: this.format.write(features, options), │ │ │ │ │ + callback: this.createCallback(this.handleCommit, response, options) │ │ │ │ │ + }); │ │ │ │ │ + return response │ │ │ │ │ + }, │ │ │ │ │ + handleCommit: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + var data = request.responseXML; │ │ │ │ │ + if (!data || !data.documentElement) { │ │ │ │ │ + data = request.responseText │ │ │ │ │ + } │ │ │ │ │ + var obj = this.format.read(data) || {}; │ │ │ │ │ + response.insertIds = obj.insertIds || []; │ │ │ │ │ + if (obj.success) { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ + response.error = obj │ │ │ │ │ + } │ │ │ │ │ + options.callback.call(options.scope, response) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + filterDelete: function(filter, options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "commit" │ │ │ │ │ + }); │ │ │ │ │ + var root = this.format.createElementNSPlus("wfs:Transaction", { │ │ │ │ │ + attributes: { │ │ │ │ │ + service: "WFS", │ │ │ │ │ + version: this.version │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + var deleteNode = this.format.createElementNSPlus("wfs:Delete", { │ │ │ │ │ + attributes: { │ │ │ │ │ + typeName: (options.featureNS ? this.featurePrefix + ":" : "") + options.featureType │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (options.featureNS) { │ │ │ │ │ + deleteNode.setAttribute("xmlns:" + this.featurePrefix, options.featureNS) │ │ │ │ │ + } │ │ │ │ │ + var filterNode = this.format.writeNode("ogc:Filter", filter); │ │ │ │ │ + deleteNode.appendChild(filterNode); │ │ │ │ │ + root.appendChild(deleteNode); │ │ │ │ │ + var data = OpenLayers.Format.XML.prototype.write.apply(this.format, [root]); │ │ │ │ │ + return OpenLayers.Request.POST({ │ │ │ │ │ + url: this.url, │ │ │ │ │ + callback: options.callback || function() {}, │ │ │ │ │ + data: data │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + abort: function(response) { │ │ │ │ │ + if (response) { │ │ │ │ │ + response.priv.abort() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.WFS.v1" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Protocol.WFS.v1_1_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ + version: "1.1.0", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Protocol.WFS.v1.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (this.outputFormat && !this.readFormat) { │ │ │ │ │ + if (this.outputFormat.toLowerCase() == "gml2") { │ │ │ │ │ + this.readFormat = new OpenLayers.Format.GML.v2({ │ │ │ │ │ + featureType: this.featureType, │ │ │ │ │ + featureNS: this.featureNS, │ │ │ │ │ + geometryName: this.geometryName │ │ │ │ │ + }) │ │ │ │ │ + } else if (this.outputFormat.toLowerCase() == "json") { │ │ │ │ │ + this.readFormat = new OpenLayers.Format.GeoJSON │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.WFS.v1_1_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.GML.v2 = OpenLayers.Class(OpenLayers.Format.GML.Base, { │ │ │ │ │ + schemaLocation: "http://www.opengis.net/gml http://schemas.opengis.net/gml/2.1.2/feature.xsd", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.GML.Base.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + gml: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + outerBoundaryIs: function(node, container) { │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + container.outer = obj.components[0] │ │ │ │ │ + }, │ │ │ │ │ + innerBoundaryIs: function(node, container) { │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + container.inner.push(obj.components[0]) │ │ │ │ │ + }, │ │ │ │ │ + Box: function(node, container) { │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + if (!container.components) { │ │ │ │ │ + container.components = [] │ │ │ │ │ } │ │ │ │ │ + var min = obj.points[0]; │ │ │ │ │ + var max = obj.points[1]; │ │ │ │ │ + container.components.push(new OpenLayers.Bounds(min.x, min.y, max.x, max.y)) │ │ │ │ │ } │ │ │ │ │ - persons.push(value) │ │ │ │ │ + }, OpenLayers.Format.GML.Base.prototype.readers["gml"]), │ │ │ │ │ + feature: OpenLayers.Format.GML.Base.prototype.readers["feature"], │ │ │ │ │ + wfs: OpenLayers.Format.GML.Base.prototype.readers["wfs"] │ │ │ │ │ + }, │ │ │ │ │ + write: function(features) { │ │ │ │ │ + var name; │ │ │ │ │ + if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ + name = "wfs:FeatureCollection" │ │ │ │ │ + } else { │ │ │ │ │ + name = "gml:featureMember" │ │ │ │ │ } │ │ │ │ │ - if (persons.length > 0) { │ │ │ │ │ - data[name + "s"] = persons │ │ │ │ │ + var root = this.writeNode(name, features); │ │ │ │ │ + this.setAttributeNS(root, this.namespaces["xsi"], "xsi:schemaLocation", this.schemaLocation); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [root]) │ │ │ │ │ + }, │ │ │ │ │ + writers: { │ │ │ │ │ + gml: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Point: function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:Point"); │ │ │ │ │ + this.writeNode("coordinates", [geometry], node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + coordinates: function(points) { │ │ │ │ │ + var numPoints = points.length; │ │ │ │ │ + var parts = new Array(numPoints); │ │ │ │ │ + var point; │ │ │ │ │ + for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ + point = points[i]; │ │ │ │ │ + if (this.xy) { │ │ │ │ │ + parts[i] = point.x + "," + point.y │ │ │ │ │ + } else { │ │ │ │ │ + parts[i] = point.y + "," + point.x │ │ │ │ │ + } │ │ │ │ │ + if (point.z != undefined) { │ │ │ │ │ + parts[i] += "," + point.z │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return this.createElementNSPlus("gml:coordinates", { │ │ │ │ │ + attributes: { │ │ │ │ │ + decimal: ".", │ │ │ │ │ + cs: ",", │ │ │ │ │ + ts: " " │ │ │ │ │ + }, │ │ │ │ │ + value: numPoints == 1 ? parts[0] : parts.join(" ") │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + LineString: function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:LineString"); │ │ │ │ │ + this.writeNode("coordinates", geometry.components, node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + Polygon: function(geometry) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:Polygon"); │ │ │ │ │ + this.writeNode("outerBoundaryIs", geometry.components[0], node); │ │ │ │ │ + for (var i = 1; i < geometry.components.length; ++i) { │ │ │ │ │ + this.writeNode("innerBoundaryIs", geometry.components[i], node) │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + outerBoundaryIs: function(ring) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:outerBoundaryIs"); │ │ │ │ │ + this.writeNode("LinearRing", ring, node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + innerBoundaryIs: function(ring) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:innerBoundaryIs"); │ │ │ │ │ + this.writeNode("LinearRing", ring, node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + LinearRing: function(ring) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:LinearRing"); │ │ │ │ │ + this.writeNode("coordinates", ring.components, node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + Box: function(bounds) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:Box"); │ │ │ │ │ + this.writeNode("coordinates", [{ │ │ │ │ │ + x: bounds.left, │ │ │ │ │ + y: bounds.bottom │ │ │ │ │ + }, { │ │ │ │ │ + x: bounds.right, │ │ │ │ │ + y: bounds.top │ │ │ │ │ + }], node); │ │ │ │ │ + if (this.srsName) { │ │ │ │ │ + node.setAttribute("srsName", this.srsName) │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.GML.Base.prototype.writers["gml"]), │ │ │ │ │ + feature: OpenLayers.Format.GML.Base.prototype.writers["feature"], │ │ │ │ │ + wfs: OpenLayers.Format.GML.Base.prototype.writers["wfs"] │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.GML.v2" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.Filter.v1_0_0 = OpenLayers.Class(OpenLayers.Format.GML.v2, OpenLayers.Format.Filter.v1, { │ │ │ │ │ + VERSION: "1.0.0", │ │ │ │ │ + schemaLocation: "http://www.opengis.net/ogc/filter/1.0.0/filter.xsd", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.GML.v2.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + ogc: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + PropertyIsEqualTo: function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.EQUAL_TO │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + obj.filters.push(filter) │ │ │ │ │ + }, │ │ │ │ │ + PropertyIsNotEqualTo: function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.NOT_EQUAL_TO │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + obj.filters.push(filter) │ │ │ │ │ + }, │ │ │ │ │ + PropertyIsLike: function(node, obj) { │ │ │ │ │ + var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ + type: OpenLayers.Filter.Comparison.LIKE │ │ │ │ │ + }); │ │ │ │ │ + this.readChildNodes(node, filter); │ │ │ │ │ + var wildCard = node.getAttribute("wildCard"); │ │ │ │ │ + var singleChar = node.getAttribute("singleChar"); │ │ │ │ │ + var esc = node.getAttribute("escape"); │ │ │ │ │ + filter.value2regex(wildCard, singleChar, esc); │ │ │ │ │ + obj.filters.push(filter) │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.Filter.v1.prototype.readers["ogc"]), │ │ │ │ │ + gml: OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ + feature: OpenLayers.Format.GML.v2.prototype.readers["feature"] │ │ │ │ │ + }, │ │ │ │ │ + writers: { │ │ │ │ │ + ogc: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + PropertyIsEqualTo: function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsEqualTo"); │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + this.writeOgcExpression(filter.value, node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + PropertyIsNotEqualTo: function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsNotEqualTo"); │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + this.writeOgcExpression(filter.value, node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + PropertyIsLike: function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:PropertyIsLike", { │ │ │ │ │ + attributes: { │ │ │ │ │ + wildCard: "*", │ │ │ │ │ + singleChar: ".", │ │ │ │ │ + escape: "!" │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + this.writeNode("Literal", filter.regex2value(), node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + BBOX: function(filter) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:BBOX"); │ │ │ │ │ + filter.property && this.writeNode("PropertyName", filter, node); │ │ │ │ │ + var box = this.writeNode("gml:Box", filter.value, node); │ │ │ │ │ + if (filter.projection) { │ │ │ │ │ + box.setAttribute("srsName", filter.projection) │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.Filter.v1.prototype.writers["ogc"]), │ │ │ │ │ + gml: OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ + feature: OpenLayers.Format.GML.v2.prototype.writers["feature"] │ │ │ │ │ + }, │ │ │ │ │ + writeSpatial: function(filter, name) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:" + name); │ │ │ │ │ + this.writeNode("PropertyName", filter, node); │ │ │ │ │ + if (filter.value instanceof OpenLayers.Filter.Function) { │ │ │ │ │ + this.writeNode("Function", filter.value, node) │ │ │ │ │ + } else { │ │ │ │ │ + var child; │ │ │ │ │ + if (filter.value instanceof OpenLayers.Geometry) { │ │ │ │ │ + child = this.writeNode("feature:_geometry", filter.value).firstChild │ │ │ │ │ + } else { │ │ │ │ │ + child = this.writeNode("gml:Box", filter.value) │ │ │ │ │ + } │ │ │ │ │ + if (filter.projection) { │ │ │ │ │ + child.setAttribute("srsName", filter.projection) │ │ │ │ │ + } │ │ │ │ │ + node.appendChild(child) │ │ │ │ │ } │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.Atom" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.Filter.v1_0_0" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.SOSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - defaultVersion: "1.0.0", │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SOSCapabilities" │ │ │ │ │ +OpenLayers.Format.WFST.v1_0_0 = OpenLayers.Class(OpenLayers.Format.Filter.v1_0_0, OpenLayers.Format.WFST.v1, { │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + srsNameInQuery: false, │ │ │ │ │ + schemaLocations: { │ │ │ │ │ + wfs: "http://schemas.opengis.net/wfs/1.0.0/WFS-transaction.xsd" │ │ │ │ │ + }, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.Filter.v1_0_0.prototype.initialize.apply(this, [options]); │ │ │ │ │ + OpenLayers.Format.WFST.v1.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ + readNode: function(node, obj, first) { │ │ │ │ │ + return OpenLayers.Format.GML.v2.prototype.readNode.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + wfs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + WFS_TransactionResponse: function(node, obj) { │ │ │ │ │ + obj.insertIds = []; │ │ │ │ │ + obj.success = false; │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + InsertResult: function(node, container) { │ │ │ │ │ + var obj = { │ │ │ │ │ + fids: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + container.insertIds = container.insertIds.concat(obj.fids) │ │ │ │ │ + }, │ │ │ │ │ + TransactionResult: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + Status: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + SUCCESS: function(node, obj) { │ │ │ │ │ + obj.success = true │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WFST.v1.prototype.readers["wfs"]), │ │ │ │ │ + gml: OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ + feature: OpenLayers.Format.GML.v2.prototype.readers["feature"], │ │ │ │ │ + ogc: OpenLayers.Format.Filter.v1_0_0.prototype.readers["ogc"] │ │ │ │ │ + }, │ │ │ │ │ + writers: { │ │ │ │ │ + wfs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Query: function(options) { │ │ │ │ │ + options = OpenLayers.Util.extend({ │ │ │ │ │ + featureNS: this.featureNS, │ │ │ │ │ + featurePrefix: this.featurePrefix, │ │ │ │ │ + featureType: this.featureType, │ │ │ │ │ + srsName: this.srsName, │ │ │ │ │ + srsNameInQuery: this.srsNameInQuery │ │ │ │ │ + }, options); │ │ │ │ │ + var prefix = options.featurePrefix; │ │ │ │ │ + var node = this.createElementNSPlus("wfs:Query", { │ │ │ │ │ + attributes: { │ │ │ │ │ + typeName: (prefix ? prefix + ":" : "") + options.featureType │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (options.srsNameInQuery && options.srsName) { │ │ │ │ │ + node.setAttribute("srsName", options.srsName) │ │ │ │ │ + } │ │ │ │ │ + if (options.featureNS) { │ │ │ │ │ + node.setAttribute("xmlns:" + prefix, options.featureNS) │ │ │ │ │ + } │ │ │ │ │ + if (options.propertyNames) { │ │ │ │ │ + for (var i = 0, len = options.propertyNames.length; i < len; i++) { │ │ │ │ │ + this.writeNode("ogc:PropertyName", { │ │ │ │ │ + property: options.propertyNames[i] │ │ │ │ │ + }, node) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (options.filter) { │ │ │ │ │ + this.setFilterProperty(options.filter); │ │ │ │ │ + this.writeNode("ogc:Filter", options.filter, node) │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WFST.v1.prototype.writers["wfs"]), │ │ │ │ │ + gml: OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ + feature: OpenLayers.Format.GML.v2.prototype.writers["feature"], │ │ │ │ │ + ogc: OpenLayers.Format.Filter.v1_0_0.prototype.writers["ogc"] │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFST.v1_0_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Protocol.WFS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.WFS.v1_0_0" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Format.SOSGetFeatureOfInterest = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ VERSION: "1.0.0", │ │ │ │ │ namespaces: { │ │ │ │ │ sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ gml: "http://www.opengis.net/gml", │ │ │ │ │ sa: "http://www.opengis.net/sampling/1.0", │ │ │ │ │ @@ -22109,241 +23135,1203 @@ │ │ │ │ │ }); │ │ │ │ │ return node │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.SOSGetFeatureOfInterest" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.WMSDescribeLayer = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - defaultVersion: "1.1.1", │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSDescribeLayer" │ │ │ │ │ +OpenLayers.Protocol.SOS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + fois: null, │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!options.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.SOSGetFeatureOfInterest(this.formatOptions) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.options && !this.options.format) { │ │ │ │ │ + this.format.destroy() │ │ │ │ │ + } │ │ │ │ │ + this.format = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ + }, │ │ │ │ │ + read: function(options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + var format = this.format; │ │ │ │ │ + var data = OpenLayers.Format.XML.prototype.write.apply(format, [format.writeNode("sos:GetFeatureOfInterest", { │ │ │ │ │ + fois: this.fois │ │ │ │ │ + })]); │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ + data: data │ │ │ │ │ + }); │ │ │ │ │ + return response │ │ │ │ │ + }, │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + response.features = this.parseFeatures(request); │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ + } │ │ │ │ │ + options.callback.call(options.scope, response) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + parseFeatures: function(request) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ + } │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ + return this.format.read(doc) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.SOS.v1_0_0" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.SOSGetObservation = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Format.CSWGetRecords = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, OpenLayers.Format.CSWGetRecords.DEFAULTS); │ │ │ │ │ + var cls = OpenLayers.Format.CSWGetRecords["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported CSWGetRecords version: " + options.version │ │ │ │ │ + } │ │ │ │ │ + return new cls(options) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Format.CSWGetRecords.DEFAULTS = { │ │ │ │ │ + version: "2.0.2" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Format.CSWGetRecords.v2_0_2 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ namespaces: { │ │ │ │ │ - ows: "http://www.opengis.net/ows", │ │ │ │ │ - gml: "http://www.opengis.net/gml", │ │ │ │ │ - sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ + csw: "http://www.opengis.net/cat/csw/2.0.2", │ │ │ │ │ + dc: "http://purl.org/dc/elements/1.1/", │ │ │ │ │ + dct: "http://purl.org/dc/terms/", │ │ │ │ │ + gmd: "http://www.isotc211.org/2005/gmd", │ │ │ │ │ + geonet: "http://www.fao.org/geonetwork", │ │ │ │ │ ogc: "http://www.opengis.net/ogc", │ │ │ │ │ - om: "http://www.opengis.net/om/1.0", │ │ │ │ │ - sa: "http://www.opengis.net/sampling/1.0", │ │ │ │ │ + ows: "http://www.opengis.net/ows", │ │ │ │ │ xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ }, │ │ │ │ │ + defaultPrefix: "csw", │ │ │ │ │ + version: "2.0.2", │ │ │ │ │ + schemaLocation: "http://www.opengis.net/cat/csw/2.0.2 http://schemas.opengis.net/csw/2.0.2/CSW-discovery.xsd", │ │ │ │ │ + requestId: null, │ │ │ │ │ + resultType: null, │ │ │ │ │ + outputFormat: null, │ │ │ │ │ + outputSchema: null, │ │ │ │ │ + startPosition: null, │ │ │ │ │ + maxRecords: null, │ │ │ │ │ + DistributedSearch: null, │ │ │ │ │ + ResponseHandler: null, │ │ │ │ │ + Query: null, │ │ │ │ │ regExes: { │ │ │ │ │ trimSpace: /^\s*|\s*$/g, │ │ │ │ │ removeSpace: /\s*/g, │ │ │ │ │ splitSpace: /\s+/, │ │ │ │ │ trimComma: /\s*,\s*/g │ │ │ │ │ }, │ │ │ │ │ - VERSION: "1.0.0", │ │ │ │ │ - schemaLocation: "http://www.opengis.net/sos/1.0 http://schemas.opengis.net/sos/1.0.0/sosGetObservation.xsd", │ │ │ │ │ - defaultPrefix: "sos", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ read: function(data) { │ │ │ │ │ if (typeof data == "string") { │ │ │ │ │ data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ } │ │ │ │ │ if (data && data.nodeType == 9) { │ │ │ │ │ data = data.documentElement │ │ │ │ │ } │ │ │ │ │ - var info = { │ │ │ │ │ - measurements: [], │ │ │ │ │ - observations: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readNode(data, info); │ │ │ │ │ - return info │ │ │ │ │ - }, │ │ │ │ │ - write: function(options) { │ │ │ │ │ - var node = this.writeNode("sos:GetObservation", options); │ │ │ │ │ - node.setAttribute("xmlns:om", this.namespaces.om); │ │ │ │ │ - node.setAttribute("xmlns:ogc", this.namespaces.ogc); │ │ │ │ │ - this.setAttributeNS(node, this.namespaces.xsi, "xsi:schemaLocation", this.schemaLocation); │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [node]) │ │ │ │ │ + var obj = {}; │ │ │ │ │ + this.readNode(data, obj); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ readers: { │ │ │ │ │ - om: { │ │ │ │ │ - ObservationCollection: function(node, obj) { │ │ │ │ │ - obj.id = this.getAttributeNS(node, this.namespaces.gml, "id"); │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ + csw: { │ │ │ │ │ + GetRecordsResponse: function(node, obj) { │ │ │ │ │ + obj.records = []; │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + var version = this.getAttributeNS(node, "", "version"); │ │ │ │ │ + if (version != "") { │ │ │ │ │ + obj.version = version │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - member: function(node, observationCollection) { │ │ │ │ │ - this.readChildNodes(node, observationCollection) │ │ │ │ │ + RequestId: function(node, obj) { │ │ │ │ │ + obj.RequestId = this.getChildValue(node) │ │ │ │ │ }, │ │ │ │ │ - Measurement: function(node, observationCollection) { │ │ │ │ │ - var measurement = {}; │ │ │ │ │ - observationCollection.measurements.push(measurement); │ │ │ │ │ - this.readChildNodes(node, measurement) │ │ │ │ │ + SearchStatus: function(node, obj) { │ │ │ │ │ + obj.SearchStatus = {}; │ │ │ │ │ + var timestamp = this.getAttributeNS(node, "", "timestamp"); │ │ │ │ │ + if (timestamp != "") { │ │ │ │ │ + obj.SearchStatus.timestamp = timestamp │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - Observation: function(node, observationCollection) { │ │ │ │ │ - var observation = {}; │ │ │ │ │ - observationCollection.observations.push(observation); │ │ │ │ │ - this.readChildNodes(node, observation) │ │ │ │ │ + SearchResults: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj); │ │ │ │ │ + var attrs = node.attributes; │ │ │ │ │ + var SearchResults = {}; │ │ │ │ │ + for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ + if (attrs[i].name == "numberOfRecordsMatched" || attrs[i].name == "numberOfRecordsReturned" || attrs[i].name == "nextRecord") { │ │ │ │ │ + SearchResults[attrs[i].name] = parseInt(attrs[i].nodeValue) │ │ │ │ │ + } else { │ │ │ │ │ + SearchResults[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + obj.SearchResults = SearchResults │ │ │ │ │ }, │ │ │ │ │ - samplingTime: function(node, measurement) { │ │ │ │ │ - var samplingTime = {}; │ │ │ │ │ - measurement.samplingTime = samplingTime; │ │ │ │ │ - this.readChildNodes(node, samplingTime) │ │ │ │ │ + SummaryRecord: function(node, obj) { │ │ │ │ │ + var record = { │ │ │ │ │ + type: "SummaryRecord" │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, record); │ │ │ │ │ + obj.records.push(record) │ │ │ │ │ }, │ │ │ │ │ - observedProperty: function(node, measurement) { │ │ │ │ │ - measurement.observedProperty = this.getAttributeNS(node, this.namespaces.xlink, "href"); │ │ │ │ │ - this.readChildNodes(node, measurement) │ │ │ │ │ + BriefRecord: function(node, obj) { │ │ │ │ │ + var record = { │ │ │ │ │ + type: "BriefRecord" │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, record); │ │ │ │ │ + obj.records.push(record) │ │ │ │ │ }, │ │ │ │ │ - procedure: function(node, measurement) { │ │ │ │ │ - measurement.procedure = this.getAttributeNS(node, this.namespaces.xlink, "href"); │ │ │ │ │ - this.readChildNodes(node, measurement) │ │ │ │ │ + DCMIRecord: function(node, obj) { │ │ │ │ │ + var record = { │ │ │ │ │ + type: "DCMIRecord" │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, record); │ │ │ │ │ + obj.records.push(record) │ │ │ │ │ }, │ │ │ │ │ - featureOfInterest: function(node, observation) { │ │ │ │ │ - var foi = { │ │ │ │ │ - features: [] │ │ │ │ │ + Record: function(node, obj) { │ │ │ │ │ + var record = { │ │ │ │ │ + type: "Record" │ │ │ │ │ }; │ │ │ │ │ - observation.fois = []; │ │ │ │ │ - observation.fois.push(foi); │ │ │ │ │ - this.readChildNodes(node, foi); │ │ │ │ │ - var features = []; │ │ │ │ │ - for (var i = 0, len = foi.features.length; i < len; i++) { │ │ │ │ │ - var feature = foi.features[i]; │ │ │ │ │ - features.push(new OpenLayers.Feature.Vector(feature.components[0], feature.attributes)) │ │ │ │ │ - } │ │ │ │ │ - foi.features = features │ │ │ │ │ + this.readChildNodes(node, record); │ │ │ │ │ + obj.records.push(record) │ │ │ │ │ }, │ │ │ │ │ - result: function(node, measurement) { │ │ │ │ │ - var result = {}; │ │ │ │ │ - measurement.result = result; │ │ │ │ │ - if (this.getChildValue(node) !== "") { │ │ │ │ │ - result.value = this.getChildValue(node); │ │ │ │ │ - result.uom = node.getAttribute("uom") │ │ │ │ │ - } else { │ │ │ │ │ - this.readChildNodes(node, result) │ │ │ │ │ + "*": function(node, obj) { │ │ │ │ │ + var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ + obj[name] = this.getChildValue(node) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + geonet: { │ │ │ │ │ + info: function(node, obj) { │ │ │ │ │ + var gninfo = {}; │ │ │ │ │ + this.readChildNodes(node, gninfo); │ │ │ │ │ + obj.gninfo = gninfo │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + dc: { │ │ │ │ │ + "*": function(node, obj) { │ │ │ │ │ + var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ + if (!OpenLayers.Util.isArray(obj[name])) { │ │ │ │ │ + obj[name] = [] │ │ │ │ │ + } │ │ │ │ │ + var dc_element = {}; │ │ │ │ │ + var attrs = node.attributes; │ │ │ │ │ + for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ + dc_element[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ + } │ │ │ │ │ + dc_element.value = this.getChildValue(node); │ │ │ │ │ + if (dc_element.value != "") { │ │ │ │ │ + obj[name].push(dc_element) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - sa: OpenLayers.Format.SOSGetFeatureOfInterest.prototype.readers.sa, │ │ │ │ │ - gml: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - TimeInstant: function(node, samplingTime) { │ │ │ │ │ - var timeInstant = {}; │ │ │ │ │ - samplingTime.timeInstant = timeInstant; │ │ │ │ │ - this.readChildNodes(node, timeInstant) │ │ │ │ │ - }, │ │ │ │ │ - timePosition: function(node, timeInstant) { │ │ │ │ │ - timeInstant.timePosition = this.getChildValue(node) │ │ │ │ │ + dct: { │ │ │ │ │ + "*": function(node, obj) { │ │ │ │ │ + var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ + if (!OpenLayers.Util.isArray(obj[name])) { │ │ │ │ │ + obj[name] = [] │ │ │ │ │ + } │ │ │ │ │ + obj[name].push(this.getChildValue(node)) │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.SOSGetFeatureOfInterest.prototype.readers.gml) │ │ │ │ │ + }, │ │ │ │ │ + ows: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + BoundingBox: function(node, obj) { │ │ │ │ │ + if (obj.bounds) { │ │ │ │ │ + obj.BoundingBox = [{ │ │ │ │ │ + crs: obj.projection, │ │ │ │ │ + value: [obj.bounds.left, obj.bounds.bottom, obj.bounds.right, obj.bounds.top] │ │ │ │ │ + }]; │ │ │ │ │ + delete obj.projection; │ │ │ │ │ + delete obj.bounds │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Format.OWSCommon.v1_0_0.prototype.readers["ows"]["BoundingBox"].apply(this, arguments) │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.OWSCommon.v1_0_0.prototype.readers["ows"]) │ │ │ │ │ + }, │ │ │ │ │ + write: function(options) { │ │ │ │ │ + var node = this.writeNode("csw:GetRecords", options); │ │ │ │ │ + node.setAttribute("xmlns:gmd", this.namespaces.gmd); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [node]) │ │ │ │ │ }, │ │ │ │ │ writers: { │ │ │ │ │ - sos: { │ │ │ │ │ - GetObservation: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("GetObservation", { │ │ │ │ │ + csw: { │ │ │ │ │ + GetRecords: function(options) { │ │ │ │ │ + if (!options) { │ │ │ │ │ + options = {} │ │ │ │ │ + } │ │ │ │ │ + var node = this.createElementNSPlus("csw:GetRecords", { │ │ │ │ │ attributes: { │ │ │ │ │ - version: this.VERSION, │ │ │ │ │ - service: "SOS" │ │ │ │ │ + service: "CSW", │ │ │ │ │ + version: this.version, │ │ │ │ │ + requestId: options.requestId || this.requestId, │ │ │ │ │ + resultType: options.resultType || this.resultType, │ │ │ │ │ + outputFormat: options.outputFormat || this.outputFormat, │ │ │ │ │ + outputSchema: options.outputSchema || this.outputSchema, │ │ │ │ │ + startPosition: options.startPosition || this.startPosition, │ │ │ │ │ + maxRecords: options.maxRecords || this.maxRecords │ │ │ │ │ } │ │ │ │ │ }); │ │ │ │ │ - this.writeNode("offering", options, node); │ │ │ │ │ - if (options.eventTime) { │ │ │ │ │ - this.writeNode("eventTime", options, node) │ │ │ │ │ + if (options.DistributedSearch || this.DistributedSearch) { │ │ │ │ │ + this.writeNode("csw:DistributedSearch", options.DistributedSearch || this.DistributedSearch, node) │ │ │ │ │ } │ │ │ │ │ - for (var procedure in options.procedures) { │ │ │ │ │ - this.writeNode("procedure", options.procedures[procedure], node) │ │ │ │ │ + var ResponseHandler = options.ResponseHandler || this.ResponseHandler; │ │ │ │ │ + if (OpenLayers.Util.isArray(ResponseHandler) && ResponseHandler.length > 0) { │ │ │ │ │ + for (var i = 0, len = ResponseHandler.length; i < len; i++) { │ │ │ │ │ + this.writeNode("csw:ResponseHandler", ResponseHandler[i], node) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - for (var observedProperty in options.observedProperties) { │ │ │ │ │ - this.writeNode("observedProperty", options.observedProperties[observedProperty], node) │ │ │ │ │ + this.writeNode("Query", options.Query || this.Query, node); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + DistributedSearch: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:DistributedSearch", { │ │ │ │ │ + attributes: { │ │ │ │ │ + hopCount: options.hopCount │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + ResponseHandler: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:ResponseHandler", { │ │ │ │ │ + value: options.value │ │ │ │ │ + }); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + Query: function(options) { │ │ │ │ │ + if (!options) { │ │ │ │ │ + options = {} │ │ │ │ │ } │ │ │ │ │ - if (options.foi) { │ │ │ │ │ - this.writeNode("featureOfInterest", options.foi, node) │ │ │ │ │ + var node = this.createElementNSPlus("csw:Query", { │ │ │ │ │ + attributes: { │ │ │ │ │ + typeNames: options.typeNames || "csw:Record" │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + var ElementName = options.ElementName; │ │ │ │ │ + if (OpenLayers.Util.isArray(ElementName) && ElementName.length > 0) { │ │ │ │ │ + for (var i = 0, len = ElementName.length; i < len; i++) { │ │ │ │ │ + this.writeNode("csw:ElementName", ElementName[i], node) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this.writeNode("csw:ElementSetName", options.ElementSetName || { │ │ │ │ │ + value: "summary" │ │ │ │ │ + }, node) │ │ │ │ │ } │ │ │ │ │ - this.writeNode("responseFormat", options, node); │ │ │ │ │ - if (options.resultModel) { │ │ │ │ │ - this.writeNode("resultModel", options, node) │ │ │ │ │ + if (options.Constraint) { │ │ │ │ │ + this.writeNode("csw:Constraint", options.Constraint, node) │ │ │ │ │ } │ │ │ │ │ - if (options.responseMode) { │ │ │ │ │ - this.writeNode("responseMode", options, node) │ │ │ │ │ + if (options.SortBy) { │ │ │ │ │ + this.writeNode("ogc:SortBy", options.SortBy, node) │ │ │ │ │ } │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - featureOfInterest: function(foi) { │ │ │ │ │ - var node = this.createElementNSPlus("featureOfInterest"); │ │ │ │ │ - this.writeNode("ObjectID", foi.objectId, node); │ │ │ │ │ + ElementName: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:ElementName", { │ │ │ │ │ + value: options.value │ │ │ │ │ + }); │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - ObjectID: function(options) { │ │ │ │ │ - return this.createElementNSPlus("ObjectID", { │ │ │ │ │ - value: options │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - responseFormat: function(options) { │ │ │ │ │ - return this.createElementNSPlus("responseFormat", { │ │ │ │ │ - value: options.responseFormat │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - procedure: function(procedure) { │ │ │ │ │ - return this.createElementNSPlus("procedure", { │ │ │ │ │ - value: procedure │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - offering: function(options) { │ │ │ │ │ - return this.createElementNSPlus("offering", { │ │ │ │ │ - value: options.offering │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - observedProperty: function(observedProperty) { │ │ │ │ │ - return this.createElementNSPlus("observedProperty", { │ │ │ │ │ - value: observedProperty │ │ │ │ │ - }) │ │ │ │ │ + ElementSetName: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:ElementSetName", { │ │ │ │ │ + attributes: { │ │ │ │ │ + typeNames: options.typeNames │ │ │ │ │ + }, │ │ │ │ │ + value: options.value │ │ │ │ │ + }); │ │ │ │ │ + return node │ │ │ │ │ }, │ │ │ │ │ - eventTime: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("eventTime"); │ │ │ │ │ - if (options.eventTime === "latest") { │ │ │ │ │ - this.writeNode("ogc:TM_Equals", options, node) │ │ │ │ │ + Constraint: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:Constraint", { │ │ │ │ │ + attributes: { │ │ │ │ │ + version: options.version │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (options.Filter) { │ │ │ │ │ + var format = new OpenLayers.Format.Filter({ │ │ │ │ │ + version: options.version │ │ │ │ │ + }); │ │ │ │ │ + node.appendChild(format.write(options.Filter)) │ │ │ │ │ + } else if (options.CqlText) { │ │ │ │ │ + var child = this.createElementNSPlus("CqlText", { │ │ │ │ │ + value: options.CqlText.value │ │ │ │ │ + }); │ │ │ │ │ + node.appendChild(child) │ │ │ │ │ } │ │ │ │ │ return node │ │ │ │ │ - }, │ │ │ │ │ - resultModel: function(options) { │ │ │ │ │ - return this.createElementNSPlus("resultModel", { │ │ │ │ │ - value: options.resultModel │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - responseMode: function(options) { │ │ │ │ │ - return this.createElementNSPlus("responseMode", { │ │ │ │ │ - value: options.responseMode │ │ │ │ │ - }) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - ogc: { │ │ │ │ │ - TM_Equals: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:TM_Equals"); │ │ │ │ │ - this.writeNode("ogc:PropertyName", { │ │ │ │ │ - property: "urn:ogc:data:time:iso8601" │ │ │ │ │ - }, node); │ │ │ │ │ - if (options.eventTime === "latest") { │ │ │ │ │ - this.writeNode("gml:TimeInstant", { │ │ │ │ │ - value: "latest" │ │ │ │ │ - }, node) │ │ │ │ │ + ogc: OpenLayers.Format.Filter.v1_1_0.prototype.writers["ogc"] │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.CSWGetRecords.v2_0_2" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Protocol.CSW.v2_0_2 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!options.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.CSWGetRecords.v2_0_2(OpenLayers.Util.extend({}, this.formatOptions)) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.options && !this.options.format) { │ │ │ │ │ + this.format.destroy() │ │ │ │ │ + } │ │ │ │ │ + this.format = null; │ │ │ │ │ + OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ + }, │ │ │ │ │ + read: function(options) { │ │ │ │ │ + options = OpenLayers.Util.extend({}, options); │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ + var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ + requestType: "read" │ │ │ │ │ + }); │ │ │ │ │ + var data = this.format.write(options.params || options); │ │ │ │ │ + response.priv = OpenLayers.Request.POST({ │ │ │ │ │ + url: options.url, │ │ │ │ │ + callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ + params: options.params, │ │ │ │ │ + headers: options.headers, │ │ │ │ │ + data: data │ │ │ │ │ + }); │ │ │ │ │ + return response │ │ │ │ │ + }, │ │ │ │ │ + handleRead: function(response, options) { │ │ │ │ │ + if (options.callback) { │ │ │ │ │ + var request = response.priv; │ │ │ │ │ + if (request.status >= 200 && request.status < 300) { │ │ │ │ │ + response.data = this.parseData(request); │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ + } else { │ │ │ │ │ + response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ + } │ │ │ │ │ + options.callback.call(options.scope, response) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + parseData: function(request) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ + } │ │ │ │ │ + if (!doc || doc.length <= 0) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ + return this.format.read(doc) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Protocol.CSW.v2_0_2" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WCSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + defaultVersion: "1.1.0", │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WCSCapabilities" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + defaultVersion: "1.1.1", │ │ │ │ │ + profile: null, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.ArcXML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + fontStyleKeys: ["antialiasing", "blockout", "font", "fontcolor", "fontsize", "fontstyle", "glowing", "interval", "outline", "printmode", "shadow", "transparency"], │ │ │ │ │ + request: null, │ │ │ │ │ + response: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.request = new OpenLayers.Format.ArcXML.Request; │ │ │ │ │ + this.response = new OpenLayers.Format.ArcXML.Response; │ │ │ │ │ + if (options) { │ │ │ │ │ + if (options.requesttype == "feature") { │ │ │ │ │ + this.request.get_image = null; │ │ │ │ │ + var qry = this.request.get_feature.query; │ │ │ │ │ + this.addCoordSys(qry.featurecoordsys, options.featureCoordSys); │ │ │ │ │ + this.addCoordSys(qry.filtercoordsys, options.filterCoordSys); │ │ │ │ │ + if (options.polygon) { │ │ │ │ │ + qry.isspatial = true; │ │ │ │ │ + qry.spatialfilter.polygon = options.polygon │ │ │ │ │ + } else if (options.envelope) { │ │ │ │ │ + qry.isspatial = true; │ │ │ │ │ + qry.spatialfilter.envelope = { │ │ │ │ │ + minx: 0, │ │ │ │ │ + miny: 0, │ │ │ │ │ + maxx: 0, │ │ │ │ │ + maxy: 0 │ │ │ │ │ + }; │ │ │ │ │ + this.parseEnvelope(qry.spatialfilter.envelope, options.envelope) │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - PropertyName: function(options) { │ │ │ │ │ - return this.createElementNSPlus("ogc:PropertyName", { │ │ │ │ │ - value: options.property │ │ │ │ │ - }) │ │ │ │ │ + } else if (options.requesttype == "image") { │ │ │ │ │ + this.request.get_feature = null; │ │ │ │ │ + var props = this.request.get_image.properties; │ │ │ │ │ + this.parseEnvelope(props.envelope, options.envelope); │ │ │ │ │ + this.addLayers(props.layerlist, options.layers); │ │ │ │ │ + this.addImageSize(props.imagesize, options.tileSize); │ │ │ │ │ + this.addCoordSys(props.featurecoordsys, options.featureCoordSys); │ │ │ │ │ + this.addCoordSys(props.filtercoordsys, options.filterCoordSys) │ │ │ │ │ + } else { │ │ │ │ │ + this.request = null │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - gml: { │ │ │ │ │ - TimeInstant: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:TimeInstant"); │ │ │ │ │ - this.writeNode("gml:timePosition", options, node); │ │ │ │ │ - return node │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ + parseEnvelope: function(env, arr) { │ │ │ │ │ + if (arr && arr.length == 4) { │ │ │ │ │ + env.minx = arr[0]; │ │ │ │ │ + env.miny = arr[1]; │ │ │ │ │ + env.maxx = arr[2]; │ │ │ │ │ + env.maxy = arr[3] │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addLayers: function(ll, lyrs) { │ │ │ │ │ + for (var lind = 0, len = lyrs.length; lind < len; lind++) { │ │ │ │ │ + ll.push(lyrs[lind]) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addImageSize: function(imsize, olsize) { │ │ │ │ │ + if (olsize !== null) { │ │ │ │ │ + imsize.width = olsize.w; │ │ │ │ │ + imsize.height = olsize.h; │ │ │ │ │ + imsize.printwidth = olsize.w; │ │ │ │ │ + imsize.printheight = olsize.h │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addCoordSys: function(featOrFilt, fsys) { │ │ │ │ │ + if (typeof fsys == "string") { │ │ │ │ │ + featOrFilt.id = parseInt(fsys); │ │ │ │ │ + featOrFilt.string = fsys │ │ │ │ │ + } else if (typeof fsys == "object" && fsys.proj !== null) { │ │ │ │ │ + featOrFilt.id = fsys.proj.srsProjNumber; │ │ │ │ │ + featOrFilt.string = fsys.proj.srsCode │ │ │ │ │ + } else { │ │ │ │ │ + featOrFilt = fsys │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + iserror: function(data) { │ │ │ │ │ + var ret = null; │ │ │ │ │ + if (!data) { │ │ │ │ │ + ret = this.response.error !== "" │ │ │ │ │ + } else { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]); │ │ │ │ │ + var errorNodes = data.documentElement.getElementsByTagName("ERROR"); │ │ │ │ │ + ret = errorNodes !== null && errorNodes.length > 0 │ │ │ │ │ + } │ │ │ │ │ + return ret │ │ │ │ │ + }, │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + var arcNode = null; │ │ │ │ │ + if (data && data.documentElement) { │ │ │ │ │ + if (data.documentElement.nodeName == "ARCXML") { │ │ │ │ │ + arcNode = data.documentElement │ │ │ │ │ + } else { │ │ │ │ │ + arcNode = data.documentElement.getElementsByTagName("ARCXML")[0] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!arcNode || arcNode.firstChild.nodeName === "parsererror") { │ │ │ │ │ + var error, source; │ │ │ │ │ + try { │ │ │ │ │ + error = data.firstChild.nodeValue; │ │ │ │ │ + source = data.firstChild.childNodes[1].firstChild.nodeValue │ │ │ │ │ + } catch (err) {} │ │ │ │ │ + throw { │ │ │ │ │ + message: "Error parsing the ArcXML request", │ │ │ │ │ + error: error, │ │ │ │ │ + source: source │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var response = this.parseResponse(arcNode); │ │ │ │ │ + return response │ │ │ │ │ + }, │ │ │ │ │ + write: function(request) { │ │ │ │ │ + if (!request) { │ │ │ │ │ + request = this.request │ │ │ │ │ + } │ │ │ │ │ + var root = this.createElementNS("", "ARCXML"); │ │ │ │ │ + root.setAttribute("version", "1.1"); │ │ │ │ │ + var reqElem = this.createElementNS("", "REQUEST"); │ │ │ │ │ + if (request.get_image != null) { │ │ │ │ │ + var getElem = this.createElementNS("", "GET_IMAGE"); │ │ │ │ │ + reqElem.appendChild(getElem); │ │ │ │ │ + var propElem = this.createElementNS("", "PROPERTIES"); │ │ │ │ │ + getElem.appendChild(propElem); │ │ │ │ │ + var props = request.get_image.properties; │ │ │ │ │ + if (props.featurecoordsys != null) { │ │ │ │ │ + var feat = this.createElementNS("", "FEATURECOORDSYS"); │ │ │ │ │ + propElem.appendChild(feat); │ │ │ │ │ + if (props.featurecoordsys.id === 0) { │ │ │ │ │ + feat.setAttribute("string", props.featurecoordsys["string"]) │ │ │ │ │ + } else { │ │ │ │ │ + feat.setAttribute("id", props.featurecoordsys.id) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (props.filtercoordsys != null) { │ │ │ │ │ + var filt = this.createElementNS("", "FILTERCOORDSYS"); │ │ │ │ │ + propElem.appendChild(filt); │ │ │ │ │ + if (props.filtercoordsys.id === 0) { │ │ │ │ │ + filt.setAttribute("string", props.filtercoordsys.string) │ │ │ │ │ + } else { │ │ │ │ │ + filt.setAttribute("id", props.filtercoordsys.id) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (props.envelope != null) { │ │ │ │ │ + var env = this.createElementNS("", "ENVELOPE"); │ │ │ │ │ + propElem.appendChild(env); │ │ │ │ │ + env.setAttribute("minx", props.envelope.minx); │ │ │ │ │ + env.setAttribute("miny", props.envelope.miny); │ │ │ │ │ + env.setAttribute("maxx", props.envelope.maxx); │ │ │ │ │ + env.setAttribute("maxy", props.envelope.maxy) │ │ │ │ │ + } │ │ │ │ │ + var imagesz = this.createElementNS("", "IMAGESIZE"); │ │ │ │ │ + propElem.appendChild(imagesz); │ │ │ │ │ + imagesz.setAttribute("height", props.imagesize.height); │ │ │ │ │ + imagesz.setAttribute("width", props.imagesize.width); │ │ │ │ │ + if (props.imagesize.height != props.imagesize.printheight || props.imagesize.width != props.imagesize.printwidth) { │ │ │ │ │ + imagesz.setAttribute("printheight", props.imagesize.printheight); │ │ │ │ │ + imagesz.setArrtibute("printwidth", props.imagesize.printwidth) │ │ │ │ │ + } │ │ │ │ │ + if (props.background != null) { │ │ │ │ │ + var backgrnd = this.createElementNS("", "BACKGROUND"); │ │ │ │ │ + propElem.appendChild(backgrnd); │ │ │ │ │ + backgrnd.setAttribute("color", props.background.color.r + "," + props.background.color.g + "," + props.background.color.b); │ │ │ │ │ + if (props.background.transcolor !== null) { │ │ │ │ │ + backgrnd.setAttribute("transcolor", props.background.transcolor.r + "," + props.background.transcolor.g + "," + props.background.transcolor.b) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (props.layerlist != null && props.layerlist.length > 0) { │ │ │ │ │ + var layerlst = this.createElementNS("", "LAYERLIST"); │ │ │ │ │ + propElem.appendChild(layerlst); │ │ │ │ │ + for (var ld = 0; ld < props.layerlist.length; ld++) { │ │ │ │ │ + var ldef = this.createElementNS("", "LAYERDEF"); │ │ │ │ │ + layerlst.appendChild(ldef); │ │ │ │ │ + ldef.setAttribute("id", props.layerlist[ld].id); │ │ │ │ │ + ldef.setAttribute("visible", props.layerlist[ld].visible); │ │ │ │ │ + if (typeof props.layerlist[ld].query == "object") { │ │ │ │ │ + var query = props.layerlist[ld].query; │ │ │ │ │ + if (query.where.length < 0) { │ │ │ │ │ + continue │ │ │ │ │ + } │ │ │ │ │ + var queryElem = null; │ │ │ │ │ + if (typeof query.spatialfilter == "boolean" && query.spatialfilter) { │ │ │ │ │ + queryElem = this.createElementNS("", "SPATIALQUERY") │ │ │ │ │ + } else { │ │ │ │ │ + queryElem = this.createElementNS("", "QUERY") │ │ │ │ │ + } │ │ │ │ │ + queryElem.setAttribute("where", query.where); │ │ │ │ │ + if (typeof query.accuracy == "number" && query.accuracy > 0) { │ │ │ │ │ + queryElem.setAttribute("accuracy", query.accuracy) │ │ │ │ │ + } │ │ │ │ │ + if (typeof query.featurelimit == "number" && query.featurelimit < 2e3) { │ │ │ │ │ + queryElem.setAttribute("featurelimit", query.featurelimit) │ │ │ │ │ + } │ │ │ │ │ + if (typeof query.subfields == "string" && query.subfields != "#ALL#") { │ │ │ │ │ + queryElem.setAttribute("subfields", query.subfields) │ │ │ │ │ + } │ │ │ │ │ + if (typeof query.joinexpression == "string" && query.joinexpression.length > 0) { │ │ │ │ │ + queryElem.setAttribute("joinexpression", query.joinexpression) │ │ │ │ │ + } │ │ │ │ │ + if (typeof query.jointables == "string" && query.jointables.length > 0) { │ │ │ │ │ + queryElem.setAttribute("jointables", query.jointables) │ │ │ │ │ + } │ │ │ │ │ + ldef.appendChild(queryElem) │ │ │ │ │ + } │ │ │ │ │ + if (typeof props.layerlist[ld].renderer == "object") { │ │ │ │ │ + this.addRenderer(ldef, props.layerlist[ld].renderer) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else if (request.get_feature != null) { │ │ │ │ │ + var getElem = this.createElementNS("", "GET_FEATURES"); │ │ │ │ │ + getElem.setAttribute("outputmode", "newxml"); │ │ │ │ │ + getElem.setAttribute("checkesc", "true"); │ │ │ │ │ + if (request.get_feature.geometry) { │ │ │ │ │ + getElem.setAttribute("geometry", request.get_feature.geometry) │ │ │ │ │ + } else { │ │ │ │ │ + getElem.setAttribute("geometry", "false") │ │ │ │ │ + } │ │ │ │ │ + if (request.get_feature.compact) { │ │ │ │ │ + getElem.setAttribute("compact", request.get_feature.compact) │ │ │ │ │ + } │ │ │ │ │ + if (request.get_feature.featurelimit == "number") { │ │ │ │ │ + getElem.setAttribute("featurelimit", request.get_feature.featurelimit) │ │ │ │ │ + } │ │ │ │ │ + getElem.setAttribute("globalenvelope", "true"); │ │ │ │ │ + reqElem.appendChild(getElem); │ │ │ │ │ + if (request.get_feature.layer != null && request.get_feature.layer.length > 0) { │ │ │ │ │ + var lyrElem = this.createElementNS("", "LAYER"); │ │ │ │ │ + lyrElem.setAttribute("id", request.get_feature.layer); │ │ │ │ │ + getElem.appendChild(lyrElem) │ │ │ │ │ + } │ │ │ │ │ + var fquery = request.get_feature.query; │ │ │ │ │ + if (fquery != null) { │ │ │ │ │ + var qElem = null; │ │ │ │ │ + if (fquery.isspatial) { │ │ │ │ │ + qElem = this.createElementNS("", "SPATIALQUERY") │ │ │ │ │ + } else { │ │ │ │ │ + qElem = this.createElementNS("", "QUERY") │ │ │ │ │ + } │ │ │ │ │ + getElem.appendChild(qElem); │ │ │ │ │ + if (typeof fquery.accuracy == "number") { │ │ │ │ │ + qElem.setAttribute("accuracy", fquery.accuracy) │ │ │ │ │ + } │ │ │ │ │ + if (fquery.featurecoordsys != null) { │ │ │ │ │ + var fcsElem1 = this.createElementNS("", "FEATURECOORDSYS"); │ │ │ │ │ + if (fquery.featurecoordsys.id == 0) { │ │ │ │ │ + fcsElem1.setAttribute("string", fquery.featurecoordsys.string) │ │ │ │ │ + } else { │ │ │ │ │ + fcsElem1.setAttribute("id", fquery.featurecoordsys.id) │ │ │ │ │ + } │ │ │ │ │ + qElem.appendChild(fcsElem1) │ │ │ │ │ + } │ │ │ │ │ + if (fquery.filtercoordsys != null) { │ │ │ │ │ + var fcsElem2 = this.createElementNS("", "FILTERCOORDSYS"); │ │ │ │ │ + if (fquery.filtercoordsys.id === 0) { │ │ │ │ │ + fcsElem2.setAttribute("string", fquery.filtercoordsys.string) │ │ │ │ │ + } else { │ │ │ │ │ + fcsElem2.setAttribute("id", fquery.filtercoordsys.id) │ │ │ │ │ + } │ │ │ │ │ + qElem.appendChild(fcsElem2) │ │ │ │ │ + } │ │ │ │ │ + if (fquery.buffer > 0) { │ │ │ │ │ + var bufElem = this.createElementNS("", "BUFFER"); │ │ │ │ │ + bufElem.setAttribute("distance", fquery.buffer); │ │ │ │ │ + qElem.appendChild(bufElem) │ │ │ │ │ + } │ │ │ │ │ + if (fquery.isspatial) { │ │ │ │ │ + var spfElem = this.createElementNS("", "SPATIALFILTER"); │ │ │ │ │ + spfElem.setAttribute("relation", fquery.spatialfilter.relation); │ │ │ │ │ + qElem.appendChild(spfElem); │ │ │ │ │ + if (fquery.spatialfilter.envelope) { │ │ │ │ │ + var envElem = this.createElementNS("", "ENVELOPE"); │ │ │ │ │ + envElem.setAttribute("minx", fquery.spatialfilter.envelope.minx); │ │ │ │ │ + envElem.setAttribute("miny", fquery.spatialfilter.envelope.miny); │ │ │ │ │ + envElem.setAttribute("maxx", fquery.spatialfilter.envelope.maxx); │ │ │ │ │ + envElem.setAttribute("maxy", fquery.spatialfilter.envelope.maxy); │ │ │ │ │ + spfElem.appendChild(envElem) │ │ │ │ │ + } else if (typeof fquery.spatialfilter.polygon == "object") { │ │ │ │ │ + spfElem.appendChild(this.writePolygonGeometry(fquery.spatialfilter.polygon)) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (fquery.where != null && fquery.where.length > 0) { │ │ │ │ │ + qElem.setAttribute("where", fquery.where) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + root.appendChild(reqElem); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [root]) │ │ │ │ │ + }, │ │ │ │ │ + addGroupRenderer: function(ldef, toprenderer) { │ │ │ │ │ + var topRelem = this.createElementNS("", "GROUPRENDERER"); │ │ │ │ │ + ldef.appendChild(topRelem); │ │ │ │ │ + for (var rind = 0; rind < toprenderer.length; rind++) { │ │ │ │ │ + var renderer = toprenderer[rind]; │ │ │ │ │ + this.addRenderer(topRelem, renderer) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addRenderer: function(topRelem, renderer) { │ │ │ │ │ + if (OpenLayers.Util.isArray(renderer)) { │ │ │ │ │ + this.addGroupRenderer(topRelem, renderer) │ │ │ │ │ + } else { │ │ │ │ │ + var renderElem = this.createElementNS("", renderer.type.toUpperCase() + "RENDERER"); │ │ │ │ │ + topRelem.appendChild(renderElem); │ │ │ │ │ + if (renderElem.tagName == "VALUEMAPRENDERER") { │ │ │ │ │ + this.addValueMapRenderer(renderElem, renderer) │ │ │ │ │ + } else if (renderElem.tagName == "VALUEMAPLABELRENDERER") { │ │ │ │ │ + this.addValueMapLabelRenderer(renderElem, renderer) │ │ │ │ │ + } else if (renderElem.tagName == "SIMPLELABELRENDERER") { │ │ │ │ │ + this.addSimpleLabelRenderer(renderElem, renderer) │ │ │ │ │ + } else if (renderElem.tagName == "SCALEDEPENDENTRENDERER") { │ │ │ │ │ + this.addScaleDependentRenderer(renderElem, renderer) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addScaleDependentRenderer: function(renderElem, renderer) { │ │ │ │ │ + if (typeof renderer.lower == "string" || typeof renderer.lower == "number") { │ │ │ │ │ + renderElem.setAttribute("lower", renderer.lower) │ │ │ │ │ + } │ │ │ │ │ + if (typeof renderer.upper == "string" || typeof renderer.upper == "number") { │ │ │ │ │ + renderElem.setAttribute("upper", renderer.upper) │ │ │ │ │ + } │ │ │ │ │ + this.addRenderer(renderElem, renderer.renderer) │ │ │ │ │ + }, │ │ │ │ │ + addValueMapLabelRenderer: function(renderElem, renderer) { │ │ │ │ │ + renderElem.setAttribute("lookupfield", renderer.lookupfield); │ │ │ │ │ + renderElem.setAttribute("labelfield", renderer.labelfield); │ │ │ │ │ + if (typeof renderer.exacts == "object") { │ │ │ │ │ + for (var ext = 0, extlen = renderer.exacts.length; ext < extlen; ext++) { │ │ │ │ │ + var exact = renderer.exacts[ext]; │ │ │ │ │ + var eelem = this.createElementNS("", "EXACT"); │ │ │ │ │ + if (typeof exact.value == "string") { │ │ │ │ │ + eelem.setAttribute("value", exact.value) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.label == "string") { │ │ │ │ │ + eelem.setAttribute("label", exact.label) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.method == "string") { │ │ │ │ │ + eelem.setAttribute("method", exact.method) │ │ │ │ │ + } │ │ │ │ │ + renderElem.appendChild(eelem); │ │ │ │ │ + if (typeof exact.symbol == "object") { │ │ │ │ │ + var selem = null; │ │ │ │ │ + if (exact.symbol.type == "text") { │ │ │ │ │ + selem = this.createElementNS("", "TEXTSYMBOL") │ │ │ │ │ + } │ │ │ │ │ + if (selem != null) { │ │ │ │ │ + var keys = this.fontStyleKeys; │ │ │ │ │ + for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ + var key = keys[i]; │ │ │ │ │ + if (exact.symbol[key]) { │ │ │ │ │ + selem.setAttribute(key, exact.symbol[key]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + eelem.appendChild(selem) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addValueMapRenderer: function(renderElem, renderer) { │ │ │ │ │ + renderElem.setAttribute("lookupfield", renderer.lookupfield); │ │ │ │ │ + if (typeof renderer.ranges == "object") { │ │ │ │ │ + for (var rng = 0, rnglen = renderer.ranges.length; rng < rnglen; rng++) { │ │ │ │ │ + var range = renderer.ranges[rng]; │ │ │ │ │ + var relem = this.createElementNS("", "RANGE"); │ │ │ │ │ + relem.setAttribute("lower", range.lower); │ │ │ │ │ + relem.setAttribute("upper", range.upper); │ │ │ │ │ + renderElem.appendChild(relem); │ │ │ │ │ + if (typeof range.symbol == "object") { │ │ │ │ │ + var selem = null; │ │ │ │ │ + if (range.symbol.type == "simplepolygon") { │ │ │ │ │ + selem = this.createElementNS("", "SIMPLEPOLYGONSYMBOL") │ │ │ │ │ + } │ │ │ │ │ + if (selem != null) { │ │ │ │ │ + if (typeof range.symbol.boundarycolor == "string") { │ │ │ │ │ + selem.setAttribute("boundarycolor", range.symbol.boundarycolor) │ │ │ │ │ + } │ │ │ │ │ + if (typeof range.symbol.fillcolor == "string") { │ │ │ │ │ + selem.setAttribute("fillcolor", range.symbol.fillcolor) │ │ │ │ │ + } │ │ │ │ │ + if (typeof range.symbol.filltransparency == "number") { │ │ │ │ │ + selem.setAttribute("filltransparency", range.symbol.filltransparency) │ │ │ │ │ + } │ │ │ │ │ + relem.appendChild(selem) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else if (typeof renderer.exacts == "object") { │ │ │ │ │ + for (var ext = 0, extlen = renderer.exacts.length; ext < extlen; ext++) { │ │ │ │ │ + var exact = renderer.exacts[ext]; │ │ │ │ │ + var eelem = this.createElementNS("", "EXACT"); │ │ │ │ │ + if (typeof exact.value == "string") { │ │ │ │ │ + eelem.setAttribute("value", exact.value) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.label == "string") { │ │ │ │ │ + eelem.setAttribute("label", exact.label) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.method == "string") { │ │ │ │ │ + eelem.setAttribute("method", exact.method) │ │ │ │ │ + } │ │ │ │ │ + renderElem.appendChild(eelem); │ │ │ │ │ + if (typeof exact.symbol == "object") { │ │ │ │ │ + var selem = null; │ │ │ │ │ + if (exact.symbol.type == "simplemarker") { │ │ │ │ │ + selem = this.createElementNS("", "SIMPLEMARKERSYMBOL") │ │ │ │ │ + } │ │ │ │ │ + if (selem != null) { │ │ │ │ │ + if (typeof exact.symbol.antialiasing == "string") { │ │ │ │ │ + selem.setAttribute("antialiasing", exact.symbol.antialiasing) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.color == "string") { │ │ │ │ │ + selem.setAttribute("color", exact.symbol.color) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.outline == "string") { │ │ │ │ │ + selem.setAttribute("outline", exact.symbol.outline) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.overlap == "string") { │ │ │ │ │ + selem.setAttribute("overlap", exact.symbol.overlap) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.shadow == "string") { │ │ │ │ │ + selem.setAttribute("shadow", exact.symbol.shadow) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.transparency == "number") { │ │ │ │ │ + selem.setAttribute("transparency", exact.symbol.transparency) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.usecentroid == "string") { │ │ │ │ │ + selem.setAttribute("usecentroid", exact.symbol.usecentroid) │ │ │ │ │ + } │ │ │ │ │ + if (typeof exact.symbol.width == "number") { │ │ │ │ │ + selem.setAttribute("width", exact.symbol.width) │ │ │ │ │ + } │ │ │ │ │ + eelem.appendChild(selem) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + addSimpleLabelRenderer: function(renderElem, renderer) { │ │ │ │ │ + renderElem.setAttribute("field", renderer.field); │ │ │ │ │ + var keys = ["featureweight", "howmanylabels", "labelbufferratio", "labelpriorities", "labelweight", "linelabelposition", "rotationalangles"]; │ │ │ │ │ + for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ + var key = keys[i]; │ │ │ │ │ + if (renderer[key]) { │ │ │ │ │ + renderElem.setAttribute(key, renderer[key]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (renderer.symbol.type == "text") { │ │ │ │ │ + var symbol = renderer.symbol; │ │ │ │ │ + var selem = this.createElementNS("", "TEXTSYMBOL"); │ │ │ │ │ + renderElem.appendChild(selem); │ │ │ │ │ + var keys = this.fontStyleKeys; │ │ │ │ │ + for (var i = 0, len = keys.length; i < len; i++) { │ │ │ │ │ + var key = keys[i]; │ │ │ │ │ + if (symbol[key]) { │ │ │ │ │ + selem.setAttribute(key, renderer[key]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + writePolygonGeometry: function(polygon) { │ │ │ │ │ + if (!(polygon instanceof OpenLayers.Geometry.Polygon)) { │ │ │ │ │ + throw { │ │ │ │ │ + message: "Cannot write polygon geometry to ArcXML with an " + polygon.CLASS_NAME + " object.", │ │ │ │ │ + geometry: polygon │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var polyElem = this.createElementNS("", "POLYGON"); │ │ │ │ │ + for (var ln = 0, lnlen = polygon.components.length; ln < lnlen; ln++) { │ │ │ │ │ + var ring = polygon.components[ln]; │ │ │ │ │ + var ringElem = this.createElementNS("", "RING"); │ │ │ │ │ + for (var rn = 0, rnlen = ring.components.length; rn < rnlen; rn++) { │ │ │ │ │ + var point = ring.components[rn]; │ │ │ │ │ + var pointElem = this.createElementNS("", "POINT"); │ │ │ │ │ + pointElem.setAttribute("x", point.x); │ │ │ │ │ + pointElem.setAttribute("y", point.y); │ │ │ │ │ + ringElem.appendChild(pointElem) │ │ │ │ │ + } │ │ │ │ │ + polyElem.appendChild(ringElem) │ │ │ │ │ + } │ │ │ │ │ + return polyElem │ │ │ │ │ + }, │ │ │ │ │ + parseResponse: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + var newData = new OpenLayers.Format.XML; │ │ │ │ │ + data = newData.read(data) │ │ │ │ │ + } │ │ │ │ │ + var response = new OpenLayers.Format.ArcXML.Response; │ │ │ │ │ + var errorNode = data.getElementsByTagName("ERROR"); │ │ │ │ │ + if (errorNode != null && errorNode.length > 0) { │ │ │ │ │ + response.error = this.getChildValue(errorNode, "Unknown error.") │ │ │ │ │ + } else { │ │ │ │ │ + var responseNode = data.getElementsByTagName("RESPONSE"); │ │ │ │ │ + if (responseNode == null || responseNode.length == 0) { │ │ │ │ │ + response.error = "No RESPONSE tag found in ArcXML response."; │ │ │ │ │ + return response │ │ │ │ │ + } │ │ │ │ │ + var rtype = responseNode[0].firstChild.nodeName; │ │ │ │ │ + if (rtype == "#text") { │ │ │ │ │ + rtype = responseNode[0].firstChild.nextSibling.nodeName │ │ │ │ │ + } │ │ │ │ │ + if (rtype == "IMAGE") { │ │ │ │ │ + var envelopeNode = data.getElementsByTagName("ENVELOPE"); │ │ │ │ │ + var outputNode = data.getElementsByTagName("OUTPUT"); │ │ │ │ │ + if (envelopeNode == null || envelopeNode.length == 0) { │ │ │ │ │ + response.error = "No ENVELOPE tag found in ArcXML response." │ │ │ │ │ + } else if (outputNode == null || outputNode.length == 0) { │ │ │ │ │ + response.error = "No OUTPUT tag found in ArcXML response." │ │ │ │ │ + } else { │ │ │ │ │ + var envAttr = this.parseAttributes(envelopeNode[0]); │ │ │ │ │ + var outputAttr = this.parseAttributes(outputNode[0]); │ │ │ │ │ + if (typeof outputAttr.type == "string") { │ │ │ │ │ + response.image = { │ │ │ │ │ + envelope: envAttr, │ │ │ │ │ + output: { │ │ │ │ │ + type: outputAttr.type, │ │ │ │ │ + data: this.getChildValue(outputNode[0]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + response.image = { │ │ │ │ │ + envelope: envAttr, │ │ │ │ │ + output: outputAttr │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else if (rtype == "FEATURES") { │ │ │ │ │ + var features = responseNode[0].getElementsByTagName("FEATURES"); │ │ │ │ │ + var featureCount = features[0].getElementsByTagName("FEATURECOUNT"); │ │ │ │ │ + response.features.featurecount = featureCount[0].getAttribute("count"); │ │ │ │ │ + if (response.features.featurecount > 0) { │ │ │ │ │ + var envelope = features[0].getElementsByTagName("ENVELOPE"); │ │ │ │ │ + response.features.envelope = this.parseAttributes(envelope[0], typeof 0); │ │ │ │ │ + var featureList = features[0].getElementsByTagName("FEATURE"); │ │ │ │ │ + for (var fn = 0; fn < featureList.length; fn++) { │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector; │ │ │ │ │ + var fields = featureList[fn].getElementsByTagName("FIELD"); │ │ │ │ │ + for (var fdn = 0; fdn < fields.length; fdn++) { │ │ │ │ │ + var fieldName = fields[fdn].getAttribute("name"); │ │ │ │ │ + var fieldValue = fields[fdn].getAttribute("value"); │ │ │ │ │ + feature.attributes[fieldName] = fieldValue │ │ │ │ │ + } │ │ │ │ │ + var geom = featureList[fn].getElementsByTagName("POLYGON"); │ │ │ │ │ + if (geom.length > 0) { │ │ │ │ │ + var ring = geom[0].getElementsByTagName("RING"); │ │ │ │ │ + var polys = []; │ │ │ │ │ + for (var rn = 0; rn < ring.length; rn++) { │ │ │ │ │ + var linearRings = []; │ │ │ │ │ + linearRings.push(this.parsePointGeometry(ring[rn])); │ │ │ │ │ + var holes = ring[rn].getElementsByTagName("HOLE"); │ │ │ │ │ + for (var hn = 0; hn < holes.length; hn++) { │ │ │ │ │ + linearRings.push(this.parsePointGeometry(holes[hn])) │ │ │ │ │ + } │ │ │ │ │ + holes = null; │ │ │ │ │ + polys.push(new OpenLayers.Geometry.Polygon(linearRings)); │ │ │ │ │ + linearRings = null │ │ │ │ │ + } │ │ │ │ │ + ring = null; │ │ │ │ │ + if (polys.length == 1) { │ │ │ │ │ + feature.geometry = polys[0] │ │ │ │ │ + } else { │ │ │ │ │ + feature.geometry = new OpenLayers.Geometry.MultiPolygon(polys) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + response.features.feature.push(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + response.error = "Unidentified response type." │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return response │ │ │ │ │ + }, │ │ │ │ │ + parseAttributes: function(node, type) { │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + for (var attr = 0; attr < node.attributes.length; attr++) { │ │ │ │ │ + if (type == "number") { │ │ │ │ │ + attributes[node.attributes[attr].nodeName] = parseFloat(node.attributes[attr].nodeValue) │ │ │ │ │ + } else { │ │ │ │ │ + attributes[node.attributes[attr].nodeName] = node.attributes[attr].nodeValue │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return attributes │ │ │ │ │ + }, │ │ │ │ │ + parsePointGeometry: function(node) { │ │ │ │ │ + var ringPoints = []; │ │ │ │ │ + var coords = node.getElementsByTagName("COORDS"); │ │ │ │ │ + if (coords.length > 0) { │ │ │ │ │ + var coordArr = this.getChildValue(coords[0]); │ │ │ │ │ + coordArr = coordArr.split(/;/); │ │ │ │ │ + for (var cn = 0; cn < coordArr.length; cn++) { │ │ │ │ │ + var coordItems = coordArr[cn].split(/ /); │ │ │ │ │ + ringPoints.push(new OpenLayers.Geometry.Point(coordItems[0], coordItems[1])) │ │ │ │ │ + } │ │ │ │ │ + coords = null │ │ │ │ │ + } else { │ │ │ │ │ + var point = node.getElementsByTagName("POINT"); │ │ │ │ │ + if (point.length > 0) { │ │ │ │ │ + for (var pn = 0; pn < point.length; pn++) { │ │ │ │ │ + ringPoints.push(new OpenLayers.Geometry.Point(parseFloat(point[pn].getAttribute("x")), parseFloat(point[pn].getAttribute("y")))) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + point = null │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.LinearRing(ringPoints) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.ArcXML" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.ArcXML.Request = OpenLayers.Class({ │ │ │ │ │ + initialize: function(params) { │ │ │ │ │ + var defaults = { │ │ │ │ │ + get_image: { │ │ │ │ │ + properties: { │ │ │ │ │ + background: null, │ │ │ │ │ + draw: true, │ │ │ │ │ + envelope: { │ │ │ │ │ + minx: 0, │ │ │ │ │ + miny: 0, │ │ │ │ │ + maxx: 0, │ │ │ │ │ + maxy: 0 │ │ │ │ │ + }, │ │ │ │ │ + featurecoordsys: { │ │ │ │ │ + id: 0, │ │ │ │ │ + string: "", │ │ │ │ │ + datumtransformid: 0, │ │ │ │ │ + datumtransformstring: "" │ │ │ │ │ + }, │ │ │ │ │ + filtercoordsys: { │ │ │ │ │ + id: 0, │ │ │ │ │ + string: "", │ │ │ │ │ + datumtransformid: 0, │ │ │ │ │ + datumtransformstring: "" │ │ │ │ │ + }, │ │ │ │ │ + imagesize: { │ │ │ │ │ + height: 0, │ │ │ │ │ + width: 0, │ │ │ │ │ + dpi: 96, │ │ │ │ │ + printheight: 0, │ │ │ │ │ + printwidth: 0, │ │ │ │ │ + scalesymbols: false │ │ │ │ │ + }, │ │ │ │ │ + layerlist: [], │ │ │ │ │ + output: { │ │ │ │ │ + baseurl: "", │ │ │ │ │ + legendbaseurl: "", │ │ │ │ │ + legendname: "", │ │ │ │ │ + legendpath: "", │ │ │ │ │ + legendurl: "", │ │ │ │ │ + name: "", │ │ │ │ │ + path: "", │ │ │ │ │ + type: "jpg", │ │ │ │ │ + url: "" │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - timePosition: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:timePosition", { │ │ │ │ │ - value: options.value │ │ │ │ │ - }); │ │ │ │ │ - return node │ │ │ │ │ + get_feature: { │ │ │ │ │ + layer: "", │ │ │ │ │ + query: { │ │ │ │ │ + isspatial: false, │ │ │ │ │ + featurecoordsys: { │ │ │ │ │ + id: 0, │ │ │ │ │ + string: "", │ │ │ │ │ + datumtransformid: 0, │ │ │ │ │ + datumtransformstring: "" │ │ │ │ │ + }, │ │ │ │ │ + filtercoordsys: { │ │ │ │ │ + id: 0, │ │ │ │ │ + string: "", │ │ │ │ │ + datumtransformid: 0, │ │ │ │ │ + datumtransformstring: "" │ │ │ │ │ + }, │ │ │ │ │ + buffer: 0, │ │ │ │ │ + where: "", │ │ │ │ │ + spatialfilter: { │ │ │ │ │ + relation: "envelope_intersection", │ │ │ │ │ + envelope: null │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + environment: { │ │ │ │ │ + separators: { │ │ │ │ │ + cs: " ", │ │ │ │ │ + ts: ";" │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + layer: [], │ │ │ │ │ + workspaces: [] │ │ │ │ │ + }; │ │ │ │ │ + return OpenLayers.Util.extend(this, defaults) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.ArcXML.Request" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.ArcXML.Response = OpenLayers.Class({ │ │ │ │ │ + initialize: function(params) { │ │ │ │ │ + var defaults = { │ │ │ │ │ + image: { │ │ │ │ │ + envelope: null, │ │ │ │ │ + output: "" │ │ │ │ │ + }, │ │ │ │ │ + features: { │ │ │ │ │ + featurecount: 0, │ │ │ │ │ + envelope: null, │ │ │ │ │ + feature: [] │ │ │ │ │ + }, │ │ │ │ │ + error: "" │ │ │ │ │ + }; │ │ │ │ │ + return OpenLayers.Util.extend(this, defaults) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.ArcXML.Response" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.Text = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ + defaultStyle: null, │ │ │ │ │ + extractStyles: true, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + if (options.extractStyles !== false) { │ │ │ │ │ + options.defaultStyle = { │ │ │ │ │ + externalGraphic: OpenLayers.Util.getImageLocation("marker.png"), │ │ │ │ │ + graphicWidth: 21, │ │ │ │ │ + graphicHeight: 25, │ │ │ │ │ + graphicXOffset: -10.5, │ │ │ │ │ + graphicYOffset: -12.5 │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Format.prototype.initialize.apply(this, [options]) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SOSGetObservation" │ │ │ │ │ + read: function(text) { │ │ │ │ │ + var lines = text.split("\n"); │ │ │ │ │ + var columns; │ │ │ │ │ + var features = []; │ │ │ │ │ + for (var lcv = 0; lcv < lines.length - 1; lcv++) { │ │ │ │ │ + var currLine = lines[lcv].replace(/^\s*/, "").replace(/\s*$/, ""); │ │ │ │ │ + if (currLine.charAt(0) != "#") { │ │ │ │ │ + if (!columns) { │ │ │ │ │ + columns = currLine.split("\t") │ │ │ │ │ + } else { │ │ │ │ │ + var vals = currLine.split("\t"); │ │ │ │ │ + var geometry = new OpenLayers.Geometry.Point(0, 0); │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + var style = this.defaultStyle ? OpenLayers.Util.applyDefaults({}, this.defaultStyle) : null; │ │ │ │ │ + var icon, iconSize, iconOffset, overflow; │ │ │ │ │ + var set = false; │ │ │ │ │ + for (var valIndex = 0; valIndex < vals.length; valIndex++) { │ │ │ │ │ + if (vals[valIndex]) { │ │ │ │ │ + if (columns[valIndex] == "point") { │ │ │ │ │ + var coords = vals[valIndex].split(","); │ │ │ │ │ + geometry.y = parseFloat(coords[0]); │ │ │ │ │ + geometry.x = parseFloat(coords[1]); │ │ │ │ │ + set = true │ │ │ │ │ + } else if (columns[valIndex] == "lat") { │ │ │ │ │ + geometry.y = parseFloat(vals[valIndex]); │ │ │ │ │ + set = true │ │ │ │ │ + } else if (columns[valIndex] == "lon") { │ │ │ │ │ + geometry.x = parseFloat(vals[valIndex]); │ │ │ │ │ + set = true │ │ │ │ │ + } else if (columns[valIndex] == "title") attributes["title"] = vals[valIndex]; │ │ │ │ │ + else if (columns[valIndex] == "image" || columns[valIndex] == "icon" && style) { │ │ │ │ │ + style["externalGraphic"] = vals[valIndex] │ │ │ │ │ + } else if (columns[valIndex] == "iconSize" && style) { │ │ │ │ │ + var size = vals[valIndex].split(","); │ │ │ │ │ + style["graphicWidth"] = parseFloat(size[0]); │ │ │ │ │ + style["graphicHeight"] = parseFloat(size[1]) │ │ │ │ │ + } else if (columns[valIndex] == "iconOffset" && style) { │ │ │ │ │ + var offset = vals[valIndex].split(","); │ │ │ │ │ + style["graphicXOffset"] = parseFloat(offset[0]); │ │ │ │ │ + style["graphicYOffset"] = parseFloat(offset[1]) │ │ │ │ │ + } else if (columns[valIndex] == "description") { │ │ │ │ │ + attributes["description"] = vals[valIndex] │ │ │ │ │ + } else if (columns[valIndex] == "overflow") { │ │ │ │ │ + attributes["overflow"] = vals[valIndex] │ │ │ │ │ + } else { │ │ │ │ │ + attributes[columns[valIndex]] = vals[valIndex] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (set) { │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ + } │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry, attributes, style); │ │ │ │ │ + features.push(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return features │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.Text" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Format.Context = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ layerOptions: null, │ │ │ │ │ layerParams: null, │ │ │ │ │ read: function(data, options) { │ │ │ │ │ var context = OpenLayers.Format.XML.VersionedOGC.prototype.read.apply(this, arguments); │ │ │ │ │ var map; │ │ │ │ │ @@ -22505,14 +24493,215 @@ │ │ │ │ │ WFS: "urn:ogc:serviceType:WFS", │ │ │ │ │ WCS: "urn:ogc:serviceType:WCS", │ │ │ │ │ GML: "urn:ogc:serviceType:GML", │ │ │ │ │ SLD: "urn:ogc:serviceType:SLD", │ │ │ │ │ FES: "urn:ogc:serviceType:FES", │ │ │ │ │ KML: "urn:ogc:serviceType:KML" │ │ │ │ │ }; │ │ │ │ │ +OpenLayers.Format.OWSContext = OpenLayers.Class(OpenLayers.Format.Context, { │ │ │ │ │ + defaultVersion: "0.3.1", │ │ │ │ │ + getVersion: function(root, options) { │ │ │ │ │ + var version = OpenLayers.Format.XML.VersionedOGC.prototype.getVersion.apply(this, arguments); │ │ │ │ │ + if (version === "0.3.0") { │ │ │ │ │ + version = this.defaultVersion │ │ │ │ │ + } │ │ │ │ │ + return version │ │ │ │ │ + }, │ │ │ │ │ + toContext: function(obj) { │ │ │ │ │ + var context = {}; │ │ │ │ │ + if (obj.CLASS_NAME == "OpenLayers.Map") { │ │ │ │ │ + context.bounds = obj.getExtent(); │ │ │ │ │ + context.maxExtent = obj.maxExtent; │ │ │ │ │ + context.projection = obj.projection; │ │ │ │ │ + context.size = obj.getSize(); │ │ │ │ │ + context.layers = obj.layers │ │ │ │ │ + } │ │ │ │ │ + return context │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.OWSContext" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.GPX = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + defaultDesc: "No description available", │ │ │ │ │ + extractWaypoints: true, │ │ │ │ │ + extractTracks: true, │ │ │ │ │ + extractRoutes: true, │ │ │ │ │ + extractAttributes: true, │ │ │ │ │ + namespaces: { │ │ │ │ │ + gpx: "http://www.topografix.com/GPX/1/1", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ + schemaLocation: "http://www.topografix.com/GPX/1/1 http://www.topografix.com/GPX/1/1/gpx.xsd", │ │ │ │ │ + creator: "OpenLayers", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ + read: function(doc) { │ │ │ │ │ + if (typeof doc == "string") { │ │ │ │ │ + doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]) │ │ │ │ │ + } │ │ │ │ │ + var features = []; │ │ │ │ │ + if (this.extractTracks) { │ │ │ │ │ + var tracks = doc.getElementsByTagName("trk"); │ │ │ │ │ + for (var i = 0, len = tracks.length; i < len; i++) { │ │ │ │ │ + var attrs = {}; │ │ │ │ │ + if (this.extractAttributes) { │ │ │ │ │ + attrs = this.parseAttributes(tracks[i]) │ │ │ │ │ + } │ │ │ │ │ + var segs = this.getElementsByTagNameNS(tracks[i], tracks[i].namespaceURI, "trkseg"); │ │ │ │ │ + for (var j = 0, seglen = segs.length; j < seglen; j++) { │ │ │ │ │ + var track = this.extractSegment(segs[j], "trkpt"); │ │ │ │ │ + features.push(new OpenLayers.Feature.Vector(track, attrs)) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.extractRoutes) { │ │ │ │ │ + var routes = doc.getElementsByTagName("rte"); │ │ │ │ │ + for (var k = 0, klen = routes.length; k < klen; k++) { │ │ │ │ │ + var attrs = {}; │ │ │ │ │ + if (this.extractAttributes) { │ │ │ │ │ + attrs = this.parseAttributes(routes[k]) │ │ │ │ │ + } │ │ │ │ │ + var route = this.extractSegment(routes[k], "rtept"); │ │ │ │ │ + features.push(new OpenLayers.Feature.Vector(route, attrs)) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.extractWaypoints) { │ │ │ │ │ + var waypoints = doc.getElementsByTagName("wpt"); │ │ │ │ │ + for (var l = 0, len = waypoints.length; l < len; l++) { │ │ │ │ │ + var attrs = {}; │ │ │ │ │ + if (this.extractAttributes) { │ │ │ │ │ + attrs = this.parseAttributes(waypoints[l]) │ │ │ │ │ + } │ │ │ │ │ + var wpt = new OpenLayers.Geometry.Point(waypoints[l].getAttribute("lon"), waypoints[l].getAttribute("lat")); │ │ │ │ │ + features.push(new OpenLayers.Feature.Vector(wpt, attrs)) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + for (var g = 0, featLength = features.length; g < featLength; g++) { │ │ │ │ │ + features[g].geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return features │ │ │ │ │ + }, │ │ │ │ │ + extractSegment: function(segment, segmentType) { │ │ │ │ │ + var points = this.getElementsByTagNameNS(segment, segment.namespaceURI, segmentType); │ │ │ │ │ + var point_features = []; │ │ │ │ │ + for (var i = 0, len = points.length; i < len; i++) { │ │ │ │ │ + point_features.push(new OpenLayers.Geometry.Point(points[i].getAttribute("lon"), points[i].getAttribute("lat"))) │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Geometry.LineString(point_features) │ │ │ │ │ + }, │ │ │ │ │ + parseAttributes: function(node) { │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + var attrNode = node.firstChild, │ │ │ │ │ + value, name; │ │ │ │ │ + while (attrNode) { │ │ │ │ │ + if (attrNode.nodeType == 1 && attrNode.firstChild) { │ │ │ │ │ + value = attrNode.firstChild; │ │ │ │ │ + if (value.nodeType == 3 || value.nodeType == 4) { │ │ │ │ │ + name = attrNode.prefix ? attrNode.nodeName.split(":")[1] : attrNode.nodeName; │ │ │ │ │ + if (name != "trkseg" && name != "rtept") { │ │ │ │ │ + attributes[name] = value.nodeValue │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + attrNode = attrNode.nextSibling │ │ │ │ │ + } │ │ │ │ │ + return attributes │ │ │ │ │ + }, │ │ │ │ │ + write: function(features, metadata) { │ │ │ │ │ + features = OpenLayers.Util.isArray(features) ? features : [features]; │ │ │ │ │ + var gpx = this.createElementNS(this.namespaces.gpx, "gpx"); │ │ │ │ │ + gpx.setAttribute("version", "1.1"); │ │ │ │ │ + gpx.setAttribute("creator", this.creator); │ │ │ │ │ + this.setAttributes(gpx, { │ │ │ │ │ + "xsi:schemaLocation": this.schemaLocation │ │ │ │ │ + }); │ │ │ │ │ + if (metadata && typeof metadata == "object") { │ │ │ │ │ + gpx.appendChild(this.buildMetadataNode(metadata)) │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + gpx.appendChild(this.buildFeatureNode(features[i])) │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [gpx]) │ │ │ │ │ + }, │ │ │ │ │ + buildMetadataNode: function(metadata) { │ │ │ │ │ + var types = ["name", "desc", "author"], │ │ │ │ │ + node = this.createElementNS(this.namespaces.gpx, "metadata"); │ │ │ │ │ + for (var i = 0; i < types.length; i++) { │ │ │ │ │ + var type = types[i]; │ │ │ │ │ + if (metadata[type]) { │ │ │ │ │ + var n = this.createElementNS(this.namespaces.gpx, type); │ │ │ │ │ + n.appendChild(this.createTextNode(metadata[type])); │ │ │ │ │ + node.appendChild(n) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + buildFeatureNode: function(feature) { │ │ │ │ │ + var geometry = feature.geometry; │ │ │ │ │ + geometry = geometry.clone(); │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + geometry.transform(this.internalProjection, this.externalProjection) │ │ │ │ │ + } │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + var wpt = this.buildWptNode(geometry); │ │ │ │ │ + this.appendAttributesNode(wpt, feature); │ │ │ │ │ + return wpt │ │ │ │ │ + } else { │ │ │ │ │ + var trkNode = this.createElementNS(this.namespaces.gpx, "trk"); │ │ │ │ │ + this.appendAttributesNode(trkNode, feature); │ │ │ │ │ + var trkSegNodes = this.buildTrkSegNode(geometry); │ │ │ │ │ + trkSegNodes = OpenLayers.Util.isArray(trkSegNodes) ? trkSegNodes : [trkSegNodes]; │ │ │ │ │ + for (var i = 0, len = trkSegNodes.length; i < len; i++) { │ │ │ │ │ + trkNode.appendChild(trkSegNodes[i]) │ │ │ │ │ + } │ │ │ │ │ + return trkNode │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + buildTrkSegNode: function(geometry) { │ │ │ │ │ + var node, i, len, point, nodes; │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.LineString" || geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ + node = this.createElementNS(this.namespaces.gpx, "trkseg"); │ │ │ │ │ + for (i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + point = geometry.components[i]; │ │ │ │ │ + node.appendChild(this.buildTrkPtNode(point)) │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ + } else { │ │ │ │ │ + nodes = []; │ │ │ │ │ + for (i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ + nodes.push(this.buildTrkSegNode(geometry.components[i])) │ │ │ │ │ + } │ │ │ │ │ + return nodes │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + buildTrkPtNode: function(point) { │ │ │ │ │ + var node = this.createElementNS(this.namespaces.gpx, "trkpt"); │ │ │ │ │ + node.setAttribute("lon", point.x); │ │ │ │ │ + node.setAttribute("lat", point.y); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + buildWptNode: function(geometry) { │ │ │ │ │ + var node = this.createElementNS(this.namespaces.gpx, "wpt"); │ │ │ │ │ + node.setAttribute("lon", geometry.x); │ │ │ │ │ + node.setAttribute("lat", geometry.y); │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + appendAttributesNode: function(node, feature) { │ │ │ │ │ + var name = this.createElementNS(this.namespaces.gpx, "name"); │ │ │ │ │ + name.appendChild(this.createTextNode(feature.attributes.name || feature.id)); │ │ │ │ │ + node.appendChild(name); │ │ │ │ │ + var desc = this.createElementNS(this.namespaces.gpx, "desc"); │ │ │ │ │ + desc.appendChild(this.createTextNode(feature.attributes.description || this.defaultDesc)); │ │ │ │ │ + node.appendChild(desc) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.GPX" │ │ │ │ │ +}); │ │ │ │ │ OpenLayers.Format.WMC = OpenLayers.Class(OpenLayers.Format.Context, { │ │ │ │ │ defaultVersion: "1.1.0", │ │ │ │ │ layerToContext: function(layer) { │ │ │ │ │ var parser = this.getParser(); │ │ │ │ │ var layerContext = { │ │ │ │ │ queryable: layer.queryable, │ │ │ │ │ visibility: layer.visibility, │ │ │ │ │ @@ -22614,14 +24803,451 @@ │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ return context │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.WMC" │ │ │ │ │ }); │ │ │ │ │ +OpenLayers.Format.WPSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + defaultVersion: "1.0.0", │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WPSCapabilities" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSGetFeatureInfo = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + layerIdentifier: "_layer", │ │ │ │ │ + featureIdentifier: "_feature", │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: /^\s*|\s*$/g, │ │ │ │ │ + removeSpace: /\s*/g, │ │ │ │ │ + splitSpace: /\s+/, │ │ │ │ │ + trimComma: /\s*,\s*/g │ │ │ │ │ + }, │ │ │ │ │ + gmlFormat: null, │ │ │ │ │ + read: function(data) { │ │ │ │ │ + var result; │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + var root = data.documentElement; │ │ │ │ │ + if (root) { │ │ │ │ │ + var scope = this; │ │ │ │ │ + var read = this["read_" + root.nodeName]; │ │ │ │ │ + if (read) { │ │ │ │ │ + result = read.call(this, root) │ │ │ │ │ + } else { │ │ │ │ │ + result = new OpenLayers.Format.GML(this.options ? this.options : {}).read(data) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + result = data │ │ │ │ │ + } │ │ │ │ │ + return result │ │ │ │ │ + }, │ │ │ │ │ + read_msGMLOutput: function(data) { │ │ │ │ │ + var response = []; │ │ │ │ │ + var layerNodes = this.getSiblingNodesByTagCriteria(data, this.layerIdentifier); │ │ │ │ │ + if (layerNodes) { │ │ │ │ │ + for (var i = 0, len = layerNodes.length; i < len; ++i) { │ │ │ │ │ + var node = layerNodes[i]; │ │ │ │ │ + var layerName = node.nodeName; │ │ │ │ │ + if (node.prefix) { │ │ │ │ │ + layerName = layerName.split(":")[1] │ │ │ │ │ + } │ │ │ │ │ + var layerName = layerName.replace(this.layerIdentifier, ""); │ │ │ │ │ + var featureNodes = this.getSiblingNodesByTagCriteria(node, this.featureIdentifier); │ │ │ │ │ + if (featureNodes) { │ │ │ │ │ + for (var j = 0; j < featureNodes.length; j++) { │ │ │ │ │ + var featureNode = featureNodes[j]; │ │ │ │ │ + var geomInfo = this.parseGeometry(featureNode); │ │ │ │ │ + var attributes = this.parseAttributes(featureNode); │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geomInfo.geometry, attributes, null); │ │ │ │ │ + feature.bounds = geomInfo.bounds; │ │ │ │ │ + feature.type = layerName; │ │ │ │ │ + response.push(feature) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return response │ │ │ │ │ + }, │ │ │ │ │ + read_FeatureInfoResponse: function(data) { │ │ │ │ │ + var response = []; │ │ │ │ │ + var featureNodes = this.getElementsByTagNameNS(data, "*", "FIELDS"); │ │ │ │ │ + for (var i = 0, len = featureNodes.length; i < len; i++) { │ │ │ │ │ + var featureNode = featureNodes[i]; │ │ │ │ │ + var geom = null; │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + var j; │ │ │ │ │ + var jlen = featureNode.attributes.length; │ │ │ │ │ + if (jlen > 0) { │ │ │ │ │ + for (j = 0; j < jlen; j++) { │ │ │ │ │ + var attribute = featureNode.attributes[j]; │ │ │ │ │ + attributes[attribute.nodeName] = attribute.nodeValue │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + var nodes = featureNode.childNodes; │ │ │ │ │ + for (j = 0, jlen = nodes.length; j < jlen; ++j) { │ │ │ │ │ + var node = nodes[j]; │ │ │ │ │ + if (node.nodeType != 3) { │ │ │ │ │ + attributes[node.getAttribute("name")] = node.getAttribute("value") │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + response.push(new OpenLayers.Feature.Vector(geom, attributes, null)) │ │ │ │ │ + } │ │ │ │ │ + return response │ │ │ │ │ + }, │ │ │ │ │ + getSiblingNodesByTagCriteria: function(node, criteria) { │ │ │ │ │ + var nodes = []; │ │ │ │ │ + var children, tagName, n, matchNodes, child; │ │ │ │ │ + if (node && node.hasChildNodes()) { │ │ │ │ │ + children = node.childNodes; │ │ │ │ │ + n = children.length; │ │ │ │ │ + for (var k = 0; k < n; k++) { │ │ │ │ │ + child = children[k]; │ │ │ │ │ + while (child && child.nodeType != 1) { │ │ │ │ │ + child = child.nextSibling; │ │ │ │ │ + k++ │ │ │ │ │ + } │ │ │ │ │ + tagName = child ? child.nodeName : ""; │ │ │ │ │ + if (tagName.length > 0 && tagName.indexOf(criteria) > -1) { │ │ │ │ │ + nodes.push(child) │ │ │ │ │ + } else { │ │ │ │ │ + matchNodes = this.getSiblingNodesByTagCriteria(child, criteria); │ │ │ │ │ + if (matchNodes.length > 0) { │ │ │ │ │ + nodes.length == 0 ? nodes = matchNodes : nodes.push(matchNodes) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return nodes │ │ │ │ │ + }, │ │ │ │ │ + parseAttributes: function(node) { │ │ │ │ │ + var attributes = {}; │ │ │ │ │ + if (node.nodeType == 1) { │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + var n = children.length; │ │ │ │ │ + for (var i = 0; i < n; ++i) { │ │ │ │ │ + var child = children[i]; │ │ │ │ │ + if (child.nodeType == 1) { │ │ │ │ │ + var grandchildren = child.childNodes; │ │ │ │ │ + var name = child.prefix ? child.nodeName.split(":")[1] : child.nodeName; │ │ │ │ │ + if (grandchildren.length == 0) { │ │ │ │ │ + attributes[name] = null │ │ │ │ │ + } else if (grandchildren.length == 1) { │ │ │ │ │ + var grandchild = grandchildren[0]; │ │ │ │ │ + if (grandchild.nodeType == 3 || grandchild.nodeType == 4) { │ │ │ │ │ + var value = grandchild.nodeValue.replace(this.regExes.trimSpace, ""); │ │ │ │ │ + attributes[name] = value │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return attributes │ │ │ │ │ + }, │ │ │ │ │ + parseGeometry: function(node) { │ │ │ │ │ + if (!this.gmlFormat) { │ │ │ │ │ + this.gmlFormat = new OpenLayers.Format.GML │ │ │ │ │ + } │ │ │ │ │ + var feature = this.gmlFormat.parseFeature(node); │ │ │ │ │ + var geometry, bounds = null; │ │ │ │ │ + if (feature) { │ │ │ │ │ + geometry = feature.geometry && feature.geometry.clone(); │ │ │ │ │ + bounds = feature.bounds && feature.bounds.clone(); │ │ │ │ │ + feature.destroy() │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + geometry: geometry, │ │ │ │ │ + bounds: bounds │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSGetFeatureInfo" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WFSDescribeFeatureType = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: /^\s*|\s*$/g │ │ │ │ │ + }, │ │ │ │ │ + namespaces: { │ │ │ │ │ + xsd: "http://www.w3.org/2001/XMLSchema" │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + xsd: { │ │ │ │ │ + schema: function(node, obj) { │ │ │ │ │ + var complexTypes = []; │ │ │ │ │ + var customTypes = {}; │ │ │ │ │ + var schema = { │ │ │ │ │ + complexTypes: complexTypes, │ │ │ │ │ + customTypes: customTypes │ │ │ │ │ + }; │ │ │ │ │ + var i, len; │ │ │ │ │ + this.readChildNodes(node, schema); │ │ │ │ │ + var attributes = node.attributes; │ │ │ │ │ + var attr, name; │ │ │ │ │ + for (i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ + attr = attributes[i]; │ │ │ │ │ + name = attr.name; │ │ │ │ │ + if (name.indexOf("xmlns") === 0) { │ │ │ │ │ + this.setNamespace(name.split(":")[1] || "", attr.value) │ │ │ │ │ + } else { │ │ │ │ │ + obj[name] = attr.value │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + obj.featureTypes = complexTypes; │ │ │ │ │ + obj.targetPrefix = this.namespaceAlias[obj.targetNamespace]; │ │ │ │ │ + var complexType, customType; │ │ │ │ │ + for (i = 0, len = complexTypes.length; i < len; ++i) { │ │ │ │ │ + complexType = complexTypes[i]; │ │ │ │ │ + customType = customTypes[complexType.typeName]; │ │ │ │ │ + if (customTypes[complexType.typeName]) { │ │ │ │ │ + complexType.typeName = customType.name │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + complexType: function(node, obj) { │ │ │ │ │ + var complexType = { │ │ │ │ │ + typeName: node.getAttribute("name") │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, complexType); │ │ │ │ │ + obj.complexTypes.push(complexType) │ │ │ │ │ + }, │ │ │ │ │ + complexContent: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + extension: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + sequence: function(node, obj) { │ │ │ │ │ + var sequence = { │ │ │ │ │ + elements: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, sequence); │ │ │ │ │ + obj.properties = sequence.elements │ │ │ │ │ + }, │ │ │ │ │ + element: function(node, obj) { │ │ │ │ │ + var type; │ │ │ │ │ + if (obj.elements) { │ │ │ │ │ + var element = {}; │ │ │ │ │ + var attributes = node.attributes; │ │ │ │ │ + var attr; │ │ │ │ │ + for (var i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ + attr = attributes[i]; │ │ │ │ │ + element[attr.name] = attr.value │ │ │ │ │ + } │ │ │ │ │ + type = element.type; │ │ │ │ │ + if (!type) { │ │ │ │ │ + type = {}; │ │ │ │ │ + this.readChildNodes(node, type); │ │ │ │ │ + element.restriction = type; │ │ │ │ │ + element.type = type.base │ │ │ │ │ + } │ │ │ │ │ + var fullType = type.base || type; │ │ │ │ │ + element.localType = fullType.split(":").pop(); │ │ │ │ │ + obj.elements.push(element); │ │ │ │ │ + this.readChildNodes(node, element) │ │ │ │ │ + } │ │ │ │ │ + if (obj.complexTypes) { │ │ │ │ │ + type = node.getAttribute("type"); │ │ │ │ │ + var localType = type.split(":").pop(); │ │ │ │ │ + obj.customTypes[localType] = { │ │ │ │ │ + name: node.getAttribute("name"), │ │ │ │ │ + type: type │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + annotation: function(node, obj) { │ │ │ │ │ + obj.annotation = {}; │ │ │ │ │ + this.readChildNodes(node, obj.annotation) │ │ │ │ │ + }, │ │ │ │ │ + appinfo: function(node, obj) { │ │ │ │ │ + if (!obj.appinfo) { │ │ │ │ │ + obj.appinfo = [] │ │ │ │ │ + } │ │ │ │ │ + obj.appinfo.push(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + documentation: function(node, obj) { │ │ │ │ │ + if (!obj.documentation) { │ │ │ │ │ + obj.documentation = [] │ │ │ │ │ + } │ │ │ │ │ + var value = this.getChildValue(node); │ │ │ │ │ + obj.documentation.push({ │ │ │ │ │ + lang: node.getAttribute("xml:lang"), │ │ │ │ │ + textContent: value.replace(this.regExes.trimSpace, "") │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + simpleType: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + restriction: function(node, obj) { │ │ │ │ │ + obj.base = node.getAttribute("base"); │ │ │ │ │ + this.readRestriction(node, obj) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + readRestriction: function(node, obj) { │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + var child, nodeName, value; │ │ │ │ │ + for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ + child = children[i]; │ │ │ │ │ + if (child.nodeType == 1) { │ │ │ │ │ + nodeName = child.nodeName.split(":").pop(); │ │ │ │ │ + value = child.getAttribute("value"); │ │ │ │ │ + if (!obj[nodeName]) { │ │ │ │ │ + obj[nodeName] = value │ │ │ │ │ + } else { │ │ │ │ │ + if (typeof obj[nodeName] == "string") { │ │ │ │ │ + obj[nodeName] = [obj[nodeName]] │ │ │ │ │ + } │ │ │ │ │ + obj[nodeName].push(value) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement │ │ │ │ │ + } │ │ │ │ │ + var schema = {}; │ │ │ │ │ + if (data.nodeName.split(":").pop() === "ExceptionReport") { │ │ │ │ │ + var parser = new OpenLayers.Format.OGCExceptionReport; │ │ │ │ │ + schema.error = parser.read(data) │ │ │ │ │ + } else { │ │ │ │ │ + this.readNode(data, schema) │ │ │ │ │ + } │ │ │ │ │ + return schema │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSDescribeFeatureType" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.CSWGetDomain = function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults(options, OpenLayers.Format.CSWGetDomain.DEFAULTS); │ │ │ │ │ + var cls = OpenLayers.Format.CSWGetDomain["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (!cls) { │ │ │ │ │ + throw "Unsupported CSWGetDomain version: " + options.version │ │ │ │ │ + } │ │ │ │ │ + return new cls(options) │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Format.CSWGetDomain.DEFAULTS = { │ │ │ │ │ + version: "2.0.2" │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Format.WMTSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + defaultVersion: "1.0.0", │ │ │ │ │ + yx: { │ │ │ │ │ + "urn:ogc:def:crs:EPSG::4326": true │ │ │ │ │ + }, │ │ │ │ │ + createLayer: function(capabilities, config) { │ │ │ │ │ + var layer; │ │ │ │ │ + if (!("layer" in config)) { │ │ │ │ │ + throw new Error("Missing property 'layer' in configuration.") │ │ │ │ │ + } │ │ │ │ │ + var contents = capabilities.contents; │ │ │ │ │ + var layers = contents.layers; │ │ │ │ │ + var layerDef; │ │ │ │ │ + for (var i = 0, ii = contents.layers.length; i < ii; ++i) { │ │ │ │ │ + if (contents.layers[i].identifier === config.layer) { │ │ │ │ │ + layerDef = contents.layers[i]; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!layerDef) { │ │ │ │ │ + throw new Error("Layer not found") │ │ │ │ │ + } │ │ │ │ │ + var format = config.format; │ │ │ │ │ + if (!format && layerDef.formats && layerDef.formats.length) { │ │ │ │ │ + format = layerDef.formats[0] │ │ │ │ │ + } │ │ │ │ │ + var matrixSet; │ │ │ │ │ + if (config.matrixSet) { │ │ │ │ │ + matrixSet = contents.tileMatrixSets[config.matrixSet] │ │ │ │ │ + } else if (layerDef.tileMatrixSetLinks.length >= 1) { │ │ │ │ │ + matrixSet = contents.tileMatrixSets[layerDef.tileMatrixSetLinks[0].tileMatrixSet] │ │ │ │ │ + } │ │ │ │ │ + if (!matrixSet) { │ │ │ │ │ + throw new Error("matrixSet not found") │ │ │ │ │ + } │ │ │ │ │ + var style; │ │ │ │ │ + for (var i = 0, ii = layerDef.styles.length; i < ii; ++i) { │ │ │ │ │ + style = layerDef.styles[i]; │ │ │ │ │ + if (style.isDefault) { │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var requestEncoding = config.requestEncoding; │ │ │ │ │ + if (!requestEncoding) { │ │ │ │ │ + requestEncoding = "KVP"; │ │ │ │ │ + if (capabilities.operationsMetadata.GetTile.dcp.http) { │ │ │ │ │ + var http = capabilities.operationsMetadata.GetTile.dcp.http; │ │ │ │ │ + if (http.get[0].constraints) { │ │ │ │ │ + var constraints = http.get[0].constraints; │ │ │ │ │ + var allowedValues = constraints.GetEncoding.allowedValues; │ │ │ │ │ + if (!allowedValues.KVP && (allowedValues.REST || allowedValues.RESTful)) { │ │ │ │ │ + requestEncoding = "REST" │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var dimensions = []; │ │ │ │ │ + var params = config.params || {}; │ │ │ │ │ + delete config.params; │ │ │ │ │ + for (var id = 0, ld = layerDef.dimensions.length; id < ld; id++) { │ │ │ │ │ + var dimension = layerDef.dimensions[id]; │ │ │ │ │ + dimensions.push(dimension.identifier); │ │ │ │ │ + if (!params.hasOwnProperty(dimension.identifier)) { │ │ │ │ │ + params[dimension.identifier] = dimension["default"] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var projection = config.projection || matrixSet.supportedCRS.replace(/urn:ogc:def:crs:(\w+):(.*:)?(\w+)$/, "$1:$3"); │ │ │ │ │ + var units = config.units || (projection === "EPSG:4326" ? "degrees" : "m"); │ │ │ │ │ + var resolutions = []; │ │ │ │ │ + for (var mid in matrixSet.matrixIds) { │ │ │ │ │ + if (matrixSet.matrixIds.hasOwnProperty(mid)) { │ │ │ │ │ + resolutions.push(matrixSet.matrixIds[mid].scaleDenominator * 28e-5 / OpenLayers.METERS_PER_INCH / OpenLayers.INCHES_PER_UNIT[units]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var url; │ │ │ │ │ + if (requestEncoding === "REST" && layerDef.resourceUrls) { │ │ │ │ │ + url = []; │ │ │ │ │ + var resourceUrls = layerDef.resourceUrls, │ │ │ │ │ + resourceUrl; │ │ │ │ │ + for (var t = 0, tt = layerDef.resourceUrls.length; t < tt; ++t) { │ │ │ │ │ + resourceUrl = layerDef.resourceUrls[t]; │ │ │ │ │ + if (resourceUrl.format === format && resourceUrl.resourceType === "tile") { │ │ │ │ │ + url.push(resourceUrl.template) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + var httpGet = capabilities.operationsMetadata.GetTile.dcp.http.get; │ │ │ │ │ + url = []; │ │ │ │ │ + var constraint; │ │ │ │ │ + for (var i = 0, ii = httpGet.length; i < ii; i++) { │ │ │ │ │ + constraint = httpGet[i].constraints; │ │ │ │ │ + if (!constraint || constraint && constraint.GetEncoding.allowedValues[requestEncoding]) { │ │ │ │ │ + url.push(httpGet[i].url) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Layer.WMTS(OpenLayers.Util.applyDefaults(config, { │ │ │ │ │ + url: url, │ │ │ │ │ + requestEncoding: requestEncoding, │ │ │ │ │ + name: layerDef.title, │ │ │ │ │ + style: style.identifier, │ │ │ │ │ + format: format, │ │ │ │ │ + matrixIds: matrixSet.matrixIds, │ │ │ │ │ + matrixSet: matrixSet.identifier, │ │ │ │ │ + projection: projection, │ │ │ │ │ + units: units, │ │ │ │ │ + resolutions: config.isBaseLayer === false ? undefined : resolutions, │ │ │ │ │ + serverResolutions: resolutions, │ │ │ │ │ + tileFullExtent: matrixSet.bounds, │ │ │ │ │ + dimensions: dimensions, │ │ │ │ │ + params: params │ │ │ │ │ + })) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMTSCapabilities" │ │ │ │ │ +}); │ │ │ │ │ OpenLayers.Format.OSM = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ checkTags: false, │ │ │ │ │ interestingTagsExclude: null, │ │ │ │ │ areaTags: null, │ │ │ │ │ initialize: function(options) { │ │ │ │ │ var layer_defaults = { │ │ │ │ │ interestingTagsExclude: ["source", "source_ref", "source:ref", "history", "attribution", "created_by"], │ │ │ │ │ @@ -22881,35 +25507,129 @@ │ │ │ │ │ if (state) { │ │ │ │ │ node.setAttribute("action", state) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.OSM" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.OWSContext = OpenLayers.Class(OpenLayers.Format.Context, { │ │ │ │ │ - defaultVersion: "0.3.1", │ │ │ │ │ - getVersion: function(root, options) { │ │ │ │ │ - var version = OpenLayers.Format.XML.VersionedOGC.prototype.getVersion.apply(this, arguments); │ │ │ │ │ - if (version === "0.3.0") { │ │ │ │ │ - version = this.defaultVersion │ │ │ │ │ +OpenLayers.Format.WFS = OpenLayers.Class(OpenLayers.Format.GML, { │ │ │ │ │ + layer: null, │ │ │ │ │ + wfsns: "http://www.opengis.net/wfs", │ │ │ │ │ + ogcns: "http://www.opengis.net/ogc", │ │ │ │ │ + initialize: function(options, layer) { │ │ │ │ │ + OpenLayers.Format.GML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + if (this.layer.featureNS) { │ │ │ │ │ + this.featureNS = this.layer.featureNS │ │ │ │ │ + } │ │ │ │ │ + if (this.layer.options.geometry_column) { │ │ │ │ │ + this.geometryName = this.layer.options.geometry_column │ │ │ │ │ + } │ │ │ │ │ + if (this.layer.options.typename) { │ │ │ │ │ + this.featureName = this.layer.options.typename │ │ │ │ │ } │ │ │ │ │ - return version │ │ │ │ │ }, │ │ │ │ │ - toContext: function(obj) { │ │ │ │ │ - var context = {}; │ │ │ │ │ - if (obj.CLASS_NAME == "OpenLayers.Map") { │ │ │ │ │ - context.bounds = obj.getExtent(); │ │ │ │ │ - context.maxExtent = obj.maxExtent; │ │ │ │ │ - context.projection = obj.projection; │ │ │ │ │ - context.size = obj.getSize(); │ │ │ │ │ - context.layers = obj.layers │ │ │ │ │ + write: function(features) { │ │ │ │ │ + var transaction = this.createElementNS(this.wfsns, "wfs:Transaction"); │ │ │ │ │ + transaction.setAttribute("version", "1.0.0"); │ │ │ │ │ + transaction.setAttribute("service", "WFS"); │ │ │ │ │ + for (var i = 0; i < features.length; i++) { │ │ │ │ │ + switch (features[i].state) { │ │ │ │ │ + case OpenLayers.State.INSERT: │ │ │ │ │ + transaction.appendChild(this.insert(features[i])); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.State.UPDATE: │ │ │ │ │ + transaction.appendChild(this.update(features[i])); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.State.DELETE: │ │ │ │ │ + transaction.appendChild(this.remove(features[i])); │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return context │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [transaction]) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.OWSContext" │ │ │ │ │ + createFeatureXML: function(feature) { │ │ │ │ │ + var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ + var geomContainer = this.createElementNS(this.featureNS, "feature:" + this.geometryName); │ │ │ │ │ + geomContainer.appendChild(geometryNode); │ │ │ │ │ + var featureContainer = this.createElementNS(this.featureNS, "feature:" + this.featureName); │ │ │ │ │ + featureContainer.appendChild(geomContainer); │ │ │ │ │ + for (var attr in feature.attributes) { │ │ │ │ │ + var attrText = this.createTextNode(feature.attributes[attr]); │ │ │ │ │ + var nodename = attr; │ │ │ │ │ + if (attr.search(":") != -1) { │ │ │ │ │ + nodename = attr.split(":")[1] │ │ │ │ │ + } │ │ │ │ │ + var attrContainer = this.createElementNS(this.featureNS, "feature:" + nodename); │ │ │ │ │ + attrContainer.appendChild(attrText); │ │ │ │ │ + featureContainer.appendChild(attrContainer) │ │ │ │ │ + } │ │ │ │ │ + return featureContainer │ │ │ │ │ + }, │ │ │ │ │ + insert: function(feature) { │ │ │ │ │ + var insertNode = this.createElementNS(this.wfsns, "wfs:Insert"); │ │ │ │ │ + insertNode.appendChild(this.createFeatureXML(feature)); │ │ │ │ │ + return insertNode │ │ │ │ │ + }, │ │ │ │ │ + update: function(feature) { │ │ │ │ │ + if (!feature.fid) { │ │ │ │ │ + OpenLayers.Console.userError(OpenLayers.i18n("noFID")) │ │ │ │ │ + } │ │ │ │ │ + var updateNode = this.createElementNS(this.wfsns, "wfs:Update"); │ │ │ │ │ + updateNode.setAttribute("typeName", this.featurePrefix + ":" + this.featureName); │ │ │ │ │ + updateNode.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ + var propertyNode = this.createElementNS(this.wfsns, "wfs:Property"); │ │ │ │ │ + var nameNode = this.createElementNS(this.wfsns, "wfs:Name"); │ │ │ │ │ + var txtNode = this.createTextNode(this.geometryName); │ │ │ │ │ + nameNode.appendChild(txtNode); │ │ │ │ │ + propertyNode.appendChild(nameNode); │ │ │ │ │ + var valueNode = this.createElementNS(this.wfsns, "wfs:Value"); │ │ │ │ │ + var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ + if (feature.layer) { │ │ │ │ │ + geometryNode.setAttribute("srsName", feature.layer.projection.getCode()) │ │ │ │ │ + } │ │ │ │ │ + valueNode.appendChild(geometryNode); │ │ │ │ │ + propertyNode.appendChild(valueNode); │ │ │ │ │ + updateNode.appendChild(propertyNode); │ │ │ │ │ + for (var propName in feature.attributes) { │ │ │ │ │ + propertyNode = this.createElementNS(this.wfsns, "wfs:Property"); │ │ │ │ │ + nameNode = this.createElementNS(this.wfsns, "wfs:Name"); │ │ │ │ │ + nameNode.appendChild(this.createTextNode(propName)); │ │ │ │ │ + propertyNode.appendChild(nameNode); │ │ │ │ │ + valueNode = this.createElementNS(this.wfsns, "wfs:Value"); │ │ │ │ │ + valueNode.appendChild(this.createTextNode(feature.attributes[propName])); │ │ │ │ │ + propertyNode.appendChild(valueNode); │ │ │ │ │ + updateNode.appendChild(propertyNode) │ │ │ │ │ + } │ │ │ │ │ + var filterNode = this.createElementNS(this.ogcns, "ogc:Filter"); │ │ │ │ │ + var filterIdNode = this.createElementNS(this.ogcns, "ogc:FeatureId"); │ │ │ │ │ + filterIdNode.setAttribute("fid", feature.fid); │ │ │ │ │ + filterNode.appendChild(filterIdNode); │ │ │ │ │ + updateNode.appendChild(filterNode); │ │ │ │ │ + return updateNode │ │ │ │ │ + }, │ │ │ │ │ + remove: function(feature) { │ │ │ │ │ + if (!feature.fid) { │ │ │ │ │ + OpenLayers.Console.userError(OpenLayers.i18n("noFID")); │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + var deleteNode = this.createElementNS(this.wfsns, "wfs:Delete"); │ │ │ │ │ + deleteNode.setAttribute("typeName", this.featurePrefix + ":" + this.featureName); │ │ │ │ │ + deleteNode.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ + var filterNode = this.createElementNS(this.ogcns, "ogc:Filter"); │ │ │ │ │ + var filterIdNode = this.createElementNS(this.ogcns, "ogc:FeatureId"); │ │ │ │ │ + filterIdNode.setAttribute("fid", feature.fid); │ │ │ │ │ + filterNode.appendChild(filterIdNode); │ │ │ │ │ + deleteNode.appendChild(filterNode); │ │ │ │ │ + return deleteNode │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.layer = null │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFS" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Format.EncodedPolyline = OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ geometryType: "linestring", │ │ │ │ │ initialize: function(options) { │ │ │ │ │ OpenLayers.Format.prototype.initialize.apply(this, [options]) │ │ │ │ │ }, │ │ │ │ │ read: function(encoded) { │ │ │ │ │ @@ -23115,263 +25835,377 @@ │ │ │ │ │ if (b < 32) break; │ │ │ │ │ shift += 5 │ │ │ │ │ } │ │ │ │ │ return result │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.EncodedPolyline" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.WMSGetFeatureInfo = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - layerIdentifier: "_layer", │ │ │ │ │ - featureIdentifier: "_feature", │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: /^\s*|\s*$/g, │ │ │ │ │ - removeSpace: /\s*/g, │ │ │ │ │ - splitSpace: /\s+/, │ │ │ │ │ - trimComma: /\s*,\s*/g │ │ │ │ │ - }, │ │ │ │ │ - gmlFormat: null, │ │ │ │ │ - read: function(data) { │ │ │ │ │ - var result; │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ +OpenLayers.Format.SOSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + defaultVersion: "1.0.0", │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SOSCapabilities" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.CQL = function() { │ │ │ │ │ + var tokens = ["PROPERTY", "COMPARISON", "VALUE", "LOGICAL"], │ │ │ │ │ + patterns = { │ │ │ │ │ + PROPERTY: /^[_a-zA-Z]\w*/, │ │ │ │ │ + COMPARISON: /^(=|<>|<=|<|>=|>|LIKE)/i, │ │ │ │ │ + IS_NULL: /^IS NULL/i, │ │ │ │ │ + COMMA: /^,/, │ │ │ │ │ + LOGICAL: /^(AND|OR)/i, │ │ │ │ │ + VALUE: /^('([^']|'')*'|\d+(\.\d*)?|\.\d+)/, │ │ │ │ │ + LPAREN: /^\(/, │ │ │ │ │ + RPAREN: /^\)/, │ │ │ │ │ + SPATIAL: /^(BBOX|INTERSECTS|DWITHIN|WITHIN|CONTAINS)/i, │ │ │ │ │ + NOT: /^NOT/i, │ │ │ │ │ + BETWEEN: /^BETWEEN/i, │ │ │ │ │ + GEOMETRY: function(text) { │ │ │ │ │ + var type = /^(POINT|LINESTRING|POLYGON|MULTIPOINT|MULTILINESTRING|MULTIPOLYGON|GEOMETRYCOLLECTION)/.exec(text); │ │ │ │ │ + if (type) { │ │ │ │ │ + var len = text.length; │ │ │ │ │ + var idx = text.indexOf("(", type[0].length); │ │ │ │ │ + if (idx > -1) { │ │ │ │ │ + var depth = 1; │ │ │ │ │ + while (idx < len && depth > 0) { │ │ │ │ │ + idx++; │ │ │ │ │ + switch (text.charAt(idx)) { │ │ │ │ │ + case "(": │ │ │ │ │ + depth++; │ │ │ │ │ + break; │ │ │ │ │ + case ")": │ │ │ │ │ + depth--; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return [text.substr(0, idx + 1)] │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + END: /^$/ │ │ │ │ │ + }, │ │ │ │ │ + follows = { │ │ │ │ │ + LPAREN: ["GEOMETRY", "SPATIAL", "PROPERTY", "VALUE", "LPAREN"], │ │ │ │ │ + RPAREN: ["NOT", "LOGICAL", "END", "RPAREN"], │ │ │ │ │ + PROPERTY: ["COMPARISON", "BETWEEN", "COMMA", "IS_NULL"], │ │ │ │ │ + BETWEEN: ["VALUE"], │ │ │ │ │ + IS_NULL: ["END"], │ │ │ │ │ + COMPARISON: ["VALUE"], │ │ │ │ │ + COMMA: ["GEOMETRY", "VALUE", "PROPERTY"], │ │ │ │ │ + VALUE: ["LOGICAL", "COMMA", "RPAREN", "END"], │ │ │ │ │ + SPATIAL: ["LPAREN"], │ │ │ │ │ + LOGICAL: ["NOT", "VALUE", "SPATIAL", "PROPERTY", "LPAREN"], │ │ │ │ │ + NOT: ["PROPERTY", "LPAREN"], │ │ │ │ │ + GEOMETRY: ["COMMA", "RPAREN"] │ │ │ │ │ + }, │ │ │ │ │ + operators = { │ │ │ │ │ + "=": OpenLayers.Filter.Comparison.EQUAL_TO, │ │ │ │ │ + "<>": OpenLayers.Filter.Comparison.NOT_EQUAL_TO, │ │ │ │ │ + "<": OpenLayers.Filter.Comparison.LESS_THAN, │ │ │ │ │ + "<=": OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO, │ │ │ │ │ + ">": OpenLayers.Filter.Comparison.GREATER_THAN, │ │ │ │ │ + ">=": OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO, │ │ │ │ │ + LIKE: OpenLayers.Filter.Comparison.LIKE, │ │ │ │ │ + BETWEEN: OpenLayers.Filter.Comparison.BETWEEN, │ │ │ │ │ + "IS NULL": OpenLayers.Filter.Comparison.IS_NULL │ │ │ │ │ + }, │ │ │ │ │ + operatorReverse = {}, │ │ │ │ │ + logicals = { │ │ │ │ │ + AND: OpenLayers.Filter.Logical.AND, │ │ │ │ │ + OR: OpenLayers.Filter.Logical.OR │ │ │ │ │ + }, │ │ │ │ │ + logicalReverse = {}, │ │ │ │ │ + precedence = { │ │ │ │ │ + RPAREN: 3, │ │ │ │ │ + LOGICAL: 2, │ │ │ │ │ + COMPARISON: 1 │ │ │ │ │ + }; │ │ │ │ │ + var i; │ │ │ │ │ + for (i in operators) { │ │ │ │ │ + if (operators.hasOwnProperty(i)) { │ │ │ │ │ + operatorReverse[operators[i]] = i │ │ │ │ │ } │ │ │ │ │ - var root = data.documentElement; │ │ │ │ │ - if (root) { │ │ │ │ │ - var scope = this; │ │ │ │ │ - var read = this["read_" + root.nodeName]; │ │ │ │ │ - if (read) { │ │ │ │ │ - result = read.call(this, root) │ │ │ │ │ - } else { │ │ │ │ │ - result = new OpenLayers.Format.GML(this.options ? this.options : {}).read(data) │ │ │ │ │ - } │ │ │ │ │ + } │ │ │ │ │ + for (i in logicals) { │ │ │ │ │ + if (logicals.hasOwnProperty(i)) { │ │ │ │ │ + logicalReverse[logicals[i]] = i │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function tryToken(text, pattern) { │ │ │ │ │ + if (pattern instanceof RegExp) { │ │ │ │ │ + return pattern.exec(text) │ │ │ │ │ } else { │ │ │ │ │ - result = data │ │ │ │ │ + return pattern(text) │ │ │ │ │ } │ │ │ │ │ - return result │ │ │ │ │ - }, │ │ │ │ │ - read_msGMLOutput: function(data) { │ │ │ │ │ - var response = []; │ │ │ │ │ - var layerNodes = this.getSiblingNodesByTagCriteria(data, this.layerIdentifier); │ │ │ │ │ - if (layerNodes) { │ │ │ │ │ - for (var i = 0, len = layerNodes.length; i < len; ++i) { │ │ │ │ │ - var node = layerNodes[i]; │ │ │ │ │ - var layerName = node.nodeName; │ │ │ │ │ - if (node.prefix) { │ │ │ │ │ - layerName = layerName.split(":")[1] │ │ │ │ │ - } │ │ │ │ │ - var layerName = layerName.replace(this.layerIdentifier, ""); │ │ │ │ │ - var featureNodes = this.getSiblingNodesByTagCriteria(node, this.featureIdentifier); │ │ │ │ │ - if (featureNodes) { │ │ │ │ │ - for (var j = 0; j < featureNodes.length; j++) { │ │ │ │ │ - var featureNode = featureNodes[j]; │ │ │ │ │ - var geomInfo = this.parseGeometry(featureNode); │ │ │ │ │ - var attributes = this.parseAttributes(featureNode); │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geomInfo.geometry, attributes, null); │ │ │ │ │ - feature.bounds = geomInfo.bounds; │ │ │ │ │ - feature.type = layerName; │ │ │ │ │ - response.push(feature) │ │ │ │ │ - } │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function nextToken(text, tokens) { │ │ │ │ │ + var i, token, len = tokens.length; │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + token = tokens[i]; │ │ │ │ │ + var pat = patterns[token]; │ │ │ │ │ + var matches = tryToken(text, pat); │ │ │ │ │ + if (matches) { │ │ │ │ │ + var match = matches[0]; │ │ │ │ │ + var remainder = text.substr(match.length).replace(/^\s*/, ""); │ │ │ │ │ + return { │ │ │ │ │ + type: token, │ │ │ │ │ + text: match, │ │ │ │ │ + remainder: remainder │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return response │ │ │ │ │ - }, │ │ │ │ │ - read_FeatureInfoResponse: function(data) { │ │ │ │ │ - var response = []; │ │ │ │ │ - var featureNodes = this.getElementsByTagNameNS(data, "*", "FIELDS"); │ │ │ │ │ - for (var i = 0, len = featureNodes.length; i < len; i++) { │ │ │ │ │ - var featureNode = featureNodes[i]; │ │ │ │ │ - var geom = null; │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - var j; │ │ │ │ │ - var jlen = featureNode.attributes.length; │ │ │ │ │ - if (jlen > 0) { │ │ │ │ │ - for (j = 0; j < jlen; j++) { │ │ │ │ │ - var attribute = featureNode.attributes[j]; │ │ │ │ │ - attributes[attribute.nodeName] = attribute.nodeValue │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - var nodes = featureNode.childNodes; │ │ │ │ │ - for (j = 0, jlen = nodes.length; j < jlen; ++j) { │ │ │ │ │ - var node = nodes[j]; │ │ │ │ │ - if (node.nodeType != 3) { │ │ │ │ │ - attributes[node.getAttribute("name")] = node.getAttribute("value") │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - response.push(new OpenLayers.Feature.Vector(geom, attributes, null)) │ │ │ │ │ + var msg = "ERROR: In parsing: [" + text + "], expected one of: "; │ │ │ │ │ + for (i = 0; i < len; i++) { │ │ │ │ │ + token = tokens[i]; │ │ │ │ │ + msg += "\n " + token + ": " + patterns[token] │ │ │ │ │ } │ │ │ │ │ - return response │ │ │ │ │ - }, │ │ │ │ │ - getSiblingNodesByTagCriteria: function(node, criteria) { │ │ │ │ │ - var nodes = []; │ │ │ │ │ - var children, tagName, n, matchNodes, child; │ │ │ │ │ - if (node && node.hasChildNodes()) { │ │ │ │ │ - children = node.childNodes; │ │ │ │ │ - n = children.length; │ │ │ │ │ - for (var k = 0; k < n; k++) { │ │ │ │ │ - child = children[k]; │ │ │ │ │ - while (child && child.nodeType != 1) { │ │ │ │ │ - child = child.nextSibling; │ │ │ │ │ - k++ │ │ │ │ │ - } │ │ │ │ │ - tagName = child ? child.nodeName : ""; │ │ │ │ │ - if (tagName.length > 0 && tagName.indexOf(criteria) > -1) { │ │ │ │ │ - nodes.push(child) │ │ │ │ │ - } else { │ │ │ │ │ - matchNodes = this.getSiblingNodesByTagCriteria(child, criteria); │ │ │ │ │ - if (matchNodes.length > 0) { │ │ │ │ │ - nodes.length == 0 ? nodes = matchNodes : nodes.push(matchNodes) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + throw new Error(msg) │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function tokenize(text) { │ │ │ │ │ + var results = []; │ │ │ │ │ + var token, expect = ["NOT", "GEOMETRY", "SPATIAL", "PROPERTY", "LPAREN"]; │ │ │ │ │ + do { │ │ │ │ │ + token = nextToken(text, expect); │ │ │ │ │ + text = token.remainder; │ │ │ │ │ + expect = follows[token.type]; │ │ │ │ │ + if (token.type != "END" && !expect) { │ │ │ │ │ + throw new Error("No follows list for " + token.type) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - return nodes │ │ │ │ │ - }, │ │ │ │ │ - parseAttributes: function(node) { │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - if (node.nodeType == 1) { │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - var n = children.length; │ │ │ │ │ - for (var i = 0; i < n; ++i) { │ │ │ │ │ - var child = children[i]; │ │ │ │ │ - if (child.nodeType == 1) { │ │ │ │ │ - var grandchildren = child.childNodes; │ │ │ │ │ - var name = child.prefix ? child.nodeName.split(":")[1] : child.nodeName; │ │ │ │ │ - if (grandchildren.length == 0) { │ │ │ │ │ - attributes[name] = null │ │ │ │ │ - } else if (grandchildren.length == 1) { │ │ │ │ │ - var grandchild = grandchildren[0]; │ │ │ │ │ - if (grandchild.nodeType == 3 || grandchild.nodeType == 4) { │ │ │ │ │ - var value = grandchild.nodeValue.replace(this.regExes.trimSpace, ""); │ │ │ │ │ - attributes[name] = value │ │ │ │ │ - } │ │ │ │ │ + results.push(token) │ │ │ │ │ + } while (token.type != "END"); │ │ │ │ │ + return results │ │ │ │ │ + } │ │ │ │ │ + │ │ │ │ │ + function buildAst(tokens) { │ │ │ │ │ + var operatorStack = [], │ │ │ │ │ + postfix = []; │ │ │ │ │ + while (tokens.length) { │ │ │ │ │ + var tok = tokens.shift(); │ │ │ │ │ + switch (tok.type) { │ │ │ │ │ + case "PROPERTY": │ │ │ │ │ + case "GEOMETRY": │ │ │ │ │ + case "VALUE": │ │ │ │ │ + postfix.push(tok); │ │ │ │ │ + break; │ │ │ │ │ + case "COMPARISON": │ │ │ │ │ + case "BETWEEN": │ │ │ │ │ + case "IS_NULL": │ │ │ │ │ + case "LOGICAL": │ │ │ │ │ + var p = precedence[tok.type]; │ │ │ │ │ + while (operatorStack.length > 0 && precedence[operatorStack[operatorStack.length - 1].type] <= p) { │ │ │ │ │ + postfix.push(operatorStack.pop()) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ + operatorStack.push(tok); │ │ │ │ │ + break; │ │ │ │ │ + case "SPATIAL": │ │ │ │ │ + case "NOT": │ │ │ │ │ + case "LPAREN": │ │ │ │ │ + operatorStack.push(tok); │ │ │ │ │ + break; │ │ │ │ │ + case "RPAREN": │ │ │ │ │ + while (operatorStack.length > 0 && operatorStack[operatorStack.length - 1].type != "LPAREN") { │ │ │ │ │ + postfix.push(operatorStack.pop()) │ │ │ │ │ + } │ │ │ │ │ + operatorStack.pop(); │ │ │ │ │ + if (operatorStack.length > 0 && operatorStack[operatorStack.length - 1].type == "SPATIAL") { │ │ │ │ │ + postfix.push(operatorStack.pop()) │ │ │ │ │ + } │ │ │ │ │ + case "COMMA": │ │ │ │ │ + case "END": │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + throw new Error("Unknown token type " + tok.type) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return attributes │ │ │ │ │ - }, │ │ │ │ │ - parseGeometry: function(node) { │ │ │ │ │ - if (!this.gmlFormat) { │ │ │ │ │ - this.gmlFormat = new OpenLayers.Format.GML │ │ │ │ │ + while (operatorStack.length > 0) { │ │ │ │ │ + postfix.push(operatorStack.pop()) │ │ │ │ │ } │ │ │ │ │ - var feature = this.gmlFormat.parseFeature(node); │ │ │ │ │ - var geometry, bounds = null; │ │ │ │ │ - if (feature) { │ │ │ │ │ - geometry = feature.geometry && feature.geometry.clone(); │ │ │ │ │ - bounds = feature.bounds && feature.bounds.clone(); │ │ │ │ │ - feature.destroy() │ │ │ │ │ + │ │ │ │ │ + function buildTree() { │ │ │ │ │ + var tok = postfix.pop(); │ │ │ │ │ + switch (tok.type) { │ │ │ │ │ + case "LOGICAL": │ │ │ │ │ + var rhs = buildTree(), │ │ │ │ │ + lhs = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Logical({ │ │ │ │ │ + filters: [lhs, rhs], │ │ │ │ │ + type: logicals[tok.text.toUpperCase()] │ │ │ │ │ + }); │ │ │ │ │ + case "NOT": │ │ │ │ │ + var operand = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Logical({ │ │ │ │ │ + filters: [operand], │ │ │ │ │ + type: OpenLayers.Filter.Logical.NOT │ │ │ │ │ + }); │ │ │ │ │ + case "BETWEEN": │ │ │ │ │ + var min, max, property; │ │ │ │ │ + postfix.pop(); │ │ │ │ │ + max = buildTree(); │ │ │ │ │ + min = buildTree(); │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Comparison({ │ │ │ │ │ + property: property, │ │ │ │ │ + lowerBoundary: min, │ │ │ │ │ + upperBoundary: max, │ │ │ │ │ + type: OpenLayers.Filter.Comparison.BETWEEN │ │ │ │ │ + }); │ │ │ │ │ + case "COMPARISON": │ │ │ │ │ + var value = buildTree(), │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Comparison({ │ │ │ │ │ + property: property, │ │ │ │ │ + value: value, │ │ │ │ │ + type: operators[tok.text.toUpperCase()] │ │ │ │ │ + }); │ │ │ │ │ + case "IS_NULL": │ │ │ │ │ + var property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Comparison({ │ │ │ │ │ + property: property, │ │ │ │ │ + type: operators[tok.text.toUpperCase()] │ │ │ │ │ + }); │ │ │ │ │ + case "VALUE": │ │ │ │ │ + var match = tok.text.match(/^'(.*)'$/); │ │ │ │ │ + if (match) { │ │ │ │ │ + return match[1].replace(/''/g, "'") │ │ │ │ │ + } else { │ │ │ │ │ + return Number(tok.text) │ │ │ │ │ + } │ │ │ │ │ + case "SPATIAL": │ │ │ │ │ + switch (tok.text.toUpperCase()) { │ │ │ │ │ + case "BBOX": │ │ │ │ │ + var maxy = buildTree(), │ │ │ │ │ + maxx = buildTree(), │ │ │ │ │ + miny = buildTree(), │ │ │ │ │ + minx = buildTree(), │ │ │ │ │ + prop = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ + property: prop, │ │ │ │ │ + value: OpenLayers.Bounds.fromArray([minx, miny, maxx, maxy]) │ │ │ │ │ + }); │ │ │ │ │ + case "INTERSECTS": │ │ │ │ │ + var value = buildTree(), │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: property, │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + case "WITHIN": │ │ │ │ │ + var value = buildTree(), │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.WITHIN, │ │ │ │ │ + property: property, │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + case "CONTAINS": │ │ │ │ │ + var value = buildTree(), │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.CONTAINS, │ │ │ │ │ + property: property, │ │ │ │ │ + value: value │ │ │ │ │ + }); │ │ │ │ │ + case "DWITHIN": │ │ │ │ │ + var distance = buildTree(), │ │ │ │ │ + value = buildTree(), │ │ │ │ │ + property = buildTree(); │ │ │ │ │ + return new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ + value: value, │ │ │ │ │ + property: property, │ │ │ │ │ + distance: Number(distance) │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + case "GEOMETRY": │ │ │ │ │ + return OpenLayers.Geometry.fromWKT(tok.text); │ │ │ │ │ + default: │ │ │ │ │ + return tok.text │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return { │ │ │ │ │ - geometry: geometry, │ │ │ │ │ - bounds: bounds │ │ │ │ │ + var result = buildTree(); │ │ │ │ │ + if (postfix.length > 0) { │ │ │ │ │ + var msg = "Remaining tokens after building AST: \n"; │ │ │ │ │ + for (var i = postfix.length - 1; i >= 0; i--) { │ │ │ │ │ + msg += postfix[i].type + ": " + postfix[i].text + "\n" │ │ │ │ │ + } │ │ │ │ │ + throw new Error(msg) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSGetFeatureInfo" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.CSWGetDomain = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, OpenLayers.Format.CSWGetDomain.DEFAULTS); │ │ │ │ │ - var cls = OpenLayers.Format.CSWGetDomain["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported CSWGetDomain version: " + options.version │ │ │ │ │ - } │ │ │ │ │ - return new cls(options) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Format.CSWGetDomain.DEFAULTS = { │ │ │ │ │ - version: "2.0.2" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Format.QueryStringFilter = function() { │ │ │ │ │ - var cmpToStr = {}; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.EQUAL_TO] = "eq"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.NOT_EQUAL_TO] = "ne"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.LESS_THAN] = "lt"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO] = "lte"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.GREATER_THAN] = "gt"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO] = "gte"; │ │ │ │ │ - cmpToStr[OpenLayers.Filter.Comparison.LIKE] = "ilike"; │ │ │ │ │ - │ │ │ │ │ - function regex2value(value) { │ │ │ │ │ - value = value.replace(/%/g, "\\%"); │ │ │ │ │ - value = value.replace(/\\\\\.(\*)?/g, function($0, $1) { │ │ │ │ │ - return $1 ? $0 : "\\\\_" │ │ │ │ │ - }); │ │ │ │ │ - value = value.replace(/\\\\\.\*/g, "\\\\%"); │ │ │ │ │ - value = value.replace(/(\\)?\.(\*)?/g, function($0, $1, $2) { │ │ │ │ │ - return $1 || $2 ? $0 : "_" │ │ │ │ │ - }); │ │ │ │ │ - value = value.replace(/(\\)?\.\*/g, function($0, $1) { │ │ │ │ │ - return $1 ? $0 : "%" │ │ │ │ │ - }); │ │ │ │ │ - value = value.replace(/\\\./g, "."); │ │ │ │ │ - value = value.replace(/(\\)?\\\*/g, function($0, $1) { │ │ │ │ │ - return $1 ? $0 : "*" │ │ │ │ │ - }); │ │ │ │ │ - return value │ │ │ │ │ + return result │ │ │ │ │ } │ │ │ │ │ return OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ - wildcarded: false, │ │ │ │ │ - srsInBBOX: false, │ │ │ │ │ - write: function(filter, params) { │ │ │ │ │ - params = params || {}; │ │ │ │ │ - var className = filter.CLASS_NAME; │ │ │ │ │ - var filterType = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ - switch (filterType) { │ │ │ │ │ - case "Spatial": │ │ │ │ │ + read: function(text) { │ │ │ │ │ + var result = buildAst(tokenize(text)); │ │ │ │ │ + if (this.keepData) { │ │ │ │ │ + this.data = result │ │ │ │ │ + } │ │ │ │ │ + return result │ │ │ │ │ + }, │ │ │ │ │ + write: function(filter) { │ │ │ │ │ + if (filter instanceof OpenLayers.Geometry) { │ │ │ │ │ + return filter.toString() │ │ │ │ │ + } │ │ │ │ │ + switch (filter.CLASS_NAME) { │ │ │ │ │ + case "OpenLayers.Filter.Spatial": │ │ │ │ │ switch (filter.type) { │ │ │ │ │ case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ - params.bbox = filter.value.toArray(); │ │ │ │ │ - if (this.srsInBBOX && filter.projection) { │ │ │ │ │ - params.bbox.push(filter.projection.getCode()) │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ + return "BBOX(" + filter.property + "," + filter.value.toBBOX() + ")"; │ │ │ │ │ case OpenLayers.Filter.Spatial.DWITHIN: │ │ │ │ │ - params.tolerance = filter.distance; │ │ │ │ │ + return "DWITHIN(" + filter.property + ", " + this.write(filter.value) + ", " + filter.distance + ")"; │ │ │ │ │ case OpenLayers.Filter.Spatial.WITHIN: │ │ │ │ │ - params.lon = filter.value.x; │ │ │ │ │ - params.lat = filter.value.y; │ │ │ │ │ - break; │ │ │ │ │ + return "WITHIN(" + filter.property + ", " + this.write(filter.value) + ")"; │ │ │ │ │ + case OpenLayers.Filter.Spatial.INTERSECTS: │ │ │ │ │ + return "INTERSECTS(" + filter.property + ", " + this.write(filter.value) + ")"; │ │ │ │ │ + case OpenLayers.Filter.Spatial.CONTAINS: │ │ │ │ │ + return "CONTAINS(" + filter.property + ", " + this.write(filter.value) + ")"; │ │ │ │ │ default: │ │ │ │ │ - OpenLayers.Console.warn("Unknown spatial filter type " + filter.type) │ │ │ │ │ + throw new Error("Unknown spatial filter type: " + filter.type) │ │ │ │ │ } │ │ │ │ │ - break; │ │ │ │ │ - case "Comparison": │ │ │ │ │ - var op = cmpToStr[filter.type]; │ │ │ │ │ - if (op !== undefined) { │ │ │ │ │ - var value = filter.value; │ │ │ │ │ - if (filter.type == OpenLayers.Filter.Comparison.LIKE) { │ │ │ │ │ - value = regex2value(value); │ │ │ │ │ - if (this.wildcarded) { │ │ │ │ │ - value = "%" + value + "%" │ │ │ │ │ + case "OpenLayers.Filter.Logical": │ │ │ │ │ + if (filter.type == OpenLayers.Filter.Logical.NOT) { │ │ │ │ │ + return "NOT (" + this.write(filter.filters[0]) + ")" │ │ │ │ │ + } else { │ │ │ │ │ + var res = "("; │ │ │ │ │ + var first = true; │ │ │ │ │ + for (var i = 0; i < filter.filters.length; i++) { │ │ │ │ │ + if (first) { │ │ │ │ │ + first = false │ │ │ │ │ + } else { │ │ │ │ │ + res += ") " + logicalReverse[filter.type] + " (" │ │ │ │ │ } │ │ │ │ │ + res += this.write(filter.filters[i]) │ │ │ │ │ } │ │ │ │ │ - params[filter.property + "__" + op] = value; │ │ │ │ │ - params.queryable = params.queryable || []; │ │ │ │ │ - params.queryable.push(filter.property) │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Console.warn("Unknown comparison filter type " + filter.type) │ │ │ │ │ + return res + ")" │ │ │ │ │ } │ │ │ │ │ - break; │ │ │ │ │ - case "Logical": │ │ │ │ │ - if (filter.type === OpenLayers.Filter.Logical.AND) { │ │ │ │ │ - for (var i = 0, len = filter.filters.length; i < len; i++) { │ │ │ │ │ - params = this.write(filter.filters[i], params) │ │ │ │ │ - } │ │ │ │ │ + case "OpenLayers.Filter.Comparison": │ │ │ │ │ + if (filter.type == OpenLayers.Filter.Comparison.BETWEEN) { │ │ │ │ │ + return filter.property + " BETWEEN " + this.write(filter.lowerBoundary) + " AND " + this.write(filter.upperBoundary) │ │ │ │ │ } else { │ │ │ │ │ - OpenLayers.Console.warn("Unsupported logical filter type " + filter.type) │ │ │ │ │ + return filter.value !== null ? filter.property + " " + operatorReverse[filter.type] + " " + this.write(filter.value) : filter.property + " " + operatorReverse[filter.type] │ │ │ │ │ + } │ │ │ │ │ + case undefined: │ │ │ │ │ + if (typeof filter === "string") { │ │ │ │ │ + return "'" + filter.replace(/'/g, "''") + "'" │ │ │ │ │ + } else if (typeof filter === "number") { │ │ │ │ │ + return String(filter) │ │ │ │ │ } │ │ │ │ │ - break; │ │ │ │ │ default: │ │ │ │ │ - OpenLayers.Console.warn("Unknown filter type " + filterType) │ │ │ │ │ + throw new Error("Can't encode: " + filter.CLASS_NAME + " " + filter) │ │ │ │ │ } │ │ │ │ │ - return params │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.QueryStringFilter" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.CQL" │ │ │ │ │ }) │ │ │ │ │ }(); │ │ │ │ │ +OpenLayers.Format.WMSDescribeLayer = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + defaultVersion: "1.1.1", │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSDescribeLayer" │ │ │ │ │ +}); │ │ │ │ │ OpenLayers.Format.KML = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ namespaces: { │ │ │ │ │ kml: "http://www.opengis.net/kml/2.2", │ │ │ │ │ gx: "http://www.google.com/kml/ext/2.2" │ │ │ │ │ }, │ │ │ │ │ kmlns: "http://earth.google.com/kml/2.0", │ │ │ │ │ placemarksDesc: "No description available", │ │ │ │ │ @@ -24144,197 +26978,109 @@ │ │ │ │ │ return null │ │ │ │ │ } else { │ │ │ │ │ return extendedData │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.KML" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.WCSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - defaultVersion: "1.1.0", │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WCSCapabilities" │ │ │ │ │ +OpenLayers.Format.SLD = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + profile: null, │ │ │ │ │ + defaultVersion: "1.0.0", │ │ │ │ │ + stringifyOutput: true, │ │ │ │ │ + namedLayersAsArray: false, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SLD" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.GPX = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - defaultDesc: "No description available", │ │ │ │ │ - extractWaypoints: true, │ │ │ │ │ - extractTracks: true, │ │ │ │ │ - extractRoutes: true, │ │ │ │ │ - extractAttributes: true, │ │ │ │ │ - namespaces: { │ │ │ │ │ - gpx: "http://www.topografix.com/GPX/1/1", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ - }, │ │ │ │ │ - schemaLocation: "http://www.topografix.com/GPX/1/1 http://www.topografix.com/GPX/1/1/gpx.xsd", │ │ │ │ │ - creator: "OpenLayers", │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.externalProjection = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ - read: function(doc) { │ │ │ │ │ - if (typeof doc == "string") { │ │ │ │ │ - doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]) │ │ │ │ │ - } │ │ │ │ │ - var features = []; │ │ │ │ │ - if (this.extractTracks) { │ │ │ │ │ - var tracks = doc.getElementsByTagName("trk"); │ │ │ │ │ - for (var i = 0, len = tracks.length; i < len; i++) { │ │ │ │ │ - var attrs = {}; │ │ │ │ │ - if (this.extractAttributes) { │ │ │ │ │ - attrs = this.parseAttributes(tracks[i]) │ │ │ │ │ - } │ │ │ │ │ - var segs = this.getElementsByTagNameNS(tracks[i], tracks[i].namespaceURI, "trkseg"); │ │ │ │ │ - for (var j = 0, seglen = segs.length; j < seglen; j++) { │ │ │ │ │ - var track = this.extractSegment(segs[j], "trkpt"); │ │ │ │ │ - features.push(new OpenLayers.Feature.Vector(track, attrs)) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.extractRoutes) { │ │ │ │ │ - var routes = doc.getElementsByTagName("rte"); │ │ │ │ │ - for (var k = 0, klen = routes.length; k < klen; k++) { │ │ │ │ │ - var attrs = {}; │ │ │ │ │ - if (this.extractAttributes) { │ │ │ │ │ - attrs = this.parseAttributes(routes[k]) │ │ │ │ │ - } │ │ │ │ │ - var route = this.extractSegment(routes[k], "rtept"); │ │ │ │ │ - features.push(new OpenLayers.Feature.Vector(route, attrs)) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.extractWaypoints) { │ │ │ │ │ - var waypoints = doc.getElementsByTagName("wpt"); │ │ │ │ │ - for (var l = 0, len = waypoints.length; l < len; l++) { │ │ │ │ │ - var attrs = {}; │ │ │ │ │ - if (this.extractAttributes) { │ │ │ │ │ - attrs = this.parseAttributes(waypoints[l]) │ │ │ │ │ - } │ │ │ │ │ - var wpt = new OpenLayers.Geometry.Point(waypoints[l].getAttribute("lon"), waypoints[l].getAttribute("lat")); │ │ │ │ │ - features.push(new OpenLayers.Feature.Vector(wpt, attrs)) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - for (var g = 0, featLength = features.length; g < featLength; g++) { │ │ │ │ │ - features[g].geometry.transform(this.externalProjection, this.internalProjection) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return features │ │ │ │ │ - }, │ │ │ │ │ - extractSegment: function(segment, segmentType) { │ │ │ │ │ - var points = this.getElementsByTagNameNS(segment, segment.namespaceURI, segmentType); │ │ │ │ │ - var point_features = []; │ │ │ │ │ - for (var i = 0, len = points.length; i < len; i++) { │ │ │ │ │ - point_features.push(new OpenLayers.Geometry.Point(points[i].getAttribute("lon"), points[i].getAttribute("lat"))) │ │ │ │ │ - } │ │ │ │ │ - return new OpenLayers.Geometry.LineString(point_features) │ │ │ │ │ - }, │ │ │ │ │ - parseAttributes: function(node) { │ │ │ │ │ - var attributes = {}; │ │ │ │ │ - var attrNode = node.firstChild, │ │ │ │ │ - value, name; │ │ │ │ │ - while (attrNode) { │ │ │ │ │ - if (attrNode.nodeType == 1 && attrNode.firstChild) { │ │ │ │ │ - value = attrNode.firstChild; │ │ │ │ │ - if (value.nodeType == 3 || value.nodeType == 4) { │ │ │ │ │ - name = attrNode.prefix ? attrNode.nodeName.split(":")[1] : attrNode.nodeName; │ │ │ │ │ - if (name != "trkseg" && name != "rtept") { │ │ │ │ │ - attributes[name] = value.nodeValue │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - attrNode = attrNode.nextSibling │ │ │ │ │ - } │ │ │ │ │ - return attributes │ │ │ │ │ - }, │ │ │ │ │ - write: function(features, metadata) { │ │ │ │ │ - features = OpenLayers.Util.isArray(features) ? features : [features]; │ │ │ │ │ - var gpx = this.createElementNS(this.namespaces.gpx, "gpx"); │ │ │ │ │ - gpx.setAttribute("version", "1.1"); │ │ │ │ │ - gpx.setAttribute("creator", this.creator); │ │ │ │ │ - this.setAttributes(gpx, { │ │ │ │ │ - "xsi:schemaLocation": this.schemaLocation │ │ │ │ │ +OpenLayers.Format.QueryStringFilter = function() { │ │ │ │ │ + var cmpToStr = {}; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.EQUAL_TO] = "eq"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.NOT_EQUAL_TO] = "ne"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.LESS_THAN] = "lt"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO] = "lte"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.GREATER_THAN] = "gt"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO] = "gte"; │ │ │ │ │ + cmpToStr[OpenLayers.Filter.Comparison.LIKE] = "ilike"; │ │ │ │ │ + │ │ │ │ │ + function regex2value(value) { │ │ │ │ │ + value = value.replace(/%/g, "\\%"); │ │ │ │ │ + value = value.replace(/\\\\\.(\*)?/g, function($0, $1) { │ │ │ │ │ + return $1 ? $0 : "\\\\_" │ │ │ │ │ }); │ │ │ │ │ - if (metadata && typeof metadata == "object") { │ │ │ │ │ - gpx.appendChild(this.buildMetadataNode(metadata)) │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ - gpx.appendChild(this.buildFeatureNode(features[i])) │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [gpx]) │ │ │ │ │ - }, │ │ │ │ │ - buildMetadataNode: function(metadata) { │ │ │ │ │ - var types = ["name", "desc", "author"], │ │ │ │ │ - node = this.createElementNS(this.namespaces.gpx, "metadata"); │ │ │ │ │ - for (var i = 0; i < types.length; i++) { │ │ │ │ │ - var type = types[i]; │ │ │ │ │ - if (metadata[type]) { │ │ │ │ │ - var n = this.createElementNS(this.namespaces.gpx, type); │ │ │ │ │ - n.appendChild(this.createTextNode(metadata[type])); │ │ │ │ │ - node.appendChild(n) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - buildFeatureNode: function(feature) { │ │ │ │ │ - var geometry = feature.geometry; │ │ │ │ │ - geometry = geometry.clone(); │ │ │ │ │ - if (this.internalProjection && this.externalProjection) { │ │ │ │ │ - geometry.transform(this.internalProjection, this.externalProjection) │ │ │ │ │ - } │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - var wpt = this.buildWptNode(geometry); │ │ │ │ │ - this.appendAttributesNode(wpt, feature); │ │ │ │ │ - return wpt │ │ │ │ │ - } else { │ │ │ │ │ - var trkNode = this.createElementNS(this.namespaces.gpx, "trk"); │ │ │ │ │ - this.appendAttributesNode(trkNode, feature); │ │ │ │ │ - var trkSegNodes = this.buildTrkSegNode(geometry); │ │ │ │ │ - trkSegNodes = OpenLayers.Util.isArray(trkSegNodes) ? trkSegNodes : [trkSegNodes]; │ │ │ │ │ - for (var i = 0, len = trkSegNodes.length; i < len; i++) { │ │ │ │ │ - trkNode.appendChild(trkSegNodes[i]) │ │ │ │ │ - } │ │ │ │ │ - return trkNode │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - buildTrkSegNode: function(geometry) { │ │ │ │ │ - var node, i, len, point, nodes; │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.LineString" || geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ - node = this.createElementNS(this.namespaces.gpx, "trkseg"); │ │ │ │ │ - for (i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - point = geometry.components[i]; │ │ │ │ │ - node.appendChild(this.buildTrkPtNode(point)) │ │ │ │ │ - } │ │ │ │ │ - return node │ │ │ │ │ - } else { │ │ │ │ │ - nodes = []; │ │ │ │ │ - for (i = 0, len = geometry.components.length; i < len; i++) { │ │ │ │ │ - nodes.push(this.buildTrkSegNode(geometry.components[i])) │ │ │ │ │ + value = value.replace(/\\\\\.\*/g, "\\\\%"); │ │ │ │ │ + value = value.replace(/(\\)?\.(\*)?/g, function($0, $1, $2) { │ │ │ │ │ + return $1 || $2 ? $0 : "_" │ │ │ │ │ + }); │ │ │ │ │ + value = value.replace(/(\\)?\.\*/g, function($0, $1) { │ │ │ │ │ + return $1 ? $0 : "%" │ │ │ │ │ + }); │ │ │ │ │ + value = value.replace(/\\\./g, "."); │ │ │ │ │ + value = value.replace(/(\\)?\\\*/g, function($0, $1) { │ │ │ │ │ + return $1 ? $0 : "*" │ │ │ │ │ + }); │ │ │ │ │ + return value │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ + wildcarded: false, │ │ │ │ │ + srsInBBOX: false, │ │ │ │ │ + write: function(filter, params) { │ │ │ │ │ + params = params || {}; │ │ │ │ │ + var className = filter.CLASS_NAME; │ │ │ │ │ + var filterType = className.substring(className.lastIndexOf(".") + 1); │ │ │ │ │ + switch (filterType) { │ │ │ │ │ + case "Spatial": │ │ │ │ │ + switch (filter.type) { │ │ │ │ │ + case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ + params.bbox = filter.value.toArray(); │ │ │ │ │ + if (this.srsInBBOX && filter.projection) { │ │ │ │ │ + params.bbox.push(filter.projection.getCode()) │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Filter.Spatial.DWITHIN: │ │ │ │ │ + params.tolerance = filter.distance; │ │ │ │ │ + case OpenLayers.Filter.Spatial.WITHIN: │ │ │ │ │ + params.lon = filter.value.x; │ │ │ │ │ + params.lat = filter.value.y; │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + OpenLayers.Console.warn("Unknown spatial filter type " + filter.type) │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "Comparison": │ │ │ │ │ + var op = cmpToStr[filter.type]; │ │ │ │ │ + if (op !== undefined) { │ │ │ │ │ + var value = filter.value; │ │ │ │ │ + if (filter.type == OpenLayers.Filter.Comparison.LIKE) { │ │ │ │ │ + value = regex2value(value); │ │ │ │ │ + if (this.wildcarded) { │ │ │ │ │ + value = "%" + value + "%" │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + params[filter.property + "__" + op] = value; │ │ │ │ │ + params.queryable = params.queryable || []; │ │ │ │ │ + params.queryable.push(filter.property) │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Console.warn("Unknown comparison filter type " + filter.type) │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + case "Logical": │ │ │ │ │ + if (filter.type === OpenLayers.Filter.Logical.AND) { │ │ │ │ │ + for (var i = 0, len = filter.filters.length; i < len; i++) { │ │ │ │ │ + params = this.write(filter.filters[i], params) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Console.warn("Unsupported logical filter type " + filter.type) │ │ │ │ │ + } │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + OpenLayers.Console.warn("Unknown filter type " + filterType) │ │ │ │ │ } │ │ │ │ │ - return nodes │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - buildTrkPtNode: function(point) { │ │ │ │ │ - var node = this.createElementNS(this.namespaces.gpx, "trkpt"); │ │ │ │ │ - node.setAttribute("lon", point.x); │ │ │ │ │ - node.setAttribute("lat", point.y); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - buildWptNode: function(geometry) { │ │ │ │ │ - var node = this.createElementNS(this.namespaces.gpx, "wpt"); │ │ │ │ │ - node.setAttribute("lon", geometry.x); │ │ │ │ │ - node.setAttribute("lat", geometry.y); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - appendAttributesNode: function(node, feature) { │ │ │ │ │ - var name = this.createElementNS(this.namespaces.gpx, "name"); │ │ │ │ │ - name.appendChild(this.createTextNode(feature.attributes.name || feature.id)); │ │ │ │ │ - node.appendChild(name); │ │ │ │ │ - var desc = this.createElementNS(this.namespaces.gpx, "desc"); │ │ │ │ │ - desc.appendChild(this.createTextNode(feature.attributes.description || this.defaultDesc)); │ │ │ │ │ - node.appendChild(desc) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.GPX" │ │ │ │ │ -}); │ │ │ │ │ + return params │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.QueryStringFilter" │ │ │ │ │ + }) │ │ │ │ │ +}(); │ │ │ │ │ OpenLayers.Format.GeoRSS = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ rssns: "http://backend.userland.com/rss2", │ │ │ │ │ featureNS: "http://mapserver.gis.umn.edu/mapserver", │ │ │ │ │ georssns: "http://www.georss.org/georss", │ │ │ │ │ geons: "http://www.w3.org/2003/01/geo/wgs84_pos#", │ │ │ │ │ featureTitle: "Untitled", │ │ │ │ │ featureDescription: "No Description", │ │ │ │ │ @@ -24534,1274 +27280,747 @@ │ │ │ │ │ } else { │ │ │ │ │ path = geometry.y + " " + geometry.x │ │ │ │ │ } │ │ │ │ │ return this.createTextNode(path) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.GeoRSS" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.WMSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - defaultVersion: "1.1.1", │ │ │ │ │ - profile: null, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities" │ │ │ │ │ +OpenLayers.Format.WFSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + defaultVersion: "1.1.0", │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSCapabilities" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.WMTSCapabilities = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - defaultVersion: "1.0.0", │ │ │ │ │ - yx: { │ │ │ │ │ - "urn:ogc:def:crs:EPSG::4326": true │ │ │ │ │ +OpenLayers.Format.Atom = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + namespaces: { │ │ │ │ │ + atom: "http://www.w3.org/2005/Atom", │ │ │ │ │ + georss: "http://www.georss.org/georss" │ │ │ │ │ }, │ │ │ │ │ - createLayer: function(capabilities, config) { │ │ │ │ │ - var layer; │ │ │ │ │ - if (!("layer" in config)) { │ │ │ │ │ - throw new Error("Missing property 'layer' in configuration.") │ │ │ │ │ + feedTitle: "untitled", │ │ │ │ │ + defaultEntryTitle: "untitled", │ │ │ │ │ + gmlParser: null, │ │ │ │ │ + xy: false, │ │ │ │ │ + read: function(doc) { │ │ │ │ │ + if (typeof doc == "string") { │ │ │ │ │ + doc = OpenLayers.Format.XML.prototype.read.apply(this, [doc]) │ │ │ │ │ } │ │ │ │ │ - var contents = capabilities.contents; │ │ │ │ │ - var layers = contents.layers; │ │ │ │ │ - var layerDef; │ │ │ │ │ - for (var i = 0, ii = contents.layers.length; i < ii; ++i) { │ │ │ │ │ - if (contents.layers[i].identifier === config.layer) { │ │ │ │ │ - layerDef = contents.layers[i]; │ │ │ │ │ - break │ │ │ │ │ + return this.parseFeatures(doc) │ │ │ │ │ + }, │ │ │ │ │ + write: function(features) { │ │ │ │ │ + var doc; │ │ │ │ │ + if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ + doc = this.createElementNSPlus("atom:feed"); │ │ │ │ │ + doc.appendChild(this.createElementNSPlus("atom:title", { │ │ │ │ │ + value: this.feedTitle │ │ │ │ │ + })); │ │ │ │ │ + for (var i = 0, ii = features.length; i < ii; i++) { │ │ │ │ │ + doc.appendChild(this.buildEntryNode(features[i])) │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + doc = this.buildEntryNode(features) │ │ │ │ │ } │ │ │ │ │ - if (!layerDef) { │ │ │ │ │ - throw new Error("Layer not found") │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [doc]) │ │ │ │ │ + }, │ │ │ │ │ + buildContentNode: function(content) { │ │ │ │ │ + var node = this.createElementNSPlus("atom:content", { │ │ │ │ │ + attributes: { │ │ │ │ │ + type: content.type || null │ │ │ │ │ + } │ │ │ │ │ + }); │ │ │ │ │ + if (content.src) { │ │ │ │ │ + node.setAttribute("src", content.src) │ │ │ │ │ + } else { │ │ │ │ │ + if (content.type == "text" || content.type == null) { │ │ │ │ │ + node.appendChild(this.createTextNode(content.value)) │ │ │ │ │ + } else if (content.type == "html") { │ │ │ │ │ + if (typeof content.value != "string") { │ │ │ │ │ + throw "HTML content must be in form of an escaped string" │ │ │ │ │ + } │ │ │ │ │ + node.appendChild(this.createTextNode(content.value)) │ │ │ │ │ + } else if (content.type == "xhtml") { │ │ │ │ │ + node.appendChild(content.value) │ │ │ │ │ + } else if (content.type == "xhtml" || content.type.match(/(\+|\/)xml$/)) { │ │ │ │ │ + node.appendChild(content.value) │ │ │ │ │ + } else { │ │ │ │ │ + node.appendChild(this.createTextNode(content.value)) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var format = config.format; │ │ │ │ │ - if (!format && layerDef.formats && layerDef.formats.length) { │ │ │ │ │ - format = layerDef.formats[0] │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + buildEntryNode: function(feature) { │ │ │ │ │ + var attrib = feature.attributes; │ │ │ │ │ + var atomAttrib = attrib.atom || {}; │ │ │ │ │ + var entryNode = this.createElementNSPlus("atom:entry"); │ │ │ │ │ + if (atomAttrib.authors) { │ │ │ │ │ + var authors = OpenLayers.Util.isArray(atomAttrib.authors) ? atomAttrib.authors : [atomAttrib.authors]; │ │ │ │ │ + for (var i = 0, ii = authors.length; i < ii; i++) { │ │ │ │ │ + entryNode.appendChild(this.buildPersonConstructNode("author", authors[i])) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var matrixSet; │ │ │ │ │ - if (config.matrixSet) { │ │ │ │ │ - matrixSet = contents.tileMatrixSets[config.matrixSet] │ │ │ │ │ - } else if (layerDef.tileMatrixSetLinks.length >= 1) { │ │ │ │ │ - matrixSet = contents.tileMatrixSets[layerDef.tileMatrixSetLinks[0].tileMatrixSet] │ │ │ │ │ + if (atomAttrib.categories) { │ │ │ │ │ + var categories = OpenLayers.Util.isArray(atomAttrib.categories) ? atomAttrib.categories : [atomAttrib.categories]; │ │ │ │ │ + var category; │ │ │ │ │ + for (var i = 0, ii = categories.length; i < ii; i++) { │ │ │ │ │ + category = categories[i]; │ │ │ │ │ + entryNode.appendChild(this.createElementNSPlus("atom:category", { │ │ │ │ │ + attributes: { │ │ │ │ │ + term: category.term, │ │ │ │ │ + scheme: category.scheme || null, │ │ │ │ │ + label: category.label || null │ │ │ │ │ + } │ │ │ │ │ + })) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (!matrixSet) { │ │ │ │ │ - throw new Error("matrixSet not found") │ │ │ │ │ + if (atomAttrib.content) { │ │ │ │ │ + entryNode.appendChild(this.buildContentNode(atomAttrib.content)) │ │ │ │ │ } │ │ │ │ │ - var style; │ │ │ │ │ - for (var i = 0, ii = layerDef.styles.length; i < ii; ++i) { │ │ │ │ │ - style = layerDef.styles[i]; │ │ │ │ │ - if (style.isDefault) { │ │ │ │ │ - break │ │ │ │ │ + if (atomAttrib.contributors) { │ │ │ │ │ + var contributors = OpenLayers.Util.isArray(atomAttrib.contributors) ? atomAttrib.contributors : [atomAttrib.contributors]; │ │ │ │ │ + for (var i = 0, ii = contributors.length; i < ii; i++) { │ │ │ │ │ + entryNode.appendChild(this.buildPersonConstructNode("contributor", contributors[i])) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var requestEncoding = config.requestEncoding; │ │ │ │ │ - if (!requestEncoding) { │ │ │ │ │ - requestEncoding = "KVP"; │ │ │ │ │ - if (capabilities.operationsMetadata.GetTile.dcp.http) { │ │ │ │ │ - var http = capabilities.operationsMetadata.GetTile.dcp.http; │ │ │ │ │ - if (http.get[0].constraints) { │ │ │ │ │ - var constraints = http.get[0].constraints; │ │ │ │ │ - var allowedValues = constraints.GetEncoding.allowedValues; │ │ │ │ │ - if (!allowedValues.KVP && (allowedValues.REST || allowedValues.RESTful)) { │ │ │ │ │ - requestEncoding = "REST" │ │ │ │ │ + if (feature.fid) { │ │ │ │ │ + entryNode.appendChild(this.createElementNSPlus("atom:id", { │ │ │ │ │ + value: feature.fid │ │ │ │ │ + })) │ │ │ │ │ + } │ │ │ │ │ + if (atomAttrib.links) { │ │ │ │ │ + var links = OpenLayers.Util.isArray(atomAttrib.links) ? atomAttrib.links : [atomAttrib.links]; │ │ │ │ │ + var link; │ │ │ │ │ + for (var i = 0, ii = links.length; i < ii; i++) { │ │ │ │ │ + link = links[i]; │ │ │ │ │ + entryNode.appendChild(this.createElementNSPlus("atom:link", { │ │ │ │ │ + attributes: { │ │ │ │ │ + href: link.href, │ │ │ │ │ + rel: link.rel || null, │ │ │ │ │ + type: link.type || null, │ │ │ │ │ + hreflang: link.hreflang || null, │ │ │ │ │ + title: link.title || null, │ │ │ │ │ + length: link.length || null │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ + })) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var dimensions = []; │ │ │ │ │ - var params = config.params || {}; │ │ │ │ │ - delete config.params; │ │ │ │ │ - for (var id = 0, ld = layerDef.dimensions.length; id < ld; id++) { │ │ │ │ │ - var dimension = layerDef.dimensions[id]; │ │ │ │ │ - dimensions.push(dimension.identifier); │ │ │ │ │ - if (!params.hasOwnProperty(dimension.identifier)) { │ │ │ │ │ - params[dimension.identifier] = dimension["default"] │ │ │ │ │ - } │ │ │ │ │ + if (atomAttrib.published) { │ │ │ │ │ + entryNode.appendChild(this.createElementNSPlus("atom:published", { │ │ │ │ │ + value: atomAttrib.published │ │ │ │ │ + })) │ │ │ │ │ } │ │ │ │ │ - var projection = config.projection || matrixSet.supportedCRS.replace(/urn:ogc:def:crs:(\w+):(.*:)?(\w+)$/, "$1:$3"); │ │ │ │ │ - var units = config.units || (projection === "EPSG:4326" ? "degrees" : "m"); │ │ │ │ │ - var resolutions = []; │ │ │ │ │ - for (var mid in matrixSet.matrixIds) { │ │ │ │ │ - if (matrixSet.matrixIds.hasOwnProperty(mid)) { │ │ │ │ │ - resolutions.push(matrixSet.matrixIds[mid].scaleDenominator * 28e-5 / OpenLayers.METERS_PER_INCH / OpenLayers.INCHES_PER_UNIT[units]) │ │ │ │ │ - } │ │ │ │ │ + if (atomAttrib.rights) { │ │ │ │ │ + entryNode.appendChild(this.createElementNSPlus("atom:rights", { │ │ │ │ │ + value: atomAttrib.rights │ │ │ │ │ + })) │ │ │ │ │ } │ │ │ │ │ - var url; │ │ │ │ │ - if (requestEncoding === "REST" && layerDef.resourceUrls) { │ │ │ │ │ - url = []; │ │ │ │ │ - var resourceUrls = layerDef.resourceUrls, │ │ │ │ │ - resourceUrl; │ │ │ │ │ - for (var t = 0, tt = layerDef.resourceUrls.length; t < tt; ++t) { │ │ │ │ │ - resourceUrl = layerDef.resourceUrls[t]; │ │ │ │ │ - if (resourceUrl.format === format && resourceUrl.resourceType === "tile") { │ │ │ │ │ - url.push(resourceUrl.template) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - var httpGet = capabilities.operationsMetadata.GetTile.dcp.http.get; │ │ │ │ │ - url = []; │ │ │ │ │ - var constraint; │ │ │ │ │ - for (var i = 0, ii = httpGet.length; i < ii; i++) { │ │ │ │ │ - constraint = httpGet[i].constraints; │ │ │ │ │ - if (!constraint || constraint && constraint.GetEncoding.allowedValues[requestEncoding]) { │ │ │ │ │ - url.push(httpGet[i].url) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + if (atomAttrib.summary || attrib.description) { │ │ │ │ │ + entryNode.appendChild(this.createElementNSPlus("atom:summary", { │ │ │ │ │ + value: atomAttrib.summary || attrib.description │ │ │ │ │ + })) │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Layer.WMTS(OpenLayers.Util.applyDefaults(config, { │ │ │ │ │ - url: url, │ │ │ │ │ - requestEncoding: requestEncoding, │ │ │ │ │ - name: layerDef.title, │ │ │ │ │ - style: style.identifier, │ │ │ │ │ - format: format, │ │ │ │ │ - matrixIds: matrixSet.matrixIds, │ │ │ │ │ - matrixSet: matrixSet.identifier, │ │ │ │ │ - projection: projection, │ │ │ │ │ - units: units, │ │ │ │ │ - resolutions: config.isBaseLayer === false ? undefined : resolutions, │ │ │ │ │ - serverResolutions: resolutions, │ │ │ │ │ - tileFullExtent: matrixSet.bounds, │ │ │ │ │ - dimensions: dimensions, │ │ │ │ │ - params: params │ │ │ │ │ - })) │ │ │ │ │ + entryNode.appendChild(this.createElementNSPlus("atom:title", { │ │ │ │ │ + value: atomAttrib.title || attrib.title || this.defaultEntryTitle │ │ │ │ │ + })); │ │ │ │ │ + if (atomAttrib.updated) { │ │ │ │ │ + entryNode.appendChild(this.createElementNSPlus("atom:updated", { │ │ │ │ │ + value: atomAttrib.updated │ │ │ │ │ + })) │ │ │ │ │ + } │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + var whereNode = this.createElementNSPlus("georss:where"); │ │ │ │ │ + whereNode.appendChild(this.buildGeometryNode(feature.geometry)); │ │ │ │ │ + entryNode.appendChild(whereNode) │ │ │ │ │ + } │ │ │ │ │ + return entryNode │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMTSCapabilities" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.CSWGetRecords = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, OpenLayers.Format.CSWGetRecords.DEFAULTS); │ │ │ │ │ - var cls = OpenLayers.Format.CSWGetRecords["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported CSWGetRecords version: " + options.version │ │ │ │ │ - } │ │ │ │ │ - return new cls(options) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Format.CSWGetRecords.DEFAULTS = { │ │ │ │ │ - version: "2.0.2" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Format.CQL = function() { │ │ │ │ │ - var tokens = ["PROPERTY", "COMPARISON", "VALUE", "LOGICAL"], │ │ │ │ │ - patterns = { │ │ │ │ │ - PROPERTY: /^[_a-zA-Z]\w*/, │ │ │ │ │ - COMPARISON: /^(=|<>|<=|<|>=|>|LIKE)/i, │ │ │ │ │ - IS_NULL: /^IS NULL/i, │ │ │ │ │ - COMMA: /^,/, │ │ │ │ │ - LOGICAL: /^(AND|OR)/i, │ │ │ │ │ - VALUE: /^('([^']|'')*'|\d+(\.\d*)?|\.\d+)/, │ │ │ │ │ - LPAREN: /^\(/, │ │ │ │ │ - RPAREN: /^\)/, │ │ │ │ │ - SPATIAL: /^(BBOX|INTERSECTS|DWITHIN|WITHIN|CONTAINS)/i, │ │ │ │ │ - NOT: /^NOT/i, │ │ │ │ │ - BETWEEN: /^BETWEEN/i, │ │ │ │ │ - GEOMETRY: function(text) { │ │ │ │ │ - var type = /^(POINT|LINESTRING|POLYGON|MULTIPOINT|MULTILINESTRING|MULTIPOLYGON|GEOMETRYCOLLECTION)/.exec(text); │ │ │ │ │ - if (type) { │ │ │ │ │ - var len = text.length; │ │ │ │ │ - var idx = text.indexOf("(", type[0].length); │ │ │ │ │ - if (idx > -1) { │ │ │ │ │ - var depth = 1; │ │ │ │ │ - while (idx < len && depth > 0) { │ │ │ │ │ - idx++; │ │ │ │ │ - switch (text.charAt(idx)) { │ │ │ │ │ - case "(": │ │ │ │ │ - depth++; │ │ │ │ │ - break; │ │ │ │ │ - case ")": │ │ │ │ │ - depth--; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return [text.substr(0, idx + 1)] │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - END: /^$/ │ │ │ │ │ - }, │ │ │ │ │ - follows = { │ │ │ │ │ - LPAREN: ["GEOMETRY", "SPATIAL", "PROPERTY", "VALUE", "LPAREN"], │ │ │ │ │ - RPAREN: ["NOT", "LOGICAL", "END", "RPAREN"], │ │ │ │ │ - PROPERTY: ["COMPARISON", "BETWEEN", "COMMA", "IS_NULL"], │ │ │ │ │ - BETWEEN: ["VALUE"], │ │ │ │ │ - IS_NULL: ["END"], │ │ │ │ │ - COMPARISON: ["VALUE"], │ │ │ │ │ - COMMA: ["GEOMETRY", "VALUE", "PROPERTY"], │ │ │ │ │ - VALUE: ["LOGICAL", "COMMA", "RPAREN", "END"], │ │ │ │ │ - SPATIAL: ["LPAREN"], │ │ │ │ │ - LOGICAL: ["NOT", "VALUE", "SPATIAL", "PROPERTY", "LPAREN"], │ │ │ │ │ - NOT: ["PROPERTY", "LPAREN"], │ │ │ │ │ - GEOMETRY: ["COMMA", "RPAREN"] │ │ │ │ │ - }, │ │ │ │ │ - operators = { │ │ │ │ │ - "=": OpenLayers.Filter.Comparison.EQUAL_TO, │ │ │ │ │ - "<>": OpenLayers.Filter.Comparison.NOT_EQUAL_TO, │ │ │ │ │ - "<": OpenLayers.Filter.Comparison.LESS_THAN, │ │ │ │ │ - "<=": OpenLayers.Filter.Comparison.LESS_THAN_OR_EQUAL_TO, │ │ │ │ │ - ">": OpenLayers.Filter.Comparison.GREATER_THAN, │ │ │ │ │ - ">=": OpenLayers.Filter.Comparison.GREATER_THAN_OR_EQUAL_TO, │ │ │ │ │ - LIKE: OpenLayers.Filter.Comparison.LIKE, │ │ │ │ │ - BETWEEN: OpenLayers.Filter.Comparison.BETWEEN, │ │ │ │ │ - "IS NULL": OpenLayers.Filter.Comparison.IS_NULL │ │ │ │ │ - }, │ │ │ │ │ - operatorReverse = {}, │ │ │ │ │ - logicals = { │ │ │ │ │ - AND: OpenLayers.Filter.Logical.AND, │ │ │ │ │ - OR: OpenLayers.Filter.Logical.OR │ │ │ │ │ - }, │ │ │ │ │ - logicalReverse = {}, │ │ │ │ │ - precedence = { │ │ │ │ │ - RPAREN: 3, │ │ │ │ │ - LOGICAL: 2, │ │ │ │ │ - COMPARISON: 1 │ │ │ │ │ - }; │ │ │ │ │ - var i; │ │ │ │ │ - for (i in operators) { │ │ │ │ │ - if (operators.hasOwnProperty(i)) { │ │ │ │ │ - operatorReverse[operators[i]] = i │ │ │ │ │ + initGmlParser: function() { │ │ │ │ │ + this.gmlParser = new OpenLayers.Format.GML.v3({ │ │ │ │ │ + xy: this.xy, │ │ │ │ │ + featureNS: "http://example.com#feature", │ │ │ │ │ + internalProjection: this.internalProjection, │ │ │ │ │ + externalProjection: this.externalProjection │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + buildGeometryNode: function(geometry) { │ │ │ │ │ + if (!this.gmlParser) { │ │ │ │ │ + this.initGmlParser() │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - for (i in logicals) { │ │ │ │ │ - if (logicals.hasOwnProperty(i)) { │ │ │ │ │ - logicalReverse[logicals[i]] = i │ │ │ │ │ + var node = this.gmlParser.writeNode("feature:_geometry", geometry); │ │ │ │ │ + return node.firstChild │ │ │ │ │ + }, │ │ │ │ │ + buildPersonConstructNode: function(name, value) { │ │ │ │ │ + var oNames = ["uri", "email"]; │ │ │ │ │ + var personNode = this.createElementNSPlus("atom:" + name); │ │ │ │ │ + personNode.appendChild(this.createElementNSPlus("atom:name", { │ │ │ │ │ + value: value.name │ │ │ │ │ + })); │ │ │ │ │ + for (var i = 0, ii = oNames.length; i < ii; i++) { │ │ │ │ │ + if (value[oNames[i]]) { │ │ │ │ │ + personNode.appendChild(this.createElementNSPlus("atom:" + oNames[i], { │ │ │ │ │ + value: value[oNames[i]] │ │ │ │ │ + })) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function tryToken(text, pattern) { │ │ │ │ │ - if (pattern instanceof RegExp) { │ │ │ │ │ - return pattern.exec(text) │ │ │ │ │ + return personNode │ │ │ │ │ + }, │ │ │ │ │ + getFirstChildValue: function(node, nsuri, name, def) { │ │ │ │ │ + var value; │ │ │ │ │ + var nodes = this.getElementsByTagNameNS(node, nsuri, name); │ │ │ │ │ + if (nodes && nodes.length > 0) { │ │ │ │ │ + value = this.getChildValue(nodes[0], def) │ │ │ │ │ } else { │ │ │ │ │ - return pattern(text) │ │ │ │ │ + value = def │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function nextToken(text, tokens) { │ │ │ │ │ - var i, token, len = tokens.length; │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - token = tokens[i]; │ │ │ │ │ - var pat = patterns[token]; │ │ │ │ │ - var matches = tryToken(text, pat); │ │ │ │ │ - if (matches) { │ │ │ │ │ - var match = matches[0]; │ │ │ │ │ - var remainder = text.substr(match.length).replace(/^\s*/, ""); │ │ │ │ │ - return { │ │ │ │ │ - type: token, │ │ │ │ │ - text: match, │ │ │ │ │ - remainder: remainder │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + return value │ │ │ │ │ + }, │ │ │ │ │ + parseFeature: function(node) { │ │ │ │ │ + var atomAttrib = {}; │ │ │ │ │ + var value = null; │ │ │ │ │ + var nodes = null; │ │ │ │ │ + var attval = null; │ │ │ │ │ + var atomns = this.namespaces.atom; │ │ │ │ │ + this.parsePersonConstructs(node, "author", atomAttrib); │ │ │ │ │ + nodes = this.getElementsByTagNameNS(node, atomns, "category"); │ │ │ │ │ + if (nodes.length > 0) { │ │ │ │ │ + atomAttrib.categories = [] │ │ │ │ │ } │ │ │ │ │ - var msg = "ERROR: In parsing: [" + text + "], expected one of: "; │ │ │ │ │ - for (i = 0; i < len; i++) { │ │ │ │ │ - token = tokens[i]; │ │ │ │ │ - msg += "\n " + token + ": " + patterns[token] │ │ │ │ │ + for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ + value = {}; │ │ │ │ │ + value.term = nodes[i].getAttribute("term"); │ │ │ │ │ + attval = nodes[i].getAttribute("scheme"); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value.scheme = attval │ │ │ │ │ + } │ │ │ │ │ + attval = nodes[i].getAttribute("label"); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value.label = attval │ │ │ │ │ + } │ │ │ │ │ + atomAttrib.categories.push(value) │ │ │ │ │ } │ │ │ │ │ - throw new Error(msg) │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function tokenize(text) { │ │ │ │ │ - var results = []; │ │ │ │ │ - var token, expect = ["NOT", "GEOMETRY", "SPATIAL", "PROPERTY", "LPAREN"]; │ │ │ │ │ - do { │ │ │ │ │ - token = nextToken(text, expect); │ │ │ │ │ - text = token.remainder; │ │ │ │ │ - expect = follows[token.type]; │ │ │ │ │ - if (token.type != "END" && !expect) { │ │ │ │ │ - throw new Error("No follows list for " + token.type) │ │ │ │ │ + nodes = this.getElementsByTagNameNS(node, atomns, "content"); │ │ │ │ │ + if (nodes.length > 0) { │ │ │ │ │ + value = {}; │ │ │ │ │ + attval = nodes[0].getAttribute("type"); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value.type = attval │ │ │ │ │ } │ │ │ │ │ - results.push(token) │ │ │ │ │ - } while (token.type != "END"); │ │ │ │ │ - return results │ │ │ │ │ - } │ │ │ │ │ - │ │ │ │ │ - function buildAst(tokens) { │ │ │ │ │ - var operatorStack = [], │ │ │ │ │ - postfix = []; │ │ │ │ │ - while (tokens.length) { │ │ │ │ │ - var tok = tokens.shift(); │ │ │ │ │ - switch (tok.type) { │ │ │ │ │ - case "PROPERTY": │ │ │ │ │ - case "GEOMETRY": │ │ │ │ │ - case "VALUE": │ │ │ │ │ - postfix.push(tok); │ │ │ │ │ - break; │ │ │ │ │ - case "COMPARISON": │ │ │ │ │ - case "BETWEEN": │ │ │ │ │ - case "IS_NULL": │ │ │ │ │ - case "LOGICAL": │ │ │ │ │ - var p = precedence[tok.type]; │ │ │ │ │ - while (operatorStack.length > 0 && precedence[operatorStack[operatorStack.length - 1].type] <= p) { │ │ │ │ │ - postfix.push(operatorStack.pop()) │ │ │ │ │ - } │ │ │ │ │ - operatorStack.push(tok); │ │ │ │ │ - break; │ │ │ │ │ - case "SPATIAL": │ │ │ │ │ - case "NOT": │ │ │ │ │ - case "LPAREN": │ │ │ │ │ - operatorStack.push(tok); │ │ │ │ │ - break; │ │ │ │ │ - case "RPAREN": │ │ │ │ │ - while (operatorStack.length > 0 && operatorStack[operatorStack.length - 1].type != "LPAREN") { │ │ │ │ │ - postfix.push(operatorStack.pop()) │ │ │ │ │ - } │ │ │ │ │ - operatorStack.pop(); │ │ │ │ │ - if (operatorStack.length > 0 && operatorStack[operatorStack.length - 1].type == "SPATIAL") { │ │ │ │ │ - postfix.push(operatorStack.pop()) │ │ │ │ │ - } │ │ │ │ │ - case "COMMA": │ │ │ │ │ - case "END": │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - throw new Error("Unknown token type " + tok.type) │ │ │ │ │ + attval = nodes[0].getAttribute("src"); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value.src = attval │ │ │ │ │ + } else { │ │ │ │ │ + if (value.type == "text" || value.type == "html" || value.type == null) { │ │ │ │ │ + value.value = this.getFirstChildValue(node, atomns, "content", null) │ │ │ │ │ + } else if (value.type == "xhtml" || value.type.match(/(\+|\/)xml$/)) { │ │ │ │ │ + value.value = this.getChildEl(nodes[0]) │ │ │ │ │ + } else { │ │ │ │ │ + value.value = this.getFirstChildValue(node, atomns, "content", null) │ │ │ │ │ + } │ │ │ │ │ + atomAttrib.content = value │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - while (operatorStack.length > 0) { │ │ │ │ │ - postfix.push(operatorStack.pop()) │ │ │ │ │ + this.parsePersonConstructs(node, "contributor", atomAttrib); │ │ │ │ │ + atomAttrib.id = this.getFirstChildValue(node, atomns, "id", null); │ │ │ │ │ + nodes = this.getElementsByTagNameNS(node, atomns, "link"); │ │ │ │ │ + if (nodes.length > 0) { │ │ │ │ │ + atomAttrib.links = new Array(nodes.length) │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - function buildTree() { │ │ │ │ │ - var tok = postfix.pop(); │ │ │ │ │ - switch (tok.type) { │ │ │ │ │ - case "LOGICAL": │ │ │ │ │ - var rhs = buildTree(), │ │ │ │ │ - lhs = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Logical({ │ │ │ │ │ - filters: [lhs, rhs], │ │ │ │ │ - type: logicals[tok.text.toUpperCase()] │ │ │ │ │ - }); │ │ │ │ │ - case "NOT": │ │ │ │ │ - var operand = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Logical({ │ │ │ │ │ - filters: [operand], │ │ │ │ │ - type: OpenLayers.Filter.Logical.NOT │ │ │ │ │ - }); │ │ │ │ │ - case "BETWEEN": │ │ │ │ │ - var min, max, property; │ │ │ │ │ - postfix.pop(); │ │ │ │ │ - max = buildTree(); │ │ │ │ │ - min = buildTree(); │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Comparison({ │ │ │ │ │ - property: property, │ │ │ │ │ - lowerBoundary: min, │ │ │ │ │ - upperBoundary: max, │ │ │ │ │ - type: OpenLayers.Filter.Comparison.BETWEEN │ │ │ │ │ - }); │ │ │ │ │ - case "COMPARISON": │ │ │ │ │ - var value = buildTree(), │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Comparison({ │ │ │ │ │ - property: property, │ │ │ │ │ - value: value, │ │ │ │ │ - type: operators[tok.text.toUpperCase()] │ │ │ │ │ - }); │ │ │ │ │ - case "IS_NULL": │ │ │ │ │ - var property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Comparison({ │ │ │ │ │ - property: property, │ │ │ │ │ - type: operators[tok.text.toUpperCase()] │ │ │ │ │ - }); │ │ │ │ │ - case "VALUE": │ │ │ │ │ - var match = tok.text.match(/^'(.*)'$/); │ │ │ │ │ - if (match) { │ │ │ │ │ - return match[1].replace(/''/g, "'") │ │ │ │ │ - } else { │ │ │ │ │ - return Number(tok.text) │ │ │ │ │ - } │ │ │ │ │ - case "SPATIAL": │ │ │ │ │ - switch (tok.text.toUpperCase()) { │ │ │ │ │ - case "BBOX": │ │ │ │ │ - var maxy = buildTree(), │ │ │ │ │ - maxx = buildTree(), │ │ │ │ │ - miny = buildTree(), │ │ │ │ │ - minx = buildTree(), │ │ │ │ │ - prop = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ - property: prop, │ │ │ │ │ - value: OpenLayers.Bounds.fromArray([minx, miny, maxx, maxy]) │ │ │ │ │ - }); │ │ │ │ │ - case "INTERSECTS": │ │ │ │ │ - var value = buildTree(), │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: property, │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - case "WITHIN": │ │ │ │ │ - var value = buildTree(), │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.WITHIN, │ │ │ │ │ - property: property, │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - case "CONTAINS": │ │ │ │ │ - var value = buildTree(), │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.CONTAINS, │ │ │ │ │ - property: property, │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - case "DWITHIN": │ │ │ │ │ - var distance = buildTree(), │ │ │ │ │ - value = buildTree(), │ │ │ │ │ - property = buildTree(); │ │ │ │ │ - return new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ - value: value, │ │ │ │ │ - property: property, │ │ │ │ │ - distance: Number(distance) │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - case "GEOMETRY": │ │ │ │ │ - return OpenLayers.Geometry.fromWKT(tok.text); │ │ │ │ │ - default: │ │ │ │ │ - return tok.text │ │ │ │ │ + var oAtts = ["rel", "type", "hreflang", "title", "length"]; │ │ │ │ │ + for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ + value = {}; │ │ │ │ │ + value.href = nodes[i].getAttribute("href"); │ │ │ │ │ + for (var j = 0, jj = oAtts.length; j < jj; j++) { │ │ │ │ │ + attval = nodes[i].getAttribute(oAtts[j]); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value[oAtts[j]] = attval │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + atomAttrib.links[i] = value │ │ │ │ │ } │ │ │ │ │ - var result = buildTree(); │ │ │ │ │ - if (postfix.length > 0) { │ │ │ │ │ - var msg = "Remaining tokens after building AST: \n"; │ │ │ │ │ - for (var i = postfix.length - 1; i >= 0; i--) { │ │ │ │ │ - msg += postfix[i].type + ": " + postfix[i].text + "\n" │ │ │ │ │ - } │ │ │ │ │ - throw new Error(msg) │ │ │ │ │ + value = this.getFirstChildValue(node, atomns, "published", null); │ │ │ │ │ + if (value) { │ │ │ │ │ + atomAttrib.published = value │ │ │ │ │ } │ │ │ │ │ - return result │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Class(OpenLayers.Format, { │ │ │ │ │ - read: function(text) { │ │ │ │ │ - var result = buildAst(tokenize(text)); │ │ │ │ │ - if (this.keepData) { │ │ │ │ │ - this.data = result │ │ │ │ │ - } │ │ │ │ │ - return result │ │ │ │ │ - }, │ │ │ │ │ - write: function(filter) { │ │ │ │ │ - if (filter instanceof OpenLayers.Geometry) { │ │ │ │ │ - return filter.toString() │ │ │ │ │ - } │ │ │ │ │ - switch (filter.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Filter.Spatial": │ │ │ │ │ - switch (filter.type) { │ │ │ │ │ - case OpenLayers.Filter.Spatial.BBOX: │ │ │ │ │ - return "BBOX(" + filter.property + "," + filter.value.toBBOX() + ")"; │ │ │ │ │ - case OpenLayers.Filter.Spatial.DWITHIN: │ │ │ │ │ - return "DWITHIN(" + filter.property + ", " + this.write(filter.value) + ", " + filter.distance + ")"; │ │ │ │ │ - case OpenLayers.Filter.Spatial.WITHIN: │ │ │ │ │ - return "WITHIN(" + filter.property + ", " + this.write(filter.value) + ")"; │ │ │ │ │ - case OpenLayers.Filter.Spatial.INTERSECTS: │ │ │ │ │ - return "INTERSECTS(" + filter.property + ", " + this.write(filter.value) + ")"; │ │ │ │ │ - case OpenLayers.Filter.Spatial.CONTAINS: │ │ │ │ │ - return "CONTAINS(" + filter.property + ", " + this.write(filter.value) + ")"; │ │ │ │ │ - default: │ │ │ │ │ - throw new Error("Unknown spatial filter type: " + filter.type) │ │ │ │ │ - } │ │ │ │ │ - case "OpenLayers.Filter.Logical": │ │ │ │ │ - if (filter.type == OpenLayers.Filter.Logical.NOT) { │ │ │ │ │ - return "NOT (" + this.write(filter.filters[0]) + ")" │ │ │ │ │ - } else { │ │ │ │ │ - var res = "("; │ │ │ │ │ - var first = true; │ │ │ │ │ - for (var i = 0; i < filter.filters.length; i++) { │ │ │ │ │ - if (first) { │ │ │ │ │ - first = false │ │ │ │ │ - } else { │ │ │ │ │ - res += ") " + logicalReverse[filter.type] + " (" │ │ │ │ │ - } │ │ │ │ │ - res += this.write(filter.filters[i]) │ │ │ │ │ - } │ │ │ │ │ - return res + ")" │ │ │ │ │ - } │ │ │ │ │ - case "OpenLayers.Filter.Comparison": │ │ │ │ │ - if (filter.type == OpenLayers.Filter.Comparison.BETWEEN) { │ │ │ │ │ - return filter.property + " BETWEEN " + this.write(filter.lowerBoundary) + " AND " + this.write(filter.upperBoundary) │ │ │ │ │ - } else { │ │ │ │ │ - return filter.value !== null ? filter.property + " " + operatorReverse[filter.type] + " " + this.write(filter.value) : filter.property + " " + operatorReverse[filter.type] │ │ │ │ │ - } │ │ │ │ │ - case undefined: │ │ │ │ │ - if (typeof filter === "string") { │ │ │ │ │ - return "'" + filter.replace(/'/g, "''") + "'" │ │ │ │ │ - } else if (typeof filter === "number") { │ │ │ │ │ - return String(filter) │ │ │ │ │ - } │ │ │ │ │ - default: │ │ │ │ │ - throw new Error("Can't encode: " + filter.CLASS_NAME + " " + filter) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.CQL" │ │ │ │ │ - }) │ │ │ │ │ -}(); │ │ │ │ │ -OpenLayers.Format.WFSDescribeFeatureType = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: /^\s*|\s*$/g │ │ │ │ │ + value = this.getFirstChildValue(node, atomns, "rights", null); │ │ │ │ │ + if (value) { │ │ │ │ │ + atomAttrib.rights = value │ │ │ │ │ + } │ │ │ │ │ + value = this.getFirstChildValue(node, atomns, "summary", null); │ │ │ │ │ + if (value) { │ │ │ │ │ + atomAttrib.summary = value │ │ │ │ │ + } │ │ │ │ │ + atomAttrib.title = this.getFirstChildValue(node, atomns, "title", null); │ │ │ │ │ + atomAttrib.updated = this.getFirstChildValue(node, atomns, "updated", null); │ │ │ │ │ + var featureAttrib = { │ │ │ │ │ + title: atomAttrib.title, │ │ │ │ │ + description: atomAttrib.summary, │ │ │ │ │ + atom: atomAttrib │ │ │ │ │ + }; │ │ │ │ │ + var geometry = this.parseLocations(node)[0]; │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry, featureAttrib); │ │ │ │ │ + feature.fid = atomAttrib.id; │ │ │ │ │ + return feature │ │ │ │ │ }, │ │ │ │ │ - namespaces: { │ │ │ │ │ - xsd: "http://www.w3.org/2001/XMLSchema" │ │ │ │ │ + parseFeatures: function(node) { │ │ │ │ │ + var features = []; │ │ │ │ │ + var entries = this.getElementsByTagNameNS(node, this.namespaces.atom, "entry"); │ │ │ │ │ + if (entries.length == 0) { │ │ │ │ │ + entries = [node] │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, ii = entries.length; i < ii; i++) { │ │ │ │ │ + features.push(this.parseFeature(entries[i])) │ │ │ │ │ + } │ │ │ │ │ + return features │ │ │ │ │ }, │ │ │ │ │ - readers: { │ │ │ │ │ - xsd: { │ │ │ │ │ - schema: function(node, obj) { │ │ │ │ │ - var complexTypes = []; │ │ │ │ │ - var customTypes = {}; │ │ │ │ │ - var schema = { │ │ │ │ │ - complexTypes: complexTypes, │ │ │ │ │ - customTypes: customTypes │ │ │ │ │ - }; │ │ │ │ │ - var i, len; │ │ │ │ │ - this.readChildNodes(node, schema); │ │ │ │ │ - var attributes = node.attributes; │ │ │ │ │ - var attr, name; │ │ │ │ │ - for (i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ - attr = attributes[i]; │ │ │ │ │ - name = attr.name; │ │ │ │ │ - if (name.indexOf("xmlns") === 0) { │ │ │ │ │ - this.setNamespace(name.split(":")[1] || "", attr.value) │ │ │ │ │ - } else { │ │ │ │ │ - obj[name] = attr.value │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - obj.featureTypes = complexTypes; │ │ │ │ │ - obj.targetPrefix = this.namespaceAlias[obj.targetNamespace]; │ │ │ │ │ - var complexType, customType; │ │ │ │ │ - for (i = 0, len = complexTypes.length; i < len; ++i) { │ │ │ │ │ - complexType = complexTypes[i]; │ │ │ │ │ - customType = customTypes[complexType.typeName]; │ │ │ │ │ - if (customTypes[complexType.typeName]) { │ │ │ │ │ - complexType.typeName = customType.name │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - complexType: function(node, obj) { │ │ │ │ │ - var complexType = { │ │ │ │ │ - typeName: node.getAttribute("name") │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, complexType); │ │ │ │ │ - obj.complexTypes.push(complexType) │ │ │ │ │ - }, │ │ │ │ │ - complexContent: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - extension: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - sequence: function(node, obj) { │ │ │ │ │ - var sequence = { │ │ │ │ │ - elements: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, sequence); │ │ │ │ │ - obj.properties = sequence.elements │ │ │ │ │ - }, │ │ │ │ │ - element: function(node, obj) { │ │ │ │ │ - var type; │ │ │ │ │ - if (obj.elements) { │ │ │ │ │ - var element = {}; │ │ │ │ │ - var attributes = node.attributes; │ │ │ │ │ - var attr; │ │ │ │ │ - for (var i = 0, len = attributes.length; i < len; ++i) { │ │ │ │ │ - attr = attributes[i]; │ │ │ │ │ - element[attr.name] = attr.value │ │ │ │ │ - } │ │ │ │ │ - type = element.type; │ │ │ │ │ - if (!type) { │ │ │ │ │ - type = {}; │ │ │ │ │ - this.readChildNodes(node, type); │ │ │ │ │ - element.restriction = type; │ │ │ │ │ - element.type = type.base │ │ │ │ │ - } │ │ │ │ │ - var fullType = type.base || type; │ │ │ │ │ - element.localType = fullType.split(":").pop(); │ │ │ │ │ - obj.elements.push(element); │ │ │ │ │ - this.readChildNodes(node, element) │ │ │ │ │ + parseLocations: function(node) { │ │ │ │ │ + var georssns = this.namespaces.georss; │ │ │ │ │ + var locations = { │ │ │ │ │ + components: [] │ │ │ │ │ + }; │ │ │ │ │ + var where = this.getElementsByTagNameNS(node, georssns, "where"); │ │ │ │ │ + if (where && where.length > 0) { │ │ │ │ │ + if (!this.gmlParser) { │ │ │ │ │ + this.initGmlParser() │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, ii = where.length; i < ii; i++) { │ │ │ │ │ + this.gmlParser.readChildNodes(where[i], locations) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var components = locations.components; │ │ │ │ │ + var point = this.getElementsByTagNameNS(node, georssns, "point"); │ │ │ │ │ + if (point && point.length > 0) { │ │ │ │ │ + for (var i = 0, ii = point.length; i < ii; i++) { │ │ │ │ │ + var xy = OpenLayers.String.trim(point[i].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ + if (xy.length != 2) { │ │ │ │ │ + xy = OpenLayers.String.trim(point[i].firstChild.nodeValue).split(/\s*,\s*/) │ │ │ │ │ } │ │ │ │ │ - if (obj.complexTypes) { │ │ │ │ │ - type = node.getAttribute("type"); │ │ │ │ │ - var localType = type.split(":").pop(); │ │ │ │ │ - obj.customTypes[localType] = { │ │ │ │ │ - name: node.getAttribute("name"), │ │ │ │ │ - type: type │ │ │ │ │ - } │ │ │ │ │ + components.push(new OpenLayers.Geometry.Point(xy[1], xy[0])) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var line = this.getElementsByTagNameNS(node, georssns, "line"); │ │ │ │ │ + if (line && line.length > 0) { │ │ │ │ │ + var coords; │ │ │ │ │ + var p; │ │ │ │ │ + var points; │ │ │ │ │ + for (var i = 0, ii = line.length; i < ii; i++) { │ │ │ │ │ + coords = OpenLayers.String.trim(line[i].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ + points = []; │ │ │ │ │ + for (var j = 0, jj = coords.length; j < jj; j += 2) { │ │ │ │ │ + p = new OpenLayers.Geometry.Point(coords[j + 1], coords[j]); │ │ │ │ │ + points.push(p) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - annotation: function(node, obj) { │ │ │ │ │ - obj.annotation = {}; │ │ │ │ │ - this.readChildNodes(node, obj.annotation) │ │ │ │ │ - }, │ │ │ │ │ - appinfo: function(node, obj) { │ │ │ │ │ - if (!obj.appinfo) { │ │ │ │ │ - obj.appinfo = [] │ │ │ │ │ + components.push(new OpenLayers.Geometry.LineString(points)) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var polygon = this.getElementsByTagNameNS(node, georssns, "polygon"); │ │ │ │ │ + if (polygon && polygon.length > 0) { │ │ │ │ │ + var coords; │ │ │ │ │ + var p; │ │ │ │ │ + var points; │ │ │ │ │ + for (var i = 0, ii = polygon.length; i < ii; i++) { │ │ │ │ │ + coords = OpenLayers.String.trim(polygon[i].firstChild.nodeValue).split(/\s+/); │ │ │ │ │ + points = []; │ │ │ │ │ + for (var j = 0, jj = coords.length; j < jj; j += 2) { │ │ │ │ │ + p = new OpenLayers.Geometry.Point(coords[j + 1], coords[j]); │ │ │ │ │ + points.push(p) │ │ │ │ │ } │ │ │ │ │ - obj.appinfo.push(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - documentation: function(node, obj) { │ │ │ │ │ - if (!obj.documentation) { │ │ │ │ │ - obj.documentation = [] │ │ │ │ │ + components.push(new OpenLayers.Geometry.Polygon([new OpenLayers.Geometry.LinearRing(points)])) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.internalProjection && this.externalProjection) { │ │ │ │ │ + for (var i = 0, ii = components.length; i < ii; i++) { │ │ │ │ │ + if (components[i]) { │ │ │ │ │ + components[i].transform(this.externalProjection, this.internalProjection) │ │ │ │ │ } │ │ │ │ │ - var value = this.getChildValue(node); │ │ │ │ │ - obj.documentation.push({ │ │ │ │ │ - lang: node.getAttribute("xml:lang"), │ │ │ │ │ - textContent: value.replace(this.regExes.trimSpace, "") │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - simpleType: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - restriction: function(node, obj) { │ │ │ │ │ - obj.base = node.getAttribute("base"); │ │ │ │ │ - this.readRestriction(node, obj) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return components │ │ │ │ │ }, │ │ │ │ │ - readRestriction: function(node, obj) { │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - var child, nodeName, value; │ │ │ │ │ - for (var i = 0, len = children.length; i < len; ++i) { │ │ │ │ │ - child = children[i]; │ │ │ │ │ - if (child.nodeType == 1) { │ │ │ │ │ - nodeName = child.nodeName.split(":").pop(); │ │ │ │ │ - value = child.getAttribute("value"); │ │ │ │ │ - if (!obj[nodeName]) { │ │ │ │ │ - obj[nodeName] = value │ │ │ │ │ - } else { │ │ │ │ │ - if (typeof obj[nodeName] == "string") { │ │ │ │ │ - obj[nodeName] = [obj[nodeName]] │ │ │ │ │ - } │ │ │ │ │ - obj[nodeName].push(value) │ │ │ │ │ + parsePersonConstructs: function(node, name, data) { │ │ │ │ │ + var persons = []; │ │ │ │ │ + var atomns = this.namespaces.atom; │ │ │ │ │ + var nodes = this.getElementsByTagNameNS(node, atomns, name); │ │ │ │ │ + var oAtts = ["uri", "email"]; │ │ │ │ │ + for (var i = 0, ii = nodes.length; i < ii; i++) { │ │ │ │ │ + var value = {}; │ │ │ │ │ + value.name = this.getFirstChildValue(nodes[i], atomns, "name", null); │ │ │ │ │ + for (var j = 0, jj = oAtts.length; j < jj; j++) { │ │ │ │ │ + var attval = this.getFirstChildValue(nodes[i], atomns, oAtts[j], null); │ │ │ │ │ + if (attval) { │ │ │ │ │ + value[oAtts[j]] = attval │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + persons.push(value) │ │ │ │ │ + } │ │ │ │ │ + if (persons.length > 0) { │ │ │ │ │ + data[name + "s"] = persons │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.Atom" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.SOSGetObservation = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + namespaces: { │ │ │ │ │ + ows: "http://www.opengis.net/ows", │ │ │ │ │ + gml: "http://www.opengis.net/gml", │ │ │ │ │ + sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ + ogc: "http://www.opengis.net/ogc", │ │ │ │ │ + om: "http://www.opengis.net/om/1.0", │ │ │ │ │ + sa: "http://www.opengis.net/sampling/1.0", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: /^\s*|\s*$/g, │ │ │ │ │ + removeSpace: /\s*/g, │ │ │ │ │ + splitSpace: /\s+/, │ │ │ │ │ + trimComma: /\s*,\s*/g │ │ │ │ │ + }, │ │ │ │ │ + VERSION: "1.0.0", │ │ │ │ │ + schemaLocation: "http://www.opengis.net/sos/1.0 http://schemas.opengis.net/sos/1.0.0/sosGetObservation.xsd", │ │ │ │ │ + defaultPrefix: "sos", │ │ │ │ │ read: function(data) { │ │ │ │ │ if (typeof data == "string") { │ │ │ │ │ data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ } │ │ │ │ │ if (data && data.nodeType == 9) { │ │ │ │ │ data = data.documentElement │ │ │ │ │ } │ │ │ │ │ - var schema = {}; │ │ │ │ │ - if (data.nodeName.split(":").pop() === "ExceptionReport") { │ │ │ │ │ - var parser = new OpenLayers.Format.OGCExceptionReport; │ │ │ │ │ - schema.error = parser.read(data) │ │ │ │ │ - } else { │ │ │ │ │ - this.readNode(data, schema) │ │ │ │ │ - } │ │ │ │ │ - return schema │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSDescribeFeatureType" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.SLD = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - profile: null, │ │ │ │ │ - defaultVersion: "1.0.0", │ │ │ │ │ - stringifyOutput: true, │ │ │ │ │ - namedLayersAsArray: false, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SLD" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WFS = OpenLayers.Class(OpenLayers.Format.GML, { │ │ │ │ │ - layer: null, │ │ │ │ │ - wfsns: "http://www.opengis.net/wfs", │ │ │ │ │ - ogcns: "http://www.opengis.net/ogc", │ │ │ │ │ - initialize: function(options, layer) { │ │ │ │ │ - OpenLayers.Format.GML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - if (this.layer.featureNS) { │ │ │ │ │ - this.featureNS = this.layer.featureNS │ │ │ │ │ - } │ │ │ │ │ - if (this.layer.options.geometry_column) { │ │ │ │ │ - this.geometryName = this.layer.options.geometry_column │ │ │ │ │ - } │ │ │ │ │ - if (this.layer.options.typename) { │ │ │ │ │ - this.featureName = this.layer.options.typename │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - write: function(features) { │ │ │ │ │ - var transaction = this.createElementNS(this.wfsns, "wfs:Transaction"); │ │ │ │ │ - transaction.setAttribute("version", "1.0.0"); │ │ │ │ │ - transaction.setAttribute("service", "WFS"); │ │ │ │ │ - for (var i = 0; i < features.length; i++) { │ │ │ │ │ - switch (features[i].state) { │ │ │ │ │ - case OpenLayers.State.INSERT: │ │ │ │ │ - transaction.appendChild(this.insert(features[i])); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.State.UPDATE: │ │ │ │ │ - transaction.appendChild(this.update(features[i])); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.State.DELETE: │ │ │ │ │ - transaction.appendChild(this.remove(features[i])); │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [transaction]) │ │ │ │ │ - }, │ │ │ │ │ - createFeatureXML: function(feature) { │ │ │ │ │ - var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ - var geomContainer = this.createElementNS(this.featureNS, "feature:" + this.geometryName); │ │ │ │ │ - geomContainer.appendChild(geometryNode); │ │ │ │ │ - var featureContainer = this.createElementNS(this.featureNS, "feature:" + this.featureName); │ │ │ │ │ - featureContainer.appendChild(geomContainer); │ │ │ │ │ - for (var attr in feature.attributes) { │ │ │ │ │ - var attrText = this.createTextNode(feature.attributes[attr]); │ │ │ │ │ - var nodename = attr; │ │ │ │ │ - if (attr.search(":") != -1) { │ │ │ │ │ - nodename = attr.split(":")[1] │ │ │ │ │ - } │ │ │ │ │ - var attrContainer = this.createElementNS(this.featureNS, "feature:" + nodename); │ │ │ │ │ - attrContainer.appendChild(attrText); │ │ │ │ │ - featureContainer.appendChild(attrContainer) │ │ │ │ │ - } │ │ │ │ │ - return featureContainer │ │ │ │ │ - }, │ │ │ │ │ - insert: function(feature) { │ │ │ │ │ - var insertNode = this.createElementNS(this.wfsns, "wfs:Insert"); │ │ │ │ │ - insertNode.appendChild(this.createFeatureXML(feature)); │ │ │ │ │ - return insertNode │ │ │ │ │ - }, │ │ │ │ │ - update: function(feature) { │ │ │ │ │ - if (!feature.fid) { │ │ │ │ │ - OpenLayers.Console.userError(OpenLayers.i18n("noFID")) │ │ │ │ │ - } │ │ │ │ │ - var updateNode = this.createElementNS(this.wfsns, "wfs:Update"); │ │ │ │ │ - updateNode.setAttribute("typeName", this.featurePrefix + ":" + this.featureName); │ │ │ │ │ - updateNode.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ - var propertyNode = this.createElementNS(this.wfsns, "wfs:Property"); │ │ │ │ │ - var nameNode = this.createElementNS(this.wfsns, "wfs:Name"); │ │ │ │ │ - var txtNode = this.createTextNode(this.geometryName); │ │ │ │ │ - nameNode.appendChild(txtNode); │ │ │ │ │ - propertyNode.appendChild(nameNode); │ │ │ │ │ - var valueNode = this.createElementNS(this.wfsns, "wfs:Value"); │ │ │ │ │ - var geometryNode = this.buildGeometryNode(feature.geometry); │ │ │ │ │ - if (feature.layer) { │ │ │ │ │ - geometryNode.setAttribute("srsName", feature.layer.projection.getCode()) │ │ │ │ │ - } │ │ │ │ │ - valueNode.appendChild(geometryNode); │ │ │ │ │ - propertyNode.appendChild(valueNode); │ │ │ │ │ - updateNode.appendChild(propertyNode); │ │ │ │ │ - for (var propName in feature.attributes) { │ │ │ │ │ - propertyNode = this.createElementNS(this.wfsns, "wfs:Property"); │ │ │ │ │ - nameNode = this.createElementNS(this.wfsns, "wfs:Name"); │ │ │ │ │ - nameNode.appendChild(this.createTextNode(propName)); │ │ │ │ │ - propertyNode.appendChild(nameNode); │ │ │ │ │ - valueNode = this.createElementNS(this.wfsns, "wfs:Value"); │ │ │ │ │ - valueNode.appendChild(this.createTextNode(feature.attributes[propName])); │ │ │ │ │ - propertyNode.appendChild(valueNode); │ │ │ │ │ - updateNode.appendChild(propertyNode) │ │ │ │ │ - } │ │ │ │ │ - var filterNode = this.createElementNS(this.ogcns, "ogc:Filter"); │ │ │ │ │ - var filterIdNode = this.createElementNS(this.ogcns, "ogc:FeatureId"); │ │ │ │ │ - filterIdNode.setAttribute("fid", feature.fid); │ │ │ │ │ - filterNode.appendChild(filterIdNode); │ │ │ │ │ - updateNode.appendChild(filterNode); │ │ │ │ │ - return updateNode │ │ │ │ │ - }, │ │ │ │ │ - remove: function(feature) { │ │ │ │ │ - if (!feature.fid) { │ │ │ │ │ - OpenLayers.Console.userError(OpenLayers.i18n("noFID")); │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - var deleteNode = this.createElementNS(this.wfsns, "wfs:Delete"); │ │ │ │ │ - deleteNode.setAttribute("typeName", this.featurePrefix + ":" + this.featureName); │ │ │ │ │ - deleteNode.setAttribute("xmlns:" + this.featurePrefix, this.featureNS); │ │ │ │ │ - var filterNode = this.createElementNS(this.ogcns, "ogc:Filter"); │ │ │ │ │ - var filterIdNode = this.createElementNS(this.ogcns, "ogc:FeatureId"); │ │ │ │ │ - filterIdNode.setAttribute("fid", feature.fid); │ │ │ │ │ - filterNode.appendChild(filterIdNode); │ │ │ │ │ - deleteNode.appendChild(filterNode); │ │ │ │ │ - return deleteNode │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.layer = null │ │ │ │ │ + var info = { │ │ │ │ │ + measurements: [], │ │ │ │ │ + observations: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readNode(data, info); │ │ │ │ │ + return info │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFS" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.XLS = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ - defaultVersion: "1.1.0", │ │ │ │ │ - stringifyOutput: true, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.XLS" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.GML.v2 = OpenLayers.Class(OpenLayers.Format.GML.Base, { │ │ │ │ │ - schemaLocation: "http://www.opengis.net/gml http://schemas.opengis.net/gml/2.1.2/feature.xsd", │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.GML.Base.prototype.initialize.apply(this, [options]) │ │ │ │ │ + write: function(options) { │ │ │ │ │ + var node = this.writeNode("sos:GetObservation", options); │ │ │ │ │ + node.setAttribute("xmlns:om", this.namespaces.om); │ │ │ │ │ + node.setAttribute("xmlns:ogc", this.namespaces.ogc); │ │ │ │ │ + this.setAttributeNS(node, this.namespaces.xsi, "xsi:schemaLocation", this.schemaLocation); │ │ │ │ │ + return OpenLayers.Format.XML.prototype.write.apply(this, [node]) │ │ │ │ │ }, │ │ │ │ │ readers: { │ │ │ │ │ - gml: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - outerBoundaryIs: function(node, container) { │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - container.outer = obj.components[0] │ │ │ │ │ + om: { │ │ │ │ │ + ObservationCollection: function(node, obj) { │ │ │ │ │ + obj.id = this.getAttributeNS(node, this.namespaces.gml, "id"); │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ }, │ │ │ │ │ - innerBoundaryIs: function(node, container) { │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - container.inner.push(obj.components[0]) │ │ │ │ │ + member: function(node, observationCollection) { │ │ │ │ │ + this.readChildNodes(node, observationCollection) │ │ │ │ │ }, │ │ │ │ │ - Box: function(node, container) { │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - if (!container.components) { │ │ │ │ │ - container.components = [] │ │ │ │ │ + Measurement: function(node, observationCollection) { │ │ │ │ │ + var measurement = {}; │ │ │ │ │ + observationCollection.measurements.push(measurement); │ │ │ │ │ + this.readChildNodes(node, measurement) │ │ │ │ │ + }, │ │ │ │ │ + Observation: function(node, observationCollection) { │ │ │ │ │ + var observation = {}; │ │ │ │ │ + observationCollection.observations.push(observation); │ │ │ │ │ + this.readChildNodes(node, observation) │ │ │ │ │ + }, │ │ │ │ │ + samplingTime: function(node, measurement) { │ │ │ │ │ + var samplingTime = {}; │ │ │ │ │ + measurement.samplingTime = samplingTime; │ │ │ │ │ + this.readChildNodes(node, samplingTime) │ │ │ │ │ + }, │ │ │ │ │ + observedProperty: function(node, measurement) { │ │ │ │ │ + measurement.observedProperty = this.getAttributeNS(node, this.namespaces.xlink, "href"); │ │ │ │ │ + this.readChildNodes(node, measurement) │ │ │ │ │ + }, │ │ │ │ │ + procedure: function(node, measurement) { │ │ │ │ │ + measurement.procedure = this.getAttributeNS(node, this.namespaces.xlink, "href"); │ │ │ │ │ + this.readChildNodes(node, measurement) │ │ │ │ │ + }, │ │ │ │ │ + featureOfInterest: function(node, observation) { │ │ │ │ │ + var foi = { │ │ │ │ │ + features: [] │ │ │ │ │ + }; │ │ │ │ │ + observation.fois = []; │ │ │ │ │ + observation.fois.push(foi); │ │ │ │ │ + this.readChildNodes(node, foi); │ │ │ │ │ + var features = []; │ │ │ │ │ + for (var i = 0, len = foi.features.length; i < len; i++) { │ │ │ │ │ + var feature = foi.features[i]; │ │ │ │ │ + features.push(new OpenLayers.Feature.Vector(feature.components[0], feature.attributes)) │ │ │ │ │ + } │ │ │ │ │ + foi.features = features │ │ │ │ │ + }, │ │ │ │ │ + result: function(node, measurement) { │ │ │ │ │ + var result = {}; │ │ │ │ │ + measurement.result = result; │ │ │ │ │ + if (this.getChildValue(node) !== "") { │ │ │ │ │ + result.value = this.getChildValue(node); │ │ │ │ │ + result.uom = node.getAttribute("uom") │ │ │ │ │ + } else { │ │ │ │ │ + this.readChildNodes(node, result) │ │ │ │ │ } │ │ │ │ │ - var min = obj.points[0]; │ │ │ │ │ - var max = obj.points[1]; │ │ │ │ │ - container.components.push(new OpenLayers.Bounds(min.x, min.y, max.x, max.y)) │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.GML.Base.prototype.readers["gml"]), │ │ │ │ │ - feature: OpenLayers.Format.GML.Base.prototype.readers["feature"], │ │ │ │ │ - wfs: OpenLayers.Format.GML.Base.prototype.readers["wfs"] │ │ │ │ │ - }, │ │ │ │ │ - write: function(features) { │ │ │ │ │ - var name; │ │ │ │ │ - if (OpenLayers.Util.isArray(features)) { │ │ │ │ │ - name = "wfs:FeatureCollection" │ │ │ │ │ - } else { │ │ │ │ │ - name = "gml:featureMember" │ │ │ │ │ - } │ │ │ │ │ - var root = this.writeNode(name, features); │ │ │ │ │ - this.setAttributeNS(root, this.namespaces["xsi"], "xsi:schemaLocation", this.schemaLocation); │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [root]) │ │ │ │ │ - }, │ │ │ │ │ - writers: { │ │ │ │ │ + }, │ │ │ │ │ + sa: OpenLayers.Format.SOSGetFeatureOfInterest.prototype.readers.sa, │ │ │ │ │ gml: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Point: function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:Point"); │ │ │ │ │ - this.writeNode("coordinates", [geometry], node); │ │ │ │ │ - return node │ │ │ │ │ + TimeInstant: function(node, samplingTime) { │ │ │ │ │ + var timeInstant = {}; │ │ │ │ │ + samplingTime.timeInstant = timeInstant; │ │ │ │ │ + this.readChildNodes(node, timeInstant) │ │ │ │ │ }, │ │ │ │ │ - coordinates: function(points) { │ │ │ │ │ - var numPoints = points.length; │ │ │ │ │ - var parts = new Array(numPoints); │ │ │ │ │ - var point; │ │ │ │ │ - for (var i = 0; i < numPoints; ++i) { │ │ │ │ │ - point = points[i]; │ │ │ │ │ - if (this.xy) { │ │ │ │ │ - parts[i] = point.x + "," + point.y │ │ │ │ │ - } else { │ │ │ │ │ - parts[i] = point.y + "," + point.x │ │ │ │ │ - } │ │ │ │ │ - if (point.z != undefined) { │ │ │ │ │ - parts[i] += "," + point.z │ │ │ │ │ + timePosition: function(node, timeInstant) { │ │ │ │ │ + timeInstant.timePosition = this.getChildValue(node) │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.SOSGetFeatureOfInterest.prototype.readers.gml) │ │ │ │ │ + }, │ │ │ │ │ + writers: { │ │ │ │ │ + sos: { │ │ │ │ │ + GetObservation: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("GetObservation", { │ │ │ │ │ + attributes: { │ │ │ │ │ + version: this.VERSION, │ │ │ │ │ + service: "SOS" │ │ │ │ │ } │ │ │ │ │ + }); │ │ │ │ │ + this.writeNode("offering", options, node); │ │ │ │ │ + if (options.eventTime) { │ │ │ │ │ + this.writeNode("eventTime", options, node) │ │ │ │ │ } │ │ │ │ │ - return this.createElementNSPlus("gml:coordinates", { │ │ │ │ │ - attributes: { │ │ │ │ │ - decimal: ".", │ │ │ │ │ - cs: ",", │ │ │ │ │ - ts: " " │ │ │ │ │ - }, │ │ │ │ │ - value: numPoints == 1 ? parts[0] : parts.join(" ") │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - LineString: function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:LineString"); │ │ │ │ │ - this.writeNode("coordinates", geometry.components, node); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - Polygon: function(geometry) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:Polygon"); │ │ │ │ │ - this.writeNode("outerBoundaryIs", geometry.components[0], node); │ │ │ │ │ - for (var i = 1; i < geometry.components.length; ++i) { │ │ │ │ │ - this.writeNode("innerBoundaryIs", geometry.components[i], node) │ │ │ │ │ + for (var procedure in options.procedures) { │ │ │ │ │ + this.writeNode("procedure", options.procedures[procedure], node) │ │ │ │ │ + } │ │ │ │ │ + for (var observedProperty in options.observedProperties) { │ │ │ │ │ + this.writeNode("observedProperty", options.observedProperties[observedProperty], node) │ │ │ │ │ + } │ │ │ │ │ + if (options.foi) { │ │ │ │ │ + this.writeNode("featureOfInterest", options.foi, node) │ │ │ │ │ + } │ │ │ │ │ + this.writeNode("responseFormat", options, node); │ │ │ │ │ + if (options.resultModel) { │ │ │ │ │ + this.writeNode("resultModel", options, node) │ │ │ │ │ + } │ │ │ │ │ + if (options.responseMode) { │ │ │ │ │ + this.writeNode("responseMode", options, node) │ │ │ │ │ } │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - outerBoundaryIs: function(ring) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:outerBoundaryIs"); │ │ │ │ │ - this.writeNode("LinearRing", ring, node); │ │ │ │ │ + featureOfInterest: function(foi) { │ │ │ │ │ + var node = this.createElementNSPlus("featureOfInterest"); │ │ │ │ │ + this.writeNode("ObjectID", foi.objectId, node); │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - innerBoundaryIs: function(ring) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:innerBoundaryIs"); │ │ │ │ │ - this.writeNode("LinearRing", ring, node); │ │ │ │ │ - return node │ │ │ │ │ + ObjectID: function(options) { │ │ │ │ │ + return this.createElementNSPlus("ObjectID", { │ │ │ │ │ + value: options │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - LinearRing: function(ring) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:LinearRing"); │ │ │ │ │ - this.writeNode("coordinates", ring.components, node); │ │ │ │ │ - return node │ │ │ │ │ + responseFormat: function(options) { │ │ │ │ │ + return this.createElementNSPlus("responseFormat", { │ │ │ │ │ + value: options.responseFormat │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - Box: function(bounds) { │ │ │ │ │ - var node = this.createElementNSPlus("gml:Box"); │ │ │ │ │ - this.writeNode("coordinates", [{ │ │ │ │ │ - x: bounds.left, │ │ │ │ │ - y: bounds.bottom │ │ │ │ │ - }, { │ │ │ │ │ - x: bounds.right, │ │ │ │ │ - y: bounds.top │ │ │ │ │ - }], node); │ │ │ │ │ - if (this.srsName) { │ │ │ │ │ - node.setAttribute("srsName", this.srsName) │ │ │ │ │ + procedure: function(procedure) { │ │ │ │ │ + return this.createElementNSPlus("procedure", { │ │ │ │ │ + value: procedure │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + offering: function(options) { │ │ │ │ │ + return this.createElementNSPlus("offering", { │ │ │ │ │ + value: options.offering │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + observedProperty: function(observedProperty) { │ │ │ │ │ + return this.createElementNSPlus("observedProperty", { │ │ │ │ │ + value: observedProperty │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + eventTime: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("eventTime"); │ │ │ │ │ + if (options.eventTime === "latest") { │ │ │ │ │ + this.writeNode("ogc:TM_Equals", options, node) │ │ │ │ │ } │ │ │ │ │ return node │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.GML.Base.prototype.writers["gml"]), │ │ │ │ │ - feature: OpenLayers.Format.GML.Base.prototype.writers["feature"], │ │ │ │ │ - wfs: OpenLayers.Format.GML.Base.prototype.writers["wfs"] │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.GML.v2" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.Filter.v1_0_0 = OpenLayers.Class(OpenLayers.Format.GML.v2, OpenLayers.Format.Filter.v1, { │ │ │ │ │ - VERSION: "1.0.0", │ │ │ │ │ - schemaLocation: "http://www.opengis.net/ogc/filter/1.0.0/filter.xsd", │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.GML.v2.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ - readers: { │ │ │ │ │ - ogc: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - PropertyIsEqualTo: function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.EQUAL_TO │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - obj.filters.push(filter) │ │ │ │ │ }, │ │ │ │ │ - PropertyIsNotEqualTo: function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.NOT_EQUAL_TO │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - obj.filters.push(filter) │ │ │ │ │ + resultModel: function(options) { │ │ │ │ │ + return this.createElementNSPlus("resultModel", { │ │ │ │ │ + value: options.resultModel │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - PropertyIsLike: function(node, obj) { │ │ │ │ │ - var filter = new OpenLayers.Filter.Comparison({ │ │ │ │ │ - type: OpenLayers.Filter.Comparison.LIKE │ │ │ │ │ - }); │ │ │ │ │ - this.readChildNodes(node, filter); │ │ │ │ │ - var wildCard = node.getAttribute("wildCard"); │ │ │ │ │ - var singleChar = node.getAttribute("singleChar"); │ │ │ │ │ - var esc = node.getAttribute("escape"); │ │ │ │ │ - filter.value2regex(wildCard, singleChar, esc); │ │ │ │ │ - obj.filters.push(filter) │ │ │ │ │ + responseMode: function(options) { │ │ │ │ │ + return this.createElementNSPlus("responseMode", { │ │ │ │ │ + value: options.responseMode │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.Filter.v1.prototype.readers["ogc"]), │ │ │ │ │ - gml: OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ - feature: OpenLayers.Format.GML.v2.prototype.readers["feature"] │ │ │ │ │ - }, │ │ │ │ │ - writers: { │ │ │ │ │ - ogc: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - PropertyIsEqualTo: function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsEqualTo"); │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - this.writeOgcExpression(filter.value, node); │ │ │ │ │ + }, │ │ │ │ │ + ogc: { │ │ │ │ │ + TM_Equals: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("ogc:TM_Equals"); │ │ │ │ │ + this.writeNode("ogc:PropertyName", { │ │ │ │ │ + property: "urn:ogc:data:time:iso8601" │ │ │ │ │ + }, node); │ │ │ │ │ + if (options.eventTime === "latest") { │ │ │ │ │ + this.writeNode("gml:TimeInstant", { │ │ │ │ │ + value: "latest" │ │ │ │ │ + }, node) │ │ │ │ │ + } │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - PropertyIsNotEqualTo: function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsNotEqualTo"); │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - this.writeOgcExpression(filter.value, node); │ │ │ │ │ + PropertyName: function(options) { │ │ │ │ │ + return this.createElementNSPlus("ogc:PropertyName", { │ │ │ │ │ + value: options.property │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + gml: { │ │ │ │ │ + TimeInstant: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:TimeInstant"); │ │ │ │ │ + this.writeNode("gml:timePosition", options, node); │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - PropertyIsLike: function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:PropertyIsLike", { │ │ │ │ │ - attributes: { │ │ │ │ │ - wildCard: "*", │ │ │ │ │ - singleChar: ".", │ │ │ │ │ - escape: "!" │ │ │ │ │ - } │ │ │ │ │ + timePosition: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("gml:timePosition", { │ │ │ │ │ + value: options.value │ │ │ │ │ }); │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - this.writeNode("Literal", filter.regex2value(), node); │ │ │ │ │ return node │ │ │ │ │ - }, │ │ │ │ │ - BBOX: function(filter) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:BBOX"); │ │ │ │ │ - filter.property && this.writeNode("PropertyName", filter, node); │ │ │ │ │ - var box = this.writeNode("gml:Box", filter.value, node); │ │ │ │ │ - if (filter.projection) { │ │ │ │ │ - box.setAttribute("srsName", filter.projection) │ │ │ │ │ - } │ │ │ │ │ - return node │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.Filter.v1.prototype.writers["ogc"]), │ │ │ │ │ - gml: OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ - feature: OpenLayers.Format.GML.v2.prototype.writers["feature"] │ │ │ │ │ - }, │ │ │ │ │ - writeSpatial: function(filter, name) { │ │ │ │ │ - var node = this.createElementNSPlus("ogc:" + name); │ │ │ │ │ - this.writeNode("PropertyName", filter, node); │ │ │ │ │ - if (filter.value instanceof OpenLayers.Filter.Function) { │ │ │ │ │ - this.writeNode("Function", filter.value, node) │ │ │ │ │ - } else { │ │ │ │ │ - var child; │ │ │ │ │ - if (filter.value instanceof OpenLayers.Geometry) { │ │ │ │ │ - child = this.writeNode("feature:_geometry", filter.value).firstChild │ │ │ │ │ - } else { │ │ │ │ │ - child = this.writeNode("gml:Box", filter.value) │ │ │ │ │ - } │ │ │ │ │ - if (filter.projection) { │ │ │ │ │ - child.setAttribute("srsName", filter.projection) │ │ │ │ │ } │ │ │ │ │ - node.appendChild(child) │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.Filter.v1_0_0" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SOSGetObservation" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.CSWGetRecords.v2_0_2 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ +OpenLayers.Format.XLS = OpenLayers.Class(OpenLayers.Format.XML.VersionedOGC, { │ │ │ │ │ + defaultVersion: "1.1.0", │ │ │ │ │ + stringifyOutput: true, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.XLS" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.CSWGetDomain.v2_0_2 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ namespaces: { │ │ │ │ │ - csw: "http://www.opengis.net/cat/csw/2.0.2", │ │ │ │ │ - dc: "http://purl.org/dc/elements/1.1/", │ │ │ │ │ - dct: "http://purl.org/dc/terms/", │ │ │ │ │ - gmd: "http://www.isotc211.org/2005/gmd", │ │ │ │ │ - geonet: "http://www.fao.org/geonetwork", │ │ │ │ │ - ogc: "http://www.opengis.net/ogc", │ │ │ │ │ - ows: "http://www.opengis.net/ows", │ │ │ │ │ xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + csw: "http://www.opengis.net/cat/csw/2.0.2" │ │ │ │ │ }, │ │ │ │ │ defaultPrefix: "csw", │ │ │ │ │ version: "2.0.2", │ │ │ │ │ schemaLocation: "http://www.opengis.net/cat/csw/2.0.2 http://schemas.opengis.net/csw/2.0.2/CSW-discovery.xsd", │ │ │ │ │ - requestId: null, │ │ │ │ │ - resultType: null, │ │ │ │ │ - outputFormat: null, │ │ │ │ │ - outputSchema: null, │ │ │ │ │ - startPosition: null, │ │ │ │ │ - maxRecords: null, │ │ │ │ │ - DistributedSearch: null, │ │ │ │ │ - ResponseHandler: null, │ │ │ │ │ - Query: null, │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: /^\s*|\s*$/g, │ │ │ │ │ - removeSpace: /\s*/g, │ │ │ │ │ - splitSpace: /\s+/, │ │ │ │ │ - trimComma: /\s*,\s*/g │ │ │ │ │ - }, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ + PropertyName: null, │ │ │ │ │ + ParameterName: null, │ │ │ │ │ read: function(data) { │ │ │ │ │ if (typeof data == "string") { │ │ │ │ │ data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ } │ │ │ │ │ if (data && data.nodeType == 9) { │ │ │ │ │ data = data.documentElement │ │ │ │ │ } │ │ │ │ │ var obj = {}; │ │ │ │ │ this.readNode(data, obj); │ │ │ │ │ return obj │ │ │ │ │ }, │ │ │ │ │ readers: { │ │ │ │ │ csw: { │ │ │ │ │ - GetRecordsResponse: function(node, obj) { │ │ │ │ │ - obj.records = []; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - var version = this.getAttributeNS(node, "", "version"); │ │ │ │ │ - if (version != "") { │ │ │ │ │ - obj.version = version │ │ │ │ │ + GetDomainResponse: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + DomainValues: function(node, obj) { │ │ │ │ │ + if (!OpenLayers.Util.isArray(obj.DomainValues)) { │ │ │ │ │ + obj.DomainValues = [] │ │ │ │ │ + } │ │ │ │ │ + var attrs = node.attributes; │ │ │ │ │ + var domainValue = {}; │ │ │ │ │ + for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ + domainValue[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ } │ │ │ │ │ + this.readChildNodes(node, domainValue); │ │ │ │ │ + obj.DomainValues.push(domainValue) │ │ │ │ │ }, │ │ │ │ │ - RequestId: function(node, obj) { │ │ │ │ │ - obj.RequestId = this.getChildValue(node) │ │ │ │ │ + PropertyName: function(node, obj) { │ │ │ │ │ + obj.PropertyName = this.getChildValue(node) │ │ │ │ │ }, │ │ │ │ │ - SearchStatus: function(node, obj) { │ │ │ │ │ - obj.SearchStatus = {}; │ │ │ │ │ - var timestamp = this.getAttributeNS(node, "", "timestamp"); │ │ │ │ │ - if (timestamp != "") { │ │ │ │ │ - obj.SearchStatus.timestamp = timestamp │ │ │ │ │ + ParameterName: function(node, obj) { │ │ │ │ │ + obj.ParameterName = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + ListOfValues: function(node, obj) { │ │ │ │ │ + if (!OpenLayers.Util.isArray(obj.ListOfValues)) { │ │ │ │ │ + obj.ListOfValues = [] │ │ │ │ │ } │ │ │ │ │ + this.readChildNodes(node, obj.ListOfValues) │ │ │ │ │ }, │ │ │ │ │ - SearchResults: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ + Value: function(node, obj) { │ │ │ │ │ var attrs = node.attributes; │ │ │ │ │ - var SearchResults = {}; │ │ │ │ │ + var value = {}; │ │ │ │ │ for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - if (attrs[i].name == "numberOfRecordsMatched" || attrs[i].name == "numberOfRecordsReturned" || attrs[i].name == "nextRecord") { │ │ │ │ │ - SearchResults[attrs[i].name] = parseInt(attrs[i].nodeValue) │ │ │ │ │ - } else { │ │ │ │ │ - SearchResults[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ - } │ │ │ │ │ + value[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ } │ │ │ │ │ - obj.SearchResults = SearchResults │ │ │ │ │ + value.value = this.getChildValue(node); │ │ │ │ │ + obj.push({ │ │ │ │ │ + Value: value │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - SummaryRecord: function(node, obj) { │ │ │ │ │ - var record = { │ │ │ │ │ - type: "SummaryRecord" │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, record); │ │ │ │ │ - obj.records.push(record) │ │ │ │ │ + ConceptualScheme: function(node, obj) { │ │ │ │ │ + obj.ConceptualScheme = {}; │ │ │ │ │ + this.readChildNodes(node, obj.ConceptualScheme) │ │ │ │ │ }, │ │ │ │ │ - BriefRecord: function(node, obj) { │ │ │ │ │ - var record = { │ │ │ │ │ - type: "BriefRecord" │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, record); │ │ │ │ │ - obj.records.push(record) │ │ │ │ │ + Name: function(node, obj) { │ │ │ │ │ + obj.Name = this.getChildValue(node) │ │ │ │ │ }, │ │ │ │ │ - DCMIRecord: function(node, obj) { │ │ │ │ │ - var record = { │ │ │ │ │ - type: "DCMIRecord" │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, record); │ │ │ │ │ - obj.records.push(record) │ │ │ │ │ + Document: function(node, obj) { │ │ │ │ │ + obj.Document = this.getChildValue(node) │ │ │ │ │ }, │ │ │ │ │ - Record: function(node, obj) { │ │ │ │ │ - var record = { │ │ │ │ │ - type: "Record" │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, record); │ │ │ │ │ - obj.records.push(record) │ │ │ │ │ + Authority: function(node, obj) { │ │ │ │ │ + obj.Authority = this.getChildValue(node) │ │ │ │ │ }, │ │ │ │ │ - "*": function(node, obj) { │ │ │ │ │ - var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ - obj[name] = this.getChildValue(node) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - geonet: { │ │ │ │ │ - info: function(node, obj) { │ │ │ │ │ - var gninfo = {}; │ │ │ │ │ - this.readChildNodes(node, gninfo); │ │ │ │ │ - obj.gninfo = gninfo │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - dc: { │ │ │ │ │ - "*": function(node, obj) { │ │ │ │ │ - var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ - if (!OpenLayers.Util.isArray(obj[name])) { │ │ │ │ │ - obj[name] = [] │ │ │ │ │ - } │ │ │ │ │ - var dc_element = {}; │ │ │ │ │ + RangeOfValues: function(node, obj) { │ │ │ │ │ + obj.RangeOfValues = {}; │ │ │ │ │ + this.readChildNodes(node, obj.RangeOfValues) │ │ │ │ │ + }, │ │ │ │ │ + MinValue: function(node, obj) { │ │ │ │ │ var attrs = node.attributes; │ │ │ │ │ + var value = {}; │ │ │ │ │ for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - dc_element[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ - } │ │ │ │ │ - dc_element.value = this.getChildValue(node); │ │ │ │ │ - if (dc_element.value != "") { │ │ │ │ │ - obj[name].push(dc_element) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - dct: { │ │ │ │ │ - "*": function(node, obj) { │ │ │ │ │ - var name = node.localName || node.nodeName.split(":").pop(); │ │ │ │ │ - if (!OpenLayers.Util.isArray(obj[name])) { │ │ │ │ │ - obj[name] = [] │ │ │ │ │ + value[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ } │ │ │ │ │ - obj[name].push(this.getChildValue(node)) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - ows: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - BoundingBox: function(node, obj) { │ │ │ │ │ - if (obj.bounds) { │ │ │ │ │ - obj.BoundingBox = [{ │ │ │ │ │ - crs: obj.projection, │ │ │ │ │ - value: [obj.bounds.left, obj.bounds.bottom, obj.bounds.right, obj.bounds.top] │ │ │ │ │ - }]; │ │ │ │ │ - delete obj.projection; │ │ │ │ │ - delete obj.bounds │ │ │ │ │ + value.value = this.getChildValue(node); │ │ │ │ │ + obj.MinValue = value │ │ │ │ │ + }, │ │ │ │ │ + MaxValue: function(node, obj) { │ │ │ │ │ + var attrs = node.attributes; │ │ │ │ │ + var value = {}; │ │ │ │ │ + for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ + value[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Format.OWSCommon.v1_0_0.prototype.readers["ows"]["BoundingBox"].apply(this, arguments) │ │ │ │ │ + value.value = this.getChildValue(node); │ │ │ │ │ + obj.MaxValue = value │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.OWSCommon.v1_0_0.prototype.readers["ows"]) │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ write: function(options) { │ │ │ │ │ - var node = this.writeNode("csw:GetRecords", options); │ │ │ │ │ - node.setAttribute("xmlns:gmd", this.namespaces.gmd); │ │ │ │ │ + var node = this.writeNode("csw:GetDomain", options); │ │ │ │ │ return OpenLayers.Format.XML.prototype.write.apply(this, [node]) │ │ │ │ │ }, │ │ │ │ │ writers: { │ │ │ │ │ csw: { │ │ │ │ │ - GetRecords: function(options) { │ │ │ │ │ - if (!options) { │ │ │ │ │ - options = {} │ │ │ │ │ - } │ │ │ │ │ - var node = this.createElementNSPlus("csw:GetRecords", { │ │ │ │ │ + GetDomain: function(options) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:GetDomain", { │ │ │ │ │ attributes: { │ │ │ │ │ service: "CSW", │ │ │ │ │ - version: this.version, │ │ │ │ │ - requestId: options.requestId || this.requestId, │ │ │ │ │ - resultType: options.resultType || this.resultType, │ │ │ │ │ - outputFormat: options.outputFormat || this.outputFormat, │ │ │ │ │ - outputSchema: options.outputSchema || this.outputSchema, │ │ │ │ │ - startPosition: options.startPosition || this.startPosition, │ │ │ │ │ - maxRecords: options.maxRecords || this.maxRecords │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (options.DistributedSearch || this.DistributedSearch) { │ │ │ │ │ - this.writeNode("csw:DistributedSearch", options.DistributedSearch || this.DistributedSearch, node) │ │ │ │ │ - } │ │ │ │ │ - var ResponseHandler = options.ResponseHandler || this.ResponseHandler; │ │ │ │ │ - if (OpenLayers.Util.isArray(ResponseHandler) && ResponseHandler.length > 0) { │ │ │ │ │ - for (var i = 0, len = ResponseHandler.length; i < len; i++) { │ │ │ │ │ - this.writeNode("csw:ResponseHandler", ResponseHandler[i], node) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.writeNode("Query", options.Query || this.Query, node); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - DistributedSearch: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:DistributedSearch", { │ │ │ │ │ - attributes: { │ │ │ │ │ - hopCount: options.hopCount │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - ResponseHandler: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:ResponseHandler", { │ │ │ │ │ - value: options.value │ │ │ │ │ - }); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - Query: function(options) { │ │ │ │ │ - if (!options) { │ │ │ │ │ - options = {} │ │ │ │ │ - } │ │ │ │ │ - var node = this.createElementNSPlus("csw:Query", { │ │ │ │ │ - attributes: { │ │ │ │ │ - typeNames: options.typeNames || "csw:Record" │ │ │ │ │ + version: this.version │ │ │ │ │ } │ │ │ │ │ }); │ │ │ │ │ - var ElementName = options.ElementName; │ │ │ │ │ - if (OpenLayers.Util.isArray(ElementName) && ElementName.length > 0) { │ │ │ │ │ - for (var i = 0, len = ElementName.length; i < len; i++) { │ │ │ │ │ - this.writeNode("csw:ElementName", ElementName[i], node) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.writeNode("csw:ElementSetName", options.ElementSetName || { │ │ │ │ │ - value: "summary" │ │ │ │ │ - }, node) │ │ │ │ │ - } │ │ │ │ │ - if (options.Constraint) { │ │ │ │ │ - this.writeNode("csw:Constraint", options.Constraint, node) │ │ │ │ │ - } │ │ │ │ │ - if (options.SortBy) { │ │ │ │ │ - this.writeNode("ogc:SortBy", options.SortBy, node) │ │ │ │ │ + if (options.PropertyName || this.PropertyName) { │ │ │ │ │ + this.writeNode("csw:PropertyName", options.PropertyName || this.PropertyName, node) │ │ │ │ │ + } else if (options.ParameterName || this.ParameterName) { │ │ │ │ │ + this.writeNode("csw:ParameterName", options.ParameterName || this.ParameterName, node) │ │ │ │ │ } │ │ │ │ │ + this.readChildNodes(node, options); │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - ElementName: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:ElementName", { │ │ │ │ │ - value: options.value │ │ │ │ │ - }); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - ElementSetName: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:ElementSetName", { │ │ │ │ │ - attributes: { │ │ │ │ │ - typeNames: options.typeNames │ │ │ │ │ - }, │ │ │ │ │ - value: options.value │ │ │ │ │ + PropertyName: function(value) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:PropertyName", { │ │ │ │ │ + value: value │ │ │ │ │ }); │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ - Constraint: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:Constraint", { │ │ │ │ │ - attributes: { │ │ │ │ │ - version: options.version │ │ │ │ │ - } │ │ │ │ │ + ParameterName: function(value) { │ │ │ │ │ + var node = this.createElementNSPlus("csw:ParameterName", { │ │ │ │ │ + value: value │ │ │ │ │ }); │ │ │ │ │ - if (options.Filter) { │ │ │ │ │ - var format = new OpenLayers.Format.Filter({ │ │ │ │ │ - version: options.version │ │ │ │ │ - }); │ │ │ │ │ - node.appendChild(format.write(options.Filter)) │ │ │ │ │ - } else if (options.CqlText) { │ │ │ │ │ - var child = this.createElementNSPlus("CqlText", { │ │ │ │ │ - value: options.CqlText.value │ │ │ │ │ - }); │ │ │ │ │ - node.appendChild(child) │ │ │ │ │ - } │ │ │ │ │ return node │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - ogc: OpenLayers.Format.Filter.v1_1_0.prototype.writers["ogc"] │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.CSWGetRecords.v2_0_2" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.CSWGetDomain.v2_0_2" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Format.SLD.v1 = OpenLayers.Class(OpenLayers.Format.Filter.v1_0_0, { │ │ │ │ │ namespaces: { │ │ │ │ │ sld: "http://www.opengis.net/sld", │ │ │ │ │ ogc: "http://www.opengis.net/ogc", │ │ │ │ │ gml: "http://www.opengis.net/gml", │ │ │ │ │ xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ @@ -26783,14 +29002,1023 @@ │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.SLD.v1" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Format.SLD.v1_0_0 = OpenLayers.Class(OpenLayers.Format.SLD.v1, { │ │ │ │ │ VERSION: "1.0.0", │ │ │ │ │ schemaLocation: "http://www.opengis.net/sld http://schemas.opengis.net/sld/1.0.0/StyledLayerDescriptor.xsd", │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.SLD.v1_0_0" │ │ │ │ │ }); │ │ │ │ │ +OpenLayers.Format.SLD.v1_0_0_GeoServer = OpenLayers.Class(OpenLayers.Format.SLD.v1_0_0, { │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + profile: "GeoServer", │ │ │ │ │ + readers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + sld: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Priority: function(node, obj) { │ │ │ │ │ + var value = this.readers.ogc._expression.call(this, node); │ │ │ │ │ + if (value) { │ │ │ │ │ + obj.priority = value │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + VendorOption: function(node, obj) { │ │ │ │ │ + if (!obj.vendorOptions) { │ │ │ │ │ + obj.vendorOptions = {} │ │ │ │ │ + } │ │ │ │ │ + obj.vendorOptions[node.getAttribute("name")] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + TextSymbolizer: function(node, rule) { │ │ │ │ │ + OpenLayers.Format.SLD.v1_0_0.prototype.readers.sld.TextSymbolizer.apply(this, arguments); │ │ │ │ │ + var symbolizer = this.multipleSymbolizers ? rule.symbolizers[rule.symbolizers.length - 1] : rule.symbolizer["Text"]; │ │ │ │ │ + if (symbolizer.graphic === undefined) { │ │ │ │ │ + symbolizer.graphic = false │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.SLD.v1_0_0.prototype.readers["sld"]) │ │ │ │ │ + }, OpenLayers.Format.SLD.v1_0_0.prototype.readers), │ │ │ │ │ + writers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + sld: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Priority: function(priority) { │ │ │ │ │ + return this.writers.sld._OGCExpression.call(this, "sld:Priority", priority) │ │ │ │ │ + }, │ │ │ │ │ + VendorOption: function(option) { │ │ │ │ │ + return this.createElementNSPlus("sld:VendorOption", { │ │ │ │ │ + attributes: { │ │ │ │ │ + name: option.name │ │ │ │ │ + }, │ │ │ │ │ + value: option.value │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + TextSymbolizer: function(symbolizer) { │ │ │ │ │ + var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ + var node = writers["sld"]["TextSymbolizer"].apply(this, arguments); │ │ │ │ │ + if (symbolizer.graphic !== false && (symbolizer.externalGraphic || symbolizer.graphicName)) { │ │ │ │ │ + this.writeNode("Graphic", symbolizer, node) │ │ │ │ │ + } │ │ │ │ │ + if ("priority" in symbolizer) { │ │ │ │ │ + this.writeNode("Priority", symbolizer.priority, node) │ │ │ │ │ + } │ │ │ │ │ + return this.addVendorOptions(node, symbolizer) │ │ │ │ │ + }, │ │ │ │ │ + PointSymbolizer: function(symbolizer) { │ │ │ │ │ + var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ + var node = writers["sld"]["PointSymbolizer"].apply(this, arguments); │ │ │ │ │ + return this.addVendorOptions(node, symbolizer) │ │ │ │ │ + }, │ │ │ │ │ + LineSymbolizer: function(symbolizer) { │ │ │ │ │ + var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ + var node = writers["sld"]["LineSymbolizer"].apply(this, arguments); │ │ │ │ │ + return this.addVendorOptions(node, symbolizer) │ │ │ │ │ + }, │ │ │ │ │ + PolygonSymbolizer: function(symbolizer) { │ │ │ │ │ + var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ + var node = writers["sld"]["PolygonSymbolizer"].apply(this, arguments); │ │ │ │ │ + return this.addVendorOptions(node, symbolizer) │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.SLD.v1_0_0.prototype.writers["sld"]) │ │ │ │ │ + }, OpenLayers.Format.SLD.v1_0_0.prototype.writers), │ │ │ │ │ + addVendorOptions: function(node, symbolizer) { │ │ │ │ │ + var options = symbolizer.vendorOptions; │ │ │ │ │ + if (options) { │ │ │ │ │ + for (var key in symbolizer.vendorOptions) { │ │ │ │ │ + this.writeNode("VendorOption", { │ │ │ │ │ + name: key, │ │ │ │ │ + value: symbolizer.vendorOptions[key] │ │ │ │ │ + }, node) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return node │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.SLD.v1_0_0_GeoServer" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.ArcXML.Features = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + read: function(data) { │ │ │ │ │ + var axl = new OpenLayers.Format.ArcXML; │ │ │ │ │ + var parsed = axl.read(data); │ │ │ │ │ + return parsed.features.feature │ │ │ │ │ + } │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WFSCapabilities.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + namespaces: { │ │ │ │ │ + wfs: "http://www.opengis.net/wfs", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + ows: "http://www.opengis.net/ows" │ │ │ │ │ + }, │ │ │ │ │ + errorProperty: "featureTypeList", │ │ │ │ │ + defaultPrefix: "wfs", │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + var raw = data; │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + return capabilities │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + wfs: { │ │ │ │ │ + WFS_Capabilities: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + FeatureTypeList: function(node, request) { │ │ │ │ │ + request.featureTypeList = { │ │ │ │ │ + featureTypes: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, request.featureTypeList) │ │ │ │ │ + }, │ │ │ │ │ + FeatureType: function(node, featureTypeList) { │ │ │ │ │ + var featureType = {}; │ │ │ │ │ + this.readChildNodes(node, featureType); │ │ │ │ │ + featureTypeList.featureTypes.push(featureType) │ │ │ │ │ + }, │ │ │ │ │ + Name: function(node, obj) { │ │ │ │ │ + var name = this.getChildValue(node); │ │ │ │ │ + if (name) { │ │ │ │ │ + var parts = name.split(":"); │ │ │ │ │ + obj.name = parts.pop(); │ │ │ │ │ + if (parts.length > 0) { │ │ │ │ │ + obj.featureNS = this.lookupNamespaceURI(node, parts[0]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Title: function(node, obj) { │ │ │ │ │ + var title = this.getChildValue(node); │ │ │ │ │ + if (title) { │ │ │ │ │ + obj.title = title │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Abstract: function(node, obj) { │ │ │ │ │ + var abst = this.getChildValue(node); │ │ │ │ │ + if (abst) { │ │ │ │ │ + obj["abstract"] = abst │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WFSCapabilities.v1_1_0 = OpenLayers.Class(OpenLayers.Format.WFSCapabilities.v1, { │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: /^\s*|\s*$/g, │ │ │ │ │ + removeSpace: /\s*/g, │ │ │ │ │ + splitSpace: /\s+/, │ │ │ │ │ + trimComma: /\s*,\s*/g │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + wfs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + DefaultSRS: function(node, obj) { │ │ │ │ │ + var defaultSRS = this.getChildValue(node); │ │ │ │ │ + if (defaultSRS) { │ │ │ │ │ + obj.srs = defaultSRS │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WFSCapabilities.v1.prototype.readers["wfs"]), │ │ │ │ │ + ows: OpenLayers.Format.OWSCommon.v1.prototype.readers.ows │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1_1_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WFSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.WFSCapabilities.v1, { │ │ │ │ │ + readers: { │ │ │ │ │ + wfs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Service: function(node, capabilities) { │ │ │ │ │ + capabilities.service = {}; │ │ │ │ │ + this.readChildNodes(node, capabilities.service) │ │ │ │ │ + }, │ │ │ │ │ + Fees: function(node, service) { │ │ │ │ │ + var fees = this.getChildValue(node); │ │ │ │ │ + if (fees && fees.toLowerCase() != "none") { │ │ │ │ │ + service.fees = fees │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + AccessConstraints: function(node, service) { │ │ │ │ │ + var constraints = this.getChildValue(node); │ │ │ │ │ + if (constraints && constraints.toLowerCase() != "none") { │ │ │ │ │ + service.accessConstraints = constraints │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + OnlineResource: function(node, service) { │ │ │ │ │ + var onlineResource = this.getChildValue(node); │ │ │ │ │ + if (onlineResource && onlineResource.toLowerCase() != "none") { │ │ │ │ │ + service.onlineResource = onlineResource │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Keywords: function(node, service) { │ │ │ │ │ + var keywords = this.getChildValue(node); │ │ │ │ │ + if (keywords && keywords.toLowerCase() != "none") { │ │ │ │ │ + service.keywords = keywords.split(", ") │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Capability: function(node, capabilities) { │ │ │ │ │ + capabilities.capability = {}; │ │ │ │ │ + this.readChildNodes(node, capabilities.capability) │ │ │ │ │ + }, │ │ │ │ │ + Request: function(node, obj) { │ │ │ │ │ + obj.request = {}; │ │ │ │ │ + this.readChildNodes(node, obj.request) │ │ │ │ │ + }, │ │ │ │ │ + GetFeature: function(node, request) { │ │ │ │ │ + request.getfeature = { │ │ │ │ │ + href: {}, │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, request.getfeature) │ │ │ │ │ + }, │ │ │ │ │ + ResultFormat: function(node, obj) { │ │ │ │ │ + var children = node.childNodes; │ │ │ │ │ + var childNode; │ │ │ │ │ + for (var i = 0; i < children.length; i++) { │ │ │ │ │ + childNode = children[i]; │ │ │ │ │ + if (childNode.nodeType == 1) { │ │ │ │ │ + obj.formats.push(childNode.nodeName) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + DCPType: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + HTTP: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj.href) │ │ │ │ │ + }, │ │ │ │ │ + Get: function(node, obj) { │ │ │ │ │ + obj.get = node.getAttribute("onlineResource") │ │ │ │ │ + }, │ │ │ │ │ + Post: function(node, obj) { │ │ │ │ │ + obj.post = node.getAttribute("onlineResource") │ │ │ │ │ + }, │ │ │ │ │ + SRS: function(node, obj) { │ │ │ │ │ + var srs = this.getChildValue(node); │ │ │ │ │ + if (srs) { │ │ │ │ │ + obj.srs = srs │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WFSCapabilities.v1.prototype.readers["wfs"]) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1_0_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + namespaces: { │ │ │ │ │ + wms: "http://www.opengis.net/wms", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ + }, │ │ │ │ │ + defaultPrefix: "wms", │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + var raw = data; │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + if (capabilities.service === undefined) { │ │ │ │ │ + var parser = new OpenLayers.Format.OGCExceptionReport; │ │ │ │ │ + capabilities.error = parser.read(raw) │ │ │ │ │ + } │ │ │ │ │ + return capabilities │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + wms: { │ │ │ │ │ + Service: function(node, obj) { │ │ │ │ │ + obj.service = {}; │ │ │ │ │ + this.readChildNodes(node, obj.service) │ │ │ │ │ + }, │ │ │ │ │ + Name: function(node, obj) { │ │ │ │ │ + obj.name = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Title: function(node, obj) { │ │ │ │ │ + obj.title = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Abstract: function(node, obj) { │ │ │ │ │ + obj["abstract"] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + BoundingBox: function(node, obj) { │ │ │ │ │ + var bbox = {}; │ │ │ │ │ + bbox.bbox = [parseFloat(node.getAttribute("minx")), parseFloat(node.getAttribute("miny")), parseFloat(node.getAttribute("maxx")), parseFloat(node.getAttribute("maxy"))]; │ │ │ │ │ + var res = { │ │ │ │ │ + x: parseFloat(node.getAttribute("resx")), │ │ │ │ │ + y: parseFloat(node.getAttribute("resy")) │ │ │ │ │ + }; │ │ │ │ │ + if (!(isNaN(res.x) && isNaN(res.y))) { │ │ │ │ │ + bbox.res = res │ │ │ │ │ + } │ │ │ │ │ + return bbox │ │ │ │ │ + }, │ │ │ │ │ + OnlineResource: function(node, obj) { │ │ │ │ │ + obj.href = this.getAttributeNS(node, this.namespaces.xlink, "href") │ │ │ │ │ + }, │ │ │ │ │ + ContactInformation: function(node, obj) { │ │ │ │ │ + obj.contactInformation = {}; │ │ │ │ │ + this.readChildNodes(node, obj.contactInformation) │ │ │ │ │ + }, │ │ │ │ │ + ContactPersonPrimary: function(node, obj) { │ │ │ │ │ + obj.personPrimary = {}; │ │ │ │ │ + this.readChildNodes(node, obj.personPrimary) │ │ │ │ │ + }, │ │ │ │ │ + ContactPerson: function(node, obj) { │ │ │ │ │ + obj.person = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + ContactOrganization: function(node, obj) { │ │ │ │ │ + obj.organization = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + ContactPosition: function(node, obj) { │ │ │ │ │ + obj.position = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + ContactAddress: function(node, obj) { │ │ │ │ │ + obj.contactAddress = {}; │ │ │ │ │ + this.readChildNodes(node, obj.contactAddress) │ │ │ │ │ + }, │ │ │ │ │ + AddressType: function(node, obj) { │ │ │ │ │ + obj.type = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Address: function(node, obj) { │ │ │ │ │ + obj.address = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + City: function(node, obj) { │ │ │ │ │ + obj.city = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + StateOrProvince: function(node, obj) { │ │ │ │ │ + obj.stateOrProvince = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + PostCode: function(node, obj) { │ │ │ │ │ + obj.postcode = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Country: function(node, obj) { │ │ │ │ │ + obj.country = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + ContactVoiceTelephone: function(node, obj) { │ │ │ │ │ + obj.phone = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + ContactFacsimileTelephone: function(node, obj) { │ │ │ │ │ + obj.fax = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + ContactElectronicMailAddress: function(node, obj) { │ │ │ │ │ + obj.email = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Fees: function(node, obj) { │ │ │ │ │ + var fees = this.getChildValue(node); │ │ │ │ │ + if (fees && fees.toLowerCase() != "none") { │ │ │ │ │ + obj.fees = fees │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + AccessConstraints: function(node, obj) { │ │ │ │ │ + var constraints = this.getChildValue(node); │ │ │ │ │ + if (constraints && constraints.toLowerCase() != "none") { │ │ │ │ │ + obj.accessConstraints = constraints │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Capability: function(node, obj) { │ │ │ │ │ + obj.capability = { │ │ │ │ │ + nestedLayers: [], │ │ │ │ │ + layers: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.capability) │ │ │ │ │ + }, │ │ │ │ │ + Request: function(node, obj) { │ │ │ │ │ + obj.request = {}; │ │ │ │ │ + this.readChildNodes(node, obj.request) │ │ │ │ │ + }, │ │ │ │ │ + GetCapabilities: function(node, obj) { │ │ │ │ │ + obj.getcapabilities = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getcapabilities) │ │ │ │ │ + }, │ │ │ │ │ + Format: function(node, obj) { │ │ │ │ │ + if (OpenLayers.Util.isArray(obj.formats)) { │ │ │ │ │ + obj.formats.push(this.getChildValue(node)) │ │ │ │ │ + } else { │ │ │ │ │ + obj.format = this.getChildValue(node) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + DCPType: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + HTTP: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + Get: function(node, obj) { │ │ │ │ │ + obj.get = {}; │ │ │ │ │ + this.readChildNodes(node, obj.get); │ │ │ │ │ + if (!obj.href) { │ │ │ │ │ + obj.href = obj.get.href │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Post: function(node, obj) { │ │ │ │ │ + obj.post = {}; │ │ │ │ │ + this.readChildNodes(node, obj.post); │ │ │ │ │ + if (!obj.href) { │ │ │ │ │ + obj.href = obj.get.href │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + GetMap: function(node, obj) { │ │ │ │ │ + obj.getmap = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getmap) │ │ │ │ │ + }, │ │ │ │ │ + GetFeatureInfo: function(node, obj) { │ │ │ │ │ + obj.getfeatureinfo = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getfeatureinfo) │ │ │ │ │ + }, │ │ │ │ │ + Exception: function(node, obj) { │ │ │ │ │ + obj.exception = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.exception) │ │ │ │ │ + }, │ │ │ │ │ + Layer: function(node, obj) { │ │ │ │ │ + var parentLayer, capability; │ │ │ │ │ + if (obj.capability) { │ │ │ │ │ + capability = obj.capability; │ │ │ │ │ + parentLayer = obj │ │ │ │ │ + } else { │ │ │ │ │ + capability = obj │ │ │ │ │ + } │ │ │ │ │ + var attrNode = node.getAttributeNode("queryable"); │ │ │ │ │ + var queryable = attrNode && attrNode.specified ? node.getAttribute("queryable") : null; │ │ │ │ │ + attrNode = node.getAttributeNode("cascaded"); │ │ │ │ │ + var cascaded = attrNode && attrNode.specified ? node.getAttribute("cascaded") : null; │ │ │ │ │ + attrNode = node.getAttributeNode("opaque"); │ │ │ │ │ + var opaque = attrNode && attrNode.specified ? node.getAttribute("opaque") : null; │ │ │ │ │ + var noSubsets = node.getAttribute("noSubsets"); │ │ │ │ │ + var fixedWidth = node.getAttribute("fixedWidth"); │ │ │ │ │ + var fixedHeight = node.getAttribute("fixedHeight"); │ │ │ │ │ + var parent = parentLayer || {}, │ │ │ │ │ + extend = OpenLayers.Util.extend; │ │ │ │ │ + var layer = { │ │ │ │ │ + nestedLayers: [], │ │ │ │ │ + styles: parentLayer ? [].concat(parentLayer.styles) : [], │ │ │ │ │ + srs: parentLayer ? extend({}, parent.srs) : {}, │ │ │ │ │ + metadataURLs: [], │ │ │ │ │ + bbox: parentLayer ? extend({}, parent.bbox) : {}, │ │ │ │ │ + llbbox: parent.llbbox, │ │ │ │ │ + dimensions: parentLayer ? extend({}, parent.dimensions) : {}, │ │ │ │ │ + authorityURLs: parentLayer ? extend({}, parent.authorityURLs) : {}, │ │ │ │ │ + identifiers: {}, │ │ │ │ │ + keywords: [], │ │ │ │ │ + queryable: queryable && queryable !== "" ? queryable === "1" || queryable === "true" : parent.queryable || false, │ │ │ │ │ + cascaded: cascaded !== null ? parseInt(cascaded) : parent.cascaded || 0, │ │ │ │ │ + opaque: opaque ? opaque === "1" || opaque === "true" : parent.opaque || false, │ │ │ │ │ + noSubsets: noSubsets !== null ? noSubsets === "1" || noSubsets === "true" : parent.noSubsets || false, │ │ │ │ │ + fixedWidth: fixedWidth != null ? parseInt(fixedWidth) : parent.fixedWidth || 0, │ │ │ │ │ + fixedHeight: fixedHeight != null ? parseInt(fixedHeight) : parent.fixedHeight || 0, │ │ │ │ │ + minScale: parent.minScale, │ │ │ │ │ + maxScale: parent.maxScale, │ │ │ │ │ + attribution: parent.attribution │ │ │ │ │ + }; │ │ │ │ │ + obj.nestedLayers.push(layer); │ │ │ │ │ + layer.capability = capability; │ │ │ │ │ + this.readChildNodes(node, layer); │ │ │ │ │ + delete layer.capability; │ │ │ │ │ + if (layer.name) { │ │ │ │ │ + var parts = layer.name.split(":"), │ │ │ │ │ + request = capability.request, │ │ │ │ │ + gfi = request.getfeatureinfo; │ │ │ │ │ + if (parts.length > 0) { │ │ │ │ │ + layer.prefix = parts[0] │ │ │ │ │ + } │ │ │ │ │ + capability.layers.push(layer); │ │ │ │ │ + if (layer.formats === undefined) { │ │ │ │ │ + layer.formats = request.getmap.formats │ │ │ │ │ + } │ │ │ │ │ + if (layer.infoFormats === undefined && gfi) { │ │ │ │ │ + layer.infoFormats = gfi.formats │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Attribution: function(node, obj) { │ │ │ │ │ + obj.attribution = {}; │ │ │ │ │ + this.readChildNodes(node, obj.attribution) │ │ │ │ │ + }, │ │ │ │ │ + LogoURL: function(node, obj) { │ │ │ │ │ + obj.logo = { │ │ │ │ │ + width: node.getAttribute("width"), │ │ │ │ │ + height: node.getAttribute("height") │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.logo) │ │ │ │ │ + }, │ │ │ │ │ + Style: function(node, obj) { │ │ │ │ │ + var style = {}; │ │ │ │ │ + obj.styles.push(style); │ │ │ │ │ + this.readChildNodes(node, style) │ │ │ │ │ + }, │ │ │ │ │ + LegendURL: function(node, obj) { │ │ │ │ │ + var legend = { │ │ │ │ │ + width: node.getAttribute("width"), │ │ │ │ │ + height: node.getAttribute("height") │ │ │ │ │ + }; │ │ │ │ │ + obj.legend = legend; │ │ │ │ │ + this.readChildNodes(node, legend) │ │ │ │ │ + }, │ │ │ │ │ + MetadataURL: function(node, obj) { │ │ │ │ │ + var metadataURL = { │ │ │ │ │ + type: node.getAttribute("type") │ │ │ │ │ + }; │ │ │ │ │ + obj.metadataURLs.push(metadataURL); │ │ │ │ │ + this.readChildNodes(node, metadataURL) │ │ │ │ │ + }, │ │ │ │ │ + DataURL: function(node, obj) { │ │ │ │ │ + obj.dataURL = {}; │ │ │ │ │ + this.readChildNodes(node, obj.dataURL) │ │ │ │ │ + }, │ │ │ │ │ + FeatureListURL: function(node, obj) { │ │ │ │ │ + obj.featureListURL = {}; │ │ │ │ │ + this.readChildNodes(node, obj.featureListURL) │ │ │ │ │ + }, │ │ │ │ │ + AuthorityURL: function(node, obj) { │ │ │ │ │ + var name = node.getAttribute("name"); │ │ │ │ │ + var authority = {}; │ │ │ │ │ + this.readChildNodes(node, authority); │ │ │ │ │ + obj.authorityURLs[name] = authority.href │ │ │ │ │ + }, │ │ │ │ │ + Identifier: function(node, obj) { │ │ │ │ │ + var authority = node.getAttribute("authority"); │ │ │ │ │ + obj.identifiers[authority] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + KeywordList: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + SRS: function(node, obj) { │ │ │ │ │ + obj.srs[this.getChildValue(node)] = true │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_1 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1, { │ │ │ │ │ + readers: { │ │ │ │ │ + wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + WMT_MS_Capabilities: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + Keyword: function(node, obj) { │ │ │ │ │ + if (obj.keywords) { │ │ │ │ │ + obj.keywords.push(this.getChildValue(node)) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + DescribeLayer: function(node, obj) { │ │ │ │ │ + obj.describelayer = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.describelayer) │ │ │ │ │ + }, │ │ │ │ │ + GetLegendGraphic: function(node, obj) { │ │ │ │ │ + obj.getlegendgraphic = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getlegendgraphic) │ │ │ │ │ + }, │ │ │ │ │ + GetStyles: function(node, obj) { │ │ │ │ │ + obj.getstyles = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.getstyles) │ │ │ │ │ + }, │ │ │ │ │ + PutStyles: function(node, obj) { │ │ │ │ │ + obj.putstyles = { │ │ │ │ │ + formats: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.putstyles) │ │ │ │ │ + }, │ │ │ │ │ + UserDefinedSymbolization: function(node, obj) { │ │ │ │ │ + var userSymbols = { │ │ │ │ │ + supportSLD: parseInt(node.getAttribute("SupportSLD")) == 1, │ │ │ │ │ + userLayer: parseInt(node.getAttribute("UserLayer")) == 1, │ │ │ │ │ + userStyle: parseInt(node.getAttribute("UserStyle")) == 1, │ │ │ │ │ + remoteWFS: parseInt(node.getAttribute("RemoteWFS")) == 1 │ │ │ │ │ + }; │ │ │ │ │ + obj.userSymbols = userSymbols │ │ │ │ │ + }, │ │ │ │ │ + LatLonBoundingBox: function(node, obj) { │ │ │ │ │ + obj.llbbox = [parseFloat(node.getAttribute("minx")), parseFloat(node.getAttribute("miny")), parseFloat(node.getAttribute("maxx")), parseFloat(node.getAttribute("maxy"))] │ │ │ │ │ + }, │ │ │ │ │ + BoundingBox: function(node, obj) { │ │ │ │ │ + var bbox = OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"].BoundingBox.apply(this, [node, obj]); │ │ │ │ │ + bbox.srs = node.getAttribute("SRS"); │ │ │ │ │ + obj.bbox[bbox.srs] = bbox │ │ │ │ │ + }, │ │ │ │ │ + ScaleHint: function(node, obj) { │ │ │ │ │ + var min = node.getAttribute("min"); │ │ │ │ │ + var max = node.getAttribute("max"); │ │ │ │ │ + var rad2 = Math.pow(2, .5); │ │ │ │ │ + var ipm = OpenLayers.INCHES_PER_UNIT["m"]; │ │ │ │ │ + if (min != 0) { │ │ │ │ │ + obj.maxScale = parseFloat((min / rad2 * ipm * OpenLayers.DOTS_PER_INCH).toPrecision(13)) │ │ │ │ │ + } │ │ │ │ │ + if (max != Number.POSITIVE_INFINITY) { │ │ │ │ │ + obj.minScale = parseFloat((max / rad2 * ipm * OpenLayers.DOTS_PER_INCH).toPrecision(13)) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Dimension: function(node, obj) { │ │ │ │ │ + var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ + var dim = { │ │ │ │ │ + name: name, │ │ │ │ │ + units: node.getAttribute("units"), │ │ │ │ │ + unitsymbol: node.getAttribute("unitSymbol") │ │ │ │ │ + }; │ │ │ │ │ + obj.dimensions[dim.name] = dim │ │ │ │ │ + }, │ │ │ │ │ + Extent: function(node, obj) { │ │ │ │ │ + var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ + if (name in obj["dimensions"]) { │ │ │ │ │ + var extent = obj.dimensions[name]; │ │ │ │ │ + extent.nearestVal = node.getAttribute("nearestValue") === "1"; │ │ │ │ │ + extent.multipleVal = node.getAttribute("multipleValues") === "1"; │ │ │ │ │ + extent.current = node.getAttribute("current") === "1"; │ │ │ │ │ + extent["default"] = node.getAttribute("default") || ""; │ │ │ │ │ + var values = this.getChildValue(node); │ │ │ │ │ + extent.values = values.split(",") │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"]) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_1_0 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1_1, { │ │ │ │ │ + version: "1.1.0", │ │ │ │ │ + readers: { │ │ │ │ │ + wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + SRS: function(node, obj) { │ │ │ │ │ + var srs = this.getChildValue(node); │ │ │ │ │ + var values = srs.split(/ +/); │ │ │ │ │ + for (var i = 0, len = values.length; i < len; i++) { │ │ │ │ │ + obj.srs[values[i]] = true │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1_1.prototype.readers["wms"]) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_1_1 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1_1, { │ │ │ │ │ + version: "1.1.1", │ │ │ │ │ + readers: { │ │ │ │ │ + wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + SRS: function(node, obj) { │ │ │ │ │ + obj.srs[this.getChildValue(node)] = true │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1_1.prototype.readers["wms"]) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_1" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_1_1_WMSC = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1_1_1, { │ │ │ │ │ + version: "1.1.1", │ │ │ │ │ + profile: "WMSC", │ │ │ │ │ + readers: { │ │ │ │ │ + wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + VendorSpecificCapabilities: function(node, obj) { │ │ │ │ │ + obj.vendorSpecific = { │ │ │ │ │ + tileSets: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, obj.vendorSpecific) │ │ │ │ │ + }, │ │ │ │ │ + TileSet: function(node, vendorSpecific) { │ │ │ │ │ + var tileset = { │ │ │ │ │ + srs: {}, │ │ │ │ │ + bbox: {}, │ │ │ │ │ + resolutions: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, tileset); │ │ │ │ │ + vendorSpecific.tileSets.push(tileset) │ │ │ │ │ + }, │ │ │ │ │ + Resolutions: function(node, tileset) { │ │ │ │ │ + var res = this.getChildValue(node).split(" "); │ │ │ │ │ + for (var i = 0, len = res.length; i < len; i++) { │ │ │ │ │ + if (res[i] != "") { │ │ │ │ │ + tileset.resolutions.push(parseFloat(res[i])) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + Width: function(node, tileset) { │ │ │ │ │ + tileset.width = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + Height: function(node, tileset) { │ │ │ │ │ + tileset.height = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + Layers: function(node, tileset) { │ │ │ │ │ + tileset.layers = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Styles: function(node, tileset) { │ │ │ │ │ + tileset.styles = this.getChildValue(node) │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1_1_1.prototype.readers["wms"]) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_1_WMSC" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_3 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1, { │ │ │ │ │ + readers: { │ │ │ │ │ + wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + WMS_Capabilities: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + LayerLimit: function(node, obj) { │ │ │ │ │ + obj.layerLimit = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + MaxWidth: function(node, obj) { │ │ │ │ │ + obj.maxWidth = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + MaxHeight: function(node, obj) { │ │ │ │ │ + obj.maxHeight = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + BoundingBox: function(node, obj) { │ │ │ │ │ + var bbox = OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"].BoundingBox.apply(this, [node, obj]); │ │ │ │ │ + bbox.srs = node.getAttribute("CRS"); │ │ │ │ │ + obj.bbox[bbox.srs] = bbox │ │ │ │ │ + }, │ │ │ │ │ + CRS: function(node, obj) { │ │ │ │ │ + this.readers.wms.SRS.apply(this, [node, obj]) │ │ │ │ │ + }, │ │ │ │ │ + EX_GeographicBoundingBox: function(node, obj) { │ │ │ │ │ + obj.llbbox = []; │ │ │ │ │ + this.readChildNodes(node, obj.llbbox) │ │ │ │ │ + }, │ │ │ │ │ + westBoundLongitude: function(node, obj) { │ │ │ │ │ + obj[0] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + eastBoundLongitude: function(node, obj) { │ │ │ │ │ + obj[2] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + southBoundLatitude: function(node, obj) { │ │ │ │ │ + obj[1] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + northBoundLatitude: function(node, obj) { │ │ │ │ │ + obj[3] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + MinScaleDenominator: function(node, obj) { │ │ │ │ │ + obj.maxScale = parseFloat(this.getChildValue(node)).toPrecision(16) │ │ │ │ │ + }, │ │ │ │ │ + MaxScaleDenominator: function(node, obj) { │ │ │ │ │ + obj.minScale = parseFloat(this.getChildValue(node)).toPrecision(16) │ │ │ │ │ + }, │ │ │ │ │ + Dimension: function(node, obj) { │ │ │ │ │ + var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ + var dim = { │ │ │ │ │ + name: name, │ │ │ │ │ + units: node.getAttribute("units"), │ │ │ │ │ + unitsymbol: node.getAttribute("unitSymbol"), │ │ │ │ │ + nearestVal: node.getAttribute("nearestValue") === "1", │ │ │ │ │ + multipleVal: node.getAttribute("multipleValues") === "1", │ │ │ │ │ + default: node.getAttribute("default") || "", │ │ │ │ │ + current: node.getAttribute("current") === "1", │ │ │ │ │ + values: this.getChildValue(node).split(",") │ │ │ │ │ + }; │ │ │ │ │ + obj.dimensions[dim.name] = dim │ │ │ │ │ + }, │ │ │ │ │ + Keyword: function(node, obj) { │ │ │ │ │ + var keyword = { │ │ │ │ │ + value: this.getChildValue(node), │ │ │ │ │ + vocabulary: node.getAttribute("vocabulary") │ │ │ │ │ + }; │ │ │ │ │ + if (obj.keywords) { │ │ │ │ │ + obj.keywords.push(keyword) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"]), │ │ │ │ │ + sld: { │ │ │ │ │ + UserDefinedSymbolization: function(node, obj) { │ │ │ │ │ + this.readers.wms.UserDefinedSymbolization.apply(this, [node, obj]); │ │ │ │ │ + obj.userSymbols.inlineFeature = parseInt(node.getAttribute("InlineFeature")) == 1; │ │ │ │ │ + obj.userSymbols.remoteWCS = parseInt(node.getAttribute("RemoteWCS")) == 1 │ │ │ │ │ + }, │ │ │ │ │ + DescribeLayer: function(node, obj) { │ │ │ │ │ + this.readers.wms.DescribeLayer.apply(this, [node, obj]) │ │ │ │ │ + }, │ │ │ │ │ + GetLegendGraphic: function(node, obj) { │ │ │ │ │ + this.readers.wms.GetLegendGraphic.apply(this, [node, obj]) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_3" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSCapabilities.v1_3_0 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1_3, { │ │ │ │ │ + version: "1.3.0", │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_3_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMTSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.OWSCommon.v1_1_0, { │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + namespaces: { │ │ │ │ │ + ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ + wmts: "http://www.opengis.net/wmts/1.0", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ + }, │ │ │ │ │ + yx: null, │ │ │ │ │ + defaultPrefix: "wmts", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.options = options; │ │ │ │ │ + var yx = OpenLayers.Util.extend({}, OpenLayers.Format.WMTSCapabilities.prototype.yx); │ │ │ │ │ + this.yx = OpenLayers.Util.extend(yx, this.yx) │ │ │ │ │ + }, │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + capabilities.version = this.version; │ │ │ │ │ + return capabilities │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + wmts: { │ │ │ │ │ + Capabilities: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + Contents: function(node, obj) { │ │ │ │ │ + obj.contents = {}; │ │ │ │ │ + obj.contents.layers = []; │ │ │ │ │ + obj.contents.tileMatrixSets = {}; │ │ │ │ │ + this.readChildNodes(node, obj.contents) │ │ │ │ │ + }, │ │ │ │ │ + Layer: function(node, obj) { │ │ │ │ │ + var layer = { │ │ │ │ │ + styles: [], │ │ │ │ │ + formats: [], │ │ │ │ │ + dimensions: [], │ │ │ │ │ + tileMatrixSetLinks: [] │ │ │ │ │ + }; │ │ │ │ │ + layer.layers = []; │ │ │ │ │ + this.readChildNodes(node, layer); │ │ │ │ │ + obj.layers.push(layer) │ │ │ │ │ + }, │ │ │ │ │ + Style: function(node, obj) { │ │ │ │ │ + var style = {}; │ │ │ │ │ + style.isDefault = node.getAttribute("isDefault") === "true"; │ │ │ │ │ + this.readChildNodes(node, style); │ │ │ │ │ + obj.styles.push(style) │ │ │ │ │ + }, │ │ │ │ │ + Format: function(node, obj) { │ │ │ │ │ + obj.formats.push(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + TileMatrixSetLink: function(node, obj) { │ │ │ │ │ + var tileMatrixSetLink = {}; │ │ │ │ │ + this.readChildNodes(node, tileMatrixSetLink); │ │ │ │ │ + obj.tileMatrixSetLinks.push(tileMatrixSetLink) │ │ │ │ │ + }, │ │ │ │ │ + TileMatrixSet: function(node, obj) { │ │ │ │ │ + if (obj.layers) { │ │ │ │ │ + var tileMatrixSet = { │ │ │ │ │ + matrixIds: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, tileMatrixSet); │ │ │ │ │ + obj.tileMatrixSets[tileMatrixSet.identifier] = tileMatrixSet │ │ │ │ │ + } else { │ │ │ │ │ + obj.tileMatrixSet = this.getChildValue(node) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + TileMatrix: function(node, obj) { │ │ │ │ │ + var tileMatrix = { │ │ │ │ │ + supportedCRS: obj.supportedCRS │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, tileMatrix); │ │ │ │ │ + obj.matrixIds.push(tileMatrix) │ │ │ │ │ + }, │ │ │ │ │ + ScaleDenominator: function(node, obj) { │ │ │ │ │ + obj.scaleDenominator = parseFloat(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + TopLeftCorner: function(node, obj) { │ │ │ │ │ + var topLeftCorner = this.getChildValue(node); │ │ │ │ │ + var coords = topLeftCorner.split(" "); │ │ │ │ │ + var yx; │ │ │ │ │ + if (obj.supportedCRS) { │ │ │ │ │ + var crs = obj.supportedCRS.replace(/urn:ogc:def:crs:(\w+):.+:(\w+)$/, "urn:ogc:def:crs:$1::$2"); │ │ │ │ │ + yx = !!this.yx[crs] │ │ │ │ │ + } │ │ │ │ │ + if (yx) { │ │ │ │ │ + obj.topLeftCorner = new OpenLayers.LonLat(coords[1], coords[0]) │ │ │ │ │ + } else { │ │ │ │ │ + obj.topLeftCorner = new OpenLayers.LonLat(coords[0], coords[1]) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + TileWidth: function(node, obj) { │ │ │ │ │ + obj.tileWidth = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + TileHeight: function(node, obj) { │ │ │ │ │ + obj.tileHeight = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + MatrixWidth: function(node, obj) { │ │ │ │ │ + obj.matrixWidth = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + MatrixHeight: function(node, obj) { │ │ │ │ │ + obj.matrixHeight = parseInt(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + ResourceURL: function(node, obj) { │ │ │ │ │ + obj.resourceUrl = obj.resourceUrl || {}; │ │ │ │ │ + var resourceType = node.getAttribute("resourceType"); │ │ │ │ │ + if (!obj.resourceUrls) { │ │ │ │ │ + obj.resourceUrls = [] │ │ │ │ │ + } │ │ │ │ │ + var resourceUrl = obj.resourceUrl[resourceType] = { │ │ │ │ │ + format: node.getAttribute("format"), │ │ │ │ │ + template: node.getAttribute("template"), │ │ │ │ │ + resourceType: resourceType │ │ │ │ │ + }; │ │ │ │ │ + obj.resourceUrls.push(resourceUrl) │ │ │ │ │ + }, │ │ │ │ │ + WSDL: function(node, obj) { │ │ │ │ │ + obj.wsdl = {}; │ │ │ │ │ + obj.wsdl.href = node.getAttribute("xlink:href") │ │ │ │ │ + }, │ │ │ │ │ + ServiceMetadataURL: function(node, obj) { │ │ │ │ │ + obj.serviceMetadataUrl = {}; │ │ │ │ │ + obj.serviceMetadataUrl.href = node.getAttribute("xlink:href") │ │ │ │ │ + }, │ │ │ │ │ + LegendURL: function(node, obj) { │ │ │ │ │ + obj.legend = {}; │ │ │ │ │ + obj.legend.href = node.getAttribute("xlink:href"); │ │ │ │ │ + obj.legend.format = node.getAttribute("format") │ │ │ │ │ + }, │ │ │ │ │ + Dimension: function(node, obj) { │ │ │ │ │ + var dimension = { │ │ │ │ │ + values: [] │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, dimension); │ │ │ │ │ + obj.dimensions.push(dimension) │ │ │ │ │ + }, │ │ │ │ │ + Default: function(node, obj) { │ │ │ │ │ + obj["default"] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Value: function(node, obj) { │ │ │ │ │ + obj.values.push(this.getChildValue(node)) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + ows: OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMTSCapabilities.v1_0_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WPSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + namespaces: { │ │ │ │ │ + ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ + wps: "http://www.opengis.net/wps/1.0.0", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ + }, │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: /^\s*|\s*$/g, │ │ │ │ │ + removeSpace: /\s*/g, │ │ │ │ │ + splitSpace: /\s+/, │ │ │ │ │ + trimComma: /\s*,\s*/g │ │ │ │ │ + }, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + return capabilities │ │ │ │ │ + }, │ │ │ │ │ + readers: { │ │ │ │ │ + wps: { │ │ │ │ │ + Capabilities: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + ProcessOfferings: function(node, obj) { │ │ │ │ │ + obj.processOfferings = {}; │ │ │ │ │ + this.readChildNodes(node, obj.processOfferings) │ │ │ │ │ + }, │ │ │ │ │ + Process: function(node, processOfferings) { │ │ │ │ │ + var processVersion = this.getAttributeNS(node, this.namespaces.wps, "processVersion"); │ │ │ │ │ + var process = { │ │ │ │ │ + processVersion: processVersion │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, process); │ │ │ │ │ + processOfferings[process.identifier] = process │ │ │ │ │ + }, │ │ │ │ │ + Languages: function(node, obj) { │ │ │ │ │ + obj.languages = []; │ │ │ │ │ + this.readChildNodes(node, obj.languages) │ │ │ │ │ + }, │ │ │ │ │ + Default: function(node, languages) { │ │ │ │ │ + var language = { │ │ │ │ │ + isDefault: true │ │ │ │ │ + }; │ │ │ │ │ + this.readChildNodes(node, language); │ │ │ │ │ + languages.push(language) │ │ │ │ │ + }, │ │ │ │ │ + Supported: function(node, languages) { │ │ │ │ │ + var language = {}; │ │ │ │ │ + this.readChildNodes(node, language); │ │ │ │ │ + languages.push(language) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + ows: OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WPSCapabilities.v1_0_0" │ │ │ │ │ +}); │ │ │ │ │ OpenLayers.Format.OWSContext.v0_3_1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ namespaces: { │ │ │ │ │ owc: "http://www.opengis.net/ows-context", │ │ │ │ │ gml: "http://www.opengis.net/gml", │ │ │ │ │ kml: "http://www.opengis.net/kml/2.2", │ │ │ │ │ ogc: "http://www.opengis.net/ogc", │ │ │ │ │ ows: "http://www.opengis.net/ows", │ │ │ │ │ @@ -27193,14 +30421,264 @@ │ │ │ │ │ }, OpenLayers.Format.GML.v2.prototype.writers.gml), │ │ │ │ │ ows: OpenLayers.Format.OWSCommon.v1_0_0.prototype.writers.ows, │ │ │ │ │ sld: OpenLayers.Format.SLD.v1_0_0.prototype.writers.sld, │ │ │ │ │ feature: OpenLayers.Format.GML.v2.prototype.writers.feature │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.OWSContext.v0_3_1" │ │ │ │ │ }); │ │ │ │ │ +OpenLayers.Format.WMSDescribeLayer.v1_1_1 = OpenLayers.Class(OpenLayers.Format.WMSDescribeLayer, { │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.WMSDescribeLayer.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + var root = data.documentElement; │ │ │ │ │ + var children = root.childNodes; │ │ │ │ │ + var describelayer = { │ │ │ │ │ + layerDescriptions: [] │ │ │ │ │ + }; │ │ │ │ │ + var childNode, nodeName; │ │ │ │ │ + for (var i = 0; i < children.length; ++i) { │ │ │ │ │ + childNode = children[i]; │ │ │ │ │ + nodeName = childNode.nodeName; │ │ │ │ │ + if (nodeName == "LayerDescription") { │ │ │ │ │ + var layerName = childNode.getAttribute("name"); │ │ │ │ │ + var owsType = ""; │ │ │ │ │ + var owsURL = ""; │ │ │ │ │ + var typeName = ""; │ │ │ │ │ + if (childNode.getAttribute("owsType")) { │ │ │ │ │ + owsType = childNode.getAttribute("owsType"); │ │ │ │ │ + owsURL = childNode.getAttribute("owsURL") │ │ │ │ │ + } else { │ │ │ │ │ + if (childNode.getAttribute("wfs") != "") { │ │ │ │ │ + owsType = "WFS"; │ │ │ │ │ + owsURL = childNode.getAttribute("wfs") │ │ │ │ │ + } else if (childNode.getAttribute("wcs") != "") { │ │ │ │ │ + owsType = "WCS"; │ │ │ │ │ + owsURL = childNode.getAttribute("wcs") │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var query = childNode.getElementsByTagName("Query"); │ │ │ │ │ + if (query.length > 0) { │ │ │ │ │ + typeName = query[0].getAttribute("typeName"); │ │ │ │ │ + if (!typeName) { │ │ │ │ │ + typeName = query[0].getAttribute("typename") │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var layerDescription = { │ │ │ │ │ + layerName: layerName, │ │ │ │ │ + owsType: owsType, │ │ │ │ │ + owsURL: owsURL, │ │ │ │ │ + typeName: typeName │ │ │ │ │ + }; │ │ │ │ │ + describelayer.layerDescriptions.push(layerDescription); │ │ │ │ │ + describelayer.length = describelayer.layerDescriptions.length; │ │ │ │ │ + describelayer[describelayer.length - 1] = layerDescription │ │ │ │ │ + } else if (nodeName == "ServiceException") { │ │ │ │ │ + var parser = new OpenLayers.Format.OGCExceptionReport; │ │ │ │ │ + return { │ │ │ │ │ + error: parser.read(data) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return describelayer │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMSDescribeLayer.v1_1_1" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WMSDescribeLayer.v1_1_0 = OpenLayers.Format.WMSDescribeLayer.v1_1_1; │ │ │ │ │ +OpenLayers.Format.WCSCapabilities.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ + regExes: { │ │ │ │ │ + trimSpace: /^\s*|\s*$/g, │ │ │ │ │ + splitSpace: /\s+/ │ │ │ │ │ + }, │ │ │ │ │ + defaultPrefix: "wcs", │ │ │ │ │ + read: function(data) { │ │ │ │ │ + if (typeof data == "string") { │ │ │ │ │ + data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + } │ │ │ │ │ + var raw = data; │ │ │ │ │ + if (data && data.nodeType == 9) { │ │ │ │ │ + data = data.documentElement │ │ │ │ │ + } │ │ │ │ │ + var capabilities = {}; │ │ │ │ │ + this.readNode(data, capabilities); │ │ │ │ │ + return capabilities │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WCSCapabilities.v1_1_0 = OpenLayers.Class(OpenLayers.Format.WCSCapabilities.v1, { │ │ │ │ │ + namespaces: { │ │ │ │ │ + wcs: "http://www.opengis.net/wcs/1.1", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + ows: "http://www.opengis.net/ows/1.1" │ │ │ │ │ + }, │ │ │ │ │ + errorProperty: "operationsMetadata", │ │ │ │ │ + readers: { │ │ │ │ │ + wcs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ + Capabilities: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + Contents: function(node, request) { │ │ │ │ │ + request.contentMetadata = []; │ │ │ │ │ + this.readChildNodes(node, request.contentMetadata) │ │ │ │ │ + }, │ │ │ │ │ + CoverageSummary: function(node, contentMetadata) { │ │ │ │ │ + var coverageSummary = {}; │ │ │ │ │ + this.readChildNodes(node, coverageSummary); │ │ │ │ │ + contentMetadata.push(coverageSummary) │ │ │ │ │ + }, │ │ │ │ │ + Identifier: function(node, coverageSummary) { │ │ │ │ │ + coverageSummary.identifier = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Title: function(node, coverageSummary) { │ │ │ │ │ + coverageSummary.title = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + Abstract: function(node, coverageSummary) { │ │ │ │ │ + coverageSummary["abstract"] = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + SupportedCRS: function(node, coverageSummary) { │ │ │ │ │ + var crs = this.getChildValue(node); │ │ │ │ │ + if (crs) { │ │ │ │ │ + if (!coverageSummary.supportedCRS) { │ │ │ │ │ + coverageSummary.supportedCRS = [] │ │ │ │ │ + } │ │ │ │ │ + coverageSummary.supportedCRS.push(crs) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + SupportedFormat: function(node, coverageSummary) { │ │ │ │ │ + var format = this.getChildValue(node); │ │ │ │ │ + if (format) { │ │ │ │ │ + if (!coverageSummary.supportedFormat) { │ │ │ │ │ + coverageSummary.supportedFormat = [] │ │ │ │ │ + } │ │ │ │ │ + coverageSummary.supportedFormat.push(format) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, OpenLayers.Format.WCSCapabilities.v1.prototype.readers["wcs"]), │ │ │ │ │ + ows: OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1_1_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Format.WCSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.WCSCapabilities.v1, { │ │ │ │ │ + namespaces: { │ │ │ │ │ + wcs: "http://www.opengis.net/wcs", │ │ │ │ │ + xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ + xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ + ows: "http://www.opengis.net/ows" │ │ │ │ │ + }, │ │ │ │ │ + errorProperty: "service", │ │ │ │ │ + readers: { │ │ │ │ │ + wcs: { │ │ │ │ │ + WCS_Capabilities: function(node, obj) { │ │ │ │ │ + this.readChildNodes(node, obj) │ │ │ │ │ + }, │ │ │ │ │ + Service: function(node, obj) { │ │ │ │ │ + obj.service = {}; │ │ │ │ │ + this.readChildNodes(node, obj.service) │ │ │ │ │ + }, │ │ │ │ │ + name: function(node, service) { │ │ │ │ │ + service.name = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + label: function(node, service) { │ │ │ │ │ + service.label = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + keywords: function(node, service) { │ │ │ │ │ + service.keywords = []; │ │ │ │ │ + this.readChildNodes(node, service.keywords) │ │ │ │ │ + }, │ │ │ │ │ + keyword: function(node, keywords) { │ │ │ │ │ + keywords.push(this.getChildValue(node)) │ │ │ │ │ + }, │ │ │ │ │ + responsibleParty: function(node, service) { │ │ │ │ │ + service.responsibleParty = {}; │ │ │ │ │ + this.readChildNodes(node, service.responsibleParty) │ │ │ │ │ + }, │ │ │ │ │ + individualName: function(node, responsibleParty) { │ │ │ │ │ + responsibleParty.individualName = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + organisationName: function(node, responsibleParty) { │ │ │ │ │ + responsibleParty.organisationName = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + positionName: function(node, responsibleParty) { │ │ │ │ │ + responsibleParty.positionName = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + contactInfo: function(node, responsibleParty) { │ │ │ │ │ + responsibleParty.contactInfo = {}; │ │ │ │ │ + this.readChildNodes(node, responsibleParty.contactInfo) │ │ │ │ │ + }, │ │ │ │ │ + phone: function(node, contactInfo) { │ │ │ │ │ + contactInfo.phone = {}; │ │ │ │ │ + this.readChildNodes(node, contactInfo.phone) │ │ │ │ │ + }, │ │ │ │ │ + voice: function(node, phone) { │ │ │ │ │ + phone.voice = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + facsimile: function(node, phone) { │ │ │ │ │ + phone.facsimile = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + address: function(node, contactInfo) { │ │ │ │ │ + contactInfo.address = {}; │ │ │ │ │ + this.readChildNodes(node, contactInfo.address) │ │ │ │ │ + }, │ │ │ │ │ + deliveryPoint: function(node, address) { │ │ │ │ │ + address.deliveryPoint = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + city: function(node, address) { │ │ │ │ │ + address.city = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + postalCode: function(node, address) { │ │ │ │ │ + address.postalCode = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + country: function(node, address) { │ │ │ │ │ + address.country = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + electronicMailAddress: function(node, address) { │ │ │ │ │ + address.electronicMailAddress = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + fees: function(node, service) { │ │ │ │ │ + service.fees = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + accessConstraints: function(node, service) { │ │ │ │ │ + service.accessConstraints = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + ContentMetadata: function(node, obj) { │ │ │ │ │ + obj.contentMetadata = []; │ │ │ │ │ + this.readChildNodes(node, obj.contentMetadata) │ │ │ │ │ + }, │ │ │ │ │ + CoverageOfferingBrief: function(node, contentMetadata) { │ │ │ │ │ + var coverageOfferingBrief = {}; │ │ │ │ │ + this.readChildNodes(node, coverageOfferingBrief); │ │ │ │ │ + contentMetadata.push(coverageOfferingBrief) │ │ │ │ │ + }, │ │ │ │ │ + name: function(node, coverageOfferingBrief) { │ │ │ │ │ + coverageOfferingBrief.name = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + label: function(node, coverageOfferingBrief) { │ │ │ │ │ + coverageOfferingBrief.label = this.getChildValue(node) │ │ │ │ │ + }, │ │ │ │ │ + lonLatEnvelope: function(node, coverageOfferingBrief) { │ │ │ │ │ + var nodeList = this.getElementsByTagNameNS(node, "http://www.opengis.net/gml", "pos"); │ │ │ │ │ + if (nodeList.length == 2) { │ │ │ │ │ + var min = {}; │ │ │ │ │ + var max = {}; │ │ │ │ │ + OpenLayers.Format.GML.v3.prototype.readers["gml"].pos.apply(this, [nodeList[0], min]); │ │ │ │ │ + OpenLayers.Format.GML.v3.prototype.readers["gml"].pos.apply(this, [nodeList[1], max]); │ │ │ │ │ + coverageOfferingBrief.lonLatEnvelope = {}; │ │ │ │ │ + coverageOfferingBrief.lonLatEnvelope.srsName = node.getAttribute("srsName"); │ │ │ │ │ + coverageOfferingBrief.lonLatEnvelope.min = min.points[0]; │ │ │ │ │ + coverageOfferingBrief.lonLatEnvelope.max = max.points[0] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1_0_0" │ │ │ │ │ +}); │ │ │ │ │ OpenLayers.Format.XLS.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ namespaces: { │ │ │ │ │ xls: "http://www.opengis.net/xls", │ │ │ │ │ gml: "http://www.opengis.net/gml", │ │ │ │ │ xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ }, │ │ │ │ │ regExes: { │ │ │ │ │ @@ -27407,789 +30885,14 @@ │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Format.XLS.v1_1_0 = OpenLayers.Class(OpenLayers.Format.XLS.v1, { │ │ │ │ │ VERSION: "1.1", │ │ │ │ │ schemaLocation: "http://www.opengis.net/xls http://schemas.opengis.net/ols/1.1.0/LocationUtilityService.xsd", │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.XLS.v1_1_0" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Format.XLS.v1_1 = OpenLayers.Format.XLS.v1_1_0; │ │ │ │ │ -OpenLayers.Format.WCSCapabilities.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: /^\s*|\s*$/g, │ │ │ │ │ - splitSpace: /\s+/ │ │ │ │ │ - }, │ │ │ │ │ - defaultPrefix: "wcs", │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ - } │ │ │ │ │ - var raw = data; │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement │ │ │ │ │ - } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - return capabilities │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WCSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.WCSCapabilities.v1, { │ │ │ │ │ - namespaces: { │ │ │ │ │ - wcs: "http://www.opengis.net/wcs", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - ows: "http://www.opengis.net/ows" │ │ │ │ │ - }, │ │ │ │ │ - errorProperty: "service", │ │ │ │ │ - readers: { │ │ │ │ │ - wcs: { │ │ │ │ │ - WCS_Capabilities: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - Service: function(node, obj) { │ │ │ │ │ - obj.service = {}; │ │ │ │ │ - this.readChildNodes(node, obj.service) │ │ │ │ │ - }, │ │ │ │ │ - name: function(node, service) { │ │ │ │ │ - service.name = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - label: function(node, service) { │ │ │ │ │ - service.label = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - keywords: function(node, service) { │ │ │ │ │ - service.keywords = []; │ │ │ │ │ - this.readChildNodes(node, service.keywords) │ │ │ │ │ - }, │ │ │ │ │ - keyword: function(node, keywords) { │ │ │ │ │ - keywords.push(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - responsibleParty: function(node, service) { │ │ │ │ │ - service.responsibleParty = {}; │ │ │ │ │ - this.readChildNodes(node, service.responsibleParty) │ │ │ │ │ - }, │ │ │ │ │ - individualName: function(node, responsibleParty) { │ │ │ │ │ - responsibleParty.individualName = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - organisationName: function(node, responsibleParty) { │ │ │ │ │ - responsibleParty.organisationName = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - positionName: function(node, responsibleParty) { │ │ │ │ │ - responsibleParty.positionName = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - contactInfo: function(node, responsibleParty) { │ │ │ │ │ - responsibleParty.contactInfo = {}; │ │ │ │ │ - this.readChildNodes(node, responsibleParty.contactInfo) │ │ │ │ │ - }, │ │ │ │ │ - phone: function(node, contactInfo) { │ │ │ │ │ - contactInfo.phone = {}; │ │ │ │ │ - this.readChildNodes(node, contactInfo.phone) │ │ │ │ │ - }, │ │ │ │ │ - voice: function(node, phone) { │ │ │ │ │ - phone.voice = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - facsimile: function(node, phone) { │ │ │ │ │ - phone.facsimile = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - address: function(node, contactInfo) { │ │ │ │ │ - contactInfo.address = {}; │ │ │ │ │ - this.readChildNodes(node, contactInfo.address) │ │ │ │ │ - }, │ │ │ │ │ - deliveryPoint: function(node, address) { │ │ │ │ │ - address.deliveryPoint = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - city: function(node, address) { │ │ │ │ │ - address.city = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - postalCode: function(node, address) { │ │ │ │ │ - address.postalCode = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - country: function(node, address) { │ │ │ │ │ - address.country = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - electronicMailAddress: function(node, address) { │ │ │ │ │ - address.electronicMailAddress = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - fees: function(node, service) { │ │ │ │ │ - service.fees = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - accessConstraints: function(node, service) { │ │ │ │ │ - service.accessConstraints = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ContentMetadata: function(node, obj) { │ │ │ │ │ - obj.contentMetadata = []; │ │ │ │ │ - this.readChildNodes(node, obj.contentMetadata) │ │ │ │ │ - }, │ │ │ │ │ - CoverageOfferingBrief: function(node, contentMetadata) { │ │ │ │ │ - var coverageOfferingBrief = {}; │ │ │ │ │ - this.readChildNodes(node, coverageOfferingBrief); │ │ │ │ │ - contentMetadata.push(coverageOfferingBrief) │ │ │ │ │ - }, │ │ │ │ │ - name: function(node, coverageOfferingBrief) { │ │ │ │ │ - coverageOfferingBrief.name = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - label: function(node, coverageOfferingBrief) { │ │ │ │ │ - coverageOfferingBrief.label = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - lonLatEnvelope: function(node, coverageOfferingBrief) { │ │ │ │ │ - var nodeList = this.getElementsByTagNameNS(node, "http://www.opengis.net/gml", "pos"); │ │ │ │ │ - if (nodeList.length == 2) { │ │ │ │ │ - var min = {}; │ │ │ │ │ - var max = {}; │ │ │ │ │ - OpenLayers.Format.GML.v3.prototype.readers["gml"].pos.apply(this, [nodeList[0], min]); │ │ │ │ │ - OpenLayers.Format.GML.v3.prototype.readers["gml"].pos.apply(this, [nodeList[1], max]); │ │ │ │ │ - coverageOfferingBrief.lonLatEnvelope = {}; │ │ │ │ │ - coverageOfferingBrief.lonLatEnvelope.srsName = node.getAttribute("srsName"); │ │ │ │ │ - coverageOfferingBrief.lonLatEnvelope.min = min.points[0]; │ │ │ │ │ - coverageOfferingBrief.lonLatEnvelope.max = max.points[0] │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1_0_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WCSCapabilities.v1_1_0 = OpenLayers.Class(OpenLayers.Format.WCSCapabilities.v1, { │ │ │ │ │ - namespaces: { │ │ │ │ │ - wcs: "http://www.opengis.net/wcs/1.1", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - ows: "http://www.opengis.net/ows/1.1" │ │ │ │ │ - }, │ │ │ │ │ - errorProperty: "operationsMetadata", │ │ │ │ │ - readers: { │ │ │ │ │ - wcs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Capabilities: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - Contents: function(node, request) { │ │ │ │ │ - request.contentMetadata = []; │ │ │ │ │ - this.readChildNodes(node, request.contentMetadata) │ │ │ │ │ - }, │ │ │ │ │ - CoverageSummary: function(node, contentMetadata) { │ │ │ │ │ - var coverageSummary = {}; │ │ │ │ │ - this.readChildNodes(node, coverageSummary); │ │ │ │ │ - contentMetadata.push(coverageSummary) │ │ │ │ │ - }, │ │ │ │ │ - Identifier: function(node, coverageSummary) { │ │ │ │ │ - coverageSummary.identifier = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Title: function(node, coverageSummary) { │ │ │ │ │ - coverageSummary.title = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Abstract: function(node, coverageSummary) { │ │ │ │ │ - coverageSummary["abstract"] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - SupportedCRS: function(node, coverageSummary) { │ │ │ │ │ - var crs = this.getChildValue(node); │ │ │ │ │ - if (crs) { │ │ │ │ │ - if (!coverageSummary.supportedCRS) { │ │ │ │ │ - coverageSummary.supportedCRS = [] │ │ │ │ │ - } │ │ │ │ │ - coverageSummary.supportedCRS.push(crs) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - SupportedFormat: function(node, coverageSummary) { │ │ │ │ │ - var format = this.getChildValue(node); │ │ │ │ │ - if (format) { │ │ │ │ │ - if (!coverageSummary.supportedFormat) { │ │ │ │ │ - coverageSummary.supportedFormat = [] │ │ │ │ │ - } │ │ │ │ │ - coverageSummary.supportedFormat.push(format) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WCSCapabilities.v1.prototype.readers["wcs"]), │ │ │ │ │ - ows: OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WCSCapabilities.v1_1_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.SLD.v1_0_0_GeoServer = OpenLayers.Class(OpenLayers.Format.SLD.v1_0_0, { │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - profile: "GeoServer", │ │ │ │ │ - readers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - sld: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Priority: function(node, obj) { │ │ │ │ │ - var value = this.readers.ogc._expression.call(this, node); │ │ │ │ │ - if (value) { │ │ │ │ │ - obj.priority = value │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - VendorOption: function(node, obj) { │ │ │ │ │ - if (!obj.vendorOptions) { │ │ │ │ │ - obj.vendorOptions = {} │ │ │ │ │ - } │ │ │ │ │ - obj.vendorOptions[node.getAttribute("name")] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - TextSymbolizer: function(node, rule) { │ │ │ │ │ - OpenLayers.Format.SLD.v1_0_0.prototype.readers.sld.TextSymbolizer.apply(this, arguments); │ │ │ │ │ - var symbolizer = this.multipleSymbolizers ? rule.symbolizers[rule.symbolizers.length - 1] : rule.symbolizer["Text"]; │ │ │ │ │ - if (symbolizer.graphic === undefined) { │ │ │ │ │ - symbolizer.graphic = false │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.SLD.v1_0_0.prototype.readers["sld"]) │ │ │ │ │ - }, OpenLayers.Format.SLD.v1_0_0.prototype.readers), │ │ │ │ │ - writers: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - sld: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Priority: function(priority) { │ │ │ │ │ - return this.writers.sld._OGCExpression.call(this, "sld:Priority", priority) │ │ │ │ │ - }, │ │ │ │ │ - VendorOption: function(option) { │ │ │ │ │ - return this.createElementNSPlus("sld:VendorOption", { │ │ │ │ │ - attributes: { │ │ │ │ │ - name: option.name │ │ │ │ │ - }, │ │ │ │ │ - value: option.value │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - TextSymbolizer: function(symbolizer) { │ │ │ │ │ - var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ - var node = writers["sld"]["TextSymbolizer"].apply(this, arguments); │ │ │ │ │ - if (symbolizer.graphic !== false && (symbolizer.externalGraphic || symbolizer.graphicName)) { │ │ │ │ │ - this.writeNode("Graphic", symbolizer, node) │ │ │ │ │ - } │ │ │ │ │ - if ("priority" in symbolizer) { │ │ │ │ │ - this.writeNode("Priority", symbolizer.priority, node) │ │ │ │ │ - } │ │ │ │ │ - return this.addVendorOptions(node, symbolizer) │ │ │ │ │ - }, │ │ │ │ │ - PointSymbolizer: function(symbolizer) { │ │ │ │ │ - var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ - var node = writers["sld"]["PointSymbolizer"].apply(this, arguments); │ │ │ │ │ - return this.addVendorOptions(node, symbolizer) │ │ │ │ │ - }, │ │ │ │ │ - LineSymbolizer: function(symbolizer) { │ │ │ │ │ - var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ - var node = writers["sld"]["LineSymbolizer"].apply(this, arguments); │ │ │ │ │ - return this.addVendorOptions(node, symbolizer) │ │ │ │ │ - }, │ │ │ │ │ - PolygonSymbolizer: function(symbolizer) { │ │ │ │ │ - var writers = OpenLayers.Format.SLD.v1_0_0.prototype.writers; │ │ │ │ │ - var node = writers["sld"]["PolygonSymbolizer"].apply(this, arguments); │ │ │ │ │ - return this.addVendorOptions(node, symbolizer) │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.SLD.v1_0_0.prototype.writers["sld"]) │ │ │ │ │ - }, OpenLayers.Format.SLD.v1_0_0.prototype.writers), │ │ │ │ │ - addVendorOptions: function(node, symbolizer) { │ │ │ │ │ - var options = symbolizer.vendorOptions; │ │ │ │ │ - if (options) { │ │ │ │ │ - for (var key in symbolizer.vendorOptions) { │ │ │ │ │ - this.writeNode("VendorOption", { │ │ │ │ │ - name: key, │ │ │ │ │ - value: symbolizer.vendorOptions[key] │ │ │ │ │ - }, node) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.SLD.v1_0_0_GeoServer" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.CSWGetDomain.v2_0_2 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - namespaces: { │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - csw: "http://www.opengis.net/cat/csw/2.0.2" │ │ │ │ │ - }, │ │ │ │ │ - defaultPrefix: "csw", │ │ │ │ │ - version: "2.0.2", │ │ │ │ │ - schemaLocation: "http://www.opengis.net/cat/csw/2.0.2 http://schemas.opengis.net/csw/2.0.2/CSW-discovery.xsd", │ │ │ │ │ - PropertyName: null, │ │ │ │ │ - ParameterName: null, │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement │ │ │ │ │ - } │ │ │ │ │ - var obj = {}; │ │ │ │ │ - this.readNode(data, obj); │ │ │ │ │ - return obj │ │ │ │ │ - }, │ │ │ │ │ - readers: { │ │ │ │ │ - csw: { │ │ │ │ │ - GetDomainResponse: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - DomainValues: function(node, obj) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(obj.DomainValues)) { │ │ │ │ │ - obj.DomainValues = [] │ │ │ │ │ - } │ │ │ │ │ - var attrs = node.attributes; │ │ │ │ │ - var domainValue = {}; │ │ │ │ │ - for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - domainValue[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ - } │ │ │ │ │ - this.readChildNodes(node, domainValue); │ │ │ │ │ - obj.DomainValues.push(domainValue) │ │ │ │ │ - }, │ │ │ │ │ - PropertyName: function(node, obj) { │ │ │ │ │ - obj.PropertyName = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ParameterName: function(node, obj) { │ │ │ │ │ - obj.ParameterName = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ListOfValues: function(node, obj) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(obj.ListOfValues)) { │ │ │ │ │ - obj.ListOfValues = [] │ │ │ │ │ - } │ │ │ │ │ - this.readChildNodes(node, obj.ListOfValues) │ │ │ │ │ - }, │ │ │ │ │ - Value: function(node, obj) { │ │ │ │ │ - var attrs = node.attributes; │ │ │ │ │ - var value = {}; │ │ │ │ │ - for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - value[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ - } │ │ │ │ │ - value.value = this.getChildValue(node); │ │ │ │ │ - obj.push({ │ │ │ │ │ - Value: value │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - ConceptualScheme: function(node, obj) { │ │ │ │ │ - obj.ConceptualScheme = {}; │ │ │ │ │ - this.readChildNodes(node, obj.ConceptualScheme) │ │ │ │ │ - }, │ │ │ │ │ - Name: function(node, obj) { │ │ │ │ │ - obj.Name = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Document: function(node, obj) { │ │ │ │ │ - obj.Document = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Authority: function(node, obj) { │ │ │ │ │ - obj.Authority = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - RangeOfValues: function(node, obj) { │ │ │ │ │ - obj.RangeOfValues = {}; │ │ │ │ │ - this.readChildNodes(node, obj.RangeOfValues) │ │ │ │ │ - }, │ │ │ │ │ - MinValue: function(node, obj) { │ │ │ │ │ - var attrs = node.attributes; │ │ │ │ │ - var value = {}; │ │ │ │ │ - for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - value[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ - } │ │ │ │ │ - value.value = this.getChildValue(node); │ │ │ │ │ - obj.MinValue = value │ │ │ │ │ - }, │ │ │ │ │ - MaxValue: function(node, obj) { │ │ │ │ │ - var attrs = node.attributes; │ │ │ │ │ - var value = {}; │ │ │ │ │ - for (var i = 0, len = attrs.length; i < len; ++i) { │ │ │ │ │ - value[attrs[i].name] = attrs[i].nodeValue │ │ │ │ │ - } │ │ │ │ │ - value.value = this.getChildValue(node); │ │ │ │ │ - obj.MaxValue = value │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - write: function(options) { │ │ │ │ │ - var node = this.writeNode("csw:GetDomain", options); │ │ │ │ │ - return OpenLayers.Format.XML.prototype.write.apply(this, [node]) │ │ │ │ │ - }, │ │ │ │ │ - writers: { │ │ │ │ │ - csw: { │ │ │ │ │ - GetDomain: function(options) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:GetDomain", { │ │ │ │ │ - attributes: { │ │ │ │ │ - service: "CSW", │ │ │ │ │ - version: this.version │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (options.PropertyName || this.PropertyName) { │ │ │ │ │ - this.writeNode("csw:PropertyName", options.PropertyName || this.PropertyName, node) │ │ │ │ │ - } else if (options.ParameterName || this.ParameterName) { │ │ │ │ │ - this.writeNode("csw:ParameterName", options.ParameterName || this.ParameterName, node) │ │ │ │ │ - } │ │ │ │ │ - this.readChildNodes(node, options); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - PropertyName: function(value) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:PropertyName", { │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - ParameterName: function(value) { │ │ │ │ │ - var node = this.createElementNSPlus("csw:ParameterName", { │ │ │ │ │ - value: value │ │ │ │ │ - }); │ │ │ │ │ - return node │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.CSWGetDomain.v2_0_2" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WMTSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.OWSCommon.v1_1_0, { │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - namespaces: { │ │ │ │ │ - ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ - wmts: "http://www.opengis.net/wmts/1.0", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ - }, │ │ │ │ │ - yx: null, │ │ │ │ │ - defaultPrefix: "wmts", │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.options = options; │ │ │ │ │ - var yx = OpenLayers.Util.extend({}, OpenLayers.Format.WMTSCapabilities.prototype.yx); │ │ │ │ │ - this.yx = OpenLayers.Util.extend(yx, this.yx) │ │ │ │ │ - }, │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement │ │ │ │ │ - } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - capabilities.version = this.version; │ │ │ │ │ - return capabilities │ │ │ │ │ - }, │ │ │ │ │ - readers: { │ │ │ │ │ - wmts: { │ │ │ │ │ - Capabilities: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - Contents: function(node, obj) { │ │ │ │ │ - obj.contents = {}; │ │ │ │ │ - obj.contents.layers = []; │ │ │ │ │ - obj.contents.tileMatrixSets = {}; │ │ │ │ │ - this.readChildNodes(node, obj.contents) │ │ │ │ │ - }, │ │ │ │ │ - Layer: function(node, obj) { │ │ │ │ │ - var layer = { │ │ │ │ │ - styles: [], │ │ │ │ │ - formats: [], │ │ │ │ │ - dimensions: [], │ │ │ │ │ - tileMatrixSetLinks: [] │ │ │ │ │ - }; │ │ │ │ │ - layer.layers = []; │ │ │ │ │ - this.readChildNodes(node, layer); │ │ │ │ │ - obj.layers.push(layer) │ │ │ │ │ - }, │ │ │ │ │ - Style: function(node, obj) { │ │ │ │ │ - var style = {}; │ │ │ │ │ - style.isDefault = node.getAttribute("isDefault") === "true"; │ │ │ │ │ - this.readChildNodes(node, style); │ │ │ │ │ - obj.styles.push(style) │ │ │ │ │ - }, │ │ │ │ │ - Format: function(node, obj) { │ │ │ │ │ - obj.formats.push(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - TileMatrixSetLink: function(node, obj) { │ │ │ │ │ - var tileMatrixSetLink = {}; │ │ │ │ │ - this.readChildNodes(node, tileMatrixSetLink); │ │ │ │ │ - obj.tileMatrixSetLinks.push(tileMatrixSetLink) │ │ │ │ │ - }, │ │ │ │ │ - TileMatrixSet: function(node, obj) { │ │ │ │ │ - if (obj.layers) { │ │ │ │ │ - var tileMatrixSet = { │ │ │ │ │ - matrixIds: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, tileMatrixSet); │ │ │ │ │ - obj.tileMatrixSets[tileMatrixSet.identifier] = tileMatrixSet │ │ │ │ │ - } else { │ │ │ │ │ - obj.tileMatrixSet = this.getChildValue(node) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - TileMatrix: function(node, obj) { │ │ │ │ │ - var tileMatrix = { │ │ │ │ │ - supportedCRS: obj.supportedCRS │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, tileMatrix); │ │ │ │ │ - obj.matrixIds.push(tileMatrix) │ │ │ │ │ - }, │ │ │ │ │ - ScaleDenominator: function(node, obj) { │ │ │ │ │ - obj.scaleDenominator = parseFloat(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - TopLeftCorner: function(node, obj) { │ │ │ │ │ - var topLeftCorner = this.getChildValue(node); │ │ │ │ │ - var coords = topLeftCorner.split(" "); │ │ │ │ │ - var yx; │ │ │ │ │ - if (obj.supportedCRS) { │ │ │ │ │ - var crs = obj.supportedCRS.replace(/urn:ogc:def:crs:(\w+):.+:(\w+)$/, "urn:ogc:def:crs:$1::$2"); │ │ │ │ │ - yx = !!this.yx[crs] │ │ │ │ │ - } │ │ │ │ │ - if (yx) { │ │ │ │ │ - obj.topLeftCorner = new OpenLayers.LonLat(coords[1], coords[0]) │ │ │ │ │ - } else { │ │ │ │ │ - obj.topLeftCorner = new OpenLayers.LonLat(coords[0], coords[1]) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - TileWidth: function(node, obj) { │ │ │ │ │ - obj.tileWidth = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - TileHeight: function(node, obj) { │ │ │ │ │ - obj.tileHeight = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - MatrixWidth: function(node, obj) { │ │ │ │ │ - obj.matrixWidth = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - MatrixHeight: function(node, obj) { │ │ │ │ │ - obj.matrixHeight = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - ResourceURL: function(node, obj) { │ │ │ │ │ - obj.resourceUrl = obj.resourceUrl || {}; │ │ │ │ │ - var resourceType = node.getAttribute("resourceType"); │ │ │ │ │ - if (!obj.resourceUrls) { │ │ │ │ │ - obj.resourceUrls = [] │ │ │ │ │ - } │ │ │ │ │ - var resourceUrl = obj.resourceUrl[resourceType] = { │ │ │ │ │ - format: node.getAttribute("format"), │ │ │ │ │ - template: node.getAttribute("template"), │ │ │ │ │ - resourceType: resourceType │ │ │ │ │ - }; │ │ │ │ │ - obj.resourceUrls.push(resourceUrl) │ │ │ │ │ - }, │ │ │ │ │ - WSDL: function(node, obj) { │ │ │ │ │ - obj.wsdl = {}; │ │ │ │ │ - obj.wsdl.href = node.getAttribute("xlink:href") │ │ │ │ │ - }, │ │ │ │ │ - ServiceMetadataURL: function(node, obj) { │ │ │ │ │ - obj.serviceMetadataUrl = {}; │ │ │ │ │ - obj.serviceMetadataUrl.href = node.getAttribute("xlink:href") │ │ │ │ │ - }, │ │ │ │ │ - LegendURL: function(node, obj) { │ │ │ │ │ - obj.legend = {}; │ │ │ │ │ - obj.legend.href = node.getAttribute("xlink:href"); │ │ │ │ │ - obj.legend.format = node.getAttribute("format") │ │ │ │ │ - }, │ │ │ │ │ - Dimension: function(node, obj) { │ │ │ │ │ - var dimension = { │ │ │ │ │ - values: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, dimension); │ │ │ │ │ - obj.dimensions.push(dimension) │ │ │ │ │ - }, │ │ │ │ │ - Default: function(node, obj) { │ │ │ │ │ - obj["default"] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Value: function(node, obj) { │ │ │ │ │ - obj.values.push(this.getChildValue(node)) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - ows: OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMTSCapabilities.v1_0_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WPSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - namespaces: { │ │ │ │ │ - ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ - wps: "http://www.opengis.net/wps/1.0.0", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ - }, │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: /^\s*|\s*$/g, │ │ │ │ │ - removeSpace: /\s*/g, │ │ │ │ │ - splitSpace: /\s+/, │ │ │ │ │ - trimComma: /\s*,\s*/g │ │ │ │ │ - }, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.XML.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ - } │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement │ │ │ │ │ - } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - return capabilities │ │ │ │ │ - }, │ │ │ │ │ - readers: { │ │ │ │ │ - wps: { │ │ │ │ │ - Capabilities: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - ProcessOfferings: function(node, obj) { │ │ │ │ │ - obj.processOfferings = {}; │ │ │ │ │ - this.readChildNodes(node, obj.processOfferings) │ │ │ │ │ - }, │ │ │ │ │ - Process: function(node, processOfferings) { │ │ │ │ │ - var processVersion = this.getAttributeNS(node, this.namespaces.wps, "processVersion"); │ │ │ │ │ - var process = { │ │ │ │ │ - processVersion: processVersion │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, process); │ │ │ │ │ - processOfferings[process.identifier] = process │ │ │ │ │ - }, │ │ │ │ │ - Languages: function(node, obj) { │ │ │ │ │ - obj.languages = []; │ │ │ │ │ - this.readChildNodes(node, obj.languages) │ │ │ │ │ - }, │ │ │ │ │ - Default: function(node, languages) { │ │ │ │ │ - var language = { │ │ │ │ │ - isDefault: true │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, language); │ │ │ │ │ - languages.push(language) │ │ │ │ │ - }, │ │ │ │ │ - Supported: function(node, languages) { │ │ │ │ │ - var language = {}; │ │ │ │ │ - this.readChildNodes(node, language); │ │ │ │ │ - languages.push(language) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - ows: OpenLayers.Format.OWSCommon.v1_1_0.prototype.readers["ows"] │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WPSCapabilities.v1_0_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WMSDescribeLayer.v1_1_1 = OpenLayers.Class(OpenLayers.Format.WMSDescribeLayer, { │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.WMSDescribeLayer.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ - } │ │ │ │ │ - var root = data.documentElement; │ │ │ │ │ - var children = root.childNodes; │ │ │ │ │ - var describelayer = { │ │ │ │ │ - layerDescriptions: [] │ │ │ │ │ - }; │ │ │ │ │ - var childNode, nodeName; │ │ │ │ │ - for (var i = 0; i < children.length; ++i) { │ │ │ │ │ - childNode = children[i]; │ │ │ │ │ - nodeName = childNode.nodeName; │ │ │ │ │ - if (nodeName == "LayerDescription") { │ │ │ │ │ - var layerName = childNode.getAttribute("name"); │ │ │ │ │ - var owsType = ""; │ │ │ │ │ - var owsURL = ""; │ │ │ │ │ - var typeName = ""; │ │ │ │ │ - if (childNode.getAttribute("owsType")) { │ │ │ │ │ - owsType = childNode.getAttribute("owsType"); │ │ │ │ │ - owsURL = childNode.getAttribute("owsURL") │ │ │ │ │ - } else { │ │ │ │ │ - if (childNode.getAttribute("wfs") != "") { │ │ │ │ │ - owsType = "WFS"; │ │ │ │ │ - owsURL = childNode.getAttribute("wfs") │ │ │ │ │ - } else if (childNode.getAttribute("wcs") != "") { │ │ │ │ │ - owsType = "WCS"; │ │ │ │ │ - owsURL = childNode.getAttribute("wcs") │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var query = childNode.getElementsByTagName("Query"); │ │ │ │ │ - if (query.length > 0) { │ │ │ │ │ - typeName = query[0].getAttribute("typeName"); │ │ │ │ │ - if (!typeName) { │ │ │ │ │ - typeName = query[0].getAttribute("typename") │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var layerDescription = { │ │ │ │ │ - layerName: layerName, │ │ │ │ │ - owsType: owsType, │ │ │ │ │ - owsURL: owsURL, │ │ │ │ │ - typeName: typeName │ │ │ │ │ - }; │ │ │ │ │ - describelayer.layerDescriptions.push(layerDescription); │ │ │ │ │ - describelayer.length = describelayer.layerDescriptions.length; │ │ │ │ │ - describelayer[describelayer.length - 1] = layerDescription │ │ │ │ │ - } else if (nodeName == "ServiceException") { │ │ │ │ │ - var parser = new OpenLayers.Format.OGCExceptionReport; │ │ │ │ │ - return { │ │ │ │ │ - error: parser.read(data) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return describelayer │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSDescribeLayer.v1_1_1" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WMSDescribeLayer.v1_1_0 = OpenLayers.Format.WMSDescribeLayer.v1_1_1; │ │ │ │ │ -OpenLayers.Format.ArcXML.Features = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - read: function(data) { │ │ │ │ │ - var axl = new OpenLayers.Format.ArcXML; │ │ │ │ │ - var parsed = axl.read(data); │ │ │ │ │ - return parsed.features.feature │ │ │ │ │ - } │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WFST.v1_0_0 = OpenLayers.Class(OpenLayers.Format.Filter.v1_0_0, OpenLayers.Format.WFST.v1, { │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - srsNameInQuery: false, │ │ │ │ │ - schemaLocations: { │ │ │ │ │ - wfs: "http://schemas.opengis.net/wfs/1.0.0/WFS-transaction.xsd" │ │ │ │ │ - }, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.Filter.v1_0_0.prototype.initialize.apply(this, [options]); │ │ │ │ │ - OpenLayers.Format.WFST.v1.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ - readNode: function(node, obj, first) { │ │ │ │ │ - return OpenLayers.Format.GML.v2.prototype.readNode.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - readers: { │ │ │ │ │ - wfs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - WFS_TransactionResponse: function(node, obj) { │ │ │ │ │ - obj.insertIds = []; │ │ │ │ │ - obj.success = false; │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - InsertResult: function(node, container) { │ │ │ │ │ - var obj = { │ │ │ │ │ - fids: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj); │ │ │ │ │ - container.insertIds = container.insertIds.concat(obj.fids) │ │ │ │ │ - }, │ │ │ │ │ - TransactionResult: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - Status: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - SUCCESS: function(node, obj) { │ │ │ │ │ - obj.success = true │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WFST.v1.prototype.readers["wfs"]), │ │ │ │ │ - gml: OpenLayers.Format.GML.v2.prototype.readers["gml"], │ │ │ │ │ - feature: OpenLayers.Format.GML.v2.prototype.readers["feature"], │ │ │ │ │ - ogc: OpenLayers.Format.Filter.v1_0_0.prototype.readers["ogc"] │ │ │ │ │ - }, │ │ │ │ │ - writers: { │ │ │ │ │ - wfs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Query: function(options) { │ │ │ │ │ - options = OpenLayers.Util.extend({ │ │ │ │ │ - featureNS: this.featureNS, │ │ │ │ │ - featurePrefix: this.featurePrefix, │ │ │ │ │ - featureType: this.featureType, │ │ │ │ │ - srsName: this.srsName, │ │ │ │ │ - srsNameInQuery: this.srsNameInQuery │ │ │ │ │ - }, options); │ │ │ │ │ - var prefix = options.featurePrefix; │ │ │ │ │ - var node = this.createElementNSPlus("wfs:Query", { │ │ │ │ │ - attributes: { │ │ │ │ │ - typeName: (prefix ? prefix + ":" : "") + options.featureType │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (options.srsNameInQuery && options.srsName) { │ │ │ │ │ - node.setAttribute("srsName", options.srsName) │ │ │ │ │ - } │ │ │ │ │ - if (options.featureNS) { │ │ │ │ │ - node.setAttribute("xmlns:" + prefix, options.featureNS) │ │ │ │ │ - } │ │ │ │ │ - if (options.propertyNames) { │ │ │ │ │ - for (var i = 0, len = options.propertyNames.length; i < len; i++) { │ │ │ │ │ - this.writeNode("ogc:PropertyName", { │ │ │ │ │ - property: options.propertyNames[i] │ │ │ │ │ - }, node) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (options.filter) { │ │ │ │ │ - this.setFilterProperty(options.filter); │ │ │ │ │ - this.writeNode("ogc:Filter", options.filter, node) │ │ │ │ │ - } │ │ │ │ │ - return node │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WFST.v1.prototype.writers["wfs"]), │ │ │ │ │ - gml: OpenLayers.Format.GML.v2.prototype.writers["gml"], │ │ │ │ │ - feature: OpenLayers.Format.GML.v2.prototype.writers["feature"], │ │ │ │ │ - ogc: OpenLayers.Format.Filter.v1_0_0.prototype.writers["ogc"] │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFST.v1_0_0" │ │ │ │ │ -}); │ │ │ │ │ OpenLayers.Format.SOSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.SOSCapabilities, { │ │ │ │ │ namespaces: { │ │ │ │ │ ows: "http://www.opengis.net/ows/1.1", │ │ │ │ │ sos: "http://www.opengis.net/sos/1.0", │ │ │ │ │ gml: "http://www.opengis.net/gml", │ │ │ │ │ xlink: "http://www.w3.org/1999/xlink" │ │ │ │ │ }, │ │ │ │ │ @@ -28930,48 +31633,14 @@ │ │ │ │ │ if (links.length > 0) { │ │ │ │ │ this.read_wmc_OnlineResource(object, links[0]) │ │ │ │ │ } │ │ │ │ │ return object.href │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.WMC.v1" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.WMC.v1_0_0 = OpenLayers.Class(OpenLayers.Format.WMC.v1, { │ │ │ │ │ - VERSION: "1.0.0", │ │ │ │ │ - schemaLocation: "http://www.opengis.net/context http://schemas.opengis.net/context/1.0.0/context.xsd", │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Format.WMC.v1.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ - read_wmc_SRS: function(layerContext, node) { │ │ │ │ │ - var srs = this.getChildValue(node); │ │ │ │ │ - if (typeof layerContext.projections != "object") { │ │ │ │ │ - layerContext.projections = {} │ │ │ │ │ - } │ │ │ │ │ - var values = srs.split(/ +/); │ │ │ │ │ - for (var i = 0, len = values.length; i < len; i++) { │ │ │ │ │ - layerContext.projections[values[i]] = true │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - write_wmc_Layer: function(context) { │ │ │ │ │ - var node = OpenLayers.Format.WMC.v1.prototype.write_wmc_Layer.apply(this, [context]); │ │ │ │ │ - if (context.srs) { │ │ │ │ │ - var projections = []; │ │ │ │ │ - for (var name in context.srs) { │ │ │ │ │ - projections.push(name) │ │ │ │ │ - } │ │ │ │ │ - node.appendChild(this.createElementDefaultNS("SRS", projections.join(" "))) │ │ │ │ │ - } │ │ │ │ │ - node.appendChild(this.write_wmc_FormatList(context)); │ │ │ │ │ - node.appendChild(this.write_wmc_StyleList(context)); │ │ │ │ │ - if (context.dimensions) { │ │ │ │ │ - node.appendChild(this.write_wmc_DimensionList(context)) │ │ │ │ │ - } │ │ │ │ │ - node.appendChild(this.write_wmc_LayerExtension(context)) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMC.v1_0_0" │ │ │ │ │ -}); │ │ │ │ │ OpenLayers.Format.WMC.v1_1_0 = OpenLayers.Class(OpenLayers.Format.WMC.v1, { │ │ │ │ │ VERSION: "1.1.0", │ │ │ │ │ schemaLocation: "http://www.opengis.net/context http://schemas.opengis.net/context/1.1.0/context.xsd", │ │ │ │ │ initialize: function(options) { │ │ │ │ │ OpenLayers.Format.WMC.v1.prototype.initialize.apply(this, [options]) │ │ │ │ │ }, │ │ │ │ │ read_sld_MinScaleDenominator: function(layerContext, node) { │ │ │ │ │ @@ -29012,761 +31681,188 @@ │ │ │ │ │ node.appendChild(this.write_wmc_DimensionList(context)) │ │ │ │ │ } │ │ │ │ │ node.appendChild(this.write_wmc_LayerExtension(context)); │ │ │ │ │ return node │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Format.WMC.v1_1_0" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - namespaces: { │ │ │ │ │ - wms: "http://www.opengis.net/wms", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance" │ │ │ │ │ +OpenLayers.Format.WMC.v1_0_0 = OpenLayers.Class(OpenLayers.Format.WMC.v1, { │ │ │ │ │ + VERSION: "1.0.0", │ │ │ │ │ + schemaLocation: "http://www.opengis.net/context http://schemas.opengis.net/context/1.0.0/context.xsd", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Format.WMC.v1.prototype.initialize.apply(this, [options]) │ │ │ │ │ }, │ │ │ │ │ - defaultPrefix: "wms", │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ + read_wmc_SRS: function(layerContext, node) { │ │ │ │ │ + var srs = this.getChildValue(node); │ │ │ │ │ + if (typeof layerContext.projections != "object") { │ │ │ │ │ + layerContext.projections = {} │ │ │ │ │ } │ │ │ │ │ - var raw = data; │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement │ │ │ │ │ + var values = srs.split(/ +/); │ │ │ │ │ + for (var i = 0, len = values.length; i < len; i++) { │ │ │ │ │ + layerContext.projections[values[i]] = true │ │ │ │ │ } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - if (capabilities.service === undefined) { │ │ │ │ │ - var parser = new OpenLayers.Format.OGCExceptionReport; │ │ │ │ │ - capabilities.error = parser.read(raw) │ │ │ │ │ + }, │ │ │ │ │ + write_wmc_Layer: function(context) { │ │ │ │ │ + var node = OpenLayers.Format.WMC.v1.prototype.write_wmc_Layer.apply(this, [context]); │ │ │ │ │ + if (context.srs) { │ │ │ │ │ + var projections = []; │ │ │ │ │ + for (var name in context.srs) { │ │ │ │ │ + projections.push(name) │ │ │ │ │ + } │ │ │ │ │ + node.appendChild(this.createElementDefaultNS("SRS", projections.join(" "))) │ │ │ │ │ } │ │ │ │ │ - return capabilities │ │ │ │ │ + node.appendChild(this.write_wmc_FormatList(context)); │ │ │ │ │ + node.appendChild(this.write_wmc_StyleList(context)); │ │ │ │ │ + if (context.dimensions) { │ │ │ │ │ + node.appendChild(this.write_wmc_DimensionList(context)) │ │ │ │ │ + } │ │ │ │ │ + node.appendChild(this.write_wmc_LayerExtension(context)) │ │ │ │ │ }, │ │ │ │ │ - readers: { │ │ │ │ │ - wms: { │ │ │ │ │ - Service: function(node, obj) { │ │ │ │ │ - obj.service = {}; │ │ │ │ │ - this.readChildNodes(node, obj.service) │ │ │ │ │ - }, │ │ │ │ │ - Name: function(node, obj) { │ │ │ │ │ - obj.name = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Title: function(node, obj) { │ │ │ │ │ - obj.title = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Abstract: function(node, obj) { │ │ │ │ │ - obj["abstract"] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - BoundingBox: function(node, obj) { │ │ │ │ │ - var bbox = {}; │ │ │ │ │ - bbox.bbox = [parseFloat(node.getAttribute("minx")), parseFloat(node.getAttribute("miny")), parseFloat(node.getAttribute("maxx")), parseFloat(node.getAttribute("maxy"))]; │ │ │ │ │ - var res = { │ │ │ │ │ - x: parseFloat(node.getAttribute("resx")), │ │ │ │ │ - y: parseFloat(node.getAttribute("resy")) │ │ │ │ │ - }; │ │ │ │ │ - if (!(isNaN(res.x) && isNaN(res.y))) { │ │ │ │ │ - bbox.res = res │ │ │ │ │ - } │ │ │ │ │ - return bbox │ │ │ │ │ - }, │ │ │ │ │ - OnlineResource: function(node, obj) { │ │ │ │ │ - obj.href = this.getAttributeNS(node, this.namespaces.xlink, "href") │ │ │ │ │ - }, │ │ │ │ │ - ContactInformation: function(node, obj) { │ │ │ │ │ - obj.contactInformation = {}; │ │ │ │ │ - this.readChildNodes(node, obj.contactInformation) │ │ │ │ │ - }, │ │ │ │ │ - ContactPersonPrimary: function(node, obj) { │ │ │ │ │ - obj.personPrimary = {}; │ │ │ │ │ - this.readChildNodes(node, obj.personPrimary) │ │ │ │ │ - }, │ │ │ │ │ - ContactPerson: function(node, obj) { │ │ │ │ │ - obj.person = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ContactOrganization: function(node, obj) { │ │ │ │ │ - obj.organization = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ContactPosition: function(node, obj) { │ │ │ │ │ - obj.position = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ContactAddress: function(node, obj) { │ │ │ │ │ - obj.contactAddress = {}; │ │ │ │ │ - this.readChildNodes(node, obj.contactAddress) │ │ │ │ │ - }, │ │ │ │ │ - AddressType: function(node, obj) { │ │ │ │ │ - obj.type = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Address: function(node, obj) { │ │ │ │ │ - obj.address = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - City: function(node, obj) { │ │ │ │ │ - obj.city = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - StateOrProvince: function(node, obj) { │ │ │ │ │ - obj.stateOrProvince = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - PostCode: function(node, obj) { │ │ │ │ │ - obj.postcode = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Country: function(node, obj) { │ │ │ │ │ - obj.country = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ContactVoiceTelephone: function(node, obj) { │ │ │ │ │ - obj.phone = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ContactFacsimileTelephone: function(node, obj) { │ │ │ │ │ - obj.fax = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - ContactElectronicMailAddress: function(node, obj) { │ │ │ │ │ - obj.email = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Fees: function(node, obj) { │ │ │ │ │ - var fees = this.getChildValue(node); │ │ │ │ │ - if (fees && fees.toLowerCase() != "none") { │ │ │ │ │ - obj.fees = fees │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - AccessConstraints: function(node, obj) { │ │ │ │ │ - var constraints = this.getChildValue(node); │ │ │ │ │ - if (constraints && constraints.toLowerCase() != "none") { │ │ │ │ │ - obj.accessConstraints = constraints │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - Capability: function(node, obj) { │ │ │ │ │ - obj.capability = { │ │ │ │ │ - nestedLayers: [], │ │ │ │ │ - layers: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.capability) │ │ │ │ │ - }, │ │ │ │ │ - Request: function(node, obj) { │ │ │ │ │ - obj.request = {}; │ │ │ │ │ - this.readChildNodes(node, obj.request) │ │ │ │ │ - }, │ │ │ │ │ - GetCapabilities: function(node, obj) { │ │ │ │ │ - obj.getcapabilities = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getcapabilities) │ │ │ │ │ - }, │ │ │ │ │ - Format: function(node, obj) { │ │ │ │ │ - if (OpenLayers.Util.isArray(obj.formats)) { │ │ │ │ │ - obj.formats.push(this.getChildValue(node)) │ │ │ │ │ - } else { │ │ │ │ │ - obj.format = this.getChildValue(node) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - DCPType: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - HTTP: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - Get: function(node, obj) { │ │ │ │ │ - obj.get = {}; │ │ │ │ │ - this.readChildNodes(node, obj.get); │ │ │ │ │ - if (!obj.href) { │ │ │ │ │ - obj.href = obj.get.href │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - Post: function(node, obj) { │ │ │ │ │ - obj.post = {}; │ │ │ │ │ - this.readChildNodes(node, obj.post); │ │ │ │ │ - if (!obj.href) { │ │ │ │ │ - obj.href = obj.get.href │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - GetMap: function(node, obj) { │ │ │ │ │ - obj.getmap = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getmap) │ │ │ │ │ - }, │ │ │ │ │ - GetFeatureInfo: function(node, obj) { │ │ │ │ │ - obj.getfeatureinfo = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getfeatureinfo) │ │ │ │ │ - }, │ │ │ │ │ - Exception: function(node, obj) { │ │ │ │ │ - obj.exception = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.exception) │ │ │ │ │ - }, │ │ │ │ │ - Layer: function(node, obj) { │ │ │ │ │ - var parentLayer, capability; │ │ │ │ │ - if (obj.capability) { │ │ │ │ │ - capability = obj.capability; │ │ │ │ │ - parentLayer = obj │ │ │ │ │ - } else { │ │ │ │ │ - capability = obj │ │ │ │ │ - } │ │ │ │ │ - var attrNode = node.getAttributeNode("queryable"); │ │ │ │ │ - var queryable = attrNode && attrNode.specified ? node.getAttribute("queryable") : null; │ │ │ │ │ - attrNode = node.getAttributeNode("cascaded"); │ │ │ │ │ - var cascaded = attrNode && attrNode.specified ? node.getAttribute("cascaded") : null; │ │ │ │ │ - attrNode = node.getAttributeNode("opaque"); │ │ │ │ │ - var opaque = attrNode && attrNode.specified ? node.getAttribute("opaque") : null; │ │ │ │ │ - var noSubsets = node.getAttribute("noSubsets"); │ │ │ │ │ - var fixedWidth = node.getAttribute("fixedWidth"); │ │ │ │ │ - var fixedHeight = node.getAttribute("fixedHeight"); │ │ │ │ │ - var parent = parentLayer || {}, │ │ │ │ │ - extend = OpenLayers.Util.extend; │ │ │ │ │ - var layer = { │ │ │ │ │ - nestedLayers: [], │ │ │ │ │ - styles: parentLayer ? [].concat(parentLayer.styles) : [], │ │ │ │ │ - srs: parentLayer ? extend({}, parent.srs) : {}, │ │ │ │ │ - metadataURLs: [], │ │ │ │ │ - bbox: parentLayer ? extend({}, parent.bbox) : {}, │ │ │ │ │ - llbbox: parent.llbbox, │ │ │ │ │ - dimensions: parentLayer ? extend({}, parent.dimensions) : {}, │ │ │ │ │ - authorityURLs: parentLayer ? extend({}, parent.authorityURLs) : {}, │ │ │ │ │ - identifiers: {}, │ │ │ │ │ - keywords: [], │ │ │ │ │ - queryable: queryable && queryable !== "" ? queryable === "1" || queryable === "true" : parent.queryable || false, │ │ │ │ │ - cascaded: cascaded !== null ? parseInt(cascaded) : parent.cascaded || 0, │ │ │ │ │ - opaque: opaque ? opaque === "1" || opaque === "true" : parent.opaque || false, │ │ │ │ │ - noSubsets: noSubsets !== null ? noSubsets === "1" || noSubsets === "true" : parent.noSubsets || false, │ │ │ │ │ - fixedWidth: fixedWidth != null ? parseInt(fixedWidth) : parent.fixedWidth || 0, │ │ │ │ │ - fixedHeight: fixedHeight != null ? parseInt(fixedHeight) : parent.fixedHeight || 0, │ │ │ │ │ - minScale: parent.minScale, │ │ │ │ │ - maxScale: parent.maxScale, │ │ │ │ │ - attribution: parent.attribution │ │ │ │ │ - }; │ │ │ │ │ - obj.nestedLayers.push(layer); │ │ │ │ │ - layer.capability = capability; │ │ │ │ │ - this.readChildNodes(node, layer); │ │ │ │ │ - delete layer.capability; │ │ │ │ │ - if (layer.name) { │ │ │ │ │ - var parts = layer.name.split(":"), │ │ │ │ │ - request = capability.request, │ │ │ │ │ - gfi = request.getfeatureinfo; │ │ │ │ │ - if (parts.length > 0) { │ │ │ │ │ - layer.prefix = parts[0] │ │ │ │ │ + CLASS_NAME: "OpenLayers.Format.WMC.v1_0_0" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Events.buttonclick = OpenLayers.Class({ │ │ │ │ │ + target: null, │ │ │ │ │ + events: ["mousedown", "mouseup", "click", "dblclick", "touchstart", "touchmove", "touchend", "keydown"], │ │ │ │ │ + startRegEx: /^mousedown|touchstart$/, │ │ │ │ │ + cancelRegEx: /^touchmove$/, │ │ │ │ │ + completeRegEx: /^mouseup|touchend$/, │ │ │ │ │ + initialize: function(target) { │ │ │ │ │ + this.target = target; │ │ │ │ │ + for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ + this.target.register(this.events[i], this, this.buttonClick, { │ │ │ │ │ + extension: true │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ + this.target.unregister(this.events[i], this, this.buttonClick) │ │ │ │ │ + } │ │ │ │ │ + delete this.target │ │ │ │ │ + }, │ │ │ │ │ + getPressedButton: function(element) { │ │ │ │ │ + var depth = 3, │ │ │ │ │ + button; │ │ │ │ │ + do { │ │ │ │ │ + if (OpenLayers.Element.hasClass(element, "olButton")) { │ │ │ │ │ + button = element; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + element = element.parentNode │ │ │ │ │ + } while (--depth > 0 && element); │ │ │ │ │ + return button │ │ │ │ │ + }, │ │ │ │ │ + ignore: function(element) { │ │ │ │ │ + var depth = 3, │ │ │ │ │ + ignore = false; │ │ │ │ │ + do { │ │ │ │ │ + if (element.nodeName.toLowerCase() === "a") { │ │ │ │ │ + ignore = true; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + element = element.parentNode │ │ │ │ │ + } while (--depth > 0 && element); │ │ │ │ │ + return ignore │ │ │ │ │ + }, │ │ │ │ │ + buttonClick: function(evt) { │ │ │ │ │ + var propagate = true, │ │ │ │ │ + element = OpenLayers.Event.element(evt); │ │ │ │ │ + if (element && (OpenLayers.Event.isLeftClick(evt) || !~evt.type.indexOf("mouse"))) { │ │ │ │ │ + var button = this.getPressedButton(element); │ │ │ │ │ + if (button) { │ │ │ │ │ + if (evt.type === "keydown") { │ │ │ │ │ + switch (evt.keyCode) { │ │ │ │ │ + case OpenLayers.Event.KEY_RETURN: │ │ │ │ │ + case OpenLayers.Event.KEY_SPACE: │ │ │ │ │ + this.target.triggerEvent("buttonclick", { │ │ │ │ │ + buttonElement: button │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false; │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - capability.layers.push(layer); │ │ │ │ │ - if (layer.formats === undefined) { │ │ │ │ │ - layer.formats = request.getmap.formats │ │ │ │ │ + } else if (this.startEvt) { │ │ │ │ │ + if (this.completeRegEx.test(evt.type)) { │ │ │ │ │ + var pos = OpenLayers.Util.pagePosition(button); │ │ │ │ │ + var viewportElement = OpenLayers.Util.getViewportElement(); │ │ │ │ │ + var scrollTop = window.pageYOffset || viewportElement.scrollTop; │ │ │ │ │ + var scrollLeft = window.pageXOffset || viewportElement.scrollLeft; │ │ │ │ │ + pos[0] = pos[0] - scrollLeft; │ │ │ │ │ + pos[1] = pos[1] - scrollTop; │ │ │ │ │ + this.target.triggerEvent("buttonclick", { │ │ │ │ │ + buttonElement: button, │ │ │ │ │ + buttonXY: { │ │ │ │ │ + x: this.startEvt.clientX - pos[0], │ │ │ │ │ + y: this.startEvt.clientY - pos[1] │ │ │ │ │ + } │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - if (layer.infoFormats === undefined && gfi) { │ │ │ │ │ - layer.infoFormats = gfi.formats │ │ │ │ │ + if (this.cancelRegEx.test(evt.type)) { │ │ │ │ │ + delete this.startEvt │ │ │ │ │ } │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - Attribution: function(node, obj) { │ │ │ │ │ - obj.attribution = {}; │ │ │ │ │ - this.readChildNodes(node, obj.attribution) │ │ │ │ │ - }, │ │ │ │ │ - LogoURL: function(node, obj) { │ │ │ │ │ - obj.logo = { │ │ │ │ │ - width: node.getAttribute("width"), │ │ │ │ │ - height: node.getAttribute("height") │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.logo) │ │ │ │ │ - }, │ │ │ │ │ - Style: function(node, obj) { │ │ │ │ │ - var style = {}; │ │ │ │ │ - obj.styles.push(style); │ │ │ │ │ - this.readChildNodes(node, style) │ │ │ │ │ - }, │ │ │ │ │ - LegendURL: function(node, obj) { │ │ │ │ │ - var legend = { │ │ │ │ │ - width: node.getAttribute("width"), │ │ │ │ │ - height: node.getAttribute("height") │ │ │ │ │ - }; │ │ │ │ │ - obj.legend = legend; │ │ │ │ │ - this.readChildNodes(node, legend) │ │ │ │ │ - }, │ │ │ │ │ - MetadataURL: function(node, obj) { │ │ │ │ │ - var metadataURL = { │ │ │ │ │ - type: node.getAttribute("type") │ │ │ │ │ - }; │ │ │ │ │ - obj.metadataURLs.push(metadataURL); │ │ │ │ │ - this.readChildNodes(node, metadataURL) │ │ │ │ │ - }, │ │ │ │ │ - DataURL: function(node, obj) { │ │ │ │ │ - obj.dataURL = {}; │ │ │ │ │ - this.readChildNodes(node, obj.dataURL) │ │ │ │ │ - }, │ │ │ │ │ - FeatureListURL: function(node, obj) { │ │ │ │ │ - obj.featureListURL = {}; │ │ │ │ │ - this.readChildNodes(node, obj.featureListURL) │ │ │ │ │ - }, │ │ │ │ │ - AuthorityURL: function(node, obj) { │ │ │ │ │ - var name = node.getAttribute("name"); │ │ │ │ │ - var authority = {}; │ │ │ │ │ - this.readChildNodes(node, authority); │ │ │ │ │ - obj.authorityURLs[name] = authority.href │ │ │ │ │ - }, │ │ │ │ │ - Identifier: function(node, obj) { │ │ │ │ │ - var authority = node.getAttribute("authority"); │ │ │ │ │ - obj.identifiers[authority] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - KeywordList: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - SRS: function(node, obj) { │ │ │ │ │ - obj.srs[this.getChildValue(node)] = true │ │ │ │ │ + if (this.startRegEx.test(evt.type)) { │ │ │ │ │ + this.startEvt = evt; │ │ │ │ │ + OpenLayers.Event.stop(evt); │ │ │ │ │ + propagate = false │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + propagate = !this.ignore(OpenLayers.Event.element(evt)); │ │ │ │ │ + delete this.startEvt │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1" │ │ │ │ │ + return propagate │ │ │ │ │ + } │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_1 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1, { │ │ │ │ │ - readers: { │ │ │ │ │ - wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - WMT_MS_Capabilities: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - Keyword: function(node, obj) { │ │ │ │ │ - if (obj.keywords) { │ │ │ │ │ - obj.keywords.push(this.getChildValue(node)) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - DescribeLayer: function(node, obj) { │ │ │ │ │ - obj.describelayer = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.describelayer) │ │ │ │ │ - }, │ │ │ │ │ - GetLegendGraphic: function(node, obj) { │ │ │ │ │ - obj.getlegendgraphic = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getlegendgraphic) │ │ │ │ │ - }, │ │ │ │ │ - GetStyles: function(node, obj) { │ │ │ │ │ - obj.getstyles = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.getstyles) │ │ │ │ │ - }, │ │ │ │ │ - PutStyles: function(node, obj) { │ │ │ │ │ - obj.putstyles = { │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.putstyles) │ │ │ │ │ - }, │ │ │ │ │ - UserDefinedSymbolization: function(node, obj) { │ │ │ │ │ - var userSymbols = { │ │ │ │ │ - supportSLD: parseInt(node.getAttribute("SupportSLD")) == 1, │ │ │ │ │ - userLayer: parseInt(node.getAttribute("UserLayer")) == 1, │ │ │ │ │ - userStyle: parseInt(node.getAttribute("UserStyle")) == 1, │ │ │ │ │ - remoteWFS: parseInt(node.getAttribute("RemoteWFS")) == 1 │ │ │ │ │ - }; │ │ │ │ │ - obj.userSymbols = userSymbols │ │ │ │ │ - }, │ │ │ │ │ - LatLonBoundingBox: function(node, obj) { │ │ │ │ │ - obj.llbbox = [parseFloat(node.getAttribute("minx")), parseFloat(node.getAttribute("miny")), parseFloat(node.getAttribute("maxx")), parseFloat(node.getAttribute("maxy"))] │ │ │ │ │ - }, │ │ │ │ │ - BoundingBox: function(node, obj) { │ │ │ │ │ - var bbox = OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"].BoundingBox.apply(this, [node, obj]); │ │ │ │ │ - bbox.srs = node.getAttribute("SRS"); │ │ │ │ │ - obj.bbox[bbox.srs] = bbox │ │ │ │ │ - }, │ │ │ │ │ - ScaleHint: function(node, obj) { │ │ │ │ │ - var min = node.getAttribute("min"); │ │ │ │ │ - var max = node.getAttribute("max"); │ │ │ │ │ - var rad2 = Math.pow(2, .5); │ │ │ │ │ - var ipm = OpenLayers.INCHES_PER_UNIT["m"]; │ │ │ │ │ - if (min != 0) { │ │ │ │ │ - obj.maxScale = parseFloat((min / rad2 * ipm * OpenLayers.DOTS_PER_INCH).toPrecision(13)) │ │ │ │ │ - } │ │ │ │ │ - if (max != Number.POSITIVE_INFINITY) { │ │ │ │ │ - obj.minScale = parseFloat((max / rad2 * ipm * OpenLayers.DOTS_PER_INCH).toPrecision(13)) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - Dimension: function(node, obj) { │ │ │ │ │ - var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ - var dim = { │ │ │ │ │ - name: name, │ │ │ │ │ - units: node.getAttribute("units"), │ │ │ │ │ - unitsymbol: node.getAttribute("unitSymbol") │ │ │ │ │ - }; │ │ │ │ │ - obj.dimensions[dim.name] = dim │ │ │ │ │ - }, │ │ │ │ │ - Extent: function(node, obj) { │ │ │ │ │ - var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ - if (name in obj["dimensions"]) { │ │ │ │ │ - var extent = obj.dimensions[name]; │ │ │ │ │ - extent.nearestVal = node.getAttribute("nearestValue") === "1"; │ │ │ │ │ - extent.multipleVal = node.getAttribute("multipleValues") === "1"; │ │ │ │ │ - extent.current = node.getAttribute("current") === "1"; │ │ │ │ │ - extent["default"] = node.getAttribute("default") || ""; │ │ │ │ │ - var values = this.getChildValue(node); │ │ │ │ │ - extent.values = values.split(",") │ │ │ │ │ - } │ │ │ │ │ +OpenLayers.Events.featureclick = OpenLayers.Class({ │ │ │ │ │ + cache: null, │ │ │ │ │ + map: null, │ │ │ │ │ + provides: ["featureclick", "nofeatureclick", "featureover", "featureout"], │ │ │ │ │ + initialize: function(target) { │ │ │ │ │ + this.target = target; │ │ │ │ │ + if (target.object instanceof OpenLayers.Map) { │ │ │ │ │ + this.setMap(target.object) │ │ │ │ │ + } else if (target.object instanceof OpenLayers.Layer.Vector) { │ │ │ │ │ + if (target.object.map) { │ │ │ │ │ + this.setMap(target.object.map) │ │ │ │ │ + } else { │ │ │ │ │ + target.object.events.register("added", this, function(evt) { │ │ │ │ │ + this.setMap(target.object.map) │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"]) │ │ │ │ │ + } else { │ │ │ │ │ + throw "Listeners for '" + this.provides.join("', '") + "' events can only be registered for OpenLayers.Layer.Vector " + "or OpenLayers.Map instances" │ │ │ │ │ + } │ │ │ │ │ + for (var i = this.provides.length - 1; i >= 0; --i) { │ │ │ │ │ + target.extensions[this.provides[i]] = true │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_1_1 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1_1, { │ │ │ │ │ - version: "1.1.1", │ │ │ │ │ - readers: { │ │ │ │ │ - wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - SRS: function(node, obj) { │ │ │ │ │ - obj.srs[this.getChildValue(node)] = true │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1_1.prototype.readers["wms"]) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_1" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_3 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1, { │ │ │ │ │ - readers: { │ │ │ │ │ - wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - WMS_Capabilities: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - LayerLimit: function(node, obj) { │ │ │ │ │ - obj.layerLimit = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - MaxWidth: function(node, obj) { │ │ │ │ │ - obj.maxWidth = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - MaxHeight: function(node, obj) { │ │ │ │ │ - obj.maxHeight = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - BoundingBox: function(node, obj) { │ │ │ │ │ - var bbox = OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"].BoundingBox.apply(this, [node, obj]); │ │ │ │ │ - bbox.srs = node.getAttribute("CRS"); │ │ │ │ │ - obj.bbox[bbox.srs] = bbox │ │ │ │ │ - }, │ │ │ │ │ - CRS: function(node, obj) { │ │ │ │ │ - this.readers.wms.SRS.apply(this, [node, obj]) │ │ │ │ │ - }, │ │ │ │ │ - EX_GeographicBoundingBox: function(node, obj) { │ │ │ │ │ - obj.llbbox = []; │ │ │ │ │ - this.readChildNodes(node, obj.llbbox) │ │ │ │ │ - }, │ │ │ │ │ - westBoundLongitude: function(node, obj) { │ │ │ │ │ - obj[0] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - eastBoundLongitude: function(node, obj) { │ │ │ │ │ - obj[2] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - southBoundLatitude: function(node, obj) { │ │ │ │ │ - obj[1] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - northBoundLatitude: function(node, obj) { │ │ │ │ │ - obj[3] = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - MinScaleDenominator: function(node, obj) { │ │ │ │ │ - obj.maxScale = parseFloat(this.getChildValue(node)).toPrecision(16) │ │ │ │ │ - }, │ │ │ │ │ - MaxScaleDenominator: function(node, obj) { │ │ │ │ │ - obj.minScale = parseFloat(this.getChildValue(node)).toPrecision(16) │ │ │ │ │ - }, │ │ │ │ │ - Dimension: function(node, obj) { │ │ │ │ │ - var name = node.getAttribute("name").toLowerCase(); │ │ │ │ │ - var dim = { │ │ │ │ │ - name: name, │ │ │ │ │ - units: node.getAttribute("units"), │ │ │ │ │ - unitsymbol: node.getAttribute("unitSymbol"), │ │ │ │ │ - nearestVal: node.getAttribute("nearestValue") === "1", │ │ │ │ │ - multipleVal: node.getAttribute("multipleValues") === "1", │ │ │ │ │ - default: node.getAttribute("default") || "", │ │ │ │ │ - current: node.getAttribute("current") === "1", │ │ │ │ │ - values: this.getChildValue(node).split(",") │ │ │ │ │ - }; │ │ │ │ │ - obj.dimensions[dim.name] = dim │ │ │ │ │ - }, │ │ │ │ │ - Keyword: function(node, obj) { │ │ │ │ │ - var keyword = { │ │ │ │ │ - value: this.getChildValue(node), │ │ │ │ │ - vocabulary: node.getAttribute("vocabulary") │ │ │ │ │ - }; │ │ │ │ │ - if (obj.keywords) { │ │ │ │ │ - obj.keywords.push(keyword) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1.prototype.readers["wms"]), │ │ │ │ │ - sld: { │ │ │ │ │ - UserDefinedSymbolization: function(node, obj) { │ │ │ │ │ - this.readers.wms.UserDefinedSymbolization.apply(this, [node, obj]); │ │ │ │ │ - obj.userSymbols.inlineFeature = parseInt(node.getAttribute("InlineFeature")) == 1; │ │ │ │ │ - obj.userSymbols.remoteWCS = parseInt(node.getAttribute("RemoteWCS")) == 1 │ │ │ │ │ - }, │ │ │ │ │ - DescribeLayer: function(node, obj) { │ │ │ │ │ - this.readers.wms.DescribeLayer.apply(this, [node, obj]) │ │ │ │ │ - }, │ │ │ │ │ - GetLegendGraphic: function(node, obj) { │ │ │ │ │ - this.readers.wms.GetLegendGraphic.apply(this, [node, obj]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_3" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_3_0 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1_3, { │ │ │ │ │ - version: "1.3.0", │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_3_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_1_1_WMSC = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1_1_1, { │ │ │ │ │ - version: "1.1.1", │ │ │ │ │ - profile: "WMSC", │ │ │ │ │ - readers: { │ │ │ │ │ - wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - VendorSpecificCapabilities: function(node, obj) { │ │ │ │ │ - obj.vendorSpecific = { │ │ │ │ │ - tileSets: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, obj.vendorSpecific) │ │ │ │ │ - }, │ │ │ │ │ - TileSet: function(node, vendorSpecific) { │ │ │ │ │ - var tileset = { │ │ │ │ │ - srs: {}, │ │ │ │ │ - bbox: {}, │ │ │ │ │ - resolutions: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, tileset); │ │ │ │ │ - vendorSpecific.tileSets.push(tileset) │ │ │ │ │ - }, │ │ │ │ │ - Resolutions: function(node, tileset) { │ │ │ │ │ - var res = this.getChildValue(node).split(" "); │ │ │ │ │ - for (var i = 0, len = res.length; i < len; i++) { │ │ │ │ │ - if (res[i] != "") { │ │ │ │ │ - tileset.resolutions.push(parseFloat(res[i])) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - Width: function(node, tileset) { │ │ │ │ │ - tileset.width = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - Height: function(node, tileset) { │ │ │ │ │ - tileset.height = parseInt(this.getChildValue(node)) │ │ │ │ │ - }, │ │ │ │ │ - Layers: function(node, tileset) { │ │ │ │ │ - tileset.layers = this.getChildValue(node) │ │ │ │ │ - }, │ │ │ │ │ - Styles: function(node, tileset) { │ │ │ │ │ - tileset.styles = this.getChildValue(node) │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1_1_1.prototype.readers["wms"]) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_1_WMSC" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WMSCapabilities.v1_1_0 = OpenLayers.Class(OpenLayers.Format.WMSCapabilities.v1_1, { │ │ │ │ │ - version: "1.1.0", │ │ │ │ │ - readers: { │ │ │ │ │ - wms: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - SRS: function(node, obj) { │ │ │ │ │ - var srs = this.getChildValue(node); │ │ │ │ │ - var values = srs.split(/ +/); │ │ │ │ │ - for (var i = 0, len = values.length; i < len; i++) { │ │ │ │ │ - obj.srs[values[i]] = true │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WMSCapabilities.v1_1.prototype.readers["wms"]) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WMSCapabilities.v1_1_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WFSCapabilities.v1 = OpenLayers.Class(OpenLayers.Format.XML, { │ │ │ │ │ - namespaces: { │ │ │ │ │ - wfs: "http://www.opengis.net/wfs", │ │ │ │ │ - xlink: "http://www.w3.org/1999/xlink", │ │ │ │ │ - xsi: "http://www.w3.org/2001/XMLSchema-instance", │ │ │ │ │ - ows: "http://www.opengis.net/ows" │ │ │ │ │ - }, │ │ │ │ │ - errorProperty: "featureTypeList", │ │ │ │ │ - defaultPrefix: "wfs", │ │ │ │ │ - read: function(data) { │ │ │ │ │ - if (typeof data == "string") { │ │ │ │ │ - data = OpenLayers.Format.XML.prototype.read.apply(this, [data]) │ │ │ │ │ - } │ │ │ │ │ - var raw = data; │ │ │ │ │ - if (data && data.nodeType == 9) { │ │ │ │ │ - data = data.documentElement │ │ │ │ │ - } │ │ │ │ │ - var capabilities = {}; │ │ │ │ │ - this.readNode(data, capabilities); │ │ │ │ │ - return capabilities │ │ │ │ │ - }, │ │ │ │ │ - readers: { │ │ │ │ │ - wfs: { │ │ │ │ │ - WFS_Capabilities: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - FeatureTypeList: function(node, request) { │ │ │ │ │ - request.featureTypeList = { │ │ │ │ │ - featureTypes: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, request.featureTypeList) │ │ │ │ │ - }, │ │ │ │ │ - FeatureType: function(node, featureTypeList) { │ │ │ │ │ - var featureType = {}; │ │ │ │ │ - this.readChildNodes(node, featureType); │ │ │ │ │ - featureTypeList.featureTypes.push(featureType) │ │ │ │ │ - }, │ │ │ │ │ - Name: function(node, obj) { │ │ │ │ │ - var name = this.getChildValue(node); │ │ │ │ │ - if (name) { │ │ │ │ │ - var parts = name.split(":"); │ │ │ │ │ - obj.name = parts.pop(); │ │ │ │ │ - if (parts.length > 0) { │ │ │ │ │ - obj.featureNS = this.lookupNamespaceURI(node, parts[0]) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - Title: function(node, obj) { │ │ │ │ │ - var title = this.getChildValue(node); │ │ │ │ │ - if (title) { │ │ │ │ │ - obj.title = title │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - Abstract: function(node, obj) { │ │ │ │ │ - var abst = this.getChildValue(node); │ │ │ │ │ - if (abst) { │ │ │ │ │ - obj["abstract"] = abst │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WFSCapabilities.v1_0_0 = OpenLayers.Class(OpenLayers.Format.WFSCapabilities.v1, { │ │ │ │ │ - readers: { │ │ │ │ │ - wfs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Service: function(node, capabilities) { │ │ │ │ │ - capabilities.service = {}; │ │ │ │ │ - this.readChildNodes(node, capabilities.service) │ │ │ │ │ - }, │ │ │ │ │ - Fees: function(node, service) { │ │ │ │ │ - var fees = this.getChildValue(node); │ │ │ │ │ - if (fees && fees.toLowerCase() != "none") { │ │ │ │ │ - service.fees = fees │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - AccessConstraints: function(node, service) { │ │ │ │ │ - var constraints = this.getChildValue(node); │ │ │ │ │ - if (constraints && constraints.toLowerCase() != "none") { │ │ │ │ │ - service.accessConstraints = constraints │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - OnlineResource: function(node, service) { │ │ │ │ │ - var onlineResource = this.getChildValue(node); │ │ │ │ │ - if (onlineResource && onlineResource.toLowerCase() != "none") { │ │ │ │ │ - service.onlineResource = onlineResource │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - Keywords: function(node, service) { │ │ │ │ │ - var keywords = this.getChildValue(node); │ │ │ │ │ - if (keywords && keywords.toLowerCase() != "none") { │ │ │ │ │ - service.keywords = keywords.split(", ") │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - Capability: function(node, capabilities) { │ │ │ │ │ - capabilities.capability = {}; │ │ │ │ │ - this.readChildNodes(node, capabilities.capability) │ │ │ │ │ - }, │ │ │ │ │ - Request: function(node, obj) { │ │ │ │ │ - obj.request = {}; │ │ │ │ │ - this.readChildNodes(node, obj.request) │ │ │ │ │ - }, │ │ │ │ │ - GetFeature: function(node, request) { │ │ │ │ │ - request.getfeature = { │ │ │ │ │ - href: {}, │ │ │ │ │ - formats: [] │ │ │ │ │ - }; │ │ │ │ │ - this.readChildNodes(node, request.getfeature) │ │ │ │ │ - }, │ │ │ │ │ - ResultFormat: function(node, obj) { │ │ │ │ │ - var children = node.childNodes; │ │ │ │ │ - var childNode; │ │ │ │ │ - for (var i = 0; i < children.length; i++) { │ │ │ │ │ - childNode = children[i]; │ │ │ │ │ - if (childNode.nodeType == 1) { │ │ │ │ │ - obj.formats.push(childNode.nodeName) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - DCPType: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj) │ │ │ │ │ - }, │ │ │ │ │ - HTTP: function(node, obj) { │ │ │ │ │ - this.readChildNodes(node, obj.href) │ │ │ │ │ - }, │ │ │ │ │ - Get: function(node, obj) { │ │ │ │ │ - obj.get = node.getAttribute("onlineResource") │ │ │ │ │ - }, │ │ │ │ │ - Post: function(node, obj) { │ │ │ │ │ - obj.post = node.getAttribute("onlineResource") │ │ │ │ │ - }, │ │ │ │ │ - SRS: function(node, obj) { │ │ │ │ │ - var srs = this.getChildValue(node); │ │ │ │ │ - if (srs) { │ │ │ │ │ - obj.srs = srs │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WFSCapabilities.v1.prototype.readers["wfs"]) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1_0_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Format.WFSCapabilities.v1_1_0 = OpenLayers.Class(OpenLayers.Format.WFSCapabilities.v1, { │ │ │ │ │ - regExes: { │ │ │ │ │ - trimSpace: /^\s*|\s*$/g, │ │ │ │ │ - removeSpace: /\s*/g, │ │ │ │ │ - splitSpace: /\s+/, │ │ │ │ │ - trimComma: /\s*,\s*/g │ │ │ │ │ - }, │ │ │ │ │ - readers: { │ │ │ │ │ - wfs: OpenLayers.Util.applyDefaults({ │ │ │ │ │ - DefaultSRS: function(node, obj) { │ │ │ │ │ - var defaultSRS = this.getChildValue(node); │ │ │ │ │ - if (defaultSRS) { │ │ │ │ │ - obj.srs = defaultSRS │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, OpenLayers.Format.WFSCapabilities.v1.prototype.readers["wfs"]), │ │ │ │ │ - ows: OpenLayers.Format.OWSCommon.v1.prototype.readers.ows │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Format.WFSCapabilities.v1_1_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Events.featureclick = OpenLayers.Class({ │ │ │ │ │ - cache: null, │ │ │ │ │ - map: null, │ │ │ │ │ - provides: ["featureclick", "nofeatureclick", "featureover", "featureout"], │ │ │ │ │ - initialize: function(target) { │ │ │ │ │ - this.target = target; │ │ │ │ │ - if (target.object instanceof OpenLayers.Map) { │ │ │ │ │ - this.setMap(target.object) │ │ │ │ │ - } else if (target.object instanceof OpenLayers.Layer.Vector) { │ │ │ │ │ - if (target.object.map) { │ │ │ │ │ - this.setMap(target.object.map) │ │ │ │ │ - } else { │ │ │ │ │ - target.object.events.register("added", this, function(evt) { │ │ │ │ │ - this.setMap(target.object.map) │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - throw "Listeners for '" + this.provides.join("', '") + "' events can only be registered for OpenLayers.Layer.Vector " + "or OpenLayers.Map instances" │ │ │ │ │ - } │ │ │ │ │ - for (var i = this.provides.length - 1; i >= 0; --i) { │ │ │ │ │ - target.extensions[this.provides[i]] = true │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.map = map; │ │ │ │ │ - this.cache = {}; │ │ │ │ │ - map.events.register("mousedown", this, this.start, { │ │ │ │ │ - extension: true │ │ │ │ │ - }); │ │ │ │ │ - map.events.register("mouseup", this, this.onClick, { │ │ │ │ │ - extension: true │ │ │ │ │ - }); │ │ │ │ │ - map.events.register("touchstart", this, this.start, { │ │ │ │ │ - extension: true │ │ │ │ │ - }); │ │ │ │ │ - map.events.register("touchmove", this, this.cancel, { │ │ │ │ │ - extension: true │ │ │ │ │ - }); │ │ │ │ │ - map.events.register("touchend", this, this.onClick, { │ │ │ │ │ - extension: true │ │ │ │ │ - }); │ │ │ │ │ - map.events.register("mousemove", this, this.onMousemove, { │ │ │ │ │ - extension: true │ │ │ │ │ - }) │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.map = map; │ │ │ │ │ + this.cache = {}; │ │ │ │ │ + map.events.register("mousedown", this, this.start, { │ │ │ │ │ + extension: true │ │ │ │ │ + }); │ │ │ │ │ + map.events.register("mouseup", this, this.onClick, { │ │ │ │ │ + extension: true │ │ │ │ │ + }); │ │ │ │ │ + map.events.register("touchstart", this, this.start, { │ │ │ │ │ + extension: true │ │ │ │ │ + }); │ │ │ │ │ + map.events.register("touchmove", this, this.cancel, { │ │ │ │ │ + extension: true │ │ │ │ │ + }); │ │ │ │ │ + map.events.register("touchend", this, this.onClick, { │ │ │ │ │ + extension: true │ │ │ │ │ + }); │ │ │ │ │ + map.events.register("mousemove", this, this.onMousemove, { │ │ │ │ │ + extension: true │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ start: function(evt) { │ │ │ │ │ this.startEvt = evt │ │ │ │ │ }, │ │ │ │ │ cancel: function(evt) { │ │ │ │ │ delete this.startEvt │ │ │ │ │ }, │ │ │ │ │ @@ -29911,2818 +32007,2933 @@ │ │ │ │ │ delete this.map; │ │ │ │ │ delete this.target │ │ │ │ │ } │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Events.nofeatureclick = OpenLayers.Events.featureclick; │ │ │ │ │ OpenLayers.Events.featureover = OpenLayers.Events.featureclick; │ │ │ │ │ OpenLayers.Events.featureout = OpenLayers.Events.featureclick; │ │ │ │ │ -OpenLayers.Events.buttonclick = OpenLayers.Class({ │ │ │ │ │ - target: null, │ │ │ │ │ - events: ["mousedown", "mouseup", "click", "dblclick", "touchstart", "touchmove", "touchend", "keydown"], │ │ │ │ │ - startRegEx: /^mousedown|touchstart$/, │ │ │ │ │ - cancelRegEx: /^touchmove$/, │ │ │ │ │ - completeRegEx: /^mouseup|touchend$/, │ │ │ │ │ - initialize: function(target) { │ │ │ │ │ - this.target = target; │ │ │ │ │ - for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ - this.target.register(this.events[i], this, this.buttonClick, { │ │ │ │ │ - extension: true │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var i = this.events.length - 1; i >= 0; --i) { │ │ │ │ │ - this.target.unregister(this.events[i], this, this.buttonClick) │ │ │ │ │ - } │ │ │ │ │ - delete this.target │ │ │ │ │ +OpenLayers.Control.ZoomBox = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ + out: false, │ │ │ │ │ + keyMask: null, │ │ │ │ │ + alwaysZoom: false, │ │ │ │ │ + zoomOnClick: true, │ │ │ │ │ + draw: function() { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Box(this, { │ │ │ │ │ + done: this.zoomBox │ │ │ │ │ + }, { │ │ │ │ │ + keyMask: this.keyMask │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - getPressedButton: function(element) { │ │ │ │ │ - var depth = 3, │ │ │ │ │ - button; │ │ │ │ │ - do { │ │ │ │ │ - if (OpenLayers.Element.hasClass(element, "olButton")) { │ │ │ │ │ - button = element; │ │ │ │ │ - break │ │ │ │ │ + zoomBox: function(position) { │ │ │ │ │ + if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ + var bounds, targetCenterPx = position.getCenterPixel(); │ │ │ │ │ + if (!this.out) { │ │ │ │ │ + var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.left, │ │ │ │ │ + y: position.bottom │ │ │ │ │ + }); │ │ │ │ │ + var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.right, │ │ │ │ │ + y: position.top │ │ │ │ │ + }); │ │ │ │ │ + bounds = new OpenLayers.Bounds(minXY.lon, minXY.lat, maxXY.lon, maxXY.lat) │ │ │ │ │ + } else { │ │ │ │ │ + var pixWidth = position.right - position.left; │ │ │ │ │ + var pixHeight = position.bottom - position.top; │ │ │ │ │ + var zoomFactor = Math.min(this.map.size.h / pixHeight, this.map.size.w / pixWidth); │ │ │ │ │ + var extent = this.map.getExtent(); │ │ │ │ │ + var center = this.map.getLonLatFromPixel(targetCenterPx); │ │ │ │ │ + var xmin = center.lon - extent.getWidth() / 2 * zoomFactor; │ │ │ │ │ + var xmax = center.lon + extent.getWidth() / 2 * zoomFactor; │ │ │ │ │ + var ymin = center.lat - extent.getHeight() / 2 * zoomFactor; │ │ │ │ │ + var ymax = center.lat + extent.getHeight() / 2 * zoomFactor; │ │ │ │ │ + bounds = new OpenLayers.Bounds(xmin, ymin, xmax, ymax) │ │ │ │ │ } │ │ │ │ │ - element = element.parentNode │ │ │ │ │ - } while (--depth > 0 && element); │ │ │ │ │ - return button │ │ │ │ │ - }, │ │ │ │ │ - ignore: function(element) { │ │ │ │ │ - var depth = 3, │ │ │ │ │ - ignore = false; │ │ │ │ │ - do { │ │ │ │ │ - if (element.nodeName.toLowerCase() === "a") { │ │ │ │ │ - ignore = true; │ │ │ │ │ - break │ │ │ │ │ + var lastZoom = this.map.getZoom(), │ │ │ │ │ + size = this.map.getSize(), │ │ │ │ │ + centerPx = { │ │ │ │ │ + x: size.w / 2, │ │ │ │ │ + y: size.h / 2 │ │ │ │ │ + }, │ │ │ │ │ + zoom = this.map.getZoomForExtent(bounds), │ │ │ │ │ + oldRes = this.map.getResolution(), │ │ │ │ │ + newRes = this.map.getResolutionForZoom(zoom); │ │ │ │ │ + if (oldRes == newRes) { │ │ │ │ │ + this.map.setCenter(this.map.getLonLatFromPixel(targetCenterPx)) │ │ │ │ │ + } else { │ │ │ │ │ + var zoomOriginPx = { │ │ │ │ │ + x: (oldRes * targetCenterPx.x - newRes * centerPx.x) / (oldRes - newRes), │ │ │ │ │ + y: (oldRes * targetCenterPx.y - newRes * centerPx.y) / (oldRes - newRes) │ │ │ │ │ + }; │ │ │ │ │ + this.map.zoomTo(zoom, zoomOriginPx) │ │ │ │ │ } │ │ │ │ │ - element = element.parentNode │ │ │ │ │ - } while (--depth > 0 && element); │ │ │ │ │ - return ignore │ │ │ │ │ - }, │ │ │ │ │ - buttonClick: function(evt) { │ │ │ │ │ - var propagate = true, │ │ │ │ │ - element = OpenLayers.Event.element(evt); │ │ │ │ │ - if (element && (OpenLayers.Event.isLeftClick(evt) || !~evt.type.indexOf("mouse"))) { │ │ │ │ │ - var button = this.getPressedButton(element); │ │ │ │ │ - if (button) { │ │ │ │ │ - if (evt.type === "keydown") { │ │ │ │ │ - switch (evt.keyCode) { │ │ │ │ │ - case OpenLayers.Event.KEY_RETURN: │ │ │ │ │ - case OpenLayers.Event.KEY_SPACE: │ │ │ │ │ - this.target.triggerEvent("buttonclick", { │ │ │ │ │ - buttonElement: button │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } else if (this.startEvt) { │ │ │ │ │ - if (this.completeRegEx.test(evt.type)) { │ │ │ │ │ - var pos = OpenLayers.Util.pagePosition(button); │ │ │ │ │ - var viewportElement = OpenLayers.Util.getViewportElement(); │ │ │ │ │ - var scrollTop = window.pageYOffset || viewportElement.scrollTop; │ │ │ │ │ - var scrollLeft = window.pageXOffset || viewportElement.scrollLeft; │ │ │ │ │ - pos[0] = pos[0] - scrollLeft; │ │ │ │ │ - pos[1] = pos[1] - scrollTop; │ │ │ │ │ - this.target.triggerEvent("buttonclick", { │ │ │ │ │ - buttonElement: button, │ │ │ │ │ - buttonXY: { │ │ │ │ │ - x: this.startEvt.clientX - pos[0], │ │ │ │ │ - y: this.startEvt.clientY - pos[1] │ │ │ │ │ - } │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - if (this.cancelRegEx.test(evt.type)) { │ │ │ │ │ - delete this.startEvt │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false │ │ │ │ │ - } │ │ │ │ │ - if (this.startRegEx.test(evt.type)) { │ │ │ │ │ - this.startEvt = evt; │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - propagate = false │ │ │ │ │ - } │ │ │ │ │ + if (lastZoom == this.map.getZoom() && this.alwaysZoom == true) { │ │ │ │ │ + this.map.zoomTo(lastZoom + (this.out ? -1 : 1)) │ │ │ │ │ + } │ │ │ │ │ + } else if (this.zoomOnClick) { │ │ │ │ │ + if (!this.out) { │ │ │ │ │ + this.map.zoomTo(this.map.getZoom() + 1, position) │ │ │ │ │ } else { │ │ │ │ │ - propagate = !this.ignore(OpenLayers.Event.element(evt)); │ │ │ │ │ - delete this.startEvt │ │ │ │ │ + this.map.zoomTo(this.map.getZoom() - 1, position) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return propagate │ │ │ │ │ - } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomBox" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Handler.Point = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - point: null, │ │ │ │ │ - layer: null, │ │ │ │ │ - multi: false, │ │ │ │ │ - citeCompliant: false, │ │ │ │ │ - mouseDown: false, │ │ │ │ │ - stoppedDown: null, │ │ │ │ │ - lastDown: null, │ │ │ │ │ - lastUp: null, │ │ │ │ │ - persist: false, │ │ │ │ │ - stopDown: false, │ │ │ │ │ - stopUp: false, │ │ │ │ │ - layerOptions: null, │ │ │ │ │ - pixelTolerance: 5, │ │ │ │ │ - lastTouchPx: null, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ - this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style["default"], {}) │ │ │ │ │ +OpenLayers.Control.DragPan = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ + panned: false, │ │ │ │ │ + interval: 0, │ │ │ │ │ + documentDrag: false, │ │ │ │ │ + kinetic: null, │ │ │ │ │ + enableKinetic: true, │ │ │ │ │ + kineticInterval: 10, │ │ │ │ │ + draw: function() { │ │ │ │ │ + if (this.enableKinetic && OpenLayers.Kinetic) { │ │ │ │ │ + var config = { │ │ │ │ │ + interval: this.kineticInterval │ │ │ │ │ + }; │ │ │ │ │ + if (typeof this.enableKinetic === "object") { │ │ │ │ │ + config = OpenLayers.Util.extend(config, this.enableKinetic) │ │ │ │ │ + } │ │ │ │ │ + this.kinetic = new OpenLayers.Kinetic(config) │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments) │ │ │ │ │ + this.handler = new OpenLayers.Handler.Drag(this, { │ │ │ │ │ + move: this.panMap, │ │ │ │ │ + done: this.panMapDone, │ │ │ │ │ + down: this.panMapStart │ │ │ │ │ + }, { │ │ │ │ │ + interval: this.interval, │ │ │ │ │ + documentDrag: this.documentDrag │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (!OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - return false │ │ │ │ │ + panMapStart: function() { │ │ │ │ │ + if (this.kinetic) { │ │ │ │ │ + this.kinetic.begin() │ │ │ │ │ } │ │ │ │ │ - var options = OpenLayers.Util.extend({ │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - calculateInRange: OpenLayers.Function.True, │ │ │ │ │ - wrapDateLine: this.citeCompliant │ │ │ │ │ - }, this.layerOptions); │ │ │ │ │ - this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ - this.map.addLayer(this.layer); │ │ │ │ │ - return true │ │ │ │ │ }, │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.point]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.point], { │ │ │ │ │ - silent: true │ │ │ │ │ + panMap: function(xy) { │ │ │ │ │ + if (this.kinetic) { │ │ │ │ │ + this.kinetic.update(xy) │ │ │ │ │ + } │ │ │ │ │ + this.panned = true; │ │ │ │ │ + this.map.pan(this.handler.last.x - xy.x, this.handler.last.y - xy.y, { │ │ │ │ │ + dragging: true, │ │ │ │ │ + animate: false │ │ │ │ │ }) │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (!OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - this.cancel(); │ │ │ │ │ - if (this.layer.map != null) { │ │ │ │ │ - this.destroyFeature(true); │ │ │ │ │ - this.layer.destroy(false) │ │ │ │ │ + panMapDone: function(xy) { │ │ │ │ │ + if (this.panned) { │ │ │ │ │ + var res = null; │ │ │ │ │ + if (this.kinetic) { │ │ │ │ │ + res = this.kinetic.end(xy) │ │ │ │ │ + } │ │ │ │ │ + this.map.pan(this.handler.last.x - xy.x, this.handler.last.y - xy.y, { │ │ │ │ │ + dragging: !!res, │ │ │ │ │ + animate: false │ │ │ │ │ + }); │ │ │ │ │ + if (res) { │ │ │ │ │ + var self = this; │ │ │ │ │ + this.kinetic.move(res, function(x, y, end) { │ │ │ │ │ + self.map.pan(x, y, { │ │ │ │ │ + dragging: !end, │ │ │ │ │ + animate: false │ │ │ │ │ + }) │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + this.panned = false │ │ │ │ │ } │ │ │ │ │ - this.layer = null; │ │ │ │ │ - return true │ │ │ │ │ }, │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - if (this.layer && (force || !this.persist)) { │ │ │ │ │ - this.layer.destroyFeatures() │ │ │ │ │ - } │ │ │ │ │ - this.point = null │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.DragPan" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Scale = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + element: null, │ │ │ │ │ + geodesic: false, │ │ │ │ │ + initialize: function(element, options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.element = OpenLayers.Util.getElement(element) │ │ │ │ │ }, │ │ │ │ │ - destroyPersistedFeature: function() { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - if (layer && layer.features.length > 1) { │ │ │ │ │ - this.layer.features[0].destroy() │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (!this.element) { │ │ │ │ │ + this.element = document.createElement("div"); │ │ │ │ │ + this.div.appendChild(this.element) │ │ │ │ │ } │ │ │ │ │ + this.map.events.register("moveend", this, this.updateScale); │ │ │ │ │ + this.updateScale(); │ │ │ │ │ + return this.div │ │ │ │ │ }, │ │ │ │ │ - finalize: function(cancel) { │ │ │ │ │ - var key = cancel ? "cancel" : "done"; │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - this.lastDown = null; │ │ │ │ │ - this.lastUp = null; │ │ │ │ │ - this.lastTouchPx = null; │ │ │ │ │ - this.callback(key, [this.geometryClone()]); │ │ │ │ │ - this.destroyFeature(cancel) │ │ │ │ │ + updateScale: function() { │ │ │ │ │ + var scale; │ │ │ │ │ + if (this.geodesic === true) { │ │ │ │ │ + var units = this.map.getUnits(); │ │ │ │ │ + if (!units) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var inches = OpenLayers.INCHES_PER_UNIT; │ │ │ │ │ + scale = (this.map.getGeodesicPixelSize().w || 1e-6) * inches["km"] * OpenLayers.DOTS_PER_INCH │ │ │ │ │ + } else { │ │ │ │ │ + scale = this.map.getScale() │ │ │ │ │ + } │ │ │ │ │ + if (!scale) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (scale >= 9500 && scale <= 95e4) { │ │ │ │ │ + scale = Math.round(scale / 1e3) + "K" │ │ │ │ │ + } else if (scale >= 95e4) { │ │ │ │ │ + scale = Math.round(scale / 1e6) + "M" │ │ │ │ │ + } else { │ │ │ │ │ + scale = Math.round(scale) │ │ │ │ │ + } │ │ │ │ │ + this.element.innerHTML = OpenLayers.i18n("Scale = 1 : ${scaleDenom}", { │ │ │ │ │ + scaleDenom: scale │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - cancel: function() { │ │ │ │ │ - this.finalize(true) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Scale" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.PinchZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ + pinchOrigin: null, │ │ │ │ │ + currentCenter: null, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + preserveCenter: false, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.handler = new OpenLayers.Handler.Pinch(this, { │ │ │ │ │ + start: this.pinchStart, │ │ │ │ │ + move: this.pinchMove, │ │ │ │ │ + done: this.pinchDone │ │ │ │ │ + }, this.handlerOptions) │ │ │ │ │ }, │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - return false │ │ │ │ │ + pinchStart: function(evt, pinchData) { │ │ │ │ │ + var xy = this.preserveCenter ? this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ + this.pinchOrigin = xy; │ │ │ │ │ + this.currentCenter = xy │ │ │ │ │ }, │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - OpenLayers.Event.stop(evt); │ │ │ │ │ - return false │ │ │ │ │ + pinchMove: function(evt, pinchData) { │ │ │ │ │ + var scale = pinchData.scale; │ │ │ │ │ + var containerOrigin = this.map.layerContainerOriginPx; │ │ │ │ │ + var pinchOrigin = this.pinchOrigin; │ │ │ │ │ + var current = this.preserveCenter ? this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ + var dx = Math.round(containerOrigin.x + current.x - pinchOrigin.x + (scale - 1) * (containerOrigin.x - pinchOrigin.x)); │ │ │ │ │ + var dy = Math.round(containerOrigin.y + current.y - pinchOrigin.y + (scale - 1) * (containerOrigin.y - pinchOrigin.y)); │ │ │ │ │ + this.map.applyTransform(dx, dy, scale); │ │ │ │ │ + this.currentCenter = current │ │ │ │ │ }, │ │ │ │ │ - modifyFeature: function(pixel) { │ │ │ │ │ - if (!this.point) { │ │ │ │ │ - this.createFeature(pixel) │ │ │ │ │ + pinchDone: function(evt, start, last) { │ │ │ │ │ + this.map.applyTransform(); │ │ │ │ │ + var zoom = this.map.getZoomForResolution(this.map.getResolution() / last.scale, true); │ │ │ │ │ + if (zoom !== this.map.getZoom() || !this.currentCenter.equals(this.pinchOrigin)) { │ │ │ │ │ + var resolution = this.map.getResolutionForZoom(zoom); │ │ │ │ │ + var location = this.map.getLonLatFromPixel(this.pinchOrigin); │ │ │ │ │ + var zoomPixel = this.currentCenter; │ │ │ │ │ + var size = this.map.getSize(); │ │ │ │ │ + location.lon += resolution * (size.w / 2 - zoomPixel.x); │ │ │ │ │ + location.lat -= resolution * (size.h / 2 - zoomPixel.y); │ │ │ │ │ + this.map.div.clientWidth = this.map.div.clientWidth; │ │ │ │ │ + this.map.setCenter(location, zoom) │ │ │ │ │ } │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point.geometry.x = lonlat.lon; │ │ │ │ │ - this.point.geometry.y = lonlat.lat; │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.point, false]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.drawFeature() │ │ │ │ │ }, │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.point, this.style) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.PinchZoom" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.TouchNavigation = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + dragPan: null, │ │ │ │ │ + dragPanOptions: null, │ │ │ │ │ + pinchZoom: null, │ │ │ │ │ + pinchZoomOptions: null, │ │ │ │ │ + clickHandlerOptions: null, │ │ │ │ │ + documentDrag: false, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.handlers = {}; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.point && this.point.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiPoint([geometry]) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + if (this.dragPan) { │ │ │ │ │ + this.dragPan.destroy() │ │ │ │ │ } │ │ │ │ │ - return geometry │ │ │ │ │ - }, │ │ │ │ │ - geometryClone: function() { │ │ │ │ │ - var geom = this.getGeometry(); │ │ │ │ │ - return geom && geom.clone() │ │ │ │ │ + this.dragPan = null; │ │ │ │ │ + if (this.pinchZoom) { │ │ │ │ │ + this.pinchZoom.destroy(); │ │ │ │ │ + delete this.pinchZoom │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - return this.down(evt) │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.dragPan.activate(); │ │ │ │ │ + this.handlers.click.activate(); │ │ │ │ │ + this.pinchZoom.activate(); │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - this.lastTouchPx = evt.xy; │ │ │ │ │ - return this.down(evt) │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.dragPan.deactivate(); │ │ │ │ │ + this.handlers.click.deactivate(); │ │ │ │ │ + this.pinchZoom.deactivate(); │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - return this.move(evt) │ │ │ │ │ + draw: function() { │ │ │ │ │ + var clickCallbacks = { │ │ │ │ │ + click: this.defaultClick, │ │ │ │ │ + dblclick: this.defaultDblClick │ │ │ │ │ + }; │ │ │ │ │ + var clickOptions = OpenLayers.Util.extend({ │ │ │ │ │ + double: true, │ │ │ │ │ + stopDouble: true, │ │ │ │ │ + pixelTolerance: 2 │ │ │ │ │ + }, this.clickHandlerOptions); │ │ │ │ │ + this.handlers.click = new OpenLayers.Handler.Click(this, clickCallbacks, clickOptions); │ │ │ │ │ + this.dragPan = new OpenLayers.Control.DragPan(OpenLayers.Util.extend({ │ │ │ │ │ + map: this.map, │ │ │ │ │ + documentDrag: this.documentDrag │ │ │ │ │ + }, this.dragPanOptions)); │ │ │ │ │ + this.dragPan.draw(); │ │ │ │ │ + this.pinchZoom = new OpenLayers.Control.PinchZoom(OpenLayers.Util.extend({ │ │ │ │ │ + map: this.map │ │ │ │ │ + }, this.pinchZoomOptions)) │ │ │ │ │ }, │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - this.lastTouchPx = evt.xy; │ │ │ │ │ - return this.move(evt) │ │ │ │ │ + defaultClick: function(evt) { │ │ │ │ │ + if (evt.lastTouches && evt.lastTouches.length == 2) { │ │ │ │ │ + this.map.zoomOut() │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - return this.up(evt) │ │ │ │ │ + defaultDblClick: function(evt) { │ │ │ │ │ + this.map.zoomTo(this.map.zoom + 1, evt.xy) │ │ │ │ │ }, │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - evt.xy = this.lastTouchPx; │ │ │ │ │ - return this.up(evt) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.TouchNavigation" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.KeyboardDefaults = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + slideFactor: 75, │ │ │ │ │ + observeElement: null, │ │ │ │ │ + draw: function() { │ │ │ │ │ + var observeElement = this.observeElement || document; │ │ │ │ │ + this.handler = new OpenLayers.Handler.Keyboard(this, { │ │ │ │ │ + keydown: this.defaultKeyPress │ │ │ │ │ + }, { │ │ │ │ │ + observeElement: observeElement │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - down: function(evt) { │ │ │ │ │ - this.mouseDown = true; │ │ │ │ │ - this.lastDown = evt.xy; │ │ │ │ │ - if (!this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy) │ │ │ │ │ + defaultKeyPress: function(evt) { │ │ │ │ │ + var size, handled = true; │ │ │ │ │ + var target = OpenLayers.Event.element(evt); │ │ │ │ │ + if (target && (target.tagName == "INPUT" || target.tagName == "TEXTAREA" || target.tagName == "SELECT")) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - return !this.stopDown │ │ │ │ │ - }, │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ - this.modifyFeature(evt.xy) │ │ │ │ │ + switch (evt.keyCode) { │ │ │ │ │ + case OpenLayers.Event.KEY_LEFT: │ │ │ │ │ + this.map.pan(-this.slideFactor, 0); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Event.KEY_RIGHT: │ │ │ │ │ + this.map.pan(this.slideFactor, 0); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Event.KEY_UP: │ │ │ │ │ + this.map.pan(0, -this.slideFactor); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Event.KEY_DOWN: │ │ │ │ │ + this.map.pan(0, this.slideFactor); │ │ │ │ │ + break; │ │ │ │ │ + case 33: │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + this.map.pan(0, -.75 * size.h); │ │ │ │ │ + break; │ │ │ │ │ + case 34: │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + this.map.pan(0, .75 * size.h); │ │ │ │ │ + break; │ │ │ │ │ + case 35: │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + this.map.pan(.75 * size.w, 0); │ │ │ │ │ + break; │ │ │ │ │ + case 36: │ │ │ │ │ + size = this.map.getSize(); │ │ │ │ │ + this.map.pan(-.75 * size.w, 0); │ │ │ │ │ + break; │ │ │ │ │ + case 43: │ │ │ │ │ + case 61: │ │ │ │ │ + case 187: │ │ │ │ │ + case 107: │ │ │ │ │ + this.map.zoomIn(); │ │ │ │ │ + break; │ │ │ │ │ + case 45: │ │ │ │ │ + case 109: │ │ │ │ │ + case 189: │ │ │ │ │ + case 95: │ │ │ │ │ + this.map.zoomOut(); │ │ │ │ │ + break; │ │ │ │ │ + default: │ │ │ │ │ + handled = false │ │ │ │ │ + } │ │ │ │ │ + if (handled) { │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ } │ │ │ │ │ - return true │ │ │ │ │ }, │ │ │ │ │ - up: function(evt) { │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - if (!this.checkModifiers(evt)) { │ │ │ │ │ - return true │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.KeyboardDefaults" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.ArgParser = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + center: null, │ │ │ │ │ + zoom: null, │ │ │ │ │ + layers: null, │ │ │ │ │ + displayProjection: null, │ │ │ │ │ + getParameters: function(url) { │ │ │ │ │ + url = url || window.location.href; │ │ │ │ │ + var parameters = OpenLayers.Util.getParameters(url); │ │ │ │ │ + var index = url.indexOf("#"); │ │ │ │ │ + if (index > 0) { │ │ │ │ │ + url = "?" + url.substring(index + 1, url.length); │ │ │ │ │ + OpenLayers.Util.extend(parameters, OpenLayers.Util.getParameters(url)) │ │ │ │ │ } │ │ │ │ │ - if (this.lastUp && this.lastUp.equals(evt.xy)) { │ │ │ │ │ - return true │ │ │ │ │ + return parameters │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + for (var i = 0, len = this.map.controls.length; i < len; i++) { │ │ │ │ │ + var control = this.map.controls[i]; │ │ │ │ │ + if (control != this && control.CLASS_NAME == "OpenLayers.Control.ArgParser") { │ │ │ │ │ + if (control.displayProjection != this.displayProjection) { │ │ │ │ │ + this.displayProjection = control.displayProjection │ │ │ │ │ + } │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (this.lastDown && this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance)) { │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy) │ │ │ │ │ + if (i == this.map.controls.length) { │ │ │ │ │ + var args = this.getParameters(); │ │ │ │ │ + if (args.layers) { │ │ │ │ │ + this.layers = args.layers; │ │ │ │ │ + this.map.events.register("addlayer", this, this.configureLayers); │ │ │ │ │ + this.configureLayers() │ │ │ │ │ } │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature() │ │ │ │ │ + if (args.lat && args.lon) { │ │ │ │ │ + this.center = new OpenLayers.LonLat(parseFloat(args.lon), parseFloat(args.lat)); │ │ │ │ │ + if (args.zoom) { │ │ │ │ │ + this.zoom = parseFloat(args.zoom) │ │ │ │ │ + } │ │ │ │ │ + this.map.events.register("changebaselayer", this, this.setCenter); │ │ │ │ │ + this.setCenter() │ │ │ │ │ } │ │ │ │ │ - this.lastUp = evt.xy; │ │ │ │ │ - this.finalize(); │ │ │ │ │ - return !this.stopUp │ │ │ │ │ - } else { │ │ │ │ │ - return true │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - mouseout: function(evt) { │ │ │ │ │ - if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - this.mouseDown = false │ │ │ │ │ + setCenter: function() { │ │ │ │ │ + if (this.map.baseLayer) { │ │ │ │ │ + this.map.events.unregister("changebaselayer", this, this.setCenter); │ │ │ │ │ + if (this.displayProjection) { │ │ │ │ │ + this.center.transform(this.displayProjection, this.map.getProjectionObject()) │ │ │ │ │ + } │ │ │ │ │ + this.map.setCenter(this.center, this.zoom) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - passesTolerance: function(pixel1, pixel2, tolerance) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - if (tolerance != null && pixel1 && pixel2) { │ │ │ │ │ - var dist = pixel1.distanceTo(pixel2); │ │ │ │ │ - if (dist > tolerance) { │ │ │ │ │ - passes = false │ │ │ │ │ + configureLayers: function() { │ │ │ │ │ + if (this.layers.length == this.map.layers.length) { │ │ │ │ │ + this.map.events.unregister("addlayer", this, this.configureLayers); │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + var c = this.layers.charAt(i); │ │ │ │ │ + if (c == "B") { │ │ │ │ │ + this.map.setBaseLayer(layer) │ │ │ │ │ + } else if (c == "T" || c == "F") { │ │ │ │ │ + layer.setVisibility(c == "T") │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return passes │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Point" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ArgParser" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Handler.Feature = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - EVENTMAP: { │ │ │ │ │ - click: { │ │ │ │ │ - in: "click", │ │ │ │ │ - out: "clickout" │ │ │ │ │ - }, │ │ │ │ │ - mousemove: { │ │ │ │ │ - in: "over", │ │ │ │ │ - out: "out" │ │ │ │ │ - }, │ │ │ │ │ - dblclick: { │ │ │ │ │ - in: "dblclick", │ │ │ │ │ - out: null │ │ │ │ │ - }, │ │ │ │ │ - mousedown: { │ │ │ │ │ - in: null, │ │ │ │ │ - out: null │ │ │ │ │ - }, │ │ │ │ │ - mouseup: { │ │ │ │ │ - in: null, │ │ │ │ │ - out: null │ │ │ │ │ - }, │ │ │ │ │ - touchstart: { │ │ │ │ │ - in: "click", │ │ │ │ │ - out: "clickout" │ │ │ │ │ +OpenLayers.Control.WMTSGetFeatureInfo = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + hover: false, │ │ │ │ │ + requestEncoding: "KVP", │ │ │ │ │ + drillDown: false, │ │ │ │ │ + maxFeatures: 10, │ │ │ │ │ + clickCallback: "click", │ │ │ │ │ + layers: null, │ │ │ │ │ + queryVisible: true, │ │ │ │ │ + infoFormat: "text/html", │ │ │ │ │ + vendorParams: {}, │ │ │ │ │ + format: null, │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + handler: null, │ │ │ │ │ + hoverRequest: null, │ │ │ │ │ + pending: 0, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!this.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.WMSGetFeatureInfo(options.formatOptions) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - feature: null, │ │ │ │ │ - lastFeature: null, │ │ │ │ │ - down: null, │ │ │ │ │ - up: null, │ │ │ │ │ - clickTolerance: 4, │ │ │ │ │ - geometryTypes: null, │ │ │ │ │ - stopClick: true, │ │ │ │ │ - stopDown: true, │ │ │ │ │ - stopUp: false, │ │ │ │ │ - initialize: function(control, layer, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ - this.layer = layer │ │ │ │ │ - }, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - return OpenLayers.Event.isMultiTouch(evt) ? true : this.mousedown(evt) │ │ │ │ │ - }, │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - OpenLayers.Event.preventDefault(evt) │ │ │ │ │ - }, │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - if (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ - this.down = evt.xy │ │ │ │ │ + if (this.drillDown === true) { │ │ │ │ │ + this.hover = false │ │ │ │ │ } │ │ │ │ │ - return this.handle(evt) ? !this.stopDown : true │ │ │ │ │ - }, │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - this.up = evt.xy; │ │ │ │ │ - return this.handle(evt) ? !this.stopUp : true │ │ │ │ │ - }, │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - return this.handle(evt) ? !this.stopClick : true │ │ │ │ │ - }, │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - if (!this.callbacks["over"] && !this.callbacks["out"]) { │ │ │ │ │ - return true │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ + move: this.cancelHover, │ │ │ │ │ + pause: this.getInfoForHover │ │ │ │ │ + }, OpenLayers.Util.extend(this.handlerOptions.hover || {}, { │ │ │ │ │ + delay: 250 │ │ │ │ │ + })) │ │ │ │ │ + } else { │ │ │ │ │ + var callbacks = {}; │ │ │ │ │ + callbacks[this.clickCallback] = this.getInfoForClick; │ │ │ │ │ + this.handler = new OpenLayers.Handler.Click(this, callbacks, this.handlerOptions.click || {}) │ │ │ │ │ } │ │ │ │ │ - this.handle(evt); │ │ │ │ │ - return true │ │ │ │ │ }, │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - return !this.handle(evt) │ │ │ │ │ + getInfoForClick: function(evt) { │ │ │ │ │ + this.request(evt.xy, {}) │ │ │ │ │ }, │ │ │ │ │ - geometryTypeMatches: function(feature) { │ │ │ │ │ - return this.geometryTypes == null || OpenLayers.Util.indexOf(this.geometryTypes, feature.geometry.CLASS_NAME) > -1 │ │ │ │ │ + getInfoForHover: function(evt) { │ │ │ │ │ + this.request(evt.xy, { │ │ │ │ │ + hover: true │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - handle: function(evt) { │ │ │ │ │ - if (this.feature && !this.feature.layer) { │ │ │ │ │ - this.feature = null │ │ │ │ │ - } │ │ │ │ │ - var type = evt.type; │ │ │ │ │ - var handled = false; │ │ │ │ │ - var previouslyIn = !!this.feature; │ │ │ │ │ - var click = type == "click" || type == "dblclick" || type == "touchstart"; │ │ │ │ │ - this.feature = this.layer.getFeatureFromEvent(evt); │ │ │ │ │ - if (this.feature && !this.feature.layer) { │ │ │ │ │ - this.feature = null │ │ │ │ │ - } │ │ │ │ │ - if (this.lastFeature && !this.lastFeature.layer) { │ │ │ │ │ - this.lastFeature = null │ │ │ │ │ - } │ │ │ │ │ - if (this.feature) { │ │ │ │ │ - if (type === "touchstart") { │ │ │ │ │ - OpenLayers.Event.preventDefault(evt) │ │ │ │ │ - } │ │ │ │ │ - var inNew = this.feature != this.lastFeature; │ │ │ │ │ - if (this.geometryTypeMatches(this.feature)) { │ │ │ │ │ - if (previouslyIn && inNew) { │ │ │ │ │ - if (this.lastFeature) { │ │ │ │ │ - this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ - } │ │ │ │ │ - this.triggerCallback(type, "in", [this.feature]) │ │ │ │ │ - } else if (!previouslyIn || click) { │ │ │ │ │ - this.triggerCallback(type, "in", [this.feature]) │ │ │ │ │ - } │ │ │ │ │ - this.lastFeature = this.feature; │ │ │ │ │ - handled = true │ │ │ │ │ - } else { │ │ │ │ │ - if (this.lastFeature && (previouslyIn && inNew || click)) { │ │ │ │ │ - this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ - } │ │ │ │ │ - this.feature = null │ │ │ │ │ + cancelHover: function() { │ │ │ │ │ + if (this.hoverRequest) { │ │ │ │ │ + --this.pending; │ │ │ │ │ + if (this.pending <= 0) { │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + this.pending = 0 │ │ │ │ │ } │ │ │ │ │ - } else if (this.lastFeature && (previouslyIn || click)) { │ │ │ │ │ - this.triggerCallback(type, "out", [this.lastFeature]) │ │ │ │ │ + this.hoverRequest.abort(); │ │ │ │ │ + this.hoverRequest = null │ │ │ │ │ } │ │ │ │ │ - return handled │ │ │ │ │ }, │ │ │ │ │ - triggerCallback: function(type, mode, args) { │ │ │ │ │ - var key = this.EVENTMAP[type][mode]; │ │ │ │ │ - if (key) { │ │ │ │ │ - if (type == "click" && this.up && this.down) { │ │ │ │ │ - var dpx = Math.sqrt(Math.pow(this.up.x - this.down.x, 2) + Math.pow(this.up.y - this.down.y, 2)); │ │ │ │ │ - if (dpx <= this.clickTolerance) { │ │ │ │ │ - this.callback(key, args) │ │ │ │ │ + findLayers: function() { │ │ │ │ │ + var candidates = this.layers || this.map.layers; │ │ │ │ │ + var layers = []; │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = candidates[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.WMTS && layer.requestEncoding === this.requestEncoding && (!this.queryVisible || layer.getVisibility())) { │ │ │ │ │ + layers.push(layer); │ │ │ │ │ + if (!this.drillDown || this.hover) { │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - this.up = this.down = null │ │ │ │ │ - } else { │ │ │ │ │ - this.callback(key, args) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return layers │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerToTop(); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - removelayer: this.handleMapEvents, │ │ │ │ │ - changelayer: this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - activated = true │ │ │ │ │ + buildRequestOptions: function(layer, xy) { │ │ │ │ │ + var loc = this.map.getLonLatFromPixel(xy); │ │ │ │ │ + var getTileUrl = layer.getURL(new OpenLayers.Bounds(loc.lon, loc.lat, loc.lon, loc.lat)); │ │ │ │ │ + var params = OpenLayers.Util.getParameters(getTileUrl); │ │ │ │ │ + var tileInfo = layer.getTileInfo(loc); │ │ │ │ │ + OpenLayers.Util.extend(params, { │ │ │ │ │ + service: "WMTS", │ │ │ │ │ + version: layer.version, │ │ │ │ │ + request: "GetFeatureInfo", │ │ │ │ │ + infoFormat: this.infoFormat, │ │ │ │ │ + i: tileInfo.i, │ │ │ │ │ + j: tileInfo.j │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Util.applyDefaults(params, this.vendorParams); │ │ │ │ │ + return { │ │ │ │ │ + url: OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url, │ │ │ │ │ + params: OpenLayers.Util.upperCaseObject(params), │ │ │ │ │ + callback: function(request) { │ │ │ │ │ + this.handleResponse(xy, request, layer) │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ } │ │ │ │ │ - return activated │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerBack(); │ │ │ │ │ - this.feature = null; │ │ │ │ │ - this.lastFeature = null; │ │ │ │ │ - this.down = null; │ │ │ │ │ - this.up = null; │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - removelayer: this.handleMapEvents, │ │ │ │ │ - changelayer: this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - deactivated = true │ │ │ │ │ + request: function(xy, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var layers = this.findLayers(); │ │ │ │ │ + if (layers.length > 0) { │ │ │ │ │ + var issue, layer; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + layer = layers[i]; │ │ │ │ │ + issue = this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ + xy: xy, │ │ │ │ │ + layer: layer │ │ │ │ │ + }); │ │ │ │ │ + if (issue !== false) { │ │ │ │ │ + ++this.pending; │ │ │ │ │ + var requestOptions = this.buildRequestOptions(layer, xy); │ │ │ │ │ + var request = OpenLayers.Request.GET(requestOptions); │ │ │ │ │ + if (options.hover === true) { │ │ │ │ │ + this.hoverRequest = request │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.pending > 0) { │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait") │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return deactivated │ │ │ │ │ }, │ │ │ │ │ - handleMapEvents: function(evt) { │ │ │ │ │ - if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ - this.moveLayerToTop() │ │ │ │ │ + handleResponse: function(xy, request, layer) { │ │ │ │ │ + --this.pending; │ │ │ │ │ + if (this.pending <= 0) { │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + this.pending = 0 │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - moveLayerToTop: function() { │ │ │ │ │ - var index = Math.max(this.map.Z_INDEX_BASE["Feature"] - 1, this.layer.getZIndex()) + 1; │ │ │ │ │ - this.layer.setZIndex(index) │ │ │ │ │ - }, │ │ │ │ │ - moveLayerBack: function() { │ │ │ │ │ - var index = this.layer.getZIndex() - 1; │ │ │ │ │ - if (index >= this.map.Z_INDEX_BASE["Feature"]) { │ │ │ │ │ - this.layer.setZIndex(index) │ │ │ │ │ + if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ + this.events.triggerEvent("exception", { │ │ │ │ │ + xy: xy, │ │ │ │ │ + request: request, │ │ │ │ │ + layer: layer │ │ │ │ │ + }) │ │ │ │ │ } else { │ │ │ │ │ - this.map.setLayerZIndex(this.layer, this.map.getLayerIndex(this.layer)) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Feature" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.Drag = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - started: false, │ │ │ │ │ - stopDown: true, │ │ │ │ │ - dragging: false, │ │ │ │ │ - last: null, │ │ │ │ │ - start: null, │ │ │ │ │ - lastMoveEvt: null, │ │ │ │ │ - oldOnselectstart: null, │ │ │ │ │ - interval: 0, │ │ │ │ │ - timeoutId: null, │ │ │ │ │ - documentDrag: false, │ │ │ │ │ - documentEvents: null, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (this.documentDrag === true) { │ │ │ │ │ - var me = this; │ │ │ │ │ - this._docMove = function(evt) { │ │ │ │ │ - me.mousemove({ │ │ │ │ │ - xy: { │ │ │ │ │ - x: evt.clientX, │ │ │ │ │ - y: evt.clientY │ │ │ │ │ - }, │ │ │ │ │ - element: document │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ + } │ │ │ │ │ + var features, except; │ │ │ │ │ + try { │ │ │ │ │ + features = this.format.read(doc) │ │ │ │ │ + } catch (error) { │ │ │ │ │ + except = true; │ │ │ │ │ + this.events.triggerEvent("exception", { │ │ │ │ │ + xy: xy, │ │ │ │ │ + request: request, │ │ │ │ │ + error: error, │ │ │ │ │ + layer: layer │ │ │ │ │ }) │ │ │ │ │ - }; │ │ │ │ │ - this._docUp = function(evt) { │ │ │ │ │ - me.mouseup({ │ │ │ │ │ - xy: { │ │ │ │ │ - x: evt.clientX, │ │ │ │ │ - y: evt.clientY │ │ │ │ │ - } │ │ │ │ │ + } │ │ │ │ │ + if (!except) { │ │ │ │ │ + this.events.triggerEvent("getfeatureinfo", { │ │ │ │ │ + text: request.responseText, │ │ │ │ │ + features: features, │ │ │ │ │ + request: request, │ │ │ │ │ + xy: xy, │ │ │ │ │ + layer: layer │ │ │ │ │ }) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - dragstart: function(evt) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - if (this.checkModifiers(evt) && (OpenLayers.Event.isLeftClick(evt) || OpenLayers.Event.isSingleTouch(evt))) { │ │ │ │ │ - this.started = true; │ │ │ │ │ - this.start = evt.xy; │ │ │ │ │ - this.last = evt.xy; │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ - this.down(evt); │ │ │ │ │ - this.callback("down", [evt.xy]); │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ - if (!this.oldOnselectstart) { │ │ │ │ │ - this.oldOnselectstart = document.onselectstart ? document.onselectstart : OpenLayers.Function.True │ │ │ │ │ - } │ │ │ │ │ - document.onselectstart = OpenLayers.Function.False; │ │ │ │ │ - propagate = !this.stopDown │ │ │ │ │ - } else { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.WMTSGetFeatureInfo" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Button = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + type: OpenLayers.Control.TYPE_BUTTON, │ │ │ │ │ + trigger: function() {}, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Button" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.ZoomOut = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ + trigger: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.zoomOut() │ │ │ │ │ } │ │ │ │ │ - return propagate │ │ │ │ │ }, │ │ │ │ │ - dragmove: function(evt) { │ │ │ │ │ - this.lastMoveEvt = evt; │ │ │ │ │ - if (this.started && !this.timeoutId && (evt.xy.x != this.last.x || evt.xy.y != this.last.y)) { │ │ │ │ │ - if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ - if (evt.element === document) { │ │ │ │ │ - this.adjustXY(evt); │ │ │ │ │ - this.setEvent(evt) │ │ │ │ │ - } else { │ │ │ │ │ - this.removeDocumentEvents() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.interval > 0) { │ │ │ │ │ - this.timeoutId = setTimeout(OpenLayers.Function.bind(this.removeTimeout, this), this.interval) │ │ │ │ │ - } │ │ │ │ │ - this.dragging = true; │ │ │ │ │ - this.move(evt); │ │ │ │ │ - this.callback("move", [evt.xy]); │ │ │ │ │ - if (!this.oldOnselectstart) { │ │ │ │ │ - this.oldOnselectstart = document.onselectstart; │ │ │ │ │ - document.onselectstart = OpenLayers.Function.False │ │ │ │ │ - } │ │ │ │ │ - this.last = evt.xy │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomOut" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.SLDSelect = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + clearOnDeactivate: false, │ │ │ │ │ + layers: null, │ │ │ │ │ + callbacks: null, │ │ │ │ │ + selectionSymbolizer: { │ │ │ │ │ + Polygon: { │ │ │ │ │ + fillColor: "#FF0000", │ │ │ │ │ + stroke: false │ │ │ │ │ + }, │ │ │ │ │ + Line: { │ │ │ │ │ + strokeColor: "#FF0000", │ │ │ │ │ + strokeWidth: 2 │ │ │ │ │ + }, │ │ │ │ │ + Point: { │ │ │ │ │ + graphicName: "square", │ │ │ │ │ + fillColor: "#FF0000", │ │ │ │ │ + pointRadius: 5 │ │ │ │ │ } │ │ │ │ │ - return true │ │ │ │ │ }, │ │ │ │ │ - dragend: function(evt) { │ │ │ │ │ - if (this.started) { │ │ │ │ │ - if (this.documentDrag === true && this.documentEvents) { │ │ │ │ │ - this.adjustXY(evt); │ │ │ │ │ - this.removeDocumentEvents() │ │ │ │ │ - } │ │ │ │ │ - var dragged = this.start != this.last; │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ - this.up(evt); │ │ │ │ │ - this.callback("up", [evt.xy]); │ │ │ │ │ - if (dragged) { │ │ │ │ │ - this.callback("done", [evt.xy]) │ │ │ │ │ + layerOptions: null, │ │ │ │ │ + sketchStyle: null, │ │ │ │ │ + wfsCache: {}, │ │ │ │ │ + layerCache: {}, │ │ │ │ │ + initialize: function(handler, options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.callbacks = OpenLayers.Util.extend({ │ │ │ │ │ + done: this.select, │ │ │ │ │ + click: this.select │ │ │ │ │ + }, this.callbacks); │ │ │ │ │ + this.handlerOptions = this.handlerOptions || {}; │ │ │ │ │ + this.layerOptions = OpenLayers.Util.applyDefaults(this.layerOptions, { │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + tileOptions: { │ │ │ │ │ + maxGetUrlLength: 2048 │ │ │ │ │ } │ │ │ │ │ - document.onselectstart = this.oldOnselectstart │ │ │ │ │ + }); │ │ │ │ │ + if (this.sketchStyle) { │ │ │ │ │ + this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults(this.handlerOptions.layerOptions, { │ │ │ │ │ + styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ + default: this.sketchStyle │ │ │ │ │ + }) │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - down: function(evt) {}, │ │ │ │ │ - move: function(evt) {}, │ │ │ │ │ - up: function(evt) {}, │ │ │ │ │ - out: function(evt) {}, │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - return this.dragstart(evt) │ │ │ │ │ - }, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - return this.dragstart(evt) │ │ │ │ │ - }, │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - return this.dragmove(evt) │ │ │ │ │ - }, │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - return this.dragmove(evt) │ │ │ │ │ + this.handler = new handler(this, this.callbacks, this.handlerOptions) │ │ │ │ │ }, │ │ │ │ │ - removeTimeout: function() { │ │ │ │ │ - this.timeoutId = null; │ │ │ │ │ - if (this.dragging) { │ │ │ │ │ - this.mousemove(this.lastMoveEvt) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + for (var key in this.layerCache) { │ │ │ │ │ + delete this.layerCache[key] │ │ │ │ │ } │ │ │ │ │ + for (var key in this.wfsCache) { │ │ │ │ │ + delete this.wfsCache[key] │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - return this.dragend(evt) │ │ │ │ │ + coupleLayerVisiblity: function(evt) { │ │ │ │ │ + this.setVisibility(evt.object.getVisibility()) │ │ │ │ │ }, │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - evt.xy = this.last; │ │ │ │ │ - return this.dragend(evt) │ │ │ │ │ + createSelectionLayer: function(source) { │ │ │ │ │ + var selectionLayer; │ │ │ │ │ + if (!this.layerCache[source.id]) { │ │ │ │ │ + selectionLayer = new OpenLayers.Layer.WMS(source.name, source.url, source.params, OpenLayers.Util.applyDefaults(this.layerOptions, source.getOptions())); │ │ │ │ │ + this.layerCache[source.id] = selectionLayer; │ │ │ │ │ + if (this.layerOptions.displayInLayerSwitcher === false) { │ │ │ │ │ + source.events.on({ │ │ │ │ │ + visibilitychanged: this.coupleLayerVisiblity, │ │ │ │ │ + scope: selectionLayer │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + this.map.addLayer(selectionLayer) │ │ │ │ │ + } else { │ │ │ │ │ + selectionLayer = this.layerCache[source.id] │ │ │ │ │ + } │ │ │ │ │ + return selectionLayer │ │ │ │ │ }, │ │ │ │ │ - mouseout: function(evt) { │ │ │ │ │ - if (this.started && OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ - if (this.documentDrag === true) { │ │ │ │ │ - this.addDocumentEvents() │ │ │ │ │ - } else { │ │ │ │ │ - var dragged = this.start != this.last; │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown"); │ │ │ │ │ - this.out(evt); │ │ │ │ │ - this.callback("out", []); │ │ │ │ │ - if (dragged) { │ │ │ │ │ - this.callback("done", [evt.xy]) │ │ │ │ │ + createSLD: function(layer, filters, geometryAttributes) { │ │ │ │ │ + var sld = { │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + namedLayers: {} │ │ │ │ │ + }; │ │ │ │ │ + var layerNames = [layer.params.LAYERS].join(",").split(","); │ │ │ │ │ + for (var i = 0, len = layerNames.length; i < len; i++) { │ │ │ │ │ + var name = layerNames[i]; │ │ │ │ │ + sld.namedLayers[name] = { │ │ │ │ │ + name: name, │ │ │ │ │ + userStyles: [] │ │ │ │ │ + }; │ │ │ │ │ + var symbolizer = this.selectionSymbolizer; │ │ │ │ │ + var geometryAttribute = geometryAttributes[i]; │ │ │ │ │ + if (geometryAttribute.type.indexOf("Polygon") >= 0) { │ │ │ │ │ + symbolizer = { │ │ │ │ │ + Polygon: this.selectionSymbolizer["Polygon"] │ │ │ │ │ } │ │ │ │ │ - if (document.onselectstart) { │ │ │ │ │ - document.onselectstart = this.oldOnselectstart │ │ │ │ │ + } else if (geometryAttribute.type.indexOf("LineString") >= 0) { │ │ │ │ │ + symbolizer = { │ │ │ │ │ + Line: this.selectionSymbolizer["Line"] │ │ │ │ │ + } │ │ │ │ │ + } else if (geometryAttribute.type.indexOf("Point") >= 0) { │ │ │ │ │ + symbolizer = { │ │ │ │ │ + Point: this.selectionSymbolizer["Point"] │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + var filter = filters[i]; │ │ │ │ │ + sld.namedLayers[name].userStyles.push({ │ │ │ │ │ + name: "default", │ │ │ │ │ + rules: [new OpenLayers.Rule({ │ │ │ │ │ + symbolizer: symbolizer, │ │ │ │ │ + filter: filter, │ │ │ │ │ + maxScaleDenominator: layer.options.minScale │ │ │ │ │ + })] │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - return this.start == this.last │ │ │ │ │ + return new OpenLayers.Format.SLD({ │ │ │ │ │ + srsName: this.map.getProjection() │ │ │ │ │ + }).write(sld) │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - activated = true │ │ │ │ │ + parseDescribeLayer: function(request) { │ │ │ │ │ + var format = new OpenLayers.Format.WMSDescribeLayer; │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.dragging = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - deactivated = true; │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDragDown") │ │ │ │ │ + var describeLayer = format.read(doc); │ │ │ │ │ + var typeNames = []; │ │ │ │ │ + var url = null; │ │ │ │ │ + for (var i = 0, len = describeLayer.length; i < len; i++) { │ │ │ │ │ + if (describeLayer[i].owsType == "WFS") { │ │ │ │ │ + typeNames.push(describeLayer[i].typeName); │ │ │ │ │ + url = describeLayer[i].owsURL │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - adjustXY: function(evt) { │ │ │ │ │ - var pos = OpenLayers.Util.pagePosition(this.map.viewPortDiv); │ │ │ │ │ - evt.xy.x -= pos[0]; │ │ │ │ │ - evt.xy.y -= pos[1] │ │ │ │ │ - }, │ │ │ │ │ - addDocumentEvents: function() { │ │ │ │ │ - OpenLayers.Element.addClass(document.body, "olDragDown"); │ │ │ │ │ - this.documentEvents = true; │ │ │ │ │ - OpenLayers.Event.observe(document, "mousemove", this._docMove); │ │ │ │ │ - OpenLayers.Event.observe(document, "mouseup", this._docUp) │ │ │ │ │ - }, │ │ │ │ │ - removeDocumentEvents: function() { │ │ │ │ │ - OpenLayers.Element.removeClass(document.body, "olDragDown"); │ │ │ │ │ - this.documentEvents = false; │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mousemove", this._docMove); │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mouseup", this._docUp) │ │ │ │ │ + var options = { │ │ │ │ │ + url: url, │ │ │ │ │ + params: { │ │ │ │ │ + SERVICE: "WFS", │ │ │ │ │ + TYPENAME: typeNames.toString(), │ │ │ │ │ + REQUEST: "DescribeFeatureType", │ │ │ │ │ + VERSION: "1.0.0" │ │ │ │ │ + }, │ │ │ │ │ + callback: function(request) { │ │ │ │ │ + var format = new OpenLayers.Format.WFSDescribeFeatureType; │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ + } │ │ │ │ │ + var describeFeatureType = format.read(doc); │ │ │ │ │ + this.control.wfsCache[this.layer.id] = describeFeatureType; │ │ │ │ │ + this.control._queue && this.control.applySelection() │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Request.GET(options) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Drag" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.RegularPolygon = OpenLayers.Class(OpenLayers.Handler.Drag, { │ │ │ │ │ - sides: 4, │ │ │ │ │ - radius: null, │ │ │ │ │ - snapAngle: null, │ │ │ │ │ - snapToggle: "shiftKey", │ │ │ │ │ - layerOptions: null, │ │ │ │ │ - persist: false, │ │ │ │ │ - irregular: false, │ │ │ │ │ - citeCompliant: false, │ │ │ │ │ - angle: null, │ │ │ │ │ - fixedRadius: false, │ │ │ │ │ - feature: null, │ │ │ │ │ - layer: null, │ │ │ │ │ - origin: null, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - if (!(options && options.layerOptions && options.layerOptions.styleMap)) { │ │ │ │ │ - this.style = OpenLayers.Util.extend(OpenLayers.Feature.Vector.style["default"], {}) │ │ │ │ │ + getGeometryAttributes: function(layer) { │ │ │ │ │ + var result = []; │ │ │ │ │ + var cache = this.wfsCache[layer.id]; │ │ │ │ │ + for (var i = 0, len = cache.featureTypes.length; i < len; i++) { │ │ │ │ │ + var typeName = cache.featureTypes[i]; │ │ │ │ │ + var properties = typeName.properties; │ │ │ │ │ + for (var j = 0, lenj = properties.length; j < lenj; j++) { │ │ │ │ │ + var property = properties[j]; │ │ │ │ │ + var type = property.type; │ │ │ │ │ + if (type.indexOf("LineString") >= 0 || type.indexOf("GeometryAssociationType") >= 0 || type.indexOf("GeometryPropertyType") >= 0 || type.indexOf("Point") >= 0 || type.indexOf("Polygon") >= 0) { │ │ │ │ │ + result.push(property) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Handler.Drag.prototype.initialize.apply(this, [control, callbacks, options]); │ │ │ │ │ - this.options = options ? options : {} │ │ │ │ │ - }, │ │ │ │ │ - setOptions: function(newOptions) { │ │ │ │ │ - OpenLayers.Util.extend(this.options, newOptions); │ │ │ │ │ - OpenLayers.Util.extend(this, newOptions) │ │ │ │ │ + return result │ │ │ │ │ }, │ │ │ │ │ activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.Drag.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - var options = OpenLayers.Util.extend({ │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - calculateInRange: OpenLayers.Function.True, │ │ │ │ │ - wrapDateLine: this.citeCompliant │ │ │ │ │ - }, this.layerOptions); │ │ │ │ │ - this.layer = new OpenLayers.Layer.Vector(this.CLASS_NAME, options); │ │ │ │ │ - this.map.addLayer(this.layer); │ │ │ │ │ - activated = true │ │ │ │ │ + var activated = OpenLayers.Control.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.layers[i]; │ │ │ │ │ + if (layer && !this.wfsCache[layer.id]) { │ │ │ │ │ + var options = { │ │ │ │ │ + url: layer.url, │ │ │ │ │ + params: { │ │ │ │ │ + SERVICE: "WMS", │ │ │ │ │ + VERSION: layer.params.VERSION, │ │ │ │ │ + LAYERS: layer.params.LAYERS, │ │ │ │ │ + REQUEST: "DescribeLayer" │ │ │ │ │ + }, │ │ │ │ │ + callback: this.parseDescribeLayer, │ │ │ │ │ + scope: { │ │ │ │ │ + layer: layer, │ │ │ │ │ + control: this │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Request.GET(options) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ return activated │ │ │ │ │ }, │ │ │ │ │ deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.Drag.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - if (this.dragging) { │ │ │ │ │ - this.cancel() │ │ │ │ │ - } │ │ │ │ │ - if (this.layer.map != null) { │ │ │ │ │ - this.layer.destroy(false); │ │ │ │ │ - if (this.feature) { │ │ │ │ │ - this.feature.destroy() │ │ │ │ │ + var deactivated = OpenLayers.Control.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.layers[i]; │ │ │ │ │ + if (layer && this.clearOnDeactivate === true) { │ │ │ │ │ + var layerCache = this.layerCache; │ │ │ │ │ + var selectionLayer = layerCache[layer.id]; │ │ │ │ │ + if (selectionLayer) { │ │ │ │ │ + layer.events.un({ │ │ │ │ │ + visibilitychanged: this.coupleLayerVisiblity, │ │ │ │ │ + scope: selectionLayer │ │ │ │ │ + }); │ │ │ │ │ + selectionLayer.destroy(); │ │ │ │ │ + delete layerCache[layer.id] │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.layer = null; │ │ │ │ │ - this.feature = null; │ │ │ │ │ - deactivated = true │ │ │ │ │ } │ │ │ │ │ return deactivated │ │ │ │ │ }, │ │ │ │ │ - down: function(evt) { │ │ │ │ │ - this.fixedRadius = !!this.radius; │ │ │ │ │ - var maploc = this.layer.getLonLatFromViewPortPx(evt.xy); │ │ │ │ │ - this.origin = new OpenLayers.Geometry.Point(maploc.lon, maploc.lat); │ │ │ │ │ - if (!this.fixedRadius || this.irregular) { │ │ │ │ │ - this.radius = this.map.getResolution() │ │ │ │ │ - } │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.clear() │ │ │ │ │ - } │ │ │ │ │ - this.feature = new OpenLayers.Feature.Vector; │ │ │ │ │ - this.createGeometry(); │ │ │ │ │ - this.callback("create", [this.origin, this.feature]); │ │ │ │ │ - this.layer.addFeatures([this.feature], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.layer.drawFeature(this.feature, this.style) │ │ │ │ │ - }, │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - var maploc = this.layer.getLonLatFromViewPortPx(evt.xy); │ │ │ │ │ - var point = new OpenLayers.Geometry.Point(maploc.lon, maploc.lat); │ │ │ │ │ - if (this.irregular) { │ │ │ │ │ - var ry = Math.sqrt(2) * Math.abs(point.y - this.origin.y) / 2; │ │ │ │ │ - this.radius = Math.max(this.map.getResolution() / 2, ry) │ │ │ │ │ - } else if (this.fixedRadius) { │ │ │ │ │ - this.origin = point │ │ │ │ │ + setLayers: function(layers) { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.layers = layers; │ │ │ │ │ + this.activate() │ │ │ │ │ } else { │ │ │ │ │ - this.calculateAngle(point, evt); │ │ │ │ │ - this.radius = Math.max(this.map.getResolution() / 2, point.distanceTo(this.origin)) │ │ │ │ │ + this.layers = layers │ │ │ │ │ } │ │ │ │ │ - this.modifyGeometry(); │ │ │ │ │ - if (this.irregular) { │ │ │ │ │ - var dx = point.x - this.origin.x; │ │ │ │ │ - var dy = point.y - this.origin.y; │ │ │ │ │ - var ratio; │ │ │ │ │ - if (dy == 0) { │ │ │ │ │ - ratio = dx / (this.radius * Math.sqrt(2)) │ │ │ │ │ + }, │ │ │ │ │ + createFilter: function(geometryAttribute, geometry) { │ │ │ │ │ + var filter = null; │ │ │ │ │ + if (this.handler instanceof OpenLayers.Handler.RegularPolygon) { │ │ │ │ │ + if (this.handler.irregular === true) { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry.getBounds() │ │ │ │ │ + }) │ │ │ │ │ } else { │ │ │ │ │ - ratio = dx / dy │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } else if (this.handler instanceof OpenLayers.Handler.Polygon) { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry │ │ │ │ │ + }) │ │ │ │ │ + } else if (this.handler instanceof OpenLayers.Handler.Path) { │ │ │ │ │ + if (geometryAttribute.type.indexOf("Point") >= 0) { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + distance: this.map.getExtent().getWidth() * .01, │ │ │ │ │ + distanceUnits: this.map.getUnits(), │ │ │ │ │ + value: geometry │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } else if (this.handler instanceof OpenLayers.Handler.Click) { │ │ │ │ │ + if (geometryAttribute.type.indexOf("Polygon") >= 0) { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + value: geometry │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ + property: geometryAttribute.name, │ │ │ │ │ + distance: this.map.getExtent().getWidth() * .01, │ │ │ │ │ + distanceUnits: this.map.getUnits(), │ │ │ │ │ + value: geometry │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - this.feature.geometry.resize(1, this.origin, ratio); │ │ │ │ │ - this.feature.geometry.move(dx / 2, dy / 2) │ │ │ │ │ } │ │ │ │ │ - this.layer.drawFeature(this.feature, this.style) │ │ │ │ │ + return filter │ │ │ │ │ }, │ │ │ │ │ - up: function(evt) { │ │ │ │ │ - this.finalize(); │ │ │ │ │ - if (this.start == this.last) { │ │ │ │ │ - this.callback("done", [evt.xy]) │ │ │ │ │ + select: function(geometry) { │ │ │ │ │ + this._queue = function() { │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.layers[i]; │ │ │ │ │ + var geometryAttributes = this.getGeometryAttributes(layer); │ │ │ │ │ + var filters = []; │ │ │ │ │ + for (var j = 0, lenj = geometryAttributes.length; j < lenj; j++) { │ │ │ │ │ + var geometryAttribute = geometryAttributes[j]; │ │ │ │ │ + if (geometryAttribute !== null) { │ │ │ │ │ + if (!(geometry instanceof OpenLayers.Geometry)) { │ │ │ │ │ + var point = this.map.getLonLatFromPixel(geometry.xy); │ │ │ │ │ + geometry = new OpenLayers.Geometry.Point(point.lon, point.lat) │ │ │ │ │ + } │ │ │ │ │ + var filter = this.createFilter(geometryAttribute, geometry); │ │ │ │ │ + if (filter !== null) { │ │ │ │ │ + filters.push(filter) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var selectionLayer = this.createSelectionLayer(layer); │ │ │ │ │ + this.events.triggerEvent("selected", { │ │ │ │ │ + layer: layer, │ │ │ │ │ + filters: filters │ │ │ │ │ + }); │ │ │ │ │ + var sld = this.createSLD(layer, filters, geometryAttributes); │ │ │ │ │ + selectionLayer.mergeNewParams({ │ │ │ │ │ + SLD_BODY: sld │ │ │ │ │ + }); │ │ │ │ │ + delete this._queue │ │ │ │ │ + } │ │ │ │ │ + }; │ │ │ │ │ + this.applySelection() │ │ │ │ │ + }, │ │ │ │ │ + applySelection: function() { │ │ │ │ │ + var canApply = true; │ │ │ │ │ + for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ + if (!this.wfsCache[this.layers[i].id]) { │ │ │ │ │ + canApply = false; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + canApply && this._queue.call(this) │ │ │ │ │ }, │ │ │ │ │ - out: function(evt) { │ │ │ │ │ - this.finalize() │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.SLDSelect" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.PanZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + slideFactor: 50, │ │ │ │ │ + slideRatio: null, │ │ │ │ │ + buttons: null, │ │ │ │ │ + position: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + this.position = new OpenLayers.Pixel(OpenLayers.Control.PanZoom.X, OpenLayers.Control.PanZoom.Y); │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - createGeometry: function() { │ │ │ │ │ - this.angle = Math.PI * (1 / this.sides - 1 / 2); │ │ │ │ │ - if (this.snapAngle) { │ │ │ │ │ - this.angle += this.snapAngle * (Math.PI / 180) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onButtonClick) │ │ │ │ │ } │ │ │ │ │ - this.feature.geometry = OpenLayers.Geometry.Polygon.createRegularPolygon(this.origin, this.radius, this.sides, this.snapAngle) │ │ │ │ │ + this.removeButtons(); │ │ │ │ │ + this.buttons = null; │ │ │ │ │ + this.position = null; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - modifyGeometry: function() { │ │ │ │ │ - var angle, point; │ │ │ │ │ - var ring = this.feature.geometry.components[0]; │ │ │ │ │ - if (ring.components.length != this.sides + 1) { │ │ │ │ │ - this.createGeometry(); │ │ │ │ │ - ring = this.feature.geometry.components[0] │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0; i < this.sides; ++i) { │ │ │ │ │ - point = ring.components[i]; │ │ │ │ │ - angle = this.angle + i * 2 * Math.PI / this.sides; │ │ │ │ │ - point.x = this.origin.x + this.radius * Math.cos(angle); │ │ │ │ │ - point.y = this.origin.y + this.radius * Math.sin(angle); │ │ │ │ │ - point.clearBounds() │ │ │ │ │ - } │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ }, │ │ │ │ │ - calculateAngle: function(point, evt) { │ │ │ │ │ - var alpha = Math.atan2(point.y - this.origin.y, point.x - this.origin.x); │ │ │ │ │ - if (this.snapAngle && (this.snapToggle && !evt[this.snapToggle])) { │ │ │ │ │ - var snapAngleRad = Math.PI / 180 * this.snapAngle; │ │ │ │ │ - this.angle = Math.round(alpha / snapAngleRad) * snapAngleRad │ │ │ │ │ - } else { │ │ │ │ │ - this.angle = alpha │ │ │ │ │ - } │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + px = this.position; │ │ │ │ │ + this.buttons = []; │ │ │ │ │ + var sz = { │ │ │ │ │ + w: 18, │ │ │ │ │ + h: 18 │ │ │ │ │ + }; │ │ │ │ │ + var centered = new OpenLayers.Pixel(px.x + sz.w / 2, px.y); │ │ │ │ │ + this._addButton("panup", "north-mini.png", centered, sz); │ │ │ │ │ + px.y = centered.y + sz.h; │ │ │ │ │ + this._addButton("panleft", "west-mini.png", px, sz); │ │ │ │ │ + this._addButton("panright", "east-mini.png", px.add(sz.w, 0), sz); │ │ │ │ │ + this._addButton("pandown", "south-mini.png", centered.add(0, sz.h * 2), sz); │ │ │ │ │ + this._addButton("zoomin", "zoom-plus-mini.png", centered.add(0, sz.h * 3 + 5), sz); │ │ │ │ │ + this._addButton("zoomworld", "zoom-world-mini.png", centered.add(0, sz.h * 4 + 5), sz); │ │ │ │ │ + this._addButton("zoomout", "zoom-minus-mini.png", centered.add(0, sz.h * 5 + 5), sz); │ │ │ │ │ + return this.div │ │ │ │ │ }, │ │ │ │ │ - cancel: function() { │ │ │ │ │ - this.callback("cancel", null); │ │ │ │ │ - this.finalize() │ │ │ │ │ + _addButton: function(id, img, xy, sz) { │ │ │ │ │ + var imgLocation = OpenLayers.Util.getImageLocation(img); │ │ │ │ │ + var btn = OpenLayers.Util.createAlphaImageDiv(this.id + "_" + id, xy, sz, imgLocation, "absolute"); │ │ │ │ │ + btn.style.cursor = "pointer"; │ │ │ │ │ + this.div.appendChild(btn); │ │ │ │ │ + btn.action = id; │ │ │ │ │ + btn.className = "olButton"; │ │ │ │ │ + this.buttons.push(btn); │ │ │ │ │ + return btn │ │ │ │ │ }, │ │ │ │ │ - finalize: function() { │ │ │ │ │ - this.origin = null; │ │ │ │ │ - this.radius = this.options.radius │ │ │ │ │ + _removeButton: function(btn) { │ │ │ │ │ + this.div.removeChild(btn); │ │ │ │ │ + OpenLayers.Util.removeItem(this.buttons, btn) │ │ │ │ │ }, │ │ │ │ │ - clear: function() { │ │ │ │ │ - if (this.layer) { │ │ │ │ │ - this.layer.renderer.clear(); │ │ │ │ │ - this.layer.destroyFeatures() │ │ │ │ │ + removeButtons: function() { │ │ │ │ │ + for (var i = this.buttons.length - 1; i >= 0; --i) { │ │ │ │ │ + this._removeButton(this.buttons[i]) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - callback: function(name, args) { │ │ │ │ │ - if (this.callbacks[name]) { │ │ │ │ │ - this.callbacks[name].apply(this.control, [this.feature.geometry.clone()]) │ │ │ │ │ - } │ │ │ │ │ - if (!this.persist && (name == "done" || name == "cancel")) { │ │ │ │ │ - this.clear() │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + var btn = evt.buttonElement; │ │ │ │ │ + switch (btn.action) { │ │ │ │ │ + case "panup": │ │ │ │ │ + this.map.pan(0, -this.getSlideFactor("h")); │ │ │ │ │ + break; │ │ │ │ │ + case "pandown": │ │ │ │ │ + this.map.pan(0, this.getSlideFactor("h")); │ │ │ │ │ + break; │ │ │ │ │ + case "panleft": │ │ │ │ │ + this.map.pan(-this.getSlideFactor("w"), 0); │ │ │ │ │ + break; │ │ │ │ │ + case "panright": │ │ │ │ │ + this.map.pan(this.getSlideFactor("w"), 0); │ │ │ │ │ + break; │ │ │ │ │ + case "zoomin": │ │ │ │ │ + this.map.zoomIn(); │ │ │ │ │ + break; │ │ │ │ │ + case "zoomout": │ │ │ │ │ + this.map.zoomOut(); │ │ │ │ │ + break; │ │ │ │ │ + case "zoomworld": │ │ │ │ │ + this.map.zoomToMaxExtent(); │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.RegularPolygon" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.Click = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - delay: 300, │ │ │ │ │ - single: true, │ │ │ │ │ - double: false, │ │ │ │ │ - pixelTolerance: 0, │ │ │ │ │ - dblclickTolerance: 13, │ │ │ │ │ - stopSingle: false, │ │ │ │ │ - stopDouble: false, │ │ │ │ │ - timerId: null, │ │ │ │ │ - down: null, │ │ │ │ │ - last: null, │ │ │ │ │ - first: null, │ │ │ │ │ - rightclickTimerId: null, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - this.startTouch(); │ │ │ │ │ - this.down = this.getEventInfo(evt); │ │ │ │ │ - this.last = this.getEventInfo(evt); │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - this.last = this.getEventInfo(evt); │ │ │ │ │ - return true │ │ │ │ │ + getSlideFactor: function(dim) { │ │ │ │ │ + return this.slideRatio ? this.map.getSize()[dim] * this.slideRatio : this.slideFactor │ │ │ │ │ }, │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - if (this.down) { │ │ │ │ │ - evt.xy = this.last.xy; │ │ │ │ │ - evt.lastTouches = this.last.touches; │ │ │ │ │ - this.handleSingle(evt); │ │ │ │ │ - this.down = null │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.PanZoom" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.PanZoom.X = 4; │ │ │ │ │ +OpenLayers.Control.PanZoom.Y = 4; │ │ │ │ │ +OpenLayers.Control.WMSGetFeatureInfo = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + hover: false, │ │ │ │ │ + drillDown: false, │ │ │ │ │ + maxFeatures: 10, │ │ │ │ │ + clickCallback: "click", │ │ │ │ │ + output: "features", │ │ │ │ │ + layers: null, │ │ │ │ │ + queryVisible: false, │ │ │ │ │ + url: null, │ │ │ │ │ + layerUrls: null, │ │ │ │ │ + infoFormat: "text/html", │ │ │ │ │ + vendorParams: {}, │ │ │ │ │ + format: null, │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + handler: null, │ │ │ │ │ + hoverRequest: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!this.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.WMSGetFeatureInfo(options.formatOptions) │ │ │ │ │ + } │ │ │ │ │ + if (this.drillDown === true) { │ │ │ │ │ + this.hover = false │ │ │ │ │ + } │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ + move: this.cancelHover, │ │ │ │ │ + pause: this.getInfoForHover │ │ │ │ │ + }, OpenLayers.Util.extend(this.handlerOptions.hover || {}, { │ │ │ │ │ + delay: 250 │ │ │ │ │ + })) │ │ │ │ │ + } else { │ │ │ │ │ + var callbacks = {}; │ │ │ │ │ + callbacks[this.clickCallback] = this.getInfoForClick; │ │ │ │ │ + this.handler = new OpenLayers.Handler.Click(this, callbacks, this.handlerOptions.click || {}) │ │ │ │ │ } │ │ │ │ │ - return true │ │ │ │ │ }, │ │ │ │ │ - mousedown: function(evt) { │ │ │ │ │ - this.down = this.getEventInfo(evt); │ │ │ │ │ - this.last = this.getEventInfo(evt); │ │ │ │ │ - return true │ │ │ │ │ + getInfoForClick: function(evt) { │ │ │ │ │ + this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ + xy: evt.xy │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + this.request(evt.xy, {}) │ │ │ │ │ }, │ │ │ │ │ - mouseup: function(evt) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - if (this.checkModifiers(evt) && this.control.handleRightClicks && OpenLayers.Event.isRightClick(evt)) { │ │ │ │ │ - propagate = this.rightclick(evt) │ │ │ │ │ - } │ │ │ │ │ - return propagate │ │ │ │ │ + getInfoForHover: function(evt) { │ │ │ │ │ + this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ + xy: evt.xy │ │ │ │ │ + }); │ │ │ │ │ + this.request(evt.xy, { │ │ │ │ │ + hover: true │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - rightclick: function(evt) { │ │ │ │ │ - if (this.passesTolerance(evt)) { │ │ │ │ │ - if (this.rightclickTimerId != null) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.callback("dblrightclick", [evt]); │ │ │ │ │ - return !this.stopDouble │ │ │ │ │ - } else { │ │ │ │ │ - var clickEvent = this["double"] ? OpenLayers.Util.extend({}, evt) : this.callback("rightclick", [evt]); │ │ │ │ │ - var delayedRightCall = OpenLayers.Function.bind(this.delayedRightCall, this, clickEvent); │ │ │ │ │ - this.rightclickTimerId = window.setTimeout(delayedRightCall, this.delay) │ │ │ │ │ - } │ │ │ │ │ + cancelHover: function() { │ │ │ │ │ + if (this.hoverRequest) { │ │ │ │ │ + this.hoverRequest.abort(); │ │ │ │ │ + this.hoverRequest = null │ │ │ │ │ } │ │ │ │ │ - return !this.stopSingle │ │ │ │ │ }, │ │ │ │ │ - delayedRightCall: function(evt) { │ │ │ │ │ - this.rightclickTimerId = null; │ │ │ │ │ - if (evt) { │ │ │ │ │ - this.callback("rightclick", [evt]) │ │ │ │ │ + findLayers: function() { │ │ │ │ │ + var candidates = this.layers || this.map.layers; │ │ │ │ │ + var layers = []; │ │ │ │ │ + var layer, url; │ │ │ │ │ + for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = candidates[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.WMS && (!this.queryVisible || layer.getVisibility())) { │ │ │ │ │ + url = OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url; │ │ │ │ │ + if (this.drillDown === false && !this.url) { │ │ │ │ │ + this.url = url │ │ │ │ │ + } │ │ │ │ │ + if (this.drillDown === true || this.urlMatches(url)) { │ │ │ │ │ + layers.push(layer) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return layers │ │ │ │ │ }, │ │ │ │ │ - click: function(evt) { │ │ │ │ │ - if (!this.last) { │ │ │ │ │ - this.last = this.getEventInfo(evt) │ │ │ │ │ + urlMatches: function(url) { │ │ │ │ │ + var matches = OpenLayers.Util.isEquivalentUrl(this.url, url); │ │ │ │ │ + if (!matches && this.layerUrls) { │ │ │ │ │ + for (var i = 0, len = this.layerUrls.length; i < len; ++i) { │ │ │ │ │ + if (OpenLayers.Util.isEquivalentUrl(this.layerUrls[i], url)) { │ │ │ │ │ + matches = true; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.handleSingle(evt); │ │ │ │ │ - return !this.stopSingle │ │ │ │ │ - }, │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - this.handleDouble(evt); │ │ │ │ │ - return !this.stopDouble │ │ │ │ │ + return matches │ │ │ │ │ }, │ │ │ │ │ - handleDouble: function(evt) { │ │ │ │ │ - if (this.passesDblclickTolerance(evt)) { │ │ │ │ │ - if (this["double"]) { │ │ │ │ │ - this.callback("dblclick", [evt]) │ │ │ │ │ + buildWMSOptions: function(url, layers, clickPosition, format) { │ │ │ │ │ + var layerNames = [], │ │ │ │ │ + styleNames = []; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + if (layers[i].params.LAYERS != null) { │ │ │ │ │ + layerNames = layerNames.concat(layers[i].params.LAYERS); │ │ │ │ │ + styleNames = styleNames.concat(this.getStyleNames(layers[i])) │ │ │ │ │ } │ │ │ │ │ - this.clearTimer() │ │ │ │ │ + } │ │ │ │ │ + var firstLayer = layers[0]; │ │ │ │ │ + var projection = this.map.getProjection(); │ │ │ │ │ + var layerProj = firstLayer.projection; │ │ │ │ │ + if (layerProj && layerProj.equals(this.map.getProjectionObject())) { │ │ │ │ │ + projection = layerProj.getCode() │ │ │ │ │ + } │ │ │ │ │ + var params = OpenLayers.Util.extend({ │ │ │ │ │ + service: "WMS", │ │ │ │ │ + version: firstLayer.params.VERSION, │ │ │ │ │ + request: "GetFeatureInfo", │ │ │ │ │ + exceptions: firstLayer.params.EXCEPTIONS, │ │ │ │ │ + bbox: this.map.getExtent().toBBOX(null, firstLayer.reverseAxisOrder()), │ │ │ │ │ + feature_count: this.maxFeatures, │ │ │ │ │ + height: this.map.getSize().h, │ │ │ │ │ + width: this.map.getSize().w, │ │ │ │ │ + format: format, │ │ │ │ │ + info_format: firstLayer.params.INFO_FORMAT || this.infoFormat │ │ │ │ │ + }, parseFloat(firstLayer.params.VERSION) >= 1.3 ? { │ │ │ │ │ + crs: projection, │ │ │ │ │ + i: parseInt(clickPosition.x), │ │ │ │ │ + j: parseInt(clickPosition.y) │ │ │ │ │ + } : { │ │ │ │ │ + srs: projection, │ │ │ │ │ + x: parseInt(clickPosition.x), │ │ │ │ │ + y: parseInt(clickPosition.y) │ │ │ │ │ + }); │ │ │ │ │ + if (layerNames.length != 0) { │ │ │ │ │ + params = OpenLayers.Util.extend({ │ │ │ │ │ + layers: layerNames, │ │ │ │ │ + query_layers: layerNames, │ │ │ │ │ + styles: styleNames │ │ │ │ │ + }, params) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.applyDefaults(params, this.vendorParams); │ │ │ │ │ + return { │ │ │ │ │ + url: url, │ │ │ │ │ + params: OpenLayers.Util.upperCaseObject(params), │ │ │ │ │ + callback: function(request) { │ │ │ │ │ + this.handleResponse(clickPosition, request, url) │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - handleSingle: function(evt) { │ │ │ │ │ - if (this.passesTolerance(evt)) { │ │ │ │ │ - if (this.timerId != null) { │ │ │ │ │ - if (this.last.touches && this.last.touches.length === 1) { │ │ │ │ │ - if (this["double"]) { │ │ │ │ │ - OpenLayers.Event.preventDefault(evt) │ │ │ │ │ - } │ │ │ │ │ - this.handleDouble(evt) │ │ │ │ │ - } │ │ │ │ │ - if (!this.last.touches || this.last.touches.length !== 2) { │ │ │ │ │ - this.clearTimer() │ │ │ │ │ - } │ │ │ │ │ + getStyleNames: function(layer) { │ │ │ │ │ + var styleNames; │ │ │ │ │ + if (layer.params.STYLES) { │ │ │ │ │ + styleNames = layer.params.STYLES │ │ │ │ │ + } else { │ │ │ │ │ + if (OpenLayers.Util.isArray(layer.params.LAYERS)) { │ │ │ │ │ + styleNames = new Array(layer.params.LAYERS.length) │ │ │ │ │ } else { │ │ │ │ │ - this.first = this.getEventInfo(evt); │ │ │ │ │ - var clickEvent = this.single ? OpenLayers.Util.extend({}, evt) : null; │ │ │ │ │ - this.queuePotentialClick(clickEvent) │ │ │ │ │ + styleNames = layer.params.LAYERS.replace(/[^,]/g, "") │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return styleNames │ │ │ │ │ }, │ │ │ │ │ - queuePotentialClick: function(evt) { │ │ │ │ │ - this.timerId = window.setTimeout(OpenLayers.Function.bind(this.delayedCall, this, evt), this.delay) │ │ │ │ │ - }, │ │ │ │ │ - passesTolerance: function(evt) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - if (this.pixelTolerance != null && this.down && this.down.xy) { │ │ │ │ │ - passes = this.pixelTolerance >= this.down.xy.distanceTo(evt.xy); │ │ │ │ │ - if (passes && this.touch && this.down.touches.length === this.last.touches.length) { │ │ │ │ │ - for (var i = 0, ii = this.down.touches.length; i < ii; ++i) { │ │ │ │ │ - if (this.getTouchDistance(this.down.touches[i], this.last.touches[i]) > this.pixelTolerance) { │ │ │ │ │ - passes = false; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ + request: function(clickPosition, options) { │ │ │ │ │ + var layers = this.findLayers(); │ │ │ │ │ + if (layers.length == 0) { │ │ │ │ │ + this.events.triggerEvent("nogetfeatureinfo"); │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + options = options || {}; │ │ │ │ │ + if (this.drillDown === false) { │ │ │ │ │ + var wmsOptions = this.buildWMSOptions(this.url, layers, clickPosition, layers[0].params.FORMAT); │ │ │ │ │ + var request = OpenLayers.Request.GET(wmsOptions); │ │ │ │ │ + if (options.hover === true) { │ │ │ │ │ + this.hoverRequest = request │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + this._requestCount = 0; │ │ │ │ │ + this._numRequests = 0; │ │ │ │ │ + this.features = []; │ │ │ │ │ + var services = {}, │ │ │ │ │ + url; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + var layer = layers[i]; │ │ │ │ │ + var service, found = false; │ │ │ │ │ + url = OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url; │ │ │ │ │ + if (url in services) { │ │ │ │ │ + services[url].push(layer) │ │ │ │ │ + } else { │ │ │ │ │ + this._numRequests++; │ │ │ │ │ + services[url] = [layer] │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + var layers; │ │ │ │ │ + for (var url in services) { │ │ │ │ │ + layers = services[url]; │ │ │ │ │ + var wmsOptions = this.buildWMSOptions(url, layers, clickPosition, layers[0].params.FORMAT); │ │ │ │ │ + OpenLayers.Request.GET(wmsOptions) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return passes │ │ │ │ │ - }, │ │ │ │ │ - getTouchDistance: function(from, to) { │ │ │ │ │ - return Math.sqrt(Math.pow(from.clientX - to.clientX, 2) + Math.pow(from.clientY - to.clientY, 2)) │ │ │ │ │ }, │ │ │ │ │ - passesDblclickTolerance: function(evt) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - if (this.down && this.first) { │ │ │ │ │ - passes = this.down.xy.distanceTo(this.first.xy) <= this.dblclickTolerance │ │ │ │ │ - } │ │ │ │ │ - return passes │ │ │ │ │ + triggerGetFeatureInfo: function(request, xy, features) { │ │ │ │ │ + this.events.triggerEvent("getfeatureinfo", { │ │ │ │ │ + text: request.responseText, │ │ │ │ │ + features: features, │ │ │ │ │ + request: request, │ │ │ │ │ + xy: xy │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait") │ │ │ │ │ }, │ │ │ │ │ - clearTimer: function() { │ │ │ │ │ - if (this.timerId != null) { │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null │ │ │ │ │ + handleResponse: function(xy, request, url) { │ │ │ │ │ + var doc = request.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = request.responseText │ │ │ │ │ } │ │ │ │ │ - if (this.rightclickTimerId != null) { │ │ │ │ │ - window.clearTimeout(this.rightclickTimerId); │ │ │ │ │ - this.rightclickTimerId = null │ │ │ │ │ + var features = this.format.read(doc); │ │ │ │ │ + if (this.drillDown === false) { │ │ │ │ │ + this.triggerGetFeatureInfo(request, xy, features) │ │ │ │ │ + } else { │ │ │ │ │ + this._requestCount++; │ │ │ │ │ + if (this.output === "object") { │ │ │ │ │ + this._features = (this._features || []).concat({ │ │ │ │ │ + url: url, │ │ │ │ │ + features: features │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + this._features = (this._features || []).concat(features) │ │ │ │ │ + } │ │ │ │ │ + if (this._requestCount === this._numRequests) { │ │ │ │ │ + this.triggerGetFeatureInfo(request, xy, this._features.concat()); │ │ │ │ │ + delete this._features; │ │ │ │ │ + delete this._requestCount; │ │ │ │ │ + delete this._numRequests │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - delayedCall: function(evt) { │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - if (evt) { │ │ │ │ │ - this.callback("click", [evt]) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.WMSGetFeatureInfo" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.ModifyFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + documentDrag: false, │ │ │ │ │ + geometryTypes: null, │ │ │ │ │ + clickout: true, │ │ │ │ │ + toggle: true, │ │ │ │ │ + standalone: false, │ │ │ │ │ + layer: null, │ │ │ │ │ + feature: null, │ │ │ │ │ + vertex: null, │ │ │ │ │ + vertices: null, │ │ │ │ │ + virtualVertices: null, │ │ │ │ │ + handlers: null, │ │ │ │ │ + deleteCodes: null, │ │ │ │ │ + virtualStyle: null, │ │ │ │ │ + vertexRenderIntent: null, │ │ │ │ │ + mode: null, │ │ │ │ │ + createVertices: true, │ │ │ │ │ + modified: false, │ │ │ │ │ + radiusHandle: null, │ │ │ │ │ + dragHandle: null, │ │ │ │ │ + onModificationStart: function() {}, │ │ │ │ │ + onModification: function() {}, │ │ │ │ │ + onModificationEnd: function() {}, │ │ │ │ │ + initialize: function(layer, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + this.virtualVertices = []; │ │ │ │ │ + this.virtualStyle = OpenLayers.Util.extend({}, this.layer.style || this.layer.styleMap.createSymbolizer(null, options.vertexRenderIntent)); │ │ │ │ │ + this.virtualStyle.fillOpacity = .3; │ │ │ │ │ + this.virtualStyle.strokeOpacity = .3; │ │ │ │ │ + this.deleteCodes = [46, 68]; │ │ │ │ │ + this.mode = OpenLayers.Control.ModifyFeature.RESHAPE; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (!OpenLayers.Util.isArray(this.deleteCodes)) { │ │ │ │ │ + this.deleteCodes = [this.deleteCodes] │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - getEventInfo: function(evt) { │ │ │ │ │ - var touches; │ │ │ │ │ - if (evt.touches) { │ │ │ │ │ - var len = evt.touches.length; │ │ │ │ │ - touches = new Array(len); │ │ │ │ │ - var touch; │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - touch = evt.touches[i]; │ │ │ │ │ - touches[i] = { │ │ │ │ │ - clientX: touch.olClientX, │ │ │ │ │ - clientY: touch.olClientY │ │ │ │ │ + var dragCallbacks = { │ │ │ │ │ + down: function(pixel) { │ │ │ │ │ + this.vertex = null; │ │ │ │ │ + var feature = this.layer.getFeatureFromEvent(this.handlers.drag.evt); │ │ │ │ │ + if (feature) { │ │ │ │ │ + this.dragStart(feature) │ │ │ │ │ + } else if (this.clickout) { │ │ │ │ │ + this._unselect = this.feature │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + move: function(pixel) { │ │ │ │ │ + delete this._unselect; │ │ │ │ │ + if (this.vertex) { │ │ │ │ │ + this.dragVertex(this.vertex, pixel) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + up: function() { │ │ │ │ │ + this.handlers.drag.stopDown = false; │ │ │ │ │ + if (this._unselect) { │ │ │ │ │ + this.unselectFeature(this._unselect); │ │ │ │ │ + delete this._unselect │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + done: function(pixel) { │ │ │ │ │ + if (this.vertex) { │ │ │ │ │ + this.dragComplete(this.vertex) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + }; │ │ │ │ │ + var dragOptions = { │ │ │ │ │ + documentDrag: this.documentDrag, │ │ │ │ │ + stopDown: false │ │ │ │ │ + }; │ │ │ │ │ + var keyboardOptions = { │ │ │ │ │ + keydown: this.handleKeypress │ │ │ │ │ + }; │ │ │ │ │ + this.handlers = { │ │ │ │ │ + keyboard: new OpenLayers.Handler.Keyboard(this, keyboardOptions), │ │ │ │ │ + drag: new OpenLayers.Handler.Drag(this, dragCallbacks, dragOptions) │ │ │ │ │ } │ │ │ │ │ - return { │ │ │ │ │ - xy: evt.xy, │ │ │ │ │ - touches: touches │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + removelayer: this.handleMapEvents, │ │ │ │ │ + changelayer: this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ + this.layer = null; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, []) │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + this.moveLayerToTop(); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + removelayer: this.handleMapEvents, │ │ │ │ │ + changelayer: this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + return this.handlers.keyboard.activate() && this.handlers.drag.activate() && OpenLayers.Control.prototype.activate.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ deactivate: function() { │ │ │ │ │ var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.down = null; │ │ │ │ │ - this.first = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.moveLayerBack(); │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + removelayer: this.handleMapEvents, │ │ │ │ │ + changelayer: this.handleMapEvents, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.layer.removeFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.layer.removeFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + this.handlers.drag.deactivate(); │ │ │ │ │ + this.handlers.keyboard.deactivate(); │ │ │ │ │ + var feature = this.feature; │ │ │ │ │ + if (feature && feature.geometry && feature.layer) { │ │ │ │ │ + this.unselectFeature(feature) │ │ │ │ │ + } │ │ │ │ │ deactivated = true │ │ │ │ │ } │ │ │ │ │ return deactivated │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Click" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.Path = OpenLayers.Class(OpenLayers.Handler.Point, { │ │ │ │ │ - line: null, │ │ │ │ │ - maxVertices: null, │ │ │ │ │ - doubleTouchTolerance: 20, │ │ │ │ │ - freehand: false, │ │ │ │ │ - freehandToggle: "shiftKey", │ │ │ │ │ - timerId: null, │ │ │ │ │ - redoStack: null, │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.line = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.LineString([this.point.geometry])); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.line, this.point], { │ │ │ │ │ - silent: true │ │ │ │ │ + beforeSelectFeature: function(feature) { │ │ │ │ │ + return this.layer.events.triggerEvent("beforefeaturemodified", { │ │ │ │ │ + feature: feature │ │ │ │ │ }) │ │ │ │ │ }, │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - OpenLayers.Handler.Point.prototype.destroyFeature.call(this, force); │ │ │ │ │ - this.line = null │ │ │ │ │ - }, │ │ │ │ │ - destroyPersistedFeature: function() { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - if (layer && layer.features.length > 2) { │ │ │ │ │ - this.layer.features[0].destroy() │ │ │ │ │ + selectFeature: function(feature) { │ │ │ │ │ + if (this.feature === feature || this.geometryTypes && OpenLayers.Util.indexOf(this.geometryTypes, feature.geometry.CLASS_NAME) == -1) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - removePoint: function() { │ │ │ │ │ - if (this.point) { │ │ │ │ │ - this.layer.removeFeatures([this.point]) │ │ │ │ │ + if (this.beforeSelectFeature(feature) !== false) { │ │ │ │ │ + if (this.feature) { │ │ │ │ │ + this.unselectFeature(this.feature) │ │ │ │ │ + } │ │ │ │ │ + this.feature = feature; │ │ │ │ │ + this.layer.selectedFeatures.push(feature); │ │ │ │ │ + this.layer.drawFeature(feature, "select"); │ │ │ │ │ + this.modified = false; │ │ │ │ │ + this.resetVertices(); │ │ │ │ │ + this.onModificationStart(this.feature) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - addPoint: function(pixel) { │ │ │ │ │ - this.layer.removeFeatures([this.point]); │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat)); │ │ │ │ │ - this.line.geometry.addComponent(this.point.geometry, this.line.geometry.components.length); │ │ │ │ │ - this.layer.addFeatures([this.point]); │ │ │ │ │ - this.callback("point", [this.point.geometry, this.getGeometry()]); │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - delete this.redoStack │ │ │ │ │ - }, │ │ │ │ │ - insertXY: function(x, y) { │ │ │ │ │ - this.line.geometry.addComponent(new OpenLayers.Geometry.Point(x, y), this.getCurrentPointIndex()); │ │ │ │ │ - this.drawFeature(); │ │ │ │ │ - delete this.redoStack │ │ │ │ │ - }, │ │ │ │ │ - insertDeltaXY: function(dx, dy) { │ │ │ │ │ - var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ - var p0 = this.line.geometry.components[previousIndex]; │ │ │ │ │ - if (p0 && !isNaN(p0.x) && !isNaN(p0.y)) { │ │ │ │ │ - this.insertXY(p0.x + dx, p0.y + dy) │ │ │ │ │ + var modified = feature.modified; │ │ │ │ │ + if (feature.geometry && !(modified && modified.geometry)) { │ │ │ │ │ + this._originalGeometry = feature.geometry.clone() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - insertDirectionLength: function(direction, length) { │ │ │ │ │ - direction *= Math.PI / 180; │ │ │ │ │ - var dx = length * Math.cos(direction); │ │ │ │ │ - var dy = length * Math.sin(direction); │ │ │ │ │ - this.insertDeltaXY(dx, dy) │ │ │ │ │ - }, │ │ │ │ │ - insertDeflectionLength: function(deflection, length) { │ │ │ │ │ - var previousIndex = this.getCurrentPointIndex() - 1; │ │ │ │ │ - if (previousIndex > 0) { │ │ │ │ │ - var p1 = this.line.geometry.components[previousIndex]; │ │ │ │ │ - var p0 = this.line.geometry.components[previousIndex - 1]; │ │ │ │ │ - var theta = Math.atan2(p1.y - p0.y, p1.x - p0.x); │ │ │ │ │ - this.insertDirectionLength(theta * 180 / Math.PI + deflection, length) │ │ │ │ │ + unselectFeature: function(feature) { │ │ │ │ │ + this.layer.removeFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + this.layer.destroyFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.virtualVertices = []; │ │ │ │ │ + if (this.dragHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.dragHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + delete this.dragHandle │ │ │ │ │ } │ │ │ │ │ + if (this.radiusHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + delete this.radiusHandle │ │ │ │ │ + } │ │ │ │ │ + this.layer.drawFeature(this.feature, "default"); │ │ │ │ │ + this.feature = null; │ │ │ │ │ + OpenLayers.Util.removeItem(this.layer.selectedFeatures, feature); │ │ │ │ │ + this.onModificationEnd(feature); │ │ │ │ │ + this.layer.events.triggerEvent("afterfeaturemodified", { │ │ │ │ │ + feature: feature, │ │ │ │ │ + modified: this.modified │ │ │ │ │ + }); │ │ │ │ │ + this.modified = false │ │ │ │ │ }, │ │ │ │ │ - getCurrentPointIndex: function() { │ │ │ │ │ - return this.line.geometry.components.length - 1 │ │ │ │ │ - }, │ │ │ │ │ - undo: function() { │ │ │ │ │ - var geometry = this.line.geometry; │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var index = this.getCurrentPointIndex() - 1; │ │ │ │ │ - var target = components[index]; │ │ │ │ │ - var undone = geometry.removeComponent(target); │ │ │ │ │ - if (undone) { │ │ │ │ │ - if (this.touch && index > 0) { │ │ │ │ │ - components = geometry.components; │ │ │ │ │ - var lastpt = components[index - 1]; │ │ │ │ │ - var curptidx = this.getCurrentPointIndex(); │ │ │ │ │ - var curpt = components[curptidx]; │ │ │ │ │ - curpt.x = lastpt.x; │ │ │ │ │ - curpt.y = lastpt.y │ │ │ │ │ - } │ │ │ │ │ - if (!this.redoStack) { │ │ │ │ │ - this.redoStack = [] │ │ │ │ │ + dragStart: function(feature) { │ │ │ │ │ + var isPoint = feature.geometry.CLASS_NAME == "OpenLayers.Geometry.Point"; │ │ │ │ │ + if (!this.standalone && (!feature._sketch && isPoint || !feature._sketch)) { │ │ │ │ │ + if (this.toggle && this.feature === feature) { │ │ │ │ │ + this._unselect = feature │ │ │ │ │ } │ │ │ │ │ - this.redoStack.push(target); │ │ │ │ │ - this.drawFeature() │ │ │ │ │ + this.selectFeature(feature) │ │ │ │ │ } │ │ │ │ │ - return undone │ │ │ │ │ - }, │ │ │ │ │ - redo: function() { │ │ │ │ │ - var target = this.redoStack && this.redoStack.pop(); │ │ │ │ │ - if (target) { │ │ │ │ │ - this.line.geometry.addComponent(target, this.getCurrentPointIndex()); │ │ │ │ │ - this.drawFeature() │ │ │ │ │ + if (feature._sketch || isPoint) { │ │ │ │ │ + this.vertex = feature; │ │ │ │ │ + this.handlers.drag.stopDown = true │ │ │ │ │ } │ │ │ │ │ - return !!target │ │ │ │ │ - }, │ │ │ │ │ - freehandMode: function(evt) { │ │ │ │ │ - return this.freehandToggle && evt[this.freehandToggle] ? !this.freehand : this.freehand │ │ │ │ │ }, │ │ │ │ │ - modifyFeature: function(pixel, drawing) { │ │ │ │ │ - if (!this.line) { │ │ │ │ │ - this.createFeature(pixel) │ │ │ │ │ + dragVertex: function(vertex, pixel) { │ │ │ │ │ + var pos = this.map.getLonLatFromViewPortPx(pixel); │ │ │ │ │ + var geom = vertex.geometry; │ │ │ │ │ + geom.move(pos.lon - geom.x, pos.lat - geom.y); │ │ │ │ │ + this.modified = true; │ │ │ │ │ + if (this.feature.geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + this.layer.events.triggerEvent("vertexmodified", { │ │ │ │ │ + vertex: vertex.geometry, │ │ │ │ │ + feature: this.feature, │ │ │ │ │ + pixel: pixel │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + if (vertex._index) { │ │ │ │ │ + vertex.geometry.parent.addComponent(vertex.geometry, vertex._index); │ │ │ │ │ + delete vertex._index; │ │ │ │ │ + OpenLayers.Util.removeItem(this.virtualVertices, vertex); │ │ │ │ │ + this.vertices.push(vertex) │ │ │ │ │ + } else if (vertex == this.dragHandle) { │ │ │ │ │ + this.layer.removeFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + if (this.radiusHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.radiusHandle = null │ │ │ │ │ + } │ │ │ │ │ + } else if (vertex !== this.radiusHandle) { │ │ │ │ │ + this.layer.events.triggerEvent("vertexmodified", { │ │ │ │ │ + vertex: vertex.geometry, │ │ │ │ │ + feature: this.feature, │ │ │ │ │ + pixel: pixel │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + if (this.virtualVertices.length > 0) { │ │ │ │ │ + this.layer.destroyFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.virtualVertices = [] │ │ │ │ │ + } │ │ │ │ │ + this.layer.drawFeature(this.feature, this.standalone ? undefined : "select") │ │ │ │ │ } │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - this.point.geometry.x = lonlat.lon; │ │ │ │ │ - this.point.geometry.y = lonlat.lat; │ │ │ │ │ - this.callback("modify", [this.point.geometry, this.getSketch(), drawing]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.drawFeature() │ │ │ │ │ - }, │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.line, this.style); │ │ │ │ │ - this.layer.drawFeature(this.point, this.style) │ │ │ │ │ - }, │ │ │ │ │ - getSketch: function() { │ │ │ │ │ - return this.line │ │ │ │ │ + this.layer.drawFeature(vertex) │ │ │ │ │ }, │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.line && this.line.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiLineString([geometry]) │ │ │ │ │ - } │ │ │ │ │ - return geometry │ │ │ │ │ + dragComplete: function(vertex) { │ │ │ │ │ + this.resetVertices(); │ │ │ │ │ + this.setFeatureState(); │ │ │ │ │ + this.onModification(this.feature); │ │ │ │ │ + this.layer.events.triggerEvent("featuremodified", { │ │ │ │ │ + feature: this.feature │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - if (this.timerId && this.passesTolerance(this.lastTouchPx, evt.xy, this.doubleTouchTolerance)) { │ │ │ │ │ - this.finishGeometry(); │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null; │ │ │ │ │ - return false │ │ │ │ │ - } else { │ │ │ │ │ - if (this.timerId) { │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null │ │ │ │ │ + setFeatureState: function() { │ │ │ │ │ + if (this.feature.state != OpenLayers.State.INSERT && this.feature.state != OpenLayers.State.DELETE) { │ │ │ │ │ + this.feature.state = OpenLayers.State.UPDATE; │ │ │ │ │ + if (this.modified && this._originalGeometry) { │ │ │ │ │ + var feature = this.feature; │ │ │ │ │ + feature.modified = OpenLayers.Util.extend(feature.modified, { │ │ │ │ │ + geometry: this._originalGeometry │ │ │ │ │ + }); │ │ │ │ │ + delete this._originalGeometry │ │ │ │ │ } │ │ │ │ │ - this.timerId = window.setTimeout(OpenLayers.Function.bind(function() { │ │ │ │ │ - this.timerId = null │ │ │ │ │ - }, this), 300); │ │ │ │ │ - return OpenLayers.Handler.Point.prototype.touchstart.call(this, evt) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - down: function(evt) { │ │ │ │ │ - var stopDown = this.stopDown; │ │ │ │ │ - if (this.freehandMode(evt)) { │ │ │ │ │ - stopDown = true; │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp); │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ - } │ │ │ │ │ + resetVertices: function() { │ │ │ │ │ + if (this.vertices.length > 0) { │ │ │ │ │ + this.layer.removeFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.vertices = [] │ │ │ │ │ } │ │ │ │ │ - if (!this.touch && (!this.lastDown || !this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance))) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp) │ │ │ │ │ + if (this.virtualVertices.length > 0) { │ │ │ │ │ + this.layer.removeFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.virtualVertices = [] │ │ │ │ │ } │ │ │ │ │ - this.mouseDown = true; │ │ │ │ │ - this.lastDown = evt.xy; │ │ │ │ │ - this.stoppedDown = stopDown; │ │ │ │ │ - return !stopDown │ │ │ │ │ - }, │ │ │ │ │ - move: function(evt) { │ │ │ │ │ - if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature() │ │ │ │ │ + if (this.dragHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.dragHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.dragHandle = null │ │ │ │ │ + } │ │ │ │ │ + if (this.radiusHandle) { │ │ │ │ │ + this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.radiusHandle = null │ │ │ │ │ + } │ │ │ │ │ + if (this.feature && this.feature.geometry.CLASS_NAME != "OpenLayers.Geometry.Point") { │ │ │ │ │ + if (this.mode & OpenLayers.Control.ModifyFeature.DRAG) { │ │ │ │ │ + this.collectDragHandle() │ │ │ │ │ } │ │ │ │ │ - if (this.maxVertices && this.line && this.line.geometry.components.length === this.maxVertices) { │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize() │ │ │ │ │ - } else { │ │ │ │ │ - this.addPoint(evt.xy) │ │ │ │ │ + if (this.mode & (OpenLayers.Control.ModifyFeature.ROTATE | OpenLayers.Control.ModifyFeature.RESIZE)) { │ │ │ │ │ + this.collectRadiusHandle() │ │ │ │ │ + } │ │ │ │ │ + if (this.mode & OpenLayers.Control.ModifyFeature.RESHAPE) { │ │ │ │ │ + if (!(this.mode & OpenLayers.Control.ModifyFeature.RESIZE)) { │ │ │ │ │ + this.collectVertices() │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return false │ │ │ │ │ } │ │ │ │ │ - if (!this.touch && (!this.mouseDown || this.stoppedDown)) { │ │ │ │ │ - this.modifyFeature(evt.xy, !!this.lastUp) │ │ │ │ │ + }, │ │ │ │ │ + handleKeypress: function(evt) { │ │ │ │ │ + var code = evt.keyCode; │ │ │ │ │ + if (this.feature && OpenLayers.Util.indexOf(this.deleteCodes, code) != -1) { │ │ │ │ │ + var vertex = this.layer.getFeatureFromEvent(this.handlers.drag.evt); │ │ │ │ │ + if (vertex && OpenLayers.Util.indexOf(this.vertices, vertex) != -1 && !this.handlers.drag.dragging && vertex.geometry.parent) { │ │ │ │ │ + vertex.geometry.parent.removeComponent(vertex.geometry); │ │ │ │ │ + this.layer.events.triggerEvent("vertexremoved", { │ │ │ │ │ + vertex: vertex.geometry, │ │ │ │ │ + feature: this.feature, │ │ │ │ │ + pixel: evt.xy │ │ │ │ │ + }); │ │ │ │ │ + this.layer.drawFeature(this.feature, this.standalone ? undefined : "select"); │ │ │ │ │ + this.modified = true; │ │ │ │ │ + this.resetVertices(); │ │ │ │ │ + this.setFeatureState(); │ │ │ │ │ + this.onModification(this.feature); │ │ │ │ │ + this.layer.events.triggerEvent("featuremodified", { │ │ │ │ │ + feature: this.feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return true │ │ │ │ │ }, │ │ │ │ │ - up: function(evt) { │ │ │ │ │ - if (this.mouseDown && (!this.lastUp || !this.lastUp.equals(evt.xy))) { │ │ │ │ │ - if (this.stoppedDown && this.freehandMode(evt)) { │ │ │ │ │ - if (this.persist) { │ │ │ │ │ - this.destroyPersistedFeature() │ │ │ │ │ - } │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize() │ │ │ │ │ + collectVertices: function() { │ │ │ │ │ + this.vertices = []; │ │ │ │ │ + this.virtualVertices = []; │ │ │ │ │ + var control = this; │ │ │ │ │ + │ │ │ │ │ + function collectComponentVertices(geometry) { │ │ │ │ │ + var i, vertex, component, len; │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + vertex = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + vertex._sketch = true; │ │ │ │ │ + vertex.renderIntent = control.vertexRenderIntent; │ │ │ │ │ + control.vertices.push(vertex) │ │ │ │ │ } else { │ │ │ │ │ - if (this.passesTolerance(this.lastDown, evt.xy, this.pixelTolerance)) { │ │ │ │ │ - if (this.touch) { │ │ │ │ │ - this.modifyFeature(evt.xy) │ │ │ │ │ - } │ │ │ │ │ - if (this.lastUp == null && this.persist) { │ │ │ │ │ - this.destroyPersistedFeature() │ │ │ │ │ + var numVert = geometry.components.length; │ │ │ │ │ + if (geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ + numVert -= 1 │ │ │ │ │ + } │ │ │ │ │ + for (i = 0; i < numVert; ++i) { │ │ │ │ │ + component = geometry.components[i]; │ │ │ │ │ + if (component.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + vertex = new OpenLayers.Feature.Vector(component); │ │ │ │ │ + vertex._sketch = true; │ │ │ │ │ + vertex.renderIntent = control.vertexRenderIntent; │ │ │ │ │ + control.vertices.push(vertex) │ │ │ │ │ + } else { │ │ │ │ │ + collectComponentVertices(component) │ │ │ │ │ } │ │ │ │ │ - this.addPoint(evt.xy); │ │ │ │ │ - this.lastUp = evt.xy; │ │ │ │ │ - if (this.line.geometry.components.length === this.maxVertices + 1) { │ │ │ │ │ - this.finishGeometry() │ │ │ │ │ + } │ │ │ │ │ + if (control.createVertices && geometry.CLASS_NAME != "OpenLayers.Geometry.MultiPoint") { │ │ │ │ │ + for (i = 0, len = geometry.components.length; i < len - 1; ++i) { │ │ │ │ │ + var prevVertex = geometry.components[i]; │ │ │ │ │ + var nextVertex = geometry.components[i + 1]; │ │ │ │ │ + if (prevVertex.CLASS_NAME == "OpenLayers.Geometry.Point" && nextVertex.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ + var x = (prevVertex.x + nextVertex.x) / 2; │ │ │ │ │ + var y = (prevVertex.y + nextVertex.y) / 2; │ │ │ │ │ + var point = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Point(x, y), null, control.virtualStyle); │ │ │ │ │ + point.geometry.parent = geometry; │ │ │ │ │ + point._index = i + 1; │ │ │ │ │ + point._sketch = true; │ │ │ │ │ + control.virtualVertices.push(point) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - this.stoppedDown = this.stopDown; │ │ │ │ │ - this.mouseDown = false; │ │ │ │ │ - return !this.stopUp │ │ │ │ │ - }, │ │ │ │ │ - finishGeometry: function() { │ │ │ │ │ - var index = this.line.geometry.components.length - 1; │ │ │ │ │ - this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize() │ │ │ │ │ - }, │ │ │ │ │ - dblclick: function(evt) { │ │ │ │ │ - if (!this.freehandMode(evt)) { │ │ │ │ │ - this.finishGeometry() │ │ │ │ │ - } │ │ │ │ │ - return false │ │ │ │ │ + collectComponentVertices.call(this, this.feature.geometry); │ │ │ │ │ + this.layer.addFeatures(this.virtualVertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.layer.addFeatures(this.vertices, { │ │ │ │ │ + silent: true │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Path" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.Polygon = OpenLayers.Class(OpenLayers.Handler.Path, { │ │ │ │ │ - holeModifier: null, │ │ │ │ │ - drawingHole: false, │ │ │ │ │ - polygon: null, │ │ │ │ │ - createFeature: function(pixel) { │ │ │ │ │ - var lonlat = this.layer.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geometry = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat); │ │ │ │ │ - this.point = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - this.line = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.LinearRing([this.point.geometry])); │ │ │ │ │ - this.polygon = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Polygon([this.line.geometry])); │ │ │ │ │ - this.callback("create", [this.point.geometry, this.getSketch()]); │ │ │ │ │ - this.point.geometry.clearBounds(); │ │ │ │ │ - this.layer.addFeatures([this.polygon, this.point], { │ │ │ │ │ + collectDragHandle: function() { │ │ │ │ │ + var geometry = this.feature.geometry; │ │ │ │ │ + var center = geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + var originGeometry = new OpenLayers.Geometry.Point(center.lon, center.lat); │ │ │ │ │ + var origin = new OpenLayers.Feature.Vector(originGeometry); │ │ │ │ │ + originGeometry.move = function(x, y) { │ │ │ │ │ + OpenLayers.Geometry.Point.prototype.move.call(this, x, y); │ │ │ │ │ + geometry.move(x, y) │ │ │ │ │ + }; │ │ │ │ │ + origin._sketch = true; │ │ │ │ │ + this.dragHandle = origin; │ │ │ │ │ + this.dragHandle.renderIntent = this.vertexRenderIntent; │ │ │ │ │ + this.layer.addFeatures([this.dragHandle], { │ │ │ │ │ silent: true │ │ │ │ │ }) │ │ │ │ │ }, │ │ │ │ │ - addPoint: function(pixel) { │ │ │ │ │ - if (!this.drawingHole && this.holeModifier && this.evt && this.evt[this.holeModifier]) { │ │ │ │ │ - var geometry = this.point.geometry; │ │ │ │ │ - var features = this.control.layer.features; │ │ │ │ │ - var candidate, polygon; │ │ │ │ │ - for (var i = features.length - 1; i >= 0; --i) { │ │ │ │ │ - candidate = features[i].geometry; │ │ │ │ │ - if ((candidate instanceof OpenLayers.Geometry.Polygon || candidate instanceof OpenLayers.Geometry.MultiPolygon) && candidate.intersects(geometry)) { │ │ │ │ │ - polygon = features[i]; │ │ │ │ │ - this.control.layer.removeFeatures([polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.control.layer.events.registerPriority("sketchcomplete", this, this.finalizeInteriorRing); │ │ │ │ │ - this.control.layer.events.registerPriority("sketchmodified", this, this.enforceTopology); │ │ │ │ │ - polygon.geometry.addComponent(this.line.geometry); │ │ │ │ │ - this.polygon = polygon; │ │ │ │ │ - this.drawingHole = true; │ │ │ │ │ - break │ │ │ │ │ + collectRadiusHandle: function() { │ │ │ │ │ + var geometry = this.feature.geometry; │ │ │ │ │ + var bounds = geometry.getBounds(); │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var originGeometry = new OpenLayers.Geometry.Point(center.lon, center.lat); │ │ │ │ │ + var radiusGeometry = new OpenLayers.Geometry.Point(bounds.right, bounds.bottom); │ │ │ │ │ + var radius = new OpenLayers.Feature.Vector(radiusGeometry); │ │ │ │ │ + var resize = this.mode & OpenLayers.Control.ModifyFeature.RESIZE; │ │ │ │ │ + var reshape = this.mode & OpenLayers.Control.ModifyFeature.RESHAPE; │ │ │ │ │ + var rotate = this.mode & OpenLayers.Control.ModifyFeature.ROTATE; │ │ │ │ │ + radiusGeometry.move = function(x, y) { │ │ │ │ │ + OpenLayers.Geometry.Point.prototype.move.call(this, x, y); │ │ │ │ │ + var dx1 = this.x - originGeometry.x; │ │ │ │ │ + var dy1 = this.y - originGeometry.y; │ │ │ │ │ + var dx0 = dx1 - x; │ │ │ │ │ + var dy0 = dy1 - y; │ │ │ │ │ + if (rotate) { │ │ │ │ │ + var a0 = Math.atan2(dy0, dx0); │ │ │ │ │ + var a1 = Math.atan2(dy1, dx1); │ │ │ │ │ + var angle = a1 - a0; │ │ │ │ │ + angle *= 180 / Math.PI; │ │ │ │ │ + geometry.rotate(angle, originGeometry) │ │ │ │ │ + } │ │ │ │ │ + if (resize) { │ │ │ │ │ + var scale, ratio; │ │ │ │ │ + if (reshape) { │ │ │ │ │ + scale = dy1 / dy0; │ │ │ │ │ + ratio = dx1 / dx0 / scale │ │ │ │ │ + } else { │ │ │ │ │ + var l0 = Math.sqrt(dx0 * dx0 + dy0 * dy0); │ │ │ │ │ + var l1 = Math.sqrt(dx1 * dx1 + dy1 * dy1); │ │ │ │ │ + scale = l1 / l0 │ │ │ │ │ } │ │ │ │ │ + geometry.resize(scale, originGeometry, ratio) │ │ │ │ │ } │ │ │ │ │ + }; │ │ │ │ │ + radius._sketch = true; │ │ │ │ │ + this.radiusHandle = radius; │ │ │ │ │ + this.radiusHandle.renderIntent = this.vertexRenderIntent; │ │ │ │ │ + this.layer.addFeatures([this.radiusHandle], { │ │ │ │ │ + silent: true │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + this.handlers.drag.setMap(map); │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + handleMapEvents: function(evt) { │ │ │ │ │ + if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ + this.moveLayerToTop() │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Handler.Path.prototype.addPoint.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - getCurrentPointIndex: function() { │ │ │ │ │ - return this.line.geometry.components.length - 2 │ │ │ │ │ + moveLayerToTop: function() { │ │ │ │ │ + var index = Math.max(this.map.Z_INDEX_BASE["Feature"] - 1, this.layer.getZIndex()) + 1; │ │ │ │ │ + this.layer.setZIndex(index) │ │ │ │ │ }, │ │ │ │ │ - enforceTopology: function(event) { │ │ │ │ │ - var point = event.vertex; │ │ │ │ │ - var components = this.line.geometry.components; │ │ │ │ │ - if (!this.polygon.geometry.intersects(point)) { │ │ │ │ │ - var last = components[components.length - 3]; │ │ │ │ │ - point.x = last.x; │ │ │ │ │ - point.y = last.y │ │ │ │ │ + moveLayerBack: function() { │ │ │ │ │ + var index = this.layer.getZIndex() - 1; │ │ │ │ │ + if (index >= this.map.Z_INDEX_BASE["Feature"]) { │ │ │ │ │ + this.layer.setZIndex(index) │ │ │ │ │ + } else { │ │ │ │ │ + this.map.setLayerZIndex(this.layer, this.map.getLayerIndex(this.layer)) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - finishGeometry: function() { │ │ │ │ │ - var index = this.line.geometry.components.length - 2; │ │ │ │ │ - this.line.geometry.removeComponent(this.line.geometry.components[index]); │ │ │ │ │ - this.removePoint(); │ │ │ │ │ - this.finalize() │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ModifyFeature" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.ModifyFeature.RESHAPE = 1; │ │ │ │ │ +OpenLayers.Control.ModifyFeature.RESIZE = 2; │ │ │ │ │ +OpenLayers.Control.ModifyFeature.ROTATE = 4; │ │ │ │ │ +OpenLayers.Control.ModifyFeature.DRAG = 8; │ │ │ │ │ +OpenLayers.Control.ZoomIn = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ + trigger: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.zoomIn() │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - finalizeInteriorRing: function() { │ │ │ │ │ - var ring = this.line.geometry; │ │ │ │ │ - var modified = ring.getArea() !== 0; │ │ │ │ │ - if (modified) { │ │ │ │ │ - var rings = this.polygon.geometry.components; │ │ │ │ │ - for (var i = rings.length - 2; i >= 0; --i) { │ │ │ │ │ - if (ring.intersects(rings[i])) { │ │ │ │ │ - modified = false; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (modified) { │ │ │ │ │ - var target; │ │ │ │ │ - outer: for (var i = rings.length - 2; i > 0; --i) { │ │ │ │ │ - var points = rings[i].components; │ │ │ │ │ - for (var j = 0, jj = points.length; j < jj; ++j) { │ │ │ │ │ - if (ring.containsPoint(points[j])) { │ │ │ │ │ - modified = false; │ │ │ │ │ - break outer │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomIn" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Zoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + zoomInText: "+", │ │ │ │ │ + zoomInId: "olZoomInLink", │ │ │ │ │ + zoomOutText: "−", │ │ │ │ │ + zoomOutId: "olZoomOutLink", │ │ │ │ │ + draw: function() { │ │ │ │ │ + var div = OpenLayers.Control.prototype.draw.apply(this), │ │ │ │ │ + links = this.getOrCreateLinks(div), │ │ │ │ │ + zoomIn = links.zoomIn, │ │ │ │ │ + zoomOut = links.zoomOut, │ │ │ │ │ + eventsInstance = this.map.events; │ │ │ │ │ + if (zoomOut.parentNode !== div) { │ │ │ │ │ + eventsInstance = this.events; │ │ │ │ │ + eventsInstance.attachToElement(zoomOut.parentNode) │ │ │ │ │ } │ │ │ │ │ - if (modified) { │ │ │ │ │ - if (this.polygon.state !== OpenLayers.State.INSERT) { │ │ │ │ │ - this.polygon.state = OpenLayers.State.UPDATE │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.polygon.geometry.removeComponent(ring) │ │ │ │ │ + eventsInstance.register("buttonclick", this, this.onZoomClick); │ │ │ │ │ + this.zoomInLink = zoomIn; │ │ │ │ │ + this.zoomOutLink = zoomOut; │ │ │ │ │ + return div │ │ │ │ │ + }, │ │ │ │ │ + getOrCreateLinks: function(el) { │ │ │ │ │ + var zoomIn = document.getElementById(this.zoomInId), │ │ │ │ │ + zoomOut = document.getElementById(this.zoomOutId); │ │ │ │ │ + if (!zoomIn) { │ │ │ │ │ + zoomIn = document.createElement("a"); │ │ │ │ │ + zoomIn.href = "#zoomIn"; │ │ │ │ │ + zoomIn.appendChild(document.createTextNode(this.zoomInText)); │ │ │ │ │ + zoomIn.className = "olControlZoomIn"; │ │ │ │ │ + el.appendChild(zoomIn) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.addClass(zoomIn, "olButton"); │ │ │ │ │ + if (!zoomOut) { │ │ │ │ │ + zoomOut = document.createElement("a"); │ │ │ │ │ + zoomOut.href = "#zoomOut"; │ │ │ │ │ + zoomOut.appendChild(document.createTextNode(this.zoomOutText)); │ │ │ │ │ + zoomOut.className = "olControlZoomOut"; │ │ │ │ │ + el.appendChild(zoomOut) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.addClass(zoomOut, "olButton"); │ │ │ │ │ + return { │ │ │ │ │ + zoomIn: zoomIn, │ │ │ │ │ + zoomOut: zoomOut │ │ │ │ │ } │ │ │ │ │ - this.restoreFeature(); │ │ │ │ │ - return false │ │ │ │ │ }, │ │ │ │ │ - cancel: function() { │ │ │ │ │ - if (this.drawingHole) { │ │ │ │ │ - this.polygon.geometry.removeComponent(this.line.geometry); │ │ │ │ │ - this.restoreFeature(true) │ │ │ │ │ + onZoomClick: function(evt) { │ │ │ │ │ + var button = evt.buttonElement; │ │ │ │ │ + if (button === this.zoomInLink) { │ │ │ │ │ + this.map.zoomIn() │ │ │ │ │ + } else if (button === this.zoomOutLink) { │ │ │ │ │ + this.map.zoomOut() │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Handler.Path.prototype.cancel.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - restoreFeature: function(cancel) { │ │ │ │ │ - this.control.layer.events.unregister("sketchcomplete", this, this.finalizeInteriorRing); │ │ │ │ │ - this.control.layer.events.unregister("sketchmodified", this, this.enforceTopology); │ │ │ │ │ - this.layer.removeFeatures([this.polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onZoomClick) │ │ │ │ │ + } │ │ │ │ │ + delete this.zoomInLink; │ │ │ │ │ + delete this.zoomOutLink; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Zoom" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.DrawFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + layer: null, │ │ │ │ │ + callbacks: null, │ │ │ │ │ + multi: false, │ │ │ │ │ + featureAdded: function() {}, │ │ │ │ │ + initialize: function(layer, handler, options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.callbacks = OpenLayers.Util.extend({ │ │ │ │ │ + done: this.drawFeature, │ │ │ │ │ + modify: function(vertex, feature) { │ │ │ │ │ + this.layer.events.triggerEvent("sketchmodified", { │ │ │ │ │ + vertex: vertex, │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + create: function(vertex, feature) { │ │ │ │ │ + this.layer.events.triggerEvent("sketchstarted", { │ │ │ │ │ + vertex: vertex, │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, this.callbacks); │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + this.handlerOptions = this.handlerOptions || {}; │ │ │ │ │ + this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults(this.handlerOptions.layerOptions, { │ │ │ │ │ + renderers: layer.renderers, │ │ │ │ │ + rendererOptions: layer.rendererOptions │ │ │ │ │ }); │ │ │ │ │ - this.control.layer.addFeatures([this.polygon], { │ │ │ │ │ - silent: true │ │ │ │ │ + if (!("multi" in this.handlerOptions)) { │ │ │ │ │ + this.handlerOptions.multi = this.multi │ │ │ │ │ + } │ │ │ │ │ + var sketchStyle = this.layer.styleMap && this.layer.styleMap.styles.temporary; │ │ │ │ │ + if (sketchStyle) { │ │ │ │ │ + this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults(this.handlerOptions.layerOptions, { │ │ │ │ │ + styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ + default: sketchStyle │ │ │ │ │ + }) │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + this.handler = new handler(this, this.callbacks, this.handlerOptions) │ │ │ │ │ + }, │ │ │ │ │ + drawFeature: function(geometry) { │ │ │ │ │ + var feature = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ + var proceed = this.layer.events.triggerEvent("sketchcomplete", { │ │ │ │ │ + feature: feature │ │ │ │ │ }); │ │ │ │ │ - this.drawingHole = false; │ │ │ │ │ - if (!cancel) { │ │ │ │ │ - this.control.layer.events.triggerEvent("sketchcomplete", { │ │ │ │ │ - feature: this.polygon │ │ │ │ │ + if (proceed !== false) { │ │ │ │ │ + feature.state = OpenLayers.State.INSERT; │ │ │ │ │ + this.layer.addFeatures([feature]); │ │ │ │ │ + this.featureAdded(feature); │ │ │ │ │ + this.events.triggerEvent("featureadded", { │ │ │ │ │ + feature: feature │ │ │ │ │ }) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - destroyFeature: function(force) { │ │ │ │ │ - OpenLayers.Handler.Path.prototype.destroyFeature.call(this, force); │ │ │ │ │ - this.polygon = null │ │ │ │ │ + insertXY: function(x, y) { │ │ │ │ │ + if (this.handler && this.handler.line) { │ │ │ │ │ + this.handler.insertXY(x, y) │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - drawFeature: function() { │ │ │ │ │ - this.layer.drawFeature(this.polygon, this.style); │ │ │ │ │ - this.layer.drawFeature(this.point, this.style) │ │ │ │ │ + insertDeltaXY: function(dx, dy) { │ │ │ │ │ + if (this.handler && this.handler.line) { │ │ │ │ │ + this.handler.insertDeltaXY(dx, dy) │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - getSketch: function() { │ │ │ │ │ - return this.polygon │ │ │ │ │ + insertDirectionLength: function(direction, length) { │ │ │ │ │ + if (this.handler && this.handler.line) { │ │ │ │ │ + this.handler.insertDirectionLength(direction, length) │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - getGeometry: function() { │ │ │ │ │ - var geometry = this.polygon && this.polygon.geometry; │ │ │ │ │ - if (geometry && this.multi) { │ │ │ │ │ - geometry = new OpenLayers.Geometry.MultiPolygon([geometry]) │ │ │ │ │ + insertDeflectionLength: function(deflection, length) { │ │ │ │ │ + if (this.handler && this.handler.line) { │ │ │ │ │ + this.handler.insertDeflectionLength(deflection, length) │ │ │ │ │ } │ │ │ │ │ - return geometry │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Polygon" │ │ │ │ │ + undo: function() { │ │ │ │ │ + return this.handler.undo && this.handler.undo() │ │ │ │ │ + }, │ │ │ │ │ + redo: function() { │ │ │ │ │ + return this.handler.redo && this.handler.redo() │ │ │ │ │ + }, │ │ │ │ │ + finishSketch: function() { │ │ │ │ │ + this.handler.finishGeometry() │ │ │ │ │ + }, │ │ │ │ │ + cancel: function() { │ │ │ │ │ + this.handler.cancel() │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.DrawFeature" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Handler.MouseWheel = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - wheelListener: null, │ │ │ │ │ - interval: 0, │ │ │ │ │ - maxDelta: Number.POSITIVE_INFINITY, │ │ │ │ │ - delta: 0, │ │ │ │ │ - cumulative: true, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.wheelListener = OpenLayers.Function.bindAsEventListener(this.onWheelEvent, this) │ │ │ │ │ +OpenLayers.Control.Permalink = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + argParserClass: OpenLayers.Control.ArgParser, │ │ │ │ │ + element: null, │ │ │ │ │ + anchor: false, │ │ │ │ │ + base: "", │ │ │ │ │ + displayProjection: null, │ │ │ │ │ + initialize: function(element, base, options) { │ │ │ │ │ + if (element !== null && typeof element == "object" && !OpenLayers.Util.isElement(element)) { │ │ │ │ │ + options = element; │ │ │ │ │ + this.base = document.location.href; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + if (this.element != null) { │ │ │ │ │ + this.element = OpenLayers.Util.getElement(this.element) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.element = OpenLayers.Util.getElement(element); │ │ │ │ │ + this.base = base || document.location.href │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ - this.wheelListener = null │ │ │ │ │ - }, │ │ │ │ │ - onWheelEvent: function(e) { │ │ │ │ │ - if (!this.map || !this.checkModifiers(e)) { │ │ │ │ │ - return │ │ │ │ │ + if (this.element && this.element.parentNode == this.div) { │ │ │ │ │ + this.div.removeChild(this.element); │ │ │ │ │ + this.element = null │ │ │ │ │ } │ │ │ │ │ - var overScrollableDiv = false; │ │ │ │ │ - var allowScroll = false; │ │ │ │ │ - var overMapDiv = false; │ │ │ │ │ - var elem = OpenLayers.Event.element(e); │ │ │ │ │ - while (elem != null && !overMapDiv && !overScrollableDiv) { │ │ │ │ │ - if (!overScrollableDiv) { │ │ │ │ │ - try { │ │ │ │ │ - var overflow; │ │ │ │ │ - if (elem.currentStyle) { │ │ │ │ │ - overflow = elem.currentStyle["overflow"] │ │ │ │ │ - } else { │ │ │ │ │ - var style = document.defaultView.getComputedStyle(elem, null); │ │ │ │ │ - overflow = style.getPropertyValue("overflow") │ │ │ │ │ - } │ │ │ │ │ - overScrollableDiv = overflow && overflow == "auto" || overflow == "scroll" │ │ │ │ │ - } catch (err) {} │ │ │ │ │ - } │ │ │ │ │ - if (!allowScroll) { │ │ │ │ │ - allowScroll = OpenLayers.Element.hasClass(elem, "olScrollable"); │ │ │ │ │ - if (!allowScroll) { │ │ │ │ │ - for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if (elem == layer.div || elem == layer.pane) { │ │ │ │ │ - allowScroll = true; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - overMapDiv = elem == this.map.div; │ │ │ │ │ - elem = elem.parentNode │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("moveend", this, this.updateLink) │ │ │ │ │ } │ │ │ │ │ - if (!overScrollableDiv && overMapDiv) { │ │ │ │ │ - if (allowScroll) { │ │ │ │ │ - var delta = 0; │ │ │ │ │ - if (e.wheelDelta) { │ │ │ │ │ - delta = e.wheelDelta; │ │ │ │ │ - if (delta % 160 === 0) { │ │ │ │ │ - delta = delta * .75 │ │ │ │ │ - } │ │ │ │ │ - delta = delta / 120 │ │ │ │ │ - } else if (e.detail) { │ │ │ │ │ - delta = -(e.detail / Math.abs(e.detail)) │ │ │ │ │ - } │ │ │ │ │ - this.delta += delta; │ │ │ │ │ - window.clearTimeout(this._timeoutId); │ │ │ │ │ - if (this.interval && Math.abs(this.delta) < this.maxDelta) { │ │ │ │ │ - var evt = OpenLayers.Util.extend({}, e); │ │ │ │ │ - this._timeoutId = window.setTimeout(OpenLayers.Function.bind(function() { │ │ │ │ │ - this.wheelZoom(evt) │ │ │ │ │ - }, this), this.interval) │ │ │ │ │ - } else { │ │ │ │ │ - this.wheelZoom(e) │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + for (var i = 0, len = this.map.controls.length; i < len; i++) { │ │ │ │ │ + var control = this.map.controls[i]; │ │ │ │ │ + if (control.CLASS_NAME == this.argParserClass.CLASS_NAME) { │ │ │ │ │ + if (control.displayProjection != this.displayProjection) { │ │ │ │ │ + this.displayProjection = control.displayProjection │ │ │ │ │ } │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Event.stop(e) │ │ │ │ │ + } │ │ │ │ │ + if (i == this.map.controls.length) { │ │ │ │ │ + this.map.addControl(new this.argParserClass({ │ │ │ │ │ + displayProjection: this.displayProjection │ │ │ │ │ + })) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - wheelZoom: function(e) { │ │ │ │ │ - var delta = this.delta; │ │ │ │ │ - this.delta = 0; │ │ │ │ │ - if (delta) { │ │ │ │ │ - e.xy = this.map.events.getMousePosition(e); │ │ │ │ │ - if (delta < 0) { │ │ │ │ │ - this.callback("down", [e, this.cumulative ? Math.max(-this.maxDelta, delta) : -1]) │ │ │ │ │ - } else { │ │ │ │ │ - this.callback("up", [e, this.cumulative ? Math.min(this.maxDelta, delta) : 1]) │ │ │ │ │ - } │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (!this.element && !this.anchor) { │ │ │ │ │ + this.element = document.createElement("a"); │ │ │ │ │ + this.element.innerHTML = OpenLayers.i18n("Permalink"); │ │ │ │ │ + this.element.href = ""; │ │ │ │ │ + this.div.appendChild(this.element) │ │ │ │ │ } │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + moveend: this.updateLink, │ │ │ │ │ + changelayer: this.updateLink, │ │ │ │ │ + changebaselayer: this.updateLink, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.updateLink(); │ │ │ │ │ + return this.div │ │ │ │ │ }, │ │ │ │ │ - activate: function(evt) { │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - var wheelListener = this.wheelListener; │ │ │ │ │ - OpenLayers.Event.observe(window, "DOMMouseScroll", wheelListener); │ │ │ │ │ - OpenLayers.Event.observe(window, "mousewheel", wheelListener); │ │ │ │ │ - OpenLayers.Event.observe(document, "mousewheel", wheelListener); │ │ │ │ │ - return true │ │ │ │ │ + updateLink: function() { │ │ │ │ │ + var separator = this.anchor ? "#" : "?"; │ │ │ │ │ + var href = this.base; │ │ │ │ │ + var anchor = null; │ │ │ │ │ + if (href.indexOf("#") != -1 && this.anchor == false) { │ │ │ │ │ + anchor = href.substring(href.indexOf("#"), href.length) │ │ │ │ │ + } │ │ │ │ │ + if (href.indexOf(separator) != -1) { │ │ │ │ │ + href = href.substring(0, href.indexOf(separator)) │ │ │ │ │ + } │ │ │ │ │ + var splits = href.split("#"); │ │ │ │ │ + href = splits[0] + separator + OpenLayers.Util.getParameterString(this.createParams()); │ │ │ │ │ + if (anchor) { │ │ │ │ │ + href += anchor │ │ │ │ │ + } │ │ │ │ │ + if (this.anchor && !this.element) { │ │ │ │ │ + window.location.href = href │ │ │ │ │ } else { │ │ │ │ │ - return false │ │ │ │ │ + this.element.href = href │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - deactivate: function(evt) { │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - var wheelListener = this.wheelListener; │ │ │ │ │ - OpenLayers.Event.stopObserving(window, "DOMMouseScroll", wheelListener); │ │ │ │ │ - OpenLayers.Event.stopObserving(window, "mousewheel", wheelListener); │ │ │ │ │ - OpenLayers.Event.stopObserving(document, "mousewheel", wheelListener); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + createParams: function(center, zoom, layers) { │ │ │ │ │ + center = center || this.map.getCenter(); │ │ │ │ │ + var params = OpenLayers.Util.getParameters(this.base); │ │ │ │ │ + if (center) { │ │ │ │ │ + params.zoom = zoom || this.map.getZoom(); │ │ │ │ │ + var lat = center.lat; │ │ │ │ │ + var lon = center.lon; │ │ │ │ │ + if (this.displayProjection) { │ │ │ │ │ + var mapPosition = OpenLayers.Projection.transform({ │ │ │ │ │ + x: lon, │ │ │ │ │ + y: lat │ │ │ │ │ + }, this.map.getProjectionObject(), this.displayProjection); │ │ │ │ │ + lon = mapPosition.x; │ │ │ │ │ + lat = mapPosition.y │ │ │ │ │ + } │ │ │ │ │ + params.lat = Math.round(lat * 1e5) / 1e5; │ │ │ │ │ + params.lon = Math.round(lon * 1e5) / 1e5; │ │ │ │ │ + layers = layers || this.map.layers; │ │ │ │ │ + params.layers = ""; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + var layer = layers[i]; │ │ │ │ │ + if (layer.isBaseLayer) { │ │ │ │ │ + params.layers += layer == this.map.baseLayer ? "B" : "0" │ │ │ │ │ + } else { │ │ │ │ │ + params.layers += layer.getVisibility() ? "T" : "F" │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return params │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.MouseWheel" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Permalink" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Handler.Keyboard = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - KEY_EVENTS: ["keydown", "keyup"], │ │ │ │ │ - eventListener: null, │ │ │ │ │ - observeElement: null, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.eventListener = OpenLayers.Function.bindAsEventListener(this.handleKeyEvent, this) │ │ │ │ │ +OpenLayers.Control.ZoomToMaxExtent = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ + trigger: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.zoomToMaxExtent() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomToMaxExtent" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Panel = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + controls: null, │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + defaultControl: null, │ │ │ │ │ + saveState: false, │ │ │ │ │ + allowDepress: false, │ │ │ │ │ + activeState: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.controls = []; │ │ │ │ │ + this.activeState = {} │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.eventListener = null; │ │ │ │ │ - OpenLayers.Handler.prototype.destroy.apply(this, arguments) │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.unregister("buttonclick", this, this.onButtonClick) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ + for (var ctl, i = this.controls.length - 1; i >= 0; i--) { │ │ │ │ │ + ctl = this.controls[i]; │ │ │ │ │ + if (ctl.events) { │ │ │ │ │ + ctl.events.un({ │ │ │ │ │ + activate: this.iconOn, │ │ │ │ │ + deactivate: this.iconOff │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + ctl.panel_div = null │ │ │ │ │ + } │ │ │ │ │ + this.activeState = null │ │ │ │ │ }, │ │ │ │ │ activate: function() { │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.observeElement = this.observeElement || document; │ │ │ │ │ - for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ - OpenLayers.Event.observe(this.observeElement, this.KEY_EVENTS[i], this.eventListener) │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + control = this.controls[i]; │ │ │ │ │ + if (control === this.defaultControl || this.saveState && this.activeState[control.id]) { │ │ │ │ │ + control.activate() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (this.saveState === true) { │ │ │ │ │ + this.defaultControl = null │ │ │ │ │ } │ │ │ │ │ + this.redraw(); │ │ │ │ │ return true │ │ │ │ │ } else { │ │ │ │ │ return false │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - for (var i = 0, len = this.KEY_EVENTS.length; i < len; i++) { │ │ │ │ │ - OpenLayers.Event.stopObserving(this.observeElement, this.KEY_EVENTS[i], this.eventListener) │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + control = this.controls[i]; │ │ │ │ │ + this.activeState[control.id] = control.deactivate() │ │ │ │ │ } │ │ │ │ │ - deactivated = true │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - handleKeyEvent: function(evt) { │ │ │ │ │ - if (this.checkModifiers(evt)) { │ │ │ │ │ - this.callback(evt.type, [evt]) │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Keyboard" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.Hover = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - delay: 500, │ │ │ │ │ - pixelTolerance: null, │ │ │ │ │ - stopMove: false, │ │ │ │ │ - px: null, │ │ │ │ │ - timerId: null, │ │ │ │ │ - mousemove: function(evt) { │ │ │ │ │ - if (this.passesTolerance(evt.xy)) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.callback("move", [evt]); │ │ │ │ │ - this.px = evt.xy; │ │ │ │ │ - evt = OpenLayers.Util.extend({}, evt); │ │ │ │ │ - this.timerId = window.setTimeout(OpenLayers.Function.bind(this.delayedCall, this, evt), this.delay) │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (this.outsideViewport) { │ │ │ │ │ + this.events.attachToElement(this.div); │ │ │ │ │ + this.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ + } else { │ │ │ │ │ + this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ } │ │ │ │ │ - return !this.stopMove │ │ │ │ │ + this.addControlsToMap(this.controls); │ │ │ │ │ + return this.div │ │ │ │ │ }, │ │ │ │ │ - mouseout: function(evt) { │ │ │ │ │ - if (OpenLayers.Util.mouseLeft(evt, this.map.viewPortDiv)) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - this.callback("move", [evt]) │ │ │ │ │ + redraw: function() { │ │ │ │ │ + for (var l = this.div.childNodes.length, i = l - 1; i >= 0; i--) { │ │ │ │ │ + this.div.removeChild(this.div.childNodes[i]) │ │ │ │ │ } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - passesTolerance: function(px) { │ │ │ │ │ - var passes = true; │ │ │ │ │ - if (this.pixelTolerance && this.px) { │ │ │ │ │ - var dpx = Math.sqrt(Math.pow(this.px.x - px.x, 2) + Math.pow(this.px.y - px.y, 2)); │ │ │ │ │ - if (dpx < this.pixelTolerance) { │ │ │ │ │ - passes = false │ │ │ │ │ + this.div.innerHTML = ""; │ │ │ │ │ + if (this.active) { │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + this.div.appendChild(this.controls[i].panel_div) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return passes │ │ │ │ │ }, │ │ │ │ │ - clearTimer: function() { │ │ │ │ │ - if (this.timerId != null) { │ │ │ │ │ - window.clearTimeout(this.timerId); │ │ │ │ │ - this.timerId = null │ │ │ │ │ + activateControl: function(control) { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - delayedCall: function(evt) { │ │ │ │ │ - this.callback("pause", [evt]) │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.clearTimer(); │ │ │ │ │ - deactivated = true │ │ │ │ │ + if (control.type == OpenLayers.Control.TYPE_BUTTON) { │ │ │ │ │ + control.trigger(); │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Hover" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.Pinch = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - started: false, │ │ │ │ │ - stopDown: false, │ │ │ │ │ - pinching: false, │ │ │ │ │ - last: null, │ │ │ │ │ - start: null, │ │ │ │ │ - touchstart: function(evt) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - if (OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ - this.started = true; │ │ │ │ │ - this.last = this.start = { │ │ │ │ │ - distance: this.getDistance(evt.touches), │ │ │ │ │ - delta: 0, │ │ │ │ │ - scale: 1 │ │ │ │ │ - }; │ │ │ │ │ - this.callback("start", [evt, this.start]); │ │ │ │ │ - propagate = !this.stopDown │ │ │ │ │ - } else if (this.started) { │ │ │ │ │ - return false │ │ │ │ │ - } else { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null │ │ │ │ │ + if (control.type == OpenLayers.Control.TYPE_TOGGLE) { │ │ │ │ │ + if (control.active) { │ │ │ │ │ + control.deactivate() │ │ │ │ │ + } else { │ │ │ │ │ + control.activate() │ │ │ │ │ + } │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Event.preventDefault(evt); │ │ │ │ │ - return propagate │ │ │ │ │ - }, │ │ │ │ │ - touchmove: function(evt) { │ │ │ │ │ - if (this.started && OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ - this.pinching = true; │ │ │ │ │ - var current = this.getPinchData(evt); │ │ │ │ │ - this.callback("move", [evt, current]); │ │ │ │ │ - this.last = current; │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ - } else if (this.started) { │ │ │ │ │ - return false │ │ │ │ │ + if (this.allowDepress && control.active) { │ │ │ │ │ + control.deactivate() │ │ │ │ │ + } else { │ │ │ │ │ + var c; │ │ │ │ │ + for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ + c = this.controls[i]; │ │ │ │ │ + if (c != control && (c.type === OpenLayers.Control.TYPE_TOOL || c.type == null)) { │ │ │ │ │ + c.deactivate() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + control.activate() │ │ │ │ │ } │ │ │ │ │ - return true │ │ │ │ │ }, │ │ │ │ │ - touchend: function(evt) { │ │ │ │ │ - if (this.started && !OpenLayers.Event.isMultiTouch(evt)) { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - this.callback("done", [evt, this.start, this.last]); │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - return false │ │ │ │ │ + addControls: function(controls) { │ │ │ │ │ + if (!OpenLayers.Util.isArray(controls)) { │ │ │ │ │ + controls = [controls] │ │ │ │ │ } │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - activated = true │ │ │ │ │ + this.controls = this.controls.concat(controls); │ │ │ │ │ + for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ + var control = controls[i], │ │ │ │ │ + element = this.createControlMarkup(control); │ │ │ │ │ + OpenLayers.Element.addClass(element, control.displayClass + "ItemInactive"); │ │ │ │ │ + OpenLayers.Element.addClass(element, "olButton"); │ │ │ │ │ + if (control.title != "" && !element.title) { │ │ │ │ │ + element.title = control.title │ │ │ │ │ + } │ │ │ │ │ + control.panel_div = element │ │ │ │ │ } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.started = false; │ │ │ │ │ - this.pinching = false; │ │ │ │ │ - this.start = null; │ │ │ │ │ - this.last = null; │ │ │ │ │ - deactivated = true │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.addControlsToMap(controls); │ │ │ │ │ + this.redraw() │ │ │ │ │ } │ │ │ │ │ - return deactivated │ │ │ │ │ }, │ │ │ │ │ - getDistance: function(touches) { │ │ │ │ │ - var t0 = touches[0]; │ │ │ │ │ - var t1 = touches[1]; │ │ │ │ │ - return Math.sqrt(Math.pow(t0.olClientX - t1.olClientX, 2) + Math.pow(t0.olClientY - t1.olClientY, 2)) │ │ │ │ │ + createControlMarkup: function(control) { │ │ │ │ │ + return document.createElement("div") │ │ │ │ │ }, │ │ │ │ │ - getPinchData: function(evt) { │ │ │ │ │ - var distance = this.getDistance(evt.touches); │ │ │ │ │ - var scale = distance / this.start.distance; │ │ │ │ │ - return { │ │ │ │ │ - distance: distance, │ │ │ │ │ - delta: this.last.distance - distance, │ │ │ │ │ - scale: scale │ │ │ │ │ + addControlsToMap: function(controls) { │ │ │ │ │ + var control; │ │ │ │ │ + for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ + control = controls[i]; │ │ │ │ │ + if (control.autoActivate === true) { │ │ │ │ │ + control.autoActivate = false; │ │ │ │ │ + this.map.addControl(control); │ │ │ │ │ + control.autoActivate = true │ │ │ │ │ + } else { │ │ │ │ │ + this.map.addControl(control); │ │ │ │ │ + control.deactivate() │ │ │ │ │ + } │ │ │ │ │ + control.events.on({ │ │ │ │ │ + activate: this.iconOn, │ │ │ │ │ + deactivate: this.iconOff │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Pinch" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Handler.Box = OpenLayers.Class(OpenLayers.Handler, { │ │ │ │ │ - dragHandler: null, │ │ │ │ │ - boxDivClassName: "olHandlerBoxZoomBox", │ │ │ │ │ - boxOffsets: null, │ │ │ │ │ - initialize: function(control, callbacks, options) { │ │ │ │ │ - OpenLayers.Handler.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.dragHandler = new OpenLayers.Handler.Drag(this, { │ │ │ │ │ - down: this.startBox, │ │ │ │ │ - move: this.moveBox, │ │ │ │ │ - out: this.removeBox, │ │ │ │ │ - up: this.endBox │ │ │ │ │ - }, { │ │ │ │ │ - keyMask: this.keyMask │ │ │ │ │ - }) │ │ │ │ │ + iconOn: function() { │ │ │ │ │ + var d = this.panel_div; │ │ │ │ │ + var re = new RegExp("\\b(" + this.displayClass + "Item)Inactive\\b"); │ │ │ │ │ + d.className = d.className.replace(re, "$1Active") │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - OpenLayers.Handler.prototype.destroy.apply(this, arguments); │ │ │ │ │ - if (this.dragHandler) { │ │ │ │ │ - this.dragHandler.destroy(); │ │ │ │ │ - this.dragHandler = null │ │ │ │ │ - } │ │ │ │ │ + iconOff: function() { │ │ │ │ │ + var d = this.panel_div; │ │ │ │ │ + var re = new RegExp("\\b(" + this.displayClass + "Item)Active\\b"); │ │ │ │ │ + d.className = d.className.replace(re, "$1Inactive") │ │ │ │ │ }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Handler.prototype.setMap.apply(this, arguments); │ │ │ │ │ - if (this.dragHandler) { │ │ │ │ │ - this.dragHandler.setMap(map) │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + var controls = this.controls, │ │ │ │ │ + button = evt.buttonElement; │ │ │ │ │ + for (var i = controls.length - 1; i >= 0; --i) { │ │ │ │ │ + if (controls[i].panel_div === button) { │ │ │ │ │ + this.activateControl(controls[i]); │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - startBox: function(xy) { │ │ │ │ │ - this.callback("start", []); │ │ │ │ │ - this.zoomBox = OpenLayers.Util.createDiv("zoomBox", { │ │ │ │ │ - x: -9999, │ │ │ │ │ - y: -9999 │ │ │ │ │ + getControlsBy: function(property, match) { │ │ │ │ │ + var test = typeof match.test == "function"; │ │ │ │ │ + var found = OpenLayers.Array.filter(this.controls, function(item) { │ │ │ │ │ + return item[property] == match || test && match.test(item[property]) │ │ │ │ │ }); │ │ │ │ │ - this.zoomBox.className = this.boxDivClassName; │ │ │ │ │ - this.zoomBox.style.zIndex = this.map.Z_INDEX_BASE["Popup"] - 1; │ │ │ │ │ - this.map.viewPortDiv.appendChild(this.zoomBox); │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, "olDrawBox") │ │ │ │ │ + return found │ │ │ │ │ }, │ │ │ │ │ - moveBox: function(xy) { │ │ │ │ │ - var startX = this.dragHandler.start.x; │ │ │ │ │ - var startY = this.dragHandler.start.y; │ │ │ │ │ - var deltaX = Math.abs(startX - xy.x); │ │ │ │ │ - var deltaY = Math.abs(startY - xy.y); │ │ │ │ │ - var offset = this.getBoxOffsets(); │ │ │ │ │ - this.zoomBox.style.width = deltaX + offset.width + 1 + "px"; │ │ │ │ │ - this.zoomBox.style.height = deltaY + offset.height + 1 + "px"; │ │ │ │ │ - this.zoomBox.style.left = (xy.x < startX ? startX - deltaX - offset.left : startX - offset.left) + "px"; │ │ │ │ │ - this.zoomBox.style.top = (xy.y < startY ? startY - deltaY - offset.top : startY - offset.top) + "px" │ │ │ │ │ + getControlsByName: function(match) { │ │ │ │ │ + return this.getControlsBy("name", match) │ │ │ │ │ }, │ │ │ │ │ - endBox: function(end) { │ │ │ │ │ - var result; │ │ │ │ │ - if (Math.abs(this.dragHandler.start.x - end.x) > 5 || Math.abs(this.dragHandler.start.y - end.y) > 5) { │ │ │ │ │ - var start = this.dragHandler.start; │ │ │ │ │ - var top = Math.min(start.y, end.y); │ │ │ │ │ - var bottom = Math.max(start.y, end.y); │ │ │ │ │ - var left = Math.min(start.x, end.x); │ │ │ │ │ - var right = Math.max(start.x, end.x); │ │ │ │ │ - result = new OpenLayers.Bounds(left, bottom, right, top) │ │ │ │ │ - } else { │ │ │ │ │ - result = this.dragHandler.start.clone() │ │ │ │ │ + getControlsByClass: function(match) { │ │ │ │ │ + return this.getControlsBy("CLASS_NAME", match) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Panel" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Pan = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ + slideFactor: 50, │ │ │ │ │ + slideRatio: null, │ │ │ │ │ + direction: null, │ │ │ │ │ + initialize: function(direction, options) { │ │ │ │ │ + this.direction = direction; │ │ │ │ │ + this.CLASS_NAME += this.direction; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]) │ │ │ │ │ + }, │ │ │ │ │ + trigger: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + var getSlideFactor = OpenLayers.Function.bind(function(dim) { │ │ │ │ │ + return this.slideRatio ? this.map.getSize()[dim] * this.slideRatio : this.slideFactor │ │ │ │ │ + }, this); │ │ │ │ │ + switch (this.direction) { │ │ │ │ │ + case OpenLayers.Control.Pan.NORTH: │ │ │ │ │ + this.map.pan(0, -getSlideFactor("h")); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Control.Pan.SOUTH: │ │ │ │ │ + this.map.pan(0, getSlideFactor("h")); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Control.Pan.WEST: │ │ │ │ │ + this.map.pan(-getSlideFactor("w"), 0); │ │ │ │ │ + break; │ │ │ │ │ + case OpenLayers.Control.Pan.EAST: │ │ │ │ │ + this.map.pan(getSlideFactor("w"), 0); │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.removeBox(); │ │ │ │ │ - this.callback("done", [result]) │ │ │ │ │ }, │ │ │ │ │ - removeBox: function() { │ │ │ │ │ - this.map.viewPortDiv.removeChild(this.zoomBox); │ │ │ │ │ - this.zoomBox = null; │ │ │ │ │ - this.boxOffsets = null; │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olDrawBox") │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Pan" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Pan.NORTH = "North"; │ │ │ │ │ +OpenLayers.Control.Pan.SOUTH = "South"; │ │ │ │ │ +OpenLayers.Control.Pan.EAST = "East"; │ │ │ │ │ +OpenLayers.Control.Pan.WEST = "West"; │ │ │ │ │ +OpenLayers.Control.Snapping = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + DEFAULTS: { │ │ │ │ │ + tolerance: 10, │ │ │ │ │ + node: true, │ │ │ │ │ + edge: true, │ │ │ │ │ + vertex: true │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Handler.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.dragHandler.activate(); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + greedy: true, │ │ │ │ │ + precedence: ["node", "vertex", "edge"], │ │ │ │ │ + resolution: null, │ │ │ │ │ + geoToleranceCache: null, │ │ │ │ │ + layer: null, │ │ │ │ │ + feature: null, │ │ │ │ │ + point: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.options = options || {}; │ │ │ │ │ + if (this.options.layer) { │ │ │ │ │ + this.setLayer(this.options.layer) │ │ │ │ │ } │ │ │ │ │ + var defaults = OpenLayers.Util.extend({}, this.options.defaults); │ │ │ │ │ + this.defaults = OpenLayers.Util.applyDefaults(defaults, this.DEFAULTS); │ │ │ │ │ + this.setTargets(this.options.targets); │ │ │ │ │ + if (this.targets.length === 0 && this.layer) { │ │ │ │ │ + this.addTargetLayer(this.layer) │ │ │ │ │ + } │ │ │ │ │ + this.geoToleranceCache = {} │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Handler.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - if (this.dragHandler.deactivate()) { │ │ │ │ │ - if (this.zoomBox) { │ │ │ │ │ - this.removeBox() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return true │ │ │ │ │ + setLayer: function(layer) { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + this.layer = layer; │ │ │ │ │ + this.activate() │ │ │ │ │ } else { │ │ │ │ │ - return false │ │ │ │ │ + this.layer = layer │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - getBoxOffsets: function() { │ │ │ │ │ - if (!this.boxOffsets) { │ │ │ │ │ - var testDiv = document.createElement("div"); │ │ │ │ │ - testDiv.style.position = "absolute"; │ │ │ │ │ - testDiv.style.border = "1px solid black"; │ │ │ │ │ - testDiv.style.width = "3px"; │ │ │ │ │ - document.body.appendChild(testDiv); │ │ │ │ │ - var w3cBoxModel = testDiv.clientWidth == 3; │ │ │ │ │ - document.body.removeChild(testDiv); │ │ │ │ │ - var left = parseInt(OpenLayers.Element.getStyle(this.zoomBox, "border-left-width")); │ │ │ │ │ - var right = parseInt(OpenLayers.Element.getStyle(this.zoomBox, "border-right-width")); │ │ │ │ │ - var top = parseInt(OpenLayers.Element.getStyle(this.zoomBox, "border-top-width")); │ │ │ │ │ - var bottom = parseInt(OpenLayers.Element.getStyle(this.zoomBox, "border-bottom-width")); │ │ │ │ │ - this.boxOffsets = { │ │ │ │ │ - left: left, │ │ │ │ │ - right: right, │ │ │ │ │ - top: top, │ │ │ │ │ - bottom: bottom, │ │ │ │ │ - width: w3cBoxModel === false ? left + right : 0, │ │ │ │ │ - height: w3cBoxModel === false ? top + bottom : 0 │ │ │ │ │ + setTargets: function(targets) { │ │ │ │ │ + this.targets = []; │ │ │ │ │ + if (targets && targets.length) { │ │ │ │ │ + var target; │ │ │ │ │ + for (var i = 0, len = targets.length; i < len; ++i) { │ │ │ │ │ + target = targets[i]; │ │ │ │ │ + if (target instanceof OpenLayers.Layer.Vector) { │ │ │ │ │ + this.addTargetLayer(target) │ │ │ │ │ + } else { │ │ │ │ │ + this.addTarget(target) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return this.boxOffsets │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Handler.Box" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Marker.Box = OpenLayers.Class(OpenLayers.Marker, { │ │ │ │ │ - bounds: null, │ │ │ │ │ - div: null, │ │ │ │ │ - initialize: function(bounds, borderColor, borderWidth) { │ │ │ │ │ - this.bounds = bounds; │ │ │ │ │ - this.div = OpenLayers.Util.createDiv(); │ │ │ │ │ - this.div.style.overflow = "hidden"; │ │ │ │ │ - this.events = new OpenLayers.Events(this, this.div); │ │ │ │ │ - this.setBorder(borderColor, borderWidth) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.bounds = null; │ │ │ │ │ - this.div = null; │ │ │ │ │ - OpenLayers.Marker.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - setBorder: function(color, width) { │ │ │ │ │ - if (!color) { │ │ │ │ │ - color = "red" │ │ │ │ │ - } │ │ │ │ │ - if (!width) { │ │ │ │ │ - width = 2 │ │ │ │ │ - } │ │ │ │ │ - this.div.style.border = width + "px solid " + color │ │ │ │ │ + addTargetLayer: function(layer) { │ │ │ │ │ + this.addTarget({ │ │ │ │ │ + layer: layer │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - draw: function(px, sz) { │ │ │ │ │ - OpenLayers.Util.modifyDOMElement(this.div, null, px, sz); │ │ │ │ │ - return this.div │ │ │ │ │ + addTarget: function(target) { │ │ │ │ │ + target = OpenLayers.Util.applyDefaults(target, this.defaults); │ │ │ │ │ + target.nodeTolerance = target.nodeTolerance || target.tolerance; │ │ │ │ │ + target.vertexTolerance = target.vertexTolerance || target.tolerance; │ │ │ │ │ + target.edgeTolerance = target.edgeTolerance || target.tolerance; │ │ │ │ │ + this.targets.push(target) │ │ │ │ │ }, │ │ │ │ │ - onScreen: function() { │ │ │ │ │ - var onScreen = false; │ │ │ │ │ - if (this.map) { │ │ │ │ │ - var screenBounds = this.map.getExtent(); │ │ │ │ │ - onScreen = screenBounds.containsBounds(this.bounds, true, true) │ │ │ │ │ + removeTargetLayer: function(layer) { │ │ │ │ │ + var target; │ │ │ │ │ + for (var i = this.targets.length - 1; i >= 0; --i) { │ │ │ │ │ + target = this.targets[i]; │ │ │ │ │ + if (target.layer === layer) { │ │ │ │ │ + this.removeTarget(target) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return onScreen │ │ │ │ │ }, │ │ │ │ │ - display: function(display) { │ │ │ │ │ - this.div.style.display = display ? "" : "none" │ │ │ │ │ + removeTarget: function(target) { │ │ │ │ │ + return OpenLayers.Util.removeItem(this.targets, target) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Marker.Box" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Strategy.Cluster = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - distance: 20, │ │ │ │ │ - threshold: null, │ │ │ │ │ - features: null, │ │ │ │ │ - clusters: null, │ │ │ │ │ - clustering: false, │ │ │ │ │ - resolution: null, │ │ │ │ │ activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + var activated = OpenLayers.Control.prototype.activate.call(this); │ │ │ │ │ if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - beforefeaturesadded: this.cacheFeatures, │ │ │ │ │ - featuresremoved: this.clearCache, │ │ │ │ │ - moveend: this.cluster, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ + if (this.layer && this.layer.events) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + sketchstarted: this.onSketchModified, │ │ │ │ │ + sketchmodified: this.onSketchModified, │ │ │ │ │ + vertexmodified: this.onVertexModified, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ return activated │ │ │ │ │ }, │ │ │ │ │ deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + var deactivated = OpenLayers.Control.prototype.deactivate.call(this); │ │ │ │ │ if (deactivated) { │ │ │ │ │ - this.clearCache(); │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - beforefeaturesadded: this.cacheFeatures, │ │ │ │ │ - featuresremoved: this.clearCache, │ │ │ │ │ - moveend: this.cluster, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ + if (this.layer && this.layer.events) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + sketchstarted: this.onSketchModified, │ │ │ │ │ + sketchmodified: this.onSketchModified, │ │ │ │ │ + vertexmodified: this.onVertexModified, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + this.feature = null; │ │ │ │ │ + this.point = null; │ │ │ │ │ return deactivated │ │ │ │ │ }, │ │ │ │ │ - cacheFeatures: function(event) { │ │ │ │ │ - var propagate = true; │ │ │ │ │ - if (!this.clustering) { │ │ │ │ │ - this.clearCache(); │ │ │ │ │ - this.features = event.features; │ │ │ │ │ - this.cluster(); │ │ │ │ │ - propagate = false │ │ │ │ │ - } │ │ │ │ │ - return propagate │ │ │ │ │ + onSketchModified: function(event) { │ │ │ │ │ + this.feature = event.feature; │ │ │ │ │ + this.considerSnapping(event.vertex, event.vertex) │ │ │ │ │ }, │ │ │ │ │ - clearCache: function() { │ │ │ │ │ - if (!this.clustering) { │ │ │ │ │ - this.features = null │ │ │ │ │ - } │ │ │ │ │ + onVertexModified: function(event) { │ │ │ │ │ + this.feature = event.feature; │ │ │ │ │ + var loc = this.layer.map.getLonLatFromViewPortPx(event.pixel); │ │ │ │ │ + this.considerSnapping(event.vertex, new OpenLayers.Geometry.Point(loc.lon, loc.lat)) │ │ │ │ │ }, │ │ │ │ │ - cluster: function(event) { │ │ │ │ │ - if ((!event || event.zoomChanged) && this.features) { │ │ │ │ │ - var resolution = this.layer.map.getResolution(); │ │ │ │ │ - if (resolution != this.resolution || !this.clustersExist()) { │ │ │ │ │ - this.resolution = resolution; │ │ │ │ │ - var clusters = []; │ │ │ │ │ - var feature, clustered, cluster; │ │ │ │ │ - for (var i = 0; i < this.features.length; ++i) { │ │ │ │ │ - feature = this.features[i]; │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - clustered = false; │ │ │ │ │ - for (var j = clusters.length - 1; j >= 0; --j) { │ │ │ │ │ - cluster = clusters[j]; │ │ │ │ │ - if (this.shouldCluster(cluster, feature)) { │ │ │ │ │ - this.addToCluster(cluster, feature); │ │ │ │ │ - clustered = true; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (!clustered) { │ │ │ │ │ - clusters.push(this.createCluster(this.features[i])) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.clustering = true; │ │ │ │ │ - this.layer.removeAllFeatures(); │ │ │ │ │ - this.clustering = false; │ │ │ │ │ - if (clusters.length > 0) { │ │ │ │ │ - if (this.threshold > 1) { │ │ │ │ │ - var clone = clusters.slice(); │ │ │ │ │ - clusters = []; │ │ │ │ │ - var candidate; │ │ │ │ │ - for (var i = 0, len = clone.length; i < len; ++i) { │ │ │ │ │ - candidate = clone[i]; │ │ │ │ │ - if (candidate.attributes.count < this.threshold) { │ │ │ │ │ - Array.prototype.push.apply(clusters, candidate.cluster) │ │ │ │ │ - } else { │ │ │ │ │ - clusters.push(candidate) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + considerSnapping: function(point, loc) { │ │ │ │ │ + var best = { │ │ │ │ │ + rank: Number.POSITIVE_INFINITY, │ │ │ │ │ + dist: Number.POSITIVE_INFINITY, │ │ │ │ │ + x: null, │ │ │ │ │ + y: null │ │ │ │ │ + }; │ │ │ │ │ + var snapped = false; │ │ │ │ │ + var result, target; │ │ │ │ │ + for (var i = 0, len = this.targets.length; i < len; ++i) { │ │ │ │ │ + target = this.targets[i]; │ │ │ │ │ + result = this.testTarget(target, loc); │ │ │ │ │ + if (result) { │ │ │ │ │ + if (this.greedy) { │ │ │ │ │ + best = result; │ │ │ │ │ + best.target = target; │ │ │ │ │ + snapped = true; │ │ │ │ │ + break │ │ │ │ │ + } else { │ │ │ │ │ + if (result.rank < best.rank || result.rank === best.rank && result.dist < best.dist) { │ │ │ │ │ + best = result; │ │ │ │ │ + best.target = target; │ │ │ │ │ + snapped = true │ │ │ │ │ } │ │ │ │ │ - this.clustering = true; │ │ │ │ │ - this.layer.addFeatures(clusters); │ │ │ │ │ - this.clustering = false │ │ │ │ │ } │ │ │ │ │ - this.clusters = clusters │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - clustersExist: function() { │ │ │ │ │ - var exist = false; │ │ │ │ │ - if (this.clusters && this.clusters.length > 0 && this.clusters.length == this.layer.features.length) { │ │ │ │ │ - exist = true; │ │ │ │ │ - for (var i = 0; i < this.clusters.length; ++i) { │ │ │ │ │ - if (this.clusters[i] != this.layer.features[i]) { │ │ │ │ │ - exist = false; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ + if (snapped) { │ │ │ │ │ + var proceed = this.events.triggerEvent("beforesnap", { │ │ │ │ │ + point: point, │ │ │ │ │ + x: best.x, │ │ │ │ │ + y: best.y, │ │ │ │ │ + distance: best.dist, │ │ │ │ │ + layer: best.target.layer, │ │ │ │ │ + snapType: this.precedence[best.rank] │ │ │ │ │ + }); │ │ │ │ │ + if (proceed !== false) { │ │ │ │ │ + point.x = best.x; │ │ │ │ │ + point.y = best.y; │ │ │ │ │ + this.point = point; │ │ │ │ │ + this.events.triggerEvent("snap", { │ │ │ │ │ + point: point, │ │ │ │ │ + snapType: this.precedence[best.rank], │ │ │ │ │ + layer: best.target.layer, │ │ │ │ │ + distance: best.dist │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + snapped = false │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return exist │ │ │ │ │ - }, │ │ │ │ │ - shouldCluster: function(cluster, feature) { │ │ │ │ │ - var cc = cluster.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - var fc = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - var distance = Math.sqrt(Math.pow(cc.lon - fc.lon, 2) + Math.pow(cc.lat - fc.lat, 2)) / this.resolution; │ │ │ │ │ - return distance <= this.distance │ │ │ │ │ - }, │ │ │ │ │ - addToCluster: function(cluster, feature) { │ │ │ │ │ - cluster.cluster.push(feature); │ │ │ │ │ - cluster.attributes.count += 1 │ │ │ │ │ - }, │ │ │ │ │ - createCluster: function(feature) { │ │ │ │ │ - var center = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - var cluster = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Point(center.lon, center.lat), { │ │ │ │ │ - count: 1 │ │ │ │ │ - }); │ │ │ │ │ - cluster.cluster = [feature]; │ │ │ │ │ - return cluster │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Cluster" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Strategy.Paging = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - features: null, │ │ │ │ │ - length: 10, │ │ │ │ │ - num: null, │ │ │ │ │ - paging: false, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - beforefeaturesadded: this.cacheFeatures, │ │ │ │ │ - scope: this │ │ │ │ │ + if (this.point && !snapped) { │ │ │ │ │ + point.x = loc.x; │ │ │ │ │ + point.y = loc.y; │ │ │ │ │ + this.point = null; │ │ │ │ │ + this.events.triggerEvent("unsnap", { │ │ │ │ │ + point: point │ │ │ │ │ }) │ │ │ │ │ } │ │ │ │ │ - return activated │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.clearCache(); │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - beforefeaturesadded: this.cacheFeatures, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ + testTarget: function(target, loc) { │ │ │ │ │ + var resolution = this.layer.map.getResolution(); │ │ │ │ │ + if ("minResolution" in target) { │ │ │ │ │ + if (resolution < target.minResolution) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - cacheFeatures: function(event) { │ │ │ │ │ - if (!this.paging) { │ │ │ │ │ - this.clearCache(); │ │ │ │ │ - this.features = event.features; │ │ │ │ │ - this.pageNext(event) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - clearCache: function() { │ │ │ │ │ - if (this.features) { │ │ │ │ │ - for (var i = 0; i < this.features.length; ++i) { │ │ │ │ │ - this.features[i].destroy() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.features = null; │ │ │ │ │ - this.num = null │ │ │ │ │ - }, │ │ │ │ │ - pageCount: function() { │ │ │ │ │ - var numFeatures = this.features ? this.features.length : 0; │ │ │ │ │ - return Math.ceil(numFeatures / this.length) │ │ │ │ │ - }, │ │ │ │ │ - pageNum: function() { │ │ │ │ │ - return this.num │ │ │ │ │ - }, │ │ │ │ │ - pageLength: function(newLength) { │ │ │ │ │ - if (newLength && newLength > 0) { │ │ │ │ │ - this.length = newLength │ │ │ │ │ - } │ │ │ │ │ - return this.length │ │ │ │ │ - }, │ │ │ │ │ - pageNext: function(event) { │ │ │ │ │ - var changed = false; │ │ │ │ │ - if (this.features) { │ │ │ │ │ - if (this.num === null) { │ │ │ │ │ - this.num = -1 │ │ │ │ │ - } │ │ │ │ │ - var start = (this.num + 1) * this.length; │ │ │ │ │ - changed = this.page(start, event) │ │ │ │ │ - } │ │ │ │ │ - return changed │ │ │ │ │ - }, │ │ │ │ │ - pagePrevious: function() { │ │ │ │ │ - var changed = false; │ │ │ │ │ - if (this.features) { │ │ │ │ │ - if (this.num === null) { │ │ │ │ │ - this.num = this.pageCount() │ │ │ │ │ + if ("maxResolution" in target) { │ │ │ │ │ + if (resolution >= target.maxResolution) { │ │ │ │ │ + return null │ │ │ │ │ } │ │ │ │ │ - var start = (this.num - 1) * this.length; │ │ │ │ │ - changed = this.page(start) │ │ │ │ │ } │ │ │ │ │ - return changed │ │ │ │ │ - }, │ │ │ │ │ - page: function(start, event) { │ │ │ │ │ - var changed = false; │ │ │ │ │ - if (this.features) { │ │ │ │ │ - if (start >= 0 && start < this.features.length) { │ │ │ │ │ - var num = Math.floor(start / this.length); │ │ │ │ │ - if (num != this.num) { │ │ │ │ │ - this.paging = true; │ │ │ │ │ - var features = this.features.slice(start, start + this.length); │ │ │ │ │ - this.layer.removeFeatures(this.layer.features); │ │ │ │ │ - this.num = num; │ │ │ │ │ - if (event && event.features) { │ │ │ │ │ - event.features = features │ │ │ │ │ - } else { │ │ │ │ │ - this.layer.addFeatures(features) │ │ │ │ │ + var tolerance = { │ │ │ │ │ + node: this.getGeoTolerance(target.nodeTolerance, resolution), │ │ │ │ │ + vertex: this.getGeoTolerance(target.vertexTolerance, resolution), │ │ │ │ │ + edge: this.getGeoTolerance(target.edgeTolerance, resolution) │ │ │ │ │ + }; │ │ │ │ │ + var maxTolerance = Math.max(tolerance.node, tolerance.vertex, tolerance.edge); │ │ │ │ │ + var result = { │ │ │ │ │ + rank: Number.POSITIVE_INFINITY, │ │ │ │ │ + dist: Number.POSITIVE_INFINITY │ │ │ │ │ + }; │ │ │ │ │ + var eligible = false; │ │ │ │ │ + var features = target.layer.features; │ │ │ │ │ + var feature, type, vertices, vertex, closest, dist, found; │ │ │ │ │ + var numTypes = this.precedence.length; │ │ │ │ │ + var ll = new OpenLayers.LonLat(loc.x, loc.y); │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (feature !== this.feature && !feature._sketch && feature.state !== OpenLayers.State.DELETE && (!target.filter || target.filter.evaluate(feature))) { │ │ │ │ │ + if (feature.atPoint(ll, maxTolerance, maxTolerance)) { │ │ │ │ │ + for (var j = 0, stop = Math.min(result.rank + 1, numTypes); j < stop; ++j) { │ │ │ │ │ + type = this.precedence[j]; │ │ │ │ │ + if (target[type]) { │ │ │ │ │ + if (type === "edge") { │ │ │ │ │ + closest = feature.geometry.distanceTo(loc, { │ │ │ │ │ + details: true │ │ │ │ │ + }); │ │ │ │ │ + dist = closest.distance; │ │ │ │ │ + if (dist <= tolerance[type] && dist < result.dist) { │ │ │ │ │ + result = { │ │ │ │ │ + rank: j, │ │ │ │ │ + dist: dist, │ │ │ │ │ + x: closest.x0, │ │ │ │ │ + y: closest.y0 │ │ │ │ │ + }; │ │ │ │ │ + eligible = true; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + vertices = feature.geometry.getVertices(type === "node"); │ │ │ │ │ + found = false; │ │ │ │ │ + for (var k = 0, klen = vertices.length; k < klen; ++k) { │ │ │ │ │ + vertex = vertices[k]; │ │ │ │ │ + dist = vertex.distanceTo(loc); │ │ │ │ │ + if (dist <= tolerance[type] && (j < result.rank || j === result.rank && dist < result.dist)) { │ │ │ │ │ + result = { │ │ │ │ │ + rank: j, │ │ │ │ │ + dist: dist, │ │ │ │ │ + x: vertex.x, │ │ │ │ │ + y: vertex.y │ │ │ │ │ + }; │ │ │ │ │ + eligible = true; │ │ │ │ │ + found = true │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (found) { │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.paging = false; │ │ │ │ │ - changed = true │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return changed │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Paging" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Strategy.Filter = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - filter: null, │ │ │ │ │ - cache: null, │ │ │ │ │ - caching: false, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.cache = []; │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - beforefeaturesadded: this.handleAdd, │ │ │ │ │ - beforefeaturesremoved: this.handleRemove, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - this.cache = null; │ │ │ │ │ - if (this.layer && this.layer.events) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - beforefeaturesadded: this.handleAdd, │ │ │ │ │ - beforefeaturesremoved: this.handleRemove, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Strategy.prototype.deactivate.apply(this, arguments) │ │ │ │ │ + return eligible ? result : null │ │ │ │ │ }, │ │ │ │ │ - handleAdd: function(event) { │ │ │ │ │ - if (!this.caching && this.filter) { │ │ │ │ │ - var features = event.features; │ │ │ │ │ - event.features = []; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0, ii = features.length; i < ii; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (this.filter.evaluate(feature)) { │ │ │ │ │ - event.features.push(feature) │ │ │ │ │ - } else { │ │ │ │ │ - this.cache.push(feature) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + getGeoTolerance: function(tolerance, resolution) { │ │ │ │ │ + if (resolution !== this.resolution) { │ │ │ │ │ + this.resolution = resolution; │ │ │ │ │ + this.geoToleranceCache = {} │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - handleRemove: function(event) { │ │ │ │ │ - if (!this.caching) { │ │ │ │ │ - this.cache = [] │ │ │ │ │ + var geoTolerance = this.geoToleranceCache[tolerance]; │ │ │ │ │ + if (geoTolerance === undefined) { │ │ │ │ │ + geoTolerance = tolerance * resolution; │ │ │ │ │ + this.geoToleranceCache[tolerance] = geoTolerance │ │ │ │ │ } │ │ │ │ │ + return geoTolerance │ │ │ │ │ }, │ │ │ │ │ - setFilter: function(filter) { │ │ │ │ │ - this.filter = filter; │ │ │ │ │ - var previousCache = this.cache; │ │ │ │ │ - this.cache = []; │ │ │ │ │ - this.handleAdd({ │ │ │ │ │ - features: this.layer.features │ │ │ │ │ - }); │ │ │ │ │ - if (this.cache.length > 0) { │ │ │ │ │ - this.caching = true; │ │ │ │ │ - this.layer.removeFeatures(this.cache.slice()); │ │ │ │ │ - this.caching = false │ │ │ │ │ - } │ │ │ │ │ - if (previousCache.length > 0) { │ │ │ │ │ - var event = { │ │ │ │ │ - features: previousCache │ │ │ │ │ - }; │ │ │ │ │ - this.handleAdd(event); │ │ │ │ │ - if (event.features.length > 0) { │ │ │ │ │ - this.caching = true; │ │ │ │ │ - this.layer.addFeatures(event.features); │ │ │ │ │ - this.caching = false │ │ │ │ │ - } │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + this.deactivate() │ │ │ │ │ } │ │ │ │ │ + delete this.layer; │ │ │ │ │ + delete this.targets; │ │ │ │ │ + OpenLayers.Control.prototype.destroy.call(this) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Filter" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Snapping" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Strategy.Refresh = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - force: false, │ │ │ │ │ - interval: 0, │ │ │ │ │ - timer: null, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - if (this.layer.visibility === true) { │ │ │ │ │ - this.start() │ │ │ │ │ - } │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - visibilitychanged: this.reset, │ │ │ │ │ - scope: this │ │ │ │ │ +OpenLayers.Control.CacheWrite = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + layers: null, │ │ │ │ │ + imageFormat: "image/png", │ │ │ │ │ + quotaRegEx: /quota/i, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + var i, layers = this.layers || map.layers; │ │ │ │ │ + for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + this.addLayer({ │ │ │ │ │ + layer: layers[i] │ │ │ │ │ }) │ │ │ │ │ } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.stop(); │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - visibilitychanged: this.reset, │ │ │ │ │ + if (!this.layers) { │ │ │ │ │ + map.events.on({ │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + removeLayer: this.removeLayer, │ │ │ │ │ scope: this │ │ │ │ │ }) │ │ │ │ │ } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - reset: function() { │ │ │ │ │ - if (this.layer.visibility === true) { │ │ │ │ │ - this.start() │ │ │ │ │ - } else { │ │ │ │ │ - this.stop() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - start: function() { │ │ │ │ │ - if (this.interval && typeof this.interval === "number" && this.interval > 0) { │ │ │ │ │ - this.timer = window.setInterval(OpenLayers.Function.bind(this.refresh, this), this.interval) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - refresh: function() { │ │ │ │ │ - if (this.layer && this.layer.refresh && typeof this.layer.refresh == "function") { │ │ │ │ │ - this.layer.refresh({ │ │ │ │ │ - force: this.force │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - stop: function() { │ │ │ │ │ - if (this.timer !== null) { │ │ │ │ │ - window.clearInterval(this.timer); │ │ │ │ │ - this.timer = null │ │ │ │ │ - } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Refresh" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Strategy.Save = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - events: null, │ │ │ │ │ - auto: false, │ │ │ │ │ - timer: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Strategy.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.events = new OpenLayers.Events(this) │ │ │ │ │ + addLayer: function(evt) { │ │ │ │ │ + evt.layer.events.on({ │ │ │ │ │ + tileloadstart: this.makeSameOrigin, │ │ │ │ │ + tileloaded: this.onTileLoaded, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - if (this.auto) { │ │ │ │ │ - if (typeof this.auto === "number") { │ │ │ │ │ - this.timer = window.setInterval(OpenLayers.Function.bind(this.save, this), this.auto * 1e3) │ │ │ │ │ - } else { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - featureadded: this.triggerSave, │ │ │ │ │ - afterfeaturemodified: this.triggerSave, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return activated │ │ │ │ │ + removeLayer: function(evt) { │ │ │ │ │ + evt.layer.events.un({ │ │ │ │ │ + tileloadstart: this.makeSameOrigin, │ │ │ │ │ + tileloaded: this.onTileLoaded, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - if (this.auto) { │ │ │ │ │ - if (typeof this.auto === "number") { │ │ │ │ │ - window.clearInterval(this.timer) │ │ │ │ │ - } else { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - featureadded: this.triggerSave, │ │ │ │ │ - afterfeaturemodified: this.triggerSave, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ + makeSameOrigin: function(evt) { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + var tile = evt.tile; │ │ │ │ │ + if (tile instanceof OpenLayers.Tile.Image && !tile.crossOriginKeyword && tile.url.substr(0, 5) !== "data:") { │ │ │ │ │ + var sameOriginUrl = OpenLayers.Request.makeSameOrigin(tile.url, OpenLayers.ProxyHost); │ │ │ │ │ + OpenLayers.Control.CacheWrite.urlMap[sameOriginUrl] = tile.url; │ │ │ │ │ + tile.url = sameOriginUrl │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return deactivated │ │ │ │ │ }, │ │ │ │ │ - triggerSave: function(event) { │ │ │ │ │ - var feature = event.feature; │ │ │ │ │ - if (feature.state === OpenLayers.State.INSERT || feature.state === OpenLayers.State.UPDATE || feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ - this.save([event.feature]) │ │ │ │ │ + onTileLoaded: function(evt) { │ │ │ │ │ + if (this.active && !evt.aborted && evt.tile instanceof OpenLayers.Tile.Image && evt.tile.url.substr(0, 5) !== "data:") { │ │ │ │ │ + this.cache({ │ │ │ │ │ + tile: evt.tile │ │ │ │ │ + }); │ │ │ │ │ + delete OpenLayers.Control.CacheWrite.urlMap[evt.tile.url] │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - save: function(features) { │ │ │ │ │ - if (!features) { │ │ │ │ │ - features = this.layer.features │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("start", { │ │ │ │ │ - features: features │ │ │ │ │ - }); │ │ │ │ │ - var remote = this.layer.projection; │ │ │ │ │ - var local = this.layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var len = features.length; │ │ │ │ │ - var clones = new Array(len); │ │ │ │ │ - var orig, clone; │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - orig = features[i]; │ │ │ │ │ - clone = orig.clone(); │ │ │ │ │ - clone.fid = orig.fid; │ │ │ │ │ - clone.state = orig.state; │ │ │ │ │ - if (orig.url) { │ │ │ │ │ - clone.url = orig.url │ │ │ │ │ + cache: function(obj) { │ │ │ │ │ + if (window.localStorage) { │ │ │ │ │ + var tile = obj.tile; │ │ │ │ │ + try { │ │ │ │ │ + var canvasContext = tile.getCanvasContext(); │ │ │ │ │ + if (canvasContext) { │ │ │ │ │ + var urlMap = OpenLayers.Control.CacheWrite.urlMap; │ │ │ │ │ + var url = urlMap[tile.url] || tile.url; │ │ │ │ │ + window.localStorage.setItem("olCache_" + url, canvasContext.canvas.toDataURL(this.imageFormat)) │ │ │ │ │ } │ │ │ │ │ - clone._original = orig; │ │ │ │ │ - clone.geometry.transform(local, remote); │ │ │ │ │ - clones[i] = clone │ │ │ │ │ - } │ │ │ │ │ - features = clones │ │ │ │ │ - } │ │ │ │ │ - this.layer.protocol.commit(features, { │ │ │ │ │ - callback: this.onCommit, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - onCommit: function(response) { │ │ │ │ │ - var evt = { │ │ │ │ │ - response: response │ │ │ │ │ - }; │ │ │ │ │ - if (response.success()) { │ │ │ │ │ - var features = response.reqFeatures; │ │ │ │ │ - var state, feature; │ │ │ │ │ - var destroys = []; │ │ │ │ │ - var insertIds = response.insertIds || []; │ │ │ │ │ - var j = 0; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - feature = feature._original || feature; │ │ │ │ │ - state = feature.state; │ │ │ │ │ - if (state) { │ │ │ │ │ - if (state == OpenLayers.State.DELETE) { │ │ │ │ │ - destroys.push(feature) │ │ │ │ │ - } else if (state == OpenLayers.State.INSERT) { │ │ │ │ │ - feature.fid = insertIds[j]; │ │ │ │ │ - ++j │ │ │ │ │ - } │ │ │ │ │ - feature.state = null │ │ │ │ │ + } catch (e) { │ │ │ │ │ + var reason = e.name || e.message; │ │ │ │ │ + if (reason && this.quotaRegEx.test(reason)) { │ │ │ │ │ + this.events.triggerEvent("cachefull", { │ │ │ │ │ + tile: tile │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Console.error(e.toString()) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (destroys.length > 0) { │ │ │ │ │ - this.layer.destroyFeatures(destroys) │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("success", evt) │ │ │ │ │ - } else { │ │ │ │ │ - this.events.triggerEvent("fail", evt) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Save" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Strategy.Fixed = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - preload: false, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - refresh: this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - if (this.layer.visibility == true || this.preload) { │ │ │ │ │ - this.load() │ │ │ │ │ - } else { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - visibilitychanged: this.load, │ │ │ │ │ - scope: this │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.layers || this.map) { │ │ │ │ │ + var i, layers = this.layers || this.map.layers; │ │ │ │ │ + for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + this.removeLayer({ │ │ │ │ │ + layer: layers[i] │ │ │ │ │ }) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - refresh: this.load, │ │ │ │ │ - visibilitychanged: this.load, │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + addlayer: this.addLayer, │ │ │ │ │ + removeLayer: this.removeLayer, │ │ │ │ │ scope: this │ │ │ │ │ }) │ │ │ │ │ } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - load: function(options) { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - layer.events.triggerEvent("loadstart", { │ │ │ │ │ - filter: layer.filter │ │ │ │ │ - }); │ │ │ │ │ - layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: this.merge, │ │ │ │ │ - filter: layer.filter, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options)); │ │ │ │ │ - layer.events.un({ │ │ │ │ │ - visibilitychanged: this.load, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - merge: function(resp) { │ │ │ │ │ - var layer = this.layer; │ │ │ │ │ - layer.destroyFeatures(); │ │ │ │ │ - var features = resp.features; │ │ │ │ │ - if (features && features.length > 0) { │ │ │ │ │ - var remote = layer.projection; │ │ │ │ │ - var local = layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var geom; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - geom = features[i].geometry; │ │ │ │ │ - if (geom) { │ │ │ │ │ - geom.transform(remote, local) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - layer.addFeatures(features) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.CacheWrite" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.CacheWrite.clearCache = function() { │ │ │ │ │ + if (!window.localStorage) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var i, key; │ │ │ │ │ + for (i = window.localStorage.length - 1; i >= 0; --i) { │ │ │ │ │ + key = window.localStorage.key(i); │ │ │ │ │ + if (key.substr(0, 8) === "olCache_") { │ │ │ │ │ + window.localStorage.removeItem(key) │ │ │ │ │ } │ │ │ │ │ - layer.events.triggerEvent("loadend", { │ │ │ │ │ - response: resp │ │ │ │ │ - }) │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Control.CacheWrite.urlMap = {}; │ │ │ │ │ +OpenLayers.Control.PanPanel = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ + slideFactor: 50, │ │ │ │ │ + slideRatio: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ + var options = { │ │ │ │ │ + slideFactor: this.slideFactor, │ │ │ │ │ + slideRatio: this.slideRatio │ │ │ │ │ + }; │ │ │ │ │ + this.addControls([new OpenLayers.Control.Pan(OpenLayers.Control.Pan.NORTH, options), new OpenLayers.Control.Pan(OpenLayers.Control.Pan.SOUTH, options), new OpenLayers.Control.Pan(OpenLayers.Control.Pan.EAST, options), new OpenLayers.Control.Pan(OpenLayers.Control.Pan.WEST, options)]) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.Fixed" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.PanPanel" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Strategy.BBOX = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ - bounds: null, │ │ │ │ │ - resolution: null, │ │ │ │ │ - ratio: 2, │ │ │ │ │ - resFactor: null, │ │ │ │ │ - response: null, │ │ │ │ │ +OpenLayers.Control.MousePosition = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + element: null, │ │ │ │ │ + prefix: "", │ │ │ │ │ + separator: ", ", │ │ │ │ │ + suffix: "", │ │ │ │ │ + numDigits: 5, │ │ │ │ │ + granularity: 10, │ │ │ │ │ + emptyString: null, │ │ │ │ │ + lastXy: null, │ │ │ │ │ + displayProjection: null, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ activate: function() { │ │ │ │ │ - var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - moveend: this.update, │ │ │ │ │ - refresh: this.update, │ │ │ │ │ - visibilitychanged: this.update, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.update() │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + this.map.events.register("mousemove", this, this.redraw); │ │ │ │ │ + this.map.events.register("mouseout", this, this.reset); │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return activated │ │ │ │ │ }, │ │ │ │ │ deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - moveend: this.update, │ │ │ │ │ - refresh: this.update, │ │ │ │ │ - visibilitychanged: this.update, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - update: function(options) { │ │ │ │ │ - var mapBounds = this.getMapBounds(); │ │ │ │ │ - if (mapBounds !== null && (options && options.force || this.layer.visibility && this.layer.calculateInRange() && this.invalidBounds(mapBounds))) { │ │ │ │ │ - this.calculateBounds(mapBounds); │ │ │ │ │ - this.resolution = this.layer.map.getResolution(); │ │ │ │ │ - this.triggerRead(options) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getMapBounds: function() { │ │ │ │ │ - if (this.layer.map === null) { │ │ │ │ │ - return null │ │ │ │ │ - } │ │ │ │ │ - var bounds = this.layer.map.getExtent(); │ │ │ │ │ - if (bounds && !this.layer.projection.equals(this.layer.map.getProjectionObject())) { │ │ │ │ │ - bounds = bounds.clone().transform(this.layer.map.getProjectionObject(), this.layer.projection) │ │ │ │ │ + if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ + this.map.events.unregister("mousemove", this, this.redraw); │ │ │ │ │ + this.map.events.unregister("mouseout", this, this.reset); │ │ │ │ │ + this.element.innerHTML = ""; │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ } │ │ │ │ │ - return bounds │ │ │ │ │ }, │ │ │ │ │ - invalidBounds: function(mapBounds) { │ │ │ │ │ - if (!mapBounds) { │ │ │ │ │ - mapBounds = this.getMapBounds() │ │ │ │ │ - } │ │ │ │ │ - var invalid = !this.bounds || !this.bounds.containsBounds(mapBounds); │ │ │ │ │ - if (!invalid && this.resFactor) { │ │ │ │ │ - var ratio = this.resolution / this.layer.map.getResolution(); │ │ │ │ │ - invalid = ratio >= this.resFactor || ratio <= 1 / this.resFactor │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (!this.element) { │ │ │ │ │ + this.div.left = ""; │ │ │ │ │ + this.div.top = ""; │ │ │ │ │ + this.element = this.div │ │ │ │ │ } │ │ │ │ │ - return invalid │ │ │ │ │ + return this.div │ │ │ │ │ }, │ │ │ │ │ - calculateBounds: function(mapBounds) { │ │ │ │ │ - if (!mapBounds) { │ │ │ │ │ - mapBounds = this.getMapBounds() │ │ │ │ │ + redraw: function(evt) { │ │ │ │ │ + var lonLat; │ │ │ │ │ + if (evt == null) { │ │ │ │ │ + this.reset(); │ │ │ │ │ + return │ │ │ │ │ + } else { │ │ │ │ │ + if (this.lastXy == null || Math.abs(evt.xy.x - this.lastXy.x) > this.granularity || Math.abs(evt.xy.y - this.lastXy.y) > this.granularity) { │ │ │ │ │ + this.lastXy = evt.xy; │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + lonLat = this.map.getLonLatFromPixel(evt.xy); │ │ │ │ │ + if (!lonLat) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (this.displayProjection) { │ │ │ │ │ + lonLat.transform(this.map.getProjectionObject(), this.displayProjection) │ │ │ │ │ + } │ │ │ │ │ + this.lastXy = evt.xy │ │ │ │ │ } │ │ │ │ │ - var center = mapBounds.getCenterLonLat(); │ │ │ │ │ - var dataWidth = mapBounds.getWidth() * this.ratio; │ │ │ │ │ - var dataHeight = mapBounds.getHeight() * this.ratio; │ │ │ │ │ - this.bounds = new OpenLayers.Bounds(center.lon - dataWidth / 2, center.lat - dataHeight / 2, center.lon + dataWidth / 2, center.lat + dataHeight / 2) │ │ │ │ │ - }, │ │ │ │ │ - triggerRead: function(options) { │ │ │ │ │ - if (this.response && !(options && options.noAbort === true)) { │ │ │ │ │ - this.layer.protocol.abort(this.response); │ │ │ │ │ - this.layer.events.triggerEvent("loadend") │ │ │ │ │ + var newHtml = this.formatOutput(lonLat); │ │ │ │ │ + if (newHtml != this.element.innerHTML) { │ │ │ │ │ + this.element.innerHTML = newHtml │ │ │ │ │ } │ │ │ │ │ - var evt = { │ │ │ │ │ - filter: this.createFilter() │ │ │ │ │ - }; │ │ │ │ │ - this.layer.events.triggerEvent("loadstart", evt); │ │ │ │ │ - this.response = this.layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ - filter: evt.filter, │ │ │ │ │ - callback: this.merge, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options)) │ │ │ │ │ }, │ │ │ │ │ - createFilter: function() { │ │ │ │ │ - var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ - value: this.bounds, │ │ │ │ │ - projection: this.layer.projection │ │ │ │ │ - }); │ │ │ │ │ - if (this.layer.filter) { │ │ │ │ │ - filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ - type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ - filters: [this.layer.filter, filter] │ │ │ │ │ - }) │ │ │ │ │ + reset: function(evt) { │ │ │ │ │ + if (this.emptyString != null) { │ │ │ │ │ + this.element.innerHTML = this.emptyString │ │ │ │ │ } │ │ │ │ │ - return filter │ │ │ │ │ }, │ │ │ │ │ - merge: function(resp) { │ │ │ │ │ - this.layer.destroyFeatures(); │ │ │ │ │ - if (resp.success()) { │ │ │ │ │ - var features = resp.features; │ │ │ │ │ - if (features && features.length > 0) { │ │ │ │ │ - var remote = this.layer.projection; │ │ │ │ │ - var local = this.layer.map.getProjectionObject(); │ │ │ │ │ - if (!local.equals(remote)) { │ │ │ │ │ - var geom; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - geom = features[i].geometry; │ │ │ │ │ - if (geom) { │ │ │ │ │ - geom.transform(remote, local) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.layer.addFeatures(features) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - this.bounds = null │ │ │ │ │ - } │ │ │ │ │ - this.response = null; │ │ │ │ │ - this.layer.events.triggerEvent("loadend", { │ │ │ │ │ - response: resp │ │ │ │ │ - }) │ │ │ │ │ + formatOutput: function(lonLat) { │ │ │ │ │ + var digits = parseInt(this.numDigits); │ │ │ │ │ + var newHtml = this.prefix + lonLat.lon.toFixed(digits) + this.separator + lonLat.lat.toFixed(digits) + this.suffix; │ │ │ │ │ + return newHtml │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Strategy.BBOX" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.MousePosition" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.Panel = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - controls: null, │ │ │ │ │ +OpenLayers.Control.Graticule = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ autoActivate: true, │ │ │ │ │ - defaultControl: null, │ │ │ │ │ - saveState: false, │ │ │ │ │ - allowDepress: false, │ │ │ │ │ - activeState: null, │ │ │ │ │ + intervals: [45, 30, 20, 10, 5, 2, 1, .5, .2, .1, .05, .01, .005, .002, .001], │ │ │ │ │ + displayInLayerSwitcher: true, │ │ │ │ │ + visible: true, │ │ │ │ │ + numPoints: 50, │ │ │ │ │ + targetSize: 200, │ │ │ │ │ + layerName: null, │ │ │ │ │ + labelled: true, │ │ │ │ │ + labelFormat: "dm", │ │ │ │ │ + lineSymbolizer: { │ │ │ │ │ + strokeColor: "#333", │ │ │ │ │ + strokeWidth: 1, │ │ │ │ │ + strokeOpacity: .5 │ │ │ │ │ + }, │ │ │ │ │ + labelSymbolizer: {}, │ │ │ │ │ + gratLayer: null, │ │ │ │ │ initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.layerName = options.layerName || OpenLayers.i18n("Graticule"); │ │ │ │ │ OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.controls = []; │ │ │ │ │ - this.activeState = {} │ │ │ │ │ + this.labelSymbolizer.stroke = false; │ │ │ │ │ + this.labelSymbolizer.fill = false; │ │ │ │ │ + this.labelSymbolizer.label = "${label}"; │ │ │ │ │ + this.labelSymbolizer.labelAlign = "${labelAlign}"; │ │ │ │ │ + this.labelSymbolizer.labelXOffset = "${xOffset}"; │ │ │ │ │ + this.labelSymbolizer.labelYOffset = "${yOffset}" │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onButtonClick) │ │ │ │ │ - } │ │ │ │ │ + this.deactivate(); │ │ │ │ │ OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - for (var ctl, i = this.controls.length - 1; i >= 0; i--) { │ │ │ │ │ - ctl = this.controls[i]; │ │ │ │ │ - if (ctl.events) { │ │ │ │ │ - ctl.events.un({ │ │ │ │ │ - activate: this.iconOn, │ │ │ │ │ - deactivate: this.iconOff │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - ctl.panel_div = null │ │ │ │ │ + if (this.gratLayer) { │ │ │ │ │ + this.gratLayer.destroy(); │ │ │ │ │ + this.gratLayer = null │ │ │ │ │ } │ │ │ │ │ - this.activeState = null │ │ │ │ │ + }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (!this.gratLayer) { │ │ │ │ │ + var gratStyle = new OpenLayers.Style({}, { │ │ │ │ │ + rules: [new OpenLayers.Rule({ │ │ │ │ │ + symbolizer: { │ │ │ │ │ + Point: this.labelSymbolizer, │ │ │ │ │ + Line: this.lineSymbolizer │ │ │ │ │ + } │ │ │ │ │ + })] │ │ │ │ │ + }); │ │ │ │ │ + this.gratLayer = new OpenLayers.Layer.Vector(this.layerName, { │ │ │ │ │ + styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ + default: gratStyle │ │ │ │ │ + }), │ │ │ │ │ + visibility: this.visible, │ │ │ │ │ + displayInLayerSwitcher: this.displayInLayerSwitcher │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + return this.div │ │ │ │ │ }, │ │ │ │ │ activate: function() { │ │ │ │ │ if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - control = this.controls[i]; │ │ │ │ │ - if (control === this.defaultControl || this.saveState && this.activeState[control.id]) { │ │ │ │ │ - control.activate() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.saveState === true) { │ │ │ │ │ - this.defaultControl = null │ │ │ │ │ - } │ │ │ │ │ - this.redraw(); │ │ │ │ │ + this.map.addLayer(this.gratLayer); │ │ │ │ │ + this.map.events.register("moveend", this, this.update); │ │ │ │ │ + this.update(); │ │ │ │ │ return true │ │ │ │ │ } else { │ │ │ │ │ return false │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ deactivate: function() { │ │ │ │ │ if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - control = this.controls[i]; │ │ │ │ │ - this.activeState[control.id] = control.deactivate() │ │ │ │ │ - } │ │ │ │ │ - this.redraw(); │ │ │ │ │ + this.map.events.unregister("moveend", this, this.update); │ │ │ │ │ + this.map.removeLayer(this.gratLayer); │ │ │ │ │ return true │ │ │ │ │ } else { │ │ │ │ │ return false │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (this.outsideViewport) { │ │ │ │ │ - this.events.attachToElement(this.div); │ │ │ │ │ - this.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ - } else { │ │ │ │ │ - this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ - } │ │ │ │ │ - this.addControlsToMap(this.controls); │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - redraw: function() { │ │ │ │ │ - for (var l = this.div.childNodes.length, i = l - 1; i >= 0; i--) { │ │ │ │ │ - this.div.removeChild(this.div.childNodes[i]) │ │ │ │ │ - } │ │ │ │ │ - this.div.innerHTML = ""; │ │ │ │ │ - if (this.active) { │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - this.div.appendChild(this.controls[i].panel_div) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - activateControl: function(control) { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - if (control.type == OpenLayers.Control.TYPE_BUTTON) { │ │ │ │ │ - control.trigger(); │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (control.type == OpenLayers.Control.TYPE_TOGGLE) { │ │ │ │ │ - if (control.active) { │ │ │ │ │ - control.deactivate() │ │ │ │ │ - } else { │ │ │ │ │ - control.activate() │ │ │ │ │ - } │ │ │ │ │ + update: function() { │ │ │ │ │ + var mapBounds = this.map.getExtent(); │ │ │ │ │ + if (!mapBounds) { │ │ │ │ │ return │ │ │ │ │ } │ │ │ │ │ - if (this.allowDepress && control.active) { │ │ │ │ │ - control.deactivate() │ │ │ │ │ - } else { │ │ │ │ │ - var c; │ │ │ │ │ - for (var i = 0, len = this.controls.length; i < len; i++) { │ │ │ │ │ - c = this.controls[i]; │ │ │ │ │ - if (c != control && (c.type === OpenLayers.Control.TYPE_TOOL || c.type == null)) { │ │ │ │ │ - c.deactivate() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - control.activate() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - addControls: function(controls) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(controls)) { │ │ │ │ │ - controls = [controls] │ │ │ │ │ - } │ │ │ │ │ - this.controls = this.controls.concat(controls); │ │ │ │ │ - for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ - var control = controls[i], │ │ │ │ │ - element = this.createControlMarkup(control); │ │ │ │ │ - OpenLayers.Element.addClass(element, control.displayClass + "ItemInactive"); │ │ │ │ │ - OpenLayers.Element.addClass(element, "olButton"); │ │ │ │ │ - if (control.title != "" && !element.title) { │ │ │ │ │ - element.title = control.title │ │ │ │ │ - } │ │ │ │ │ - control.panel_div = element │ │ │ │ │ - } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.addControlsToMap(controls); │ │ │ │ │ - this.redraw() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - createControlMarkup: function(control) { │ │ │ │ │ - return document.createElement("div") │ │ │ │ │ - }, │ │ │ │ │ - addControlsToMap: function(controls) { │ │ │ │ │ - var control; │ │ │ │ │ - for (var i = 0, len = controls.length; i < len; i++) { │ │ │ │ │ - control = controls[i]; │ │ │ │ │ - if (control.autoActivate === true) { │ │ │ │ │ - control.autoActivate = false; │ │ │ │ │ - this.map.addControl(control); │ │ │ │ │ - control.autoActivate = true │ │ │ │ │ - } else { │ │ │ │ │ - this.map.addControl(control); │ │ │ │ │ - control.deactivate() │ │ │ │ │ - } │ │ │ │ │ - control.events.on({ │ │ │ │ │ - activate: this.iconOn, │ │ │ │ │ - deactivate: this.iconOff │ │ │ │ │ - }) │ │ │ │ │ + this.gratLayer.destroyFeatures(); │ │ │ │ │ + var llProj = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ + var mapProj = this.map.getProjectionObject(); │ │ │ │ │ + var mapRes = this.map.getResolution(); │ │ │ │ │ + if (mapProj.proj && mapProj.proj.projName == "longlat") { │ │ │ │ │ + this.numPoints = 1 │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - iconOn: function() { │ │ │ │ │ - var d = this.panel_div; │ │ │ │ │ - var re = new RegExp("\\b(" + this.displayClass + "Item)Inactive\\b"); │ │ │ │ │ - d.className = d.className.replace(re, "$1Active") │ │ │ │ │ - }, │ │ │ │ │ - iconOff: function() { │ │ │ │ │ - var d = this.panel_div; │ │ │ │ │ - var re = new RegExp("\\b(" + this.displayClass + "Item)Active\\b"); │ │ │ │ │ - d.className = d.className.replace(re, "$1Inactive") │ │ │ │ │ - }, │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - var controls = this.controls, │ │ │ │ │ - button = evt.buttonElement; │ │ │ │ │ - for (var i = controls.length - 1; i >= 0; --i) { │ │ │ │ │ - if (controls[i].panel_div === button) { │ │ │ │ │ - this.activateControl(controls[i]); │ │ │ │ │ + var mapCenter = this.map.getCenter(); │ │ │ │ │ + var mapCenterLL = new OpenLayers.Pixel(mapCenter.lon, mapCenter.lat); │ │ │ │ │ + OpenLayers.Projection.transform(mapCenterLL, mapProj, llProj); │ │ │ │ │ + var testSq = this.targetSize * mapRes; │ │ │ │ │ + testSq *= testSq; │ │ │ │ │ + var llInterval; │ │ │ │ │ + for (var i = 0; i < this.intervals.length; ++i) { │ │ │ │ │ + llInterval = this.intervals[i]; │ │ │ │ │ + var delta = llInterval / 2; │ │ │ │ │ + var p1 = mapCenterLL.offset({ │ │ │ │ │ + x: -delta, │ │ │ │ │ + y: -delta │ │ │ │ │ + }); │ │ │ │ │ + var p2 = mapCenterLL.offset({ │ │ │ │ │ + x: delta, │ │ │ │ │ + y: delta │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Projection.transform(p1, llProj, mapProj); │ │ │ │ │ + OpenLayers.Projection.transform(p2, llProj, mapProj); │ │ │ │ │ + var distSq = (p1.x - p2.x) * (p1.x - p2.x) + (p1.y - p2.y) * (p1.y - p2.y); │ │ │ │ │ + if (distSq <= testSq) { │ │ │ │ │ break │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - getControlsBy: function(property, match) { │ │ │ │ │ - var test = typeof match.test == "function"; │ │ │ │ │ - var found = OpenLayers.Array.filter(this.controls, function(item) { │ │ │ │ │ - return item[property] == match || test && match.test(item[property]) │ │ │ │ │ - }); │ │ │ │ │ - return found │ │ │ │ │ - }, │ │ │ │ │ - getControlsByName: function(match) { │ │ │ │ │ - return this.getControlsBy("name", match) │ │ │ │ │ - }, │ │ │ │ │ - getControlsByClass: function(match) { │ │ │ │ │ - return this.getControlsBy("CLASS_NAME", match) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Panel" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.ZoomBox = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ - out: false, │ │ │ │ │ - keyMask: null, │ │ │ │ │ - alwaysZoom: false, │ │ │ │ │ - zoomOnClick: true, │ │ │ │ │ - draw: function() { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Box(this, { │ │ │ │ │ - done: this.zoomBox │ │ │ │ │ - }, { │ │ │ │ │ - keyMask: this.keyMask │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - zoomBox: function(position) { │ │ │ │ │ - if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ - var bounds, targetCenterPx = position.getCenterPixel(); │ │ │ │ │ - if (!this.out) { │ │ │ │ │ - var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.left, │ │ │ │ │ - y: position.bottom │ │ │ │ │ - }); │ │ │ │ │ - var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.right, │ │ │ │ │ - y: position.top │ │ │ │ │ - }); │ │ │ │ │ - bounds = new OpenLayers.Bounds(minXY.lon, minXY.lat, maxXY.lon, maxXY.lat) │ │ │ │ │ - } else { │ │ │ │ │ - var pixWidth = position.right - position.left; │ │ │ │ │ - var pixHeight = position.bottom - position.top; │ │ │ │ │ - var zoomFactor = Math.min(this.map.size.h / pixHeight, this.map.size.w / pixWidth); │ │ │ │ │ - var extent = this.map.getExtent(); │ │ │ │ │ - var center = this.map.getLonLatFromPixel(targetCenterPx); │ │ │ │ │ - var xmin = center.lon - extent.getWidth() / 2 * zoomFactor; │ │ │ │ │ - var xmax = center.lon + extent.getWidth() / 2 * zoomFactor; │ │ │ │ │ - var ymin = center.lat - extent.getHeight() / 2 * zoomFactor; │ │ │ │ │ - var ymax = center.lat + extent.getHeight() / 2 * zoomFactor; │ │ │ │ │ - bounds = new OpenLayers.Bounds(xmin, ymin, xmax, ymax) │ │ │ │ │ + mapCenterLL.x = Math.floor(mapCenterLL.x / llInterval) * llInterval; │ │ │ │ │ + mapCenterLL.y = Math.floor(mapCenterLL.y / llInterval) * llInterval; │ │ │ │ │ + var iter = 0; │ │ │ │ │ + var centerLonPoints = [mapCenterLL.clone()]; │ │ │ │ │ + var newPoint = mapCenterLL.clone(); │ │ │ │ │ + var mapXY; │ │ │ │ │ + do { │ │ │ │ │ + newPoint = newPoint.offset({ │ │ │ │ │ + x: 0, │ │ │ │ │ + y: llInterval │ │ │ │ │ + }); │ │ │ │ │ + mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ + centerLonPoints.unshift(newPoint) │ │ │ │ │ + } while (mapBounds.containsPixel(mapXY) && ++iter < 1e3); │ │ │ │ │ + newPoint = mapCenterLL.clone(); │ │ │ │ │ + do { │ │ │ │ │ + newPoint = newPoint.offset({ │ │ │ │ │ + x: 0, │ │ │ │ │ + y: -llInterval │ │ │ │ │ + }); │ │ │ │ │ + mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ + centerLonPoints.push(newPoint) │ │ │ │ │ + } while (mapBounds.containsPixel(mapXY) && ++iter < 1e3); │ │ │ │ │ + iter = 0; │ │ │ │ │ + var centerLatPoints = [mapCenterLL.clone()]; │ │ │ │ │ + newPoint = mapCenterLL.clone(); │ │ │ │ │ + do { │ │ │ │ │ + newPoint = newPoint.offset({ │ │ │ │ │ + x: -llInterval, │ │ │ │ │ + y: 0 │ │ │ │ │ + }); │ │ │ │ │ + mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ + centerLatPoints.unshift(newPoint) │ │ │ │ │ + } while (mapBounds.containsPixel(mapXY) && ++iter < 1e3); │ │ │ │ │ + newPoint = mapCenterLL.clone(); │ │ │ │ │ + do { │ │ │ │ │ + newPoint = newPoint.offset({ │ │ │ │ │ + x: llInterval, │ │ │ │ │ + y: 0 │ │ │ │ │ + }); │ │ │ │ │ + mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ + centerLatPoints.push(newPoint) │ │ │ │ │ + } while (mapBounds.containsPixel(mapXY) && ++iter < 1e3); │ │ │ │ │ + var lines = []; │ │ │ │ │ + for (var i = 0; i < centerLatPoints.length; ++i) { │ │ │ │ │ + var lon = centerLatPoints[i].x; │ │ │ │ │ + var pointList = []; │ │ │ │ │ + var labelPoint = null; │ │ │ │ │ + var latEnd = Math.min(centerLonPoints[0].y, 90); │ │ │ │ │ + var latStart = Math.max(centerLonPoints[centerLonPoints.length - 1].y, -90); │ │ │ │ │ + var latDelta = (latEnd - latStart) / this.numPoints; │ │ │ │ │ + var lat = latStart; │ │ │ │ │ + for (var j = 0; j <= this.numPoints; ++j) { │ │ │ │ │ + var gridPoint = new OpenLayers.Geometry.Point(lon, lat); │ │ │ │ │ + gridPoint.transform(llProj, mapProj); │ │ │ │ │ + pointList.push(gridPoint); │ │ │ │ │ + lat += latDelta; │ │ │ │ │ + if (gridPoint.y >= mapBounds.bottom && !labelPoint) { │ │ │ │ │ + labelPoint = gridPoint │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var lastZoom = this.map.getZoom(), │ │ │ │ │ - size = this.map.getSize(), │ │ │ │ │ - centerPx = { │ │ │ │ │ - x: size.w / 2, │ │ │ │ │ - y: size.h / 2 │ │ │ │ │ - }, │ │ │ │ │ - zoom = this.map.getZoomForExtent(bounds), │ │ │ │ │ - oldRes = this.map.getResolution(), │ │ │ │ │ - newRes = this.map.getResolutionForZoom(zoom); │ │ │ │ │ - if (oldRes == newRes) { │ │ │ │ │ - this.map.setCenter(this.map.getLonLatFromPixel(targetCenterPx)) │ │ │ │ │ - } else { │ │ │ │ │ - var zoomOriginPx = { │ │ │ │ │ - x: (oldRes * targetCenterPx.x - newRes * centerPx.x) / (oldRes - newRes), │ │ │ │ │ - y: (oldRes * targetCenterPx.y - newRes * centerPx.y) / (oldRes - newRes) │ │ │ │ │ + if (this.labelled) { │ │ │ │ │ + var labelPos = new OpenLayers.Geometry.Point(labelPoint.x, mapBounds.bottom); │ │ │ │ │ + var labelAttrs = { │ │ │ │ │ + value: lon, │ │ │ │ │ + label: this.labelled ? OpenLayers.Util.getFormattedLonLat(lon, "lon", this.labelFormat) : "", │ │ │ │ │ + labelAlign: "cb", │ │ │ │ │ + xOffset: 0, │ │ │ │ │ + yOffset: 2 │ │ │ │ │ }; │ │ │ │ │ - this.map.zoomTo(zoom, zoomOriginPx) │ │ │ │ │ - } │ │ │ │ │ - if (lastZoom == this.map.getZoom() && this.alwaysZoom == true) { │ │ │ │ │ - this.map.zoomTo(lastZoom + (this.out ? -1 : 1)) │ │ │ │ │ - } │ │ │ │ │ - } else if (this.zoomOnClick) { │ │ │ │ │ - if (!this.out) { │ │ │ │ │ - this.map.zoomTo(this.map.getZoom() + 1, position) │ │ │ │ │ - } else { │ │ │ │ │ - this.map.zoomTo(this.map.getZoom() - 1, position) │ │ │ │ │ + this.gratLayer.addFeatures(new OpenLayers.Feature.Vector(labelPos, labelAttrs)) │ │ │ │ │ } │ │ │ │ │ + var geom = new OpenLayers.Geometry.LineString(pointList); │ │ │ │ │ + lines.push(new OpenLayers.Feature.Vector(geom)) │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomBox" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.DragPan = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ - panned: false, │ │ │ │ │ - interval: 0, │ │ │ │ │ - documentDrag: false, │ │ │ │ │ - kinetic: null, │ │ │ │ │ - enableKinetic: true, │ │ │ │ │ - kineticInterval: 10, │ │ │ │ │ - draw: function() { │ │ │ │ │ - if (this.enableKinetic && OpenLayers.Kinetic) { │ │ │ │ │ - var config = { │ │ │ │ │ - interval: this.kineticInterval │ │ │ │ │ - }; │ │ │ │ │ - if (typeof this.enableKinetic === "object") { │ │ │ │ │ - config = OpenLayers.Util.extend(config, this.enableKinetic) │ │ │ │ │ + for (var j = 0; j < centerLonPoints.length; ++j) { │ │ │ │ │ + lat = centerLonPoints[j].y; │ │ │ │ │ + if (lat < -90 || lat > 90) { │ │ │ │ │ + continue │ │ │ │ │ } │ │ │ │ │ - this.kinetic = new OpenLayers.Kinetic(config) │ │ │ │ │ - } │ │ │ │ │ - this.handler = new OpenLayers.Handler.Drag(this, { │ │ │ │ │ - move: this.panMap, │ │ │ │ │ - done: this.panMapDone, │ │ │ │ │ - down: this.panMapStart │ │ │ │ │ - }, { │ │ │ │ │ - interval: this.interval, │ │ │ │ │ - documentDrag: this.documentDrag │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - panMapStart: function() { │ │ │ │ │ - if (this.kinetic) { │ │ │ │ │ - this.kinetic.begin() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - panMap: function(xy) { │ │ │ │ │ - if (this.kinetic) { │ │ │ │ │ - this.kinetic.update(xy) │ │ │ │ │ - } │ │ │ │ │ - this.panned = true; │ │ │ │ │ - this.map.pan(this.handler.last.x - xy.x, this.handler.last.y - xy.y, { │ │ │ │ │ - dragging: true, │ │ │ │ │ - animate: false │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - panMapDone: function(xy) { │ │ │ │ │ - if (this.panned) { │ │ │ │ │ - var res = null; │ │ │ │ │ - if (this.kinetic) { │ │ │ │ │ - res = this.kinetic.end(xy) │ │ │ │ │ + var pointList = []; │ │ │ │ │ + var lonStart = centerLatPoints[0].x; │ │ │ │ │ + var lonEnd = centerLatPoints[centerLatPoints.length - 1].x; │ │ │ │ │ + var lonDelta = (lonEnd - lonStart) / this.numPoints; │ │ │ │ │ + var lon = lonStart; │ │ │ │ │ + var labelPoint = null; │ │ │ │ │ + for (var i = 0; i <= this.numPoints; ++i) { │ │ │ │ │ + var gridPoint = new OpenLayers.Geometry.Point(lon, lat); │ │ │ │ │ + gridPoint.transform(llProj, mapProj); │ │ │ │ │ + pointList.push(gridPoint); │ │ │ │ │ + lon += lonDelta; │ │ │ │ │ + if (gridPoint.x < mapBounds.right) { │ │ │ │ │ + labelPoint = gridPoint │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.map.pan(this.handler.last.x - xy.x, this.handler.last.y - xy.y, { │ │ │ │ │ - dragging: !!res, │ │ │ │ │ - animate: false │ │ │ │ │ - }); │ │ │ │ │ - if (res) { │ │ │ │ │ - var self = this; │ │ │ │ │ - this.kinetic.move(res, function(x, y, end) { │ │ │ │ │ - self.map.pan(x, y, { │ │ │ │ │ - dragging: !end, │ │ │ │ │ - animate: false │ │ │ │ │ - }) │ │ │ │ │ - }) │ │ │ │ │ + if (this.labelled) { │ │ │ │ │ + var labelPos = new OpenLayers.Geometry.Point(mapBounds.right, labelPoint.y); │ │ │ │ │ + var labelAttrs = { │ │ │ │ │ + value: lat, │ │ │ │ │ + label: this.labelled ? OpenLayers.Util.getFormattedLonLat(lat, "lat", this.labelFormat) : "", │ │ │ │ │ + labelAlign: "rb", │ │ │ │ │ + xOffset: -2, │ │ │ │ │ + yOffset: 2 │ │ │ │ │ + }; │ │ │ │ │ + this.gratLayer.addFeatures(new OpenLayers.Feature.Vector(labelPos, labelAttrs)) │ │ │ │ │ } │ │ │ │ │ - this.panned = false │ │ │ │ │ + var geom = new OpenLayers.Geometry.LineString(pointList); │ │ │ │ │ + lines.push(new OpenLayers.Feature.Vector(geom)) │ │ │ │ │ } │ │ │ │ │ + this.gratLayer.addFeatures(lines) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.DragPan" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Graticule" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Control.Navigation = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ dragPan: null, │ │ │ │ │ dragPanOptions: null, │ │ │ │ │ pinchZoom: null, │ │ │ │ │ pinchZoomOptions: null, │ │ │ │ │ documentDrag: false, │ │ │ │ │ @@ -32940,1610 +35151,14 @@ │ │ │ │ │ scope: this │ │ │ │ │ }) │ │ │ │ │ } │ │ │ │ │ OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.CacheRead" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.ModifyFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - documentDrag: false, │ │ │ │ │ - geometryTypes: null, │ │ │ │ │ - clickout: true, │ │ │ │ │ - toggle: true, │ │ │ │ │ - standalone: false, │ │ │ │ │ - layer: null, │ │ │ │ │ - feature: null, │ │ │ │ │ - vertex: null, │ │ │ │ │ - vertices: null, │ │ │ │ │ - virtualVertices: null, │ │ │ │ │ - handlers: null, │ │ │ │ │ - deleteCodes: null, │ │ │ │ │ - virtualStyle: null, │ │ │ │ │ - vertexRenderIntent: null, │ │ │ │ │ - mode: null, │ │ │ │ │ - createVertices: true, │ │ │ │ │ - modified: false, │ │ │ │ │ - radiusHandle: null, │ │ │ │ │ - dragHandle: null, │ │ │ │ │ - onModificationStart: function() {}, │ │ │ │ │ - onModification: function() {}, │ │ │ │ │ - onModificationEnd: function() {}, │ │ │ │ │ - initialize: function(layer, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - this.virtualVertices = []; │ │ │ │ │ - this.virtualStyle = OpenLayers.Util.extend({}, this.layer.style || this.layer.styleMap.createSymbolizer(null, options.vertexRenderIntent)); │ │ │ │ │ - this.virtualStyle.fillOpacity = .3; │ │ │ │ │ - this.virtualStyle.strokeOpacity = .3; │ │ │ │ │ - this.deleteCodes = [46, 68]; │ │ │ │ │ - this.mode = OpenLayers.Control.ModifyFeature.RESHAPE; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!OpenLayers.Util.isArray(this.deleteCodes)) { │ │ │ │ │ - this.deleteCodes = [this.deleteCodes] │ │ │ │ │ - } │ │ │ │ │ - var dragCallbacks = { │ │ │ │ │ - down: function(pixel) { │ │ │ │ │ - this.vertex = null; │ │ │ │ │ - var feature = this.layer.getFeatureFromEvent(this.handlers.drag.evt); │ │ │ │ │ - if (feature) { │ │ │ │ │ - this.dragStart(feature) │ │ │ │ │ - } else if (this.clickout) { │ │ │ │ │ - this._unselect = this.feature │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - move: function(pixel) { │ │ │ │ │ - delete this._unselect; │ │ │ │ │ - if (this.vertex) { │ │ │ │ │ - this.dragVertex(this.vertex, pixel) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - up: function() { │ │ │ │ │ - this.handlers.drag.stopDown = false; │ │ │ │ │ - if (this._unselect) { │ │ │ │ │ - this.unselectFeature(this._unselect); │ │ │ │ │ - delete this._unselect │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - done: function(pixel) { │ │ │ │ │ - if (this.vertex) { │ │ │ │ │ - this.dragComplete(this.vertex) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - var dragOptions = { │ │ │ │ │ - documentDrag: this.documentDrag, │ │ │ │ │ - stopDown: false │ │ │ │ │ - }; │ │ │ │ │ - var keyboardOptions = { │ │ │ │ │ - keydown: this.handleKeypress │ │ │ │ │ - }; │ │ │ │ │ - this.handlers = { │ │ │ │ │ - keyboard: new OpenLayers.Handler.Keyboard(this, keyboardOptions), │ │ │ │ │ - drag: new OpenLayers.Handler.Drag(this, dragCallbacks, dragOptions) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - removelayer: this.handleMapEvents, │ │ │ │ │ - changelayer: this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.layer = null; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, []) │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - this.moveLayerToTop(); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - removelayer: this.handleMapEvents, │ │ │ │ │ - changelayer: this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - return this.handlers.keyboard.activate() && this.handlers.drag.activate() && OpenLayers.Control.prototype.activate.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = false; │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.moveLayerBack(); │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - removelayer: this.handleMapEvents, │ │ │ │ │ - changelayer: this.handleMapEvents, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.layer.removeFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.layer.removeFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - this.handlers.drag.deactivate(); │ │ │ │ │ - this.handlers.keyboard.deactivate(); │ │ │ │ │ - var feature = this.feature; │ │ │ │ │ - if (feature && feature.geometry && feature.layer) { │ │ │ │ │ - this.unselectFeature(feature) │ │ │ │ │ - } │ │ │ │ │ - deactivated = true │ │ │ │ │ - } │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - beforeSelectFeature: function(feature) { │ │ │ │ │ - return this.layer.events.triggerEvent("beforefeaturemodified", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - selectFeature: function(feature) { │ │ │ │ │ - if (this.feature === feature || this.geometryTypes && OpenLayers.Util.indexOf(this.geometryTypes, feature.geometry.CLASS_NAME) == -1) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (this.beforeSelectFeature(feature) !== false) { │ │ │ │ │ - if (this.feature) { │ │ │ │ │ - this.unselectFeature(this.feature) │ │ │ │ │ - } │ │ │ │ │ - this.feature = feature; │ │ │ │ │ - this.layer.selectedFeatures.push(feature); │ │ │ │ │ - this.layer.drawFeature(feature, "select"); │ │ │ │ │ - this.modified = false; │ │ │ │ │ - this.resetVertices(); │ │ │ │ │ - this.onModificationStart(this.feature) │ │ │ │ │ - } │ │ │ │ │ - var modified = feature.modified; │ │ │ │ │ - if (feature.geometry && !(modified && modified.geometry)) { │ │ │ │ │ - this._originalGeometry = feature.geometry.clone() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - unselectFeature: function(feature) { │ │ │ │ │ - this.layer.removeFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - this.layer.destroyFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.virtualVertices = []; │ │ │ │ │ - if (this.dragHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.dragHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - delete this.dragHandle │ │ │ │ │ - } │ │ │ │ │ - if (this.radiusHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - delete this.radiusHandle │ │ │ │ │ - } │ │ │ │ │ - this.layer.drawFeature(this.feature, "default"); │ │ │ │ │ - this.feature = null; │ │ │ │ │ - OpenLayers.Util.removeItem(this.layer.selectedFeatures, feature); │ │ │ │ │ - this.onModificationEnd(feature); │ │ │ │ │ - this.layer.events.triggerEvent("afterfeaturemodified", { │ │ │ │ │ - feature: feature, │ │ │ │ │ - modified: this.modified │ │ │ │ │ - }); │ │ │ │ │ - this.modified = false │ │ │ │ │ - }, │ │ │ │ │ - dragStart: function(feature) { │ │ │ │ │ - var isPoint = feature.geometry.CLASS_NAME == "OpenLayers.Geometry.Point"; │ │ │ │ │ - if (!this.standalone && (!feature._sketch && isPoint || !feature._sketch)) { │ │ │ │ │ - if (this.toggle && this.feature === feature) { │ │ │ │ │ - this._unselect = feature │ │ │ │ │ - } │ │ │ │ │ - this.selectFeature(feature) │ │ │ │ │ - } │ │ │ │ │ - if (feature._sketch || isPoint) { │ │ │ │ │ - this.vertex = feature; │ │ │ │ │ - this.handlers.drag.stopDown = true │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - dragVertex: function(vertex, pixel) { │ │ │ │ │ - var pos = this.map.getLonLatFromViewPortPx(pixel); │ │ │ │ │ - var geom = vertex.geometry; │ │ │ │ │ - geom.move(pos.lon - geom.x, pos.lat - geom.y); │ │ │ │ │ - this.modified = true; │ │ │ │ │ - if (this.feature.geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - this.layer.events.triggerEvent("vertexmodified", { │ │ │ │ │ - vertex: vertex.geometry, │ │ │ │ │ - feature: this.feature, │ │ │ │ │ - pixel: pixel │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - if (vertex._index) { │ │ │ │ │ - vertex.geometry.parent.addComponent(vertex.geometry, vertex._index); │ │ │ │ │ - delete vertex._index; │ │ │ │ │ - OpenLayers.Util.removeItem(this.virtualVertices, vertex); │ │ │ │ │ - this.vertices.push(vertex) │ │ │ │ │ - } else if (vertex == this.dragHandle) { │ │ │ │ │ - this.layer.removeFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - if (this.radiusHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.radiusHandle = null │ │ │ │ │ - } │ │ │ │ │ - } else if (vertex !== this.radiusHandle) { │ │ │ │ │ - this.layer.events.triggerEvent("vertexmodified", { │ │ │ │ │ - vertex: vertex.geometry, │ │ │ │ │ - feature: this.feature, │ │ │ │ │ - pixel: pixel │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - if (this.virtualVertices.length > 0) { │ │ │ │ │ - this.layer.destroyFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.virtualVertices = [] │ │ │ │ │ - } │ │ │ │ │ - this.layer.drawFeature(this.feature, this.standalone ? undefined : "select") │ │ │ │ │ - } │ │ │ │ │ - this.layer.drawFeature(vertex) │ │ │ │ │ - }, │ │ │ │ │ - dragComplete: function(vertex) { │ │ │ │ │ - this.resetVertices(); │ │ │ │ │ - this.setFeatureState(); │ │ │ │ │ - this.onModification(this.feature); │ │ │ │ │ - this.layer.events.triggerEvent("featuremodified", { │ │ │ │ │ - feature: this.feature │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - setFeatureState: function() { │ │ │ │ │ - if (this.feature.state != OpenLayers.State.INSERT && this.feature.state != OpenLayers.State.DELETE) { │ │ │ │ │ - this.feature.state = OpenLayers.State.UPDATE; │ │ │ │ │ - if (this.modified && this._originalGeometry) { │ │ │ │ │ - var feature = this.feature; │ │ │ │ │ - feature.modified = OpenLayers.Util.extend(feature.modified, { │ │ │ │ │ - geometry: this._originalGeometry │ │ │ │ │ - }); │ │ │ │ │ - delete this._originalGeometry │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - resetVertices: function() { │ │ │ │ │ - if (this.vertices.length > 0) { │ │ │ │ │ - this.layer.removeFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.vertices = [] │ │ │ │ │ - } │ │ │ │ │ - if (this.virtualVertices.length > 0) { │ │ │ │ │ - this.layer.removeFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.virtualVertices = [] │ │ │ │ │ - } │ │ │ │ │ - if (this.dragHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.dragHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.dragHandle = null │ │ │ │ │ - } │ │ │ │ │ - if (this.radiusHandle) { │ │ │ │ │ - this.layer.destroyFeatures([this.radiusHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.radiusHandle = null │ │ │ │ │ - } │ │ │ │ │ - if (this.feature && this.feature.geometry.CLASS_NAME != "OpenLayers.Geometry.Point") { │ │ │ │ │ - if (this.mode & OpenLayers.Control.ModifyFeature.DRAG) { │ │ │ │ │ - this.collectDragHandle() │ │ │ │ │ - } │ │ │ │ │ - if (this.mode & (OpenLayers.Control.ModifyFeature.ROTATE | OpenLayers.Control.ModifyFeature.RESIZE)) { │ │ │ │ │ - this.collectRadiusHandle() │ │ │ │ │ - } │ │ │ │ │ - if (this.mode & OpenLayers.Control.ModifyFeature.RESHAPE) { │ │ │ │ │ - if (!(this.mode & OpenLayers.Control.ModifyFeature.RESIZE)) { │ │ │ │ │ - this.collectVertices() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - handleKeypress: function(evt) { │ │ │ │ │ - var code = evt.keyCode; │ │ │ │ │ - if (this.feature && OpenLayers.Util.indexOf(this.deleteCodes, code) != -1) { │ │ │ │ │ - var vertex = this.layer.getFeatureFromEvent(this.handlers.drag.evt); │ │ │ │ │ - if (vertex && OpenLayers.Util.indexOf(this.vertices, vertex) != -1 && !this.handlers.drag.dragging && vertex.geometry.parent) { │ │ │ │ │ - vertex.geometry.parent.removeComponent(vertex.geometry); │ │ │ │ │ - this.layer.events.triggerEvent("vertexremoved", { │ │ │ │ │ - vertex: vertex.geometry, │ │ │ │ │ - feature: this.feature, │ │ │ │ │ - pixel: evt.xy │ │ │ │ │ - }); │ │ │ │ │ - this.layer.drawFeature(this.feature, this.standalone ? undefined : "select"); │ │ │ │ │ - this.modified = true; │ │ │ │ │ - this.resetVertices(); │ │ │ │ │ - this.setFeatureState(); │ │ │ │ │ - this.onModification(this.feature); │ │ │ │ │ - this.layer.events.triggerEvent("featuremodified", { │ │ │ │ │ - feature: this.feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - collectVertices: function() { │ │ │ │ │ - this.vertices = []; │ │ │ │ │ - this.virtualVertices = []; │ │ │ │ │ - var control = this; │ │ │ │ │ - │ │ │ │ │ - function collectComponentVertices(geometry) { │ │ │ │ │ - var i, vertex, component, len; │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - vertex = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - vertex._sketch = true; │ │ │ │ │ - vertex.renderIntent = control.vertexRenderIntent; │ │ │ │ │ - control.vertices.push(vertex) │ │ │ │ │ - } else { │ │ │ │ │ - var numVert = geometry.components.length; │ │ │ │ │ - if (geometry.CLASS_NAME == "OpenLayers.Geometry.LinearRing") { │ │ │ │ │ - numVert -= 1 │ │ │ │ │ - } │ │ │ │ │ - for (i = 0; i < numVert; ++i) { │ │ │ │ │ - component = geometry.components[i]; │ │ │ │ │ - if (component.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - vertex = new OpenLayers.Feature.Vector(component); │ │ │ │ │ - vertex._sketch = true; │ │ │ │ │ - vertex.renderIntent = control.vertexRenderIntent; │ │ │ │ │ - control.vertices.push(vertex) │ │ │ │ │ - } else { │ │ │ │ │ - collectComponentVertices(component) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (control.createVertices && geometry.CLASS_NAME != "OpenLayers.Geometry.MultiPoint") { │ │ │ │ │ - for (i = 0, len = geometry.components.length; i < len - 1; ++i) { │ │ │ │ │ - var prevVertex = geometry.components[i]; │ │ │ │ │ - var nextVertex = geometry.components[i + 1]; │ │ │ │ │ - if (prevVertex.CLASS_NAME == "OpenLayers.Geometry.Point" && nextVertex.CLASS_NAME == "OpenLayers.Geometry.Point") { │ │ │ │ │ - var x = (prevVertex.x + nextVertex.x) / 2; │ │ │ │ │ - var y = (prevVertex.y + nextVertex.y) / 2; │ │ │ │ │ - var point = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Point(x, y), null, control.virtualStyle); │ │ │ │ │ - point.geometry.parent = geometry; │ │ │ │ │ - point._index = i + 1; │ │ │ │ │ - point._sketch = true; │ │ │ │ │ - control.virtualVertices.push(point) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - collectComponentVertices.call(this, this.feature.geometry); │ │ │ │ │ - this.layer.addFeatures(this.virtualVertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }); │ │ │ │ │ - this.layer.addFeatures(this.vertices, { │ │ │ │ │ - silent: true │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - collectDragHandle: function() { │ │ │ │ │ - var geometry = this.feature.geometry; │ │ │ │ │ - var center = geometry.getBounds().getCenterLonLat(); │ │ │ │ │ - var originGeometry = new OpenLayers.Geometry.Point(center.lon, center.lat); │ │ │ │ │ - var origin = new OpenLayers.Feature.Vector(originGeometry); │ │ │ │ │ - originGeometry.move = function(x, y) { │ │ │ │ │ - OpenLayers.Geometry.Point.prototype.move.call(this, x, y); │ │ │ │ │ - geometry.move(x, y) │ │ │ │ │ - }; │ │ │ │ │ - origin._sketch = true; │ │ │ │ │ - this.dragHandle = origin; │ │ │ │ │ - this.dragHandle.renderIntent = this.vertexRenderIntent; │ │ │ │ │ - this.layer.addFeatures([this.dragHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - collectRadiusHandle: function() { │ │ │ │ │ - var geometry = this.feature.geometry; │ │ │ │ │ - var bounds = geometry.getBounds(); │ │ │ │ │ - var center = bounds.getCenterLonLat(); │ │ │ │ │ - var originGeometry = new OpenLayers.Geometry.Point(center.lon, center.lat); │ │ │ │ │ - var radiusGeometry = new OpenLayers.Geometry.Point(bounds.right, bounds.bottom); │ │ │ │ │ - var radius = new OpenLayers.Feature.Vector(radiusGeometry); │ │ │ │ │ - var resize = this.mode & OpenLayers.Control.ModifyFeature.RESIZE; │ │ │ │ │ - var reshape = this.mode & OpenLayers.Control.ModifyFeature.RESHAPE; │ │ │ │ │ - var rotate = this.mode & OpenLayers.Control.ModifyFeature.ROTATE; │ │ │ │ │ - radiusGeometry.move = function(x, y) { │ │ │ │ │ - OpenLayers.Geometry.Point.prototype.move.call(this, x, y); │ │ │ │ │ - var dx1 = this.x - originGeometry.x; │ │ │ │ │ - var dy1 = this.y - originGeometry.y; │ │ │ │ │ - var dx0 = dx1 - x; │ │ │ │ │ - var dy0 = dy1 - y; │ │ │ │ │ - if (rotate) { │ │ │ │ │ - var a0 = Math.atan2(dy0, dx0); │ │ │ │ │ - var a1 = Math.atan2(dy1, dx1); │ │ │ │ │ - var angle = a1 - a0; │ │ │ │ │ - angle *= 180 / Math.PI; │ │ │ │ │ - geometry.rotate(angle, originGeometry) │ │ │ │ │ - } │ │ │ │ │ - if (resize) { │ │ │ │ │ - var scale, ratio; │ │ │ │ │ - if (reshape) { │ │ │ │ │ - scale = dy1 / dy0; │ │ │ │ │ - ratio = dx1 / dx0 / scale │ │ │ │ │ - } else { │ │ │ │ │ - var l0 = Math.sqrt(dx0 * dx0 + dy0 * dy0); │ │ │ │ │ - var l1 = Math.sqrt(dx1 * dx1 + dy1 * dy1); │ │ │ │ │ - scale = l1 / l0 │ │ │ │ │ - } │ │ │ │ │ - geometry.resize(scale, originGeometry, ratio) │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - radius._sketch = true; │ │ │ │ │ - this.radiusHandle = radius; │ │ │ │ │ - this.radiusHandle.renderIntent = this.vertexRenderIntent; │ │ │ │ │ - this.layer.addFeatures([this.radiusHandle], { │ │ │ │ │ - silent: true │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - this.handlers.drag.setMap(map); │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - handleMapEvents: function(evt) { │ │ │ │ │ - if (evt.type == "removelayer" || evt.property == "order") { │ │ │ │ │ - this.moveLayerToTop() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - moveLayerToTop: function() { │ │ │ │ │ - var index = Math.max(this.map.Z_INDEX_BASE["Feature"] - 1, this.layer.getZIndex()) + 1; │ │ │ │ │ - this.layer.setZIndex(index) │ │ │ │ │ - }, │ │ │ │ │ - moveLayerBack: function() { │ │ │ │ │ - var index = this.layer.getZIndex() - 1; │ │ │ │ │ - if (index >= this.map.Z_INDEX_BASE["Feature"]) { │ │ │ │ │ - this.layer.setZIndex(index) │ │ │ │ │ - } else { │ │ │ │ │ - this.map.setLayerZIndex(this.layer, this.map.getLayerIndex(this.layer)) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ModifyFeature" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.ModifyFeature.RESHAPE = 1; │ │ │ │ │ -OpenLayers.Control.ModifyFeature.RESIZE = 2; │ │ │ │ │ -OpenLayers.Control.ModifyFeature.ROTATE = 4; │ │ │ │ │ -OpenLayers.Control.ModifyFeature.DRAG = 8; │ │ │ │ │ -OpenLayers.Control.DrawFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - layer: null, │ │ │ │ │ - callbacks: null, │ │ │ │ │ - multi: false, │ │ │ │ │ - featureAdded: function() {}, │ │ │ │ │ - initialize: function(layer, handler, options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.callbacks = OpenLayers.Util.extend({ │ │ │ │ │ - done: this.drawFeature, │ │ │ │ │ - modify: function(vertex, feature) { │ │ │ │ │ - this.layer.events.triggerEvent("sketchmodified", { │ │ │ │ │ - vertex: vertex, │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - create: function(vertex, feature) { │ │ │ │ │ - this.layer.events.triggerEvent("sketchstarted", { │ │ │ │ │ - vertex: vertex, │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, this.callbacks); │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - this.handlerOptions = this.handlerOptions || {}; │ │ │ │ │ - this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults(this.handlerOptions.layerOptions, { │ │ │ │ │ - renderers: layer.renderers, │ │ │ │ │ - rendererOptions: layer.rendererOptions │ │ │ │ │ - }); │ │ │ │ │ - if (!("multi" in this.handlerOptions)) { │ │ │ │ │ - this.handlerOptions.multi = this.multi │ │ │ │ │ - } │ │ │ │ │ - var sketchStyle = this.layer.styleMap && this.layer.styleMap.styles.temporary; │ │ │ │ │ - if (sketchStyle) { │ │ │ │ │ - this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults(this.handlerOptions.layerOptions, { │ │ │ │ │ - styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ - default: sketchStyle │ │ │ │ │ - }) │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.handler = new handler(this, this.callbacks, this.handlerOptions) │ │ │ │ │ - }, │ │ │ │ │ - drawFeature: function(geometry) { │ │ │ │ │ - var feature = new OpenLayers.Feature.Vector(geometry); │ │ │ │ │ - var proceed = this.layer.events.triggerEvent("sketchcomplete", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - if (proceed !== false) { │ │ │ │ │ - feature.state = OpenLayers.State.INSERT; │ │ │ │ │ - this.layer.addFeatures([feature]); │ │ │ │ │ - this.featureAdded(feature); │ │ │ │ │ - this.events.triggerEvent("featureadded", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - insertXY: function(x, y) { │ │ │ │ │ - if (this.handler && this.handler.line) { │ │ │ │ │ - this.handler.insertXY(x, y) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - insertDeltaXY: function(dx, dy) { │ │ │ │ │ - if (this.handler && this.handler.line) { │ │ │ │ │ - this.handler.insertDeltaXY(dx, dy) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - insertDirectionLength: function(direction, length) { │ │ │ │ │ - if (this.handler && this.handler.line) { │ │ │ │ │ - this.handler.insertDirectionLength(direction, length) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - insertDeflectionLength: function(deflection, length) { │ │ │ │ │ - if (this.handler && this.handler.line) { │ │ │ │ │ - this.handler.insertDeflectionLength(deflection, length) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - undo: function() { │ │ │ │ │ - return this.handler.undo && this.handler.undo() │ │ │ │ │ - }, │ │ │ │ │ - redo: function() { │ │ │ │ │ - return this.handler.redo && this.handler.redo() │ │ │ │ │ - }, │ │ │ │ │ - finishSketch: function() { │ │ │ │ │ - this.handler.finishGeometry() │ │ │ │ │ - }, │ │ │ │ │ - cancel: function() { │ │ │ │ │ - this.handler.cancel() │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.DrawFeature" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.ArgParser = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - center: null, │ │ │ │ │ - zoom: null, │ │ │ │ │ - layers: null, │ │ │ │ │ - displayProjection: null, │ │ │ │ │ - getParameters: function(url) { │ │ │ │ │ - url = url || window.location.href; │ │ │ │ │ - var parameters = OpenLayers.Util.getParameters(url); │ │ │ │ │ - var index = url.indexOf("#"); │ │ │ │ │ - if (index > 0) { │ │ │ │ │ - url = "?" + url.substring(index + 1, url.length); │ │ │ │ │ - OpenLayers.Util.extend(parameters, OpenLayers.Util.getParameters(url)) │ │ │ │ │ - } │ │ │ │ │ - return parameters │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - for (var i = 0, len = this.map.controls.length; i < len; i++) { │ │ │ │ │ - var control = this.map.controls[i]; │ │ │ │ │ - if (control != this && control.CLASS_NAME == "OpenLayers.Control.ArgParser") { │ │ │ │ │ - if (control.displayProjection != this.displayProjection) { │ │ │ │ │ - this.displayProjection = control.displayProjection │ │ │ │ │ - } │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (i == this.map.controls.length) { │ │ │ │ │ - var args = this.getParameters(); │ │ │ │ │ - if (args.layers) { │ │ │ │ │ - this.layers = args.layers; │ │ │ │ │ - this.map.events.register("addlayer", this, this.configureLayers); │ │ │ │ │ - this.configureLayers() │ │ │ │ │ - } │ │ │ │ │ - if (args.lat && args.lon) { │ │ │ │ │ - this.center = new OpenLayers.LonLat(parseFloat(args.lon), parseFloat(args.lat)); │ │ │ │ │ - if (args.zoom) { │ │ │ │ │ - this.zoom = parseFloat(args.zoom) │ │ │ │ │ - } │ │ │ │ │ - this.map.events.register("changebaselayer", this, this.setCenter); │ │ │ │ │ - this.setCenter() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setCenter: function() { │ │ │ │ │ - if (this.map.baseLayer) { │ │ │ │ │ - this.map.events.unregister("changebaselayer", this, this.setCenter); │ │ │ │ │ - if (this.displayProjection) { │ │ │ │ │ - this.center.transform(this.displayProjection, this.map.getProjectionObject()) │ │ │ │ │ - } │ │ │ │ │ - this.map.setCenter(this.center, this.zoom) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - configureLayers: function() { │ │ │ │ │ - if (this.layers.length == this.map.layers.length) { │ │ │ │ │ - this.map.events.unregister("addlayer", this, this.configureLayers); │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - var c = this.layers.charAt(i); │ │ │ │ │ - if (c == "B") { │ │ │ │ │ - this.map.setBaseLayer(layer) │ │ │ │ │ - } else if (c == "T" || c == "F") { │ │ │ │ │ - layer.setVisibility(c == "T") │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ArgParser" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.Geolocate = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - geolocation: null, │ │ │ │ │ - available: "geolocation" in navigator, │ │ │ │ │ - bind: true, │ │ │ │ │ - watch: false, │ │ │ │ │ - geolocationOptions: null, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (this.available && !this.geolocation) { │ │ │ │ │ - this.geolocation = navigator.geolocation │ │ │ │ │ - } │ │ │ │ │ - if (!this.geolocation) { │ │ │ │ │ - this.events.triggerEvent("locationuncapable"); │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - if (this.watch) { │ │ │ │ │ - this.watchId = this.geolocation.watchPosition(OpenLayers.Function.bind(this.geolocate, this), OpenLayers.Function.bind(this.failure, this), this.geolocationOptions) │ │ │ │ │ - } else { │ │ │ │ │ - this.getCurrentLocation() │ │ │ │ │ - } │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active && this.watchId !== null) { │ │ │ │ │ - this.geolocation.clearWatch(this.watchId) │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Control.prototype.deactivate.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - geolocate: function(position) { │ │ │ │ │ - var center = new OpenLayers.LonLat(position.coords.longitude, position.coords.latitude).transform(new OpenLayers.Projection("EPSG:4326"), this.map.getProjectionObject()); │ │ │ │ │ - if (this.bind) { │ │ │ │ │ - this.map.setCenter(center) │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("locationupdated", { │ │ │ │ │ - position: position, │ │ │ │ │ - point: new OpenLayers.Geometry.Point(center.lon, center.lat) │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - getCurrentLocation: function() { │ │ │ │ │ - if (!this.active || this.watch) { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - this.geolocation.getCurrentPosition(OpenLayers.Function.bind(this.geolocate, this), OpenLayers.Function.bind(this.failure, this), this.geolocationOptions); │ │ │ │ │ - return true │ │ │ │ │ - }, │ │ │ │ │ - failure: function(error) { │ │ │ │ │ - this.events.triggerEvent("locationfailed", { │ │ │ │ │ - error: error │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Geolocate" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.CacheWrite = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - layers: null, │ │ │ │ │ - imageFormat: "image/png", │ │ │ │ │ - quotaRegEx: /quota/i, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - var i, layers = this.layers || map.layers; │ │ │ │ │ - for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - this.addLayer({ │ │ │ │ │ - layer: layers[i] │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - if (!this.layers) { │ │ │ │ │ - map.events.on({ │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - removeLayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - addLayer: function(evt) { │ │ │ │ │ - evt.layer.events.on({ │ │ │ │ │ - tileloadstart: this.makeSameOrigin, │ │ │ │ │ - tileloaded: this.onTileLoaded, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - removeLayer: function(evt) { │ │ │ │ │ - evt.layer.events.un({ │ │ │ │ │ - tileloadstart: this.makeSameOrigin, │ │ │ │ │ - tileloaded: this.onTileLoaded, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - makeSameOrigin: function(evt) { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - var tile = evt.tile; │ │ │ │ │ - if (tile instanceof OpenLayers.Tile.Image && !tile.crossOriginKeyword && tile.url.substr(0, 5) !== "data:") { │ │ │ │ │ - var sameOriginUrl = OpenLayers.Request.makeSameOrigin(tile.url, OpenLayers.ProxyHost); │ │ │ │ │ - OpenLayers.Control.CacheWrite.urlMap[sameOriginUrl] = tile.url; │ │ │ │ │ - tile.url = sameOriginUrl │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - onTileLoaded: function(evt) { │ │ │ │ │ - if (this.active && !evt.aborted && evt.tile instanceof OpenLayers.Tile.Image && evt.tile.url.substr(0, 5) !== "data:") { │ │ │ │ │ - this.cache({ │ │ │ │ │ - tile: evt.tile │ │ │ │ │ - }); │ │ │ │ │ - delete OpenLayers.Control.CacheWrite.urlMap[evt.tile.url] │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - cache: function(obj) { │ │ │ │ │ - if (window.localStorage) { │ │ │ │ │ - var tile = obj.tile; │ │ │ │ │ - try { │ │ │ │ │ - var canvasContext = tile.getCanvasContext(); │ │ │ │ │ - if (canvasContext) { │ │ │ │ │ - var urlMap = OpenLayers.Control.CacheWrite.urlMap; │ │ │ │ │ - var url = urlMap[tile.url] || tile.url; │ │ │ │ │ - window.localStorage.setItem("olCache_" + url, canvasContext.canvas.toDataURL(this.imageFormat)) │ │ │ │ │ - } │ │ │ │ │ - } catch (e) { │ │ │ │ │ - var reason = e.name || e.message; │ │ │ │ │ - if (reason && this.quotaRegEx.test(reason)) { │ │ │ │ │ - this.events.triggerEvent("cachefull", { │ │ │ │ │ - tile: tile │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Console.error(e.toString()) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.layers || this.map) { │ │ │ │ │ - var i, layers = this.layers || this.map.layers; │ │ │ │ │ - for (i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ - this.removeLayer({ │ │ │ │ │ - layer: layers[i] │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - addlayer: this.addLayer, │ │ │ │ │ - removeLayer: this.removeLayer, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.CacheWrite" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.CacheWrite.clearCache = function() { │ │ │ │ │ - if (!window.localStorage) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var i, key; │ │ │ │ │ - for (i = window.localStorage.length - 1; i >= 0; --i) { │ │ │ │ │ - key = window.localStorage.key(i); │ │ │ │ │ - if (key.substr(0, 8) === "olCache_") { │ │ │ │ │ - window.localStorage.removeItem(key) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Control.CacheWrite.urlMap = {}; │ │ │ │ │ -OpenLayers.Control.Button = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - type: OpenLayers.Control.TYPE_BUTTON, │ │ │ │ │ - trigger: function() {}, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Button" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.Pan = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ - slideFactor: 50, │ │ │ │ │ - slideRatio: null, │ │ │ │ │ - direction: null, │ │ │ │ │ - initialize: function(direction, options) { │ │ │ │ │ - this.direction = direction; │ │ │ │ │ - this.CLASS_NAME += this.direction; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]) │ │ │ │ │ - }, │ │ │ │ │ - trigger: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - var getSlideFactor = OpenLayers.Function.bind(function(dim) { │ │ │ │ │ - return this.slideRatio ? this.map.getSize()[dim] * this.slideRatio : this.slideFactor │ │ │ │ │ - }, this); │ │ │ │ │ - switch (this.direction) { │ │ │ │ │ - case OpenLayers.Control.Pan.NORTH: │ │ │ │ │ - this.map.pan(0, -getSlideFactor("h")); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Control.Pan.SOUTH: │ │ │ │ │ - this.map.pan(0, getSlideFactor("h")); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Control.Pan.WEST: │ │ │ │ │ - this.map.pan(-getSlideFactor("w"), 0); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Control.Pan.EAST: │ │ │ │ │ - this.map.pan(getSlideFactor("w"), 0); │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Pan" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.Pan.NORTH = "North"; │ │ │ │ │ -OpenLayers.Control.Pan.SOUTH = "South"; │ │ │ │ │ -OpenLayers.Control.Pan.EAST = "East"; │ │ │ │ │ -OpenLayers.Control.Pan.WEST = "West"; │ │ │ │ │ -OpenLayers.Control.PanPanel = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ - slideFactor: 50, │ │ │ │ │ - slideRatio: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ - var options = { │ │ │ │ │ - slideFactor: this.slideFactor, │ │ │ │ │ - slideRatio: this.slideRatio │ │ │ │ │ - }; │ │ │ │ │ - this.addControls([new OpenLayers.Control.Pan(OpenLayers.Control.Pan.NORTH, options), new OpenLayers.Control.Pan(OpenLayers.Control.Pan.SOUTH, options), new OpenLayers.Control.Pan(OpenLayers.Control.Pan.EAST, options), new OpenLayers.Control.Pan(OpenLayers.Control.Pan.WEST, options)]) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.PanPanel" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.ZoomIn = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ - trigger: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.zoomIn() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomIn" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.ZoomOut = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ - trigger: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.zoomOut() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomOut" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.ZoomToMaxExtent = OpenLayers.Class(OpenLayers.Control.Button, { │ │ │ │ │ - trigger: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.zoomToMaxExtent() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomToMaxExtent" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.ZoomPanel = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.addControls([new OpenLayers.Control.ZoomIn, new OpenLayers.Control.ZoomToMaxExtent, new OpenLayers.Control.ZoomOut]) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ZoomPanel" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.GetFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - protocol: null, │ │ │ │ │ - multipleKey: null, │ │ │ │ │ - toggleKey: null, │ │ │ │ │ - modifiers: null, │ │ │ │ │ - multiple: false, │ │ │ │ │ - click: true, │ │ │ │ │ - single: true, │ │ │ │ │ - clickout: true, │ │ │ │ │ - toggle: false, │ │ │ │ │ - clickTolerance: 5, │ │ │ │ │ - hover: false, │ │ │ │ │ - box: false, │ │ │ │ │ - maxFeatures: 10, │ │ │ │ │ - features: null, │ │ │ │ │ - hoverFeature: null, │ │ │ │ │ - handlers: null, │ │ │ │ │ - hoverResponse: null, │ │ │ │ │ - filterType: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.features = {}; │ │ │ │ │ - this.handlers = {}; │ │ │ │ │ - if (this.click) { │ │ │ │ │ - this.handlers.click = new OpenLayers.Handler.Click(this, { │ │ │ │ │ - click: this.selectClick │ │ │ │ │ - }, this.handlerOptions.click || {}) │ │ │ │ │ - } │ │ │ │ │ - if (this.box) { │ │ │ │ │ - this.handlers.box = new OpenLayers.Handler.Box(this, { │ │ │ │ │ - done: this.selectBox │ │ │ │ │ - }, OpenLayers.Util.extend(this.handlerOptions.box, { │ │ │ │ │ - boxDivClassName: "olHandlerBoxSelectFeature" │ │ │ │ │ - })) │ │ │ │ │ - } │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - this.handlers.hover = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ - move: this.cancelHover, │ │ │ │ │ - pause: this.selectHover │ │ │ │ │ - }, OpenLayers.Util.extend(this.handlerOptions.hover, { │ │ │ │ │ - delay: 250, │ │ │ │ │ - pixelTolerance: 2 │ │ │ │ │ - })) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (!this.active) { │ │ │ │ │ - for (var i in this.handlers) { │ │ │ │ │ - this.handlers[i].activate() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Control.prototype.activate.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - for (var i in this.handlers) { │ │ │ │ │ - this.handlers[i].deactivate() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return OpenLayers.Control.prototype.deactivate.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - selectClick: function(evt) { │ │ │ │ │ - var bounds = this.pixelToBounds(evt.xy); │ │ │ │ │ - this.setModifiers(evt); │ │ │ │ │ - this.request(bounds, { │ │ │ │ │ - single: this.single │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - selectBox: function(position) { │ │ │ │ │ - var bounds; │ │ │ │ │ - if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ - var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.left, │ │ │ │ │ - y: position.bottom │ │ │ │ │ - }); │ │ │ │ │ - var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ - x: position.right, │ │ │ │ │ - y: position.top │ │ │ │ │ - }); │ │ │ │ │ - bounds = new OpenLayers.Bounds(minXY.lon, minXY.lat, maxXY.lon, maxXY.lat) │ │ │ │ │ - } else { │ │ │ │ │ - if (this.click) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - bounds = this.pixelToBounds(position) │ │ │ │ │ - } │ │ │ │ │ - this.setModifiers(this.handlers.box.dragHandler.evt); │ │ │ │ │ - this.request(bounds) │ │ │ │ │ - }, │ │ │ │ │ - selectHover: function(evt) { │ │ │ │ │ - var bounds = this.pixelToBounds(evt.xy); │ │ │ │ │ - this.request(bounds, { │ │ │ │ │ - single: true, │ │ │ │ │ - hover: true │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - cancelHover: function() { │ │ │ │ │ - if (this.hoverResponse) { │ │ │ │ │ - this.protocol.abort(this.hoverResponse); │ │ │ │ │ - this.hoverResponse = null; │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait") │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - request: function(bounds, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: this.filterType, │ │ │ │ │ - value: bounds │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - var response = this.protocol.read({ │ │ │ │ │ - maxFeatures: options.single == true ? this.maxFeatures : undefined, │ │ │ │ │ - filter: filter, │ │ │ │ │ - callback: function(result) { │ │ │ │ │ - if (result.success()) { │ │ │ │ │ - if (result.features.length) { │ │ │ │ │ - if (options.single == true) { │ │ │ │ │ - this.selectBestFeature(result.features, bounds.getCenterLonLat(), options) │ │ │ │ │ - } else { │ │ │ │ │ - this.select(result.features) │ │ │ │ │ - } │ │ │ │ │ - } else if (options.hover) { │ │ │ │ │ - this.hoverSelect() │ │ │ │ │ - } else { │ │ │ │ │ - this.events.triggerEvent("clickout"); │ │ │ │ │ - if (this.clickout) { │ │ │ │ │ - this.unselectAll() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait") │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - if (options.hover == true) { │ │ │ │ │ - this.hoverResponse = response │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - selectBestFeature: function(features, clickPosition, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - if (features.length) { │ │ │ │ │ - var point = new OpenLayers.Geometry.Point(clickPosition.lon, clickPosition.lat); │ │ │ │ │ - var feature, resultFeature, dist; │ │ │ │ │ - var minDist = Number.MAX_VALUE; │ │ │ │ │ - for (var i = 0; i < features.length; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - dist = point.distanceTo(feature.geometry, { │ │ │ │ │ - edge: false │ │ │ │ │ - }); │ │ │ │ │ - if (dist < minDist) { │ │ │ │ │ - minDist = dist; │ │ │ │ │ - resultFeature = feature; │ │ │ │ │ - if (minDist == 0) { │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (options.hover == true) { │ │ │ │ │ - this.hoverSelect(resultFeature) │ │ │ │ │ - } else { │ │ │ │ │ - this.select(resultFeature || features) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setModifiers: function(evt) { │ │ │ │ │ - this.modifiers = { │ │ │ │ │ - multiple: this.multiple || this.multipleKey && evt[this.multipleKey], │ │ │ │ │ - toggle: this.toggle || this.toggleKey && evt[this.toggleKey] │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - select: function(features) { │ │ │ │ │ - if (!this.modifiers.multiple && !this.modifiers.toggle) { │ │ │ │ │ - this.unselectAll() │ │ │ │ │ - } │ │ │ │ │ - if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ - features = [features] │ │ │ │ │ - } │ │ │ │ │ - var cont = this.events.triggerEvent("beforefeaturesselected", { │ │ │ │ │ - features: features │ │ │ │ │ - }); │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - var selectedFeatures = []; │ │ │ │ │ - var feature; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (this.features[feature.fid || feature.id]) { │ │ │ │ │ - if (this.modifiers.toggle) { │ │ │ │ │ - this.unselect(this.features[feature.fid || feature.id]) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - cont = this.events.triggerEvent("beforefeatureselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - if (cont !== false) { │ │ │ │ │ - this.features[feature.fid || feature.id] = feature; │ │ │ │ │ - selectedFeatures.push(feature); │ │ │ │ │ - this.events.triggerEvent("featureselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.events.triggerEvent("featuresselected", { │ │ │ │ │ - features: selectedFeatures │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - hoverSelect: function(feature) { │ │ │ │ │ - var fid = feature ? feature.fid || feature.id : null; │ │ │ │ │ - var hfid = this.hoverFeature ? this.hoverFeature.fid || this.hoverFeature.id : null; │ │ │ │ │ - if (hfid && hfid != fid) { │ │ │ │ │ - this.events.triggerEvent("outfeature", { │ │ │ │ │ - feature: this.hoverFeature │ │ │ │ │ - }); │ │ │ │ │ - this.hoverFeature = null │ │ │ │ │ - } │ │ │ │ │ - if (fid && fid != hfid) { │ │ │ │ │ - this.events.triggerEvent("hoverfeature", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }); │ │ │ │ │ - this.hoverFeature = feature │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - unselect: function(feature) { │ │ │ │ │ - delete this.features[feature.fid || feature.id]; │ │ │ │ │ - this.events.triggerEvent("featureunselected", { │ │ │ │ │ - feature: feature │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - unselectAll: function() { │ │ │ │ │ - for (var fid in this.features) { │ │ │ │ │ - this.unselect(this.features[fid]) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - for (var i in this.handlers) { │ │ │ │ │ - this.handlers[i].setMap(map) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - pixelToBounds: function(pixel) { │ │ │ │ │ - var llPx = pixel.add(-this.clickTolerance / 2, this.clickTolerance / 2); │ │ │ │ │ - var urPx = pixel.add(this.clickTolerance / 2, -this.clickTolerance / 2); │ │ │ │ │ - var ll = this.map.getLonLatFromPixel(llPx); │ │ │ │ │ - var ur = this.map.getLonLatFromPixel(urPx); │ │ │ │ │ - return new OpenLayers.Bounds(ll.lon, ll.lat, ur.lon, ur.lat) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.GetFeature" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.Snapping = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - DEFAULTS: { │ │ │ │ │ - tolerance: 10, │ │ │ │ │ - node: true, │ │ │ │ │ - edge: true, │ │ │ │ │ - vertex: true │ │ │ │ │ - }, │ │ │ │ │ - greedy: true, │ │ │ │ │ - precedence: ["node", "vertex", "edge"], │ │ │ │ │ - resolution: null, │ │ │ │ │ - geoToleranceCache: null, │ │ │ │ │ - layer: null, │ │ │ │ │ - feature: null, │ │ │ │ │ - point: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.options = options || {}; │ │ │ │ │ - if (this.options.layer) { │ │ │ │ │ - this.setLayer(this.options.layer) │ │ │ │ │ - } │ │ │ │ │ - var defaults = OpenLayers.Util.extend({}, this.options.defaults); │ │ │ │ │ - this.defaults = OpenLayers.Util.applyDefaults(defaults, this.DEFAULTS); │ │ │ │ │ - this.setTargets(this.options.targets); │ │ │ │ │ - if (this.targets.length === 0 && this.layer) { │ │ │ │ │ - this.addTargetLayer(this.layer) │ │ │ │ │ - } │ │ │ │ │ - this.geoToleranceCache = {} │ │ │ │ │ - }, │ │ │ │ │ - setLayer: function(layer) { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.layer = layer; │ │ │ │ │ - this.activate() │ │ │ │ │ - } else { │ │ │ │ │ - this.layer = layer │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - setTargets: function(targets) { │ │ │ │ │ - this.targets = []; │ │ │ │ │ - if (targets && targets.length) { │ │ │ │ │ - var target; │ │ │ │ │ - for (var i = 0, len = targets.length; i < len; ++i) { │ │ │ │ │ - target = targets[i]; │ │ │ │ │ - if (target instanceof OpenLayers.Layer.Vector) { │ │ │ │ │ - this.addTargetLayer(target) │ │ │ │ │ - } else { │ │ │ │ │ - this.addTarget(target) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - addTargetLayer: function(layer) { │ │ │ │ │ - this.addTarget({ │ │ │ │ │ - layer: layer │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - addTarget: function(target) { │ │ │ │ │ - target = OpenLayers.Util.applyDefaults(target, this.defaults); │ │ │ │ │ - target.nodeTolerance = target.nodeTolerance || target.tolerance; │ │ │ │ │ - target.vertexTolerance = target.vertexTolerance || target.tolerance; │ │ │ │ │ - target.edgeTolerance = target.edgeTolerance || target.tolerance; │ │ │ │ │ - this.targets.push(target) │ │ │ │ │ - }, │ │ │ │ │ - removeTargetLayer: function(layer) { │ │ │ │ │ - var target; │ │ │ │ │ - for (var i = this.targets.length - 1; i >= 0; --i) { │ │ │ │ │ - target = this.targets[i]; │ │ │ │ │ - if (target.layer === layer) { │ │ │ │ │ - this.removeTarget(target) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - removeTarget: function(target) { │ │ │ │ │ - return OpenLayers.Util.removeItem(this.targets, target) │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - var activated = OpenLayers.Control.prototype.activate.call(this); │ │ │ │ │ - if (activated) { │ │ │ │ │ - if (this.layer && this.layer.events) { │ │ │ │ │ - this.layer.events.on({ │ │ │ │ │ - sketchstarted: this.onSketchModified, │ │ │ │ │ - sketchmodified: this.onSketchModified, │ │ │ │ │ - vertexmodified: this.onVertexModified, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return activated │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Control.prototype.deactivate.call(this); │ │ │ │ │ - if (deactivated) { │ │ │ │ │ - if (this.layer && this.layer.events) { │ │ │ │ │ - this.layer.events.un({ │ │ │ │ │ - sketchstarted: this.onSketchModified, │ │ │ │ │ - sketchmodified: this.onSketchModified, │ │ │ │ │ - vertexmodified: this.onVertexModified, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.feature = null; │ │ │ │ │ - this.point = null; │ │ │ │ │ - return deactivated │ │ │ │ │ - }, │ │ │ │ │ - onSketchModified: function(event) { │ │ │ │ │ - this.feature = event.feature; │ │ │ │ │ - this.considerSnapping(event.vertex, event.vertex) │ │ │ │ │ - }, │ │ │ │ │ - onVertexModified: function(event) { │ │ │ │ │ - this.feature = event.feature; │ │ │ │ │ - var loc = this.layer.map.getLonLatFromViewPortPx(event.pixel); │ │ │ │ │ - this.considerSnapping(event.vertex, new OpenLayers.Geometry.Point(loc.lon, loc.lat)) │ │ │ │ │ - }, │ │ │ │ │ - considerSnapping: function(point, loc) { │ │ │ │ │ - var best = { │ │ │ │ │ - rank: Number.POSITIVE_INFINITY, │ │ │ │ │ - dist: Number.POSITIVE_INFINITY, │ │ │ │ │ - x: null, │ │ │ │ │ - y: null │ │ │ │ │ - }; │ │ │ │ │ - var snapped = false; │ │ │ │ │ - var result, target; │ │ │ │ │ - for (var i = 0, len = this.targets.length; i < len; ++i) { │ │ │ │ │ - target = this.targets[i]; │ │ │ │ │ - result = this.testTarget(target, loc); │ │ │ │ │ - if (result) { │ │ │ │ │ - if (this.greedy) { │ │ │ │ │ - best = result; │ │ │ │ │ - best.target = target; │ │ │ │ │ - snapped = true; │ │ │ │ │ - break │ │ │ │ │ - } else { │ │ │ │ │ - if (result.rank < best.rank || result.rank === best.rank && result.dist < best.dist) { │ │ │ │ │ - best = result; │ │ │ │ │ - best.target = target; │ │ │ │ │ - snapped = true │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (snapped) { │ │ │ │ │ - var proceed = this.events.triggerEvent("beforesnap", { │ │ │ │ │ - point: point, │ │ │ │ │ - x: best.x, │ │ │ │ │ - y: best.y, │ │ │ │ │ - distance: best.dist, │ │ │ │ │ - layer: best.target.layer, │ │ │ │ │ - snapType: this.precedence[best.rank] │ │ │ │ │ - }); │ │ │ │ │ - if (proceed !== false) { │ │ │ │ │ - point.x = best.x; │ │ │ │ │ - point.y = best.y; │ │ │ │ │ - this.point = point; │ │ │ │ │ - this.events.triggerEvent("snap", { │ │ │ │ │ - point: point, │ │ │ │ │ - snapType: this.precedence[best.rank], │ │ │ │ │ - layer: best.target.layer, │ │ │ │ │ - distance: best.dist │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - snapped = false │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.point && !snapped) { │ │ │ │ │ - point.x = loc.x; │ │ │ │ │ - point.y = loc.y; │ │ │ │ │ - this.point = null; │ │ │ │ │ - this.events.triggerEvent("unsnap", { │ │ │ │ │ - point: point │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - testTarget: function(target, loc) { │ │ │ │ │ - var resolution = this.layer.map.getResolution(); │ │ │ │ │ - if ("minResolution" in target) { │ │ │ │ │ - if (resolution < target.minResolution) { │ │ │ │ │ - return null │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if ("maxResolution" in target) { │ │ │ │ │ - if (resolution >= target.maxResolution) { │ │ │ │ │ - return null │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var tolerance = { │ │ │ │ │ - node: this.getGeoTolerance(target.nodeTolerance, resolution), │ │ │ │ │ - vertex: this.getGeoTolerance(target.vertexTolerance, resolution), │ │ │ │ │ - edge: this.getGeoTolerance(target.edgeTolerance, resolution) │ │ │ │ │ - }; │ │ │ │ │ - var maxTolerance = Math.max(tolerance.node, tolerance.vertex, tolerance.edge); │ │ │ │ │ - var result = { │ │ │ │ │ - rank: Number.POSITIVE_INFINITY, │ │ │ │ │ - dist: Number.POSITIVE_INFINITY │ │ │ │ │ - }; │ │ │ │ │ - var eligible = false; │ │ │ │ │ - var features = target.layer.features; │ │ │ │ │ - var feature, type, vertices, vertex, closest, dist, found; │ │ │ │ │ - var numTypes = this.precedence.length; │ │ │ │ │ - var ll = new OpenLayers.LonLat(loc.x, loc.y); │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - if (feature !== this.feature && !feature._sketch && feature.state !== OpenLayers.State.DELETE && (!target.filter || target.filter.evaluate(feature))) { │ │ │ │ │ - if (feature.atPoint(ll, maxTolerance, maxTolerance)) { │ │ │ │ │ - for (var j = 0, stop = Math.min(result.rank + 1, numTypes); j < stop; ++j) { │ │ │ │ │ - type = this.precedence[j]; │ │ │ │ │ - if (target[type]) { │ │ │ │ │ - if (type === "edge") { │ │ │ │ │ - closest = feature.geometry.distanceTo(loc, { │ │ │ │ │ - details: true │ │ │ │ │ - }); │ │ │ │ │ - dist = closest.distance; │ │ │ │ │ - if (dist <= tolerance[type] && dist < result.dist) { │ │ │ │ │ - result = { │ │ │ │ │ - rank: j, │ │ │ │ │ - dist: dist, │ │ │ │ │ - x: closest.x0, │ │ │ │ │ - y: closest.y0 │ │ │ │ │ - }; │ │ │ │ │ - eligible = true; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - vertices = feature.geometry.getVertices(type === "node"); │ │ │ │ │ - found = false; │ │ │ │ │ - for (var k = 0, klen = vertices.length; k < klen; ++k) { │ │ │ │ │ - vertex = vertices[k]; │ │ │ │ │ - dist = vertex.distanceTo(loc); │ │ │ │ │ - if (dist <= tolerance[type] && (j < result.rank || j === result.rank && dist < result.dist)) { │ │ │ │ │ - result = { │ │ │ │ │ - rank: j, │ │ │ │ │ - dist: dist, │ │ │ │ │ - x: vertex.x, │ │ │ │ │ - y: vertex.y │ │ │ │ │ - }; │ │ │ │ │ - eligible = true; │ │ │ │ │ - found = true │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (found) { │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return eligible ? result : null │ │ │ │ │ - }, │ │ │ │ │ - getGeoTolerance: function(tolerance, resolution) { │ │ │ │ │ - if (resolution !== this.resolution) { │ │ │ │ │ - this.resolution = resolution; │ │ │ │ │ - this.geoToleranceCache = {} │ │ │ │ │ - } │ │ │ │ │ - var geoTolerance = this.geoToleranceCache[tolerance]; │ │ │ │ │ - if (geoTolerance === undefined) { │ │ │ │ │ - geoTolerance = tolerance * resolution; │ │ │ │ │ - this.geoToleranceCache[tolerance] = geoTolerance │ │ │ │ │ - } │ │ │ │ │ - return geoTolerance │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.deactivate() │ │ │ │ │ - } │ │ │ │ │ - delete this.layer; │ │ │ │ │ - delete this.targets; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.call(this) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Snapping" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.WMTSGetFeatureInfo = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - hover: false, │ │ │ │ │ - requestEncoding: "KVP", │ │ │ │ │ - drillDown: false, │ │ │ │ │ - maxFeatures: 10, │ │ │ │ │ - clickCallback: "click", │ │ │ │ │ - layers: null, │ │ │ │ │ - queryVisible: true, │ │ │ │ │ - infoFormat: "text/html", │ │ │ │ │ - vendorParams: {}, │ │ │ │ │ - format: null, │ │ │ │ │ - formatOptions: null, │ │ │ │ │ - handler: null, │ │ │ │ │ - hoverRequest: null, │ │ │ │ │ - pending: 0, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!this.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.WMSGetFeatureInfo(options.formatOptions) │ │ │ │ │ - } │ │ │ │ │ - if (this.drillDown === true) { │ │ │ │ │ - this.hover = false │ │ │ │ │ - } │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ - move: this.cancelHover, │ │ │ │ │ - pause: this.getInfoForHover │ │ │ │ │ - }, OpenLayers.Util.extend(this.handlerOptions.hover || {}, { │ │ │ │ │ - delay: 250 │ │ │ │ │ - })) │ │ │ │ │ - } else { │ │ │ │ │ - var callbacks = {}; │ │ │ │ │ - callbacks[this.clickCallback] = this.getInfoForClick; │ │ │ │ │ - this.handler = new OpenLayers.Handler.Click(this, callbacks, this.handlerOptions.click || {}) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getInfoForClick: function(evt) { │ │ │ │ │ - this.request(evt.xy, {}) │ │ │ │ │ - }, │ │ │ │ │ - getInfoForHover: function(evt) { │ │ │ │ │ - this.request(evt.xy, { │ │ │ │ │ - hover: true │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - cancelHover: function() { │ │ │ │ │ - if (this.hoverRequest) { │ │ │ │ │ - --this.pending; │ │ │ │ │ - if (this.pending <= 0) { │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - this.pending = 0 │ │ │ │ │ - } │ │ │ │ │ - this.hoverRequest.abort(); │ │ │ │ │ - this.hoverRequest = null │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - findLayers: function() { │ │ │ │ │ - var candidates = this.layers || this.map.layers; │ │ │ │ │ - var layers = []; │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = candidates[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.WMTS && layer.requestEncoding === this.requestEncoding && (!this.queryVisible || layer.getVisibility())) { │ │ │ │ │ - layers.push(layer); │ │ │ │ │ - if (!this.drillDown || this.hover) { │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return layers │ │ │ │ │ - }, │ │ │ │ │ - buildRequestOptions: function(layer, xy) { │ │ │ │ │ - var loc = this.map.getLonLatFromPixel(xy); │ │ │ │ │ - var getTileUrl = layer.getURL(new OpenLayers.Bounds(loc.lon, loc.lat, loc.lon, loc.lat)); │ │ │ │ │ - var params = OpenLayers.Util.getParameters(getTileUrl); │ │ │ │ │ - var tileInfo = layer.getTileInfo(loc); │ │ │ │ │ - OpenLayers.Util.extend(params, { │ │ │ │ │ - service: "WMTS", │ │ │ │ │ - version: layer.version, │ │ │ │ │ - request: "GetFeatureInfo", │ │ │ │ │ - infoFormat: this.infoFormat, │ │ │ │ │ - i: tileInfo.i, │ │ │ │ │ - j: tileInfo.j │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Util.applyDefaults(params, this.vendorParams); │ │ │ │ │ - return { │ │ │ │ │ - url: OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url, │ │ │ │ │ - params: OpenLayers.Util.upperCaseObject(params), │ │ │ │ │ - callback: function(request) { │ │ │ │ │ - this.handleResponse(xy, request, layer) │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - request: function(xy, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var layers = this.findLayers(); │ │ │ │ │ - if (layers.length > 0) { │ │ │ │ │ - var issue, layer; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - layer = layers[i]; │ │ │ │ │ - issue = this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ - xy: xy, │ │ │ │ │ - layer: layer │ │ │ │ │ - }); │ │ │ │ │ - if (issue !== false) { │ │ │ │ │ - ++this.pending; │ │ │ │ │ - var requestOptions = this.buildRequestOptions(layer, xy); │ │ │ │ │ - var request = OpenLayers.Request.GET(requestOptions); │ │ │ │ │ - if (options.hover === true) { │ │ │ │ │ - this.hoverRequest = request │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.pending > 0) { │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait") │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - handleResponse: function(xy, request, layer) { │ │ │ │ │ - --this.pending; │ │ │ │ │ - if (this.pending <= 0) { │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - this.pending = 0 │ │ │ │ │ - } │ │ │ │ │ - if (request.status && (request.status < 200 || request.status >= 300)) { │ │ │ │ │ - this.events.triggerEvent("exception", { │ │ │ │ │ - xy: xy, │ │ │ │ │ - request: request, │ │ │ │ │ - layer: layer │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ - } │ │ │ │ │ - var features, except; │ │ │ │ │ - try { │ │ │ │ │ - features = this.format.read(doc) │ │ │ │ │ - } catch (error) { │ │ │ │ │ - except = true; │ │ │ │ │ - this.events.triggerEvent("exception", { │ │ │ │ │ - xy: xy, │ │ │ │ │ - request: request, │ │ │ │ │ - error: error, │ │ │ │ │ - layer: layer │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - if (!except) { │ │ │ │ │ - this.events.triggerEvent("getfeatureinfo", { │ │ │ │ │ - text: request.responseText, │ │ │ │ │ - features: features, │ │ │ │ │ - request: request, │ │ │ │ │ - xy: xy, │ │ │ │ │ - layer: layer │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.WMTSGetFeatureInfo" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.PinchZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - type: OpenLayers.Control.TYPE_TOOL, │ │ │ │ │ - pinchOrigin: null, │ │ │ │ │ - currentCenter: null, │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - preserveCenter: false, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.handler = new OpenLayers.Handler.Pinch(this, { │ │ │ │ │ - start: this.pinchStart, │ │ │ │ │ - move: this.pinchMove, │ │ │ │ │ - done: this.pinchDone │ │ │ │ │ - }, this.handlerOptions) │ │ │ │ │ - }, │ │ │ │ │ - pinchStart: function(evt, pinchData) { │ │ │ │ │ - var xy = this.preserveCenter ? this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ - this.pinchOrigin = xy; │ │ │ │ │ - this.currentCenter = xy │ │ │ │ │ - }, │ │ │ │ │ - pinchMove: function(evt, pinchData) { │ │ │ │ │ - var scale = pinchData.scale; │ │ │ │ │ - var containerOrigin = this.map.layerContainerOriginPx; │ │ │ │ │ - var pinchOrigin = this.pinchOrigin; │ │ │ │ │ - var current = this.preserveCenter ? this.map.getPixelFromLonLat(this.map.getCenter()) : evt.xy; │ │ │ │ │ - var dx = Math.round(containerOrigin.x + current.x - pinchOrigin.x + (scale - 1) * (containerOrigin.x - pinchOrigin.x)); │ │ │ │ │ - var dy = Math.round(containerOrigin.y + current.y - pinchOrigin.y + (scale - 1) * (containerOrigin.y - pinchOrigin.y)); │ │ │ │ │ - this.map.applyTransform(dx, dy, scale); │ │ │ │ │ - this.currentCenter = current │ │ │ │ │ - }, │ │ │ │ │ - pinchDone: function(evt, start, last) { │ │ │ │ │ - this.map.applyTransform(); │ │ │ │ │ - var zoom = this.map.getZoomForResolution(this.map.getResolution() / last.scale, true); │ │ │ │ │ - if (zoom !== this.map.getZoom() || !this.currentCenter.equals(this.pinchOrigin)) { │ │ │ │ │ - var resolution = this.map.getResolutionForZoom(zoom); │ │ │ │ │ - var location = this.map.getLonLatFromPixel(this.pinchOrigin); │ │ │ │ │ - var zoomPixel = this.currentCenter; │ │ │ │ │ - var size = this.map.getSize(); │ │ │ │ │ - location.lon += resolution * (size.w / 2 - zoomPixel.x); │ │ │ │ │ - location.lat -= resolution * (size.h / 2 - zoomPixel.y); │ │ │ │ │ - this.map.div.clientWidth = this.map.div.clientWidth; │ │ │ │ │ - this.map.setCenter(location, zoom) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.PinchZoom" │ │ │ │ │ -}); │ │ │ │ │ OpenLayers.Control.OverviewMap = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ element: null, │ │ │ │ │ ovmap: null, │ │ │ │ │ size: { │ │ │ │ │ w: 180, │ │ │ │ │ h: 90 │ │ │ │ │ }, │ │ │ │ │ @@ -34903,529 +35518,251 @@ │ │ │ │ │ x: Math.round(1 / res * (lonlat.lon - extent.left)), │ │ │ │ │ y: Math.round(1 / res * (extent.top - lonlat.lat)) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.OverviewMap" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.WMSGetFeatureInfo = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - hover: false, │ │ │ │ │ - drillDown: false, │ │ │ │ │ - maxFeatures: 10, │ │ │ │ │ - clickCallback: "click", │ │ │ │ │ - output: "features", │ │ │ │ │ - layers: null, │ │ │ │ │ - queryVisible: false, │ │ │ │ │ - url: null, │ │ │ │ │ - layerUrls: null, │ │ │ │ │ - infoFormat: "text/html", │ │ │ │ │ - vendorParams: {}, │ │ │ │ │ - format: null, │ │ │ │ │ - formatOptions: null, │ │ │ │ │ - handler: null, │ │ │ │ │ - hoverRequest: null, │ │ │ │ │ +OpenLayers.Control.LayerSwitcher = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + layerStates: null, │ │ │ │ │ + layersDiv: null, │ │ │ │ │ + baseLayersDiv: null, │ │ │ │ │ + baseLayers: null, │ │ │ │ │ + dataLbl: null, │ │ │ │ │ + dataLayersDiv: null, │ │ │ │ │ + dataLayers: null, │ │ │ │ │ + minimizeDiv: null, │ │ │ │ │ + maximizeDiv: null, │ │ │ │ │ + ascending: true, │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!this.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.WMSGetFeatureInfo(options.formatOptions) │ │ │ │ │ - } │ │ │ │ │ - if (this.drillDown === true) { │ │ │ │ │ - this.hover = false │ │ │ │ │ - } │ │ │ │ │ - if (this.hover) { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ - move: this.cancelHover, │ │ │ │ │ - pause: this.getInfoForHover │ │ │ │ │ - }, OpenLayers.Util.extend(this.handlerOptions.hover || {}, { │ │ │ │ │ - delay: 250 │ │ │ │ │ - })) │ │ │ │ │ - } else { │ │ │ │ │ - var callbacks = {}; │ │ │ │ │ - callbacks[this.clickCallback] = this.getInfoForClick; │ │ │ │ │ - this.handler = new OpenLayers.Handler.Click(this, callbacks, this.handlerOptions.click || {}) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getInfoForClick: function(evt) { │ │ │ │ │ - this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ - xy: evt.xy │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - this.request(evt.xy, {}) │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.layerStates = [] │ │ │ │ │ }, │ │ │ │ │ - getInfoForHover: function(evt) { │ │ │ │ │ - this.events.triggerEvent("beforegetfeatureinfo", { │ │ │ │ │ - xy: evt.xy │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.clearLayersArray("base"); │ │ │ │ │ + this.clearLayersArray("data"); │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + buttonclick: this.onButtonClick, │ │ │ │ │ + addlayer: this.redraw, │ │ │ │ │ + changelayer: this.redraw, │ │ │ │ │ + removelayer: this.redraw, │ │ │ │ │ + changebaselayer: this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ }); │ │ │ │ │ - this.request(evt.xy, { │ │ │ │ │ - hover: true │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - cancelHover: function() { │ │ │ │ │ - if (this.hoverRequest) { │ │ │ │ │ - this.hoverRequest.abort(); │ │ │ │ │ - this.hoverRequest = null │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - findLayers: function() { │ │ │ │ │ - var candidates = this.layers || this.map.layers; │ │ │ │ │ - var layers = []; │ │ │ │ │ - var layer, url; │ │ │ │ │ - for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = candidates[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.WMS && (!this.queryVisible || layer.getVisibility())) { │ │ │ │ │ - url = OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url; │ │ │ │ │ - if (this.drillDown === false && !this.url) { │ │ │ │ │ - this.url = url │ │ │ │ │ - } │ │ │ │ │ - if (this.drillDown === true || this.urlMatches(url)) { │ │ │ │ │ - layers.push(layer) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return layers │ │ │ │ │ - }, │ │ │ │ │ - urlMatches: function(url) { │ │ │ │ │ - var matches = OpenLayers.Util.isEquivalentUrl(this.url, url); │ │ │ │ │ - if (!matches && this.layerUrls) { │ │ │ │ │ - for (var i = 0, len = this.layerUrls.length; i < len; ++i) { │ │ │ │ │ - if (OpenLayers.Util.isEquivalentUrl(this.layerUrls[i], url)) { │ │ │ │ │ - matches = true; │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return matches │ │ │ │ │ + this.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - buildWMSOptions: function(url, layers, clickPosition, format) { │ │ │ │ │ - var layerNames = [], │ │ │ │ │ - styleNames = []; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - if (layers[i].params.LAYERS != null) { │ │ │ │ │ - layerNames = layerNames.concat(layers[i].params.LAYERS); │ │ │ │ │ - styleNames = styleNames.concat(this.getStyleNames(layers[i])) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var firstLayer = layers[0]; │ │ │ │ │ - var projection = this.map.getProjection(); │ │ │ │ │ - var layerProj = firstLayer.projection; │ │ │ │ │ - if (layerProj && layerProj.equals(this.map.getProjectionObject())) { │ │ │ │ │ - projection = layerProj.getCode() │ │ │ │ │ - } │ │ │ │ │ - var params = OpenLayers.Util.extend({ │ │ │ │ │ - service: "WMS", │ │ │ │ │ - version: firstLayer.params.VERSION, │ │ │ │ │ - request: "GetFeatureInfo", │ │ │ │ │ - exceptions: firstLayer.params.EXCEPTIONS, │ │ │ │ │ - bbox: this.map.getExtent().toBBOX(null, firstLayer.reverseAxisOrder()), │ │ │ │ │ - feature_count: this.maxFeatures, │ │ │ │ │ - height: this.map.getSize().h, │ │ │ │ │ - width: this.map.getSize().w, │ │ │ │ │ - format: format, │ │ │ │ │ - info_format: firstLayer.params.INFO_FORMAT || this.infoFormat │ │ │ │ │ - }, parseFloat(firstLayer.params.VERSION) >= 1.3 ? { │ │ │ │ │ - crs: projection, │ │ │ │ │ - i: parseInt(clickPosition.x), │ │ │ │ │ - j: parseInt(clickPosition.y) │ │ │ │ │ - } : { │ │ │ │ │ - srs: projection, │ │ │ │ │ - x: parseInt(clickPosition.x), │ │ │ │ │ - y: parseInt(clickPosition.y) │ │ │ │ │ - }); │ │ │ │ │ - if (layerNames.length != 0) { │ │ │ │ │ - params = OpenLayers.Util.extend({ │ │ │ │ │ - layers: layerNames, │ │ │ │ │ - query_layers: layerNames, │ │ │ │ │ - styles: styleNames │ │ │ │ │ - }, params) │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Util.applyDefaults(params, this.vendorParams); │ │ │ │ │ - return { │ │ │ │ │ - url: url, │ │ │ │ │ - params: OpenLayers.Util.upperCaseObject(params), │ │ │ │ │ - callback: function(request) { │ │ │ │ │ - this.handleResponse(clickPosition, request, url) │ │ │ │ │ - }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + addlayer: this.redraw, │ │ │ │ │ + changelayer: this.redraw, │ │ │ │ │ + removelayer: this.redraw, │ │ │ │ │ + changebaselayer: this.redraw, │ │ │ │ │ scope: this │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getStyleNames: function(layer) { │ │ │ │ │ - var styleNames; │ │ │ │ │ - if (layer.params.STYLES) { │ │ │ │ │ - styleNames = layer.params.STYLES │ │ │ │ │ + }); │ │ │ │ │ + if (this.outsideViewport) { │ │ │ │ │ + this.events.attachToElement(this.div); │ │ │ │ │ + this.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ } else { │ │ │ │ │ - if (OpenLayers.Util.isArray(layer.params.LAYERS)) { │ │ │ │ │ - styleNames = new Array(layer.params.LAYERS.length) │ │ │ │ │ - } else { │ │ │ │ │ - styleNames = layer.params.LAYERS.replace(/[^,]/g, "") │ │ │ │ │ - } │ │ │ │ │ + this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ } │ │ │ │ │ - return styleNames │ │ │ │ │ }, │ │ │ │ │ - request: function(clickPosition, options) { │ │ │ │ │ - var layers = this.findLayers(); │ │ │ │ │ - if (layers.length == 0) { │ │ │ │ │ - this.events.triggerEvent("nogetfeatureinfo"); │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ - return │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this); │ │ │ │ │ + this.loadContents(); │ │ │ │ │ + if (!this.outsideViewport) { │ │ │ │ │ + this.minimizeControl() │ │ │ │ │ } │ │ │ │ │ - options = options || {}; │ │ │ │ │ - if (this.drillDown === false) { │ │ │ │ │ - var wmsOptions = this.buildWMSOptions(this.url, layers, clickPosition, layers[0].params.FORMAT); │ │ │ │ │ - var request = OpenLayers.Request.GET(wmsOptions); │ │ │ │ │ - if (options.hover === true) { │ │ │ │ │ - this.hoverRequest = request │ │ │ │ │ + this.redraw(); │ │ │ │ │ + return this.div │ │ │ │ │ + }, │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + var button = evt.buttonElement; │ │ │ │ │ + if (button === this.minimizeDiv) { │ │ │ │ │ + this.minimizeControl() │ │ │ │ │ + } else if (button === this.maximizeDiv) { │ │ │ │ │ + this.maximizeControl() │ │ │ │ │ + } else if (button._layerSwitcher === this.id) { │ │ │ │ │ + if (button["for"]) { │ │ │ │ │ + button = document.getElementById(button["for"]) │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - this._requestCount = 0; │ │ │ │ │ - this._numRequests = 0; │ │ │ │ │ - this.features = []; │ │ │ │ │ - var services = {}, │ │ │ │ │ - url; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - var layer = layers[i]; │ │ │ │ │ - var service, found = false; │ │ │ │ │ - url = OpenLayers.Util.isArray(layer.url) ? layer.url[0] : layer.url; │ │ │ │ │ - if (url in services) { │ │ │ │ │ - services[url].push(layer) │ │ │ │ │ + if (!button.disabled) { │ │ │ │ │ + if (button.type == "radio") { │ │ │ │ │ + button.checked = true; │ │ │ │ │ + this.map.setBaseLayer(this.map.getLayer(button._layer)) │ │ │ │ │ } else { │ │ │ │ │ - this._numRequests++; │ │ │ │ │ - services[url] = [layer] │ │ │ │ │ + button.checked = !button.checked; │ │ │ │ │ + this.updateMap() │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var layers; │ │ │ │ │ - for (var url in services) { │ │ │ │ │ - layers = services[url]; │ │ │ │ │ - var wmsOptions = this.buildWMSOptions(url, layers, clickPosition, layers[0].params.FORMAT); │ │ │ │ │ - OpenLayers.Request.GET(wmsOptions) │ │ │ │ │ - } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - triggerGetFeatureInfo: function(request, xy, features) { │ │ │ │ │ - this.events.triggerEvent("getfeatureinfo", { │ │ │ │ │ - text: request.responseText, │ │ │ │ │ - features: features, │ │ │ │ │ - request: request, │ │ │ │ │ - xy: xy │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait") │ │ │ │ │ + clearLayersArray: function(layersType) { │ │ │ │ │ + this[layersType + "LayersDiv"].innerHTML = ""; │ │ │ │ │ + this[layersType + "Layers"] = [] │ │ │ │ │ }, │ │ │ │ │ - handleResponse: function(xy, request, url) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ + checkRedraw: function() { │ │ │ │ │ + if (!this.layerStates.length || this.map.layers.length != this.layerStates.length) { │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ - var features = this.format.read(doc); │ │ │ │ │ - if (this.drillDown === false) { │ │ │ │ │ - this.triggerGetFeatureInfo(request, xy, features) │ │ │ │ │ - } else { │ │ │ │ │ - this._requestCount++; │ │ │ │ │ - if (this.output === "object") { │ │ │ │ │ - this._features = (this._features || []).concat({ │ │ │ │ │ - url: url, │ │ │ │ │ - features: features │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - this._features = (this._features || []).concat(features) │ │ │ │ │ - } │ │ │ │ │ - if (this._requestCount === this._numRequests) { │ │ │ │ │ - this.triggerGetFeatureInfo(request, xy, this._features.concat()); │ │ │ │ │ - delete this._features; │ │ │ │ │ - delete this._requestCount; │ │ │ │ │ - delete this._numRequests │ │ │ │ │ + for (var i = 0, len = this.layerStates.length; i < len; i++) { │ │ │ │ │ + var layerState = this.layerStates[i]; │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if (layerState.name != layer.name || layerState.inRange != layer.inRange || layerState.id != layer.id || layerState.visibility != layer.visibility) { │ │ │ │ │ + return true │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + return false │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.WMSGetFeatureInfo" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.EditingToolbar = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ - citeCompliant: false, │ │ │ │ │ - initialize: function(layer, options) { │ │ │ │ │ - OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.addControls([new OpenLayers.Control.Navigation]); │ │ │ │ │ - var controls = [new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Point, { │ │ │ │ │ - displayClass: "olControlDrawFeaturePoint", │ │ │ │ │ - handlerOptions: { │ │ │ │ │ - citeCompliant: this.citeCompliant │ │ │ │ │ - } │ │ │ │ │ - }), new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Path, { │ │ │ │ │ - displayClass: "olControlDrawFeaturePath", │ │ │ │ │ - handlerOptions: { │ │ │ │ │ - citeCompliant: this.citeCompliant │ │ │ │ │ - } │ │ │ │ │ - }), new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Polygon, { │ │ │ │ │ - displayClass: "olControlDrawFeaturePolygon", │ │ │ │ │ - handlerOptions: { │ │ │ │ │ - citeCompliant: this.citeCompliant │ │ │ │ │ + redraw: function() { │ │ │ │ │ + if (!this.checkRedraw()) { │ │ │ │ │ + return this.div │ │ │ │ │ + } │ │ │ │ │ + this.clearLayersArray("base"); │ │ │ │ │ + this.clearLayersArray("data"); │ │ │ │ │ + var containsOverlays = false; │ │ │ │ │ + var containsBaseLayers = false; │ │ │ │ │ + var len = this.map.layers.length; │ │ │ │ │ + this.layerStates = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + this.layerStates[i] = { │ │ │ │ │ + name: layer.name, │ │ │ │ │ + visibility: layer.visibility, │ │ │ │ │ + inRange: layer.inRange, │ │ │ │ │ + id: layer.id │ │ │ │ │ } │ │ │ │ │ - })]; │ │ │ │ │ - this.addControls(controls) │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - var div = OpenLayers.Control.Panel.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (this.defaultControl === null) { │ │ │ │ │ - this.defaultControl = this.controls[0] │ │ │ │ │ } │ │ │ │ │ - return div │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.EditingToolbar" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.Attribution = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - separator: ", ", │ │ │ │ │ - template: "${layers}", │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - removelayer: this.updateAttribution, │ │ │ │ │ - addlayer: this.updateAttribution, │ │ │ │ │ - changelayer: this.updateAttribution, │ │ │ │ │ - changebaselayer: this.updateAttribution, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - changebaselayer: this.updateAttribution, │ │ │ │ │ - changelayer: this.updateAttribution, │ │ │ │ │ - addlayer: this.updateAttribution, │ │ │ │ │ - removelayer: this.updateAttribution, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.updateAttribution(); │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - updateAttribution: function() { │ │ │ │ │ - var attributions = []; │ │ │ │ │ - if (this.map && this.map.layers) { │ │ │ │ │ - for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if (layer.attribution && layer.getVisibility()) { │ │ │ │ │ - if (OpenLayers.Util.indexOf(attributions, layer.attribution) === -1) { │ │ │ │ │ - attributions.push(layer.attribution) │ │ │ │ │ - } │ │ │ │ │ + var layers = this.map.layers.slice(); │ │ │ │ │ + if (!this.ascending) { │ │ │ │ │ + layers.reverse() │ │ │ │ │ + } │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + var layer = layers[i]; │ │ │ │ │ + var baseLayer = layer.isBaseLayer; │ │ │ │ │ + if (layer.displayInLayerSwitcher) { │ │ │ │ │ + if (baseLayer) { │ │ │ │ │ + containsBaseLayers = true │ │ │ │ │ + } else { │ │ │ │ │ + containsOverlays = true │ │ │ │ │ + } │ │ │ │ │ + var checked = baseLayer ? layer == this.map.baseLayer : layer.getVisibility(); │ │ │ │ │ + var inputElem = document.createElement("input"), │ │ │ │ │ + inputId = OpenLayers.Util.createUniqueID(this.id + "_input_"); │ │ │ │ │ + inputElem.id = inputId; │ │ │ │ │ + inputElem.name = baseLayer ? this.id + "_baseLayers" : layer.name; │ │ │ │ │ + inputElem.type = baseLayer ? "radio" : "checkbox"; │ │ │ │ │ + inputElem.value = layer.name; │ │ │ │ │ + inputElem.checked = checked; │ │ │ │ │ + inputElem.defaultChecked = checked; │ │ │ │ │ + inputElem.className = "olButton"; │ │ │ │ │ + inputElem._layer = layer.id; │ │ │ │ │ + inputElem._layerSwitcher = this.id; │ │ │ │ │ + if (!baseLayer && !layer.inRange) { │ │ │ │ │ + inputElem.disabled = true │ │ │ │ │ + } │ │ │ │ │ + var labelSpan = document.createElement("label"); │ │ │ │ │ + labelSpan["for"] = inputElem.id; │ │ │ │ │ + OpenLayers.Element.addClass(labelSpan, "labelSpan olButton"); │ │ │ │ │ + labelSpan._layer = layer.id; │ │ │ │ │ + labelSpan._layerSwitcher = this.id; │ │ │ │ │ + if (!baseLayer && !layer.inRange) { │ │ │ │ │ + labelSpan.style.color = "gray" │ │ │ │ │ } │ │ │ │ │ + labelSpan.innerHTML = layer.name; │ │ │ │ │ + labelSpan.style.verticalAlign = baseLayer ? "bottom" : "baseline"; │ │ │ │ │ + var br = document.createElement("br"); │ │ │ │ │ + var groupArray = baseLayer ? this.baseLayers : this.dataLayers; │ │ │ │ │ + groupArray.push({ │ │ │ │ │ + layer: layer, │ │ │ │ │ + inputElem: inputElem, │ │ │ │ │ + labelSpan: labelSpan │ │ │ │ │ + }); │ │ │ │ │ + var groupDiv = baseLayer ? this.baseLayersDiv : this.dataLayersDiv; │ │ │ │ │ + groupDiv.appendChild(inputElem); │ │ │ │ │ + groupDiv.appendChild(labelSpan); │ │ │ │ │ + groupDiv.appendChild(br) │ │ │ │ │ } │ │ │ │ │ - this.div.innerHTML = OpenLayers.String.format(this.template, { │ │ │ │ │ - layers: attributions.join(this.separator) │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Attribution" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.PanZoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - slideFactor: 50, │ │ │ │ │ - slideRatio: null, │ │ │ │ │ - buttons: null, │ │ │ │ │ - position: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.position = new OpenLayers.Pixel(OpenLayers.Control.PanZoom.X, OpenLayers.Control.PanZoom.Y); │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onButtonClick) │ │ │ │ │ } │ │ │ │ │ - this.removeButtons(); │ │ │ │ │ - this.buttons = null; │ │ │ │ │ - this.position = null; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ - }, │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - px = this.position; │ │ │ │ │ - this.buttons = []; │ │ │ │ │ - var sz = { │ │ │ │ │ - w: 18, │ │ │ │ │ - h: 18 │ │ │ │ │ - }; │ │ │ │ │ - var centered = new OpenLayers.Pixel(px.x + sz.w / 2, px.y); │ │ │ │ │ - this._addButton("panup", "north-mini.png", centered, sz); │ │ │ │ │ - px.y = centered.y + sz.h; │ │ │ │ │ - this._addButton("panleft", "west-mini.png", px, sz); │ │ │ │ │ - this._addButton("panright", "east-mini.png", px.add(sz.w, 0), sz); │ │ │ │ │ - this._addButton("pandown", "south-mini.png", centered.add(0, sz.h * 2), sz); │ │ │ │ │ - this._addButton("zoomin", "zoom-plus-mini.png", centered.add(0, sz.h * 3 + 5), sz); │ │ │ │ │ - this._addButton("zoomworld", "zoom-world-mini.png", centered.add(0, sz.h * 4 + 5), sz); │ │ │ │ │ - this._addButton("zoomout", "zoom-minus-mini.png", centered.add(0, sz.h * 5 + 5), sz); │ │ │ │ │ + this.dataLbl.style.display = containsOverlays ? "" : "none"; │ │ │ │ │ + this.baseLbl.style.display = containsBaseLayers ? "" : "none"; │ │ │ │ │ return this.div │ │ │ │ │ }, │ │ │ │ │ - _addButton: function(id, img, xy, sz) { │ │ │ │ │ - var imgLocation = OpenLayers.Util.getImageLocation(img); │ │ │ │ │ - var btn = OpenLayers.Util.createAlphaImageDiv(this.id + "_" + id, xy, sz, imgLocation, "absolute"); │ │ │ │ │ - btn.style.cursor = "pointer"; │ │ │ │ │ - this.div.appendChild(btn); │ │ │ │ │ - btn.action = id; │ │ │ │ │ - btn.className = "olButton"; │ │ │ │ │ - this.buttons.push(btn); │ │ │ │ │ - return btn │ │ │ │ │ - }, │ │ │ │ │ - _removeButton: function(btn) { │ │ │ │ │ - this.div.removeChild(btn); │ │ │ │ │ - OpenLayers.Util.removeItem(this.buttons, btn) │ │ │ │ │ - }, │ │ │ │ │ - removeButtons: function() { │ │ │ │ │ - for (var i = this.buttons.length - 1; i >= 0; --i) { │ │ │ │ │ - this._removeButton(this.buttons[i]) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - var btn = evt.buttonElement; │ │ │ │ │ - switch (btn.action) { │ │ │ │ │ - case "panup": │ │ │ │ │ - this.map.pan(0, -this.getSlideFactor("h")); │ │ │ │ │ - break; │ │ │ │ │ - case "pandown": │ │ │ │ │ - this.map.pan(0, this.getSlideFactor("h")); │ │ │ │ │ - break; │ │ │ │ │ - case "panleft": │ │ │ │ │ - this.map.pan(-this.getSlideFactor("w"), 0); │ │ │ │ │ - break; │ │ │ │ │ - case "panright": │ │ │ │ │ - this.map.pan(this.getSlideFactor("w"), 0); │ │ │ │ │ - break; │ │ │ │ │ - case "zoomin": │ │ │ │ │ - this.map.zoomIn(); │ │ │ │ │ - break; │ │ │ │ │ - case "zoomout": │ │ │ │ │ - this.map.zoomOut(); │ │ │ │ │ - break; │ │ │ │ │ - case "zoomworld": │ │ │ │ │ - this.map.zoomToMaxExtent(); │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - getSlideFactor: function(dim) { │ │ │ │ │ - return this.slideRatio ? this.map.getSize()[dim] * this.slideRatio : this.slideFactor │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.PanZoom" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.PanZoom.X = 4; │ │ │ │ │ -OpenLayers.Control.PanZoom.Y = 4; │ │ │ │ │ -OpenLayers.Control.Permalink = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - argParserClass: OpenLayers.Control.ArgParser, │ │ │ │ │ - element: null, │ │ │ │ │ - anchor: false, │ │ │ │ │ - base: "", │ │ │ │ │ - displayProjection: null, │ │ │ │ │ - initialize: function(element, base, options) { │ │ │ │ │ - if (element !== null && typeof element == "object" && !OpenLayers.Util.isElement(element)) { │ │ │ │ │ - options = element; │ │ │ │ │ - this.base = document.location.href; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (this.element != null) { │ │ │ │ │ - this.element = OpenLayers.Util.getElement(this.element) │ │ │ │ │ + updateMap: function() { │ │ │ │ │ + for (var i = 0, len = this.baseLayers.length; i < len; i++) { │ │ │ │ │ + var layerEntry = this.baseLayers[i]; │ │ │ │ │ + if (layerEntry.inputElem.checked) { │ │ │ │ │ + this.map.setBaseLayer(layerEntry.layer, false) │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.element = OpenLayers.Util.getElement(element); │ │ │ │ │ - this.base = base || document.location.href │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.element && this.element.parentNode == this.div) { │ │ │ │ │ - this.div.removeChild(this.element); │ │ │ │ │ - this.element = null │ │ │ │ │ } │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("moveend", this, this.updateLink) │ │ │ │ │ + for (var i = 0, len = this.dataLayers.length; i < len; i++) { │ │ │ │ │ + var layerEntry = this.dataLayers[i]; │ │ │ │ │ + layerEntry.layer.setVisibility(layerEntry.inputElem.checked) │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - for (var i = 0, len = this.map.controls.length; i < len; i++) { │ │ │ │ │ - var control = this.map.controls[i]; │ │ │ │ │ - if (control.CLASS_NAME == this.argParserClass.CLASS_NAME) { │ │ │ │ │ - if (control.displayProjection != this.displayProjection) { │ │ │ │ │ - this.displayProjection = control.displayProjection │ │ │ │ │ - } │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (i == this.map.controls.length) { │ │ │ │ │ - this.map.addControl(new this.argParserClass({ │ │ │ │ │ - displayProjection: this.displayProjection │ │ │ │ │ - })) │ │ │ │ │ + maximizeControl: function(e) { │ │ │ │ │ + this.div.style.width = ""; │ │ │ │ │ + this.div.style.height = ""; │ │ │ │ │ + this.showControls(false); │ │ │ │ │ + if (e != null) { │ │ │ │ │ + OpenLayers.Event.stop(e) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (!this.element && !this.anchor) { │ │ │ │ │ - this.element = document.createElement("a"); │ │ │ │ │ - this.element.innerHTML = OpenLayers.i18n("Permalink"); │ │ │ │ │ - this.element.href = ""; │ │ │ │ │ - this.div.appendChild(this.element) │ │ │ │ │ + minimizeControl: function(e) { │ │ │ │ │ + this.div.style.width = "0px"; │ │ │ │ │ + this.div.style.height = "0px"; │ │ │ │ │ + this.showControls(true); │ │ │ │ │ + if (e != null) { │ │ │ │ │ + OpenLayers.Event.stop(e) │ │ │ │ │ } │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - moveend: this.updateLink, │ │ │ │ │ - changelayer: this.updateLink, │ │ │ │ │ - changebaselayer: this.updateLink, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.updateLink(); │ │ │ │ │ - return this.div │ │ │ │ │ }, │ │ │ │ │ - updateLink: function() { │ │ │ │ │ - var separator = this.anchor ? "#" : "?"; │ │ │ │ │ - var href = this.base; │ │ │ │ │ - var anchor = null; │ │ │ │ │ - if (href.indexOf("#") != -1 && this.anchor == false) { │ │ │ │ │ - anchor = href.substring(href.indexOf("#"), href.length) │ │ │ │ │ - } │ │ │ │ │ - if (href.indexOf(separator) != -1) { │ │ │ │ │ - href = href.substring(0, href.indexOf(separator)) │ │ │ │ │ - } │ │ │ │ │ - var splits = href.split("#"); │ │ │ │ │ - href = splits[0] + separator + OpenLayers.Util.getParameterString(this.createParams()); │ │ │ │ │ - if (anchor) { │ │ │ │ │ - href += anchor │ │ │ │ │ - } │ │ │ │ │ - if (this.anchor && !this.element) { │ │ │ │ │ - window.location.href = href │ │ │ │ │ - } else { │ │ │ │ │ - this.element.href = href │ │ │ │ │ - } │ │ │ │ │ + showControls: function(minimize) { │ │ │ │ │ + this.maximizeDiv.style.display = minimize ? "" : "none"; │ │ │ │ │ + this.minimizeDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ + this.layersDiv.style.display = minimize ? "none" : "" │ │ │ │ │ }, │ │ │ │ │ - createParams: function(center, zoom, layers) { │ │ │ │ │ - center = center || this.map.getCenter(); │ │ │ │ │ - var params = OpenLayers.Util.getParameters(this.base); │ │ │ │ │ - if (center) { │ │ │ │ │ - params.zoom = zoom || this.map.getZoom(); │ │ │ │ │ - var lat = center.lat; │ │ │ │ │ - var lon = center.lon; │ │ │ │ │ - if (this.displayProjection) { │ │ │ │ │ - var mapPosition = OpenLayers.Projection.transform({ │ │ │ │ │ - x: lon, │ │ │ │ │ - y: lat │ │ │ │ │ - }, this.map.getProjectionObject(), this.displayProjection); │ │ │ │ │ - lon = mapPosition.x; │ │ │ │ │ - lat = mapPosition.y │ │ │ │ │ - } │ │ │ │ │ - params.lat = Math.round(lat * 1e5) / 1e5; │ │ │ │ │ - params.lon = Math.round(lon * 1e5) / 1e5; │ │ │ │ │ - layers = layers || this.map.layers; │ │ │ │ │ - params.layers = ""; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - var layer = layers[i]; │ │ │ │ │ - if (layer.isBaseLayer) { │ │ │ │ │ - params.layers += layer == this.map.baseLayer ? "B" : "0" │ │ │ │ │ - } else { │ │ │ │ │ - params.layers += layer.getVisibility() ? "T" : "F" │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + loadContents: function() { │ │ │ │ │ + this.layersDiv = document.createElement("div"); │ │ │ │ │ + this.layersDiv.id = this.id + "_layersDiv"; │ │ │ │ │ + OpenLayers.Element.addClass(this.layersDiv, "layersDiv"); │ │ │ │ │ + this.baseLbl = document.createElement("div"); │ │ │ │ │ + this.baseLbl.innerHTML = OpenLayers.i18n("Base Layer"); │ │ │ │ │ + OpenLayers.Element.addClass(this.baseLbl, "baseLbl"); │ │ │ │ │ + this.baseLayersDiv = document.createElement("div"); │ │ │ │ │ + OpenLayers.Element.addClass(this.baseLayersDiv, "baseLayersDiv"); │ │ │ │ │ + this.dataLbl = document.createElement("div"); │ │ │ │ │ + this.dataLbl.innerHTML = OpenLayers.i18n("Overlays"); │ │ │ │ │ + OpenLayers.Element.addClass(this.dataLbl, "dataLbl"); │ │ │ │ │ + this.dataLayersDiv = document.createElement("div"); │ │ │ │ │ + OpenLayers.Element.addClass(this.dataLayersDiv, "dataLayersDiv"); │ │ │ │ │ + if (this.ascending) { │ │ │ │ │ + this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLayersDiv) │ │ │ │ │ + } else { │ │ │ │ │ + this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ + this.layersDiv.appendChild(this.baseLayersDiv) │ │ │ │ │ } │ │ │ │ │ - return params │ │ │ │ │ + this.div.appendChild(this.layersDiv); │ │ │ │ │ + var img = OpenLayers.Util.getImageLocation("layer-switcher-maximize.png"); │ │ │ │ │ + this.maximizeDiv = OpenLayers.Util.createAlphaImageDiv("OpenLayers_Control_MaximizeDiv", null, null, img, "absolute"); │ │ │ │ │ + OpenLayers.Element.addClass(this.maximizeDiv, "maximizeDiv olButton"); │ │ │ │ │ + this.maximizeDiv.style.display = "none"; │ │ │ │ │ + this.div.appendChild(this.maximizeDiv); │ │ │ │ │ + var img = OpenLayers.Util.getImageLocation("layer-switcher-minimize.png"); │ │ │ │ │ + this.minimizeDiv = OpenLayers.Util.createAlphaImageDiv("OpenLayers_Control_MinimizeDiv", null, null, img, "absolute"); │ │ │ │ │ + OpenLayers.Element.addClass(this.minimizeDiv, "minimizeDiv olButton"); │ │ │ │ │ + this.minimizeDiv.style.display = "none"; │ │ │ │ │ + this.div.appendChild(this.minimizeDiv) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Permalink" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.LayerSwitcher" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Control.Split = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ layer: null, │ │ │ │ │ source: null, │ │ │ │ │ sourceOptions: null, │ │ │ │ │ tolerance: null, │ │ │ │ │ edge: true, │ │ │ │ │ @@ -35644,230 +35981,14 @@ │ │ │ │ │ if (this.active) { │ │ │ │ │ this.deactivate() │ │ │ │ │ } │ │ │ │ │ OpenLayers.Control.prototype.destroy.call(this) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.Split" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.PanZoomBar = OpenLayers.Class(OpenLayers.Control.PanZoom, { │ │ │ │ │ - zoomStopWidth: 18, │ │ │ │ │ - zoomStopHeight: 11, │ │ │ │ │ - slider: null, │ │ │ │ │ - sliderEvents: null, │ │ │ │ │ - zoombarDiv: null, │ │ │ │ │ - zoomWorldIcon: false, │ │ │ │ │ - panIcons: true, │ │ │ │ │ - forceFixedZoomLevel: false, │ │ │ │ │ - mouseDragStart: null, │ │ │ │ │ - deltaY: null, │ │ │ │ │ - zoomStart: null, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this._removeZoomBar(); │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - changebaselayer: this.redraw, │ │ │ │ │ - updatesize: this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Control.PanZoom.prototype.destroy.apply(this, arguments); │ │ │ │ │ - delete this.mouseDragStart; │ │ │ │ │ - delete this.zoomStart │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.PanZoom.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - changebaselayer: this.redraw, │ │ │ │ │ - updatesize: this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - redraw: function() { │ │ │ │ │ - if (this.div != null) { │ │ │ │ │ - this.removeButtons(); │ │ │ │ │ - this._removeZoomBar() │ │ │ │ │ - } │ │ │ │ │ - this.draw() │ │ │ │ │ - }, │ │ │ │ │ - draw: function(px) { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - px = this.position.clone(); │ │ │ │ │ - this.buttons = []; │ │ │ │ │ - var sz = { │ │ │ │ │ - w: 18, │ │ │ │ │ - h: 18 │ │ │ │ │ - }; │ │ │ │ │ - if (this.panIcons) { │ │ │ │ │ - var centered = new OpenLayers.Pixel(px.x + sz.w / 2, px.y); │ │ │ │ │ - var wposition = sz.w; │ │ │ │ │ - if (this.zoomWorldIcon) { │ │ │ │ │ - centered = new OpenLayers.Pixel(px.x + sz.w, px.y) │ │ │ │ │ - } │ │ │ │ │ - this._addButton("panup", "north-mini.png", centered, sz); │ │ │ │ │ - px.y = centered.y + sz.h; │ │ │ │ │ - this._addButton("panleft", "west-mini.png", px, sz); │ │ │ │ │ - if (this.zoomWorldIcon) { │ │ │ │ │ - this._addButton("zoomworld", "zoom-world-mini.png", px.add(sz.w, 0), sz); │ │ │ │ │ - wposition *= 2 │ │ │ │ │ - } │ │ │ │ │ - this._addButton("panright", "east-mini.png", px.add(wposition, 0), sz); │ │ │ │ │ - this._addButton("pandown", "south-mini.png", centered.add(0, sz.h * 2), sz); │ │ │ │ │ - this._addButton("zoomin", "zoom-plus-mini.png", centered.add(0, sz.h * 3 + 5), sz); │ │ │ │ │ - centered = this._addZoomBar(centered.add(0, sz.h * 4 + 5)); │ │ │ │ │ - this._addButton("zoomout", "zoom-minus-mini.png", centered, sz) │ │ │ │ │ - } else { │ │ │ │ │ - this._addButton("zoomin", "zoom-plus-mini.png", px, sz); │ │ │ │ │ - centered = this._addZoomBar(px.add(0, sz.h)); │ │ │ │ │ - this._addButton("zoomout", "zoom-minus-mini.png", centered, sz); │ │ │ │ │ - if (this.zoomWorldIcon) { │ │ │ │ │ - centered = centered.add(0, sz.h + 3); │ │ │ │ │ - this._addButton("zoomworld", "zoom-world-mini.png", centered, sz) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - _addZoomBar: function(centered) { │ │ │ │ │ - var imgLocation = OpenLayers.Util.getImageLocation("slider.png"); │ │ │ │ │ - var id = this.id + "_" + this.map.id; │ │ │ │ │ - var minZoom = this.map.getMinZoom(); │ │ │ │ │ - var zoomsToEnd = this.map.getNumZoomLevels() - 1 - this.map.getZoom(); │ │ │ │ │ - var slider = OpenLayers.Util.createAlphaImageDiv(id, centered.add(-1, zoomsToEnd * this.zoomStopHeight), { │ │ │ │ │ - w: 20, │ │ │ │ │ - h: 9 │ │ │ │ │ - }, imgLocation, "absolute"); │ │ │ │ │ - slider.style.cursor = "move"; │ │ │ │ │ - this.slider = slider; │ │ │ │ │ - this.sliderEvents = new OpenLayers.Events(this, slider, null, true, { │ │ │ │ │ - includeXY: true │ │ │ │ │ - }); │ │ │ │ │ - this.sliderEvents.on({ │ │ │ │ │ - touchstart: this.zoomBarDown, │ │ │ │ │ - touchmove: this.zoomBarDrag, │ │ │ │ │ - touchend: this.zoomBarUp, │ │ │ │ │ - mousedown: this.zoomBarDown, │ │ │ │ │ - mousemove: this.zoomBarDrag, │ │ │ │ │ - mouseup: this.zoomBarUp │ │ │ │ │ - }); │ │ │ │ │ - var sz = { │ │ │ │ │ - w: this.zoomStopWidth, │ │ │ │ │ - h: this.zoomStopHeight * (this.map.getNumZoomLevels() - minZoom) │ │ │ │ │ - }; │ │ │ │ │ - var imgLocation = OpenLayers.Util.getImageLocation("zoombar.png"); │ │ │ │ │ - var div = null; │ │ │ │ │ - if (OpenLayers.Util.alphaHack()) { │ │ │ │ │ - var id = this.id + "_" + this.map.id; │ │ │ │ │ - div = OpenLayers.Util.createAlphaImageDiv(id, centered, { │ │ │ │ │ - w: sz.w, │ │ │ │ │ - h: this.zoomStopHeight │ │ │ │ │ - }, imgLocation, "absolute", null, "crop"); │ │ │ │ │ - div.style.height = sz.h + "px" │ │ │ │ │ - } else { │ │ │ │ │ - div = OpenLayers.Util.createDiv("OpenLayers_Control_PanZoomBar_Zoombar" + this.map.id, centered, sz, imgLocation) │ │ │ │ │ - } │ │ │ │ │ - div.style.cursor = "pointer"; │ │ │ │ │ - div.className = "olButton"; │ │ │ │ │ - this.zoombarDiv = div; │ │ │ │ │ - this.div.appendChild(div); │ │ │ │ │ - this.startTop = parseInt(div.style.top); │ │ │ │ │ - this.div.appendChild(slider); │ │ │ │ │ - this.map.events.register("zoomend", this, this.moveZoomBar); │ │ │ │ │ - centered = centered.add(0, this.zoomStopHeight * (this.map.getNumZoomLevels() - minZoom)); │ │ │ │ │ - return centered │ │ │ │ │ - }, │ │ │ │ │ - _removeZoomBar: function() { │ │ │ │ │ - this.sliderEvents.un({ │ │ │ │ │ - touchstart: this.zoomBarDown, │ │ │ │ │ - touchmove: this.zoomBarDrag, │ │ │ │ │ - touchend: this.zoomBarUp, │ │ │ │ │ - mousedown: this.zoomBarDown, │ │ │ │ │ - mousemove: this.zoomBarDrag, │ │ │ │ │ - mouseup: this.zoomBarUp │ │ │ │ │ - }); │ │ │ │ │ - this.sliderEvents.destroy(); │ │ │ │ │ - this.div.removeChild(this.zoombarDiv); │ │ │ │ │ - this.zoombarDiv = null; │ │ │ │ │ - this.div.removeChild(this.slider); │ │ │ │ │ - this.slider = null; │ │ │ │ │ - this.map.events.unregister("zoomend", this, this.moveZoomBar) │ │ │ │ │ - }, │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - OpenLayers.Control.PanZoom.prototype.onButtonClick.apply(this, arguments); │ │ │ │ │ - if (evt.buttonElement === this.zoombarDiv) { │ │ │ │ │ - var levels = evt.buttonXY.y / this.zoomStopHeight; │ │ │ │ │ - if (this.forceFixedZoomLevel || !this.map.fractionalZoom) { │ │ │ │ │ - levels = Math.floor(levels) │ │ │ │ │ - } │ │ │ │ │ - var zoom = this.map.getNumZoomLevels() - 1 - levels; │ │ │ │ │ - zoom = Math.min(Math.max(zoom, 0), this.map.getNumZoomLevels() - 1); │ │ │ │ │ - this.map.zoomTo(zoom) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - passEventToSlider: function(evt) { │ │ │ │ │ - this.sliderEvents.handleBrowserEvent(evt) │ │ │ │ │ - }, │ │ │ │ │ - zoomBarDown: function(evt) { │ │ │ │ │ - if (!OpenLayers.Event.isLeftClick(evt) && !OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - touchmove: this.passEventToSlider, │ │ │ │ │ - mousemove: this.passEventToSlider, │ │ │ │ │ - mouseup: this.passEventToSlider, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.mouseDragStart = evt.xy.clone(); │ │ │ │ │ - this.zoomStart = evt.xy.clone(); │ │ │ │ │ - this.div.style.cursor = "move"; │ │ │ │ │ - this.zoombarDiv.offsets = null; │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ - }, │ │ │ │ │ - zoomBarDrag: function(evt) { │ │ │ │ │ - if (this.mouseDragStart != null) { │ │ │ │ │ - var deltaY = this.mouseDragStart.y - evt.xy.y; │ │ │ │ │ - var offsets = OpenLayers.Util.pagePosition(this.zoombarDiv); │ │ │ │ │ - if (evt.clientY - offsets[1] > 0 && evt.clientY - offsets[1] < parseInt(this.zoombarDiv.style.height) - 2) { │ │ │ │ │ - var newTop = parseInt(this.slider.style.top) - deltaY; │ │ │ │ │ - this.slider.style.top = newTop + "px"; │ │ │ │ │ - this.mouseDragStart = evt.xy.clone() │ │ │ │ │ - } │ │ │ │ │ - this.deltaY = this.zoomStart.y - evt.xy.y; │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - zoomBarUp: function(evt) { │ │ │ │ │ - if (!OpenLayers.Event.isLeftClick(evt) && evt.type !== "touchend") { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - if (this.mouseDragStart) { │ │ │ │ │ - this.div.style.cursor = ""; │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - touchmove: this.passEventToSlider, │ │ │ │ │ - mouseup: this.passEventToSlider, │ │ │ │ │ - mousemove: this.passEventToSlider, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - var zoomLevel = this.map.zoom; │ │ │ │ │ - if (!this.forceFixedZoomLevel && this.map.fractionalZoom) { │ │ │ │ │ - zoomLevel += this.deltaY / this.zoomStopHeight; │ │ │ │ │ - zoomLevel = Math.min(Math.max(zoomLevel, 0), this.map.getNumZoomLevels() - 1) │ │ │ │ │ - } else { │ │ │ │ │ - zoomLevel += this.deltaY / this.zoomStopHeight; │ │ │ │ │ - zoomLevel = Math.max(Math.round(zoomLevel), 0) │ │ │ │ │ - } │ │ │ │ │ - this.map.zoomTo(zoomLevel); │ │ │ │ │ - this.mouseDragStart = null; │ │ │ │ │ - this.zoomStart = null; │ │ │ │ │ - this.deltaY = 0; │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - moveZoomBar: function() { │ │ │ │ │ - var newTop = (this.map.getNumZoomLevels() - 1 - this.map.getZoom()) * this.zoomStopHeight + this.startTop + 1; │ │ │ │ │ - this.slider.style.top = newTop + "px" │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.PanZoomBar" │ │ │ │ │ -}); │ │ │ │ │ OpenLayers.Control.DragFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ geometryTypes: null, │ │ │ │ │ onStart: function(feature, pixel) {}, │ │ │ │ │ onDrag: function(feature, pixel) {}, │ │ │ │ │ onComplete: function(feature, pixel) {}, │ │ │ │ │ onEnter: function(feature) {}, │ │ │ │ │ onLeave: function(feature) {}, │ │ │ │ │ @@ -35985,105 +36106,14 @@ │ │ │ │ │ setMap: function(map) { │ │ │ │ │ this.handlers.drag.setMap(map); │ │ │ │ │ this.handlers.feature.setMap(map); │ │ │ │ │ OpenLayers.Control.prototype.setMap.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.DragFeature" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.ScaleLine = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - maxWidth: 100, │ │ │ │ │ - topOutUnits: "km", │ │ │ │ │ - topInUnits: "m", │ │ │ │ │ - bottomOutUnits: "mi", │ │ │ │ │ - bottomInUnits: "ft", │ │ │ │ │ - eTop: null, │ │ │ │ │ - eBottom: null, │ │ │ │ │ - geodesic: false, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (!this.eTop) { │ │ │ │ │ - this.eTop = document.createElement("div"); │ │ │ │ │ - this.eTop.className = this.displayClass + "Top"; │ │ │ │ │ - var theLen = this.topInUnits.length; │ │ │ │ │ - this.div.appendChild(this.eTop); │ │ │ │ │ - if (this.topOutUnits == "" || this.topInUnits == "") { │ │ │ │ │ - this.eTop.style.visibility = "hidden" │ │ │ │ │ - } else { │ │ │ │ │ - this.eTop.style.visibility = "visible" │ │ │ │ │ - } │ │ │ │ │ - this.eBottom = document.createElement("div"); │ │ │ │ │ - this.eBottom.className = this.displayClass + "Bottom"; │ │ │ │ │ - this.div.appendChild(this.eBottom); │ │ │ │ │ - if (this.bottomOutUnits == "" || this.bottomInUnits == "") { │ │ │ │ │ - this.eBottom.style.visibility = "hidden" │ │ │ │ │ - } else { │ │ │ │ │ - this.eBottom.style.visibility = "visible" │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.map.events.register("moveend", this, this.update); │ │ │ │ │ - this.update(); │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - getBarLen: function(maxLen) { │ │ │ │ │ - var digits = parseInt(Math.log(maxLen) / Math.log(10)); │ │ │ │ │ - var pow10 = Math.pow(10, digits); │ │ │ │ │ - var firstChar = parseInt(maxLen / pow10); │ │ │ │ │ - var barLen; │ │ │ │ │ - if (firstChar > 5) { │ │ │ │ │ - barLen = 5 │ │ │ │ │ - } else if (firstChar > 2) { │ │ │ │ │ - barLen = 2 │ │ │ │ │ - } else { │ │ │ │ │ - barLen = 1 │ │ │ │ │ - } │ │ │ │ │ - return barLen * pow10 │ │ │ │ │ - }, │ │ │ │ │ - update: function() { │ │ │ │ │ - var res = this.map.getResolution(); │ │ │ │ │ - if (!res) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var curMapUnits = this.map.getUnits(); │ │ │ │ │ - var inches = OpenLayers.INCHES_PER_UNIT; │ │ │ │ │ - var maxSizeData = this.maxWidth * res * inches[curMapUnits]; │ │ │ │ │ - var geodesicRatio = 1; │ │ │ │ │ - if (this.geodesic === true) { │ │ │ │ │ - var maxSizeGeodesic = (this.map.getGeodesicPixelSize().w || 1e-6) * this.maxWidth; │ │ │ │ │ - var maxSizeKilometers = maxSizeData / inches["km"]; │ │ │ │ │ - geodesicRatio = maxSizeGeodesic / maxSizeKilometers; │ │ │ │ │ - maxSizeData *= geodesicRatio │ │ │ │ │ - } │ │ │ │ │ - var topUnits; │ │ │ │ │ - var bottomUnits; │ │ │ │ │ - if (maxSizeData > 1e5) { │ │ │ │ │ - topUnits = this.topOutUnits; │ │ │ │ │ - bottomUnits = this.bottomOutUnits │ │ │ │ │ - } else { │ │ │ │ │ - topUnits = this.topInUnits; │ │ │ │ │ - bottomUnits = this.bottomInUnits │ │ │ │ │ - } │ │ │ │ │ - var topMax = maxSizeData / inches[topUnits]; │ │ │ │ │ - var bottomMax = maxSizeData / inches[bottomUnits]; │ │ │ │ │ - var topRounded = this.getBarLen(topMax); │ │ │ │ │ - var bottomRounded = this.getBarLen(bottomMax); │ │ │ │ │ - topMax = topRounded / inches[curMapUnits] * inches[topUnits]; │ │ │ │ │ - bottomMax = bottomRounded / inches[curMapUnits] * inches[bottomUnits]; │ │ │ │ │ - var topPx = topMax / res / geodesicRatio; │ │ │ │ │ - var bottomPx = bottomMax / res / geodesicRatio; │ │ │ │ │ - if (this.eBottom.style.visibility == "visible") { │ │ │ │ │ - this.eBottom.style.width = Math.round(bottomPx) + "px"; │ │ │ │ │ - this.eBottom.innerHTML = bottomRounded + " " + bottomUnits │ │ │ │ │ - } │ │ │ │ │ - if (this.eTop.style.visibility == "visible") { │ │ │ │ │ - this.eTop.style.width = Math.round(topPx) + "px"; │ │ │ │ │ - this.eTop.innerHTML = topRounded + " " + topUnits │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.ScaleLine" │ │ │ │ │ -}); │ │ │ │ │ OpenLayers.Control.TransformFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ geometryTypes: null, │ │ │ │ │ layer: null, │ │ │ │ │ preserveAspectRatio: false, │ │ │ │ │ rotate: true, │ │ │ │ │ feature: null, │ │ │ │ │ renderIntent: "temporary", │ │ │ │ │ @@ -36408,312 +36438,623 @@ │ │ │ │ │ this.layer = null; │ │ │ │ │ this.dragControl.destroy(); │ │ │ │ │ this.dragControl = null; │ │ │ │ │ OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.TransformFeature" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.KeyboardDefaults = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - slideFactor: 75, │ │ │ │ │ - observeElement: null, │ │ │ │ │ - draw: function() { │ │ │ │ │ - var observeElement = this.observeElement || document; │ │ │ │ │ - this.handler = new OpenLayers.Handler.Keyboard(this, { │ │ │ │ │ - keydown: this.defaultKeyPress │ │ │ │ │ - }, { │ │ │ │ │ - observeElement: observeElement │ │ │ │ │ - }) │ │ │ │ │ - }, │ │ │ │ │ - defaultKeyPress: function(evt) { │ │ │ │ │ - var size, handled = true; │ │ │ │ │ - var target = OpenLayers.Event.element(evt); │ │ │ │ │ - if (target && (target.tagName == "INPUT" || target.tagName == "TEXTAREA" || target.tagName == "SELECT")) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - switch (evt.keyCode) { │ │ │ │ │ - case OpenLayers.Event.KEY_LEFT: │ │ │ │ │ - this.map.pan(-this.slideFactor, 0); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Event.KEY_RIGHT: │ │ │ │ │ - this.map.pan(this.slideFactor, 0); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Event.KEY_UP: │ │ │ │ │ - this.map.pan(0, -this.slideFactor); │ │ │ │ │ - break; │ │ │ │ │ - case OpenLayers.Event.KEY_DOWN: │ │ │ │ │ - this.map.pan(0, this.slideFactor); │ │ │ │ │ - break; │ │ │ │ │ - case 33: │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - this.map.pan(0, -.75 * size.h); │ │ │ │ │ - break; │ │ │ │ │ - case 34: │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - this.map.pan(0, .75 * size.h); │ │ │ │ │ - break; │ │ │ │ │ - case 35: │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - this.map.pan(.75 * size.w, 0); │ │ │ │ │ - break; │ │ │ │ │ - case 36: │ │ │ │ │ - size = this.map.getSize(); │ │ │ │ │ - this.map.pan(-.75 * size.w, 0); │ │ │ │ │ - break; │ │ │ │ │ - case 43: │ │ │ │ │ - case 61: │ │ │ │ │ - case 187: │ │ │ │ │ - case 107: │ │ │ │ │ - this.map.zoomIn(); │ │ │ │ │ - break; │ │ │ │ │ - case 45: │ │ │ │ │ - case 109: │ │ │ │ │ - case 189: │ │ │ │ │ - case 95: │ │ │ │ │ - this.map.zoomOut(); │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - handled = false │ │ │ │ │ - } │ │ │ │ │ - if (handled) { │ │ │ │ │ - OpenLayers.Event.stop(evt) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.KeyboardDefaults" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.UTFGrid = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - layers: null, │ │ │ │ │ - defaultHandlerOptions: { │ │ │ │ │ - delay: 300, │ │ │ │ │ - pixelTolerance: 4, │ │ │ │ │ - stopMove: false, │ │ │ │ │ - single: true, │ │ │ │ │ - double: false, │ │ │ │ │ - stopSingle: false, │ │ │ │ │ - stopDouble: false │ │ │ │ │ - }, │ │ │ │ │ - handlerMode: "click", │ │ │ │ │ - setHandler: function(hm) { │ │ │ │ │ - this.handlerMode = hm; │ │ │ │ │ - this.resetHandler() │ │ │ │ │ - }, │ │ │ │ │ - resetHandler: function() { │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - this.handler.deactivate(); │ │ │ │ │ - this.handler.destroy(); │ │ │ │ │ - this.handler = null │ │ │ │ │ - } │ │ │ │ │ - if (this.handlerMode == "hover") { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ - pause: this.handleEvent, │ │ │ │ │ - move: this.reset │ │ │ │ │ - }, this.handlerOptions) │ │ │ │ │ - } else if (this.handlerMode == "click") { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Click(this, { │ │ │ │ │ - click: this.handleEvent │ │ │ │ │ - }, this.handlerOptions) │ │ │ │ │ - } else if (this.handlerMode == "move") { │ │ │ │ │ - this.handler = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ - pause: this.handleEvent, │ │ │ │ │ - move: this.handleEvent │ │ │ │ │ - }, this.handlerOptions) │ │ │ │ │ - } │ │ │ │ │ - if (this.handler) { │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ +OpenLayers.Control.GetFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + protocol: null, │ │ │ │ │ + multipleKey: null, │ │ │ │ │ + toggleKey: null, │ │ │ │ │ + modifiers: null, │ │ │ │ │ + multiple: false, │ │ │ │ │ + click: true, │ │ │ │ │ + single: true, │ │ │ │ │ + clickout: true, │ │ │ │ │ + toggle: false, │ │ │ │ │ + clickTolerance: 5, │ │ │ │ │ + hover: false, │ │ │ │ │ + box: false, │ │ │ │ │ + maxFeatures: 10, │ │ │ │ │ + features: null, │ │ │ │ │ + hoverFeature: null, │ │ │ │ │ + handlers: null, │ │ │ │ │ + hoverResponse: null, │ │ │ │ │ + filterType: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.handlerOptions = options.handlerOptions || this.defaultHandlerOptions; │ │ │ │ │ + options.handlerOptions = options.handlerOptions || {}; │ │ │ │ │ OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.resetHandler() │ │ │ │ │ - }, │ │ │ │ │ - handleEvent: function(evt) { │ │ │ │ │ - if (evt == null) { │ │ │ │ │ - this.reset(); │ │ │ │ │ - return │ │ │ │ │ + this.features = {}; │ │ │ │ │ + this.handlers = {}; │ │ │ │ │ + if (this.click) { │ │ │ │ │ + this.handlers.click = new OpenLayers.Handler.Click(this, { │ │ │ │ │ + click: this.selectClick │ │ │ │ │ + }, this.handlerOptions.click || {}) │ │ │ │ │ } │ │ │ │ │ - var lonLat = this.map.getLonLatFromPixel(evt.xy); │ │ │ │ │ - if (!lonLat) { │ │ │ │ │ - return │ │ │ │ │ + if (this.box) { │ │ │ │ │ + this.handlers.box = new OpenLayers.Handler.Box(this, { │ │ │ │ │ + done: this.selectBox │ │ │ │ │ + }, OpenLayers.Util.extend(this.handlerOptions.box, { │ │ │ │ │ + boxDivClassName: "olHandlerBoxSelectFeature" │ │ │ │ │ + })) │ │ │ │ │ } │ │ │ │ │ - var layers = this.findLayers(); │ │ │ │ │ - if (layers.length > 0) { │ │ │ │ │ - var infoLookup = {}; │ │ │ │ │ - var layer, idx; │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - layer = layers[i]; │ │ │ │ │ - idx = OpenLayers.Util.indexOf(this.map.layers, layer); │ │ │ │ │ - infoLookup[idx] = layer.getFeatureInfo(lonLat) │ │ │ │ │ - } │ │ │ │ │ - this.callback(infoLookup, lonLat, evt.xy) │ │ │ │ │ + if (this.hover) { │ │ │ │ │ + this.handlers.hover = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ + move: this.cancelHover, │ │ │ │ │ + pause: this.selectHover │ │ │ │ │ + }, OpenLayers.Util.extend(this.handlerOptions.hover, { │ │ │ │ │ + delay: 250, │ │ │ │ │ + pixelTolerance: 2 │ │ │ │ │ + })) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - callback: function(infoLookup) {}, │ │ │ │ │ - reset: function(evt) { │ │ │ │ │ - this.callback(null) │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (!this.active) { │ │ │ │ │ + for (var i in this.handlers) { │ │ │ │ │ + this.handlers[i].activate() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Control.prototype.activate.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - findLayers: function() { │ │ │ │ │ - var candidates = this.layers || this.map.layers; │ │ │ │ │ - var layers = []; │ │ │ │ │ - var layer; │ │ │ │ │ - for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ - layer = candidates[i]; │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.UTFGrid) { │ │ │ │ │ - layers.push(layer) │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active) { │ │ │ │ │ + for (var i in this.handlers) { │ │ │ │ │ + this.handlers[i].deactivate() │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return layers │ │ │ │ │ + return OpenLayers.Control.prototype.deactivate.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.UTFGrid" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.MousePosition = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - element: null, │ │ │ │ │ - prefix: "", │ │ │ │ │ - separator: ", ", │ │ │ │ │ - suffix: "", │ │ │ │ │ - numDigits: 5, │ │ │ │ │ - granularity: 10, │ │ │ │ │ - emptyString: null, │ │ │ │ │ - lastXy: null, │ │ │ │ │ - displayProjection: null, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ + selectClick: function(evt) { │ │ │ │ │ + var bounds = this.pixelToBounds(evt.xy); │ │ │ │ │ + this.setModifiers(evt); │ │ │ │ │ + this.request(bounds, { │ │ │ │ │ + single: this.single │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.map.events.register("mousemove", this, this.redraw); │ │ │ │ │ - this.map.events.register("mouseout", this, this.reset); │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return true │ │ │ │ │ + selectBox: function(position) { │ │ │ │ │ + var bounds; │ │ │ │ │ + if (position instanceof OpenLayers.Bounds) { │ │ │ │ │ + var minXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.left, │ │ │ │ │ + y: position.bottom │ │ │ │ │ + }); │ │ │ │ │ + var maxXY = this.map.getLonLatFromPixel({ │ │ │ │ │ + x: position.right, │ │ │ │ │ + y: position.top │ │ │ │ │ + }); │ │ │ │ │ + bounds = new OpenLayers.Bounds(minXY.lon, minXY.lat, maxXY.lon, maxXY.lat) │ │ │ │ │ } else { │ │ │ │ │ - return false │ │ │ │ │ + if (this.click) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + bounds = this.pixelToBounds(position) │ │ │ │ │ } │ │ │ │ │ + this.setModifiers(this.handlers.box.dragHandler.evt); │ │ │ │ │ + this.request(bounds) │ │ │ │ │ }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.map.events.unregister("mousemove", this, this.redraw); │ │ │ │ │ - this.map.events.unregister("mouseout", this, this.reset); │ │ │ │ │ - this.element.innerHTML = ""; │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + selectHover: function(evt) { │ │ │ │ │ + var bounds = this.pixelToBounds(evt.xy); │ │ │ │ │ + this.request(bounds, { │ │ │ │ │ + single: true, │ │ │ │ │ + hover: true │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + cancelHover: function() { │ │ │ │ │ + if (this.hoverResponse) { │ │ │ │ │ + this.protocol.abort(this.hoverResponse); │ │ │ │ │ + this.hoverResponse = null; │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait") │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (!this.element) { │ │ │ │ │ - this.div.left = ""; │ │ │ │ │ - this.div.top = ""; │ │ │ │ │ - this.element = this.div │ │ │ │ │ + request: function(bounds, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: this.filterType, │ │ │ │ │ + value: bounds │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Element.addClass(this.map.viewPortDiv, "olCursorWait"); │ │ │ │ │ + var response = this.protocol.read({ │ │ │ │ │ + maxFeatures: options.single == true ? this.maxFeatures : undefined, │ │ │ │ │ + filter: filter, │ │ │ │ │ + callback: function(result) { │ │ │ │ │ + if (result.success()) { │ │ │ │ │ + if (result.features.length) { │ │ │ │ │ + if (options.single == true) { │ │ │ │ │ + this.selectBestFeature(result.features, bounds.getCenterLonLat(), options) │ │ │ │ │ + } else { │ │ │ │ │ + this.select(result.features) │ │ │ │ │ + } │ │ │ │ │ + } else if (options.hover) { │ │ │ │ │ + this.hoverSelect() │ │ │ │ │ + } else { │ │ │ │ │ + this.events.triggerEvent("clickout"); │ │ │ │ │ + if (this.clickout) { │ │ │ │ │ + this.unselectAll() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Element.removeClass(this.map.viewPortDiv, "olCursorWait") │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + if (options.hover == true) { │ │ │ │ │ + this.hoverResponse = response │ │ │ │ │ } │ │ │ │ │ - return this.div │ │ │ │ │ }, │ │ │ │ │ - redraw: function(evt) { │ │ │ │ │ - var lonLat; │ │ │ │ │ - if (evt == null) { │ │ │ │ │ - this.reset(); │ │ │ │ │ - return │ │ │ │ │ - } else { │ │ │ │ │ - if (this.lastXy == null || Math.abs(evt.xy.x - this.lastXy.x) > this.granularity || Math.abs(evt.xy.y - this.lastXy.y) > this.granularity) { │ │ │ │ │ - this.lastXy = evt.xy; │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - lonLat = this.map.getLonLatFromPixel(evt.xy); │ │ │ │ │ - if (!lonLat) { │ │ │ │ │ - return │ │ │ │ │ + selectBestFeature: function(features, clickPosition, options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + if (features.length) { │ │ │ │ │ + var point = new OpenLayers.Geometry.Point(clickPosition.lon, clickPosition.lat); │ │ │ │ │ + var feature, resultFeature, dist; │ │ │ │ │ + var minDist = Number.MAX_VALUE; │ │ │ │ │ + for (var i = 0; i < features.length; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + dist = point.distanceTo(feature.geometry, { │ │ │ │ │ + edge: false │ │ │ │ │ + }); │ │ │ │ │ + if (dist < minDist) { │ │ │ │ │ + minDist = dist; │ │ │ │ │ + resultFeature = feature; │ │ │ │ │ + if (minDist == 0) { │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (this.displayProjection) { │ │ │ │ │ - lonLat.transform(this.map.getProjectionObject(), this.displayProjection) │ │ │ │ │ + if (options.hover == true) { │ │ │ │ │ + this.hoverSelect(resultFeature) │ │ │ │ │ + } else { │ │ │ │ │ + this.select(resultFeature || features) │ │ │ │ │ } │ │ │ │ │ - this.lastXy = evt.xy │ │ │ │ │ } │ │ │ │ │ - var newHtml = this.formatOutput(lonLat); │ │ │ │ │ - if (newHtml != this.element.innerHTML) { │ │ │ │ │ - this.element.innerHTML = newHtml │ │ │ │ │ + }, │ │ │ │ │ + setModifiers: function(evt) { │ │ │ │ │ + this.modifiers = { │ │ │ │ │ + multiple: this.multiple || this.multipleKey && evt[this.multipleKey], │ │ │ │ │ + toggle: this.toggle || this.toggleKey && evt[this.toggleKey] │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - reset: function(evt) { │ │ │ │ │ - if (this.emptyString != null) { │ │ │ │ │ - this.element.innerHTML = this.emptyString │ │ │ │ │ + select: function(features) { │ │ │ │ │ + if (!this.modifiers.multiple && !this.modifiers.toggle) { │ │ │ │ │ + this.unselectAll() │ │ │ │ │ + } │ │ │ │ │ + if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ + features = [features] │ │ │ │ │ + } │ │ │ │ │ + var cont = this.events.triggerEvent("beforefeaturesselected", { │ │ │ │ │ + features: features │ │ │ │ │ + }); │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + var selectedFeatures = []; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (this.features[feature.fid || feature.id]) { │ │ │ │ │ + if (this.modifiers.toggle) { │ │ │ │ │ + this.unselect(this.features[feature.fid || feature.id]) │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + cont = this.events.triggerEvent("beforefeatureselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + if (cont !== false) { │ │ │ │ │ + this.features[feature.fid || feature.id] = feature; │ │ │ │ │ + selectedFeatures.push(feature); │ │ │ │ │ + this.events.triggerEvent("featureselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("featuresselected", { │ │ │ │ │ + features: selectedFeatures │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - formatOutput: function(lonLat) { │ │ │ │ │ - var digits = parseInt(this.numDigits); │ │ │ │ │ - var newHtml = this.prefix + lonLat.lon.toFixed(digits) + this.separator + lonLat.lat.toFixed(digits) + this.suffix; │ │ │ │ │ - return newHtml │ │ │ │ │ + hoverSelect: function(feature) { │ │ │ │ │ + var fid = feature ? feature.fid || feature.id : null; │ │ │ │ │ + var hfid = this.hoverFeature ? this.hoverFeature.fid || this.hoverFeature.id : null; │ │ │ │ │ + if (hfid && hfid != fid) { │ │ │ │ │ + this.events.triggerEvent("outfeature", { │ │ │ │ │ + feature: this.hoverFeature │ │ │ │ │ + }); │ │ │ │ │ + this.hoverFeature = null │ │ │ │ │ + } │ │ │ │ │ + if (fid && fid != hfid) { │ │ │ │ │ + this.events.triggerEvent("hoverfeature", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }); │ │ │ │ │ + this.hoverFeature = feature │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.MousePosition" │ │ │ │ │ + unselect: function(feature) { │ │ │ │ │ + delete this.features[feature.fid || feature.id]; │ │ │ │ │ + this.events.triggerEvent("featureunselected", { │ │ │ │ │ + feature: feature │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + unselectAll: function() { │ │ │ │ │ + for (var fid in this.features) { │ │ │ │ │ + this.unselect(this.features[fid]) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + for (var i in this.handlers) { │ │ │ │ │ + this.handlers[i].setMap(map) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Control.prototype.setMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + pixelToBounds: function(pixel) { │ │ │ │ │ + var llPx = pixel.add(-this.clickTolerance / 2, this.clickTolerance / 2); │ │ │ │ │ + var urPx = pixel.add(this.clickTolerance / 2, -this.clickTolerance / 2); │ │ │ │ │ + var ll = this.map.getLonLatFromPixel(llPx); │ │ │ │ │ + var ur = this.map.getLonLatFromPixel(urPx); │ │ │ │ │ + return new OpenLayers.Bounds(ll.lon, ll.lat, ur.lon, ur.lat) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.GetFeature" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.Zoom = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - zoomInText: "+", │ │ │ │ │ - zoomInId: "olZoomInLink", │ │ │ │ │ - zoomOutText: "−", │ │ │ │ │ - zoomOutId: "olZoomOutLink", │ │ │ │ │ +OpenLayers.Control.PanZoomBar = OpenLayers.Class(OpenLayers.Control.PanZoom, { │ │ │ │ │ + zoomStopWidth: 18, │ │ │ │ │ + zoomStopHeight: 11, │ │ │ │ │ + slider: null, │ │ │ │ │ + sliderEvents: null, │ │ │ │ │ + zoombarDiv: null, │ │ │ │ │ + zoomWorldIcon: false, │ │ │ │ │ + panIcons: true, │ │ │ │ │ + forceFixedZoomLevel: false, │ │ │ │ │ + mouseDragStart: null, │ │ │ │ │ + deltaY: null, │ │ │ │ │ + zoomStart: null, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this._removeZoomBar(); │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + changebaselayer: this.redraw, │ │ │ │ │ + updatesize: this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Control.PanZoom.prototype.destroy.apply(this, arguments); │ │ │ │ │ + delete this.mouseDragStart; │ │ │ │ │ + delete this.zoomStart │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Control.PanZoom.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + changebaselayer: this.redraw, │ │ │ │ │ + updatesize: this.redraw, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + redraw: function() { │ │ │ │ │ + if (this.div != null) { │ │ │ │ │ + this.removeButtons(); │ │ │ │ │ + this._removeZoomBar() │ │ │ │ │ + } │ │ │ │ │ + this.draw() │ │ │ │ │ + }, │ │ │ │ │ + draw: function(px) { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + px = this.position.clone(); │ │ │ │ │ + this.buttons = []; │ │ │ │ │ + var sz = { │ │ │ │ │ + w: 18, │ │ │ │ │ + h: 18 │ │ │ │ │ + }; │ │ │ │ │ + if (this.panIcons) { │ │ │ │ │ + var centered = new OpenLayers.Pixel(px.x + sz.w / 2, px.y); │ │ │ │ │ + var wposition = sz.w; │ │ │ │ │ + if (this.zoomWorldIcon) { │ │ │ │ │ + centered = new OpenLayers.Pixel(px.x + sz.w, px.y) │ │ │ │ │ + } │ │ │ │ │ + this._addButton("panup", "north-mini.png", centered, sz); │ │ │ │ │ + px.y = centered.y + sz.h; │ │ │ │ │ + this._addButton("panleft", "west-mini.png", px, sz); │ │ │ │ │ + if (this.zoomWorldIcon) { │ │ │ │ │ + this._addButton("zoomworld", "zoom-world-mini.png", px.add(sz.w, 0), sz); │ │ │ │ │ + wposition *= 2 │ │ │ │ │ + } │ │ │ │ │ + this._addButton("panright", "east-mini.png", px.add(wposition, 0), sz); │ │ │ │ │ + this._addButton("pandown", "south-mini.png", centered.add(0, sz.h * 2), sz); │ │ │ │ │ + this._addButton("zoomin", "zoom-plus-mini.png", centered.add(0, sz.h * 3 + 5), sz); │ │ │ │ │ + centered = this._addZoomBar(centered.add(0, sz.h * 4 + 5)); │ │ │ │ │ + this._addButton("zoomout", "zoom-minus-mini.png", centered, sz) │ │ │ │ │ + } else { │ │ │ │ │ + this._addButton("zoomin", "zoom-plus-mini.png", px, sz); │ │ │ │ │ + centered = this._addZoomBar(px.add(0, sz.h)); │ │ │ │ │ + this._addButton("zoomout", "zoom-minus-mini.png", centered, sz); │ │ │ │ │ + if (this.zoomWorldIcon) { │ │ │ │ │ + centered = centered.add(0, sz.h + 3); │ │ │ │ │ + this._addButton("zoomworld", "zoom-world-mini.png", centered, sz) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return this.div │ │ │ │ │ + }, │ │ │ │ │ + _addZoomBar: function(centered) { │ │ │ │ │ + var imgLocation = OpenLayers.Util.getImageLocation("slider.png"); │ │ │ │ │ + var id = this.id + "_" + this.map.id; │ │ │ │ │ + var minZoom = this.map.getMinZoom(); │ │ │ │ │ + var zoomsToEnd = this.map.getNumZoomLevels() - 1 - this.map.getZoom(); │ │ │ │ │ + var slider = OpenLayers.Util.createAlphaImageDiv(id, centered.add(-1, zoomsToEnd * this.zoomStopHeight), { │ │ │ │ │ + w: 20, │ │ │ │ │ + h: 9 │ │ │ │ │ + }, imgLocation, "absolute"); │ │ │ │ │ + slider.style.cursor = "move"; │ │ │ │ │ + this.slider = slider; │ │ │ │ │ + this.sliderEvents = new OpenLayers.Events(this, slider, null, true, { │ │ │ │ │ + includeXY: true │ │ │ │ │ + }); │ │ │ │ │ + this.sliderEvents.on({ │ │ │ │ │ + touchstart: this.zoomBarDown, │ │ │ │ │ + touchmove: this.zoomBarDrag, │ │ │ │ │ + touchend: this.zoomBarUp, │ │ │ │ │ + mousedown: this.zoomBarDown, │ │ │ │ │ + mousemove: this.zoomBarDrag, │ │ │ │ │ + mouseup: this.zoomBarUp │ │ │ │ │ + }); │ │ │ │ │ + var sz = { │ │ │ │ │ + w: this.zoomStopWidth, │ │ │ │ │ + h: this.zoomStopHeight * (this.map.getNumZoomLevels() - minZoom) │ │ │ │ │ + }; │ │ │ │ │ + var imgLocation = OpenLayers.Util.getImageLocation("zoombar.png"); │ │ │ │ │ + var div = null; │ │ │ │ │ + if (OpenLayers.Util.alphaHack()) { │ │ │ │ │ + var id = this.id + "_" + this.map.id; │ │ │ │ │ + div = OpenLayers.Util.createAlphaImageDiv(id, centered, { │ │ │ │ │ + w: sz.w, │ │ │ │ │ + h: this.zoomStopHeight │ │ │ │ │ + }, imgLocation, "absolute", null, "crop"); │ │ │ │ │ + div.style.height = sz.h + "px" │ │ │ │ │ + } else { │ │ │ │ │ + div = OpenLayers.Util.createDiv("OpenLayers_Control_PanZoomBar_Zoombar" + this.map.id, centered, sz, imgLocation) │ │ │ │ │ + } │ │ │ │ │ + div.style.cursor = "pointer"; │ │ │ │ │ + div.className = "olButton"; │ │ │ │ │ + this.zoombarDiv = div; │ │ │ │ │ + this.div.appendChild(div); │ │ │ │ │ + this.startTop = parseInt(div.style.top); │ │ │ │ │ + this.div.appendChild(slider); │ │ │ │ │ + this.map.events.register("zoomend", this, this.moveZoomBar); │ │ │ │ │ + centered = centered.add(0, this.zoomStopHeight * (this.map.getNumZoomLevels() - minZoom)); │ │ │ │ │ + return centered │ │ │ │ │ + }, │ │ │ │ │ + _removeZoomBar: function() { │ │ │ │ │ + this.sliderEvents.un({ │ │ │ │ │ + touchstart: this.zoomBarDown, │ │ │ │ │ + touchmove: this.zoomBarDrag, │ │ │ │ │ + touchend: this.zoomBarUp, │ │ │ │ │ + mousedown: this.zoomBarDown, │ │ │ │ │ + mousemove: this.zoomBarDrag, │ │ │ │ │ + mouseup: this.zoomBarUp │ │ │ │ │ + }); │ │ │ │ │ + this.sliderEvents.destroy(); │ │ │ │ │ + this.div.removeChild(this.zoombarDiv); │ │ │ │ │ + this.zoombarDiv = null; │ │ │ │ │ + this.div.removeChild(this.slider); │ │ │ │ │ + this.slider = null; │ │ │ │ │ + this.map.events.unregister("zoomend", this, this.moveZoomBar) │ │ │ │ │ + }, │ │ │ │ │ + onButtonClick: function(evt) { │ │ │ │ │ + OpenLayers.Control.PanZoom.prototype.onButtonClick.apply(this, arguments); │ │ │ │ │ + if (evt.buttonElement === this.zoombarDiv) { │ │ │ │ │ + var levels = evt.buttonXY.y / this.zoomStopHeight; │ │ │ │ │ + if (this.forceFixedZoomLevel || !this.map.fractionalZoom) { │ │ │ │ │ + levels = Math.floor(levels) │ │ │ │ │ + } │ │ │ │ │ + var zoom = this.map.getNumZoomLevels() - 1 - levels; │ │ │ │ │ + zoom = Math.min(Math.max(zoom, 0), this.map.getNumZoomLevels() - 1); │ │ │ │ │ + this.map.zoomTo(zoom) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + passEventToSlider: function(evt) { │ │ │ │ │ + this.sliderEvents.handleBrowserEvent(evt) │ │ │ │ │ + }, │ │ │ │ │ + zoomBarDown: function(evt) { │ │ │ │ │ + if (!OpenLayers.Event.isLeftClick(evt) && !OpenLayers.Event.isSingleTouch(evt)) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + touchmove: this.passEventToSlider, │ │ │ │ │ + mousemove: this.passEventToSlider, │ │ │ │ │ + mouseup: this.passEventToSlider, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.mouseDragStart = evt.xy.clone(); │ │ │ │ │ + this.zoomStart = evt.xy.clone(); │ │ │ │ │ + this.div.style.cursor = "move"; │ │ │ │ │ + this.zoombarDiv.offsets = null; │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ + }, │ │ │ │ │ + zoomBarDrag: function(evt) { │ │ │ │ │ + if (this.mouseDragStart != null) { │ │ │ │ │ + var deltaY = this.mouseDragStart.y - evt.xy.y; │ │ │ │ │ + var offsets = OpenLayers.Util.pagePosition(this.zoombarDiv); │ │ │ │ │ + if (evt.clientY - offsets[1] > 0 && evt.clientY - offsets[1] < parseInt(this.zoombarDiv.style.height) - 2) { │ │ │ │ │ + var newTop = parseInt(this.slider.style.top) - deltaY; │ │ │ │ │ + this.slider.style.top = newTop + "px"; │ │ │ │ │ + this.mouseDragStart = evt.xy.clone() │ │ │ │ │ + } │ │ │ │ │ + this.deltaY = this.zoomStart.y - evt.xy.y; │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + zoomBarUp: function(evt) { │ │ │ │ │ + if (!OpenLayers.Event.isLeftClick(evt) && evt.type !== "touchend") { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + if (this.mouseDragStart) { │ │ │ │ │ + this.div.style.cursor = ""; │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + touchmove: this.passEventToSlider, │ │ │ │ │ + mouseup: this.passEventToSlider, │ │ │ │ │ + mousemove: this.passEventToSlider, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + var zoomLevel = this.map.zoom; │ │ │ │ │ + if (!this.forceFixedZoomLevel && this.map.fractionalZoom) { │ │ │ │ │ + zoomLevel += this.deltaY / this.zoomStopHeight; │ │ │ │ │ + zoomLevel = Math.min(Math.max(zoomLevel, 0), this.map.getNumZoomLevels() - 1) │ │ │ │ │ + } else { │ │ │ │ │ + zoomLevel += this.deltaY / this.zoomStopHeight; │ │ │ │ │ + zoomLevel = Math.max(Math.round(zoomLevel), 0) │ │ │ │ │ + } │ │ │ │ │ + this.map.zoomTo(zoomLevel); │ │ │ │ │ + this.mouseDragStart = null; │ │ │ │ │ + this.zoomStart = null; │ │ │ │ │ + this.deltaY = 0; │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + moveZoomBar: function() { │ │ │ │ │ + var newTop = (this.map.getNumZoomLevels() - 1 - this.map.getZoom()) * this.zoomStopHeight + this.startTop + 1; │ │ │ │ │ + this.slider.style.top = newTop + "px" │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.PanZoomBar" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.ScaleLine = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + maxWidth: 100, │ │ │ │ │ + topOutUnits: "km", │ │ │ │ │ + topInUnits: "m", │ │ │ │ │ + bottomOutUnits: "mi", │ │ │ │ │ + bottomInUnits: "ft", │ │ │ │ │ + eTop: null, │ │ │ │ │ + eBottom: null, │ │ │ │ │ + geodesic: false, │ │ │ │ │ draw: function() { │ │ │ │ │ - var div = OpenLayers.Control.prototype.draw.apply(this), │ │ │ │ │ - links = this.getOrCreateLinks(div), │ │ │ │ │ - zoomIn = links.zoomIn, │ │ │ │ │ - zoomOut = links.zoomOut, │ │ │ │ │ - eventsInstance = this.map.events; │ │ │ │ │ - if (zoomOut.parentNode !== div) { │ │ │ │ │ - eventsInstance = this.events; │ │ │ │ │ - eventsInstance.attachToElement(zoomOut.parentNode) │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (!this.eTop) { │ │ │ │ │ + this.eTop = document.createElement("div"); │ │ │ │ │ + this.eTop.className = this.displayClass + "Top"; │ │ │ │ │ + var theLen = this.topInUnits.length; │ │ │ │ │ + this.div.appendChild(this.eTop); │ │ │ │ │ + if (this.topOutUnits == "" || this.topInUnits == "") { │ │ │ │ │ + this.eTop.style.visibility = "hidden" │ │ │ │ │ + } else { │ │ │ │ │ + this.eTop.style.visibility = "visible" │ │ │ │ │ + } │ │ │ │ │ + this.eBottom = document.createElement("div"); │ │ │ │ │ + this.eBottom.className = this.displayClass + "Bottom"; │ │ │ │ │ + this.div.appendChild(this.eBottom); │ │ │ │ │ + if (this.bottomOutUnits == "" || this.bottomInUnits == "") { │ │ │ │ │ + this.eBottom.style.visibility = "hidden" │ │ │ │ │ + } else { │ │ │ │ │ + this.eBottom.style.visibility = "visible" │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - eventsInstance.register("buttonclick", this, this.onZoomClick); │ │ │ │ │ - this.zoomInLink = zoomIn; │ │ │ │ │ - this.zoomOutLink = zoomOut; │ │ │ │ │ - return div │ │ │ │ │ + this.map.events.register("moveend", this, this.update); │ │ │ │ │ + this.update(); │ │ │ │ │ + return this.div │ │ │ │ │ }, │ │ │ │ │ - getOrCreateLinks: function(el) { │ │ │ │ │ - var zoomIn = document.getElementById(this.zoomInId), │ │ │ │ │ - zoomOut = document.getElementById(this.zoomOutId); │ │ │ │ │ - if (!zoomIn) { │ │ │ │ │ - zoomIn = document.createElement("a"); │ │ │ │ │ - zoomIn.href = "#zoomIn"; │ │ │ │ │ - zoomIn.appendChild(document.createTextNode(this.zoomInText)); │ │ │ │ │ - zoomIn.className = "olControlZoomIn"; │ │ │ │ │ - el.appendChild(zoomIn) │ │ │ │ │ + getBarLen: function(maxLen) { │ │ │ │ │ + var digits = parseInt(Math.log(maxLen) / Math.log(10)); │ │ │ │ │ + var pow10 = Math.pow(10, digits); │ │ │ │ │ + var firstChar = parseInt(maxLen / pow10); │ │ │ │ │ + var barLen; │ │ │ │ │ + if (firstChar > 5) { │ │ │ │ │ + barLen = 5 │ │ │ │ │ + } else if (firstChar > 2) { │ │ │ │ │ + barLen = 2 │ │ │ │ │ + } else { │ │ │ │ │ + barLen = 1 │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Element.addClass(zoomIn, "olButton"); │ │ │ │ │ - if (!zoomOut) { │ │ │ │ │ - zoomOut = document.createElement("a"); │ │ │ │ │ - zoomOut.href = "#zoomOut"; │ │ │ │ │ - zoomOut.appendChild(document.createTextNode(this.zoomOutText)); │ │ │ │ │ - zoomOut.className = "olControlZoomOut"; │ │ │ │ │ - el.appendChild(zoomOut) │ │ │ │ │ + return barLen * pow10 │ │ │ │ │ + }, │ │ │ │ │ + update: function() { │ │ │ │ │ + var res = this.map.getResolution(); │ │ │ │ │ + if (!res) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Element.addClass(zoomOut, "olButton"); │ │ │ │ │ - return { │ │ │ │ │ - zoomIn: zoomIn, │ │ │ │ │ - zoomOut: zoomOut │ │ │ │ │ + var curMapUnits = this.map.getUnits(); │ │ │ │ │ + var inches = OpenLayers.INCHES_PER_UNIT; │ │ │ │ │ + var maxSizeData = this.maxWidth * res * inches[curMapUnits]; │ │ │ │ │ + var geodesicRatio = 1; │ │ │ │ │ + if (this.geodesic === true) { │ │ │ │ │ + var maxSizeGeodesic = (this.map.getGeodesicPixelSize().w || 1e-6) * this.maxWidth; │ │ │ │ │ + var maxSizeKilometers = maxSizeData / inches["km"]; │ │ │ │ │ + geodesicRatio = maxSizeGeodesic / maxSizeKilometers; │ │ │ │ │ + maxSizeData *= geodesicRatio │ │ │ │ │ + } │ │ │ │ │ + var topUnits; │ │ │ │ │ + var bottomUnits; │ │ │ │ │ + if (maxSizeData > 1e5) { │ │ │ │ │ + topUnits = this.topOutUnits; │ │ │ │ │ + bottomUnits = this.bottomOutUnits │ │ │ │ │ + } else { │ │ │ │ │ + topUnits = this.topInUnits; │ │ │ │ │ + bottomUnits = this.bottomInUnits │ │ │ │ │ + } │ │ │ │ │ + var topMax = maxSizeData / inches[topUnits]; │ │ │ │ │ + var bottomMax = maxSizeData / inches[bottomUnits]; │ │ │ │ │ + var topRounded = this.getBarLen(topMax); │ │ │ │ │ + var bottomRounded = this.getBarLen(bottomMax); │ │ │ │ │ + topMax = topRounded / inches[curMapUnits] * inches[topUnits]; │ │ │ │ │ + bottomMax = bottomRounded / inches[curMapUnits] * inches[bottomUnits]; │ │ │ │ │ + var topPx = topMax / res / geodesicRatio; │ │ │ │ │ + var bottomPx = bottomMax / res / geodesicRatio; │ │ │ │ │ + if (this.eBottom.style.visibility == "visible") { │ │ │ │ │ + this.eBottom.style.width = Math.round(bottomPx) + "px"; │ │ │ │ │ + this.eBottom.innerHTML = bottomRounded + " " + bottomUnits │ │ │ │ │ + } │ │ │ │ │ + if (this.eTop.style.visibility == "visible") { │ │ │ │ │ + this.eTop.style.width = Math.round(topPx) + "px"; │ │ │ │ │ + this.eTop.innerHTML = topRounded + " " + topUnits │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - onZoomClick: function(evt) { │ │ │ │ │ - var button = evt.buttonElement; │ │ │ │ │ - if (button === this.zoomInLink) { │ │ │ │ │ - this.map.zoomIn() │ │ │ │ │ - } else if (button === this.zoomOutLink) { │ │ │ │ │ - this.map.zoomOut() │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ScaleLine" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Vector.RootContainer = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ + displayInLayerSwitcher: false, │ │ │ │ │ + layers: null, │ │ │ │ │ + display: function() {}, │ │ │ │ │ + getFeatureFromEvent: function(evt) { │ │ │ │ │ + var layers = this.layers; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0; i < layers.length; i++) { │ │ │ │ │ + feature = layers[i].getFeatureFromEvent(evt); │ │ │ │ │ + if (feature) { │ │ │ │ │ + return feature │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.map) { │ │ │ │ │ - this.map.events.unregister("buttonclick", this, this.onZoomClick) │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.collectRoots(); │ │ │ │ │ + map.events.register("changelayer", this, this.handleChangeLayer) │ │ │ │ │ + }, │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + map.events.unregister("changelayer", this, this.handleChangeLayer); │ │ │ │ │ + this.resetRoots(); │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + collectRoots: function() { │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = 0; i < this.map.layers.length; ++i) { │ │ │ │ │ + layer = this.map.layers[i]; │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ + layer.renderer.moveRoot(this.renderer) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - delete this.zoomInLink; │ │ │ │ │ - delete this.zoomOutLink; │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Zoom" │ │ │ │ │ + resetRoots: function() { │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = 0; i < this.layers.length; ++i) { │ │ │ │ │ + layer = this.layers[i]; │ │ │ │ │ + if (this.renderer && layer.renderer.getRenderLayerId() == this.id) { │ │ │ │ │ + this.renderer.moveRoot(layer.renderer) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + handleChangeLayer: function(evt) { │ │ │ │ │ + var layer = evt.layer; │ │ │ │ │ + if (evt.property == "order" && OpenLayers.Util.indexOf(this.layers, layer) != -1) { │ │ │ │ │ + this.resetRoots(); │ │ │ │ │ + this.collectRoots() │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Vector.RootContainer" │ │ │ │ │ }); │ │ │ │ │ OpenLayers.Control.SelectFeature = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ multipleKey: null, │ │ │ │ │ toggleKey: null, │ │ │ │ │ multiple: false, │ │ │ │ │ clickout: true, │ │ │ │ │ toggle: false, │ │ │ │ │ @@ -37002,14 +37343,236 @@ │ │ │ │ │ this.handlers.feature.layer = this.layer; │ │ │ │ │ if (isActive) { │ │ │ │ │ this.activate() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.SelectFeature" │ │ │ │ │ }); │ │ │ │ │ +OpenLayers.Control.EditingToolbar = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ + citeCompliant: false, │ │ │ │ │ + initialize: function(layer, options) { │ │ │ │ │ + OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.addControls([new OpenLayers.Control.Navigation]); │ │ │ │ │ + var controls = [new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Point, { │ │ │ │ │ + displayClass: "olControlDrawFeaturePoint", │ │ │ │ │ + handlerOptions: { │ │ │ │ │ + citeCompliant: this.citeCompliant │ │ │ │ │ + } │ │ │ │ │ + }), new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Path, { │ │ │ │ │ + displayClass: "olControlDrawFeaturePath", │ │ │ │ │ + handlerOptions: { │ │ │ │ │ + citeCompliant: this.citeCompliant │ │ │ │ │ + } │ │ │ │ │ + }), new OpenLayers.Control.DrawFeature(layer, OpenLayers.Handler.Polygon, { │ │ │ │ │ + displayClass: "olControlDrawFeaturePolygon", │ │ │ │ │ + handlerOptions: { │ │ │ │ │ + citeCompliant: this.citeCompliant │ │ │ │ │ + } │ │ │ │ │ + })]; │ │ │ │ │ + this.addControls(controls) │ │ │ │ │ + }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + var div = OpenLayers.Control.Panel.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (this.defaultControl === null) { │ │ │ │ │ + this.defaultControl = this.controls[0] │ │ │ │ │ + } │ │ │ │ │ + return div │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.EditingToolbar" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Attribution = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + separator: ", ", │ │ │ │ │ + template: "${layers}", │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.map.events.un({ │ │ │ │ │ + removelayer: this.updateAttribution, │ │ │ │ │ + addlayer: this.updateAttribution, │ │ │ │ │ + changelayer: this.updateAttribution, │ │ │ │ │ + changebaselayer: this.updateAttribution, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + draw: function() { │ │ │ │ │ + OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ + this.map.events.on({ │ │ │ │ │ + changebaselayer: this.updateAttribution, │ │ │ │ │ + changelayer: this.updateAttribution, │ │ │ │ │ + addlayer: this.updateAttribution, │ │ │ │ │ + removelayer: this.updateAttribution, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.updateAttribution(); │ │ │ │ │ + return this.div │ │ │ │ │ + }, │ │ │ │ │ + updateAttribution: function() { │ │ │ │ │ + var attributions = []; │ │ │ │ │ + if (this.map && this.map.layers) { │ │ │ │ │ + for (var i = 0, len = this.map.layers.length; i < len; i++) { │ │ │ │ │ + var layer = this.map.layers[i]; │ │ │ │ │ + if (layer.attribution && layer.getVisibility()) { │ │ │ │ │ + if (OpenLayers.Util.indexOf(attributions, layer.attribution) === -1) { │ │ │ │ │ + attributions.push(layer.attribution) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.div.innerHTML = OpenLayers.String.format(this.template, { │ │ │ │ │ + layers: attributions.join(this.separator) │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Attribution" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.Geolocate = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + geolocation: null, │ │ │ │ │ + available: "geolocation" in navigator, │ │ │ │ │ + bind: true, │ │ │ │ │ + watch: false, │ │ │ │ │ + geolocationOptions: null, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.deactivate(); │ │ │ │ │ + OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + if (this.available && !this.geolocation) { │ │ │ │ │ + this.geolocation = navigator.geolocation │ │ │ │ │ + } │ │ │ │ │ + if (!this.geolocation) { │ │ │ │ │ + this.events.triggerEvent("locationuncapable"); │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ + if (this.watch) { │ │ │ │ │ + this.watchId = this.geolocation.watchPosition(OpenLayers.Function.bind(this.geolocate, this), OpenLayers.Function.bind(this.failure, this), this.geolocationOptions) │ │ │ │ │ + } else { │ │ │ │ │ + this.getCurrentLocation() │ │ │ │ │ + } │ │ │ │ │ + return true │ │ │ │ │ + } │ │ │ │ │ + return false │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + if (this.active && this.watchId !== null) { │ │ │ │ │ + this.geolocation.clearWatch(this.watchId) │ │ │ │ │ + } │ │ │ │ │ + return OpenLayers.Control.prototype.deactivate.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + geolocate: function(position) { │ │ │ │ │ + var center = new OpenLayers.LonLat(position.coords.longitude, position.coords.latitude).transform(new OpenLayers.Projection("EPSG:4326"), this.map.getProjectionObject()); │ │ │ │ │ + if (this.bind) { │ │ │ │ │ + this.map.setCenter(center) │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("locationupdated", { │ │ │ │ │ + position: position, │ │ │ │ │ + point: new OpenLayers.Geometry.Point(center.lon, center.lat) │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + getCurrentLocation: function() { │ │ │ │ │ + if (!this.active || this.watch) { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + this.geolocation.getCurrentPosition(OpenLayers.Function.bind(this.geolocate, this), OpenLayers.Function.bind(this.failure, this), this.geolocationOptions); │ │ │ │ │ + return true │ │ │ │ │ + }, │ │ │ │ │ + failure: function(error) { │ │ │ │ │ + this.events.triggerEvent("locationfailed", { │ │ │ │ │ + error: error │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.Geolocate" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Control.UTFGrid = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ + autoActivate: true, │ │ │ │ │ + layers: null, │ │ │ │ │ + defaultHandlerOptions: { │ │ │ │ │ + delay: 300, │ │ │ │ │ + pixelTolerance: 4, │ │ │ │ │ + stopMove: false, │ │ │ │ │ + single: true, │ │ │ │ │ + double: false, │ │ │ │ │ + stopSingle: false, │ │ │ │ │ + stopDouble: false │ │ │ │ │ + }, │ │ │ │ │ + handlerMode: "click", │ │ │ │ │ + setHandler: function(hm) { │ │ │ │ │ + this.handlerMode = hm; │ │ │ │ │ + this.resetHandler() │ │ │ │ │ + }, │ │ │ │ │ + resetHandler: function() { │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + this.handler.deactivate(); │ │ │ │ │ + this.handler.destroy(); │ │ │ │ │ + this.handler = null │ │ │ │ │ + } │ │ │ │ │ + if (this.handlerMode == "hover") { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ + pause: this.handleEvent, │ │ │ │ │ + move: this.reset │ │ │ │ │ + }, this.handlerOptions) │ │ │ │ │ + } else if (this.handlerMode == "click") { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Click(this, { │ │ │ │ │ + click: this.handleEvent │ │ │ │ │ + }, this.handlerOptions) │ │ │ │ │ + } else if (this.handlerMode == "move") { │ │ │ │ │ + this.handler = new OpenLayers.Handler.Hover(this, { │ │ │ │ │ + pause: this.handleEvent, │ │ │ │ │ + move: this.handleEvent │ │ │ │ │ + }, this.handlerOptions) │ │ │ │ │ + } │ │ │ │ │ + if (this.handler) { │ │ │ │ │ + return true │ │ │ │ │ + } else { │ │ │ │ │ + return false │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = options || {}; │ │ │ │ │ + options.handlerOptions = options.handlerOptions || this.defaultHandlerOptions; │ │ │ │ │ + OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.resetHandler() │ │ │ │ │ + }, │ │ │ │ │ + handleEvent: function(evt) { │ │ │ │ │ + if (evt == null) { │ │ │ │ │ + this.reset(); │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var lonLat = this.map.getLonLatFromPixel(evt.xy); │ │ │ │ │ + if (!lonLat) { │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ + var layers = this.findLayers(); │ │ │ │ │ + if (layers.length > 0) { │ │ │ │ │ + var infoLookup = {}; │ │ │ │ │ + var layer, idx; │ │ │ │ │ + for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ + layer = layers[i]; │ │ │ │ │ + idx = OpenLayers.Util.indexOf(this.map.layers, layer); │ │ │ │ │ + infoLookup[idx] = layer.getFeatureInfo(lonLat) │ │ │ │ │ + } │ │ │ │ │ + this.callback(infoLookup, lonLat, evt.xy) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + callback: function(infoLookup) {}, │ │ │ │ │ + reset: function(evt) { │ │ │ │ │ + this.callback(null) │ │ │ │ │ + }, │ │ │ │ │ + findLayers: function() { │ │ │ │ │ + var candidates = this.layers || this.map.layers; │ │ │ │ │ + var layers = []; │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = candidates.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = candidates[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.UTFGrid) { │ │ │ │ │ + layers.push(layer) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + return layers │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.UTFGrid" │ │ │ │ │ +}); │ │ │ │ │ OpenLayers.Control.NavigationHistory = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ type: OpenLayers.Control.TYPE_TOGGLE, │ │ │ │ │ previous: null, │ │ │ │ │ previousOptions: null, │ │ │ │ │ next: null, │ │ │ │ │ nextOptions: null, │ │ │ │ │ limit: 50, │ │ │ │ │ @@ -37187,549 +37750,14 @@ │ │ │ │ │ deactivated = true │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ return deactivated │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.NavigationHistory" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.TouchNavigation = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - dragPan: null, │ │ │ │ │ - dragPanOptions: null, │ │ │ │ │ - pinchZoom: null, │ │ │ │ │ - pinchZoomOptions: null, │ │ │ │ │ - clickHandlerOptions: null, │ │ │ │ │ - documentDrag: false, │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - this.handlers = {}; │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - if (this.dragPan) { │ │ │ │ │ - this.dragPan.destroy() │ │ │ │ │ - } │ │ │ │ │ - this.dragPan = null; │ │ │ │ │ - if (this.pinchZoom) { │ │ │ │ │ - this.pinchZoom.destroy(); │ │ │ │ │ - delete this.pinchZoom │ │ │ │ │ - } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.dragPan.activate(); │ │ │ │ │ - this.handlers.click.activate(); │ │ │ │ │ - this.pinchZoom.activate(); │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.dragPan.deactivate(); │ │ │ │ │ - this.handlers.click.deactivate(); │ │ │ │ │ - this.pinchZoom.deactivate(); │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - var clickCallbacks = { │ │ │ │ │ - click: this.defaultClick, │ │ │ │ │ - dblclick: this.defaultDblClick │ │ │ │ │ - }; │ │ │ │ │ - var clickOptions = OpenLayers.Util.extend({ │ │ │ │ │ - double: true, │ │ │ │ │ - stopDouble: true, │ │ │ │ │ - pixelTolerance: 2 │ │ │ │ │ - }, this.clickHandlerOptions); │ │ │ │ │ - this.handlers.click = new OpenLayers.Handler.Click(this, clickCallbacks, clickOptions); │ │ │ │ │ - this.dragPan = new OpenLayers.Control.DragPan(OpenLayers.Util.extend({ │ │ │ │ │ - map: this.map, │ │ │ │ │ - documentDrag: this.documentDrag │ │ │ │ │ - }, this.dragPanOptions)); │ │ │ │ │ - this.dragPan.draw(); │ │ │ │ │ - this.pinchZoom = new OpenLayers.Control.PinchZoom(OpenLayers.Util.extend({ │ │ │ │ │ - map: this.map │ │ │ │ │ - }, this.pinchZoomOptions)) │ │ │ │ │ - }, │ │ │ │ │ - defaultClick: function(evt) { │ │ │ │ │ - if (evt.lastTouches && evt.lastTouches.length == 2) { │ │ │ │ │ - this.map.zoomOut() │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - defaultDblClick: function(evt) { │ │ │ │ │ - this.map.zoomTo(this.map.zoom + 1, evt.xy) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.TouchNavigation" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.LayerSwitcher = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - layerStates: null, │ │ │ │ │ - layersDiv: null, │ │ │ │ │ - baseLayersDiv: null, │ │ │ │ │ - baseLayers: null, │ │ │ │ │ - dataLbl: null, │ │ │ │ │ - dataLayersDiv: null, │ │ │ │ │ - dataLayers: null, │ │ │ │ │ - minimizeDiv: null, │ │ │ │ │ - maximizeDiv: null, │ │ │ │ │ - ascending: true, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.layerStates = [] │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.clearLayersArray("base"); │ │ │ │ │ - this.clearLayersArray("data"); │ │ │ │ │ - this.map.events.un({ │ │ │ │ │ - buttonclick: this.onButtonClick, │ │ │ │ │ - addlayer: this.redraw, │ │ │ │ │ - changelayer: this.redraw, │ │ │ │ │ - removelayer: this.redraw, │ │ │ │ │ - changebaselayer: this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - this.events.unregister("buttonclick", this, this.onButtonClick); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - setMap: function(map) { │ │ │ │ │ - OpenLayers.Control.prototype.setMap.apply(this, arguments); │ │ │ │ │ - this.map.events.on({ │ │ │ │ │ - addlayer: this.redraw, │ │ │ │ │ - changelayer: this.redraw, │ │ │ │ │ - removelayer: this.redraw, │ │ │ │ │ - changebaselayer: this.redraw, │ │ │ │ │ - scope: this │ │ │ │ │ - }); │ │ │ │ │ - if (this.outsideViewport) { │ │ │ │ │ - this.events.attachToElement(this.div); │ │ │ │ │ - this.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ - } else { │ │ │ │ │ - this.map.events.register("buttonclick", this, this.onButtonClick) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this); │ │ │ │ │ - this.loadContents(); │ │ │ │ │ - if (!this.outsideViewport) { │ │ │ │ │ - this.minimizeControl() │ │ │ │ │ - } │ │ │ │ │ - this.redraw(); │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - onButtonClick: function(evt) { │ │ │ │ │ - var button = evt.buttonElement; │ │ │ │ │ - if (button === this.minimizeDiv) { │ │ │ │ │ - this.minimizeControl() │ │ │ │ │ - } else if (button === this.maximizeDiv) { │ │ │ │ │ - this.maximizeControl() │ │ │ │ │ - } else if (button._layerSwitcher === this.id) { │ │ │ │ │ - if (button["for"]) { │ │ │ │ │ - button = document.getElementById(button["for"]) │ │ │ │ │ - } │ │ │ │ │ - if (!button.disabled) { │ │ │ │ │ - if (button.type == "radio") { │ │ │ │ │ - button.checked = true; │ │ │ │ │ - this.map.setBaseLayer(this.map.getLayer(button._layer)) │ │ │ │ │ - } else { │ │ │ │ │ - button.checked = !button.checked; │ │ │ │ │ - this.updateMap() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - clearLayersArray: function(layersType) { │ │ │ │ │ - this[layersType + "LayersDiv"].innerHTML = ""; │ │ │ │ │ - this[layersType + "Layers"] = [] │ │ │ │ │ - }, │ │ │ │ │ - checkRedraw: function() { │ │ │ │ │ - if (!this.layerStates.length || this.map.layers.length != this.layerStates.length) { │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = this.layerStates.length; i < len; i++) { │ │ │ │ │ - var layerState = this.layerStates[i]; │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - if (layerState.name != layer.name || layerState.inRange != layer.inRange || layerState.id != layer.id || layerState.visibility != layer.visibility) { │ │ │ │ │ - return true │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - redraw: function() { │ │ │ │ │ - if (!this.checkRedraw()) { │ │ │ │ │ - return this.div │ │ │ │ │ - } │ │ │ │ │ - this.clearLayersArray("base"); │ │ │ │ │ - this.clearLayersArray("data"); │ │ │ │ │ - var containsOverlays = false; │ │ │ │ │ - var containsBaseLayers = false; │ │ │ │ │ - var len = this.map.layers.length; │ │ │ │ │ - this.layerStates = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - var layer = this.map.layers[i]; │ │ │ │ │ - this.layerStates[i] = { │ │ │ │ │ - name: layer.name, │ │ │ │ │ - visibility: layer.visibility, │ │ │ │ │ - inRange: layer.inRange, │ │ │ │ │ - id: layer.id │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - var layers = this.map.layers.slice(); │ │ │ │ │ - if (!this.ascending) { │ │ │ │ │ - layers.reverse() │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = layers.length; i < len; i++) { │ │ │ │ │ - var layer = layers[i]; │ │ │ │ │ - var baseLayer = layer.isBaseLayer; │ │ │ │ │ - if (layer.displayInLayerSwitcher) { │ │ │ │ │ - if (baseLayer) { │ │ │ │ │ - containsBaseLayers = true │ │ │ │ │ - } else { │ │ │ │ │ - containsOverlays = true │ │ │ │ │ - } │ │ │ │ │ - var checked = baseLayer ? layer == this.map.baseLayer : layer.getVisibility(); │ │ │ │ │ - var inputElem = document.createElement("input"), │ │ │ │ │ - inputId = OpenLayers.Util.createUniqueID(this.id + "_input_"); │ │ │ │ │ - inputElem.id = inputId; │ │ │ │ │ - inputElem.name = baseLayer ? this.id + "_baseLayers" : layer.name; │ │ │ │ │ - inputElem.type = baseLayer ? "radio" : "checkbox"; │ │ │ │ │ - inputElem.value = layer.name; │ │ │ │ │ - inputElem.checked = checked; │ │ │ │ │ - inputElem.defaultChecked = checked; │ │ │ │ │ - inputElem.className = "olButton"; │ │ │ │ │ - inputElem._layer = layer.id; │ │ │ │ │ - inputElem._layerSwitcher = this.id; │ │ │ │ │ - if (!baseLayer && !layer.inRange) { │ │ │ │ │ - inputElem.disabled = true │ │ │ │ │ - } │ │ │ │ │ - var labelSpan = document.createElement("label"); │ │ │ │ │ - labelSpan["for"] = inputElem.id; │ │ │ │ │ - OpenLayers.Element.addClass(labelSpan, "labelSpan olButton"); │ │ │ │ │ - labelSpan._layer = layer.id; │ │ │ │ │ - labelSpan._layerSwitcher = this.id; │ │ │ │ │ - if (!baseLayer && !layer.inRange) { │ │ │ │ │ - labelSpan.style.color = "gray" │ │ │ │ │ - } │ │ │ │ │ - labelSpan.innerHTML = layer.name; │ │ │ │ │ - labelSpan.style.verticalAlign = baseLayer ? "bottom" : "baseline"; │ │ │ │ │ - var br = document.createElement("br"); │ │ │ │ │ - var groupArray = baseLayer ? this.baseLayers : this.dataLayers; │ │ │ │ │ - groupArray.push({ │ │ │ │ │ - layer: layer, │ │ │ │ │ - inputElem: inputElem, │ │ │ │ │ - labelSpan: labelSpan │ │ │ │ │ - }); │ │ │ │ │ - var groupDiv = baseLayer ? this.baseLayersDiv : this.dataLayersDiv; │ │ │ │ │ - groupDiv.appendChild(inputElem); │ │ │ │ │ - groupDiv.appendChild(labelSpan); │ │ │ │ │ - groupDiv.appendChild(br) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - this.dataLbl.style.display = containsOverlays ? "" : "none"; │ │ │ │ │ - this.baseLbl.style.display = containsBaseLayers ? "" : "none"; │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - updateMap: function() { │ │ │ │ │ - for (var i = 0, len = this.baseLayers.length; i < len; i++) { │ │ │ │ │ - var layerEntry = this.baseLayers[i]; │ │ │ │ │ - if (layerEntry.inputElem.checked) { │ │ │ │ │ - this.map.setBaseLayer(layerEntry.layer, false) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - for (var i = 0, len = this.dataLayers.length; i < len; i++) { │ │ │ │ │ - var layerEntry = this.dataLayers[i]; │ │ │ │ │ - layerEntry.layer.setVisibility(layerEntry.inputElem.checked) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - maximizeControl: function(e) { │ │ │ │ │ - this.div.style.width = ""; │ │ │ │ │ - this.div.style.height = ""; │ │ │ │ │ - this.showControls(false); │ │ │ │ │ - if (e != null) { │ │ │ │ │ - OpenLayers.Event.stop(e) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - minimizeControl: function(e) { │ │ │ │ │ - this.div.style.width = "0px"; │ │ │ │ │ - this.div.style.height = "0px"; │ │ │ │ │ - this.showControls(true); │ │ │ │ │ - if (e != null) { │ │ │ │ │ - OpenLayers.Event.stop(e) │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - showControls: function(minimize) { │ │ │ │ │ - this.maximizeDiv.style.display = minimize ? "" : "none"; │ │ │ │ │ - this.minimizeDiv.style.display = minimize ? "none" : ""; │ │ │ │ │ - this.layersDiv.style.display = minimize ? "none" : "" │ │ │ │ │ - }, │ │ │ │ │ - loadContents: function() { │ │ │ │ │ - this.layersDiv = document.createElement("div"); │ │ │ │ │ - this.layersDiv.id = this.id + "_layersDiv"; │ │ │ │ │ - OpenLayers.Element.addClass(this.layersDiv, "layersDiv"); │ │ │ │ │ - this.baseLbl = document.createElement("div"); │ │ │ │ │ - this.baseLbl.innerHTML = OpenLayers.i18n("Base Layer"); │ │ │ │ │ - OpenLayers.Element.addClass(this.baseLbl, "baseLbl"); │ │ │ │ │ - this.baseLayersDiv = document.createElement("div"); │ │ │ │ │ - OpenLayers.Element.addClass(this.baseLayersDiv, "baseLayersDiv"); │ │ │ │ │ - this.dataLbl = document.createElement("div"); │ │ │ │ │ - this.dataLbl.innerHTML = OpenLayers.i18n("Overlays"); │ │ │ │ │ - OpenLayers.Element.addClass(this.dataLbl, "dataLbl"); │ │ │ │ │ - this.dataLayersDiv = document.createElement("div"); │ │ │ │ │ - OpenLayers.Element.addClass(this.dataLayersDiv, "dataLayersDiv"); │ │ │ │ │ - if (this.ascending) { │ │ │ │ │ - this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLayersDiv); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLayersDiv) │ │ │ │ │ - } else { │ │ │ │ │ - this.layersDiv.appendChild(this.dataLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.dataLayersDiv); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLbl); │ │ │ │ │ - this.layersDiv.appendChild(this.baseLayersDiv) │ │ │ │ │ - } │ │ │ │ │ - this.div.appendChild(this.layersDiv); │ │ │ │ │ - var img = OpenLayers.Util.getImageLocation("layer-switcher-maximize.png"); │ │ │ │ │ - this.maximizeDiv = OpenLayers.Util.createAlphaImageDiv("OpenLayers_Control_MaximizeDiv", null, null, img, "absolute"); │ │ │ │ │ - OpenLayers.Element.addClass(this.maximizeDiv, "maximizeDiv olButton"); │ │ │ │ │ - this.maximizeDiv.style.display = "none"; │ │ │ │ │ - this.div.appendChild(this.maximizeDiv); │ │ │ │ │ - var img = OpenLayers.Util.getImageLocation("layer-switcher-minimize.png"); │ │ │ │ │ - this.minimizeDiv = OpenLayers.Util.createAlphaImageDiv("OpenLayers_Control_MinimizeDiv", null, null, img, "absolute"); │ │ │ │ │ - OpenLayers.Element.addClass(this.minimizeDiv, "minimizeDiv olButton"); │ │ │ │ │ - this.minimizeDiv.style.display = "none"; │ │ │ │ │ - this.div.appendChild(this.minimizeDiv) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.LayerSwitcher" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.Graticule = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - autoActivate: true, │ │ │ │ │ - intervals: [45, 30, 20, 10, 5, 2, 1, .5, .2, .1, .05, .01, .005, .002, .001], │ │ │ │ │ - displayInLayerSwitcher: true, │ │ │ │ │ - visible: true, │ │ │ │ │ - numPoints: 50, │ │ │ │ │ - targetSize: 200, │ │ │ │ │ - layerName: null, │ │ │ │ │ - labelled: true, │ │ │ │ │ - labelFormat: "dm", │ │ │ │ │ - lineSymbolizer: { │ │ │ │ │ - strokeColor: "#333", │ │ │ │ │ - strokeWidth: 1, │ │ │ │ │ - strokeOpacity: .5 │ │ │ │ │ - }, │ │ │ │ │ - labelSymbolizer: {}, │ │ │ │ │ - gratLayer: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - options.layerName = options.layerName || OpenLayers.i18n("Graticule"); │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.labelSymbolizer.stroke = false; │ │ │ │ │ - this.labelSymbolizer.fill = false; │ │ │ │ │ - this.labelSymbolizer.label = "${label}"; │ │ │ │ │ - this.labelSymbolizer.labelAlign = "${labelAlign}"; │ │ │ │ │ - this.labelSymbolizer.labelXOffset = "${xOffset}"; │ │ │ │ │ - this.labelSymbolizer.labelYOffset = "${yOffset}" │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments); │ │ │ │ │ - if (this.gratLayer) { │ │ │ │ │ - this.gratLayer.destroy(); │ │ │ │ │ - this.gratLayer = null │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (!this.gratLayer) { │ │ │ │ │ - var gratStyle = new OpenLayers.Style({}, { │ │ │ │ │ - rules: [new OpenLayers.Rule({ │ │ │ │ │ - symbolizer: { │ │ │ │ │ - Point: this.labelSymbolizer, │ │ │ │ │ - Line: this.lineSymbolizer │ │ │ │ │ - } │ │ │ │ │ - })] │ │ │ │ │ - }); │ │ │ │ │ - this.gratLayer = new OpenLayers.Layer.Vector(this.layerName, { │ │ │ │ │ - styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ - default: gratStyle │ │ │ │ │ - }), │ │ │ │ │ - visibility: this.visible, │ │ │ │ │ - displayInLayerSwitcher: this.displayInLayerSwitcher │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - return this.div │ │ │ │ │ - }, │ │ │ │ │ - activate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.activate.apply(this, arguments)) { │ │ │ │ │ - this.map.addLayer(this.gratLayer); │ │ │ │ │ - this.map.events.register("moveend", this, this.update); │ │ │ │ │ - this.update(); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - deactivate: function() { │ │ │ │ │ - if (OpenLayers.Control.prototype.deactivate.apply(this, arguments)) { │ │ │ │ │ - this.map.events.unregister("moveend", this, this.update); │ │ │ │ │ - this.map.removeLayer(this.gratLayer); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ - }, │ │ │ │ │ - update: function() { │ │ │ │ │ - var mapBounds = this.map.getExtent(); │ │ │ │ │ - if (!mapBounds) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - this.gratLayer.destroyFeatures(); │ │ │ │ │ - var llProj = new OpenLayers.Projection("EPSG:4326"); │ │ │ │ │ - var mapProj = this.map.getProjectionObject(); │ │ │ │ │ - var mapRes = this.map.getResolution(); │ │ │ │ │ - if (mapProj.proj && mapProj.proj.projName == "longlat") { │ │ │ │ │ - this.numPoints = 1 │ │ │ │ │ - } │ │ │ │ │ - var mapCenter = this.map.getCenter(); │ │ │ │ │ - var mapCenterLL = new OpenLayers.Pixel(mapCenter.lon, mapCenter.lat); │ │ │ │ │ - OpenLayers.Projection.transform(mapCenterLL, mapProj, llProj); │ │ │ │ │ - var testSq = this.targetSize * mapRes; │ │ │ │ │ - testSq *= testSq; │ │ │ │ │ - var llInterval; │ │ │ │ │ - for (var i = 0; i < this.intervals.length; ++i) { │ │ │ │ │ - llInterval = this.intervals[i]; │ │ │ │ │ - var delta = llInterval / 2; │ │ │ │ │ - var p1 = mapCenterLL.offset({ │ │ │ │ │ - x: -delta, │ │ │ │ │ - y: -delta │ │ │ │ │ - }); │ │ │ │ │ - var p2 = mapCenterLL.offset({ │ │ │ │ │ - x: delta, │ │ │ │ │ - y: delta │ │ │ │ │ - }); │ │ │ │ │ - OpenLayers.Projection.transform(p1, llProj, mapProj); │ │ │ │ │ - OpenLayers.Projection.transform(p2, llProj, mapProj); │ │ │ │ │ - var distSq = (p1.x - p2.x) * (p1.x - p2.x) + (p1.y - p2.y) * (p1.y - p2.y); │ │ │ │ │ - if (distSq <= testSq) { │ │ │ │ │ - break │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - mapCenterLL.x = Math.floor(mapCenterLL.x / llInterval) * llInterval; │ │ │ │ │ - mapCenterLL.y = Math.floor(mapCenterLL.y / llInterval) * llInterval; │ │ │ │ │ - var iter = 0; │ │ │ │ │ - var centerLonPoints = [mapCenterLL.clone()]; │ │ │ │ │ - var newPoint = mapCenterLL.clone(); │ │ │ │ │ - var mapXY; │ │ │ │ │ - do { │ │ │ │ │ - newPoint = newPoint.offset({ │ │ │ │ │ - x: 0, │ │ │ │ │ - y: llInterval │ │ │ │ │ - }); │ │ │ │ │ - mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ - centerLonPoints.unshift(newPoint) │ │ │ │ │ - } while (mapBounds.containsPixel(mapXY) && ++iter < 1e3); │ │ │ │ │ - newPoint = mapCenterLL.clone(); │ │ │ │ │ - do { │ │ │ │ │ - newPoint = newPoint.offset({ │ │ │ │ │ - x: 0, │ │ │ │ │ - y: -llInterval │ │ │ │ │ - }); │ │ │ │ │ - mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ - centerLonPoints.push(newPoint) │ │ │ │ │ - } while (mapBounds.containsPixel(mapXY) && ++iter < 1e3); │ │ │ │ │ - iter = 0; │ │ │ │ │ - var centerLatPoints = [mapCenterLL.clone()]; │ │ │ │ │ - newPoint = mapCenterLL.clone(); │ │ │ │ │ - do { │ │ │ │ │ - newPoint = newPoint.offset({ │ │ │ │ │ - x: -llInterval, │ │ │ │ │ - y: 0 │ │ │ │ │ - }); │ │ │ │ │ - mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ - centerLatPoints.unshift(newPoint) │ │ │ │ │ - } while (mapBounds.containsPixel(mapXY) && ++iter < 1e3); │ │ │ │ │ - newPoint = mapCenterLL.clone(); │ │ │ │ │ - do { │ │ │ │ │ - newPoint = newPoint.offset({ │ │ │ │ │ - x: llInterval, │ │ │ │ │ - y: 0 │ │ │ │ │ - }); │ │ │ │ │ - mapXY = OpenLayers.Projection.transform(newPoint.clone(), llProj, mapProj); │ │ │ │ │ - centerLatPoints.push(newPoint) │ │ │ │ │ - } while (mapBounds.containsPixel(mapXY) && ++iter < 1e3); │ │ │ │ │ - var lines = []; │ │ │ │ │ - for (var i = 0; i < centerLatPoints.length; ++i) { │ │ │ │ │ - var lon = centerLatPoints[i].x; │ │ │ │ │ - var pointList = []; │ │ │ │ │ - var labelPoint = null; │ │ │ │ │ - var latEnd = Math.min(centerLonPoints[0].y, 90); │ │ │ │ │ - var latStart = Math.max(centerLonPoints[centerLonPoints.length - 1].y, -90); │ │ │ │ │ - var latDelta = (latEnd - latStart) / this.numPoints; │ │ │ │ │ - var lat = latStart; │ │ │ │ │ - for (var j = 0; j <= this.numPoints; ++j) { │ │ │ │ │ - var gridPoint = new OpenLayers.Geometry.Point(lon, lat); │ │ │ │ │ - gridPoint.transform(llProj, mapProj); │ │ │ │ │ - pointList.push(gridPoint); │ │ │ │ │ - lat += latDelta; │ │ │ │ │ - if (gridPoint.y >= mapBounds.bottom && !labelPoint) { │ │ │ │ │ - labelPoint = gridPoint │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.labelled) { │ │ │ │ │ - var labelPos = new OpenLayers.Geometry.Point(labelPoint.x, mapBounds.bottom); │ │ │ │ │ - var labelAttrs = { │ │ │ │ │ - value: lon, │ │ │ │ │ - label: this.labelled ? OpenLayers.Util.getFormattedLonLat(lon, "lon", this.labelFormat) : "", │ │ │ │ │ - labelAlign: "cb", │ │ │ │ │ - xOffset: 0, │ │ │ │ │ - yOffset: 2 │ │ │ │ │ - }; │ │ │ │ │ - this.gratLayer.addFeatures(new OpenLayers.Feature.Vector(labelPos, labelAttrs)) │ │ │ │ │ - } │ │ │ │ │ - var geom = new OpenLayers.Geometry.LineString(pointList); │ │ │ │ │ - lines.push(new OpenLayers.Feature.Vector(geom)) │ │ │ │ │ - } │ │ │ │ │ - for (var j = 0; j < centerLonPoints.length; ++j) { │ │ │ │ │ - lat = centerLonPoints[j].y; │ │ │ │ │ - if (lat < -90 || lat > 90) { │ │ │ │ │ - continue │ │ │ │ │ - } │ │ │ │ │ - var pointList = []; │ │ │ │ │ - var lonStart = centerLatPoints[0].x; │ │ │ │ │ - var lonEnd = centerLatPoints[centerLatPoints.length - 1].x; │ │ │ │ │ - var lonDelta = (lonEnd - lonStart) / this.numPoints; │ │ │ │ │ - var lon = lonStart; │ │ │ │ │ - var labelPoint = null; │ │ │ │ │ - for (var i = 0; i <= this.numPoints; ++i) { │ │ │ │ │ - var gridPoint = new OpenLayers.Geometry.Point(lon, lat); │ │ │ │ │ - gridPoint.transform(llProj, mapProj); │ │ │ │ │ - pointList.push(gridPoint); │ │ │ │ │ - lon += lonDelta; │ │ │ │ │ - if (gridPoint.x < mapBounds.right) { │ │ │ │ │ - labelPoint = gridPoint │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (this.labelled) { │ │ │ │ │ - var labelPos = new OpenLayers.Geometry.Point(mapBounds.right, labelPoint.y); │ │ │ │ │ - var labelAttrs = { │ │ │ │ │ - value: lat, │ │ │ │ │ - label: this.labelled ? OpenLayers.Util.getFormattedLonLat(lat, "lat", this.labelFormat) : "", │ │ │ │ │ - labelAlign: "rb", │ │ │ │ │ - xOffset: -2, │ │ │ │ │ - yOffset: 2 │ │ │ │ │ - }; │ │ │ │ │ - this.gratLayer.addFeatures(new OpenLayers.Feature.Vector(labelPos, labelAttrs)) │ │ │ │ │ - } │ │ │ │ │ - var geom = new OpenLayers.Geometry.LineString(pointList); │ │ │ │ │ - lines.push(new OpenLayers.Feature.Vector(geom)) │ │ │ │ │ - } │ │ │ │ │ - this.gratLayer.addFeatures(lines) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Graticule" │ │ │ │ │ -}); │ │ │ │ │ OpenLayers.Control.Measure = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ callbacks: null, │ │ │ │ │ displaySystem: "metric", │ │ │ │ │ geodesic: false, │ │ │ │ │ displaySystemUnits: { │ │ │ │ │ geographic: ["dd"], │ │ │ │ │ english: ["mi", "ft", "in"], │ │ │ │ │ @@ -37878,3493 +37906,3465 @@ │ │ │ │ │ var inPerMapUnit = OpenLayers.INCHES_PER_UNIT[geomUnits]; │ │ │ │ │ length *= inPerMapUnit / inPerDisplayUnit │ │ │ │ │ } │ │ │ │ │ return length │ │ │ │ │ }, │ │ │ │ │ CLASS_NAME: "OpenLayers.Control.Measure" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Control.Scale = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - element: null, │ │ │ │ │ - geodesic: false, │ │ │ │ │ - initialize: function(element, options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.element = OpenLayers.Util.getElement(element) │ │ │ │ │ +OpenLayers.Control.ZoomPanel = OpenLayers.Class(OpenLayers.Control.Panel, { │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Control.Panel.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.addControls([new OpenLayers.Control.ZoomIn, new OpenLayers.Control.ZoomToMaxExtent, new OpenLayers.Control.ZoomOut]) │ │ │ │ │ }, │ │ │ │ │ - draw: function() { │ │ │ │ │ - OpenLayers.Control.prototype.draw.apply(this, arguments); │ │ │ │ │ - if (!this.element) { │ │ │ │ │ - this.element = document.createElement("div"); │ │ │ │ │ - this.div.appendChild(this.element) │ │ │ │ │ - } │ │ │ │ │ - this.map.events.register("moveend", this, this.updateScale); │ │ │ │ │ - this.updateScale(); │ │ │ │ │ - return this.div │ │ │ │ │ + CLASS_NAME: "OpenLayers.Control.ZoomPanel" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Tile.UTFGrid = OpenLayers.Class(OpenLayers.Tile, { │ │ │ │ │ + url: null, │ │ │ │ │ + utfgridResolution: 2, │ │ │ │ │ + json: null, │ │ │ │ │ + format: null, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.clear(); │ │ │ │ │ + OpenLayers.Tile.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - updateScale: function() { │ │ │ │ │ - var scale; │ │ │ │ │ - if (this.geodesic === true) { │ │ │ │ │ - var units = this.map.getUnits(); │ │ │ │ │ - if (!units) { │ │ │ │ │ - return │ │ │ │ │ + draw: function() { │ │ │ │ │ + var drawn = OpenLayers.Tile.prototype.draw.apply(this, arguments); │ │ │ │ │ + if (drawn) { │ │ │ │ │ + if (this.isLoading) { │ │ │ │ │ + this.abortLoading(); │ │ │ │ │ + this.events.triggerEvent("reload") │ │ │ │ │ + } else { │ │ │ │ │ + this.isLoading = true; │ │ │ │ │ + this.events.triggerEvent("loadstart") │ │ │ │ │ + } │ │ │ │ │ + this.url = this.layer.getURL(this.bounds); │ │ │ │ │ + if (this.layer.useJSONP) { │ │ │ │ │ + var ols = new OpenLayers.Protocol.Script({ │ │ │ │ │ + url: this.url, │ │ │ │ │ + callback: function(response) { │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + this.json = response.data │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + ols.read(); │ │ │ │ │ + this.request = ols │ │ │ │ │ + } else { │ │ │ │ │ + this.request = OpenLayers.Request.GET({ │ │ │ │ │ + url: this.url, │ │ │ │ │ + callback: function(response) { │ │ │ │ │ + this.isLoading = false; │ │ │ │ │ + this.events.triggerEvent("loadend"); │ │ │ │ │ + if (response.status === 200) { │ │ │ │ │ + this.parseData(response.responseText) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - var inches = OpenLayers.INCHES_PER_UNIT; │ │ │ │ │ - scale = (this.map.getGeodesicPixelSize().w || 1e-6) * inches["km"] * OpenLayers.DOTS_PER_INCH │ │ │ │ │ } else { │ │ │ │ │ - scale = this.map.getScale() │ │ │ │ │ - } │ │ │ │ │ - if (!scale) { │ │ │ │ │ - return │ │ │ │ │ + this.unload() │ │ │ │ │ } │ │ │ │ │ - if (scale >= 9500 && scale <= 95e4) { │ │ │ │ │ - scale = Math.round(scale / 1e3) + "K" │ │ │ │ │ - } else if (scale >= 95e4) { │ │ │ │ │ - scale = Math.round(scale / 1e6) + "M" │ │ │ │ │ - } else { │ │ │ │ │ - scale = Math.round(scale) │ │ │ │ │ + return drawn │ │ │ │ │ + }, │ │ │ │ │ + abortLoading: function() { │ │ │ │ │ + if (this.request) { │ │ │ │ │ + this.request.abort(); │ │ │ │ │ + delete this.request │ │ │ │ │ } │ │ │ │ │ - this.element.innerHTML = OpenLayers.i18n("Scale = 1 : ${scaleDenom}", { │ │ │ │ │ - scaleDenom: scale │ │ │ │ │ - }) │ │ │ │ │ + this.isLoading = false │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.Scale" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Control.SLDSelect = OpenLayers.Class(OpenLayers.Control, { │ │ │ │ │ - clearOnDeactivate: false, │ │ │ │ │ - layers: null, │ │ │ │ │ - callbacks: null, │ │ │ │ │ - selectionSymbolizer: { │ │ │ │ │ - Polygon: { │ │ │ │ │ - fillColor: "#FF0000", │ │ │ │ │ - stroke: false │ │ │ │ │ - }, │ │ │ │ │ - Line: { │ │ │ │ │ - strokeColor: "#FF0000", │ │ │ │ │ - strokeWidth: 2 │ │ │ │ │ - }, │ │ │ │ │ - Point: { │ │ │ │ │ - graphicName: "square", │ │ │ │ │ - fillColor: "#FF0000", │ │ │ │ │ - pointRadius: 5 │ │ │ │ │ + getFeatureInfo: function(i, j) { │ │ │ │ │ + var info = null; │ │ │ │ │ + if (this.json) { │ │ │ │ │ + var id = this.getFeatureId(i, j); │ │ │ │ │ + if (id !== null) { │ │ │ │ │ + info = { │ │ │ │ │ + id: id, │ │ │ │ │ + data: this.json.data[id] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return info │ │ │ │ │ }, │ │ │ │ │ - layerOptions: null, │ │ │ │ │ - sketchStyle: null, │ │ │ │ │ - wfsCache: {}, │ │ │ │ │ - layerCache: {}, │ │ │ │ │ - initialize: function(handler, options) { │ │ │ │ │ - OpenLayers.Control.prototype.initialize.apply(this, [options]); │ │ │ │ │ - this.callbacks = OpenLayers.Util.extend({ │ │ │ │ │ - done: this.select, │ │ │ │ │ - click: this.select │ │ │ │ │ - }, this.callbacks); │ │ │ │ │ - this.handlerOptions = this.handlerOptions || {}; │ │ │ │ │ - this.layerOptions = OpenLayers.Util.applyDefaults(this.layerOptions, { │ │ │ │ │ - displayInLayerSwitcher: false, │ │ │ │ │ - tileOptions: { │ │ │ │ │ - maxGetUrlLength: 2048 │ │ │ │ │ + getFeatureId: function(i, j) { │ │ │ │ │ + var id = null; │ │ │ │ │ + if (this.json) { │ │ │ │ │ + var resolution = this.utfgridResolution; │ │ │ │ │ + var row = Math.floor(j / resolution); │ │ │ │ │ + var col = Math.floor(i / resolution); │ │ │ │ │ + var charCode = this.json.grid[row].charCodeAt(col); │ │ │ │ │ + var index = this.indexFromCharCode(charCode); │ │ │ │ │ + var keys = this.json.keys; │ │ │ │ │ + if (!isNaN(index) && index in keys) { │ │ │ │ │ + id = keys[index] │ │ │ │ │ } │ │ │ │ │ - }); │ │ │ │ │ - if (this.sketchStyle) { │ │ │ │ │ - this.handlerOptions.layerOptions = OpenLayers.Util.applyDefaults(this.handlerOptions.layerOptions, { │ │ │ │ │ - styleMap: new OpenLayers.StyleMap({ │ │ │ │ │ - default: this.sketchStyle │ │ │ │ │ - }) │ │ │ │ │ - }) │ │ │ │ │ } │ │ │ │ │ - this.handler = new handler(this, this.callbacks, this.handlerOptions) │ │ │ │ │ + return id │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - for (var key in this.layerCache) { │ │ │ │ │ - delete this.layerCache[key] │ │ │ │ │ + indexFromCharCode: function(charCode) { │ │ │ │ │ + if (charCode >= 93) { │ │ │ │ │ + charCode-- │ │ │ │ │ } │ │ │ │ │ - for (var key in this.wfsCache) { │ │ │ │ │ - delete this.wfsCache[key] │ │ │ │ │ + if (charCode >= 35) { │ │ │ │ │ + charCode-- │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Control.prototype.destroy.apply(this, arguments) │ │ │ │ │ - }, │ │ │ │ │ - coupleLayerVisiblity: function(evt) { │ │ │ │ │ - this.setVisibility(evt.object.getVisibility()) │ │ │ │ │ + return charCode - 32 │ │ │ │ │ }, │ │ │ │ │ - createSelectionLayer: function(source) { │ │ │ │ │ - var selectionLayer; │ │ │ │ │ - if (!this.layerCache[source.id]) { │ │ │ │ │ - selectionLayer = new OpenLayers.Layer.WMS(source.name, source.url, source.params, OpenLayers.Util.applyDefaults(this.layerOptions, source.getOptions())); │ │ │ │ │ - this.layerCache[source.id] = selectionLayer; │ │ │ │ │ - if (this.layerOptions.displayInLayerSwitcher === false) { │ │ │ │ │ - source.events.on({ │ │ │ │ │ - visibilitychanged: this.coupleLayerVisiblity, │ │ │ │ │ - scope: selectionLayer │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - this.map.addLayer(selectionLayer) │ │ │ │ │ - } else { │ │ │ │ │ - selectionLayer = this.layerCache[source.id] │ │ │ │ │ + parseData: function(str) { │ │ │ │ │ + if (!this.format) { │ │ │ │ │ + this.format = new OpenLayers.Format.JSON │ │ │ │ │ } │ │ │ │ │ - return selectionLayer │ │ │ │ │ + this.json = this.format.read(str) │ │ │ │ │ }, │ │ │ │ │ - createSLD: function(layer, filters, geometryAttributes) { │ │ │ │ │ - var sld = { │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - namedLayers: {} │ │ │ │ │ - }; │ │ │ │ │ - var layerNames = [layer.params.LAYERS].join(",").split(","); │ │ │ │ │ - for (var i = 0, len = layerNames.length; i < len; i++) { │ │ │ │ │ - var name = layerNames[i]; │ │ │ │ │ - sld.namedLayers[name] = { │ │ │ │ │ - name: name, │ │ │ │ │ - userStyles: [] │ │ │ │ │ - }; │ │ │ │ │ - var symbolizer = this.selectionSymbolizer; │ │ │ │ │ - var geometryAttribute = geometryAttributes[i]; │ │ │ │ │ - if (geometryAttribute.type.indexOf("Polygon") >= 0) { │ │ │ │ │ - symbolizer = { │ │ │ │ │ - Polygon: this.selectionSymbolizer["Polygon"] │ │ │ │ │ - } │ │ │ │ │ - } else if (geometryAttribute.type.indexOf("LineString") >= 0) { │ │ │ │ │ - symbolizer = { │ │ │ │ │ - Line: this.selectionSymbolizer["Line"] │ │ │ │ │ + clear: function() { │ │ │ │ │ + this.json = null │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Tile.UTFGrid" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Tile.Image.IFrame = { │ │ │ │ │ + useIFrame: null, │ │ │ │ │ + blankImageUrl: "data:image/gif;base64,R0lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAQAIBRAA7", │ │ │ │ │ + draw: function() { │ │ │ │ │ + var draw = OpenLayers.Tile.Image.prototype.shouldDraw.call(this); │ │ │ │ │ + if (draw) { │ │ │ │ │ + var url = this.layer.getURL(this.bounds); │ │ │ │ │ + var usedIFrame = this.useIFrame; │ │ │ │ │ + this.useIFrame = this.maxGetUrlLength !== null && !this.layer.async && url.length > this.maxGetUrlLength; │ │ │ │ │ + var fromIFrame = usedIFrame && !this.useIFrame; │ │ │ │ │ + var toIFrame = !usedIFrame && this.useIFrame; │ │ │ │ │ + if (fromIFrame || toIFrame) { │ │ │ │ │ + if (this.imgDiv && this.imgDiv.parentNode === this.frame) { │ │ │ │ │ + this.frame.removeChild(this.imgDiv) │ │ │ │ │ } │ │ │ │ │ - } else if (geometryAttribute.type.indexOf("Point") >= 0) { │ │ │ │ │ - symbolizer = { │ │ │ │ │ - Point: this.selectionSymbolizer["Point"] │ │ │ │ │ + this.imgDiv = null; │ │ │ │ │ + if (fromIFrame) { │ │ │ │ │ + this.frame.removeChild(this.frame.firstChild) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var filter = filters[i]; │ │ │ │ │ - sld.namedLayers[name].userStyles.push({ │ │ │ │ │ - name: "default", │ │ │ │ │ - rules: [new OpenLayers.Rule({ │ │ │ │ │ - symbolizer: symbolizer, │ │ │ │ │ - filter: filter, │ │ │ │ │ - maxScaleDenominator: layer.options.minScale │ │ │ │ │ - })] │ │ │ │ │ - }) │ │ │ │ │ } │ │ │ │ │ - return new OpenLayers.Format.SLD({ │ │ │ │ │ - srsName: this.map.getProjection() │ │ │ │ │ - }).write(sld) │ │ │ │ │ + return OpenLayers.Tile.Image.prototype.draw.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - parseDescribeLayer: function(request) { │ │ │ │ │ - var format = new OpenLayers.Format.WMSDescribeLayer; │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ - } │ │ │ │ │ - var describeLayer = format.read(doc); │ │ │ │ │ - var typeNames = []; │ │ │ │ │ - var url = null; │ │ │ │ │ - for (var i = 0, len = describeLayer.length; i < len; i++) { │ │ │ │ │ - if (describeLayer[i].owsType == "WFS") { │ │ │ │ │ - typeNames.push(describeLayer[i].typeName); │ │ │ │ │ - url = describeLayer[i].owsURL │ │ │ │ │ + getImage: function() { │ │ │ │ │ + if (this.useIFrame === true) { │ │ │ │ │ + if (!this.frame.childNodes.length) { │ │ │ │ │ + var eventPane = document.createElement("div"), │ │ │ │ │ + style = eventPane.style; │ │ │ │ │ + style.position = "absolute"; │ │ │ │ │ + style.width = "100%"; │ │ │ │ │ + style.height = "100%"; │ │ │ │ │ + style.zIndex = 1; │ │ │ │ │ + style.backgroundImage = "url(" + this.blankImageUrl + ")"; │ │ │ │ │ + this.frame.appendChild(eventPane) │ │ │ │ │ } │ │ │ │ │ + var id = this.id + "_iFrame", │ │ │ │ │ + iframe; │ │ │ │ │ + if (parseFloat(navigator.appVersion.split("MSIE")[1]) < 9) { │ │ │ │ │ + iframe = document.createElement('<iframe name="' + id + '">'); │ │ │ │ │ + iframe.style.backgroundColor = "#FFFFFF"; │ │ │ │ │ + iframe.style.filter = "chroma(color=#FFFFFF)" │ │ │ │ │ + } else { │ │ │ │ │ + iframe = document.createElement("iframe"); │ │ │ │ │ + iframe.style.backgroundColor = "transparent"; │ │ │ │ │ + iframe.name = id │ │ │ │ │ + } │ │ │ │ │ + iframe.scrolling = "no"; │ │ │ │ │ + iframe.marginWidth = "0px"; │ │ │ │ │ + iframe.marginHeight = "0px"; │ │ │ │ │ + iframe.frameBorder = "0"; │ │ │ │ │ + iframe.style.position = "absolute"; │ │ │ │ │ + iframe.style.width = "100%"; │ │ │ │ │ + iframe.style.height = "100%"; │ │ │ │ │ + if (this.layer.opacity < 1) { │ │ │ │ │ + OpenLayers.Util.modifyDOMElement(iframe, null, null, null, null, null, null, this.layer.opacity) │ │ │ │ │ + } │ │ │ │ │ + this.frame.appendChild(iframe); │ │ │ │ │ + this.imgDiv = iframe; │ │ │ │ │ + return iframe │ │ │ │ │ + } else { │ │ │ │ │ + return OpenLayers.Tile.Image.prototype.getImage.apply(this, arguments) │ │ │ │ │ } │ │ │ │ │ - var options = { │ │ │ │ │ - url: url, │ │ │ │ │ - params: { │ │ │ │ │ - SERVICE: "WFS", │ │ │ │ │ - TYPENAME: typeNames.toString(), │ │ │ │ │ - REQUEST: "DescribeFeatureType", │ │ │ │ │ - VERSION: "1.0.0" │ │ │ │ │ - }, │ │ │ │ │ - callback: function(request) { │ │ │ │ │ - var format = new OpenLayers.Format.WFSDescribeFeatureType; │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ - } │ │ │ │ │ - var describeFeatureType = format.read(doc); │ │ │ │ │ - this.control.wfsCache[this.layer.id] = describeFeatureType; │ │ │ │ │ - this.control._queue && this.control.applySelection() │ │ │ │ │ - }, │ │ │ │ │ - scope: this │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Request.GET(options) │ │ │ │ │ }, │ │ │ │ │ - getGeometryAttributes: function(layer) { │ │ │ │ │ - var result = []; │ │ │ │ │ - var cache = this.wfsCache[layer.id]; │ │ │ │ │ - for (var i = 0, len = cache.featureTypes.length; i < len; i++) { │ │ │ │ │ - var typeName = cache.featureTypes[i]; │ │ │ │ │ - var properties = typeName.properties; │ │ │ │ │ - for (var j = 0, lenj = properties.length; j < lenj; j++) { │ │ │ │ │ - var property = properties[j]; │ │ │ │ │ - var type = property.type; │ │ │ │ │ - if (type.indexOf("LineString") >= 0 || type.indexOf("GeometryAssociationType") >= 0 || type.indexOf("GeometryPropertyType") >= 0 || type.indexOf("Point") >= 0 || type.indexOf("Polygon") >= 0) { │ │ │ │ │ - result.push(property) │ │ │ │ │ - } │ │ │ │ │ + createRequestForm: function() { │ │ │ │ │ + var form = document.createElement("form"); │ │ │ │ │ + form.method = "POST"; │ │ │ │ │ + var cacheId = this.layer.params["_OLSALT"]; │ │ │ │ │ + cacheId = (cacheId ? cacheId + "_" : "") + this.bounds.toBBOX(); │ │ │ │ │ + form.action = OpenLayers.Util.urlAppend(this.layer.url, cacheId); │ │ │ │ │ + form.target = this.id + "_iFrame"; │ │ │ │ │ + var imageSize = this.layer.getImageSize(), │ │ │ │ │ + params = OpenLayers.Util.getParameters(this.url), │ │ │ │ │ + field; │ │ │ │ │ + for (var par in params) { │ │ │ │ │ + field = document.createElement("input"); │ │ │ │ │ + field.type = "hidden"; │ │ │ │ │ + field.name = par; │ │ │ │ │ + field.value = params[par]; │ │ │ │ │ + form.appendChild(field) │ │ │ │ │ + } │ │ │ │ │ + return form │ │ │ │ │ + }, │ │ │ │ │ + setImgSrc: function(url) { │ │ │ │ │ + if (this.useIFrame === true) { │ │ │ │ │ + if (url) { │ │ │ │ │ + var form = this.createRequestForm(); │ │ │ │ │ + this.frame.appendChild(form); │ │ │ │ │ + form.submit(); │ │ │ │ │ + this.frame.removeChild(form) │ │ │ │ │ + } else if (this.imgDiv.parentNode === this.frame) { │ │ │ │ │ + this.frame.removeChild(this.imgDiv); │ │ │ │ │ + this.imgDiv = null │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Tile.Image.prototype.setImgSrc.apply(this, arguments) │ │ │ │ │ } │ │ │ │ │ - return result │ │ │ │ │ }, │ │ │ │ │ + onImageLoad: function() { │ │ │ │ │ + OpenLayers.Tile.Image.prototype.onImageLoad.apply(this, arguments); │ │ │ │ │ + if (this.useIFrame === true) { │ │ │ │ │ + this.imgDiv.style.opacity = 1; │ │ │ │ │ + this.frame.style.opacity = this.layer.opacity │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + createBackBuffer: function() { │ │ │ │ │ + var backBuffer; │ │ │ │ │ + if (this.useIFrame === false) { │ │ │ │ │ + backBuffer = OpenLayers.Tile.Image.prototype.createBackBuffer.call(this) │ │ │ │ │ + } │ │ │ │ │ + return backBuffer │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Strategy.BBOX = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + bounds: null, │ │ │ │ │ + resolution: null, │ │ │ │ │ + ratio: 2, │ │ │ │ │ + resFactor: null, │ │ │ │ │ + response: null, │ │ │ │ │ activate: function() { │ │ │ │ │ - var activated = OpenLayers.Control.prototype.activate.call(this); │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ if (activated) { │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.layers[i]; │ │ │ │ │ - if (layer && !this.wfsCache[layer.id]) { │ │ │ │ │ - var options = { │ │ │ │ │ - url: layer.url, │ │ │ │ │ - params: { │ │ │ │ │ - SERVICE: "WMS", │ │ │ │ │ - VERSION: layer.params.VERSION, │ │ │ │ │ - LAYERS: layer.params.LAYERS, │ │ │ │ │ - REQUEST: "DescribeLayer" │ │ │ │ │ - }, │ │ │ │ │ - callback: this.parseDescribeLayer, │ │ │ │ │ - scope: { │ │ │ │ │ - layer: layer, │ │ │ │ │ - control: this │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - OpenLayers.Request.GET(options) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + moveend: this.update, │ │ │ │ │ + refresh: this.update, │ │ │ │ │ + visibilitychanged: this.update, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.update() │ │ │ │ │ } │ │ │ │ │ return activated │ │ │ │ │ }, │ │ │ │ │ deactivate: function() { │ │ │ │ │ - var deactivated = OpenLayers.Control.prototype.deactivate.call(this); │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ if (deactivated) { │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.layers[i]; │ │ │ │ │ - if (layer && this.clearOnDeactivate === true) { │ │ │ │ │ - var layerCache = this.layerCache; │ │ │ │ │ - var selectionLayer = layerCache[layer.id]; │ │ │ │ │ - if (selectionLayer) { │ │ │ │ │ - layer.events.un({ │ │ │ │ │ - visibilitychanged: this.coupleLayerVisiblity, │ │ │ │ │ - scope: selectionLayer │ │ │ │ │ - }); │ │ │ │ │ - selectionLayer.destroy(); │ │ │ │ │ - delete layerCache[layer.id] │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + moveend: this.update, │ │ │ │ │ + refresh: this.update, │ │ │ │ │ + visibilitychanged: this.update, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ return deactivated │ │ │ │ │ }, │ │ │ │ │ - setLayers: function(layers) { │ │ │ │ │ - if (this.active) { │ │ │ │ │ - this.deactivate(); │ │ │ │ │ - this.layers = layers; │ │ │ │ │ - this.activate() │ │ │ │ │ - } else { │ │ │ │ │ - this.layers = layers │ │ │ │ │ + update: function(options) { │ │ │ │ │ + var mapBounds = this.getMapBounds(); │ │ │ │ │ + if (mapBounds !== null && (options && options.force || this.layer.visibility && this.layer.calculateInRange() && this.invalidBounds(mapBounds))) { │ │ │ │ │ + this.calculateBounds(mapBounds); │ │ │ │ │ + this.resolution = this.layer.map.getResolution(); │ │ │ │ │ + this.triggerRead(options) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - createFilter: function(geometryAttribute, geometry) { │ │ │ │ │ - var filter = null; │ │ │ │ │ - if (this.handler instanceof OpenLayers.Handler.RegularPolygon) { │ │ │ │ │ - if (this.handler.irregular === true) { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry.getBounds() │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } else if (this.handler instanceof OpenLayers.Handler.Polygon) { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry │ │ │ │ │ + getMapBounds: function() { │ │ │ │ │ + if (this.layer.map === null) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ + var bounds = this.layer.map.getExtent(); │ │ │ │ │ + if (bounds && !this.layer.projection.equals(this.layer.map.getProjectionObject())) { │ │ │ │ │ + bounds = bounds.clone().transform(this.layer.map.getProjectionObject(), this.layer.projection) │ │ │ │ │ + } │ │ │ │ │ + return bounds │ │ │ │ │ + }, │ │ │ │ │ + invalidBounds: function(mapBounds) { │ │ │ │ │ + if (!mapBounds) { │ │ │ │ │ + mapBounds = this.getMapBounds() │ │ │ │ │ + } │ │ │ │ │ + var invalid = !this.bounds || !this.bounds.containsBounds(mapBounds); │ │ │ │ │ + if (!invalid && this.resFactor) { │ │ │ │ │ + var ratio = this.resolution / this.layer.map.getResolution(); │ │ │ │ │ + invalid = ratio >= this.resFactor || ratio <= 1 / this.resFactor │ │ │ │ │ + } │ │ │ │ │ + return invalid │ │ │ │ │ + }, │ │ │ │ │ + calculateBounds: function(mapBounds) { │ │ │ │ │ + if (!mapBounds) { │ │ │ │ │ + mapBounds = this.getMapBounds() │ │ │ │ │ + } │ │ │ │ │ + var center = mapBounds.getCenterLonLat(); │ │ │ │ │ + var dataWidth = mapBounds.getWidth() * this.ratio; │ │ │ │ │ + var dataHeight = mapBounds.getHeight() * this.ratio; │ │ │ │ │ + this.bounds = new OpenLayers.Bounds(center.lon - dataWidth / 2, center.lat - dataHeight / 2, center.lon + dataWidth / 2, center.lat + dataHeight / 2) │ │ │ │ │ + }, │ │ │ │ │ + triggerRead: function(options) { │ │ │ │ │ + if (this.response && !(options && options.noAbort === true)) { │ │ │ │ │ + this.layer.protocol.abort(this.response); │ │ │ │ │ + this.layer.events.triggerEvent("loadend") │ │ │ │ │ + } │ │ │ │ │ + var evt = { │ │ │ │ │ + filter: this.createFilter() │ │ │ │ │ + }; │ │ │ │ │ + this.layer.events.triggerEvent("loadstart", evt); │ │ │ │ │ + this.response = this.layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ + filter: evt.filter, │ │ │ │ │ + callback: this.merge, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options)) │ │ │ │ │ + }, │ │ │ │ │ + createFilter: function() { │ │ │ │ │ + var filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ + type: OpenLayers.Filter.Spatial.BBOX, │ │ │ │ │ + value: this.bounds, │ │ │ │ │ + projection: this.layer.projection │ │ │ │ │ + }); │ │ │ │ │ + if (this.layer.filter) { │ │ │ │ │ + filter = new OpenLayers.Filter.Logical({ │ │ │ │ │ + type: OpenLayers.Filter.Logical.AND, │ │ │ │ │ + filters: [this.layer.filter, filter] │ │ │ │ │ }) │ │ │ │ │ - } else if (this.handler instanceof OpenLayers.Handler.Path) { │ │ │ │ │ - if (geometryAttribute.type.indexOf("Point") >= 0) { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - distance: this.map.getExtent().getWidth() * .01, │ │ │ │ │ - distanceUnits: this.map.getUnits(), │ │ │ │ │ - value: geometry │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ - } else if (this.handler instanceof OpenLayers.Handler.Click) { │ │ │ │ │ - if (geometryAttribute.type.indexOf("Polygon") >= 0) { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.INTERSECTS, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - value: geometry │ │ │ │ │ - }) │ │ │ │ │ - } else { │ │ │ │ │ - filter = new OpenLayers.Filter.Spatial({ │ │ │ │ │ - type: OpenLayers.Filter.Spatial.DWITHIN, │ │ │ │ │ - property: geometryAttribute.name, │ │ │ │ │ - distance: this.map.getExtent().getWidth() * .01, │ │ │ │ │ - distanceUnits: this.map.getUnits(), │ │ │ │ │ - value: geometry │ │ │ │ │ - }) │ │ │ │ │ - } │ │ │ │ │ } │ │ │ │ │ return filter │ │ │ │ │ }, │ │ │ │ │ - select: function(geometry) { │ │ │ │ │ - this._queue = function() { │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - var layer = this.layers[i]; │ │ │ │ │ - var geometryAttributes = this.getGeometryAttributes(layer); │ │ │ │ │ - var filters = []; │ │ │ │ │ - for (var j = 0, lenj = geometryAttributes.length; j < lenj; j++) { │ │ │ │ │ - var geometryAttribute = geometryAttributes[j]; │ │ │ │ │ - if (geometryAttribute !== null) { │ │ │ │ │ - if (!(geometry instanceof OpenLayers.Geometry)) { │ │ │ │ │ - var point = this.map.getLonLatFromPixel(geometry.xy); │ │ │ │ │ - geometry = new OpenLayers.Geometry.Point(point.lon, point.lat) │ │ │ │ │ - } │ │ │ │ │ - var filter = this.createFilter(geometryAttribute, geometry); │ │ │ │ │ - if (filter !== null) { │ │ │ │ │ - filters.push(filter) │ │ │ │ │ + merge: function(resp) { │ │ │ │ │ + this.layer.destroyFeatures(); │ │ │ │ │ + if (resp.success()) { │ │ │ │ │ + var features = resp.features; │ │ │ │ │ + if (features && features.length > 0) { │ │ │ │ │ + var remote = this.layer.projection; │ │ │ │ │ + var local = this.layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var geom; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + geom = features[i].geometry; │ │ │ │ │ + if (geom) { │ │ │ │ │ + geom.transform(remote, local) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var selectionLayer = this.createSelectionLayer(layer); │ │ │ │ │ - this.events.triggerEvent("selected", { │ │ │ │ │ - layer: layer, │ │ │ │ │ - filters: filters │ │ │ │ │ - }); │ │ │ │ │ - var sld = this.createSLD(layer, filters, geometryAttributes); │ │ │ │ │ - selectionLayer.mergeNewParams({ │ │ │ │ │ - SLD_BODY: sld │ │ │ │ │ - }); │ │ │ │ │ - delete this._queue │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - this.applySelection() │ │ │ │ │ - }, │ │ │ │ │ - applySelection: function() { │ │ │ │ │ - var canApply = true; │ │ │ │ │ - for (var i = 0, len = this.layers.length; i < len; i++) { │ │ │ │ │ - if (!this.wfsCache[this.layers[i].id]) { │ │ │ │ │ - canApply = false; │ │ │ │ │ - break │ │ │ │ │ + this.layer.addFeatures(features) │ │ │ │ │ } │ │ │ │ │ + } else { │ │ │ │ │ + this.bounds = null │ │ │ │ │ } │ │ │ │ │ - canApply && this._queue.call(this) │ │ │ │ │ + this.response = null; │ │ │ │ │ + this.layer.events.triggerEvent("loadend", { │ │ │ │ │ + response: resp │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Control.SLDSelect" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.BBOX" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Renderer.Canvas = OpenLayers.Class(OpenLayers.Renderer, { │ │ │ │ │ - hitDetection: true, │ │ │ │ │ - hitOverflow: 0, │ │ │ │ │ - canvas: null, │ │ │ │ │ - features: null, │ │ │ │ │ - pendingRedraw: false, │ │ │ │ │ - cachedSymbolBounds: {}, │ │ │ │ │ - initialize: function(containerID, options) { │ │ │ │ │ - OpenLayers.Renderer.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.root = document.createElement("canvas"); │ │ │ │ │ - this.container.appendChild(this.root); │ │ │ │ │ - this.canvas = this.root.getContext("2d"); │ │ │ │ │ - this.features = {}; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitCanvas = document.createElement("canvas"); │ │ │ │ │ - this.hitContext = this.hitCanvas.getContext("2d") │ │ │ │ │ +OpenLayers.Strategy.Filter = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + filter: null, │ │ │ │ │ + cache: null, │ │ │ │ │ + caching: false, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.cache = []; │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + beforefeaturesadded: this.handleAdd, │ │ │ │ │ + beforefeaturesremoved: this.handleRemove, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - setExtent: function() { │ │ │ │ │ - OpenLayers.Renderer.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - return false │ │ │ │ │ - }, │ │ │ │ │ - eraseGeometry: function(geometry, featureId) { │ │ │ │ │ - this.eraseFeatures(this.features[featureId][0]) │ │ │ │ │ - }, │ │ │ │ │ - supported: function() { │ │ │ │ │ - return OpenLayers.CANVAS_SUPPORTED │ │ │ │ │ - }, │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - this.size = size.clone(); │ │ │ │ │ - var root = this.root; │ │ │ │ │ - root.style.width = size.w + "px"; │ │ │ │ │ - root.style.height = size.h + "px"; │ │ │ │ │ - root.width = size.w; │ │ │ │ │ - root.height = size.h; │ │ │ │ │ - this.resolution = null; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - var hitCanvas = this.hitCanvas; │ │ │ │ │ - hitCanvas.style.width = size.w + "px"; │ │ │ │ │ - hitCanvas.style.height = size.h + "px"; │ │ │ │ │ - hitCanvas.width = size.w; │ │ │ │ │ - hitCanvas.height = size.h │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + this.cache = null; │ │ │ │ │ + if (this.layer && this.layer.events) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + beforefeaturesadded: this.handleAdd, │ │ │ │ │ + beforefeaturesremoved: this.handleRemove, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ + return OpenLayers.Strategy.prototype.deactivate.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - drawFeature: function(feature, style) { │ │ │ │ │ - var rendered; │ │ │ │ │ - if (feature.geometry) { │ │ │ │ │ - style = this.applyDefaultSymbolizer(style || feature.style); │ │ │ │ │ - var bounds = feature.geometry.getBounds(); │ │ │ │ │ - var worldBounds; │ │ │ │ │ - if (this.map.baseLayer && this.map.baseLayer.wrapDateLine) { │ │ │ │ │ - worldBounds = this.map.getMaxExtent() │ │ │ │ │ - } │ │ │ │ │ - var intersects = bounds && bounds.intersectsBounds(this.extent, { │ │ │ │ │ - worldBounds: worldBounds │ │ │ │ │ - }); │ │ │ │ │ - rendered = style.display !== "none" && !!bounds && intersects; │ │ │ │ │ - if (rendered) { │ │ │ │ │ - this.features[feature.id] = [feature, style] │ │ │ │ │ - } else { │ │ │ │ │ - delete this.features[feature.id] │ │ │ │ │ + handleAdd: function(event) { │ │ │ │ │ + if (!this.caching && this.filter) { │ │ │ │ │ + var features = event.features; │ │ │ │ │ + event.features = []; │ │ │ │ │ + var feature; │ │ │ │ │ + for (var i = 0, ii = features.length; i < ii; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + if (this.filter.evaluate(feature)) { │ │ │ │ │ + event.features.push(feature) │ │ │ │ │ + } else { │ │ │ │ │ + this.cache.push(feature) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.pendingRedraw = true │ │ │ │ │ } │ │ │ │ │ - if (this.pendingRedraw && !this.locked) { │ │ │ │ │ - this.redraw(); │ │ │ │ │ - this.pendingRedraw = false │ │ │ │ │ + }, │ │ │ │ │ + handleRemove: function(event) { │ │ │ │ │ + if (!this.caching) { │ │ │ │ │ + this.cache = [] │ │ │ │ │ } │ │ │ │ │ - return rendered │ │ │ │ │ }, │ │ │ │ │ - drawGeometry: function(geometry, style, featureId) { │ │ │ │ │ - var className = geometry.CLASS_NAME; │ │ │ │ │ - if (className == "OpenLayers.Geometry.Collection" || className == "OpenLayers.Geometry.MultiPoint" || className == "OpenLayers.Geometry.MultiLineString" || className == "OpenLayers.Geometry.MultiPolygon") { │ │ │ │ │ - for (var i = 0; i < geometry.components.length; i++) { │ │ │ │ │ - this.drawGeometry(geometry.components[i], style, featureId) │ │ │ │ │ + setFilter: function(filter) { │ │ │ │ │ + this.filter = filter; │ │ │ │ │ + var previousCache = this.cache; │ │ │ │ │ + this.cache = []; │ │ │ │ │ + this.handleAdd({ │ │ │ │ │ + features: this.layer.features │ │ │ │ │ + }); │ │ │ │ │ + if (this.cache.length > 0) { │ │ │ │ │ + this.caching = true; │ │ │ │ │ + this.layer.removeFeatures(this.cache.slice()); │ │ │ │ │ + this.caching = false │ │ │ │ │ + } │ │ │ │ │ + if (previousCache.length > 0) { │ │ │ │ │ + var event = { │ │ │ │ │ + features: previousCache │ │ │ │ │ + }; │ │ │ │ │ + this.handleAdd(event); │ │ │ │ │ + if (event.features.length > 0) { │ │ │ │ │ + this.caching = true; │ │ │ │ │ + this.layer.addFeatures(event.features); │ │ │ │ │ + this.caching = false │ │ │ │ │ } │ │ │ │ │ - return │ │ │ │ │ } │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - this.drawPoint(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - this.drawLineString(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - this.drawLinearRing(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - this.drawPolygon(geometry, style, featureId); │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Filter" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Strategy.Refresh = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + force: false, │ │ │ │ │ + interval: 0, │ │ │ │ │ + timer: null, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + if (this.layer.visibility === true) { │ │ │ │ │ + this.start() │ │ │ │ │ + } │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + visibilitychanged: this.reset, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - drawExternalGraphic: function(geometry, style, featureId) { │ │ │ │ │ - var img = new Image; │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - img.title = title │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.stop(); │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + visibilitychanged: this.reset, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ - var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ - var onLoad = function() { │ │ │ │ │ - if (!this.features[featureId]) { │ │ │ │ │ - return │ │ │ │ │ - } │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ - if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ - var x = p0 + xOffset | 0; │ │ │ │ │ - var y = p1 + yOffset | 0; │ │ │ │ │ - var canvas = this.canvas; │ │ │ │ │ - canvas.globalAlpha = opacity; │ │ │ │ │ - var factor = OpenLayers.Renderer.Canvas.drawImageScaleFactor || (OpenLayers.Renderer.Canvas.drawImageScaleFactor = /android 2.1/.test(navigator.userAgent.toLowerCase()) ? 320 / window.screen.width : 1); │ │ │ │ │ - canvas.drawImage(img, x * factor, y * factor, width * factor, height * factor); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId); │ │ │ │ │ - this.hitContext.fillRect(x, y, width, height) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - }; │ │ │ │ │ - img.onload = OpenLayers.Function.bind(onLoad, this); │ │ │ │ │ - img.src = style.externalGraphic │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - drawNamedSymbol: function(geometry, style, featureId) { │ │ │ │ │ - var x, y, cx, cy, i, symbolBounds, scaling, angle; │ │ │ │ │ - var unscaledStrokeWidth; │ │ │ │ │ - var deg2rad = Math.PI / 180; │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[style.graphicName]; │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(style.graphicName + " is not a valid symbol name") │ │ │ │ │ + reset: function() { │ │ │ │ │ + if (this.layer.visibility === true) { │ │ │ │ │ + this.start() │ │ │ │ │ + } else { │ │ │ │ │ + this.stop() │ │ │ │ │ } │ │ │ │ │ - if (!symbol.length || symbol.length < 2) return; │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ - if (isNaN(p0) || isNaN(p1)) return; │ │ │ │ │ - this.canvas.lineCap = "round"; │ │ │ │ │ - this.canvas.lineJoin = "round"; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.lineCap = "round"; │ │ │ │ │ - this.hitContext.lineJoin = "round" │ │ │ │ │ + }, │ │ │ │ │ + start: function() { │ │ │ │ │ + if (this.interval && typeof this.interval === "number" && this.interval > 0) { │ │ │ │ │ + this.timer = window.setInterval(OpenLayers.Function.bind(this.refresh, this), this.interval) │ │ │ │ │ } │ │ │ │ │ - if (style.graphicName in this.cachedSymbolBounds) { │ │ │ │ │ - symbolBounds = this.cachedSymbolBounds[style.graphicName] │ │ │ │ │ - } else { │ │ │ │ │ - symbolBounds = new OpenLayers.Bounds; │ │ │ │ │ - for (i = 0; i < symbol.length; i += 2) { │ │ │ │ │ - symbolBounds.extend(new OpenLayers.LonLat(symbol[i], symbol[i + 1])) │ │ │ │ │ - } │ │ │ │ │ - this.cachedSymbolBounds[style.graphicName] = symbolBounds │ │ │ │ │ + }, │ │ │ │ │ + refresh: function() { │ │ │ │ │ + if (this.layer && this.layer.refresh && typeof this.layer.refresh == "function") { │ │ │ │ │ + this.layer.refresh({ │ │ │ │ │ + force: this.force │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - this.canvas.save(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.save() │ │ │ │ │ + }, │ │ │ │ │ + stop: function() { │ │ │ │ │ + if (this.timer !== null) { │ │ │ │ │ + window.clearInterval(this.timer); │ │ │ │ │ + this.timer = null │ │ │ │ │ } │ │ │ │ │ - this.canvas.translate(p0, p1); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.translate(p0, p1) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Refresh" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Strategy.Cluster = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + distance: 20, │ │ │ │ │ + threshold: null, │ │ │ │ │ + features: null, │ │ │ │ │ + clusters: null, │ │ │ │ │ + clustering: false, │ │ │ │ │ + resolution: null, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + beforefeaturesadded: this.cacheFeatures, │ │ │ │ │ + featuresremoved: this.clearCache, │ │ │ │ │ + moveend: this.cluster, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - angle = deg2rad * style.rotation; │ │ │ │ │ - if (!isNaN(angle)) { │ │ │ │ │ - this.canvas.rotate(angle); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.rotate(angle) │ │ │ │ │ - } │ │ │ │ │ + return activated │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.clearCache(); │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + beforefeaturesadded: this.cacheFeatures, │ │ │ │ │ + featuresremoved: this.clearCache, │ │ │ │ │ + moveend: this.cluster, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - scaling = 2 * style.pointRadius / Math.max(symbolBounds.getWidth(), symbolBounds.getHeight()); │ │ │ │ │ - this.canvas.scale(scaling, scaling); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.scale(scaling, scaling) │ │ │ │ │ + return deactivated │ │ │ │ │ + }, │ │ │ │ │ + cacheFeatures: function(event) { │ │ │ │ │ + var propagate = true; │ │ │ │ │ + if (!this.clustering) { │ │ │ │ │ + this.clearCache(); │ │ │ │ │ + this.features = event.features; │ │ │ │ │ + this.cluster(); │ │ │ │ │ + propagate = false │ │ │ │ │ } │ │ │ │ │ - cx = symbolBounds.getCenterLonLat().lon; │ │ │ │ │ - cy = symbolBounds.getCenterLonLat().lat; │ │ │ │ │ - this.canvas.translate(-cx, -cy); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.translate(-cx, -cy) │ │ │ │ │ + return propagate │ │ │ │ │ + }, │ │ │ │ │ + clearCache: function() { │ │ │ │ │ + if (!this.clustering) { │ │ │ │ │ + this.features = null │ │ │ │ │ } │ │ │ │ │ - unscaledStrokeWidth = style.strokeWidth; │ │ │ │ │ - style.strokeWidth = unscaledStrokeWidth / scaling; │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.canvas.lineTo(x, y) │ │ │ │ │ - } │ │ │ │ │ - this.canvas.closePath(); │ │ │ │ │ - this.canvas.fill(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.hitContext.lineTo(x, y) │ │ │ │ │ + }, │ │ │ │ │ + cluster: function(event) { │ │ │ │ │ + if ((!event || event.zoomChanged) && this.features) { │ │ │ │ │ + var resolution = this.layer.map.getResolution(); │ │ │ │ │ + if (resolution != this.resolution || !this.clustersExist()) { │ │ │ │ │ + this.resolution = resolution; │ │ │ │ │ + var clusters = []; │ │ │ │ │ + var feature, clustered, cluster; │ │ │ │ │ + for (var i = 0; i < this.features.length; ++i) { │ │ │ │ │ + feature = this.features[i]; │ │ │ │ │ + if (feature.geometry) { │ │ │ │ │ + clustered = false; │ │ │ │ │ + for (var j = clusters.length - 1; j >= 0; --j) { │ │ │ │ │ + cluster = clusters[j]; │ │ │ │ │ + if (this.shouldCluster(cluster, feature)) { │ │ │ │ │ + this.addToCluster(cluster, feature); │ │ │ │ │ + clustered = true; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (!clustered) { │ │ │ │ │ + clusters.push(this.createCluster(this.features[i])) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.hitContext.closePath(); │ │ │ │ │ - this.hitContext.fill() │ │ │ │ │ + this.clustering = true; │ │ │ │ │ + this.layer.removeAllFeatures(); │ │ │ │ │ + this.clustering = false; │ │ │ │ │ + if (clusters.length > 0) { │ │ │ │ │ + if (this.threshold > 1) { │ │ │ │ │ + var clone = clusters.slice(); │ │ │ │ │ + clusters = []; │ │ │ │ │ + var candidate; │ │ │ │ │ + for (var i = 0, len = clone.length; i < len; ++i) { │ │ │ │ │ + candidate = clone[i]; │ │ │ │ │ + if (candidate.attributes.count < this.threshold) { │ │ │ │ │ + Array.prototype.push.apply(clusters, candidate.cluster) │ │ │ │ │ + } else { │ │ │ │ │ + clusters.push(candidate) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.clustering = true; │ │ │ │ │ + this.layer.addFeatures(clusters); │ │ │ │ │ + this.clustering = false │ │ │ │ │ + } │ │ │ │ │ + this.clusters = clusters │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.canvas.moveTo(x, y); │ │ │ │ │ - this.canvas.lineTo(x, y) │ │ │ │ │ - } │ │ │ │ │ - this.canvas.closePath(); │ │ │ │ │ - this.canvas.stroke(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style, scaling); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - for (i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - if (i == 0) this.hitContext.moveTo(x, y); │ │ │ │ │ - this.hitContext.lineTo(x, y) │ │ │ │ │ + }, │ │ │ │ │ + clustersExist: function() { │ │ │ │ │ + var exist = false; │ │ │ │ │ + if (this.clusters && this.clusters.length > 0 && this.clusters.length == this.layer.features.length) { │ │ │ │ │ + exist = true; │ │ │ │ │ + for (var i = 0; i < this.clusters.length; ++i) { │ │ │ │ │ + if (this.clusters[i] != this.layer.features[i]) { │ │ │ │ │ + exist = false; │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - this.hitContext.closePath(); │ │ │ │ │ - this.hitContext.stroke() │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - style.strokeWidth = unscaledStrokeWidth; │ │ │ │ │ - this.canvas.restore(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.restore() │ │ │ │ │ + return exist │ │ │ │ │ + }, │ │ │ │ │ + shouldCluster: function(cluster, feature) { │ │ │ │ │ + var cc = cluster.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + var fc = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + var distance = Math.sqrt(Math.pow(cc.lon - fc.lon, 2) + Math.pow(cc.lat - fc.lat, 2)) / this.resolution; │ │ │ │ │ + return distance <= this.distance │ │ │ │ │ + }, │ │ │ │ │ + addToCluster: function(cluster, feature) { │ │ │ │ │ + cluster.cluster.push(feature); │ │ │ │ │ + cluster.attributes.count += 1 │ │ │ │ │ + }, │ │ │ │ │ + createCluster: function(feature) { │ │ │ │ │ + var center = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + var cluster = new OpenLayers.Feature.Vector(new OpenLayers.Geometry.Point(center.lon, center.lat), { │ │ │ │ │ + count: 1 │ │ │ │ │ + }); │ │ │ │ │ + cluster.cluster = [feature]; │ │ │ │ │ + return cluster │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Cluster" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Strategy.Save = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + events: null, │ │ │ │ │ + auto: false, │ │ │ │ │ + timer: null, │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + OpenLayers.Strategy.prototype.initialize.apply(this, [options]); │ │ │ │ │ + this.events = new OpenLayers.Events(this) │ │ │ │ │ + }, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + if (this.auto) { │ │ │ │ │ + if (typeof this.auto === "number") { │ │ │ │ │ + this.timer = window.setInterval(OpenLayers.Function.bind(this.save, this), this.auto * 1e3) │ │ │ │ │ + } else { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + featureadded: this.triggerSave, │ │ │ │ │ + afterfeaturemodified: this.triggerSave, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - this.setCanvasStyle("reset") │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - setCanvasStyle: function(type, style) { │ │ │ │ │ - if (type === "fill") { │ │ │ │ │ - this.canvas.globalAlpha = style["fillOpacity"]; │ │ │ │ │ - this.canvas.fillStyle = style["fillColor"] │ │ │ │ │ - } else if (type === "stroke") { │ │ │ │ │ - this.canvas.globalAlpha = style["strokeOpacity"]; │ │ │ │ │ - this.canvas.strokeStyle = style["strokeColor"]; │ │ │ │ │ - this.canvas.lineWidth = style["strokeWidth"] │ │ │ │ │ - } else { │ │ │ │ │ - this.canvas.globalAlpha = 0; │ │ │ │ │ - this.canvas.lineWidth = 1 │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + if (this.auto) { │ │ │ │ │ + if (typeof this.auto === "number") { │ │ │ │ │ + window.clearInterval(this.timer) │ │ │ │ │ + } else { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + featureadded: this.triggerSave, │ │ │ │ │ + afterfeaturemodified: this.triggerSave, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - featureIdToHex: function(featureId) { │ │ │ │ │ - var id = Number(featureId.split("_").pop()) + 1; │ │ │ │ │ - if (id >= 16777216) { │ │ │ │ │ - this.hitOverflow = id - 16777215; │ │ │ │ │ - id = id % 16777216 + 1 │ │ │ │ │ + triggerSave: function(event) { │ │ │ │ │ + var feature = event.feature; │ │ │ │ │ + if (feature.state === OpenLayers.State.INSERT || feature.state === OpenLayers.State.UPDATE || feature.state === OpenLayers.State.DELETE) { │ │ │ │ │ + this.save([event.feature]) │ │ │ │ │ } │ │ │ │ │ - var hex = "000000" + id.toString(16); │ │ │ │ │ - var len = hex.length; │ │ │ │ │ - hex = "#" + hex.substring(len - 6, len); │ │ │ │ │ - return hex │ │ │ │ │ }, │ │ │ │ │ - setHitContextStyle: function(type, featureId, symbolizer, strokeScaling) { │ │ │ │ │ - var hex = this.featureIdToHex(featureId); │ │ │ │ │ - if (type == "fill") { │ │ │ │ │ - this.hitContext.globalAlpha = 1; │ │ │ │ │ - this.hitContext.fillStyle = hex │ │ │ │ │ - } else if (type == "stroke") { │ │ │ │ │ - this.hitContext.globalAlpha = 1; │ │ │ │ │ - this.hitContext.strokeStyle = hex; │ │ │ │ │ - if (typeof strokeScaling === "undefined") { │ │ │ │ │ - this.hitContext.lineWidth = symbolizer.strokeWidth + 2 │ │ │ │ │ - } else { │ │ │ │ │ - if (!isNaN(strokeScaling)) { │ │ │ │ │ - this.hitContext.lineWidth = symbolizer.strokeWidth + 2 / strokeScaling │ │ │ │ │ + save: function(features) { │ │ │ │ │ + if (!features) { │ │ │ │ │ + features = this.layer.features │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("start", { │ │ │ │ │ + features: features │ │ │ │ │ + }); │ │ │ │ │ + var remote = this.layer.projection; │ │ │ │ │ + var local = this.layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var len = features.length; │ │ │ │ │ + var clones = new Array(len); │ │ │ │ │ + var orig, clone; │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + orig = features[i]; │ │ │ │ │ + clone = orig.clone(); │ │ │ │ │ + clone.fid = orig.fid; │ │ │ │ │ + clone.state = orig.state; │ │ │ │ │ + if (orig.url) { │ │ │ │ │ + clone.url = orig.url │ │ │ │ │ } │ │ │ │ │ + clone._original = orig; │ │ │ │ │ + clone.geometry.transform(local, remote); │ │ │ │ │ + clones[i] = clone │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - this.hitContext.globalAlpha = 0; │ │ │ │ │ - this.hitContext.lineWidth = 1 │ │ │ │ │ + features = clones │ │ │ │ │ } │ │ │ │ │ + this.layer.protocol.commit(features, { │ │ │ │ │ + callback: this.onCommit, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - drawPoint: function(geometry, style, featureId) { │ │ │ │ │ - if (style.graphic !== false) { │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - this.drawExternalGraphic(geometry, style, featureId) │ │ │ │ │ - } else if (style.graphicName && style.graphicName != "circle") { │ │ │ │ │ - this.drawNamedSymbol(geometry, style, featureId) │ │ │ │ │ - } else { │ │ │ │ │ - var pt = this.getLocalXY(geometry); │ │ │ │ │ - var p0 = pt[0]; │ │ │ │ │ - var p1 = pt[1]; │ │ │ │ │ - if (!isNaN(p0) && !isNaN(p1)) { │ │ │ │ │ - var twoPi = Math.PI * 2; │ │ │ │ │ - var radius = style.pointRadius; │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.canvas.fill(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.hitContext.fill() │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.canvas.beginPath(); │ │ │ │ │ - this.canvas.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.canvas.stroke(); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ - this.hitContext.beginPath(); │ │ │ │ │ - this.hitContext.arc(p0, p1, radius, 0, twoPi, true); │ │ │ │ │ - this.hitContext.stroke() │ │ │ │ │ - } │ │ │ │ │ - this.setCanvasStyle("reset") │ │ │ │ │ + onCommit: function(response) { │ │ │ │ │ + var evt = { │ │ │ │ │ + response: response │ │ │ │ │ + }; │ │ │ │ │ + if (response.success()) { │ │ │ │ │ + var features = response.reqFeatures; │ │ │ │ │ + var state, feature; │ │ │ │ │ + var destroys = []; │ │ │ │ │ + var insertIds = response.insertIds || []; │ │ │ │ │ + var j = 0; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + feature = features[i]; │ │ │ │ │ + feature = feature._original || feature; │ │ │ │ │ + state = feature.state; │ │ │ │ │ + if (state) { │ │ │ │ │ + if (state == OpenLayers.State.DELETE) { │ │ │ │ │ + destroys.push(feature) │ │ │ │ │ + } else if (state == OpenLayers.State.INSERT) { │ │ │ │ │ + feature.fid = insertIds[j]; │ │ │ │ │ + ++j │ │ │ │ │ } │ │ │ │ │ + feature.state = null │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + if (destroys.length > 0) { │ │ │ │ │ + this.layer.destroyFeatures(destroys) │ │ │ │ │ + } │ │ │ │ │ + this.events.triggerEvent("success", evt) │ │ │ │ │ + } else { │ │ │ │ │ + this.events.triggerEvent("fail", evt) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - drawLineString: function(geometry, style, featureId) { │ │ │ │ │ - style = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - fill: false │ │ │ │ │ - }, style); │ │ │ │ │ - this.drawLinearRing(geometry, style, featureId) │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Save" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Strategy.Paging = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + features: null, │ │ │ │ │ + length: 10, │ │ │ │ │ + num: null, │ │ │ │ │ + paging: false, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.call(this); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + beforefeaturesadded: this.cacheFeatures, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + return activated │ │ │ │ │ }, │ │ │ │ │ - drawLinearRing: function(geometry, style, featureId) { │ │ │ │ │ - if (style.fill !== false) { │ │ │ │ │ - this.setCanvasStyle("fill", style); │ │ │ │ │ - this.renderPath(this.canvas, geometry, style, featureId, "fill"); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("fill", featureId, style); │ │ │ │ │ - this.renderPath(this.hitContext, geometry, style, featureId, "fill") │ │ │ │ │ - } │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.clearCache(); │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + beforefeaturesadded: this.cacheFeatures, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - if (style.stroke !== false) { │ │ │ │ │ - this.setCanvasStyle("stroke", style); │ │ │ │ │ - this.renderPath(this.canvas, geometry, style, featureId, "stroke"); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.setHitContextStyle("stroke", featureId, style); │ │ │ │ │ - this.renderPath(this.hitContext, geometry, style, featureId, "stroke") │ │ │ │ │ - } │ │ │ │ │ + return deactivated │ │ │ │ │ + }, │ │ │ │ │ + cacheFeatures: function(event) { │ │ │ │ │ + if (!this.paging) { │ │ │ │ │ + this.clearCache(); │ │ │ │ │ + this.features = event.features; │ │ │ │ │ + this.pageNext(event) │ │ │ │ │ } │ │ │ │ │ - this.setCanvasStyle("reset") │ │ │ │ │ }, │ │ │ │ │ - renderPath: function(context, geometry, style, featureId, type) { │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - context.beginPath(); │ │ │ │ │ - var start = this.getLocalXY(components[0]); │ │ │ │ │ - var x = start[0]; │ │ │ │ │ - var y = start[1]; │ │ │ │ │ - if (!isNaN(x) && !isNaN(y)) { │ │ │ │ │ - context.moveTo(start[0], start[1]); │ │ │ │ │ - for (var i = 1; i < len; ++i) { │ │ │ │ │ - var pt = this.getLocalXY(components[i]); │ │ │ │ │ - context.lineTo(pt[0], pt[1]) │ │ │ │ │ - } │ │ │ │ │ - if (type === "fill") { │ │ │ │ │ - context.fill() │ │ │ │ │ - } else { │ │ │ │ │ - context.stroke() │ │ │ │ │ + clearCache: function() { │ │ │ │ │ + if (this.features) { │ │ │ │ │ + for (var i = 0; i < this.features.length; ++i) { │ │ │ │ │ + this.features[i].destroy() │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ + this.features = null; │ │ │ │ │ + this.num = null │ │ │ │ │ }, │ │ │ │ │ - drawPolygon: function(geometry, style, featureId) { │ │ │ │ │ - var components = geometry.components; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - this.drawLinearRing(components[0], style, featureId); │ │ │ │ │ - for (var i = 1; i < len; ++i) { │ │ │ │ │ - this.canvas.globalCompositeOperation = "destination-out"; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.globalCompositeOperation = "destination-out" │ │ │ │ │ - } │ │ │ │ │ - this.drawLinearRing(components[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - stroke: false, │ │ │ │ │ - fillOpacity: 1 │ │ │ │ │ - }, style), featureId); │ │ │ │ │ - this.canvas.globalCompositeOperation = "source-over"; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.globalCompositeOperation = "source-over" │ │ │ │ │ - } │ │ │ │ │ - this.drawLinearRing(components[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - fill: false │ │ │ │ │ - }, style), featureId) │ │ │ │ │ + pageCount: function() { │ │ │ │ │ + var numFeatures = this.features ? this.features.length : 0; │ │ │ │ │ + return Math.ceil(numFeatures / this.length) │ │ │ │ │ + }, │ │ │ │ │ + pageNum: function() { │ │ │ │ │ + return this.num │ │ │ │ │ + }, │ │ │ │ │ + pageLength: function(newLength) { │ │ │ │ │ + if (newLength && newLength > 0) { │ │ │ │ │ + this.length = newLength │ │ │ │ │ } │ │ │ │ │ + return this.length │ │ │ │ │ }, │ │ │ │ │ - drawText: function(location, style) { │ │ │ │ │ - var pt = this.getLocalXY(location); │ │ │ │ │ - this.setCanvasStyle("reset"); │ │ │ │ │ - this.canvas.fillStyle = style.fontColor; │ │ │ │ │ - this.canvas.globalAlpha = style.fontOpacity || 1; │ │ │ │ │ - var fontStyle = [style.fontStyle ? style.fontStyle : "normal", "normal", style.fontWeight ? style.fontWeight : "normal", style.fontSize ? style.fontSize : "1em", style.fontFamily ? style.fontFamily : "sans-serif"].join(" "); │ │ │ │ │ - var labelRows = style.label.split("\n"); │ │ │ │ │ - var numRows = labelRows.length; │ │ │ │ │ - if (this.canvas.fillText) { │ │ │ │ │ - this.canvas.font = fontStyle; │ │ │ │ │ - this.canvas.textAlign = OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[0]] || "center"; │ │ │ │ │ - this.canvas.textBaseline = OpenLayers.Renderer.Canvas.LABEL_ALIGN[style.labelAlign[1]] || "middle"; │ │ │ │ │ - var vfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5 │ │ │ │ │ - } │ │ │ │ │ - var lineHeight = this.canvas.measureText("Mg").height || this.canvas.measureText("xx").width; │ │ │ │ │ - pt[1] += lineHeight * vfactor * (numRows - 1); │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - if (style.labelOutlineWidth) { │ │ │ │ │ - this.canvas.save(); │ │ │ │ │ - this.canvas.globalAlpha = style.labelOutlineOpacity || style.fontOpacity || 1; │ │ │ │ │ - this.canvas.strokeStyle = style.labelOutlineColor; │ │ │ │ │ - this.canvas.lineWidth = style.labelOutlineWidth; │ │ │ │ │ - this.canvas.strokeText(labelRows[i], pt[0], pt[1] + lineHeight * i + 1); │ │ │ │ │ - this.canvas.restore() │ │ │ │ │ - } │ │ │ │ │ - this.canvas.fillText(labelRows[i], pt[0], pt[1] + lineHeight * i) │ │ │ │ │ - } │ │ │ │ │ - } else if (this.canvas.mozDrawText) { │ │ │ │ │ - this.canvas.mozTextStyle = fontStyle; │ │ │ │ │ - var hfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[0]]; │ │ │ │ │ - if (hfactor == null) { │ │ │ │ │ - hfactor = -.5 │ │ │ │ │ - } │ │ │ │ │ - var vfactor = OpenLayers.Renderer.Canvas.LABEL_FACTOR[style.labelAlign[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5 │ │ │ │ │ - } │ │ │ │ │ - var lineHeight = this.canvas.mozMeasureText("xx"); │ │ │ │ │ - pt[1] += lineHeight * (1 + vfactor * numRows); │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - var x = pt[0] + hfactor * this.canvas.mozMeasureText(labelRows[i]); │ │ │ │ │ - var y = pt[1] + i * lineHeight; │ │ │ │ │ - this.canvas.translate(x, y); │ │ │ │ │ - this.canvas.mozDrawText(labelRows[i]); │ │ │ │ │ - this.canvas.translate(-x, -y) │ │ │ │ │ + pageNext: function(event) { │ │ │ │ │ + var changed = false; │ │ │ │ │ + if (this.features) { │ │ │ │ │ + if (this.num === null) { │ │ │ │ │ + this.num = -1 │ │ │ │ │ } │ │ │ │ │ + var start = (this.num + 1) * this.length; │ │ │ │ │ + changed = this.page(start, event) │ │ │ │ │ } │ │ │ │ │ - this.setCanvasStyle("reset") │ │ │ │ │ - }, │ │ │ │ │ - getLocalXY: function(point) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var extent = this.extent; │ │ │ │ │ - var x = (point.x - this.featureDx) / resolution + -extent.left / resolution; │ │ │ │ │ - var y = extent.top / resolution - point.y / resolution; │ │ │ │ │ - return [x, y] │ │ │ │ │ + return changed │ │ │ │ │ }, │ │ │ │ │ - clear: function() { │ │ │ │ │ - var height = this.root.height; │ │ │ │ │ - var width = this.root.width; │ │ │ │ │ - this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ - this.features = {}; │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.clearRect(0, 0, width, height) │ │ │ │ │ + pagePrevious: function() { │ │ │ │ │ + var changed = false; │ │ │ │ │ + if (this.features) { │ │ │ │ │ + if (this.num === null) { │ │ │ │ │ + this.num = this.pageCount() │ │ │ │ │ + } │ │ │ │ │ + var start = (this.num - 1) * this.length; │ │ │ │ │ + changed = this.page(start) │ │ │ │ │ } │ │ │ │ │ + return changed │ │ │ │ │ }, │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var featureId, feature; │ │ │ │ │ - if (this.hitDetection && this.root.style.display !== "none") { │ │ │ │ │ - if (!this.map.dragging) { │ │ │ │ │ - var xy = evt.xy; │ │ │ │ │ - var x = xy.x | 0; │ │ │ │ │ - var y = xy.y | 0; │ │ │ │ │ - var data = this.hitContext.getImageData(x, y, 1, 1).data; │ │ │ │ │ - if (data[3] === 255) { │ │ │ │ │ - var id = data[2] + 256 * (data[1] + 256 * data[0]); │ │ │ │ │ - if (id) { │ │ │ │ │ - featureId = "OpenLayers_Feature_Vector_" + (id - 1 + this.hitOverflow); │ │ │ │ │ - try { │ │ │ │ │ - feature = this.features[featureId][0] │ │ │ │ │ - } catch (err) {} │ │ │ │ │ + page: function(start, event) { │ │ │ │ │ + var changed = false; │ │ │ │ │ + if (this.features) { │ │ │ │ │ + if (start >= 0 && start < this.features.length) { │ │ │ │ │ + var num = Math.floor(start / this.length); │ │ │ │ │ + if (num != this.num) { │ │ │ │ │ + this.paging = true; │ │ │ │ │ + var features = this.features.slice(start, start + this.length); │ │ │ │ │ + this.layer.removeFeatures(this.layer.features); │ │ │ │ │ + this.num = num; │ │ │ │ │ + if (event && event.features) { │ │ │ │ │ + event.features = features │ │ │ │ │ + } else { │ │ │ │ │ + this.layer.addFeatures(features) │ │ │ │ │ } │ │ │ │ │ + this.paging = false; │ │ │ │ │ + changed = true │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - return feature │ │ │ │ │ + return changed │ │ │ │ │ }, │ │ │ │ │ - eraseFeatures: function(features) { │ │ │ │ │ - if (!OpenLayers.Util.isArray(features)) { │ │ │ │ │ - features = [features] │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Paging" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Strategy.Fixed = OpenLayers.Class(OpenLayers.Strategy, { │ │ │ │ │ + preload: false, │ │ │ │ │ + activate: function() { │ │ │ │ │ + var activated = OpenLayers.Strategy.prototype.activate.apply(this, arguments); │ │ │ │ │ + if (activated) { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + refresh: this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + if (this.layer.visibility == true || this.preload) { │ │ │ │ │ + this.load() │ │ │ │ │ + } else { │ │ │ │ │ + this.layer.events.on({ │ │ │ │ │ + visibilitychanged: this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - for (var i = 0; i < features.length; ++i) { │ │ │ │ │ - delete this.features[features[i].id] │ │ │ │ │ + return activated │ │ │ │ │ + }, │ │ │ │ │ + deactivate: function() { │ │ │ │ │ + var deactivated = OpenLayers.Strategy.prototype.deactivate.call(this); │ │ │ │ │ + if (deactivated) { │ │ │ │ │ + this.layer.events.un({ │ │ │ │ │ + refresh: this.load, │ │ │ │ │ + visibilitychanged: this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - this.redraw() │ │ │ │ │ + return deactivated │ │ │ │ │ }, │ │ │ │ │ - redraw: function() { │ │ │ │ │ - if (!this.locked) { │ │ │ │ │ - var height = this.root.height; │ │ │ │ │ - var width = this.root.width; │ │ │ │ │ - this.canvas.clearRect(0, 0, width, height); │ │ │ │ │ - if (this.hitDetection) { │ │ │ │ │ - this.hitContext.clearRect(0, 0, width, height) │ │ │ │ │ - } │ │ │ │ │ - var labelMap = []; │ │ │ │ │ - var feature, geometry, style; │ │ │ │ │ - var worldBounds = this.map.baseLayer && this.map.baseLayer.wrapDateLine && this.map.getMaxExtent(); │ │ │ │ │ - for (var id in this.features) { │ │ │ │ │ - if (!this.features.hasOwnProperty(id)) { │ │ │ │ │ - continue │ │ │ │ │ - } │ │ │ │ │ - feature = this.features[id][0]; │ │ │ │ │ - geometry = feature.geometry; │ │ │ │ │ - this.calculateFeatureDx(geometry.getBounds(), worldBounds); │ │ │ │ │ - style = this.features[id][1]; │ │ │ │ │ - this.drawGeometry(geometry, style, feature.id); │ │ │ │ │ - if (style.label) { │ │ │ │ │ - labelMap.push([feature, style]) │ │ │ │ │ + load: function(options) { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + layer.events.triggerEvent("loadstart", { │ │ │ │ │ + filter: layer.filter │ │ │ │ │ + }); │ │ │ │ │ + layer.protocol.read(OpenLayers.Util.applyDefaults({ │ │ │ │ │ + callback: this.merge, │ │ │ │ │ + filter: layer.filter, │ │ │ │ │ + scope: this │ │ │ │ │ + }, options)); │ │ │ │ │ + layer.events.un({ │ │ │ │ │ + visibilitychanged: this.load, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + merge: function(resp) { │ │ │ │ │ + var layer = this.layer; │ │ │ │ │ + layer.destroyFeatures(); │ │ │ │ │ + var features = resp.features; │ │ │ │ │ + if (features && features.length > 0) { │ │ │ │ │ + var remote = layer.projection; │ │ │ │ │ + var local = layer.map.getProjectionObject(); │ │ │ │ │ + if (!local.equals(remote)) { │ │ │ │ │ + var geom; │ │ │ │ │ + for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ + geom = features[i].geometry; │ │ │ │ │ + if (geom) { │ │ │ │ │ + geom.transform(remote, local) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var item; │ │ │ │ │ - for (var i = 0, len = labelMap.length; i < len; ++i) { │ │ │ │ │ - item = labelMap[i]; │ │ │ │ │ - this.drawText(item[0].geometry.getCentroid(), item[1]) │ │ │ │ │ - } │ │ │ │ │ + layer.addFeatures(features) │ │ │ │ │ } │ │ │ │ │ + layer.events.triggerEvent("loadend", { │ │ │ │ │ + response: resp │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.Canvas" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Strategy.Fixed" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Renderer.Canvas.LABEL_ALIGN = { │ │ │ │ │ - l: "left", │ │ │ │ │ - r: "right", │ │ │ │ │ - t: "top", │ │ │ │ │ - b: "bottom" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.Canvas.LABEL_FACTOR = { │ │ │ │ │ - l: 0, │ │ │ │ │ - r: -1, │ │ │ │ │ - t: 0, │ │ │ │ │ - b: -1 │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.Canvas.drawImageScaleFactor = null; │ │ │ │ │ -OpenLayers.Renderer.VML = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ - xmlns: "urn:schemas-microsoft-com:vml", │ │ │ │ │ - symbolCache: {}, │ │ │ │ │ - offset: null, │ │ │ │ │ - initialize: function(containerID) { │ │ │ │ │ - if (!this.supported()) { │ │ │ │ │ - return │ │ │ │ │ +OpenLayers.Marker.Box = OpenLayers.Class(OpenLayers.Marker, { │ │ │ │ │ + bounds: null, │ │ │ │ │ + div: null, │ │ │ │ │ + initialize: function(bounds, borderColor, borderWidth) { │ │ │ │ │ + this.bounds = bounds; │ │ │ │ │ + this.div = OpenLayers.Util.createDiv(); │ │ │ │ │ + this.div.style.overflow = "hidden"; │ │ │ │ │ + this.events = new OpenLayers.Events(this, this.div); │ │ │ │ │ + this.setBorder(borderColor, borderWidth) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.bounds = null; │ │ │ │ │ + this.div = null; │ │ │ │ │ + OpenLayers.Marker.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + setBorder: function(color, width) { │ │ │ │ │ + if (!color) { │ │ │ │ │ + color = "red" │ │ │ │ │ } │ │ │ │ │ - if (!document.namespaces.olv) { │ │ │ │ │ - document.namespaces.add("olv", this.xmlns); │ │ │ │ │ - var style = document.createStyleSheet(); │ │ │ │ │ - var shapes = ["shape", "rect", "oval", "fill", "stroke", "imagedata", "group", "textbox"]; │ │ │ │ │ - for (var i = 0, len = shapes.length; i < len; i++) { │ │ │ │ │ - style.addRule("olv\\:" + shapes[i], "behavior: url(#default#VML); " + "position: absolute; display: inline-block;") │ │ │ │ │ - } │ │ │ │ │ + if (!width) { │ │ │ │ │ + width = 2 │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.initialize.apply(this, arguments) │ │ │ │ │ + this.div.style.border = width + "px solid " + color │ │ │ │ │ }, │ │ │ │ │ - supported: function() { │ │ │ │ │ - return !!document.namespaces │ │ │ │ │ + draw: function(px, sz) { │ │ │ │ │ + OpenLayers.Util.modifyDOMElement(this.div, null, px, sz); │ │ │ │ │ + return this.div │ │ │ │ │ }, │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var left = extent.left / resolution | 0; │ │ │ │ │ - var top = extent.top / resolution - this.size.h | 0; │ │ │ │ │ - if (resolutionChanged || !this.offset) { │ │ │ │ │ - this.offset = { │ │ │ │ │ - x: left, │ │ │ │ │ - y: top │ │ │ │ │ - }; │ │ │ │ │ - left = 0; │ │ │ │ │ - top = 0 │ │ │ │ │ - } else { │ │ │ │ │ - left = left - this.offset.x; │ │ │ │ │ - top = top - this.offset.y │ │ │ │ │ + onScreen: function() { │ │ │ │ │ + var onScreen = false; │ │ │ │ │ + if (this.map) { │ │ │ │ │ + var screenBounds = this.map.getExtent(); │ │ │ │ │ + onScreen = screenBounds.containsBounds(this.bounds, true, true) │ │ │ │ │ } │ │ │ │ │ - var org = left - this.xOffset + " " + top; │ │ │ │ │ - this.root.coordorigin = org; │ │ │ │ │ - var roots = [this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ - var root; │ │ │ │ │ - for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ - root = roots[i]; │ │ │ │ │ - var size = this.size.w + " " + this.size.h; │ │ │ │ │ - root.coordsize = size │ │ │ │ │ + return onScreen │ │ │ │ │ + }, │ │ │ │ │ + display: function(display) { │ │ │ │ │ + this.div.style.display = display ? "" : "none" │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Marker.Box" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.XYZ = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + sphericalMercator: false, │ │ │ │ │ + zoomOffset: 0, │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + if (options && options.sphericalMercator || this.sphericalMercator) { │ │ │ │ │ + options = OpenLayers.Util.extend({ │ │ │ │ │ + projection: "EPSG:900913", │ │ │ │ │ + numZoomLevels: 19 │ │ │ │ │ + }, options) │ │ │ │ │ } │ │ │ │ │ - this.root.style.flip = "y"; │ │ │ │ │ - return coordSysUnchanged │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name || this.name, url || this.url, {}, options]) │ │ │ │ │ }, │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ - var roots = [this.rendererRoot, this.root, this.vectorRoot, this.textRoot]; │ │ │ │ │ - var w = this.size.w + "px"; │ │ │ │ │ - var h = this.size.h + "px"; │ │ │ │ │ - var root; │ │ │ │ │ - for (var i = 0, len = roots.length; i < len; ++i) { │ │ │ │ │ - root = roots[i]; │ │ │ │ │ - root.style.width = w; │ │ │ │ │ - root.style.height = h │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.XYZ(this.name, this.url, this.getOptions()) │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var xyz = this.getXYZ(bounds); │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + var s = "" + xyz.x + xyz.y + xyz.z; │ │ │ │ │ + url = this.selectUrl(s, url) │ │ │ │ │ } │ │ │ │ │ + return OpenLayers.String.format(url, xyz) │ │ │ │ │ }, │ │ │ │ │ - getNodeType: function(geometry, style) { │ │ │ │ │ - var nodeType = null; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - nodeType = "olv:rect" │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - nodeType = "olv:shape" │ │ │ │ │ - } else { │ │ │ │ │ - nodeType = "olv:oval" │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - nodeType = "olv:rect"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - case "OpenLayers.Geometry.Curve": │ │ │ │ │ - nodeType = "olv:shape"; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break │ │ │ │ │ + getXYZ: function(bounds) { │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.maxExtent.left) / (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((this.maxExtent.top - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getServerZoom(); │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + var limit = Math.pow(2, z); │ │ │ │ │ + x = (x % limit + limit) % limit │ │ │ │ │ + } │ │ │ │ │ + return { │ │ │ │ │ + x: x, │ │ │ │ │ + y: y, │ │ │ │ │ + z: z │ │ │ │ │ } │ │ │ │ │ - return nodeType │ │ │ │ │ }, │ │ │ │ │ - setStyle: function(node, style, options, geometry) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - options = options || node._options; │ │ │ │ │ - var fillColor = style.fillColor; │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - node.title = title │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (!this.tileOrigin) { │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, this.maxExtent.bottom) │ │ │ │ │ } │ │ │ │ │ - if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - options.isFilled = true; │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ - var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ - node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x + xOffset | 0) + "px"; │ │ │ │ │ - node.style.top = (geometry.y / resolution - this.offset.y - (yOffset + height) | 0) + "px"; │ │ │ │ │ - node.style.width = width + "px"; │ │ │ │ │ - node.style.height = height + "px"; │ │ │ │ │ - node.style.flip = "y"; │ │ │ │ │ - fillColor = "none"; │ │ │ │ │ - options.isStroked = false │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - var cache = this.importSymbol(style.graphicName); │ │ │ │ │ - node.path = cache.path; │ │ │ │ │ - node.coordorigin = cache.left + "," + cache.bottom; │ │ │ │ │ - var size = cache.size; │ │ │ │ │ - node.coordsize = size + "," + size; │ │ │ │ │ - this.drawCircle(node, geometry, style.pointRadius); │ │ │ │ │ - node.style.flip = "y" │ │ │ │ │ - } else { │ │ │ │ │ - this.drawCircle(node, geometry, style.pointRadius) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.XYZ" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Text = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ + location: null, │ │ │ │ │ + features: null, │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + selectedFeature: null, │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.features = [] │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.destroy.apply(this, arguments); │ │ │ │ │ + this.clearFeatures(); │ │ │ │ │ + this.features = null │ │ │ │ │ + }, │ │ │ │ │ + loadText: function() { │ │ │ │ │ + if (!this.loaded) { │ │ │ │ │ + if (this.location != null) { │ │ │ │ │ + var onFail = function(e) { │ │ │ │ │ + this.events.triggerEvent("loadend") │ │ │ │ │ + }; │ │ │ │ │ + this.events.triggerEvent("loadstart"); │ │ │ │ │ + OpenLayers.Request.GET({ │ │ │ │ │ + url: this.location, │ │ │ │ │ + success: this.parseData, │ │ │ │ │ + failure: onFail, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.loaded = true │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (options.isFilled) { │ │ │ │ │ - node.fillcolor = fillColor │ │ │ │ │ - } else { │ │ │ │ │ - node.filled = "false" │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(bounds, zoomChanged, minor) { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + if (this.visibility && !this.loaded) { │ │ │ │ │ + this.loadText() │ │ │ │ │ } │ │ │ │ │ - var fills = node.getElementsByTagName("fill"); │ │ │ │ │ - var fill = fills.length == 0 ? null : fills[0]; │ │ │ │ │ - if (!options.isFilled) { │ │ │ │ │ - if (fill) { │ │ │ │ │ - node.removeChild(fill) │ │ │ │ │ + }, │ │ │ │ │ + parseData: function(ajaxRequest) { │ │ │ │ │ + var text = ajaxRequest.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 parser = new OpenLayers.Format.Text(options); │ │ │ │ │ + var features = parser.read(text); │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + var data = {}; │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + var location; │ │ │ │ │ + var iconSize, iconOffset; │ │ │ │ │ + location = new OpenLayers.LonLat(feature.geometry.x, feature.geometry.y); │ │ │ │ │ + if (feature.style.graphicWidth && feature.style.graphicHeight) { │ │ │ │ │ + iconSize = new OpenLayers.Size(feature.style.graphicWidth, feature.style.graphicHeight) │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - if (!fill) { │ │ │ │ │ - fill = this.createNode("olv:fill", node.id + "_fill") │ │ │ │ │ + if (feature.style.graphicXOffset !== undefined && feature.style.graphicYOffset !== undefined) { │ │ │ │ │ + iconOffset = new OpenLayers.Pixel(feature.style.graphicXOffset, feature.style.graphicYOffset) │ │ │ │ │ } │ │ │ │ │ - fill.opacity = style.fillOpacity; │ │ │ │ │ - if (node._geometryClass === "OpenLayers.Geometry.Point" && style.externalGraphic) { │ │ │ │ │ - if (style.graphicOpacity) { │ │ │ │ │ - fill.opacity = style.graphicOpacity │ │ │ │ │ - } │ │ │ │ │ - fill.src = style.externalGraphic; │ │ │ │ │ - fill.type = "frame"; │ │ │ │ │ - if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ - fill.aspect = "atmost" │ │ │ │ │ + if (feature.style.externalGraphic != null) { │ │ │ │ │ + data.icon = new OpenLayers.Icon(feature.style.externalGraphic, iconSize, iconOffset) │ │ │ │ │ + } else { │ │ │ │ │ + data.icon = OpenLayers.Marker.defaultIcon(); │ │ │ │ │ + if (iconSize != null) { │ │ │ │ │ + data.icon.setSize(iconSize) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (fill.parentNode != node) { │ │ │ │ │ - node.appendChild(fill) │ │ │ │ │ + if (feature.attributes.title != null && feature.attributes.description != null) { │ │ │ │ │ + data["popupContentHTML"] = "<h2>" + feature.attributes.title + "</h2>" + "<p>" + feature.attributes.description + "</p>" │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - var rotation = style.rotation; │ │ │ │ │ - if (rotation !== undefined || node._rotation !== undefined) { │ │ │ │ │ - node._rotation = rotation; │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - this.graphicRotate(node, xOffset, yOffset, style); │ │ │ │ │ - fill.opacity = 0 │ │ │ │ │ - } else if (node._geometryClass === "OpenLayers.Geometry.Point") { │ │ │ │ │ - node.style.rotation = rotation || 0 │ │ │ │ │ + data["overflow"] = feature.attributes.overflow || "auto"; │ │ │ │ │ + var markerFeature = new OpenLayers.Feature(this, location, data); │ │ │ │ │ + this.features.push(markerFeature); │ │ │ │ │ + var marker = markerFeature.createMarker(); │ │ │ │ │ + if (feature.attributes.title != null && feature.attributes.description != null) { │ │ │ │ │ + marker.events.register("click", markerFeature, this.markerClick) │ │ │ │ │ } │ │ │ │ │ + this.addMarker(marker) │ │ │ │ │ } │ │ │ │ │ - var strokes = node.getElementsByTagName("stroke"); │ │ │ │ │ - var stroke = strokes.length == 0 ? null : strokes[0]; │ │ │ │ │ - if (!options.isStroked) { │ │ │ │ │ - node.stroked = false; │ │ │ │ │ - if (stroke) { │ │ │ │ │ - stroke.on = false │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - if (!stroke) { │ │ │ │ │ - stroke = this.createNode("olv:stroke", node.id + "_stroke"); │ │ │ │ │ - node.appendChild(stroke) │ │ │ │ │ + this.events.triggerEvent("loadend") │ │ │ │ │ + }, │ │ │ │ │ + markerClick: function(evt) { │ │ │ │ │ + var sameMarkerClicked = this == this.layer.selectedFeature; │ │ │ │ │ + this.layer.selectedFeature = !sameMarkerClicked ? this : null; │ │ │ │ │ + for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ + this.layer.map.removePopup(this.layer.map.popups[i]) │ │ │ │ │ + } │ │ │ │ │ + if (!sameMarkerClicked) { │ │ │ │ │ + this.layer.map.addPopup(this.createPopup()) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ + }, │ │ │ │ │ + clearFeatures: function() { │ │ │ │ │ + if (this.features != null) { │ │ │ │ │ + while (this.features.length > 0) { │ │ │ │ │ + var feature = this.features[0]; │ │ │ │ │ + OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ + feature.destroy() │ │ │ │ │ } │ │ │ │ │ - stroke.on = true; │ │ │ │ │ - stroke.color = style.strokeColor; │ │ │ │ │ - stroke.weight = style.strokeWidth + "px"; │ │ │ │ │ - stroke.opacity = style.strokeOpacity; │ │ │ │ │ - stroke.endcap = style.strokeLinecap == "butt" ? "flat" : style.strokeLinecap || "round"; │ │ │ │ │ - if (style.strokeDashstyle) { │ │ │ │ │ - stroke.dashstyle = this.dashStyle(style) │ │ │ │ │ + } │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Text" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.ArcGISCache = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ + url: null, │ │ │ │ │ + tileOrigin: null, │ │ │ │ │ + tileSize: new OpenLayers.Size(256, 256), │ │ │ │ │ + useArcGISServer: true, │ │ │ │ │ + type: "png", │ │ │ │ │ + useScales: false, │ │ │ │ │ + overrideDPI: false, │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (this.resolutions) { │ │ │ │ │ + this.serverResolutions = this.resolutions; │ │ │ │ │ + this.maxExtent = this.getMaxExtentForResolution(this.resolutions[0]) │ │ │ │ │ + } │ │ │ │ │ + if (this.layerInfo) { │ │ │ │ │ + var info = this.layerInfo; │ │ │ │ │ + var startingTileExtent = new OpenLayers.Bounds(info.fullExtent.xmin, info.fullExtent.ymin, info.fullExtent.xmax, info.fullExtent.ymax); │ │ │ │ │ + this.projection = "EPSG:" + info.spatialReference.wkid; │ │ │ │ │ + this.sphericalMercator = info.spatialReference.wkid == 102100; │ │ │ │ │ + this.units = info.units == "esriFeet" ? "ft" : "m"; │ │ │ │ │ + if (!!info.tileInfo) { │ │ │ │ │ + this.tileSize = new OpenLayers.Size(info.tileInfo.width || info.tileInfo.cols, info.tileInfo.height || info.tileInfo.rows); │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(info.tileInfo.origin.x, info.tileInfo.origin.y); │ │ │ │ │ + var upperLeft = new OpenLayers.Geometry.Point(startingTileExtent.left, startingTileExtent.top); │ │ │ │ │ + var bottomRight = new OpenLayers.Geometry.Point(startingTileExtent.right, startingTileExtent.bottom); │ │ │ │ │ + if (this.useScales) { │ │ │ │ │ + this.scales = [] │ │ │ │ │ + } else { │ │ │ │ │ + this.resolutions = [] │ │ │ │ │ + } │ │ │ │ │ + this.lods = []; │ │ │ │ │ + for (var key in info.tileInfo.lods) { │ │ │ │ │ + if (info.tileInfo.lods.hasOwnProperty(key)) { │ │ │ │ │ + var lod = info.tileInfo.lods[key]; │ │ │ │ │ + if (this.useScales) { │ │ │ │ │ + this.scales.push(lod.scale) │ │ │ │ │ + } else { │ │ │ │ │ + this.resolutions.push(lod.resolution) │ │ │ │ │ + } │ │ │ │ │ + var start = this.getContainingTileCoords(upperLeft, lod.resolution); │ │ │ │ │ + lod.startTileCol = start.x; │ │ │ │ │ + lod.startTileRow = start.y; │ │ │ │ │ + var end = this.getContainingTileCoords(bottomRight, lod.resolution); │ │ │ │ │ + lod.endTileCol = end.x; │ │ │ │ │ + lod.endTileRow = end.y; │ │ │ │ │ + this.lods.push(lod) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + this.maxExtent = this.calculateMaxExtentWithLOD(this.lods[0]); │ │ │ │ │ + this.serverResolutions = this.resolutions; │ │ │ │ │ + if (this.overrideDPI && info.tileInfo.dpi) { │ │ │ │ │ + OpenLayers.DOTS_PER_INCH = info.tileInfo.dpi │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ - node.style.cursor = style.cursor │ │ │ │ │ + }, │ │ │ │ │ + getContainingTileCoords: function(point, res) { │ │ │ │ │ + return new OpenLayers.Pixel(Math.max(Math.floor((point.x - this.tileOrigin.lon) / (this.tileSize.w * res)), 0), Math.max(Math.floor((this.tileOrigin.lat - point.y) / (this.tileSize.h * res)), 0)) │ │ │ │ │ + }, │ │ │ │ │ + calculateMaxExtentWithLOD: function(lod) { │ │ │ │ │ + var numTileCols = lod.endTileCol - lod.startTileCol + 1; │ │ │ │ │ + var numTileRows = lod.endTileRow - lod.startTileRow + 1; │ │ │ │ │ + var minX = this.tileOrigin.lon + lod.startTileCol * this.tileSize.w * lod.resolution; │ │ │ │ │ + var maxX = minX + numTileCols * this.tileSize.w * lod.resolution; │ │ │ │ │ + var maxY = this.tileOrigin.lat - lod.startTileRow * this.tileSize.h * lod.resolution; │ │ │ │ │ + var minY = maxY - numTileRows * this.tileSize.h * lod.resolution; │ │ │ │ │ + return new OpenLayers.Bounds(minX, minY, maxX, maxY) │ │ │ │ │ + }, │ │ │ │ │ + calculateMaxExtentWithExtent: function(extent, res) { │ │ │ │ │ + var upperLeft = new OpenLayers.Geometry.Point(extent.left, extent.top); │ │ │ │ │ + var bottomRight = new OpenLayers.Geometry.Point(extent.right, extent.bottom); │ │ │ │ │ + var start = this.getContainingTileCoords(upperLeft, res); │ │ │ │ │ + var end = this.getContainingTileCoords(bottomRight, res); │ │ │ │ │ + var lod = { │ │ │ │ │ + resolution: res, │ │ │ │ │ + startTileCol: start.x, │ │ │ │ │ + startTileRow: start.y, │ │ │ │ │ + endTileCol: end.x, │ │ │ │ │ + endTileRow: end.y │ │ │ │ │ + }; │ │ │ │ │ + return this.calculateMaxExtentWithLOD(lod) │ │ │ │ │ + }, │ │ │ │ │ + getUpperLeftTileCoord: function(res) { │ │ │ │ │ + var upperLeft = new OpenLayers.Geometry.Point(this.maxExtent.left, this.maxExtent.top); │ │ │ │ │ + return this.getContainingTileCoords(upperLeft, res) │ │ │ │ │ + }, │ │ │ │ │ + getLowerRightTileCoord: function(res) { │ │ │ │ │ + var bottomRight = new OpenLayers.Geometry.Point(this.maxExtent.right, this.maxExtent.bottom); │ │ │ │ │ + return this.getContainingTileCoords(bottomRight, res) │ │ │ │ │ + }, │ │ │ │ │ + getMaxExtentForResolution: function(res) { │ │ │ │ │ + var start = this.getUpperLeftTileCoord(res); │ │ │ │ │ + var end = this.getLowerRightTileCoord(res); │ │ │ │ │ + var numTileCols = end.x - start.x + 1; │ │ │ │ │ + var numTileRows = end.y - start.y + 1; │ │ │ │ │ + var minX = this.tileOrigin.lon + start.x * this.tileSize.w * res; │ │ │ │ │ + var maxX = minX + numTileCols * this.tileSize.w * res; │ │ │ │ │ + var maxY = this.tileOrigin.lat - start.y * this.tileSize.h * res; │ │ │ │ │ + var minY = maxY - numTileRows * this.tileSize.h * res; │ │ │ │ │ + return new OpenLayers.Bounds(minX, minY, maxX, maxY) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.ArcGISCache(this.name, this.url, this.options) │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ + return OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]) │ │ │ │ │ }, │ │ │ │ │ - graphicRotate: function(node, xOffset, yOffset, style) { │ │ │ │ │ - var style = style || node._style; │ │ │ │ │ - var rotation = style.rotation || 0; │ │ │ │ │ - var aspectRatio, size; │ │ │ │ │ - if (!(style.graphicWidth && style.graphicHeight)) { │ │ │ │ │ - var img = new Image; │ │ │ │ │ - img.onreadystatechange = OpenLayers.Function.bind(function() { │ │ │ │ │ - if (img.readyState == "complete" || img.readyState == "interactive") { │ │ │ │ │ - aspectRatio = img.width / img.height; │ │ │ │ │ - size = Math.max(style.pointRadius * 2, style.graphicWidth || 0, style.graphicHeight || 0); │ │ │ │ │ - xOffset = xOffset * aspectRatio; │ │ │ │ │ - style.graphicWidth = size * aspectRatio; │ │ │ │ │ - style.graphicHeight = size; │ │ │ │ │ - this.graphicRotate(node, xOffset, yOffset, style) │ │ │ │ │ - } │ │ │ │ │ - }, this); │ │ │ │ │ - img.src = style.externalGraphic; │ │ │ │ │ - return │ │ │ │ │ + initGriddedTiles: function(bounds) { │ │ │ │ │ + delete this._tileOrigin; │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initGriddedTiles.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + getMaxExtent: function() { │ │ │ │ │ + var resolution = this.map.getResolution(); │ │ │ │ │ + return this.maxExtent = this.getMaxExtentForResolution(resolution) │ │ │ │ │ + }, │ │ │ │ │ + getTileOrigin: function() { │ │ │ │ │ + if (!this._tileOrigin) { │ │ │ │ │ + var extent = this.getMaxExtent(); │ │ │ │ │ + this._tileOrigin = new OpenLayers.LonLat(extent.left, extent.bottom) │ │ │ │ │ + } │ │ │ │ │ + return this._tileOrigin │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var res = this.getResolution(); │ │ │ │ │ + var originTileX = this.tileOrigin.lon + res * this.tileSize.w / 2; │ │ │ │ │ + var originTileY = this.tileOrigin.lat - res * this.tileSize.h / 2; │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var point = { │ │ │ │ │ + x: center.lon, │ │ │ │ │ + y: center.lat │ │ │ │ │ + }; │ │ │ │ │ + var x = Math.round(Math.abs((center.lon - originTileX) / (res * this.tileSize.w))); │ │ │ │ │ + var y = Math.round(Math.abs((originTileY - center.lat) / (res * this.tileSize.h))); │ │ │ │ │ + var z = this.map.getZoom(); │ │ │ │ │ + if (this.lods) { │ │ │ │ │ + var lod = this.lods[this.map.getZoom()]; │ │ │ │ │ + if (x < lod.startTileCol || x > lod.endTileCol || (y < lod.startTileRow || y > lod.endTileRow)) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ } else { │ │ │ │ │ - size = Math.max(style.graphicWidth, style.graphicHeight); │ │ │ │ │ - aspectRatio = style.graphicWidth / style.graphicHeight │ │ │ │ │ + var start = this.getUpperLeftTileCoord(res); │ │ │ │ │ + var end = this.getLowerRightTileCoord(res); │ │ │ │ │ + if (x < start.x || x >= end.x || (y < start.y || y >= end.y)) { │ │ │ │ │ + return null │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var width = Math.round(style.graphicWidth || size * aspectRatio); │ │ │ │ │ - var height = Math.round(style.graphicHeight || size); │ │ │ │ │ - node.style.width = width + "px"; │ │ │ │ │ - node.style.height = height + "px"; │ │ │ │ │ - var image = document.getElementById(node.id + "_image"); │ │ │ │ │ - if (!image) { │ │ │ │ │ - image = this.createNode("olv:imagedata", node.id + "_image"); │ │ │ │ │ - node.appendChild(image) │ │ │ │ │ + var url = this.url; │ │ │ │ │ + var s = "" + x + y + z; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(s, url) │ │ │ │ │ } │ │ │ │ │ - image.style.width = width + "px"; │ │ │ │ │ - image.style.height = height + "px"; │ │ │ │ │ - image.src = style.externalGraphic; │ │ │ │ │ - image.style.filter = "progid:DXImageTransform.Microsoft.AlphaImageLoader(" + "src='', sizingMethod='scale')"; │ │ │ │ │ - var rot = rotation * Math.PI / 180; │ │ │ │ │ - var sintheta = Math.sin(rot); │ │ │ │ │ - var costheta = Math.cos(rot); │ │ │ │ │ - var filter = "progid:DXImageTransform.Microsoft.Matrix(M11=" + costheta + ",M12=" + -sintheta + ",M21=" + sintheta + ",M22=" + costheta + ",SizingMethod='auto expand')\n"; │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ - if (opacity && opacity != 1) { │ │ │ │ │ - filter += "progid:DXImageTransform.Microsoft.BasicImage(opacity=" + opacity + ")\n" │ │ │ │ │ + if (this.useArcGISServer) { │ │ │ │ │ + url = url + "/tile/${z}/${y}/${x}" │ │ │ │ │ + } else { │ │ │ │ │ + x = "C" + OpenLayers.Number.zeroPad(x, 8, 16); │ │ │ │ │ + y = "R" + OpenLayers.Number.zeroPad(y, 8, 16); │ │ │ │ │ + z = "L" + OpenLayers.Number.zeroPad(z, 2, 10); │ │ │ │ │ + url = url + "/${z}/${y}/${x}." + this.type │ │ │ │ │ } │ │ │ │ │ - node.style.filter = filter; │ │ │ │ │ - var centerPoint = new OpenLayers.Geometry.Point(-xOffset, -yOffset); │ │ │ │ │ - var imgBox = new OpenLayers.Bounds(0, 0, width, height).toGeometry(); │ │ │ │ │ - imgBox.rotate(style.rotation, centerPoint); │ │ │ │ │ - var imgBounds = imgBox.getBounds(); │ │ │ │ │ - node.style.left = Math.round(parseInt(node.style.left) + imgBounds.left) + "px"; │ │ │ │ │ - node.style.top = Math.round(parseInt(node.style.top) - imgBounds.bottom) + "px" │ │ │ │ │ + url = OpenLayers.String.format(url, { │ │ │ │ │ + x: x, │ │ │ │ │ + y: y, │ │ │ │ │ + z: z │ │ │ │ │ + }); │ │ │ │ │ + return OpenLayers.Util.urlAppend(url, OpenLayers.Util.getParameterString(this.params)) │ │ │ │ │ }, │ │ │ │ │ - postDraw: function(node) { │ │ │ │ │ - node.style.visibility = "visible"; │ │ │ │ │ - var fillColor = node._style.fillColor; │ │ │ │ │ - var strokeColor = node._style.strokeColor; │ │ │ │ │ - if (fillColor == "none" && node.fillcolor != fillColor) { │ │ │ │ │ - node.fillcolor = fillColor │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.ArcGISCache" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.ArcIMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + ClientVersion: "9.2", │ │ │ │ │ + ServiceName: "" │ │ │ │ │ + }, │ │ │ │ │ + featureCoordSys: "4326", │ │ │ │ │ + filterCoordSys: "4326", │ │ │ │ │ + layers: null, │ │ │ │ │ + async: true, │ │ │ │ │ + name: "ArcIMS", │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + DEFAULT_OPTIONS: { │ │ │ │ │ + tileSize: new OpenLayers.Size(512, 512), │ │ │ │ │ + featureCoordSys: "4326", │ │ │ │ │ + filterCoordSys: "4326", │ │ │ │ │ + layers: null, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + async: true, │ │ │ │ │ + name: "ArcIMS" │ │ │ │ │ + }, │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + this.tileSize = new OpenLayers.Size(512, 512); │ │ │ │ │ + this.params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + ServiceName: options.serviceName │ │ │ │ │ + }, this.DEFAULT_PARAMS); │ │ │ │ │ + this.options = OpenLayers.Util.applyDefaults(options, this.DEFAULT_OPTIONS); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name, url, this.params, options]); │ │ │ │ │ + if (this.transparent) { │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + this.isBaseLayer = false │ │ │ │ │ + } │ │ │ │ │ + if (this.format == "image/jpeg") { │ │ │ │ │ + this.format = OpenLayers.Util.alphaHack() ? "image/gif" : "image/png" │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (strokeColor == "none" && node.strokecolor != strokeColor) { │ │ │ │ │ - node.strokecolor = strokeColor │ │ │ │ │ + if (this.options.layers === null) { │ │ │ │ │ + this.options.layers = [] │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - setNodeDimension: function(node, geometry) { │ │ │ │ │ - var bbox = geometry.getBounds(); │ │ │ │ │ - if (bbox) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var scaledBox = new OpenLayers.Bounds((bbox.left - this.featureDx) / resolution - this.offset.x | 0, bbox.bottom / resolution - this.offset.y | 0, (bbox.right - this.featureDx) / resolution - this.offset.x | 0, bbox.top / resolution - this.offset.y | 0); │ │ │ │ │ - node.style.left = scaledBox.left + "px"; │ │ │ │ │ - node.style.top = scaledBox.top + "px"; │ │ │ │ │ - node.style.width = scaledBox.getWidth() + "px"; │ │ │ │ │ - node.style.height = scaledBox.getHeight() + "px"; │ │ │ │ │ - node.coordorigin = scaledBox.left + " " + scaledBox.top; │ │ │ │ │ - node.coordsize = scaledBox.getWidth() + " " + scaledBox.getHeight() │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var url = ""; │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var axlReq = new OpenLayers.Format.ArcXML(OpenLayers.Util.extend(this.options, { │ │ │ │ │ + requesttype: "image", │ │ │ │ │ + envelope: bounds.toArray(), │ │ │ │ │ + tileSize: this.tileSize │ │ │ │ │ + })); │ │ │ │ │ + var req = new OpenLayers.Request.POST({ │ │ │ │ │ + url: this.getFullRequestString(), │ │ │ │ │ + data: axlReq.write(), │ │ │ │ │ + async: false │ │ │ │ │ + }); │ │ │ │ │ + if (req != null) { │ │ │ │ │ + var doc = req.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = req.responseText │ │ │ │ │ + } │ │ │ │ │ + var axlResp = new OpenLayers.Format.ArcXML; │ │ │ │ │ + var arcxml = axlResp.read(doc); │ │ │ │ │ + url = this.getUrlOrImage(arcxml.image.output) │ │ │ │ │ } │ │ │ │ │ + return url │ │ │ │ │ }, │ │ │ │ │ - dashStyle: function(style) { │ │ │ │ │ - var dash = style.strokeDashstyle; │ │ │ │ │ - switch (dash) { │ │ │ │ │ - case "solid": │ │ │ │ │ - case "dot": │ │ │ │ │ - case "dash": │ │ │ │ │ - case "dashdot": │ │ │ │ │ - case "longdash": │ │ │ │ │ - case "longdashdot": │ │ │ │ │ - return dash; │ │ │ │ │ - default: │ │ │ │ │ - var parts = dash.split(/[ ,]/); │ │ │ │ │ - if (parts.length == 2) { │ │ │ │ │ - if (1 * parts[0] >= 2 * parts[1]) { │ │ │ │ │ - return "longdash" │ │ │ │ │ - } │ │ │ │ │ - return parts[0] == 1 || parts[1] == 1 ? "dot" : "dash" │ │ │ │ │ - } else if (parts.length == 4) { │ │ │ │ │ - return 1 * parts[0] >= 2 * parts[1] ? "longdashdot" : "dashdot" │ │ │ │ │ + getURLasync: function(bounds, callback, scope) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var axlReq = new OpenLayers.Format.ArcXML(OpenLayers.Util.extend(this.options, { │ │ │ │ │ + requesttype: "image", │ │ │ │ │ + envelope: bounds.toArray(), │ │ │ │ │ + tileSize: this.tileSize │ │ │ │ │ + })); │ │ │ │ │ + OpenLayers.Request.POST({ │ │ │ │ │ + url: this.getFullRequestString(), │ │ │ │ │ + async: true, │ │ │ │ │ + data: axlReq.write(), │ │ │ │ │ + callback: function(req) { │ │ │ │ │ + var doc = req.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = req.responseText │ │ │ │ │ } │ │ │ │ │ - return "solid" │ │ │ │ │ + var axlResp = new OpenLayers.Format.ArcXML; │ │ │ │ │ + var arcxml = axlResp.read(doc); │ │ │ │ │ + callback.call(scope, this.getUrlOrImage(arcxml.image.output)) │ │ │ │ │ + }, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + getUrlOrImage: function(output) { │ │ │ │ │ + var ret = ""; │ │ │ │ │ + if (output.url) { │ │ │ │ │ + ret = output.url │ │ │ │ │ + } else if (output.data) { │ │ │ │ │ + ret = "data:image/" + output.type + ";base64," + output.data │ │ │ │ │ } │ │ │ │ │ + return ret │ │ │ │ │ }, │ │ │ │ │ - createNode: function(type, id) { │ │ │ │ │ - var node = document.createElement(type); │ │ │ │ │ - if (id) { │ │ │ │ │ - node.id = id │ │ │ │ │ + setLayerQuery: function(id, querydef) { │ │ │ │ │ + for (var lyr = 0; lyr < this.options.layers.length; lyr++) { │ │ │ │ │ + if (id == this.options.layers[lyr].id) { │ │ │ │ │ + this.options.layers[lyr].query = querydef; │ │ │ │ │ + return │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - node.unselectable = "on"; │ │ │ │ │ - node.onselectstart = OpenLayers.Function.False; │ │ │ │ │ - return node │ │ │ │ │ + this.options.layers.push({ │ │ │ │ │ + id: id, │ │ │ │ │ + visible: true, │ │ │ │ │ + query: querydef │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - nodeTypeCompare: function(node, type) { │ │ │ │ │ - var subType = type; │ │ │ │ │ - var splitIndex = subType.indexOf(":"); │ │ │ │ │ - if (splitIndex != -1) { │ │ │ │ │ - subType = subType.substr(splitIndex + 1) │ │ │ │ │ + getFeatureInfo: function(geometry, layer, options) { │ │ │ │ │ + var buffer = options.buffer || 1; │ │ │ │ │ + var callback = options.callback || function() {}; │ │ │ │ │ + var scope = options.scope || window; │ │ │ │ │ + var requestOptions = {}; │ │ │ │ │ + OpenLayers.Util.extend(requestOptions, this.options); │ │ │ │ │ + requestOptions.requesttype = "feature"; │ │ │ │ │ + if (geometry instanceof OpenLayers.LonLat) { │ │ │ │ │ + requestOptions.polygon = null; │ │ │ │ │ + requestOptions.envelope = [geometry.lon - buffer, geometry.lat - buffer, geometry.lon + buffer, geometry.lat + buffer] │ │ │ │ │ + } else if (geometry instanceof OpenLayers.Geometry.Polygon) { │ │ │ │ │ + requestOptions.envelope = null; │ │ │ │ │ + requestOptions.polygon = geometry │ │ │ │ │ } │ │ │ │ │ - var nodeName = node.nodeName; │ │ │ │ │ - splitIndex = nodeName.indexOf(":"); │ │ │ │ │ - if (splitIndex != -1) { │ │ │ │ │ - nodeName = nodeName.substr(splitIndex + 1) │ │ │ │ │ + var arcxml = new OpenLayers.Format.ArcXML(requestOptions); │ │ │ │ │ + OpenLayers.Util.extend(arcxml.request.get_feature, options); │ │ │ │ │ + arcxml.request.get_feature.layer = layer.id; │ │ │ │ │ + if (typeof layer.query.accuracy == "number") { │ │ │ │ │ + arcxml.request.get_feature.query.accuracy = layer.query.accuracy │ │ │ │ │ + } else { │ │ │ │ │ + var mapCenter = this.map.getCenter(); │ │ │ │ │ + var viewPx = this.map.getViewPortPxFromLonLat(mapCenter); │ │ │ │ │ + viewPx.x++; │ │ │ │ │ + var mapOffCenter = this.map.getLonLatFromPixel(viewPx); │ │ │ │ │ + arcxml.request.get_feature.query.accuracy = mapOffCenter.lon - mapCenter.lon │ │ │ │ │ } │ │ │ │ │ - return subType == nodeName │ │ │ │ │ + arcxml.request.get_feature.query.where = layer.query.where; │ │ │ │ │ + arcxml.request.get_feature.query.spatialfilter.relation = "area_intersection"; │ │ │ │ │ + OpenLayers.Request.POST({ │ │ │ │ │ + url: this.getFullRequestString({ │ │ │ │ │ + CustomService: "Query" │ │ │ │ │ + }), │ │ │ │ │ + data: arcxml.write(), │ │ │ │ │ + callback: function(request) { │ │ │ │ │ + var response = arcxml.parseResponse(request.responseText); │ │ │ │ │ + if (!arcxml.iserror()) { │ │ │ │ │ + callback.call(scope, response.features) │ │ │ │ │ + } else { │ │ │ │ │ + callback.call(scope, null) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - createRenderRoot: function() { │ │ │ │ │ - return this.nodeFactory(this.container.id + "_vmlRoot", "div") │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.ArcIMS(this.name, this.url, this.getOptions()) │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - createRoot: function(suffix) { │ │ │ │ │ - return this.nodeFactory(this.container.id + suffix, "olv:group") │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.ArcIMS" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.PointGrid = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ + dx: null, │ │ │ │ │ + dy: null, │ │ │ │ │ + ratio: 1.5, │ │ │ │ │ + maxFeatures: 250, │ │ │ │ │ + rotation: 0, │ │ │ │ │ + origin: null, │ │ │ │ │ + gridBounds: null, │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ + config = config || {}; │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.initialize.apply(this, [config.name, config]) │ │ │ │ │ }, │ │ │ │ │ - drawPoint: function(node, geometry) { │ │ │ │ │ - return this.drawCircle(node, geometry, 1) │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.setMap.apply(this, arguments); │ │ │ │ │ + map.events.register("moveend", this, this.onMoveEnd) │ │ │ │ │ }, │ │ │ │ │ - drawCircle: function(node, geometry, radius) { │ │ │ │ │ - if (!isNaN(geometry.x) && !isNaN(geometry.y)) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x | 0) - radius + "px"; │ │ │ │ │ - node.style.top = (geometry.y / resolution - this.offset.y | 0) - radius + "px"; │ │ │ │ │ - var diameter = radius * 2; │ │ │ │ │ - node.style.width = diameter + "px"; │ │ │ │ │ - node.style.height = diameter + "px"; │ │ │ │ │ - return node │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + map.events.unregister("moveend", this, this.onMoveEnd); │ │ │ │ │ + OpenLayers.Layer.Vector.prototype.removeMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + setRatio: function(ratio) { │ │ │ │ │ + this.ratio = ratio; │ │ │ │ │ + this.updateGrid(true) │ │ │ │ │ + }, │ │ │ │ │ + setMaxFeatures: function(maxFeatures) { │ │ │ │ │ + this.maxFeatures = maxFeatures; │ │ │ │ │ + this.updateGrid(true) │ │ │ │ │ + }, │ │ │ │ │ + setSpacing: function(dx, dy) { │ │ │ │ │ + this.dx = dx; │ │ │ │ │ + this.dy = dy || dx; │ │ │ │ │ + this.updateGrid(true) │ │ │ │ │ + }, │ │ │ │ │ + setOrigin: function(origin) { │ │ │ │ │ + this.origin = origin; │ │ │ │ │ + this.updateGrid(true) │ │ │ │ │ + }, │ │ │ │ │ + getOrigin: function() { │ │ │ │ │ + if (!this.origin) { │ │ │ │ │ + this.origin = this.map.getExtent().getCenterLonLat() │ │ │ │ │ } │ │ │ │ │ - return false │ │ │ │ │ + return this.origin │ │ │ │ │ }, │ │ │ │ │ - drawLineString: function(node, geometry) { │ │ │ │ │ - return this.drawLine(node, geometry, false) │ │ │ │ │ + setRotation: function(rotation) { │ │ │ │ │ + this.rotation = rotation; │ │ │ │ │ + this.updateGrid(true) │ │ │ │ │ }, │ │ │ │ │ - drawLinearRing: function(node, geometry) { │ │ │ │ │ - return this.drawLine(node, geometry, true) │ │ │ │ │ + onMoveEnd: function() { │ │ │ │ │ + this.updateGrid() │ │ │ │ │ }, │ │ │ │ │ - drawLine: function(node, geometry, closeLine) { │ │ │ │ │ - this.setNodeDimension(node, geometry); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var numComponents = geometry.components.length; │ │ │ │ │ - var parts = new Array(numComponents); │ │ │ │ │ - var comp, x, y; │ │ │ │ │ - for (var i = 0; i < numComponents; i++) { │ │ │ │ │ - comp = geometry.components[i]; │ │ │ │ │ - x = (comp.x - this.featureDx) / resolution - this.offset.x | 0; │ │ │ │ │ - y = comp.y / resolution - this.offset.y | 0; │ │ │ │ │ - parts[i] = " " + x + "," + y + " l " │ │ │ │ │ + getViewBounds: function() { │ │ │ │ │ + var bounds = this.map.getExtent(); │ │ │ │ │ + if (this.rotation) { │ │ │ │ │ + var origin = this.getOrigin(); │ │ │ │ │ + var rotationOrigin = new OpenLayers.Geometry.Point(origin.lon, origin.lat); │ │ │ │ │ + var rect = bounds.toGeometry(); │ │ │ │ │ + rect.rotate(-this.rotation, rotationOrigin); │ │ │ │ │ + bounds = rect.getBounds() │ │ │ │ │ } │ │ │ │ │ - var end = closeLine ? " x e" : " e"; │ │ │ │ │ - node.path = "m" + parts.join("") + end; │ │ │ │ │ - return node │ │ │ │ │ + return bounds │ │ │ │ │ }, │ │ │ │ │ - drawPolygon: function(node, geometry) { │ │ │ │ │ - this.setNodeDimension(node, geometry); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var path = []; │ │ │ │ │ - var j, jj, points, area, first, second, i, ii, comp, pathComp, x, y; │ │ │ │ │ - for (j = 0, jj = geometry.components.length; j < jj; j++) { │ │ │ │ │ - path.push("m"); │ │ │ │ │ - points = geometry.components[j].components; │ │ │ │ │ - area = j === 0; │ │ │ │ │ - first = null; │ │ │ │ │ - second = null; │ │ │ │ │ - for (i = 0, ii = points.length; i < ii; i++) { │ │ │ │ │ - comp = points[i]; │ │ │ │ │ - x = (comp.x - this.featureDx) / resolution - this.offset.x | 0; │ │ │ │ │ - y = comp.y / resolution - this.offset.y | 0; │ │ │ │ │ - pathComp = " " + x + "," + y; │ │ │ │ │ - path.push(pathComp); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - path.push(" l") │ │ │ │ │ - } │ │ │ │ │ - if (!area) { │ │ │ │ │ - if (!first) { │ │ │ │ │ - first = pathComp │ │ │ │ │ - } else if (first != pathComp) { │ │ │ │ │ - if (!second) { │ │ │ │ │ - second = pathComp │ │ │ │ │ - } else if (second != pathComp) { │ │ │ │ │ - area = true │ │ │ │ │ - } │ │ │ │ │ + updateGrid: function(force) { │ │ │ │ │ + if (force || this.invalidBounds()) { │ │ │ │ │ + var viewBounds = this.getViewBounds(); │ │ │ │ │ + var origin = this.getOrigin(); │ │ │ │ │ + var rotationOrigin = new OpenLayers.Geometry.Point(origin.lon, origin.lat); │ │ │ │ │ + var viewBoundsWidth = viewBounds.getWidth(); │ │ │ │ │ + var viewBoundsHeight = viewBounds.getHeight(); │ │ │ │ │ + var aspectRatio = viewBoundsWidth / viewBoundsHeight; │ │ │ │ │ + var maxHeight = Math.sqrt(this.dx * this.dy * this.maxFeatures / aspectRatio); │ │ │ │ │ + var maxWidth = maxHeight * aspectRatio; │ │ │ │ │ + var gridWidth = Math.min(viewBoundsWidth * this.ratio, maxWidth); │ │ │ │ │ + var gridHeight = Math.min(viewBoundsHeight * this.ratio, maxHeight); │ │ │ │ │ + var center = viewBounds.getCenterLonLat(); │ │ │ │ │ + this.gridBounds = new OpenLayers.Bounds(center.lon - gridWidth / 2, center.lat - gridHeight / 2, center.lon + gridWidth / 2, center.lat + gridHeight / 2); │ │ │ │ │ + var rows = Math.floor(gridHeight / this.dy); │ │ │ │ │ + var cols = Math.floor(gridWidth / this.dx); │ │ │ │ │ + var gridLeft = origin.lon + this.dx * Math.ceil((this.gridBounds.left - origin.lon) / this.dx); │ │ │ │ │ + var gridBottom = origin.lat + this.dy * Math.ceil((this.gridBounds.bottom - origin.lat) / this.dy); │ │ │ │ │ + var features = new Array(rows * cols); │ │ │ │ │ + var x, y, point; │ │ │ │ │ + for (var i = 0; i < cols; ++i) { │ │ │ │ │ + x = gridLeft + i * this.dx; │ │ │ │ │ + for (var j = 0; j < rows; ++j) { │ │ │ │ │ + y = gridBottom + j * this.dy; │ │ │ │ │ + point = new OpenLayers.Geometry.Point(x, y); │ │ │ │ │ + if (this.rotation) { │ │ │ │ │ + point.rotate(this.rotation, rotationOrigin) │ │ │ │ │ } │ │ │ │ │ + features[i * rows + j] = new OpenLayers.Feature.Vector(point) │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - path.push(area ? " x " : " ") │ │ │ │ │ + this.destroyFeatures(this.features, { │ │ │ │ │ + silent: true │ │ │ │ │ + }); │ │ │ │ │ + this.addFeatures(features, { │ │ │ │ │ + silent: true │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - path.push("e"); │ │ │ │ │ - node.path = path.join(""); │ │ │ │ │ - return node │ │ │ │ │ }, │ │ │ │ │ - drawRectangle: function(node, geometry) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - node.style.left = ((geometry.x - this.featureDx) / resolution - this.offset.x | 0) + "px"; │ │ │ │ │ - node.style.top = (geometry.y / resolution - this.offset.y | 0) + "px"; │ │ │ │ │ - node.style.width = (geometry.width / resolution | 0) + "px"; │ │ │ │ │ - node.style.height = (geometry.height / resolution | 0) + "px"; │ │ │ │ │ - return node │ │ │ │ │ + invalidBounds: function() { │ │ │ │ │ + return !this.gridBounds || !this.gridBounds.containsBounds(this.getViewBounds()) │ │ │ │ │ }, │ │ │ │ │ - drawText: function(featureId, style, location) { │ │ │ │ │ - var label = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX, "olv:rect"); │ │ │ │ │ - var textbox = this.nodeFactory(featureId + this.LABEL_ID_SUFFIX + "_textbox", "olv:textbox"); │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - label.style.left = ((location.x - this.featureDx) / resolution - this.offset.x | 0) + "px"; │ │ │ │ │ - label.style.top = (location.y / resolution - this.offset.y | 0) + "px"; │ │ │ │ │ - label.style.flip = "y"; │ │ │ │ │ - textbox.innerText = style.label; │ │ │ │ │ - if (style.cursor != "inherit" && style.cursor != null) { │ │ │ │ │ - textbox.style.cursor = style.cursor │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.PointGrid" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.TileCache = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + format: "image/png", │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + initialize: function(name, url, layername, options) { │ │ │ │ │ + this.layername = layername; │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name, url, {}, options]); │ │ │ │ │ + this.extension = this.format.split("/")[1].toLowerCase(); │ │ │ │ │ + this.extension = this.extension == "jpg" ? "jpeg" : this.extension │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.TileCache(this.name, this.url, this.layername, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - if (style.fontColor) { │ │ │ │ │ - textbox.style.color = style.fontColor │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var bbox = this.maxExtent; │ │ │ │ │ + var size = this.tileSize; │ │ │ │ │ + var tileX = Math.round((bounds.left - bbox.left) / (res * size.w)); │ │ │ │ │ + var tileY = Math.round((bounds.bottom - bbox.bottom) / (res * size.h)); │ │ │ │ │ + var tileZ = this.serverResolutions != null ? OpenLayers.Util.indexOf(this.serverResolutions, res) : this.map.getZoom(); │ │ │ │ │ + var components = [this.layername, OpenLayers.Number.zeroPad(tileZ, 2), OpenLayers.Number.zeroPad(parseInt(tileX / 1e6), 3), OpenLayers.Number.zeroPad(parseInt(tileX / 1e3) % 1e3, 3), OpenLayers.Number.zeroPad(parseInt(tileX) % 1e3, 3), OpenLayers.Number.zeroPad(parseInt(tileY / 1e6), 3), OpenLayers.Number.zeroPad(parseInt(tileY / 1e3) % 1e3, 3), OpenLayers.Number.zeroPad(parseInt(tileY) % 1e3, 3) + "." + this.extension]; │ │ │ │ │ + var path = components.join("/"); │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(path, url) │ │ │ │ │ } │ │ │ │ │ - if (style.fontOpacity) { │ │ │ │ │ - textbox.style.filter = "alpha(opacity=" + style.fontOpacity * 100 + ")" │ │ │ │ │ + url = url.charAt(url.length - 1) == "/" ? url : url + "/"; │ │ │ │ │ + return url + path │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.TileCache" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.WMTS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + version: "1.0.0", │ │ │ │ │ + requestEncoding: "KVP", │ │ │ │ │ + url: null, │ │ │ │ │ + layer: null, │ │ │ │ │ + matrixSet: null, │ │ │ │ │ + style: null, │ │ │ │ │ + format: "image/jpeg", │ │ │ │ │ + tileOrigin: null, │ │ │ │ │ + tileFullExtent: null, │ │ │ │ │ + formatSuffix: null, │ │ │ │ │ + matrixIds: null, │ │ │ │ │ + dimensions: null, │ │ │ │ │ + params: null, │ │ │ │ │ + zoomOffset: 0, │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + formatSuffixMap: { │ │ │ │ │ + "image/png": "png", │ │ │ │ │ + "image/png8": "png", │ │ │ │ │ + "image/png24": "png", │ │ │ │ │ + "image/png32": "png", │ │ │ │ │ + png: "png", │ │ │ │ │ + "image/jpeg": "jpg", │ │ │ │ │ + "image/jpg": "jpg", │ │ │ │ │ + jpeg: "jpg", │ │ │ │ │ + jpg: "jpg" │ │ │ │ │ + }, │ │ │ │ │ + matrix: null, │ │ │ │ │ + initialize: function(config) { │ │ │ │ │ + var required = { │ │ │ │ │ + url: true, │ │ │ │ │ + layer: true, │ │ │ │ │ + style: true, │ │ │ │ │ + matrixSet: true │ │ │ │ │ + }; │ │ │ │ │ + for (var prop in required) { │ │ │ │ │ + if (!(prop in config)) { │ │ │ │ │ + throw new Error("Missing property '" + prop + "' in layer configuration.") │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (style.fontFamily) { │ │ │ │ │ - textbox.style.fontFamily = style.fontFamily │ │ │ │ │ + config.params = OpenLayers.Util.upperCaseObject(config.params); │ │ │ │ │ + var args = [config.name, config.url, config.params, config]; │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, args); │ │ │ │ │ + if (!this.formatSuffix) { │ │ │ │ │ + this.formatSuffix = this.formatSuffixMap[this.format] || this.format.split("/").pop() │ │ │ │ │ } │ │ │ │ │ - if (style.fontSize) { │ │ │ │ │ - textbox.style.fontSize = style.fontSize │ │ │ │ │ + if (this.matrixIds) { │ │ │ │ │ + var len = this.matrixIds.length; │ │ │ │ │ + if (len && typeof this.matrixIds[0] === "string") { │ │ │ │ │ + var ids = this.matrixIds; │ │ │ │ │ + this.matrixIds = new Array(len); │ │ │ │ │ + for (var i = 0; i < len; ++i) { │ │ │ │ │ + this.matrixIds[i] = { │ │ │ │ │ + identifier: ids[i] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (style.fontWeight) { │ │ │ │ │ - textbox.style.fontWeight = style.fontWeight │ │ │ │ │ + }, │ │ │ │ │ + setMap: function() { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + updateMatrixProperties: function() { │ │ │ │ │ + this.matrix = this.getMatrix(); │ │ │ │ │ + if (this.matrix) { │ │ │ │ │ + if (this.matrix.topLeftCorner) { │ │ │ │ │ + this.tileOrigin = this.matrix.topLeftCorner │ │ │ │ │ + } │ │ │ │ │ + if (this.matrix.tileWidth && this.matrix.tileHeight) { │ │ │ │ │ + this.tileSize = new OpenLayers.Size(this.matrix.tileWidth, this.matrix.tileHeight) │ │ │ │ │ + } │ │ │ │ │ + if (!this.tileOrigin) { │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.maxExtent.left, this.maxExtent.top) │ │ │ │ │ + } │ │ │ │ │ + if (!this.tileFullExtent) { │ │ │ │ │ + this.tileFullExtent = this.maxExtent │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (style.fontStyle) { │ │ │ │ │ - textbox.style.fontStyle = style.fontStyle │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + if (zoomChanged || !this.matrix) { │ │ │ │ │ + this.updateMatrixProperties() │ │ │ │ │ } │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - label._featureId = featureId; │ │ │ │ │ - textbox._featureId = featureId; │ │ │ │ │ - textbox._geometry = location; │ │ │ │ │ - textbox._geometryClass = location.CLASS_NAME │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.moveTo.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.WMTS(this.options) │ │ │ │ │ } │ │ │ │ │ - textbox.style.whiteSpace = "nowrap"; │ │ │ │ │ - textbox.inset = "1px,0px,0px,0px"; │ │ │ │ │ - if (!label.parentNode) { │ │ │ │ │ - label.appendChild(textbox); │ │ │ │ │ - this.textRoot.appendChild(label) │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + getIdentifier: function() { │ │ │ │ │ + return this.getServerZoom() │ │ │ │ │ + }, │ │ │ │ │ + getMatrix: function() { │ │ │ │ │ + var matrix; │ │ │ │ │ + if (!this.matrixIds || this.matrixIds.length === 0) { │ │ │ │ │ + matrix = { │ │ │ │ │ + identifier: this.getIdentifier() │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + if ("scaleDenominator" in this.matrixIds[0]) { │ │ │ │ │ + var denom = OpenLayers.METERS_PER_INCH * OpenLayers.INCHES_PER_UNIT[this.units] * this.getServerResolution() / 28e-5; │ │ │ │ │ + var diff = Number.POSITIVE_INFINITY; │ │ │ │ │ + var delta; │ │ │ │ │ + for (var i = 0, ii = this.matrixIds.length; i < ii; ++i) { │ │ │ │ │ + delta = Math.abs(1 - this.matrixIds[i].scaleDenominator / denom); │ │ │ │ │ + if (delta < diff) { │ │ │ │ │ + diff = delta; │ │ │ │ │ + matrix = this.matrixIds[i] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } else { │ │ │ │ │ + matrix = this.matrixIds[this.getIdentifier()] │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - var align = style.labelAlign || "cm"; │ │ │ │ │ - if (align.length == 1) { │ │ │ │ │ - align += "m" │ │ │ │ │ + return matrix │ │ │ │ │ + }, │ │ │ │ │ + getTileInfo: function(loc) { │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var fx = (loc.lon - this.tileOrigin.lon) / (res * this.tileSize.w); │ │ │ │ │ + var fy = (this.tileOrigin.lat - loc.lat) / (res * this.tileSize.h); │ │ │ │ │ + var col = Math.floor(fx); │ │ │ │ │ + var row = Math.floor(fy); │ │ │ │ │ + return { │ │ │ │ │ + col: col, │ │ │ │ │ + row: row, │ │ │ │ │ + i: Math.floor((fx - col) * this.tileSize.w), │ │ │ │ │ + j: Math.floor((fy - row) * this.tileSize.h) │ │ │ │ │ } │ │ │ │ │ - var xshift = textbox.clientWidth * OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(0, 1)]; │ │ │ │ │ - var yshift = textbox.clientHeight * OpenLayers.Renderer.VML.LABEL_SHIFT[align.substr(1, 1)]; │ │ │ │ │ - label.style.left = parseInt(label.style.left) - xshift - 1 + "px"; │ │ │ │ │ - label.style.top = parseInt(label.style.top) + yshift + "px" │ │ │ │ │ }, │ │ │ │ │ - moveRoot: function(renderer) { │ │ │ │ │ - var layer = this.map.getLayer(renderer.container.id); │ │ │ │ │ - if (layer instanceof OpenLayers.Layer.Vector.RootContainer) { │ │ │ │ │ - layer = this.map.getLayer(this.container.id) │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var url = ""; │ │ │ │ │ + if (!this.tileFullExtent || this.tileFullExtent.intersectsBounds(bounds)) { │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var info = this.getTileInfo(center); │ │ │ │ │ + var matrixId = this.matrix.identifier; │ │ │ │ │ + var dimensions = this.dimensions, │ │ │ │ │ + params; │ │ │ │ │ + if (OpenLayers.Util.isArray(this.url)) { │ │ │ │ │ + url = this.selectUrl([this.version, this.style, this.matrixSet, this.matrix.identifier, info.row, info.col].join(","), this.url) │ │ │ │ │ + } else { │ │ │ │ │ + url = this.url │ │ │ │ │ + } │ │ │ │ │ + if (this.requestEncoding.toUpperCase() === "REST") { │ │ │ │ │ + params = this.params; │ │ │ │ │ + if (url.indexOf("{") !== -1) { │ │ │ │ │ + var template = url.replace(/\{/g, "${"); │ │ │ │ │ + var context = { │ │ │ │ │ + style: this.style, │ │ │ │ │ + Style: this.style, │ │ │ │ │ + TileMatrixSet: this.matrixSet, │ │ │ │ │ + TileMatrix: this.matrix.identifier, │ │ │ │ │ + TileRow: info.row, │ │ │ │ │ + TileCol: info.col │ │ │ │ │ + }; │ │ │ │ │ + if (dimensions) { │ │ │ │ │ + var dimension, i; │ │ │ │ │ + for (i = dimensions.length - 1; i >= 0; --i) { │ │ │ │ │ + dimension = dimensions[i]; │ │ │ │ │ + context[dimension] = params[dimension.toUpperCase()] │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + url = OpenLayers.String.format(template, context) │ │ │ │ │ + } else { │ │ │ │ │ + var path = this.version + "/" + this.layer + "/" + this.style + "/"; │ │ │ │ │ + if (dimensions) { │ │ │ │ │ + for (var i = 0; i < dimensions.length; i++) { │ │ │ │ │ + if (params[dimensions[i]]) { │ │ │ │ │ + path = path + params[dimensions[i]] + "/" │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + path = path + this.matrixSet + "/" + this.matrix.identifier + "/" + info.row + "/" + info.col + "." + this.formatSuffix; │ │ │ │ │ + if (!url.match(/\/$/)) { │ │ │ │ │ + url = url + "/" │ │ │ │ │ + } │ │ │ │ │ + url = url + path │ │ │ │ │ + } │ │ │ │ │ + } else if (this.requestEncoding.toUpperCase() === "KVP") { │ │ │ │ │ + params = { │ │ │ │ │ + SERVICE: "WMTS", │ │ │ │ │ + REQUEST: "GetTile", │ │ │ │ │ + VERSION: this.version, │ │ │ │ │ + LAYER: this.layer, │ │ │ │ │ + STYLE: this.style, │ │ │ │ │ + TILEMATRIXSET: this.matrixSet, │ │ │ │ │ + TILEMATRIX: this.matrix.identifier, │ │ │ │ │ + TILEROW: info.row, │ │ │ │ │ + TILECOL: info.col, │ │ │ │ │ + FORMAT: this.format │ │ │ │ │ + }; │ │ │ │ │ + url = OpenLayers.Layer.Grid.prototype.getFullRequestString.apply(this, [params]) │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - layer && layer.renderer.clear(); │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.moveRoot.apply(this, arguments); │ │ │ │ │ - layer && layer.redraw() │ │ │ │ │ + return url │ │ │ │ │ }, │ │ │ │ │ - importSymbol: function(graphicName) { │ │ │ │ │ - var id = this.container.id + "-" + graphicName; │ │ │ │ │ - var cache = this.symbolCache[id]; │ │ │ │ │ - if (cache) { │ │ │ │ │ - return cache │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + if (this.requestEncoding.toUpperCase() === "KVP") { │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, [OpenLayers.Util.upperCaseObject(newParams)]) │ │ │ │ │ } │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(graphicName + " is not a valid symbol name") │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.WMTS" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.PointTrack = OpenLayers.Class(OpenLayers.Layer.Vector, { │ │ │ │ │ + dataFrom: null, │ │ │ │ │ + styleFrom: null, │ │ │ │ │ + addNodes: function(pointFeatures, options) { │ │ │ │ │ + if (pointFeatures.length < 2) { │ │ │ │ │ + throw new Error("At least two point features have to be added to " + "create a line from") │ │ │ │ │ } │ │ │ │ │ - var symbolExtent = new OpenLayers.Bounds(Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ - var pathitems = ["m"]; │ │ │ │ │ - for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - var x = symbol[i]; │ │ │ │ │ - var y = symbol[i + 1]; │ │ │ │ │ - symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ - symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ - symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ - symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ - pathitems.push(x); │ │ │ │ │ - pathitems.push(y); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - pathitems.push("l") │ │ │ │ │ + var lines = new Array(pointFeatures.length - 1); │ │ │ │ │ + var pointFeature, startPoint, endPoint; │ │ │ │ │ + for (var i = 0, len = pointFeatures.length; i < len; i++) { │ │ │ │ │ + pointFeature = pointFeatures[i]; │ │ │ │ │ + endPoint = pointFeature.geometry; │ │ │ │ │ + if (!endPoint) { │ │ │ │ │ + var lonlat = pointFeature.lonlat; │ │ │ │ │ + endPoint = new OpenLayers.Geometry.Point(lonlat.lon, lonlat.lat) │ │ │ │ │ + } else if (endPoint.CLASS_NAME != "OpenLayers.Geometry.Point") { │ │ │ │ │ + throw new TypeError("Only features with point geometries are supported.") │ │ │ │ │ } │ │ │ │ │ + if (i > 0) { │ │ │ │ │ + var attributes = this.dataFrom != null ? pointFeatures[i + this.dataFrom].data || pointFeatures[i + this.dataFrom].attributes : null; │ │ │ │ │ + var style = this.styleFrom != null ? pointFeatures[i + this.styleFrom].style : null; │ │ │ │ │ + var line = new OpenLayers.Geometry.LineString([startPoint, endPoint]); │ │ │ │ │ + lines[i - 1] = new OpenLayers.Feature.Vector(line, attributes, style) │ │ │ │ │ + } │ │ │ │ │ + startPoint = endPoint │ │ │ │ │ } │ │ │ │ │ - pathitems.push("x e"); │ │ │ │ │ - var path = pathitems.join(" "); │ │ │ │ │ - var diff = (symbolExtent.getWidth() - symbolExtent.getHeight()) / 2; │ │ │ │ │ - if (diff > 0) { │ │ │ │ │ - symbolExtent.bottom = symbolExtent.bottom - diff; │ │ │ │ │ - symbolExtent.top = symbolExtent.top + diff │ │ │ │ │ - } else { │ │ │ │ │ - symbolExtent.left = symbolExtent.left + diff; │ │ │ │ │ - symbolExtent.right = symbolExtent.right - diff │ │ │ │ │ - } │ │ │ │ │ - cache = { │ │ │ │ │ - path: path, │ │ │ │ │ - size: symbolExtent.getWidth(), │ │ │ │ │ - left: symbolExtent.left, │ │ │ │ │ - bottom: symbolExtent.bottom │ │ │ │ │ - }; │ │ │ │ │ - this.symbolCache[id] = cache; │ │ │ │ │ - return cache │ │ │ │ │ + this.addFeatures(lines, options) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.VML" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.PointTrack" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Renderer.VML.LABEL_SHIFT = { │ │ │ │ │ - l: 0, │ │ │ │ │ - c: .5, │ │ │ │ │ - r: 1, │ │ │ │ │ - t: 0, │ │ │ │ │ - m: .5, │ │ │ │ │ - b: 1 │ │ │ │ │ +OpenLayers.Layer.PointTrack.SOURCE_NODE = -1; │ │ │ │ │ +OpenLayers.Layer.PointTrack.TARGET_NODE = 0; │ │ │ │ │ +OpenLayers.Layer.PointTrack.dataFrom = { │ │ │ │ │ + SOURCE_NODE: -1, │ │ │ │ │ + TARGET_NODE: 0 │ │ │ │ │ }; │ │ │ │ │ -OpenLayers.Renderer.SVG = OpenLayers.Class(OpenLayers.Renderer.Elements, { │ │ │ │ │ - xmlns: "http://www.w3.org/2000/svg", │ │ │ │ │ - xlinkns: "http://www.w3.org/1999/xlink", │ │ │ │ │ - MAX_PIXEL: 15e3, │ │ │ │ │ - translationParameters: null, │ │ │ │ │ - symbolMetrics: null, │ │ │ │ │ - initialize: function(containerID) { │ │ │ │ │ - if (!this.supported()) { │ │ │ │ │ - return │ │ │ │ │ +OpenLayers.Layer.Zoomify = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + size: null, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + standardTileSize: 256, │ │ │ │ │ + tileOriginCorner: "tl", │ │ │ │ │ + numberOfTiers: 0, │ │ │ │ │ + tileCountUpToTier: null, │ │ │ │ │ + tierSizeInTiles: null, │ │ │ │ │ + tierImageSize: null, │ │ │ │ │ + initialize: function(name, url, size, options) { │ │ │ │ │ + this.initializeZoomify(size); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [name, url, size, {}, options]) │ │ │ │ │ + }, │ │ │ │ │ + initializeZoomify: function(size) { │ │ │ │ │ + var imageSize = size.clone(); │ │ │ │ │ + this.size = size.clone(); │ │ │ │ │ + var tiles = new OpenLayers.Size(Math.ceil(imageSize.w / this.standardTileSize), Math.ceil(imageSize.h / this.standardTileSize)); │ │ │ │ │ + this.tierSizeInTiles = [tiles]; │ │ │ │ │ + this.tierImageSize = [imageSize]; │ │ │ │ │ + while (imageSize.w > this.standardTileSize || imageSize.h > this.standardTileSize) { │ │ │ │ │ + imageSize = new OpenLayers.Size(Math.floor(imageSize.w / 2), Math.floor(imageSize.h / 2)); │ │ │ │ │ + tiles = new OpenLayers.Size(Math.ceil(imageSize.w / this.standardTileSize), Math.ceil(imageSize.h / this.standardTileSize)); │ │ │ │ │ + this.tierSizeInTiles.push(tiles); │ │ │ │ │ + this.tierImageSize.push(imageSize) │ │ │ │ │ + } │ │ │ │ │ + this.tierSizeInTiles.reverse(); │ │ │ │ │ + this.tierImageSize.reverse(); │ │ │ │ │ + this.numberOfTiers = this.tierSizeInTiles.length; │ │ │ │ │ + var resolutions = [1]; │ │ │ │ │ + this.tileCountUpToTier = [0]; │ │ │ │ │ + for (var i = 1; i < this.numberOfTiers; i++) { │ │ │ │ │ + resolutions.unshift(Math.pow(2, i)); │ │ │ │ │ + this.tileCountUpToTier.push(this.tierSizeInTiles[i - 1].w * this.tierSizeInTiles[i - 1].h + this.tileCountUpToTier[i - 1]) │ │ │ │ │ + } │ │ │ │ │ + if (!this.serverResolutions) { │ │ │ │ │ + this.serverResolutions = resolutions │ │ │ │ │ } │ │ │ │ │ - OpenLayers.Renderer.Elements.prototype.initialize.apply(this, arguments); │ │ │ │ │ - this.translationParameters = { │ │ │ │ │ - x: 0, │ │ │ │ │ - y: 0 │ │ │ │ │ - }; │ │ │ │ │ - this.symbolMetrics = {} │ │ │ │ │ }, │ │ │ │ │ - supported: function() { │ │ │ │ │ - var svgFeature = "http://www.w3.org/TR/SVG11/feature#"; │ │ │ │ │ - return document.implementation && (document.implementation.hasFeature("org.w3c.svg", "1.0") || document.implementation.hasFeature(svgFeature + "SVG", "1.1") || document.implementation.hasFeature(svgFeature + "BasicStructure", "1.1")) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.destroy.apply(this, arguments); │ │ │ │ │ + this.tileCountUpToTier.length = 0; │ │ │ │ │ + this.tierSizeInTiles.length = 0; │ │ │ │ │ + this.tierImageSize.length = 0 │ │ │ │ │ }, │ │ │ │ │ - inValidRange: function(x, y, xyOnly) { │ │ │ │ │ - var left = x + (xyOnly ? 0 : this.translationParameters.x); │ │ │ │ │ - var top = y + (xyOnly ? 0 : this.translationParameters.y); │ │ │ │ │ - return left >= -this.MAX_PIXEL && left <= this.MAX_PIXEL && top >= -this.MAX_PIXEL && top <= this.MAX_PIXEL │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Zoomify(this.name, this.url, this.size, this.options) │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - setExtent: function(extent, resolutionChanged) { │ │ │ │ │ - var coordSysUnchanged = OpenLayers.Renderer.Elements.prototype.setExtent.apply(this, arguments); │ │ │ │ │ - var resolution = this.getResolution(), │ │ │ │ │ - left = -extent.left / resolution, │ │ │ │ │ - top = extent.top / resolution; │ │ │ │ │ - if (resolutionChanged) { │ │ │ │ │ - this.left = left; │ │ │ │ │ - this.top = top; │ │ │ │ │ - var extentString = "0 0 " + this.size.w + " " + this.size.h; │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "viewBox", extentString); │ │ │ │ │ - this.translate(this.xOffset, 0); │ │ │ │ │ - return true │ │ │ │ │ - } else { │ │ │ │ │ - var inRange = this.translate(left - this.left + this.xOffset, top - this.top); │ │ │ │ │ - if (!inRange) { │ │ │ │ │ - this.setExtent(extent, true) │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((this.tileOrigin.lat - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getZoomForResolution(res); │ │ │ │ │ + var tileIndex = x + y * this.tierSizeInTiles[z].w + this.tileCountUpToTier[z]; │ │ │ │ │ + var path = "TileGroup" + Math.floor(tileIndex / 256) + "/" + z + "-" + x + "-" + y + ".jpg"; │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(path, url) │ │ │ │ │ + } │ │ │ │ │ + return url + path │ │ │ │ │ + }, │ │ │ │ │ + getImageSize: function() { │ │ │ │ │ + if (arguments.length > 0) { │ │ │ │ │ + var bounds = this.adjustBounds(arguments[0]); │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((this.tileOrigin.lat - bounds.top) / (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getZoomForResolution(res); │ │ │ │ │ + var w = this.standardTileSize; │ │ │ │ │ + var h = this.standardTileSize; │ │ │ │ │ + if (x == this.tierSizeInTiles[z].w - 1) { │ │ │ │ │ + var w = this.tierImageSize[z].w % this.standardTileSize │ │ │ │ │ } │ │ │ │ │ - return coordSysUnchanged && inRange │ │ │ │ │ + if (y == this.tierSizeInTiles[z].h - 1) { │ │ │ │ │ + var h = this.tierImageSize[z].h % this.standardTileSize │ │ │ │ │ + } │ │ │ │ │ + return new OpenLayers.Size(w, h) │ │ │ │ │ + } else { │ │ │ │ │ + return this.tileSize │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - translate: function(x, y) { │ │ │ │ │ - if (!this.inValidRange(x, y, true)) { │ │ │ │ │ - return false │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.map.maxExtent.left, this.map.maxExtent.top) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Zoomify" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Boxes = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ + drawMarker: function(marker) { │ │ │ │ │ + var topleft = this.map.getLayerPxFromLonLat({ │ │ │ │ │ + lon: marker.bounds.left, │ │ │ │ │ + lat: marker.bounds.top │ │ │ │ │ + }); │ │ │ │ │ + var botright = this.map.getLayerPxFromLonLat({ │ │ │ │ │ + lon: marker.bounds.right, │ │ │ │ │ + lat: marker.bounds.bottom │ │ │ │ │ + }); │ │ │ │ │ + if (botright == null || topleft == null) { │ │ │ │ │ + marker.display(false) │ │ │ │ │ } else { │ │ │ │ │ - var transformString = ""; │ │ │ │ │ - if (x || y) { │ │ │ │ │ - transformString = "translate(" + x + "," + y + ")" │ │ │ │ │ + var markerDiv = marker.draw(topleft, { │ │ │ │ │ + w: Math.max(1, botright.x - topleft.x), │ │ │ │ │ + h: Math.max(1, botright.y - topleft.y) │ │ │ │ │ + }); │ │ │ │ │ + if (!marker.drawn) { │ │ │ │ │ + this.div.appendChild(markerDiv); │ │ │ │ │ + marker.drawn = true │ │ │ │ │ } │ │ │ │ │ - this.root.setAttributeNS(null, "transform", transformString); │ │ │ │ │ - this.translationParameters = { │ │ │ │ │ - x: x, │ │ │ │ │ - y: y │ │ │ │ │ - }; │ │ │ │ │ - return true │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - setSize: function(size) { │ │ │ │ │ - OpenLayers.Renderer.prototype.setSize.apply(this, arguments); │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "width", this.size.w); │ │ │ │ │ - this.rendererRoot.setAttributeNS(null, "height", this.size.h) │ │ │ │ │ + removeMarker: function(marker) { │ │ │ │ │ + OpenLayers.Util.removeItem(this.markers, marker); │ │ │ │ │ + if (marker.div != null && marker.div.parentNode == this.div) { │ │ │ │ │ + this.div.removeChild(marker.div) │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - getNodeType: function(geometry, style) { │ │ │ │ │ - var nodeType = null; │ │ │ │ │ - switch (geometry.CLASS_NAME) { │ │ │ │ │ - case "OpenLayers.Geometry.Point": │ │ │ │ │ - if (style.externalGraphic) { │ │ │ │ │ - nodeType = "image" │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - nodeType = "svg" │ │ │ │ │ - } else { │ │ │ │ │ - nodeType = "circle" │ │ │ │ │ - } │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Rectangle": │ │ │ │ │ - nodeType = "rect"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LineString": │ │ │ │ │ - nodeType = "polyline"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.LinearRing": │ │ │ │ │ - nodeType = "polygon"; │ │ │ │ │ - break; │ │ │ │ │ - case "OpenLayers.Geometry.Polygon": │ │ │ │ │ - case "OpenLayers.Geometry.Curve": │ │ │ │ │ - nodeType = "path"; │ │ │ │ │ - break; │ │ │ │ │ - default: │ │ │ │ │ - break │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Boxes" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Bing = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ + key: null, │ │ │ │ │ + serverResolutions: [156543.03390625, 78271.516953125, 39135.7584765625, 19567.87923828125, 9783.939619140625, 4891.9698095703125, 2445.9849047851562, 1222.9924523925781, 611.4962261962891, 305.74811309814453, 152.87405654907226, 76.43702827453613, 38.218514137268066, 19.109257068634033, 9.554628534317017, 4.777314267158508, 2.388657133579254, 1.194328566789627, .5971642833948135, .29858214169740677, .14929107084870338, .07464553542435169], │ │ │ │ │ + attributionTemplate: '<span class="olBingAttribution ${type}">' + '<div><a target="_blank" href="http://www.bing.com/maps/">' + '<img src="${logo}" /></a></div>${copyrights}' + '<a style="white-space: nowrap" target="_blank" ' + 'href="http://www.microsoft.com/maps/product/terms.html">' + "Terms of Use</a></span>", │ │ │ │ │ + metadata: null, │ │ │ │ │ + protocolRegex: /^http:/i, │ │ │ │ │ + type: "Road", │ │ │ │ │ + culture: "en-US", │ │ │ │ │ + metadataParams: null, │ │ │ │ │ + tileOptions: null, │ │ │ │ │ + protocol: ~window.location.href.indexOf("http") ? "" : "http:", │ │ │ │ │ + initialize: function(options) { │ │ │ │ │ + options = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + sphericalMercator: true │ │ │ │ │ + }, options); │ │ │ │ │ + var name = options.name || "Bing " + (options.type || this.type); │ │ │ │ │ + var newArgs = [name, null, options]; │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, newArgs); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + crossOriginKeyword: "anonymous" │ │ │ │ │ + }, this.options.tileOptions); │ │ │ │ │ + this.loadMetadata() │ │ │ │ │ + }, │ │ │ │ │ + loadMetadata: function() { │ │ │ │ │ + this._callbackId = "_callback_" + this.id.replace(/\./g, "_"); │ │ │ │ │ + window[this._callbackId] = OpenLayers.Function.bind(OpenLayers.Layer.Bing.processMetadata, this); │ │ │ │ │ + var params = OpenLayers.Util.applyDefaults({ │ │ │ │ │ + key: this.key, │ │ │ │ │ + jsonp: this._callbackId, │ │ │ │ │ + include: "ImageryProviders" │ │ │ │ │ + }, this.metadataParams); │ │ │ │ │ + var url = this.protocol + "//dev.virtualearth.net/REST/v1/Imagery/Metadata/" + this.type + "?" + OpenLayers.Util.getParameterString(params); │ │ │ │ │ + var script = document.createElement("script"); │ │ │ │ │ + script.type = "text/javascript"; │ │ │ │ │ + script.src = url; │ │ │ │ │ + script.id = this._callbackId; │ │ │ │ │ + document.getElementsByTagName("head")[0].appendChild(script) │ │ │ │ │ + }, │ │ │ │ │ + initLayer: function() { │ │ │ │ │ + var res = this.metadata.resourceSets[0].resources[0]; │ │ │ │ │ + var url = res.imageUrl.replace("{quadkey}", "${quadkey}"); │ │ │ │ │ + url = url.replace("{culture}", this.culture); │ │ │ │ │ + url = url.replace(this.protocolRegex, this.protocol); │ │ │ │ │ + this.url = []; │ │ │ │ │ + for (var i = 0; i < res.imageUrlSubdomains.length; ++i) { │ │ │ │ │ + this.url.push(url.replace("{subdomain}", res.imageUrlSubdomains[i])) │ │ │ │ │ } │ │ │ │ │ - return nodeType │ │ │ │ │ + this.addOptions({ │ │ │ │ │ + maxResolution: Math.min(this.serverResolutions[res.zoomMin], this.maxResolution || Number.POSITIVE_INFINITY), │ │ │ │ │ + numZoomLevels: Math.min(res.zoomMax + 1 - res.zoomMin, this.numZoomLevels) │ │ │ │ │ + }, true); │ │ │ │ │ + if (!this.isBaseLayer) { │ │ │ │ │ + this.redraw() │ │ │ │ │ + } │ │ │ │ │ + this.updateAttribution() │ │ │ │ │ }, │ │ │ │ │ - setStyle: function(node, style, options) { │ │ │ │ │ - style = style || node._style; │ │ │ │ │ - options = options || node._options; │ │ │ │ │ - var title = style.title || style.graphicTitle; │ │ │ │ │ - if (title) { │ │ │ │ │ - node.setAttributeNS(null, "title", title); │ │ │ │ │ - var titleNode = node.getElementsByTagName("title"); │ │ │ │ │ - if (titleNode.length > 0) { │ │ │ │ │ - titleNode[0].firstChild.textContent = title │ │ │ │ │ - } else { │ │ │ │ │ - var label = this.nodeFactory(null, "title"); │ │ │ │ │ - label.textContent = title; │ │ │ │ │ - node.appendChild(label) │ │ │ │ │ - } │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + if (!this.url) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - var r = parseFloat(node.getAttributeNS(null, "r")); │ │ │ │ │ - var widthFactor = 1; │ │ │ │ │ - var pos; │ │ │ │ │ - if (node._geometryClass == "OpenLayers.Geometry.Point" && r) { │ │ │ │ │ - node.style.visibility = ""; │ │ │ │ │ - if (style.graphic === false) { │ │ │ │ │ - node.style.visibility = "hidden" │ │ │ │ │ - } else if (style.externalGraphic) { │ │ │ │ │ - pos = this.getPosition(node); │ │ │ │ │ - if (style.graphicWidth && style.graphicHeight) { │ │ │ │ │ - node.setAttributeNS(null, "preserveAspectRatio", "none") │ │ │ │ │ - } │ │ │ │ │ - var width = style.graphicWidth || style.graphicHeight; │ │ │ │ │ - var height = style.graphicHeight || style.graphicWidth; │ │ │ │ │ - width = width ? width : style.pointRadius * 2; │ │ │ │ │ - height = height ? height : style.pointRadius * 2; │ │ │ │ │ - var xOffset = style.graphicXOffset != undefined ? style.graphicXOffset : -(.5 * width); │ │ │ │ │ - var yOffset = style.graphicYOffset != undefined ? style.graphicYOffset : -(.5 * height); │ │ │ │ │ - var opacity = style.graphicOpacity || style.fillOpacity; │ │ │ │ │ - node.setAttributeNS(null, "x", (pos.x + xOffset).toFixed()); │ │ │ │ │ - node.setAttributeNS(null, "y", (pos.y + yOffset).toFixed()); │ │ │ │ │ - node.setAttributeNS(null, "width", width); │ │ │ │ │ - node.setAttributeNS(null, "height", height); │ │ │ │ │ - node.setAttributeNS(this.xlinkns, "xlink:href", style.externalGraphic); │ │ │ │ │ - node.setAttributeNS(null, "style", "opacity: " + opacity); │ │ │ │ │ - node.onclick = OpenLayers.Event.preventDefault │ │ │ │ │ - } else if (this.isComplexSymbol(style.graphicName)) { │ │ │ │ │ - var offset = style.pointRadius * 3; │ │ │ │ │ - var size = offset * 2; │ │ │ │ │ - var src = this.importSymbol(style.graphicName); │ │ │ │ │ - pos = this.getPosition(node); │ │ │ │ │ - widthFactor = this.symbolMetrics[src.id][0] * 3 / size; │ │ │ │ │ - var parent = node.parentNode; │ │ │ │ │ - var nextSibling = node.nextSibling; │ │ │ │ │ - if (parent) { │ │ │ │ │ - parent.removeChild(node) │ │ │ │ │ - } │ │ │ │ │ - node.firstChild && node.removeChild(node.firstChild); │ │ │ │ │ - node.appendChild(src.firstChild.cloneNode(true)); │ │ │ │ │ - node.setAttributeNS(null, "viewBox", src.getAttributeNS(null, "viewBox")); │ │ │ │ │ - node.setAttributeNS(null, "width", size); │ │ │ │ │ - node.setAttributeNS(null, "height", size); │ │ │ │ │ - node.setAttributeNS(null, "x", pos.x - offset); │ │ │ │ │ - node.setAttributeNS(null, "y", pos.y - offset); │ │ │ │ │ - if (nextSibling) { │ │ │ │ │ - parent.insertBefore(node, nextSibling) │ │ │ │ │ - } else if (parent) { │ │ │ │ │ - parent.appendChild(node) │ │ │ │ │ - } │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "r", style.pointRadius) │ │ │ │ │ + var xyz = this.getXYZ(bounds), │ │ │ │ │ + x = xyz.x, │ │ │ │ │ + y = xyz.y, │ │ │ │ │ + z = xyz.z; │ │ │ │ │ + var quadDigits = []; │ │ │ │ │ + for (var i = z; i > 0; --i) { │ │ │ │ │ + var digit = "0"; │ │ │ │ │ + var mask = 1 << i - 1; │ │ │ │ │ + if ((x & mask) != 0) { │ │ │ │ │ + digit++ │ │ │ │ │ } │ │ │ │ │ - var rotation = style.rotation; │ │ │ │ │ - if ((rotation !== undefined || node._rotation !== undefined) && pos) { │ │ │ │ │ - node._rotation = rotation; │ │ │ │ │ - rotation |= 0; │ │ │ │ │ - if (node.nodeName !== "svg") { │ │ │ │ │ - node.setAttributeNS(null, "transform", "rotate(" + rotation + " " + pos.x + " " + pos.y + ")") │ │ │ │ │ - } else { │ │ │ │ │ - var metrics = this.symbolMetrics[src.id]; │ │ │ │ │ - node.firstChild.setAttributeNS(null, "transform", "rotate(" + rotation + " " + metrics[1] + " " + metrics[2] + ")") │ │ │ │ │ - } │ │ │ │ │ + if ((y & mask) != 0) { │ │ │ │ │ + digit++; │ │ │ │ │ + digit++ │ │ │ │ │ } │ │ │ │ │ + quadDigits.push(digit) │ │ │ │ │ } │ │ │ │ │ - if (options.isFilled) { │ │ │ │ │ - node.setAttributeNS(null, "fill", style.fillColor); │ │ │ │ │ - node.setAttributeNS(null, "fill-opacity", style.fillOpacity) │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "fill", "none") │ │ │ │ │ - } │ │ │ │ │ - if (options.isStroked) { │ │ │ │ │ - node.setAttributeNS(null, "stroke", style.strokeColor); │ │ │ │ │ - node.setAttributeNS(null, "stroke-opacity", style.strokeOpacity); │ │ │ │ │ - node.setAttributeNS(null, "stroke-width", style.strokeWidth * widthFactor); │ │ │ │ │ - node.setAttributeNS(null, "stroke-linecap", style.strokeLinecap || "round"); │ │ │ │ │ - node.setAttributeNS(null, "stroke-linejoin", "round"); │ │ │ │ │ - style.strokeDashstyle && node.setAttributeNS(null, "stroke-dasharray", this.dashStyle(style, widthFactor)) │ │ │ │ │ - } else { │ │ │ │ │ - node.setAttributeNS(null, "stroke", "none") │ │ │ │ │ - } │ │ │ │ │ - if (style.pointerEvents) { │ │ │ │ │ - node.setAttributeNS(null, "pointer-events", style.pointerEvents) │ │ │ │ │ + var quadKey = quadDigits.join(""); │ │ │ │ │ + var url = this.selectUrl("" + x + y + z, this.url); │ │ │ │ │ + return OpenLayers.String.format(url, { │ │ │ │ │ + quadkey: quadKey │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + updateAttribution: function() { │ │ │ │ │ + var metadata = this.metadata; │ │ │ │ │ + if (!metadata.resourceSets || !this.map || !this.map.center) { │ │ │ │ │ + return │ │ │ │ │ } │ │ │ │ │ - if (style.cursor != null) { │ │ │ │ │ - node.setAttributeNS(null, "cursor", style.cursor) │ │ │ │ │ + var res = metadata.resourceSets[0].resources[0]; │ │ │ │ │ + var extent = this.map.getExtent().transform(this.map.getProjectionObject(), new OpenLayers.Projection("EPSG:4326")); │ │ │ │ │ + var providers = res.imageryProviders || [], │ │ │ │ │ + zoom = OpenLayers.Util.indexOf(this.serverResolutions, this.getServerResolution()), │ │ │ │ │ + copyrights = "", │ │ │ │ │ + provider, i, ii, j, jj, bbox, coverage; │ │ │ │ │ + for (i = 0, ii = providers.length; i < ii; ++i) { │ │ │ │ │ + provider = providers[i]; │ │ │ │ │ + for (j = 0, jj = provider.coverageAreas.length; j < jj; ++j) { │ │ │ │ │ + coverage = provider.coverageAreas[j]; │ │ │ │ │ + bbox = OpenLayers.Bounds.fromArray(coverage.bbox, true); │ │ │ │ │ + if (extent.intersectsBounds(bbox) && zoom <= coverage.zoomMax && zoom >= coverage.zoomMin) { │ │ │ │ │ + copyrights += provider.attribution + " " │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ + var logo = metadata.brandLogoUri.replace(this.protocolRegex, this.protocol); │ │ │ │ │ + this.attribution = OpenLayers.String.format(this.attributionTemplate, { │ │ │ │ │ + type: this.type.toLowerCase(), │ │ │ │ │ + logo: logo, │ │ │ │ │ + copyrights: copyrights │ │ │ │ │ + }); │ │ │ │ │ + this.map && this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "attribution" │ │ │ │ │ + }) │ │ │ │ │ }, │ │ │ │ │ - dashStyle: function(style, widthFactor) { │ │ │ │ │ - var w = style.strokeWidth * widthFactor; │ │ │ │ │ - var str = style.strokeDashstyle; │ │ │ │ │ - switch (str) { │ │ │ │ │ - case "solid": │ │ │ │ │ - return "none"; │ │ │ │ │ - case "dot": │ │ │ │ │ - return [1, 4 * w].join(); │ │ │ │ │ - case "dash": │ │ │ │ │ - return [4 * w, 4 * w].join(); │ │ │ │ │ - case "dashdot": │ │ │ │ │ - return [4 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ - case "longdash": │ │ │ │ │ - return [8 * w, 4 * w].join(); │ │ │ │ │ - case "longdashdot": │ │ │ │ │ - return [8 * w, 4 * w, 1, 4 * w].join(); │ │ │ │ │ - default: │ │ │ │ │ - return OpenLayers.String.trim(str).replace(/\s+/g, ",") │ │ │ │ │ - } │ │ │ │ │ + setMap: function() { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.setMap.apply(this, arguments); │ │ │ │ │ + this.map.events.register("moveend", this, this.updateAttribution) │ │ │ │ │ }, │ │ │ │ │ - createNode: function(type, id) { │ │ │ │ │ - var node = document.createElementNS(this.xmlns, type); │ │ │ │ │ - if (id) { │ │ │ │ │ - node.setAttributeNS(null, "id", id) │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Bing(this.options) │ │ │ │ │ } │ │ │ │ │ - return node │ │ │ │ │ - }, │ │ │ │ │ - nodeTypeCompare: function(node, type) { │ │ │ │ │ - return type == node.nodeName │ │ │ │ │ + obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - createRenderRoot: function() { │ │ │ │ │ - var svg = this.nodeFactory(this.container.id + "_svgRoot", "svg"); │ │ │ │ │ - svg.style.display = "block"; │ │ │ │ │ - return svg │ │ │ │ │ + destroy: function() { │ │ │ │ │ + this.map && this.map.events.unregister("moveend", this, this.updateAttribution); │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - createRoot: function(suffix) { │ │ │ │ │ - return this.nodeFactory(this.container.id + suffix, "g") │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Bing" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Bing.processMetadata = function(metadata) { │ │ │ │ │ + this.metadata = metadata; │ │ │ │ │ + this.initLayer(); │ │ │ │ │ + var script = document.getElementById(this._callbackId); │ │ │ │ │ + script.parentNode.removeChild(script); │ │ │ │ │ + window[this._callbackId] = undefined; │ │ │ │ │ + delete this._callbackId │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Layer.MapGuide = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + useHttpTile: false, │ │ │ │ │ + singleTile: false, │ │ │ │ │ + useOverlay: false, │ │ │ │ │ + useAsyncOverlay: true, │ │ │ │ │ + TILE_PARAMS: { │ │ │ │ │ + operation: "GETTILEIMAGE", │ │ │ │ │ + version: "1.2.0" │ │ │ │ │ }, │ │ │ │ │ - createDefs: function() { │ │ │ │ │ - var defs = this.nodeFactory(this.container.id + "_defs", "defs"); │ │ │ │ │ - this.rendererRoot.appendChild(defs); │ │ │ │ │ - return defs │ │ │ │ │ + SINGLE_TILE_PARAMS: { │ │ │ │ │ + operation: "GETMAPIMAGE", │ │ │ │ │ + format: "PNG", │ │ │ │ │ + locale: "en", │ │ │ │ │ + clip: "1", │ │ │ │ │ + version: "1.0.0" │ │ │ │ │ }, │ │ │ │ │ - drawPoint: function(node, geometry) { │ │ │ │ │ - return this.drawCircle(node, geometry, 1) │ │ │ │ │ + OVERLAY_PARAMS: { │ │ │ │ │ + operation: "GETDYNAMICMAPOVERLAYIMAGE", │ │ │ │ │ + format: "PNG", │ │ │ │ │ + locale: "en", │ │ │ │ │ + clip: "1", │ │ │ │ │ + version: "2.0.0" │ │ │ │ │ }, │ │ │ │ │ - drawCircle: function(node, geometry, radius) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = (geometry.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y = this.top - geometry.y / resolution; │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - node.setAttributeNS(null, "cx", x); │ │ │ │ │ - node.setAttributeNS(null, "cy", y); │ │ │ │ │ - node.setAttributeNS(null, "r", radius); │ │ │ │ │ - return node │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ - } │ │ │ │ │ + FOLDER_PARAMS: { │ │ │ │ │ + tileColumnsPerFolder: 30, │ │ │ │ │ + tileRowsPerFolder: 30, │ │ │ │ │ + format: "png", │ │ │ │ │ + querystring: null │ │ │ │ │ }, │ │ │ │ │ - drawLineString: function(node, geometry) { │ │ │ │ │ - var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ - if (componentsResult.path) { │ │ │ │ │ - node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ - return componentsResult.complete ? node : null │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + defaultSize: new OpenLayers.Size(300, 300), │ │ │ │ │ + tileOriginCorner: "tl", │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (options == null || options.isBaseLayer == null) { │ │ │ │ │ + this.isBaseLayer = this.transparent != "true" && this.transparent != true │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - drawLinearRing: function(node, geometry) { │ │ │ │ │ - var componentsResult = this.getComponentsString(geometry.components); │ │ │ │ │ - if (componentsResult.path) { │ │ │ │ │ - node.setAttributeNS(null, "points", componentsResult.path); │ │ │ │ │ - return componentsResult.complete ? node : null │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + if (options && options.useOverlay != null) { │ │ │ │ │ + this.useOverlay = options.useOverlay │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - drawPolygon: function(node, geometry) { │ │ │ │ │ - var d = ""; │ │ │ │ │ - var draw = true; │ │ │ │ │ - var complete = true; │ │ │ │ │ - var linearRingResult, path; │ │ │ │ │ - for (var j = 0, len = geometry.components.length; j < len; j++) { │ │ │ │ │ - d += " M"; │ │ │ │ │ - linearRingResult = this.getComponentsString(geometry.components[j].components, " "); │ │ │ │ │ - path = linearRingResult.path; │ │ │ │ │ - if (path) { │ │ │ │ │ - d += " " + path; │ │ │ │ │ - complete = linearRingResult.complete && complete │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + if (this.useOverlay) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(this.params, this.OVERLAY_PARAMS); │ │ │ │ │ + if (!this.useAsyncOverlay) { │ │ │ │ │ + this.params.version = "1.0.0" │ │ │ │ │ + } │ │ │ │ │ } else { │ │ │ │ │ - draw = false │ │ │ │ │ + OpenLayers.Util.applyDefaults(this.params, this.SINGLE_TILE_PARAMS) │ │ │ │ │ } │ │ │ │ │ - } │ │ │ │ │ - d += " z"; │ │ │ │ │ - if (draw) { │ │ │ │ │ - node.setAttributeNS(null, "d", d); │ │ │ │ │ - node.setAttributeNS(null, "fill-rule", "evenodd"); │ │ │ │ │ - return complete ? node : null │ │ │ │ │ } else { │ │ │ │ │ - return false │ │ │ │ │ + if (this.useHttpTile) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(this.params, this.FOLDER_PARAMS) │ │ │ │ │ + } else { │ │ │ │ │ + OpenLayers.Util.applyDefaults(this.params, this.TILE_PARAMS) │ │ │ │ │ + } │ │ │ │ │ + this.setTileSize(this.defaultSize) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - drawRectangle: function(node, geometry) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = (geometry.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y = this.top - geometry.y / resolution; │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - node.setAttributeNS(null, "x", x); │ │ │ │ │ - node.setAttributeNS(null, "y", y); │ │ │ │ │ - node.setAttributeNS(null, "width", geometry.width / resolution); │ │ │ │ │ - node.setAttributeNS(null, "height", geometry.height / resolution); │ │ │ │ │ - return node │ │ │ │ │ - } else { │ │ │ │ │ - return false │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.MapGuide(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - drawText: function(featureId, style, location) { │ │ │ │ │ - var drawOutline = !!style.labelOutlineWidth; │ │ │ │ │ - if (drawOutline) { │ │ │ │ │ - var outlineStyle = OpenLayers.Util.extend({}, style); │ │ │ │ │ - outlineStyle.fontColor = outlineStyle.labelOutlineColor; │ │ │ │ │ - outlineStyle.fontStrokeColor = outlineStyle.labelOutlineColor; │ │ │ │ │ - outlineStyle.fontStrokeWidth = style.labelOutlineWidth; │ │ │ │ │ - if (style.labelOutlineOpacity) { │ │ │ │ │ - outlineStyle.fontOpacity = style.labelOutlineOpacity │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + var url; │ │ │ │ │ + var center = bounds.getCenterLonLat(); │ │ │ │ │ + var mapSize = this.map.getSize(); │ │ │ │ │ + if (this.singleTile) { │ │ │ │ │ + var params = { │ │ │ │ │ + setdisplaydpi: OpenLayers.DOTS_PER_INCH, │ │ │ │ │ + setdisplayheight: mapSize.h * this.ratio, │ │ │ │ │ + setdisplaywidth: mapSize.w * this.ratio, │ │ │ │ │ + setviewcenterx: center.lon, │ │ │ │ │ + setviewcentery: center.lat, │ │ │ │ │ + setviewscale: this.map.getScale() │ │ │ │ │ + }; │ │ │ │ │ + if (this.useOverlay && !this.useAsyncOverlay) { │ │ │ │ │ + var getVisParams = {}; │ │ │ │ │ + getVisParams = OpenLayers.Util.extend(getVisParams, params); │ │ │ │ │ + getVisParams.operation = "GETVISIBLEMAPEXTENT"; │ │ │ │ │ + getVisParams.version = "1.0.0"; │ │ │ │ │ + getVisParams.session = this.params.session; │ │ │ │ │ + getVisParams.mapName = this.params.mapName; │ │ │ │ │ + getVisParams.format = "text/xml"; │ │ │ │ │ + url = this.getFullRequestString(getVisParams); │ │ │ │ │ + OpenLayers.Request.GET({ │ │ │ │ │ + url: url, │ │ │ │ │ + async: false │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + url = this.getFullRequestString(params) │ │ │ │ │ + } else { │ │ │ │ │ + var currentRes = this.map.getResolution(); │ │ │ │ │ + var colidx = Math.floor((bounds.left - this.maxExtent.left) / currentRes); │ │ │ │ │ + colidx = Math.round(colidx / this.tileSize.w); │ │ │ │ │ + var rowidx = Math.floor((this.maxExtent.top - bounds.top) / currentRes); │ │ │ │ │ + rowidx = Math.round(rowidx / this.tileSize.h); │ │ │ │ │ + if (this.useHttpTile) { │ │ │ │ │ + url = this.getImageFilePath({ │ │ │ │ │ + tilecol: colidx, │ │ │ │ │ + tilerow: rowidx, │ │ │ │ │ + scaleindex: this.resolutions.length - this.map.zoom - 1 │ │ │ │ │ + }) │ │ │ │ │ + } else { │ │ │ │ │ + url = this.getFullRequestString({ │ │ │ │ │ + tilecol: colidx, │ │ │ │ │ + tilerow: rowidx, │ │ │ │ │ + scaleindex: this.resolutions.length - this.map.zoom - 1 │ │ │ │ │ + }) │ │ │ │ │ } │ │ │ │ │ - delete outlineStyle.labelOutlineWidth; │ │ │ │ │ - this.drawText(featureId, outlineStyle, location) │ │ │ │ │ - } │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = (location.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y = location.y / resolution - this.top; │ │ │ │ │ - var suffix = drawOutline ? this.LABEL_OUTLINE_SUFFIX : this.LABEL_ID_SUFFIX; │ │ │ │ │ - var label = this.nodeFactory(featureId + suffix, "text"); │ │ │ │ │ - label.setAttributeNS(null, "x", x); │ │ │ │ │ - label.setAttributeNS(null, "y", -y); │ │ │ │ │ - if (style.fontColor) { │ │ │ │ │ - label.setAttributeNS(null, "fill", style.fontColor) │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStrokeColor) { │ │ │ │ │ - label.setAttributeNS(null, "stroke", style.fontStrokeColor) │ │ │ │ │ - } │ │ │ │ │ - if (style.fontStrokeWidth) { │ │ │ │ │ - label.setAttributeNS(null, "stroke-width", style.fontStrokeWidth) │ │ │ │ │ } │ │ │ │ │ - if (style.fontOpacity) { │ │ │ │ │ - label.setAttributeNS(null, "opacity", style.fontOpacity) │ │ │ │ │ + return url │ │ │ │ │ + }, │ │ │ │ │ + getFullRequestString: function(newParams, altUrl) { │ │ │ │ │ + var url = altUrl == null ? this.url : altUrl; │ │ │ │ │ + if (typeof url == "object") { │ │ │ │ │ + url = url[Math.floor(Math.random() * url.length)] │ │ │ │ │ } │ │ │ │ │ - if (style.fontFamily) { │ │ │ │ │ - label.setAttributeNS(null, "font-family", style.fontFamily) │ │ │ │ │ + var requestString = url; │ │ │ │ │ + var allParams = OpenLayers.Util.extend({}, this.params); │ │ │ │ │ + allParams = OpenLayers.Util.extend(allParams, newParams); │ │ │ │ │ + var urlParams = OpenLayers.Util.upperCaseObject(OpenLayers.Util.getParameters(url)); │ │ │ │ │ + for (var key in allParams) { │ │ │ │ │ + if (key.toUpperCase() in urlParams) { │ │ │ │ │ + delete allParams[key] │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (style.fontSize) { │ │ │ │ │ - label.setAttributeNS(null, "font-size", style.fontSize) │ │ │ │ │ + var paramsString = OpenLayers.Util.getParameterString(allParams); │ │ │ │ │ + paramsString = paramsString.replace(/,/g, "+"); │ │ │ │ │ + if (paramsString != "") { │ │ │ │ │ + var lastServerChar = url.charAt(url.length - 1); │ │ │ │ │ + if (lastServerChar == "&" || lastServerChar == "?") { │ │ │ │ │ + requestString += paramsString │ │ │ │ │ + } else { │ │ │ │ │ + if (url.indexOf("?") == -1) { │ │ │ │ │ + requestString += "?" + paramsString │ │ │ │ │ + } else { │ │ │ │ │ + requestString += "&" + paramsString │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - if (style.fontWeight) { │ │ │ │ │ - label.setAttributeNS(null, "font-weight", style.fontWeight) │ │ │ │ │ + return requestString │ │ │ │ │ + }, │ │ │ │ │ + getImageFilePath: function(newParams, altUrl) { │ │ │ │ │ + var url = altUrl == null ? this.url : altUrl; │ │ │ │ │ + if (typeof url == "object") { │ │ │ │ │ + url = url[Math.floor(Math.random() * url.length)] │ │ │ │ │ } │ │ │ │ │ - if (style.fontStyle) { │ │ │ │ │ - label.setAttributeNS(null, "font-style", style.fontStyle) │ │ │ │ │ + var requestString = url; │ │ │ │ │ + var tileRowGroup = ""; │ │ │ │ │ + var tileColGroup = ""; │ │ │ │ │ + if (newParams.tilerow < 0) { │ │ │ │ │ + tileRowGroup = "-" │ │ │ │ │ } │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - label.setAttributeNS(null, "pointer-events", "visible"); │ │ │ │ │ - label._featureId = featureId │ │ │ │ │ + if (newParams.tilerow == 0) { │ │ │ │ │ + tileRowGroup += "0" │ │ │ │ │ } else { │ │ │ │ │ - label.setAttributeNS(null, "pointer-events", "none") │ │ │ │ │ - } │ │ │ │ │ - var align = style.labelAlign || OpenLayers.Renderer.defaultSymbolizer.labelAlign; │ │ │ │ │ - label.setAttributeNS(null, "text-anchor", OpenLayers.Renderer.SVG.LABEL_ALIGN[align[0]] || "middle"); │ │ │ │ │ - if (OpenLayers.IS_GECKO === true) { │ │ │ │ │ - label.setAttributeNS(null, "dominant-baseline", OpenLayers.Renderer.SVG.LABEL_ALIGN[align[1]] || "central") │ │ │ │ │ + tileRowGroup += Math.floor(Math.abs(newParams.tilerow / this.params.tileRowsPerFolder)) * this.params.tileRowsPerFolder │ │ │ │ │ } │ │ │ │ │ - var labelRows = style.label.split("\n"); │ │ │ │ │ - var numRows = labelRows.length; │ │ │ │ │ - while (label.childNodes.length > numRows) { │ │ │ │ │ - label.removeChild(label.lastChild) │ │ │ │ │ + if (newParams.tilecol < 0) { │ │ │ │ │ + tileColGroup = "-" │ │ │ │ │ } │ │ │ │ │ - for (var i = 0; i < numRows; i++) { │ │ │ │ │ - var tspan = this.nodeFactory(featureId + suffix + "_tspan_" + i, "tspan"); │ │ │ │ │ - if (style.labelSelect === true) { │ │ │ │ │ - tspan._featureId = featureId; │ │ │ │ │ - tspan._geometry = location; │ │ │ │ │ - tspan._geometryClass = location.CLASS_NAME │ │ │ │ │ - } │ │ │ │ │ - if (OpenLayers.IS_GECKO === false) { │ │ │ │ │ - tspan.setAttributeNS(null, "baseline-shift", OpenLayers.Renderer.SVG.LABEL_VSHIFT[align[1]] || "-35%") │ │ │ │ │ - } │ │ │ │ │ - tspan.setAttribute("x", x); │ │ │ │ │ - if (i == 0) { │ │ │ │ │ - var vfactor = OpenLayers.Renderer.SVG.LABEL_VFACTOR[align[1]]; │ │ │ │ │ - if (vfactor == null) { │ │ │ │ │ - vfactor = -.5 │ │ │ │ │ - } │ │ │ │ │ - tspan.setAttribute("dy", vfactor * (numRows - 1) + "em") │ │ │ │ │ - } else { │ │ │ │ │ - tspan.setAttribute("dy", "1em") │ │ │ │ │ - } │ │ │ │ │ - tspan.textContent = labelRows[i] === "" ? " " : labelRows[i]; │ │ │ │ │ - if (!tspan.parentNode) { │ │ │ │ │ - label.appendChild(tspan) │ │ │ │ │ - } │ │ │ │ │ + if (newParams.tilecol == 0) { │ │ │ │ │ + tileColGroup += "0" │ │ │ │ │ + } else { │ │ │ │ │ + tileColGroup += Math.floor(Math.abs(newParams.tilecol / this.params.tileColumnsPerFolder)) * this.params.tileColumnsPerFolder │ │ │ │ │ } │ │ │ │ │ - if (!label.parentNode) { │ │ │ │ │ - this.textRoot.appendChild(label) │ │ │ │ │ + var tilePath = "/S" + Math.floor(newParams.scaleindex) + "/" + this.params.basemaplayergroupname + "/R" + tileRowGroup + "/C" + tileColGroup + "/" + newParams.tilerow % this.params.tileRowsPerFolder + "_" + newParams.tilecol % this.params.tileColumnsPerFolder + "." + this.params.format; │ │ │ │ │ + if (this.params.querystring) { │ │ │ │ │ + tilePath += "?" + this.params.querystring │ │ │ │ │ } │ │ │ │ │ + requestString += tilePath; │ │ │ │ │ + return requestString │ │ │ │ │ }, │ │ │ │ │ - getComponentsString: function(components, separator) { │ │ │ │ │ - var renderCmp = []; │ │ │ │ │ - var complete = true; │ │ │ │ │ - var len = components.length; │ │ │ │ │ - var strings = []; │ │ │ │ │ - var str, component; │ │ │ │ │ - for (var i = 0; i < len; i++) { │ │ │ │ │ - component = components[i]; │ │ │ │ │ - renderCmp.push(component); │ │ │ │ │ - str = this.getShortString(component); │ │ │ │ │ - if (str) { │ │ │ │ │ - strings.push(str) │ │ │ │ │ - } else { │ │ │ │ │ - if (i > 0) { │ │ │ │ │ - if (this.getShortString(components[i - 1])) { │ │ │ │ │ - strings.push(this.clipLine(components[i], components[i - 1])) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - if (i < len - 1) { │ │ │ │ │ - if (this.getShortString(components[i + 1])) { │ │ │ │ │ - strings.push(this.clipLine(components[i], components[i + 1])) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - complete = false │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return { │ │ │ │ │ - path: strings.join(separator || ","), │ │ │ │ │ - complete: complete │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.MapGuide" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.TMS = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + serviceVersion: "1.0.0", │ │ │ │ │ + layername: null, │ │ │ │ │ + type: null, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + tileOrigin: null, │ │ │ │ │ + serverResolutions: null, │ │ │ │ │ + zoomOffset: 0, │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + newArguments.push(name, url, {}, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.TMS(this.name, this.url, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - clipLine: function(badComponent, goodComponent) { │ │ │ │ │ - if (goodComponent.equals(badComponent)) { │ │ │ │ │ - return "" │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var res = this.getServerResolution(); │ │ │ │ │ + var x = Math.round((bounds.left - this.tileOrigin.lon) / (res * this.tileSize.w)); │ │ │ │ │ + var y = Math.round((bounds.bottom - this.tileOrigin.lat) / (res * this.tileSize.h)); │ │ │ │ │ + var z = this.getServerZoom(); │ │ │ │ │ + var path = this.serviceVersion + "/" + this.layername + "/" + z + "/" + x + "/" + y + "." + this.type; │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(path, url) │ │ │ │ │ } │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var maxX = this.MAX_PIXEL - this.translationParameters.x; │ │ │ │ │ - var maxY = this.MAX_PIXEL - this.translationParameters.y; │ │ │ │ │ - var x1 = (goodComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y1 = this.top - goodComponent.y / resolution; │ │ │ │ │ - var x2 = (badComponent.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y2 = this.top - badComponent.y / resolution; │ │ │ │ │ - var k; │ │ │ │ │ - if (x2 < -maxX || x2 > maxX) { │ │ │ │ │ - k = (y2 - y1) / (x2 - x1); │ │ │ │ │ - x2 = x2 < 0 ? -maxX : maxX; │ │ │ │ │ - y2 = y1 + (x2 - x1) * k │ │ │ │ │ + return url + path │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.setMap.apply(this, arguments); │ │ │ │ │ + if (!this.tileOrigin) { │ │ │ │ │ + this.tileOrigin = new OpenLayers.LonLat(this.map.maxExtent.left, this.map.maxExtent.bottom) │ │ │ │ │ } │ │ │ │ │ - if (y2 < -maxY || y2 > maxY) { │ │ │ │ │ - k = (x2 - x1) / (y2 - y1); │ │ │ │ │ - y2 = y2 < 0 ? -maxY : maxY; │ │ │ │ │ - x2 = x1 + (y2 - y1) * k │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.TMS" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.OSM = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ + name: "OpenStreetMap", │ │ │ │ │ + url: ["http://a.tile.openstreetmap.org/${z}/${x}/${y}.png", "http://b.tile.openstreetmap.org/${z}/${x}/${y}.png", "http://c.tile.openstreetmap.org/${z}/${x}/${y}.png"], │ │ │ │ │ + attribution: "© <a href='http://www.openstreetmap.org/copyright'>OpenStreetMap</a> contributors", │ │ │ │ │ + sphericalMercator: true, │ │ │ │ │ + wrapDateLine: true, │ │ │ │ │ + tileOptions: null, │ │ │ │ │ + initialize: function(name, url, options) { │ │ │ │ │ + OpenLayers.Layer.XYZ.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + crossOriginKeyword: "anonymous" │ │ │ │ │ + }, this.options && this.options.tileOptions) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.OSM(this.name, this.url, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - return x2 + "," + y2 │ │ │ │ │ + obj = OpenLayers.Layer.XYZ.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - getShortString: function(point) { │ │ │ │ │ - var resolution = this.getResolution(); │ │ │ │ │ - var x = (point.x - this.featureDx) / resolution + this.left; │ │ │ │ │ - var y = this.top - point.y / resolution; │ │ │ │ │ - if (this.inValidRange(x, y)) { │ │ │ │ │ - return x + "," + y │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.OSM" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.WorldWind = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + DEFAULT_PARAMS: {}, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + lzd: null, │ │ │ │ │ + zoomLevels: null, │ │ │ │ │ + initialize: function(name, url, lzd, zoomLevels, params, options) { │ │ │ │ │ + this.lzd = lzd; │ │ │ │ │ + this.zoomLevels = zoomLevels; │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + newArguments.push(name, url, params, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + this.params = OpenLayers.Util.applyDefaults(this.params, this.DEFAULT_PARAMS) │ │ │ │ │ + }, │ │ │ │ │ + getZoom: function() { │ │ │ │ │ + var zoom = this.map.getZoom(); │ │ │ │ │ + var extent = this.map.getMaxExtent(); │ │ │ │ │ + zoom = zoom - Math.log(this.maxResolution / (this.lzd / 512)) / Math.log(2); │ │ │ │ │ + return zoom │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var zoom = this.getZoom(); │ │ │ │ │ + var extent = this.map.getMaxExtent(); │ │ │ │ │ + var deg = this.lzd / Math.pow(2, this.getZoom()); │ │ │ │ │ + var x = Math.floor((bounds.left - extent.left) / deg); │ │ │ │ │ + var y = Math.floor((bounds.bottom - extent.bottom) / deg); │ │ │ │ │ + if (this.map.getResolution() <= this.lzd / 512 && this.getZoom() <= this.zoomLevels) { │ │ │ │ │ + return this.getFullRequestString({ │ │ │ │ │ + L: zoom, │ │ │ │ │ + X: x, │ │ │ │ │ + Y: y │ │ │ │ │ + }) │ │ │ │ │ } else { │ │ │ │ │ - return false │ │ │ │ │ + return OpenLayers.Util.getImageLocation("blank.gif") │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - getPosition: function(node) { │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.WorldWind" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.KaMap = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + i: "jpeg", │ │ │ │ │ + map: "" │ │ │ │ │ + }, │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.params = OpenLayers.Util.applyDefaults(this.params, this.DEFAULT_PARAMS) │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var mapRes = this.map.getResolution(); │ │ │ │ │ + var scale = Math.round(this.map.getScale() * 1e4) / 1e4; │ │ │ │ │ + var pX = Math.round(bounds.left / mapRes); │ │ │ │ │ + var pY = -Math.round(bounds.top / mapRes); │ │ │ │ │ + return this.getFullRequestString({ │ │ │ │ │ + t: pY, │ │ │ │ │ + l: pX, │ │ │ │ │ + s: scale │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + calculateGridLayout: function(bounds, origin, resolution) { │ │ │ │ │ + var tilelon = resolution * this.tileSize.w; │ │ │ │ │ + var tilelat = resolution * this.tileSize.h; │ │ │ │ │ + var offsetlon = bounds.left; │ │ │ │ │ + var tilecol = Math.floor(offsetlon / tilelon) - this.buffer; │ │ │ │ │ + var offsetlat = bounds.top; │ │ │ │ │ + var tilerow = Math.floor(offsetlat / tilelat) + this.buffer; │ │ │ │ │ return { │ │ │ │ │ - x: parseFloat(node.getAttributeNS(null, "cx")), │ │ │ │ │ - y: parseFloat(node.getAttributeNS(null, "cy")) │ │ │ │ │ + tilelon: tilelon, │ │ │ │ │ + tilelat: tilelat, │ │ │ │ │ + startcol: tilecol, │ │ │ │ │ + startrow: tilerow │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - importSymbol: function(graphicName) { │ │ │ │ │ - if (!this.defs) { │ │ │ │ │ - this.defs = this.createDefs() │ │ │ │ │ - } │ │ │ │ │ - var id = this.container.id + "-" + graphicName; │ │ │ │ │ - var existing = document.getElementById(id); │ │ │ │ │ - if (existing != null) { │ │ │ │ │ - return existing │ │ │ │ │ - } │ │ │ │ │ - var symbol = OpenLayers.Renderer.symbol[graphicName]; │ │ │ │ │ - if (!symbol) { │ │ │ │ │ - throw new Error(graphicName + " is not a valid symbol name") │ │ │ │ │ + getTileBoundsForGridIndex: function(row, col) { │ │ │ │ │ + var origin = this.getTileOrigin(); │ │ │ │ │ + var tileLayout = this.gridLayout; │ │ │ │ │ + var tilelon = tileLayout.tilelon; │ │ │ │ │ + var tilelat = tileLayout.tilelat; │ │ │ │ │ + var minX = (tileLayout.startcol + col) * tilelon; │ │ │ │ │ + var minY = (tileLayout.startrow - row) * tilelat; │ │ │ │ │ + return new OpenLayers.Bounds(minX, minY, minX + tilelon, minY + tilelat) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.KaMap(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - var symbolNode = this.nodeFactory(id, "symbol"); │ │ │ │ │ - var node = this.nodeFactory(null, "polygon"); │ │ │ │ │ - symbolNode.appendChild(node); │ │ │ │ │ - var symbolExtent = new OpenLayers.Bounds(Number.MAX_VALUE, Number.MAX_VALUE, 0, 0); │ │ │ │ │ - var points = []; │ │ │ │ │ - var x, y; │ │ │ │ │ - for (var i = 0; i < symbol.length; i = i + 2) { │ │ │ │ │ - x = symbol[i]; │ │ │ │ │ - y = symbol[i + 1]; │ │ │ │ │ - symbolExtent.left = Math.min(symbolExtent.left, x); │ │ │ │ │ - symbolExtent.bottom = Math.min(symbolExtent.bottom, y); │ │ │ │ │ - symbolExtent.right = Math.max(symbolExtent.right, x); │ │ │ │ │ - symbolExtent.top = Math.max(symbolExtent.top, y); │ │ │ │ │ - points.push(x, ",", y) │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + if (this.tileSize != null) { │ │ │ │ │ + obj.tileSize = this.tileSize.clone() │ │ │ │ │ } │ │ │ │ │ - node.setAttributeNS(null, "points", points.join(" ")); │ │ │ │ │ - var width = symbolExtent.getWidth(); │ │ │ │ │ - var height = symbolExtent.getHeight(); │ │ │ │ │ - var viewBox = [symbolExtent.left - width, symbolExtent.bottom - height, width * 3, height * 3]; │ │ │ │ │ - symbolNode.setAttributeNS(null, "viewBox", viewBox.join(" ")); │ │ │ │ │ - this.symbolMetrics[id] = [Math.max(width, height), symbolExtent.getCenterLonLat().lon, symbolExtent.getCenterLonLat().lat]; │ │ │ │ │ - this.defs.appendChild(symbolNode); │ │ │ │ │ - return symbolNode │ │ │ │ │ + obj.grid = []; │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - getFeatureIdFromEvent: function(evt) { │ │ │ │ │ - var featureId = OpenLayers.Renderer.Elements.prototype.getFeatureIdFromEvent.apply(this, arguments); │ │ │ │ │ - if (!featureId) { │ │ │ │ │ - var target = evt.target; │ │ │ │ │ - featureId = target.parentNode && target != this.rendererRoot ? target.parentNode._featureId : undefined │ │ │ │ │ - } │ │ │ │ │ - return featureId │ │ │ │ │ + getTileBounds: function(viewPortPx) { │ │ │ │ │ + var resolution = this.getResolution(); │ │ │ │ │ + var tileMapWidth = resolution * this.tileSize.w; │ │ │ │ │ + var tileMapHeight = resolution * this.tileSize.h; │ │ │ │ │ + var mapPoint = this.getLonLatFromViewPortPx(viewPortPx); │ │ │ │ │ + var tileLeft = tileMapWidth * Math.floor(mapPoint.lon / tileMapWidth); │ │ │ │ │ + var tileBottom = tileMapHeight * Math.floor(mapPoint.lat / tileMapHeight); │ │ │ │ │ + return new OpenLayers.Bounds(tileLeft, tileBottom, tileLeft + tileMapWidth, tileBottom + tileMapHeight) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Renderer.SVG" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.KaMap" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_ALIGN = { │ │ │ │ │ - l: "start", │ │ │ │ │ - r: "end", │ │ │ │ │ - b: "bottom", │ │ │ │ │ - t: "hanging" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_VSHIFT = { │ │ │ │ │ - t: "-70%", │ │ │ │ │ - b: "0" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.SVG.LABEL_VFACTOR = { │ │ │ │ │ - t: 0, │ │ │ │ │ - b: -1 │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Renderer.SVG.preventDefault = function(e) { │ │ │ │ │ - OpenLayers.Event.preventDefault(e) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Tile.Image.IFrame = { │ │ │ │ │ - useIFrame: null, │ │ │ │ │ - blankImageUrl: "data:image/gif;base64,R0lGODlhAQABAIAAAAAAAP///yH5BAEAAAAALAAAAAABAAEAQAIBRAA7", │ │ │ │ │ - draw: function() { │ │ │ │ │ - var draw = OpenLayers.Tile.Image.prototype.shouldDraw.call(this); │ │ │ │ │ - if (draw) { │ │ │ │ │ - var url = this.layer.getURL(this.bounds); │ │ │ │ │ - var usedIFrame = this.useIFrame; │ │ │ │ │ - this.useIFrame = this.maxGetUrlLength !== null && !this.layer.async && url.length > this.maxGetUrlLength; │ │ │ │ │ - var fromIFrame = usedIFrame && !this.useIFrame; │ │ │ │ │ - var toIFrame = !usedIFrame && this.useIFrame; │ │ │ │ │ - if (fromIFrame || toIFrame) { │ │ │ │ │ - if (this.imgDiv && this.imgDiv.parentNode === this.frame) { │ │ │ │ │ - this.frame.removeChild(this.imgDiv) │ │ │ │ │ - } │ │ │ │ │ - this.imgDiv = null; │ │ │ │ │ - if (fromIFrame) { │ │ │ │ │ - this.frame.removeChild(this.frame.firstChild) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ +OpenLayers.Layer.KaMapCache = OpenLayers.Class(OpenLayers.Layer.KaMap, { │ │ │ │ │ + IMAGE_EXTENSIONS: { │ │ │ │ │ + jpeg: "jpg", │ │ │ │ │ + gif: "gif", │ │ │ │ │ + png: "png", │ │ │ │ │ + png8: "png", │ │ │ │ │ + png24: "png", │ │ │ │ │ + dithered: "png" │ │ │ │ │ + }, │ │ │ │ │ + DEFAULT_FORMAT: "jpeg", │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + OpenLayers.Layer.KaMap.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.extension = this.IMAGE_EXTENSIONS[this.params.i.toLowerCase() || this.DEFAULT_FORMAT] │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var mapRes = this.map.getResolution(); │ │ │ │ │ + var scale = Math.round(this.map.getScale() * 1e4) / 1e4; │ │ │ │ │ + var pX = Math.round(bounds.left / mapRes); │ │ │ │ │ + var pY = -Math.round(bounds.top / mapRes); │ │ │ │ │ + var metaX = Math.floor(pX / this.tileSize.w / this.params.metaTileSize.w) * this.tileSize.w * this.params.metaTileSize.w; │ │ │ │ │ + var metaY = Math.floor(pY / this.tileSize.h / this.params.metaTileSize.h) * this.tileSize.h * this.params.metaTileSize.h; │ │ │ │ │ + var components = ["/", this.params.map, "/", scale, "/", this.params.g.replace(/\s/g, "_"), "/def/t", metaY, "/l", metaX, "/t", pY, "l", pX, ".", this.extension]; │ │ │ │ │ + var url = this.url; │ │ │ │ │ + if (OpenLayers.Util.isArray(url)) { │ │ │ │ │ + url = this.selectUrl(components.join(""), url) │ │ │ │ │ } │ │ │ │ │ - return OpenLayers.Tile.Image.prototype.draw.apply(this, arguments) │ │ │ │ │ + return url + components.join("") │ │ │ │ │ }, │ │ │ │ │ - getImage: function() { │ │ │ │ │ - if (this.useIFrame === true) { │ │ │ │ │ - if (!this.frame.childNodes.length) { │ │ │ │ │ - var eventPane = document.createElement("div"), │ │ │ │ │ - style = eventPane.style; │ │ │ │ │ - style.position = "absolute"; │ │ │ │ │ - style.width = "100%"; │ │ │ │ │ - style.height = "100%"; │ │ │ │ │ - style.zIndex = 1; │ │ │ │ │ - style.backgroundImage = "url(" + this.blankImageUrl + ")"; │ │ │ │ │ - this.frame.appendChild(eventPane) │ │ │ │ │ - } │ │ │ │ │ - var id = this.id + "_iFrame", │ │ │ │ │ - iframe; │ │ │ │ │ - if (parseFloat(navigator.appVersion.split("MSIE")[1]) < 9) { │ │ │ │ │ - iframe = document.createElement('<iframe name="' + id + '">'); │ │ │ │ │ - iframe.style.backgroundColor = "#FFFFFF"; │ │ │ │ │ - iframe.style.filter = "chroma(color=#FFFFFF)" │ │ │ │ │ - } else { │ │ │ │ │ - iframe = document.createElement("iframe"); │ │ │ │ │ - iframe.style.backgroundColor = "transparent"; │ │ │ │ │ - iframe.name = id │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.KaMapCache" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.ArcGIS93Rest = OpenLayers.Class(OpenLayers.Layer.Grid, { │ │ │ │ │ + DEFAULT_PARAMS: { │ │ │ │ │ + format: "png" │ │ │ │ │ + }, │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ + initialize: function(name, url, params, options) { │ │ │ │ │ + var newArguments = []; │ │ │ │ │ + params = OpenLayers.Util.upperCaseObject(params); │ │ │ │ │ + newArguments.push(name, url, params, options); │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, newArguments); │ │ │ │ │ + OpenLayers.Util.applyDefaults(this.params, OpenLayers.Util.upperCaseObject(this.DEFAULT_PARAMS)); │ │ │ │ │ + if (this.params.TRANSPARENT && this.params.TRANSPARENT.toString().toLowerCase() == "true") { │ │ │ │ │ + if (options == null || !options.isBaseLayer) { │ │ │ │ │ + this.isBaseLayer = false │ │ │ │ │ } │ │ │ │ │ - iframe.scrolling = "no"; │ │ │ │ │ - iframe.marginWidth = "0px"; │ │ │ │ │ - iframe.marginHeight = "0px"; │ │ │ │ │ - iframe.frameBorder = "0"; │ │ │ │ │ - iframe.style.position = "absolute"; │ │ │ │ │ - iframe.style.width = "100%"; │ │ │ │ │ - iframe.style.height = "100%"; │ │ │ │ │ - if (this.layer.opacity < 1) { │ │ │ │ │ - OpenLayers.Util.modifyDOMElement(iframe, null, null, null, null, null, null, this.layer.opacity) │ │ │ │ │ + if (this.params.FORMAT == "jpg") { │ │ │ │ │ + this.params.FORMAT = OpenLayers.Util.alphaHack() ? "gif" : "png" │ │ │ │ │ } │ │ │ │ │ - this.frame.appendChild(iframe); │ │ │ │ │ - this.imgDiv = iframe; │ │ │ │ │ - return iframe │ │ │ │ │ - } else { │ │ │ │ │ - return OpenLayers.Tile.Image.prototype.getImage.apply(this, arguments) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - createRequestForm: function() { │ │ │ │ │ - var form = document.createElement("form"); │ │ │ │ │ - form.method = "POST"; │ │ │ │ │ - var cacheId = this.layer.params["_OLSALT"]; │ │ │ │ │ - cacheId = (cacheId ? cacheId + "_" : "") + this.bounds.toBBOX(); │ │ │ │ │ - form.action = OpenLayers.Util.urlAppend(this.layer.url, cacheId); │ │ │ │ │ - form.target = this.id + "_iFrame"; │ │ │ │ │ - var imageSize = this.layer.getImageSize(), │ │ │ │ │ - params = OpenLayers.Util.getParameters(this.url), │ │ │ │ │ - field; │ │ │ │ │ - for (var par in params) { │ │ │ │ │ - field = document.createElement("input"); │ │ │ │ │ - field.type = "hidden"; │ │ │ │ │ - field.name = par; │ │ │ │ │ - field.value = params[par]; │ │ │ │ │ - form.appendChild(field) │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.ArcGIS93Rest(this.name, this.url, this.params, this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - return form │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ }, │ │ │ │ │ - setImgSrc: function(url) { │ │ │ │ │ - if (this.useIFrame === true) { │ │ │ │ │ - if (url) { │ │ │ │ │ - var form = this.createRequestForm(); │ │ │ │ │ - this.frame.appendChild(form); │ │ │ │ │ - form.submit(); │ │ │ │ │ - this.frame.removeChild(form) │ │ │ │ │ - } else if (this.imgDiv.parentNode === this.frame) { │ │ │ │ │ - this.frame.removeChild(this.imgDiv); │ │ │ │ │ - this.imgDiv = null │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + bounds = this.adjustBounds(bounds); │ │ │ │ │ + var projWords = this.projection.getCode().split(":"); │ │ │ │ │ + var srid = projWords[projWords.length - 1]; │ │ │ │ │ + var imageSize = this.getImageSize(); │ │ │ │ │ + var newParams = { │ │ │ │ │ + BBOX: bounds.toBBOX(), │ │ │ │ │ + SIZE: imageSize.w + "," + imageSize.h, │ │ │ │ │ + F: "image", │ │ │ │ │ + BBOXSR: srid, │ │ │ │ │ + IMAGESR: srid │ │ │ │ │ + }; │ │ │ │ │ + if (this.layerDefs) { │ │ │ │ │ + var layerDefStrList = []; │ │ │ │ │ + var layerID; │ │ │ │ │ + for (layerID in this.layerDefs) { │ │ │ │ │ + if (this.layerDefs.hasOwnProperty(layerID)) { │ │ │ │ │ + if (this.layerDefs[layerID]) { │ │ │ │ │ + layerDefStrList.push(layerID); │ │ │ │ │ + layerDefStrList.push(":"); │ │ │ │ │ + layerDefStrList.push(this.layerDefs[layerID]); │ │ │ │ │ + layerDefStrList.push(";") │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + if (layerDefStrList.length > 0) { │ │ │ │ │ + newParams["LAYERDEFS"] = layerDefStrList.join("") │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - OpenLayers.Tile.Image.prototype.setImgSrc.apply(this, arguments) │ │ │ │ │ } │ │ │ │ │ + var requestString = this.getFullRequestString(newParams); │ │ │ │ │ + return requestString │ │ │ │ │ }, │ │ │ │ │ - onImageLoad: function() { │ │ │ │ │ - OpenLayers.Tile.Image.prototype.onImageLoad.apply(this, arguments); │ │ │ │ │ - if (this.useIFrame === true) { │ │ │ │ │ - this.imgDiv.style.opacity = 1; │ │ │ │ │ - this.frame.style.opacity = this.layer.opacity │ │ │ │ │ + setLayerFilter: function(id, queryDef) { │ │ │ │ │ + if (!this.layerDefs) { │ │ │ │ │ + this.layerDefs = {} │ │ │ │ │ + } │ │ │ │ │ + if (queryDef) { │ │ │ │ │ + this.layerDefs[id] = queryDef │ │ │ │ │ + } else { │ │ │ │ │ + delete this.layerDefs[id] │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - createBackBuffer: function() { │ │ │ │ │ - var backBuffer; │ │ │ │ │ - if (this.useIFrame === false) { │ │ │ │ │ - backBuffer = OpenLayers.Tile.Image.prototype.createBackBuffer.call(this) │ │ │ │ │ + clearLayerFilter: function(id) { │ │ │ │ │ + if (id) { │ │ │ │ │ + delete this.layerDefs[id] │ │ │ │ │ + } else { │ │ │ │ │ + delete this.layerDefs │ │ │ │ │ } │ │ │ │ │ - return backBuffer │ │ │ │ │ - } │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Lang["cs-CZ"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Nezpracovaná návratová hodnota ${statusText}", │ │ │ │ │ - Permalink: "Trvalý odkaz", │ │ │ │ │ - Overlays: "Překryvné vrstvy", │ │ │ │ │ - "Base Layer": "Podkladové vrstvy", │ │ │ │ │ - noFID: "Nelze aktualizovat prvek, pro který neexistuje FID.", │ │ │ │ │ - browserNotSupported: "Váš prohlížeč nepodporuje vykreslování vektorů. Momentálně podporované nástroje jsou::\n${renderers}", │ │ │ │ │ - minZoomLevelError: "Vlastnost minZoomLevel by se měla používat pouze s potomky FixedZoomLevels vrstvami. To znamená, že vrstva wfs kontroluje, zda-li minZoomLevel není zbytek z minulosti.Nelze to ovšem vyjmout bez možnosti, že bychom rozbili aplikace postavené na OL, které by na tom mohly záviset. Proto tuto vlastnost nedoporučujeme používat -- kontrola minZoomLevel bude odstraněna ve verzi 3.0. Použijte prosím raději nastavení min/max podle příkaldu popsaného na: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "WFS Transaction: ÚSPĚCH ${response}", │ │ │ │ │ - commitFailed: "WFS Transaction: CHYBA ${response}", │ │ │ │ │ - googleWarning: "Nepodařilo se správně načíst vrstvu Google.<br><br>Abyste se zbavili této zprávy, zvolte jinou základní vrstvu v přepínači vrstev.<br><br>To se většinou stává, pokud nebyl načten skript, nebo neobsahuje správný klíč pro API pro tuto stránku.<br><br>Vývojáři: Pro pomoc, aby tohle fungovalo , <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>klikněte sem</a>", │ │ │ │ │ - getLayerWarning: "The ${layerType} Layer was unable to load correctly.<br><br>To get rid of this message, select a new BaseLayer in the layer switcher in the upper-right corner.<br><br>Most likely, this is because the ${layerLib} library script was either not correctly included.<br><br>Developers: For help getting this working correctly, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>click here</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Měřítko = 1 : ${scaleDenom}", │ │ │ │ │ - reprojectDeprecated: "Použil jste volbu 'reproject' ve vrstvě ${layerName}. Tato volba není doporučená: byla zde proto, aby bylo možno zobrazovat data z okomerčních serverů, ale tato funkce je nyní zajištěna pomocí podpory Spherical Mercator. Více informací naleznete na http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Tato metoda je zavržená a bude ve verzi 3.0 odstraněna. Prosím, použijte raději ${newMethod}." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang.ro = { │ │ │ │ │ - unhandledRequest: "Cerere nesoluționată return ${statusText}", │ │ │ │ │ - Permalink: "Legatură permanentă", │ │ │ │ │ - Overlays: "Straturi vector", │ │ │ │ │ - "Base Layer": "Straturi de bază", │ │ │ │ │ - noFID: "Nu pot actualiza un feature pentru care nu există FID.", │ │ │ │ │ - browserNotSupported: "Browserul tău nu suportă afișarea vectorilor. Supoetul curent pentru randare:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "Proprietatea minZoomLevel este doar pentru a fi folosită " + "cu straturile FixedZoomLevels-descendent. De aceea acest " + "strat wfs verifică dacă minZoomLevel este o relicvă" + ". Nu îl putem , oricum, înlătura fără " + "a afecta aplicațiile Openlayers care depind de ea." + " De aceea considerăm depreciat -- minZoomLevel " + "și îl vom înlătura în 3.0. Folosește " + "min/max resolution cum este descrisă aici: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "Tranzacție WFS: SUCCES ${response}", │ │ │ │ │ - commitFailed: "Tranzacție WFS : EȘEC ${response}", │ │ │ │ │ - googleWarning: "Stratul Google nu a putut fi încărcat corect.<br><br>" + "Pentru a elimina acest mesaj, selectează un nou strat de bază " + "în Layerswitcher din colțul dreata-sus.<br><br>" + "Asta datorită, faptului că Google Maps library " + "script nu este inclus, sau nu conține " + "cheia API corectă pentru situl tău.<br><br>" + "Developeri: Pentru ajutor, " + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>apăsați aici</a>", │ │ │ │ │ - getLayerWarning: "Stratul ${layerType} nu a putut fi încărcat corect.<br><br>" + "pentru a înlătura acest mesaj, selectează un nou strat de bază " + "Acesta eroare apare de obicei când ${layerLib} library " + "script nu a fost încărcat corect.<br><br>" + "Developeri: Pentru ajutor privind utilizarea corectă, " + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>apasă aici</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Scara = 1 : ${scaleDenom}", │ │ │ │ │ - W: "V", │ │ │ │ │ - E: "E", │ │ │ │ │ - N: "N", │ │ │ │ │ - S: "S", │ │ │ │ │ - Graticule: "Graticule", │ │ │ │ │ - reprojectDeprecated: "folosești opțiunea 'reproject' " + "pentru stratul ${layerName} . Această opțiune este depreciată: " + "a fost utilizată pentru afișarea straturilor de bază comerciale " + "Mai multe informații despre proiecția Mercator sunt disponibile aici " + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Această metodă este depreciată și va fi înlăturată in versiunea 3.0. " + "folosește metoda ${newMethod}.", │ │ │ │ │ - end: "" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Lang["pl"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Nieobsługiwane żądanie zwróciło ${statusText}", │ │ │ │ │ - Permalink: "Permalink", │ │ │ │ │ - Overlays: "Nakładki", │ │ │ │ │ - "Base Layer": "Warstwa podstawowa", │ │ │ │ │ - noFID: "Nie można zaktualizować funkcji, dla których nie ma FID.", │ │ │ │ │ - browserNotSupported: "Twoja przeglądarka nie obsługuje renderowania wektorów. Obecnie obsługiwane renderowanie to:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "Właściwość minZoomLevel jest przeznaczona tylko do użytku " + "z warstwami FixedZoomLevels-descendent." + "Warstwa wfs, która sprawdza minZoomLevel jest reliktem przeszłości." + "Nie możemy jej jednak usunąc bez mozliwości łamania OL aplikacji, " + "które mogą być od niej zależne. " + "Dlatego jesteśmy za deprecjację -- minZoomLevel " + "zostanie usunięta w wersji 3.0. W zamian prosze użyj " + "min/max rozdzielczości w sposób opisany tutaj: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "Transakcja WFS: SUKCES ${response}", │ │ │ │ │ - commitFailed: "Transakcja WFS: FAILED ${response}", │ │ │ │ │ - googleWarning: "Warstwa Google nie był w stanie załadować się poprawnie.<br><br>" + "Aby pozbyć się tej wiadomości, wybierz nową Warstwe podstawową " + "w przełączniku warstw w górnym prawym rogu mapy.<br><br>" + "Najprawdopodobniej jest to spowodowane tym, że biblioteka Google Maps " + "nie jest załadowana, lub nie zawiera poprawnego klucza do API dla twojej strony<br><br>" + "Programisto: Aby uzyskać pomoc , " + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>kliknij tutaj</a>", │ │ │ │ │ - getLayerWarning: "Warstwa ${layerType} nie mogła zostać załadowana poprawnie.<br><br>" + "Aby pozbyć się tej wiadomości, wybierz nową Warstwe podstawową " + "w przełączniku warstw w górnym prawym rogu mapy.<br><br>" + "Najprawdopodobniej jest to spowodowane tym, że biblioteka ${layerLib} " + "nie jest załadowana, lub może(o ile biblioteka tego wymaga) " + "byc potrzebny klucza do API dla twojej strony<br><br>" + "Programisto: Aby uzyskać pomoc , " + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>kliknij tutaj</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Skala = 1 : ${scaleDenom}", │ │ │ │ │ - W: "ZACH", │ │ │ │ │ - E: "WSCH", │ │ │ │ │ - N: "PN", │ │ │ │ │ - S: "PD", │ │ │ │ │ - Graticule: "Siatka", │ │ │ │ │ - reprojectDeprecated: "w warstwie ${layerName} używasz opcji 'reproject'. " + "Ta opcja jest przestarzała: " + "jej zastosowanie został zaprojektowany, aby wspierać wyświetlania danych przez komercyjne mapy, " + "jednak obecnie ta funkcjonalność powinien zostać osiągnięty za pomocą Spherical Mercator " + "its use was designed to support displaying data over commercial. Więcje informacji na ten temat możesz znaleźć na stronie " + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Ta metoda jest przestarzała i będzie usunięta od wersji 3.0. " + "W zamian użyj ${newMethod}." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["nb"] = { │ │ │ │ │ - unhandledRequest: "Ubehandlet forespørsel returnerte ${statusText}", │ │ │ │ │ - Permalink: "Kobling til denne siden", │ │ │ │ │ - Overlays: "Kartlag", │ │ │ │ │ - "Base Layer": "Bakgrunnskart", │ │ │ │ │ - noFID: "Kan ikke oppdatere et feature (et objekt) som ikke har FID.", │ │ │ │ │ - browserNotSupported: "Din nettleser støtter ikke vektortegning. Tegnemetodene som støttes er:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "Egenskapen minZoomLevel er kun ment til bruk på lag " + "basert på FixedZoomLevels. At dette wfs-laget sjekker " + "minZoomLevel er en etterlevning fra tidligere versjoner. Det kan dog ikke " + "tas bort uten å risikere at OL-baserte applikasjoner " + "slutter å virke, så det er merket som foreldet: " + "minZoomLevel i sjekken nedenfor vil fjernes i 3.0. " + "Vennligst bruk innstillingene for min/maks oppløsning " + "som er beskrevet her: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "WFS-transaksjon: LYKTES ${response}", │ │ │ │ │ - commitFailed: "WFS-transaksjon: MISLYKTES ${response}", │ │ │ │ │ - googleWarning: "Google-laget kunne ikke lastes.<br><br>" + "Bytt til et annet bakgrunnslag i lagvelgeren i " + "øvre høyre hjørne for å slippe denne meldingen.<br><br>" + "Sannsynligvis forårsakes feilen av at Google Maps-biblioteket " + "ikke er riktig inkludert på nettsiden, eller at det ikke er " + "angitt riktig API-nøkkel for nettstedet.<br><br>" + "Utviklere: For hjelp til å få dette til å virke se " + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>her</a>.", │ │ │ │ │ - getLayerWarning: "${layerType}-laget kunne ikke lastes.<br><br>" + "Bytt til et annet bakgrunnslag i lagvelgeren i " + "øvre høyre hjørne for å slippe denne meldingen.<br><br>" + "Sannsynligvis forårsakes feilen av at " + "${layerLib}-biblioteket ikke var riktig inkludert " + "på nettsiden.<br><br>" + "Utviklere: For hjelp til å få dette til å virke se " + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>her</a>.", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "<strong>Skala</strong> 1 : ${scaleDenom}", │ │ │ │ │ - reprojectDeprecated: "Du bruker innstillingen 'reproject' på laget ${layerName}. " + "Denne innstillingen er foreldet, den var ment for å støtte " + "visning av kartdata over kommersielle bakgrunnskart, men det " + "bør nå gjøres med støtten for Spherical Mercator. Mer informasjon " + "finnes på http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Denne metoden er markert som foreldet og vil bli fjernet i 3.0. " + "Vennligst bruk ${newMethod} i stedet.", │ │ │ │ │ - end: "" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Lang["no"] = OpenLayers.Lang["nb"]; │ │ │ │ │ -OpenLayers.Lang["is"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Permalink: "Varanlegur tengill", │ │ │ │ │ - Overlays: "Þekjur", │ │ │ │ │ - "Base Layer": "Grunnlag", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Skali = 1 : ${scaleDenom}", │ │ │ │ │ - methodDeprecated: "Þetta fall hefur verið úrelt og verður fjarlægt í 3.0. Notaðu ${newMethod} í staðin." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["de"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Unbehandelte Anfragerückmeldung ${statusText}", │ │ │ │ │ - Permalink: "Permalink", │ │ │ │ │ - Overlays: "Overlays", │ │ │ │ │ - "Base Layer": "Grundkarte", │ │ │ │ │ - noFID: "Ein Feature, für das keine FID existiert, kann nicht aktualisiert werden.", │ │ │ │ │ - browserNotSupported: "Ihr Browser unterstützt keine Vektordarstellung. Aktuell unterstützte Renderer:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "Die <code>minZoomLevel</code>-Eigenschaft ist nur für die Verwendung mit <code>FixedZoomLevels</code>-untergeordneten Layers vorgesehen. Das dieser <tt>wfs</tt>-Layer die <code>minZoomLevel</code>-Eigenschaft überprüft ist ein Relikt der Vergangenheit. Wir können diese Überprüfung nicht entfernen, ohne das OL basierende Applikationen nicht mehr funktionieren. Daher markieren wir es als veraltet - die <code>minZoomLevel</code>-Überprüfung wird in Version 3.0 entfernt werden. Bitte verwenden Sie stattdessen die Min-/Max-Lösung, wie sie unter http://trac.openlayers.org/wiki/SettingZoomLevels beschrieben ist.", │ │ │ │ │ - commitSuccess: "WFS-Transaktion: Erfolgreich ${response}", │ │ │ │ │ - commitFailed: "WFS-Transaktion: Fehlgeschlagen ${response}", │ │ │ │ │ - googleWarning: "Der Google-Layer konnte nicht korrekt geladen werden.<br><br>Um diese Meldung nicht mehr zu erhalten, wählen Sie einen anderen Hintergrundlayer aus dem LayerSwitcher in der rechten oberen Ecke.<br><br>Sehr wahrscheinlich tritt dieser Fehler auf, weil das Skript der Google-Maps-Bibliothek nicht eingebunden wurde oder keinen gültigen API-Schlüssel für Ihre URL enthält.<br><br>Entwickler: Besuche <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>das Wiki</a> für Hilfe zum korrekten Einbinden des Google-Layers", │ │ │ │ │ - getLayerWarning: "Der ${layerType}-Layer konnte nicht korrekt geladen werden.<br><br>Um diese Meldung nicht mehr zu erhalten, wählen Sie einen anderen Hintergrundlayer aus dem LayerSwitcher in der rechten oberen Ecke.<br><br>Sehr wahrscheinlich tritt dieser Fehler auf, weil das Skript der '${layerLib}'-Bibliothek nicht eingebunden wurde.<br><br>Entwickler: Besuche <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>das Wiki</a> für Hilfe zum korrekten Einbinden von Layern", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Maßstab = 1 : ${scaleDenom}", │ │ │ │ │ - W: "W", │ │ │ │ │ - E: "O", │ │ │ │ │ - N: "N", │ │ │ │ │ - S: "S", │ │ │ │ │ - reprojectDeprecated: "Sie verwenden die „Reproject“-Option des Layers ${layerName}. Diese Option ist veraltet: Sie wurde entwickelt um die Anzeige von Daten auf kommerziellen Basiskarten zu unterstützen, aber diese Funktion sollte jetzt durch Unterstützung der „Spherical Mercator“ erreicht werden. Weitere Informationen sind unter http://trac.openlayers.org/wiki/SphericalMercator verfügbar.", │ │ │ │ │ - methodDeprecated: "Die Methode ist veraltet und wird in 3.0 entfernt. Bitte verwende stattdessen ${newMethod}." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang.en = { │ │ │ │ │ - unhandledRequest: "Unhandled request return ${statusText}", │ │ │ │ │ - Permalink: "Permalink", │ │ │ │ │ - Overlays: "Overlays", │ │ │ │ │ - "Base Layer": "Base Layer", │ │ │ │ │ - noFID: "Can't update a feature for which there is no FID.", │ │ │ │ │ - browserNotSupported: "Your browser does not support vector rendering. Currently supported renderers are:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "The minZoomLevel property is only intended for use " + "with the FixedZoomLevels-descendent layers. That this " + "wfs layer checks for minZoomLevel is a relic of the" + "past. We cannot, however, remove it without possibly " + "breaking OL based applications that may depend on it." + " Therefore we are deprecating it -- the minZoomLevel " + "check below will be removed at 3.0. Please instead " + "use min/max resolution setting as described here: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "WFS Transaction: SUCCESS ${response}", │ │ │ │ │ - commitFailed: "WFS Transaction: FAILED ${response}", │ │ │ │ │ - googleWarning: "The Google Layer was unable to load correctly.<br><br>" + "To get rid of this message, select a new BaseLayer " + "in the layer switcher in the upper-right corner.<br><br>" + "Most likely, this is because the Google Maps library " + "script was either not included, or does not contain the " + "correct API key for your site.<br><br>" + "Developers: For help getting this working correctly, " + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>click here</a>", │ │ │ │ │ - getLayerWarning: "The ${layerType} Layer was unable to load correctly.<br><br>" + "To get rid of this message, select a new BaseLayer " + "in the layer switcher in the upper-right corner.<br><br>" + "Most likely, this is because the ${layerLib} library " + "script was not correctly included.<br><br>" + "Developers: For help getting this working correctly, " + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>click here</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Scale = 1 : ${scaleDenom}", │ │ │ │ │ - W: "W", │ │ │ │ │ - E: "E", │ │ │ │ │ - N: "N", │ │ │ │ │ - S: "S", │ │ │ │ │ - Graticule: "Graticule", │ │ │ │ │ - reprojectDeprecated: "You are using the 'reproject' option " + "on the ${layerName} layer. This option is deprecated: " + "its use was designed to support displaying data over commercial " + "basemaps, but that functionality should now be achieved by using " + "Spherical Mercator support. More information is available from " + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "This method has been deprecated and will be removed in 3.0. " + "Please use ${newMethod} instead.", │ │ │ │ │ - end: "" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Lang["en-CA"] = OpenLayers.Util.applyDefaults({}, OpenLayers.Lang["en"]); │ │ │ │ │ -OpenLayers.Lang["el"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Κλίμακα ~ 1 : ${scaleDenom}" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["oc"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Requèsta pas gerida, retorna ${statusText}", │ │ │ │ │ - Permalink: "Permaligam", │ │ │ │ │ - Overlays: "Calques", │ │ │ │ │ - "Base Layer": "Calc de basa", │ │ │ │ │ - noFID: "Impossible de metre a jorn un objècte sens identificant (fid).", │ │ │ │ │ - browserNotSupported: "Vòstre navegidor supòrta pas lo rendut vectorial. Los renderers actualament suportats son : \n${renderers}", │ │ │ │ │ - minZoomLevelError: "La proprietat minZoomLevel deu èsser utilizada solament per de jaces FixedZoomLevels-descendent. Lo fach qu'aqueste jaç WFS verifique la preséncia de minZoomLevel es una relica del passat. Çaquelà, la podèm suprimir sens copar d'aplicacions que ne poirián dependre. Es per aquò que la depreciam -- la verificacion del minZoomLevel serà suprimida en version 3.0. A la plaça, mercés d'utilizar los paramètres de resolucions min/max tal coma descrich sus : http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "Transaccion WFS : SUCCES ${response}", │ │ │ │ │ - commitFailed: "Transaccion WFS : FRACAS ${response}", │ │ │ │ │ - googleWarning: "Lo jaç Google es pas estat en mesura de se cargar corrèctament.<br><br>Per suprimir aqueste messatge, causissètz una BaseLayer novèla dins lo selector de jaç en naut a drecha.<br><br>Aquò es possiblament causat par la non-inclusion de la librariá Google Maps, o alara perque que la clau de l'API correspond pas a vòstre site.<br><br>Desvolopaires : per saber cossí corregir aquò, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>clicatz aicí</a>", │ │ │ │ │ - getLayerWarning: "Lo jaç ${layerType} es pas en mesura de se cargar corrèctament.<br><br>Per suprimir aqueste messatge, causissètz una BaseLayer novèla dins lo selector de jaç en naut a drecha.<br><br>Aquò es possiblament causat per la non-inclusion de la librariá ${layerLib}.<br><br>Desvolopaires : per saber cossí corregir aquí, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>clicatz aicí</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Escala ~ 1 : ${scaleDenom}", │ │ │ │ │ - W: "O", │ │ │ │ │ - E: "È", │ │ │ │ │ - N: "N", │ │ │ │ │ - S: "S", │ │ │ │ │ - reprojectDeprecated: "Utilizatz l'opcion 'reproject' sul jaç ${layerName}. Aquesta opcion es despreciada : Son usatge permetiá d'afichar de donadas al dessús de jaces raster comercials. Aquesta foncionalitat ara es suportada en utilizant lo supòrt de la projeccion Mercator Esferica. Mai d'informacion es disponibla sus http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Aqueste metòde es despreciada, e serà suprimida a la version 3.0. Mercés d'utilizar ${newMethod} a la plaça." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["lt"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Neapdorota užklausa gražino ${statusText}", │ │ │ │ │ - Permalink: "Pastovi nuoroda", │ │ │ │ │ - Overlays: "Papildomi sluoksniai", │ │ │ │ │ - "Base Layer": "Pagrindinis sluoksnis", │ │ │ │ │ - noFID: "Negaliu atnaujinti objekto, kuris neturi FID.", │ │ │ │ │ - browserNotSupported: "Jūsų naršyklė nemoka parodyti vektorių. Šiuo metu galima naudotis tokiais rodymo varikliais:\n{renderers}", │ │ │ │ │ - commitSuccess: "WFS Tranzakcija: PAVYKO ${response}", │ │ │ │ │ - commitFailed: "WFS Tranzakcija: ŽLUGO ${response}", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Mastelis = 1 : ${scaleDenom}", │ │ │ │ │ - W: "V", │ │ │ │ │ - E: "R", │ │ │ │ │ - N: "Š", │ │ │ │ │ - S: "P", │ │ │ │ │ - Graticule: "Tinklelis", │ │ │ │ │ - methodDeprecated: "Šis metodas yra pasenęs ir 3.0 versijoje bus pašalintas. " + "Prašome naudoti ${newMethod}.", │ │ │ │ │ - end: "" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["sv"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Ej hanterad fråga retur ${statusText}", │ │ │ │ │ - Permalink: "Permalänk", │ │ │ │ │ - Overlays: "Kartlager", │ │ │ │ │ - "Base Layer": "Bakgrundskarta", │ │ │ │ │ - noFID: "Kan ej uppdatera feature (objekt) för vilket FID saknas.", │ │ │ │ │ - browserNotSupported: "Din webbläsare stöder inte vektorvisning. För närvarande stöds följande visning:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "Egenskapen minZoomLevel är endast avsedd att användas med lager med FixedZoomLevels. Att detta WFS-lager kontrollerar minZoomLevel är en relik från äldre versioner. Vi kan dock inte ta bort det utan att riskera att OL-baserade tillämpningar som använder detta slutar fungera. Därför är det satt som deprecated, minZoomLevel kommer att tas bort i version 3.0. Använd i stället inställning av min/max resolution som beskrivs här: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "WFS-transaktion: LYCKADES ${response}", │ │ │ │ │ - commitFailed: "WFS-transaktion: MISSLYCKADES ${response}", │ │ │ │ │ - googleWarning: "Google-lagret kunde inte laddas korrekt.<br><br>För att slippa detta meddelande, välj en annan bakgrundskarta i lagerväljaren i övre högra hörnet.<br><br>Sannolikt beror felet på att Google Maps-biblioteket inte är inkluderat på webbsidan eller på att sidan inte anger korrekt API-nyckel för webbplatsen.<br><br>Utvecklare: hjälp för att åtgärda detta, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>klicka här</a>.", │ │ │ │ │ - getLayerWarning: "${layerType}-lagret kunde inte laddas korrekt.<br><br>För att slippa detta meddelande, välj en annan bakgrundskarta i lagerväljaren i övre högra hörnet.<br><br>Sannolikt beror felet på att ${layerLib}-biblioteket inte är inkluderat på webbsidan.<br><br>Utvecklare: hjälp för att åtgärda detta, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>klicka här</a>.", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "<strong>Skala</strong> 1 : ${scaleDenom}", │ │ │ │ │ - reprojectDeprecated: "Du använder inställningen 'reproject' på lagret ${layerName}. Denna inställning markerad som deprecated: den var avsedd att användas för att stödja visning av kartdata på kommersiella bakgrundskartor, men nu bör man i stället använda Spherical Mercator-stöd för den funktionaliteten. Mer information finns på http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Denna metod är markerad som deprecated och kommer att tas bort i 3.0. Använd ${newMethod} i stället." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang.es = { │ │ │ │ │ - unhandledRequest: "Respuesta a petición no gestionada ${statusText}", │ │ │ │ │ - Permalink: "Enlace permanente", │ │ │ │ │ - Overlays: "Capas superpuestas", │ │ │ │ │ - "Base Layer": "Capa Base", │ │ │ │ │ - noFID: "No se puede actualizar un elemento para el que no existe FID.", │ │ │ │ │ - browserNotSupported: "Su navegador no soporta renderización vectorial. Los renderizadores soportados actualmente son:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "La propiedad minZoomLevel debe sólo utilizarse " + "con las capas que tienen FixedZoomLevels. El hecho de que " + "una capa wfs compruebe minZoomLevel es una reliquia del " + "pasado. Sin embargo, no podemos eliminarla sin discontinuar " + "probablemente las aplicaciones OL que puedan depender de ello. " + "Así pues estamos haciéndolo obsoleto --la comprobación " + "minZoomLevel se eliminará en la versión 3.0. Utilice el ajuste " + "de resolution min/max en su lugar, tal como se describe aquí: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "Transacción WFS: ÉXITO ${response}", │ │ │ │ │ - commitFailed: "Transacción WFS: FALLÓ ${response}", │ │ │ │ │ - googleWarning: "La capa Google no pudo ser cargada correctamente.<br><br>" + "Para evitar este mensaje, seleccione una nueva Capa Base " + "en el selector de capas en la esquina superior derecha.<br><br>" + "Probablemente, esto se debe a que el script de la biblioteca de " + "Google Maps no fue correctamente incluido en su página, o no " + "contiene la clave del API correcta para su sitio.<br><br>" + "Desarrolladores: Para ayudar a hacer funcionar esto correctamente, " + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>haga clic aquí</a>", │ │ │ │ │ - getLayerWarning: "La capa ${layerType} no pudo ser cargada correctamente.<br><br>" + "Para evitar este mensaje, seleccione una nueva Capa Base " + "en el selector de capas en la esquina superior derecha.<br><br>" + "Probablemente, esto se debe a que el script de " + "la biblioteca ${layerLib} " + "no fue correctamente incluido en su página.<br><br>" + "Desarrolladores: Para ayudar a hacer funcionar esto correctamente, " + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>haga clic aquí</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Escala = 1 : ${scaleDenom}", │ │ │ │ │ - W: "O", │ │ │ │ │ - E: "E", │ │ │ │ │ - N: "N", │ │ │ │ │ - S: "S", │ │ │ │ │ - Graticule: "Retícula", │ │ │ │ │ - reprojectDeprecated: "Está usando la opción 'reproject' en la capa " + "${layerName}. Esta opción es obsoleta: su uso fue diseñado " + "para soportar la visualización de datos sobre mapas base comerciales, " + "pero ahora esa funcionalidad debería conseguirse mediante el soporte " + "de la proyección Spherical Mercator. Más información disponible en " + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Este método es obsoleto y se eliminará en la versión 3.0. " + "Por favor utilice el método ${newMethod} en su lugar.", │ │ │ │ │ - end: "" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Lang["vi"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Không xử lý được phản hồi ${statusText} cho yêu cầu", │ │ │ │ │ - Permalink: "Liên kết thường trực", │ │ │ │ │ - Overlays: "Lấp bản đồ", │ │ │ │ │ - "Base Layer": "Lớp nền", │ │ │ │ │ - noFID: "Không thể cập nhật tính năng thiếu FID.", │ │ │ │ │ - browserNotSupported: "Trình duyệt của bạn không hỗ trợ chức năng vẽ bằng vectơ. Hiện hỗ trợ các bộ kết xuất:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "Chỉ nên sử dụng thuộc tính minZoomLevel với các lớp FixedZoomLevels-descendent. Việc lớp wfs này tìm cho minZoomLevel là di tích còn lại từ xưa. Tuy nhiên, nếu chúng tôi dời nó thì sẽ vỡ các chương trình OpenLayers mà dựa trên nó. Bởi vậy chúng tôi phản đối sử dụng nó – bước tìm cho minZoomLevel sẽ được dời vào phiên bản 3.0. Xin sử dụng thiết lập độ phân tích tối thiểu / tối đa thay thế, theo hướng dẫn này: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "Giao dịch WFS: THÀNH CÔNG ${response}", │ │ │ │ │ - commitFailed: "Giao dịch WFS: THẤT BẠI ${response}", │ │ │ │ │ - googleWarning: "Không thể tải lớp Google đúng đắn.<br><br>Để tránh thông báo này lần sau, hãy chọn BaseLayer mới dùng điều khiển chọn lớp ở góc trên phải.<br><br>Chắc script thư viện Google Maps hoặc không được bao gồm hoặc không chứa khóa API hợp với website của bạn.<br><br><a href='http://trac.openlayers.org/wiki/Google' target='_blank'>Trợ giúp về tính năng này</a> cho người phát triển.", │ │ │ │ │ - getLayerWarning: "Không thể tải lớp ${layerType} đúng đắn.<br><br>Để tránh thông báo này lần sau, hãy chọn BaseLayer mới dùng điều khiển chọn lớp ở góc trên phải.<br><br>Chắc script thư viện ${layerLib} không được bao gồm đúng kiểu.<br><br><a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>Trợ giúp về tính năng này</a> cho người phát triển.", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Tỷ lệ = 1 : ${scaleDenom}", │ │ │ │ │ - W: "T", │ │ │ │ │ - E: "Đ", │ │ │ │ │ - N: "B", │ │ │ │ │ - S: "N", │ │ │ │ │ - reprojectDeprecated: "Bạn đang áp dụng chế độ “reproject” vào lớp ${layerName}. Chế độ này đã bị phản đối: nó có mục đích hỗ trợ lấp dữ liệu trên các nền bản đồ thương mại; nên thực hiện hiệu ứng đó dùng tính năng Mercator Hình cầu. Có sẵn thêm chi tiết tại http://trac.openlayers.org/wiki/SphericalMercator .", │ │ │ │ │ - methodDeprecated: "Phương thức này đã bị phản đối và sẽ bị dời vào phiên bản 3.0. Xin hãy sử dụng ${newMethod} thay thế." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["io"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Skalo = 1 : ${scaleDenom}" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["nn"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Skala = 1 : ${scaleDenom}" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["sk"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Neobslúžené požiadavky vracajú ${statusText}", │ │ │ │ │ - Permalink: "Trvalý odkaz", │ │ │ │ │ - Overlays: "Prekrytia", │ │ │ │ │ - "Base Layer": "Základná vrstva", │ │ │ │ │ - noFID: "Nie je možné aktualizovať vlastnosť, pre ktorú neexistuje FID.", │ │ │ │ │ - browserNotSupported: "Váš prehliadač nepodporuje vykresľovanie vektorov. Momentálne podporované vykresľovače sú:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "Vlastnosť minZoomLevel je určený iba na použitie s vrstvami odvodenými od FixedZoomLevels. To, že táto wfs vrstva kontroluje minZoomLevel je pozostatok z minulosti. Nemôžeme ho však odstrániť, aby sme sa vyhli možnému porušeniu aplikácií založených na Open Layers, ktoré na tomto môže závisieť. Preto ho označujeme ako zavrhovaný - dolu uvedená kontrola minZoomLevel bude odstránená vo verzii 3.0. Použite prosím namiesto toho kontrolu min./max. rozlíšenia podľa tu uvedeného popisu: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "Transakcia WFS: ÚSPEŠNÁ ${response}", │ │ │ │ │ - commitFailed: "Transakcia WFS: ZLYHALA ${response}", │ │ │ │ │ - googleWarning: "Vrstvu Google nebolo možné správne načítať.<br><br>Aby ste sa tejto správy zbavili vyberte novú BaseLayer v prepínači vrstiev v pravom hornom rohu.<br><br>Toto sa stalo pravdepodobne preto, že skript knižnice Google Maps buď nebol načítaný alebo neobsahuje správny kľúč API pre vašu lokalitu.<br><br>Vývojári: Tu môžete získať <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>pomoc so sfunkčnením</a>", │ │ │ │ │ - getLayerWarning: "Vrstvu ${layerType} nebolo možné správne načítať.<br><br>Aby ste sa tejto správy zbavili vyberte novú BaseLayer v prepínači vrstiev v pravom hornom rohu.<br><br>Toto sa stalo pravdepodobne preto, že skript knižnice ${layerType} buď nebol načítaný alebo neobsahuje správny kľúč API pre vašu lokalitu.<br><br>Vývojári: Tu môžete získať <a href='http://trac.openlayers.org/wiki/${layerType}' target='_blank'>pomoc so sfunkčnením</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Mierka = 1 : ${scaleDenom}", │ │ │ │ │ - reprojectDeprecated: "Používate voľby „reproject“ vrstvy ${layerType}. Táto voľba je zzavrhovaná: jej použitie bolo navrhnuté na podporu zobrazovania údajov nad komerčnými základovými mapami, ale túto funkcionalitu je teraz možné dosiahnuť pomocou Spherical Mercator. Ďalšie informácie získate na stránke http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Táto metóda je zavrhovaná a bude odstránená vo verzii 3.0. Použite prosím namiesto nej metódu ${newMethod}." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["gsw"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Nit behandleti Aafrogsruckmäldig ${statusText}", │ │ │ │ │ - Permalink: "Permalink", │ │ │ │ │ - Overlays: "Iberlagerige", │ │ │ │ │ - "Base Layer": "Grundcharte", │ │ │ │ │ - noFID: "E Feature, wu s kei FID derfir git, cha nit aktualisiert wäre.", │ │ │ │ │ - browserNotSupported: "Dyy Browser unterstitzt kei Vektordarstellig. Aktuäll unterstitzti Renderer:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "D minZoomLevel-Eigeschaft isch nume dänk fir d Layer, wu vu dr FixedZoomLevels abstamme. Ass dää wfs-Layer minZoomLevel prieft, scih e Relikt us dr Vergangeheit. Mir chenne s aber nit ändere ohni OL_basierti Aawändige villicht kaputt gehn, wu dervu abhänge. Us däm Grund het die Funktion d Eigeschaft 'deprecated' iberchuu. D minZoomLevel-Priefig unte wird in dr Version 3.0 usegnuu. Bitte verwänd statt däm e min/max-Uflesig wie s do bschriben isch: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "WFS-Transaktion: ERFOLGRYCH ${response}", │ │ │ │ │ - commitFailed: "WFS-Transaktion: FÄHLGSCHLAA ${response}", │ │ │ │ │ - googleWarning: "Dr Google-Layer het nit korräkt chenne glade wäre.<br><br>Go die Mäldig nimi z kriege, wehl e andere Hintergrundlayer us em LayerSwitcher im rächte obere Ecke.<br><br>Dää Fähler git s seli hyfig, wel s Skript vu dr Google-Maps-Bibliothek nit yybunde woren isch oder wel s kei giltige API-Schlissel fir Dyy URL din het.<br><br>Entwickler: Fir Hilf zum korräkte Yybinde vum Google-Layer <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>doo drucke</a>", │ │ │ │ │ - getLayerWarning: "Dr ${layerType}-Layer het nit korräkt chenne glade wäre.<br><br>Go die Mäldig nimi z kriege, wehl e andere Hintergrundlayer us em LayerSwitcher im rächte obere Ecke.<br><br>Dää Fähler git s seli hyfig, wel s Skript vu dr '${layerLib}'-Bibliothek nit yybunde woren isch oder wel s kei giltige API-Schlissel fir Dyy URL din het.<br><br>Entwickler: Fir Hilf zum korräkte Yybinde vu Layer <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>doo drucke</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Maßstab = 1 : ${scaleDenom}", │ │ │ │ │ - W: "W", │ │ │ │ │ - E: "O", │ │ │ │ │ - N: "N", │ │ │ │ │ - S: "S", │ │ │ │ │ - reprojectDeprecated: "Du bruchsch d 'reproject'-Option bim ${layerName}-Layer. Die Option isch nimi giltig: si isch aagleit wore go Date iber kommerziälli Grundcharte lege, aber des sott mer jetz mache mit dr Unterstitzig vu Spherical Mercator. Meh Informatione git s uf http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Die Methode isch veraltet un wird us dr Version 3.0 usegnuu. Bitte verwäbnd statt däm ${newMethod}." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["br"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Distro evel reked anveret ${statusText}", │ │ │ │ │ - Permalink: "Peurliamm", │ │ │ │ │ - Overlays: "Gwiskadoù", │ │ │ │ │ - "Base Layer": "Gwiskad diazez", │ │ │ │ │ - noFID: "N'haller ket hizivaat un elfenn ma n'eus ket a niverenn-anaout (FID) eviti.", │ │ │ │ │ - browserNotSupported: "N'eo ket skoret an daskor vektorel gant ho merdeer. Setu aze an daskorerioù skoret evit ar poent :\n${renderers}", │ │ │ │ │ - minZoomLevelError: "Ne zleer implijout ar perzh minZoomLevel nemet evit gwiskadoù FixedZoomLevels-descendent. Ar fed ma wiria ar gwiskad WHS-se hag-eñ ez eus eus minZoomLevel zo un aspadenn gozh. Koulskoude n'omp ket evit e ziverkañ kuit da derriñ arloadoù diazezet war OL a c'hallfe bezañ stag outañ. Setu perak eo dispredet -- Lamet kuit e vo ar gwiriañ minZoomLevel a-is er stumm 3.0. Ober gant an arventennoù bihanañ/brasañ evel deskrivet amañ e plas : http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "Treuzgread WFS : MAT EO ${response}", │ │ │ │ │ - commitFailed: "Treuzgread WFS Transaction: C'HWITET ${response}", │ │ │ │ │ - googleWarning: "N'eus ket bet gallet kargañ ar gwiskad Google ent reizh.<br><br>Evit en em zizober eus ar c'hemenn-mañ, dibabit ur BaseLayer nevez en diuzer gwiskadoù er c'horn dehoù el laez.<br><br>Sur a-walc'h eo peogwir n'eo ket bet ensoc'het levraoueg Google Maps pe neuze ne glot ket an alc'hwez API gant ho lec'hienn.<br><br>Diorroerien : Evit reizhañ an dra-se, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>click here</a>", │ │ │ │ │ - getLayerWarning: "N'haller ket kargañ ar gwiskad ${layerType} ent reizh.<br><br>Evit en em zizober eus ar c'hemenn-mañ, dibabit ur BaseLayer nevez en diuzer gwiskadoù er c'horn dehoù el laez.<br><br>Sur a-walc'h eo peogwir n'eo ket bet ensoc'het mat al levraoueg ${layerLib}.<br><br>Diorroerien : Evit gouzout penaos reizhañ an dra-se, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>click here</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Skeul = 1 : ${scaleDenom}", │ │ │ │ │ - W: "K", │ │ │ │ │ - E: "R", │ │ │ │ │ - N: "N", │ │ │ │ │ - S: "S", │ │ │ │ │ - reprojectDeprecated: "Emaoc'h oc'h implijout an dibarzh 'reproject' war ar gwiskad ${layerName}. Dispredet eo an dibarzh-mañ : bet eo hag e talveze da ziskwel roadennoù war-c'horre kartennoù diazez kenwerzhel, un dra hag a c'haller ober bremañ gant an arc'hwel dre skor banndres boullek Mercator. Muioc'h a ditouroù a c'haller da gaout war http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Dispredet eo an daore-se ha tennet e vo kuit eus ar stumm 3.0. Grit gant ${newMethod} e plas." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["da-DK"] = { │ │ │ │ │ - unhandledRequest: "En ikke håndteret forespørgsel returnerede ${statusText}", │ │ │ │ │ - Permalink: "Permalink", │ │ │ │ │ - Overlays: "Kortlag", │ │ │ │ │ - "Base Layer": "Baggrundslag", │ │ │ │ │ - noFID: "Kan ikke opdateret en feature (et objekt) der ikke har et FID.", │ │ │ │ │ - browserNotSupported: "Din browser understøtter ikke vektor visning. Følgende vektor visninger understøttes:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "Egenskaben minZoomLevel er kun beregnet til brug " + "med FixedZoomLevels. At dette WFS lag kontrollerer " + "minZoomLevel egenskaben, er et levn fra en tidligere " + "version. Vi kan desværre ikke fjerne dette uden at risikere " + "at ødelægge eksisterende OL baserede programmer der " + " benytter denne funktionalitet. " + "Egenskaben bør derfor ikke anvendes, og minZoomLevel " + "kontrollen herunder vil blive fjernet i version 3.0. " + "Benyt istedet min/max opløsnings indstillingerne, som " + "er beskrevet her: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "WFS transaktion: LYKKEDES ${response}", │ │ │ │ │ - commitFailed: "WFS transaktion: MISLYKKEDES ${response}", │ │ │ │ │ - googleWarning: "Google laget kunne ikke indlæses.<br><br>" + "For at fjerne denne besked, vælg et nyt bagrundskort i " + "lagskifteren i øverste højre hjørne.<br><br>" + "Fejlen skyldes formentlig at Google Maps bibliotekts " + "scriptet ikke er inkluderet, eller ikke indeholder den " + "korrkte API nøgle for dit site.<br><br>" + "Udviklere: For hjælp til at få dette til at fungere, " + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>klik her</a>", │ │ │ │ │ - getLayerWarning: "${layerType}-laget kunne ikke indlæses.<br><br>" + "For at fjerne denne besked, vælg et nyt bagrundskort i " + "lagskifteren i øverste højre hjørne.<br><br>" + "Fejlen skyldes formentlig at ${layerLib} bibliotekts " + "scriptet ikke er inkluderet.<br><br>" + "Udviklere: For hjælp til at få dette til at fungere, " + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>klik her</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Målforhold = 1 : ${scaleDenom}", │ │ │ │ │ - reprojectDeprecated: "Du anvender indstillingen 'reproject' på laget ${layerName}." + "Denne indstilling bør ikke længere anvendes. Den var beregnet " + "til at vise data ovenpå kommercielle grundkort, men den funktionalitet " + "bør nu opnås ved at anvende Spherical Mercator understøttelsen. " + "Mere information er tilgængelig her: " + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Denne funktion bør ikke længere anvendes, og vil blive fjernet i version 3.0. " + "Anvend venligst funktionen ${newMethod} istedet." │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Lang["km"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Permalink: "តំណភ្ជាប់អចិន្ត្រៃយ៍", │ │ │ │ │ - "Base Layer": "ស្រទាប់បាត​", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "មាត្រដ្ឋាន = ១ ៖ ${scaleDenom}" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["nds"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Unbehannelt Trüchmellels för de Anfraag ${statusText}", │ │ │ │ │ - Permalink: "Permalink", │ │ │ │ │ - Overlays: "Overlays", │ │ │ │ │ - "Base Layer": "Achtergrundkoort", │ │ │ │ │ - noFID: "En Feature, dat keen FID hett, kann nich aktuell maakt warrn.", │ │ │ │ │ - browserNotSupported: "Dien Browser ünnerstütt keen Vektorbiller. Ünnerstütt Renderers:\n${renderers}", │ │ │ │ │ - commitSuccess: "WFS-Transakschoon: hett klappt ${response}", │ │ │ │ │ - commitFailed: "WFS-Transakschoon: hett nich klappt ${response}", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Skaal = 1 : ${scaleDenom}", │ │ │ │ │ - methodDeprecated: "Disse Methood is oold un schall dat in 3.0 nich mehr geven. Bruuk dor man beter ${newMethod} för." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["zh-TW"] = { │ │ │ │ │ - unhandledRequest: "未處理的請求,傳回值為 ${statusText}。", │ │ │ │ │ - Permalink: "永久連結", │ │ │ │ │ - Overlays: "額外圖層", │ │ │ │ │ - "Base Layer": "基礎圖層", │ │ │ │ │ - noFID: "因為沒有 FID 所以無法更新 feature。", │ │ │ │ │ - browserNotSupported: "您的瀏覽器未支援向量渲染. 目前支援的渲染方式是:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "minZoomLevel 屬性僅適合用在 " + "FixedZoomLevels-descendent 類型的圖層. 這個" + "wfs layer 的 minZoomLevel 是過去所遺留下來的," + "然而我們不能移除它而不讓它將" + "過去的程式相容性給破壞掉。" + "因此我們將會迴避使用它 -- minZoomLevel " + "會在3.0被移除,請改" + "用在這邊描述的 min/max resolution 設定: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "WFS Transaction: 成功 ${response}", │ │ │ │ │ - commitFailed: "WFS Transaction: 失敗 ${response}", │ │ │ │ │ - googleWarning: "The Google Layer 圖層無法被正確的載入。<br><br>" + "要迴避這個訊息, 請在右上角的圖層改變器裡," + "選一個新的基礎圖層。<br><br>" + "很有可能是因為 Google Maps 的函式庫" + "腳本沒有被正確的置入,或沒有包含 " + "您網站上正確的 API key <br><br>" + "開發者: 要幫助這個行為正確完成," + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>請按這裡</a>", │ │ │ │ │ - getLayerWarning: "${layerType} 圖層無法被正確的載入。<br><br>" + "要迴避這個訊息, 請在右上角的圖層改變器裡," + "選一個新的基礎圖層。<br><br>" + "很有可能是因為 ${layerLib} 的函式庫" + "腳本沒有被正確的置入。<br><br>" + "開發者: 要幫助這個行為正確完成," + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>請按這裡</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Scale = 1 : ${scaleDenom}", │ │ │ │ │ - reprojectDeprecated: "你正使用 'reproject' 這個選項 " + "在 ${layerName} 層。這個選項已經不再使用:" + "它的使用原本是設計用來支援在商業地圖上秀出資料," + "但這個功能已經被" + "Spherical Mercator所取代。更多的資訊可以在 " + "http://trac.openlayers.org/wiki/SphericalMercator 找到。", │ │ │ │ │ - methodDeprecated: "這個方法已經不再使用且在3.0將會被移除," + "請使用 ${newMethod} 來代替。", │ │ │ │ │ - end: "" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Lang["ja"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "未処理の要求は ${statusText} を返します", │ │ │ │ │ - Permalink: "パーマリンク", │ │ │ │ │ - Overlays: "オーバーレイ", │ │ │ │ │ - "Base Layer": "基底レイヤー", │ │ │ │ │ - noFID: "FID のない地物は更新できません。", │ │ │ │ │ - browserNotSupported: "あなたのブラウザはベクターグラフィックスの描写に対応していません。現時点で対応しているソフトウェアは以下のものです。\n${renderers}", │ │ │ │ │ - minZoomLevelError: "minZoomLevel プロパティは FixedZoomLevels を継承するレイヤーでの使用のみを想定しています。この minZoomLevel に対する WFS レイヤーの検査は歴史的なものです。しかしながら、この検査を除去するとそれに依存する OpenLayers ベースのアプリケーションを破壊してしまう可能性があります。よって廃止が予定されており、この minZoomLevel 検査はバージョン3.0で除去されます。代わりに、http://trac.openlayers.org/wiki/SettingZoomLevels で解説されている、最小および最大解像度設定を使用してください。", │ │ │ │ │ - commitSuccess: "WFS トランザクション: 成功 ${response}", │ │ │ │ │ - commitFailed: "WFS トランザクション: 失敗 ${response}", │ │ │ │ │ - googleWarning: "Google レイヤーが正しく読み込みを行えませんでした。<br><br>このメッセージを消すには、右上の隅にあるレイヤー切り替え部分で新しい基底レイヤーを選んでください。<br><br>おそらく、これは Google マップ用ライブラリのスクリプトが組み込まれていないか、あなたのサイトに対応する正しい API キーが設定されていないためです。<br><br>開発者の方へ: 正しい動作をさせるために<a href='http://trac.openlayers.org/wiki/Google' target='_blank'>こちらのウィキ</a>を参照してください。", │ │ │ │ │ - getLayerWarning: "${layerType} レイヤーが正しく読み込みを行えませんでした。<br><br>このメッセージを消すには、右上の隅にあるレイヤー切り替え部分で新しい基底レイヤーを選んでください。<br><br>おそらく、これは ${layerLib} ライブラリのスクリプトが正しく組み込まれていないためです。<br><br>開発者の方へ: 正しい動作をさせるために<a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>こちらのウィキ</a>を参照してください。", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "縮尺 = 1 : ${scaleDenom}", │ │ │ │ │ - W: "西", │ │ │ │ │ - E: "東", │ │ │ │ │ - N: "北", │ │ │ │ │ - S: "南", │ │ │ │ │ - reprojectDeprecated: "あなたは「${layerName}」レイヤーで reproject オプションを使っています。このオプションは商用の基底地図上に情報を表示する目的で設計されましたが、現在ではその機能は Spherical Mercator サポートを利用して実現されており、このオプションの使用は非推奨です。追加の情報は http://trac.openlayers.org/wiki/SphericalMercator で入手できます。", │ │ │ │ │ - methodDeprecated: "このメソッドは廃止が予定されており、バージョン3.0で除去されます。代わりに ${newMethod} を使用してください。" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["bg"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Permalink: "Постоянна препратка", │ │ │ │ │ - "Base Layer": "Основен слой", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Мащаб = 1 : ${scaleDenom}", │ │ │ │ │ - methodDeprecated: "Този метод е остарял и ще бъде премахват в 3.0. Вместо него използвайте ${newMethod}." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["be-tarask"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Неапрацаваны вынік запыту ${statusText}", │ │ │ │ │ - Permalink: "Сталая спасылка", │ │ │ │ │ - Overlays: "Слаі", │ │ │ │ │ - "Base Layer": "Базавы слой", │ │ │ │ │ - noFID: "Немагчыма абнавіць магчымасьць, для якога не існуе FID.", │ │ │ │ │ - browserNotSupported: "Ваш браўзэр не падтрымлівае вэктарную графіку. У цяперашні момант падтрымліваюцца: ${renderers}", │ │ │ │ │ - minZoomLevelError: "Уласьцівасьць minZoomLevel прызначана толькі для выкарыстаньня са слаямі вытворнымі ад FixedZoomLevels. Тое, што гэты wfs-слой правяраецца на minZoomLevel — рэха прошлага. Але мы ня можам выдаліць гэтую магчымасьць, таму што ад яе залежаць некаторыя заснаваныя на OL дастасаваньні. Тым ня менш, праверка minZoomLevel будзе выдаленая ў вэрсіі 3.0. Калі ласка, выкарыстоўваеце замест яе ўстаноўкі мінімальнага/максымальнага памераў, як апісана тут: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "WFS-транзакцыя: ПОСЬПЕХ ${response}", │ │ │ │ │ - commitFailed: "WFS-транзакцыя: ПАМЫЛКА ${response}", │ │ │ │ │ - googleWarning: "Не атрымалася загрузіць слой Google. <br><br>Каб пазбавіцца гэтага паведамленьня, выберыце новы базавы слой у сьпісе ў верхнім правым куце.<br><br> Хутчэй за ўсё, прычына ў тым, што скрыпт бібліятэкі Google Maps ня быў уключаныя альбо не ўтрымлівае слушны API-ключ для Вашага сайта.<br><br>Распрацоўшчыкам: Для таго, каб даведацца як зрабіць так, каб усё працавала, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>націсьніце тут</a>", │ │ │ │ │ - getLayerWarning: "Немагчыма загрузіць слой ${layerType}.<br><br>Каб пазбавіцца гэтага паведамленьня, выберыце новы базавы слой у сьпісе ў верхнім правым куце.<br><br>Хутчэй за ўсё, прычына ў тым, што скрыпт бібліятэкі ${layerLib} ня быў слушна ўключаны.<br><br>Распрацоўшчыкам: Для таго, каб даведацца як зрабіць так, каб усё працавала, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>націсьніце тут</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Маштаб = 1 : ${scaleDenom}", │ │ │ │ │ - W: "З", │ │ │ │ │ - E: "У", │ │ │ │ │ - N: "Пн", │ │ │ │ │ - S: "Пд", │ │ │ │ │ - reprojectDeprecated: "Вы выкарыстоўваеце ўстаноўку 'reproject' для слоя ${layerName}. Гэтая ўстаноўка зьяўляецца састарэлай: яна выкарыстоўвалася для падтрымкі паказу зьвестак на камэрцыйных базавых мапах, але гэта функцыя цяпер рэалізаваная ў убудаванай падтрымцы сфэрычнай праекцыі Мэркатара. Дадатковая інфармацыя ёсьць на http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Гэты мэтад састарэлы і будзе выдалены ў вэрсіі 3.0. Калі ласка, замест яго выкарыстоўвайце ${newMethod}." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang.ca = { │ │ │ │ │ - unhandledRequest: "Resposta a petició no gestionada ${statusText}", │ │ │ │ │ - Permalink: "Enllaç permanent", │ │ │ │ │ - Overlays: "Capes addicionals", │ │ │ │ │ - "Base Layer": "Capa Base", │ │ │ │ │ - noFID: "No es pot actualitzar un element per al que no existeix FID.", │ │ │ │ │ - browserNotSupported: "El seu navegador no suporta renderització vectorial. Els renderitzadors suportats actualment són:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "La propietat minZoomLevel s'ha d'utilitzar només " + "amb les capes que tenen FixedZoomLevels. El fet que " + "una capa wfs comprovi minZoomLevel és una relíquia del " + "passat. No podem, però, eliminar-la sense trencar " + "les aplicacions d'OpenLayers que en puguin dependre. " + "Així doncs estem fent-la obsoleta -- la comprovació " + "minZoomLevel s'eliminarà a la versió 3.0. Feu servir " + "els paràmetres min/max resolution en substitució, tal com es descriu aquí: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "Transacció WFS: CORRECTA ${response}", │ │ │ │ │ - commitFailed: "Transacció WFS: HA FALLAT ${response}", │ │ │ │ │ - googleWarning: "La capa Google no s'ha pogut carregar correctament.<br><br>" + "Per evitar aquest missatge, seleccioneu una nova Capa Base " + "al gestor de capes de la cantonada superior dreta.<br><br>" + "Probablement això és degut a que l'script de la biblioteca de " + "Google Maps no ha estat inclòs a la vostra pàgina, o no " + "conté la clau de l'API correcta per a la vostra adreça.<br><br>" + "Desenvolupadors: Per obtenir consells sobre com fer anar això, " + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>féu clic aquí</a>", │ │ │ │ │ - getLayerWarning: "Per evitar aquest missatge, seleccioneu una nova Capa Base " + "al gestor de capes de la cantonada superior dreta.<br><br>" + "Probablement això és degut a que l'script de la biblioteca " + "${layerLib} " + "no ha estat inclòs a la vostra pàgina.<br><br>" + "Desenvolupadors: Per obtenir consells sobre com fer anar això, " + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>féu clic aquí</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Escala = 1 : ${scaleDenom}", │ │ │ │ │ - W: "O", │ │ │ │ │ - E: "E", │ │ │ │ │ - N: "N", │ │ │ │ │ - S: "S", │ │ │ │ │ - Graticule: "Retícula", │ │ │ │ │ - reprojectDeprecated: "Esteu fent servir l'opció 'reproject' a la capa " + "${layerName}. Aquesta opció és obsoleta: el seu ús fou concebut " + "per suportar la visualització de dades sobre mapes base comercials, " + "però ara aquesta funcionalitat s'hauria d'assolir mitjançant el suport " + "de la projecció Spherical Mercator. Més informació disponible a " + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Aquest mètode és obsolet i s'eliminarà a la versió 3.0. " + "Si us plau feu servir em mètode alternatiu ${newMethod}.", │ │ │ │ │ - end: "" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Lang["ksh"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Met dä Antwoot op en Aanfrooch ham_mer nix aanjefange: ${statusText}", │ │ │ │ │ - Permalink: "Lengk op Duuer", │ │ │ │ │ - Overlays: "Drövver jelaat", │ │ │ │ │ - "Base Layer": "Jrund-Nivoh", │ │ │ │ │ - noFID: 'En Saach, woh kein <i lang="en">FID</i> för doh es, löht sesch nit ändere.', │ │ │ │ │ - browserNotSupported: "Dinge Brauser kann kein Väktore ußjävve. De Zoote Ußjaabe, di em Momang jon, sen:\n${renderers}", │ │ │ │ │ - minZoomLevelError: 'De Eijeschaff „<code lang="en">minZoomLevel</code>“ es bloß doför jedaach, dat mer se met dä Nivvohß bruch, di vun <code lang="en">FixedZoomLevels</code> affhange don. Dat dat <i lang="en">WFS</i>-Nivvoh övverhoup de Eijeschaff „<code lang="en">minZoomLevel</code>“ pröhfe deiht, es noch övveresch vun fröhjer. Mer künne dat ävver jez nit fott lohße, oohne dat mer Jevaa loufe, dat Aanwendunge vun OpenLayers nit mieh loufe, di sesch doh velleijsch noch drop am verlohße sin. Dröm sare mer, dat mer et nit mieh han welle, un de „<code lang="en">minZoomLevel</code>“-Eijeschaff weed hee vun de Version 3.0 af nit mieh jeprööf wäde. Nemm doför de Enstellung för de hühßte un de kleinßte Oplöhsung, esu wi et en http://trac.openlayers.org/wiki/SettingZoomLevels opjeschrevve es.', │ │ │ │ │ - commitSuccess: 'Dä <i lang="en">WFS</i>-Vörjang es joot jeloufe: ${response}', │ │ │ │ │ - commitFailed: 'Dä <i lang="en">WFS</i>-Vörjang es scheif jejange: ${response}', │ │ │ │ │ - googleWarning: 'Dat Nivvoh <code lang="en">Google</code> kunnt nit reschtesch jelaade wääde.<br /><br />Öm hee di Nohreesch loß ze krijje, donn en ander Jrund-Nivvoh ußsöhke, rähß bovve en de Äk.<br /><br />Wascheinlesch es dat wiel dat <i lang="en">Google-Maps</i>-Skrepp entweeder nit reschtesch enjebonge wood, udder nit dä reschtejje <i lang="en">API</i>-Schlößel för Ding Web-ßait scheke deiht.<br /><br />För Projrammierer jidd_et Hölp do_drövver, <a href="http://trac.openlayers.org/wiki/Google" target="_blank">wi mer dat aan et Loufe brengk</a>.', │ │ │ │ │ - getLayerWarning: 'Dat Nivvoh <code>${layerType}</code> kunnt nit reschtesch jelaade wääde.<br /><br />Öm hee di Nohreesch loß ze krijje, donn en ander Jrund-Nivvoh ußsöhkre, rähß bovve en de Äk.<br /><br />Wascheinlesch es dat, wiel dat Skrepp <code>${layerLib}</code> nit reschtesch enjebonge wood.<br /><br />För Projrammierer jidd_Et Hölp do_drövver, <a href="http://trac.openlayers.org/wiki/${layerLib}" target="_blank">wi mer dat aan et Loufe brengk</a>.', │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Mohßshtaab = 1 : ${scaleDenom}", │ │ │ │ │ - W: "W", │ │ │ │ │ - E: "O", │ │ │ │ │ - N: "N", │ │ │ │ │ - S: "S", │ │ │ │ │ - reprojectDeprecated: "Do bruchs de Ußwahl <code>reproject</code> op däm Nivvoh <code>${layerName}</code>. Di Ußwahl es nit mieh jähn jesinn. Se wohr doför jedaach, öm Date op jeschääfsmäßesch eruß jejovve Kaate bovve drop ze moole, wat ävver enzwesche besser met dä Öngershtözung för de ßfääresche Mäkaator Beldscher jeiht. Doh kanns De mieh drövver fenge op dä Sigg: http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Hee di Metood es nim_mih aktoäll un et weed se en dä Version 3.0 nit mieh jävve. Nemm <code>${newMethod}</code> doföör." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["hr"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Nepodržani zahtjev ${statusText}", │ │ │ │ │ - Permalink: "Permalink", │ │ │ │ │ - Overlays: "Overlays", │ │ │ │ │ - "Base Layer": "Osnovna karta", │ │ │ │ │ - noFID: "Ne mogu ažurirati značajku za koju ne postoji FID.", │ │ │ │ │ - browserNotSupported: "Vaš preglednik ne podržava vektorsko renderiranje. Trenutno podržani rendereri su: ${renderers}", │ │ │ │ │ - commitSuccess: "WFS Transakcija: USPJEŠNA ${response}", │ │ │ │ │ - commitFailed: "WFS Transakcija: NEUSPJEŠNA ${response}", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Mjerilo = 1 : ${scaleDenom}", │ │ │ │ │ - methodDeprecated: "Ova metoda nije odobrena i biti će maknuta u 3.0. Koristite ${newMethod}." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["hu"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Nem kezelt kérés visszatérése ${statusText}", │ │ │ │ │ - Permalink: "Permalink", │ │ │ │ │ - Overlays: "Rávetítések", │ │ │ │ │ - "Base Layer": "Alapréteg", │ │ │ │ │ - noFID: "Nem frissíthető olyan jellemző, amely nem rendelkezik FID-del.", │ │ │ │ │ - browserNotSupported: "A böngészője nem támogatja a vektoros renderelést. A jelenleg támogatott renderelők:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "A minZoomLevel tulajdonságot csak a következővel való használatra szánták: FixedZoomLevels-leszármazott fóliák. Ez azt jelenti, hogy a minZoomLevel wfs fólia jelölőnégyzetei már a múlté. Mi azonban nem távolíthatjuk el annak a veszélye nélkül, hogy az esetlegesen ettől függő OL alapú alkalmazásokat tönkretennénk. Ezért ezt érvénytelenítjük -- a minZoomLevel az alul levő jelölőnégyzet a 3.0-s verzióból el lesz távolítva. Kérjük, helyette használja a min/max felbontás beállítást, amelyről az alábbi helyen talál leírást: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "WFS tranzakció: SIKERES ${response}", │ │ │ │ │ - commitFailed: "WFS tranzakció: SIKERTELEN ${response}", │ │ │ │ │ - googleWarning: "A Google fólia betöltése sikertelen.<br><br>Ahhoz, hogy ez az üzenet eltűnjön, válasszon egy új BaseLayer fóliát a jobb felső sarokban található fóliakapcsoló segítségével.<br><br>Nagy valószínűséggel ez azért van, mert a Google Maps könyvtár parancsfájlja nem található, vagy nem tartalmazza az Ön oldalához tartozó megfelelő API-kulcsot.<br><br>Fejlesztőknek: A helyes működtetésre vonatkozó segítség az alábbi helyen érhető el, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>kattintson ide</a>", │ │ │ │ │ - getLayerWarning: "A(z) ${layerType} fólia nem töltődött be helyesen.<br><br>Ahhoz, hogy ez az üzenet eltűnjön, válasszon egy új BaseLayer fóliát a jobb felső sarokban található fóliakapcsoló segítségével.<br><br>Nagy valószínűséggel ez azért van, mert a(z) ${layerLib} könyvtár parancsfájlja helytelen.<br><br>Fejlesztőknek: A helyes működtetésre vonatkozó segítség az alábbi helyen érhető el, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>kattintson ide</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Lépték = 1 : ${scaleDenom}", │ │ │ │ │ - W: "Ny", │ │ │ │ │ - E: "K", │ │ │ │ │ - N: "É", │ │ │ │ │ - S: "D", │ │ │ │ │ - reprojectDeprecated: "Ön a 'reproject' beállítást használja a(z) ${layerName} fólián. Ez a beállítás érvénytelen: használata az üzleti alaptérképek fölötti adatok megjelenítésének támogatására szolgált, de ezt a funkció ezentúl a Gömbi Mercator használatával érhető el. További információ az alábbi helyen érhető el: http://trac.openlayers.org/wiki/SphericalMercator", │ │ │ │ │ - methodDeprecated: "Ez a módszer érvénytelenítve lett és a 3.0-s verzióból el lesz távolítva. Használja a(z) ${newMethod} módszert helyette." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["te"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Permalink: "స్థిరలింకు", │ │ │ │ │ - W: "ప", │ │ │ │ │ - E: "తూ", │ │ │ │ │ - N: "ఉ", │ │ │ │ │ - S: "ద" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["ru"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Необработанный запрос вернул ${statusText}", │ │ │ │ │ - Permalink: "Постоянная ссылка", │ │ │ │ │ - Overlays: "Слои", │ │ │ │ │ - "Base Layer": "Основной слой", │ │ │ │ │ - noFID: "Невозможно обновить объект, для которого нет FID.", │ │ │ │ │ - browserNotSupported: "Ваш браузер не поддерживает векторную графику. На данный момент поддерживаются:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "Свойство minZoomLevel предназначено только для использования со слоями, являющимися потомками FixedZoomLevels. То, что этот WFS-слой проверяется на minZoomLevel — реликт прошлого. Однако мы не можем удалить эту функцию, так как, возможно, от неё зависят некоторые основанные на OpenLayers приложения. Функция объявлена устаревшей — проверка minZoomLevel будет удалена в 3.0. Пожалуйста, используйте вместо неё настройку мин/макс разрешения, описанную здесь: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "Транзакция WFS: УСПЕШНО ${response}", │ │ │ │ │ - commitFailed: "Транзакция WFS: ОШИБКА ${response}", │ │ │ │ │ - googleWarning: "Слой Google не удалось нормально загрузить.<br><br>Чтобы избавиться от этого сообщения, выбите другой основной слой в переключателе в правом верхнем углу.<br><br>Скорее всего, причина в том, что библиотека Google Maps не была включена или не содержит корректного API-ключа для вашего сайта.<br><br>Разработчикам: чтобы узнать, как сделать, чтобы всё заработало, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>щёлкните тут</a>", │ │ │ │ │ - getLayerWarning: "Слой ${layerType} не удалось нормально загрузить. <br><br>Чтобы избавиться от этого сообщения, выбите другой основной слой в переключателе в правом верхнем углу.<br><br>Скорее всего, причина в том, что библиотека ${layerLib} не была включена или была включена некорректно.<br><br>Разработчикам: чтобы узнать, как сделать, чтобы всё заработало, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>щёлкните тут</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Масштаб = 1 : ${scaleDenom}", │ │ │ │ │ - W: "З", │ │ │ │ │ - E: "В", │ │ │ │ │ - N: "С", │ │ │ │ │ - S: "Ю", │ │ │ │ │ - reprojectDeprecated: "Вы используете опцию 'reproject' для слоя ${layerName}. Эта опция является устаревшей: ее использование предполагалось для поддержки показа данных поверх коммерческих базовых карт, но теперь этот функционал несёт встроенная поддержка сферической проекции Меркатора. Больше сведений доступно на http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Этот метод считается устаревшим и будет удалён в версии 3.0. Пожалуйста, пользуйтесь ${newMethod}." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["gl"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Solicitude non xerada; a resposta foi: ${statusText}", │ │ │ │ │ - Permalink: "Ligazón permanente", │ │ │ │ │ - Overlays: "Capas superpostas", │ │ │ │ │ - "Base Layer": "Capa base", │ │ │ │ │ - noFID: "Non se pode actualizar a funcionalidade para a que non hai FID.", │ │ │ │ │ - browserNotSupported: "O seu navegador non soporta a renderización de vectores. Os renderizadores soportados actualmente son:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "A propiedade minZoomLevel é só para uso conxuntamente coas capas FixedZoomLevels-descendent. O feito de que esa capa wfs verifique o minZoomLevel é unha reliquia do pasado. Non podemos, con todo, eliminala sen a posibilidade de non romper as aplicacións baseadas en OL que poidan depender dela. Por iso a estamos deixando obsoleta (a comprobación minZoomLevel de embaixo será eliminada na versión 3.0). Por favor, no canto diso use o axuste de resolución mín/máx tal e como está descrito aquí: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "Transacción WFS: ÉXITO ${response}", │ │ │ │ │ - commitFailed: "Transacción WFS: FALLIDA ${response}", │ │ │ │ │ - googleWarning: "A capa do Google non puido cargarse correctamente.<br><br>Para evitar esta mensaxe, escolla unha nova capa base no seleccionador de capas na marxe superior dereita.<br><br>Probablemente, isto acontece porque a escritura da libraría do Google Maps ou ben non foi incluída ou ben non contén a clave API correcta para o seu sitio.<br><br>Desenvolvedores: para axudar a facer funcionar isto correctamente, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>premede aquí</a>", │ │ │ │ │ - getLayerWarning: "A capa ${layerType} foi incapaz de cargarse correctamente.<br><br>Para evitar esta mensaxe, escolla unha nova capa base no seleccionador de capas na marxe superior dereita.<br><br>Probablemente, isto acontece porque a escritura da libraría ${layerLib} non foi ben incluída.<br><br>Desenvolvedores: para axudar a facer funcionar isto correctamente, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>premede aquí</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Escala = 1 : ${scaleDenom}", │ │ │ │ │ - W: "O", │ │ │ │ │ - E: "L", │ │ │ │ │ - N: "N", │ │ │ │ │ - S: "S", │ │ │ │ │ - reprojectDeprecated: 'Está usando a opción "reproject" na capa ${layerName}. Esta opción está obsoleta: o seu uso foi deseñado para a visualización de datos sobre mapas base comerciais, pero esta funcionalidade debera agora ser obtida utilizando a proxección Spherical Mercator. Hai dispoñible máis información en http://trac.openlayers.org/wiki/SphericalMercator.', │ │ │ │ │ - methodDeprecated: "Este método está obsoleto e será eliminado na versión 3.0. Por favor, no canto deste use ${newMethod}." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["pt"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Servidor devolveu erro não contemplado ${statusText}", │ │ │ │ │ - Permalink: "Ligação permanente", │ │ │ │ │ - Overlays: "Sobreposições", │ │ │ │ │ - "Base Layer": "Camada Base", │ │ │ │ │ - noFID: "Não é possível atualizar um elemento para a qual não há FID.", │ │ │ │ │ - browserNotSupported: "O seu navegador não suporta renderização vetorial. Actualmente os renderizadores suportados são:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "A propriedade minZoomLevel só deve ser usada com as camadas descendentes da FixedZoomLevels. A verificação da propriedade por esta camada wfs é uma relíquia do passado. No entanto, não podemos removê-la sem correr o risco de afectar aplicações OL que dependam dela. Portanto, estamos a torná-la obsoleta -- a verificação minZoomLevel será removida na versão 3.0. Em vez dela, por favor, use as opções de resolução min/max descritas aqui: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "Transacção WFS: SUCESSO ${response}", │ │ │ │ │ - commitFailed: "Transacção WFS: FALHOU ${response}", │ │ │ │ │ - googleWarning: "A Camada Google não foi correctamente carregada.<br><br>Para deixar de receber esta mensagem, seleccione uma nova Camada-Base no ''switcher'' de camadas no canto superior direito.<br><br>Provavelmente, isto acontece porque o ''script'' da biblioteca do Google Maps não foi incluído ou não contém a chave API correcta para o seu sítio.<br><br>Programadores: Para ajuda sobre como solucionar o problema <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>clique aqui</a> .", │ │ │ │ │ - getLayerWarning: "A camada ${layerType} não foi correctamente carregada.<br><br>Para desactivar esta mensagem, seleccione uma nova Camada-Base no ''switcher'' de camadas no canto superior direito.<br><br>Provavelmente, isto acontece porque o ''script'' da biblioteca ${layerLib} não foi incluído correctamente.<br><br>Programadores: Para ajuda sobre como solucionar o problema <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>clique aqui</a> .", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Escala = 1 : ${scaleDenom}", │ │ │ │ │ - W: "O", │ │ │ │ │ - E: "E", │ │ │ │ │ - N: "N", │ │ │ │ │ - S: "S", │ │ │ │ │ - reprojectDeprecated: "Está usando a opção 'reproject' na camada ${layerName}. Esta opção é obsoleta: foi concebida para permitir a apresentação de dados sobre mapas-base comerciais, mas esta funcionalidade é agora suportada pelo Mercator Esférico. Mais informação está disponível em http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Este método foi declarado obsoleto e será removido na versão 3.0. Por favor, use ${newMethod} em vez disso." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["ar"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Permalink: "وصلة دائمة", │ │ │ │ │ - "Base Layer": "الطبقة الاساسية", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "النسبة = 1 : ${scaleDenom}", │ │ │ │ │ - W: "غ", │ │ │ │ │ - E: "شر", │ │ │ │ │ - N: "شم", │ │ │ │ │ - S: "ج" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["ia"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Le responsa a un requesta non esseva maneate: ${statusText}", │ │ │ │ │ - Permalink: "Permaligamine", │ │ │ │ │ - Overlays: "Superpositiones", │ │ │ │ │ - "Base Layer": "Strato de base", │ │ │ │ │ - noFID: "Non pote actualisar un elemento sin FID.", │ │ │ │ │ - browserNotSupported: "Tu navigator non supporta le rendition de vectores. Le renditores actualmente supportate es:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "Le proprietate minZoomLevel es solmente pro uso con le stratos descendente de FixedZoomLevels. Le facto que iste strato WFS verifica minZoomLevel es un reliquia del passato. Nonobstante, si nos lo remove immediatemente, nos pote rumper applicationes a base de OL que depende de illo. Ergo nos lo declara obsolete; le verification de minZoomLevel in basso essera removite in version 3.0. Per favor usa in su loco le configuration de resolutiones min/max como describite a: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "Transaction WFS: SUCCESSO ${response}", │ │ │ │ │ - commitFailed: "Transaction WFS: FALLEVA ${response}", │ │ │ │ │ - googleWarning: "Le strato Google non poteva esser cargate correctemente.<br><br>Pro disfacer te de iste message, selige un nove BaseLayer in le selector de strato in alto a dextra.<br><br>Multo probabilemente, isto es proque le script del libreria de Google Maps non esseva includite o non contine le clave API correcte pro tu sito.<br><br>Disveloppatores: Pro adjuta de corriger isto, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>clicca hic</a", │ │ │ │ │ - getLayerWarning: "Le strato ${layerType} non poteva esser cargate correctemente.<br><br>Pro disfacer te de iste message, selige un nove BaseLayer in le selector de strato in alto a dextra.<br><br>Multo probabilemente, isto es proque le script del libreria de ${layerLib} non esseva correctemente includite.<br><br>Disveloppatores: Pro adjuta de corriger isto, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>clicca hic</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Scala = 1 : ${scaleDenom}", │ │ │ │ │ - W: "W", │ │ │ │ │ - E: "E", │ │ │ │ │ - N: "N", │ │ │ │ │ - S: "S", │ │ │ │ │ - reprojectDeprecated: "Tu usa le option 'reproject' in le strato ${layerName} layer. Iste option es obsolescente: illo esseva pro poter monstrar datos super cartas de base commercial, ma iste functionalitate pote ora esser attingite con le uso de Spherical Mercator. Ulterior information es disponibile a http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Iste methodo ha essite declarate obsolescente e essera removite in version 3.0. Per favor usa ${newMethod} in su loco." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["zh-CN"] = { │ │ │ │ │ - unhandledRequest: "未处理的请求,返回值为 ${statusText}", │ │ │ │ │ - Permalink: "永久链接", │ │ │ │ │ - Overlays: "叠加层", │ │ │ │ │ - "Base Layer": "基础图层", │ │ │ │ │ - noFID: "无法更新feature,缺少FID。", │ │ │ │ │ - browserNotSupported: "你使用的浏览器不支持矢量渲染。当前支持的渲染方式包括:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "minZoomLevel属性仅适合用于" + "使用了固定缩放级别的图层。这个 " + "wfs 图层检查 minZoomLevel 是过去遗留下来的。" + "然而,我们不能移除它," + "而破坏依赖于它的基于OL的应用程序。" + "因此,我们废除了它 -- minZoomLevel " + "将会在3.0中被移除。请改用 " + "min/max resolution 设置,参考:" + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "WFS Transaction: 成功。 ${response}", │ │ │ │ │ - commitFailed: "WFS Transaction: 失败。 ${response}", │ │ │ │ │ - googleWarning: "Google图层不能正确加载。<br><br>" + "要消除这个信息,请在右上角的" + "图层控制面板中选择其他的基础图层。<br><br>" + "这种情况很可能是没有正确的包含Google地图脚本库," + "或者是没有包含在你的站点上" + "使用的正确的Google Maps API密匙。<br><br>" + "开发者:获取使其正确工作的帮助信息," + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>点击这里</a>", │ │ │ │ │ - getLayerWarning: "${layerType} 图层不能正确加载。<br><br>" + "要消除这个信息,请在右上角的" + "图层控制面板中选择其他的基础图层。<br><br>" + "这种情况很可能是没有正确的包含" + "${layerLib} 脚本库。<br><br>" + "开发者:获取使其正确工作的帮助信息," + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>点击这里</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "比例尺 = 1 : ${scaleDenom}", │ │ │ │ │ - reprojectDeprecated: "你正在使用 ${layerName} 图层上的'reproject'选项。" + "这个选项已经不再使用:" + "它是被设计用来支持显示商业的地图数据," + "不过现在该功能可以通过使用Spherical Mercator来实现。" + "更多信息可以参阅" + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "该方法已经不再被支持,并且将在3.0中被移除。" + "请使用 ${newMethod} 方法来替代。", │ │ │ │ │ - end: "" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Lang["pt-BR"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "A requisição retornou um erro não tratado: ${statusText}", │ │ │ │ │ - Permalink: "Link para essa página", │ │ │ │ │ - Overlays: "Camadas de Sobreposição", │ │ │ │ │ - "Base Layer": "Camada Base", │ │ │ │ │ - noFID: "Não é possível atualizar uma feição que não tenha um FID.", │ │ │ │ │ - browserNotSupported: "Seu navegador não suporta renderização de vetores. Os renderizadores suportados atualmente são:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "A propriedade minZoomLevel é de uso restrito das camadas descendentes de FixedZoomLevels. A verificação dessa propriedade pelas camadas wfs é um resíduo do passado. Não podemos, entretanto não é possível removê-la sem possívelmente quebrar o funcionamento de aplicações OL que possuem depência com ela. Portanto estamos tornando seu uso obsoleto -- a verificação desse atributo será removida na versão 3.0. Ao invés, use as opções de resolução min/max como descrito em: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "Transação WFS : SUCESSO ${response}", │ │ │ │ │ - commitFailed: "Transação WFS : ERRO ${response}", │ │ │ │ │ - googleWarning: "Não foi possível carregar a camada Google corretamente.<br><br>Para se livrar dessa mensagem, selecione uma nova Camada Base, na ferramenta de alternação de camadas localização do canto superior direito.<br><br>Muito provavelmente, isso foi causado porque o script da biblioteca do Google Maps não foi incluído, ou porque ele não contém a chave correta da API para o seu site.<br><br>Desenvolvedores: Para obter ajuda em solucionar esse problema <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>cliquem aqui</a>", │ │ │ │ │ - getLayerWarning: "Não foi possível carregar a camada ${layerType} corretamente.<br><br>Para se livrar dessa mensagem, selecione uma nova Camada Base, na ferramenta de alternação de camadas localização do canto superior direito.<br><br>Muito provavelmente, isso foi causado porque o script da biblioteca ${layerLib} não foi incluído corretamente.<br><br>Desenvolvedores: Para obter ajuda em solucionar esse problema <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>cliquem aqui</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Escala = 1 : ${scaleDenom}", │ │ │ │ │ - W: "O", │ │ │ │ │ - E: "L", │ │ │ │ │ - N: "N", │ │ │ │ │ - S: "S", │ │ │ │ │ - reprojectDeprecated: "Você está usando a opção 'reproject' na camada ${layerName}. Essa opção está obsoleta: seu uso foi projetado para suportar a visualização de dados sobre bases de mapas comerciais, entretanto essa funcionalidade deve agora ser alcançada usando o suporte à projeção Mercator. Mais informação está disponível em: http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Esse método está obsoleto e será removido na versão 3.0. Ao invés, por favor use ${newMethod}." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["fi"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Permalink: "Ikilinkki", │ │ │ │ │ - Overlays: "Kerrokset", │ │ │ │ │ - "Base Layer": "Peruskerros", │ │ │ │ │ - W: "L", │ │ │ │ │ - E: "I", │ │ │ │ │ - N: "P", │ │ │ │ │ - S: "E" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["nl"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Het verzoek is niet afgehandeld met de volgende melding: ${statusText}", │ │ │ │ │ - Permalink: "Permanente verwijzing", │ │ │ │ │ - Overlays: "Overlays", │ │ │ │ │ - "Base Layer": "Achtergrondkaart", │ │ │ │ │ - noFID: "Een optie die geen FID heeft kan niet bijgewerkt worden.", │ │ │ │ │ - browserNotSupported: "Uw browser ondersteunt het weergeven van vectoren niet.\nMomenteel ondersteunde weergavemogelijkheden:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "De eigenschap minZoomLevel is alleen bedoeld voor gebruik lagen met die afstammen van FixedZoomLevels-lagen.\nDat deze WFS-laag minZoomLevel controleert, is een overblijfsel uit het verleden.\nWe kunnen deze controle echter niet verwijderen zonder op OL gebaseerde applicaties die hervan afhankelijk zijn stuk te maken.\nDaarom heeft deze functionaliteit de eigenschap 'deprecated' gekregen - de minZoomLevel wordt verwijderd in versie 3.0.\nGebruik in plaats van deze functie de mogelijkheid om min/max voor resolutie in te stellen zoals op de volgende pagina wordt beschreven:\nhttp://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "WFS-transactie: succesvol ${response}", │ │ │ │ │ - commitFailed: "WFS-transactie: mislukt ${response}", │ │ │ │ │ - googleWarning: "De Google-Layer kon niet correct geladen worden.<br /><br />\nOm deze melding niet meer te krijgen, moet u een andere achtergrondkaart kiezen in de laagwisselaar in de rechterbovenhoek.<br /><br />\nDit komt waarschijnlijk doordat de bibliotheek ${layerLib} niet correct ingevoegd is.<br /><br />\nOntwikkelaars: <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>klik hier</a> om dit werkend te krijgen.", │ │ │ │ │ - getLayerWarning: "De laag ${layerType} kon niet goed geladen worden.<br /><br />\nOm deze melding niet meer te krijgen, moet u een andere achtergrondkaart kiezen in de laagwisselaar in de rechterbovenhoek.<br /><br />\nDit komt waarschijnlijk doordat de bibliotheek ${layerLib} niet correct is ingevoegd.<br /><br />\nOntwikkelaars: <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>klik hier</a> om dit werkend te krijgen.", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Schaal = 1 : ${scaleDenom}", │ │ │ │ │ - W: "W", │ │ │ │ │ - E: "O", │ │ │ │ │ - N: "N", │ │ │ │ │ - S: "Z", │ │ │ │ │ - reprojectDeprecated: "U gebruikt de optie 'reproject' op de laag ${layerName}.\nDeze optie is vervallen: deze optie was ontwikkeld om gegevens over commerciële basiskaarten weer te geven, maar deze functionaliteit wordt nu bereikt door ondersteuning van Spherical Mercator.\nMeer informatie is beschikbaar op http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Deze methode is verouderd en wordt verwijderd in versie 3.0.\nGebruik ${newMethod}." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["fr"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Requête non gérée, retournant ${statusText}", │ │ │ │ │ - Permalink: "Permalien", │ │ │ │ │ - Overlays: "Calques", │ │ │ │ │ - "Base Layer": "Calque de base", │ │ │ │ │ - noFID: "Impossible de mettre à jour un objet sans identifiant (fid).", │ │ │ │ │ - browserNotSupported: "Votre navigateur ne supporte pas le rendu vectoriel. Les renderers actuellement supportés sont : \n${renderers}", │ │ │ │ │ - minZoomLevelError: "La propriété minZoomLevel doit seulement être utilisée pour des couches FixedZoomLevels-descendent. Le fait que cette couche WFS vérifie la présence de minZoomLevel est une relique du passé. Nous ne pouvons toutefois la supprimer sans casser des applications qui pourraient en dépendre. C'est pourquoi nous la déprécions -- la vérification du minZoomLevel sera supprimée en version 3.0. A la place, merci d'utiliser les paramètres de résolutions min/max tel que décrit sur : http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "Transaction WFS : SUCCES ${response}", │ │ │ │ │ - commitFailed: "Transaction WFS : ECHEC ${response}", │ │ │ │ │ - googleWarning: "La couche Google n'a pas été en mesure de se charger correctement.<br><br>Pour supprimer ce message, choisissez une nouvelle BaseLayer dans le sélecteur de couche en haut à droite.<br><br>Cela est possiblement causé par la non-inclusion de la librairie Google Maps, ou alors parce que la clé de l'API ne correspond pas à votre site.<br><br>Développeurs : pour savoir comment corriger ceci, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>cliquez ici</a>", │ │ │ │ │ - getLayerWarning: "La couche ${layerType} n'est pas en mesure de se charger correctement.<br><br>Pour supprimer ce message, choisissez une nouvelle BaseLayer dans le sélecteur de couche en haut à droite.<br><br>Cela est possiblement causé par la non-inclusion de la librairie ${layerLib}.<br><br>Développeurs : pour savoir comment corriger ceci, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>cliquez ici</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Echelle ~ 1 : ${scaleDenom}", │ │ │ │ │ - W: "O", │ │ │ │ │ - E: "E", │ │ │ │ │ - N: "N", │ │ │ │ │ - S: "S", │ │ │ │ │ - reprojectDeprecated: "Vous utilisez l'option 'reproject' sur la couche ${layerName}. Cette option est dépréciée : Son usage permettait d'afficher des données au dessus de couches raster commerciales.Cette fonctionalité est maintenant supportée en utilisant le support de la projection Mercator Sphérique. Plus d'information est disponible sur http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Cette méthode est dépréciée, et sera supprimée à la version 3.0. Merci d'utiliser ${newMethod} à la place." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["fur"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - Permalink: "Leam Permanent", │ │ │ │ │ - Overlays: "Livei parsore", │ │ │ │ │ - "Base Layer": "Livel di base", │ │ │ │ │ - browserNotSupported: "Il to sgarfadôr nol supuarte la renderizazion vetoriâl. Al moment a son supuartâts:\n${renderers}", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Scjale = 1 : ${scaleDenom}", │ │ │ │ │ - W: "O", │ │ │ │ │ - E: "E", │ │ │ │ │ - N: "N", │ │ │ │ │ - S: "S" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang["id"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Permintaan yang tak tertangani menghasilkan ${statusText}", │ │ │ │ │ - Permalink: "Pranala permanen", │ │ │ │ │ - Overlays: "Hamparan", │ │ │ │ │ - "Base Layer": "Lapisan Dasar", │ │ │ │ │ - noFID: "Tidak dapat memperbarui fitur yang tidak memiliki FID.", │ │ │ │ │ - browserNotSupported: "Peramban Anda tidak mendukung penggambaran vektor. Penggambar yang didukung saat ini adalah:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "Properti minZoomLevel hanya ditujukan bekerja dengan lapisan FixedZoomLevels-descendent. Pengecekan minZoomLevel oleh lapisan wfs adalah peninggalan masa lalu. Kami tidak dapat menghapusnya tanpa kemungkinan merusak aplikasi berbasis OL yang mungkin bergantung padanya. Karenanya, kami menganggapnya tidak berlaku -- Cek minZoomLevel di bawah ini akan dihapus pada 3.0. Silakan gunakan penyetelan resolusi min/maks seperti dijabarkan di sini: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "WFS Transaksi: BERHASIL ${respon}", │ │ │ │ │ - commitFailed: "WFS Transaksi: GAGAL ${respon}", │ │ │ │ │ - googleWarning: "Lapisan Google tidak dapat dimuat dengan benar.<br><br>Untuk menghilangkan pesan ini, pilih suatu BaseLayer baru melalui penukar lapisan (layer switcher) di ujung kanan atas.<br><br>Kemungkinan besar ini karena pustaka skrip Google Maps tidak disertakan atau tidak mengandung kunci API yang tepat untuk situs Anda.<br><br>Pengembang: Untuk bantuan mengatasi masalah ini, <a href='http://trac.openlayers.org/wiki/Google' target='_blank'>klik di sini</a>", │ │ │ │ │ - getLayerWarning: "Lapisan ${layerType} tidak dapat dimuat dengan benar.<br><br>Untuk menghilangkan pesan ini, pilih suatu BaseLayer baru melalui penukar lapisan (layer switcher) di ujung kanan atas.<br><br>Kemungkinan besar ini karena pustaka skrip Google Maps tidak disertakan dengan benar.<br><br>Pengembang: Untuk bantuan mengatasi masalah ini, <a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>klik di sini</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Sekala = 1 : ${scaleDenom}", │ │ │ │ │ - W: "B", │ │ │ │ │ - E: "T", │ │ │ │ │ - N: "U", │ │ │ │ │ - S: "S", │ │ │ │ │ - reprojectDeprecated: "Anda menggunakan opsi 'reproject' pada lapisan ${layerName}. Opsi ini telah ditinggalkan: penggunaannya dirancang untuk mendukung tampilan data melalui peta dasar komersial, tapi fungsionalitas tersebut saat ini harus dilakukan dengan menggunakan dukungan Spherical Mercator. Informasi lebih lanjut tersedia di http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Metode ini telah usang dan akan dihapus di 3.0. Sebaliknya, harap gunakan ${newMethod}." │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Lang.it = { │ │ │ │ │ - unhandledRequest: "Codice di ritorno della richiesta ${statusText}", │ │ │ │ │ - Permalink: "Permalink", │ │ │ │ │ - Overlays: "Overlays", │ │ │ │ │ - "Base Layer": "Livello base", │ │ │ │ │ - noFID: "Impossibile aggiornare un elemento grafico che non abbia il FID.", │ │ │ │ │ - browserNotSupported: "Il tuo browser non supporta il rendering vettoriale. I renderizzatore attualemnte supportati sono:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "La proprietà minZoomLevel è da utilizzare solamente " + "con livelli che abbiano FixedZoomLevels. Il fatto che " + "questo livello wfs controlli la proprietà minZoomLevel è " + "un retaggio del passato. Non possiamo comunque rimuoverla " + "senza rompere le vecchie applicazioni che dipendono su di essa." + "Quindi siamo costretti a deprecarla -- minZoomLevel " + "e sarà rimossa dalla vesione 3.0. Si prega di utilizzare i " + "settaggi di risoluzione min/max come descritto qui: " + "http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "Transazione WFS: SUCCESS ${response}", │ │ │ │ │ - commitFailed: "Transazione WFS: FAILED ${response}", │ │ │ │ │ - googleWarning: "Il livello Google non è riuscito a caricare correttamente.<br><br>" + "Per evitare questo messaggio, seleziona un nuovo BaseLayer " + "nel selettore di livelli nell'angolo in alto a destra.<br><br>" + "Più precisamente, ciò accade perchè la libreria Google Maps " + "non è stata inclusa nella pagina, oppure non contiene la " + "corretta API key per il tuo sito.<br><br>" + "Sviluppatori: Per aiuto su come farlo funzionare correttamente, " + "<a href='http://trac.openlayers.org/wiki/Google' " + "target='_blank'>clicca qui</a>", │ │ │ │ │ - getLayerWarning: "Il livello ${layerType} non è riuscito a caricare correttamente.<br><br>" + "Per evitare questo messaggio, seleziona un nuovo BaseLayer " + "nel selettore di livelli nell'angolo in alto a destra.<br><br>" + "Più precisamente, ciò accade perchè la libreria ${layerLib} " + "non è stata inclusa nella pagina.<br><br>" + "Sviluppatori: Per aiuto su come farlo funzionare correttamente, " + "<a href='http://trac.openlayers.org/wiki/${layerLib}' " + "target='_blank'>clicca qui</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Scala = 1 : ${scaleDenom}", │ │ │ │ │ - reprojectDeprecated: "Stai utilizzando l'opzione 'reproject' sul livello ${layerName}. " + "Questa opzione è deprecata: il suo utilizzo è stato introdotto per" + "supportare il disegno dei dati sopra mappe commerciali, ma tale " + "funzionalità dovrebbe essere ottenuta tramite l'utilizzo della proiezione " + "Spherical Mercator. Per maggiori informazioni consultare qui " + "http://trac.openlayers.org/wiki/SphericalMercator.", │ │ │ │ │ - methodDeprecated: "Questo metodo è stato deprecato e sarà rimosso dalla versione 3.0. " + "Si prega di utilizzare il metodo ${newMethod} in alternativa.", │ │ │ │ │ - end: "" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Lang["hsb"] = OpenLayers.Util.applyDefaults({ │ │ │ │ │ - unhandledRequest: "Wotmołwa njewobdźěłaneho naprašowanja ${statusText}", │ │ │ │ │ - Permalink: "Trajny wotkaz", │ │ │ │ │ - Overlays: "Naworštowanja", │ │ │ │ │ - "Base Layer": "Zakładna runina", │ │ │ │ │ - noFID: "Funkcija, za kotruž FID njeje, njeda so aktualizować.", │ │ │ │ │ - browserNotSupported: "Twój wobhladowak wektorowe rysowanje njepodpěruje. Tuchwilu podpěrowane rysowaki su:\n${renderers}", │ │ │ │ │ - minZoomLevelError: "Kajkosć minZoomLevel je jenož za wužiwanje z worštami myslena, kotrež wot FixedZoomLevels pochadźeja. Zo tuta woršta wfs za minZoomLevel přepruwuje, je relikt zańdźenosće. Njemóžemy wšak ju wotstronić, bjeztoho zo aplikacije, kotrež na OpenLayers bazěruja a snano tutu kajkosć wužiwaja, hižo njefunguja. Tohodla smy ju jako zestarjenu woznamjenili -- přepruwowanje za minZoomLevel budu so we wersiji 3.0 wotstronjeć. Prošu wužij město toho nastajenje min/max, kaž je tu wopisane: http://trac.openlayers.org/wiki/SettingZoomLevels", │ │ │ │ │ - commitSuccess: "WFS-Transakcija: WUSPĚŠNA ${response}", │ │ │ │ │ - commitFailed: "WFS-Transakcija: NJEPORADŹENA ${response}", │ │ │ │ │ - googleWarning: "Woršta Google njemóžeše so korektnje začitać.<br><br>Zo by tutu zdźělenku wotbył, wubjer nowy BaseLayer z wuběra worštow horjeka naprawo.<br><br>Najskerje so to stawa, dokelž skript biblioteki Google Maps pak njebu zapřijaty pak njewobsahuje korektny kluč API za twoje sydło.<br><br>Wuwiwarjo: Za pomoc ke korektnemu fungowanju worštow\n<a href='http://trac.openlayers.org/wiki/Google' target='_blank'>tu kliknyć</a>", │ │ │ │ │ - getLayerWarning: "Woršta ${layerType} njemóžeše so korektnje začitać.<br><br>Zo by tutu zdźělenku wotbył, wubjer nowy BaseLayer z wuběra worštow horjeka naprawo.<br><br>Najskerje so to stawa, dokelž skript biblioteki ${layerLib} njebu korektnje zapřijaty.<br><br>Wuwiwarjo: Za pomoc ke korektnemu fungowanju worštow\n<a href='http://trac.openlayers.org/wiki/${layerLib}' target='_blank'>tu kliknyć</a>", │ │ │ │ │ - "Scale = 1 : ${scaleDenom}": "Měritko = 1 : ${scaleDenom}", │ │ │ │ │ - W: "Z", │ │ │ │ │ - E: "W", │ │ │ │ │ - N: "S", │ │ │ │ │ - S: "J", │ │ │ │ │ - reprojectDeprecated: 'Wužiwaš opciju "reproject" wořšty ${layerName}. Tuta opcija je zestarjena: jeje wužiwanje bě myslene, zo by zwobraznjenje datow nad komercielnymi bazowymi kartami podpěrało, ale funkcionalnosć měła so nětko z pomocu Sperical Mercator docpěć. Dalše informacije steja na http://trac.openlayers.org/wiki/SphericalMercator k dispoziciji.', │ │ │ │ │ - methodDeprecated: "Tuta metoda je so njeschwaliła a budźe so w 3.0 wotstronjeć. Prošu wužij ${newMethod} město toho." │ │ │ │ │ + }, │ │ │ │ │ + mergeNewParams: function(newParams) { │ │ │ │ │ + var upperParams = OpenLayers.Util.upperCaseObject(newParams); │ │ │ │ │ + var newArguments = [upperParams]; │ │ │ │ │ + return OpenLayers.Layer.Grid.prototype.mergeNewParams.apply(this, newArguments) │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.ArcGIS93Rest" │ │ │ │ │ }); │ │ │ │ │ -OpenLayers.Protocol.SOS = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, OpenLayers.Protocol.SOS.DEFAULTS); │ │ │ │ │ - var cls = OpenLayers.Protocol.SOS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported SOS version: " + options.version │ │ │ │ │ - } │ │ │ │ │ - return new cls(options) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.SOS.DEFAULTS = { │ │ │ │ │ - version: "1.0.0" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.CSW = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, OpenLayers.Protocol.CSW.DEFAULTS); │ │ │ │ │ - var cls = OpenLayers.Protocol.CSW["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported CSW version: " + options.version │ │ │ │ │ - } │ │ │ │ │ - return new cls(options) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.CSW.DEFAULTS = { │ │ │ │ │ - version: "2.0.2" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.WFS = function(options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, OpenLayers.Protocol.WFS.DEFAULTS); │ │ │ │ │ - var cls = OpenLayers.Protocol.WFS["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ - if (!cls) { │ │ │ │ │ - throw "Unsupported WFS version: " + options.version │ │ │ │ │ - } │ │ │ │ │ - return new cls(options) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.WFS.fromWMSLayer = function(layer, options) { │ │ │ │ │ - var typeName, featurePrefix; │ │ │ │ │ - var param = layer.params["LAYERS"]; │ │ │ │ │ - var parts = (OpenLayers.Util.isArray(param) ? param[0] : param).split(":"); │ │ │ │ │ - if (parts.length > 1) { │ │ │ │ │ - featurePrefix = parts[0] │ │ │ │ │ - } │ │ │ │ │ - typeName = parts.pop(); │ │ │ │ │ - var protocolOptions = { │ │ │ │ │ - url: layer.url, │ │ │ │ │ - featureType: typeName, │ │ │ │ │ - featurePrefix: featurePrefix, │ │ │ │ │ - srsName: layer.projection && layer.projection.getCode() || layer.map && layer.map.getProjectionObject().getCode(), │ │ │ │ │ - version: "1.1.0" │ │ │ │ │ - }; │ │ │ │ │ - return new OpenLayers.Protocol.WFS(OpenLayers.Util.applyDefaults(options, protocolOptions)) │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.WFS.DEFAULTS = { │ │ │ │ │ - version: "1.0.0" │ │ │ │ │ -}; │ │ │ │ │ -OpenLayers.Protocol.HTTP = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ +OpenLayers.Layer.Image = OpenLayers.Class(OpenLayers.Layer, { │ │ │ │ │ + isBaseLayer: true, │ │ │ │ │ url: null, │ │ │ │ │ - headers: null, │ │ │ │ │ - params: null, │ │ │ │ │ - callback: null, │ │ │ │ │ - scope: null, │ │ │ │ │ - readWithPOST: false, │ │ │ │ │ - updateWithPOST: false, │ │ │ │ │ - deleteWithPOST: false, │ │ │ │ │ - wildcarded: false, │ │ │ │ │ - srsInBBOX: false, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - this.params = {}; │ │ │ │ │ - this.headers = {}; │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ - var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ - wildcarded: this.wildcarded, │ │ │ │ │ - srsInBBOX: this.srsInBBOX │ │ │ │ │ + extent: null, │ │ │ │ │ + size: null, │ │ │ │ │ + tile: null, │ │ │ │ │ + aspectRatio: null, │ │ │ │ │ + initialize: function(name, url, extent, size, options) { │ │ │ │ │ + this.url = url; │ │ │ │ │ + this.extent = extent; │ │ │ │ │ + this.maxExtent = extent; │ │ │ │ │ + this.size = size; │ │ │ │ │ + OpenLayers.Layer.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ + this.aspectRatio = this.extent.getHeight() / this.size.h / (this.extent.getWidth() / this.size.w) │ │ │ │ │ + }, │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.tile) { │ │ │ │ │ + this.removeTileMonitoringHooks(this.tile); │ │ │ │ │ + this.tile.destroy(); │ │ │ │ │ + this.tile = null │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.prototype.destroy.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.Image(this.name, this.url, this.extent, this.size, this.getOptions()) │ │ │ │ │ + } │ │ │ │ │ + obj = OpenLayers.Layer.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + setMap: function(map) { │ │ │ │ │ + if (this.options.maxResolution == null) { │ │ │ │ │ + this.options.maxResolution = this.aspectRatio * this.extent.getWidth() / this.size.w │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.prototype.setMap.apply(this, arguments) │ │ │ │ │ + }, │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + OpenLayers.Layer.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + var firstRendering = this.tile == null; │ │ │ │ │ + if (zoomChanged || firstRendering) { │ │ │ │ │ + this.setTileSize(); │ │ │ │ │ + var ulPx = this.map.getLayerPxFromLonLat({ │ │ │ │ │ + lon: this.extent.left, │ │ │ │ │ + lat: this.extent.top │ │ │ │ │ }); │ │ │ │ │ - this.filterToParams = function(filter, params) { │ │ │ │ │ - return format.write(filter, params) │ │ │ │ │ + if (firstRendering) { │ │ │ │ │ + this.tile = new OpenLayers.Tile.Image(this, ulPx, this.extent, null, this.tileSize); │ │ │ │ │ + this.addTileMonitoringHooks(this.tile) │ │ │ │ │ + } else { │ │ │ │ │ + this.tile.size = this.tileSize.clone(); │ │ │ │ │ + this.tile.position = ulPx.clone() │ │ │ │ │ } │ │ │ │ │ + this.tile.draw() │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.params = null; │ │ │ │ │ - this.headers = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ + setTileSize: function() { │ │ │ │ │ + var tileWidth = this.extent.getWidth() / this.map.getResolution(); │ │ │ │ │ + var tileHeight = this.extent.getHeight() / this.map.getResolution(); │ │ │ │ │ + this.tileSize = new OpenLayers.Size(tileWidth, tileHeight) │ │ │ │ │ }, │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ + addTileMonitoringHooks: function(tile) { │ │ │ │ │ + tile.onLoadStart = function() { │ │ │ │ │ + this.events.triggerEvent("loadstart") │ │ │ │ │ + }; │ │ │ │ │ + tile.events.register("loadstart", this, tile.onLoadStart); │ │ │ │ │ + tile.onLoadEnd = function() { │ │ │ │ │ + this.events.triggerEvent("loadend") │ │ │ │ │ + }; │ │ │ │ │ + tile.events.register("loadend", this, tile.onLoadEnd); │ │ │ │ │ + tile.events.register("unload", this, tile.onLoadEnd) │ │ │ │ │ + }, │ │ │ │ │ + removeTileMonitoringHooks: function(tile) { │ │ │ │ │ + tile.unload(); │ │ │ │ │ + tile.events.un({ │ │ │ │ │ + loadstart: tile.onLoadStart, │ │ │ │ │ + loadend: tile.onLoadEnd, │ │ │ │ │ + unload: tile.onLoadEnd, │ │ │ │ │ + scope: this │ │ │ │ │ + }) │ │ │ │ │ + }, │ │ │ │ │ + setUrl: function(newUrl) { │ │ │ │ │ + this.url = newUrl; │ │ │ │ │ + this.tile.draw() │ │ │ │ │ + }, │ │ │ │ │ + getURL: function(bounds) { │ │ │ │ │ + return this.url │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Image" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Google = OpenLayers.Class(OpenLayers.Layer.EventPane, OpenLayers.Layer.FixedZoomLevels, { │ │ │ │ │ + MIN_ZOOM_LEVEL: 0, │ │ │ │ │ + MAX_ZOOM_LEVEL: 21, │ │ │ │ │ + RESOLUTIONS: [1.40625, .703125, .3515625, .17578125, .087890625, .0439453125, .02197265625, .010986328125, .0054931640625, .00274658203125, .001373291015625, .0006866455078125, .00034332275390625, .000171661376953125, 858306884765625e-19, 4291534423828125e-20, 2145767211914062e-20, 1072883605957031e-20, 536441802978515e-20, 268220901489257e-20, 1341104507446289e-21, 6.705522537231445e-7], │ │ │ │ │ + type: null, │ │ │ │ │ + wrapDateLine: true, │ │ │ │ │ + sphericalMercator: false, │ │ │ │ │ + version: null, │ │ │ │ │ + initialize: function(name, options) { │ │ │ │ │ options = options || {}; │ │ │ │ │ - options.params = OpenLayers.Util.applyDefaults(options.params, this.options.params); │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - if (options.filter && this.filterToParams) { │ │ │ │ │ - options.params = this.filterToParams(options.filter, options.params) │ │ │ │ │ + if (!options.version) { │ │ │ │ │ + options.version = typeof GMap2 === "function" ? "2" : "3" │ │ │ │ │ } │ │ │ │ │ - var readWithPOST = options.readWithPOST !== undefined ? options.readWithPOST : this.readWithPOST; │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - if (readWithPOST) { │ │ │ │ │ - var headers = options.headers || {}; │ │ │ │ │ - headers["Content-Type"] = "application/x-www-form-urlencoded"; │ │ │ │ │ - resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ - data: OpenLayers.Util.getParameterString(options.params), │ │ │ │ │ - headers: headers │ │ │ │ │ - }) │ │ │ │ │ + var mixin = OpenLayers.Layer.Google["v" + options.version.replace(/\./g, "_")]; │ │ │ │ │ + if (mixin) { │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, mixin) │ │ │ │ │ } else { │ │ │ │ │ - resp.priv = OpenLayers.Request.GET({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, resp, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers │ │ │ │ │ - }) │ │ │ │ │ + throw "Unsupported Google Maps API version: " + options.version │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Util.applyDefaults(options, mixin.DEFAULTS); │ │ │ │ │ + if (options.maxExtent) { │ │ │ │ │ + options.maxExtent = options.maxExtent.clone() │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ + OpenLayers.Layer.FixedZoomLevels.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + OpenLayers.Util.extend(this, OpenLayers.Layer.SphericalMercator); │ │ │ │ │ + this.initMercatorParameters() │ │ │ │ │ } │ │ │ │ │ - return resp │ │ │ │ │ - }, │ │ │ │ │ - handleRead: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options) │ │ │ │ │ }, │ │ │ │ │ - create: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: features, │ │ │ │ │ - requestType: "create" │ │ │ │ │ - }); │ │ │ │ │ - resp.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleCreate, resp, options), │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(features) │ │ │ │ │ - }); │ │ │ │ │ - return resp │ │ │ │ │ + clone: function() { │ │ │ │ │ + return new OpenLayers.Layer.Google(this.name, this.getOptions()) │ │ │ │ │ }, │ │ │ │ │ - handleCreate: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options) │ │ │ │ │ + setVisibility: function(visible) { │ │ │ │ │ + var opacity = this.opacity == null ? 1 : this.opacity; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.setVisibility.apply(this, arguments); │ │ │ │ │ + this.setOpacity(opacity) │ │ │ │ │ }, │ │ │ │ │ - update: function(feature, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var url = options.url || feature.url || this.options.url + "/" + feature.fid; │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: feature, │ │ │ │ │ - requestType: "update" │ │ │ │ │ - }); │ │ │ │ │ - var method = this.updateWithPOST ? "POST" : "PUT"; │ │ │ │ │ - resp.priv = OpenLayers.Request[method]({ │ │ │ │ │ - url: url, │ │ │ │ │ - callback: this.createCallback(this.handleUpdate, resp, options), │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(feature) │ │ │ │ │ - }); │ │ │ │ │ - return resp │ │ │ │ │ + display: function(visible) { │ │ │ │ │ + if (!this._dragging) { │ │ │ │ │ + this.setGMapVisibility(visible) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.display.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - handleUpdate: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options) │ │ │ │ │ + moveTo: function(bounds, zoomChanged, dragging) { │ │ │ │ │ + this._dragging = dragging; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + delete this._dragging │ │ │ │ │ }, │ │ │ │ │ - delete: function(feature, options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - var url = options.url || feature.url || this.options.url + "/" + feature.fid; │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: feature, │ │ │ │ │ - requestType: "delete" │ │ │ │ │ - }); │ │ │ │ │ - var method = this.deleteWithPOST ? "POST" : "DELETE"; │ │ │ │ │ - var requestOptions = { │ │ │ │ │ - url: url, │ │ │ │ │ - callback: this.createCallback(this.handleDelete, resp, options), │ │ │ │ │ - headers: options.headers │ │ │ │ │ - }; │ │ │ │ │ - if (this.deleteWithPOST) { │ │ │ │ │ - requestOptions.data = this.format.write(feature) │ │ │ │ │ + setOpacity: function(opacity) { │ │ │ │ │ + if (opacity !== this.opacity) { │ │ │ │ │ + if (this.map != null) { │ │ │ │ │ + this.map.events.triggerEvent("changelayer", { │ │ │ │ │ + layer: this, │ │ │ │ │ + property: "opacity" │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + this.opacity = opacity │ │ │ │ │ + } │ │ │ │ │ + if (this.getVisibility()) { │ │ │ │ │ + var container = this.getMapContainer(); │ │ │ │ │ + OpenLayers.Util.modifyDOMElement(container, null, null, null, null, null, null, opacity) │ │ │ │ │ } │ │ │ │ │ - resp.priv = OpenLayers.Request[method](requestOptions); │ │ │ │ │ - return resp │ │ │ │ │ }, │ │ │ │ │ - handleDelete: function(resp, options) { │ │ │ │ │ - this.handleResponse(resp, options) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + if (this.map) { │ │ │ │ │ + this.setGMapVisibility(false); │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache && cache.count <= 1) { │ │ │ │ │ + this.removeGMapElements() │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - handleResponse: function(resp, options) { │ │ │ │ │ - var request = resp.priv; │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - if (resp.requestType != "delete") { │ │ │ │ │ - resp.features = this.parseFeatures(request) │ │ │ │ │ - } │ │ │ │ │ - resp.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ - } else { │ │ │ │ │ - resp.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ + removeGMapElements: function() { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + var container = this.mapObject && this.getMapContainer(); │ │ │ │ │ + if (container && container.parentNode) { │ │ │ │ │ + container.parentNode.removeChild(container) │ │ │ │ │ + } │ │ │ │ │ + var termsOfUse = cache.termsOfUse; │ │ │ │ │ + if (termsOfUse && termsOfUse.parentNode) { │ │ │ │ │ + termsOfUse.parentNode.removeChild(termsOfUse) │ │ │ │ │ + } │ │ │ │ │ + var poweredBy = cache.poweredBy; │ │ │ │ │ + if (poweredBy && poweredBy.parentNode) { │ │ │ │ │ + poweredBy.parentNode.removeChild(poweredBy) │ │ │ │ │ + } │ │ │ │ │ + if (this.mapObject && window.google && google.maps && google.maps.event && google.maps.event.clearListeners) { │ │ │ │ │ + google.maps.event.clearListeners(this.mapObject, "tilesloaded") │ │ │ │ │ } │ │ │ │ │ - options.callback.call(options.scope, resp) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - parseFeatures: function(request) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ + removeMap: function(map) { │ │ │ │ │ + if (this.visibility && this.mapObject) { │ │ │ │ │ + this.setGMapVisibility(false) │ │ │ │ │ } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + if (cache.count <= 1) { │ │ │ │ │ + this.removeGMapElements(); │ │ │ │ │ + delete OpenLayers.Layer.Google.cache[map.id] │ │ │ │ │ + } else { │ │ │ │ │ + --cache.count │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ - return this.format.read(doc) │ │ │ │ │ + delete this.termsOfUse; │ │ │ │ │ + delete this.poweredBy; │ │ │ │ │ + delete this.mapObject; │ │ │ │ │ + delete this.dragObject; │ │ │ │ │ + OpenLayers.Layer.EventPane.prototype.removeMap.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - commit: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var resp = [], │ │ │ │ │ - nResponses = 0; │ │ │ │ │ - var types = {}; │ │ │ │ │ - types[OpenLayers.State.INSERT] = []; │ │ │ │ │ - types[OpenLayers.State.UPDATE] = []; │ │ │ │ │ - types[OpenLayers.State.DELETE] = []; │ │ │ │ │ - var feature, list, requestFeatures = []; │ │ │ │ │ - for (var i = 0, len = features.length; i < len; ++i) { │ │ │ │ │ - feature = features[i]; │ │ │ │ │ - list = types[feature.state]; │ │ │ │ │ - if (list) { │ │ │ │ │ - list.push(feature); │ │ │ │ │ - requestFeatures.push(feature) │ │ │ │ │ + getOLBoundsFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + var olBounds = null; │ │ │ │ │ + if (moBounds != null) { │ │ │ │ │ + var sw = moBounds.getSouthWest(); │ │ │ │ │ + var ne = moBounds.getNorthEast(); │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + sw = this.forwardMercator(sw.lng(), sw.lat()); │ │ │ │ │ + ne = this.forwardMercator(ne.lng(), ne.lat()) │ │ │ │ │ + } else { │ │ │ │ │ + sw = new OpenLayers.LonLat(sw.lng(), sw.lat()); │ │ │ │ │ + ne = new OpenLayers.LonLat(ne.lng(), ne.lat()) │ │ │ │ │ } │ │ │ │ │ + olBounds = new OpenLayers.Bounds(sw.lon, sw.lat, ne.lon, ne.lat) │ │ │ │ │ } │ │ │ │ │ - var nRequests = (types[OpenLayers.State.INSERT].length > 0 ? 1 : 0) + types[OpenLayers.State.UPDATE].length + types[OpenLayers.State.DELETE].length; │ │ │ │ │ - var success = true; │ │ │ │ │ - var finalResponse = new OpenLayers.Protocol.Response({ │ │ │ │ │ - reqFeatures: requestFeatures │ │ │ │ │ - }); │ │ │ │ │ - │ │ │ │ │ - function insertCallback(response) { │ │ │ │ │ - var len = response.features ? response.features.length : 0; │ │ │ │ │ - var fids = new Array(len); │ │ │ │ │ - for (var i = 0; i < len; ++i) { │ │ │ │ │ - fids[i] = response.features[i].fid │ │ │ │ │ - } │ │ │ │ │ - finalResponse.insertIds = fids; │ │ │ │ │ - callback.apply(this, [response]) │ │ │ │ │ + return olBounds │ │ │ │ │ + }, │ │ │ │ │ + getWarningHTML: function() { │ │ │ │ │ + return OpenLayers.i18n("googleWarning") │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectCenter: function() { │ │ │ │ │ + return this.mapObject.getCenter() │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectZoom: function() { │ │ │ │ │ + return this.mapObject.getZoom() │ │ │ │ │ + }, │ │ │ │ │ + getLongitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + return this.sphericalMercator ? this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lon : moLonLat.lng() │ │ │ │ │ + }, │ │ │ │ │ + getLatitudeFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + var lat = this.sphericalMercator ? this.forwardMercator(moLonLat.lng(), moLonLat.lat()).lat : moLonLat.lat(); │ │ │ │ │ + return lat │ │ │ │ │ + }, │ │ │ │ │ + getXFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return moPixel.x │ │ │ │ │ + }, │ │ │ │ │ + getYFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return moPixel.y │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.Google" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Google.cache = {}; │ │ │ │ │ +OpenLayers.Layer.Google.v2 = { │ │ │ │ │ + termsOfUse: null, │ │ │ │ │ + poweredBy: null, │ │ │ │ │ + dragObject: null, │ │ │ │ │ + loadMapObject: function() { │ │ │ │ │ + if (!this.type) { │ │ │ │ │ + this.type = G_NORMAL_MAP │ │ │ │ │ } │ │ │ │ │ - │ │ │ │ │ - function callback(response) { │ │ │ │ │ - this.callUserCallback(response, options); │ │ │ │ │ - success = success && response.success(); │ │ │ │ │ - nResponses++; │ │ │ │ │ - if (nResponses >= nRequests) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - finalResponse.code = success ? OpenLayers.Protocol.Response.SUCCESS : OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - options.callback.apply(options.scope, [finalResponse]) │ │ │ │ │ - } │ │ │ │ │ + var mapObject, termsOfUse, poweredBy; │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + mapObject = cache.mapObject; │ │ │ │ │ + termsOfUse = cache.termsOfUse; │ │ │ │ │ + poweredBy = cache.poweredBy; │ │ │ │ │ + ++cache.count │ │ │ │ │ + } else { │ │ │ │ │ + var container = this.map.viewPortDiv; │ │ │ │ │ + var div = document.createElement("div"); │ │ │ │ │ + div.id = this.map.id + "_GMap2Container"; │ │ │ │ │ + div.style.position = "absolute"; │ │ │ │ │ + div.style.width = "100%"; │ │ │ │ │ + div.style.height = "100%"; │ │ │ │ │ + container.appendChild(div); │ │ │ │ │ + try { │ │ │ │ │ + mapObject = new GMap2(div); │ │ │ │ │ + termsOfUse = div.lastChild; │ │ │ │ │ + container.appendChild(termsOfUse); │ │ │ │ │ + termsOfUse.style.zIndex = "1100"; │ │ │ │ │ + termsOfUse.style.right = ""; │ │ │ │ │ + termsOfUse.style.bottom = ""; │ │ │ │ │ + termsOfUse.className = "olLayerGoogleCopyright"; │ │ │ │ │ + poweredBy = div.lastChild; │ │ │ │ │ + container.appendChild(poweredBy); │ │ │ │ │ + poweredBy.style.zIndex = "1100"; │ │ │ │ │ + poweredBy.style.right = ""; │ │ │ │ │ + poweredBy.style.bottom = ""; │ │ │ │ │ + poweredBy.className = "olLayerGooglePoweredBy gmnoprint" │ │ │ │ │ + } catch (e) { │ │ │ │ │ + throw e │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Layer.Google.cache[this.map.id] = { │ │ │ │ │ + mapObject: mapObject, │ │ │ │ │ + termsOfUse: termsOfUse, │ │ │ │ │ + poweredBy: poweredBy, │ │ │ │ │ + count: 1 │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - var queue = types[OpenLayers.State.INSERT]; │ │ │ │ │ - if (queue.length > 0) { │ │ │ │ │ - resp.push(this.create(queue, OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: insertCallback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options.create))) │ │ │ │ │ + this.mapObject = mapObject; │ │ │ │ │ + this.termsOfUse = termsOfUse; │ │ │ │ │ + this.poweredBy = poweredBy; │ │ │ │ │ + if (OpenLayers.Util.indexOf(this.mapObject.getMapTypes(), this.type) === -1) { │ │ │ │ │ + this.mapObject.addMapType(this.type) │ │ │ │ │ } │ │ │ │ │ - queue = types[OpenLayers.State.UPDATE]; │ │ │ │ │ - for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ - resp.push(this.update(queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: callback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options.update))) │ │ │ │ │ + if (typeof mapObject.getDragObject == "function") { │ │ │ │ │ + this.dragObject = mapObject.getDragObject() │ │ │ │ │ + } else { │ │ │ │ │ + this.dragPanMapObject = null │ │ │ │ │ } │ │ │ │ │ - queue = types[OpenLayers.State.DELETE]; │ │ │ │ │ - for (var i = queue.length - 1; i >= 0; --i) { │ │ │ │ │ - resp.push(this["delete"](queue[i], OpenLayers.Util.applyDefaults({ │ │ │ │ │ - callback: callback, │ │ │ │ │ - scope: this │ │ │ │ │ - }, options["delete"]))) │ │ │ │ │ + if (this.isBaseLayer === false) { │ │ │ │ │ + this.setGMapVisibility(this.div.style.display !== "none") │ │ │ │ │ } │ │ │ │ │ - return resp │ │ │ │ │ }, │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - response.priv.abort() │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + if (this.visibility && this.mapObject.isLoaded()) { │ │ │ │ │ + this.mapObject.checkResize() │ │ │ │ │ + } else { │ │ │ │ │ + if (!this._resized) { │ │ │ │ │ + var layer = this; │ │ │ │ │ + var handle = GEvent.addListener(this.mapObject, "load", function() { │ │ │ │ │ + GEvent.removeListener(handle); │ │ │ │ │ + delete layer._resized; │ │ │ │ │ + layer.mapObject.checkResize(); │ │ │ │ │ + layer.moveTo(layer.map.getCenter(), layer.map.getZoom()) │ │ │ │ │ + }) │ │ │ │ │ + } │ │ │ │ │ + this._resized = true │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - callUserCallback: function(resp, options) { │ │ │ │ │ - var opt = options[resp.requestType]; │ │ │ │ │ - if (opt && opt.callback) { │ │ │ │ │ - opt.callback.call(opt.scope, resp) │ │ │ │ │ + setGMapVisibility: function(visible) { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + var container = this.mapObject.getContainer(); │ │ │ │ │ + if (visible === true) { │ │ │ │ │ + this.mapObject.setMapType(this.type); │ │ │ │ │ + container.style.display = ""; │ │ │ │ │ + this.termsOfUse.style.left = ""; │ │ │ │ │ + this.termsOfUse.style.display = ""; │ │ │ │ │ + this.poweredBy.style.display = ""; │ │ │ │ │ + cache.displayed = this.id │ │ │ │ │ + } else { │ │ │ │ │ + if (cache.displayed === this.id) { │ │ │ │ │ + delete cache.displayed │ │ │ │ │ + } │ │ │ │ │ + if (!cache.displayed) { │ │ │ │ │ + container.style.display = "none"; │ │ │ │ │ + this.termsOfUse.style.display = "none"; │ │ │ │ │ + this.termsOfUse.style.left = "-9999px"; │ │ │ │ │ + this.poweredBy.style.display = "none" │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.HTTP" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Protocol.Script = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ - url: null, │ │ │ │ │ - params: null, │ │ │ │ │ - callback: null, │ │ │ │ │ - callbackTemplate: "OpenLayers.Protocol.Script.registry.${id}", │ │ │ │ │ - callbackKey: "callback", │ │ │ │ │ - callbackPrefix: "", │ │ │ │ │ - scope: null, │ │ │ │ │ - format: null, │ │ │ │ │ - pendingRequests: null, │ │ │ │ │ - srsInBBOX: false, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - options = options || {}; │ │ │ │ │ - this.params = {}; │ │ │ │ │ - this.pendingRequests = {}; │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (!this.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.GeoJSON │ │ │ │ │ - } │ │ │ │ │ - if (!this.filterToParams && OpenLayers.Format.QueryStringFilter) { │ │ │ │ │ - var format = new OpenLayers.Format.QueryStringFilter({ │ │ │ │ │ - srsInBBOX: this.srsInBBOX │ │ │ │ │ - }); │ │ │ │ │ - this.filterToParams = function(filter, params) { │ │ │ │ │ - return format.write(filter, params) │ │ │ │ │ - } │ │ │ │ │ + getMapContainer: function() { │ │ │ │ │ + return this.mapObject.getContainer() │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ + var moBounds = null; │ │ │ │ │ + if (olBounds != null) { │ │ │ │ │ + var sw = this.sphericalMercator ? this.inverseMercator(olBounds.bottom, olBounds.left) : new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ + var ne = this.sphericalMercator ? this.inverseMercator(olBounds.top, olBounds.right) : new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ + moBounds = new GLatLngBounds(new GLatLng(sw.lat, sw.lon), new GLatLng(ne.lat, ne.lon)) │ │ │ │ │ } │ │ │ │ │ + return moBounds │ │ │ │ │ }, │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ - options = OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - options.params = OpenLayers.Util.applyDefaults(options.params, this.options.params); │ │ │ │ │ - if (options.filter && this.filterToParams) { │ │ │ │ │ - options.params = this.filterToParams(options.filter, options.params) │ │ │ │ │ + setMapObjectCenter: function(center, zoom) { │ │ │ │ │ + this.mapObject.setCenter(center, zoom) │ │ │ │ │ + }, │ │ │ │ │ + dragPanMapObject: function(dX, dY) { │ │ │ │ │ + this.dragObject.moveBy(new GSize(-dX, dY)) │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + return this.mapObject.fromContainerPixelToLatLng(moPixel) │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + return this.mapObject.fromLatLngToContainerPixel(moLonLat) │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + return this.mapObject.getBoundsZoomLevel(moBounds) │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ + var gLatLng; │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ + gLatLng = new GLatLng(lonlat.lat, lonlat.lon) │ │ │ │ │ + } else { │ │ │ │ │ + gLatLng = new GLatLng(lat, lon) │ │ │ │ │ } │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - var request = this.createRequest(options.url, options.params, OpenLayers.Function.bind(function(data) { │ │ │ │ │ - response.data = data; │ │ │ │ │ - this.handleRead(response, options) │ │ │ │ │ - }, this)); │ │ │ │ │ - response.priv = request; │ │ │ │ │ - return response │ │ │ │ │ + return gLatLng │ │ │ │ │ }, │ │ │ │ │ - createRequest: function(url, params, callback) { │ │ │ │ │ - var id = OpenLayers.Protocol.Script.register(callback); │ │ │ │ │ - var name = OpenLayers.String.format(this.callbackTemplate, { │ │ │ │ │ - id: id │ │ │ │ │ - }); │ │ │ │ │ - params = OpenLayers.Util.extend({}, params); │ │ │ │ │ - params[this.callbackKey] = this.callbackPrefix + name; │ │ │ │ │ - url = OpenLayers.Util.urlAppend(url, OpenLayers.Util.getParameterString(params)); │ │ │ │ │ - var script = document.createElement("script"); │ │ │ │ │ - script.type = "text/javascript"; │ │ │ │ │ - script.src = url; │ │ │ │ │ - script.id = "OpenLayers_Protocol_Script_" + id; │ │ │ │ │ - this.pendingRequests[script.id] = script; │ │ │ │ │ - var head = document.getElementsByTagName("head")[0]; │ │ │ │ │ - head.appendChild(script); │ │ │ │ │ - return script │ │ │ │ │ + getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ + return new GPoint(x, y) │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Layer.GeoRSS = OpenLayers.Class(OpenLayers.Layer.Markers, { │ │ │ │ │ + location: null, │ │ │ │ │ + features: null, │ │ │ │ │ + formatOptions: null, │ │ │ │ │ + selectedFeature: null, │ │ │ │ │ + icon: null, │ │ │ │ │ + popupSize: null, │ │ │ │ │ + useFeedTitle: true, │ │ │ │ │ + initialize: function(name, location, options) { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.initialize.apply(this, [name, options]); │ │ │ │ │ + this.location = location; │ │ │ │ │ + this.features = [] │ │ │ │ │ }, │ │ │ │ │ - destroyRequest: function(script) { │ │ │ │ │ - OpenLayers.Protocol.Script.unregister(script.id.split("_").pop()); │ │ │ │ │ - delete this.pendingRequests[script.id]; │ │ │ │ │ - if (script.parentNode) { │ │ │ │ │ - script.parentNode.removeChild(script) │ │ │ │ │ + destroy: function() { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.destroy.apply(this, arguments); │ │ │ │ │ + this.clearFeatures(); │ │ │ │ │ + this.features = null │ │ │ │ │ + }, │ │ │ │ │ + loadRSS: function() { │ │ │ │ │ + if (!this.loaded) { │ │ │ │ │ + this.events.triggerEvent("loadstart"); │ │ │ │ │ + OpenLayers.Request.GET({ │ │ │ │ │ + url: this.location, │ │ │ │ │ + success: this.parseData, │ │ │ │ │ + scope: this │ │ │ │ │ + }); │ │ │ │ │ + this.loaded = true │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - this.handleResponse(response, options) │ │ │ │ │ + moveTo: function(bounds, zoomChanged, minor) { │ │ │ │ │ + OpenLayers.Layer.Markers.prototype.moveTo.apply(this, arguments); │ │ │ │ │ + if (this.visibility && !this.loaded) { │ │ │ │ │ + this.loadRSS() │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - handleResponse: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - if (response.data) { │ │ │ │ │ - response.features = this.parseFeatures(response.data); │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ + parseData: function(ajaxRequest) { │ │ │ │ │ + var doc = ajaxRequest.responseXML; │ │ │ │ │ + if (!doc || !doc.documentElement) { │ │ │ │ │ + doc = OpenLayers.Format.XML.prototype.read(ajaxRequest.responseText) │ │ │ │ │ + } │ │ │ │ │ + if (this.useFeedTitle) { │ │ │ │ │ + var name = null; │ │ │ │ │ + try { │ │ │ │ │ + name = doc.getElementsByTagNameNS("*", "title")[0].firstChild.nodeValue │ │ │ │ │ + } catch (e) { │ │ │ │ │ + try { │ │ │ │ │ + name = doc.getElementsByTagName("title")[0].firstChild.nodeValue │ │ │ │ │ + } catch (e) {} │ │ │ │ │ } │ │ │ │ │ - this.destroyRequest(response.priv); │ │ │ │ │ - options.callback.call(options.scope, response) │ │ │ │ │ + if (name) { │ │ │ │ │ + this.setName(name) │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + 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 format = new OpenLayers.Format.GeoRSS(options); │ │ │ │ │ + var features = format.read(doc); │ │ │ │ │ + for (var i = 0, len = features.length; i < len; i++) { │ │ │ │ │ + var data = {}; │ │ │ │ │ + var feature = features[i]; │ │ │ │ │ + if (!feature.geometry) { │ │ │ │ │ + continue │ │ │ │ │ + } │ │ │ │ │ + var title = feature.attributes.title ? feature.attributes.title : "Untitled"; │ │ │ │ │ + var description = feature.attributes.description ? feature.attributes.description : "No description."; │ │ │ │ │ + var link = feature.attributes.link ? feature.attributes.link : ""; │ │ │ │ │ + var location = feature.geometry.getBounds().getCenterLonLat(); │ │ │ │ │ + data.icon = this.icon == null ? OpenLayers.Marker.defaultIcon() : this.icon.clone(); │ │ │ │ │ + data.popupSize = this.popupSize ? this.popupSize.clone() : new OpenLayers.Size(250, 120); │ │ │ │ │ + if (title || description) { │ │ │ │ │ + data.title = title; │ │ │ │ │ + data.description = description; │ │ │ │ │ + var contentHTML = '<div class="olLayerGeoRSSClose">[x]</div>'; │ │ │ │ │ + contentHTML += '<div class="olLayerGeoRSSTitle">'; │ │ │ │ │ + if (link) { │ │ │ │ │ + contentHTML += '<a class="link" href="' + link + '" target="_blank">' │ │ │ │ │ + } │ │ │ │ │ + contentHTML += title; │ │ │ │ │ + if (link) { │ │ │ │ │ + contentHTML += "</a>" │ │ │ │ │ + } │ │ │ │ │ + contentHTML += "</div>"; │ │ │ │ │ + contentHTML += '<div style="" class="olLayerGeoRSSDescription">'; │ │ │ │ │ + contentHTML += description; │ │ │ │ │ + contentHTML += "</div>"; │ │ │ │ │ + data["popupContentHTML"] = contentHTML │ │ │ │ │ + } │ │ │ │ │ + var feature = new OpenLayers.Feature(this, location, data); │ │ │ │ │ + this.features.push(feature); │ │ │ │ │ + var marker = feature.createMarker(); │ │ │ │ │ + marker.events.register("click", feature, this.markerClick); │ │ │ │ │ + this.addMarker(marker) │ │ │ │ │ } │ │ │ │ │ + this.events.triggerEvent("loadend") │ │ │ │ │ }, │ │ │ │ │ - parseFeatures: function(data) { │ │ │ │ │ - return this.format.read(data) │ │ │ │ │ + markerClick: function(evt) { │ │ │ │ │ + var sameMarkerClicked = this == this.layer.selectedFeature; │ │ │ │ │ + this.layer.selectedFeature = !sameMarkerClicked ? this : null; │ │ │ │ │ + for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ + this.layer.map.removePopup(this.layer.map.popups[i]) │ │ │ │ │ + } │ │ │ │ │ + if (!sameMarkerClicked) { │ │ │ │ │ + var popup = this.createPopup(); │ │ │ │ │ + OpenLayers.Event.observe(popup.div, "click", OpenLayers.Function.bind(function() { │ │ │ │ │ + for (var i = 0, len = this.layer.map.popups.length; i < len; i++) { │ │ │ │ │ + this.layer.map.removePopup(this.layer.map.popups[i]) │ │ │ │ │ + } │ │ │ │ │ + }, this)); │ │ │ │ │ + this.layer.map.addPopup(popup) │ │ │ │ │ + } │ │ │ │ │ + OpenLayers.Event.stop(evt) │ │ │ │ │ }, │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - this.destroyRequest(response.priv) │ │ │ │ │ - } else { │ │ │ │ │ - for (var key in this.pendingRequests) { │ │ │ │ │ - this.destroyRequest(this.pendingRequests[key]) │ │ │ │ │ + clearFeatures: function() { │ │ │ │ │ + if (this.features != null) { │ │ │ │ │ + while (this.features.length > 0) { │ │ │ │ │ + var feature = this.features[0]; │ │ │ │ │ + OpenLayers.Util.removeItem(this.features, feature); │ │ │ │ │ + feature.destroy() │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - this.abort(); │ │ │ │ │ - delete this.params; │ │ │ │ │ - delete this.format; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.Script" │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.GeoRSS" │ │ │ │ │ }); │ │ │ │ │ -(function() { │ │ │ │ │ - var o = OpenLayers.Protocol.Script; │ │ │ │ │ - var counter = 0; │ │ │ │ │ - o.registry = {}; │ │ │ │ │ - o.register = function(callback) { │ │ │ │ │ - var id = "c" + ++counter; │ │ │ │ │ - o.registry[id] = function() { │ │ │ │ │ - callback.apply(this, arguments) │ │ │ │ │ - }; │ │ │ │ │ - return id │ │ │ │ │ - }; │ │ │ │ │ - o.unregister = function(id) { │ │ │ │ │ - delete o.registry[id] │ │ │ │ │ - } │ │ │ │ │ -})(); │ │ │ │ │ -OpenLayers.Protocol.WFS.v1 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ - version: null, │ │ │ │ │ - srsName: "EPSG:4326", │ │ │ │ │ - featureType: null, │ │ │ │ │ - featureNS: null, │ │ │ │ │ - geometryName: "the_geom", │ │ │ │ │ - schema: null, │ │ │ │ │ - featurePrefix: "feature", │ │ │ │ │ - formatOptions: null, │ │ │ │ │ - readFormat: null, │ │ │ │ │ - readOptions: null, │ │ │ │ │ +OpenLayers.Layer.UTFGrid = OpenLayers.Class(OpenLayers.Layer.XYZ, { │ │ │ │ │ + isBaseLayer: false, │ │ │ │ │ + projection: new OpenLayers.Projection("EPSG:900913"), │ │ │ │ │ + useJSONP: false, │ │ │ │ │ + tileClass: OpenLayers.Tile.UTFGrid, │ │ │ │ │ initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!options.format) { │ │ │ │ │ - this.format = OpenLayers.Format.WFST(OpenLayers.Util.extend({ │ │ │ │ │ - version: this.version, │ │ │ │ │ - featureType: this.featureType, │ │ │ │ │ - featureNS: this.featureNS, │ │ │ │ │ - featurePrefix: this.featurePrefix, │ │ │ │ │ - geometryName: this.geometryName, │ │ │ │ │ - srsName: this.srsName, │ │ │ │ │ - schema: this.schema │ │ │ │ │ - }, this.formatOptions)) │ │ │ │ │ + OpenLayers.Layer.Grid.prototype.initialize.apply(this, [options.name, options.url, {}, options]); │ │ │ │ │ + this.tileOptions = OpenLayers.Util.extend({ │ │ │ │ │ + utfgridResolution: this.utfgridResolution │ │ │ │ │ + }, this.tileOptions) │ │ │ │ │ + }, │ │ │ │ │ + createBackBuffer: function() {}, │ │ │ │ │ + clone: function(obj) { │ │ │ │ │ + if (obj == null) { │ │ │ │ │ + obj = new OpenLayers.Layer.UTFGrid(this.getOptions()) │ │ │ │ │ } │ │ │ │ │ - if (!options.geometryName && parseFloat(this.format.version) > 1) { │ │ │ │ │ - this.setGeometryName(null) │ │ │ │ │ + obj = OpenLayers.Layer.Grid.prototype.clone.apply(this, [obj]); │ │ │ │ │ + return obj │ │ │ │ │ + }, │ │ │ │ │ + getFeatureInfo: function(location) { │ │ │ │ │ + var info = null; │ │ │ │ │ + var tileInfo = this.getTileData(location); │ │ │ │ │ + if (tileInfo && tileInfo.tile) { │ │ │ │ │ + info = tileInfo.tile.getFeatureInfo(tileInfo.i, tileInfo.j) │ │ │ │ │ } │ │ │ │ │ + return info │ │ │ │ │ }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.options && !this.options.format) { │ │ │ │ │ - this.format.destroy() │ │ │ │ │ + getFeatureId: function(location) { │ │ │ │ │ + var id = null; │ │ │ │ │ + var info = this.getTileData(location); │ │ │ │ │ + if (info.tile) { │ │ │ │ │ + id = info.tile.getFeatureId(info.i, info.j) │ │ │ │ │ } │ │ │ │ │ - this.format = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ + return id │ │ │ │ │ }, │ │ │ │ │ - read: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.read.apply(this, arguments); │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - var data = OpenLayers.Format.XML.prototype.write.apply(this.format, [this.format.writeNode("wfs:GetFeature", options)]); │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: data │ │ │ │ │ - }); │ │ │ │ │ - return response │ │ │ │ │ + CLASS_NAME: "OpenLayers.Layer.UTFGrid" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Layer.Google.v3 = { │ │ │ │ │ + DEFAULTS: { │ │ │ │ │ + sphericalMercator: true, │ │ │ │ │ + projection: "EPSG:900913" │ │ │ │ │ }, │ │ │ │ │ - setFeatureType: function(featureType) { │ │ │ │ │ - this.featureType = featureType; │ │ │ │ │ - this.format.featureType = featureType │ │ │ │ │ + animationEnabled: true, │ │ │ │ │ + loadMapObject: function() { │ │ │ │ │ + if (!this.type) { │ │ │ │ │ + this.type = google.maps.MapTypeId.ROADMAP │ │ │ │ │ + } │ │ │ │ │ + var mapObject; │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + if (cache) { │ │ │ │ │ + mapObject = cache.mapObject; │ │ │ │ │ + ++cache.count │ │ │ │ │ + } else { │ │ │ │ │ + var center = this.map.getCenter(); │ │ │ │ │ + var container = document.createElement("div"); │ │ │ │ │ + container.className = "olForeignContainer"; │ │ │ │ │ + container.style.width = "100%"; │ │ │ │ │ + container.style.height = "100%"; │ │ │ │ │ + mapObject = new google.maps.Map(container, { │ │ │ │ │ + center: center ? new google.maps.LatLng(center.lat, center.lon) : new google.maps.LatLng(0, 0), │ │ │ │ │ + zoom: this.map.getZoom() || 0, │ │ │ │ │ + mapTypeId: this.type, │ │ │ │ │ + disableDefaultUI: true, │ │ │ │ │ + keyboardShortcuts: false, │ │ │ │ │ + draggable: false, │ │ │ │ │ + disableDoubleClickZoom: true, │ │ │ │ │ + scrollwheel: false, │ │ │ │ │ + streetViewControl: false │ │ │ │ │ + }); │ │ │ │ │ + var googleControl = document.createElement("div"); │ │ │ │ │ + googleControl.style.width = "100%"; │ │ │ │ │ + googleControl.style.height = "100%"; │ │ │ │ │ + mapObject.controls[google.maps.ControlPosition.TOP_LEFT].push(googleControl); │ │ │ │ │ + cache = { │ │ │ │ │ + googleControl: googleControl, │ │ │ │ │ + mapObject: mapObject, │ │ │ │ │ + count: 1 │ │ │ │ │ + }; │ │ │ │ │ + OpenLayers.Layer.Google.cache[this.map.id] = cache │ │ │ │ │ + } │ │ │ │ │ + this.mapObject = mapObject; │ │ │ │ │ + this.setGMapVisibility(this.visibility) │ │ │ │ │ }, │ │ │ │ │ - setGeometryName: function(geometryName) { │ │ │ │ │ - this.geometryName = geometryName; │ │ │ │ │ - this.format.geometryName = geometryName │ │ │ │ │ + onMapResize: function() { │ │ │ │ │ + if (this.visibility) { │ │ │ │ │ + google.maps.event.trigger(this.mapObject, "resize") │ │ │ │ │ + } │ │ │ │ │ }, │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - var result = this.parseResponse(request, options.readOptions); │ │ │ │ │ - if (result && result.success !== false) { │ │ │ │ │ - if (options.readOptions && options.readOptions.output == "object") { │ │ │ │ │ - OpenLayers.Util.extend(response, result) │ │ │ │ │ + setGMapVisibility: function(visible) { │ │ │ │ │ + var cache = OpenLayers.Layer.Google.cache[this.map.id]; │ │ │ │ │ + var map = this.map; │ │ │ │ │ + if (cache) { │ │ │ │ │ + var type = this.type; │ │ │ │ │ + var layers = map.layers; │ │ │ │ │ + var layer; │ │ │ │ │ + for (var i = layers.length - 1; i >= 0; --i) { │ │ │ │ │ + layer = layers[i]; │ │ │ │ │ + if (layer instanceof OpenLayers.Layer.Google && layer.visibility === true && layer.inRange === true) { │ │ │ │ │ + type = layer.type; │ │ │ │ │ + visible = true; │ │ │ │ │ + break │ │ │ │ │ + } │ │ │ │ │ + } │ │ │ │ │ + var container = this.mapObject.getDiv(); │ │ │ │ │ + if (visible === true) { │ │ │ │ │ + if (container.parentNode !== map.div) { │ │ │ │ │ + if (!cache.rendered) { │ │ │ │ │ + var me = this; │ │ │ │ │ + google.maps.event.addListenerOnce(this.mapObject, "tilesloaded", function() { │ │ │ │ │ + cache.rendered = true; │ │ │ │ │ + me.setGMapVisibility(me.getVisibility()); │ │ │ │ │ + me.moveTo(me.map.getCenter()) │ │ │ │ │ + }) │ │ │ │ │ } else { │ │ │ │ │ - response.features = result │ │ │ │ │ + map.div.appendChild(container); │ │ │ │ │ + cache.googleControl.appendChild(map.viewPortDiv); │ │ │ │ │ + google.maps.event.trigger(this.mapObject, "resize") │ │ │ │ │ } │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - response.error = result │ │ │ │ │ } │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ + this.mapObject.setMapTypeId(type) │ │ │ │ │ + } else if (cache.googleControl.hasChildNodes()) { │ │ │ │ │ + map.div.appendChild(map.viewPortDiv); │ │ │ │ │ + map.div.removeChild(container) │ │ │ │ │ } │ │ │ │ │ - options.callback.call(options.scope, response) │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - parseResponse: function(request, options) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null │ │ │ │ │ - } │ │ │ │ │ - var result = this.readFormat !== null ? this.readFormat.read(doc) : this.format.read(doc, options); │ │ │ │ │ - if (!this.featureNS) { │ │ │ │ │ - var format = this.readFormat || this.format; │ │ │ │ │ - this.featureNS = format.featureNS; │ │ │ │ │ - format.autoConfig = false; │ │ │ │ │ - if (!this.geometryName) { │ │ │ │ │ - this.setGeometryName(format.geometryName) │ │ │ │ │ - } │ │ │ │ │ - } │ │ │ │ │ - return result │ │ │ │ │ + getMapContainer: function() { │ │ │ │ │ + return this.mapObject.getDiv() │ │ │ │ │ }, │ │ │ │ │ - commit: function(features, options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "commit", │ │ │ │ │ - reqFeatures: features │ │ │ │ │ - }); │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: this.format.write(features, options), │ │ │ │ │ - callback: this.createCallback(this.handleCommit, response, options) │ │ │ │ │ - }); │ │ │ │ │ - return response │ │ │ │ │ + getMapObjectBoundsFromOLBounds: function(olBounds) { │ │ │ │ │ + var moBounds = null; │ │ │ │ │ + if (olBounds != null) { │ │ │ │ │ + var sw = this.sphericalMercator ? this.inverseMercator(olBounds.bottom, olBounds.left) : new OpenLayers.LonLat(olBounds.bottom, olBounds.left); │ │ │ │ │ + var ne = this.sphericalMercator ? this.inverseMercator(olBounds.top, olBounds.right) : new OpenLayers.LonLat(olBounds.top, olBounds.right); │ │ │ │ │ + moBounds = new google.maps.LatLngBounds(new google.maps.LatLng(sw.lat, sw.lon), new google.maps.LatLng(ne.lat, ne.lon)) │ │ │ │ │ + } │ │ │ │ │ + return moBounds │ │ │ │ │ }, │ │ │ │ │ - handleCommit: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - var data = request.responseXML; │ │ │ │ │ - if (!data || !data.documentElement) { │ │ │ │ │ - data = request.responseText │ │ │ │ │ - } │ │ │ │ │ - var obj = this.format.read(data) || {}; │ │ │ │ │ - response.insertIds = obj.insertIds || []; │ │ │ │ │ - if (obj.success) { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE; │ │ │ │ │ - response.error = obj │ │ │ │ │ - } │ │ │ │ │ - options.callback.call(options.scope, response) │ │ │ │ │ + getMapObjectLonLatFromMapObjectPixel: function(moPixel) { │ │ │ │ │ + var size = this.map.getSize(); │ │ │ │ │ + var lon = this.getLongitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ + var lat = this.getLatitudeFromMapObjectLonLat(this.mapObject.center); │ │ │ │ │ + var res = this.map.getResolution(); │ │ │ │ │ + var delta_x = moPixel.x - size.w / 2; │ │ │ │ │ + var delta_y = moPixel.y - size.h / 2; │ │ │ │ │ + var lonlat = new OpenLayers.LonLat(lon + delta_x * res, lat - delta_y * res); │ │ │ │ │ + if (this.wrapDateLine) { │ │ │ │ │ + lonlat = lonlat.wrapDateLine(this.maxExtent) │ │ │ │ │ } │ │ │ │ │ + return this.getMapObjectLonLatFromLonLat(lonlat.lon, lonlat.lat) │ │ │ │ │ }, │ │ │ │ │ - filterDelete: function(filter, options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "commit" │ │ │ │ │ - }); │ │ │ │ │ - var root = this.format.createElementNSPlus("wfs:Transaction", { │ │ │ │ │ - attributes: { │ │ │ │ │ - service: "WFS", │ │ │ │ │ - version: this.version │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - var deleteNode = this.format.createElementNSPlus("wfs:Delete", { │ │ │ │ │ - attributes: { │ │ │ │ │ - typeName: (options.featureNS ? this.featurePrefix + ":" : "") + options.featureType │ │ │ │ │ - } │ │ │ │ │ - }); │ │ │ │ │ - if (options.featureNS) { │ │ │ │ │ - deleteNode.setAttribute("xmlns:" + this.featurePrefix, options.featureNS) │ │ │ │ │ + getMapObjectPixelFromMapObjectLonLat: function(moLonLat) { │ │ │ │ │ + var lon = this.getLongitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + var lat = this.getLatitudeFromMapObjectLonLat(moLonLat); │ │ │ │ │ + var res = this.map.getResolution(); │ │ │ │ │ + var extent = this.map.getExtent(); │ │ │ │ │ + return this.getMapObjectPixelFromXY(1 / res * (lon - extent.left), 1 / res * (extent.top - lat)) │ │ │ │ │ + }, │ │ │ │ │ + setMapObjectCenter: function(center, zoom) { │ │ │ │ │ + if (this.animationEnabled === false && zoom != this.mapObject.zoom) { │ │ │ │ │ + var mapContainer = this.getMapContainer(); │ │ │ │ │ + google.maps.event.addListenerOnce(this.mapObject, "idle", function() { │ │ │ │ │ + mapContainer.style.visibility = "" │ │ │ │ │ + }); │ │ │ │ │ + mapContainer.style.visibility = "hidden" │ │ │ │ │ } │ │ │ │ │ - var filterNode = this.format.writeNode("ogc:Filter", filter); │ │ │ │ │ - deleteNode.appendChild(filterNode); │ │ │ │ │ - root.appendChild(deleteNode); │ │ │ │ │ - var data = OpenLayers.Format.XML.prototype.write.apply(this.format, [root]); │ │ │ │ │ - return OpenLayers.Request.POST({ │ │ │ │ │ - url: this.url, │ │ │ │ │ - callback: options.callback || function() {}, │ │ │ │ │ - data: data │ │ │ │ │ + this.mapObject.setOptions({ │ │ │ │ │ + center: center, │ │ │ │ │ + zoom: zoom │ │ │ │ │ }) │ │ │ │ │ }, │ │ │ │ │ - abort: function(response) { │ │ │ │ │ - if (response) { │ │ │ │ │ - response.priv.abort() │ │ │ │ │ + getMapObjectZoomFromMapObjectBounds: function(moBounds) { │ │ │ │ │ + return this.mapObject.getBoundsZoomLevel(moBounds) │ │ │ │ │ + }, │ │ │ │ │ + getMapObjectLonLatFromLonLat: function(lon, lat) { │ │ │ │ │ + var gLatLng; │ │ │ │ │ + if (this.sphericalMercator) { │ │ │ │ │ + var lonlat = this.inverseMercator(lon, lat); │ │ │ │ │ + gLatLng = new google.maps.LatLng(lonlat.lat, lonlat.lon) │ │ │ │ │ + } else { │ │ │ │ │ + gLatLng = new google.maps.LatLng(lat, lon) │ │ │ │ │ } │ │ │ │ │ + return gLatLng │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.WFS.v1" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Protocol.WFS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ - version: "1.0.0", │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.WFS.v1_0_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Protocol.WFS.v1_1_0 = OpenLayers.Class(OpenLayers.Protocol.WFS.v1, { │ │ │ │ │ - version: "1.1.0", │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.WFS.v1.prototype.initialize.apply(this, arguments); │ │ │ │ │ - if (this.outputFormat && !this.readFormat) { │ │ │ │ │ - if (this.outputFormat.toLowerCase() == "gml2") { │ │ │ │ │ - this.readFormat = new OpenLayers.Format.GML.v2({ │ │ │ │ │ - featureType: this.featureType, │ │ │ │ │ - featureNS: this.featureNS, │ │ │ │ │ - geometryName: this.geometryName │ │ │ │ │ - }) │ │ │ │ │ - } else if (this.outputFormat.toLowerCase() == "json") { │ │ │ │ │ - this.readFormat = new OpenLayers.Format.GeoJSON │ │ │ │ │ + getMapObjectPixelFromXY: function(x, y) { │ │ │ │ │ + return new google.maps.Point(x, y) │ │ │ │ │ + } │ │ │ │ │ +}; │ │ │ │ │ +OpenLayers.Popup.Framed = OpenLayers.Class(OpenLayers.Popup.Anchored, { │ │ │ │ │ + imageSrc: null, │ │ │ │ │ + imageSize: null, │ │ │ │ │ + isAlphaImage: false, │ │ │ │ │ + positionBlocks: null, │ │ │ │ │ + blocks: null, │ │ │ │ │ + fixedRelativePosition: false, │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, closeBoxCallback) { │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.initialize.apply(this, arguments); │ │ │ │ │ + if (this.fixedRelativePosition) { │ │ │ │ │ + this.updateRelativePosition(); │ │ │ │ │ + this.calculateRelativePosition = function(px) { │ │ │ │ │ + return this.relativePosition │ │ │ │ │ } │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.WFS.v1_1_0" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Protocol.CSW.v2_0_2 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ - formatOptions: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!options.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.CSWGetRecords.v2_0_2(OpenLayers.Util.extend({}, this.formatOptions)) │ │ │ │ │ + this.contentDiv.style.position = "absolute"; │ │ │ │ │ + this.contentDiv.style.zIndex = 1; │ │ │ │ │ + if (closeBox) { │ │ │ │ │ + this.closeDiv.style.zIndex = 1 │ │ │ │ │ } │ │ │ │ │ + this.groupDiv.style.position = "absolute"; │ │ │ │ │ + this.groupDiv.style.top = "0px"; │ │ │ │ │ + this.groupDiv.style.left = "0px"; │ │ │ │ │ + this.groupDiv.style.height = "100%"; │ │ │ │ │ + this.groupDiv.style.width = "100%" │ │ │ │ │ }, │ │ │ │ │ destroy: function() { │ │ │ │ │ - if (this.options && !this.options.format) { │ │ │ │ │ - this.format.destroy() │ │ │ │ │ + this.imageSrc = null; │ │ │ │ │ + this.imageSize = null; │ │ │ │ │ + this.isAlphaImage = null; │ │ │ │ │ + this.fixedRelativePosition = false; │ │ │ │ │ + this.positionBlocks = null; │ │ │ │ │ + for (var i = 0; i < this.blocks.length; i++) { │ │ │ │ │ + var block = this.blocks[i]; │ │ │ │ │ + if (block.image) { │ │ │ │ │ + block.div.removeChild(block.image) │ │ │ │ │ + } │ │ │ │ │ + block.image = null; │ │ │ │ │ + if (block.div) { │ │ │ │ │ + this.groupDiv.removeChild(block.div) │ │ │ │ │ + } │ │ │ │ │ + block.div = null │ │ │ │ │ } │ │ │ │ │ - this.format = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ + this.blocks = null; │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.destroy.apply(this, arguments) │ │ │ │ │ }, │ │ │ │ │ - read: function(options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - var data = this.format.write(options.params || options); │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ - params: options.params, │ │ │ │ │ - headers: options.headers, │ │ │ │ │ - data: data │ │ │ │ │ - }); │ │ │ │ │ - return response │ │ │ │ │ + setBackgroundColor: function(color) {}, │ │ │ │ │ + setBorder: function() {}, │ │ │ │ │ + setOpacity: function(opacity) {}, │ │ │ │ │ + setSize: function(contentSize) { │ │ │ │ │ + OpenLayers.Popup.Anchored.prototype.setSize.apply(this, arguments); │ │ │ │ │ + this.updateBlocks() │ │ │ │ │ }, │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - response.data = this.parseData(request); │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ - } │ │ │ │ │ - options.callback.call(options.scope, response) │ │ │ │ │ + updateRelativePosition: function() { │ │ │ │ │ + this.padding = this.positionBlocks[this.relativePosition].padding; │ │ │ │ │ + if (this.closeDiv) { │ │ │ │ │ + var contentDivPadding = this.getContentDivPadding(); │ │ │ │ │ + this.closeDiv.style.right = contentDivPadding.right + this.padding.right + "px"; │ │ │ │ │ + this.closeDiv.style.top = contentDivPadding.top + this.padding.top + "px" │ │ │ │ │ } │ │ │ │ │ + this.updateBlocks() │ │ │ │ │ }, │ │ │ │ │ - parseData: function(request) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null │ │ │ │ │ - } │ │ │ │ │ - return this.format.read(doc) │ │ │ │ │ + calculateNewPx: function(px) { │ │ │ │ │ + var newPx = OpenLayers.Popup.Anchored.prototype.calculateNewPx.apply(this, arguments); │ │ │ │ │ + newPx = newPx.offset(this.positionBlocks[this.relativePosition].offset); │ │ │ │ │ + return newPx │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.CSW.v2_0_2" │ │ │ │ │ -}); │ │ │ │ │ -OpenLayers.Protocol.SOS.v1_0_0 = OpenLayers.Class(OpenLayers.Protocol, { │ │ │ │ │ - fois: null, │ │ │ │ │ - formatOptions: null, │ │ │ │ │ - initialize: function(options) { │ │ │ │ │ - OpenLayers.Protocol.prototype.initialize.apply(this, [options]); │ │ │ │ │ - if (!options.format) { │ │ │ │ │ - this.format = new OpenLayers.Format.SOSGetFeatureOfInterest(this.formatOptions) │ │ │ │ │ + createBlocks: function() { │ │ │ │ │ + this.blocks = []; │ │ │ │ │ + var firstPosition = null; │ │ │ │ │ + for (var key in this.positionBlocks) { │ │ │ │ │ + firstPosition = key; │ │ │ │ │ + break │ │ │ │ │ } │ │ │ │ │ - }, │ │ │ │ │ - destroy: function() { │ │ │ │ │ - if (this.options && !this.options.format) { │ │ │ │ │ - this.format.destroy() │ │ │ │ │ + var position = this.positionBlocks[firstPosition]; │ │ │ │ │ + for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ + var block = {}; │ │ │ │ │ + this.blocks.push(block); │ │ │ │ │ + var divId = this.id + "_FrameDecorationDiv_" + i; │ │ │ │ │ + block.div = OpenLayers.Util.createDiv(divId, null, null, null, "absolute", null, "hidden", null); │ │ │ │ │ + var imgId = this.id + "_FrameDecorationImg_" + i; │ │ │ │ │ + var imageCreator = this.isAlphaImage ? OpenLayers.Util.createAlphaImageDiv : OpenLayers.Util.createImage; │ │ │ │ │ + block.image = imageCreator(imgId, null, this.imageSize, this.imageSrc, "absolute", null, null, null); │ │ │ │ │ + block.div.appendChild(block.image); │ │ │ │ │ + this.groupDiv.appendChild(block.div) │ │ │ │ │ } │ │ │ │ │ - this.format = null; │ │ │ │ │ - OpenLayers.Protocol.prototype.destroy.apply(this) │ │ │ │ │ - }, │ │ │ │ │ - read: function(options) { │ │ │ │ │ - options = OpenLayers.Util.extend({}, options); │ │ │ │ │ - OpenLayers.Util.applyDefaults(options, this.options || {}); │ │ │ │ │ - var response = new OpenLayers.Protocol.Response({ │ │ │ │ │ - requestType: "read" │ │ │ │ │ - }); │ │ │ │ │ - var format = this.format; │ │ │ │ │ - var data = OpenLayers.Format.XML.prototype.write.apply(format, [format.writeNode("sos:GetFeatureOfInterest", { │ │ │ │ │ - fois: this.fois │ │ │ │ │ - })]); │ │ │ │ │ - response.priv = OpenLayers.Request.POST({ │ │ │ │ │ - url: options.url, │ │ │ │ │ - callback: this.createCallback(this.handleRead, response, options), │ │ │ │ │ - data: data │ │ │ │ │ - }); │ │ │ │ │ - return response │ │ │ │ │ }, │ │ │ │ │ - handleRead: function(response, options) { │ │ │ │ │ - if (options.callback) { │ │ │ │ │ - var request = response.priv; │ │ │ │ │ - if (request.status >= 200 && request.status < 300) { │ │ │ │ │ - response.features = this.parseFeatures(request); │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.SUCCESS │ │ │ │ │ - } else { │ │ │ │ │ - response.code = OpenLayers.Protocol.Response.FAILURE │ │ │ │ │ + updateBlocks: function() { │ │ │ │ │ + if (!this.blocks) { │ │ │ │ │ + this.createBlocks() │ │ │ │ │ + } │ │ │ │ │ + if (this.size && this.relativePosition) { │ │ │ │ │ + var position = this.positionBlocks[this.relativePosition]; │ │ │ │ │ + for (var i = 0; i < position.blocks.length; i++) { │ │ │ │ │ + var positionBlock = position.blocks[i]; │ │ │ │ │ + var block = this.blocks[i]; │ │ │ │ │ + var l = positionBlock.anchor.left; │ │ │ │ │ + var b = positionBlock.anchor.bottom; │ │ │ │ │ + var r = positionBlock.anchor.right; │ │ │ │ │ + var t = positionBlock.anchor.top; │ │ │ │ │ + var w = isNaN(positionBlock.size.w) ? this.size.w - (r + l) : positionBlock.size.w; │ │ │ │ │ + var h = isNaN(positionBlock.size.h) ? this.size.h - (b + t) : positionBlock.size.h; │ │ │ │ │ + block.div.style.width = (w < 0 ? 0 : w) + "px"; │ │ │ │ │ + block.div.style.height = (h < 0 ? 0 : h) + "px"; │ │ │ │ │ + block.div.style.left = l != null ? l + "px" : ""; │ │ │ │ │ + block.div.style.bottom = b != null ? b + "px" : ""; │ │ │ │ │ + block.div.style.right = r != null ? r + "px" : ""; │ │ │ │ │ + block.div.style.top = t != null ? t + "px" : ""; │ │ │ │ │ + block.image.style.left = positionBlock.position.x + "px"; │ │ │ │ │ + block.image.style.top = positionBlock.position.y + "px" │ │ │ │ │ } │ │ │ │ │ - options.callback.call(options.scope, response) │ │ │ │ │ + this.contentDiv.style.left = this.padding.left + "px"; │ │ │ │ │ + this.contentDiv.style.top = this.padding.top + "px" │ │ │ │ │ } │ │ │ │ │ }, │ │ │ │ │ - parseFeatures: function(request) { │ │ │ │ │ - var doc = request.responseXML; │ │ │ │ │ - if (!doc || !doc.documentElement) { │ │ │ │ │ - doc = request.responseText │ │ │ │ │ - } │ │ │ │ │ - if (!doc || doc.length <= 0) { │ │ │ │ │ - return null │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup.Framed" │ │ │ │ │ +}); │ │ │ │ │ +OpenLayers.Popup.FramedCloud = OpenLayers.Class(OpenLayers.Popup.Framed, { │ │ │ │ │ + contentDisplayClass: "olFramedCloudPopupContent", │ │ │ │ │ + autoSize: true, │ │ │ │ │ + panMapIfOutOfView: true, │ │ │ │ │ + imageSize: new OpenLayers.Size(1276, 736), │ │ │ │ │ + isAlphaImage: false, │ │ │ │ │ + fixedRelativePosition: false, │ │ │ │ │ + positionBlocks: { │ │ │ │ │ + tl: { │ │ │ │ │ + offset: new OpenLayers.Pixel(44, 0), │ │ │ │ │ + padding: new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ + blocks: [{ │ │ │ │ │ + size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size("auto", 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, 18), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -632) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(81, 35), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -688) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + tr: { │ │ │ │ │ + offset: new OpenLayers.Pixel(-45, 0), │ │ │ │ │ + padding: new OpenLayers.Bounds(8, 40, 8, 9), │ │ │ │ │ + blocks: [{ │ │ │ │ │ + size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 51, 22, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 50, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size("auto", 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 32, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -631) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, 19), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 32, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -631) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(81, 35), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, null, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-215, -687) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + bl: { │ │ │ │ │ + offset: new OpenLayers.Pixel(45, 0), │ │ │ │ │ + padding: new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ + blocks: [{ │ │ │ │ │ + size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size("auto", 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(81, 33), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, null, 0, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-101, -674) │ │ │ │ │ + }] │ │ │ │ │ + }, │ │ │ │ │ + br: { │ │ │ │ │ + offset: new OpenLayers.Pixel(-44, 0), │ │ │ │ │ + padding: new OpenLayers.Bounds(8, 9, 8, 40), │ │ │ │ │ + blocks: [{ │ │ │ │ │ + size: new OpenLayers.Size("auto", "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 21, 22, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(0, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, "auto"), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 21, 0, 32), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, 0) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size("auto", 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, 0, 22, null), │ │ │ │ │ + position: new OpenLayers.Pixel(0, -629) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(22, 21), │ │ │ │ │ + anchor: new OpenLayers.Bounds(null, 0, 0, null), │ │ │ │ │ + position: new OpenLayers.Pixel(-1238, -629) │ │ │ │ │ + }, { │ │ │ │ │ + size: new OpenLayers.Size(81, 33), │ │ │ │ │ + anchor: new OpenLayers.Bounds(0, null, null, 0), │ │ │ │ │ + position: new OpenLayers.Pixel(-311, -674) │ │ │ │ │ + }] │ │ │ │ │ } │ │ │ │ │ - return this.format.read(doc) │ │ │ │ │ }, │ │ │ │ │ - CLASS_NAME: "OpenLayers.Protocol.SOS.v1_0_0" │ │ │ │ │ + minSize: new OpenLayers.Size(105, 10), │ │ │ │ │ + maxSize: new OpenLayers.Size(1200, 660), │ │ │ │ │ + initialize: function(id, lonlat, contentSize, contentHTML, anchor, closeBox, closeBoxCallback) { │ │ │ │ │ + this.imageSrc = OpenLayers.Util.getImageLocation("cloud-popup-relative.png"); │ │ │ │ │ + OpenLayers.Popup.Framed.prototype.initialize.apply(this, arguments); │ │ │ │ │ + this.contentDiv.className = this.contentDisplayClass │ │ │ │ │ + }, │ │ │ │ │ + CLASS_NAME: "OpenLayers.Popup.FramedCloud" │ │ │ │ │ });